From f9a559303bab0516eee2406780eff6054306f0a1 Mon Sep 17 00:00:00 2001 From: msweet Date: Fri, 10 May 2013 18:56:23 +0000 Subject: [PATCH 1/1] Import cups.org releases git-svn-id: svn+ssh://src.apple.com/svn/cups/cups.org/tags/release-1.5.1@4306 a1ca3aef-8c08-0410-bb20-df032aa958be --- CHANGES-1.0.txt | 217 + CHANGES-1.1.txt | 3462 + CHANGES-1.2.txt | 1261 + CHANGES-1.3.txt | 856 + CHANGES-1.4.txt | 843 + CHANGES-1.5.txt | 228 + CHANGES-IPPTOOL.txt | 107 + CHANGES.txt | 40 + CREDITS.txt | 49 + INSTALL.txt | 207 + IPPTOOL.txt | 145 + LICENSE.txt | 971 + Makedefs.in | 260 + Makefile | 334 + README.txt | 164 + backend/Dependencies | 72 + backend/Makefile | 291 + backend/backend-private.h | 324 + backend/dnssd.c | 965 + backend/ieee1284.c | 491 + backend/ipp.c | 3152 + backend/lpd.c | 1342 + backend/network.c | 300 + backend/pseudo | 30 + backend/runloop.c | 523 + backend/snmp-supplies.c | 986 + backend/snmp.c | 1387 + backend/snmp.txt | 172 + backend/socket.c | 527 + backend/test1284.c | 84 + backend/testbackend.c | 670 + backend/testsupplies.c | 83 + backend/usb-darwin.c | 2269 + backend/usb-libusb.c | 942 + backend/usb-unix.c | 623 + backend/usb.c | 264 + berkeley/Dependencies | 28 + berkeley/Makefile | 167 + berkeley/lpc.c | 449 + berkeley/lpq.c | 678 + berkeley/lpr.c | 417 + berkeley/lprm.c | 217 + cgi-bin/Dependencies | 73 + cgi-bin/Makefile | 382 + cgi-bin/admin.c | 4200 + cgi-bin/api-cgi.header | 34 + cgi-bin/api-cgi.shtml | 17 + cgi-bin/cgi-private.h | 36 + cgi-bin/cgi.h | 119 + cgi-bin/classes.c | 558 + cgi-bin/help-index.c | 1328 + cgi-bin/help-index.h | 87 + cgi-bin/help.c | 392 + cgi-bin/html.c | 239 + cgi-bin/ipp-var.c | 1592 + cgi-bin/jobs.c | 214 + cgi-bin/libcupscgi.exp | 42 + cgi-bin/makedocset.c | 486 + cgi-bin/multipart.dat | Bin 0 -> 50411 bytes cgi-bin/printers.c | 578 + cgi-bin/search.c | 381 + cgi-bin/template.c | 733 + cgi-bin/testcgi.c | 75 + cgi-bin/testhi.c | 113 + cgi-bin/testhi.html | 31 + cgi-bin/testtemplate.c | 103 + cgi-bin/var.c | 1305 + cgi-bin/websearch.c | 116 + conf/Makefile | 143 + conf/cupsd.conf.in | 140 + conf/mime.convs.in | 63 + conf/mime.types | 174 + conf/pam.opendirectory | 5 + conf/pam.securityserver | 7 + conf/pam.std.in | 2 + conf/snmp.conf.in | 13 + config-scripts/cups-common.m4 | 453 + config-scripts/cups-compiler.m4 | 299 + config-scripts/cups-defaults.m4 | 396 + config-scripts/cups-directories.m4 | 440 + config-scripts/cups-dnssd.m4 | 64 + config-scripts/cups-gssapi.m4 | 171 + config-scripts/cups-largefile.m4 | 52 + config-scripts/cups-launchd.m4 | 43 + config-scripts/cups-libtool.m4 | 39 + config-scripts/cups-manpages.m4 | 96 + config-scripts/cups-network.m4 | 74 + config-scripts/cups-opsys.m4 | 35 + config-scripts/cups-pam.m4 | 102 + config-scripts/cups-poll.m4 | 22 + config-scripts/cups-scripting.m4 | 89 + config-scripts/cups-sharedlibs.m4 | 238 + config-scripts/cups-ssl.m4 | 179 + config-scripts/cups-threads.m4 | 54 + config.h.in | 751 + configure.in | 93 + cups-config.in | 146 + cups/Dependencies | 250 + cups/Makefile | 625 + cups/adminutil.c | 2341 + cups/adminutil.h | 78 + cups/api-array.header | 34 + cups/api-array.shtml | 196 + cups/api-cups.header | 40 + cups/api-cups.shtml | 443 + cups/api-filedir.header | 36 + cups/api-filedir.shtml | 31 + cups/api-filter.header | 41 + cups/api-filter.shtml | 765 + cups/api-httpipp.header | 37 + cups/api-httpipp.shtml | 323 + cups/api-overview.header | 53 + cups/api-overview.shtml | 94 + cups/api-ppd.header | 38 + cups/api-ppd.shtml | 219 + cups/array-private.h | 51 + cups/array.c | 1326 + cups/array.h | 92 + cups/attr.c | 335 + cups/auth.c | 878 + cups/backchannel.c | 199 + cups/backend.c | 143 + cups/backend.h | 78 + cups/conflicts.c | 1214 + cups/cups-private.h | 262 + cups/cups.h | 598 + cups/custom.c | 122 + cups/debug-private.h | 117 + cups/debug.c | 668 + cups/dest-job.c | 145 + cups/dest-localization.c | 76 + cups/dest-options.c | 1164 + cups/dest.c | 3476 + cups/dir.c | 472 + cups/dir.h | 69 + cups/emit.c | 1217 + cups/encode.c | 651 + cups/file-private.h | 137 + cups/file.c | 2714 + cups/file.h | 116 + cups/getdevices.c | 283 + cups/getifaddrs.c | 266 + cups/getputfile.c | 502 + cups/globals.c | 354 + cups/http-addr.c | 705 + cups/http-addrlist.c | 832 + cups/http-private.h | 408 + cups/http-support.c | 1919 + cups/http.c | 4740 + cups/http.h | 486 + cups/ipp-private.h | 175 + cups/ipp-support.c | 1085 + cups/ipp.c | 5565 + cups/ipp.h | 546 + cups/langprintf.c | 352 + cups/language-private.h | 85 + cups/language.c | 1518 + cups/language.h | 115 + cups/libcups2.def | 382 + cups/libcups_s.exp | 85 + cups/localize.c | 779 + cups/mark.c | 1101 + cups/md5-private.h | 79 + cups/md5.c | 346 + cups/md5passwd.c | 142 + cups/notify.c | 202 + cups/options.c | 711 + cups/page.c | 396 + cups/ppd-cache.c | 2605 + cups/ppd-private.h | 214 + cups/ppd.c | 3398 + cups/ppd.h | 468 + cups/pwg-media.c | 851 + cups/pwg-private.h | 102 + cups/raster-private.h | 69 + cups/raster.h | 405 + cups/request.c | 1149 + cups/sidechannel.c | 642 + cups/sidechannel.h | 147 + cups/snmp-private.h | 145 + cups/snmp.c | 1737 + cups/snprintf.c | 362 + cups/sspi-private.h | 82 + cups/sspi.c | 1485 + cups/string-private.h | 200 + cups/string.c | 759 + cups/tempfile.c | 233 + cups/test.ppd | 262 + cups/test2.ppd | 252 + cups/testadmin.c | 120 + cups/testarray.c | 480 + cups/testconflicts.c | 127 + cups/testcups.c | 472 + cups/testfile.c | 821 + cups/testhttp.c | 600 + cups/testi18n.c | 619 + cups/testipp.c | 1005 + cups/testlang.c | 114 + cups/testoptions.c | 116 + cups/testppd.c | 1102 + cups/testpwg.c | 497 + cups/testsnmp.c | 304 + cups/thread-private.h | 97 + cups/thread.c | 317 + cups/transcode.c | 720 + cups/transcode.h | 81 + cups/usersys.c | 1059 + cups/utf8demo.txt | 213 + cups/util.c | 1800 + cups/versioning.h | 92 + data/Makefile | 150 + data/classified | 6 + data/confidential | 6 + data/cups.irix | 3 + data/cups.pam | 2 + data/cups.suse | 2 + data/epson.h | 27 + data/font.defs | 55 + data/hp.h | 24 + data/label.h | 28 + data/media.defs | 208 + data/raster.defs | 94 + data/secret | 6 + data/smiley.ps | 28 + data/standard | 6 + data/testprint.in | 7 + data/topsecret | 6 + data/unclassified | 6 + desktop/Makefile | 135 + desktop/cups-128.png | Bin 0 -> 4888 bytes desktop/cups-16.png | Bin 0 -> 503 bytes desktop/cups-256.png | Bin 0 -> 10147 bytes desktop/cups-32.png | Bin 0 -> 1042 bytes desktop/cups-512.png | Bin 0 -> 21029 bytes desktop/cups-64.png | Bin 0 -> 2234 bytes desktop/cups.conf | 13 + desktop/cups.desktop.in | 39 + desktop/cups.icns | Bin 0 -> 107501 bytes desktop/cups.svg | 533 + doc/Makefile | 266 + doc/cups-printable.css | 329 + doc/cups.css | 462 + doc/help/accounting.html | 62 + doc/help/api-array.html | 1062 + doc/help/api-cgi.html | 1126 + doc/help/api-cups.html | 2521 + doc/help/api-driver.html | 1156 + doc/help/api-filedir.html | 995 + doc/help/api-filter.html | 1625 + doc/help/api-httpipp.html | 5358 + doc/help/api-mime.html | 833 + doc/help/api-overview.html | 501 + doc/help/api-ppd.html | 2193 + doc/help/api-ppdc.html | 2197 + doc/help/api-raster.html | 1416 + doc/help/cgi.html | 86 + doc/help/glossary.html | 219 + doc/help/kerberos.html | 89 + doc/help/license.html | 1076 + doc/help/man-ipptool.html | 144 + doc/help/man-ipptoolfile.html | 607 + doc/help/network.html | 684 + doc/help/options.html | 778 + doc/help/overview.html | 70 + doc/help/policies.html | 601 + doc/help/postscript-driver.html | 663 + doc/help/ppd-compiler.html | 1279 + doc/help/raster-driver.html | 579 + doc/help/ref-access_log.html | 140 + doc/help/ref-classes-conf.html | 566 + doc/help/ref-client-conf.html | 75 + doc/help/ref-cupsd-conf.html.in | 2709 + doc/help/ref-error_log.html | 55 + doc/help/ref-mailto-conf.html | 108 + doc/help/ref-page_log.html | 77 + doc/help/ref-ppdcfile.html | 2449 + doc/help/ref-printers-conf.html | 720 + doc/help/ref-snmp-conf.html | 146 + doc/help/ref-subscriptions-conf.html | 354 + doc/help/security.html | 172 + doc/help/sharing.html | 184 + doc/help/spec-banner.html | 156 + doc/help/spec-browsing.html | 118 + doc/help/spec-cmp.html | 1218 + doc/help/spec-command.html | 218 + doc/help/spec-design.html | 184 + doc/help/spec-ipp.html | 2744 + doc/help/spec-pdf.html | 24 + doc/help/spec-postscript.html | 148 + doc/help/spec-ppd.html | 2340 + doc/help/spec-raster.html | 720 + doc/help/spec-stp.html | 133 + doc/help/standard.html.in | 181 + doc/help/translation.html | 852 + doc/help/whatsnew.html | 61 + doc/images/color-wheel.png | Bin 0 -> 13148 bytes doc/images/cups-block-diagram.png | Bin 0 -> 76386 bytes doc/images/cups-block-diagram.svg | 841 + doc/images/cups-command-chain.png | Bin 0 -> 14902 bytes doc/images/cups-command-chain.svg | 439 + doc/images/cups-icon.png | Bin 0 -> 4888 bytes doc/images/cups-postscript-chain.png | Bin 0 -> 17498 bytes doc/images/cups-postscript-chain.svg | 531 + doc/images/cups-raster-chain.png | Bin 0 -> 16916 bytes doc/images/cups-raster-chain.svg | 534 + doc/images/cups.png | Bin 0 -> 4888 bytes doc/images/cups.svg | 533 + doc/images/generic.png | Bin 0 -> 16913 bytes doc/images/left.gif | Bin 0 -> 1492 bytes doc/images/left.xcf.gz | Bin 0 -> 1194 bytes doc/images/raster-organization.png | Bin 0 -> 20974 bytes doc/images/raster-organization.svg | 189 + doc/images/raster.png | Bin 0 -> 37656 bytes doc/images/raster.svg | 386 + doc/images/right.gif | Bin 0 -> 341 bytes doc/images/sample-image.png | Bin 0 -> 3541 bytes doc/images/sel.gif | Bin 0 -> 362 bytes doc/images/smiley.jpg | Bin 0 -> 14120 bytes doc/images/unsel.gif | Bin 0 -> 127 bytes doc/images/wait.gif | Bin 0 -> 1810 bytes doc/images/webinterface.png | Bin 0 -> 150003 bytes doc/index.html.in | 107 + doc/robots.txt | 31 + examples/Makefile | 128 + examples/color.drv | 44 + examples/constraint.drv | 48 + examples/custom.drv | 41 + examples/grouping.drv | 36 + examples/laserjet-basic.drv | 88 + examples/laserjet-pjl.drv | 101 + examples/minimum.drv | 26 + examples/postscript.drv | 46 + examples/r300-basic.drv | 75 + examples/r300-colorman.drv | 85 + examples/r300-remote.drv | 85 + filter/Dependencies | 59 + filter/Makefile | 400 + filter/api-raster.header | 37 + filter/api-raster.shtml | 160 + filter/commandtops.c | 371 + filter/common.c | 535 + filter/common.h | 78 + filter/error.c | 286 + filter/gziptoany.c | 112 + filter/interpret.c | 1688 + filter/libcupsimage2.def | 14 + filter/libcupsimage_s.exp | 16 + filter/postscript-driver.header | 32 + filter/postscript-driver.shtml | 276 + filter/ppd-compiler.header | 34 + filter/ppd-compiler.shtml | 883 + filter/pstops.c | 3425 + filter/raster-driver.header | 32 + filter/raster-driver.shtml | 194 + filter/raster.c | 1476 + filter/rasterbench.c | 355 + filter/rastertoepson.c | 1160 + filter/rastertohp.c | 889 + filter/rastertolabel.c | 1315 + filter/rastertopwg.c | 461 + filter/spec-ppd.header | 32 + filter/spec-ppd.shtml | 1898 + filter/testraster.c | 1078 + install-sh | 222 + locale/Dependencies | 22 + locale/Makefile | 219 + locale/checkpo.c | 413 + locale/cups.footer | 5 + locale/cups.header | 27 + locale/cups.pot | 6824 + locale/cups.strings | 1517 + locale/locale.txt | 32 + locale/po2strings.c | 281 + locale/strings2po.c | 175 + locale/translate.c | 439 + man/Makefile | 240 + man/backend.man | 196 + man/cancel.man | 75 + man/classes.conf.man | 110 + man/client.conf.man.in | 55 + man/cups-config.man | 117 + man/cups-deviced.man.in | 44 + man/cups-driverd.man.in | 122 + man/cups-lpd.man.in | 124 + man/cups-polld.man | 38 + man/cups-snmp.conf.man | 73 + man/cupsaccept.man | 79 + man/cupsaddsmb.man.in | 214 + man/cupsctl.man | 107 + man/cupsd.conf.man.in | 640 + man/cupsd.man.in | 72 + man/cupsenable.man | 93 + man/cupsfilter.man | 91 + man/cupstestdsc.man | 50 + man/cupstestppd.man | 165 + man/filter.man | 256 + man/ipptool.man | 130 + man/ipptoolfile.man | 523 + man/lp.man | 258 + man/lpadmin.man | 228 + man/lpc.man | 71 + man/lpinfo.man | 115 + man/lpmove.man | 66 + man/lpoptions.man.in | 135 + man/lppasswd.man | 68 + man/lpq.man | 72 + man/lpr.man | 122 + man/lprm.man | 65 + man/lpstat.man | 143 + man/mailto.conf.man | 60 + man/mantohtml.c | 720 + man/mime.convs.man | 46 + man/mime.types.man | 115 + man/notifier.man | 157 + man/ppdc.man | 80 + man/ppdcfile.man | 171 + man/ppdhtml.man | 46 + man/ppdi.man | 46 + man/ppdmerge.man | 47 + man/ppdpo.man | 52 + man/printers.conf.man | 124 + man/subscriptions.conf.man | 89 + monitor/Dependencies | 14 + monitor/Makefile | 146 + monitor/bcp.c | 292 + monitor/tbcp.c | 285 + notifier/Dependencies | 20 + notifier/Makefile | 162 + notifier/dbus.c | 627 + notifier/mailto.c | 646 + notifier/rss.c | 732 + notifier/testnotify.c | 127 + packaging/InstallationCheck | 11 + packaging/LICENSE.rtf | 434 + packaging/WELCOME.rtf | 24 + packaging/cups-desc.plist.in | 14 + packaging/cups-info.plist.in | 26 + packaging/cups.list.in | 738 + packaging/cups.spec.in | 351 + packaging/installer.gif | Bin 0 -> 3392 bytes packaging/installer.tif | Bin 0 -> 5538 bytes ppdc/Dependencies | 208 + ppdc/Makefile | 404 + ppdc/api-ppdc.header | 34 + ppdc/api-ppdc.shtml | 18 + ppdc/foo-fr.po | 11 + ppdc/foo.drv | 547 + ppdc/genstrings.cxx | 215 + ppdc/ppdc-array.cxx | 168 + ppdc/ppdc-attr.cxx | 66 + ppdc/ppdc-catalog.cxx | 897 + ppdc/ppdc-choice.cxx | 61 + ppdc/ppdc-constraint.cxx | 64 + ppdc/ppdc-driver.cxx | 1339 + ppdc/ppdc-file.cxx | 109 + ppdc/ppdc-filter.cxx | 60 + ppdc/ppdc-font.cxx | 66 + ppdc/ppdc-group.cxx | 103 + ppdc/ppdc-import.cxx | 343 + ppdc/ppdc-mediasize.cxx | 85 + ppdc/ppdc-message.cxx | 58 + ppdc/ppdc-option.cxx | 129 + ppdc/ppdc-private.h | 40 + ppdc/ppdc-profile.cxx | 65 + ppdc/ppdc-shared.cxx | 88 + ppdc/ppdc-source.cxx | 3850 + ppdc/ppdc-string.cxx | 62 + ppdc/ppdc-variable.cxx | 71 + ppdc/ppdc.cxx | 469 + ppdc/ppdc.h | 532 + ppdc/ppdhtml.cxx | 186 + ppdc/ppdi.cxx | 142 + ppdc/ppdmerge.cxx | 379 + ppdc/ppdpo.cxx | 268 + ppdc/sample.drv | 1254 + ppdc/testcatalog.cxx | 63 + scheduler/Dependencies | 287 + scheduler/Makefile | 551 + scheduler/api-mime.header | 34 + scheduler/api-mime.shtml | 17 + scheduler/auth.c | 2599 + scheduler/auth.h | 151 + scheduler/banners.c | 224 + scheduler/banners.h | 45 + scheduler/cert.c | 442 + scheduler/cert.h | 56 + scheduler/classes.c | 818 + scheduler/classes.h | 35 + scheduler/client.c | 5255 + scheduler/client.h | 137 + scheduler/conf.c | 3790 + scheduler/conf.h | 296 + scheduler/cups-deviced.c | 810 + scheduler/cups-driverd.cxx | 2594 + scheduler/cups-exec.c | 108 + scheduler/cups-lpd.c | 1627 + scheduler/cups-lpd.xinetd.in | 12 + scheduler/cups.sh.in | 237 + scheduler/cups.xml.in | 214 + scheduler/cupsd.h | 242 + scheduler/cupsfilter.c | 1495 + scheduler/dirsvc.c | 1392 + scheduler/dirsvc.h | 77 + scheduler/env.c | 271 + scheduler/file.c | 450 + scheduler/filter.c | 504 + scheduler/ipp.c | 11989 ++ scheduler/job.c | 4930 + scheduler/job.h | 166 + scheduler/libcupsmime.exp | 22 + scheduler/listen.c | 431 + scheduler/log.c | 1102 + scheduler/main.c | 2005 + scheduler/mime-private.h | 45 + scheduler/mime.c | 960 + scheduler/mime.h | 162 + scheduler/network.c | 300 + scheduler/network.h | 52 + scheduler/org.cups.cups-lpd.plist.in | 33 + scheduler/org.cups.cupsd.plist | 62 + scheduler/policy.c | 517 + scheduler/policy.h | 63 + scheduler/printers.c | 5273 + scheduler/printers.h | 172 + scheduler/process.c | 656 + scheduler/quotas.c | 244 + scheduler/select.c | 951 + scheduler/server.c | 172 + scheduler/statbuf.c | 330 + scheduler/statbuf.h | 49 + scheduler/subscriptions.c | 1638 + scheduler/subscriptions.h | 166 + scheduler/sysman.c | 1004 + scheduler/sysman.h | 64 + scheduler/testlpd.c | 550 + scheduler/testmime.c | 531 + scheduler/testspeed.c | 365 + scheduler/testsub.c | 523 + scheduler/type.c | 1216 + scheduler/util.c | 471 + scheduler/util.h | 71 + systemv/Dependencies | 92 + systemv/Makefile | 293 + systemv/cancel.c | 376 + systemv/cupsaccept.c | 239 + systemv/cupsaddsmb.c | 303 + systemv/cupsctl.c | 227 + systemv/cupstestdsc.c | 442 + systemv/cupstestppd.c | 3956 + systemv/lp.c | 723 + systemv/lpadmin.c | 1518 + systemv/lpinfo.c | 498 + systemv/lpmove.c | 213 + systemv/lpoptions.c | 565 + systemv/lppasswd.c | 489 + systemv/lpstat.c | 2054 + templates/Makefile | 204 + templates/add-class.tmpl | 40 + templates/add-printer.tmpl | 47 + templates/add-rss-subscription.tmpl | 44 + templates/admin.tmpl | 99 + templates/choose-device.tmpl | 53 + templates/choose-make.tmpl | 64 + templates/choose-model.tmpl | 60 + templates/choose-serial.tmpl | 52 + templates/choose-uri.tmpl | 44 + templates/class-added.tmpl | 8 + templates/class-confirm.tmpl | 10 + templates/class-deleted.tmpl | 7 + templates/class-jobs-header.tmpl | 3 + templates/class-modified.tmpl | 8 + templates/class.tmpl | 44 + templates/classes-header.tmpl | 1 + templates/classes.tmpl | 11 + templates/command.tmpl | 12 + templates/edit-config.tmpl | 24 + templates/error-op.tmpl | 9 + templates/error.tmpl | 9 + templates/header.tmpl.in | 29 + templates/help-header.tmpl | 51 + templates/help-printable.tmpl | 9 + templates/help-trailer.tmpl | 1 + templates/job-cancel.tmpl | 7 + templates/job-hold.tmpl | 7 + templates/job-move.tmpl | 27 + templates/job-moved.tmpl | 8 + templates/job-release.tmpl | 7 + templates/job-restart.tmpl | 7 + templates/jobs-header.tmpl | 5 + templates/jobs.tmpl | 36 + templates/list-available-printers.tmpl | 11 + templates/modify-class.tmpl | 34 + templates/modify-printer.tmpl | 42 + templates/norestart.tmpl | 8 + templates/option-boolean.tmpl | 6 + templates/option-conflict.tmpl | 7 + templates/option-header.tmpl | 5 + templates/option-pickmany.tmpl | 6 + templates/option-pickone.tmpl | 18 + templates/option-trailer.tmpl | 5 + templates/pager.tmpl | 6 + templates/printer-accept.tmpl | 9 + templates/printer-added.tmpl | 8 + templates/printer-configured.tmpl | 8 + templates/printer-confirm.tmpl | 10 + templates/printer-default.tmpl | 13 + templates/printer-deleted.tmpl | 7 + templates/printer-jobs-header.tmpl | 3 + templates/printer-modified.tmpl | 8 + templates/printer-purge.tmpl | 9 + templates/printer-reject.tmpl | 9 + templates/printer-start.tmpl | 9 + templates/printer-stop.tmpl | 9 + templates/printer.tmpl | 47 + templates/printers-header.tmpl | 1 + templates/printers.tmpl | 11 + templates/restart.tmpl | 8 + templates/samba-export.tmpl | 55 + templates/samba-exported.tmpl | 1 + templates/search.tmpl | 10 + templates/set-printer-options-header.tmpl | 26 + templates/set-printer-options-trailer.tmpl | 16 + templates/subscription-added.tmpl | 5 + templates/subscription-canceled.tmpl | 5 + templates/test-page.tmpl | 8 + templates/trailer.tmpl | 8 + templates/users.tmpl | 30 + test/4.1-requests.test | 159 + test/4.2-cups-printer-ops.test | 327 + test/4.3-job-ops.test | 330 + test/4.4-subscription-ops.test | 153 + test/5.1-lpadmin.sh | 55 + test/5.2-lpc.sh | 31 + test/5.3-lpq.sh | 31 + test/5.4-lpstat.sh | 43 + test/5.5-lp.sh | 84 + test/5.6-lpr.sh | 84 + test/5.7-lprm.sh | 47 + test/5.8-cancel.sh | 45 + test/5.9-lpinfo.sh | 55 + test/Dependencies | 17 + test/Makefile | 204 + test/color.jpg | Bin 0 -> 90720 bytes test/create-job-format.test | 56 + test/create-job-sheets.test | 55 + test/create-job-timeout.test | 54 + test/create-job.test | 52 + test/create-printer-subscription.test | 49 + test/document-a4.pdf | Bin 0 -> 1062429 bytes test/document-a4.ps | 135164 +++++++++++++++++ test/document-a4.sla | 277 + test/document-letter.pdf | Bin 0 -> 1430525 bytes test/document-letter.ps | 136971 ++++++++++++++++++ test/document-letter.sla | 279 + test/get-completed-jobs.test | 51 + test/get-devices.test | 21 + test/get-job-attributes.test | 27 + test/get-job-attributes2.test | 28 + test/get-jobs.test | 53 + test/get-ppd-printer.test | 20 + test/get-ppd.test | 20 + test/get-ppds-drv-only.test | 24 + test/get-ppds-language.test | 21 + test/get-ppds-make-and-model.test | 21 + test/get-ppds-make.test | 21 + test/get-ppds-product.test | 21 + test/get-ppds-psversion.test | 21 + test/get-ppds.test | 21 + test/get-printer-attributes-2.0.test | 48 + test/get-printer-attributes.test | 38 + test/get-printers.test | 19 + test/get-subscriptions.test | 21 + test/gray.jpg | Bin 0 -> 134055 bytes test/ipp-1.1.test | 2320 + test/ipp-2.0.test | 115 + test/ipp-2.1.test | 96 + test/ipp-2.2.test | 86 + test/ipp-backend.test | 22 + test/ippserver.c | 5255 + test/ipptool.c | 5249 + test/onepage-a4.pdf | Bin 0 -> 50961 bytes test/onepage-a4.ps | 34422 +++++ test/onepage-a4.sla | 156 + test/onepage-letter.pdf | Bin 0 -> 49476 bytes test/onepage-letter.ps | 31301 ++++ test/onepage-letter.sla | 160 + test/print-job-hold.test | 41 + test/print-job-media-col.test | 46 + test/print-job.test | 29 + test/print-uri.test | 27 + test/printer.opacity | Bin 0 -> 18493 bytes test/printer.png | Bin 0 -> 5133 bytes test/run-stp-tests.sh | 907 + test/set-attrs-hold.test | 180 + test/str-header.html | 30 + test/str-trailer.html | 2 + test/testfile.jpg | Bin 0 -> 204156 bytes test/testfile.pdf | Bin 0 -> 279746 bytes test/testfile.ps | 598 + test/testfile.txt | 60 + test/testhp.ppd | 187 + test/testps.ppd | 183 + test/waitjobs.sh | 60 + test/xmltotest.c | 529 + tools/checkglobals | 32 + tools/listpublic | 12 + tools/makeipptoolpkg | 88 + tools/makesrcdist | 90 + tools/pdftops-darwin.sh | 44 + tools/products.php | 50 + tools/testosx | 137 + tools/testrpm | 32 + vc2005/cups.sln | 78 + vc2005/cupstestppd.vcproj | 356 + vc2005/ipptool.vcproj | 199 + vc2005/libcups2.vcproj | 1611 + vc2005/libcupsimage2.vcproj | 395 + vc2005/testfile.vcproj | 202 + vc2005/testhttp.vcproj | 202 + vcnet/config.h | 788 + vcnet/cups.sln | 101 + vcnet/cupstestppd.vcproj | 353 + vcnet/ipptool-installer.vdproj | 1577 + vcnet/ipptool.vcproj | 385 + vcnet/libcups2.vcproj | 1612 + vcnet/libcupsimage2.vcproj | 388 + vcnet/regex/COPYRIGHT | 20 + vcnet/regex/Makefile | 130 + vcnet/regex/README | 32 + vcnet/regex/WHATSNEW | 108 + vcnet/regex/cclass.h | 31 + vcnet/regex/cname.h | 102 + vcnet/regex/debug.c | 242 + vcnet/regex/debug.ih | 14 + vcnet/regex/engine.c | 1019 + vcnet/regex/engine.ih | 35 + vcnet/regex/main.c | 510 + vcnet/regex/main.ih | 19 + vcnet/regex/mkh | 76 + vcnet/regex/regcomp.c | 1603 + vcnet/regex/regcomp.ih | 51 + vcnet/regex/regerror.c | 126 + vcnet/regex/regerror.ih | 12 + vcnet/regex/regex.3 | 509 + vcnet/regex/regex.7 | 235 + vcnet/regex/regex.h | 74 + vcnet/regex/regex2.h | 134 + vcnet/regex/regexec.c | 138 + vcnet/regex/regfree.c | 37 + vcnet/regex/split.c | 316 + vcnet/regex/tests | 477 + vcnet/regex/utils.h | 22 + vcnet/setdebug.bat | 5 + vcnet/testfile.vcproj | 201 + vcnet/testhttp.vcproj | 201 + xcode/CUPS.xcodeproj/project.pbxproj | 4554 + xcode/config.h | 736 + 757 files changed, 672803 insertions(+) create mode 100644 CHANGES-1.0.txt create mode 100644 CHANGES-1.1.txt create mode 100644 CHANGES-1.2.txt create mode 100644 CHANGES-1.3.txt create mode 100644 CHANGES-1.4.txt create mode 100644 CHANGES-1.5.txt create mode 100644 CHANGES-IPPTOOL.txt create mode 100644 CHANGES.txt create mode 100644 CREDITS.txt create mode 100644 INSTALL.txt create mode 100644 IPPTOOL.txt create mode 100644 LICENSE.txt create mode 100644 Makedefs.in create mode 100644 Makefile create mode 100644 README.txt create mode 100644 backend/Dependencies create mode 100644 backend/Makefile create mode 100644 backend/backend-private.h create mode 100644 backend/dnssd.c create mode 100644 backend/ieee1284.c create mode 100644 backend/ipp.c create mode 100644 backend/lpd.c create mode 100644 backend/network.c create mode 100755 backend/pseudo create mode 100644 backend/runloop.c create mode 100644 backend/snmp-supplies.c create mode 100644 backend/snmp.c create mode 100644 backend/snmp.txt create mode 100644 backend/socket.c create mode 100644 backend/test1284.c create mode 100644 backend/testbackend.c create mode 100644 backend/testsupplies.c create mode 100644 backend/usb-darwin.c create mode 100644 backend/usb-libusb.c create mode 100644 backend/usb-unix.c create mode 100644 backend/usb.c create mode 100644 berkeley/Dependencies create mode 100644 berkeley/Makefile create mode 100644 berkeley/lpc.c create mode 100644 berkeley/lpq.c create mode 100644 berkeley/lpr.c create mode 100644 berkeley/lprm.c create mode 100644 cgi-bin/Dependencies create mode 100644 cgi-bin/Makefile create mode 100644 cgi-bin/admin.c create mode 100644 cgi-bin/api-cgi.header create mode 100644 cgi-bin/api-cgi.shtml create mode 100644 cgi-bin/cgi-private.h create mode 100644 cgi-bin/cgi.h create mode 100644 cgi-bin/classes.c create mode 100644 cgi-bin/help-index.c create mode 100644 cgi-bin/help-index.h create mode 100644 cgi-bin/help.c create mode 100644 cgi-bin/html.c create mode 100644 cgi-bin/ipp-var.c create mode 100644 cgi-bin/jobs.c create mode 100644 cgi-bin/libcupscgi.exp create mode 100644 cgi-bin/makedocset.c create mode 100644 cgi-bin/multipart.dat create mode 100644 cgi-bin/printers.c create mode 100644 cgi-bin/search.c create mode 100644 cgi-bin/template.c create mode 100644 cgi-bin/testcgi.c create mode 100644 cgi-bin/testhi.c create mode 100644 cgi-bin/testhi.html create mode 100644 cgi-bin/testtemplate.c create mode 100644 cgi-bin/var.c create mode 100644 cgi-bin/websearch.c create mode 100644 conf/Makefile create mode 100644 conf/cupsd.conf.in create mode 100644 conf/mime.convs.in create mode 100644 conf/mime.types create mode 100644 conf/pam.opendirectory create mode 100644 conf/pam.securityserver create mode 100644 conf/pam.std.in create mode 100644 conf/snmp.conf.in create mode 100644 config-scripts/cups-common.m4 create mode 100644 config-scripts/cups-compiler.m4 create mode 100644 config-scripts/cups-defaults.m4 create mode 100644 config-scripts/cups-directories.m4 create mode 100644 config-scripts/cups-dnssd.m4 create mode 100644 config-scripts/cups-gssapi.m4 create mode 100644 config-scripts/cups-largefile.m4 create mode 100644 config-scripts/cups-launchd.m4 create mode 100644 config-scripts/cups-libtool.m4 create mode 100644 config-scripts/cups-manpages.m4 create mode 100644 config-scripts/cups-network.m4 create mode 100644 config-scripts/cups-opsys.m4 create mode 100644 config-scripts/cups-pam.m4 create mode 100644 config-scripts/cups-poll.m4 create mode 100644 config-scripts/cups-scripting.m4 create mode 100644 config-scripts/cups-sharedlibs.m4 create mode 100644 config-scripts/cups-ssl.m4 create mode 100644 config-scripts/cups-threads.m4 create mode 100644 config.h.in create mode 100644 configure.in create mode 100755 cups-config.in create mode 100644 cups/Dependencies create mode 100644 cups/Makefile create mode 100644 cups/adminutil.c create mode 100644 cups/adminutil.h create mode 100644 cups/api-array.header create mode 100644 cups/api-array.shtml create mode 100644 cups/api-cups.header create mode 100644 cups/api-cups.shtml create mode 100644 cups/api-filedir.header create mode 100644 cups/api-filedir.shtml create mode 100644 cups/api-filter.header create mode 100644 cups/api-filter.shtml create mode 100644 cups/api-httpipp.header create mode 100644 cups/api-httpipp.shtml create mode 100644 cups/api-overview.header create mode 100644 cups/api-overview.shtml create mode 100644 cups/api-ppd.header create mode 100644 cups/api-ppd.shtml create mode 100644 cups/array-private.h create mode 100644 cups/array.c create mode 100644 cups/array.h create mode 100644 cups/attr.c create mode 100644 cups/auth.c create mode 100644 cups/backchannel.c create mode 100644 cups/backend.c create mode 100644 cups/backend.h create mode 100644 cups/conflicts.c create mode 100644 cups/cups-private.h create mode 100644 cups/cups.h create mode 100644 cups/custom.c create mode 100644 cups/debug-private.h create mode 100644 cups/debug.c create mode 100644 cups/dest-job.c create mode 100644 cups/dest-localization.c create mode 100644 cups/dest-options.c create mode 100644 cups/dest.c create mode 100644 cups/dir.c create mode 100644 cups/dir.h create mode 100644 cups/emit.c create mode 100644 cups/encode.c create mode 100644 cups/file-private.h create mode 100644 cups/file.c create mode 100644 cups/file.h create mode 100644 cups/getdevices.c create mode 100644 cups/getifaddrs.c create mode 100644 cups/getputfile.c create mode 100644 cups/globals.c create mode 100644 cups/http-addr.c create mode 100644 cups/http-addrlist.c create mode 100644 cups/http-private.h create mode 100644 cups/http-support.c create mode 100644 cups/http.c create mode 100644 cups/http.h create mode 100644 cups/ipp-private.h create mode 100644 cups/ipp-support.c create mode 100644 cups/ipp.c create mode 100644 cups/ipp.h create mode 100644 cups/langprintf.c create mode 100644 cups/language-private.h create mode 100644 cups/language.c create mode 100644 cups/language.h create mode 100644 cups/libcups2.def create mode 100644 cups/libcups_s.exp create mode 100644 cups/localize.c create mode 100644 cups/mark.c create mode 100644 cups/md5-private.h create mode 100644 cups/md5.c create mode 100644 cups/md5passwd.c create mode 100644 cups/notify.c create mode 100644 cups/options.c create mode 100644 cups/page.c create mode 100644 cups/ppd-cache.c create mode 100644 cups/ppd-private.h create mode 100644 cups/ppd.c create mode 100644 cups/ppd.h create mode 100644 cups/pwg-media.c create mode 100644 cups/pwg-private.h create mode 100644 cups/raster-private.h create mode 100644 cups/raster.h create mode 100644 cups/request.c create mode 100644 cups/sidechannel.c create mode 100644 cups/sidechannel.h create mode 100644 cups/snmp-private.h create mode 100644 cups/snmp.c create mode 100644 cups/snprintf.c create mode 100644 cups/sspi-private.h create mode 100644 cups/sspi.c create mode 100644 cups/string-private.h create mode 100644 cups/string.c create mode 100644 cups/tempfile.c create mode 100644 cups/test.ppd create mode 100644 cups/test2.ppd create mode 100644 cups/testadmin.c create mode 100644 cups/testarray.c create mode 100644 cups/testconflicts.c create mode 100644 cups/testcups.c create mode 100644 cups/testfile.c create mode 100644 cups/testhttp.c create mode 100644 cups/testi18n.c create mode 100644 cups/testipp.c create mode 100644 cups/testlang.c create mode 100644 cups/testoptions.c create mode 100644 cups/testppd.c create mode 100644 cups/testpwg.c create mode 100644 cups/testsnmp.c create mode 100644 cups/thread-private.h create mode 100644 cups/thread.c create mode 100644 cups/transcode.c create mode 100644 cups/transcode.h create mode 100644 cups/usersys.c create mode 100644 cups/utf8demo.txt create mode 100644 cups/util.c create mode 100644 cups/versioning.h create mode 100644 data/Makefile create mode 100644 data/classified create mode 100644 data/confidential create mode 100644 data/cups.irix create mode 100644 data/cups.pam create mode 100644 data/cups.suse create mode 100644 data/epson.h create mode 100644 data/font.defs create mode 100644 data/hp.h create mode 100644 data/label.h create mode 100644 data/media.defs create mode 100644 data/raster.defs create mode 100644 data/secret create mode 100644 data/smiley.ps create mode 100644 data/standard create mode 100644 data/testprint.in create mode 100644 data/topsecret create mode 100644 data/unclassified create mode 100644 desktop/Makefile create mode 100644 desktop/cups-128.png create mode 100644 desktop/cups-16.png create mode 100644 desktop/cups-256.png create mode 100644 desktop/cups-32.png create mode 100644 desktop/cups-512.png create mode 100644 desktop/cups-64.png create mode 100644 desktop/cups.conf create mode 100644 desktop/cups.desktop.in create mode 100644 desktop/cups.icns create mode 100644 desktop/cups.svg create mode 100644 doc/Makefile create mode 100644 doc/cups-printable.css create mode 100644 doc/cups.css create mode 100644 doc/help/accounting.html create mode 100644 doc/help/api-array.html create mode 100644 doc/help/api-cgi.html create mode 100644 doc/help/api-cups.html create mode 100644 doc/help/api-driver.html create mode 100644 doc/help/api-filedir.html create mode 100644 doc/help/api-filter.html create mode 100644 doc/help/api-httpipp.html create mode 100644 doc/help/api-mime.html create mode 100644 doc/help/api-overview.html create mode 100644 doc/help/api-ppd.html create mode 100644 doc/help/api-ppdc.html create mode 100644 doc/help/api-raster.html create mode 100644 doc/help/cgi.html create mode 100644 doc/help/glossary.html create mode 100644 doc/help/kerberos.html create mode 100644 doc/help/license.html create mode 100644 doc/help/man-ipptool.html create mode 100644 doc/help/man-ipptoolfile.html create mode 100644 doc/help/network.html create mode 100644 doc/help/options.html create mode 100644 doc/help/overview.html create mode 100644 doc/help/policies.html create mode 100644 doc/help/postscript-driver.html create mode 100644 doc/help/ppd-compiler.html create mode 100644 doc/help/raster-driver.html create mode 100644 doc/help/ref-access_log.html create mode 100644 doc/help/ref-classes-conf.html create mode 100644 doc/help/ref-client-conf.html create mode 100644 doc/help/ref-cupsd-conf.html.in create mode 100644 doc/help/ref-error_log.html create mode 100644 doc/help/ref-mailto-conf.html create mode 100644 doc/help/ref-page_log.html create mode 100644 doc/help/ref-ppdcfile.html create mode 100644 doc/help/ref-printers-conf.html create mode 100644 doc/help/ref-snmp-conf.html create mode 100644 doc/help/ref-subscriptions-conf.html create mode 100644 doc/help/security.html create mode 100644 doc/help/sharing.html create mode 100644 doc/help/spec-banner.html create mode 100644 doc/help/spec-browsing.html create mode 100644 doc/help/spec-cmp.html create mode 100644 doc/help/spec-command.html create mode 100644 doc/help/spec-design.html create mode 100644 doc/help/spec-ipp.html create mode 100644 doc/help/spec-pdf.html create mode 100644 doc/help/spec-postscript.html create mode 100644 doc/help/spec-ppd.html create mode 100644 doc/help/spec-raster.html create mode 100644 doc/help/spec-stp.html create mode 100644 doc/help/standard.html.in create mode 100644 doc/help/translation.html create mode 100644 doc/help/whatsnew.html create mode 100644 doc/images/color-wheel.png create mode 100644 doc/images/cups-block-diagram.png create mode 100644 doc/images/cups-block-diagram.svg create mode 100644 doc/images/cups-command-chain.png create mode 100644 doc/images/cups-command-chain.svg create mode 100644 doc/images/cups-icon.png create mode 100644 doc/images/cups-postscript-chain.png create mode 100644 doc/images/cups-postscript-chain.svg create mode 100644 doc/images/cups-raster-chain.png create mode 100644 doc/images/cups-raster-chain.svg create mode 100644 doc/images/cups.png create mode 100644 doc/images/cups.svg create mode 100644 doc/images/generic.png create mode 100644 doc/images/left.gif create mode 100644 doc/images/left.xcf.gz create mode 100644 doc/images/raster-organization.png create mode 100644 doc/images/raster-organization.svg create mode 100644 doc/images/raster.png create mode 100644 doc/images/raster.svg create mode 100644 doc/images/right.gif create mode 100644 doc/images/sample-image.png create mode 100644 doc/images/sel.gif create mode 100644 doc/images/smiley.jpg create mode 100644 doc/images/unsel.gif create mode 100644 doc/images/wait.gif create mode 100644 doc/images/webinterface.png create mode 100644 doc/index.html.in create mode 100644 doc/robots.txt create mode 100644 examples/Makefile create mode 100644 examples/color.drv create mode 100644 examples/constraint.drv create mode 100644 examples/custom.drv create mode 100644 examples/grouping.drv create mode 100644 examples/laserjet-basic.drv create mode 100644 examples/laserjet-pjl.drv create mode 100644 examples/minimum.drv create mode 100644 examples/postscript.drv create mode 100644 examples/r300-basic.drv create mode 100644 examples/r300-colorman.drv create mode 100644 examples/r300-remote.drv create mode 100644 filter/Dependencies create mode 100644 filter/Makefile create mode 100644 filter/api-raster.header create mode 100644 filter/api-raster.shtml create mode 100644 filter/commandtops.c create mode 100644 filter/common.c create mode 100644 filter/common.h create mode 100644 filter/error.c create mode 100644 filter/gziptoany.c create mode 100644 filter/interpret.c create mode 100644 filter/libcupsimage2.def create mode 100644 filter/libcupsimage_s.exp create mode 100644 filter/postscript-driver.header create mode 100644 filter/postscript-driver.shtml create mode 100644 filter/ppd-compiler.header create mode 100644 filter/ppd-compiler.shtml create mode 100644 filter/pstops.c create mode 100644 filter/raster-driver.header create mode 100644 filter/raster-driver.shtml create mode 100644 filter/raster.c create mode 100644 filter/rasterbench.c create mode 100644 filter/rastertoepson.c create mode 100644 filter/rastertohp.c create mode 100644 filter/rastertolabel.c create mode 100644 filter/rastertopwg.c create mode 100644 filter/spec-ppd.header create mode 100644 filter/spec-ppd.shtml create mode 100644 filter/testraster.c create mode 100755 install-sh create mode 100644 locale/Dependencies create mode 100644 locale/Makefile create mode 100644 locale/checkpo.c create mode 100644 locale/cups.footer create mode 100644 locale/cups.header create mode 100644 locale/cups.pot create mode 100644 locale/cups.strings create mode 100644 locale/locale.txt create mode 100644 locale/po2strings.c create mode 100644 locale/strings2po.c create mode 100644 locale/translate.c create mode 100644 man/Makefile create mode 100644 man/backend.man create mode 100644 man/cancel.man create mode 100644 man/classes.conf.man create mode 100644 man/client.conf.man.in create mode 100644 man/cups-config.man create mode 100644 man/cups-deviced.man.in create mode 100644 man/cups-driverd.man.in create mode 100644 man/cups-lpd.man.in create mode 100644 man/cups-polld.man create mode 100644 man/cups-snmp.conf.man create mode 100644 man/cupsaccept.man create mode 100644 man/cupsaddsmb.man.in create mode 100644 man/cupsctl.man create mode 100644 man/cupsd.conf.man.in create mode 100644 man/cupsd.man.in create mode 100644 man/cupsenable.man create mode 100644 man/cupsfilter.man create mode 100644 man/cupstestdsc.man create mode 100644 man/cupstestppd.man create mode 100644 man/filter.man create mode 100644 man/ipptool.man create mode 100644 man/ipptoolfile.man create mode 100644 man/lp.man create mode 100644 man/lpadmin.man create mode 100644 man/lpc.man create mode 100644 man/lpinfo.man create mode 100644 man/lpmove.man create mode 100644 man/lpoptions.man.in create mode 100644 man/lppasswd.man create mode 100644 man/lpq.man create mode 100644 man/lpr.man create mode 100644 man/lprm.man create mode 100644 man/lpstat.man create mode 100644 man/mailto.conf.man create mode 100644 man/mantohtml.c create mode 100644 man/mime.convs.man create mode 100644 man/mime.types.man create mode 100644 man/notifier.man create mode 100644 man/ppdc.man create mode 100644 man/ppdcfile.man create mode 100644 man/ppdhtml.man create mode 100644 man/ppdi.man create mode 100644 man/ppdmerge.man create mode 100644 man/ppdpo.man create mode 100644 man/printers.conf.man create mode 100644 man/subscriptions.conf.man create mode 100644 monitor/Dependencies create mode 100644 monitor/Makefile create mode 100644 monitor/bcp.c create mode 100644 monitor/tbcp.c create mode 100644 notifier/Dependencies create mode 100644 notifier/Makefile create mode 100644 notifier/dbus.c create mode 100644 notifier/mailto.c create mode 100644 notifier/rss.c create mode 100644 notifier/testnotify.c create mode 100755 packaging/InstallationCheck create mode 100644 packaging/LICENSE.rtf create mode 100644 packaging/WELCOME.rtf create mode 100644 packaging/cups-desc.plist.in create mode 100644 packaging/cups-info.plist.in create mode 100644 packaging/cups.list.in create mode 100644 packaging/cups.spec.in create mode 100644 packaging/installer.gif create mode 100644 packaging/installer.tif create mode 100644 ppdc/Dependencies create mode 100644 ppdc/Makefile create mode 100644 ppdc/api-ppdc.header create mode 100644 ppdc/api-ppdc.shtml create mode 100644 ppdc/foo-fr.po create mode 100644 ppdc/foo.drv create mode 100644 ppdc/genstrings.cxx create mode 100644 ppdc/ppdc-array.cxx create mode 100644 ppdc/ppdc-attr.cxx create mode 100644 ppdc/ppdc-catalog.cxx create mode 100644 ppdc/ppdc-choice.cxx create mode 100644 ppdc/ppdc-constraint.cxx create mode 100644 ppdc/ppdc-driver.cxx create mode 100644 ppdc/ppdc-file.cxx create mode 100644 ppdc/ppdc-filter.cxx create mode 100644 ppdc/ppdc-font.cxx create mode 100644 ppdc/ppdc-group.cxx create mode 100644 ppdc/ppdc-import.cxx create mode 100644 ppdc/ppdc-mediasize.cxx create mode 100644 ppdc/ppdc-message.cxx create mode 100644 ppdc/ppdc-option.cxx create mode 100644 ppdc/ppdc-private.h create mode 100644 ppdc/ppdc-profile.cxx create mode 100644 ppdc/ppdc-shared.cxx create mode 100644 ppdc/ppdc-source.cxx create mode 100644 ppdc/ppdc-string.cxx create mode 100644 ppdc/ppdc-variable.cxx create mode 100644 ppdc/ppdc.cxx create mode 100644 ppdc/ppdc.h create mode 100644 ppdc/ppdhtml.cxx create mode 100644 ppdc/ppdi.cxx create mode 100644 ppdc/ppdmerge.cxx create mode 100644 ppdc/ppdpo.cxx create mode 100644 ppdc/sample.drv create mode 100644 ppdc/testcatalog.cxx create mode 100644 scheduler/Dependencies create mode 100644 scheduler/Makefile create mode 100644 scheduler/api-mime.header create mode 100644 scheduler/api-mime.shtml create mode 100644 scheduler/auth.c create mode 100644 scheduler/auth.h create mode 100644 scheduler/banners.c create mode 100644 scheduler/banners.h create mode 100644 scheduler/cert.c create mode 100644 scheduler/cert.h create mode 100644 scheduler/classes.c create mode 100644 scheduler/classes.h create mode 100644 scheduler/client.c create mode 100644 scheduler/client.h create mode 100644 scheduler/conf.c create mode 100644 scheduler/conf.h create mode 100644 scheduler/cups-deviced.c create mode 100644 scheduler/cups-driverd.cxx create mode 100644 scheduler/cups-exec.c create mode 100644 scheduler/cups-lpd.c create mode 100644 scheduler/cups-lpd.xinetd.in create mode 100644 scheduler/cups.sh.in create mode 100644 scheduler/cups.xml.in create mode 100644 scheduler/cupsd.h create mode 100644 scheduler/cupsfilter.c create mode 100644 scheduler/dirsvc.c create mode 100644 scheduler/dirsvc.h create mode 100644 scheduler/env.c create mode 100644 scheduler/file.c create mode 100644 scheduler/filter.c create mode 100644 scheduler/ipp.c create mode 100644 scheduler/job.c create mode 100644 scheduler/job.h create mode 100644 scheduler/libcupsmime.exp create mode 100644 scheduler/listen.c create mode 100644 scheduler/log.c create mode 100644 scheduler/main.c create mode 100644 scheduler/mime-private.h create mode 100644 scheduler/mime.c create mode 100644 scheduler/mime.h create mode 100644 scheduler/network.c create mode 100644 scheduler/network.h create mode 100644 scheduler/org.cups.cups-lpd.plist.in create mode 100644 scheduler/org.cups.cupsd.plist create mode 100644 scheduler/policy.c create mode 100644 scheduler/policy.h create mode 100644 scheduler/printers.c create mode 100644 scheduler/printers.h create mode 100644 scheduler/process.c create mode 100644 scheduler/quotas.c create mode 100644 scheduler/select.c create mode 100644 scheduler/server.c create mode 100644 scheduler/statbuf.c create mode 100644 scheduler/statbuf.h create mode 100644 scheduler/subscriptions.c create mode 100644 scheduler/subscriptions.h create mode 100644 scheduler/sysman.c create mode 100644 scheduler/sysman.h create mode 100644 scheduler/testlpd.c create mode 100644 scheduler/testmime.c create mode 100644 scheduler/testspeed.c create mode 100644 scheduler/testsub.c create mode 100644 scheduler/type.c create mode 100644 scheduler/util.c create mode 100644 scheduler/util.h create mode 100644 systemv/Dependencies create mode 100644 systemv/Makefile create mode 100644 systemv/cancel.c create mode 100644 systemv/cupsaccept.c create mode 100644 systemv/cupsaddsmb.c create mode 100644 systemv/cupsctl.c create mode 100644 systemv/cupstestdsc.c create mode 100644 systemv/cupstestppd.c create mode 100644 systemv/lp.c create mode 100644 systemv/lpadmin.c create mode 100644 systemv/lpinfo.c create mode 100644 systemv/lpmove.c create mode 100644 systemv/lpoptions.c create mode 100644 systemv/lppasswd.c create mode 100644 systemv/lpstat.c create mode 100644 templates/Makefile create mode 100644 templates/add-class.tmpl create mode 100644 templates/add-printer.tmpl create mode 100644 templates/add-rss-subscription.tmpl create mode 100644 templates/admin.tmpl create mode 100644 templates/choose-device.tmpl create mode 100644 templates/choose-make.tmpl create mode 100644 templates/choose-model.tmpl create mode 100644 templates/choose-serial.tmpl create mode 100644 templates/choose-uri.tmpl create mode 100644 templates/class-added.tmpl create mode 100644 templates/class-confirm.tmpl create mode 100644 templates/class-deleted.tmpl create mode 100644 templates/class-jobs-header.tmpl create mode 100644 templates/class-modified.tmpl create mode 100644 templates/class.tmpl create mode 100644 templates/classes-header.tmpl create mode 100644 templates/classes.tmpl create mode 100644 templates/command.tmpl create mode 100644 templates/edit-config.tmpl create mode 100644 templates/error-op.tmpl create mode 100644 templates/error.tmpl create mode 100644 templates/header.tmpl.in create mode 100644 templates/help-header.tmpl create mode 100644 templates/help-printable.tmpl create mode 100644 templates/help-trailer.tmpl create mode 100644 templates/job-cancel.tmpl create mode 100644 templates/job-hold.tmpl create mode 100644 templates/job-move.tmpl create mode 100644 templates/job-moved.tmpl create mode 100644 templates/job-release.tmpl create mode 100644 templates/job-restart.tmpl create mode 100644 templates/jobs-header.tmpl create mode 100644 templates/jobs.tmpl create mode 100644 templates/list-available-printers.tmpl create mode 100644 templates/modify-class.tmpl create mode 100644 templates/modify-printer.tmpl create mode 100644 templates/norestart.tmpl create mode 100644 templates/option-boolean.tmpl create mode 100644 templates/option-conflict.tmpl create mode 100644 templates/option-header.tmpl create mode 100644 templates/option-pickmany.tmpl create mode 100644 templates/option-pickone.tmpl create mode 100644 templates/option-trailer.tmpl create mode 100644 templates/pager.tmpl create mode 100644 templates/printer-accept.tmpl create mode 100644 templates/printer-added.tmpl create mode 100644 templates/printer-configured.tmpl create mode 100644 templates/printer-confirm.tmpl create mode 100644 templates/printer-default.tmpl create mode 100644 templates/printer-deleted.tmpl create mode 100644 templates/printer-jobs-header.tmpl create mode 100644 templates/printer-modified.tmpl create mode 100644 templates/printer-purge.tmpl create mode 100644 templates/printer-reject.tmpl create mode 100644 templates/printer-start.tmpl create mode 100644 templates/printer-stop.tmpl create mode 100644 templates/printer.tmpl create mode 100644 templates/printers-header.tmpl create mode 100644 templates/printers.tmpl create mode 100644 templates/restart.tmpl create mode 100644 templates/samba-export.tmpl create mode 100644 templates/samba-exported.tmpl create mode 100644 templates/search.tmpl create mode 100644 templates/set-printer-options-header.tmpl create mode 100644 templates/set-printer-options-trailer.tmpl create mode 100644 templates/subscription-added.tmpl create mode 100644 templates/subscription-canceled.tmpl create mode 100644 templates/test-page.tmpl create mode 100644 templates/trailer.tmpl create mode 100644 templates/users.tmpl create mode 100644 test/4.1-requests.test create mode 100644 test/4.2-cups-printer-ops.test create mode 100644 test/4.3-job-ops.test create mode 100644 test/4.4-subscription-ops.test create mode 100755 test/5.1-lpadmin.sh create mode 100755 test/5.2-lpc.sh create mode 100755 test/5.3-lpq.sh create mode 100755 test/5.4-lpstat.sh create mode 100755 test/5.5-lp.sh create mode 100755 test/5.6-lpr.sh create mode 100755 test/5.7-lprm.sh create mode 100755 test/5.8-cancel.sh create mode 100755 test/5.9-lpinfo.sh create mode 100644 test/Dependencies create mode 100644 test/Makefile create mode 100644 test/color.jpg create mode 100644 test/create-job-format.test create mode 100644 test/create-job-sheets.test create mode 100644 test/create-job-timeout.test create mode 100644 test/create-job.test create mode 100644 test/create-printer-subscription.test create mode 100644 test/document-a4.pdf create mode 100644 test/document-a4.ps create mode 100644 test/document-a4.sla create mode 100644 test/document-letter.pdf create mode 100644 test/document-letter.ps create mode 100644 test/document-letter.sla create mode 100644 test/get-completed-jobs.test create mode 100644 test/get-devices.test create mode 100644 test/get-job-attributes.test create mode 100644 test/get-job-attributes2.test create mode 100644 test/get-jobs.test create mode 100644 test/get-ppd-printer.test create mode 100644 test/get-ppd.test create mode 100644 test/get-ppds-drv-only.test create mode 100644 test/get-ppds-language.test create mode 100644 test/get-ppds-make-and-model.test create mode 100644 test/get-ppds-make.test create mode 100644 test/get-ppds-product.test create mode 100644 test/get-ppds-psversion.test create mode 100644 test/get-ppds.test create mode 100644 test/get-printer-attributes-2.0.test create mode 100644 test/get-printer-attributes.test create mode 100644 test/get-printers.test create mode 100644 test/get-subscriptions.test create mode 100644 test/gray.jpg create mode 100644 test/ipp-1.1.test create mode 100644 test/ipp-2.0.test create mode 100644 test/ipp-2.1.test create mode 100644 test/ipp-2.2.test create mode 100644 test/ipp-backend.test create mode 100644 test/ippserver.c create mode 100644 test/ipptool.c create mode 100644 test/onepage-a4.pdf create mode 100644 test/onepage-a4.ps create mode 100644 test/onepage-a4.sla create mode 100644 test/onepage-letter.pdf create mode 100644 test/onepage-letter.ps create mode 100644 test/onepage-letter.sla create mode 100644 test/print-job-hold.test create mode 100644 test/print-job-media-col.test create mode 100644 test/print-job.test create mode 100644 test/print-uri.test create mode 100644 test/printer.opacity create mode 100644 test/printer.png create mode 100755 test/run-stp-tests.sh create mode 100644 test/set-attrs-hold.test create mode 100644 test/str-header.html create mode 100644 test/str-trailer.html create mode 100644 test/testfile.jpg create mode 100644 test/testfile.pdf create mode 100644 test/testfile.ps create mode 100644 test/testfile.txt create mode 100644 test/testhp.ppd create mode 100644 test/testps.ppd create mode 100755 test/waitjobs.sh create mode 100644 test/xmltotest.c create mode 100755 tools/checkglobals create mode 100755 tools/listpublic create mode 100755 tools/makeipptoolpkg create mode 100755 tools/makesrcdist create mode 100755 tools/pdftops-darwin.sh create mode 100755 tools/products.php create mode 100755 tools/testosx create mode 100755 tools/testrpm create mode 100644 vc2005/cups.sln create mode 100644 vc2005/cupstestppd.vcproj create mode 100644 vc2005/ipptool.vcproj create mode 100644 vc2005/libcups2.vcproj create mode 100644 vc2005/libcupsimage2.vcproj create mode 100644 vc2005/testfile.vcproj create mode 100644 vc2005/testhttp.vcproj create mode 100644 vcnet/config.h create mode 100644 vcnet/cups.sln create mode 100644 vcnet/cupstestppd.vcproj create mode 100644 vcnet/ipptool-installer.vdproj create mode 100644 vcnet/ipptool.vcproj create mode 100644 vcnet/libcups2.vcproj create mode 100644 vcnet/libcupsimage2.vcproj create mode 100644 vcnet/regex/COPYRIGHT create mode 100644 vcnet/regex/Makefile create mode 100644 vcnet/regex/README create mode 100644 vcnet/regex/WHATSNEW create mode 100644 vcnet/regex/cclass.h create mode 100644 vcnet/regex/cname.h create mode 100644 vcnet/regex/debug.c create mode 100644 vcnet/regex/debug.ih create mode 100644 vcnet/regex/engine.c create mode 100644 vcnet/regex/engine.ih create mode 100644 vcnet/regex/main.c create mode 100644 vcnet/regex/main.ih create mode 100644 vcnet/regex/mkh create mode 100644 vcnet/regex/regcomp.c create mode 100644 vcnet/regex/regcomp.ih create mode 100644 vcnet/regex/regerror.c create mode 100644 vcnet/regex/regerror.ih create mode 100644 vcnet/regex/regex.3 create mode 100644 vcnet/regex/regex.7 create mode 100644 vcnet/regex/regex.h create mode 100644 vcnet/regex/regex2.h create mode 100644 vcnet/regex/regexec.c create mode 100644 vcnet/regex/regfree.c create mode 100644 vcnet/regex/split.c create mode 100644 vcnet/regex/tests create mode 100644 vcnet/regex/utils.h create mode 100644 vcnet/setdebug.bat create mode 100644 vcnet/testfile.vcproj create mode 100755 vcnet/testhttp.vcproj create mode 100644 xcode/CUPS.xcodeproj/project.pbxproj create mode 100644 xcode/config.h diff --git a/CHANGES-1.0.txt b/CHANGES-1.0.txt new file mode 100644 index 0000000000..296d89f791 --- /dev/null +++ b/CHANGES-1.0.txt @@ -0,0 +1,217 @@ +CHANGES-1.0.txt +--------------- + +CHANGES IN CUPS v1.0.5 + + - The HP-GL/2 filter did not correctly set the pen color + for pens other than #1. + - The scheduler would only accept 26 simultaneous jobs + under some OS releases (mkstemp() limitation.) It now + handles up to 2^32 simultaneous jobs. + - The PostScript filter loaded the printer's PPD file + twice. + - The PAM authentication code now uses pam_strerror() to + provide a textual error message in the error_log file. + - The scheduler now copies PPD and interface script + files instead of moving them; this fixes installations + with a separate requests directory. + - The PostScript RIP did not generate correct 6-color + output. + - Several filters were marking PPD options twice when + they didn't need to. + - The scheduler did not save the printer or class state + after an accept-jobs or reject-jobs operation. + - The cupsGetDefault() function now ignores the PRINTER + environment variable if it is set to "lp". + - New ippErrorString() function to get textual error + messages. + - Better error reporting in the System V commands. + - The lpadmin and lpstat commands always tried to + connect to the default server. + - The text filter didn't load the charset files from the + correct location. + - Wasn't sending a WWW-Authenticate: field to HTTP + clients when authentication was required. + - httpSeparate() didn't always set the default port + number for known methods. + - The HP-GL/2 filter now looks for "PSwidth,length" + instead of (the correct) "PSlength,width" as + documented by HP. It appears that many major CAD + applications are broken and this change allows the + auto-rotation to work with them. + - The IPP "printer-resolution" option was not being + translated. + - The charset files did not include the Microsoft + "standard" characters from 128 to 159 (unused by the + ISO-8859-x charsets) + - The scheduler was chunking the Content-Type field from + CGI programs; this problem was most noticeable with + Microsoft Internet Explorer 5. + - By popular demand, the printers, jobs, and classes + CGIs no longer force a reload of the page every 10/30 + seconds. + - The scheduler incorrectly required that the IPP client + provide a document-format attribute for the + validate-job operation. + - Clients that sent bad IPP requests without the + required attributes-natural-language and + attributes-charset attributes would crash the + scheduler. + + +CHANGES IN CUPS v1.0.4 + + - Documentation updates. + - Jobs would get stuck in the queue and wouldn't print + until you enabled the queue. + - The lp and lpr commands now catch SIGHUP and SIGINTR. + - The lp and lpr commands now use sigaction or sigset + when available. + - CUPS library updates for WIN32/OS-2 + + +CHANGES IN CUPS v1.0.3 + + - Documentation updates. + - The lpq man page was missing. + - The configure script was not properly detecting the + image libraries. + - The top-level makefile was calling "make" instead of + "$(MAKE)". + - PostScript filter fixes for number-up, OutputOrder, + and %Trailer. + - The imagetops filter didn't end the base-85 encoding + properly if the image data was not a multiple of 4 + bytes in length. + - The imagetoraster filter didn't generate good banded + RGB or CMY data (was dividing the line width by 4 + instead of 3...) + - The imagetoraster filter now records the bounding + box of the image on the page. + - The CUPS image library cache code wasn't working as + designed; images larger than the maximum RIP cache + would eventually thrash using the same cache tile. + - The CUPS image library TIFF loading code didn't + handle unknown resolution units properly; the fixed + code uses a default resolution of 128 PPI. + - cupsGetClasses() and cupsGetPrinters() did not free + existing strings if they ran out of memory. + - The scheduler logs incorrectly contained 3 digits for + the timezone offset instead of 4. + - The scheduler now does a lookup for the default user + and group ID; the previous hardcoded values caused + problems with the LPD backend. + - The cancel-job operation now allows any user in the + system group to cancel any job. + - The cancel-job operation stopped the print queue if + the job was being printed. + - Now only stop printers if the backend fails. If the + filter fails then the failure is noted in the + error_log and printing continues with the next file in + the queue. + - Now log whether a filter fails because of a signal + or because it returned a non-zero exit status. + - The root user now always passes the system group test. + - Printers with an interface script and remote printers + and classes didn't have a printer-make-and-model + attribute. + - Added logging of lost/timed-out remote printers. + - The HP-GL/2 filter was scaling the pen width twice. + - Updated the HP-GL/2 filter to use a single SP (Set + Pen) procedure. This makes the output smaller and is + more appropriate since the filter keeps track of the + pen states already. + - The scheduler didn't handle passwords with spaces. + - The IPP backend now does multiple copies and retries + if the destination server requires it (e.g. HP + JetDirect.) + - The disable command didn't implement the "-c" option + (cancel all jobs.) + - Changed the CMYK generation function for the image file + and PostScript RIPs. + - The lp command didn't support the "-h" option as + documented. + - The AppSocket, IPP, and LPD backends now retry on all + network errors. This should prevent stopped queues + caused by a printer being disconnected from the + network or powered off. + - The scheduler now restarts a job if the corresponding + printer is modified. + - The image RIPs now rotate the image if needed to fit + on the page. + + +CHANGES IN CUPS v1.0.2 + + - The HP-GL/2 filter didn't always scale the output + correctly. + - The HP-GL/2 filter now supports changing the page size + automatically when the "fitplot" option is not used. + - The cancel-job operation was expecting a resource name + of the form "/job/#" instead of "/jobs/#"; this + prevented the cancel and lprm commands from working. + - The backends didn't log pages when files were printed + using the "-oraw" option. + - The authorization code did not work with the Slackware + long shadow password package because its crypt() can + return NULL. + - The chunking code didn't work for reading the response + of a POST request. + - cupsGetPPD() now does authentication as needed. + - The N-up code in the PostScript filter didn't work + with some printers (grestoreall would restore the + default blank page and device settings). + - The N-up code in the PostScript filter didn't scale + the pages to fit within the imageable area of the + page. + - Wasn't doing an fchown() on the request files. This + caused problems when the default root account group + and CUPS group were not the same. + + +CHANGES IN CUPS v1.0.1 + + - Documentation updates. + - Fixed a bunch of possible buffer-overflow conditions. + - The scheduler now supports authentication using PAM. + - Updated the Italian message file. + - httpEncode64() didn't add an extra "=" if there was + only one byte in the last three-byte group. + - Now drop any trailing character set from the locale + string (e.g. "en_US.ISO_8859-1" becomes "en_US") + - Fixed "timezone" vs "tm_gmtoff" usage for BSD-based + operating systems. + - Updated IPP security so that "get" operations can be + done from any resource name; this allows the CGIs to + work with printer authentication enabled so long as + authentication isn't turned on for the whole "site". + - The IPP code didn't properly handle the "unsupported" + group; this caused problems with the HP JetDirect since + it doesn't seem to support the "copies" attribute. + - The HTTP chunking code was missing a CR LF pair at the + end of a 0-length chunk. + - The httpSeparate() function didn't handle embedded + usernames and passwords in the URI properly. + - Doing "lpadmin -p printer -E" didn't restart printing + if there were pending jobs. + - The cancel-job operation now requires either a + requesting-user-name attribute or an authenticated + username. + - The add-printer code did not report errors if the + interface script or PPD file could not be renamed. + - Request files are now created without world read + permissions. + - Added a cupsLastError() function to the CUPS API to + retrieve the IPP error code from the last request. + - Options are now case-insensitive. + - The lpq command now provides 10 characters for the + username instead of the original (Berkeley standard) + 7. + - The cancel command needed a local CUPS server to work + (or the appropriate ServerName in cupsd.conf) + - The cancel and lprm commands didn't report the IPP + error if the job could not be cancelled. + - The lp and lpr commands didn't intercept SIGTERM to + remove temporary files when printing from stdin. + - The lp and lpr commands didn't report the IPP error if + the job could not be printed. diff --git a/CHANGES-1.1.txt b/CHANGES-1.1.txt new file mode 100644 index 0000000000..c641e794a3 --- /dev/null +++ b/CHANGES-1.1.txt @@ -0,0 +1,3462 @@ +CHANGES-1.1.txt +--------------- + +CHANGES IN CUPS V1.1.23 + + - Updated the Spanish man pages (STR #1041) + - The lpstat man page contained a typo (STR #1040) + - The scheduler's is_path_absolute() code could cause a + DoS (STR #1042) + - The scheduler's device loading code used the wrong + size limits for the make/model and info parameters + (STR #1035) + - The PNG loading code did not use a "long unsigned + integer" format specifier for the width and height + (STR #1032) + - The web interface only showed the first 4 or 8 + characters of "{variable-name}" for undefined template + variables (STR #1031) + - The hpgltops filter did not handle a common PCL + command to enter HP-GL/2 mode (STR #1037) + + +CHANGES IN CUPS V1.1.23rc1 + + - The lpr man page did not document the "-U" option (STR + #998) + - The scheduler no longer sends the page-set option when + printing banner pages (STR #995) + - Fixed a debug message in the imagetops filter (STR + #1012) + - The lprm man page listed the "-" option in the wrong + order (STR #911) + - The hpgltops filter contained two buffer overflows + that could potentially allow remote access to the "lp" + account (STR #1024) + - The lppasswd command did not protect against file + descriptor or ulimit attacks (STR #1023) + - The "lpc status" command used the wrong resource path + when querying the list of printers and jobs, causing + unnecessary authentication requests (STR #1018) + - The httpWait() function did not handle signal + interruptions (STR #1020) + - The USB backend used the wrong size status variable + when checking the printer status (STR #1017) + - The scheduler did not delete classes from other + classes or implicit classes, which could cause a crash + (STR #1015) + - The IPP backend now logs the remote print job ID at + log level NOTICE instead of INFO (so it shows up in + the error_log file...) + + +CHANGES IN CUPS V1.1.22 + + - The lpstat man page incorrectly listed the "-s" option + as using the equivalent of the "-p" option to list the + printers; it uses the "-v" option to list the printers + (STR #986) + - Now allow 0-length reads in the CUPS file API (STR + #985) + - cupsDoFileRequest() now sets cupsLastError() to + IPP_ERROR on network errors (STR #953) + - The pdftops filter didn't scale small pages up to the + output page size when the fitplot option was used (STR + #984) + - Fixed the ipptest program usage message (STR #959) + - Added Spanish man pages (STR #963) + - Fixed the order of comparisons in the client.conf + reading code (STR #971) + - cupsLangGet() incorrectly set the current locale (STR + #970) + + +CHANGES IN CUPS V1.1.22rc2 + + - The pdftops filter didn't check the range of all + integer attributes (STR #972) + - Documentation corrections (STR #944, STR #946) + - Also sanitize device URI in argv[0] (STR #933) + - cupsRasterReadHeader() didn't swap bytes for the + numeric fields properly (STR #930) + + +CHANGES IN CUPS V1.1.22rc1 + + - Now sanitize the device URI that is reported in the + error_log file (STR #920) + - Fixed some memory and file descriptor leaks in the job + dispatch code (STR #921) + - Deleting a printer could cause a crash with browsing + enabled (STR #865, STR #881, STR #928) + - Browsing would turn off if the scheduler got an EAGAIN + error (STR #924) + - The mime.types file didn't recognize PostScript as a + PJL language name (STR #925) + + +CHANGES IN CUPS V1.1.21 + + - The scheduler did not separate Digest authentication + parameters with commas (STR #882) + - Fixed some problems with image printing to custom page + sizes (STR #891) + - Removed the remaining scheduler code that did not use + the "close-on-exec" file descriptor flag to speed up + program invocations (STR #890) + - The "lpr -r" command removed the print file even if it + was not printed. It now only removes the file if the + job is successfully created (STR #886) + - Revamped the custom page size orientation fix (STR + #127) + - The lp, lpq, lpr, and lpstat commands now report when + an environment variable is pointing to a non-existent + printer instead of just saying "no default + destination" (STR #879) + - Queue names with 2 periods (e.g. "printer..2") were + not supported (STR #866) + + +CHANGES IN CUPS V1.1.21rc2 + + - Fixed a denial-of-service bug in the CUPS browse + protocol support (STR #863) + - The scheduler used a select() timeout of INT_MAX + seconds when there was nothing to do, which doesn't + work on IRIX (STR #864) + - Updated the cupsaddsmb program to use the new Windows + 2000 PostScript drivers instead of the Windows NT + printer drivers (STR #390) + - The gziptoany filter did not produce copies for raw + print jobs (STR #808) + - The cupsLangGet() function now uses nl_langinfo(), + when available, to get the current encoding (STR #856) + - Added a ReloadTimeout directive to control how long + the scheduler waits for jobs to complete before + restarting the scheduler (STR #861) + - Added a note to the default cupsd.conf file which + mentions that you must allow connections from + localhost for the command-line and web interfaces to + work (STR #850) + - The IPP backend incorrectly used the local port when + communicating with a remote server; this caused + problems with some custom configurations (STR #852) + - The cups-lpd mini-daemon wasn't using the right + default banner option (STR #851) + - Updated the new httpDecode64_2() and httpEncode64_2() + functions to handle arbitrary binary data, not just + text (STR #860) + - String options with quotes in their values were not + quoted properly by the scheduler (STR #839) + - Configure script changes for GNU/Hurd (STR #838) + - The lppasswd program was not installed properly by GNU + install when the installer was not root (STR #836) + - Updated the cups-lpd man page (STR #843) + - Fixed a typo in the cupsd man page (STR #833) + - The USB backend now defaults to using the newer + /dev/usb/lpN filenames; this helps on systems which + use the devfs filesystem type on Linux (STR #818) + - The config.h file did not define the HAVE_USERSEC_H + constant when the configure script detected the + usersec.h header file. This caused authentication + errors on AIX (STR #832) + - The lp and lpr commands now report the temporary + filename and error if they are unable to create a + temporary file (STR #812) + - Added ServerTokens directive to control the Server + header in HTTP responses (STR #792) + - Added new httpDecode64_2(), httpEncode64_2(), and + httpSeparate2() functions which offer buffer size + arguments (STR #797) + - The cupsGetFile() and cupsPutFile() code did not + support CDSA or GNUTLS (STR #794) + - The httpSeparate() function did not decode all + character escapes (STR #795) + - The cupstestppd program now checks for invalid Duplex + option choices and fails PPD files that use + non-standard values (STR #791) + - Updated the printer name error message to indicate + that spaces are not allowed (STR #675) + - The scheduler didn't handle HTTP GET form data + properly (STR #744) + - The pstops filter now makes sure that the prolog code + is sent before the setup code (STR #776) + - The pstops filter now handles print files that + incorrectly start @PJL commands without a language + escape (STR #734) + - Miscellaneous build fixes for NetBSD (STR #788) + - Added support for quoted system group names (STR #784) + - Added "version" option to IPP backend to workaround + serious bug in Linksys's IPP implementation (STR #767) + - Added Spanish translation of web interface (STR #772, + STR #802) + - The LPD backend now uses geteuid() instead of getuid() + when it is available (STR #752) + - The IPP backend did not report the printer state if + the wait option was set to "no" (STR #761) + - The printer state was not updated for "STATE: foo,bar" + messages (STR #745) + - Added new CUPS API convenience functions which accept + a HTTP connection to eliminate extra username/password + prompts. This resolves a previous authentication + caching issue (STR #729, STR #743) + - The scheduler did not correctly throttle the browse + broadcasts, resulting in missing printers on client + machines (STR #754) + - The scheduler did not pass the correct CUPS_ENCRYPTION + setting to CGI programs which caused problems on + systems which used non-standard encryption settings + (STR #773) + - The lpq command showed 11st, 12nd, and 13rd instead of + 11th, 12th, and 13th for the rank (STR #769) + - "make install" didn't work on some platforms due to an + error in the man page makefiles (STR #775) + - Changed some calls to snprintf() in the scheduler to + SetStringf() (STR #740) + + +CHANGES IN CUPS V1.1.21rc1 + + - Fixed some "type-punned" warnings produced by GCC when + -fstrict-aliasing is specified (STR #679) + - The PDF filter incorrectly calculated the bounding box + of a page (STR #682) + - The IPP backend did not use SSL when printing over a + port other than 443 (STR #730) + - The scheduler could crash when processing a Limit or + LimitExcept directive (STR #728) + - The lpq, lpr, and lp commands did not differentiate + between the server being unresponsive and the lack of + a default printer (STR #728) + - The PAM checks in the configure script did not stop + after the first match (STR #728) + - The cups-config man page was incorrectly placed in + section 3 (STR #728) + - The cupstestppd utility did not show a warning message + when a PPD file indicated BCP protocol support with + PJL (STR #720) + - The scheduler did not return the correct exit code + when startup failed (STR #718) + - The cupsRasterReadPixels() function checked for + EAGAIN, which caused problems on FreeBSD (STR #723) + - The cupsGetDests() function did not use the current + encryption setting (STR #653) + - The scheduler did not properly parse name-based + BrowseRelay directives in the cupsd.conf file (STR + #711) + - The IPP backend now supports the following options in + the device URI: encryption, waitjob, and waitprinter + (STR #699) + - The parallel, serial, socket, and USB backends did not + return a non-zero exit status when a job failed to + print in the middle of sending it (STR #715) + - Location directives in the cupsd.conf file were + case-sensitive for printer and class names, so + queue-specific access control was not reliable (STR + #700) + - cupsDoFileRequest() did not handle HTTP continue + status messages in all cases, causing sporatic + problems with IPP printers from some vendors (STR + #716) + - The rastertodymo driver now supports the Zebra ZPL + language (STR #713) + - The test suite no longer generates a printcap file, + which caused problems when testing as the root user + (STR #693) + - The scheduler now updates the accepting state of an + implicit class based upon the accepting state of its + member printers (STR #697) + - The pstops filter didn't properly skip leading PJL + commands (STR #664) + - The reinterpret_cast keyword was not highlighted when + printing C/C++ source files in prettyprint mode (STR + #694) + - Fixed a segfault problem with some of the client + programs (STR #668) + - When using RunAsUser, the scheduler did not correctly + set the ownership of the log files, preventing log + file rotation (STR #686) + - The image filters did not correctly load 1-bit PNG + files (STR #687) + - The pdftops filter did not show all annotation objects + in a PDF file (STR #674) + - The pdftops filter did not print the contents of + textual form elements, making it impossible to print a + filled-in form (STR #663) + - Integrated the MacOS X/Darwin USB backend into the + CUPS baseline (STR #661) + - The USB backend incorrectly reported "media tray + empty" (STR #660) + - The scheduler did not use a case-insensitive + comparison when checking for group membership, which + caused problems with Win9x clients printing via SAMBA + (STR #647) + - The scheduler did not report the addresses associated + with certain network errors, making troubleshooting + difficult (STR #648, #649) + - The cupstestppd program did not allow a default choice + of "Unknown" as required by the PPD spec (STR #651) + - The select() buffers are now allocated to be at least + as large as sizeof(fd_set) (STR #639) + - The LPD backend now supports overriding the print job + username via the device URI (STR #631) + - The scheduler did not handle an unknown MIME type when + checking for a CGI script (STR #603) + - Added a timeout optimization to the scheduler's main + loop to allow CUPS to sleep more of the time (STR + #629) + - The USB backend now retries printing to devices of the + form "usb://make/model" if any USB port shows up as + "busy" (STR #617) + - The httpGetHostByName() function did not range check + IP address values (STR #608) + - The httpUpdate() function could return HTTP_ERROR + instead of the HTTP status if the server closed the + connection before the client received the whole + response (STR #611) + - The LPD mini-daemon did not allow the administrator to + force banner pages on (STR #605) + - Added PAM support for Darwin/MacOS X (STR #550) + - The web interface now provides a "Set As Default" + button to set the default printer or class on a server + (STR #577) + - The HTTP authentication cache was broken (STR #517) + - The cupstestppd utility now fails PPD files that have + a DefaultOption keyword for a non-existance option + name (STR #476) + - Optimized the scanning of new PPD files on scheduler + startup (STR #424) + - The EPM list file did not include the bin, lib, or + sbin directories (STR #598) + - The web interface did not redirect administration + tasks to the primary server for a class or printer + (STR #491, #652) + - The cups-lpd mini-daemon did not reject print jobs to + queues that were rejecting new print jobs (STR #515) + - Some calls to the ctype functions did not account for + platforms that use a signed char type by default (STR + #518) + - The scheduler could use excess amounts of CPU if a CGI + program was sending data faster than the client could + take it (STR #595) + - Updated the Ghostscript 8.x integration stuff (STR + #484) + - The lpd backend used a source port of 732 by default, + which is outside of the range defined by RFC 1179; + also added a new (default) "reserve=any" option for + any priviledged port from 1 to 1023 (STR #474) + - The scheduler did not check for a valid Listen/Port + configuration (STR #499) + - The cupsPrintFiles() function did not always set the + last IPP error message (STR #538) + - The pstops filter did not write the PostScript header + line if the file began with a PJL escape sequence (STR + #574) + - The printer-is-accepting-jobs status of remote + printers was not sent to clients via browsing or + polling (STR #571) + - Browse packets did not indicate whether a printer + was accepting or rejecting jobs. + - The web interface did not show the printer state + history information (STR #592) + - The rastertoepson filter would crash under certain + cirsumstances (STR #583) + - The USB backend did not handle serial numbers using + the (incorrect) SN keyword and did not terminate the + make and model name strings properly (STR #471, STR + #588) + - The USB backend did not build on Solaris x86 (STR + #585) + - The cupsDoAuthentication() function did not use the + method name for Digest authentication (STR #584) + - The scheduler could crash if a print job could not be + printed and the PreserveJobHistory option was turned + off (STR #535) + - cups-lpd now logs the temporary filenames that could + not be opened in order to make troubleshooting easier + (STR #565) + - cupsGetJobs() now returns -1 on error (STR #569) + - Added localization for Belarusian (STR #575) + - The LPD backend used the full length of the hostname + when creating the data and control filenames, which + causes problems with older systems that can't handle + long filenames (STR #560) + - The scheduler did not refresh the common printer data + after a fast reload; this prevented banner and other + information from being updated (STR #562) + - The scheduler did not send common or history data to + the client when processing a CUPS-Get-Default request + (STR #559) + - The httpFlush() function did not always flush the + remaining response data in requests (STR #558) + - The scheduler could complete a job before it collected + the exit status from all filters and the backend (STR + #448) + - The PPD conformance tests did not catch group + translation strings that exceeded the maximum allowed + size (STR #454) + - Updated the client code in the scheduler to close the + client connection on errors rather than shutting down + the receive end of the socket; this caused resource + problems on some systems (STR #434) + - cups-polld didn't compile on Tru64 5.1B (STR #436) + - "lpc stat" crashed if the device URI was empty (STR + #548) + - The scheduler did not compile without zlib (STR #433) + - std:floor() cast needed on IRIX 6.5 with SGI C++ + compiler (STR #497) + - cupsRasterReadPixels() and cupsRasterWritePixels() did + not handle EAGAIN and EINTR properly (STR #473) + - RequiresPageRegion should not be consulted for Manual + Feed (STR #514) + - International characters were not substituted in + banner files properly (STR #468) + - Updated pdftops to Xpdf 2.03 code to fix printing bugs + (STR #470) + - The Digest authentication code did not include the + (required) "uri" attribute in the Authorization + response, preventing interoperation with Apache + (STR #408) + - The web interface could lockup when displaying certain + URLs (STR #459) + - The PostScript filters now convert underscores ("_") + to spaces for custom classification names (STR #555) + + +CHANGES IN CUPS V1.1.20 + + - The pstops filter didn't properly handle collated, + duplexed copies of documents with an odd number of + pages on printers that did not do their own collated + copies (STR #389) + - Tru64 doesn't define a prototype for hstrerror() (STR + #430) + - Updated the pdftops filter to use the annotation flags + instead of the subtype to determine whether to print + an annotation (STR #425) + - The French web interface localization did not use + absolute paths for the navigation bar (STR #428) + - The CUPS test suite did not undefine the PRINTER and + LPDEST environment variables. This could lead to bogus + test results (STR #380) + - The cupsLangDefault() function now works if you don't + have the base OS localization installed (STR #418) + - The pdftops filter no longer needs to create temporary + files with tmpnam (STR #406) + - The HTTP code did not use a case-insensitive + comparison when checking for the Basic authentication + method (STR #407) + - The httpEncode() function always added a trailing "=" + character, which is not required by the Base64 + encoding specification (STR #407) + - The signal handlers did not need to call sigset(); + this caused a recursion problem on some versions of + IRIX (STR #422) + - Moved the scheduler termination code into the mainline + to be consistent with the way other signals are + handled (STR #423) + - The cupsaddsmb program didn't export the new CUPS + driver for Windows properly (STR #390) + - The ppdOpen() functions did not issue an error when a + translation string exceeded the maximum allowed by the + Adobe PPD specification (STR #399) + - The default landscape orientation was not the same as + that defined in the PPD file (STR #397) + - Updated the pstoraster patch files and CUPS driver to + work with Ghostscript 8 (STR #402) + - The hpgltops filter did not skip PJL commands (STR + #379) + + +CHANGES IN CUPS V1.1.20rc6 + + - "lp -i jobid -H restart" would often return an error + even though the job restarted successfully (STR #362) + - The scheduler did not check for invalid allow/deny + addresses such as "11.22.33.44/24". It now masks off + the extra address bits and logs a warning message in + the error_log file (STR #337) + - The cupstestppd utility now checks for missing + ImageableArea and PaperDimension attributes for each + defined PageSize (STR #365) + - The IPP code did not wait for a reply indefinitely on + HTTP connections in "blocking" mode (STR #377) + - The web interfaces did not rewrite the default printer + URI properly (STR #299 and #369) + - The LPD backend passed the C and L commands in the + wrong order (STR #378) + - The Dymo label printer driver did not set the label + length properly (STR #373) + - The scheduler did not support job IDs higher than + 99999 (STR #371) + - The Visual C++ project files did not work (STR #366) + - The scheduler's cupsLangSeek() function did not reset + the "EOF" flag, preventing compressed files from being + typed properly in some cases (STR #368) + - The cupsLangGet() cache was only used if the locale + name provided an explicit character set name (STR + #354) + - The CUPS API convenience functions did not call + cupsLangFree() when they were done with the + localization data (STR #354) + - The scheduler did not return the + job-hold-until-supported or job-hold-until-default + attributes (STR #356) + - The cupsaddsmb program did not support the new CUPS + driver for Windows (STR #357) + + +CHANGES IN CUPS V1.1.20rc5 + + - The scheduler did not initialize the browse socket + file descriptor properly when only SLP browsing was + enabled (STR #259) + - The scheduler accessed the job attributes before they + were set (STR #347, fix to STR #335) + - The cupsCancelJob() function did not return 0 when the + job could not be canceled (STR #340) + + +CHANGES IN CUPS V1.1.20rc4 + + - The scheduler did not move the incoming job attributes + in the operation group to the job group (STR #335) + - The cupsDoFileRequest() function did not check for an + early HTTP response while sending the file (STR #314) + - The web interfaces did not quote #, ?, or . in printer + names, which caused some problems with the generated + URLs (STR #320) + - CUPS couldn't be completely compiled with the -dDEBUG + option (STR #331) + + +CHANGES IN CUPS V1.1.20rc3 + + - More SLP changes (STR #259) + - Revamped the child signal handling code to completely + avoid deadlock issues on Solaris (STR #325) + - The lpadmin command displayed an incorrect error + message when the "-u" option was provided with no + arguments (STR #313) + - The web admin interface did not display an error + message if the PPD file could not be loaded (STR #308) + - The ppdEmit() functions did not use the correct + orientation value position for custom page sizes (STR + #292) + + +CHANGES IN CUPS V1.1.20rc2 + + - The serial backend set the IXANY option on the port + for XON/XOFF flow control; this caused problems with + printers that returned status info but were not ready + for more print data (STR #287) + - The scheduler didn't support scripted index files + (index.php, index.pl, etc. - STR #290) + - The scheduler did not correctly localize script files + with "GET" variables (STR #268) + - Changes in job classification are now logged (STR + #289) + - Fixed a few more SLP-related bugs (STR #259) + - Updated the user/group configure checks for MacOS X + 10.3 (STR #270) + - Fixed an offset bug in the PDF filter (STR #284) + - The cupsDoRequest() and cupsDoFileRequest() functions + did not map several HTTP status codes to their IPP + counterparts. This made detecting certain conditions + very difficult (STR #277) + - Config, spool, and status files are now owned by the + scheduler user (usually root) with read permission for + the filter group (STR #283) + - The HP-GL/2 filter did not support the SI command, + some values for the AD and SD commands, and did not + rotate labels properly via the DI command (STR #282) + - The fax support did not update/set the job-hold-until + attribute when a fax job fails (STR #269) + - The cupsLangGet() function didn't support locales of + the form "ll.charset" (STR #271) + - The scheduler did not use the charset when getting the + language localization for a request; this caused extra + disk IO for every request (STR #271) + - The scheduler did not support requests with more than + one language specified (STR #267) + + +CHANGES IN CUPS V1.1.20rc1 + + - The scheduler now waits up to 60 seconds before + restarting to allow active jobs to complete printing + and pending requests to be processed (STR #226) + - The web interface did not work on systems where time_t + is 64 bits (STR #262) + - Added backend tweeks and content-length check from Red + Hat (STR #253) + - The USB backend now uses the 8255 constants instead of + the standard constants when reporting printer status + bits on Linux (STR #254) + - Added new cupsDoAuthentication(), cupsGetFd(), + cupsGetFile(), cupsPutFd(), and cupsPutFile() functions + to the CUPS API (STR #112) + - The PDF filter always scaled and offset pages; this + caused problems under MacOS X, so now the "fitplot" + option controls whether PDF files are scaled to fit + within the printable area of the page (STR #250) + - The LPD backend did not support the port number in a + URI (STR #247) + - Some filters didn't properly support boolean options + (STR #249) + - Landscape PDF files were not always offset by the + correct amount when rotating (STR #243) + - The scheduler could hang in a call to localtime() when + logging messages from the signal handler (STR #242) + - The PDF filter no longer prints form widgets; this + duplicates the behavior of Acrobat Reader (STR #241) + - cupsGetPPD() didn't handle a late termination of a + HTTP connection with the server (STR #220) + - ppdOpen() did not correctly check for "*PPD-Adobe-4." + on the first line of a PPD file. This caused incorrect + PASS results for some PPD files (STR #233) + - cupsEncodeOptions() did not allow boolean options to + use "yes" and "on" for true values (STR #227) + - The pstops filter only sent the TBCP exit sequence if + it was defined in the JCLEnd attribute in the PPD file + (STR #224) + - Support for more than 1024 files was broken on Solaris + 9 (STR #217) + - The setgroups() calls now pass in 1 group (the + configured group) instead of 0 for compatibility with + BSD and Darwin (STR #213) + - The scheduler's built-in broadcast throttling was + ineffective since incoming packets would cause the + next group of outgoing packets to be sent immediately + rather than waiting for the next time slot (STR #211) + - Added a new ppdSetConformance() function to set the + conformance requirements for PPD files. Currently only + two levels are defined, PPD_CONFORM_RELAXED and + PPD_CONFORM_STRICT, and the default is the relaxed + level (STR #212) + - The IPP backend did not correctly execute the + pictwpstops filter on OSX (STR #210) + - The LPD backend did not set the banner class when the + "banner=yes" option was specified in the device URI + (STR #209) + - The imagetoraster filter did not support all of the + page device attributes (STR #208) + - The pdftops filter incorrectly auto-rotated pages when + the user already had specified the proper orientation + (STR #207) + - Fixed AIX shared library support (STR #201) + - Added support for live testing with Valgrind (STR + #193) + - The CGI programs now collect the list of needed + attributes for the class, job, and printer template + files (STR #192) + - The scheduler now passes the first port that is bound + to the local loopback or "any" addresses to the CGI + programs rather than the port that the browser + connected to (STR #103) + - The cupstestppd program now checks for bad + JobPatchFile attributes and incorrect versions of the + Manufacturer attribute for HP printers (STR #155) + - The filter makefile incorrectly installed + libcupsimage.a in the filter directory (STR #180) + - The scheduler did not verify that the job history + files define the job-priority and + job-originating-user-name attributes (STR #178) + - The pstops filter didn't handle poorly-formed binary + PostScript files that had CTRL-D's in them (STR #156) + - The ppdOpen*() and cupsLangGet() functions did not + make a copy of the old locale strings when using the + POSIX locale when reading files, which apparently + caused problems with some implementations of the + standard C library. (STR #159) + - The pdftops filter did not work properly with some + embedded Type1C fonts (STR #177) + - Updated the pdftops filter to be based upon Xpdf + 2.02pl1 (STR #191) + - The scheduler did not reset the group list when + running CGI and filter processes (STR #185) + - The scheduler no longer calls malloc and free from the + signal handlers (STR #190) + - The USB backend now uses the manufacturer and model + strings if the description string is not available + (STR #174) + - The ppdOpen functions still supported the + VariablePaperSize attribute, which was removed in v4.0 + of the PPD spec. This caused problems with PPD files + that relocated the PageSize option to a non-standard + group (STR #158) + - The cups.list file referenced MAN1EXT, MAN3EXT, and + MAN5EXT, but none of those were actually defined (STR + #147) + - Chunked requests could cause a Denial of Service if + the connection is terminated before the first byte of + chunk data is sent/received (STR #143) + - Printers with special characters in their names were + not accessible from the web interface (STR #120) + - The lpstat command now shows the correct interface + script or PPD file, if any, for a print queue (STR #89) + - The lpstat command now shows the printer-state-message + and printer-state-reasons attributes whenever they are + not blank (STR #152) + - The French and German option-conflict.tmpl template + files did not get installed (STR #148) + - The cups.list.in file did not work when compiling + without shared libraries (STR #149) + - The DSOFLAGS included the LDFLAGS, which causes + problems on at least HP-UX (STR #150) + - The fax printer support did not keep track of the fax + capability bit (STR #144) + - The appleLangDefault() function could leak a small + amount of memory (STR #145) + - The ppdOpen() functions now mirror all normal + attributes to the attribute list; previously only + certain unassigned attributes would be added (STR + #139) + - The ppdEmitJCL() function wrote JCL commands to stdout + instead of the passed file pointer (STR #142) + - The httpGets() function could, in certain states, + block waiting for data (STR #132) + - The cupsEmitJCL() function not outputs an empty @PJL + command after the PJL language escape to work around + bugs in certain PJL implementations (STR #131) + - The cupsEmit*() functions didn't set the orientation + value properly (STR #127) + - The cups.spec file didn't list the rc2.d init + directory or the cupstestppd file (STR #134) + + +CHANGES IN CUPS V1.1.19 + + - The GNU TLS code incorrectly used + gnutls_check_pending() instead of + gnutls_record_check_pending() (STR #128) + - The ppdEmit() functions output "PageSize Custom" + instead of "CustomPageSize True" in the DSC comments. + Also, the custom page size code did not use the + ParamCustomPageSize attributes (STR #127) + - The cupstestppd command did not list the conflicting + options (STR #123) + - The lpq command did not ensure that there was + whitespace between the fields in the job listing (STR + #117) + - The German web templates had errors (STR #119) + - The configure script didn't specify the static + libraries properly when configuring with the + --disable-shared option (STR #104) + - The cups.list file used file dependencies for package + formats other than portable, RPM, and Debian (STR #98) + - cupsLangGet() didn't use its language cache (STR #97) + - "lpq -P" would segfault instead of showing a usage + message (STR #94) + - Fixed compiler warnings in pdftops filter (STR #96) + + +CHANGES IN CUPS V1.1.19rc5 + + - Jobs with banner pages that were printed to implicit + classes would get double banner pages for each + file/banner in the job (STR #68) + - The mime.convs file was missing the filter definition + for Windows BMP (image/x-bitmap) files (STR #85) + - The scheduler allowed some READ-ONLY job attributes to + be set, which could cause the scheduler to fail on the + next restart (STR #82) + - The lp and lpr commands did not report when the + scheduler was not responding; instead, the user would + incorrectly see a "no default destination" error (STR + #70) + - cupsLangGet() could fail on OSX due to a corrupt + language preference (STR #78) + - Added more checks for HTTP request timeouts. + - The scheduler dropped the first non-alpha character + after an open brace when doing attribute substitutions + in banner pages (STR #77) + - The scheduler child might send SIGUSR1 to the parent + before the signal handler was installed; this didn't + prevent the scheduler from starting but produced an + annoying error message (STR #45) + + +CHANGES IN CUPS V1.1.19rc4 + + - The lp command did not accept "-" for printing from + the standard input as required by POSIX 1003.1 (STR + #59) + - Added the job-originating-host-name information for + the page_log file documentation in the SAM (STR #31) + - The German web interface templates did not use the + right paths for job operations (STR #54) + - The scheduler would consume all available CPU if + started with a pending job in the queue (STR #35) + - The polling daemon allocated an extra localization + buffer but did not free it, causing cups-polld to + eventually use all available memory (STR #40) + + +CHANGES IN CUPS V1.1.19rc3 + + - The scheduler could get in an infinite loop cancelling + jobs using "cancel -u user dest" (STR #48) + - The "cancel -u user" command did nothing (it should + cancel all jobs on all printers owned by the named + user - STR #48) + - The scheduler would write 0-length job control files + (STR #46) + - Updated the French man pages (translation provided by + Gilles QUERRET) + - The scheduler would delete all printers from + printers.conf if a job was active when a HUP signal + was handled (STR #47) + - The cups-polld program would leak memory if it was + unable to send browse packets to the loopback + interface (STR #40) + - The scheduler did not put the + job-originating-host-name attribute in the job + attributes group. + - The text filter did not default to wrapping text as + defined by the IPP implementation document. + - Scan backends first, PPDs second (STR #37) + - Updated the Netatalk documentation in the SAM (STR #38 + and #39) + - The test suite sent text files to a non-PS print queue, + which requires ESP Ghostscript (provided separately). + Now send the JPEG test file (STR #33) + - The test suite did not show the estimated disk space + requirements (STR #33) + - The test suite did not set the MaxLogSize directive to + 0 to prevent log file rotation (STR #33) + - The test suite still setup the old CUPS Ghostscript + symlinks (STR #33) + - The pstops filter did not report the correct number of + copies for the page_log file when printing collated + copies to a printer that doesn't support them in + hardware (STR #32) + - cupsLangGet() needs to set the CTYPE locale to "C" + (POSIX) to avoid erroneous tolower/toupper values (fix + suggested by Bjoern Jacke) + - Fixed a typo in the cups.list.in file. + - Updated all of the Western European locales to default + to ISO-8859-15 (for Euro support, suggested by Bjoern + Jacke) + - Updated the German message catalog (update provided by + Bjoern Jacke) + + +CHANGES IN CUPS V1.1.19rc2 + + - cupsLangGet() now sets the encoding field based on the + trailing charset in the locale name, and doesn't look + for a message catalog in a specific locale.charset + directory. This fixes STR #26 and is more in line + with the CUPS 1.2 implementation. + - The configure script now aborts if the "ar" command or + compilers cannot be found. + - The static cupsimage library was not built by default. + - The path for the "ln" command was hardcoded in + Makedefs.in instead of being checked at configure time + (STR #28). + - Banner pages containing unescaped { characters would + not work. + - The printer-state-time collection attribute was + encoded as an enumeration instead of an integer. + - The printer-is-accepting-jobs collection attribute was + was not added to the collection value. + - The printer-state-sequence-number collection attribute + was not added to the collection value. + - Fixed typo and const mismatch in IPP backend. + - Updated the man pages for the new configuration + directives. + - Updated the SAM for MacOS 10.2, the CUPS drivers for + windows, the available LPD backend options, and the + new configuration directives. + - The imagetops filter didn't position images properly + on the page (STR #18) + - The configure script didn't add CPPFLAGS to the + compiler options or LDFLAGS to the DSO options (STR + #13) + - The scheduler would try to write a debug log message + when starting a job that contained a NULL string. + Since not all versions of snprintf() support NULL + string pointers this caused some problems (STR #20) + - The testipp program now supports reading of IPP + message files such as those used for the job history + in /var/spool/cups. + + +CHANGES IN CUPS V1.1.19rc1 + + - Added CUPS support files for Java, Perl, and PHP + (located in the "scripting" subdirectory...) + - The scheduler now supports fast-reloads of the + cupsd.conf file when it is updated via HTTP. + - The scheduler always changed the ownership of log + files; it now only does so if they are not in the /dev + directory (i.e. don't want to change the ownership and + permissions of /dev/null...) + - Added libpaper support (patch from Jeff Licquia) + - Added a new istring() rule for MIME types files that + does a case-insensitive comparison of strings. + - The cups-lpd mini-daemon now sends jobs to the default + queue when an empty queue name (or "lp" and there is + no "lp" queue) is sent. + - The scheduler now supports fax queues identified by a + "*cupsFax: True" attribute in the PPD file. When a job + can't be sent, it is held for 5 minutes by default + while other jobs are attempted. The FaxRetryLimit and + FaxRetryInterval directives control the number of + retries and the time between retries. + - The scheduler now preserves the default options of PPD + files when modifying/upgrading an existing PPD file. + When installing a new printer, the scheduler sets the + default media size to Letter or A4 as appropriate for + your locale. + - The scheduler no longer limits the number of + BrowseAddress, BrowsePoll, BrowseRelay, Listen, Port, + SSLListen, and SSLPort directives to 10. + - The scheduler now supports print files that have been + compressed using gzip. + - The scheduler used the stdio functions to read any job + ticket information in a PostScript print job. Since + some platforms limit the number of stdio files to 256, + job ticket information was ignored when the server had + a large number of clients connected to the system. + - Filters and backends may now report the total number + of pages ("PAGE: total NNN") to the scheduler. + - The LPD backend now supports timeout and + sanitize_title options (default to 300 and yes, + respectively) and has some additional changes to + reduce the chances of multiple copies being printed + when only one copy was requested. + - Fixed a polygon drawing bug in the HP-GL/2 filter. + - Added a robots.txt file to the standard install to + prevent search engines from indexing the CUPS server. + - Added support for STATE: messages + (printer-state-reasons), printer-state-history, and + printer-state-time to the scheduler. + - When using RunAsUser, the scheduler would initially + start any previously queued (pending) jobs with + RunAsUser disabled - all backends would be running as + root. + - If a backend failed for a printer, CUPS would + incorrectly requeue the job for printing again. + - Added support for IPP collections and files. + - Added experimental support for generic CGI scripts and + programs, Java, Perl, PHP, and Python to the + scheduler. See the file "CGI.txt" for more + information. + - The CUPS API now supports HTTP cookies and the Expect: + field. + - The cancel command now correctly supports the "-u + user" option to cancel all jobs for the named user. + - The Purge-Jobs operation now supports the my-jobs + boolean attribute and a new purge-jobs boolean + attribute to control whether job history data is + purged from the scheduler; the default is false for + my-jobs and true for purge-jobs to match the original + implementation. + - The scheduler would not timeout printers when only + using SLP browsing. + - If the scheduler was unable to execute a filter, it + would try to restart the job indefinitely until the + filter could be executed. + - When writing BSD printcap files, the scheduler now + includes the rm and rp attributes, allowing the file + to be exported to LPD clients. [Patch from Dominic + Kubla] + - The scheduler optimization to reference IPP attribute + data instead of performing a full copy caused problems + when the referenced data was deleted before it was + sent. It now only references attributes that change + only when the scheduler is restarted. The change also + reduced the memory footprint of a printer object to + 2k. + - The scheduler now holds signals while logging messages + to avoid potential deadlock issues when handling + signals on Solaris 8. + - The lpadmin command now allows printer access control + by group name as well as user name. + - "lpoptions -l" got in an infinite loop if no default + printer was available. + - The scheduler now logs the job-originating-host-name + attribute in the page_log file, and uses "-" for any + empty fields (patch from Dominik Kubla). + - The pdftops filter now scales PDF pages within the + printable area of the page. + - The pstops filter didn't include the page-label and + classification boxes when printing EPS or non- + conformant PS files. + - The imagetops filter didn't always correctly position + the image on the page when printing in landscape + orientation. + - The ppdEmit() functions now support the + RequiresPageRegion attribute when sending InputSlot + and ManualFeed commands. + - The PPD loading code now supports standard options + outside of OpenUI/CloseUI as required by the PPD spec. + - The cupstestppd program has been upgraded to provide a + concise PASS/FAIL report, additional detailed + conformance testing, and support for gzip'd PPD files. + - The PPD loading code is now much more strict when + loading a PPD file, and tracks more format errors. + - The scheduler ignored child signals when gathering the + list of available devices, when it should have been + using the default signal handler. + - The cupsEncodeOptions() function could encode an + option with a NULL last string. + - The socket backend could report the wrong number of + backchannel bytes if an error occurred on the link. + - The cups-polld program now only sleeps after getting + all printers and classes. This allows for longer + intervals without excessive delays before classes show + up... + - Added a new httpWait() function to support waiting for + data for a specific number of milliseconds. + - httpGets() now times out after 1 second on + non-blocking HTTP connections. + - The scheduler no longer accepts rangeOfInteger values + that are out of order (e.g. 5-1) + - The sides attribute was incorrectly sent as a name + value; it is a keyword value. + - The IPP backend now detects if the destination queue + has gone away and reports an error. + - The scheduler and HTTP API now allocate their select() + sets to support larger numbers of clients on systems + that support it. + - The scheduler now sets the CFProcessPath environment + variable under MacOS X. + - The cupsLangDefault() function now uses the + CoreFoundation localization API under MacOS X. + - The httpSeparate() function didn't handle file URIs of + the form "file:///path" properly. + - The lpadmin command now supports a "protocol" option + for specifying the binary communications protocol to + use when printing binary PostScript data. + - The scheduler did not properly parse the SystemGroup + directive, so only the first group would be used. + - Revamped how strings are stored in the scheduler, + providing a substantial improvement in memory usage + for systems with large numbers of printers. + - The PostScript filter now supports binary PostScript + files and files beginning with the PJL language escape + sequence. + - The PPD API now provides additional information from + the PPD file. + - The USB backend didn't compile on Solaris Intel. + - The cupstestppd utility now supports the "-q" option + (quiet) for use in scripts, etc. + - Merged several weight-reducing changes into the CUPS + baseline donated by Apple. + - Added preliminary support for CDSA; patch provided by + Apple. + - Implicit classes are now created from identical + printer classes on the network. + - The lp command now supports a "-H restart" option to + restart previously printed jobs. This functionality + only works if you have enabled the PreserveJobFiles + option. + - The scheduler now supports URIs in HTTP request lines + to conform to the HTTP/1.1 specification. + - The time-at-xyz attributes were not recognized in + banner files if prefixed by a question mark, e.g. + "{?time-at-creation}". + - Added support for pre-filtering application/pictwps + files on MacOS clients before sending them to a server + via IPP. + - The scheduler now allows file:/dev/null device URIs + even if FileDevices is set to No. + - CUPS uses strerror() for hostname resolution errors, + when it should have used hstrerror(). + - The USB backend no longer tries to guess the serial + number of a device from the USB devices file; this + means that printers that don't report their serial + numbers in the device ID string will not be + individually selectable. + - The pstops filter didn't handle page ranges properly + when a page contained an embedded document. + - Added a translation of the web interface to German. + - When printing using the OutputOrder=Reverse option + with duplexing, the output order is now truly + reversed; the order of sub-pages when printing N-up is + the same. + - The pstops filter did not always output the extra + blank page when printing a document with an odd number + of pages with duplexing enabled. + - The ippAddXYZ functions no longer allow the + application to add less than 1 value. + - Fixed a URL rewrite bug in the web interface - local + access was sometimes redirected away from localhost... + - The ppdOpen() functions could get in an infinite loop + if the PPD file contained a keyword or text that was + too large for the buffer. + - Added preliminary support for GNU TLS; patch provided + by Jeff Licquia. + - Now timeout IPP attribute reads after 1 second inside + an attribute definition. + - Now timeout connections that have been shutdown (due + to errors) after 30 seconds instead of the Timeout + setting (300 seconds by default). This provides + faster recovery from DoS attacks. + - A denial-of-service attack warning message was being + written to the log files by the scheduler for every + detection. This caused a DoS of its own in some + situations. The warning message is now written no more + than once per minute. + - Fixed the CIE colorspace support code in the image and + PS RIPs. + - The job-quota-period, job-page-limit, and job-k-limit + attributes were not flagged as integers, so setting + quotas would not work. + - Added an additional response check in the scheduler to + more quickly recover from denial-of-service attacks. + - The cupstestppd file was incorrectly installed in the + /usr/sbin directory instead of /usr/bin. + - The EPM list file did not include the cupstestppd + program or man page files. + + +CHANGES IN CUPS V1.1.18 + + - Fixed a bug in the Set-Job-Attributes code in the + scheduler that would cause it to crash or continuously + write a job control file. + - SECURITY FIX: The scheduler now provides a FileDevice + directive to control whether new printers can be added + using device URIs of the form "file:/filename". The + default is to not allow printers with these device + URIs. + - The scheduler did not compute the cost of filters + properly, nor did it choose a multi-filter solution + with a lower cost than a single filter solution. + - Now install CUPS PPD file test utility (cupstestppd) + to support basic conformance testing of PPD files. + - The scheduler now logs an error message when it sees a + non-conforming PPD file. + - Upgraded pdftops filter to Xpdf 2.01 with fixes for + TrueType fonts. + - Added a MaxClientsPerHost configuration directive to + provide limited protection against Denial of Service + attacks. + - SECURITY FIX: Potential underflow/overflow bug in web + interface. + - SECURITY FIX: Race condition in certificate creation. + - SECURITY FIX: Bad URIs in browse packets could be used + to exploint the web interface underflow/overflow bug. + - SECURITY FIX: Some types of Denial of Service attacks + were not handled properly, so once the attack was over + the scheduler did not close the connections + immediately on all platforms. + - SECURITY FIXES: Added integer overflow/underflow + checks for all image formats. + - The pstops filter didn't reset the showpage operator + back to its original at the end of a job; this + prevented the concatenation of documents (used + primarily for CUPS 1.2...) + - The cupsGetPPD() function didn't always set the + cupsLastError() value when an error occurred. + - The IPP media, output-bin, and sides attributes took + precedence over the corresponding PPD options, which + caused inconsistent behavior under MacOS X with some + PPD files. + - The cupsaddsmb utility specified the wrong number of + arguments to the adddriver command when adding the + Win9x PostScript drivers. + - The web interface did not always report the correct + error message. + - The scheduler did not clear the POSIX signal action + structure when waiting for the child to send it a + SIGUSR1 signal; this could cause the signal handler + not to be called properly, preventing the parent + process from returning. + + +CHANGES IN CUPS V1.1.17 + + - The "manual_copies" option did not work when the LPD + backend had to retry a print job. + - The image filters did not convert GIF images properly. + - The RunAsUser option was incompatible with the new + daemon-mode code in 1.1.16. + - Fixed a problem with the Set-Job-Attributes and + PostScript job ticket code in the scheduler - the + "last" attribute pointer was never updated, which + could cause the scheduler to crash when applying job + ticket data. + - Fixed a problem in the scheduler that caused it to + continue processing HTTP requests on a connection + after it was shutdown. + - The scheduler now allows accounts authenticated via + PAM to not have a corresponding UNIX account, but + group membership still requires the account name to be + listed in the UNIX group file(s)... + - The scheduler used a fixed-size (16k) buffer for + encoding job options for filters; it now dynamically + allocates and expands the buffer as needed depending + on the options that are sent in a job. + - The pdftops filter didn't support all of the MacOS + characters for MacRoman encoded fonts. + - The cupsEncodeOptions() and cupsParseOptions() + functions now conform to the grammer defined by the + current draft of the PAPI specification. The main + difference is that option=yes and option=no are no + longer treated as boolean options. + - The IPP backend didn't honor the encryption settings + in /etc/cups/client.conf. + - Fixed a potential bug in the HTTP code which was + caused by servers sending the status line and + newline(s) in separate packets. + - User-defined classification strings are now printed + verbatim - previously the classification box would be + empty. + - Re-added Spanish to the list of PPD languages that + CUPS supports. + - CUPS API library user and temp file updates for + Windows. + - The image filters did not properly handle grayscale + printing of Sun Raster images. + - The scheduler never reset the NumJobs variable before + loading the job list (previously this only happened on + a full start, so the problem was never apparent...) + - The HTTP and IPP read/write code didn't handle EINTR + (interrupted system call) errors. + - When under high load, the scheduler could abort due to + the wrong errno value after a select() call. This was + caused by the child signal handler. + - Added new load tests to the test target to verify that + cupsd can handle hundreds of simultaneous jobs without + error. + - The Solaris USB backend now supports the new device + URI syntax. + - The ppdOpen*() functions now reset the numeric locale + settings while loading a PPD file. + - Fixed the libtool build rules. + - The manpage make rules didn't use $(MAKE) and + $(MFLAGS) for the language subdirectories. + - Now set the LC_TIME locale category to get the + properly localized time string. + - Fixed a problem in the scheduler that would cause the + web interface problems when adding, modifying, or + configuring a printer or class. + - The backends now ignore SIGPIPE so that failed job + filters will not stop a print queue. + - The lpstat command did not allow for destination lists + ("lpstat -v printer1,printer2") + - Fixed parsing of long filter status messages in the + scheduler. + - Added some startup performance enhancements to the + scheduler so that the printer object information is + regenerated fewer times and the MIME type database is + not filled with lots of empty filters for raw/direct + queues. + - The LPD backend now sends the job title as the print + filename. + - Added support for variable sizes in the EPSON dot + matrix printer drivers. This allows for pages as + short as 1/2" (1 row of labels) and does not do an + automatic form feed. + - French translation updates. + - The filters did not quote the page label string when + embedding it in PostScript output. + - The serial backend now enumerates serial ports under + MacOS X. + - The pdftops filter contained font rasterizer code that + wasn't being used and that depended on X11. This code + has been removed. + + +CHANGES IN CUPS V1.1.16 + + - The cancel and lprm commands now both display an error + message and return a non-zero exit status if an + attempt is made to cancel a job on a non-existent + printer. + - The lpoptions command incorrectly complained if a + request to delete a non-existent printer was made. + - If the client.conf file defines an alternate server + name, the "configure printer" action in the web + interface might not work. + - The lpstat command now supports a "-W" option so that + you can display completed jobs as well as + not-completed (pending) jobs. + - The lp and lpr commands did not return an error when + one or more files in a set of files for printing could + not be printed. + - The lp, lpadmin, and lpstat commands now consistently + return with a non-zero exit status when an error + occurs. + - The scheduler would not accept print jobs sent to a + stopped remote printer. + - The texttops filter incorrectly converted the page + numbers in the prettyprint header to double-byte + characters when printing a non-Unicode text file. This + caused an extra space to appear between each digit in + the page number. + - The scheduler did not use a case-insensitive + comparison when adding filters for a printer. + - Upgraded the pdftops filter to Xpdf 1.01. + - The scheduler no longer passes the page-border and + number-up-layout attributes to filters when printing + banner pages. + - The LPD backend now uses a 30-second timeout when + sending commands and control files, and a 30-second + timeout when retrieving responses from an LPD server. + If a timeout occurs, it retries indefinitely. This + helps to make LPD printing over VPNs work more + reliably. + - The USB backend now supports device URIs based on the + printer serial number and/or model number under Linux. + This avoids the "wrong device filename" problem when + using more than one USB printer. + - Now just shutdown the receiving end of a client + connection when sending an error that requires the + server to disconnect from the client afterwards. This + fixes a problem when doing remote administration with + encryption enabled. + - The scheduler did not send a printer-state-message + attribute if the string was empty; it now always sends + this attribute. This caused the printer message to be + displayed for other printers in the web interface. + - The LPD backend now supports a "manual_copies" option, + e.g.: "lpd://server/queue?manual_copies=no", in order + to handle copies for raw jobs to printers that don't + implement the LPD protocol properly... + - The "mirror" option was not being handled by the + PostScript or image filters. + - Updated the cupsaddsmb command to support the new CUPS + driver for Windows NT/2k/XP. + - Filter status lines longer than 1023 characters could + cause the scheduler to get into an infinite loop. + - The scheduler didn't reset the job state to pending + when modifying an active printer. + - Now limit the maximum number of recursion steps when + searching for a filter for a job, in case a user + defines a circular filter rule. + - The PostScript filter would embed an invalid + requirements comment in some cases. + - Added support for embedded job tickets in PostScript + files. + - The PostScript filter now detects EPS files and should + better handle printing EPS files. + - The cancel command now ignores a trailing destination + name when cancelling a specific job ID (Solaris + compatibility). + - The scheduler now rejects jobs with copies outside the + range of 1 to MaxCopies, inclusive. + - Added new MaxCopies directive to set the maximum + number of copies that a user can request. + - The scheduler didn't block signals while it processed + others and when it forked processes. + - The scheduler checked for new jobs to print when + stopping a job. This caused jobs to restart before a + shutdown. + - Updated the CUPS startup script to better support + different timezones and to support the RedHat/Mandrake + init script functions, if available. + - The scheduler did not properly handle backslashes in + banner files; it incorrectly assumed that "\c" should + always be replaced by "c", instead of only looking for + "\{" and replacing it by "{". + - The texttops filter didn't handle prettyprint=no. + - The text and HP-GL/2 filters didn't check for other + common duplex option names like cupsMarkOptions() did. + - "lpoptions -x printer" no longer clears the "default + printer" status of the printer. + - cupsTempFd() now stops trying to create a temporary + file after 1000 tries, and aborts on any error other + than EEXIST. This should prevent lp/lpr hangs due to + a bad or missing temporary directory. + - The lpadmin command did not send the right URI to the + scheduler when setting options on classes. This + caused a client-error-bad-request error. + - The CUPS API convenience functions would attempt to + connect to the remote server name in a + "printer@server" printer name instead of dealing with + the default (usually local) server. Aside from + causing user confusion, the remote server name might + not be resolved properly, causing further problems. + - "lp -q" would cause the "lp" command to segfault, as + the program would try to print the option letter that + caused the error using the wrong index into the + command-line; bugfix from Debian. + - Fixed a minor inconsistancy in the encoding of boolean + attributes from printer options in + cupsEncodeOptions(). + - Added a FilterNice directive which sets the priority + of job filter processes that are run by the scheduler. + - Added Solaris x86 USB printer support. + - The USB backend now reports both the ulpt and unlpt + devices under *BSD. + - The "lpstat -o" command would truncate the + "printer-jobid" string if it was longer than 21 + characters. + - The PJL-based MIME type rules now look in the first + 1024 bytes instead of just the first 512 bytes to find + the language mode. + - The image file types are now listed explicitly in the + mime.convs file so that additional image file formats + do not use the standard CUPS image filters by default. + - Updated the Software Programmers Manual to include + all of the CUPS API functions. + - ppdOpen*() no longer sorts choices for an option. + - The web interface now enforces constraints in PPD + files when configuring a printer. + - When stopping a printer, the scheduler didn't set the + printer state before stopping the current job. + - The cupsaddsmb utility now lists all data files for + Win9x and WinMe clients when installing that Windows + driver. + - Jobs submitted to a class now bounce immediately to + the next available printer rather than waiting until + that printer is available. + - Filters and backends now also get the CLASS + environment variable set when a job is printed to a + printer class instead of a normal printer. + - Added French translations of the web interface, CUPS + Overview, Software Administrators Manual, and Software + Users Manual contributed by Marian REYT-LLABRES. + - Added several "hint" messages for common configuration + problems that are stored in the error_log file. + - httpSeparate() now unquotes %xx characters in the + username:password field of a URI. + - When starting the scheduler in daemon mode, the parent + process now waits for the child to signal it is ready + to accept connections. + - Added -F option to cupsd to run cupsd in the + foreground but detach from the controlling terminal + and current directory. + - The scheduler did not reload jobs when receiving a HUP + signal; this would cause problems since the pointers + into the file type database would no longer be valid + for existing jobs. + - The scheduler did not save the network interface list + update time, thus no caching of the network data was + actually provided. + - Updated the SuSE PAM configuration file. + - The LPD backend now supports a "reserve" option and no + longer reserves a priviledged port by default. + - The cupsaddsmb command now continues past printers + that do not have a PPD file to export. + - The lpstat command didn't treat printer names as + case-insensitive. + - The lpstat command now reports the printer location + attribute with "lpstat -l -p". + - Fixed a bug in the vsnprintf() emulation function, + which was used on old versions of HP-UX, IRIX, and + Solaris. + - The number-up option was incorrectly being used when + printing banner pages. + - Added support for Greek and Slovak PPD files. + - CUPS now supports printer names containing any + printable character, e.g. "123-abc", "foo-bar", etc. + - The null filter was not supported in mime.convs due to + a bug in the filter validation code. + - Changes in the default printer and printer attributes + were not always reflected in the generated printcap + file. + - Implicit classes did not inherit the location or + description from member printers. + - The httpGetHostByName() function did not handle + hostnames that started with a number. + - Updated the filters to use the %cupsRotation comment + instead of %%Orientation to auto-rotate pages, since + the use of %%Orientation is inconsistent. + - Added the RootCertDuration directive to control how + often the root authentication certificate is updated. + - Increased the size of the IPP write buffer to 32k to + allow for larger attribute values and to provide more + efficient output of large numbers of attributes. + - The polling daemon now retries the initial connection + to the remote server; this fixes a problem when the + remote server is unavailable when the scheduler starts + up... + - The scheduler didn't validate Digest users against the + system group(s), so Digest and BasicDigest + authentication didn't work for administration + operations. + - The scheduler now passes the SHLIB_PATH environment + variable to child processes (HP-UX shared libraries) + - The scheduler now maps accesses from the loopback + interface to "localhost". + - The cups-lpd mini-daemon sent a status code byte in + response to queue state commands, but those commands + only return textual data. + + +CHANGES IN CUPS V1.1.15-1 + + - The lpc and lprm sources didn't include the CUPS + string function header, which is required on systems + that don't have their own snprintf() function. + - The French manpage Makefile tried to install the + language subdirectories when it (obviously) didn't + have to. + + +CHANGES IN CUPS V1.1.15 + + - Updated the CUPS license agreement for the new MacOS + license exception. + - The printer-info attribute now defaults to the printer + name if no value has been set. + - ppdOpen() and friends now add an "Auto" InputSlot + option if none is provided to automatically select the + correct tray. + - Updated the ppdEmit() and ppdEmitFd() functions to + (re)mark the correct PageSize or PageRegion option + depending on the selected ManualFeed or InputSlot + options. + - ppdEmitFd() didn't handle custom page sizes. + - Darwin uses instead of + . + - The jobs.cgi web interface now handles all job + operations, allowing the administrator to allow "job + administrators" or operators to manage jobs (but not + queues) on the server. + - The cupsDoFileRequest() function now checks if the + filename passed into the function is a directory, and + returns the IPP_NOT_POSSIBLE error if so. + - New SCSI printer backend. + - Cleaned up handling of locales with trailing character + set definitions. + - Fixed handling of invalid PPD attributes inside + OpenUI/CloseUI. + - Fixed a problem with SSL and the job, printer, and + admin CGIs on ports other than 443. + - The scheduler didn't handle AuthClass properly. + - Added French translation of man pages. + - Updated the text filter to support the const_cast, + dynamic_cast, and static_cast keywords in ISO C++. + - Now use strlcat() and strlcpy() (or emulation + functions) for easier string/buffer protection. + - The auto-generated printcap/printers.conf files now + have a small comment header explaining where the file + comes from... + - The PostScript filter now supports 6, 9, and 16-up + output, as well as new page-border and + number-up-layout options. + - The lpoptions command didn't set options properly when + using the default printer. + - Added ConfigFilePerm and LogFilePerm directives. + - Increased maximum size of MIME types to IPP_MAX_NAME + to allow for longer printer names. + - No longer create remote printers when loading job + history data. + - The printer-make-and-model attribute wasn't set when + the PPD file didn't contain a NickName attribute. + - Now handle PPD files with translation strings longer + than 80 bytes - they are truncated if they go over... + - The scheduler didn't handle signals until after it + loaded the configuration files the first time; this + caused problems on some installations that would + restart the scheduler as the system booted into run + level 3. + - Now throttle broadcasts like we do for polling. + - Fixed a bug in the reading of PPD files using CR's + instead of CR LF's or LF's. + - The scheduler would crash if cupsd.conf contained a + BrowseProtocols line with no protocols listed. + - The HTML job operation templates now link back to the + destination printer or class. + - The serial backend now detects USB serial devices. + - The LPD mini-daemon (cups-lpd) now passes the + job-originating-host-name attribute to the scheduler + (cupsd). + - Updated the IPP backend to reconnect after downgrading + from IPP/1.1 to 1.0, and when sending requests to HP + JetDirect interfaces that don't support HTTP + Keep-Alive like they should. + - Now pass NLSPATH and DYLD_LIBRARY_PATH environment + variables, if defined, to CGI and job processes. + - Removed the pstoraster filter (based on GNU + Ghostscript 5.50) and now provide the raster "driver" + and patch file necessary to use the current GNU + Ghostscript 7.05 release. + - Removed unnecessary fonts and updated the Courier and + Symbol fonts to the latest versions to better support + non-ISOLatin1 text. + - The text filter now always embeds the Courier and + Symbol fonts to ensure that they contain the full set + of glyphs. + - The lp and lpr commands now only override the SIGINT + handler if it is not being ignored (patch from Robert + Ambrose for some interactive software that catches + SIGINT and will gracefully cancel the print...) + - The PostScript image filter (imagetops) now supports + printing CMYK images using the CMYK colorspace. + - The image filters now support CMYK JPEG files, and + correctly handles the inverted files from Photoshop + (which seems to save RGBW data, not CMYK...) + - Added a "check" target to the top-level makefile to + conform with GNU standards (same as "test"). + - The IPP code didn't always map the POSIX locale "C" to + the proper IPP language code. + - The cupsaddsmb program was updated to use the + setdriver command instead of addprinter. + - Banner pages were not handled properly for implicit + classes. + - When tunneling to a remote system using SSH, the + printer URIs for local printers on the remote system + did not reflect the correct port number. + - The Allow, Deny, BrowseAllow, BrowseDeny, and + BrowseAddress directives now support the network + interface names "@LOCAL" and "@IF(name)" for access + control and browsing based on the current interface + addresses instead of fixed names or IP addresses. + - The texttops filter did not properly recognize the + "nowrap" (wrap=false) option. + - The InstallableOptions group name in a PPD file is now + translated separately (CUPS_MSG_OPTIONS_INSTALLED) so + that UIs can accurately detect the presence of this + group. + - The scheduler no longer keeps job history data for + remote printers on the client (just on the server.) + - The parallel and USB backends now retry if the backend + detects that the printer is not connected to the + system (rather than stopping the queue...) + - The network backends now retry if the backend detects + that the printer is not connected to the network or is + unreachable (rather than stopping the queue...) + - The cupsGetDests() function no longer lists options + and instances for printers that no longer exist. + - The scheduler now converts the document language to + the correct LANG string. + - The cupsaddsmb program now supports alternative CUPS + and SAMBA server names. + - The PostScript filter now supports the Orientation + comment and rotates the page as needed automatically. + - Revamped the makefiles slightly to use automatically + generated dependencies. + - Build fixes for OS X. + - The TIFF reading code depended on the newest version + of libtiff; now conditionally compile that portion of + the loader. + - The PPD code now decodes all JCL options in the + JCLSetup group, not just those options that start with + the prefix "JCL". + - The backends now read print data using the read() + system call to ensure that the current page is printed + while the next page is being processed. + - The pdftops filter did not support shading type 3 + (radial fill) for the "sh" operator. + - The cups-polld program now throttles the local + broadcasts of polled printers and classes so that the + local system is not overwhelmed with hundreds of + printers and classes all at once. + - Updated the serial backend to support 230,400 baud for + the Linux PPC port. + - The cupsGetJobs() function wouldn't report completed + jobs that did not have a document-format attribute + value. + - The cupsEncodeOptions() function now maintains a table + of known boolean and numeric options, and encodes all + other options as strings. + - Now add a newline before the end-of-page code in the + PostScript filter; this fixes a problem with files + that don't end with a newline. + - The image filters looked for the "orientation" option + instead of the correctly named "orientation-requested" + option. + - The cupsEncodeOptions() function now handles mixed + integers and ranges. + - New translation guide for developers to provide native + language support for CUPS. + + +CHANGES IN CUPS V1.1.14 + + - The ippRead() function did not verify that the + attribute name length or string with language value + was not larger than the read buffer. + - The scheduler set the signal handlers before loading + the configuration files the first time; this prevented + the RunAsUser directive from blocking server reloads. + - Added Swedish message catalog. + - The parallel backend now recognizes the /dev/printers + device directory under Linux 2.4.x. + - MacOS X fixes. + - The cupsaddsmb utility sent the server name after the + user information when executing the rpcclient program. + This caused problems with some versions of SAMBA + 2.2.x. + - The IPP backend did not pass the requesting user name + when checking on the print job status. This prevented + it from waiting for the job to complete when + communicating with some IPP implementations that + require it. + + +CHANGES IN CUPS V1.1.13 + + - The lpstat command did not report jobs submitted to + regular printer classes. + - The texttops filter didn't use sufficient precision + when positioning text with some values of cpi and lpi. + This could cause the alignment of text to stray. + - cupsGetDests() didn't merge the options from the + /etc/cups/lpoptions file with ~/.lpoptions - options + in ~/.lpoptions overrode them completely. + - Added support for KOI8-R and KOI8-U character sets, + and added several Russian message catalogs. + - The scheduler put the wrong timezone offset in the log + files (e.g. +0500 instead of -0500 for EST...) + - The scheduler did not ignore trailing whitespace in + *.convs files. + - The scheduler now forces all processes to exit (kill + -9) when it is stopped. This prevents parallel and + USB devices from running in the background after cupsd + goes away. + - The cupsParseOptions() function didn't skip trailing + whitespace after quoted values. + - More changes to support CUPS on OS/2. + - Added Simplified Chinese message catalog. + - Added PAM support for IRIX. + - The cupsGetPPD() function didn't remove the @server + portion of the printer name, and since it would + connect immediately to the remote server instead of + the local server, the printer would not be found. + - Classification and page labels were not rotated to + match the page orientation. + - Now set the TCP "no delay" option on network + connections to improve performance/response time. + - Improved the IRIX printing tools support with patches + from Andrea Suatoni. + - Added a new PrintcapGUI directive to specify the GUI + option panel program to use for the IRIX printing + tools support. + - The cupsGetDests() function did not check to see if a + user-defined default printer (set via lpoptions) still + existed. + - The pstops filter no longer assumes that the default + dictionary is writable when doing N-up processing. + - The pstops filter now supports printing N-up with the + page-set option. + - The imagetoraster filter now supports direct printing + of CMYK image data without conversion/correction. + - The IPP backend now reports printer state/error + conditions when possible (toner low, media empty, + etc.) + - The lpstat command now supports the (undocumented) + IRIX -l option ("-lprintername") for a compact job + listing for a printer. + - The lpstat command now includes printer date/time + information in the output (always Jan 01 00:00) to + make third-party tools happy. + - The text filter now supports non-integer cpi and lpi + values. + - The Margins field in the CUPS raster header was not + initialized by the pstoraster filter. + - Added --with-optim="flags" option to configure script. + - Updated the Italian message translations. + - Updated the cups.list file to install the correct + files. + - The pstoraster filter accessed the third element of a + 2 element array. + - The scheduler did not setup a status pipe for polling + processes, so error messages went to whatever file + descriptor 2 was pointing to when they were started. + - The httpMD5Final() function didn't put a colon between + the password and nonce strings. + - The pstops filter did not default to Binary data for + "%%BeginData:". + - The pstops filter did not stop processing when a line + containing a CTRL-D is seen. + - The scheduler no longer replaces the JobSheets values + from the printers.conf and classes.conf files with the + classification level, if set. This way the original + banner settings are preserved when classification + levels are changed or turned off. + - The serial backend didn't drain the output queue, nor + did it restore the original settings. + - Updated the default system group under MacOS X. + - If no SystemGroup was defined in cupsd.conf, the + system default group was not used. + - The cups-lpd mini-daemon now supports LPD clients that + send multiple control files. + - httpConnectEncrypt() now always uses encryption for + connections on port 443, since port 443 is reserved + for the "https" scheme. + - Group authentication via certificates did not work + from the web interface for accounts other than + "root". + - The serial port backend did not clear the OPOST + option, which could cause problems with some printers. + - The cups-lpd mini-daemon didn't lookup the client IP + address properly. + - The parallel backend now identifies the polled and + interrupt-driven devices under *BSD. + - The scheduler allowed the "always" encryption mode + inside a Location, which is not valid. + - The CUPS startup script now checks for the timezone + information under Linux. + - Now also map the sides attribute to the JCLDuplex + option (if present) in PPD files. + - Updated pdftops to Xpdf 0.93a. + - Added support for MD5 passwords under Slackware. + - Added new AuthType BasicDigest that does Basic + authentication using the MD5 password file managed by + the lppasswd command. + - The banner page attribute substitution code now + retains {name} sequences in banner files when the + named attribute is undefined. Use {?name} to + conditionally substitute an IPP attribute. + - The scheduler now ensures that the ServerRoot + directory and configuration files are owned by and + writable by the User and Group in cupsd.conf. + - The USB backend now lists all USB printer devices + regardless of whether a printer is connected or not. + This allows new USB printers to be connected without + restarting cupsd. + - Added some more minor performance tweeks to the IPP + protocol code to reduce copying and array indexing. + - The cupsaddsmb utility now uses the -c option with + smbclient and rpcclient to avoid the read length limit + for commands on the standard input. + - Added an include file to the CRD handling code in + pstoraster so that it would compile properly on 64-bit + pointer platforms... + + +CHANGES IN CUPS V1.1.12 + + - Added "Polish" to the list of known languages for PPD + files. + - Added missing directory definition to cups-config. + - The CUPS-Move-Job operation did not set the + destination type for the new destination. + - The CUPS-Add-Printer operation did not support the + allow=all or deny=none values to clear the per-user + printer ACLs. + - The SetPrinterAttrs() function did not handle invalid + PPD files that were missing the required NickName + attribute. It now looks for NickName, ModelName, and + then substitutes the string "Bad PPD File" for the + printer-make-and-model attribute. + + +CHANGES IN CUPS V1.1.11 + + - Added support for embedded TrueType fonts in PDF + files. + - Added support for PostScript functions in PDF + files. + - Added new "cupsaddsmb" utility for exporting + CUPS printer drivers to SAMBA/Windows clients. + - Added preliminary support for Darwin/MacOS X. + - The CUPS-Add-Printer operation no longer allows + arbitrary scheme names in device URIs to be used - it + now restricts the available schemes to those found in + the device list (lpinfo -m). + - The ippRead() and ipp_read_file() functions could not + handle more than IPP_MAX_VALUES (100) values in a + 1setOf attribute. These functions have been updated + to dynamically allocate more memory as needed, and the + IPP_MAX_VALUES constant now represents the allocation + increment. [this caused some versions of the + GIMP-print drivers to fail since the number of media + options exceeded 100...] + - The scheduler could crash when BrowseShortNames + was set to "No". + - The scheduler did not prevent MaxClients from being + set to 0, which could cause the scheduler to go in an + infinite loop when accepting a request. + - Made some performance optimizations in the ippRead() + functions to make IPP request/response processing + faster. + - The accept/reject/enable/disable command did not + support properly support the "-h" or default + server name. + - The scheduler did not save the quota configuration + when the job-quota-period attribute was set to 0. + - The LPDEST and PRINTER environment variables did not + support printer instances. + - The text filter now handles more types of boldface and + underline formatting. + - The cupsTempFd() function did not fail if the + temporary directory did not exist; this would cause it + to loop indefinitely instead of returning an error + (-1). + - Stopping (disabling) a printer class did not stop jobs + from printing to printers in that class. + - The cupsGetDests() function was sending the + requested-attributes attribute as a name instead of a + keyword; this caused a serious performance problem on + slower systems since more information had to be + transferred from server to client. + - The web interfaces did not always quote < and & in + things like the job title. This had the potential for + browser-based security violations (on the browser's + machine); bug report from SuSE. + - The scheduler now treats unauthenticated usernames as + case-insensitive when doing quota and allow/deny + processing. + - The lp command sent the "request ID is ..." message + to stderr instead of stdout... + - The PostScript filter (pstops) now handles EPS files, + adding a showpage command to the files as needed. + - The configure script checked for the header + file before the JPEG libraries; since the JPEG headers + can define HAVE_STDLIB_H, the configure check would + cause the JPEG check to fail on some systems. + - The scheduler now supports localized banner files, + using the subdirectory approach, e.g. the "es" + subdirectory under /usr/share/cups/banners is used for + the Spanish banner files. + - Updated the scheduler so it knows the correct + language abbreviation to use for all supported + PPD LanguageVersion values. The new code also + supports country codes as well, so "English-GB" + maps to the "en_GB" locale. + - The cups-lpd mini-daemon did not support + anonymous printing (no username specified). + While the username is REQUIRED by RFC-1179, + MacOS clients do not send the REQUIRED username + information when printing via LPD. + - Added many warning and informational messages + to cups-lpd where they were missing. + - Added Czech message file contributed by SuSE. + - The cups-lpd mini-daemon now returns a non-zero + status if an invalid destination or job ID is + provided. + - The scheduler did not honor the KeepAlive setting in + cupsd.conf. + - Increased the size of the file read/write buffers to + 32k. + - *BSD static library creation fixes. + - Use mkstemps() instead of tmpnam() in pdftops whenever + possible. + - Added httpGetHostByName() function as a wrapper around + gethostbyname() - some implementations of this + function do not support IP addresses (e.g. MacOS X.) + - Added casts to all printf's of file lengths, since + there is currently no standard way of formatting long + long values. + - The client filename field was not cleared in all + instances, resulting in old form data being submitted + to CGIs. + - The httpConnect*() functions now try all available + addresses for a host when connecting for the first + time. + - The pstoraster filter would "lose" all drawing + commands when the PageSize was set but the printer + bitmap was not reallocated. This was most noticeable + with the output from StarOffice 6 beta and would + result in a blank page being output... + - The IPP backend was sending a PAGE comment even when + printing the output from a filter (it should only send + page comments when printing files directly...) + - The pdftops filter didn't properly map glyph names of + embedded Asian TrueType fonts. + - Changed the CUPS startup script to look for a program + named "cupsd", not just any program with "cupsd" in + the name (this caused the apcupsd UPS monitoring + daemon to be stopped/restarted...) + - The CUPS-Move-Job operation did not change the + internal destination name for held jobs, so moved (but + held) jobs would still show up as queued on the + original destination. + - The cups-polld program didn't send the + requested-attributes attribute in the + CUPS-Get-Printers and CUPS-Get-Classes requests, which + made it use more CPU and bandwidth than required. + - The scheduler and CUPS API incorrectly added a + job-sheets-default attribute for remote printers. This + caused banner pages to be omitted from client system + prints. + + +CHANGES IN CUPS V1.1.10-1 + + - Minor fixes to the filter, systemv, and template + makefiles to install files properly. + + +CHANGES IN CUPS V1.1.10 + + - Added a driver for DYMO label printers. + - Added new ClassifyOverride directive to allow users + to override the classification of individual jobs. + - Added new BrowseProtocols directive to control which + browse protocols are used (currently CUPS and SLP). + - Added SLPv2 support (thanks to Matt Peterson for + contributing the initial implementation for CUPS.) + - Adding a raw printer on a remote CUPS server now + correctly redirects PPD file requests to the remote + server. + - The serial backend now limits writes to 1/10th + second worth of data to avoid buffer overflows + with some types of flow control. + - The scheduler did not properly process PUT requests, + so configuration files could not be uploaded to the + server. + - The scheduler did not strip trailing whitespace on + lines in the configuration files. + - The httpWrite() function did not transition the PUT + request to the HTTP_STATUS state to get the status + from the server. + - The scheduler did not properly handle trailing null + ("-") filters when testing a driver that sent data + to the file: pseudo-backend. + - The IPP backend now only sends a document-format of + "application/vnd.cups-raw" when printing to another + CUPS server using a local printer driver or interface + script. Previously the job's document format was + used, which was incorrect. + - The lpadmin command didn't use the ppd-name attribute + with the -m option; this prevented the use of the + "raw" model from the command-line. + - The pstoraster filter output draft (1-bit) 6-color + output in the wrong order; this resulted in yellow + being printed instead of black on Stylus Photo + printers. + - The pdftops filter did not have the Japanese and + Chinese text support compiled into it. + - The IPP and AppSocket backends did not clear the + "waiting for print job to complete" status message, + which caused some confusion... :) + - The serial backend now opens the port in "no delay" + mode to avoid DCD detection problems with some OS's. + + +CHANGES IN CUPS V1.1.9-1 + + - The configure script did not substitute the + correct user and group names. + - The configure script did not use the full path + to the install-sh script when it was used. + - The pstoraster filter did not correctly support + DuplexTumble mode for printers that used flip + duplexing. + - The cups.list.in file was missing from the + distribution. + - The New DeskJet series driver did not use the + correct OrderDependency for the Duplex option. + - Use read() instead of fread() to read piped + print files in lpr/lp. This avoids a bug in the + HP-UX 10.20 fread() function. + - Updated the pstoraster filter to use the MIPS_FIXADE + system call under IRIX to fix bus error problems on + R12000 processors (Ghostscript is not 64-bit clean...) + - Some Xerox PPD files (most notably the Phaser 790) + have illegal whitespace in the option keyword in the + OpenUI line. This caused the PageRegion option to not + be recognized properly for the Phaser 790. + + +CHANGES IN CUPS V1.1.9 + + - Revamped the configure script to use a modular + approach for the various tests. + - Added --with-openssl-* options to properly reference + the OpenSSL libraries in DSOs. + - Added --with-cups-user and --with-cups-group + options to specify the default user and group for + CUPS. + - Added AIX shared library support. + - Added AIX device discovery for the serial and + parallel ports. + - Now use install program or script to install + directories, files, and symlinks. + - Updated pstops filter to use strict handling of EPS + files embedded in a PostScript document. The %%EOF + handling in 1.1.8 caused some dvips files not to + print. + - Fixed yet another memory allocation bug in pstoraster + that would cause it to crash. This fix also ensures + that all memory allocations are done on (at least) a + 64-bit boundary. + - Fixed Digest authentication - httpGetSubField() didn't + skip the Digest keyword. + - The scheduler did not properly handle Digest + authentication with the new multiple-group support. + - The scheduler did not allow usernames that were + not in the UNIX password file to be used for Digest + authentication from passwd.md5. + - The scheduler could not scan PPD files that only used + a carriage return (i.e. MacOS PPD files); the new code + is also about 40% faster, so servers with thousands of + PPD files should start much faster now. + - The scheduler now stores the PPD file size and + modification times in the ppds.dat file, so it can now + incrementally update the PPD database from the model + directory, resulting in significantly faster startup + times. + - The lpinfo command did not return a non-zero status + code if an error occurred. + - Fixed a bug in the scheduler's UpdateJob() function. + Basically, all jobs shared the same status buffer, and + the "buffer start" pointer could point to 1 byte + before the beginning of the buffer. The new + implementation uses a separate buffer for each job and + eliminates the buffer start bug. + - The IPP backend would send N copies of a document if + the receiving device didn't support the copies + attribute, even if the upstream driver already added + the necessary commands to generate the copies. This + was most noticeable with HP printers where N * N + copies would come out instead of N. + - The PostScript filter (pstops) did not properly handle + duplex printing on inkjet printers that provide this + option. Copies would be put on the front and back + sides of the duplexed page, and the filter did not + output an even number of pages. + - The backends always caught SIGTERM after they + connected to the printer. This prevented raw jobs + from being cancelled early. + - The cupsSetDests() function now removes any printers, + instances, and options that are not defined by the + user or server. This should prevent old system-wide + options from being used in individual user accounts. + - Updated the EPSON printer driver and added PPDs for + the newer EPSON Stylus printers that only support the + "ESC i" graphics command. + - The lpadmin command didn't allow you to add remote + printers to a local class. + - The lpadmin command didn't allow you to set the + options (quotas, etc.) for a class. + - The scheduler did not load or save the + job-sheets-default attribute for classes. + - The scheduler did not automatically recreate remote + printers that were part of a class. + - It was possible for a printer class to list the same + printer more than once. + - The scheduler now makes a backup copy of classes.conf + and printers.conf before writing the new file. + - The lppasswd program incorrectly asked for a new + password when deleting an existing MD5 password + account. + - The scheduler did not match "/printers/name.ppd" + against a location of "/printers/name". + - The client code did not always handle HTTP encryption + upgrades properly. + - The client code now caches the last Digest password so + it can retry using a new resource path or nonce value, + which are included in the MD5 sum sent to the server. + This should eliminate unnecessary password prompts + when using Digest authentication. + - The lppasswd command didn't have a man page. + - Updated the PJL detection rules to allow the universal + escape to occur anywhere in the first 128 bytes of the + file. + - The cups-polld program would poll servers continuously + with no delay if there was an error contacting the + server. + - The IPP backend would send an empty job-name or + requesting-user-name attribute if the corresponding + job attribute was an empty string. While this is + allowed by the IPP specification, some HP JetDirect + implementations return a client-error-bad-request + error if an empty name attribute value is received. + The new code only sends these attributes if they are + not the empty string. + - At least some versions of the HP JetDirect firmware + do not correctly implement IPP. Added additional + checks to the IPP backend to eliminate extra, + unsupported attributes which should normally be + ignored by a compliant IPP device. + - The scheduler did not copy the complete list of + supported file types into the + document-format-supported attribute. This caused + clients to not send the local file type (such as + application/vnd.cups-raw for raw print files) and the + corresponding bad output in some cases. + - The scheduler did not fully copy attributes from a + set-job-attributes request - string attributes were + only referenced, which could cause cupsd to crash + or behave irratically. + - The lp command didn't send the right value for the + job-hold-until attribute when "-H resume" was + specified. + - The IPP backend now returns as soon as a job is + completed or reported as "pending-held". + - Added new ImplicitAnyClasses and HideImplicitMembers + directives to the cupsd.conf file to make implicit + classes more usable/transparent to the user. + - Clients can now (with the appropriate authentication) + retrieve and update the server configuration files + using HTTP GET and PUT requests. + - The web interface didn't allow you to modify the + location or description of the printer. + - The pdftops filter now uses its own temporary file + function to work with PDF files using LZW compression + (which use the uncompress program or gunzip) + - The SystemGroup directive now supports specification of + multiple groups. + - Added new Include directive to cupsd.conf, a la + Apache. + - Added new pseudo-driver/PPD called "raw" that can be + used to create/convert a raw queue. This also allows + raw queues to be created in the web interface. + - The pdftops filter didn't handle image objects that + used JPEG and Flate compression together. + - The pstops filter counted pages wrong when using the + N-up and even/odd printing options. This prevented + the page-ranges option from working properly. + - Added another fix to pstoraster for a bus error + condition caused by a lack of parenthesis in the + Ghostscript code. + - Added new "natural-scaling" option which scales the + natural size of the image (percent of natural image + size instead of percent of page size.) + - The lppasswd program is now setuid to the CUPS user + instead of root. + - The PPD functions did not allow for PPD files that + defined the page sizes and margins before the page + size options. + - The mime.types file now checks for the PJL "LANGUAGE = + Postscript" command for PostScript files. + - The scheduler did not truncate file: output files. + - The PPD file reading code did not handle options with + raw quotes (") in the human-readable names. + - The pdftops filter now remaps the space character when + (bad) PDF files contain a .notdef glyph for the space + character. + + +CHANGES IN CUPS V1.1.8 + + - Updated spec file to generate separate cups-pstoraster + package for pstoraster. + - The spec file wasn't setting LOGDIR in the install. + - The scheduler might restart a stopped printer after + stopping a print job. Thanks to Florent + Guiliani for finding this bug! + - The init script showed run level 0 for the Red Hat + chkconfig program. This is incorrect because Red Hat + doesn't use run level 0 for shutdown scripts. + - The IPP backend did not handle the + client-error-not-found error when checking the status + of the job that was sent. This caused remote queues + to stop on client machines when the server had job + history disabled. + - Added httpConnectEncrypt() function to avoid + performance penalty for setting up encrypted + connections initially. + - Use httpConnectEncrypt() in all client apps and in the + CUPS API to ensure consistent usage of encryption + throughout. + - Jobs weren't queued to remote classes (fix from + Richard Begg.) + - AIX changes from Richard Begg. + - Fixed the pstops fix for GNOME output - no longer use + the page numbers in the %%Page: comment since GNOME + puts a filename instead (!?@!#?!). There is still an + issue with N-up printing since GNOME defines its fonts + in the first page instead of the document setup section + (pages must be independent according to the DSC spec) + People with GNOME printing problems should consult bug + #54489... + - The imagetops filter produced PAGE: messages when + generating PostScript for a non-PostScript printer + (only affects page-label and Classification + options.) + - The updated pdftops filter was looking for an options + file called xpdf.conf instead of pdftops.conf. + + +CHANGES IN CUPS V1.1.7 + + - Configuration script changes, including new + "--with-docdir=/dir" option to relocate CUPS + documentation and web content according to your + favorite version of the FHS. + - Documentation updates for encryption, SLP, etc. + - New Software Test Plan and automated test script to + test CUPS prior to installation. + - All scheduler configuration files are now case + insensitive to match Apache. + - Added support for Apache ListenBackLog, Require, + Satisfy, , , and LimitRequestSize + directives. + - Added support for all Apache log levels... + - Added support for "double" HostNameLookups. + - Added new "RunAsUser" directive to support non-root + configurations on the standard (priviledged) ports. + - Added support for non-root invocation of the lpd + backend (does no reserve a priviledged port, which + might not work with some LPD servers...) + - Added new PrintcapFormat directive to control the + output format of the printcap file (BSD or Solaris + formats are supported at present.) + - The CUPS directory service routines now handle + ECONNREFUSED errors gracefully rather than shutting + all browsing off. + - ippErrorString() now returns the recommended error + messages from the IPP/1.1 Model and Semantics + document. + - Fixed a minor IPP compliance issue with responses + to requests without the attributes-charset or + attributes-natural-language attributes. + - Sun fix: need httpFlush() call for chunked IPP + requests in cupsDoFileRequest(). + - httpConnect() now looks up "localhost" by name and + by address (127.0.0.1) for users the go to the + trouble of removing the required localhost entry + in /etc/hosts or on their DNS server... + - Added support for Linux 2.4.x devfs parallel port + filenames (/dev/parallel/N). + - cupsDo[File]Request() and cupsGetPPD() no longer + block trying to reconnect to a crashed or inaccessable + server. + - Added new ppdEmitJCL() function to better handle + PJL commands from PPD files. + - A bug in UpdateJob() would cause the scheduler to + consume 100% CPU until another request was submitted. + - The cancel command did not support the "-" option to + cancel all jobs on all printers. + - The cancel and lprm commands did not support cancelling + the next/current job in the queue. + - The pdftops and pstoraster filters were using unsafe + temporary file functions; while this is not a problem + in normal configurations (the CUPS temporary directory + is restricted), they now use the cupsTempFd() function. + - The mime.types file was missing the recognition rule + for Sun Raster images. + - The admin CGI was passing a printer make string to + ippSetCGIVars() that was being replaced in that + function. + - "lpoptions -l" would resave the options... + - The EPSON drivers now send the "end packet mode" + command when printing to USB devices. + - The scheduler initialized certificates before loading + the cupsd.conf file. + - The scheduler used /dev/random to collect random data, + which could block if insufficient entropy information + had been collected by the kernel. Now use + /dev/urandom. + - Fixed a bug in the whitespace skipping code in + httpGetSubField(). + - The LPD backend now supports a new "order" option: + "lpd://server/queue?order=control,data" (default) and + "lpd://server/queue?order=data,control". + - The scheduler enforced a 30 second timeout on all + clients regardless of the Timeout directive and if a + CGI was currently running. + - cupsParseOptions() now sets boolean options to + option=true or option=false. + - The "percent complete" calculations in the LPD backend + could overflow on large files, causing the percentage + to wrap to 0 every 40MB or so. + - Fixed a memory reallocation bug in pstoraster that + could cause it to crash. + - The LPD backend now sanitizes the job title to avoid + potential problems on remote LPD servers. + - The lp command did not send the requesting-user-name + attribute when altering a job. + - The pstops filter did not handle PostScript files with + lines longer than 8191 bytes. + - The scheduler no longer uses inet_addr() to convert IP + addresses in dot format (mmm.nnn.ooo.ppp) to the + 32-bit format, since it will not work for IPv6 + addresses. + - New "Classification" directive to force labeling of + the current classification on each page. + - New "page-label" attribute to add per-page labels + ("For Official Use Only", "Draft", etc.) + - The scheduler now sets the HTTPS environment variable + for CGI programs when a client connects using + encryption. + - Fixed a recursion bug in the scheduler that could + cause cupsd to crash when a printer was removed. + - The LPDEST and PRINTER environment variables didn't + support instances. + - Dropped the "file" backend from the device list that + is reported, since it is only available for *testing* + and should never be used in a production environment. + The file: device can still be used, but it won't show + up in the list of devices from lpinfo or the web + interface. + - Added support for /dev/lpa# parallel ports under *BSD. + - Added META variables to the CGI header template to + prevent caching of the results. + - Fixed an unaligned memory buffer for the pstoraster + clist states; this caused bus errors for some + combinations of printers, drivers, and options. + - Re-added black reduction for colorful colors; this + helps to prevent dark colors from getting desaturated. + (only used when converting RGB to CMYK) + - Added two new directives - MaxJobsPerPrinter and + MaxJobsPerUser - to allow an administrator to set + the maximum number of pending jobs in a queue or + submitted by a user. + - The scheduler no longer stops a printer if it can't + create the status pipe or run the filters or backend. + This will allow heavily loaded servers to service + clients or start print jobs as the load allows. + - Fixed a bug in the Set-Job-Attributes code that could + crash the scheduler (patch from Martin Zielinski) + - cupsSetDests() did not quote option values with + embedded spaces. + - Added support for the Enable-Printer and + Disable-Printer extension operations (same as + CUPS-Accept-Jobs and CUPS-Reject-Jobs.) + - The AppSocket and IPP backends now wait for the print + job to be finished before exiting; this should prevent + the loss of print jobs with older JetDirect firmware + and make consecutive print jobs print faster. + - The BMP loading code did not handle resolution values + of 0. This is a problem with BMP image files produced + by the GIMP. + - The HTTP Upgrade code (upgrade to TLS encryption) + bypassed the authentication checks. + - The HTTP Upgrade code did not send a 426 status code + to the client and end the current request. This caused + a race condition between the client and server for the + upgrade to TLS. + - Fixed a bug in the EOF and Trailer detection code in + the pstops filter. + - The imagetoraster filter did not add the margins to + the custom page size in the raster header. + - The imagetops filter did not adjust the custom page + size to the size of the printed image. + - The imagetops filter did not include DSC comments + which are required by some printers. + - The imagetops filter did not insert newlines in + Base85 encoded output, causing files to contain + lines longer than 255 characters (violation of the + DSC). + - Added support for the DeskJet 900 series duplexer + and CRET color modes in the HP driver. + - Added support for PPD-defined margins in the HP + driver. + - Fixed the debugging output from pstoraster - the + font list was not terminated by a newline. + - Some versions of the HP-UX pam_unix authentication + module apparently do not pass the appdata_ptr argument + to the conversation function, preventing the scheduler + from authenticating users using PAM under HP-UX. A + workaround using a static variable has been added to + address this problem. + - Fixed a bug in the scheduler SortPrinters() function + that could cause printers to disappear or the + scheduler to crash when adding a printer. + - Changed the pstops filter to not do per-page filtering + if the file does not conform to at least version 3.0 + of the document structuring conventions. This seems + to "fix" printing with broken apps. + - The image filters did not handle older TIFF files that + lacked the samples-per-pixel and bits-per-pixel tags. + - Added new cupsGetJobs() and cupsFreeJobs() functions + to manage print jobs. + - cupsEncodeOptions() would encode names of 0 length and + cupsAddOption() and cupsParseOptions() would add names + of 0 length. + - The scheduler might block waiting for status messages + after starting a new print job. Thanks to Florent + Guiliani for finding this bug! + + +CHANGES IN CUPS V1.1.6-3 + + - The configure script put the JPEG library before the + TIFF library; this caused problems in some + configurations since the TIFF library also supports + JPEG compression of TIFF images. + - Updated the configure script and makefiles to handle + admin man pages with the "1m" extension (HP-UX, IRIX, + Solaris, Tru64) and in odd directories (IRIX) + - The updated cupsTempFile() function did not return + the filename when called with a filename buffer of + NULL (previously it used a static buffer.) + - FreeBSD uses /dev/unlptN, but NetBSD and OpenBSD use + /dev/ulptN. + - DeletePrinter() didn't remove the printer from any + classes it was a member of. + - DeletePrinterFromClass() didn't preserve the + implicit status of a class. + - DeletePrinterFromClasses() didn't remove printers + from implicit classes. + - StartJob() didn't send the job-sheets, job-priority, + and job-hold-until attributes to remote printers. + - LoadAllJobs() was looking for job-sheets-completed + instead of job-media-sheets-completed. This would + prevent accumulation of page data after a restart + of the scheduler. + - The pstops and imagetops filters now generate copies + using the appropriate method for a Level 1, 2, or 3 + printer since some Level 2/3 printers don't support + the /#copies variable anymore. + - The man page for cups-lpd did not mention the "-o" + option. + - The IPP backend didn't handle version-not-supported + errors and revert to IPP/1.0 (previously it only checked + for a bad-request error) + - Caldera fix: lpc now reports unimplemented commands as + unimplemented, not invalid. + - Caldera fix: lpq didn't recognize BSD lpq "-a" option. + - Caldera fix: lpr didn't recognize BSD lpr "-1", "-2", + "-3", "-4", "-q", or "-U" options. + - RedHat fixes: patches to GNU Ghostscript + - SuSE fix: temp file creation patch to GNU Ghostscript + (pstoraster). + - SuSE fix: remove cgi-bin/abort.c and cgi-bin/email.c, + which are not used. + - SuSE fix: missing NULL check in cgi_initialize_post(). + - SuSE fix: potential buffer overflows in + cgi_initialize_string(). + - SuSE fix: potential buffer overflows in + ippSetCGIVars() + - SuSE fix: more NULL checks in ppdOpen(); also make + sure that all memory is freed on error to avoid memory + leaks. + - SuSE fix: Exit from child if setgid() or setuid() + fails. + - SuSE fix: Added setgroups() calls after setgid() and + setuid() calls. + - SuSE fix: potential buffer overflows in httpEncode64() + calls. + - SuSE fix: potential buffer overflows in httpSeparate() + - SuSE fix: potential buffer overflows in ippWrite() for + bad input. + - SuSE fix: potential nul skip in ppd_decode() for + missing hex digits. + + +CHANGES IN CUPS V1.1.6-2 + + - Added changes to support NetBSD startup scripts. + - Added separate compiler options for pstoraster + (Ghostscript) to avoid compiler-induced errors + from Ghostscript's twisted code. + - The mime.types file contained syntax errors. + - Updated the *BSD USB device filenames to use + the /dev/unlptN files so that the USB device + is not reset prior to printing (causes print + corruption on many printers) + - Added new cupsTempFd() function to avoid serious + security bug in glibc fopen() function. The glibc + fopen() function unlinks a file before creating it, + which opens up possible symlink attacks. + - Now reject 0-length names in add-printer and add-class + requests. + - Fix for pstoraster when ZLIB is not available. + - cupsGetPPD() didn't reconnect when a HTTP connection + was lost. + - SuSE fix: httpConnect() didn't check that the + value from gethostbyname() was a valid IPv4 address. + - SuSE fix: httpConnect() didn't allow file descriptor 0 + to be used for a socket. + - SuSE fix: ippRead() didn't confirm that all values in + a set were numeric or string types. + - SuSE fix: lppasswd race condition fixes. + - SuSE fix: directive names could overflow buffer when + reading *.conf files. + - SuSE fix: HEAD requests for PPD files did not use the + same logic as GET requests. + - SuSE fix: possible buffer overflow when adding + /index.html to requested directory name. + - SuSE fix: possible buffer overflow when converting + IPP attributes to string options for filters. + - SuSE fix: creating file: device output with mode 0666 + instead of mode 0600. + - SuSE fix: creating job info files with mode 0640 + instead of 0600. + - SuSE fix: don't rely on snprintf() for including + system name in log filenames. + - SuSE fix: add bounds checking when copying quoted + and hex strings. + + +CHANGES IN CUPS V1.1.6-1 + + - Added configure check for getting the correct + strftime() format string; %c is not Y2k safe, + and %KC and NULL are not universally supported. + + +CHANGES IN CUPS V1.1.6 + + - Fixed another possible DoS attack in httpGets() + - Added check for "LANGUAGE = PCL" and "LANGUAGE = + POSTSCRIPT" in mime.types. + - Resolution options were not being passed into the + filter programs properly. + - The default compiler options for GCC no longer include + "-g3", which apparently is deprecated in newer + versions of GCC. + - CheckJobs() could cause cupsd to crash if a job is + cancelled in StartJob(). + - The printers.conf and classes.conf files are now + written with restricted permissions. + - The round-robin algorithm used by FindAvailablePrinter() + had problems; fixes contributed by Joel Fredrikson. + - If LoadAllJobs() is unable to determine the file type + of a print job, assume "application/vnd.cups-raw". + - The web interface now provides a job_printer_name + value for any corresponding job_printer_uri value. + - The cups-lpd mini-daemon now logs the client address + and hostname as well as all commands and errors in the + syslog file. + - The IPP backend now detects the supported file formats + and only specifies the document format if it is + supported. This makes IPP printing to network print + servers and cards more reliable without affecting the + capabilities of CUPS servers. + - The time_at_xyz attributes are now converted to human- + readable dates and times for the web interfaces. + - The HP and EPSON sample drivers now correctly catch + signals and eject the current page when a job is + cancelled. + - Fixed bug in CGI code - did not ignore control + characters (e.g. newlines) in form data. This caused + sporatic web interface problems. + - The file type logging code in the scheduler referenced + the optional document-format attribute; the new code + uses the resolved MIME type instead. + - The client.conf parsing code now removes trailing + whitespace. + - The MaxJobs directive was being treated as a boolean + instead of an integer. + - The scheduler would not timeout remote printers if + BrowseInterval was set to 0. + - The lpadmin command now supports setting of options + and user-level access control. + - Added "-E" option to all printing commands to force + encryption. + - The client code did not consume the response to the + OPTIONS request when switching to secure mode. + - The scheduler did not output a Content-Length field + when responding to an OPTIONS request. + - Added documentation on using cups-lpd with xinetd + to the man page. + - The socket backend now starts retries at 5 seconds and + increases the interval to 30 seconds. This should + provide faster printing when multiple jobs/files are + queued for a printer. + - The filters and backends no longer buffer output to + stderr. This should provide much more accurate status + reporting. + + +CHANGES IN CUPS V1.1.5-2 + + - Fixed configure check for OpenSSL to work with RSA + code. + - Added configure check for , and use this + check in backend/serial.c. + - Updated configure script handling of data, + configuration, and state directories to use datadir, + sysconfdir, and localstatedir variables. + - NetBSD uses different serial port filenames than + FreeBSD and OpenBSD. + - The pdftops filter didn't need some X-specific files. + - The scheduler makefile doesn't do a chown anymore when + installing (cupsd did this automatically on startup + anyways) + + +CHANGES IN CUPS V1.1.5-1 + + - There was a typo in the top-level Makefile + - The top-level Makefile did not install an init script + for run level 5. + - The configure script did not add the "crypto" library + when checking for the OpenSSL library. + - The OKIDATA PPD files were missing. + - The config.h.in file defined the wrong version number. + - The serial backend did not define "funky_hex" under *BSD. + - Updated the Visual C++ project files and some of the + CUPS API sources to compile under Windows again. + + +CHANGES IN CUPS V1.1.5 + + - Security updates - new default configuration does + not broadcast printer information and only allows + access from the local system. + - EXPERIMENTAL encryption support - CUPS now optionally + supports TLS/SSL encryption via the OpenSSL library. + - Documentation updates. + - Makefile/configure script updates. + - The RPM spec file didn't work out-of-the-box under + RedHat or Mandrake. + - Minor code cleanup to remove extraneous compiler + warnings. + - cupsTempFile() was using %p for the temporary + filename; this should have been %08x (just 8 digit + hex) + - Deleting a printer with active print jobs would still + crash the server. + - ippWrite() and ipp_write_file() didn't send the + correct value length for name-with-language and + text-with-language attributes. + - Updated IPP code to support copied strings (that + should not be freed); this provides slightly more + efficient IPP server performance. + - Updated PDF filter to Xpdf 0.91. + - httpGets() could go into an infinite loop if a line + longer than the input buffer size was sent by a + client. This could be used in a Denial-of-Service + attack. + - The lpstat and CUPS API functions now request only the + data required when getting the list of printer or + class information. This should improve performance + with large numbers of printers on slower machines. + - The scheduler was always enforcing the FilterLimit, + even if FilterLimit was set to 0. + - Updated the Linux USB backend to support Mandrake's + /dev/usb/usblp# filenames. + - The PRINTER and LPDEST environment variables did not + override the lpoptions default printer. + - The PPD read functions incorrectly included trailing + characters (usually whitespace) after quoted string + attributes. + - The multiple-document-handling attribute handling code + did not check for the correct value for collated + copies (separate-documents-uncollated-copies). + - The EPSON driver did not work with OKIDATA printers in + EPSON emulation mode (needed change-emulation command) + - The HP-GL/2 filter did not scale the plot properly in + scale mode 2. + - Added PPD files for 9-pin and 24-pin OKIDATA printers. + - The httpSeparate() function didn't handle passwords + that started with a number. + - ippDelete() could free the character set string + multiple times in name-with-language and + text-with-language attributes. + - The scheduler would access freed memory right after + freeing it (for debug messages); these parts of the + code have been reordered to avoid this situation + which was causing sporatic errors and crashes. + - The ppdClose() function didn't free all of the strings + in the ppd_file_t structure. + - The LoadAllJobs() function in the scheduler did not + close the spool directory. + - Changed all sprintf's that use string formats to + snprintf's, even if the destination buffer is + larger than the source string(s); this protects + against buffer overflows caused outside of CUPS... + - Changed all strcpy's to strncpy's between local and + global variables, even if the destination buffer is + larger than the source string; this protects + against buffer overflows caused outside of CUPS... + - The CUPS certificate functions didn't use the + CUPS_SERVERROOT environment variable when set. + - The directory services code was copying instead of + comparing the remote printer info, resulting in + unnecessary updates of the printer attributes for + remote printers. + - Added new mime.types rules to allow automatic raw + printing of PCL and ESC/P files; PJL headers are + parsed to differentiate between PostScript and + PCL job files. This should eliminate a lot of + the reports of SAMBA printing problems due to + the missing "-oraw" or "-l" options. + - The mimeLoadType() function didn't handle the + 3-argument contains() function. + - The LoadPPDs() function in the scheduler didn't + properly set the alloc_ppds variable or handle a PPD + database containing 0 printers. + - The scheduler FindAvailablePrinter() function didn't + use the same queuing logic as the CheckJobs() + function. This caused classes to stall if a remote + printer was always busy. + - Jobs are now assigned to printers in a class + round-robin style. This should prevent the first + server in the class from bearing the brunt of the + jobs. + - The scheduler's LoadAllJobs() function didn't always + restore remote printers for queued jobs on startup. + - The serial backend didn't support the higher baud + rates with the old termios interface. It now supports + 57600 and 115200 baud. + - The serial backend now supports different types of + flow control; previously it ignored the flow=XYZ + option in the device URI. + - The serial backend now supports DTR/DSR flow control, + which is popular on dot-matrix printers (access with + "flow=dtrdsr" in the device URI) + - Added new job-originating-host-name attribute for + jobs. The new attribute provides the hostname or + IP address of the machine that submitted the job. + - The set-job-attributes code no longer allows read-only + job attributes to be changed. + - Expanded the click area for the navigation bar in the + web interface. + - Updated the lp and cancel commands to support all of + the Solaris print options (some are simply ignored + since they do not map) + - Updated the scheduler to limit the number of file + descriptors to the maximum select() set size. This + was causing problems on Solaris systems where the + max FD count was increased beyond 1024. + - The scheduler's LoadDevices() function was getting + interrupted by the SIGCHLD signal handler; now ignore + child signals while loading devices. + - Added quota and allow/deny user support for printers + and classes. + - Removed black/CMY adjustment code from the PS and + image file RIPs; it was interfering with some CUPS + driver dithering code. + - The lpc program stopped listing the queue statuses + after the first active printer. + - The cups-lpd program used an output format that the + Solaris printing system did not understand. + - Updated the lpq program to use the Solaris format + except under Tru64 UNIX. + - Some DEC PPD files incorrectly use "Off" for the null + value in UI constraints. Added "Off" to the list of + accepted null values. + - Changed the *BSD define constants to __*BSD__ in all + of the backends. + - Added support for "lpstat printername", which is an + undocumented feature in Solaris. + - The HP-GL/2 filter now only sets the plot size if it + is set in the plot file. + - The lpmove command wasn't sending the requesting + user name, causing it to always fail. + - Updated the cupsTempFile() code to use GetTempPath() + under Windows. + - The cups-lpd mini-daemon didn't limit the number of + data files accepted, didn't use cupsTempFile(), + didn't handle control file job information in any + order, and didn't free job options after printing + a file. + - The scheduler copy_banner() function did not + explicitly set the owner and permissions of the banner + files, which could prevent the banner pages from + printing on some systems. + - The lpstat program wasn't listing remote classes. + - The scheduler did not verify that the printer-uri + attribute was specified in all requests that required + it. + + +CHANGES IN CUPS v1.1.4 + + - Makefile and configure script fixes. + - **** Changed the default Printcap setting **** to + /etc/printcap. There are just too many people asking + why application XYZ doesn't see their printers! + - The web admin interface now displays an error if it + can't get the list of printer drivers from cupsd. + - The IPP backend was putting the copies option before + the other job options were set. This caused the IPP + request to contain attribute groups in the wrong + order, which prevented remote printing. + - Added checks in scheduler to free memory used for + IPP requests and language information when closing + a client connection. + - Fixed the duplex option in the HP LaserJet driver. It + should now work with all LaserJet printers (and + compatibles) + - The add-printer web interface didn't initialize the + "old info" data pointer, which caused random crashes + on many OS's. + - Fixed many page sizes defined in the Level 1 + compatibility file "gs_statd.ps" to match reality. + - Fixed another bug in the setpagedevice "code" in + Ghostscript. It should now accept all standard + Adobe attributes on all platforms. + - Fixed pstoraster so that it reallocates memory for + color depth changes as well as size/resolution + changes. This removes an ordering constraint on + the color, page size, and resolution options in + PPD files. + - The IPP backend didn't use the job's character set + when the destination printer supported it. This + caused problems when printing text files to other + CUPS servers. + - Updated the logic used to determine when to rebuild + the PPD file database. The scheduler now checks the + dates and the number of PPD files (was just checking + the dates.) + - Updated the ippSetCGIVars() function (used by the + web interfaces) to only filter valid string values. + - The PostScript filter was scaling 2-up pages + incorrectly. This caused the edges of some pages to + be clipped. + + +CHANGES IN CUPS v1.1.3 + + - Makefile fixes. + - RPM spec file changes. + - Documentation updates. + - Enabled pstoraster debug messages for everything + (only logged when LogLevel set to "debug"...) + - Changed the Input/OutputAttributes fix in + pstoraster so that it works on all platforms. + - The HP-GL/2 filter didn't set the right green + color value in encoded polylines or text. + - Updated the "fitplot" code to handle plot sizes + specified as "PSwidth,length" and "PSlength,width". + - Updated the Linux parallel and USB backends to open + the device files prior to looking in /proc for + autoprobe info. This makes sure that loadable device + driver modules are in fact loaded... + - Added new FilterLimit directive to limit the number + of processing jobs/filters on a system. + - set-job-attributes didn't change the job-state to + held/pending when the job-hold-until attribute was + specified. + - set-job-attributes didn't save the new job attributes. + - Now change the "requesting-user-name" attribute in + requests from remote systems to "remroot" when an + unauthenticated "root" user is sent. This can be + changed using the new RemoteRoot directive in + cupsd.conf. + - The cancel-job, hold-job, release-job, and restart-job + operations didn't log the authenticated username. + - The cups-lpd mini-daemon now checks for a + document-format option before forcing raw mode with + filter mode 'l'. + - The cups-lpd mini-daemon now supports "-o" options + on the command-line (passed by inetd) to set global + defaults for all print queues. + - The pstops filter assumed that a file with a Trailer + comment would also have an EOF comment. + - Added new cupsSetPasswordCB(), cupsSetServer(), + cupsSetUser(), and ippSetPort() functions to better + support client applications (especially GUIs...) + - The CUPS-add-class and CUPS-add-printer operations + didn't reset the printer-name attribute on remote + print queues that had to be renamed when a local + printer was defined with the same name. + - The lpoptions command now supports a "-r" option to + remove options for a printer or instance. + - The lpadmin and admin.cgi programs no longer allow + class and printer names to begin with a number; this + caused the command-line utilities to become confused. + - The Linux USB backend now looks for both the parallel + and usblp driver names in the device list. + - Added a new FontPath directive to cupsd.conf, and also + a "--with-fontpath" option for the configure script to + specify alternate font paths for pstoraster. + - The CUPS-move-job operation didn't update the + job-printer-uri attribute. + - The scheduler only looked up printers and classes by + name in IPP requests, instead of using the full URI. + This caused problems with KUPS and friends with + remote printers. + - The scheduler now handles better localization of + hostnames (e.g. server is host.foo.com, remote is + host.subfoo.foo.com, localized is not host.subfoo...) + - The scheduler logging functions now use a common + log file checking/rotation function (courtesy of + Crutcher Dunnavant at Red Hat) + - The scheduler could accept more client connections + than it allocated for if more than one Port or Listen + line was present in cupsd.conf. + - Other minor scheduler performance tweeks. + - The lpq and lprm commands didn't support the default + printer set using lpoptions. + - The lpoptions command now supports a "-l" option to + list the printer-specific options and their current + settings. + - The web printer and class lists now show a link to the + default printer or class at the top of the page. + - The text filter now supports pretty printing of shell + and perl scripts as well as C/C++ source files. + - The top and bottom margins were reversed for landscape + text printing. + - The lpq and lprm commands didn't understand printer + instances. + - The scheduler only selected on the first 100 file + descriptors instead of the maximum file descriptor + limit. + - The scheduler client, listener, and mainline functions + now share code to disable and enable monitoring for + new client connections. + - The imagetoraster filter didn't support all of the + required pagedevice parameters. + - The serial backend now checks for 100 serial ports + under Linux. + - The scheduler used sscanf() to pull out the remote + printer location, description, and make/model strings, + but if any of these options was empty then sscanf() + would stop processing. + - Added "debug2" log level to provide a little less + verbose debugging information at the "debug" level. + - The scheduler would crash if you stopped a printer + that was currently printing a job. + - The scheduler incorrectly allowed jobs in the cancelled, + aborted, or completed state to be cancelled. + - The image filters did not load TIFF images properly + for bottom-to-top and right-to-left orientations. + - Added new cupsEncodeOptions() function to encode + CUPS options as IPP job attributes. + - The IPP backend, LPD mini-daemon, client commands, + and CUPS API did not properly encode multiple + option values separated by commas. + - Added new scheduler malloc logging in debug mode + (provides summary of total arena size, allocated, + and free bytes once a minute) + - The EPM-based distributions didn't install the + correct symlinks for a few man pages. + - Fixed a memory leak in the scheduler - wasn't + freeing old filters when deleting or renaming + printers. + - The scheduler now queries the primary IP address + for the name of the server and maps any incoming + requests from that address to the server name. + This fixes web admin mapping problems from + server.domain.com to localhost. + - The web printer modify interface now remembers + the previous device and driver settings (except + for serial ports.) + - The job-k-octets attribute is now stored as part of + the job attributes; this preserves the information + after a job is completed when job file history is + turned off. + - Dropped option sub-group parsing code for the moment, + since many Xerox PPD files abuse this feature in PPD + files and don't follow the hierarchy rules. + - Added new wrapper code around options so that duplex + options for some HP printers don't prevent prints. + - Added support for Digital UNIX/Tru64 UNIX/OSF/1 format + for "lpstat -v" output. + - Now show the URI for remote printers instead of + /dev/null in "lpstat -v" output. + - Creating classes and adding printers to a class with + the lpadmin command didn't work. + - The banner pages and test page should now format + correctly in both portrait and landscape orientations. + - Updated banner page substitution so that { can appear + by itself without quoting. + + +CHANGES IN CUPS v1.1.2 + + - Makefile/configure fixes + - RPM spec file and EPM list file fixes + - The cupsTempFile() function now uses a different + algorithm for generating temp files and "reserves" + them to avoid possible security exploitation. + - Now use /dev/random (if available) to seed the random + number generator for certificates. + - The /var/spool/cups and /var/spool/cups/tmp directories + were incorrectly owned by root; they are now owned by + the filter user, typically "lp". + - The scheduler now resets the permissions on the spool + and temp directories as needed to match the filter + user. + - Now expose ppdCollect() as an externally callable + function. + - The image filters now support filtering from the + standard input. + - The imagetoraster filter now collects all printer + options and job patch files and applies them to the + page header as needed. + - Added format and banner options to LPD backend. + - The send-document operation didn't start a job + immediately when last-document was true. + - The set-job-attributes operation didn't correctly + replace the current job-hold-until value. + - Removed the option wrapper code from ppdEmit() and + friends since it caused problems with Ghostscript + and many PS printers. + - Was setting TZ environment variable twice for job + filters. + - Added syslog logging in cups-lpd to aide in + debugging problems. + - The HP-UX parallel port backend did not list the + available parallel ports on some systems (printf + calling problem...) + - The lp and lpr commands overrode user options if + -d/-P were specified after -o. + - The scheduler would crash with a */* filter. + - Added support for a "default" filter for unknown file + types. The example provided in the mime.types and + mime.convs file prints unknown files as if "-oraw" was + specified for the job. This functionality is disabled + by default. + - The "compatibility" mode fix for older backends did not + work for smbspool. Added a workaround for it. + - The HP-GL/2 filter didn't perform the right pen scaling + with some files and the "fitplot" option. + - New Software Performance Specification document that + describes the memory, disk, and CPU usage of all the + CUPS software. + + +CHANGES IN CUPS v1.1.1 + + - The pstoraster Makefile still referenced one of the + old PDF filter files. + - The filter Makefile used INSTALL_DATA instead of + INSTALL_LIB to install the CUPS image library. + - The administration CGI didn't work properly with + network devices. + - The BrowseACL variable was not updated after the + cupsd.conf file was loaded. + - The lpd mini-daemon didn't support printer instances. + - Now use a default umask of 077 for child processes. + - Now put temp files in /var/spool/cups/tmp for child + processes and the root user, unless TMPDIR or TempDir + is defined otherwise. + - cupsGetPPD() no longer uses easy-to-guess filenames. + - The CUPS-Delete-Class and CUPS-Delete-Printer + operations now save classes.conf file as needed. + - The lppasswd command wouldn't add a user. + - The ppdOpen() function could cause a segfault if a + 0-length PPD file was read. + - The image filters were not handling images with + different X and Y resolutions properly. + - The imagetoraster filter defaulted to RGB output + instead of black output like pstoraster. + - The pstops filter didn't handle binary data properly. + - The pstops filter didn't handle copies properly for + PS files lacking DSC comments. + - The pstops filter now appends %%EOF to the end of + documents if they don't have it. + - The cupsGetPPD() function didn't work with remote + printers lacking the @server in the name. + - The configure script didn't work right when only + --prefix was specified. + - The ppdEmit() code now wraps all printer commands so + that buggy PostScript printers will still print a file + after receiving an option that isn't available. + - Fixed the DeskJet margin bug, and disabled 600dpi + color mode until it can be fixed. + - The cupsAddDest() function didn't sort instances + correctly in all cases. + - The time-at-xyz attributes now expand to the date and + time in banner files. + + +CHANGES IN CUPS v1.1 + + - Documentation updates. + - Configuration script updates. + - Didn't map charset and language value strings to lowercase + and _ to - as required by SLP and IPP. + - ppdLoadXYZ() didn't add the list of available fonts to the + ppd_file_t structure. + - The text filter common code was freeing the PPD file data + before it was used. + - The text filter now embeds missing fonts. + - The CGI interface now maps local access to the server to + the localhost address. + - The HP-GL/2 filter didn't use the specified (or default) + color ranges, resulting in strange colors. + - The HP-GL/2 filter didn't default to no input window, which + caused unnecessary clipping of plots. + - Integrated Xpdf's pdftops filter into CUPS, which is a + lightweight and reliable replacement for Ghostscript's + PDF support. + - Removed all PDF support from Ghostscript. + - Updated HP driver to set top margin; this seems to fix + the offset problem seen on HP DeskJet printers. + - Fixed dependencies on the ZLIB and JPEG libraries in + pstoraster. + - The lpr command wasn't using the lpoptions defined by + the user. + - The lpr command would segfault if the CUPS server was + not running. + - The top-level makefile was not installing the CUPS + initialization script. It now does so if it sees there + is an init.d directory in /sbin, /etc/rc.d, or /etc. + - "lpstat -v all" didn't work. + - pstoraster would crash on some platforms doing the + setpagedevice operator. + - The web administration interface now allows you to set + the default banner pages. + - Images can now be positioned on the page using the new + "position" option. + - The AccessLog, ErrorLog, and PageLog directives now + support "%s" to insert the server name. + - Added a new BrowseShortNames directive to allow for + short remote printer names ("printer" instead of + "printer@server") when possible. + - The scheduler could crash if given an invalid PPD file + with no PageSize attributes. + - Updated the serial, parallel, and usb backends to do + multiple writes and ignore ioctl() errors as needed; + this should fix problems with serial printing on old + serial drivers and with the UltraSPARC parallel port + driver under Solaris 2.7. + - Now propagate LD_LIBRARY_PATH to child processes from + cupsd. + - New DataDir directive for installing in alternate + locations. + - New CUPS_SERVERROOT and CUPS_DATADIR environment + variables to specify installation directories as + needed. + - Queued remote jobs recreate remote printers as needed + when the scheduler is started. + - Deleting a printer also purges all jobs on that + printer. + - Old job and control files that don't belong to a + printer are automatically deleted. + - Wasn't updating time-at-processing and + time-at-completed attributes in job. + - Didn't send required multiple-operation-time-out + attribute in response to a get-printer-attributes + request. + - cups-lpd now supports options set with lpoptions. + - The job-hold-until attribute is now provided with all + jobs. For jobs that are not currently held the value + is "no-hold". + - The scheduler was not sending "unknown" values in IPP + responses. + - The lpoptions command now accumulates options from + previous runs rather than replacing all options for a + printer. + - The IPP backend now switches to IPP/1.0 if a 1.1 + request fails. + - The lpadmin and admin.cgi programs now validate new + printer and class names. + - The access_log file now includes the number of IPP bytes + received in a POST request. + + +CHANGES IN CUPS v1.1b5 + + - Documentation updates. + - The pstoraster filter didn't compile without the JPEG library. + - The cupsd server didn't support the HTTP OPTIONS request + method. + - Dropped the "CLOSE" method supported by the cupsd server. + (not defined in HTTP specification) + - Makefile/configure script fixes. + - Missing the job-restart template. + - Added IPP test suite for testing. + - Missing IPP documentation from binary distributions. + - Fixed multiple-document handling code when last-document + not specified. + - Added more checks to IPP requests to prevent bad requests + from getting through. + - Not all of the Ghostscript error output was being sent to + stderr. + - The PostScript filter now added PJL commands to set the + job name and display string, if supported. + - The scheduler would crash if the browse socket could not + be bound. Now disables browsing if port 631 (reserved for + IPP) is being used by a misbehaving daemon. + - The USB backend now looks for the older Linux 2.2.x USB + printer device filenames as well as the newer ones. + - The IPP backend now uses the UTF-8 charset exclusively, + since apparently only CUPS handles more than US-ASCII and + UTF-8... + - Wasn't quoting ( in PostScript banners... + - Send-document requests with no document-format attribute + could cause cupsd to crash. + - Old jobs in the spool directory might cause cupsd to + crash. + - CUPS now supports all of the recommended job-hold-until + keywords as well as name values of the form "HH:MM" and + "HH:MM:SS". + - Added placeholder pointer for TLS encryption to the HTTP + connection structure. + - Fixed the "fast poll" bug reported by DISA - the + status pipe wasn't being closed for multi-file jobs. + - Revamped put_params code in pstoraster to fix bitmap + allocation bug with FrameMaker output. + - Ripped out filename, etc. code from pstoraster as it + is a potential security hole. + - Added support for RIP_CACHE environment variable in the + new pstoraster. + - Fixed USB device filenames for Linux; now support new + pre-2.4 devices (/dev/usb/lp#) and 2.2 devices + (/dev/usblp#) + - Fixed accept-jobs crash with classes. + - Didn't include dot-matrix EPSON drivers in previous + release. + + +CHANGES IN CUPS v1.1b4 + + - Documentation updates. + - Many makefile and configuration script fixes (should + now compile better under *BSD.) + - The MediaPosition attribute was being mishandled by + GhostScript, causing the RIP to fail whenever a paper + tray was selected. + - The scheduler now logs the final line of log information + from a filter, even if it doesn't end with a newline; this + primarily affects GhostScript error output. + - The scheduler was saving implicit classes, so after a few + restarts you'll end up with AnyPrinter, AnyAnyPrinter, etc. + - The JPEG autodetection didn't work with some JPEG files that + came from digital cameras (JPEG but not JFIF); the new + magic types should work with all images that the JPEG library + can handle. + - Fixed a bug in the new contains() MIME type rule that could + cause cupsd to crash. + - Switched to using strtol() in the MIME type code so that you + can use hex, octal, or decimal constants as desired in the + mime.types file. + - Banner files are now treated as templates, allowing any type + of file to be used as a banner. + - Added a 30-second timeout to backend device reports so that a + hung backend will not prevent the scheduler from starting. + - Backends are once again terminated when jobs are stopped; the + CUPS-supplied backends will stay alive until the downstream + filters have had a chance to clear out old page data. + - The charset lookup in the CUPS localization support was wrong + (iso8859-x instead of iso-8859-x) + - Changed the "cpNNNN" code page files to "windows-NNNN" to match + the IANA registrations. + - New PostScript banner pages. + - Added Windows BMP and Alias PIX image file support to the image + filter. + - The PNG reading coded didn't free all of its buffers. + - Added Digest authentication support to the client and server + code. + - Added Solaris options to System V commands. + - Now support the output-bin job template attribute. + - Now log the job-billing attribute in the page_log file, and + keep track of the total number of pages in the + job-media-sheets-completed attribute. + - The penwidth option is now in micrometers to support more + accurate width specification. + - The image filters now support interlaced and transparent PNG + files. + - Didn't handle Keep-Alive for HTTP/1.0 clients. + - The BrowsePoll support didn't handle when BrowseInterval + was set to 0 (now uses 30 seconds if BrowseInterval is 0) + - The DeskJet driver now supports 600 DPI color for printers + that support it. + - New lpinfo and lpmove commands. + - The lpq command now supports the Digital UNIX output format. + - The LPD mini-daemon now supports all required LPD operations. + - Implemented timeouts for multi-file documents. + - New cupsPrintFiles() function in the CUPS API library to + print multiple files using create-job and send-document + requests (1 job ID for multiple files) + - The lp command now sends multiple files as a single job, + matching the behavior of the System V command. + - The "cancel -a" command now purges job history files. + + +CHANGES IN CUPS v1.1b3 + + - Documentation updates. + - The startup script redirected stderr before stdout, + which caused problems with some versions of Bourne + shell and Bash. + - Fixed a bug in the scheduler's PPD language reading + code. + - Fixed a bug in the scheduler's check for the + manufacturer in the PPD. + - The pstoraster filter didn't allow some input and + output attributes to be set. + - Added banner page support. + - Added missing PAM configuration file. + - Configuration script fixes for Linux and *BSD. + - The log file code was using the wrong sign for the + timezone offset. + - The default printcap file is now empty (no printcap + file is generated). + - The scheduler did not start jobs destined for remote + printers when they became available. + - The scheduler now sends jobs to remote printers + immediately. (when sending jobs to a class, the remote + printer is only used when it becomes available) + - The scheduler now supports printing of banner pages + via the job-sheets attribute (banner files go in + /usr/share/cups/banners) + - The cupsd process now forks itself into the background + (override with -f) + - Added several *BSD enhancements. + - Added UNSUPPORTED libtool option to configuration + script to allow the use of libtool. Note that this is + UNSUPPORTED by us, but added by request of the *BSD + folks. + - The parallel, serial, and usb backends now retry the + opening of their ports. This allows multiple print + queues to be associated with a single physical port, + and will allow CUPS to support several types of + parallel port auto-switches in the near future. + - Set-Job-Attributes now supports adding, changing, and + deleting job template attributes, and no longer allows + job-printer-uri to be set (see CUPS-Move-Job) + - Added CUPS-Move-Job operation to support moving of jobs. + - The CGI template functionality now supports multiple + languages (still only have templates for English) + - The CUPS-Get-Printers and CUPS-Get-Classes operations + now support filtering as defined in the IDD. + - The Get-Jobs, CUPS-Get-Printers, and CUPS-Get-Classes + operations no longer limit themselves to 1000 jobs, + printers, or classes (believe it or not, this is + needed for some sites) + - The web interfaces now support language-specific + templates. + - The web admin interface now supports class management. + - The web admin interface now shows a list of + manufacturers before selecting the PPD/driver for a + specific printer. + - The web admin interface now supports configuration of + the default printer options in the PPD file. + - The web interface now uses printer/class + authentication for the test page instead of admin + authentication. + - Updated the RPM spec file for the current release. + - Updated language support for Windows code pages. + - 8-bit character set files can now use multiple fonts + (needed for Arabic, Greek, Hebrew, etc.) + - Added basic right-to-left text support in the text + filter. + - The POSIX locale now uses ISO-8859-1 instead of + US-ASCII. + - Fixed PDF printing problems. + - Fixed PostScript RIP page device dictionary elements + that weren't getting passed in cups_get_params(). + - Added a new "contains" rule for the magic file typing. + - The "printable" rule now accepts characters from 128 to 255 + (needed for Microsoft character sets) + - Added support for ~/.cupsrc as well as /etc/cups/client.conf + so that the default server can be configured on a per-user + basis without environment variables. + - Added LPD mini-daemon to support incoming LPD jobs. + + +CHANGES IN CUPS v1.1b2 + + - Documentation updates. + - The lp command didn't always load the user-defined + destinations, preventing it from seeing the default + printer. + - Many configure script and makefile fixes. + - The Microsoft code page files were missing from the + distribution. + - Added a workaround for the HP IPP client (which is sending + an invalid printer-uri in requests) + - Fixed the encoding of text-with-language and name-with-language + to match the IPP spec. + - Added support for unknown value tags in the IPP routines + (previously they would be ignored) + - Integrated GNU GhostScript 5.50 into the pstoraster filter. + - Client hostname resolution was broken on little-endian + machines. + - Now look at client.conf file for client's default server + and printer. + - The cupsServer() function did not close the client.conf file + if it contained a ServerName directive. + - Added BrowseAllow, BrowseDeny, BrowseOrder, BrowsePoll, and + BrowseRelay directives. + - BrowseInterval 0 disables advertising of local printers, but + still receives information on remote printers. + - New browse polling daemon (for polling servers on different + networks) + - New PPD cache file for faster startup times with large numbers + of PPD files. + - The Host: field was incorrectly required for HTTP/1.0 clients. + - New set-job-attributes operation now supported. + - The mime_load_types() and mime_load_convs() functions did not + close their input files. + + +CHANGES IN CUPS v1.1b1 + + - NEW web-based administration interface. + - NEW EPSON printer drivers. + - NEW user-defined printers and options. + - NEW persistent jobs and job history + - NEW IPP/1.1 support + - NEW template-based web interfaces. + - NEW CUPS-get-devices and CUPS-get-ppds operations. + - NEW support for create-job and send-file operations. + - NEW certificate-based authentication for local + administration. + - NEW USB backend. + - The lpr command now produces human-readable error messages. + - The lpq command now produces BSD standard format output + instead of OSF/1 output. This should resolve the SAMBA + print queue problems that have been reported. + - The IPP backend did not always detect when the "raw" option + was being used. + - The "lpstat -p" command would stop after the first active + printer. + - The "lpstat -v" command would stop before the first remote + printer. diff --git a/CHANGES-1.2.txt b/CHANGES-1.2.txt new file mode 100644 index 0000000000..7b0b1f2de1 --- /dev/null +++ b/CHANGES-1.2.txt @@ -0,0 +1,1261 @@ +CHANGES-1.2.txt +--------------- + +CHANGES IN CUPS V1.2.12 + + - The PHP cups_print_file() function crashed if the options + array contained non-string option values (STR #2430) + - The image/tiff file matching rule incorrectly identified + some text files as TIFF files (STR #2431) + - The filter(7) man page incorrectly documented the + "PAGE: total #-pages" message (STR #2427) + - PCL text files were mis-identified as HP-GL/2 and + caused the HP-GL/2 filter to hang (STR #2423) + - When printing to a queue with user ACLs, the scheduler + incorrectly returned a quota error instead of a "not + allowed to print" error (STR #2409) + - cupsaddsmb could get in a loop if no printer drivers + were installed (STR #2407) + - cupsRasterReadHeader() did not byte-swap the header + properly when compiled with certain versions of GCC. + - The IPP backend did not send the document-format + attribute for filtered jobs (STR #2411) + - Some PPD files could cause a crash in ppdOpen2 (STR + #2408) + - The web admin interface incorrectly handled the "share + printers" and "show remote printers" settings (STR + #2393) + - The scheduler's log messages about AuthClass and + AuthGroupName advised using a replacement directive but + had the wrong syntax (STR #2400) + - Updated the PostScript/PJL and HP-GL/2 MIME rules to + look in the first 4k of the file, not just the first 1k + (STR #2386) + - Updated the Italian localization (STR #2382) + + +CHANGES IN CUPS V1.2.11 + + - Fixed the "relaying from" log message (STR #2376) + - Updated the launchd support on Mac OS X to better + support reconfiguration. + - "make distclean" didn't remove all generated files + (STR #2366) + - Fixed a bug in the advertisement of classes (STR + #2373) + - The IPP backend now stays running until the job is + actually printed by the remote server; previously + it would stop monitoring the job if it was held or + temporarily stopped (STR #2352) + - PDF files were not always printed using the correct + orientation (STR #2348) + - The scheduler could crash if you specified a bad file: + URI for a printer (STR #2351) + - The Renew-Subscription operation now returns the + notify-lease-duration value that was used (STR #2346) + - The IPP backend sent job options to IPP printers, + however some printers tried to override the options + embedded in the PS/PCL stream with those job options + (STR #2349) + - ppdLocalize() now also tries a country-specific + localization for when localizing to a generic locale + name. + - The cupstestppd program now allows for partial + localizations to reduce the size of universal PPD + files. + - Chinese PPD files were incorrectly tagged with the + "cn" locale (should have been "zh") + - The backends now manage the printer-state-reasons + attribute more accurately (STR #2345) + - Java, PHP, Perl, and Python scripts did not work + properly (STR #2342) + - The scheduler would take forever to start if the + maximum number of file descriptors was set to + "unlimited" (STR #2329) + - The page-ranges option was incorrectly applied to the + banner pages (STR #2336) + - Fixed some GCC compile warnings (STR #2340) + - The DBUS notification code was broken for older + versions of DBUS (STR #2327) + - The IPv6 code did not compile on HP-UX 11.23 (STR + #2331) + - PPD constraints did not work properly with custom + options. + - Regular PPD options with the name "CustomFoo" did + not work. + - The USB backend did not work on NetBSD (STR #2324) + - The printer-state-reasons attribute was incorrectly + cleared after a job completed (STR #2323) + - The scheduler did not set the printer operation policy + on startup, only on soft reload (STR #2319) + - The AP_FIRSTPAGE_InputSlot option did not clear any + ManualFeed setting that was made, which caused problems + with some PPD files (STR #2318) + - cupsDoFileRequest() and cupsDoRequest() did not abort + when getting an error in the response (STR #2315) + - The scheduler did not schedule jobs properly to remote + or nested classes (STR #2317) + - Updated the mime.types and mime.convs headers to warn + that the files are overwritten when CUPS is installed. + Local changes should go in local.types or local.convs, + respectively (STR #2310) + - The scheduler could get in an infinite loop if a + printer in an implicit class disappeared (STR #2311) + - The pstops filter did not handle %%EndFeature comments + properly (STR #2306) + - Fixed a problem with the Polish web page printer icons + (STR #2305) + - ppdLocalize() now also localizes the cupsICCProfile + attributes. + - The scheduler still had a reference to the incorrect + "notify-recipient" attribute (STR #2307) + - The "make check" and "make test" subscription tests did + not set the locale (STR #2307) + - The "make check" and "make test" subscription tests + incorrectly used the notify-recipient attribute instead + of notify-recipient-uri (STR #2307) + - cupsRasterInterpretPPD() incorrectly limited the + cupsBorderlessScalingFactor when specified in the + job options. + + +CHANGES IN CUPS V1.2.10 + + - ppdLocalize() now supports localizing for Japanese + using the "jp" locale name used by the ppdmerge + program from the CUPS DDK 1.1.0 (STR #2301) + - _cupsAdminSetServerSettings() did not support changing + of top-level directives as designed. + - The init script path check was broken. + - CUPS incorrectly used the attribute "notify-recipient" + instead of "notify-recicpient-uri" in several places + (STR #2297) + - Fixed a configure script bug on MirBSD (STR #2294) + - The pdftops filter did not limit the amount of recursion + of page sets (STR #2293) + - Custom page sizes with fractional point sizes did not + work (STR #2296) + - The lpoptions command would crash when adding or removing + options on a system with no printers (STR #2295) + + +CHANGES IN CUPS V1.2.9 + + - The scheduler did not use the default job-sheets + (banners) for implicit classes (STR #2284) + - The scheduler could crash when listing complete jobs + that had been unloaded from memory (STR #2288) + - The French localization was doubled up (STR #2287) + - Build system fixes for several platforms (STR #2260, + STR #2275) + - The scheduler's openssl certificate generation code was + broken on some platforms (STR #2282) + - The scheduler's log rotation check for devices was + broken (STR #2278) + - The LPD mini-daemon did not handle the document-format + option correctly (STR #2266) + - The pdftops filter ignored the "match" size option in the + pdftops.conf file (STR #2285) + - cupstestppd now validates UTF-8 text strings in + globalized PPD files (STR #2283) + - The outputorder=reverse option did not work with all + printers (STR #2279) + - Classes containing other classes did not always work + (STR #2255) + - Printer location and description information was lost + if the corresponding string contained the "#" character + (STR #2254) + - cupsRemoveOption() did not work properly (STR #2264) + - The USB backend did not work with some USB to parallel + cables on Mac OS X. + - The test page did not print the rulers properly on + large media sizes (STR #2252) + - The text filter could crash when pretty printing certain + types of files (STR #2158) + + +CHANGES IN CUPS V1.2.8 + + - Documentation fixes (STR #2141, STR #2157) + - The HTTP upgrade redirection used by the scheduler did + not work with Internet Explorer (STR #2235) + - Members of a class with Unicode names did not appear + correctly in the web interface (STR #2154) + - Changing the "Save debugging information" setting in + the web interface no longer affects the other server + settings (STR #1993) + - The scheduler did not choose SSL certificates correctly + on Mac OS X (STR #2225) + - The scheduler could get in an infinite loop when + printing to a remote class (STR #2228) + - The jobs web page did not have separating space after + the number of pages column (STR #2230) + - Added French localization (STR #2221) + - Updated Spanish localization (STR #2223) + - Updated Japanese localization (STR #2216) + - cupsBorderlessScalingFacter was limited to a range of + 0.9 to 1.1, but some printers need larger values (STR + #2222) + - Landscape printing of PDF files did not always work + (STR #2149) + - Fixed slow USB printing on Minolta printers (STR #2104, + STR #2219) + - The ZPL label printer driver could produce stretched + output (PR #6448) + - The IPP backend now clears the printer-state-message + when there are no outstanding errors or warnings (STR + #2126) + - The CUPS Java scripting support did not work with + recent versions of Java due to the use of Sun's private + Base64 class (STR #2152) + - The scheduler did not pass HTTP GET form variables to + custom CGI programs (STR #2173) + - The lpoptions command now displays the reason why a PPD + file cannot be found (STR #2184) + - The scheduler did not accept "none" as a browse + protocol name (STR #2200) + - The scheduler still loaded the remote printer cache, + even when browsing was disabled (STR #2198) + - The SNMP backend now shows OfficeJet printers with the + "HP" manufacturer prefix (STR #2151) + - Web interface HTML cleanup (STR #2153) + - The parallel backend consumed 100% CPU on FreeBSD due + to an apparently common parallel port driver bug (STR + #2161) + - ippReadIO() incorrectly returned IPP_IDLE when the + initial IPP message header could not be read (STR + #2179) + - cupsRasterInterpretPPD() did not support custom options + (STR #1960) + - Collated output produced by the PostScript filter could + lose some options (STR #2137) + - job-hold-until with time values for the next day would + be held for 60 days (STR #2144) + - Some types of Sun raster files did not print correctly + (STR #2107) + - Raw PBM files did not print correctly (STR #2106) + - The SNMP backend no longer uses IPP with HP printers, + as some recent firmware versions appear to not work + (STR #2055) + - cupsMarkOptions() did not handle the + multiple-document-handling option (STR #2135) + - lpstat did not show the local job ID of active printers + (STR #2125) + - The backends incorrectly used STATUS: + media-tray-empty-error messages for out-of-paper + conditions (STR #2123, STR #2124) + - cupsGetPPD2() returned the wrong error when the PPD + file did not exist (STR #2122) + - cupsDoAuthentication() did not translate the password + prompt (STR #2121) + - httpGetLength2() did not handle error messages without + content correctly (STR #2133) + - Added support for 32/64-bit libraries on HP-UX Itanium + systems (STR #2115) + - Fixed a configure script problem with the 32/64-bit + library support (STR #2114) + - The PostScript filter did not properly output document + setup commands for reversed output (STR #2111) + - The scheduler did not parse IPv6 netmasks properly (STR + #2117) + + +CHANGES IN CUPS V1.2.7 + + - Documentation updates (STR #2089) + - Added an Italian translation (STR #2105) + - The PostScript filter now rotates the bounding box + values as needed (STR #2079) + - The scheduler no longer loads the remote printer cache + when browsing is disabled (STR #2084) + - The scheduler no longer writes a new launchd + configuration file if it doesn't have to (STR #2083) + - Updated the USB and PAP backends for Mac OS X (STR + #2086) + - The scheduler now picks up on changes to IPv6 and DNS + configuration on Mac OS X (STR #2085) + - The lpstat program could still hang (STR #2098) + - Fixed an inefficiency in the SNMP IPP detection code + (STR #2100) + - The SSL negotiation code did not implement short + timeouts (STR #2091) + + +CHANGES IN CUPS V1.2.6 + + - The web interface was not localized on Mac OS X (STR + #2075) + - "lpc status" did not show the number of queued jobs for + disabled queues (STR #2069) + - The lpstat program could hang (STR #2073) + - The serial backend did not support the new USB serial + filenames on Linux (STR #2061) + - The parallel backend did not support bidirectional I/O + properly (STR #2056) + - The network backends now log the numeric address that + is being used (STR #2046) + - Fixed a compile error when using libpaper. + - Fixed a compile error when compiling on Solaris with + threading enabled (STR #2049, STR #2050) + - Missing printer-state-changed event for + printer-state-message updates (STR #2047) + + +CHANGES IN CUPS V1.2.5 + + - Documentation updates (STR #2038) + - The SNMP backend no longer uses IPP for Epson printers + (STR #2028) + - Updated the configure script for Tru64 UNIX 5.1 (STR + #2033) + - Tru64 5.1B's getaddrinfo() and getnameinfo() functions + leak file descriptors (STR #2034) + - cupsAddDest() didn't add the parent destination's + options and attributes. + - ppdConflicts() did not handle custom option + constraints. + - Raw printing of gzip'd files did not work (STR #2009) + - The scheduler no longer preserves default option + choices when the new PPD no longer provides the old + default choice (STR #1929) + - The Linux SCSI backend is now only built if the SCSI + development headers are installed. + - USB printing to Minolta printers did not work (STR + #2019) + - Windows clients could not monitor the queue status (STR + #2006) + - The scheduler didn't log the operation name in the + access_log file for Create-Job and Print-Job requests. + - The PostScript filter now separates collated copies + with any required JCL commands so that JCL-based + finishing options act on the individual copies and not + all of the copies as a single document. + - The PostScript filter now disables duplex printing when + printing a 1-page document. + - cups-lpd didn't pass the correct + job-originating-host-name value (STR #2023) + - Fixed some speling errors in the German message catalog + (STR #2012) + - cupstestppd did not catch PPD files with bad + UIConstraints values (STR #2016) + - The USB backend did not work with the current udev- + created printers if the first printer was disconnected + (STR #2017) + - Mirrored and rotated printing did not work with some + documents (STR #2004) + - 2-sided printing with banners did not work properly on + some printers (STR #2018) + - Updated the raw type rule to handle PJL within the + first 4k of a print job (STR #1969) + - Added an Estonian translation (STR #1957) + - Clarified the documentation for the cupsd.conf @LOCAL + and @IF(name) allow/deny functionality (STR #1992) + - The PostScript filters did not escape the Title and For + comments in the print job header (STR #1988) + - The scheduler would use 100% CPU if browsing was + disabled and the cupsd.conf file contained BrowsePoll + lines (STR #1994) + - The cupsDirRead() function did not work properly on + non-POSIX-compliant systems (STR #2001) + - The cupsFile functions didn't handle read/write errors + properly (STR #1996) + - The DBUS support now works with older versions of the + DBUS library. + + +CHANGES IN CUPS V1.2.4 + + - The --with-printcap configure option did not work (STR + #1984) + - The character set reported by cupsLangGet() did not + always reflect the default character set of a given + locale (STR #1983) + - Older Lexmark and Tektronix printers did not work with + IPP (STR #1980) + - Failsafe printing did not work (PR #6328) + - Some web interface redirects did not work (STR #1978) + - The web interface change settings button could + introduce a "Port 0" line in cupsd.conf if there was no + loopback connection available (STR #1979) + - The web interface change settings and edit + configuration file buttons would truncate the + cupsd.conf file (STR #1976) + - The German web interface used the wrong printer icon + images (STR #1973) + - The "All Documents" link in the on-line help was + missing a trailing slash (STR #1971) + - The Polish web interface translation used the wrong + URLs for the job history (STR #1963) + - The "reprint job" button did not work (STR #1956) + - The scheduler did not always report printer or job + events properly (STR #1955) + - The scheduler always stopped the queue on error, + regardless of the exit code, if the error policy was + set to "stop-printer" (STR #1959) + - ppdEmitJCL() included UTF-8 characters in the JCL job + name, which caused problems on some printers (STR + #1959) + - Fixed a buffering problem that cause high CPU usage + (STR #1968) + - The command-line applications did not convert + command-line strings to UTF-8 as needed (STR #1958) + - cupsDirRead() incorrectly aborted when reading a + symbolic link that pointed to a file/directory that did + not exist (STR #1953) + - The cupsInterpretRasterPPD() function did not handle + custom page sizes properly. + + +CHANGES IN CUPS V1.2.3 + + - The scheduler did not send job-state or + job-config-changed events when a job was held, + released, or changed (STR #1947) + - The scheduler now aborts if the configuration file and + directory checks fail (STR #1941) + - Fixed a problem with ippPort() not using the port + number that was set via the client.conf file or + CUPS_SERVER environment variable (STR #1945) + - HTTP headers were not buffered (STR #1899) + - Some IPP printers (HP) did not like UTF-8 job names + (STR #1837) + - The CUPS desktop icon is now localized for Polish (STR + #1920) + - Printer options were not always honored when printing + from Windows clients (STR #1839) + - The openssl command would lock up the scheduler when + generating an encryption certificate on some platforms + due to a lack of entropy for the random number + generator (STR #1876) + - The web admin page did not recognize that "Listen 631" + enabled remote access (STR #1908) + - The web admin page did not check whether changes were + made to the Basic Server Settings check boxes (STR + #1908) + - The IPP backend could generate N*N copies in certain + edge cases. + - The scheduler did not restore remote printers properly + when BrowseShortNames was enabled (STR #1893) + - Polling did not handle changes to the network + environment on Mac OS X (STR #1896) + - The "make test" subscription tests used invalid + notify-recipient-uri values (STR #1910) + - Printers could be left in an undefined state on system + sleep (STR #1905) + - The Berkeley and System V commands did not always use + the expected character set (STR #1915) + - Remote printing fixes (STR #1881) + - The cupstestppd utility did not validate translation + strings for custom options properly. + - Multi-language PPD files were not properly localized in + the web interface (STR #1913) + - The admin page's simple settings options did not check + for local domain socket or IPv6 addresses and did not + use "localhost" as the listen address. + - An empty BrowseProtocols, BrowseLocalProtocols, or + BrowseRemoteProtocols line would crash the scheduler + instead of disabling the corresponding browsing options. + - The scheduler now logs IPP operation status as debug + messages instead of info or error. + - cupsFileRewind() didn't clear the end-of-file state. + - cupstestppd didn't report the actual misspelling of the + 1284DeviceID attribute (STR #1849) + - BrowseRelay didn't work on Debian (STR #1887) + - configure --without-languages didn't work (STR #1879) + - Manually added remote printers did not work (STR #1881) + - The header was not installed. + - Updated the build files for Autoconf 2.60 (STR #1853) + - The scheduler incorrectly terminated the polling + processes after receiving a partial log line. + - The cups-lpd mini-daemon reported "No printer-state + attribute found" errors when reporting the queue status + (PR #6250, STR #1821) + - SNMP backend improvements (STR #1737, STR #1742, STR + #1790, STR #1835, STR #1880) + - The scheduler erroneously reported an error with the + CGI pipe (STR #1860) + - Fixed HP-UX compile problems (STR #1858, STR #1859) + - cupstestppd crashed with some PPD files (STR #1864) + - The and header files did not + work with C++. + + +CHANGES IN CUPS V1.2.2 + + - Documentation updates (STR #1765, STR #1780) + - CUPS didn't know about alternate character set names + for Asian text (STR #1819) + - The lpoptions -o and -r options did not work unless you + specified a printer. + - The lpoptions command incorrectly allowed users to set + printer attributes like printer-type (STR #1791) + - httpWait() did not flush the write buffer, causing "bad + request" errors when communicating with CUPS 1.1.x + servers (STR #1717) + - Polling did not sanitize the printer description, + location, or make and model strings like broadcasts + did. + - Polled printers did not show the server's default + job-sheets option value. + - The Samba password prompt was not properly localized + (STR #1814) + - Added a German translation (STR #1842) + - The scheduler now creates self-signed SSL certficates + automatically when using OpenSSL and CDSA for + encryption, just as for GNU TLS. + - The SNMP backend sporatically reported some printers as + "unknown" (STR #1774) + - The scheduler now forces BrowseTimeout to be at least + twice the BrowseInterval value and non-zero to avoid + common configuration errors. + - The scheduler incorrectly returned printer URIs of the + form "ipp://server/printers/classname" for classes (STR + #1813) + - Updated Japanese localization (STR #1805) + - The scheduler's SSL certificate/key directory was not + created on installation (STR #1788) + - Added a mailto.conf man page and help page (STR #1754) + - The parallel and USB backends no longer wait for the + printer to go on-line - this caused problems with + certain printers that don't follow with the IEEE-1284 + standard (STR #1738) + - The scheduler could crash on a reload when implicit + classes were present (STR #1828) + - The IPP backend incorrectly used the CUPS_ENCRYPTION + environment variable to determine the default + encryption mode when printing (STR #1820) + - USB printing did not work on Solaris (STR #1756) + - The scheduler sorted job priorities in the wrong order + (STR #1811) + - The scheduler did not automatically restart notifiers + that exited or crashed (STR #1793) + - IPv6 support did not work on NetBSD (STR #1834) + - The EPM packaging file did not work (STR #1804) + - The scheduler used up the CPU if BrowseRemoteProtocols + was empty (STR #1792) + - Custom page sizes did not work (STR #1787) + - The SNMP backend could crash on some systems when SNMP + logging was enabled (STR #1789) + - Browsing could produce some funny printer names when + ServerName was set to an IP address (STR #1799) + - Fixed the log message for BrowseRelay (STR #1798) + - Fixes to allow CUPS to compile on MirBSD (STR #1796) + - The scheduler incorrectly set the FINAL_CONTENT_TYPE + environment variable (STR #1795) + - The pdftops filter incorrectly embedded a "produced by" + comment, causing PDF printing not to work on some + operating systems (STR #1801) + - Sending raw jobs from a client system could cause the + client's scheduler to eventually crash (STR #1786) + - The scheduler now checks that the notifier exists prior + to accepting a new subscription request. + - The scheduler now reports the supported + notify-recipient schemes based on the contents of the + ServerBin/notifier directory. + - Event notifications did not include the + notify-sequence-number or other required attributes + (STR #1747) + - Allow/Deny addresses of the form "11.22.33.*" did not + work on Linux (STR #1769) + - cupsGetPPD() did not work if the scheduler was only + listening on a domain socket (STR #1766) + - The scheduler could crash advertising a class (STR + #1768) + - The scheduler could crash if the default printer was + deleted (STR #1776) + - Added a new default CUPS raster format (v3) which does + not compress the raster stream in order to provide the + same cupsRasterReadPixels() and cupsRasterWritePixels() + performance as CUPS 1.1.x. + - The cupsaddsmb man page listed the wrong files for the + CUPS driver. + - Some configure --with options did not work (STR #1746) + - "Allow @IF(name)" didn't work if "name" wasn't the + first network interface (STR #1758) + - The lpstat command did not use the correct character + set when reporting the date and time (STR #1751) + - The cupsaddsmb command and web interface did not update + the Windows PPD files properly, resulting in corrupt + PPD files for the Windows client to use (STR #1750) + - The cupsd.conf man page didn't describe the Listen + domain socket syntax (STR #1753) + - The scheduler no longer tries to support more than + FD_SETSIZE file descriptors. + - CDSA (encryption) support fixes for MacOS X. + - The lppasswd program needs to be setuid to root to + create and update the /etc/cups/passwd.md5 file (STR + #1735) + - 32/64-bit library installation was broken (STR #1741) + - The USB backend now reports a "no such device" error + when using the old filename-based USB URIs instead of + the "success" error. + - Increased the HTTP and IPP read timeouts to 10 seconds, + as 1 second was too short on congested networks (STR + #1719) + - The SNMP backend now uses the device description over + the printer-make-and-model attribute when the attribute + contains a generic name (STR #1728) + - Fixed another file descriptor leak when printing raw + files (STR #1736) + - Raw queues were not shared via LDAP (STR #1739) + - The pstops filter didn't always embed PageSetup + commands from the PPD file (STR #1740) + - "make install" didn't work if you disabled all of the + localizations. + - The scheduler didn't always choose the least costly + filter. + - Fixed parsing of IPv6 addresses in Allow, Deny, + BrowseAllow, BrowseDeny, and BrowseRelay directives + (STR #1713) + - Printers that were shared via LDAP did not get added to + the LDAP server properly (STR #1733) + - LDAP browsing would crash the scheduler if a required + value was missing (STR #1731) + - Special cases for the "localhost" hostname did not + work, causing printing to not work when the /etc/hosts + file did not contain a localhost entry (STR #1723) + - Updated the Spanish translation (STR #1720, STR #1770) + - Reverse-order page output was broken when N-up or + landscape orientations were used (STR #1725) + - The parallel, serial, socket, and USB backends needed + print data before they would report back-channel data, + causing problems with several new drivers (STR #1724) + + +CHANGES IN CUPS V1.2.1 + + - "lprm -h hostname" did not work (STR #1800) + - The web interface did not handle reloads properly for + MSIE (STR #1716) + - The configure script no longer adds linker rpath + options when they are unnecessary. + - The scheduler could crash printing a debug message on + Solaris (STR #1714) + - The --enable-32bit and --enable-64bit configure options + did not always work. + - The password prompt showed the domain socket address + instead of "localhost" for local authentication (STR + #1706) + - The web interface filtered the list of printers even if + the user wasn't logged in (STR #1700) + - The IPP backend did not work reliably with some Xerox + printers (STR #1704) + - Trailing banners were not added when printing a single + file (STR #1698) + - The web interface support programs crashed on Solaris + (STR #1699) + - cupstestppd incorrectly reported problems with + *1284DeviceID attributes (STR #1710) + - Browsing could get disabled after a restart (STR #1670) + - Custom page sizes were not parsed properly (STR #1709) + - The -U option wasn't supported by lpadmin (STR #1702) + - The -u option didn't work with lpadmin (STR #1703) + - The scheduler did not create non-blocking back-channel + pipes, which caused problems when the printer driver + did not read the back-channel data (STR #1705) + - The scheduler no longer uses chunking in responses to + clients - this caused problems with older versions of + CUPS like 1.1.17 (PR #6143) + - Automatic raw printing was broken (STR #1667) + - 6-up printing was broken (STR #1697) + - The pstops filter did not disable CTRL-D processing on + the printer/RIP. + - ppdOpen*() did not load custom options properly (STR + #1680) + - "Set Printer Options" in the web interface did not + update the DefaultImageableArea or + DefaultPaperDimension attributes in the PPD file (STR + #1689) + - Fixed compile errors (STR #1682, STR #1684, STR #1685, + STR #1690) + - The lpstat command displayed the wrong error message + for a missing destination (STR #1683) + - Revised and completed the Polish translation (STR + #1669) + - Stopped jobs did not show up in the list of active jobs + (STR #1676) + - The configure script did not use the GNU TLS + "libgnutls-config" script to find the proper compiler + and linker options. + - The imagetoraster filter did not correctly generate + several 1, 2, and 4-bit color modes. + - cupsRasterWritePixels() could lose track of the current + output row. + - cupsRasterReadPixels() did not automatically swap + 12/16-bit chunked pixel data. + - Moved the private _cups_raster_s structure out of the + public header. + - Updated the CUPS raster format specification to include + encoding rules and colorspace definitions. + - The Zebra PPD files had the wrong PostScript code for + the "default" option choices. + - The imagetoraster filter did not generate correct CIE + XYZ or Lab color data. + - The cups-config script did not work when invoked from a + source directory (STR #1673) + - The SNMP backend did not compile on systems that used + the getifaddrs emulation functions (STR #1668) + + +CHANGES IN CUPS V1.2.0 + + - Documentation updates (STR #1618, STR #1620, STR #1622, + STR #1637) + - Static file copy buffers reduced from 64k to 32k to + work around bogus MallocDebug library assumptions (STR + #1660) + - The scheduler did not decode the backend exit code + properly (STR #1648) + - The MacOS X USB backend did not report the 1284 device ID, + nor did it fix device IDs returned by HP printers. + - The scheduler started more slowly than 1.1.x with large + numbers of printers (STR #1653) + - cupsRasterInterpretPPD() didn't support the + cupsPreferredBitsPerColor attribute, and imagetoraster + didn't use the new API. + - The "make test" script did not create all of the necessary + subdirectories for testing (STR #1638) + - The scheduler did not prevent rotation of logs + redirected to /dev/null (STR #1651) + - "make test" did not include the SNMP backend in the + test environment (STR #1625) + - The EPM packaging files did not work (STR #1621) + - "Use Default Configuration" inserted a broken + configuration file (STR #1624) + - Redirects in the web interface did not always preserve + the encrypted status of a connection (STR #1603) + - Added the Apple "pap" backend. + - Added CUPS library to CUPS Image shared library + linkage to support Linux --as-needed linker option + (STR #1606) + - Fixed support for --enable-pie (STR #1609) + - The pdftops filter did not validate the length of the + encryption key (STR #1608) + - Updated the Polish localization. + - "Encryption Required" in the cupsd.conf file now only + requires encryption when the connection is not over the + loopback interface or domain socket. + - Printer names containing "+" were not quoted properly in + the web interface (STR #1600) + - The SNMP backend now reports the make and model in the + information string so that the auto-generated printer + name is more useful than just an IP address. + + +CHANGES IN CUPS V1.2rc3 + + - The cups-lpd program always did reverse lookups on the + client address, which could be a performance problem. + Added a "-n" option to disable lookups. + - When configured with SSL support, require encryption by + default when displaying the /admin location (STR #1592) + - The next job ID was not computed correctly if the job + cache file got out of sync with the spool directory + (STR #1582) + - The PNG image handling code used deprecated functions + from libpng (STR #1587) + - Added a Polish translation (STR #1584, STR #1586) + - More changes to the scheduler to improve battery life + on portable devices (STR #1583) + - Changed the default log level for status messages back + to "DEBUG" to be consistent with CUPS 1.1.x (STR #1579) + - The error string was not set properly when + cupsDoFileRequest() was given the name of a directory + (STR #1578) + - Fixed handling of job-hold-until (STR #1581) + - Added explicit notes to the cupsaddsmb man page + explaining that the driver filenames are case-sensitive + under UNIX and that they must be all lowercase (Windows + 2000) or all UPPERCASE (Windows 95/98/Me) to work (STR + #1568) + - The USB backend incorrectly split the manufacturer name + if it contained spaces (STR #1566) + - The scheduler would hang when listing PPD files for a + manufacturer whose name contained spaces (STR #1567) + - Added the SNMP backend for network printer discovery + (STR #1555) + - cupstestppd now fails PPD files with 1284DeviceId + instead of 1284DeviceID, and cups-driverd uses a + case-insensitive comparison when looking for it (STR + #1573) + - cupsDoFileRequest() and cupsDoRequest() now work + properly with non-blocking HTTP connections. + - Added Swedish translation (STR #1569) + - "make install" now installs the MIME files with world + read permissions (STR #1565) + - More CDSA encryption support fixes (STR #1563) + - Updated the default mime.types file to support printing + of files that do not have a locally-recognized MIME + media type to raw or System V queues. + - Updated the serial port detection code on Linux (STR + #1562) + - Added some more error checking to httpGetHostname() + (STR #1561) + - The title of some administration pages was not + localized (STR #1548) + - The edit-config.tmpl file was not generated or + installed for the Spanish or Japanese localizations + (STR #1547) + - The mimeDelete() function freed the types before the + filters, but the filters needed the type data (STR #1558) + - The scheduler didn't keep track of the status pipes + properly, leading to a bad select() for multi-file jobs + (STR #1559) + - The cupstestdsc program didn't validate the ordinal + page number value for %%Page: comments. + + +CHANGES IN CUPS V1.2rc2 + + - The scheduler was not always using the string pool, + causing random crashes. + - The lpmove and the web interface's Move Job button did + not work with stopped jobs (STR #1534) + - The PostScript filter did not handle the page-set + option properly with number-up printing (STR #1543) + - The scheduler now only warns about unsupported ACLs + once (STR #1532) + - The "fitplot" option did not work with output from + Mozilla (STR #1542) + - The imagetops filter did not work with Level 2 or 3 + printers (STR #1533) + - The scheduler now recognizes PostScript files with PJL + commands that do not include an ENTER LANGUAGE command. + - Added --with-printcap configure option. + - 64-bit SSL fixes for MacOS X. + - The scheduler didn't send some printer state change + events. + - The scheduler didn't send jobs to busy remote printers. + - Fixed some problems with the launchd support. + - Added new USB printer backend for MacOS X. + - The PostScript filter now handles files that start with + an incomplete PJL header (PR #6076) + - The web interface language selection code did not try + the generic language localization (STR #1531) + - The language cache, string pool, and transcoding caches + are now process global instead of per-thread to avoid + problems with GNOME and to allow for data sharing + between threads (STR #1530) + - Fixed a CUPS 1.1.x compatibility bug (STR #1528) + - The web interface redirection after certain printer + administration tasks was broken (STR #1516) + - Web interface authorization could get stuck (STR #1512) + - Localization updates (STR #1513, STR #1518, STR #1520) + - The pstops filter didn't work with some files (STR + #1523) + - "./configure --enable-static" didn't work (STR #1522) + - The scheduler was not using the configured default + Group (STR #1521) + - The web interface still did not show the localized time + and date for some locales and systems (STR #1509) + - httpAddrGetList() would crash on systems without + getaddrinfo(). + - Socket URIs without a trailing slash would cause the + port number to not be accepted (STR #1519) + - Local raw and System V printers were not advertised as + such for printer browsing (STR #1502) + - The RPM spec file incorrectly put duplicate copies of + the Japanese and Spanish web interface templates in the + main cups package (STR #1517) + - cupsSetDests() did not explicitly set the permissions + of the /etc/cups/lpoptions file (STR #1508) + - The lpq command crashed with the -h option (STR #1515) + + +CHANGES IN CUPS V1.2rc1 + + - Documentation updates (STR #1497, STR #1498) + - The scheduler now redirects browsers to https: URLs + when encryption is required. + - The scheduler would crash when printing with a banner + (STR #1500) + - cups-driverd did not use the LanguageEncoding attribute + in PPD files to convert the NickName to UTF-8 (STR + #1503) + - The lpadmin command could not set the + printer-error-policy attribute (STR #1504) + - The web interface did not show the time and date in the + correct format for the locale (STR #1505) + - CUPS no longer accepts print jobs if a printer does not + support the file format (STR #1501) + - Cleaned up the PostScript filter (pstops) so that it + properly supports %%IncludeFeature and page scaling + (STR #1453) + - Fixed the cupsFileRewind() and cupsFileSeek() functions + to work properly with uncompressed files. + - Added cupsFileGetLine(), cupsFileStderr(), + cupsFileStdin(), and cupsFileStdout() functions to the + CUPS library. + - Added a new cupstestdsc program to test the DSC + conformance of PostScript files. + - Added KDE/GNOME icons and a Manage Printers menu item. + - Added --enable-image and --enable-pdftops configure + options to control whether the image and PDF filters + are built and installed (default = yes for all + platforms but MacOS X) + - Fixed a minor memory leak in the PPD API. + - Fixed transcoding issues (STR #1493) + - The scheduler now enforces a minimum job cost of 100 + when doing FilterLimit checks. + - The scheduler would leak file descriptors when printing + to raw queues (STR #1491) + - The IPv6 support did not compile on Tru64 UNIX (STR + #1488) + - ppdOpen2() now converts the NickName and all UI text to + UTF-8 (STR #1475) + - The Set Allowed Users web page did not work (STR #1486) + - When the default policy was not set or set to a non- + existing policy, the scheduler did not set the default + policy name to "default" (STR #1484) + - The Zebra CPCL driver did not use the correct righthand + margin for the 4" wide label sizes. + - Fixed a problem with the parsing of fractional real + numbers in PPD files. + - Added Spanish localization files (STR #1480) + - Fixed localization of a few scheduler messages (STR + #1478) + - Fixed support for HEAD requests in the scheduler (STR + #1481) + + +CHANGES IN CUPS V1.2b2 + + - Updated the CUPS design description. + - Added --enable-32bit and --enable-64bit configure + options to allow building of separate 32/64-bit + libraries on systems that support both environments + (STR #1472) + - Various compiler warning fixes. + - Fixes for Solaris 10 builds against old GNU TLS and + LDAP libraries. + - Added a cupsArrayUserData() function to retrieve the + user data pointer for an array (useful for typing + arrays) + - The ppdEmitString() function did not compute the + required buffer size properly, leading to dropped + characters on the end of the printer commands in pstops + and imagetops (STR #1470) + + +CHANGES IN CUPS V1.2b1 + + - The serial backend now supports Equinox 8-port serial + hubs (STR #526) + - The IPP backend now supports a compression option to + compress print files as they are sent to the remote + server (STR #956) + - The CUPS browse protocol now supports passing of + default options and browse timeout values from the + server to the clients (STR #800) + - Implicit classes that timed out could cause the + scheduler to crash (STR #1439) + - Added DragonFly support in local device backends (STR + #1362) + - Added LDAP printer browsing support (STR #338) + - Added official support for printer maintenance commands + via the CUPS Command file format and hooks in the + printer-type and web interfaces (STR #932) + - The HP-GL/2 filter could get in an infinite loop trying + to convert HP-PCL files (STR #1415) + - CUPS now implements the HTTP/1.1 Expect header (STR + #1407) + - Options in PPD files are no longer automatically put in + an "Extra" group; rather, all options that are not + inside an Open/CloseGroup will be placed in the + "General" group (STR #1385) + - The scheduler now creates a job-uuid attribute that + uniquely identifies a job on a network (STR #1410) + - The init script now unsets the TMPDIR environment + variable to prevent user temporary directories from + being used by cupsd accidentally (STR #1424) + - Added support for launchd on MacOS X. + - Added support for notify_post on MacOS X. + - Added support for DBUS on Linux. + - All of the Berkeley (except for lpc) and System V + commands now support specification of user, host, and + port (STR #1028, STR #1029, STR #1087) + - The lpmove command now allows you to move all jobs for + a given queue (STR #56) + - The web interface now supports moving of a job or jobs + to another queue (STR #56) + - The web interface now provides searching, paging, and + changing of the sort/display order of classes, jobs, + and printers. + - cupsaddsmb now accepts a password on the command-line + and supports passwords with special characters (STR + #822, STR #1236) + - ppdLoad*() no longer tries to "fix" bad characters in + UI text (STR #1101) + - Printer names can now (reliably) contain Unicode + characters (STR #896) + - The lpstat command now shows the time and date of the + last printer state change instead of the hardcoded "Jan + 01 00:00" (STR #659) + - The scheduler now adds a job-actual-printer-uri + attribute to job objects when printing to a class (STR + #116) + - The scheduler now logs log file open errors to the + system log (STR #1289) + - The scheduler now sets the job-originating-user-name to + the authenticated username, if available (STR #1318) + - The scheduler now only updates the permissions of SSL + keys and certificates when they are under the + ServerRoot directory (STR #1324) + - The rastertodymo driver has been renamed to + rastertolabel (a symlink is installed so that existing + queues continue to work) and now also supports Zebra's + CPCL language. + - The lpstat command could show the wrong active job for + a printer (STR #1301) + - Fixed a potential crash problem in the scheduler when + aborting a CGI program (STR #1290) + - Added a "cancel all jobs" button to the class and + printer web interfaces (STR #1140) + - The add-printer web page now shows the + set-printer-options page after the printer has been + added (STR #690) + - The classes web page now provides links to each of the + member printers (STR #307) + - CUPS now handles HTTP request/response lines up to 32k + in length; this is mainly for better cookie support + (STR #1274) + - Added support for the Apache PassEnv and SetEnv + directives to cupsd.conf (STR #853) + - Added large file (64-bit) support (STR #541) + - Fixed a performance issue with the ippReadIO() + implementation (STR #1284) + - Fixed a performance issue with the scheduler's implicit + class implementation (STR #1283) + - The pdftops filter now adds the Title and Creator + fields from the PDF file to the PostScript document + comments section (STR #539, STR #830) + - Added a new cups_array_t and cupsArray*() functions to + the CUPS API to support sorted lists of data. + - Made the CUPS API library thread-safe (STR #1276) + - Added "media" option support for EFI EFMediaType option + (STR #902) + - Added write buffering to the HTTP code to improve + performance (STR #547) + - The scheduler now uses the attributes-natural-language + attribute to localize banner pages (STR #386) + - The scheduler now returns the address that was used to + connect to it (STR #1076) + - Fixed a problem with N-up printing and OpenOffice (STR + #576) + - Added support for the GCC position independent + executable options (STR #1209) + - Added new BrowseLocalProtocols and + BrowseRemoteProtocols directives to cupsd.conf, + allowing for different browse protocols for local and + remote printers (STR #877) + - PPD files can now contain strings up to 256k in length + (STR #1215) + - The pstops filter now supports the IncludeFeature DSC + comment (STR #1212) + - The pstops filter now disables the setpagedevice + procedure when doing N-up printing (STR #1161) + - The serial backend now supports "stop=1", "stop=2", + "parity=space", and "parity=mark" options (STR #1155) + - "make install" no longer overwrites an existing PAM + configuration file (STR #1064) + - The scheduler now closes all files on startup when run + in daemon mode (STR #1009) + - Added a new RGBW colorspace to the CUPS raster format + (STR #1071) + - The pdftops filter now sets the page size based on the + media box when not scaling the output (STR #912) + - The pdftops filter now supports masked images (STR + #281) + - The pdftops filter produced large output when rendering + PDF files containing lot of repeated images (STR #327) + - The pdftops filter now minimizes print processing of + PDF files when using the page-ranges option (STR #273) + - Updated pdftops filter to Xpdf 3.01. + - Added new cupsBackchannelRead() and + cupsBackchannelWrite() functions, as well as + backchannel support to the parallel, serial, socket, + and USB backends (STR #1252) + - The parallel and USB backends now treat a "no space + available" error as an out-of-paper condition (STR + #1225) + - The "lpc" command now supports the "status all" command + (STR #1004) + - ippReadIO() did not read collections properly (STR + #1249) + - The "make test" script now creates the test files in + "/tmp/cups-$USER" instead of "/tmp/$USER" (STR #981) + - All backends now abort on error when printing a job to + a class - this allows the next printer in the class to + print the job (STR #1084) + - The scheduler now verifies that a printer supports + Letter or A4 media sizes before setting them as the + initial default (STR #1250) + - The cupstestppd program now flags bad Resolution + options (STR #1269) + - The USB backend now retries printing when the printer + is disconnected or turned off (STR #1267) + - Added new httpGetHostname() function to CUPS API, and + use it instead of gethostname() so that the web + interface will work correctly on systems whose hostname + is not the FQDN (STR #1266) + - The scheduler now stops printers if the backend for the + queue is missing on startup (STR #1265) + - The configure script now supports "--disable-library" + to disable particular image file format support + libraries, even if they are available on the build + system (STR #1248) + - The IPP backend did not always report on the total + number of pages that were printed (STR #1251) + - The lpstat program could display garbage date and time + values for locales whose date format exceeded 31 + characters (STR #1263) + - The cupstestppd program would segfault when testing + certain broken PPD files (STR #1268) + - Dramatically reduced the overhead of implicit classes. + - Added new cupsDir*() functions to CUPS API. + - Printers can now be published individually for sharing. + - Fixed a bug in the scheduler's startup signalling code + which caused cupsd to send the SIGUSR1 signal to the + init process instead of the original parent process + (STR #1258) + - Added new on-line help CGI to web interface to provide + searchable help. + - Devices are now tracked dynamically, with each query + doing a new device scan. This eliminates a previous + startup delay caused by slow backends and allows new + printers to be seen without restarting the server, + however it limits the amount of device URI checking + that can be done (basically now the scheduler only + requires a URI with a method that is a listed backend) + - Added new printer auto-detection, server configuration, + and log file viewing to the administration web page. + - Added new "set allowed users" web interface to set the + list of allowed users for a printer or class. + - The scheduler, command-line, and web interfaces now + limit the list of printers and classes to those + accessible by a user. + - cupsMarkOptions() now handles more non-standard + duplexing options and choices (STR #915) + - cups-lpd now honors remote banner requests with the + "standard" banner whenever a printer does not have one + defined (STR #1220) + - The scheduler's denial-of-service checks did not work + properly with IPv6 addresses (STR #1134) + - The lp and lpr commands did not error out properly when + they were unable to write to a temporary file (STR + #1129) + - The pstops filter did not handle Adobe-specific + comments in Windows NT driver output (STR #1085) + - "lpstat -l -p" incorrectly reported the printer + interface (STR #936) + - The web interface now operates exclusively with the + UTF-8 encoding, and sends the appropriate character set + and header information to the web browser (STR #919, + STR #1007) + - Added a "set allowed users" interface to the web + interface so that you can set the list of allowed or + denied users/groups for a printer or class. + - Disallow the "#" character in printer names, since it + has special meaning in the shell, config files, and in + URIs (STR #917, STR #1202) + - Added a new application/x-csource MIME type, and + support for it to the texttops filter so that you can + pretty print plain text files without the C/C++ + keywords being highlighted. + - The pdftops filter did not compile with GCC 4.0 (STR + #1226) + - The texttops filter did not highlight preprocessor + directives followed by a tab properly. + - HP PJL output now uses both JOB DISPLAY and RDYMSG + commands to show the current job on the printer's + display (STR #1218) + - Local authentication certificates are now stored in + /var/run/cups/certs by default instead of + /etc/cups/certs (STR #1211) + - Backends now use "&" to separate options in device + URIs; "+" is still recognized but is deprecated (STR + #842) + - The USB backend no longer supports the usb:/dev/foo + format on systems that support device ID queries. + - Forced classification markings did not work when the + job-sheets parameters were "none,none". + - "lpstat -l -p" incorrectly showed all users as allowed, + even if the queue was restricted to certain users (STR + #801) + - The scheduler now automatically detects SSL/TLS clients + without using the SSLPort/SSLListen directives. + - The CUPS API and scheduler no longer support SSLv2- + encrypted connections. + - Updated the cupsaddsmb utility to correctly export the + CUPS driver for Windows. + - Fixed a signal-handling bug in httpRead() which + ultimately caused the server to print multiple copies + when it was busy (STR #1184) + - The cupsFile API now uses the O_APPEND option when + opening files in append mode (STR #990) + - The md5.h header and md5_* functions are now officially + private and have been renamed to avoid conflicts with + other implementations with the same name. + - The pdftops filter incorrectly embedded some Type1 + fonts (STR #1093) + - The scheduler didn't detect a closed connection in the + middle of an IPP request (STR #1153) + - The scheduler could block trying to read the job status + if there was input pending and the job was cancelled in + the same input cycle (STR #1157) + - The scheduler could crash when deleting a class due to + infinite recursion. + - Updated the Zebra ZPL label printer driver to use the + run-length encoding and support more options. + - Updated serial backend to scan for /dev/ttyC* as well + as /dev/ttyc* for Cyclades serial ports (STR #1049) + - The scheduler could hang reading the job status under + certain circumstances (STR #1068) + - The USB backend termination signal code was inverted + (STR #1046) + - Moved enable and disable commands to sbindir to be + consistent. + - Added new cupsRasterInterpretPPD() function for RIP + filters to setup the raster page header from + PostScript commands in a PPD file. + - The CUPS browsing protocol now offers a "delete" bit + to remove printers as soon as they are deleted on the + server or as soon as the server shuts down gracefully + (STR #793) + - The CUPS_SERVER and ServerName directives (client.conf + and ~/.cupsrc) may now contain names of the form + "server:port" and "/path/to/domain/socket". + - The "cancel -u user" command now works for ordinary + users (STR #751) + - Added test run support to "make test" target (STR #64) + - Added domain socket support (STR #656) + - Added BrowseLocalOptions directive to allow the + administrator to add printer URI options to the browse + URI, e.g. "encryption=required" (STR #732) + - Added BrowseRemoteOptions directive to allow the + administrator to add standard URI options to the + remote printer URI, e.g. "encryption=required" (STR + #732) + - Now put "-I.." compiler option in front of all others + to ensure that local CUPS headers are used before + installed headers (STR #437) + - New cupsLangPrintf() and cupsLangPuts() for localized + interfaces. + - Now support custom attributes and extended options in + PPD files. + - Now provide functions to save PPD files. + - New policy mechanism allows per-operation and + per-printer control over what users and groups are + allowed to do various IPP operations. + - New error policy mechanism to control how aborted + backend errors are handled by the scheduler + (abort-job, retry-job, requeue-job, stop-printer) + - Updated the printer test page with a better color + wheel and a separate grayscale ramp. + - A single backend process is now run to send all print + data for a job. + - Backends and filters can now send and receive + backchannel data over file descriptor 3. + - Updated the raster stream format to support more + user-defined attributes and to do compression of the + page data. diff --git a/CHANGES-1.3.txt b/CHANGES-1.3.txt new file mode 100644 index 0000000000..9af534f9bf --- /dev/null +++ b/CHANGES-1.3.txt @@ -0,0 +1,856 @@ +CHANGES-1.3.txt +--------------- + +CHANGES IN CUPS V1.3.11 + + - The scheduler did not prevent nested classes (STR #3211) + - The scheduler did not reprint processing jobs that were moved to + another destination (STR #3222) + - The scheduler did not reset the current job file when stopping a + printer (STR #3226) + - The scheduler did not handle POSTs to custom CGIs properly (STR #3221) + - The pdftops filter did not print landscape PDF pages properly + (STR #2881) + - The scheduler did not handle partial header lines properly from CGI + programs (STR #3194) + - The web interface could hang on OpenBSD (STR #3176, STR #3196) + - The scheduler and cupsfilter utility did not handle rules starting + with a negation operator properly (STR #3160) + - The scheduler and cupsfilter utility would crash with certain MIME + .types rules (STR #3159) + - httpSetField wasn't bracketing IPv6 numeric addresses for the Host: + field (STR #3164) + - The ServerName, if specified, was not treated as a valid alias for the + local system (STR #3167) + - "make epm" did not work (STR #3166) + - "lpstat -h server" showed non-shared printers (STR #3147) + - "make check" did not work on Linux (STR #3161) + + +CHANGES IN CUPS V1.3.10 + + - Documentation fixes (STR #2994, STR #2995, STR #3008, STR #3056, + STR #3057) + - SECURITY: The scheduler now protects against DNS rebinding attacks + (STR #3118) + - SECURITY: Fixed TIFF integer overflow in image filters (STR #3031) + - The scheduler did not support the job-hold-until attribute with the + Restart-Job operation (STR #3130) + - SECURITY: The PNG image reading code did not validate the + image size properly, leading to a potential buffer overflow + (STR #2974) + - The rastertohp driver did not set the 1-sided printing mode when + needed (STR #3131) + - Now use a wrapper program instead of our fork of the Xpdf code to + support printing of PDF files. The new wrapper supports using Xpdf, + poppler, or Ghostscript to convert PDF files to PostScript (STR #3129) + - Long job names caused problems with some PJL printers (STR #3125) + - The lpq command did not work when showing all destinations (STR #3117) + - The scheduler used a codeset name of UTF8 which is not supported on + Solaris (STR #3113) + - cupsGetJobs() did not work with a NULL destination (STR #3107) + - Fixed a localization problem for option choices (incorrectly) named + "Custom" (STR #3106) + - The fallback OpenSSL random number seeding would not work (STR #3079) + - The scheduler might miss a child signal, causing high CPU usage. + - The scheduler did not enforce quotas after the job history was + unloaded (STR #3078) + - The job-k-limit, job-page-limit, and job-quota-period attributes + could not be set using the lpadmin command (STR #3077) + - httpSeparateURI() did not error out on URIs with a missing port + number after a colon. + - Fixed a Valgrind-detected initialization error when creating a + missing directory on startup. + - The scheduler did not always read all of the HTTP headers from a + CGI script/program. + - The scheduler did not always set the "air" property in Bonjour/DNS-SD + registrations. + - The scheduler incorrectly compared Mac OS X UUIDs for access + control, preventing access in certain configurations. + - The IPP backend incorrectly reset the required authentication + to Kerberos when authentication failed. + - The scheduler no longer looks up the local hostname by default; + turn on hostname lookups to restore the previous behavior. + - The scheduler did not always load MIME type rules correctly + (STR #3059) + - The test page did not format correctly on A4 paper (STR #3060) + - The web interface sometimes incorrectly redirected users to + 127.0.0.1 (STR #3022) + - cupsPrintFile*() did not send the document filename for single + file submissions (STR #3055) + - The scheduler did not update the member-names attribute when + removing the last printer from a class. + - The scheduler did not report PPD Products with parenthesis + in them properly (STR #3046) + - The wrong italic fonts were listed in the UTF-8 charset file + for the text filter. + - The backends did not return an OK status for the + CUPS_SC_CMD_GET_BIDI side-channel command (STR #3029) + - The scheduler did not purge jobs that were missing a + time-at-creation attribute, indicating a bad job control file + (STR #3030) + - The "-o job-hold-until=week-end" option did not work properly + (STR #3025) + - The Solaris USB printer device does not support select or poll + (STR #3028) + - The scheduler would crash if you exceeded the MaxSubscriptions + limit. + - The lp "-H immediate" option did not specify that the job + should not be held (STR #3013) + - The scheduler did not support the "Connection: close" + HTTP header (STR #3010) + - The mailto notifier didn't terminate messages properly + (STR #3011) + - Backends could spin trying to read back-channel data + (STR #3001) + - The HP-GL/2 filter was using the wrong default colors + (STR #2966) + - The scheduler incorrectly allowed Get-Jobs operations without a + printer-uri (STR #2996) + - The compression option was not being encoded properly + (STR #2997) + - Added a missing character map for JIS-X0213/ShiftJIS. + - The scheduler now rejects ATTR: messages with empty values. + - The scheduler could consume all CPU handling closed connections + (STR #2988) + - Fixed some configure script bugs with rc/xinetd directories + (STR #2970) + - The Epson sample driver PPDs contained errors (STR #2979) + + +CHANGES IN CUPS V1.3.9 + + - SECURITY: The HP-GL/2 filter did not range check pen numbers + (STR #2911) + - SECURITY: The SGI image file reader did not range check + 16-bit run lengths (STR #2918) + - SECURITY: The text filter did not range check cpi, lpi, or + column values (STR #2919) + - Documentation updates (STR #2904, STR #2944) + - The French web admin page was never updated (STR #2963) + - The IPP backend did not retry print jobs when the printer + reported itself as busy or unavailable (STR #2951) + - The "Set Allowed Users" web interface did not handle trailing + whitespace correctly (STR #2956) + - The PostScript filter did not work with Adobe applications + using custom page sizes (STR #2968) + - The Mac OS X USB backend did not work with some printers + that reported a bad 1284 device ID. + - The scheduler incorrectly resolved the client connection + address when HostNameLookups was set to Off (STR #2946) + - The IPP backend incorrectly stopped the local queue if + the remote server reported the "paused" state. + - The cupsGetDests() function did not catch all types of + request errors. + - The scheduler did not always log "job queued" messages + (STR #2943) + - The scheduler did not support destination filtering using + the printer-location attribute properly (STR #2945) + - The scheduler did not send the server-started, + server-restarted, or server-stopped events (STR #2927) + - The scheduler no longer enforces configuration file + permissions on symlinked files (STR #2937) + - CUPS now reinitializes the DNS resolver on failures + (STR #2920) + - The CUPS desktop menu item was broken (STR #2924) + - The PPD parser was too strict about missing keyword + values in "relaxed" mode. + - The PostScript filter incorrectly mirrored landscape + documents. + - The scheduler did not correctly update the + auth-info-required value(s) if the AuthType was Default. + - The scheduler required Kerberos authentication for + all operations on remote Kerberized printers instead + of just for the operations that needed it. + - The socket backend could wait indefinitely for back- + channel data with some devices. + - PJL panel messages were not reset correctly on older + printers (STR #2909) + - cupsfilter used the wrong default path (STR #2908) + - Fixed address matching for "BrowseAddress @IF(name)" + (STR #2910) + - Fixed compiles on AIX. + - Firefox 3 did not work with the CUPS web interface in SSL + mode (STR #2892) + - Custom options with multiple parameters were not emitted + correctly. + - Refined the cupstestppd utility. + - ppdEmit*() did not support custom JCL options (STR #2889) + - The cupstestppd utility incorrectly reported missing + "en" base translations (STR #2887) + + +CHANGES IN CUPS V1.3.8 + + - Documentation updates (STR #2785, STR #2861, STR #2862) + - The scheduler did not add the ending job sheet when the + job was released. + - The IPP backend did not relay marker-* attributes. + - The CUPS GNOME/KDE menu item was not localized for + Chinese (STR #2880) + - The CUPS GNOME/KDE menu item was not localized for + Japanese (STR #2876) + - The cupstestppd utility reported mixed line endings for + Mac OS and Windows PPD files (STR #2874) + - The pdftops filter did not print landscape orientation PDF + pages correctly on all printers (STR #2850) + - The scheduler did not handle expiring of implicit classes + or their members properly, leading to a configuration where + one of the members would have a short name (STR #2766) + - The scheduler and cupstestppd utilities did not support + cupsFilter and cupsPreFilter programs with spaces in their + names (STR #2866) + - Removed unused variables and assignments found by the + LLVM "clang" tool. + - Added NULL checks recommended by the LLVM "clang" tool. + - The scheduler would crash if you started a printer that + pointed to a backend that did not exist (STR #2865) + - The ppdLocalize functions incorrectly mapped all generic + locales to country-specific locales. + - The cups-driverd program did not support Simplified Chinese + or Traditional Chinese language version strings (STR #2851) + - Added an Indonesian translation (STR #2792) + - Fixed a timing issue in the backends that could cause data + corruption with the CUPS_SC_CMD_DRAIN_OUTPUT side-channel + command (STR #2858) + - The scheduler did not support "HostNameLookups" with all of + the boolean names (STR #2861) + - Fixed a compile problem with glibc 2.8 (STR #2860) + - The scheduler incorrectly filtered out queues with ACLs and + authentication. + - The PostScript filter did not support %%IncludeFeature lines + in the page setup section of each page (STR #2831) + - The scheduler did not generate printer-state events when the + default printer was changed (STR #2764) + - cupstestppd incorrectly reported a warning about the PPD format + version in some locales (STR #2854) + - cupsGetPPD() and friends incorrectly returned a PPD file for + a class with no printers. + - The member-uris values for local printers in a class returned + by the scheduler did not reflect the connected hostname or + port. + - The CUPS PHP extension was not thread-safe (STR #2828) + - The scheduler incorrectly added the document-format-default + attribute to the list of "common" printer attributes, which + over time would slow down the printing system (STR #2755, + STR #2836) + - The cups-deviced and cups-driverd helper programs did not set + the CFProcessPath environment variable on Mac OS X (STR #2837) + - "lpstat -p" could report the wrong job as printing (STR #2845) + - The scheduler would crash when some cupsd.conf directives + were missing values (STR #2849) + - The web interface "move jobs" operation redirected users to + the wrong URL (STR #2815) + - The Polish web interface translation contained errors + (STR #2815) + - The scheduler did not report PostScript printer PPDs with + filters as PostScript devices. + - The scheduler did not set the job document-format attribute + for jobs submitted using Create-Job and Send-Document. + - cupsFileTell() did not work for log files opened in append + mode (STR #2810) + - The scheduler did not set QUERY_STRING all of the time + for CGI scripts (STR #2781, STR #2816) + - The scheduler now returns an error for bad job-sheets + values (STR #2775) + - Authenticated remote printing did not work over domain + sockets (STR #2750) + - The scheduler incorrectly logged errors for print filters + when a job was canceled (STR #2806, #2808) + - The scheduler no longer allows multiple RSS subscriptions + with the same URI (STR #2789) + - The scheduler now supports Kerberized printing with + multiple server names (STR #2783) + - "Satisfy any" did not work in IPP policies (STR #2782) + - The CUPS imaging library would crash with very large + images - more than 16Mx16M pixels (STR #2805) + - The PNG image loading code would crash with large images + (STR #2790) + - The scheduler did not limit the total number of filters. + - The scheduler now ensures that the RSS directory has + the correct permissions. + - The RSS notifier did not quote the feed URL in the RSS + file it created (STR #2801) + - The web interface allowed the creation and cancellation + of RSS subscriptions without a username (STR #2774) + - Increased the default MaxCopies value on Mac OS X to + 9999 to match the limit imposed by the print dialog. + - The scheduler did not reject requests with an empty + Content-Length field (STR #2787) + - The scheduler did not log the current date and time and + did not escape special characters in request URIs when + logging bad requests to the access_log file (STR #2788) + + +CHANGES IN CUPS V1.3.7 + + - CVE-2008-0047: cgiCompileSearch buffer overflow (STR #2729) + - CVE-2008-1373: CUPS GIF image filter overflow (STR #2765) + - Updated the "make check" tests to do a more thorough + automated test. + - cups-driverd complained about missing directories (STR + #2777) + - cupsaddsmb would leave the Samba username and password on + disk if no Windows drivers were installed (STR #2779) + - The Linux USB backend used 100% CPU when a printer was + disconnected (STR #2769) + - The sample raster drivers did not properly handle SIGTERM + (STR #2770) + - The scheduler sent notify_post() messages too often on + Mac OS X. + - Kerberos access to the web interface did not work + (STR #2748) + - The scheduler did not support "AuthType Default" in IPP + policies (STR #2749) + - The scheduler did not support the "HideImplicitMembers" + directive as documented (STR #2760) + - "make check" didn't return a non-zero exit code on + error (STR #2758) + - The scheduler incorrectly logged AUTH_foo environment + variables in debug mode (STR #2751) + - The image filters inverted PBM files (STR #2746) + - cupsctl would crash if the scheduler was not running + (STR #2741) + - The scheduler could crash when printing using a port + monitor (STR #2742) + - The scheduler would crash if PAM was broken (STR #2734) + - The image filters did not work with some CMYK JPEG files + produced by Adobe applications (STR #2727) + - The Mac OS X USB backend did not work with printers that + did not report a make or model. + - The job-sheets option was not encoded properly (STR #2715) + - The scheduler incorrectly complained about missing LSB + PPD directories. + + +CHANGES IN CUPS V1.3.6 + + - Documentation updates (STR #2646, STR #2647, STR #2649) + - Fixed a problem with the web interface "Use Kerberos + Authentication" check box (STR #2703) + - The scheduler unconditionally overwrote the printer-state- + message with "process-name failed" when a filter or backend + failed, preventing a useful error message from being shown + to the user. + - Policies on CUPS-Move-Job didn't work as expected (STR + #2699) + - The configure script only supported D-BUS on Linux + (STR #2702) + - The scheduler did not support (STR #2701) + - The scheduler did not reset the job-hold-until attribute + after a job's hold time was reached. + - The scheduler did not support printer supply attributes + (STR #1307) + - The Kerberos credentials provided by some Windows KDCs + were still too large - now use a dynamic buffer to + support credentials up to 64k in size (STR #2695) + - Printing a test page from the web interface incorrectly + defaulted to the "guest" user (STR #2688) + - The cupsEncodeOptions2() function did not parse multiple- + value attribute values properly (STR #2690) + - The scheduler incorrectly sent printer-stopped events for + status updates from the print filters (STR #2680) + - The IPP backend could crash when handling printer errors + (STR #2667) + - Multi-file jobs did not print to remote CUPS servers + (STR #2673) + - The scheduler did not provide the Apple language ID to + job filters. + - Kerberos authentication did not work with the web + interface (STR #2606, STR #2669) + - The requesing-user-name-allowed and -denied functionality + did not work for Kerberos-authenticated usernames (STR + #2670) + - CUPS didn't compile on HP-UX 11i (STR #2679) + - cupsEncodeOptions2() did not handle option values like + "What's up, doc?" properly. + - Added lots of memory allocation checks (Fortify) + - The scheduler would crash if it was unable to add a job + file (Fortify) + - ppdOpen*() did not check all memory allocations (Coverity) + - ippReadIO() did not check all memory allocations (Coverity) + - The PostScript filter did not detect read errors (Coverity) + - The scheduler did not check for a missing job-sheets-completed + attribute when sending an event notification (Coverity) + - "Set Printer Options" might not work with raw queues (Coverity) + - cupsRasterInterpretPPD() could crash on certain PostScript + errors (Coverity) + - The USB backend did not check for back-channel support + properly on all systems (Coverity) + - Fixed memory leaks in the GIF and PNM image loading code + (Coverity) + - Removed some dead code in the CUPS API and scheduler (Coverity) + - Fixed two overflow bugs in the HP-GL/2 filter (Coverity) + - Fixed another ASN1 string parsing bug (STR #2665) + - The RSS notifier directory was not installed with the + correct permissions. + - The standard CUPS backends could use 100% CPU while waiting + for print data (STR #2664) + - Filename-based MIME rules did not work (STR #2659) + - The cups-polld program did not exit if the scheduler crashed + (STR #2640) + - The scheduler would crash if you tried to set the port-monitor + on a raw queue (STR #2639) + - The scheduler could crash if a polled remote printer was + converted to a class (STR #2656) + - The web interface and cupsctl did not correctly reflect + the "allow printing from the Internet" state (STR #2650) + - The scheduler incorrectly treated MIME types as case- + sensitive (STR #2657) + - The Java support classes did not send UTF-8 strings to + the scheduler (STR #2651) + - The CGI code did not handle interrupted POST requests + properly (STR #2652) + - The PostScript filter incorrectly handled number-up when + the number of pages was evenly divisible by the number-up + value. + - The PDF filter incorrectly filtered pages when page-ranges + and number-up were both specified (STR #2643) + - The IPP backend did not handle printing of pictwps files + to a non-Mac CUPS server properly. + - The scheduler did not detect network interface changes + on operating systems other than Mac OS X (STR #2631) + - The scheduler now logs the UNIX error message when it + is unable to create a request file such as a print job. + - Added support for --enable-pie on Mac OS X. + + +CHANGES IN CUPS V1.3.5 + + - The SNMP backend did not check for negative string + lengths (STR #2589) + - The scheduler incorrectly removed auth-info attributes, + potentially leading to a loss of all options for a job. + - The scheduler stopped sending CUPS browse packets on a + restart when using fixed addresses (STR #2618) + - Fixed PDF filter security issues (CVE-2007-4352 + CVE-2007-5392 CVE-2007-5393) + - Changing settings would always change the DefaultAuthType + and Allow lines (STR #2580) + - The scheduler would crash when submitting an undefined + format file from Samba with LogLevel debug2 (STR #2600) + - The scheduler did not use poll() when epoll() was not + supported by the running kernel (STR #2582) + - Fixed a compile problem with Heimdal Kerberos (STR #2592) + - The USB backend now retries connections to a printer + indefinitely rather than stopping the queue. + - Printers with untranslated JCL options were not exported + to Samba correctly (STR #2570) + - The USB backend did not work with some Minolta USB + printers (STR #2604) + - The strcasecmp() emulation code did not compile (STR + #2612) + - The scheduler would crash if a job was sent to an empty + class (STR #2605) + - The lpc command did not work in non-UTF-8 locales (STR + #2595) + - Subscriptions for printer-stopped events also received + other state changes (STR #2572) + - cupstestppd incorrectly reported translation errors for + the "en" locale. + - ppdOpen() did not handle custom options properly when the + Custom attribute appeared before the OpenUI for that + option. + - The scheduler could crash when deleting a printer or + listing old jobs. + - The Mac OS X USB backend did not allow for requeuing of + jobs submitted to a class. + - lpmove didn't accept a job ID by itself. + - The scheduler incorrectly removed job history information + for remote print jobs. + - The scheduler incorrectly sent the + "com.apple.printerListChanged" message for printer state + changes. + - The PostScript filter drew the page borders (when enabled) + outside the imageable area. + - The LPD and IPP backends did not default to the correct + port numbers when using alternate scheme names. + - The scheduler incorrectly deleted hardwired remote + printers on system sleep. + - The scheduler would abort if a bad browse protocol name + was listed in the cupsd.conf file. + - The online cupsd.conf help file incorrectly showed + "dns-sd" instead of "dnssd" for Bonjour sharing. + - The scheduler could crash changing the port-monitor value. + - The scheduler generated CoreFoundation errors when run as + a background process. + - When printing with number-up > 1, it was possible to get + an extra blank page. + + +CHANGES IN CUPS V1.3.4 + + - Documentation updates (STR #2560, STR #2563, STR #2569) + - CUPS now maps the "nb" locale to "no" on all platforms + (STR #2575) + - CUPS did not work with a Windows 2003 R2 KDC (STR #2568) + - ippReadIO() could read past the end of a buffer (STR + #2561) + - The scheduler would crash on shutdown if it was unable + to create a Kerberos context. + - Multiple AuthTypes in cupsd.conf did not work (STR + #2545) + - The snmp.conf file referenced the wrong man page (STR + #2564) + - The cupsaddsmb program didn't handle domain sockets + properly (STR #2556) + - The scheduler now validates device URIs when adding + printers. + - Updated httpSeparateURI() to support hostnames with + the backslash character. + - Updated the Japanese localization (STR #2546) + - The parallel backend now gets the current IEEE-1284 + device ID string on Linux (STR #2553) + - The IPP backend now checks the job status at + variable intervals (from 1 to 10 seconds) instead + of every 10 seconds for faster remote printing + (STR #2548) + - "lpr -p" and "lpr -l" did not work (STR #2544) + - Compilation failed when a previous version of CUPS + was installed and was included in the SSL include + path (STR #2538) + - The scheduler did not reject requests with charsets + other than US-ASCII or UTF-8, and the CUPS API + incorrectly passed the locale charset to the scheduler + instead of UTF-8 (STR #2537) + - cups-deviced did not filter out duplicate devices. + - The AppleTalk backend incorrectly added a scheme + listing when AppleTalk was disabled or no printers + were found. + - The PostScript filter generated N^2 copies when the + printer supported collated copies and user requested + reverse-order output. + - The scheduler did not reprint all of the files in a + job that was held. + - The scheduler did not update the printcap file after + removing stale remote queues. + - The cupsd.conf man page incorrectly referenced + "AuthType Kerberos" instead of "AuthType Negotiate". + + +CHANGES IN CUPS V1.3.3 + + - The scheduler did not use the attributes-natural-language + attribute when passing the LANG environment variable to + cups-deviced or cups-driverd. + - The scheduler did not use the printer-op-policy when + modifying classes or printers (STR #2525) + - The auth-info-required attribute was not always updated + for remote queues that required authentication. + - The German web interface localization contained errors + (STR #2523) + - The Swedish localization contained errors (STR #2522) + + +CHANGES IN CUPS V1.3.2 + + - The 1.3.1 release was incorrectly created from the + 1.4.x source tree (STR #2519) + - Added support for 32/64-bit libraries on HP-UX + (STR #2520) + - The scheduler incorrectly used portrait as the default + orientation (STR #2513) + - The scheduler no longer writes the printcap file for + every remote printer update (STR #2512) + - Remote raw printing with multiple copies did not work + (STR #2518) + - Updated the configure script to require at least autoconf + 2.60 (STR #2515) + - Some gzip'd PPD files were not read in their entirety + (STR #2510) + + +CHANGES IN CUPS V1.3.1 + + - Documentation updates. + - The USB backend on Mac OS X could hang if the driver and + printer did not match. + - Delegated Kerberos credentials were not working. + - "make distclean" incorrectly removed the edit-config.tmpl + files (STR #2508) + - Fix compile problem on HP-UX (STR #2501) + - The cupstestppd utility now tests for resolutions greater + than 99999 DPI to detect a missing "x" between the X and Y + resolutions. + - Fixed many problems in the various translations and added + a new "checkpo" utility to validate them. + - The cupstestppd utility now tests the custom page size code + for CUPS raster drivers. + - cupsLangDefault() did not attempt to return a language that + was supported by the calling application. + - If a remote printer stopped while a job was being sent, the + local queue would also get stopped and the job re-queued, + resulting in duplicate prints in some cases. + - A few Apple-specific job options needed to be omitted when + printing a banner page. + - The new peer credential support did not compile on FreeBSD + (STR #2495) + - Direct links to help files did not set the current section + so the table-of-contents was not shown. + - The configure script did not support --localedir=foo (STR #2488) + - The backends were not displaying their localized messages. + - CUPS-Authenticate-Job did not require Kerberos authentication + on queues protected by Kerberos. + - The Zebra ZPL driver did not work with Brady label printers + (STR #2487) + - Norwegian wasn't localized on Mac OS X. + - getnameinfo() returns an error on some systems when DNS is + not available, leading to numerous problems (STR #2486) + - The cupsfilter command did not work properly on Mac OS X. + - The scheduler makefile contained a typo (STR #2483) + - The TBCP and BCP port monitors did not handle the trailing + CTRL-D in some PostScript output properly. + - Fixed the localization instructions and German template for + the "Find New Printers" button (STR #2478) + - The web interface did not work with the Chinese localization + (STR #2477) + - The web interface home page did not work for languages that + were only partially localized (STR #2472) + - Updated the Spanish web interface localization (STR #2473) + - ppdLocalize() did not work for country-specific localizations. + + +CHANGES IN CUPS V1.3.0 + + - The scheduler did not handle out-of-file conditions + gracefully when accepting new connections, leading to + heavy CPU usage. + - The scheduler did not detect ServerBin misconfigurations + (STR #2470) + - "AuthType Default" did not work as expected when the + "DefaultAuthType foo" line appeared after it in the + cupsd.conf file. + - The on-line help did not describe many common printing + options (STR #1846) + - The IPP backend did not return the "auth required" status + when printing to a Kerberos-protected queue. + - The scheduler was not looking in the correct directories + for LSB PPD files (STR #2464) + - Changed references to ESP Ghostscript to GPL Ghostscript + (STR #2463) + - The PostScript filter did not cleanly terminate when + the job was canceled or stopped. + - Fixed generation of Kerberos credentials for remote + printing. Note that this requires a recent version of + MIT Kerberos with a working krb5_cc_new_unique() + function or Heimdal Kerberos. + - Added Portuguese and updated Italian message catalogs. + + +CHANGES IN CUPS V1.3rc2 + + - Added more range checking to the pdftops filter. + - The scheduler would crash if a remote IPP queue was stopped + (STR #2460) + - The scheduler did not allow "DefaultAuthType None". + + +CHANGES IN CUPS V1.3rc1 + + - Updated the German localization (STR #2443) + - cupsAdminGetServerSettings() did not handle properly. + - When lprm and cancel are run with no job ID, they now will + cancel the first stopped job if no pending or processing + jobs are left in the queue. + - The scheduler now logs successful print jobs, filter + failures, and the job file types at the default log + level (STR #2458) + - The scheduler now logs the usernames it is using for + authorization at LogLevel debug instead of debug2 (STR #2448) + - Added Intellitech Intellibar and Zebra CPCL PPDs to the list + of installed PPDs. + - Added 6" and 8" wide label sizes for the Zebra ZPL Label + Printer driver (STR #2442) + - The cupsaddsmb program and web interface now support + exporting of 64-bit Windows drivers, when available + (STR #2439) + - Moving a job that was printing did not stop the job on the + original printer (STR #2262) + - The cups-lpd mini-daemon did not work on Mac OS X server. + - Added httpGetAuthString() and httpSetAuthString() APIs to get + and set the current (cached) authorization string to use for + HTTP requests. + - Updated the default cupsd.conf policy to list the + "administrative" operations separately from the "printer + control" operations so that it is easier to define a + group of users that are "printer operators". + - The web interface now pulls the default cupsd.conf file + from cupsd.conf.default in the CUPS config directory. + - Added a help file for using Kerberos with CUPS. + - The scheduler now strips the "@KDC" portion of Kerberos + usernames since those usernames typically do not appear in + the group membership lists used by CUPS. + - cupsMarkOptions() could (incorrectly) leave multiple option + choices marked. + - Backends could (incorrectly) run as root during discovery + (STR #2454) + - Avahi is now supported for DNS-SD (Bonjour) printer sharing + (STR #2455) + - The default cupsd.conf file had typos and old operation names + (STR #2450) + - The scheduler now erases authentication cache files using the + 7-pass US DoD algorithm. + - Delegated Kerberos credentials (proxy authentication) did not + work. + - The filter makefile did not optimize the libcupsimage.2.dylib + with a sectorder file. + - The IPP backend incorrectly wrote an empty printer message + when processing the "none" state reason. + - The USB backend could deadlock on Mac OS X while performing + a side-channel command. + - The scheduler did not prevent remote queues from being + shared/published. + - The scheduler did not remove the temporary request file on + authentication errors. + - ppdLocalizeIPPReason() did not handle "scheme:" schemes or + "file" URLs. + - ppdLocalizeIPPReason() was not exported on Mac OS X. + + +CHANGES IN CUPS V1.3b1 + + - Copyright updates - CUPS is now owned by Apple Inc. + - Documentation updates (STR #1775, STR #2027, STR #2130, + STR #2131, STR #2263, STR #2356, STR #2397) + - Added new cupsfilter utility (STR #1734) + - Added new job-printer-state-message and + job-printer-state-reasons attributes to jobs (STR #2418) + - Added LDAP+SSL support (STR #1967) + - CUPS now supports authentication via peer credentials + over domain sockets (STR #2242, STR #2277) + - The CUPS sample driver PPDs are now generated by the PPD + compiler and include all of the localized languages by + default (STR #2164) + - You can now specify "AuthType Default" in the cupsd.conf + file to use the default authentication defined by the + DefaultAuthType directive. + - The SNMP backend no longer adds a default Address line + when none is specified in the snmp.conf file; this allows + the backend to be easily disabled as needed (STR #2434) + - Added a new cupsctl command for doing basic changes to + the cupsd.conf file (STR #1777) + - Added a new ppdLocalizeIPPReason() function to get the + localized text/URI for a given IPP reason keyword for a + driver. + - Removed the deskjet2.ppd driver, as it only worked with + a very small subset of HP DeskJet printers and was + confusing to users. The rastertohp driver still + supports the deskjet2.ppd options for existing queues. + - The scheduler did not add a trailing banner page if a + client did not specify the last document in a job (STR + #1711) + - The scheduler did not report Bonjour shared printers as + remote printers (STR #2384) + - Added new -R and -W options to the cupstestppd program + for greater control over the testing of PPDs. + - Added a new cupsGetServerPPD() function for getting + an available PPD from the server (STR #2334) + - Added a new cupsDoIORequest() function for reading + and writing files via IPP requests (STR #2334) + - Added a new CUPS_GET_PPD operation for getting an + available PPD file on the server (STR #2334) + - CUPS_GET_PPDS now reports multiple ppd-product values + based on the PPD ModelName and Product strings (STR + #2334, STR #2383) + - CUPS_GET_PPDS now reports the PSVersion attributes + from a PPD file in the ppd-psversion attribute + (STR #2334) + - CUPS_GET_PPDS now reports the cupsModelNumber attribute + from a PPD file in the ppd-model-number attribute (STR + #2383) + - CUPS_GET_PPDS now reports a driver type string in the + ppd-type attribute based on the cupsFax and cupsFilter + attributes in a PPD file (STR #2383) + - Added a new printer attribute called "cups-version" + which reports the version of CUPS that is running + (STR #2240) + - backendRunLoop() now aborts immediately on SIGTERM + if no data has been written yet (STR #2103) + - Due to poor IPP support from the vendors, the SNMP + backend no longer tries IPP connections; instead, + it now uses a lookup file with fallback to port 9100 + (socket://address) and 515 (lpd://address) printing + (STR #2035, STR #2354) + - The scheduler now recreates the CUPS log directory as + needed (STR #2353) + - cupsLangDefault() now maps new-style Apple locale names + to the traditional ll_CC form (STR #2357) + - Add new cupsArrayNew2() API to support hashed lookups + of array elements (STR #2358) + - ppdConflicts() optimizations (STR #2358) + - The cupstestppd program now tests for existing filters, + icons, profiles, and dialog extensions (STR #2326) + - The web interface no longer lists new printers on the + main administration page. Instead, a new "List Available + Printers" button is provided that shows a separate page + with the list of printers. + - The web interface now supports setting the banner and + policy options on raw printers and classes (STR #2238) + - The socket backend now reads any pending back-channel + data before shutting down the socket (STR #2325) + - Added a new ErrorPolicy directive in the cupsd.conf + file (STR #1871) + - Printers that use JCL options are now exported to Samba + correctly (STR #1985) + - The IPP backend now relays printer-state-message values + from the server to the client (STR #2109) + - Added support for the PWG printer-alert and + printer-alert-description attributes (STR #2088) + - Added support for LPD "stream" mode (STR #2036) + - The scheduler now reports the PostScript product string + from PPD files in CUPS-Get-PPDs responses (STR #1900) + - Raw printing with queues pointing to the file pseudo- + device and multiple files and/or banners now works (STR + #1933) + - Added new public cupsAdminGetServerSettings() and + cupsAdminSetServerSettings() APIs. + - Added new "makebuttons" script in the "tools" directory + for creating web interface buttons (STR #2231) + - Added support for DNS-SD (aka "Bonjour") printer sharing + (STR #1171) + - Job operations (cancel, hold, release, etc.) from the + web interface now return back to the original page (STR + #2239) + - The classes or printers list is now shown after a + successful deletion from the web interface (STR #1999) + - The default configuration now allows browse packets from + any address (STR #2008) + - The web interface now provides an "allow printing from the + Internet" check box (STR #1897) + - The notify-events-default and + notify-lease-duration-default attributes can now be set + (STR #1671) + - Server-side default options are now sent to clients when + the "printer-defaults" attribute group is requested (STR + #1923) + - Added support for Linux "relro" linker option (STR #1614) + - CUPS now validates the number-up option value (STR #1329) + - The on-line help now provides better search capabilities + (STR #1701) + - The web interface "Add This Printer" button now allows you + to change the printer name, description, and location + (STR #1646) + - Added support for Mac OS X authorization services + (STR #2206) + - Added support for driver-specific pre-filters (STR #2108) + - Added a new side-channel API for drivers and backends + for basic device control and information queries (STR + #1898) + - The scheduler now uses poll(), epoll(), or /dev/kqueue + instead of select() when possible (STR #1261) + - Added new cupsArrayGetIndex() and cupsArrayGetInsert() + functions to get the current index and insertion + positions of an array. + - Added a new --with-max-copies configure option (STR + #2090) + - Added new cupsRemoveDest() and cupsSetDefaultDest() + functions. + - Added support for cupsPJLCharset attribute in PPD files + which specifies the character set that is used in PJL + strings (STR #1969) + - Moved the definition of the (private) _http_s structure + to http-private.h; code that directly accesses the + http_t members will no longer compile! + - Added support for setting the document-format-default + attribute on a per-printer basis. + - Added support for IntelliBar label printers. diff --git a/CHANGES-1.4.txt b/CHANGES-1.4.txt new file mode 100644 index 0000000000..09326b2886 --- /dev/null +++ b/CHANGES-1.4.txt @@ -0,0 +1,843 @@ +CHANGES-1.4.txt +--------------- + +CHANGES IN CUPS V1.4.8 + + - The scheduler would delete job data files when restarted (STR #3880) + - The network backends could crash if a printer returned a value of 0 + for the maximum capacity for a supply (STR #3875) + + +CHANGES IN CUPS V1.4.7 + + - Documentation changes (STR #3710, STR #3720, STR #3745, STR #3750, + STR #3757, STR #3758, STR #3782, STR #3826, STR #3829, STR #3837) + - Web interface fixes (STR #3412, STR #3345, STR #3455, STR #3707, + STR #3755, STR #3769, STR #3783) + - Configure script fixes (STR #3659, STR #3691) + - Compilation fixes (STR #3718, STR #3771, STR #3774) + - The imageto* filters could crash with bad GIF files (STR #3867) + - The scheduler might leave old job data files in the spool directory + (STR #3795) + - CUPS did not work with locales using the ASCII character set + (STR #3832) + - httpAddrString() did not return a URI-style IPv6 numeric address + (STR #3814) + - Fixed an issue when reading compressed CUPS raster streams (STR #3812) + - Fixed an issue with PostScript printer auto-configuration (STR #3443) + - Fixed some compatibility issues with the libusb-based USB backend + (STR #3799) + - The network backends no longer try to collect SNMP supply and status + information for raw queues (STR #3809) + - The DBUS notifier did not report job state changes (STR #3805) + - The scheduler did not always report that the "normal" print-quality + value was supported (STR #3803) + - The gziptoany filter did not report the correct error if it was unable + to write the uncompressed document to the next filter or backend in + the chain (STR #3797) + - The Epson and Oki 9-pin drivers had a bad resolution option + (STR #3798) + - The scheduler did not always register the correct default ICC profile + on Mac OS X. + - The scheduler did not use the job owner when authorizing access for + the CUPS-Get-Document operation, preventing non-admins from accessing + their own jobs. + - CUPS did not work with some printers that incorrectly implemented the + HTTP/1.1 standard (STR #3778, STR #3791) + - The scheduler did not retry fax jobs properly. + - The scheduler now recognizes an empty cupsCommands PPD keyword as + meaning that CUPS commands are not supported for a printer (STR #3773) + - Fixed a crash bug in the scheduler when the application/octet-stream + MIME type was not defined (STR #3690) + - Polled printers were advertised more slowly than necessary (STR #3574) + - cupsResolveConflicts() did not handle resolving multiple UIConstraints + issues (STR #3705) + - The SetEnv and PassEnv directives had no effect (STR #3664) + - The libusb-based USB backend printed slowly to the LaserJet 1300 and + other printers (STR #3405) + - "lp" and "lpr" failed to print with Kerberos enabled (STR #3768) + - The cupsctl program now displays an error if you try to directly set + the Port or Listen directives (STR #3749) + - PPD files with "*JobPatchFile: bla" no longer fail to load in relaxed + conformance mode (STR #3747) + - The scheduler generated a bad notify-text string for printer state + change notifications (STR #3739) + - The scheduler incorrectly updated printers.conf when it really needed + to update classes.conf or remote.cache (STR #3726) + - Hardwired remote printers with options did not work (STR #3717) + - Accessing the CUPS web interface using a CNAME-based hostname would + sometimes fail due to redirection to the actual hostname (STR #3701) + - Subscription events had a misspelled attribute (STR #3693) + - "make check" failed if LC_MESSAGES was set (STR #3765) + - Fixed the configure script to always look for the pkg-config script + (STR #3761) + - The scheduler now only looks up interface hostnames if HostNameLookups + are enabled (STR #3737) + - Fixed a compilation problem on DragonFly BSD (STR #3738) + - The default PageLogFormat value had the username and job ID swapped + from CUPS 1.3.x (STR #3727) + - The scheduler could crash if a browsed printer times out while a job + is printing (STR #3754) + - The scheduler incorrectly mapped custom page sizes to standard sizes + (STR #3764) + - cupsfilter and pstops did not map IPP attributes to PPD options due to + a change in cupsMarkOptions (STR #3756) + - The scheduler did not always show the most recent status message from + the print filters (STR #3731) + - The PostScript filter did not apply the mirror and number-up options + properly, leading to offset and clipped output (STR #3732) + - The network backends always reported "low toner" or "out of toner" + states, even for inkjet printers (STR #3733) + + +CHANGES IN CUPS V1.4.6 + + - Fixed a "make check" issue on Solaris (STR #3729) + - Regression: The pstops filter did not support landscape printing of + PostScript files (STR #3722) + - The scheduler killed retried (fax) jobs after restarting them + (STR #3697) + - The cupsAdminSetServerSettings() function disabled sharing when + debug logging was enabled (STR #3712) + + +CHANGES IN CUPS V1.4.5 + + - Documentation fixes (STR #3542, STR #3650) + - Localization fixes (STR #3635, STR #3636, STR #3647, STR #3666) + - Security: Fixed a memory corruption bug reported in CVE-2010-2941 + (STR #3648) + - The CUPS API incorrectly mapped the HTTP_UNAUTHORIZED status to the + IPP_NOT_AUTHORIZED status code, when IPP_NOT_AUTHENTICATED would be + the correct mapping (STR #3684) + - The scheduler would restart jobs while shutting down (STR #3679) + - Fixed a PPD loader bug that could cause a crash in cupsd (STR #3680) + - Improved the mapping of non-standard PPD and PWG names (STR #3671) + - The scheduler did not initialize Kerberos in all cases (STR #3662) + - cupsAdminSetServerSettings duplicated Listen and Order lines + (STR #3645) + - Added DeviceN colorspace support to the CUPS Raster format (STR #3419) + - ppdMarkDefaults() did not clear the marked field of the previous + choices (STR #3642) + - The serial backend would not allow a raw job to be canceled + (STR #3649) + - The socket backend could go into an infinite loop with certain + printers (STR #3622) + - Setting the PRINTER or LPDEST environment variables to "name/instance" + did not work (STR #3485) + - The scheduler did not handle the JobRetryLimit setting properly + (STR #3466) + - The lpstat command always showed a remote job ID of 0 for shared + printers (STR #3627) + - Increased the write timeout for the libusb-based USB backend to 5 + minutes (STR #3595) + - The libusb-base USB backend did not check whether the printer has a + serial number (STR #3590) + - The lpadmin command did not support setting of custom option values + (STR #3631) + - The lpadmin command did not support setting of the location or + description of a class (STR #3613) + - The cupsaddsmb command did not give up after too many failed attempts + (STR #3615) + - The CUPS library no longer uses certain problematic ctype macros that + change based on the locale's character set. + - PJL value substitution of more than 9 values was broken (STR #3621) + - Custom options with missing string values caused ppdEmit* to segfault + (STR #3620) + - Fixed an issue with the Italian version of the web interface + (STR #3624) + - Fixed the Solaris SMF configuration file for cups-lpd (STR #3611) + - The scheduler did not set the notify-subscribed-event attribute when + delivering printer-added or printer-modified events (STR #3608) + - The mailto notifier could get into an infinite loop (STR #3609) + - Date/time information was not shown in banner pages. + - Relational operators were broken in #if/#elif/#else/#endif expressions + for the PPD compiler. + - Moving a job via the web interface failed without asking for + authentication (STR #3559) + - The scheduler now clears the printer-state-reasons when the driver is + changed (STR #3570) + - The web interface did not allow a user to change the driver + (STR #3537, STR #3601) + - The scheduler was not setting the PATH_INFO environment variable when + needed (STR #3600) + - The scheduler incorrectly set the CUPSD_AUTH_TYPE environment + variable instead of AUTH_TYPE (STR #3599) + - Fixed a buffer overrun in the PPD compiler (STR #3594) + - Fixed some additional IPP job template attribute mapping issues in the + scheduler. + + +CHANGES IN CUPS V1.4.4 + + - Documentation updates (STR #3453, STR #3527, STR #3528, STR #3529) + - Security: The fix for CVE-2009-3553 was incomplete (STR #3490) + - Security: The texttops filter did not check the results of allocations + (STR #3516) + - Security: The web admin interface could disclose the contents of + memory (STR #3577) + - Security: CUPS could overwrite files as root in directories owned or + writable by non-root users (STR #3510) + - The cups-config utility did not return the correct linker options on + AIX (STR #3587) + - Fixed some IPP conformance issues with the scheduler's + ippget-event-life, operations-supported, output-bin, and sides + attributes (STR #3554) + - The OpenSSL interfaces have been made thread-safe and the GNU TLS + interface is explicitly forbidden when threading is enabled + (STR #3461) + - Fixed an IPP conformance issue with the scheduler's Send-Document + implementation (STR #3514) + - Added additional validation checks for the 1284 device ID (STR #3534) + - Fixed a problem with the RPM spec file (STR #3544) + - The lpstat command did not limit the job list to the specified + printers (STR #3541) + - The cupsfilter command did not set the RIP_MAX_CACHE environment + variable (STR #3531) + - Fixed support for media-col and page size variants (STR #3394) + - The PostScript filter did not support all media selection options for + the first page (STR #3525) + - The scheduler did not always remove job control files (STR #3425) + - The scheduler could crash on restart if classes were defined + (STR #3524) + - The scheduler no longer looks up network interface hostnames by + default on Mac OS X (STR #3523) + - ippWriteIO did not write collection (member) attributes properly in + all cases (STR #3521) + - The "cupsctl --remote-any" and corresponding web interface check box + (allow printing from the Internet) did not work reliably (STR #3520) + - The lpq and lpr commands would sometimes choose different default + printers (STR #3503) + - cupsDo*Request did not flush error text, leading to multiple issues + (STR #3325, STR #3519) + - cupsDoAuthentication did not cancel password authentication after 3 + failures (STR #3518) + - Fixed several LDAP browsing bugs (STR #3392) + - The Dymo driver did not support copies (STR #3457) + - The scheduler did not update the classes.conf file when deleting a + printer belonging to a class (STR #3505) + - The lppasswd command did not use localized password prompts + (STR #3492) + - The socket backend no longer waits for back-channel data on platforms + other than Mac OS X (STR #3495) + - The scheduler didn't send events when a printer started accepting or + rejecting jobs (STR #3480) + - The web interface now includes additional CSRF protection (STR #3498) + + +CHANGES IN CUPS V1.4.3 + + - SECURITY: The scheduler could try responding on a closed client + connection, leading to a crash (STR #3200) + - SECURITY: The lppasswd program allowed the localization files to be + overridden when running in setuid mode (STR #3482) + - Localization updates (STR #3352, STR #3409, STR #3422, STR #3452, + STR #3473, STR #3502) + - Documentation updates (STR #3451, STR #3504) + - The IPP backend now sets the printer-state-message to "Ready to + print." at the end of a successful job (STR #3460) + - The PPD compiler did not correctly add the manufacturer to the output + filename when using the "-m" option (STR #3469) + - The IPP backend did not handle authentication properly for the Get- + Printer-Attributes operation (STR 3458) + - Getting SNMP values larger than 127 bytes did not work. + - IPP conformance: Get-Jobs has a default value for requested-attributes + (STR #3383) + - cupsPrintFiles() did not report all errors (STR #3449) + - cupsAddDest() could read freed memory (STR #3448) + - The DBUS notifier did not build (STR #3447) + - The scheduler would crash when an active printer was deleted. + - The snmp backend did not work with some printers (STR #3413) + - The web interface did not show the conflicting values when setting + options (STR #3440) + - Setting options in the web interface did not always work (STR #3439) + - The scheduler did not use the Get-Job-Attributes policy for a printer + (STR #3431) + - The scheduler added two job-name attributes to each job object + (STR #3428) + - CSS files would not print (STR #3442) + - The scheduler did not clean out completed jobs when PreserveJobHistory + was turned off (STR #3425) + - The web interface did not show completed jobs for a printer + (STR #3436) + - Authenticated printing did not always work when printing directly to + a remote server (STR #3435) + - The USB backend did not work on Solaris (STR #3423) + - cupstestppd didn't catch problems with JobPatchFile definitions + (STR #3421) + - The socket backend could crash if a SNMP string had a negative length. + - Fixed some termination issues with the USB backend on Mac OS X. + - The side-channel APIs did not handle interrupts properly. + - The network backends incorrectly cleared the media-empty-warning + state. + - The web interface did not allow users to successfully add serial + printers (STR #3391) + - cupsTempFd() did not work in some situations (STR #3382) + - Some C API headers were missing C++ wrapper logic. + - The PPD compiler did not localize single-language PPD options properly + (STR #3386) + - Modifying a printer from the web interface sometimes caused the wrong + driver to be selected (STR #3418) + - The scheduler did not handle out-of-memory conditions properly when + loading a job (STR #3407) + - When adding printers from the web interface, the dynamic updates of + the device list made it hard to pick a device (STR #3406) + - Fixed a typo in the web interface admin page template (STR 3403) + - The web interface did not preserve the "printer is shared" state when + modifying a printer (STR #3390) + - The PPD compiler incorrectly inserted translations of empty strings + (STR #3411) + - The scheduler did not reset the SIGPIPE handler of child processes + (STR #3399) + - cupsGetNamedDest() incorrectly returned the default printer if the + named printer did not exist (STR #3397) + - Fixed a GNU TLS error handling bug (STR #3381) + + +CHANGES IN CUPS V1.4.2 + + - SECURITY: The CUPS web interface was vulnerable to several XSS and + HTTP header/body attacks via attribute injection (STR #3367, + STR #3401) + - Fixed localization errors (STR #3359, STR #3372, STR #3380, STR #3387) + - The documentation for classes.conf and printers.conf did not provide + the correct instructions for manual changes (STR #3351) + - The scheduler did not always rebuild printer cache files when the + driver was changed (STR #3356) + - The documentation makefile failed to install localizations when using + newer versions of Bash (STR #3360) + - The configure script did not use the --with-xinetd value for the + default LPD configuration path (STR #3347) + - The configure script incorrectly required glib for DBUS support + (STR #3346) + - The cupstestppd program incorrectly reported filters with bad + permisssions as missing (STR #3363) + - The cups.desktop file used the wrong locale names (STR #3358) + - cupsSideChannelRead() did not return an error for short reads. + - The installed PAM configuration file did not use the correct options + with the pam_unix2 module (STR #3313) + - The scheduler did not preserve default options that contained special + characters (STR #3340) + - The scheduler did not remove old pre-filters when updating a printer + driver (STR #3342) + - The HP/GL-2 filter did not check for early end-of-file (STR #3319) + - The USB backend did not compile on some platforms (STR #3332) + - cupsSideChannelSNMPWalk() could go into an infinite loop with broken + SNMP implementations. + + +CHANGES IN CUPS V1.4.1 + + - Documention fixes (STR #3296) + - SNMP supply levels and states were wrong for some printers. + - The IPP backend did not update the auth-info-required value. + - The libusb-based USB backend would hang at the end of the job + (STR #3315, STR #3318) + - DNS-SD registrations for raw queues had an empty "ty" key (STR #3299) + - The JPEG and BMP MIME type rules were broken (STR #3284) + - cupsGetNamedDest returned the default printer when the named + destination did not exist (STR #3285) + - The JobKillDelay was not triggered for canceled jobs (STR #3292) + - The PPD compiler could get in an infinite loop (STR #3293) + - The configure check for dns-sd.h was broken (STR #3297) + - The "Query Printer for Default Options" page did not go away if the + query job was held (STR #3302) + - Boolean options did not show up as selected in the web interface + (STR #3303) + - The scheduler did not cache or report driver information files + correctly, leading to a variety of issues (STR #3283, STR #3297, + STR #3305) + - cupsDoIORequest() did not abort on permanent errors (STR #3311) + - Modifying a class in the web interface did not work (STR #3312) + - BrowseLocalProtocols could be cleared when changing the sharing + setting (STR #3287) + - The scheduler could return an empty supported document format + (STR #3308) + - The PPD compiler generated invalid PPD files when the locale used + something other than "." for the decimal point (STR #3300) + - The IPP backend did not handle some non-comforming IPP printer + implementations (STR #3262) + - The scheduler leaked three file descriptors to each job filter + (STR #3263) + - The scheduler now uses a default CUPS-Get-Devices timeout of 15 + seconds (STR #3307) + + +CHANGES IN CUPS V1.4.0 + + - Localization updates (STR #3223, STR #3246, STR #3248, STR #3250) + - Documentation updates (STR #3225, STR #3230, STR #3242, STR #3260) + - The --with-pdftops configure option did not accept a full path to the + filter (STR #3278) + - The banner filter did not position the back side image correctly + (STR #3277) + - The dnssd backend could crash (STR #3272) + - The 1284 device ID sometimes contained trailing garbage (STR #3266) + - The USB backend returned different URIs for some printers than in + CUPS 1.3 (STR #3259) + - The scheduler did not do local job-hold-until processing for remote + queues (STR #3258) + - The scheduler did not try all possible SSL certificates on Mac OS X. + - The scheduler did not always remove a file descriptor when using the + kqueue interface (STR #3256) + - The scheduler did not protect against bad job control files in all + cases (STR #3253) + - The scheduler did not encode "+" in model names (STR #3254) + - The web interface didn't show the default options (STR #3244) + - The IPP and LPD backends needed print data before they would do an + SNMP query. + - Fixed a GNU TLS compatibility issue (STR #3231) + - Fixed a HTML error in the add and modify printer web interface + templates (STR #3229) + - The scheduler did not minimize the number of printer state events that + were generated by filter STATE: messages, which could lead to poor + performance. + - The USB backend on Mac OS X did not cleanly cancel a job. + - The network backends now set the connecting-to-device printer-state- + reasons value when looking up the address and copying the print data + for consistency. + - The scheduler now supports the com.apple.print.recoverable-warning + reason on all platforms. + + +CHANGES IN CUPS V1.4rc1 + + - The PPD compiler documentation was missing information on localization + (STR #3212) + - The IPP backend now reconnects after every request when talking to + printers that claim IPP support but only use HTTP/1.0. + - The PPD compiler crashed when both "Resolution" and "Group foo Option + Resolution" were specified in the .drv file. + - The PPD compiler's #if/#elif/#else/#endif did not work for undefined + variables (STR #3210) + - Static libraries could not be installed by a non-root user on systems + needing a ranlib program (STR #3209) + - The scheduler incorrectly always tried to copy Kerberos credentials + for print jobs. + - Updated the Spanish localization (STR #3204) + - The scheduler crashed when getting the default paper size from + libpaper (STR #3205, STR #3206) + - The PPD compiler now defines six variables: CUPS_VERSION, + CUPS_VERSION_MAJOR, CUPS_VERSION_MINOR, CUPS_VERSION_PATCH, + PLATFORM_NAME, and PLATFORM_ARCH (STR #3203) + - Fixed a whitespace skipping bug in cupsRasterInterpretPPD. + - The scheduler did not return HTTP 403 (Forbidden) for authenticated + users that were not authorized to do IPP operations (STR #3193) + - The scheduler did not report more than 8 Product strings from a PPD + file. Some PPD files have as many as 24. + - ppdOpen*() could crash if a keyword had no value string (something + that cupstestppd looks for...) + - cupsLangDefault() did not return the correct language on Mac OS X. + - The Mac OS X USB backend did not handle aborted or stalled pipe + conditions properly, which prevented drivers from ejecting partial + pages when a job was canceled or held. + + +CHANGES IN CUPS V1.4b3 + + - Documentation fixes (STR #3044, STR #3057, STR #3153, STR #3158, + STR #3173) + - Added complete localizations for German, Japanese, Polish, and + Russian and partial localizations for Chinese, Danish, Finnish, + French, Italian, Korean, Norwegian, Portuguese, and Swedish + (STR #3096, STR #3098, STR #3109, STR #3111, STR #3141) + - Updated the configure check for -fstack-protector (STR #3198) + - The network backends now correctly convert SNMP supply descriptions to + UTF-8 encoding as needed. + - The scheduler could crash when deleting an attribute (STR #3197) + - The cups-driverd program did not detect symlink loops (STR #3185) + - The EPSON 24-pin series driver should now feed the correct amount + (STR #2624) + - The scheduler now automatically logs the last N debug messages for + failed print jobs. + - You can now modify a raw print queue (STR #3133) + - Fixed a number of ppdi issues and added a unit test to validate that + ppdc + ppdi can generate and import the same data (STR #3152) + - Moving jobs in the web interface now shows an error if you only have + one printer or class added (STR #3094) + - Since classes have never truly supported the printer-error-policy + stuff added in CUPS 1.2, update the code to reflect the current + reality and support only the retry-current-job policy for now + (STR #3171) + - Revised the password callback support (STR #2953) + - ppdEmit*() did not choose between PageSize and PageRegion properly. + - Make some fairly substantial changes to the Kerberos support code so + that CUPS can work in multi-realm environments and does not require + delegatable credentials. Shared printing still requires delegation, + however "delegation by policy" can be enabled in the KDC to make this + all work. + - "AccessLogLevel actions" did not hide client-error-not-found errors. + - AP_FIRST_InputSlot did not work with number-up. + - cupsBackChannelRead() and cupsBackChannelWrite() could fail due to a + lack of kernel buffers. + - The IPP and LPD backends did not respond to side-channel requests + while copying print data to a temporary file. + - cupsWriteRequestData() flushed the output buffer unnecessarily, + causing reduced performance in some situations. + - If a CGI process died before sending its MIME headers, the request + would hang on the client. + - The printer/class/job search feature on the web interface did not + work (STR #3132) + - The scheduler did not write the printers out for classes. + - CUPS-Get-PPDs did not work properly when filtering by language, + product, or psversion (STR #3136) + - The scheduler now kills job filters when it takes more than 30 seconds + (configurable) to cancel or hold the job. + - The cupstestppd program did not validate the capitalization of + filenames in the PPD file. + - The cupstestppd program did not validate the PageSize and PageRegion + values. + - The cups-deviced helper program could miss reporting some backend + devices (STR #3108) + - The cupsSideChannelSNMP* functions did not work. + - The scheduler could consume 100% CPU when jobs were canceled. + - Clicking on "Select Another Make/Manufacturer" in the web interface + incorrectly added the printer (STR #3095) + - The scheduler no longer uses programs with insecure file + permissions. + - httpAssembleURI*() did not escape backslashes in hostnames. + - The dnssd backend did not unquote "full names" before creating the + device URI. + - The scheduler now supports JobRetryInterval values less than 10 + seconds. + - Updated the Spanish localization (STR #3090) + - The scheduler did not redo Bonjour/DNS-SD registrations when updating + them failed. + - The "authenticated" policy incorrectly required authentication for + status operations. + - ppdOpen*() incorrectly loaded PPDs with multiple JobPatchFile + keywords. + - The network backends no longer report the SNMP "offline" or + maintenance status bits since they are inconsistently implemented and + often unreliable. + - The scheduler no longer logs child processes killed via SIGKILL as + "crashed". + - The printer link shown on the "job moved" template was bad (STR #3085) + - Updated the HTML templates to use the final HTML 4 DOCTYPE (STR #3086) + - The scheduler did not track the "paused" reason properly if a + printer had other reasons associated with it. + - cupsSendRequest() did not clear old local certificate auth data. + - The PPD compiler did not search for localization files properly + (STR #3084) + - cupsGetNamedDest() did not use the fallback default like + cupsGetDests*() (STR #3082) + - The scheduler now provides a LogTimeFormat directive to enable + microseconds in the date and time that are logged. + - The scheduler now provides a MultipleOperationTimeout directive to + control the timeout for multi-file print jobs. + - The configure script incorrectly allowed Avahi to be used for DNS-SD + printer discovery (STR #3065) + - The web interface and scheduler did not support URIs up to 1024 bytes + in length (STR #3072) + - Fixed pdftops issues with page sizes (STR #3063) + - Fixed pdftops issues with Ghostscript (STR #3062) + - The scheduler incorrectly registered default profiles for PostScript + printers with no specified colorspace. + - The scheduler incorrectly created an empty org.cups.printers.plist + file on Mac OS X. + - cupsGetPPD3() did not look for local PPDs in the right directory. + - SNMP lookups via side-channel did not work for NULL-VALUE and + and OCTET-STRING OIDs containing nul characters. + - The libusb-based USB backend did not work. + - The scheduler did not set the printer-commands attribute correctly + for some PPDs. + - The ppdi utility did not work. + - The web interface no longer uses multi-part output with old or broken + web browsers (STR #3049) + - CUPS now conforms to the draft IPP/2.0 and IPP/2.1 specification. + - Added a new cupsGetConflicts() API to get a list of conflicting + options. + - The PPD compiler didn't localize options or choices that did not + have associated translation text (STR #3045) + - Updated the Spanish localization (STR #3043) + - Fixed build problems (STR #3040, STR #3047) + - cupsResolveConflicts() did not resolve using the default option + choice in some cases due to the mirror UIConstraints that are + present in most PPD files. + - The scheduler did not honor MIME type priorities. + - The commandtops filter incorrectly used the JCLBegin code to end + its jobs. + - The default BrowseLocalProtocols value was not set properly. + - Since the commandtops filter does not actually support ReportLevels + all on its own, don't list that printer command by default for PS + printers. + - The scheduler did not give filters a chance to log errors or update + printer attributes when a job was canceled. + - The scheduler did not clear the "connecting-to-device" reason keyword + when a job finished. + + +CHANGES IN CUPS V1.4b2 + + - Documentation updates (STR #2983, STR #2998, STR #3021) + - The cupstestppd utility now validates the FileVersion and + FormatVersion values in PPD files. + - The default cupsd.conf file did not reflect the + --with-local-protocols value set at compile-time (STR #3037) + - The cupsGetPPD* APIs now create symlinks to local PPD files + rather than copying them whenever possible. + - Various performance optimizations in the string pool, dests, and + options implementations. + - The cupsGetDests* APIs now return the marker and printer-commands + attributes. + - Side-channel SNMP lookups would not work when cupsSNMPSupplies + was set to False in the PPD file. + - Localized the device descriptions for the SCSI, serial, + and network backends (STR #3014) + - Added a Spanish localization (STR #3015) + - Added support for marker-low-levels and marker-high-levels + attributes. + - The scheduler could hang writing a long log line. + - The cupsGetDevices() function now has an "include_schemes" + parameter. + - The lpinfo command now supports --include-schemes and + --exclude-schemes options. + - The CUPS-Get-PPDs operation now supports the include-schemes + and exclude-schemes attributes. + - The CUPS-Get-Devices operation now supports the include-schemes + attribute. + - The print filters now support a replacement for the fitplot + option called "fit-to-page". + - The LPD backend no longer tries to collect page accounting + information since the LPD protocol does not allow us to + prevent race conditions. + - The scheduler did not save the last marker-change-time value. + - Fixed a problem with printing to some IPP printers, including + CUPS 1.1.x. + - Fixed a redirection problem with the printer web page (STR #3012) + - Fixed a PPD compiler problem with the loading of message + catalogs (STR #2990) + - Fixed a PPD compiler problem with the loading of .strings files + (STR #2989) + - The cupsfilter utility did not set the CONTENT_TYPE environment + variable when running filters. + - The scheduler now waits to allow system sleep until the jobs + have all stopped. + - The IPP, LPD, and socket backends used different "connecting" + progress messages. + + +CHANGES IN CUPS V1.4b1 + + - Documentation updates (STR #2567) + - The PPD compiler now allows local message catalogs to + override the standard CUPS translations (STR #2642) + - The ppdmerge command did not merge custom option strings + (STR #2863) + - The scheduler now supports the Hold-New-Jobs and + Release-Held-New-Jobs operations; these are exposed via the + cupsdisable and cupsenable commands (STR #2332) + - The lpstat command is now much faster when displaying the + status of a single printer (STR #2843) + - The scheduler now caches information from PPD files to provide + significantly faster startup time with large numbers of PPDs + (STR #1293) + - CUPS-Get-Driver now provides much better driver matching based + on the IEEE-1284 device ID and make/model strings (STR #2707) + - Now support the cupsSNMPSupplies keyword to control whether + the network backends query the SNMP Printer MIB for supply + levels. + - Now support and use a new banner file format for better text + support and easier customization (STR #2490) + - The scheduler now sets the PRINTER_INFO and PRINTER_LOCATION + environment variables from the corresponding IPP attributes. + - The ippRead*() and ippWrite*() functions no longer use a + stack-based buffer (STR #2388) + - The CUPS-Add-Modify-Printer operation now allows you to set + the printer-state-reasons attribute. + - The "set printer options" page now supports auto-configuration + of printer options (STR #1440) + - The web interface now provides an advanced server settings + form. + - The web interface's "modify printer" pages now make it + easier to change just one setting (STR #1919) + - The scheduler now supports a plist PrintcapFormat. + - The scheduler now supports multiple addresses in Allow and + Deny lines, just like Apache (STR #2947) + - Added CUPS_JOBTYPE environment variable for job filters so + they know whether they are printing a banner or document + file (STR #2799) + - Added support for printer filtering by the cupsfilter + command (STR #2562) + - Added a SSLOptions directive to allow Windows clients to + talk to CUPS in FIPS mode (STR #2827) + - Renamed the accept and reject commands to cupsaccept and + cupsreject; the old names are still available (STR #2936) + - The locale/translate utility needed an update to work with + Google (STR #2882) + - The lpstat command now supports a -H option to display the + default server (STR #2833) + - The scheduler now supports a FatalErrors directive to control + which errors should cause the scheduler to exit (STR #2536) + - The scheduler now uses the php-cgi program if it is available + (STR #2923) + - The scheduler now supports a DefaultPaperSize directive + (STR #2848) + - The scheduler now passes the job-originating-host-name + value to filters in the options argument (STR #2558) + - CUPS now supports job tickets in PDF files (STR #2903) + - Added a DBUS notifier (STR #2529) + - The LPD mini-daemon now passes the document name when queuing + print jobs (STR #2482) + - The IPP backend did not relay com.apple.print.recoverable-message + values. + - The scheduler now supports a job-media-progress attribute to + track the progress of individual pages. + - The sample HP driver now supports A5 (STR #2798) + - The CUPS web interface menu item now uses the xdg-open + command, when available (STR #2724) + - The cups-lpd program now supports the -h option (STR #2794) + - The scheduler now sets the PAM_TTY parameter and the + PAM_ESTABLISH_CRED credential flag (STR #2745) + - The scheduler now logs unsuccessful requests to the error_log + file as errors (STR #2616) + - Added support for a "retry-current-job" error policy that + retries the current job immediately when the backend encounters + an error (STR #2555) + - The scheduler now returns a "forbidden" error when a user + correctly authenticates but does not have permission to + continue further (STR #2101) + - The scheduler now loads both the server and CA certificates + (if present) from the ServerCertificate file (STR #2146) + - New RSS subscriptions now create their feed files immediately + (STR #2853) + - Added support for a device-location attribute which provides + the physical location of a printer device. + - Added a cupsBackendReport() API which handles quoting of the + device data by a backend. + - Added support for custom options in the web interface + (STR #1729) + - Added support for Mozilla LDAP, reconnection to LDAP servers, + and improved LDAP performance (STR #1962) + - Added Solaris SMF support (STR #1477) + - Added optional support for using TCP wrappers to limit access + to CUPS (STR #263) + - Added ppdPageSizeLimits API. + - Added support for new cupsMediaQualifier2, cupsMediaQualifier3, + cupsMinSize, and cupsMaxSize attributes. + - Added cupsResolveConflicts and ppdInstallableConflict APIs. + - Added support for new cupsUIConstraints and cupsUIResolver + attributes for better option conflict detection and + resolution. + - Increased the maximum size of 1284 device ID strings to + 256 bytes (STR #2877) + - Added an AccessLogLevel directive to cupsd.conf to control + what is logged to the access_log file. + - The default LogLevel is now "warn" instead of "info" to reduce + the amount of logging that is done to disk by default. + - The PPD compiler did not include OID query keywords in PPD + files (STR #2871) + - The cups-driverd helper program now directly supports driver + information files. + - The USB backend now uses libusb when available (STR #1575) + - Added ppdLocalizeAttr function to get the localized version + of an attribute. + - MIME types now support a priority() attribute (STR #2719) + - The standard MIME types are now installed in + DataDir/mime (STR #2719) + - The lpoptions command now describes custom options and + the necessary parameters (STR #2660) + - The ppdmerge program did not support Simplified Chinese + or Traditional Chinese language version strings (STR #2851) + - The PPD compiler now supports localizable attributes + (STR #2738) + - The ppdpo utility now includes cupsIPPReasons values in + the message catalogs it generates (STR #2754) + - The PPD compiler now supports conditional directives + (STR #2636) + - The ppdc utility now supports a "-t" option to test PPD + files (STR #2739) + - The ppdc utility now supports a "-m" option to use the + ModelName value as the output filename. + - The ppdc utility now supports a FileName directive to + set an alternate output filename (STR #2740) + - The side-channel API now supports SNMP queries for the + standard network backends. + - Added a PageLogFormat directive to the cupsd.conf file to + control the format of lines in the page_log file. + - Filters can now send PPD: messages to stderr to set PPD + keywords like DefaultPageSize while a job is printing. + - Added a mdns backend for discovery and printing to printers + that advertise themselves via DNS-SD (Bonjour) + - The ipp, lpd, and socket backends now support DNS-SD service + name resolution. + - The scheduler now uses a single shared file descriptor for + all DNS-SD registrations (STR #2674) + - The ipp, lpd, and socket backends now support SNMP-based + page accounting and supply level monitoring (STR #1655) + - Added support for cupsPJLDisplay attribute to control what + PJL commands are used to display the job information. + - Driver information files can now be installed in + /Library/Printers/PPDs.drv on Mac OS X. + - The CUPS image library now supports reading images larger + than 2GB. + - The scheduler now delays writing config and state files to + reduce disk activity (STR #2684) + - The CUPS-Get-Devices operation now supports the + exclude-schemes and timeout attributes to control which + backends are polled and for how long. + - The cups-deviced helper application now runs backends in + parallel to get the list of devices faster. + - Added --enable-pap configure option. + - The default cupsd.conf file now includes an "authenticated" + policy which requires authentication for remote print jobs. + - Added support for Czech and Hungarian in PPD files + (STR #2735, STR #2736) + - The PPD compiler tools now support Mac OS X .strings files + for localization (STR #2737) + - ppdOpen*() now default the colorspace member to PPD_CS_N + when no DefaultColorSpace attribute is present in the PPD + file. + - The build system has been updated to support separate + installation of data, program, header, and library files. + - All support libraries are now built as shared libraries + by default. + - The scheduler now manages ICC color profiles on Mac OS X. + - The network backends (ipp, lpd, socket) now support + SNMP-based supply and page count monitoring (STR #1655) + - The lppasswd program is no longer installed setuid to + root to make the default installation more secure. + - Added a new ppdLocalizeMarkerName() function to get + the localized version of a marker-names value. + - The scheduler now provides the printer-dns-sd-name + attribute for printers shared via DNS-SD/Bonjour. + - The pdftops filter now executes the Xpdf or poppler + pdftops utility to convert PDF files (STR #1471) + - Bonjour printer registrations now advertise as local or + global based on the current access policies for the + printer. + - cupsGetDests*() and cupsSetDests*() now track the last + used printer preference on Mac OS X. + - Added a new streaming request API (STR #2261) + - Added a new cupsGetNamedDest() function to the CUPS + library for faster printing with lp and lpr (STR #2638) + - The scheduler now sets the PAM RHOST value on systems + that support it (STR #2637) + - The scheduler now sandboxes child processes when + possible. + - The Cancel-Job operation now supports a purge-job + attriibute to purge a specified job. + - ppdEmit* and ppdCollect* now use the NonUIOrderDependency + attributes for custom option selections. + - The web interface now enables/disables the printer + sharing (formerly publishing) controls based on the + server-is-sharing-printers state (STR #2233) + - The scheduler now tracks printer sharing via the + server-is-sharing-printers attribute, and manages LPD + and SMB sharing as well (STR #2233) + - The web interface now allows you to go back to the make/ + manufacturer page if there is no matching printer driver + on the model page (STR #2436) + - The printer list now shows the default media, banner, and + duplex options as well as the color and duplex capabilities + of printers (STR #1175) + - The web interface look-n-feel has been updated (STR #2492) + - The scheduler now supports a CUPS-Get-Document operation + that returns the specified print job document (STR #118) + - The cupsfilter utility now supports a "-J jobid" option + to filter the document from the specified job. + - The scheduler (cupsd) now supports a new option (-t) to + do a syntax check of the cupsd.conf file (STR #2003) + - Added new cupsGetPPD3() API to allow applications to + cache PPDs safely (STR #1473) + - Added generic PostScript and PCL printer driver PPDs. diff --git a/CHANGES-1.5.txt b/CHANGES-1.5.txt new file mode 100644 index 0000000000..94debd4dd9 --- /dev/null +++ b/CHANGES-1.5.txt @@ -0,0 +1,228 @@ +CHANGES-1.5.txt +--------------- + +CHANGES IN CUPS V1.5.1 + + - Documentation updates (STR #3885, STR #3886, STR #3946, STR #3969) + - Localization updates (STR #3840, STR #3989, STR #3997) + - Build fixes (STR #3956, STR #3999) + - The SNMP backend did not validate the device URIs reported by printers + (STR #4004) + - cupsBackendReport() did not handle newlines in 1284 Device IDs + (STR #4005) + - USB backend fixes for libusb (STR #3965, STR #3978) + - The DBUS notifier did not validate string parameters (STR #3984) + - Group quota ACLs did not work with Kerberos (STR #3972) + - The IPP backend did not retry when a printer responded with + client-error-not-possible (STR #3963) + - PostScript PPDs with filters used the wrong command filter (STR #3973) + - The scheduler incorrectly used free() on a POSIX ACL value, which + could cause a crash (STR #3970) + - PPD files using the MacStandard encoding did not work. + - The web interface did not work on some platforms (STR #3902) + - The lpstat command would crash when then "-u" option was used by a + non-administrator (STR #3953) + - Japanese supply level reporting did not always work. + - The DBUS notifier could crash (STR #3947) + - Relaxed some of the page size checks in cupstestppd. + - The ipptool program now reports attributes that are repeated within + the same attribute group. + - Updated the PWG raster support to match the current draft + specification. + - Fixed some IPP conformance issues in the scheduler. + - Added ipptool support for repeating requests. + - Added IPP/2.2 conformance tests and greatly improved the IPP/1.1, + IPP/2.0, and IPP/2.1 conformance testing. + - IPP messages containing mixed integer/rangeOfInteger values did not + work (STR #3942) + - The ipptool program now provides additional diagnostics for badly- + formatted responses (STR #3857) + - When possible, the IPP backend now stops sending job data early on a + cancel. + - cupsSendRequest and cupsWriteRequestData did not properly read all + HTTP headers, preventing authentication and encryption upgrades from + working in all cases. + - The client.conf Server directive is no longer supported on Mac OS X + 10.7 and later. + - The IPP backend sent the wrong margins in media-col. + - The scheduler did not save or restore large Kerberos credentials for + jobs. + - The dnssd backend did not properly browse for secure IPP printers. + - httpAssembleURI* did not properly escape all special characters in the + username/password field. + - The scheduler now logs config file errors to stderr (STR #3936) + - The configure script incorrectly used bundle-based localizations on + Linux (STR #3938) + - The cups-driverd helper program did not cache .drv files properly, + sometimes leading to a crash (STR #3921) + - CUPS did not build on stock Mac OS X installations. + - Encryption was broken with OpenSSL. + - ipptool's XML output used date/time values with timezone offsets, + which are not supported by Mac OS X's NSDate class. + - Several programs did not support the cupsFilter2 keyword in PPD files. + - The IPP backend incorrectly reported spool-area-full states. + - cupsMarkOptions() did not protect against a bad PPD that was missing + one or more standard Duplex options. + - The PostScript filter did not mirror N-up output properly. + - The ipptool program did not validate UTF-8 strings in XML output. + - Fixed supply level reporting for some printers. + - The scheduler no longer automatically logs debug messages for jobs + that were held or canceled. + - The cupsSendRequest function did not flush remaining response data + from a previous request, leading to apparent chunking issues. + - The scheduler did not report the correct version in the Server: header + (STR #3903) + - The scheduler did not support 1284 device IDs reported by driver + interface programs longer than 127 characters (STR #3871) + - The image filters did not support loading images larger than the + RIPCache setting (STR #3901) + - "PAGE: total NNN" messages did not get logged properly (STR #3887) + - Updated the PWG Raster support to conform to the current draft of the + PWG Raster Format specification. + - The PWG Raster filter did not always write the correct number of + padding lines on the bottom of the page (STR #3904) + - When reporting a denial-of-service attack from the domain socket, the + address reported does not always contain the correct path (STR #3888) + - Badly formed GIF files could cause the image filters to crash + (STR #3914) + - Jobs canceled at the printer were retried by the IPP backend. + - "cupsfilter -u" deleted the input file instead of the PPD file. + - The scheduler did not compute the cost of PPD filters defined using + the cupsFilter2 keyword properly. + - The scheduler did not correctly support the maxsize() attribute for + PPD filters. + + +CHANGES IN CUPS V1.5.0 + + - Documentation updates. + - Localization update (STR #3865) + - Needed to limit TLS to v1.0 on some versions of Mac OS X. + - The snmp backend did not work with some printers. + + +CHANGES IN CUPS V1.5rc1 + + - Compile fixes (STR #3849, STR #3850) + - The scheduler didn't check for empty values for several configuration + directives (STR #3861) + - ipptool didn't generate valid XML when a test was skipped. + - Added additional error checking to the 1284 device ID code (STR #3858) + - Fixed some compatibility issues migrating from the old usblp backend + to the libusb backend (STR #3860) + - Fixed the wake-from-sleep printing behavior on Mac OS X. + - The scheduler incorrectly allowed jobs to be held from a terminating + state. + - The cups-driverd program could crash when a PPD was renamed. + - The dnssd backend took too long to discover printers on large or busy + networks with the new default timeout used by lpinfo and the web + interface. This resulted in "lost" printers. + + +CHANGES IN CUPS V1.5b2 + + - Documentation updates. + - Localization updates (STR #3845) + - Compiler warning cleanup. + - Fixed PIE support for Linux (STR #3846) + - Made httpSetTimeout API public and use it in the IPP backend to avoid + timeout errors. + - The scheduler incorrectly set the "authenticated" printer-type bit for + remote queues using authentication. + + +CHANGES IN CUPS V1.5b1 + + - The CUPS library now supports per-connection HTTP timeouts and + callbacks. + - The CUPS library now supports (limited) SSL/TLS X.509 certificate + validation and revocation (STR #1616) + - Updated the PostScript filter to support IncludeFeature in more + circumstances (STR #3417) + - The schedule did not correctly parse some IPv6 addresses and masks in + the cupsd.conf file (STR #3533) + - Fixed a case-insensitive string comparison issue for locales that do + not treat "I" and "i" as equivalent (STR #3800) + - The scheduler reported an incorrect job-printer-uri value when sharing + was not enabled (STR #3639) + - The scheduler now allows the ServerAlias directive to contain multiple + hostnames separated by spaces or commas (STR #3813) + - The scheduler now sets the process group for child processes and + manages the group (STR #2829) + - Fixed some minor issues discovered by a Coverity scan (STR #3838) + - The scheduler now more carefully creates and removes configuration, + cache, and state files (STR #3715) + - The lpadmin command now allows default option values to be deleted + (STR #2959) + - The lpadmin command now allows the cupsIPPSupplies and + cupsSNMPSupplies keywords to be set in a PPD file (STR #3825) + - Moving a held job no longer releases it (STR #3839) + - Restored support for GNU TLS and OpenSSL with threading enabled + (STR #3605) + - Fixed a confusing error message from cups-polld (STR #3806) + - Increased the default RIPCache value to 128MB (STR #3535) + - MIME errors are now routed to the error_log file (STR #2410) + - Updated PDF filter to support new Ghostscript ps2write device + (STR #3766) + - Updated PDF filter to support new Poppler option to preserve page + sizes in PDF files when the user has not selected a particular media + size (STR #3689) + - Added new PWG Raster filter for IPP Everywhere printer support. + - Added job-uuid, printer-uuid, and subscription-uuid attributes. + - Added support for the cupsSingleFile PPD keyword. + - Dropped support for the printer-state-history attribute (STR #3654) + - Added support for a new cupsIPPSupplies keyword in PPD files to allow + drivers to disable IPP supply level reporting. + - Added support for a new cupsFilter2 keyword in PPD files to allow for + the propagation of the actual MIME media type produced by a filter. + - The scheduler did not always get the correct Kerberos username when + authenticating (STR #3670) + - Added new cupsRasterOpenIO function and CUPS_RASTER_WRITE_PWG to the + CUPS imaging library to support printing to IPP Everywhere raster + printers. + - The scheduler now provides default values for the pages-per-minute and + pages-per-minute-color attributes for PPD files that lack a + Throughput keyword. + - Email notifications did not work on Mac OS X. + - The cupstestppd program now shows an error for files missing a + CloseGroup keyword (STR #3668) + - Name resolution errors no longer cause queues to stop (STR #3719, + STR #3753) + - Added a new cups-exec helper program that applies security profiles + to filters, port monitors, backends, CGI programs, and mini-daemons. + - The web interface can now be disabled using the WebInterface directive + in cupsd.conf (STR #2625) + - The scheduler now provides privacy controls for jobs and subscriptions + (STR #2969) + - Added new cupsArrayNew3 API which offers memory management of array + elements. + - Added several new color spaces to the CUPS raster format (STR #3419) + - The Validate-Job operation now uses the same policy as Print-Job by + default. + - CUPS now uses iconv to implement all of its character encoding + support (STR #3097) + - The scheduler now implements the Cancel-Jobs, Cancel-My-Jobs, and + Close-Job operations along with the job-ids operation attribute from + PWG 5100.11. + - The main CUPS header () no longer includes the PPD header + (). + - The scheduler and CUPS API now support the print-quality job template + attribute. + - The scheduler no longer supports the old Mac OS X Server quota + plugin. + - The scheduler now allows writing to /Users/Shared from print filters + on Mac OS X. + - CUPS no longer supports the old ~/.cupsrc or ~/.lpoptions files from + CUPS 1.1.x. The ~/.cups/client.conf and ~/.cups/lpoptions files that + were introduced in CUPS 1.2 must now be used. + - The ipptest tool is now a first-class user program and has several + improvements along with new documentation (STR #3484) + - The cupstestppd tool now warns about non-unique filenames and + provides a way to ignore all filename warnings. + - Dropped support for the recoverable: and recovered: message prefixes. + - The scheduler now requires that filters and backends have group write + permissions disabled. + - The PPD compiler now checks for overlapping filenames when writing + PPD files. + - The HP-GL/2 filter is no longer included with CUPS (STR #3322) + - The SCSI backend is no longer included with CUPS (STR #3500) diff --git a/CHANGES-IPPTOOL.txt b/CHANGES-IPPTOOL.txt new file mode 100644 index 0000000000..9c307156d4 --- /dev/null +++ b/CHANGES-IPPTOOL.txt @@ -0,0 +1,107 @@ +CHANGES-IPPTOOL.txt - 2012-01-10 +-------------------------------- + +This file provides a list of changes to the ipptool binary distribution posted +on cups.org. + + +2012-01-10 + + - Fixed an issue with the IPP/1.1 test file where the "waiting for job + completion" test did not work. + - The attribute groups out of order error was not shown for the + operation-attributes-tag group. + + +2011-12-02 + + - Fixed a truncated XML output bug that would happen for certain errors. + - Fixed the order-of-groups tests. + - Fixed "WITH-VALUE >N" for rangeOfInteger attributes. + - The Windows installer was missing the get-printer-attributes.test + file. + - The Linux binaries are now compiled for all LSB 4.x-compliant Linux + distributions. + - The Linux binaries no longer support SSL or TLS. + + +2011-10-05 + + - Fixed a crasher bug that showed up on Windows. + - The IPP/1.1 test would hang if the initial Print-Job test failed. + - Fixed a typo in the IPP/2.0 test. + + +2011-10-03 + + - Using OF-TYPE with the "no-value" or "unknown" out-of-band value tags + now works without special WITH-VALUE strings. + - ipptool now shows an error when an attribute appears more than once + within the same attribute group. + - ipptool did not display attributes as specified by the DISPLAY + directive when a test error failed. + - Fixed one spot where DEFINE-VALUE was still treated as an error when + the EXPECT test failed. + - uriScheme values were not displayed properly. + - Updated the IPP/1.1 tests to properly validate support for + job-hold-until-default and job-hold-until-supported when the Hold-Job + operation is supported. + - Updated the IPP/1.1 tests to properly check for the various standard + media sizes for the printing tests. + - Updated the IPP/1.1 tests to accept "no-value" for media-default an + orientation-requested-default. + - Updated the IPP/1.1 tests to accept "unknown" for job-state. + - Updated the IPP/1.1 tests to not perform the bogus URI printing tests + unless the "document-uri" variable is defined. + - Updated the IPP/2.0 tests to properly validate media-col + "media-xxx-supported" attributes. + - Updated the IPP/2.2 tests to properly validate media-ready and + media-col-ready. + + +2011-09-28 + + - Test output now includes a summary and overall score at the end. + - The MATCH-VALUE predicate now correctly deals with a failed EXPECT + condition. + - The IPP/1.1 test suite now looks for legacy media names and uses them + if the corresponding PWG standard names are not present. + - The IPP/1.1 test suite now tests the Print-Job+Release-Job when the + printer supports the job-hold-until attribute, Hold-Job operation, and + Release-Job operation. + + +2011-09-21 + + - Fixes for HTTP chunking, timeout, and encryption issues reported by + various users. + - Greatly improved IPP tests with added IPP/2.2 tests. + - New test documents - 1-page and 4-page mixed A4/Letter PDF/PS and a + couple JPEGs. + - New REPEAT directives to programmatically repeat tests as needed. + + +2011-08-16 + + - The Windows version no longer requires Visual Studio to be installed. + - The Windows version now supports SSL. + - Added "ipps" URI support. + - Added a new "-T" option. + - Added support for fractional seconds for the -i option and DELAY + directive. + - Added support for authentication. + - Added DEFINE-MATCH, DEFINE-NO-MATCH, DEFINE-VALUE, IF-NOT-DEFINED, + IGNORE-ERRORS, SKIP-IF-DEFINED, SKIP-IF-NOT-DEFINED, + SKIP-PREVIOUS-ERROR directives. + - WITH-VALUE now supports variable expansion. + - Updated the IPP/1.1 conformance test to skip the "my-jobs different + user" test if the printer URI contains a username. + - Updated the IPP conformance tests to validate media, media-default, + and media-supported values. + - No longer error out if a Printer returns a different version number in + the response when the request contains the version 0.0. + + +2010-10-16 + + - Initial release of standalone binary. diff --git a/CHANGES.txt b/CHANGES.txt new file mode 100644 index 0000000000..454e98dc97 --- /dev/null +++ b/CHANGES.txt @@ -0,0 +1,40 @@ +CHANGES.txt - 1.6b1 - 2012-01-30 +-------------------------------- + +CHANGES IN CUPS V1.6b1 + + - The configure script and build system no longer support building of + separate 32-bit and 64-bit libraries. + - The "brightness", "columns", "fitplot", "gamma", "hue", + "natural-scaling", "penwidth", "position", "ppi", "saturation", and + "scaling" options are not longer supported (STR #4010) + - The "page-bottom", "page-left", "page-right", "page-top", + "prettyprint", and "wrap" options have been deprecated (STR #4010) + - The scheduler now reports the standard "number-of-documents" attribute + instead of the CUPS-specific "document-count" attribute in + job objects. + - Added new destination connection and enumeration functions (STR #3924) + - Added new option, localization, and job submission functions that do + not depend on PPD files (STR #3925) + - Added a new MaxJobTime directive for cupsd that specifies the maximum + amount of time allowed for a job to complete before it is canceled. + - The default password callback now supports passwords up to 127 + characters. + - The scheduler now supports a DefaultAuthType of "auto" to + automatically choose between Basic (username/password) and Negotiate + (Kerberos) authentication. + - cupsSideChannelSNMPGet/Walk now support OIDs and values up to 64k in + length. + - CUPS no longer supports automatic remote printers or implicit classes + via the CUPS, LDAP, or SLP protocols (STR #3922, STR #3923) + - The PPD APIs are now deprecated and will be removed in a future + version of CUPS (STR #3927) + - The default IPP version for requests is now 2.0 (STR #3929) + - The IPP APIs no longer expose the ipp_t or ipp_attribute_t structures + and instead provide accessor functions (STR #3928) + - The scheduler will no longer run programs with group write permission. + - The PHP module has been removed (STR #3932) + - The bannertops, commandtoescpx, commandtopclx, imagetops, + imagetoraster, pdftops, rastertoescpx, rastertopclx, and texttops + filters have been removed (STR #3930) + - The serial and parallel backends have been removed (STR 3935) diff --git a/CREDITS.txt b/CREDITS.txt new file mode 100644 index 0000000000..a241eef4db --- /dev/null +++ b/CREDITS.txt @@ -0,0 +1,49 @@ +CREDITS.txt - 2010-03-13 +------------------------ + +Few projects are completed by one person, and CUPS is no exception. We'd +like to thank the following individuals for their contributions: + + Nathaniel Barbour - Lots of testing and feedback. + N. Becker - setsid(). + Philippe Combes - French localization and buttons script. + Jean-Eric Cuendet - GhostScript filters for CUPS. + Van Dang - HTTP and IPP policeman. + L. Peter Deutsch - MD5 code. + Dr. ZP Han - setgid()/setuid(). + Guy Harris - *BSD shared libraries and lots of other + fixes. + Bjoern Jacke - I18N stuff. + Wang Jian - CUPS RPM corrections. + Roderick Johnstone - Beta tester of the millenium. + Till Kamppeter - Bug fixes, beta testing, evangelism. + I–aki Larra–aga - Basque localization. + Kenshi Muto - Japanese localization, patches, and + testing. + Tomohiro Kato - Japanese localization. + Kiko - Bug fixes. + Sergey V. Kovalyov - ESP Print Pro and CUPS beta tester. + Marek Laane - Estonian translation. + Mark Lawrence - Microsoft interoperability testing. + Jeff Licquia - Bug fixes, beta testing, evangelism. + Jason McMullan - Original CUPS RPM distributions. + Wes Morgan - *BSD fixes. + Daniel Nylander - Swedish localization. + Niklas 'Nille' kerstršm - Swedish localization. + Giulio Orsero - Bug fixes and testing. + Michal Osowiecki - Polish localization. + Citra Paska - Indonesian localization. + Kurt Pfeifle - Bug fixes, beta testing, evangelism. + Vincenzo Reale - Italian localization. + Petter Reinholdtsen - HP-UX compiler stuff. + Juan Pablo Gonz‡lez Riopedre - Spanish localization. + Opher Shachar - Hebrew localization. + Stuart Stevens - HP JetDirect IPP information. + Andrea Suatoni - IRIX desktop integration and testing. + Teppo Turliainen - Finnish localization. + Tim Waugh - Lots of patches, testing, and Linux + integration. + Yugami - LDAP browsing support. + +If I've missed someone, please let me know by sending an email to +"msweet@apple.com". diff --git a/INSTALL.txt b/INSTALL.txt new file mode 100644 index 0000000000..a7b5a5869b --- /dev/null +++ b/INSTALL.txt @@ -0,0 +1,207 @@ +INSTALL - CUPS v1.5.0 - 2011-07-25 +---------------------------------- + +This file describes how to compile and install CUPS from source code. For more +information on CUPS see the file called "README.txt". A complete change log can +be found in "CHANGES.txt". + +**** IF YOU HAVE A NON-POSTSCRIPT PRINTER AND ARE NOT **** +**** RUNNING MAC OS X, YOU WILL ALSO NEED TO INSTALL GPL **** +**** GHOSTSCRIPT WITH THE "cups" DRIVER AFTER YOU INSTALL **** +**** CUPS. **** + + +BEFORE YOU BEGIN + + You'll need ANSI-compliant C and C++ compilers, plus a make program and + POSIX-compliant shell (/bin/sh). The GNU compiler tools and Bash work well + and we have tested the current CUPS code against several versions of GCC + with excellent results. + + The makefiles used by the project should work with most versions of make. + We've tested them with GNU make as well as the make programs shipped by + Compaq, HP, SGI, and Sun. BSD users should use GNU make (gmake) since BSD + make does not support "include". + + Besides these tools you'll want the JPEG, PNG, TIFF, and ZLIB libraries for + image support, the CDSA, GNU TLS, or OpenSSL libraries for encryption + support, the OpenLDAP and OpenSLP libraries for directory services support, + and either MIT (1.6.3 or higher) or Heimdal Kerberos for Kerberos support. + CUPS will compile and run without these, however you'll miss out on many of + the features provided by CUPS. + + Also, please note that CUPS does not include the Ghostscript-based + PostScript filter needed by non-PostScript printers. You *must* download + GPL Ghostscript separately from the CUPS web site if you want to print + PostScript files to non-PostScript printers on operating systems other than + Mac OS X. + + +COMPILING THE SUBVERSION REPOSITORY CODE + + The CUPS Subversion repository doesn't hold a copy of the pre-built + configure script. You'll need to run the GNU autoconf software (2.60 or + higher) to create it: + + autoconf + + +CONFIGURATION + + CUPS uses GNU autoconf, so you should find the usual "configure" script in + the main CUPS source directory. To configure CUPS for your system, type: + + ./configure + + The default installation will put the CUPS software in the "/etc", "/usr", + and "/var" directories on your system, which will overwrite any existing + printing commands on your system. Use the "--prefix" option to install the + CUPS software in another location: + + ./configure --prefix=/some/directory + + To see a complete list of configuration options, use the --help option: + + ./configure --help + + If any of the dependent libraries are not installed in a system default + location (typically "/usr/include" and "/usr/lib") you'll need to set the + CFLAGS, CPPFLAGS, CXXFLAGS, DSOFLAGS, and LDFLAGS environment variables + prior to running configure: + + setenv CFLAGS "-I/some/directory" + setenv CPPFLAGS "-I/some/directory" + setenv CXXFLAGS "-I/some/directory" + setenv DSOFLAGS "-L/some/directory" + setenv LDFLAGS "-L/some/directory" + ./configure ... + + or: + + CFLAGS="-I/some/directory" \ + CPPFLAGS="-I/some/directory" \ + CXXFLAGS="-I/some/directory" \ + DSOFLAGS="-L/some/directory" \ + LDFLAGS="-L/some/directory" \ + ./configure ... + + The "--enable-debug" option compiles CUPS with debugging information + enabled. Additional debug logging support can be enabled using the + "--enable-debug-printfs" option - these debug messages are enabled using the + CUPS_DEBUG_LOG environment variable at run-time. + + CUPS also includes an extensive set of unit tests that can be used to find + and diagnose a variety of common problems - use the "--enable-unit-tests" + configure option to run them at build time. + + Once you have configured things, just type: + + make ENTER + + or if you have FreeBSD, NetBSD, or OpenBSD type: + + gmake ENTER + + to build the software. + + +TESTING THE SOFTWARE + + Aside from the built-in unit tests, CUPS includes an automated test + framework for testing the entire printing system. To run the tests, just + type: + + make check ENTER + + or if you have FreeBSD, NetBSD, or OpenBSD type: + + gmake check ENTER + + The test framework runs a copy of the CUPS scheduler (cupsd) on port 8631 + in /tmp/cups-$USER and produces a nice HTML report of the results. + + +INSTALLING THE SOFTWARE + + Once you have built the software you need to install it. The "install" + target provides a quick way to install the software on your local system: + + make install ENTER + + or for FreeBSD, NetBSD, or OpenBSD: + + gmake install ENTER + + You can also build binary packages that can be installed on other machines + using the RPM spec file ("packaging/cups.spec") or EPM list file + ("packaging/cups.list"). The latter also supports building of binary RPMs, + so it may be more convenient to use. + + You can find the RPM software at: + + http://www.rpm.org/ + + The EPM software is available at: + + http://www.epmhome.org/ + + +CREATING BINARY DISTRIBUTIONS WITH EPM + + The top level makefile supports generation of many types of binary + distributions using EPM. To build a binary distribution type: + + make ENTER + + or + + gmake ENTER + + for FreeBSD, NetBSD, and OpenBSD. The target is one of the + following: + + epm - Builds a script + tarfile package + aix - Builds an AIX package + bsd - Builds a *BSD package + deb - Builds a Debian package + depot - Builds a HP-UX package (also swinstall) + inst - Builds an IRIX package (also tardist) + pkg - Builds a Solaris package + rpm - Builds a RPM package + setld - Build a Tru64 UNIX package + slackware - Build a Slackware package + swinstall - Build a HP-UX package (also depot) + tardist - Builds an IRIX package (also inst) + + +GETTING DEBUG LOGGING FROM CUPS + + When configured with the "--enable-debug-printfs" option, CUPS compiles in + additional debug logging support in the scheduler, CUPS API, and CUPS + Imaging API. The following environment variables are used to enable and + control debug logging: + + CUPS_DEBUG_FILTER Specifies a POSIX regular expression to control + which messages are logged. + CUPS_DEBUG_LEVEL Specifies a number from 0 to 9 to control the + verbosity of the logging. The default level is 1. + CUPS_DEBUG_LOG Specifies a log file to use. Specify the name "-" + to send the messages to stderr. Prefix a filename + with "+" to append to an existing file. + + +REPORTING PROBLEMS + + If you have problems, READ THE DOCUMENTATION FIRST! If the documentation + does not solve your problems, please post a message on the "cups.general" + forum at: + + http://www.cups.org/newsgroups.php + + Include your operating system and version, compiler and version, and any + errors or problems you've run into. The "config.log" file and the output + from the configure script and make should also be sent, as it often helps to + determine the cause of your problem. + + If you are running a version of Linux, be sure to provide the Linux + distribution you have, too. diff --git a/IPPTOOL.txt b/IPPTOOL.txt new file mode 100644 index 0000000000..3447b19398 --- /dev/null +++ b/IPPTOOL.txt @@ -0,0 +1,145 @@ +IPPTOOL.txt - 2011-12-02 +------------------------ + +See the file CHANGES-IPPTOOL.txt for a list of changes to this software. + + +INTRODUCTION + + Starting with CUPS 1.5, CUPS now installs a user program called ipptool that + can be used to send arbitrary IPP requests to a CUPS server or IPP printer. + This tool started life as part of the CUPS automated test suite and has + grown to support complex conformance tests and a simple way to query + printer, job, and subscription attributes. + + +BASIC USAGE + + The ipptool command requires a printer URI and one or more "test" files that + describe the operations, attributes to display, and expected status and + attribute values. Several standard files are included with CUPS, for example + to show a list of pending print jobs on a CUPS printer called "myprinter" + you'd run: + + ipptool ipp://localhost/printers/myprinter get-jobs.test + + which would produce something like this: + + job-id job-state job-name job-originating-user-name + ------ ------------ ------------ ------------------------- + 72 pending testfile.pdf msweet + 73 pending testfile.ps msweet + 74 pending-held testfile.jpg msweet + 75 pending-held testfile.txt msweet + + To get output suitable for import into a spreadsheet, use the "-c" (CSV) + option: + + ipptool -c ipp://localhost/printers/myprinter get-jobs.test + + which will produce something like this: + + job-id,job-state,job-name,job-originating-user-name + 72,pending,testfile.pdf,msweet + 73,pending,testfile.ps,msweet + 74,pending-held,testfile.jpg,msweet + 75,pending-held,testfile.txt,msweet + + +CONFORMANCE TESTS + + We provide basic IPP conformance tests for IPP/1.1, IPP/2.0, IPP/2.1, and + IPP/2.2. For a given printer URI, the following commands perform tests at + each level: + + ipptool -tf filename [options] -I printer-uri ipp-1.1.test + ipptool -tf filename [options] -I -V 2.0 printer-uri ipp-2.0.test + ipptool -tf filename [options] -I -V 2.1 printer-uri ipp-2.1.test + ipptool -tf filename [options] -I -V 2.2 printer-uri ipp-2.2.test + + The filename must use a format supported by the printer; ipptool will guess + the MIME media type using the extension, otherwise application/octet stream + will be used. The following standard test files are included: + + color.jpg + document-a4.pdf + document-a4.ps + document-letter.pdf + document-letter.ps + gray.jpg + onepage-a4.pdf + onepage-a4.ps + onepage-letter.pdf + onepage-letter.ps + + Print-by-reference (URL) printing can be tested by defining the document-uri + variable to a URL, for example: + + ipptool -tf filename -d document-uri=url -I printer-uri ipp-1.1.test + + The standard test files are available on cups.org under the "test" + directory, for example: + + http://www.cups.org/test/document-a4.pdf + + The "document" test files contain 4 pages each. Doing the IPP conformance + tests will will produce up to 90 pages on various media, depending on the + printer. + + +READING THE DOCUMENTATION + + The command usage is described in the ipptest(1) man page, while the file + format is described in the ipptestfile(5) man page. + + +GETTING SUPPORT AND OTHER RESOURCES + + If you have problems, READ THE DOCUMENTATION FIRST! We also provide many + discussion forums which are available at: + + http://www.cups.org/newsgroups.php + + See the CUPS web site at "http://www.cups.org/" for other resources. + + +REPORTING BUGS + + If you believe you have discovered a bug in ipptool, please fill out the + bug form at: + + http://www.cups.org/str.php + + Be sure to identify the version of CUPS and ipptool (if you downloaded the + standalone version) you are using, the printer (if any) and firmware + version, and include any files that apply. + + If you downloaded the standalone version of ipptool, please also re-run the + test with debug logging enabled. Run the following commands on Windows to + enable debug logging: + + set CUPS_DEBUG_LOG=ipptool.log + set CUPS_DEBUG_LEVEL=6 + + For Linux and Mac OS X use: + + CUPS_DEBUG_LOG=ipptool.log; export CUPS_DEBUG_LOG + CUPS_DEBUG_LEVEL=6; export CUPS_DEBUG_LEVEL + + Then when you run the ipptool command a new "ipptool.log" file will be + created with detailed information - attach this file to the bug you file + as well. + + +LEGAL STUFF + + CUPS is Copyright 2007-2011 by Apple Inc. CUPS and the CUPS logo are + trademarks of Apple Inc. + + The MD5 Digest code is Copyright 1999 Aladdin Enterprises. + + CUPS is provided under the terms of version 2 of the GNU General Public + License and GNU Library General Public License. This program is distributed + in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even + the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + See the "doc/help/license.html" or "LICENSE.txt" files for more information. diff --git a/LICENSE.txt b/LICENSE.txt new file mode 100644 index 0000000000..7d80518a35 --- /dev/null +++ b/LICENSE.txt @@ -0,0 +1,971 @@ + CUPS License Agreement + + Copyright 2007-2011 by Apple Inc. + 1 Infinite Loop + Cupertino, CA 95014 USA + + WWW: http://www.cups.org/ + + +INTRODUCTION + +CUPS(tm) is provided under the GNU General Public License ("GPL") +and GNU Library General Public License ("LGPL"), Version 2, with +exceptions for Apple operating systems and the OpenSSL toolkit. A +copy of the exceptions and licenses follow this introduction. + +The GNU LGPL applies to the CUPS and CUPS Imaging libraries +located in the "cups" and "filter" subdirectories of the CUPS +source distribution and the files in the "test" subdirectory. The +GNU GPL applies to the remainder of the CUPS distribution. + +For those not familiar with the GNU GPL, the license basically +allows you to: + + - Use the CUPS software at no charge. + - Distribute verbatim copies of the software in source or + binary form. + - Sell verbatim copies of the software for a media fee, or + sell support for the software. + +What this license *does not* allow you to do is make changes or +add features to CUPS and then sell a binary distribution without +source code. You must provide source for any changes or additions +to the software, and all code must be provided under the GPL or +LGPL as appropriate. The only exceptions to this are the portions +of the CUPS software covered by the Apple operating system +license exceptions outlined later in this license agreement. + +The GNU LGPL relaxes the "link-to" restriction, allowing you to +develop applications that use the CUPS and CUPS Imaging libraries +under other licenses and/or conditions as appropriate for your +application, driver, or filter. + + +LICENSE EXCEPTIONS + +In addition, as the copyright holder of CUPS, Apple Inc. grants +the following special exceptions: + + 1. Apple Operating System Development License Exception; + + a. Software that is developed by any person or entity + for an Apple Operating System ("Apple OS-Developed + Software"), including but not limited to Apple and + third party printer drivers, filters, and backends + for an Apple Operating System, that is linked to the + CUPS imaging library or based on any sample filters + or backends provided with CUPS shall not be + considered to be a derivative work or collective work + based on the CUPS program and is exempt from the + mandatory source code release clauses of the GNU GPL. + You may therefore distribute linked combinations of + the CUPS imaging library with Apple OS-Developed + Software without releasing the source code of the + Apple OS-Developed Software. You may also use sample + filters and backends provided with CUPS to develop + Apple OS-Developed Software without releasing the + source code of the Apple OS-Developed Software. + + b. An Apple Operating System means any operating system + software developed and/or marketed by Apple Computer, + Inc., including but not limited to all existing + releases and versions of Apple's Darwin, Mac OS X, + and Mac OS X Server products and all follow-on + releases and future versions thereof. + + c. This exception is only available for Apple + OS-Developed Software and does not apply to software + that is distributed for use on other operating + systems. + + d. All CUPS software that falls under this license + exception have the following text at the top of each + source file: + + This file is subject to the Apple OS-Developed + Software exception. + + 2. OpenSSL Toolkit License Exception; + + a. Apple Inc. explicitly allows the compilation and + distribution of the CUPS software with the OpenSSL + Toolkit. + +No developer is required to provide these exceptions in a +derived work. + + +KERBEROS SUPPORT CODE + +The Kerberos support code ("KSC") is copyright 2006 by Jelmer +Vernooij and is provided 'as-is', without any express or implied +warranty. In no event will the author or Apple Inc. be held +liable for any damages arising from the use of the KSC. + +Sources files containing KSC have the following text at the top +of each source file: + + This file contains Kerberos support code, copyright 2006 by + Jelmer Vernooij. + +The KSC copyright and license apply only to Kerberos-related +feature code in CUPS. Such code is typically conditionally +compiled based on the present of the HAVE_GSSAPI preprocessor +definition. + +Permission is granted to anyone to use the KSC for any purpose, +including commercial applications, and to alter it and +redistribute it freely, subject to the following restrictions: + + 1. The origin of the KSC must not be misrepresented; you + must not claim that you wrote the original software. If + you use the KSC in a product, an acknowledgment in the + product documentation would be appreciated but is not + required. + + 2. Altered source versions must be plainly marked as such, + and must not be misrepresented as being the original + software. + + 3. This notice may not be removed or altered from any source + distribution. + + +TRADEMARKS + +CUPS and the CUPS logo (the "CUPS Marks") are trademarks of Apple +Inc. Apple grants you a non-exclusive and non-transferable right +to use the CUPS Marks in any direct port or binary distribution +incorporating CUPS software and in any promotional material +therefor. You agree that your products will meet the highest +levels of quality and integrity for similar goods, not be unlawful, +and be developed, manufactured, and distributed in compliance with +this license. You will not interfere with Apple's rights in the +CUPS Marks, and all use of the CUPS Marks shall inure to the +benefit of Apple. This license does not apply to use of the CUPS +Marks in a derivative products, which requires prior written +permission from Apple Inc. + + GNU GENERAL PUBLIC LICENSE + Version 2, June 1991 + + Copyright (C) 1989, 1991 Free Software Foundation, Inc. + 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The licenses for most software are designed to take away your +freedom to share and change it. By contrast, the GNU General Public +License is intended to guarantee your freedom to share and change free +software--to make sure the software is free for all its users. This +General Public License applies to most of the Free Software +Foundation's software and to any other program whose authors commit to +using it. (Some other Free Software Foundation software is covered by +the GNU Library General Public License instead.) You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +this service if you wish), that you receive source code or can get it +if you want it, that you can change the software or use pieces of it +in new free programs; and that you know you can do these things. + + To protect your rights, we need to make restrictions that forbid +anyone to deny you these rights or to ask you to surrender the rights. +These restrictions translate to certain responsibilities for you if you +distribute copies of the software, or if you modify it. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must give the recipients all the rights that +you have. You must make sure that they, too, receive or can get the +source code. And you must show them these terms so they know their +rights. + + We protect your rights with two steps: (1) copyright the software, and +(2) offer you this license which gives you legal permission to copy, +distribute and/or modify the software. + + Also, for each author's protection and ours, we want to make certain +that everyone understands that there is no warranty for this free +software. If the software is modified by someone else and passed on, we +want its recipients to know that what they have is not the original, so +that any problems introduced by others will not reflect on the original +authors' reputations. + + Finally, any free program is threatened constantly by software +patents. We wish to avoid the danger that redistributors of a free +program will individually obtain patent licenses, in effect making the +program proprietary. To prevent this, we have made it clear that any +patent must be licensed for everyone's free use or not licensed at all. + + The precise terms and conditions for copying, distribution and +modification follow. + + GNU GENERAL PUBLIC LICENSE + TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION + + 0. This License applies to any program or other work which contains +a notice placed by the copyright holder saying it may be distributed +under the terms of this General Public License. The "Program", below, +refers to any such program or work, and a "work based on the Program" +means either the Program or any derivative work under copyright law: +that is to say, a work containing the Program or a portion of it, +either verbatim or with modifications and/or translated into another +language. (Hereinafter, translation is included without limitation in +the term "modification".) Each licensee is addressed as "you". + +Activities other than copying, distribution and modification are not +covered by this License; they are outside its scope. The act of +running the Program is not restricted, and the output from the Program +is covered only if its contents constitute a work based on the +Program (independent of having been made by running the Program). +Whether that is true depends on what the Program does. + + 1. You may copy and distribute verbatim copies of the Program's +source code as you receive it, in any medium, provided that you +conspicuously and appropriately publish on each copy an appropriate +copyright notice and disclaimer of warranty; keep intact all the +notices that refer to this License and to the absence of any warranty; +and give any other recipients of the Program a copy of this License +along with the Program. + +You may charge a fee for the physical act of transferring a copy, and +you may at your option offer warranty protection in exchange for a fee. + + 2. You may modify your copy or copies of the Program or any portion +of it, thus forming a work based on the Program, and copy and +distribute such modifications or work under the terms of Section 1 +above, provided that you also meet all of these conditions: + + a) You must cause the modified files to carry prominent notices + stating that you changed the files and the date of any change. + + b) You must cause any work that you distribute or publish, that in + whole or in part contains or is derived from the Program or any + part thereof, to be licensed as a whole at no charge to all third + parties under the terms of this License. + + c) If the modified program normally reads commands interactively + when run, you must cause it, when started running for such + interactive use in the most ordinary way, to print or display an + announcement including an appropriate copyright notice and a + notice that there is no warranty (or else, saying that you provide + a warranty) and that users may redistribute the program under + these conditions, and telling the user how to view a copy of this + License. (Exception: if the Program itself is interactive but + does not normally print such an announcement, your work based on + the Program is not required to print an announcement.) + +These requirements apply to the modified work as a whole. If +identifiable sections of that work are not derived from the Program, +and can be reasonably considered independent and separate works in +themselves, then this License, and its terms, do not apply to those +sections when you distribute them as separate works. But when you +distribute the same sections as part of a whole which is a work based +on the Program, the distribution of the whole must be on the terms of +this License, whose permissions for other licensees extend to the +entire whole, and thus to each and every part regardless of who wrote it. + +Thus, it is not the intent of this section to claim rights or contest +your rights to work written entirely by you; rather, the intent is to +exercise the right to control the distribution of derivative or +collective works based on the Program. + +In addition, mere aggregation of another work not based on the Program +with the Program (or with a work based on the Program) on a volume of +a storage or distribution medium does not bring the other work under +the scope of this License. + + 3. You may copy and distribute the Program (or a work based on it, +under Section 2) in object code or executable form under the terms of +Sections 1 and 2 above provided that you also do one of the following: + + a) Accompany it with the complete corresponding machine-readable + source code, which must be distributed under the terms of Sections + 1 and 2 above on a medium customarily used for software interchange; or, + + b) Accompany it with a written offer, valid for at least three + years, to give any third party, for a charge no more than your + cost of physically performing source distribution, a complete + machine-readable copy of the corresponding source code, to be + distributed under the terms of Sections 1 and 2 above on a medium + customarily used for software interchange; or, + + c) Accompany it with the information you received as to the offer + to distribute corresponding source code. (This alternative is + allowed only for noncommercial distribution and only if you + received the program in object code or executable form with such + an offer, in accord with Subsection b above.) + +The source code for a work means the preferred form of the work for +making modifications to it. For an executable work, complete source +code means all the source code for all modules it contains, plus any +associated interface definition files, plus the scripts used to +control compilation and installation of the executable. However, as a +special exception, the source code distributed need not include +anything that is normally distributed (in either source or binary +form) with the major components (compiler, kernel, and so on) of the +operating system on which the executable runs, unless that component +itself accompanies the executable. + +If distribution of executable or object code is made by offering +access to copy from a designated place, then offering equivalent +access to copy the source code from the same place counts as +distribution of the source code, even though third parties are not +compelled to copy the source along with the object code. + + 4. You may not copy, modify, sublicense, or distribute the Program +except as expressly provided under this License. Any attempt +otherwise to copy, modify, sublicense or distribute the Program is +void, and will automatically terminate your rights under this License. +However, parties who have received copies, or rights, from you under +this License will not have their licenses terminated so long as such +parties remain in full compliance. + + 5. You are not required to accept this License, since you have not +signed it. However, nothing else grants you permission to modify or +distribute the Program or its derivative works. These actions are +prohibited by law if you do not accept this License. Therefore, by +modifying or distributing the Program (or any work based on the +Program), you indicate your acceptance of this License to do so, and +all its terms and conditions for copying, distributing or modifying +the Program or works based on it. + + 6. Each time you redistribute the Program (or any work based on the +Program), the recipient automatically receives a license from the +original licensor to copy, distribute or modify the Program subject to +these terms and conditions. You may not impose any further +restrictions on the recipients' exercise of the rights granted herein. +You are not responsible for enforcing compliance by third parties to +this License. + + 7. If, as a consequence of a court judgment or allegation of patent +infringement or for any other reason (not limited to patent issues), +conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot +distribute so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you +may not distribute the Program at all. For example, if a patent +license would not permit royalty-free redistribution of the Program by +all those who receive copies directly or indirectly through you, then +the only way you could satisfy both it and this License would be to +refrain entirely from distribution of the Program. + +If any portion of this section is held invalid or unenforceable under +any particular circumstance, the balance of the section is intended to +apply and the section as a whole is intended to apply in other +circumstances. + +It is not the purpose of this section to induce you to infringe any +patents or other property right claims or to contest validity of any +such claims; this section has the sole purpose of protecting the +integrity of the free software distribution system, which is +implemented by public license practices. Many people have made +generous contributions to the wide range of software distributed +through that system in reliance on consistent application of that +system; it is up to the author/donor to decide if he or she is willing +to distribute software through any other system and a licensee cannot +impose that choice. + +This section is intended to make thoroughly clear what is believed to +be a consequence of the rest of this License. + + 8. If the distribution and/or use of the Program is restricted in +certain countries either by patents or by copyrighted interfaces, the +original copyright holder who places the Program under this License +may add an explicit geographical distribution limitation excluding +those countries, so that distribution is permitted only in or among +countries not thus excluded. In such case, this License incorporates +the limitation as if written in the body of this License. + + 9. The Free Software Foundation may publish revised and/or new versions +of the General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + +Each version is given a distinguishing version number. If the Program +specifies a version number of this License which applies to it and "any +later version", you have the option of following the terms and conditions +either of that version or of any later version published by the Free +Software Foundation. If the Program does not specify a version number of +this License, you may choose any version ever published by the Free Software +Foundation. + + 10. If you wish to incorporate parts of the Program into other free +programs whose distribution conditions are different, write to the author +to ask for permission. For software which is copyrighted by the Free +Software Foundation, write to the Free Software Foundation; we sometimes +make exceptions for this. Our decision will be guided by the two goals +of preserving the free status of all derivatives of our free software and +of promoting the sharing and reuse of software generally. + + NO WARRANTY + + 11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY +FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN +OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES +PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED +OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF +MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS +TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE +PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, +REPAIR OR CORRECTION. + + 12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR +REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, +INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING +OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED +TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY +YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER +PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE +POSSIBILITY OF SUCH DAMAGES. + + END OF TERMS AND CONDITIONS + + Appendix: How to Apply These Terms to Your New Programs + + If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + + To do so, attach the following notices to the program. It is safest +to attach them to the start of each source file to most effectively +convey the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + + + Copyright (C) 19yy + + This program is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 2 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program; if not, write to the Free Software + Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. + +Also add information on how to contact you by electronic and paper mail. + +If the program is interactive, make it output a short notice like this +when it starts in an interactive mode: + + Gnomovision version 69, Copyright (C) 19yy name of author + Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'. + This is free software, and you are welcome to redistribute it + under certain conditions; type `show c' for details. + +The hypothetical commands `show w' and `show c' should show the appropriate +parts of the General Public License. Of course, the commands you use may +be called something other than `show w' and `show c'; they could even be +mouse-clicks or menu items--whatever suits your program. + +You should also get your employer (if you work as a programmer) or your +school, if any, to sign a "copyright disclaimer" for the program, if +necessary. Here is a sample; alter the names: + + Yoyodyne, Inc., hereby disclaims all copyright interest in the program + `Gnomovision' (which makes passes at compilers) written by James Hacker. + + , 1 April 1989 + Ty Coon, President of Vice + +This General Public License does not permit incorporating your program into +proprietary programs. If your program is a subroutine library, you may +consider it more useful to permit linking proprietary applications with the +library. If this is what you want to do, use the GNU Library General +Public License instead of this License. + + GNU LIBRARY GENERAL PUBLIC LICENSE + Version 2, June 1991 + + Copyright (C) 1991 Free Software Foundation, Inc. + 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + [This is the first released version of the library GPL. It is + numbered 2 because it goes with version 2 of the ordinary GPL.] + + Preamble + + The licenses for most software are designed to take away your +freedom to share and change it. By contrast, the GNU General Public +Licenses are intended to guarantee your freedom to share and change +free software--to make sure the software is free for all its users. + + This license, the Library General Public License, applies to some +specially designated Free Software Foundation software, and to any +other libraries whose authors decide to use it. You can use it for +your libraries, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +this service if you wish), that you receive source code or can get it +if you want it, that you can change the software or use pieces of it +in new free programs; and that you know you can do these things. + + To protect your rights, we need to make restrictions that forbid +anyone to deny you these rights or to ask you to surrender the rights. +These restrictions translate to certain responsibilities for you if +you distribute copies of the library, or if you modify it. + + For example, if you distribute copies of the library, whether gratis +or for a fee, you must give the recipients all the rights that we gave +you. You must make sure that they, too, receive or can get the source +code. If you link a program with the library, you must provide +complete object files to the recipients so that they can relink them +with the library, after making changes to the library and recompiling +it. And you must show them these terms so they know their rights. + + Our method of protecting your rights has two steps: (1) copyright +the library, and (2) offer you this license which gives you legal +permission to copy, distribute and/or modify the library. + + Also, for each distributor's protection, we want to make certain +that everyone understands that there is no warranty for this free +library. If the library is modified by someone else and passed on, we +want its recipients to know that what they have is not the original +version, so that any problems introduced by others will not reflect on +the original authors' reputations. + + Finally, any free program is threatened constantly by software +patents. We wish to avoid the danger that companies distributing free +software will individually obtain patent licenses, thus in effect +transforming the program into proprietary software. To prevent this, +we have made it clear that any patent must be licensed for everyone's +free use or not licensed at all. + + Most GNU software, including some libraries, is covered by the ordinary +GNU General Public License, which was designed for utility programs. This +license, the GNU Library General Public License, applies to certain +designated libraries. This license is quite different from the ordinary +one; be sure to read it in full, and don't assume that anything in it is +the same as in the ordinary license. + + The reason we have a separate public license for some libraries is that +they blur the distinction we usually make between modifying or adding to a +program and simply using it. Linking a program with a library, without +changing the library, is in some sense simply using the library, and is +analogous to running a utility program or application program. However, in +a textual and legal sense, the linked executable is a combined work, a +derivative of the original library, and the ordinary General Public License +treats it as such. + + Because of this blurred distinction, using the ordinary General +Public License for libraries did not effectively promote software +sharing, because most developers did not use the libraries. We +concluded that weaker conditions might promote sharing better. + + However, unrestricted linking of non-free programs would deprive the +users of those programs of all benefit from the free status of the +libraries themselves. This Library General Public License is intended to +permit developers of non-free programs to use free libraries, while +preserving your freedom as a user of such programs to change the free +libraries that are incorporated in them. (We have not seen how to achieve +this as regards changes in header files, but we have achieved it as regards +changes in the actual functions of the Library.) The hope is that this +will lead to faster development of free libraries. + + The precise terms and conditions for copying, distribution and +modification follow. Pay close attention to the difference between a +"work based on the library" and a "work that uses the library". The +former contains code derived from the library, while the latter only +works together with the library. + + Note that it is possible for a library to be covered by the ordinary +General Public License rather than by this special one. + + GNU LIBRARY GENERAL PUBLIC LICENSE + TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION + + 0. This License Agreement applies to any software library which +contains a notice placed by the copyright holder or other authorized +party saying it may be distributed under the terms of this Library +General Public License (also called "this License"). Each licensee is +addressed as "you". + + A "library" means a collection of software functions and/or data +prepared so as to be conveniently linked with application programs +(which use some of those functions and data) to form executables. + + The "Library", below, refers to any such software library or work +which has been distributed under these terms. A "work based on the +Library" means either the Library or any derivative work under +copyright law: that is to say, a work containing the Library or a +portion of it, either verbatim or with modifications and/or translated +straightforwardly into another language. (Hereinafter, translation is +included without limitation in the term "modification".) + + "Source code" for a work means the preferred form of the work for +making modifications to it. For a library, complete source code means +all the source code for all modules it contains, plus any associated +interface definition files, plus the scripts used to control compilation +and installation of the library. + + Activities other than copying, distribution and modification are not +covered by this License; they are outside its scope. The act of +running a program using the Library is not restricted, and output from +such a program is covered only if its contents constitute a work based +on the Library (independent of the use of the Library in a tool for +writing it). Whether that is true depends on what the Library does +and what the program that uses the Library does. + + 1. You may copy and distribute verbatim copies of the Library's +complete source code as you receive it, in any medium, provided that +you conspicuously and appropriately publish on each copy an +appropriate copyright notice and disclaimer of warranty; keep intact +all the notices that refer to this License and to the absence of any +warranty; and distribute a copy of this License along with the +Library. + + You may charge a fee for the physical act of transferring a copy, +and you may at your option offer warranty protection in exchange for a +fee. + + 2. You may modify your copy or copies of the Library or any portion +of it, thus forming a work based on the Library, and copy and +distribute such modifications or work under the terms of Section 1 +above, provided that you also meet all of these conditions: + + a) The modified work must itself be a software library. + + b) You must cause the files modified to carry prominent notices + stating that you changed the files and the date of any change. + + c) You must cause the whole of the work to be licensed at no + charge to all third parties under the terms of this License. + + d) If a facility in the modified Library refers to a function or a + table of data to be supplied by an application program that uses + the facility, other than as an argument passed when the facility + is invoked, then you must make a good faith effort to ensure that, + in the event an application does not supply such function or + table, the facility still operates, and performs whatever part of + its purpose remains meaningful. + + (For example, a function in a library to compute square roots has + a purpose that is entirely well-defined independent of the + application. Therefore, Subsection 2d requires that any + application-supplied function or table used by this function must + be optional: if the application does not supply it, the square + root function must still compute square roots.) + +These requirements apply to the modified work as a whole. If +identifiable sections of that work are not derived from the Library, +and can be reasonably considered independent and separate works in +themselves, then this License, and its terms, do not apply to those +sections when you distribute them as separate works. But when you +distribute the same sections as part of a whole which is a work based +on the Library, the distribution of the whole must be on the terms of +this License, whose permissions for other licensees extend to the +entire whole, and thus to each and every part regardless of who wrote +it. + +Thus, it is not the intent of this section to claim rights or contest +your rights to work written entirely by you; rather, the intent is to +exercise the right to control the distribution of derivative or +collective works based on the Library. + +In addition, mere aggregation of another work not based on the Library +with the Library (or with a work based on the Library) on a volume of +a storage or distribution medium does not bring the other work under +the scope of this License. + + 3. You may opt to apply the terms of the ordinary GNU General Public +License instead of this License to a given copy of the Library. To do +this, you must alter all the notices that refer to this License, so +that they refer to the ordinary GNU General Public License, version 2, +instead of to this License. (If a newer version than version 2 of the +ordinary GNU General Public License has appeared, then you can specify +that version instead if you wish.) Do not make any other change in +these notices. + + Once this change is made in a given copy, it is irreversible for +that copy, so the ordinary GNU General Public License applies to all +subsequent copies and derivative works made from that copy. + + This option is useful when you wish to copy part of the code of +the Library into a program that is not a library. + + 4. You may copy and distribute the Library (or a portion or +derivative of it, under Section 2) in object code or executable form +under the terms of Sections 1 and 2 above provided that you accompany +it with the complete corresponding machine-readable source code, which +must be distributed under the terms of Sections 1 and 2 above on a +medium customarily used for software interchange. + + If distribution of object code is made by offering access to copy +from a designated place, then offering equivalent access to copy the +source code from the same place satisfies the requirement to +distribute the source code, even though third parties are not +compelled to copy the source along with the object code. + + 5. A program that contains no derivative of any portion of the +Library, but is designed to work with the Library by being compiled or +linked with it, is called a "work that uses the Library". Such a +work, in isolation, is not a derivative work of the Library, and +therefore falls outside the scope of this License. + + However, linking a "work that uses the Library" with the Library +creates an executable that is a derivative of the Library (because it +contains portions of the Library), rather than a "work that uses the +library". The executable is therefore covered by this License. +Section 6 states terms for distribution of such executables. + + When a "work that uses the Library" uses material from a header file +that is part of the Library, the object code for the work may be a +derivative work of the Library even though the source code is not. +Whether this is true is especially significant if the work can be +linked without the Library, or if the work is itself a library. The +threshold for this to be true is not precisely defined by law. + + If such an object file uses only numerical parameters, data +structure layouts and accessors, and small macros and small inline +functions (ten lines or less in length), then the use of the object +file is unrestricted, regardless of whether it is legally a derivative +work. (Executables containing this object code plus portions of the +Library will still fall under Section 6.) + + Otherwise, if the work is a derivative of the Library, you may +distribute the object code for the work under the terms of Section 6. +Any executables containing that work also fall under Section 6, +whether or not they are linked directly with the Library itself. + + 6. As an exception to the Sections above, you may also compile or +link a "work that uses the Library" with the Library to produce a +work containing portions of the Library, and distribute that work +under terms of your choice, provided that the terms permit +modification of the work for the customer's own use and reverse +engineering for debugging such modifications. + + You must give prominent notice with each copy of the work that the +Library is used in it and that the Library and its use are covered by +this License. You must supply a copy of this License. If the work +during execution displays copyright notices, you must include the +copyright notice for the Library among them, as well as a reference +directing the user to the copy of this License. Also, you must do one +of these things: + + a) Accompany the work with the complete corresponding + machine-readable source code for the Library including whatever + changes were used in the work (which must be distributed under + Sections 1 and 2 above); and, if the work is an executable linked + with the Library, with the complete machine-readable "work that + uses the Library", as object code and/or source code, so that the + user can modify the Library and then relink to produce a modified + executable containing the modified Library. (It is understood + that the user who changes the contents of definitions files in the + Library will not necessarily be able to recompile the application + to use the modified definitions.) + + b) Accompany the work with a written offer, valid for at + least three years, to give the same user the materials + specified in Subsection 6a, above, for a charge no more + than the cost of performing this distribution. + + c) If distribution of the work is made by offering access to copy + from a designated place, offer equivalent access to copy the above + specified materials from the same place. + + d) Verify that the user has already received a copy of these + materials or that you have already sent this user a copy. + + For an executable, the required form of the "work that uses the +Library" must include any data and utility programs needed for +reproducing the executable from it. However, as a special exception, +the source code distributed need not include anything that is normally +distributed (in either source or binary form) with the major +components (compiler, kernel, and so on) of the operating system on +which the executable runs, unless that component itself accompanies +the executable. + + It may happen that this requirement contradicts the license +restrictions of other proprietary libraries that do not normally +accompany the operating system. Such a contradiction means you cannot +use both them and the Library together in an executable that you +distribute. + + 7. You may place library facilities that are a work based on the +Library side-by-side in a single library together with other library +facilities not covered by this License, and distribute such a combined +library, provided that the separate distribution of the work based on +the Library and of the other library facilities is otherwise +permitted, and provided that you do these two things: + + a) Accompany the combined library with a copy of the same work + based on the Library, uncombined with any other library + facilities. This must be distributed under the terms of the + Sections above. + + b) Give prominent notice with the combined library of the fact + that part of it is a work based on the Library, and explaining + where to find the accompanying uncombined form of the same work. + + 8. You may not copy, modify, sublicense, link with, or distribute +the Library except as expressly provided under this License. Any +attempt otherwise to copy, modify, sublicense, link with, or +distribute the Library is void, and will automatically terminate your +rights under this License. However, parties who have received copies, +or rights, from you under this License will not have their licenses +terminated so long as such parties remain in full compliance. + + 9. You are not required to accept this License, since you have not +signed it. However, nothing else grants you permission to modify or +distribute the Library or its derivative works. These actions are +prohibited by law if you do not accept this License. Therefore, by +modifying or distributing the Library (or any work based on the +Library), you indicate your acceptance of this License to do so, and +all its terms and conditions for copying, distributing or modifying +the Library or works based on it. + + 10. Each time you redistribute the Library (or any work based on the +Library), the recipient automatically receives a license from the +original licensor to copy, distribute, link with or modify the Library +subject to these terms and conditions. You may not impose any further +restrictions on the recipients' exercise of the rights granted herein. +You are not responsible for enforcing compliance by third parties to +this License. + + 11. If, as a consequence of a court judgment or allegation of patent +infringement or for any other reason (not limited to patent issues), +conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot +distribute so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you +may not distribute the Library at all. For example, if a patent +license would not permit royalty-free redistribution of the Library by +all those who receive copies directly or indirectly through you, then +the only way you could satisfy both it and this License would be to +refrain entirely from distribution of the Library. + +If any portion of this section is held invalid or unenforceable under any +particular circumstance, the balance of the section is intended to apply, +and the section as a whole is intended to apply in other circumstances. + +It is not the purpose of this section to induce you to infringe any +patents or other property right claims or to contest validity of any +such claims; this section has the sole purpose of protecting the +integrity of the free software distribution system which is +implemented by public license practices. Many people have made +generous contributions to the wide range of software distributed +through that system in reliance on consistent application of that +system; it is up to the author/donor to decide if he or she is willing +to distribute software through any other system and a licensee cannot +impose that choice. + +This section is intended to make thoroughly clear what is believed to +be a consequence of the rest of this License. + + 12. If the distribution and/or use of the Library is restricted in +certain countries either by patents or by copyrighted interfaces, the +original copyright holder who places the Library under this License may add +an explicit geographical distribution limitation excluding those countries, +so that distribution is permitted only in or among countries not thus +excluded. In such case, this License incorporates the limitation as if +written in the body of this License. + + 13. The Free Software Foundation may publish revised and/or new +versions of the Library General Public License from time to time. +Such new versions will be similar in spirit to the present version, +but may differ in detail to address new problems or concerns. + +Each version is given a distinguishing version number. If the Library +specifies a version number of this License which applies to it and +"any later version", you have the option of following the terms and +conditions either of that version or of any later version published by +the Free Software Foundation. If the Library does not specify a +license version number, you may choose any version ever published by +the Free Software Foundation. + + 14. If you wish to incorporate parts of the Library into other free +programs whose distribution conditions are incompatible with these, +write to the author to ask for permission. For software which is +copyrighted by the Free Software Foundation, write to the Free +Software Foundation; we sometimes make exceptions for this. Our +decision will be guided by the two goals of preserving the free status +of all derivatives of our free software and of promoting the sharing +and reuse of software generally. + + NO WARRANTY + + 15. BECAUSE THE LIBRARY IS LICENSED FREE OF CHARGE, THERE IS NO +WARRANTY FOR THE LIBRARY, TO THE EXTENT PERMITTED BY APPLICABLE LAW. +EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR +OTHER PARTIES PROVIDE THE LIBRARY "AS IS" WITHOUT WARRANTY OF ANY +KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE +IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE +LIBRARY IS WITH YOU. SHOULD THE LIBRARY PROVE DEFECTIVE, YOU ASSUME +THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + + 16. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN +WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY +AND/OR REDISTRIBUTE THE LIBRARY AS PERMITTED ABOVE, BE LIABLE TO YOU +FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR +CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE +LIBRARY (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING +RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A +FAILURE OF THE LIBRARY TO OPERATE WITH ANY OTHER SOFTWARE), EVEN IF +SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH +DAMAGES. + + END OF TERMS AND CONDITIONS + + Appendix: How to Apply These Terms to Your New Libraries + + If you develop a new library, and you want it to be of the greatest +possible use to the public, we recommend making it free software that +everyone can redistribute and change. You can do so by permitting +redistribution under these terms (or, alternatively, under the terms of the +ordinary General Public License). + + To apply these terms, attach the following notices to the library. It is +safest to attach them to the start of each source file to most effectively +convey the exclusion of warranty; and each file should have at least the +"copyright" line and a pointer to where the full notice is found. + + + Copyright (C) + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + This library is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + Library General Public License for more details. + + You should have received a copy of the GNU Library General Public + License along with this library; if not, write to the Free + Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. + +Also add information on how to contact you by electronic and paper mail. + +You should also get your employer (if you work as a programmer) or your +school, if any, to sign a "copyright disclaimer" for the library, if +necessary. Here is a sample; alter the names: + + Yoyodyne, Inc., hereby disclaims all copyright interest in the + library `Frob' (a library for tweaking knobs) written by James Random Hacker. + + , 1 April 1990 + Ty Coon, President of Vice + +That's all there is to it! diff --git a/Makedefs.in b/Makedefs.in new file mode 100644 index 0000000000..db54231581 --- /dev/null +++ b/Makedefs.in @@ -0,0 +1,260 @@ +# +# "$Id$" +# +# Common makefile definitions for CUPS. +# +# Copyright 2007-2012 by Apple Inc. +# Copyright 1997-2007 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +# +# Programs... +# + +AR = @AR@ +AWK = @AWK@ +CC = @LIBTOOL@ @CC@ +CHMOD = @CHMOD@ +CXX = @LIBTOOL@ @CXX@ +DSO = @DSO@ +DSOXX = @DSOXX@ +HTMLDOC = @HTMLDOC@ +INSTALL = @INSTALL@ +LD = @LD@ +LIBTOOL = @LIBTOOL@ +LN = @LN@ -sf +MV = @MV@ +RANLIB = @RANLIB@ +RM = @RM@ -f +RMDIR = @RMDIR@ +SED = @SED@ +SHELL = /bin/sh + +# +# Installation programs... +# + +INSTALL_BIN = $(LIBTOOL) $(INSTALL) -c -m 555 @INSTALL_STRIP@ +INSTALL_CONFIG = $(INSTALL) -c -m @CUPS_CONFIG_FILE_PERM@ +INSTALL_DATA = $(INSTALL) -c -m 444 +INSTALL_DIR = $(INSTALL) -d +INSTALL_LIB = $(LIBTOOL) $(INSTALL) -c -m 555 @INSTALL_STRIP@ +INSTALL_MAN = $(INSTALL) -c -m 444 +INSTALL_SCRIPT = $(INSTALL) -c -m 555 + +# +# Default user, group, and system groups for the scheduler... +# + +CUPS_USER = @CUPS_USER@ +CUPS_GROUP = @CUPS_GROUP@ +CUPS_SYSTEM_GROUPS = @CUPS_SYSTEM_GROUPS@ +CUPS_PRIMARY_SYSTEM_GROUP = @CUPS_PRIMARY_SYSTEM_GROUP@ + +# +# Default permissions... +# + +CUPS_CONFIG_FILE_PERM = @CUPS_CONFIG_FILE_PERM@ +CUPS_LOG_FILE_PERM = @CUPS_LOG_FILE_PERM@ + +# +# Languages to install... +# + +LANGUAGES = @LANGUAGES@ +INSTALL_LANGUAGES = @INSTALL_LANGUAGES@ +UNINSTALL_LANGUAGES = @UNINSTALL_LANGUAGES@ + +# +# Libraries... +# + +LIBCUPS = @LIBCUPS@ +LIBCUPSCGI = @LIBCUPSCGI@ +LIBCUPSIMAGE = @LIBCUPSIMAGE@ +LIBCUPSMIME = @LIBCUPSMIME@ +LIBCUPSPPDC = @LIBCUPSPPDC@ +LIBCUPSSTATIC = @LIBCUPSSTATIC@ +LIBGSSAPI = @LIBGSSAPI@ +LIBMALLOC = @LIBMALLOC@ +LIBMXML = @LIBMXML@ +LIBPAPER = @LIBPAPER@ +LIBUSB = @LIBUSB@ +LIBWRAP = @LIBWRAP@ +LIBZ = @LIBZ@ + +# +# Install static libraries? +# + +INSTALLSTATIC = @INSTALLSTATIC@ + +# +# IPP backend aliases... +# + +IPPALIASES = @IPPALIASES@ + +# +# Install XPC backends? +# + +INSTALLXPC = @INSTALLXPC@ + +# +# Program options... +# +# ARCHFLAGS Defines the default architecture build options. +# OPTIM Defines the common compiler optimization/debugging options +# for all architectures. +# OPTIONS Defines other compile-time options (currently only -DDEBUG +# for extra debug info) +# + +ALL_CFLAGS = -I.. -D_CUPS_SOURCE $(CFLAGS) $(SSLFLAGS) \ + @LARGEFILE@ @PTHREAD_FLAGS@ $(OPTIONS) +ALL_CXXFLAGS = -I.. -D_CUPS_SOURCE $(CXXFLAGS) $(SSLFLAGS) \ + @LARGEFILE@ @PTHREAD_FLAGS@ $(OPTIONS) +ARCHFLAGS = @ARCHFLAGS@ +ARFLAGS = @ARFLAGS@ +BACKLIBS = @BACKLIBS@ +BUILDDIRS = @BUILDDIRS@ +CFLAGS = @CPPFLAGS@ @CFLAGS@ +COMMONLIBS = @LIBS@ +CXXFLAGS = @CPPFLAGS@ @CXXFLAGS@ +CXXLIBS = @CXXLIBS@ +DBUS_NOTIFIER = @DBUS_NOTIFIER@ +DBUS_NOTIFIERLIBS = @DBUS_NOTIFIERLIBS@ +DNSSD_BACKEND = @DNSSD_BACKEND@ +DSOFLAGS = -L../cups @DSOFLAGS@ +DSOLIBS = @DSOLIBS@ $(COMMONLIBS) +DNSSDLIBS = @DNSSDLIBS@ +LAUNCHDLIBS = @LAUNCHDLIBS@ +LDFLAGS = -L../cgi-bin -L../cups -L../filter -L../ppdc \ + -L../scheduler @LDARCHFLAGS@ \ + @LDFLAGS@ @RELROFLAGS@ @PIEFLAGS@ $(OPTIM) +LINKCUPS = @LINKCUPS@ $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(LIBZ) +LINKCUPSIMAGE = @LINKCUPSIMAGE@ +LIBS = $(LINKCUPS) $(COMMONLIBS) +OPTIM = @OPTIM@ +OPTIONS = +PAMLIBS = @PAMLIBS@ +SERVERLIBS = @SERVERLIBS@ +SSLFLAGS = @SSLFLAGS@ +SSLLIBS = @SSLLIBS@ +UNITTESTS = @UNITTESTS@ + + +# +# Directories... +# +# The first section uses the GNU names (which are *extremely* +# difficult to find in a makefile because they are lowercase...) +# We have to define these first because autoconf uses ${prefix} +# and ${exec_prefix} for most of the other directories... +# +# The "datarootdir" variable may not get defined if you are using +# a version of autoconf prior to 2.60. +# +# This is immediately followed by definition in ALL CAPS for the +# needed directories... +# + +bindir = @bindir@ +datadir = @datadir@ +datarootdir = @datarootdir@ +exec_prefix = @exec_prefix@ +includedir = @includedir@ +infodir = @infodir@ +libdir = @libdir@ +libexecdir = @libexecdir@ +localstatedir = @localstatedir@ +mandir = @mandir@ +oldincludedir = @oldincludedir@ +prefix = @prefix@ +privateinclude = @privateinclude@ +sbindir = @sbindir@ +sharedstatedir = @sharedstatedir@ +srcdir = @srcdir@ +sysconfdir = @sysconfdir@ +top_srcdir = @top_srcdir@ + +BUILDROOT = $(DSTROOT) + +AMANDIR = $(BUILDROOT)@AMANDIR@ +BINDIR = $(BUILDROOT)@bindir@ +BUNDLEDIR = @CUPS_BUNDLEDIR@ +CACHEDIR = $(BUILDROOT)@CUPS_CACHEDIR@ +DATADIR = $(BUILDROOT)@CUPS_DATADIR@ +DOCDIR = $(BUILDROOT)@CUPS_DOCROOT@ +ICONDIR = @ICONDIR@ +INCLUDEDIR = $(BUILDROOT)$(includedir) +INITDIR = @INITDIR@ +INITDDIR = @INITDDIR@ +LIBDIR = $(BUILDROOT)$(libdir) +LOCALEDIR = $(BUILDROOT)@CUPS_LOCALEDIR@ +LOGDIR = $(BUILDROOT)@CUPS_LOGDIR@ +MANDIR = $(BUILDROOT)@mandir@ +MENUDIR = @MENUDIR@ +PMANDIR = $(BUILDROOT)@PMANDIR@ +PRIVATEINCLUDE = $(BUILDROOT)@PRIVATEINCLUDE@ +RCLEVELS = @RCLEVELS@ +RCSTART = @RCSTART@ +RCSTOP = @RCSTOP@ +REQUESTS = $(BUILDROOT)@CUPS_REQUESTS@ +SBINDIR = $(BUILDROOT)@sbindir@ +SERVERBIN = $(BUILDROOT)@CUPS_SERVERBIN@ +SERVERROOT = $(BUILDROOT)@CUPS_SERVERROOT@ +SMFMANIFESTDIR = @SMFMANIFESTDIR@ +STATEDIR = $(BUILDROOT)@CUPS_STATEDIR@ +XINETD = @XINETD@ + +MAN1EXT = @MAN1EXT@ +MAN5EXT = @MAN5EXT@ +MAN7EXT = @MAN7EXT@ +MAN8EXT = @MAN8EXT@ +MAN8DIR = @MAN8DIR@ + +PAMDIR = @PAMDIR@ +PAMFILE = @PAMFILE@ + +DEFAULT_LAUNCHD_CONF = @DEFAULT_LAUNCHD_CONF@ +DBUSDIR = @DBUSDIR@ + + +# +# Rules... +# + +.SILENT: +.SUFFIXES: .1 .1.gz .1m .1m.gz .3 .3.gz .5 .5.gz .7 .7.gz .8 .8.gz .a .c .cxx .h .man .o .gz + +.c.o: + echo Compiling $<... + $(CC) $(ARCHFLAGS) $(OPTIM) $(ALL_CFLAGS) -c -o $@ $< + +.cxx.o: + echo Compiling $<... + $(CXX) $(ARCHFLAGS) $(OPTIM) $(ALL_CXXFLAGS) -c -o $@ $< + +.man.1 .man.1m .man.3 .man.5 .man.7 .man.8: + echo Linking $<... + $(RM) $@ + $(LN) $< $@ + +.man.1.gz .man.1m.gz .man.3.gz .man.5.gz .man.7.gz .man.8.gz .man.gz: + echo -n Compressing $<... + $(RM) $@ + gzip -v9 <$< >$@ + + +# +# End of "$Id$" +# diff --git a/Makefile b/Makefile new file mode 100644 index 0000000000..baf491d59f --- /dev/null +++ b/Makefile @@ -0,0 +1,334 @@ +# +# "$Id$" +# +# Top-level Makefile for CUPS. +# +# Copyright 2007-2010 by Apple Inc. +# Copyright 1997-2007 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +include Makedefs + + +# +# Directories to make... +# + +DIRS = cups test $(BUILDDIRS) + + +# +# Make all targets... +# + +all: + chmod +x cups-config + echo Using ARCHFLAGS="$(ARCHFLAGS)" + echo Using ALL_CFLAGS="$(ALL_CFLAGS)" + echo Using ALL_CXXFLAGS="$(ALL_CXXFLAGS)" + echo Using CC="$(CC)" + echo Using CXX="$(CC)" + echo Using DSOFLAGS="$(DSOFLAGS)" + echo Using LDFLAGS="$(LDFLAGS)" + echo Using LIBS="$(LIBS)" + for dir in $(DIRS); do\ + echo Making all in $$dir... ;\ + (cd $$dir ; $(MAKE) $(MFLAGS) all $(UNITTESTS)) || exit 1;\ + done + + +# +# Make library targets... +# + +libs: + echo Using ARCHFLAGS="$(ARCHFLAGS)" + echo Using ALL_CFLAGS="$(ALL_CFLAGS)" + echo Using ALL_CXXFLAGS="$(ALL_CXXFLAGS)" + echo Using CC="$(CC)" + echo Using CXX="$(CC)" + echo Using DSOFLAGS="$(DSOFLAGS)" + echo Using LDFLAGS="$(LDFLAGS)" + echo Using LIBS="$(LIBS)" + for dir in $(DIRS); do\ + echo Making libraries in $$dir... ;\ + (cd $$dir ; $(MAKE) $(MFLAGS) libs) || exit 1;\ + done + + +# +# Make unit test targets... +# + +unittests: + echo Using ARCHFLAGS="$(ARCHFLAGS)" + echo Using ALL_CFLAGS="$(ALL_CFLAGS)" + echo Using ALL_CXXFLAGS="$(ALL_CXXFLAGS)" + echo Using CC="$(CC)" + echo Using CXX="$(CC)" + echo Using DSOFLAGS="$(DSOFLAGS)" + echo Using LDFLAGS="$(LDFLAGS)" + echo Using LIBS="$(LIBS)" + for dir in $(DIRS); do\ + echo Making all in $$dir... ;\ + (cd $$dir ; $(MAKE) $(MFLAGS) unittests) || exit 1;\ + done + + +# +# Remove object and target files... +# + +clean: + for dir in $(DIRS); do\ + echo Cleaning in $$dir... ;\ + (cd $$dir; $(MAKE) $(MFLAGS) clean) || exit 1;\ + done + + +# +# Remove all non-distribution files... +# + +distclean: clean + $(RM) Makedefs config.h config.log config.status + $(RM) cups-config + $(RM) conf/cupsd.conf conf/mime.convs conf/pam.std conf/snmp.conf + $(RM) doc/help/ref-cupsd-conf.html doc/help/standard.html doc/index.html + $(RM) man/client.conf.man + $(RM) man/cups-deviced.man man/cups-driverd.man + $(RM) man/cups-lpd.man man/cupsaddsmb.man man/cupsd.man + $(RM) man/cupsd.conf.man man/drv.man man/lpoptions.man + $(RM) packaging/cups.list + $(RM) packaging/cups-desc.plist packaging/cups-info.plist + $(RM) templates/header.tmpl + $(RM) desktop/cups.desktop + $(RM) scheduler/cups.sh scheduler/cups-lpd.xinetd + $(RM) scheduler/org.cups.cups-lpd.plist scheduler/cups.xml + -$(RM) doc/*/index.html + -$(RM) templates/*/header.tmpl + -$(RM) -r autom4te*.cache clang cups/charmaps cups/locale driver/test + + +# +# Make dependencies +# + +depend: + for dir in $(DIRS); do\ + echo Making dependencies in $$dir... ;\ + (cd $$dir; $(MAKE) $(MFLAGS) depend) || exit 1;\ + done + + +# +# Run the clang.llvm.org static code analysis tool on the C sources. +# (at least checker-231 is required for scan-build to work this way) +# + +.PHONY: clang clang-changes +clang: + $(RM) -r clang + scan-build -V -k -o `pwd`/clang $(MAKE) $(MFLAGS) clean all +clang-changes: + scan-build -V -k -o `pwd`/clang $(MAKE) $(MFLAGS) all + + +# +# Generate a ctags file... +# + +ctags: + ctags -R . + + +# +# Install everything... +# + +install: install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: + echo Making all in cups... + (cd cups; $(MAKE) $(MFLAGS) all) + for dir in $(DIRS); do\ + echo Installing data files in $$dir... ;\ + (cd $$dir; $(MAKE) $(MFLAGS) install-data) || exit 1;\ + done + echo Installing cups-config script... + $(INSTALL_DIR) -m 755 $(BINDIR) + $(INSTALL_SCRIPT) cups-config $(BINDIR)/cups-config + + +# +# Install header files... +# + +install-headers: + for dir in $(DIRS); do\ + echo Installing header files in $$dir... ;\ + (cd $$dir; $(MAKE) $(MFLAGS) install-headers) || exit 1;\ + done + if test "x$(privateinclude)" != x; then \ + echo Installing config.h into $(PRIVATEINCLUDE)...; \ + $(INSTALL_DIR) -m 755 $(PRIVATEINCLUDE); \ + $(INSTALL_DATA) config.h $(PRIVATEINCLUDE)/config.h; \ + fi + + +# +# Install programs... +# + +install-exec: all + for dir in $(DIRS); do\ + echo Installing programs in $$dir... ;\ + (cd $$dir; $(MAKE) $(MFLAGS) install-exec) || exit 1;\ + done + + +# +# Install libraries... +# + +install-libs: libs + for dir in $(DIRS); do\ + echo Installing libraries in $$dir... ;\ + (cd $$dir; $(MAKE) $(MFLAGS) install-libs) || exit 1;\ + done + + +# +# Uninstall object and target files... +# + +uninstall: + for dir in $(DIRS); do\ + echo Uninstalling in $$dir... ;\ + (cd $$dir; $(MAKE) $(MFLAGS) uninstall) || exit 1;\ + done + echo Uninstalling cups-config script... + $(RM) $(BINDIR)/cups-config + -$(RMDIR) $(BINDIR) + + +# +# Run the test suite... +# + +test: all unittests + echo Running CUPS test suite... + cd test; ./run-stp-tests.sh + + +check: all unittests + echo Running CUPS test suite with defaults... + cd test; ./run-stp-tests.sh 1 0 n n + +debugcheck: all unittests + echo Running CUPS test suite with debug printfs... + cd test; ./run-stp-tests.sh 1 0 n y + + +# +# Create HTML documentation... +# + +apihelp: + for dir in cgi-bin cups filter ppdc scheduler; do\ + echo Generating API help in $$dir... ;\ + (cd $$dir; $(MAKE) $(MFLAGS) apihelp) || exit 1;\ + done + +framedhelp: + for dir in cgi-bin cups filter ppdc scheduler; do\ + echo Generating framed API help in $$dir... ;\ + (cd $$dir; $(MAKE) $(MFLAGS) framedhelp) || exit 1;\ + done + + +# +# Create an Xcode docset... +# + +docset: apihelp + echo Generating docset directory tree... + $(RM) -r org.cups.docset + mkdir -p org.cups.docset/Contents/Resources/Documentation/help + mkdir -p org.cups.docset/Contents/Resources/Documentation/images + cd man; $(MAKE) $(MFLAGS) html + cd doc; $(MAKE) $(MFLAGS) docset + cd cgi-bin; $(MAKE) $(MFLAGS) makedocset + cgi-bin/makedocset org.cups.docset \ + `svnversion . | sed -e '1,$$s/[a-zA-Z]//g'` \ + doc/help/api-*.tokens + $(RM) doc/help/api-*.tokens + echo Indexing docset... + /Developer/usr/bin/docsetutil index org.cups.docset + echo Generating docset archive and feed... + $(RM) org.cups.docset.atom + /Developer/usr/bin/docsetutil package --output org.cups.docset.xar \ + --atom org.cups.docset.atom \ + --download-url http://www.cups.org/org.cups.docset.xar \ + org.cups.docset + + +# +# Lines of code computation... +# + +sloc: + for dir in cups scheduler; do \ + (cd $$dir; $(MAKE) $(MFLAGS) sloc) || exit 1;\ + done + + +# +# Make software distributions using EPM (http://www.epmhome.org/)... +# + +EPMFLAGS = -v --output-dir dist $(EPMARCH) + +aix bsd deb depot inst pkg setld slackware swinstall tardist: + epm $(EPMFLAGS) -f $@ cups packaging/cups.list + +epm: + epm $(EPMFLAGS) -s packaging/installer.gif cups packaging/cups.list + +rpm: + epm $(EPMFLAGS) -f rpm -s packaging/installer.gif cups packaging/cups.list + +.PHONEY: dist +dist: all + $(RM) -r dist + $(MAKE) $(MFLAGS) epm + case `uname` in \ + *BSD*) $(MAKE) $(MFLAGS) bsd;; \ + Darwin*) $(MAKE) $(MFLAGS) osx;; \ + IRIX*) $(MAKE) $(MFLAGS) tardist;; \ + Linux*) test ! -x /usr/bin/rpm || $(MAKE) $(MFLAGS) rpm;; \ + SunOS*) $(MAKE) $(MFLAGS) pkg;; \ + esac + + +# +# Don't run top-level build targets in parallel... +# + +.NOTPARALLEL: + + +# +# End of "$Id$". +# diff --git a/README.txt b/README.txt new file mode 100644 index 0000000000..f278e4dec3 --- /dev/null +++ b/README.txt @@ -0,0 +1,164 @@ +README - CUPS v1.5.0 - 2011-07-25 +---------------------------------- + +Looking for compile instructions? Read the file "INSTALL.txt" +instead... + + +INTRODUCTION + + CUPS is a standards-based, open source printing system developed by Apple + Inc. for Mac OS® X and other UNIX®-like operating systems. CUPS uses the + Internet Printing Protocol ("IPP") and provides System V and Berkeley + command-line interfaces, a web interface, and a C API to manage printers and + print jobs. It supports printing to both local (parallel, serial, USB) and + networked printers, and printers can be shared from one computer to another, + even over the Internet! + + Internally, CUPS uses PostScript Printer Description ("PPD") files to + describe printer capabilities and features and a wide variety of generic + and device-specific programs to convert and print many types of files. + Sample drivers are included with CUPS to support many Dymo, EPSON, HP, + Intellitech, OKIDATA, and Zebra printers. Many more drivers are available + online and (in some cases) on the driver CD-ROM that came with your printer. + + CUPS is licensed under the GNU General Public License and GNU Library + General Public License versions 2. See the file "LICENSE.txt" for more + information. + + +READING THE DOCUMENTATION + + Once you have installed the software you can access the documentation (and + a bunch of other stuff) online at: + + http://localhost:631/ + + If you're having trouble getting that far, the documentation is located + under the "doc/help" directory. + + Please read the documentation before asking questions. + + +GETTING SUPPORT AND OTHER RESOURCES + + If you have problems, READ THE DOCUMENTATION FIRST! We also provide many + discussion forums which are available at: + + http://www.cups.org/newsgroups.php + + See the CUPS web site at "http://www.cups.org/" for other resources. + + +SETTING UP PRINTER QUEUES USING YOUR WEB BROWSER + + CUPS includes a web-based administration tool that allows you to manage + printers, classes, and jobs on your server. Open the following URL in your + browser to access the printer administration tools: + + http://localhost:631/admin/ + + DO NOT use the hostname for your machine - it will not work with the default + CUPS configuration. To enable administration access on other addresses, + check the "Allow Remote Administration" box and click on the "Change + Settings" button. + + You will be asked for the administration password (root or any other user in + the sys/system/root/admin/lpadmin group on your system) when performing any + administrative function. + + +SETTING UP PRINTER QUEUES FROM THE COMMAND-LINE + + CUPS works best with PPD (PostScript Printer Description) files. In a pinch + you can also use System V style printer interface scripts. + + CUPS includes several sample PPD files you can use: + + Driver PPD Name + ----------------------------- ------------------------------ + Dymo Label Printers drv:///sample.drv/dymo.ppd + Intellitech Intellibar drv:///sample.drv/intelbar.ppd + EPSON Stylus Color Series drv:///sample.drv/stcolor.ppd + EPSON Stylus Photo Series drv:///sample.drv/stphoto.ppd + EPSON Stylus New Color Series drv:///sample.drv/stcolor2.ppd + EPSON Stylus New Photo Series drv:///sample.drv/stphoto2.ppd + EPSON 9-pin Series drv:///sample.drv/epson9.ppd + EPSON 24-pin Series drv:///sample.drv/epson24.ppd + Generic PCL Laser Printer drv:///sample.drv/generpcl.ppd + Generic PostScript Printer drv:///sample.drv/generic.ppd + HP DeskJet Series drv:///sample.drv/deskjet.ppd + HP LaserJet Series drv:///sample.drv/laserjet.ppd + OKIDATA 9-Pin Series drv:///sample.drv/okidata9.ppd + OKIDATA 24-Pin Series drv:///sample.drv/okidat24.ppd + Zebra CPCL Label Printer drv:///sample.drv/zebracpl.ppd + Zebra EPL1 Label Printer drv:///sample.drv/zebraep1.ppd + Zebra EPL2 Label Printer drv:///sample.drv/zebraep2.ppd + Zebra ZPL Label Printer drv:///sample.drv/zebra.ppd + + Run the "lpinfo -m" command to list the available drivers: + + lpinfo -m + + Run the "lpinfo -v" command to list the available printers: + + lpinfo -v + + Then use the correct URI to add the printer using the "lpadmin" command: + + lpadmin -p printername -E -v device-uri -m ppd-name + + Network printers typically use "socket" or "lpd" URIs: + + lpadmin -p printername -E -v socket://11.22.33.44 -m ppd-name + lpadmin -p printername -E -v lpd://11.22.33.44/ -m ppd-name + + The sample drivers provide basic printing capabilities, but generally do not + exercise the full potential of the printers or CUPS. The CUPS web site + provides links and drivers: + + http://www.cups.org/ppd.php PPD files + http://www.cups.org/links.php Links to other drivers + + +PRINTING FILES + + CUPS provides both the System V "lp" and Berkeley "lpr" commands for + printing: + + lp filename + lpr filename + + Both the "lp" and "lpr" commands support printing options for the driver: + + lp -o media=A4 -o resolution=600dpi filename + lpr -o media=A4 -o resolution=600dpi filename + + CUPS recognizes many types of images files as well as PDF, PostScript, + HP-GL/2, and text files, so you can print those files directly rather than + through an application. + + If you have an application that generates output specifically for your + printer then you need to use the "-oraw" or "-l" options: + + lp -o raw filename + lpr -l filename + + This will prevent the filters from misinterpreting your print + file. + + +LEGAL STUFF + + CUPS is Copyright 2007-2011 by Apple Inc. CUPS and the CUPS logo are + trademarks of Apple Inc. + + The MD5 Digest code is Copyright 1999 Aladdin Enterprises. + + This software is based in part on the work of the Independent JPEG Group. + + CUPS is provided under the terms of version 2 of the GNU General Public + License and GNU Library General Public License. This program is distributed + in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even + the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + See the "doc/help/license.html" or "LICENSE.txt" files for more information. diff --git a/backend/Dependencies b/backend/Dependencies new file mode 100644 index 0000000000..4c20479773 --- /dev/null +++ b/backend/Dependencies @@ -0,0 +1,72 @@ +ipp.o: ipp.c backend-private.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/snmp-private.h \ + ../cups/backend.h ../cups/sidechannel.h ../cups/array-private.h +lpd.o: lpd.c ../cups/http-private.h ../config.h ../cups/http.h \ + ../cups/versioning.h ../cups/array.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/ipp.h backend-private.h \ + ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/language.h ../cups/string-private.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h ../cups/snmp-private.h ../cups/backend.h \ + ../cups/sidechannel.h +dnssd.o: dnssd.c backend-private.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/snmp-private.h \ + ../cups/backend.h ../cups/sidechannel.h +snmp.o: snmp.c backend-private.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/snmp-private.h \ + ../cups/backend.h ../cups/sidechannel.h +socket.o: socket.c ../cups/http-private.h ../config.h ../cups/http.h \ + ../cups/versioning.h ../cups/array.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/ipp.h backend-private.h \ + ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/language.h ../cups/string-private.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h ../cups/snmp-private.h ../cups/backend.h \ + ../cups/sidechannel.h +test1284.o: test1284.c ../cups/string-private.h ../config.h ieee1284.c \ + backend-private.h ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/snmp-private.h \ + ../cups/backend.h ../cups/sidechannel.h +testbackend.o: testbackend.c ../cups/string-private.h ../config.h \ + ../cups/cups.h ../cups/file.h ../cups/versioning.h ../cups/ipp.h \ + ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/sidechannel.h +testsupplies.o: testsupplies.c backend-private.h ../cups/cups-private.h \ + ../cups/cups.h ../cups/file.h ../cups/versioning.h ../cups/ipp.h \ + ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h ../cups/snmp-private.h ../cups/backend.h \ + ../cups/sidechannel.h +usb.o: usb.c backend-private.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/snmp-private.h \ + ../cups/backend.h ../cups/sidechannel.h usb-darwin.c \ + ../cups/file-private.h diff --git a/backend/Makefile b/backend/Makefile new file mode 100644 index 0000000000..601db7123d --- /dev/null +++ b/backend/Makefile @@ -0,0 +1,291 @@ +# +# "$Id$" +# +# Backend makefile for CUPS. +# +# Copyright 2007-2012 by Apple Inc. +# Copyright 1997-2007 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# +# This file is subject to the Apple OS-Developed Software exception. +# + +include ../Makedefs + +# +# Object files... +# + +RBACKENDS = \ + ipp \ + lpd \ + $(DNSSD_BACKEND) +UBACKENDS = \ + snmp \ + socket \ + usb +UNITTESTS = \ + test1284 \ + testbackend \ + testsupplies +TARGETS = \ + libbackend.a \ + $(RBACKENDS) \ + $(UBACKENDS) +LIBOBJS = \ + ieee1284.o \ + network.o \ + runloop.o \ + snmp-supplies.o +OBJS = \ + ipp.o \ + lpd.o \ + dnssd.o \ + snmp.o \ + socket.o \ + test1284.o \ + testbackend.o \ + testsupplies.o \ + usb.o + + +# +# Make all targets... +# + +all: $(TARGETS) + + +# +# Make library targets... +# + +libs: + + +# +# Make unit tests... +# + +unittests: $(UNITTESTS) + + +# +# Clean all object files... +# + +clean: + $(RM) $(OBJS) $(TARGETS) $(UNITTESTS) $(LIBOBJS) http https ipps mdns + + +# +# Update dependencies (without system header dependencies...) +# + +depend: + $(CC) -MM $(ALL_CFLAGS) $(OBJS:.o=.c) >Dependencies + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: + + +# +# Install programs... +# + +install-exec: $(INSTALLXPC) + echo Installing backends in $(SERVERBIN)/backend + $(INSTALL_DIR) -m 755 $(SERVERBIN)/backend + for file in $(RBACKENDS); do \ + $(LIBTOOL) $(INSTALL_BIN) -m 700 $$file $(SERVERBIN)/backend; \ + done + for file in $(UBACKENDS); do \ + $(INSTALL_BIN) $$file $(SERVERBIN)/backend; \ + done + for file in $(IPPALIASES); do \ + $(RM) $(SERVERBIN)/backend/$$file; \ + $(LN) ipp $(SERVERBIN)/backend/$$file; \ + done + if test "x$(DNSSD_BACKEND)" != x; then \ + $(RM) $(SERVERBIN)/backend/mdns; \ + $(LN) $(DNSSD_BACKEND) $(SERVERBIN)/backend/mdns; \ + fi + if test "x$(SYMROOT)" != "x"; then \ + $(INSTALL_DIR) $(SYMROOT); \ + for file in $(TARGETS); do \ + cp $$file $(SYMROOT); \ + done \ + fi + +install-xpc: ipp + echo Installing XPC backends in $(SERVERBIN)/apple + $(INSTALL_DIR) -m 755 $(SERVERBIN)/apple + $(LIBTOOL) $(INSTALL_BIN) ipp $(SERVERBIN)/apple + for file in $(IPPALIASES); do \ + $(RM) $(SERVERBIN)/apple/$$file; \ + $(LN) ipp $(SERVERBIN)/apple/$$file; \ + done + + +# +# Install headers... +# + +install-headers: + + +# +# Install libraries... +# + +install-libs: + + +# +# Uninstall all targets... +# + +uninstall: + $(RM) $(SERVERBIN)/apple/ipp + for file in $(IPPALIASES); do \ + $(RM) $(SERVERBIN)/apple/$$file; \ + done + -$(RMDIR) $(SERVERBIN)/apple + for file in $(RBACKENDS) $(UBACKENDS); do \ + $(RM) $(SERVERBIN)/backend/$$file; \ + done + for file in $(IPPALIASES); do \ + $(RM) $(SERVERBIN)/backend/$$file; \ + done + -$(RMDIR) $(SERVERBIN)/backend + -$(RMDIR) $(SERVERBIN) + + +# +# test1284 +# + +test1284: test1284.o ../cups/$(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(LDFLAGS) -o test1284 test1284.o ../cups/$(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + + +# +# testbackend +# + +testbackend: testbackend.o ../cups/$(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(LDFLAGS) -o testbackend testbackend.o ../cups/$(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + + +# +# testsupplies +# + +testsupplies: testsupplies.o libbackend.a ../cups/$(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(LDFLAGS) -o testsupplies testsupplies.o libbackend.a \ + ../cups/$(LIBCUPSSTATIC) $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) \ + $(COMMONLIBS) $(LIBZ) + + +# +# libbackend.a +# + +libbackend.a: $(LIBOBJS) + echo Archiving $@... + $(RM) $@ + $(AR) $(ARFLAGS) $@ $(LIBOBJS) + $(RANLIB) $@ + + +# +# dnssd +# + +dnssd: dnssd.o ../cups/$(LIBCUPS) libbackend.a + echo Linking $@... + $(CC) $(LDFLAGS) -o dnssd dnssd.o libbackend.a $(LIBS) + $(RM) mdns + $(LN) dnssd mdns + + +# +# ipp +# + +ipp: ipp.o ../cups/$(LIBCUPS) libbackend.a + echo Linking $@... + $(CC) $(LDFLAGS) -o ipp ipp.o libbackend.a $(LIBS) + $(RM) http + $(LN) ipp http + + +# +# lpd +# + +lpd: lpd.o ../cups/$(LIBCUPS) libbackend.a + echo Linking $@... + $(CC) $(LDFLAGS) -o lpd lpd.o libbackend.a $(LIBS) + + +# +# snmp +# + +snmp: snmp.o ../cups/$(LIBCUPS) libbackend.a + echo Linking $@... + $(CC) $(LDFLAGS) -o snmp snmp.o libbackend.a $(LIBS) + + +# +# socket +# + +socket: socket.o ../cups/$(LIBCUPS) libbackend.a + echo Linking $@... + $(CC) $(LDFLAGS) -o socket socket.o libbackend.a $(LIBS) + + +# +# usb +# + +usb: usb.o ../cups/$(LIBCUPS) libbackend.a + echo Linking $@... + $(CC) $(LDFLAGS) -o usb usb.o libbackend.a $(LIBUSB) \ + $(BACKLIBS) $(LIBS) +usb.o: usb.c usb-darwin.c usb-libusb.c usb-unix.c + + +# +# Dependencies... +# + +include Dependencies + + +# +# End of "$Id$". +# diff --git a/backend/backend-private.h b/backend/backend-private.h new file mode 100644 index 0000000000..30d0bd7bb5 --- /dev/null +++ b/backend/backend-private.h @@ -0,0 +1,324 @@ +/* + * "$Id$" + * + * Backend support definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_BACKEND_PRIVATE_H_ +# define _CUPS_BACKEND_PRIVATE_H_ + + +/* + * Include necessary headers. + */ + +# include +# include +# include +# include +# include + +# ifdef __linux +# include +# include +# define IOCNR_GET_DEVICE_ID 1 +# define LPIOC_GET_DEVICE_ID(len) _IOC(_IOC_READ, 'P', IOCNR_GET_DEVICE_ID, len) +# include +# include +# include +# include +# endif /* __linux */ + +# ifdef __sun +# ifdef __sparc +# include +# else +# include +# include +# endif /* __sparc */ +# endif /* __sun */ + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * OID constants... + */ + +/* Host MIB */ +#define CUPS_OID_mib2 1,3,6,1,2,1 + +#define CUPS_OID_system CUPS_OID_mib2,1 +#define CUPS_OID_sysLocation CUPS_OID_system,6 + +#define CUPS_OID_host CUPS_OID_mib2,25 + +#define CUPS_OID_hrSystem CUPS_OID_host,1 + +#define CUPS_OID_hrStorage CUPS_OID_host,2 + +#define CUPS_OID_hrDevice CUPS_OID_host,3 +#define CUPS_OID_hrDeviceTable CUPS_OID_hrDevice,2 +#define CUPS_OID_hrDeviceEntry CUPS_OID_hrDeviceTable,1 +#define CUPS_OID_hrDeviceIndex CUPS_OID_hrDeviceEntry,1 +#define CUPS_OID_hrDeviceType CUPS_OID_hrDeviceEntry,2 +#define CUPS_OID_hrDeviceDescr CUPS_OID_hrDeviceEntry,3 + +#define CUPS_OID_hrPrinterTable CUPS_OID_hrDevice,5 +#define CUPS_OID_hrPrinterEntry CUPS_OID_hrPrinterTable,1 +#define CUPS_OID_hrPrinterStatus CUPS_OID_hrPrinterEntry,1 +#define CUPS_OID_hrPrinterDetectedErrorState CUPS_OID_hrPrinterEntry,2 + +/* Printer MIB */ +#define CUPS_OID_printmib CUPS_OID_mib2,43 + +#define CUPS_OID_prtGeneral CUPS_OID_printmib,5 +#define CUPS_OID_prtGeneralTable CUPS_OID_prtGeneral,1 +#define CUPS_OID_prtGeneralEntry CUPS_OID_prtGeneralTable,1 +#define CUPS_OID_prtGeneralCurrentLocalization CUPS_OID_prtGeneralEntry,2 +#define CUPS_OID_prtGeneralPrinterName CUPS_OID_prtGeneralEntry,16 +#define CUPS_OID_prtGeneralSerialNumber CUPS_OID_prtGeneralEntry,17 + +#define CUPS_OID_prtCover CUPS_OID_printmib,6 +#define CUPS_OID_prtCoverTable CUPS_OID_prtCover,1 +#define CUPS_OID_prtCoverEntry CUPS_OID_prtCoverTable,1 +#define CUPS_OID_prtCoverDescription CUPS_OID_prtCoverEntry,2 +#define CUPS_OID_prtCoverStatus CUPS_OID_prtCoverEntry,3 + +#define CUPS_OID_prtLocalization CUPS_OID_printmib,7 +#define CUPS_OID_prtLocalizationTable CUPS_OID_prtLocalization,1 +#define CUPS_OID_prtLocalizationEntry CUPS_OID_prtLocalizationTable,1 +#define CUPS_OID_prtLocalizationCharacterSet CUPS_OID_prtLocalizationEntry,4 + +#define CUPS_OID_prtMarker CUPS_OID_printmib,10 +#define CUPS_OID_prtMarkerTable CUPS_OID_prtMarker,2 +#define CUPS_OID_prtMarkerEntry CUPS_OID_prtMarkerTable,1 +#define CUPS_OID_prtMarkerLifeCount CUPS_OID_prtMarkerEntry,4 + +#define CUPS_OID_prtMarkerSupplies CUPS_OID_printmib,11 +#define CUPS_OID_prtMarkerSuppliesTable CUPS_OID_prtMarkerSupplies,1 +#define CUPS_OID_prtMarkerSuppliesEntry CUPS_OID_prtMarkerSuppliesTable,1 +#define CUPS_OID_prtMarkerSuppliesIndex CUPS_OID_prtMarkerSuppliesEntry,1 +#define CUPS_OID_prtMarkerSuppliesMarkerIndex CUPS_OID_prtMarkerSuppliesEntry,2 +#define CUPS_OID_prtMarkerSuppliesColorantIndex CUPS_OID_prtMarkerSuppliesEntry,3 +#define CUPS_OID_prtMarkerSuppliesClass CUPS_OID_prtMarkerSuppliesEntry,4 +#define CUPS_OID_prtMarkerSuppliesType CUPS_OID_prtMarkerSuppliesEntry,5 +#define CUPS_OID_prtMarkerSuppliesDescription CUPS_OID_prtMarkerSuppliesEntry,6 +#define CUPS_OID_prtMarkerSuppliesSupplyUnit CUPS_OID_prtMarkerSuppliesEntry,7 +#define CUPS_OID_prtMarkerSuppliesMaxCapacity CUPS_OID_prtMarkerSuppliesEntry,8 +#define CUPS_OID_prtMarkerSuppliesLevel CUPS_OID_prtMarkerSuppliesEntry,9 + +#define CUPS_OID_prtMarkerColorant CUPS_OID_printmib,12 +#define CUPS_OID_prtMarkerColorantTable CUPS_OID_prtMarkerColorant,1 +#define CUPS_OID_prtMarkerColorantEntry CUPS_OID_prtMarkerColorantTable,1 +#define CUPS_OID_prtMarkerColorantIndex CUPS_OID_prtMarkerColorantEntry,1 +#define CUPS_OID_prtMarkerColorantMarkerIndex CUPS_OID_prtMarkerColorantEntry,2 +#define CUPS_OID_prtMarkerColorantRole CUPS_OID_prtMarkerColorantEntry,3 +#define CUPS_OID_prtMarkerColorantValue CUPS_OID_prtMarkerColorantEntry,4 +#define CUPS_OID_prtMarkerColorantTonality CUPS_OID_prtMarkerColorantEntry,5 + +#define CUPS_OID_prtInterpreter CUPS_OID_printmib,15 +#define CUPS_OID_prtInterpreterTable CUPS_OID_prtInterpreter,1 +#define CUPS_OID_prtInterpreterEntry CUPS_OID_prtInterpreterTable,1 +#define CUPS_OID_prtInterpreterLangFamily CUPS_OID_prtInterpreterEntry,2 +#define CUPS_OID_prtInterpreterLangLevel CUPS_OID_prtInterpreterEntry,3 + +/* Printer Port Monitor MIB */ +#define CUPS_OID_enterprises 1,3,6,1,4,1 +#define CUPS_OID_pwg CUPS_OID_enterprises,2699,1 +#define CUPS_OID_ppmMIB CUPS_OID_pwg,2 +#define CUPS_OID_ppmMIBObjects CUPS_OID_ppmMIB,1 + +#define CUPS_OID_ppmGeneral CUPS_OID_ppmMIBObjects,1 + +#define CUPS_OID_ppmPrinter CUPS_OID_ppmMIBObjects,2 +#define CUPS_OID_ppmPrinterTable CUPS_OID_ppmPrinter,1 +#define CUPS_OID_ppmPrinterEntry CUPS_OID_ppmPrinterTable,1 +#define CUPS_OID_ppmPrinterIndex CUPS_OID_ppmPrinterEntry,1 +#define CUPS_OID_ppmPrinterName CUPS_OID_ppmPrinterEntry,2 +#define CUPS_OID_ppmPrinterIEEE1284DeviceId CUPS_OID_ppmPrinterEntry,3 +#define CUPS_OID_ppmPrinterNumberOfPorts CUPS_OID_ppmPrinterEntry,4 +#define CUPS_OID_ppmPrinterPreferredPortIndex CUPS_OID_ppmPrinterEntry,5 +#define CUPS_OID_ppmPrinterHrDeviceIndex CUPS_OID_ppmPrinterEntry,6 +#define CUPS_OID_ppmPrinterSnmpCommunityName CUPS_OID_ppmPrinterEntry,7 +#define CUPS_OID_ppmPrinterSnmpQueryEnabled CUPS_OID_ppmPrinterEntry,8 + +#define CUPS_OID_ppmPort CUPS_OID_ppmMIBObjects,3 +#define CUPS_OID_ppmPortTable CUPS_OID_ppmPort,1 +#define CUPS_OID_ppmPortEntry CUPS_OID_ppmPortTable,1 +#define CUPS_OID_ppmPortIndex CUPS_OID_ppmPortEntry,1 +#define CUPS_OID_ppmPortEnabled CUPS_OID_ppmPortEntry,2 +#define CUPS_OID_ppmPortName CUPS_OID_ppmPortEntry,3 +#define CUPS_OID_ppmPortServiceNameOrURI CUPS_OID_ppmPortEntry,4 +#define CUPS_OID_ppmPortProtocolType CUPS_OID_ppmPortEntry,5 +#define CUPS_OID_ppmPortProtocolTargetPort CUPS_OID_ppmPortEntry,6 +#define CUPS_OID_ppmPortProtocolAltSourceEnabled CUPS_OID_ppmPortEntry,7 +#define CUPS_OID_ppmPortPrtChannelIndex CUPS_OID_ppmPortEntry,8 +#define CUPS_OID_ppmPortLprByteCountEnabled CUPS_OID_ppmPortEntry,9 + + +/* + * State constants... + */ + +#define CUPS_TC_other 1 +#define CUPS_TC_unknown 2 + +#define CUPS_TC_idle 3 +#define CUPS_TC_printing 4 +#define CUPS_TC_warmup 5 + +/* These come from the hrPrinterDetectedErrorState OCTET-STRING */ +#define CUPS_TC_lowPaper 0x8000 +#define CUPS_TC_noPaper 0x4000 +#define CUPS_TC_lowToner 0x2000 +#define CUPS_TC_noToner 0x1000 +#define CUPS_TC_doorOpen 0x0800 +#define CUPS_TC_jammed 0x0400 +#define CUPS_TC_offline 0x0200 +#define CUPS_TC_serviceRequested 0x0100 +#define CUPS_TC_inputTrayMissing 0x0080 +#define CUPS_TC_outputTrayMissing 0x0040 +#define CUPS_TC_markerSupplyMissing 0x0020 +#define CUPS_TC_outputNearFull 0x0010 +#define CUPS_TC_outputFull 0x0008 +#define CUPS_TC_inputTrayEmpty 0x0004 +#define CUPS_TC_overduePreventMaint 0x0002 + +#define CUPS_TC_prtCoverStatus_coverOpen 3 +#define CUPS_TC_prtCoverStatus_coverClosed 4 +#define CUPS_TC_prtCoverStatus_interlockOpen 5 +#define CUPS_TC_prtCoverStatus_interlockClosed 6 + +#define CUPS_TC_langPCL 3 +#define CUPS_TC_langHPGL 4 +#define CUPS_TC_langPJL 5 +#define CUPS_TC_langPS 6 +#define CUPS_TC_langEscapeP 9 +#define CUPS_TC_langCCITT 26 +#define CUPS_TC_langLIPS 39 +#define CUPS_TC_langTIFF 40 +#define CUPS_TC_langPCLXL 47 +#define CUPS_TC_langPDF 54 +#define CUPS_TC_langJPEG 61 + +#define CUPS_TC_supplyThatIsConsumed 3 +#define CUPS_TC_receptacleThatIsFilled 4 + +#define CUPS_TC_process 3 +#define CUPS_TC_spot 4 + +#define CUPS_TC_toner 3 +#define CUPS_TC_wasteToner 4 +#define CUPS_TC_ink 5 +#define CUPS_TC_inkCartridge 6 +#define CUPS_TC_inkRibbon 7 +#define CUPS_TC_wasteInk 8 +#define CUPS_TC_opc 9 +#define CUPS_TC_developer 10 +#define CUPS_TC_fuserOil 11 +#define CUPS_TC_solidWax 12 +#define CUPS_TC_ribbonWax 13 +#define CUPS_TC_wasteWax 14 +#define CUPS_TC_fuser 15 +#define CUPS_TC_coronaWire 16 +#define CUPS_TC_fuserOilWick 17 +#define CUPS_TC_cleanerUnit 18 +#define CUPS_TC_fuserCleaningPad 19 +#define CUPS_TC_transferUnit 20 +#define CUPS_TC_tonerCartridge 21 +#define CUPS_TC_fuserOiler 22 +#define CUPS_TC_water 23 +#define CUPS_TC_wasteWater 24 +#define CUPS_TC_glueWaterAdditive 25 +#define CUPS_TC_wastePaper 26 +#define CUPS_TC_bindingSupply 27 +#define CUPS_TC_bandingSupply 28 +#define CUPS_TC_stitchingWire 29 +#define CUPS_TC_shrinkWrap 30 +#define CUPS_TC_paperWrap 31 +#define CUPS_TC_staples 32 +#define CUPS_TC_inserts 33 +#define CUPS_TC_covers 34 + +/* These come from RFC 3808 to define character sets we support */ +/* Also see http://www.iana.org/assignments/character-sets */ +#define CUPS_TC_csASCII 3 +#define CUPS_TC_csISOLatin1 4 +#define CUPS_TC_csShiftJIS 17 +#define CUPS_TC_csUTF8 106 +#define CUPS_TC_csUnicode 1000 /* UCS2 BE */ +#define CUPS_TC_csUCS4 1001 /* UCS4 BE */ +#define CUPS_TC_csUnicodeASCII 1002 +#define CUPS_TC_csUnicodeLatin1 1003 +#define CUPS_TC_csUTF16BE 1013 +#define CUPS_TC_csUTF16LE 1014 +#define CUPS_TC_csUTF32 1017 +#define CUPS_TC_csUTF32BE 1018 +#define CUPS_TC_csUTF32LE 1019 +#define CUPS_TC_csWindows31J 2024 + + +/* + * Types... + */ + +typedef int (*_cups_sccb_t)(int print_fd, int device_fd, int snmp_fd, + http_addr_t *addr, int use_bc); + + +/* + * Prototypes... + */ + +extern void backendCheckSideChannel(int snmp_fd, http_addr_t *addr); +extern int backendDrainOutput(int print_fd, int device_fd); +extern int backendGetDeviceID(int fd, char *device_id, + int device_id_size, + char *make_model, + int make_model_size, + const char *scheme, char *uri, + int uri_size); +extern int backendGetMakeModel(const char *device_id, + char *make_model, + int make_model_size); +extern int backendNetworkSideCB(int print_fd, int device_fd, + int snmp_fd, http_addr_t *addr, + int use_bc); +extern ssize_t backendRunLoop(int print_fd, int device_fd, int snmp_fd, + http_addr_t *addr, int use_bc, + int update_state, _cups_sccb_t side_cb); +extern int backendSNMPSupplies(int snmp_fd, http_addr_t *addr, + int *page_count, + int *printer_state); +extern int backendWaitLoop(int snmp_fd, http_addr_t *addr, + int use_bc, _cups_sccb_t side_cb); + + +# ifdef __cplusplus +} +# endif /* __cplusplus */ +#endif /* !_CUPS_BACKEND_PRIVATE_H_ */ + + +/* + * End of "$Id$". + */ diff --git a/backend/dnssd.c b/backend/dnssd.c new file mode 100644 index 0000000000..fd86e08148 --- /dev/null +++ b/backend/dnssd.c @@ -0,0 +1,965 @@ +/* + * "$Id$" + * + * DNS-SD discovery backend for CUPS. + * + * Copyright 2008-2011 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Browse for printers. + * browse_callback() - Browse devices. + * browse_local_callback() - Browse local devices. + * compare_devices() - Compare two devices. + * exec_backend() - Execute the backend that corresponds to the + * resolved service name. + * get_device() - Create or update a device. + * query_callback() - Process query data. + * sigterm_handler() - Handle termination signals... + * unquote() - Unquote a name string. + */ + +/* + * Include necessary headers. + */ + +#include "backend-private.h" +#include +#include + + +/* + * Device structure... + */ + +typedef enum +{ + CUPS_DEVICE_PRINTER = 0, /* lpd://... */ + CUPS_DEVICE_IPP, /* ipp://... */ + CUPS_DEVICE_IPPS, /* ipps://... */ + CUPS_DEVICE_FAX_IPP, /* ipp://... */ + CUPS_DEVICE_PDL_DATASTREAM, /* socket://... */ + CUPS_DEVICE_RIOUSBPRINT /* riousbprint://... */ +} cups_devtype_t; + + +typedef struct +{ + DNSServiceRef ref; /* Service reference for resolve */ + char *name, /* Service name */ + *domain, /* Domain name */ + *fullName, /* Full name */ + *make_and_model, /* Make and model from TXT record */ + *device_id; /* 1284 device ID from TXT record */ + cups_devtype_t type; /* Device registration type */ + int priority, /* Priority associated with type */ + cups_shared, /* CUPS shared printer? */ + sent; /* Did we list the device? */ +} cups_device_t; + + +/* + * Local globals... + */ + +static int job_canceled = 0; + /* Set to 1 on SIGTERM */ + + +/* + * Local functions... + */ + +static void browse_callback(DNSServiceRef sdRef, + DNSServiceFlags flags, + uint32_t interfaceIndex, + DNSServiceErrorType errorCode, + const char *serviceName, + const char *regtype, + const char *replyDomain, void *context); +static void browse_local_callback(DNSServiceRef sdRef, + DNSServiceFlags flags, + uint32_t interfaceIndex, + DNSServiceErrorType errorCode, + const char *serviceName, + const char *regtype, + const char *replyDomain, + void *context); +static int compare_devices(cups_device_t *a, cups_device_t *b); +static void exec_backend(char **argv); +static cups_device_t *get_device(cups_array_t *devices, + const char *serviceName, + const char *regtype, + const char *replyDomain); +static void query_callback(DNSServiceRef sdRef, + DNSServiceFlags flags, + uint32_t interfaceIndex, + DNSServiceErrorType errorCode, + const char *fullName, uint16_t rrtype, + uint16_t rrclass, uint16_t rdlen, + const void *rdata, uint32_t ttl, + void *context); +static void sigterm_handler(int sig); +static void unquote(char *dst, const char *src, size_t dstsize); + + +/* + * 'main()' - Browse for printers. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + const char *name; /* Backend name */ + DNSServiceRef main_ref, /* Main service reference */ + fax_ipp_ref, /* IPP fax service reference */ + ipp_ref, /* IPP service reference */ + ipp_tls_ref, /* IPP w/TLS service reference */ + ipps_ref, /* IPP service reference */ + local_fax_ipp_ref, /* Local IPP fax service reference */ + local_ipp_ref, /* Local IPP service reference */ + local_ipp_tls_ref, /* Local IPP w/TLS service reference */ + local_ipps_ref, /* Local IPP service reference */ + local_printer_ref, /* Local LPD service reference */ + pdl_datastream_ref, /* AppSocket service reference */ + printer_ref, /* LPD service reference */ + riousbprint_ref; /* Remote IO service reference */ + int fd; /* Main file descriptor */ + fd_set input; /* Input set for select() */ + struct timeval timeout; /* Timeout for select() */ + cups_array_t *devices; /* Device array */ + cups_device_t *device; /* Current device */ + char uriName[1024]; /* Unquoted fullName for URI */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* Actions for POSIX signals */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ + + + /* + * Don't buffer stderr, and catch SIGTERM... + */ + + setbuf(stderr, NULL); + +#ifdef HAVE_SIGSET /* Use System V signals over POSIX to avoid bugs */ + sigset(SIGTERM, sigterm_handler); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + + sigemptyset(&action.sa_mask); + action.sa_handler = sigterm_handler; + sigaction(SIGTERM, &action, NULL); +#else + signal(SIGTERM, sigterm_handler); +#endif /* HAVE_SIGSET */ + + /* + * Check command-line... + */ + + if (argc >= 6) + exec_backend(argv); + else if (argc != 1) + { + _cupsLangPrintf(stderr, + _("Usage: %s job-id user title copies options [file]"), + argv[0]); + return (1); + } + + /* + * Only do discovery when run as "dnssd"... + */ + + if ((name = strrchr(argv[0], '/')) != NULL) + name ++; + else + name = argv[0]; + + if (strcmp(name, "dnssd")) + return (0); + + /* + * Create an array to track devices... + */ + + devices = cupsArrayNew((cups_array_func_t)compare_devices, NULL); + + /* + * Browse for different kinds of printers... + */ + + if (DNSServiceCreateConnection(&main_ref) != kDNSServiceErr_NoError) + { + perror("ERROR: Unable to create service connection"); + return (1); + } + + fd = DNSServiceRefSockFD(main_ref); + + fax_ipp_ref = main_ref; + DNSServiceBrowse(&fax_ipp_ref, kDNSServiceFlagsShareConnection, 0, + "_fax-ipp._tcp", NULL, browse_callback, devices); + + ipp_ref = main_ref; + DNSServiceBrowse(&ipp_ref, kDNSServiceFlagsShareConnection, 0, + "_ipp._tcp", NULL, browse_callback, devices); + + ipp_tls_ref = main_ref; + DNSServiceBrowse(&ipp_tls_ref, kDNSServiceFlagsShareConnection, 0, + "_ipp-tls._tcp", NULL, browse_callback, devices); + + ipps_ref = main_ref; + DNSServiceBrowse(&ipps_ref, kDNSServiceFlagsShareConnection, 0, + "_ipps._tcp", NULL, browse_callback, devices); + + local_fax_ipp_ref = main_ref; + DNSServiceBrowse(&local_fax_ipp_ref, kDNSServiceFlagsShareConnection, + kDNSServiceInterfaceIndexLocalOnly, + "_fax-ipp._tcp", NULL, browse_local_callback, devices); + + local_ipp_ref = main_ref; + DNSServiceBrowse(&local_ipp_ref, kDNSServiceFlagsShareConnection, + kDNSServiceInterfaceIndexLocalOnly, + "_ipp._tcp", NULL, browse_local_callback, devices); + + local_ipp_tls_ref = main_ref; + DNSServiceBrowse(&local_ipp_tls_ref, kDNSServiceFlagsShareConnection, + kDNSServiceInterfaceIndexLocalOnly, + "_ipp-tls._tcp", NULL, browse_local_callback, devices); + + local_ipps_ref = main_ref; + DNSServiceBrowse(&local_ipps_ref, kDNSServiceFlagsShareConnection, + kDNSServiceInterfaceIndexLocalOnly, + "_ipps._tcp", NULL, browse_local_callback, devices); + + local_printer_ref = main_ref; + DNSServiceBrowse(&local_printer_ref, kDNSServiceFlagsShareConnection, + kDNSServiceInterfaceIndexLocalOnly, + "_printer._tcp", NULL, browse_local_callback, devices); + + pdl_datastream_ref = main_ref; + DNSServiceBrowse(&pdl_datastream_ref, kDNSServiceFlagsShareConnection, 0, + "_pdl-datastream._tcp", NULL, browse_callback, devices); + + printer_ref = main_ref; + DNSServiceBrowse(&printer_ref, kDNSServiceFlagsShareConnection, 0, + "_printer._tcp", NULL, browse_callback, devices); + + riousbprint_ref = main_ref; + DNSServiceBrowse(&riousbprint_ref, kDNSServiceFlagsShareConnection, 0, + "_riousbprint._tcp", NULL, browse_callback, devices); + + /* + * Loop until we are killed... + */ + + while (!job_canceled) + { + FD_ZERO(&input); + FD_SET(fd, &input); + + timeout.tv_sec = 0; + timeout.tv_usec = 250000; + + if (select(fd + 1, &input, NULL, NULL, &timeout) < 0) + continue; + + if (FD_ISSET(fd, &input)) + { + /* + * Process results of our browsing... + */ + + DNSServiceProcessResult(main_ref); + } + else + { + /* + * Announce any devices we've found... + */ + + DNSServiceErrorType status; /* DNS query status */ + cups_device_t *best; /* Best matching device */ + char device_uri[1024]; /* Device URI */ + int count; /* Number of queries */ + int sent; /* Number of sent */ + + for (device = (cups_device_t *)cupsArrayFirst(devices), + best = NULL, count = 0, sent = 0; + device; + device = (cups_device_t *)cupsArrayNext(devices)) + { + if (device->sent) + sent ++; + + if (device->ref) + count ++; + + if (!device->ref && !device->sent) + { + /* + * Found the device, now get the TXT record(s) for it... + */ + + if (count < 20) + { + device->ref = main_ref; + + fprintf(stderr, "DEBUG: Querying \"%s\"...\n", device->fullName); + + status = DNSServiceQueryRecord(&(device->ref), + kDNSServiceFlagsShareConnection, + 0, device->fullName, + kDNSServiceType_TXT, + kDNSServiceClass_IN, query_callback, + devices); + if (status != kDNSServiceErr_NoError) + { + fputs("ERROR: Unable to query for TXT records!\n", stderr); + fprintf(stderr, "DEBUG: DNSServiceQueryRecord returned %d\n", + status); + } + else + count ++; + } + } + else if (!device->sent) + { + /* + * Got the TXT records, now report the device... + */ + + DNSServiceRefDeallocate(device->ref); + device->ref = 0; + + if (!best) + best = device; + else if (_cups_strcasecmp(best->name, device->name) || + _cups_strcasecmp(best->domain, device->domain)) + { + unquote(uriName, best->fullName, sizeof(uriName)); + + httpAssembleURI(HTTP_URI_CODING_ALL, device_uri, sizeof(device_uri), + "dnssd", NULL, uriName, 0, + best->cups_shared ? "/cups" : "/"); + + cupsBackendReport("network", device_uri, best->make_and_model, + best->name, best->device_id, NULL); + best->sent = 1; + best = device; + + sent ++; + } + else if (best->priority > device->priority || + (best->priority == device->priority && + best->type < device->type)) + { + best->sent = 1; + best = device; + + sent ++; + } + else + { + device->sent = 1; + + sent ++; + } + } + } + + if (best) + { + unquote(uriName, best->fullName, sizeof(uriName)); + + httpAssembleURI(HTTP_URI_CODING_ALL, device_uri, sizeof(device_uri), + "dnssd", NULL, uriName, 0, + best->cups_shared ? "/cups" : "/"); + + cupsBackendReport("network", device_uri, best->make_and_model, + best->name, best->device_id, NULL); + best->sent = 1; + sent ++; + } + + if (sent == cupsArrayCount(devices)) + break; + } + } + + return (CUPS_BACKEND_OK); +} + + +/* + * 'browse_callback()' - Browse devices. + */ + +static void +browse_callback( + DNSServiceRef sdRef, /* I - Service reference */ + DNSServiceFlags flags, /* I - Option flags */ + uint32_t interfaceIndex, /* I - Interface number */ + DNSServiceErrorType errorCode, /* I - Error, if any */ + const char *serviceName, /* I - Name of service/device */ + const char *regtype, /* I - Type of service */ + const char *replyDomain, /* I - Service domain */ + void *context) /* I - Devices array */ +{ + fprintf(stderr, "DEBUG2: browse_callback(sdRef=%p, flags=%x, " + "interfaceIndex=%d, errorCode=%d, serviceName=\"%s\", " + "regtype=\"%s\", replyDomain=\"%s\", context=%p)\n", + sdRef, flags, interfaceIndex, errorCode, + serviceName ? serviceName : "(null)", + regtype ? regtype : "(null)", + replyDomain ? replyDomain : "(null)", + context); + + /* + * Only process "add" data... + */ + + if (errorCode != kDNSServiceErr_NoError || !(flags & kDNSServiceFlagsAdd)) + return; + + /* + * Get the device... + */ + + get_device((cups_array_t *)context, serviceName, regtype, replyDomain); +} + + +/* + * 'browse_local_callback()' - Browse local devices. + */ + +static void +browse_local_callback( + DNSServiceRef sdRef, /* I - Service reference */ + DNSServiceFlags flags, /* I - Option flags */ + uint32_t interfaceIndex, /* I - Interface number */ + DNSServiceErrorType errorCode, /* I - Error, if any */ + const char *serviceName, /* I - Name of service/device */ + const char *regtype, /* I - Type of service */ + const char *replyDomain, /* I - Service domain */ + void *context) /* I - Devices array */ +{ + cups_device_t *device; /* Device */ + + + fprintf(stderr, "DEBUG2: browse_local_callback(sdRef=%p, flags=%x, " + "interfaceIndex=%d, errorCode=%d, serviceName=\"%s\", " + "regtype=\"%s\", replyDomain=\"%s\", context=%p)\n", + sdRef, flags, interfaceIndex, errorCode, + serviceName ? serviceName : "(null)", + regtype ? regtype : "(null)", + replyDomain ? replyDomain : "(null)", + context); + + /* + * Only process "add" data... + */ + + if (errorCode != kDNSServiceErr_NoError || !(flags & kDNSServiceFlagsAdd)) + return; + + /* + * Get the device... + */ + + device = get_device((cups_array_t *)context, serviceName, regtype, + replyDomain); + + /* + * Hide locally-registered devices... + */ + + fprintf(stderr, "DEBUG: Hiding local printer \"%s\"...\n", + device->fullName); + device->sent = 1; +} + + +/* + * 'compare_devices()' - Compare two devices. + */ + +static int /* O - Result of comparison */ +compare_devices(cups_device_t *a, /* I - First device */ + cups_device_t *b) /* I - Second device */ +{ + return (strcmp(a->name, b->name)); +} + + +/* + * 'exec_backend()' - Execute the backend that corresponds to the + * resolved service name. + */ + +static void +exec_backend(char **argv) /* I - Command-line arguments */ +{ + const char *resolved_uri, /* Resolved device URI */ + *cups_serverbin; /* Location of programs */ + char scheme[1024], /* Scheme from URI */ + *ptr, /* Pointer into scheme */ + filename[1024]; /* Backend filename */ + + + /* + * Resolve the device URI... + */ + + job_canceled = -1; + + while ((resolved_uri = cupsBackendDeviceURI(argv)) == NULL) + { + _cupsLangPrintFilter(stderr, "INFO", _("Unable to locate printer.")); + sleep(10); + + if (getenv("CLASS") != NULL) + exit(CUPS_BACKEND_FAILED); + } + + /* + * Extract the scheme from the URI... + */ + + strlcpy(scheme, resolved_uri, sizeof(scheme)); + if ((ptr = strchr(scheme, ':')) != NULL) + *ptr = '\0'; + + /* + * Get the filename of the backend... + */ + + if ((cups_serverbin = getenv("CUPS_SERVERBIN")) == NULL) + cups_serverbin = CUPS_SERVERBIN; + + snprintf(filename, sizeof(filename), "%s/backend/%s", cups_serverbin, scheme); + + /* + * Overwrite the device URI and run the new backend... + */ + + setenv("DEVICE_URI", resolved_uri, 1); + + argv[0] = (char *)resolved_uri; + + fprintf(stderr, "DEBUG: Executing backend \"%s\"...\n", filename); + + execv(filename, argv); + + fprintf(stderr, "ERROR: Unable to execute backend \"%s\": %s\n", filename, + strerror(errno)); + exit(CUPS_BACKEND_STOP); +} + + +/* + * 'get_device()' - Create or update a device. + */ + +static cups_device_t * /* O - Device */ +get_device(cups_array_t *devices, /* I - Device array */ + const char *serviceName, /* I - Name of service/device */ + const char *regtype, /* I - Type of service */ + const char *replyDomain) /* I - Service domain */ +{ + cups_device_t key, /* Search key */ + *device; /* Device */ + char fullName[kDNSServiceMaxDomainName]; + /* Full name for query */ + + + /* + * See if this is a new device... + */ + + key.name = (char *)serviceName; + + if (!strcmp(regtype, "_ipp._tcp.")) + key.type = CUPS_DEVICE_IPP; + else if (!strcmp(regtype, "_ipps._tcp.") || + !strcmp(regtype, "_ipp-tls._tcp.")) + key.type = CUPS_DEVICE_IPPS; + else if (!strcmp(regtype, "_fax-ipp._tcp.")) + key.type = CUPS_DEVICE_FAX_IPP; + else if (!strcmp(regtype, "_printer._tcp.")) + key.type = CUPS_DEVICE_PRINTER; + else if (!strcmp(regtype, "_pdl-datastream._tcp.")) + key.type = CUPS_DEVICE_PDL_DATASTREAM; + else + key.type = CUPS_DEVICE_RIOUSBPRINT; + + for (device = cupsArrayFind(devices, &key); + device; + device = cupsArrayNext(devices)) + if (_cups_strcasecmp(device->name, key.name)) + break; + else if (device->type == key.type) + { + if (!_cups_strcasecmp(device->domain, "local.") && + _cups_strcasecmp(device->domain, replyDomain)) + { + /* + * Update the .local listing to use the "global" domain name instead. + * The backend will try local lookups first, then the global domain name. + */ + + free(device->domain); + device->domain = strdup(replyDomain); + + DNSServiceConstructFullName(fullName, device->name, regtype, + replyDomain); + free(device->fullName); + device->fullName = strdup(fullName); + } + + return (device); + } + + /* + * Yes, add the device... + */ + + fprintf(stderr, "DEBUG: Found \"%s.%s%s\"...\n", serviceName, regtype, + replyDomain); + + device = calloc(sizeof(cups_device_t), 1); + device->name = strdup(serviceName); + device->domain = strdup(replyDomain); + device->type = key.type; + device->priority = 50; + + cupsArrayAdd(devices, device); + + /* + * Set the "full name" of this service, which is used for queries... + */ + + DNSServiceConstructFullName(fullName, serviceName, regtype, replyDomain); + device->fullName = strdup(fullName); + + return (device); +} + + +/* + * 'query_callback()' - Process query data. + */ + +static void +query_callback( + DNSServiceRef sdRef, /* I - Service reference */ + DNSServiceFlags flags, /* I - Data flags */ + uint32_t interfaceIndex, /* I - Interface */ + DNSServiceErrorType errorCode, /* I - Error, if any */ + const char *fullName, /* I - Full service name */ + uint16_t rrtype, /* I - Record type */ + uint16_t rrclass, /* I - Record class */ + uint16_t rdlen, /* I - Length of record data */ + const void *rdata, /* I - Record data */ + uint32_t ttl, /* I - Time-to-live */ + void *context) /* I - Devices array */ +{ + cups_array_t *devices; /* Device array */ + char name[1024], /* Service name */ + *ptr; /* Pointer into string */ + cups_device_t dkey, /* Search key */ + *device; /* Device */ + + + fprintf(stderr, "DEBUG2: query_callback(sdRef=%p, flags=%x, " + "interfaceIndex=%d, errorCode=%d, fullName=\"%s\", " + "rrtype=%u, rrclass=%u, rdlen=%u, rdata=%p, ttl=%u, " + "context=%p)\n", + sdRef, flags, interfaceIndex, errorCode, + fullName ? fullName : "(null)", rrtype, rrclass, rdlen, rdata, ttl, + context); + + /* + * Only process "add" data... + */ + + if (errorCode != kDNSServiceErr_NoError || !(flags & kDNSServiceFlagsAdd)) + return; + + /* + * Lookup the service in the devices array. + */ + + devices = (cups_array_t *)context; + dkey.name = name; + + unquote(name, fullName, sizeof(name)); + + if ((dkey.domain = strstr(name, "._tcp.")) != NULL) + dkey.domain += 6; + else + dkey.domain = (char *)"local."; + + if ((ptr = strstr(name, "._")) != NULL) + *ptr = '\0'; + + if (strstr(fullName, "_ipp._tcp.")) + dkey.type = CUPS_DEVICE_IPP; + else if (strstr(fullName, "_ipps._tcp.") || + strstr(fullName, "_ipp-tls._tcp.")) + dkey.type = CUPS_DEVICE_IPPS; + else if (strstr(fullName, "_fax-ipp._tcp.")) + dkey.type = CUPS_DEVICE_FAX_IPP; + else if (strstr(fullName, "_printer._tcp.")) + dkey.type = CUPS_DEVICE_PRINTER; + else if (strstr(fullName, "_pdl-datastream._tcp.")) + dkey.type = CUPS_DEVICE_PDL_DATASTREAM; + else + dkey.type = CUPS_DEVICE_RIOUSBPRINT; + + for (device = cupsArrayFind(devices, &dkey); + device; + device = cupsArrayNext(devices)) + { + if (_cups_strcasecmp(device->name, dkey.name) || + _cups_strcasecmp(device->domain, dkey.domain)) + { + device = NULL; + break; + } + else if (device->type == dkey.type) + { + /* + * Found it, pull out the priority and make and model from the TXT + * record and save it... + */ + + const uint8_t *data, /* Pointer into data */ + *datanext, /* Next key/value pair */ + *dataend; /* End of entire TXT record */ + uint8_t datalen; /* Length of current key/value pair */ + char key[256], /* Key string */ + value[256], /* Value string */ + make_and_model[512], + /* Manufacturer and model */ + model[256], /* Model */ + device_id[2048];/* 1284 device ID */ + + + device_id[0] = '\0'; + make_and_model[0] = '\0'; + + strcpy(model, "Unknown"); + + for (data = rdata, dataend = data + rdlen; + data < dataend; + data = datanext) + { + /* + * Read a key/value pair starting with an 8-bit length. Since the + * length is 8 bits and the size of the key/value buffers is 256, we + * don't need to check for overflow... + */ + + datalen = *data++; + + if (!datalen || (data + datalen) > dataend) + break; + + datanext = data + datalen; + + for (ptr = key; data < datanext && *data != '='; data ++) + *ptr++ = *data; + *ptr = '\0'; + + if (data < datanext && *data == '=') + { + data ++; + + if (data < datanext) + memcpy(value, data, datanext - data); + value[datanext - data] = '\0'; + + fprintf(stderr, "DEBUG2: query_callback: \"%s=%s\".\n", + key, value); + } + else + { + fprintf(stderr, "DEBUG2: query_callback: \"%s\" with no value.\n", + key); + continue; + } + + if (!_cups_strncasecmp(key, "usb_", 4)) + { + /* + * Add USB device ID information... + */ + + ptr = device_id + strlen(device_id); + snprintf(ptr, sizeof(device_id) - (ptr - device_id), "%s:%s;", + key + 4, value); + } + + if (!_cups_strcasecmp(key, "usb_MFG") || !_cups_strcasecmp(key, "usb_MANU") || + !_cups_strcasecmp(key, "usb_MANUFACTURER")) + strcpy(make_and_model, value); + else if (!_cups_strcasecmp(key, "usb_MDL") || !_cups_strcasecmp(key, "usb_MODEL")) + strcpy(model, value); + else if (!_cups_strcasecmp(key, "product") && !strstr(value, "Ghostscript")) + { + if (value[0] == '(') + { + /* + * Strip parenthesis... + */ + + if ((ptr = value + strlen(value) - 1) > value && *ptr == ')') + *ptr = '\0'; + + strcpy(model, value + 1); + } + else + strcpy(model, value); + } + else if (!_cups_strcasecmp(key, "ty")) + { + strcpy(model, value); + + if ((ptr = strchr(model, ',')) != NULL) + *ptr = '\0'; + } + else if (!_cups_strcasecmp(key, "priority")) + device->priority = atoi(value); + else if ((device->type == CUPS_DEVICE_IPP || + device->type == CUPS_DEVICE_IPPS || + device->type == CUPS_DEVICE_PRINTER) && + !_cups_strcasecmp(key, "printer-type")) + { + /* + * This is a CUPS printer! + */ + + device->cups_shared = 1; + + if (device->type == CUPS_DEVICE_PRINTER) + device->sent = 1; + } + } + + if (device->device_id) + free(device->device_id); + + if (!device_id[0] && strcmp(model, "Unknown")) + { + if (make_and_model[0]) + snprintf(device_id, sizeof(device_id), "MFG:%s;MDL:%s;", + make_and_model, model); + else if (!_cups_strncasecmp(model, "designjet ", 10)) + snprintf(device_id, sizeof(device_id), "MFG:HP;MDL:%s", model + 10); + else if (!_cups_strncasecmp(model, "stylus ", 7)) + snprintf(device_id, sizeof(device_id), "MFG:EPSON;MDL:%s", model + 7); + else if ((ptr = strchr(model, ' ')) != NULL) + { + /* + * Assume the first word is the make... + */ + + memcpy(make_and_model, model, ptr - model); + make_and_model[ptr - model] = '\0'; + + snprintf(device_id, sizeof(device_id), "MFG:%s;MDL:%s", + make_and_model, ptr + 1); + } + } + + if (device_id[0]) + device->device_id = strdup(device_id); + else + device->device_id = NULL; + + if (device->make_and_model) + free(device->make_and_model); + + if (make_and_model[0]) + { + strlcat(make_and_model, " ", sizeof(make_and_model)); + strlcat(make_and_model, model, sizeof(make_and_model)); + + device->make_and_model = strdup(make_and_model); + } + else + device->make_and_model = strdup(model); + break; + } + } + + if (!device) + fprintf(stderr, "DEBUG: Ignoring TXT record for \"%s\"...\n", fullName); +} + + +/* + * 'sigterm_handler()' - Handle termination signals... + */ + +static void +sigterm_handler(int sig) /* I - Signal number (unused) */ +{ + (void)sig; + + if (job_canceled) + exit(CUPS_BACKEND_OK); + else + job_canceled = 1; +} + + +/* + * 'unquote()' - Unquote a name string. + */ + +static void +unquote(char *dst, /* I - Destination buffer */ + const char *src, /* I - Source string */ + size_t dstsize) /* I - Size of destination buffer */ +{ + char *dstend = dst + dstsize - 1; /* End of destination buffer */ + + + while (*src && dst < dstend) + { + if (*src == '\\') + { + src ++; + if (isdigit(src[0] & 255) && isdigit(src[1] & 255) && + isdigit(src[2] & 255)) + { + *dst++ = ((((src[0] - '0') * 10) + src[1] - '0') * 10) + src[2] - '0'; + src += 3; + } + else + *dst++ = *src++; + } + else + *dst++ = *src ++; + } + + *dst = '\0'; +} + + +/* + * End of "$Id$". + */ diff --git a/backend/ieee1284.c b/backend/ieee1284.c new file mode 100644 index 0000000000..bc4b344a60 --- /dev/null +++ b/backend/ieee1284.c @@ -0,0 +1,491 @@ +/* + * "$Id$" + * + * IEEE-1284 support functions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * backendGetDeviceID() - Get the IEEE-1284 device ID string and + * corresponding URI. + * backendGetMakeModel() - Get the make and model string from the device ID. + */ + +/* + * Include necessary headers. + */ + +#include "backend-private.h" +#include + + +/* + * 'backendGetDeviceID()' - Get the IEEE-1284 device ID string and + * corresponding URI. + */ + +int /* O - 0 on success, -1 on failure */ +backendGetDeviceID( + int fd, /* I - File descriptor */ + char *device_id, /* O - 1284 device ID */ + int device_id_size, /* I - Size of buffer */ + char *make_model, /* O - Make/model */ + int make_model_size, /* I - Size of buffer */ + const char *scheme, /* I - URI scheme */ + char *uri, /* O - Device URI */ + int uri_size) /* I - Size of buffer */ +{ +#ifdef __APPLE__ /* This function is a no-op */ + (void)fd; + (void)device_id; + (void)device_id_size; + (void)make_model; + (void)make_model_size; + (void)scheme; + (void)uri; + (void)uri_size; + + return (-1); + +#else /* Get the device ID from the specified file descriptor... */ +# ifdef __linux + int length; /* Length of device ID info */ + int got_id = 0; +# endif /* __linux */ +# if defined(__sun) && defined(ECPPIOC_GETDEVID) + struct ecpp_device_id did; /* Device ID buffer */ +# endif /* __sun && ECPPIOC_GETDEVID */ + char *ptr; /* Pointer into device ID */ + + + DEBUG_printf(("backendGetDeviceID(fd=%d, device_id=%p, device_id_size=%d, " + "make_model=%p, make_model_size=%d, scheme=\"%s\", " + "uri=%p, uri_size=%d)\n", fd, device_id, device_id_size, + make_model, make_model_size, scheme ? scheme : "(null)", + uri, uri_size)); + + /* + * Range check input... + */ + + if (!device_id || device_id_size < 32) + { + DEBUG_puts("backendGetDeviceID: Bad args!"); + return (-1); + } + + if (make_model) + *make_model = '\0'; + + if (fd >= 0) + { + /* + * Get the device ID string... + */ + + *device_id = '\0'; + +# ifdef __linux + if (ioctl(fd, LPIOC_GET_DEVICE_ID(device_id_size), device_id)) + { + /* + * Linux has to implement things differently for every device it seems. + * Since the standard parallel port driver does not provide a simple + * ioctl() to get the 1284 device ID, we have to open the "raw" parallel + * device corresponding to this port and do some negotiation trickery + * to get the current device ID. + */ + + if (uri && !strncmp(uri, "parallel:/dev/", 14)) + { + char devparport[16]; /* /dev/parportN */ + int devparportfd, /* File descriptor for raw device */ + mode; /* Port mode */ + + + /* + * Since the Linux parallel backend only supports 4 parallel port + * devices, just grab the trailing digit and use it to construct a + * /dev/parportN filename... + */ + + snprintf(devparport, sizeof(devparport), "/dev/parport%s", + uri + strlen(uri) - 1); + + if ((devparportfd = open(devparport, O_RDWR | O_NOCTTY)) != -1) + { + /* + * Claim the device... + */ + + if (!ioctl(devparportfd, PPCLAIM)) + { + fcntl(devparportfd, F_SETFL, fcntl(devparportfd, F_GETFL) | O_NONBLOCK); + + mode = IEEE1284_MODE_COMPAT; + + if (!ioctl(devparportfd, PPNEGOT, &mode)) + { + /* + * Put the device into Device ID mode... + */ + + mode = IEEE1284_MODE_NIBBLE | IEEE1284_DEVICEID; + + if (!ioctl(devparportfd, PPNEGOT, &mode)) + { + /* + * Read the 1284 device ID... + */ + + if ((length = read(devparportfd, device_id, + device_id_size - 1)) >= 2) + { + device_id[length] = '\0'; + got_id = 1; + } + } + } + + /* + * Release the device... + */ + + ioctl(devparportfd, PPRELEASE); + } + + close(devparportfd); + } + } + } + else + got_id = 1; + + if (got_id) + { + /* + * Extract the length of the device ID string from the first two + * bytes. The 1284 spec says the length is stored MSB first... + */ + + length = (((unsigned)device_id[0] & 255) << 8) + + ((unsigned)device_id[1] & 255); + + /* + * Check to see if the length is larger than our buffer; first + * assume that the vendor incorrectly implemented the 1284 spec, + * and then limit the length to the size of our buffer... + */ + + if (length > device_id_size || length < 14) + length = (((unsigned)device_id[1] & 255) << 8) + + ((unsigned)device_id[0] & 255); + + if (length > device_id_size) + length = device_id_size; + + /* + * The length field counts the number of bytes in the string + * including the length field itself (2 bytes). The minimum + * length for a valid/usable device ID is 14 bytes: + * + * MFG: ;MDL: ; + * 2 + 4 + 1 + 5 + 1 + 1 + */ + + if (length < 14) + { + /* + * Can't use this device ID, so don't try to copy it... + */ + + device_id[0] = '\0'; + got_id = 0; + } + else + { + /* + * Copy the device ID text to the beginning of the buffer and + * nul-terminate. + */ + + length -= 2; + + memmove(device_id, device_id + 2, length); + device_id[length] = '\0'; + } + } + else + { + DEBUG_printf(("backendGetDeviceID: ioctl failed - %s\n", + strerror(errno))); + *device_id = '\0'; + } +# endif /* __linux */ + +# if defined(__sun) && defined(ECPPIOC_GETDEVID) + did.mode = ECPP_CENTRONICS; + did.len = device_id_size - 1; + did.rlen = 0; + did.addr = device_id; + + if (!ioctl(fd, ECPPIOC_GETDEVID, &did)) + { + /* + * Nul-terminate the device ID text. + */ + + if (did.rlen < (device_id_size - 1)) + device_id[did.rlen] = '\0'; + else + device_id[device_id_size - 1] = '\0'; + } +# ifdef DEBUG + else + DEBUG_printf(("backendGetDeviceID: ioctl failed - %s\n", + strerror(errno))); +# endif /* DEBUG */ +# endif /* __sun && ECPPIOC_GETDEVID */ + } + + /* + * Check whether device ID is valid. Turn line breaks and tabs to spaces and + * reject device IDs with non-printable characters. + */ + + for (ptr = device_id; *ptr; ptr ++) + if (_cups_isspace(*ptr)) + *ptr = ' '; + else if ((*ptr & 255) < ' ' || *ptr == 127) + { + DEBUG_printf(("backendGetDeviceID: Bad device_id character %d.", + *ptr & 255)); + *device_id = '\0'; + break; + } + + DEBUG_printf(("backendGetDeviceID: device_id=\"%s\"\n", device_id)); + + if (scheme && uri) + *uri = '\0'; + + if (!*device_id) + return (-1); + + /* + * Get the make and model... + */ + + if (make_model) + backendGetMakeModel(device_id, make_model, make_model_size); + + /* + * Then generate a device URI... + */ + + if (scheme && uri && uri_size > 32) + { + int num_values; /* Number of keys and values */ + cups_option_t *values; /* Keys and values in device ID */ + const char *mfg, /* Manufacturer */ + *mdl, /* Model */ + *sern; /* Serial number */ + char temp[256], /* Temporary manufacturer string */ + *tempptr; /* Pointer into temp string */ + + + /* + * Get the make, model, and serial numbers... + */ + + num_values = _cupsGet1284Values(device_id, &values); + + if ((sern = cupsGetOption("SERIALNUMBER", num_values, values)) == NULL) + if ((sern = cupsGetOption("SERN", num_values, values)) == NULL) + sern = cupsGetOption("SN", num_values, values); + + if ((mfg = cupsGetOption("MANUFACTURER", num_values, values)) == NULL) + mfg = cupsGetOption("MFG", num_values, values); + + if ((mdl = cupsGetOption("MODEL", num_values, values)) == NULL) + mdl = cupsGetOption("MDL", num_values, values); + + if (mfg) + { + if (!_cups_strcasecmp(mfg, "Hewlett-Packard")) + mfg = "HP"; + else if (!_cups_strcasecmp(mfg, "Lexmark International")) + mfg = "Lexmark"; + } + else + { + strlcpy(temp, make_model, sizeof(temp)); + + if ((tempptr = strchr(temp, ' ')) != NULL) + *tempptr = '\0'; + + mfg = temp; + } + + if (!mdl) + mdl = ""; + + if (!_cups_strncasecmp(mdl, mfg, strlen(mfg))) + { + mdl += strlen(mfg); + + while (isspace(*mdl & 255)) + mdl ++; + } + + /* + * Generate the device URI from the manufacturer, make_model, and + * serial number strings. + */ + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, uri_size, scheme, NULL, mfg, 0, + "/%s%s%s", mdl, sern ? "?serial=" : "", sern ? sern : ""); + + cupsFreeOptions(num_values, values); + } + + return (0); +#endif /* __APPLE__ */ +} + + +/* + * 'backendGetMakeModel()' - Get the make and model string from the device ID. + */ + +int /* O - 0 on success, -1 on failure */ +backendGetMakeModel( + const char *device_id, /* O - 1284 device ID */ + char *make_model, /* O - Make/model */ + int make_model_size) /* I - Size of buffer */ +{ + int num_values; /* Number of keys and values */ + cups_option_t *values; /* Keys and values */ + const char *mfg, /* Manufacturer string */ + *mdl, /* Model string */ + *des; /* Description string */ + + + DEBUG_printf(("backendGetMakeModel(device_id=\"%s\", " + "make_model=%p, make_model_size=%d)\n", device_id, + make_model, make_model_size)); + + /* + * Range check input... + */ + + if (!device_id || !*device_id || !make_model || make_model_size < 32) + { + DEBUG_puts("backendGetMakeModel: Bad args!"); + return (-1); + } + + *make_model = '\0'; + + /* + * Look for the description field... + */ + + num_values = _cupsGet1284Values(device_id, &values); + + if ((mdl = cupsGetOption("MODEL", num_values, values)) == NULL) + mdl = cupsGetOption("MDL", num_values, values); + + if (mdl) + { + /* + * Build a make-model string from the manufacturer and model attributes... + */ + + if ((mfg = cupsGetOption("MANUFACTURER", num_values, values)) == NULL) + mfg = cupsGetOption("MFG", num_values, values); + + if (!mfg || !_cups_strncasecmp(mdl, mfg, strlen(mfg))) + { + /* + * Just copy the model string, since it has the manufacturer... + */ + + _ppdNormalizeMakeAndModel(mdl, make_model, make_model_size); + } + else + { + /* + * Concatenate the make and model... + */ + + char temp[1024]; /* Temporary make and model */ + + snprintf(temp, sizeof(temp), "%s %s", mfg, mdl); + + _ppdNormalizeMakeAndModel(temp, make_model, make_model_size); + } + } + else if ((des = cupsGetOption("DESCRIPTION", num_values, values)) != NULL || + (des = cupsGetOption("DES", num_values, values)) != NULL) + { + /* + * Make sure the description contains something useful, since some + * printer manufacturers (HP) apparently don't follow the standards + * they helped to define... + * + * Here we require the description to be 8 or more characters in length, + * containing at least one space and one letter. + */ + + if (strlen(des) >= 8) + { + const char *ptr; /* Pointer into description */ + int letters, /* Number of letters seen */ + spaces; /* Number of spaces seen */ + + + for (ptr = des, letters = 0, spaces = 0; *ptr; ptr ++) + { + if (isspace(*ptr & 255)) + spaces ++; + else if (isalpha(*ptr & 255)) + letters ++; + + if (spaces && letters) + break; + } + + if (spaces && letters) + _ppdNormalizeMakeAndModel(des, make_model, make_model_size); + } + } + + if (!make_model[0]) + { + /* + * Use "Unknown" as the printer make and model... + */ + + strlcpy(make_model, "Unknown", make_model_size); + } + + cupsFreeOptions(num_values, values); + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/backend/ipp.c b/backend/ipp.c new file mode 100644 index 0000000000..62c14b7032 --- /dev/null +++ b/backend/ipp.c @@ -0,0 +1,3152 @@ +/* + * "$Id$" + * + * IPP backend for CUPS. + * + * Copyright 2007-2012 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Send a file to the printer or server. + * cancel_job() - Cancel a print job. + * check_printer_state() - Check the printer state. + * compress_files() - Compress print files. + * monitor_printer() - Monitor the printer state. + * new_request() - Create a new print creation or validation request. + * password_cb() - Disable the password prompt for + * cupsDoFileRequest(). + * report_attr() - Report an IPP attribute value. + * report_printer_state() - Report the printer state. + * run_as_user() - Run the IPP backend as the printing user. + * timeout_cb() - Handle HTTP timeouts. + * sigterm_handler() - Handle 'terminate' signals that stop the backend. + */ + +/* + * Include necessary headers. + */ + +#include "backend-private.h" +#include +#include +#include +#include +#if defined(HAVE_GSSAPI) && defined(HAVE_XPC) +# include +# define kPMPrintUIToolAgent "com.apple.printuitool.agent" +# define kPMStartJob 100 +# define kPMWaitForJob 101 +extern void xpc_connection_set_target_uid(xpc_connection_t connection, + uid_t uid); +#endif /* HAVE_GSSAPI && HAVE_XPC */ + + +/* + * Types... + */ + +typedef struct _cups_monitor_s /**** Monitoring data ****/ +{ + const char *uri, /* Printer URI */ + *hostname, /* Hostname */ + *user, /* Username */ + *resource; /* Resource path */ + int port, /* Port number */ + version, /* IPP version */ + job_id; /* Job ID for submitted job */ + const char *job_name; /* Job name for submitted job */ + http_encryption_t encryption; /* Use encryption? */ + ipp_jstate_t job_state; /* Current job state */ + ipp_pstate_t printer_state; /* Current printer state */ +} _cups_monitor_t; + + +/* + * Globals... + */ + +static const char *auth_info_required; + /* New auth-info-required value */ +#if defined(HAVE_GSSAPI) && defined(HAVE_XPC) +static int child_pid = 0; /* Child process ID */ +#endif /* HAVE_GSSAPI && HAVE_XPC */ +static const char * const jattrs[] = /* Job attributes we want */ +{ + "job-impressions-completed", + "job-media-sheets-completed", + "job-name", + "job-originating-user-name", + "job-state", + "job-state-reasons" +}; +static int job_canceled = 0; + /* Job cancelled? */ +static char *password = NULL; + /* Password for device URI */ +static int password_tries = 0; + /* Password tries */ +static const char * const pattrs[] = /* Printer attributes we want */ +{ + "copies-supported", + "cups-version", + "document-format-supported", + "marker-colors", + "marker-high-levels", + "marker-levels", + "marker-low-levels", + "marker-message", + "marker-names", + "marker-types", + "media-col-supported", + "multiple-document-handling-supported", + "operations-supported", + "printer-alert", + "printer-alert-description", + "printer-is-accepting-jobs", + "printer-state", + "printer-state-message", + "printer-state-reasons" +}; +static const char * const remote_job_states[] = +{ /* Remote job state keywords */ + "+cups-remote-pending", + "+cups-remote-pending-held", + "+cups-remote-processing", + "+cups-remote-stopped", + "+cups-remote-canceled", + "+cups-remote-aborted", + "+cups-remote-completed" +}; +static _cups_mutex_t report_mutex = _CUPS_MUTEX_INITIALIZER; + /* Mutex to control access */ +static int num_attr_cache = 0; + /* Number of cached attributes */ +static cups_option_t *attr_cache = NULL; + /* Cached attributes */ +static cups_array_t *state_reasons; /* Array of printe-state-reasons keywords */ +static char tmpfilename[1024] = ""; + /* Temporary spool file name */ + + +/* + * Local functions... + */ + +static void cancel_job(http_t *http, const char *uri, int id, + const char *resource, const char *user, + int version); +static ipp_pstate_t check_printer_state(http_t *http, const char *uri, + const char *resource, + const char *user, int version); +#ifdef HAVE_LIBZ +static void compress_files(int num_files, char **files); +#endif /* HAVE_LIBZ */ +static void *monitor_printer(_cups_monitor_t *monitor); +static ipp_t *new_request(ipp_op_t op, int version, const char *uri, + const char *user, const char *title, + int num_options, cups_option_t *options, + const char *compression, int copies, + const char *format, _ppd_cache_t *pc, + ipp_attribute_t *media_col_sup, + ipp_attribute_t *doc_handling_sup); +static const char *password_cb(const char *); +static void report_attr(ipp_attribute_t *attr); +static void report_printer_state(ipp_t *ipp); +#if defined(HAVE_GSSAPI) && defined(HAVE_XPC) +static int run_as_user(int argc, char *argv[], uid_t uid, + const char *device_uri, int fd); +#endif /* HAVE_GSSAPI && HAVE_XPC */ +static void sigterm_handler(int sig); +static int timeout_cb(http_t *http, void *user_data); +static void update_reasons(ipp_attribute_t *attr, const char *s); + + +/* + * 'main()' - Send a file to the printer or server. + * + * Usage: + * + * printer-uri job-id user title copies options [file] + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + int send_options; /* Send job options? */ + int num_options; /* Number of printer options */ + cups_option_t *options; /* Printer options */ + const char *device_uri; /* Device URI */ + char scheme[255], /* Scheme in URI */ + hostname[1024], /* Hostname */ + username[255], /* Username info */ + resource[1024], /* Resource info (printer name) */ + addrname[256], /* Address name */ + *optptr, /* Pointer to URI options */ + *name, /* Name of option */ + *value, /* Value of option */ + sep; /* Separator character */ + http_addrlist_t *addrlist; /* Address of printer */ + int snmp_fd, /* SNMP socket */ + start_count, /* Page count via SNMP at start */ + page_count, /* Page count via SNMP */ + have_supplies; /* Printer supports supply levels? */ + int num_files; /* Number of files to print */ + char **files, /* Files to print */ + *compatfile = NULL; /* Compatibility filename */ + off_t compatsize = 0; /* Size of compatibility file */ + int port; /* Port number (not used) */ + char portname[255]; /* Port name */ + char uri[HTTP_MAX_URI]; /* Updated URI without user/pass */ + char print_job_name[1024]; /* Update job-name for Print-Job */ + http_status_t http_status; /* Status of HTTP request */ + ipp_status_t ipp_status; /* Status of IPP request */ + http_t *http; /* HTTP connection */ + ipp_t *request, /* IPP request */ + *response, /* IPP response */ + *supported; /* get-printer-attributes response */ + time_t start_time; /* Time of first connect */ + int contimeout; /* Connection timeout */ + int delay, /* Delay for retries */ + prev_delay; /* Previous delay */ + const char *compression; /* Compression mode */ + int waitjob, /* Wait for job complete? */ + waitprinter; /* Wait for printer ready? */ + _cups_monitor_t monitor; /* Monitoring data */ + ipp_attribute_t *job_id_attr; /* job-id attribute */ + int job_id; /* job-id value */ + ipp_attribute_t *job_sheets; /* job-media-sheets-completed */ + ipp_attribute_t *job_state; /* job-state */ + ipp_attribute_t *copies_sup; /* copies-supported */ + ipp_attribute_t *cups_version; /* cups-version */ + ipp_attribute_t *format_sup; /* document-format-supported */ + ipp_attribute_t *media_col_sup; /* media-col-supported */ + ipp_attribute_t *operations_sup; /* operations-supported */ + ipp_attribute_t *doc_handling_sup; /* multiple-document-handling-supported */ + ipp_attribute_t *printer_state; /* printer-state attribute */ + ipp_attribute_t *printer_accepting; /* printer-is-accepting-jobs */ + int create_job = 0, /* Does printer support Create-Job? */ + send_document = 0, /* Does printer support Send-Document? */ + validate_job = 0; /* Does printer support Validate-Job? */ + int copies, /* Number of copies for job */ + copies_remaining; /* Number of copies remaining */ + const char *content_type, /* CONTENT_TYPE environment variable */ + *final_content_type, /* FINAL_CONTENT_TYPE environment var */ + *document_format; /* document-format value */ + int fd; /* File descriptor */ + off_t bytes = 0; /* Bytes copied */ + char buffer[16384]; /* Copy buffer */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* Actions for POSIX signals */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ + int version; /* IPP version */ + ppd_file_t *ppd; /* PPD file */ + _ppd_cache_t *pc; /* PPD cache and mapping data */ + fd_set input; /* Input set for select() */ + + + /* + * Make sure status messages are not buffered... + */ + + setbuf(stderr, NULL); + + /* + * Ignore SIGPIPE and catch SIGTERM signals... + */ + +#ifdef HAVE_SIGSET + sigset(SIGPIPE, SIG_IGN); + sigset(SIGTERM, sigterm_handler); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + action.sa_handler = SIG_IGN; + sigaction(SIGPIPE, &action, NULL); + + sigemptyset(&action.sa_mask); + sigaddset(&action.sa_mask, SIGTERM); + action.sa_handler = sigterm_handler; + sigaction(SIGTERM, &action, NULL); +#else + signal(SIGPIPE, SIG_IGN); + signal(SIGTERM, sigterm_handler); +#endif /* HAVE_SIGSET */ + + /* + * Check command-line... + */ + + if (argc == 1) + { + char *s; + + if ((s = strrchr(argv[0], '/')) != NULL) + s ++; + else + s = argv[0]; + + printf("network %s \"Unknown\" \"%s (%s)\"\n", + s, _cupsLangString(cupsLangDefault(), + _("Internet Printing Protocol")), s); + return (CUPS_BACKEND_OK); + } + else if (argc < 6) + { + _cupsLangPrintf(stderr, + _("Usage: %s job-id user title copies options [file]"), + argv[0]); + return (CUPS_BACKEND_STOP); + } + + /* + * Get the device URI... + */ + + while ((device_uri = cupsBackendDeviceURI(argv)) == NULL) + { + _cupsLangPrintFilter(stderr, "INFO", _("Unable to locate printer.")); + sleep(10); + + if (getenv("CLASS") != NULL) + return (CUPS_BACKEND_FAILED); + } + + if ((auth_info_required = getenv("AUTH_INFO_REQUIRED")) == NULL) + auth_info_required = "none"; + + state_reasons = _cupsArrayNewStrings(getenv("PRINTER_STATE_REASONS")); + +#ifdef HAVE_GSSAPI + /* + * For Kerberos, become the printing user (if we can) to get the credentials + * that way. + */ + + if (!getuid() && (value = getenv("AUTH_UID")) != NULL) + { + uid_t uid = (uid_t)atoi(value); + /* User ID */ + +# ifdef HAVE_XPC + if (uid > 0) + { + if (argc == 6) + return (run_as_user(argc, argv, uid, device_uri, 0)); + else + { + int status = 0; /* Exit status */ + + for (i = 6; i < argc && !status && !job_canceled; i ++) + { + if ((fd = open(argv[i], O_RDONLY)) >= 0) + { + status = run_as_user(argc, argv, uid, device_uri, fd); + close(fd); + } + else + { + _cupsLangPrintError("ERROR", _("Unable to open print file")); + status = CUPS_BACKEND_FAILED; + } + } + + return (status); + } + } + +# else /* No XPC, just try to run as the user ID */ + if (uid > 0) + seteuid(uid); +# endif /* HAVE_XPC */ + } +#endif /* HAVE_GSSAPI */ + + /* + * Get the (final) content type... + */ + + if ((content_type = getenv("CONTENT_TYPE")) == NULL) + content_type = "application/octet-stream"; + + if ((final_content_type = getenv("FINAL_CONTENT_TYPE")) == NULL) + { + final_content_type = content_type; + + if (!strncmp(final_content_type, "printer/", 8)) + final_content_type = "application/vnd.cups-raw"; + } + + /* + * Extract the hostname and printer name from the URI... + */ + + httpSeparateURI(HTTP_URI_CODING_ALL, device_uri, scheme, sizeof(scheme), + username, sizeof(username), hostname, sizeof(hostname), &port, + resource, sizeof(resource)); + + if (!port) + port = IPP_PORT; /* Default to port 631 */ + + if (!strcmp(scheme, "https")) + cupsSetEncryption(HTTP_ENCRYPT_ALWAYS); + else + cupsSetEncryption(HTTP_ENCRYPT_IF_REQUESTED); + + /* + * See if there are any options... + */ + + compression = NULL; + version = 20; + waitjob = 1; + waitprinter = 1; + contimeout = 7 * 24 * 60 * 60; + + if ((optptr = strchr(resource, '?')) != NULL) + { + /* + * Yup, terminate the device name string and move to the first + * character of the optptr... + */ + + *optptr++ = '\0'; + + /* + * Then parse the optptr... + */ + + while (*optptr) + { + /* + * Get the name... + */ + + name = optptr; + + while (*optptr && *optptr != '=' && *optptr != '+' && *optptr != '&') + optptr ++; + + if ((sep = *optptr) != '\0') + *optptr++ = '\0'; + + if (sep == '=') + { + /* + * Get the value... + */ + + value = optptr; + + while (*optptr && *optptr != '+' && *optptr != '&') + optptr ++; + + if (*optptr) + *optptr++ = '\0'; + } + else + value = (char *)""; + + /* + * Process the option... + */ + + if (!_cups_strcasecmp(name, "waitjob")) + { + /* + * Wait for job completion? + */ + + waitjob = !_cups_strcasecmp(value, "on") || + !_cups_strcasecmp(value, "yes") || + !_cups_strcasecmp(value, "true"); + } + else if (!_cups_strcasecmp(name, "waitprinter")) + { + /* + * Wait for printer idle? + */ + + waitprinter = !_cups_strcasecmp(value, "on") || + !_cups_strcasecmp(value, "yes") || + !_cups_strcasecmp(value, "true"); + } + else if (!_cups_strcasecmp(name, "encryption")) + { + /* + * Enable/disable encryption? + */ + + if (!_cups_strcasecmp(value, "always")) + cupsSetEncryption(HTTP_ENCRYPT_ALWAYS); + else if (!_cups_strcasecmp(value, "required")) + cupsSetEncryption(HTTP_ENCRYPT_REQUIRED); + else if (!_cups_strcasecmp(value, "never")) + cupsSetEncryption(HTTP_ENCRYPT_NEVER); + else if (!_cups_strcasecmp(value, "ifrequested")) + cupsSetEncryption(HTTP_ENCRYPT_IF_REQUESTED); + else + { + _cupsLangPrintFilter(stderr, "ERROR", + _("Unknown encryption option value: \"%s\"."), + value); + } + } + else if (!_cups_strcasecmp(name, "version")) + { + if (!strcmp(value, "1.0")) + version = 10; + else if (!strcmp(value, "1.1")) + version = 11; + else if (!strcmp(value, "2.0")) + version = 20; + else if (!strcmp(value, "2.1")) + version = 21; + else if (!strcmp(value, "2.2")) + version = 22; + else + { + _cupsLangPrintFilter(stderr, "ERROR", + _("Unknown version option value: \"%s\"."), + value); + } + } +#ifdef HAVE_LIBZ + else if (!_cups_strcasecmp(name, "compression")) + { + if (!_cups_strcasecmp(value, "true") || !_cups_strcasecmp(value, "yes") || + !_cups_strcasecmp(value, "on") || !_cups_strcasecmp(value, "gzip")) + compression = "gzip"; + } +#endif /* HAVE_LIBZ */ + else if (!_cups_strcasecmp(name, "contimeout")) + { + /* + * Set the connection timeout... + */ + + if (atoi(value) > 0) + contimeout = atoi(value); + } + else + { + /* + * Unknown option... + */ + + _cupsLangPrintFilter(stderr, "ERROR", + _("Unknown option \"%s\" with value \"%s\"."), + name, value); + } + } + } + + /* + * If we have 7 arguments, print the file named on the command-line. + * Otherwise, copy stdin to a temporary file and print the temporary + * file. + */ + + if (argc == 6) + { + num_files = 0; + files = NULL; + send_options = !_cups_strcasecmp(final_content_type, "application/pdf") || + !_cups_strcasecmp(final_content_type, "application/vnd.cups-pdf") || + !_cups_strncasecmp(final_content_type, "image/", 6); + + fputs("DEBUG: Sending stdin for job...\n", stderr); + } + else + { + /* + * Point to the files on the command-line... + */ + + num_files = argc - 6; + files = argv + 6; + send_options = 1; + +#ifdef HAVE_LIBZ + if (compression) + compress_files(num_files, files); +#endif /* HAVE_LIBZ */ + + fprintf(stderr, "DEBUG: %d files to send in job...\n", num_files); + } + + /* + * Set the authentication info, if any... + */ + + cupsSetPasswordCB(password_cb); + + if (username[0]) + { + /* + * Use authenticaion information in the device URI... + */ + + if ((password = strchr(username, ':')) != NULL) + *password++ = '\0'; + + cupsSetUser(username); + } + else + { + /* + * Try loading authentication information from the environment. + */ + + const char *ptr = getenv("AUTH_USERNAME"); + + if (ptr) + cupsSetUser(ptr); + + password = getenv("AUTH_PASSWORD"); + } + + /* + * Try finding the remote server... + */ + + start_time = time(NULL); + + sprintf(portname, "%d", port); + + update_reasons(NULL, "+connecting-to-device"); + fprintf(stderr, "DEBUG: Looking up \"%s\"...\n", hostname); + + while ((addrlist = httpAddrGetList(hostname, AF_UNSPEC, portname)) == NULL) + { + _cupsLangPrintFilter(stderr, "INFO", + _("Unable to locate printer \"%s\"."), hostname); + sleep(10); + + if (getenv("CLASS") != NULL) + { + update_reasons(NULL, "-connecting-to-device"); + return (CUPS_BACKEND_STOP); + } + } + + http = _httpCreate(hostname, port, addrlist, cupsEncryption(), AF_UNSPEC); + httpSetTimeout(http, 30.0, timeout_cb, NULL); + + /* + * See if the printer supports SNMP... + */ + + if ((snmp_fd = _cupsSNMPOpen(addrlist->addr.addr.sa_family)) >= 0) + { + have_supplies = !backendSNMPSupplies(snmp_fd, &(addrlist->addr), + &start_count, NULL); + } + else + have_supplies = start_count = 0; + + /* + * Wait for data from the filter... + */ + + if (num_files == 0) + { + if (!backendWaitLoop(snmp_fd, &(addrlist->addr), 0, backendNetworkSideCB)) + return (CUPS_BACKEND_OK); + else if ((bytes = read(0, buffer, sizeof(buffer))) <= 0) + return (CUPS_BACKEND_OK); + } + + /* + * Try connecting to the remote server... + */ + + delay = _cupsNextDelay(0, &prev_delay); + + do + { + fprintf(stderr, "DEBUG: Connecting to %s:%d\n", hostname, port); + _cupsLangPrintFilter(stderr, "INFO", _("Connecting to printer.")); + + if (httpReconnect(http)) + { + int error = errno; /* Connection error */ + + if (http->status == HTTP_PKI_ERROR) + update_reasons(NULL, "+cups-certificate-error"); + + if (job_canceled) + break; + + if (getenv("CLASS") != NULL) + { + /* + * If the CLASS environment variable is set, the job was submitted + * to a class and not to a specific queue. In this case, we want + * to abort immediately so that the job can be requeued on the next + * available printer in the class. + */ + + _cupsLangPrintFilter(stderr, "INFO", + _("Unable to contact printer, queuing on next " + "printer in class.")); + + /* + * Sleep 5 seconds to keep the job from requeuing too rapidly... + */ + + sleep(5); + + update_reasons(NULL, "-connecting-to-device"); + + return (CUPS_BACKEND_FAILED); + } + + fprintf(stderr, "DEBUG: Connection error: %s\n", strerror(errno)); + + if (errno == ECONNREFUSED || errno == EHOSTDOWN || + errno == EHOSTUNREACH) + { + if (contimeout && (time(NULL) - start_time) > contimeout) + { + _cupsLangPrintFilter(stderr, "ERROR", + _("The printer is not responding.")); + update_reasons(NULL, "-connecting-to-device"); + return (CUPS_BACKEND_FAILED); + } + + switch (error) + { + case EHOSTDOWN : + _cupsLangPrintFilter(stderr, "WARNING", + _("The printer may not exist or " + "is unavailable at this time.")); + break; + + case EHOSTUNREACH : + _cupsLangPrintFilter(stderr, "WARNING", + _("The printer is unreachable at this " + "time.")); + break; + + case ECONNREFUSED : + default : + _cupsLangPrintFilter(stderr, "WARNING", + _("The printer is busy.")); + break; + } + + sleep(delay); + + delay = _cupsNextDelay(delay, &prev_delay); + } + else + { + _cupsLangPrintFilter(stderr, "ERROR", + _("The printer is not responding.")); + sleep(30); + } + + if (job_canceled) + break; + } + else + update_reasons(NULL, "-cups-certificate-error"); + } + while (http->fd < 0); + + if (job_canceled || !http) + return (CUPS_BACKEND_FAILED); + + update_reasons(NULL, "-connecting-to-device"); + _cupsLangPrintFilter(stderr, "INFO", _("Connected to printer.")); + + fprintf(stderr, "DEBUG: Connected to %s:%d...\n", + httpAddrString(http->hostaddr, addrname, sizeof(addrname)), + _httpAddrPort(http->hostaddr)); + + /* + * Build a URI for the printer and fill the standard IPP attributes for + * an IPP_PRINT_FILE request. We can't use the URI in argv[0] because it + * might contain username:password information... + */ + + httpAssembleURI(HTTP_URI_CODING_ALL, uri, sizeof(uri), scheme, NULL, hostname, + port, resource); + + /* + * First validate the destination and see if the device supports multiple + * copies... + */ + + copies_sup = NULL; + cups_version = NULL; + format_sup = NULL; + media_col_sup = NULL; + supported = NULL; + operations_sup = NULL; + doc_handling_sup = NULL; + + do + { + /* + * Check for side-channel requests... + */ + + backendCheckSideChannel(snmp_fd, http->hostaddr); + + /* + * Build the IPP request... + */ + + request = ippNewRequest(IPP_GET_PRINTER_ATTRIBUTES); + request->request.op.version[0] = version / 10; + request->request.op.version[1] = version % 10; + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", sizeof(pattrs) / sizeof(pattrs[0]), + NULL, pattrs); + + /* + * Do the request... + */ + + fputs("DEBUG: Getting supported attributes...\n", stderr); + + if (http->version < HTTP_1_1) + { + fprintf(stderr, "DEBUG: Printer responded with HTTP version %d.%d.\n", + http->version / 100, http->version % 100); + update_reasons(NULL, "+cups-ipp-conformance-failure-report," + "cups-ipp-wrong-http-version"); + } + + supported = cupsDoRequest(http, request, resource); + ipp_status = cupsLastError(); + + fprintf(stderr, "DEBUG: Get-Printer-Attributes: %s (%s)\n", + ippErrorString(ipp_status), cupsLastErrorString()); + + if (ipp_status > IPP_OK_CONFLICT) + { + fprintf(stderr, "DEBUG: Get-Printer-Attributes returned %s.\n", + ippErrorString(ipp_status)); + + if (ipp_status == IPP_PRINTER_BUSY || + ipp_status == IPP_SERVICE_UNAVAILABLE) + { + if (contimeout && (time(NULL) - start_time) > contimeout) + { + _cupsLangPrintFilter(stderr, "ERROR", + _("The printer is not responding.")); + return (CUPS_BACKEND_FAILED); + } + + _cupsLangPrintFilter(stderr, "INFO", _("The printer is busy.")); + + report_printer_state(supported); + + sleep(delay); + + delay = _cupsNextDelay(delay, &prev_delay); + } + else if ((ipp_status == IPP_BAD_REQUEST || + ipp_status == IPP_VERSION_NOT_SUPPORTED) && version > 10) + { + /* + * Switch to IPP/1.1 or IPP/1.0... + */ + + if (version >= 20) + { + _cupsLangPrintFilter(stderr, "INFO", + _("Printer does not support IPP/%d.%d, trying " + "IPP/%s."), version / 10, version % 10, "1.1"); + version = 11; + } + else + { + _cupsLangPrintFilter(stderr, "INFO", + _("Printer does not support IPP/%d.%d, trying " + "IPP/%s."), version / 10, version % 10, "1.0"); + version = 10; + } + + httpReconnect(http); + } + else if (ipp_status == IPP_NOT_FOUND) + { + _cupsLangPrintFilter(stderr, "ERROR", + _("The printer URI is incorrect or no longer " + "exists.")); + + ippDelete(supported); + + return (CUPS_BACKEND_STOP); + } + else if (ipp_status == IPP_NOT_AUTHORIZED || ipp_status == IPP_FORBIDDEN) + { + if (!strncmp(httpGetField(http, HTTP_FIELD_WWW_AUTHENTICATE), + "Negotiate", 9)) + auth_info_required = "negotiate"; + + fprintf(stderr, "ATTR: auth-info-required=%s\n", auth_info_required); + return (CUPS_BACKEND_AUTH_REQUIRED); + } + else + { + _cupsLangPrintFilter(stderr, "ERROR", + _("Unable to get printer status.")); + sleep(10); + } + + ippDelete(supported); + supported = NULL; + continue; + } + + if (!getenv("CLASS")) + { + /* + * Check printer-is-accepting-jobs = false and printer-state-reasons for the + * "spool-area-full" keyword... + */ + + int busy = 0; + + if ((printer_accepting = ippFindAttribute(supported, + "printer-is-accepting-jobs", + IPP_TAG_BOOLEAN)) != NULL && + !printer_accepting->values[0].boolean) + busy = 1; + else if (!printer_accepting) + update_reasons(NULL, "+cups-ipp-conformance-failure-report," + "cups-ipp-missing-printer-is-accepting-jobs"); + + if ((printer_state = ippFindAttribute(supported, + "printer-state-reasons", + IPP_TAG_KEYWORD)) != NULL && !busy) + { + for (i = 0; i < printer_state->num_values; i ++) + if (!strcmp(printer_state->values[0].string.text, + "spool-area-full") || + !strncmp(printer_state->values[0].string.text, "spool-area-full-", + 16)) + { + busy = 1; + break; + } + } + else + update_reasons(NULL, "+cups-ipp-conformance-failure-report," + "cups-ipp-missing-printer-state-reasons"); + + if (busy) + { + _cupsLangPrintFilter(stderr, "INFO", _("The printer is busy.")); + + report_printer_state(supported); + + sleep(delay); + + delay = _cupsNextDelay(delay, &prev_delay); + + ippDelete(supported); + supported = NULL; + continue; + } + } + + /* + * Check for supported attributes... + */ + + if ((copies_sup = ippFindAttribute(supported, "copies-supported", + IPP_TAG_RANGE)) != NULL) + { + /* + * Has the "copies-supported" attribute - does it have an upper + * bound > 1? + */ + + fprintf(stderr, "DEBUG: copies-supported=%d-%d\n", + copies_sup->values[0].range.lower, + copies_sup->values[0].range.upper); + + if (copies_sup->values[0].range.upper <= 1) + copies_sup = NULL; /* No */ + } + + cups_version = ippFindAttribute(supported, "cups-version", IPP_TAG_TEXT); + + if ((format_sup = ippFindAttribute(supported, "document-format-supported", + IPP_TAG_MIMETYPE)) != NULL) + { + fprintf(stderr, "DEBUG: document-format-supported (%d values)\n", + format_sup->num_values); + for (i = 0; i < format_sup->num_values; i ++) + fprintf(stderr, "DEBUG: [%d] = \"%s\"\n", i, + format_sup->values[i].string.text); + } + + if ((media_col_sup = ippFindAttribute(supported, "media-col-supported", + IPP_TAG_KEYWORD)) != NULL) + { + fprintf(stderr, "DEBUG: media-col-supported (%d values)\n", + media_col_sup->num_values); + for (i = 0; i < media_col_sup->num_values; i ++) + fprintf(stderr, "DEBUG: [%d] = \"%s\"\n", i, + media_col_sup->values[i].string.text); + } + + if ((operations_sup = ippFindAttribute(supported, "operations-supported", + IPP_TAG_ENUM)) != NULL) + { + for (i = 0; i < operations_sup->num_values; i ++) + if (operations_sup->values[i].integer == IPP_PRINT_JOB) + break; + + if (i >= operations_sup->num_values) + update_reasons(NULL, "+cups-ipp-conformance-failure-report," + "cups-ipp-missing-print-job"); + + for (i = 0; i < operations_sup->num_values; i ++) + if (operations_sup->values[i].integer == IPP_CANCEL_JOB) + break; + + if (i >= operations_sup->num_values) + update_reasons(NULL, "+cups-ipp-conformance-failure-report," + "cups-ipp-missing-cancel-job"); + + for (i = 0; i < operations_sup->num_values; i ++) + if (operations_sup->values[i].integer == IPP_GET_JOB_ATTRIBUTES) + break; + + if (i >= operations_sup->num_values) + update_reasons(NULL, "+cups-ipp-conformance-failure-report," + "cups-ipp-missing-get-job-attributes"); + + for (i = 0; i < operations_sup->num_values; i ++) + if (operations_sup->values[i].integer == IPP_GET_PRINTER_ATTRIBUTES) + break; + + if (i >= operations_sup->num_values) + update_reasons(NULL, "+cups-ipp-conformance-failure-report," + "cups-ipp-missing-get-printer-attributes"); + + for (i = 0; i < operations_sup->num_values; i ++) + { + if (operations_sup->values[i].integer == IPP_VALIDATE_JOB) + validate_job = 1; + else if (operations_sup->values[i].integer == IPP_CREATE_JOB) + create_job = 1; + else if (operations_sup->values[i].integer == IPP_SEND_DOCUMENT) + send_document = 1; + } + + if (!send_document) + { + fputs("DEBUG: Printer supports Create-Job but not Send-Document.\n", + stderr); + create_job = 0; + } + + if (!validate_job) + update_reasons(NULL, "+cups-ipp-conformance-failure-report," + "cups-ipp-missing-validate-job"); + } + else + update_reasons(NULL, "+cups-ipp-conformance-failure-report," + "cups-ipp-missing-operations-supported"); + + doc_handling_sup = ippFindAttribute(supported, + "multiple-document-handling-supported", + IPP_TAG_KEYWORD); + + report_printer_state(supported); + } + while (ipp_status > IPP_OK_CONFLICT); + + /* + * See if the printer is accepting jobs and is not stopped; if either + * condition is true and we are printing to a class, requeue the job... + */ + + if (getenv("CLASS") != NULL) + { + printer_state = ippFindAttribute(supported, "printer-state", + IPP_TAG_ENUM); + printer_accepting = ippFindAttribute(supported, "printer-is-accepting-jobs", + IPP_TAG_BOOLEAN); + + if (printer_state == NULL || + (printer_state->values[0].integer > IPP_PRINTER_PROCESSING && + waitprinter) || + printer_accepting == NULL || + !printer_accepting->values[0].boolean) + { + /* + * If the CLASS environment variable is set, the job was submitted + * to a class and not to a specific queue. In this case, we want + * to abort immediately so that the job can be requeued on the next + * available printer in the class. + */ + + _cupsLangPrintFilter(stderr, "INFO", + _("Unable to contact printer, queuing on next " + "printer in class.")); + + ippDelete(supported); + httpClose(http); + + /* + * Sleep 5 seconds to keep the job from requeuing too rapidly... + */ + + sleep(5); + + return (CUPS_BACKEND_FAILED); + } + } + + /* + * See if the printer supports multiple copies... + */ + + copies = atoi(argv[4]); + + if (copies_sup || argc < 7) + { + copies_remaining = 1; + + if (argc < 7 && !_cups_strncasecmp(final_content_type, "image/", 6)) + copies = 1; + } + else + copies_remaining = copies; + + /* + * Prepare remaining printing options... + */ + + options = NULL; + pc = NULL; + + if (send_options) + { + num_options = cupsParseOptions(argv[5], 0, &options); + + if (!cups_version && media_col_sup) + { + /* + * Load the PPD file and generate PWG attribute mapping information... + */ + + ppd = ppdOpenFile(getenv("PPD")); + pc = _ppdCacheCreateWithPPD(ppd); + + ppdClose(ppd); + } + } + else + num_options = 0; + + document_format = NULL; + + if (format_sup != NULL) + { + for (i = 0; i < format_sup->num_values; i ++) + if (!_cups_strcasecmp(final_content_type, format_sup->values[i].string.text)) + { + document_format = final_content_type; + break; + } + + if (!document_format) + { + for (i = 0; i < format_sup->num_values; i ++) + if (!_cups_strcasecmp("application/octet-stream", + format_sup->values[i].string.text)) + { + document_format = "application/octet-stream"; + break; + } + } + } + + /* + * If the printer does not support HTTP/1.1 (which IPP requires), copy stdin + * to a temporary file so that we can do a HTTP/1.0 submission... + * + * (I hate compatibility hacks!) + */ + + if (http->version < HTTP_1_1 && num_files == 0) + { + if ((fd = cupsTempFd(tmpfilename, sizeof(tmpfilename))) < 0) + { + perror("DEBUG: Unable to create temporary file"); + return (CUPS_BACKEND_FAILED); + } + + _cupsLangPrintFilter(stderr, "INFO", _("Copying print data.")); + + compatsize = backendRunLoop(-1, fd, snmp_fd, &(addrlist->addr), 0, 0, + backendNetworkSideCB); + + close(fd); + + compatfile = tmpfilename; + files = &compatfile; + num_files = 1; + } + else if (http->version < HTTP_1_1 && num_files == 1) + { + struct stat fileinfo; /* File information */ + + if (!stat(files[0], &fileinfo)) + compatsize = fileinfo.st_size; + } + + /* + * Start monitoring the printer in the background... + */ + + monitor.uri = uri; + monitor.hostname = hostname; + monitor.user = argv[2]; + monitor.resource = resource; + monitor.port = port; + monitor.version = version; + monitor.job_id = 0; + monitor.encryption = cupsEncryption(); + monitor.job_state = IPP_JOB_PENDING; + monitor.printer_state = IPP_PRINTER_IDLE; + + if (create_job) + { + monitor.job_name = argv[3]; + } + else + { + snprintf(print_job_name, sizeof(print_job_name), "%s - %s", argv[1], + argv[3]); + monitor.job_name = print_job_name; + } + + _cupsThreadCreate((_cups_thread_func_t)monitor_printer, &monitor); + + /* + * Validate access to the printer... + */ + + while (!job_canceled && validate_job) + { + request = new_request(IPP_VALIDATE_JOB, version, uri, argv[2], + monitor.job_name, num_options, options, compression, + copies_sup ? copies : 1, document_format, pc, + media_col_sup, doc_handling_sup); + + ippDelete(cupsDoRequest(http, request, resource)); + + ipp_status = cupsLastError(); + + fprintf(stderr, "DEBUG: Validate-Job: %s (%s)\n", + ippErrorString(ipp_status), cupsLastErrorString()); + + if (job_canceled) + break; + + if (ipp_status == IPP_SERVICE_UNAVAILABLE || ipp_status == IPP_PRINTER_BUSY) + { + _cupsLangPrintFilter(stderr, "INFO", _("The printer is busy.")); + sleep(10); + } + else if (ipp_status == IPP_NOT_AUTHORIZED || ipp_status == IPP_FORBIDDEN || + ipp_status == IPP_AUTHENTICATION_CANCELED) + { + /* + * Update auth-info-required as needed... + */ + + fprintf(stderr, "DEBUG: WWW-Authenticate=\"%s\"\n", + httpGetField(http, HTTP_FIELD_WWW_AUTHENTICATE)); + + /* + * Normal authentication goes through the password callback, which sets + * auth_info_required to "username,password". Kerberos goes directly + * through GSSAPI, so look for Negotiate in the WWW-Authenticate header + * here and set auth_info_required as needed... + */ + + if (!strncmp(httpGetField(http, HTTP_FIELD_WWW_AUTHENTICATE), + "Negotiate", 9)) + auth_info_required = "negotiate"; + + goto cleanup; + } + else if (ipp_status == IPP_OPERATION_NOT_SUPPORTED) + { + /* + * This is all too common... + */ + + update_reasons(NULL, "+cups-ipp-conformance-failure-report," + "cups-ipp-missing-validate-job"); + break; + } + else if (ipp_status < IPP_REDIRECTION_OTHER_SITE) + break; + } + + /* + * Then issue the print-job request... + */ + + job_id = 0; + + while (!job_canceled && copies_remaining > 0) + { + /* + * Check for side-channel requests... + */ + + backendCheckSideChannel(snmp_fd, http->hostaddr); + + /* + * Build the IPP job creation request... + */ + + if (job_canceled) + break; + + request = new_request((num_files > 1 || create_job) ? IPP_CREATE_JOB : + IPP_PRINT_JOB, + version, uri, argv[2], monitor.job_name, num_options, + options, compression, copies_sup ? copies : 1, + document_format, pc, media_col_sup, doc_handling_sup); + + /* + * Do the request... + */ + + if (num_files > 1 || create_job) + response = cupsDoRequest(http, request, resource); + else + { + size_t length = 0; /* Length of request */ + + if (compatsize > 0) + { + fputs("DEBUG: Sending file using HTTP/1.0 Content-Length...\n", stderr); + length = ippLength(request) + (size_t)compatsize; + } + else + fputs("DEBUG: Sending file using HTTP/1.1 chunking...\n", stderr); + + http_status = cupsSendRequest(http, request, resource, length); + if (http_status == HTTP_CONTINUE && request->state == IPP_DATA) + { + if (num_files == 1) + { + if ((fd = open(files[0], O_RDONLY)) < 0) + { + _cupsLangPrintError("ERROR", _("Unable to open print file")); + return (CUPS_BACKEND_FAILED); + } + } + else + { + fd = 0; + http_status = cupsWriteRequestData(http, buffer, bytes); + } + + while (http_status == HTTP_CONTINUE && + (!job_canceled || compatsize > 0)) + { + /* + * Check for side-channel requests and more print data... + */ + + FD_ZERO(&input); + FD_SET(fd, &input); + FD_SET(snmp_fd, &input); + + while (select(fd > snmp_fd ? fd + 1 : snmp_fd + 1, &input, NULL, NULL, + NULL) <= 0 && !job_canceled); + + if (FD_ISSET(snmp_fd, &input)) + backendCheckSideChannel(snmp_fd, http->hostaddr); + + if (FD_ISSET(fd, &input)) + { + if ((bytes = read(fd, buffer, sizeof(buffer))) > 0) + { + fprintf(stderr, "DEBUG: Read %d bytes...\n", (int)bytes); + + if (cupsWriteRequestData(http, buffer, bytes) != HTTP_CONTINUE) + break; + } + else if (bytes == 0 || (errno != EINTR && errno != EAGAIN)) + break; + } + } + + if (num_files == 1) + close(fd); + } + + response = cupsGetResponse(http, resource); + ippDelete(request); + } + + ipp_status = cupsLastError(); + + fprintf(stderr, "DEBUG: %s: %s (%s)\n", + (num_files > 1 || create_job) ? "Create-Job" : "Print-Job", + ippErrorString(ipp_status), cupsLastErrorString()); + + if (ipp_status > IPP_OK_CONFLICT) + { + job_id = 0; + + if (job_canceled) + break; + + if (ipp_status == IPP_SERVICE_UNAVAILABLE || + ipp_status == IPP_NOT_POSSIBLE || + ipp_status == IPP_PRINTER_BUSY) + { + _cupsLangPrintFilter(stderr, "INFO", _("The printer is busy.")); + sleep(10); + + if (num_files == 0) + { + /* + * We can't re-submit when we have no files to print, so exit + * immediately with the right status code... + */ + + goto cleanup; + } + } + else if (ipp_status == IPP_ERROR_JOB_CANCELED) + goto cleanup; + else + { + /* + * Update auth-info-required as needed... + */ + + _cupsLangPrintFilter(stderr, "ERROR", + _("Print file was not accepted.")); + + if (ipp_status == IPP_NOT_AUTHORIZED || ipp_status == IPP_FORBIDDEN) + { + fprintf(stderr, "DEBUG: WWW-Authenticate=\"%s\"\n", + httpGetField(http, HTTP_FIELD_WWW_AUTHENTICATE)); + + /* + * Normal authentication goes through the password callback, which sets + * auth_info_required to "username,password". Kerberos goes directly + * through GSSAPI, so look for Negotiate in the WWW-Authenticate header + * here and set auth_info_required as needed... + */ + + if (!strncmp(httpGetField(http, HTTP_FIELD_WWW_AUTHENTICATE), + "Negotiate", 9)) + auth_info_required = "negotiate"; + } + else + sleep(10); + + if (num_files == 0) + { + /* + * We can't re-submit when we have no files to print, so exit + * immediately with the right status code... + */ + + goto cleanup; + } + } + } + else if ((job_id_attr = ippFindAttribute(response, "job-id", + IPP_TAG_INTEGER)) == NULL) + { + _cupsLangPrintFilter(stderr, "INFO", + _("Print file accepted - job ID unknown.")); + update_reasons(NULL, "+cups-ipp-conformance-failure-report," + "cups-ipp-missing-job-id"); + job_id = 0; + } + else + { + monitor.job_id = job_id = job_id_attr->values[0].integer; + _cupsLangPrintFilter(stderr, "INFO", + _("Print file accepted - job ID %d."), job_id); + } + + fprintf(stderr, "DEBUG: job-id=%d\n", job_id); + ippDelete(response); + + if (job_canceled) + break; + + if (job_id && (num_files > 1 || create_job)) + { + for (i = 0; num_files == 0 || i < num_files; i ++) + { + /* + * Check for side-channel requests... + */ + + backendCheckSideChannel(snmp_fd, http->hostaddr); + + /* + * Send the next file in the job... + */ + + request = ippNewRequest(IPP_SEND_DOCUMENT); + request->request.op.version[0] = version / 10; + request->request.op.version[1] = version % 10; + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_INTEGER, "job-id", + job_id); + + if (argv[2][0]) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, argv[2]); + + if ((i + 1) >= num_files) + ippAddBoolean(request, IPP_TAG_OPERATION, "last-document", 1); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_MIMETYPE, + "document-format", NULL, document_format); + + fprintf(stderr, "DEBUG: Sending file %d using chunking...\n", i + 1); + http_status = cupsSendRequest(http, request, resource, 0); + if (http_status == HTTP_CONTINUE && request->state == IPP_DATA) + { + if (num_files == 0) + { + fd = 0; + http_status = cupsWriteRequestData(http, buffer, bytes); + } + else + { + if ((fd = open(files[i], O_RDONLY)) < 0) + { + _cupsLangPrintError("ERROR", _("Unable to open print file")); + return (CUPS_BACKEND_FAILED); + } + } + } + else + fd = -1; + + if (fd >= 0) + { + while (!job_canceled && + (bytes = read(fd, buffer, sizeof(buffer))) > 0) + { + if (cupsWriteRequestData(http, buffer, bytes) != HTTP_CONTINUE) + break; + else + { + /* + * Check for side-channel requests... + */ + + backendCheckSideChannel(snmp_fd, http->hostaddr); + } + } + + if (fd > 0) + close(fd); + } + + ippDelete(cupsGetResponse(http, resource)); + ippDelete(request); + + fprintf(stderr, "DEBUG: Send-Document: %s (%s)\n", + ippErrorString(cupsLastError()), cupsLastErrorString()); + + if (cupsLastError() > IPP_OK_CONFLICT) + { + ipp_status = cupsLastError(); + + _cupsLangPrintFilter(stderr, "ERROR", + _("Unable to add document to print job.")); + break; + } + else if (num_files == 0 || fd < 0) + break; + } + } + + if (ipp_status <= IPP_OK_CONFLICT && argc > 6) + { + fprintf(stderr, "PAGE: 1 %d\n", copies_sup ? atoi(argv[4]) : 1); + copies_remaining --; + } + else if (ipp_status == IPP_SERVICE_UNAVAILABLE || + ipp_status == IPP_NOT_POSSIBLE || + ipp_status == IPP_PRINTER_BUSY) + continue; + else + copies_remaining --; + + /* + * Wait for the job to complete... + */ + + if (!job_id || !waitjob) + continue; + + _cupsLangPrintFilter(stderr, "INFO", _("Waiting for job to complete.")); + + for (delay = _cupsNextDelay(0, &prev_delay); !job_canceled;) + { + /* + * Check for side-channel requests... + */ + + backendCheckSideChannel(snmp_fd, http->hostaddr); + + /* + * Build an IPP_GET_JOB_ATTRIBUTES request... + */ + + request = ippNewRequest(IPP_GET_JOB_ATTRIBUTES); + request->request.op.version[0] = version / 10; + request->request.op.version[1] = version % 10; + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_INTEGER, "job-id", + job_id); + + if (argv[2][0]) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, argv[2]); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", sizeof(jattrs) / sizeof(jattrs[0]), + NULL, jattrs); + + /* + * Do the request... + */ + + httpReconnect(http); + response = cupsDoRequest(http, request, resource); + ipp_status = cupsLastError(); + + if (ipp_status == IPP_NOT_FOUND) + { + /* + * Job has gone away and/or the server has no job history... + */ + + update_reasons(NULL, "+cups-ipp-conformance-failure-report," + "cups-ipp-missing-job-history"); + ippDelete(response); + + ipp_status = IPP_OK; + break; + } + + fprintf(stderr, "DEBUG: Get-Job-Attributes: %s (%s)\n", + ippErrorString(ipp_status), cupsLastErrorString()); + + if (ipp_status > IPP_OK_CONFLICT) + { + if (ipp_status != IPP_SERVICE_UNAVAILABLE && + ipp_status != IPP_NOT_POSSIBLE && + ipp_status != IPP_PRINTER_BUSY) + { + ippDelete(response); + + _cupsLangPrintFilter(stderr, "ERROR", + _("Unable to get print job status.")); + ipp_status = IPP_OK; + break; + } + } + + if (response) + { + if ((job_state = ippFindAttribute(response, "job-state", + IPP_TAG_ENUM)) != NULL) + { + /* + * Reflect the remote job state in the local queue... + */ + + if (cups_version && + job_state->values[0].integer >= IPP_JOB_PENDING && + job_state->values[0].integer <= IPP_JOB_COMPLETED) + update_reasons(NULL, + remote_job_states[job_state->values[0].integer - + IPP_JOB_PENDING]); + + if ((job_sheets = ippFindAttribute(response, + "job-media-sheets-completed", + IPP_TAG_INTEGER)) == NULL) + job_sheets = ippFindAttribute(response, + "job-impressions-completed", + IPP_TAG_INTEGER); + + if (job_sheets) + fprintf(stderr, "PAGE: total %d\n", + job_sheets->values[0].integer); + + /* + * Stop polling if the job is finished or pending-held... + */ + + if (job_state->values[0].integer > IPP_JOB_STOPPED) + { + ippDelete(response); + break; + } + } + else if (ipp_status != IPP_SERVICE_UNAVAILABLE && + ipp_status != IPP_NOT_POSSIBLE && + ipp_status != IPP_PRINTER_BUSY) + { + /* + * If the printer does not return a job-state attribute, it does not + * conform to the IPP specification - break out immediately and fail + * the job... + */ + + update_reasons(NULL, "+cups-ipp-conformance-failure-report," + "cups-ipp-missing-job-state"); + ipp_status = IPP_INTERNAL_ERROR; + break; + } + } + + ippDelete(response); + + /* + * Wait before polling again... + */ + + sleep(delay); + + delay = _cupsNextDelay(delay, &prev_delay); + } + } + + /* + * Cancel the job as needed... + */ + + if (job_canceled && job_id) + cancel_job(http, uri, job_id, resource, argv[2], version); + + /* + * Check the printer state and report it if necessary... + */ + + check_printer_state(http, uri, resource, argv[2], version); + + /* + * Collect the final page count as needed... + */ + + if (have_supplies && + !backendSNMPSupplies(snmp_fd, http->hostaddr, &page_count, NULL) && + page_count > start_count) + fprintf(stderr, "PAGE: total %d\n", page_count - start_count); + +#ifdef HAVE_GSSAPI + /* + * See if we used Kerberos at all... + */ + + if (http->gssctx) + auth_info_required = "negotiate"; +#endif /* HAVE_GSSAPI */ + + /* + * Free memory... + */ + + cleanup: + + cupsFreeOptions(num_options, options); + _ppdCacheDestroy(pc); + + httpClose(http); + + ippDelete(supported); + + /* + * Remove the temporary file(s) if necessary... + */ + + if (tmpfilename[0]) + unlink(tmpfilename); + +#ifdef HAVE_LIBZ + if (compression) + { + for (i = 0; i < num_files; i ++) + unlink(files[i]); + } +#endif /* HAVE_LIBZ */ + + /* + * Return the queue status... + */ + + if (ipp_status == IPP_NOT_AUTHORIZED || ipp_status == IPP_FORBIDDEN || + ipp_status == IPP_AUTHENTICATION_CANCELED || + ipp_status <= IPP_OK_CONFLICT) + fprintf(stderr, "ATTR: auth-info-required=%s\n", auth_info_required); + + if (ipp_status == IPP_NOT_AUTHORIZED || ipp_status == IPP_FORBIDDEN || + ipp_status == IPP_AUTHENTICATION_CANCELED) + return (CUPS_BACKEND_AUTH_REQUIRED); + else if (ipp_status == IPP_INTERNAL_ERROR) + return (CUPS_BACKEND_STOP); + else if (ipp_status == IPP_DOCUMENT_FORMAT || + ipp_status == IPP_CONFLICT) + return (CUPS_BACKEND_FAILED); + else if (ipp_status > IPP_OK_CONFLICT && ipp_status != IPP_ERROR_JOB_CANCELED) + return (CUPS_BACKEND_RETRY_CURRENT); + else + { + _cupsLangPrintFilter(stderr, "INFO", _("Ready to print.")); + return (CUPS_BACKEND_OK); + } +} + + +/* + * 'cancel_job()' - Cancel a print job. + */ + +static void +cancel_job(http_t *http, /* I - HTTP connection */ + const char *uri, /* I - printer-uri */ + int id, /* I - job-id */ + const char *resource, /* I - Resource path */ + const char *user, /* I - requesting-user-name */ + int version) /* I - IPP version */ +{ + ipp_t *request; /* Cancel-Job request */ + + + _cupsLangPrintFilter(stderr, "INFO", _("Canceling print job.")); + + request = ippNewRequest(IPP_CANCEL_JOB); + request->request.op.version[0] = version / 10; + request->request.op.version[1] = version % 10; + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_INTEGER, "job-id", id); + + if (user && user[0]) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, user); + + /* + * Do the request... + */ + + ippDelete(cupsDoRequest(http, request, resource)); + + if (cupsLastError() > IPP_OK_CONFLICT) + _cupsLangPrintFilter(stderr, "ERROR", _("Unable to cancel print job.")); +} + + +/* + * 'check_printer_state()' - Check the printer state. + */ + +static ipp_pstate_t /* O - Current printer-state */ +check_printer_state( + http_t *http, /* I - HTTP connection */ + const char *uri, /* I - Printer URI */ + const char *resource, /* I - Resource path */ + const char *user, /* I - Username, if any */ + int version) /* I - IPP version */ + { + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + ipp_attribute_t *attr; /* Attribute in response */ + ipp_pstate_t printer_state = IPP_PRINTER_STOPPED; + /* Current printer-state */ + + + /* + * Send a Get-Printer-Attributes request and log the results... + */ + + request = ippNewRequest(IPP_GET_PRINTER_ATTRIBUTES); + request->request.op.version[0] = version / 10; + request->request.op.version[1] = version % 10; + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + if (user && user[0]) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, user); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", + (int)(sizeof(pattrs) / sizeof(pattrs[0])), NULL, pattrs); + + if ((response = cupsDoRequest(http, request, resource)) != NULL) + { + report_printer_state(response); + + if ((attr = ippFindAttribute(response, "printer-state", + IPP_TAG_ENUM)) != NULL) + printer_state = (ipp_pstate_t)attr->values[0].integer; + + ippDelete(response); + } + + fprintf(stderr, "DEBUG: Get-Printer-Attributes: %s (%s)\n", + ippErrorString(cupsLastError()), cupsLastErrorString()); + + /* + * Return the printer-state value... + */ + + return (printer_state); +} + + +#ifdef HAVE_LIBZ +/* + * 'compress_files()' - Compress print files. + */ + +static void +compress_files(int num_files, /* I - Number of files */ + char **files) /* I - Files */ +{ + int i, /* Looping var */ + fd; /* Temporary file descriptor */ + ssize_t bytes; /* Bytes read/written */ + size_t total; /* Total bytes read */ + cups_file_t *in, /* Input file */ + *out; /* Output file */ + struct stat outinfo; /* Output file information */ + char filename[1024], /* Temporary filename */ + buffer[32768]; /* Copy buffer */ + + + fprintf(stderr, "DEBUG: Compressing %d job files...\n", num_files); + for (i = 0; i < num_files; i ++) + { + if ((fd = cupsTempFd(filename, sizeof(filename))) < 0) + { + _cupsLangPrintError("ERROR", _("Unable to create compressed print file")); + exit(CUPS_BACKEND_FAILED); + } + + if ((out = cupsFileOpenFd(fd, "w9")) == NULL) + { + _cupsLangPrintError("ERROR", _("Unable to open compressed print file")); + exit(CUPS_BACKEND_FAILED); + } + + if ((in = cupsFileOpen(files[i], "r")) == NULL) + { + _cupsLangPrintError("ERROR", _("Unable to open print file")); + cupsFileClose(out); + exit(CUPS_BACKEND_FAILED); + } + + total = 0; + while ((bytes = cupsFileRead(in, buffer, sizeof(buffer))) > 0) + if (cupsFileWrite(out, buffer, bytes) < bytes) + { + _cupsLangPrintError("ERROR", + _("Unable to generate compressed print file")); + cupsFileClose(in); + cupsFileClose(out); + exit(CUPS_BACKEND_FAILED); + } + else + total += bytes; + + cupsFileClose(out); + cupsFileClose(in); + + files[i] = strdup(filename); + + if (!stat(filename, &outinfo)) + fprintf(stderr, + "DEBUG: File %d compressed to %.1f%% of original size, " + CUPS_LLFMT " bytes...\n", + i + 1, 100.0 * outinfo.st_size / total, + CUPS_LLCAST outinfo.st_size); + } +} +#endif /* HAVE_LIBZ */ + + +/* + * 'monitor_printer()' - Monitor the printer state. + */ + +static void * /* O - Thread exit code */ +monitor_printer( + _cups_monitor_t *monitor) /* I - Monitoring data */ +{ + http_t *http; /* Connection to printer */ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + ipp_attribute_t *attr; /* Attribute in response */ + int delay, /* Current delay */ + prev_delay; /* Previous delay */ + ipp_op_t job_op; /* Operation to use */ + int job_id; /* Job ID */ + const char *job_name; /* Job name */ + ipp_jstate_t job_state; /* Job state */ + const char *job_user; /* Job originating user name */ + + + /* + * Make a copy of the printer connection... + */ + + http = _httpCreate(monitor->hostname, monitor->port, NULL, monitor->encryption, + AF_UNSPEC); + httpSetTimeout(http, 30.0, timeout_cb, NULL); + cupsSetPasswordCB(password_cb); + + /* + * Loop until the job is canceled, aborted, or completed. + */ + + delay = _cupsNextDelay(0, &prev_delay); + + while (monitor->job_state < IPP_JOB_CANCELED && !job_canceled) + { + /* + * Reconnect to the printer... + */ + + if (!httpReconnect(http)) + { + /* + * Connected, so check on the printer state... + */ + + monitor->printer_state = check_printer_state(http, monitor->uri, + monitor->resource, + monitor->user, + monitor->version); + + /* + * Check the status of the job itself... + */ + + job_op = monitor->job_id > 0 ? IPP_GET_JOB_ATTRIBUTES : IPP_GET_JOBS; + request = ippNewRequest(job_op); + request->request.op.version[0] = monitor->version / 10; + request->request.op.version[1] = monitor->version % 10; + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, monitor->uri); + if (job_op == IPP_GET_JOB_ATTRIBUTES) + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_INTEGER, "job-id", + monitor->job_id); + + if (monitor->user && monitor->user[0]) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, monitor->user); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", + (int)(sizeof(jattrs) / sizeof(jattrs[0])), NULL, jattrs); + + /* + * Do the request... + */ + + response = cupsDoRequest(http, request, monitor->resource); + + fprintf(stderr, "DEBUG: %s: %s (%s)\n", ippOpString(job_op), + ippErrorString(cupsLastError()), cupsLastErrorString()); + + if (job_op == IPP_GET_JOB_ATTRIBUTES) + { + if ((attr = ippFindAttribute(response, "job-state", + IPP_TAG_ENUM)) != NULL) + monitor->job_state = (ipp_jstate_t)attr->values[0].integer; + else + monitor->job_state = IPP_JOB_COMPLETED; + } + else if (response) + { + for (attr = response->attrs; attr; attr = attr->next) + { + job_id = 0; + job_name = NULL; + job_state = IPP_JOB_PENDING; + job_user = NULL; + + while (attr && attr->group_tag != IPP_TAG_JOB) + attr = attr->next; + + if (!attr) + break; + + while (attr && attr->group_tag == IPP_TAG_JOB) + { + if (!strcmp(attr->name, "job-id") && + attr->value_tag == IPP_TAG_INTEGER) + job_id = attr->values[0].integer; + else if (!strcmp(attr->name, "job-name") && + (attr->value_tag == IPP_TAG_NAME || + attr->value_tag == IPP_TAG_NAMELANG)) + job_name = attr->values[0].string.text; + else if (!strcmp(attr->name, "job-state") && + attr->value_tag == IPP_TAG_ENUM) + job_state = attr->values[0].integer; + else if (!strcmp(attr->name, "job-originating-user-name") && + (attr->value_tag == IPP_TAG_NAME || + attr->value_tag == IPP_TAG_NAMELANG)) + job_user = attr->values[0].string.text; + + attr = attr->next; + } + + if (job_id > 0 && job_name && !strcmp(job_name, monitor->job_name) && + job_user && monitor->user && !strcmp(job_user, monitor->user)) + { + monitor->job_id = job_id; + monitor->job_state = job_state; + break; + } + + if (!attr) + break; + } + } + + ippDelete(response); + + /* + * Disconnect from the printer - we'll reconnect on the next poll... + */ + + _httpDisconnect(http); + } + + /* + * Sleep for N seconds... + */ + + sleep(delay); + + delay = _cupsNextDelay(delay, &prev_delay); + } + + /* + * Cancel the job if necessary... + */ + + if (job_canceled && monitor->job_id > 0) + if (!httpReconnect(http)) + cancel_job(http, monitor->uri, monitor->job_id, monitor->resource, + monitor->user, monitor->version); + + /* + * Cleanup and return... + */ + + httpClose(http); + + return (NULL); +} + + +/* + * 'new_request()' - Create a new print creation or validation request. + */ + +static ipp_t * /* O - Request data */ +new_request( + ipp_op_t op, /* I - IPP operation code */ + int version, /* I - IPP version number */ + const char *uri, /* I - printer-uri value */ + const char *user, /* I - requesting-user-name value */ + const char *title, /* I - job-name value */ + int num_options, /* I - Number of options to send */ + cups_option_t *options, /* I - Options to send */ + const char *compression, /* I - compression value or NULL */ + int copies, /* I - copies value or 0 */ + const char *format, /* I - document-format value or NULL */ + _ppd_cache_t *pc, /* I - PPD cache and mapping data */ + ipp_attribute_t *media_col_sup, /* I - media-col-supported values */ + ipp_attribute_t *doc_handling_sup) /* I - multiple-document-handling-supported values */ +{ + int i; /* Looping var */ + ipp_t *request; /* Request data */ + const char *keyword; /* PWG keyword */ + _pwg_size_t *size; /* PWG media size */ + ipp_t *media_col, /* media-col value */ + *media_size; /* media-size value */ + const char *media_source, /* media-source value */ + *media_type, /* media-type value */ + *collate_str; /* multiple-document-handling value */ + + + /* + * Create the IPP request... + */ + + request = ippNewRequest(op); + request->request.op.version[0] = version / 10; + request->request.op.version[1] = version % 10; + + fprintf(stderr, "DEBUG: %s IPP/%d.%d\n", + ippOpString(request->request.op.operation_id), + request->request.op.version[0], + request->request.op.version[1]); + + /* + * Add standard attributes... + */ + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + fprintf(stderr, "DEBUG: printer-uri=\"%s\"\n", uri); + + if (user && *user) + { + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, user); + fprintf(stderr, "DEBUG: requesting-user-name=\"%s\"\n", user); + } + + if (title && *title) + { + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "job-name", NULL, + title); + fprintf(stderr, "DEBUG: job-name=\"%s\"\n", title); + } + + if (format) + { + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_MIMETYPE, + "document-format", NULL, format); + fprintf(stderr, "DEBUG: document-format=\"%s\"\n", format); + } + +#ifdef HAVE_LIBZ + if (compression) + { + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "compression", NULL, compression); + fprintf(stderr, "DEBUG: compression=\"%s\"\n", compression); + } +#endif /* HAVE_LIBZ */ + + /* + * Handle options on the command-line... + */ + + if (num_options > 0) + { + if (pc) + { + int num_finishings = 0, /* Number of finishing values */ + finishings[10]; /* Finishing enum values */ + + /* + * Send standard IPP attributes... + */ + + if ((keyword = cupsGetOption("PageSize", num_options, options)) == NULL) + keyword = cupsGetOption("media", num_options, options); + + if ((size = _ppdCacheGetSize(pc, keyword)) != NULL) + { + /* + * Add a media-col value... + */ + + media_size = ippNew(); + ippAddInteger(media_size, IPP_TAG_ZERO, IPP_TAG_INTEGER, + "x-dimension", size->width); + ippAddInteger(media_size, IPP_TAG_ZERO, IPP_TAG_INTEGER, + "y-dimension", size->length); + + media_col = ippNew(); + ippAddCollection(media_col, IPP_TAG_ZERO, "media-size", media_size); + + media_source = _ppdCacheGetSource(pc, cupsGetOption("InputSlot", + num_options, + options)); + media_type = _ppdCacheGetType(pc, cupsGetOption("MediaType", + num_options, + options)); + + for (i = 0; i < media_col_sup->num_values; i ++) + { + if (!strcmp(media_col_sup->values[i].string.text, + "media-left-margin")) + ippAddInteger(media_col, IPP_TAG_ZERO, IPP_TAG_INTEGER, + "media-left-margin", size->left); + else if (!strcmp(media_col_sup->values[i].string.text, + "media-bottom-margin")) + ippAddInteger(media_col, IPP_TAG_ZERO, IPP_TAG_INTEGER, + "media-bottom-margin", size->bottom); + else if (!strcmp(media_col_sup->values[i].string.text, + "media-right-margin")) + ippAddInteger(media_col, IPP_TAG_ZERO, IPP_TAG_INTEGER, + "media-right-margin", size->right); + else if (!strcmp(media_col_sup->values[i].string.text, + "media-top-margin")) + ippAddInteger(media_col, IPP_TAG_ZERO, IPP_TAG_INTEGER, + "media-top-margin", size->top); + else if (!strcmp(media_col_sup->values[i].string.text, + "media-source") && media_source) + ippAddString(media_col, IPP_TAG_ZERO, IPP_TAG_KEYWORD, + "media-source", NULL, media_source); + else if (!strcmp(media_col_sup->values[i].string.text, + "media-type") && media_type) + ippAddString(media_col, IPP_TAG_ZERO, IPP_TAG_KEYWORD, + "media-type", NULL, media_type); + } + + ippAddCollection(request, IPP_TAG_JOB, "media-col", media_col); + } + + if ((keyword = cupsGetOption("output-bin", num_options, + options)) == NULL) + keyword = _ppdCacheGetBin(pc, cupsGetOption("OutputBin", num_options, + options)); + + if (keyword) + ippAddString(request, IPP_TAG_JOB, IPP_TAG_KEYWORD, "output-bin", + NULL, keyword); + + if ((keyword = cupsGetOption("output-mode", num_options, + options)) != NULL) + ippAddString(request, IPP_TAG_JOB, IPP_TAG_KEYWORD, "output-mode", + NULL, keyword); + else if ((keyword = cupsGetOption("ColorModel", num_options, + options)) != NULL) + { + if (!_cups_strcasecmp(keyword, "Gray")) + ippAddString(request, IPP_TAG_JOB, IPP_TAG_KEYWORD, "output-mode", + NULL, "monochrome"); + else + ippAddString(request, IPP_TAG_JOB, IPP_TAG_KEYWORD, "output-mode", + NULL, "color"); + } + + if ((keyword = cupsGetOption("print-quality", num_options, + options)) != NULL) + ippAddInteger(request, IPP_TAG_JOB, IPP_TAG_ENUM, "print-quality", + atoi(keyword)); + else if ((keyword = cupsGetOption("cupsPrintQuality", num_options, + options)) != NULL) + { + if (!_cups_strcasecmp(keyword, "draft")) + ippAddInteger(request, IPP_TAG_JOB, IPP_TAG_ENUM, "print-quality", + IPP_QUALITY_DRAFT); + else if (!_cups_strcasecmp(keyword, "normal")) + ippAddInteger(request, IPP_TAG_JOB, IPP_TAG_ENUM, "print-quality", + IPP_QUALITY_NORMAL); + else if (!_cups_strcasecmp(keyword, "high")) + ippAddInteger(request, IPP_TAG_JOB, IPP_TAG_ENUM, "print-quality", + IPP_QUALITY_HIGH); + } + + if ((keyword = cupsGetOption("sides", num_options, options)) != NULL) + ippAddString(request, IPP_TAG_JOB, IPP_TAG_KEYWORD, "sides", + NULL, keyword); + else if (pc->sides_option && + (keyword = cupsGetOption(pc->sides_option, num_options, + options)) != NULL) + { + if (!_cups_strcasecmp(keyword, pc->sides_1sided)) + ippAddString(request, IPP_TAG_JOB, IPP_TAG_KEYWORD, "sides", + NULL, "one-sided"); + else if (!_cups_strcasecmp(keyword, pc->sides_2sided_long)) + ippAddString(request, IPP_TAG_JOB, IPP_TAG_KEYWORD, "sides", + NULL, "two-sided-long-edge"); + if (!_cups_strcasecmp(keyword, pc->sides_2sided_short)) + ippAddString(request, IPP_TAG_JOB, IPP_TAG_KEYWORD, "sides", + NULL, "two-sided-short-edge"); + } + + if (doc_handling_sup && + (!format || _cups_strncasecmp(format, "image/", 6)) && + (keyword = cupsGetOption("collate", num_options, options)) != NULL) + { + if (!_cups_strcasecmp(keyword, "true")) + collate_str = "separate-documents-collated-copies"; + else + collate_str = "separate-documents-uncollated-copies"; + + for (i = 0; i < doc_handling_sup->num_values; i ++) + if (!strcmp(doc_handling_sup->values[i].string.text, collate_str)) + { + ippAddString(request, IPP_TAG_JOB, IPP_TAG_KEYWORD, + "multiple-document-handling", NULL, collate_str); + break; + } + } + + /* + * Map finishing options... + */ + + num_finishings = _ppdCacheGetFinishingValues(pc, num_options, options, + (int)(sizeof(finishings) / + sizeof(finishings[0])), + finishings); + if (num_finishings > 0) + ippAddIntegers(request, IPP_TAG_JOB, IPP_TAG_ENUM, "finishings", + num_finishings, finishings); + + /* + * Map FaxOut options... + */ + + if ((keyword = cupsGetOption("phone", num_options, options)) != NULL) + { + ipp_t *destination; /* destination collection */ + char tel_uri[1024]; /* tel: URI */ + + destination = ippNew(); + + httpAssembleURI(HTTP_URI_CODING_ALL, tel_uri, sizeof(tel_uri), "tel", + NULL, NULL, 0, keyword); + ippAddString(destination, IPP_TAG_JOB, IPP_TAG_URI, "destination-uri", + NULL, tel_uri); + + if ((keyword = cupsGetOption("faxPrefix", num_options, + options)) != NULL && *keyword) + ippAddString(destination, IPP_TAG_JOB, IPP_TAG_TEXT, + "pre-dial-string", NULL, keyword); + + ippAddCollection(request, IPP_TAG_JOB, "destination-uris", destination); + ippDelete(destination); + } + } + else + { + /* + * When talking to another CUPS server, send all options... + */ + + cupsEncodeOptions(request, num_options, options); + } + + if (copies > 1) + ippAddInteger(request, IPP_TAG_JOB, IPP_TAG_INTEGER, "copies", copies); + } + + return (request); +} + + +/* + * 'password_cb()' - Disable the password prompt for cupsDoFileRequest(). + */ + +static const char * /* O - Password */ +password_cb(const char *prompt) /* I - Prompt (not used) */ +{ + (void)prompt; + + /* + * Remember that we need to authenticate... + */ + + auth_info_required = "username,password"; + + if (password && *password && password_tries < 3) + { + password_tries ++; + + return (password); + } + else + { + /* + * Give up after 3 tries or if we don't have a password to begin with... + */ + + return (NULL); + } +} + + +/* + * 'report_attr()' - Report an IPP attribute value. + */ + +static void +report_attr(ipp_attribute_t *attr) /* I - Attribute */ +{ + int i; /* Looping var */ + char value[1024], /* Value string */ + *valptr, /* Pointer into value string */ + *attrptr; /* Pointer into attribute value */ + const char *cached; /* Cached attribute */ + + + /* + * Convert the attribute values into quoted strings... + */ + + for (i = 0, valptr = value; + i < attr->num_values && valptr < (value + sizeof(value) - 10); + i ++) + { + if (i > 0) + *valptr++ = ','; + + switch (attr->value_tag) + { + case IPP_TAG_INTEGER : + case IPP_TAG_ENUM : + snprintf(valptr, sizeof(value) - (valptr - value), "%d", + attr->values[i].integer); + valptr += strlen(valptr); + break; + + case IPP_TAG_TEXT : + case IPP_TAG_NAME : + case IPP_TAG_KEYWORD : + *valptr++ = '\"'; + for (attrptr = attr->values[i].string.text; + *attrptr && valptr < (value + sizeof(value) - 10); + attrptr ++) + { + if (*attrptr == '\\' || *attrptr == '\"') + *valptr++ = '\\'; + + *valptr++ = *attrptr; + } + *valptr++ = '\"'; + break; + + default : + /* + * Unsupported value type... + */ + + return; + } + } + + *valptr = '\0'; + + _cupsMutexLock(&report_mutex); + + if ((cached = cupsGetOption(attr->name, num_attr_cache, + attr_cache)) == NULL || strcmp(cached, value)) + { + /* + * Tell the scheduler about the new values... + */ + + num_attr_cache = cupsAddOption(attr->name, value, num_attr_cache, + &attr_cache); + fprintf(stderr, "ATTR: %s=%s\n", attr->name, value); + } + + _cupsMutexUnlock(&report_mutex); +} + + +/* + * 'report_printer_state()' - Report the printer state. + */ + +static void +report_printer_state(ipp_t *ipp) /* I - IPP response */ +{ + ipp_attribute_t *pa, /* printer-alert */ + *pam, /* printer-alert-message */ + *psm, /* printer-state-message */ + *reasons, /* printer-state-reasons */ + *marker; /* marker-* attributes */ + char value[1024], /* State/message string */ + *valptr; /* Pointer into string */ + static int ipp_supplies = -1; + /* Report supply levels? */ + + + /* + * Report alerts and messages... + */ + + if ((pa = ippFindAttribute(ipp, "printer-alert", IPP_TAG_TEXT)) != NULL) + report_attr(pa); + + if ((pam = ippFindAttribute(ipp, "printer-alert-message", + IPP_TAG_TEXT)) != NULL) + report_attr(pam); + + if ((psm = ippFindAttribute(ipp, "printer-state-message", + IPP_TAG_TEXT)) != NULL) + { + char *ptr; /* Pointer into message */ + + + strlcpy(value, "INFO: ", sizeof(value)); + for (ptr = psm->values[0].string.text, valptr = value + 6; + *ptr && valptr < (value + sizeof(value) - 6); + ptr ++) + { + if (*ptr < ' ' && *ptr > 0 && *ptr != '\t') + { + /* + * Substitute "" for the control character; sprintf is safe because + * we always leave 6 chars free at the end... + */ + + sprintf(valptr, "<%02X>", *ptr); + valptr += 4; + } + else + *valptr++ = *ptr; + } + + *valptr++ = '\n'; + *valptr = '\0'; + + fputs(value, stderr); + } + + /* + * Now report printer-state-reasons, filtering out some of the reasons we never + * want to set... + */ + + if ((reasons = ippFindAttribute(ipp, "printer-state-reasons", + IPP_TAG_KEYWORD)) == NULL) + return; + + update_reasons(reasons, NULL); + + /* + * Relay the current marker-* attribute values... + */ + + if (ipp_supplies < 0) + { + ppd_file_t *ppd; /* PPD file */ + ppd_attr_t *ppdattr; /* Attribute in PPD file */ + + if ((ppd = ppdOpenFile(getenv("PPD"))) != NULL && + (ppdattr = ppdFindAttr(ppd, "cupsIPPSupplies", NULL)) != NULL && + ppdattr->value && _cups_strcasecmp(ppdattr->value, "true")) + ipp_supplies = 0; + else + ipp_supplies = 1; + + ppdClose(ppd); + } + + if (ipp_supplies > 0) + { + if ((marker = ippFindAttribute(ipp, "marker-colors", IPP_TAG_NAME)) != NULL) + report_attr(marker); + if ((marker = ippFindAttribute(ipp, "marker-high-levels", + IPP_TAG_INTEGER)) != NULL) + report_attr(marker); + if ((marker = ippFindAttribute(ipp, "marker-levels", + IPP_TAG_INTEGER)) != NULL) + report_attr(marker); + if ((marker = ippFindAttribute(ipp, "marker-low-levels", + IPP_TAG_INTEGER)) != NULL) + report_attr(marker); + if ((marker = ippFindAttribute(ipp, "marker-message", + IPP_TAG_TEXT)) != NULL) + report_attr(marker); + if ((marker = ippFindAttribute(ipp, "marker-names", IPP_TAG_NAME)) != NULL) + report_attr(marker); + if ((marker = ippFindAttribute(ipp, "marker-types", + IPP_TAG_KEYWORD)) != NULL) + report_attr(marker); + } +} + + +#if defined(HAVE_GSSAPI) && defined(HAVE_XPC) +/* + * 'run_as_user()' - Run the IPP backend as the printing user. + * + * This function uses an XPC-based user agent to run the backend as the printing + * user. We need to do this in order to have access to the user's Kerberos + * credentials. + */ + +static int /* O - Exit status */ +run_as_user(int argc, /* I - Number of command-line args */ + char *argv[], /* I - Command-line arguments */ + uid_t uid, /* I - User ID */ + const char *device_uri, /* I - Device URI */ + int fd) /* I - File to print */ +{ + const char *auth_negotiate;/* AUTH_NEGOTIATE env var */ + xpc_connection_t conn; /* Connection to XPC service */ + xpc_object_t request; /* Request message dictionary */ + __block xpc_object_t response; /* Response message dictionary */ + dispatch_semaphore_t sem; /* Semaphore for waiting for response */ + int status = CUPS_BACKEND_FAILED; + /* Status of request */ + + + fprintf(stderr, "DEBUG: Running IPP backend as UID %d.\n", (int)uid); + + /* + * Connect to the user agent for the specified UID... + */ + + conn = xpc_connection_create_mach_service(kPMPrintUIToolAgent, + dispatch_get_global_queue(0, 0), 0); + if (!conn) + { + _cupsLangPrintFilter(stderr, "ERROR", + _("Unable to start backend process.")); + fputs("DEBUG: Unable to create connection to agent.\n", stderr); + goto cleanup; + } + + xpc_connection_set_event_handler(conn, + ^(xpc_object_t event) + { + xpc_type_t messageType = xpc_get_type(event); + + if (messageType == XPC_TYPE_ERROR) + { + if (event == XPC_ERROR_CONNECTION_INTERRUPTED) + fprintf(stderr, "DEBUG: Interrupted connection to service %s.\n", + xpc_connection_get_name(conn)); + else if (event == XPC_ERROR_CONNECTION_INVALID) + fprintf(stderr, "DEBUG: Connection invalid for service %s.\n", + xpc_connection_get_name(conn)); + else + fprintf(stderr, "DEBUG: Unxpected error for service %s: %s\n", + xpc_connection_get_name(conn), + xpc_dictionary_get_string(event, XPC_ERROR_KEY_DESCRIPTION)); + } + }); + xpc_connection_set_target_uid(conn, uid); + xpc_connection_resume(conn); + + /* + * Try starting the backend... + */ + + request = xpc_dictionary_create(NULL, NULL, 0); + xpc_dictionary_set_int64(request, "command", kPMStartJob); + xpc_dictionary_set_string(request, "device-uri", device_uri); + xpc_dictionary_set_string(request, "job-id", argv[1]); + xpc_dictionary_set_string(request, "user", argv[2]); + xpc_dictionary_set_string(request, "title", argv[3]); + xpc_dictionary_set_string(request, "copies", argv[4]); + xpc_dictionary_set_string(request, "options", argv[5]); + xpc_dictionary_set_string(request, "auth-info-required", + getenv("AUTH_INFO_REQUIRED")); + if ((auth_negotiate = getenv("AUTH_NEGOTIATE")) != NULL) + xpc_dictionary_set_string(request, "auth-negotiate", auth_negotiate); + xpc_dictionary_set_fd(request, "stdin", fd); + xpc_dictionary_set_fd(request, "stderr", 2); + xpc_dictionary_set_fd(request, "side-channel", CUPS_SC_FD); + + sem = dispatch_semaphore_create(0); + response = NULL; + + xpc_connection_send_message_with_reply(conn, request, + dispatch_get_global_queue(0,0), + ^(xpc_object_t reply) + { + /* Save the response and wake up */ + if (xpc_get_type(reply) + == XPC_TYPE_DICTIONARY) + response = xpc_retain(reply); + + dispatch_semaphore_signal(sem); + }); + + dispatch_semaphore_wait(sem, DISPATCH_TIME_FOREVER); + xpc_release(request); + dispatch_release(sem); + + if (response) + { + child_pid = xpc_dictionary_get_int64(response, "child-pid"); + + xpc_release(response); + + if (child_pid) + fprintf(stderr, "DEBUG: Child PID=%d.\n", child_pid); + else + { + _cupsLangPrintFilter(stderr, "ERROR", + _("Unable to start backend process.")); + fputs("DEBUG: No child PID.\n", stderr); + goto cleanup; + } + } + else + { + _cupsLangPrintFilter(stderr, "ERROR", + _("Unable to start backend process.")); + fputs("DEBUG: No reply from agent.\n", stderr); + goto cleanup; + } + + /* + * Then wait for the backend to finish... + */ + + request = xpc_dictionary_create(NULL, NULL, 0); + xpc_dictionary_set_int64(request, "command", kPMWaitForJob); + xpc_dictionary_set_fd(request, "stderr", 2); + + sem = dispatch_semaphore_create(0); + response = NULL; + + xpc_connection_send_message_with_reply(conn, request, + dispatch_get_global_queue(0,0), + ^(xpc_object_t reply) + { + /* Save the response and wake up */ + if (xpc_get_type(reply) + == XPC_TYPE_DICTIONARY) + response = xpc_retain(reply); + + dispatch_semaphore_signal(sem); + }); + + dispatch_semaphore_wait(sem, DISPATCH_TIME_FOREVER); + xpc_release(request); + dispatch_release(sem); + + if (response) + { + status = xpc_dictionary_get_int64(response, "status"); + + if (status == SIGTERM || status == SIGKILL || status == SIGPIPE) + { + fprintf(stderr, "DEBUG: Child terminated on signal %d.\n", status); + status = CUPS_BACKEND_FAILED; + } + else if (WIFSIGNALED(status)) + { + fprintf(stderr, "DEBUG: Child crashed on signal %d.\n", status); + status = CUPS_BACKEND_STOP; + } + else if (WIFEXITED(status)) + { + status = WEXITSTATUS(status); + fprintf(stderr, "DEBUG: Child exited with status %d.\n", status); + } + + xpc_release(response); + } + else + _cupsLangPrintFilter(stderr, "ERROR", + _("Unable to get backend exit status.")); + + cleanup: + + if (conn) + { + xpc_connection_suspend(conn); + xpc_connection_cancel(conn); + xpc_release(conn); + } + + return (status); +} +#endif /* HAVE_GSSAPI && HAVE_XPC */ + + +/* + * 'sigterm_handler()' - Handle 'terminate' signals that stop the backend. + */ + +static void +sigterm_handler(int sig) /* I - Signal */ +{ + (void)sig; /* remove compiler warnings... */ + +#if defined(HAVE_GSSAPI) && defined(HAVE_XPC) + if (child_pid) + { + kill(child_pid, sig); + child_pid = 0; + } +#endif /* HAVE_GSSAPI && HAVE_XPC */ + + if (!job_canceled) + { + /* + * Flag that the job should be canceled... + */ + + job_canceled = 1; + return; + } + + /* + * The scheduler already tried to cancel us once, now just terminate + * after removing our temp file! + */ + + if (tmpfilename[0]) + unlink(tmpfilename); + + exit(1); +} + + +/* + * 'timeout_cb()' - Handle HTTP timeouts. + */ + +static int /* O - 1 to continue, 0 to cancel */ +timeout_cb(http_t *http, /* I - Connection to server (unused) */ + void *user_data) /* I - User data (unused) */ +{ + (void)http; + (void)user_data; + + return (!job_canceled); +} + + +/* + * 'update_reasons()' - Update the printer-state-reasons values. + */ + +static void +update_reasons(ipp_attribute_t *attr, /* I - printer-state-reasons or NULL */ + const char *s) /* I - STATE: string or NULL */ +{ + char op; /* Add (+), remove (-), replace (\0) */ + cups_array_t *new_reasons; /* New reasons array */ + char *reason, /* Current reason */ + add[2048], /* Reasons added string */ + *addptr, /* Pointer into add string */ + rem[2048], /* Reasons removed string */ + *remptr; /* Pointer into remove string */ + const char *addprefix, /* Current add string prefix */ + *remprefix; /* Current remove string prefix */ + + + fprintf(stderr, "DEBUG: update_reasons(attr=%d(%s%s), s=\"%s\")\n", + attr ? attr->num_values : 0, attr ? attr->values[0].string.text : "", + attr && attr->num_values > 1 ? ",..." : "", s ? s : "(null)"); + + /* + * Create an array of new reason keyword strings... + */ + + if (attr) + { + int i; /* Looping var */ + + new_reasons = cupsArrayNew((cups_array_func_t)strcmp, NULL); + op = '\0'; + + for (i = 0; i < attr->num_values; i ++) + { + reason = attr->values[i].string.text; + + if (strcmp(reason, "none") && + strcmp(reason, "none-report") && + strcmp(reason, "paused") && + strncmp(reason, "spool-area-full", 15) && + strcmp(reason, "com.apple.print.recoverable-warning") && + strncmp(reason, "cups-", 5)) + cupsArrayAdd(new_reasons, reason); + } + } + else if (s) + { + if (*s == '+' || *s == '-') + op = *s++; + else + op = '\0'; + + new_reasons = _cupsArrayNewStrings(s); + } + else + return; + + /* + * Compute the changes... + */ + + add[0] = '\0'; + addprefix = "STATE: +"; + addptr = add; + rem[0] = '\0'; + remprefix = "STATE: -"; + remptr = rem; + + fprintf(stderr, "DEBUG2: op='%c', new_reasons=%d, state_reasons=%d\n", + op ? op : ' ', cupsArrayCount(new_reasons), + cupsArrayCount(state_reasons)); + + _cupsMutexLock(&report_mutex); + + if (op == '+') + { + /* + * Add reasons... + */ + + for (reason = (char *)cupsArrayFirst(new_reasons); + reason; + reason = (char *)cupsArrayNext(new_reasons)) + { + if (!cupsArrayFind(state_reasons, reason)) + { + if (!strncmp(reason, "cups-remote-", 12)) + { + /* + * If we are setting cups-remote-xxx, remove all other cups-remote-xxx + * keywords... + */ + + char *temp; /* Current reason in state_reasons */ + + cupsArraySave(state_reasons); + + for (temp = (char *)cupsArrayFirst(state_reasons); + temp; + temp = (char *)cupsArrayNext(state_reasons)) + if (!strncmp(temp, "cups-remote-", 12)) + { + snprintf(remptr, sizeof(rem) - (remptr - rem), "%s%s", remprefix, + temp); + remptr += strlen(remptr); + remprefix = ","; + + cupsArrayRemove(state_reasons, temp); + break; + } + + cupsArrayRestore(state_reasons); + } + + cupsArrayAdd(state_reasons, reason); + + snprintf(addptr, sizeof(add) - (addptr - add), "%s%s", addprefix, + reason); + addptr += strlen(addptr); + addprefix = ","; + } + } + } + else if (op == '-') + { + /* + * Remove reasons... + */ + + for (reason = (char *)cupsArrayFirst(new_reasons); + reason; + reason = (char *)cupsArrayNext(new_reasons)) + { + if (cupsArrayFind(state_reasons, reason)) + { + snprintf(remptr, sizeof(rem) - (remptr - rem), "%s%s", remprefix, + reason); + remptr += strlen(remptr); + remprefix = ","; + + cupsArrayRemove(state_reasons, reason); + } + } + } + else + { + /* + * Replace reasons... + */ + + for (reason = (char *)cupsArrayFirst(state_reasons); + reason; + reason = (char *)cupsArrayNext(state_reasons)) + { + if (strncmp(reason, "cups-", 5) && !cupsArrayFind(new_reasons, reason)) + { + snprintf(remptr, sizeof(rem) - (remptr - rem), "%s%s", remprefix, + reason); + remptr += strlen(remptr); + remprefix = ","; + + cupsArrayRemove(state_reasons, reason); + } + } + + for (reason = (char *)cupsArrayFirst(new_reasons); + reason; + reason = (char *)cupsArrayNext(new_reasons)) + { + if (!cupsArrayFind(state_reasons, reason)) + { + cupsArrayAdd(state_reasons, reason); + + snprintf(addptr, sizeof(add) - (addptr - add), "%s%s", addprefix, + reason); + addptr += strlen(addptr); + addprefix = ","; + } + } + } + + _cupsMutexUnlock(&report_mutex); + + /* + * Report changes and return... + */ + + if (add[0] && rem[0]) + fprintf(stderr, "%s\n%s\n", add, rem); + else if (add[0]) + fprintf(stderr, "%s\n", add); + else if (rem[0]) + fprintf(stderr, "%s\n", rem); +} + +/* + * End of "$Id$". + */ diff --git a/backend/lpd.c b/backend/lpd.c new file mode 100644 index 0000000000..006bb5831e --- /dev/null +++ b/backend/lpd.c @@ -0,0 +1,1342 @@ +/* + * "$Id$" + * + * Line Printer Daemon backend for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Send a file to the printer or server. + * lpd_command() - Send an LPR command sequence and wait for a reply. + * lpd_queue() - Queue a file using the Line Printer Daemon protocol. + * lpd_timeout() - Handle timeout alarms... + * lpd_write() - Write a buffer of data to an LPD server. + * rresvport_af() - A simple implementation of rresvport_af(). + * sigterm_handler() - Handle 'terminate' signals that stop the backend. + */ + +/* + * Include necessary headers. + */ + +#include +#include "backend-private.h" +#include +#include +#include +#include + +#ifdef WIN32 +# include +#else +# include +# include +# include +# include +#endif /* WIN32 */ +#ifdef __APPLE__ +# include +# include +#endif /* __APPLE__ */ + + +/* + * Globals... + */ + +static char tmpfilename[1024] = ""; /* Temporary spool file name */ +static int abort_job = 0; /* Non-zero if we get SIGTERM */ + + +/* + * Print mode... + */ + +#define MODE_STANDARD 0 /* Queue a copy */ +#define MODE_STREAM 1 /* Stream a copy */ + + +/* + * The order for control and data files in LPD requests... + */ + +#define ORDER_CONTROL_DATA 0 /* Control file first, then data */ +#define ORDER_DATA_CONTROL 1 /* Data file first, then control */ + + +/* + * What to reserve... + */ + +#define RESERVE_NONE 0 /* Don't reserve a priviledged port */ +#define RESERVE_RFC1179 1 /* Reserve port 721-731 */ +#define RESERVE_ANY 2 /* Reserve port 1-1023 */ + + +/* + * Local functions... + */ + +static int lpd_command(int lpd_fd, int timeout, char *format, ...); +static int lpd_queue(const char *hostname, http_addrlist_t *addrlist, + const char *printer, int print_fd, int snmp_fd, + int mode, const char *user, const char *title, + int copies, int banner, int format, int order, + int reserve, int manual_copies, int timeout, + int contimeout); +static void lpd_timeout(int sig); +static int lpd_write(int lpd_fd, char *buffer, int length); +#ifndef HAVE_RRESVPORT_AF +static int rresvport_af(int *port, int family); +#endif /* !HAVE_RRESVPORT_AF */ +static void sigterm_handler(int sig); + + +/* + * 'main()' - Send a file to the printer or server. + * + * Usage: + * + * printer-uri job-id user title copies options [file] + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments (6 or 7) */ + char *argv[]) /* I - Command-line arguments */ +{ + const char *device_uri; /* Device URI */ + char scheme[255], /* Scheme in URI */ + hostname[1024], /* Hostname */ + username[255], /* Username info */ + resource[1024], /* Resource info (printer name) */ + *options, /* Pointer to options */ + *name, /* Name of option */ + *value, /* Value of option */ + sep, /* Separator character */ + *filename, /* File to print */ + title[256]; /* Title string */ + int port; /* Port number */ + char portname[256]; /* Port name (string) */ + http_addrlist_t *addrlist; /* List of addresses for printer */ + int snmp_fd; /* SNMP socket */ + int fd; /* Print file */ + int status; /* Status of LPD job */ + int mode; /* Print mode */ + int banner; /* Print banner page? */ + int format; /* Print format */ + int order; /* Order of control/data files */ + int reserve; /* Reserve priviledged port? */ + int sanitize_title; /* Sanitize title string? */ + int manual_copies, /* Do manual copies? */ + timeout, /* Timeout */ + contimeout, /* Connection timeout */ + copies; /* Number of copies */ + ssize_t bytes = 0; /* Initial bytes read */ + char buffer[16384]; /* Initial print buffer */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* Actions for POSIX signals */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ + + + /* + * Make sure status messages are not buffered... + */ + + setbuf(stderr, NULL); + + /* + * Ignore SIGPIPE and catch SIGTERM signals... + */ + +#ifdef HAVE_SIGSET + sigset(SIGPIPE, SIG_IGN); + sigset(SIGTERM, sigterm_handler); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + action.sa_handler = SIG_IGN; + sigaction(SIGPIPE, &action, NULL); + + sigemptyset(&action.sa_mask); + sigaddset(&action.sa_mask, SIGTERM); + action.sa_handler = sigterm_handler; + sigaction(SIGTERM, &action, NULL); +#else + signal(SIGPIPE, SIG_IGN); + signal(SIGTERM, sigterm_handler); +#endif /* HAVE_SIGSET */ + + /* + * Check command-line... + */ + + if (argc == 1) + { + printf("network lpd \"Unknown\" \"%s\"\n", + _cupsLangString(cupsLangDefault(), _("LPD/LPR Host or Printer"))); + return (CUPS_BACKEND_OK); + } + else if (argc < 6 || argc > 7) + { + _cupsLangPrintf(stderr, + _("Usage: %s job-id user title copies options [file]"), + argv[0]); + return (CUPS_BACKEND_FAILED); + } + + /* + * Extract the hostname and printer name from the URI... + */ + + while ((device_uri = cupsBackendDeviceURI(argv)) == NULL) + { + _cupsLangPrintFilter(stderr, "INFO", _("Unable to locate printer.")); + sleep(10); + + if (getenv("CLASS") != NULL) + return (CUPS_BACKEND_FAILED); + } + + httpSeparateURI(HTTP_URI_CODING_ALL, device_uri, scheme, sizeof(scheme), + username, sizeof(username), hostname, sizeof(hostname), &port, + resource, sizeof(resource)); + + if (!port) + port = 515; /* Default to port 515 */ + + if (!username[0]) + { + /* + * If no username is in the device URI, then use the print job user... + */ + + strlcpy(username, argv[2], sizeof(username)); + } + + /* + * See if there are any options... + */ + + mode = MODE_STANDARD; + banner = 0; + format = 'l'; + order = ORDER_CONTROL_DATA; + reserve = RESERVE_ANY; + manual_copies = 1; + timeout = 300; + contimeout = 7 * 24 * 60 * 60; + +#ifdef __APPLE__ + /* + * We want to pass UTF-8 characters by default, not re-map them (3071945) + */ + + sanitize_title = 0; +#else + /* + * Otherwise we want to re-map UTF-8 to "safe" characters by default... + */ + + sanitize_title = 1; +#endif /* __APPLE__ */ + + if ((options = strchr(resource, '?')) != NULL) + { + /* + * Yup, terminate the device name string and move to the first + * character of the options... + */ + + *options++ = '\0'; + + /* + * Parse options... + */ + + while (*options) + { + /* + * Get the name... + */ + + name = options; + + while (*options && *options != '=' && *options != '+' && *options != '&') + options ++; + + if ((sep = *options) != '\0') + *options++ = '\0'; + + if (sep == '=') + { + /* + * Get the value... + */ + + value = options; + + while (*options && *options != '+' && *options != '&') + options ++; + + if (*options) + *options++ = '\0'; + } + else + value = (char *)""; + + /* + * Process the option... + */ + + if (!_cups_strcasecmp(name, "banner")) + { + /* + * Set the banner... + */ + + banner = !value[0] || !_cups_strcasecmp(value, "on") || + !_cups_strcasecmp(value, "yes") || !_cups_strcasecmp(value, "true"); + } + else if (!_cups_strcasecmp(name, "format") && value[0]) + { + /* + * Set output format... + */ + + if (strchr("cdfglnoprtv", value[0])) + format = value[0]; + else + _cupsLangPrintFilter(stderr, "ERROR", + _("Unknown format character: \"%c\"."), + value[0]); + } + else if (!_cups_strcasecmp(name, "mode") && value[0]) + { + /* + * Set control/data order... + */ + + if (!_cups_strcasecmp(value, "standard")) + mode = MODE_STANDARD; + else if (!_cups_strcasecmp(value, "stream")) + mode = MODE_STREAM; + else + _cupsLangPrintFilter(stderr, "ERROR", + _("Unknown print mode: \"%s\"."), value); + } + else if (!_cups_strcasecmp(name, "order") && value[0]) + { + /* + * Set control/data order... + */ + + if (!_cups_strcasecmp(value, "control,data")) + order = ORDER_CONTROL_DATA; + else if (!_cups_strcasecmp(value, "data,control")) + order = ORDER_DATA_CONTROL; + else + _cupsLangPrintFilter(stderr, "ERROR", + _("Unknown file order: \"%s\"."), value); + } + else if (!_cups_strcasecmp(name, "reserve")) + { + /* + * Set port reservation mode... + */ + + if (!value[0] || !_cups_strcasecmp(value, "on") || + !_cups_strcasecmp(value, "yes") || !_cups_strcasecmp(value, "true") || + !_cups_strcasecmp(value, "rfc1179")) + reserve = RESERVE_RFC1179; + else if (!_cups_strcasecmp(value, "any")) + reserve = RESERVE_ANY; + else + reserve = RESERVE_NONE; + } + else if (!_cups_strcasecmp(name, "manual_copies")) + { + /* + * Set manual copies... + */ + + manual_copies = !value[0] || !_cups_strcasecmp(value, "on") || + !_cups_strcasecmp(value, "yes") || !_cups_strcasecmp(value, "true"); + } + else if (!_cups_strcasecmp(name, "sanitize_title")) + { + /* + * Set sanitize title... + */ + + sanitize_title = !value[0] || !_cups_strcasecmp(value, "on") || + !_cups_strcasecmp(value, "yes") || !_cups_strcasecmp(value, "true"); + } + else if (!_cups_strcasecmp(name, "timeout")) + { + /* + * Set the timeout... + */ + + if (atoi(value) > 0) + timeout = atoi(value); + } + else if (!_cups_strcasecmp(name, "contimeout")) + { + /* + * Set the connection timeout... + */ + + if (atoi(value) > 0) + contimeout = atoi(value); + } + } + } + + if (mode == MODE_STREAM) + order = ORDER_CONTROL_DATA; + + /* + * Find the printer... + */ + + snprintf(portname, sizeof(portname), "%d", port); + + fputs("STATE: +connecting-to-device\n", stderr); + fprintf(stderr, "DEBUG: Looking up \"%s\"...\n", hostname); + + while ((addrlist = httpAddrGetList(hostname, AF_UNSPEC, portname)) == NULL) + { + _cupsLangPrintFilter(stderr, "INFO", + _("Unable to locate printer \"%s\"."), hostname); + sleep(10); + + if (getenv("CLASS") != NULL) + { + fputs("STATE: -connecting-to-device\n", stderr); + exit(CUPS_BACKEND_FAILED); + } + } + + snmp_fd = _cupsSNMPOpen(addrlist->addr.addr.sa_family); + + /* + * Wait for data from the filter... + */ + + if (argc == 6) + { + if (!backendWaitLoop(snmp_fd, &(addrlist->addr), 0, backendNetworkSideCB)) + return (CUPS_BACKEND_OK); + else if (mode == MODE_STANDARD && + (bytes = read(0, buffer, sizeof(buffer))) <= 0) + return (CUPS_BACKEND_OK); + } + + /* + * If we have 7 arguments, print the file named on the command-line. + * Otherwise, copy stdin to a temporary file and print the temporary + * file. + */ + + if (argc == 6 && mode == MODE_STANDARD) + { + /* + * Copy stdin to a temporary file... + */ + + if ((fd = cupsTempFd(tmpfilename, sizeof(tmpfilename))) < 0) + { + perror("DEBUG: Unable to create temporary file"); + return (CUPS_BACKEND_FAILED); + } + + _cupsLangPrintFilter(stderr, "INFO", _("Copying print data.")); + + if (bytes > 0) + write(fd, buffer, bytes); + + backendRunLoop(-1, fd, snmp_fd, &(addrlist->addr), 0, 0, + backendNetworkSideCB); + } + else if (argc == 6) + { + /* + * Stream from stdin... + */ + + filename = NULL; + fd = 0; + } + else + { + filename = argv[6]; + fd = open(filename, O_RDONLY); + + if (fd == -1) + { + _cupsLangPrintError("ERROR", _("Unable to open print file")); + return (CUPS_BACKEND_FAILED); + } + } + + /* + * Sanitize the document title... + */ + + strlcpy(title, argv[3], sizeof(title)); + + if (sanitize_title) + { + /* + * Sanitize the title string so that we don't cause problems on + * the remote end... + */ + + char *ptr; + + for (ptr = title; *ptr; ptr ++) + if (!isalnum(*ptr & 255) && !isspace(*ptr & 255)) + *ptr = '_'; + } + + /* + * Queue the job... + */ + + if (argc > 6) + { + if (manual_copies) + { + manual_copies = atoi(argv[4]); + copies = 1; + } + else + { + manual_copies = 1; + copies = atoi(argv[4]); + } + + status = lpd_queue(hostname, addrlist, resource + 1, fd, snmp_fd, mode, + username, title, copies, banner, format, order, reserve, + manual_copies, timeout, contimeout); + + if (!status) + fprintf(stderr, "PAGE: 1 %d\n", atoi(argv[4])); + } + else + status = lpd_queue(hostname, addrlist, resource + 1, fd, snmp_fd, mode, + username, title, 1, banner, format, order, reserve, 1, + timeout, contimeout); + + /* + * Remove the temporary file if necessary... + */ + + if (tmpfilename[0]) + unlink(tmpfilename); + + if (fd) + close(fd); + + if (snmp_fd >= 0) + _cupsSNMPClose(snmp_fd); + + /* + * Return the queue status... + */ + + return (status); +} + + +/* + * 'lpd_command()' - Send an LPR command sequence and wait for a reply. + */ + +static int /* O - Status of command */ +lpd_command(int fd, /* I - Socket connection to LPD host */ + int timeout, /* I - Seconds to wait for a response */ + char *format, /* I - printf()-style format string */ + ...) /* I - Additional args as necessary */ +{ + va_list ap; /* Argument pointer */ + char buf[1024]; /* Output buffer */ + int bytes; /* Number of bytes to output */ + char status; /* Status from command */ + + + /* + * Don't try to send commands if the job has been canceled... + */ + + if (abort_job) + return (-1); + + /* + * Format the string... + */ + + va_start(ap, format); + bytes = vsnprintf(buf, sizeof(buf), format, ap); + va_end(ap); + + fprintf(stderr, "DEBUG: lpd_command %2.2x %s", buf[0], buf + 1); + + /* + * Send the command... + */ + + fprintf(stderr, "DEBUG: Sending command string (%d bytes)...\n", bytes); + + if (lpd_write(fd, buf, bytes) < bytes) + { + perror("DEBUG: Unable to send LPD command"); + return (-1); + } + + /* + * Read back the status from the command and return it... + */ + + fputs("DEBUG: Reading command status...\n", stderr); + + alarm(timeout); + + if (recv(fd, &status, 1, 0) < 1) + { + _cupsLangPrintFilter(stderr, "WARNING", + _("Printer did not respond after %d seconds."), + timeout); + status = errno; + } + + alarm(0); + + fprintf(stderr, "DEBUG: lpd_command returning %d\n", status); + + return (status); +} + + +/* + * 'lpd_queue()' - Queue a file using the Line Printer Daemon protocol. + */ + +static int /* O - Zero on success, non-zero on failure */ +lpd_queue(const char *hostname, /* I - Host to connect to */ + http_addrlist_t *addrlist, /* I - List of host addresses */ + const char *printer, /* I - Printer/queue name */ + int print_fd, /* I - File to print */ + int snmp_fd, /* I - SNMP socket */ + int mode, /* I - Print mode */ + const char *user, /* I - Requesting user */ + const char *title, /* I - Job title */ + int copies, /* I - Number of copies */ + int banner, /* I - Print LPD banner? */ + int format, /* I - Format specifier */ + int order, /* I - Order of data/control files */ + int reserve, /* I - Reserve ports? */ + int manual_copies,/* I - Do copies by hand... */ + int timeout, /* I - Timeout... */ + int contimeout) /* I - Connection timeout */ +{ + char localhost[255]; /* Local host name */ + int error; /* Error number */ + struct stat filestats; /* File statistics */ + int lport; /* LPD connection local port */ + int fd; /* LPD socket */ + char control[10240], /* LPD control 'file' */ + *cptr; /* Pointer into control file string */ + char status; /* Status byte from command */ + int delay; /* Delay for retries... */ + char addrname[256]; /* Address name */ + http_addrlist_t *addr; /* Socket address */ + int have_supplies; /* Printer supports supply levels? */ + int copy; /* Copies written */ + time_t start_time; /* Time of first connect */ + size_t nbytes; /* Number of bytes written */ + off_t tbytes; /* Total bytes written */ + char buffer[32768]; /* Output buffer */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* Actions for POSIX signals */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ + + + /* + * Setup an alarm handler for timeouts... + */ + +#ifdef HAVE_SIGSET /* Use System V signals over POSIX to avoid bugs */ + sigset(SIGALRM, lpd_timeout); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + + sigemptyset(&action.sa_mask); + action.sa_handler = lpd_timeout; + sigaction(SIGALRM, &action, NULL); +#else + signal(SIGALRM, lpd_timeout); +#endif /* HAVE_SIGSET */ + + /* + * Remember when we started trying to connect to the printer... + */ + + start_time = time(NULL); + + /* + * Loop forever trying to print the file... + */ + + while (!abort_job) + { + /* + * First try to reserve a port for this connection... + */ + + fprintf(stderr, "DEBUG: Connecting to %s:%d for printer %s\n", hostname, + _httpAddrPort(&(addrlist->addr)), printer); + _cupsLangPrintFilter(stderr, "INFO", _("Connecting to printer.")); + + for (lport = reserve == RESERVE_RFC1179 ? 732 : 1024, addr = addrlist, + delay = 5;; + addr = addr->next) + { + /* + * Stop if this job has been canceled... + */ + + if (abort_job) + return (CUPS_BACKEND_FAILED); + + /* + * Choose the next priviledged port... + */ + + if (!addr) + addr = addrlist; + + lport --; + + if (lport < 721 && reserve == RESERVE_RFC1179) + lport = 731; + else if (lport < 1) + lport = 1023; + +#ifdef HAVE_GETEUID + if (geteuid() || !reserve) +#else + if (getuid() || !reserve) +#endif /* HAVE_GETEUID */ + { + /* + * Just create a regular socket... + */ + + if ((fd = socket(addr->addr.addr.sa_family, SOCK_STREAM, 0)) < 0) + { + perror("DEBUG: Unable to create socket"); + sleep(1); + + continue; + } + + lport = 0; + } + else + { + /* + * We're running as root and want to comply with RFC 1179. Reserve a + * priviledged lport between 721 and 731... + */ + + if ((fd = rresvport_af(&lport, addr->addr.addr.sa_family)) < 0) + { + perror("DEBUG: Unable to reserve port"); + sleep(1); + + continue; + } + } + + /* + * Connect to the printer or server... + */ + + if (abort_job) + { + close(fd); + + return (CUPS_BACKEND_FAILED); + } + + if (!connect(fd, &(addr->addr.addr), httpAddrLength(&(addr->addr)))) + break; + + error = errno; + close(fd); + + if (addr->next) + continue; + + if (getenv("CLASS") != NULL) + { + /* + * If the CLASS environment variable is set, the job was submitted + * to a class and not to a specific queue. In this case, we want + * to abort immediately so that the job can be requeued on the next + * available printer in the class. + */ + + _cupsLangPrintFilter(stderr, "INFO", + _("Unable to contact printer, queuing on next " + "printer in class.")); + + /* + * Sleep 5 seconds to keep the job from requeuing too rapidly... + */ + + sleep(5); + + return (CUPS_BACKEND_FAILED); + } + + fprintf(stderr, "DEBUG: Connection error: %s\n", strerror(error)); + + if (error == ECONNREFUSED || error == EHOSTDOWN || + error == EHOSTUNREACH) + { + if (contimeout && (time(NULL) - start_time) > contimeout) + { + _cupsLangPrintFilter(stderr, "ERROR", + _("The printer is not responding.")); + return (CUPS_BACKEND_FAILED); + } + + switch (error) + { + case EHOSTDOWN : + _cupsLangPrintFilter(stderr, "WARNING", + _("The printer may not exist or " + "is unavailable at this time.")); + break; + + case EHOSTUNREACH : + _cupsLangPrintFilter(stderr, "WARNING", + _("The printer is unreachable at " + "this time.")); + break; + + case ECONNREFUSED : + default : + _cupsLangPrintFilter(stderr, "WARNING", + _("The printer is busy.")); + break; + } + + sleep(delay); + + if (delay < 30) + delay += 5; + } + else if (error == EADDRINUSE) + { + /* + * Try on another port... + */ + + sleep(1); + } + else + { + _cupsLangPrintFilter(stderr, "ERROR", + _("The printer is not responding.")); + sleep(30); + } + } + + fputs("STATE: -connecting-to-device\n", stderr); + _cupsLangPrintFilter(stderr, "INFO", _("Connected to printer.")); + + fprintf(stderr, "DEBUG: Connected to %s:%d (local port %d)...\n", + httpAddrString(&(addr->addr), addrname, sizeof(addrname)), + _httpAddrPort(&(addr->addr)), lport); + + /* + * See if the printer supports SNMP... + */ + + if (snmp_fd >= 0) + have_supplies = !backendSNMPSupplies(snmp_fd, &(addrlist->addr), NULL, + NULL); + else + have_supplies = 0; + + /* + * Check for side-channel requests... + */ + + backendCheckSideChannel(snmp_fd, &(addrlist->addr)); + + /* + * Next, open the print file and figure out its size... + */ + + if (print_fd) + { + /* + * Use the size from the print file... + */ + + if (fstat(print_fd, &filestats)) + { + close(fd); + + perror("DEBUG: unable to stat print file"); + return (CUPS_BACKEND_FAILED); + } + + filestats.st_size *= manual_copies; + } + else + { + /* + * Use a "very large value" for the size so that the printer will + * keep printing until we close the connection... + */ + +#ifdef _LARGEFILE_SOURCE + filestats.st_size = (size_t)(999999999999.0); +#else + filestats.st_size = 2147483647; +#endif /* _LARGEFILE_SOURCE */ + } + + /* + * Send a job header to the printer, specifying no banner page and + * literal output... + */ + + if (lpd_command(fd, timeout, "\002%s\n", + printer)) /* Receive print job(s) */ + { + close(fd); + return (CUPS_BACKEND_FAILED); + } + + httpGetHostname(NULL, localhost, sizeof(localhost)); + + snprintf(control, sizeof(control), + "H%.31s\n" /* RFC 1179, Section 7.2 - host name <= 31 chars */ + "P%.31s\n" /* RFC 1179, Section 7.2 - user name <= 31 chars */ + "J%.99s\n", /* RFC 1179, Section 7.2 - job name <= 99 chars */ + localhost, user, title); + cptr = control + strlen(control); + + if (banner) + { + snprintf(cptr, sizeof(control) - (cptr - control), + "C%.31s\n" /* RFC 1179, Section 7.2 - class name <= 31 chars */ + "L%s\n", + localhost, user); + cptr += strlen(cptr); + } + + while (copies > 0) + { + snprintf(cptr, sizeof(control) - (cptr - control), "%cdfA%03d%.15s\n", + format, (int)getpid() % 1000, localhost); + cptr += strlen(cptr); + copies --; + } + + snprintf(cptr, sizeof(control) - (cptr - control), + "UdfA%03d%.15s\n" + "N%.131s\n", /* RFC 1179, Section 7.2 - sourcefile name <= 131 chars */ + (int)getpid() % 1000, localhost, title); + + fprintf(stderr, "DEBUG: Control file is:\n%s", control); + + if (order == ORDER_CONTROL_DATA) + { + /* + * Check for side-channel requests... + */ + + backendCheckSideChannel(snmp_fd, &(addr->addr)); + + /* + * Send the control file... + */ + + if (lpd_command(fd, timeout, "\002%d cfA%03.3d%.15s\n", strlen(control), + (int)getpid() % 1000, localhost)) + { + close(fd); + + return (CUPS_BACKEND_FAILED); + } + + fprintf(stderr, "DEBUG: Sending control file (%u bytes)\n", + (unsigned)strlen(control)); + + if (lpd_write(fd, control, strlen(control) + 1) < (strlen(control) + 1)) + { + status = errno; + perror("DEBUG: Unable to write control file"); + + } + else + { + alarm(timeout); + + if (read(fd, &status, 1) < 1) + { + _cupsLangPrintFilter(stderr, "WARNING", + _("Printer did not respond after %d seconds."), + timeout); + status = errno; + } + + alarm(0); + } + + if (status != 0) + _cupsLangPrintFilter(stderr, "ERROR", + _("Remote host did not accept control file (%d)."), + status); + else + _cupsLangPrintFilter(stderr, "INFO", + _("Control file sent successfully.")); + } + else + status = 0; + + if (status == 0) + { + /* + * Check for side-channel requests... + */ + + backendCheckSideChannel(snmp_fd, &(addr->addr)); + + /* + * Send the print file... + */ + + if (lpd_command(fd, timeout, "\003" CUPS_LLFMT " dfA%03.3d%.15s\n", + CUPS_LLCAST filestats.st_size, (int)getpid() % 1000, + localhost)) + { + close(fd); + + return (CUPS_BACKEND_FAILED); + } + + fprintf(stderr, "DEBUG: Sending data file (" CUPS_LLFMT " bytes)\n", + CUPS_LLCAST filestats.st_size); + + tbytes = 0; + for (copy = 0; copy < manual_copies; copy ++) + { + lseek(print_fd, 0, SEEK_SET); + + while ((nbytes = read(print_fd, buffer, sizeof(buffer))) > 0) + { + _cupsLangPrintFilter(stderr, "INFO", + _("Spooling job, %.0f%% complete."), + 100.0 * tbytes / filestats.st_size); + + if (lpd_write(fd, buffer, nbytes) < nbytes) + { + perror("DEBUG: Unable to send print file to printer"); + break; + } + else + tbytes += nbytes; + } + } + + if (mode == MODE_STANDARD) + { + if (tbytes < filestats.st_size) + status = errno; + else if (lpd_write(fd, "", 1) < 1) + { + perror("DEBUG: Unable to send trailing nul to printer"); + status = errno; + } + else + { + /* + * Read the status byte from the printer; if we can't read the byte + * back now, we should set status to "errno", however at this point + * we know the printer got the whole file and we don't necessarily + * want to requeue it over and over... + */ + + alarm(timeout); + + if (recv(fd, &status, 1, 0) < 1) + { + _cupsLangPrintFilter(stderr, "WARNING", + _("Printer did not respond after %d seconds."), + timeout); + status = 0; + } + + alarm(0); + } + } + else + status = 0; + + if (status != 0) + _cupsLangPrintFilter(stderr, "ERROR", + _("Remote host did not accept data file (%d)."), + status); + else + _cupsLangPrintFilter(stderr, "INFO", + _("Data file sent successfully.")); + } + + if (status == 0 && order == ORDER_DATA_CONTROL) + { + /* + * Check for side-channel requests... + */ + + backendCheckSideChannel(snmp_fd, &(addr->addr)); + + /* + * Send control file... + */ + + if (lpd_command(fd, timeout, "\002%d cfA%03.3d%.15s\n", strlen(control), + (int)getpid() % 1000, localhost)) + { + close(fd); + + return (CUPS_BACKEND_FAILED); + } + + fprintf(stderr, "DEBUG: Sending control file (%lu bytes)\n", + (unsigned long)strlen(control)); + + if (lpd_write(fd, control, strlen(control) + 1) < (strlen(control) + 1)) + { + status = errno; + perror("DEBUG: Unable to write control file"); + } + else + { + alarm(timeout); + + if (read(fd, &status, 1) < 1) + { + _cupsLangPrintFilter(stderr, "WARNING", + _("Printer did not respond after %d seconds."), + timeout); + status = errno; + } + + alarm(0); + } + + if (status != 0) + _cupsLangPrintFilter(stderr, "ERROR", + _("Remote host did not accept control file (%d)."), + status); + else + _cupsLangPrintFilter(stderr, "INFO", + _("Control file sent successfully.")); + } + + /* + * Collect the final supply levels as needed... + */ + + if (have_supplies) + backendSNMPSupplies(snmp_fd, &(addr->addr), NULL, NULL); + + /* + * Close the socket connection and input file... + */ + + close(fd); + + if (status == 0) + return (CUPS_BACKEND_OK); + + /* + * Waiting for a retry... + */ + + sleep(30); + } + + /* + * If we get here, then the job has been canceled... + */ + + return (CUPS_BACKEND_FAILED); +} + + +/* + * 'lpd_timeout()' - Handle timeout alarms... + */ + +static void +lpd_timeout(int sig) /* I - Signal number */ +{ + (void)sig; + +#if !defined(HAVE_SIGSET) && !defined(HAVE_SIGACTION) + signal(SIGALRM, lpd_timeout); +#endif /* !HAVE_SIGSET && !HAVE_SIGACTION */ +} + + +/* + * 'lpd_write()' - Write a buffer of data to an LPD server. + */ + +static int /* O - Number of bytes written or -1 on error */ +lpd_write(int lpd_fd, /* I - LPD socket */ + char *buffer, /* I - Buffer to write */ + int length) /* I - Number of bytes to write */ +{ + int bytes, /* Number of bytes written */ + total; /* Total number of bytes written */ + + + if (abort_job) + return (-1); + + total = 0; + while ((bytes = send(lpd_fd, buffer, length - total, 0)) >= 0) + { + total += bytes; + buffer += bytes; + + if (total == length) + break; + } + + if (bytes < 0) + return (-1); + else + return (length); +} + + +#ifndef HAVE_RRESVPORT_AF +/* + * 'rresvport_af()' - A simple implementation of rresvport_af(). + */ + +static int /* O - Socket or -1 on error */ +rresvport_af(int *port, /* IO - Port number to bind to */ + int family) /* I - Address family */ +{ + http_addr_t addr; /* Socket address */ + int fd; /* Socket file descriptor */ + + + /* + * Try to create an IPv4 socket... + */ + + if ((fd = socket(family, SOCK_STREAM, 0)) < 0) + return (-1); + + /* + * Initialize the address buffer... + */ + + memset(&addr, 0, sizeof(addr)); + addr.addr.sa_family = family; + + /* + * Try to bind the socket to a reserved port... + */ + + while (*port > 511) + { + /* + * Set the port number... + */ + + _httpAddrSetPort(&addr, *port); + + /* + * Try binding the port to the socket; return if all is OK... + */ + + if (!bind(fd, (struct sockaddr *)&addr, sizeof(addr))) + return (fd); + + /* + * Stop if we have any error other than "address already in use"... + */ + + if (errno != EADDRINUSE) + { +# ifdef WIN32 + closesocket(fd); +# else + close(fd); +# endif /* WIN32 */ + + return (-1); + } + + /* + * Try the next port... + */ + + (*port)--; + } + + /* + * Wasn't able to bind to a reserved port, so close the socket and return + * -1... + */ + +# ifdef WIN32 + closesocket(fd); +# else + close(fd); +# endif /* WIN32 */ + + return (-1); +} +#endif /* !HAVE_RRESVPORT_AF */ + + +/* + * 'sigterm_handler()' - Handle 'terminate' signals that stop the backend. + */ + +static void +sigterm_handler(int sig) /* I - Signal */ +{ + (void)sig; /* remove compiler warnings... */ + + abort_job = 1; +} + + +/* + * End of "$Id$". + */ diff --git a/backend/network.c b/backend/network.c new file mode 100644 index 0000000000..6dbdca03bd --- /dev/null +++ b/backend/network.c @@ -0,0 +1,300 @@ +/* + * "$Id$" + * + * Common backend network APIs for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 2006-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * backendCheckSideChannel() - Check the side-channel for pending requests. + * backendNetworkSideCB() - Handle common network side-channel commands. + */ + +/* + * Include necessary headers. + */ + +#include "backend-private.h" +#include +#ifdef __hpux +# include +#else +# include +#endif /* __hpux */ + + +/* + * 'backendCheckSideChannel()' - Check the side-channel for pending requests. + */ + + +void +backendCheckSideChannel( + int snmp_fd, /* I - SNMP socket */ + http_addr_t *addr) /* I - Address of device */ +{ + fd_set input; /* Select input set */ + struct timeval timeout; /* Select timeout */ + + + FD_ZERO(&input); + FD_SET(CUPS_SC_FD, &input); + + timeout.tv_sec = timeout.tv_usec = 0; + + if (select(CUPS_SC_FD + 1, &input, NULL, NULL, &timeout) > 0) + backendNetworkSideCB(-1, -1, snmp_fd, addr, 0); +} + + +/* + * 'backendNetworkSideCB()' - Handle common network side-channel commands. + */ + +int /* O - -1 on error, 0 on success */ +backendNetworkSideCB( + int print_fd, /* I - Print file or -1 */ + int device_fd, /* I - Device file or -1 */ + int snmp_fd, /* I - SNMP socket */ + http_addr_t *addr, /* I - Address of device */ + int use_bc) /* I - Use back-channel data? */ +{ + cups_sc_command_t command; /* Request command */ + cups_sc_status_t status; /* Request/response status */ + char data[65536]; /* Request/response data */ + int datalen; /* Request/response data size */ + const char *device_id; /* 1284DEVICEID env var */ + + + datalen = sizeof(data); + + if (cupsSideChannelRead(&command, &status, data, &datalen, 1.0)) + return (-1); + + switch (command) + { + case CUPS_SC_CMD_DRAIN_OUTPUT : + /* + * Our sockets disable the Nagle algorithm and data is sent immediately. + */ + + if (device_fd < 0) + status = CUPS_SC_STATUS_NOT_IMPLEMENTED; + else if (backendDrainOutput(print_fd, device_fd)) + status = CUPS_SC_STATUS_IO_ERROR; + else + status = CUPS_SC_STATUS_OK; + + datalen = 0; + break; + + case CUPS_SC_CMD_GET_BIDI : + status = CUPS_SC_STATUS_OK; + data[0] = use_bc; + datalen = 1; + break; + + case CUPS_SC_CMD_SNMP_GET : + case CUPS_SC_CMD_SNMP_GET_NEXT : + fprintf(stderr, "DEBUG: CUPS_SC_CMD_SNMP_%s: %d (%s)\n", + command == CUPS_SC_CMD_SNMP_GET ? "GET" : "GET_NEXT", datalen, + data); + + if (datalen < 2) + { + status = CUPS_SC_STATUS_BAD_MESSAGE; + datalen = 0; + break; + } + + if (snmp_fd >= 0) + { + cups_snmp_t packet; /* Packet from printer */ + + + if (!_cupsSNMPStringToOID(data, packet.object_name, CUPS_SNMP_MAX_OID)) + { + status = CUPS_SC_STATUS_BAD_MESSAGE; + datalen = 0; + break; + } + + status = CUPS_SC_STATUS_IO_ERROR; + datalen = 0; + + if (_cupsSNMPWrite(snmp_fd, addr, CUPS_SNMP_VERSION_1, + _cupsSNMPDefaultCommunity(), + command == CUPS_SC_CMD_SNMP_GET ? + CUPS_ASN1_GET_REQUEST : + CUPS_ASN1_GET_NEXT_REQUEST, 1, + packet.object_name)) + { + if (_cupsSNMPRead(snmp_fd, &packet, 1.0)) + { + char *dataptr; /* Pointer into data */ + int i; /* Looping var */ + + + if (!_cupsSNMPOIDToString(packet.object_name, data, sizeof(data))) + { + fputs("DEBUG: Bad OID returned!\n", stderr); + break; + } + + datalen = (int)strlen(data) + 1; + dataptr = data + datalen; + + switch (packet.object_type) + { + case CUPS_ASN1_BOOLEAN : + snprintf(dataptr, sizeof(data) - (dataptr - data), "%d", + packet.object_value.boolean); + datalen += (int)strlen(dataptr); + break; + + case CUPS_ASN1_INTEGER : + snprintf(dataptr, sizeof(data) - (dataptr - data), "%d", + packet.object_value.integer); + datalen += (int)strlen(dataptr); + break; + + case CUPS_ASN1_BIT_STRING : + case CUPS_ASN1_OCTET_STRING : + if (packet.object_value.string.num_bytes < 0) + i = 0; + else if (packet.object_value.string.num_bytes < + (sizeof(data) - (dataptr - data))) + i = packet.object_value.string.num_bytes; + else + i = (int)(sizeof(data) - (dataptr - data)); + + memcpy(dataptr, packet.object_value.string.bytes, i); + + datalen += i; + break; + + case CUPS_ASN1_OID : + _cupsSNMPOIDToString(packet.object_value.oid, dataptr, + sizeof(data) - (dataptr - data)); + datalen += (int)strlen(dataptr); + break; + + case CUPS_ASN1_HEX_STRING : + for (i = 0; + i < packet.object_value.string.num_bytes && + dataptr < (data + sizeof(data) - 3); + i ++, dataptr += 2) + sprintf(dataptr, "%02X", + packet.object_value.string.bytes[i]); + datalen += (int)strlen(dataptr); + break; + + case CUPS_ASN1_COUNTER : + snprintf(dataptr, sizeof(data) - (dataptr - data), "%d", + packet.object_value.counter); + datalen += (int)strlen(dataptr); + break; + + case CUPS_ASN1_GAUGE : + snprintf(dataptr, sizeof(data) - (dataptr - data), "%u", + packet.object_value.gauge); + datalen += (int)strlen(dataptr); + break; + + case CUPS_ASN1_TIMETICKS : + snprintf(dataptr, sizeof(data) - (dataptr - data), "%u", + packet.object_value.timeticks); + datalen += (int)strlen(dataptr); + break; + + default : + fprintf(stderr, "DEBUG: Unknown OID value type %02X!\n", + packet.object_type); + + case CUPS_ASN1_NULL_VALUE : + dataptr[0] = '\0'; + break; + } + + fprintf(stderr, "DEBUG: Returning %s %s\n", data, data + datalen); + + status = CUPS_SC_STATUS_OK; + } + else + fputs("DEBUG: SNMP read error...\n", stderr); + } + else + fputs("DEBUG: SNMP write error...\n", stderr); + break; + } + + status = CUPS_SC_STATUS_NOT_IMPLEMENTED; + datalen = 0; + break; + + case CUPS_SC_CMD_GET_DEVICE_ID : + if (snmp_fd >= 0) + { + cups_snmp_t packet; /* Packet from printer */ + static const int ppmPrinterIEEE1284DeviceId[] = + { CUPS_OID_ppmPrinterIEEE1284DeviceId,1,-1 }; + + + status = CUPS_SC_STATUS_IO_ERROR; + datalen = 0; + + if (_cupsSNMPWrite(snmp_fd, addr, CUPS_SNMP_VERSION_1, + _cupsSNMPDefaultCommunity(), + CUPS_ASN1_GET_REQUEST, 1, + ppmPrinterIEEE1284DeviceId)) + { + if (_cupsSNMPRead(snmp_fd, &packet, 1.0) && + packet.object_type == CUPS_ASN1_OCTET_STRING) + { + strlcpy(data, (char *)packet.object_value.string.bytes, + sizeof(data)); + datalen = (int)strlen(data); + status = CUPS_SC_STATUS_OK; + } + } + + break; + } + + if ((device_id = getenv("1284DEVICEID")) != NULL) + { + strlcpy(data, device_id, sizeof(data)); + datalen = (int)strlen(data); + status = CUPS_SC_STATUS_OK; + break; + } + + case CUPS_SC_CMD_GET_CONNECTED : + status = CUPS_SC_STATUS_OK; + data[0] = device_fd != -1; + datalen = 1; + break; + + default : + status = CUPS_SC_STATUS_NOT_IMPLEMENTED; + datalen = 0; + break; + } + + return (cupsSideChannelWrite(command, status, data, datalen, 1.0)); +} + + +/* + * End of "$Id$". + */ diff --git a/backend/pseudo b/backend/pseudo new file mode 100755 index 0000000000..3a82a522a4 --- /dev/null +++ b/backend/pseudo @@ -0,0 +1,30 @@ +#!/bin/sh +# +# "$Id$" +# +# Psuedo-backend for CUPS testing purposes. +# +# Copyright 2011 by Apple Inc. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# +# This file is subject to the Apple OS-Developed Software exception. +# + +if test $# = 0; then + echo 'direct pseudo:///deskjet "HP DeskJet" "HP DeskJet (pseudo)" "MFG:HP;MDL:DeskJet;CMD:PCL;" "Nowhere"' + echo 'direct pseudo:///laserjet "HP LaserJet" "HP LaserJet (pseudo)" "MFG:HP;MDL:LaserJet;CMD:PCL;" "Nowhere"' + exit 0 +fi + +cat $6 >/dev/null +sleep 5 +exit 0 + +# +# End of "$Id$". +# diff --git a/backend/runloop.c b/backend/runloop.c new file mode 100644 index 0000000000..fdcaf29aaa --- /dev/null +++ b/backend/runloop.c @@ -0,0 +1,523 @@ +/* + * "$Id$" + * + * Common run loop APIs for CUPS backends. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 2006-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * backendDrainOutput() - Drain pending print data to the device. + * backendRunLoop() - Read and write print and back-channel data. + * backendWaitLoop() - Wait for input from stdin while handling + * side-channel queries. + */ + +/* + * Include necessary headers. + */ + +#include "backend-private.h" +#include +#ifdef __hpux +# include +#else +# include +#endif /* __hpux */ + + +/* + * 'backendDrainOutput()' - Drain pending print data to the device. + */ + +int /* O - 0 on success, -1 on error */ +backendDrainOutput(int print_fd, /* I - Print file descriptor */ + int device_fd) /* I - Device file descriptor */ +{ + int nfds; /* Maximum file descriptor value + 1 */ + fd_set input; /* Input set for reading */ + ssize_t print_bytes, /* Print bytes read */ + bytes; /* Bytes written */ + char print_buffer[8192], /* Print data buffer */ + *print_ptr; /* Pointer into print data buffer */ + struct timeval timeout; /* Timeout for read... */ + + + fprintf(stderr, "DEBUG: backendDrainOutput(print_fd=%d, device_fd=%d)\n", + print_fd, device_fd); + + /* + * Figure out the maximum file descriptor value to use with select()... + */ + + nfds = (print_fd > device_fd ? print_fd : device_fd) + 1; + + /* + * Now loop until we are out of data from print_fd... + */ + + for (;;) + { + /* + * Use select() to determine whether we have data to copy around... + */ + + FD_ZERO(&input); + FD_SET(print_fd, &input); + + timeout.tv_sec = 0; + timeout.tv_usec = 0; + + if (select(nfds, &input, NULL, NULL, &timeout) < 0) + return (-1); + + if (!FD_ISSET(print_fd, &input)) + return (0); + + if ((print_bytes = read(print_fd, print_buffer, + sizeof(print_buffer))) < 0) + { + /* + * Read error - bail if we don't see EAGAIN or EINTR... + */ + + if (errno != EAGAIN || errno != EINTR) + { + _cupsLangPrintError("ERROR", _("Unable to read print data")); + return (-1); + } + + print_bytes = 0; + } + else if (print_bytes == 0) + { + /* + * End of file, return... + */ + + return (0); + } + + fprintf(stderr, "DEBUG: Read %d bytes of print data...\n", + (int)print_bytes); + + for (print_ptr = print_buffer; print_bytes > 0;) + { + if ((bytes = write(device_fd, print_ptr, print_bytes)) < 0) + { + /* + * Write error - bail if we don't see an error we can retry... + */ + + if (errno != ENOSPC && errno != ENXIO && errno != EAGAIN && + errno != EINTR && errno != ENOTTY) + { + _cupsLangPrintError("ERROR", _("Unable to write print data")); + return (-1); + } + } + else + { + fprintf(stderr, "DEBUG: Wrote %d bytes of print data...\n", (int)bytes); + + print_bytes -= bytes; + print_ptr += bytes; + } + } + } +} + + +/* + * 'backendRunLoop()' - Read and write print and back-channel data. + */ + +ssize_t /* O - Total bytes on success, -1 on error */ +backendRunLoop( + int print_fd, /* I - Print file descriptor */ + int device_fd, /* I - Device file descriptor */ + int snmp_fd, /* I - SNMP socket or -1 if none */ + http_addr_t *addr, /* I - Address of device */ + int use_bc, /* I - Use back-channel? */ + int update_state, /* I - Update printer-state-reasons? */ + _cups_sccb_t side_cb) /* I - Side-channel callback */ +{ + int nfds; /* Maximum file descriptor value + 1 */ + fd_set input, /* Input set for reading */ + output; /* Output set for writing */ + ssize_t print_bytes, /* Print bytes read */ + bc_bytes, /* Backchannel bytes read */ + total_bytes, /* Total bytes written */ + bytes; /* Bytes written */ + int paperout; /* "Paper out" status */ + int offline; /* "Off-line" status */ + char print_buffer[8192], /* Print data buffer */ + *print_ptr, /* Pointer into print data buffer */ + bc_buffer[1024]; /* Back-channel data buffer */ + struct timeval timeout; /* Timeout for select() */ + time_t curtime, /* Current time */ + snmp_update = 0; +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* Actions for POSIX signals */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ + + + fprintf(stderr, + "DEBUG: backendRunLoop(print_fd=%d, device_fd=%d, snmp_fd=%d, " + "addr=%p, use_bc=%d, side_cb=%p)\n", + print_fd, device_fd, snmp_fd, addr, use_bc, side_cb); + + /* + * If we are printing data from a print driver on stdin, ignore SIGTERM + * so that the driver can finish out any page data, e.g. to eject the + * current page. We only do this for stdin printing as otherwise there + * is no way to cancel a raw print job... + */ + + if (!print_fd) + { +#ifdef HAVE_SIGSET /* Use System V signals over POSIX to avoid bugs */ + sigset(SIGTERM, SIG_IGN); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + + sigemptyset(&action.sa_mask); + action.sa_handler = SIG_IGN; + sigaction(SIGTERM, &action, NULL); +#else + signal(SIGTERM, SIG_IGN); +#endif /* HAVE_SIGSET */ + } + else if (print_fd < 0) + { + /* + * Copy print data from stdin, but don't mess with the signal handlers... + */ + + print_fd = 0; + } + + /* + * Figure out the maximum file descriptor value to use with select()... + */ + + nfds = (print_fd > device_fd ? print_fd : device_fd) + 1; + + /* + * Now loop until we are out of data from print_fd... + */ + + for (print_bytes = 0, print_ptr = print_buffer, offline = -1, + paperout = -1, total_bytes = 0;;) + { + /* + * Use select() to determine whether we have data to copy around... + */ + + FD_ZERO(&input); + if (!print_bytes) + FD_SET(print_fd, &input); + if (use_bc) + FD_SET(device_fd, &input); + if (!print_bytes && side_cb) + FD_SET(CUPS_SC_FD, &input); + + FD_ZERO(&output); + if (print_bytes || (!use_bc && !side_cb)) + FD_SET(device_fd, &output); + + if (use_bc || side_cb) + { + timeout.tv_sec = 5; + timeout.tv_usec = 0; + + if (select(nfds, &input, &output, NULL, &timeout) < 0) + { + /* + * Pause printing to clear any pending errors... + */ + + if (errno == ENXIO && offline != 1 && update_state) + { + fputs("STATE: +offline-report\n", stderr); + _cupsLangPrintFilter(stderr, "INFO", + _("Printer is not currently connected.")); + offline = 1; + } + else if (errno == EINTR && total_bytes == 0) + { + fputs("DEBUG: Received an interrupt before any bytes were " + "written, aborting.\n", stderr); + return (0); + } + + sleep(1); + continue; + } + } + + /* + * Check if we have a side-channel request ready... + */ + + if (side_cb && FD_ISSET(CUPS_SC_FD, &input)) + { + /* + * Do the side-channel request, then start back over in the select + * loop since it may have read from print_fd... + */ + + if ((*side_cb)(print_fd, device_fd, snmp_fd, addr, use_bc)) + side_cb = NULL; + continue; + } + + /* + * Check if we have back-channel data ready... + */ + + if (FD_ISSET(device_fd, &input)) + { + if ((bc_bytes = read(device_fd, bc_buffer, sizeof(bc_buffer))) > 0) + { + fprintf(stderr, + "DEBUG: Received " CUPS_LLFMT " bytes of back-channel data\n", + CUPS_LLCAST bc_bytes); + cupsBackChannelWrite(bc_buffer, bc_bytes, 1.0); + } + else if (bc_bytes < 0 && errno != EAGAIN && errno != EINTR) + { + fprintf(stderr, "DEBUG: Error reading back-channel data: %s\n", + strerror(errno)); + use_bc = 0; + } + else if (bc_bytes == 0) + use_bc = 0; + } + + /* + * Check if we have print data ready... + */ + + if (FD_ISSET(print_fd, &input)) + { + if ((print_bytes = read(print_fd, print_buffer, + sizeof(print_buffer))) < 0) + { + /* + * Read error - bail if we don't see EAGAIN or EINTR... + */ + + if (errno != EAGAIN || errno != EINTR) + { + _cupsLangPrintError("ERROR", _("Unable to read print data")); + return (-1); + } + + print_bytes = 0; + } + else if (print_bytes == 0) + { + /* + * End of file, break out of the loop... + */ + + break; + } + + print_ptr = print_buffer; + + fprintf(stderr, "DEBUG: Read %d bytes of print data...\n", + (int)print_bytes); + } + + /* + * Check if the device is ready to receive data and we have data to + * send... + */ + + if (print_bytes && FD_ISSET(device_fd, &output)) + { + if ((bytes = write(device_fd, print_ptr, print_bytes)) < 0) + { + /* + * Write error - bail if we don't see an error we can retry... + */ + + if (errno == ENOSPC) + { + if (paperout != 1 && update_state) + { + fputs("STATE: +media-empty-warning\n", stderr); + fputs("DEBUG: Out of paper\n", stderr); + paperout = 1; + } + } + else if (errno == ENXIO) + { + if (offline != 1 && update_state) + { + fputs("STATE: +offline-report\n", stderr); + _cupsLangPrintFilter(stderr, "INFO", + _("Printer is not currently connected.")); + offline = 1; + } + } + else if (errno != EAGAIN && errno != EINTR && errno != ENOTTY) + { + _cupsLangPrintError("ERROR", _("Unable to write print data")); + return (-1); + } + } + else + { + if (paperout && update_state) + { + fputs("STATE: -media-empty-warning\n", stderr); + paperout = 0; + } + + if (offline && update_state) + { + fputs("STATE: -offline-report\n", stderr); + _cupsLangPrintFilter(stderr, "INFO", _("Printer is now connected.")); + offline = 0; + } + + fprintf(stderr, "DEBUG: Wrote %d bytes of print data...\n", (int)bytes); + + print_bytes -= bytes; + print_ptr += bytes; + total_bytes += bytes; + } + } + + /* + * Do SNMP updates periodically... + */ + + if (snmp_fd >= 0 && time(&curtime) >= snmp_update) + { + if (backendSNMPSupplies(snmp_fd, addr, NULL, NULL)) + snmp_update = INT_MAX; + else + snmp_update = curtime + 5; + } + } + + /* + * Return with success... + */ + + return (total_bytes); +} + + +/* + * 'backendWaitLoop()' - Wait for input from stdin while handling side-channel + * queries. + */ + +int /* O - 1 if data is ready, 0 if not */ +backendWaitLoop( + int snmp_fd, /* I - SNMP socket or -1 if none */ + http_addr_t *addr, /* I - Address of device */ + int use_bc, /* I - Use back-channel? */ + _cups_sccb_t side_cb) /* I - Side-channel callback */ +{ + fd_set input; /* Input set for reading */ + time_t curtime, /* Current time */ + snmp_update = 0; /* Last SNMP status update */ + + + fprintf(stderr, "DEBUG: backendWaitLoop(snmp_fd=%d, addr=%p, side_cb=%p)\n", + snmp_fd, addr, side_cb); + + /* + * Now loop until we receive data from stdin... + */ + + for (;;) + { + /* + * Use select() to determine whether we have data to copy around... + */ + + FD_ZERO(&input); + FD_SET(0, &input); + if (side_cb) + FD_SET(CUPS_SC_FD, &input); + + if (select(CUPS_SC_FD + 1, &input, NULL, NULL, NULL) < 0) + { + /* + * Pause printing to clear any pending errors... + */ + + if (errno == EINTR) + { + fputs("DEBUG: Received an interrupt before any bytes were " + "written, aborting.\n", stderr); + return (0); + } + + sleep(1); + continue; + } + + /* + * Check for input on stdin... + */ + + if (FD_ISSET(0, &input)) + break; + + /* + * Check if we have a side-channel request ready... + */ + + if (side_cb && FD_ISSET(CUPS_SC_FD, &input)) + { + /* + * Do the side-channel request, then start back over in the select + * loop since it may have read from print_fd... + */ + + if ((*side_cb)(0, -1, snmp_fd, addr, use_bc)) + side_cb = NULL; + continue; + } + + /* + * Do SNMP updates periodically... + */ + + if (snmp_fd >= 0 && time(&curtime) >= snmp_update) + { + if (backendSNMPSupplies(snmp_fd, addr, NULL, NULL)) + snmp_update = INT_MAX; + else + snmp_update = curtime + 5; + } + } + + /* + * Return with success... + */ + + return (1); +} + + +/* + * End of "$Id$". + */ diff --git a/backend/snmp-supplies.c b/backend/snmp-supplies.c new file mode 100644 index 0000000000..b823520635 --- /dev/null +++ b/backend/snmp-supplies.c @@ -0,0 +1,986 @@ +/* + * "$Id$" + * + * SNMP supplies functions for CUPS. + * + * Copyright 2008-2011 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * backendSNMPSupplies() - Get the current supplies for a device. + * backend_init_supplies() - Initialize the supplies list. + * backend_walk_cb() - Interpret the supply value responses. + * utf16_to_utf8() - Convert UTF-16 text to UTF-8. + */ + +/* + * Include necessary headers. + */ + +#include "backend-private.h" +#include + + +/* + * Local constants... + */ + +#define CUPS_MAX_SUPPLIES 32 /* Maximum number of supplies for a printer */ +#define CUPS_SUPPLY_TIMEOUT 2.0 /* Timeout for SNMP lookups */ + +#define CUPS_DEVELOPER_LOW 1 +#define CUPS_DEVELOPER_EMPTY 2 +#define CUPS_MARKER_SUPPLY_LOW 4 +#define CUPS_MARKER_SUPPLY_EMPTY 8 +#define CUPS_OPC_NEAR_EOL 16 +#define CUPS_OPC_LIFE_OVER 32 +#define CUPS_TONER_LOW 64 +#define CUPS_TONER_EMPTY 128 + + +/* + * Local structures... + */ + +typedef struct /**** Printer supply data ****/ +{ + char name[CUPS_SNMP_MAX_STRING], /* Name of supply */ + color[8]; /* Color: "#RRGGBB" or "none" */ + int colorant, /* Colorant index */ + type, /* Supply type */ + max_capacity, /* Maximum capacity */ + level; /* Current level value */ +} backend_supplies_t; + +typedef struct /**** Printer state table ****/ +{ + int bit; /* State bit */ + const char *keyword; /* IPP printer-state-reasons keyword */ +} backend_state_t; + + +/* + * Local globals... + */ + +static http_addr_t current_addr; /* Current address */ +static int current_state = -1; + /* Current device state bits */ +static int charset = -1; /* Character set for supply names */ +static int num_supplies = 0; + /* Number of supplies found */ +static backend_supplies_t supplies[CUPS_MAX_SUPPLIES]; + /* Supply information */ +static int supply_state = -1; + /* Supply state info */ + +static const int hrDeviceDescr[] = + { CUPS_OID_hrDeviceDescr, 1, -1 }; + /* Device description OID */ +static const int hrPrinterStatus[] = + { CUPS_OID_hrPrinterStatus, 1, -1 }; + /* Current state OID */ +static const int hrPrinterDetectedErrorState[] = + { CUPS_OID_hrPrinterDetectedErrorState, 1, -1 }; + /* Current printer state bits OID */ +static const int prtGeneralCurrentLocalization[] = + { CUPS_OID_prtGeneralCurrentLocalization, 1, -1 }; +static const int prtLocalizationCharacterSet[] = + { CUPS_OID_prtLocalizationCharacterSet, 1, 1, -1 }, + prtLocalizationCharacterSetOffset = + (sizeof(prtLocalizationCharacterSet) / + sizeof(prtLocalizationCharacterSet[0])); +static const int prtMarkerColorantValue[] = + { CUPS_OID_prtMarkerColorantValue, -1 }, + /* Colorant OID */ + prtMarkerColorantValueOffset = + (sizeof(prtMarkerColorantValue) / + sizeof(prtMarkerColorantValue[0])); + /* Offset to colorant index */ +static const int prtMarkerLifeCount[] = + { CUPS_OID_prtMarkerLifeCount, 1, 1, -1 }; + /* Page counter OID */ +static const int prtMarkerSuppliesEntry[] = + { CUPS_OID_prtMarkerSuppliesEntry, -1 }; + /* Supplies OID */ +static const int prtMarkerSuppliesColorantIndex[] = + { CUPS_OID_prtMarkerSuppliesColorantIndex, -1 }, + /* Colorant index OID */ + prtMarkerSuppliesColorantIndexOffset = + (sizeof(prtMarkerSuppliesColorantIndex) / + sizeof(prtMarkerSuppliesColorantIndex[0])); + /* Offset to supply index */ +static const int prtMarkerSuppliesDescription[] = + { CUPS_OID_prtMarkerSuppliesDescription, -1 }, + /* Description OID */ + prtMarkerSuppliesDescriptionOffset = + (sizeof(prtMarkerSuppliesDescription) / + sizeof(prtMarkerSuppliesDescription[0])); + /* Offset to supply index */ +static const int prtMarkerSuppliesLevel[] = + { CUPS_OID_prtMarkerSuppliesLevel, -1 }, + /* Level OID */ + prtMarkerSuppliesLevelOffset = + (sizeof(prtMarkerSuppliesLevel) / + sizeof(prtMarkerSuppliesLevel[0])); + /* Offset to supply index */ +static const int prtMarkerSuppliesMaxCapacity[] = + { CUPS_OID_prtMarkerSuppliesMaxCapacity, -1 }, + /* Max capacity OID */ + prtMarkerSuppliesMaxCapacityOffset = + (sizeof(prtMarkerSuppliesMaxCapacity) / + sizeof(prtMarkerSuppliesMaxCapacity[0])); + /* Offset to supply index */ +static const int prtMarkerSuppliesType[] = + { CUPS_OID_prtMarkerSuppliesType, -1 }, + /* Type OID */ + prtMarkerSuppliesTypeOffset = + (sizeof(prtMarkerSuppliesType) / + sizeof(prtMarkerSuppliesType[0])); + /* Offset to supply index */ + +static const backend_state_t const printer_states[] = + { + { CUPS_TC_lowPaper, "media-low-report" }, + { CUPS_TC_noPaper | CUPS_TC_inputTrayEmpty, "media-empty-warning" }, + /* { CUPS_TC_lowToner, "toner-low-report" }, */ /* now use prtMarkerSupplies */ + /* { CUPS_TC_noToner, "toner-empty-warning" }, */ /* now use prtMarkerSupplies */ + { CUPS_TC_doorOpen, "door-open-report" }, + { CUPS_TC_jammed, "media-jam-warning" }, + /* { CUPS_TC_offline, "offline-report" }, */ /* unreliable */ + /* { CUPS_TC_serviceRequested | CUPS_TC_overduePreventMaint, "service-needed-warning" }, */ /* unreliable */ + { CUPS_TC_inputTrayMissing, "input-tray-missing-warning" }, + { CUPS_TC_outputTrayMissing, "output-tray-missing-warning" }, + { CUPS_TC_markerSupplyMissing, "marker-supply-missing-warning" }, + { CUPS_TC_outputNearFull, "output-area-almost-full-report" }, + { CUPS_TC_outputFull, "output-area-full-warning" } + }; + +static const backend_state_t const supply_states[] = + { + { CUPS_DEVELOPER_LOW, "developer-low-report" }, + { CUPS_DEVELOPER_EMPTY, "developer-empty-warning" }, + { CUPS_MARKER_SUPPLY_LOW, "marker-supply-low-report" }, + { CUPS_MARKER_SUPPLY_EMPTY, "marker-supply-empty-warning" }, + { CUPS_OPC_NEAR_EOL, "opc-near-eol-report" }, + { CUPS_OPC_LIFE_OVER, "opc-life-over-warning" }, + { CUPS_TONER_LOW, "toner-low-report" }, + { CUPS_TONER_EMPTY, "toner-empty-warning" } + }; + + +/* + * Local functions... + */ + +static void backend_init_supplies(int snmp_fd, http_addr_t *addr); +static void backend_walk_cb(cups_snmp_t *packet, void *data); +static void utf16_to_utf8(cups_utf8_t *dst, const unsigned char *src, + size_t srcsize, size_t dstsize, int le); + + +/* + * 'backendSNMPSupplies()' - Get the current supplies for a device. + */ + +int /* O - 0 on success, -1 on error */ +backendSNMPSupplies( + int snmp_fd, /* I - SNMP socket */ + http_addr_t *addr, /* I - Printer address */ + int *page_count, /* O - Page count */ + int *printer_state) /* O - Printer state */ +{ + if (!httpAddrEqual(addr, ¤t_addr)) + backend_init_supplies(snmp_fd, addr); + else if (num_supplies > 0) + _cupsSNMPWalk(snmp_fd, ¤t_addr, CUPS_SNMP_VERSION_1, + _cupsSNMPDefaultCommunity(), prtMarkerSuppliesLevel, + CUPS_SUPPLY_TIMEOUT, backend_walk_cb, NULL); + + if (page_count) + *page_count = -1; + + if (printer_state) + *printer_state = -1; + + if (num_supplies > 0) + { + int i, /* Looping var */ + percent, /* Percent full */ + new_state, /* New state value */ + change_state, /* State change */ + new_supply_state = 0; /* Supply state */ + char value[CUPS_MAX_SUPPLIES * 4], + /* marker-levels value string */ + *ptr; /* Pointer into value string */ + cups_snmp_t packet; /* SNMP response packet */ + + /* + * Generate the marker-levels value string... + */ + + for (i = 0, ptr = value; i < num_supplies; i ++, ptr += strlen(ptr)) + { + if (supplies[i].max_capacity > 0 && supplies[i].level >= 0) + percent = 100 * supplies[i].level / supplies[i].max_capacity; + else + percent = 50; + + if (percent <= 5) + { + switch (supplies[i].type) + { + case CUPS_TC_toner : + case CUPS_TC_tonerCartridge : + if (percent <= 1) + new_supply_state |= CUPS_TONER_EMPTY; + else + new_supply_state |= CUPS_TONER_LOW; + break; + case CUPS_TC_wasteToner : + case CUPS_TC_wasteInk : + break; + case CUPS_TC_ink : + case CUPS_TC_inkCartridge : + case CUPS_TC_inkRibbon : + case CUPS_TC_solidWax : + case CUPS_TC_ribbonWax : + if (percent <= 1) + new_supply_state |= CUPS_MARKER_SUPPLY_EMPTY; + else + new_supply_state |= CUPS_MARKER_SUPPLY_LOW; + break; + case CUPS_TC_developer : + if (percent <= 1) + new_supply_state |= CUPS_DEVELOPER_EMPTY; + else + new_supply_state |= CUPS_DEVELOPER_LOW; + break; + case CUPS_TC_coronaWire : + case CUPS_TC_fuser : + case CUPS_TC_opc : + case CUPS_TC_transferUnit : + if (percent <= 1) + new_supply_state |= CUPS_OPC_LIFE_OVER; + else + new_supply_state |= CUPS_OPC_NEAR_EOL; + break; + } + } + + if (i) + *ptr++ = ','; + + if (supplies[i].max_capacity > 0 && supplies[i].level >= 0) + sprintf(ptr, "%d", percent); + else + strcpy(ptr, "-1"); + } + + fprintf(stderr, "ATTR: marker-levels=%s\n", value); + + if (supply_state < 0) + change_state = 0xffff; + else + change_state = supply_state ^ new_supply_state; + + fprintf(stderr, "DEBUG: new_supply_state=%x, change_state=%x\n", + new_supply_state, change_state); + + for (i = 0; + i < (int)(sizeof(supply_states) / sizeof(supply_states[0])); + i ++) + if (change_state & supply_states[i].bit) + { + fprintf(stderr, "STATE: %c%s\n", + (new_supply_state & supply_states[i].bit) ? '+' : '-', + supply_states[i].keyword); + } + + supply_state = new_supply_state; + + /* + * Get the current printer status bits... + */ + + if (!_cupsSNMPWrite(snmp_fd, addr, CUPS_SNMP_VERSION_1, + _cupsSNMPDefaultCommunity(), CUPS_ASN1_GET_REQUEST, 1, + hrPrinterDetectedErrorState)) + return (-1); + + if (!_cupsSNMPRead(snmp_fd, &packet, CUPS_SUPPLY_TIMEOUT) || + packet.object_type != CUPS_ASN1_OCTET_STRING) + return (-1); + + if (packet.object_value.string.num_bytes == 2) + new_state = (packet.object_value.string.bytes[0] << 8) | + packet.object_value.string.bytes[1]; + else if (packet.object_value.string.num_bytes == 1) + new_state = (packet.object_value.string.bytes[0] << 8); + else + new_state = 0; + + if (current_state < 0) + change_state = 0xffff; + else + change_state = current_state ^ new_state; + + fprintf(stderr, "DEBUG: new_state=%x, change_state=%x\n", new_state, + change_state); + + for (i = 0; + i < (int)(sizeof(printer_states) / sizeof(printer_states[0])); + i ++) + if (change_state & printer_states[i].bit) + { + fprintf(stderr, "STATE: %c%s\n", + (new_state & printer_states[i].bit) ? '+' : '-', + printer_states[i].keyword); + } + + current_state = new_state; + + /* + * Get the current printer state... + */ + + if (printer_state) + { + if (!_cupsSNMPWrite(snmp_fd, addr, CUPS_SNMP_VERSION_1, + _cupsSNMPDefaultCommunity(), CUPS_ASN1_GET_REQUEST, 1, + hrPrinterStatus)) + return (-1); + + if (!_cupsSNMPRead(snmp_fd, &packet, CUPS_SUPPLY_TIMEOUT) || + packet.object_type != CUPS_ASN1_INTEGER) + return (-1); + + *printer_state = packet.object_value.integer; + } + + /* + * Get the current page count... + */ + + if (page_count) + { + if (!_cupsSNMPWrite(snmp_fd, addr, CUPS_SNMP_VERSION_1, + _cupsSNMPDefaultCommunity(), CUPS_ASN1_GET_REQUEST, 1, + prtMarkerLifeCount)) + return (-1); + + if (!_cupsSNMPRead(snmp_fd, &packet, CUPS_SUPPLY_TIMEOUT) || + packet.object_type != CUPS_ASN1_COUNTER) + return (-1); + + *page_count = packet.object_value.counter; + } + + return (0); + } + else + return (-1); +} + + +/* + * 'backend_init_supplies()' - Initialize the supplies list. + */ + +static void +backend_init_supplies( + int snmp_fd, /* I - SNMP socket */ + http_addr_t *addr) /* I - Printer address */ +{ + int i, /* Looping var */ + type; /* Current marker type */ + cups_file_t *cachefile; /* Cache file */ + const char *cachedir; /* CUPS_CACHEDIR value */ + char addrstr[1024], /* Address string */ + cachefilename[1024], /* Cache filename */ + description[CUPS_SNMP_MAX_STRING], + /* Device description string */ + value[CUPS_MAX_SUPPLIES * (CUPS_SNMP_MAX_STRING * 2 + 3)], + /* Value string */ + *ptr, /* Pointer into value string */ + *name_ptr; /* Pointer into name string */ + cups_snmp_t packet; /* SNMP response packet */ + ppd_file_t *ppd; /* PPD file for this queue */ + ppd_attr_t *ppdattr; /* cupsSNMPSupplies attribute */ + static const char * const types[] = /* Supply types */ + { + "other", + "unknown", + "toner", + "wasteToner", + "ink", + "inkCartridge", + "inkRibbon", + "wasteInk", + "opc", + "developer", + "fuserOil", + "solidWax", + "ribbonWax", + "wasteWax", + "fuser", + "coronaWire", + "fuserOilWick", + "cleanerUnit", + "fuserCleaningPad", + "transferUnit", + "tonerCartridge", + "fuserOiler", + "water", + "wasteWater", + "glueWaterAdditive", + "wastePaper", + "bindingSupply", + "bandingSupply", + "stitchingWire", + "shrinkWrap", + "paperWrap", + "staples", + "inserts", + "covers" + }; + + + /* + * Reset state information... + */ + + current_addr = *addr; + current_state = -1; + num_supplies = -1; + charset = -1; + + memset(supplies, 0, sizeof(supplies)); + + /* + * See if we should be getting supply levels via SNMP... + */ + + if ((ppd = ppdOpenFile(getenv("PPD"))) == NULL || + ((ppdattr = ppdFindAttr(ppd, "cupsSNMPSupplies", NULL)) != NULL && + ppdattr->value && _cups_strcasecmp(ppdattr->value, "true"))) + { + ppdClose(ppd); + return; + } + + ppdClose(ppd); + + /* + * Get the device description... + */ + + if (!_cupsSNMPWrite(snmp_fd, addr, CUPS_SNMP_VERSION_1, + _cupsSNMPDefaultCommunity(), CUPS_ASN1_GET_REQUEST, 1, + hrDeviceDescr)) + return; + + if (!_cupsSNMPRead(snmp_fd, &packet, CUPS_SUPPLY_TIMEOUT) || + packet.object_type != CUPS_ASN1_OCTET_STRING) + { + strlcpy(description, "Unknown", sizeof(description)); + num_supplies = 0; + } + else + strlcpy(description, (char *)packet.object_value.string.bytes, + sizeof(description)); + + fprintf(stderr, "DEBUG2: hrDeviceDesc=\"%s\"\n", description); + + /* + * See if we have already queried this device... + */ + + httpAddrString(addr, addrstr, sizeof(addrstr)); + + if ((cachedir = getenv("CUPS_CACHEDIR")) == NULL) + cachedir = CUPS_CACHEDIR; + + snprintf(cachefilename, sizeof(cachefilename), "%s/%s.snmp", cachedir, + addrstr); + + if ((cachefile = cupsFileOpen(cachefilename, "r")) != NULL) + { + /* + * Yes, read the cache file: + * + * 2 num_supplies charset + * device description + * supply structures... + */ + + if (cupsFileGets(cachefile, value, sizeof(value))) + { + if (sscanf(value, "2 %d%d", &num_supplies, &charset) == 2 && + num_supplies <= CUPS_MAX_SUPPLIES && + cupsFileGets(cachefile, value, sizeof(value))) + { + if (!strcmp(description, value)) + cupsFileRead(cachefile, (char *)supplies, + num_supplies * sizeof(backend_supplies_t)); + else + { + num_supplies = -1; + charset = -1; + } + } + else + { + num_supplies = -1; + charset = -1; + } + } + + cupsFileClose(cachefile); + } + + /* + * If the cache information isn't correct, scan for supplies... + */ + + if (charset < 0) + { + /* + * Get the configured character set... + */ + + int oid[CUPS_SNMP_MAX_OID]; /* OID for character set */ + + + if (!_cupsSNMPWrite(snmp_fd, ¤t_addr, CUPS_SNMP_VERSION_1, + _cupsSNMPDefaultCommunity(), CUPS_ASN1_GET_REQUEST, 1, + prtGeneralCurrentLocalization)) + return; + + if (!_cupsSNMPRead(snmp_fd, &packet, CUPS_SUPPLY_TIMEOUT) || + packet.object_type != CUPS_ASN1_INTEGER) + { + fprintf(stderr, + "DEBUG: prtGeneralCurrentLocalization type is %x, expected %x!\n", + packet.object_type, CUPS_ASN1_INTEGER); + return; + } + + fprintf(stderr, "DEBUG2: prtGeneralCurrentLocalization=%d\n", + packet.object_value.integer); + + _cupsSNMPCopyOID(oid, prtLocalizationCharacterSet, CUPS_SNMP_MAX_OID); + oid[prtLocalizationCharacterSetOffset - 2] = packet.object_value.integer; + + + if (!_cupsSNMPWrite(snmp_fd, ¤t_addr, CUPS_SNMP_VERSION_1, + _cupsSNMPDefaultCommunity(), CUPS_ASN1_GET_REQUEST, 1, + oid)) + return; + + if (!_cupsSNMPRead(snmp_fd, &packet, CUPS_SUPPLY_TIMEOUT) || + packet.object_type != CUPS_ASN1_INTEGER) + { + fprintf(stderr, + "DEBUG: prtLocalizationCharacterSet type is %x, expected %x!\n", + packet.object_type, CUPS_ASN1_INTEGER); + return; + } + + fprintf(stderr, "DEBUG2: prtLocalizationCharacterSet=%d\n", + packet.object_value.integer); + charset = packet.object_value.integer; + } + + if (num_supplies < 0) + { + /* + * Walk the printer configuration information... + */ + + _cupsSNMPWalk(snmp_fd, ¤t_addr, CUPS_SNMP_VERSION_1, + _cupsSNMPDefaultCommunity(), prtMarkerSuppliesEntry, + CUPS_SUPPLY_TIMEOUT, backend_walk_cb, NULL); + } + + /* + * Save the cached information... + */ + + if (num_supplies < 0) + num_supplies = 0; + + if ((cachefile = cupsFileOpen(cachefilename, "w")) != NULL) + { + cupsFilePrintf(cachefile, "2 %d %d\n", num_supplies, charset); + cupsFilePrintf(cachefile, "%s\n", description); + + if (num_supplies > 0) + cupsFileWrite(cachefile, (char *)supplies, + num_supplies * sizeof(backend_supplies_t)); + + cupsFileClose(cachefile); + } + + if (num_supplies <= 0) + return; + + /* + * Get the colors... + */ + + for (i = 0; i < num_supplies; i ++) + strcpy(supplies[i].color, "none"); + + _cupsSNMPWalk(snmp_fd, ¤t_addr, CUPS_SNMP_VERSION_1, + _cupsSNMPDefaultCommunity(), prtMarkerColorantValue, + CUPS_SUPPLY_TIMEOUT, backend_walk_cb, NULL); + + /* + * Output the marker-colors attribute... + */ + + for (i = 0, ptr = value; i < num_supplies; i ++, ptr += strlen(ptr)) + { + if (i) + *ptr++ = ','; + + strcpy(ptr, supplies[i].color); + } + + fprintf(stderr, "ATTR: marker-colors=%s\n", value); + + /* + * Output the marker-names attribute... + */ + + for (i = 0, ptr = value; i < num_supplies; i ++) + { + if (i) + *ptr++ = ','; + + *ptr++ = '\"'; + for (name_ptr = supplies[i].name; *name_ptr;) + { + if (*name_ptr == '\\' || *name_ptr == '\"') + *ptr++ = '\\'; + + *ptr++ = *name_ptr++; + } + *ptr++ = '\"'; + } + + *ptr = '\0'; + + fprintf(stderr, "ATTR: marker-names=%s\n", value); + + /* + * Output the marker-types attribute... + */ + + for (i = 0, ptr = value; i < num_supplies; i ++, ptr += strlen(ptr)) + { + if (i) + *ptr++ = ','; + + type = supplies[i].type; + + if (type < CUPS_TC_other || type > CUPS_TC_covers) + strcpy(ptr, "unknown"); + else + strcpy(ptr, types[type - CUPS_TC_other]); + } + + fprintf(stderr, "ATTR: marker-types=%s\n", value); +} + + +/* + * 'backend_walk_cb()' - Interpret the supply value responses. + */ + +static void +backend_walk_cb(cups_snmp_t *packet, /* I - SNMP packet */ + void *data) /* I - User data (unused) */ +{ + int i, j, k; /* Looping vars */ + static const char * const colors[8][2] = + { /* Standard color names */ + { "black", "#000000" }, + { "blue", "#0000FF" }, + { "cyan", "#00FFFF" }, + { "green", "#00FF00" }, + { "magenta", "#FF00FF" }, + { "red", "#FF0000" }, + { "white", "#FFFFFF" }, + { "yellow", "#FFFF00" } + }; + + + (void)data; + + if (_cupsSNMPIsOIDPrefixed(packet, prtMarkerColorantValue) && + packet->object_type == CUPS_ASN1_OCTET_STRING) + { + /* + * Get colorant... + */ + + i = packet->object_name[prtMarkerColorantValueOffset]; + + fprintf(stderr, "DEBUG2: prtMarkerColorantValue.1.%d = \"%s\"\n", i, + (char *)packet->object_value.string.bytes); + + for (j = 0; j < num_supplies; j ++) + if (supplies[j].colorant == i) + { + for (k = 0; k < (int)(sizeof(colors) / sizeof(colors[0])); k ++) + if (!strcmp(colors[k][0], (char *)packet->object_value.string.bytes)) + { + strcpy(supplies[j].color, colors[k][1]); + break; + } + } + } + else if (_cupsSNMPIsOIDPrefixed(packet, prtMarkerSuppliesColorantIndex)) + { + /* + * Get colorant index... + */ + + i = packet->object_name[prtMarkerSuppliesColorantIndexOffset]; + if (i < 1 || i > CUPS_MAX_SUPPLIES || + packet->object_type != CUPS_ASN1_INTEGER) + return; + + fprintf(stderr, "DEBUG2: prtMarkerSuppliesColorantIndex.1.%d = %d\n", i, + packet->object_value.integer); + + if (i > num_supplies) + num_supplies = i; + + supplies[i - 1].colorant = packet->object_value.integer; + } + else if (_cupsSNMPIsOIDPrefixed(packet, prtMarkerSuppliesDescription)) + { + /* + * Get supply name/description... + */ + + i = packet->object_name[prtMarkerSuppliesDescriptionOffset]; + if (i < 1 || i > CUPS_MAX_SUPPLIES || + packet->object_type != CUPS_ASN1_OCTET_STRING) + return; + + if (i > num_supplies) + num_supplies = i; + + switch (charset) + { + case CUPS_TC_csASCII : + case CUPS_TC_csUTF8 : + case CUPS_TC_csUnicodeASCII : + strlcpy(supplies[i - 1].name, + (char *)packet->object_value.string.bytes, + sizeof(supplies[0].name)); + break; + + case CUPS_TC_csISOLatin1 : + case CUPS_TC_csUnicodeLatin1 : + cupsCharsetToUTF8((cups_utf8_t *)supplies[i - 1].name, + (char *)packet->object_value.string.bytes, + sizeof(supplies[0].name), CUPS_ISO8859_1); + break; + + case CUPS_TC_csShiftJIS : + case CUPS_TC_csWindows31J : /* Close enough for our purposes */ + cupsCharsetToUTF8((cups_utf8_t *)supplies[i - 1].name, + (char *)packet->object_value.string.bytes, + sizeof(supplies[0].name), CUPS_JIS_X0213); + break; + + case CUPS_TC_csUCS4 : + case CUPS_TC_csUTF32 : + case CUPS_TC_csUTF32BE : + case CUPS_TC_csUTF32LE : + cupsUTF32ToUTF8((cups_utf8_t *)supplies[i - 1].name, + (cups_utf32_t *)packet->object_value.string.bytes, + sizeof(supplies[0].name)); + break; + + case CUPS_TC_csUnicode : + case CUPS_TC_csUTF16BE : + case CUPS_TC_csUTF16LE : + utf16_to_utf8((cups_utf8_t *)supplies[i - 1].name, + packet->object_value.string.bytes, + packet->object_value.string.num_bytes, + sizeof(supplies[0].name), charset == CUPS_TC_csUTF16LE); + break; + + default : + /* + * If we get here, the printer is using an unknown character set and + * we just want to copy characters that look like ASCII... + */ + + { + char *src, *dst; /* Pointers into strings */ + + + /* + * Loop safe because both the object_value and supplies char arrays + * are CUPS_SNMP_MAX_STRING elements long. + */ + + for (src = (char *)packet->object_value.string.bytes, + dst = supplies[i - 1].name; + *src; + src ++) + { + if ((*src & 0x80) || *src < ' ' || *src == 0x7f) + *dst++ = '?'; + else + *dst++ = *src; + } + + *dst = '\0'; + } + break; + } + + fprintf(stderr, "DEBUG2: prtMarkerSuppliesDescription.1.%d = \"%s\"\n", i, + supplies[i - 1].name); + + } + else if (_cupsSNMPIsOIDPrefixed(packet, prtMarkerSuppliesLevel)) + { + /* + * Get level... + */ + + i = packet->object_name[prtMarkerSuppliesLevelOffset]; + if (i < 1 || i > CUPS_MAX_SUPPLIES || + packet->object_type != CUPS_ASN1_INTEGER) + return; + + fprintf(stderr, "DEBUG2: prtMarkerSuppliesLevel.1.%d = %d\n", i, + packet->object_value.integer); + + if (i > num_supplies) + num_supplies = i; + + supplies[i - 1].level = packet->object_value.integer; + } + else if (_cupsSNMPIsOIDPrefixed(packet, prtMarkerSuppliesMaxCapacity)) + { + /* + * Get max capacity... + */ + + i = packet->object_name[prtMarkerSuppliesMaxCapacityOffset]; + if (i < 1 || i > CUPS_MAX_SUPPLIES || + packet->object_type != CUPS_ASN1_INTEGER) + return; + + fprintf(stderr, "DEBUG2: prtMarkerSuppliesMaxCapacity.1.%d = %d\n", i, + packet->object_value.integer); + + if (i > num_supplies) + num_supplies = i; + + supplies[i - 1].max_capacity = packet->object_value.integer; + } + else if (_cupsSNMPIsOIDPrefixed(packet, prtMarkerSuppliesType)) + { + /* + * Get marker type... + */ + + i = packet->object_name[prtMarkerSuppliesTypeOffset]; + if (i < 1 || i > CUPS_MAX_SUPPLIES || + packet->object_type != CUPS_ASN1_INTEGER) + return; + + fprintf(stderr, "DEBUG2: prtMarkerSuppliesType.1.%d = %d\n", i, + packet->object_value.integer); + + if (i > num_supplies) + num_supplies = i; + + supplies[i - 1].type = packet->object_value.integer; + } +} + + +/* + * 'utf16_to_utf8()' - Convert UTF-16 text to UTF-8. + */ + +static void +utf16_to_utf8( + cups_utf8_t *dst, /* I - Destination buffer */ + const unsigned char *src, /* I - Source string */ + size_t srcsize, /* I - Size of source string */ + size_t dstsize, /* I - Size of destination buffer */ + int le) /* I - Source is little-endian? */ +{ + cups_utf32_t ch, /* Current character */ + temp[CUPS_SNMP_MAX_STRING], + /* UTF-32 string */ + *ptr; /* Pointer into UTF-32 string */ + + + for (ptr = temp; srcsize >= 2;) + { + if (le) + ch = src[0] | (src[1] << 8); + else + ch = (src[0] << 8) | src[1]; + + src += 2; + srcsize -= 2; + + if (ch >= 0xd800 && ch <= 0xdbff && srcsize >= 2) + { + /* + * Multi-word UTF-16 char... + */ + + int lch; /* Lower word */ + + + if (le) + lch = src[0] | (src[1] << 8); + else + lch = (src[0] << 8) | src[1]; + + if (lch >= 0xdc00 && lch <= 0xdfff) + { + src += 2; + srcsize -= 2; + + ch = (((ch & 0x3ff) << 10) | (lch & 0x3ff)) + 0x10000; + } + } + + if (ptr < (temp + CUPS_SNMP_MAX_STRING - 1)) + *ptr++ = ch; + } + + *ptr = '\0'; + + cupsUTF32ToUTF8(dst, temp, dstsize); +} + + +/* + * End of "$Id$". + */ diff --git a/backend/snmp.c b/backend/snmp.c new file mode 100644 index 0000000000..8819fcb3cc --- /dev/null +++ b/backend/snmp.c @@ -0,0 +1,1387 @@ +/* + * "$Id$" + * + * SNMP discovery backend for CUPS. + * + * Copyright 2007-2012 by Apple Inc. + * Copyright 2006-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Discover printers via SNMP. + * add_array() - Add a string to an array. + * add_cache() - Add a cached device... + * add_device_uri() - Add a device URI to the cache. + * alarm_handler() - Handle alarm signals... + * compare_cache() - Compare two cache entries. + * debug_printf() - Display some debugging information. + * fix_make_model() - Fix common problems in the make-and-model + * string. + * free_array() - Free an array of strings. + * free_cache() - Free the array of cached devices. + * get_interface_addresses() - Get the broadcast address(es) associated with + * an interface. + * list_device() - List a device we found... + * password_cb() - Handle authentication requests. + * probe_device() - Probe a device to discover whether it is a + * printer. + * read_snmp_conf() - Read the snmp.conf file. + * read_snmp_response() - Read and parse a SNMP response... + * run_time() - Return the total running time... + * scan_devices() - Scan for devices using SNMP. + * try_connect() - Try connecting on a port... + * update_cache() - Update a cached device... + */ + +/* + * Include necessary headers. + */ + +#include "backend-private.h" +#include +#include +#include +#include + + +/* + * This backend implements SNMP printer discovery. It uses a broadcast- + * based approach to get SNMP response packets from potential printers, + * requesting OIDs from the Host and Port Monitor MIBs, does a URI + * lookup based on the device description string, and finally a probe of + * port 9100 (AppSocket) and 515 (LPD). + * + * The current focus is on printers with internal network cards, although + * the code also works with many external print servers as well. + * + * The backend reads the snmp.conf file from the CUPS_SERVERROOT directory + * which can contain comments, blank lines, or any number of the following + * directives: + * + * Address ip-address + * Address @LOCAL + * Address @IF(name) + * Community name + * DebugLevel N + * DeviceURI "regex pattern" uri + * HostNameLookups on + * HostNameLookups off + * MaxRunTime N + * + * The default is to use: + * + * Address @LOCAL + * Community public + * DebugLevel 0 + * HostNameLookups off + * MaxRunTime 120 + * + * This backend is known to work with the following network printers and + * print servers: + * + * Axis OfficeBasic, 5400, 5600 + * Brother + * EPSON + * Genicom + * HP JetDirect + * Lexmark + * Sharp + * Tektronix + * Xerox + * + * It does not currently work with: + * + * DLink + * Linksys + * Netgear + * Okidata + * + * (for all of these, they do not support the Host MIB) + */ + +/* + * Types... + */ + +enum /**** Request IDs for each field ****/ +{ + DEVICE_TYPE = 1, + DEVICE_DESCRIPTION, + DEVICE_LOCATION, + DEVICE_ID, + DEVICE_URI, + DEVICE_PRODUCT +}; + +typedef struct device_uri_s /**** DeviceURI values ****/ +{ + regex_t re; /* Regular expression to match */ + cups_array_t *uris; /* URIs */ +} device_uri_t; + +typedef struct snmp_cache_s /**** SNMP scan cache ****/ +{ + http_addr_t address; /* Address of device */ + char *addrname, /* Name of device */ + *uri, /* device-uri */ + *id, /* device-id */ + *info, /* device-info */ + *location, /* device-location */ + *make_and_model; /* device-make-and-model */ + int sent; /* Has this device been listed? */ +} snmp_cache_t; + + +/* + * Local functions... + */ + +static char *add_array(cups_array_t *a, const char *s); +static void add_cache(http_addr_t *addr, const char *addrname, + const char *uri, const char *id, + const char *make_and_model); +static device_uri_t *add_device_uri(char *value); +static void alarm_handler(int sig); +static int compare_cache(snmp_cache_t *a, snmp_cache_t *b); +static void debug_printf(const char *format, ...); +static void fix_make_model(char *make_model, + const char *old_make_model, + int make_model_size); +static void free_array(cups_array_t *a); +static void free_cache(void); +static http_addrlist_t *get_interface_addresses(const char *ifname); +static void list_device(snmp_cache_t *cache); +static const char *password_cb(const char *prompt); +static void probe_device(snmp_cache_t *device); +static void read_snmp_conf(const char *address); +static void read_snmp_response(int fd); +static double run_time(void); +static void scan_devices(int ipv4, int ipv6); +static int try_connect(http_addr_t *addr, const char *addrname, + int port); +static void update_cache(snmp_cache_t *device, const char *uri, + const char *id, const char *make_model); + + +/* + * Local globals... + */ + +static cups_array_t *Addresses = NULL; +static cups_array_t *Communities = NULL; +static cups_array_t *Devices = NULL; +static int DebugLevel = 0; +static const int DescriptionOID[] = { CUPS_OID_hrDeviceDescr, 1, -1 }; +static const int LocationOID[] = { CUPS_OID_sysLocation, 0, -1 }; +static const int DeviceTypeOID[] = { CUPS_OID_hrDeviceType, 1, -1 }; +static const int DeviceIdOID[] = { CUPS_OID_ppmPrinterIEEE1284DeviceId, 1, -1 }; +static const int UriOID[] = { CUPS_OID_ppmPortServiceNameOrURI, 1, 1, -1 }; +static const int LexmarkProductOID[] = { 1,3,6,1,4,1,641,2,1,2,1,2,1,-1 }; +static const int LexmarkProductOID2[] = { 1,3,6,1,4,1,674,10898,100,2,1,2,1,2,1,-1 }; +static const int LexmarkDeviceIdOID[] = { 1,3,6,1,4,1,641,2,1,2,1,3,1,-1 }; +static const int XeroxProductOID[] = { 1,3,6,1,4,1,128,2,1,3,1,2,0,-1 }; +static cups_array_t *DeviceURIs = NULL; +static int HostNameLookups = 0; +static int MaxRunTime = 120; +static struct timeval StartTime; + + +/* + * 'main()' - Discover printers via SNMP. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments (6 or 7) */ + char *argv[]) /* I - Command-line arguments */ +{ + int ipv4, /* SNMP IPv4 socket */ + ipv6; /* SNMP IPv6 socket */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* Actions for POSIX signals */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ + + + /* + * Check command-line options... + */ + + if (argc > 2) + { + _cupsLangPuts(stderr, _("Usage: snmp [host-or-ip-address]")); + return (1); + } + + /* + * Set the password callback for IPP operations... + */ + + cupsSetPasswordCB(password_cb); + + /* + * Catch SIGALRM signals... + */ + +#ifdef HAVE_SIGSET + sigset(SIGALRM, alarm_handler); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + + sigemptyset(&action.sa_mask); + sigaddset(&action.sa_mask, SIGALRM); + action.sa_handler = alarm_handler; + sigaction(SIGALRM, &action, NULL); +#else + signal(SIGALRM, alarm_handler); +#endif /* HAVE_SIGSET */ + + /* + * Open the SNMP socket... + */ + + if ((ipv4 = _cupsSNMPOpen(AF_INET)) < 0) + return (1); + +#ifdef AF_INET6 + if ((ipv6 = _cupsSNMPOpen(AF_INET6)) < 0) + return (1); +#else + ipv6 = -1; +#endif /* AF_INET6 */ + + /* + * Read the configuration file and any cache data... + */ + + read_snmp_conf(argv[1]); + + _cupsSNMPSetDebug(DebugLevel); + + Devices = cupsArrayNew((cups_array_func_t)compare_cache, NULL); + + /* + * Scan for devices... + */ + + scan_devices(ipv4, ipv6); + + /* + * Close, free, and return with no errors... + */ + + _cupsSNMPClose(ipv4); + if (ipv6 >= 0) + _cupsSNMPClose(ipv6); + + free_array(Addresses); + free_array(Communities); + free_cache(); + + return (0); +} + + +/* + * 'add_array()' - Add a string to an array. + */ + +static char * /* O - New string */ +add_array(cups_array_t *a, /* I - Array */ + const char *s) /* I - String to add */ +{ + char *dups; /* New string */ + + + dups = strdup(s); + + cupsArrayAdd(a, dups); + + return (dups); +} + + +/* + * 'add_cache()' - Add a cached device... + */ + +static void +add_cache(http_addr_t *addr, /* I - Device IP address */ + const char *addrname, /* I - IP address or name string */ + const char *uri, /* I - Device URI */ + const char *id, /* I - 1284 device ID */ + const char *make_and_model) /* I - Make and model */ +{ + snmp_cache_t *temp; /* New device entry */ + + + debug_printf("DEBUG: add_cache(addr=%p, addrname=\"%s\", uri=\"%s\", " + "id=\"%s\", make_and_model=\"%s\")\n", + addr, addrname, uri ? uri : "(null)", id ? id : "(null)", + make_and_model ? make_and_model : "(null)"); + + temp = calloc(1, sizeof(snmp_cache_t)); + memcpy(&(temp->address), addr, sizeof(temp->address)); + + temp->addrname = strdup(addrname); + + if (uri) + temp->uri = strdup(uri); + + if (id) + temp->id = strdup(id); + + if (make_and_model) + temp->make_and_model = strdup(make_and_model); + + cupsArrayAdd(Devices, temp); + + if (uri) + list_device(temp); +} + + +/* + * 'add_device_uri()' - Add a device URI to the cache. + * + * The value string is modified (chopped up) as needed. + */ + +static device_uri_t * /* O - Device URI */ +add_device_uri(char *value) /* I - Value from snmp.conf */ +{ + device_uri_t *device_uri; /* Device URI */ + char *start; /* Start of value */ + + + /* + * Allocate memory as needed... + */ + + if (!DeviceURIs) + DeviceURIs = cupsArrayNew(NULL, NULL); + + if (!DeviceURIs) + return (NULL); + + if ((device_uri = calloc(1, sizeof(device_uri_t))) == NULL) + return (NULL); + + if ((device_uri->uris = cupsArrayNew(NULL, NULL)) == NULL) + { + free(device_uri); + return (NULL); + } + + /* + * Scan the value string for the regular expression and URI(s)... + */ + + value ++; /* Skip leading " */ + + for (start = value; *value && *value != '\"'; value ++) + if (*value == '\\' && value[1]) + _cups_strcpy(value, value + 1); + + if (!*value) + { + fputs("ERROR: Missing end quote for DeviceURI!\n", stderr); + + cupsArrayDelete(device_uri->uris); + free(device_uri); + + return (NULL); + } + + *value++ = '\0'; + + if (regcomp(&(device_uri->re), start, REG_EXTENDED | REG_ICASE)) + { + fputs("ERROR: Bad regular expression for DeviceURI!\n", stderr); + + cupsArrayDelete(device_uri->uris); + free(device_uri); + + return (NULL); + } + + while (*value) + { + while (isspace(*value & 255)) + value ++; + + if (!*value) + break; + + for (start = value; *value && !isspace(*value & 255); value ++); + + if (*value) + *value++ = '\0'; + + cupsArrayAdd(device_uri->uris, strdup(start)); + } + + /* + * Add the device URI to the list and return it... + */ + + cupsArrayAdd(DeviceURIs, device_uri); + + return (device_uri); +} + + +/* + * 'alarm_handler()' - Handle alarm signals... + */ + +static void +alarm_handler(int sig) /* I - Signal number */ +{ + /* + * Do nothing... + */ + + (void)sig; + +#if !defined(HAVE_SIGSET) && !defined(HAVE_SIGACTION) + signal(SIGALRM, alarm_handler); +#endif /* !HAVE_SIGSET && !HAVE_SIGACTION */ + + if (DebugLevel) + write(2, "DEBUG: ALARM!\n", 14); +} + + +/* + * 'compare_cache()' - Compare two cache entries. + */ + +static int /* O - Result of comparison */ +compare_cache(snmp_cache_t *a, /* I - First cache entry */ + snmp_cache_t *b) /* I - Second cache entry */ +{ + return (_cups_strcasecmp(a->addrname, b->addrname)); +} + + +/* + * 'debug_printf()' - Display some debugging information. + */ + +static void +debug_printf(const char *format, /* I - Printf-style format string */ + ...) /* I - Additional arguments as needed */ +{ + va_list ap; /* Pointer to arguments */ + + + if (!DebugLevel) + return; + + va_start(ap, format); + vfprintf(stderr, format, ap); + va_end(ap); +} + + +/* + * 'fix_make_model()' - Fix common problems in the make-and-model string. + */ + +static void +fix_make_model( + char *make_model, /* I - New make-and-model string */ + const char *old_make_model, /* I - Old make-and-model string */ + int make_model_size) /* I - Size of new string buffer */ +{ + char *mmptr; /* Pointer into make-and-model string */ + + + /* + * Fix some common problems with the make-and-model string so + * that printer driver detection works better... + */ + + if (!_cups_strncasecmp(old_make_model, "Hewlett-Packard", 15)) + { + /* + * Strip leading Hewlett-Packard and hp prefixes and replace + * with a single HP manufacturer prefix... + */ + + mmptr = (char *)old_make_model + 15; + + while (isspace(*mmptr & 255)) + mmptr ++; + + if (!_cups_strncasecmp(mmptr, "hp", 2)) + { + mmptr += 2; + + while (isspace(*mmptr & 255)) + mmptr ++; + } + + make_model[0] = 'H'; + make_model[1] = 'P'; + make_model[2] = ' '; + strlcpy(make_model + 3, mmptr, make_model_size - 3); + } + else if (!_cups_strncasecmp(old_make_model, "deskjet", 7)) + snprintf(make_model, make_model_size, "HP DeskJet%s", old_make_model + 7); + else if (!_cups_strncasecmp(old_make_model, "officejet", 9)) + snprintf(make_model, make_model_size, "HP OfficeJet%s", old_make_model + 9); + else if (!_cups_strncasecmp(old_make_model, "stylus_pro_", 11)) + snprintf(make_model, make_model_size, "EPSON Stylus Pro %s", + old_make_model + 11); + else + strlcpy(make_model, old_make_model, make_model_size); + + if ((mmptr = strstr(make_model, ", Inc.,")) != NULL) + { + /* + * Strip inc. from name, e.g. "Tektronix, Inc., Phaser 560" + * becomes "Tektronix Phaser 560"... + */ + + _cups_strcpy(mmptr, mmptr + 7); + } + + if ((mmptr = strstr(make_model, " Network")) != NULL) + { + /* + * Drop unnecessary informational text, e.g. "Xerox DocuPrint N2025 + * Network LaserJet - 2.12" becomes "Xerox DocuPrint N2025"... + */ + + *mmptr = '\0'; + } + + if ((mmptr = strchr(make_model, ',')) != NULL) + { + /* + * Drop anything after a trailing comma... + */ + + *mmptr = '\0'; + } +} + + +/* + * 'free_array()' - Free an array of strings. + */ + +static void +free_array(cups_array_t *a) /* I - Array */ +{ + char *s; /* Current string */ + + + for (s = (char *)cupsArrayFirst(a); s; s = (char *)cupsArrayNext(a)) + free(s); + + cupsArrayDelete(a); +} + + +/* + * 'free_cache()' - Free the array of cached devices. + */ + +static void +free_cache(void) +{ + snmp_cache_t *cache; /* Cached device */ + + + for (cache = (snmp_cache_t *)cupsArrayFirst(Devices); + cache; + cache = (snmp_cache_t *)cupsArrayNext(Devices)) + { + free(cache->addrname); + + if (cache->uri) + free(cache->uri); + + if (cache->id) + free(cache->id); + + if (cache->make_and_model) + free(cache->make_and_model); + + free(cache); + } + + cupsArrayDelete(Devices); + Devices = NULL; +} + + +/* + * 'get_interface_addresses()' - Get the broadcast address(es) associated + * with an interface. + */ + +static http_addrlist_t * /* O - List of addresses */ +get_interface_addresses( + const char *ifname) /* I - Interface name */ +{ + struct ifaddrs *addrs, /* Interface address list */ + *addr; /* Current interface address */ + http_addrlist_t *first, /* First address in list */ + *last, /* Last address in list */ + *current; /* Current address */ + + + if (getifaddrs(&addrs) < 0) + return (NULL); + + for (addr = addrs, first = NULL, last = NULL; addr; addr = addr->ifa_next) + if ((addr->ifa_flags & IFF_BROADCAST) && addr->ifa_broadaddr && + addr->ifa_broadaddr->sa_family == AF_INET && + (!ifname || !strcmp(ifname, addr->ifa_name))) + { + current = calloc(1, sizeof(http_addrlist_t)); + + memcpy(&(current->addr), addr->ifa_broadaddr, + sizeof(struct sockaddr_in)); + + if (!last) + first = current; + else + last->next = current; + + last = current; + } + + freeifaddrs(addrs); + + return (first); +} + + +/* + * 'list_device()' - List a device we found... + */ + +static void +list_device(snmp_cache_t *cache) /* I - Cached device */ +{ + if (cache->uri) + cupsBackendReport("network", cache->uri, cache->make_and_model, + cache->info, cache->id, cache->location); +} + + +/* + * 'password_cb()' - Handle authentication requests. + * + * All we do right now is return NULL, indicating that no authentication + * is possible. + */ + +static const char * /* O - Password (NULL) */ +password_cb(const char *prompt) /* I - Prompt message */ +{ + (void)prompt; /* Anti-compiler-warning-code */ + + return (NULL); +} + + +/* + * 'probe_device()' - Probe a device to discover whether it is a printer. + * + * TODO: Try using the Port Monitor MIB to discover the correct protocol + * to use - first need a commercially-available printer that supports + * it, though... + */ + +static void +probe_device(snmp_cache_t *device) /* I - Device */ +{ + char uri[1024], /* Full device URI */ + *uriptr, /* Pointer into URI */ + *format; /* Format string for device */ + device_uri_t *device_uri; /* Current DeviceURI match */ + + + debug_printf("DEBUG: %.3f Probing %s...\n", run_time(), device->addrname); + +#ifdef __APPLE__ + /* + * If the printer supports Bonjour/mDNS, don't report it from the SNMP backend. + */ + + if (!try_connect(&(device->address), device->addrname, 5353)) + { + debug_printf("DEBUG: %s supports mDNS, not reporting!\n", device->addrname); + return; + } +#endif /* __APPLE__ */ + + /* + * Lookup the device in the match table... + */ + + for (device_uri = (device_uri_t *)cupsArrayFirst(DeviceURIs); + device_uri; + device_uri = (device_uri_t *)cupsArrayNext(DeviceURIs)) + if (device->make_and_model && + !regexec(&(device_uri->re), device->make_and_model, 0, NULL, 0)) + { + /* + * Found a match, add the URIs... + */ + + for (format = (char *)cupsArrayFirst(device_uri->uris); + format; + format = (char *)cupsArrayNext(device_uri->uris)) + { + for (uriptr = uri; *format && uriptr < (uri + sizeof(uri) - 1);) + if (*format == '%' && format[1] == 's') + { + /* + * Insert hostname/address... + */ + + strlcpy(uriptr, device->addrname, sizeof(uri) - (uriptr - uri)); + uriptr += strlen(uriptr); + format += 2; + } + else + *uriptr++ = *format++; + + *uriptr = '\0'; + + update_cache(device, uri, NULL, NULL); + } + + return; + } + + /* + * Then try the standard ports... + */ + + if (!try_connect(&(device->address), device->addrname, 9100)) + { + debug_printf("DEBUG: %s supports AppSocket!\n", device->addrname); + + snprintf(uri, sizeof(uri), "socket://%s", device->addrname); + update_cache(device, uri, NULL, NULL); + } + else if (!try_connect(&(device->address), device->addrname, 515)) + { + debug_printf("DEBUG: %s supports LPD!\n", device->addrname); + + snprintf(uri, sizeof(uri), "lpd://%s/", device->addrname); + update_cache(device, uri, NULL, NULL); + } +} + + +/* + * 'read_snmp_conf()' - Read the snmp.conf file. + */ + +static void +read_snmp_conf(const char *address) /* I - Single address to probe */ +{ + cups_file_t *fp; /* File pointer */ + char filename[1024], /* Filename */ + line[1024], /* Line from file */ + *value; /* Value on line */ + int linenum; /* Line number */ + const char *cups_serverroot; /* CUPS_SERVERROOT env var */ + const char *debug; /* CUPS_DEBUG_LEVEL env var */ + const char *runtime; /* CUPS_MAX_RUN_TIME env var */ + + + /* + * Initialize the global address and community lists... + */ + + Addresses = cupsArrayNew(NULL, NULL); + Communities = cupsArrayNew(NULL, NULL); + + if (address) + add_array(Addresses, address); + + if ((debug = getenv("CUPS_DEBUG_LEVEL")) != NULL) + DebugLevel = atoi(debug); + + if ((runtime = getenv("CUPS_MAX_RUN_TIME")) != NULL) + MaxRunTime = atoi(runtime); + + /* + * Find the snmp.conf file... + */ + + if ((cups_serverroot = getenv("CUPS_SERVERROOT")) == NULL) + cups_serverroot = CUPS_SERVERROOT; + + snprintf(filename, sizeof(filename), "%s/snmp.conf", cups_serverroot); + + if ((fp = cupsFileOpen(filename, "r")) != NULL) + { + /* + * Read the snmp.conf file... + */ + + linenum = 0; + + while (cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + { + if (!value) + fprintf(stderr, "ERROR: Missing value on line %d of %s!\n", linenum, + filename); + else if (!_cups_strcasecmp(line, "Address")) + { + if (!address) + add_array(Addresses, value); + } + else if (!_cups_strcasecmp(line, "Community")) + add_array(Communities, value); + else if (!_cups_strcasecmp(line, "DebugLevel")) + DebugLevel = atoi(value); + else if (!_cups_strcasecmp(line, "DeviceURI")) + { + if (*value != '\"') + fprintf(stderr, + "ERROR: Missing double quote for regular expression on " + "line %d of %s!\n", linenum, filename); + else + add_device_uri(value); + } + else if (!_cups_strcasecmp(line, "HostNameLookups")) + HostNameLookups = !_cups_strcasecmp(value, "on") || + !_cups_strcasecmp(value, "yes") || + !_cups_strcasecmp(value, "true") || + !_cups_strcasecmp(value, "double"); + else if (!_cups_strcasecmp(line, "MaxRunTime")) + MaxRunTime = atoi(value); + else + fprintf(stderr, "ERROR: Unknown directive %s on line %d of %s!\n", + line, linenum, filename); + } + + cupsFileClose(fp); + } + + /* + * Use defaults if parameters are undefined... + */ + + if (cupsArrayCount(Addresses) == 0) + { + /* + * If we have no addresses, exit immediately... + */ + + fprintf(stderr, + "DEBUG: No address specified and no Address line in %s...\n", + filename); + exit(0); + } + + if (cupsArrayCount(Communities) == 0) + { + fputs("INFO: Using default SNMP Community public\n", stderr); + add_array(Communities, "public"); + } +} + + +/* + * 'read_snmp_response()' - Read and parse a SNMP response... + */ + +static void +read_snmp_response(int fd) /* I - SNMP socket file descriptor */ +{ + char addrname[256]; /* Source address name */ + cups_snmp_t packet; /* Decoded packet */ + snmp_cache_t key, /* Search key */ + *device; /* Matching device */ + + + /* + * Read the response data... + */ + + if (!_cupsSNMPRead(fd, &packet, -1.0)) + { + fprintf(stderr, "ERROR: Unable to read data from socket: %s\n", + strerror(errno)); + return; + } + + if (HostNameLookups) + httpAddrLookup(&(packet.address), addrname, sizeof(addrname)); + else + httpAddrString(&(packet.address), addrname, sizeof(addrname)); + + debug_printf("DEBUG: %.3f Received data from %s...\n", run_time(), addrname); + + /* + * Look for the response status code in the SNMP message header... + */ + + if (packet.error) + { + fprintf(stderr, "ERROR: Bad SNMP packet from %s: %s\n", addrname, + packet.error); + + return; + } + + debug_printf("DEBUG: community=\"%s\"\n", packet.community); + debug_printf("DEBUG: request-id=%d\n", packet.request_id); + debug_printf("DEBUG: error-status=%d\n", packet.error_status); + + if (packet.error_status && packet.request_id != DEVICE_TYPE) + return; + + /* + * Find a matching device in the cache... + */ + + key.addrname = addrname; + device = (snmp_cache_t *)cupsArrayFind(Devices, &key); + + /* + * Process the message... + */ + + switch (packet.request_id) + { + case DEVICE_TYPE : + /* + * Got the device type response... + */ + + if (device) + { + debug_printf("DEBUG: Discarding duplicate device type for \"%s\"...\n", + addrname); + return; + } + + /* + * Add the device and request the device data... + */ + + add_cache(&(packet.address), addrname, NULL, NULL, NULL); + + _cupsSNMPWrite(fd, &(packet.address), CUPS_SNMP_VERSION_1, + packet.community, CUPS_ASN1_GET_REQUEST, + DEVICE_DESCRIPTION, DescriptionOID); + _cupsSNMPWrite(fd, &(packet.address), CUPS_SNMP_VERSION_1, + packet.community, CUPS_ASN1_GET_REQUEST, + DEVICE_ID, DeviceIdOID); + _cupsSNMPWrite(fd, &(packet.address), CUPS_SNMP_VERSION_1, + packet.community, CUPS_ASN1_GET_REQUEST, + DEVICE_URI, UriOID); + _cupsSNMPWrite(fd, &(packet.address), CUPS_SNMP_VERSION_1, + packet.community, CUPS_ASN1_GET_REQUEST, + DEVICE_LOCATION, LocationOID); + _cupsSNMPWrite(fd, &(packet.address), CUPS_SNMP_VERSION_1, + packet.community, CUPS_ASN1_GET_REQUEST, + DEVICE_PRODUCT, LexmarkProductOID); + _cupsSNMPWrite(fd, &(packet.address), CUPS_SNMP_VERSION_1, + packet.community, CUPS_ASN1_GET_REQUEST, + DEVICE_PRODUCT, LexmarkProductOID2); + _cupsSNMPWrite(fd, &(packet.address), CUPS_SNMP_VERSION_1, + packet.community, CUPS_ASN1_GET_REQUEST, + DEVICE_ID, LexmarkDeviceIdOID); + _cupsSNMPWrite(fd, &(packet.address), CUPS_SNMP_VERSION_1, + packet.community, CUPS_ASN1_GET_REQUEST, + DEVICE_PRODUCT, XeroxProductOID); + break; + + case DEVICE_DESCRIPTION : + if (device && packet.object_type == CUPS_ASN1_OCTET_STRING) + { + /* + * Update an existing cache entry... + */ + + char make_model[256]; /* Make and model */ + + + if (strchr((char *)packet.object_value.string.bytes, ':') && + strchr((char *)packet.object_value.string.bytes, ';')) + { + /* + * Description is the IEEE-1284 device ID... + */ + + if (!device->id) + device->id = strdup((char *)packet.object_value.string.bytes); + + backendGetMakeModel((char *)packet.object_value.string.bytes, + make_model, sizeof(make_model)); + + if (device->info) + free(device->info); + + device->info = strdup(make_model); + } + else + { + /* + * Description is plain text... + */ + + fix_make_model(make_model, (char *)packet.object_value.string.bytes, + sizeof(make_model)); + + if (device->info) + free(device->info); + + device->info = strdup((char *)packet.object_value.string.bytes); + } + + if (!device->make_and_model) + device->make_and_model = strdup(make_model); + } + break; + + case DEVICE_ID : + if (device && packet.object_type == CUPS_ASN1_OCTET_STRING && + (!device->id || + strlen(device->id) < packet.object_value.string.num_bytes)) + { + /* + * Update an existing cache entry... + */ + + char make_model[256]; /* Make and model */ + + + if (device->id) + free(device->id); + + device->id = strdup((char *)packet.object_value.string.bytes); + + /* + * Convert the ID to a make and model string... + */ + + backendGetMakeModel((char *)packet.object_value.string.bytes, + make_model, sizeof(make_model)); + if (device->make_and_model) + free(device->make_and_model); + + device->make_and_model = strdup(make_model); + } + break; + + case DEVICE_LOCATION : + if (device && packet.object_type == CUPS_ASN1_OCTET_STRING && + !device->location) + device->location = strdup((char *)packet.object_value.string.bytes); + break; + + case DEVICE_PRODUCT : + if (device && packet.object_type == CUPS_ASN1_OCTET_STRING && + !device->id) + { + /* + * Update an existing cache entry... + */ + + if (!device->info) + device->info = strdup((char *)packet.object_value.string.bytes); + + if (device->make_and_model) + free(device->make_and_model); + + device->make_and_model = strdup((char *)packet.object_value.string.bytes); + } + break; + + case DEVICE_URI : + if (device && packet.object_type == CUPS_ASN1_OCTET_STRING && + !device->uri && packet.object_value.string.num_bytes > 3) + { + /* + * Update an existing cache entry... + */ + + char scheme[32], /* URI scheme */ + userpass[256], /* Username:password in URI */ + hostname[256], /* Hostname in URI */ + resource[1024]; /* Resource path in URI */ + int port; /* Port number in URI */ + + if (!strncmp((char *)packet.object_value.string.bytes, "lpr:", 4)) + { + /* + * We want "lpd://..." for the URI... + */ + + packet.object_value.string.bytes[2] = 'd'; + } + + if (httpSeparateURI(HTTP_URI_CODING_ALL, + (char *)packet.object_value.string.bytes, + scheme, sizeof(scheme), + userpass, sizeof(userpass), + hostname, sizeof(hostname), &port, + resource, sizeof(resource)) >= HTTP_URI_OK) + device->uri = strdup((char *)packet.object_value.string.bytes); + } + break; + } +} + + +/* + * 'run_time()' - Return the total running time... + */ + +static double /* O - Number of seconds */ +run_time(void) +{ + struct timeval curtime; /* Current time */ + + + gettimeofday(&curtime, NULL); + + return (curtime.tv_sec - StartTime.tv_sec + + 0.000001 * (curtime.tv_usec - StartTime.tv_usec)); +} + + +/* + * 'scan_devices()' - Scan for devices using SNMP. + */ + +static void +scan_devices(int ipv4, /* I - SNMP IPv4 socket */ + int ipv6) /* I - SNMP IPv6 socket */ +{ + int fd, /* File descriptor for this address */ + busy; /* Are we busy processing something? */ + char *address, /* Current address */ + *community; /* Current community */ + fd_set input; /* Input set for select() */ + struct timeval timeout; /* Timeout for select() */ + time_t endtime; /* End time for scan */ + http_addrlist_t *addrs, /* List of addresses */ + *addr; /* Current address */ + snmp_cache_t *device; /* Current device */ + char temp[1024]; /* Temporary address string */ + + + gettimeofday(&StartTime, NULL); + + /* + * First send all of the broadcast queries... + */ + + for (address = (char *)cupsArrayFirst(Addresses); + address; + address = (char *)cupsArrayNext(Addresses)) + { + if (!strcmp(address, "@LOCAL")) + addrs = get_interface_addresses(NULL); + else if (!strncmp(address, "@IF(", 4)) + { + char ifname[255]; /* Interface name */ + + strlcpy(ifname, address + 4, sizeof(ifname)); + if (ifname[0]) + ifname[strlen(ifname) - 1] = '\0'; + + addrs = get_interface_addresses(ifname); + } + else + addrs = httpAddrGetList(address, AF_UNSPEC, NULL); + + if (!addrs) + { + fprintf(stderr, "ERROR: Unable to scan \"%s\"!\n", address); + continue; + } + + for (community = (char *)cupsArrayFirst(Communities); + community; + community = (char *)cupsArrayNext(Communities)) + { + debug_printf("DEBUG: Scanning for devices in \"%s\" via \"%s\"...\n", + community, address); + + for (addr = addrs; addr; addr = addr->next) + { +#ifdef AF_INET6 + if (_httpAddrFamily(&(addr->addr)) == AF_INET6) + fd = ipv6; + else +#endif /* AF_INET6 */ + fd = ipv4; + + debug_printf("DEBUG: Sending get request to %s...\n", + httpAddrString(&(addr->addr), temp, sizeof(temp))); + + _cupsSNMPWrite(fd, &(addr->addr), CUPS_SNMP_VERSION_1, community, + CUPS_ASN1_GET_REQUEST, DEVICE_TYPE, DeviceTypeOID); + } + } + + httpAddrFreeList(addrs); + } + + /* + * Then read any responses that come in over the next 3 seconds... + */ + + endtime = time(NULL) + MaxRunTime; + + FD_ZERO(&input); + + while (time(NULL) < endtime) + { + timeout.tv_sec = 2; + timeout.tv_usec = 0; + + FD_SET(ipv4, &input); + if (ipv6 >= 0) + FD_SET(ipv6, &input); + + fd = ipv4 > ipv6 ? ipv4 : ipv6; + if (select(fd + 1, &input, NULL, NULL, &timeout) < 0) + { + fprintf(stderr, "ERROR: %.3f select() for %d/%d failed: %s\n", run_time(), + ipv4, ipv6, strerror(errno)); + break; + } + + busy = 0; + + if (FD_ISSET(ipv4, &input)) + { + read_snmp_response(ipv4); + busy = 1; + } + + if (ipv6 >= 0 && FD_ISSET(ipv6, &input)) + { + read_snmp_response(ipv6); + busy = 1; + } + + if (!busy) + { + /* + * List devices with complete information... + */ + + int sent_something = 0; + + for (device = (snmp_cache_t *)cupsArrayFirst(Devices); + device; + device = (snmp_cache_t *)cupsArrayNext(Devices)) + if (!device->sent && device->info && device->make_and_model) + { + if (device->uri) + list_device(device); + else + probe_device(device); + + device->sent = sent_something = 1; + } + + if (!sent_something) + break; + } + } + + debug_printf("DEBUG: %.3f Scan complete!\n", run_time()); +} + + +/* + * 'try_connect()' - Try connecting on a port... + */ + +static int /* O - 0 on success or -1 on error */ +try_connect(http_addr_t *addr, /* I - Socket address */ + const char *addrname, /* I - Hostname or IP address */ + int port) /* I - Port number */ +{ + int fd; /* Socket */ + int status; /* Connection status */ + + + debug_printf("DEBUG: %.3f Trying %s://%s:%d...\n", run_time(), + port == 515 ? "lpd" : "socket", addrname, port); + + if ((fd = socket(_httpAddrFamily(addr), SOCK_STREAM, 0)) < 0) + { + fprintf(stderr, "ERROR: Unable to create socket: %s\n", + strerror(errno)); + return (-1); + } + + _httpAddrSetPort(addr, port); + + alarm(1); + + status = connect(fd, (void *)addr, httpAddrLength(addr)); + + close(fd); + alarm(0); + + return (status); +} + + +/* + * 'update_cache()' - Update a cached device... + */ + +static void +update_cache(snmp_cache_t *device, /* I - Device */ + const char *uri, /* I - Device URI */ + const char *id, /* I - Device ID */ + const char *make_model) /* I - Device make and model */ +{ + if (device->uri) + free(device->uri); + + device->uri = strdup(uri); + + if (id) + { + if (device->id) + free(device->id); + + device->id = strdup(id); + } + + if (make_model) + { + if (device->make_and_model) + free(device->make_and_model); + + device->make_and_model = strdup(make_model); + } + + list_device(device); +} + + +/* + * End of "$Id$". + */ diff --git a/backend/snmp.txt b/backend/snmp.txt new file mode 100644 index 0000000000..77a38c498d --- /dev/null +++ b/backend/snmp.txt @@ -0,0 +1,172 @@ +snmp.txt - 2006-04-19 +--------------------- + +This file lists the "interesting" bits from the command: + + snmpwalk -v 1 -c public HOST .1 + +for many network print servers and internal cards. It is mainly here +for SNMP documentation and development purposes. + + +AXIS 5600 + +SNMPv2-MIB::sysDescr.0 = STRING: HP ETHERNET MULTI-ENVIRONMENT,ROM J.sp.00,JETDIRECT EX,JD28,EEPROM 6.16.5 +SNMPv2-MIB::sysName.0 = STRING: +HOST-RESOURCES-MIB::hrDeviceType.1 = OID: HOST-RESOURCES-TYPES::hrDevicePrinter +HOST-RESOURCES-MIB::hrDeviceType.2 = OID: HOST-RESOURCES-TYPES::hrDevicePrinter +HOST-RESOURCES-MIB::hrDeviceType.3 = OID: HOST-RESOURCES-TYPES::hrDevicePrinter +HOST-RESOURCES-MIB::hrDeviceDescr.1 = STRING: Hewlett-Packard hp LaserJet 3380 +HOST-RESOURCES-MIB::hrDeviceDescr.2 = STRING: Axis AXIS 5600 +HOST-RESOURCES-MIB::hrDeviceDescr.3 = STRING: Axis AXIS 5600 +HOST-RESOURCES-MIB::hrDeviceID.1 = OID: SNMPv2-SMI::zeroDotZero +HOST-RESOURCES-MIB::hrDeviceID.2 = OID: SNMPv2-SMI::zeroDotZero +HOST-RESOURCES-MIB::hrDeviceID.3 = OID: SNMPv2-SMI::zeroDotZero +SNMPv2-SMI::enterprises.11.2.4.3.10.8.0 = STRING: "AXIS433AE8" +SNMPv2-SMI::enterprises.368.2.3.2.601.0 = INTEGER: 9100 +SNMPv2-SMI::enterprises.368.2.3.2.602.0 = INTEGER: 9101 +SNMPv2-SMI::enterprises.368.2.3.2.603.0 = INTEGER: 9102 +SNMPv2-SMI::enterprises.368.2.3.10.901.0 = STRING: "AXIS433AE8" + + +AXIS OfficeBasic + +SNMPv2-MIB::sysDescr.0 = STRING: AXIS OfficeBasic Parallel Network Print Server V6.43 Sep 4 2003 +HOST-RESOURCES-MIB::hrDeviceType.1 = OID: HOST-RESOURCES-TYPES::hrDevicePrinter +HOST-RESOURCES-MIB::hrDeviceDescr.1 = STRING: EPSON Stylus Photo 870 +HOST-RESOURCES-MIB::hrDeviceID.1 = OID: SNMPv2-SMI::zeroDotZero + + +DLink DP-301P+ + +SNMPv2-MIB::sysDescr.0 = STRING: D-Link DP-301P+ Print Server + + +Genicom ML280 + +SNMPv2-MIB::sysDescr.0 = STRING: GENICOM microLaser 280 +SNMPv2-MIB::sysName.0 = STRING: PRQ_004F75 +HOST-RESOURCES-MIB::hrDeviceType.1 = OID: HOST-RESOURCES-TYPES::hrDevicePrinter +HOST-RESOURCES-MIB::hrDeviceDescr.1 = STRING: MANUFACTURER:GENICOM;MODEL:microLaser 280; +HOST-RESOURCES-MIB::hrDeviceID.1 = OID: SNMPv2-SMI::enterprises.3369.1.1.2.4 + + +EPSON Type-B Network Card + +SNMPv2-MIB::sysDescr.0 = STRING: EPSON Type-B 10Base-T/100Base-TX Print Server +SNMPv2-MIB::sysName.0 = STRING: StylusPro7600-BB87A8 +HOST-RESOURCES-MIB::hrDeviceType.1 = OID: HOST-RESOURCES-TYPES::hrDevicePrinter +HOST-RESOURCES-MIB::hrDeviceDescr.1 = STRING: EPSON Stylus Pro 7600 +HOST-RESOURCES-MIB::hrDeviceID.1 = OID: SNMPv2-SMI::enterprises.1248.1.2.1.22.69.109.117.108.97.116.101.83.116.121.108.117.115.32.80.114.111.32.55.54.48.48 +SNMPv2-SMI::enterprises.11.2.3.9.1.1.7.0 = STRING: "MFG:EPSON;CMD:ESCPL2,BDC;MDL:Stylus Pro 7600;CLS:PRINTER;DES:EPSON Stylus Pro 7600;" +SNMPv2-SMI::enterprises.1248.1.2.2.1.1.1.1.1 = STRING: "MFG:EPSON;CMD:ESCPL2,BDC;MDL:Stylus Pro 7600;CLS:PRINTER;DES:EPSON Stylus Pro 7600;" + + +EPSON Wireless 802.11b Print Server + +SNMPv2-MIB::sysDescr.0 = STRING: EPSON Wireless LAN Print Interface compatible with an HP JETDIRECT EX +SNMPv2-MIB::sysName.0 = STRING: EAI_0F550B +HOST-RESOURCES-MIB::hrDeviceType.1 = OID: HOST-RESOURCES-TYPES::hrDevicePrinter +HOST-RESOURCES-MIB::hrDeviceDescr.1 = STRING: +HOST-RESOURCES-MIB::hrDeviceID.1 = OID: SNMPv2-SMI::zeroDotZero + + +HP JetDirect EX3plus + +SNMPv2-MIB::sysDescr.0 = STRING: HP ETHERNET MULTI-ENVIRONMENT,ROM D.04.03,JETDIRECT EX,JD26,EEPROM D.05.22 +SNMPv2-MIB::sysName.0 = STRING: NPID1EC0F + + +HP LJ4000 + +SNMPv2-MIB::sysDescr.0 = STRING: HP ETHERNET MULTI-ENVIRONMENT,ROM G.05.34,JETDIRECT,JD30,EEPROM G.08.32 +SNMPv2-MIB::sysName.0 = STRING: +HOST-RESOURCES-MIB::hrDeviceType.1 = OID: HOST-RESOURCES-TYPES::hrDevicePrinter +HOST-RESOURCES-MIB::hrDeviceDescr.1 = STRING: HP LaserJet 4000 Series +HOST-RESOURCES-MIB::hrDeviceID.1 = OID: SNMPv2-SMI::enterprises.11.2.3.9.1.2.5 + + +HP CLJ4550 + +SNMPv2-MIB::sysDescr.0 = STRING: HP ETHERNET MULTI-ENVIRONMENT,ROM L.20.07,JETDIRECT,JD84,EEPROM L.21.22,CIDATE 07/06/2001 +SNMPv2-MIB::sysName.0 = STRING: NPI02FDE7 +HOST-RESOURCES-MIB::hrDeviceDescr.1 = STRING: HP Color LaserJet 4550 +HOST-RESOURCES-MIB::hrDeviceDescr.2 = STRING: Hewlett-Packard Dynamic RAM Disk +HOST-RESOURCES-MIB::hrDeviceID.1 = OID: SNMPv2-SMI::enterprises.11.2.3.9.1.2.14 + + +Lexmark C522 + +SNMPv2-MIB::sysDescr.0 = STRING: Lexmark C522 version NS.NP.N212 kernel 2.6.6 All-N-1 +SNMPv2-MIB::sysName.0 = STRING: ET0004000D0CCA +HOST-RESOURCES-MIB::hrDeviceType.1 = OID: HOST-RESOURCES-TYPES::hrDevicePrinter +HOST-RESOURCES-MIB::hrDeviceType.2 = OID: HOST-RESOURCES-TYPES::hrDeviceNonVolatileMemory +HOST-RESOURCES-MIB::hrDeviceType.3 = OID: HOST-RESOURCES-TYPES::hrDeviceProcessor +HOST-RESOURCES-MIB::hrDeviceType.4 = OID: HOST-RESOURCES-TYPES::hrDeviceSerialPort +HOST-RESOURCES-MIB::hrDeviceType.5 = OID: HOST-RESOURCES-TYPES::hrDeviceNetwork +HOST-RESOURCES-MIB::hrDeviceDescr.1 = STRING: Lexmark C522 9421TTV LS.FA.P129 +HOST-RESOURCES-MIB::hrDeviceDescr.2 = STRING: Nonvolatile RAM +HOST-RESOURCES-MIB::hrDeviceDescr.3 = STRING: IBM 750 Rev CXr +HOST-RESOURCES-MIB::hrDeviceDescr.4 = STRING: USB Interface +HOST-RESOURCES-MIB::hrDeviceDescr.5 = STRING: Network Interface +HOST-RESOURCES-MIB::hrDeviceID.1 = OID: SNMPv2-SMI::zeroDotZero +HOST-RESOURCES-MIB::hrDeviceID.2 = OID: SNMPv2-SMI::zeroDotZero +HOST-RESOURCES-MIB::hrDeviceID.3 = OID: SNMPv2-SMI::zeroDotZero +HOST-RESOURCES-MIB::hrDeviceID.4 = OID: SNMPv2-SMI::zeroDotZero +HOST-RESOURCES-MIB::hrDeviceID.5 = OID: SNMPv2-SMI::enterprises.641.1 +SNMPv2-SMI::enterprises.641.2.1.2.1.2.1 = STRING: "Lexmark C522" +SNMPv2-SMI::enterprises.641.2.1.2.1.3.1 = STRING: "MANUFACTURER:Lexmark International;COMMAND SET:;MODEL:Lexmark C522" + + +Linksys EPSX3 + +SNMPv2-MIB::sysDescr.0 = STRING: ETHERNET MULTI-ENVIRONMENT.ROM, JETDIRECT EX, EEPROM 6016 + + +NetGear PS113 + +SNMPv2-MIB::sysDescr.0 = STRING: A SNMP proxy agent. + + +Okidata C7200 + +SNMPv2-MIB::sysDescr.0 = STRING: OkiLAN 6200e +SNMPv2-MIB::sysName.0 = STRING: OKI7009715 +HOST-RESOURCES-MIB::hrDeviceType.1 = OID: HOST-RESOURCES-TYPES::hrDevicePrinter +HOST-RESOURCES-MIB::hrDeviceType.2 = OID: HOST-RESOURCES-TYPES::hrDeviceNonVolat +ileMemory +HOST-RESOURCES-MIB::hrDeviceDescr.1 = STRING: C7200 +HOST-RESOURCES-MIB::hrDeviceDescr.2 = STRING: FLASH0 +HOST-RESOURCES-MIB::hrDeviceID.1 = OID: SNMPv2-SMI::enterprises.2001.1.1.1.1 +HOST-RESOURCES-MIB::hrDeviceID.2 = OID: SNMPv2-SMI::zeroDotZero +SNMPv2-SMI::mib-2.43.14.1.1.3.1.1 = STRING: "IEEE 1284" +SNMPv2-SMI::mib-2.43.14.1.1.3.1.2 = STRING: "EtherTalk Phase 2" +SNMPv2-SMI::mib-2.43.14.1.1.3.1.3 = STRING: "LPD" +SNMPv2-SMI::mib-2.43.14.1.1.3.1.4 = STRING: "Netware Rprinter" +SNMPv2-SMI::mib-2.43.14.1.1.3.1.5 = STRING: "Netware Bindery or NDS Pserver" +SNMPv2-SMI::mib-2.43.14.1.1.3.1.6 = STRING: "Raw TCP Port 9100" +SNMPv2-SMI::mib-2.43.14.1.1.3.1.7 = STRING: "FTP" +SNMPv2-SMI::mib-2.43.14.1.1.3.1.8 = STRING: "DLC/LLC" +SNMPv2-SMI::enterprises.2001.1.1.1.1.1.3530.0 = STRING: "C7200" + + +Xerox N2025 + +SNMPv2-MIB::sysDescr.0 = STRING: Xerox DocuPrint N2025 Network Laser Printer - 2.12-02 +SNMPv2-MIB::sysName.0 = STRING: +HOST-RESOURCES-MIB::hrDeviceType.1 = OID: HOST-RESOURCES-TYPES::hrDevicePrinter +HOST-RESOURCES-MIB::hrDeviceType.2 = OID: HOST-RESOURCES-TYPES::hrDeviceParallelPort +HOST-RESOURCES-MIB::hrDeviceType.3 = OID: HOST-RESOURCES-TYPES::hrDeviceNetwork +HOST-RESOURCES-MIB::hrDeviceType.6 = OID: HOST-RESOURCES-TYPES::hrDeviceProcessor +HOST-RESOURCES-MIB::hrDeviceType.7 = OID: HOST-RESOURCES-TYPES::hrDeviceOther +HOST-RESOURCES-MIB::hrDeviceType.9 = OID: HOST-RESOURCES-TYPES::hrDeviceVolatileMemory +HOST-RESOURCES-MIB::hrDeviceType.10 = OID: HOST-RESOURCES-TYPES::hrDeviceNonVolatileMemory +HOST-RESOURCES-MIB::hrDeviceDescr.1 = STRING: Xerox DocuPrint N2025 Network Laser Printer - 2.12-02 +HOST-RESOURCES-MIB::hrDeviceDescr.2 = STRING: IEEE 1284 port +HOST-RESOURCES-MIB::hrDeviceDescr.3 = STRING: Ethernet port +HOST-RESOURCES-MIB::hrDeviceDescr.6 = STRING: Motorola Power PC +HOST-RESOURCES-MIB::hrDeviceDescr.7 = STRING: USB Port +HOST-RESOURCES-MIB::hrDeviceDescr.9 = STRING: RAM Memory +HOST-RESOURCES-MIB::hrDeviceDescr.10 = STRING: ROM Memory +HOST-RESOURCES-MIB::hrDeviceID.1 = OID: SNMPv2-SMI::enterprises.253.8.62.1.3.2.17.1 + diff --git a/backend/socket.c b/backend/socket.c new file mode 100644 index 0000000000..39c31fb099 --- /dev/null +++ b/backend/socket.c @@ -0,0 +1,527 @@ +/* + * "$Id$" + * + * AppSocket backend for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Send a file to the printer or server. + * wait_bc() - Wait for back-channel data... + */ + +/* + * Include necessary headers. + */ + +#include +#include "backend-private.h" +#include +#include +#include + +#ifdef WIN32 +# include +#else +# include +# include +# include +# include +# include +# include +#endif /* WIN32 */ + + +/* + * Local functions... + */ + +static int wait_bc(int device_fd, int secs); + + +/* + * 'main()' - Send a file to the printer or server. + * + * Usage: + * + * printer-uri job-id user title copies options [file] + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments (6 or 7) */ + char *argv[]) /* I - Command-line arguments */ +{ + const char *device_uri; /* Device URI */ + char scheme[255], /* Scheme in URI */ + hostname[1024], /* Hostname */ + username[255], /* Username info (not used) */ + resource[1024], /* Resource info (not used) */ + *options, /* Pointer to options */ + *name, /* Name of option */ + *value, /* Value of option */ + sep; /* Option separator */ + int print_fd; /* Print file */ + int copies; /* Number of copies to print */ + time_t start_time; /* Time of first connect */ +#ifdef __APPLE__ + time_t current_time, /* Current time */ + wait_time; /* Time to wait before shutting down socket */ +#endif /* __APPLE__ */ + int contimeout; /* Connection timeout */ + int waiteof; /* Wait for end-of-file? */ + int port; /* Port number */ + char portname[255]; /* Port name */ + int delay; /* Delay for retries... */ + int device_fd; /* AppSocket */ + int error; /* Error code (if any) */ + http_addrlist_t *addrlist, /* Address list */ + *addr; /* Connected address */ + char addrname[256]; /* Address name */ + int snmp_fd, /* SNMP socket */ + start_count, /* Page count via SNMP at start */ + page_count, /* Page count via SNMP */ + have_supplies; /* Printer supports supply levels? */ + ssize_t bytes = 0, /* Initial bytes read */ + tbytes; /* Total number of bytes written */ + char buffer[1024]; /* Initial print buffer */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* Actions for POSIX signals */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ + + + /* + * Make sure status messages are not buffered... + */ + + setbuf(stderr, NULL); + + /* + * Ignore SIGPIPE signals... + */ + +#ifdef HAVE_SIGSET + sigset(SIGPIPE, SIG_IGN); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + action.sa_handler = SIG_IGN; + sigaction(SIGPIPE, &action, NULL); +#else + signal(SIGPIPE, SIG_IGN); +#endif /* HAVE_SIGSET */ + + /* + * Check command-line... + */ + + if (argc == 1) + { + printf("network socket \"Unknown\" \"%s\"\n", + _cupsLangString(cupsLangDefault(), _("AppSocket/HP JetDirect"))); + return (CUPS_BACKEND_OK); + } + else if (argc < 6 || argc > 7) + { + _cupsLangPrintf(stderr, + _("Usage: %s job-id user title copies options [file]"), + argv[0]); + return (CUPS_BACKEND_FAILED); + } + + /* + * If we have 7 arguments, print the file named on the command-line. + * Otherwise, send stdin instead... + */ + + if (argc == 6) + { + print_fd = 0; + copies = 1; + } + else + { + /* + * Try to open the print file... + */ + + if ((print_fd = open(argv[6], O_RDONLY)) < 0) + { + _cupsLangPrintError("ERROR", _("Unable to open print file")); + return (CUPS_BACKEND_FAILED); + } + + copies = atoi(argv[4]); + } + + /* + * Extract the hostname and port number from the URI... + */ + + while ((device_uri = cupsBackendDeviceURI(argv)) == NULL) + { + _cupsLangPrintFilter(stderr, "INFO", _("Unable to locate printer.")); + sleep(10); + + if (getenv("CLASS") != NULL) + return (CUPS_BACKEND_FAILED); + } + + httpSeparateURI(HTTP_URI_CODING_ALL, device_uri, scheme, sizeof(scheme), + username, sizeof(username), hostname, sizeof(hostname), &port, + resource, sizeof(resource)); + + if (port == 0) + port = 9100; /* Default to HP JetDirect/Tektronix PhaserShare */ + + /* + * Get options, if any... + */ + + waiteof = 1; + contimeout = 7 * 24 * 60 * 60; + + if ((options = strchr(resource, '?')) != NULL) + { + /* + * Yup, terminate the device name string and move to the first + * character of the options... + */ + + *options++ = '\0'; + + /* + * Parse options... + */ + + while (*options) + { + /* + * Get the name... + */ + + name = options; + + while (*options && *options != '=' && *options != '+' && *options != '&') + options ++; + + if ((sep = *options) != '\0') + *options++ = '\0'; + + if (sep == '=') + { + /* + * Get the value... + */ + + value = options; + + while (*options && *options != '+' && *options != '&') + options ++; + + if (*options) + *options++ = '\0'; + } + else + value = (char *)""; + + /* + * Process the option... + */ + + if (!_cups_strcasecmp(name, "waiteof")) + { + /* + * Set the wait-for-eof value... + */ + + waiteof = !value[0] || !_cups_strcasecmp(value, "on") || + !_cups_strcasecmp(value, "yes") || !_cups_strcasecmp(value, "true"); + } + else if (!_cups_strcasecmp(name, "contimeout")) + { + /* + * Set the connection timeout... + */ + + if (atoi(value) > 0) + contimeout = atoi(value); + } + } + } + + /* + * Then try finding the remote host... + */ + + start_time = time(NULL); + + sprintf(portname, "%d", port); + + fputs("STATE: +connecting-to-device\n", stderr); + fprintf(stderr, "DEBUG: Looking up \"%s\"...\n", hostname); + + while ((addrlist = httpAddrGetList(hostname, AF_UNSPEC, portname)) == NULL) + { + _cupsLangPrintFilter(stderr, "INFO", + _("Unable to locate printer \"%s\"."), hostname); + sleep(10); + + if (getenv("CLASS") != NULL) + { + fputs("STATE: -connecting-to-device\n", stderr); + return (CUPS_BACKEND_STOP); + } + } + + /* + * See if the printer supports SNMP... + */ + + if ((snmp_fd = _cupsSNMPOpen(addrlist->addr.addr.sa_family)) >= 0) + { + have_supplies = !backendSNMPSupplies(snmp_fd, &(addrlist->addr), + &start_count, NULL); + } + else + have_supplies = start_count = 0; + + /* + * Wait for data from the filter... + */ + + if (print_fd == 0) + { + if (!backendWaitLoop(snmp_fd, &(addrlist->addr), 1, backendNetworkSideCB)) + return (CUPS_BACKEND_OK); + else if ((bytes = read(0, buffer, sizeof(buffer))) <= 0) + return (CUPS_BACKEND_OK); + } + + /* + * Connect to the printer... + */ + + fprintf(stderr, "DEBUG: Connecting to %s:%d\n", hostname, port); + _cupsLangPrintFilter(stderr, "INFO", _("Connecting to printer.")); + + for (delay = 5;;) + { + if ((addr = httpAddrConnect(addrlist, &device_fd)) == NULL) + { + error = errno; + device_fd = -1; + + if (getenv("CLASS") != NULL) + { + /* + * If the CLASS environment variable is set, the job was submitted + * to a class and not to a specific queue. In this case, we want + * to abort immediately so that the job can be requeued on the next + * available printer in the class. + */ + + _cupsLangPrintFilter(stderr, "INFO", + _("Unable to contact printer, queuing on next " + "printer in class.")); + + /* + * Sleep 5 seconds to keep the job from requeuing too rapidly... + */ + + sleep(5); + + return (CUPS_BACKEND_FAILED); + } + + fprintf(stderr, "DEBUG: Connection error: %s\n", strerror(error)); + + if (error == ECONNREFUSED || error == EHOSTDOWN || + error == EHOSTUNREACH) + { + if (contimeout && (time(NULL) - start_time) > contimeout) + { + _cupsLangPrintFilter(stderr, "ERROR", + _("The printer is not responding.")); + return (CUPS_BACKEND_FAILED); + } + + switch (error) + { + case EHOSTDOWN : + _cupsLangPrintFilter(stderr, "WARNING", + _("The printer may not exist or " + "is unavailable at this time.")); + break; + + case EHOSTUNREACH : + _cupsLangPrintFilter(stderr, "WARNING", + _("The printer is unreachable at this " + "time.")); + break; + + case ECONNREFUSED : + default : + _cupsLangPrintFilter(stderr, "WARNING", + _("The printer is busy.")); + break; + } + + sleep(delay); + + if (delay < 30) + delay += 5; + } + else + { + _cupsLangPrintFilter(stderr, "ERROR", + _("The printer is not responding.")); + sleep(30); + } + } + else + break; + } + + fputs("STATE: -connecting-to-device\n", stderr); + _cupsLangPrintFilter(stderr, "INFO", _("Connected to printer.")); + + fprintf(stderr, "DEBUG: Connected to %s:%d...\n", + httpAddrString(&(addr->addr), addrname, sizeof(addrname)), + _httpAddrPort(&(addr->addr))); + + /* + * Print everything... + */ + + tbytes = 0; + + if (bytes > 0) + tbytes += write(device_fd, buffer, bytes); + + while (copies > 0 && tbytes >= 0) + { + copies --; + + if (print_fd != 0) + { + fputs("PAGE: 1 1\n", stderr); + lseek(print_fd, 0, SEEK_SET); + } + + tbytes = backendRunLoop(print_fd, device_fd, snmp_fd, &(addrlist->addr), 1, + 0, backendNetworkSideCB); + + if (print_fd != 0 && tbytes >= 0) + _cupsLangPrintFilter(stderr, "INFO", _("Print file sent.")); + } + +#ifdef __APPLE__ + /* + * Wait up to 5 seconds to get any pending back-channel data... + */ + + wait_time = time(NULL) + 5; + while (wait_time >= time(¤t_time)) + if (wait_bc(device_fd, wait_time - current_time) <= 0) + break; +#endif /* __APPLE__ */ + + if (waiteof) + { + /* + * Shutdown the socket and wait for the other end to finish... + */ + + _cupsLangPrintFilter(stderr, "INFO", _("Waiting for printer to finish.")); + + shutdown(device_fd, 1); + + while (wait_bc(device_fd, 90) > 0); + } + + /* + * Collect the final page count as needed... + */ + + if (have_supplies && + !backendSNMPSupplies(snmp_fd, &(addrlist->addr), &page_count, NULL) && + page_count > start_count) + fprintf(stderr, "PAGE: total %d\n", page_count - start_count); + + /* + * Close the socket connection... + */ + + close(device_fd); + + httpAddrFreeList(addrlist); + + /* + * Close the input file and return... + */ + + if (print_fd != 0) + close(print_fd); + + _cupsLangPrintFilter(stderr, "INFO", _("Ready to print.")); + + return (CUPS_BACKEND_OK); +} + + +/* + * 'wait_bc()' - Wait for back-channel data... + */ + +static int /* O - # bytes read or -1 on error */ +wait_bc(int device_fd, /* I - Socket */ + int secs) /* I - Seconds to wait */ +{ + struct timeval timeout; /* Timeout for select() */ + fd_set input; /* Input set for select() */ + ssize_t bytes; /* Number of back-channel bytes read */ + char buffer[1024]; /* Back-channel buffer */ + + + /* + * Wait up to "secs" seconds for backchannel data... + */ + + timeout.tv_sec = secs; + timeout.tv_usec = 0; + + FD_ZERO(&input); + FD_SET(device_fd, &input); + + if (select(device_fd + 1, &input, NULL, NULL, &timeout) > 0) + { + /* + * Grab the data coming back and spit it out to stderr... + */ + + if ((bytes = read(device_fd, buffer, sizeof(buffer))) > 0) + { + fprintf(stderr, "DEBUG: Received %d bytes of back-channel data\n", + (int)bytes); + cupsBackChannelWrite(buffer, bytes, 1.0); + } + + return (bytes); + } + else + return (-1); +} + + +/* + * End of "$Id$". + */ diff --git a/backend/test1284.c b/backend/test1284.c new file mode 100644 index 0000000000..49bef87ee3 --- /dev/null +++ b/backend/test1284.c @@ -0,0 +1,84 @@ +/* + * "$Id$" + * + * IEEE-1284 support functions test program for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Test the device-ID functions. + */ + +/* + * Include necessary headers. + */ + +#include +#ifdef WIN32 +# include +#else +# include +# include +#endif /* WIN32 */ + +#include "ieee1284.c" + + +/* + * 'main()' - Test the device-ID functions. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int i, /* Looping var */ + fd; /* File descriptor */ + char device_id[1024], /* 1284 device ID string */ + make_model[1024], /* make-and-model string */ + uri[1024]; /* URI string */ + + + if (argc < 2) + { + puts("Usage: test1284 device-file [... device-file-N]"); + exit(1); + } + + for (i = 1; i < argc; i ++) + { + if ((fd = open(argv[i], O_RDWR)) < 0) + { + perror(argv[i]); + return (errno); + } + + printf("%s:\n", argv[i]); + + backendGetDeviceID(fd, device_id, sizeof(device_id), make_model, + sizeof(make_model), "test", uri, sizeof(uri)); + + printf(" device_id=\"%s\"\n", device_id); + printf(" make_model=\"%s\"\n", make_model); + printf(" uri=\"%s\"\n", uri); + + close(fd); + } + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/backend/testbackend.c b/backend/testbackend.c new file mode 100644 index 0000000000..6182471d12 --- /dev/null +++ b/backend/testbackend.c @@ -0,0 +1,670 @@ +/* + * "$Id$" + * + * Backend test program for CUPS. + * + * Copyright 2007-2012 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Run the named backend. + * sigterm_handler() - Flag when we get SIGTERM. + * usage() - Show usage information. + * walk_cb() - Show results of cupsSideChannelSNMPWalk... + */ + +/* + * Include necessary headers. + */ + +#include +#include +#include +#include +#include +#include +#include + + +/* + * Local globals... + */ + +static int job_canceled = 0; + + +/* + * Local functions... + */ + +static void sigterm_handler(int sig); +static void usage(void) __attribute__((noreturn)); +static void walk_cb(const char *oid, const char *data, int datalen, + void *context); + + +/* + * 'main()' - Run the named backend. + * + * Usage: + * + * testbackend [-s] [-t] device-uri job-id user title copies options [file] + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int first_arg, /* First argument for backend */ + do_cancel = 0, /* Simulate a cancel-job via SIGTERM */ + do_ps = 0, /* Do PostScript query+test? */ + do_pcl = 0, /* Do PCL query+test? */ + do_side_tests = 0, /* Test side-channel ops? */ + do_trickle = 0, /* Trickle data to backend */ + do_walk = 0, /* Do OID lookup (0) or walking (1) */ + show_log = 0; /* Show log messages from backends? */ + const char *oid = ".1.3.6.1.2.1.43.10.2.1.4.1.1"; + /* OID to lookup or walk */ + char scheme[255], /* Scheme in URI == backend */ + backend[1024], /* Backend path */ + libpath[1024], /* Path for libcups */ + *ptr; /* Pointer into path */ + const char *serverbin; /* CUPS_SERVERBIN environment variable */ + int fd, /* Temporary file descriptor */ + back_fds[2], /* Back-channel pipe */ + side_fds[2], /* Side-channel socket */ + data_fds[2], /* Data pipe */ + back_pid = -1, /* Backend process ID */ + data_pid = -1, /* Trickle process ID */ + pid, /* Process ID */ + status; /* Exit status */ + + + /* + * Get the current directory and point the run-time linker at the "cups" + * subdirectory... + */ + + if (getcwd(libpath, sizeof(libpath)) && + (ptr = strrchr(libpath, '/')) != NULL && !strcmp(ptr, "/backend")) + { + strlcpy(ptr, "/cups", sizeof(libpath) - (ptr - libpath)); + if (access(libpath, 0)) +#ifdef __APPLE__ + setenv("DYLD_LIBRARY_PATH", libpath, 1); +#else + setenv("LD_LIBRARY_PATH", libpath, 1); +#endif /* __APPLE__ */ + } + + /* + * See if we have side-channel tests to do... + */ + + for (first_arg = 1; + argv[first_arg] && argv[first_arg][0] == '-'; + first_arg ++) + if (!strcmp(argv[first_arg], "-d")) + show_log = 1; + else if (!strcmp(argv[first_arg], "-cancel")) + do_cancel = 1; + else if (!strcmp(argv[first_arg], "-pcl")) + do_pcl = 1; + else if (!strcmp(argv[first_arg], "-ps")) + do_ps = 1; + else if (!strcmp(argv[first_arg], "-s")) + do_side_tests = 1; + else if (!strcmp(argv[first_arg], "-t")) + do_trickle = 1; + else if (!strcmp(argv[first_arg], "-get") && (first_arg + 1) < argc) + { + first_arg ++; + + do_side_tests = 1; + oid = argv[first_arg]; + } + else if (!strcmp(argv[first_arg], "-walk") && (first_arg + 1) < argc) + { + first_arg ++; + + do_side_tests = 1; + do_walk = 1; + oid = argv[first_arg]; + } + else + usage(); + + argc -= first_arg; + if (argc < 6 || argc > 7 || (argc == 7 && do_trickle)) + usage(); + + /* + * Extract the scheme from the device-uri - that's the program we want to + * execute. + */ + + if (sscanf(argv[first_arg], "%254[^:]", scheme) != 1) + { + fputs("testbackend: Bad device-uri - no colon!\n", stderr); + return (1); + } + + if (!access(scheme, X_OK)) + strlcpy(backend, scheme, sizeof(backend)); + else + { + if ((serverbin = getenv("CUPS_SERVERBIN")) == NULL) + serverbin = CUPS_SERVERBIN; + + snprintf(backend, sizeof(backend), "%s/backend/%s", serverbin, scheme); + if (access(backend, X_OK)) + { + fprintf(stderr, "testbackend: Unknown device scheme \"%s\"!\n", scheme); + return (1); + } + } + + /* + * Create the back-channel pipe and side-channel socket... + */ + + open("/dev/null", O_WRONLY); /* Make sure fd 3 and 4 are used */ + open("/dev/null", O_WRONLY); + + pipe(back_fds); + fcntl(back_fds[0], F_SETFL, fcntl(back_fds[0], F_GETFL) | O_NONBLOCK); + fcntl(back_fds[1], F_SETFL, fcntl(back_fds[1], F_GETFL) | O_NONBLOCK); + + socketpair(AF_LOCAL, SOCK_STREAM, 0, side_fds); + fcntl(side_fds[0], F_SETFL, fcntl(side_fds[0], F_GETFL) | O_NONBLOCK); + fcntl(side_fds[1], F_SETFL, fcntl(side_fds[1], F_GETFL) | O_NONBLOCK); + + /* + * Execute the trickle process as needed... + */ + + if (do_trickle || do_pcl || do_ps || do_cancel) + { + pipe(data_fds); + + signal(SIGTERM, sigterm_handler); + + if ((data_pid = fork()) == 0) + { + /* + * Trickle/query child comes here. Rearrange file descriptors so that + * FD 1, 3, and 4 point to the backend... + */ + + if ((fd = open("/dev/null", O_RDONLY)) != 0) + { + dup2(fd, 0); + close(fd); + } + + if (data_fds[1] != 1) + { + dup2(data_fds[1], 1); + close(data_fds[1]); + } + close(data_fds[0]); + + if (back_fds[0] != 3) + { + dup2(back_fds[0], 3); + close(back_fds[0]); + } + close(back_fds[1]); + + if (side_fds[0] != 4) + { + dup2(side_fds[0], 4); + close(side_fds[0]); + } + close(side_fds[1]); + + if (do_trickle) + { + /* + * Write 10 spaces, 1 per second... + */ + + int i; /* Looping var */ + + for (i = 0; i < 10; i ++) + { + write(1, " ", 1); + sleep(1); + } + } + else if (do_cancel) + { + /* + * Write PS or PCL lines until we see SIGTERM... + */ + + int line = 0, page = 0; /* Current line and page */ + ssize_t bytes; /* Number of bytes of response data */ + char buffer[1024]; /* Output buffer */ + + + if (do_pcl) + write(1, "\033E", 2); + else + write(1, "%!\n/Courier findfont 12 scalefont setfont 0 setgray\n", 52); + + while (!job_canceled) + { + if (line == 0) + { + page ++; + + if (do_pcl) + snprintf(buffer, sizeof(buffer), "PCL Page %d\r\n\r\n", page); + else + snprintf(buffer, sizeof(buffer), + "18 732 moveto (PS Page %d) show\n", page); + + write(1, buffer, strlen(buffer)); + } + + line ++; + + if (do_pcl) + snprintf(buffer, sizeof(buffer), "Line %d\r\n", line); + else + snprintf(buffer, sizeof(buffer), "18 %d moveto (Line %d) show\n", + 720 - line * 12, line); + + write(1, buffer, strlen(buffer)); + + if (line >= 55) + { + /* + * Eject after 55 lines... + */ + + line = 0; + if (do_pcl) + write(1, "\014", 1); + else + write(1, "showpage\n", 9); + } + + /* + * Check for back-channel data... + */ + + if ((bytes = cupsBackChannelRead(buffer, sizeof(buffer), 0)) > 0) + write(2, buffer, bytes); + + /* + * Throttle output to ~100hz... + */ + + usleep(10000); + } + + /* + * Eject current page with info... + */ + + if (do_pcl) + snprintf(buffer, sizeof(buffer), + "Canceled on line %d of page %d\r\n\014\033E", line, page); + else + snprintf(buffer, sizeof(buffer), + "\n18 %d moveto (Canceled on line %d of page %d)\nshowpage\n", + 720 - line * 12, line, page); + + write(1, buffer, strlen(buffer)); + + /* + * See if we get any back-channel data... + */ + + while ((bytes = cupsBackChannelRead(buffer, sizeof(buffer), 5.0)) > 0) + write(2, buffer, bytes); + + exit(0); + } + else + { + /* + * Do PS or PCL query + test pages. + */ + + char buffer[1024]; /* Buffer for response data */ + ssize_t bytes; /* Number of bytes of response data */ + double timeout; /* Timeout */ + const char *data; /* Data to send */ + static const char *pcl_data = /* PCL data */ + "\033%-12345X@PJL\r\n" + "@PJL JOB NAME = \"Hello, World!\"\r\n" + "@PJL INFO USTATUS\r\n" + "@PJL ENTER LANGUAGE = PCL\r\n" + "\033E" + "Hello, World!\n" + "\014" + "\033%-12345X@PJL\r\n" + "@PJL EOJ NAME=\"Hello, World!\"\r\n" + "\033%-12345X"; + static const char *ps_data = /* PostScript data */ + "%!\n" + "save\n" + "product = flush\n" + "currentpagedevice /PageSize get aload pop\n" + "2 copy gt {exch} if\n" + "(Unknown)\n" + "19 dict\n" + "dup [612 792] (Letter) put\n" + "dup [612 1008] (Legal) put\n" + "dup [612 935] (w612h935) put\n" + "dup [522 756] (Executive) put\n" + "dup [595 842] (A4) put\n" + "dup [420 595] (A5) put\n" + "dup [499 709] (ISOB5) put\n" + "dup [516 728] (B5) put\n" + "dup [612 936] (w612h936) put\n" + "dup [284 419] (Postcard) put\n" + "dup [419.5 567] (DoublePostcard) put\n" + "dup [558 774] (w558h774) put\n" + "dup [553 765] (w553h765) put\n" + "dup [522 737] (w522h737) put\n" + "dup [499 709] (EnvISOB5) put\n" + "dup [297 684] (Env10) put\n" + "dup [459 649] (EnvC5) put\n" + "dup [312 624] (EnvDL) put\n" + "dup [279 540] (EnvMonarch) put\n" + "{ exch aload pop 4 index sub abs 5 le exch\n" + " 5 index sub abs 5 le and\n" + " {exch pop exit} {pop} ifelse\n" + "} bind forall\n" + "= flush pop pop\n" + "/Courier findfont 12 scalefont setfont\n" + "0 setgray 36 720 moveto (Hello, ) show product show (!) show\n" + "showpage\n" + "restore\n" + "\004"; + + + if (do_pcl) + data = pcl_data; + else + data = ps_data; + + write(1, data, strlen(data)); + write(2, "DEBUG: START\n", 13); + timeout = 60.0; + while ((bytes = cupsBackChannelRead(buffer, sizeof(buffer), + timeout)) > 0) + { + write(2, buffer, bytes); + timeout = 5.0; + } + write(2, "\nDEBUG: END\n", 12); + } + + exit(0); + } + else if (data_pid < 0) + { + perror("testbackend: Unable to fork"); + return (1); + } + } + else + data_fds[0] = data_fds[1] = -1; + + /* + * Execute the backend... + */ + + if ((back_pid = fork()) == 0) + { + /* + * Child comes here... + */ + + if (do_trickle || do_ps || do_pcl || do_cancel) + { + if (data_fds[0] != 0) + { + dup2(data_fds[0], 0); + close(data_fds[0]); + } + close(data_fds[1]); + } + + if (!show_log) + { + if ((fd = open("/dev/null", O_WRONLY)) != 2) + { + dup2(fd, 2); + close(fd); + } + } + + if (back_fds[1] != 3) + { + dup2(back_fds[1], 3); + close(back_fds[0]); + } + close(back_fds[1]); + + if (side_fds[1] != 4) + { + dup2(side_fds[1], 4); + close(side_fds[0]); + } + close(side_fds[1]); + + execv(backend, argv + first_arg); + fprintf(stderr, "testbackend: Unable to execute \"%s\": %s\n", backend, + strerror(errno)); + return (errno); + } + else if (back_pid < 0) + { + perror("testbackend: Unable to fork"); + return (1); + } + + /* + * Parent comes here, setup back and side channel file descriptors... + */ + + if (do_trickle || do_ps || do_pcl || do_cancel) + { + close(data_fds[0]); + close(data_fds[1]); + } + + if (back_fds[0] != 3) + { + dup2(back_fds[0], 3); + close(back_fds[0]); + } + close(back_fds[1]); + + if (side_fds[0] != 4) + { + dup2(side_fds[0], 4); + close(side_fds[0]); + } + close(side_fds[1]); + + /* + * Do side-channel tests as needed, then wait for the backend... + */ + + if (do_side_tests) + { + int length; /* Length of buffer */ + char buffer[2049]; /* Buffer for reponse */ + cups_sc_status_t scstatus; /* Status of side-channel command */ + static const char * const statuses[] = + { + "CUPS_SC_STATUS_NONE", /* No status */ + "CUPS_SC_STATUS_OK", /* Operation succeeded */ + "CUPS_SC_STATUS_IO_ERROR", /* An I/O error occurred */ + "CUPS_SC_STATUS_TIMEOUT", /* The backend did not respond */ + "CUPS_SC_STATUS_NO_RESPONSE", /* The device did not respond */ + "CUPS_SC_STATUS_BAD_MESSAGE", /* The command/response message was invalid */ + "CUPS_SC_STATUS_TOO_BIG", /* Response too big */ + "CUPS_SC_STATUS_NOT_IMPLEMENTED" /* Command not implemented */ + }; + + + sleep(2); + + length = 0; + scstatus = cupsSideChannelDoRequest(CUPS_SC_CMD_DRAIN_OUTPUT, buffer, + &length, 60.0); + printf("CUPS_SC_CMD_DRAIN_OUTPUT returned %s\n", statuses[scstatus]); + + length = 1; + scstatus = cupsSideChannelDoRequest(CUPS_SC_CMD_GET_BIDI, buffer, + &length, 5.0); + printf("CUPS_SC_CMD_GET_BIDI returned %s, %d\n", statuses[scstatus], buffer[0]); + + length = sizeof(buffer) - 1; + scstatus = cupsSideChannelDoRequest(CUPS_SC_CMD_GET_DEVICE_ID, buffer, + &length, 5.0); + buffer[length] = '\0'; + printf("CUPS_SC_CMD_GET_DEVICE_ID returned %s, \"%s\"\n", + statuses[scstatus], buffer); + + length = 1; + scstatus = cupsSideChannelDoRequest(CUPS_SC_CMD_GET_STATE, buffer, + &length, 5.0); + printf("CUPS_SC_CMD_GET_STATE returned %s, %02X\n", statuses[scstatus], + buffer[0] & 255); + + if (do_walk) + { + /* + * Walk the OID tree... + */ + + scstatus = cupsSideChannelSNMPWalk(oid, 5.0, walk_cb, NULL); + printf("CUPS_SC_CMD_SNMP_WALK returned %s\n", statuses[scstatus]); + } + else + { + /* + * Lookup the same OID twice... + */ + + length = sizeof(buffer); + scstatus = cupsSideChannelSNMPGet(oid, buffer, &length, 5.0); + printf("CUPS_SC_CMD_SNMP_GET %s returned %s, %s\n", oid, + statuses[scstatus], buffer); + + length = sizeof(buffer); + scstatus = cupsSideChannelSNMPGet(oid, buffer, &length, 5.0); + printf("CUPS_SC_CMD_SNMP_GET %s returned %s, %s\n", oid, + statuses[scstatus], buffer); + } + + length = 0; + scstatus = cupsSideChannelDoRequest(CUPS_SC_CMD_SOFT_RESET, buffer, + &length, 5.0); + printf("CUPS_SC_CMD_SOFT_RESET returned %s\n", statuses[scstatus]); + } + + if (do_cancel) + { + sleep(1); + kill(data_pid, SIGTERM); + kill(back_pid, SIGTERM); + } + + while ((pid = wait(&status)) > 0) + { + if (status) + { + if (WIFEXITED(status)) + printf("%s exited with status %d!\n", + pid == back_pid ? backend : "test", + WEXITSTATUS(status)); + else + printf("%s crashed with signal %d!\n", + pid == back_pid ? backend : "test", + WTERMSIG(status)); + } + } + + /* + * Exit accordingly... + */ + + return (status != 0); +} + + +/* + * 'sigterm_handler()' - Flag when we get SIGTERM. + */ + +static void +sigterm_handler(int sig) /* I - Signal */ +{ + (void)sig; + + job_canceled = 1; +} + + +/* + * 'usage()' - Show usage information. + */ + +static void +usage(void) +{ + puts("Usage: testbackend [-cancel] [-d] [-ps | -pcl] [-s [-oid OID] " + "[-walk OID]] [-t] device-uri job-id user title copies options [file]"); + puts(""); + puts("Options:"); + puts(" -cancel Simulate a canceled print job after 2 seconds."); + puts(" -d Show log messages from backend."); + puts(" -oid OID Lookup the specified SNMP OID."); + puts(" (.1.3.6.1.2.1.43.10.2.1.4.1.1 is a good one for printers)"); + puts(" -pcl Send PCL+PJL query and test page to backend."); + puts(" -ps Send PostScript query and test page to backend."); + puts(" -s Do side-channel + SNMP tests."); + puts(" -t Send spaces slowly to backend ('trickle')."); + puts(" -walk OID Walk the specified SNMP OID."); + puts(" (.1.3.6.1.2.1.43 is a good one for printers)"); + + exit(1); +} + + +/* + * 'walk_cb()' - Show results of cupsSideChannelSNMPWalk... + */ + +static void +walk_cb(const char *oid, /* I - OID */ + const char *data, /* I - Data */ + int datalen, /* I - Length of data */ + void *context) /* I - Context (unused) */ +{ + printf("CUPS_SC_CMD_SNMP_WALK %s=%s (%d bytes)\n", oid, data, datalen); +} + + +/* + * End of "$Id$". + */ diff --git a/backend/testsupplies.c b/backend/testsupplies.c new file mode 100644 index 0000000000..6cbdf8091a --- /dev/null +++ b/backend/testsupplies.c @@ -0,0 +1,83 @@ +/* + * "$Id$" + * + * SNMP supplies test program for CUPS. + * + * Copyright 2008-2011 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Show the supplies state of a printer. + */ + +/* + * Include necessary headers. + */ + +#include "backend-private.h" + + +/* + * 'main()' - Show the supplies state of a printer. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + http_addrlist_t *host; /* Host addresses */ + int snmp_fd; /* SNMP socket */ + int page_count, /* Current page count */ + printer_state; /* Current printer state */ + + + if (argc != 2) + { + puts("Usage: testsupplies ip-or-hostname"); + return (1); + } + + if ((host = httpAddrGetList(argv[1], AF_UNSPEC, "9100")) == NULL) + { + perror(argv[1]); + return (1); + } + + if ((snmp_fd = _cupsSNMPOpen(host->addr.addr.sa_family)) < 0) + { + perror(argv[1]); + return (1); + } + + for (;;) + { + fputs("backendSNMPSupplies: ", stdout); + + if (backendSNMPSupplies(snmp_fd, &(host->addr), &page_count, + &printer_state)) + { + puts("FAIL"); + return (1); + } + + printf("backendSNMPSupplies: %s (page_count=%d, printer_state=%d)\n", + page_count < 0 || printer_state < CUPS_TC_other || + printer_state > CUPS_TC_warmup ? "FAIL" : "PASS", + page_count, printer_state); + + sleep(5); + } +} + + +/* + * End of "$Id$". + */ diff --git a/backend/usb-darwin.c b/backend/usb-darwin.c new file mode 100644 index 0000000000..1de83b209a --- /dev/null +++ b/backend/usb-darwin.c @@ -0,0 +1,2269 @@ +/* +* "$Id$" +* +* Copyright 2005-2011 Apple Inc. All rights reserved. +* +* IMPORTANT: This Apple software is supplied to you by Apple Computer, +* Inc. ("Apple") in consideration of your agreement to the following +* terms, and your use, installation, modification or redistribution of +* this Apple software constitutes acceptance of these terms. If you do +* not agree with these terms, please do not use, install, modify or +* redistribute this Apple software. +* +* In consideration of your agreement to abide by the following terms, and +* subject to these terms, Apple grants you a personal, non-exclusive +* license, under Apple's copyrights in this original Apple software (the +* "Apple Software"), to use, reproduce, modify and redistribute the Apple +* Software, with or without modifications, in source and/or binary forms; +* provided that if you redistribute the Apple Software in its entirety and +* without modifications, you must retain this notice and the following +* text and disclaimers in all such redistributions of the Apple Software. +* Neither the name, trademarks, service marks or logos of Apple Computer, +* Inc. may be used to endorse or promote products derived from the Apple +* Software without specific prior written permission from Apple. Except +* as expressly stated in this notice, no other rights or licenses, express +* or implied, are granted by Apple herein, including but not limited to +* any patent rights that may be infringed by your derivative works or by +* other works in which the Apple Software may be incorporated. +* +* The Apple Software is provided by Apple on an "AS IS" basis. APPLE +* MAKES NO WARRANTIES, EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION +* THE IMPLIED WARRANTIES OF NON-INFRINGEMENT, MERCHANTABILITY AND FITNESS +* FOR A PARTICULAR PURPOSE, REGARDING THE APPLE SOFTWARE OR ITS USE AND +* OPERATION ALONE OR IN COMBINATION WITH YOUR PRODUCTS. +* +* IN NO EVENT SHALL APPLE BE LIABLE FOR ANY SPECIAL, INDIRECT, INCIDENTAL +* OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF +* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS +* INTERRUPTION) ARISING IN ANY WAY OUT OF THE USE, REPRODUCTION, +* MODIFICATION AND/OR DISTRIBUTION OF THE APPLE SOFTWARE, HOWEVER CAUSED +* AND WHETHER UNDER THEORY OF CONTRACT, TORT (INCLUDING NEGLIGENCE), +* STRICT LIABILITY OR OTHERWISE, EVEN IF APPLE HAS BEEN ADVISED OF THE +* POSSIBILITY OF SUCH DAMAGE. +* +* Contents: +* + * list_devices() - List all USB devices. + * print_device() - Print a file to a USB device. + * read_thread() - Thread to read the backchannel data on. + * sidechannel_thread() - Handle side-channel requests. + * iterate_printers() - Iterate over all the printers. + * device_added() - Device added notifier. + * list_device_cb() - list_device iterator callback. + * find_device_cb() - print_device iterator callback. + * status_timer_cb() - Status timer callback. + * copy_deviceinfo() - Copy strings from the 1284 device ID. + * release_deviceinfo() - Release deviceinfo strings. + * load_classdriver() - Load a classdriver. + * unload_classdriver() - Unload a classdriver. + * load_printerdriver() - Load vendor's classdriver. + * registry_open() - Open a connection to the printer. + * registry_close() - Close the connection to the printer. + * copy_deviceid() - Copy the 1284 device id string. + * copy_devicestring() - Copy the 1284 device id string. + * copy_value_for_key() - Copy value string associated with a key. + * cfstr_create_trim() - Create CFString and trim whitespace characters. + * parse_options() - Parse URI options. + * sigterm_handler() - SIGTERM handler. + * next_line() - Find the next line in a buffer. + * parse_pserror() - Scan the backchannel data for postscript errors. + * soft_reset() - Send a soft reset to the device. + * get_device_id() - Return IEEE-1284 device ID. +*/ + +/* + * Include necessary headers. + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include "backend-private.h" +#include +#include +#include + +#include +#include + +extern char **environ; + + +/* + * DEBUG_WRITES, if defined, causes the backend to write data to the printer in + * 512 byte increments, up to 8192 bytes, to make debugging with a USB bus + * analyzer easier. + */ + +#define DEBUG_WRITES 0 + +/* + * WAIT_EOF_DELAY is number of seconds we'll wait for responses from + * the printer after we've finished sending all the data + */ +#define WAIT_EOF_DELAY 7 +#define WAIT_SIDE_DELAY 3 +#define DEFAULT_TIMEOUT 5000L + +#define USB_INTERFACE_KIND CFUUIDGetUUIDBytes(kIOUSBInterfaceInterfaceID190) +#define kUSBLanguageEnglish 0x409 + +#define PRINTER_POLLING_INTERVAL 5 /* seconds */ +#define INITIAL_LOG_INTERVAL PRINTER_POLLING_INTERVAL +#define SUBSEQUENT_LOG_INTERVAL 3 * INITIAL_LOG_INTERVAL + +#define kUSBPrinterClassTypeID CFUUIDGetConstantUUIDWithBytes(NULL, 0x06, 0x04, 0x7D, 0x16, 0x53, 0xA2, 0x11, 0xD6, 0x92, 0x06, 0x00, 0x30, 0x65, 0x52, 0x45, 0x92) +#define kUSBPrinterClassInterfaceID CFUUIDGetConstantUUIDWithBytes(NULL, 0x03, 0x34, 0x6D, 0x74, 0x53, 0xA3, 0x11, 0xD6, 0x9E, 0xA1, 0x76, 0x30, 0x65, 0x52, 0x45, 0x92) + +#define kUSBClassDriverProperty CFSTR("USB Printing Class") + +#define kUSBGenericTOPrinterClassDriver CFSTR("/System/Library/Printers/Libraries/USBGenericPrintingClass.plugin") +#define kUSBPrinterClassDeviceNotOpen -9664 /*kPMInvalidIOMContext*/ + + +/* + * Section 5.3 USB Printing Class spec + */ +#define kUSBPrintingSubclass 1 +#define kUSBPrintingProtocolNoOpen 0 +#define kUSBPrintingProtocolUnidirectional 1 +#define kUSBPrintingProtocolBidirectional 2 + +typedef IOUSBInterfaceInterface190 **printer_interface_t; + +typedef struct iodevice_request_s /**** Device request ****/ +{ + UInt8 requestType; + UInt8 request; + UInt16 value; + UInt16 index; + UInt16 length; + void *buffer; +} iodevice_request_t; + +typedef union /**** Centronics status byte ****/ +{ + char b; + struct + { + unsigned reserved0:2; + unsigned paperError:1; + unsigned select:1; + unsigned notError:1; + unsigned reserved1:3; + } status; +} centronics_status_t; + +typedef struct classdriver_s /**** g.classdriver context ****/ +{ + IUNKNOWN_C_GUTS; + CFPlugInRef plugin; /* release plugin */ + IUnknownVTbl **factory; /* Factory */ + void *vendorReference; /* vendor class specific usage */ + UInt32 location; /* unique location in bus topology */ + UInt8 interfaceNumber; /* Interface number */ + UInt16 vendorID; /* Vendor id */ + UInt16 productID; /* Product id */ + printer_interface_t interface; /* identify the device to IOKit */ + UInt8 outpipe; /* mandatory bulkOut pipe */ + UInt8 inpipe; /* optional bulkIn pipe */ + + /* general class requests */ + kern_return_t (*DeviceRequest)(struct classdriver_s **printer, iodevice_request_t *iorequest, UInt16 timeout); + kern_return_t (*GetString)(struct classdriver_s **printer, UInt8 whichString, UInt16 language, UInt16 timeout, CFStringRef *result); + + /* standard printer class requests */ + kern_return_t (*SoftReset)(struct classdriver_s **printer, UInt16 timeout); + kern_return_t (*GetCentronicsStatus)(struct classdriver_s **printer, centronics_status_t *result, UInt16 timeout); + kern_return_t (*GetDeviceID)(struct classdriver_s **printer, CFStringRef *devid, UInt16 timeout); + + /* standard bulk device requests */ + kern_return_t (*ReadPipe)(struct classdriver_s **printer, UInt8 *buffer, UInt32 *count); + kern_return_t (*WritePipe)(struct classdriver_s **printer, UInt8 *buffer, UInt32 *count, Boolean eoj); + + /* interface requests */ + kern_return_t (*Open)(struct classdriver_s **printer, UInt32 location, UInt8 protocol); + kern_return_t (*Abort)(struct classdriver_s **printer); + kern_return_t (*Close)(struct classdriver_s **printer); + + /* initialize and terminate */ + kern_return_t (*Initialize)(struct classdriver_s **printer, struct classdriver_s **baseclass); + kern_return_t (*Terminate)(struct classdriver_s **printer); + +} classdriver_t; + +typedef Boolean (*iterator_callback_t)(void *refcon, io_service_t obj); + +typedef struct iterator_reference_s /**** Iterator reference data */ +{ + iterator_callback_t callback; + void *userdata; + Boolean keepRunning; +} iterator_reference_t; + +typedef struct globals_s +{ + io_service_t printer_obj; + classdriver_t **classdriver; + + pthread_mutex_t read_thread_mutex; + pthread_cond_t read_thread_cond; + int read_thread_stop; + int read_thread_done; + + pthread_mutex_t readwrite_lock_mutex; + pthread_cond_t readwrite_lock_cond; + int readwrite_lock; + + CFStringRef make; + CFStringRef model; + CFStringRef serial; + UInt32 location; + UInt8 interfaceNum; + + CFRunLoopTimerRef status_timer; + + int print_fd; /* File descriptor to print */ + ssize_t print_bytes; /* Print bytes read */ +#if DEBUG_WRITES + ssize_t debug_bytes; /* Current bytes to read */ +#endif /* DEBUG_WRITES */ + + Boolean wait_eof; + int drain_output; /* Drain all pending output */ + int bidi_flag; /* 0=unidirectional, 1=bidirectional */ + + pthread_mutex_t sidechannel_thread_mutex; + pthread_cond_t sidechannel_thread_cond; + int sidechannel_thread_stop; + int sidechannel_thread_done; +} globals_t; + + +/* + * Globals... + */ + +globals_t g = { 0 }; /* Globals */ + + +/* + * Local functions... + */ + +static Boolean find_device_cb(void *refcon, io_service_t obj); +static Boolean list_device_cb(void *refcon, io_service_t obj); +static CFStringRef cfstr_create_trim(const char *cstr); +static CFStringRef copy_value_for_key(CFStringRef deviceID, CFStringRef *keys); +static kern_return_t load_classdriver(CFStringRef driverPath, printer_interface_t interface, classdriver_t ***printerDriver); +static kern_return_t load_printerdriver(CFStringRef *driverBundlePath); +static kern_return_t registry_close(void); +static kern_return_t registry_open(CFStringRef *driverBundlePath); +static kern_return_t unload_classdriver(classdriver_t ***classdriver); +static OSStatus copy_deviceid(classdriver_t **printer, CFStringRef *deviceID); +static void *read_thread(void *reference); +static void *sidechannel_thread(void *reference); +static void copy_deviceinfo(CFStringRef deviceIDString, CFStringRef *make, CFStringRef *model, CFStringRef *serial); +static void copy_devicestring(io_service_t usbInterface, CFStringRef *deviceID, UInt32 *deviceLocation, UInt8 *interfaceNum); +static void device_added(void *userdata, io_iterator_t iterator); +static void get_device_id(cups_sc_status_t *status, char *data, int *datalen); +static void iterate_printers(iterator_callback_t callBack, void *userdata); +static void parse_options(char *options, char *serial, int serial_size, UInt32 *location, Boolean *wait_eof); +static void release_deviceinfo(CFStringRef *make, CFStringRef *model, CFStringRef *serial); +static void setup_cfLanguage(void); +static void soft_reset(void); +static void status_timer_cb(CFRunLoopTimerRef timer, void *info); + +#if defined(__i386__) || defined(__x86_64__) +static pid_t child_pid; /* Child PID */ +static void run_legacy_backend(int argc, char *argv[], int fd); /* Starts child backend process running as a ppc executable */ +#endif /* __i386__ || __x86_64__ */ +static void sigterm_handler(int sig); /* SIGTERM handler */ + +#ifdef PARSE_PS_ERRORS +static const char *next_line (const char *buffer); +static void parse_pserror (char *sockBuffer, int len); +#endif /* PARSE_PS_ERRORS */ + +#pragma mark - + +/* + * 'list_devices()' - List all USB devices. + */ + +void list_devices() +{ + iterate_printers(list_device_cb, NULL); +} + + +/* + * 'print_device()' - Print a file to a USB device. + */ + +int /* O - Exit status */ +print_device(const char *uri, /* I - Device URI */ + const char *hostname, /* I - Hostname/manufacturer */ + const char *resource, /* I - Resource/modelname */ + char *options, /* I - Device options/serial number */ + int print_fd, /* I - File descriptor to print */ + int copies, /* I - Copies to print */ + int argc, /* I - Number of command-line arguments (6 or 7) */ + char *argv[]) /* I - Command-line arguments */ +{ + char serial[1024]; /* Serial number buffer */ + OSStatus status; /* Function results */ + IOReturn iostatus; /* Current IO status */ + pthread_t read_thread_id, /* Read thread */ + sidechannel_thread_id;/* Side-channel thread */ + int have_sidechannel = 0; /* Was the side-channel thread started? */ + struct stat sidechannel_info; /* Side-channel file descriptor info */ + char print_buffer[8192], /* Print data buffer */ + *print_ptr; /* Pointer into print data buffer */ + UInt32 location; /* Unique location in bus topology */ + fd_set input_set; /* Input set for select() */ + CFStringRef driverBundlePath; /* Class driver path */ + int countdown, /* Logging interval */ + nfds; /* Number of file descriptors */ + ssize_t total_bytes; /* Total bytes written */ + UInt32 bytes; /* Bytes written */ + struct timeval *timeout, /* Timeout pointer */ + tv; /* Time value */ + struct timespec cond_timeout; /* pthread condition timeout */ + + + (void)uri; + + /* + * See if the side-channel descriptor is valid... + */ + + have_sidechannel = !fstat(CUPS_SC_FD, &sidechannel_info) && + S_ISSOCK(sidechannel_info.st_mode); + + /* + * Localize using CoreFoundation... + */ + + setup_cfLanguage(); + + parse_options(options, serial, sizeof(serial), &location, &g.wait_eof); + + if (resource[0] == '/') + resource++; + + g.print_fd = print_fd; + g.make = cfstr_create_trim(hostname); + g.model = cfstr_create_trim(resource); + g.serial = cfstr_create_trim(serial); + g.location = location; + + if (!g.make || !g.model) + { + fprintf(stderr, "DEBUG: Fatal USB error.\n"); + _cupsLangPrintFilter(stderr, "ERROR", + _("There was an unrecoverable USB error.")); + + if (!g.make) + fputs("DEBUG: USB make string is NULL\n", stderr); + if (!g.model) + fputs("DEBUG: USB model string is NULL\n", stderr); + + return (CUPS_BACKEND_STOP); + } + + fputs("STATE: +connecting-to-device\n", stderr); + + countdown = INITIAL_LOG_INTERVAL; + + do + { + if (g.printer_obj) + { + IOObjectRelease(g.printer_obj); + unload_classdriver(&g.classdriver); + g.printer_obj = 0x0; + g.classdriver = 0x0; + } + + fprintf(stderr, "DEBUG: Looking for '%s %s'\n", hostname, resource); + + iterate_printers(find_device_cb, NULL); + + fputs("DEBUG: Opening connection\n", stderr); + + driverBundlePath = NULL; + + status = registry_open(&driverBundlePath); + +#if defined(__i386__) || defined(__x86_64__) + /* + * If we were unable to load the class drivers for this printer it's + * probably because they're ppc or i386. In this case try to run this + * backend as i386 or ppc executables so we can use them... + */ + if (status == -2) + { + run_legacy_backend(argc, argv, print_fd); + /* Never returns here */ + } +#endif /* __i386__ || __x86_64__ */ + + if (status == -2) + { + /* + * If we still were unable to load the class drivers for this printer log + * the error and stop the queue... + */ + + if (driverBundlePath == NULL || !CFStringGetCString(driverBundlePath, print_buffer, sizeof(print_buffer), kCFStringEncodingUTF8)) + strlcpy(print_buffer, "USB class driver", sizeof(print_buffer)); + + fputs("STATE: +apple-missing-usbclassdriver-error\n", stderr); + _cupsLangPrintFilter(stderr, "ERROR", + _("There was an unrecoverable USB error.")); + fprintf(stderr, "DEBUG: Could not load %s\n", print_buffer); + + if (driverBundlePath) + CFRelease(driverBundlePath); + + return (CUPS_BACKEND_STOP); + } + + if (driverBundlePath) + CFRelease(driverBundlePath); + + if (status != noErr) + { + sleep(PRINTER_POLLING_INTERVAL); + countdown -= PRINTER_POLLING_INTERVAL; + if (countdown <= 0) + { + _cupsLangPrintFilter(stderr, "INFO", + _("Waiting for printer to become available.")); + fprintf(stderr, "DEBUG: USB printer status: 0x%08x\n", (int)status); + countdown = SUBSEQUENT_LOG_INTERVAL; /* subsequent log entries, every 15 seconds */ + } + } + } while (status != noErr); + + fputs("STATE: -connecting-to-device\n", stderr); + + /* + * Now that we are "connected" to the port, ignore SIGTERM so that we + * can finish out any page data the driver sends (e.g. to eject the + * current page... Only ignore SIGTERM if we are printing data from + * stdin (otherwise you can't cancel raw jobs...) + */ + + if (!print_fd) + { + struct sigaction action; /* POSIX signal action */ + + + memset(&action, 0, sizeof(action)); + + sigemptyset(&action.sa_mask); + action.sa_handler = SIG_IGN; + sigaction(SIGTERM, &action, NULL); + } + + /* + * Start the side channel thread if the descriptor is valid... + */ + + pthread_mutex_init(&g.readwrite_lock_mutex, NULL); + pthread_cond_init(&g.readwrite_lock_cond, NULL); + g.readwrite_lock = 1; + + if (have_sidechannel) + { + g.sidechannel_thread_stop = 0; + g.sidechannel_thread_done = 0; + + pthread_cond_init(&g.sidechannel_thread_cond, NULL); + pthread_mutex_init(&g.sidechannel_thread_mutex, NULL); + + if (pthread_create(&sidechannel_thread_id, NULL, sidechannel_thread, NULL)) + { + fprintf(stderr, "DEBUG: Fatal USB error.\n"); + _cupsLangPrintFilter(stderr, "ERROR", + _("There was an unrecoverable USB error.")); + fputs("DEBUG: Couldn't create side-channel thread\n", stderr); + registry_close(); + return (CUPS_BACKEND_STOP); + } + } + + /* + * Get the read thread going... + */ + + g.read_thread_stop = 0; + g.read_thread_done = 0; + + pthread_cond_init(&g.read_thread_cond, NULL); + pthread_mutex_init(&g.read_thread_mutex, NULL); + + if (pthread_create(&read_thread_id, NULL, read_thread, NULL)) + { + fprintf(stderr, "DEBUG: Fatal USB error.\n"); + _cupsLangPrintFilter(stderr, "ERROR", + _("There was an unrecoverable USB error.")); + fputs("DEBUG: Couldn't create read thread\n", stderr); + registry_close(); + return (CUPS_BACKEND_STOP); + } + + /* + * The main thread sends the print file... + */ + + g.drain_output = 0; + g.print_bytes = 0; + total_bytes = 0; + print_ptr = print_buffer; + + while (status == noErr && copies-- > 0) + { + _cupsLangPrintFilter(stderr, "INFO", _("Sending data to printer.")); + + if (print_fd != STDIN_FILENO) + { + fputs("PAGE: 1 1\n", stderr); + lseek(print_fd, 0, SEEK_SET); + } + + while (status == noErr) + { + FD_ZERO(&input_set); + + if (!g.print_bytes) + FD_SET(print_fd, &input_set); + + /* + * Calculate select timeout... + * If we have data waiting to send timeout is 100ms. + * else if we're draining print_fd timeout is 0. + * else we're waiting forever... + */ + + if (g.print_bytes) + { + tv.tv_sec = 0; + tv.tv_usec = 100000; /* 100ms */ + timeout = &tv; + } + else if (g.drain_output) + { + tv.tv_sec = 0; + tv.tv_usec = 0; + timeout = &tv; + } + else + timeout = NULL; + + /* + * I/O is unlocked around select... + */ + + pthread_mutex_lock(&g.readwrite_lock_mutex); + g.readwrite_lock = 0; + pthread_cond_signal(&g.readwrite_lock_cond); + pthread_mutex_unlock(&g.readwrite_lock_mutex); + + nfds = select(print_fd + 1, &input_set, NULL, NULL, timeout); + + /* + * Reacquire the lock... + */ + + pthread_mutex_lock(&g.readwrite_lock_mutex); + while (g.readwrite_lock) + pthread_cond_wait(&g.readwrite_lock_cond, &g.readwrite_lock_mutex); + g.readwrite_lock = 1; + pthread_mutex_unlock(&g.readwrite_lock_mutex); + + if (nfds < 0) + { + if (errno == EINTR && total_bytes == 0) + { + fputs("DEBUG: Received an interrupt before any bytes were " + "written, aborting\n", stderr); + registry_close(); + return (CUPS_BACKEND_OK); + } + else if (errno != EAGAIN && errno != EINTR) + { + _cupsLangPrintFilter(stderr, "ERROR", + _("Unable to read print data.")); + perror("DEBUG: select"); + registry_close(); + return (CUPS_BACKEND_FAILED); + } + } + + /* + * If drain output has finished send a response... + */ + + if (g.drain_output && !nfds && !g.print_bytes) + { + /* Send a response... */ + cupsSideChannelWrite(CUPS_SC_CMD_DRAIN_OUTPUT, CUPS_SC_STATUS_OK, NULL, 0, 1.0); + g.drain_output = 0; + } + + /* + * Check if we have print data ready... + */ + + if (FD_ISSET(print_fd, &input_set)) + { +#if DEBUG_WRITES + g.debug_bytes += 512; + if (g.debug_bytes > sizeof(print_buffer)) + g.debug_bytes = 512; + + g.print_bytes = read(print_fd, print_buffer, g.debug_bytes); + +#else + g.print_bytes = read(print_fd, print_buffer, sizeof(print_buffer)); +#endif /* DEBUG_WRITES */ + + if (g.print_bytes < 0) + { + /* + * Read error - bail if we don't see EAGAIN or EINTR... + */ + + if (errno != EAGAIN && errno != EINTR) + { + _cupsLangPrintFilter(stderr, "ERROR", + _("Unable to read print data.")); + perror("DEBUG: read"); + registry_close(); + return (CUPS_BACKEND_FAILED); + } + + g.print_bytes = 0; + } + else if (g.print_bytes == 0) + { + /* + * End of file, break out of the loop... + */ + + break; + } + + print_ptr = print_buffer; + + fprintf(stderr, "DEBUG: Read %d bytes of print data...\n", + (int)g.print_bytes); + } + + if (g.print_bytes) + { + bytes = g.print_bytes; + iostatus = (*g.classdriver)->WritePipe(g.classdriver, (UInt8*)print_ptr, &bytes, 0); + + /* + * Ignore timeout errors, but retain the number of bytes written to + * avoid sending duplicate data ()... + */ + + if (iostatus == kIOUSBTransactionTimeout) + { + fputs("DEBUG: Got USB transaction timeout during write\n", stderr); + iostatus = 0; + } + + /* + * If we've stalled, retry the write... + */ + + else if (iostatus == kIOUSBPipeStalled) + { + fputs("DEBUG: Got USB pipe stalled during write\n", stderr); + + bytes = g.print_bytes; + iostatus = (*g.classdriver)->WritePipe(g.classdriver, (UInt8*)print_ptr, &bytes, 0); + } + + /* + * Retry a write after an aborted write since we probably just got + * SIGTERM ()... + */ + + else if (iostatus == kIOReturnAborted) + { + fputs("DEBUG: Got USB return aborted during write\n", stderr); + + IOReturn err = (*g.classdriver)->Abort(g.classdriver); + fprintf(stderr, "DEBUG: USB class driver Abort returned %x\n", err); + +#if DEBUG_WRITES + sleep(5); +#endif /* DEBUG_WRITES */ + + bytes = g.print_bytes; + iostatus = (*g.classdriver)->WritePipe(g.classdriver, (UInt8*)print_ptr, &bytes, 0); + } + + if (iostatus) + { + /* + * Write error - bail if we don't see an error we can retry... + */ + + _cupsLangPrintFilter(stderr, "ERROR", + _("Unable to send data to printer.")); + fprintf(stderr, "DEBUG: USB class driver WritePipe returned %x\n", + iostatus); + + IOReturn err = (*g.classdriver)->Abort(g.classdriver); + fprintf(stderr, "DEBUG: USB class driver Abort returned %x\n", + err); + + status = CUPS_BACKEND_FAILED; + break; + } + else if (bytes > 0) + { + fprintf(stderr, "DEBUG: Wrote %d bytes of print data...\n", (int)bytes); + + g.print_bytes -= bytes; + print_ptr += bytes; + total_bytes += bytes; + } + } + + if (print_fd != 0 && status == noErr) + fprintf(stderr, "DEBUG: Sending print file, %lld bytes...\n", + (off_t)total_bytes); + } + } + + fprintf(stderr, "DEBUG: Sent %lld bytes...\n", (off_t)total_bytes); + + /* + * Signal the side channel thread to exit... + */ + + if (have_sidechannel) + { + close(CUPS_SC_FD); + pthread_mutex_lock(&g.readwrite_lock_mutex); + g.readwrite_lock = 0; + pthread_cond_signal(&g.readwrite_lock_cond); + pthread_mutex_unlock(&g.readwrite_lock_mutex); + + g.sidechannel_thread_stop = 1; + pthread_mutex_lock(&g.sidechannel_thread_mutex); + + if (!g.sidechannel_thread_done) + { + gettimeofday(&tv, NULL); + cond_timeout.tv_sec = tv.tv_sec + WAIT_SIDE_DELAY; + cond_timeout.tv_nsec = tv.tv_usec * 1000; + + while (!g.sidechannel_thread_done) + { + if (pthread_cond_timedwait(&g.sidechannel_thread_cond, + &g.sidechannel_thread_mutex, + &cond_timeout) != 0) + break; + } + } + + pthread_mutex_unlock(&g.sidechannel_thread_mutex); + } + + /* + * Signal the read thread to exit then wait 7 seconds for it to complete... + */ + + g.read_thread_stop = 1; + + pthread_mutex_lock(&g.read_thread_mutex); + + if (!g.read_thread_done) + { + fputs("DEBUG: Waiting for read thread to exit...\n", stderr); + + gettimeofday(&tv, NULL); + cond_timeout.tv_sec = tv.tv_sec + WAIT_EOF_DELAY; + cond_timeout.tv_nsec = tv.tv_usec * 1000; + + while (!g.read_thread_done) + { + if (pthread_cond_timedwait(&g.read_thread_cond, &g.read_thread_mutex, + &cond_timeout) != 0) + break; + } + + /* + * If it didn't exit abort the pending read and wait an additional second... + */ + + if (!g.read_thread_done) + { + fputs("DEBUG: Read thread still active, aborting the pending read...\n", + stderr); + + g.wait_eof = 0; + + (*g.classdriver)->Abort(g.classdriver); + + gettimeofday(&tv, NULL); + cond_timeout.tv_sec = tv.tv_sec + 1; + cond_timeout.tv_nsec = tv.tv_usec * 1000; + + while (!g.read_thread_done) + { + if (pthread_cond_timedwait(&g.read_thread_cond, &g.read_thread_mutex, + &cond_timeout) != 0) + break; + } + } + } + + pthread_mutex_unlock(&g.read_thread_mutex); + + /* + * Close the connection and input file and general clean up... + */ + + registry_close(); + + if (print_fd != STDIN_FILENO) + close(print_fd); + + if (g.make != NULL) + CFRelease(g.make); + + if (g.model != NULL) + CFRelease(g.model); + + if (g.serial != NULL) + CFRelease(g.serial); + + if (g.printer_obj != 0x0) + IOObjectRelease(g.printer_obj); + + return status; +} + + +/* + * 'read_thread()' - Thread to read the backchannel data on. + */ + +static void *read_thread(void *reference) +{ + UInt8 readbuffer[512]; + UInt32 rbytes; + kern_return_t readstatus; + struct mach_timebase_info timeBaseInfo; + uint64_t start, + delay; + + + (void)reference; + + /* Calculate what 250 milliSeconds are in mach absolute time... + */ + mach_timebase_info(&timeBaseInfo); + delay = ((uint64_t)250000000 * (uint64_t)timeBaseInfo.denom) / (uint64_t)timeBaseInfo.numer; + + do + { + /* + * Remember when we started so we can throttle the loop after the read call... + */ + + start = mach_absolute_time(); + + rbytes = sizeof(readbuffer); + readstatus = (*g.classdriver)->ReadPipe(g.classdriver, readbuffer, &rbytes); + if (readstatus == kIOReturnSuccess && rbytes > 0) + { + fprintf(stderr, "DEBUG: Read %d bytes of back-channel data...\n", + (int)rbytes); + cupsBackChannelWrite((char*)readbuffer, rbytes, 1.0); + + /* cntrl-d is echoed by the printer. + * NOTES: + * Xerox Phaser 6250D doesn't echo the cntrl-d. + * Xerox Phaser 6250D doesn't always send the product query. + */ + if (g.wait_eof && readbuffer[rbytes-1] == 0x4) + break; + +#ifdef PARSE_PS_ERRORS + parse_pserror(readbuffer, rbytes); +#endif + } + else if (readstatus == kIOUSBTransactionTimeout) + fputs("DEBUG: Got USB transaction timeout during read\n", stderr); + else if (readstatus == kIOUSBPipeStalled) + fputs("DEBUG: Got USB pipe stalled during read\n", stderr); + else if (readstatus == kIOReturnAborted) + fputs("DEBUG: Got USB return aborted during read\n", stderr); + + /* + * Make sure this loop executes no more than once every 250 miliseconds... + */ + + if ((readstatus != kIOReturnSuccess || rbytes == 0) && (g.wait_eof || !g.read_thread_stop)) + mach_wait_until(start + delay); + + } while (g.wait_eof || !g.read_thread_stop); /* Abort from main thread tests error here */ + + /* + * Let the main thread know that we have completed the read thread... + */ + + pthread_mutex_lock(&g.read_thread_mutex); + g.read_thread_done = 1; + pthread_cond_signal(&g.read_thread_cond); + pthread_mutex_unlock(&g.read_thread_mutex); + + return NULL; +} + + +/* + * 'sidechannel_thread()' - Handle side-channel requests. + */ + +static void* +sidechannel_thread(void *reference) +{ + cups_sc_command_t command; /* Request command */ + cups_sc_status_t status; /* Request/response status */ + char data[2048]; /* Request/response data */ + int datalen; /* Request/response data size */ + + + (void)reference; + + do + { + datalen = sizeof(data); + + if (cupsSideChannelRead(&command, &status, data, &datalen, 1.0)) + { + if (status == CUPS_SC_STATUS_TIMEOUT) + continue; + else + break; + } + + switch (command) + { + case CUPS_SC_CMD_SOFT_RESET: /* Do a soft reset */ + fputs("DEBUG: CUPS_SC_CMD_SOFT_RESET received from driver...\n", + stderr); + + if ((*g.classdriver)->SoftReset != NULL) + { + soft_reset(); + cupsSideChannelWrite(command, CUPS_SC_STATUS_OK, NULL, 0, 1.0); + fputs("DEBUG: Returning status CUPS_STATUS_OK with no bytes...\n", + stderr); + } + else + { + cupsSideChannelWrite(command, CUPS_SC_STATUS_NOT_IMPLEMENTED, + NULL, 0, 1.0); + fputs("DEBUG: Returning status CUPS_STATUS_NOT_IMPLEMENTED with " + "no bytes...\n", stderr); + } + break; + + case CUPS_SC_CMD_DRAIN_OUTPUT: /* Drain all pending output */ + fputs("DEBUG: CUPS_SC_CMD_DRAIN_OUTPUT received from driver...\n", + stderr); + + g.drain_output = 1; + break; + + case CUPS_SC_CMD_GET_BIDI: /* Is the connection bidirectional? */ + fputs("DEBUG: CUPS_SC_CMD_GET_BIDI received from driver...\n", + stderr); + + data[0] = g.bidi_flag; + cupsSideChannelWrite(command, CUPS_SC_STATUS_OK, data, 1, 1.0); + + fprintf(stderr, + "DEBUG: Returned CUPS_SC_STATUS_OK with 1 byte (%02X)...\n", + data[0]); + break; + + case CUPS_SC_CMD_GET_DEVICE_ID: /* Return IEEE-1284 device ID */ + fputs("DEBUG: CUPS_SC_CMD_GET_DEVICE_ID received from driver...\n", + stderr); + + datalen = sizeof(data); + get_device_id(&status, data, &datalen); + cupsSideChannelWrite(command, CUPS_SC_STATUS_OK, data, datalen, 1.0); + + if (datalen < sizeof(data)) + data[datalen] = '\0'; + else + data[sizeof(data) - 1] = '\0'; + + fprintf(stderr, + "DEBUG: Returning CUPS_SC_STATUS_OK with %d bytes (%s)...\n", + datalen, data); + break; + + case CUPS_SC_CMD_GET_STATE: /* Return device state */ + fputs("DEBUG: CUPS_SC_CMD_GET_STATE received from driver...\n", + stderr); + + data[0] = CUPS_SC_STATE_ONLINE; + cupsSideChannelWrite(command, CUPS_SC_STATUS_OK, data, 1, 1.0); + + fprintf(stderr, + "DEBUG: Returned CUPS_SC_STATUS_OK with 1 byte (%02X)...\n", + data[0]); + break; + + default: + fprintf(stderr, "DEBUG: Unknown side-channel command (%d) received " + "from driver...\n", command); + + cupsSideChannelWrite(command, CUPS_SC_STATUS_NOT_IMPLEMENTED, + NULL, 0, 1.0); + + fputs("DEBUG: Returned CUPS_SC_STATUS_NOT_IMPLEMENTED with no bytes...\n", + stderr); + break; + } + } + while (!g.sidechannel_thread_stop); + + pthread_mutex_lock(&g.sidechannel_thread_mutex); + g.sidechannel_thread_done = 1; + pthread_cond_signal(&g.sidechannel_thread_cond); + pthread_mutex_unlock(&g.sidechannel_thread_mutex); + + return NULL; +} + + +#pragma mark - +/* + * 'iterate_printers()' - Iterate over all the printers. + */ + +static void iterate_printers(iterator_callback_t callBack, + void *userdata) +{ + mach_port_t masterPort = 0x0; + kern_return_t kr = IOMasterPort (bootstrap_port, &masterPort); + + if (kr == kIOReturnSuccess && masterPort != 0x0) + { + io_iterator_t addIterator = 0x0; + + iterator_reference_t reference = { callBack, userdata, true }; + IONotificationPortRef addNotification = IONotificationPortCreate(masterPort); + + int klass = kUSBPrintingClass; + int subklass = kUSBPrintingSubclass; + + CFNumberRef usb_klass = CFNumberCreate(NULL, kCFNumberIntType, &klass); + CFNumberRef usb_subklass = CFNumberCreate(NULL, kCFNumberIntType, &subklass); + CFMutableDictionaryRef usbPrinterMatchDictionary = IOServiceMatching(kIOUSBInterfaceClassName); + + CFDictionaryAddValue(usbPrinterMatchDictionary, CFSTR("bInterfaceClass"), usb_klass); + CFDictionaryAddValue(usbPrinterMatchDictionary, CFSTR("bInterfaceSubClass"), usb_subklass); + + CFRelease(usb_klass); + CFRelease(usb_subklass); + + kr = IOServiceAddMatchingNotification(addNotification, kIOMatchedNotification, usbPrinterMatchDictionary, &device_added, &reference, &addIterator); + if (addIterator != 0x0) + { + device_added (&reference, addIterator); + + if (reference.keepRunning) + { + CFRunLoopAddSource(CFRunLoopGetCurrent(), IONotificationPortGetRunLoopSource(addNotification), kCFRunLoopDefaultMode); + CFRunLoopRun(); + } + IOObjectRelease(addIterator); + } + mach_port_deallocate(mach_task_self(), masterPort); + } +} + + +/* + * 'device_added()' - Device added notifier. + */ + +static void device_added(void *userdata, + io_iterator_t iterator) +{ + iterator_reference_t *reference = userdata; + + io_service_t obj; + while (reference->keepRunning && (obj = IOIteratorNext(iterator)) != 0x0) + { + if (reference->callback != NULL) + reference->keepRunning = reference->callback(reference->userdata, obj); + + IOObjectRelease(obj); + } + + /* One last call to the call back now that we are not longer have printers left to iterate... + */ + if (reference->keepRunning) + reference->keepRunning = reference->callback(reference->userdata, 0x0); + + if (!reference->keepRunning) + CFRunLoopStop(CFRunLoopGetCurrent()); +} + + +/* + * 'list_device_cb()' - list_device iterator callback. + */ + +static Boolean list_device_cb(void *refcon, + io_service_t obj) +{ + Boolean keepRunning = (obj != 0x0); + + + (void)refcon; + + if (keepRunning) + { + CFStringRef deviceIDString = NULL; + UInt32 deviceLocation = 0; + UInt8 interfaceNum = 0; + + copy_devicestring(obj, &deviceIDString, &deviceLocation, &interfaceNum); + if (deviceIDString != NULL) + { + CFStringRef make = NULL, model = NULL, serial = NULL; + char uristr[1024], makestr[1024], modelstr[1024], serialstr[1024]; + char optionsstr[1024], idstr[1024], make_modelstr[1024]; + + copy_deviceinfo(deviceIDString, &make, &model, &serial); + CFStringGetCString(deviceIDString, idstr, sizeof(idstr), + kCFStringEncodingUTF8); + backendGetMakeModel(idstr, make_modelstr, sizeof(make_modelstr)); + + modelstr[0] = '/'; + + if (!make || + !CFStringGetCString(make, makestr, sizeof(makestr), + kCFStringEncodingUTF8)) + strcpy(makestr, "Unknown"); + + if (!model || + !CFStringGetCString(model, &modelstr[1], sizeof(modelstr)-1, + kCFStringEncodingUTF8)) + strcpy(modelstr + 1, "Printer"); + + optionsstr[0] = '\0'; + if (serial != NULL) + { + CFStringGetCString(serial, serialstr, sizeof(serialstr), kCFStringEncodingUTF8); + snprintf(optionsstr, sizeof(optionsstr), "?serial=%s", serialstr); + } + else if (deviceLocation != 0) + snprintf(optionsstr, sizeof(optionsstr), "?location=%x", (unsigned)deviceLocation); + + httpAssembleURI(HTTP_URI_CODING_ALL, uristr, sizeof(uristr), "usb", NULL, makestr, 0, modelstr); + strlcat(uristr, optionsstr, sizeof(uristr)); + + cupsBackendReport("direct", uristr, make_modelstr, make_modelstr, idstr, + NULL); + + release_deviceinfo(&make, &model, &serial); + CFRelease(deviceIDString); + } + } + + return keepRunning; +} + + +/* + * 'find_device_cb()' - print_device iterator callback. + */ + +static Boolean find_device_cb(void *refcon, + io_service_t obj) +{ + Boolean keepLooking = true; + + if (obj != 0x0) + { + CFStringRef idString = NULL; + UInt32 location = -1; + UInt8 interfaceNum = 0; + + copy_devicestring(obj, &idString, &location, &interfaceNum); + if (idString != NULL) + { + CFStringRef make = NULL, model = NULL, serial = NULL; + + copy_deviceinfo(idString, &make, &model, &serial); + if (make && CFStringCompare(make, g.make, kCFCompareCaseInsensitive) == kCFCompareEqualTo) + { + if (model && CFStringCompare(model, g.model, kCFCompareCaseInsensitive) == kCFCompareEqualTo) + { + if (g.serial != NULL && CFStringGetLength(g.serial) > 0) + { + if (serial != NULL && CFStringCompare(serial, g.serial, kCFCompareCaseInsensitive) == kCFCompareEqualTo) + { + IOObjectRetain(obj); + g.printer_obj = obj; + keepLooking = false; + } + } + else + { + if (g.printer_obj != 0) + IOObjectRelease(g.printer_obj); + + g.printer_obj = obj; + IOObjectRetain(obj); + + if (g.location == 0 || g.location == location) + keepLooking = false; + } + if ( !keepLooking ) + g.interfaceNum = interfaceNum; + } + } + + release_deviceinfo(&make, &model, &serial); + CFRelease(idString); + } + } + else + { + keepLooking = (g.printer_obj == 0); + if (obj == 0x0 && keepLooking) + { + CFRunLoopTimerContext context = { 0, refcon, NULL, NULL, NULL }; + CFRunLoopTimerRef timer = CFRunLoopTimerCreate(NULL, CFAbsoluteTimeGetCurrent() + 1.0, 10, 0x0, 0x0, status_timer_cb, &context); + if (timer != NULL) + { + CFRunLoopAddTimer(CFRunLoopGetCurrent(), timer, kCFRunLoopDefaultMode); + g.status_timer = timer; + } + } + } + + if (!keepLooking && g.status_timer != NULL) + { + fputs("STATE: -offline-report\n", stderr); + _cupsLangPrintFilter(stderr, "INFO", _("Printer is now online.")); + CFRunLoopRemoveTimer(CFRunLoopGetCurrent(), g.status_timer, kCFRunLoopDefaultMode); + CFRelease(g.status_timer); + g.status_timer = NULL; + } + + return keepLooking; +} + + +/* + * 'status_timer_cb()' - Status timer callback. + */ + +static void status_timer_cb(CFRunLoopTimerRef timer, + void *info) +{ + (void)timer; + (void)info; + + fputs("STATE: +offline-report\n", stderr); + _cupsLangPrintFilter(stderr, "INFO", _("Printer is offline.")); + + if (getenv("CLASS") != NULL) + { + /* + * If the CLASS environment variable is set, the job was submitted + * to a class and not to a specific queue. In this case, we want + * to abort immediately so that the job can be requeued on the next + * available printer in the class. + * + * Sleep 5 seconds to keep the job from requeuing too rapidly... + */ + + sleep(5); + + exit(CUPS_BACKEND_FAILED); + } +} + + +#pragma mark - +/* + * 'copy_deviceinfo()' - Copy strings from the 1284 device ID. + */ + +static void copy_deviceinfo(CFStringRef deviceIDString, + CFStringRef *make, + CFStringRef *model, + CFStringRef *serial) +{ + CFStringRef modelKeys[] = { CFSTR("MDL:"), CFSTR("MODEL:"), NULL }; + CFStringRef makeKeys[] = { CFSTR("MFG:"), CFSTR("MANUFACTURER:"), NULL }; + CFStringRef serialKeys[] = { CFSTR("SN:"), CFSTR("SERN:"), NULL }; + + if (make != NULL) + *make = copy_value_for_key(deviceIDString, makeKeys); + + if (model != NULL) + *model = copy_value_for_key(deviceIDString, modelKeys); + + if (serial != NULL) + *serial = copy_value_for_key(deviceIDString, serialKeys); +} + + +/* + * 'release_deviceinfo()' - Release deviceinfo strings. + */ + +static void release_deviceinfo(CFStringRef *make, + CFStringRef *model, + CFStringRef *serial) +{ + if (make != NULL && *make != NULL) + { + CFRelease(*make); + *make = NULL; + } + + if (model != NULL && *model != NULL) + { + CFRelease(*model); + *model = NULL; + } + + if (serial != NULL && *serial != NULL) + { + CFRelease(*serial); + *serial = NULL; + } +} + + +#pragma mark - +/* + * 'load_classdriver()' - Load a classdriver. + */ + +static kern_return_t load_classdriver(CFStringRef driverPath, + printer_interface_t interface, + classdriver_t ***printerDriver) +{ + kern_return_t kr = kUSBPrinterClassDeviceNotOpen; + classdriver_t **driver = NULL; + CFStringRef bundle = driverPath ? driverPath : kUSBGenericTOPrinterClassDriver; + char bundlestr[1024]; /* Bundle path */ + CFURLRef url; /* URL for driver */ + CFPlugInRef plugin = NULL; /* Plug-in address */ + + + CFStringGetCString(bundle, bundlestr, sizeof(bundlestr), kCFStringEncodingUTF8); + + /* + * Validate permissions for the class driver... + */ + + _cups_fc_result_t result = _cupsFileCheck(bundlestr, + _CUPS_FILE_CHECK_DIRECTORY, 1, + _cupsFileCheckFilter, NULL); + + if (result && driverPath) + return (load_classdriver(NULL, interface, printerDriver)); + else if (result) + return (kr); + + /* + * Try loading the class driver... + */ + + url = CFURLCreateWithFileSystemPath(NULL, bundle, kCFURLPOSIXPathStyle, true); + + if (url) + { + plugin = CFPlugInCreate(NULL, url); + CFRelease(url); + } + else + plugin = NULL; + + if (plugin) + { + CFArrayRef factories = CFPlugInFindFactoriesForPlugInTypeInPlugIn(kUSBPrinterClassTypeID, plugin); + if (factories != NULL && CFArrayGetCount(factories) > 0) + { + CFUUIDRef factoryID = CFArrayGetValueAtIndex(factories, 0); + IUnknownVTbl **iunknown = CFPlugInInstanceCreate(NULL, factoryID, kUSBPrinterClassTypeID); + if (iunknown != NULL) + { + kr = (*iunknown)->QueryInterface(iunknown, CFUUIDGetUUIDBytes(kUSBPrinterClassInterfaceID), (LPVOID *)&driver); + if (kr == kIOReturnSuccess && driver != NULL) + { + classdriver_t **genericDriver = NULL; + if (driverPath != NULL && CFStringCompare(driverPath, kUSBGenericTOPrinterClassDriver, 0) != kCFCompareEqualTo) + kr = load_classdriver(NULL, interface, &genericDriver); + + if (kr == kIOReturnSuccess) + { + (*driver)->interface = interface; + (*driver)->Initialize(driver, genericDriver); + + (*driver)->plugin = plugin; + (*driver)->interface = interface; + *printerDriver = driver; + } + } + (*iunknown)->Release(iunknown); + } + CFRelease(factories); + } + } + + fprintf(stderr, "DEBUG: load_classdriver(%s) (kr:0x%08x)\n", bundlestr, (int)kr); + + return (kr); +} + + +/* + * 'unload_classdriver()' - Unload a classdriver. + */ + +static kern_return_t unload_classdriver(classdriver_t ***classdriver) +{ + if (*classdriver != NULL) + { + (**classdriver)->Release(*classdriver); + *classdriver = NULL; + } + + return kIOReturnSuccess; +} + + +/* + * 'load_printerdriver()' - Load vendor's classdriver. + * + * If driverBundlePath is not NULL on return it is the callers responsbility to release it! + */ + +static kern_return_t load_printerdriver(CFStringRef *driverBundlePath) +{ + IOCFPlugInInterface **iodev = NULL; + SInt32 score; + kern_return_t kr; + printer_interface_t interface; + HRESULT res; + + kr = IOCreatePlugInInterfaceForService(g.printer_obj, kIOUSBInterfaceUserClientTypeID, kIOCFPlugInInterfaceID, &iodev, &score); + if (kr == kIOReturnSuccess) + { + if ((res = (*iodev)->QueryInterface(iodev, USB_INTERFACE_KIND, (LPVOID *) &interface)) == noErr) + { + *driverBundlePath = IORegistryEntryCreateCFProperty(g.printer_obj, kUSBClassDriverProperty, NULL, kNilOptions); + + kr = load_classdriver(*driverBundlePath, interface, &g.classdriver); + + if (kr != kIOReturnSuccess) + (*interface)->Release(interface); + } + IODestroyPlugInInterface(iodev); + } + return kr; +} + + +/* + * 'registry_open()' - Open a connection to the printer. + */ + +static kern_return_t registry_open(CFStringRef *driverBundlePath) +{ + g.bidi_flag = 0; /* 0=unidirectional */ + + kern_return_t kr = load_printerdriver(driverBundlePath); + if (kr != kIOReturnSuccess) + kr = -2; + + if (g.classdriver != NULL) + { + (*g.classdriver)->interfaceNumber = g.interfaceNum; + kr = (*g.classdriver)->Open(g.classdriver, g.location, kUSBPrintingProtocolBidirectional); + if (kr != kIOReturnSuccess || (*g.classdriver)->interface == NULL) + { + kr = (*g.classdriver)->Open(g.classdriver, g.location, kUSBPrintingProtocolUnidirectional); + if (kr == kIOReturnSuccess) + { + if ((*g.classdriver)->interface == NULL) + { + (*g.classdriver)->Close(g.classdriver); + kr = -1; + } + } + } + else + g.bidi_flag = 1; /* 1=bidirectional */ + } + + if (kr != kIOReturnSuccess) + unload_classdriver(&g.classdriver); + + return kr; +} + + +/* + * 'registry_close()' - Close the connection to the printer. + */ + +static kern_return_t registry_close(void) +{ + if (g.classdriver != NULL) + (*g.classdriver)->Close(g.classdriver); + + unload_classdriver(&g.classdriver); + return kIOReturnSuccess; +} + + +/* + * 'copy_deviceid()' - Copy the 1284 device id string. + */ + +static OSStatus copy_deviceid(classdriver_t **classdriver, + CFStringRef *deviceID) +{ + CFStringRef devID = NULL, + + deviceMake = NULL, + deviceModel = NULL, + deviceSerial = NULL; + + OSStatus err = (*classdriver)->GetDeviceID(classdriver, &devID, DEFAULT_TIMEOUT); + + copy_deviceinfo(devID, &deviceMake, &deviceModel, &deviceSerial); + + if (deviceMake == NULL || deviceModel == NULL || deviceSerial == NULL) + { + IOUSBDeviceDescriptor desc; + iodevice_request_t request; + + request.requestType = USBmakebmRequestType(kUSBIn, kUSBStandard, kUSBDevice); + request.request = kUSBRqGetDescriptor; + request.value = (kUSBDeviceDesc << 8) | 0; + request.index = 0; + request.length = sizeof(desc); + request.buffer = &desc; + err = (*classdriver)->DeviceRequest(classdriver, &request, DEFAULT_TIMEOUT); + if (err == kIOReturnSuccess) + { + CFMutableStringRef newDevID = CFStringCreateMutable(NULL, 0); + + if (deviceMake == NULL) + { + CFStringRef data = NULL; + err = (*classdriver)->GetString(classdriver, desc.iManufacturer, kUSBLanguageEnglish, DEFAULT_TIMEOUT, &data); + if (data != NULL) + { + CFStringAppendFormat(newDevID, NULL, CFSTR("MFG:%@;"), data); + CFRelease(data); + } + } + + if (deviceModel == NULL) + { + CFStringRef data = NULL; + err = (*classdriver)->GetString(classdriver, desc.iProduct, kUSBLanguageEnglish, DEFAULT_TIMEOUT, &data); + if (data != NULL) + { + CFStringAppendFormat(newDevID, NULL, CFSTR("MDL:%@;"), data); + CFRelease(data); + } + } + + if (deviceSerial == NULL && desc.iSerialNumber != 0) + { + CFStringRef data = NULL; + err = (*classdriver)->GetString(classdriver, desc.iSerialNumber, kUSBLanguageEnglish, DEFAULT_TIMEOUT, &data); + if (data != NULL) + { + CFStringAppendFormat(newDevID, NULL, CFSTR("SERN:%@;"), data); + CFRelease(data); + } + } + + if (devID != NULL) + { + CFStringAppend(newDevID, devID); + CFRelease(devID); + } + + *deviceID = newDevID; + } + } + else + { + *deviceID = devID; + } + release_deviceinfo(&deviceMake, &deviceModel, &deviceSerial); + + return err; +} + + +/* + * 'copy_devicestring()' - Copy the 1284 device id string. + */ + +static void copy_devicestring(io_service_t usbInterface, + CFStringRef *deviceID, + UInt32 *deviceLocation, + UInt8 *interfaceNumber ) +{ + IOCFPlugInInterface **iodev = NULL; + SInt32 score; + kern_return_t kr; + printer_interface_t interface; + HRESULT res; + classdriver_t **klassDriver = NULL; + CFStringRef driverBundlePath; + + if ((kr = IOCreatePlugInInterfaceForService(usbInterface, + kIOUSBInterfaceUserClientTypeID, + kIOCFPlugInInterfaceID, + &iodev, &score)) == kIOReturnSuccess) + { + if ((res = (*iodev)->QueryInterface(iodev, USB_INTERFACE_KIND, (LPVOID *) + &interface)) == noErr) + { + (*interface)->GetLocationID(interface, deviceLocation); + (*interface)->GetInterfaceNumber(interface, interfaceNumber); + + driverBundlePath = IORegistryEntryCreateCFProperty(usbInterface, + kUSBClassDriverProperty, + NULL, kNilOptions); + + kr = load_classdriver(driverBundlePath, interface, &klassDriver); + + if (kr != kIOReturnSuccess && driverBundlePath != NULL) + kr = load_classdriver(NULL, interface, &klassDriver); + + if (kr == kIOReturnSuccess && klassDriver != NULL) + kr = copy_deviceid(klassDriver, deviceID); + + unload_classdriver(&klassDriver); + + if (driverBundlePath != NULL) + CFRelease(driverBundlePath); + + /* (*interface)->Release(interface); */ + } + IODestroyPlugInInterface(iodev); + } +} + + +#pragma mark - +/* + * 'copy_value_for_key()' - Copy value string associated with a key. + */ + +static CFStringRef copy_value_for_key(CFStringRef deviceID, + CFStringRef *keys) +{ + CFStringRef value = NULL; + CFArrayRef kvPairs = deviceID != NULL ? CFStringCreateArrayBySeparatingStrings(NULL, deviceID, CFSTR(";")) : NULL; + CFIndex max = kvPairs != NULL ? CFArrayGetCount(kvPairs) : 0; + CFIndex idx = 0; + + while (idx < max && value == NULL) + { + CFStringRef kvpair = CFArrayGetValueAtIndex(kvPairs, idx); + CFIndex idxx = 0; + while (keys[idxx] != NULL && value == NULL) + { + CFRange range = CFStringFind(kvpair, keys[idxx], kCFCompareCaseInsensitive); + if (range.length != -1) + { + if (range.location != 0) + { + CFMutableStringRef theString = CFStringCreateMutableCopy(NULL, 0, kvpair); + CFStringTrimWhitespace(theString); + range = CFStringFind(theString, keys[idxx], kCFCompareCaseInsensitive); + if (range.location == 0) + value = CFStringCreateWithSubstring(NULL, theString, CFRangeMake(range.length, CFStringGetLength(theString) - range.length)); + + CFRelease(theString); + } + else + { + CFStringRef theString = CFStringCreateWithSubstring(NULL, kvpair, CFRangeMake(range.length, CFStringGetLength(kvpair) - range.length)); + CFMutableStringRef theString2 = CFStringCreateMutableCopy(NULL, 0, theString); + CFRelease(theString); + + CFStringTrimWhitespace(theString2); + value = theString2; + } + } + idxx++; + } + idx++; + } + + if (kvPairs != NULL) + CFRelease(kvPairs); + return value; +} + + +/* + * 'cfstr_create_trim()' - Create CFString and trim whitespace characters. + */ + +CFStringRef cfstr_create_trim(const char *cstr) +{ + CFStringRef cfstr; + CFMutableStringRef cfmutablestr = NULL; + + if ((cfstr = CFStringCreateWithCString(NULL, cstr, kCFStringEncodingUTF8)) != NULL) + { + if ((cfmutablestr = CFStringCreateMutableCopy(NULL, 1024, cfstr)) != NULL) + CFStringTrimWhitespace(cfmutablestr); + + CFRelease(cfstr); + } + return (CFStringRef) cfmutablestr; +} + + +#pragma mark - +/* + * 'parse_options()' - Parse URI options. + */ + +static void parse_options(char *options, + char *serial, + int serial_size, + UInt32 *location, + Boolean *wait_eof) +{ + char sep, /* Separator character */ + *name, /* Name of option */ + *value; /* Value of option */ + + + if (serial) + *serial = '\0'; + if (location) + *location = 0; + + if (!options) + return; + + while (*options) + { + /* + * Get the name... + */ + + name = options; + + while (*options && *options != '=' && *options != '+' && *options != '&') + options ++; + + if ((sep = *options) != '\0') + *options++ = '\0'; + + if (sep == '=') + { + /* + * Get the value... + */ + + value = options; + + while (*options && *options != '+' && *options != '&') + options ++; + + if (*options) + *options++ = '\0'; + } + else + value = (char *)""; + + /* + * Process the option... + */ + + if (!_cups_strcasecmp(name, "waiteof")) + { + if (!_cups_strcasecmp(value, "on") || + !_cups_strcasecmp(value, "yes") || + !_cups_strcasecmp(value, "true")) + *wait_eof = true; + else if (!_cups_strcasecmp(value, "off") || + !_cups_strcasecmp(value, "no") || + !_cups_strcasecmp(value, "false")) + *wait_eof = false; + else + _cupsLangPrintFilter(stderr, "WARNING", + _("Boolean expected for waiteof option \"%s\"."), + value); + } + else if (!_cups_strcasecmp(name, "serial")) + strlcpy(serial, value, serial_size); + else if (!_cups_strcasecmp(name, "location") && location) + *location = strtol(value, NULL, 16); + } +} + + +/*! + * @function setup_cfLanguage + * @abstract Convert the contents of the CUPS 'APPLE_LANGUAGE' environment + * variable into a one element CF array of languages. + * + * @discussion Each submitted job comes with a natural language. CUPS passes + * that language in an environment variable. We take that language + * and jam it into the AppleLanguages array so that CF will use + * it when reading localized resources. We need to do this before + * any CF code reads and caches the languages array, so this function + * should be called early in main() + */ +static void setup_cfLanguage(void) +{ + CFStringRef lang[1] = {NULL}; + CFArrayRef langArray = NULL; + const char *requestedLang = NULL; + + if ((requestedLang = getenv("APPLE_LANGUAGE")) == NULL) + requestedLang = getenv("LANG"); + + if (requestedLang != NULL) + { + lang[0] = CFStringCreateWithCString(kCFAllocatorDefault, requestedLang, kCFStringEncodingUTF8); + langArray = CFArrayCreate(kCFAllocatorDefault, (const void **)lang, sizeof(lang) / sizeof(lang[0]), &kCFTypeArrayCallBacks); + + CFPreferencesSetValue(CFSTR("AppleLanguages"), langArray, kCFPreferencesCurrentApplication, kCFPreferencesAnyUser, kCFPreferencesAnyHost); + fprintf(stderr, "DEBUG: usb: AppleLanguages=\"%s\"\n", requestedLang); + + CFRelease(lang[0]); + CFRelease(langArray); + } + else + fputs("DEBUG: usb: LANG and APPLE_LANGUAGE environment variables missing.\n", stderr); +} + +#pragma mark - +#if defined(__i386__) || defined(__x86_64__) +/*! + * @function run_legacy_backend + * + * @abstract Starts child backend process running as a ppc or i386 executable. + * + * @result Never returns; always calls exit(). + * + * @discussion + */ +static void run_legacy_backend(int argc, + char *argv[], + int fd) +{ + int i; + int exitstatus = 0; + int childstatus; + pid_t waitpid_status; + char *my_argv[32]; + char *usb_legacy_status; + + /* + * If we're running as x86_64 or i386 and couldn't load the class driver + * (because it's ppc or i386), then try to re-exec ourselves in ppc or i386 + * mode to try again. If we don't have a ppc or i386 architecture we may be + * running with the same architecture again so guard against this by setting + * and testing an environment variable... + */ + +# ifdef __x86_64__ + usb_legacy_status = getenv("USB_I386_STATUS"); +# else + usb_legacy_status = getenv("USB_PPC_STATUS"); +# endif /* __x86_64__ */ + + if (!usb_legacy_status) + { + /* + * Setup a SIGTERM handler then block it before forking... + */ + + int err; /* posix_spawn result */ + struct sigaction action; /* POSIX signal action */ + sigset_t newmask, /* New signal mask */ + oldmask; /* Old signal mask */ + char usbpath[1024]; /* Path to USB backend */ + const char *cups_serverbin;/* Path to CUPS binaries */ + + + memset(&action, 0, sizeof(action)); + sigaddset(&action.sa_mask, SIGTERM); + action.sa_handler = sigterm_handler; + sigaction(SIGTERM, &action, NULL); + + sigemptyset(&newmask); + sigaddset(&newmask, SIGTERM); + sigprocmask(SIG_BLOCK, &newmask, &oldmask); + + /* + * Set the environment variable... + */ + +# ifdef __x86_64__ + setenv("USB_I386_STATUS", "1", false); +# else + setenv("USB_PPC_STATUS", "1", false); +# endif /* __x86_64__ */ + + /* + * Tell the kernel to use the specified CPU architecture... + */ + +# ifdef __x86_64__ + cpu_type_t cpu = CPU_TYPE_I386; +# else + cpu_type_t cpu = CPU_TYPE_POWERPC; +# endif /* __x86_64__ */ + size_t ocount = 1; + posix_spawnattr_t attrs; + + if (!posix_spawnattr_init(&attrs)) + { + posix_spawnattr_setsigdefault(&attrs, &oldmask); + if (posix_spawnattr_setbinpref_np(&attrs, 1, &cpu, &ocount) || ocount != 1) + { +# ifdef __x86_64__ + perror("DEBUG: Unable to set binary preference to i386"); +# else + perror("DEBUG: Unable to set binary preference to ppc"); +# endif /* __x86_64__ */ + _cupsLangPrintFilter(stderr, "ERROR", + _("Unable to use legacy USB class driver.")); + exit(CUPS_BACKEND_STOP); + } + } + + /* + * Set up the arguments and call posix_spawn... + */ + + if ((cups_serverbin = getenv("CUPS_SERVERBIN")) == NULL) + cups_serverbin = CUPS_SERVERBIN; + snprintf(usbpath, sizeof(usbpath), "%s/backend/usb", cups_serverbin); + + for (i = 0; i < argc && i < (sizeof(my_argv) / sizeof(my_argv[0])) - 1; i ++) + my_argv[i] = argv[i]; + + my_argv[i] = NULL; + + if ((err = posix_spawn(&child_pid, usbpath, NULL, &attrs, my_argv, + environ)) != 0) + { + fprintf(stderr, "DEBUG: Unable to exec %s: %s\n", usbpath, + strerror(err)); + _cupsLangPrintFilter(stderr, "ERROR", + _("Unable to use legacy USB class driver.")); + exit(CUPS_BACKEND_STOP); + } + + /* + * Unblock signals... + */ + + sigprocmask(SIG_SETMASK, &oldmask, NULL); + + /* + * Close the fds we won't be using then wait for the child backend to exit. + */ + + close(fd); + close(1); + + fprintf(stderr, "DEBUG: Started usb(legacy) backend (PID %d)\n", + (int)child_pid); + + while ((waitpid_status = waitpid(child_pid, &childstatus, 0)) == (pid_t)-1 && errno == EINTR) + usleep(1000); + + if (WIFSIGNALED(childstatus)) + { + exitstatus = CUPS_BACKEND_STOP; + fprintf(stderr, "DEBUG: usb(legacy) backend %d crashed on signal %d\n", + child_pid, WTERMSIG(childstatus)); + } + else + { + if ((exitstatus = WEXITSTATUS(childstatus)) != 0) + fprintf(stderr, + "DEBUG: usb(legacy) backend %d stopped with status %d\n", + child_pid, exitstatus); + else + fprintf(stderr, "DEBUG: usb(legacy) backend %d exited with no errors\n", + child_pid); + } + } + else + { + fputs("DEBUG: usb(legacy) backend running native again\n", stderr); + exitstatus = CUPS_BACKEND_STOP; + } + + exit(exitstatus); +} +#endif /* __i386__ || __x86_64__ */ + + +/* + * 'sigterm_handler()' - SIGTERM handler. + */ + +static void +sigterm_handler(int sig) /* I - Signal */ +{ +#if defined(__i386__) || defined(__x86_64__) + /* + * If we started a child process pass the signal on to it... + */ + + if (child_pid) + { + /* + * If we started a child process pass the signal on to it... + */ + + int status; + + kill(child_pid, sig); + while (waitpid(child_pid, &status, 0) < 0 && errno == EINTR); + + if (WIFEXITED(status)) + exit(WEXITSTATUS(status)); + else if (status == SIGTERM || status == SIGKILL) + exit(0); + else + { + fprintf(stderr, "DEBUG: Child crashed on signal %d\n", status); + exit(CUPS_BACKEND_STOP); + } + } +#endif /* __i386__ || __x86_64__ */ +} + + +#ifdef PARSE_PS_ERRORS +/* + * 'next_line()' - Find the next line in a buffer. + */ + +static const char *next_line (const char *buffer) +{ + const char *cptr, *lptr = NULL; + + for (cptr = buffer; *cptr && lptr == NULL; cptr++) + if (*cptr == '\n' || *cptr == '\r') + lptr = cptr; + return lptr; +} + + +/* + * 'parse_pserror()' - Scan the backchannel data for postscript errors. + */ + +static void parse_pserror(char *sockBuffer, + int len) +{ + static char gErrorBuffer[1024] = ""; + static char *gErrorBufferPtr = gErrorBuffer; + static char *gErrorBufferEndPtr = gErrorBuffer + sizeof(gErrorBuffer); + + char *pCommentBegin, *pCommentEnd, *pLineEnd; + char *logLevel; + char logstr[1024]; + int logstrlen; + + if (gErrorBufferPtr + len > gErrorBufferEndPtr - 1) + gErrorBufferPtr = gErrorBuffer; + if (len > sizeof(gErrorBuffer) - 1) + len = sizeof(gErrorBuffer) - 1; + + memcpy(gErrorBufferPtr, (const void *)sockBuffer, len); + gErrorBufferPtr += len; + *(gErrorBufferPtr + 1) = '\0'; + + pLineEnd = (char *)next_line((const char *)gErrorBuffer); + while (pLineEnd != NULL) + { + *pLineEnd++ = '\0'; + + pCommentBegin = strstr(gErrorBuffer,"%%["); + pCommentEnd = strstr(gErrorBuffer, "]%%"); + if (pCommentBegin != gErrorBuffer && pCommentEnd != NULL) + { + pCommentEnd += 3; /* Skip past "]%%" */ + *pCommentEnd = '\0'; /* There's always room for the nul */ + + if (_cups_strncasecmp(pCommentBegin, "%%[ Error:", 10) == 0) + logLevel = "DEBUG"; + else if (_cups_strncasecmp(pCommentBegin, "%%[ Flushing", 12) == 0) + logLevel = "DEBUG"; + else + logLevel = "INFO"; + + if ((logstrlen = snprintf(logstr, sizeof(logstr), "%s: %s\n", logLevel, pCommentBegin)) >= sizeof(logstr)) + { + /* If the string was trucnated make sure it has a linefeed before the nul */ + logstrlen = sizeof(logstr) - 1; + logstr[logstrlen - 1] = '\n'; + } + write(STDERR_FILENO, logstr, logstrlen); + } + + /* move everything over... */ + strcpy(gErrorBuffer, pLineEnd); + gErrorBufferPtr = gErrorBuffer; + pLineEnd = (char *)next_line((const char *)gErrorBuffer); + } +} +#endif /* PARSE_PS_ERRORS */ + + +/* + * 'soft_reset()' - Send a soft reset to the device. + */ + +static void soft_reset(void) +{ + fd_set input_set; /* Input set for select() */ + struct timeval tv; /* Time value */ + char buffer[2048]; /* Buffer */ + struct timespec cond_timeout; /* pthread condition timeout */ + + /* + * Send an abort once a second until the I/O lock is released by the main thread... + */ + + pthread_mutex_lock(&g.readwrite_lock_mutex); + while (g.readwrite_lock) + { + (*g.classdriver)->Abort(g.classdriver); + + gettimeofday(&tv, NULL); + cond_timeout.tv_sec = tv.tv_sec + 1; + cond_timeout.tv_nsec = tv.tv_usec * 1000; + + while (g.readwrite_lock) + { + if (pthread_cond_timedwait(&g.readwrite_lock_cond, + &g.readwrite_lock_mutex, + &cond_timeout) != 0) + break; + } + } + + g.readwrite_lock = 1; + pthread_mutex_unlock(&g.readwrite_lock_mutex); + + /* + * Flush bytes waiting on print_fd... + */ + + g.print_bytes = 0; + + FD_ZERO(&input_set); + FD_SET(g.print_fd, &input_set); + + tv.tv_sec = 0; + tv.tv_usec = 0; + + while (select(g.print_fd+1, &input_set, NULL, NULL, &tv) > 0) + if (read(g.print_fd, buffer, sizeof(buffer)) <= 0) + break; + + /* + * Send the reset... + */ + + (*g.classdriver)->SoftReset(g.classdriver, DEFAULT_TIMEOUT); + + /* + * Release the I/O lock... + */ + + pthread_mutex_lock(&g.readwrite_lock_mutex); + g.readwrite_lock = 0; + pthread_cond_signal(&g.readwrite_lock_cond); + pthread_mutex_unlock(&g.readwrite_lock_mutex); +} + + +/* + * 'get_device_id()' - Return IEEE-1284 device ID. + */ + +static void get_device_id(cups_sc_status_t *status, + char *data, + int *datalen) +{ + CFStringRef deviceIDString = NULL; + + /* GetDeviceID */ + copy_deviceid(g.classdriver, &deviceIDString); + + if (deviceIDString) + { + CFStringGetCString(deviceIDString, data, *datalen, kCFStringEncodingUTF8); + *datalen = strlen(data); + CFRelease(deviceIDString); + } + *status = CUPS_SC_STATUS_OK; +} + + +/* + * End of "$Id$". + */ diff --git a/backend/usb-libusb.c b/backend/usb-libusb.c new file mode 100644 index 0000000000..914414982d --- /dev/null +++ b/backend/usb-libusb.c @@ -0,0 +1,942 @@ +/* + * "$Id$" + * + * Libusb interface code for CUPS. + * + * Copyright 2007-2012 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * list_devices() - List the available printers. + * print_device() - Print a file to a USB device. + * close_device() - Close the connection to the USB printer. + * find_device() - Find or enumerate USB printers. + * get_device_id() - Get the IEEE-1284 device ID for the printer. + * list_cb() - List USB printers for discovery. + * make_device_uri() - Create a device URI for a USB printer. + * open_device() - Open a connection to the USB printer. + * print_cb() - Find a USB printer for printing. + * side_cb() - Handle side-channel requests. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include + + +/* + * Local types... + */ + +typedef struct usb_printer_s /**** USB Printer Data ****/ +{ + struct usb_device *device; /* Device info */ + int conf, /* Configuration */ + iface, /* Interface */ + altset, /* Alternate setting */ + write_endp, /* Write endpoint */ + read_endp; /* Read endpoint */ + struct usb_dev_handle *handle; /* Open handle to device */ +} usb_printer_t; + +typedef int (*usb_cb_t)(usb_printer_t *, const char *, const char *, + const void *); + + +/* + * Local functions... + */ + +static int close_device(usb_printer_t *printer); +static usb_printer_t *find_device(usb_cb_t cb, const void *data); +static int get_device_id(usb_printer_t *printer, char *buffer, + size_t bufsize); +static int list_cb(usb_printer_t *printer, const char *device_uri, + const char *device_id, const void *data); +static char *make_device_uri(usb_printer_t *printer, + const char *device_id, + char *uri, size_t uri_size); +static int open_device(usb_printer_t *printer, int verbose); +static int print_cb(usb_printer_t *printer, const char *device_uri, + const char *device_id, const void *data); +static ssize_t side_cb(usb_printer_t *printer, int print_fd); + + +/* + * 'list_devices()' - List the available printers. + */ + +void +list_devices(void) +{ + fputs("DEBUG: list_devices\n", stderr); + find_device(list_cb, NULL); +} + + +/* + * 'print_device()' - Print a file to a USB device. + */ + +int /* O - Exit status */ +print_device(const char *uri, /* I - Device URI */ + const char *hostname, /* I - Hostname/manufacturer */ + const char *resource, /* I - Resource/modelname */ + char *options, /* I - Device options/serial number */ + int print_fd, /* I - File descriptor to print */ + int copies, /* I - Copies to print */ + int argc, /* I - Number of command-line arguments (6 or 7) */ + char *argv[]) /* I - Command-line arguments */ +{ + usb_printer_t *printer; /* Printer */ + ssize_t bytes, /* Bytes read/written */ + tbytes; /* Total bytes written */ + char buffer[512]; /* Print data buffer */ + struct sigaction action; /* Actions for POSIX signals */ + struct pollfd pfds[2]; /* Poll descriptors */ + + + fputs("DEBUG: print_device\n", stderr); + + /* + * Connect to the printer... + */ + + while ((printer = find_device(print_cb, uri)) == NULL) + { + _cupsLangPrintFilter(stderr, "INFO", + _("Waiting for printer to become available.")); + sleep(5); + } + + + /* + * If we are printing data from a print driver on stdin, ignore SIGTERM + * so that the driver can finish out any page data, e.g. to eject the + * current page. We only do this for stdin printing as otherwise there + * is no way to cancel a raw print job... + */ + + if (!print_fd) + { + memset(&action, 0, sizeof(action)); + + sigemptyset(&action.sa_mask); + action.sa_handler = SIG_IGN; + sigaction(SIGTERM, &action, NULL); + } + + tbytes = 0; + + pfds[0].fd = print_fd; + pfds[0].events = POLLIN; + pfds[1].fd = CUPS_SC_FD; + pfds[1].events = POLLIN; + + while (copies > 0 && tbytes >= 0) + { + copies --; + + if (print_fd != 0) + { + fputs("PAGE: 1 1\n", stderr); + lseek(print_fd, 0, SEEK_SET); + } + + /* + * TODO: Add back-channel support, along with better write error handling. + */ + + while (poll(pfds, 2, -1) > 0) + { + /* + * CUPS STR #3318: USB process hangs on end-of-file, making further + * printing impossible + * + * From a strict interpretation of POSIX poll(), POLLHUP should never be + * set without POLLIN, since POLLIN is the event you request. That said, + * it appears that some versions of Linux break this. + */ + + if (pfds[0].revents & (POLLIN | POLLHUP)) + { + if ((bytes = read(print_fd, buffer, sizeof(buffer))) > 0) + { + if (usb_bulk_write(printer->handle, printer->write_endp, buffer, + bytes, 3600000) < 0) + { + _cupsLangPrintFilter(stderr, "ERROR", + _("Unable to send data to printer.")); + tbytes = -1; + break; + } + + tbytes += bytes; + } + else if (bytes == 0 || (bytes < 0 && errno != EAGAIN && errno != EINTR)) + break; + } + + if (pfds[1].revents & (POLLIN | POLLHUP)) + { + if ((bytes = side_cb(printer, print_fd)) < 0) + pfds[1].events = 0; /* Filter has gone away... */ + else + tbytes += bytes; + } + } + } + + /* + * Close our connection and return... + */ + + close_device(printer); + + return (CUPS_BACKEND_OK); +} + + +/* + * 'close_device()' - Close the connection to the USB printer. + */ + +static int /* I - 0 on success, -1 on failure */ +close_device(usb_printer_t *printer) /* I - Printer */ +{ + if (printer->handle) + { + /* + * Release interfaces before closing so that we know all data is written + * to the device... + */ + + int number = printer->device->config[printer->conf]. + interface[printer->iface]. + altsetting[printer->altset].bInterfaceNumber; + usb_release_interface(printer->handle, number); + + if (number != 0) + usb_release_interface(printer->handle, 0); + + /* + * Close the interface and return... + */ + + usb_close(printer->handle); + printer->handle = NULL; + } + + return (0); +} + + +/* + * 'find_device()' - Find or enumerate USB printers. + */ + +static usb_printer_t * /* O - Found printer */ +find_device(usb_cb_t cb, /* I - Callback function */ + const void *data) /* I - User data for callback */ +{ + struct usb_bus *bus; /* Current bus */ + struct usb_device *device; /* Current device */ + struct usb_config_descriptor *confptr;/* Pointer to current configuration */ + struct usb_interface *ifaceptr; /* Pointer to current interface */ + struct usb_interface_descriptor *altptr; + /* Pointer to current alternate setting */ + struct usb_endpoint_descriptor *endpptr; + /* Pointer to current endpoint */ + int conf, /* Current configuration */ + iface, /* Current interface */ + altset, /* Current alternate setting */ + protocol, /* Current protocol */ + endp, /* Current endpoint */ + read_endp, /* Current read endpoint */ + write_endp; /* Current write endpoint */ + char device_id[1024],/* IEEE-1284 device ID */ + device_uri[1024]; + /* Device URI */ + static usb_printer_t printer; /* Current printer */ + + + /* + * Initialize libusb... + */ + + usb_init(); + fprintf(stderr, "DEBUG: usb_find_busses=%d\n", usb_find_busses()); + fprintf(stderr, "DEBUG: usb_find_devices=%d\n", usb_find_devices()); + + /* + * Then loop through the devices it found... + */ + + for (bus = usb_get_busses(); bus; bus = bus->next) + for (device = bus->devices; device; device = device->next) + { + /* + * Ignore devices with no configuration data and anything that is not + * a printer... + */ + + if (!device->config || !device->descriptor.idVendor || + !device->descriptor.idProduct) + continue; + + for (conf = 0, confptr = device->config; + conf < device->descriptor.bNumConfigurations; + conf ++, confptr ++) + for (iface = 0, ifaceptr = confptr->interface; + iface < confptr->bNumInterfaces; + iface ++, ifaceptr ++) + { + /* + * Some printers offer multiple interfaces... + */ + + protocol = 0; + + for (altset = 0, altptr = ifaceptr->altsetting; + altset < ifaceptr->num_altsetting; + altset ++, altptr ++) + { + /* + * Currently we only support unidirectional and bidirectional + * printers. Future versions of this code will support the + * 1284.4 (packet mode) protocol as well. + */ + + if (altptr->bInterfaceClass != USB_CLASS_PRINTER || + altptr->bInterfaceSubClass != 1 || + (altptr->bInterfaceProtocol != 1 && /* Unidirectional */ + altptr->bInterfaceProtocol != 2) || /* Bidirectional */ + altptr->bInterfaceProtocol < protocol) + continue; + + read_endp = -1; + write_endp = -1; + + for (endp = 0, endpptr = altptr->endpoint; + endp < altptr->bNumEndpoints; + endp ++, endpptr ++) + if ((endpptr->bmAttributes & USB_ENDPOINT_TYPE_MASK) == + USB_ENDPOINT_TYPE_BULK) + { + if (endpptr->bEndpointAddress & USB_ENDPOINT_DIR_MASK) + read_endp = endp; + else + write_endp = endp; + } + + if (write_endp >= 0) + { + /* + * Save the best match so far... + */ + + protocol = altptr->bInterfaceProtocol; + printer.altset = altset; + printer.write_endp = write_endp; + printer.read_endp = read_endp; + } + } + + if (protocol > 0) + { + printer.device = device; + printer.conf = conf; + printer.iface = iface; + printer.handle = NULL; + + if (!open_device(&printer, data != NULL)) + { + if (!get_device_id(&printer, device_id, sizeof(device_id))) + { + make_device_uri(&printer, device_id, device_uri, + sizeof(device_uri)); + + if ((*cb)(&printer, device_uri, device_id, data)) + { + printer.read_endp = printer.device->config[printer.conf]. + interface[printer.iface]. + altsetting[printer.altset]. + endpoint[printer.read_endp]. + bEndpointAddress; + printer.write_endp = printer.device->config[printer.conf]. + interface[printer.iface]. + altsetting[printer.altset]. + endpoint[printer.write_endp]. + bEndpointAddress; + return (&printer); + } + } + + close_device(&printer); + } + } + } + } + + /* + * If we get this far without returning, then we haven't found a printer + * to print to... + */ + + return (NULL); +} + + +/* + * 'get_device_id()' - Get the IEEE-1284 device ID for the printer. + */ + +static int /* O - 0 on success, -1 on error */ +get_device_id(usb_printer_t *printer, /* I - Printer */ + char *buffer, /* I - String buffer */ + size_t bufsize) /* I - Number of bytes in buffer */ +{ + int length; /* Length of device ID */ + + + if (usb_control_msg(printer->handle, + USB_TYPE_CLASS | USB_ENDPOINT_IN | USB_RECIP_INTERFACE, + 0, printer->conf, (printer->iface << 8) | printer->altset, + buffer, bufsize, 5000) < 0) + { + *buffer = '\0'; + return (-1); + } + + /* + * Extract the length of the device ID string from the first two + * bytes. The 1284 spec says the length is stored MSB first... + */ + + length = (((unsigned)buffer[0] & 255) << 8) | + ((unsigned)buffer[1] & 255); + + /* + * Check to see if the length is larger than our buffer or less than 14 bytes + * (the minimum valid device ID is "MFG:x;MDL:y;" with 2 bytes for the length). + * + * If the length is out-of-range, assume that the vendor incorrectly + * implemented the 1284 spec and re-read the length as LSB first,.. + */ + + if (length > bufsize || length < 14) + length = (((unsigned)buffer[1] & 255) << 8) | + ((unsigned)buffer[0] & 255); + + if (length > bufsize) + length = bufsize; + + if (length < 14) + { + /* + * Invalid device ID, clear it! + */ + + *buffer = '\0'; + return (-1); + } + + length -= 2; + + /* + * Copy the device ID text to the beginning of the buffer and + * nul-terminate. + */ + + memmove(buffer, buffer + 2, length); + buffer[length] = '\0'; + + return (0); +} + + +/* + * 'list_cb()' - List USB printers for discovery. + */ + +static int /* O - 0 to continue, 1 to stop */ +list_cb(usb_printer_t *printer, /* I - Printer */ + const char *device_uri, /* I - Device URI */ + const char *device_id, /* I - IEEE-1284 device ID */ + const void *data) /* I - User data (not used) */ +{ + char make_model[1024]; /* Make and model */ + + + /* + * Get the device URI and make/model strings... + */ + + backendGetMakeModel(device_id, make_model, sizeof(make_model)); + + /* + * Report the printer... + */ + + cupsBackendReport("direct", device_uri, make_model, make_model, device_id, + NULL); + + /* + * Keep going... + */ + + return (0); +} + + +/* + * 'make_device_uri()' - Create a device URI for a USB printer. + */ + +static char * /* O - Device URI */ +make_device_uri( + usb_printer_t *printer, /* I - Printer */ + const char *device_id, /* I - IEEE-1284 device ID */ + char *uri, /* I - Device URI buffer */ + size_t uri_size) /* I - Size of device URI buffer */ +{ + char options[1024]; /* Device URI options */ + int num_values; /* Number of 1284 parameters */ + cups_option_t *values; /* 1284 parameters */ + const char *mfg, /* Manufacturer */ + *mdl, /* Model */ + *des, /* Description */ + *sern; /* Serial number */ + size_t mfglen; /* Length of manufacturer string */ + char tempmfg[256], /* Temporary manufacturer string */ + tempsern[256], /* Temporary serial number string */ + *tempptr; /* Pointer into temp string */ + + + /* + * Get the make, model, and serial numbers... + */ + + num_values = _cupsGet1284Values(device_id, &values); + + if ((sern = cupsGetOption("SERIALNUMBER", num_values, values)) == NULL) + if ((sern = cupsGetOption("SERN", num_values, values)) == NULL) + if ((sern = cupsGetOption("SN", num_values, values)) == NULL && + printer->device->descriptor.iSerialNumber) + { + /* + * Try getting the serial number from the device itself... + */ + + int length = usb_get_string_simple(printer->handle, + printer->device->descriptor. + iSerialNumber, + tempsern, sizeof(tempsern) - 1); + if (length > 0) + { + tempsern[length] = '\0'; + sern = tempsern; + } + } + + if ((mfg = cupsGetOption("MANUFACTURER", num_values, values)) == NULL) + mfg = cupsGetOption("MFG", num_values, values); + + if ((mdl = cupsGetOption("MODEL", num_values, values)) == NULL) + mdl = cupsGetOption("MDL", num_values, values); + +#ifdef __APPLE__ + /* + * To maintain compatibility with the original IOKit-based backend on Mac OS X, + * don't map manufacturer names... + */ + + if (!mfg) + +#else + /* + * To maintain compatibility with the original character device backend on + * Linux and *BSD, map manufacturer names... + */ + + if (mfg) + { + if (!_cups_strcasecmp(mfg, "Hewlett-Packard")) + mfg = "HP"; + else if (!_cups_strcasecmp(mfg, "Lexmark International")) + mfg = "Lexmark"; + } + else +#endif /* __APPLE__ */ + { + /* + * No manufacturer? Use the model string or description... + */ + + if (mdl) + _ppdNormalizeMakeAndModel(mdl, tempmfg, sizeof(tempmfg)); + else if ((des = cupsGetOption("DESCRIPTION", num_values, values)) != NULL || + (des = cupsGetOption("DES", num_values, values)) != NULL) + _ppdNormalizeMakeAndModel(des, tempmfg, sizeof(tempmfg)); + else + strlcpy(tempmfg, "Unknown", sizeof(tempmfg)); + + if ((tempptr = strchr(tempmfg, ' ')) != NULL) + *tempptr = '\0'; + + mfg = tempmfg; + } + + mfglen = strlen(mfg); + + if (!strncasecmp(mdl, mfg, mfglen) && _cups_isspace(mdl[mfglen])) + { + mdl += mfglen + 1; + + while (_cups_isspace(*mdl)) + mdl ++; + } + + /* + * Generate the device URI from the manufacturer, model, serial number, + * and interface number... + */ + + if (sern) + { + if (printer->iface > 0) + snprintf(options, sizeof(options), "?serial=%s&interface=%d", sern, + printer->iface); + else + snprintf(options, sizeof(options), "?serial=%s", sern); + } + else if (printer->iface > 0) + snprintf(options, sizeof(options), "?interface=%d", printer->iface); + else + options[0] = '\0'; + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, uri_size, "usb", NULL, mfg, 0, + "/%s%s", mdl, options); + + cupsFreeOptions(num_values, values); + + return (uri); +} + + +/* + * 'open_device()' - Open a connection to the USB printer. + */ + +static int /* O - 0 on success, -1 on error */ +open_device(usb_printer_t *printer, /* I - Printer */ + int verbose) /* I - Update connecting-to-device state? */ +{ + int number; /* Configuration/interface/altset numbers */ + char current; /* Current configuration */ + + + /* + * Return immediately if we are already connected... + */ + + if (printer->handle) + return (0); + + /* + * Try opening the printer... + */ + + if ((printer->handle = usb_open(printer->device)) == NULL) + return (-1); + + if (verbose) + fputs("STATE: +connecting-to-device\n", stderr); + + /* + * Set the desired configuration, but only if it needs changing. Some + * printers (e.g., Samsung) don't like usb_set_configuration. It will succeed, + * but the following print job is sometimes silently lost by the printer. + */ + + if (usb_control_msg(printer->handle, + USB_TYPE_STANDARD | USB_ENDPOINT_IN | USB_RECIP_DEVICE, + 8, /* GET_CONFIGURATION */ + 0, 0, ¤t, 1, 5000) != 1) + current = 0; /* Assume not configured */ + + number = printer->device->config[printer->conf].bConfigurationValue; + if (number != current) + { + if (usb_set_configuration(printer->handle, number) < 0) + { + /* + * If the set fails, chances are that the printer only supports a + * single configuration. Technically these printers don't conform to + * the USB printer specification, but otherwise they'll work... + */ + + if (errno != EBUSY) + fprintf(stderr, "DEBUG: Failed to set configuration %d for %04x:%04x\n", + number, printer->device->descriptor.idVendor, + printer->device->descriptor.idProduct); + } + } + + /* + * Claim interfaces as needed... + */ + + number = printer->device->config[printer->conf].interface[printer->iface]. + altsetting[printer->altset].bInterfaceNumber; + while (usb_claim_interface(printer->handle, number) < 0) + { + if (errno != EBUSY) + fprintf(stderr, + "DEBUG: Failed to claim interface %d for %04x:%04x: %s\n", + number, printer->device->descriptor.idVendor, + printer->device->descriptor.idProduct, strerror(errno)); + + goto error; + } + + /* + * Set alternate setting, but only if there is more than one option. Some + * printers (e.g., Samsung) don't like usb_set_altinterface. + */ + + if (printer->device->config[printer->conf].interface[printer->iface]. + num_altsetting > 1) + { + number = printer->device->config[printer->conf].interface[printer->iface]. + altsetting[printer->altset].bAlternateSetting; + + while (usb_set_altinterface(printer->handle, number) < 0) + { + if (errno != EBUSY) + fprintf(stderr, + "DEBUG: Failed to set alternate interface %d for %04x:%04x: " + "%s\n", number, printer->device->descriptor.idVendor, + printer->device->descriptor.idProduct, strerror(errno)); + + goto error; + } + } + + if (verbose) + fputs("STATE: -connecting-to-device\n", stderr); + + return (0); + + /* + * If we get here, there was a hard error... + */ + + error: + + if (verbose) + fputs("STATE: -connecting-to-device\n", stderr); + + usb_close(printer->handle); + printer->handle = NULL; + + return (-1); +} + + +/* + * 'print_cb()' - Find a USB printer for printing. + */ + +static int /* O - 0 to continue, 1 to stop (found) */ +print_cb(usb_printer_t *printer, /* I - Printer */ + const char *device_uri, /* I - Device URI */ + const char *device_id, /* I - IEEE-1284 device ID */ + const void *data) /* I - User data (make, model, S/N) */ +{ + char requested_uri[1024], /* Requested URI */ + *requested_ptr, /* Pointer into requested URI */ + detected_uri[1024], /* Detected URI */ + *detected_ptr; /* Pointer into detected URI */ + + + /* + * If we have an exact match, stop now... + */ + + if (!strcmp((char *)data, device_uri)) + return (1); + + /* + * Work on copies of the URIs... + */ + + strlcpy(requested_uri, (char *)data, sizeof(requested_uri)); + strlcpy(detected_uri, device_uri, sizeof(detected_uri)); + + /* + * libusb-discovered URIs can have an "interface" specification and this + * never happens for usblp-discovered URIs, so remove the "interface" + * specification from the URI which we are checking currently. This way a + * queue for a usblp-discovered printer can now be accessed via libusb. + * + * Similarly, strip "?serial=NNN...NNN" as needed. + */ + + if ((requested_ptr = strstr(requested_uri, "?interface=")) == NULL) + requested_ptr = strstr(requested_uri, "&interface="); + if ((detected_ptr = strstr(detected_uri, "?interface=")) == NULL) + detected_ptr = strstr(detected_uri, "&interface="); + + if (!requested_ptr && detected_ptr) + { + /* + * Strip "[?&]interface=nnn" from the detected printer. + */ + + *detected_ptr = '\0'; + } + else if (requested_ptr && !detected_ptr) + { + /* + * Strip "[?&]interface=nnn" from the requested printer. + */ + + *requested_ptr = '\0'; + } + + if ((requested_ptr = strstr(requested_uri, "?serial=?")) != NULL) + { + /* + * Strip "?serial=?" from the requested printer. This is a special + * case, as "?serial=?" means no serial number and not the serial + * number '?'. This is not covered by the checks below... + */ + + *requested_ptr = '\0'; + } + + if ((requested_ptr = strstr(requested_uri, "?serial=")) == NULL && + (detected_ptr = strstr(detected_uri, "?serial=")) != NULL) + { + /* + * Strip "?serial=nnn" from the detected printer. + */ + + *detected_ptr = '\0'; + } + else if (requested_ptr && !detected_ptr) + { + /* + * Strip "?serial=nnn" from the requested printer. + */ + + *requested_ptr = '\0'; + } + + return (!strcmp(requested_uri, detected_uri)); +} + + +/* + * 'side_cb()' - Handle side-channel requests. + */ + +static ssize_t /* O - Number of bytes written */ +side_cb(usb_printer_t *printer, /* I - Printer */ + int print_fd) /* I - File to print */ +{ + ssize_t bytes, /* Bytes read/written */ + tbytes; /* Total bytes written */ + char buffer[512]; /* Print data buffer */ + struct pollfd pfd; /* Poll descriptor */ + cups_sc_command_t command; /* Request command */ + cups_sc_status_t status; /* Request/response status */ + char data[2048]; /* Request/response data */ + int datalen; /* Request/response data size */ + + + tbytes = 0; + datalen = sizeof(data); + + if (cupsSideChannelRead(&command, &status, data, &datalen, 1.0)) + return (-1); + + switch (command) + { + case CUPS_SC_CMD_DRAIN_OUTPUT : + pfd.fd = print_fd; + pfd.events = POLLIN; + + while (poll(&pfd, 1, 1000) > 0) + { + if ((bytes = read(print_fd, buffer, sizeof(buffer))) > 0) + { + while (usb_bulk_write(printer->handle, printer->write_endp, buffer, + bytes, 5000) < 0) + { + _cupsLangPrintFilter(stderr, "ERROR", + _("Unable to send data to printer.")); + tbytes = -1; + break; + } + + tbytes += bytes; + } + else if (bytes < 0 && errno != EAGAIN && errno != EINTR) + break; + } + + if (tbytes < 0) + status = CUPS_SC_STATUS_IO_ERROR; + else + status = CUPS_SC_STATUS_OK; + + datalen = 0; + break; + + case CUPS_SC_CMD_GET_BIDI : + status = CUPS_SC_STATUS_OK; + data[0] = 0; /* TODO: Change to 1 when read supported */ + datalen = 1; + break; + + case CUPS_SC_CMD_GET_DEVICE_ID : + if (get_device_id(printer, data, sizeof(data))) + { + status = CUPS_SC_STATUS_IO_ERROR; + datalen = 0; + } + else + { + status = CUPS_SC_STATUS_OK; + datalen = strlen(data); + } + break; + + default : + status = CUPS_SC_STATUS_NOT_IMPLEMENTED; + datalen = 0; + break; + } + + cupsSideChannelWrite(command, status, data, datalen, 1.0); + + return (tbytes); +} + + +/* + * End of "$Id$". + */ + diff --git a/backend/usb-unix.c b/backend/usb-unix.c new file mode 100644 index 0000000000..b0365e4701 --- /dev/null +++ b/backend/usb-unix.c @@ -0,0 +1,623 @@ +/* + * "$Id$" + * + * USB port backend for CUPS. + * + * This file is included from "usb.c" when compiled on UNIX/Linux. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * print_device() - Print a file to a USB device. + * list_devices() - List all USB devices. + * open_device() - Open a USB device... + * side_cb() - Handle side-channel requests... + */ + +/* + * Include necessary headers. + */ + +#include + + +/* + * Local functions... + */ + +static int open_device(const char *uri, int *use_bc); +static int side_cb(int print_fd, int device_fd, int snmp_fd, + http_addr_t *addr, int use_bc); + + +/* + * 'print_device()' - Print a file to a USB device. + */ + +int /* O - Exit status */ +print_device(const char *uri, /* I - Device URI */ + const char *hostname, /* I - Hostname/manufacturer */ + const char *resource, /* I - Resource/modelname */ + char *options, /* I - Device options/serial number */ + int print_fd, /* I - File descriptor to print */ + int copies, /* I - Copies to print */ + int argc, /* I - Number of command-line arguments (6 or 7) */ + char *argv[]) /* I - Command-line arguments */ +{ + int use_bc; /* Use backchannel path? */ + int device_fd; /* USB device */ + ssize_t tbytes; /* Total number of bytes written */ + struct termios opts; /* Parallel port options */ + + + (void)argc; + (void)argv; + + /* + * Open the USB port device... + */ + + fputs("STATE: +connecting-to-device\n", stderr); + + do + { +#if defined(__FreeBSD__) || defined(__NetBSD__) || defined(__OpenBSD__) || defined(__DragonFly__) + /* + * *BSD's ulpt driver currently does not support the + * back-channel, incorrectly returns data ready on a select(), + * and locks up on read()... + */ + + use_bc = 0; + +#elif defined(__sun) + /* + * CUPS STR #3028: Solaris' usbprn driver apparently does not support + * select() or poll(), so we can't support backchannel... + */ + + use_bc = 0; + +#else + /* + * Disable backchannel data when printing to Brother, Canon, or + * Minolta USB printers - apparently these printers will return + * the IEEE-1284 device ID over and over and over when they get + * a read request... + */ + + use_bc = _cups_strcasecmp(hostname, "Brother") && + _cups_strcasecmp(hostname, "Canon") && + _cups_strncasecmp(hostname, "Konica", 6) && + _cups_strncasecmp(hostname, "Minolta", 7); +#endif /* __FreeBSD__ || __NetBSD__ || __OpenBSD__ || __DragonFly__ */ + + if ((device_fd = open_device(uri, &use_bc)) == -1) + { + if (getenv("CLASS") != NULL) + { + /* + * If the CLASS environment variable is set, the job was submitted + * to a class and not to a specific queue. In this case, we want + * to abort immediately so that the job can be requeued on the next + * available printer in the class. + */ + + _cupsLangPrintFilter(stderr, "INFO", + _("Unable to contact printer, queuing on next " + "printer in class.")); + + /* + * Sleep 5 seconds to keep the job from requeuing too rapidly... + */ + + sleep(5); + + return (CUPS_BACKEND_FAILED); + } + + if (errno == EBUSY) + { + _cupsLangPrintFilter(stderr, "INFO", + _("Printer busy, will retry in 10 seconds.")); + sleep(10); + } + else if (errno == ENXIO || errno == EIO || errno == ENOENT || + errno == ENODEV) + { + _cupsLangPrintFilter(stderr, "INFO", + _("Printer not connected, will retry in 30 " + "seconds.")); + sleep(30); + } + else + { + _cupsLangPrintError("ERROR", _("Unable to open device file")); + return (CUPS_BACKEND_FAILED); + } + } + } + while (device_fd < 0); + + fputs("STATE: -connecting-to-device\n", stderr); + + /* + * Set any options provided... + */ + + tcgetattr(device_fd, &opts); + + opts.c_lflag &= ~(ICANON | ECHO | ISIG); /* Raw mode */ + + /**** No options supported yet ****/ + + tcsetattr(device_fd, TCSANOW, &opts); + + /* + * Finally, send the print file... + */ + + tbytes = 0; + + while (copies > 0 && tbytes >= 0) + { + copies --; + + if (print_fd != 0) + { + fputs("PAGE: 1 1\n", stderr); + lseek(print_fd, 0, SEEK_SET); + } + +#ifdef __sun + /* + * CUPS STR #3028: Solaris' usbprn driver apparently does not support + * select() or poll(), so we can't support the sidechannel either... + */ + + tbytes = backendRunLoop(print_fd, device_fd, -1, NULL, use_bc, 1, NULL); + +#else + tbytes = backendRunLoop(print_fd, device_fd, -1, NULL, use_bc, 1, side_cb); +#endif /* __sun */ + + if (print_fd != 0 && tbytes >= 0) + _cupsLangPrintFilter(stderr, "INFO", _("Print file sent.")); + } + + /* + * Close the USB port and return... + */ + + close(device_fd); + + return (CUPS_BACKEND_OK); +} + + +/* + * 'list_devices()' - List all USB devices. + */ + +void +list_devices(void) +{ +#ifdef __linux + int i; /* Looping var */ + int fd; /* File descriptor */ + char device[255], /* Device filename */ + device_id[1024], /* Device ID string */ + device_uri[1024], /* Device URI string */ + make_model[1024]; /* Make and model */ + + + /* + * Try to open each USB device... + */ + + for (i = 0; i < 16; i ++) + { + /* + * Linux has a long history of changing the standard filenames used + * for USB printer devices. We get the honor of trying them all... + */ + + sprintf(device, "/dev/usblp%d", i); + + if ((fd = open(device, O_RDWR | O_EXCL)) < 0) + { + if (errno != ENOENT) + continue; + + sprintf(device, "/dev/usb/lp%d", i); + + if ((fd = open(device, O_RDWR | O_EXCL)) < 0) + { + if (errno != ENOENT) + continue; + + sprintf(device, "/dev/usb/usblp%d", i); + + if ((fd = open(device, O_RDWR | O_EXCL)) < 0) + continue; + } + } + + if (!backendGetDeviceID(fd, device_id, sizeof(device_id), + make_model, sizeof(make_model), + "usb", device_uri, sizeof(device_uri))) + cupsBackendReport("direct", device_uri, make_model, make_model, + device_id, NULL); + + close(fd); + } +#elif defined(__sgi) +#elif defined(__sun) && defined(ECPPIOC_GETDEVID) + int i; /* Looping var */ + int fd; /* File descriptor */ + char device[255], /* Device filename */ + device_id[1024], /* Device ID string */ + device_uri[1024], /* Device URI string */ + make_model[1024]; /* Make and model */ + + + /* + * Open each USB device... + */ + + for (i = 0; i < 8; i ++) + { + sprintf(device, "/dev/usb/printer%d", i); + + if ((fd = open(device, O_WRONLY | O_EXCL)) >= 0) + { + if (!backendGetDeviceID(fd, device_id, sizeof(device_id), + make_model, sizeof(make_model), + "usb", device_uri, sizeof(device_uri))) + cupsBackendReport("direct", device_uri, make_model, make_model, + device_id, NULL); + + close(fd); + } + } +#elif defined(__hpux) +#elif defined(__osf) +#elif defined(__FreeBSD__) || defined(__NetBSD__) || defined(__OpenBSD__) || defined(__DragonFly__) || defined(__FreeBSD_kernel__) + int i; /* Looping var */ + char device[255]; /* Device filename */ + + + for (i = 0; i < 8; i ++) + { + sprintf(device, "/dev/ulpt%d", i); + if (!access(device, 0)) + printf("direct usb:%s \"Unknown\" \"USB Printer #%d\"\n", device, i + 1); + + sprintf(device, "/dev/unlpt%d", i); + if (!access(device, 0)) + printf("direct usb:%s \"Unknown\" \"USB Printer #%d (no reset)\"\n", device, i + 1); + } +#endif +} + + +/* + * 'open_device()' - Open a USB device... + */ + +static int /* O - File descriptor or -1 on error */ +open_device(const char *uri, /* I - Device URI */ + int *use_bc) /* O - Set to 0 for unidirectional */ +{ + int fd; /* File descriptor */ + + + /* + * The generic implementation just treats the URI as a device filename... + * Specific operating systems may also support using the device serial + * number and/or make/model. + */ + + if (!strncmp(uri, "usb:/dev/", 9)) +#ifdef __linux + { + /* + * Do not allow direct devices anymore... + */ + + errno = ENODEV; + return (-1); + } + else if (!strncmp(uri, "usb://", 6)) + { + /* + * For Linux, try looking up the device serial number or model... + */ + + int i; /* Looping var */ + int busy; /* Are any ports busy? */ + char device[255], /* Device filename */ + device_id[1024], /* Device ID string */ + make_model[1024], /* Make and model */ + device_uri[1024]; /* Device URI string */ + + + /* + * Find the correct USB device... + */ + + for (;;) + { + for (busy = 0, i = 0; i < 16; i ++) + { + /* + * Linux has a long history of changing the standard filenames used + * for USB printer devices. We get the honor of trying them all... + */ + + sprintf(device, "/dev/usblp%d", i); + + if ((fd = open(device, O_RDWR | O_EXCL)) < 0 && errno == ENOENT) + { + sprintf(device, "/dev/usb/lp%d", i); + + if ((fd = open(device, O_RDWR | O_EXCL)) < 0 && errno == ENOENT) + { + sprintf(device, "/dev/usb/usblp%d", i); + + if ((fd = open(device, O_RDWR | O_EXCL)) < 0 && errno == ENOENT) + continue; + } + } + + if (fd >= 0) + { + backendGetDeviceID(fd, device_id, sizeof(device_id), + make_model, sizeof(make_model), + "usb", device_uri, sizeof(device_uri)); + } + else + { + /* + * If the open failed because it was busy, flag it so we retry + * as needed... + */ + + if (errno == EBUSY) + busy = 1; + + device_uri[0] = '\0'; + } + + if (!strcmp(uri, device_uri)) + { + /* + * Yes, return this file descriptor... + */ + + fprintf(stderr, "DEBUG: Printer using device file \"%s\"...\n", + device); + + return (fd); + } + + /* + * This wasn't the one... + */ + + if (fd >= 0) + close(fd); + } + + /* + * If we get here and at least one of the printer ports showed up + * as "busy", then sleep for a bit and retry... + */ + + if (busy) + _cupsLangPrintFilter(stderr, "INFO", + _("Printer is busy, will retry in 5 seconds.")); + + sleep(5); + } + } +#elif defined(__sun) && defined(ECPPIOC_GETDEVID) + { + /* + * Do not allow direct devices anymore... + */ + + errno = ENODEV; + return (-1); + } + else if (!strncmp(uri, "usb://", 6)) + { + /* + * For Solaris, try looking up the device serial number or model... + */ + + int i; /* Looping var */ + int busy; /* Are any ports busy? */ + char device[255], /* Device filename */ + device_id[1024], /* Device ID string */ + make_model[1024], /* Make and model */ + device_uri[1024]; /* Device URI string */ + + + /* + * Find the correct USB device... + */ + + do + { + for (i = 0, busy = 0; i < 8; i ++) + { + sprintf(device, "/dev/usb/printer%d", i); + + if ((fd = open(device, O_WRONLY | O_EXCL)) >= 0) + backendGetDeviceID(fd, device_id, sizeof(device_id), + make_model, sizeof(make_model), + "usb", device_uri, sizeof(device_uri)); + else + { + /* + * If the open failed because it was busy, flag it so we retry + * as needed... + */ + + if (errno == EBUSY) + busy = 1; + + device_uri[0] = '\0'; + } + + if (!strcmp(uri, device_uri)) + { + /* + * Yes, return this file descriptor... + */ + + fputs("DEBUG: Setting use_bc to 0!\n", stderr); + + *use_bc = 0; + + return (fd); + } + + /* + * This wasn't the one... + */ + + if (fd >= 0) + close(fd); + } + + /* + * If we get here and at least one of the printer ports showed up + * as "busy", then sleep for a bit and retry... + */ + + if (busy) + { + _cupsLangPrintFilter(stderr, "INFO", + _("Printer is busy, will retry in 5 seconds.")); + sleep(5); + } + } + while (busy); + + /* + * Couldn't find the printer, return "no such device or address"... + */ + + errno = ENODEV; + + return (-1); + } +#else + { + if (*use_bc) + fd = open(uri + 4, O_RDWR | O_EXCL); + else + fd = -1; + + if (fd < 0) + { + fd = open(uri + 4, O_WRONLY | O_EXCL); + *use_bc = 0; + } + + return (fd); + } +#endif /* __linux */ + else + { + errno = ENODEV; + return (-1); + } +} + + +/* + * 'side_cb()' - Handle side-channel requests... + */ + +static int /* O - 0 on success, -1 on error */ +side_cb(int print_fd, /* I - Print file */ + int device_fd, /* I - Device file */ + int snmp_fd, /* I - SNMP socket (unused) */ + http_addr_t *addr, /* I - Device address (unused) */ + int use_bc) /* I - Using back-channel? */ +{ + cups_sc_command_t command; /* Request command */ + cups_sc_status_t status; /* Request/response status */ + char data[2048]; /* Request/response data */ + int datalen; /* Request/response data size */ + + + (void)snmp_fd; + (void)addr; + + datalen = sizeof(data); + + if (cupsSideChannelRead(&command, &status, data, &datalen, 1.0)) + return (-1); + + switch (command) + { + case CUPS_SC_CMD_DRAIN_OUTPUT : + if (backendDrainOutput(print_fd, device_fd)) + status = CUPS_SC_STATUS_IO_ERROR; + else if (tcdrain(device_fd)) + status = CUPS_SC_STATUS_IO_ERROR; + else + status = CUPS_SC_STATUS_OK; + + datalen = 0; + break; + + case CUPS_SC_CMD_GET_BIDI : + status = CUPS_SC_STATUS_OK; + data[0] = use_bc; + datalen = 1; + break; + + case CUPS_SC_CMD_GET_DEVICE_ID : + memset(data, 0, sizeof(data)); + + if (backendGetDeviceID(device_fd, data, sizeof(data) - 1, + NULL, 0, NULL, NULL, 0)) + { + status = CUPS_SC_STATUS_NOT_IMPLEMENTED; + datalen = 0; + } + else + { + status = CUPS_SC_STATUS_OK; + datalen = strlen(data); + } + break; + + default : + status = CUPS_SC_STATUS_NOT_IMPLEMENTED; + datalen = 0; + break; + } + + return (cupsSideChannelWrite(command, status, data, datalen, 1.0)); +} + + +/* + * End of "$Id$". + */ diff --git a/backend/usb.c b/backend/usb.c new file mode 100644 index 0000000000..d72535e938 --- /dev/null +++ b/backend/usb.c @@ -0,0 +1,264 @@ +/* + * "$Id$" + * + * USB port backend for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * list_devices() - List all available USB devices to stdout. + * print_device() - Print a file to a USB device. + * main() - Send a file to the specified USB port. + */ + +/* + * Include necessary headers. + */ + +#ifdef __APPLE__ + /* A header order dependency requires this be first */ +# include +#endif /* __APPLE__ */ + +#include "backend-private.h" + +#ifdef WIN32 +# include +#else +# include +# include +# include +#endif /* WIN32 */ + + +/* + * Local functions... + */ + +void list_devices(void); +int print_device(const char *uri, const char *hostname, + const char *resource, char *options, + int print_fd, int copies, int argc, char *argv[]); + + +/* + * Include the vendor-specific USB implementation... + */ + +#ifdef HAVE_USB_H +# include "usb-libusb.c" +#elif defined(__APPLE__) +# include "usb-darwin.c" +#elif defined(__linux) || defined(__sun) || defined(__FreeBSD__) || defined(__NetBSD__) || defined(__OpenBSD__) || defined(__DragonFly__) || defined(__FreeBSD_kernel__) +# include "usb-unix.c" +#else +/* + * Use dummy functions that do nothing on unsupported platforms... + * These can be used as templates for implementing USB printing on new + * platforms... + */ + +/* + * 'list_devices()' - List all available USB devices to stdout. + */ + +void +list_devices(void) +{ + /* + * Don't have any devices to list... Use output of the form: + * + * direct usb:/make/model?serial=foo "Make Model" "USB Printer" + * + * Note that "Hewlett Packard" or any other variation MUST be mapped to + * "HP" for compatibility with the PPD and ICC specs. + */ +} + + +/* + * 'print_device()' - Print a file to a USB device. + */ + +int /* O - Exit status */ +print_device(const char *uri, /* I - Device URI */ + const char *hostname, /* I - Hostname/manufacturer */ + const char *resource, /* I - Resource/modelname */ + char *options, /* I - Device options/serial number */ + int print_fd, /* I - File descriptor to print */ + int copies, /* I - Copies to print */ + int argc, /* I - Number of command-line arguments (6 or 7) */ + char *argv[]) /* I - Command-line arguments */ +{ + /* + * Can't print, so just reference the arguments to eliminate compiler + * warnings and return and exit status of 1. Normally you would use the + * arguments to send a file to the printer and return 0 if everything + * worked OK and non-zero if there was an error. + */ + + (void)uri; + (void)hostname; + (void)resource; + (void)options; + (void)print_fd; + (void)copies; + (void)argc; + (void)argv; + + return (CUPS_BACKEND_FAILED); +} +#endif /* __APPLE__ */ + + +/* + * 'main()' - Send a file to the specified USB port. + * + * Usage: + * + * printer-uri job-id user title copies options [file] + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments (6 or 7) */ + char *argv[]) /* I - Command-line arguments */ +{ + int print_fd; /* Print file */ + int copies; /* Number of copies to print */ + int status; /* Exit status */ + int port; /* Port number (not used) */ + const char *uri; /* Device URI */ + char method[255], /* Method in URI */ + hostname[1024], /* Hostname */ + username[255], /* Username info (not used) */ + resource[1024], /* Resource info (device and options) */ + *options; /* Pointer to options */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* Actions for POSIX signals */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ + + + /* + * Make sure status messages are not buffered... + */ + + setbuf(stderr, NULL); + + /* + * Ignore SIGPIPE signals... + */ + +#ifdef HAVE_SIGSET + sigset(SIGPIPE, SIG_IGN); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + action.sa_handler = SIG_IGN; + sigaction(SIGPIPE, &action, NULL); +#else + signal(SIGPIPE, SIG_IGN); +#endif /* HAVE_SIGSET */ + + /* + * Check command-line... + */ + + if (argc == 1) + { + list_devices(); + return (CUPS_BACKEND_OK); + } + else if (argc < 6 || argc > 7) + { + _cupsLangPrintf(stderr, + _("Usage: %s job-id user title copies options [file]"), + argv[0]); + return (CUPS_BACKEND_FAILED); + } + + /* + * Extract the device name and options from the URI... + */ + + uri = cupsBackendDeviceURI(argv); + + if (httpSeparateURI(HTTP_URI_CODING_ALL, uri, + method, sizeof(method), username, sizeof(username), + hostname, sizeof(hostname), &port, + resource, sizeof(resource)) < HTTP_URI_OK) + { + _cupsLangPrintFilter(stderr, "ERROR", + _("No device URI found in argv[0] or in DEVICE_URI " + "environment variable.")); + return (1); + } + + /* + * See if there are any options... + */ + + if ((options = strchr(resource, '?')) != NULL) + { + /* + * Yup, terminate the device name string and move to the first + * character of the options... + */ + + *options++ = '\0'; + } + + /* + * If we have 7 arguments, print the file named on the command-line. + * Otherwise, send stdin instead... + */ + + if (argc == 6) + { + print_fd = 0; + copies = 1; + } + else + { + /* + * Try to open the print file... + */ + + if ((print_fd = open(argv[6], O_RDONLY)) < 0) + { + _cupsLangPrintError("ERROR", _("Unable to open print file")); + return (CUPS_BACKEND_FAILED); + } + + copies = atoi(argv[4]); + } + + /* + * Finally, send the print file... + */ + + status = print_device(uri, hostname, resource, options, print_fd, copies, + argc, argv); + + /* + * Close the input file and return... + */ + + if (print_fd != 0) + close(print_fd); + + return (status); +} + + +/* + * End of "$Id$". + */ diff --git a/berkeley/Dependencies b/berkeley/Dependencies new file mode 100644 index 0000000000..9beac1b39c --- /dev/null +++ b/berkeley/Dependencies @@ -0,0 +1,28 @@ +lpc.o: lpc.c ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +lpq.o: lpq.c ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +lpr.o: lpr.c ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +lprm.o: lprm.c ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h diff --git a/berkeley/Makefile b/berkeley/Makefile new file mode 100644 index 0000000000..e075228021 --- /dev/null +++ b/berkeley/Makefile @@ -0,0 +1,167 @@ +# +# "$Id$" +# +# Berkeley commands makefile for CUPS. +# +# Copyright 2007-2012 by Apple Inc. +# Copyright 1997-2006 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +include ../Makedefs + + +TARGETS = lpc lpq lpr lprm +OBJS = lpc.o lpq.o lpr.o lprm.o + + +# +# Make all targets... +# + +all: $(TARGETS) + + +# +# Make library targets... +# + +libs: + + +# +# Make unit tests... +# + +unittests: + + +# +# Clean all object files... +# + +clean: + $(RM) $(OBJS) $(TARGETS) + + +# +# Update dependencies (without system header dependencies...) +# + +depend: + $(CC) -MM $(ALL_CFLAGS) $(OBJS:.o=.c) >Dependencies + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: + + +# +# Install programs... +# + +install-exec: + echo Installing Berkeley user printing commands in $(BINDIR)... + $(INSTALL_DIR) -m 755 $(BINDIR) + $(INSTALL_BIN) lpq $(BINDIR) + $(INSTALL_BIN) lpr $(BINDIR) + $(INSTALL_BIN) lprm $(BINDIR) + echo Installing Berkeley admin printing commands in $(BINDIR)... + $(INSTALL_DIR) -m 755 $(SBINDIR) + $(INSTALL_BIN) lpc $(SBINDIR) + if test "x$(SYMROOT)" != "x"; then \ + $(INSTALL_DIR) $(SYMROOT); \ + for file in $(TARGETS); do \ + cp $$file $(SYMROOT); \ + done \ + fi + + +# +# Install headers... +# + +install-headers: + + +# +# Install libraries... +# + +install-libs: + + +# +# Uninstall all targets... +# + +uninstall: + $(RM) $(BINDIR)/lpq + $(RM) $(BINDIR)/lpr + $(RM) $(BINDIR)/lprm + $(RM) $(SBINDIR)/lpc + -$(RMDIR) $(SBINDIR) + -$(RMDIR) $(BINDIR) + + +# +# lpc +# + +lpc: lpc.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o lpc lpc.o $(LIBS) + + +# +# lpq +# + +lpq: lpq.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o lpq lpq.o $(LIBS) + + +# +# lpr +# + +lpr: lpr.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o lpr lpr.o $(LIBS) + + +# +# lprm +# + +lprm: lprm.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o lprm lprm.o $(LIBS) + + +# +# Dependencies... +# + +include Dependencies + + +# +# End of "$Id$". +# diff --git a/berkeley/lpc.c b/berkeley/lpc.c new file mode 100644 index 0000000000..e410d0b584 --- /dev/null +++ b/berkeley/lpc.c @@ -0,0 +1,449 @@ +/* + * "$Id$" + * + * "lpc" command for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Parse options and commands. + * compare_strings() - Compare two command-line strings. + * do_command() - Do an lpc command... + * show_help() - Show help messages. + * show_status() - Show printers. + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * Local functions... + */ + +static int compare_strings(const char *, const char *, int); +static void do_command(http_t *, const char *, const char *); +static void show_help(const char *); +static void show_status(http_t *, const char *); + + +/* + * 'main()' - Parse options and commands. + */ + +int +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + http_t *http; /* Connection to server */ + char line[1024], /* Input line from user */ + *params; /* Pointer to parameters */ + + + _cupsSetLocale(argv); + + /* + * Connect to the scheduler... + */ + + http = httpConnectEncrypt(cupsServer(), ippPort(), cupsEncryption()); + + if (argc > 1) + { + /* + * Process a single command on the command-line... + */ + + do_command(http, argv[1], argv[2]); + } + else + { + /* + * Do the command prompt thing... + */ + + _cupsLangPuts(stdout, _("lpc> ")); /* TODO: Need no-newline version */ + while (fgets(line, sizeof(line), stdin) != NULL) + { + /* + * Strip trailing whitespace... + */ + + for (params = line + strlen(line) - 1; params >= line;) + if (!isspace(*params & 255)) + break; + else + *params-- = '\0'; + + /* + * Strip leading whitespace... + */ + + for (params = line; isspace(*params & 255); params ++); + + if (params > line) + _cups_strcpy(line, params); + + if (!line[0]) + { + /* + * Nothing left, just show a prompt... + */ + + _cupsLangPuts(stdout, _("lpc> ")); /* TODO: Need no newline version */ + continue; + } + + /* + * Find any options in the string... + */ + + for (params = line; *params != '\0'; params ++) + if (isspace(*params & 255)) + break; + + /* + * Remove whitespace between the command and parameters... + */ + + while (isspace(*params & 255)) + *params++ = '\0'; + + /* + * The "quit" and "exit" commands exit; otherwise, process as needed... + */ + + if (!compare_strings(line, "quit", 1) || + !compare_strings(line, "exit", 2)) + break; + + if (*params == '\0') + do_command(http, line, NULL); + else + do_command(http, line, params); + + /* + * Put another prompt out to the user... + */ + + _cupsLangPuts(stdout, _("lpc> ")); /* TODO: Need no newline version */ + } + } + + /* + * Close the connection to the server and return... + */ + + httpClose(http); + + return (0); +} + + +/* + * 'compare_strings()' - Compare two command-line strings. + */ + +static int /* O - -1 or 1 = no match, 0 = match */ +compare_strings(const char *s, /* I - Command-line string */ + const char *t, /* I - Option string */ + int tmin) /* I - Minimum number of unique chars in option */ +{ + int slen; /* Length of command-line string */ + + + slen = strlen(s); + if (slen < tmin) + return (-1); + else + return (strncmp(s, t, slen)); +} + + +/* + * 'do_command()' - Do an lpc command... + */ + +static void +do_command(http_t *http, /* I - HTTP connection to server */ + const char *command, /* I - Command string */ + const char *params) /* I - Parameters for command */ +{ + if (!compare_strings(command, "status", 4)) + show_status(http, params); + else if (!compare_strings(command, "help", 1) || !strcmp(command, "?")) + show_help(params); + else + _cupsLangPrintf(stdout, + _("%s is not implemented by the CUPS version of lpc."), + command); +} + + +/* + * 'show_help()' - Show help messages. + */ + +static void +show_help(const char *command) /* I - Command to describe or NULL */ +{ + if (!command) + { + _cupsLangPrintf(stdout, + _("Commands may be abbreviated. Commands are:\n" + "\n" + "exit help quit status ?")); + } + else if (!compare_strings(command, "help", 1) || !strcmp(command, "?")) + _cupsLangPrintf(stdout, _("help\t\tGet help on commands.")); + else if (!compare_strings(command, "status", 4)) + _cupsLangPrintf(stdout, _("status\t\tShow status of daemon and queue.")); + else + _cupsLangPrintf(stdout, _("?Invalid help command unknown.")); +} + + +/* + * 'show_status()' - Show printers. + */ + +static void +show_status(http_t *http, /* I - HTTP connection to server */ + const char *dests) /* I - Destinations */ +{ + ipp_t *request, /* IPP Request */ + *response; /* IPP Response */ + ipp_attribute_t *attr; /* Current attribute */ + char *printer, /* Printer name */ + *device, /* Device URI */ + *delimiter; /* Char search result */ + ipp_pstate_t pstate; /* Printer state */ + int accepting; /* Is printer accepting jobs? */ + int jobcount; /* Count of current jobs */ + const char *dptr, /* Pointer into destination list */ + *ptr; /* Pointer into printer name */ + int match; /* Non-zero if this job matches */ + static const char *requested[] = /* Requested attributes */ + { + "device-uri", + "printer-is-accepting-jobs", + "printer-name", + "printer-state", + "queued-job-count" + }; + + + DEBUG_printf(("show_status(http=%p, dests=\"%s\")\n", http, dests)); + + if (http == NULL) + return; + + /* + * Build a CUPS_GET_PRINTERS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + */ + + request = ippNewRequest(CUPS_GET_PRINTERS); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", sizeof(requested) / sizeof(requested[0]), + NULL, requested); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + DEBUG_puts("show_status: request succeeded..."); + + /* + * Loop through the printers returned in the list and display + * their status... + */ + + for (attr = response->attrs; attr != NULL; attr = attr->next) + { + /* + * Skip leading attributes until we hit a job... + */ + + while (attr != NULL && attr->group_tag != IPP_TAG_PRINTER) + attr = attr->next; + + if (attr == NULL) + break; + + /* + * Pull the needed attributes from this job... + */ + + printer = NULL; + device = "file:/dev/null"; + pstate = IPP_PRINTER_IDLE; + jobcount = 0; + accepting = 1; + + while (attr != NULL && attr->group_tag == IPP_TAG_PRINTER) + { + if (!strcmp(attr->name, "device-uri") && + attr->value_tag == IPP_TAG_URI) + device = attr->values[0].string.text; + else if (!strcmp(attr->name, "printer-is-accepting-jobs") && + attr->value_tag == IPP_TAG_BOOLEAN) + accepting = attr->values[0].boolean; + else if (!strcmp(attr->name, "printer-name") && + attr->value_tag == IPP_TAG_NAME) + printer = attr->values[0].string.text; + else if (!strcmp(attr->name, "printer-state") && + attr->value_tag == IPP_TAG_ENUM) + pstate = (ipp_pstate_t)attr->values[0].integer; + else if (!strcmp(attr->name, "queued-job-count") && + attr->value_tag == IPP_TAG_INTEGER) + jobcount = attr->values[0].integer; + + attr = attr->next; + } + + /* + * See if we have everything needed... + */ + + if (printer == NULL) + { + if (attr == NULL) + break; + else + continue; + } + + /* + * A single 'all' printer name is special, meaning all printers. + */ + + if (dests != NULL && !strcmp(dests, "all")) + dests = NULL; + + /* + * See if this is a printer we're interested in... + */ + + match = dests == NULL; + + if (dests != NULL) + { + for (dptr = dests; *dptr != '\0';) + { + /* + * Skip leading whitespace and commas... + */ + + while (isspace(*dptr & 255) || *dptr == ',') + dptr ++; + + if (*dptr == '\0') + break; + + /* + * Compare names... + */ + + for (ptr = printer; + *ptr != '\0' && *dptr != '\0' && *ptr == *dptr; + ptr ++, dptr ++); + + if (*ptr == '\0' && (*dptr == '\0' || *dptr == ',' || + isspace(*dptr & 255))) + { + match = 1; + break; + } + + /* + * Skip trailing junk... + */ + + while (!isspace(*dptr & 255) && *dptr != '\0') + dptr ++; + while (isspace(*dptr & 255) || *dptr == ',') + dptr ++; + + if (*dptr == '\0') + break; + } + } + + /* + * Display the printer entry if needed... + */ + + if (match) + { + /* + * Display it... + */ + + printf("%s:\n", printer); + if (!strncmp(device, "file:", 5)) + _cupsLangPrintf(stdout, + _("\tprinter is on device \'%s\' speed -1"), + device + 5); + else + { + /* + * Just show the scheme... + */ + + if ((delimiter = strchr(device, ':')) != NULL ) + { + *delimiter = '\0'; + _cupsLangPrintf(stdout, + _("\tprinter is on device \'%s\' speed -1"), + device); + } + } + + if (accepting) + _cupsLangPuts(stdout, _("\tqueuing is enabled")); + else + _cupsLangPuts(stdout, _("\tqueuing is disabled")); + + if (pstate != IPP_PRINTER_STOPPED) + _cupsLangPuts(stdout, _("\tprinting is enabled")); + else + _cupsLangPuts(stdout, _("\tprinting is disabled")); + + if (jobcount == 0) + _cupsLangPuts(stdout, _("\tno entries")); + else + _cupsLangPrintf(stdout, _("\t%d entries"), jobcount); + + _cupsLangPuts(stdout, _("\tdaemon present")); + } + + if (attr == NULL) + break; + } + + ippDelete(response); + } +} + + +/* + * End of "$Id$". + */ diff --git a/berkeley/lpq.c b/berkeley/lpq.c new file mode 100644 index 0000000000..59f76fc216 --- /dev/null +++ b/berkeley/lpq.c @@ -0,0 +1,678 @@ +/* + * "$Id$" + * + * "lpq" command for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Parse options and commands. + * show_jobs() - Show jobs. + * show_printer() - Show printer status. + * usage() - Show program usage. + */ + +/* + * Include necessary headers... + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * Local functions... + */ + +static http_t *connect_server(const char *, http_t *); +static int show_jobs(const char *, http_t *, const char *, + const char *, const int, const int); +static void show_printer(const char *, http_t *, const char *); +static void usage(void) __attribute__((noreturn)); + + +/* + * 'main()' - Parse options and commands. + */ + +int +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + http_t *http; /* Connection to server */ + const char *dest, /* Desired printer */ + *user, /* Desired user */ + *val; /* Environment variable name */ + char *instance; /* Printer instance */ + int id, /* Desired job ID */ + all, /* All printers */ + interval, /* Reporting interval */ + longstatus; /* Show file details */ + int num_dests; /* Number of destinations */ + cups_dest_t *dests; /* Destinations */ + + + _cupsSetLocale(argv); + + /* + * Check for command-line options... + */ + + http = NULL; + dest = NULL; + user = NULL; + id = 0; + interval = 0; + longstatus = 0; + all = 0; + num_dests = 0; + dests = NULL; + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '+') + interval = atoi(argv[i] + 1); + else if (argv[i][0] == '-') + { + switch (argv[i][1]) + { + case 'E' : /* Encrypt */ +#ifdef HAVE_SSL + cupsSetEncryption(HTTP_ENCRYPT_REQUIRED); + + if (http) + httpEncryption(http, HTTP_ENCRYPT_REQUIRED); +#else + _cupsLangPrintf(stderr, _("%s: Sorry, no encryption support."), + argv[0]); +#endif /* HAVE_SSL */ + break; + + case 'U' : /* Username */ + if (argv[i][2] != '\0') + cupsSetUser(argv[i] + 2); + else + { + i ++; + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected username after " + "\"-U\" option."), argv[0]); + return (1); + } + + cupsSetUser(argv[i]); + } + break; + + case 'P' : /* Printer */ + if (argv[i][2]) + dest = argv[i] + 2; + else + { + i ++; + + if (i >= argc) + { + httpClose(http); + cupsFreeDests(num_dests, dests); + + usage(); + } + + dest = argv[i]; + } + + if ((instance = strchr(dest, '/')) != NULL) + *instance++ = '\0'; + + http = connect_server(argv[0], http); + + if (num_dests == 0) + num_dests = cupsGetDests2(http, &dests); + + if (cupsGetDest(dest, instance, num_dests, dests) == NULL) + { + if (instance) + _cupsLangPrintf(stderr, + _("%s: Error - unknown destination \"%s/%s\"."), + argv[0], dest, instance); + else + _cupsLangPrintf(stderr, _("%s: Unknown destination \"%s\"."), + argv[0], dest); + + return (1); + } + break; + + case 'a' : /* All printers */ + all = 1; + break; + + case 'h' : /* Connect to host */ + if (http) + { + httpClose(http); + http = NULL; + } + + if (argv[i][2] != '\0') + cupsSetServer(argv[i] + 2); + else + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected hostname after " + "\"-h\" option."), argv[0]); + return (1); + } + else + cupsSetServer(argv[i]); + } + break; + + case 'l' : /* Long status */ + longstatus = 1; + break; + + default : + httpClose(http); + cupsFreeDests(num_dests, dests); + + usage(); + break; + } + } + else if (isdigit(argv[i][0] & 255)) + id = atoi(argv[i]); + else + user = argv[i]; + + http = connect_server(argv[0], http); + + if (dest == NULL && !all) + { + if (num_dests == 0) + num_dests = cupsGetDests2(http, &dests); + + for (i = 0; i < num_dests; i ++) + if (dests[i].is_default) + dest = dests[i].name; + + if (dest == NULL) + { + val = NULL; + + if ((dest = getenv("LPDEST")) == NULL) + { + if ((dest = getenv("PRINTER")) != NULL) + { + if (!strcmp(dest, "lp")) + dest = NULL; + else + val = "PRINTER"; + } + } + else + val = "LPDEST"; + + if (dest && !cupsGetDest(dest, NULL, num_dests, dests)) + _cupsLangPrintf(stderr, + _("%s: Error - %s environment variable names " + "non-existent destination \"%s\"."), argv[0], val, + dest); + else + _cupsLangPrintf(stderr, + _("%s: Error - no default destination available."), + argv[0]); + httpClose(http); + cupsFreeDests(num_dests, dests); + return (1); + } + } + + /* + * Show the status in a loop... + */ + + for (;;) + { + if (dest) + show_printer(argv[0], http, dest); + + i = show_jobs(argv[0], http, dest, user, id, longstatus); + + if (i && interval) + { + fflush(stdout); + sleep(interval); + } + else + break; + } + + /* + * Close the connection to the server and return... + */ + + cupsFreeDests(num_dests, dests); + httpClose(http); + + return (0); +} + + +/* + * 'connect_server()' - Connect to the server as necessary... + */ + +static http_t * /* O - New HTTP connection */ +connect_server(const char *command, /* I - Command name */ + http_t *http) /* I - Current HTTP connection */ +{ + if (!http) + { + http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption()); + + if (http == NULL) + { + _cupsLangPrintf(stderr, _("%s: Unable to connect to server."), command); + exit(1); + } + } + + return (http); +} + + +/* + * 'show_jobs()' - Show jobs. + */ + +static int /* O - Number of jobs in queue */ +show_jobs(const char *command, /* I - Command name */ + http_t *http, /* I - HTTP connection to server */ + const char *dest, /* I - Destination */ + const char *user, /* I - User */ + const int id, /* I - Job ID */ + const int longstatus) /* I - 1 if long report desired */ +{ + ipp_t *request, /* IPP Request */ + *response; /* IPP Response */ + ipp_attribute_t *attr; /* Current attribute */ + const char *jobdest, /* Pointer into job-printer-uri */ + *jobuser, /* Pointer to job-originating-user-name */ + *jobname; /* Pointer to job-name */ + ipp_jstate_t jobstate; /* job-state */ + int jobid, /* job-id */ + jobsize, /* job-k-octets */ +#ifdef __osf__ + jobpriority, /* job-priority */ +#endif /* __osf__ */ + jobcount, /* Number of jobs */ + jobcopies, /* Number of copies */ + rank; /* Rank of job */ + char resource[1024]; /* Resource string */ + char rankstr[255]; /* Rank string */ + char namestr[1024]; /* Job name string */ + static const char * const jobattrs[] =/* Job attributes we want to see */ + { + "copies", + "job-id", + "job-k-octets", + "job-name", + "job-originating-user-name", + "job-printer-uri", + "job-priority", + "job-state" + }; + static const char * const ranks[10] = /* Ranking strings */ + { + "th", + "st", + "nd", + "rd", + "th", + "th", + "th", + "th", + "th", + "th" + }; + + + DEBUG_printf(("show_jobs(http=%p, dest=%p, user=%p, id=%d, longstatus%d)\n", + http, dest, user, id, longstatus)); + + if (http == NULL) + return (0); + + /* + * Build an IPP_GET_JOBS or IPP_GET_JOB_ATTRIBUTES request, which requires + * the following attributes: + * + * attributes-charset + * attributes-natural-language + * job-uri or printer-uri + * requested-attributes + */ + + request = ippNewRequest(id ? IPP_GET_JOB_ATTRIBUTES : IPP_GET_JOBS); + + if (id) + { + snprintf(resource, sizeof(resource), "ipp://localhost/jobs/%d", id); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-uri", + NULL, resource); + } + else if (dest) + { + httpAssembleURIf(HTTP_URI_CODING_ALL, resource, sizeof(resource), "ipp", + NULL, "localhost", 0, "/printers/%s", dest); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, resource); + } + else + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, "ipp://localhost/"); + + if (user) + { + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, user); + ippAddBoolean(request, IPP_TAG_OPERATION, "my-jobs", 1); + } + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", + (int)(sizeof(jobattrs) / sizeof(jobattrs[0])), NULL, jobattrs); + + /* + * Do the request and get back a response... + */ + + jobcount = 0; + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + if (response->request.status.status_code > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, "%s: %s", command, cupsLastErrorString()); + ippDelete(response); + return (0); + } + + rank = 1; + + /* + * Loop through the job list and display them... + */ + + for (attr = response->attrs; attr != NULL; attr = attr->next) + { + /* + * Skip leading attributes until we hit a job... + */ + + while (attr != NULL && attr->group_tag != IPP_TAG_JOB) + attr = attr->next; + + if (attr == NULL) + break; + + /* + * Pull the needed attributes from this job... + */ + + jobid = 0; + jobsize = 0; +#ifdef __osf__ + jobpriority = 50; +#endif /* __osf__ */ + jobstate = IPP_JOB_PENDING; + jobname = "untitled"; + jobuser = NULL; + jobdest = NULL; + jobcopies = 1; + + while (attr != NULL && attr->group_tag == IPP_TAG_JOB) + { + if (!strcmp(attr->name, "job-id") && + attr->value_tag == IPP_TAG_INTEGER) + jobid = attr->values[0].integer; + + if (!strcmp(attr->name, "job-k-octets") && + attr->value_tag == IPP_TAG_INTEGER) + jobsize = attr->values[0].integer; + +#ifdef __osf__ + if (!strcmp(attr->name, "job-priority") && + attr->value_tag == IPP_TAG_INTEGER) + jobpriority = attr->values[0].integer; +#endif /* __osf__ */ + + if (!strcmp(attr->name, "job-state") && + attr->value_tag == IPP_TAG_ENUM) + jobstate = (ipp_jstate_t)attr->values[0].integer; + + if (!strcmp(attr->name, "job-printer-uri") && + attr->value_tag == IPP_TAG_URI) + if ((jobdest = strrchr(attr->values[0].string.text, '/')) != NULL) + jobdest ++; + + if (!strcmp(attr->name, "job-originating-user-name") && + attr->value_tag == IPP_TAG_NAME) + jobuser = attr->values[0].string.text; + + if (!strcmp(attr->name, "job-name") && + attr->value_tag == IPP_TAG_NAME) + jobname = attr->values[0].string.text; + + if (!strcmp(attr->name, "copies") && + attr->value_tag == IPP_TAG_INTEGER) + jobcopies = attr->values[0].integer; + + attr = attr->next; + } + + /* + * See if we have everything needed... + */ + + if (jobdest == NULL || jobid == 0) + { + if (attr == NULL) + break; + else + continue; + } + + if (!longstatus && jobcount == 0) +#ifdef __osf__ + _cupsLangPuts(stdout, + /* TRANSLATORS: Pri is job priority. */ + _("Rank Owner Pri Job Files" + " Total Size")); +#else + _cupsLangPuts(stdout, + _("Rank Owner Job File(s)" + " Total Size")); +#endif /* __osf__ */ + + jobcount ++; + + /* + * Display the job... + */ + + if (jobstate == IPP_JOB_PROCESSING) + strcpy(rankstr, "active"); + else + { + /* + * Make the rank show the "correct" suffix for each number + * (11-13 are the only special cases, for English anyways...) + */ + + if ((rank % 100) >= 11 && (rank % 100) <= 13) + snprintf(rankstr, sizeof(rankstr), "%dth", rank); + else + snprintf(rankstr, sizeof(rankstr), "%d%s", rank, ranks[rank % 10]); + + rank ++; + } + + if (longstatus) + { + _cupsLangPuts(stdout, "\n"); + + if (jobcopies > 1) + snprintf(namestr, sizeof(namestr), "%d copies of %s", jobcopies, + jobname); + else + strlcpy(namestr, jobname, sizeof(namestr)); + + _cupsLangPrintf(stdout, _("%s: %-33.33s [job %d localhost]"), + jobuser, rankstr, jobid); + _cupsLangPrintf(stdout, _(" %-39.39s %.0f bytes"), + namestr, 1024.0 * jobsize); + } + else +#ifdef __osf__ + _cupsLangPrintf(stdout, + _("%-6s %-10.10s %-4d %-10d %-27.27s %.0f bytes"), + rankstr, jobuser, jobpriority, jobid, jobname, + 1024.0 * jobsize); +#else + _cupsLangPrintf(stdout, + _("%-7s %-7.7s %-7d %-31.31s %.0f bytes"), + rankstr, jobuser, jobid, jobname, 1024.0 * jobsize); +#endif /* __osf */ + + if (attr == NULL) + break; + } + + ippDelete(response); + } + else + { + _cupsLangPrintf(stderr, "%s: %s", command, cupsLastErrorString()); + return (0); + } + + if (jobcount == 0) + _cupsLangPuts(stdout, _("no entries")); + + return (jobcount); +} + + +/* + * 'show_printer()' - Show printer status. + */ + +static void +show_printer(const char *command, /* I - Command name */ + http_t *http, /* I - HTTP connection to server */ + const char *dest) /* I - Destination */ +{ + ipp_t *request, /* IPP Request */ + *response; /* IPP Response */ + ipp_attribute_t *attr; /* Current attribute */ + ipp_pstate_t state; /* Printer state */ + char uri[HTTP_MAX_URI]; /* Printer URI */ + + + if (http == NULL) + return; + + /* + * Build an IPP_GET_PRINTER_ATTRIBUTES request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + */ + + request = ippNewRequest(IPP_GET_PRINTER_ATTRIBUTES); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", dest); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, uri); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + if (response->request.status.status_code > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, "%s: %s", command, cupsLastErrorString()); + ippDelete(response); + return; + } + + if ((attr = ippFindAttribute(response, "printer-state", IPP_TAG_ENUM)) != NULL) + state = (ipp_pstate_t)attr->values[0].integer; + else + state = IPP_PRINTER_STOPPED; + + switch (state) + { + case IPP_PRINTER_IDLE : + _cupsLangPrintf(stdout, _("%s is ready"), dest); + break; + case IPP_PRINTER_PROCESSING : + _cupsLangPrintf(stdout, _("%s is ready and printing"), + dest); + break; + case IPP_PRINTER_STOPPED : + _cupsLangPrintf(stdout, _("%s is not ready"), dest); + break; + } + + ippDelete(response); + } + else + _cupsLangPrintf(stderr, "%s: %s", command, cupsLastErrorString()); +} + + +/* + * 'usage()' - Show program usage. + */ + +static void +usage(void) +{ + _cupsLangPuts(stderr, + _("Usage: lpq [-P dest] [-U username] [-h hostname[:port]] " + "[-l] [+interval]")); + exit(1); +} + + +/* + * End of "$Id$". + */ diff --git a/berkeley/lpr.c b/berkeley/lpr.c new file mode 100644 index 0000000000..bf46820577 --- /dev/null +++ b/berkeley/lpr.c @@ -0,0 +1,417 @@ +/* + * "$Id$" + * + * "lpr" command for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Parse options and send files for printing. + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * 'main()' - Parse options and send files for printing. + */ + +int +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i, j; /* Looping var */ + int job_id; /* Job ID */ + char ch; /* Option character */ + char *printer, /* Destination printer or class */ + *instance; /* Instance */ + const char *title, /* Job title */ + *val; /* Environment variable name */ + int num_copies; /* Number of copies per file */ + int num_files; /* Number of files to print */ + const char *files[1000]; /* Files to print */ + cups_dest_t *dest; /* Selected destination */ + int num_options; /* Number of options */ + cups_option_t *options; /* Options */ + int deletefile; /* Delete file after print? */ + char buffer[8192]; /* Copy buffer */ + + + _cupsSetLocale(argv); + + deletefile = 0; + printer = NULL; + dest = NULL; + num_options = 0; + options = NULL; + num_files = 0; + title = NULL; + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-') + switch (ch = argv[i][1]) + { + case 'E' : /* Encrypt */ +#ifdef HAVE_SSL + cupsSetEncryption(HTTP_ENCRYPT_REQUIRED); +#else + _cupsLangPrintf(stderr, _("%s: Sorry, no encryption support."), + argv[0]); +#endif /* HAVE_SSL */ + break; + + case 'U' : /* Username */ + if (argv[i][2] != '\0') + cupsSetUser(argv[i] + 2); + else + { + i ++; + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected username after " + "\"-U\" option."), argv[0]); + return (1); + } + + cupsSetUser(argv[i]); + } + break; + + case 'H' : /* Connect to host */ + if (argv[i][2] != '\0') + cupsSetServer(argv[i] + 2); + else + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected hostname after " + "\"-H\" option."), argv[0]); + return (1); + } + else + cupsSetServer(argv[i]); + } + break; + + case '1' : /* TROFF font set 1 */ + case '2' : /* TROFF font set 2 */ + case '3' : /* TROFF font set 3 */ + case '4' : /* TROFF font set 4 */ + case 'i' : /* indent */ + case 'w' : /* width */ + if (argv[i][2] == '\0') + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected value after \"-%c\" " + "option."), argv[0], ch); + return (1); + } + } + + case 'c' : /* CIFPLOT */ + case 'd' : /* DVI */ + case 'f' : /* FORTRAN */ + case 'g' : /* plot */ + case 'n' : /* Ditroff */ + case 't' : /* Troff */ + case 'v' : /* Raster image */ + _cupsLangPrintf(stderr, + _("%s: Warning - \"%c\" format modifier not " + "supported - output may not be correct."), + argv[0], ch); + break; + + case 'o' : /* Option */ + if (argv[i][2] != '\0') + num_options = cupsParseOptions(argv[i] + 2, num_options, &options); + else + { + i ++; + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected option=value after " + "\"-o\" option."), argv[0]); + return (1); + } + + num_options = cupsParseOptions(argv[i], num_options, &options); + } + break; + + case 'l' : /* Literal/raw */ + num_options = cupsAddOption("raw", "true", num_options, &options); + break; + + case 'p' : /* Prettyprint */ + num_options = cupsAddOption("prettyprint", "true", num_options, + &options); + break; + + case 'h' : /* Suppress burst page */ + num_options = cupsAddOption("job-sheets", "none", num_options, + &options); + break; + + case 's' : /* Don't use symlinks */ + break; + + case 'm' : /* Mail on completion */ + { + char email[1024]; /* EMail address */ + + + snprintf(email, sizeof(email), "mailto:%s@%s", cupsUser(), + httpGetHostname(NULL, buffer, sizeof(buffer))); + num_options = cupsAddOption("notify-recipient-uri", email, + num_options, &options); + } + break; + + case 'q' : /* Queue file but don't print */ + num_options = cupsAddOption("job-hold-until", "indefinite", + num_options, &options); + break; + + case 'r' : /* Remove file after printing */ + deletefile = 1; + break; + + case 'P' : /* Destination printer or class */ + if (argv[i][2] != '\0') + printer = argv[i] + 2; + else + { + i ++; + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected destination after " + "\"-P\" option."), argv[0]); + return (1); + } + + printer = argv[i]; + } + + if ((instance = strrchr(printer, '/')) != NULL) + *instance++ = '\0'; + + if ((dest = cupsGetNamedDest(NULL, printer, instance)) != NULL) + { + for (j = 0; j < dest->num_options; j ++) + if (cupsGetOption(dest->options[j].name, num_options, + options) == NULL) + num_options = cupsAddOption(dest->options[j].name, + dest->options[j].value, + num_options, &options); + } + break; + + case '#' : /* Number of copies */ + if (argv[i][2] != '\0') + num_copies = atoi(argv[i] + 2); + else + { + i ++; + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected copies after " + "\"-#\" option."), argv[0]); + return (1); + } + + num_copies = atoi(argv[i]); + } + + sprintf(buffer, "%d", num_copies); + num_options = cupsAddOption("copies", buffer, num_options, &options); + break; + + case 'C' : /* Class */ + case 'J' : /* Job name */ + case 'T' : /* Title */ + if (argv[i][2] != '\0') + title = argv[i] + 2; + else + { + i ++; + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected name after \"-%c\" " + "option."), argv[0], ch); + return (1); + } + + title = argv[i]; + } + break; + + default : + _cupsLangPrintf(stderr, + _("%s: Error - unknown option \"%c\"."), argv[0], + argv[i][1]); + return (1); + } + else if (num_files < 1000) + { + /* + * Print a file... + */ + + if (access(argv[i], R_OK) != 0) + { + _cupsLangPrintf(stderr, + _("%s: Error - unable to access \"%s\" - %s"), + argv[0], argv[i], strerror(errno)); + return (1); + } + + files[num_files] = argv[i]; + num_files ++; + + if (title == NULL) + { + if ((title = strrchr(argv[i], '/')) != NULL) + title ++; + else + title = argv[i]; + } + } + else + _cupsLangPrintf(stderr, + _("%s: Error - too many files - \"%s\"."), argv[0], + argv[i]); + /* + * See if we have any files to print; if not, print from stdin... + */ + + if (printer == NULL) + { + if ((dest = cupsGetNamedDest(NULL, NULL, NULL)) != NULL) + { + printer = dest->name; + + for (j = 0; j < dest->num_options; j ++) + if (cupsGetOption(dest->options[j].name, num_options, options) == NULL) + num_options = cupsAddOption(dest->options[j].name, + dest->options[j].value, + num_options, &options); + } + } + + if (printer == NULL) + { + val = NULL; + + if ((printer = getenv("LPDEST")) == NULL) + { + if ((printer = getenv("PRINTER")) != NULL) + { + if (!strcmp(printer, "lp")) + printer = NULL; + else + val = "PRINTER"; + } + } + else + val = "LPDEST"; + + if (printer && !cupsGetNamedDest(NULL, printer, NULL)) + _cupsLangPrintf(stderr, + _("%s: Error - %s environment variable names " + "non-existent destination \"%s\"."), argv[0], val, + printer); + else if (cupsLastError() == IPP_NOT_FOUND) + _cupsLangPrintf(stderr, + _("%s: Error - no default destination available."), + argv[0]); + else + _cupsLangPrintf(stderr, _("%s: Error - scheduler not responding."), + argv[0]); + + return (1); + } + + if (num_files > 0) + { + job_id = cupsPrintFiles(printer, num_files, files, title, num_options, options); + + if (deletefile && job_id > 0) + { + /* + * Delete print files after printing... + */ + + for (i = 0; i < num_files; i ++) + unlink(files[i]); + } + } + else if ((job_id = cupsCreateJob(CUPS_HTTP_DEFAULT, printer, + title ? title : "(stdin)", + num_options, options)) > 0) + { + http_status_t status; /* Write status */ + const char *format; /* Document format */ + ssize_t bytes; /* Bytes read */ + + + if (cupsGetOption("raw", num_options, options)) + format = CUPS_FORMAT_RAW; + else if ((format = cupsGetOption("document-format", num_options, + options)) == NULL) + format = CUPS_FORMAT_AUTO; + + status = cupsStartDocument(CUPS_HTTP_DEFAULT, printer, job_id, NULL, + format, 1); + + while (status == HTTP_CONTINUE && + (bytes = read(0, buffer, sizeof(buffer))) > 0) + status = cupsWriteRequestData(CUPS_HTTP_DEFAULT, buffer, bytes); + + if (status != HTTP_CONTINUE) + { + _cupsLangPrintf(stderr, _("%s: Error - unable to queue from stdin - %s."), + argv[0], httpStatus(status)); + return (1); + } + + if (cupsFinishDocument(CUPS_HTTP_DEFAULT, printer) != IPP_OK) + job_id = 0; + } + + if (job_id < 1) + { + _cupsLangPrintf(stderr, "%s: %s", argv[0], cupsLastErrorString()); + return (1); + } + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/berkeley/lprm.c b/berkeley/lprm.c new file mode 100644 index 0000000000..e2d03e359f --- /dev/null +++ b/berkeley/lprm.c @@ -0,0 +1,217 @@ +/* + * "$Id$" + * + * "lprm" command for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Parse options and cancel jobs. + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * 'main()' - Parse options and cancel jobs. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + int job_id; /* Job ID */ + const char *name; /* Destination printer */ + char *instance; /* Pointer to instance name */ + cups_dest_t *dest, /* Destination */ + *defdest; /* Default destination */ + int did_cancel; /* Did we cancel something? */ + + + _cupsSetLocale(argv); + + /* + * Setup to cancel individual print jobs... + */ + + did_cancel = 0; + defdest = cupsGetNamedDest(CUPS_HTTP_DEFAULT, NULL, NULL); + name = defdest ? defdest->name : NULL; + + /* + * Process command-line arguments... + */ + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-' && argv[i][1] != '\0') + switch (argv[i][1]) + { + case 'E' : /* Encrypt */ +#ifdef HAVE_SSL + cupsSetEncryption(HTTP_ENCRYPT_REQUIRED); +#else + _cupsLangPrintf(stderr, + _("%s: Sorry, no encryption support."), argv[0]); +#endif /* HAVE_SSL */ + break; + + case 'P' : /* Cancel jobs on a printer */ + if (argv[i][2]) + name = argv[i] + 2; + else + { + i ++; + name = argv[i]; + } + + if ((instance = strchr(name, '/')) != NULL) + *instance = '\0'; + + if ((dest = cupsGetNamedDest(CUPS_HTTP_DEFAULT, name, + NULL)) == NULL) + { + _cupsLangPrintf(stderr, + _("%s: Error - unknown destination \"%s\"."), + argv[0], name); + goto error; + } + + cupsFreeDests(1, dest); + break; + + case 'U' : /* Username */ + if (argv[i][2] != '\0') + cupsSetUser(argv[i] + 2); + else + { + i ++; + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected username after " + "\"-U\" option."), argv[0]); + goto error; + } + + cupsSetUser(argv[i]); + } + break; + + case 'h' : /* Connect to host */ + if (argv[i][2] != '\0') + cupsSetServer(argv[i] + 2); + else + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected hostname after " + "\"-h\" option."), argv[0]); + goto error; + } + else + cupsSetServer(argv[i]); + } + + if (defdest) + cupsFreeDests(1, defdest); + + defdest = cupsGetNamedDest(CUPS_HTTP_DEFAULT, NULL, NULL); + name = defdest ? defdest->name : NULL; + break; + + default : + _cupsLangPrintf(stderr, _("%s: Error - unknown option \"%c\"."), + argv[0], argv[i][1]); + goto error; + } + else + { + /* + * Cancel a job or printer... + */ + + if ((dest = cupsGetNamedDest(CUPS_HTTP_DEFAULT, argv[i], NULL)) != NULL) + cupsFreeDests(1, dest); + + if (dest) + { + name = argv[i]; + job_id = 0; + } + else if (isdigit(argv[i][0] & 255)) + { + name = NULL; + job_id = atoi(argv[i]); + } + else if (!strcmp(argv[i], "-")) + { + /* + * Cancel all jobs + */ + + job_id = -1; + } + else + { + _cupsLangPrintf(stderr, _("%s: Error - unknown destination \"%s\"."), + argv[0], argv[i]); + goto error; + } + + if (cupsCancelJob2(CUPS_HTTP_DEFAULT, name, job_id, 0) != IPP_OK) + { + _cupsLangPrintf(stderr, "%s: %s", argv[0], cupsLastErrorString()); + goto error; + } + + did_cancel = 1; + } + + /* + * If nothing has been canceled yet, cancel the current job on the specified + * (or default) printer... + */ + + if (!did_cancel && cupsCancelJob2(CUPS_HTTP_DEFAULT, name, 0, 0) != IPP_OK) + { + _cupsLangPrintf(stderr, "%s: %s", argv[0], cupsLastErrorString()); + goto error; + } + + if (defdest) + cupsFreeDests(1, defdest); + + return (0); + + /* + * If we get here there was an error, so clean up... + */ + + error: + + if (defdest) + cupsFreeDests(1, defdest); + + return (1); +} + + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/Dependencies b/cgi-bin/Dependencies new file mode 100644 index 0000000000..ff16e48490 --- /dev/null +++ b/cgi-bin/Dependencies @@ -0,0 +1,73 @@ +help-index.o: help-index.c cgi-private.h cgi.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h help-index.h \ + ../cups/debug-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/string-private.h ../config.h ../cups/ipp-private.h \ + ../cups/dir.h +html.o: html.c cgi-private.h cgi.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h help-index.h ../cups/debug-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/string-private.h ../config.h ../cups/ipp-private.h +ipp-var.o: ipp-var.c cgi-private.h cgi.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h help-index.h ../cups/debug-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/string-private.h ../config.h ../cups/ipp-private.h +search.o: search.c cgi-private.h cgi.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h help-index.h ../cups/debug-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/string-private.h ../config.h ../cups/ipp-private.h +template.o: template.c cgi-private.h cgi.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h help-index.h ../cups/debug-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/string-private.h ../config.h ../cups/ipp-private.h +var.o: var.c cgi-private.h cgi.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h help-index.h ../cups/debug-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/string-private.h ../config.h ../cups/ipp-private.h \ + ../cups/md5-private.h +admin.o: admin.c cgi-private.h cgi.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h help-index.h ../cups/debug-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/string-private.h ../config.h ../cups/ipp-private.h \ + ../cups/adminutil.h ../cups/ppd.h +classes.o: classes.c cgi-private.h cgi.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h help-index.h ../cups/debug-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/string-private.h ../config.h ../cups/ipp-private.h +help.o: help.c cgi-private.h cgi.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h help-index.h ../cups/debug-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/string-private.h ../config.h ../cups/ipp-private.h +jobs.o: jobs.c cgi-private.h cgi.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h help-index.h ../cups/debug-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/string-private.h ../config.h ../cups/ipp-private.h +makedocset.o: makedocset.c cgi.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h help-index.h +printers.o: printers.c cgi-private.h cgi.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h help-index.h ../cups/debug-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/string-private.h ../config.h ../cups/ipp-private.h +testcgi.o: testcgi.c cgi.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h help-index.h +testhi.o: testhi.c cgi.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h help-index.h +testtemplate.o: testtemplate.c cgi.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h help-index.h +websearch.o: websearch.c cgi.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h help-index.h diff --git a/cgi-bin/Makefile b/cgi-bin/Makefile new file mode 100644 index 0000000000..05bf60107d --- /dev/null +++ b/cgi-bin/Makefile @@ -0,0 +1,382 @@ +# +# "$Id$" +# +# CGI makefile for CUPS. +# +# Copyright 2007-2012 by Apple Inc. +# Copyright 1997-2006 by Easy Software Products. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +include ../Makedefs + +LIBOBJS = \ + help-index.o \ + html.o \ + ipp-var.o \ + search.o \ + template.o \ + var.o +OBJS = \ + $(LIBOBJS) \ + admin.o \ + classes.o \ + help.o \ + jobs.o \ + makedocset.o \ + printers.o \ + testcgi.o \ + testhi.o \ + testtemplate.o \ + websearch.o +CGIS = \ + admin.cgi \ + classes.cgi \ + help.cgi \ + jobs.cgi \ + printers.cgi +LIBTARGETS = \ + libcupscgi.a \ + $(LIBCUPSCGI) \ + websearch + +UNITTARGETS = \ + testcgi \ + testhi \ + testtemplate + +TARGETS = \ + $(LIBTARGETS) \ + $(CGIS) + + +# +# Make all targets... +# + +all: $(TARGETS) + + +# +# Make library targets... +# + +libs: $(LIBTARGETS) $(UNITTESTS) + + +# +# Make unit tests... +# + +unittests: $(UNITTARGETS) + + +# +# Clean all object files... +# + +clean: + $(RM) $(OBJS) $(TARGETS) $(UNITTARGETS) makedocset + $(RM) libcupscgi.so libcupscgi.sl libcupscgi.dylib + + +# +# Update dependencies (without system header dependencies...) +# + +depend: + $(CC) -MM $(ALL_CFLAGS) $(OBJS:.o=.c) >Dependencies + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: + + +# +# Install programs... +# + +install-exec: + $(INSTALL_DIR) -m 755 $(SERVERBIN)/cgi-bin + for file in $(CGIS); do \ + $(INSTALL_BIN) $$file $(SERVERBIN)/cgi-bin; \ + done + if test "x$(SYMROOT)" != "x"; then \ + $(INSTALL_DIR) $(SYMROOT); \ + for file in $(CGIS); do \ + cp $$file $(SYMROOT); \ + done \ + fi + + +# +# Install headers... +# + +install-headers: + echo Installing header files in $(INCLUDEDIR)/cups... + $(INSTALL_DIR) -m 755 $(INCLUDEDIR)/cups + $(INSTALL_DATA) cgi.h $(INCLUDEDIR)/cups + $(INSTALL_DATA) help-index.h $(INCLUDEDIR)/cups + + +# +# Install libraries... +# + +install-libs: $(INSTALLSTATIC) + echo Installing libraries in $(LIBDIR)... + $(INSTALL_DIR) -m 755 $(LIBDIR) + $(INSTALL_LIB) $(LIBCUPSCGI) $(LIBDIR) + if test $(LIBCUPSCGI) = "libcupscgi.so.1" -o $(LIBCUPSCGI) = "libcupscgi.sl.1"; then \ + $(RM) $(LIBDIR)/`basename $(LIBCUPSCGI) .1`; \ + $(LN) $(LIBCUPSCGI) $(LIBDIR)/`basename $(LIBCUPSCGI) .1`; \ + fi + if test $(LIBCUPSCGI) = "libcupscgi.1.dylib"; then \ + $(RM) $(LIBDIR)/libcupscgi.dylib; \ + $(LN) $(LIBCUPSCGI) $(LIBDIR)/libcupscgi.dylib; \ + fi + if test "x$(SYMROOT)" != "x"; then \ + $(INSTALL_DIR) $(SYMROOT); \ + cp $(LIBCUPSCGI) $(SYMROOT); \ + fi + +installstatic: + $(INSTALL_DIR) -m 755 $(LIBDIR) + $(INSTALL_LIB) -m 755 libcupscgi.a $(LIBDIR) + $(RANLIB) $(LIBDIR)/libcupscgi.a + $(CHMOD) 555 $(LIBDIR)/libcupscgi.a + + +# +# Uninstall all targets... +# + +uninstall: + for file in $(CGIS); do \ + $(RM) $(SERVERBIN)/cgi-bin/$$file; \ + done + -$(RMDIR) $(SERVERBIN)/cgi-bin + $(RM) $(LIBDIR)/libcupscgi.1.dylib + $(RM) $(LIBDIR)/libcupscgi.a + $(RM) $(LIBDIR)/libcupscgi.dylib + $(RM) $(LIBDIR)/libcupscgi_s.a + $(RM) $(LIBDIR)/libcupscgi.sl + $(RM) $(LIBDIR)/libcupscgi.sl.1 + $(RM) $(LIBDIR)/libcupscgi.so + $(RM) $(LIBDIR)/libcupscgi.so.1 + -$(RMDIR) $(LIBDIR) + $(RM) $(INCLUDEDIR)/cups/cgi.h + $(RM) $(INCLUDEDIR)/cups/help-index.h + -$(RMDIR) $(INCLUDEDIR)/cups + + +# +# Automatic API help files... +# + +apihelp: + mxmldoc --section "Programming" \ + --title "CGI API" \ + --css ../doc/cups-printable.css \ + --header api-cgi.header --intro api-cgi.shtml \ + cgi.h help-index.h $(LIBOBJS:.o=.c) >../doc/help/api-cgi.html + mxmldoc --tokens help/api-cgi.html api-cgi.xml >../doc/help/api-cgi.tokens + $(RM) api-cgi.xml + +framedhelp: + mxmldoc --framed api-cgi \ + --section "Programming" \ + --title "CGI API" \ + --css ../doc/cups-printable.css \ + --header api-cgi.header --intro api-cgi.shtml \ + cgi.h help-index.h $(LIBOBJS:.o=.c) + + +# +# libcupscgi.so.1, libcupscgi.sl.1 +# + +libcupscgi.so.1 libcupscgi.sl.1: $(LIBOBJS) + echo Linking $@... + $(DSO) $(ARCHFLAGS) $(DSOFLAGS) -o $@ $(LIBOBJS) $(LIBS) + $(RM) `basename $@ .1` + $(LN) $@ `basename $@ .1` + + +# +# libcupscgi.1.dylib +# + +libcupscgi.1.dylib: $(LIBOBJS) libcupscgi.exp + echo Linking $@... + $(DSO) $(ARCHFLAGS) $(DSOFLAGS) -o $@ \ + -install_name $(libdir)/$@ \ + -current_version 1.0.0 \ + -compatibility_version 1.0.0 \ + -exported_symbols_list libcupscgi.exp \ + $(LIBOBJS) $(LIBS) + $(RM) libcupscgi.dylib + $(LN) $@ libcupscgi.dylib + + +# +# libcupscgi_s.a +# + +libcupscgi_s.a: $(LIBOBJS) + echo Creating $@... + $(DSO) $(DSOFLAGS) -o libcupscgi_s.o $(LIBOBJS) $(LIBS) + $(RM) $@ + $(AR) $(ARFLAGS) $@ libcupscgi_s.o + + +# +# libcupscgi.la +# + +libcupscgi.la: $(LIBOBJS) + echo Linking $@... + $(CC) $(ARCHFLAGS) $(DSOFLAGS) -o $@ $(LIBOBJS:.o=.lo) -rpath $(LIBDIR) \ + -version-info 1:0 $(LIBS) + + +# +# libcupscgi.a +# + +libcupscgi.a: $(LIBOBJS) + echo Archiving $@... + $(RM) $@ + $(AR) $(ARFLAGS) $@ $(LIBOBJS) + $(RANLIB) $@ + + +# +# admin.cgi +# + +admin.cgi: admin.o ../Makedefs ../cups/$(LIBCUPS) $(LIBCUPSCGI) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ admin.o -lcupscgi $(LIBS) + + +# +# classes.cgi +# + +classes.cgi: classes.o ../Makedefs ../cups/$(LIBCUPS) $(LIBCUPSCGI) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ classes.o -lcupscgi $(LIBS) + + +# +# help.cgi +# + +help.cgi: help.o ../Makedefs ../cups/$(LIBCUPS) $(LIBCUPSCGI) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ help.o -lcupscgi $(LIBS) + + +# +# jobs.cgi +# + +jobs.cgi: jobs.o ../Makedefs ../cups/$(LIBCUPS) $(LIBCUPSCGI) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ jobs.o -lcupscgi $(LIBS) + + +# +# makedocset +# + +makedocset: makedocset.o ../Makedefs libcupscgi.a ../cups/$(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o $@ makedocset.o libcupscgi.a \ + ../cups/$(LIBCUPSSTATIC) $(COMMONLIBS) $(SSLLIBS) $(DNSSDLIBS) \ + $(LIBZ) $(LIBGSSAPI) + + +# +# printers.cgi +# + +printers.cgi: printers.o ../Makedefs ../cups/$(LIBCUPS) $(LIBCUPSCGI) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ printers.o -L. -lcupscgi $(LIBS) + + +# +# testcgi +# + +testcgi: testcgi.o ../Makedefs libcupscgi.a ../cups/$(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o $@ testcgi.o libcupscgi.a \ + ../cups/$(LIBCUPSSTATIC) $(COMMONLIBS) $(SSLLIBS) $(DNSSDLIBS) \ + $(LIBZ) $(LIBGSSAPI) + echo Testing CGI API... + ./testcgi + + +# +# testhi +# + +testhi: testhi.o ../Makedefs libcupscgi.a ../cups/$(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o $@ testhi.o libcupscgi.a \ + ../cups/$(LIBCUPSSTATIC) $(COMMONLIBS) $(SSLLIBS) $(DNSSDLIBS) \ + $(LIBZ) $(LIBGSSAPI) + echo Testing help index API... + ./testhi + + +# +# testtemplate +# + +testtemplate: testtemplate.o ../Makedefs libcupscgi.a ../cups/$(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ testtemplate.o libcupscgi.a ../cups/$(LIBCUPSSTATIC) \ + $(COMMONLIBS) $(SSLLIBS) $(DNSSDLIBS) $(LIBZ) $(LIBGSSAPI) + + +# +# websearch +# + +websearch: websearch.o ../Makedefs libcupscgi.a ../cups/$(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o $@ websearch.o libcupscgi.a \ + ../cups/$(LIBCUPSSTATIC) $(COMMONLIBS) $(SSLLIBS) $(DNSSDLIBS) \ + $(LIBZ) $(LIBGSSAPI) + + +# +# Dependencies... +# + +include Dependencies + + +# +# End of "$Id$". +# diff --git a/cgi-bin/admin.c b/cgi-bin/admin.c new file mode 100644 index 0000000000..c2b2900f83 --- /dev/null +++ b/cgi-bin/admin.c @@ -0,0 +1,4200 @@ +/* + * "$Id$" + * + * Administration CGI for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Main entry for CGI. + * choose_device_cb() - Add a device to the device selection page. + * do_add_rss_subscription() - Add a RSS subscription. + * do_am_class() - Add or modify a class. + * do_am_printer() - Add or modify a printer. + * do_cancel_subscription() - Cancel a subscription. + * do_config_server() - Configure server settings. + * do_delete_class() - Delete a class. + * do_delete_printer() - Delete a printer. + * do_export() - Export printers to Samba. + * do_list_printers() - List available printers. + * do_menu() - Show the main menu. + * do_set_allowed_users() - Set the allowed/denied users for a queue. + * do_set_default() - Set the server default printer/class. + * do_set_options() - Configure the default options for a queue. + * do_set_sharing() - Set printer-is-shared value. + * get_option_value() - Return the value of an option. + * get_points() - Get a value in points. + */ + +/* + * Include necessary headers... + */ + +#include "cgi-private.h" +#include +#include +#include +#include +#include +#include +#include + + +/* + * Local globals... + */ + +static int current_device = 0; /* Current device shown */ + + +/* + * Local functions... + */ + +static void choose_device_cb(const char *device_class, + const char *device_id, const char *device_info, + const char *device_make_and_model, + const char *device_uri, + const char *device_location, + const char *title); +static void do_add_rss_subscription(http_t *http); +static void do_am_class(http_t *http, int modify); +static void do_am_printer(http_t *http, int modify); +static void do_cancel_subscription(http_t *http); +static void do_config_server(http_t *http); +static void do_delete_class(http_t *http); +static void do_delete_printer(http_t *http); +static void do_export(http_t *http); +static void do_list_printers(http_t *http); +static void do_menu(http_t *http); +static void do_set_allowed_users(http_t *http); +static void do_set_default(http_t *http); +static void do_set_options(http_t *http, int is_class); +static void do_set_sharing(http_t *http); +static char *get_option_value(ppd_file_t *ppd, const char *name, + char *buffer, size_t bufsize); +static double get_points(double number, const char *uval); + + +/* + * 'main()' - Main entry for CGI. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + http_t *http; /* Connection to the server */ + const char *op; /* Operation name */ + + + /* + * Connect to the HTTP server... + */ + + fputs("DEBUG: admin.cgi started...\n", stderr); + + http = httpConnectEncrypt(cupsServer(), ippPort(), cupsEncryption()); + + if (!http) + { + perror("ERROR: Unable to connect to cupsd"); + fprintf(stderr, "DEBUG: cupsServer()=\"%s\"\n", + cupsServer() ? cupsServer() : "(null)"); + fprintf(stderr, "DEBUG: ippPort()=%d\n", ippPort()); + fprintf(stderr, "DEBUG: cupsEncryption()=%d\n", cupsEncryption()); + exit(1); + } + + fprintf(stderr, "DEBUG: http=%p\n", http); + + /* + * Set the web interface section... + */ + + cgiSetVariable("SECTION", "admin"); + cgiSetVariable("REFRESH_PAGE", ""); + + /* + * See if we have form data... + */ + + if (!cgiInitialize() || !cgiGetVariable("OP")) + { + /* + * Nope, send the administration menu... + */ + + fputs("DEBUG: No form data, showing main menu...\n", stderr); + + do_menu(http); + } + else if ((op = cgiGetVariable("OP")) != NULL && cgiIsPOST()) + { + /* + * Do the operation... + */ + + fprintf(stderr, "DEBUG: op=\"%s\"...\n", op); + + if (!*op) + { + const char *printer = getenv("PRINTER_NAME"), + /* Printer or class name */ + *server_port = getenv("SERVER_PORT"); + /* Port number string */ + int port = atoi(server_port ? server_port : "0"); + /* Port number */ + char uri[1024]; /* URL */ + + if (printer) + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), + getenv("HTTPS") ? "https" : "http", NULL, + getenv("SERVER_NAME"), port, "/%s/%s", + cgiGetVariable("IS_CLASS") ? "classes" : "printers", + printer); + else + httpAssembleURI(HTTP_URI_CODING_ALL, uri, sizeof(uri), + getenv("HTTPS") ? "https" : "http", NULL, + getenv("SERVER_NAME"), port, "/admin"); + + printf("Location: %s\n\n", uri); + } + else if (!strcmp(op, "set-allowed-users")) + do_set_allowed_users(http); + else if (!strcmp(op, "set-as-default")) + do_set_default(http); + else if (!strcmp(op, "set-sharing")) + do_set_sharing(http); + else if (!strcmp(op, "find-new-printers") || + !strcmp(op, "list-available-printers")) + do_list_printers(http); + else if (!strcmp(op, "add-class")) + do_am_class(http, 0); + else if (!strcmp(op, "add-printer")) + do_am_printer(http, 0); + else if (!strcmp(op, "modify-class")) + do_am_class(http, 1); + else if (!strcmp(op, "modify-printer")) + do_am_printer(http, 1); + else if (!strcmp(op, "delete-class")) + do_delete_class(http); + else if (!strcmp(op, "delete-printer")) + do_delete_printer(http); + else if (!strcmp(op, "set-class-options")) + do_set_options(http, 1); + else if (!strcmp(op, "set-printer-options")) + do_set_options(http, 0); + else if (!strcmp(op, "config-server")) + do_config_server(http); + else if (!strcmp(op, "export-samba")) + do_export(http); + else if (!strcmp(op, "add-rss-subscription")) + do_add_rss_subscription(http); + else if (!strcmp(op, "cancel-subscription")) + do_cancel_subscription(http); + else + { + /* + * Bad operation code - display an error... + */ + + cgiStartHTML(cgiText(_("Administration"))); + cgiCopyTemplateLang("error-op.tmpl"); + cgiEndHTML(); + } + } + else if (op && !strcmp(op, "redirect")) + { + const char *url; /* Redirection URL... */ + char prefix[1024]; /* URL prefix */ + + + if (getenv("HTTPS")) + snprintf(prefix, sizeof(prefix), "https://%s:%s", + getenv("SERVER_NAME"), getenv("SERVER_PORT")); + else + snprintf(prefix, sizeof(prefix), "http://%s:%s", + getenv("SERVER_NAME"), getenv("SERVER_PORT")); + + fprintf(stderr, "DEBUG: redirecting with prefix %s!\n", prefix); + + if ((url = cgiGetVariable("URL")) != NULL) + { + char encoded[1024], /* Encoded URL string */ + *ptr; /* Pointer into encoded string */ + + + ptr = encoded; + if (*url != '/') + *ptr++ = '/'; + + for (; *url && ptr < (encoded + sizeof(encoded) - 4); url ++) + { + if (strchr("%@&+ <>#=", *url) || *url < ' ' || *url & 128) + { + /* + * Percent-encode this character; safe because we have at least 4 + * bytes left in the array... + */ + + sprintf(ptr, "%%%02X", *url & 255); + ptr += 3; + } + else + *ptr++ = *url; + } + + *ptr = '\0'; + + if (*url) + { + /* + * URL was too long, just redirect to the admin page... + */ + + printf("Location: %s/admin\n\n", prefix); + } + else + { + /* + * URL is OK, redirect there... + */ + + printf("Location: %s%s\n\n", prefix, encoded); + } + } + else + printf("Location: %s/admin\n\n", prefix); + } + else + { + /* + * Form data but no operation code - display an error... + */ + + cgiStartHTML(cgiText(_("Administration"))); + cgiCopyTemplateLang("error-op.tmpl"); + cgiEndHTML(); + } + + /* + * Close the HTTP server connection... + */ + + httpClose(http); + + /* + * Return with no errors... + */ + + return (0); +} + + +/* + * 'choose_device_cb()' - Add a device to the device selection page. + */ + +static void +choose_device_cb( + const char *device_class, /* I - Class */ + const char *device_id, /* I - 1284 device ID */ + const char *device_info, /* I - Description */ + const char *device_make_and_model, /* I - Make and model */ + const char *device_uri, /* I - Device URI */ + const char *device_location, /* I - Location */ + const char *title) /* I - Page title */ +{ + /* + * For modern browsers, start a multi-part page so we can show that something + * is happening. Non-modern browsers just get everything at the end... + */ + + if (current_device == 0 && cgiSupportsMultipart()) + { + cgiStartMultipart(); + cgiStartHTML(title); + cgiCopyTemplateLang("choose-device.tmpl"); + cgiEndHTML(); + fflush(stdout); + } + + + /* + * Add the device to the array... + */ + + cgiSetArray("device_class", current_device, device_class); + cgiSetArray("device_id", current_device, device_id); + cgiSetArray("device_info", current_device, device_info); + cgiSetArray("device_make_and_model", current_device, device_make_and_model); + cgiSetArray("device_uri", current_device, device_uri); + cgiSetArray("device_location", current_device, device_location); + + current_device ++; +} + + +/* + * 'do_add_rss_subscription()' - Add a RSS subscription. + */ + +static void +do_add_rss_subscription(http_t *http) /* I - HTTP connection */ +{ + ipp_t *request, /* IPP request data */ + *response; /* IPP response data */ + char rss_uri[1024]; /* RSS notify-recipient URI */ + int num_events; /* Number of events */ + const char *events[12], /* Subscribed events */ + *subscription_name, /* Subscription name */ + *printer_uri, /* Printer URI */ + *ptr, /* Pointer into name */ + *user; /* Username */ + int max_events; /* Maximum number of events */ + + + /* + * See if we have all of the required information... + */ + + subscription_name = cgiGetVariable("SUBSCRIPTION_NAME"); + printer_uri = cgiGetVariable("PRINTER_URI"); + num_events = 0; + + if (cgiGetVariable("EVENT_JOB_CREATED")) + events[num_events ++] = "job-created"; + if (cgiGetVariable("EVENT_JOB_COMPLETED")) + events[num_events ++] = "job-completed"; + if (cgiGetVariable("EVENT_JOB_STOPPED")) + events[num_events ++] = "job-stopped"; + if (cgiGetVariable("EVENT_JOB_CONFIG_CHANGED")) + events[num_events ++] = "job-config-changed"; + if (cgiGetVariable("EVENT_PRINTER_STOPPED")) + events[num_events ++] = "printer-stopped"; + if (cgiGetVariable("EVENT_PRINTER_ADDED")) + events[num_events ++] = "printer-added"; + if (cgiGetVariable("EVENT_PRINTER_MODIFIED")) + events[num_events ++] = "printer-modified"; + if (cgiGetVariable("EVENT_PRINTER_DELETED")) + events[num_events ++] = "printer-deleted"; + if (cgiGetVariable("EVENT_SERVER_STARTED")) + events[num_events ++] = "server-started"; + if (cgiGetVariable("EVENT_SERVER_STOPPED")) + events[num_events ++] = "server-stopped"; + if (cgiGetVariable("EVENT_SERVER_RESTARTED")) + events[num_events ++] = "server-restarted"; + if (cgiGetVariable("EVENT_SERVER_AUDIT")) + events[num_events ++] = "server-audit"; + + if ((ptr = cgiGetVariable("MAX_EVENTS")) != NULL) + max_events = atoi(ptr); + else + max_events = 0; + + if (!subscription_name || !printer_uri || !num_events || + max_events <= 0 || max_events > 9999) + { + /* + * Don't have everything we need, so get the available printers + * and classes and (re)show the add page... + */ + + if (cgiGetVariable("EVENT_JOB_CREATED")) + cgiSetVariable("EVENT_JOB_CREATED", "CHECKED"); + if (cgiGetVariable("EVENT_JOB_COMPLETED")) + cgiSetVariable("EVENT_JOB_COMPLETED", "CHECKED"); + if (cgiGetVariable("EVENT_JOB_STOPPED")) + cgiSetVariable("EVENT_JOB_STOPPED", "CHECKED"); + if (cgiGetVariable("EVENT_JOB_CONFIG_CHANGED")) + cgiSetVariable("EVENT_JOB_CONFIG_CHANGED", "CHECKED"); + if (cgiGetVariable("EVENT_PRINTER_STOPPED")) + cgiSetVariable("EVENT_PRINTER_STOPPED", "CHECKED"); + if (cgiGetVariable("EVENT_PRINTER_ADDED")) + cgiSetVariable("EVENT_PRINTER_ADDED", "CHECKED"); + if (cgiGetVariable("EVENT_PRINTER_MODIFIED")) + cgiSetVariable("EVENT_PRINTER_MODIFIED", "CHECKED"); + if (cgiGetVariable("EVENT_PRINTER_DELETED")) + cgiSetVariable("EVENT_PRINTER_DELETED", "CHECKED"); + if (cgiGetVariable("EVENT_SERVER_STARTED")) + cgiSetVariable("EVENT_SERVER_STARTED", "CHECKED"); + if (cgiGetVariable("EVENT_SERVER_STOPPED")) + cgiSetVariable("EVENT_SERVER_STOPPED", "CHECKED"); + if (cgiGetVariable("EVENT_SERVER_RESTARTED")) + cgiSetVariable("EVENT_SERVER_RESTARTED", "CHECKED"); + if (cgiGetVariable("EVENT_SERVER_AUDIT")) + cgiSetVariable("EVENT_SERVER_AUDIT", "CHECKED"); + + request = ippNewRequest(CUPS_GET_PRINTERS); + response = cupsDoRequest(http, request, "/"); + + cgiSetIPPVars(response, NULL, NULL, NULL, 0); + + ippDelete(response); + + cgiStartHTML(cgiText(_("Add RSS Subscription"))); + + cgiCopyTemplateLang("add-rss-subscription.tmpl"); + + cgiEndHTML(); + return; + } + + /* + * Make sure we have a username... + */ + + if ((user = getenv("REMOTE_USER")) == NULL) + { + puts("Status: 401\n"); + exit(0); + } + + /* + * Validate the subscription name... + */ + + for (ptr = subscription_name; *ptr; ptr ++) + if ((*ptr >= 0 && *ptr <= ' ') || *ptr == 127 || *ptr == '/' || + *ptr == '?' || *ptr == '#') + break; + + if (*ptr) + { + cgiSetVariable("ERROR", + cgiText(_("The subscription name may not " + "contain spaces, slashes (/), question marks (?), " + "or the pound sign (#)."))); + cgiStartHTML(_("Add RSS Subscription")); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + return; + } + + /* + * Add the subscription... + */ + + ptr = subscription_name + strlen(subscription_name) - 4; + if (ptr < subscription_name || strcmp(ptr, ".rss")) + httpAssembleURIf(HTTP_URI_CODING_ALL, rss_uri, sizeof(rss_uri), "rss", + NULL, NULL, 0, "/%s.rss?max_events=%d", subscription_name, + max_events); + else + httpAssembleURIf(HTTP_URI_CODING_ALL, rss_uri, sizeof(rss_uri), "rss", + NULL, NULL, 0, "/%s?max_events=%d", subscription_name, + max_events); + + request = ippNewRequest(IPP_CREATE_PRINTER_SUBSCRIPTION); + + if (!_cups_strcasecmp(printer_uri, "#ALL#")) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, "ipp://localhost/"); + else + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, printer_uri); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, user); + + ippAddString(request, IPP_TAG_SUBSCRIPTION, IPP_TAG_URI, + "notify-recipient-uri", NULL, rss_uri); + ippAddStrings(request, IPP_TAG_SUBSCRIPTION, IPP_TAG_KEYWORD, "notify-events", + num_events, NULL, events); + ippAddInteger(request, IPP_TAG_SUBSCRIPTION, IPP_TAG_INTEGER, + "notify-lease-duration", 0); + + ippDelete(cupsDoRequest(http, request, "/")); + + if (cupsLastError() == IPP_NOT_AUTHORIZED) + { + puts("Status: 401\n"); + exit(0); + } + else if (cupsLastError() > IPP_OK_CONFLICT) + { + cgiStartHTML(_("Add RSS Subscription")); + cgiShowIPPError(_("Unable to add RSS subscription:")); + } + else + { + /* + * Redirect successful updates back to the admin page... + */ + + cgiSetVariable("refresh_page", "5;URL=/admin"); + cgiStartHTML(_("Add RSS Subscription")); + cgiCopyTemplateLang("subscription-added.tmpl"); + } + + cgiEndHTML(); +} + + +/* + * 'do_am_class()' - Add or modify a class. + */ + +static void +do_am_class(http_t *http, /* I - HTTP connection */ + int modify) /* I - Modify the printer? */ +{ + int i, j; /* Looping vars */ + int element; /* Element number */ + int num_printers; /* Number of printers */ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + ipp_attribute_t *attr; /* member-uris attribute */ + char uri[HTTP_MAX_URI]; /* Device or printer URI */ + const char *name, /* Pointer to class name */ + *op, /* Operation name */ + *ptr; /* Pointer to CGI variable */ + const char *title; /* Title of page */ + static const char * const pattrs[] = /* Requested printer attributes */ + { + "member-names", + "printer-info", + "printer-location" + }; + + + title = cgiText(modify ? _("Modify Class") : _("Add Class")); + op = cgiGetVariable("OP"); + name = cgiGetVariable("PRINTER_NAME"); + + if (cgiGetVariable("PRINTER_LOCATION") == NULL) + { + /* + * Build a CUPS_GET_PRINTERS request, which requires the + * following attributes: + * + * attributes-charset + * attributes-natural-language + */ + + request = ippNewRequest(CUPS_GET_PRINTERS); + + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_ENUM, "printer-type", + CUPS_PRINTER_LOCAL); + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_ENUM, "printer-type-mask", + CUPS_PRINTER_CLASS | CUPS_PRINTER_REMOTE); + + /* + * Do the request and get back a response... + */ + + cgiClearVariables(); + if (op) + cgiSetVariable("OP", op); + if (name) + cgiSetVariable("PRINTER_NAME", name); + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + /* + * Create MEMBER_URIS and MEMBER_NAMES arrays... + */ + + for (element = 0, attr = response->attrs; + attr != NULL; + attr = attr->next) + if (attr->name && !strcmp(attr->name, "printer-uri-supported")) + { + if ((ptr = strrchr(attr->values[0].string.text, '/')) != NULL && + (!name || _cups_strcasecmp(name, ptr + 1))) + { + /* + * Don't show the current class... + */ + + cgiSetArray("MEMBER_URIS", element, attr->values[0].string.text); + element ++; + } + } + + for (element = 0, attr = response->attrs; + attr != NULL; + attr = attr->next) + if (attr->name && !strcmp(attr->name, "printer-name")) + { + if (!name || _cups_strcasecmp(name, attr->values[0].string.text)) + { + /* + * Don't show the current class... + */ + + cgiSetArray("MEMBER_NAMES", element, attr->values[0].string.text); + element ++; + } + } + + num_printers = cgiGetSize("MEMBER_URIS"); + + ippDelete(response); + } + else + num_printers = 0; + + if (modify) + { + /* + * Build an IPP_GET_PRINTER_ATTRIBUTES request, which requires the + * following attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + */ + + request = ippNewRequest(IPP_GET_PRINTER_ATTRIBUTES); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/classes/%s", name); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", + (int)(sizeof(pattrs) / sizeof(pattrs[0])), + NULL, pattrs); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + if ((attr = ippFindAttribute(response, "member-names", + IPP_TAG_NAME)) != NULL) + { + /* + * Mark any current members in the class... + */ + + for (j = 0; j < num_printers; j ++) + cgiSetArray("MEMBER_SELECTED", j, ""); + + for (i = 0; i < attr->num_values; i ++) + { + for (j = 0; j < num_printers; j ++) + { + if (!_cups_strcasecmp(attr->values[i].string.text, + cgiGetArray("MEMBER_NAMES", j))) + { + cgiSetArray("MEMBER_SELECTED", j, "SELECTED"); + break; + } + } + } + } + + if ((attr = ippFindAttribute(response, "printer-info", + IPP_TAG_TEXT)) != NULL) + cgiSetVariable("PRINTER_INFO", attr->values[0].string.text); + + if ((attr = ippFindAttribute(response, "printer-location", + IPP_TAG_TEXT)) != NULL) + cgiSetVariable("PRINTER_LOCATION", attr->values[0].string.text); + + ippDelete(response); + } + + /* + * Update the location and description of an existing printer... + */ + + cgiStartHTML(title); + cgiCopyTemplateLang("modify-class.tmpl"); + } + else + { + /* + * Get the name, location, and description for a new printer... + */ + + cgiStartHTML(title); + cgiCopyTemplateLang("add-class.tmpl"); + } + + cgiEndHTML(); + + return; + } + + if (!name) + { + cgiStartHTML(title); + cgiSetVariable("ERROR", cgiText(_("Missing form variable"))); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + return; + } + + for (ptr = name; *ptr; ptr ++) + if ((*ptr >= 0 && *ptr <= ' ') || *ptr == 127 || *ptr == '/' || *ptr == '#') + break; + + if (*ptr || ptr == name || strlen(name) > 127) + { + cgiSetVariable("ERROR", + cgiText(_("The class name may only contain up to " + "127 printable characters and may not " + "contain spaces, slashes (/), or the " + "pound sign (#)."))); + cgiStartHTML(title); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + return; + } + + /* + * Build a CUPS_ADD_CLASS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * printer-location + * printer-info + * printer-is-accepting-jobs + * printer-state + * member-uris + */ + + request = ippNewRequest(CUPS_ADD_CLASS); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/classes/%s", name); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + ippAddString(request, IPP_TAG_PRINTER, IPP_TAG_TEXT, "printer-location", + NULL, cgiGetVariable("PRINTER_LOCATION")); + + ippAddString(request, IPP_TAG_PRINTER, IPP_TAG_TEXT, "printer-info", + NULL, cgiGetVariable("PRINTER_INFO")); + + ippAddBoolean(request, IPP_TAG_PRINTER, "printer-is-accepting-jobs", 1); + + ippAddInteger(request, IPP_TAG_PRINTER, IPP_TAG_ENUM, "printer-state", + IPP_PRINTER_IDLE); + + if ((num_printers = cgiGetSize("MEMBER_URIS")) > 0) + { + attr = ippAddStrings(request, IPP_TAG_PRINTER, IPP_TAG_URI, "member-uris", + num_printers, NULL, NULL); + for (i = 0; i < num_printers; i ++) + attr->values[i].string.text = _cupsStrAlloc(cgiGetArray("MEMBER_URIS", i)); + } + + /* + * Do the request and get back a response... + */ + + ippDelete(cupsDoRequest(http, request, "/admin/")); + + if (cupsLastError() == IPP_NOT_AUTHORIZED) + { + puts("Status: 401\n"); + exit(0); + } + else if (cupsLastError() > IPP_OK_CONFLICT) + { + cgiStartHTML(title); + cgiShowIPPError(modify ? _("Unable to modify class:") : + _("Unable to add class:")); + } + else + { + /* + * Redirect successful updates back to the class page... + */ + + char refresh[1024]; /* Refresh URL */ + + cgiFormEncode(uri, name, sizeof(uri)); + snprintf(refresh, sizeof(refresh), "5;URL=/admin/?OP=redirect&URL=/classes/%s", + uri); + cgiSetVariable("refresh_page", refresh); + + cgiStartHTML(title); + + if (modify) + cgiCopyTemplateLang("class-modified.tmpl"); + else + cgiCopyTemplateLang("class-added.tmpl"); + } + + cgiEndHTML(); +} + + +/* + * 'do_am_printer()' - Add or modify a printer. + */ + +static void +do_am_printer(http_t *http, /* I - HTTP connection */ + int modify) /* I - Modify the printer? */ +{ + int i; /* Looping var */ + ipp_attribute_t *attr; /* Current attribute */ + ipp_t *request, /* IPP request */ + *response, /* IPP response */ + *oldinfo; /* Old printer information */ + const cgi_file_t *file; /* Uploaded file, if any */ + const char *var; /* CGI variable */ + char uri[HTTP_MAX_URI], /* Device or printer URI */ + *uriptr; /* Pointer into URI */ + int maxrate; /* Maximum baud rate */ + char baudrate[255]; /* Baud rate string */ + const char *name, /* Pointer to class name */ + *ptr; /* Pointer to CGI variable */ + const char *title; /* Title of page */ + static int baudrates[] = /* Baud rates */ + { + 1200, + 2400, + 4800, + 9600, + 19200, + 38400, + 57600, + 115200, + 230400, + 460800 + }; + + + ptr = cgiGetVariable("DEVICE_URI"); + fprintf(stderr, "DEBUG: do_am_printer: DEVICE_URI=\"%s\"\n", + ptr ? ptr : "(null)"); + + title = cgiText(modify ? _("Modify Printer") : _("Add Printer")); + + if (modify) + { + /* + * Build an IPP_GET_PRINTER_ATTRIBUTES request, which requires the + * following attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + */ + + request = ippNewRequest(IPP_GET_PRINTER_ATTRIBUTES); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", + cgiGetVariable("PRINTER_NAME")); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + /* + * Do the request and get back a response... + */ + + oldinfo = cupsDoRequest(http, request, "/"); + } + else + oldinfo = NULL; + + file = cgiGetFile(); + + if (file) + { + fprintf(stderr, "DEBUG: file->tempfile=%s\n", file->tempfile); + fprintf(stderr, "DEBUG: file->name=%s\n", file->name); + fprintf(stderr, "DEBUG: file->filename=%s\n", file->filename); + fprintf(stderr, "DEBUG: file->mimetype=%s\n", file->mimetype); + } + + if ((name = cgiGetVariable("PRINTER_NAME")) != NULL) + { + for (ptr = name; *ptr; ptr ++) + if ((*ptr >= 0 && *ptr <= ' ') || *ptr == 127 || *ptr == '/' || *ptr == '#') + break; + + if (*ptr || ptr == name || strlen(name) > 127) + { + cgiSetVariable("ERROR", + cgiText(_("The printer name may only contain up to " + "127 printable characters and may not " + "contain spaces, slashes (/), or the " + "pound sign (#)."))); + cgiStartHTML(title); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + return; + } + } + + if ((var = cgiGetVariable("DEVICE_URI")) != NULL) + { + if ((uriptr = strrchr(var, '|')) != NULL) + { + /* + * Extract make and make/model from device URI string... + */ + + char make[1024], /* Make string */ + *makeptr; /* Pointer into make */ + + + *uriptr++ = '\0'; + + strlcpy(make, uriptr, sizeof(make)); + + if ((makeptr = strchr(make, ' ')) != NULL) + *makeptr = '\0'; + else if ((makeptr = strchr(make, '-')) != NULL) + *makeptr = '\0'; + else if (!_cups_strncasecmp(make, "laserjet", 8) || + !_cups_strncasecmp(make, "deskjet", 7) || + !_cups_strncasecmp(make, "designjet", 9)) + strcpy(make, "HP"); + else if (!_cups_strncasecmp(make, "phaser", 6)) + strcpy(make, "Xerox"); + else if (!_cups_strncasecmp(make, "stylus", 6)) + strcpy(make, "Epson"); + else + strcpy(make, "Generic"); + + if (!cgiGetVariable("CURRENT_MAKE")) + cgiSetVariable("CURRENT_MAKE", make); + + if (!cgiGetVariable("CURRENT_MAKE_AND_MODEL")) + cgiSetVariable("CURRENT_MAKE_AND_MODEL", uriptr); + + if (!modify) + { + char template[128], /* Template name */ + *tptr; /* Pointer into template name */ + + cgiSetVariable("PRINTER_INFO", uriptr); + + for (tptr = template; + tptr < (template + sizeof(template) - 1) && *uriptr; + uriptr ++) + if (isalnum(*uriptr & 255) || *uriptr == '_' || *uriptr == '-' || + *uriptr == '.') + *tptr++ = *uriptr; + else if ((*uriptr == ' ' || *uriptr == '/') && tptr > template && + tptr[-1] != '_') + *tptr++ = '_'; + else if (*uriptr == '?' || *uriptr == '(') + break; + + *tptr = '\0'; + + cgiSetVariable("TEMPLATE_NAME", template); + } + } + } + + if (!var) + { + /* + * Look for devices so the user can pick something... + */ + + if ((attr = ippFindAttribute(oldinfo, "device-uri", IPP_TAG_URI)) != NULL) + { + strlcpy(uri, attr->values[0].string.text, sizeof(uri)); + if ((uriptr = strchr(uri, ':')) != NULL && strncmp(uriptr, "://", 3) == 0) + *uriptr = '\0'; + + cgiSetVariable("CURRENT_DEVICE_URI", attr->values[0].string.text); + cgiSetVariable("CURRENT_DEVICE_SCHEME", uri); + } + + /* + * Scan for devices for up to 30 seconds... + */ + + fputs("DEBUG: Getting list of devices...\n", stderr); + + current_device = 0; + if (cupsGetDevices(http, 5, CUPS_INCLUDE_ALL, CUPS_EXCLUDE_NONE, + (cups_device_cb_t)choose_device_cb, + (void *)title) == IPP_OK) + { + fputs("DEBUG: Got device list!\n", stderr); + + if (cgiSupportsMultipart()) + cgiStartMultipart(); + + cgiSetVariable("CUPS_GET_DEVICES_DONE", "1"); + cgiStartHTML(title); + cgiCopyTemplateLang("choose-device.tmpl"); + cgiEndHTML(); + + if (cgiSupportsMultipart()) + cgiEndMultipart(); + } + else + { + fprintf(stderr, + "ERROR: CUPS-Get-Devices request failed with status %x: %s\n", + cupsLastError(), cupsLastErrorString()); + if (cupsLastError() == IPP_NOT_AUTHORIZED) + { + puts("Status: 401\n"); + exit(0); + } + else + { + cgiStartHTML(title); + cgiShowIPPError(modify ? _("Unable to modify printer:") : + _("Unable to add printer:")); + cgiEndHTML(); + return; + } + } + } + else if (!strchr(var, '/') || + (!strncmp(var, "lpd://", 6) && !strchr(var + 6, '/'))) + { + if ((attr = ippFindAttribute(oldinfo, "device-uri", IPP_TAG_URI)) != NULL) + { + /* + * Set the current device URI for the form to the old one... + */ + + if (strncmp(attr->values[0].string.text, var, strlen(var)) == 0) + cgiSetVariable("CURRENT_DEVICE_URI", attr->values[0].string.text); + } + + /* + * User needs to set the full URI... + */ + + cgiStartHTML(title); + cgiCopyTemplateLang("choose-uri.tmpl"); + cgiEndHTML(); + } + else if (!strncmp(var, "serial:", 7) && !cgiGetVariable("BAUDRATE")) + { + /* + * Need baud rate, parity, etc. + */ + + if ((var = strchr(var, '?')) != NULL && + strncmp(var, "?baud=", 6) == 0) + maxrate = atoi(var + 6); + else + maxrate = 19200; + + for (i = 0; i < 10; i ++) + if (baudrates[i] > maxrate) + break; + else + { + sprintf(baudrate, "%d", baudrates[i]); + cgiSetArray("BAUDRATES", i, baudrate); + } + + cgiStartHTML(title); + cgiCopyTemplateLang("choose-serial.tmpl"); + cgiEndHTML(); + } + else if (!name || !cgiGetVariable("PRINTER_LOCATION")) + { + cgiStartHTML(title); + + if (modify) + { + /* + * Update the location and description of an existing printer... + */ + + if (oldinfo) + { + if ((attr = ippFindAttribute(oldinfo, "printer-info", + IPP_TAG_TEXT)) != NULL) + cgiSetVariable("PRINTER_INFO", attr->values[0].string.text); + + if ((attr = ippFindAttribute(oldinfo, "printer-location", + IPP_TAG_TEXT)) != NULL) + cgiSetVariable("PRINTER_LOCATION", attr->values[0].string.text); + + if ((attr = ippFindAttribute(oldinfo, "printer-is-shared", + IPP_TAG_BOOLEAN)) != NULL) + cgiSetVariable("PRINTER_IS_SHARED", + attr->values[0].boolean ? "1" : "0"); + } + + cgiCopyTemplateLang("modify-printer.tmpl"); + } + else + { + /* + * Get the name, location, and description for a new printer... + */ + +#ifdef __APPLE__ + if (!strncmp(var, "usb:", 4)) + cgiSetVariable("printer_is_shared", "1"); + else +#endif /* __APPLE__ */ + cgiSetVariable("printer_is_shared", "0"); + + cgiCopyTemplateLang("add-printer.tmpl"); + } + + cgiEndHTML(); + + if (oldinfo) + ippDelete(oldinfo); + + return; + } + else if (!file && + (!cgiGetVariable("PPD_NAME") || cgiGetVariable("SELECT_MAKE"))) + { + if (modify && !cgiGetVariable("SELECT_MAKE")) + { + /* + * Get the PPD file... + */ + + int fd; /* PPD file */ + char filename[1024]; /* PPD filename */ + ppd_file_t *ppd; /* PPD information */ + char buffer[1024]; /* Buffer */ + int bytes; /* Number of bytes */ + http_status_t get_status; /* Status of GET */ + + + /* TODO: Use cupsGetFile() API... */ + snprintf(uri, sizeof(uri), "/printers/%s.ppd", name); + + if (httpGet(http, uri)) + httpGet(http, uri); + + while ((get_status = httpUpdate(http)) == HTTP_CONTINUE); + + if (get_status != HTTP_OK) + { + httpFlush(http); + + fprintf(stderr, "ERROR: Unable to get PPD file %s: %d - %s\n", + uri, get_status, httpStatus(get_status)); + } + else if ((fd = cupsTempFd(filename, sizeof(filename))) >= 0) + { + while ((bytes = httpRead2(http, buffer, sizeof(buffer))) > 0) + write(fd, buffer, bytes); + + close(fd); + + if ((ppd = ppdOpenFile(filename)) != NULL) + { + if (ppd->manufacturer) + cgiSetVariable("CURRENT_MAKE", ppd->manufacturer); + + if (ppd->nickname) + cgiSetVariable("CURRENT_MAKE_AND_MODEL", ppd->nickname); + + ppdClose(ppd); + unlink(filename); + } + else + { + fprintf(stderr, "ERROR: Unable to open PPD file %s: %s\n", + filename, ppdErrorString(ppdLastError(&bytes))); + } + } + else + { + httpFlush(http); + + fprintf(stderr, + "ERROR: Unable to create temporary file for PPD file: %s\n", + strerror(errno)); + } + } + + /* + * Build a CUPS_GET_PPDS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + */ + + request = ippNewRequest(CUPS_GET_PPDS); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, "ipp://localhost/printers/"); + + if ((var = cgiGetVariable("PPD_MAKE")) == NULL) + var = cgiGetVariable("CURRENT_MAKE"); + if (var && !cgiGetVariable("SELECT_MAKE")) + { + const char *make_model; /* Make and model */ + + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_TEXT, + "ppd-make", NULL, var); + + if ((make_model = cgiGetVariable("CURRENT_MAKE_AND_MODEL")) != NULL) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_TEXT, + "ppd-make-and-model", NULL, make_model); + } + else + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", NULL, "ppd-make"); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + /* + * Got the list of PPDs, see if the user has selected a make... + */ + + if (cgiSetIPPVars(response, NULL, NULL, NULL, 0) == 0 && !modify) + { + /* + * No PPD files with this make, try again with all makes... + */ + + ippDelete(response); + + request = ippNewRequest(CUPS_GET_PPDS); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, "ipp://localhost/printers/"); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", NULL, "ppd-make"); + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + cgiSetIPPVars(response, NULL, NULL, NULL, 0); + + cgiStartHTML(title); + cgiCopyTemplateLang("choose-make.tmpl"); + cgiEndHTML(); + } + else if (!var || cgiGetVariable("SELECT_MAKE")) + { + cgiStartHTML(title); + cgiCopyTemplateLang("choose-make.tmpl"); + cgiEndHTML(); + } + else + { + /* + * Let the user choose a model... + */ + + cgiStartHTML(title); + if (!cgiGetVariable("PPD_MAKE")) + cgiSetVariable("PPD_MAKE", cgiGetVariable("CURRENT_MAKE")); + if (!modify) + cgiSetVariable("CURRENT_MAKE_AND_MODEL", + cgiGetArray("PPD_MAKE_AND_MODEL", 0)); + cgiCopyTemplateLang("choose-model.tmpl"); + cgiEndHTML(); + } + + ippDelete(response); + } + else + { + cgiStartHTML(title); + cgiShowIPPError(_("Unable to get list of printer drivers:")); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + } + } + else + { + /* + * Build a CUPS_ADD_PRINTER request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * printer-location + * printer-info + * ppd-name + * device-uri + * printer-is-accepting-jobs + * printer-is-shared + * printer-state + */ + + request = ippNewRequest(CUPS_ADD_PRINTER); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", + cgiGetVariable("PRINTER_NAME")); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + ippAddString(request, IPP_TAG_PRINTER, IPP_TAG_TEXT, "printer-location", + NULL, cgiGetVariable("PRINTER_LOCATION")); + + ippAddString(request, IPP_TAG_PRINTER, IPP_TAG_TEXT, "printer-info", + NULL, cgiGetVariable("PRINTER_INFO")); + + if (!file) + { + var = cgiGetVariable("PPD_NAME"); + if (strcmp(var, "__no_change__")) + ippAddString(request, IPP_TAG_PRINTER, IPP_TAG_NAME, "ppd-name", + NULL, var); + } + + strlcpy(uri, cgiGetVariable("DEVICE_URI"), sizeof(uri)); + + /* + * Strip make and model from URI... + */ + + if ((uriptr = strrchr(uri, '|')) != NULL) + *uriptr = '\0'; + + if (!strncmp(uri, "serial:", 7)) + { + /* + * Update serial port URI to include baud rate, etc. + */ + + if ((uriptr = strchr(uri, '?')) == NULL) + uriptr = uri + strlen(uri); + + snprintf(uriptr, sizeof(uri) - (uriptr - uri), + "?baud=%s+bits=%s+parity=%s+flow=%s", + cgiGetVariable("BAUDRATE"), cgiGetVariable("BITS"), + cgiGetVariable("PARITY"), cgiGetVariable("FLOW")); + } + + ippAddString(request, IPP_TAG_PRINTER, IPP_TAG_URI, "device-uri", + NULL, uri); + + ippAddBoolean(request, IPP_TAG_PRINTER, "printer-is-accepting-jobs", 1); + + var = cgiGetVariable("printer_is_shared"); + ippAddBoolean(request, IPP_TAG_PRINTER, "printer-is-shared", + var && (!strcmp(var, "1") || !strcmp(var, "on"))); + + ippAddInteger(request, IPP_TAG_PRINTER, IPP_TAG_ENUM, "printer-state", + IPP_PRINTER_IDLE); + + /* + * Do the request and get back a response... + */ + + if (file) + ippDelete(cupsDoFileRequest(http, request, "/admin/", file->tempfile)); + else + ippDelete(cupsDoRequest(http, request, "/admin/")); + + if (cupsLastError() == IPP_NOT_AUTHORIZED) + { + puts("Status: 401\n"); + exit(0); + } + else if (cupsLastError() > IPP_OK_CONFLICT) + { + cgiStartHTML(title); + cgiShowIPPError(modify ? _("Unable to modify printer:") : + _("Unable to add printer:")); + } + else if (modify) + { + /* + * Redirect successful updates back to the printer page... + */ + + char refresh[1024]; /* Refresh URL */ + + + cgiFormEncode(uri, name, sizeof(uri)); + + snprintf(refresh, sizeof(refresh), + "5;/admin/?OP=redirect&URL=/printers/%s", uri); + + cgiSetVariable("refresh_page", refresh); + + cgiStartHTML(title); + + cgiCopyTemplateLang("printer-modified.tmpl"); + } + else + { + /* + * Set the printer options... + */ + + cgiSetVariable("OP", "set-printer-options"); + do_set_options(http, 0); + return; + } + + cgiEndHTML(); + } + + if (oldinfo) + ippDelete(oldinfo); +} + + +/* + * 'do_cancel_subscription()' - Cancel a subscription. + */ + +static void +do_cancel_subscription(http_t *http)/* I - HTTP connection */ +{ + ipp_t *request; /* IPP request data */ + const char *var, /* Form variable */ + *user; /* Username */ + int id; /* Subscription ID */ + + + /* + * See if we have all of the required information... + */ + + if ((var = cgiGetVariable("NOTIFY_SUBSCRIPTION_ID")) != NULL) + id = atoi(var); + else + id = 0; + + if (id <= 0) + { + cgiSetVariable("ERROR", cgiText(_("Bad subscription ID"))); + cgiStartHTML(_("Cancel RSS Subscription")); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + return; + } + + /* + * Require a username... + */ + + if ((user = getenv("REMOTE_USER")) == NULL) + { + puts("Status: 401\n"); + exit(0); + } + + /* + * Cancel the subscription... + */ + + request = ippNewRequest(IPP_CANCEL_SUBSCRIPTION); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, "ipp://localhost/"); + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_INTEGER, + "notify-subscription-id", id); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, user); + + ippDelete(cupsDoRequest(http, request, "/")); + + if (cupsLastError() == IPP_NOT_AUTHORIZED) + { + puts("Status: 401\n"); + exit(0); + } + else if (cupsLastError() > IPP_OK_CONFLICT) + { + cgiStartHTML(_("Cancel RSS Subscription")); + cgiShowIPPError(_("Unable to cancel RSS subscription:")); + } + else + { + /* + * Redirect successful updates back to the admin page... + */ + + cgiSetVariable("refresh_page", "5;URL=/admin"); + cgiStartHTML(_("Cancel RSS Subscription")); + cgiCopyTemplateLang("subscription-canceled.tmpl"); + } + + cgiEndHTML(); +} + + +/* + * 'do_config_server()' - Configure server settings. + */ + +static void +do_config_server(http_t *http) /* I - HTTP connection */ +{ + if (cgiGetVariable("CHANGESETTINGS")) + { + /* + * Save basic setting changes... + */ + + int num_settings; /* Number of server settings */ + cups_option_t *settings; /* Server settings */ + int advanced, /* Advanced settings shown? */ + changed; /* Have settings changed? */ + const char *debug_logging, /* DEBUG_LOGGING value */ + *remote_admin, /* REMOTE_ADMIN value */ + *remote_any, /* REMOTE_ANY value */ + *share_printers,/* SHARE_PRINTERS value */ + *user_cancel_any, + /* USER_CANCEL_ANY value */ + *browse_web_if = NULL, + /* BrowseWebIF value */ + *preserve_job_history = NULL, + /* PreserveJobHistory value */ + *preserve_job_files = NULL, + /* PreserveJobFiles value */ + *max_clients = NULL, + /* MaxClients value */ + *max_jobs = NULL, + /* MaxJobs value */ + *max_log_size = NULL; + /* MaxLogSize value */ + const char *current_browse_web_if, + /* BrowseWebIF value */ + *current_preserve_job_history, + /* PreserveJobHistory value */ + *current_preserve_job_files, + /* PreserveJobFiles value */ + *current_max_clients, + /* MaxClients value */ + *current_max_jobs, + /* MaxJobs value */ + *current_max_log_size; + /* MaxLogSize value */ +#ifdef HAVE_GSSAPI + char default_auth_type[255]; + /* DefaultAuthType value */ + const char *val; /* Setting value */ +#endif /* HAVE_GSSAPI */ + + + /* + * Get the checkbox values from the form... + */ + + debug_logging = cgiGetVariable("DEBUG_LOGGING") ? "1" : "0"; + remote_admin = cgiGetVariable("REMOTE_ADMIN") ? "1" : "0"; + remote_any = cgiGetVariable("REMOTE_ANY") ? "1" : "0"; + share_printers = cgiGetVariable("SHARE_PRINTERS") ? "1" : "0"; + user_cancel_any = cgiGetVariable("USER_CANCEL_ANY") ? "1" : "0"; + + advanced = cgiGetVariable("ADVANCEDSETTINGS") != NULL; + if (advanced) + { + /* + * Get advanced settings... + */ + + browse_web_if = cgiGetVariable("BROWSE_WEB_IF") ? "Yes" : "No"; + preserve_job_history = cgiGetVariable("PRESERVE_JOB_HISTORY") ? "Yes" : "No"; + preserve_job_files = cgiGetVariable("PRESERVE_JOB_FILES") ? "Yes" : "No"; + max_clients = cgiGetVariable("MAX_CLIENTS"); + max_jobs = cgiGetVariable("MAX_JOBS"); + max_log_size = cgiGetVariable("MAX_LOG_SIZE"); + + if (!max_clients || atoi(max_clients) <= 0) + max_clients = "100"; + + if (!max_jobs || atoi(max_jobs) <= 0) + max_jobs = "500"; + + if (!max_log_size || atof(max_log_size) <= 0.0) + max_log_size = "1m"; + } + + /* + * Get the current server settings... + */ + + if (!cupsAdminGetServerSettings(http, &num_settings, &settings)) + { + cgiStartHTML(cgiText(_("Change Settings"))); + cgiSetVariable("MESSAGE", + cgiText(_("Unable to change server settings:"))); + cgiSetVariable("ERROR", cupsLastErrorString()); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + return; + } + +#ifdef HAVE_GSSAPI + /* + * Get authentication settings... + */ + + if (cgiGetVariable("KERBEROS")) + strlcpy(default_auth_type, "Negotiate", sizeof(default_auth_type)); + else + { + val = cupsGetOption("DefaultAuthType", num_settings, settings); + + if (!val || !_cups_strcasecmp(val, "Negotiate")) + strlcpy(default_auth_type, "Basic", sizeof(default_auth_type)); + else + strlcpy(default_auth_type, val, sizeof(default_auth_type)); + } + + fprintf(stderr, "DEBUG: DefaultAuthType %s\n", default_auth_type); +#endif /* HAVE_GSSAPI */ + + if ((current_browse_web_if = cupsGetOption("BrowseWebIF", num_settings, + settings)) == NULL) + current_browse_web_if = "No"; + + if ((current_preserve_job_history = cupsGetOption("PreserveJobHistory", + num_settings, + settings)) == NULL) + current_preserve_job_history = "Yes"; + + if ((current_preserve_job_files = cupsGetOption("PreserveJobFiles", + num_settings, + settings)) == NULL) + current_preserve_job_files = "No"; + + if ((current_max_clients = cupsGetOption("MaxClients", num_settings, + settings)) == NULL) + current_max_clients = "100"; + + if ((current_max_jobs = cupsGetOption("MaxJobs", num_settings, + settings)) == NULL) + current_max_jobs = "500"; + + if ((current_max_log_size = cupsGetOption("MaxLogSize", num_settings, + settings)) == NULL) + current_max_log_size = "1m"; + + /* + * See if the settings have changed... + */ + + changed = strcmp(debug_logging, cupsGetOption(CUPS_SERVER_DEBUG_LOGGING, + num_settings, settings)) || + strcmp(remote_admin, cupsGetOption(CUPS_SERVER_REMOTE_ADMIN, + num_settings, settings)) || + strcmp(remote_any, cupsGetOption(CUPS_SERVER_REMOTE_ANY, + num_settings, settings)) || + strcmp(share_printers, cupsGetOption(CUPS_SERVER_SHARE_PRINTERS, + num_settings, settings)) || +#ifdef HAVE_GSSAPI + !cupsGetOption("DefaultAuthType", num_settings, settings) || + strcmp(default_auth_type, cupsGetOption("DefaultAuthType", + num_settings, settings)) || +#endif /* HAVE_GSSAPI */ + strcmp(user_cancel_any, cupsGetOption(CUPS_SERVER_USER_CANCEL_ANY, + num_settings, settings)); + + if (advanced && !changed) + changed = _cups_strcasecmp(browse_web_if, current_browse_web_if) || + _cups_strcasecmp(preserve_job_history, current_preserve_job_history) || + _cups_strcasecmp(preserve_job_files, current_preserve_job_files) || + _cups_strcasecmp(max_clients, current_max_clients) || + _cups_strcasecmp(max_jobs, current_max_jobs) || + _cups_strcasecmp(max_log_size, current_max_log_size); + + if (changed) + { + /* + * Settings *have* changed, so save the changes... + */ + + cupsFreeOptions(num_settings, settings); + + num_settings = 0; + num_settings = cupsAddOption(CUPS_SERVER_DEBUG_LOGGING, + debug_logging, num_settings, &settings); + num_settings = cupsAddOption(CUPS_SERVER_REMOTE_ADMIN, + remote_admin, num_settings, &settings); + num_settings = cupsAddOption(CUPS_SERVER_REMOTE_ANY, + remote_any, num_settings, &settings); + num_settings = cupsAddOption(CUPS_SERVER_SHARE_PRINTERS, + share_printers, num_settings, &settings); + num_settings = cupsAddOption(CUPS_SERVER_USER_CANCEL_ANY, + user_cancel_any, num_settings, &settings); +#ifdef HAVE_GSSAPI + num_settings = cupsAddOption("DefaultAuthType", default_auth_type, + num_settings, &settings); +#endif /* HAVE_GSSAPI */ + + if (advanced) + { + /* + * Add advanced settings... + */ + + if (_cups_strcasecmp(browse_web_if, current_browse_web_if)) + num_settings = cupsAddOption("BrowseWebIF", browse_web_if, + num_settings, &settings); + if (_cups_strcasecmp(preserve_job_history, current_preserve_job_history)) + num_settings = cupsAddOption("PreserveJobHistory", + preserve_job_history, num_settings, + &settings); + if (_cups_strcasecmp(preserve_job_files, current_preserve_job_files)) + num_settings = cupsAddOption("PreserveJobFiles", preserve_job_files, + num_settings, &settings); + if (_cups_strcasecmp(max_clients, current_max_clients)) + num_settings = cupsAddOption("MaxClients", max_clients, num_settings, + &settings); + if (_cups_strcasecmp(max_jobs, current_max_jobs)) + num_settings = cupsAddOption("MaxJobs", max_jobs, num_settings, + &settings); + if (_cups_strcasecmp(max_log_size, current_max_log_size)) + num_settings = cupsAddOption("MaxLogSize", max_log_size, num_settings, + &settings); + } + + if (!cupsAdminSetServerSettings(http, num_settings, settings)) + { + if (cupsLastError() == IPP_NOT_AUTHORIZED) + { + puts("Status: 401\n"); + exit(0); + } + + cgiStartHTML(cgiText(_("Change Settings"))); + cgiSetVariable("MESSAGE", + cgiText(_("Unable to change server settings:"))); + cgiSetVariable("ERROR", cupsLastErrorString()); + cgiCopyTemplateLang("error.tmpl"); + } + else + { + if (advanced) + cgiSetVariable("refresh_page", "5;URL=/admin/?OP=redirect&" + "URL=/admin/?ADVANCEDSETTINGS=YES"); + else + cgiSetVariable("refresh_page", "5;URL=/admin/?OP=redirect"); + cgiStartHTML(cgiText(_("Change Settings"))); + cgiCopyTemplateLang("restart.tmpl"); + } + } + else + { + /* + * No changes... + */ + + cgiSetVariable("refresh_page", "5;URL=/admin/?OP=redirect"); + cgiStartHTML(cgiText(_("Change Settings"))); + cgiCopyTemplateLang("norestart.tmpl"); + } + + cupsFreeOptions(num_settings, settings); + + cgiEndHTML(); + } + else if (cgiGetVariable("SAVECHANGES") && cgiGetVariable("CUPSDCONF")) + { + /* + * Save hand-edited config file... + */ + + http_status_t status; /* PUT status */ + char tempfile[1024]; /* Temporary new cupsd.conf */ + int tempfd; /* Temporary file descriptor */ + cups_file_t *temp; /* Temporary file */ + const char *start, /* Start of line */ + *end; /* End of line */ + + + /* + * Create a temporary file for the new cupsd.conf file... + */ + + if ((tempfd = cupsTempFd(tempfile, sizeof(tempfile))) < 0) + { + cgiStartHTML(cgiText(_("Edit Configuration File"))); + cgiSetVariable("MESSAGE", cgiText(_("Unable to create temporary file:"))); + cgiSetVariable("ERROR", strerror(errno)); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + + perror(tempfile); + return; + } + + if ((temp = cupsFileOpenFd(tempfd, "w")) == NULL) + { + cgiStartHTML(cgiText(_("Edit Configuration File"))); + cgiSetVariable("MESSAGE", cgiText(_("Unable to create temporary file:"))); + cgiSetVariable("ERROR", strerror(errno)); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + + perror(tempfile); + close(tempfd); + unlink(tempfile); + return; + } + + /* + * Copy the cupsd.conf text from the form variable... + */ + + start = cgiGetVariable("CUPSDCONF"); + while (start) + { + if ((end = strstr(start, "\r\n")) == NULL) + if ((end = strstr(start, "\n")) == NULL) + end = start + strlen(start); + + cupsFileWrite(temp, start, end - start); + cupsFilePutChar(temp, '\n'); + + if (*end == '\r') + start = end + 2; + else if (*end == '\n') + start = end + 1; + else + start = NULL; + } + + cupsFileClose(temp); + + /* + * Upload the configuration file to the server... + */ + + status = cupsPutFile(http, "/admin/conf/cupsd.conf", tempfile); + + if (status == HTTP_UNAUTHORIZED) + { + puts("Status: 401\n"); + unlink(tempfile); + exit(0); + } + else if (status != HTTP_CREATED) + { + cgiSetVariable("MESSAGE", + cgiText(_("Unable to upload cupsd.conf file:"))); + cgiSetVariable("ERROR", httpStatus(status)); + + cgiStartHTML(cgiText(_("Edit Configuration File"))); + cgiCopyTemplateLang("error.tmpl"); + } + else + { + cgiSetVariable("refresh_page", "5;URL=/admin/?OP=redirect"); + + cgiStartHTML(cgiText(_("Edit Configuration File"))); + cgiCopyTemplateLang("restart.tmpl"); + } + + cgiEndHTML(); + + unlink(tempfile); + } + else + { + struct stat info; /* cupsd.conf information */ + cups_file_t *cupsd; /* cupsd.conf file */ + char *buffer, /* Buffer for entire file */ + *bufptr, /* Pointer into buffer */ + *bufend; /* End of buffer */ + int ch; /* Character from file */ + char filename[1024]; /* Filename */ + const char *server_root; /* Location of config files */ + + + /* + * Locate the cupsd.conf file... + */ + + if ((server_root = getenv("CUPS_SERVERROOT")) == NULL) + server_root = CUPS_SERVERROOT; + + snprintf(filename, sizeof(filename), "%s/cupsd.conf", server_root); + + /* + * Figure out the size... + */ + + if (stat(filename, &info)) + { + cgiStartHTML(cgiText(_("Edit Configuration File"))); + cgiSetVariable("MESSAGE", + cgiText(_("Unable to access cupsd.conf file:"))); + cgiSetVariable("ERROR", strerror(errno)); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + + perror(filename); + return; + } + + if (info.st_size > (1024 * 1024)) + { + cgiStartHTML(cgiText(_("Edit Configuration File"))); + cgiSetVariable("MESSAGE", + cgiText(_("Unable to access cupsd.conf file:"))); + cgiSetVariable("ERROR", + cgiText(_("Unable to edit cupsd.conf files larger than " + "1MB"))); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + + fprintf(stderr, "ERROR: \"%s\" too large (%ld) to edit!\n", filename, + (long)info.st_size); + return; + } + + /* + * Open the cupsd.conf file... + */ + + if ((cupsd = cupsFileOpen(filename, "r")) == NULL) + { + /* + * Unable to open - log an error... + */ + + cgiStartHTML(cgiText(_("Edit Configuration File"))); + cgiSetVariable("MESSAGE", + cgiText(_("Unable to access cupsd.conf file:"))); + cgiSetVariable("ERROR", strerror(errno)); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + + perror(filename); + return; + } + + /* + * Allocate memory and load the file into a string buffer... + */ + + if ((buffer = calloc(1, info.st_size + 1)) != NULL) + { + cupsFileRead(cupsd, buffer, info.st_size); + cgiSetVariable("CUPSDCONF", buffer); + free(buffer); + } + + cupsFileClose(cupsd); + + /* + * Then get the default cupsd.conf file and put that into a string as + * well... + */ + + strlcat(filename, ".default", sizeof(filename)); + + if (!stat(filename, &info) && info.st_size < (1024 * 1024) && + (cupsd = cupsFileOpen(filename, "r")) != NULL) + { + if ((buffer = calloc(1, 2 * info.st_size + 1)) != NULL) + { + bufend = buffer + 2 * info.st_size - 1; + + for (bufptr = buffer; + bufptr < bufend && (ch = cupsFileGetChar(cupsd)) != EOF;) + { + if (ch == '\\' || ch == '\"') + { + *bufptr++ = '\\'; + *bufptr++ = ch; + } + else if (ch == '\n') + { + *bufptr++ = '\\'; + *bufptr++ = 'n'; + } + else if (ch == '\t') + { + *bufptr++ = '\\'; + *bufptr++ = 't'; + } + else if (ch >= ' ') + *bufptr++ = ch; + } + + *bufptr = '\0'; + + cgiSetVariable("CUPSDCONF_DEFAULT", buffer); + free(buffer); + } + + cupsFileClose(cupsd); + } + + /* + * Show the current config file... + */ + + cgiStartHTML(cgiText(_("Edit Configuration File"))); + + cgiCopyTemplateLang("edit-config.tmpl"); + + cgiEndHTML(); + } +} + + +/* + * 'do_delete_class()' - Delete a class. + */ + +static void +do_delete_class(http_t *http) /* I - HTTP connection */ +{ + ipp_t *request; /* IPP request */ + char uri[HTTP_MAX_URI]; /* Job URI */ + const char *pclass; /* Printer class name */ + + + /* + * Get form variables... + */ + + if (cgiGetVariable("CONFIRM") == NULL) + { + cgiStartHTML(cgiText(_("Delete Class"))); + cgiCopyTemplateLang("class-confirm.tmpl"); + cgiEndHTML(); + return; + } + + if ((pclass = cgiGetVariable("PRINTER_NAME")) != NULL) + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/classes/%s", pclass); + else + { + cgiStartHTML(cgiText(_("Delete Class"))); + cgiSetVariable("ERROR", cgiText(_("Missing form variable"))); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + return; + } + + /* + * Build a CUPS_DELETE_CLASS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + */ + + request = ippNewRequest(CUPS_DELETE_CLASS); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + /* + * Do the request and get back a response... + */ + + ippDelete(cupsDoRequest(http, request, "/admin/")); + + /* + * Show the results... + */ + + if (cupsLastError() == IPP_NOT_AUTHORIZED) + { + puts("Status: 401\n"); + exit(0); + } + else if (cupsLastError() <= IPP_OK_CONFLICT) + { + /* + * Redirect successful updates back to the classes page... + */ + + cgiSetVariable("refresh_page", "5;URL=/admin/?OP=redirect&URL=/classes"); + } + + cgiStartHTML(cgiText(_("Delete Class"))); + + if (cupsLastError() > IPP_OK_CONFLICT) + cgiShowIPPError(_("Unable to delete class:")); + else + cgiCopyTemplateLang("class-deleted.tmpl"); + + cgiEndHTML(); +} + + +/* + * 'do_delete_printer()' - Delete a printer. + */ + +static void +do_delete_printer(http_t *http) /* I - HTTP connection */ +{ + ipp_t *request; /* IPP request */ + char uri[HTTP_MAX_URI]; /* Job URI */ + const char *printer; /* Printer printer name */ + + + /* + * Get form variables... + */ + + if (cgiGetVariable("CONFIRM") == NULL) + { + cgiStartHTML(cgiText(_("Delete Printer"))); + cgiCopyTemplateLang("printer-confirm.tmpl"); + cgiEndHTML(); + return; + } + + if ((printer = cgiGetVariable("PRINTER_NAME")) != NULL) + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", printer); + else + { + cgiStartHTML(cgiText(_("Delete Printer"))); + cgiSetVariable("ERROR", cgiText(_("Missing form variable"))); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + return; + } + + /* + * Build a CUPS_DELETE_PRINTER request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + */ + + request = ippNewRequest(CUPS_DELETE_PRINTER); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + /* + * Do the request and get back a response... + */ + + ippDelete(cupsDoRequest(http, request, "/admin/")); + + /* + * Show the results... + */ + + if (cupsLastError() == IPP_NOT_AUTHORIZED) + { + puts("Status: 401\n"); + exit(0); + } + else if (cupsLastError() <= IPP_OK_CONFLICT) + { + /* + * Redirect successful updates back to the printers page... + */ + + cgiSetVariable("refresh_page", "5;URL=/admin/?OP=redirect&URL=/printers"); + } + + cgiStartHTML(cgiText(_("Delete Printer"))); + + if (cupsLastError() > IPP_OK_CONFLICT) + cgiShowIPPError(_("Unable to delete printer:")); + else + cgiCopyTemplateLang("printer-deleted.tmpl"); + + cgiEndHTML(); +} + + +/* + * 'do_export()' - Export printers to Samba. + */ + +static void +do_export(http_t *http) /* I - HTTP connection */ +{ + int i, j; /* Looping vars */ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + const char *username, /* Samba username */ + *password, /* Samba password */ + *export_all; /* Export all printers? */ + int export_count, /* Number of printers to export */ + printer_count; /* Number of available printers */ + const char *name, /* What name to pull */ + *dest; /* Current destination */ + char ppd[1024]; /* PPD file */ + + + /* + * Get form data... + */ + + username = cgiGetVariable("USERNAME"); + password = cgiGetVariable("PASSWORD"); + export_all = cgiGetVariable("EXPORT_ALL"); + export_count = cgiGetSize("EXPORT_NAME"); + + /* + * Get list of available printers... + */ + + cgiSetSize("PRINTER_NAME", 0); + cgiSetSize("PRINTER_EXPORT", 0); + + request = ippNewRequest(CUPS_GET_PRINTERS); + + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_ENUM, + "printer-type", 0); + + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_ENUM, + "printer-type-mask", CUPS_PRINTER_CLASS | CUPS_PRINTER_REMOTE); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", NULL, "printer-name"); + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + cgiSetIPPVars(response, NULL, NULL, NULL, 0); + ippDelete(response); + + if (!export_all) + { + printer_count = cgiGetSize("PRINTER_NAME"); + + for (i = 0; i < printer_count; i ++) + { + dest = cgiGetArray("PRINTER_NAME", i); + + for (j = 0; j < export_count; j ++) + if (!_cups_strcasecmp(dest, cgiGetArray("EXPORT_NAME", j))) + break; + + cgiSetArray("PRINTER_EXPORT", i, j < export_count ? "Y" : ""); + } + } + } + + /* + * Export or get the printers to export... + */ + + if (username && *username && password && *password && + (export_all || export_count > 0)) + { + /* + * Do export... + */ + + fputs("DEBUG: Export printers...\n", stderr); + + if (export_all) + { + name = "PRINTER_NAME"; + export_count = cgiGetSize("PRINTER_NAME"); + } + else + name = "EXPORT_NAME"; + + for (i = 0; i < export_count; i ++) + { + dest = cgiGetArray(name, i); + + if (!cupsAdminCreateWindowsPPD(http, dest, ppd, sizeof(ppd))) + break; + + j = cupsAdminExportSamba(dest, ppd, "localhost", username, password, + stderr); + + unlink(ppd); + + if (!j) + break; + } + + if (i < export_count) + cgiSetVariable("ERROR", cupsLastErrorString()); + else + { + cgiStartHTML(cgiText(_("Export Printers to Samba"))); + cgiCopyTemplateLang("samba-exported.tmpl"); + cgiEndHTML(); + return; + } + } + else if (username && !*username) + cgiSetVariable("ERROR", + cgiText(_("A Samba username is required to export " + "printer drivers"))); + else if (username && (!password || !*password)) + cgiSetVariable("ERROR", + cgiText(_("A Samba password is required to export " + "printer drivers"))); + + /* + * Show form... + */ + + cgiStartHTML(cgiText(_("Export Printers to Samba"))); + cgiCopyTemplateLang("samba-export.tmpl"); + cgiEndHTML(); +} + + +/* + * 'do_list_printers()' - List available printers. + */ + +static void +do_list_printers(http_t *http) /* I - HTTP connection */ +{ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + ipp_attribute_t *attr; /* IPP attribute */ + + + cgiStartHTML(cgiText(_("List Available Printers"))); + fflush(stdout); + + /* + * Get the list of printers and their devices... + */ + + request = ippNewRequest(CUPS_GET_PRINTERS); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", NULL, "device-uri"); + + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_ENUM, "printer-type", + CUPS_PRINTER_LOCAL); + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_ENUM, "printer-type-mask", + CUPS_PRINTER_LOCAL); + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + /* + * Got the printer list, now load the devices... + */ + + int i; /* Looping var */ + cups_array_t *printer_devices; /* Printer devices for local printers */ + char *printer_device; /* Current printer device */ + + + /* + * Allocate an array and copy the device strings... + */ + + printer_devices = cupsArrayNew((cups_array_func_t)strcmp, NULL); + + for (attr = ippFindAttribute(response, "device-uri", IPP_TAG_URI); + attr; + attr = ippFindNextAttribute(response, "device-uri", IPP_TAG_URI)) + { + cupsArrayAdd(printer_devices, _cupsStrAlloc(attr->values[0].string.text)); + } + + /* + * Free the printer list and get the device list... + */ + + ippDelete(response); + + request = ippNewRequest(CUPS_GET_DEVICES); + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + /* + * Got the device list, let's parse it... + */ + + const char *device_uri, /* device-uri attribute value */ + *device_make_and_model, /* device-make-and-model value */ + *device_info; /* device-info value */ + + + for (i = 0, attr = response->attrs; attr; attr = attr->next) + { + /* + * Skip leading attributes until we hit a device... + */ + + while (attr && attr->group_tag != IPP_TAG_PRINTER) + attr = attr->next; + + if (!attr) + break; + + /* + * Pull the needed attributes from this device... + */ + + device_info = NULL; + device_make_and_model = NULL; + device_uri = NULL; + + while (attr && attr->group_tag == IPP_TAG_PRINTER) + { + if (!strcmp(attr->name, "device-info") && + attr->value_tag == IPP_TAG_TEXT) + device_info = attr->values[0].string.text; + + if (!strcmp(attr->name, "device-make-and-model") && + attr->value_tag == IPP_TAG_TEXT) + device_make_and_model = attr->values[0].string.text; + + if (!strcmp(attr->name, "device-uri") && + attr->value_tag == IPP_TAG_URI) + device_uri = attr->values[0].string.text; + + attr = attr->next; + } + + /* + * See if we have everything needed... + */ + + if (device_info && device_make_and_model && device_uri && + _cups_strcasecmp(device_make_and_model, "unknown") && + strchr(device_uri, ':')) + { + /* + * Yes, now see if there is already a printer for this + * device... + */ + + if (!cupsArrayFind(printer_devices, (void *)device_uri)) + { + /* + * Not found, so it must be a new printer... + */ + + char option[1024], /* Form variables for this device */ + *option_ptr; /* Pointer into string */ + const char *ptr; /* Pointer into device string */ + + + /* + * Format the printer name variable for this device... + * + * We use the device-info string first, then device-uri, + * and finally device-make-and-model to come up with a + * suitable name. + */ + + if (_cups_strncasecmp(device_info, "unknown", 7)) + ptr = device_info; + else if ((ptr = strstr(device_uri, "://")) != NULL) + ptr += 3; + else + ptr = device_make_and_model; + + for (option_ptr = option; + option_ptr < (option + sizeof(option) - 1) && *ptr; + ptr ++) + if (isalnum(*ptr & 255) || *ptr == '_' || *ptr == '-' || + *ptr == '.') + *option_ptr++ = *ptr; + else if ((*ptr == ' ' || *ptr == '/') && option_ptr > option && + option_ptr[-1] != '_') + *option_ptr++ = '_'; + else if (*ptr == '?' || *ptr == '(') + break; + + *option_ptr = '\0'; + + cgiSetArray("TEMPLATE_NAME", i, option); + + /* + * Finally, set the form variables for this printer... + */ + + cgiSetArray("device_info", i, device_info); + cgiSetArray("device_make_and_model", i, device_make_and_model); + cgiSetArray("device_uri", i, device_uri); + i ++; + } + } + + if (!attr) + break; + } + + ippDelete(response); + + /* + * Free the device list... + */ + + for (printer_device = (char *)cupsArrayFirst(printer_devices); + printer_device; + printer_device = (char *)cupsArrayNext(printer_devices)) + _cupsStrFree(printer_device); + + cupsArrayDelete(printer_devices); + } + } + + /* + * Finally, show the printer list... + */ + + cgiCopyTemplateLang("list-available-printers.tmpl"); + + cgiEndHTML(); +} + + +/* + * 'do_menu()' - Show the main menu. + */ + +static void +do_menu(http_t *http) /* I - HTTP connection */ +{ + int num_settings; /* Number of server settings */ + cups_option_t *settings; /* Server settings */ + const char *val; /* Setting value */ + char filename[1024]; /* Temporary filename */ + const char *datadir; /* Location of data files */ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + + + /* + * Get the current server settings... + */ + + if (!cupsAdminGetServerSettings(http, &num_settings, &settings)) + { + cgiSetVariable("SETTINGS_MESSAGE", + cgiText(_("Unable to open cupsd.conf file:"))); + cgiSetVariable("SETTINGS_ERROR", cupsLastErrorString()); + } + + if ((val = cupsGetOption(CUPS_SERVER_DEBUG_LOGGING, num_settings, + settings)) != NULL && atoi(val)) + cgiSetVariable("DEBUG_LOGGING", "CHECKED"); + + if ((val = cupsGetOption(CUPS_SERVER_REMOTE_ADMIN, num_settings, + settings)) != NULL && atoi(val)) + cgiSetVariable("REMOTE_ADMIN", "CHECKED"); + + if ((val = cupsGetOption(CUPS_SERVER_REMOTE_ANY, num_settings, + settings)) != NULL && atoi(val)) + cgiSetVariable("REMOTE_ANY", "CHECKED"); + + if ((val = cupsGetOption(CUPS_SERVER_SHARE_PRINTERS, num_settings, + settings)) != NULL && atoi(val)) + cgiSetVariable("SHARE_PRINTERS", "CHECKED"); + + if ((val = cupsGetOption(CUPS_SERVER_USER_CANCEL_ANY, num_settings, + settings)) != NULL && atoi(val)) + cgiSetVariable("USER_CANCEL_ANY", "CHECKED"); + +#ifdef HAVE_GSSAPI + cgiSetVariable("HAVE_GSSAPI", "1"); + + if ((val = cupsGetOption("DefaultAuthType", num_settings, + settings)) != NULL && !_cups_strcasecmp(val, "Negotiate")) + cgiSetVariable("KERBEROS", "CHECKED"); + else +#endif /* HAVE_GSSAPI */ + cgiSetVariable("KERBEROS", ""); + +#ifdef HAVE_DNSSD + cgiSetVariable("HAVE_DNSSD", "1"); +#endif /* HAVE_DNSSD */ + + if ((val = cupsGetOption("BrowseWebIF", num_settings, + settings)) == NULL) + val = "No"; + + if (!_cups_strcasecmp(val, "yes") || !_cups_strcasecmp(val, "on") || + !_cups_strcasecmp(val, "true")) + cgiSetVariable("BROWSE_WEB_IF", "CHECKED"); + + if ((val = cupsGetOption("PreserveJobHistory", num_settings, + settings)) == NULL) + val = "Yes"; + + if (!_cups_strcasecmp(val, "yes") || !_cups_strcasecmp(val, "on") || + !_cups_strcasecmp(val, "true")) + { + cgiSetVariable("PRESERVE_JOB_HISTORY", "CHECKED"); + + if ((val = cupsGetOption("PreserveJobFiles", num_settings, + settings)) == NULL) + val = "No"; + + if (!_cups_strcasecmp(val, "yes") || !_cups_strcasecmp(val, "on") || + !_cups_strcasecmp(val, "true")) + cgiSetVariable("PRESERVE_JOB_FILES", "CHECKED"); + } + + if ((val = cupsGetOption("MaxClients", num_settings, settings)) == NULL) + val = "100"; + + cgiSetVariable("MAX_CLIENTS", val); + + if ((val = cupsGetOption("MaxJobs", num_settings, settings)) == NULL) + val = "500"; + + cgiSetVariable("MAX_JOBS", val); + + if ((val = cupsGetOption("MaxLogSize", num_settings, settings)) == NULL) + val = "1m"; + + cgiSetVariable("MAX_LOG_SIZE", val); + + cupsFreeOptions(num_settings, settings); + + /* + * See if Samba and the Windows drivers are installed... + */ + + if ((datadir = getenv("CUPS_DATADIR")) == NULL) + datadir = CUPS_DATADIR; + + snprintf(filename, sizeof(filename), "%s/drivers/pscript5.dll", datadir); + if (!access(filename, R_OK)) + { + /* + * Found Windows 2000 driver file, see if we have smbclient and + * rpcclient... + */ + + if (cupsFileFind("smbclient", getenv("PATH"), 1, filename, + sizeof(filename)) && + cupsFileFind("rpcclient", getenv("PATH"), 1, filename, + sizeof(filename))) + cgiSetVariable("HAVE_SAMBA", "Y"); + else + { + if (!cupsFileFind("smbclient", getenv("PATH"), 1, filename, + sizeof(filename))) + fputs("ERROR: smbclient not found!\n", stderr); + + if (!cupsFileFind("rpcclient", getenv("PATH"), 1, filename, + sizeof(filename))) + fputs("ERROR: rpcclient not found!\n", stderr); + } + } + else + perror(filename); + + /* + * Subscriptions... + */ + + request = ippNewRequest(IPP_GET_SUBSCRIPTIONS); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, "ipp://localhost/"); + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + cgiSetIPPVars(response, NULL, NULL, NULL, 0); + ippDelete(response); + } + + /* + * Finally, show the main menu template... + */ + + cgiStartHTML(cgiText(_("Administration"))); + + cgiCopyTemplateLang("admin.tmpl"); + + cgiEndHTML(); +} + + +/* + * 'do_set_allowed_users()' - Set the allowed/denied users for a queue. + */ + +static void +do_set_allowed_users(http_t *http) /* I - HTTP connection */ +{ + int i; /* Looping var */ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + char uri[HTTP_MAX_URI]; /* Printer URI */ + const char *printer, /* Printer name (purge-jobs) */ + *is_class, /* Is a class? */ + *users, /* List of users or groups */ + *type; /* Allow/deny type */ + int num_users; /* Number of users */ + char *ptr, /* Pointer into users string */ + *end, /* Pointer to end of users string */ + quote; /* Quote character */ + ipp_attribute_t *attr; /* Attribute */ + static const char * const attrs[] = /* Requested attributes */ + { + "requesting-user-name-allowed", + "requesting-user-name-denied" + }; + + + is_class = cgiGetVariable("IS_CLASS"); + printer = cgiGetVariable("PRINTER_NAME"); + + if (!printer) + { + cgiSetVariable("ERROR", cgiText(_("Missing form variable"))); + cgiStartHTML(cgiText(_("Set Allowed Users"))); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + return; + } + + users = cgiGetVariable("users"); + type = cgiGetVariable("type"); + + if (!users || !type || + (strcmp(type, "requesting-user-name-allowed") && + strcmp(type, "requesting-user-name-denied"))) + { + /* + * Build a Get-Printer-Attributes request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requested-attributes + */ + + request = ippNewRequest(IPP_GET_PRINTER_ATTRIBUTES); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, is_class ? "/classes/%s" : "/printers/%s", + printer); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", + (int)(sizeof(attrs) / sizeof(attrs[0])), NULL, attrs); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + cgiSetIPPVars(response, NULL, NULL, NULL, 0); + + ippDelete(response); + } + + cgiStartHTML(cgiText(_("Set Allowed Users"))); + + if (cupsLastError() == IPP_NOT_AUTHORIZED) + { + puts("Status: 401\n"); + exit(0); + } + else if (cupsLastError() > IPP_OK_CONFLICT) + cgiShowIPPError(_("Unable to get printer attributes:")); + else + cgiCopyTemplateLang("users.tmpl"); + + cgiEndHTML(); + } + else + { + /* + * Save the changes... + */ + + for (num_users = 0, ptr = (char *)users; *ptr; num_users ++) + { + /* + * Skip whitespace and commas... + */ + + while (*ptr == ',' || isspace(*ptr & 255)) + ptr ++; + + if (!*ptr) + break; + + if (*ptr == '\'' || *ptr == '\"') + { + /* + * Scan quoted name... + */ + + quote = *ptr++; + + for (end = ptr; *end; end ++) + if (*end == quote) + break; + } + else + { + /* + * Scan space or comma-delimited name... + */ + + for (end = ptr; *end; end ++) + if (isspace(*end & 255) || *end == ',') + break; + } + + /* + * Advance to the next name... + */ + + ptr = end; + } + + /* + * Build a CUPS-Add-Printer/Class request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requesting-user-name-{allowed,denied} + */ + + request = ippNewRequest(is_class ? CUPS_ADD_CLASS : CUPS_ADD_PRINTER); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, is_class ? "/classes/%s" : "/printers/%s", + printer); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + if (num_users == 0) + ippAddString(request, IPP_TAG_PRINTER, IPP_TAG_NAME, + "requesting-user-name-allowed", NULL, "all"); + else + { + attr = ippAddStrings(request, IPP_TAG_PRINTER, IPP_TAG_NAME, + type, num_users, NULL, NULL); + + for (i = 0, ptr = (char *)users; *ptr; i ++) + { + /* + * Skip whitespace and commas... + */ + + while (*ptr == ',' || isspace(*ptr & 255)) + ptr ++; + + if (!*ptr) + break; + + if (*ptr == '\'' || *ptr == '\"') + { + /* + * Scan quoted name... + */ + + quote = *ptr++; + + for (end = ptr; *end; end ++) + if (*end == quote) + break; + } + else + { + /* + * Scan space or comma-delimited name... + */ + + for (end = ptr; *end; end ++) + if (isspace(*end & 255) || *end == ',') + break; + } + + /* + * Terminate the name... + */ + + if (*end) + *end++ = '\0'; + + /* + * Add the name... + */ + + attr->values[i].string.text = _cupsStrAlloc(ptr); + + /* + * Advance to the next name... + */ + + ptr = end; + } + } + + /* + * Do the request and get back a response... + */ + + ippDelete(cupsDoRequest(http, request, "/admin/")); + + if (cupsLastError() == IPP_NOT_AUTHORIZED) + { + puts("Status: 401\n"); + exit(0); + } + else if (cupsLastError() > IPP_OK_CONFLICT) + { + cgiStartHTML(cgiText(_("Set Allowed Users"))); + cgiShowIPPError(_("Unable to change printer:")); + } + else + { + /* + * Redirect successful updates back to the printer page... + */ + + char url[1024], /* Printer/class URL */ + refresh[1024]; /* Refresh URL */ + + + cgiRewriteURL(uri, url, sizeof(url), NULL); + cgiFormEncode(uri, url, sizeof(uri)); + snprintf(refresh, sizeof(refresh), "5;URL=/admin/?OP=redirect&URL=%s", + uri); + cgiSetVariable("refresh_page", refresh); + + cgiStartHTML(cgiText(_("Set Allowed Users"))); + + cgiCopyTemplateLang(is_class ? "class-modified.tmpl" : + "printer-modified.tmpl"); + } + + cgiEndHTML(); + } +} + + +/* + * 'do_set_default()' - Set the server default printer/class. + */ + +static void +do_set_default(http_t *http) /* I - HTTP connection */ +{ + const char *title; /* Page title */ + ipp_t *request; /* IPP request */ + char uri[HTTP_MAX_URI]; /* Printer URI */ + const char *printer, /* Printer name (purge-jobs) */ + *is_class; /* Is a class? */ + + + is_class = cgiGetVariable("IS_CLASS"); + printer = cgiGetVariable("PRINTER_NAME"); + title = cgiText(_("Set As Server Default")); + + if (!printer) + { + cgiSetVariable("ERROR", cgiText(_("Missing form variable"))); + cgiStartHTML(title); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + return; + } + + /* + * Build a printer request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + */ + + request = ippNewRequest(CUPS_SET_DEFAULT); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, is_class ? "/classes/%s" : "/printers/%s", + printer); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + /* + * Do the request and get back a response... + */ + + ippDelete(cupsDoRequest(http, request, "/admin/")); + + if (cupsLastError() == IPP_NOT_AUTHORIZED) + { + puts("Status: 401\n"); + exit(0); + } + else if (cupsLastError() > IPP_OK_CONFLICT) + { + cgiStartHTML(title); + cgiShowIPPError(_("Unable to set server default:")); + } + else + { + /* + * Redirect successful updates back to the printer page... + */ + + char url[1024], /* Printer/class URL */ + refresh[1024]; /* Refresh URL */ + + + cgiRewriteURL(uri, url, sizeof(url), NULL); + cgiFormEncode(uri, url, sizeof(uri)); + snprintf(refresh, sizeof(refresh), "5;URL=/admin/?OP=redirect&URL=%s", uri); + cgiSetVariable("refresh_page", refresh); + + cgiStartHTML(title); + cgiCopyTemplateLang("printer-default.tmpl"); + } + + cgiEndHTML(); +} + + +/* + * 'do_set_options()' - Configure the default options for a queue. + */ + +static void +do_set_options(http_t *http, /* I - HTTP connection */ + int is_class) /* I - Set options for class? */ +{ + int i, j, k, m; /* Looping vars */ + int have_options; /* Have options? */ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + ipp_attribute_t *attr; /* IPP attribute */ + char uri[HTTP_MAX_URI]; /* Job URI */ + const char *var; /* Variable value */ + const char *printer; /* Printer printer name */ + const char *filename; /* PPD filename */ + char tempfile[1024]; /* Temporary filename */ + cups_file_t *in, /* Input file */ + *out; /* Output file */ + char line[1024], /* Line from PPD file */ + value[1024], /* Option value */ + keyword[1024], /* Keyword from Default line */ + *keyptr; /* Pointer into keyword... */ + ppd_file_t *ppd; /* PPD file */ + ppd_group_t *group; /* Option group */ + ppd_option_t *option; /* Option */ + ppd_coption_t *coption; /* Custom option */ + ppd_cparam_t *cparam; /* Custom parameter */ + ppd_attr_t *ppdattr; /* PPD attribute */ + const char *title; /* Page title */ + + + title = cgiText(is_class ? _("Set Class Options") : _("Set Printer Options")); + + fprintf(stderr, "DEBUG: do_set_options(http=%p, is_class=%d)\n", http, + is_class); + + /* + * Get the printer name... + */ + + if ((printer = cgiGetVariable("PRINTER_NAME")) != NULL) + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, is_class ? "/classes/%s" : "/printers/%s", + printer); + else + { + cgiSetVariable("ERROR", cgiText(_("Missing form variable"))); + cgiStartHTML(title); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + return; + } + + fprintf(stderr, "DEBUG: printer=\"%s\", uri=\"%s\"...\n", printer, uri); + + /* + * If the user clicks on the Auto-Configure button, send an AutoConfigure + * command file to the printer... + */ + + if (cgiGetVariable("AUTOCONFIGURE")) + { + cgiPrintCommand(http, printer, "AutoConfigure", "Set Default Options"); + return; + } + + /* + * Get the PPD file... + */ + + if (is_class) + filename = NULL; + else + filename = cupsGetPPD2(http, printer); + + if (filename) + { + fprintf(stderr, "DEBUG: Got PPD file: \"%s\"\n", filename); + + if ((ppd = ppdOpenFile(filename)) == NULL) + { + cgiSetVariable("ERROR", ppdErrorString(ppdLastError(&i))); + cgiSetVariable("MESSAGE", cgiText(_("Unable to open PPD file:"))); + cgiStartHTML(title); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + return; + } + } + else + { + fputs("DEBUG: No PPD file\n", stderr); + ppd = NULL; + } + + if (cgiGetVariable("job_sheets_start") != NULL || + cgiGetVariable("job_sheets_end") != NULL) + have_options = 1; + else + have_options = 0; + + if (ppd) + { + ppdMarkDefaults(ppd); + + for (option = ppdFirstOption(ppd); + option; + option = ppdNextOption(ppd)) + { + if ((var = cgiGetVariable(option->keyword)) != NULL) + { + have_options = 1; + ppdMarkOption(ppd, option->keyword, var); + fprintf(stderr, "DEBUG: Set %s to %s...\n", option->keyword, var); + } + else + fprintf(stderr, "DEBUG: Didn't find %s...\n", option->keyword); + } + } + + if (!have_options || ppdConflicts(ppd)) + { + /* + * Show the options to the user... + */ + + fputs("DEBUG: Showing options...\n", stderr); + + /* + * Show auto-configure button if supported... + */ + + if (ppd) + { + if (ppd->num_filters == 0 || + ((ppdattr = ppdFindAttr(ppd, "cupsCommands", NULL)) != NULL && + ppdattr->value && strstr(ppdattr->value, "AutoConfigure"))) + cgiSetVariable("HAVE_AUTOCONFIGURE", "YES"); + else + { + for (i = 0; i < ppd->num_filters; i ++) + if (!strncmp(ppd->filters[i], "application/vnd.cups-postscript", 31)) + { + cgiSetVariable("HAVE_AUTOCONFIGURE", "YES"); + break; + } + } + } + + /* + * Get the printer attributes... + */ + + request = ippNewRequest(IPP_GET_PRINTER_ATTRIBUTES); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", printer); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + response = cupsDoRequest(http, request, "/"); + + /* + * List the groups used as "tabs"... + */ + + i = 0; + + if (ppd) + { + for (group = ppd->groups; + i < ppd->num_groups; + i ++, group ++) + { + cgiSetArray("GROUP_ID", i, group->name); + + if (!strcmp(group->name, "InstallableOptions")) + cgiSetArray("GROUP", i, cgiText(_("Options Installed"))); + else + cgiSetArray("GROUP", i, group->text); + } + } + + if (ippFindAttribute(response, "job-sheets-supported", IPP_TAG_ZERO)) + { + cgiSetArray("GROUP_ID", i, "CUPS_BANNERS"); + cgiSetArray("GROUP", i ++, cgiText(_("Banners"))); + } + + if (ippFindAttribute(response, "printer-error-policy-supported", + IPP_TAG_ZERO) || + ippFindAttribute(response, "printer-op-policy-supported", + IPP_TAG_ZERO)) + { + cgiSetArray("GROUP_ID", i, "CUPS_POLICIES"); + cgiSetArray("GROUP", i ++, cgiText(_("Policies"))); + } + + if ((attr = ippFindAttribute(response, "port-monitor-supported", + IPP_TAG_NAME)) != NULL && attr->num_values > 1) + { + cgiSetArray("GROUP_ID", i, "CUPS_PORT_MONITOR"); + cgiSetArray("GROUP", i, cgiText(_("Port Monitor"))); + } + + cgiStartHTML(cgiText(_("Set Printer Options"))); + cgiCopyTemplateLang("set-printer-options-header.tmpl"); + + if (ppd) + { + ppdLocalize(ppd); + + if (ppdConflicts(ppd)) + { + for (i = ppd->num_groups, k = 0, group = ppd->groups; + i > 0; + i --, group ++) + for (j = group->num_options, option = group->options; + j > 0; + j --, option ++) + if (option->conflicted) + { + cgiSetArray("ckeyword", k, option->keyword); + cgiSetArray("ckeytext", k, option->text); + + for (m = 0; m < option->num_choices; m ++) + { + if (option->choices[m].marked) + { + cgiSetArray("cchoice", k, option->choices[m].text); + break; + } + } + + k ++; + } + + cgiCopyTemplateLang("option-conflict.tmpl"); + } + + for (i = ppd->num_groups, group = ppd->groups; + i > 0; + i --, group ++) + { + for (j = group->num_options, option = group->options; + j > 0; + j --, option ++) + { + if (!strcmp(option->keyword, "PageRegion")) + continue; + + if (option->num_choices > 1) + break; + } + + if (j == 0) + continue; + + cgiSetVariable("GROUP_ID", group->name); + + if (!strcmp(group->name, "InstallableOptions")) + cgiSetVariable("GROUP", cgiText(_("Options Installed"))); + else + cgiSetVariable("GROUP", group->text); + + cgiCopyTemplateLang("option-header.tmpl"); + + for (j = group->num_options, option = group->options; + j > 0; + j --, option ++) + { + if (!strcmp(option->keyword, "PageRegion") || option->num_choices < 2) + continue; + + cgiSetVariable("KEYWORD", option->keyword); + cgiSetVariable("KEYTEXT", option->text); + + if (option->conflicted) + cgiSetVariable("CONFLICTED", "1"); + else + cgiSetVariable("CONFLICTED", "0"); + + cgiSetSize("CHOICES", 0); + cgiSetSize("TEXT", 0); + for (k = 0, m = 0; k < option->num_choices; k ++) + { + cgiSetArray("CHOICES", m, option->choices[k].choice); + cgiSetArray("TEXT", m, option->choices[k].text); + + m ++; + + if (option->choices[k].marked) + cgiSetVariable("DEFCHOICE", option->choices[k].choice); + } + + cgiSetSize("PARAMS", 0); + cgiSetSize("PARAMTEXT", 0); + cgiSetSize("PARAMVALUE", 0); + cgiSetSize("INPUTTYPE", 0); + + if ((coption = ppdFindCustomOption(ppd, option->keyword))) + { + const char *units = NULL; /* Units value, if any */ + + cgiSetVariable("ISCUSTOM", "1"); + + for (cparam = ppdFirstCustomParam(coption), m = 0; + cparam; + cparam = ppdNextCustomParam(coption), m ++) + { + if (!_cups_strcasecmp(option->keyword, "PageSize") && + _cups_strcasecmp(cparam->name, "Width") && + _cups_strcasecmp(cparam->name, "Height")) + { + m --; + continue; + } + + cgiSetArray("PARAMS", m, cparam->name); + cgiSetArray("PARAMTEXT", m, cparam->text); + cgiSetArray("INPUTTYPE", m, "text"); + + switch (cparam->type) + { + case PPD_CUSTOM_POINTS : + if (!_cups_strncasecmp(option->defchoice, "Custom.", 7)) + { + units = option->defchoice + strlen(option->defchoice) - 2; + + if (strcmp(units, "mm") && strcmp(units, "cm") && + strcmp(units, "in") && strcmp(units, "ft")) + { + if (units[1] == 'm') + units ++; + else + units = "pt"; + } + } + else + units = "pt"; + + if (!strcmp(units, "mm")) + snprintf(value, sizeof(value), "%g", + cparam->current.custom_points / 72.0 * 25.4); + else if (!strcmp(units, "cm")) + snprintf(value, sizeof(value), "%g", + cparam->current.custom_points / 72.0 * 2.54); + else if (!strcmp(units, "in")) + snprintf(value, sizeof(value), "%g", + cparam->current.custom_points / 72.0); + else if (!strcmp(units, "ft")) + snprintf(value, sizeof(value), "%g", + cparam->current.custom_points / 72.0 / 12.0); + else if (!strcmp(units, "m")) + snprintf(value, sizeof(value), "%g", + cparam->current.custom_points / 72.0 * 0.0254); + else + snprintf(value, sizeof(value), "%g", + cparam->current.custom_points); + cgiSetArray("PARAMVALUE", m, value); + break; + + case PPD_CUSTOM_CURVE : + case PPD_CUSTOM_INVCURVE : + case PPD_CUSTOM_REAL : + snprintf(value, sizeof(value), "%g", + cparam->current.custom_real); + cgiSetArray("PARAMVALUE", m, value); + break; + + case PPD_CUSTOM_INT: + snprintf(value, sizeof(value), "%d", + cparam->current.custom_int); + cgiSetArray("PARAMVALUE", m, value); + break; + + case PPD_CUSTOM_PASSCODE: + case PPD_CUSTOM_PASSWORD: + if (cparam->current.custom_password) + cgiSetArray("PARAMVALUE", m, + cparam->current.custom_password); + else + cgiSetArray("PARAMVALUE", m, ""); + cgiSetArray("INPUTTYPE", m, "password"); + break; + + case PPD_CUSTOM_STRING: + if (cparam->current.custom_string) + cgiSetArray("PARAMVALUE", m, + cparam->current.custom_string); + else + cgiSetArray("PARAMVALUE", m, ""); + break; + } + } + + if (units) + { + cgiSetArray("PARAMS", m, "Units"); + cgiSetArray("PARAMTEXT", m, cgiText(_("Units"))); + cgiSetArray("PARAMVALUE", m, units); + } + } + else + cgiSetVariable("ISCUSTOM", "0"); + + switch (option->ui) + { + case PPD_UI_BOOLEAN : + cgiCopyTemplateLang("option-boolean.tmpl"); + break; + case PPD_UI_PICKONE : + cgiCopyTemplateLang("option-pickone.tmpl"); + break; + case PPD_UI_PICKMANY : + cgiCopyTemplateLang("option-pickmany.tmpl"); + break; + } + } + + cgiCopyTemplateLang("option-trailer.tmpl"); + } + } + + if ((attr = ippFindAttribute(response, "job-sheets-supported", + IPP_TAG_ZERO)) != NULL) + { + /* + * Add the job sheets options... + */ + + cgiSetVariable("GROUP_ID", "CUPS_BANNERS"); + cgiSetVariable("GROUP", cgiText(_("Banners"))); + cgiCopyTemplateLang("option-header.tmpl"); + + cgiSetSize("CHOICES", attr->num_values); + cgiSetSize("TEXT", attr->num_values); + for (k = 0; k < attr->num_values; k ++) + { + cgiSetArray("CHOICES", k, attr->values[k].string.text); + cgiSetArray("TEXT", k, attr->values[k].string.text); + } + + attr = ippFindAttribute(response, "job-sheets-default", IPP_TAG_ZERO); + + cgiSetVariable("KEYWORD", "job_sheets_start"); + cgiSetVariable("KEYTEXT", + /* TRANSLATORS: Banner/cover sheet before the print job. */ + cgiText(_("Starting Banner"))); + cgiSetVariable("DEFCHOICE", attr != NULL ? + attr->values[0].string.text : ""); + + cgiCopyTemplateLang("option-pickone.tmpl"); + + cgiSetVariable("KEYWORD", "job_sheets_end"); + cgiSetVariable("KEYTEXT", + /* TRANSLATORS: Banner/cover sheet after the print job. */ + cgiText(_("Ending Banner"))); + cgiSetVariable("DEFCHOICE", attr != NULL && attr->num_values > 1 ? + attr->values[1].string.text : ""); + + cgiCopyTemplateLang("option-pickone.tmpl"); + + cgiCopyTemplateLang("option-trailer.tmpl"); + } + + if (ippFindAttribute(response, "printer-error-policy-supported", + IPP_TAG_ZERO) || + ippFindAttribute(response, "printer-op-policy-supported", + IPP_TAG_ZERO)) + { + /* + * Add the error and operation policy options... + */ + + cgiSetVariable("GROUP_ID", "CUPS_POLICIES"); + cgiSetVariable("GROUP", cgiText(_("Policies"))); + cgiCopyTemplateLang("option-header.tmpl"); + + /* + * Error policy... + */ + + attr = ippFindAttribute(response, "printer-error-policy-supported", + IPP_TAG_ZERO); + + if (attr) + { + cgiSetSize("CHOICES", attr->num_values); + cgiSetSize("TEXT", attr->num_values); + for (k = 0; k < attr->num_values; k ++) + { + cgiSetArray("CHOICES", k, attr->values[k].string.text); + cgiSetArray("TEXT", k, attr->values[k].string.text); + } + + attr = ippFindAttribute(response, "printer-error-policy", + IPP_TAG_ZERO); + + cgiSetVariable("KEYWORD", "printer_error_policy"); + cgiSetVariable("KEYTEXT", cgiText(_("Error Policy"))); + cgiSetVariable("DEFCHOICE", attr == NULL ? + "" : attr->values[0].string.text); + } + + cgiCopyTemplateLang("option-pickone.tmpl"); + + /* + * Operation policy... + */ + + attr = ippFindAttribute(response, "printer-op-policy-supported", + IPP_TAG_ZERO); + + if (attr) + { + cgiSetSize("CHOICES", attr->num_values); + cgiSetSize("TEXT", attr->num_values); + for (k = 0; k < attr->num_values; k ++) + { + cgiSetArray("CHOICES", k, attr->values[k].string.text); + cgiSetArray("TEXT", k, attr->values[k].string.text); + } + + attr = ippFindAttribute(response, "printer-op-policy", IPP_TAG_ZERO); + + cgiSetVariable("KEYWORD", "printer_op_policy"); + cgiSetVariable("KEYTEXT", cgiText(_("Operation Policy"))); + cgiSetVariable("DEFCHOICE", attr == NULL ? + "" : attr->values[0].string.text); + + cgiCopyTemplateLang("option-pickone.tmpl"); + } + + cgiCopyTemplateLang("option-trailer.tmpl"); + } + + /* + * Binary protocol support... + */ + + if ((attr = ippFindAttribute(response, "port-monitor-supported", + IPP_TAG_NAME)) != NULL && attr->num_values > 1) + { + cgiSetVariable("GROUP_ID", "CUPS_PORT_MONITOR"); + cgiSetVariable("GROUP", cgiText(_("Port Monitor"))); + + cgiSetSize("CHOICES", attr->num_values); + cgiSetSize("TEXT", attr->num_values); + + for (i = 0; i < attr->num_values; i ++) + { + cgiSetArray("CHOICES", i, attr->values[i].string.text); + cgiSetArray("TEXT", i, attr->values[i].string.text); + } + + attr = ippFindAttribute(response, "port-monitor", IPP_TAG_NAME); + cgiSetVariable("KEYWORD", "port_monitor"); + cgiSetVariable("KEYTEXT", cgiText(_("Port Monitor"))); + cgiSetVariable("DEFCHOICE", attr ? attr->values[0].string.text : "none"); + + cgiCopyTemplateLang("option-header.tmpl"); + cgiCopyTemplateLang("option-pickone.tmpl"); + cgiCopyTemplateLang("option-trailer.tmpl"); + } + + cgiCopyTemplateLang("set-printer-options-trailer.tmpl"); + cgiEndHTML(); + + ippDelete(response); + } + else + { + /* + * Set default options... + */ + + fputs("DEBUG: Setting options...\n", stderr); + + if (filename) + { + out = cupsTempFile2(tempfile, sizeof(tempfile)); + in = cupsFileOpen(filename, "r"); + + if (!in || !out) + { + cgiSetVariable("ERROR", strerror(errno)); + cgiStartHTML(cgiText(_("Set Printer Options"))); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + + if (in) + cupsFileClose(in); + + if (out) + { + cupsFileClose(out); + unlink(tempfile); + } + + unlink(filename); + return; + } + + while (cupsFileGets(in, line, sizeof(line))) + { + if (!strncmp(line, "*cupsProtocol:", 14)) + continue; + else if (strncmp(line, "*Default", 8)) + cupsFilePrintf(out, "%s\n", line); + else + { + /* + * Get default option name... + */ + + strlcpy(keyword, line + 8, sizeof(keyword)); + + for (keyptr = keyword; *keyptr; keyptr ++) + if (*keyptr == ':' || isspace(*keyptr & 255)) + break; + + *keyptr = '\0'; + + if (!strcmp(keyword, "PageRegion") || + !strcmp(keyword, "PaperDimension") || + !strcmp(keyword, "ImageableArea")) + var = get_option_value(ppd, "PageSize", value, sizeof(value)); + else + var = get_option_value(ppd, keyword, value, sizeof(value)); + + if (!var) + cupsFilePrintf(out, "%s\n", line); + else + cupsFilePrintf(out, "*Default%s: %s\n", keyword, var); + } + } + + cupsFileClose(in); + cupsFileClose(out); + } + else + { + /* + * Make sure temporary filename is cleared when there is no PPD... + */ + + tempfile[0] = '\0'; + } + + /* + * Build a CUPS_ADD_MODIFY_CLASS/PRINTER request, which requires the + * following attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * job-sheets-default + * printer-error-policy + * printer-op-policy + * [ppd file] + */ + + request = ippNewRequest(is_class ? CUPS_ADD_MODIFY_CLASS : + CUPS_ADD_MODIFY_PRINTER); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + attr = ippAddStrings(request, IPP_TAG_PRINTER, IPP_TAG_NAME, + "job-sheets-default", 2, NULL, NULL); + attr->values[0].string.text = _cupsStrAlloc(cgiGetVariable("job_sheets_start")); + attr->values[1].string.text = _cupsStrAlloc(cgiGetVariable("job_sheets_end")); + + if ((var = cgiGetVariable("printer_error_policy")) != NULL) + ippAddString(request, IPP_TAG_PRINTER, IPP_TAG_NAME, + "printer-error-policy", NULL, var); + + if ((var = cgiGetVariable("printer_op_policy")) != NULL) + ippAddString(request, IPP_TAG_PRINTER, IPP_TAG_NAME, + "printer-op-policy", NULL, var); + + if ((var = cgiGetVariable("port_monitor")) != NULL) + ippAddString(request, IPP_TAG_PRINTER, IPP_TAG_NAME, + "port-monitor", NULL, var); + + /* + * Do the request and get back a response... + */ + + if (filename) + ippDelete(cupsDoFileRequest(http, request, "/admin/", tempfile)); + else + ippDelete(cupsDoRequest(http, request, "/admin/")); + + if (cupsLastError() == IPP_NOT_AUTHORIZED) + { + puts("Status: 401\n"); + exit(0); + } + else if (cupsLastError() > IPP_OK_CONFLICT) + { + cgiStartHTML(title); + cgiShowIPPError(_("Unable to set options:")); + } + else + { + /* + * Redirect successful updates back to the printer page... + */ + + char refresh[1024]; /* Refresh URL */ + + + cgiFormEncode(uri, printer, sizeof(uri)); + snprintf(refresh, sizeof(refresh), "5;URL=/admin/?OP=redirect&URL=/%s/%s", + is_class ? "classes" : "printers", uri); + cgiSetVariable("refresh_page", refresh); + + cgiStartHTML(title); + + cgiCopyTemplateLang("printer-configured.tmpl"); + } + + cgiEndHTML(); + + if (filename) + unlink(tempfile); + } + + if (filename) + unlink(filename); +} + + +/* + * 'do_set_sharing()' - Set printer-is-shared value. + */ + +static void +do_set_sharing(http_t *http) /* I - HTTP connection */ +{ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + char uri[HTTP_MAX_URI]; /* Printer URI */ + const char *printer, /* Printer name */ + *is_class, /* Is a class? */ + *shared; /* Sharing value */ + + + is_class = cgiGetVariable("IS_CLASS"); + printer = cgiGetVariable("PRINTER_NAME"); + shared = cgiGetVariable("SHARED"); + + if (!printer || !shared) + { + cgiSetVariable("ERROR", cgiText(_("Missing form variable"))); + cgiStartHTML(cgiText(_("Set Publishing"))); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + return; + } + + /* + * Build a CUPS-Add-Printer/CUPS-Add-Class request, which requires the + * following attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * printer-is-shared + */ + + request = ippNewRequest(is_class ? CUPS_ADD_CLASS : CUPS_ADD_PRINTER); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, is_class ? "/classes/%s" : "/printers/%s", + printer); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + ippAddBoolean(request, IPP_TAG_OPERATION, "printer-is-shared", atoi(shared)); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(http, request, "/admin/")) != NULL) + { + cgiSetIPPVars(response, NULL, NULL, NULL, 0); + + ippDelete(response); + } + + if (cupsLastError() == IPP_NOT_AUTHORIZED) + { + puts("Status: 401\n"); + exit(0); + } + else if (cupsLastError() > IPP_OK_CONFLICT) + { + cgiStartHTML(cgiText(_("Set Publishing"))); + cgiShowIPPError(_("Unable to change printer-is-shared attribute:")); + } + else + { + /* + * Redirect successful updates back to the printer page... + */ + + char url[1024], /* Printer/class URL */ + refresh[1024]; /* Refresh URL */ + + + cgiRewriteURL(uri, url, sizeof(url), NULL); + cgiFormEncode(uri, url, sizeof(uri)); + snprintf(refresh, sizeof(refresh), "5;URL=/admin/?OP=redirect&URL=%s", uri); + cgiSetVariable("refresh_page", refresh); + + cgiStartHTML(cgiText(_("Set Publishing"))); + cgiCopyTemplateLang(is_class ? "class-modified.tmpl" : + "printer-modified.tmpl"); + } + + cgiEndHTML(); +} + + +/* + * 'get_option_value()' - Return the value of an option. + * + * This function also handles generation of custom option values. + */ + +static char * /* O - Value string or NULL on error */ +get_option_value( + ppd_file_t *ppd, /* I - PPD file */ + const char *name, /* I - Option name */ + char *buffer, /* I - String buffer */ + size_t bufsize) /* I - Size of buffer */ +{ + char *bufptr, /* Pointer into buffer */ + *bufend; /* End of buffer */ + ppd_coption_t *coption; /* Custom option */ + ppd_cparam_t *cparam; /* Current custom parameter */ + char keyword[256]; /* Parameter name */ + const char *val, /* Parameter value */ + *uval; /* Units value */ + long integer; /* Integer value */ + double number, /* Number value */ + number_points; /* Number in points */ + + + /* + * See if we have a custom option choice... + */ + + if ((val = cgiGetVariable(name)) == NULL) + { + /* + * Option not found! + */ + + return (NULL); + } + else if (_cups_strcasecmp(val, "Custom") || + (coption = ppdFindCustomOption(ppd, name)) == NULL) + { + /* + * Not a custom choice... + */ + + strlcpy(buffer, val, bufsize); + return (buffer); + } + + /* + * OK, we have a custom option choice, format it... + */ + + *buffer = '\0'; + + if (!strcmp(coption->keyword, "PageSize")) + { + const char *lval; /* Length string value */ + double width, /* Width value */ + width_points, /* Width in points */ + length, /* Length value */ + length_points; /* Length in points */ + + + val = cgiGetVariable("PageSize.Width"); + lval = cgiGetVariable("PageSize.Height"); + uval = cgiGetVariable("PageSize.Units"); + + if (!val || !lval || !uval || + (width = strtod(val, NULL)) == 0.0 || + (length = strtod(lval, NULL)) == 0.0 || + (strcmp(uval, "pt") && strcmp(uval, "in") && strcmp(uval, "ft") && + strcmp(uval, "cm") && strcmp(uval, "mm") && strcmp(uval, "m"))) + return (NULL); + + width_points = get_points(width, uval); + length_points = get_points(length, uval); + + if (width_points < ppd->custom_min[0] || + width_points > ppd->custom_max[0] || + length_points < ppd->custom_min[1] || + length_points > ppd->custom_max[1]) + return (NULL); + + snprintf(buffer, bufsize, "Custom.%gx%g%s", width, length, uval); + } + else if (cupsArrayCount(coption->params) == 1) + { + cparam = ppdFirstCustomParam(coption); + snprintf(keyword, sizeof(keyword), "%s.%s", coption->keyword, cparam->name); + + if ((val = cgiGetVariable(keyword)) == NULL) + return (NULL); + + switch (cparam->type) + { + case PPD_CUSTOM_CURVE : + case PPD_CUSTOM_INVCURVE : + case PPD_CUSTOM_REAL : + if ((number = strtod(val, NULL)) == 0.0 || + number < cparam->minimum.custom_real || + number > cparam->maximum.custom_real) + return (NULL); + + snprintf(buffer, bufsize, "Custom.%g", number); + break; + + case PPD_CUSTOM_INT : + if (!*val || (integer = strtol(val, NULL, 10)) == LONG_MIN || + integer == LONG_MAX || + integer < cparam->minimum.custom_int || + integer > cparam->maximum.custom_int) + return (NULL); + + snprintf(buffer, bufsize, "Custom.%ld", integer); + break; + + case PPD_CUSTOM_POINTS : + snprintf(keyword, sizeof(keyword), "%s.Units", coption->keyword); + + if ((number = strtod(val, NULL)) == 0.0 || + (uval = cgiGetVariable(keyword)) == NULL || + (strcmp(uval, "pt") && strcmp(uval, "in") && strcmp(uval, "ft") && + strcmp(uval, "cm") && strcmp(uval, "mm") && strcmp(uval, "m"))) + return (NULL); + + number_points = get_points(number, uval); + if (number_points < cparam->minimum.custom_points || + number_points > cparam->maximum.custom_points) + return (NULL); + + snprintf(buffer, bufsize, "Custom.%g%s", number, uval); + break; + + case PPD_CUSTOM_PASSCODE : + for (uval = val; *uval; uval ++) + if (!isdigit(*uval & 255)) + return (NULL); + + case PPD_CUSTOM_PASSWORD : + case PPD_CUSTOM_STRING : + integer = (long)strlen(val); + if (integer < cparam->minimum.custom_string || + integer > cparam->maximum.custom_string) + return (NULL); + + snprintf(buffer, bufsize, "Custom.%s", val); + break; + } + } + else + { + const char *prefix = "{"; /* Prefix string */ + + + bufptr = buffer; + bufend = buffer + bufsize; + + for (cparam = ppdFirstCustomParam(coption); + cparam; + cparam = ppdNextCustomParam(coption)) + { + snprintf(keyword, sizeof(keyword), "%s.%s", coption->keyword, + cparam->name); + + if ((val = cgiGetVariable(keyword)) == NULL) + return (NULL); + + snprintf(bufptr, bufend - bufptr, "%s%s=", prefix, cparam->name); + bufptr += strlen(bufptr); + prefix = " "; + + switch (cparam->type) + { + case PPD_CUSTOM_CURVE : + case PPD_CUSTOM_INVCURVE : + case PPD_CUSTOM_REAL : + if ((number = strtod(val, NULL)) == 0.0 || + number < cparam->minimum.custom_real || + number > cparam->maximum.custom_real) + return (NULL); + + snprintf(bufptr, bufend - bufptr, "%g", number); + break; + + case PPD_CUSTOM_INT : + if (!*val || (integer = strtol(val, NULL, 10)) == LONG_MIN || + integer == LONG_MAX || + integer < cparam->minimum.custom_int || + integer > cparam->maximum.custom_int) + return (NULL); + + snprintf(bufptr, bufend - bufptr, "%ld", integer); + break; + + case PPD_CUSTOM_POINTS : + snprintf(keyword, sizeof(keyword), "%s.Units", coption->keyword); + + if ((number = strtod(val, NULL)) == 0.0 || + (uval = cgiGetVariable(keyword)) == NULL || + (strcmp(uval, "pt") && strcmp(uval, "in") && + strcmp(uval, "ft") && strcmp(uval, "cm") && + strcmp(uval, "mm") && strcmp(uval, "m"))) + return (NULL); + + number_points = get_points(number, uval); + if (number_points < cparam->minimum.custom_points || + number_points > cparam->maximum.custom_points) + return (NULL); + + snprintf(bufptr, bufend - bufptr, "%g%s", number, uval); + break; + + case PPD_CUSTOM_PASSCODE : + for (uval = val; *uval; uval ++) + if (!isdigit(*uval & 255)) + return (NULL); + + case PPD_CUSTOM_PASSWORD : + case PPD_CUSTOM_STRING : + integer = (long)strlen(val); + if (integer < cparam->minimum.custom_string || + integer > cparam->maximum.custom_string) + return (NULL); + + if ((bufptr + 2) > bufend) + return (NULL); + + bufend --; + *bufptr++ = '\"'; + + while (*val && bufptr < bufend) + { + if (*val == '\\' || *val == '\"') + { + if ((bufptr + 1) >= bufend) + return (NULL); + + *bufptr++ = '\\'; + } + + *bufptr++ = *val++; + } + + if (bufptr >= bufend) + return (NULL); + + *bufptr++ = '\"'; + *bufptr = '\0'; + bufend ++; + break; + } + + bufptr += strlen(bufptr); + } + + if (bufptr == buffer || (bufend - bufptr) < 2) + return (NULL); + + strcpy(bufptr, "}"); + } + + return (buffer); +} + + +/* + * 'get_points()' - Get a value in points. + */ + +static double /* O - Number in points */ +get_points(double number, /* I - Original number */ + const char *uval) /* I - Units */ +{ + if (!strcmp(uval, "mm")) /* Millimeters */ + return (number * 72.0 / 25.4); + else if (!strcmp(uval, "cm")) /* Centimeters */ + return (number * 72.0 / 2.54); + else if (!strcmp(uval, "in")) /* Inches */ + return (number * 72.0); + else if (!strcmp(uval, "ft")) /* Feet */ + return (number * 72.0 * 12.0); + else if (!strcmp(uval, "m")) /* Meters */ + return (number * 72.0 / 0.0254); + else /* Points */ + return (number); +} + + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/api-cgi.header b/cgi-bin/api-cgi.header new file mode 100644 index 0000000000..e3355a6b99 --- /dev/null +++ b/cgi-bin/api-cgi.header @@ -0,0 +1,34 @@ + + +

CGI API

+ +
+ + + + + + + + + + + + + + + + +
Headercups/cgi.h
Library-lcupscgi
See AlsoProgramming: Introduction to CUPS Programming
diff --git a/cgi-bin/api-cgi.shtml b/cgi-bin/api-cgi.shtml new file mode 100644 index 0000000000..cf0413a5e4 --- /dev/null +++ b/cgi-bin/api-cgi.shtml @@ -0,0 +1,17 @@ + + +

Overview

+ +

The CGI API provides Common Gateway Interface functions for CUPS.

diff --git a/cgi-bin/cgi-private.h b/cgi-bin/cgi-private.h new file mode 100644 index 0000000000..1f96902739 --- /dev/null +++ b/cgi-bin/cgi-private.h @@ -0,0 +1,36 @@ +/* + * "$Id$" + * + * Private CGI definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +/* + * Include necessary headers... + */ + +#include "cgi.h" +#include +#include +#include +#include /* TODO: Update so we don't need this */ + + +/* + * Limits... + */ + +#define CUPS_PAGE_MAX 100 /* Maximum items per page */ + + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/cgi.h b/cgi-bin/cgi.h new file mode 100644 index 0000000000..61eb7fc754 --- /dev/null +++ b/cgi-bin/cgi.h @@ -0,0 +1,119 @@ +/* + * "$Id$" + * + * CGI support library definitions for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#ifndef _CUPS_CGI_H_ +# define _CUPS_CGI_H_ + +# include +# include +# include +# include + +# ifdef WIN32 +# include +# include +# else +# include +# endif /* WIN32 */ + +# include +# include +# include "help-index.h" + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + +/* + * Types... + */ + +typedef struct cgi_file_s /**** Uploaded file data ****/ +{ + char tempfile[1024], /* Temporary file containing data */ + *name, /* Variable name */ + *filename, /* Original filename */ + *mimetype; /* MIME media type */ + size_t filesize; /* Size of uploaded file */ +} cgi_file_t; + + +/* + * Prototypes... + */ + +extern void cgiAbort(const char *title, const char *stylesheet, + const char *format, ...); +extern int cgiCheckVariables(const char *names); +extern void cgiClearVariables(void); +extern void *cgiCompileSearch(const char *query); +extern void cgiCopyTemplateFile(FILE *out, const char *tmpl); +extern void cgiCopyTemplateLang(const char *tmpl); +extern int cgiDoSearch(void *search, const char *text); +extern void cgiEndHTML(void); +extern void cgiEndMultipart(void); +extern char *cgiFormEncode(char *dst, const char *src, + size_t dstsize); +extern void cgiFreeSearch(void *search); +extern const char *cgiGetArray(const char *name, int element); +extern void cgiGetAttributes(ipp_t *request, const char *tmpl); +extern const char *cgiGetCookie(const char *name); +extern const cgi_file_t *cgiGetFile(void); +extern cups_array_t *cgiGetIPPObjects(ipp_t *response, void *search); +extern int cgiGetSize(const char *name); +extern char *cgiGetTemplateDir(void); +extern const char *cgiGetVariable(const char *name); +extern int cgiInitialize(void); +extern int cgiIsPOST(void); +extern void cgiMoveJobs(http_t *http, const char *dest, int job_id); +extern void cgiPrintCommand(http_t *http, const char *dest, + const char *command, const char *title); +extern void cgiPrintTestPage(http_t *http, const char *dest); +extern char *cgiRewriteURL(const char *uri, char *url, int urlsize, + const char *newresource); +extern void cgiSetArray(const char *name, int element, + const char *value); +extern void cgiSetCookie(const char *name, const char *value, + const char *path, const char *domain, + time_t expires, int secure); +extern ipp_attribute_t *cgiSetIPPObjectVars(ipp_attribute_t *obj, + const char *prefix, int element); +extern int cgiSetIPPVars(ipp_t *response, const char *filter_name, + const char *filter_value, + const char *prefix, int parent_el); +extern void cgiSetServerVersion(void); +extern void cgiSetSize(const char *name, int size); +extern void cgiSetVariable(const char *name, const char *value); +extern void cgiShowIPPError(const char *message); +extern void cgiShowJobs(http_t *http, const char *dest); +extern void cgiStartHTML(const char *title); +extern void cgiStartMultipart(void); +extern int cgiSupportsMultipart(void); +extern const char *cgiText(const char *message); + +# ifdef __cplusplus +} +# endif /* __cplusplus */ + +#endif /* !_CUPS_CGI_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/classes.c b/cgi-bin/classes.c new file mode 100644 index 0000000000..293b2c3a8c --- /dev/null +++ b/cgi-bin/classes.c @@ -0,0 +1,558 @@ +/* + * "$Id$" + * + * Class status CGI for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Main entry for CGI. + * do_class_op() - Do a class operation. + * show_all_classes() - Show all classes... + * show_class() - Show a single class. + */ + +/* + * Include necessary headers... + */ + +#include "cgi-private.h" + + +/* + * Local functions... + */ + +static void do_class_op(http_t *http, const char *printer, ipp_op_t op, + const char *title); +static void show_all_classes(http_t *http, const char *username); +static void show_class(http_t *http, const char *printer); + + +/* + * 'main()' - Main entry for CGI. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + const char *pclass; /* Class name */ + const char *user; /* Username */ + http_t *http; /* Connection to the server */ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + ipp_attribute_t *attr; /* IPP attribute */ + const char *op; /* Operation to perform, if any */ + static const char *def_attrs[] = /* Attributes for default printer */ + { + "printer-name", + "printer-uri-supported" + }; + + + /* + * Get any form variables... + */ + + cgiInitialize(); + + op = cgiGetVariable("OP"); + + /* + * Set the web interface section... + */ + + cgiSetVariable("SECTION", "classes"); + cgiSetVariable("REFRESH_PAGE", ""); + + /* + * See if we are displaying a printer or all classes... + */ + + if ((pclass = getenv("PATH_INFO")) != NULL) + { + pclass ++; + + if (!*pclass) + pclass = NULL; + + if (pclass) + cgiSetVariable("PRINTER_NAME", pclass); + } + + /* + * See who is logged in... + */ + + user = getenv("REMOTE_USER"); + + /* + * Connect to the HTTP server... + */ + + http = httpConnectEncrypt(cupsServer(), ippPort(), cupsEncryption()); + + /* + * Get the default printer... + */ + + if (!op || !cgiIsPOST()) + { + /* + * Get the default destination... + */ + + request = ippNewRequest(CUPS_GET_DEFAULT); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", + sizeof(def_attrs) / sizeof(def_attrs[0]), NULL, def_attrs); + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + if ((attr = ippFindAttribute(response, "printer-name", IPP_TAG_NAME)) != NULL) + cgiSetVariable("DEFAULT_NAME", attr->values[0].string.text); + + if ((attr = ippFindAttribute(response, "printer-uri-supported", IPP_TAG_URI)) != NULL) + { + char url[HTTP_MAX_URI]; /* New URL */ + + + cgiSetVariable("DEFAULT_URI", + cgiRewriteURL(attr->values[0].string.text, + url, sizeof(url), NULL)); + } + + ippDelete(response); + } + + /* + * See if we need to show a list of classes or the status of a + * single printer... + */ + + if (!pclass) + show_all_classes(http, user); + else + show_class(http, pclass); + } + else if (pclass) + { + if (!*op) + { + const char *server_port = getenv("SERVER_PORT"); + /* Port number string */ + int port = atoi(server_port ? server_port : "0"); + /* Port number */ + char uri[1024]; /* URL */ + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), + getenv("HTTPS") ? "https" : "http", NULL, + getenv("SERVER_NAME"), port, "/classes/%s", pclass); + + printf("Location: %s\n\n", uri); + } + else if (!strcmp(op, "start-class")) + do_class_op(http, pclass, IPP_RESUME_PRINTER, cgiText(_("Resume Class"))); + else if (!strcmp(op, "stop-class")) + do_class_op(http, pclass, IPP_PAUSE_PRINTER, cgiText(_("Pause Class"))); + else if (!strcmp(op, "accept-jobs")) + do_class_op(http, pclass, CUPS_ACCEPT_JOBS, cgiText(_("Accept Jobs"))); + else if (!strcmp(op, "reject-jobs")) + do_class_op(http, pclass, CUPS_REJECT_JOBS, cgiText(_("Reject Jobs"))); + else if (!strcmp(op, "purge-jobs")) + do_class_op(http, pclass, IPP_PURGE_JOBS, cgiText(_("Purge Jobs"))); + else if (!_cups_strcasecmp(op, "print-test-page")) + cgiPrintTestPage(http, pclass); + else if (!_cups_strcasecmp(op, "move-jobs")) + cgiMoveJobs(http, pclass, 0); + else + { + /* + * Unknown/bad operation... + */ + + cgiStartHTML(pclass); + cgiCopyTemplateLang("error-op.tmpl"); + cgiEndHTML(); + } + } + else + { + /* + * Unknown/bad operation... + */ + + cgiStartHTML(cgiText(_("Classes"))); + cgiCopyTemplateLang("error-op.tmpl"); + cgiEndHTML(); + } + + /* + * Close the HTTP server connection... + */ + + httpClose(http); + + /* + * Return with no errors... + */ + + return (0); +} + + +/* + * 'do_class_op()' - Do a class operation. + */ + +static void +do_class_op(http_t *http, /* I - HTTP connection */ + const char *printer, /* I - Printer name */ + ipp_op_t op, /* I - Operation to perform */ + const char *title) /* I - Title of page */ +{ + ipp_t *request; /* IPP request */ + char uri[HTTP_MAX_URI], /* Printer URI */ + resource[HTTP_MAX_URI]; /* Path for request */ + + + /* + * Build a printer request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + */ + + request = ippNewRequest(op); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/classes/%s", printer); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + /* + * Do the request and get back a response... + */ + + snprintf(resource, sizeof(resource), "/classes/%s", printer); + ippDelete(cupsDoRequest(http, request, resource)); + + if (cupsLastError() == IPP_NOT_AUTHORIZED) + { + puts("Status: 401\n"); + exit(0); + } + else if (cupsLastError() > IPP_OK_CONFLICT) + { + cgiStartHTML(title); + cgiShowIPPError(_("Unable to do maintenance command:")); + } + else + { + /* + * Redirect successful updates back to the printer page... + */ + + char url[1024], /* Printer/class URL */ + refresh[1024]; /* Refresh URL */ + + + cgiRewriteURL(uri, url, sizeof(url), NULL); + cgiFormEncode(uri, url, sizeof(uri)); + snprintf(refresh, sizeof(refresh), "5;URL=%s", uri); + cgiSetVariable("refresh_page", refresh); + + cgiStartHTML(title); + + cgiSetVariable("IS_CLASS", "YES"); + + if (op == IPP_PAUSE_PRINTER) + cgiCopyTemplateLang("printer-stop.tmpl"); + else if (op == IPP_RESUME_PRINTER) + cgiCopyTemplateLang("printer-start.tmpl"); + else if (op == CUPS_ACCEPT_JOBS) + cgiCopyTemplateLang("printer-accept.tmpl"); + else if (op == CUPS_REJECT_JOBS) + cgiCopyTemplateLang("printer-reject.tmpl"); + else if (op == IPP_PURGE_JOBS) + cgiCopyTemplateLang("printer-purge.tmpl"); + } + + cgiEndHTML(); +} + + +/* + * 'show_all_classes()' - Show all classes... + */ + +static void +show_all_classes(http_t *http, /* I - Connection to server */ + const char *user) /* I - Username */ +{ + int i; /* Looping var */ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + cups_array_t *classes; /* Array of class objects */ + ipp_attribute_t *pclass; /* Class object */ + int ascending, /* Order of classes (0 = descending) */ + first, /* First class to show */ + count; /* Number of classes */ + const char *var; /* Form variable */ + void *search; /* Search data */ + char val[1024]; /* Form variable */ + + + /* + * Show the standard header... + */ + + cgiStartHTML(cgiText(_("Classes"))); + + /* + * Build a CUPS_GET_CLASSES request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * requesting-user-name + */ + + request = ippNewRequest(CUPS_GET_CLASSES); + + if (user) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, user); + + cgiGetAttributes(request, "classes.tmpl"); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + /* + * Get a list of matching job objects. + */ + + if ((var = cgiGetVariable("QUERY")) != NULL && + !cgiGetVariable("CLEAR")) + search = cgiCompileSearch(var); + else + search = NULL; + + classes = cgiGetIPPObjects(response, search); + count = cupsArrayCount(classes); + + if (search) + cgiFreeSearch(search); + + /* + * Figure out which classes to display... + */ + + if ((var = cgiGetVariable("FIRST")) != NULL) + first = atoi(var); + else + first = 0; + + if (first >= count) + first = count - CUPS_PAGE_MAX; + + first = (first / CUPS_PAGE_MAX) * CUPS_PAGE_MAX; + + if (first < 0) + first = 0; + + sprintf(val, "%d", count); + cgiSetVariable("TOTAL", val); + + if ((var = cgiGetVariable("ORDER")) != NULL) + ascending = !_cups_strcasecmp(var, "asc"); + else + ascending = 1; + + if (ascending) + { + for (i = 0, pclass = (ipp_attribute_t *)cupsArrayIndex(classes, first); + i < CUPS_PAGE_MAX && pclass; + i ++, pclass = (ipp_attribute_t *)cupsArrayNext(classes)) + cgiSetIPPObjectVars(pclass, NULL, i); + } + else + { + for (i = 0, pclass = (ipp_attribute_t *)cupsArrayIndex(classes, count - first - 1); + i < CUPS_PAGE_MAX && pclass; + i ++, pclass = (ipp_attribute_t *)cupsArrayPrev(classes)) + cgiSetIPPObjectVars(pclass, NULL, i); + } + + /* + * Save navigation URLs... + */ + + cgiSetVariable("THISURL", "/classes/"); + + if (first > 0) + { + sprintf(val, "%d", first - CUPS_PAGE_MAX); + cgiSetVariable("PREV", val); + } + + if ((first + CUPS_PAGE_MAX) < count) + { + sprintf(val, "%d", first + CUPS_PAGE_MAX); + cgiSetVariable("NEXT", val); + } + + /* + * Then show everything... + */ + + cgiCopyTemplateLang("search.tmpl"); + + cgiCopyTemplateLang("classes-header.tmpl"); + + if (count > CUPS_PAGE_MAX) + cgiCopyTemplateLang("pager.tmpl"); + + cgiCopyTemplateLang("classes.tmpl"); + + if (count > CUPS_PAGE_MAX) + cgiCopyTemplateLang("pager.tmpl"); + + /* + * Delete the response... + */ + + cupsArrayDelete(classes); + ippDelete(response); + } + else + { + /* + * Show the error... + */ + + cgiShowIPPError(_("Unable to get class list:")); + } + + cgiEndHTML(); +} + + +/* + * 'show_class()' - Show a single class. + */ + +static void +show_class(http_t *http, /* I - Connection to server */ + const char *pclass) /* I - Name of class */ +{ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + ipp_attribute_t *attr; /* IPP attribute */ + char uri[HTTP_MAX_URI]; /* Printer URI */ + char refresh[1024]; /* Refresh URL */ + + + /* + * Build an IPP_GET_PRINTER_ATTRIBUTES request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + */ + + request = ippNewRequest(IPP_GET_PRINTER_ATTRIBUTES); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/classes/%s", pclass); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", NULL, + uri); + + cgiGetAttributes(request, "class.tmpl"); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + /* + * Got the result; set the CGI variables and check the status of a + * single-queue request... + */ + + cgiSetIPPVars(response, NULL, NULL, NULL, 0); + + if (pclass && (attr = ippFindAttribute(response, "printer-state", + IPP_TAG_ENUM)) != NULL && + attr->values[0].integer == IPP_PRINTER_PROCESSING) + { + /* + * Class is processing - automatically refresh the page until we + * are done printing... + */ + + cgiFormEncode(uri, pclass, sizeof(uri)); + snprintf(refresh, sizeof(refresh), "10;URL=/classes/%s", uri); + cgiSetVariable("refresh_page", refresh); + } + + /* + * Delete the response... + */ + + ippDelete(response); + + /* + * Show the standard header... + */ + + cgiStartHTML(pclass); + + /* + * Show the class status... + */ + + cgiCopyTemplateLang("class.tmpl"); + + /* + * Show jobs for the specified class... + */ + + cgiCopyTemplateLang("class-jobs-header.tmpl"); + cgiShowJobs(http, pclass); + } + else + { + /* + * Show the IPP error... + */ + + cgiStartHTML(pclass); + cgiShowIPPError(_("Unable to get class status:")); + } + + cgiEndHTML(); +} + + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/help-index.c b/cgi-bin/help-index.c new file mode 100644 index 0000000000..53d482021c --- /dev/null +++ b/cgi-bin/help-index.c @@ -0,0 +1,1328 @@ +/* + * "$Id$" + * + * Online help index routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * helpDeleteIndex() - Delete an index, freeing all memory used. + * helpFindNode() - Find a node in an index. + * helpLoadIndex() - Load a help index from disk. + * helpSaveIndex() - Save a help index to disk. + * helpSearchIndex() - Search an index. + * help_add_word() - Add a word to a node. + * help_compile_search() - Convert a search string into a regular expression. + * help_delete_node() - Free all memory used by a node. + * help_delete_word() - Free all memory used by a word. + * help_load_directory() - Load a directory of files into an index. + * help_load_file() - Load a HTML files into an index. + * help_new_node() - Create a new node and add it to an index. + * help_sort_nodes_by_name() - Sort nodes by section, filename, and anchor. + * help_sort_nodes_by_score() - Sort nodes by score and text. + * help_sort_words() - Sort words alphabetically. + */ + +/* + * Include necessary headers... + */ + +#include "cgi-private.h" +#include + + +/* + * List of common English words that should not be indexed... + */ + +static char help_common_words[][6] = + { + "about", + "all", + "an", + "and", + "are", + "as", + "at", + "be", + "been", + "but", + "by", + "call", + "can", + "come", + "could", + "day", + "did", + "do", + "down", + "each", + "find", + "first", + "for", + "from", + "go", + "had", + "has", + "have", + "he", + "her", + "him", + "his", + "hot", + "how", + "if", + "in", + "is", + "it", + "know", + "like", + "long", + "look", + "make", + "many", + "may", + "more", + "most", + "my", + "no", + "now", + "of", + "on", + "one", + "or", + "other", + "out", + "over", + "said", + "see", + "she", + "side", + "so", + "some", + "sound", + "than", + "that", + "the", + "their", + "them", + "then", + "there", + "these", + "they", + "thing", + "this", + "time", + "to", + "two", + "up", + "use", + "was", + "water", + "way", + "we", + "were", + "what", + "when", + "which", + "who", + "will", + "with", + "word", + "would", + "write", + "you", + "your" + }; + + +/* + * Local functions... + */ + +static help_word_t *help_add_word(help_node_t *n, const char *text); +static void help_delete_node(help_node_t *n); +static void help_delete_word(help_word_t *w); +static int help_load_directory(help_index_t *hi, + const char *directory, + const char *relative); +static int help_load_file(help_index_t *hi, + const char *filename, + const char *relative, + time_t mtime); +static help_node_t *help_new_node(const char *filename, const char *anchor, + const char *section, const char *text, + time_t mtime, off_t offset, + size_t length) + __attribute__((nonnull(1,3,4))); +static int help_sort_by_name(help_node_t *p1, help_node_t *p2); +static int help_sort_by_score(help_node_t *p1, help_node_t *p2); +static int help_sort_words(help_word_t *w1, help_word_t *w2); + + +/* + * 'helpDeleteIndex()' - Delete an index, freeing all memory used. + */ + +void +helpDeleteIndex(help_index_t *hi) /* I - Help index */ +{ + help_node_t *node; /* Current node */ + + + DEBUG_printf(("helpDeleteIndex(hi=%p)", hi)); + + if (!hi) + return; + + for (node = (help_node_t *)cupsArrayFirst(hi->nodes); + node; + node = (help_node_t *)cupsArrayNext(hi->nodes)) + { + if (!hi->search) + help_delete_node(node); + } + + cupsArrayDelete(hi->nodes); + cupsArrayDelete(hi->sorted); + + free(hi); +} + + +/* + * 'helpFindNode()' - Find a node in an index. + */ + +help_node_t * /* O - Node pointer or NULL */ +helpFindNode(help_index_t *hi, /* I - Index */ + const char *filename, /* I - Filename */ + const char *anchor) /* I - Anchor */ +{ + help_node_t key; /* Search key */ + + + DEBUG_printf(("helpFindNode(hi=%p, filename=\"%s\", anchor=\"%s\")", + hi, filename, anchor)); + + /* + * Range check input... + */ + + if (!hi || !filename) + return (NULL); + + /* + * Initialize the search key... + */ + + key.filename = (char *)filename; + key.anchor = (char *)anchor; + + /* + * Return any match... + */ + + return ((help_node_t *)cupsArrayFind(hi->nodes, &key)); +} + + +/* + * 'helpLoadIndex()' - Load a help index from disk. + */ + +help_index_t * /* O - Index pointer or NULL */ +helpLoadIndex(const char *hifile, /* I - Index filename */ + const char *directory) /* I - Directory that is indexed */ +{ + help_index_t *hi; /* Help index */ + cups_file_t *fp; /* Current file */ + char line[2048], /* Line from file */ + *ptr, /* Pointer into line */ + *filename, /* Filename in line */ + *anchor, /* Anchor in line */ + *sectptr, /* Section pointer in line */ + section[1024], /* Section name */ + *text; /* Text in line */ + time_t mtime; /* Modification time */ + off_t offset; /* Offset into file */ + size_t length; /* Length in bytes */ + int update; /* Update? */ + help_node_t *node; /* Current node */ + help_word_t *word; /* Current word */ + + + DEBUG_printf(("helpLoadIndex(hifile=\"%s\", directory=\"%s\")", + hifile, directory)); + + /* + * Create a new, empty index. + */ + + if ((hi = (help_index_t *)calloc(1, sizeof(help_index_t))) == NULL) + return (NULL); + + hi->nodes = cupsArrayNew((cups_array_func_t)help_sort_by_name, NULL); + hi->sorted = cupsArrayNew((cups_array_func_t)help_sort_by_score, NULL); + + if (!hi->nodes || !hi->sorted) + { + cupsArrayDelete(hi->nodes); + cupsArrayDelete(hi->sorted); + free(hi); + return (NULL); + } + + /* + * Try loading the existing index file... + */ + + if ((fp = cupsFileOpen(hifile, "r")) != NULL) + { + /* + * Lock the file and then read the first line... + */ + + cupsFileLock(fp, 1); + + if (cupsFileGets(fp, line, sizeof(line)) && !strcmp(line, "HELPV2")) + { + /* + * Got a valid header line, now read the data lines... + */ + + node = NULL; + + while (cupsFileGets(fp, line, sizeof(line))) + { + /* + * Each line looks like one of the following: + * + * filename mtime offset length "section" "text" + * filename#anchor offset length "text" + * SP count word + */ + + if (line[0] == ' ') + { + /* + * Read a word in the current node... + */ + + if (!node || (ptr = strrchr(line, ' ')) == NULL) + continue; + + if ((word = help_add_word(node, ptr + 1)) != NULL) + word->count = atoi(line + 1); + } + else + { + /* + * Add a node... + */ + + filename = line; + + if ((ptr = strchr(line, ' ')) == NULL) + break; + + while (isspace(*ptr & 255)) + *ptr++ = '\0'; + + if ((anchor = strrchr(filename, '#')) != NULL) + { + *anchor++ = '\0'; + mtime = 0; + } + else + mtime = strtol(ptr, &ptr, 10); + + offset = strtoll(ptr, &ptr, 10); + length = strtoll(ptr, &ptr, 10); + + while (isspace(*ptr & 255)) + ptr ++; + + if (!anchor) + { + /* + * Get section... + */ + + if (*ptr != '\"') + break; + + ptr ++; + sectptr = ptr; + + while (*ptr && *ptr != '\"') + ptr ++; + + if (*ptr != '\"') + break; + + *ptr++ = '\0'; + + strlcpy(section, sectptr, sizeof(section)); + + while (isspace(*ptr & 255)) + ptr ++; + } + + if (*ptr != '\"') + break; + + ptr ++; + text = ptr; + + while (*ptr && *ptr != '\"') + ptr ++; + + if (*ptr != '\"') + break; + + *ptr++ = '\0'; + + if ((node = help_new_node(filename, anchor, section, text, + mtime, offset, length)) == NULL) + break; + + node->score = -1; + + cupsArrayAdd(hi->nodes, node); + } + } + } + + cupsFileClose(fp); + } + + /* + * Scan for new/updated files... + */ + + update = help_load_directory(hi, directory, NULL); + + /* + * Remove any files that are no longer installed... + */ + + for (node = (help_node_t *)cupsArrayFirst(hi->nodes); + node; + node = (help_node_t *)cupsArrayNext(hi->nodes)) + if (node->score < 0) + { + /* + * Delete this node... + */ + + cupsArrayRemove(hi->nodes, node); + help_delete_node(node); + } + + /* + * Add nodes to the sorted array... + */ + + for (node = (help_node_t *)cupsArrayFirst(hi->nodes); + node; + node = (help_node_t *)cupsArrayNext(hi->nodes)) + cupsArrayAdd(hi->sorted, node); + + /* + * Save the index if we updated it... + */ + + if (update) + helpSaveIndex(hi, hifile); + + /* + * Return the index... + */ + + return (hi); +} + + +/* + * 'helpSaveIndex()' - Save a help index to disk. + */ + +int /* O - 0 on success, -1 on error */ +helpSaveIndex(help_index_t *hi, /* I - Index */ + const char *hifile) /* I - Index filename */ +{ + cups_file_t *fp; /* Index file */ + help_node_t *node; /* Current node */ + help_word_t *word; /* Current word */ + + + DEBUG_printf(("helpSaveIndex(hi=%p, hifile=\"%s\")", hi, hifile)); + + /* + * Try creating a new index file... + */ + + if ((fp = cupsFileOpen(hifile, "w9")) == NULL) + return (-1); + + /* + * Lock the file while we write it... + */ + + cupsFileLock(fp, 1); + + cupsFilePuts(fp, "HELPV2\n"); + + for (node = (help_node_t *)cupsArrayFirst(hi->nodes); + node; + node = (help_node_t *)cupsArrayNext(hi->nodes)) + { + /* + * Write the current node with/without the anchor... + */ + + if (node->anchor) + { + if (cupsFilePrintf(fp, "%s#%s " CUPS_LLFMT " " CUPS_LLFMT " \"%s\"\n", + node->filename, node->anchor, + CUPS_LLCAST node->offset, CUPS_LLCAST node->length, + node->text) < 0) + break; + } + else + { + if (cupsFilePrintf(fp, "%s %d " CUPS_LLFMT " " CUPS_LLFMT " \"%s\" \"%s\"\n", + node->filename, (int)node->mtime, + CUPS_LLCAST node->offset, CUPS_LLCAST node->length, + node->section ? node->section : "", node->text) < 0) + break; + } + + /* + * Then write the words associated with the node... + */ + + for (word = (help_word_t *)cupsArrayFirst(node->words); + word; + word = (help_word_t *)cupsArrayNext(node->words)) + if (cupsFilePrintf(fp, " %d %s\n", word->count, word->text) < 0) + break; + } + + cupsFileFlush(fp); + + if (cupsFileClose(fp) < 0) + return (-1); + else if (node) + return (-1); + else + return (0); +} + + +/* + * 'helpSearchIndex()' - Search an index. + */ + +help_index_t * /* O - Search index */ +helpSearchIndex(help_index_t *hi, /* I - Index */ + const char *query, /* I - Query string */ + const char *section, /* I - Limit search to this section */ + const char *filename) /* I - Limit search to this file */ +{ + help_index_t *search; /* Search index */ + help_node_t *node; /* Current node */ + help_word_t *word; /* Current word */ + void *sc; /* Search context */ + int matches; /* Number of matches */ + + + DEBUG_printf(("helpSearchIndex(hi=%p, query=\"%s\", filename=\"%s\")", + hi, query, filename)); + + /* + * Range check... + */ + + if (!hi || !query) + return (NULL); + + /* + * Reset the scores of all nodes to 0... + */ + + for (node = (help_node_t *)cupsArrayFirst(hi->nodes); + node; + node = (help_node_t *)cupsArrayNext(hi->nodes)) + node->score = 0; + + /* + * Find the first node to search in... + */ + + if (filename) + { + node = helpFindNode(hi, filename, NULL); + if (!node) + return (NULL); + } + else + node = (help_node_t *)cupsArrayFirst(hi->nodes); + + /* + * Convert the query into a regular expression... + */ + + sc = cgiCompileSearch(query); + if (!sc) + return (NULL); + + /* + * Allocate a search index... + */ + + search = calloc(1, sizeof(help_index_t)); + if (!search) + { + cgiFreeSearch(sc); + return (NULL); + } + + search->nodes = cupsArrayNew((cups_array_func_t)help_sort_by_name, NULL); + search->sorted = cupsArrayNew((cups_array_func_t)help_sort_by_score, NULL); + + if (!search->nodes || !search->sorted) + { + cupsArrayDelete(search->nodes); + cupsArrayDelete(search->sorted); + free(search); + cgiFreeSearch(sc); + return (NULL); + } + + search->search = 1; + + /* + * Check each node in the index, adding matching nodes to the + * search index... + */ + + for (; node; node = (help_node_t *)cupsArrayNext(hi->nodes)) + if (section && strcmp(node->section, section)) + continue; + else if (filename && strcmp(node->filename, filename)) + continue; + else + { + matches = cgiDoSearch(sc, node->text); + + for (word = (help_word_t *)cupsArrayFirst(node->words); + word; + word = (help_word_t *)cupsArrayNext(node->words)) + if (cgiDoSearch(sc, word->text) > 0) + matches += word->count; + + if (matches > 0) + { + /* + * Found a match, add the node to the search index... + */ + + node->score = matches; + + cupsArrayAdd(search->nodes, node); + cupsArrayAdd(search->sorted, node); + } + } + + /* + * Free the search context... + */ + + cgiFreeSearch(sc); + + /* + * Return the results... + */ + + return (search); +} + + +/* + * 'help_add_word()' - Add a word to a node. + */ + +static help_word_t * /* O - New word */ +help_add_word(help_node_t *n, /* I - Node */ + const char *text) /* I - Word text */ +{ + help_word_t *w, /* New word */ + key; /* Search key */ + + + DEBUG_printf(("2help_add_word(n=%p, text=\"%s\")", n, text)); + + /* + * Create the words array as needed... + */ + + if (!n->words) + n->words = cupsArrayNew((cups_array_func_t)help_sort_words, NULL); + + /* + * See if the word is already added... + */ + + key.text = (char *)text; + + if ((w = (help_word_t *)cupsArrayFind(n->words, &key)) == NULL) + { + /* + * Create a new word... + */ + + if ((w = calloc(1, sizeof(help_word_t))) == NULL) + return (NULL); + + if ((w->text = strdup(text)) == NULL) + { + free(w); + return (NULL); + } + + cupsArrayAdd(n->words, w); + } + + /* + * Bump the counter for this word and return it... + */ + + w->count ++; + + return (w); +} + + +/* + * 'help_delete_node()' - Free all memory used by a node. + */ + +static void +help_delete_node(help_node_t *n) /* I - Node */ +{ + help_word_t *w; /* Current word */ + + + DEBUG_printf(("2help_delete_node(n=%p)", n)); + + if (!n) + return; + + if (n->filename) + free(n->filename); + + if (n->anchor) + free(n->anchor); + + if (n->section) + free(n->section); + + if (n->text) + free(n->text); + + for (w = (help_word_t *)cupsArrayFirst(n->words); + w; + w = (help_word_t *)cupsArrayNext(n->words)) + help_delete_word(w); + + cupsArrayDelete(n->words); + + free(n); +} + + +/* + * 'help_delete_word()' - Free all memory used by a word. + */ + +static void +help_delete_word(help_word_t *w) /* I - Word */ +{ + DEBUG_printf(("2help_delete_word(w=%p)", w)); + + if (!w) + return; + + if (w->text) + free(w->text); + + free(w); +} + + +/* + * 'help_load_directory()' - Load a directory of files into an index. + */ + +static int /* O - 0 = success, -1 = error, 1 = updated */ +help_load_directory( + help_index_t *hi, /* I - Index */ + const char *directory, /* I - Directory */ + const char *relative) /* I - Relative path */ +{ + cups_dir_t *dir; /* Directory file */ + cups_dentry_t *dent; /* Directory entry */ + char *ext, /* Pointer to extension */ + filename[1024], /* Full filename */ + relname[1024]; /* Relative filename */ + int update; /* Updated? */ + help_node_t *node; /* Current node */ + + + DEBUG_printf(("2help_load_directory(hi=%p, directory=\"%s\", relative=\"%s\")", + hi, directory, relative)); + + /* + * Open the directory and scan it... + */ + + if ((dir = cupsDirOpen(directory)) == NULL) + return (0); + + update = 0; + + while ((dent = cupsDirRead(dir)) != NULL) + { + /* + * Skip "." files... + */ + + if (dent->filename[0] == '.') + continue; + + /* + * Get absolute and relative filenames... + */ + + snprintf(filename, sizeof(filename), "%s/%s", directory, dent->filename); + if (relative) + snprintf(relname, sizeof(relname), "%s/%s", relative, dent->filename); + else + strlcpy(relname, dent->filename, sizeof(relname)); + + /* + * Check if we have a HTML file... + */ + + if ((ext = strstr(dent->filename, ".html")) != NULL && + (!ext[5] || !strcmp(ext + 5, ".gz"))) + { + /* + * HTML file, see if we have already indexed the file... + */ + + if ((node = helpFindNode(hi, relname, NULL)) != NULL) + { + /* + * File already indexed - check dates to confirm that the + * index is up-to-date... + */ + + if (node->mtime == dent->fileinfo.st_mtime) + { + /* + * Same modification time, so mark all of the nodes + * for this file as up-to-date... + */ + + for (; node; node = (help_node_t *)cupsArrayNext(hi->nodes)) + if (!strcmp(node->filename, relname)) + node->score = 0; + else + break; + + continue; + } + } + + update = 1; + + help_load_file(hi, filename, relname, dent->fileinfo.st_mtime); + } + else if (S_ISDIR(dent->fileinfo.st_mode)) + { + /* + * Process sub-directory... + */ + + if (help_load_directory(hi, filename, relname) == 1) + update = 1; + } + } + + cupsDirClose(dir); + + return (update); +} + + +/* + * 'help_load_file()' - Load a HTML files into an index. + */ + +static int /* O - 0 = success, -1 = error */ +help_load_file( + help_index_t *hi, /* I - Index */ + const char *filename, /* I - Filename */ + const char *relative, /* I - Relative path */ + time_t mtime) /* I - Modification time */ +{ + cups_file_t *fp; /* HTML file */ + help_node_t *node; /* Current node */ + char line[1024], /* Line from file */ + temp[1024], /* Temporary word */ + section[1024], /* Section */ + *ptr, /* Pointer into line */ + *anchor, /* Anchor name */ + *text; /* Text for anchor */ + off_t offset; /* File offset */ + char quote; /* Quote character */ + help_word_t *word; /* Current word */ + int wordlen; /* Length of word */ + + + DEBUG_printf(("2help_load_file(hi=%p, filename=\"%s\", relative=\"%s\", " + "mtime=%ld)", hi, filename, relative, mtime)); + + if ((fp = cupsFileOpen(filename, "r")) == NULL) + return (-1); + + node = NULL; + offset = 0; + + strcpy(section, "Other"); + + while (cupsFileGets(fp, line, sizeof(line))) + { + /* + * Look for "", "<A NAME", or "<!-- SECTION:" prefix... + */ + + if (!_cups_strncasecmp(line, "<!-- SECTION:", 13)) + { + /* + * Got section line, copy it! + */ + + for (ptr = line + 13; isspace(*ptr & 255); ptr ++); + + strlcpy(section, ptr, sizeof(section)); + if ((ptr = strstr(section, "-->")) != NULL) + { + /* + * Strip comment stuff from end of line... + */ + + for (*ptr-- = '\0'; ptr > line && isspace(*ptr & 255); *ptr-- = '\0'); + + if (isspace(*ptr & 255)) + *ptr = '\0'; + } + continue; + } + + for (ptr = line; (ptr = strchr(ptr, '<')) != NULL;) + { + ptr ++; + + if (!_cups_strncasecmp(ptr, "TITLE>", 6)) + { + /* + * Found the title... + */ + + anchor = NULL; + ptr += 6; + } + else if (!_cups_strncasecmp(ptr, "A NAME=", 7)) + { + /* + * Found an anchor... + */ + + ptr += 7; + + if (*ptr == '\"' || *ptr == '\'') + { + /* + * Get quoted anchor... + */ + + quote = *ptr; + anchor = ptr + 1; + if ((ptr = strchr(anchor, quote)) != NULL) + *ptr++ = '\0'; + else + break; + } + else + { + /* + * Get unquoted anchor... + */ + + anchor = ptr + 1; + + for (ptr = anchor; *ptr && *ptr != '>' && !isspace(*ptr & 255); ptr ++); + + if (*ptr) + *ptr++ = '\0'; + else + break; + } + + /* + * Got the anchor, now lets find the end... + */ + + while (*ptr && *ptr != '>') + ptr ++; + + if (*ptr != '>') + break; + + ptr ++; + } + else + continue; + + /* + * Now collect text for the link... + */ + + text = ptr; + while ((ptr = strchr(text, '<')) == NULL) + { + ptr = text + strlen(text); + if (ptr >= (line + sizeof(line) - 2)) + break; + + *ptr++ = ' '; + + if (!cupsFileGets(fp, ptr, sizeof(line) - (ptr - line) - 1)) + break; + } + + *ptr = '\0'; + + if (node) + node->length = offset - node->offset; + + if (!*text) + { + node = NULL; + break; + } + + if ((node = helpFindNode(hi, relative, anchor)) != NULL) + { + /* + * Node already in the index, so replace the text and other + * data... + */ + + cupsArrayRemove(hi->nodes, node); + + if (node->section) + free(node->section); + + if (node->text) + free(node->text); + + if (node->words) + { + for (word = (help_word_t *)cupsArrayFirst(node->words); + word; + word = (help_word_t *)cupsArrayNext(node->words)) + help_delete_word(word); + + cupsArrayDelete(node->words); + node->words = NULL; + } + + node->section = section[0] ? strdup(section) : NULL; + node->text = strdup(text); + node->mtime = mtime; + node->offset = offset; + node->score = 0; + } + else + { + /* + * New node... + */ + + node = help_new_node(relative, anchor, section, text, mtime, offset, 0); + } + + /* + * Go through the text value and replace tabs and newlines with + * whitespace and eliminate extra whitespace... + */ + + for (ptr = node->text, text = node->text; *ptr;) + if (isspace(*ptr & 255)) + { + while (isspace(*ptr & 255)) + ptr ++; + + *text++ = ' '; + } + else if (text != ptr) + *text++ = *ptr++; + else + { + text ++; + ptr ++; + } + + *text = '\0'; + + /* + * (Re)add the node to the array... + */ + + cupsArrayAdd(hi->nodes, node); + + if (!anchor) + node = NULL; + break; + } + + if (node) + { + /* + * Scan this line for words... + */ + + for (ptr = line; *ptr; ptr ++) + { + /* + * Skip HTML stuff... + */ + + if (*ptr == '<') + { + if (!strncmp(ptr, "<!--", 4)) + { + /* + * Skip HTML comment... + */ + + if ((text = strstr(ptr + 4, "-->")) == NULL) + ptr += strlen(ptr) - 1; + else + ptr = text + 2; + } + else + { + /* + * Skip HTML element... + */ + + for (ptr ++; *ptr && *ptr != '>'; ptr ++) + { + if (*ptr == '\"' || *ptr == '\'') + { + for (quote = *ptr++; *ptr && *ptr != quote; ptr ++); + + if (!*ptr) + ptr --; + } + } + + if (!*ptr) + ptr --; + } + + continue; + } + else if (*ptr == '&') + { + /* + * Skip HTML entity... + */ + + for (ptr ++; *ptr && *ptr != ';'; ptr ++); + + if (!*ptr) + ptr --; + + continue; + } + else if (!isalnum(*ptr & 255)) + continue; + + /* + * Found the start of a word, search until we find the end... + */ + + for (text = ptr, ptr ++; *ptr && isalnum(*ptr & 255); ptr ++); + + wordlen = ptr - text; + + memcpy(temp, text, wordlen); + temp[wordlen] = '\0'; + + ptr --; + + if (wordlen > 1 && !bsearch(temp, help_common_words, + (sizeof(help_common_words) / + sizeof(help_common_words[0])), + sizeof(help_common_words[0]), + (int (*)(const void *, const void *)) + _cups_strcasecmp)) + help_add_word(node, temp); + } + } + + /* + * Get the offset of the next line... + */ + + offset = cupsFileTell(fp); + } + + cupsFileClose(fp); + + if (node) + node->length = offset - node->offset; + + return (0); +} + + +/* + * 'help_new_node()' - Create a new node and add it to an index. + */ + +static help_node_t * /* O - Node pointer or NULL on error */ +help_new_node(const char *filename, /* I - Filename */ + const char *anchor, /* I - Anchor */ + const char *section, /* I - Section */ + const char *text, /* I - Text */ + time_t mtime, /* I - Modification time */ + off_t offset, /* I - Offset in file */ + size_t length) /* I - Length in bytes */ +{ + help_node_t *n; /* Node */ + + + DEBUG_printf(("2help_new_node(filename=\"%s\", anchor=\"%s\", text=\"%s\", " + "mtime=%ld, offset=%ld, length=%ld)", filename, anchor, text, + (long)mtime, (long)offset, (long)length)); + + n = (help_node_t *)calloc(1, sizeof(help_node_t)); + if (!n) + return (NULL); + + n->filename = strdup(filename); + n->anchor = anchor ? strdup(anchor) : NULL; + n->section = (section && *section) ? strdup(section) : NULL; + n->text = strdup(text); + n->mtime = mtime; + n->offset = offset; + n->length = length; + + return (n); +} + + +/* + * 'help_sort_nodes_by_name()' - Sort nodes by section, filename, and anchor. + */ + +static int /* O - Difference */ +help_sort_by_name(help_node_t *n1, /* I - First node */ + help_node_t *n2) /* I - Second node */ +{ + int diff; /* Difference */ + + + DEBUG_printf(("2help_sort_by_name(n1=%p(%s#%s), n2=%p(%s#%s)", + n1, n1->filename, n1->anchor, + n2, n2->filename, n2->anchor)); + + if ((diff = strcmp(n1->filename, n2->filename)) != 0) + return (diff); + + if (!n1->anchor && !n2->anchor) + return (0); + else if (!n1->anchor) + return (-1); + else if (!n2->anchor) + return (1); + else + return (strcmp(n1->anchor, n2->anchor)); +} + + +/* + * 'help_sort_nodes_by_score()' - Sort nodes by score and text. + */ + +static int /* O - Difference */ +help_sort_by_score(help_node_t *n1, /* I - First node */ + help_node_t *n2) /* I - Second node */ +{ + int diff; /* Difference */ + + + DEBUG_printf(("2help_sort_by_score(n1=%p(%d \"%s\" \"%s\"), " + "n2=%p(%d \"%s\" \"%s\")", + n1, n1->score, n1->section, n1->text, + n2, n2->score, n2->section, n2->text)); + + if (n1->score != n2->score) + return (n2->score - n1->score); + + if (n1->section && !n2->section) + return (1); + else if (!n1->section && n2->section) + return (-1); + else if (n1->section && n2->section && + (diff = strcmp(n1->section, n2->section)) != 0) + return (diff); + + return (_cups_strcasecmp(n1->text, n2->text)); +} + + +/* + * 'help_sort_words()' - Sort words alphabetically. + */ + +static int /* O - Difference */ +help_sort_words(help_word_t *w1, /* I - Second word */ + help_word_t *w2) /* I - Second word */ +{ + DEBUG_printf(("2help_sort_words(w1=%p(\"%s\"), w2=%p(\"%s\"))", + w1, w1->text, w2, w2->text)); + + return (_cups_strcasecmp(w1->text, w2->text)); +} + + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/help-index.h b/cgi-bin/help-index.h new file mode 100644 index 0000000000..41d8d93db5 --- /dev/null +++ b/cgi-bin/help-index.h @@ -0,0 +1,87 @@ +/* + * "$Id$" + * + * Online help index definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#ifndef _CUPS_HELP_INDEX_H_ +# define _CUPS_HELP_INDEX_H_ + +/* + * Include necessary headers... + */ + +# include <cups/array.h> + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + +/* + * Data structures... + */ + +typedef struct help_word_s /**** Help word structure... ****/ +{ + int count; /* Number of occurrences */ + char *text; /* Word text */ +} help_word_t; + +typedef struct help_node_s /**** Help node structure... ****/ +{ + char *filename; /* Filename, relative to help dir */ + char *section; /* Section name (NULL if none) */ + char *anchor; /* Anchor name (NULL if none) */ + char *text; /* Text in anchor */ + cups_array_t *words; /* Words after this node */ + time_t mtime; /* Last modification time */ + off_t offset; /* Offset in file */ + size_t length; /* Length in bytes */ + int score; /* Search score */ +} help_node_t; + +typedef struct help_index_s /**** Help index structure ****/ +{ + int search; /* 1 = search index, 0 = normal */ + cups_array_t *nodes; /* Nodes sorted by filename */ + cups_array_t *sorted; /* Nodes sorted by score + text */ +} help_index_t; + + +/* + * Functions... + */ + +extern void helpDeleteIndex(help_index_t *hi); +extern help_node_t *helpFindNode(help_index_t *hi, const char *filename, + const char *anchor); +extern help_index_t *helpLoadIndex(const char *hifile, const char *directory); +extern int helpSaveIndex(help_index_t *hi, const char *hifile); +extern help_index_t *helpSearchIndex(help_index_t *hi, const char *query, + const char *section, + const char *filename); + + +# ifdef __cplusplus +} +# endif /* __cplusplus */ + +#endif /* !_CUPS_HELP_INDEX_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/help.c b/cgi-bin/help.c new file mode 100644 index 0000000000..32b6db6939 --- /dev/null +++ b/cgi-bin/help.c @@ -0,0 +1,392 @@ +/* + * "$Id$" + * + * Online help CGI for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Main entry for CGI. + */ + +/* + * Include necessary headers... + */ + +#include "cgi-private.h" + + +/* + * 'main()' - Main entry for CGI. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + help_index_t *hi, /* Help index */ + *si; /* Search index */ + help_node_t *n; /* Current help node */ + int i; /* Looping var */ + const char *query; /* Search query */ + const char *cache_dir; /* CUPS_CACHEDIR environment variable */ + const char *docroot; /* CUPS_DOCROOT environment variable */ + const char *helpfile; /* Current help file */ + const char *topic; /* Current topic */ + char topic_data[1024]; /* Topic form data */ + const char *section; /* Current section */ + char filename[1024], /* Filename */ + directory[1024]; /* Directory */ + cups_file_t *fp; /* Help file */ + char line[1024]; /* Line from file */ + int printable; /* Show printable version? */ + + + /* + * Get any form variables... + */ + + cgiInitialize(); + + printable = cgiGetVariable("PRINTABLE") != NULL; + + /* + * Set the web interface section... + */ + + cgiSetVariable("SECTION", "help"); + cgiSetVariable("REFRESH_PAGE", ""); + + /* + * Load the help index... + */ + + if ((cache_dir = getenv("CUPS_CACHEDIR")) == NULL) + cache_dir = CUPS_CACHEDIR; + + snprintf(filename, sizeof(filename), "%s/help.index", cache_dir); + + if ((docroot = getenv("CUPS_DOCROOT")) == NULL) + docroot = CUPS_DOCROOT; + + snprintf(directory, sizeof(directory), "%s/help", docroot); + + fprintf(stderr, "DEBUG: helpLoadIndex(filename=\"%s\", directory=\"%s\")\n", + filename, directory); + + hi = helpLoadIndex(filename, directory); + if (!hi) + { + perror(filename); + + cgiStartHTML(cgiText(_("Online Help"))); + cgiSetVariable("ERROR", "Unable to load help index!"); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + + return (1); + } + + fprintf(stderr, "DEBUG: %d nodes in help index...\n", + cupsArrayCount(hi->nodes)); + + /* + * See if we are viewing a file... + */ + + for (i = 0; i < argc; i ++) + fprintf(stderr, "DEBUG: argv[%d]=\"%s\"\n", i, argv[i]); + + if ((helpfile = getenv("PATH_INFO")) != NULL) + { + helpfile ++; + + if (!*helpfile) + helpfile = NULL; + } + + if (helpfile) + { + /* + * Verify that the help file exists and is part of the index... + */ + + snprintf(filename, sizeof(filename), "%s/help/%s", docroot, helpfile); + + fprintf(stderr, "DEBUG: helpfile=\"%s\", filename=\"%s\"\n", + helpfile, filename); + + if (access(filename, R_OK)) + { + perror(filename); + + cgiStartHTML(cgiText(_("Online Help"))); + cgiSetVariable("ERROR", "Unable to access help file!"); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + + return (1); + } + + if ((n = helpFindNode(hi, helpfile, NULL)) == NULL) + { + cgiStartHTML(cgiText(_("Online Help"))); + cgiSetVariable("ERROR", "Help file not in index!"); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + + return (1); + } + + /* + * Set the page title and save the help file... + */ + + cgiSetVariable("HELPFILE", helpfile); + cgiSetVariable("HELPTITLE", n->text); + cgiSetVariable("TOPIC", n->section); + + /* + * Send a standard page header... + */ + + if (printable) + puts("Content-Type: text/html;charset=utf-8\n"); + else + cgiStartHTML(n->text); + } + else + { + /* + * Send a standard page header... + */ + + cgiStartHTML(cgiText(_("Online Help"))); + } + + /* + * Do a search as needed... + */ + + if (cgiGetVariable("CLEAR")) + cgiSetVariable("QUERY", ""); + + query = cgiGetVariable("QUERY"); + topic = cgiGetVariable("TOPIC"); + si = helpSearchIndex(hi, query, topic, helpfile); + + cgiClearVariables(); + if (query) + cgiSetVariable("QUERY", query); + if (topic) + cgiSetVariable("TOPIC", topic); + + fprintf(stderr, "DEBUG: query=\"%s\", topic=\"%s\"\n", + query ? query : "(null)", topic ? topic : "(null)"); + + if (si) + { + help_node_t *nn; /* Parent node */ + + + fprintf(stderr, + "DEBUG: si=%p, si->sorted=%p, cupsArrayCount(si->sorted)=%d\n", si, + si->sorted, cupsArrayCount(si->sorted)); + + for (i = 0, n = (help_node_t *)cupsArrayFirst(si->sorted); + n; + i ++, n = (help_node_t *)cupsArrayNext(si->sorted)) + { + if (helpfile && n->anchor) + snprintf(line, sizeof(line), "#%s", n->anchor); + else if (n->anchor) + snprintf(line, sizeof(line), "/help/%s?QUERY=%s#%s", n->filename, + query ? query : "", n->anchor); + else + snprintf(line, sizeof(line), "/help/%s?QUERY=%s", n->filename, + query ? query : ""); + + cgiSetArray("QTEXT", i, n->text); + cgiSetArray("QLINK", i, line); + + if (!helpfile && n->anchor) + { + nn = helpFindNode(hi, n->filename, NULL); + + snprintf(line, sizeof(line), "/help/%s?QUERY=%s", nn->filename, + query ? query : ""); + + cgiSetArray("QPTEXT", i, nn->text); + cgiSetArray("QPLINK", i, line); + } + else + { + cgiSetArray("QPTEXT", i, ""); + cgiSetArray("QPLINK", i, ""); + } + + fprintf(stderr, "DEBUG: [%d] = \"%s\" @ \"%s\"\n", i, n->text, line); + } + + helpDeleteIndex(si); + } + + /* + * OK, now list the bookmarks within the index... + */ + + for (i = 0, section = NULL, n = (help_node_t *)cupsArrayFirst(hi->sorted); + n; + n = (help_node_t *)cupsArrayNext(hi->sorted)) + { + if (n->anchor) + continue; + + /* + * Add a section link as needed... + */ + + if (n->section && + (!section || strcmp(n->section, section))) + { + /* + * Add a link for this node... + */ + + snprintf(line, sizeof(line), "/help/?TOPIC=%s&QUERY=%s", + cgiFormEncode(topic_data, n->section, sizeof(topic_data)), + query ? query : ""); + cgiSetArray("BMLINK", i, line); + cgiSetArray("BMTEXT", i, n->section); + cgiSetArray("BMINDENT", i, "0"); + + i ++; + section = n->section; + } + + if (!topic || strcmp(n->section, topic)) + continue; + + /* + * Add a link for this node... + */ + + snprintf(line, sizeof(line), "/help/%s?TOPIC=%s&QUERY=%s", n->filename, + cgiFormEncode(topic_data, n->section, sizeof(topic_data)), + query ? query : ""); + cgiSetArray("BMLINK", i, line); + cgiSetArray("BMTEXT", i, n->text); + cgiSetArray("BMINDENT", i, "1"); + + i ++; + + if (helpfile && !strcmp(helpfile, n->filename)) + { + help_node_t *nn; /* Pointer to sub-node */ + + + cupsArraySave(hi->sorted); + + for (nn = (help_node_t *)cupsArrayFirst(hi->sorted); + nn; + nn = (help_node_t *)cupsArrayNext(hi->sorted)) + if (nn->anchor && !strcmp(helpfile, nn->filename)) + { + /* + * Add a link for this node... + */ + + snprintf(line, sizeof(line), "#%s", nn->anchor); + cgiSetArray("BMLINK", i, line); + cgiSetArray("BMTEXT", i, nn->text); + cgiSetArray("BMINDENT", i, "2"); + + i ++; + } + + cupsArrayRestore(hi->sorted); + } + } + + /* + * Show the search and bookmark content... + */ + + if (!helpfile || !printable) + cgiCopyTemplateLang("help-header.tmpl"); + else + cgiCopyTemplateLang("help-printable.tmpl"); + + /* + * If we are viewing a file, copy it in now... + */ + + if (helpfile) + { + if ((fp = cupsFileOpen(filename, "r")) != NULL) + { + int inbody; /* Are we inside the body? */ + + + inbody = 0; + + while (cupsFileGets(fp, line, sizeof(line))) + { + if (inbody) + { + if (!_cups_strncasecmp(line, "</BODY>", 7)) + break; + + printf("%s\n", line); + } + else if (!_cups_strncasecmp(line, "<BODY", 5)) + inbody = 1; + } + + cupsFileClose(fp); + } + else + { + perror(filename); + cgiSetVariable("ERROR", "Unable to open help file."); + cgiCopyTemplateLang("error.tmpl"); + } + } + + /* + * Send a standard trailer... + */ + + if (!printable) + { + cgiCopyTemplateLang("help-trailer.tmpl"); + cgiEndHTML(); + } + else + puts("</BODY>\n</HTML>"); + + /* + * Delete the index... + */ + + helpDeleteIndex(hi); + + /* + * Return with no errors... + */ + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/html.c b/cgi-bin/html.c new file mode 100644 index 0000000000..9db5438ebf --- /dev/null +++ b/cgi-bin/html.c @@ -0,0 +1,239 @@ +/* + * "$Id$" + * + * HTML support functions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cgiEndHTML() - End a HTML page. + * cgiEndMultipart() - End the delivery of a multipart web page. + * cgiFormEncode() - Encode a string as a form variable. + * cgiStartHTML() - Start a HTML page. + * cgiStartMultipart() - Start a multipart delivery of a web page. + * cgiSupportsMultipart() - Does the browser support multi-part documents? + * cgi_null_passwd() - Return a NULL password for authentication. + */ + +/* + * Include necessary headers... + */ + +#include "cgi-private.h" + + +/* + * Local globals... + */ + +static const char *cgi_multipart = NULL; + /* Multipart separator, if any */ + + +/* + * Local functions... + */ + +static const char *cgi_null_passwd(const char *prompt); + + +/* + * 'cgiEndHTML()' - End a HTML page. + */ + +void +cgiEndHTML(void) +{ + /* + * Send the standard trailer... + */ + + cgiCopyTemplateLang("trailer.tmpl"); +} + + +/* + * 'cgiEndMultipart()' - End the delivery of a multipart web page. + */ + +void +cgiEndMultipart(void) +{ + if (cgi_multipart) + { + printf("\n%s--\n", cgi_multipart); + fflush(stdout); + } +} + + +/* + * 'cgiFormEncode()' - Encode a string as a form variable. + */ + +char * /* O - Destination string */ +cgiFormEncode(char *dst, /* I - Destination string */ + const char *src, /* I - Source string */ + size_t dstsize) /* I - Size of destination string */ +{ + char *dstptr, /* Pointer into destination */ + *dstend; /* End of destination */ + static const char *hex = /* Hexadecimal characters */ + "0123456789ABCDEF"; + + + /* + * Mark the end of the string... + */ + + dstend = dst + dstsize - 1; + + /* + * Loop through the source string and copy... + */ + + for (dstptr = dst; *src && dstptr < dstend;) + { + switch (*src) + { + case ' ' : + /* + * Encode spaces with a "+"... + */ + + *dstptr++ = '+'; + src ++; + break; + + case '&' : + case '%' : + case '+' : + /* + * Encode special characters with %XX escape... + */ + + if (dstptr < (dstend - 2)) + { + *dstptr++ = '%'; + *dstptr++ = hex[(*src & 255) >> 4]; + *dstptr++ = hex[*src & 15]; + src ++; + } + break; + + default : + /* + * Copy other characters literally... + */ + + *dstptr++ = *src++; + break; + } + } + + /* + * Nul-terminate the destination string... + */ + + *dstptr = '\0'; + + /* + * Return the encoded string... + */ + + return (dst); +} + + +/* + * 'cgiStartHTML()' - Start a HTML page. + */ + +void +cgiStartHTML(const char *title) /* I - Title of page */ +{ + /* + * Disable any further authentication attempts... + */ + + cupsSetPasswordCB(cgi_null_passwd); + + /* + * Tell the client to expect UTF-8 encoded HTML... + */ + + if (cgi_multipart) + puts(cgi_multipart); + + puts("Content-Type: text/html;charset=utf-8\n"); + + /* + * Send a standard header... + */ + + cgiSetVariable("TITLE", title); + cgiSetServerVersion(); + + cgiCopyTemplateLang("header.tmpl"); +} + + +/* + * 'cgiStartMultipart()' - Start a multipart delivery of a web page. + */ + +void +cgiStartMultipart(void) +{ + puts("MIME-Version: 1.0\n" + "Content-Type: multipart/x-mixed-replace; boundary=\"CUPS-MULTIPART\"\n"); + fflush(stdout); + + cgi_multipart = "--CUPS-MULTIPART"; +} + + +/* + * 'cgiSupportsMultipart()' - Does the browser support multi-part documents? + */ + +int /* O - 1 if multi-part supported, 0 otherwise */ +cgiSupportsMultipart(void) +{ + /* + * Too many bug reports for browsers that don't support it, and too much pain + * to whitelist known-good browsers, so for now we just punt on multi-part + * support... :( + */ + + return (0); +} + + +/* + * 'cgi_null_passwd()' - Return a NULL password for authentication. + */ + +static const char * /* O - NULL */ +cgi_null_passwd(const char *prompt) /* I - Prompt string (unused) */ +{ + (void)prompt; + + fprintf(stderr, "DEBUG: cgi_null_passwd(prompt=\"%s\") called!\n", + prompt ? prompt : "(null)"); + + return (NULL); +} + + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/ipp-var.c b/cgi-bin/ipp-var.c new file mode 100644 index 0000000000..09288f28c6 --- /dev/null +++ b/cgi-bin/ipp-var.c @@ -0,0 +1,1592 @@ +/* + * "$Id$" + * + * CGI <-> IPP variable routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cgiGetAttributes() - Get the list of attributes that are needed by the + * template file. + * cgiGetIPPObjects() - Get the objects in an IPP response. + * cgiMoveJobs() - Move one or more jobs. + * cgiPrintCommand() - Print a CUPS command job. + * cgiPrintTestPage() - Print a test page. + * cgiRewriteURL() - Rewrite a printer URI into a web browser URL... + * cgiSetIPPObjectVars() - Set CGI variables from an IPP object. + * cgiSetIPPVars() - Set CGI variables from an IPP response. + * cgiShowIPPError() - Show the last IPP error message. + * cgiShowJobs() - Show print jobs. + * cgiText() - Return localized text. + */ + +/* + * Include necessary headers... + */ + +#include "cgi-private.h" + + +/* + * 'cgiGetAttributes()' - Get the list of attributes that are needed + * by the template file. + */ + +void +cgiGetAttributes(ipp_t *request, /* I - IPP request */ + const char *tmpl) /* I - Base filename */ +{ + int num_attrs; /* Number of attributes */ + char *attrs[1000]; /* Attributes */ + int i; /* Looping var */ + char filename[1024], /* Filename */ + locale[16]; /* Locale name */ + const char *directory, /* Directory */ + *lang; /* Language */ + FILE *in; /* Input file */ + int ch; /* Character from file */ + char name[255], /* Name of variable */ + *nameptr; /* Pointer into name */ + + + /* + * Convert the language to a locale name... + */ + + if ((lang = getenv("LANG")) != NULL) + { + for (i = 0; lang[i] && i < 15; i ++) + if (isalnum(lang[i] & 255)) + locale[i] = tolower(lang[i]); + else + locale[i] = '_'; + + locale[i] = '\0'; + } + else + locale[0] = '\0'; + + /* + * See if we have a template file for this language... + */ + + directory = cgiGetTemplateDir(); + + snprintf(filename, sizeof(filename), "%s/%s/%s", directory, locale, tmpl); + if (access(filename, 0)) + { + locale[2] = '\0'; + + snprintf(filename, sizeof(filename), "%s/%s/%s", directory, locale, tmpl); + if (access(filename, 0)) + snprintf(filename, sizeof(filename), "%s/%s", directory, tmpl); + } + + /* + * Open the template file... + */ + + if ((in = fopen(filename, "r")) == NULL) + return; + + /* + * Loop through the file adding attribute names as needed... + */ + + num_attrs = 0; + attrs[0] = NULL; /* Eliminate compiler warning */ + + while ((ch = getc(in)) != EOF) + if (ch == '\\') + getc(in); + else if (ch == '{' && num_attrs < (sizeof(attrs) / sizeof(attrs[0]))) + { + /* + * Grab the name... + */ + + for (nameptr = name; (ch = getc(in)) != EOF;) + if (strchr("}]<>=!~ \t\n", ch)) + break; + else if (nameptr > name && ch == '?') + break; + else if (nameptr < (name + sizeof(name) - 1)) + { + if (ch == '_') + *nameptr++ = '-'; + else + *nameptr++ = ch; + } + + *nameptr = '\0'; + + if (!strncmp(name, "printer_state_history", 21)) + strcpy(name, "printer_state_history"); + + /* + * Possibly add it to the list of attributes... + */ + + for (i = 0; i < num_attrs; i ++) + if (!strcmp(attrs[i], name)) + break; + + if (i >= num_attrs) + { + attrs[num_attrs] = strdup(name); + num_attrs ++; + } + } + + /* + * If we have attributes, add a requested-attributes attribute to the + * request... + */ + + if (num_attrs > 0) + { + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", num_attrs, NULL, (const char **)attrs); + + for (i = 0; i < num_attrs; i ++) + free(attrs[i]); + } + + fclose(in); +} + + +/* + * 'cgiGetIPPObjects()' - Get the objects in an IPP response. + */ + +cups_array_t * /* O - Array of objects */ +cgiGetIPPObjects(ipp_t *response, /* I - IPP response */ + void *search) /* I - Search filter */ +{ + int i; /* Looping var */ + cups_array_t *objs; /* Array of objects */ + ipp_attribute_t *attr, /* Current attribute */ + *first; /* First attribute for object */ + ipp_tag_t group; /* Current group tag */ + int add; /* Add this object to the array? */ + + + if (!response) + return (0); + + for (add = 0, first = NULL, objs = cupsArrayNew(NULL, NULL), + group = IPP_TAG_ZERO, attr = response->attrs; + attr; + attr = attr->next) + { + if (attr->group_tag != group) + { + group = attr->group_tag; + + if (group != IPP_TAG_ZERO && group != IPP_TAG_OPERATION) + { + first = attr; + add = 0; + } + else if (add && first) + { + cupsArrayAdd(objs, first); + + add = 0; + first = NULL; + } + } + + if (attr->name && attr->group_tag != IPP_TAG_OPERATION && !add) + { + if (!search) + { + /* + * Add all objects if there is no search... + */ + + add = 1; + } + else + { + /* + * Check the search string against the string and integer values. + */ + + switch (attr->value_tag) + { + case IPP_TAG_TEXTLANG : + case IPP_TAG_NAMELANG : + case IPP_TAG_TEXT : + case IPP_TAG_NAME : + case IPP_TAG_KEYWORD : + case IPP_TAG_URI : + case IPP_TAG_MIMETYPE : + for (i = 0; !add && i < attr->num_values; i ++) + if (cgiDoSearch(search, attr->values[i].string.text)) + add = 1; + break; + + case IPP_TAG_INTEGER : + for (i = 0; !add && i < attr->num_values; i ++) + { + char buf[255]; /* Number buffer */ + + + sprintf(buf, "%d", attr->values[i].integer); + + if (cgiDoSearch(search, buf)) + add = 1; + } + break; + + default : + break; + } + } + } + } + + if (add && first) + cupsArrayAdd(objs, first); + + return (objs); +} + + +/* + * 'cgiMoveJobs()' - Move one or more jobs. + * + * At least one of dest or job_id must be non-zero/NULL. + */ + +void +cgiMoveJobs(http_t *http, /* I - Connection to server */ + const char *dest, /* I - Destination or NULL */ + int job_id) /* I - Job ID or 0 for all */ +{ + int i; /* Looping var */ + const char *user; /* Username */ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + ipp_attribute_t *attr; /* Current attribute */ + const char *name; /* Destination name */ + const char *job_printer_uri; /* JOB_PRINTER_URI form variable */ + char current_dest[1024]; /* Current destination */ + + + /* + * Make sure we have a username... + */ + + if ((user = getenv("REMOTE_USER")) == NULL) + { + puts("Status: 401\n"); + exit(0); + } + + /* + * See if the user has already selected a new destination... + */ + + if ((job_printer_uri = cgiGetVariable("JOB_PRINTER_URI")) == NULL) + { + /* + * Make sure necessary form variables are set... + */ + + if (job_id) + { + char temp[255]; /* Temporary string */ + + + sprintf(temp, "%d", job_id); + cgiSetVariable("JOB_ID", temp); + } + + if (dest) + cgiSetVariable("PRINTER_NAME", dest); + + /* + * No new destination specified, show the user what the available + * printers/classes are... + */ + + if (!dest) + { + /* + * Get the current destination for job N... + */ + + char job_uri[1024]; /* Job URI */ + + + request = ippNewRequest(IPP_GET_JOB_ATTRIBUTES); + + snprintf(job_uri, sizeof(job_uri), "ipp://localhost/jobs/%d", job_id); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-uri", + NULL, job_uri); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", NULL, "job-printer-uri"); + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + if ((attr = ippFindAttribute(response, "job-printer-uri", + IPP_TAG_URI)) != NULL) + { + /* + * Pull the name from the URI... + */ + + strlcpy(current_dest, strrchr(attr->values[0].string.text, '/') + 1, + sizeof(current_dest)); + dest = current_dest; + } + + ippDelete(response); + } + + if (!dest) + { + /* + * Couldn't get the current destination... + */ + + cgiStartHTML(cgiText(_("Move Job"))); + cgiShowIPPError(_("Unable to find destination for job")); + cgiEndHTML(); + return; + } + } + + /* + * Get the list of available destinations... + */ + + request = ippNewRequest(CUPS_GET_PRINTERS); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", NULL, "printer-uri-supported"); + + if (user) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, user); + + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_ENUM, "printer-type", + CUPS_PRINTER_LOCAL); + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_ENUM, "printer-type-mask", + CUPS_PRINTER_SCANNER); + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + for (i = 0, attr = ippFindAttribute(response, "printer-uri-supported", + IPP_TAG_URI); + attr; + attr = ippFindNextAttribute(response, "printer-uri-supported", + IPP_TAG_URI)) + { + /* + * Pull the name from the URI... + */ + + name = strrchr(attr->values[0].string.text, '/') + 1; + + /* + * If the name is not the same as the current destination, add it! + */ + + if (_cups_strcasecmp(name, dest)) + { + cgiSetArray("JOB_PRINTER_URI", i, attr->values[0].string.text); + cgiSetArray("JOB_PRINTER_NAME", i, name); + i ++; + } + } + + ippDelete(response); + } + + /* + * Show the form... + */ + + if (job_id) + cgiStartHTML(cgiText(_("Move Job"))); + else + cgiStartHTML(cgiText(_("Move All Jobs"))); + + if (cgiGetSize("JOB_PRINTER_NAME") > 0) + cgiCopyTemplateLang("job-move.tmpl"); + else + { + if (job_id) + cgiSetVariable("MESSAGE", cgiText(_("Unable to move job"))); + else + cgiSetVariable("MESSAGE", cgiText(_("Unable to move jobs"))); + + cgiSetVariable("ERROR", cgiText(_("No destinations added."))); + cgiCopyTemplateLang("error.tmpl"); + } + } + else + { + /* + * Try moving the job or jobs... + */ + + char uri[1024], /* Job/printer URI */ + resource[1024], /* Post resource */ + refresh[1024]; /* Refresh URL */ + const char *job_printer_name; /* New printer name */ + + + request = ippNewRequest(CUPS_MOVE_JOB); + + if (job_id) + { + /* + * Move 1 job... + */ + + snprintf(resource, sizeof(resource), "/jobs/%d", job_id); + + snprintf(uri, sizeof(uri), "ipp://localhost/jobs/%d", job_id); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-uri", + NULL, uri); + } + else + { + /* + * Move all active jobs on a destination... + */ + + snprintf(resource, sizeof(resource), "/%s/%s", + cgiGetVariable("SECTION"), dest); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", ippPort(), "/%s/%s", + cgiGetVariable("SECTION"), dest); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + } + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-printer-uri", + NULL, job_printer_uri); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, user); + + ippDelete(cupsDoRequest(http, request, resource)); + + /* + * Show the results... + */ + + job_printer_name = strrchr(job_printer_uri, '/') + 1; + + if (cupsLastError() <= IPP_OK_CONFLICT) + { + const char *path = strstr(job_printer_uri, "/printers/"); + if (!path) + { + path = strstr(job_printer_uri, "/classes/"); + cgiSetVariable("IS_CLASS", "YES"); + } + + if (path) + { + cgiFormEncode(uri, path, sizeof(uri)); + snprintf(refresh, sizeof(refresh), "2;URL=%s", uri); + cgiSetVariable("refresh_page", refresh); + } + } + + if (job_id) + cgiStartHTML(cgiText(_("Move Job"))); + else + cgiStartHTML(cgiText(_("Move All Jobs"))); + + if (cupsLastError() > IPP_OK_CONFLICT) + { + if (job_id) + cgiShowIPPError(_("Unable to move job")); + else + cgiShowIPPError(_("Unable to move jobs")); + } + else + { + cgiSetVariable("JOB_PRINTER_NAME", job_printer_name); + cgiCopyTemplateLang("job-moved.tmpl"); + } + } + + cgiEndHTML(); +} + + +/* + * 'cgiPrintCommand()' - Print a CUPS command job. + */ + +void +cgiPrintCommand(http_t *http, /* I - Connection to server */ + const char *dest, /* I - Destination printer */ + const char *command, /* I - Command to send */ + const char *title) /* I - Page/job title */ +{ + int job_id; /* Command file job */ + char uri[HTTP_MAX_URI], /* Job URI */ + resource[1024], /* Printer resource path */ + refresh[1024], /* Refresh URL */ + command_file[1024]; /* Command "file" */ + http_status_t status; /* Document status */ + cups_option_t hold_option; /* job-hold-until option */ + const char *user; /* User name */ + ipp_t *request, /* Get-Job-Attributes request */ + *response; /* Get-Job-Attributes response */ + ipp_attribute_t *attr; /* Current job attribute */ + static const char const *job_attrs[] =/* Job attributes we want */ + { + "job-state", + "job-printer-state-message" + }; + + + /* + * Create the CUPS command file... + */ + + snprintf(command_file, sizeof(command_file), "#CUPS-COMMAND\n%s\n", command); + + /* + * Show status... + */ + + if (cgiSupportsMultipart()) + { + cgiStartMultipart(); + cgiStartHTML(title); + cgiCopyTemplateLang("command.tmpl"); + cgiEndHTML(); + fflush(stdout); + } + + /* + * Send the command file job... + */ + + hold_option.name = "job-hold-until"; + hold_option.value = "no-hold"; + + if ((user = getenv("REMOTE_USER")) != NULL) + cupsSetUser(user); + else + cupsSetUser("anonymous"); + + if ((job_id = cupsCreateJob(http, dest, title, + 1, &hold_option)) < 1) + { + cgiSetVariable("MESSAGE", cgiText(_("Unable to send command to printer driver"))); + cgiSetVariable("ERROR", cupsLastErrorString()); + cgiStartHTML(title); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + + if (cgiSupportsMultipart()) + cgiEndMultipart(); + return; + } + + status = cupsStartDocument(http, dest, job_id, NULL, CUPS_FORMAT_COMMAND, 1); + if (status == HTTP_CONTINUE) + status = cupsWriteRequestData(http, command_file, + strlen(command_file)); + if (status == HTTP_CONTINUE) + cupsFinishDocument(http, dest); + + if (cupsLastError() >= IPP_REDIRECTION_OTHER_SITE) + { + cgiSetVariable("MESSAGE", cgiText(_("Unable to send command to printer driver"))); + cgiSetVariable("ERROR", cupsLastErrorString()); + cgiStartHTML(title); + cgiCopyTemplateLang("error.tmpl"); + cgiEndHTML(); + + if (cgiSupportsMultipart()) + cgiEndMultipart(); + + cupsCancelJob(dest, job_id); + return; + } + + /* + * Wait for the job to complete... + */ + + if (cgiSupportsMultipart()) + { + for (;;) + { + /* + * Get the current job state... + */ + + snprintf(uri, sizeof(uri), "ipp://localhost/jobs/%d", job_id); + request = ippNewRequest(IPP_GET_JOB_ATTRIBUTES); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-uri", + NULL, uri); + if (user) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, user); + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", 2, NULL, job_attrs); + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + cgiSetIPPVars(response, NULL, NULL, NULL, 0); + + attr = ippFindAttribute(response, "job-state", IPP_TAG_ENUM); + if (!attr || attr->values[0].integer >= IPP_JOB_STOPPED || + attr->values[0].integer == IPP_JOB_HELD) + { + ippDelete(response); + break; + } + + /* + * Job not complete, so update the status... + */ + + ippDelete(response); + + cgiStartHTML(title); + cgiCopyTemplateLang("command.tmpl"); + cgiEndHTML(); + fflush(stdout); + + sleep(5); + } + } + + /* + * Send the final page that reloads the printer's page... + */ + + snprintf(resource, sizeof(resource), "/printers/%s", dest); + + cgiFormEncode(uri, resource, sizeof(uri)); + snprintf(refresh, sizeof(refresh), "5;URL=%s", uri); + cgiSetVariable("refresh_page", refresh); + + cgiStartHTML(title); + cgiCopyTemplateLang("command.tmpl"); + cgiEndHTML(); + + if (cgiSupportsMultipart()) + cgiEndMultipart(); +} + + +/* + * 'cgiPrintTestPage()' - Print a test page. + */ + +void +cgiPrintTestPage(http_t *http, /* I - Connection to server */ + const char *dest) /* I - Destination printer/class */ +{ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + char uri[HTTP_MAX_URI], /* Printer URI */ + resource[1024], /* POST resource path */ + refresh[1024], /* Refresh URL */ + filename[1024]; /* Test page filename */ + const char *datadir; /* CUPS_DATADIR env var */ + const char *user; /* Username */ + + + /* + * See who is logged in... + */ + + user = getenv("REMOTE_USER"); + + /* + * Locate the test page file... + */ + + if ((datadir = getenv("CUPS_DATADIR")) == NULL) + datadir = CUPS_DATADIR; + + snprintf(filename, sizeof(filename), "%s/data/testprint", datadir); + + /* + * Point to the printer/class... + */ + + snprintf(resource, sizeof(resource), "/%s/%s", cgiGetVariable("SECTION"), + dest); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", ippPort(), "/%s/%s", cgiGetVariable("SECTION"), + dest); + + /* + * Build an IPP_PRINT_JOB request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requesting-user-name + */ + + request = ippNewRequest(IPP_PRINT_JOB); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + if (user) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, user); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "job-name", + NULL, "Test Page"); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoFileRequest(http, request, resource, + filename)) != NULL) + { + cgiSetIPPVars(response, NULL, NULL, NULL, 0); + + ippDelete(response); + } + + if (cupsLastError() <= IPP_OK_CONFLICT) + { + /* + * Automatically reload the printer status page... + */ + + cgiFormEncode(uri, resource, sizeof(uri)); + snprintf(refresh, sizeof(refresh), "2;URL=%s", uri); + cgiSetVariable("refresh_page", refresh); + } + else if (cupsLastError() == IPP_NOT_AUTHORIZED) + { + puts("Status: 401\n"); + exit(0); + } + + cgiStartHTML(cgiText(_("Print Test Page"))); + + if (cupsLastError() > IPP_OK_CONFLICT) + cgiShowIPPError(_("Unable to print test page:")); + else + { + cgiSetVariable("PRINTER_NAME", dest); + + cgiCopyTemplateLang("test-page.tmpl"); + } + + cgiEndHTML(); +} + + +/* + * 'cgiRewriteURL()' - Rewrite a printer URI into a web browser URL... + */ + +char * /* O - New URL */ +cgiRewriteURL(const char *uri, /* I - Current URI */ + char *url, /* O - New URL */ + int urlsize, /* I - Size of URL buffer */ + const char *newresource) /* I - Replacement resource */ +{ + char scheme[HTTP_MAX_URI], + userpass[HTTP_MAX_URI], + hostname[HTTP_MAX_URI], + rawresource[HTTP_MAX_URI], + resource[HTTP_MAX_URI], + /* URI components... */ + *rawptr, /* Pointer into rawresource */ + *resptr; /* Pointer into resource */ + int port; /* Port number */ + static int ishttps = -1; /* Using encryption? */ + static const char *server; /* Name of server */ + static char servername[1024]; + /* Local server name */ + static const char hexchars[] = "0123456789ABCDEF"; + /* Hexadecimal conversion characters */ + + + /* + * Check if we have been called before... + */ + + if (ishttps < 0) + { + /* + * No, initialize static vars for the conversion... + * + * First get the server name associated with the client interface as + * well as the locally configured hostname. We'll check *both* of + * these to see if the printer URL is local... + */ + + if ((server = getenv("SERVER_NAME")) == NULL) + server = ""; + + httpGetHostname(NULL, servername, sizeof(servername)); + + /* + * Then flag whether we are using SSL on this connection... + */ + + ishttps = getenv("HTTPS") != NULL; + } + + /* + * Convert the URI to a URL... + */ + + httpSeparateURI(HTTP_URI_CODING_ALL, uri, scheme, sizeof(scheme), userpass, + sizeof(userpass), hostname, sizeof(hostname), &port, + rawresource, sizeof(rawresource)); + + if (!strcmp(scheme, "ipp") || + !strcmp(scheme, "http") || + !strcmp(scheme, "https")) + { + if (newresource) + { + /* + * Force the specified resource name instead of the one in the URL... + */ + + strlcpy(resource, newresource, sizeof(resource)); + } + else + { + /* + * Rewrite the resource string so it doesn't contain any + * illegal chars... + */ + + for (rawptr = rawresource, resptr = resource; *rawptr; rawptr ++) + if ((*rawptr & 128) || *rawptr == '%' || *rawptr == ' ' || + *rawptr == '#' || *rawptr == '?' || + *rawptr == '.') /* For MSIE */ + { + if (resptr < (resource + sizeof(resource) - 3)) + { + *resptr++ = '%'; + *resptr++ = hexchars[(*rawptr >> 4) & 15]; + *resptr++ = hexchars[*rawptr & 15]; + } + } + else if (resptr < (resource + sizeof(resource) - 1)) + *resptr++ = *rawptr; + + *resptr = '\0'; + } + + /* + * Map local access to a local URI... + */ + + if (!_cups_strcasecmp(hostname, "127.0.0.1") || + !_cups_strcasecmp(hostname, "[::1]") || + !_cups_strcasecmp(hostname, "localhost") || + !_cups_strncasecmp(hostname, "localhost.", 10) || + !_cups_strcasecmp(hostname, server) || + !_cups_strcasecmp(hostname, servername)) + { + /* + * Make URI relative to the current server... + */ + + strlcpy(url, resource, urlsize); + } + else + { + /* + * Rewrite URI with HTTP/HTTPS scheme... + */ + + if (userpass[0]) + snprintf(url, urlsize, "%s://%s@%s:%d%s", + ishttps ? "https" : "http", + userpass, hostname, port, resource); + else + snprintf(url, urlsize, "%s://%s:%d%s", + ishttps ? "https" : "http", + hostname, port, resource); + } + } + else + strlcpy(url, uri, urlsize); + + return (url); +} + + +/* + * 'cgiSetIPPObjectVars()' - Set CGI variables from an IPP object. + */ + +ipp_attribute_t * /* O - Next object */ +cgiSetIPPObjectVars( + ipp_attribute_t *obj, /* I - Response data to be copied... */ + const char *prefix, /* I - Prefix for name or NULL */ + int element) /* I - Parent element number */ +{ + ipp_attribute_t *attr; /* Attribute in response... */ + int i; /* Looping var */ + char name[1024], /* Name of attribute */ + *nameptr, /* Pointer into name */ + value[16384], /* Value(s) */ + *valptr; /* Pointer into value */ + struct tm *date; /* Date information */ + + + fprintf(stderr, "DEBUG2: cgiSetIPPObjectVars(obj=%p, prefix=\"%s\", " + "element=%d)\n", + obj, prefix ? prefix : "(null)", element); + + /* + * Set common CGI template variables... + */ + + if (!prefix) + cgiSetServerVersion(); + + /* + * Loop through the attributes and set them for the template... + */ + + for (attr = obj; attr && attr->group_tag != IPP_TAG_ZERO; attr = attr->next) + { + /* + * Copy the attribute name, substituting "_" for "-"... + */ + + if (!attr->name) + continue; + + if (prefix) + { + snprintf(name, sizeof(name), "%s.", prefix); + nameptr = name + strlen(name); + } + else + nameptr = name; + + for (i = 0; attr->name[i] && nameptr < (name + sizeof(name) - 1); i ++) + if (attr->name[i] == '-') + *nameptr++ = '_'; + else + *nameptr++ = attr->name[i]; + + *nameptr = '\0'; + + /* + * Add "job_printer_name" variable if we have a "job_printer_uri" + * attribute... + */ + + if (!strcmp(name, "job_printer_uri")) + { + if ((valptr = strrchr(attr->values[0].string.text, '/')) == NULL) + valptr = "unknown"; + else + valptr ++; + + cgiSetArray("job_printer_name", element, valptr); + } + + /* + * Localize event names in "notify_events" variable... + */ + + if (!strcmp(name, "notify_events")) + { + size_t remaining; /* Remaining bytes in buffer */ + + + value[0] = '\0'; + valptr = value; + + for (i = 0; i < attr->num_values; i ++) + { + if (valptr >= (value + sizeof(value) - 3)) + break; + + if (i) + { + *valptr++ = ','; + *valptr++ = ' '; + } + + remaining = sizeof(value) - (valptr - value); + + if (!strcmp(attr->values[i].string.text, "printer-stopped")) + strlcpy(valptr, _("Printer Paused"), remaining); + else if (!strcmp(attr->values[i].string.text, "printer-added")) + strlcpy(valptr, _("Printer Added"), remaining); + else if (!strcmp(attr->values[i].string.text, "printer-modified")) + strlcpy(valptr, _("Printer Modified"), remaining); + else if (!strcmp(attr->values[i].string.text, "printer-deleted")) + strlcpy(valptr, _("Printer Deleted"), remaining); + else if (!strcmp(attr->values[i].string.text, "job-created")) + strlcpy(valptr, _("Job Created"), remaining); + else if (!strcmp(attr->values[i].string.text, "job-completed")) + strlcpy(valptr, _("Job Completed"), remaining); + else if (!strcmp(attr->values[i].string.text, "job-stopped")) + strlcpy(valptr, _("Job Stopped"), remaining); + else if (!strcmp(attr->values[i].string.text, "job-config-changed")) + strlcpy(valptr, _("Job Options Changed"), remaining); + else if (!strcmp(attr->values[i].string.text, "server-restarted")) + strlcpy(valptr, _("Server Restarted"), remaining); + else if (!strcmp(attr->values[i].string.text, "server-started")) + strlcpy(valptr, _("Server Started"), remaining); + else if (!strcmp(attr->values[i].string.text, "server-stopped")) + strlcpy(valptr, _("Server Stopped"), remaining); + else if (!strcmp(attr->values[i].string.text, "server-audit")) + strlcpy(valptr, _("Server Security Auditing"), remaining); + else + strlcpy(valptr, attr->values[i].string.text, remaining); + + valptr += strlen(valptr); + } + + cgiSetArray("notify_events", element, value); + continue; + } + + /* + * Add "notify_printer_name" variable if we have a "notify_printer_uri" + * attribute... + */ + + if (!strcmp(name, "notify_printer_uri")) + { + if ((valptr = strrchr(attr->values[0].string.text, '/')) == NULL) + valptr = "unknown"; + else + valptr ++; + + cgiSetArray("notify_printer_name", element, valptr); + } + + /* + * Add "notify_recipient_name" variable if we have a "notify_recipient_uri" + * attribute, and rewrite recipient URI... + */ + + if (!strcmp(name, "notify_recipient_uri")) + { + char uri[1024], /* New URI */ + scheme[32], /* Scheme portion of URI */ + userpass[256], /* Username/password portion of URI */ + host[1024], /* Hostname portion of URI */ + resource[1024], /* Resource portion of URI */ + *options; /* Options in URI */ + int port; /* Port number */ + + + httpSeparateURI(HTTP_URI_CODING_ALL, attr->values[0].string.text, + scheme, sizeof(scheme), userpass, sizeof(userpass), + host, sizeof(host), &port, resource, sizeof(resource)); + + if (!strcmp(scheme, "rss")) + { + /* + * RSS notification... + */ + + if ((options = strchr(resource, '?')) != NULL) + *options = '\0'; + + if (host[0]) + { + /* + * Link to remote feed... + */ + + httpAssembleURI(HTTP_URI_CODING_ALL, uri, sizeof(uri), "http", + userpass, host, port, resource); + strlcpy(name, uri, sizeof(name)); + } + else + { + /* + * Link to local feed... + */ + + snprintf(uri, sizeof(uri), "/rss%s", resource); + strlcpy(name, resource + 1, sizeof(name)); + } + } + else + { + /* + * Other... + */ + + strlcpy(uri, attr->values[0].string.text, sizeof(uri)); + strlcpy(name, resource, sizeof(name)); + } + + cgiSetArray("notify_recipient_uri", element, uri); + cgiSetArray("notify_recipient_name", element, name); + continue; + } + + /* + * Add "admin_uri" variable if we have a "printer_uri_supported" + * attribute... + */ + + if (!strcmp(name, "printer_uri_supported")) + { + cgiRewriteURL(attr->values[0].string.text, value, sizeof(value), + "/admin/"); + + cgiSetArray("admin_uri", element, value); + } + + /* + * Copy values... + */ + + value[0] = '\0'; /* Initially an empty string */ + valptr = value; /* Start at the beginning */ + + for (i = 0; i < attr->num_values; i ++) + { + if (i) + strlcat(valptr, ", ", sizeof(value) - (valptr - value)); + + valptr += strlen(valptr); + + switch (attr->value_tag) + { + case IPP_TAG_INTEGER : + case IPP_TAG_ENUM : + if (strncmp(name, "time_at_", 8) == 0) + { + time_t t; /* Temporary time value */ + + t = (time_t)attr->values[i].integer; + date = localtime(&t); + + strftime(valptr, sizeof(value) - (valptr - value), "%c", date); + } + else + snprintf(valptr, sizeof(value) - (valptr - value), + "%d", attr->values[i].integer); + break; + + case IPP_TAG_BOOLEAN : + snprintf(valptr, sizeof(value) - (valptr - value), + "%d", attr->values[i].boolean); + break; + + case IPP_TAG_NOVALUE : + strlcat(valptr, "novalue", sizeof(value) - (valptr - value)); + break; + + case IPP_TAG_RANGE : + snprintf(valptr, sizeof(value) - (valptr - value), + "%d-%d", attr->values[i].range.lower, + attr->values[i].range.upper); + break; + + case IPP_TAG_RESOLUTION : + snprintf(valptr, sizeof(value) - (valptr - value), + "%dx%d%s", attr->values[i].resolution.xres, + attr->values[i].resolution.yres, + attr->values[i].resolution.units == IPP_RES_PER_INCH ? + "dpi" : "dpc"); + break; + + case IPP_TAG_URI : + if (strchr(attr->values[i].string.text, ':') && + strcmp(name, "device_uri")) + { + /* + * Rewrite URIs... + */ + + if (!strcmp(name, "member_uris")) + { + char url[1024]; /* URL for class member... */ + + + cgiRewriteURL(attr->values[i].string.text, url, + sizeof(url), NULL); + + snprintf(valptr, sizeof(value) - (valptr - value), + "<A HREF=\"%s\">%s</A>", url, + strrchr(attr->values[i].string.text, '/') + 1); + } + else + cgiRewriteURL(attr->values[i].string.text, valptr, + sizeof(value) - (valptr - value), NULL); + break; + } + + case IPP_TAG_STRING : + case IPP_TAG_TEXT : + case IPP_TAG_NAME : + case IPP_TAG_KEYWORD : + case IPP_TAG_CHARSET : + case IPP_TAG_LANGUAGE : + case IPP_TAG_MIMETYPE : + strlcat(valptr, attr->values[i].string.text, + sizeof(value) - (valptr - value)); + break; + + case IPP_TAG_BEGIN_COLLECTION : + snprintf(value, sizeof(value), "%s%d", name, i + 1); + cgiSetIPPVars(attr->values[i].collection, NULL, NULL, value, + element); + break; + + default : + break; /* anti-compiler-warning-code */ + } + } + + /* + * Add the element... + */ + + if (attr->value_tag != IPP_TAG_BEGIN_COLLECTION) + { + cgiSetArray(name, element, value); + + fprintf(stderr, "DEBUG2: %s[%d]=\"%s\"\n", name, element, value); + } + } + + return (attr ? attr->next : NULL); +} + + +/* + * 'cgiSetIPPVars()' - Set CGI variables from an IPP response. + */ + +int /* O - Maximum number of elements */ +cgiSetIPPVars(ipp_t *response, /* I - Response data to be copied... */ + const char *filter_name, /* I - Filter name */ + const char *filter_value, /* I - Filter value */ + const char *prefix, /* I - Prefix for name or NULL */ + int parent_el) /* I - Parent element number */ +{ + int element; /* Element in CGI array */ + ipp_attribute_t *attr, /* Attribute in response... */ + *filter; /* Filtering attribute */ + + + fprintf(stderr, "DEBUG2: cgiSetIPPVars(response=%p, filter_name=\"%s\", " + "filter_value=\"%s\", prefix=\"%s\", parent_el=%d)\n", + response, filter_name ? filter_name : "(null)", + filter_value ? filter_value : "(null)", + prefix ? prefix : "(null)", parent_el); + + /* + * Set common CGI template variables... + */ + + if (!prefix) + cgiSetServerVersion(); + + /* + * Loop through the attributes and set them for the template... + */ + + attr = response->attrs; + + if (!prefix) + while (attr && attr->group_tag == IPP_TAG_OPERATION) + attr = attr->next; + + for (element = parent_el; attr; element ++) + { + /* + * Copy attributes to a separator... + */ + + while (attr && attr->group_tag == IPP_TAG_ZERO) + attr= attr->next; + + if (!attr) + break; + + if (filter_name) + { + for (filter = attr; + filter != NULL && filter->group_tag != IPP_TAG_ZERO; + filter = filter->next) + if (filter->name && !strcmp(filter->name, filter_name) && + (filter->value_tag == IPP_TAG_STRING || + (filter->value_tag >= IPP_TAG_TEXTLANG && + filter->value_tag <= IPP_TAG_MIMETYPE)) && + filter->values[0].string.text != NULL && + !_cups_strcasecmp(filter->values[0].string.text, filter_value)) + break; + + if (!filter) + return (element + 1); + + if (filter->group_tag == IPP_TAG_ZERO) + { + attr = filter; + element --; + continue; + } + } + + attr = cgiSetIPPObjectVars(attr, prefix, element); + } + + fprintf(stderr, "DEBUG2: Returing %d from cgiSetIPPVars()...\n", element); + + return (element); +} + + +/* + * 'cgiShowIPPError()' - Show the last IPP error message. + * + * The caller must still call cgiStartHTML() and cgiEndHTML(). + */ + +void +cgiShowIPPError(const char *message) /* I - Contextual message */ +{ + cgiSetVariable("MESSAGE", cgiText(message)); + cgiSetVariable("ERROR", cupsLastErrorString()); + cgiCopyTemplateLang("error.tmpl"); +} + + +/* + * 'cgiShowJobs()' - Show print jobs. + */ + +void +cgiShowJobs(http_t *http, /* I - Connection to server */ + const char *dest) /* I - Destination name or NULL */ +{ + int i; /* Looping var */ + const char *which_jobs; /* Which jobs to show */ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + cups_array_t *jobs; /* Array of job objects */ + ipp_attribute_t *job; /* Job object */ + int ascending, /* Order of jobs (0 = descending) */ + first, /* First job to show */ + count; /* Number of jobs */ + const char *var, /* Form variable */ + *query, /* Query string */ + *section; /* Section in web interface */ + void *search; /* Search data */ + char url[1024], /* Printer URI */ + val[1024]; /* Form variable */ + + + /* + * Build an IPP_GET_JOBS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + */ + + request = ippNewRequest(IPP_GET_JOBS); + + if (dest) + { + httpAssembleURIf(HTTP_URI_CODING_ALL, url, sizeof(url), "ipp", NULL, + "localhost", ippPort(), "/printers/%s", dest); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, url); + } + else + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", NULL, + "ipp://localhost/"); + + if ((which_jobs = cgiGetVariable("which_jobs")) != NULL) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, "which-jobs", + NULL, which_jobs); + + cgiGetAttributes(request, "jobs.tmpl"); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + /* + * Get a list of matching job objects. + */ + + if ((query = cgiGetVariable("QUERY")) != NULL && + !cgiGetVariable("CLEAR")) + search = cgiCompileSearch(query); + else + { + query = NULL; + search = NULL; + } + + jobs = cgiGetIPPObjects(response, search); + count = cupsArrayCount(jobs); + + if (search) + cgiFreeSearch(search); + + /* + * Figure out which jobs to display... + */ + + if ((var = cgiGetVariable("FIRST")) != NULL) + first = atoi(var); + else + first = 0; + + if (first >= count) + first = count - CUPS_PAGE_MAX; + + first = (first / CUPS_PAGE_MAX) * CUPS_PAGE_MAX; + + if (first < 0) + first = 0; + + if ((var = cgiGetVariable("ORDER")) != NULL) + ascending = !_cups_strcasecmp(var, "asc"); + else + ascending = !which_jobs || !_cups_strcasecmp(which_jobs, "not-completed"); + + section = cgiGetVariable("SECTION"); + + cgiClearVariables(); + + if (query) + cgiSetVariable("QUERY", query); + + cgiSetVariable("ORDER", ascending ? "asc" : "dec"); + + cgiSetVariable("SECTION", section); + + sprintf(val, "%d", count); + cgiSetVariable("TOTAL", val); + + if (which_jobs) + cgiSetVariable("WHICH_JOBS", which_jobs); + + if (ascending) + { + for (i = 0, job = (ipp_attribute_t *)cupsArrayIndex(jobs, first); + i < CUPS_PAGE_MAX && job; + i ++, job = (ipp_attribute_t *)cupsArrayNext(jobs)) + cgiSetIPPObjectVars(job, NULL, i); + } + else + { + for (i = 0, job = (ipp_attribute_t *)cupsArrayIndex(jobs, count - first - 1); + i < CUPS_PAGE_MAX && job; + i ++, job = (ipp_attribute_t *)cupsArrayPrev(jobs)) + cgiSetIPPObjectVars(job, NULL, i); + } + + /* + * Save navigation URLs... + */ + + if (dest) + { + snprintf(val, sizeof(val), "/%s/%s", section, dest); + cgiSetVariable("PRINTER_NAME", dest); + cgiSetVariable("PRINTER_URI_SUPPORTED", val); + } + else + strlcpy(val, "/jobs/", sizeof(val)); + + cgiSetVariable("THISURL", val); + + if (first > 0) + { + sprintf(val, "%d", first - CUPS_PAGE_MAX); + cgiSetVariable("PREV", val); + } + + if ((first + CUPS_PAGE_MAX) < count) + { + sprintf(val, "%d", first + CUPS_PAGE_MAX); + cgiSetVariable("NEXT", val); + } + + /* + * Then show everything... + */ + + if (dest) + cgiSetVariable("SEARCH_DEST", dest); + + cgiCopyTemplateLang("search.tmpl"); + + cgiCopyTemplateLang("jobs-header.tmpl"); + + if (count > CUPS_PAGE_MAX) + cgiCopyTemplateLang("pager.tmpl"); + + cgiCopyTemplateLang("jobs.tmpl"); + + if (count > CUPS_PAGE_MAX) + cgiCopyTemplateLang("pager.tmpl"); + + cupsArrayDelete(jobs); + ippDelete(response); + } +} + + +/* + * 'cgiText()' - Return localized text. + */ + +const char * /* O - Localized message */ +cgiText(const char *message) /* I - Message */ +{ + static cups_lang_t *language = NULL; + /* Language */ + + + if (!language) + language = cupsLangDefault(); + + return (_cupsLangString(language, message)); +} + + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/jobs.c b/cgi-bin/jobs.c new file mode 100644 index 0000000000..6c4f09852e --- /dev/null +++ b/cgi-bin/jobs.c @@ -0,0 +1,214 @@ +/* + * "$Id$" + * + * Job status CGI for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Main entry for CGI. + * do_job_op() - Do a job operation. + */ + +/* + * Include necessary headers... + */ + +#include "cgi-private.h" + + +/* + * Local functions... + */ + +static void do_job_op(http_t *http, int job_id, ipp_op_t op); + + +/* + * 'main()' - Main entry for CGI. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + http_t *http; /* Connection to the server */ + const char *op; /* Operation name */ + const char *job_id_var; /* Job ID form variable */ + int job_id; /* Job ID */ + + + /* + * Get any form variables... + */ + + cgiInitialize(); + + /* + * Set the web interface section... + */ + + cgiSetVariable("SECTION", "jobs"); + cgiSetVariable("REFRESH_PAGE", ""); + + /* + * Connect to the HTTP server... + */ + + http = httpConnectEncrypt(cupsServer(), ippPort(), cupsEncryption()); + + /* + * Get the job ID, if any... + */ + + if ((job_id_var = cgiGetVariable("JOB_ID")) != NULL) + job_id = atoi(job_id_var); + else + job_id = 0; + + /* + * Do the operation... + */ + + if ((op = cgiGetVariable("OP")) != NULL && job_id > 0 && cgiIsPOST()) + { + /* + * Do the operation... + */ + + if (!strcmp(op, "cancel-job")) + do_job_op(http, job_id, IPP_CANCEL_JOB); + else if (!strcmp(op, "hold-job")) + do_job_op(http, job_id, IPP_HOLD_JOB); + else if (!strcmp(op, "move-job")) + cgiMoveJobs(http, NULL, job_id); + else if (!strcmp(op, "release-job")) + do_job_op(http, job_id, IPP_RELEASE_JOB); + else if (!strcmp(op, "restart-job")) + do_job_op(http, job_id, IPP_RESTART_JOB); + else + { + /* + * Bad operation code... Display an error... + */ + + cgiStartHTML(cgiText(_("Jobs"))); + cgiCopyTemplateLang("error-op.tmpl"); + cgiEndHTML(); + } + } + else + { + /* + * Show a list of jobs... + */ + + cgiStartHTML(cgiText(_("Jobs"))); + cgiShowJobs(http, NULL); + cgiEndHTML(); + } + + /* + * Close the HTTP server connection... + */ + + httpClose(http); + + /* + * Return with no errors... + */ + + return (0); +} + + +/* + * 'do_job_op()' - Do a job operation. + */ + +static void +do_job_op(http_t *http, /* I - HTTP connection */ + int job_id, /* I - Job ID */ + ipp_op_t op) /* I - Operation to perform */ +{ + ipp_t *request; /* IPP request */ + char uri[HTTP_MAX_URI]; /* Job URI */ + const char *user; /* Username */ + + + /* + * Build a job request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * job-uri or printer-uri (purge-jobs) + * requesting-user-name + */ + + request = ippNewRequest(op); + + snprintf(uri, sizeof(uri), "ipp://localhost/jobs/%d", job_id); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-uri", + NULL, uri); + + if ((user = getenv("REMOTE_USER")) == NULL) + user = "guest"; + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, user); + + /* + * Do the request and get back a response... + */ + + ippDelete(cupsDoRequest(http, request, "/jobs")); + + if (cupsLastError() <= IPP_OK_CONFLICT && getenv("HTTP_REFERER")) + { + /* + * Redirect successful updates back to the parent page... + */ + + char url[1024]; /* Encoded URL */ + + + strcpy(url, "5;URL="); + cgiFormEncode(url + 6, getenv("HTTP_REFERER"), sizeof(url) - 6); + cgiSetVariable("refresh_page", url); + } + else if (cupsLastError() == IPP_NOT_AUTHORIZED) + { + puts("Status: 401\n"); + exit(0); + } + + cgiStartHTML(cgiText(_("Jobs"))); + + if (cupsLastError() > IPP_OK_CONFLICT) + cgiShowIPPError(_("Job operation failed:")); + else if (op == IPP_CANCEL_JOB) + cgiCopyTemplateLang("job-cancel.tmpl"); + else if (op == IPP_HOLD_JOB) + cgiCopyTemplateLang("job-hold.tmpl"); + else if (op == IPP_RELEASE_JOB) + cgiCopyTemplateLang("job-release.tmpl"); + else if (op == IPP_RESTART_JOB) + cgiCopyTemplateLang("job-restart.tmpl"); + + cgiEndHTML(); +} + + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/libcupscgi.exp b/cgi-bin/libcupscgi.exp new file mode 100644 index 0000000000..3a26229d79 --- /dev/null +++ b/cgi-bin/libcupscgi.exp @@ -0,0 +1,42 @@ +_cgiCheckVariables +_cgiClearVariables +_cgiCompileSearch +_cgiCopyTemplateFile +_cgiCopyTemplateLang +_cgiDoSearch +_cgiEndHTML +_cgiEndMultipart +_cgiFormEncode +_cgiFreeSearch +_cgiGetArray +_cgiGetAttributes +_cgiGetCookie +_cgiGetFile +_cgiGetIPPObjects +_cgiGetSize +_cgiGetTemplateDir +_cgiGetVariable +_cgiInitialize +_cgiIsPOST +_cgiMoveJobs +_cgiPrintCommand +_cgiPrintTestPage +_cgiRewriteURL +_cgiSetArray +_cgiSetIPPObjectVars +_cgiSetIPPVars +_cgiSetCookie +_cgiSetServerVersion +_cgiSetSize +_cgiSetVariable +_cgiShowIPPError +_cgiShowJobs +_cgiStartHTML +_cgiStartMultipart +_cgiSupportsMultipart +_cgiText +_helpDeleteIndex +_helpFindNode +_helpLoadIndex +_helpSaveIndex +_helpSearchIndex diff --git a/cgi-bin/makedocset.c b/cgi-bin/makedocset.c new file mode 100644 index 0000000000..5772828893 --- /dev/null +++ b/cgi-bin/makedocset.c @@ -0,0 +1,486 @@ +/* + * "$Id$" + * + * Xcode documentation set generator. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Usage: + * + * makedocset directory *.tokens + * + * Contents: + * + * main() - Test the help index code. + * compare_html() - Compare the titles of two HTML files. + * compare_sections() - Compare the names of two help sections. + * compare_sections_files() - Compare the number of files and section names. + * write_index() - Write an index file for the CUPS help. + * write_info() - Write the Info.plist file. + * write_nodes() - Write the Nodes.xml file. + */ + +/* + * Include necessary headers... + */ + +#include "cgi.h" +#include <errno.h> + + +/* + * Local structures... + */ + +typedef struct _cups_html_s /**** Help file ****/ +{ + char *path; /* Path to help file */ + char *title; /* Title of help file */ +} _cups_html_t; + +typedef struct _cups_section_s /**** Help section ****/ +{ + char *name; /* Section name */ + cups_array_t *files; /* Files in this section */ +} _cups_section_t; + + +/* + * Local functions... + */ + +static int compare_html(_cups_html_t *a, _cups_html_t *b); +static int compare_sections(_cups_section_t *a, _cups_section_t *b); +static int compare_sections_files(_cups_section_t *a, _cups_section_t *b); +static void write_index(const char *path, help_index_t *hi); +static void write_info(const char *path, const char *revision); +static void write_nodes(const char *path, help_index_t *hi); + + +/* + * 'main()' - Test the help index code. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + char path[1024], /* Path to documentation */ + line[1024]; /* Line from file */ + help_index_t *hi; /* Help index */ + cups_file_t *tokens, /* Tokens.xml file */ + *fp; /* Current file */ + + + if (argc < 4) + { + puts("Usage: makedocset directory revision *.tokens"); + return (1); + } + + /* + * Index the help documents... + */ + + snprintf(path, sizeof(path), "%s/Contents/Resources/Documentation", argv[1]); + if ((hi = helpLoadIndex(NULL, path)) == NULL) + { + fputs("makedocset: Unable to index help files!\n", stderr); + return (1); + } + + snprintf(path, sizeof(path), "%s/Contents/Resources/Documentation/index.html", + argv[1]); + write_index(path, hi); + + snprintf(path, sizeof(path), "%s/Contents/Resources/Nodes.xml", argv[1]); + write_nodes(path, hi); + + /* + * Write the Info.plist file... + */ + + snprintf(path, sizeof(path), "%s/Contents/Info.plist", argv[1]); + write_info(path, argv[2]); + + /* + * Merge the Tokens.xml files... + */ + + snprintf(path, sizeof(path), "%s/Contents/Resources/Tokens.xml", argv[1]); + if ((tokens = cupsFileOpen(path, "w")) == NULL) + { + fprintf(stderr, "makedocset: Unable to create \"%s\": %s\n", path, + strerror(errno)); + return (1); + } + + cupsFilePuts(tokens, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"); + cupsFilePuts(tokens, "<Tokens version=\"1.0\">\n"); + + for (i = 3; i < argc; i ++) + { + if ((fp = cupsFileOpen(argv[i], "r")) == NULL) + { + fprintf(stderr, "makedocset: Unable to open \"%s\": %s\n", argv[i], + strerror(errno)); + return (1); + } + + if (!cupsFileGets(fp, line, sizeof(line)) || strncmp(line, "<?xml ", 6) || + !cupsFileGets(fp, line, sizeof(line)) || strncmp(line, "<Tokens ", 8)) + { + fprintf(stderr, "makedocset: Bad Tokens.xml file \"%s\"!\n", argv[i]); + return (1); + } + + while (cupsFileGets(fp, line, sizeof(line))) + { + if (strcmp(line, "</Tokens>")) + cupsFilePrintf(tokens, "%s\n", line); + } + + cupsFileClose(fp); + } + + cupsFilePuts(tokens, "</Tokens>\n"); + + cupsFileClose(tokens); + + /* + * Return with no errors... + */ + + return (0); +} + + +/* + * 'compare_html()' - Compare the titles of two HTML files. + */ + +static int /* O - Result of comparison */ +compare_html(_cups_html_t *a, /* I - First file */ + _cups_html_t *b) /* I - Second file */ +{ + return (_cups_strcasecmp(a->title, b->title)); +} + + +/* + * 'compare_sections()' - Compare the names of two help sections. + */ + +static int /* O - Result of comparison */ +compare_sections(_cups_section_t *a, /* I - First section */ + _cups_section_t *b) /* I - Second section */ +{ + return (_cups_strcasecmp(a->name, b->name)); +} + + +/* + * 'compare_sections_files()' - Compare the number of files and section names. + */ + +static int /* O - Result of comparison */ +compare_sections_files( + _cups_section_t *a, /* I - First section */ + _cups_section_t *b) /* I - Second section */ +{ + int ret = cupsArrayCount(b->files) - cupsArrayCount(a->files); + + if (ret) + return (ret); + else + return (_cups_strcasecmp(a->name, b->name)); +} + + +/* + * 'write_index()' - Write an index file for the CUPS help. + */ + +static void +write_index(const char *path, /* I - File to write */ + help_index_t *hi) /* I - Index of files */ +{ + cups_file_t *fp; /* Output file */ + help_node_t *node; /* Current help node */ + _cups_section_t *section, /* Current section */ + key; /* Section search key */ + _cups_html_t *html; /* Current HTML file */ + cups_array_t *sections, /* Sections in index */ + *sections_files,/* Sections sorted by size */ + *columns[3]; /* Columns in final HTML file */ + int column, /* Current column */ + lines[3], /* Number of lines in each column */ + min_column, /* Smallest column */ + min_lines; /* Smallest number of lines */ + + + /* + * Build an array of sections and their files. + */ + + sections = cupsArrayNew((cups_array_func_t)compare_sections, NULL); + + for (node = (help_node_t *)cupsArrayFirst(hi->nodes); + node; + node = (help_node_t *)cupsArrayNext(hi->nodes)) + { + if (node->anchor) + continue; + + key.name = node->section ? node->section : "Miscellaneous"; + if ((section = (_cups_section_t *)cupsArrayFind(sections, &key)) == NULL) + { + section = (_cups_section_t *)calloc(1, sizeof(_cups_section_t)); + section->name = key.name; + section->files = cupsArrayNew((cups_array_func_t)compare_html, NULL); + + cupsArrayAdd(sections, section); + } + + html = (_cups_html_t *)calloc(1, sizeof(_cups_html_t)); + html->path = node->filename; + html->title = node->text; + + cupsArrayAdd(section->files, html); + } + + /* + * Build a sorted list of sections based on the number of files in each section + * and the section name... + */ + + sections_files = cupsArrayNew((cups_array_func_t)compare_sections_files, + NULL); + for (section = (_cups_section_t *)cupsArrayFirst(sections); + section; + section = (_cups_section_t *)cupsArrayNext(sections)) + cupsArrayAdd(sections_files, section); + + /* + * Then build three columns to hold everything, trying to balance the number of + * lines in each column... + */ + + for (column = 0; column < 3; column ++) + { + columns[column] = cupsArrayNew((cups_array_func_t)compare_sections, NULL); + lines[column] = 0; + } + + for (section = (_cups_section_t *)cupsArrayFirst(sections_files); + section; + section = (_cups_section_t *)cupsArrayNext(sections_files)) + { + for (min_column = 0, min_lines = lines[0], column = 1; + column < 3; + column ++) + { + if (lines[column] < min_lines) + { + min_column = column; + min_lines = lines[column]; + } + } + + cupsArrayAdd(columns[min_column], section); + lines[min_column] += cupsArrayCount(section->files) + 2; + } + + /* + * Write the HTML file... + */ + + if ((fp = cupsFileOpen(path, "w")) == NULL) + { + fprintf(stderr, "makedocset: Unable to create %s: %s\n", path, + strerror(errno)); + exit(1); + } + + cupsFilePuts(fp, "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.01 " + "Transitional//EN\" " + "\"http://www.w3.org/TR/html4/loose.dtd\">\n" + "<html>\n" + "<head>\n" + "<title>CUPS Documentation\n" + "\n" + "\n" + "\n" + "

CUPS Documentation

\n" + "\n" + "\n"); + + for (column = 0; column < 3; column ++) + { + if (column) + cupsFilePuts(fp, "\n"); + + cupsFilePuts(fp, "\n"); + } + cupsFilePuts(fp, "\n" + "
     "); + for (section = (_cups_section_t *)cupsArrayFirst(columns[column]); + section; + section = (_cups_section_t *)cupsArrayNext(columns[column])) + { + cupsFilePrintf(fp, "

%s

\n", section->name); + for (html = (_cups_html_t *)cupsArrayFirst(section->files); + html; + html = (_cups_html_t *)cupsArrayNext(section->files)) + cupsFilePrintf(fp, "

%s

\n", + html->path, html->title); + } + cupsFilePuts(fp, "
\n" + "\n" + "\n"); + cupsFileClose(fp); +} + + +/* + * 'write_info()' - Write the Info.plist file. + */ + +static void +write_info(const char *path, /* I - File to write */ + const char *revision) /* I - Subversion revision number */ +{ + cups_file_t *fp; /* File */ + + + if ((fp = cupsFileOpen(path, "w")) == NULL) + { + fprintf(stderr, "makedocset: Unable to create %s: %s\n", path, + strerror(errno)); + exit(1); + } + + cupsFilePrintf(fp, "\n" + "\n" + "\n" + "\n" + "\tCFBundleIdentifier\n" + "\torg.cups.docset\n" + "\tCFBundleName\n" + "\tCUPS Documentation\n" + "\tCFBundleVersion\n" + "\t%d.%d.%s\n" + "\tCFBundleShortVersionString\n" + "\t%d.%d.%d\n" + "\tDocSetFeedName\n" + "\tcups.org\n" + "\tDocSetFeedURL\n" + "\thttp://www.cups.org/org.cups.docset.atom" + "\n" + "\tDocSetPublisherIdentifier\n" + "\torg.cups\n" + "\tDocSetPublisherName\n" + "\tCUPS\n" + "\n" + "\n", + CUPS_VERSION_MAJOR, CUPS_VERSION_MINOR, revision, + CUPS_VERSION_MAJOR, CUPS_VERSION_MINOR, CUPS_VERSION_PATCH); + + cupsFileClose(fp); +} + + +/* + * 'write_nodes()' - Write the Nodes.xml file. + */ + +static void +write_nodes(const char *path, /* I - File to write */ + help_index_t *hi) /* I - Index of files */ +{ + cups_file_t *fp; /* Output file */ + int id; /* Current node ID */ + help_node_t *node; /* Current help node */ + int subnodes; /* Currently in Subnodes for file? */ + int needclose; /* Need to close the current node? */ + + + if ((fp = cupsFileOpen(path, "w")) == NULL) + { + fprintf(stderr, "makedocset: Unable to create %s: %s\n", path, + strerror(errno)); + exit(1); + } + + cupsFilePuts(fp, "\n" + "\n" + "\n" + "\n" + "CUPS Documentation\n" + "Documentation/index.html\n" + "\n"); + + for (node = (help_node_t *)cupsArrayFirst(hi->nodes), id = 1, subnodes = 0, + needclose = 0; + node; + node = (help_node_t *)cupsArrayNext(hi->nodes), id ++) + { + if (node->anchor) + { + if (!subnodes) + { + cupsFilePuts(fp, "\n"); + subnodes = 1; + } + + cupsFilePrintf(fp, "\n" + "Documentation/%s\n" + "%s\n" + "%s\n" + "\n", id, node->filename, node->anchor, + node->text); + } + else + { + if (subnodes) + { + cupsFilePuts(fp, "\n"); + subnodes = 0; + } + + if (needclose) + cupsFilePuts(fp, "\n"); + + cupsFilePrintf(fp, "\n" + "Documentation/%s\n" + "%s\n", id, node->filename, node->text); + needclose = 1; + } + } + + if (subnodes) + cupsFilePuts(fp, "\n"); + + if (needclose) + cupsFilePuts(fp, "\n"); + + cupsFilePuts(fp, "\n" + "\n"); + + cupsFileClose(fp); +} + + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/multipart.dat b/cgi-bin/multipart.dat new file mode 100644 index 0000000000000000000000000000000000000000..91c5bb5b4c0e039c6692bc116be57983b836be68 GIT binary patch literal 50411 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 + + +/* + * Local functions... + */ + +static void do_printer_op(http_t *http, const char *printer, ipp_op_t op, + const char *title); +static void show_all_printers(http_t *http, const char *username); +static void show_printer(http_t *http, const char *printer); + + +/* + * 'main()' - Main entry for CGI. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + const char *printer; /* Printer name */ + const char *user; /* Username */ + http_t *http; /* Connection to the server */ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + ipp_attribute_t *attr; /* IPP attribute */ + const char *op; /* Operation to perform, if any */ + static const char *def_attrs[] = /* Attributes for default printer */ + { + "printer-name", + "printer-uri-supported" + }; + + + /* + * Get any form variables... + */ + + cgiInitialize(); + + op = cgiGetVariable("OP"); + + /* + * Set the web interface section... + */ + + cgiSetVariable("SECTION", "printers"); + cgiSetVariable("REFRESH_PAGE", ""); + + /* + * See if we are displaying a printer or all printers... + */ + + if ((printer = getenv("PATH_INFO")) != NULL) + { + printer ++; + + if (!*printer) + printer = NULL; + + if (printer) + cgiSetVariable("PRINTER_NAME", printer); + } + + /* + * See who is logged in... + */ + + user = getenv("REMOTE_USER"); + + /* + * Connect to the HTTP server... + */ + + http = httpConnectEncrypt(cupsServer(), ippPort(), cupsEncryption()); + + /* + * Get the default printer... + */ + + if (!op || !cgiIsPOST()) + { + /* + * Get the default destination... + */ + + request = ippNewRequest(CUPS_GET_DEFAULT); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", + sizeof(def_attrs) / sizeof(def_attrs[0]), NULL, def_attrs); + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + if ((attr = ippFindAttribute(response, "printer-name", IPP_TAG_NAME)) != NULL) + cgiSetVariable("DEFAULT_NAME", attr->values[0].string.text); + + if ((attr = ippFindAttribute(response, "printer-uri-supported", IPP_TAG_URI)) != NULL) + { + char url[HTTP_MAX_URI]; /* New URL */ + + + cgiSetVariable("DEFAULT_URI", + cgiRewriteURL(attr->values[0].string.text, + url, sizeof(url), NULL)); + } + + ippDelete(response); + } + + /* + * See if we need to show a list of printers or the status of a + * single printer... + */ + + if (!printer) + show_all_printers(http, user); + else + show_printer(http, printer); + } + else if (printer) + { + if (!*op) + { + const char *server_port = getenv("SERVER_PORT"); + /* Port number string */ + int port = atoi(server_port ? server_port : "0"); + /* Port number */ + char uri[1024]; /* URL */ + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), + getenv("HTTPS") ? "https" : "http", NULL, + getenv("SERVER_NAME"), port, "/printers/%s", printer); + + printf("Location: %s\n\n", uri); + } + else if (!strcmp(op, "start-printer")) + do_printer_op(http, printer, IPP_RESUME_PRINTER, + cgiText(_("Resume Printer"))); + else if (!strcmp(op, "stop-printer")) + do_printer_op(http, printer, IPP_PAUSE_PRINTER, + cgiText(_("Pause Printer"))); + else if (!strcmp(op, "accept-jobs")) + do_printer_op(http, printer, CUPS_ACCEPT_JOBS, cgiText(_("Accept Jobs"))); + else if (!strcmp(op, "reject-jobs")) + do_printer_op(http, printer, CUPS_REJECT_JOBS, cgiText(_("Reject Jobs"))); + else if (!strcmp(op, "purge-jobs")) + do_printer_op(http, printer, IPP_PURGE_JOBS, cgiText(_("Purge Jobs"))); + else if (!_cups_strcasecmp(op, "print-self-test-page")) + cgiPrintCommand(http, printer, "PrintSelfTestPage", + cgiText(_("Print Self-Test Page"))); + else if (!_cups_strcasecmp(op, "clean-print-heads")) + cgiPrintCommand(http, printer, "Clean all", + cgiText(_("Clean Print Heads"))); + else if (!_cups_strcasecmp(op, "print-test-page")) + cgiPrintTestPage(http, printer); + else if (!_cups_strcasecmp(op, "move-jobs")) + cgiMoveJobs(http, printer, 0); + else + { + /* + * Unknown/bad operation... + */ + + cgiStartHTML(printer); + cgiCopyTemplateLang("error-op.tmpl"); + cgiEndHTML(); + } + } + else + { + /* + * Unknown/bad operation... + */ + + cgiStartHTML(cgiText(_("Printers"))); + cgiCopyTemplateLang("error-op.tmpl"); + cgiEndHTML(); + } + + /* + * Close the HTTP server connection... + */ + + httpClose(http); + + /* + * Return with no errors... + */ + + return (0); +} + + +/* + * 'do_printer_op()' - Do a printer operation. + */ + +static void +do_printer_op(http_t *http, /* I - HTTP connection */ + const char *printer, /* I - Printer name */ + ipp_op_t op, /* I - Operation to perform */ + const char *title) /* I - Title of page */ +{ + ipp_t *request; /* IPP request */ + char uri[HTTP_MAX_URI], /* Printer URI */ + resource[HTTP_MAX_URI]; /* Path for request */ + + + /* + * Build a printer request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + */ + + request = ippNewRequest(op); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", printer); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + /* + * Do the request and get back a response... + */ + + snprintf(resource, sizeof(resource), "/printers/%s", printer); + ippDelete(cupsDoRequest(http, request, resource)); + + if (cupsLastError() == IPP_NOT_AUTHORIZED) + { + puts("Status: 401\n"); + exit(0); + } + else if (cupsLastError() > IPP_OK_CONFLICT) + { + cgiStartHTML(title); + cgiShowIPPError(_("Unable to do maintenance command:")); + } + else + { + /* + * Redirect successful updates back to the printer page... + */ + + char url[1024], /* Printer/class URL */ + refresh[1024]; /* Refresh URL */ + + + cgiRewriteURL(uri, url, sizeof(url), NULL); + cgiFormEncode(uri, url, sizeof(uri)); + snprintf(refresh, sizeof(refresh), "5;URL=%s", uri); + cgiSetVariable("refresh_page", refresh); + + cgiStartHTML(title); + + if (op == IPP_PAUSE_PRINTER) + cgiCopyTemplateLang("printer-stop.tmpl"); + else if (op == IPP_RESUME_PRINTER) + cgiCopyTemplateLang("printer-start.tmpl"); + else if (op == CUPS_ACCEPT_JOBS) + cgiCopyTemplateLang("printer-accept.tmpl"); + else if (op == CUPS_REJECT_JOBS) + cgiCopyTemplateLang("printer-reject.tmpl"); + else if (op == IPP_PURGE_JOBS) + cgiCopyTemplateLang("printer-purge.tmpl"); + } + + cgiEndHTML(); +} + + +/* + * 'show_all_printers()' - Show all printers... + */ + +static void +show_all_printers(http_t *http, /* I - Connection to server */ + const char *user) /* I - Username */ +{ + int i; /* Looping var */ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + cups_array_t *printers; /* Array of printer objects */ + ipp_attribute_t *printer; /* Printer object */ + int ascending, /* Order of printers (0 = descending) */ + first, /* First printer to show */ + count; /* Number of printers */ + const char *var; /* Form variable */ + void *search; /* Search data */ + char val[1024]; /* Form variable */ + + + fprintf(stderr, "DEBUG: show_all_printers(http=%p, user=\"%s\")\n", + http, user ? user : "(null)"); + + /* + * Show the standard header... + */ + + cgiStartHTML(cgiText(_("Printers"))); + + /* + * Build a CUPS_GET_PRINTERS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-type + * printer-type-mask + * requesting-user-name + */ + + request = ippNewRequest(CUPS_GET_PRINTERS); + + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_ENUM, + "printer-type", 0); + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_ENUM, + "printer-type-mask", CUPS_PRINTER_CLASS); + + if (user) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, user); + + cgiGetAttributes(request, "printers.tmpl"); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + /* + * Get a list of matching job objects. + */ + + if ((var = cgiGetVariable("QUERY")) != NULL && + !cgiGetVariable("CLEAR")) + search = cgiCompileSearch(var); + else + search = NULL; + + printers = cgiGetIPPObjects(response, search); + count = cupsArrayCount(printers); + + if (search) + cgiFreeSearch(search); + + /* + * Figure out which printers to display... + */ + + if ((var = cgiGetVariable("FIRST")) != NULL) + first = atoi(var); + else + first = 0; + + if (first >= count) + first = count - CUPS_PAGE_MAX; + + first = (first / CUPS_PAGE_MAX) * CUPS_PAGE_MAX; + + if (first < 0) + first = 0; + + sprintf(val, "%d", count); + cgiSetVariable("TOTAL", val); + + if ((var = cgiGetVariable("ORDER")) != NULL) + ascending = !_cups_strcasecmp(var, "asc"); + else + ascending = 1; + + if (ascending) + { + for (i = 0, printer = (ipp_attribute_t *)cupsArrayIndex(printers, first); + i < CUPS_PAGE_MAX && printer; + i ++, printer = (ipp_attribute_t *)cupsArrayNext(printers)) + cgiSetIPPObjectVars(printer, NULL, i); + } + else + { + for (i = 0, printer = (ipp_attribute_t *)cupsArrayIndex(printers, count - first - 1); + i < CUPS_PAGE_MAX && printer; + i ++, printer = (ipp_attribute_t *)cupsArrayPrev(printers)) + cgiSetIPPObjectVars(printer, NULL, i); + } + + /* + * Save navigation URLs... + */ + + cgiSetVariable("THISURL", "/printers/"); + + if (first > 0) + { + sprintf(val, "%d", first - CUPS_PAGE_MAX); + cgiSetVariable("PREV", val); + } + + if ((first + CUPS_PAGE_MAX) < count) + { + sprintf(val, "%d", first + CUPS_PAGE_MAX); + cgiSetVariable("NEXT", val); + } + + /* + * Then show everything... + */ + + cgiCopyTemplateLang("search.tmpl"); + + cgiCopyTemplateLang("printers-header.tmpl"); + + if (count > CUPS_PAGE_MAX) + cgiCopyTemplateLang("pager.tmpl"); + + cgiCopyTemplateLang("printers.tmpl"); + + if (count > CUPS_PAGE_MAX) + cgiCopyTemplateLang("pager.tmpl"); + + /* + * Delete the response... + */ + + cupsArrayDelete(printers); + ippDelete(response); + } + else + { + /* + * Show the error... + */ + + cgiShowIPPError(_("Unable to get printer list:")); + } + + cgiEndHTML(); +} + + +/* + * 'show_printer()' - Show a single printer. + */ + +static void +show_printer(http_t *http, /* I - Connection to server */ + const char *printer) /* I - Name of printer */ +{ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + ipp_attribute_t *attr; /* IPP attribute */ + char uri[HTTP_MAX_URI]; /* Printer URI */ + char refresh[1024]; /* Refresh URL */ + + + fprintf(stderr, "DEBUG: show_printer(http=%p, printer=\"%s\")\n", + http, printer ? printer : "(null)"); + + /* + * Build an IPP_GET_PRINTER_ATTRIBUTES request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + */ + + request = ippNewRequest(IPP_GET_PRINTER_ATTRIBUTES); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", printer); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", NULL, + uri); + + cgiGetAttributes(request, "printer.tmpl"); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + /* + * Got the result; set the CGI variables and check the status of a + * single-queue request... + */ + + cgiSetIPPVars(response, NULL, NULL, NULL, 0); + + if (printer && (attr = ippFindAttribute(response, "printer-state", + IPP_TAG_ENUM)) != NULL && + attr->values[0].integer == IPP_PRINTER_PROCESSING) + { + /* + * Printer is processing - automatically refresh the page until we + * are done printing... + */ + + cgiFormEncode(uri, printer, sizeof(uri)); + snprintf(refresh, sizeof(refresh), "10;URL=/printers/%s", uri); + cgiSetVariable("refresh_page", refresh); + } + + /* + * Delete the response... + */ + + ippDelete(response); + + /* + * Show the standard header... + */ + + cgiStartHTML(printer); + + /* + * Show the printer status... + */ + + cgiCopyTemplateLang("printer.tmpl"); + + /* + * Show jobs for the specified printer... + */ + + cgiCopyTemplateLang("printer-jobs-header.tmpl"); + cgiShowJobs(http, printer); + } + else + { + /* + * Show the IPP error... + */ + + cgiStartHTML(printer); + cgiShowIPPError(_("Unable to get printer status:")); + } + + cgiEndHTML(); +} + + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/search.c b/cgi-bin/search.c new file mode 100644 index 0000000000..e2a1c5c5d0 --- /dev/null +++ b/cgi-bin/search.c @@ -0,0 +1,381 @@ +/* + * "$Id$" + * + * Search routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cgiCompileSearch() - Compile a search string. + * cgiDoSearch() - Do a search of some text. + * cgiFreeSearch() - Free a compiled search context. + */ + +/* + * Include necessary headers... + */ + +#include "cgi-private.h" +#include + + +/* + * 'cgiCompileSearch()' - Compile a search string. + */ + +void * /* O - Search context */ +cgiCompileSearch(const char *query) /* I - Query string */ +{ + regex_t *re; /* Regular expression */ + char *s, /* Regular expression string */ + *sptr, /* Pointer into RE string */ + *sword; /* Pointer to start of word */ + int slen; /* Allocated size of RE string */ + const char *qptr, /* Pointer into query string */ + *qend; /* End of current word */ + const char *prefix; /* Prefix to add to next word */ + int quoted; /* Word is quoted */ + int wlen; /* Word length */ + char *lword; /* Last word in query */ + + + DEBUG_printf(("cgiCompileSearch(query=\"%s\")\n", query)); + + /* + * Range check input... + */ + + if (!query) + return (NULL); + + /* + * Allocate a regular expression storage structure... + */ + + if ((re = (regex_t *)calloc(1, sizeof(regex_t))) == NULL) + return (NULL); + + /* + * Allocate a buffer to hold the regular expression string, starting + * at 1024 bytes or 3 times the length of the query string, whichever + * is greater. We'll expand the string as needed... + */ + + slen = strlen(query) * 3; + if (slen < 1024) + slen = 1024; + + if ((s = (char *)malloc(slen)) == NULL) + { + free(re); + return (NULL); + } + + /* + * Copy the query string to the regular expression, handling basic + * AND and OR logic... + */ + + prefix = ".*"; + qptr = query; + sptr = s; + lword = NULL; + + while (*qptr) + { + /* + * Skip leading whitespace... + */ + + while (isspace(*qptr & 255)) + qptr ++; + + if (!*qptr) + break; + + /* + * Find the end of the current word... + */ + + if (*qptr == '\"' || *qptr == '\'') + { + /* + * Scan quoted string... + */ + + quoted = *qptr ++; + for (qend = qptr; *qend && *qend != quoted; qend ++); + + if (!*qend) + { + /* + * No closing quote, error out! + */ + + free(s); + free(re); + + if (lword) + free(lword); + + return (NULL); + } + } + else + { + /* + * Scan whitespace-delimited string... + */ + + quoted = 0; + for (qend = qptr + 1; *qend && !isspace(*qend); qend ++); + } + + wlen = qend - qptr; + + /* + * Look for logic words: AND, OR + */ + + if (wlen == 3 && !_cups_strncasecmp(qptr, "AND", 3)) + { + /* + * Logical AND with the following text... + */ + + if (sptr > s) + prefix = ".*"; + + qptr = qend; + } + else if (wlen == 2 && !_cups_strncasecmp(qptr, "OR", 2)) + { + /* + * Logical OR with the following text... + */ + + if (sptr > s) + prefix = ".*|.*"; + + qptr = qend; + } + else + { + /* + * Add a search word, making sure we have enough room for the + * string + RE overhead... + */ + + wlen = (sptr - s) + 2 * 4 * wlen + 2 * strlen(prefix) + 11; + if (lword) + wlen += strlen(lword); + + if (wlen > slen) + { + /* + * Expand the RE string buffer... + */ + + char *temp; /* Temporary string pointer */ + + + slen = wlen + 128; + temp = (char *)realloc(s, slen); + if (!temp) + { + free(s); + free(re); + + if (lword) + free(lword); + + return (NULL); + } + + sptr = temp + (sptr - s); + s = temp; + } + + /* + * Add the prefix string... + */ + + strcpy(sptr, prefix); + sptr += strlen(sptr); + + /* + * Then quote the remaining word characters as needed for the + * RE... + */ + + sword = sptr; + + while (qptr < qend) + { + /* + * Quote: ^ . [ $ ( ) | * + ? { \ + */ + + if (strchr("^.[$()|*+?{\\", *qptr)) + *sptr++ = '\\'; + + *sptr++ = *qptr++; + } + + *sptr = '\0'; + + /* + * For "word1 AND word2", add reciprocal "word2 AND word1"... + */ + + if (!strcmp(prefix, ".*") && lword) + { + char *lword2; /* New "last word" */ + + + if ((lword2 = strdup(sword)) == NULL) + { + free(lword); + free(s); + free(re); + return (NULL); + } + + strcpy(sptr, ".*|.*"); + sptr += 5; + + strcpy(sptr, lword2); + sptr += strlen(sptr); + + strcpy(sptr, ".*"); + sptr += 2; + + strcpy(sptr, lword); + sptr += strlen(sptr); + + free(lword); + lword = lword2; + } + else + { + if (lword) + free(lword); + + lword = strdup(sword); + } + + prefix = ".*|.*"; + } + + /* + * Advance to the next string... + */ + + if (quoted) + qptr ++; + } + + if (lword) + free(lword); + + if (sptr > s) + strcpy(sptr, ".*"); + else + { + /* + * No query data, return NULL... + */ + + free(s); + free(re); + + return (NULL); + } + + /* + * Compile the regular expression... + */ + + DEBUG_printf((" s=\"%s\"\n", s)); + + if (regcomp(re, s, REG_EXTENDED | REG_ICASE)) + { + free(re); + free(s); + + return (NULL); + } + + /* + * Free the RE string and return the new regular expression we compiled... + */ + + free(s); + + return ((void *)re); +} + + +/* + * 'cgiDoSearch()' - Do a search of some text. + */ + +int /* O - Number of matches */ +cgiDoSearch(void *search, /* I - Search context */ + const char *text) /* I - Text to search */ +{ + int i; /* Looping var */ + regmatch_t matches[100]; /* RE matches */ + + + /* + * Range check... + */ + + if (!search || !text) + return (0); + + /* + * Do a lookup... + */ + + if (!regexec((regex_t *)search, text, sizeof(matches) / sizeof(matches[0]), + matches, 0)) + { + /* + * Figure out the number of matches in the string... + */ + + for (i = 0; i < (int)(sizeof(matches) / sizeof(matches[0])); i ++) + if (matches[i].rm_so < 0) + break; + + return (i); + } + else + return (0); +} + + +/* + * 'cgiFreeSearch()' - Free a compiled search context. + */ + +void +cgiFreeSearch(void *search) /* I - Search context */ +{ + regfree((regex_t *)search); +} + + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/template.c b/cgi-bin/template.c new file mode 100644 index 0000000000..cdf1544bbe --- /dev/null +++ b/cgi-bin/template.c @@ -0,0 +1,733 @@ +/* + * "$Id$" + * + * CGI template function. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cgiCopyTemplateFile() - Copy a template file and replace all the + * '{variable}' strings with the variable value. + * cgiCopyTemplateLang() - Copy a template file using a language... + * cgiGetTemplateDir() - Get the templates directory... + * cgiSetServerVersion() - Set the server name and CUPS version... + * cgi_copy() - Copy the template file, substituting as needed... + * cgi_puts() - Put a string to the output file, quoting as + * needed... + */ + +#include "cgi-private.h" +#include +#include + + +/* + * Local functions... + */ + +static void cgi_copy(FILE *out, FILE *in, int element, char term, + int indent); +static void cgi_puts(const char *s, FILE *out); +static void cgi_puturi(const char *s, FILE *out); + + +/* + * 'cgiCopyTemplateFile()' - Copy a template file and replace all the + * '{variable}' strings with the variable value. + */ + +void +cgiCopyTemplateFile(FILE *out, /* I - Output file */ + const char *tmpl) /* I - Template file to read */ +{ + FILE *in; /* Input file */ + + + fprintf(stderr, "DEBUG2: cgiCopyTemplateFile(out=%p, tmpl=\"%s\")\n", out, + tmpl ? tmpl : "(null)"); + + /* + * Range check input... + */ + + if (!tmpl || !out) + return; + + /* + * Open the template file... + */ + + if ((in = fopen(tmpl, "r")) == NULL) + { + fprintf(stderr, "ERROR: Unable to open template file \"%s\" - %s\n", + tmpl ? tmpl : "(null)", strerror(errno)); + return; + } + + /* + * Parse the file to the end... + */ + + cgi_copy(out, in, 0, 0, 0); + + /* + * Close the template file and return... + */ + + fclose(in); +} + + +/* + * 'cgiCopyTemplateLang()' - Copy a template file using a language... + */ + +void +cgiCopyTemplateLang(const char *tmpl) /* I - Base filename */ +{ + char filename[1024], /* Filename */ + locale[16], /* Locale name */ + *locptr; /* Pointer into locale name */ + const char *directory, /* Directory for templates */ + *lang; /* Language */ + FILE *in; /* Input file */ + + + fprintf(stderr, "DEBUG2: cgiCopyTemplateLang(tmpl=\"%s\")\n", + tmpl ? tmpl : "(null)"); + + /* + * Convert the language to a locale name... + */ + + locale[0] = '\0'; + + if ((lang = getenv("LANG")) != NULL) + { + locale[0] = '/'; + strlcpy(locale + 1, lang, sizeof(locale) - 1); + + if ((locptr = strchr(locale, '.')) != NULL) + *locptr = '\0'; /* Strip charset */ + } + + fprintf(stderr, "DEBUG2: lang=\"%s\", locale=\"%s\"...\n", + lang ? lang : "(null)", locale); + + /* + * See if we have a template file for this language... + */ + + directory = cgiGetTemplateDir(); + + snprintf(filename, sizeof(filename), "%s%s/%s", directory, locale, tmpl); + if ((in = fopen(filename, "r")) == NULL) + { + locale[3] = '\0'; + + snprintf(filename, sizeof(filename), "%s%s/%s", directory, locale, tmpl); + if ((in = fopen(filename, "r")) == NULL) + { + snprintf(filename, sizeof(filename), "%s/%s", directory, tmpl); + in = fopen(filename, "r"); + } + } + + fprintf(stderr, "DEBUG2: Template file is \"%s\"...\n", filename); + + /* + * Open the template file... + */ + + if (!in) + { + fprintf(stderr, "ERROR: Unable to open template file \"%s\" - %s\n", + filename, strerror(errno)); + return; + } + + /* + * Parse the file to the end... + */ + + cgi_copy(stdout, in, 0, 0, 0); + + /* + * Close the template file and return... + */ + + fclose(in); +} + + +/* + * 'cgiGetTemplateDir()' - Get the templates directory... + */ + +char * /* O - Template directory */ +cgiGetTemplateDir(void) +{ + const char *datadir; /* CUPS_DATADIR env var */ + static char templates[1024] = ""; /* Template directory */ + + + if (!templates[0]) + { + /* + * Build the template directory pathname... + */ + + if ((datadir = getenv("CUPS_DATADIR")) == NULL) + datadir = CUPS_DATADIR; + + snprintf(templates, sizeof(templates), "%s/templates", datadir); + } + + return (templates); +} + + +/* + * 'cgiSetServerVersion()' - Set the server name and CUPS version... + */ + +void +cgiSetServerVersion(void) +{ + cgiSetVariable("SERVER_NAME", getenv("SERVER_NAME")); + cgiSetVariable("REMOTE_USER", getenv("REMOTE_USER")); + cgiSetVariable("CUPS_VERSION", CUPS_SVERSION); + +#ifdef LC_TIME + setlocale(LC_TIME, ""); +#endif /* LC_TIME */ +} + + +/* + * 'cgi_copy()' - Copy the template file, substituting as needed... + */ + +static void +cgi_copy(FILE *out, /* I - Output file */ + FILE *in, /* I - Input file */ + int element, /* I - Element number (0 to N) */ + char term, /* I - Terminating character */ + int indent) /* I - Debug info indentation */ +{ + int ch; /* Character from file */ + char op; /* Operation */ + char name[255], /* Name of variable */ + *nameptr, /* Pointer into name */ + innername[255], /* Inner comparison name */ + *innerptr, /* Pointer into inner name */ + *s; /* String pointer */ + const char *value; /* Value of variable */ + const char *innerval; /* Inner value */ + const char *outptr; /* Output string pointer */ + char outval[1024], /* Formatted output string */ + compare[1024]; /* Comparison string */ + int result; /* Result of comparison */ + int uriencode; /* Encode as URI */ + regex_t re; /* Regular expression to match */ + + + fprintf(stderr, "DEBUG2: %*sStarting at file position %ld...\n", indent, "", + ftell(in)); + + /* + * Parse the file to the end... + */ + + while ((ch = getc(in)) != EOF) + if (ch == term) + break; + else if (ch == '{') + { + /* + * Get a variable name... + */ + + uriencode = 0; + + for (s = name; (ch = getc(in)) != EOF;) + if (strchr("}]<>=!~ \t\n", ch)) + break; + else if (s == name && ch == '%') + uriencode = 1; + else if (s > name && ch == '?') + break; + else if (s < (name + sizeof(name) - 1)) + *s++ = ch; + + *s = '\0'; + + if (s == name && isspace(ch & 255)) + { + fprintf(stderr, "DEBUG2: %*sLone { at %ld...\n", indent, "", ftell(in)); + + if (out) + { + putc('{', out); + putc(ch, out); + } + + continue; + } + + if (ch == '}') + fprintf(stderr, "DEBUG2: %*s\"{%s}\" at %ld...\n", indent, "", name, + ftell(in)); + + /* + * See if it has a value... + */ + + if (name[0] == '?') + { + /* + * Insert value only if it exists... + */ + + if ((nameptr = strrchr(name, '-')) != NULL && isdigit(nameptr[1] & 255)) + { + *nameptr++ = '\0'; + + if ((value = cgiGetArray(name + 1, atoi(nameptr) - 1)) != NULL) + outptr = value; + else + { + outval[0] = '\0'; + outptr = outval; + } + } + else if ((value = cgiGetArray(name + 1, element)) != NULL) + outptr = value; + else + { + outval[0] = '\0'; + outptr = outval; + } + } + else if (name[0] == '#') + { + /* + * Insert count... + */ + + if (name[1]) + sprintf(outval, "%d", cgiGetSize(name + 1)); + else + sprintf(outval, "%d", element + 1); + + outptr = outval; + } + else if (name[0] == '[') + { + /* + * Loop for # of elements... + */ + + int i; /* Looping var */ + long pos; /* File position */ + int count; /* Number of elements */ + + + if (isdigit(name[1] & 255)) + count = atoi(name + 1); + else + count = cgiGetSize(name + 1); + + pos = ftell(in); + + fprintf(stderr, "DEBUG2: %*sLooping on \"%s\" at %ld, count=%d...\n", + indent, "", name + 1, pos, count); + + if (count > 0) + { + for (i = 0; i < count; i ++) + { + if (i) + fseek(in, pos, SEEK_SET); + + cgi_copy(out, in, i, '}', indent + 2); + } + } + else + cgi_copy(NULL, in, 0, '}', indent + 2); + + fprintf(stderr, "DEBUG2: %*sFinished looping on \"%s\"...\n", indent, + "", name + 1); + + continue; + } + else if (name[0] == '$') + { + /* + * Insert cookie value or nothing if not defined. + */ + + if ((value = cgiGetCookie(name + 1)) != NULL) + outptr = value; + else + { + outval[0] = '\0'; + outptr = outval; + } + } + else + { + /* + * Insert variable or variable name (if element is NULL)... + */ + + if ((nameptr = strrchr(name, '-')) != NULL && isdigit(nameptr[1] & 255)) + { + *nameptr++ = '\0'; + if ((value = cgiGetArray(name, atoi(nameptr) - 1)) == NULL) + { + snprintf(outval, sizeof(outval), "{%s}", name); + outptr = outval; + } + else + outptr = value; + } + else if ((value = cgiGetArray(name, element)) == NULL) + { + snprintf(outval, sizeof(outval), "{%s}", name); + outptr = outval; + } + else + outptr = value; + } + + /* + * See if the terminating character requires another test... + */ + + if (ch == '}') + { + /* + * End of substitution... + */ + + if (out) + { + if (uriencode) + cgi_puturi(outptr, out); + else if (!_cups_strcasecmp(name, "?cupsdconf_default")) + fputs(outptr, stdout); + else + cgi_puts(outptr, out); + } + + continue; + } + + /* + * OK, process one of the following checks: + * + * {name?exist:not-exist} Exists? + * {name=value?true:false} Equal + * {namevalue?true:false} Greater than + * {name!value?true:false} Not equal + * {name~refex?true:false} Regex match + */ + + op = ch; + + if (ch == '?') + { + /* + * Test for existance... + */ + + if (name[0] == '?') + result = cgiGetArray(name + 1, element) != NULL; + else if (name[0] == '#') + result = cgiGetVariable(name + 1) != NULL; + else + result = cgiGetArray(name, element) != NULL; + + result = result && outptr[0]; + compare[0] = '\0'; + } + else + { + /* + * Compare to a string... + */ + + for (s = compare; (ch = getc(in)) != EOF;) + if (ch == '?') + break; + else if (s >= (compare + sizeof(compare) - 1)) + continue; + else if (ch == '#') + { + sprintf(s, "%d", element + 1); + s += strlen(s); + } + else if (ch == '{') + { + /* + * Grab the value of a variable... + */ + + innerptr = innername; + while ((ch = getc(in)) != EOF && ch != '}') + if (innerptr < (innername + sizeof(innername) - 1)) + *innerptr++ = ch; + *innerptr = '\0'; + + if (innername[0] == '#') + sprintf(s, "%d", cgiGetSize(innername + 1)); + else if ((innerptr = strrchr(innername, '-')) != NULL && + isdigit(innerptr[1] & 255)) + { + *innerptr++ = '\0'; + if ((innerval = cgiGetArray(innername, atoi(innerptr) - 1)) == NULL) + *s = '\0'; + else + strlcpy(s, innerval, sizeof(compare) - (s - compare)); + } + else if (innername[0] == '?') + { + if ((innerval = cgiGetArray(innername + 1, element)) == NULL) + *s = '\0'; + else + strlcpy(s, innerval, sizeof(compare) - (s - compare)); + } + else if ((innerval = cgiGetArray(innername, element)) == NULL) + snprintf(s, sizeof(compare) - (s - compare), "{%s}", innername); + else + strlcpy(s, innerval, sizeof(compare) - (s - compare)); + + s += strlen(s); + } + else if (ch == '\\') + *s++ = getc(in); + else + *s++ = ch; + + *s = '\0'; + + if (ch != '?') + { + fprintf(stderr, + "DEBUG2: %*sBad terminator '%c' at file position %ld...\n", + indent, "", ch, ftell(in)); + return; + } + + /* + * Do the comparison... + */ + + switch (op) + { + case '<' : + result = _cups_strcasecmp(outptr, compare) < 0; + break; + case '>' : + result = _cups_strcasecmp(outptr, compare) > 0; + break; + case '=' : + result = _cups_strcasecmp(outptr, compare) == 0; + break; + case '!' : + result = _cups_strcasecmp(outptr, compare) != 0; + break; + case '~' : + fprintf(stderr, "DEBUG: Regular expression \"%s\"\n", compare); + + if (regcomp(&re, compare, REG_EXTENDED | REG_ICASE)) + { + fprintf(stderr, + "ERROR: Unable to compile regular expresion \"%s\"!\n", + compare); + result = 0; + } + else + { + regmatch_t matches[10]; + + result = 0; + + if (!regexec(&re, outptr, 10, matches, 0)) + { + int i; + for (i = 0; i < 10; i ++) + { + fprintf(stderr, "DEBUG: matches[%d].rm_so=%d\n", i, + (int)matches[i].rm_so); + if (matches[i].rm_so < 0) + break; + + result ++; + } + } + + regfree(&re); + } + break; + default : + result = 1; + break; + } + } + + fprintf(stderr, + "DEBUG2: %*sStarting \"{%s%c%s\" at %ld, result=%d...\n", + indent, "", name, op, compare, ftell(in), result); + + if (result) + { + /* + * Comparison true; output first part and ignore second... + */ + + fprintf(stderr, "DEBUG2: %*sOutput first part...\n", indent, ""); + cgi_copy(out, in, element, ':', indent + 2); + + fprintf(stderr, "DEBUG2: %*sSkip second part...\n", indent, ""); + cgi_copy(NULL, in, element, '}', indent + 2); + } + else + { + /* + * Comparison false; ignore first part and output second... + */ + + fprintf(stderr, "DEBUG2: %*sSkip first part...\n", indent, ""); + cgi_copy(NULL, in, element, ':', indent + 2); + + fprintf(stderr, "DEBUG2: %*sOutput second part...\n", indent, ""); + cgi_copy(out, in, element, '}', indent + 2); + } + + fprintf(stderr, "DEBUG2: %*sFinished \"{%s%c%s\", out=%p...\n", indent, "", + name, op, compare, out); + } + else if (ch == '\\') /* Quoted char */ + { + if (out) + putc(getc(in), out); + else + getc(in); + } + else if (out) + putc(ch, out); + + if (ch == EOF) + fprintf(stderr, "DEBUG2: %*sReturning at file position %ld on EOF...\n", + indent, "", ftell(in)); + else + fprintf(stderr, + "DEBUG2: %*sReturning at file position %ld on character '%c'...\n", + indent, "", ftell(in), ch); + + if (ch == EOF && term) + fprintf(stderr, "ERROR: %*sSaw EOF, expected '%c'!\n", indent, "", term); + + /* + * Flush any pending output... + */ + + if (out) + fflush(out); +} + + +/* + * 'cgi_puts()' - Put a string to the output file, quoting as needed... + */ + +static void +cgi_puts(const char *s, /* I - String to output */ + FILE *out) /* I - Output file */ +{ + while (*s) + { + if (*s == '<') + { + /* + * Pass and , otherwise quote it... + */ + + if (!_cups_strncasecmp(s, "", out); + } + else if (!_cups_strncasecmp(s, "", 4)) + { + fputs("", out); + s += 3; + } + else + fputs("<", out); + } + else if (*s == '>') + fputs(">", out); + else if (*s == '\"') + fputs(""", out); + else if (*s == '\'') + fputs("'", out); + else if (*s == '&') + fputs("&", out); + else + putc(*s, out); + + s ++; + } +} + + +/* + * 'cgi_puturi()' - Put a URI string to the output file, quoting as needed... + */ + +static void +cgi_puturi(const char *s, /* I - String to output */ + FILE *out) /* I - Output file */ +{ + while (*s) + { + if (strchr("%@&+ <>#=", *s) || *s < ' ' || *s & 128) + fprintf(out, "%%%02X", *s & 255); + else + putc(*s, out); + + s ++; + } +} + + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/testcgi.c b/cgi-bin/testcgi.c new file mode 100644 index 0000000000..dfda3d7ea9 --- /dev/null +++ b/cgi-bin/testcgi.c @@ -0,0 +1,75 @@ +/* + * "$Id$" + * + * CGI test program for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Test the help index code. + * list_nodes() - List nodes in an array... + */ + +/* + * Include necessary headers... + */ + +#include "cgi.h" + + +/* + * 'main()' - Test the CGI code. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + /* + * Test file upload/multi-part submissions... + */ + + freopen("multipart.dat", "rb", stdin); + + putenv("CONTENT_TYPE=multipart/form-data; " + "boundary=---------------------------1977426492562745908748943111"); + putenv("REQUEST_METHOD=POST"); + + printf("cgiInitialize: "); + if (cgiInitialize()) + { + const cgi_file_t *file; /* Upload file */ + + if ((file = cgiGetFile()) != NULL) + { + puts("PASS"); + printf(" tempfile=\"%s\"\n", file->tempfile); + printf(" name=\"%s\"\n", file->name); + printf(" filename=\"%s\"\n", file->filename); + printf(" mimetype=\"%s\"\n", file->mimetype); + } + else + puts("FAIL (no file!)"); + } + else + puts("FAIL (init)"); + + /* + * Return with no errors... + */ + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/testhi.c b/cgi-bin/testhi.c new file mode 100644 index 0000000000..ce1b94b014 --- /dev/null +++ b/cgi-bin/testhi.c @@ -0,0 +1,113 @@ +/* + * "$Id$" + * + * Help index test program for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Test the help index code. + * list_nodes() - List nodes in an array... + */ + +/* + * Include necessary headers... + */ + +#include "cgi.h" + + +/* + * Local functions... + */ + +static void list_nodes(const char *title, cups_array_t *nodes); + + +/* + * 'main()' - Test the help index code. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + help_index_t *hi, /* Help index */ + *search; /* Search index */ + + + /* + * Load the help index... + */ + + hi = helpLoadIndex("testhi.index", "."); + + list_nodes("nodes", hi->nodes); + list_nodes("sorted", hi->sorted); + + /* + * Do any searches... + */ + + if (argc > 1) + { + search = helpSearchIndex(hi, argv[1], NULL, argv[2]); + + if (search) + { + list_nodes(argv[1], search->sorted); + helpDeleteIndex(search); + } + else + printf("%s (0 nodes)\n", argv[1]); + } + + helpDeleteIndex(hi); + + /* + * Return with no errors... + */ + + return (0); +} + + +/* + * 'list_nodes()' - List nodes in an array... + */ + +static void +list_nodes(const char *title, /* I - Title string */ + cups_array_t *nodes) /* I - Nodes */ +{ + int i; /* Looping var */ + help_node_t *node; /* Current node */ + + + printf("%s (%d nodes):\n", title, cupsArrayCount(nodes)); + for (i = 1, node = (help_node_t *)cupsArrayFirst(nodes); + node; + i ++, node = (help_node_t *)cupsArrayNext(nodes)) + { + if (node->anchor) + printf(" %d: %s#%s \"%s\"", i, node->filename, node->anchor, + node->text); + else + printf(" %d: %s \"%s\"", i, node->filename, node->text); + + printf(" (%d words)\n", cupsArrayCount(node->words)); + } +} + + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/testhi.html b/cgi-bin/testhi.html new file mode 100644 index 0000000000..0000e9eff8 --- /dev/null +++ b/cgi-bin/testhi.html @@ -0,0 +1,31 @@ + + + Test File for Help Index Code + + + +

This is a test file for the help index code. The help index +code reads plain HTML and indexes the title and any anchored +text, ignoring all other markup. Anchor tags must be on a single +line, although the text they wrap may cross multiple lines and be +up to 1024 bytes in length.

+ +

This is the First Anchor

+ +

This is some text for the first anchor.

+ + +

This is the Second Anchor

+ +

This is some text for the first anchor.

+ +

John asked Mary to the dance.

+ + +

This is the Third Anchor

+ +

This is some text for the third anchor. This +is an in-line anchor that crosses a line.

+ + + diff --git a/cgi-bin/testtemplate.c b/cgi-bin/testtemplate.c new file mode 100644 index 0000000000..02c0173658 --- /dev/null +++ b/cgi-bin/testtemplate.c @@ -0,0 +1,103 @@ +/* + * "$Id$" + * + * CGI template test program for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Test the template code. + */ + +/* + * Include necessary headers... + */ + +#include "cgi.h" + + +/* + * 'main()' - Test the template code. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + char *value; /* Value in name=value */ + FILE *out; /* Where to send output */ + + + /* + * Don't buffer stdout or stderr so that the mixed output is sane... + */ + + setbuf(stdout, NULL); + setbuf(stderr, NULL); + + /* + * Loop through the command-line, assigning variables for any args with + * "name=value"... + */ + + out = stdout; + + for (i = 1; i < argc; i ++) + { + if (!strcmp(argv[i], "-o")) + { + i ++; + if (i < argc) + { + out = fopen(argv[i], "w"); + if (!out) + { + perror(argv[i]); + return (1); + } + } + } + else if (!strcmp(argv[i], "-e")) + { + i ++; + + if (i < argc) + { + if (!freopen(argv[i], "w", stderr)) + { + perror(argv[i]); + return (1); + } + } + } + else if (!strcmp(argv[i], "-q")) + freopen("/dev/null", "w", stderr); + else if ((value = strchr(argv[i], '=')) != NULL) + { + *value++ = '\0'; + cgiSetVariable(argv[i], value); + } + else + cgiCopyTemplateFile(out, argv[i]); + } + + /* + * Return with no errors... + */ + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/var.c b/cgi-bin/var.c new file mode 100644 index 0000000000..803677fe9a --- /dev/null +++ b/cgi-bin/var.c @@ -0,0 +1,1305 @@ +/* + * "$Id$" + * + * CGI form variable and array functions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cgiCheckVariables() - Check for the presence of "required" + * variables. + * cgiClearVariables() - Clear all form variables. + * cgiGetArray() - Get an element from a form array. + * cgiGetCookie() - Get a cookie value. + * cgiGetFile() - Get the file (if any) that was submitted in + * the form. + * cgiGetSize() - Get the size of a form array value. + * cgiGetVariable() - Get a CGI variable from the database. + * cgiInitialize() - Initialize the CGI variable "database". + * cgiIsPOST() - Determine whether this page was POSTed. + * cgiSetArray() - Set array element N to the specified string. + * cgiSetCookie() - Set a cookie value. + * cgiSetSize() - Set the array size. + * cgiSetVariable() - Set a CGI variable in the database. + * cgi_add_variable() - Add a form variable. + * cgi_compare_variables() - Compare two variables. + * cgi_find_variable() - Find a variable. + * cgi_initialize_cookies() - Initialize cookies. + * cgi_initialize_get() - Initialize form variables using the GET + * method. + * cgi_initialize_multipart() - Initialize variables and file using the POST + * method. + * cgi_initialize_post() - Initialize variables using the POST method. + * cgi_initialize_string() - Initialize form variables from a string. + * cgi_passwd() - Catch authentication requests and notify the + * server. + * cgi_set_sid() - Set the CUPS session ID. + * cgi_sort_variables() - Sort all form variables for faster lookup. + * cgi_unlink_file() - Remove the uploaded form. + */ + +/*#define DEBUG*/ +#include "cgi-private.h" +#include +#include + + +/* + * Session ID name + */ + +#define CUPS_SID "org.cups.sid" + + +/* + * Data structure to hold all the CGI form variables and arrays... + */ + +typedef struct /**** Form variable structure ****/ +{ + const char *name; /* Name of variable */ + int nvalues, /* Number of values */ + avalues; /* Number of values allocated */ + const char **values; /* Value(s) of variable */ +} _cgi_var_t; + + +/* + * Local globals... + */ + +static int num_cookies = 0;/* Number of cookies */ +static cups_option_t *cookies = NULL;/* Cookies */ +static int form_count = 0, /* Form variable count */ + form_alloc = 0; /* Number of variables allocated */ +static _cgi_var_t *form_vars = NULL; + /* Form variables */ +static cgi_file_t *form_file = NULL; + /* Uploaded file */ + + +/* + * Local functions... + */ + +static void cgi_add_variable(const char *name, int element, + const char *value); +static int cgi_compare_variables(const _cgi_var_t *v1, + const _cgi_var_t *v2); +static _cgi_var_t *cgi_find_variable(const char *name); +static void cgi_initialize_cookies(void); +static int cgi_initialize_get(void); +static int cgi_initialize_multipart(const char *boundary); +static int cgi_initialize_post(void); +static int cgi_initialize_string(const char *data); +static const char *cgi_passwd(const char *prompt); +static const char *cgi_set_sid(void); +static void cgi_sort_variables(void); +static void cgi_unlink_file(void); + + +/* + * 'cgiCheckVariables()' - Check for the presence of "required" variables. + * + * Names may be separated by spaces and/or commas. + */ + +int /* O - 1 if all variables present, 0 otherwise */ +cgiCheckVariables(const char *names) /* I - Variables to look for */ +{ + char name[255], /* Current variable name */ + *s; /* Pointer in string */ + const char *val; /* Value of variable */ + int element; /* Array element number */ + + + if (names == NULL) + return (1); + + while (*names != '\0') + { + while (*names == ' ' || *names == ',') + names ++; + + for (s = name; *names != '\0' && *names != ' ' && *names != ','; s ++, names ++) + *s = *names; + + *s = 0; + if (name[0] == '\0') + break; + + if ((s = strrchr(name, '-')) != NULL) + { + *s = '\0'; + element = atoi(s + 1) - 1; + val = cgiGetArray(name, element); + } + else + val = cgiGetVariable(name); + + if (val == NULL) + return (0); + + if (*val == '\0') + return (0); /* Can't be blank, either! */ + } + + return (1); +} + + +/* + * 'cgiClearVariables()' - Clear all form variables. + */ + +void +cgiClearVariables(void) +{ + int i, j; /* Looping vars */ + _cgi_var_t *v; /* Current variable */ + + + for (v = form_vars, i = form_count; i > 0; v ++, i --) + { + _cupsStrFree(v->name); + for (j = 0; j < v->nvalues; j ++) + if (v->values[j]) + _cupsStrFree(v->values[j]); + } + + form_count = 0; + + cgi_unlink_file(); +} + + +/* + * 'cgiGetArray()' - Get an element from a form array. + */ + +const char * /* O - Element value or NULL */ +cgiGetArray(const char *name, /* I - Name of array variable */ + int element) /* I - Element number (0 to N) */ +{ + _cgi_var_t *var; /* Pointer to variable */ + + + if ((var = cgi_find_variable(name)) == NULL) + return (NULL); + + if (element < 0 || element >= var->nvalues) + return (NULL); + + return (_cupsStrRetain(var->values[element])); +} + + +/* + * 'cgiGetCookie()' - Get a cookie value. + */ + +const char * /* O - Value or NULL */ +cgiGetCookie(const char *name) /* I - Name of cookie */ +{ + return (cupsGetOption(name, num_cookies, cookies)); +} + + +/* + * 'cgiGetFile()' - Get the file (if any) that was submitted in the form. + */ + +const cgi_file_t * /* O - Attached file or NULL */ +cgiGetFile(void) +{ + return (form_file); +} + + +/* + * 'cgiGetSize()' - Get the size of a form array value. + */ + +int /* O - Number of elements */ +cgiGetSize(const char *name) /* I - Name of variable */ +{ + _cgi_var_t *var; /* Pointer to variable */ + + + if ((var = cgi_find_variable(name)) == NULL) + return (0); + + return (var->nvalues); +} + + +/* + * 'cgiGetVariable()' - Get a CGI variable from the database. + * + * Returns NULL if the variable doesn't exist. If the variable is an + * array of values, returns the last element. + */ + +const char * /* O - Value of variable */ +cgiGetVariable(const char *name) /* I - Name of variable */ +{ + const _cgi_var_t *var; /* Returned variable */ + + + var = cgi_find_variable(name); + +#ifdef DEBUG + if (var == NULL) + DEBUG_printf(("cgiGetVariable(\"%s\") is returning NULL...\n", name)); + else + DEBUG_printf(("cgiGetVariable(\"%s\") is returning \"%s\"...\n", name, + var->values[var->nvalues - 1])); +#endif /* DEBUG */ + + return ((var == NULL) ? NULL : _cupsStrRetain(var->values[var->nvalues - 1])); +} + + +/* + * 'cgiInitialize()' - Initialize the CGI variable "database". + */ + +int /* O - Non-zero if there was form data */ +cgiInitialize(void) +{ + const char *method, /* Form posting method */ + *content_type, /* Content-Type of post data */ + *cups_sid_cookie, /* SID cookie */ + *cups_sid_form; /* SID form variable */ + + + /* + * Setup a password callback for authentication... + */ + + cupsSetPasswordCB(cgi_passwd); + + /* + * Set the locale so that times, etc. are formatted properly... + */ + + setlocale(LC_ALL, ""); + +#ifdef DEBUG + /* + * Disable output buffering to find bugs... + */ + + setbuf(stdout, NULL); +#endif /* DEBUG */ + + /* + * Get cookies... + */ + + cgi_initialize_cookies(); + + if ((cups_sid_cookie = cgiGetCookie(CUPS_SID)) == NULL) + { + fputs("DEBUG: " CUPS_SID " cookie not found, initializing!\n", stderr); + cups_sid_cookie = cgi_set_sid(); + } + + fprintf(stderr, "DEBUG: " CUPS_SID " cookie is \"%s\"\n", cups_sid_cookie); + + /* + * Get the request method (GET or POST)... + */ + + method = getenv("REQUEST_METHOD"); + content_type = getenv("CONTENT_TYPE"); + if (!method) + return (0); + + /* + * Grab form data from the corresponding location... + */ + + if (!_cups_strcasecmp(method, "GET")) + return (cgi_initialize_get()); + else if (!_cups_strcasecmp(method, "POST") && content_type) + { + const char *boundary = strstr(content_type, "boundary="); + + if (boundary) + boundary += 9; + + if (content_type && !strncmp(content_type, "multipart/form-data; ", 21)) + { + if (!cgi_initialize_multipart(boundary)) + return (0); + } + else if (!cgi_initialize_post()) + return (0); + + if ((cups_sid_form = cgiGetVariable(CUPS_SID)) == NULL || + strcmp(cups_sid_cookie, cups_sid_form)) + { + if (cups_sid_form) + fprintf(stderr, "DEBUG: " CUPS_SID " form variable is \"%s\"\n", + cups_sid_form); + else + fputs("DEBUG: " CUPS_SID " form variable is not present.\n", stderr); + + cgiClearVariables(); + return (0); + } + else + return (1); + } + else + return (0); +} + + +/* + * 'cgiIsPOST()' - Determine whether this page was POSTed. + */ + +int /* O - 1 if POST, 0 if GET */ +cgiIsPOST(void) +{ + const char *method; /* REQUEST_METHOD environment variable */ + + + if ((method = getenv("REQUEST_METHOD")) == NULL) + return (0); + else + return (!strcmp(method, "POST")); +} + + +/* + * 'cgiSetArray()' - Set array element N to the specified string. + * + * If the variable array is smaller than (element + 1), the intervening + * elements are set to NULL. + */ + +void +cgiSetArray(const char *name, /* I - Name of variable */ + int element, /* I - Element number (0 to N) */ + const char *value) /* I - Value of variable */ +{ + int i; /* Looping var */ + _cgi_var_t *var; /* Returned variable */ + + + if (name == NULL || value == NULL || element < 0 || element > 100000) + return; + + if ((var = cgi_find_variable(name)) == NULL) + { + cgi_add_variable(name, element, value); + cgi_sort_variables(); + } + else + { + if (element >= var->avalues) + { + const char **temp; /* Temporary pointer */ + + temp = (const char **)realloc((void *)(var->values), + sizeof(char *) * (element + 16)); + if (!temp) + return; + + var->avalues = element + 16; + var->values = temp; + } + + if (element >= var->nvalues) + { + for (i = var->nvalues; i < element; i ++) + var->values[i] = NULL; + + var->nvalues = element + 1; + } + else if (var->values[element]) + _cupsStrFree((char *)var->values[element]); + + var->values[element] = _cupsStrAlloc(value); + } +} + + +/* + * 'cgiSetCookie()' - Set a cookie value. + */ + +void +cgiSetCookie(const char *name, /* I - Name */ + const char *value, /* I - Value */ + const char *path, /* I - Path (typically "/") */ + const char *domain, /* I - Domain name */ + time_t expires, /* I - Expiration date (0 for session) */ + int secure) /* I - Require SSL */ +{ + num_cookies = cupsAddOption(name, value, num_cookies, &cookies); + + printf("Set-Cookie: %s=%s;", name, value); + if (path) + printf(" path=%s;", path); + if (domain) + printf(" domain=%s;", domain); + if (expires) + { + char date[256]; /* Date string */ + + printf(" expires=%s;", httpGetDateString2(expires, date, sizeof(date))); + } + if (secure) + puts(" secure;"); + else + putchar('\n'); +} + + +/* + * 'cgiSetSize()' - Set the array size. + */ + +void +cgiSetSize(const char *name, /* I - Name of variable */ + int size) /* I - Number of elements (0 to N) */ +{ + int i; /* Looping var */ + _cgi_var_t *var; /* Returned variable */ + + + if (name == NULL || size < 0 || size > 100000) + return; + + if ((var = cgi_find_variable(name)) == NULL) + return; + + if (size >= var->avalues) + { + const char **temp; /* Temporary pointer */ + + temp = (const char **)realloc((void *)(var->values), + sizeof(char *) * (size + 16)); + if (!temp) + return; + + var->avalues = size + 16; + var->values = temp; + } + + if (size > var->nvalues) + { + for (i = var->nvalues; i < size; i ++) + var->values[i] = NULL; + } + else if (size < var->nvalues) + { + for (i = size; i < var->nvalues; i ++) + if (var->values[i]) + _cupsStrFree((void *)(var->values[i])); + } + + var->nvalues = size; +} + + +/* + * 'cgiSetVariable()' - Set a CGI variable in the database. + * + * If the variable is an array, this truncates the array to a single element. + */ + +void +cgiSetVariable(const char *name, /* I - Name of variable */ + const char *value) /* I - Value of variable */ +{ + int i; /* Looping var */ + _cgi_var_t *var; /* Returned variable */ + + + if (name == NULL || value == NULL) + return; + + if ((var = cgi_find_variable(name)) == NULL) + { + cgi_add_variable(name, 0, value); + cgi_sort_variables(); + } + else + { + for (i = 0; i < var->nvalues; i ++) + if (var->values[i]) + _cupsStrFree((char *)var->values[i]); + + var->values[0] = _cupsStrAlloc(value); + var->nvalues = 1; + } +} + + +/* + * 'cgi_add_variable()' - Add a form variable. + */ + +static void +cgi_add_variable(const char *name, /* I - Variable name */ + int element, /* I - Array element number */ + const char *value) /* I - Variable value */ +{ + _cgi_var_t *var; /* New variable */ + + + if (name == NULL || value == NULL || element < 0 || element > 100000) + return; + + DEBUG_printf(("cgi_add_variable: Adding variable \'%s\' with value " + "\'%s\'...\n", name, value)); + + if (form_count >= form_alloc) + { + _cgi_var_t *temp_vars; /* Temporary form pointer */ + + + if (form_alloc == 0) + temp_vars = malloc(sizeof(_cgi_var_t) * 16); + else + temp_vars = realloc(form_vars, (form_alloc + 16) * sizeof(_cgi_var_t)); + + if (!temp_vars) + return; + + form_vars = temp_vars; + form_alloc += 16; + } + + var = form_vars + form_count; + + if ((var->values = calloc(element + 1, sizeof(char *))) == NULL) + return; + + var->name = _cupsStrAlloc(name); + var->nvalues = element + 1; + var->avalues = element + 1; + var->values[element] = _cupsStrAlloc(value); + + form_count ++; +} + + +/* + * 'cgi_compare_variables()' - Compare two variables. + */ + +static int /* O - Result of comparison */ +cgi_compare_variables( + const _cgi_var_t *v1, /* I - First variable */ + const _cgi_var_t *v2) /* I - Second variable */ +{ + return (_cups_strcasecmp(v1->name, v2->name)); +} + + +/* + * 'cgi_find_variable()' - Find a variable. + */ + +static _cgi_var_t * /* O - Variable pointer or NULL */ +cgi_find_variable(const char *name) /* I - Name of variable */ +{ + _cgi_var_t key; /* Search key */ + + + if (form_count < 1 || name == NULL) + return (NULL); + + key.name = name; + + return ((_cgi_var_t *)bsearch(&key, form_vars, form_count, sizeof(_cgi_var_t), + (int (*)(const void *, const void *))cgi_compare_variables)); +} + + +/* + * 'cgi_initialize_cookies()' - Initialize cookies. + */ + +static void +cgi_initialize_cookies(void) +{ + const char *cookie; /* HTTP_COOKIE environment variable */ + char name[128], /* Name string */ + value[512], /* Value string */ + *ptr; /* Pointer into name/value */ + + + if ((cookie = getenv("HTTP_COOKIE")) == NULL) + return; + + while (*cookie) + { + /* + * Skip leading whitespace... + */ + + while (isspace(*cookie & 255)) + cookie ++; + if (!*cookie) + break; + + /* + * Copy the name... + */ + + for (ptr = name; *cookie && *cookie != '=';) + if (ptr < (name + sizeof(name) - 1)) + *ptr++ = *cookie++; + else + break; + + if (*cookie != '=') + break; + + *ptr = '\0'; + cookie ++; + + /* + * Then the value... + */ + + if (*cookie == '\"') + { + for (cookie ++, ptr = value; *cookie && *cookie != '\"';) + if (ptr < (value + sizeof(value) - 1)) + *ptr++ = *cookie++; + else + break; + + if (*cookie == '\"') + cookie ++; + } + else + { + for (ptr = value; *cookie && *cookie != ';';) + if (ptr < (value + sizeof(value) - 1)) + *ptr++ = *cookie++; + else + break; + } + + if (*cookie == ';') + cookie ++; + else if (*cookie) + break; + + *ptr = '\0'; + + /* + * Then add the cookie to an array as long as the name doesn't start with + * "$"... + */ + + if (name[0] != '$') + num_cookies = cupsAddOption(name, value, num_cookies, &cookies); + } +} + + +/* + * 'cgi_initialize_get()' - Initialize form variables using the GET method. + */ + +static int /* O - 1 if form data read */ +cgi_initialize_get(void) +{ + char *data; /* Pointer to form data string */ + + + DEBUG_puts("cgi_initialize_get: Initializing variables using GET method..."); + + /* + * Check to see if there is anything for us to read... + */ + + data = getenv("QUERY_STRING"); + if (data == NULL || strlen(data) == 0) + return (0); + + /* + * Parse it out and return... + */ + + return (cgi_initialize_string(data)); +} + + +/* + * 'cgi_initialize_multipart()' - Initialize variables and file using the POST + * method. + * + * TODO: Update to support files > 2GB. + */ + +static int /* O - 1 if form data was read */ +cgi_initialize_multipart( + const char *boundary) /* I - Boundary string */ +{ + char line[10240], /* MIME header line */ + name[1024], /* Form variable name */ + filename[1024], /* Form filename */ + mimetype[1024], /* MIME media type */ + bstring[256], /* Boundary string to look for */ + *ptr, /* Pointer into name/filename */ + *end; /* End of buffer */ + int ch, /* Character from file */ + fd, /* Temporary file descriptor */ + blen; /* Length of boundary string */ + + + DEBUG_printf(("cgi_initialize_multipart(boundary=\"%s\")\n", boundary)); + + /* + * Read multipart form data until we run out... + */ + + name[0] = '\0'; + filename[0] = '\0'; + mimetype[0] = '\0'; + + snprintf(bstring, sizeof(bstring), "\r\n--%s", boundary); + blen = strlen(bstring); + + while (fgets(line, sizeof(line), stdin)) + { + if (!strcmp(line, "\r\n")) + { + /* + * End of headers, grab value... + */ + + if (filename[0]) + { + /* + * Read an embedded file... + */ + + if (form_file) + { + /* + * Remove previous file... + */ + + cgi_unlink_file(); + } + + /* + * Allocate memory for the new file... + */ + + if ((form_file = calloc(1, sizeof(cgi_file_t))) == NULL) + return (0); + + form_file->name = strdup(name); + form_file->filename = strdup(filename); + form_file->mimetype = strdup(mimetype); + + fd = cupsTempFd(form_file->tempfile, sizeof(form_file->tempfile)); + + if (fd < 0) + return (0); + + atexit(cgi_unlink_file); + + /* + * Copy file data to the temp file... + */ + + ptr = line; + + while ((ch = getchar()) != EOF) + { + *ptr++ = ch; + + if ((ptr - line) >= blen && !memcmp(ptr - blen, bstring, blen)) + { + ptr -= blen; + break; + } + + if ((ptr - line - blen) >= 8192) + { + /* + * Write out the first 8k of the buffer... + */ + + write(fd, line, 8192); + memmove(line, line + 8192, ptr - line - 8192); + ptr -= 8192; + } + } + + /* + * Write the rest of the data and close the temp file... + */ + + if (ptr > line) + write(fd, line, ptr - line); + + close(fd); + } + else + { + /* + * Just get a form variable; the current code only handles + * form values up to 10k in size... + */ + + ptr = line; + end = line + sizeof(line) - 1; + + while ((ch = getchar()) != EOF) + { + if (ptr < end) + *ptr++ = ch; + + if ((ptr - line) >= blen && !memcmp(ptr - blen, bstring, blen)) + { + ptr -= blen; + break; + } + } + + *ptr = '\0'; + + /* + * Set the form variable... + */ + + if ((ptr = strrchr(name, '-')) != NULL && isdigit(ptr[1] & 255)) + { + /* + * Set a specific index in the array... + */ + + *ptr++ = '\0'; + if (line[0]) + cgiSetArray(name, atoi(ptr) - 1, line); + } + else if (cgiGetVariable(name)) + { + /* + * Add another element in the array... + */ + + cgiSetArray(name, cgiGetSize(name), line); + } + else + { + /* + * Just set the line... + */ + + cgiSetVariable(name, line); + } + } + + /* + * Read the rest of the current line... + */ + + fgets(line, sizeof(line), stdin); + + /* + * Clear the state vars... + */ + + name[0] = '\0'; + filename[0] = '\0'; + mimetype[0] = '\0'; + } + else if (!_cups_strncasecmp(line, "Content-Disposition:", 20)) + { + if ((ptr = strstr(line + 20, " name=\"")) != NULL) + { + strlcpy(name, ptr + 7, sizeof(name)); + + if ((ptr = strchr(name, '\"')) != NULL) + *ptr = '\0'; + } + + if ((ptr = strstr(line + 20, " filename=\"")) != NULL) + { + strlcpy(filename, ptr + 11, sizeof(filename)); + + if ((ptr = strchr(filename, '\"')) != NULL) + *ptr = '\0'; + } + } + else if (!_cups_strncasecmp(line, "Content-Type:", 13)) + { + for (ptr = line + 13; isspace(*ptr & 255); ptr ++); + + strlcpy(mimetype, ptr, sizeof(mimetype)); + + for (ptr = mimetype + strlen(mimetype) - 1; + ptr > mimetype && isspace(*ptr & 255); + *ptr-- = '\0'); + } + } + + /* + * Return 1 for "form data found"... + */ + + return (1); +} + + +/* + * 'cgi_initialize_post()' - Initialize variables using the POST method. + */ + +static int /* O - 1 if form data was read */ +cgi_initialize_post(void) +{ + char *content_length, /* Length of input data (string) */ + *data; /* Pointer to form data string */ + int length, /* Length of input data */ + nbytes, /* Number of bytes read this read() */ + tbytes, /* Total number of bytes read */ + status; /* Return status */ + + + DEBUG_puts("cgi_initialize_post: Initializing variables using POST method..."); + + /* + * Check to see if there is anything for us to read... + */ + + content_length = getenv("CONTENT_LENGTH"); + if (content_length == NULL || atoi(content_length) <= 0) + return (0); + + /* + * Get the length of the input stream and allocate a buffer for it... + */ + + length = atoi(content_length); + data = malloc(length + 1); + + if (data == NULL) + return (0); + + /* + * Read the data into the buffer... + */ + + for (tbytes = 0; tbytes < length; tbytes += nbytes) + if ((nbytes = read(0, data + tbytes, length - tbytes)) < 0) + { + if (errno != EAGAIN) + { + free(data); + return (0); + } + else + nbytes = 0; + } + else if (nbytes == 0) + { + /* + * CUPS STR #3176: OpenBSD: Early end-of-file on POST data causes 100% CPU + * + * This should never happen, but does on OpenBSD. If we see early end-of- + * file, treat this as an error and process no data. + */ + + free(data); + return (0); + } + + data[length] = '\0'; + + /* + * Parse it out... + */ + + status = cgi_initialize_string(data); + + /* + * Free the data and return... + */ + + free(data); + + return (status); +} + + +/* + * 'cgi_initialize_string()' - Initialize form variables from a string. + */ + +static int /* O - 1 if form data was processed */ +cgi_initialize_string(const char *data) /* I - Form data string */ +{ + int done; /* True if we're done reading a form variable */ + char *s, /* Pointer to current form string */ + ch, /* Temporary character */ + name[255], /* Name of form variable */ + value[65536]; /* Variable value */ + + + /* + * Check input... + */ + + if (data == NULL) + return (0); + + /* + * Loop until we've read all the form data... + */ + + while (*data != '\0') + { + /* + * Get the variable name... + */ + + for (s = name; *data != '\0'; data ++) + if (*data == '=') + break; + else if (*data >= ' ' && s < (name + sizeof(name) - 1)) + *s++ = *data; + + *s = '\0'; + if (*data == '=') + data ++; + else + return (0); + + /* + * Read the variable value... + */ + + for (s = value, done = 0; !done && *data != '\0'; data ++) + switch (*data) + { + case '&' : /* End of data... */ + done = 1; + break; + + case '+' : /* Escaped space character */ + if (s < (value + sizeof(value) - 1)) + *s++ = ' '; + break; + + case '%' : /* Escaped control character */ + /* + * Read the hex code... + */ + + if (!isxdigit(data[1] & 255) || !isxdigit(data[2] & 255)) + return (0); + + if (s < (value + sizeof(value) - 1)) + { + data ++; + ch = *data - '0'; + if (ch > 9) + ch -= 7; + *s = ch << 4; + + data ++; + ch = *data - '0'; + if (ch > 9) + ch -= 7; + *s++ |= ch; + } + else + data += 2; + break; + + default : /* Other characters come straight through */ + if (*data >= ' ' && s < (value + sizeof(value) - 1)) + *s++ = *data; + break; + } + + *s = '\0'; /* nul terminate the string */ + + /* + * Remove trailing whitespace... + */ + + if (s > value) + s --; + + while (s >= value && isspace(*s & 255)) + *s-- = '\0'; + + /* + * Add the string to the variable "database"... + */ + + if ((s = strrchr(name, '-')) != NULL && isdigit(s[1] & 255)) + { + *s++ = '\0'; + if (value[0]) + cgiSetArray(name, atoi(s) - 1, value); + } + else if (cgiGetVariable(name) != NULL) + cgiSetArray(name, cgiGetSize(name), value); + else + cgiSetVariable(name, value); + } + + return (1); +} + + +/* + * 'cgi_passwd()' - Catch authentication requests and notify the server. + * + * This function sends a Status header and exits, forcing authentication + * for this request. + */ + +static const char * /* O - NULL (no return) */ +cgi_passwd(const char *prompt) /* I - Prompt (not used) */ +{ + (void)prompt; + + fprintf(stderr, "DEBUG: cgi_passwd(prompt=\"%s\") called!\n", + prompt ? prompt : "(null)"); + + /* + * Send a 401 (unauthorized) status to the server, so it can notify + * the client that authentication is required. + */ + + puts("Status: 401\n"); + exit(0); + + /* + * This code is never executed, but is present to satisfy the compiler. + */ + + return (NULL); +} + + +/* + * 'cgi_set_sid()' - Set the CUPS session ID. + */ + +static const char * /* O - New session ID */ +cgi_set_sid(void) +{ + char buffer[512], /* SID data */ + sid[33]; /* SID string */ + _cups_md5_state_t md5; /* MD5 state */ + unsigned char sum[16]; /* MD5 sum */ + const char *remote_addr, /* REMOTE_ADDR */ + *server_name, /* SERVER_NAME */ + *server_port; /* SERVER_PORT */ + + + if ((remote_addr = getenv("REMOTE_ADDR")) == NULL) + remote_addr = "REMOTE_ADDR"; + if ((server_name = getenv("SERVER_NAME")) == NULL) + server_name = "SERVER_NAME"; + if ((server_port = getenv("SERVER_PORT")) == NULL) + server_port = "SERVER_PORT"; + + CUPS_SRAND(time(NULL)); + snprintf(buffer, sizeof(buffer), "%s:%s:%s:%02X%02X%02X%02X%02X%02X%02X%02X", + remote_addr, server_name, server_port, + (unsigned)CUPS_RAND() & 255, (unsigned)CUPS_RAND() & 255, + (unsigned)CUPS_RAND() & 255, (unsigned)CUPS_RAND() & 255, + (unsigned)CUPS_RAND() & 255, (unsigned)CUPS_RAND() & 255, + (unsigned)CUPS_RAND() & 255, (unsigned)CUPS_RAND() & 255); + _cupsMD5Init(&md5); + _cupsMD5Append(&md5, (unsigned char *)buffer, (int)strlen(buffer)); + _cupsMD5Finish(&md5, sum); + + cgiSetCookie(CUPS_SID, httpMD5String(sum, sid), "/", NULL, 0, 0); + + return (cupsGetOption(CUPS_SID, num_cookies, cookies)); +} + + +/* + * 'cgi_sort_variables()' - Sort all form variables for faster lookup. + */ + +static void +cgi_sort_variables(void) +{ +#ifdef DEBUG + int i; + + + DEBUG_puts("cgi_sort_variables: Sorting variables..."); +#endif /* DEBUG */ + + if (form_count < 2) + return; + + qsort(form_vars, form_count, sizeof(_cgi_var_t), + (int (*)(const void *, const void *))cgi_compare_variables); + +#ifdef DEBUG + DEBUG_puts("cgi_sort_variables: Sorted variable list is:"); + for (i = 0; i < form_count; i ++) + DEBUG_printf(("cgi_sort_variables: %d: %s (%d) = \"%s\" ...\n", i, + form_vars[i].name, form_vars[i].nvalues, + form_vars[i].values[0])); +#endif /* DEBUG */ +} + + +/* + * 'cgi_unlink_file()' - Remove the uploaded form. + */ + +static void +cgi_unlink_file(void) +{ + if (form_file) + { + /* + * Remove the temporary file... + */ + + unlink(form_file->tempfile); + + /* + * Free memory used... + */ + + free(form_file->name); + free(form_file->filename); + free(form_file->mimetype); + free(form_file); + + form_file = NULL; + } +} + + +/* + * End of "$Id$". + */ diff --git a/cgi-bin/websearch.c b/cgi-bin/websearch.c new file mode 100644 index 0000000000..351f92db08 --- /dev/null +++ b/cgi-bin/websearch.c @@ -0,0 +1,116 @@ +/* + * "$Id$" + * + * Web search program for www.cups.org. + * + * Copyright 2007-2009 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Usage: + * + * websearch directory "search string" + * + * Contents: + * + * main() - Search a directory of help files. + * list_nodes() - List matching nodes. + */ + +/* + * Include necessary headers... + */ + +#include "cgi.h" + + +/* + * Local functions... + */ + +static void list_nodes(help_index_t *hi, const char *title, + cups_array_t *nodes); + + +/* + * 'main()' - Test the help index code. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + help_index_t *hi, /* Help index */ + *search; /* Search index */ + char indexname[1024]; /* Name of index file */ + + + if (argc != 3) + { + puts("Usage: websearch directory \"search terms\""); + return (1); + } + + /* + * Load the help index... + */ + + snprintf(indexname, sizeof(indexname), "%s/.index", argv[1]); + hi = helpLoadIndex(indexname, argv[1]); + + /* + * Do any searches... + */ + + search = helpSearchIndex(hi, argv[2], NULL, NULL); + + if (search) + list_nodes(hi, argv[1], search->sorted); + + /* + * Return with no errors... + */ + + return (0); +} + + +/* + * 'list_nodes()' - List nodes in an array... + */ + +static void +list_nodes(help_index_t *hi, /* I - Help index */ + const char *title, /* I - Title string */ + cups_array_t *nodes) /* I - Nodes */ +{ + help_node_t *node, /* Current node */ + *file; /* File node */ + + + printf("%d\n", cupsArrayCount(nodes)); + for (node = (help_node_t *)cupsArrayFirst(nodes); + node; + node = (help_node_t *)cupsArrayNext(nodes)) + { + if (node->anchor) + { + file = helpFindNode(hi, node->filename, NULL); + printf("%d|%s#%s|%s|%s\n", node->score, node->filename, node->anchor, + node->text, file ? file->text : node->filename); + } + else + printf("%d|%s|%s|%s\n", node->score, node->filename, node->text, + node->text); + } +} + + +/* + * End of "$Id$". + */ diff --git a/conf/Makefile b/conf/Makefile new file mode 100644 index 0000000000..b35b593f78 --- /dev/null +++ b/conf/Makefile @@ -0,0 +1,143 @@ +# +# "$Id$" +# +# Configuration file makefile for CUPS. +# +# Copyright 2007-2011 by Apple Inc. +# Copyright 1993-2006 by Easy Software Products. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +include ../Makedefs + +# +# Config files... +# + +KEEP = cupsd.conf snmp.conf +REPLACE = mime.convs mime.types + + +# +# Make everything... +# + +all: + + +# +# Make library targets... +# + +libs: + + +# +# Make unit tests... +# + +unittests: + + +# +# Clean all config and object files... +# + +clean: + + +# +# Dummy depend... +# + +depend: + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: + for file in $(KEEP); do \ + if test -r $(SERVERROOT)/$$file ; then \ + $(INSTALL_CONFIG) -g $(CUPS_GROUP) $$file $(SERVERROOT)/$$file.N ; \ + else \ + $(INSTALL_CONFIG) -g $(CUPS_GROUP) $$file $(SERVERROOT) ; \ + fi ; \ + done + $(INSTALL_CONFIG) -g $(CUPS_GROUP) cupsd.conf $(SERVERROOT)/cupsd.conf.default + $(INSTALL_DIR) -m 755 $(DATADIR)/mime + for file in $(REPLACE); do \ + if test -r $(DATADIR)/mime/$$file ; then \ + $(MV) $(DATADIR)/mime/$$file $(DATADIR)/mime/$$file.O ; \ + fi ; \ + if test -r $(SERVERROOT)/$$file ; then \ + $(MV) $(SERVERROOT)/$$file $(DATADIR)/mime/$$file.O ; \ + fi ; \ + $(INSTALL_DATA) $$file $(DATADIR)/mime ; \ + done + -if test x$(PAMDIR) != x; then \ + $(INSTALL_DIR) -m 755 $(BUILDROOT)$(PAMDIR); \ + if test -r $(BUILDROOT)$(PAMDIR)/cups ; then \ + $(INSTALL_DATA) $(PAMFILE) $(BUILDROOT)$(PAMDIR)/cups.N ; \ + else \ + $(INSTALL_DATA) $(PAMFILE) $(BUILDROOT)$(PAMDIR)/cups ; \ + fi ; \ + fi + + +# +# Install programs... +# + +install-exec: + + +# +# Install headers... +# + +install-headers: + + +# +# Install libraries... +# + +install-libs: + + +# +# Uninstall files... +# + +uninstall: + for file in $(KEEP) $(REPLACE) cupsd.conf.default; do \ + $(RM) $(SERVERROOT)/$$file; \ + done + -$(RMDIR) $(SERVERROOT) + for file in $(REPLACE); do \ + $(RM) $(DATADIR)/mime/$$file; \ + done + -$(RMDIR) $(DATADIR)/mime + -if test x$(PAMDIR) != x; then \ + $(RM) $(BUILDROOT)$(PAMDIR)/cups; \ + $(RMDIR) $(BUILDROOT)$(PAMDIR); \ + fi + + +# +# End of "$Id$". +# diff --git a/conf/cupsd.conf.in b/conf/cupsd.conf.in new file mode 100644 index 0000000000..2ce33e3d79 --- /dev/null +++ b/conf/cupsd.conf.in @@ -0,0 +1,140 @@ +# +# "$Id$" +# +# Sample configuration file for the CUPS scheduler. See "man cupsd.conf" for a +# complete description of this file. +# + +# Log general information in error_log - change "@CUPS_LOG_LEVEL@" to "debug" +# for troubleshooting... +LogLevel @CUPS_LOG_LEVEL@ + +# Administrator user group... +SystemGroup @CUPS_SYSTEM_GROUPS@ +@CUPS_SYSTEM_AUTHKEY@ + +# Only listen for connections from the local machine. +Listen localhost:@DEFAULT_IPP_PORT@ +@CUPS_LISTEN_DOMAINSOCKET@ + +# Show shared printers on the local network. +Browsing On +BrowseOrder allow,deny +BrowseAllow all +BrowseLocalProtocols @CUPS_BROWSE_LOCAL_PROTOCOLS@ + +# Default authentication type, when authentication is required... +DefaultAuthType Basic + +# Web interface setting... +WebInterface @CUPS_WEBIF@ + +# Restrict access to the server... + + Order allow,deny + + +# Restrict access to the admin pages... + + Order allow,deny + + +# Restrict access to configuration files... + + AuthType Default + Require user @SYSTEM + Order allow,deny + + +# Set the default printer/job policies... + + # Job/subscription privacy... + JobPrivateAccess default + JobPrivateValues default + SubscriptionPrivateAccess default + SubscriptionPrivateValues default + + # Job-related operations must be done by the owner or an administrator... + + Order deny,allow + + + + Require user @OWNER @SYSTEM + Order deny,allow + + + # All administration operations require an administrator to authenticate... + + AuthType Default + Require user @SYSTEM + Order deny,allow + + + # All printer operations require a printer operator to authenticate... + + AuthType Default + Require user @CUPS_DEFAULT_PRINTOPERATOR_AUTH@ + Order deny,allow + + + # Only the owner or an administrator can cancel or authenticate a job... + + Require user @OWNER @CUPS_DEFAULT_PRINTOPERATOR_AUTH@ + Order deny,allow + + + + Order deny,allow + + + +# Set the authenticated printer/job policies... + + # Job/subscription privacy... + JobPrivateAccess default + JobPrivateValues default + SubscriptionPrivateAccess default + SubscriptionPrivateValues default + + # Job-related operations must be done by the owner or an administrator... + + AuthType Default + Order deny,allow + + + + AuthType Default + Require user @OWNER @SYSTEM + Order deny,allow + + + # All administration operations require an administrator to authenticate... + + AuthType Default + Require user @SYSTEM + Order deny,allow + + + # All printer operations require a printer operator to authenticate... + + AuthType Default + Require user @CUPS_DEFAULT_PRINTOPERATOR_AUTH@ + Order deny,allow + + + # Only the owner or an administrator can cancel or authenticate a job... + + AuthType Default + Require user @OWNER @CUPS_DEFAULT_PRINTOPERATOR_AUTH@ + Order deny,allow + + + + Order deny,allow + + + +# +# End of "$Id$". +# diff --git a/conf/mime.convs.in b/conf/mime.convs.in new file mode 100644 index 0000000000..f460000dd1 --- /dev/null +++ b/conf/mime.convs.in @@ -0,0 +1,63 @@ +# +# "$Id$" +# +# DO NOT EDIT THIS FILE, AS IT IS OVERWRITTEN WHEN YOU INSTALL NEW +# VERSIONS OF CUPS. Instead, create a "local.convs" file that +# reflects your local configuration changes. +# +# Base MIME conversions file for CUPS. +# +# Copyright 2007-2011 by Apple Inc. +# Copyright 1997-2007 by Easy Software Products. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +######################################################################## +# +# Format of Lines: +# +# source/type destination/type cost filter +# +# General Notes: +# +# The "cost" field is used to find the least costly filters to run +# when converting a job file to a printable format. +# +# All filters *must* accept the standard command-line arguments +# (job-id, user, title, copies, options, [filename or stdin]) to +# work with CUPS. +# + +######################################################################## +# +# PostScript filters +# + +application/postscript application/vnd.cups-postscript 66 pstops + +######################################################################## +# +# Raster filters... +# + +# PWG Raster filter for IPP Everywhere... +application/vnd.cups-raster image/pwg-raster 100 rastertopwg + +######################################################################## +# +# Raw filter... +# +# Uncomment the following filter to allow printing of arbitrary files +# without the -oraw option. +# + +@DEFAULT_RAW_PRINTING@application/octet-stream application/vnd.cups-raw 0 - + +# +# End of "$Id$". +# diff --git a/conf/mime.types b/conf/mime.types new file mode 100644 index 0000000000..21ec3bd04e --- /dev/null +++ b/conf/mime.types @@ -0,0 +1,174 @@ +# +# "$Id: mime.types 9871 2011-08-06 06:34:46Z mike $" +# +# Base MIME types file for CUPS. +# +# DO NOT EDIT THIS FILE, AS IT IS OVERWRITTEN WHEN YOU INSTALL NEW +# VERSIONS OF CUPS. Instead, create a "local.types" file that +# reflects your local configuration changes. +# +# Copyright 2007-2011 by Apple Inc. +# Copyright 1997-2007 by Easy Software Products. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +######################################################################## +# +# Format of Lines: +# +# super/type rules +# +# "rules" can be any combination of: +# +# ( expr ) Parenthesis for expression grouping +# + Logical AND +# , or whitespace Logical OR +# ! Logical NOT +# match("pattern") Pattern match on filename +# extension Pattern match on "*.extension" +# ascii(offset,length) True if bytes are valid printable ASCII +# (CR, NL, TAB, BS, 32-126) +# priority(number) Sets priority of type (0=lowest, +# 100=default, 200=highest) +# printable(offset,length) True if bytes are printable 8-bit chars +# (CR, NL, TAB, BS, 32-126, 128-254) +# string(offset,"string") True if bytes are identical to string +# istring(offset,"string") True if bytes are identical to +# case-insensitive string +# char(offset,value) True if byte is identical +# short(offset,value) True if 16-bit integer is identical +# int(offset,value) True if 32-bit integer is identical +# locale("string") True if current locale matches string +# contains(offset,range,"string") True if the range contains the string +# +# General Notes: +# +# MIME type names are case-insensitive. Internally they are converted +# to lowercase. Multiple occurrences of a type will cause the provided +# rules to be appended to the existing definition. If two types use the same +# rules to resolve a type and have the same priority, e.g. "doc" extension for +# "text/bar" and "text/foo", the returned type will be the first type as +# sorted in alphanumerically ascending order without regard to case. Thus, +# the "text/bar" type will match the "doc" extension first unless the +# "text/foo" type has specified a higher priority. +# +# The "printable" rule differs from the "ascii" rule in that it also +# accepts 8-bit characters in the range 128-255. +# +# String constants must be surrounded by "" if they contain whitespace. +# To insert binary data into a string, use the notation. +# + +######################################################################## +# +# Application-generated files... +# + +#application/msword doc string(0,) +application/pdf pdf string(0,%PDF) +application/postscript ai eps ps string(0,%!) string(0,<04>%!) \ + contains(0,128,<1B>%-12345X) + \ + (contains(0,4096,"LANGUAGE=POSTSCRIPT") \ + contains(0,4096,"LANGUAGE = Postscript") \ + contains(0,4096,"LANGUAGE = PostScript") \ + contains(0,4096,"LANGUAGE = POSTSCRIPT") \ + (contains(0,4096,<0a>%!) + \ + !contains(0,4096,"ENTER LANGUAGE"))) + +######################################################################## +# +# Image files... +# + +image/gif gif string(0,GIF87a) string(0,GIF89a) +image/png png string(0,<89>PNG) +image/jpeg jpeg jpg jpe string(0,) +\ + (char(3,0xe0) char(3,0xe1) char(3,0xe2) char(3,0xe3)\ + char(3,0xe4) char(3,0xe5) char(3,0xe6) char(3,0xe7)\ + char(3,0xe8) char(3,0xe9) char(3,0xea) char(3,0xeb)\ + char(3,0xec) char(3,0xed) char(3,0xee) char(3,0xef)) +image/pwg-raster string(0,"RaS2") + string(4,PwgRaster<00>) priority(100) +image/tiff tiff tif string(0,MM<002A>) string(0,II<2A00>) +image/x-photocd pcd string(2048,PCD_IPI) +image/x-portable-anymap pnm +image/x-portable-bitmap pbm string(0,P1) string(0,P4) +image/x-portable-graymap pgm string(0,P2) string(0,P5) +image/x-portable-pixmap ppm string(0,P3) string(0,P6) +image/x-sgi-rgb rgb sgi bw icon short(0,474) +image/x-xbitmap xbm +image/x-xpixmap xpm ascii(0,1024) + string(3,"XPM") +#image/x-xwindowdump xwd string(4,<00000007>) +image/x-sun-raster ras string(0,<59a66a95>) + +#image/fpx fpx +image/x-alias pix short(8,8) short(8,24) +image/x-bitmap bmp string(0,BM) + !printable(2,14) +image/x-icon ico + +######################################################################## +# +# Text files... +# + +application/x-cshell csh printable(0,1024) + string(0,#!) +\ + (contains(2,80,/csh) contains(2,80,/tcsh)) +application/x-perl pl printable(0,1024) + string(0,#!) +\ + contains(2,80,/perl) +application/x-shell sh printable(0,1024) + string(0,#!) +\ + (contains(2,80,/bash) contains(2,80,/ksh)\ + contains(2,80,/sh) contains(2,80,/zsh)) +application/x-csource c cxx cpp cc C h hpp \ + printable(0,1024) + \ + (string(0,/*) string(0,//) + string(0,#include) contains(0,1024,<0a>#include) \ + string(0,#define) contains(0,1024,<0a>#define)) +text/html html htm printable(0,1024) +\ + (istring(0,"") istring(0,"E) + !string(2,<1B>%0B)) \ + string(0,<1B>@) \ + (contains(0,128,<1B>%-12345X) + \ + (contains(0,4096,"LANGUAGE=PCL") \ + contains(0,4096,"LANGUAGE = PCL"))) + +######################################################################## +# +# Raw print file support... +# +# Comment the following type to prevent raw file printing. +# + +application/octet-stream + +# +# End of "$Id: mime.types 9871 2011-08-06 06:34:46Z mike $". +# diff --git a/conf/pam.opendirectory b/conf/pam.opendirectory new file mode 100644 index 0000000000..b336306a88 --- /dev/null +++ b/conf/pam.opendirectory @@ -0,0 +1,5 @@ +# cups: auth account password session +auth required pam_opendirectory.so +account required pam_permit.so +password required pam_deny.so +session required pam_permit.so diff --git a/conf/pam.securityserver b/conf/pam.securityserver new file mode 100644 index 0000000000..ff724da4d6 --- /dev/null +++ b/conf/pam.securityserver @@ -0,0 +1,7 @@ +# cups: auth account password session +auth sufficient pam_securityserver.so +auth sufficient pam_unix.so +auth required pam_deny.so +account required pam_permit.so +password required pam_deny.so +session required pam_permit.so diff --git a/conf/pam.std.in b/conf/pam.std.in new file mode 100644 index 0000000000..68fb50af3d --- /dev/null +++ b/conf/pam.std.in @@ -0,0 +1,2 @@ +auth required @PAMMODAUTH@ +account required @PAMMOD@ diff --git a/conf/snmp.conf.in b/conf/snmp.conf.in new file mode 100644 index 0000000000..43855ab66e --- /dev/null +++ b/conf/snmp.conf.in @@ -0,0 +1,13 @@ +# +# "$Id$" +# +# Sample SNMP configuration file for CUPS. See "man cups-snmp.conf" for a +# complete description of this file. +# + +@CUPS_SNMP_ADDRESS@ +@CUPS_SNMP_COMMUNITY@ + +# +# End of "$Id$". +# diff --git a/config-scripts/cups-common.m4 b/config-scripts/cups-common.m4 new file mode 100644 index 0000000000..5ab10b3c0d --- /dev/null +++ b/config-scripts/cups-common.m4 @@ -0,0 +1,453 @@ +dnl +dnl "$Id: cups-common.m4 10192 2012-01-20 21:49:02Z mike $" +dnl +dnl Common configuration stuff for CUPS. +dnl +dnl Copyright 2007-2011 by Apple Inc. +dnl Copyright 1997-2007 by Easy Software Products, all rights reserved. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +dnl We need at least autoconf 2.60... +AC_PREREQ(2.60) + +dnl Set the name of the config header file... +AC_CONFIG_HEADER(config.h) + +dnl Version number information... +CUPS_VERSION="1.6svn" +CUPS_REVISION="" +if test -z "$CUPS_REVISION" -a -d .svn; then + CUPS_REVISION="-r`svnversion . | awk -F: '{print $NF}' | sed -e '1,$s/[[a-zA-Z]]*//g'`" +fi +CUPS_BUILD="cups-$CUPS_VERSION" + +AC_ARG_WITH(cups_build, [ --with-cups-build set "cups-config --build" string ], + CUPS_BUILD="$withval") + +AC_SUBST(CUPS_VERSION) +AC_SUBST(CUPS_REVISION) +AC_SUBST(CUPS_BUILD) +AC_DEFINE_UNQUOTED(CUPS_SVERSION, "CUPS v$CUPS_VERSION$CUPS_REVISION") +AC_DEFINE_UNQUOTED(CUPS_MINIMAL, "CUPS/$CUPS_VERSION$CUPS_REVISION") + +dnl Default compiler flags... +CFLAGS="${CFLAGS:=}" +CPPFLAGS="${CPPFLAGS:=}" +CXXFLAGS="${CXXFLAGS:=}" +LDFLAGS="${LDFLAGS:=}" + +dnl Checks for programs... +AC_PROG_AWK +AC_PROG_CC +AC_PROG_CPP +AC_PROG_CXX +AC_PROG_RANLIB +AC_PATH_PROG(AR,ar) +AC_PATH_PROG(CHMOD,chmod) +AC_PATH_PROG(HTMLDOC,htmldoc) +AC_PATH_PROG(LD,ld) +AC_PATH_PROG(LN,ln) +AC_PATH_PROG(MV,mv) +AC_PATH_PROG(RM,rm) +AC_PATH_PROG(RMDIR,rmdir) +AC_PATH_PROG(SED,sed) +AC_PATH_PROG(XDGOPEN,xdg-open) +if test "x$XDGOPEN" = x; then + CUPS_HTMLVIEW="htmlview" +else + CUPS_HTMLVIEW="$XDGOPEN" +fi +AC_SUBST(CUPS_HTMLVIEW) + +AC_MSG_CHECKING(for install-sh script) +INSTALL="`pwd`/install-sh" +AC_SUBST(INSTALL) +AC_MSG_RESULT(using $INSTALL) + +if test "x$AR" = x; then + AC_MSG_ERROR([Unable to find required library archive command.]) +fi +if test "x$CC" = x; then + AC_MSG_ERROR([Unable to find required C compiler command.]) +fi + +dnl Static library option... +INSTALLSTATIC="" +AC_ARG_ENABLE(static, [ --enable-static install static libraries]) + +if test x$enable_static = xyes; then + echo Installing static libraries... + INSTALLSTATIC="installstatic" +fi + +AC_SUBST(INSTALLSTATIC) + +dnl Check for pkg-config, which is used for some other tests later on... +AC_PATH_PROG(PKGCONFIG, pkg-config) + +dnl Check for libraries... +AC_SEARCH_LIBS(fmod, m) +AC_SEARCH_LIBS(crypt, crypt) +AC_SEARCH_LIBS(getspent, sec gen) + +LIBMALLOC="" +AC_ARG_ENABLE(mallinfo, [ --enable-mallinfo build with malloc debug logging]) + +if test x$enable_mallinfo = xyes; then + SAVELIBS="$LIBS" + LIBS="" + AC_SEARCH_LIBS(mallinfo, malloc, AC_DEFINE(HAVE_MALLINFO)) + LIBMALLOC="$LIBS" + LIBS="$SAVELIBS" +fi + +AC_SUBST(LIBMALLOC) + +dnl Check for libpaper support... +AC_ARG_ENABLE(libpaper, [ --enable-libpaper build with libpaper support]) + +if test x$enable_libpaper = xyes; then + AC_CHECK_LIB(paper,systempapername, + AC_DEFINE(HAVE_LIBPAPER) + LIBPAPER="-lpaper", + LIBPAPER="") +else + LIBPAPER="" +fi +AC_SUBST(LIBPAPER) + +dnl Checks for header files. +AC_HEADER_STDC +AC_CHECK_HEADER(stdlib.h,AC_DEFINE(HAVE_STDLIB_H)) +AC_CHECK_HEADER(crypt.h,AC_DEFINE(HAVE_CRYPT_H)) +AC_CHECK_HEADER(langinfo.h,AC_DEFINE(HAVE_LANGINFO_H)) +AC_CHECK_HEADER(malloc.h,AC_DEFINE(HAVE_MALLOC_H)) +AC_CHECK_HEADER(shadow.h,AC_DEFINE(HAVE_SHADOW_H)) +AC_CHECK_HEADER(string.h,AC_DEFINE(HAVE_STRING_H)) +AC_CHECK_HEADER(strings.h,AC_DEFINE(HAVE_STRINGS_H)) +AC_CHECK_HEADER(bstring.h,AC_DEFINE(HAVE_BSTRING_H)) +AC_CHECK_HEADER(usersec.h,AC_DEFINE(HAVE_USERSEC_H)) +AC_CHECK_HEADER(sys/ioctl.h,AC_DEFINE(HAVE_SYS_IOCTL_H)) +AC_CHECK_HEADER(sys/param.h,AC_DEFINE(HAVE_SYS_PARAM_H)) +AC_CHECK_HEADER(sys/ucred.h,AC_DEFINE(HAVE_SYS_UCRED_H)) +AC_CHECK_HEADER(scsi/sg.h,AC_DEFINE(HAVE_SCSI_SG_H)) + +dnl Checks for iconv.h and iconv_open +AC_CHECK_HEADER(iconv.h, + SAVELIBS="$LIBS" + LIBS="" + AC_SEARCH_LIBS(iconv_open,iconv, + AC_DEFINE(HAVE_ICONV_H) + SAVELIBS="$SAVELIBS $LIBS") + LIBS="$SAVELIBS") + +dnl Checks for Mini-XML (www.minixml.org)... +LIBMXML="" +AC_CHECK_HEADER(mxml.h, + SAVELIBS="$LIBS" + AC_SEARCH_LIBS(mmxlNewElement,mxml, + AC_DEFINE(HAVE_MXML_H) + LIBMXML="-lmxml") + LIBS="$SAVELIBS") +AC_SUBST(LIBMXML) + +dnl Checks for statfs and its many headers... +AC_CHECK_HEADER(sys/mount.h,AC_DEFINE(HAVE_SYS_MOUNT_H)) +AC_CHECK_HEADER(sys/statfs.h,AC_DEFINE(HAVE_SYS_STATFS_H)) +AC_CHECK_HEADER(sys/statvfs.h,AC_DEFINE(HAVE_SYS_STATVFS_H)) +AC_CHECK_HEADER(sys/vfs.h,AC_DEFINE(HAVE_SYS_VFS_H)) +AC_CHECK_FUNCS(statfs statvfs) + +dnl Checks for string functions. +AC_CHECK_FUNCS(strdup strlcat strlcpy) +if test "$uname" = "HP-UX" -a "$uversion" = "1020"; then + echo Forcing snprintf emulation for HP-UX. +else + AC_CHECK_FUNCS(snprintf vsnprintf) +fi + +dnl Check for random number functions... +AC_CHECK_FUNCS(random lrand48 arc4random) + +dnl Check for geteuid function. +AC_CHECK_FUNCS(geteuid) + +dnl Check for setpgid function. +AC_CHECK_FUNCS(setpgid) + +dnl Check for vsyslog function. +AC_CHECK_FUNCS(vsyslog) + +dnl Checks for signal functions. +case "$uname" in + Linux | GNU) + # Do not use sigset on Linux or GNU HURD + ;; + *) + # Use sigset on other platforms, if available + AC_CHECK_FUNCS(sigset) + ;; +esac + +AC_CHECK_FUNCS(sigaction) + +dnl Checks for wait functions. +AC_CHECK_FUNCS(waitpid wait3) + +dnl See if the tm structure has the tm_gmtoff member... +AC_MSG_CHECKING(for tm_gmtoff member in tm structure) +AC_TRY_COMPILE([#include ],[struct tm t; + int o = t.tm_gmtoff;], + AC_MSG_RESULT(yes) + AC_DEFINE(HAVE_TM_GMTOFF), + AC_MSG_RESULT(no)) + +dnl See if the stat structure has the st_gen member... +AC_MSG_CHECKING(for st_gen member in stat structure) +AC_TRY_COMPILE([#include ],[struct stat t; + int o = t.st_gen;], + AC_MSG_RESULT(yes) + AC_DEFINE(HAVE_ST_GEN), + AC_MSG_RESULT(no)) + +dnl See if we have the removefile(3) function for securely removing files +AC_CHECK_FUNCS(removefile) + +dnl See if we have libusb... +AC_ARG_ENABLE(libusb, [ --enable-libusb use libusb for USB printing]) + +LIBUSB="" +AC_SUBST(LIBUSB) + +if test x$enable_libusb = xyes; then + check_libusb=yes +elif test x$enable_libusb != xno -a $uname != Darwin; then + check_libusb=yes +else + check_libusb=no +fi + +if test $check_libusb = yes; then + AC_CHECK_LIB(usb, usb_get_string_simple,[ + AC_CHECK_HEADER(usb.h, + AC_DEFINE(HAVE_USB_H) + LIBUSB="-lusb")]) +fi + +dnl See if we have libwrap for TCP wrappers support... +AC_ARG_ENABLE(tcp_wrappers, [ --enable-tcp-wrappers use libwrap for TCP wrappers support]) + +LIBWRAP="" +AC_SUBST(LIBWRAP) + +if test x$enable_tcp_wrappers = xyes; then + AC_CHECK_LIB(wrap, hosts_access,[ + AC_CHECK_HEADER(tcpd.h, + AC_DEFINE(HAVE_TCPD_H) + LIBWRAP="-lwrap")]) +fi + +dnl ZLIB +LIBZ="" +AC_CHECK_HEADER(zlib.h, + AC_CHECK_LIB(z, gzgets, + AC_DEFINE(HAVE_LIBZ) + LIBZ="-lz" + LIBS="$LIBS -lz")) +AC_SUBST(LIBZ) + +dnl Flags for "ar" command... +case $uname in + Darwin* | *BSD*) + ARFLAGS="-rcv" + ;; + *) + ARFLAGS="crvs" + ;; +esac + +AC_SUBST(ARFLAGS) + +dnl Prep libraries specifically for cupsd and backends... +BACKLIBS="" +SERVERLIBS="" +AC_SUBST(BACKLIBS) +AC_SUBST(SERVERLIBS) + +dnl See if we have POSIX ACL support... +SAVELIBS="$LIBS" +LIBS="" +AC_ARG_ENABLE(acl, [ --enable-acl build with POSIX ACL support]) +if test "x$enable_acl" != xno; then + AC_SEARCH_LIBS(acl_init, acl, AC_DEFINE(HAVE_ACL_INIT)) + SERVERLIBS="$SERVERLIBS $LIBS" +fi +LIBS="$SAVELIBS" + +dnl Check for DBUS support +if test -d /etc/dbus-1; then + DBUSDIR="/etc/dbus-1" +else + DBUSDIR="" +fi + +AC_ARG_ENABLE(dbus, [ --enable-dbus build with DBUS support]) +AC_ARG_WITH(dbusdir, [ --with-dbusdir set DBUS configuration directory ], + DBUSDIR="$withval") + +DBUS_NOTIFIER="" +DBUS_NOTIFIERLIBS="" + +if test "x$enable_dbus" != xno -a "x$PKGCONFIG" != x; then + AC_MSG_CHECKING(for DBUS) + if $PKGCONFIG --exists dbus-1; then + AC_MSG_RESULT(yes) + AC_DEFINE(HAVE_DBUS) + CFLAGS="$CFLAGS `$PKGCONFIG --cflags dbus-1` -DDBUS_API_SUBJECT_TO_CHANGE" + SERVERLIBS="$SERVERLIBS `$PKGCONFIG --libs dbus-1`" + DBUS_NOTIFIER="dbus" + DBUS_NOTIFIERLIBS="`$PKGCONFIG --libs dbus-1`" + SAVELIBS="$LIBS" + LIBS="$LIBS $DBUS_NOTIFIERLIBS" + AC_CHECK_FUNC(dbus_message_iter_init_append, + AC_DEFINE(HAVE_DBUS_MESSAGE_ITER_INIT_APPEND)) + LIBS="$SAVELIBS" + else + AC_MSG_RESULT(no) + fi +fi + +AC_SUBST(DBUSDIR) +AC_SUBST(DBUS_NOTIFIER) +AC_SUBST(DBUS_NOTIFIERLIBS) + +dnl Extra platform-specific libraries... +CUPS_DEFAULT_PRINTOPERATOR_AUTH="@SYSTEM" +CUPS_SYSTEM_AUTHKEY="" +INSTALLXPC="" + +case $uname in + Darwin*) + BACKLIBS="$BACKLIBS -framework IOKit" + SERVERLIBS="$SERVERLIBS -framework IOKit -weak_framework ApplicationServices" + LIBS="-framework SystemConfiguration -framework CoreFoundation -framework Security $LIBS" + + dnl Check for framework headers... + AC_CHECK_HEADER(ApplicationServices/ApplicationServices.h,AC_DEFINE(HAVE_APPLICATIONSERVICES_H)) + AC_CHECK_HEADER(CoreFoundation/CoreFoundation.h,AC_DEFINE(HAVE_COREFOUNDATION_H)) + AC_CHECK_HEADER(CoreFoundation/CFPriv.h,AC_DEFINE(HAVE_CFPRIV_H)) + AC_CHECK_HEADER(CoreFoundation/CFBundlePriv.h,AC_DEFINE(HAVE_CFBUNDLEPRIV_H)) + AC_CHECK_HEADER(IOKit/pwr_mgt/IOPMLibPrivate.h,AC_DEFINE(HAVE_IOKIT_PWR_MGT_IOPMLIBPRIVATE_H)) + + dnl Check for dynamic store function... + AC_CHECK_FUNCS(SCDynamicStoreCopyComputerName) + + dnl Check for new ColorSync APIs... + SAVELIBS="$LIBS" + LIBS="$LIBS -framework ApplicationServices" + AC_CHECK_FUNCS(ColorSyncRegisterDevice) + LIBS="$SAVELIBS" + + dnl Check for the new membership functions in MacOSX 10.4... + AC_CHECK_HEADER(membership.h,AC_DEFINE(HAVE_MEMBERSHIP_H)) + AC_CHECK_HEADER(membershipPriv.h,AC_DEFINE(HAVE_MEMBERSHIPPRIV_H)) + AC_CHECK_FUNCS(mbr_uid_to_uuid) + + dnl Check for the vproc_transaction_begin/end stuff... + AC_CHECK_FUNCS(vproc_transaction_begin) + + dnl Need header... + AC_CHECK_HEADER(dlfcn.h,AC_DEFINE(HAVE_DLFCN_H)) + + dnl Check for notify_post support + AC_CHECK_HEADER(notify.h,AC_DEFINE(HAVE_NOTIFY_H)) + AC_CHECK_FUNCS(notify_post) + + dnl Check for Authorization Services support + AC_ARG_WITH(adminkey, [ --with-adminkey set the default SystemAuthKey value], + default_adminkey="$withval", + default_adminkey="default") + AC_ARG_WITH(operkey, [ --with-operkey set the default operator @AUTHKEY value], + default_operkey="$withval", + default_operkey="default") + + AC_CHECK_HEADER(Security/Authorization.h, [ + AC_DEFINE(HAVE_AUTHORIZATION_H) + + if test "x$default_adminkey" != xdefault; then + CUPS_SYSTEM_AUTHKEY="SystemGroupAuthKey $default_adminkey" + elif grep -q system.print.operator /etc/authorization; then + CUPS_SYSTEM_AUTHKEY="SystemGroupAuthKey system.print.admin" + else + CUPS_SYSTEM_AUTHKEY="SystemGroupAuthKey system.preferences" + fi + + if test "x$default_operkey" != xdefault; then + CUPS_DEFAULT_PRINTOPERATOR_AUTH="@AUTHKEY($default_operkey) @admin @lpadmin" + elif grep -q system.print.operator /etc/authorization; then + CUPS_DEFAULT_PRINTOPERATOR_AUTH="@AUTHKEY(system.print.operator) @admin @lpadmin" + else + CUPS_DEFAULT_PRINTOPERATOR_AUTH="@AUTHKEY(system.print.admin) @admin @lpadmin" + fi]) + AC_CHECK_HEADER(Security/SecBasePriv.h,AC_DEFINE(HAVE_SECBASEPRIV_H)) + + dnl Check for sandbox/Seatbelt support + if test $uversion -ge 100; then + AC_CHECK_HEADER(sandbox.h,AC_DEFINE(HAVE_SANDBOX_H)) + fi + if test $uversion -ge 110; then + # Broken public headers in 10.7... + AC_MSG_CHECKING(for sandbox/private.h presence) + if test -f /usr/local/include/sandbox/private.h; then + AC_MSG_RESULT(yes) + else + AC_MSG_RESULT(no) + AC_MSG_ERROR(Run 'sudo mkdir -p /usr/local/include/sandbox' and 'sudo touch /usr/local/include/sandbox/private.h' to build CUPS.) + fi + fi + + dnl Check for XPC support + AC_CHECK_HEADER(xpc/xpc.h, + AC_DEFINE(HAVE_XPC) + INSTALLXPC="install-xpc") + ;; +esac + +AC_SUBST(CUPS_DEFAULT_PRINTOPERATOR_AUTH) +AC_DEFINE_UNQUOTED(CUPS_DEFAULT_PRINTOPERATOR_AUTH, "$CUPS_DEFAULT_PRINTOPERATOR_AUTH") +AC_SUBST(CUPS_SYSTEM_AUTHKEY) +AC_SUBST(INSTALLXPC) + +dnl Check for build components +COMPONENTS="all" + +AC_ARG_WITH(components, [ --with-components set components to build: + - "all" (default) builds everything + - "core" builds libcups and ipptool], + COMPONENTS="$withval") + +case "$COMPONENTS" in + all) + BUILDDIRS="filter backend berkeley cgi-bin monitor notifier ppdc scheduler systemv conf data desktop locale man doc examples templates" + ;; + + core) + BUILDDIRS="data locale" + ;; + + *) + AC_MSG_ERROR([Bad build component "$COMPONENT" specified!]) + ;; +esac + +AC_SUBST(BUILDDIRS) + +dnl +dnl End of "$Id: cups-common.m4 10192 2012-01-20 21:49:02Z mike $". +dnl diff --git a/config-scripts/cups-compiler.m4 b/config-scripts/cups-compiler.m4 new file mode 100644 index 0000000000..fc415c5588 --- /dev/null +++ b/config-scripts/cups-compiler.m4 @@ -0,0 +1,299 @@ +dnl +dnl "$Id: cups-compiler.m4 10190 2012-01-20 16:22:58Z mike $" +dnl +dnl Compiler stuff for CUPS. +dnl +dnl Copyright 2007-2012 by Apple Inc. +dnl Copyright 1997-2007 by Easy Software Products, all rights reserved. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +dnl Clear the debugging and non-shared library options unless the user asks +dnl for them... +INSTALL_STRIP="" +OPTIM="" +AC_SUBST(INSTALL_STRIP) +AC_SUBST(OPTIM) + +AC_ARG_WITH(optim, [ --with-optim set optimization flags ]) +AC_ARG_ENABLE(debug, [ --enable-debug build with debugging symbols]) +AC_ARG_ENABLE(debug_guards, [ --enable-debug-guards build with memory allocation guards]) +AC_ARG_ENABLE(debug_printfs, [ --enable-debug-printfs build with CUPS_DEBUG_LOG support]) +AC_ARG_ENABLE(unit_tests, [ --enable-unit-tests build and run unit tests]) + +dnl For debugging, keep symbols, otherwise strip them... +if test x$enable_debug = xyes; then + OPTIM="-g" +else + INSTALL_STRIP="-s" +fi + +dnl Debug printfs can slow things down, so provide a separate option for that +if test x$enable_debug_printfs = xyes; then + CFLAGS="$CFLAGS -DDEBUG" + CXXFLAGS="$CXXFLAGS -DDEBUG" +fi + +dnl Debug guards use an extra 4 bytes for some structures like strings in the +dnl string pool, so provide a separate option for that +if test x$enable_debug_guards = xyes; then + CFLAGS="$CFLAGS -DDEBUG_GUARDS" + CXXFLAGS="$CXXFLAGS -DDEBUG_GUARDS" +fi + +dnl Unit tests take up time during a compile... +if test x$enable_unit_tests = xyes; then + UNITTESTS="unittests" +else + UNITTESTS="" +fi +AC_SUBST(UNITTESTS) + +dnl Setup general architecture flags... +AC_ARG_WITH(archflags, [ --with-archflags set default architecture flags ]) +AC_ARG_WITH(ldarchflags, [ --with-ldarchflags set program architecture flags ]) + +if test -z "$with_archflags"; then + ARCHFLAGS="" +else + ARCHFLAGS="$with_archflags" +fi + +if test -z "$with_ldarchflags"; then + if test "$uname" = Darwin; then + # Only create Intel programs by default + LDARCHFLAGS="`echo $ARCHFLAGS | sed -e '1,$s/-arch ppc64//'`" + else + LDARCHFLAGS="$ARCHFLAGS" + fi +else + LDARCHFLAGS="$with_ldarchflags" +fi + +AC_SUBST(ARCHFLAGS) +AC_SUBST(LDARCHFLAGS) + +dnl Read-only data/program support on Linux... +AC_ARG_ENABLE(relro, [ --enable-relro build with the GCC relro option]) + +dnl Update compiler options... +CXXLIBS="${CXXLIBS:=}" +AC_SUBST(CXXLIBS) + +PIEFLAGS="" +AC_SUBST(PIEFLAGS) + +RELROFLAGS="" +AC_SUBST(RELROFLAGS) + +if test -n "$GCC"; then + # Add GCC-specific compiler options... + if test -z "$OPTIM"; then + if test "x$with_optim" = x; then + # Default to optimize-for-size and debug + OPTIM="-Os -g" + else + OPTIM="$with_optim $OPTIM" + fi + fi + + # Generate position-independent code as needed... + if test $PICFLAG = 1 -a $uname != AIX; then + OPTIM="-fPIC $OPTIM" + fi + + # The -fstack-protector option is available with some versions of + # GCC and adds "stack canaries" which detect when the return address + # has been overwritten, preventing many types of exploit attacks. + AC_MSG_CHECKING(if GCC supports -fstack-protector) + OLDCFLAGS="$CFLAGS" + CFLAGS="$CFLAGS -fstack-protector" + AC_TRY_LINK(,, + OPTIM="$OPTIM -fstack-protector" + AC_MSG_RESULT(yes), + AC_MSG_RESULT(no)) + CFLAGS="$OLDCFLAGS" + + # The -fPIE option is available with some versions of GCC and adds + # randomization of addresses, which avoids another class of exploits + # that depend on a fixed address for common functions. + AC_MSG_CHECKING(if GCC supports -fPIE) + OLDCFLAGS="$CFLAGS" + CFLAGS="$CFLAGS -fPIE" + AC_TRY_COMPILE(,, + [case "$CC" in + *clang) + PIEFLAGS="-fPIE -Wl,-pie" + ;; + *) + PIEFLAGS="-fPIE -pie" + ;; + esac + AC_MSG_RESULT(yes)], + AC_MSG_RESULT(no)) + CFLAGS="$OLDCFLAGS" + + if test "x$with_optim" = x; then + # Add useful warning options for tracking down problems... + OPTIM="-Wall -Wno-format-y2k -Wunused $OPTIM" + + # Additional warning options for development testing... + if test -d .svn; then + OPTIM="-Wshadow -Werror $OPTIM" + else + AC_MSG_CHECKING(if GCC supports -Wno-tautological-compare) + OLDCFLAGS="$CFLAGS" + CFLAGS="$CFLAGS -Werror -Wno-tautological-compare" + AC_TRY_COMPILE(,, + [OPTIM="$OPTIM -Wno-tautological-compare" + AC_MSG_RESULT(yes)], + AC_MSG_RESULT(no)) + CFLAGS="$OLDCFLAGS" + fi + fi + + case "$uname" in + Darwin*) + # -D_FORTIFY_SOURCE=2 adds additional object size + # checking, basically wrapping all string functions + # with buffer-limited ones. Not strictly needed for + # CUPS since we already use buffer-limited calls, but + # this will catch any additions that are broken. + CFLAGS="$CFLAGS -D_FORTIFY_SOURCE=2" + ;; + + Linux*) + # The -z relro option is provided by the Linux linker command to + # make relocatable data read-only. + if test x$enable_relro = xyes; then + RELROFLAGS="-Wl,-z,relro" + fi + ;; + esac +else + # Add vendor-specific compiler options... + case $uname in + AIX*) + if test -z "$OPTIM"; then + if test "x$with_optim" = x; then + OPTIM="-O2 -qmaxmem=6000" + else + OPTIM="$with_optim $OPTIM" + fi + fi + ;; + HP-UX*) + if test -z "$OPTIM"; then + if test "x$with_optim" = x; then + OPTIM="+O2" + else + OPTIM="$with_optim $OPTIM" + fi + fi + + CFLAGS="-Ae $CFLAGS" + + if test $PICFLAG = 1; then + OPTIM="+z $OPTIM" + fi + ;; + IRIX) + if test -z "$OPTIM"; then + if test "x$with_optim" = x; then + OPTIM="-O2" + else + OPTIM="$with_optim $OPTIM" + fi + fi + + if test "x$with_optim" = x; then + OPTIM="-fullwarn -woff 1183,1209,1349,1506,3201 $OPTIM" + fi + ;; + OSF*) + # Tru64 UNIX aka Digital UNIX aka OSF/1 + if test -z "$OPTIM"; then + if test "x$with_optim" = x; then + OPTIM="-O" + else + OPTIM="$with_optim" + fi + fi + ;; + SunOS*) + # Solaris + if test -z "$OPTIM"; then + if test "x$with_optim" = x; then + OPTIM="-xO2" + else + OPTIM="$with_optim $OPTIM" + fi + fi + + if test $PICFLAG = 1; then + OPTIM="-KPIC $OPTIM" + fi + ;; + UNIX_SVR*) + # UnixWare + if test -z "$OPTIM"; then + if test "x$with_optim" = x; then + OPTIM="-O" + else + OPTIM="$with_optim $OPTIM" + fi + fi + + if test $PICFLAG = 1; then + OPTIM="-KPIC $OPTIM" + fi + ;; + *) + # Running some other operating system; inform the user they + # should contribute the necessary options to + # cups-support@cups.org... + echo "Building CUPS with default compiler optimizations; contact" + echo "cups-bugs@cups.org with uname and compiler options needed" + echo "for your platform, or set the CFLAGS and LDFLAGS environment" + echo "variables before running configure." + ;; + esac +fi + +# Add general compiler options per platform... +case $uname in + HP-UX*) + # HP-UX 10.20 (at least) needs this definition to get the + # h_errno global... + OPTIM="$OPTIM -D_XOPEN_SOURCE_EXTENDED" + + # HP-UX 11.00 (at least) needs this definition to get the + # u_short type used by the IP headers... + OPTIM="$OPTIM -D_INCLUDE_HPUX_SOURCE" + + # HP-UX 11.23 (at least) needs this definition to get the + # IPv6 header to work... + OPTIM="$OPTIM -D_HPUX_SOURCE" + ;; + + Linux*) + # glibc 2.8 and higher breaks peer credentials unless you + # define _GNU_SOURCE... + OPTIM="$OPTIM -D_GNU_SOURCE" + ;; + + OSF*) + # Tru64 UNIX aka Digital UNIX aka OSF/1 need to be told + # to be POSIX-compliant... + OPTIM="$OPTIM -D_XOPEN_SOURCE=500 -D_XOPEN_SOURCE_EXTENDED -D_OSF_SOURCE" + ;; +esac + +dnl +dnl End of "$Id: cups-compiler.m4 10190 2012-01-20 16:22:58Z mike $". +dnl diff --git a/config-scripts/cups-defaults.m4 b/config-scripts/cups-defaults.m4 new file mode 100644 index 0000000000..749a0d944e --- /dev/null +++ b/config-scripts/cups-defaults.m4 @@ -0,0 +1,396 @@ +dnl +dnl "$Id: cups-defaults.m4 10104 2011-11-04 06:37:03Z mike $" +dnl +dnl Default cupsd configuration settings for CUPS. +dnl +dnl Copyright 2007-2011 by Apple Inc. +dnl Copyright 2006-2007 by Easy Software Products, all rights reserved. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +dnl Default languages... +LANGUAGES="`ls -1 locale/cups_*.po | sed -e '1,$s/locale\/cups_//' -e '1,$s/\.po//' | tr '\n' ' '`" + +AC_ARG_WITH(languages, [ --with-languages set installed languages, default=all ],[ + case "$withval" in + none | no) LANGUAGES="" ;; + all) ;; + *) LANGUAGES="$withval" ;; + esac]) +AC_SUBST(LANGUAGES) + +dnl Mac OS X bundle-based localization support +AC_ARG_WITH(bundledir, [ --with-bundledir set Mac OS X localization bundle directory ], + CUPS_BUNDLEDIR="$withval", + if test "x$uname" = xDarwin -a $uversion -ge 100; then + CUPS_BUNDLEDIR="/System/Library/Frameworks/ApplicationServices.framework/Versions/A/Frameworks/PrintCore.framework/Versions/A" + LANGUAGES="" + else + CUPS_BUNDLEDIR="" + fi) + +AC_SUBST(CUPS_BUNDLEDIR) +if test "x$CUPS_BUNDLEDIR" != x; then + AC_DEFINE_UNQUOTED(CUPS_BUNDLEDIR, "$CUPS_BUNDLEDIR") +fi + +dnl Default ConfigFilePerm +AC_ARG_WITH(config_file_perm, [ --with-config-file-perm set default ConfigFilePerm value, default=0640], + CUPS_CONFIG_FILE_PERM="$withval", + if test "x$uname" = xDarwin; then + CUPS_CONFIG_FILE_PERM="644" + else + CUPS_CONFIG_FILE_PERM="640" + fi) +AC_SUBST(CUPS_CONFIG_FILE_PERM) +AC_DEFINE_UNQUOTED(CUPS_DEFAULT_CONFIG_FILE_PERM, 0$CUPS_CONFIG_FILE_PERM) + +dnl Default LogFilePerm +AC_ARG_WITH(log_file_perm, [ --with-log-file-perm set default LogFilePerm value, default=0644], + CUPS_LOG_FILE_PERM="$withval", + CUPS_LOG_FILE_PERM="644") +AC_SUBST(CUPS_LOG_FILE_PERM) +AC_DEFINE_UNQUOTED(CUPS_DEFAULT_LOG_FILE_PERM, 0$CUPS_LOG_FILE_PERM) + +dnl Default FatalErrors +AC_ARG_WITH(fatal_errors, [ --with-fatal-errors set default FatalErrors value, default=config], + CUPS_FATAL_ERRORS="$withval", + CUPS_FATAL_ERRORS="config") +AC_SUBST(CUPS_FATAL_ERRORS) +AC_DEFINE_UNQUOTED(CUPS_DEFAULT_FATAL_ERRORS, "$CUPS_FATAL_ERRORS") + +dnl Default LogLevel +AC_ARG_WITH(log_level, [ --with-log-level set default LogLevel value, default=warn], + CUPS_LOG_LEVEL="$withval", + CUPS_LOG_LEVEL="warn") +AC_SUBST(CUPS_LOG_LEVEL) +AC_DEFINE_UNQUOTED(CUPS_DEFAULT_LOG_LEVEL, "$CUPS_LOG_LEVEL") + +dnl Default AccessLogLevel +AC_ARG_WITH(access_log_level, [ --with-access-log-level set default AccessLogLevel value, default=actions], + CUPS_ACCESS_LOG_LEVEL="$withval", + CUPS_ACCESS_LOG_LEVEL="actions") +AC_SUBST(CUPS_ACCESS_LOG_LEVEL) +AC_DEFINE_UNQUOTED(CUPS_DEFAULT_ACCESS_LOG_LEVEL, "$CUPS_ACCESS_LOG_LEVEL") + +dnl Default Browsing +AC_ARG_ENABLE(browsing, [ --disable-browsing disable Browsing by default]) +if test "x$enable_browsing" = xno; then + CUPS_BROWSING="No" + AC_DEFINE_UNQUOTED(CUPS_DEFAULT_BROWSING, 0) +else + CUPS_BROWSING="Yes" + AC_DEFINE_UNQUOTED(CUPS_DEFAULT_BROWSING, 1) +fi +AC_SUBST(CUPS_BROWSING) + +dnl Default BrowseLocalProtocols +AC_ARG_WITH(local_protocols, [ --with-local-protocols set default BrowseLocalProtocols, default=""], + default_local_protocols="$withval", + default_local_protocols="default") + +if test x$with_local_protocols != xno; then + if test "x$default_local_protocols" = "xdefault"; then + if test "x$DNSSDLIBS" != "x"; then + CUPS_BROWSE_LOCAL_PROTOCOLS="dnssd" + else + CUPS_BROWSE_LOCAL_PROTOCOLS="" + fi + else + CUPS_BROWSE_LOCAL_PROTOCOLS="$default_local_protocols" + fi +else + CUPS_BROWSE_LOCAL_PROTOCOLS="" +fi + +AC_SUBST(CUPS_BROWSE_LOCAL_PROTOCOLS) +AC_DEFINE_UNQUOTED(CUPS_DEFAULT_BROWSE_LOCAL_PROTOCOLS, + "$CUPS_BROWSE_LOCAL_PROTOCOLS") + +dnl Default DefaultShared +AC_ARG_ENABLE(default_shared, [ --disable-default-shared + disable DefaultShared by default]) +if test "x$enable_default_shared" = xno; then + CUPS_DEFAULT_SHARED="No" + AC_DEFINE_UNQUOTED(CUPS_DEFAULT_DEFAULT_SHARED, 0) +else + CUPS_DEFAULT_SHARED="Yes" + AC_DEFINE_UNQUOTED(CUPS_DEFAULT_DEFAULT_SHARED, 1) +fi +AC_SUBST(CUPS_DEFAULT_SHARED) + +dnl Determine the correct username and group for this OS... +AC_ARG_WITH(cups_user, [ --with-cups-user set default user for CUPS], + CUPS_USER="$withval", + AC_MSG_CHECKING(for default print user) + if test x$uname = xDarwin; then + if test x`id -u _lp 2>/dev/null` = x; then + CUPS_USER="lp"; + else + CUPS_USER="_lp"; + fi + AC_MSG_RESULT($CUPS_USER) + elif test -f /etc/passwd; then + CUPS_USER="" + for user in lp lpd guest daemon nobody; do + if test "`grep \^${user}: /etc/passwd`" != ""; then + CUPS_USER="$user" + AC_MSG_RESULT($user) + break; + fi + done + + if test x$CUPS_USER = x; then + CUPS_USER="nobody" + AC_MSG_RESULT(not found, using "$CUPS_USER") + fi + else + CUPS_USER="nobody" + AC_MSG_RESULT(no password file, using "$CUPS_USER") + fi) + +if test "x$CUPS_USER" = "xroot" -o "x$CUPS_USER" = "x0"; then + AC_MSG_ERROR([The default user for CUPS cannot be root!]) +fi + +AC_ARG_WITH(cups_group, [ --with-cups-group set default group for CUPS], + CUPS_GROUP="$withval", + AC_MSG_CHECKING(for default print group) + if test x$uname = xDarwin; then + if test x`id -g _lp 2>/dev/null` = x; then + CUPS_GROUP="lp"; + else + CUPS_GROUP="_lp"; + fi + AC_MSG_RESULT($CUPS_GROUP) + elif test -f /etc/group; then + GROUP_LIST="_lp lp nobody" + CUPS_GROUP="" + for group in $GROUP_LIST; do + if test "`grep \^${group}: /etc/group`" != ""; then + CUPS_GROUP="$group" + AC_MSG_RESULT($group) + break; + fi + done + + if test x$CUPS_GROUP = x; then + CUPS_GROUP="nobody" + AC_MSG_RESULT(not found, using "$CUPS_GROUP") + fi + else + CUPS_GROUP="nobody" + AC_MSG_RESULT(no group file, using "$CUPS_GROUP") + fi) + +if test "x$CUPS_GROUP" = "xroot" -o "x$CUPS_GROUP" = "xwheel" -o "x$CUPS_GROUP" = "x0"; then + AC_MSG_ERROR([The default group for CUPS cannot be root!]) +fi + +AC_ARG_WITH(system_groups, [ --with-system-groups set default system groups for CUPS], + CUPS_SYSTEM_GROUPS="$withval", + if test x$uname = xDarwin; then + CUPS_SYSTEM_GROUPS="admin" + else + AC_MSG_CHECKING(for default system groups) + if test -f /etc/group; then + CUPS_SYSTEM_GROUPS="" + GROUP_LIST="lpadmin sys system root" + for group in $GROUP_LIST; do + if test "`grep \^${group}: /etc/group`" != ""; then + if test "x$CUPS_SYSTEM_GROUPS" = x; then + CUPS_SYSTEM_GROUPS="$group" + else + CUPS_SYSTEM_GROUPS="$CUPS_SYSTEM_GROUPS $group" + fi + fi + done + + if test "x$CUPS_SYSTEM_GROUPS" = x; then + CUPS_SYSTEM_GROUPS="$GROUP_LIST" + AC_MSG_RESULT(no groups found, using "$CUPS_SYSTEM_GROUPS") + else + AC_MSG_RESULT("$CUPS_SYSTEM_GROUPS") + fi + else + CUPS_SYSTEM_GROUPS="$GROUP_LIST" + AC_MSG_RESULT(no group file, using "$CUPS_SYSTEM_GROUPS") + fi + fi) + +CUPS_PRIMARY_SYSTEM_GROUP="`echo $CUPS_SYSTEM_GROUPS | awk '{print $1}'`" + +for group in $CUPS_SYSTEM_GROUPS; do + if test "x$CUPS_GROUP" = "x$group"; then + AC_MSG_ERROR([The default system groups cannot contain the default CUPS group!]) + fi +done + +AC_SUBST(CUPS_USER) +AC_SUBST(CUPS_GROUP) +AC_SUBST(CUPS_SYSTEM_GROUPS) +AC_SUBST(CUPS_PRIMARY_SYSTEM_GROUP) + +AC_DEFINE_UNQUOTED(CUPS_DEFAULT_USER, "$CUPS_USER") +AC_DEFINE_UNQUOTED(CUPS_DEFAULT_GROUP, "$CUPS_GROUP") +AC_DEFINE_UNQUOTED(CUPS_DEFAULT_SYSTEM_GROUPS, "$CUPS_SYSTEM_GROUPS") + +dnl Default printcap file... +AC_ARG_WITH(printcap, [ --with-printcap set default printcap file], + default_printcap="$withval", + default_printcap="default") + +if test x$default_printcap != xno; then + if test "x$default_printcap" = "xdefault"; then + case $uname in + Darwin*) + if test $uversion -ge 90; then + CUPS_DEFAULT_PRINTCAP="/Library/Preferences/org.cups.printers.plist" + else + CUPS_DEFAULT_PRINTCAP="/etc/printcap" + fi + ;; + SunOS*) + CUPS_DEFAULT_PRINTCAP="/etc/printers.conf" + ;; + *) + CUPS_DEFAULT_PRINTCAP="/etc/printcap" + ;; + esac + else + CUPS_DEFAULT_PRINTCAP="$default_printcap" + fi +else + CUPS_DEFAULT_PRINTCAP="" +fi + +AC_SUBST(CUPS_DEFAULT_PRINTCAP) +AC_DEFINE_UNQUOTED(CUPS_DEFAULT_PRINTCAP, "$CUPS_DEFAULT_PRINTCAP") + +dnl Default LPD config file... +AC_ARG_WITH(lpdconfigfile, [ --with-lpdconfigfile set default LPDConfigFile URI], + default_lpdconfigfile="$withval", + default_lpdconfigfile="default") + +if test x$default_lpdconfigfile != xno; then + if test "x$default_lpdconfigfile" = "xdefault"; then + case $uname in + Darwin*) + CUPS_DEFAULT_LPD_CONFIG_FILE="launchd:///System/Library/LaunchDaemons/org.cups.cups-lpd.plist" + ;; + *) + if test "x$XINETD" != x; then + CUPS_DEFAULT_LPD_CONFIG_FILE="xinetd://$XINETD/cups-lpd" + else + CUPS_DEFAULT_LPD_CONFIG_FILE="" + fi + ;; + esac + else + CUPS_DEFAULT_LPD_CONFIG_FILE="$default_lpdconfigfile" + fi +else + CUPS_DEFAULT_LPD_CONFIG_FILE="" +fi + +AC_DEFINE_UNQUOTED(CUPS_DEFAULT_LPD_CONFIG_FILE, "$CUPS_DEFAULT_LPD_CONFIG_FILE") + +dnl Default SMB config file... +AC_ARG_WITH(smbconfigfile, [ --with-smbconfigfile set default SMBConfigFile URI], + default_smbconfigfile="$withval", + default_smbconfigfile="default") + +if test x$default_smbconfigfile != xno; then + if test "x$default_smbconfigfile" = "xdefault"; then + if test -f /etc/smb.conf; then + CUPS_DEFAULT_SMB_CONFIG_FILE="samba:///etc/smb.conf" + else + CUPS_DEFAULT_SMB_CONFIG_FILE="" + fi + else + CUPS_DEFAULT_SMB_CONFIG_FILE="$default_smbconfigfile" + fi +else + CUPS_DEFAULT_SMB_CONFIG_FILE="" +fi + +AC_DEFINE_UNQUOTED(CUPS_DEFAULT_SMB_CONFIG_FILE, "$CUPS_DEFAULT_SMB_CONFIG_FILE") + +dnl Default MaxCopies value... +AC_ARG_WITH(max-copies, [ --with-max-copies set default max copies value, default=9999 ], + CUPS_MAX_COPIES="$withval", + CUPS_MAX_COPIES="9999") + +AC_SUBST(CUPS_MAX_COPIES) +AC_DEFINE_UNQUOTED(CUPS_DEFAULT_MAX_COPIES, $CUPS_MAX_COPIES) + +dnl Default raw printing state +AC_ARG_ENABLE(raw_printing, [ --disable-raw-printing do not allow raw printing by default]) +if test "x$enable_raw_printing" != xno; then + DEFAULT_RAW_PRINTING="" +else + DEFAULT_RAW_PRINTING="#" +fi +AC_SUBST(DEFAULT_RAW_PRINTING) + +dnl Default SNMP options... +AC_ARG_WITH(snmp-address, [ --with-snmp-address set SNMP query address, default=auto ], + if test "x$withval" = x; then + CUPS_SNMP_ADDRESS="" + else + CUPS_SNMP_ADDRESS="Address $withval" + fi, + if test "x$uname" = xDarwin; then + CUPS_SNMP_ADDRESS="" + else + CUPS_SNMP_ADDRESS="Address @LOCAL" + fi) + +AC_ARG_WITH(snmp-community, [ --with-snmp-community set SNMP community, default=public ], + CUPS_SNMP_COMMUNITY="Community $withval", + CUPS_SNMP_COMMUNITY="Community public") + +AC_SUBST(CUPS_SNMP_ADDRESS) +AC_SUBST(CUPS_SNMP_COMMUNITY) + +dnl New default port definition for IPP... +AC_ARG_WITH(ipp-port, [ --with-ipp-port set port number for IPP, default=631 ], + DEFAULT_IPP_PORT="$withval", + DEFAULT_IPP_PORT="631") + +AC_SUBST(DEFAULT_IPP_PORT) +AC_DEFINE_UNQUOTED(CUPS_DEFAULT_IPP_PORT,$DEFAULT_IPP_PORT) + +dnl Web interface... +AC_ARG_ENABLE(webif, [ --enable-webif enable the web interface by default, default=no for Mac OS X]) +case "x$enable_webif" in + xno) + CUPS_WEBIF=No + CUPS_DEFAULT_WEBIF=0 + ;; + xyes) + CUPS_WEBIF=Yes + CUPS_DEFAULT_WEBIF=1 + ;; + *) + if test $uname = Darwin; then + CUPS_WEBIF=No + CUPS_DEFAULT_WEBIF=0 + else + CUPS_WEBIF=Yes + CUPS_DEFAULT_WEBIF=1 + fi + ;; +esac + +AC_SUBST(CUPS_WEBIF) +AC_DEFINE_UNQUOTED(CUPS_DEFAULT_WEBIF, $CUPS_DEFAULT_WEBIF) + +dnl +dnl End of "$Id: cups-defaults.m4 10104 2011-11-04 06:37:03Z mike $". +dnl diff --git a/config-scripts/cups-directories.m4 b/config-scripts/cups-directories.m4 new file mode 100644 index 0000000000..4159f4cf89 --- /dev/null +++ b/config-scripts/cups-directories.m4 @@ -0,0 +1,440 @@ +dnl +dnl "$Id: cups-directories.m4 9771 2011-05-12 05:21:56Z mike $" +dnl +dnl Directory stuff for CUPS. +dnl +dnl Copyright 2007-2011 by Apple Inc. +dnl Copyright 1997-2007 by Easy Software Products, all rights reserved. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +AC_PREFIX_DEFAULT(/) + +dnl Fix "prefix" variable if it hasn't been specified... +if test "$prefix" = "NONE"; then + prefix="/" +fi + +dnl Fix "exec_prefix" variable if it hasn't been specified... +if test "$exec_prefix" = "NONE"; then + if test "$prefix" = "/"; then + exec_prefix="/usr" + else + exec_prefix="$prefix" + fi +fi + +dnl Fix "bindir" variable... +if test "$bindir" = "\${exec_prefix}/bin"; then + bindir="$exec_prefix/bin" +fi + +AC_DEFINE_UNQUOTED(CUPS_BINDIR, "$bindir") + +dnl Fix "sbindir" variable... +if test "$sbindir" = "\${exec_prefix}/sbin"; then + sbindir="$exec_prefix/sbin" +fi + +AC_DEFINE_UNQUOTED(CUPS_SBINDIR, "$sbindir") + +dnl Fix "sharedstatedir" variable if it hasn't been specified... +if test "$sharedstatedir" = "\${prefix}/com" -a "$prefix" = "/"; then + sharedstatedir="/usr/com" +fi + +dnl Fix "datarootdir" variable if it hasn't been specified... +if test "$datarootdir" = "\${prefix}/share"; then + if test "$prefix" = "/"; then + datarootdir="/usr/share" + else + datarootdir="$prefix/share" + fi +fi + +dnl Fix "datadir" variable if it hasn't been specified... +if test "$datadir" = "\${prefix}/share"; then + if test "$prefix" = "/"; then + datadir="/usr/share" + else + datadir="$prefix/share" + fi +elif test "$datadir" = "\${datarootdir}"; then + datadir="$datarootdir" +fi + +dnl Fix "includedir" variable if it hasn't been specified... +if test "$includedir" = "\${prefix}/include" -a "$prefix" = "/"; then + includedir="/usr/include" +fi + +dnl Fix "localstatedir" variable if it hasn't been specified... +if test "$localstatedir" = "\${prefix}/var"; then + if test "$prefix" = "/"; then + if test "$uname" = Darwin; then + localstatedir="/private/var" + else + localstatedir="/var" + fi + else + localstatedir="$prefix/var" + fi +fi + +dnl Fix "sysconfdir" variable if it hasn't been specified... +if test "$sysconfdir" = "\${prefix}/etc"; then + if test "$prefix" = "/"; then + if test "$uname" = Darwin; then + sysconfdir="/private/etc" + else + sysconfdir="/etc" + fi + else + sysconfdir="$prefix/etc" + fi +fi + +dnl Fix "libdir" variable... +if test "$libdir" = "\${exec_prefix}/lib"; then + case "$uname" in + IRIX*) + libdir="$exec_prefix/lib32" + ;; + Linux*) + if test -d /usr/lib64; then + libdir="$exec_prefix/lib64" + fi + ;; + HP-UX*) + if test -d /usr/lib/hpux32; then + libdir="$exec_prefix/lib/hpux32" + fi + ;; + esac +fi + +dnl Setup private include directory... +AC_ARG_WITH(privateinclude, [ --with-privateinclude set path for private include files, default=none],privateinclude="$withval",privateinclude="") +if test "x$privateinclude" != x -a "x$privateinclude" != xnone; then + PRIVATEINCLUDE="$privateinclude/cups" +else + privateinclude="" + PRIVATEINCLUDE="" +fi +AC_SUBST(privateinclude) +AC_SUBST(PRIVATEINCLUDE) + +dnl Setup init.d locations... +AC_ARG_WITH(rcdir, [ --with-rcdir set path for rc scripts],rcdir="$withval",rcdir="") +AC_ARG_WITH(rclevels, [ --with-rclevels set run levels for rc scripts],rclevels="$withval",rclevels="2 3 5") +AC_ARG_WITH(rcstart, [ --with-rcstart set start number for rc scripts],rcstart="$withval",rcstart="99") +AC_ARG_WITH(rcstop, [ --with-rcstop set stop number for rc scripts],rcstop="$withval",rcstop="00") +AC_ARG_WITH(smfmanifestdir, [ --with-smfmanifestdir set path for Solaris SMF manifest],smfmanifestdir="$withval",smfmanifestdir="") + +INITDIR="" +INITDDIR="" +RCLEVELS="$rclevels" +RCSTART="$rcstart" +RCSTOP="$rcstop" +SMFMANIFESTDIR="" + +if test x$rcdir = x; then + case "$uname" in + AIX*) + INITDIR="/etc/rc.d" + ;; + + Darwin*) + # Darwin and MacOS X... + if test -x /sbin/launchd; then + INITDDIR="/System/Library/LaunchDaemons" + else + INITDDIR="/System/Library/StartupItems/PrintingServices" + fi + ;; + + FreeBSD* | OpenBSD* | MirBSD* | ekkoBSD*) + # FreeBSD and OpenBSD + ;; + + HP-UX*) + INITDIR="/sbin" + RCLEVELS="2" + RCSTART="380" + RCSTOP="620" + ;; + + IRIX*) + # IRIX + INITDIR="/etc" + RCSTART="60" + RCSTOP="25" + ;; + + Linux | GNU | GNU/k*BSD*) + # Linux/HURD seems to choose an init.d directory at random... + if test -d /sbin/init.d; then + # SuSE + INITDIR="/sbin/init.d" + else + if test -d /etc/init.d; then + # Others + INITDIR="/etc" + else + # RedHat + INITDIR="/etc/rc.d" + fi + fi + RCSTART="81" + RCSTOP="36" + ;; + + NetBSD*) + # NetBSD + INITDDIR="/etc/rc.d" + ;; + + OSF1*) + INITDIR="/sbin" + ;; + + SunOS*) + # Solaris + if test "x$smfmanifestdir" != x; then + SMFMANIFESTDIR=$smfmanifestdir + else + INITDIR="/etc" + RCSTART="81" + fi + ;; + + *) + INITDIR="/etc" + ;; + + esac +elif test "x$rcdir" != xno; then + if test "x$rclevels" = x; then + INITDDIR="$rcdir" + else + INITDIR="$rcdir" + fi +fi + +AC_SUBST(INITDIR) +AC_SUBST(INITDDIR) +AC_SUBST(RCLEVELS) +AC_SUBST(RCSTART) +AC_SUBST(RCSTOP) +AC_SUBST(SMFMANIFESTDIR) + +dnl Xinetd support... +AC_ARG_WITH(xinetd, [ --with-xinetd set path for xinetd config files],XINETD="$withval",XINETD="") + +if test "x$XINETD" = x -a ! -x /sbin/launchd; then + for dir in /private/etc/xinetd.d /etc/xinetd.d /usr/local/etc/xinetd.d; do + if test -d $dir; then + XINETD="$dir" + break + fi + done +elif test "x$XINETD" = xno; then + XINETD="" +fi + +AC_SUBST(XINETD) + +dnl LPD sharing support... +AC_ARG_WITH(lpdconfig, [ --with-lpdconfig set URI for LPD config file], + LPDCONFIG="$withval", LPDCONFIG="") + +if test "x$LPDCONFIG" = x; then + if test -f /System/Library/LaunchDaemons/org.cups.cups-lpd.plist; then + LPDCONFIG="launchd:///System/Library/LaunchDaemons/org.cups.cups-lpd.plist" + elif test "x$XINETD" != x; then + LPDCONFIG="xinetd://$XINETD/cups-lpd" + fi +fi + +if test "x$LPDCONFIG" = xoff; then + AC_DEFINE_UNQUOTED(CUPS_DEFAULT_LPD_CONFIG, "") +else + AC_DEFINE_UNQUOTED(CUPS_DEFAULT_LPD_CONFIG, "$LPDCONFIG") +fi + +dnl SMB sharing support... +AC_ARG_WITH(smbconfig, [ --with-smbconfig set URI for Samba config file], + SMBCONFIG="$withval", SMBCONFIG="") + +if test "x$SMBCONFIG" = x; then + if test -f /System/Library/LaunchDaemons/smbd.plist; then + SMBCONFIG="launchd:///System/Library/LaunchDaemons/smbd.plist" + else + for dir in /etc /etc/samba /usr/local/etc; do + if test -f $dir/smb.conf; then + SMBCONFIG="samba://$dir/smb.conf" + break + fi + done + fi +fi + +if test "x$SMBCONFIG" = xoff; then + AC_DEFINE_UNQUOTED(CUPS_DEFAULT_SMB_CONFIG, "") +else + AC_DEFINE_UNQUOTED(CUPS_DEFAULT_SMB_CONFIG, "$SMBCONFIG") +fi + +dnl Setup default locations... +# Cache data... +AC_ARG_WITH(cachedir, [ --with-cachedir set path for cache files],cachedir="$withval",cachedir="") + +if test x$cachedir = x; then + if test "x$uname" = xDarwin; then + CUPS_CACHEDIR="$localstatedir/spool/cups/cache" + else + CUPS_CACHEDIR="$localstatedir/cache/cups" + fi +else + CUPS_CACHEDIR="$cachedir" +fi +AC_DEFINE_UNQUOTED(CUPS_CACHEDIR, "$CUPS_CACHEDIR") +AC_SUBST(CUPS_CACHEDIR) + +# Data files +CUPS_DATADIR="$datadir/cups" +AC_DEFINE_UNQUOTED(CUPS_DATADIR, "$datadir/cups") +AC_SUBST(CUPS_DATADIR) + +# Icon directory +AC_ARG_WITH(icondir, [ --with-icondir set path for application icons],icondir="$withval",icondir="") + +if test "x$icondir" = x -a -d /usr/share/icons; then + ICONDIR="/usr/share/icons" +else + ICONDIR="$icondir" +fi + +AC_SUBST(ICONDIR) + +# Menu directory +AC_ARG_WITH(menudir, [ --with-menudir set path for application menus],menudir="$withval",menudir="") + +if test "x$menudir" = x -a -d /usr/share/applications; then + MENUDIR="/usr/share/applications" +else + MENUDIR="$menudir" +fi + +AC_SUBST(MENUDIR) + +# Documentation files +AC_ARG_WITH(docdir, [ --with-docdir set path for documentation],docdir="$withval",docdir="") + +if test x$docdir = x; then + CUPS_DOCROOT="$datadir/doc/cups" + docdir="$datadir/doc/cups" +else + CUPS_DOCROOT="$docdir" +fi + +AC_DEFINE_UNQUOTED(CUPS_DOCROOT, "$docdir") +AC_SUBST(CUPS_DOCROOT) + +# Fonts +AC_ARG_WITH(fontpath, [ --with-fontpath set font path for pstoraster],fontpath="$withval",fontpath="") + +if test "x$fontpath" = "x"; then + CUPS_FONTPATH="$datadir/cups/fonts" +else + CUPS_FONTPATH="$fontpath" +fi + +AC_SUBST(CUPS_FONTPATH) +AC_DEFINE_UNQUOTED(CUPS_FONTPATH, "$CUPS_FONTPATH") + +# Locale data +if test "$localedir" = "\${datarootdir}/locale"; then + case "$uname" in + Linux | GNU | *BSD* | Darwin*) + CUPS_LOCALEDIR="$datarootdir/locale" + ;; + + OSF1* | AIX*) + CUPS_LOCALEDIR="$exec_prefix/lib/nls/msg" + ;; + + *) + # This is the standard System V location... + CUPS_LOCALEDIR="$exec_prefix/lib/locale" + ;; + esac +else + CUPS_LOCALEDIR="$localedir" +fi + +AC_DEFINE_UNQUOTED(CUPS_LOCALEDIR, "$CUPS_LOCALEDIR") +AC_SUBST(CUPS_LOCALEDIR) + +# Log files... +AC_ARG_WITH(logdir, [ --with-logdir set path for log files],logdir="$withval",logdir="") + +if test x$logdir = x; then + CUPS_LOGDIR="$localstatedir/log/cups" + AC_DEFINE_UNQUOTED(CUPS_LOGDIR, "$localstatedir/log/cups") +else + CUPS_LOGDIR="$logdir" +fi +AC_DEFINE_UNQUOTED(CUPS_LOGDIR, "$CUPS_LOGDIR") +AC_SUBST(CUPS_LOGDIR) + +# Longer-term spool data +CUPS_REQUESTS="$localstatedir/spool/cups" +AC_DEFINE_UNQUOTED(CUPS_REQUESTS, "$localstatedir/spool/cups") +AC_SUBST(CUPS_REQUESTS) + +# Server executables... +case "$uname" in + *BSD* | Darwin*) + # *BSD and Darwin (MacOS X) + INSTALL_SYSV="" + CUPS_SERVERBIN="$exec_prefix/libexec/cups" + ;; + *) + # All others + INSTALL_SYSV="install-sysv" + CUPS_SERVERBIN="$exec_prefix/lib/cups" + ;; +esac + +AC_DEFINE_UNQUOTED(CUPS_SERVERBIN, "$CUPS_SERVERBIN") +AC_SUBST(CUPS_SERVERBIN) +AC_SUBST(INSTALL_SYSV) + +# Configuration files +CUPS_SERVERROOT="$sysconfdir/cups" +AC_DEFINE_UNQUOTED(CUPS_SERVERROOT, "$sysconfdir/cups") +AC_SUBST(CUPS_SERVERROOT) + +# Transient run-time state +case "$uname" in + Darwin*) + # Darwin (Mac OS X) + CUPS_STATEDIR="$CUPS_SERVERROOT" + ;; + *) + # All others + CUPS_STATEDIR="$localstatedir/run/cups" + ;; +esac +AC_DEFINE_UNQUOTED(CUPS_STATEDIR, "$CUPS_STATEDIR") +AC_SUBST(CUPS_STATEDIR) + +dnl +dnl End of "$Id: cups-directories.m4 9771 2011-05-12 05:21:56Z mike $". +dnl diff --git a/config-scripts/cups-dnssd.m4 b/config-scripts/cups-dnssd.m4 new file mode 100644 index 0000000000..e6bb8555da --- /dev/null +++ b/config-scripts/cups-dnssd.m4 @@ -0,0 +1,64 @@ +dnl +dnl "$Id: cups-dnssd.m4 9771 2011-05-12 05:21:56Z mike $" +dnl +dnl DNS Service Discovery (aka Bonjour) stuff for CUPS. +dnl +dnl Copyright 2007-2011 by Apple Inc. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +AC_ARG_ENABLE(dnssd, [ --disable-dnssd disable DNS Service Discovery support]) +AC_ARG_WITH(dnssd-libs, [ --with-dnssd-libs set directory for DNS Service Discovery library], + LDFLAGS="-L$withval $LDFLAGS" + DSOFLAGS="-L$withval $DSOFLAGS",) +AC_ARG_WITH(dnssd-includes, [ --with-dnssd-includes set directory for DNS Service Discovery includes], + CFLAGS="-I$withval $CFLAGS" + CPPFLAGS="-I$withval $CPPFLAGS",) + +DNSSDLIBS="" +DNSSD_BACKEND="" + +if test x$enable_dnssd != xno; then + AC_CHECK_HEADER(dns_sd.h, [ + case "$uname" in + Darwin*) + # Darwin and MacOS X... + AC_DEFINE(HAVE_DNSSD) + AC_DEFINE(HAVE_COREFOUNDATION) + AC_DEFINE(HAVE_SYSTEMCONFIGURATION) + DNSSDLIBS="-framework CoreFoundation -framework SystemConfiguration" + DNSSD_BACKEND="dnssd" + ;; + *) + # All others... + AC_MSG_CHECKING(for current version of dns_sd library) + SAVELIBS="$LIBS" + LIBS="$LIBS -ldns_sd" + AC_TRY_COMPILE([#include ], + [int constant = kDNSServiceFlagsShareConnection; + unsigned char txtRecord[100]; + uint8_t valueLen; + TXTRecordGetValuePtr(sizeof(txtRecord), + txtRecord, "value", &valueLen);], + AC_MSG_RESULT(yes) + AC_DEFINE(HAVE_DNSSD) + DNSSDLIBS="-ldns_sd" + DNSSD_BACKEND="dnssd", + AC_MSG_RESULT(no)) + LIBS="$SAVELIBS" + ;; + esac + ]) +fi + +AC_SUBST(DNSSDLIBS) +AC_SUBST(DNSSD_BACKEND) + +dnl +dnl End of "$Id: cups-dnssd.m4 9771 2011-05-12 05:21:56Z mike $". +dnl diff --git a/config-scripts/cups-gssapi.m4 b/config-scripts/cups-gssapi.m4 new file mode 100644 index 0000000000..ca8435e9f7 --- /dev/null +++ b/config-scripts/cups-gssapi.m4 @@ -0,0 +1,171 @@ +dnl +dnl "$Id: cups-gssapi.m4 10137 2011-12-02 04:04:49Z mike $" +dnl +dnl GSSAPI/Kerberos library detection for CUPS. +dnl +dnl Copyright 2007-2011 by Apple Inc. +dnl Copyright 2006-2007 by Easy Software Products. +dnl +dnl This file contains Kerberos support code, copyright 2006 by +dnl Jelmer Vernooij. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +AC_ARG_ENABLE(gssapi, [ --disable-gssapi disable GSSAPI support]) + +LIBGSSAPI="" +AC_SUBST(LIBGSSAPI) + +if test x$enable_gssapi != xno; then + AC_PATH_PROG(KRB5CONFIG, krb5-config) + if test "x$KRB5CONFIG" != x; then + case "$uname" in + Darwin) + # Mac OS X weak-links to the Kerberos framework... + LIBGSSAPI="-weak_framework Kerberos" + AC_MSG_CHECKING(for GSS framework) + if test -d /System/Library/Frameworks/GSS.framework; then + AC_MSG_RESULT(yes) + LIBGSSAPI="$LIBGSSAPI -weak_framework GSS" + else + AC_MSG_RESULT(no) + fi + ;; + SunOS*) + # Solaris has a non-standard krb5-config, don't use it! + AC_CHECK_LIB(gss, gss_display_status, + AC_DEFINE(HAVE_GSSAPI, 1, [Whether GSSAPI is available]) + CFLAGS="`$KRB5CONFIG --cflags` $CFLAGS" + CPPFLAGS="`$KRB5CONFIG --cflags` $CPPFLAGS" + LIBGSSAPI="-lgss `$KRB5CONFIG --libs`") + ;; + *) + # Other platforms just ask for GSSAPI + CFLAGS="`$KRB5CONFIG --cflags gssapi` $CFLAGS" + CPPFLAGS="`$KRB5CONFIG --cflags gssapi` $CPPFLAGS" + LIBGSSAPI="`$KRB5CONFIG --libs gssapi`" + ;; + esac + AC_DEFINE(HAVE_GSSAPI, 1, [Whether GSSAPI is available]) + else + # Check for vendor-specific implementations... + case "$uname" in + HP-UX*) + AC_CHECK_LIB(gss, gss_display_status, + AC_DEFINE(HAVE_GSSAPI, 1, [Whether GSSAPI is available]) + LIBGSSAPI="-lgss -lgssapi_krb5") + ;; + SunOS*) + AC_CHECK_LIB(gss, gss_display_status, + AC_DEFINE(HAVE_GSSAPI, 1, [Whether GSSAPI is available]) + LIBGSSAPI="-lgss") + ;; + esac + fi + + if test "x$LIBGSSAPI" != x; then + AC_CHECK_HEADER(krb5.h, AC_DEFINE(HAVE_KRB5_H)) + if test -d /System/Library/Frameworks/GSS.framework; then + gssdir="/System/Library/Frameworks/GSS.framework" + AC_MSG_CHECKING(for GSS/gssapi.h presence) + if test -f $gssdir/Headers/gssapi.h; then + AC_DEFINE(HAVE_GSS_GSSAPI_H) + AC_MSG_RESULT(yes) + else + AC_MSG_RESULT(no) + fi + AC_MSG_CHECKING(for GSS/gssapi_generic.h presence) + if test -f $gssdir/Headers/gssapi_generic.h; then + AC_DEFINE(HAVE_GSSAPI_GENERIC_H) + AC_MSG_RESULT(yes) + else + AC_MSG_RESULT(no) + fi + AC_MSG_CHECKING(for GSS/gssapi_krb5.h presence) + if test -f $gssdir/Headers/gssapi_krb5.h; then + AC_DEFINE(HAVE_GSSAPI_KRB5_H) + AC_MSG_RESULT(yes) + else + AC_MSG_RESULT(no) + fi + AC_MSG_CHECKING(for GSS/gssapi_spi.h presence) + if test -f $gssdir/PrivateHeaders/gssapi_spi.h; then + AC_MSG_RESULT(yes) + AC_MSG_CHECKING(for GSS/gssapi_spi.h usability) + if test -s $gssdir/PrivateHeaders/gssapi_spi.h; then + AC_MSG_RESULT(yes) + AC_DEFINE(HAVE_GSS_GSSAPI_SPI_H) + else + AC_MSG_RESULT(no) + fi + else + AC_MSG_RESULT(no) + if test $uversion -ge 110; then + # Broken public headers in 10.7... + AC_MSG_ERROR(Run 'sudo mkdir -p $gssdir/PrivateHeaders' and 'sudo touch $gssdir/PrivateHeaders/gssapi_spi.h' to build CUPS.) + fi + fi + else + AC_CHECK_HEADER(gssapi.h, AC_DEFINE(HAVE_GSSAPI_H)) + AC_CHECK_HEADER(gssapi/gssapi.h, AC_DEFINE(HAVE_GSSAPI_GSSAPI_H)) + AC_CHECK_HEADER(gssapi/gssapi_generic.h, AC_DEFINE(HAVE_GSSAPI_GENERIC_H)) + AC_CHECK_HEADER(gssapi/gssapi_krb5.h, AC_DEFINE(HAVE_GSSAPI_KRB5_H)) + fi + + SAVELIBS="$LIBS" + LIBS="$LIBS $LIBGSSAPI" + + AC_CHECK_FUNC(__ApplePrivate_gss_acquire_cred_ex_f, + AC_DEFINE(HAVE_GSS_ACQUIRE_CRED_EX_F)) + + AC_MSG_CHECKING(for GSS_C_NT_HOSTBASED_SERVICE) + if test x$ac_cv_header_gssapi_gssapi_h = xyes; then + AC_TRY_COMPILE([ #include ], + [ gss_OID foo = GSS_C_NT_HOSTBASED_SERVICE; ], + AC_DEFINE(HAVE_GSS_C_NT_HOSTBASED_SERVICE) + AC_MSG_RESULT(yes), + AC_MSG_RESULT(no)) + elif test x$ac_cv_header_gss_gssapi_h = xyes; then + AC_TRY_COMPILE([ #include ], + [ gss_OID foo = GSS_C_NT_HOSTBASED_SERVICE; ], + AC_DEFINE(HAVE_GSS_C_NT_HOSTBASED_SERVICE) + AC_MSG_RESULT(yes), + AC_MSG_RESULT(no)) + else + AC_TRY_COMPILE([ #include ], + [ gss_OID foo = GSS_C_NT_HOSTBASED_SERVICE; ], + AC_DEFINE(HAVE_GSS_C_NT_HOSTBASED_SERVICE) + AC_MSG_RESULT(yes), + AC_MSG_RESULT(no)) + fi + + LIBS="$SAVELIBS" + fi +fi + +dnl Default GSS service name... +AC_ARG_WITH(gssservicename, [ --with-gssservicename set default gss service name], + default_gssservicename="$withval", + default_gssservicename="default") + +if test x$default_gssservicename != xno; then + if test "x$default_gssservicename" = "xdefault"; then + CUPS_DEFAULT_GSSSERVICENAME="host" + else + CUPS_DEFAULT_GSSSERVICENAME="$default_gssservicename" + fi +else + CUPS_DEFAULT_GSSSERVICENAME="" +fi + +AC_SUBST(CUPS_DEFAULT_GSSSERVICENAME) +AC_DEFINE_UNQUOTED(CUPS_DEFAULT_GSSSERVICENAME, "$CUPS_DEFAULT_GSSSERVICENAME") + +dnl +dnl End of "$Id: cups-gssapi.m4 10137 2011-12-02 04:04:49Z mike $". +dnl diff --git a/config-scripts/cups-largefile.m4 b/config-scripts/cups-largefile.m4 new file mode 100644 index 0000000000..c4eab9a84e --- /dev/null +++ b/config-scripts/cups-largefile.m4 @@ -0,0 +1,52 @@ +dnl +dnl "$Id: cups-largefile.m4 9771 2011-05-12 05:21:56Z mike $" +dnl +dnl Large file support stuff for CUPS. +dnl +dnl Copyright 2007-2011 by Apple Inc. +dnl Copyright 1997-2005 by Easy Software Products, all rights reserved. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +dnl Check for largefile support... +AC_SYS_LARGEFILE + +dnl Define largefile options as needed... +LARGEFILE="" +if test x$enable_largefile != xno; then + LARGEFILE="-D_LARGEFILE_SOURCE -D_LARGEFILE64_SOURCE" + + if test x$ac_cv_sys_large_files = x1; then + LARGEFILE="$LARGEFILE -D_LARGE_FILES" + fi + + if test x$ac_cv_sys_file_offset_bits = x64; then + LARGEFILE="$LARGEFILE -D_FILE_OFFSET_BITS=64" + fi +fi +AC_SUBST(LARGEFILE) + +dnl Check for "long long" support... +AC_CACHE_CHECK(for long long int, ac_cv_c_long_long, + [if test "$GCC" = yes; then + ac_cv_c_long_long=yes + else + AC_TRY_COMPILE(,[long long int i;], + ac_cv_c_long_long=yes, + ac_cv_c_long_long=no) + fi]) + +if test $ac_cv_c_long_long = yes; then + AC_DEFINE(HAVE_LONG_LONG) +fi + +AC_CHECK_FUNC(strtoll, AC_DEFINE(HAVE_STRTOLL)) + +dnl +dnl End of "$Id: cups-largefile.m4 9771 2011-05-12 05:21:56Z mike $". +dnl diff --git a/config-scripts/cups-launchd.m4 b/config-scripts/cups-launchd.m4 new file mode 100644 index 0000000000..5da5e077e7 --- /dev/null +++ b/config-scripts/cups-launchd.m4 @@ -0,0 +1,43 @@ +dnl +dnl "$Id: cups-launchd.m4 8983 2010-02-13 02:20:23Z mike $" +dnl +dnl launchd stuff for CUPS. +dnl +dnl Copyright 2007-2010 by Apple Inc. +dnl Copyright 1997-2005 by Easy Software Products, all rights reserved. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + + +AC_ARG_ENABLE(launchd, [ --disable-launchd disable launchd support]) + +DEFAULT_LAUNCHD_CONF="" +LAUNCHDLIBS="" + +if test x$enable_launchd != xno; then + AC_CHECK_FUNC(launch_msg, AC_DEFINE(HAVE_LAUNCHD)) + AC_CHECK_HEADER(launch.h, AC_DEFINE(HAVE_LAUNCH_H)) + + case "$uname" in + Darwin*) + # Darwin, MacOS X + DEFAULT_LAUNCHD_CONF="/System/Library/LaunchDaemons/org.cups.cupsd.plist" + # liblaunch is already part of libSystem + ;; + *) + # All others; this test will need to be updated + ;; + esac +fi + +AC_SUBST(DEFAULT_LAUNCHD_CONF) +AC_SUBST(LAUNCHDLIBS) + +dnl +dnl End of "$Id: cups-launchd.m4 8983 2010-02-13 02:20:23Z mike $". +dnl diff --git a/config-scripts/cups-libtool.m4 b/config-scripts/cups-libtool.m4 new file mode 100644 index 0000000000..5b95b39971 --- /dev/null +++ b/config-scripts/cups-libtool.m4 @@ -0,0 +1,39 @@ +dnl +dnl "$Id: cups-libtool.m4 9771 2011-05-12 05:21:56Z mike $" +dnl +dnl Libtool stuff for CUPS. +dnl +dnl Copyright 2007-2011 by Apple Inc. +dnl Copyright 1997-2005 by Easy Software Products, all rights reserved. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +AC_ARG_ENABLE(libtool_unsupported, [ --enable-libtool-unsupported + build with libtool (UNSUPPORTED!)], + [if test x$enable_libtool_unsupported != xno; then + LIBTOOL="$enable_libtool_unsupported" + enable_shared=no + echo "WARNING: libtool is not supported or endorsed by Apple Inc." + echo " WE DO NOT PROVIDE SUPPORT FOR LIBTOOL PROBLEMS." + else + LIBTOOL="" + fi]) + +AC_SUBST(LIBTOOL) + +if test x$LIBTOOL != x; then + LIBCUPS="libcups.la" + LIBCUPSIMAGE="libcupsimage.la" + LINKCUPS="../cups/\$(LIBCUPS)" + LINKCUPSIMAGE="../filter/\$(LIBCUPSIMAGE)" + DSO="\$(CC)" +fi + +dnl +dnl End of "$Id: cups-libtool.m4 9771 2011-05-12 05:21:56Z mike $". +dnl diff --git a/config-scripts/cups-manpages.m4 b/config-scripts/cups-manpages.m4 new file mode 100644 index 0000000000..3a7afd7304 --- /dev/null +++ b/config-scripts/cups-manpages.m4 @@ -0,0 +1,96 @@ +dnl +dnl "$Id: cups-manpages.m4 9771 2011-05-12 05:21:56Z mike $" +dnl +dnl Manpage stuff for CUPS. +dnl +dnl Copyright 2007-2011 by Apple Inc. +dnl Copyright 1997-2006 by Easy Software Products, all rights reserved. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +dnl Fix "mandir" variable... +if test "$mandir" = "\${datarootdir}/man" -a "$prefix" = "/"; then + # New GNU "standards" break previous ones, so make sure we use + # the right default location for the operating system... + mandir="\${prefix}/man" +fi + +if test "$mandir" = "\${prefix}/man" -a "$prefix" = "/"; then + case "$uname" in + Darwin* | Linux | GNU | *BSD* | AIX*) + # Darwin, MacOS X, Linux, GNU HURD, *BSD, and AIX + mandir="/usr/share/man" + AMANDIR="/usr/share/man" + PMANDIR="/usr/share/man" + ;; + IRIX) + # SGI IRIX + mandir="/usr/share/catman/u_man" + AMANDIR="/usr/share/catman/a_man" + PMANDIR="/usr/share/catman/p_man" + ;; + *) + # All others + mandir="/usr/man" + AMANDIR="/usr/man" + PMANDIR="/usr/man" + ;; + esac +else + AMANDIR="$mandir" + PMANDIR="$mandir" +fi + +AC_SUBST(AMANDIR) +AC_SUBST(PMANDIR) + +dnl Setup manpage extensions... +case "$uname" in + IRIX*) + # SGI IRIX + MAN1EXT=1 + MAN5EXT=5 + MAN7EXT=7 + MAN8EXT=1m + MAN8DIR=1 + ;; + SunOS* | HP-UX*) + # Solaris and HP-UX + MAN1EXT=1 + MAN5EXT=5 + MAN7EXT=7 + MAN8EXT=1m + MAN8DIR=1m + ;; + Linux* | GNU* | Darwin*) + # Linux, GNU Hurd, and Mac OS X + MAN1EXT=1.gz + MAN5EXT=5.gz + MAN7EXT=7.gz + MAN8EXT=8.gz + MAN8DIR=8 + ;; + *) + # All others + MAN1EXT=1 + MAN5EXT=5 + MAN7EXT=7 + MAN8EXT=8 + MAN8DIR=8 + ;; +esac + +AC_SUBST(MAN1EXT) +AC_SUBST(MAN5EXT) +AC_SUBST(MAN7EXT) +AC_SUBST(MAN8EXT) +AC_SUBST(MAN8DIR) + +dnl +dnl End of "$Id: cups-manpages.m4 9771 2011-05-12 05:21:56Z mike $". +dnl diff --git a/config-scripts/cups-network.m4 b/config-scripts/cups-network.m4 new file mode 100644 index 0000000000..9b591ed12b --- /dev/null +++ b/config-scripts/cups-network.m4 @@ -0,0 +1,74 @@ +dnl +dnl "$Id: cups-network.m4 9771 2011-05-12 05:21:56Z mike $" +dnl +dnl Networking stuff for CUPS. +dnl +dnl Copyright 2007-2011 by Apple Inc. +dnl Copyright 1997-2005 by Easy Software Products, all rights reserved. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +AC_CHECK_HEADER(resolv.h,AC_DEFINE(HAVE_RESOLV_H)) +AC_SEARCH_LIBS(socket, socket) +AC_SEARCH_LIBS(gethostbyaddr, nsl) +AC_SEARCH_LIBS(getifaddrs, nsl, AC_DEFINE(HAVE_GETIFADDRS)) +AC_SEARCH_LIBS(hstrerror, nsl socket resolv, AC_DEFINE(HAVE_HSTRERROR)) +AC_SEARCH_LIBS(rresvport_af, nsl, AC_DEFINE(HAVE_RRESVPORT_AF)) +AC_SEARCH_LIBS(__res_init, resolv bind, AC_DEFINE(HAVE_RES_INIT), + AC_SEARCH_LIBS(res_9_init, resolv bind, AC_DEFINE(HAVE_RES_INIT), + AC_SEARCH_LIBS(res_init, resolv bind, AC_DEFINE(HAVE_RES_INIT)))) + +# Tru64 5.1b leaks file descriptors with these functions; disable until +# we can come up with a test for this... +if test "$uname" != "OSF1"; then + AC_SEARCH_LIBS(getaddrinfo, nsl, AC_DEFINE(HAVE_GETADDRINFO)) + AC_SEARCH_LIBS(getnameinfo, nsl, AC_DEFINE(HAVE_GETNAMEINFO)) +fi + +AC_CHECK_MEMBER(struct sockaddr.sa_len,,, [#include ]) +AC_CHECK_HEADER(sys/sockio.h, AC_DEFINE(HAVE_SYS_SOCKIO_H)) + +CUPS_DEFAULT_DOMAINSOCKET="" + +dnl Domain socket support... +AC_ARG_WITH(domainsocket, [ --with-domainsocket set unix domain socket name], + default_domainsocket="$withval", + default_domainsocket="") + +if test x$enable_domainsocket != xno -a x$default_domainsocket != xno; then + if test "x$default_domainsocket" = x; then + case "$uname" in + Darwin*) + # Darwin and MaxOS X do their own thing... + CUPS_DEFAULT_DOMAINSOCKET="$localstatedir/run/cupsd" + ;; + *) + # All others use FHS standard... + CUPS_DEFAULT_DOMAINSOCKET="$CUPS_STATEDIR/cups.sock" + ;; + esac + else + CUPS_DEFAULT_DOMAINSOCKET="$default_domainsocket" + fi + + CUPS_LISTEN_DOMAINSOCKET="Listen $CUPS_DEFAULT_DOMAINSOCKET" + + AC_DEFINE_UNQUOTED(CUPS_DEFAULT_DOMAINSOCKET, "$CUPS_DEFAULT_DOMAINSOCKET") +else + CUPS_LISTEN_DOMAINSOCKET="" +fi + +AC_SUBST(CUPS_DEFAULT_DOMAINSOCKET) +AC_SUBST(CUPS_LISTEN_DOMAINSOCKET) + +AC_CHECK_HEADERS(AppleTalk/at_proto.h,AC_DEFINE(HAVE_APPLETALK_AT_PROTO_H),, + [#include ]) + +dnl +dnl End of "$Id: cups-network.m4 9771 2011-05-12 05:21:56Z mike $". +dnl diff --git a/config-scripts/cups-opsys.m4 b/config-scripts/cups-opsys.m4 new file mode 100644 index 0000000000..0effd6284c --- /dev/null +++ b/config-scripts/cups-opsys.m4 @@ -0,0 +1,35 @@ +dnl +dnl "$Id: cups-opsys.m4 9771 2011-05-12 05:21:56Z mike $" +dnl +dnl Operating system stuff for CUPS. +dnl +dnl Copyright 2007-2011 by Apple Inc. +dnl Copyright 1997-2006 by Easy Software Products, all rights reserved. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +dnl Get the operating system, version number, and architecture... +uname=`uname` +uversion=`uname -r | sed -e '1,$s/^[[^0-9]]*\([[0-9]]*\)\.\([[0-9]]*\).*/\1\2/'` +uarch=`uname -m` + +case "$uname" in + GNU* | GNU/*) + uname="GNU" + ;; + IRIX*) + uname="IRIX" + ;; + Linux*) + uname="Linux" + ;; +esac + +dnl +dnl "$Id: cups-opsys.m4 9771 2011-05-12 05:21:56Z mike $" +dnl diff --git a/config-scripts/cups-pam.m4 b/config-scripts/cups-pam.m4 new file mode 100644 index 0000000000..9a770bfcd6 --- /dev/null +++ b/config-scripts/cups-pam.m4 @@ -0,0 +1,102 @@ +dnl +dnl "$Id: cups-pam.m4 9771 2011-05-12 05:21:56Z mike $" +dnl +dnl PAM stuff for CUPS. +dnl +dnl Copyright 2007-2011 by Apple Inc. +dnl Copyright 1997-2005 by Easy Software Products, all rights reserved. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +AC_ARG_ENABLE(pam, [ --disable-pam disable PAM support]) +AC_ARG_WITH(pam_module, [ --with-pam-module specify the PAM module to use]) + +dnl Don't use PAM with AIX... +if test $uname = AIX; then + enable_pam=no +fi + +PAMDIR="" +PAMFILE="pam.std" +PAMLIBS="" +PAMMOD="pam_unknown.so" +PAMMODAUTH="pam_unknown.so" + +if test x$enable_pam != xno; then + SAVELIBS="$LIBS" + + AC_CHECK_LIB(dl,dlopen) + AC_CHECK_LIB(pam,pam_start) + AC_CHECK_LIB(pam,pam_set_item,AC_DEFINE(HAVE_PAM_SET_ITEM)) + AC_CHECK_LIB(pam,pam_setcred,AC_DEFINE(HAVE_PAM_SETCRED)) + AC_CHECK_HEADER(security/pam_appl.h) + if test x$ac_cv_header_security_pam_appl_h != xyes; then + AC_CHECK_HEADER(pam/pam_appl.h, + AC_DEFINE(HAVE_PAM_PAM_APPL_H)) + fi + + if test x$ac_cv_lib_pam_pam_start != xno; then + # Set the necessary libraries for PAM... + if test x$ac_cv_lib_dl_dlopen != xno; then + PAMLIBS="-lpam -ldl" + else + PAMLIBS="-lpam" + fi + + # Find the PAM configuration directory, if any... + for dir in /private/etc/pam.d /etc/pam.d; do + if test -d $dir; then + PAMDIR=$dir + break; + fi + done + fi + + LIBS="$SAVELIBS" + + case "$uname" in + Darwin*) + # Darwin/Mac OS X + if test "x$with_pam_module" != x; then + PAMFILE="pam.$with_pam_module" + elif test -f /usr/lib/pam/pam_opendirectory.so.2; then + PAMFILE="pam.opendirectory" + else + PAMFILE="pam.securityserver" + fi + ;; + + *) + # All others; this test might need to be updated + # as Linux distributors move things around... + if test "x$with_pam_module" != x; then + PAMMOD="pam_${with_pam_module}.so" + elif test -f /lib/security/pam_unix2.so; then + PAMMOD="pam_unix2.so" + elif test -f /lib/security/pam_unix.so; then + PAMMOD="pam_unix.so" + fi + + if test "x$PAMMOD" = xpam_unix.so; then + PAMMODAUTH="$PAMMOD shadow nodelay" + else + PAMMODAUTH="$PAMMOD nodelay" + fi + ;; + esac +fi + +AC_SUBST(PAMDIR) +AC_SUBST(PAMFILE) +AC_SUBST(PAMLIBS) +AC_SUBST(PAMMOD) +AC_SUBST(PAMMODAUTH) + +dnl +dnl End of "$Id: cups-pam.m4 9771 2011-05-12 05:21:56Z mike $". +dnl diff --git a/config-scripts/cups-poll.m4 b/config-scripts/cups-poll.m4 new file mode 100644 index 0000000000..46d1af5b4a --- /dev/null +++ b/config-scripts/cups-poll.m4 @@ -0,0 +1,22 @@ +dnl +dnl "$Id: cups-poll.m4 9771 2011-05-12 05:21:56Z mike $" +dnl +dnl Select/poll stuff for CUPS. +dnl +dnl Copyright 2007-2011 by Apple Inc. +dnl Copyright 2006 by Easy Software Products, all rights reserved. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +AC_CHECK_FUNC(poll, AC_DEFINE(HAVE_POLL)) +AC_CHECK_FUNC(epoll_create, AC_DEFINE(HAVE_EPOLL)) +AC_CHECK_FUNC(kqueue, AC_DEFINE(HAVE_KQUEUE)) + +dnl +dnl End of "$Id: cups-poll.m4 9771 2011-05-12 05:21:56Z mike $". +dnl diff --git a/config-scripts/cups-scripting.m4 b/config-scripts/cups-scripting.m4 new file mode 100644 index 0000000000..75abd7e264 --- /dev/null +++ b/config-scripts/cups-scripting.m4 @@ -0,0 +1,89 @@ +dnl +dnl "$Id: cups-scripting.m4 9927 2011-08-27 09:28:30Z mike $" +dnl +dnl Scripting configuration stuff for CUPS. +dnl +dnl Copyright 2007-2010 by Apple Inc. +dnl Copyright 1997-2006 by Easy Software Products, all rights reserved. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +dnl Do we have Java? +AC_ARG_WITH(java, [ --with-java set Java interpreter for web interfaces ], + CUPS_JAVA="$withval", + CUPS_JAVA="") + +if test "x$CUPS_JAVA" = x; then + AC_PATH_PROG(JAVA,java) + CUPS_JAVA="$JAVA" +fi + +AC_DEFINE_UNQUOTED(CUPS_JAVA, "$CUPS_JAVA") + +if test "x$CUPS_JAVA" != x; then + AC_DEFINE(HAVE_JAVA) +fi + +dnl Do we have Perl? +AC_ARG_WITH(perl, [ --with-perl set Perl interpreter for web interfaces ], + CUPS_PERL="$withval", + CUPS_PERL="") + +if test "x$CUPS_PERL" = x; then + AC_PATH_PROG(PERL,perl) + CUPS_PERL="$PERL" +fi + +AC_DEFINE_UNQUOTED(CUPS_PERL, "$CUPS_PERL") + +if test "x$CUPS_PERL" != x; then + AC_DEFINE(HAVE_PERL) +fi + +dnl Do we have PHP? +AC_ARG_WITH(php, [ --with-php set PHP interpreter for web interfaces ], + CUPS_PHP="$withval", + CUPS_PHP="") + +if test "x$CUPS_PHP" = x; then + AC_PATH_PROG(PHPCGI,php-cgi) + if test "x$PHPCGI" = x; then + AC_PATH_PROG(PHP,php) + CUPS_PHP="$PHP" + else + CUPS_PHP="$PHPCGI" + fi +fi + +AC_DEFINE_UNQUOTED(CUPS_PHP, "$CUPS_PHP") + +if test "x$CUPS_PHP" = x; then + CUPS_PHP="no" +else + AC_DEFINE(HAVE_PHP) +fi + +dnl Do we have Python? +AC_ARG_WITH(python, [ --with-python set Python interpreter for web interfaces ], + CUPS_PYTHON="$withval", + CUPS_PYTHON="") + +if test "x$CUPS_PYTHON" = x; then + AC_PATH_PROG(PYTHON,python) + CUPS_PYTHON="$PYTHON" +fi + +AC_DEFINE_UNQUOTED(CUPS_PYTHON, "$CUPS_PYTHON") + +if test "x$CUPS_PYTHON" != x; then + AC_DEFINE(HAVE_PYTHON) +fi + +dnl +dnl End of "$Id: cups-scripting.m4 9927 2011-08-27 09:28:30Z mike $". +dnl diff --git a/config-scripts/cups-sharedlibs.m4 b/config-scripts/cups-sharedlibs.m4 new file mode 100644 index 0000000000..09084c6fde --- /dev/null +++ b/config-scripts/cups-sharedlibs.m4 @@ -0,0 +1,238 @@ +dnl +dnl "$Id: cups-sharedlibs.m4 10190 2012-01-20 16:22:58Z mike $" +dnl +dnl Shared library support for CUPS. +dnl +dnl Copyright 2007-2012 by Apple Inc. +dnl Copyright 1997-2005 by Easy Software Products, all rights reserved. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +PICFLAG=1 +DSOFLAGS="${DSOFLAGS:=}" + +AC_ARG_ENABLE(shared, [ --disable-shared do not create shared libraries]) + +cupsbase="cups" +LIBCUPSBASE="lib$cupsbase" +LIBCUPSSTATIC="lib$cupsbase.a" + +if test x$enable_shared != xno; then + case "$uname" in + SunOS*) + LIBCUPS="lib$cupsbase.so.2" + LIBCUPSCGI="libcupscgi.so.1" + LIBCUPSIMAGE="libcupsimage.so.2" + LIBCUPSMIME="libcupsmime.so.1" + LIBCUPSPPDC="libcupsppdc.so.1" + DSO="\$(CC)" + DSOXX="\$(CXX)" + DSOFLAGS="$DSOFLAGS -Wl,-h\`basename \$@\` -G \$(OPTIM)" + ;; + UNIX_S*) + LIBCUPS="lib$cupsbase.so.2" + LIBCUPSCGI="libcupscgi.so.1" + LIBCUPSIMAGE="libcupsimage.so.2" + LIBCUPSMIME="libcupsmime.so.1" + LIBCUPSPPDC="libcupsppdc.so.1" + DSO="\$(CC)" + DSOXX="\$(CXX)" + DSOFLAGS="$DSOFLAGS -Wl,-h,\`basename \$@\` -G \$(OPTIM)" + ;; + HP-UX*) + case "$uarch" in + ia64) + LIBCUPS="lib$cupsbase.so.2" + LIBCUPSCGI="libcupscgi.so.1" + LIBCUPSIMAGE="libcupsimage.so.2" + LIBCUPSMIME="libcupsmime.so.1" + LIBCUPSPPDC="libcupsppdc.so.1" + DSO="\$(CC)" + DSOXX="\$(CXX)" + DSOFLAGS="$DSOFLAGS -Wl,-b,-z,+h,\`basename \$@\`" + ;; + *) + LIBCUPS="lib$cupsbase.sl.2" + LIBCUPSCGI="libcupscgi.sl.1" + LIBCUPSIMAGE="libcupsimage.sl.2" + LIBCUPSMIME="libcupsmime.sl.1" + LIBCUPSPPDC="libcupsppdc.sl.1" + DSO="\$(LD)" + DSOXX="\$(LD)" + DSOFLAGS="$DSOFLAGS -b -z +h \`basename \$@\`" + ;; + esac + ;; + IRIX) + LIBCUPS="lib$cupsbase.so.2" + LIBCUPSCGI="libcupscgi.so.1" + LIBCUPSIMAGE="libcupsimage.so.2" + LIBCUPSMIME="libcupsmime.so.1" + LIBCUPSPPDC="libcupsppdc.so.1" + DSO="\$(CC)" + DSOXX="\$(CXX)" + DSOFLAGS="$DSOFLAGS -set_version,sgi2.6,-soname,\`basename \$@\` -shared \$(OPTIM)" + ;; + OSF1* | Linux | GNU | *BSD*) + LIBCUPS="lib$cupsbase.so.2" + LIBCUPSCGI="libcupscgi.so.1" + LIBCUPSIMAGE="libcupsimage.so.2" + LIBCUPSMIME="libcupsmime.so.1" + LIBCUPSPPDC="libcupsppdc.so.1" + DSO="\$(CC)" + DSOXX="\$(CXX)" + DSOFLAGS="$DSOFLAGS -Wl,-soname,\`basename \$@\` -shared \$(OPTIM)" + ;; + Darwin*) + LIBCUPS="lib$cupsbase.2.dylib" + LIBCUPSCGI="libcupscgi.1.dylib" + LIBCUPSIMAGE="libcupsimage.2.dylib" + LIBCUPSMIME="libcupsmime.1.dylib" + LIBCUPSPPDC="libcupsppdc.1.dylib" + DSO="\$(CC)" + DSOXX="\$(CXX)" + DSOFLAGS="$DSOFLAGS -dynamiclib -single_module -lc" + ;; + AIX*) + LIBCUPS="lib${cupsbase}_s.a" + LIBCUPSBASE="${cupsbase}_s" + LIBCUPSCGI="libcupscgi_s.a" + LIBCUPSIMAGE="libcupsimage_s.a" + LIBCUPSMIME="libcupsmime_s.a" + LIBCUPSPPDC="libcupsppdc_s.a" + DSO="\$(CC)" + DSOXX="\$(CXX)" + DSOFLAGS="$DSOFLAGS -Wl,-bexpall,-bM:SRE,-bnoentry,-blibpath:\$(libdir)" + ;; + *) + echo "Warning: shared libraries may not be supported. Trying -shared" + echo " option with compiler." + LIBCUPS="lib$cupsbase.so.2" + LIBCUPSCGI="libcupscgi.so.1" + LIBCUPSIMAGE="libcupsimage.so.2" + LIBCUPSMIME="libcupsmime.so.1" + LIBCUPSPPDC="libcupsppdc.so.1" + DSO="\$(CC)" + DSOXX="\$(CXX)" + DSOFLAGS="$DSOFLAGS -Wl,-soname,\`basename \$@\` -shared \$(OPTIM)" + ;; + esac +else + PICFLAG=0 + LIBCUPS="lib$cupsbase.a" + LIBCUPSCGI="libcupscgi.a" + LIBCUPSIMAGE="libcupsimage.a" + LIBCUPSMIME="libcupsmime.a" + LIBCUPSPPDC="libcupsppdc.a" + DSO=":" + DSOXX=":" +fi + +AC_SUBST(DSO) +AC_SUBST(DSOXX) +AC_SUBST(DSOFLAGS) +AC_SUBST(LIBCUPS) +AC_SUBST(LIBCUPSBASE) +AC_SUBST(LIBCUPSCGI) +AC_SUBST(LIBCUPSIMAGE) +AC_SUBST(LIBCUPSMIME) +AC_SUBST(LIBCUPSPPDC) +AC_SUBST(LIBCUPSSTATIC) + +if test x$enable_shared = xno; then + LINKCUPS="../cups/lib$cupsbase.a" + LINKCUPSIMAGE="../filter/libcupsimage.a" + + EXTLINKCUPS="-lcups" + EXTLINKCUPSIMAGE="-lcupsimage" +else + if test $uname = AIX; then + LINKCUPS="-l${cupsbase}_s" + LINKCUPSIMAGE="-lcupsimage_s" + + EXTLINKCUPS="-lcups_s" + EXTLINKCUPSIMAGE="-lcupsimage_s" + else + LINKCUPS="-l${cupsbase}" + LINKCUPSIMAGE="-lcupsimage" + + EXTLINKCUPS="-lcups" + EXTLINKCUPSIMAGE="-lcupsimage" + fi +fi + +AC_SUBST(EXTLINKCUPS) +AC_SUBST(EXTLINKCUPSIMAGE) +AC_SUBST(LINKCUPS) +AC_SUBST(LINKCUPSIMAGE) + +dnl Update libraries for DSOs... +EXPORT_LDFLAGS="" + +if test "$DSO" != ":"; then + # When using DSOs the image libraries are linked to libcupsimage.so + # rather than to the executables. This makes things smaller if you + # are using any static libraries, and it also allows us to distribute + # a single DSO rather than a bunch... + DSOLIBS="\$(LIBZ)" + IMGLIBS="" + + # Tell the run-time linkers where to find a DSO. Some platforms + # need this option, even when the library is installed in a + # standard location... + case $uname in + HP-UX*) + # HP-UX needs the path, even for /usr/lib... + case "$uarch" in + ia64) + DSOFLAGS="-Wl,+s,+b,$libdir $DSOFLAGS" + ;; + *) + DSOFLAGS="+s +b $libdir $DSOFLAGS" + ;; + esac + LDFLAGS="$LDFLAGS -Wl,+s,+b,$libdir" + EXPORT_LDFLAGS="-Wl,+s,+b,$libdir" + ;; + SunOS*) + # Solaris... + if test $exec_prefix != /usr; then + DSOFLAGS="-R$libdir $DSOFLAGS" + LDFLAGS="$LDFLAGS -R$libdir" + EXPORT_LDFLAGS="-R$libdir" + fi + ;; + *BSD*) + # *BSD... + if test $exec_prefix != /usr; then + DSOFLAGS="-Wl,-R$libdir $DSOFLAGS" + LDFLAGS="$LDFLAGS -Wl,-R$libdir" + EXPORT_LDFLAGS="-Wl,-R$libdir" + fi + ;; + IRIX | Linux | GNU) + # IRIX, Linux, and HURD... + if test $exec_prefix != /usr; then + DSOFLAGS="-Wl,-rpath,$libdir $DSOFLAGS" + LDFLAGS="$LDFLAGS -Wl,-rpath,$libdir" + EXPORT_LDFLAGS="-Wl,-rpath,$libdir" + fi + ;; + esac +else + DSOLIBS="" + IMGLIBS="\$(LIBZ)" +fi + +AC_SUBST(DSOLIBS) +AC_SUBST(IMGLIBS) +AC_SUBST(EXPORT_LDFLAGS) + +dnl +dnl End of "$Id: cups-sharedlibs.m4 10190 2012-01-20 16:22:58Z mike $". +dnl diff --git a/config-scripts/cups-ssl.m4 b/config-scripts/cups-ssl.m4 new file mode 100644 index 0000000000..5f665d3866 --- /dev/null +++ b/config-scripts/cups-ssl.m4 @@ -0,0 +1,179 @@ +dnl +dnl "$Id: cups-ssl.m4 9837 2011-06-16 20:12:16Z mike $" +dnl +dnl OpenSSL/GNUTLS stuff for CUPS. +dnl +dnl Copyright 2007-2011 by Apple Inc. +dnl Copyright 1997-2007 by Easy Software Products, all rights reserved. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +AC_ARG_ENABLE(ssl, [ --disable-ssl disable SSL/TLS support]) +AC_ARG_ENABLE(cdsassl, [ --enable-cdsassl use CDSA for SSL/TLS support, default=first]) +AC_ARG_ENABLE(gnutls, [ --enable-gnutls use GNU TLS for SSL/TLS support, default=second]) +AC_ARG_ENABLE(openssl, [ --enable-openssl use OpenSSL for SSL/TLS support, default=third]) +AC_ARG_WITH(openssl-libs, [ --with-openssl-libs set directory for OpenSSL library], + LDFLAGS="-L$withval $LDFLAGS" + DSOFLAGS="-L$withval $DSOFLAGS",) +AC_ARG_WITH(openssl-includes, [ --with-openssl-includes set directory for OpenSSL includes], + CFLAGS="-I$withval $CFLAGS" + CPPFLAGS="-I$withval $CPPFLAGS",) + +SSLFLAGS="" +SSLLIBS="" +have_ssl=0 + +if test x$enable_ssl != xno; then + dnl Look for CDSA... + if test $have_ssl = 0 -a "x$enable_cdsassl" != "xno"; then + if test $uname = Darwin; then + AC_CHECK_HEADER(Security/SecureTransport.h, [ + have_ssl=1 + AC_DEFINE(HAVE_SSL) + AC_DEFINE(HAVE_CDSASSL) + + dnl Check for the various security headers... + AC_CHECK_HEADER(Security/SecureTransportPriv.h, + AC_DEFINE(HAVE_SECURETRANSPORTPRIV_H)) + AC_CHECK_HEADER(Security/SecCertificate.h, + AC_DEFINE(HAVE_SECCERTIFICATE_H)) + AC_CHECK_HEADER(Security/SecItem.h, + AC_DEFINE(HAVE_SECITEM_H)) + AC_CHECK_HEADER(Security/SecItemPriv.h, + AC_DEFINE(HAVE_SECITEMPRIV_H),, + [#include ]) + AC_CHECK_HEADER(Security/SecPolicy.h, + AC_DEFINE(HAVE_SECPOLICY_H)) + AC_CHECK_HEADER(Security/SecPolicyPriv.h, + AC_DEFINE(HAVE_SECPOLICYPRIV_H)) + AC_CHECK_HEADER(Security/SecBasePriv.h, + AC_DEFINE(HAVE_SECBASEPRIV_H)) + AC_CHECK_HEADER(Security/SecIdentitySearchPriv.h, + AC_DEFINE(HAVE_SECIDENTITYSEARCHPRIV_H)) + + dnl Check for SSLSetProtocolVersionMax... + SAVELIBS="$LIBS" + LIBS="$LIBS -framework Security" + AC_CHECK_FUNC(SSLSetProtocolVersionMax) + LIBS="$SAVELIBS" + + dnl Check for SecCertificateCopyData.. + AC_MSG_CHECKING(for SecCertificateCopyData) + if test $uversion -ge 100; then + AC_DEFINE(HAVE_SECCERTIFICATECOPYDATA) + AC_MSG_RESULT(yes) + else + AC_MSG_RESULT(no) + fi + + dnl Check for SecIdentitySearchCreateWithPolicy... + AC_MSG_CHECKING(for SecIdentitySearchCreateWithPolicy) + if test $uversion -ge 80; then + AC_DEFINE(HAVE_SECIDENTITYSEARCHCREATEWITHPOLICY) + AC_MSG_RESULT(yes) + else + AC_MSG_RESULT(no) + fi + + dnl Check for SecPolicyCreateSSL... + AC_MSG_CHECKING(for SecPolicyCreateSSL) + if test $uversion -ge 110; then + AC_DEFINE(HAVE_SECPOLICYCREATESSL) + AC_MSG_RESULT(yes) + else + AC_MSG_RESULT(no) + fi]) + + AC_DEFINE(HAVE_CSSMERRORSTRING) + fi + fi + + dnl Then look for GNU TLS... + if test $have_ssl = 0 -a "x$enable_gnutls" != "xno" -a "x$PKGCONFIG" != x; then + AC_PATH_PROG(LIBGNUTLSCONFIG,libgnutls-config) + AC_PATH_PROG(LIBGCRYPTCONFIG,libgcrypt-config) + if $PKGCONFIG --exists gnutls; then + have_ssl=1 + SSLLIBS=`$PKGCONFIG --libs gnutls` + SSLFLAGS=`$PKGCONFIG --cflags gnutls` + AC_DEFINE(HAVE_SSL) + AC_DEFINE(HAVE_GNUTLS) + elif test "x$LIBGNUTLSCONFIG" != x; then + have_ssl=1 + SSLLIBS=`$LIBGNUTLSCONFIG --libs` + SSLFLAGS=`$LIBGNUTLSCONFIG --cflags` + AC_DEFINE(HAVE_SSL) + AC_DEFINE(HAVE_GNUTLS) + fi + + if test $have_ssl = 1; then + if $PKGCONFIG --exists gcrypt; then + SSLLIBS="$SSLLIBS `$PKGCONFIG --libs gcrypt`" + SSLFLAGS="$SSLFLAGS `$PKGCONFIG --cflags gcrypt`" + elif test "x$LIBGCRYPTCONFIG" != x; then + SSLLIBS="$SSLLIBS `$LIBGCRYPTCONFIG --libs`" + SSLFLAGS="$SSLFLAGS `$LIBGCRYPTCONFIG --cflags`" + fi + fi + fi + + dnl Check for the OpenSSL library last... + if test $have_ssl = 0 -a "x$enable_openssl" != "xno"; then + AC_CHECK_HEADER(openssl/ssl.h, + dnl Save the current libraries so the crypto stuff isn't always + dnl included... + SAVELIBS="$LIBS" + + dnl Some ELF systems can't resolve all the symbols in libcrypto + dnl if libcrypto was linked against RSAREF, and fail to link the + dnl test program correctly, even though a correct installation + dnl of OpenSSL exists. So we test the linking three times in + dnl case the RSAREF libraries are needed. + + for libcrypto in \ + "-lcrypto" \ + "-lcrypto -lrsaref" \ + "-lcrypto -lRSAglue -lrsaref" + do + AC_CHECK_LIB(ssl,SSL_new, + [have_ssl=1 + SSLFLAGS="-DOPENSSL_DISABLE_OLD_DES_SUPPORT" + SSLLIBS="-lssl $libcrypto" + AC_DEFINE(HAVE_SSL) + AC_DEFINE(HAVE_LIBSSL)],, + $libcrypto) + + if test "x${SSLLIBS}" != "x"; then + break + fi + done + + LIBS="$SAVELIBS") + fi +fi + +IPPALIASES="http" +if test $have_ssl = 1; then + AC_MSG_RESULT([ Using SSLLIBS="$SSLLIBS"]) + AC_MSG_RESULT([ Using SSLFLAGS="$SSLFLAGS"]) + IPPALIASES="http https ipps" +elif test x$enable_cdsa = xyes -o x$enable_gnutls = xyes -o x$enable_openssl = xyes; then + AC_MSG_ERROR([Unable to enable SSL support.]) +fi + +AC_SUBST(IPPALIASES) +AC_SUBST(SSLFLAGS) +AC_SUBST(SSLLIBS) + +EXPORT_SSLLIBS="$SSLLIBS" +AC_SUBST(EXPORT_SSLLIBS) + + +dnl +dnl End of "$Id: cups-ssl.m4 9837 2011-06-16 20:12:16Z mike $". +dnl diff --git a/config-scripts/cups-threads.m4 b/config-scripts/cups-threads.m4 new file mode 100644 index 0000000000..0743e3416f --- /dev/null +++ b/config-scripts/cups-threads.m4 @@ -0,0 +1,54 @@ +dnl +dnl "$Id: cups-threads.m4 9771 2011-05-12 05:21:56Z mike $" +dnl +dnl Threading stuff for CUPS. +dnl +dnl Copyright 2007-2011 by Apple Inc. +dnl Copyright 1997-2005 by Easy Software Products, all rights reserved. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +AC_ARG_ENABLE(threads, [ --disable-threads disable multi-threading support]) + +have_pthread=no +PTHREAD_FLAGS="" + +if test "x$enable_threads" != xno; then + AC_CHECK_HEADER(pthread.h, AC_DEFINE(HAVE_PTHREAD_H)) + + if test x$ac_cv_header_pthread_h = xyes; then + dnl Check various threading options for the platforms we support + for flag in -lpthreads -lpthread -pthread; do + AC_MSG_CHECKING([for pthread_create using $flag]) + SAVELIBS="$LIBS" + LIBS="$flag $LIBS" + AC_TRY_LINK([#include ], + [pthread_create(0, 0, 0, 0);], + have_pthread=yes, + LIBS="$SAVELIBS") + AC_MSG_RESULT([$have_pthread]) + + if test $have_pthread = yes; then + PTHREAD_FLAGS="-D_THREAD_SAFE -D_REENTRANT" + + # Solaris requires -D_POSIX_PTHREAD_SEMANTICS to + # be POSIX-compliant... :( + if test $uname = SunOS; then + PTHREAD_FLAGS="$PTHREAD_FLAGS -D_POSIX_PTHREAD_SEMANTICS" + fi + break + fi + done + fi +fi + +AC_SUBST(PTHREAD_FLAGS) + +dnl +dnl End of "$Id: cups-threads.m4 9771 2011-05-12 05:21:56Z mike $". +dnl diff --git a/config.h.in b/config.h.in new file mode 100644 index 0000000000..1bcf7fe0b7 --- /dev/null +++ b/config.h.in @@ -0,0 +1,751 @@ +/* + * "$Id$" + * + * Configuration file for CUPS. + * + * Copyright 2007-2012 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#ifndef _CUPS_CONFIG_H_ +#define _CUPS_CONFIG_H_ + +/* + * Version of software... + */ + +#define CUPS_SVERSION "" +#define CUPS_MINIMAL "" + + +/* + * Default user and groups... + */ + +#define CUPS_DEFAULT_USER "lp" +#define CUPS_DEFAULT_GROUP "sys" +#define CUPS_DEFAULT_SYSTEM_GROUPS "sys root system" +#define CUPS_DEFAULT_PRINTOPERATOR_AUTH "@SYSTEM" + + +/* + * Default file permissions... + */ + +#define CUPS_DEFAULT_CONFIG_FILE_PERM 0640 +#define CUPS_DEFAULT_LOG_FILE_PERM 0644 + + +/* + * Default logging settings... + */ + +#define CUPS_DEFAULT_LOG_LEVEL "warn" +#define CUPS_DEFAULT_ACCESS_LOG_LEVEL "actions" + + +/* + * Default fatal error settings... + */ + +#define CUPS_DEFAULT_FATAL_ERRORS "config" + + +/* + * Default browsing settings... + */ + +#define CUPS_DEFAULT_BROWSING 1 +#define CUPS_DEFAULT_BROWSE_LOCAL_PROTOCOLS "" +#define CUPS_DEFAULT_DEFAULT_SHARED 1 + + +/* + * Default IPP port... + */ + +#define CUPS_DEFAULT_IPP_PORT 631 + + +/* + * Default printcap file... + */ + +#define CUPS_DEFAULT_PRINTCAP "/etc/printcap" + + +/* + * Default Samba and LPD config files... + */ + +#define CUPS_DEFAULT_SMB_CONFIG_FILE "" +#define CUPS_DEFAULT_LPD_CONFIG_FILE "" + + +/* + * Default MaxCopies value... + */ + +#define CUPS_DEFAULT_MAX_COPIES 100 + + +/* + * Do we have domain socket support, and if so what is the default one? + */ + +#undef CUPS_DEFAULT_DOMAINSOCKET + + +/* + * Default WebInterface value... + */ + +#undef CUPS_DEFAULT_WEBIF + + +/* + * Where are files stored? + * + * Note: These are defaults, which can be overridden by environment + * variables at run-time... + */ + +#define CUPS_BINDIR "/usr/bin" +#define CUPS_CACHEDIR "/var/cache/cups" +#define CUPS_DATADIR "/usr/share/cups" +#define CUPS_DOCROOT "/usr/share/doc/cups" +#define CUPS_FONTPATH "/usr/share/cups/fonts" +#define CUPS_LOCALEDIR "/usr/share/locale" +#define CUPS_LOGDIR "/var/logs/cups" +#define CUPS_REQUESTS "/var/spool/cups" +#define CUPS_SBINDIR "/usr/sbin" +#define CUPS_SERVERBIN "/usr/lib/cups" +#define CUPS_SERVERROOT "/etc/cups" +#define CUPS_STATEDIR "/var/run/cups" + + +/* + * Do we have various image libraries? + */ + +#undef HAVE_LIBPNG +#undef HAVE_LIBZ +#undef HAVE_LIBJPEG +#undef HAVE_LIBTIFF + + +/* + * Do we have PAM stuff? + */ + +#ifndef HAVE_LIBPAM +#define HAVE_LIBPAM 0 +#endif /* !HAVE_LIBPAM */ + +#undef HAVE_PAM_PAM_APPL_H +#undef HAVE_PAM_SET_ITEM +#undef HAVE_PAM_SETCRED + + +/* + * Do we have ? + */ + +#undef HAVE_SHADOW_H + + +/* + * Do we have ? + */ + +#undef HAVE_CRYPT_H + + +/* + * Do we have ? + */ + +#undef HAVE_SCSI_SG_H + + +/* + * Use , , and/or ? + */ + +#undef HAVE_STRING_H +#undef HAVE_STRINGS_H +#undef HAVE_BSTRING_H + + +/* + * Do we have the long long type? + */ + +#undef HAVE_LONG_LONG + +#ifdef HAVE_LONG_LONG +# define CUPS_LLFMT "%lld" +# define CUPS_LLCAST (long long) +#else +# define CUPS_LLFMT "%ld" +# define CUPS_LLCAST (long) +#endif /* HAVE_LONG_LONG */ + + +/* + * Do we have the strtoll() function? + */ + +#undef HAVE_STRTOLL + +#ifndef HAVE_STRTOLL +# define strtoll(nptr,endptr,base) strtol((nptr), (endptr), (base)) +#endif /* !HAVE_STRTOLL */ + + +/* + * Do we have the strXXX() functions? + */ + +#undef HAVE_STRDUP +#undef HAVE_STRLCAT +#undef HAVE_STRLCPY + + +/* + * Do we have the geteuid() function? + */ + +#undef HAVE_GETEUID + + +/* + * Do we have the setpgid() function? + */ + +#undef HAVE_SETPGID + + +/* + * Do we have the vsyslog() function? + */ + +#undef HAVE_VSYSLOG + + +/* + * Do we have the (v)snprintf() functions? + */ + +#undef HAVE_SNPRINTF +#undef HAVE_VSNPRINTF + + +/* + * What signal functions to use? + */ + +#undef HAVE_SIGSET +#undef HAVE_SIGACTION + + +/* + * What wait functions to use? + */ + +#undef HAVE_WAITPID +#undef HAVE_WAIT3 + + +/* + * Do we have the mallinfo function and malloc.h? + */ + +#undef HAVE_MALLINFO +#undef HAVE_MALLOC_H + + +/* + * Do we have the POSIX ACL functions? + */ + +#undef HAVE_ACL_INIT + + +/* + * Do we have the langinfo.h header file? + */ + +#undef HAVE_LANGINFO_H + + +/* + * Which encryption libraries do we have? + */ + +#undef HAVE_CDSASSL +#undef HAVE_GNUTLS +#undef HAVE_LIBSSL +#undef HAVE_SSL + + +/* + * What Security framework headers do we have? + */ + +#undef HAVE_AUTHORIZATION_H +#undef HAVE_SECBASEPRIV_H +#undef HAVE_SECCERTIFICATE_H +#undef HAVE_SECIDENTITYSEARCHPRIV_H +#undef HAVE_SECITEM_H +#undef HAVE_SECITEMPRIV_H +#undef HAVE_SECPOLICY_H +#undef HAVE_SECPOLICYPRIV_H +#undef HAVE_SECURETRANSPORTPRIV_H + + +/* + * Do we have the SecCertificateCopyData function? + */ + +#undef HAVE_SECCERTIFICATECOPYDATA + + +/* + * Do we have the SecIdentitySearchCreateWithPolicy function? + */ + +#undef HAVE_SECIDENTITYSEARCHCREATEWITHPOLICY + + +/* + * Do we have the SecPolicyCreateSSL function? + */ + +#undef HAVE_SECPOLICYCREATESSL + + +/* + * Do we have the SecPolicyCreateSSL function? + */ + +#undef HAVE_SECPOLICYCREATESSL + + +/* + * Do we have the SSLSetProtocolVersionMax function? + */ + +#undef HAVE_SSLSETPROTOCOLVERSIONMAX + + +/* + * Do we have the cssmErrorString function? + */ + +#undef HAVE_CSSMERRORSTRING + + +/* + * Do we have the SLP library? + */ + +#undef HAVE_LIBSLP + + +/* + * Do we have an LDAP library? + */ + +#undef HAVE_LDAP +#undef HAVE_OPENLDAP +#undef HAVE_MOZILLA_LDAP +#undef HAVE_LDAP_SSL_H +#undef HAVE_LDAP_SSL +#undef HAVE_LDAP_REBIND_PROC + + +/* + * Do we have libpaper? + */ + +#undef HAVE_LIBPAPER + + +/* + * Do we have DNS Service Discovery (aka Bonjour)? + */ + +#undef HAVE_DNSSD + + +/* + * Do we have ? + */ + +#undef HAVE_SYS_IOCTL_H + + +/* + * Does the "stat" structure contain the "st_gen" member? + */ + +#undef HAVE_ST_GEN + + +/* + * Does the "tm" structure contain the "tm_gmtoff" member? + */ + +#undef HAVE_TM_GMTOFF + + +/* + * Do we have rresvport_af()? + */ + +#undef HAVE_RRESVPORT_AF + + +/* + * Do we have getaddrinfo()? + */ + +#undef HAVE_GETADDRINFO + + +/* + * Do we have getnameinfo()? + */ + +#undef HAVE_GETNAMEINFO + + +/* + * Do we have getifaddrs()? + */ + +#undef HAVE_GETIFADDRS + + +/* + * Do we have hstrerror()? + */ + +#undef HAVE_HSTRERROR + + +/* + * Do we have res_init()? + */ + +#undef HAVE_RES_INIT + + +/* + * Do we have + */ + +#undef HAVE_RESOLV_H + + +/* + * Do we have the header file? + */ + +#undef HAVE_SYS_SOCKIO_H + + +/* + * Does the sockaddr structure contain an sa_len parameter? + */ + +#undef HAVE_STRUCT_SOCKADDR_SA_LEN + + +/* + * Do we have the AIX usersec.h header file? + */ + +#undef HAVE_USERSEC_H + + +/* + * Do we have pthread support? + */ + +#undef HAVE_PTHREAD_H + + +/* + * Do we have launchd support? + */ + +#undef HAVE_LAUNCH_H +#undef HAVE_LAUNCHD + + +/* + * Various scripting languages... + */ + +#undef HAVE_JAVA +#define CUPS_JAVA "/usr/bin/java" +#undef HAVE_PERL +#define CUPS_PERL "/usr/bin/perl" +#undef HAVE_PHP +#define CUPS_PHP "/usr/bin/php" +#undef HAVE_PYTHON +#define CUPS_PYTHON "/usr/bin/python" + + +/* + * Location of the poppler/Xpdf pdftops program... + */ + +#undef HAVE_PDFTOPS +#undef HAVE_PDFTOPS_WITH_ORIGPAGESIZES +#define CUPS_PDFTOPS "/usr/bin/pdftops" + + +/* + * Location of the Ghostscript gs program... + */ + +#undef HAVE_GHOSTSCRIPT +#undef HAVE_GHOSTSCRIPT_PS2WRITE +#define CUPS_GHOSTSCRIPT "/usr/bin/gs" + + +/* + * Do we have Darwin's CoreFoundation and SystemConfiguration frameworks? + */ + +#undef HAVE_COREFOUNDATION +#undef HAVE_SYSTEMCONFIGURATION + + +/* + * Do we have CoreFoundation public and private headers? + */ + +#undef HAVE_COREFOUNDATION_H +#undef HAVE_CFPRIV_H +#undef HAVE_CFBUNDLEPRIV_H + + +/* + * Do we have ApplicationServices public headers? + */ + +#undef HAVE_APPLICATIONSERVICES_H + + +/* + * Do we have the SCDynamicStoreCopyComputerName function? + */ + +#undef HAVE_SCDYNAMICSTORECOPYCOMPUTERNAME + + +/* + * Do we have Mac OS X 10.4's mbr_XXX functions? + */ + +#undef HAVE_MEMBERSHIP_H +#undef HAVE_MEMBERSHIPPRIV_H +#undef HAVE_MBR_UID_TO_UUID + + +/* + * Do we have Darwin's notify_post header and function? + */ + +#undef HAVE_NOTIFY_H +#undef HAVE_NOTIFY_POST + + +/* + * Do we have Darwin's IOKit private headers? + */ + +#undef HAVE_IOKIT_PWR_MGT_IOPMLIBPRIVATE_H + + +/* + * Do we have DBUS? + */ + +#undef HAVE_DBUS +#undef HAVE_DBUS_MESSAGE_ITER_INIT_APPEND + + +/* + * Do we have the GSSAPI support library (for Kerberos support)? + */ + +#undef HAVE_GSS_ACQUIRE_CRED_EX_F +#undef HAVE_GSS_C_NT_HOSTBASED_SERVICE +#undef HAVE_GSS_GSSAPI_H +#undef HAVE_GSS_GSSAPI_SPI_H +#undef HAVE_GSSAPI +#undef HAVE_GSSAPI_GENERIC_H +#undef HAVE_GSSAPI_GSSAPI_H +#undef HAVE_GSSAPI_H +#undef HAVE_GSSAPI_KRB5_H +#undef HAVE_KRB5_H + + +/* + * Default GSS service name... + */ + +#define CUPS_DEFAULT_GSSSERVICENAME "" + + +/* + * Select/poll interfaces... + */ + +#undef HAVE_POLL +#undef HAVE_EPOLL +#undef HAVE_KQUEUE + + +/* + * Do we have the header? + */ + +#undef HAVE_DLFCN_H + + +/* + * Do we have ? + */ + +#undef HAVE_SYS_PARAM_H + + +/* + * Do we have ? + */ + +#undef HAVE_SYS_UCRED_H + + +/* + * Do we have removefile()? + */ + +#undef HAVE_REMOVEFILE + + +/* + * Do we have ? + */ + +#undef HAVE_SANDBOX_H + + +/* + * Which random number generator function to use... + */ + +#undef HAVE_ARC4RANDOM +#undef HAVE_RANDOM +#undef HAVE_LRAND48 + +#ifdef HAVE_ARC4RANDOM +# define CUPS_RAND() arc4random() +# define CUPS_SRAND(v) arc4random_stir() +#elif defined(HAVE_RANDOM) +# define CUPS_RAND() random() +# define CUPS_SRAND(v) srandom(v) +#elif defined(HAVE_LRAND48) +# define CUPS_RAND() lrand48() +# define CUPS_SRAND(v) srand48(v) +#else +# define CUPS_RAND() rand() +# define CUPS_SRAND(v) srand(v) +#endif /* HAVE_ARC4RANDOM */ + + +/* + * Do we have vproc_transaction_begin/end? + */ + +#undef HAVE_VPROC_TRANSACTION_BEGIN + + +/* + * Do we have libusb? + */ + +#undef HAVE_USB_H + + +/* + * Do we have libwrap and tcpd.h? + */ + +#undef HAVE_TCPD_H + + +/* + * Do we have ? + */ + +#undef HAVE_ICONV_H + + +/* + * Do we have statfs or statvfs and one of the corresponding headers? + */ + +#undef HAVE_STATFS +#undef HAVE_STATVFS +#undef HAVE_SYS_MOUNT_H +#undef HAVE_SYS_STATFS_H +#undef HAVE_SYS_STATVFS_H +#undef HAVE_SYS_VFS_H + + +/* + * Location of Mac OS X localization bundle, if any. + */ + +#undef CUPS_BUNDLEDIR + + +/* + * Do we have the ColorSyncRegisterDevice function? + */ + +#undef HAVE_COLORSYNCREGISTERDEVICE + + +/* + * Do we have XPC? + */ + +#undef HAVE_XPC + + +/* + * Do we have Mini-XML? + */ + +#undef HAVE_MXML_H + + +#endif /* !_CUPS_CONFIG_H_ */ + +/* + * End of "$Id$". + */ diff --git a/configure.in b/configure.in new file mode 100644 index 0000000000..fe47cbf566 --- /dev/null +++ b/configure.in @@ -0,0 +1,93 @@ +dnl +dnl "$Id$" +dnl +dnl Configuration script for CUPS. +dnl +dnl Copyright 2007-2012 by Apple Inc. +dnl Copyright 1997-2007 by Easy Software Products, all rights reserved. +dnl +dnl These coded instructions, statements, and computer programs are the +dnl property of Apple Inc. and are protected by Federal copyright +dnl law. Distribution and use rights are outlined in the file "LICENSE.txt" +dnl which should have been included with this file. If this file is +dnl file is missing or damaged, see the license at "http://www.cups.org/". +dnl + +AC_INIT(cups/cups.h) + +sinclude(config-scripts/cups-opsys.m4) +sinclude(config-scripts/cups-common.m4) +sinclude(config-scripts/cups-directories.m4) +sinclude(config-scripts/cups-manpages.m4) + +sinclude(config-scripts/cups-sharedlibs.m4) +sinclude(config-scripts/cups-libtool.m4) +sinclude(config-scripts/cups-compiler.m4) + +sinclude(config-scripts/cups-network.m4) +sinclude(config-scripts/cups-poll.m4) +sinclude(config-scripts/cups-gssapi.m4) +sinclude(config-scripts/cups-threads.m4) +sinclude(config-scripts/cups-ssl.m4) +sinclude(config-scripts/cups-pam.m4) +sinclude(config-scripts/cups-largefile.m4) +sinclude(config-scripts/cups-dnssd.m4) +sinclude(config-scripts/cups-launchd.m4) +sinclude(config-scripts/cups-defaults.m4) +sinclude(config-scripts/cups-scripting.m4) + +INSTALL_LANGUAGES="" +UNINSTALL_LANGUAGES="" +LANGFILES="" +if test "x$LANGUAGES" != x; then + INSTALL_LANGUAGES="install-languages" + UNINSTALL_LANGUAGES="uninstall-languages" + for lang in $LANGUAGES; do + if test -f doc/$lang/index.html.in; then + LANGFILES="$LANGFILES doc/$lang/index.html" + fi + + if test -f templates/$lang/header.tmpl.in; then + LANGFILES="$LANGFILES templates/$lang/header.tmpl" + fi + done +elif test "x$CUPS_BUNDLEDIR" != x; then + INSTALL_LANGUAGES="install-langbundle" + UNINSTALL_LANGUAGES="uninstall-langbundle" +fi + +AC_SUBST(INSTALL_LANGUAGES) +AC_SUBST(UNINSTALL_LANGUAGES) + +AC_OUTPUT(Makedefs + conf/cupsd.conf + conf/mime.convs + conf/pam.std + conf/snmp.conf + cups-config + data/testprint + desktop/cups.desktop + doc/help/ref-cupsd-conf.html + doc/help/standard.html + doc/index.html + man/client.conf.man + man/cups-deviced.man + man/cups-driverd.man + man/cups-lpd.man + man/cupsaddsmb.man + man/cupsd.conf.man + man/cupsd.man + man/lpoptions.man + scheduler/cups-lpd.xinetd + scheduler/cups.sh + scheduler/cups.xml + scheduler/org.cups.cups-lpd.plist + templates/header.tmpl + packaging/cups.list + $LANGFILES) + +chmod +x cups-config + +dnl +dnl End of "$Id$". +dnl diff --git a/cups-config.in b/cups-config.in new file mode 100755 index 0000000000..436800a640 --- /dev/null +++ b/cups-config.in @@ -0,0 +1,146 @@ +#! /bin/sh +# +# "$Id$" +# +# CUPS configuration utility. +# +# Copyright 2007-2011 by Apple Inc. +# Copyright 2001-2006 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +VERSION="@CUPS_VERSION@" +APIVERSION="1.6" +BUILD="@CUPS_BUILD@" + +prefix=@prefix@ +exec_prefix=@exec_prefix@ +bindir=@bindir@ +includedir=@includedir@ +libdir=@libdir@ +imagelibdir=@libdir@ +datarootdir=@datadir@ +datadir=@datadir@ +sysconfdir=@sysconfdir@ +cups_datadir=@CUPS_DATADIR@ +cups_serverbin=@CUPS_SERVERBIN@ +cups_serverroot=@CUPS_SERVERROOT@ +INSTALLSTATIC=@INSTALLSTATIC@ + +# flags for C++ compiler: +CFLAGS="" +LDFLAGS="@EXPORT_LDFLAGS@" +LIBS="@LIBGSSAPI@ @EXPORT_SSLLIBS@ @LIBZ@ @LIBS@" + +# Check for local invocation... +selfdir=`dirname $0` + +if test -f "$selfdir/cups/cups.h"; then + CFLAGS="-I$selfdir" + LDFLAGS="-L$selfdir/cups -L$selfdir/filter $LDFLAGS" + libdir="$selfdir/cups" + imagelibdir="$selfdir/filter" +else + if test $includedir != /usr/include; then + CFLAGS="$CFLAGS -I$includedir" + fi + + if test $libdir != /usr/lib -a $libdir != /usr/lib32 -a $libdir != /usr/lib64; then + LDFLAGS="$LDFLAGS -L$libdir" + fi +fi + + +usage () +{ + echo "Usage: cups-config --api-version" + echo " cups-config --build" + echo " cups-config --cflags" + echo " cups-config --datadir" + echo " cups-config --help" + echo " cups-config --ldflags" + echo " cups-config [--image] [--static] --libs" + echo " cups-config --serverbin" + echo " cups-config --serverroot" + echo " cups-config --version" + + exit $1 +} + +if test $# -eq 0; then + usage 1 +fi + +# Parse command line options +static=no +image=no + +while test $# -gt 0; do + case $1 in + --api-version) + echo $APIVERSION + ;; + --build) + echo $BUILD + ;; + --cflags) + echo $CFLAGS + ;; + --datadir) + echo $cups_datadir + ;; + --help) + usage 0 + ;; + --image) + image=yes + ;; + --ldflags) + echo $LDFLAGS + ;; + --libs) + if test $static = no; then + libs="@EXTLINKCUPS@ $LIBS"; + if test $image = yes; then + libs="@EXTLINKCUPSIMAGE@ $libs" + fi + else + libs="$libdir/libcups.a $LIBS"; + if test $image = yes; then + libs="$libdir/libcupsimage.a $libs" + fi + fi + echo $libs + ;; + --serverbin) + echo $cups_serverbin + ;; + --serverroot) + echo $cups_serverroot + ;; + --static) + if test -z "$INSTALLSTATIC"; then + echo "WARNING: Static libraries not installed!" >&2 + else + static=yes + fi + ;; + --version) + echo $VERSION + ;; + *) + usage 1 + ;; + esac + + shift +done + +# +# End of "$Id$". +# diff --git a/cups/Dependencies b/cups/Dependencies new file mode 100644 index 0000000000..276dd90c65 --- /dev/null +++ b/cups/Dependencies @@ -0,0 +1,250 @@ +adminutil.o: adminutil.c cups-private.h ../cups/cups.h file.h \ + versioning.h ipp.h http.h array.h language.h string-private.h \ + ../config.h debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h adminutil.h +array.o: array.c string-private.h ../config.h debug-private.h \ + ../cups/versioning.h array-private.h ../cups/array.h +attr.o: attr.c cups-private.h ../cups/cups.h file.h versioning.h ipp.h \ + http.h array.h language.h string-private.h ../config.h debug-private.h \ + ppd-private.h ../cups/ppd.h pwg-private.h http-private.h md5-private.h \ + ipp-private.h language-private.h ../cups/transcode.h thread-private.h +auth.o: auth.c cups-private.h ../cups/cups.h file.h versioning.h ipp.h \ + http.h array.h language.h string-private.h ../config.h debug-private.h \ + ppd-private.h ../cups/ppd.h pwg-private.h http-private.h md5-private.h \ + ipp-private.h language-private.h ../cups/transcode.h thread-private.h +backchannel.o: backchannel.c cups.h file.h versioning.h ipp.h http.h \ + array.h language.h +backend.o: backend.c cups-private.h ../cups/cups.h file.h versioning.h \ + ipp.h http.h array.h language.h string-private.h ../config.h \ + debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h backend.h +conflicts.o: conflicts.c cups-private.h ../cups/cups.h file.h \ + versioning.h ipp.h http.h array.h language.h string-private.h \ + ../config.h debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +custom.o: custom.c cups-private.h ../cups/cups.h file.h versioning.h \ + ipp.h http.h array.h language.h string-private.h ../config.h \ + debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +debug.o: debug.c cups-private.h ../cups/cups.h file.h versioning.h ipp.h \ + http.h array.h language.h string-private.h ../config.h debug-private.h \ + ppd-private.h ../cups/ppd.h pwg-private.h http-private.h md5-private.h \ + ipp-private.h language-private.h ../cups/transcode.h thread-private.h +dest.o: dest.c cups-private.h ../cups/cups.h file.h versioning.h ipp.h \ + http.h array.h language.h string-private.h ../config.h debug-private.h \ + ppd-private.h ../cups/ppd.h pwg-private.h http-private.h md5-private.h \ + ipp-private.h language-private.h ../cups/transcode.h thread-private.h +dest-job.o: dest-job.c cups-private.h ../cups/cups.h file.h versioning.h \ + ipp.h http.h array.h language.h string-private.h ../config.h \ + debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +dest-localization.o: dest-localization.c cups-private.h ../cups/cups.h \ + file.h versioning.h ipp.h http.h array.h language.h string-private.h \ + ../config.h debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +dest-options.o: dest-options.c cups-private.h ../cups/cups.h file.h \ + versioning.h ipp.h http.h array.h language.h string-private.h \ + ../config.h debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +dir.o: dir.c string-private.h ../config.h debug-private.h \ + ../cups/versioning.h dir.h +emit.o: emit.c cups-private.h ../cups/cups.h file.h versioning.h ipp.h \ + http.h array.h language.h string-private.h ../config.h debug-private.h \ + ppd-private.h ../cups/ppd.h pwg-private.h http-private.h md5-private.h \ + ipp-private.h language-private.h ../cups/transcode.h thread-private.h +encode.o: encode.c cups-private.h ../cups/cups.h file.h versioning.h \ + ipp.h http.h array.h language.h string-private.h ../config.h \ + debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +file.o: file.c file-private.h cups-private.h ../cups/cups.h file.h \ + versioning.h ipp.h http.h array.h language.h string-private.h \ + ../config.h debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +getdevices.o: getdevices.c cups-private.h ../cups/cups.h file.h \ + versioning.h ipp.h http.h array.h language.h string-private.h \ + ../config.h debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +getifaddrs.o: getifaddrs.c http-private.h ../config.h ../cups/http.h \ + versioning.h array.h md5-private.h ipp-private.h ../cups/ipp.h +getputfile.o: getputfile.c cups-private.h ../cups/cups.h file.h \ + versioning.h ipp.h http.h array.h language.h string-private.h \ + ../config.h debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +globals.o: globals.c cups-private.h ../cups/cups.h file.h versioning.h \ + ipp.h http.h array.h language.h string-private.h ../config.h \ + debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +http.o: http.c cups-private.h ../cups/cups.h file.h versioning.h ipp.h \ + http.h array.h language.h string-private.h ../config.h debug-private.h \ + ppd-private.h ../cups/ppd.h pwg-private.h http-private.h md5-private.h \ + ipp-private.h language-private.h ../cups/transcode.h thread-private.h +http-addr.o: http-addr.c cups-private.h ../cups/cups.h file.h \ + versioning.h ipp.h http.h array.h language.h string-private.h \ + ../config.h debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +http-addrlist.o: http-addrlist.c cups-private.h ../cups/cups.h file.h \ + versioning.h ipp.h http.h array.h language.h string-private.h \ + ../config.h debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +http-support.o: http-support.c cups-private.h ../cups/cups.h file.h \ + versioning.h ipp.h http.h array.h language.h string-private.h \ + ../config.h debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +ipp.o: ipp.c cups-private.h ../cups/cups.h file.h versioning.h ipp.h \ + http.h array.h language.h string-private.h ../config.h debug-private.h \ + ppd-private.h ../cups/ppd.h pwg-private.h http-private.h md5-private.h \ + ipp-private.h language-private.h ../cups/transcode.h thread-private.h +ipp-support.o: ipp-support.c cups-private.h ../cups/cups.h file.h \ + versioning.h ipp.h http.h array.h language.h string-private.h \ + ../config.h debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +langprintf.o: langprintf.c cups-private.h ../cups/cups.h file.h \ + versioning.h ipp.h http.h array.h language.h string-private.h \ + ../config.h debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +language.o: language.c cups-private.h ../cups/cups.h file.h versioning.h \ + ipp.h http.h array.h language.h string-private.h ../config.h \ + debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +localize.o: localize.c cups-private.h ../cups/cups.h file.h versioning.h \ + ipp.h http.h array.h language.h string-private.h ../config.h \ + debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +mark.o: mark.c cups-private.h ../cups/cups.h file.h versioning.h ipp.h \ + http.h array.h language.h string-private.h ../config.h debug-private.h \ + ppd-private.h ../cups/ppd.h pwg-private.h http-private.h md5-private.h \ + ipp-private.h language-private.h ../cups/transcode.h thread-private.h +md5.o: md5.c md5-private.h string-private.h ../config.h +md5passwd.o: md5passwd.c http-private.h ../config.h ../cups/http.h \ + versioning.h array.h md5-private.h ipp-private.h ../cups/ipp.h \ + string-private.h +notify.o: notify.c cups-private.h ../cups/cups.h file.h versioning.h \ + ipp.h http.h array.h language.h string-private.h ../config.h \ + debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +options.o: options.c cups-private.h ../cups/cups.h file.h versioning.h \ + ipp.h http.h array.h language.h string-private.h ../config.h \ + debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +page.o: page.c string-private.h ../config.h debug-private.h \ + ../cups/versioning.h ppd.h cups.h file.h ipp.h http.h array.h \ + language.h +ppd.o: ppd.c cups-private.h ../cups/cups.h file.h versioning.h ipp.h \ + http.h array.h language.h string-private.h ../config.h debug-private.h \ + ppd-private.h ../cups/ppd.h pwg-private.h http-private.h md5-private.h \ + ipp-private.h language-private.h ../cups/transcode.h thread-private.h +ppd-cache.o: ppd-cache.c cups-private.h ../cups/cups.h file.h \ + versioning.h ipp.h http.h array.h language.h string-private.h \ + ../config.h debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +pwg-media.o: pwg-media.c cups-private.h ../cups/cups.h file.h \ + versioning.h ipp.h http.h array.h language.h string-private.h \ + ../config.h debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +request.o: request.c cups-private.h ../cups/cups.h file.h versioning.h \ + ipp.h http.h array.h language.h string-private.h ../config.h \ + debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +sidechannel.o: sidechannel.c sidechannel.h versioning.h cups-private.h \ + ../cups/cups.h file.h ipp.h http.h array.h language.h string-private.h \ + ../config.h debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +snmp.o: snmp.c cups-private.h ../cups/cups.h file.h versioning.h ipp.h \ + http.h array.h language.h string-private.h ../config.h debug-private.h \ + ppd-private.h ../cups/ppd.h pwg-private.h http-private.h md5-private.h \ + ipp-private.h language-private.h ../cups/transcode.h thread-private.h \ + snmp-private.h +snprintf.o: snprintf.c string-private.h ../config.h +string.o: string.c string-private.h ../config.h debug-private.h \ + ../cups/versioning.h thread-private.h array.h +tempfile.o: tempfile.c cups-private.h ../cups/cups.h file.h versioning.h \ + ipp.h http.h array.h language.h string-private.h ../config.h \ + debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +thread.o: thread.c cups-private.h ../cups/cups.h file.h versioning.h \ + ipp.h http.h array.h language.h string-private.h ../config.h \ + debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +transcode.o: transcode.c cups-private.h ../cups/cups.h file.h \ + versioning.h ipp.h http.h array.h language.h string-private.h \ + ../config.h debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +usersys.o: usersys.c cups-private.h ../cups/cups.h file.h versioning.h \ + ipp.h http.h array.h language.h string-private.h ../config.h \ + debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +util.o: util.c cups-private.h ../cups/cups.h file.h versioning.h ipp.h \ + http.h array.h language.h string-private.h ../config.h debug-private.h \ + ppd-private.h ../cups/ppd.h pwg-private.h http-private.h md5-private.h \ + ipp-private.h language-private.h ../cups/transcode.h thread-private.h +testadmin.o: testadmin.c adminutil.h cups.h file.h versioning.h ipp.h \ + http.h array.h language.h string-private.h ../config.h +testarray.o: testarray.c string-private.h ../config.h debug-private.h \ + ../cups/versioning.h array.h dir.h +testconflicts.o: testconflicts.c cups.h file.h versioning.h ipp.h http.h \ + array.h language.h ppd.h string-private.h ../config.h +testcups.o: testcups.c string-private.h ../config.h cups.h file.h \ + versioning.h ipp.h http.h array.h language.h ppd.h +testfile.o: testfile.c string-private.h ../config.h debug-private.h \ + ../cups/versioning.h file.h +testhttp.o: testhttp.c string-private.h ../config.h http-private.h \ + ../cups/http.h versioning.h array.h md5-private.h ipp-private.h \ + ../cups/ipp.h +testi18n.o: testi18n.c string-private.h ../config.h language-private.h \ + ../cups/transcode.h language.h array.h versioning.h +testipp.o: testipp.c file.h versioning.h string-private.h ../config.h \ + ipp-private.h ../cups/ipp.h http.h array.h +testoptions.o: testoptions.c cups-private.h ../cups/cups.h file.h \ + versioning.h ipp.h http.h array.h language.h string-private.h \ + ../config.h debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +testlang.o: testlang.c cups-private.h ../cups/cups.h file.h versioning.h \ + ipp.h http.h array.h language.h string-private.h ../config.h \ + debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +testppd.o: testppd.c cups-private.h ../cups/cups.h file.h versioning.h \ + ipp.h http.h array.h language.h string-private.h ../config.h \ + debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h +testpwg.o: testpwg.c ppd-private.h ../cups/cups.h file.h versioning.h \ + ipp.h http.h array.h language.h ../cups/ppd.h pwg-private.h \ + file-private.h cups-private.h string-private.h ../config.h \ + debug-private.h http-private.h md5-private.h ipp-private.h \ + language-private.h ../cups/transcode.h thread-private.h +testsnmp.o: testsnmp.c cups-private.h ../cups/cups.h file.h versioning.h \ + ipp.h http.h array.h language.h string-private.h ../config.h \ + debug-private.h ppd-private.h ../cups/ppd.h pwg-private.h \ + http-private.h md5-private.h ipp-private.h language-private.h \ + ../cups/transcode.h thread-private.h snmp-private.h diff --git a/cups/Makefile b/cups/Makefile new file mode 100644 index 0000000000..da920d72e2 --- /dev/null +++ b/cups/Makefile @@ -0,0 +1,625 @@ +# +# "$Id$" +# +# API library Makefile for CUPS. +# +# Copyright 2007-2012 by Apple Inc. +# Copyright 1997-2006 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# +# This file is subject to the Apple OS-Developed Software exception. +# + +include ../Makedefs + +# +# Object files... +# + +LIBOBJS = \ + adminutil.o \ + array.o \ + attr.o \ + auth.o \ + backchannel.o \ + backend.o \ + conflicts.o \ + custom.o \ + debug.o \ + dest.o \ + dest-job.o \ + dest-localization.o \ + dest-options.o \ + dir.o \ + emit.o \ + encode.o \ + file.o \ + getdevices.o \ + getifaddrs.o \ + getputfile.o \ + globals.o \ + http.o \ + http-addr.o \ + http-addrlist.o \ + http-support.o \ + ipp.o \ + ipp-support.o \ + langprintf.o \ + language.o \ + localize.o \ + mark.o \ + md5.o \ + md5passwd.o \ + notify.o \ + options.o \ + page.o \ + ppd.o \ + ppd-cache.o \ + pwg-media.o \ + request.o \ + sidechannel.o \ + snmp.o \ + snprintf.o \ + string.o \ + tempfile.o \ + thread.o \ + transcode.o \ + usersys.o \ + util.o +TESTOBJS = \ + testadmin.o \ + testarray.o \ + testconflicts.o \ + testcups.o \ + testfile.o \ + testhttp.o \ + testi18n.o \ + testipp.o \ + testoptions.o \ + testlang.o \ + testppd.o \ + testpwg.o \ + testsnmp.o +OBJS = \ + $(LIBOBJS) \ + $(TESTOBJS) + + +# +# Header files to install... +# + +HEADERS = \ + adminutil.h \ + array.h \ + backend.h \ + cups.h \ + dir.h \ + file.h \ + http.h \ + ipp.h \ + language.h \ + ppd.h \ + raster.h \ + sidechannel.h \ + transcode.h \ + versioning.h + +HEADERSPRIV = \ + array-private.h \ + cups-private.h \ + debug-private.h \ + file-private.h \ + http-private.h \ + ipp-private.h \ + language-private.h \ + md5-private.h \ + ppd-private.h \ + pwg-private.h \ + raster-private.h \ + snmp-private.h \ + string-private.h \ + thread-private.h + + +# +# Targets in this directory... +# + +LIBTARGETS = \ + $(LIBCUPSSTATIC) \ + $(LIBCUPS) + +UNITTARGETS = \ + testadmin \ + testarray \ + testconflicts \ + testcups \ + testfile \ + testhttp \ + testi18n \ + testipp \ + testlang \ + testoptions \ + testppd \ + testpwg \ + testsnmp + +TARGETS = \ + $(LIBTARGETS) + + +# +# Make all targets... +# + +all: $(TARGETS) + + +# +# Make library targets... +# + +libs: $(LIBTARGETS) + + +# +# Make unit tests... +# + +unittests: $(UNITTARGETS) + + +# +# Remove object and target files... +# + +clean: + $(RM) $(OBJS) $(TARGETS) $(UNITTARGETS) + $(RM) libcups.so libcups.sl libcups.dylib + + +# +# Update dependencies (without system header dependencies...) +# + +depend: + $(CC) -MM $(ALL_CFLAGS) $(OBJS:.o=.c) >Dependencies + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: + + +# +# Install programs... +# + +install-exec: + + +# +# Install headers... +# + +install-headers: + echo Installing header files into $(INCLUDEDIR)/cups... + $(INSTALL_DIR) -m 755 $(INCLUDEDIR)/cups + for file in $(HEADERS); do \ + $(INSTALL_DATA) $$file $(INCLUDEDIR)/cups; \ + done + if test "x$(privateinclude)" != x; then \ + echo Installing private header files into $(PRIVATEINCLUDE)...; \ + $(INSTALL_DIR) -m 755 $(PRIVATEINCLUDE); \ + for file in $(HEADERSPRIV); do \ + $(INSTALL_DATA) $$file $(PRIVATEINCLUDE)/$$file; \ + done; \ + fi + + +# +# Install libraries... +# + +install-libs: $(INSTALLSTATIC) + echo Installing libraries in $(LIBDIR)... + $(INSTALL_DIR) -m 755 $(LIBDIR) + $(INSTALL_LIB) $(LIBCUPS) $(LIBDIR) + if test $(LIBCUPS) = "libcups.so.2" -o $(LIBCUPS) = "libcups.sl.2"; then \ + $(RM) $(LIBDIR)/`basename $(LIBCUPS) .2`; \ + $(LN) $(LIBCUPS) $(LIBDIR)/`basename $(LIBCUPS) .2`; \ + fi + if test $(LIBCUPS) = "libcups.2.dylib"; then \ + $(RM) $(LIBDIR)/libcups.dylib; \ + $(LN) $(LIBCUPS) $(LIBDIR)/libcups.dylib; \ + fi + if test "x$(SYMROOT)" != "x"; then \ + $(INSTALL_DIR) $(SYMROOT); \ + cp $(LIBCUPS) $(SYMROOT); \ + fi + +installstatic: + $(INSTALL_DIR) -m 755 $(LIBDIR) + $(INSTALL_LIB) -m 755 $(LIBCUPSSTATIC) $(LIBDIR) + $(RANLIB) $(LIBDIR)/$(LIBCUPSSTATIC) + $(CHMOD) 555 $(LIBDIR)/$(LIBCUPSSTATIC) + + +# +# Uninstall object and target files... +# + +uninstall: + $(RM) $(LIBDIR)/libcups.2.dylib + $(RM) $(LIBDIR)/$(LIBCUPSSTATIC) + $(RM) $(LIBDIR)/libcups.dylib + $(RM) $(LIBDIR)/libcups_s.a + $(RM) $(LIBDIR)/libcups.sl + $(RM) $(LIBDIR)/libcups.sl.2 + $(RM) $(LIBDIR)/libcups.so + $(RM) $(LIBDIR)/libcups.so.2 + -$(RMDIR) $(LIBDIR) + for file in $(HEADERS); do \ + $(RM) $(INCLUDEDIR)/cups/$$file; \ + done + -$(RMDIR) $(INCLUDEDIR)/cups + + +# +# libcups.so.2, libcups.sl.2 +# + +libcups.so.2 libcups.sl.2: $(LIBOBJS) + echo Linking $@... + $(DSO) $(ARCHFLAGS) $(DSOFLAGS) -o $@ $(LIBOBJS) $(LIBGSSAPI) \ + $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + $(RM) `basename $@ .2` + $(LN) $@ `basename $@ .2` + + +# +# libcups.2.dylib +# + +libcups.2.dylib: $(LIBOBJS) $(LIBCUPSORDER) + echo Creating export list for $@... + nm $(LIBOBJS) 2>/dev/null | grep "T _" | awk '{print $$3}' | \ + grep -v -e '^(_cupsConnect|_cupsCharset|_cupsEncodingName|_cupsSetDefaults|_cupsSetHTTPError|_cupsUserDefault|_httpWait)$$' | \ + sort >t.exp + echo Linking $@... + $(DSO) $(ARCHFLAGS) $(DSOFLAGS) -o $@ \ + -install_name $(libdir)/$@ \ + -current_version 2.9.0 \ + -compatibility_version 2.0.0 \ + -exported_symbols_list t.exp \ + $(LIBOBJS) $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) \ + $(COMMONLIBS) $(LIBZ) + $(RM) libcups.dylib t.exp + $(LN) $@ libcups.dylib + + +# +# libcups_s.a +# + +libcups_s.a: $(LIBOBJS) libcups_s.exp + echo Creating $@... + $(DSO) $(DSOFLAGS) -Wl,-bexport:libcups_s.exp -o libcups_s.o \ + $(LIBOBJS) $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) \ + $(COMMONLIBS) $(LIBZ) + $(RM) $@ + $(AR) $(ARFLAGS) $@ libcups_s.o + + +# +# libcups.la +# + +libcups.la: $(LIBOBJS) + echo Linking $@... + $(CC) $(ARCHFLAGS) $(DSOFLAGS) -o $@ $(LIBOBJS:.o=.lo) \ + -rpath $(LIBDIR) -version-info 2:9 $(LIBGSSAPI) $(SSLLIBS) \ + $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + + +# +# libcups.a +# + +libcups.a: $(LIBOBJS) + echo Archiving $@... + $(RM) $@ + $(AR) $(ARFLAGS) $@ $(LIBOBJS) + $(RANLIB) $@ + + +# +# testadmin (dependency on static CUPS library is intentional) +# + +testadmin: testadmin.o $(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ testadmin.o $(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + + +# +# testarray (dependency on static CUPS library is intentional) +# + +testarray: testarray.o $(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o $@ testarray.o $(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + echo Running array API tests... + ./testarray + + +# +# testconflicts (dependency on static CUPS library is intentional) +# + +testconflicts: testconflicts.o $(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ testconflicts.o $(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + + +# +# testcups (dependency on static CUPS library is intentional) +# + +testcups: testcups.o $(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ testcups.o $(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + + +# +# testfile (dependency on static CUPS library is intentional) +# + +testfile: testfile.o $(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o $@ testfile.o $(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + echo Running file API tests... + ./testfile + + +# +# testhttp (dependency on static CUPS library is intentional) +# + +testhttp: testhttp.o $(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o $@ testhttp.o $(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + echo Running HTTP API tests... + ./testhttp + + +# +# testipp (dependency on static CUPS library is intentional) +# + +testipp: testipp.o $(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o $@ testipp.o $(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + echo Running IPP API tests... + ./testipp + + +# +# testi18n (dependency on static CUPS library is intentional) +# + +testi18n: testi18n.o $(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o $@ testi18n.o $(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + echo Running internationalization API tests... + ./testi18n + + +# +# testlang (dependency on static CUPS library is intentional) +# + +testlang: testlang.o $(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o $@ testlang.o $(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + echo Running language API tests... + ./testlang + + +# +# testoptions (dependency on static CUPS library is intentional) +# + +testoptions: testoptions.o $(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o $@ testoptions.o $(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + echo Running option API tests... + ./testoptions + + +# +# testppd (dependency on static CUPS library is intentional) +# + +testppd: testppd.o $(LIBCUPSSTATIC) test.ppd test2.ppd + echo Linking $@... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o $@ testppd.o $(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + echo Running PPD API tests... + ./testppd + + +# +# testpwg (dependency on static CUPS library is intentional) +# + +testpwg: testpwg.o $(LIBCUPSSTATIC) test.ppd + echo Linking $@... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o $@ testpwg.o $(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + echo Running PWG API tests... + ./testpwg test.ppd + + +# +# testsnmp (dependency on static CUPS library is intentional) +# + +testsnmp: testsnmp.o $(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ testsnmp.o $(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + + +# +# Automatic API help files... +# + +apihelp: + echo Generating CUPS API help files... + mxmldoc --section "Programming" \ + --title "Introduction to CUPS Programming" \ + --css ../doc/cups-printable.css \ + --header api-overview.header --intro api-overview.shtml \ + >../doc/help/api-overview.html + mxmldoc --section "Programming" --title "Array API" \ + --css ../doc/cups-printable.css \ + --header api-array.header --intro api-array.shtml \ + api-array.xml \ + array.h array.c >../doc/help/api-array.html + mxmldoc --tokens help/api-array.html api-array.xml >../doc/help/api-array.tokens + $(RM) api-array.xml + mxmldoc --section "Programming" --title "CUPS API" \ + --css ../doc/cups-printable.css \ + --header api-cups.header --intro api-cups.shtml \ + api-cups.xml \ + cups.h adminutil.c dest.c language.c notify.c \ + options.c tempfile.c usersys.c \ + util.c >../doc/help/api-cups.html + mxmldoc --tokens help/api-cups.html api-cups.xml >../doc/help/api-cups.tokens + $(RM) api-cups.xml + mxmldoc --section "Programming" --title "File and Directory APIs" \ + --css ../doc/cups-printable.css \ + --header api-filedir.header --intro api-filedir.shtml \ + api-filedir.xml \ + file.h file.c dir.h dir.c >../doc/help/api-filedir.html + mxmldoc --tokens api-filedir.xml >../doc/help/api-filedir.tokens + $(RM) api-filedir.xml + mxmldoc --section "Programming" --title "PPD API (DEPRECATED)" \ + --css ../doc/cups-printable.css \ + --header api-ppd.header --intro api-ppd.shtml \ + api-ppd.xml \ + ppd.h attr.c conflicts.c custom.c emit.c localize.c mark.c page.c \ + ppd.c >../doc/help/api-ppd.html + mxmldoc --tokens help/api-ppd.html api-ppd.xml >../doc/help/api-ppd.tokens + $(RM) api-ppd.xml + mxmldoc --section "Programming" --title "HTTP and IPP APIs" \ + --css ../doc/cups-printable.css \ + --header api-httpipp.header --intro api-httpipp.shtml \ + api-httpipp.xml \ + http.h ipp.h auth.c getdevices.c getputfile.c encode.c \ + http.c http-addr.c http-support.c ipp.c ipp-support.c \ + md5passwd.c request.c >../doc/help/api-httpipp.html + mxmldoc --tokens help/api-httpipp.html api-httpipp.xml >../doc/help/api-httpipp.tokens + $(RM) api-httpipp.xml + mxmldoc --section "Programming" \ + --title "Filter and Backend Programming" \ + --css ../doc/cups-printable.css \ + --header api-filter.header --intro api-filter.shtml \ + api-filter.xml \ + backchannel.c backend.h backend.c sidechannel.c sidechannel.h \ + >../doc/help/api-filter.html + mxmldoc --tokens help/api-filter.html api-filter.xml >../doc/help/api-filter.tokens + $(RM) api-filter.xml + +framedhelp: + echo Generating CUPS API help files... + mxmldoc --framed api-overview \ + --section "Programming" \ + --title "Introduction to CUPS Programming" \ + --css ../doc/cups-printable.css \ + --header api-overview.header --intro api-overview.shtml + mxmldoc --framed api-array \ + --section "Programming" --title "Array API" \ + --css ../doc/cups-printable.css \ + --header api-array.header --intro api-array.shtml \ + array.h array.c + mxmldoc --framed api-cups \ + --section "Programming" --title "CUPS API" \ + --css ../doc/cups-printable.css \ + --header api-cups.header --intro api-cups.shtml \ + cups.h adminutil.c dest.c language.c notify.c \ + options.c tempfile.c usersys.c \ + util.c + mxmldoc --framed api-filedir \ + --section "Programming" --title "File and Directory APIs" \ + --css ../doc/cups-printable.css \ + --header api-filedir.header --intro api-filedir.shtml \ + file.h file.c dir.h dir.c + mxmldoc --framed api-ppd \ + --section "Programming" --title "PPD API (DEPRECATED)" \ + --css ../doc/cups-printable.css \ + --header api-ppd.header --intro api-ppd.shtml \ + ppd.h attr.c conflicts.c custom.c emit.c localize.c mark.c \ + page.c ppd.c + mxmldoc --framed api-httpipp \ + --section "Programming" --title "HTTP and IPP APIs" \ + --css ../doc/cups-printable.css \ + --header api-httpipp.header --intro api-httpipp.shtml \ + http.h ipp.h auth.c getdevices.c getputfile.c encode.c \ + http.c http-addr.c http-support.c ipp.c ipp-support.c \ + md5passwd.c request.c + mxmldoc --framed api-filter \ + --section "Programming" \ + --title "Filter and Backend Programming" \ + --css ../doc/cups-printable.css \ + --header api-filter.header --intro api-filter.shtml \ + backchannel.c backend.h backend.c sidechannel.c sidechannel.h + + +# +# Lines of code computation... +# + +sloc: + echo "libcupslite: \c" + sloccount $(LITEOBJS:.o=.c) 2>/dev/null | grep "Total Physical" | awk '{print $$9}' + echo "libcups: \c" + sloccount $(LIBOBJS:.o=.c) 2>/dev/null | grep "Total Physical" | awk '{print $$9}' + + +# +# Dependencies... +# + +include Dependencies + + +# +# End of "$Id$". +# diff --git a/cups/adminutil.c b/cups/adminutil.c new file mode 100644 index 0000000000..dba4643137 --- /dev/null +++ b/cups/adminutil.c @@ -0,0 +1,2341 @@ +/* + * "$Id$" + * + * Administration utility API definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 2001-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsAdminCreateWindowsPPD() - Create the Windows PPD file for a printer. + * cupsAdminExportSamba() - Export a printer to Samba. + * cupsAdminGetServerSettings() - Get settings from the server. + * cupsAdminSetServerSettings() - Set settings on the server. + * do_samba_command() - Do a SAMBA command. + * get_cupsd_conf() - Get the current cupsd.conf file. + * invalidate_cupsd_cache() - Invalidate the cached cupsd.conf settings. + * write_option() - Write a CUPS option to a PPD file. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include "adminutil.h" +#include +#include +#ifdef WIN32 +#else +# include +# include +#endif /* WIN32 */ + + +/* + * Local functions... + */ + +static int do_samba_command(const char *command, + const char *address, + const char *subcommand, + const char *authfile, + FILE *logfile); +static http_status_t get_cupsd_conf(http_t *http, _cups_globals_t *cg, + time_t last_update, char *name, + int namelen, int *remote); +static void invalidate_cupsd_cache(_cups_globals_t *cg); +static void write_option(cups_file_t *dstfp, int order, + const char *name, const char *text, + const char *attrname, + ipp_attribute_t *suppattr, + ipp_attribute_t *defattr, int defval, + int valcount); + + +/* + * 'cupsAdminCreateWindowsPPD()' - Create the Windows PPD file for a printer. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +char * /* O - PPD file or NULL */ +cupsAdminCreateWindowsPPD( + http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + const char *dest, /* I - Printer or class */ + char *buffer, /* I - Filename buffer */ + int bufsize) /* I - Size of filename buffer */ +{ + const char *src; /* Source PPD filename */ + cups_file_t *srcfp, /* Source PPD file */ + *dstfp; /* Destination PPD file */ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + ipp_attribute_t *suppattr, /* IPP -supported attribute */ + *defattr; /* IPP -default attribute */ + cups_lang_t *language; /* Current language */ + char line[256], /* Line from PPD file */ + junk[256], /* Extra junk to throw away */ + *ptr, /* Pointer into line */ + uri[1024], /* Printer URI */ + option[41], /* Option */ + choice[41]; /* Choice */ + int jcloption, /* In a JCL option? */ + jclorder, /* Next JCL order dependency */ + linenum; /* Current line number */ + time_t curtime; /* Current time */ + struct tm *curdate; /* Current date */ + static const char * const pattrs[] = /* Printer attributes we want */ + { + "job-hold-until-supported", + "job-hold-until-default", + "job-sheets-supported", + "job-sheets-default", + "job-priority-supported", + "job-priority-default" + }; + + + /* + * Range check the input... + */ + + if (buffer) + *buffer = '\0'; + + if (!http) + http = _cupsConnect(); + + if (!http || !dest || !buffer || bufsize < 2) + return (NULL); + + /* + * Get the PPD file... + */ + + if ((src = cupsGetPPD2(http, dest)) == NULL) + return (NULL); + + /* + * Get the supported banner pages, etc. for the printer... + */ + + request = ippNewRequest(IPP_GET_PRINTER_ATTRIBUTES); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", dest); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, uri); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", sizeof(pattrs) / sizeof(pattrs[0]), + NULL, pattrs); + + /* + * Do the request and get back a response... + */ + + response = cupsDoRequest(http, request, "/"); + if (!response || cupsLastError() > IPP_OK_CONFLICT) + { + unlink(src); + return (NULL); + } + + /* + * Open the original PPD file... + */ + + if ((srcfp = cupsFileOpen(src, "rb")) == NULL) + return (NULL); + + /* + * Create a temporary output file using the destination buffer... + */ + + if ((dstfp = cupsTempFile2(buffer, bufsize)) == NULL) + { + cupsFileClose(srcfp); + + unlink(src); + + return (NULL); + } + + /* + * Write a new header explaining that this isn't the original PPD... + */ + + cupsFilePuts(dstfp, "*PPD-Adobe: \"4.3\"\n"); + + curtime = time(NULL); + curdate = gmtime(&curtime); + + cupsFilePrintf(dstfp, "*%% Modified on %04d%02d%02d%02d%02d%02d+0000 " + "for CUPS Windows Driver\n", + curdate->tm_year + 1900, curdate->tm_mon + 1, curdate->tm_mday, + curdate->tm_hour, curdate->tm_min, curdate->tm_sec); + + /* + * Read the existing PPD file, converting all PJL commands to CUPS + * job ticket comments... + */ + + jcloption = 0; + jclorder = 0; + linenum = 0; + language = cupsLangDefault(); + + while (cupsFileGets(srcfp, line, sizeof(line))) + { + linenum ++; + + if (!strncmp(line, "*PPD-Adobe:", 11)) + { + /* + * Already wrote the PPD header... + */ + + continue; + } + else if (!strncmp(line, "*JCLBegin:", 10) || + !strncmp(line, "*JCLToPSInterpreter:", 20) || + !strncmp(line, "*JCLEnd:", 8) || + !strncmp(line, "*Protocols:", 11)) + { + /* + * Don't use existing JCL keywords; we'll create our own, below... + */ + + cupsFilePrintf(dstfp, "*%% Commented out for CUPS Windows Driver...\n" + "*%%%s\n", line + 1); + continue; + } + else if (!strncmp(line, "*JCLOpenUI", 10)) + { + jcloption = 1; + cupsFilePrintf(dstfp, "%s\n", line); + } + else if (!strncmp(line, "*JCLCloseUI", 11)) + { + jcloption = 0; + cupsFilePrintf(dstfp, "%s\n", line); + } + else if (jcloption && !strncmp(line, "*OrderDependency:", 17)) + { + for (ptr = line + 17; _cups_isspace(*ptr); ptr ++); + + ptr = strchr(ptr, ' '); + + if (ptr) + { + cupsFilePrintf(dstfp, "*OrderDependency: %d%s\n", jclorder, ptr); + jclorder ++; + } + else + cupsFilePrintf(dstfp, "%s\n", line); + } + else if (jcloption && + strncmp(line, "*End", 4) && + strncmp(line, "*Default", 8)) + { + if ((ptr = strchr(line, ':')) == NULL) + { + snprintf(line, sizeof(line), + _cupsLangString(language, _("Missing value on line %d.")), + linenum); + _cupsSetError(IPP_DOCUMENT_FORMAT_ERROR, line, 0); + + cupsFileClose(srcfp); + cupsFileClose(dstfp); + + unlink(src); + unlink(buffer); + + *buffer = '\0'; + + return (NULL); + } + + if ((ptr = strchr(ptr, '\"')) == NULL) + { + snprintf(line, sizeof(line), + _cupsLangString(language, + _("Missing double quote on line %d.")), + linenum); + _cupsSetError(IPP_DOCUMENT_FORMAT_ERROR, line, 0); + + cupsFileClose(srcfp); + cupsFileClose(dstfp); + + unlink(src); + unlink(buffer); + + *buffer = '\0'; + + return (NULL); + } + + if (sscanf(line, "*%40s%*[ \t]%40[^:/]", option, choice) != 2) + { + snprintf(line, sizeof(line), + _cupsLangString(language, + _("Bad option + choice on line %d.")), + linenum); + _cupsSetError(IPP_DOCUMENT_FORMAT_ERROR, line, 0); + + cupsFileClose(srcfp); + cupsFileClose(dstfp); + + unlink(src); + unlink(buffer); + + *buffer = '\0'; + + return (NULL); + } + + if (strchr(ptr + 1, '\"') == NULL) + { + /* + * Skip remaining... + */ + + while (cupsFileGets(srcfp, junk, sizeof(junk)) != NULL) + { + linenum ++; + + if (!strncmp(junk, "*End", 4)) + break; + } + } + + snprintf(ptr + 1, sizeof(line) - (ptr - line + 1), + "%%cupsJobTicket: %s=%s\n\"\n*End", option, choice); + + cupsFilePrintf(dstfp, "*%% Changed for CUPS Windows Driver...\n%s\n", + line); + } + else + cupsFilePrintf(dstfp, "%s\n", line); + } + + cupsFileClose(srcfp); + unlink(src); + + if (linenum == 0) + { + _cupsSetError(IPP_DOCUMENT_FORMAT_ERROR, _("Empty PPD file."), 1); + + cupsFileClose(dstfp); + unlink(buffer); + + *buffer = '\0'; + + return (NULL); + } + + /* + * Now add the CUPS-specific attributes and options... + */ + + cupsFilePuts(dstfp, "\n*% CUPS Job Ticket support and options...\n"); + cupsFilePuts(dstfp, "*Protocols: PJL\n"); + cupsFilePuts(dstfp, "*JCLBegin: \"%!PS-Adobe-3.0<0A>\"\n"); + cupsFilePuts(dstfp, "*JCLToPSInterpreter: \"\"\n"); + cupsFilePuts(dstfp, "*JCLEnd: \"\"\n"); + + cupsFilePuts(dstfp, "\n*OpenGroup: CUPS/CUPS Options\n\n"); + + if ((defattr = ippFindAttribute(response, "job-hold-until-default", + IPP_TAG_ZERO)) != NULL && + (suppattr = ippFindAttribute(response, "job-hold-until-supported", + IPP_TAG_ZERO)) != NULL) + write_option(dstfp, jclorder ++, "cupsJobHoldUntil", "Hold Until", + "job-hold-until", suppattr, defattr, 0, 1); + + if ((defattr = ippFindAttribute(response, "job-priority-default", + IPP_TAG_INTEGER)) != NULL && + (suppattr = ippFindAttribute(response, "job-priority-supported", + IPP_TAG_RANGE)) != NULL) + write_option(dstfp, jclorder ++, "cupsJobPriority", "Priority", + "job-priority", suppattr, defattr, 0, 1); + + if ((defattr = ippFindAttribute(response, "job-sheets-default", + IPP_TAG_ZERO)) != NULL && + (suppattr = ippFindAttribute(response, "job-sheets-supported", + IPP_TAG_ZERO)) != NULL) + { + write_option(dstfp, jclorder ++, "cupsJobSheetsStart", "Start Banner", + "job-sheets", suppattr, defattr, 0, 2); + write_option(dstfp, jclorder, "cupsJobSheetsEnd", "End Banner", + "job-sheets", suppattr, defattr, 1, 2); + } + + cupsFilePuts(dstfp, "*CloseGroup: CUPS\n"); + cupsFileClose(dstfp); + + ippDelete(response); + + return (buffer); +} + + +/* + * 'cupsAdminExportSamba()' - Export a printer to Samba. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 1 on success, 0 on failure */ +cupsAdminExportSamba( + const char *dest, /* I - Destination to export */ + const char *ppd, /* I - PPD file */ + const char *samba_server, /* I - Samba server */ + const char *samba_user, /* I - Samba username */ + const char *samba_password, /* I - Samba password */ + FILE *logfile) /* I - Log file, if any */ +{ + int status; /* Status of Samba commands */ + int have_drivers; /* Have drivers? */ + char file[1024], /* File to test for */ + authfile[1024], /* Temporary authentication file */ + address[1024], /* Address for command */ + subcmd[1024], /* Sub-command */ + message[1024]; /* Error message */ + cups_file_t *fp; /* Authentication file */ + cups_lang_t *language; /* Current language */ + _cups_globals_t *cg = _cupsGlobals(); + /* Global data */ + + + /* + * Range check input... + */ + + if (!dest || !ppd || !samba_server || !samba_user || !samba_password) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + return (0); + } + + /* + * Create a temporary authentication file for Samba... + */ + + if ((fp = cupsTempFile2(authfile, sizeof(authfile))) == NULL) + { + _cupsSetError(IPP_INTERNAL_ERROR, NULL, 0); + return (0); + } + + cupsFilePrintf(fp, "username = %s\n", samba_user); + cupsFilePrintf(fp, "password = %s\n", samba_password); + cupsFileClose(fp); + + /* + * See which drivers are available; the new CUPS v6 and Adobe drivers + * depend on the Windows 2k PS driver, so copy that driver first: + * + * Files: + * + * ps5ui.dll + * pscript.hlp + * pscript.ntf + * pscript5.dll + */ + + have_drivers = 0; + language = cupsLangDefault(); + + snprintf(file, sizeof(file), "%s/drivers/pscript5.dll", cg->cups_datadir); + if (!access(file, 0)) + { + have_drivers |= 1; + + /* + * Windows 2k driver is installed; do the smbclient commands needed + * to copy the Win2k drivers over... + */ + + snprintf(address, sizeof(address), "//%s/print$", samba_server); + + snprintf(subcmd, sizeof(subcmd), + "mkdir W32X86;" + "put %s W32X86/%s.ppd;" + "put %s/drivers/ps5ui.dll W32X86/ps5ui.dll;" + "put %s/drivers/pscript.hlp W32X86/pscript.hlp;" + "put %s/drivers/pscript.ntf W32X86/pscript.ntf;" + "put %s/drivers/pscript5.dll W32X86/pscript5.dll", + ppd, dest, cg->cups_datadir, cg->cups_datadir, + cg->cups_datadir, cg->cups_datadir); + + if ((status = do_samba_command("smbclient", address, subcmd, + authfile, logfile)) != 0) + { + snprintf(message, sizeof(message), + _cupsLangString(language, + _("Unable to copy Windows 2000 printer " + "driver files (%d).")), status); + + _cupsSetError(IPP_INTERNAL_ERROR, message, 0); + + if (logfile) + _cupsLangPuts(logfile, message); + + unlink(authfile); + + return (0); + } + + /* + * See if we also have the CUPS driver files; if so, use them! + */ + + snprintf(file, sizeof(file), "%s/drivers/cupsps6.dll", cg->cups_datadir); + if (!access(file, 0)) + { + /* + * Copy the CUPS driver files over... + */ + + snprintf(subcmd, sizeof(subcmd), + "put %s/drivers/cups6.ini W32X86/cups6.ini;" + "put %s/drivers/cupsps6.dll W32X86/cupsps6.dll;" + "put %s/drivers/cupsui6.dll W32X86/cupsui6.dll", + cg->cups_datadir, cg->cups_datadir, cg->cups_datadir); + + if ((status = do_samba_command("smbclient", address, subcmd, + authfile, logfile)) != 0) + { + snprintf(message, sizeof(message), + _cupsLangString(language, + _("Unable to copy CUPS printer driver " + "files (%d).")), status); + + _cupsSetError(IPP_INTERNAL_ERROR, message, 0); + + if (logfile) + _cupsLangPuts(logfile, message); + + unlink(authfile); + + return (0); + } + + /* + * Do the rpcclient command needed for the CUPS drivers... + */ + + snprintf(subcmd, sizeof(subcmd), + "adddriver \"Windows NT x86\" \"%s:" + "pscript5.dll:%s.ppd:ps5ui.dll:pscript.hlp:NULL:RAW:" + "pscript5.dll,%s.ppd,ps5ui.dll,pscript.hlp,pscript.ntf," + "cups6.ini,cupsps6.dll,cupsui6.dll\"", + dest, dest, dest); + } + else + { + /* + * Don't have the CUPS drivers, so just use the standard Windows + * drivers... + */ + + snprintf(subcmd, sizeof(subcmd), + "adddriver \"Windows NT x86\" \"%s:" + "pscript5.dll:%s.ppd:ps5ui.dll:pscript.hlp:NULL:RAW:" + "pscript5.dll,%s.ppd,ps5ui.dll,pscript.hlp,pscript.ntf\"", + dest, dest, dest); + } + + if ((status = do_samba_command("rpcclient", samba_server, subcmd, + authfile, logfile)) != 0) + { + snprintf(message, sizeof(message), + _cupsLangString(language, + _("Unable to install Windows 2000 printer " + "driver files (%d).")), status); + + _cupsSetError(IPP_INTERNAL_ERROR, message, 0); + + if (logfile) + _cupsLangPuts(logfile, message); + + unlink(authfile); + + return (0); + } + } + + /* + * See if we have the Win9x PS driver... + */ + + snprintf(file, sizeof(file), "%s/drivers/ADOBEPS4.DRV", cg->cups_datadir); + if (!access(file, 0)) + { + have_drivers |= 2; + + /* + * Do the smbclient commands needed for the Adobe Win9x drivers... + */ + + snprintf(address, sizeof(address), "//%s/print$", samba_server); + + snprintf(subcmd, sizeof(subcmd), + "mkdir WIN40;" + "put %s WIN40/%s.PPD;" + "put %s/drivers/ADFONTS.MFM WIN40/ADFONTS.MFM;" + "put %s/drivers/ADOBEPS4.DRV WIN40/ADOBEPS4.DRV;" + "put %s/drivers/ADOBEPS4.HLP WIN40/ADOBEPS4.HLP;" + "put %s/drivers/ICONLIB.DLL WIN40/ICONLIB.DLL;" + "put %s/drivers/PSMON.DLL WIN40/PSMON.DLL;", + ppd, dest, cg->cups_datadir, cg->cups_datadir, + cg->cups_datadir, cg->cups_datadir, cg->cups_datadir); + + if ((status = do_samba_command("smbclient", address, subcmd, + authfile, logfile)) != 0) + { + snprintf(message, sizeof(message), + _cupsLangString(language, + _("Unable to copy Windows 9x printer " + "driver files (%d).")), status); + + _cupsSetError(IPP_INTERNAL_ERROR, message, 0); + + if (logfile) + _cupsLangPuts(logfile, message); + + unlink(authfile); + + return (0); + } + + /* + * Do the rpcclient commands needed for the Adobe Win9x drivers... + */ + + snprintf(subcmd, sizeof(subcmd), + "adddriver \"Windows 4.0\" \"%s:ADOBEPS4.DRV:%s.PPD:NULL:" + "ADOBEPS4.HLP:PSMON.DLL:RAW:" + "ADOBEPS4.DRV,%s.PPD,ADOBEPS4.HLP,PSMON.DLL,ADFONTS.MFM," + "ICONLIB.DLL\"", + dest, dest, dest); + + if ((status = do_samba_command("rpcclient", samba_server, subcmd, + authfile, logfile)) != 0) + { + snprintf(message, sizeof(message), + _cupsLangString(language, + _("Unable to install Windows 9x printer " + "driver files (%d).")), status); + + _cupsSetError(IPP_INTERNAL_ERROR, message, 0); + + if (logfile) + _cupsLangPuts(logfile, message); + + unlink(authfile); + + return (0); + } + } + + /* + * See if we have the 64-bit Windows PS driver... + * + * Files: + * + * x64/ps5ui.dll + * x64/pscript.hlp + * x64/pscript.ntf + * x64/pscript5.dll + */ + + snprintf(file, sizeof(file), "%s/drivers/x64/pscript5.dll", cg->cups_datadir); + if (!access(file, 0)) + { + have_drivers |= 4; + + /* + * 64-bit Windows driver is installed; do the smbclient commands needed + * to copy the Win64 drivers over... + */ + + snprintf(address, sizeof(address), "//%s/print$", samba_server); + + snprintf(subcmd, sizeof(subcmd), + "mkdir x64;" + "put %s x64/%s.ppd;" + "put %s/drivers/x64/ps5ui.dll x64/ps5ui.dll;" + "put %s/drivers/x64/pscript.hlp x64/pscript.hlp;" + "put %s/drivers/x64/pscript.ntf x64/pscript.ntf;" + "put %s/drivers/x64/pscript5.dll x64/pscript5.dll", + ppd, dest, cg->cups_datadir, cg->cups_datadir, + cg->cups_datadir, cg->cups_datadir); + + if ((status = do_samba_command("smbclient", address, subcmd, + authfile, logfile)) != 0) + { + snprintf(message, sizeof(message), + _cupsLangString(language, + _("Unable to copy 64-bit Windows printer " + "driver files (%d).")), status); + + _cupsSetError(IPP_INTERNAL_ERROR, message, 0); + + if (logfile) + _cupsLangPuts(logfile, message); + + unlink(authfile); + + return (0); + } + + /* + * See if we also have the CUPS driver files; if so, use them! + */ + + snprintf(file, sizeof(file), "%s/drivers/x64/cupsps6.dll", cg->cups_datadir); + if (!access(file, 0)) + { + /* + * Copy the CUPS driver files over... + */ + + snprintf(subcmd, sizeof(subcmd), + "put %s/drivers/x64/cups6.ini x64/cups6.ini;" + "put %s/drivers/x64/cupsps6.dll x64/cupsps6.dll;" + "put %s/drivers/x64/cupsui6.dll x64/cupsui6.dll", + cg->cups_datadir, cg->cups_datadir, cg->cups_datadir); + + if ((status = do_samba_command("smbclient", address, subcmd, + authfile, logfile)) != 0) + { + snprintf(message, sizeof(message), + _cupsLangString(language, + _("Unable to copy 64-bit CUPS printer driver " + "files (%d).")), status); + + _cupsSetError(IPP_INTERNAL_ERROR, message, 0); + + if (logfile) + _cupsLangPuts(logfile, message); + + unlink(authfile); + + return (0); + } + + /* + * Do the rpcclient command needed for the CUPS drivers... + */ + + snprintf(subcmd, sizeof(subcmd), + "adddriver \"Windows x64\" \"%s:" + "pscript5.dll:%s.ppd:ps5ui.dll:pscript.hlp:NULL:RAW:" + "pscript5.dll,%s.ppd,ps5ui.dll,pscript.hlp,pscript.ntf," + "cups6.ini,cupsps6.dll,cupsui6.dll\"", + dest, dest, dest); + } + else + { + /* + * Don't have the CUPS drivers, so just use the standard Windows + * drivers... + */ + + snprintf(subcmd, sizeof(subcmd), + "adddriver \"Windows x64\" \"%s:" + "pscript5.dll:%s.ppd:ps5ui.dll:pscript.hlp:NULL:RAW:" + "pscript5.dll,%s.ppd,ps5ui.dll,pscript.hlp,pscript.ntf\"", + dest, dest, dest); + } + + if ((status = do_samba_command("rpcclient", samba_server, subcmd, + authfile, logfile)) != 0) + { + snprintf(message, sizeof(message), + _cupsLangString(language, + _("Unable to install Windows 2000 printer " + "driver files (%d).")), status); + + _cupsSetError(IPP_INTERNAL_ERROR, message, 0); + + if (logfile) + _cupsLangPuts(logfile, message); + + unlink(authfile); + + return (0); + } + } + + if (logfile && !(have_drivers & 1)) + { + if (!have_drivers) + strlcpy(message, + _cupsLangString(language, + _("No Windows printer drivers are installed.")), + sizeof(message)); + else + strlcpy(message, + _cupsLangString(language, + _("Warning, no Windows 2000 printer drivers " + "are installed.")), + sizeof(message)); + + _cupsSetError(IPP_NOT_FOUND, message, 0); + _cupsLangPuts(logfile, message); + } + + if (have_drivers == 0) + { + _cupsSetError(IPP_NOT_FOUND, message, 0); + + unlink(authfile); + + return (0); + } + + /* + * Finally, associate the drivers we just added with the queue... + */ + + snprintf(subcmd, sizeof(subcmd), "setdriver %s %s", dest, dest); + + if ((status = do_samba_command("rpcclient", samba_server, subcmd, + authfile, logfile)) != 0) + { + snprintf(message, sizeof(message), + _cupsLangString(language, + _("Unable to set Windows printer driver (%d).")), + status); + + _cupsSetError(IPP_INTERNAL_ERROR, message, 0); + + if (logfile) + _cupsLangPuts(logfile, message); + + unlink(authfile); + + return (0); + } + + unlink(authfile); + + return (1); +} + + +/* + * 'cupsAdminGetServerSettings()' - Get settings from the server. + * + * The returned settings should be freed with cupsFreeOptions() when + * you are done with them. + * + * @since CUPS 1.3/Mac OS X 10.5@ + */ + +int /* O - 1 on success, 0 on failure */ +cupsAdminGetServerSettings( + http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + int *num_settings, /* O - Number of settings */ + cups_option_t **settings) /* O - Settings */ +{ + int i; /* Looping var */ + cups_file_t *cupsd; /* cupsd.conf file */ + char cupsdconf[1024]; /* cupsd.conf filename */ + int remote; /* Remote cupsd.conf file? */ + http_status_t status; /* Status of getting cupsd.conf */ + char line[1024], /* Line from cupsd.conf file */ + *value; /* Value on line */ + cups_option_t *setting; /* Current setting */ + _cups_globals_t *cg = _cupsGlobals(); /* Global data */ + + + /* + * Range check input... + */ + + if (!http) + { + /* + * See if we are connected to the same server... + */ + + if (cg->http) + { + /* + * Compare the connection hostname, port, and encryption settings to + * the cached defaults; these were initialized the first time we + * connected... + */ + + if (strcmp(cg->http->hostname, cg->server) || + cg->ipp_port != _httpAddrPort(cg->http->hostaddr) || + (cg->http->encryption != cg->encryption && + cg->http->encryption == HTTP_ENCRYPT_NEVER)) + { + /* + * Need to close the current connection because something has changed... + */ + + httpClose(cg->http); + cg->http = NULL; + } + } + + /* + * (Re)connect as needed... + */ + + if (!cg->http) + { + if ((cg->http = _httpCreate(cupsServer(), ippPort(), NULL, + cupsEncryption(), AF_UNSPEC)) == NULL) + { + if (errno) + _cupsSetError(IPP_SERVICE_UNAVAILABLE, NULL, 0); + else + _cupsSetError(IPP_SERVICE_UNAVAILABLE, + _("Unable to connect to host."), 1); + + if (num_settings) + *num_settings = 0; + + if (settings) + *settings = NULL; + + return (0); + } + } + + http = cg->http; + } + + if (!http || !num_settings || !settings) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + + if (num_settings) + *num_settings = 0; + + if (settings) + *settings = NULL; + + return (0); + } + + *num_settings = 0; + *settings = NULL; + + /* + * Get the cupsd.conf file... + */ + + if ((status = get_cupsd_conf(http, cg, cg->cupsd_update, cupsdconf, + sizeof(cupsdconf), &remote)) == HTTP_OK) + { + if ((cupsd = cupsFileOpen(cupsdconf, "r")) == NULL) + { + char message[1024]; /* Message string */ + + + snprintf(message, sizeof(message), + _cupsLangString(cupsLangDefault(), _("Open of %s failed: %s")), + cupsdconf, strerror(errno)); + _cupsSetError(IPP_INTERNAL_ERROR, message, 0); + } + } + else + cupsd = NULL; + + if (cupsd) + { + /* + * Read the file, keeping track of what settings are enabled... + */ + + int remote_access = 0, /* Remote access allowed? */ + remote_admin = 0, /* Remote administration allowed? */ + remote_any = 0, /* Remote access from anywhere allowed? */ + browsing = 1, /* Browsing enabled? */ + cancel_policy = 1, /* Cancel-job policy set? */ + debug_logging = 0; /* LogLevel debug set? */ + int linenum = 0, /* Line number in file */ + in_location = 0, /* In a location section? */ + in_policy = 0, /* In a policy section? */ + in_cancel_job = 0, /* In a cancel-job section? */ + in_admin_location = 0; /* In the /admin location? */ + + + invalidate_cupsd_cache(cg); + + cg->cupsd_update = time(NULL); + httpGetHostname(http, cg->cupsd_hostname, sizeof(cg->cupsd_hostname)); + + while (cupsFileGetConf(cupsd, line, sizeof(line), &value, &linenum)) + { + if (!value && strncmp(line, "")) + { + in_policy = 0; + } + else if (!_cups_strcasecmp(line, "")) + { + in_cancel_job = 0; + } + else if (!_cups_strcasecmp(line, "Require") && in_cancel_job) + { + cancel_policy = 0; + } + else if (!_cups_strcasecmp(line, "")) + { + in_admin_location = 0; + in_location = 0; + } + else if (!_cups_strcasecmp(line, "Allow") && value && + _cups_strcasecmp(value, "localhost") && + _cups_strcasecmp(value, "127.0.0.1") +#ifdef AF_LOCAL + && *value != '/' +#endif /* AF_LOCAL */ +#ifdef AF_INET6 + && strcmp(value, "::1") +#endif /* AF_INET6 */ + ) + { + if (in_admin_location) + remote_admin = 1; + else if (!_cups_strcasecmp(value, "all")) + remote_any = 1; + } + else if (line[0] != '<' && !in_location && !in_policy && + _cups_strcasecmp(line, "Allow") && + _cups_strcasecmp(line, "AuthType") && + _cups_strcasecmp(line, "Deny") && + _cups_strcasecmp(line, "Order") && + _cups_strcasecmp(line, "Require") && + _cups_strcasecmp(line, "Satisfy")) + cg->cupsd_num_settings = cupsAddOption(line, value, + cg->cupsd_num_settings, + &(cg->cupsd_settings)); + } + + cupsFileClose(cupsd); + + cg->cupsd_num_settings = cupsAddOption(CUPS_SERVER_DEBUG_LOGGING, + debug_logging ? "1" : "0", + cg->cupsd_num_settings, + &(cg->cupsd_settings)); + + cg->cupsd_num_settings = cupsAddOption(CUPS_SERVER_REMOTE_ADMIN, + (remote_access && remote_admin) ? + "1" : "0", + cg->cupsd_num_settings, + &(cg->cupsd_settings)); + + cg->cupsd_num_settings = cupsAddOption(CUPS_SERVER_REMOTE_ANY, + remote_any ? "1" : "0", + cg->cupsd_num_settings, + &(cg->cupsd_settings)); + + cg->cupsd_num_settings = cupsAddOption(CUPS_SERVER_SHARE_PRINTERS, + (remote_access && browsing) ? "1" : + "0", + cg->cupsd_num_settings, + &(cg->cupsd_settings)); + + cg->cupsd_num_settings = cupsAddOption(CUPS_SERVER_USER_CANCEL_ANY, + cancel_policy ? "1" : "0", + cg->cupsd_num_settings, + &(cg->cupsd_settings)); + } + else if (status != HTTP_NOT_MODIFIED) + invalidate_cupsd_cache(cg); + + /* + * Remove any temporary files and copy the settings array... + */ + + if (remote) + unlink(cupsdconf); + + for (i = cg->cupsd_num_settings, setting = cg->cupsd_settings; + i > 0; + i --, setting ++) + *num_settings = cupsAddOption(setting->name, setting->value, + *num_settings, settings); + + return (cg->cupsd_num_settings > 0); +} + + +/* + * 'cupsAdminSetServerSettings()' - Set settings on the server. + * + * @since CUPS 1.3/Mac OS X 10.5@ + */ + +int /* O - 1 on success, 0 on failure */ +cupsAdminSetServerSettings( + http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + int num_settings, /* I - Number of settings */ + cups_option_t *settings) /* I - Settings */ +{ + int i; /* Looping var */ + http_status_t status; /* GET/PUT status */ + const char *server_port_env; /* SERVER_PORT env var */ + int server_port; /* IPP port for server */ + cups_file_t *cupsd; /* cupsd.conf file */ + char cupsdconf[1024]; /* cupsd.conf filename */ + int remote; /* Remote cupsd.conf file? */ + char tempfile[1024]; /* Temporary new cupsd.conf */ + cups_file_t *temp; /* Temporary file */ + char line[1024], /* Line from cupsd.conf file */ + *value; /* Value on line */ + int linenum, /* Line number in file */ + in_location, /* In a location section? */ + in_policy, /* In a policy section? */ + in_default_policy, /* In the default policy section? */ + in_cancel_job, /* In a cancel-job section? */ + in_admin_location, /* In the /admin location? */ + in_conf_location, /* In the /admin/conf location? */ + in_root_location; /* In the / location? */ + const char *val; /* Setting value */ + int share_printers, /* Share local printers */ + remote_admin, /* Remote administration allowed? */ + remote_any, /* Remote access from anywhere? */ + user_cancel_any, /* Cancel-job policy set? */ + debug_logging; /* LogLevel debug set? */ + int wrote_port_listen, /* Wrote the port/listen lines? */ + wrote_browsing, /* Wrote the browsing lines? */ + wrote_policy, /* Wrote the policy? */ + wrote_loglevel, /* Wrote the LogLevel line? */ + wrote_admin_location, /* Wrote the /admin location? */ + wrote_conf_location, /* Wrote the /admin/conf location? */ + wrote_root_location; /* Wrote the / location? */ + int indent; /* Indentation */ + int cupsd_num_settings; /* New number of settings */ + int old_share_printers, /* Share local printers */ + old_remote_admin, /* Remote administration allowed? */ + old_user_cancel_any, /* Cancel-job policy set? */ + old_debug_logging; /* LogLevel debug set? */ + cups_option_t *cupsd_settings, /* New settings */ + *setting; /* Current setting */ + _cups_globals_t *cg = _cupsGlobals(); /* Global data */ + + + /* + * Range check input... + */ + + if (!http) + http = _cupsConnect(); + + if (!http || !num_settings || !settings) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + + return (0); + } + + /* + * Get the cupsd.conf file... + */ + + if (get_cupsd_conf(http, cg, 0, cupsdconf, sizeof(cupsdconf), + &remote) == HTTP_OK) + { + if ((cupsd = cupsFileOpen(cupsdconf, "r")) == NULL) + { + _cupsSetError(IPP_INTERNAL_ERROR, NULL, 0); + return (0); + } + } + else + return (0); + + /* + * Get current settings... + */ + + if (!cupsAdminGetServerSettings(http, &cupsd_num_settings, + &cupsd_settings)) + return (0); + + if ((val = cupsGetOption(CUPS_SERVER_DEBUG_LOGGING, cupsd_num_settings, + cupsd_settings)) != NULL) + old_debug_logging = atoi(val); + else + old_debug_logging = 0; + + DEBUG_printf(("1cupsAdminSetServerSettings: old debug_logging=%d", + old_debug_logging)); + + if ((val = cupsGetOption(CUPS_SERVER_REMOTE_ADMIN, cupsd_num_settings, + cupsd_settings)) != NULL) + old_remote_admin = atoi(val); + else + old_remote_admin = 0; + + DEBUG_printf(("1cupsAdminSetServerSettings: old remote_admin=%d", + old_remote_admin)); + + if ((val = cupsGetOption(CUPS_SERVER_REMOTE_ANY, cupsd_num_settings, + cupsd_settings)) != NULL) + remote_any = atoi(val); + else + remote_any = 0; + + DEBUG_printf(("1cupsAdminSetServerSettings: old remote_any=%d", + remote_any)); + + if ((val = cupsGetOption(CUPS_SERVER_SHARE_PRINTERS, cupsd_num_settings, + cupsd_settings)) != NULL) + old_share_printers = atoi(val); + else + old_share_printers = 0; + + DEBUG_printf(("1cupsAdminSetServerSettings: old share_printers=%d", + old_share_printers)); + + if ((val = cupsGetOption(CUPS_SERVER_USER_CANCEL_ANY, cupsd_num_settings, + cupsd_settings)) != NULL) + old_user_cancel_any = atoi(val); + else + old_user_cancel_any = 0; + + DEBUG_printf(("1cupsAdminSetServerSettings: old user_cancel_any=%d", + old_user_cancel_any)); + + cupsFreeOptions(cupsd_num_settings, cupsd_settings); + + /* + * Get basic settings... + */ + + if ((val = cupsGetOption(CUPS_SERVER_DEBUG_LOGGING, num_settings, + settings)) != NULL) + { + debug_logging = atoi(val); + + if (debug_logging == old_debug_logging) + { + /* + * No change to this setting... + */ + + debug_logging = -1; + } + } + else + debug_logging = -1; + + DEBUG_printf(("1cupsAdminSetServerSettings: debug_logging=%d", + debug_logging)); + + if ((val = cupsGetOption(CUPS_SERVER_REMOTE_ANY, num_settings, + settings)) != NULL) + remote_any = atoi(val); + + DEBUG_printf(("1cupsAdminSetServerSettings: remote_any=%d", + remote_any)); + + if ((val = cupsGetOption(CUPS_SERVER_REMOTE_ADMIN, num_settings, + settings)) != NULL) + { + remote_admin = atoi(val); + + if (remote_admin == old_remote_admin) + { + /* + * No change to this setting... + */ + + remote_admin = -1; + } + } + else + remote_admin = -1; + + DEBUG_printf(("1cupsAdminSetServerSettings: remote_admin=%d", + remote_admin)); + + if ((val = cupsGetOption(CUPS_SERVER_SHARE_PRINTERS, num_settings, + settings)) != NULL) + { + share_printers = atoi(val); + + if (share_printers == old_share_printers) + { + /* + * No change to this setting... + */ + + share_printers = -1; + } + } + else + share_printers = -1; + + DEBUG_printf(("1cupsAdminSetServerSettings: share_printers=%d", + share_printers)); + + if ((val = cupsGetOption(CUPS_SERVER_USER_CANCEL_ANY, num_settings, + settings)) != NULL) + { + user_cancel_any = atoi(val); + + if (user_cancel_any == old_user_cancel_any) + { + /* + * No change to this setting... + */ + + user_cancel_any = -1; + } + } + else + user_cancel_any = -1; + + DEBUG_printf(("1cupsAdminSetServerSettings: user_cancel_any=%d", + user_cancel_any)); + + /* + * Create a temporary file for the new cupsd.conf file... + */ + + if ((temp = cupsTempFile2(tempfile, sizeof(tempfile))) == NULL) + { + cupsFileClose(cupsd); + + if (remote) + unlink(cupsdconf); + + _cupsSetError(IPP_INTERNAL_ERROR, NULL, 0); + return (0); + } + + /* + * Copy the old file to the new, making changes along the way... + */ + + cupsd_num_settings = 0; + in_admin_location = 0; + in_cancel_job = 0; + in_conf_location = 0; + in_default_policy = 0; + in_location = 0; + in_policy = 0; + in_root_location = 0; + linenum = 0; + wrote_admin_location = 0; + wrote_browsing = 0; + wrote_conf_location = 0; + wrote_loglevel = 0; + wrote_policy = 0; + wrote_port_listen = 0; + wrote_root_location = 0; + indent = 0; + + if ((server_port_env = getenv("SERVER_PORT")) != NULL) + { + if ((server_port = atoi(server_port_env)) <= 0) + server_port = ippPort(); + } + else + server_port = ippPort(); + + if (server_port <= 0) + server_port = IPP_PORT; + + while (cupsFileGetConf(cupsd, line, sizeof(line), &value, &linenum)) + { + if ((!_cups_strcasecmp(line, "Port") || !_cups_strcasecmp(line, "Listen")) && + (remote_admin >= 0 || remote_any > 0 || share_printers >= 0)) + { + if (!wrote_port_listen) + { + wrote_port_listen = 1; + + if (remote_admin > 0 || remote_any > 0 || share_printers > 0) + { + cupsFilePuts(temp, "# Allow remote access\n"); + cupsFilePrintf(temp, "Port %d\n", server_port); + } + else + { + cupsFilePuts(temp, "# Only listen for connections from the local " + "machine.\n"); + cupsFilePrintf(temp, "Listen localhost:%d\n", server_port); + } + +#ifdef CUPS_DEFAULT_DOMAINSOCKET + if ((!value || strcmp(CUPS_DEFAULT_DOMAINSOCKET, value)) && + !access(CUPS_DEFAULT_DOMAINSOCKET, 0)) + cupsFilePuts(temp, "Listen " CUPS_DEFAULT_DOMAINSOCKET "\n"); +#endif /* CUPS_DEFAULT_DOMAINSOCKET */ + } + else if (value && value[0] == '/' +#ifdef CUPS_DEFAULT_DOMAINSOCKET + && strcmp(CUPS_DEFAULT_DOMAINSOCKET, value) +#endif /* CUPS_DEFAULT_DOMAINSOCKET */ + ) + cupsFilePrintf(temp, "Listen %s\n", value); + } + else if ((!_cups_strcasecmp(line, "Browsing") || + !_cups_strcasecmp(line, "BrowseLocalProtocols")) && + share_printers >= 0) + { + if (!wrote_browsing) + { + int new_share_printers = (share_printers > 0 || + (share_printers == -1 && + old_share_printers > 0)); + + wrote_browsing = 1; + + if (new_share_printers) + { + const char *localp = cupsGetOption("BrowseLocalProtocols", + num_settings, settings); + + if (!localp || !localp[0]) + localp = cupsGetOption("BrowseLocalProtocols", cupsd_num_settings, + cupsd_settings); + + cupsFilePuts(temp, "# Share local printers on the local network.\n"); + cupsFilePuts(temp, "Browsing On\n"); + + if (!localp) + localp = CUPS_DEFAULT_BROWSE_LOCAL_PROTOCOLS; + + cupsFilePrintf(temp, "BrowseLocalProtocols %s\n", localp); + + cupsd_num_settings = cupsAddOption("BrowseLocalProtocols", localp, + cupsd_num_settings, + &cupsd_settings); + } + else + { + cupsFilePuts(temp, "# Disable printer sharing.\n"); + cupsFilePuts(temp, "Browsing Off\n"); + } + } + } + else if (!_cups_strcasecmp(line, "LogLevel") && debug_logging >= 0) + { + wrote_loglevel = 1; + + if (debug_logging) + { + cupsFilePuts(temp, + "# Show troubleshooting information in error_log.\n"); + cupsFilePuts(temp, "LogLevel debug\n"); + } + else + { + cupsFilePuts(temp, "# Show general information in error_log.\n"); + cupsFilePuts(temp, "LogLevel " CUPS_DEFAULT_LOG_LEVEL "\n"); + } + } + else if (!_cups_strcasecmp(line, "\n", line, value); + indent += 2; + } + else if (!_cups_strcasecmp(line, "")) + { + indent -= 2; + if (!wrote_policy && in_default_policy) + { + wrote_policy = 1; + + if (!user_cancel_any) + cupsFilePuts(temp, " # Only the owner or an administrator can " + "cancel a job...\n" + " \n" + " Order deny,allow\n" + " Require user @OWNER " + CUPS_DEFAULT_PRINTOPERATOR_AUTH "\n" + " \n"); + } + + in_policy = 0; + in_default_policy = 0; + + cupsFilePuts(temp, "\n"); + } + else if (!_cups_strcasecmp(line, "\n", line, value); + } + else if (!_cups_strcasecmp(line, "")) + { + in_location = 0; + indent -= 2; + if (in_admin_location && remote_admin >= 0) + { + wrote_admin_location = 1; + + if (remote_admin) + cupsFilePuts(temp, " # Allow remote administration...\n"); + else if (remote_admin == 0) + cupsFilePuts(temp, " # Restrict access to the admin pages...\n"); + + cupsFilePuts(temp, " Order allow,deny\n"); + + if (remote_admin) + cupsFilePrintf(temp, " Allow %s\n", + remote_any > 0 ? "all" : "@LOCAL"); + } + else if (in_conf_location && remote_admin >= 0) + { + wrote_conf_location = 1; + + if (remote_admin) + cupsFilePuts(temp, " # Allow remote access to the configuration " + "files...\n"); + else + cupsFilePuts(temp, " # Restrict access to the configuration " + "files...\n"); + + cupsFilePuts(temp, " Order allow,deny\n"); + + if (remote_admin) + cupsFilePrintf(temp, " Allow %s\n", + remote_any > 0 ? "all" : "@LOCAL"); + } + else if (in_root_location && + (remote_admin >= 0 || remote_any > 0 || share_printers >= 0)) + { + wrote_root_location = 1; + + if (remote_admin > 0 && share_printers > 0) + cupsFilePuts(temp, " # Allow shared printing and remote " + "administration...\n"); + else if (remote_admin > 0) + cupsFilePuts(temp, " # Allow remote administration...\n"); + else if (share_printers > 0) + cupsFilePuts(temp, " # Allow shared printing...\n"); + else if (remote_any > 0) + cupsFilePuts(temp, " # Allow remote access...\n"); + else + cupsFilePuts(temp, " # Restrict access to the server...\n"); + + cupsFilePuts(temp, " Order allow,deny\n"); + + if (remote_admin > 0 || remote_any > 0 || share_printers > 0) + cupsFilePrintf(temp, " Allow %s\n", + remote_any > 0 ? "all" : "@LOCAL"); + } + + in_admin_location = 0; + in_conf_location = 0; + in_root_location = 0; + + cupsFilePuts(temp, "\n"); + } + else if (!_cups_strcasecmp(line, "= 0) + { + /* + * Don't write anything for this limit section... + */ + + in_cancel_job = 2; + } + else + { + cupsFilePrintf(temp, "%*s%s", indent, "", line); + + while (*value) + { + for (valptr = value; *valptr && !_cups_isspace(*valptr); valptr ++); + + if (*valptr) + *valptr++ = '\0'; + + if (!_cups_strcasecmp(value, "cancel-job") && user_cancel_any >= 0) + { + /* + * Write everything except for this definition... + */ + + in_cancel_job = 1; + } + else + cupsFilePrintf(temp, " %s", value); + + for (value = valptr; _cups_isspace(*value); value ++); + } + + cupsFilePuts(temp, ">\n"); + } + } + else + cupsFilePrintf(temp, "%*s%s %s>\n", indent, "", line, value); + + indent += 2; + } + else if (!_cups_strcasecmp(line, "
") && in_cancel_job) + { + indent -= 2; + + if (in_cancel_job == 1) + cupsFilePuts(temp, " \n"); + + wrote_policy = 1; + + if (!user_cancel_any) + cupsFilePuts(temp, " # Only the owner or an administrator can cancel " + "a job...\n" + " \n" + " Order deny,allow\n" + " Require user @OWNER " + CUPS_DEFAULT_PRINTOPERATOR_AUTH "\n" + " \n"); + + in_cancel_job = 0; + } + else if ((((in_admin_location || in_conf_location || in_root_location) && + (remote_admin >= 0 || remote_any > 0)) || + (in_root_location && share_printers >= 0)) && + (!_cups_strcasecmp(line, "Allow") || !_cups_strcasecmp(line, "Deny") || + !_cups_strcasecmp(line, "Order"))) + continue; + else if (in_cancel_job == 2) + continue; + else if (line[0] == '<') + { + if (value) + { + cupsFilePrintf(temp, "%*s%s %s>\n", indent, "", line, value); + indent += 2; + } + else + { + if (line[1] == '/') + indent -= 2; + + cupsFilePrintf(temp, "%*s%s\n", indent, "", line); + } + } + else if (!in_policy && !in_location && + (val = cupsGetOption(line, num_settings, settings)) != NULL) + { + /* + * Replace this directive's value with the new one... + */ + + cupsd_num_settings = cupsAddOption(line, val, cupsd_num_settings, + &cupsd_settings); + + /* + * Write the new value in its place, without indentation since we + * only support setting root directives, not in sections... + */ + + cupsFilePrintf(temp, "%s %s\n", line, val); + } + else if (value) + { + if (!in_policy && !in_location) + { + /* + * Record the non-policy, non-location directives that we find + * in the server settings, since we cache this info and record it + * in cupsAdminGetServerSettings()... + */ + + cupsd_num_settings = cupsAddOption(line, value, cupsd_num_settings, + &cupsd_settings); + } + + cupsFilePrintf(temp, "%*s%s %s\n", indent, "", line, value); + } + else + cupsFilePrintf(temp, "%*s%s\n", indent, "", line); + } + + /* + * Write any missing info... + */ + + if (!wrote_browsing && share_printers >= 0) + { + if (share_printers > 0) + { + cupsFilePuts(temp, "# Share local printers on the local network.\n"); + cupsFilePuts(temp, "Browsing On\n"); + } + else + { + cupsFilePuts(temp, "# Disable printer sharing and shared printers.\n"); + cupsFilePuts(temp, "Browsing Off\n"); + } + } + + if (!wrote_loglevel && debug_logging >= 0) + { + if (debug_logging) + { + cupsFilePuts(temp, "# Show troubleshooting information in error_log.\n"); + cupsFilePuts(temp, "LogLevel debug\n"); + } + else + { + cupsFilePuts(temp, "# Show general information in error_log.\n"); + cupsFilePuts(temp, "LogLevel " CUPS_DEFAULT_LOG_LEVEL "\n"); + } + } + + if (!wrote_port_listen && + (remote_admin >= 0 || remote_any > 0 || share_printers >= 0)) + { + if (remote_admin > 0 || remote_any > 0 || share_printers > 0) + { + cupsFilePuts(temp, "# Allow remote access\n"); + cupsFilePrintf(temp, "Port %d\n", ippPort()); + } + else + { + cupsFilePuts(temp, + "# Only listen for connections from the local machine.\n"); + cupsFilePrintf(temp, "Listen localhost:%d\n", ippPort()); + } + +#ifdef CUPS_DEFAULT_DOMAINSOCKET + if (!access(CUPS_DEFAULT_DOMAINSOCKET, 0)) + cupsFilePuts(temp, "Listen " CUPS_DEFAULT_DOMAINSOCKET "\n"); +#endif /* CUPS_DEFAULT_DOMAINSOCKET */ + } + + if (!wrote_root_location && + (remote_admin >= 0 || remote_any > 0 || share_printers >= 0)) + { + if (remote_admin > 0 && share_printers > 0) + cupsFilePuts(temp, + "# Allow shared printing and remote administration...\n"); + else if (remote_admin > 0) + cupsFilePuts(temp, "# Allow remote administration...\n"); + else if (share_printers > 0) + cupsFilePuts(temp, "# Allow shared printing...\n"); + else if (remote_any > 0) + cupsFilePuts(temp, "# Allow remote access...\n"); + else + cupsFilePuts(temp, "# Restrict access to the server...\n"); + + cupsFilePuts(temp, "\n" + " Order allow,deny\n"); + + if (remote_admin > 0 || remote_any > 0 || share_printers > 0) + cupsFilePrintf(temp, " Allow %s\n", remote_any > 0 ? "all" : "@LOCAL"); + + cupsFilePuts(temp, "\n"); + } + + if (!wrote_admin_location && remote_admin >= 0) + { + if (remote_admin) + cupsFilePuts(temp, "# Allow remote administration...\n"); + else + cupsFilePuts(temp, "# Restrict access to the admin pages...\n"); + + cupsFilePuts(temp, "\n" + " Order allow,deny\n"); + + if (remote_admin) + cupsFilePrintf(temp, " Allow %s\n", remote_any > 0 ? "all" : "@LOCAL"); + + cupsFilePuts(temp, "\n"); + } + + if (!wrote_conf_location && remote_admin >= 0) + { + if (remote_admin) + cupsFilePuts(temp, + "# Allow remote access to the configuration files...\n"); + else + cupsFilePuts(temp, "# Restrict access to the configuration files...\n"); + + cupsFilePuts(temp, "\n" + " AuthType Default\n" + " Require user @SYSTEM\n" + " Order allow,deny\n"); + + if (remote_admin) + cupsFilePrintf(temp, " Allow %s\n", remote_any > 0 ? "all" : "@LOCAL"); + + cupsFilePuts(temp, "\n"); + } + + if (!wrote_policy && user_cancel_any >= 0) + { + cupsFilePuts(temp, "\n" + " # Job-related operations must be done by the owner " + "or an administrator...\n" + " \n" + " Require user @OWNER @SYSTEM\n" + " Order deny,allow\n" + " \n" + " # All administration operations require an " + "administrator to authenticate...\n" + " \n" + " AuthType Default\n" + " Require user @SYSTEM\n" + " Order deny,allow\n" + "\n"); + + if (!user_cancel_any) + cupsFilePuts(temp, " # Only the owner or an administrator can cancel " + "a job...\n" + " \n" + " Order deny,allow\n" + " Require user @OWNER " + CUPS_DEFAULT_PRINTOPERATOR_AUTH "\n" + " \n"); + + cupsFilePuts(temp, " \n" + " Order deny,allow\n" + " \n" + "\n"); + } + + for (i = num_settings, setting = settings; i > 0; i --, setting ++) + if (setting->name[0] != '_' && + _cups_strcasecmp(setting->name, "Listen") && + _cups_strcasecmp(setting->name, "Port") && + !cupsGetOption(setting->name, cupsd_num_settings, cupsd_settings)) + { + /* + * Add this directive to the list of directives we have written... + */ + + cupsd_num_settings = cupsAddOption(setting->name, setting->value, + cupsd_num_settings, &cupsd_settings); + + /* + * Write the new value, without indentation since we only support + * setting root directives, not in sections... + */ + + cupsFilePrintf(temp, "%s %s\n", setting->name, setting->value); + } + + cupsFileClose(cupsd); + cupsFileClose(temp); + + /* + * Upload the configuration file to the server... + */ + + status = cupsPutFile(http, "/admin/conf/cupsd.conf", tempfile); + + if (status == HTTP_CREATED) + { + /* + * Updated OK, add the basic settings... + */ + + if (debug_logging >= 0) + cupsd_num_settings = cupsAddOption(CUPS_SERVER_DEBUG_LOGGING, + debug_logging ? "1" : "0", + cupsd_num_settings, &cupsd_settings); + else + cupsd_num_settings = cupsAddOption(CUPS_SERVER_DEBUG_LOGGING, + old_debug_logging ? "1" : "0", + cupsd_num_settings, &cupsd_settings); + + if (remote_admin >= 0) + cupsd_num_settings = cupsAddOption(CUPS_SERVER_REMOTE_ADMIN, + remote_admin ? "1" : "0", + cupsd_num_settings, &cupsd_settings); + else + cupsd_num_settings = cupsAddOption(CUPS_SERVER_REMOTE_ADMIN, + old_remote_admin ? "1" : "0", + cupsd_num_settings, &cupsd_settings); + + cupsd_num_settings = cupsAddOption(CUPS_SERVER_REMOTE_ANY, + remote_any ? "1" : "0", + cupsd_num_settings, &cupsd_settings); + + if (share_printers >= 0) + cupsd_num_settings = cupsAddOption(CUPS_SERVER_SHARE_PRINTERS, + share_printers ? "1" : "0", + cupsd_num_settings, &cupsd_settings); + else + cupsd_num_settings = cupsAddOption(CUPS_SERVER_SHARE_PRINTERS, + old_share_printers ? "1" : "0", + cupsd_num_settings, &cupsd_settings); + + if (user_cancel_any >= 0) + cupsd_num_settings = cupsAddOption(CUPS_SERVER_USER_CANCEL_ANY, + user_cancel_any ? "1" : "0", + cupsd_num_settings, &cupsd_settings); + else + cupsd_num_settings = cupsAddOption(CUPS_SERVER_USER_CANCEL_ANY, + old_user_cancel_any ? "1" : "0", + cupsd_num_settings, &cupsd_settings); + + /* + * Save the new values... + */ + + invalidate_cupsd_cache(cg); + + cg->cupsd_num_settings = cupsd_num_settings; + cg->cupsd_settings = cupsd_settings; + cg->cupsd_update = time(NULL); + + httpGetHostname(http, cg->cupsd_hostname, sizeof(cg->cupsd_hostname)); + } + else + cupsFreeOptions(cupsd_num_settings, cupsd_settings); + + /* + * Remote our temp files and return... + */ + + if (remote) + unlink(cupsdconf); + + unlink(tempfile); + + return (status == HTTP_CREATED); +} + + +/* + * 'do_samba_command()' - Do a SAMBA command. + */ + +static int /* O - Status of command */ +do_samba_command(const char *command, /* I - Command to run */ + const char *address, /* I - Address for command */ + const char *subcmd, /* I - Sub-command */ + const char *authfile, /* I - Samba authentication file */ + FILE *logfile) /* I - Optional log file */ +{ +#ifdef WIN32 + return (1); /* Always fail on Windows... */ + +#else + int status; /* Status of command */ + int pid; /* Process ID of child */ + + + if (logfile) + _cupsLangPrintf(logfile, + _("Running command: %s %s -N -A %s -c \'%s\'"), + command, address, authfile, subcmd); + + if ((pid = fork()) == 0) + { + /* + * Child goes here, redirect stdin/out/err and execute the command... + */ + + int fd = open("/dev/null", O_RDONLY); + + if (fd > 0) + { + dup2(fd, 0); + close(fd); + } + + if (logfile) + dup2(fileno(logfile), 1); + else if ((fd = open("/dev/null", O_WRONLY)) > 1) + { + dup2(fd, 1); + close(fd); + } + + dup2(1, 2); + + execlp(command, command, address, "-N", "-A", authfile, "-c", subcmd, + (char *)0); + exit(errno); + } + else if (pid < 0) + { + status = -1; + + if (logfile) + _cupsLangPrintf(logfile, _("Unable to run \"%s\": %s"), + command, strerror(errno)); + } + else + { + /* + * Wait for the process to complete... + */ + + while (wait(&status) != pid); + } + + if (logfile) + _cupsLangPuts(logfile, ""); + + DEBUG_printf(("9do_samba_command: status=%d", status)); + + if (WIFEXITED(status)) + return (WEXITSTATUS(status)); + else + return (-WTERMSIG(status)); +#endif /* WIN32 */ +} + + +/* + * 'get_cupsd_conf()' - Get the current cupsd.conf file. + */ + +static http_status_t /* O - Status of request */ +get_cupsd_conf( + http_t *http, /* I - Connection to server */ + _cups_globals_t *cg, /* I - Global data */ + time_t last_update, /* I - Last update time for file */ + char *name, /* I - Filename buffer */ + int namesize, /* I - Size of filename buffer */ + int *remote) /* O - Remote file? */ +{ + int fd; /* Temporary file descriptor */ +#ifndef WIN32 + struct stat info; /* cupsd.conf file information */ +#endif /* WIN32 */ + http_status_t status; /* Status of getting cupsd.conf */ + char host[HTTP_MAX_HOST]; /* Hostname for connection */ + + + /* + * See if we already have the data we need... + */ + + httpGetHostname(http, host, sizeof(host)); + + if (_cups_strcasecmp(cg->cupsd_hostname, host)) + invalidate_cupsd_cache(cg); + + snprintf(name, namesize, "%s/cupsd.conf", cg->cups_serverroot); + *remote = 0; + +#ifndef WIN32 + if (!_cups_strcasecmp(host, "localhost") && !access(name, R_OK)) + { + /* + * Read the local file rather than using HTTP... + */ + + if (stat(name, &info)) + { + char message[1024]; /* Message string */ + + + snprintf(message, sizeof(message), + _cupsLangString(cupsLangDefault(), _("stat of %s failed: %s")), + name, strerror(errno)); + _cupsSetError(IPP_INTERNAL_ERROR, message, 0); + + *name = '\0'; + + return (HTTP_SERVER_ERROR); + } + else if (last_update && info.st_mtime <= last_update) + status = HTTP_NOT_MODIFIED; + else + status = HTTP_OK; + } + else +#endif /* !WIN32 */ + { + /* + * Read cupsd.conf via a HTTP GET request... + */ + + if ((fd = cupsTempFd(name, namesize)) < 0) + { + *name = '\0'; + + _cupsSetError(IPP_INTERNAL_ERROR, NULL, 0); + + invalidate_cupsd_cache(cg); + + return (HTTP_SERVER_ERROR); + } + + *remote = 1; + + httpClearFields(http); + + if (last_update) + httpSetField(http, HTTP_FIELD_IF_MODIFIED_SINCE, + httpGetDateString(last_update)); + + status = cupsGetFd(http, "/admin/conf/cupsd.conf", fd); + + close(fd); + + if (status != HTTP_OK) + { + unlink(name); + *name = '\0'; + } + } + + return (status); +} + + +/* + * 'invalidate_cupsd_cache()' - Invalidate the cached cupsd.conf settings. + */ + +static void +invalidate_cupsd_cache( + _cups_globals_t *cg) /* I - Global data */ +{ + cupsFreeOptions(cg->cupsd_num_settings, cg->cupsd_settings); + + cg->cupsd_hostname[0] = '\0'; + cg->cupsd_update = 0; + cg->cupsd_num_settings = 0; + cg->cupsd_settings = NULL; +} + + +/* + * 'write_option()' - Write a CUPS option to a PPD file. + */ + +static void +write_option(cups_file_t *dstfp, /* I - PPD file */ + int order, /* I - Order dependency */ + const char *name, /* I - Option name */ + const char *text, /* I - Option text */ + const char *attrname, /* I - Attribute name */ + ipp_attribute_t *suppattr, /* I - IPP -supported attribute */ + ipp_attribute_t *defattr, /* I - IPP -default attribute */ + int defval, /* I - Default value number */ + int valcount) /* I - Number of values */ +{ + int i; /* Looping var */ + + + cupsFilePrintf(dstfp, "*JCLOpenUI *%s/%s: PickOne\n" + "*OrderDependency: %d JCLSetup *%s\n", + name, text, order, name); + + if (defattr->value_tag == IPP_TAG_INTEGER) + { + /* + * Do numeric options with a range or list... + */ + + cupsFilePrintf(dstfp, "*Default%s: %d\n", name, + defattr->values[defval].integer); + + if (suppattr->value_tag == IPP_TAG_RANGE) + { + /* + * List each number in the range... + */ + + for (i = suppattr->values[0].range.lower; + i <= suppattr->values[0].range.upper; + i ++) + { + cupsFilePrintf(dstfp, "*%s %d: \"", name, i); + + if (valcount == 1) + cupsFilePrintf(dstfp, "%%cupsJobTicket: %s=%d\n\"\n*End\n", + attrname, i); + else if (defval == 0) + cupsFilePrintf(dstfp, "%%cupsJobTicket: %s=%d\"\n", attrname, i); + else if (defval < (valcount - 1)) + cupsFilePrintf(dstfp, ",%d\"\n", i); + else + cupsFilePrintf(dstfp, ",%d\n\"\n*End\n", i); + } + } + else + { + /* + * List explicit numbers... + */ + + for (i = 0; i < suppattr->num_values; i ++) + { + cupsFilePrintf(dstfp, "*%s %d: \"", name, suppattr->values[i].integer); + + if (valcount == 1) + cupsFilePrintf(dstfp, "%%cupsJobTicket: %s=%d\n\"\n*End\n", attrname, + suppattr->values[i].integer); + else if (defval == 0) + cupsFilePrintf(dstfp, "%%cupsJobTicket: %s=%d\"\n", attrname, + suppattr->values[i].integer); + else if (defval < (valcount - 1)) + cupsFilePrintf(dstfp, ",%d\"\n", suppattr->values[i].integer); + else + cupsFilePrintf(dstfp, ",%d\n\"\n*End\n", suppattr->values[i].integer); + } + } + } + else + { + /* + * Do text options with a list... + */ + + cupsFilePrintf(dstfp, "*Default%s: %s\n", name, + defattr->values[defval].string.text); + + for (i = 0; i < suppattr->num_values; i ++) + { + cupsFilePrintf(dstfp, "*%s %s: \"", name, + suppattr->values[i].string.text); + + if (valcount == 1) + cupsFilePrintf(dstfp, "%%cupsJobTicket: %s=%s\n\"\n*End\n", attrname, + suppattr->values[i].string.text); + else if (defval == 0) + cupsFilePrintf(dstfp, "%%cupsJobTicket: %s=%s\"\n", attrname, + suppattr->values[i].string.text); + else if (defval < (valcount - 1)) + cupsFilePrintf(dstfp, ",%s\"\n", suppattr->values[i].string.text); + else + cupsFilePrintf(dstfp, ",%s\n\"\n*End\n", + suppattr->values[i].string.text); + } + } + + cupsFilePrintf(dstfp, "*JCLCloseUI: *%s\n\n", name); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/adminutil.h b/cups/adminutil.h new file mode 100644 index 0000000000..37f9c1e0d4 --- /dev/null +++ b/cups/adminutil.h @@ -0,0 +1,78 @@ +/* + * "$Id$" + * + * Administration utility API definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 2001-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_ADMINUTIL_H_ +# define _CUPS_ADMINUTIL_H_ + +/* + * Include necessary headers... + */ + +# include +# include "cups.h" + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Constants... + */ + +# define CUPS_SERVER_DEBUG_LOGGING "_debug_logging" +# define CUPS_SERVER_REMOTE_ADMIN "_remote_admin" +# define CUPS_SERVER_REMOTE_ANY "_remote_any" +/*# define CUPS_SERVER_REMOTE_PRINTERS "_remote_printers"*/ +# define CUPS_SERVER_SHARE_PRINTERS "_share_printers" +# define CUPS_SERVER_USER_CANCEL_ANY "_user_cancel_any" + + +/* + * Functions... + */ + +extern int cupsAdminExportSamba(const char *dest, const char *ppd, + const char *samba_server, + const char *samba_user, + const char *samba_password, + FILE *logfile) _CUPS_API_1_2; +extern char *cupsAdminCreateWindowsPPD(http_t *http, const char *dest, + char *buffer, int bufsize) _CUPS_API_1_2; + +extern int cupsAdminGetServerSettings(http_t *http, + int *num_settings, + cups_option_t **settings) _CUPS_API_1_3; +extern int cupsAdminSetServerSettings(http_t *http, + int num_settings, + cups_option_t *settings) _CUPS_API_1_3; + + +# ifdef __cplusplus +} +# endif /* __cplusplus */ + +#endif /* !_CUPS_ADMINUTIL_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/api-array.header b/cups/api-array.header new file mode 100644 index 0000000000..88e5341f86 --- /dev/null +++ b/cups/api-array.header @@ -0,0 +1,34 @@ + + +

Array API

+ +
+ + + + + + + + + + + + + + + + +
Headercups/array.h
Library-lcups
See AlsoProgramming: Introduction to CUPS Programming
diff --git a/cups/api-array.shtml b/cups/api-array.shtml new file mode 100644 index 0000000000..7406a0d45a --- /dev/null +++ b/cups/api-array.shtml @@ -0,0 +1,196 @@ + + +

Overview

+ +

The CUPS array API provides a high-performance generic array container. +The contents of the array container can be sorted and the container itself is +designed for optimal speed and memory usage under a wide variety of conditions. +Sorted arrays use a binary search algorithm from the last found or inserted +element to quickly find matching elements in the array. Arrays created with the +optional hash function can often find elements with a single lookup. The +cups_array_t type is used when +referring to a CUPS array.

+ +

The CUPS scheduler (cupsd) and many of the CUPS API +functions use the array API to efficiently manage large lists of +data.

+ +

Managing Arrays

+ +

Arrays are created using either the +cupsArrayNew, +cupsArrayNew2, or +cupsArrayNew3 functions. The +first function creates a new array with the specified callback function +and user data pointer:

+ +
+#include <cups/array.h>
+
+static int compare_func(void *first, void *second, void *user_data);
+
+void *user_data;
+cups_array_t *array = cupsArrayNew(compare_func, user_data);
+
+ +

The comparison function (type +cups_arrayfunc_t) is called +whenever an element is added to the array and can be NULL to +create an unsorted array. The function returns -1 if the first element should +come before the second, 0 if the first and second elements should have the same +ordering, and 1 if the first element should come after the second.

+ +

The "user_data" pointer is passed to your comparison function. Pass +NULL if you do not need to associate the elements in your array +with additional information.

+ +

The cupsArrayNew2 function adds +two more arguments to support hashed lookups, which can potentially provide +instantaneous ("O(1)") lookups in your array:

+ +
+#include <cups/array.h>
+
+#define HASH_SIZE 512 /* Size of hash table */
+
+static int compare_func(void *first, void *second, void *user_data);
+static int hash_func(void *element, void *user_data);
+
+void *user_data;
+cups_array_t *hash_array = cupsArrayNew2(compare_func, user_data, hash_func, HASH_SIZE);
+
+ +

The hash function (type +cups_ahash_func_t) should return a +number from 0 to (hash_size-1) that (hopefully) uniquely identifies the +element and is called whenever you look up an element in the array with +cupsArrayFind. The hash size is +only limited by available memory, but generally should not be larger than +16384 to realize any performance improvement.

+ +

The cupsArrayNew3 function adds +copy and free callbacks to support basic memory management of elements:

+ +
+#include <cups/array.h>
+
+#define HASH_SIZE 512 /* Size of hash table */
+
+static int compare_func(void *first, void *second, void *user_data);
+static void *copy_func(void *element, void *user_data);
+static void free_func(void *element, void *user_data);
+static int hash_func(void *element, void *user_data);
+
+void *user_data;
+cups_array_t *array = cupsArrayNew3(compare_func, user_data, NULL, 0, copy_func, free_func);
+
+cups_array_t *hash_array = cupsArrayNew3(compare_func, user_data, hash_func, HASH_SIZE, copy_func, free_func);
+
+ +

Once you have created the array, you add elements using the +cupsArrayAdd +cupsArrayInsert functions. +The first function adds an element to the array, adding the new element +after any elements that have the same order, while the second inserts the +element before others with the same order. For unsorted arrays, +cupsArrayAdd appends the element to +the end of the array while +cupsArrayInsert inserts the +element at the beginning of the array. For example, the following code +creates a sorted array of character strings:

+ +
+#include <cups/array.h>
+
+/* Use strcmp() to compare strings - it will ignore the user_data pointer */
+cups_array_t *array = cupsArrayNew((cups_array_func_t)strcmp, NULL);
+
+/* Add four strings to the array */
+cupsArrayAdd(array, "One Fish");
+cupsArrayAdd(array, "Two Fish");
+cupsArrayAdd(array, "Red Fish");
+cupsArrayAdd(array, "Blue Fish");
+
+ +

Elements are removed using the +cupsArrayRemove function, for +example:

+ +
+#include <cups/array.h>
+
+/* Use strcmp() to compare strings - it will ignore the user_data pointer */
+cups_array_t *array = cupsArrayNew((cups_array_func_t)strcmp, NULL);
+
+/* Add four strings to the array */
+cupsArrayAdd(array, "One Fish");
+cupsArrayAdd(array, "Two Fish");
+cupsArrayAdd(array, "Red Fish");
+cupsArrayAdd(array, "Blue Fish");
+
+/* Remove "Red Fish" */
+cupsArrayRemove(array, "Red Fish");
+
+ +

Finally, you free the memory used by the array using the +cupsArrayDelete function. All +of the memory for the array and hash table (if any) is freed, however CUPS +does not free the elements unless you provide copy and free functions.

+ +

Finding and Enumerating Elements

+ +

CUPS provides several functions to find and enumerate elements in an +array. Each one sets or updates a "current index" into the array, such that +future lookups will start where the last one left off:

+ +
+
cupsArrayFind
+
Returns the first matching element.
+
cupsArrayFirst
+
Returns the first element in the array.
+
cupsArrayIndex
+
Returns the Nth element in the array, starting at 0.
+
cupsArrayLast
+
Returns the last element in the array.
+
cupsArrayNext
+
Returns the next element in the array.
+
cupsArrayPrev
+
Returns the previous element in the array.
+
+ +

Each of these functions returns NULL when there is no +corresponding element. For example, a simple for loop using the +cupsArrayFirst and +cupsArrayNext functions will +enumerate all of the strings in our previous example:

+ +
+#include <cups/array.h>
+
+/* Use strcmp() to compare strings - it will ignore the user_data pointer */
+cups_array_t *array = cupsArrayNew((cups_array_func_t)strcmp, NULL);
+
+/* Add four strings to the array */
+cupsArrayAdd(array, "One Fish");
+cupsArrayAdd(array, "Two Fish");
+cupsArrayAdd(array, "Red Fish");
+cupsArrayAdd(array, "Blue Fish");
+
+/* Show all of the strings in the array */
+char *s;
+for (s = (char *)cupsArrayFirst(array); s != NULL; s = (char *)cupsArrayNext(array))
+  puts(s);
+
diff --git a/cups/api-cups.header b/cups/api-cups.header new file mode 100644 index 0000000000..8e6d3c0b5d --- /dev/null +++ b/cups/api-cups.header @@ -0,0 +1,40 @@ + + +

CUPS API

+ +
+ + + + + + + + + + + + + + + + +
Headercups/cups.h
Library-lcups
See AlsoProgramming: Introduction to CUPS Programming
+ Programming: Array API
+ Programming: File and Directory APIs
+ Programming: Filter and Backend Programming
+ Programming: HTTP and IPP APIs
+ Programming: PPD API
+ Programming: Raster API
diff --git a/cups/api-cups.shtml b/cups/api-cups.shtml new file mode 100644 index 0000000000..dcc4fe98a1 --- /dev/null +++ b/cups/api-cups.shtml @@ -0,0 +1,443 @@ + + +

Overview

+ +

The CUPS API provides the convenience functions needed to support +applications, filters, printer drivers, and backends that need to interface +with the CUPS scheduler.

+ +

Clients and Servers

+ +

CUPS is based on the Internet Printing Protocol ("IPP"), which allows +clients (applications) to communicate with a server (the scheduler) to get a +list of printers, send print jobs, and so forth. You identify which server +you want to communicate with using a pointer to the opaque structure +http_t. All of the examples in this document use the +CUPS_HTTP_DEFAULT constant, referring to the default connection +to the scheduler. The HTTP and IPP +APIs document provides more information on server connections.

+ +

Printers and Classes

+ +

Printers and classes (collections of printers) are accessed through +the cups_dest_t structure which +includes the name (name), instance (instance - +a way of selecting certain saved options/settings), and the options and +attributes associated with that destination (num_options and +options). Destinations are created using the +cupsGetDests function and freed +using the cupsFreeDests function. +The cupsGetDest function finds a +specific destination for printing:

+ +
+#include <cups/cups.h>
+
+cups_dest_t *dests;
+int num_dests = cupsGetDests(&dests);
+cups_dest_t *dest = cupsGetDest("name", NULL, num_dests, dests);
+
+/* do something with dest */
+
+cupsFreeDests(num_dests, dests);
+
+ +

Passing NULL to +cupsGetDest for the destination name +will return the default destination. Similarly, passing a NULL +instance will return the default instance for that destination.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 1: Printer Attributes
Attribute NameDescription
"auth-info-required"The type of authentication required for printing to this + destination: "none", "username,password", "domain,username,password", + or "negotiate" (Kerberos)
"printer-info"The human-readable description of the destination such as "My + Laser Printer".
"printer-is-accepting-jobs""true" if the destination is accepting new jobs, "false" if + not.
"printer-is-shared""true" if the destination is being shared with other computers, + "false" if not.
"printer-location"The human-readable location of the destination such as "Lab 4".
"printer-make-and-model"The human-readable make and model of the destination such as "HP + LaserJet 4000 Series".
"printer-state""3" if the destination is idle, "4" if the destination is printing + a job, and "5" if the destination is stopped.
"printer-state-change-time"The UNIX time when the destination entered the current state.
"printer-state-reasons"Additional comma-delimited state keywords for the destination + such as "media-tray-empty-error" and "toner-low-warning".
"printer-type"The cups_printer_t + value associated with the destination.
+ +

Options

+ +

Options are stored in arrays of +cups_option_t structures. Each +option has a name (name) and value (value) +associated with it. The cups_dest_t +num_options and options members contain the +default options for a particular destination, along with several informational +attributes about the destination as shown in Table 1. +The cupsGetOption function gets +the value for the named option. For example, the following code lists the +available destinations and their human-readable descriptions:

+ +
+#include <cups/cups.h>
+
+cups_dest_t *dests;
+int num_dests = cupsGetDests(&dests);
+cups_dest_t *dest;
+int i;
+const char *value;
+
+for (i = num_dests, dest = dests; i > 0; i --, dest ++)
+  if (dest->instance == NULL)
+  {
+    value = cupsGetOption("printer-info", dest->num_options, dest->options);
+    printf("%s (%s)\n", dest->name, value ? value : "no description");
+  }
+
+cupsFreeDests(num_dests, dests);
+
+ +

You can create your own option arrays using the +cupsAddOption function, which +adds a single named option to an array:

+ +
+#include <cups/cups.h>
+
+int num_options = 0;
+cups_option_t *options = NULL;
+
+/* The returned num_options value is updated as needed */
+num_options = cupsAddOption("first", "value", num_options, &options);
+
+/* This adds a second option value */
+num_options = cupsAddOption("second", "value", num_options, &options);
+
+/* This replaces the first option we added */
+num_options = cupsAddOption("first", "new value", num_options, &options);
+
+ +

Use a for loop to copy the options from a destination:

+ +
+#include <cups/cups.h>
+
+int i;
+int num_options = 0;
+cups_option_t *options = NULL;
+cups_dest_t *dest;
+
+for (i = 0; i < dest->num_options; i ++)
+  num_options = cupsAddOption(dest->options[i].name, dest->options[i].value,
+                              num_options, &options);
+
+ +

Use the cupsFreeOptions +function to free the options array when you are done using it:

+ +
+cupsFreeOptions(num_options, options);
+
+ +

Print Jobs

+ +

Print jobs are identified by a locally-unique job ID number from 1 to +231-1 and have options and one or more files for printing to a +single destination. The cupsPrintFile +function creates a new job with one file. The following code prints the CUPS +test page file:

+ +
+#include <cups/cups.h>
+
+cups_dest_t *dest;
+int num_options;
+cups_option_t *options;
+int job_id;
+
+/* Print a single file */
+job_id = cupsPrintFile(dest->name, "/usr/share/cups/data/testprint.ps",
+                        "Test Print", num_options, options);
+
+ +

The cupsPrintFiles function +creates a job with multiple files. The files are provided in a +char * array:

+ +
+#include <cups/cups.h>
+
+cups_dest_t *dest;
+int num_options;
+cups_option_t *options;
+int job_id;
+char *files[3] = { "file1.pdf", "file2.pdf", "file3.pdf" };
+
+/* Print three files */
+job_id = cupsPrintFiles(dest->name, 3, files, "Test Print", num_options, options);
+
+ +

Finally, the cupsCreateJob +function creates a new job with no files in it. Files are added using the +cupsStartDocument, +cupsWriteRequestData, +and cupsFinishDocument functions. +The following example creates a job with 10 text files for printing:

+ +
+#include <cups/cups.h>
+
+cups_dest_t *dest;
+int num_options;
+cups_option_t *options;
+int job_id;
+int i;
+char buffer[1024];
+
+/* Create the job */
+job_id = cupsCreateJob(CUPS_HTTP_DEFAULT, dest->name, "10 Text Files",
+                       num_options, options);
+
+/* If the job is created, add 10 files */
+if (job_id > 0)
+{
+  for (i = 1; i <= 10; i ++)
+  {
+    snprintf(buffer, sizeof(buffer), "file%d.txt", i);
+
+    cupsStartDocument(CUPS_HTTP_DEFAULT, dest->name, job_id, buffer,
+                      CUPS_FORMAT_TEXT, i == 10);
+
+    snprintf(buffer, sizeof(buffer),
+             "File %d\n"
+             "\n"
+             "One fish,\n"
+             "Two fish,\n
+             "Red fish,\n
+             "Blue fish\n", i);
+
+    /* cupsWriteRequestData can be called as many times as needed */
+    cupsWriteRequestData(CUPS_HTTP_DEFAULT, buffer, strlen(buffer));
+
+    cupsFinishDocument(CUPS_HTTP_DEFAULT, dest->name);
+  }
+}
+
+ +

Once you have created a job, you can monitor its status using the +cupsGetJobs function, which returns +an array of cups_job_t structures. +Each contains the job ID (id), destination name +(dest), title (title), and other information +associated with the job. The job array is freed using the +cupsFreeJobs function. The following +example monitors a specific job ID, showing the current job state once every +5 seconds until the job is completed:

+ +
+#include <cups/cups.h>
+
+cups_dest_t *dest;
+int job_id;
+int num_jobs;
+cups_job_t *jobs;
+int i;
+ipp_jstate_t job_state = IPP_JOB_PENDING;
+ 
+while (job_state < IPP_JOB_STOPPED)
+{
+  /* Get my jobs (1) with any state (-1) */
+  num_jobs = cupsGetJobs(&jobs, dest->name, 1, -1);
+
+  /* Loop to find my job */
+  job_state = IPP_JOB_COMPLETED;
+
+  for (i = 0; i < num_jobs; i ++)
+    if (jobs[i].id == job_id)
+    {
+      job_state = jobs[i].state;
+      break;
+    }
+
+  /* Free the job array */
+  cupsFreeJobs(num_jobs, jobs);
+
+  /* Show the current state */
+  switch (job_state)
+  {
+    case IPP_JOB_PENDING :
+        printf("Job %d is pending.\n", job_id);
+        break;
+    case IPP_JOB_HELD :
+        printf("Job %d is held.\n", job_id);
+        break;
+    case IPP_JOB_PROCESSING :
+        printf("Job %d is processing.\n", job_id);
+        break;
+    case IPP_JOB_STOPPED :
+        printf("Job %d is stopped.\n", job_id);
+        break;
+    case IPP_JOB_CANCELED :
+        printf("Job %d is canceled.\n", job_id);
+        break;
+    case IPP_JOB_ABORTED :
+        printf("Job %d is aborted.\n", job_id);
+        break;
+    case IPP_JOB_COMPLETED :
+        printf("Job %d is completed.\n", job_id);
+        break;
+  }
+
+  /* Sleep if the job is not finished */
+  if (job_state < IPP_JOB_STOPPED)
+    sleep(5);
+}
+
+ +

To cancel a job, use the +cupsCancelJob function with the +job ID:

+ +
+#include <cups/cups.h>
+
+cups_dest_t *dest;
+int job_id;
+
+cupsCancelJob(dest->name, job_id);
+
+ +

Error Handling

+ +

If any of the CUPS API printing functions returns an error, the reason for +that error can be found by calling the +cupsLastError and +cupsLastErrorString functions. +cupsLastError returns the last IPP +error code +(ipp_status_t) +that was encountered, while +cupsLastErrorString returns +a (localized) human-readable string that can be shown to the user. For example, +if any of the job creation functions returns a job ID of 0, you can use +cupsLastErrorString to show +the reason why the job could not be created:

+ +
+#include <cups/cups.h>
+
+int job_id;
+
+if (job_id == 0)
+  puts(cupsLastErrorString());
+
+ +

Passwords and Authentication

+ +

CUPS supports authentication of any request, including submission of print +jobs. The default mechanism for getting the username and password is to use the +login user and a password from the console.

+ +

To support other types of applications, in particular Graphical User +Interfaces ("GUIs"), the CUPS API provides functions to set the default +username and to register a callback function that returns a password string.

+ +

The cupsSetPasswordCB +function is used to set a password callback in your program. Only one +function can be used at any time.

+ +

The cupsSetUser function sets the +current username for authentication. This function can be called by your +password callback function to change the current username as needed.

+ +

The following example shows a simple password callback that gets a +username and password from the user:

+ +
+#include <cups/cups.h>
+
+const char *
+my_password_cb(const char *prompt)
+{
+  char	user[65];
+
+
+  puts(prompt);
+
+  /* Get a username from the user */
+  printf("Username: ");
+  if (fgets(user, sizeof(user), stdin) == NULL)
+    return (NULL);
+
+  /* Strip the newline from the string and set the user */
+  user[strlen(user) - 1] = '\0';
+
+  cupsSetUser(user);
+
+  /* Use getpass() to ask for the password... */
+  return (getpass("Password: "));
+}
+
+cupsSetPasswordCB(my_password_cb);
+
+ +

Similarly, a GUI could display the prompt string in a window with input +fields for the username and password. The username should default to the +string returned by the cupsUser +function.

diff --git a/cups/api-filedir.header b/cups/api-filedir.header new file mode 100644 index 0000000000..63f629661e --- /dev/null +++ b/cups/api-filedir.header @@ -0,0 +1,36 @@ + + +

File and Directory APIs

+ +
+ + + + + + + + + + + + + + + + +
Headerscups/file.h
+ cups/dir.h
Library-lcups
See AlsoProgramming: Introduction to CUPS Programming
+ Programming: CUPS API
diff --git a/cups/api-filedir.shtml b/cups/api-filedir.shtml new file mode 100644 index 0000000000..4a4a16ffd8 --- /dev/null +++ b/cups/api-filedir.shtml @@ -0,0 +1,31 @@ + + +

Overview

+ +

The CUPS file and directory APIs provide portable interfaces +for manipulating files and listing files and directories. Unlike +stdio FILE streams, the cupsFile functions +allow you to open more than 256 files at any given time. They +also manage the platform-specific details of locking, large file +support, line endings (CR, LF, or CR LF), and reading and writing +files using Flate ("gzip") compression. Finally, you can also +connect, read from, and write to network connections using the +cupsFile functions.

+ +

The cupsDir functions manage the platform-specific +details of directory access/listing and provide a convenient way +to get both a list of files and the information (permissions, +size, timestamp, etc.) for each of those files.

diff --git a/cups/api-filter.header b/cups/api-filter.header new file mode 100644 index 0000000000..54e1d894ae --- /dev/null +++ b/cups/api-filter.header @@ -0,0 +1,41 @@ + + +

Filter and Backend Programming

+ +
+ + + + + + + + + + + + + + + + +
Headerscups/backend.h
+ cups/sidechannel.h
Library-lcups
See AlsoProgramming: Introduction to CUPS Programming
+ Programming: CUPS API
+ Programming: PPD API
+ Programming: Raster API
+ Programming: Developing PostScript Printer Drivers
+ Programming: Developing Raster Printer Drivers
+ Specifications: CUPS Design Description
diff --git a/cups/api-filter.shtml b/cups/api-filter.shtml new file mode 100644 index 0000000000..edc822cdcc --- /dev/null +++ b/cups/api-filter.shtml @@ -0,0 +1,765 @@ + + +

Overview

+ +

Filters (which include printer drivers and port monitors) and backends +are used to convert job files to a printable format and send that data to the +printer itself. All of these programs use a common interface for processing +print jobs and communicating status information to the scheduler. Each is run +with a standard set of command-line arguments:

+ +

+ +
argv[1]
+
The job ID
+ +
argv[2]
+
The user printing the job
+ +
argv[3]
+
The job name/title
+ +
argv[4]
+
The number of copies to print
+ +
argv[5]
+
The options that were provided when the job was submitted
+ +
argv[6]
+
The file to print (first program only)
+
+ +

The scheduler runs one or more of these programs to print any given job. The +first filter reads from the print file and writes to the standard output, while +the remaining filters read from the standard input and write to the standard +output. The backend is the last filter in the chain and writes to the +device.

+ +

Filters are always run as a non-privileged user, typically "lp", with no +connection to the user's desktop. Backends are run either as a non-privileged +user or as root if the file permissions do not allow user or group execution. +The file permissions section talks about this in +more detail.

+ +

Security Considerations

+ +

It is always important to use security programming practices. Filters and +most backends are run as a non-privileged user, so the major security +consideration is resource utilization - filters should not depend on unlimited +amounts of CPU, memory, or disk space, and should protect against conditions +that could lead to excess usage of any resource like infinite loops and +unbounded recursion. In addition, filters must never allow the user to +specify an arbitrary file path to a separator page, template, or other file +used by the filter since that can lead to an unauthorized disclosure of +information. Always treat input as suspect and validate it!

+ +

If you are developing a backend that runs as root, make sure to check for +potential buffer overflows, integer under/overflow conditions, and file +accesses since these can lead to privilege escalations. When writing files, +always validate the file path and never allow a user to determine +where to store a file.

+ +
Note: + +

Never write files to a user's home directory. Aside from the +security implications, CUPS is a network print service and as such the network +user may not be the same as the local user and/or there may not be a local home +directory to write to.

+ +

In addition, some operating systems provide additional security mechanisms +that further limit file system access, even for backends running as root. On +Mac OS X, for example, no backend may write to a user's home directory.

+
+ +

Canceled Jobs and Signal Handling

+ +

The scheduler sends SIGTERM when a printing job is canceled or +held. Filters, backends, and port monitors must catch +SIGTERM and perform any cleanup necessary to produce a valid output +file or return the printer to a known good state. The recommended behavior is to +end the output on the current page, preferably on the current line or object +being printed.

+ +

Filters and backends may also receive SIGPIPE when an upstream or downstream filter/backend exits with a non-zero status. Developers should generally ignore SIGPIPE at the beginning of main() with the following function call:

+ +
+#include <signal.h>>
+
+...
+
+int
+main(int argc, char *argv[])
+{
+  signal(SIGPIPE, SIG_IGN);
+
+  ...
+}
+
+ +

File Permissions

+ +

For security reasons, CUPS will only run filters and backends that are owned +by root and do not have world or group write permissions. The recommended +permissions for filters and backends are 0555 - read and execute but no write. +Backends that must run as root should use permissions of 0500 - read and execute +by root, no access for other users. Write permissions can be enabled for the +root user only.

+ +

To avoid a warning message, the directory containing your filter(s) must also +be owned by root and have world and group write disabled - permissions of 0755 +or 0555 are strongly encouraged.

+ +

Temporary Files

+ +

Temporary files should be created in the directory specified by the +"TMPDIR" environment variable. The +cupsTempFile2 function can be +used to safely create temporary files in this directory.

+ +

Copy Generation

+ +

The argv[4] argument specifies the number of copies to produce +of the input file. In general, you should only generate copies if the +filename argument is supplied. The only exception to this are +filters that produce device-independent PostScript output, since the PostScript +filter pstops is responsible for generating copies of PostScript +files.

+ +

Exit Codes

+ +

Filters must exit with status 0 when they successfully generate print data +or 1 when they encounter an error. Backends can return any of the +cups_backend_t constants.

+ +

Environment Variables

+ +

The following environment variables are defined by the printing system +when running print filters and backends:

+ +
+ +
APPLE_LANGUAGE
+
The Apple language identifier associated with the job + (Mac OS X only).
+ +
CHARSET
+
The job character set, typically "utf-8".
+ +
CLASS
+
When a job is submitted to a printer class, contains the name of + the destination printer class. Otherwise this environment + variable will not be set.
+ +
CONTENT_TYPE
+
The MIME type associated with the file (e.g. + application/postscript).
+ +
CUPS_CACHEDIR
+
The directory where cache files can be stored. Cache files can be + used to retain information between jobs or files in a job.
+ +
CUPS_DATADIR
+
The directory where (read-only) CUPS data files can be found.
+ +
CUPS_FILETYPE
+
The type of file being printed: "job-sheet" for a banner page and + "document" for a regular print file.
+ +
CUPS_SERVERROOT
+
The root directory of the server.
+ +
DEVICE_URI
+
The device-uri associated with the printer.
+ +
FINAL_CONTENT_TYPE
+
The MIME type associated with the printer (e.g. + application/vnd.cups-postscript).
+ +
LANG
+
The language locale associated with the job.
+ +
PPD
+
The full pathname of the PostScript Printer Description (PPD) + file for this printer.
+ +
PRINTER
+
The queue name of the class or printer.
+ +
RIP_CACHE
+
The recommended amount of memory to use for Raster Image + Processors (RIPs).
+ +
TMPDIR
+
The directory where temporary files should be created.
+ +
+ +

Communicating with the Scheduler

+ +

Filters and backends communicate with the scheduler by writing messages +to the standard error file. The scheduler reads messages from all filters in +a job and processes the message based on its prefix. For example, the following +code sets the current printer state message to "Printing page 5":

+ +
+int page = 5;
+
+fprintf(stderr, "INFO: Printing page %d\n", page);
+
+ +

Each message is a single line of text starting with one of the following +prefix strings:

+ +
+ +
ALERT: message
+
Sets the printer-state-message attribute and adds the specified + message to the current error log file using the "alert" log level.
+ +
ATTR: attribute=value [attribute=value]
+
Sets the named printer or job attribute(s). Typically this is used + to set the marker-colors, marker-high-levels, + marker-levels, marker-low-levels, + marker-message, marker-names, + marker-types, printer-alert, and + printer-alert-description printer attributes. Standard + marker-types values are listed in Table + 1.
+ +
CRIT: message
+
Sets the printer-state-message attribute and adds the specified + message to the current error log file using the "critical" log + level.
+ +
DEBUG: message
+
Sets the printer-state-message attribute and adds the specified + message to the current error log file using the "debug" log level.
+ +
DEBUG2: message
+
Sets the printer-state-message attribute and adds the specified + message to the current error log file using the "debug2" log level.
+ +
EMERG: message
+
Sets the printer-state-message attribute and adds the specified + message to the current error log file using the "emergency" log + level.
+ +
ERROR: message
+
Sets the printer-state-message attribute and adds the specified + message to the current error log file using the "error" log level. + Use "ERROR:" messages for non-persistent processing errors.
+ +
INFO: message
+
Sets the printer-state-message attribute. If the current log level + is set to "debug2", also adds the specified message to the current error + log file using the "info" log level.
+ +
NOTICE: message
+
Sets the printer-state-message attribute and adds the specified + message to the current error log file using the "notice" log level.
+ +
PAGE: page-number #-copies
+
PAGE: total #-pages
+
Adds an entry to the current page log file. The first form adds + #-copies to the job-media-sheets-completed attribute. The second + form sets the job-media-sheets-completed attribute to #-pages.
+ +
PPD: keyword=value [keyword=value ...]
+
Changes or adds keywords to the printer's PPD file. Typically + this is used to update installable options or default media settings + based on the printer configuration.
+ +
STATE: + printer-state-reason [printer-state-reason ...]
+
STATE: - printer-state-reason [printer-state-reason ...]
+
Sets or clears printer-state-reason keywords for the current queue. + Typically this is used to indicate persistent media, ink, toner, and + configuration conditions or errors on a printer. + Table 2 lists the standard state keywords - + use vendor-prefixed ("com.example.foo") keywords for custom states. See + Managing Printer State in a Filter for more + information. + +
WARNING: message
+
Sets the printer-state-message attribute and adds the specified + message to the current error log file using the "warning" log + level.
+ +
+ +

Messages without one of these prefixes are treated as if they began with +the "DEBUG:" prefix string.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 1: Standard marker-types Values
marker-typeDescription
developerDeveloper unit
fuserFuser unit
fuserCleaningPadFuser cleaning pad
fuserOilFuser oil
inkInk supply
opcPhoto conductor
solidWaxWax supply
staplesStaple supply
tonerToner supply
transferUnitTransfer unit
wasteInkWaste ink tank
wasteTonerWaste toner tank
wasteWaxWaste wax tank
+ +
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 2: Standard State Keywords
KeywordDescription
connecting-to-deviceConnecting to printer but not printing yet.
cover-openThe printer's cover is open.
input-tray-missingThe paper tray is missing.
marker-supply-emptyThe printer is out of ink.
marker-supply-lowThe printer is almost out of ink.
marker-waste-almost-fullThe printer's waste bin is almost full.
marker-waste-fullThe printer's waste bin is full.
media-emptyThe paper tray (any paper tray) is empty.
media-jamThere is a paper jam.
media-lowThe paper tray (any paper tray) is almost empty.
media-neededThe paper tray needs to be filled (for a job that is printing).
pausedStop the printer.
timed-outUnable to connect to printer.
toner-emptyThe printer is out of toner.
toner-lowThe printer is low on toner.
+ +

Managing Printer State in a Filter

+ +

Filters are responsible for managing the state keywords they set using +"STATE:" messages. Typically you will update all of the keywords that +are used by the filter at startup, for example:

+ +
+if (foo_condition != 0)
+  fputs("STATE: +com.example.foo\n", stderr);
+else
+  fputs("STATE: -com.example.foo\n", stderr);
+
+if (bar_condition != 0)
+  fputs("STATE: +com.example.bar\n", stderr);
+else
+  fputs("STATE: -com.example.bar\n", stderr);
+
+ +

Then as conditions change, your filter sends "STATE: +keyword" or "STATE: +-keyword" messages as necessary to set or clear the corresponding keyword, +respectively.

+ +

State keywords are often used to notify the user of issues that span across +jobs, for example "media-empty-warning" that indicates one or more paper trays +are empty. These keywords should not be cleared unless the corresponding issue +no longer exists.

+ +

Filters should clear job-related keywords on startup and exit so that they +do not remain set between jobs. For example, "connecting-to-device" is a job +sub-state and not an issue that applies when a job is not printing.

+ +
Note: + +

"STATE:" messages often provide visible alerts to the user. For example, +on Mac OS X setting a printer-state-reason value with an "-error" or +"-warning" suffix will cause the printer's dock item to bounce if the +corresponding reason is localized with a cupsIPPReason keyword in the +printer's PPD file.

+ +

When providing a vendor-prefixed keyword, always provide the +corresponding standard keyword (if any) to allow clients to respond to the +condition correctly. For example, if you provide a vendor-prefixed keyword +for a low cyan ink condition ("com.example.cyan-ink-low") you must also set the +"marker-supply-low-warning" keyword. In such cases you should also refrain +from localizing the vendor-prefixed keyword in the PPD file - otherwise both +the generic and vendor-specific keyword will be shown in the user +interface.

+ +
+ +

Reporting Supply Levels

+ +

CUPS tracks several "marker-*" attributes for ink/toner supply level +reporting. These attributes allow applications to display the current supply +levels for a printer without printer-specific software. Table 3 lists the marker attributes and what they represent.

+ +

Filters set marker attributes by sending "ATTR:" messages to stderr. For +example, a filter supporting an inkjet printer with black and tri-color ink +cartridges would use the following to initialize the supply attributes:

+ +
+fputs("ATTR: marker-colors=#000000,#00FFFF#FF00FF#FFFF00\n", stderr);
+fputs("ATTR: marker-low-levels=5,10\n", stderr);
+fputs("ATTR: marker-names=Black,Tri-Color\n", stderr);
+fputs("ATTR: marker-types=ink,ink\n", stderr);
+
+ +

Then periodically the filter queries the printer for its current supply +levels and updates them with a separate "ATTR:" message:

+ +
+int black_level, tri_level;
+...
+fprintf(stderr, "ATTR: marker-levels=%d,%d\n", black_level, tri_level);
+
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 3: Supply Level Attributes
AttributeDescription
marker-colorsA list of comma-separated colors; each color is either "none" or one or + more hex-encoded sRGB colors of the form "#RRGGBB".
marker-high-levelsA list of comma-separated "almost full" level values from 0 to 100; a + value of 100 should be used for supplies that are consumed/emptied like ink + cartridges.
marker-levelsA list of comma-separated level values for each supply. A value of -1 + indicates the level is unavailable, -2 indicates unknown, and -3 indicates + the level is unknown but has not yet reached capacity. Values from 0 to 100 + indicate the corresponding percentage.
marker-low-levelsA list of comma-separated "almost empty" level values from 0 to 100; a + value of 0 should be used for supplies that are filled like waste ink + tanks.
marker-messageA human-readable supply status message for the user like "12 pages of + ink remaining."
marker-namesA list of comma-separated supply names like "Cyan Ink", "Fuser", + etc.
marker-typesA list of comma-separated supply types; the types are listed in + Table 1.
+ +

Communicating with the Backend

+ +

Filters can communicate with the backend via the +cupsBackChannelRead and +cupsSideChannelDoRequest +functions. The +cupsBackChannelRead function +reads data that has been sent back from the device and is typically used to +obtain status and configuration information. For example, the following code +polls the backend for back-channel data:

+ +
+#include <cups/cups.h>
+
+char buffer[8192];
+ssize_t bytes;
+
+/* Use a timeout of 0.0 seconds to poll for back-channel data */
+bytes = cupsBackChannelRead(buffer, sizeof(buffer), 0.0);
+
+ +

Filters can also use select() or poll() on the +back-channel file descriptor (3 or CUPS_BC_FD) to read data only +when it is available.

+ +

The +cupsSideChannelDoRequest +function allows you to get out-of-band status information and do synchronization +with the device. For example, the following code gets the current IEEE-1284 +device ID string from the backend:

+ +
+#include <cups/sidechannel.h>
+
+char data[2049];
+int datalen;
+cups_sc_status_t status;
+
+/* Tell cupsSideChannelDoRequest() how big our buffer is, less 1 byte for
+   nul-termination... */
+datalen = sizeof(data) - 1;
+
+/* Get the IEEE-1284 device ID, waiting for up to 1 second */
+status = cupsSideChannelDoRequest(CUPS_SC_CMD_GET_DEVICE_ID, data, &datalen, 1.0);
+
+/* Use the returned value if OK was returned and the length is non-zero */
+if (status == CUPS_SC_STATUS_OK && datalen > 0)
+  data[datalen] = '\0';
+else
+  data[0] = '\0';
+
+ +

Forcing All Output to a Printer

+ +

The +cupsSideChannelDoRequest +function allows you to tell the backend to send all pending data to the printer. +This is most often needed when sending query commands to the printer. For example:

+ +
+#include <cups/cups.h>
+#include <cups/sidechannel.h>
+
+char data[1024];
+int datalen = sizeof(data);
+cups_sc_status_t status;
+
+/* Flush pending output to stdout */
+fflush(stdout);
+
+/* Drain output to backend, waiting for up to 30 seconds */
+status = cupsSideChannelDoRequest(CUPS_SC_CMD_DRAIN_OUTPUT, data, &datalen, 30.0);
+
+/* Read the response if the output was sent */
+if (status == CUPS_SC_STATUS_OK)
+{
+  ssize_t bytes;
+
+  /* Wait up to 10.0 seconds for back-channel data */
+  bytes = cupsBackChannelRead(data, sizeof(data), 10.0);
+  /* do something with the data from the printer */
+}
+
+ +

Communicating with Filters

+ +

Backends communicate with filters using the reciprocal functions +cupsBackChannelWrite, +cupsSideChannelRead, and +cupsSideChannelWrite. We +recommend writing back-channel data using a timeout of 1.0 seconds:

+ +
+#include <cups/cups.h>
+
+char buffer[8192];
+ssize_t bytes;
+
+/* Obtain data from printer/device */
+...
+
+/* Use a timeout of 1.0 seconds to give filters a chance to read */
+cupsBackChannelWrite(buffer, bytes, 1.0);
+
+ +

The cupsSideChannelRead +function reads a side-channel command from a filter, driver, or port monitor. +Backends can either poll for commands using a timeout of 0.0, wait +indefinitely for commands using a timeout of -1.0 (probably in a +separate thread for that purpose), or use select or +poll on the CUPS_SC_FD file descriptor (4) to handle +input and output on several file descriptors at the same time.

+ +

Once a command is processed, the backend uses the +cupsSideChannelWrite function +to send its response. For example, the following code shows how to poll for a +side-channel command and respond to it:

+ +
+#include <cups/sidechannel.h>
+
+cups_sc_command_t command;
+cups_sc_status_t status;
+char data[2048];
+int datalen = sizeof(data);
+
+/* Poll for a command... */
+if (!cupsSideChannelRead(&command, &status, data, &datalen, 0.0))
+{
+  switch (command)
+  {
+    /* handle supported commands, fill data/datalen/status with values as needed */
+
+    default :
+        status  = CUPS_SC_STATUS_NOT_IMPLEMENTED;
+	datalen = 0;
+	break;
+  }
+
+  /* Send a response... */
+  cupsSideChannelWrite(command, status, data, datalen, 1.0);
+}
+
+ +

Doing SNMP Queries with Network Printers

+ +

The Simple Network Management Protocol (SNMP) allows you to get the current +status, page counter, and supply levels from most network printers. Every +piece of information is associated with an Object Identifier (OID), and +every printer has a community name associated with it. OIDs can be +queried directly or by "walking" over a range of OIDs with a common prefix.

+ +

The two CUPS SNMP functions provide a simple API for querying network +printers through the side-channel interface. Each accepts a string containing +an OID like ".1.3.6.1.2.1.43.10.2.1.4.1.1" (the standard page counter OID) +along with a timeout for the query.

+ +

The cupsSideChannelSNMPGet +function queries a single OID and returns the value as a string in a buffer +you supply:

+ +
+#include <cups/sidechannel.h>
+
+char data[512];
+int datalen = sizeof(data);
+
+if (cupsSideChannelSNMPGet(".1.3.6.1.2.1.43.10.2.1.4.1.1", data, &datalen, 5.0)
+        == CUPS_SC_STATUS_OK)
+{
+  /* Do something with the value */
+  printf("Page counter is: %s\n", data);
+}
+
+ +

The +cupsSideChannelSNMPWalk +function allows you to query a whole group of OIDs, calling a function of your +choice for each OID that is found:

+ +
+#include <cups/sidechannel.h>
+
+void
+my_callback(const char *oid, const char *data, int datalen, void *context)
+{
+  /* Do something with the value */
+  printf("%s=%s\n", oid, data);
+}
+
+...
+
+void *my_data;
+
+cupsSNMPSideChannelWalk(".1.3.6.1.2.1.43", 5.0, my_callback, my_data);
+
diff --git a/cups/api-httpipp.header b/cups/api-httpipp.header new file mode 100644 index 0000000000..42dffb1ea0 --- /dev/null +++ b/cups/api-httpipp.header @@ -0,0 +1,37 @@ + + +

HTTP and IPP APIs

+ +
+ + + + + + + + + + + + + + + + +
Headercups/cups.h
Library-lcups
See AlsoProgramming: Introduction to CUPS Programming
+ Programming: CUPS API
+ References: CUPS Implementation of IPP
diff --git a/cups/api-httpipp.shtml b/cups/api-httpipp.shtml new file mode 100644 index 0000000000..484c00daee --- /dev/null +++ b/cups/api-httpipp.shtml @@ -0,0 +1,323 @@ + + +

Overview

+ +

The CUPS HTTP and IPP APIs provide low-level access to the HTTP and IPP +protocols and CUPS scheduler. They are typically used by monitoring and +administration programs to perform specific functions not supported by the +high-level CUPS API functions.

+ +

The HTTP APIs use an opaque structure called +http_t to manage connections to +a particular HTTP or IPP server. The +httpConnectEncrypt function is +used to create an instance of this structure for a particular server. +The constant CUPS_HTTP_DEFAULT can be used with all of the +cups functions to refer to the default CUPS server - the functions +create a per-thread http_t as needed.

+ +

The IPP APIs use two structures for requests (messages sent to the CUPS +scheduler) and responses (messages sent back to your application from the +scheduler). The ipp_t structure holds a +complete request or response and is allocated using the +ippNew or +ippNewRequest functions and +freed using the ippDelete function.

+ +

The second structure is called +ipp_attribute_t and holds a +single IPP attribute which consists of a group tag (group_tag), a +value type tag (value_tag), the attribute name (name), +and 1 or more values (values[]). Attributes are added to an +ipp_t structure using one of the +ippAdd functions. For example, use +ippAddString to add a +"requesting-user-name" string attribute to a request:

+ +
+ipp_t *request = ippNewRequest(IPP_GET_JOBS);
+
+ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name",
+             NULL, cupsUser());
+
+ +

Once you have created an IPP request, use the cups +functions to send the request to and read the response from the server. +For example, the cupsDoRequest +function can be used for simple query operations that do not involve files:

+ +
+#include <cups/cups.h>
+
+
+ipp_t *get_jobs(void)
+{
+  ipp_t *request = ippNewRequest(IPP_GET_JOBS);
+
+  ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name",
+               NULL, cupsUser());
+
+  return (cupsDoRequest(CUPS_HTTP_DEFAULT, request, "/"));
+}
+
+ +

The cupsDoRequest function frees +the request structure and returns an IPP response structure or NULL pointer if +the request could not be sent to the server. Once you have a response from +the server, you can either use the +ippFindAttribute and +ippFindNextAttribute functions +to find specific attributes, for example:

+ +
+ipp_t *response;
+ipp_attribute_t *attr;
+
+attr = ippFindAttribute(response, "printer-state", IPP_TAG_ENUM);
+
+ +

You can also walk the list of attributes with a simple for loop +like this:

+ +
+ipp_t *response;
+ipp_attribute_t *attr;
+
+for (attr = response->attrs; attr != NULL; attr = attr->next)
+  if (attr->name == NULL)
+    puts("--SEPARATOR--");
+  else
+    puts(attr->name);
+
+ +

The for loop approach is normally used when collecting +attributes for multiple objects (jobs, printers, etc.) in a response. Attributes +with NULL names indicate a separator between the attributes of +each object. For example, the following code will list the jobs returned from +our previous get_jobs example code:

+ +
+ipp_t *response = get_jobs();
+
+if (response != NULL)
+{
+  ipp_attribute_t *attr;
+  int job_id = 0;
+  char *job_name = NULL;
+  char *job_originating_user_name = NULL;
+
+  puts("Job ID  Owner             Title");
+  puts("------  ----------------  ---------------------------------");
+
+  for (attr = response->attrs; attr != NULL; attr = attr->next)
+  {
+   /* Attributes without names are separators between jobs */
+    if (attr->name == NULL)
+    {
+      if (job_id > 0 && job_name != NULL && job_originating_user_name != NULL)
+        printf("%5d  %-16s  %s\n", job_id, job_originating_user_name, job_name);
+
+      job_id = 0;
+      job_name = NULL;
+      job_originating_user_name = NULL;
+      continue;
+    }
+    else if (!strcmp(attr->name, "job-id") && attr->value_tag == IPP_TAG_INTEGER)
+      job_id = attr->values[0].integer;
+    else if (!strcmp(attr->name, "job-name") && attr->value_tag == IPP_TAG_NAME)
+      job_name = attr->values[0].string.text;
+    else if (!strcmp(attr->name, "job-originating-user-name") &&
+             attr->value_tag == IPP_TAG_NAME)
+      job_originating_user_name = attr->values[0].string.text;
+  }
+
+  if (job_id > 0 && job_name != NULL && job_originating_user_name != NULL)
+    printf("%5d  %-16s  %s\n", job_id, job_originating_user_name, job_name);
+}
+
+ +

Creating URI Strings

+ +

To ensure proper encoding, the +httpAssembleURIf function must be +used to format a "printer-uri" string for all printer-based requests:

+ +
+const char *name = "Foo";
+char uri[1024];
+ipp_t *request;
+
+httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, cupsServer(),
+                 ippPort(), "/printers/%s", name);
+ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", NULL, uri);
+
+ +

Sending Requests with Files

+ +

The cupsDoFileRequest and +cupsDoIORequest functions are +used for requests involving files. The +cupsDoFileRequest function +attaches the named file to a request and is typically used when sending a print +file or changing a printer's PPD file:

+ +
+const char *filename = "/usr/share/cups/data/testprint.ps";
+const char *name = "Foo";
+char uri[1024];
+char resource[1024];
+ipp_t *request = ippNewRequest(IPP_PRINT_JOB);
+ipp_t *response;
+
+/* Use httpAssembleURIf for the printer-uri string */
+httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, cupsServer(),
+                 ippPort(), "/printers/%s", name);
+ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", NULL, uri);
+ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name",
+             NULL, cupsUser());
+ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "job-name",
+             NULL, "testprint.ps");
+
+/* Use snprintf for the resource path */
+snprintf(resource, sizeof(resource), "/printers/%s", name);
+
+response = cupsDoFileRequest(CUPS_HTTP_DEFAULT, request, resource, filename);
+
+ +

The cupsDoIORequest function +optionally attaches a file to the request and optionally saves a file in the +response from the server. It is used when using a pipe for the request +attachment or when using a request that returns a file, currently only +CUPS_GET_DOCUMENT and CUPS_GET_PPD. For example, +the following code will download the PPD file for the sample HP LaserJet +printer driver:

+ +
+char tempfile[1024];
+int tempfd;
+ipp_t *request = ippNewRequest(CUPS_GET_PPD);
+ipp_t *response;
+
+ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "ppd-name",
+             NULL, "laserjet.ppd");
+
+tempfd = cupsTempFd(tempfile, sizeof(tempfile));
+
+response = cupsDoIORequest(CUPS_HTTP_DEFAULT, request, "/", -1, tempfd);
+
+ +

The example passes -1 for the input file descriptor to specify +that no file is to be attached to the request. The PPD file attached to the +response is written to the temporary file descriptor we created using the +cupsTempFd function.

+ +

Asynchronous Request Processing

+ +

The cupsSendRequest and +cupsGetResponse support +asynchronous communications with the server. Unlike the other request +functions, the IPP request is not automatically freed, so remember to +free your request with the ippDelete +function.

+ +

File data is attached to the request using the +cupsWriteRequestData +function, while file data returned from the server is read using the +cupsReadResponseData +function. We can rewrite the previous CUPS_GET_PPD example +to use the asynchronous functions quite easily:

+ +
+char tempfile[1024];
+int tempfd;
+ipp_t *request = ippNewRequest(CUPS_GET_PPD);
+ipp_t *response;
+
+ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "ppd-name",
+             NULL, "laserjet.ppd");
+
+tempfd = cupsTempFd(tempfile, sizeof(tempfile));
+
+if (cupsSendRequest(CUPS_HTTP_DEFAULT, request, "/") == HTTP_CONTINUE)
+{
+  response = cupsGetResponse(CUPS_HTTP_DEFAULT, "/");
+
+  if (response != NULL)
+  {
+    ssize_t bytes;
+    char buffer[8192];
+
+    while ((bytes = cupsReadResponseData(CUPS_HTTP_DEFAULT, buffer, sizeof(buffer))) > 0)
+      write(tempfd, buffer, bytes);
+  }
+}
+
+/* Free the request! */
+ippDelete(request);
+
+ +

The cupsSendRequest function +returns the initial HTTP request status, typically either +HTTP_CONTINUE or HTTP_UNAUTHORIZED. The latter status +is returned when the request requires authentication of some sort. The +cupsDoAuthentication function +must be called when your see HTTP_UNAUTHORIZED and the request +re-sent. We can add authentication support to our example code by using a +do ... while loop:

+ +
+char tempfile[1024];
+int tempfd;
+ipp_t *request = ippNewRequest(CUPS_GET_PPD);
+ipp_t *response;
+http_status_t status;
+
+ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "ppd-name",
+             NULL, "laserjet.ppd");
+
+tempfd = cupsTempFd(tempfile, sizeof(tempfile));
+
+/* Loop for authentication */
+do
+{
+  status = cupsSendRequest(CUPS_HTTP_DEFAULT, request, "/");
+
+  if (status == HTTP_UNAUTHORIZED)
+  {
+    /* Try to authenticate, break out of the loop if that fails */
+    if (cupsDoAuthentication(CUPS_HTTP_DEFAULT, "POST", "/"))
+      break;
+  }
+}
+while (status != HTTP_CONTINUE && status != HTTP_UNAUTHORIZED);
+
+if (status == HTTP_CONTINUE)
+{
+  response = cupsGetResponse(CUPS_HTTP_DEFAULT, "/");
+
+  if (response != NULL)
+  {
+    ssize_t bytes;
+    char buffer[8192];
+
+    while ((bytes = cupsReadResponseData(CUPS_HTTP_DEFAULT, buffer, sizeof(buffer))) > 0)
+      write(tempfd, buffer, bytes);
+  }
+}
+
+/* Free the request! */
+ippDelete(request);
+
diff --git a/cups/api-overview.header b/cups/api-overview.header new file mode 100644 index 0000000000..ecb14b601e --- /dev/null +++ b/cups/api-overview.header @@ -0,0 +1,53 @@ + + +

Introduction to CUPS Programming

+ +
+ + + + + + + + + + + + + + + + +
Headerscups/cups.h
+ cups/array.h
+ cups/backend.h
+ cups/dir.h
+ cups/file.h
+ cups/ppd.h
+ cups/raster.h
+ cups/sidechannel.h
Libraries-lcups
+ -lcupsimage
See AlsoProgramming: Developing Raster Printer Drivers
+ Programming: Developing PostScript Printer Drivers
+ Programming: Filter and Backend Programming
+ Programming: Introduction to the PPD Compiler
+ Programming: Array API
+ Programming: CUPS API
+ Programming: File and Directory APIs
+ Programming: HTTP and IPP APIs
+ Programming: PPD API
+ Programming: Raster API
+ References: PPD Compiler Driver Information File Reference
+ Specifications: CUPS PPD Extensions
diff --git a/cups/api-overview.shtml b/cups/api-overview.shtml new file mode 100644 index 0000000000..3ece1033ca --- /dev/null +++ b/cups/api-overview.shtml @@ -0,0 +1,94 @@ + + +

Overview

+ +

CUPS provides two libraries that interface with the different parts of the +printing system. The "cups" library provides all of the common application and +filter functions while the "cupsimage" library provides all of the imaging +functions used in raster printer drivers. The "cups" library functions are +accessed by including the <cups/cups.h> header, while +"cupsimage" functions are found in the <cups/raster.h> +header.

+ +

Compiling Programs

+ +

The CUPS libraries can be used from any C, C++, or Objective C program. +The method of compiling against the libraries varies depending on the +operating system and installation of CUPS. The following sections show how +to compile a simple program (shown below) in two common environments.

+ +

The following simple program lists the available printers on the system:

+ +
+#include <stdio.h>
+#include <cups/cups.h>
+
+int main(void)
+{
+  int i;
+  cups_dest_t *dests, *dest;
+  int num_dests = cupsGetDests(&dests);
+
+  for (i = num_dests, dest = dests; i > 0; i --, dest ++)
+  {
+    if (dest->instance)
+      printf("%s/%s\n", dest->name, dest->instance);
+    else
+      puts(dest->name);
+  }
+
+  return (0);
+}
+
+ +

Compiling with Xcode

+ +

In Xcode, choose New Project... from the File menu, +then select the Standard Tool project type under Command Line +Utility. Click Next and choose a project directory. Click +Next to create the project.

+ +

In the project window, double-click on the Targets group and +control-click on the simple target to show the context menu. Choose +Existing Framework... from the Add submenu. When the file +chooser sheet appears, press the / key and enter "/usr/lib". Scroll +down the file list and select the libcups.dylib file. Click the +Add button in the file chooser and attributes sheets.

+ +

In the project window, double-click on the main.c source file. +Replace the template source code with the listing above and save it. Click the +Build and Go button to build the sample program and run it.

+ +

Compiling with GCC

+ +

From the command-line, create a file called sample.c using your +favorite editor and then run the following command to compile it with GCC and +run it:

+ +
+gcc -o simple `cups-config --cflags` simple.c `cups-config --libs`
+./simple
+
+ +

The cups-config command provides the compiler flags +("cups-config --cflags") and libraries ("cups-config --libs") needed for the +local system.

+ +

Where to Go Next

+ +

If you are developing a print filter, driver, or backend, see the +Filter and Backend Programming +guide. Raster printer driver developers should also read the +Raster API reference.

diff --git a/cups/api-ppd.header b/cups/api-ppd.header new file mode 100644 index 0000000000..c0d264fe46 --- /dev/null +++ b/cups/api-ppd.header @@ -0,0 +1,38 @@ + + +

PPD API (DEPRECATED)

+ +
The PPD API is deprecated starting in CUPS 1.6. Please use the new Job Ticket APIs in the CUPS API documentation. These functions will be removed in a future release of CUPS.
+ +
+ + + + + + + + + + + + + + + + +
Headercups/ppd.h
Library-lcups
See AlsoProgramming: Introduction to CUPS Programming
+ Programming: CUPS API
+ Specifications: CUPS PPD Extensions
diff --git a/cups/api-ppd.shtml b/cups/api-ppd.shtml new file mode 100644 index 0000000000..c8d3c640f6 --- /dev/null +++ b/cups/api-ppd.shtml @@ -0,0 +1,219 @@ + + +

Overview

+ +
The PPD API is deprecated starting in CUPS 1.6. Please use the new Job Ticket APIs in the CUPS API documentation. These functions will be removed in a future release of CUPS.
+ +

The CUPS PPD API provides read-only access the data in PostScript Printer +Description ("PPD") files which are used for all printers with a driver. With +it you can obtain the data necessary to display printer options to users, mark +option choices and check for conflicting choices, and output marked choices in +PostScript output. The ppd_file_t +structure contains all of the information in a PPD file.

+ +
Note: + +

The CUPS PPD API uses the terms "option" and "choice" instead of the Adobe +terms "MainKeyword" and "OptionKeyword" to refer to specific printer options and +features. CUPS also treats option ("MainKeyword") and choice ("OptionKeyword") +values as case-insensitive strings, so option "InputSlot" and choice "Upper" +are equivalent to "inputslot" and "upper", respectively.

+
+ +

Loading a PPD File

+ +

The ppdOpenFile function "opens" a +PPD file and loads it into memory. For example, the following code opens the +current printer's PPD file in a CUPS filter:

+ +
+#include <cups/ppd.h>
+
+ppd_file_t *ppd = ppdOpenFile(getenv("PPD"));
+
+ +

The return value is a pointer to a new +ppd_file_t structure or NULL +if the PPD file does not exist or cannot be loaded. The +ppdClose function frees the memory used +by the structure:

+ +
+#include <cups/ppd.h>
+
+ppd_file_t *ppd;
+
+ppdClose(ppd);
+
+ +

Once closed, pointers to the ppd_file_t +structure and any data in it will no longer be valid.

+ +

Options and Groups

+ +

PPD files support multiple options, which are stored in arrays of +ppd_option_t and +ppd_choice_t structures.

+ +

Each option in turn is associated with a group stored in a +ppd_group_t structure. Groups can be +specified in the PPD file; if an option is not associated with a group +then it is put in an automatically-generated "General" group. Groups can also +have sub-groups, however CUPS currently ignores sub-groups because of past +abuses of this functionality.

+ +

Option choices are selected by marking them using one of three functions. The +first is ppdMarkDefaults which +selects all of the default options in the PPD file:

+ +
+#include <cups/ppd.h>
+
+ppd_file_t *ppd;
+
+ppdMarkDefaults(ppd);
+
+ +

The second is ppdMarkOption +which selects a single option choice in the PPD file. For example, the following +code selects the upper paper tray:

+ +
+#include <cups/ppd.h>
+
+ppd_file_t *ppd;
+
+ppdMarkOption(ppd, "InputSlot", "Upper");
+
+ +

The last function is +cupsMarkOptions which selects +multiple option choices in the PPD file from an array of CUPS options, mapping +IPP attributes like "media" and "sides" to their corresponding PPD options. You +typically use this function in a print filter with +cupsParseOptions and +ppdMarkDefaults to select all of +the option choices needed for the job, for example:

+ +
+#include <cups/ppd.h>
+
+ppd_file_t *ppd = ppdOpenFile(getenv("PPD"));
+cups_option_t *options = NULL;
+int num_options = cupsParseOptions(argv[5], 0, &options);
+
+ppdMarkDefaults(ppd);
+cupsMarkOptions(ppd, num_options, options);
+cupsFreeOptions(num_options, options);
+
+ +

Constraints

+ +

PPD files support specification of conflict conditions, called +constraints, between different options. Constraints are stored in an array of +ppd_const_t structures which specify +the options and choices that conflict with each other. The +ppdConflicts function tells you +how many of the selected options are incompatible. Since constraints are +normally specified in pairs, the returned value is typically an even number.

+ +

Page Sizes

+ +

Page sizes are special options which have physical dimensions and margins +associated with them. The size information is stored in +ppd_size_t structures and is available +by looking up the named size with the +ppdPageSize function. The page size and +margins are returned in units called points; there are 72 points per inch. If +you pass NULL for the size, the currently selected size is +returned:

+ +
+#include <cups/ppd.h>
+
+ppd_file_t *ppd;
+ppd_size_t *size = ppdPageSize(ppd, NULL);
+
+ +

Besides the standard page sizes listed in a PPD file, some printers +support variable or custom page sizes. Custom page sizes are supported if the +variables_sizes member of the +ppd_file_t structure is non-zero. +The custom_min, custom_max, and +custom_margins members of the +ppd_file_t structure define the limits +of the printable area. To get the resulting media size, use a page size string +of the form "Custom.widthxlength", where "width" and "length" are +in points. Custom page size names can also be specified in inches +("Custom.widthxheightin"), centimeters +("Custom.widthxheightcm"), or millimeters +("Custom.widthxheightmm"):

+ +
+#include <cups/ppd.h>
+
+ppd_file_t *ppd;
+
+/* Get an 576x720 point custom page size */
+ppd_size_t *size = ppdPageSize(ppd, "Custom.576x720");
+
+/* Get an 8x10 inch custom page size */
+ppd_size_t *size = ppdPageSize(ppd, "Custom.8x10in");
+
+/* Get a 100x200 millimeter custom page size */
+ppd_size_t *size = ppdPageSize(ppd, "Custom.100x200mm");
+
+/* Get a 12.7x34.5 centimeter custom page size */
+ppd_size_t *size = ppdPageSize(ppd, "Custom.12.7x34.5cm");
+
+ +

If the PPD does not support variable page sizes, the +ppdPageSize function will return +NULL.

+ +

Attributes

+ +

Every PPD file is composed of one or more attributes. Most of these +attributes are used to define groups, options, choices, and page sizes, +however several informational attributes may be present which you can access +in your program or filter. Attributes normally look like one of the following +examples in a PPD file:

+ +
+*name: "value"
+*name spec: "value"
+*name spec/text: "value"
+
+ +

The ppdFindAttr and +ppdFindNextAttr functions find the +first and next instances, respectively, of the named attribute with the given +"spec" string and return a ppd_attr_t +structure. If you provide a NULL specifier string, all attributes with the +given name will be returned. For example, the following code lists all of the +Product attributes in a PPD file:

+ +
+#include <cups/ppd.h>
+
+ppd_file_t *ppd;
+ppd_attr_t *attr;
+
+for (attr = ppdFindAttr(ppd, "Product", NULL);
+     attr != NULL;
+     attr = ppdFindNextAttr(ppd, "Product", NULL))
+  puts(attr->value);
+
diff --git a/cups/array-private.h b/cups/array-private.h new file mode 100644 index 0000000000..c0d9a92e87 --- /dev/null +++ b/cups/array-private.h @@ -0,0 +1,51 @@ +/* + * "$Id$" + * + * Private array definitions for CUPS. + * + * Copyright 2011 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_ARRAY_PRIVATE_H_ +# define _CUPS_ARRAY_PRIVATE_H_ + +/* + * Include necessary headers... + */ + +# include + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Functions... + */ + +extern int _cupsArrayAddStrings(cups_array_t *a, const char *s) + _CUPS_API_1_5; +extern cups_array_t *_cupsArrayNewStrings(const char *s) _CUPS_API_1_5; + +# ifdef __cplusplus +} +# endif /* __cplusplus */ +#endif /* !_CUPS_ARRAY_PRIVATE_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/array.c b/cups/array.c new file mode 100644 index 0000000000..71c161c7d6 --- /dev/null +++ b/cups/array.c @@ -0,0 +1,1326 @@ +/* + * "$Id$" + * + * Sorted array routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsArrayAdd() - Add an element to the array. + * _cupsArrayAddStrings() - Add zero or more comma-delimited strings to an + * array. + * cupsArrayClear() - Clear the array. + * cupsArrayCount() - Get the number of elements in the array. + * cupsArrayCurrent() - Return the current element in the array. + * cupsArrayDelete() - Free all memory used by the array. + * cupsArrayDup() - Duplicate the array. + * cupsArrayFind() - Find an element in the array. + * cupsArrayFirst() - Get the first element in the array. + * cupsArrayGetIndex() - Get the index of the current element. + * cupsArrayGetInsert() - Get the index of the last inserted element. + * cupsArrayIndex() - Get the N-th element in the array. + * cupsArrayInsert() - Insert an element in the array. + * cupsArrayLast() - Get the last element in the array. + * cupsArrayNew() - Create a new array. + * cupsArrayNew2() - Create a new array with hash. + * cupsArrayNew3() - Create a new array with hash and/or free function. + * _cupsArrayNewStrings() - Create a new array of comma-delimited strings. + * cupsArrayNext() - Get the next element in the array. + * cupsArrayPrev() - Get the previous element in the array. + * cupsArrayRemove() - Remove an element from the array. + * cupsArrayRestore() - Reset the current element to the last @link + * cupsArraySave@. + * cupsArraySave() - Mark the current element for a later @link + * cupsArrayRestore@. + * cupsArrayUserData() - Return the user data for an array. + * cups_array_add() - Insert or append an element to the array. + * cups_array_find() - Find an element in the array. + */ + +/* + * Include necessary headers... + */ + +#include "string-private.h" +#include "debug-private.h" +#include "array-private.h" + + +/* + * Limits... + */ + +#define _CUPS_MAXSAVE 32 /**** Maximum number of saves ****/ + + +/* + * Types and structures... + */ + +struct _cups_array_s /**** CUPS array structure ****/ +{ + /* + * The current implementation uses an insertion sort into an array of + * sorted pointers. We leave the array type private/opaque so that we + * can change the underlying implementation without affecting the users + * of this API. + */ + + int num_elements, /* Number of array elements */ + alloc_elements, /* Allocated array elements */ + current, /* Current element */ + insert, /* Last inserted element */ + unique, /* Are all elements unique? */ + num_saved, /* Number of saved elements */ + saved[_CUPS_MAXSAVE]; + /* Saved elements */ + void **elements; /* Array elements */ + cups_array_func_t compare; /* Element comparison function */ + void *data; /* User data passed to compare */ + cups_ahash_func_t hashfunc; /* Hash function */ + int hashsize, /* Size of hash */ + *hash; /* Hash array */ + cups_acopy_func_t copyfunc; /* Copy function */ + cups_afree_func_t freefunc; /* Free function */ +}; + + +/* + * Local functions... + */ + +static int cups_array_add(cups_array_t *a, void *e, int insert); +static int cups_array_find(cups_array_t *a, void *e, int prev, int *rdiff); + + +/* + * 'cupsArrayAdd()' - Add an element to the array. + * + * When adding an element to a sorted array, non-unique elements are + * appended at the end of the run of identical elements. For unsorted arrays, + * the element is appended to the end of the array. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 1 on success, 0 on failure */ +cupsArrayAdd(cups_array_t *a, /* I - Array */ + void *e) /* I - Element */ +{ + DEBUG_printf(("2cupsArrayAdd(a=%p, e=%p)", a, e)); + + /* + * Range check input... + */ + + if (!a || !e) + { + DEBUG_puts("3cupsArrayAdd: returning 0"); + return (0); + } + + /* + * Append the element... + */ + + return (cups_array_add(a, e, 0)); +} + + +/* + * '_cupsArrayAddStrings()' - Add zero or more comma-delimited strings to an + * array. + * + * Note: The array MUST be created using the @link _cupsArrayNewStrings@ + * function. Duplicate strings are NOT added. If the string pointer "s" is NULL + * or the empty string, no strings are added to the array. + */ + +int /* O - 1 on success, 0 on failure */ +_cupsArrayAddStrings(cups_array_t *a, /* I - Array */ + const char *s) /* I - Comma-delimited strings or NULL */ +{ + char *buffer, /* Copy of string */ + *start, /* Start of string */ + *end; /* End of string */ + int status = 1; /* Status of add */ + + + if (!a || !s || !*s) + return (0); + + if (!strchr(s, ',')) + { + /* + * String doesn't contain a comma, so add it as a single value... + */ + + if (!cupsArrayFind(a, (void *)s)) + status = cupsArrayAdd(a, (void *)s); + } + else if ((buffer = strdup(s)) == NULL) + status = 0; + else + { + for (start = end = buffer; *end; start = end) + { + /* + * Find the end of the current delimited string and see if we need to add + * it... + */ + + if ((end = strchr(start, ',')) != NULL) + *end++ = '\0'; + else + end = start + strlen(start); + + if (!cupsArrayFind(a, start)) + status &= cupsArrayAdd(a, start); + } + + free(buffer); + } + + return (status); +} + + +/* + * 'cupsArrayClear()' - Clear the array. + * + * This function is equivalent to removing all elements in the array. + * The caller is responsible for freeing the memory used by the + * elements themselves. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void +cupsArrayClear(cups_array_t *a) /* I - Array */ +{ + /* + * Range check input... + */ + + if (!a) + return; + + /* + * Free the existing elements as needed.. + */ + + if (a->freefunc) + { + int i; /* Looping var */ + void **e; /* Current element */ + + for (i = a->num_elements, e = a->elements; i > 0; i --, e ++) + (a->freefunc)(*e, a->data); + } + + /* + * Set the number of elements to 0; we don't actually free the memory + * here - that is done in cupsArrayDelete()... + */ + + a->num_elements = 0; + a->current = -1; + a->insert = -1; + a->unique = 1; + a->num_saved = 0; +} + + +/* + * 'cupsArrayCount()' - Get the number of elements in the array. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - Number of elements */ +cupsArrayCount(cups_array_t *a) /* I - Array */ +{ + /* + * Range check input... + */ + + if (!a) + return (0); + + /* + * Return the number of elements... + */ + + return (a->num_elements); +} + + +/* + * 'cupsArrayCurrent()' - Return the current element in the array. + * + * The current element is undefined until you call @link cupsArrayFind@, + * @link cupsArrayFirst@, or @link cupsArrayIndex@, or @link cupsArrayLast@. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void * /* O - Element */ +cupsArrayCurrent(cups_array_t *a) /* I - Array */ +{ + /* + * Range check input... + */ + + if (!a) + return (NULL); + + /* + * Return the current element... + */ + + if (a->current >= 0 && a->current < a->num_elements) + return (a->elements[a->current]); + else + return (NULL); +} + + +/* + * 'cupsArrayDelete()' - Free all memory used by the array. + * + * The caller is responsible for freeing the memory used by the + * elements themselves. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void +cupsArrayDelete(cups_array_t *a) /* I - Array */ +{ + /* + * Range check input... + */ + + if (!a) + return; + + /* + * Free the elements if we have a free function (otherwise the caller is + * responsible for doing the dirty work...) + */ + + if (a->freefunc) + { + int i; /* Looping var */ + void **e; /* Current element */ + + for (i = a->num_elements, e = a->elements; i > 0; i --, e ++) + (a->freefunc)(*e, a->data); + } + + /* + * Free the array of element pointers... + */ + + if (a->alloc_elements) + free(a->elements); + + if (a->hashsize) + free(a->hash); + + free(a); +} + + +/* + * 'cupsArrayDup()' - Duplicate the array. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +cups_array_t * /* O - Duplicate array */ +cupsArrayDup(cups_array_t *a) /* I - Array */ +{ + cups_array_t *da; /* Duplicate array */ + + + /* + * Range check input... + */ + + if (!a) + return (NULL); + + /* + * Allocate memory for the array... + */ + + da = calloc(1, sizeof(cups_array_t)); + if (!da) + return (NULL); + + da->compare = a->compare; + da->data = a->data; + da->current = a->current; + da->insert = a->insert; + da->unique = a->unique; + da->num_saved = a->num_saved; + + memcpy(da->saved, a->saved, sizeof(a->saved)); + + if (a->num_elements) + { + /* + * Allocate memory for the elements... + */ + + da->elements = malloc(a->num_elements * sizeof(void *)); + if (!da->elements) + { + free(da); + return (NULL); + } + + /* + * Copy the element pointers... + */ + + if (a->copyfunc) + { + /* + * Use the copy function to make a copy of each element... + */ + + int i; /* Looping var */ + + for (i = 0; i < a->num_elements; i ++) + da->elements[i] = (a->copyfunc)(a->elements[i], a->data); + } + else + { + /* + * Just copy raw pointers... + */ + + memcpy(da->elements, a->elements, a->num_elements * sizeof(void *)); + } + + da->num_elements = a->num_elements; + da->alloc_elements = a->num_elements; + } + + /* + * Return the new array... + */ + + return (da); +} + + +/* + * 'cupsArrayFind()' - Find an element in the array. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void * /* O - Element found or @code NULL@ */ +cupsArrayFind(cups_array_t *a, /* I - Array */ + void *e) /* I - Element */ +{ + int current, /* Current element */ + diff, /* Difference */ + hash; /* Hash index */ + + + /* + * Range check input... + */ + + if (!a || !e) + return (NULL); + + /* + * See if we have any elements... + */ + + if (!a->num_elements) + return (NULL); + + /* + * Yes, look for a match... + */ + + if (a->hash) + { + hash = (*(a->hashfunc))(e, a->data); + + if (hash < 0 || hash >= a->hashsize) + { + current = a->current; + hash = -1; + } + else + { + current = a->hash[hash]; + + if (current < 0 || current >= a->num_elements) + current = a->current; + } + } + else + { + current = a->current; + hash = -1; + } + + current = cups_array_find(a, e, current, &diff); + if (!diff) + { + /* + * Found a match! If the array does not contain unique values, find + * the first element that is the same... + */ + + if (!a->unique && a->compare) + { + /* + * The array is not unique, find the first match... + */ + + while (current > 0 && !(*(a->compare))(e, a->elements[current - 1], + a->data)) + current --; + } + + a->current = current; + + if (hash >= 0) + a->hash[hash] = current; + + return (a->elements[current]); + } + else + { + /* + * No match... + */ + + a->current = -1; + + return (NULL); + } +} + + +/* + * 'cupsArrayFirst()' - Get the first element in the array. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void * /* O - First element or @code NULL@ if the array is empty */ +cupsArrayFirst(cups_array_t *a) /* I - Array */ +{ + /* + * Range check input... + */ + + if (!a) + return (NULL); + + /* + * Return the first element... + */ + + a->current = 0; + + return (cupsArrayCurrent(a)); +} + + +/* + * 'cupsArrayGetIndex()' - Get the index of the current element. + * + * The current element is undefined until you call @link cupsArrayFind@, + * @link cupsArrayFirst@, or @link cupsArrayIndex@, or @link cupsArrayLast@. + * + * @since CUPS 1.3/Mac OS X 10.5@ + */ + +int /* O - Index of the current element, starting at 0 */ +cupsArrayGetIndex(cups_array_t *a) /* I - Array */ +{ + if (!a) + return (-1); + else + return (a->current); +} + + +/* + * 'cupsArrayGetInsert()' - Get the index of the last inserted element. + * + * @since CUPS 1.3/Mac OS X 10.5@ + */ + +int /* O - Index of the last inserted element, starting at 0 */ +cupsArrayGetInsert(cups_array_t *a) /* I - Array */ +{ + if (!a) + return (-1); + else + return (a->insert); +} + + +/* + * 'cupsArrayIndex()' - Get the N-th element in the array. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void * /* O - N-th element or @code NULL@ */ +cupsArrayIndex(cups_array_t *a, /* I - Array */ + int n) /* I - Index into array, starting at 0 */ +{ + if (!a) + return (NULL); + + a->current = n; + + return (cupsArrayCurrent(a)); +} + + +/* + * 'cupsArrayInsert()' - Insert an element in the array. + * + * When inserting an element in a sorted array, non-unique elements are + * inserted at the beginning of the run of identical elements. For unsorted + * arrays, the element is inserted at the beginning of the array. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 0 on failure, 1 on success */ +cupsArrayInsert(cups_array_t *a, /* I - Array */ + void *e) /* I - Element */ +{ + DEBUG_printf(("2cupsArrayInsert(a=%p, e=%p)", a, e)); + + /* + * Range check input... + */ + + if (!a || !e) + { + DEBUG_puts("3cupsArrayInsert: returning 0"); + return (0); + } + + /* + * Insert the element... + */ + + return (cups_array_add(a, e, 1)); +} + + +/* + * 'cupsArrayLast()' - Get the last element in the array. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void * /* O - Last element or @code NULL@ if the array is empty */ +cupsArrayLast(cups_array_t *a) /* I - Array */ +{ + /* + * Range check input... + */ + + if (!a) + return (NULL); + + /* + * Return the last element... + */ + + a->current = a->num_elements - 1; + + return (cupsArrayCurrent(a)); +} + + +/* + * 'cupsArrayNew()' - Create a new array. + * + * The comparison function ("f") is used to create a sorted array. The function + * receives pointers to two elements and the user data pointer ("d") - the user + * data pointer argument can safely be omitted when not required so functions + * like @code strcmp@ can be used for sorted string arrays. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +cups_array_t * /* O - Array */ +cupsArrayNew(cups_array_func_t f, /* I - Comparison function or @code NULL@ for an unsorted array */ + void *d) /* I - User data pointer or @code NULL@ */ +{ + return (cupsArrayNew3(f, d, 0, 0, 0, 0)); +} + + +/* + * 'cupsArrayNew2()' - Create a new array with hash. + * + * The comparison function ("f") is used to create a sorted array. The function + * receives pointers to two elements and the user data pointer ("d") - the user + * data pointer argument can safely be omitted when not required so functions + * like @code strcmp@ can be used for sorted string arrays. + * + * The hash function ("h") is used to implement cached lookups with the + * specified hash size ("hsize"). + * + * @since CUPS 1.3/Mac OS X 10.5@ + */ + +cups_array_t * /* O - Array */ +cupsArrayNew2(cups_array_func_t f, /* I - Comparison function or @code NULL@ for an unsorted array */ + void *d, /* I - User data or @code NULL@ */ + cups_ahash_func_t h, /* I - Hash function or @code NULL@ for unhashed lookups */ + int hsize) /* I - Hash size (>= 0) */ +{ + return (cupsArrayNew3(f, d, h, hsize, 0, 0)); +} + + +/* + * 'cupsArrayNew3()' - Create a new array with hash and/or free function. + * + * The comparison function ("f") is used to create a sorted array. The function + * receives pointers to two elements and the user data pointer ("d") - the user + * data pointer argument can safely be omitted when not required so functions + * like @code strcmp@ can be used for sorted string arrays. + * + * The hash function ("h") is used to implement cached lookups with the + * specified hash size ("hsize"). + * + * The copy function ("cf") is used to automatically copy/retain elements when + * added or the array is copied. + * + * The free function ("cf") is used to automatically free/release elements when + * removed or the array is deleted. + * + * @since CUPS 1.5/Mac OS X 10.7@ + */ + +cups_array_t * /* O - Array */ +cupsArrayNew3(cups_array_func_t f, /* I - Comparison function or @code NULL@ for an unsorted array */ + void *d, /* I - User data or @code NULL@ */ + cups_ahash_func_t h, /* I - Hash function or @code NULL@ for unhashed lookups */ + int hsize, /* I - Hash size (>= 0) */ + cups_acopy_func_t cf, /* I - Copy function */ + cups_afree_func_t ff) /* I - Free function */ +{ + cups_array_t *a; /* Array */ + + + /* + * Allocate memory for the array... + */ + + a = calloc(1, sizeof(cups_array_t)); + if (!a) + return (NULL); + + a->compare = f; + a->data = d; + a->current = -1; + a->insert = -1; + a->num_saved = 0; + a->unique = 1; + + if (hsize > 0 && h) + { + a->hashfunc = h; + a->hashsize = hsize; + a->hash = malloc(hsize * sizeof(int)); + + if (!a->hash) + { + free(a); + return (NULL); + } + + memset(a->hash, -1, hsize * sizeof(int)); + } + + a->copyfunc = cf; + a->freefunc = ff; + + return (a); +} + + +/* + * '_cupsArrayNewStrings()' - Create a new array of comma-delimited strings. + * + * Note: The array automatically manages copies of the strings passed. If the + * string pointer "s" is NULL or the empty string, no strings are added to the + * newly created array. + */ + +cups_array_t * /* O - Array */ +_cupsArrayNewStrings(const char *s) /* I - Comma-delimited strings or NULL */ +{ + cups_array_t *a; /* Array */ + + + if ((a = cupsArrayNew3((cups_array_func_t)strcmp, NULL, NULL, 0, + (cups_acopy_func_t)_cupsStrAlloc, + (cups_afree_func_t)_cupsStrFree)) != NULL) + _cupsArrayAddStrings(a, s); + + return (a); +} + + +/* + * 'cupsArrayNext()' - Get the next element in the array. + * + * This function is equivalent to "cupsArrayIndex(a, cupsArrayGetIndex(a) + 1)". + * + * The next element is undefined until you call @link cupsArrayFind@, + * @link cupsArrayFirst@, or @link cupsArrayIndex@, or @link cupsArrayLast@ + * to set the current element. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void * /* O - Next element or @code NULL@ */ +cupsArrayNext(cups_array_t *a) /* I - Array */ +{ + /* + * Range check input... + */ + + if (!a) + return (NULL); + + /* + * Return the next element... + */ + + if (a->current < a->num_elements) + a->current ++; + + return (cupsArrayCurrent(a)); +} + + +/* + * 'cupsArrayPrev()' - Get the previous element in the array. + * + * This function is equivalent to "cupsArrayIndex(a, cupsArrayGetIndex(a) - 1)". + * + * The previous element is undefined until you call @link cupsArrayFind@, + * @link cupsArrayFirst@, or @link cupsArrayIndex@, or @link cupsArrayLast@ + * to set the current element. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void * /* O - Previous element or @code NULL@ */ +cupsArrayPrev(cups_array_t *a) /* I - Array */ +{ + /* + * Range check input... + */ + + if (!a) + return (NULL); + + /* + * Return the previous element... + */ + + if (a->current >= 0) + a->current --; + + return (cupsArrayCurrent(a)); +} + + +/* + * 'cupsArrayRemove()' - Remove an element from the array. + * + * If more than one element matches "e", only the first matching element is + * removed. + * + * The caller is responsible for freeing the memory used by the + * removed element. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 1 on success, 0 on failure */ +cupsArrayRemove(cups_array_t *a, /* I - Array */ + void *e) /* I - Element */ +{ + int i, /* Looping var */ + current, /* Current element */ + diff; /* Difference */ + + + /* + * Range check input... + */ + + if (!a || !e) + return (0); + + /* + * See if the element is in the array... + */ + + if (!a->num_elements) + return (0); + + current = cups_array_find(a, e, a->current, &diff); + if (diff) + return (0); + + /* + * Yes, now remove it... + */ + + a->num_elements --; + + if (a->freefunc) + (a->freefunc)(a->elements[current], a->data); + + if (current < a->num_elements) + memmove(a->elements + current, a->elements + current + 1, + (a->num_elements - current) * sizeof(void *)); + + if (current <= a->current) + a->current --; + + if (current < a->insert) + a->insert --; + else if (current == a->insert) + a->insert = -1; + + for (i = 0; i < a->num_saved; i ++) + if (current <= a->saved[i]) + a->saved[i] --; + + if (a->num_elements <= 1) + a->unique = 1; + + return (1); +} + + +/* + * 'cupsArrayRestore()' - Reset the current element to the last @link cupsArraySave@. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void * /* O - New current element */ +cupsArrayRestore(cups_array_t *a) /* I - Array */ +{ + if (!a) + return (NULL); + + if (a->num_saved <= 0) + return (NULL); + + a->num_saved --; + a->current = a->saved[a->num_saved]; + + if (a->current >= 0 && a->current < a->num_elements) + return (a->elements[a->current]); + else + return (NULL); +} + + +/* + * 'cupsArraySave()' - Mark the current element for a later @link cupsArrayRestore@. + * + * The current element is undefined until you call @link cupsArrayFind@, + * @link cupsArrayFirst@, or @link cupsArrayIndex@, or @link cupsArrayLast@ + * to set the current element. + * + * The save/restore stack is guaranteed to be at least 32 elements deep. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 1 on success, 0 on failure */ +cupsArraySave(cups_array_t *a) /* I - Array */ +{ + if (!a) + return (0); + + if (a->num_saved >= _CUPS_MAXSAVE) + return (0); + + a->saved[a->num_saved] = a->current; + a->num_saved ++; + + return (1); +} + + +/* + * 'cupsArrayUserData()' - Return the user data for an array. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void * /* O - User data */ +cupsArrayUserData(cups_array_t *a) /* I - Array */ +{ + if (a) + return (a->data); + else + return (NULL); +} + + +/* + * 'cups_array_add()' - Insert or append an element to the array. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +static int /* O - 1 on success, 0 on failure */ +cups_array_add(cups_array_t *a, /* I - Array */ + void *e, /* I - Element to add */ + int insert) /* I - 1 = insert, 0 = append */ +{ + int i, /* Looping var */ + current, /* Current element */ + diff; /* Comparison with current element */ + + + DEBUG_printf(("7cups_array_add(a=%p, e=%p, insert=%d)", a, e, insert)); + + /* + * Verify we have room for the new element... + */ + + if (a->num_elements >= a->alloc_elements) + { + /* + * Allocate additional elements; start with 16 elements, then + * double the size until 1024 elements, then add 1024 elements + * thereafter... + */ + + void **temp; /* New array elements */ + int count; /* New allocation count */ + + + if (a->alloc_elements == 0) + { + count = 16; + temp = malloc(count * sizeof(void *)); + } + else + { + if (a->alloc_elements < 1024) + count = a->alloc_elements * 2; + else + count = a->alloc_elements + 1024; + + temp = realloc(a->elements, count * sizeof(void *)); + } + + DEBUG_printf(("9cups_array_add: count=%d", count)); + + if (!temp) + { + DEBUG_puts("9cups_array_add: allocation failed, returning 0"); + return (0); + } + + a->alloc_elements = count; + a->elements = temp; + } + + /* + * Find the insertion point for the new element; if there is no + * compare function or elements, just add it to the beginning or end... + */ + + if (!a->num_elements || !a->compare) + { + /* + * No elements or comparison function, insert/append as needed... + */ + + if (insert) + current = 0; /* Insert at beginning */ + else + current = a->num_elements; /* Append to the end */ + } + else + { + /* + * Do a binary search for the insertion point... + */ + + current = cups_array_find(a, e, a->insert, &diff); + + if (diff > 0) + { + /* + * Insert after the current element... + */ + + current ++; + } + else if (!diff) + { + /* + * Compared equal, make sure we add to the begining or end of + * the current run of equal elements... + */ + + a->unique = 0; + + if (insert) + { + /* + * Insert at beginning of run... + */ + + while (current > 0 && !(*(a->compare))(e, a->elements[current - 1], + a->data)) + current --; + } + else + { + /* + * Append at end of run... + */ + + do + { + current ++; + } + while (current < a->num_elements && + !(*(a->compare))(e, a->elements[current], a->data)); + } + } + } + + /* + * Insert or append the element... + */ + + if (current < a->num_elements) + { + /* + * Shift other elements to the right... + */ + + memmove(a->elements + current + 1, a->elements + current, + (a->num_elements - current) * sizeof(void *)); + + if (a->current >= current) + a->current ++; + + for (i = 0; i < a->num_saved; i ++) + if (a->saved[i] >= current) + a->saved[i] ++; + + DEBUG_printf(("9cups_array_add: insert element at index %d...", current)); + } +#ifdef DEBUG + else + DEBUG_printf(("9cups_array_add: append element at %d...", current)); +#endif /* DEBUG */ + + if (a->copyfunc) + { + if ((a->elements[current] = (a->copyfunc)(e, a->data)) == NULL) + { + DEBUG_puts("8cups_array_add: Copy function returned NULL, returning 0"); + return (0); + } + } + else + a->elements[current] = e; + + a->num_elements ++; + a->insert = current; + +#ifdef DEBUG + for (current = 0; current < a->num_elements; current ++) + DEBUG_printf(("9cups_array_add: a->elements[%d]=%p", current, + a->elements[current])); +#endif /* DEBUG */ + + DEBUG_puts("9cups_array_add: returning 1"); + + return (1); +} + + +/* + * 'cups_array_find()' - Find an element in the array. + */ + +static int /* O - Index of match */ +cups_array_find(cups_array_t *a, /* I - Array */ + void *e, /* I - Element */ + int prev, /* I - Previous index */ + int *rdiff) /* O - Difference of match */ +{ + int left, /* Left side of search */ + right, /* Right side of search */ + current, /* Current element */ + diff; /* Comparison with current element */ + + + DEBUG_printf(("7cups_array_find(a=%p, e=%p, prev=%d, rdiff=%p)", a, e, prev, + rdiff)); + + if (a->compare) + { + /* + * Do a binary search for the element... + */ + + DEBUG_puts("9cups_array_find: binary search"); + + if (prev >= 0 && prev < a->num_elements) + { + /* + * Start search on either side of previous... + */ + + if ((diff = (*(a->compare))(e, a->elements[prev], a->data)) == 0 || + (diff < 0 && prev == 0) || + (diff > 0 && prev == (a->num_elements - 1))) + { + /* + * Exact or edge match, return it! + */ + + DEBUG_printf(("9cups_array_find: Returning %d, diff=%d", prev, diff)); + + *rdiff = diff; + + return (prev); + } + else if (diff < 0) + { + /* + * Start with previous on right side... + */ + + left = 0; + right = prev; + } + else + { + /* + * Start wih previous on left side... + */ + + left = prev; + right = a->num_elements - 1; + } + } + else + { + /* + * Start search in the middle... + */ + + left = 0; + right = a->num_elements - 1; + } + + do + { + current = (left + right) / 2; + diff = (*(a->compare))(e, a->elements[current], a->data); + + DEBUG_printf(("9cups_array_find: left=%d, right=%d, current=%d, diff=%d", + left, right, current, diff)); + + if (diff == 0) + break; + else if (diff < 0) + right = current; + else + left = current; + } + while ((right - left) > 1); + + if (diff != 0) + { + /* + * Check the last 1 or 2 elements... + */ + + if ((diff = (*(a->compare))(e, a->elements[left], a->data)) <= 0) + current = left; + else + { + diff = (*(a->compare))(e, a->elements[right], a->data); + current = right; + } + } + } + else + { + /* + * Do a linear pointer search... + */ + + DEBUG_puts("9cups_array_find: linear search"); + + diff = 1; + + for (current = 0; current < a->num_elements; current ++) + if (a->elements[current] == e) + { + diff = 0; + break; + } + } + + /* + * Return the closest element and the difference... + */ + + DEBUG_printf(("8cups_array_find: Returning %d, diff=%d", current, diff)); + + *rdiff = diff; + + return (current); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/array.h b/cups/array.h new file mode 100644 index 0000000000..385b7dffcf --- /dev/null +++ b/cups/array.h @@ -0,0 +1,92 @@ +/* + * "$Id$" + * + * Sorted array definitions for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_ARRAY_H_ +# define _CUPS_ARRAY_H_ + +/* + * Include necessary headers... + */ + +# include "versioning.h" +# include + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Types and structures... + */ + +typedef struct _cups_array_s cups_array_t; + /**** CUPS array type ****/ +typedef int (*cups_array_func_t)(void *first, void *second, void *data); + /**** Array comparison function ****/ +typedef int (*cups_ahash_func_t)(void *element, void *data); + /**** Array hash function ****/ +typedef void *(*cups_acopy_func_t)(void *element, void *data); + /**** Array element copy function ****/ +typedef void (*cups_afree_func_t)(void *element, void *data); + /**** Array element free function ****/ + + +/* + * Functions... + */ + +extern int cupsArrayAdd(cups_array_t *a, void *e) _CUPS_API_1_2; +extern void cupsArrayClear(cups_array_t *a) _CUPS_API_1_2; +extern int cupsArrayCount(cups_array_t *a) _CUPS_API_1_2; +extern void *cupsArrayCurrent(cups_array_t *a) _CUPS_API_1_2; +extern void cupsArrayDelete(cups_array_t *a) _CUPS_API_1_2; +extern cups_array_t *cupsArrayDup(cups_array_t *a) _CUPS_API_1_2; +extern void *cupsArrayFind(cups_array_t *a, void *e) _CUPS_API_1_2; +extern void *cupsArrayFirst(cups_array_t *a) _CUPS_API_1_2; +extern int cupsArrayGetIndex(cups_array_t *a) _CUPS_API_1_3; +extern int cupsArrayGetInsert(cups_array_t *a) _CUPS_API_1_3; +extern void *cupsArrayIndex(cups_array_t *a, int n) _CUPS_API_1_2; +extern int cupsArrayInsert(cups_array_t *a, void *e) _CUPS_API_1_2; +extern void *cupsArrayLast(cups_array_t *a) _CUPS_API_1_2; +extern cups_array_t *cupsArrayNew(cups_array_func_t f, void *d) _CUPS_API_1_2; +extern cups_array_t *cupsArrayNew2(cups_array_func_t f, void *d, + cups_ahash_func_t h, int hsize) _CUPS_API_1_3; +extern cups_array_t *cupsArrayNew3(cups_array_func_t f, void *d, + cups_ahash_func_t h, int hsize, + cups_acopy_func_t cf, + cups_afree_func_t ff) _CUPS_API_1_5; +extern void *cupsArrayNext(cups_array_t *a) _CUPS_API_1_2; +extern void *cupsArrayPrev(cups_array_t *a) _CUPS_API_1_2; +extern int cupsArrayRemove(cups_array_t *a, void *e) _CUPS_API_1_2; +extern void *cupsArrayRestore(cups_array_t *a) _CUPS_API_1_2; +extern int cupsArraySave(cups_array_t *a) _CUPS_API_1_2; +extern void *cupsArrayUserData(cups_array_t *a) _CUPS_API_1_2; + +# ifdef __cplusplus +} +# endif /* __cplusplus */ +#endif /* !_CUPS_ARRAY_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/attr.c b/cups/attr.c new file mode 100644 index 0000000000..10e1dfd14e --- /dev/null +++ b/cups/attr.c @@ -0,0 +1,335 @@ +/* + * "$Id$" + * + * PPD model-specific attribute routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * ppdFindAttr() - Find the first matching attribute. + * ppdFindNextAttr() - Find the next matching attribute. + * _ppdNormalizeMakeAndModel() - Normalize a product/make-and-model string. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include "ppd-private.h" + + +/* + * 'ppdFindAttr()' - Find the first matching attribute. + * + * @since CUPS 1.1.19/Mac OS X 10.3@ + */ + +ppd_attr_t * /* O - Attribute or @code NULL@ if not found */ +ppdFindAttr(ppd_file_t *ppd, /* I - PPD file data */ + const char *name, /* I - Attribute name */ + const char *spec) /* I - Specifier string or @code NULL@ */ +{ + ppd_attr_t key, /* Search key */ + *attr; /* Current attribute */ + + + DEBUG_printf(("2ppdFindAttr(ppd=%p, name=\"%s\", spec=\"%s\")", ppd, name, + spec)); + + /* + * Range check input... + */ + + if (!ppd || !name || ppd->num_attrs == 0) + return (NULL); + + /* + * Search for a matching attribute... + */ + + memset(&key, 0, sizeof(key)); + strlcpy(key.name, name, sizeof(key.name)); + + /* + * Return the first matching attribute, if any... + */ + + if ((attr = (ppd_attr_t *)cupsArrayFind(ppd->sorted_attrs, &key)) != NULL) + { + if (spec) + { + /* + * Loop until we find the first matching attribute for "spec"... + */ + + while (attr && _cups_strcasecmp(spec, attr->spec)) + { + if ((attr = (ppd_attr_t *)cupsArrayNext(ppd->sorted_attrs)) != NULL && + _cups_strcasecmp(attr->name, name)) + attr = NULL; + } + } + } + + return (attr); +} + + +/* + * 'ppdFindNextAttr()' - Find the next matching attribute. + * + * @since CUPS 1.1.19/Mac OS X 10.3@ + */ + +ppd_attr_t * /* O - Attribute or @code NULL@ if not found */ +ppdFindNextAttr(ppd_file_t *ppd, /* I - PPD file data */ + const char *name, /* I - Attribute name */ + const char *spec) /* I - Specifier string or @code NULL@ */ +{ + ppd_attr_t *attr; /* Current attribute */ + + + /* + * Range check input... + */ + + if (!ppd || !name || ppd->num_attrs == 0) + return (NULL); + + /* + * See if there are more attributes to return... + */ + + while ((attr = (ppd_attr_t *)cupsArrayNext(ppd->sorted_attrs)) != NULL) + { + /* + * Check the next attribute to see if it is a match... + */ + + if (_cups_strcasecmp(attr->name, name)) + { + /* + * Nope, reset the current pointer to the end of the array... + */ + + cupsArrayIndex(ppd->sorted_attrs, cupsArrayCount(ppd->sorted_attrs)); + + return (NULL); + } + + if (!spec || !_cups_strcasecmp(attr->spec, spec)) + break; + } + + /* + * Return the next attribute's value... + */ + + return (attr); +} + + +/* + * '_ppdNormalizeMakeAndModel()' - Normalize a product/make-and-model string. + * + * This function tries to undo the mistakes made by many printer manufacturers + * to produce a clean make-and-model string we can use. + */ + +char * /* O - Normalized make-and-model string or NULL on error */ +_ppdNormalizeMakeAndModel( + const char *make_and_model, /* I - Original make-and-model string */ + char *buffer, /* I - String buffer */ + size_t bufsize) /* I - Size of string buffer */ +{ + char *bufptr; /* Pointer into buffer */ + + + if (!make_and_model || !buffer || bufsize < 1) + { + if (buffer) + *buffer = '\0'; + + return (NULL); + } + + /* + * Skip leading whitespace... + */ + + while (_cups_isspace(*make_and_model)) + make_and_model ++; + + /* + * Remove parenthesis and add manufacturers as needed... + */ + + if (make_and_model[0] == '(') + { + strlcpy(buffer, make_and_model + 1, bufsize); + + if ((bufptr = strrchr(buffer, ')')) != NULL) + *bufptr = '\0'; + } + else if (!_cups_strncasecmp(make_and_model, "XPrint", 6)) + { + /* + * Xerox XPrint... + */ + + snprintf(buffer, bufsize, "Xerox %s", make_and_model); + } + else if (!_cups_strncasecmp(make_and_model, "Eastman", 7)) + { + /* + * Kodak... + */ + + snprintf(buffer, bufsize, "Kodak %s", make_and_model + 7); + } + else if (!_cups_strncasecmp(make_and_model, "laserwriter", 11)) + { + /* + * Apple LaserWriter... + */ + + snprintf(buffer, bufsize, "Apple LaserWriter%s", make_and_model + 11); + } + else if (!_cups_strncasecmp(make_and_model, "colorpoint", 10)) + { + /* + * Seiko... + */ + + snprintf(buffer, bufsize, "Seiko %s", make_and_model); + } + else if (!_cups_strncasecmp(make_and_model, "fiery", 5)) + { + /* + * EFI... + */ + + snprintf(buffer, bufsize, "EFI %s", make_and_model); + } + else if (!_cups_strncasecmp(make_and_model, "ps ", 3) || + !_cups_strncasecmp(make_and_model, "colorpass", 9)) + { + /* + * Canon... + */ + + snprintf(buffer, bufsize, "Canon %s", make_and_model); + } + else if (!_cups_strncasecmp(make_and_model, "primera", 7)) + { + /* + * Fargo... + */ + + snprintf(buffer, bufsize, "Fargo %s", make_and_model); + } + else if (!_cups_strncasecmp(make_and_model, "designjet", 9) || + !_cups_strncasecmp(make_and_model, "deskjet", 7)) + { + /* + * HP... + */ + + snprintf(buffer, bufsize, "HP %s", make_and_model); + } + else + strlcpy(buffer, make_and_model, bufsize); + + /* + * Clean up the make... + */ + + if (!_cups_strncasecmp(buffer, "agfa", 4)) + { + /* + * Replace with AGFA (all uppercase)... + */ + + buffer[0] = 'A'; + buffer[1] = 'G'; + buffer[2] = 'F'; + buffer[3] = 'A'; + } + else if (!_cups_strncasecmp(buffer, "Hewlett-Packard hp ", 19)) + { + /* + * Just put "HP" on the front... + */ + + buffer[0] = 'H'; + buffer[1] = 'P'; + _cups_strcpy(buffer + 2, buffer + 18); + } + else if (!_cups_strncasecmp(buffer, "Hewlett-Packard ", 16)) + { + /* + * Just put "HP" on the front... + */ + + buffer[0] = 'H'; + buffer[1] = 'P'; + _cups_strcpy(buffer + 2, buffer + 15); + } + else if (!_cups_strncasecmp(buffer, "Lexmark International", 21)) + { + /* + * Strip "International"... + */ + + _cups_strcpy(buffer + 8, buffer + 21); + } + else if (!_cups_strncasecmp(buffer, "herk", 4)) + { + /* + * Replace with LHAG... + */ + + buffer[0] = 'L'; + buffer[1] = 'H'; + buffer[2] = 'A'; + buffer[3] = 'G'; + } + else if (!_cups_strncasecmp(buffer, "linotype", 8)) + { + /* + * Replace with LHAG... + */ + + buffer[0] = 'L'; + buffer[1] = 'H'; + buffer[2] = 'A'; + buffer[3] = 'G'; + _cups_strcpy(buffer + 4, buffer + 8); + } + + /* + * Remove trailing whitespace and return... + */ + + for (bufptr = buffer + strlen(buffer) - 1; + bufptr >= buffer && _cups_isspace(*bufptr); + bufptr --); + + bufptr[1] = '\0'; + + return (buffer[0] ? buffer : NULL); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/auth.c b/cups/auth.c new file mode 100644 index 0000000000..0734c401d0 --- /dev/null +++ b/cups/auth.c @@ -0,0 +1,878 @@ +/* + * "$Id$" + * + * Authentication functions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * This file contains Kerberos support code, copyright 2006 by + * Jelmer Vernooij. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsDoAuthentication() - Authenticate a request. + * _cupsSetNegotiateAuthString() - Set the Kerberos authentication string. + * cups_gss_acquire() - Kerberos credentials callback. + * cups_gss_getname() - Get CUPS service credentials for + * authentication. + * cups_gss_printf() - Show debug error messages from GSSAPI. + * cups_local_auth() - Get the local authorization certificate if + * available/applicable. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include +#include +#if defined(WIN32) || defined(__EMX__) +# include +#else +# include +#endif /* WIN32 || __EMX__ */ + +#if HAVE_AUTHORIZATION_H +# include +# ifdef HAVE_SECBASEPRIV_H +# include +# else +extern const char *cssmErrorString(int error); +# endif /* HAVE_SECBASEPRIV_H */ +#endif /* HAVE_AUTHORIZATION_H */ + +#if defined(SO_PEERCRED) && defined(AF_LOCAL) +# include +#endif /* SO_PEERCRED && AF_LOCAL */ + + +/* + * Local functions... + */ + +#ifdef HAVE_GSSAPI +# ifdef HAVE_GSS_ACQUIRE_CRED_EX_F +# ifdef HAVE_GSS_GSSAPI_SPI_H +# include +# else +# define GSS_AUTH_IDENTITY_TYPE_1 1 +# define gss_acquire_cred_ex_f __ApplePrivate_gss_acquire_cred_ex_f +typedef struct gss_auth_identity +{ + uint32_t type; + uint32_t flags; + char *username; + char *realm; + char *password; + gss_buffer_t *credentialsRef; +} gss_auth_identity_desc; +extern OM_uint32 gss_acquire_cred_ex_f(gss_status_id_t, const gss_name_t, + OM_uint32, OM_uint32, const gss_OID, + gss_cred_usage_t, gss_auth_identity_t, + void *, void (*)(void *, OM_uint32, + gss_status_id_t, + gss_cred_id_t, + gss_OID_set, + OM_uint32)); +# endif /* HAVE_GSS_GSSAPI_SPI_H */ +# include +typedef struct _cups_gss_acquire_s /* Acquire callback data */ +{ + dispatch_semaphore_t sem; /* Synchronization semaphore */ + OM_uint32 major; /* Returned status code */ + gss_cred_id_t creds; /* Returned credentials */ +} _cups_gss_acquire_t; + +static void cups_gss_acquire(void *ctx, OM_uint32 major, + gss_status_id_t status, + gss_cred_id_t creds, gss_OID_set oids, + OM_uint32 time_rec); +# endif /* HAVE_GSS_ACQUIRE_CRED_EX_F */ +static gss_name_t cups_gss_getname(http_t *http, const char *service_name); +# ifdef DEBUG +static void cups_gss_printf(OM_uint32 major_status, OM_uint32 minor_status, + const char *message); +# else +# define cups_gss_printf(major, minor, message) +# endif /* DEBUG */ +#endif /* HAVE_GSSAPI */ +static int cups_local_auth(http_t *http); + + +/* + * 'cupsDoAuthentication()' - Authenticate a request. + * + * This function should be called in response to a @code HTTP_UNAUTHORIZED@ + * status, prior to resubmitting your request. + * + * @since CUPS 1.1.20/Mac OS X 10.4@ + */ + +int /* O - 0 on success, -1 on error */ +cupsDoAuthentication( + http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + const char *method, /* I - Request method ("GET", "POST", "PUT") */ + const char *resource) /* I - Resource path */ +{ + const char *password; /* Password string */ + char prompt[1024], /* Prompt for user */ + realm[HTTP_MAX_VALUE], /* realm="xyz" string */ + nonce[HTTP_MAX_VALUE]; /* nonce="xyz" string */ + int localauth; /* Local authentication result */ + _cups_globals_t *cg; /* Global data */ + + + DEBUG_printf(("cupsDoAuthentication(http=%p, method=\"%s\", resource=\"%s\")", + http, method, resource)); + + if (!http) + http = _cupsConnect(); + + if (!http || !method || !resource) + return (-1); + + DEBUG_printf(("2cupsDoAuthentication: digest_tries=%d, userpass=\"%s\"", + http->digest_tries, http->userpass)); + DEBUG_printf(("2cupsDoAuthentication: WWW-Authenticate=\"%s\"", + httpGetField(http, HTTP_FIELD_WWW_AUTHENTICATE))); + + /* + * Clear the current authentication string... + */ + + httpSetAuthString(http, NULL, NULL); + + /* + * See if we can do local authentication... + */ + + if (http->digest_tries < 3) + { + if ((localauth = cups_local_auth(http)) == 0) + { + DEBUG_printf(("2cupsDoAuthentication: authstring=\"%s\"", + http->authstring)); + + if (http->status == HTTP_UNAUTHORIZED) + http->digest_tries ++; + + return (0); + } + else if (localauth == -1) + { + http->status = HTTP_AUTHORIZATION_CANCELED; + return (-1); /* Error or canceled */ + } + } + + /* + * Nope, see if we should retry the current username:password... + */ + + if ((http->digest_tries > 1 || !http->userpass[0]) && + (!strncmp(http->fields[HTTP_FIELD_WWW_AUTHENTICATE], "Basic", 5) || + !strncmp(http->fields[HTTP_FIELD_WWW_AUTHENTICATE], "Digest", 6))) + { + /* + * Nope - get a new password from the user... + */ + + cg = _cupsGlobals(); + + if (!cg->lang_default) + cg->lang_default = cupsLangDefault(); + + snprintf(prompt, sizeof(prompt), + _cupsLangString(cg->lang_default, _("Password for %s on %s? ")), + cupsUser(), + http->hostname[0] == '/' ? "localhost" : http->hostname); + + http->digest_tries = _cups_strncasecmp(http->fields[HTTP_FIELD_WWW_AUTHENTICATE], + "Digest", 5) != 0; + http->userpass[0] = '\0'; + + if ((password = cupsGetPassword2(prompt, http, method, resource)) == NULL) + { + http->status = HTTP_AUTHORIZATION_CANCELED; + return (-1); + } + + snprintf(http->userpass, sizeof(http->userpass), "%s:%s", cupsUser(), + password); + } + else if (http->status == HTTP_UNAUTHORIZED) + http->digest_tries ++; + + if (http->status == HTTP_UNAUTHORIZED && http->digest_tries >= 3) + { + DEBUG_printf(("1cupsDoAuthentication: Too many authentication tries (%d)", + http->digest_tries)); + + http->status = HTTP_AUTHORIZATION_CANCELED; + return (-1); + } + + /* + * Got a password; encode it for the server... + */ + +#ifdef HAVE_GSSAPI + if (!strncmp(http->fields[HTTP_FIELD_WWW_AUTHENTICATE], "Negotiate", 9)) + { + /* + * Kerberos authentication... + */ + + if (_cupsSetNegotiateAuthString(http, method, resource)) + { + http->status = HTTP_AUTHORIZATION_CANCELED; + return (-1); + } + } + else +#endif /* HAVE_GSSAPI */ + if (!strncmp(http->fields[HTTP_FIELD_WWW_AUTHENTICATE], "Basic", 5)) + { + /* + * Basic authentication... + */ + + char encode[256]; /* Base64 buffer */ + + + httpEncode64_2(encode, sizeof(encode), http->userpass, + (int)strlen(http->userpass)); + httpSetAuthString(http, "Basic", encode); + } + else if (!strncmp(http->fields[HTTP_FIELD_WWW_AUTHENTICATE], "Digest", 6)) + { + /* + * Digest authentication... + */ + + char encode[33], /* MD5 buffer */ + digest[1024]; /* Digest auth data */ + + + httpGetSubField(http, HTTP_FIELD_WWW_AUTHENTICATE, "realm", realm); + httpGetSubField(http, HTTP_FIELD_WWW_AUTHENTICATE, "nonce", nonce); + + httpMD5(cupsUser(), realm, strchr(http->userpass, ':') + 1, encode); + httpMD5Final(nonce, method, resource, encode); + snprintf(digest, sizeof(digest), + "username=\"%s\", realm=\"%s\", nonce=\"%s\", uri=\"%s\", " + "response=\"%s\"", cupsUser(), realm, nonce, resource, encode); + httpSetAuthString(http, "Digest", digest); + } + else + { + DEBUG_printf(("1cupsDoAuthentication: Unknown auth type: \"%s\"", + http->fields[HTTP_FIELD_WWW_AUTHENTICATE])); + http->status = HTTP_AUTHORIZATION_CANCELED; + return (-1); + } + + DEBUG_printf(("1cupsDoAuthentication: authstring=\"%s\"", http->authstring)); + + return (0); +} + + +#ifdef HAVE_GSSAPI +/* + * '_cupsSetNegotiateAuthString()' - Set the Kerberos authentication string. + */ + +int /* O - 0 on success, -1 on error */ +_cupsSetNegotiateAuthString( + http_t *http, /* I - Connection to server */ + const char *method, /* I - Request method ("GET", "POST", "PUT") */ + const char *resource) /* I - Resource path */ +{ + OM_uint32 minor_status, /* Minor status code */ + major_status; /* Major status code */ + gss_buffer_desc output_token = GSS_C_EMPTY_BUFFER; + /* Output token */ + + + (void)method; + (void)resource; + +# ifdef __APPLE__ + /* + * If the weak-linked GSSAPI/Kerberos library is not present, don't try + * to use it... + */ + + if (gss_init_sec_context == NULL) + { + DEBUG_puts("1_cupsSetNegotiateAuthString: Weak-linked GSSAPI/Kerberos " + "framework is not present"); + return (-1); + } +# endif /* __APPLE__ */ + + if (http->gssname == GSS_C_NO_NAME) + { + http->gssname = cups_gss_getname(http, _cupsGSSServiceName()); + } + + if (http->gssctx != GSS_C_NO_CONTEXT) + { + gss_delete_sec_context(&minor_status, &http->gssctx, GSS_C_NO_BUFFER); + http->gssctx = GSS_C_NO_CONTEXT; + } + + major_status = gss_init_sec_context(&minor_status, GSS_C_NO_CREDENTIAL, + &http->gssctx, + http->gssname, http->gssmech, + GSS_C_MUTUAL_FLAG | GSS_C_INTEG_FLAG, + GSS_C_INDEFINITE, + GSS_C_NO_CHANNEL_BINDINGS, + GSS_C_NO_BUFFER, &http->gssmech, + &output_token, NULL, NULL); + +#ifdef HAVE_GSS_ACQUIRE_CRED_EX_F + if (major_status == GSS_S_NO_CRED) + { + /* + * Ask the user for credentials... + */ + + char prompt[1024], /* Prompt for user */ + userbuf[256]; /* Kerberos username */ + const char *username, /* Username string */ + *password; /* Password string */ + _cups_gss_acquire_t data; /* Callback data */ + gss_auth_identity_desc identity; /* Kerberos user identity */ + _cups_globals_t *cg = _cupsGlobals(); + /* Per-thread global data */ + + if (!cg->lang_default) + cg->lang_default = cupsLangDefault(); + + snprintf(prompt, sizeof(prompt), + _cupsLangString(cg->lang_default, _("Password for %s on %s? ")), + cupsUser(), http->gsshost); + + if ((password = cupsGetPassword2(prompt, http, method, resource)) == NULL) + return (-1); + + /* + * Try to acquire credentials... + */ + + username = cupsUser(); + if (!strchr(username, '@')) + { + snprintf(userbuf, sizeof(userbuf), "%s@%s", username, http->gsshost); + username = userbuf; + } + + identity.type = GSS_AUTH_IDENTITY_TYPE_1; + identity.flags = 0; + identity.username = (char *)username; + identity.realm = (char *)""; + identity.password = (char *)password; + identity.credentialsRef = NULL; + + data.sem = dispatch_semaphore_create(0); + data.major = 0; + data.creds = NULL; + + if (data.sem) + { + major_status = gss_acquire_cred_ex_f(NULL, GSS_C_NO_NAME, 0, + GSS_C_INDEFINITE, GSS_KRB5_MECHANISM, + GSS_C_INITIATE, &identity, &data, + cups_gss_acquire); + + if (major_status == GSS_S_COMPLETE) + { + dispatch_semaphore_wait(data.sem, DISPATCH_TIME_FOREVER); + major_status = data.major; + } + + dispatch_release(data.sem); + + if (major_status == GSS_S_COMPLETE) + { + OM_uint32 release_minor; /* Minor status from releasing creds */ + + major_status = gss_init_sec_context(&minor_status, data.creds, + &http->gssctx, + http->gssname, http->gssmech, + GSS_C_MUTUAL_FLAG | GSS_C_INTEG_FLAG, + GSS_C_INDEFINITE, + GSS_C_NO_CHANNEL_BINDINGS, + GSS_C_NO_BUFFER, &http->gssmech, + &output_token, NULL, NULL); + gss_release_cred(&release_minor, &data.creds); + } + } + } +#endif /* HAVE_GSS_ACQUIRED_CRED_EX_F */ + + if (GSS_ERROR(major_status)) + { + cups_gss_printf(major_status, minor_status, + "_cupsSetNegotiateAuthString: Unable to initialize " + "security context"); + return (-1); + } + +#ifdef DEBUG + else if (major_status == GSS_S_CONTINUE_NEEDED) + cups_gss_printf(major_status, minor_status, + "_cupsSetNegotiateAuthString: Continuation needed!"); +#endif /* DEBUG */ + + if (output_token.length > 0 && output_token.length <= 65536) + { + /* + * Allocate the authorization string since Windows KDCs can have + * arbitrarily large credentials... + */ + + int authsize = 10 + /* "Negotiate " */ + output_token.length * 4 / 3 + 1 + /* Base64 */ + 1; /* nul */ + + httpSetAuthString(http, NULL, NULL); + + if ((http->authstring = malloc(authsize)) == NULL) + { + http->authstring = http->_authstring; + authsize = sizeof(http->_authstring); + } + + strcpy(http->authstring, "Negotiate "); + httpEncode64_2(http->authstring + 10, authsize - 10, output_token.value, + output_token.length); + + gss_release_buffer(&minor_status, &output_token); + } + else + { + DEBUG_printf(("1_cupsSetNegotiateAuthString: Kerberos credentials too " + "large - %d bytes!", (int)output_token.length)); + gss_release_buffer(&minor_status, &output_token); + + return (-1); + } + + return (0); +} + + +# ifdef HAVE_GSS_ACQUIRE_CRED_EX_F +/* + * 'cups_gss_acquire()' - Kerberos credentials callback. + */ +static void +cups_gss_acquire( + void *ctx, /* I - Caller context */ + OM_uint32 major, /* I - Major error code */ + gss_status_id_t status, /* I - Status (unused) */ + gss_cred_id_t creds, /* I - Credentials (if any) */ + gss_OID_set oids, /* I - Mechanism OIDs (unused) */ + OM_uint32 time_rec) /* I - Timestamp (unused) */ +{ + uint32_t min; /* Minor error code */ + _cups_gss_acquire_t *data; /* Callback data */ + + + (void)status; + (void)time_rec; + + data = (_cups_gss_acquire_t *)ctx; + data->major = major; + data->creds = creds; + + gss_release_oid_set(&min, &oids); + dispatch_semaphore_signal(data->sem); +} +# endif /* HAVE_GSS_ACQUIRE_CRED_EX_F */ + + +/* + * 'cups_gss_getname()' - Get CUPS service credentials for authentication. + */ + +static gss_name_t /* O - Server name */ +cups_gss_getname( + http_t *http, /* I - Connection to server */ + const char *service_name) /* I - Service name */ +{ + gss_buffer_desc token = GSS_C_EMPTY_BUFFER; + /* Service token */ + OM_uint32 major_status, /* Major status code */ + minor_status; /* Minor status code */ + gss_name_t server_name; /* Server name */ + char buf[1024]; /* Name buffer */ + + + DEBUG_printf(("7cups_gss_getname(http=%p, service_name=\"%s\")", http, + service_name)); + + + /* + * Get the hostname... + */ + + if (!http->gsshost[0]) + { + httpGetHostname(http, http->gsshost, sizeof(http->gsshost)); + + if (!strcmp(http->gsshost, "localhost")) + { + if (gethostname(http->gsshost, sizeof(http->gsshost)) < 0) + { + DEBUG_printf(("1cups_gss_getname: gethostname() failed: %s", + strerror(errno))); + http->gsshost[0] = '\0'; + return (NULL); + } + + if (!strchr(http->gsshost, '.')) + { + /* + * The hostname is not a FQDN, so look it up... + */ + + struct hostent *host; /* Host entry to get FQDN */ + + if ((host = gethostbyname(http->gsshost)) != NULL && host->h_name) + { + /* + * Use the resolved hostname... + */ + + strlcpy(http->gsshost, host->h_name, sizeof(http->gsshost)); + } + else + { + DEBUG_printf(("1cups_gss_getname: gethostbyname(\"%s\") failed.", + http->gsshost)); + http->gsshost[0] = '\0'; + return (NULL); + } + } + } + } + + /* + * Get a service name we can use for authentication purposes... + */ + + snprintf(buf, sizeof(buf), "%s@%s", service_name, http->gsshost); + + DEBUG_printf(("8cups_gss_getname: Looking up \"%s\".", buf)); + + token.value = buf; + token.length = strlen(buf); + server_name = GSS_C_NO_NAME; + major_status = gss_import_name(&minor_status, &token, + GSS_C_NT_HOSTBASED_SERVICE, + &server_name); + + if (GSS_ERROR(major_status)) + { + cups_gss_printf(major_status, minor_status, + "cups_gss_getname: gss_import_name() failed"); + return (NULL); + } + + return (server_name); +} + + +# ifdef DEBUG +/* + * 'cups_gss_printf()' - Show debug error messages from GSSAPI. + */ + +static void +cups_gss_printf(OM_uint32 major_status,/* I - Major status code */ + OM_uint32 minor_status,/* I - Minor status code */ + const char *message) /* I - Prefix for error message */ +{ + OM_uint32 err_major_status, /* Major status code for display */ + err_minor_status; /* Minor status code for display */ + OM_uint32 msg_ctx; /* Message context */ + gss_buffer_desc major_status_string = GSS_C_EMPTY_BUFFER, + /* Major status message */ + minor_status_string = GSS_C_EMPTY_BUFFER; + /* Minor status message */ + + + msg_ctx = 0; + err_major_status = gss_display_status(&err_minor_status, + major_status, + GSS_C_GSS_CODE, + GSS_C_NO_OID, + &msg_ctx, + &major_status_string); + + if (!GSS_ERROR(err_major_status)) + gss_display_status(&err_minor_status, minor_status, GSS_C_MECH_CODE, + GSS_C_NULL_OID, &msg_ctx, &minor_status_string); + + DEBUG_printf(("1%s: %s, %s", message, (char *)major_status_string.value, + (char *)minor_status_string.value)); + + gss_release_buffer(&err_minor_status, &major_status_string); + gss_release_buffer(&err_minor_status, &minor_status_string); +} +# endif /* DEBUG */ +#endif /* HAVE_GSSAPI */ + + +/* + * 'cups_local_auth()' - Get the local authorization certificate if + * available/applicable. + */ + +static int /* O - 0 if available */ + /* 1 if not available */ + /* -1 error */ +cups_local_auth(http_t *http) /* I - HTTP connection to server */ +{ +#if defined(WIN32) || defined(__EMX__) + /* + * Currently WIN32 and OS-2 do not support the CUPS server... + */ + + return (1); +#else + int pid; /* Current process ID */ + FILE *fp; /* Certificate file */ + char trc[16], /* Try Root Certificate parameter */ + filename[1024], /* Certificate filename */ + certificate[33];/* Certificate string */ + _cups_globals_t *cg = _cupsGlobals(); /* Global data */ +# if defined(HAVE_AUTHORIZATION_H) + OSStatus status; /* Status */ + AuthorizationItem auth_right; /* Authorization right */ + AuthorizationRights auth_rights; /* Authorization rights */ + AuthorizationFlags auth_flags; /* Authorization flags */ + AuthorizationExternalForm auth_extrn; /* Authorization ref external */ + char auth_key[1024]; /* Buffer */ + char buffer[1024]; /* Buffer */ +# endif /* HAVE_AUTHORIZATION_H */ + + + DEBUG_printf(("7cups_local_auth(http=%p) hostaddr=%s, hostname=\"%s\"", + http, httpAddrString(http->hostaddr, filename, sizeof(filename)), http->hostname)); + + /* + * See if we are accessing localhost... + */ + + if (!httpAddrLocalhost(http->hostaddr) && + _cups_strcasecmp(http->hostname, "localhost") != 0) + { + DEBUG_puts("8cups_local_auth: Not a local connection!"); + return (1); + } + +# if defined(HAVE_AUTHORIZATION_H) + /* + * Delete any previous authorization reference... + */ + + if (http->auth_ref) + { + AuthorizationFree(http->auth_ref, kAuthorizationFlagDefaults); + http->auth_ref = NULL; + } + + if (!getenv("GATEWAY_INTERFACE") && + httpGetSubField2(http, HTTP_FIELD_WWW_AUTHENTICATE, "authkey", + auth_key, sizeof(auth_key))) + { + status = AuthorizationCreate(NULL, kAuthorizationEmptyEnvironment, + kAuthorizationFlagDefaults, &http->auth_ref); + if (status != errAuthorizationSuccess) + { + DEBUG_printf(("8cups_local_auth: AuthorizationCreate() returned %d (%s)", + (int)status, cssmErrorString(status))); + return (-1); + } + + auth_right.name = auth_key; + auth_right.valueLength = 0; + auth_right.value = NULL; + auth_right.flags = 0; + + auth_rights.count = 1; + auth_rights.items = &auth_right; + + auth_flags = kAuthorizationFlagDefaults | + kAuthorizationFlagPreAuthorize | + kAuthorizationFlagInteractionAllowed | + kAuthorizationFlagExtendRights; + + status = AuthorizationCopyRights(http->auth_ref, &auth_rights, + kAuthorizationEmptyEnvironment, + auth_flags, NULL); + if (status == errAuthorizationSuccess) + status = AuthorizationMakeExternalForm(http->auth_ref, &auth_extrn); + + if (status == errAuthorizationSuccess) + { + /* + * Set the authorization string and return... + */ + + httpEncode64_2(buffer, sizeof(buffer), (void *)&auth_extrn, + sizeof(auth_extrn)); + + httpSetAuthString(http, "AuthRef", buffer); + + DEBUG_printf(("8cups_local_auth: Returning authstring=\"%s\"", + http->authstring)); + return (0); + } + else if (status == errAuthorizationCanceled) + return (-1); + + DEBUG_printf(("9cups_local_auth: AuthorizationCopyRights() returned %d (%s)", + (int)status, cssmErrorString(status))); + + /* + * Fall through to try certificates... + */ + } +# endif /* HAVE_AUTHORIZATION_H */ + +# if defined(SO_PEERCRED) && defined(AF_LOCAL) + /* + * See if we can authenticate using the peer credentials provided over a + * domain socket; if so, specify "PeerCred username" as the authentication + * information... + */ + + if ( +# ifdef HAVE_GSSAPI + strncmp(http->fields[HTTP_FIELD_WWW_AUTHENTICATE], "Negotiate", 9) && +# endif /* HAVE_GSSAPI */ +# ifdef HAVE_AUTHORIZATION_H + !httpGetSubField2(http, HTTP_FIELD_WWW_AUTHENTICATE, "authkey", + auth_key, sizeof(auth_key)) && +# endif /* HAVE_AUTHORIZATION_H */ + http->hostaddr->addr.sa_family == AF_LOCAL && + !getenv("GATEWAY_INTERFACE")) /* Not via CGI programs... */ + { + /* + * Verify that the current cupsUser() matches the current UID... + */ + + struct passwd *pwd; /* Password information */ + const char *username; /* Current username */ + + username = cupsUser(); + + if ((pwd = getpwnam(username)) != NULL && pwd->pw_uid == getuid()) + { + httpSetAuthString(http, "PeerCred", username); + + DEBUG_printf(("8cups_local_auth: Returning authstring=\"%s\"", + http->authstring)); + + return (0); + } + } +# endif /* SO_PEERCRED && AF_LOCAL */ + + /* + * Try opening a certificate file for this PID. If that fails, + * try the root certificate... + */ + + pid = getpid(); + snprintf(filename, sizeof(filename), "%s/certs/%d", cg->cups_statedir, pid); + if ((fp = fopen(filename, "r")) == NULL && pid > 0) + { + /* + * No certificate for this PID; see if we can get the root certificate... + */ + + DEBUG_printf(("9cups_local_auth: Unable to open file %s: %s", + filename, strerror(errno))); + +# ifdef HAVE_GSSAPI + if (!strncmp(http->fields[HTTP_FIELD_WWW_AUTHENTICATE], "Negotiate", 9)) + { + /* + * Kerberos required, don't try the root certificate... + */ + + return (1); + } +# endif /* HAVE_GSSAPI */ + +# ifdef HAVE_AUTHORIZATION_H + if (httpGetSubField2(http, HTTP_FIELD_WWW_AUTHENTICATE, "authkey", + auth_key, sizeof(auth_key))) + { + /* + * Don't use the root certificate as a replacement for an authkey... + */ + + return (1); + } +# endif /* HAVE_AUTHORIZATION_H */ + if (!httpGetSubField2(http, HTTP_FIELD_WWW_AUTHENTICATE, "trc", trc, + sizeof(trc))) + { + /* + * Scheduler doesn't want us to use the root certificate... + */ + + return (1); + } + + snprintf(filename, sizeof(filename), "%s/certs/0", cg->cups_statedir); + fp = fopen(filename, "r"); + } + + if (fp) + { + /* + * Read the certificate from the file... + */ + + fgets(certificate, sizeof(certificate), fp); + fclose(fp); + + /* + * Set the authorization string and return... + */ + + httpSetAuthString(http, "Local", certificate); + + DEBUG_printf(("8cups_local_auth: Returning authstring=\"%s\"", + http->authstring)); + + return (0); + } + + return (1); +#endif /* WIN32 || __EMX__ */ +} + + +/* + * End of "$Id$". + */ diff --git a/cups/backchannel.c b/cups/backchannel.c new file mode 100644 index 0000000000..faf8ec069d --- /dev/null +++ b/cups/backchannel.c @@ -0,0 +1,199 @@ +/* + * "$Id$" + * + * Backchannel functions for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsBackChannelRead() - Read data from the backchannel. + * cupsBackChannelWrite() - Write data to the backchannel. + * cups_setup() - Setup select() + */ + +/* + * Include necessary headers... + */ + +#include "cups.h" +#include +#ifdef WIN32 +# include +# include +#else +# include +#endif /* WIN32 */ + + +/* + * Local functions... + */ + +static void cups_setup(fd_set *set, struct timeval *tval, + double timeout); + + +/* + * 'cupsBackChannelRead()' - Read data from the backchannel. + * + * Reads up to "bytes" bytes from the backchannel/backend. The "timeout" + * parameter controls how many seconds to wait for the data - use 0.0 to + * return immediately if there is no data, -1.0 to wait for data indefinitely. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ssize_t /* O - Bytes read or -1 on error */ +cupsBackChannelRead(char *buffer, /* I - Buffer to read into */ + size_t bytes, /* I - Bytes to read */ + double timeout) /* I - Timeout in seconds, typically 0.0 to poll */ +{ + fd_set input; /* Input set */ + struct timeval tval; /* Timeout value */ + int status; /* Select status */ + + + /* + * Wait for input ready. + */ + + do + { + cups_setup(&input, &tval, timeout); + + if (timeout < 0.0) + status = select(4, &input, NULL, NULL, NULL); + else + status = select(4, &input, NULL, NULL, &tval); + } + while (status < 0 && errno != EINTR && errno != EAGAIN); + + if (status < 0) + return (-1); /* Timeout! */ + + /* + * Read bytes from the pipe... + */ + +#ifdef WIN32 + return ((ssize_t)_read(3, buffer, (unsigned)bytes)); +#else + return (read(3, buffer, bytes)); +#endif /* WIN32 */ +} + + +/* + * 'cupsBackChannelWrite()' - Write data to the backchannel. + * + * Writes "bytes" bytes to the backchannel/filter. The "timeout" parameter + * controls how many seconds to wait for the data to be written - use + * 0.0 to return immediately if the data cannot be written, -1.0 to wait + * indefinitely. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ssize_t /* O - Bytes written or -1 on error */ +cupsBackChannelWrite( + const char *buffer, /* I - Buffer to write */ + size_t bytes, /* I - Bytes to write */ + double timeout) /* I - Timeout in seconds, typically 1.0 */ +{ + fd_set output; /* Output set */ + struct timeval tval; /* Timeout value */ + int status; /* Select status */ + ssize_t count; /* Current bytes */ + size_t total; /* Total bytes */ + + + /* + * Write all bytes... + */ + + total = 0; + + while (total < bytes) + { + /* + * Wait for write-ready... + */ + + do + { + cups_setup(&output, &tval, timeout); + + if (timeout < 0.0) + status = select(4, NULL, &output, NULL, NULL); + else + status = select(4, NULL, &output, NULL, &tval); + } + while (status < 0 && errno != EINTR && errno != EAGAIN); + + if (status <= 0) + return (-1); /* Timeout! */ + + /* + * Write bytes to the pipe... + */ + +#ifdef WIN32 + count = (ssize_t)_write(3, buffer, (unsigned)(bytes - total)); +#else + count = write(3, buffer, bytes - total); +#endif /* WIN32 */ + + if (count < 0) + { + /* + * Write error - abort on fatal errors... + */ + + if (errno != EINTR && errno != EAGAIN) + return (-1); + } + else + { + /* + * Write succeeded, update buffer pointer and total count... + */ + + buffer += count; + total += count; + } + } + + return ((ssize_t)bytes); +} + + +/* + * 'cups_setup()' - Setup select() + */ + +static void +cups_setup(fd_set *set, /* I - Set for select() */ + struct timeval *tval, /* I - Timer value */ + double timeout) /* I - Timeout in seconds */ +{ + tval->tv_sec = (int)timeout; + tval->tv_usec = (int)(1000000.0 * (timeout - tval->tv_sec)); + + FD_ZERO(set); + FD_SET(3, set); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/backend.c b/cups/backend.c new file mode 100644 index 0000000000..031a3919c4 --- /dev/null +++ b/cups/backend.c @@ -0,0 +1,143 @@ +/* + * "$Id$" + * + * Backend functions for CUPS. + * + * Copyright 2007-2012 by Apple Inc. + * Copyright 2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsBackendDeviceURI() - Get the device URI for a backend. + * cupsBackendReport() - Write a device line from a backend. + * quote_string() - Write a quoted string to stdout, escaping \ and ". + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include "backend.h" + + +/* + * Local functions... + */ + +static void quote_string(const char *s); + + +/* + * 'cupsBackendDeviceURI()' - Get the device URI for a backend. + * + * The "argv" argument is the argv argument passed to main(). This + * function returns the device URI passed in the DEVICE_URI environment + * variable or the device URI passed in argv[0], whichever is found + * first. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +const char * /* O - Device URI or @code NULL@ */ +cupsBackendDeviceURI(char **argv) /* I - Command-line arguments */ +{ + const char *device_uri, /* Device URI */ + *auth_info_required; /* AUTH_INFO_REQUIRED env var */ + _cups_globals_t *cg = _cupsGlobals(); /* Global info */ + int options; /* Resolve options */ + + + if ((device_uri = getenv("DEVICE_URI")) == NULL) + { + if (!argv || !argv[0] || !strchr(argv[0], ':')) + return (NULL); + + device_uri = argv[0]; + } + + options = _HTTP_RESOLVE_STDERR; + if ((auth_info_required = getenv("AUTH_INFO_REQUIRED")) != NULL && + !strcmp(auth_info_required, "negotiate")) + options |= _HTTP_RESOLVE_FQDN; + + return (_httpResolveURI(device_uri, cg->resolved_uri, + sizeof(cg->resolved_uri), options, NULL, NULL)); +} + + +/* + * 'cupsBackendReport()' - Write a device line from a backend. + * + * This function writes a single device line to stdout for a backend. + * It handles quoting of special characters in the device-make-and-model, + * device-info, device-id, and device-location strings. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +void +cupsBackendReport( + const char *device_scheme, /* I - device-scheme string */ + const char *device_uri, /* I - device-uri string */ + const char *device_make_and_model, /* I - device-make-and-model string or @code NULL@ */ + const char *device_info, /* I - device-info string or @code NULL@ */ + const char *device_id, /* I - device-id string or @code NULL@ */ + const char *device_location) /* I - device-location string or @code NULL@ */ +{ + if (!device_scheme || !device_uri) + return; + + printf("%s %s", device_scheme, device_uri); + if (device_make_and_model && *device_make_and_model) + quote_string(device_make_and_model); + else + quote_string("unknown"); + quote_string(device_info); + quote_string(device_id); + quote_string(device_location); + putchar('\n'); + fflush(stdout); +} + + +/* + * 'quote_string()' - Write a quoted string to stdout, escaping \ and ". + */ + +static void +quote_string(const char *s) /* I - String to write */ +{ + fputs(" \"", stdout); + + if (s) + { + while (*s) + { + if (*s == '\\' || *s == '\"') + putchar('\\'); + + if (*s == '\n') + putchar(' '); + else + putchar(*s); + + s ++; + } + } + + putchar('\"'); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/backend.h b/cups/backend.h new file mode 100644 index 0000000000..85941c71a7 --- /dev/null +++ b/cups/backend.h @@ -0,0 +1,78 @@ +/* + * "$Id$" + * + * Backend definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_BACKEND_H_ +# define _CUPS_BACKEND_H_ + + +/* + * Include necessary headers... + */ + +# include "versioning.h" + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + +/* + * Constants... + */ + +enum cups_backend_e /**** Backend exit codes ****/ +{ + CUPS_BACKEND_OK = 0, /* Job completed successfully */ + CUPS_BACKEND_FAILED = 1, /* Job failed, use error-policy */ + CUPS_BACKEND_AUTH_REQUIRED = 2, /* Job failed, authentication required */ + CUPS_BACKEND_HOLD = 3, /* Job failed, hold job */ + CUPS_BACKEND_STOP = 4, /* Job failed, stop queue */ + CUPS_BACKEND_CANCEL = 5, /* Job failed, cancel job */ + CUPS_BACKEND_RETRY = 6, /* Job failed, retry this job later */ + CUPS_BACKEND_RETRY_CURRENT = 7 /* Job failed, retry this job immediately */ +}; +typedef enum cups_backend_e cups_backend_t; + /**** Backend exit codes ****/ + + +/* + * Prototypes... + */ + +extern const char *cupsBackendDeviceURI(char **argv) _CUPS_API_1_2; +extern void cupsBackendReport(const char *device_scheme, + const char *device_uri, + const char *device_make_and_model, + const char *device_info, + const char *device_id, + const char *device_location) + _CUPS_API_1_4; + + +# ifdef __cplusplus +} +# endif /* __cplusplus */ + +#endif /* !_CUPS_BACKEND_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/conflicts.c b/cups/conflicts.c new file mode 100644 index 0000000000..cded4585ee --- /dev/null +++ b/cups/conflicts.c @@ -0,0 +1,1214 @@ +/* + * "$Id$" + * + * Option marking routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * PostScript is a trademark of Adobe Systems, Inc. + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsGetConflicts() - Get a list of conflicting options in a marked + * PPD. + * cupsResolveConflicts() - Resolve conflicts in a marked PPD. + * ppdConflicts() - Check to see if there are any conflicts among + * the marked option choices. + * ppdInstallableConflict() - Test whether an option choice conflicts with an + * installable option. + * ppd_is_installable() - Determine whether an option is in the + * InstallableOptions group. + * ppd_load_constraints() - Load constraints from a PPD file. + * ppd_test_constraints() - See if any constraints are active. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include "ppd-private.h" + + +/* + * Local constants... + */ + +enum +{ + _PPD_NORMAL_CONSTRAINTS, + _PPD_OPTION_CONSTRAINTS, + _PPD_INSTALLABLE_CONSTRAINTS, + _PPD_ALL_CONSTRAINTS +}; + + +/* + * Local functions... + */ + +static int ppd_is_installable(ppd_group_t *installable, + const char *option); +static void ppd_load_constraints(ppd_file_t *ppd); +static cups_array_t *ppd_test_constraints(ppd_file_t *ppd, + const char *option, + const char *choice, + int num_options, + cups_option_t *options, + int which); + + +/* + * 'cupsGetConflicts()' - Get a list of conflicting options in a marked PPD. + * + * This function gets a list of options that would conflict if "option" and + * "choice" were marked in the PPD. You would typically call this function + * after marking the currently selected options in the PPD in order to + * determine whether a new option selection would cause a conflict. + * + * The number of conflicting options are returned with "options" pointing to + * the conflicting options. The returned option array must be freed using + * @link cupsFreeOptions@. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +int /* O - Number of conflicting options */ +cupsGetConflicts( + ppd_file_t *ppd, /* I - PPD file */ + const char *option, /* I - Option to test */ + const char *choice, /* I - Choice to test */ + cups_option_t **options) /* O - Conflicting options */ +{ + int i, /* Looping var */ + num_options; /* Number of conflicting options */ + cups_array_t *active; /* Active conflicts */ + _ppd_cups_uiconsts_t *c; /* Current constraints */ + _ppd_cups_uiconst_t *cptr; /* Current constraint */ + ppd_choice_t *marked; /* Marked choice */ + + + /* + * Range check input... + */ + + if (options) + *options = NULL; + + if (!ppd || !option || !choice || !options) + return (0); + + /* + * Test for conflicts... + */ + + active = ppd_test_constraints(ppd, option, choice, 0, NULL, + _PPD_ALL_CONSTRAINTS); + + /* + * Loop through all of the UI constraints and add any options that conflict... + */ + + for (num_options = 0, c = (_ppd_cups_uiconsts_t *)cupsArrayFirst(active); + c; + c = (_ppd_cups_uiconsts_t *)cupsArrayNext(active)) + { + for (i = c->num_constraints, cptr = c->constraints; + i > 0; + i --, cptr ++) + if (_cups_strcasecmp(cptr->option->keyword, option)) + { + if (cptr->choice) + num_options = cupsAddOption(cptr->option->keyword, + cptr->choice->choice, num_options, + options); + else if ((marked = ppdFindMarkedChoice(ppd, + cptr->option->keyword)) != NULL) + num_options = cupsAddOption(cptr->option->keyword, marked->choice, + num_options, options); + } + } + + cupsArrayDelete(active); + + return (num_options); +} + + +/* + * 'cupsResolveConflicts()' - Resolve conflicts in a marked PPD. + * + * This function attempts to resolve any conflicts in a marked PPD, returning + * a list of option changes that are required to resolve them. On input, + * "num_options" and "options" contain any pending option changes that have + * not yet been marked, while "option" and "choice" contain the most recent + * selection which may or may not be in "num_options" or "options". + * + * On successful return, "num_options" and "options" are updated to contain + * "option" and "choice" along with any changes required to resolve conflicts + * specified in the PPD file and 1 is returned. + * + * If option conflicts cannot be resolved, "num_options" and "options" are not + * changed and 0 is returned. + * + * When resolving conflicts, @code cupsResolveConflicts@ does not consider + * changes to the current page size (@code media@, @code PageSize@, and + * @code PageRegion@) or to the most recent option specified in "option". + * Thus, if the only way to resolve a conflict is to change the page size + * or the option the user most recently changed, @code cupsResolveConflicts@ + * will return 0 to indicate it was unable to resolve the conflicts. + * + * The @code cupsResolveConflicts@ function uses one of two sources of option + * constraint information. The preferred constraint information is defined by + * @code cupsUIConstraints@ and @code cupsUIResolver@ attributes - in this + * case, the PPD file provides constraint resolution actions. + * + * The backup constraint information is defined by the + * @code UIConstraints@ and @code NonUIConstraints@ attributes. These + * constraints are resolved algorithmically by first selecting the default + * choice for the conflicting option, then iterating over all possible choices + * until a non-conflicting option choice is found. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +int /* O - 1 on success, 0 on failure */ +cupsResolveConflicts( + ppd_file_t *ppd, /* I - PPD file */ + const char *option, /* I - Newly selected option or @code NULL@ for none */ + const char *choice, /* I - Newly selected choice or @code NULL@ for none */ + int *num_options, /* IO - Number of additional selected options */ + cups_option_t **options) /* IO - Additional selected options */ +{ + int i, /* Looping var */ + tries, /* Number of tries */ + num_newopts; /* Number of new options */ + cups_option_t *newopts; /* New options */ + cups_array_t *active, /* Active constraints */ + *pass, /* Resolvers for this pass */ + *resolvers, /* Resolvers we have used */ + *test; /* Test array for conflicts */ + _ppd_cups_uiconsts_t *consts; /* Current constraints */ + _ppd_cups_uiconst_t *constptr; /* Current constraint */ + ppd_attr_t *resolver; /* Current resolver */ + const char *resval; /* Pointer into resolver value */ + char resoption[PPD_MAX_NAME], + /* Current resolver option */ + reschoice[PPD_MAX_NAME], + /* Current resolver choice */ + *resptr, /* Pointer into option/choice */ + firstpage[255]; /* AP_FIRSTPAGE_Keyword string */ + const char *value; /* Selected option value */ + int changed; /* Did we change anything? */ + ppd_choice_t *marked; /* Marked choice */ + + + /* + * Range check input... + */ + + if (!ppd || !num_options || !options || (option == NULL) != (choice == NULL)) + return (0); + + /* + * Build a shadow option array... + */ + + num_newopts = 0; + newopts = NULL; + + for (i = 0; i < *num_options; i ++) + num_newopts = cupsAddOption((*options)[i].name, (*options)[i].value, + num_newopts, &newopts); + if (option && _cups_strcasecmp(option, "Collate")) + num_newopts = cupsAddOption(option, choice, num_newopts, &newopts); + + /* + * Loop until we have no conflicts... + */ + + cupsArraySave(ppd->sorted_attrs); + + resolvers = NULL; + pass = cupsArrayNew((cups_array_func_t)_cups_strcasecmp, NULL); + tries = 0; + + while (tries < 100 && + (active = ppd_test_constraints(ppd, NULL, NULL, num_newopts, newopts, + _PPD_ALL_CONSTRAINTS)) != NULL) + { + tries ++; + + if (!resolvers) + resolvers = cupsArrayNew((cups_array_func_t)_cups_strcasecmp, NULL); + + for (consts = (_ppd_cups_uiconsts_t *)cupsArrayFirst(active), changed = 0; + consts; + consts = (_ppd_cups_uiconsts_t *)cupsArrayNext(active)) + { + if (consts->resolver[0]) + { + /* + * Look up the resolver... + */ + + if (cupsArrayFind(pass, consts->resolver)) + continue; /* Already applied this resolver... */ + + if (cupsArrayFind(resolvers, consts->resolver)) + { + /* + * Resolver loop! + */ + + DEBUG_printf(("1cupsResolveConflicts: Resolver loop with %s!", + consts->resolver)); + goto error; + } + + if ((resolver = ppdFindAttr(ppd, "cupsUIResolver", + consts->resolver)) == NULL) + { + DEBUG_printf(("1cupsResolveConflicts: Resolver %s not found!", + consts->resolver)); + goto error; + } + + if (!resolver->value) + { + DEBUG_printf(("1cupsResolveConflicts: Resolver %s has no value!", + consts->resolver)); + goto error; + } + + /* + * Add the options from the resolver... + */ + + cupsArrayAdd(pass, consts->resolver); + cupsArrayAdd(resolvers, consts->resolver); + + for (resval = resolver->value; *resval && !changed;) + { + while (_cups_isspace(*resval)) + resval ++; + + if (*resval != '*') + break; + + for (resval ++, resptr = resoption; + *resval && !_cups_isspace(*resval); + resval ++) + if (resptr < (resoption + sizeof(resoption) - 1)) + *resptr++ = *resval; + + *resptr = '\0'; + + while (_cups_isspace(*resval)) + resval ++; + + for (resptr = reschoice; + *resval && !_cups_isspace(*resval); + resval ++) + if (resptr < (reschoice + sizeof(reschoice) - 1)) + *resptr++ = *resval; + + *resptr = '\0'; + + if (!resoption[0] || !reschoice[0]) + break; + + /* + * Is this the option we are changing? + */ + + snprintf(firstpage, sizeof(firstpage), "AP_FIRSTPAGE_%s", resoption); + + if (option && + (!_cups_strcasecmp(resoption, option) || + !_cups_strcasecmp(firstpage, option) || + (!_cups_strcasecmp(option, "PageSize") && + !_cups_strcasecmp(resoption, "PageRegion")) || + (!_cups_strcasecmp(option, "AP_FIRSTPAGE_PageSize") && + !_cups_strcasecmp(resoption, "PageSize")) || + (!_cups_strcasecmp(option, "AP_FIRSTPAGE_PageSize") && + !_cups_strcasecmp(resoption, "PageRegion")) || + (!_cups_strcasecmp(option, "PageRegion") && + !_cups_strcasecmp(resoption, "PageSize")) || + (!_cups_strcasecmp(option, "AP_FIRSTPAGE_PageRegion") && + !_cups_strcasecmp(resoption, "PageSize")) || + (!_cups_strcasecmp(option, "AP_FIRSTPAGE_PageRegion") && + !_cups_strcasecmp(resoption, "PageRegion")))) + continue; + + /* + * Try this choice... + */ + + if ((test = ppd_test_constraints(ppd, resoption, reschoice, + num_newopts, newopts, + _PPD_ALL_CONSTRAINTS)) == NULL) + { + /* + * That worked... + */ + + changed = 1; + } + else + cupsArrayDelete(test); + + /* + * Add the option/choice from the resolver regardless of whether it + * worked; this makes sure that we can cascade several changes to + * make things resolve... + */ + + num_newopts = cupsAddOption(resoption, reschoice, num_newopts, + &newopts); + } + } + else + { + /* + * Try resolving by choosing the default values for non-installable + * options, then by iterating through the possible choices... + */ + + int j; /* Looping var */ + ppd_choice_t *cptr; /* Current choice */ + ppd_size_t *size; /* Current page size */ + + + for (i = consts->num_constraints, constptr = consts->constraints; + i > 0 && !changed; + i --, constptr ++) + { + /* + * Can't resolve by changing an installable option... + */ + + if (constptr->installable) + continue; + + /* + * Is this the option we are changing? + */ + + if (option && + (!_cups_strcasecmp(constptr->option->keyword, option) || + (!_cups_strcasecmp(option, "PageSize") && + !_cups_strcasecmp(constptr->option->keyword, "PageRegion")) || + (!_cups_strcasecmp(option, "PageRegion") && + !_cups_strcasecmp(constptr->option->keyword, "PageSize")))) + continue; + + /* + * Get the current option choice... + */ + + if ((value = cupsGetOption(constptr->option->keyword, num_newopts, + newopts)) == NULL) + { + if (!_cups_strcasecmp(constptr->option->keyword, "PageSize") || + !_cups_strcasecmp(constptr->option->keyword, "PageRegion")) + { + if ((value = cupsGetOption("PageSize", num_newopts, + newopts)) == NULL) + value = cupsGetOption("PageRegion", num_newopts, newopts); + + if (!value) + { + if ((size = ppdPageSize(ppd, NULL)) != NULL) + value = size->name; + else + value = ""; + } + } + else + { + marked = ppdFindMarkedChoice(ppd, constptr->option->keyword); + value = marked ? marked->choice : ""; + } + } + + if (!_cups_strncasecmp(value, "Custom.", 7)) + value = "Custom"; + + /* + * Try the default choice... + */ + + test = NULL; + + if (_cups_strcasecmp(value, constptr->option->defchoice) && + (test = ppd_test_constraints(ppd, constptr->option->keyword, + constptr->option->defchoice, + num_newopts, newopts, + _PPD_OPTION_CONSTRAINTS)) == NULL) + { + /* + * That worked... + */ + + num_newopts = cupsAddOption(constptr->option->keyword, + constptr->option->defchoice, + num_newopts, &newopts); + changed = 1; + } + else + { + /* + * Try each choice instead... + */ + + for (j = constptr->option->num_choices, + cptr = constptr->option->choices; + j > 0; + j --, cptr ++) + { + cupsArrayDelete(test); + test = NULL; + + if (_cups_strcasecmp(value, cptr->choice) && + _cups_strcasecmp(constptr->option->defchoice, cptr->choice) && + _cups_strcasecmp("Custom", cptr->choice) && + (test = ppd_test_constraints(ppd, constptr->option->keyword, + cptr->choice, num_newopts, + newopts, + _PPD_OPTION_CONSTRAINTS)) == NULL) + { + /* + * This choice works... + */ + + num_newopts = cupsAddOption(constptr->option->keyword, + cptr->choice, num_newopts, + &newopts); + changed = 1; + break; + } + } + + cupsArrayDelete(test); + } + } + } + } + + if (!changed) + { + DEBUG_puts("1cupsResolveConflicts: Unable to automatically resolve " + "constraint!"); + goto error; + } + + cupsArrayClear(pass); + cupsArrayDelete(active); + active = NULL; + } + + if (tries >= 100) + goto error; + + /* + * Free the caller's option array... + */ + + cupsFreeOptions(*num_options, *options); + + /* + * If Collate is the option we are testing, add it here. Otherwise, remove + * any Collate option from the resolve list since the filters automatically + * handle manual collation... + */ + + if (option && !_cups_strcasecmp(option, "Collate")) + num_newopts = cupsAddOption(option, choice, num_newopts, &newopts); + else + num_newopts = cupsRemoveOption("Collate", num_newopts, &newopts); + + /* + * Return the new list of options to the caller... + */ + + *num_options = num_newopts; + *options = newopts; + + cupsArrayDelete(pass); + cupsArrayDelete(resolvers); + + cupsArrayRestore(ppd->sorted_attrs); + + DEBUG_printf(("1cupsResolveConflicts: Returning %d options:", num_newopts)); +#ifdef DEBUG + for (i = 0; i < num_newopts; i ++) + DEBUG_printf(("1cupsResolveConflicts: options[%d]: %s=%s", i, + newopts[i].name, newopts[i].value)); +#endif /* DEBUG */ + + return (1); + + /* + * If we get here, we failed to resolve... + */ + + error: + + cupsFreeOptions(num_newopts, newopts); + + cupsArrayDelete(active); + cupsArrayDelete(pass); + cupsArrayDelete(resolvers); + + cupsArrayRestore(ppd->sorted_attrs); + + DEBUG_puts("1cupsResolveConflicts: Unable to resolve conflicts!"); + + return (0); +} + + +/* + * 'ppdConflicts()' - Check to see if there are any conflicts among the + * marked option choices. + * + * The returned value is the same as returned by @link ppdMarkOption@. + */ + +int /* O - Number of conflicts found */ +ppdConflicts(ppd_file_t *ppd) /* I - PPD to check */ +{ + int i, /* Looping variable */ + conflicts; /* Number of conflicts */ + cups_array_t *active; /* Active conflicts */ + _ppd_cups_uiconsts_t *c; /* Current constraints */ + _ppd_cups_uiconst_t *cptr; /* Current constraint */ + ppd_option_t *o; /* Current option */ + + + if (!ppd) + return (0); + + /* + * Clear all conflicts... + */ + + cupsArraySave(ppd->options); + + for (o = ppdFirstOption(ppd); o; o = ppdNextOption(ppd)) + o->conflicted = 0; + + cupsArrayRestore(ppd->options); + + /* + * Test for conflicts... + */ + + active = ppd_test_constraints(ppd, NULL, NULL, 0, NULL, + _PPD_ALL_CONSTRAINTS); + conflicts = cupsArrayCount(active); + + /* + * Loop through all of the UI constraints and flag any options + * that conflict... + */ + + for (c = (_ppd_cups_uiconsts_t *)cupsArrayFirst(active); + c; + c = (_ppd_cups_uiconsts_t *)cupsArrayNext(active)) + { + for (i = c->num_constraints, cptr = c->constraints; + i > 0; + i --, cptr ++) + cptr->option->conflicted = 1; + } + + cupsArrayDelete(active); + + /* + * Return the number of conflicts found... + */ + + return (conflicts); +} + + +/* + * 'ppdInstallableConflict()' - Test whether an option choice conflicts with + * an installable option. + * + * This function tests whether a particular option choice is available based + * on constraints against options in the "InstallableOptions" group. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +int /* O - 1 if conflicting, 0 if not conflicting */ +ppdInstallableConflict( + ppd_file_t *ppd, /* I - PPD file */ + const char *option, /* I - Option */ + const char *choice) /* I - Choice */ +{ + cups_array_t *active; /* Active conflicts */ + + + DEBUG_printf(("2ppdInstallableConflict(ppd=%p, option=\"%s\", choice=\"%s\")", + ppd, option, choice)); + + /* + * Range check input... + */ + + if (!ppd || !option || !choice) + return (0); + + /* + * Test constraints using the new option... + */ + + active = ppd_test_constraints(ppd, option, choice, 0, NULL, + _PPD_INSTALLABLE_CONSTRAINTS); + + cupsArrayDelete(active); + + return (active != NULL); +} + + +/* + * 'ppd_is_installable()' - Determine whether an option is in the + * InstallableOptions group. + */ + +static int /* O - 1 if installable, 0 if normal */ +ppd_is_installable( + ppd_group_t *installable, /* I - InstallableOptions group */ + const char *name) /* I - Option name */ +{ + if (installable) + { + int i; /* Looping var */ + ppd_option_t *option; /* Current option */ + + + for (i = installable->num_options, option = installable->options; + i > 0; + i --, option ++) + if (!_cups_strcasecmp(option->keyword, name)) + return (1); + } + + return (0); +} + + +/* + * 'ppd_load_constraints()' - Load constraints from a PPD file. + */ + +static void +ppd_load_constraints(ppd_file_t *ppd) /* I - PPD file */ +{ + int i; /* Looping var */ + ppd_const_t *oldconst; /* Current UIConstraints data */ + ppd_attr_t *constattr; /* Current cupsUIConstraints attribute */ + _ppd_cups_uiconsts_t *consts; /* Current cupsUIConstraints data */ + _ppd_cups_uiconst_t *constptr; /* Current constraint */ + ppd_group_t *installable; /* Installable options group */ + const char *vptr; /* Pointer into constraint value */ + char option[PPD_MAX_NAME], /* Option name/MainKeyword */ + choice[PPD_MAX_NAME], /* Choice/OptionKeyword */ + *ptr; /* Pointer into option or choice */ + + + DEBUG_printf(("7ppd_load_constraints(ppd=%p)", ppd)); + + /* + * Create an array to hold the constraint data... + */ + + ppd->cups_uiconstraints = cupsArrayNew(NULL, NULL); + + /* + * Find the installable options group if it exists... + */ + + for (i = ppd->num_groups, installable = ppd->groups; + i > 0; + i --, installable ++) + if (!_cups_strcasecmp(installable->name, "InstallableOptions")) + break; + + if (i <= 0) + installable = NULL; + + /* + * Load old-style [Non]UIConstraints data... + */ + + for (i = ppd->num_consts, oldconst = ppd->consts; i > 0; i --, oldconst ++) + { + /* + * Weed out nearby duplicates, since the PPD spec requires that you + * define both "*Foo foo *Bar bar" and "*Bar bar *Foo foo"... + */ + + if (i > 1 && + !_cups_strcasecmp(oldconst[0].option1, oldconst[1].option2) && + !_cups_strcasecmp(oldconst[0].choice1, oldconst[1].choice2) && + !_cups_strcasecmp(oldconst[0].option2, oldconst[1].option1) && + !_cups_strcasecmp(oldconst[0].choice2, oldconst[1].choice1)) + continue; + + /* + * Allocate memory... + */ + + if ((consts = calloc(1, sizeof(_ppd_cups_uiconsts_t))) == NULL) + { + DEBUG_puts("8ppd_load_constraints: Unable to allocate memory for " + "UIConstraints!"); + return; + } + + if ((constptr = calloc(2, sizeof(_ppd_cups_uiconst_t))) == NULL) + { + free(consts); + DEBUG_puts("8ppd_load_constraints: Unable to allocate memory for " + "UIConstraints!"); + return; + } + + /* + * Fill in the information... + */ + + consts->num_constraints = 2; + consts->constraints = constptr; + + if (!_cups_strncasecmp(oldconst->option1, "Custom", 6) && + !_cups_strcasecmp(oldconst->choice1, "True")) + { + constptr[0].option = ppdFindOption(ppd, oldconst->option1 + 6); + constptr[0].choice = ppdFindChoice(constptr[0].option, "Custom"); + constptr[0].installable = 0; + } + else + { + constptr[0].option = ppdFindOption(ppd, oldconst->option1); + constptr[0].choice = ppdFindChoice(constptr[0].option, + oldconst->choice1); + constptr[0].installable = ppd_is_installable(installable, + oldconst->option1); + } + + if (!constptr[0].option || (!constptr[0].choice && oldconst->choice1[0])) + { + DEBUG_printf(("8ppd_load_constraints: Unknown option *%s %s!", + oldconst->option1, oldconst->choice1)); + free(consts->constraints); + free(consts); + continue; + } + + if (!_cups_strncasecmp(oldconst->option2, "Custom", 6) && + !_cups_strcasecmp(oldconst->choice2, "True")) + { + constptr[1].option = ppdFindOption(ppd, oldconst->option2 + 6); + constptr[1].choice = ppdFindChoice(constptr[1].option, "Custom"); + constptr[1].installable = 0; + } + else + { + constptr[1].option = ppdFindOption(ppd, oldconst->option2); + constptr[1].choice = ppdFindChoice(constptr[1].option, + oldconst->choice2); + constptr[1].installable = ppd_is_installable(installable, + oldconst->option2); + } + + if (!constptr[1].option || (!constptr[1].choice && oldconst->choice2[0])) + { + DEBUG_printf(("8ppd_load_constraints: Unknown option *%s %s!", + oldconst->option2, oldconst->choice2)); + free(consts->constraints); + free(consts); + continue; + } + + consts->installable = constptr[0].installable || constptr[1].installable; + + /* + * Add it to the constraints array... + */ + + cupsArrayAdd(ppd->cups_uiconstraints, consts); + } + + /* + * Then load new-style constraints... + */ + + for (constattr = ppdFindAttr(ppd, "cupsUIConstraints", NULL); + constattr; + constattr = ppdFindNextAttr(ppd, "cupsUIConstraints", NULL)) + { + if (!constattr->value) + { + DEBUG_puts("8ppd_load_constraints: Bad cupsUIConstraints value!"); + continue; + } + + for (i = 0, vptr = strchr(constattr->value, '*'); + vptr; + i ++, vptr = strchr(vptr + 1, '*')); + + if (i == 0) + { + DEBUG_puts("8ppd_load_constraints: Bad cupsUIConstraints value!"); + continue; + } + + if ((consts = calloc(1, sizeof(_ppd_cups_uiconsts_t))) == NULL) + { + DEBUG_puts("8ppd_load_constraints: Unable to allocate memory for " + "cupsUIConstraints!"); + return; + } + + if ((constptr = calloc(i, sizeof(_ppd_cups_uiconst_t))) == NULL) + { + free(consts); + DEBUG_puts("8ppd_load_constraints: Unable to allocate memory for " + "cupsUIConstraints!"); + return; + } + + consts->num_constraints = i; + consts->constraints = constptr; + + strlcpy(consts->resolver, constattr->spec, sizeof(consts->resolver)); + + for (i = 0, vptr = strchr(constattr->value, '*'); + vptr; + i ++, vptr = strchr(vptr, '*'), constptr ++) + { + /* + * Extract "*Option Choice" or just "*Option"... + */ + + for (vptr ++, ptr = option; *vptr && !_cups_isspace(*vptr); vptr ++) + if (ptr < (option + sizeof(option) - 1)) + *ptr++ = *vptr; + + *ptr = '\0'; + + while (_cups_isspace(*vptr)) + vptr ++; + + if (*vptr == '*') + choice[0] = '\0'; + else + { + for (ptr = choice; *vptr && !_cups_isspace(*vptr); vptr ++) + if (ptr < (choice + sizeof(choice) - 1)) + *ptr++ = *vptr; + + *ptr = '\0'; + } + + if (!_cups_strncasecmp(option, "Custom", 6) && !_cups_strcasecmp(choice, "True")) + { + _cups_strcpy(option, option + 6); + strcpy(choice, "Custom"); + } + + constptr->option = ppdFindOption(ppd, option); + constptr->choice = ppdFindChoice(constptr->option, choice); + constptr->installable = ppd_is_installable(installable, option); + consts->installable |= constptr->installable; + + if (!constptr->option || (!constptr->choice && choice[0])) + { + DEBUG_printf(("8ppd_load_constraints: Unknown option *%s %s!", + option, choice)); + break; + } + } + + if (!vptr) + cupsArrayAdd(ppd->cups_uiconstraints, consts); + else + { + free(consts->constraints); + free(consts); + } + } +} + + +/* + * 'ppd_test_constraints()' - See if any constraints are active. + */ + +static cups_array_t * /* O - Array of active constraints */ +ppd_test_constraints( + ppd_file_t *ppd, /* I - PPD file */ + const char *option, /* I - Current option */ + const char *choice, /* I - Current choice */ + int num_options, /* I - Number of additional options */ + cups_option_t *options, /* I - Additional options */ + int which) /* I - Which constraints to test */ +{ + int i; /* Looping var */ + _ppd_cups_uiconsts_t *consts; /* Current constraints */ + _ppd_cups_uiconst_t *constptr; /* Current constraint */ + ppd_choice_t key, /* Search key */ + *marked; /* Marked choice */ + cups_array_t *active = NULL; /* Active constraints */ + const char *value, /* Current value */ + *firstvalue; /* AP_FIRSTPAGE_Keyword value */ + char firstpage[255]; /* AP_FIRSTPAGE_Keyword string */ + + + DEBUG_printf(("7ppd_test_constraints(ppd=%p, option=\"%s\", choice=\"%s\", " + "num_options=%d, options=%p, which=%d)", ppd, option, choice, + num_options, options, which)); + + if (!ppd->cups_uiconstraints) + ppd_load_constraints(ppd); + + DEBUG_printf(("9ppd_test_constraints: %d constraints!", + cupsArrayCount(ppd->cups_uiconstraints))); + + cupsArraySave(ppd->marked); + + for (consts = (_ppd_cups_uiconsts_t *)cupsArrayFirst(ppd->cups_uiconstraints); + consts; + consts = (_ppd_cups_uiconsts_t *)cupsArrayNext(ppd->cups_uiconstraints)) + { + DEBUG_printf(("9ppd_test_constraints: installable=%d, resolver=\"%s\", " + "num_constraints=%d option1=\"%s\", choice1=\"%s\", " + "option2=\"%s\", choice2=\"%s\", ...", + consts->installable, consts->resolver, consts->num_constraints, + consts->constraints[0].option->keyword, + consts->constraints[0].choice ? + consts->constraints[0].choice->choice : "", + consts->constraints[1].option->keyword, + consts->constraints[1].choice ? + consts->constraints[1].choice->choice : "")); + + if (consts->installable && which < _PPD_INSTALLABLE_CONSTRAINTS) + continue; /* Skip installable option constraint */ + + if (!consts->installable && which == _PPD_INSTALLABLE_CONSTRAINTS) + continue; /* Skip non-installable option constraint */ + + if (which == _PPD_OPTION_CONSTRAINTS && option) + { + /* + * Skip constraints that do not involve the current option... + */ + + for (i = consts->num_constraints, constptr = consts->constraints; + i > 0; + i --, constptr ++) + { + if (!_cups_strcasecmp(constptr->option->keyword, option)) + break; + + if (!_cups_strncasecmp(option, "AP_FIRSTPAGE_", 13) && + !_cups_strcasecmp(constptr->option->keyword, option + 13)) + break; + } + + if (!i) + continue; + } + + DEBUG_puts("9ppd_test_constraints: Testing..."); + + for (i = consts->num_constraints, constptr = consts->constraints; + i > 0; + i --, constptr ++) + { + DEBUG_printf(("9ppd_test_constraints: %s=%s?", constptr->option->keyword, + constptr->choice ? constptr->choice->choice : "")); + + if (constptr->choice && + (!_cups_strcasecmp(constptr->option->keyword, "PageSize") || + !_cups_strcasecmp(constptr->option->keyword, "PageRegion"))) + { + /* + * PageSize and PageRegion are used depending on the selected input slot + * and manual feed mode. Validate against the selected page size instead + * of an individual option... + */ + + if (option && choice && + (!_cups_strcasecmp(option, "PageSize") || + !_cups_strcasecmp(option, "PageRegion"))) + { + value = choice; + } + else if ((value = cupsGetOption("PageSize", num_options, + options)) == NULL) + if ((value = cupsGetOption("PageRegion", num_options, + options)) == NULL) + if ((value = cupsGetOption("media", num_options, options)) == NULL) + { + ppd_size_t *size = ppdPageSize(ppd, NULL); + + if (size) + value = size->name; + } + + if (value && !_cups_strncasecmp(value, "Custom.", 7)) + value = "Custom"; + + if (option && choice && + (!_cups_strcasecmp(option, "AP_FIRSTPAGE_PageSize") || + !_cups_strcasecmp(option, "AP_FIRSTPAGE_PageRegion"))) + { + firstvalue = choice; + } + else if ((firstvalue = cupsGetOption("AP_FIRSTPAGE_PageSize", + num_options, options)) == NULL) + firstvalue = cupsGetOption("AP_FIRSTPAGE_PageRegion", num_options, + options); + + if (firstvalue && !_cups_strncasecmp(firstvalue, "Custom.", 7)) + firstvalue = "Custom"; + + if ((!value || _cups_strcasecmp(value, constptr->choice->choice)) && + (!firstvalue || _cups_strcasecmp(firstvalue, constptr->choice->choice))) + { + DEBUG_puts("9ppd_test_constraints: NO"); + break; + } + } + else if (constptr->choice) + { + /* + * Compare against the constrained choice... + */ + + if (option && choice && !_cups_strcasecmp(option, constptr->option->keyword)) + { + if (!_cups_strncasecmp(choice, "Custom.", 7)) + value = "Custom"; + else + value = choice; + } + else if ((value = cupsGetOption(constptr->option->keyword, num_options, + options)) != NULL) + { + if (!_cups_strncasecmp(value, "Custom.", 7)) + value = "Custom"; + } + else if (constptr->choice->marked) + value = constptr->choice->choice; + else + value = NULL; + + /* + * Now check AP_FIRSTPAGE_option... + */ + + snprintf(firstpage, sizeof(firstpage), "AP_FIRSTPAGE_%s", + constptr->option->keyword); + + if (option && choice && !_cups_strcasecmp(option, firstpage)) + { + if (!_cups_strncasecmp(choice, "Custom.", 7)) + firstvalue = "Custom"; + else + firstvalue = choice; + } + else if ((firstvalue = cupsGetOption(firstpage, num_options, + options)) != NULL) + { + if (!_cups_strncasecmp(firstvalue, "Custom.", 7)) + firstvalue = "Custom"; + } + else + firstvalue = NULL; + + DEBUG_printf(("9ppd_test_constraints: value=%s, firstvalue=%s", value, + firstvalue)); + + if ((!value || _cups_strcasecmp(value, constptr->choice->choice)) && + (!firstvalue || _cups_strcasecmp(firstvalue, constptr->choice->choice))) + { + DEBUG_puts("9ppd_test_constraints: NO"); + break; + } + } + else if (option && choice && + !_cups_strcasecmp(option, constptr->option->keyword)) + { + if (!_cups_strcasecmp(choice, "None") || !_cups_strcasecmp(choice, "Off") || + !_cups_strcasecmp(choice, "False")) + { + DEBUG_puts("9ppd_test_constraints: NO"); + break; + } + } + else if ((value = cupsGetOption(constptr->option->keyword, num_options, + options)) != NULL) + { + if (!_cups_strcasecmp(value, "None") || !_cups_strcasecmp(value, "Off") || + !_cups_strcasecmp(value, "False")) + { + DEBUG_puts("9ppd_test_constraints: NO"); + break; + } + } + else + { + key.option = constptr->option; + + if ((marked = (ppd_choice_t *)cupsArrayFind(ppd->marked, &key)) + == NULL || + (!_cups_strcasecmp(marked->choice, "None") || + !_cups_strcasecmp(marked->choice, "Off") || + !_cups_strcasecmp(marked->choice, "False"))) + { + DEBUG_puts("9ppd_test_constraints: NO"); + break; + } + } + } + + if (i <= 0) + { + if (!active) + active = cupsArrayNew(NULL, NULL); + + cupsArrayAdd(active, consts); + DEBUG_puts("9ppd_test_constraints: Added..."); + } + } + + cupsArrayRestore(ppd->marked); + + DEBUG_printf(("8ppd_test_constraints: Found %d active constraints!", + cupsArrayCount(active))); + + return (active); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/cups-private.h b/cups/cups-private.h new file mode 100644 index 0000000000..1950a421f3 --- /dev/null +++ b/cups/cups-private.h @@ -0,0 +1,262 @@ +/* + * "$Id$" + * + * Private definitions for CUPS. + * + * Copyright 2007-2012 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_CUPS_PRIVATE_H_ +# define _CUPS_CUPS_PRIVATE_H_ + +/* + * Include necessary headers... + */ + +# include +# include "string-private.h" +# include "debug-private.h" +# include "ppd-private.h" +# include "http-private.h" +# include "ipp-private.h" +# include "language-private.h" +# include "pwg-private.h" +# include "thread-private.h" +# ifdef __APPLE__ +# include +# include +# endif /* __APPLE__ */ + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Types... + */ + +typedef struct _cups_buffer_s /**** Read/write buffer ****/ +{ + struct _cups_buffer_s *next; /* Next buffer in list */ + size_t size; /* Size of buffer */ + char used, /* Is this buffer used? */ + d[1]; /* Data buffer */ +} _cups_buffer_t; + +typedef struct _cups_globals_s /**** CUPS global state data ****/ +{ + /* Multiple places... */ + const char *cups_datadir, /* CUPS_DATADIR environment var */ + *cups_serverbin,/* CUPS_SERVERBIN environment var */ + *cups_serverroot, + /* CUPS_SERVERROOT environment var */ + *cups_statedir, /* CUPS_STATEDIR environment var */ + *localedir; /* LOCALDIR environment var */ + + /* adminutil.c */ + time_t cupsd_update; /* Last time we got or set cupsd.conf */ + char cupsd_hostname[HTTP_MAX_HOST]; + /* Hostname for connection */ + int cupsd_num_settings; + /* Number of server settings */ + cups_option_t *cupsd_settings;/* Server settings */ + + /* auth.c */ +# ifdef HAVE_GSSAPI + char gss_service_name[32]; + /* Kerberos service name */ +# endif /* HAVE_GSSAPI */ + + /* backend.c */ + char resolved_uri[1024]; + /* Buffer for cupsBackendDeviceURI */ + + /* file.c */ + cups_file_t *stdio_files[3];/* stdin, stdout, stderr */ + + /* http.c */ + char http_date[256]; /* Date+time buffer */ + + /* http-addr.c */ + unsigned ip_addr; /* Packed IPv4 address */ + char *ip_ptrs[2]; /* Pointer to packed address */ + struct hostent hostent; /* Host entry for IP address */ +# ifdef HAVE_GETADDRINFO + char hostname[1024]; /* Hostname */ +# endif /* HAVE_GETADDRINFO */ + int need_res_init; /* Need to reinitialize resolver? */ + + /* ipp.c */ + ipp_uchar_t ipp_date[11]; /* RFC-1903 date/time data */ + _cups_buffer_t *cups_buffers; /* Buffer list */ + + /* ipp-support.c */ + int ipp_port; /* IPP port number */ + char ipp_unknown[255]; + /* Unknown error statuses */ + + /* language.c */ + cups_lang_t *lang_default; /* Default language */ +# ifdef __APPLE__ + char language[32]; /* Cached language */ +# endif /* __APPLE__ */ + + /* ppd.c */ + ppd_status_t ppd_status; /* Status of last ppdOpen*() */ + int ppd_line; /* Current line number */ + ppd_conform_t ppd_conform; /* Level of conformance required */ + + /* pwg-media.c */ + cups_array_t *leg_size_lut, /* Lookup table for legacy names */ + *ppd_size_lut, /* Lookup table for PPD names */ + *pwg_size_lut; /* Lookup table for PWG names */ + _pwg_media_t pwg_media; /* PWG media data for custom size */ + char pwg_name[65]; /* PWG media name for custom size */ + + /* request.c */ + http_t *http; /* Current server connection */ + ipp_status_t last_error; /* Last IPP error */ + char *last_status_message; + /* Last IPP status-message */ + + /* snmp.c */ + char snmp_community[255]; + /* Default SNMP community name */ + int snmp_debug; /* Log SNMP IO to stderr? */ + + /* tempfile.c */ + char tempfile[1024]; /* cupsTempFd/File buffer */ + + /* usersys.c */ + http_encryption_t encryption; /* Encryption setting */ + char user[65], /* User name */ + server[256], /* Server address */ + servername[256],/* Server hostname */ + password[128]; /* Password for default callback */ + cups_password_cb2_t password_cb; /* Password callback */ + void *password_data; /* Password user data */ + http_tls_credentials_t tls_credentials; + /* Default client credentials */ + cups_client_cert_cb_t client_cert_cb; /* Client certificate callback */ + void *client_cert_data; + /* Client certificate user data */ + cups_server_cert_cb_t server_cert_cb; /* Server certificate callback */ + void *server_cert_data; + /* Server certificate user data */ + int any_root, /* Allow any root */ + expired_certs, /* Allow expired certs */ + expired_root; /* Allow expired root */ + + /* util.c */ + char def_printer[256]; + /* Default printer */ + char ppd_filename[HTTP_MAX_URI]; + /* PPD filename */ +} _cups_globals_t; + +typedef struct _cups_media_db_s /* Media database */ +{ + char *color, /* Media color, if any */ + *key, /* Media key, if any */ + *info, /* Media human-readable name, if any */ + *size_name, /* Media PWG size name, if provided */ + *source, /* Media source, if any */ + *type; /* Media type, if any */ + int width, /* Width in hundredths of millimeters */ + length, /* Length in hundredths of + * millimeters */ + bottom, /* Bottom margin in hundredths of + * millimeters */ + left, /* Left margin in hundredths of + * millimeters */ + right, /* Right margin in hundredths of + * millimeters */ + top; /* Top margin in hundredths of + * millimeters */ +} _cups_media_db_t; + +struct _cups_dinfo_s /* Destination capability and status + * information */ +{ + const char *uri; /* Printer URI */ + char *resource; /* Resource path */ + ipp_t *attrs; /* Printer attributes */ + cups_array_t *constraints; /* Job constraints */ + cups_array_t *localizations; /* Localization information */ + cups_array_t *media_db; /* Media database */ + _cups_media_db_t min_size, /* Minimum size */ + max_size; /* Maximum size */ +}; + + +/* + * Prototypes... + */ + +# ifdef __APPLE__ +extern CFStringRef _cupsAppleCopyDefaultPaperID(void); +extern CFStringRef _cupsAppleCopyDefaultPrinter(void); +extern int _cupsAppleGetUseLastPrinter(void); +extern void _cupsAppleSetDefaultPaperID(CFStringRef name); +extern void _cupsAppleSetDefaultPrinter(CFStringRef name); +extern void _cupsAppleSetUseLastPrinter(int uselast); +# endif /* __APPLE__ */ + +extern char *_cupsBufferGet(size_t size); +extern void _cupsBufferRelease(char *b); + +extern http_t *_cupsConnect(void); +extern int _cupsGet1284Values(const char *device_id, + cups_option_t **values); +extern const char *_cupsGetDestResource(cups_dest_t *dest, char *resource, + size_t resourcesize); +extern int _cupsGetDests(http_t *http, ipp_op_t op, + const char *name, cups_dest_t **dests, + cups_ptype_t type, cups_ptype_t mask); +extern const char *_cupsGetPassword(const char *prompt); +extern void _cupsGlobalLock(void); +extern _cups_globals_t *_cupsGlobals(void); +extern void _cupsGlobalUnlock(void); +# ifdef HAVE_GSSAPI +extern const char *_cupsGSSServiceName(void); +# endif /* HAVE_GSSAPI */ +extern int _cupsNextDelay(int current, int *previous); +extern void _cupsSetDefaults(void); +extern void _cupsSetError(ipp_status_t status, const char *message, + int localize); +extern void _cupsSetHTTPError(http_status_t status); +# ifdef HAVE_GSSAPI +extern int _cupsSetNegotiateAuthString(http_t *http, + const char *method, + const char *resource); +# endif /* HAVE_GSSAPI */ +extern char *_cupsUserDefault(char *name, size_t namesize); + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +} +# endif /* __cplusplus */ +#endif /* !_CUPS_CUPS_PRIVATE_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/cups.h b/cups/cups.h new file mode 100644 index 0000000000..7a33411f4d --- /dev/null +++ b/cups/cups.h @@ -0,0 +1,598 @@ +/* + * "$Id$" + * + * API definitions for CUPS. + * + * Copyright 2007-2012 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_CUPS_H_ +# define _CUPS_CUPS_H_ + +/* + * Include necessary headers... + */ + +# include +# if defined(WIN32) && !defined(__CUPS_SSIZE_T_DEFINED) +# define __CUPS_SSIZE_T_DEFINED +# include +/* Windows does not support the ssize_t type, so map it to off_t... */ +typedef off_t ssize_t; /* @private@ */ +# endif /* WIN32 && !__CUPS_SSIZE_T_DEFINED */ + +# ifdef __BLOCKS__ +# include +# endif /* __BLOCKS__ */ + +# include "file.h" +# include "ipp.h" +# include "language.h" + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Constants... + */ + +# define CUPS_VERSION 1.0599 +# define CUPS_VERSION_MAJOR 1 +# define CUPS_VERSION_MINOR 6 +# define CUPS_VERSION_PATCH -1 + +# define CUPS_BC_FD 3 + /* Back-channel file descriptor for + * select/poll */ +# define CUPS_DATE_ANY (time_t)-1 +# define CUPS_EXCLUDE_NONE (const char *)0 +# define CUPS_FORMAT_AUTO "application/octet-stream" +# define CUPS_FORMAT_COMMAND "application/vnd.cups-command" +# define CUPS_FORMAT_JPEG "image/jpeg" +# define CUPS_FORMAT_PDF "application/pdf" +# define CUPS_FORMAT_POSTSCRIPT "application/postscript" +# define CUPS_FORMAT_RAW "application/vnd.cups-raw" +# define CUPS_FORMAT_TEXT "text/plain" +# define CUPS_HTTP_DEFAULT (http_t *)0 +# define CUPS_INCLUDE_ALL (const char *)0 +# define CUPS_JOBID_ALL -1 +# define CUPS_JOBID_CURRENT 0 +# define CUPS_LENGTH_VARIABLE (ssize_t)0 +# define CUPS_TIMEOUT_DEFAULT 0 +# define CUPS_WHICHJOBS_ALL -1 +# define CUPS_WHICHJOBS_ACTIVE 0 +# define CUPS_WHICHJOBS_COMPLETED 1 + +/* Flags for cupsConnectDest and cupsEnumDests */ +# define CUPS_DEST_FLAGS_NONE 0x00 + /* No flags are set */ +# define CUPS_DEST_FLAGS_UNCONNECTED 0x01 + /* There is not connection */ +# define CUPS_DEST_FLAGS_MORE 0x02 + /* There are more destinations */ +# define CUPS_DEST_FLAGS_REMOVED 0x04 + /* The destination has gone away */ +# define CUPS_DEST_FLAGS_ERROR 0x08 + /* An error occurred */ +# define CUPS_DEST_FLAGS_RESOLVING 0x10 + /* The destination address is being + * resolved */ +# define CUPS_DEST_FLAGS_CONNECTING 0x20 + /* A connection is being established */ +# define CUPS_DEST_FLAGS_CANCELED 0x40 + /* Operation was canceled */ + +/* Flags for cupsGetDestMediaByName/Size */ +# define CUPS_MEDIA_FLAGS_DEFAULT 0x00 + /* Find the closest size supported by + * the printer */ +# define CUPS_MEDIA_FLAGS_BORDERLESS 0x01 + /* Find a borderless size */ +# define CUPS_MEDIA_FLAGS_DUPLEX 0x02 + /* Find a size compatible with 2-sided + * printing */ +# define CUPS_MEDIA_FLAGS_EXACT 0x04 + /* Find an exact match for the size */ +# define CUPS_MEDIA_FLAGS_READY 0x08 + /* If the printer supports media + * sensing, find the size amongst the + * "ready" media. */ + +/* Options and values */ +# define CUPS_COPIES "copies" +# define CUPS_COPIES_SUPPORTED "copies-supported" + +# define CUPS_FINISHINGS "finishings" +# define CUPS_FINISHINGS_SUPPORTED "finishings-supported" + +# define CUPS_FINISHINGS_BIND "7" +# define CUPS_FINISHINGS_COVER "6" +# define CUPS_FINISHINGS_FOLD "10" +# define CUPS_FINISHINGS_NONE "3" +# define CUPS_FINISHINGS_PUNCH "5" +# define CUPS_FINISHINGS_STAPLE "4" +# define CUPS_FINISHINGS_TRIM "11" + +# define CUPS_MEDIA "media" +# define CUPS_MEDIA_READY "media-ready" +# define CUPS_MEDIA_SUPPORTED "media-supported" + +# define CUPS_MEDIA_3X5 "na_index-3x5_3x5in" +# define CUPS_MEDIA_4X6 "na_index-4x6_4x6in" +# define CUPS_MEDIA_5X7 "na_5x7_5x7in" +# define CUPS_MEDIA_8X10 "na_govt-letter_8x10in" +# define CUPS_MEDIA_A3 "iso_a3_297x420mm" +# define CUPS_MEDIA_A4 "iso_a4_210x297mm" +# define CUPS_MEDIA_A5 "iso_a5_148x210mm" +# define CUPS_MEDIA_A6 "iso_a6_105x148mm" +# define CUPS_MEDIA_ENV10 "na_number-10_4.125x9.5in" +# define CUPS_MEDIA_ENVDL "iso_dl_110x220mm" +# define CUPS_MEDIA_LEGAL "na_legal_8.5x14in" +# define CUPS_MEDIA_LETTER "na_letter_8.5x11in" +# define CUPS_MEDIA_PHOTO_L "oe_photo-l_3.5x5in" +# define CUPS_MEDIA_SUPERBA3 "na_super-b_13x19in" +# define CUPS_MEDIA_TABLOID "na_ledger_11x17in" + +# define CUPS_MEDIA_SOURCE "media-source" +# define CUPS_MEDIA_SOURCE_SUPPORTED "media-source-supported" + +# define CUPS_MEDIA_SOURCE_AUTO "auto" +# define CUPS_MEDIA_SOURCE_MANUAL "manual" + +# define CUPS_MEDIA_TYPE "media-type" +# define CUPS_MEDIA_TYPE_SUPPORTED "media-type-supported" + +# define CUPS_MEDIA_TYPE_AUTO "auto" +# define CUPS_MEDIA_TYPE_ENVELOPE "envelope" +# define CUPS_MEDIA_TYPE_LABELS "labels" +# define CUPS_MEDIA_TYPE_LETTERHEAD "stationery-letterhead" +# define CUPS_MEDIA_TYPE_PHOTO "photographic" +# define CUPS_MEDIA_TYPE_PHOTO_GLOSSY "photographic-glossy" +# define CUPS_MEDIA_TYPE_PHOTO_MATTE "photographic-matte" +# define CUPS_MEDIA_TYPE_PLAIN "stationery" +# define CUPS_MEDIA_TYPE_TRANSPARENCY "transparency" + +# define CUPS_NUMBER_UP "number-up" +# define CUPS_NUMBER_UP_SUPPORTED "number-up-supported" + +# define CUPS_ORIENTATION "orientation-requested" +# define CUPS_ORIENTATION_SUPPORTED "orientation-requested-supported" + +# define CUPS_ORIENTATION_PORTRAIT "3" +# define CUPS_ORIENTATION_LANDSCAPE "4" + +# define CUPS_PRINT_COLOR_MODE "print-color-mode" +# define CUPS_PRINT_COLOR_MODE_SUPPORTED "print-color-mode-supported" + +# define CUPS_PRINT_COLOR_MODE_AUTO "auto" +# define CUPS_PRINT_COLOR_MODE_MONOCHROME "monochrome" +# define CUPS_PRINT_COLOR_MODE_COLOR "color" + +# define CUPS_PRINT_QUALITY "print-quality" +# define CUPS_PRINT_QUALITY_SUPPORTED "print-quality-supported" + +# define CUPS_PRINT_QUALITY_DRAFT "3" +# define CUPS_PRINT_QUALITY_NORMAL "4" +# define CUPS_PRINT_QUALITY_HIGH "5" + +# define CUPS_SIDES "sides" +# define CUPS_SIDES_SUPPORTED "sides-supported" + +# define CUPS_SIDES_ONE_SIDED "one-sided" +# define CUPS_SIDES_TWO_SIDED_PORTRAIT "two-sided-long-edge" +# define CUPS_SIDES_TWO_SIDED_LANDSCAPE "two-sided-short-edge" + + +/* + * Types and structures... + */ + +typedef unsigned cups_ptype_t; /* Printer type/capability bits */ +enum cups_ptype_e /* Printer type/capability bit + * constants */ +{ /* Not a typedef'd enum so we can OR */ + CUPS_PRINTER_LOCAL = 0x0000, /* Local printer or class */ + CUPS_PRINTER_CLASS = 0x0001, /* Printer class */ + CUPS_PRINTER_REMOTE = 0x0002, /* Remote printer or class */ + CUPS_PRINTER_BW = 0x0004, /* Can do B&W printing */ + CUPS_PRINTER_COLOR = 0x0008, /* Can do color printing */ + CUPS_PRINTER_DUPLEX = 0x0010, /* Can do duplexing */ + CUPS_PRINTER_STAPLE = 0x0020, /* Can staple output */ + CUPS_PRINTER_COPIES = 0x0040, /* Can do copies */ + CUPS_PRINTER_COLLATE = 0x0080, /* Can collage copies */ + CUPS_PRINTER_PUNCH = 0x0100, /* Can punch output */ + CUPS_PRINTER_COVER = 0x0200, /* Can cover output */ + CUPS_PRINTER_BIND = 0x0400, /* Can bind output */ + CUPS_PRINTER_SORT = 0x0800, /* Can sort output */ + CUPS_PRINTER_SMALL = 0x1000, /* Can do Letter/Legal/A4 */ + CUPS_PRINTER_MEDIUM = 0x2000, /* Can do Tabloid/B/C/A3/A2 */ + CUPS_PRINTER_LARGE = 0x4000, /* Can do D/E/A1/A0 */ + CUPS_PRINTER_VARIABLE = 0x8000, /* Can do variable sizes */ + CUPS_PRINTER_IMPLICIT = 0x10000, /* Implicit class @private@ + * @since Deprecated@ */ + CUPS_PRINTER_DEFAULT = 0x20000, /* Default printer on network */ + CUPS_PRINTER_FAX = 0x40000, /* Fax queue */ + CUPS_PRINTER_REJECTING = 0x80000, /* Printer is rejecting jobs */ + CUPS_PRINTER_DELETE = 0x100000, /* Delete printer + * @since CUPS 1.2/Mac OS X 10.5@ */ + CUPS_PRINTER_NOT_SHARED = 0x200000, /* Printer is not shared + * @since CUPS 1.2/Mac OS X 10.5@ */ + CUPS_PRINTER_AUTHENTICATED = 0x400000,/* Printer requires authentication + * @since CUPS 1.2/Mac OS X 10.5@ */ + CUPS_PRINTER_COMMANDS = 0x800000, /* Printer supports maintenance commands + * @since CUPS 1.2/Mac OS X 10.5@ */ + CUPS_PRINTER_DISCOVERED = 0x1000000, /* Printer was automatically discovered + * and added @private@ + * @since Deprecated@ */ + CUPS_PRINTER_SCANNER = 0x2000000, /* Scanner-only device + * @since CUPS 1.4/Mac OS X 10.6@ */ + CUPS_PRINTER_MFP = 0x4000000, /* Printer with scanning capabilities + * @since CUPS 1.4/Mac OS X 10.6@ */ + CUPS_PRINTER_OPTIONS = 0x6fffc /* ~(CLASS | REMOTE | IMPLICIT | + * DEFAULT | FAX | REJECTING | DELETE | + * NOT_SHARED | AUTHENTICATED | + * COMMANDS | DISCOVERED) @private@ */ +}; + +typedef struct cups_option_s /**** Printer Options ****/ +{ + char *name; /* Name of option */ + char *value; /* Value of option */ +} cups_option_t; + +typedef struct cups_dest_s /**** Destination ****/ +{ + char *name, /* Printer or class name */ + *instance; /* Local instance name or NULL */ + int is_default; /* Is this printer the default? */ + int num_options; /* Number of options */ + cups_option_t *options; /* Options */ +} cups_dest_t; + +typedef struct _cups_dinfo_s cups_dinfo_t; + /* Destination capability and status + * information @since CUPS 1.6@ */ + +typedef struct cups_job_s /**** Job ****/ +{ + int id; /* The job ID */ + char *dest; /* Printer or class name */ + char *title; /* Title/job name */ + char *user; /* User the submitted the job */ + char *format; /* Document format */ + ipp_jstate_t state; /* Job state */ + int size; /* Size in kilobytes */ + int priority; /* Priority (1-100) */ + time_t completed_time; /* Time the job was completed */ + time_t creation_time; /* Time the job was created */ + time_t processing_time; /* Time the job was processed */ +} cups_job_t; + +typedef struct cups_size_s /**** Media Size @since CUPS 1.6@ ****/ +{ + char media[128]; /* Media name to use */ + int width, /* Width in hundredths of millimeters */ + length, /* Length in hundredths of + * millimeters */ + bottom, /* Bottom margin in hundredths of + * millimeters */ + left, /* Left margin in hundredths of + * millimeters */ + right, /* Right margin in hundredths of + * millimeters */ + top; /* Top margin in hundredths of + * millimeters */ +} cups_size_t; + +typedef int (*cups_client_cert_cb_t)(http_t *http, void *tls, + cups_array_t *distinguished_names, + void *user_data); + /* Client credentials callback + * @since CUPS 1.5/Mac OS X 10.7@ */ + +typedef int (*cups_dest_cb_t)(void *user_data, unsigned flags, + cups_dest_t *dest); + /* Destination enumeration callback + * @since CUPS 1.6@ */ + +# ifdef __BLOCKS__ +typedef int (^cups_dest_block_t)(unsigned flags, cups_dest_t *dest); + /* Destination enumeration block + * @since CUPS 1.6@ */ +# endif /* __BLOCKS__ */ + +typedef void (*cups_device_cb_t)(const char *device_class, + const char *device_id, const char *device_info, + const char *device_make_and_model, + const char *device_uri, + const char *device_location, void *user_data); + /* Device callback + * @since CUPS 1.4/Mac OS X 10.6@ */ + +typedef const char *(*cups_password_cb_t)(const char *prompt); + /* Password callback */ + +typedef const char *(*cups_password_cb2_t)(const char *prompt, http_t *http, + const char *method, + const char *resource, + void *user_data); + /* New password callback + * @since CUPS 1.4/Mac OS X 10.6@ */ + +typedef int (*cups_server_cert_cb_t)(http_t *http, void *tls, + cups_array_t *certs, void *user_data); + /* Server credentials callback + * @since CUPS 1.5/Mac OS X 10.7@ */ + + +/* + * Functions... + */ + +extern int cupsCancelJob(const char *name, int job_id); +extern ipp_t *cupsDoFileRequest(http_t *http, ipp_t *request, + const char *resource, + const char *filename); +extern ipp_t *cupsDoRequest(http_t *http, ipp_t *request, + const char *resource); +extern http_encryption_t cupsEncryption(void); +extern void cupsFreeJobs(int num_jobs, cups_job_t *jobs); +extern int cupsGetClasses(char ***classes) _CUPS_DEPRECATED; +extern const char *cupsGetDefault(void); +extern int cupsGetJobs(cups_job_t **jobs, const char *name, + int myjobs, int whichjobs); +extern const char *cupsGetPPD(const char *name); +extern int cupsGetPrinters(char ***printers) _CUPS_DEPRECATED; +extern ipp_status_t cupsLastError(void); +extern int cupsPrintFile(const char *name, const char *filename, + const char *title, int num_options, + cups_option_t *options); +extern int cupsPrintFiles(const char *name, int num_files, + const char **files, const char *title, + int num_options, cups_option_t *options); +extern char *cupsTempFile(char *filename, int len) _CUPS_DEPRECATED; +extern int cupsTempFd(char *filename, int len); + +extern int cupsAddDest(const char *name, const char *instance, + int num_dests, cups_dest_t **dests); +extern void cupsFreeDests(int num_dests, cups_dest_t *dests); +extern cups_dest_t *cupsGetDest(const char *name, const char *instance, + int num_dests, cups_dest_t *dests); +extern int cupsGetDests(cups_dest_t **dests); +extern void cupsSetDests(int num_dests, cups_dest_t *dests); + +extern int cupsAddOption(const char *name, const char *value, + int num_options, cups_option_t **options); +extern void cupsEncodeOptions(ipp_t *ipp, int num_options, + cups_option_t *options); +extern void cupsFreeOptions(int num_options, + cups_option_t *options); +extern const char *cupsGetOption(const char *name, int num_options, + cups_option_t *options); +extern int cupsParseOptions(const char *arg, int num_options, + cups_option_t **options); + +extern const char *cupsGetPassword(const char *prompt); +extern const char *cupsServer(void); +extern void cupsSetEncryption(http_encryption_t e); +extern void cupsSetPasswordCB(cups_password_cb_t cb); +extern void cupsSetServer(const char *server); +extern void cupsSetUser(const char *user); +extern const char *cupsUser(void); + +/**** New in CUPS 1.1.20 ****/ +extern int cupsDoAuthentication(http_t *http, const char *method, + const char *resource) + _CUPS_API_1_1_20; +extern http_status_t cupsGetFile(http_t *http, const char *resource, + const char *filename) _CUPS_API_1_1_20; +extern http_status_t cupsGetFd(http_t *http, const char *resource, int fd); +extern http_status_t cupsPutFile(http_t *http, const char *resource, + const char *filename) _CUPS_API_1_1_20; +extern http_status_t cupsPutFd(http_t *http, const char *resource, int fd) + _CUPS_API_1_1_20; + +/**** New in CUPS 1.1.21 ****/ +extern const char *cupsGetDefault2(http_t *http) _CUPS_API_1_1_21; +extern int cupsGetDests2(http_t *http, cups_dest_t **dests) + _CUPS_API_1_1_21; +extern int cupsGetJobs2(http_t *http, cups_job_t **jobs, + const char *name, int myjobs, + int whichjobs) _CUPS_API_1_1_21; +extern const char *cupsGetPPD2(http_t *http, const char *name) + _CUPS_API_1_1_21; +extern int cupsPrintFile2(http_t *http, const char *name, + const char *filename, + const char *title, int num_options, + cups_option_t *options) _CUPS_API_1_1_21; +extern int cupsPrintFiles2(http_t *http, const char *name, + int num_files, const char **files, + const char *title, int num_options, + cups_option_t *options) + _CUPS_API_1_1_21; +extern int cupsSetDests2(http_t *http, int num_dests, + cups_dest_t *dests) _CUPS_API_1_1_21; + +/**** New in CUPS 1.2/Mac OS X 10.5 ****/ +extern ssize_t cupsBackChannelRead(char *buffer, size_t bytes, + double timeout) _CUPS_API_1_2; +extern ssize_t cupsBackChannelWrite(const char *buffer, size_t bytes, + double timeout) _CUPS_API_1_2; +extern void cupsEncodeOptions2(ipp_t *ipp, int num_options, + cups_option_t *options, + ipp_tag_t group_tag) _CUPS_API_1_2; +extern const char *cupsLastErrorString(void) _CUPS_API_1_2; +extern char *cupsNotifySubject(cups_lang_t *lang, ipp_t *event) + _CUPS_API_1_2; +extern char *cupsNotifyText(cups_lang_t *lang, ipp_t *event) + _CUPS_API_1_2; +extern int cupsRemoveOption(const char *name, int num_options, + cups_option_t **options) _CUPS_API_1_2; +extern cups_file_t *cupsTempFile2(char *filename, int len) _CUPS_API_1_2; + +/**** New in CUPS 1.3/Mac OS X 10.5 ****/ +extern ipp_t *cupsDoIORequest(http_t *http, ipp_t *request, + const char *resource, int infile, + int outfile) _CUPS_API_1_3; +extern char *cupsGetServerPPD(http_t *http, const char *name) + _CUPS_API_1_3; +extern int cupsRemoveDest(const char *name, + const char *instance, + int num_dests, cups_dest_t **dests) + _CUPS_API_1_3; +extern void cupsSetDefaultDest(const char *name, + const char *instance, + int num_dests, + cups_dest_t *dests) _CUPS_API_1_3; + +/**** New in CUPS 1.4/Mac OS X 10.6 ****/ +extern ipp_status_t cupsCancelJob2(http_t *http, const char *name, + int job_id, int purge) _CUPS_API_1_4; +extern int cupsCreateJob(http_t *http, const char *name, + const char *title, int num_options, + cups_option_t *options) _CUPS_API_1_4; +extern ipp_status_t cupsFinishDocument(http_t *http, + const char *name) _CUPS_API_1_4; +extern ipp_status_t cupsGetDevices(http_t *http, int timeout, + const char *include_schemes, + const char *exclude_schemes, + cups_device_cb_t callback, + void *user_data) _CUPS_API_1_4; +extern cups_dest_t *cupsGetNamedDest(http_t *http, const char *name, + const char *instance) _CUPS_API_1_4; +extern const char *cupsGetPassword2(const char *prompt, http_t *http, + const char *method, + const char *resource) _CUPS_API_1_4; +extern http_status_t cupsGetPPD3(http_t *http, const char *name, + time_t *modtime, char *buffer, + size_t bufsize) _CUPS_API_1_4; +extern ipp_t *cupsGetResponse(http_t *http, + const char *resource) _CUPS_API_1_4; +extern ssize_t cupsReadResponseData(http_t *http, char *buffer, + size_t length) _CUPS_API_1_4; +extern http_status_t cupsSendRequest(http_t *http, ipp_t *request, + const char *resource, + size_t length) _CUPS_API_1_4; +extern void cupsSetPasswordCB2(cups_password_cb2_t cb, + void *user_data) _CUPS_API_1_4; +extern http_status_t cupsStartDocument(http_t *http, const char *name, + int job_id, const char *docname, + const char *format, + int last_document) _CUPS_API_1_4; +extern http_status_t cupsWriteRequestData(http_t *http, const char *buffer, + size_t length) _CUPS_API_1_4; + +/**** New in CUPS 1.5 ****/ +extern void cupsSetClientCertCB(cups_client_cert_cb_t cb, + void *user_data) _CUPS_API_1_5; +extern int cupsSetCredentials(cups_array_t *certs) _CUPS_API_1_5; +extern void cupsSetServerCertCB(cups_server_cert_cb_t cb, + void *user_data) _CUPS_API_1_5; + +/**** New in CUPS 1.6 ****/ +extern ipp_status_t cupsCancelDestJob(http_t *http, cups_dest_t *dest, + int job_id) _CUPS_API_1_6; +extern int cupsCheckDestSupported(http_t *http, cups_dest_t *dest, + cups_dinfo_t *info, + const char *option, + const char *value) _CUPS_API_1_6; +extern ipp_status_t cupsCloseDestJob(http_t *http, cups_dest_t *dest, + int job_id) _CUPS_API_1_6; +extern http_t *cupsConnectDest(cups_dest_t *dest, unsigned flags, + int msec, int *cancel, + char *resource, size_t resourcesize, + cups_dest_cb_t cb, void *user_data) + _CUPS_API_1_6; +# ifdef __BLOCKS__ +extern http_t *cupsConnectDestBlock(cups_dest_t *dest, + unsigned flags, int msec, + int *cancel, char *resource, + size_t resourcesize, + cups_dest_block_t block) + _CUPS_API_1_6; +# endif /* __BLOCKS__ */ +extern int cupsCopyDest(cups_dest_t *dest, int num_dests, + cups_dest_t **dests) _CUPS_API_1_6; +extern cups_dinfo_t *cupsCopyDestInfo(http_t *http, cups_dest_t *dest) + _CUPS_API_1_6; +extern int cupsCopyDestConflicts(http_t *http, cups_dest_t *dest, + cups_dinfo_t *info, + 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) + _CUPS_API_1_6; +extern ipp_status_t cupsCreateDestJob(http_t *http, cups_dest_t *dest, + cups_dinfo_t *info, int *job_id, + const char *title, int num_options, + cups_option_t *options) _CUPS_API_1_6; +extern int cupsEnumDests(unsigned flags, int msec, int *cancel, + cups_ptype_t type, cups_ptype_t mask, + cups_dest_cb_t cb, void *user_data) + _CUPS_API_1_6; +# ifdef __BLOCKS__ +extern int cupsEnumDestsBlock(unsigned flags, int msec, + int *cancel, cups_ptype_t type, + cups_ptype_t mask, + cups_dest_block_t block) + _CUPS_API_1_6; +# endif /* __BLOCKS__ */ +extern ipp_status_t cupsFinishDestDocument(http_t *http, + cups_dest_t *dest) _CUPS_API_1_6; +extern void cupsFreeDestInfo(cups_dinfo_t *dinfo) _CUPS_API_1_6; +extern int cupsGetDestMediaByName(http_t *http, cups_dest_t *dest, + cups_dinfo_t *dinfo, + const char *media, + unsigned flags, + cups_size_t *size) _CUPS_API_1_6; +extern int cupsGetDestMediaBySize(http_t *http, cups_dest_t *dest, + cups_dinfo_t *dinfo, + int width, int length, + unsigned flags, + cups_size_t *size) _CUPS_API_1_6; +extern const char *cupsLocalizeDestOption(http_t *http, cups_dest_t *dest, + cups_dinfo_t *info, + const char *option) + _CUPS_API_1_6; +extern const char *cupsLocalizeDestValue(http_t *http, cups_dest_t *dest, + cups_dinfo_t *info, + const char *option, + const char *value) + _CUPS_API_1_6; +extern http_status_t cupsStartDestDocument(http_t *http, cups_dest_t *dest, + cups_dinfo_t *info, int job_id, + const char *docname, + const char *format, + int num_options, + cups_option_t *options, + int last_document) _CUPS_API_1_6; + +# ifdef __cplusplus +} +# endif /* __cplusplus */ + +#endif /* !_CUPS_CUPS_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/custom.c b/cups/custom.c new file mode 100644 index 0000000000..94ff9bb46a --- /dev/null +++ b/cups/custom.c @@ -0,0 +1,122 @@ +/* + * "$Id$" + * + * PPD custom option routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * PostScript is a trademark of Adobe Systems, Inc. + * + * This code and any derivative of it may be used and distributed + * freely under the terms of the GNU General Public License when + * used with GNU Ghostscript or its derivatives. Use of the code + * (or any derivative of it) with software other than GNU + * GhostScript (or its derivatives) is governed by the CUPS license + * agreement. + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * ppdFindCustomOption() - Find a custom option. + * ppdFindCustomParam() - Find a parameter for a custom option. + * ppdFirstCustomParam() - Return the first parameter for a custom option. + * ppdNextCustomParam() - Return the next parameter for a custom option. + */ + +/* + * Include necessary headers. + */ + +#include "cups-private.h" + + +/* + * 'ppdFindCustomOption()' - Find a custom option. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ppd_coption_t * /* O - Custom option or NULL */ +ppdFindCustomOption(ppd_file_t *ppd, /* I - PPD file */ + const char *keyword)/* I - Custom option name */ +{ + ppd_coption_t key; /* Custom option search key */ + + + if (!ppd) + return (NULL); + + strlcpy(key.keyword, keyword, sizeof(key.keyword)); + return ((ppd_coption_t *)cupsArrayFind(ppd->coptions, &key)); +} + + +/* + * 'ppdFindCustomParam()' - Find a parameter for a custom option. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ppd_cparam_t * /* O - Custom parameter or NULL */ +ppdFindCustomParam(ppd_coption_t *opt, /* I - Custom option */ + const char *name) /* I - Parameter name */ +{ + ppd_cparam_t *param; /* Current custom parameter */ + + + if (!opt) + return (NULL); + + for (param = (ppd_cparam_t *)cupsArrayFirst(opt->params); + param; + param = (ppd_cparam_t *)cupsArrayNext(opt->params)) + if (!_cups_strcasecmp(param->name, name)) + break; + + return (param); +} + + +/* + * 'ppdFirstCustomParam()' - Return the first parameter for a custom option. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ppd_cparam_t * /* O - Custom parameter or NULL */ +ppdFirstCustomParam(ppd_coption_t *opt) /* I - Custom option */ +{ + if (!opt) + return (NULL); + + return ((ppd_cparam_t *)cupsArrayFirst(opt->params)); +} + + +/* + * 'ppdNextCustomParam()' - Return the next parameter for a custom option. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ppd_cparam_t * /* O - Custom parameter or NULL */ +ppdNextCustomParam(ppd_coption_t *opt) /* I - Custom option */ +{ + if (!opt) + return (NULL); + + return ((ppd_cparam_t *)cupsArrayNext(opt->params)); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/debug-private.h b/cups/debug-private.h new file mode 100644 index 0000000000..26c75a379b --- /dev/null +++ b/cups/debug-private.h @@ -0,0 +1,117 @@ +/* + * "$Id$" + * + * Private debugging macros for CUPS. + * + * Copyright 2007-2012 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_DEBUG_PRIVATE_H_ +# define _CUPS_DEBUG_PRIVATE_H_ + + +/* + * Include necessary headers... + */ + +# include + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * The debug macros are used if you compile with DEBUG defined. + * + * Usage: + * + * DEBUG_puts("string") + * DEBUG_printf(("format string", arg, arg, ...)); + * + * Note the extra parenthesis around the DEBUG_printf macro... + * + * Newlines are not required on the end of messages, as both add one when + * writing the output. + * + * If the first character is a digit, then it represents the "log level" of the + * message from 0 to 9. The default level is 1. The following defines the + * current levels we use: + * + * 0 = public APIs, other than value accessor functions + * 1 = return values for public APIs + * 2 = public value accessor APIs, progress for public APIs + * 3 = return values for value accessor APIs + * 4 = private APIs, progress for value accessor APIs + * 5 = return values for private APIs + * 6 = progress for private APIs + * 7 = static functions + * 8 = return values for static functions + * 9 = progress for static functions + * + * The DEBUG_set macro allows an application to programmatically enable (or + * disable) debug logging. The arguments correspond to the CUPS_DEBUG_LOG, + * CUPS_DEBUG_LEVEL, and CUPS_DEBUG_FILTER environment variables. + */ + +# ifdef DEBUG +# ifdef WIN32 +# ifdef LIBCUPS2_EXPORTS +# define DLLExport __declspec(dllexport) +# else +# define DLLExport +# endif /* LIBCUPS2_EXPORTS */ +# else +# define DLLExport +# endif /* WIN32 */ +# define DEBUG_puts(x) _cups_debug_puts(x) +# define DEBUG_printf(x) _cups_debug_printf x +# define DEBUG_set(logfile,level,filter) _cups_debug_set(logfile,level,filter,1) +# else +# define DLLExport +# define DEBUG_puts(x) +# define DEBUG_printf(x) +# define DEBUG_set(logfile,level,filter) +# endif /* DEBUG */ + + +/* + * Prototypes... + */ + +extern int _cups_debug_fd; +extern int _cups_debug_level; +extern void DLLExport _cups_debug_printf(const char *format, ...) + __attribute__ ((__format__ (__printf__, 1, 2))); +extern void DLLExport _cups_debug_puts(const char *s); +extern void DLLExport _cups_debug_set(const char *logfile, + const char *level, const char *filter, + int force); +# ifdef WIN32 +extern int _cups_gettimeofday(struct timeval *tv, void *tz); +# define gettimeofday(a,b) _cups_gettimeofday(a, b) +# endif /* WIN32 */ + +# ifdef __cplusplus +} +# endif /* __cplusplus */ + +#endif /* !_CUPS_DEBUG_PRIVATE_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/debug.c b/cups/debug.c new file mode 100644 index 0000000000..b8a743c126 --- /dev/null +++ b/cups/debug.c @@ -0,0 +1,668 @@ +/* + * "$Id$" + * + * Debugging functions for CUPS. + * + * Copyright 2008-2012 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * debug_vsnprintf() - Format a string into a fixed size buffer. + * _cups_debug_printf() - Write a formatted line to the log. + * _cups_debug_puts() - Write a single line to the log. + * _cups_debug_set() - Enable or disable debug logging. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include "thread-private.h" +#ifdef WIN32 +# include +# include +# include +# define getpid (int)GetCurrentProcessId +int /* O - 0 on success, -1 on failure */ +_cups_gettimeofday(struct timeval *tv, /* I - Timeval struct */ + void *tz) /* I - Timezone */ +{ + struct _timeb timebuffer; /* Time buffer struct */ + _ftime(&timebuffer); + tv->tv_sec = (long)timebuffer.time; + tv->tv_usec = timebuffer.millitm * 1000; + return 0; +} +#else +# include +# include +# include +#endif /* WIN32 */ +#include + + +/* + * Globals... + */ + +int _cups_debug_fd = -1; + /* Debug log file descriptor */ +int _cups_debug_level = 1; + /* Log level (0 to 9) */ + + +#ifdef DEBUG +/* + * Local globals... + */ + +# ifndef WIN32 +static regex_t *debug_filter = NULL; + /* Filter expression for messages */ +# endif /* !WIN32 */ +static int debug_init = 0; /* Did we initialize debugging? */ +static _cups_mutex_t debug_mutex = _CUPS_MUTEX_INITIALIZER; + /* Mutex to control initialization */ + + +/* + * 'debug_vsnprintf()' - Format a string into a fixed size buffer. + */ + +static int /* O - Number of bytes formatted */ +debug_vsnprintf(char *buffer, /* O - Output buffer */ + size_t bufsize, /* O - Size of output buffer */ + const char *format, /* I - printf-style format string */ + va_list ap) /* I - Pointer to additional arguments */ +{ + char *bufptr, /* Pointer to position in buffer */ + *bufend, /* Pointer to end of buffer */ + size, /* Size character (h, l, L) */ + type; /* Format type character */ + int width, /* Width of field */ + prec; /* Number of characters of precision */ + char tformat[100], /* Temporary format string for sprintf() */ + *tptr, /* Pointer into temporary format */ + temp[1024]; /* Buffer for formatted numbers */ + char *s; /* Pointer to string */ + int bytes; /* Total number of bytes needed */ + + + if (!buffer || bufsize < 2 || !format) + return (-1); + + /* + * Loop through the format string, formatting as needed... + */ + + bufptr = buffer; + bufend = buffer + bufsize - 1; + bytes = 0; + + while (*format) + { + if (*format == '%') + { + tptr = tformat; + *tptr++ = *format++; + + if (*format == '%') + { + if (bufptr < bufend) + *bufptr++ = *format; + bytes ++; + format ++; + continue; + } + else if (strchr(" -+#\'", *format)) + *tptr++ = *format++; + + if (*format == '*') + { + /* + * Get width from argument... + */ + + format ++; + width = va_arg(ap, int); + + snprintf(tptr, sizeof(tformat) - (tptr - tformat), "%d", width); + tptr += strlen(tptr); + } + else + { + width = 0; + + while (isdigit(*format & 255)) + { + if (tptr < (tformat + sizeof(tformat) - 1)) + *tptr++ = *format; + + width = width * 10 + *format++ - '0'; + } + } + + if (*format == '.') + { + if (tptr < (tformat + sizeof(tformat) - 1)) + *tptr++ = *format; + + format ++; + + if (*format == '*') + { + /* + * Get precision from argument... + */ + + format ++; + prec = va_arg(ap, int); + + snprintf(tptr, sizeof(tformat) - (tptr - tformat), "%d", prec); + tptr += strlen(tptr); + } + else + { + prec = 0; + + while (isdigit(*format & 255)) + { + if (tptr < (tformat + sizeof(tformat) - 1)) + *tptr++ = *format; + + prec = prec * 10 + *format++ - '0'; + } + } + } + + if (*format == 'l' && format[1] == 'l') + { + size = 'L'; + + if (tptr < (tformat + sizeof(tformat) - 2)) + { + *tptr++ = 'l'; + *tptr++ = 'l'; + } + + format += 2; + } + else if (*format == 'h' || *format == 'l' || *format == 'L') + { + if (tptr < (tformat + sizeof(tformat) - 1)) + *tptr++ = *format; + + size = *format++; + } + else + size = 0; + + if (!*format) + break; + + if (tptr < (tformat + sizeof(tformat) - 1)) + *tptr++ = *format; + + type = *format++; + *tptr = '\0'; + + switch (type) + { + case 'E' : /* Floating point formats */ + case 'G' : + case 'e' : + case 'f' : + case 'g' : + if ((width + 2) > sizeof(temp)) + break; + + sprintf(temp, tformat, va_arg(ap, double)); + + bytes += (int)strlen(temp); + + if (bufptr) + { + if ((bufptr + strlen(temp)) > bufend) + { + strncpy(bufptr, temp, (size_t)(bufend - bufptr)); + bufptr = bufend; + } + else + { + strcpy(bufptr, temp); + bufptr += strlen(temp); + } + } + break; + + case 'B' : /* Integer formats */ + case 'X' : + case 'b' : + case 'd' : + case 'i' : + case 'o' : + case 'u' : + case 'x' : + if ((width + 2) > sizeof(temp)) + break; + +# ifdef HAVE_LONG_LONG + if (size == 'L') + sprintf(temp, tformat, va_arg(ap, long long)); + else +# endif /* HAVE_LONG_LONG */ + if (size == 'l') + sprintf(temp, tformat, va_arg(ap, long)); + else + sprintf(temp, tformat, va_arg(ap, int)); + + bytes += (int)strlen(temp); + + if (bufptr) + { + if ((bufptr + strlen(temp)) > bufend) + { + strncpy(bufptr, temp, (size_t)(bufend - bufptr)); + bufptr = bufend; + } + else + { + strcpy(bufptr, temp); + bufptr += strlen(temp); + } + } + break; + + case 'p' : /* Pointer value */ + if ((width + 2) > sizeof(temp)) + break; + + sprintf(temp, tformat, va_arg(ap, void *)); + + bytes += (int)strlen(temp); + + if (bufptr) + { + if ((bufptr + strlen(temp)) > bufend) + { + strncpy(bufptr, temp, (size_t)(bufend - bufptr)); + bufptr = bufend; + } + else + { + strcpy(bufptr, temp); + bufptr += strlen(temp); + } + } + break; + + case 'c' : /* Character or character array */ + bytes += width; + + if (bufptr) + { + if (width <= 1) + *bufptr++ = va_arg(ap, int); + else + { + if ((bufptr + width) > bufend) + width = (int)(bufend - bufptr); + + memcpy(bufptr, va_arg(ap, char *), (size_t)width); + bufptr += width; + } + } + break; + + case 's' : /* String */ + if ((s = va_arg(ap, char *)) == NULL) + s = "(null)"; + + /* + * Copy the C string, replacing control chars and \ with + * C character escapes... + */ + + for (bufend --; *s && bufptr < bufend; s ++) + { + if (*s == '\n') + { + *bufptr++ = '\\'; + *bufptr++ = 'n'; + bytes += 2; + } + else if (*s == '\r') + { + *bufptr++ = '\\'; + *bufptr++ = 'r'; + bytes += 2; + } + else if (*s == '\t') + { + *bufptr++ = '\\'; + *bufptr++ = 't'; + bytes += 2; + } + else if (*s == '\\') + { + *bufptr++ = '\\'; + *bufptr++ = '\\'; + bytes += 2; + } + else if (*s == '\'') + { + *bufptr++ = '\\'; + *bufptr++ = '\''; + bytes += 2; + } + else if (*s == '\"') + { + *bufptr++ = '\\'; + *bufptr++ = '\"'; + bytes += 2; + } + else if ((*s & 255) < ' ') + { + if ((bufptr + 2) >= bufend) + break; + + *bufptr++ = '\\'; + *bufptr++ = '0'; + *bufptr++ = '0' + *s / 8; + *bufptr++ = '0' + (*s & 7); + bytes += 4; + } + else + { + *bufptr++ = *s; + bytes ++; + } + } + + bufend ++; + break; + + case 'n' : /* Output number of chars so far */ + *(va_arg(ap, int *)) = bytes; + break; + } + } + else + { + bytes ++; + + if (bufptr < bufend) + *bufptr++ = *format; + + format ++; + } + } + + /* + * Nul-terminate the string and return the number of characters needed. + */ + + *bufptr = '\0'; + + return (bytes); +} + + +/* + * '_cups_debug_printf()' - Write a formatted line to the log. + */ + +void DLLExport +_cups_debug_printf(const char *format, /* I - Printf-style format string */ + ...) /* I - Additional arguments as needed */ +{ + va_list ap; /* Pointer to arguments */ + struct timeval curtime; /* Current time */ + char buffer[2048]; /* Output buffer */ + size_t bytes; /* Number of bytes in buffer */ + int level; /* Log level in message */ + + + /* + * See if we need to do any logging... + */ + + if (!debug_init) + _cups_debug_set(getenv("CUPS_DEBUG_LOG"), getenv("CUPS_DEBUG_LEVEL"), + getenv("CUPS_DEBUG_FILTER"), 0); + + if (_cups_debug_fd < 0) + return; + + /* + * Filter as needed... + */ + + if (isdigit(format[0])) + level = *format++ - '0'; + else + level = 0; + + if (level > _cups_debug_level) + return; + +# ifndef WIN32 + if (debug_filter) + { + int result; /* Filter result */ + + _cupsMutexLock(&debug_mutex); + result = regexec(debug_filter, format, 0, NULL, 0); + _cupsMutexUnlock(&debug_mutex); + + if (result) + return; + } +# endif /* !WIN32 */ + + /* + * Format the message... + */ + + gettimeofday(&curtime, NULL); + snprintf(buffer, sizeof(buffer), "%02d:%02d:%02d.%03d ", + (int)((curtime.tv_sec / 3600) % 24), + (int)((curtime.tv_sec / 60) % 60), + (int)(curtime.tv_sec % 60), (int)(curtime.tv_usec / 1000)); + + va_start(ap, format); + bytes = debug_vsnprintf(buffer + 13, sizeof(buffer) - 14, format, ap) + 13; + va_end(ap); + + if (bytes >= (sizeof(buffer) - 1)) + { + buffer[sizeof(buffer) - 2] = '\n'; + bytes = sizeof(buffer) - 1; + } + else if (buffer[bytes - 1] != '\n') + { + buffer[bytes++] = '\n'; + buffer[bytes] = '\0'; + } + + /* + * Write it out... + */ + + write(_cups_debug_fd, buffer, bytes); +} + + +/* + * '_cups_debug_puts()' - Write a single line to the log. + */ + +void DLLExport +_cups_debug_puts(const char *s) /* I - String to output */ +{ + struct timeval curtime; /* Current time */ + char buffer[2048]; /* Output buffer */ + size_t bytes; /* Number of bytes in buffer */ + int level; /* Log level in message */ + + + /* + * See if we need to do any logging... + */ + + if (!debug_init) + _cups_debug_set(getenv("CUPS_DEBUG_LOG"), getenv("CUPS_DEBUG_LEVEL"), + getenv("CUPS_DEBUG_FILTER"), 0); + + if (_cups_debug_fd < 0) + return; + + /* + * Filter as needed... + */ + + if (isdigit(s[0])) + level = *s++ - '0'; + else + level = 0; + + if (level > _cups_debug_level) + return; + +# ifndef WIN32 + if (debug_filter) + { + int result; /* Filter result */ + + _cupsMutexLock(&debug_mutex); + result = regexec(debug_filter, s, 0, NULL, 0); + _cupsMutexUnlock(&debug_mutex); + + if (result) + return; + } +# endif /* !WIN32 */ + + /* + * Format the message... + */ + + gettimeofday(&curtime, NULL); + bytes = snprintf(buffer, sizeof(buffer), "%02d:%02d:%02d.%03d %s", + (int)((curtime.tv_sec / 3600) % 24), + (int)((curtime.tv_sec / 60) % 60), + (int)(curtime.tv_sec % 60), (int)(curtime.tv_usec / 1000), + s); + + if (bytes >= (sizeof(buffer) - 1)) + { + buffer[sizeof(buffer) - 2] = '\n'; + bytes = sizeof(buffer) - 1; + } + else if (buffer[bytes - 1] != '\n') + { + buffer[bytes++] = '\n'; + buffer[bytes] = '\0'; + } + + /* + * Write it out... + */ + + write(_cups_debug_fd, buffer, bytes); +} + + +/* + * '_cups_debug_set()' - Enable or disable debug logging. + */ + +void DLLExport +_cups_debug_set(const char *logfile, /* I - Log file or NULL */ + const char *level, /* I - Log level or NULL */ + const char *filter, /* I - Filter string or NULL */ + int force) /* I - Force initialization */ +{ + _cupsMutexLock(&debug_mutex); + + if (!debug_init || force) + { + /* + * Restore debug settings to defaults... + */ + + if (_cups_debug_fd != -1) + { + close(_cups_debug_fd); + _cups_debug_fd = -1; + } + +# ifndef WIN32 + if (debug_filter) + { + regfree((regex_t *)debug_filter); + debug_filter = NULL; + } +# endif /* !WIN32 */ + + _cups_debug_level = 1; + + /* + * Open logs, set log levels, etc. + */ + + if (!logfile) + _cups_debug_fd = -1; + else if (!strcmp(logfile, "-")) + _cups_debug_fd = 2; + else + { + char buffer[1024]; /* Filename buffer */ + + snprintf(buffer, sizeof(buffer), logfile, getpid()); + + if (buffer[0] == '+') + _cups_debug_fd = open(buffer + 1, O_WRONLY | O_APPEND | O_CREAT, 0644); + else + _cups_debug_fd = open(buffer, O_WRONLY | O_TRUNC | O_CREAT, 0644); + } + + if (level) + _cups_debug_level = atoi(level); + +# ifndef WIN32 + if (filter) + { + if ((debug_filter = (regex_t *)calloc(1, sizeof(regex_t))) == NULL) + fputs("Unable to allocate memory for CUPS_DEBUG_FILTER - results not " + "filtered!\n", stderr); + else if (regcomp(debug_filter, filter, REG_EXTENDED)) + { + fputs("Bad regular expression in CUPS_DEBUG_FILTER - results not " + "filtered!\n", stderr); + free(debug_filter); + debug_filter = NULL; + } + } +# endif /* !WIN32 */ + + debug_init = 1; + } + + _cupsMutexUnlock(&debug_mutex); +} +#endif /* DEBUG */ + + +/* + * End of "$Id$". + */ diff --git a/cups/dest-job.c b/cups/dest-job.c new file mode 100644 index 0000000000..31cac7903c --- /dev/null +++ b/cups/dest-job.c @@ -0,0 +1,145 @@ +/* + * "$Id$" + * + * Destination job support for CUPS. + * + * Copyright 2012 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsCancelDestJob() - Cancel a job on a destination. + * cupsCloseDestJob() - Close a job and start printing. + * cupsCreateDestJob() - Create a job on a destination. + * cupsFinishDestDocument() - Finish the current document. + * cupsStartDestDocument() - Start a new document. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" + + +/* + * 'cupsCancelDestJob()' - Cancel a job on a destination. + * + * The "job_id" is the number returned by cupsCreateDestJob. + * + * Returns IPP_OK on success and IPP_NOT_AUTHORIZED or IPP_FORBIDDEN on + * failure. + * + * @since CUPS 1.6@ + */ + +ipp_status_t +cupsCancelDestJob(http_t *http, /* I - Connection to destination */ + cups_dest_t *dest, /* I - Destination */ + int job_id) /* I - Job ID */ +{ + return (IPP_NOT_FOUND); +} + + +/* + * 'cupsCloseDestJob()' - Close a job and start printing. + * + * Use when the last call to cupsStartDocument passed 0 for "last_document". + * "job_id" is the job ID returned by cupsCreateDestJob. Returns IPP_OK on + * success. + * + * @since CUPS 1.6@ + */ + +ipp_status_t +cupsCloseDestJob( + http_t *http, /* I - Connection to destination */ + cups_dest_t *dest, /* I - Destination */ + int job_id) /* I - Job ID */ +{ + return (IPP_NOT_FOUND); +} + + +/* + * 'cupsCreateDestJob()' - Create a job on a destination. + * + * Returns IPP_OK or IPP_OK_SUBST on success, saving the job ID in the variable + * pointed to by "job_id". + * + * @since CUPS 1.6@ + */ + +ipp_status_t /* O - IPP status code */ +cupsCreateDestJob( + http_t *http, /* I - Connection to destination */ + cups_dest_t *dest, /* I - Destination */ + cups_dinfo_t *info, /* I - Destination information */ + int *job_id, /* O - Job ID or 0 on error */ + const char *title, /* I - Job name */ + int num_options, /* I - Number of job options */ + cups_option_t *options) /* I - Job options */ +{ + *job_id = 0; + + return (IPP_NOT_POSSIBLE); +} + + +/* + * 'cupsFinishDestDocument()' - Finish the current document. + * + * Returns IPP_OK on success. + * + * @since CUPS 1.6@ + */ + +ipp_status_t +cupsFinishDestDocument( + http_t *http, /* I - Connection to destination */ + cups_dest_t *dest) /* I - Destination */ +{ + return (IPP_NOT_FOUND); +} + + +/* + * 'cupsStartDestDocument()' - Start a new document. + * + * "job_id" is the job ID returned by cupsCreateDestJob. "docname" is the name + * of the document/file being printed, "format" is the MIME media type for the + * document (see CUPS_FORMAT_xxx constants), and "num_options" and "options" + * are the options do be applied to the document. "last_document" should be 1 + * if this is the last document to be submitted in the job. Returns + * HTTP_CONTINUE on success. + * + * @since CUPS 1.6@ + */ + +http_status_t +cupsStartDestDocument( + http_t *http, /* I - Connection to destination */ + cups_dest_t *dest, /* I - Destination */ + cups_dinfo_t *info, /* I - Destination information */ + int job_id, /* I - Job ID */ + const char *docname, /* I - Document name */ + const char *format, /* I - Document format */ + int num_options, /* I - Number of document options */ + cups_option_t *options, /* I - Document options */ + int last_document) /* I - 1 if this is the last document */ +{ + return (HTTP_CONTINUE); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/dest-localization.c b/cups/dest-localization.c new file mode 100644 index 0000000000..9d0d8b1d7c --- /dev/null +++ b/cups/dest-localization.c @@ -0,0 +1,76 @@ +/* + * "$Id$" + * + * Destination localization support for CUPS. + * + * Copyright 2012 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsLocalizeDestOption() - Get the localized string for a destination + * option. + * cupsLocalizeDestValue() - Get the localized string for a destination + * option+value pair. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" + + +/* + * 'cupsLocalizeDestOption()' - Get the localized string for a destination + * option. + * + * The returned string is stored in the localization array and will become + * invalid if the localization array is deleted. + * + * @since CUPS 1.6@ + */ + +const char * /* O - Localized string */ +cupsLocalizeDestOption( + http_t *http, /* I - Connection to destination */ + cups_dest_t *dest, /* I - Destination */ + cups_dinfo_t *dinfo, /* I - Destination information */ + const char *option) /* I - Option to localize */ +{ + return (option); +} + + +/* + * 'cupsLocalizeDestValue()' - Get the localized string for a destination + * option+value pair. + * + * The returned string is stored in the localization array and will become + * invalid if the localization array is deleted. + * + * @since CUPS 1.6@ + */ + +const char * /* O - Localized string */ +cupsLocalizeDestValue( + http_t *http, /* I - Connection to destination */ + cups_dest_t *dest, /* I - Destination */ + cups_dinfo_t *dinfo, /* I - Destination information */ + const char *option, /* I - Option to localize */ + const char *value) /* I - Value to localize */ +{ + return (value); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/dest-options.c b/cups/dest-options.c new file mode 100644 index 0000000000..eff70dcb00 --- /dev/null +++ b/cups/dest-options.c @@ -0,0 +1,1164 @@ +/* + * "$Id$" + * + * Destination option/media support for CUPS. + * + * Copyright 2012 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsCheckDestSupported() - Check that the option and value are supported + * by the destination. + * cupsCopyDestConflicts() - Get conflicts and resolutions for a new + * option/value pair. + * cupsCopyDestInfo() - Get the supported values/capabilities for the + * destination. + * cupsFreeDestInfo() - Free destination information obtained using + * @link cupsCopyDestInfo@. + * cupsGetDestMediaByName() - Get media names, dimensions, and margins. + * cupsGetDestMediaBySize() - Get media names, dimensions, and margins. + * cups_compare_media_db() - Compare two media entries. + * cups_copy_media_db() - Copy a media entry. + * cups_create_media_db() - Create the media database. + * cups_free_media_cb() - Free a media entry. + * cups_get_media_db() - Lookup the media entry for a given size. + * cups_is_close_media_db() - Compare two media entries to see if they are + * close to the same size. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" + + +/* + * Local functions... + */ + +static int cups_compare_media_db(_cups_media_db_t *a, + _cups_media_db_t *b); +static _cups_media_db_t *cups_copy_media_db(_cups_media_db_t *mdb); +static void cups_create_media_db(cups_dinfo_t *dinfo); +static void cups_free_media_db(_cups_media_db_t *mdb); +static int cups_get_media_db(cups_dinfo_t *dinfo, + _pwg_media_t *pwg, unsigned flags, + cups_size_t *size); +static int cups_is_close_media_db(_cups_media_db_t *a, + _cups_media_db_t *b); + +/* + * 'cupsCheckDestSupported()' - Check that the option and value are supported + * by the destination. + * + * Returns 1 if supported, 0 otherwise. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 if supported, 0 otherwise */ +cupsCheckDestSupported( + http_t *http, /* I - Connection to destination */ + cups_dest_t *dest, /* I - Destination */ + cups_dinfo_t *dinfo, /* I - Destination information */ + const char *option, /* I - Option */ + const char *value) /* I - Value */ +{ + int i; /* Looping var */ + char temp[1024]; /* Temporary string */ + int int_value; /* Integer value */ + int xres_value, /* Horizontal resolution */ + yres_value; /* Vertical resolution */ + ipp_res_t units_value; /* Resolution units */ + ipp_attribute_t *attr; /* Attribute */ + _ipp_value_t *attrval; /* Current attribute value */ + + + /* + * Range check input... + */ + + if (!http || !dest || !dinfo || !option || !value) + return (0); + + /* + * Lookup the attribute... + */ + + if (strstr(option, "-supported")) + attr = ippFindAttribute(dinfo->attrs, option, IPP_TAG_ZERO); + else + { + snprintf(temp, sizeof(temp), "%s-supported", option); + attr = ippFindAttribute(dinfo->attrs, temp, IPP_TAG_ZERO); + } + + if (!attr) + return (0); + + /* + * Compare values... + */ + + if (!strcmp(option, "media") && !strncmp(value, "custom_", 7)) + { + /* + * Check range of custom media sizes... + */ + + _pwg_media_t *pwg; /* Current PWG media size info */ + int min_width, /* Minimum width */ + min_length, /* Minimum length */ + max_width, /* Maximum width */ + max_length; /* Maximum length */ + + /* + * Get the minimum and maximum size... + */ + + min_width = min_length = INT_MAX; + max_width = max_length = 0; + + for (i = attr->num_values, attrval = attr->values; + i > 0; + i --, attrval ++) + { + if (!strncmp(attrval->string.text, "custom_min_", 11) && + (pwg = _pwgMediaForPWG(attrval->string.text)) != NULL) + { + min_width = pwg->width; + min_length = pwg->length; + } + else if (!strncmp(attrval->string.text, "custom_max_", 11) && + (pwg = _pwgMediaForPWG(attrval->string.text)) != NULL) + { + max_width = pwg->width; + max_length = pwg->length; + } + } + + /* + * Check the range... + */ + + if (min_width < INT_MAX && max_width > 0 && + (pwg = _pwgMediaForPWG(value)) != NULL && + pwg->width >= min_width && pwg->width <= max_width && + pwg->length >= min_length && pwg->length <= max_length) + return (1); + } + else + { + /* + * Check literal values... + */ + + switch (attr->value_tag) + { + case IPP_TAG_INTEGER : + case IPP_TAG_ENUM : + int_value = atoi(value); + + for (i = 0; i < attr->num_values; i ++) + if (attr->values[i].integer == int_value) + return (1); + break; + + case IPP_TAG_BOOLEAN : + return (attr->values[0].boolean); + + case IPP_TAG_RESOLUTION : + if (sscanf(value, "%dx%d%15s", &xres_value, &yres_value, temp) != 3) + { + if (sscanf(value, "%d%15s", &xres_value, temp) != 2) + return (0); + + yres_value = xres_value; + } + + if (!strcmp(temp, "dpi")) + units_value = IPP_RES_PER_INCH; + else if (!strcmp(temp, "dpc")) + units_value = IPP_RES_PER_CM; + else + return (0); + + for (i = attr->num_values, attrval = attr->values; + i > 0; + i --, attrval ++) + { + if (attrval->resolution.xres == xres_value && + attrval->resolution.yres == yres_value && + attrval->resolution.units == units_value) + return (1); + } + break; + + case IPP_TAG_TEXT : + case IPP_TAG_NAME : + case IPP_TAG_KEYWORD : + case IPP_TAG_CHARSET : + case IPP_TAG_URI : + case IPP_TAG_URISCHEME : + case IPP_TAG_MIMETYPE : + case IPP_TAG_LANGUAGE : + case IPP_TAG_TEXTLANG : + case IPP_TAG_NAMELANG : + for (i = 0; i < attr->num_values; i ++) + if (!strcmp(attr->values[i].string.text, value)) + return (1); + break; + + default : + break; + } + } + + /* + * If we get there the option+value is not supported... + */ + + return (0); +} + + +/* + * 'cupsCopyDestConflicts()' - Get conflicts and resolutions for a new + * option/value pair. + * + * "num_options" and "options" represent the currently selected options by the + * user. "new_option" and "new_value" are the setting the user has just + * changed. + * + * Returns 1 if there is a conflict and 0 otherwise. + * + * If "num_conflicts" and "conflicts" are not NULL, they are set to contain the + * list of conflicting option/value pairs. Similarly, if "num_resolved" and + * "resolved" are not NULL they will be set to the list of changes needed to + * resolve the conflict. + * + * If cupsCopyDestConflicts returns 1 but "num_resolved" and "resolved" are set + * to 0 and NULL, respectively, then the conflict cannot be resolved. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 if there is a conflict */ +cupsCopyDestConflicts( + http_t *http, /* I - Connection to destination */ + cups_dest_t *dest, /* I - Destination */ + cups_dinfo_t *dinfo, /* I - Destination information */ + int num_options, /* I - Number of current options */ + cups_option_t *options, /* I - Current options */ + const char *new_option, /* I - New option */ + const char *new_value, /* I - New value */ + int *num_conflicts, /* O - Number of conflicting options */ + cups_option_t **conflicts, /* O - Conflicting options */ + int *num_resolved, /* O - Number of options to resolve */ + cups_option_t **resolved) /* O - Resolved options */ +{ + /* + * Clear returned values... + */ + + if (num_conflicts) + *num_conflicts = 0; + + if (conflicts) + *conflicts = NULL; + + if (num_resolved) + *num_resolved = 0; + + if (resolved) + *resolved = NULL; + + /* + * Range check input... + */ + + if (!http || !dest || !dinfo || !new_option || !new_value || + (num_conflicts != NULL) != (conflicts != NULL) || + (num_resolved != NULL) != (resolved != NULL)) + return (0); + + /* + * Check for an resolve any conflicts... + */ + + /* TODO: implement me! */ + + return (0); +} + + +/* + * 'cupsCopyDestInfo()' - Get the supported values/capabilities for the + * destination. + * + * The caller is responsible for calling @link cupsFreeDestInfo@ on the return + * value. @code NULL@ is returned on error. + * + * @since CUPS 1.6@ + */ + +cups_dinfo_t * /* O - Destination information */ +cupsCopyDestInfo( + http_t *http, /* I - Connection to destination */ + cups_dest_t *dest) /* I - Destination */ +{ + cups_dinfo_t *dinfo; /* Destination information */ + ipp_t *request, /* Get-Printer-Attributes request */ + *response; /* Supported attributes */ + const char *uri; /* Printer URI */ + char resource[1024]; /* Resource path */ + int version; /* IPP version */ + ipp_status_t status; /* Status of request */ + static const char * const requested_attrs[] = + { /* Requested attributes */ + "job-template", + "media-col-database", + "printer-description" + }; + + + DEBUG_printf(("cupsCopyDestSupported(http=%p, dest=%p(%s))", http, dest, + dest ? dest->name : "")); + + /* + * Range check input... + */ + + if (!http || !dest) + return (NULL); + + /* + * Get the printer URI and resource path... + */ + + if ((uri = _cupsGetDestResource(dest, resource, sizeof(resource))) == NULL) + return (NULL); + + /* + * Get the supported attributes... + */ + + version = 20; + + do + { + /* + * Send a Get-Printer-Attributes request... + */ + + request = ippNewRequest(IPP_GET_PRINTER_ATTRIBUTES); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", NULL, + uri); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, cupsUser()); + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", + (int)(sizeof(requested_attrs) / sizeof(requested_attrs[0])), + NULL, requested_attrs); + response = cupsDoRequest(http, request, resource); + status = cupsLastError(); + + if (status > IPP_OK_SUBST) + { + DEBUG_printf(("cupsCopyDestSupported: Get-Printer-Attributes for '%s' " + "returned %s (%s)", dest->name, ippErrorString(status), + cupsLastErrorString())); + + ippDelete(response); + response = NULL; + + if (status == IPP_VERSION_NOT_SUPPORTED && version > 11) + version = 11; + else + return (NULL); + } + } + while (!response); + + /* + * Allocate a cups_dinfo_t structure and return it... + */ + + if ((dinfo = calloc(1, sizeof(cups_dinfo_t))) == NULL) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(errno), 0); + ippDelete(response); + return (NULL); + } + + dinfo->uri = uri; + dinfo->resource = _cupsStrAlloc(resource); + dinfo->attrs = response; + + return (dinfo); +} + + +/* + * 'cupsFreeDestInfo()' - Free destination information obtained using + * @link cupsCopyDestInfo@. + */ + +void +cupsFreeDestInfo(cups_dinfo_t *dinfo) /* I - Destination information */ +{ + /* + * Range check input... + */ + + if (!dinfo) + return; + + /* + * Free memory and return... + */ + + _cupsStrFree(dinfo->resource); + + ippDelete(dinfo->attrs); + + cupsArrayDelete(dinfo->constraints); + + cupsArrayDelete(dinfo->localizations); + + cupsArrayDelete(dinfo->media_db); + + free(dinfo); +} + + +/* + * 'cupsGetDestMediaByName()' - Get media names, dimensions, and margins. + * + * The "media" string is a PWG media name, while "width" and "length" are the + * dimensions in hundredths of millimeters. "flags" provides some matching + * guidance (multiple flags can be combined): + * + * CUPS_MEDIA_FLAGS_DEFAULT = find the closest size supported by the printer + * CUPS_MEDIA_FLAGS_BORDERLESS = find a borderless size + * CUPS_MEDIA_FLAGS_DUPLEX = find a size compatible with 2-sided printing + * CUPS_MEDIA_FLAGS_EXACT = find an exact match for the size + * CUPS_MEDIA_FLAGS_READY = if the printer supports media sensing, find the + * size amongst the "ready" media. + * + * The matching result (if any) is returned in the "cups_size_t" structure. + * + * Returns 1 when there is a match and 0 if there is not a match. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on match, 0 on failure */ +cupsGetDestMediaByName( + http_t *http, /* I - Connection to destination */ + cups_dest_t *dest, /* I - Destination */ + cups_dinfo_t *dinfo, /* I - Destination information */ + const char *media, /* I - Media name */ + unsigned flags, /* I - Media matching flags */ + cups_size_t *size) /* O - Media size information */ +{ + _pwg_media_t *pwg; /* PWG media info */ + + + /* + * Range check input... + */ + + if (size) + memset(size, 0, sizeof(cups_size_t)); + + if (!http || !dest || !dinfo || !media || !size) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + return (0); + } + + /* + * Lookup the media size name... + */ + + if ((pwg = _pwgMediaForPWG(media)) == NULL) + if ((pwg = _pwgMediaForLegacy(media)) == NULL) + { + DEBUG_printf(("1cupsGetDestMediaByName: Unknown size '%s'.", media)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Unknown media size name."), 1); + return (0); + } + + /* + * Lookup the size... + */ + + return (cups_get_media_db(dinfo, pwg, flags, size)); +} + + +/* + * 'cupsGetDestMediaBySize()' - Get media names, dimensions, and margins. + * + * The "media" string is a PWG media name, while "width" and "length" are the + * dimensions in hundredths of millimeters. "flags" provides some matching + * guidance (multiple flags can be combined): + * + * CUPS_MEDIA_FLAGS_DEFAULT = find the closest size supported by the printer + * CUPS_MEDIA_FLAGS_BORDERLESS = find a borderless size + * CUPS_MEDIA_FLAGS_DUPLEX = find a size compatible with 2-sided printing + * CUPS_MEDIA_FLAGS_EXACT = find an exact match for the size + * CUPS_MEDIA_FLAGS_READY = if the printer supports media sensing, find the + * size amongst the "ready" media. + * + * The matching result (if any) is returned in the "cups_size_t" structure. + * + * Returns 1 when there is a match and 0 if there is not a match. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on match, 0 on failure */ +cupsGetDestMediaBySize( + http_t *http, /* I - Connection to destination */ + cups_dest_t *dest, /* I - Destination */ + cups_dinfo_t *dinfo, /* I - Destination information */ + int width, /* I - Media width in hundredths of + * of millimeters */ + int length, /* I - Media length in hundredths of + * of millimeters */ + unsigned flags, /* I - Media matching flags */ + cups_size_t *size) /* O - Media size information */ +{ + _pwg_media_t *pwg; /* PWG media info */ + + + /* + * Range check input... + */ + + if (size) + memset(size, 0, sizeof(cups_size_t)); + + if (!http || !dest || !dinfo || width <= 0 || length <= 0 || !size) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + return (0); + } + + /* + * Lookup the media size name... + */ + + if ((pwg = _pwgMediaForSize(width, length)) == NULL) + { + DEBUG_printf(("1cupsGetDestMediaBySize: Invalid size %dx%d.", width, + length)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Invalid media size."), 1); + return (0); + } + + /* + * Lookup the size... + */ + + return (cups_get_media_db(dinfo, pwg, flags, size)); +} + + +/* + * 'cups_compare_media_db()' - Compare two media entries. + */ + +static int /* O - Result of comparison */ +cups_compare_media_db( + _cups_media_db_t *a, /* I - First media entries */ + _cups_media_db_t *b) /* I - Second media entries */ +{ + int result; /* Result of comparison */ + + + if ((result = a->width - b->width) == 0) + result = a->length - b->length; + + return (result); +} + + +/* + * 'cups_copy_media_db()' - Copy a media entry. + */ + +static _cups_media_db_t * /* O - New media entry */ +cups_copy_media_db( + _cups_media_db_t *mdb) /* I - Media entry to copy */ +{ + _cups_media_db_t *temp; /* New media entry */ + + + if ((temp = calloc(1, sizeof(_cups_media_db_t))) == NULL) + return (NULL); + + if (mdb->color) + temp->color = _cupsStrAlloc(mdb->color); + if (mdb->key) + temp->key = _cupsStrAlloc(mdb->key); + if (mdb->info) + temp->info = _cupsStrAlloc(mdb->info); + if (mdb->size_name) + temp->size_name = _cupsStrAlloc(mdb->size_name); + if (mdb->source) + temp->source = _cupsStrAlloc(mdb->source); + if (mdb->type) + temp->type = _cupsStrAlloc(mdb->type); + + temp->width = mdb->width; + temp->length = mdb->length; + temp->bottom = mdb->bottom; + temp->left = mdb->left; + temp->right = mdb->right; + temp->top = mdb->top; + + return (temp); +} + + +/* + * 'cups_create_media_db()' - Create the media database. + */ + +static void +cups_create_media_db( + cups_dinfo_t *dinfo) /* I - Destination information */ +{ + int i; /* Looping var */ + _ipp_value_t *val; /* Current value */ + ipp_attribute_t *media_col_db, /* media-col-database */ + *media_attr, /* media-xxx */ + *x_dimension, /* x-dimension */ + *y_dimension; /* y-dimension */ + _pwg_media_t *pwg; /* PWG media info */ + _cups_media_db_t mdb; /* Media entry */ + + + dinfo->media_db = cupsArrayNew3((cups_array_func_t)cups_compare_media_db, + NULL, NULL, 0, + (cups_acopy_func_t)cups_copy_media_db, + (cups_afree_func_t)cups_free_media_db); + dinfo->min_size.width = INT_MAX; + dinfo->min_size.length = INT_MAX; + dinfo->max_size.width = 0; + dinfo->max_size.length = 0; + + if ((media_col_db = ippFindAttribute(dinfo->attrs, "media-col-database", + IPP_TAG_BEGIN_COLLECTION)) != NULL) + { + _ipp_value_t *custom = NULL; /* Custom size range value */ + + for (i = media_col_db->num_values, val = media_col_db->values; + i > 0; + i --, val ++) + { + memset(&mdb, 0, sizeof(mdb)); + + if ((media_attr = ippFindAttribute(val->collection, "media-size", + IPP_TAG_BEGIN_COLLECTION)) != NULL) + { + ipp_t *media_size = media_attr->values[0].collection; + /* media-size collection value */ + + if ((x_dimension = ippFindAttribute(media_size, "x-dimension", + IPP_TAG_INTEGER)) != NULL && + (y_dimension = ippFindAttribute(media_size, "y-dimension", + IPP_TAG_INTEGER)) != NULL) + { + mdb.width = x_dimension->values[0].integer; + mdb.length = y_dimension->values[0].integer; + } + else if ((x_dimension = ippFindAttribute(media_size, "x-dimension", + IPP_TAG_RANGE)) != NULL && + (y_dimension = ippFindAttribute(media_size, "y-dimension", + IPP_TAG_RANGE)) != NULL) + { + /* + * Custom size range; save this as the custom size value with default + * margins, then continue; we'll capture the real margins below... + */ + + custom = val; + + dinfo->min_size.width = x_dimension->values[0].range.lower; + dinfo->min_size.length = y_dimension->values[0].range.lower; + dinfo->min_size.left = + dinfo->min_size.right = 635; /* Default 1/4" side margins */ + dinfo->min_size.top = + dinfo->min_size.bottom = 1270; /* Default 1/2" top/bottom margins */ + + dinfo->max_size.width = x_dimension->values[0].range.upper; + dinfo->max_size.length = y_dimension->values[0].range.upper; + dinfo->max_size.left = + dinfo->max_size.right = 635; /* Default 1/4" side margins */ + dinfo->max_size.top = + dinfo->max_size.bottom = 1270; /* Default 1/2" top/bottom margins */ + + continue; + } + } + + if ((media_attr = ippFindAttribute(val->collection, "media-color", + IPP_TAG_ZERO)) != NULL && + (media_attr->value_tag == IPP_TAG_NAME || + media_attr->value_tag == IPP_TAG_NAMELANG || + media_attr->value_tag == IPP_TAG_KEYWORD)) + mdb.color = media_attr->values[0].string.text; + + if ((media_attr = ippFindAttribute(val->collection, "media-info", + IPP_TAG_TEXT)) != NULL) + mdb.info = media_attr->values[0].string.text; + + if ((media_attr = ippFindAttribute(val->collection, "media-key", + IPP_TAG_ZERO)) != NULL && + (media_attr->value_tag == IPP_TAG_NAME || + media_attr->value_tag == IPP_TAG_NAMELANG || + media_attr->value_tag == IPP_TAG_KEYWORD)) + mdb.key = media_attr->values[0].string.text; + + if ((media_attr = ippFindAttribute(val->collection, "media-size-name", + IPP_TAG_ZERO)) != NULL && + (media_attr->value_tag == IPP_TAG_NAME || + media_attr->value_tag == IPP_TAG_NAMELANG || + media_attr->value_tag == IPP_TAG_KEYWORD)) + mdb.size_name = media_attr->values[0].string.text; + + if ((media_attr = ippFindAttribute(val->collection, "media-source", + IPP_TAG_ZERO)) != NULL && + (media_attr->value_tag == IPP_TAG_NAME || + media_attr->value_tag == IPP_TAG_NAMELANG || + media_attr->value_tag == IPP_TAG_KEYWORD)) + mdb.source = media_attr->values[0].string.text; + + if ((media_attr = ippFindAttribute(val->collection, "media-type", + IPP_TAG_ZERO)) != NULL && + (media_attr->value_tag == IPP_TAG_NAME || + media_attr->value_tag == IPP_TAG_NAMELANG || + media_attr->value_tag == IPP_TAG_KEYWORD)) + mdb.type = media_attr->values[0].string.text; + + if ((media_attr = ippFindAttribute(val->collection, "media-bottom-margin", + IPP_TAG_INTEGER)) != NULL) + mdb.bottom = media_attr->values[0].integer; + + if ((media_attr = ippFindAttribute(val->collection, "media-left-margin", + IPP_TAG_INTEGER)) != NULL) + mdb.left = media_attr->values[0].integer; + + if ((media_attr = ippFindAttribute(val->collection, "media-right-margin", + IPP_TAG_INTEGER)) != NULL) + mdb.right = media_attr->values[0].integer; + + if ((media_attr = ippFindAttribute(val->collection, "media-top-margin", + IPP_TAG_INTEGER)) != NULL) + mdb.top = media_attr->values[0].integer; + + cupsArrayAdd(dinfo->media_db, &mdb); + } + + if (custom) + { + if ((media_attr = ippFindAttribute(custom->collection, + "media-bottom-margin", + IPP_TAG_INTEGER)) != NULL) + { + dinfo->min_size.top = + dinfo->max_size.top = media_attr->values[0].integer; + } + + if ((media_attr = ippFindAttribute(custom->collection, + "media-left-margin", + IPP_TAG_INTEGER)) != NULL) + { + dinfo->min_size.left = + dinfo->max_size.left = media_attr->values[0].integer; + } + + if ((media_attr = ippFindAttribute(custom->collection, + "media-right-margin", + IPP_TAG_INTEGER)) != NULL) + { + dinfo->min_size.right = + dinfo->max_size.right = media_attr->values[0].integer; + } + + if ((media_attr = ippFindAttribute(custom->collection, + "media-top-margin", + IPP_TAG_INTEGER)) != NULL) + { + dinfo->min_size.top = + dinfo->max_size.top = media_attr->values[0].integer; + } + } + } + else if ((media_attr = ippFindAttribute(dinfo->attrs, "media-supported", + IPP_TAG_ZERO)) != NULL && + (media_attr->value_tag == IPP_TAG_NAME || + media_attr->value_tag == IPP_TAG_NAMELANG || + media_attr->value_tag == IPP_TAG_KEYWORD)) + { + memset(&mdb, 0, sizeof(mdb)); + + mdb.left = + mdb.right = 635; /* Default 1/4" side margins */ + mdb.top = + mdb.bottom = 1270; /* Default 1/2" top/bottom margins */ + + for (i = media_col_db->num_values, val = media_col_db->values; + i > 0; + i --, val ++) + { + if ((pwg = _pwgMediaForPWG(val->string.text)) == NULL) + if ((pwg = _pwgMediaForLegacy(val->string.text)) == NULL) + { + DEBUG_printf(("3cups_create_media_db: Ignoring unknown size '%s'.", + val->string.text)); + continue; + } + + mdb.width = pwg->width; + mdb.length = pwg->length; + + if (!strncmp(val->string.text, "custom_min_", 11)) + { + mdb.size_name = NULL; + dinfo->min_size = mdb; + } + else if (!strncmp(val->string.text, "custom_max_", 11)) + { + mdb.size_name = NULL; + dinfo->max_size = mdb; + } + else + { + mdb.size_name = val->string.text; + + cupsArrayAdd(dinfo->media_db, &mdb); + } + } + } +} + + +/* + * 'cups_free_media_cb()' - Free a media entry. + */ + +static void +cups_free_media_db( + _cups_media_db_t *mdb) /* I - Media entry to free */ +{ + if (mdb->color) + _cupsStrFree(mdb->color); + if (mdb->key) + _cupsStrFree(mdb->key); + if (mdb->info) + _cupsStrFree(mdb->info); + if (mdb->size_name) + _cupsStrFree(mdb->size_name); + if (mdb->source) + _cupsStrFree(mdb->source); + if (mdb->type) + _cupsStrFree(mdb->type); + + free(mdb); +} + + +/* + * 'cups_get_media_db()' - Lookup the media entry for a given size. + */ + +static int /* O - 1 on match, 0 on failure */ +cups_get_media_db(cups_dinfo_t *dinfo, /* I - Destination information */ + _pwg_media_t *pwg, /* I - PWG media info */ + unsigned flags, /* I - Media matching flags */ + cups_size_t *size) /* O - Media size/margin/name info */ +{ + _cups_media_db_t *mdb, /* Current media database entry */ + *best = NULL, /* Best matching entry */ + key; /* Search key */ + + + /* + * Create the media database as needed... + */ + + if (!dinfo->media_db) + cups_create_media_db(dinfo); + + /* + * Find a match... + */ + + memset(&key, 0, sizeof(key)); + key.width = pwg->width; + key.length = pwg->length; + + if ((mdb = cupsArrayFind(dinfo->media_db, &key)) != NULL) + { + /* + * Found an exact match, let's figure out the best margins for the flags + * supplied... + */ + + best = mdb; + + if (flags & CUPS_MEDIA_FLAGS_BORDERLESS) + { + /* + * Look for the smallest margins... + */ + + if (best->left != 0 || best->right != 0 || best->top != 0 || + best->bottom != 0) + { + for (mdb = (_cups_media_db_t *)cupsArrayNext(dinfo->media_db); + mdb && !cups_compare_media_db(mdb, &key); + mdb = (_cups_media_db_t *)cupsArrayNext(dinfo->media_db)) + { + if (mdb->left <= best->left && mdb->right <= best->right && + mdb->top <= best->top && mdb->bottom <= best->bottom) + { + best = mdb; + if (mdb->left == 0 && mdb->right == 0 && mdb->bottom == 0 && + mdb->top == 0) + break; + } + } + } + + /* + * If we need an exact match, return no-match if the size is not + * borderless. + */ + + if ((flags & CUPS_MEDIA_FLAGS_EXACT) && + (best->left || best->right || best->top || best->bottom)) + return (0); + } + else if (flags & CUPS_MEDIA_FLAGS_DUPLEX) + { + /* + * Look for the largest margins... + */ + + for (mdb = (_cups_media_db_t *)cupsArrayNext(dinfo->media_db); + mdb && !cups_compare_media_db(mdb, &key); + mdb = (_cups_media_db_t *)cupsArrayNext(dinfo->media_db)) + { + if (mdb->left >= best->left && mdb->right >= best->right && + mdb->top >= best->top && mdb->bottom >= best->bottom) + best = mdb; + } + } + else + { + /* + * Look for the smallest non-zero margins... + */ + + for (mdb = (_cups_media_db_t *)cupsArrayNext(dinfo->media_db); + mdb && !cups_compare_media_db(mdb, &key); + mdb = (_cups_media_db_t *)cupsArrayNext(dinfo->media_db)) + { + if (((mdb->left > 0 && mdb->left <= best->left) || best->left == 0) && + ((mdb->right > 0 && mdb->right <= best->right) || + best->right == 0) && + ((mdb->top > 0 && mdb->top <= best->top) || best->top == 0) && + ((mdb->bottom > 0 && mdb->bottom <= best->bottom) || + best->bottom == 0)) + best = mdb; + } + } + } + else if (flags & CUPS_MEDIA_FLAGS_EXACT) + { + /* + * See if we can do this as a custom size... + */ + + if (pwg->width < dinfo->min_size.width || + pwg->width > dinfo->max_size.width || + pwg->length < dinfo->min_size.length || + pwg->length > dinfo->max_size.length) + return (0); /* Out of range */ + + if ((flags & CUPS_MEDIA_FLAGS_BORDERLESS) && + (dinfo->min_size.left > 0 || dinfo->min_size.right > 0 || + dinfo->min_size.top > 0 || dinfo->min_size.bottom > 0)) + return (0); /* Not borderless */ + + key.size_name = (char *)pwg->pwg; + key.bottom = dinfo->min_size.bottom; + key.left = dinfo->min_size.left; + key.right = dinfo->min_size.right; + key.top = dinfo->min_size.top; + + best = &key; + } + else if (pwg->width >= dinfo->min_size.width && + pwg->width <= dinfo->max_size.width && + pwg->length >= dinfo->min_size.length && + pwg->length <= dinfo->max_size.length) + { + /* + * Map to custom size... + */ + + key.size_name = (char *)pwg->pwg; + key.bottom = dinfo->min_size.bottom; + key.left = dinfo->min_size.left; + key.right = dinfo->min_size.right; + key.top = dinfo->min_size.top; + + best = &key; + } + else + { + /* + * Find a close size... + */ + + for (mdb = (_cups_media_db_t *)cupsArrayFirst(dinfo->media_db); + mdb; + mdb = (_cups_media_db_t *)cupsArrayNext(dinfo->media_db)) + if (cups_is_close_media_db(mdb, &key)) + break; + + if (!mdb) + return (0); + + best = mdb; + + if (flags & CUPS_MEDIA_FLAGS_BORDERLESS) + { + /* + * Look for the smallest margins... + */ + + if (best->left != 0 || best->right != 0 || best->top != 0 || + best->bottom != 0) + { + for (mdb = (_cups_media_db_t *)cupsArrayNext(dinfo->media_db); + mdb && cups_is_close_media_db(mdb, &key); + mdb = (_cups_media_db_t *)cupsArrayNext(dinfo->media_db)) + { + if (mdb->left <= best->left && mdb->right <= best->right && + mdb->top <= best->top && mdb->bottom <= best->bottom) + { + best = mdb; + if (mdb->left == 0 && mdb->right == 0 && mdb->bottom == 0 && + mdb->top == 0) + break; + } + } + } + } + else if (flags & CUPS_MEDIA_FLAGS_DUPLEX) + { + /* + * Look for the largest margins... + */ + + for (mdb = (_cups_media_db_t *)cupsArrayNext(dinfo->media_db); + mdb && cups_is_close_media_db(mdb, &key); + mdb = (_cups_media_db_t *)cupsArrayNext(dinfo->media_db)) + { + if (mdb->left >= best->left && mdb->right >= best->right && + mdb->top >= best->top && mdb->bottom >= best->bottom) + best = mdb; + } + } + else + { + /* + * Look for the smallest non-zero margins... + */ + + for (mdb = (_cups_media_db_t *)cupsArrayNext(dinfo->media_db); + mdb && cups_is_close_media_db(mdb, &key); + mdb = (_cups_media_db_t *)cupsArrayNext(dinfo->media_db)) + { + if (((mdb->left > 0 && mdb->left <= best->left) || best->left == 0) && + ((mdb->right > 0 && mdb->right <= best->right) || + best->right == 0) && + ((mdb->top > 0 && mdb->top <= best->top) || best->top == 0) && + ((mdb->bottom > 0 && mdb->bottom <= best->bottom) || + best->bottom == 0)) + best = mdb; + } + } + } + + if (best) + { + /* + * Return the matching size... + */ + + if (best->size_name) + strlcpy(size->media, best->size_name, sizeof(size->media)); + else if (best->key) + strlcpy(size->media, best->key, sizeof(size->media)); + else + strlcpy(size->media, pwg->pwg, sizeof(size->media)); + + size->width = best->width; + size->length = best->length; + size->bottom = best->bottom; + size->left = best->left; + size->right = best->right; + size->top = best->top; + + return (1); + } + + return (0); +} + + +/* + * 'cups_is_close_media_db()' - Compare two media entries to see if they are + * close to the same size. + * + * Currently we use 5 points (from PostScript) as the matching range... + */ + +static int /* O - 1 if the sizes are close */ +cups_is_close_media_db( + _cups_media_db_t *a, /* I - First media entries */ + _cups_media_db_t *b) /* I - Second media entries */ +{ + int dwidth, /* Difference in width */ + dlength; /* Difference in length */ + + + dwidth = a->width - b->width; + dlength = a->length - b->length; + + return (dwidth >= -176 && dwidth <= 176 && + dlength >= -176 && dlength <= 176); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/dest.c b/cups/dest.c new file mode 100644 index 0000000000..4a8e74f981 --- /dev/null +++ b/cups/dest.c @@ -0,0 +1,3476 @@ +/* + * "$Id$" + * + * User-defined destination (and option) support for CUPS. + * + * Copyright 2007-2012 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsAddDest() - Add a destination to the list of + * destinations. + * _cupsAppleCopyDefaultPaperID() - Get the default paper ID. + * _cupsAppleCopyDefaultPrinter() - Get the default printer at this location. + * _cupsAppleGetUseLastPrinter() - Get whether to use the last used printer. + * _cupsAppleSetDefaultPaperID() - Set the default paper id. + * _cupsAppleSetDefaultPrinter() - Set the default printer for this + * location. + * _cupsAppleSetUseLastPrinter() - Set whether to use the last used printer. + * cupsConnectDest() - Connect to the server for a destination. + * cupsConnectDestBlock() - Connect to the server for a destination. + * cupsCopyDest() - Copy a destination. + * cupsEnumDests() - Enumerate available destinations with a + * callback function. + * cupsEnumDestsBlock() - Enumerate available destinations with a + * block. + * cupsFreeDests() - Free the memory used by the list of + * destinations. + * cupsGetDest() - Get the named destination from the list. + * _cupsGetDestResource() - Get the resource path and URI for a + * destination. + * _cupsGetDests() - Get destinations from a server. + * cupsGetDests() - Get the list of destinations from the + * default server. + * cupsGetDests2() - Get the list of destinations from the + * specified server. + * cupsGetNamedDest() - Get options for the named destination. + * cupsRemoveDest() - Remove a destination from the destination + * list. + * cupsSetDefaultDest() - Set the default destination. + * cupsSetDests() - Save the list of destinations for the + * default server. + * cupsSetDests2() - Save the list of destinations for the + * specified server. + * _cupsUserDefault() - Get the user default printer from + * environment variables and location + * information. + * appleCopyLocations() - Copy the location history array. + * appleCopyNetwork() - Get the network ID for the current + * location. + * appleGetPaperSize() - Get the default paper size. + * appleGetPrinter() - Get a printer from the history array. + * cups_add_dest() - Add a destination to the array. + * cups_block_cb() - Enumeration callback for block API. + * cups_compare_dests() - Compare two destinations. + * cups_dnssd_browse_cb() - Browse for printers. + * cups_dnssd_compare_device() - Compare two devices. + * cups_dnssd_free_device() - Free the memory used by a device. + * cups_dnssd_get_device() - Lookup a device and create it as needed. + * cups_dnssd_local_cb() - Browse for local printers. + * cups_dnssd_query_cb() - Process query data. + * cups_dnssd_resolve() - Resolve a Bonjour printer URI. + * cups_dnssd_resolve_cb() - See if we should continue resolving. + * cups_dnssd_unquote() - Unquote a name string. + * cups_find_dest() - Find a destination using a binary search. + * cups_get_default() - Get the default destination from an + * lpoptions file. + * cups_get_dests() - Get destinations from a file. + * cups_make_string() - Make a comma-separated string of values + * from an IPP attribute. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include + +#ifdef HAVE_NOTIFY_H +# include +#endif /* HAVE_NOTIFY_H */ + +#ifdef HAVE_POLL +# include +#endif /* HAVE_POLL */ + +#ifdef HAVE_DNSSD +# include +#endif /* HAVE_DNSSD */ + + +/* + * Constants... + */ + +#ifdef __APPLE__ +# include +# define kCUPSPrintingPrefs CFSTR("org.cups.PrintingPrefs") +# define kDefaultPaperIDKey CFSTR("DefaultPaperID") +# define kLastUsedPrintersKey CFSTR("LastUsedPrinters") +# define kLocationNetworkKey CFSTR("Network") +# define kLocationPrinterIDKey CFSTR("PrinterID") +# define kUseLastPrinter CFSTR("UseLastPrinter") +#endif /* __APPLE__ */ + + +/* + * Types... + */ + +#ifdef HAVE_DNSSD +typedef enum _cups_dnssd_state_e /* Enumerated device state */ +{ + _CUPS_DNSSD_NEW, + _CUPS_DNSSD_QUERY, + _CUPS_DNSSD_PENDING, + _CUPS_DNSSD_ACTIVE, + _CUPS_DNSSD_LOCAL, + _CUPS_DNSSD_ERROR +} _cups_dnssd_state_t; + +typedef struct _cups_dnssd_data_s /* Enumeration data */ +{ + DNSServiceRef main_ref; /* Main service reference */ + cups_dest_cb_t cb; /* Callback */ + void *user_data; /* User data pointer */ + cups_ptype_t type, /* Printer type filter */ + mask; /* Printer type mask */ + cups_array_t *devices; /* Devices found so far */ +} _cups_dnssd_data_t; + +typedef struct _cups_dnssd_device_s /* Enumerated device */ +{ + _cups_dnssd_state_t state; /* State of device listing */ + DNSServiceRef ref; /* Service reference for query */ + char *domain, /* Domain name */ + *fullName, /* Full name */ + *regtype; /* Registration type */ + cups_ptype_t type; /* Device registration type */ + cups_dest_t dest; /* Destination record */ +} _cups_dnssd_device_t; + +typedef struct _cups_dnssd_resolve_s /* Data for resolving URI */ +{ + int *cancel; /* Pointer to "cancel" variable */ + struct timeval end_time; /* Ending time */ +} _cups_dnssd_resolve_t; +#endif /* HAVE_DNSSD */ + + +/* + * Local functions... + */ + +#ifdef __APPLE__ +static CFArrayRef appleCopyLocations(void); +static CFStringRef appleCopyNetwork(void); +static char *appleGetPaperSize(char *name, int namesize); +static CFStringRef appleGetPrinter(CFArrayRef locations, CFStringRef network, + CFIndex *locindex); +#endif /* __APPLE__ */ +static cups_dest_t *cups_add_dest(const char *name, const char *instance, + int *num_dests, cups_dest_t **dests); +#ifdef __BLOCKS__ +static int cups_block_cb(cups_dest_block_t block, unsigned flags, + cups_dest_t *dest); +#endif /* __BLOCKS__ */ +static int cups_compare_dests(cups_dest_t *a, cups_dest_t *b); +#ifdef HAVE_DNSSD +static void cups_dnssd_browse_cb(DNSServiceRef sdRef, + DNSServiceFlags flags, + uint32_t interfaceIndex, + DNSServiceErrorType errorCode, + const char *serviceName, + const char *regtype, + const char *replyDomain, + void *context); +static int cups_dnssd_compare_devices(_cups_dnssd_device_t *a, + _cups_dnssd_device_t *b); +static void cups_dnssd_free_device(_cups_dnssd_device_t *device, + _cups_dnssd_data_t *data); +static _cups_dnssd_device_t * + cups_dnssd_get_device(_cups_dnssd_data_t *data, + const char *serviceName, + const char *regtype, + const char *replyDomain); +static void cups_dnssd_local_cb(DNSServiceRef sdRef, + DNSServiceFlags flags, + uint32_t interfaceIndex, + DNSServiceErrorType errorCode, + const char *serviceName, + const char *regtype, + const char *replyDomain, + void *context); +static void cups_dnssd_query_cb(DNSServiceRef sdRef, + DNSServiceFlags flags, + uint32_t interfaceIndex, + DNSServiceErrorType errorCode, + const char *fullName, + uint16_t rrtype, uint16_t rrclass, + uint16_t rdlen, const void *rdata, + uint32_t ttl, void *context); +static const char *cups_dnssd_resolve(cups_dest_t *dest, const char *uri, + int msec, int *cancel, + cups_dest_cb_t cb, void *user_data); +static int cups_dnssd_resolve_cb(void *context); +static void cups_dnssd_unquote(char *dst, const char *src, + size_t dstsize); +#endif /* HAVE_DNSSD */ +static int cups_find_dest(const char *name, const char *instance, + int num_dests, cups_dest_t *dests, int prev, + int *rdiff); +static char *cups_get_default(const char *filename, char *namebuf, + size_t namesize, const char **instance); +static int cups_get_dests(const char *filename, const char *match_name, + const char *match_inst, int user_default_set, + int num_dests, cups_dest_t **dests); +static char *cups_make_string(ipp_attribute_t *attr, char *buffer, + size_t bufsize); + + +/* + * 'cupsAddDest()' - Add a destination to the list of destinations. + * + * This function cannot be used to add a new class or printer queue, + * it only adds a new container of saved options for the named + * destination or instance. + * + * If the named destination already exists, the destination list is + * returned unchanged. Adding a new instance of a destination creates + * a copy of that destination's options. + * + * Use the @link cupsSaveDests@ function to save the updated list of + * destinations to the user's lpoptions file. + */ + +int /* O - New number of destinations */ +cupsAddDest(const char *name, /* I - Destination name */ + const char *instance, /* I - Instance name or @code NULL@ for none/primary */ + int num_dests, /* I - Number of destinations */ + cups_dest_t **dests) /* IO - Destinations */ +{ + int i; /* Looping var */ + cups_dest_t *dest; /* Destination pointer */ + cups_dest_t *parent = NULL; /* Parent destination */ + cups_option_t *doption, /* Current destination option */ + *poption; /* Current parent option */ + + + if (!name || !dests) + return (0); + + if (!cupsGetDest(name, instance, num_dests, *dests)) + { + if (instance && !cupsGetDest(name, NULL, num_dests, *dests)) + return (num_dests); + + dest = cups_add_dest(name, instance, &num_dests, dests); + + /* + * Find the base dest again now the array has been realloc'd. + */ + + parent = cupsGetDest(name, NULL, num_dests, *dests); + + if (instance && parent && parent->num_options > 0) + { + /* + * Copy options from parent... + */ + + dest->options = calloc(sizeof(cups_option_t), parent->num_options); + + if (dest->options) + { + dest->num_options = parent->num_options; + + for (i = dest->num_options, doption = dest->options, + poption = parent->options; + i > 0; + i --, doption ++, poption ++) + { + doption->name = _cupsStrRetain(poption->name); + doption->value = _cupsStrRetain(poption->value); + } + } + } + } + + return (num_dests); +} + + +#ifdef __APPLE__ +/* + * '_cupsAppleCopyDefaultPaperID()' - Get the default paper ID. + */ + +CFStringRef /* O - Default paper ID */ +_cupsAppleCopyDefaultPaperID(void) +{ + return (CFPreferencesCopyAppValue(kDefaultPaperIDKey, + kCUPSPrintingPrefs)); +} + + +/* + * '_cupsAppleCopyDefaultPrinter()' - Get the default printer at this location. + */ + +CFStringRef /* O - Default printer name */ +_cupsAppleCopyDefaultPrinter(void) +{ + CFStringRef network; /* Network location */ + CFArrayRef locations; /* Location array */ + CFStringRef locprinter; /* Current printer */ + + + /* + * Use location-based defaults only if "use last printer" is selected in the + * system preferences... + */ + + if (!_cupsAppleGetUseLastPrinter()) + { + DEBUG_puts("1_cupsAppleCopyDefaultPrinter: Not using last printer as " + "default."); + return (NULL); + } + + /* + * Get the current location... + */ + + if ((network = appleCopyNetwork()) == NULL) + { + DEBUG_puts("1_cupsAppleCopyDefaultPrinter: Unable to get current " + "network."); + return (NULL); + } + +//# ifdef DEBUG +// CFStringGetCString(network, name, namesize, kCFStringEncodingUTF8); +// DEBUG_printf(("2_cupsUserDefault: network=\"%s\"", name)); +//# endif /* DEBUG */ + + /* + * Lookup the network in the preferences... + */ + + if ((locations = appleCopyLocations()) == NULL) + { + /* + * Missing or bad location array, so no location-based default... + */ + + DEBUG_puts("1_cupsAppleCopyDefaultPrinter: Missing or bad last used " + "printer array."); + + CFRelease(network); + + return (NULL); + } + + DEBUG_printf(("1_cupsAppleCopyDefaultPrinter: Got locations, %d entries.", + (int)CFArrayGetCount(locations))); + + if ((locprinter = appleGetPrinter(locations, network, NULL)) != NULL) + CFRetain(locprinter); + + CFRelease(network); + CFRelease(locations); + + return (locprinter); +} + + +/* + * '_cupsAppleGetUseLastPrinter()' - Get whether to use the last used printer. + */ + +int /* O - 1 to use last printer, 0 otherwise */ +_cupsAppleGetUseLastPrinter(void) +{ + Boolean uselast, /* Use last printer preference value */ + uselast_set; /* Valid is set? */ + + + if (getenv("CUPS_DISABLE_APPLE_DEFAULT")) + return (0); + + uselast = CFPreferencesGetAppBooleanValue(kUseLastPrinter, + kCUPSPrintingPrefs, + &uselast_set); + if (!uselast_set) + return (1); + else + return (uselast); +} + + +/* + * '_cupsAppleSetDefaultPaperID()' - Set the default paper id. + */ + +void +_cupsAppleSetDefaultPaperID( + CFStringRef name) /* I - New paper ID */ +{ + CFPreferencesSetAppValue(kDefaultPaperIDKey, name, kCUPSPrintingPrefs); + CFPreferencesAppSynchronize(kCUPSPrintingPrefs); + notify_post("com.apple.printerPrefsChange"); +} + + +/* + * '_cupsAppleSetDefaultPrinter()' - Set the default printer for this location. + */ + +void +_cupsAppleSetDefaultPrinter( + CFStringRef name) /* I - Default printer/class name */ +{ + CFStringRef network; /* Current network */ + CFArrayRef locations; /* Old locations array */ + CFIndex locindex; /* Index in locations array */ + CFStringRef locprinter; /* Current printer */ + CFMutableArrayRef newlocations; /* New locations array */ + CFMutableDictionaryRef newlocation; /* New location */ + + + /* + * Get the current location... + */ + + if ((network = appleCopyNetwork()) == NULL) + { + DEBUG_puts("1_cupsAppleSetDefaultPrinter: Unable to get current network..."); + return; + } + + /* + * Lookup the network in the preferences... + */ + + if ((locations = appleCopyLocations()) != NULL) + locprinter = appleGetPrinter(locations, network, &locindex); + else + { + locprinter = NULL; + locindex = -1; + } + + if (!locprinter || CFStringCompare(locprinter, name, 0) != kCFCompareEqualTo) + { + /* + * Need to change the locations array... + */ + + if (locations) + { + newlocations = CFArrayCreateMutableCopy(kCFAllocatorDefault, 0, + locations); + + if (locprinter) + CFArrayRemoveValueAtIndex(newlocations, locindex); + } + else + newlocations = CFArrayCreateMutable(kCFAllocatorDefault, 0, + &kCFTypeArrayCallBacks); + + newlocation = CFDictionaryCreateMutable(kCFAllocatorDefault, 0, + &kCFTypeDictionaryKeyCallBacks, + &kCFTypeDictionaryValueCallBacks); + + if (newlocation && newlocations) + { + /* + * Put the new location at the front of the array... + */ + + CFDictionaryAddValue(newlocation, kLocationNetworkKey, network); + CFDictionaryAddValue(newlocation, kLocationPrinterIDKey, name); + CFArrayInsertValueAtIndex(newlocations, 0, newlocation); + + /* + * Limit the number of locations to 10... + */ + + while (CFArrayGetCount(newlocations) > 10) + CFArrayRemoveValueAtIndex(newlocations, 10); + + /* + * Push the changes out... + */ + + CFPreferencesSetAppValue(kLastUsedPrintersKey, newlocations, + kCUPSPrintingPrefs); + CFPreferencesAppSynchronize(kCUPSPrintingPrefs); + notify_post("com.apple.printerPrefsChange"); + } + + if (newlocations) + CFRelease(newlocations); + + if (newlocation) + CFRelease(newlocation); + } + + if (locations) + CFRelease(locations); + + CFRelease(network); +} + + +/* + * '_cupsAppleSetUseLastPrinter()' - Set whether to use the last used printer. + */ + +void +_cupsAppleSetUseLastPrinter( + int uselast) /* O - 1 to use last printer, 0 otherwise */ +{ + CFPreferencesSetAppValue(kUseLastPrinter, + uselast ? kCFBooleanTrue : kCFBooleanFalse, + kCUPSPrintingPrefs); + CFPreferencesAppSynchronize(kCUPSPrintingPrefs); + notify_post("com.apple.printerPrefsChange"); +} +#endif /* __APPLE__ */ + + +/* + * 'cupsConnectDest()' - Connect to the server for a destination. + * + * Connect to the destination, returning a new http_t connection object and + * optionally the resource path to use for the destination. These calls will + * block until a connection is made, the timeout expires, the integer pointed + * to by "cancel" is non-zero, or the callback function (or block) returns 0, + * The caller is responsible for calling httpClose() on the returned object. + * + * @since CUPS 1.6@ + */ + +http_t * /* O - Connection to server or @code NULL@ */ +cupsConnectDest( + cups_dest_t *dest, /* I - Destination */ + unsigned flags, /* I - Connection flags */ + int msec, /* I - Timeout in milliseconds */ + int *cancel, /* I - Pointer to "cancel" variable */ + char *resource, /* I - Resource buffer */ + size_t resourcesize, /* I - Size of resource buffer */ + cups_dest_cb_t cb, /* I - Callback function */ + void *user_data) /* I - User data pointer */ +{ + const char *uri; /* Printer URI */ + char scheme[32], /* URI scheme */ + userpass[256], /* Username and password (unused) */ + hostname[256], /* Hostname */ + tempresource[1024]; /* Temporary resource buffer */ + int port; /* Port number */ + char portstr[16]; /* Port number string */ + http_encryption_t encryption; /* Encryption to use */ + http_addrlist_t *addrlist; /* Address list for server */ + http_t *http; /* Connection to server */ + + + /* + * Range check input... + */ + + if (!dest) + { + if (resource) + *resource = '\0'; + + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + return (NULL); + } + + if (!resource || resourcesize < 1) + { + resource = tempresource; + resourcesize = sizeof(tempresource); + } + + /* + * Grab the printer URI... + */ + + if ((uri = cupsGetOption("printer-uri-supported", dest->num_options, + dest->options)) == NULL) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(ENOENT), 0); + + if (cb) + (*cb)(user_data, CUPS_DEST_FLAGS_UNCONNECTED | CUPS_DEST_FLAGS_ERROR, + dest); + + return (NULL); + } + +#ifdef HAVE_DNSSD + if (strstr(uri, "._tcp")) + { + if ((uri = cups_dnssd_resolve(dest, uri, msec, cancel, cb, + user_data)) == NULL) + return (NULL); + } +#endif /* HAVE_DNSSD */ + + if (httpSeparateURI(HTTP_URI_CODING_ALL, uri, scheme, sizeof(scheme), + userpass, sizeof(userpass), hostname, sizeof(hostname), + &port, resource, resourcesize) < HTTP_URI_OK) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad printer URI."), 1); + + if (cb) + (*cb)(user_data, CUPS_DEST_FLAGS_UNCONNECTED | CUPS_DEST_FLAGS_ERROR, + dest); + + return (NULL); + } + + /* + * Lookup the address for the server... + */ + + if (cb) + (*cb)(user_data, CUPS_DEST_FLAGS_UNCONNECTED | CUPS_DEST_FLAGS_RESOLVING, + dest); + + snprintf(portstr, sizeof(portstr), "%d", port); + + if ((addrlist = httpAddrGetList(hostname, AF_UNSPEC, portstr)) == NULL) + { + if (cb) + (*cb)(user_data, CUPS_DEST_FLAGS_UNCONNECTED | CUPS_DEST_FLAGS_ERROR, + dest); + + return (NULL); + } + + if (cancel && *cancel) + { + httpAddrFreeList(addrlist); + + if (cb) + (*cb)(user_data, CUPS_DEST_FLAGS_UNCONNECTED | CUPS_DEST_FLAGS_CANCELED, + dest); + + return (NULL); + } + + /* + * Create the HTTP object pointing to the server referenced by the URI... + */ + + if (!strcmp(scheme, "ipps") || port == 443) + encryption = HTTP_ENCRYPT_ALWAYS; + else + encryption = HTTP_ENCRYPT_IF_REQUESTED; + + http = _httpCreate(hostname, port, addrlist, encryption, AF_UNSPEC); + + /* + * Connect if requested... + */ + + if (flags & CUPS_DEST_FLAGS_UNCONNECTED) + { + if (cb) + (*cb)(user_data, CUPS_DEST_FLAGS_UNCONNECTED, dest); + } + else + { + if (cb) + (*cb)(user_data, CUPS_DEST_FLAGS_UNCONNECTED | CUPS_DEST_FLAGS_CONNECTING, + dest); + + if (!httpReconnect2(http, msec, cancel) && cb) + { + if (cancel && *cancel) + (*cb)(user_data, + CUPS_DEST_FLAGS_UNCONNECTED | CUPS_DEST_FLAGS_CONNECTING, dest); + else + (*cb)(user_data, CUPS_DEST_FLAGS_UNCONNECTED | CUPS_DEST_FLAGS_ERROR, + dest); + } + else if (cb) + (*cb)(user_data, CUPS_DEST_FLAGS_NONE, dest); + } + + return (http); +} + + +#ifdef __BLOCKS__ +/* + * 'cupsConnectDestBlock()' - Connect to the server for a destination. + * + * Connect to the destination, returning a new http_t connection object and + * optionally the resource path to use for the destination. These calls will + * block until a connection is made, the timeout expires, the integer pointed + * to by "cancel" is non-zero, or the callback function (or block) returns 0, + * The caller is responsible for calling httpClose() on the returned object. + * + * @since CUPS 1.6@ + */ + +http_t * /* O - Connection to server or @code NULL@ */ +cupsConnectDestBlock( + cups_dest_t *dest, /* I - Destination */ + unsigned flags, /* I - Connection flags */ + int msec, /* I - Timeout in milliseconds */ + int *cancel, /* I - Pointer to "cancel" variable */ + char *resource, /* I - Resource buffer */ + size_t resourcesize, /* I - Size of resource buffer */ + cups_dest_block_t block) /* I - Callback block */ +{ + return (cupsConnectDest(dest, flags, msec, cancel, resource, resourcesize, + (cups_dest_cb_t)cups_block_cb, (void *)block)); +} +#endif /* __BLOCKS__ */ + + +/* + * 'cupsCopyDest()' - Copy a destination. + * + * Make a copy of the destination to an array of destinations (or just a single + * copy) - for use with the cupsEnumDests* functions. The caller is responsible + * for calling cupsFreeDests() on the returned object(s). + * + * @since CUPS 1.6@ + */ + +int +cupsCopyDest(cups_dest_t *dest, + int num_dests, + cups_dest_t **dests) +{ + int i; /* Looping var */ + cups_dest_t *new_dest; /* New destination pointer */ + cups_option_t *new_option, /* Current destination option */ + *option; /* Current parent option */ + + + /* + * Range check input... + */ + + if (!dest || num_dests < 0 || !dests) + return (num_dests); + + /* + * See if the destination already exists... + */ + + if ((new_dest = cupsGetDest(dest->name, dest->instance, num_dests, + *dests)) != NULL) + { + /* + * Protect against copying destination to itself... + */ + + if (new_dest == dest) + return (num_dests); + + /* + * Otherwise, free the options... + */ + + cupsFreeOptions(new_dest->num_options, new_dest->options); + + new_dest->num_options = 0; + new_dest->options = NULL; + } + else + new_dest = cups_add_dest(dest->name, dest->instance, &num_dests, dests); + + if (new_dest) + { + if ((new_dest->options = calloc(sizeof(cups_option_t), + dest->num_options)) == NULL) + return (cupsRemoveDest(dest->name, dest->instance, num_dests, dests)); + + new_dest->num_options = dest->num_options; + + for (i = dest->num_options, option = dest->options, + new_option = new_dest->options; + i > 0; + i --, option ++, new_option ++) + { + new_option->name = _cupsStrRetain(option->name); + new_option->value = _cupsStrRetain(option->value); + } + } + + return (num_dests); +} + + +/* + * 'cupsEnumDests()' - Enumerate available destinations with a callback function. + * + * Destinations are enumerated from one or more sources. The callback function + * receives the @code user_data@ pointer, destination name, instance, number of + * options, and options which can be used as input to the @link cupsAddDest@ + * function. The function must return 1 to continue enumeration or 0 to stop. + * + * Enumeration happens on the current thread and does not return until all + * destinations have been enumerated or the callback function returns 0. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +cupsEnumDests( + unsigned flags, /* I - Enumeration flags */ + int msec, /* I - Timeout in milliseconds, + * -1 for indefinite */ + int *cancel, /* I - Pointer to "cancel" variable */ + cups_ptype_t type, /* I - Printer type bits */ + cups_ptype_t mask, /* I - Mask for printer type bits */ + cups_dest_cb_t cb, /* I - Callback function */ + void *user_data) /* I - User data */ +{ + int i, /* Looping var */ + num_dests; /* Number of destinations */ + cups_dest_t *dests, /* Destinations */ + *dest; /* Current destination */ +#ifdef HAVE_DNSSD + int nfds, /* Number of files responded */ + count, /* Number of queries started */ + remaining; /* Remainder of timeout */ + _cups_dnssd_data_t data; /* Data for callback */ + _cups_dnssd_device_t *device; /* Current device */ + int main_fd; /* File descriptor for lookups */ + DNSServiceRef ipp_ref, /* IPP browser */ + local_ipp_ref; /* Local IPP browser */ +# ifdef HAVE_SSL + DNSServiceRef ipps_ref, /* IPPS browser */ + local_ipps_ref; /* Local IPPS browser */ +# endif /* HAVE_SSL */ +# ifdef HAVE_POLL + struct pollfd pfd; /* Polling data */ +# else + fd_set input; /* Input set for select() */ + struct timeval timeout; /* Timeout for select() */ +# endif /* HAVE_POLL */ +#endif /* HAVE_DNSSD */ + + + /* + * Range check input... + */ + + if (!cb) + return (0); + + /* + * Get the list of local printers and pass them to the callback function... + */ + + num_dests = _cupsGetDests(CUPS_HTTP_DEFAULT, CUPS_GET_PRINTERS, NULL, &dests, + type, mask); + + for (i = num_dests, dest = dests; + i > 0 && (!cancel || !*cancel); + i --, dest ++) + if (!(*cb)(user_data, i > 1 ? CUPS_DEST_FLAGS_MORE : CUPS_DEST_FLAGS_NONE, + dest)) + break; + + cupsFreeDests(num_dests, dests); + + if (i > 0 || msec == 0) + return (1); + +#ifdef HAVE_DNSSD + /* + * Get Bonjour-shared printers... + */ + + data.type = type; + data.mask = mask; + data.devices = cupsArrayNew3((cups_array_func_t)cups_dnssd_compare_devices, + NULL, NULL, 0, NULL, + (cups_afree_func_t)cups_dnssd_free_device); + + if (DNSServiceCreateConnection(&data.main_ref) != kDNSServiceErr_NoError) + return (0); + + main_fd = DNSServiceRefSockFD(data.main_ref); + + ipp_ref = data.main_ref; + DNSServiceBrowse(&ipp_ref, kDNSServiceFlagsShareConnection, 0, + "_ipp._tcp,_cups", NULL, + (DNSServiceBrowseReply)cups_dnssd_browse_cb, &data); + + local_ipp_ref = data.main_ref; + DNSServiceBrowse(&local_ipp_ref, kDNSServiceFlagsShareConnection, + kDNSServiceInterfaceIndexLocalOnly, + "_ipp._tcp,_cups", NULL, + (DNSServiceBrowseReply)cups_dnssd_local_cb, &data); + +# ifdef HAVE_SSL + ipps_ref = data.main_ref; + DNSServiceBrowse(&ipps_ref, kDNSServiceFlagsShareConnection, 0, + "_ipps._tcp,_cups", NULL, + (DNSServiceBrowseReply)cups_dnssd_browse_cb, &data); + + local_ipps_ref = data.main_ref; + DNSServiceBrowse(&local_ipps_ref, kDNSServiceFlagsShareConnection, + kDNSServiceInterfaceIndexLocalOnly, + "_ipps._tcp,_cups", NULL, + (DNSServiceBrowseReply)cups_dnssd_local_cb, &data); +# endif /* HAVE_SSL */ + + if (msec < 0) + remaining = INT_MAX; + else + remaining = msec; + + while (remaining > 0 && (!cancel || !*cancel)) + { + /* + * Check for input... + */ + +# ifdef HAVE_POLL + pfd.fd = main_fd; + pfd.events = POLLIN; + + nfds = poll(&pfd, 1, remaining > 250 ? 250 : remaining); + +# else + FD_ZERO(&input); + FD_SET(main_fd, &input); + + timeout.tv_sec = 0; + timeout.tv_usec = remaining > 250 ? 250000 : remaining * 1000; + + nfds = select(main_fd + 1, &input, NULL, NULL, &timeout); +# endif /* HAVE_POLL */ + + if (nfds > 0) + DNSServiceProcessResult(data.main_ref); + else if (nfds == 0) + remaining -= 250; + + for (device = (_cups_dnssd_device_t *)cupsArrayFirst(data.devices), + count = 0; + device; + device = (_cups_dnssd_device_t *)cupsArrayNext(data.devices)) + { + if (device->ref) + count ++; + + if (!device->ref && device->state == _CUPS_DNSSD_NEW) + { + device->ref = data.main_ref; + + DEBUG_printf(("1cupsEnumDests: Querying '%s'.", device->fullName)); + + if (DNSServiceQueryRecord(&(device->ref), + kDNSServiceFlagsShareConnection, + 0, device->fullName, + kDNSServiceType_TXT, + kDNSServiceClass_IN, + (DNSServiceQueryRecordReply)cups_dnssd_query_cb, + &data) == kDNSServiceErr_NoError) + { + count ++; + } + else + { + device->ref = 0; + device->state = _CUPS_DNSSD_ERROR; + + DEBUG_puts("1cupsEnumDests: Query failed."); + } + } + else if (device->ref && device->state == _CUPS_DNSSD_PENDING) + { + if (!(*cb)(user_data, CUPS_DEST_FLAGS_NONE, &device->dest)) + { + remaining = -1; + break; + } + + device->state = _CUPS_DNSSD_ACTIVE; + } + } + } + + cupsArrayDelete(data.devices); + + DNSServiceRefDeallocate(ipp_ref); + DNSServiceRefDeallocate(local_ipp_ref); + +# ifdef HAVE_SSL + DNSServiceRefDeallocate(ipp_ref); + DNSServiceRefDeallocate(local_ipp_ref); +# endif /* HAVE_SSL */ + + DNSServiceRefDeallocate(data.main_ref); +#endif /* HAVE_DNSSD */ + + return (1); +} + + +# ifdef __BLOCKS__ +/* + * 'cupsEnumDestsBlock()' - Enumerate available destinations with a block. + * + * Destinations are enumerated from one or more sources. The block receives the + * destination name, instance, number of options, and options which can be used + * as input to the @link cupsAddDest@ function. The block must return 1 to + * continue enumeration or 0 to stop. + * + * Enumeration happens on the current thread and does not return until all + * destinations have been enumerated or the block returns 0. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +cupsEnumDestsBlock( + unsigned flags, /* I - Enumeration flags */ + int timeout, /* I - Timeout in milliseconds, 0 for indefinite */ + int *cancel, /* I - Pointer to "cancel" variable */ + cups_ptype_t type, /* I - Printer type bits */ + cups_ptype_t mask, /* I - Mask for printer type bits */ + cups_dest_block_t block) /* I - Block */ +{ + return (cupsEnumDests(flags, timeout, cancel, type, mask, + (cups_dest_cb_t)cups_block_cb, (void *)block)); +} +# endif /* __BLOCKS__ */ + + +/* + * 'cupsFreeDests()' - Free the memory used by the list of destinations. + */ + +void +cupsFreeDests(int num_dests, /* I - Number of destinations */ + cups_dest_t *dests) /* I - Destinations */ +{ + int i; /* Looping var */ + cups_dest_t *dest; /* Current destination */ + + + if (num_dests == 0 || dests == NULL) + return; + + for (i = num_dests, dest = dests; i > 0; i --, dest ++) + { + _cupsStrFree(dest->name); + _cupsStrFree(dest->instance); + + cupsFreeOptions(dest->num_options, dest->options); + } + + free(dests); +} + + +/* + * 'cupsGetDest()' - Get the named destination from the list. + * + * Use the @link cupsGetDests@ or @link cupsGetDests2@ functions to get a + * list of supported destinations for the current user. + */ + +cups_dest_t * /* O - Destination pointer or @code NULL@ */ +cupsGetDest(const char *name, /* I - Destination name or @code NULL@ for the default destination */ + const char *instance, /* I - Instance name or @code NULL@ */ + int num_dests, /* I - Number of destinations */ + cups_dest_t *dests) /* I - Destinations */ +{ + int diff, /* Result of comparison */ + match; /* Matching index */ + + + if (num_dests <= 0 || !dests) + return (NULL); + + if (!name) + { + /* + * NULL name for default printer. + */ + + while (num_dests > 0) + { + if (dests->is_default) + return (dests); + + num_dests --; + dests ++; + } + } + else + { + /* + * Lookup name and optionally the instance... + */ + + match = cups_find_dest(name, instance, num_dests, dests, -1, &diff); + + if (!diff) + return (dests + match); + } + + return (NULL); +} + + +/* + * '_cupsGetDestResource()' - Get the resource path and URI for a destination. + */ + +const char * /* O - Printer URI */ +_cupsGetDestResource( + cups_dest_t *dest, /* I - Destination */ + char *resource, /* I - Resource buffer */ + size_t resourcesize) /* I - Size of resource buffer */ +{ + const char *uri; /* Printer URI */ + char scheme[32], /* URI scheme */ + userpass[256], /* Username and password (unused) */ + hostname[256]; /* Hostname */ + int port; /* Port number */ + + + /* + * Range check input... + */ + + if (!dest || !resource || resourcesize < 1) + { + if (resource) + *resource = '\0'; + + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + return (NULL); + } + + /* + * Grab the printer URI... + */ + + if ((uri = cupsGetOption("printer-uri-supported", dest->num_options, + dest->options)) == NULL) + { + if (resource) + *resource = '\0'; + + _cupsSetError(IPP_INTERNAL_ERROR, strerror(ENOENT), 0); + + return (NULL); + } + +#ifdef HAVE_DNSSD + if (strstr(uri, "._tcp")) + { + if ((uri = cups_dnssd_resolve(dest, uri, 5000, NULL, NULL, NULL)) == NULL) + return (NULL); + } +#endif /* HAVE_DNSSD */ + + if (httpSeparateURI(HTTP_URI_CODING_ALL, uri, scheme, sizeof(scheme), + userpass, sizeof(userpass), hostname, sizeof(hostname), + &port, resource, resourcesize) < HTTP_URI_OK) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad printer URI."), 1); + + return (NULL); + } + + return (uri); +} + + +/* + * '_cupsGetDests()' - Get destinations from a server. + * + * "op" is CUPS_GET_PRINTERS to get a full list, CUPS_GET_DEFAULT to get the + * system-wide default printer, or IPP_GET_PRINTER_ATTRIBUTES for a known + * printer. + * + * "name" is the name of an existing printer and is only used when "op" is + * IPP_GET_PRINTER_ATTRIBUTES. + * + * "dest" is initialized to point to the array of destinations. + * + * 0 is returned if there are no printers, no default printer, or the named + * printer does not exist, respectively. + * + * Free the memory used by the destination array using the @link cupsFreeDests@ + * function. + * + * Note: On Mac OS X this function also gets the default paper from the system + * preferences (~/L/P/org.cups.PrintingPrefs.plist) and includes it in the + * options array for each destination that supports it. + */ + +int /* O - Number of destinations */ +_cupsGetDests(http_t *http, /* I - Connection to server or + * @code CUPS_HTTP_DEFAULT@ */ + ipp_op_t op, /* I - IPP operation */ + const char *name, /* I - Name of destination */ + cups_dest_t **dests, /* IO - Destinations */ + cups_ptype_t type, /* I - Printer type bits */ + cups_ptype_t mask) /* I - Printer type mask */ +{ + int num_dests = 0; /* Number of destinations */ + cups_dest_t *dest; /* Current destination */ + ipp_t *request, /* IPP Request */ + *response; /* IPP Response */ + ipp_attribute_t *attr; /* Current attribute */ + const char *printer_name; /* printer-name attribute */ + char uri[1024]; /* printer-uri value */ + int num_options; /* Number of options */ + cups_option_t *options; /* Options */ +#ifdef __APPLE__ + char media_default[41]; /* Default paper size */ +#endif /* __APPLE__ */ + char optname[1024], /* Option name */ + value[2048], /* Option value */ + *ptr; /* Pointer into name/value */ + static const char * const pattrs[] = /* Attributes we're interested in */ + { + "auth-info-required", + "device-uri", + "job-sheets-default", + "marker-change-time", + "marker-colors", + "marker-high-levels", + "marker-levels", + "marker-low-levels", + "marker-message", + "marker-names", + "marker-types", +#ifdef __APPLE__ + "media-supported", +#endif /* __APPLE__ */ + "printer-commands", + "printer-defaults", + "printer-info", + "printer-is-accepting-jobs", + "printer-is-shared", + "printer-location", + "printer-make-and-model", + "printer-name", + "printer-state", + "printer-state-change-time", + "printer-state-reasons", + "printer-type", + "printer-uri-supported" + }; + + +#ifdef __APPLE__ + /* + * Get the default paper size... + */ + + appleGetPaperSize(media_default, sizeof(media_default)); +#endif /* __APPLE__ */ + + /* + * Build a CUPS_GET_PRINTERS or IPP_GET_PRINTER_ATTRIBUTES request, which + * require the following attributes: + * + * attributes-charset + * attributes-natural-language + * requesting-user-name + * printer-uri [for IPP_GET_PRINTER_ATTRIBUTES] + */ + + request = ippNewRequest(op); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", sizeof(pattrs) / sizeof(pattrs[0]), + NULL, pattrs); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, cupsUser()); + + if (name && op != CUPS_GET_DEFAULT) + { + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", ippPort(), "/printers/%s", name); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", NULL, + uri); + } + else if (mask) + { + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_ENUM, "printer-type", + type); + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_ENUM, "printer-type-mask", + mask); + } + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + for (attr = response->attrs; attr != NULL; attr = attr->next) + { + /* + * Skip leading attributes until we hit a printer... + */ + + while (attr != NULL && attr->group_tag != IPP_TAG_PRINTER) + attr = attr->next; + + if (attr == NULL) + break; + + /* + * Pull the needed attributes from this printer... + */ + + printer_name = NULL; + num_options = 0; + options = NULL; + + for (; attr && attr->group_tag == IPP_TAG_PRINTER; attr = attr->next) + { + if (attr->value_tag != IPP_TAG_INTEGER && + attr->value_tag != IPP_TAG_ENUM && + attr->value_tag != IPP_TAG_BOOLEAN && + attr->value_tag != IPP_TAG_TEXT && + attr->value_tag != IPP_TAG_TEXTLANG && + attr->value_tag != IPP_TAG_NAME && + attr->value_tag != IPP_TAG_NAMELANG && + attr->value_tag != IPP_TAG_KEYWORD && + attr->value_tag != IPP_TAG_RANGE && + attr->value_tag != IPP_TAG_URI) + continue; + + if (!strcmp(attr->name, "auth-info-required") || + !strcmp(attr->name, "device-uri") || + !strcmp(attr->name, "marker-change-time") || + !strcmp(attr->name, "marker-colors") || + !strcmp(attr->name, "marker-high-levels") || + !strcmp(attr->name, "marker-levels") || + !strcmp(attr->name, "marker-low-levels") || + !strcmp(attr->name, "marker-message") || + !strcmp(attr->name, "marker-names") || + !strcmp(attr->name, "marker-types") || + !strcmp(attr->name, "printer-commands") || + !strcmp(attr->name, "printer-info") || + !strcmp(attr->name, "printer-is-shared") || + !strcmp(attr->name, "printer-make-and-model") || + !strcmp(attr->name, "printer-state") || + !strcmp(attr->name, "printer-state-change-time") || + !strcmp(attr->name, "printer-type") || + !strcmp(attr->name, "printer-is-accepting-jobs") || + !strcmp(attr->name, "printer-location") || + !strcmp(attr->name, "printer-state-reasons") || + !strcmp(attr->name, "printer-uri-supported")) + { + /* + * Add a printer description attribute... + */ + + num_options = cupsAddOption(attr->name, + cups_make_string(attr, value, + sizeof(value)), + num_options, &options); + } +#ifdef __APPLE__ + else if (!strcmp(attr->name, "media-supported")) + { + /* + * See if we can set a default media size... + */ + + int i; /* Looping var */ + + for (i = 0; i < attr->num_values; i ++) + if (!_cups_strcasecmp(media_default, attr->values[i].string.text)) + { + num_options = cupsAddOption("media", media_default, num_options, + &options); + break; + } + } +#endif /* __APPLE__ */ + else if (!strcmp(attr->name, "printer-name") && + attr->value_tag == IPP_TAG_NAME) + printer_name = attr->values[0].string.text; + else if (strncmp(attr->name, "notify-", 7) && + (attr->value_tag == IPP_TAG_BOOLEAN || + attr->value_tag == IPP_TAG_ENUM || + attr->value_tag == IPP_TAG_INTEGER || + attr->value_tag == IPP_TAG_KEYWORD || + attr->value_tag == IPP_TAG_NAME || + attr->value_tag == IPP_TAG_RANGE) && + (ptr = strstr(attr->name, "-default")) != NULL) + { + /* + * Add a default option... + */ + + strlcpy(optname, attr->name, sizeof(optname)); + optname[ptr - attr->name] = '\0'; + + if (_cups_strcasecmp(optname, "media") || + !cupsGetOption("media", num_options, options)) + num_options = cupsAddOption(optname, + cups_make_string(attr, value, + sizeof(value)), + num_options, &options); + } + } + + /* + * See if we have everything needed... + */ + + if (!printer_name) + { + cupsFreeOptions(num_options, options); + + if (attr == NULL) + break; + else + continue; + } + + if ((dest = cups_add_dest(printer_name, NULL, &num_dests, dests)) != NULL) + { + dest->num_options = num_options; + dest->options = options; + } + else + cupsFreeOptions(num_options, options); + + if (attr == NULL) + break; + } + + ippDelete(response); + } + + /* + * Return the count... + */ + + return (num_dests); +} + + +/* + * 'cupsGetDests()' - Get the list of destinations from the default server. + * + * Starting with CUPS 1.2, the returned list of destinations include the + * printer-info, printer-is-accepting-jobs, printer-is-shared, + * printer-make-and-model, printer-state, printer-state-change-time, + * printer-state-reasons, and printer-type attributes as options. CUPS 1.4 + * adds the marker-change-time, marker-colors, marker-high-levels, + * marker-levels, marker-low-levels, marker-message, marker-names, + * marker-types, and printer-commands attributes as well. + * + * Use the @link cupsFreeDests@ function to free the destination list and + * the @link cupsGetDest@ function to find a particular destination. + */ + +int /* O - Number of destinations */ +cupsGetDests(cups_dest_t **dests) /* O - Destinations */ +{ + return (cupsGetDests2(CUPS_HTTP_DEFAULT, dests)); +} + + +/* + * 'cupsGetDests2()' - Get the list of destinations from the specified server. + * + * Starting with CUPS 1.2, the returned list of destinations include the + * printer-info, printer-is-accepting-jobs, printer-is-shared, + * printer-make-and-model, printer-state, printer-state-change-time, + * printer-state-reasons, and printer-type attributes as options. CUPS 1.4 + * adds the marker-change-time, marker-colors, marker-high-levels, + * marker-levels, marker-low-levels, marker-message, marker-names, + * marker-types, and printer-commands attributes as well. + * + * Use the @link cupsFreeDests@ function to free the destination list and + * the @link cupsGetDest@ function to find a particular destination. + * + * @since CUPS 1.1.21/Mac OS X 10.4@ + */ + +int /* O - Number of destinations */ +cupsGetDests2(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + cups_dest_t **dests) /* O - Destinations */ +{ + int i; /* Looping var */ + int num_dests; /* Number of destinations */ + cups_dest_t *dest; /* Destination pointer */ + const char *home; /* HOME environment variable */ + char filename[1024]; /* Local ~/.cups/lpoptions file */ + const char *defprinter; /* Default printer */ + char name[1024], /* Copy of printer name */ + *instance, /* Pointer to instance name */ + *user_default; /* User default printer */ + int num_reals; /* Number of real queues */ + cups_dest_t *reals; /* Real queues */ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + /* + * Range check the input... + */ + + if (!dests) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad NULL dests pointer"), 1); + return (0); + } + + /* + * Grab the printers and classes... + */ + + *dests = (cups_dest_t *)0; + num_dests = _cupsGetDests(http, CUPS_GET_PRINTERS, NULL, dests, 0, 0); + + if (cupsLastError() >= IPP_REDIRECTION_OTHER_SITE) + { + cupsFreeDests(num_dests, *dests); + *dests = (cups_dest_t *)0; + return (0); + } + + /* + * Make a copy of the "real" queues for a later sanity check... + */ + + if (num_dests > 0) + { + num_reals = num_dests; + reals = calloc(num_reals, sizeof(cups_dest_t)); + + if (reals) + memcpy(reals, *dests, num_reals * sizeof(cups_dest_t)); + else + num_reals = 0; + } + else + { + num_reals = 0; + reals = NULL; + } + + /* + * Grab the default destination... + */ + + if ((user_default = _cupsUserDefault(name, sizeof(name))) != NULL) + defprinter = name; + else if ((defprinter = cupsGetDefault2(http)) != NULL) + { + strlcpy(name, defprinter, sizeof(name)); + defprinter = name; + } + + if (defprinter) + { + /* + * Separate printer and instance name... + */ + + if ((instance = strchr(name, '/')) != NULL) + *instance++ = '\0'; + + /* + * Lookup the printer and instance and make it the default... + */ + + if ((dest = cupsGetDest(name, instance, num_dests, *dests)) != NULL) + dest->is_default = 1; + } + else + instance = NULL; + + /* + * Load the /etc/cups/lpoptions and ~/.cups/lpoptions files... + */ + + snprintf(filename, sizeof(filename), "%s/lpoptions", cg->cups_serverroot); + num_dests = cups_get_dests(filename, NULL, NULL, user_default != NULL, + num_dests, dests); + + if ((home = getenv("HOME")) != NULL) + { + snprintf(filename, sizeof(filename), "%s/.cups/lpoptions", home); + + num_dests = cups_get_dests(filename, NULL, NULL, user_default != NULL, + num_dests, dests); + } + + /* + * Validate the current default destination - this prevents old + * Default lines in /etc/cups/lpoptions and ~/.cups/lpoptions from + * pointing to a non-existent printer or class... + */ + + if (num_reals) + { + /* + * See if we have a default printer... + */ + + if ((dest = cupsGetDest(NULL, NULL, num_dests, *dests)) != NULL) + { + /* + * Have a default; see if it is real... + */ + + dest = cupsGetDest(dest->name, NULL, num_reals, reals); + } + + /* + * If dest is NULL, then no default (that exists) is set, so we + * need to set a default if one exists... + */ + + if (dest == NULL && defprinter != NULL) + { + for (i = 0; i < num_dests; i ++) + (*dests)[i].is_default = 0; + + if ((dest = cupsGetDest(name, instance, num_dests, *dests)) != NULL) + dest->is_default = 1; + } + + /* + * Free memory... + */ + + free(reals); + } + + /* + * Return the number of destinations... + */ + + if (num_dests > 0) + _cupsSetError(IPP_OK, NULL, 0); + + return (num_dests); +} + + +/* + * 'cupsGetNamedDest()' - Get options for the named destination. + * + * This function is optimized for retrieving a single destination and should + * be used instead of @link cupsGetDests@ and @link cupsGetDest@ when you either + * know the name of the destination or want to print to the default destination. + * If @code NULL@ is returned, the destination does not exist or there is no + * default destination. + * + * If "http" is @code CUPS_HTTP_DEFAULT@, the connection to the default print + * server will be used. + * + * If "name" is @code NULL@, the default printer for the current user will be + * returned. + * + * The returned destination must be freed using @link cupsFreeDests@ with a + * "num_dests" value of 1. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +cups_dest_t * /* O - Destination or @code NULL@ */ +cupsGetNamedDest(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + const char *name, /* I - Destination name or @code NULL@ for the default destination */ + const char *instance) /* I - Instance name or @code NULL@ */ +{ + cups_dest_t *dest; /* Destination */ + char filename[1024], /* Path to lpoptions */ + defname[256]; /* Default printer name */ + const char *home = getenv("HOME"); /* Home directory */ + int set_as_default = 0; /* Set returned destination as default */ + ipp_op_t op = IPP_GET_PRINTER_ATTRIBUTES; + /* IPP operation to get server ops */ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + /* + * If "name" is NULL, find the default destination... + */ + + if (!name) + { + set_as_default = 1; + name = _cupsUserDefault(defname, sizeof(defname)); + + if (name) + { + char *ptr; /* Temporary pointer... */ + + if ((ptr = strchr(defname, '/')) != NULL) + { + *ptr++ = '\0'; + instance = ptr; + } + else + instance = NULL; + } + else if (home) + { + /* + * No default in the environment, try the user's lpoptions files... + */ + + snprintf(filename, sizeof(filename), "%s/.cups/lpoptions", home); + + name = cups_get_default(filename, defname, sizeof(defname), &instance); + } + + if (!name) + { + /* + * Still not there? Try the system lpoptions file... + */ + + snprintf(filename, sizeof(filename), "%s/lpoptions", + cg->cups_serverroot); + name = cups_get_default(filename, defname, sizeof(defname), &instance); + } + + if (!name) + { + /* + * No locally-set default destination, ask the server... + */ + + op = CUPS_GET_DEFAULT; + } + } + + /* + * Get the printer's attributes... + */ + + if (!_cupsGetDests(http, op, name, &dest, 0, 0)) + { + if (op == CUPS_GET_DEFAULT || (name && !set_as_default)) + return (NULL); + + /* + * The default printer from environment variables or from a + * configuration file does not exist. Find out the real default. + */ + + if (!_cupsGetDests(http, CUPS_GET_DEFAULT, NULL, &dest, 0, 0)) + return (NULL); + } + + if (instance) + dest->instance = _cupsStrAlloc(instance); + + if (set_as_default) + dest->is_default = 1; + + /* + * Then add local options... + */ + + snprintf(filename, sizeof(filename), "%s/lpoptions", cg->cups_serverroot); + cups_get_dests(filename, name, instance, 1, 1, &dest); + + if (home) + { + snprintf(filename, sizeof(filename), "%s/.cups/lpoptions", home); + + cups_get_dests(filename, name, instance, 1, 1, &dest); + } + + /* + * Return the result... + */ + + return (dest); +} + + +/* + * 'cupsRemoveDest()' - Remove a destination from the destination list. + * + * Removing a destination/instance does not delete the class or printer + * queue, merely the lpoptions for that destination/instance. Use the + * @link cupsSetDests@ or @link cupsSetDests2@ functions to save the new + * options for the user. + * + * @since CUPS 1.3/Mac OS X 10.5@ + */ + +int /* O - New number of destinations */ +cupsRemoveDest(const char *name, /* I - Destination name */ + const char *instance, /* I - Instance name or @code NULL@ */ + int num_dests, /* I - Number of destinations */ + cups_dest_t **dests) /* IO - Destinations */ +{ + int i; /* Index into destinations */ + cups_dest_t *dest; /* Pointer to destination */ + + + /* + * Find the destination... + */ + + if ((dest = cupsGetDest(name, instance, num_dests, *dests)) == NULL) + return (num_dests); + + /* + * Free memory... + */ + + _cupsStrFree(dest->name); + _cupsStrFree(dest->instance); + cupsFreeOptions(dest->num_options, dest->options); + + /* + * Remove the destination from the array... + */ + + num_dests --; + + i = dest - *dests; + + if (i < num_dests) + memmove(dest, dest + 1, (num_dests - i) * sizeof(cups_dest_t)); + + return (num_dests); +} + + +/* + * 'cupsSetDefaultDest()' - Set the default destination. + * + * @since CUPS 1.3/Mac OS X 10.5@ + */ + +void +cupsSetDefaultDest( + const char *name, /* I - Destination name */ + const char *instance, /* I - Instance name or @code NULL@ */ + int num_dests, /* I - Number of destinations */ + cups_dest_t *dests) /* I - Destinations */ +{ + int i; /* Looping var */ + cups_dest_t *dest; /* Current destination */ + + + /* + * Range check input... + */ + + if (!name || num_dests <= 0 || !dests) + return; + + /* + * Loop through the array and set the "is_default" flag for the matching + * destination... + */ + + for (i = num_dests, dest = dests; i > 0; i --, dest ++) + dest->is_default = !_cups_strcasecmp(name, dest->name) && + ((!instance && !dest->instance) || + (instance && dest->instance && + !_cups_strcasecmp(instance, dest->instance))); +} + + +/* + * 'cupsSetDests()' - Save the list of destinations for the default server. + * + * This function saves the destinations to /etc/cups/lpoptions when run + * as root and ~/.cups/lpoptions when run as a normal user. + */ + +void +cupsSetDests(int num_dests, /* I - Number of destinations */ + cups_dest_t *dests) /* I - Destinations */ +{ + cupsSetDests2(CUPS_HTTP_DEFAULT, num_dests, dests); +} + + +/* + * 'cupsSetDests2()' - Save the list of destinations for the specified server. + * + * This function saves the destinations to /etc/cups/lpoptions when run + * as root and ~/.cups/lpoptions when run as a normal user. + * + * @since CUPS 1.1.21/Mac OS X 10.4@ + */ + +int /* O - 0 on success, -1 on error */ +cupsSetDests2(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + int num_dests, /* I - Number of destinations */ + cups_dest_t *dests) /* I - Destinations */ +{ + int i, j; /* Looping vars */ + int wrote; /* Wrote definition? */ + cups_dest_t *dest; /* Current destination */ + cups_option_t *option; /* Current option */ + _ipp_option_t *match; /* Matching attribute for option */ + FILE *fp; /* File pointer */ +#ifndef WIN32 + const char *home; /* HOME environment variable */ +#endif /* WIN32 */ + char filename[1024]; /* lpoptions file */ + int num_temps; /* Number of temporary destinations */ + cups_dest_t *temps, /* Temporary destinations */ + *temp; /* Current temporary dest */ + const char *val; /* Value of temporary option */ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + /* + * Range check the input... + */ + + if (!num_dests || !dests) + return (-1); + + /* + * Get the server destinations... + */ + + num_temps = _cupsGetDests(http, CUPS_GET_PRINTERS, NULL, &temps, 0, 0); + + if (cupsLastError() >= IPP_REDIRECTION_OTHER_SITE) + { + cupsFreeDests(num_temps, temps); + return (-1); + } + + /* + * Figure out which file to write to... + */ + + snprintf(filename, sizeof(filename), "%s/lpoptions", cg->cups_serverroot); + +#ifndef WIN32 + if (getuid()) + { + /* + * Merge in server defaults... + */ + + num_temps = cups_get_dests(filename, NULL, NULL, 0, num_temps, &temps); + + /* + * Point to user defaults... + */ + + if ((home = getenv("HOME")) != NULL) + { + /* + * Create ~/.cups subdirectory... + */ + + snprintf(filename, sizeof(filename), "%s/.cups", home); + if (access(filename, 0)) + mkdir(filename, 0700); + + snprintf(filename, sizeof(filename), "%s/.cups/lpoptions", home); + } + } +#endif /* !WIN32 */ + + /* + * Try to open the file... + */ + + if ((fp = fopen(filename, "w")) == NULL) + { + cupsFreeDests(num_temps, temps); + return (-1); + } + +#ifndef WIN32 + /* + * Set the permissions to 0644 when saving to the /etc/cups/lpoptions + * file... + */ + + if (!getuid()) + fchmod(fileno(fp), 0644); +#endif /* !WIN32 */ + + /* + * Write each printer; each line looks like: + * + * Dest name[/instance] options + * Default name[/instance] options + */ + + for (i = num_dests, dest = dests; i > 0; i --, dest ++) + if (dest->instance != NULL || dest->num_options != 0 || dest->is_default) + { + if (dest->is_default) + { + fprintf(fp, "Default %s", dest->name); + if (dest->instance) + fprintf(fp, "/%s", dest->instance); + + wrote = 1; + } + else + wrote = 0; + + if ((temp = cupsGetDest(dest->name, dest->instance, num_temps, temps)) == NULL) + temp = cupsGetDest(dest->name, NULL, num_temps, temps); + + for (j = dest->num_options, option = dest->options; j > 0; j --, option ++) + { + /* + * See if this option is a printer attribute; if so, skip it... + */ + + if ((match = _ippFindOption(option->name)) != NULL && + match->group_tag == IPP_TAG_PRINTER) + continue; + + /* + * See if the server/global options match these; if so, don't + * write 'em. + */ + + if (temp && + (val = cupsGetOption(option->name, temp->num_options, + temp->options)) != NULL && + !_cups_strcasecmp(val, option->value)) + continue; + + /* + * Options don't match, write to the file... + */ + + if (!wrote) + { + fprintf(fp, "Dest %s", dest->name); + if (dest->instance) + fprintf(fp, "/%s", dest->instance); + wrote = 1; + } + + if (option->value[0]) + { + if (strchr(option->value, ' ') || + strchr(option->value, '\\') || + strchr(option->value, '\"') || + strchr(option->value, '\'')) + { + /* + * Quote the value... + */ + + fprintf(fp, " %s=\"", option->name); + + for (val = option->value; *val; val ++) + { + if (strchr("\"\'\\", *val)) + putc('\\', fp); + + putc(*val, fp); + } + + putc('\"', fp); + } + else + { + /* + * Store the literal value... + */ + + fprintf(fp, " %s=%s", option->name, option->value); + } + } + else + fprintf(fp, " %s", option->name); + } + + if (wrote) + fputs("\n", fp); + } + + /* + * Free the temporary destinations and close the file... + */ + + cupsFreeDests(num_temps, temps); + + fclose(fp); + +#ifdef __APPLE__ + /* + * Set the default printer for this location - this allows command-line + * and GUI applications to share the same default destination... + */ + + if ((dest = cupsGetDest(NULL, NULL, num_dests, dests)) != NULL) + { + CFStringRef name = CFStringCreateWithCString(kCFAllocatorDefault, + dest->name, + kCFStringEncodingUTF8); + /* Default printer name */ + + if (name) + { + _cupsAppleSetDefaultPrinter(name); + CFRelease(name); + } + } +#endif /* __APPLE__ */ + +#ifdef HAVE_NOTIFY_POST + /* + * Send a notification so that MacOS X applications can know about the + * change, too. + */ + + notify_post("com.apple.printerListChange"); +#endif /* HAVE_NOTIFY_POST */ + + return (0); +} + + +/* + * '_cupsUserDefault()' - Get the user default printer from environment + * variables and location information. + */ + +char * /* O - Default printer or NULL */ +_cupsUserDefault(char *name, /* I - Name buffer */ + size_t namesize) /* I - Size of name buffer */ +{ + const char *env; /* LPDEST or PRINTER env variable */ +#ifdef __APPLE__ + CFStringRef locprinter; /* Last printer as this location */ +#endif /* __APPLE__ */ + + + if ((env = getenv("LPDEST")) == NULL) + if ((env = getenv("PRINTER")) != NULL && !strcmp(env, "lp")) + env = NULL; + + if (env) + { + strlcpy(name, env, namesize); + return (name); + } + +#ifdef __APPLE__ + /* + * Use location-based defaults if "use last printer" is selected in the + * system preferences... + */ + + if ((locprinter = _cupsAppleCopyDefaultPrinter()) != NULL) + { + CFStringGetCString(locprinter, name, namesize, kCFStringEncodingUTF8); + CFRelease(locprinter); + } + else + name[0] = '\0'; + + DEBUG_printf(("1_cupsUserDefault: Returning \"%s\".", name)); + + return (*name ? name : NULL); + +#else + /* + * No location-based defaults on this platform... + */ + + name[0] = '\0'; + return (NULL); +#endif /* __APPLE__ */ +} + + +#ifdef __APPLE__ +/* + * 'appleCopyLocations()' - Copy the location history array. + */ + +static CFArrayRef /* O - Location array or NULL */ +appleCopyLocations(void) +{ + CFArrayRef locations; /* Location array */ + + + /* + * Look up the location array in the preferences... + */ + + if ((locations = CFPreferencesCopyAppValue(kLastUsedPrintersKey, + kCUPSPrintingPrefs)) == NULL) + return (NULL); + + if (CFGetTypeID(locations) != CFArrayGetTypeID()) + { + CFRelease(locations); + return (NULL); + } + + return (locations); +} + + +/* + * 'appleCopyNetwork()' - Get the network ID for the current location. + */ + +static CFStringRef /* O - Network ID */ +appleCopyNetwork(void) +{ + SCDynamicStoreRef dynamicStore; /* System configuration data */ + CFStringRef key; /* Current network configuration key */ + CFDictionaryRef ip_dict; /* Network configuration data */ + CFStringRef network = NULL; /* Current network ID */ + + + if ((dynamicStore = SCDynamicStoreCreate(NULL, CFSTR("libcups"), NULL, + NULL)) != NULL) + { + /* + * First use the IPv6 router address, if available, since that will generally + * be a globally-unique link-local address. + */ + + if ((key = SCDynamicStoreKeyCreateNetworkGlobalEntity( + NULL, kSCDynamicStoreDomainState, kSCEntNetIPv6)) != NULL) + { + if ((ip_dict = SCDynamicStoreCopyValue(dynamicStore, key)) != NULL) + { + if ((network = CFDictionaryGetValue(ip_dict, + kSCPropNetIPv6Router)) != NULL) + CFRetain(network); + + CFRelease(ip_dict); + } + + CFRelease(key); + } + + /* + * If that doesn't work, try the IPv4 router address. This isn't as unique + * and will likely be a 10.x.y.z or 192.168.y.z address... + */ + + if (!network) + { + if ((key = SCDynamicStoreKeyCreateNetworkGlobalEntity( + NULL, kSCDynamicStoreDomainState, kSCEntNetIPv4)) != NULL) + { + if ((ip_dict = SCDynamicStoreCopyValue(dynamicStore, key)) != NULL) + { + if ((network = CFDictionaryGetValue(ip_dict, + kSCPropNetIPv4Router)) != NULL) + CFRetain(network); + + CFRelease(ip_dict); + } + + CFRelease(key); + } + } + + CFRelease(dynamicStore); + } + + return (network); +} + + +/* + * 'appleGetPaperSize()' - Get the default paper size. + */ + +char * /* O - Default paper size */ +appleGetPaperSize(char *name, /* I - Paper size name buffer */ + int namesize) /* I - Size of buffer */ +{ + CFStringRef defaultPaperID; /* Default paper ID */ + _pwg_media_t *pwgmedia; /* PWG media size */ + + + defaultPaperID = _cupsAppleCopyDefaultPaperID(); + if (!defaultPaperID || + CFGetTypeID(defaultPaperID) != CFStringGetTypeID() || + !CFStringGetCString(defaultPaperID, name, namesize, + kCFStringEncodingUTF8)) + name[0] = '\0'; + else if ((pwgmedia = _pwgMediaForLegacy(name)) != NULL) + strlcpy(name, pwgmedia->pwg, namesize); + + if (defaultPaperID) + CFRelease(defaultPaperID); + + return (name); +} + + +/* + * 'appleGetPrinter()' - Get a printer from the history array. + */ + +static CFStringRef /* O - Printer name or NULL */ +appleGetPrinter(CFArrayRef locations, /* I - Location array */ + CFStringRef network, /* I - Network name */ + CFIndex *locindex) /* O - Index in array */ +{ + CFIndex i, /* Looping var */ + count; /* Number of locations */ + CFDictionaryRef location; /* Current location */ + CFStringRef locnetwork, /* Current network */ + locprinter; /* Current printer */ + + + for (i = 0, count = CFArrayGetCount(locations); i < count; i ++) + if ((location = CFArrayGetValueAtIndex(locations, i)) != NULL && + CFGetTypeID(location) == CFDictionaryGetTypeID()) + { + if ((locnetwork = CFDictionaryGetValue(location, + kLocationNetworkKey)) != NULL && + CFGetTypeID(locnetwork) == CFStringGetTypeID() && + CFStringCompare(network, locnetwork, 0) == kCFCompareEqualTo && + (locprinter = CFDictionaryGetValue(location, + kLocationPrinterIDKey)) != NULL && + CFGetTypeID(locprinter) == CFStringGetTypeID()) + { + if (locindex) + *locindex = i; + + return (locprinter); + } + } + + return (NULL); +} +#endif /* __APPLE__ */ + + +/* + * 'cups_add_dest()' - Add a destination to the array. + * + * Unlike cupsAddDest(), this function does not check for duplicates. + */ + +static cups_dest_t * /* O - New destination */ +cups_add_dest(const char *name, /* I - Name of destination */ + const char *instance, /* I - Instance or NULL */ + int *num_dests, /* IO - Number of destinations */ + cups_dest_t **dests) /* IO - Destinations */ +{ + int insert, /* Insertion point */ + diff; /* Result of comparison */ + cups_dest_t *dest; /* Destination pointer */ + + + /* + * Add new destination... + */ + + if (*num_dests == 0) + dest = malloc(sizeof(cups_dest_t)); + else + dest = realloc(*dests, sizeof(cups_dest_t) * (*num_dests + 1)); + + if (!dest) + return (NULL); + + *dests = dest; + + /* + * Find where to insert the destination... + */ + + if (*num_dests == 0) + insert = 0; + else + { + insert = cups_find_dest(name, instance, *num_dests, *dests, *num_dests - 1, + &diff); + + if (diff > 0) + insert ++; + } + + /* + * Move the array elements as needed... + */ + + if (insert < *num_dests) + memmove(*dests + insert + 1, *dests + insert, + (*num_dests - insert) * sizeof(cups_dest_t)); + + (*num_dests) ++; + + /* + * Initialize the destination... + */ + + dest = *dests + insert; + dest->name = _cupsStrAlloc(name); + dest->instance = _cupsStrAlloc(instance); + dest->is_default = 0; + dest->num_options = 0; + dest->options = (cups_option_t *)0; + + return (dest); +} + + +# ifdef __BLOCKS__ +/* + * 'cups_block_cb()' - Enumeration callback for block API. + */ + +static int /* O - 1 to continue, 0 to stop */ +cups_block_cb( + cups_dest_block_t block, /* I - Block */ + unsigned flags, /* I - Destination flags */ + cups_dest_t *dest) /* I - Destination */ +{ + return ((block)(flags, dest)); +} +# endif /* __BLOCKS__ */ + + +/* + * 'cups_compare_dests()' - Compare two destinations. + */ + +static int /* O - Result of comparison */ +cups_compare_dests(cups_dest_t *a, /* I - First destination */ + cups_dest_t *b) /* I - Second destination */ +{ + int diff; /* Difference */ + + + if ((diff = _cups_strcasecmp(a->name, b->name)) != 0) + return (diff); + else if (a->instance && b->instance) + return (_cups_strcasecmp(a->instance, b->instance)); + else + return ((a->instance && !b->instance) - (!a->instance && b->instance)); +} + + +#ifdef HAVE_DNSSD +/* + * 'cups_dnssd_browse_cb()' - Browse for printers. + */ + +static void +cups_dnssd_browse_cb( + DNSServiceRef sdRef, /* I - Service reference */ + DNSServiceFlags flags, /* I - Option flags */ + uint32_t interfaceIndex, /* I - Interface number */ + DNSServiceErrorType errorCode, /* I - Error, if any */ + const char *serviceName, /* I - Name of service/device */ + const char *regtype, /* I - Type of service */ + const char *replyDomain, /* I - Service domain */ + void *context) /* I - Enumeration data */ +{ + _cups_dnssd_data_t *data = (_cups_dnssd_data_t *)context; + /* Enumeration data */ + + + DEBUG_printf(("5cups_dnssd_browse_cb(sdRef=%p, flags=%x, " + "interfaceIndex=%d, errorCode=%d, serviceName=\"%s\", " + "regtype=\"%s\", replyDomain=\"%s\", context=%p)", + sdRef, flags, interfaceIndex, errorCode, serviceName, regtype, + replyDomain, context)); + + /* + * Don't do anything on error... + */ + + if (errorCode != kDNSServiceErr_NoError) + return; + + /* + * Get the device... + */ + + cups_dnssd_get_device(data, serviceName, regtype, replyDomain); +} + + +/* + * 'cups_dnssd_compare_device()' - Compare two devices. + */ + +static int /* O - Result of comparison */ +cups_dnssd_compare_devices( + _cups_dnssd_device_t *a, /* I - First device */ + _cups_dnssd_device_t *b) /* I - Second device */ +{ + return (strcmp(a->dest.name, b->dest.name)); +} + + +/* + * 'cups_dnssd_free_device()' - Free the memory used by a device. + */ + +static void +cups_dnssd_free_device( + _cups_dnssd_device_t *device, /* I - Device */ + _cups_dnssd_data_t *data) /* I - Enumeration data */ +{ + DEBUG_printf(("5cups_dnssd_free_device(device=%p(%s), data=%p)", device, + device->dest.name, data)); + + if (device->ref) + DNSServiceRefDeallocate(device->ref); + + _cupsStrFree(device->domain); + _cupsStrFree(device->fullName); + _cupsStrFree(device->regtype); + _cupsStrFree(device->dest.name); + + cupsFreeOptions(device->dest.num_options, device->dest.options); + + free(device); +} + + +/* + * 'cups_dnssd_get_device()' - Lookup a device and create it as needed. + */ + +static _cups_dnssd_device_t * /* O - Device */ +cups_dnssd_get_device( + _cups_dnssd_data_t *data, /* I - Enumeration data */ + const char *serviceName, /* I - Service name */ + const char *regtype, /* I - Registration type */ + const char *replyDomain) /* I - Domain name */ +{ + _cups_dnssd_device_t key, /* Search key */ + *device; /* Device */ + char fullName[kDNSServiceMaxDomainName]; + /* Full name for query */ + + + DEBUG_printf(("5cups_dnssd_get_device(data=%p, serviceName=\"%s\", " + "regtype=\"%s\", replyDomain=\"%s\")", data, serviceName, + regtype, replyDomain)); + + /* + * See if this is an existing device... + */ + + key.dest.name = (char *)serviceName; + + if ((device = cupsArrayFind(data->devices, &key)) != NULL) + { + /* + * Yes, see if we need to do anything with this... + */ + + int update = 0; /* Non-zero if we need to update */ + + if (!_cups_strcasecmp(replyDomain, "local.") && + _cups_strcasecmp(device->domain, replyDomain)) + { + /* + * Update the "global" listing to use the .local domain name instead. + */ + + _cupsStrFree(device->domain); + device->domain = _cupsStrAlloc(replyDomain); + + DEBUG_printf(("6cups_dnssd_get_device: Updating '%s' to use local " + "domain.", device->dest.name)); + + update = 1; + } + + if (!_cups_strcasecmp(regtype, "_ipps._tcp") && + _cups_strcasecmp(device->regtype, regtype)) + { + /* + * Prefer IPPS over IPP. + */ + + _cupsStrFree(device->regtype); + device->regtype = _cupsStrAlloc(regtype); + + DEBUG_printf(("6cups_dnssd_get_device: Updating '%s' to use IPPS.", + device->dest.name)); + + update = 1; + } + + if (!update) + { + DEBUG_printf(("6cups_dnssd_get_device: No changes to '%s'.", + device->dest.name)); + return (device); + } + } + else + { + /* + * No, add the device... + */ + + DEBUG_printf(("6cups_dnssd_get_device: Adding '%s' for %s with domain " + "'%s'.", serviceName, + !strcmp(regtype, "_ipps._tcp") ? "IPPS" : "IPP", + replyDomain)); + + device = calloc(sizeof(_cups_dnssd_device_t), 1); + device->dest.name = _cupsStrAlloc(serviceName); + device->domain = _cupsStrAlloc(replyDomain); + device->regtype = _cupsStrAlloc(regtype); + + cupsArrayAdd(data->devices, device); + } + + /* + * Set the "full name" of this service, which is used for queries... + */ + + DNSServiceConstructFullName(fullName, device->dest.name, device->regtype, + device->domain); + _cupsStrFree(device->fullName); + device->fullName = _cupsStrAlloc(fullName); + + if (device->ref) + { + DNSServiceRefDeallocate(device->ref); + device->ref = 0; + } + + if (device->state == _CUPS_DNSSD_ACTIVE) + { + (*data->cb)(data->user_data, CUPS_DEST_FLAGS_REMOVED, &device->dest); + device->state = _CUPS_DNSSD_NEW; + } + + return (device); +} + + +/* + * 'cups_dnssd_local_cb()' - Browse for local printers. + */ + +static void +cups_dnssd_local_cb( + DNSServiceRef sdRef, /* I - Service reference */ + DNSServiceFlags flags, /* I - Option flags */ + uint32_t interfaceIndex, /* I - Interface number */ + DNSServiceErrorType errorCode, /* I - Error, if any */ + const char *serviceName, /* I - Name of service/device */ + const char *regtype, /* I - Type of service */ + const char *replyDomain, /* I - Service domain */ + void *context) /* I - Devices array */ +{ + _cups_dnssd_data_t *data = (_cups_dnssd_data_t *)context; + /* Enumeration data */ + _cups_dnssd_device_t *device; /* Device */ + + + DEBUG_printf(("5cups_dnssd_local_cb(sdRef=%p, flags=%x, " + "interfaceIndex=%d, errorCode=%d, serviceName=\"%s\", " + "regtype=\"%s\", replyDomain=\"%s\", context=%p)", + sdRef, flags, interfaceIndex, errorCode, serviceName, + regtype, replyDomain, context)); + + /* + * Only process "add" data... + */ + + if (errorCode != kDNSServiceErr_NoError || !(flags & kDNSServiceFlagsAdd)) + return; + + /* + * Get the device... + */ + + device = cups_dnssd_get_device(data, serviceName, regtype, replyDomain); + + /* + * Hide locally-registered devices... + */ + + DEBUG_printf(("6cups_dnssd_local_cb: Hiding local printer '%s'.", + serviceName)); + + if (device->ref) + { + DNSServiceRefDeallocate(device->ref); + device->ref = 0; + } + + if (device->state == _CUPS_DNSSD_ACTIVE) + (*data->cb)(data->user_data, CUPS_DEST_FLAGS_REMOVED, &device->dest); + + device->state = _CUPS_DNSSD_LOCAL; +} + + +/* + * 'cups_dnssd_query_cb()' - Process query data. + */ + +static void +cups_dnssd_query_cb( + DNSServiceRef sdRef, /* I - Service reference */ + DNSServiceFlags flags, /* I - Data flags */ + uint32_t interfaceIndex, /* I - Interface */ + DNSServiceErrorType errorCode, /* I - Error, if any */ + const char *fullName, /* I - Full service name */ + uint16_t rrtype, /* I - Record type */ + uint16_t rrclass, /* I - Record class */ + uint16_t rdlen, /* I - Length of record data */ + const void *rdata, /* I - Record data */ + uint32_t ttl, /* I - Time-to-live */ + void *context) /* I - Enumeration data */ +{ + _cups_dnssd_data_t *data = (_cups_dnssd_data_t *)context; + /* Enumeration data */ + char name[1024], /* Service name */ + *ptr; /* Pointer into string */ + _cups_dnssd_device_t dkey, /* Search key */ + *device; /* Device */ + + + DEBUG_printf(("5cups_dnssd_query_cb(sdRef=%p, flags=%x, " + "interfaceIndex=%d, errorCode=%d, fullName=\"%s\", " + "rrtype=%u, rrclass=%u, rdlen=%u, rdata=%p, ttl=%u, " + "context=%p)", sdRef, flags, interfaceIndex, errorCode, + fullName, rrtype, rrclass, rdlen, rdata, ttl, context)); + + /* + * Only process "add" data... + */ + + if (errorCode != kDNSServiceErr_NoError || !(flags & kDNSServiceFlagsAdd)) + return; + + /* + * Lookup the service in the devices array. + */ + + dkey.dest.name = name; + + cups_dnssd_unquote(name, fullName, sizeof(name)); + + if ((ptr = strstr(name, "._")) != NULL) + *ptr = '\0'; + + if ((device = cupsArrayFind(data->devices, &dkey)) != NULL) + { + /* + * Found it, pull out the priority and make and model from the TXT + * record and save it... + */ + + const uint8_t *txt, /* Pointer into data */ + *txtnext, /* Next key/value pair */ + *txtend; /* End of entire TXT record */ + uint8_t txtlen; /* Length of current key/value pair */ + char key[256], /* Key string */ + value[256], /* Value string */ + make_and_model[512], + /* Manufacturer and model */ + model[256], /* Model */ + uriname[1024], /* Name for URI */ + uri[1024]; /* Printer URI */ + cups_ptype_t type; /* Device type */ + + device->state = _CUPS_DNSSD_PENDING; + make_and_model[0] = '\0'; + type = CUPS_PRINTER_REMOTE; + + strcpy(model, "Unknown"); + + for (txt = rdata, txtend = txt + rdlen; + txt < txtend; + txt = txtnext) + { + /* + * Read a key/value pair starting with an 8-bit length. Since the + * length is 8 bits and the size of the key/value buffers is 256, we + * don't need to check for overflow... + */ + + txtlen = *txt++; + + if (!txtlen || (txt + txtlen) > txtend) + break; + + txtnext = txt + txtlen; + + for (ptr = key; txt < txtnext && *txt != '='; txt ++) + *ptr++ = *txt; + *ptr = '\0'; + + if (txt < txtnext && *txt == '=') + { + txt ++; + + if (txt < txtnext) + memcpy(value, txt, txtnext - txt); + value[txtnext - txt] = '\0'; + + DEBUG_printf(("6cups_dnssd_query_cb: %s=%s", key, value)); + } + else + { + DEBUG_printf(("6cups_dnssd_query_cb: '%s' with no value.", key)); + continue; + } + + if (!_cups_strcasecmp(key, "usb_MFG") || + !_cups_strcasecmp(key, "usb_MANU") || + !_cups_strcasecmp(key, "usb_MANUFACTURER")) + strcpy(make_and_model, value); + else if (!_cups_strcasecmp(key, "usb_MDL") || + !_cups_strcasecmp(key, "usb_MODEL")) + strcpy(model, value); + else if (!_cups_strcasecmp(key, "product") && !strstr(value, "Ghostscript")) + { + if (value[0] == '(') + { + /* + * Strip parenthesis... + */ + + if ((ptr = value + strlen(value) - 1) > value && *ptr == ')') + *ptr = '\0'; + + strcpy(model, value + 1); + } + else + strcpy(model, value); + } + else if (!_cups_strcasecmp(key, "ty")) + { + strcpy(model, value); + + if ((ptr = strchr(model, ',')) != NULL) + *ptr = '\0'; + } + else if (!_cups_strcasecmp(key, "printer-type")) + { + device->dest.num_options = cupsAddOption("printer-type", value, + device->dest.num_options, + &device->dest.options); + type = strtol(value, NULL, 0); + } + } + + /* + * Save the make-and-model... + */ + + if (make_and_model[0]) + { + strlcat(make_and_model, " ", sizeof(make_and_model)); + strlcat(make_and_model, model, sizeof(make_and_model)); + + device->dest.num_options = cupsAddOption("printer-make-and-model", + make_and_model, + device->dest.num_options, + &device->dest.options); + } + else + device->dest.num_options = cupsAddOption("printer-make-and-model", + model, + device->dest.num_options, + &device->dest.options); + + /* + * Save the URI... + */ + + cups_dnssd_unquote(uriname, device->fullName, sizeof(uriname)); + httpAssembleURI(HTTP_URI_CODING_ALL, uri, sizeof(uri), + !strcmp(device->regtype, "_ipps._tcp") ? "ipps" : "ipp", + NULL, uriname, 0, "/cups"); + + DEBUG_printf(("6cups_dnssd_query: printer-uri-supported=\"%s\"", uri)); + + device->dest.num_options = cupsAddOption("printer-uri-supported", uri, + device->dest.num_options, + &device->dest.options); + } + else + DEBUG_printf(("6cups_dnssd_query: Ignoring TXT record for '%s'.", + fullName)); +} + + +/* + * 'cups_dnssd_resolve()' - Resolve a Bonjour printer URI. + */ + +static const char * /* O - Resolved URI or NULL */ +cups_dnssd_resolve( + cups_dest_t *dest, /* I - Destination */ + const char *uri, /* I - Current printer URI */ + int msec, /* I - Time in milliseconds */ + int *cancel, /* I - Pointer to "cancel" variable */ + cups_dest_cb_t cb, /* I - Callback */ + void *user_data) /* I - User data for callback */ +{ + char tempuri[1024]; /* Temporary URI buffer */ + _cups_dnssd_resolve_t resolve; /* Resolve data */ + + + /* + * Resolve the URI... + */ + + resolve.cancel = cancel; + gettimeofday(&resolve.end_time, NULL); + if (msec > 0) + { + resolve.end_time.tv_sec += msec / 1000; + resolve.end_time.tv_usec += (msec % 1000) * 1000; + + while (resolve.end_time.tv_usec >= 1000000) + { + resolve.end_time.tv_sec ++; + resolve.end_time.tv_usec -= 1000000; + } + } + else + resolve.end_time.tv_sec += 75; + + if (cb) + (*cb)(user_data, CUPS_DEST_FLAGS_UNCONNECTED | CUPS_DEST_FLAGS_RESOLVING, + dest); + + if ((uri = _httpResolveURI(uri, tempuri, sizeof(tempuri), + _HTTP_RESOLVE_FQDN, cups_dnssd_resolve_cb, + &resolve)) == NULL) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("Unable to resolve printer URI."), 1); + + if (cb) + (*cb)(user_data, CUPS_DEST_FLAGS_UNCONNECTED | CUPS_DEST_FLAGS_ERROR, + dest); + + return (NULL); + } + + /* + * Save the resolved URI... + */ + + dest->num_options = cupsAddOption("printer-uri-supported", uri, + dest->num_options, &dest->options); + + return (cupsGetOption("printer-uri-supported", dest->num_options, + dest->options)); +} + + +/* + * 'cups_dnssd_resolve_cb()' - See if we should continue resolving. + */ + +static int /* O - 1 to continue, 0 to stop */ +cups_dnssd_resolve_cb(void *context) /* I - Resolve data */ +{ + _cups_dnssd_resolve_t *resolve = (_cups_dnssd_resolve_t *)context; + /* Resolve data */ + struct timeval curtime; /* Current time */ + + + /* + * If the cancel variable is set, return immediately. + */ + + if (*resolve->cancel) + return (0); + + /* + * Otherwise check the end time... + */ + + gettimeofday(&curtime, NULL); + + return (curtime.tv_sec > resolve->end_time.tv_sec || + (curtime.tv_sec == resolve->end_time.tv_sec && + curtime.tv_usec > resolve->end_time.tv_usec)); +} + + +/* + * 'cups_dnssd_unquote()' - Unquote a name string. + */ + +static void +cups_dnssd_unquote(char *dst, /* I - Destination buffer */ + const char *src, /* I - Source string */ + size_t dstsize) /* I - Size of destination buffer */ +{ + char *dstend = dst + dstsize - 1; /* End of destination buffer */ + + + while (*src && dst < dstend) + { + if (*src == '\\') + { + src ++; + if (isdigit(src[0] & 255) && isdigit(src[1] & 255) && + isdigit(src[2] & 255)) + { + *dst++ = ((((src[0] - '0') * 10) + src[1] - '0') * 10) + src[2] - '0'; + src += 3; + } + else + *dst++ = *src++; + } + else + *dst++ = *src ++; + } + + *dst = '\0'; +} +#endif /* HAVE_DNSSD */ + + +/* + * 'cups_find_dest()' - Find a destination using a binary search. + */ + +static int /* O - Index of match */ +cups_find_dest(const char *name, /* I - Destination name */ + const char *instance, /* I - Instance or NULL */ + int num_dests, /* I - Number of destinations */ + cups_dest_t *dests, /* I - Destinations */ + int prev, /* I - Previous index */ + int *rdiff) /* O - Difference of match */ +{ + int left, /* Low mark for binary search */ + right, /* High mark for binary search */ + current, /* Current index */ + diff; /* Result of comparison */ + cups_dest_t key; /* Search key */ + + + key.name = (char *)name; + key.instance = (char *)instance; + + if (prev >= 0) + { + /* + * Start search on either side of previous... + */ + + if ((diff = cups_compare_dests(&key, dests + prev)) == 0 || + (diff < 0 && prev == 0) || + (diff > 0 && prev == (num_dests - 1))) + { + *rdiff = diff; + return (prev); + } + else if (diff < 0) + { + /* + * Start with previous on right side... + */ + + left = 0; + right = prev; + } + else + { + /* + * Start wih previous on left side... + */ + + left = prev; + right = num_dests - 1; + } + } + else + { + /* + * Start search in the middle... + */ + + left = 0; + right = num_dests - 1; + } + + do + { + current = (left + right) / 2; + diff = cups_compare_dests(&key, dests + current); + + if (diff == 0) + break; + else if (diff < 0) + right = current; + else + left = current; + } + while ((right - left) > 1); + + if (diff != 0) + { + /* + * Check the last 1 or 2 elements... + */ + + if ((diff = cups_compare_dests(&key, dests + left)) <= 0) + current = left; + else + { + diff = cups_compare_dests(&key, dests + right); + current = right; + } + } + + /* + * Return the closest destination and the difference... + */ + + *rdiff = diff; + + return (current); +} + + +/* + * 'cups_get_default()' - Get the default destination from an lpoptions file. + */ + +static char * /* O - Default destination or NULL */ +cups_get_default(const char *filename, /* I - File to read */ + char *namebuf, /* I - Name buffer */ + size_t namesize, /* I - Size of name buffer */ + const char **instance) /* I - Instance */ +{ + cups_file_t *fp; /* lpoptions file */ + char line[8192], /* Line from file */ + *value, /* Value for line */ + *nameptr; /* Pointer into name */ + int linenum; /* Current line */ + + + *namebuf = '\0'; + + if ((fp = cupsFileOpen(filename, "r")) != NULL) + { + linenum = 0; + + while (cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + { + if (!_cups_strcasecmp(line, "default") && value) + { + strlcpy(namebuf, value, namesize); + + if ((nameptr = strchr(namebuf, ' ')) != NULL) + *nameptr = '\0'; + if ((nameptr = strchr(namebuf, '\t')) != NULL) + *nameptr = '\0'; + + if ((nameptr = strchr(namebuf, '/')) != NULL) + *nameptr++ = '\0'; + + *instance = nameptr; + break; + } + } + + cupsFileClose(fp); + } + + return (*namebuf ? namebuf : NULL); +} + + +/* + * 'cups_get_dests()' - Get destinations from a file. + */ + +static int /* O - Number of destinations */ +cups_get_dests( + const char *filename, /* I - File to read from */ + const char *match_name, /* I - Destination name we want */ + const char *match_inst, /* I - Instance name we want */ + int user_default_set, /* I - User default printer set? */ + int num_dests, /* I - Number of destinations */ + cups_dest_t **dests) /* IO - Destinations */ +{ + int i; /* Looping var */ + cups_dest_t *dest; /* Current destination */ + cups_file_t *fp; /* File pointer */ + char line[8192], /* Line from file */ + *lineptr, /* Pointer into line */ + *name, /* Name of destination/option */ + *instance; /* Instance of destination */ + int linenum; /* Current line number */ + + + DEBUG_printf(("7cups_get_dests(filename=\"%s\", match_name=\"%s\", " + "match_inst=\"%s\", user_default_set=%d, num_dests=%d, " + "dests=%p)", filename, match_name, match_inst, + user_default_set, num_dests, dests)); + + /* + * Try to open the file... + */ + + if ((fp = cupsFileOpen(filename, "r")) == NULL) + return (num_dests); + + /* + * Read each printer; each line looks like: + * + * Dest name[/instance] options + * Default name[/instance] options + */ + + linenum = 0; + + while (cupsFileGetConf(fp, line, sizeof(line), &lineptr, &linenum)) + { + /* + * See what type of line it is... + */ + + DEBUG_printf(("9cups_get_dests: linenum=%d line=\"%s\" lineptr=\"%s\"", + linenum, line, lineptr)); + + if ((_cups_strcasecmp(line, "dest") && _cups_strcasecmp(line, "default")) || !lineptr) + { + DEBUG_puts("9cups_get_dests: Not a dest or default line..."); + continue; + } + + name = lineptr; + + /* + * Search for an instance... + */ + + while (!isspace(*lineptr & 255) && *lineptr && *lineptr != '/') + lineptr ++; + + if (*lineptr == '/') + { + /* + * Found an instance... + */ + + *lineptr++ = '\0'; + instance = lineptr; + + /* + * Search for an instance... + */ + + while (!isspace(*lineptr & 255) && *lineptr) + lineptr ++; + } + else + instance = NULL; + + if (*lineptr) + *lineptr++ = '\0'; + + DEBUG_printf(("9cups_get_dests: name=\"%s\", instance=\"%s\"", name, + instance)); + + /* + * See if the primary instance of the destination exists; if not, + * ignore this entry and move on... + */ + + if (match_name) + { + if (_cups_strcasecmp(name, match_name) || + (!instance && match_inst) || + (instance && !match_inst) || + (instance && _cups_strcasecmp(instance, match_inst))) + continue; + + dest = *dests; + } + else if (cupsGetDest(name, NULL, num_dests, *dests) == NULL) + { + DEBUG_puts("9cups_get_dests: Not found!"); + continue; + } + else + { + /* + * Add the destination... + */ + + num_dests = cupsAddDest(name, instance, num_dests, dests); + + if ((dest = cupsGetDest(name, instance, num_dests, *dests)) == NULL) + { + /* + * Out of memory! + */ + + DEBUG_puts("9cups_get_dests: Out of memory!"); + break; + } + } + + /* + * Add options until we hit the end of the line... + */ + + dest->num_options = cupsParseOptions(lineptr, dest->num_options, + &(dest->options)); + + /* + * If we found what we were looking for, stop now... + */ + + if (match_name) + break; + + /* + * Set this as default if needed... + */ + + if (!user_default_set && !_cups_strcasecmp(line, "default")) + { + DEBUG_puts("9cups_get_dests: Setting as default..."); + + for (i = 0; i < num_dests; i ++) + (*dests)[i].is_default = 0; + + dest->is_default = 1; + } + } + + /* + * Close the file and return... + */ + + cupsFileClose(fp); + + return (num_dests); +} + + +/* + * 'cups_make_string()' - Make a comma-separated string of values from an IPP + * attribute. + */ + +static char * /* O - New string */ +cups_make_string( + ipp_attribute_t *attr, /* I - Attribute to convert */ + char *buffer, /* I - Buffer */ + size_t bufsize) /* I - Size of buffer */ +{ + int i; /* Looping var */ + char *ptr, /* Pointer into buffer */ + *end, /* Pointer to end of buffer */ + *valptr; /* Pointer into string attribute */ + + + /* + * Return quickly if we have a single string value... + */ + + if (attr->num_values == 1 && + attr->value_tag != IPP_TAG_INTEGER && + attr->value_tag != IPP_TAG_ENUM && + attr->value_tag != IPP_TAG_BOOLEAN && + attr->value_tag != IPP_TAG_RANGE) + return (attr->values[0].string.text); + + /* + * Copy the values to the string, separating with commas and escaping strings + * as needed... + */ + + end = buffer + bufsize - 1; + + for (i = 0, ptr = buffer; i < attr->num_values && ptr < end; i ++) + { + if (i) + *ptr++ = ','; + + switch (attr->value_tag) + { + case IPP_TAG_INTEGER : + case IPP_TAG_ENUM : + snprintf(ptr, end - ptr + 1, "%d", attr->values[i].integer); + break; + + case IPP_TAG_BOOLEAN : + if (attr->values[i].boolean) + strlcpy(ptr, "true", end - ptr + 1); + else + strlcpy(ptr, "false", end - ptr + 1); + break; + + case IPP_TAG_RANGE : + if (attr->values[i].range.lower == attr->values[i].range.upper) + snprintf(ptr, end - ptr + 1, "%d", attr->values[i].range.lower); + else + snprintf(ptr, end - ptr + 1, "%d-%d", attr->values[i].range.lower, + attr->values[i].range.upper); + break; + + default : + for (valptr = attr->values[i].string.text; + *valptr && ptr < end;) + { + if (strchr(" \t\n\\\'\"", *valptr)) + { + if (ptr >= (end - 1)) + break; + + *ptr++ = '\\'; + } + + *ptr++ = *valptr++; + } + + *ptr = '\0'; + break; + } + + ptr += strlen(ptr); + } + + *ptr = '\0'; + + return (buffer); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/dir.c b/cups/dir.c new file mode 100644 index 0000000000..490296e091 --- /dev/null +++ b/cups/dir.c @@ -0,0 +1,472 @@ +/* + * "$Id$" + * + * Directory routines for CUPS. + * + * This set of APIs abstracts enumeration of directory entries. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * _cups_dir_time() - Convert a FILETIME value to a UNIX time value. + * cupsDirClose() - Close a directory. + * cupsDirOpen() - Open a directory. + * cupsDirRead() - Read the next directory entry. + * cupsDirRewind() - Rewind to the start of the directory. + * cupsDirClose() - Close a directory. + * cupsDirOpen() - Open a directory. + * cupsDirRead() - Read the next directory entry. + * cupsDirRewind() - Rewind to the start of the directory. + */ + +/* + * Include necessary headers... + */ + +#include "string-private.h" +#include "debug-private.h" +#include "dir.h" + + +/* + * Windows implementation... + */ + +#ifdef WIN32 +# include + +/* + * Types and structures... + */ + +struct _cups_dir_s /**** Directory data structure ****/ +{ + char directory[1024]; /* Directory filename */ + HANDLE dir; /* Directory handle */ + cups_dentry_t entry; /* Directory entry */ +}; + + +/* + * '_cups_dir_time()' - Convert a FILETIME value to a UNIX time value. + */ + +time_t /* O - UNIX time */ +_cups_dir_time(FILETIME ft) /* I - File time */ +{ + ULONGLONG val; /* File time in 0.1 usecs */ + + + /* + * Convert file time (1/10 microseconds since Jan 1, 1601) to UNIX + * time (seconds since Jan 1, 1970). There are 11,644,732,800 seconds + * between them... + */ + + val = ft.dwLowDateTime + ((ULONGLONG)ft.dwHighDateTime << 32); + return ((time_t)(val / 10000000 - 11644732800)); +} + + +/* + * 'cupsDirClose()' - Close a directory. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void +cupsDirClose(cups_dir_t *dp) /* I - Directory pointer */ +{ + /* + * Range check input... + */ + + if (!dp) + return; + + /* + * Close an open directory handle... + */ + + if (dp->dir != INVALID_HANDLE_VALUE) + FindClose(dp->dir); + + /* + * Free memory used... + */ + + free(dp); +} + + +/* + * 'cupsDirOpen()' - Open a directory. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +cups_dir_t * /* O - Directory pointer or @code NULL@ if the directory could not be opened. */ +cupsDirOpen(const char *directory) /* I - Directory name */ +{ + cups_dir_t *dp; /* Directory */ + + + /* + * Range check input... + */ + + if (!directory) + return (NULL); + + /* + * Allocate memory for the directory structure... + */ + + dp = (cups_dir_t *)calloc(1, sizeof(cups_dir_t)); + if (!dp) + return (NULL); + + /* + * Copy the directory name for later use... + */ + + dp->dir = INVALID_HANDLE_VALUE; + + strlcpy(dp->directory, directory, sizeof(dp->directory)); + + /* + * Return the new directory structure... + */ + + return (dp); +} + + +/* + * 'cupsDirRead()' - Read the next directory entry. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +cups_dentry_t * /* O - Directory entry or @code NULL@ if there are no more */ +cupsDirRead(cups_dir_t *dp) /* I - Directory pointer */ +{ + WIN32_FIND_DATA entry; /* Directory entry data */ + + + /* + * Range check input... + */ + + if (!dp) + return (NULL); + + /* + * See if we have already started finding files... + */ + + if (dp->dir == INVALID_HANDLE_VALUE) + { + /* + * No, find the first file... + */ + + dp->dir = FindFirstFile(dp->directory, &entry); + if (dp->dir == INVALID_HANDLE_VALUE) + return (NULL); + } + else if (!FindNextFile(dp->dir, &entry)) + return (NULL); + + /* + * Copy the name over and convert the file information... + */ + + strlcpy(dp->entry.filename, entry.cFileName, sizeof(dp->entry.filename)); + + if (entry.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) + dp->entry.fileinfo.st_mode = 0755 | S_IFDIR; + else + dp->entry.fileinfo.st_mode = 0644; + + dp->entry.fileinfo.st_atime = _cups_dir_time(entry.ftLastAccessTime); + dp->entry.fileinfo.st_ctime = _cups_dir_time(entry.ftCreationTime); + dp->entry.fileinfo.st_mtime = _cups_dir_time(entry.ftLastWriteTime); + dp->entry.fileinfo.st_size = entry.nFileSizeLow + ((unsigned long long)entry.nFileSizeHigh << 32); + + /* + * Return the entry... + */ + + return (&(dp->entry)); +} + + +/* + * 'cupsDirRewind()' - Rewind to the start of the directory. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void +cupsDirRewind(cups_dir_t *dp) /* I - Directory pointer */ +{ + /* + * Range check input... + */ + + if (!dp) + return; + + /* + * Close an open directory handle... + */ + + if (dp->dir != INVALID_HANDLE_VALUE) + { + FindClose(dp->dir); + dp->dir = INVALID_HANDLE_VALUE; + } +} + + +#else + +/* + * POSIX implementation... + */ + +# include +# include + + +/* + * Types and structures... + */ + +struct _cups_dir_s /**** Directory data structure ****/ +{ + char directory[1024]; /* Directory filename */ + DIR *dir; /* Directory file */ + cups_dentry_t entry; /* Directory entry */ +}; + + +/* + * 'cupsDirClose()' - Close a directory. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void +cupsDirClose(cups_dir_t *dp) /* I - Directory pointer */ +{ + DEBUG_printf(("cupsDirClose(dp=%p)", dp)); + + /* + * Range check input... + */ + + if (!dp) + return; + + /* + * Close the directory and free memory... + */ + + closedir(dp->dir); + free(dp); +} + + +/* + * 'cupsDirOpen()' - Open a directory. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +cups_dir_t * /* O - Directory pointer or @code NULL@ if the directory could not be opened. */ +cupsDirOpen(const char *directory) /* I - Directory name */ +{ + cups_dir_t *dp; /* Directory */ + + + DEBUG_printf(("cupsDirOpen(directory=\"%s\")", directory)); + + /* + * Range check input... + */ + + if (!directory) + return (NULL); + + /* + * Allocate memory for the directory structure... + */ + + dp = (cups_dir_t *)calloc(1, sizeof(cups_dir_t)); + if (!dp) + return (NULL); + + /* + * Open the directory... + */ + + dp->dir = opendir(directory); + if (!dp->dir) + { + free(dp); + return (NULL); + } + + /* + * Copy the directory name for later use... + */ + + strlcpy(dp->directory, directory, sizeof(dp->directory)); + + /* + * Return the new directory structure... + */ + + return (dp); +} + + +/* + * 'cupsDirRead()' - Read the next directory entry. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +cups_dentry_t * /* O - Directory entry or @code NULL@ when there are no more */ +cupsDirRead(cups_dir_t *dp) /* I - Directory pointer */ +{ + struct dirent *entry; /* Pointer to entry */ + char filename[1024]; /* Full filename */ +# ifdef HAVE_PTHREAD_H + char buffer[sizeof(struct dirent) + 1024]; + /* Directory entry buffer */ +# endif /* HAVE_PTHREAD_H */ + + + DEBUG_printf(("2cupsDirRead(dp=%p)", dp)); + + /* + * Range check input... + */ + + if (!dp) + return (NULL); + + /* + * Try reading an entry that is not "." or ".."... + */ + + for (;;) + { +# ifdef HAVE_PTHREAD_H + /* + * Read the next entry using the reentrant version of readdir... + */ + + if (readdir_r(dp->dir, (struct dirent *)buffer, &entry)) + { + DEBUG_printf(("3cupsDirRead: readdir_r() failed - %s\n", strerror(errno))); + return (NULL); + } + + if (!entry) + { + DEBUG_puts("3cupsDirRead: readdir_r() returned a NULL pointer!"); + return (NULL); + } + + DEBUG_printf(("4cupsDirRead: readdir_r() returned \"%s\"...", + entry->d_name)); + +# else + /* + * Read the next entry using the original version of readdir... + */ + + if ((entry = readdir(dp->dir)) == NULL) + { + DEBUG_puts("3cupsDirRead: readdir() returned a NULL pointer!"); + return (NULL); + } + + DEBUG_printf(("4cupsDirRead: readdir() returned \"%s\"...", entry->d_name)); + +# endif /* HAVE_PTHREAD_H */ + + /* + * Skip "." and ".."... + */ + + if (!strcmp(entry->d_name, ".") || !strcmp(entry->d_name, "..")) + continue; + + /* + * Copy the name over and get the file information... + */ + + strlcpy(dp->entry.filename, entry->d_name, sizeof(dp->entry.filename)); + + snprintf(filename, sizeof(filename), "%s/%s", dp->directory, entry->d_name); + + if (stat(filename, &(dp->entry.fileinfo))) + { + DEBUG_printf(("3cupsDirRead: stat() failed for \"%s\" - %s...", filename, + strerror(errno))); + continue; + } + + /* + * Return the entry... + */ + + return (&(dp->entry)); + } +} + + +/* + * 'cupsDirRewind()' - Rewind to the start of the directory. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void +cupsDirRewind(cups_dir_t *dp) /* I - Directory pointer */ +{ + DEBUG_printf(("cupsDirRewind(dp=%p)", dp)); + + /* + * Range check input... + */ + + if (!dp) + return; + + /* + * Rewind the directory... + */ + + rewinddir(dp->dir); +} + + +#endif /* WIN32 */ + +/* + * End of "$Id$". + */ diff --git a/cups/dir.h b/cups/dir.h new file mode 100644 index 0000000000..75e5b1e610 --- /dev/null +++ b/cups/dir.h @@ -0,0 +1,69 @@ +/* + * "$Id$" + * + * Public directory definitions for CUPS. + * + * This set of APIs abstracts enumeration of directory entries. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#ifndef _CUPS_DIR_H_ +# define _CUPS_DIR_H_ + + +/* + * Include necessary headers... + */ + +# include "versioning.h" +# include + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Data types... + */ + +typedef struct _cups_dir_s cups_dir_t; /**** Directory type ****/ + +typedef struct cups_dentry_s /**** Directory entry type ****/ +{ + char filename[260]; /* File name */ + struct stat fileinfo; /* File information */ +} cups_dentry_t; + + +/* + * Prototypes... + */ + +extern void cupsDirClose(cups_dir_t *dp) _CUPS_API_1_2; +extern cups_dir_t *cupsDirOpen(const char *directory) _CUPS_API_1_2; +extern cups_dentry_t *cupsDirRead(cups_dir_t *dp) _CUPS_API_1_2; +extern void cupsDirRewind(cups_dir_t *dp) _CUPS_API_1_2; + + +# ifdef __cplusplus +} +# endif /* __cplusplus */ +#endif /* !_CUPS_DIR_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/emit.c b/cups/emit.c new file mode 100644 index 0000000000..3beb025148 --- /dev/null +++ b/cups/emit.c @@ -0,0 +1,1217 @@ +/* + * "$Id$" + * + * PPD code emission routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * PostScript is a trademark of Adobe Systems, Inc. + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * ppdCollect() - Collect all marked options that reside in the + * specified section. + * ppdCollect2() - Collect all marked options that reside in the + * specified section and minimum order. + * ppdEmit() - Emit code for marked options to a file. + * ppdEmitAfterOrder() - Emit a subset of the code for marked options to a + * file. + * ppdEmitFd() - Emit code for marked options to a file. + * ppdEmitJCL() - Emit code for JCL options to a file. + * ppdEmitJCLEnd() - Emit JCLEnd code to a file. + * ppdEmitString() - Get a string containing the code for marked + * options. + * ppd_compare_cparams() - Compare the order of two custom parameters. + * ppd_handle_media() - Handle media selection... + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#if defined(WIN32) || defined(__EMX__) +# include +#else +# include +#endif /* WIN32 || __EMX__ */ + + +/* + * Local functions... + */ + +static int ppd_compare_cparams(ppd_cparam_t *a, ppd_cparam_t *b); +static void ppd_handle_media(ppd_file_t *ppd); + + +/* + * Local globals... + */ + +static const char ppd_custom_code[] = + "pop pop pop\n" + "<>setpagedevice\n"; + + +/* + * 'ppdCollect()' - Collect all marked options that reside in the specified + * section. + * + * The choices array should be freed using @code free@ when you are + * finished with it. + */ + +int /* O - Number of options marked */ +ppdCollect(ppd_file_t *ppd, /* I - PPD file data */ + ppd_section_t section, /* I - Section to collect */ + ppd_choice_t ***choices) /* O - Pointers to choices */ +{ + return (ppdCollect2(ppd, section, 0.0, choices)); +} + + +/* + * 'ppdCollect2()' - Collect all marked options that reside in the + * specified section and minimum order. + * + * The choices array should be freed using @code free@ when you are + * finished with it. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - Number of options marked */ +ppdCollect2(ppd_file_t *ppd, /* I - PPD file data */ + ppd_section_t section, /* I - Section to collect */ + float min_order, /* I - Minimum OrderDependency value */ + ppd_choice_t ***choices) /* O - Pointers to choices */ +{ + ppd_choice_t *c; /* Current choice */ + ppd_section_t csection; /* Current section */ + float corder; /* Current OrderDependency value */ + int count; /* Number of choices collected */ + ppd_choice_t **collect; /* Collected choices */ + float *orders; /* Collected order values */ + + + DEBUG_printf(("ppdCollect2(ppd=%p, section=%d, min_order=%f, choices=%p)", + ppd, section, min_order, choices)); + + if (!ppd || !choices) + { + if (choices) + *choices = NULL; + + return (0); + } + + /* + * Allocate memory for up to N selected choices... + */ + + count = 0; + if ((collect = calloc(sizeof(ppd_choice_t *), + cupsArrayCount(ppd->marked))) == NULL) + { + *choices = NULL; + return (0); + } + + if ((orders = calloc(sizeof(float), cupsArrayCount(ppd->marked))) == NULL) + { + *choices = NULL; + free(collect); + return (0); + } + + /* + * Loop through all options and add choices as needed... + */ + + for (c = (ppd_choice_t *)cupsArrayFirst(ppd->marked); + c; + c = (ppd_choice_t *)cupsArrayNext(ppd->marked)) + { + csection = c->option->section; + corder = c->option->order; + + if (!strcmp(c->choice, "Custom")) + { + ppd_attr_t *attr; /* NonUIOrderDependency value */ + float aorder; /* Order value */ + char asection[17], /* Section name */ + amain[PPD_MAX_NAME + 1], + aoption[PPD_MAX_NAME]; + /* *CustomFoo and True */ + + + for (attr = ppdFindAttr(ppd, "NonUIOrderDependency", NULL); + attr; + attr = ppdFindNextAttr(ppd, "NonUIOrderDependency", NULL)) + if (attr->value && + sscanf(attr->value, "%f%16s%41s%40s", &aorder, asection, amain, + aoption) == 4 && + !strncmp(amain, "*Custom", 7) && + !strcmp(amain + 7, c->option->keyword) && !strcmp(aoption, "True")) + { + /* + * Use this NonUIOrderDependency... + */ + + corder = aorder; + + if (!strcmp(asection, "DocumentSetup")) + csection = PPD_ORDER_DOCUMENT; + else if (!strcmp(asection, "ExitServer")) + csection = PPD_ORDER_EXIT; + else if (!strcmp(asection, "JCLSetup")) + csection = PPD_ORDER_JCL; + else if (!strcmp(asection, "PageSetup")) + csection = PPD_ORDER_PAGE; + else if (!strcmp(asection, "Prolog")) + csection = PPD_ORDER_PROLOG; + else + csection = PPD_ORDER_ANY; + + break; + } + } + + if (csection == section && corder >= min_order) + { + collect[count] = c; + orders[count] = corder; + count ++; + } + } + + /* + * If we have more than 1 marked choice, sort them... + */ + + if (count > 1) + { + int i, j; /* Looping vars */ + + for (i = 0; i < (count - 1); i ++) + for (j = i + 1; j < count; j ++) + if (orders[i] > orders[j]) + { + c = collect[i]; + corder = orders[i]; + collect[i] = collect[j]; + orders[i] = orders[j]; + collect[j] = c; + orders[j] = corder; + } + } + + free(orders); + + DEBUG_printf(("2ppdCollect2: %d marked choices...", count)); + + /* + * Return the array and number of choices; if 0, free the array since + * it isn't needed. + */ + + if (count > 0) + { + *choices = collect; + return (count); + } + else + { + *choices = NULL; + free(collect); + return (0); + } +} + + +/* + * 'ppdEmit()' - Emit code for marked options to a file. + */ + +int /* O - 0 on success, -1 on failure */ +ppdEmit(ppd_file_t *ppd, /* I - PPD file record */ + FILE *fp, /* I - File to write to */ + ppd_section_t section) /* I - Section to write */ +{ + return (ppdEmitAfterOrder(ppd, fp, section, 0, 0.0)); +} + + +/* + * 'ppdEmitAfterOrder()' - Emit a subset of the code for marked options to a file. + * + * When "limit" is non-zero, this function only emits options whose + * OrderDependency value is greater than or equal to "min_order". + * + * When "limit" is zero, this function is identical to ppdEmit(). + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 0 on success, -1 on failure */ +ppdEmitAfterOrder( + ppd_file_t *ppd, /* I - PPD file record */ + FILE *fp, /* I - File to write to */ + ppd_section_t section, /* I - Section to write */ + int limit, /* I - Non-zero to use min_order */ + float min_order) /* I - Lowest OrderDependency */ +{ + char *buffer; /* Option code */ + int status; /* Return status */ + + + /* + * Range check input... + */ + + if (!ppd || !fp) + return (-1); + + /* + * Get the string... + */ + + buffer = ppdEmitString(ppd, section, limit ? min_order : 0.0f); + + /* + * Write it as needed and return... + */ + + if (buffer) + { + status = fputs(buffer, fp) < 0 ? -1 : 0; + + free(buffer); + } + else + status = 0; + + return (status); +} + + +/* + * 'ppdEmitFd()' - Emit code for marked options to a file. + */ + +int /* O - 0 on success, -1 on failure */ +ppdEmitFd(ppd_file_t *ppd, /* I - PPD file record */ + int fd, /* I - File to write to */ + ppd_section_t section) /* I - Section to write */ +{ + char *buffer, /* Option code */ + *bufptr; /* Pointer into code */ + size_t buflength; /* Length of option code */ + ssize_t bytes; /* Bytes written */ + int status; /* Return status */ + + + /* + * Range check input... + */ + + if (!ppd || fd < 0) + return (-1); + + /* + * Get the string... + */ + + buffer = ppdEmitString(ppd, section, 0.0); + + /* + * Write it as needed and return... + */ + + if (buffer) + { + buflength = strlen(buffer); + bufptr = buffer; + bytes = 0; + + while (buflength > 0) + { +#ifdef WIN32 + if ((bytes = (ssize_t)write(fd, bufptr, (unsigned)buflength)) < 0) +#else + if ((bytes = write(fd, bufptr, buflength)) < 0) +#endif /* WIN32 */ + { + if (errno == EAGAIN || errno == EINTR) + continue; + + break; + } + + buflength -= bytes; + bufptr += bytes; + } + + status = bytes < 0 ? -1 : 0; + + free(buffer); + } + else + status = 0; + + return (status); +} + + +/* + * 'ppdEmitJCL()' - Emit code for JCL options to a file. + */ + +int /* O - 0 on success, -1 on failure */ +ppdEmitJCL(ppd_file_t *ppd, /* I - PPD file record */ + FILE *fp, /* I - File to write to */ + int job_id, /* I - Job ID */ + const char *user, /* I - Username */ + const char *title) /* I - Title */ +{ + char *ptr; /* Pointer into JCL string */ + char temp[65], /* Local title string */ + displaymsg[33]; /* Local display string */ + + + /* + * Range check the input... + */ + + if (!ppd || !ppd->jcl_begin || !ppd->jcl_ps) + return (0); + + /* + * See if the printer supports HP PJL... + */ + + if (!strncmp(ppd->jcl_begin, "\033%-12345X@", 10)) + { + /* + * This printer uses HP PJL commands for output; filter the output + * so that we only have a single "@PJL JOB" command in the header... + * + * To avoid bugs in the PJL implementation of certain vendors' products + * (Xerox in particular), we add a dummy "@PJL" command at the beginning + * of the PJL commands to initialize PJL processing. + */ + + ppd_attr_t *charset; /* PJL charset */ + ppd_attr_t *display; /* PJL display command */ + + + if ((charset = ppdFindAttr(ppd, "cupsPJLCharset", NULL)) != NULL) + { + if (!charset->value || _cups_strcasecmp(charset->value, "UTF-8")) + charset = NULL; + } + + if ((display = ppdFindAttr(ppd, "cupsPJLDisplay", NULL)) != NULL) + { + if (!display->value) + display = NULL; + } + + fputs("\033%-12345X@PJL\n", fp); + for (ptr = ppd->jcl_begin + 9; *ptr;) + if (!strncmp(ptr, "@PJL JOB", 8)) + { + /* + * Skip job command... + */ + + for (;*ptr; ptr ++) + if (*ptr == '\n') + break; + + if (*ptr) + ptr ++; + } + else + { + /* + * Copy line... + */ + + for (;*ptr; ptr ++) + { + putc(*ptr, fp); + if (*ptr == '\n') + break; + } + + if (*ptr) + ptr ++; + } + + /* + * Clean up the job title... + */ + + if ((ptr = strrchr(title, '/')) != NULL) + { + /* + * Only show basename of file path... + */ + + title = ptr + 1; + } + + if (!strncmp(title, "smbprn.", 7)) + { + /* + * Skip leading smbprn.######## from Samba jobs... + */ + + for (title += 7; *title && isdigit(*title & 255); title ++); + while (_cups_isspace(*title)) + title ++; + + if ((ptr = strstr(title, " - ")) != NULL) + { + /* + * Skip application name in "Some Application - Title of job"... + */ + + title = ptr + 3; + } + } + + /* + * Replace double quotes with single quotes and UTF-8 characters with + * question marks so that the title does not cause a PJL syntax error. + */ + + strlcpy(temp, title, sizeof(temp)); + + for (ptr = temp; *ptr; ptr ++) + if (*ptr == '\"') + *ptr = '\''; + else if (!charset && (*ptr & 128)) + *ptr = '?'; + + /* + * CUPS STR #3125: Long PJL JOB NAME causes problems with some printers + * + * Generate the display message, truncating at 32 characters + nul to avoid + * issues with some printer's PJL implementations... + */ + + snprintf(displaymsg, sizeof(displaymsg), "%d %s %s", job_id, user, temp); + + /* + * Send PJL JOB and PJL RDYMSG commands before we enter PostScript mode... + */ + + if (display && strcmp(display->value, "job")) + { + fprintf(fp, "@PJL JOB NAME = \"%s\"\n", temp); + + if (display && !strcmp(display->value, "rdymsg")) + fprintf(fp, "@PJL RDYMSG DISPLAY = \"%s\"\n", displaymsg); + } + else + fprintf(fp, "@PJL JOB NAME = \"%s\" DISPLAY = \"%s\"\n", temp, + displaymsg); + } + else + fputs(ppd->jcl_begin, fp); + + ppdEmit(ppd, fp, PPD_ORDER_JCL); + fputs(ppd->jcl_ps, fp); + + return (0); +} + + +/* + * 'ppdEmitJCLEnd()' - Emit JCLEnd code to a file. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 0 on success, -1 on failure */ +ppdEmitJCLEnd(ppd_file_t *ppd, /* I - PPD file record */ + FILE *fp) /* I - File to write to */ +{ + /* + * Range check the input... + */ + + if (!ppd) + return (0); + + if (!ppd->jcl_end) + { + if (ppd->num_filters == 0) + putc(0x04, fp); + + return (0); + } + + /* + * See if the printer supports HP PJL... + */ + + if (!strncmp(ppd->jcl_end, "\033%-12345X@", 10)) + { + /* + * This printer uses HP PJL commands for output; filter the output + * so that we only have a single "@PJL JOB" command in the header... + * + * To avoid bugs in the PJL implementation of certain vendors' products + * (Xerox in particular), we add a dummy "@PJL" command at the beginning + * of the PJL commands to initialize PJL processing. + */ + + fputs("\033%-12345X@PJL\n", fp); + fputs("@PJL RDYMSG DISPLAY = \"\"\n", fp); + fputs(ppd->jcl_end + 9, fp); + } + else + fputs(ppd->jcl_end, fp); + + return (0); +} + + +/* + * 'ppdEmitString()' - Get a string containing the code for marked options. + * + * When "min_order" is greater than zero, this function only includes options + * whose OrderDependency value is greater than or equal to "min_order". + * Otherwise, all options in the specified section are included in the + * returned string. + * + * The return string is allocated on the heap and should be freed using + * @code free@ when you are done with it. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +char * /* O - String containing option code or @code NULL@ if there is no option code */ +ppdEmitString(ppd_file_t *ppd, /* I - PPD file record */ + ppd_section_t section, /* I - Section to write */ + float min_order) /* I - Lowest OrderDependency */ +{ + int i, j, /* Looping vars */ + count; /* Number of choices */ + ppd_choice_t **choices; /* Choices */ + ppd_size_t *size; /* Custom page size */ + ppd_coption_t *coption; /* Custom option */ + ppd_cparam_t *cparam; /* Custom parameter */ + size_t bufsize; /* Size of string buffer needed */ + char *buffer, /* String buffer */ + *bufptr, /* Pointer into buffer */ + *bufend; /* End of buffer */ + struct lconv *loc; /* Locale data */ + + + DEBUG_printf(("ppdEmitString(ppd=%p, section=%d, min_order=%f)", + ppd, section, min_order)); + + /* + * Range check input... + */ + + if (!ppd) + return (NULL); + + /* + * Use PageSize or PageRegion as required... + */ + + ppd_handle_media(ppd); + + /* + * Collect the options we need to emit... + */ + + if ((count = ppdCollect2(ppd, section, min_order, &choices)) == 0) + return (NULL); + + /* + * Count the number of bytes that are required to hold all of the + * option code... + */ + + for (i = 0, bufsize = 1; i < count; i ++) + { + if (section == PPD_ORDER_JCL) + { + if (!_cups_strcasecmp(choices[i]->choice, "Custom") && + (coption = ppdFindCustomOption(ppd, choices[i]->option->keyword)) + != NULL) + { + /* + * Add space to account for custom parameter substitution... + */ + + for (cparam = (ppd_cparam_t *)cupsArrayFirst(coption->params); + cparam; + cparam = (ppd_cparam_t *)cupsArrayNext(coption->params)) + { + switch (cparam->type) + { + case PPD_CUSTOM_CURVE : + case PPD_CUSTOM_INVCURVE : + case PPD_CUSTOM_POINTS : + case PPD_CUSTOM_REAL : + case PPD_CUSTOM_INT : + bufsize += 10; + break; + + case PPD_CUSTOM_PASSCODE : + case PPD_CUSTOM_PASSWORD : + case PPD_CUSTOM_STRING : + if (cparam->current.custom_string) + bufsize += strlen(cparam->current.custom_string); + break; + } + } + } + } + else if (section != PPD_ORDER_EXIT) + { + bufsize += 3; /* [{\n */ + + if ((!_cups_strcasecmp(choices[i]->option->keyword, "PageSize") || + !_cups_strcasecmp(choices[i]->option->keyword, "PageRegion")) && + !_cups_strcasecmp(choices[i]->choice, "Custom")) + { + DEBUG_puts("2ppdEmitString: Custom size set!"); + + bufsize += 37; /* %%BeginFeature: *CustomPageSize True\n */ + bufsize += 50; /* Five 9-digit numbers + newline */ + } + else if (!_cups_strcasecmp(choices[i]->choice, "Custom") && + (coption = ppdFindCustomOption(ppd, + choices[i]->option->keyword)) + != NULL) + { + bufsize += 23 + strlen(choices[i]->option->keyword) + 6; + /* %%BeginFeature: *Customkeyword True\n */ + + + for (cparam = (ppd_cparam_t *)cupsArrayFirst(coption->params); + cparam; + cparam = (ppd_cparam_t *)cupsArrayNext(coption->params)) + { + switch (cparam->type) + { + case PPD_CUSTOM_CURVE : + case PPD_CUSTOM_INVCURVE : + case PPD_CUSTOM_POINTS : + case PPD_CUSTOM_REAL : + case PPD_CUSTOM_INT : + bufsize += 10; + break; + + case PPD_CUSTOM_PASSCODE : + case PPD_CUSTOM_PASSWORD : + case PPD_CUSTOM_STRING : + bufsize += 3; + if (cparam->current.custom_string) + bufsize += 4 * strlen(cparam->current.custom_string); + break; + } + } + } + else + bufsize += 17 + strlen(choices[i]->option->keyword) + 1 + + strlen(choices[i]->choice) + 1; + /* %%BeginFeature: *keyword choice\n */ + + bufsize += 13; /* %%EndFeature\n */ + bufsize += 22; /* } stopped cleartomark\n */ + } + + if (choices[i]->code) + bufsize += strlen(choices[i]->code) + 1; + else + bufsize += strlen(ppd_custom_code); + } + + /* + * Allocate memory... + */ + + DEBUG_printf(("2ppdEmitString: Allocating %d bytes for string...", + (int)bufsize)); + + if ((buffer = calloc(1, bufsize)) == NULL) + { + free(choices); + return (NULL); + } + + bufend = buffer + bufsize - 1; + loc = localeconv(); + + /* + * Copy the option code to the buffer... + */ + + for (i = 0, bufptr = buffer; i < count; i ++, bufptr += strlen(bufptr)) + if (section == PPD_ORDER_JCL) + { + if (!_cups_strcasecmp(choices[i]->choice, "Custom") && + choices[i]->code && + (coption = ppdFindCustomOption(ppd, choices[i]->option->keyword)) + != NULL) + { + /* + * Handle substitutions in custom JCL options... + */ + + char *cptr; /* Pointer into code */ + int pnum; /* Parameter number */ + + + for (cptr = choices[i]->code; *cptr && bufptr < bufend;) + { + if (*cptr == '\\') + { + cptr ++; + + if (isdigit(*cptr & 255)) + { + /* + * Substitute parameter... + */ + + pnum = *cptr++ - '0'; + while (isdigit(*cptr & 255)) + pnum = pnum * 10 + *cptr++ - '0'; + + for (cparam = (ppd_cparam_t *)cupsArrayFirst(coption->params); + cparam; + cparam = (ppd_cparam_t *)cupsArrayNext(coption->params)) + if (cparam->order == pnum) + break; + + if (cparam) + { + switch (cparam->type) + { + case PPD_CUSTOM_CURVE : + case PPD_CUSTOM_INVCURVE : + case PPD_CUSTOM_POINTS : + case PPD_CUSTOM_REAL : + bufptr = _cupsStrFormatd(bufptr, bufend, + cparam->current.custom_real, + loc); + break; + + case PPD_CUSTOM_INT : + snprintf(bufptr, bufend - bufptr, "%d", + cparam->current.custom_int); + bufptr += strlen(bufptr); + break; + + case PPD_CUSTOM_PASSCODE : + case PPD_CUSTOM_PASSWORD : + case PPD_CUSTOM_STRING : + if (cparam->current.custom_string) + { + strlcpy(bufptr, cparam->current.custom_string, + bufend - bufptr); + bufptr += strlen(bufptr); + } + break; + } + } + } + else if (*cptr) + *bufptr++ = *cptr++; + } + else + *bufptr++ = *cptr++; + } + } + else + { + /* + * Otherwise just copy the option code directly... + */ + + strlcpy(bufptr, choices[i]->code, bufend - bufptr + 1); + bufptr += strlen(bufptr); + } + } + else if (section != PPD_ORDER_EXIT) + { + /* + * Add wrapper commands to prevent printer errors for unsupported + * options... + */ + + strlcpy(bufptr, "[{\n", bufend - bufptr + 1); + bufptr += 3; + + /* + * Send DSC comments with option... + */ + + DEBUG_printf(("2ppdEmitString: Adding code for %s=%s...", + choices[i]->option->keyword, choices[i]->choice)); + + if ((!_cups_strcasecmp(choices[i]->option->keyword, "PageSize") || + !_cups_strcasecmp(choices[i]->option->keyword, "PageRegion")) && + !_cups_strcasecmp(choices[i]->choice, "Custom")) + { + /* + * Variable size; write out standard size options, using the + * parameter positions defined in the PPD file... + */ + + ppd_attr_t *attr; /* PPD attribute */ + int pos, /* Position of custom value */ + orientation; /* Orientation to use */ + float values[5]; /* Values for custom command */ + + + strlcpy(bufptr, "%%BeginFeature: *CustomPageSize True\n", + bufend - bufptr + 1); + bufptr += 37; + + size = ppdPageSize(ppd, "Custom"); + + memset(values, 0, sizeof(values)); + + if ((attr = ppdFindAttr(ppd, "ParamCustomPageSize", "Width")) != NULL) + { + pos = atoi(attr->value) - 1; + + if (pos < 0 || pos > 4) + pos = 0; + } + else + pos = 0; + + values[pos] = size->width; + + if ((attr = ppdFindAttr(ppd, "ParamCustomPageSize", "Height")) != NULL) + { + pos = atoi(attr->value) - 1; + + if (pos < 0 || pos > 4) + pos = 1; + } + else + pos = 1; + + values[pos] = size->length; + + /* + * According to the Adobe PPD specification, an orientation of 1 + * will produce a print that comes out upside-down with the X + * axis perpendicular to the direction of feed, which is exactly + * what we want to be consistent with non-PS printers. + * + * We could also use an orientation of 3 to produce output that + * comes out rightside-up (this is the default for many large format + * printer PPDs), however for consistency we will stick with the + * value 1. + * + * If we wanted to get fancy, we could use orientations of 0 or + * 2 and swap the width and length, however we don't want to get + * fancy, we just want it to work consistently. + * + * The orientation value is range limited by the Orientation + * parameter definition, so certain non-PS printer drivers that + * only support an Orientation of 0 will get the value 0 as + * expected. + */ + + orientation = 1; + + if ((attr = ppdFindAttr(ppd, "ParamCustomPageSize", + "Orientation")) != NULL) + { + int min_orient, max_orient; /* Minimum and maximum orientations */ + + + if (sscanf(attr->value, "%d%*s%d%d", &pos, &min_orient, + &max_orient) != 3) + pos = 4; + else + { + pos --; + + if (pos < 0 || pos > 4) + pos = 4; + + if (orientation > max_orient) + orientation = max_orient; + else if (orientation < min_orient) + orientation = min_orient; + } + } + else + pos = 4; + + values[pos] = (float)orientation; + + for (pos = 0; pos < 5; pos ++) + { + bufptr = _cupsStrFormatd(bufptr, bufend, values[pos], loc); + *bufptr++ = '\n'; + } + + if (!choices[i]->code) + { + /* + * This can happen with certain buggy PPD files that don't include + * a CustomPageSize command sequence... We just use a generic + * Level 2 command sequence... + */ + + strlcpy(bufptr, ppd_custom_code, bufend - bufptr + 1); + bufptr += strlen(bufptr); + } + } + else if (!_cups_strcasecmp(choices[i]->choice, "Custom") && + (coption = ppdFindCustomOption(ppd, choices[i]->option->keyword)) + != NULL) + { + /* + * Custom option... + */ + + const char *s; /* Pointer into string value */ + cups_array_t *params; /* Parameters in the correct output order */ + + + params = cupsArrayNew((cups_array_func_t)ppd_compare_cparams, NULL); + + for (cparam = (ppd_cparam_t *)cupsArrayFirst(coption->params); + cparam; + cparam = (ppd_cparam_t *)cupsArrayNext(coption->params)) + cupsArrayAdd(params, cparam); + + snprintf(bufptr, bufend - bufptr + 1, + "%%%%BeginFeature: *Custom%s True\n", coption->keyword); + bufptr += strlen(bufptr); + + for (cparam = (ppd_cparam_t *)cupsArrayFirst(params); + cparam; + cparam = (ppd_cparam_t *)cupsArrayNext(params)) + { + switch (cparam->type) + { + case PPD_CUSTOM_CURVE : + case PPD_CUSTOM_INVCURVE : + case PPD_CUSTOM_POINTS : + case PPD_CUSTOM_REAL : + bufptr = _cupsStrFormatd(bufptr, bufend, + cparam->current.custom_real, loc); + *bufptr++ = '\n'; + break; + + case PPD_CUSTOM_INT : + snprintf(bufptr, bufend - bufptr + 1, "%d\n", + cparam->current.custom_int); + bufptr += strlen(bufptr); + break; + + case PPD_CUSTOM_PASSCODE : + case PPD_CUSTOM_PASSWORD : + case PPD_CUSTOM_STRING : + *bufptr++ = '('; + + if (cparam->current.custom_string) + { + for (s = cparam->current.custom_string; *s; s ++) + { + if (*s < ' ' || *s == '(' || *s == ')' || *s >= 127) + { + snprintf(bufptr, bufend - bufptr + 1, "\\%03o", *s & 255); + bufptr += strlen(bufptr); + } + else + *bufptr++ = *s; + } + } + + *bufptr++ = ')'; + *bufptr++ = '\n'; + break; + } + } + + cupsArrayDelete(params); + } + else + { + snprintf(bufptr, bufend - bufptr + 1, "%%%%BeginFeature: *%s %s\n", + choices[i]->option->keyword, choices[i]->choice); + bufptr += strlen(bufptr); + } + + if (choices[i]->code && choices[i]->code[0]) + { + j = (int)strlen(choices[i]->code); + memcpy(bufptr, choices[i]->code, j); + bufptr += j; + + if (choices[i]->code[j - 1] != '\n') + *bufptr++ = '\n'; + } + + strlcpy(bufptr, "%%EndFeature\n" + "} stopped cleartomark\n", bufend - bufptr + 1); + bufptr += strlen(bufptr); + + DEBUG_printf(("2ppdEmitString: Offset in string is %d...", + (int)(bufptr - buffer))); + } + else + { + strlcpy(bufptr, choices[i]->code, bufend - bufptr + 1); + bufptr += strlen(bufptr); + } + + /* + * Nul-terminate, free, and return... + */ + + *bufptr = '\0'; + + free(choices); + + return (buffer); +} + + +/* + * 'ppd_compare_cparams()' - Compare the order of two custom parameters. + */ + +static int /* O - Result of comparison */ +ppd_compare_cparams(ppd_cparam_t *a, /* I - First parameter */ + ppd_cparam_t *b) /* I - Second parameter */ +{ + return (a->order - b->order); +} + + +/* + * 'ppd_handle_media()' - Handle media selection... + */ + +static void +ppd_handle_media(ppd_file_t *ppd) /* I - PPD file */ +{ + ppd_choice_t *manual_feed, /* ManualFeed choice, if any */ + *input_slot; /* InputSlot choice, if any */ + ppd_size_t *size; /* Current media size */ + ppd_attr_t *rpr; /* RequiresPageRegion value */ + + + /* + * This function determines what page size code to use, if any, for the + * current media size, InputSlot, and ManualFeed selections. + * + * We use the PageSize code if: + * + * 1. A custom media size is selected. + * 2. ManualFeed and InputSlot are not selected (or do not exist). + * 3. ManualFeed is selected but is False and InputSlot is not selected or + * the selection has no code - the latter check done to support "auto" or + * "printer default" InputSlot options. + * + * We use the PageRegion code if: + * + * 4. RequiresPageRegion does not exist and the PPD contains cupsFilter + * keywords, indicating this is a CUPS-based driver. + * 5. RequiresPageRegion exists for the selected InputSlot (or "All" for any + * InputSlot or ManualFeed selection) and is True. + * + * If none of the 5 conditions are true, no page size code is used and we + * unmark any existing PageSize or PageRegion choices. + */ + + if ((size = ppdPageSize(ppd, NULL)) == NULL) + return; + + manual_feed = ppdFindMarkedChoice(ppd, "ManualFeed"); + input_slot = ppdFindMarkedChoice(ppd, "InputSlot"); + + if (input_slot != NULL) + rpr = ppdFindAttr(ppd, "RequiresPageRegion", input_slot->choice); + else + rpr = NULL; + + if (!rpr) + rpr = ppdFindAttr(ppd, "RequiresPageRegion", "All"); + + if (!_cups_strcasecmp(size->name, "Custom") || + (!manual_feed && !input_slot) || + (manual_feed && !_cups_strcasecmp(manual_feed->choice, "False") && + (!input_slot || (input_slot->code && !input_slot->code[0]))) || + (!rpr && ppd->num_filters > 0)) + { + /* + * Use PageSize code... + */ + + ppdMarkOption(ppd, "PageSize", size->name); + } + else if (rpr && rpr->value && !_cups_strcasecmp(rpr->value, "True")) + { + /* + * Use PageRegion code... + */ + + ppdMarkOption(ppd, "PageRegion", size->name); + } + else + { + /* + * Do not use PageSize or PageRegion code... + */ + + ppd_choice_t *page; /* PageSize/Region choice, if any */ + + if ((page = ppdFindMarkedChoice(ppd, "PageSize")) != NULL) + { + /* + * Unmark PageSize... + */ + + page->marked = 0; + cupsArrayRemove(ppd->marked, page); + } + + if ((page = ppdFindMarkedChoice(ppd, "PageRegion")) != NULL) + { + /* + * Unmark PageRegion... + */ + + page->marked = 0; + cupsArrayRemove(ppd->marked, page); + } + } +} + + +/* + * End of "$Id$". + */ diff --git a/cups/encode.c b/cups/encode.c new file mode 100644 index 0000000000..d7beac1d48 --- /dev/null +++ b/cups/encode.c @@ -0,0 +1,651 @@ +/* + * "$Id$" + * + * Option encoding routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsEncodeOptions() - Encode printer options into IPP attributes. + * cupsEncodeOptions2() - Encode printer options into IPP attributes for + * a group. + * _ippFindOption() - Find the attribute information for an option. + * compare_ipp_options() - Compare two IPP options. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" + + +/* + * Local list of option names and the value tags they should use... + * + * **** THIS LIST MUST BE SORTED **** + */ + +static const _ipp_option_t ipp_options[] = +{ + { 1, "auth-info", IPP_TAG_TEXT, IPP_TAG_JOB }, + { 1, "auth-info-required", IPP_TAG_KEYWORD, IPP_TAG_PRINTER }, + { 0, "blackplot", IPP_TAG_BOOLEAN, IPP_TAG_JOB }, + { 0, "blackplot-default", IPP_TAG_BOOLEAN, IPP_TAG_PRINTER }, + { 0, "brightness", IPP_TAG_INTEGER, IPP_TAG_JOB }, + { 0, "brightness-default", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 0, "columns", IPP_TAG_INTEGER, IPP_TAG_JOB }, + { 0, "columns-default", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 0, "compression", IPP_TAG_KEYWORD, IPP_TAG_OPERATION }, + { 0, "copies", IPP_TAG_INTEGER, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 0, "copies-default", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 0, "device-uri", IPP_TAG_URI, IPP_TAG_PRINTER }, + { 1, "document-copies", IPP_TAG_RANGE, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 0, "document-format", IPP_TAG_MIMETYPE, IPP_TAG_OPERATION }, + { 0, "document-format-default", IPP_TAG_MIMETYPE, IPP_TAG_PRINTER }, + { 1, "document-numbers", IPP_TAG_RANGE, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 1, "exclude-schemes", IPP_TAG_NAME, IPP_TAG_OPERATION }, + { 1, "finishings", IPP_TAG_ENUM, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 1, "finishings-default", IPP_TAG_ENUM, IPP_TAG_PRINTER }, + { 0, "fit-to-page", IPP_TAG_BOOLEAN, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 0, "fit-to-page-default", IPP_TAG_BOOLEAN, IPP_TAG_PRINTER }, + { 0, "fitplot", IPP_TAG_BOOLEAN, IPP_TAG_JOB }, + { 0, "fitplot-default", IPP_TAG_BOOLEAN, IPP_TAG_PRINTER }, + { 0, "gamma", IPP_TAG_INTEGER, IPP_TAG_JOB }, + { 0, "gamma-default", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 0, "hue", IPP_TAG_INTEGER, IPP_TAG_JOB }, + { 0, "hue-default", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 1, "include-schemes", IPP_TAG_NAME, IPP_TAG_OPERATION }, + { 0, "job-impressions", IPP_TAG_INTEGER, IPP_TAG_JOB }, + { 0, "job-k-limit", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 0, "job-page-limit", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 0, "job-priority", IPP_TAG_INTEGER, IPP_TAG_JOB }, + { 0, "job-quota-period", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 1, "job-sheets", IPP_TAG_NAME, IPP_TAG_JOB }, + { 1, "job-sheets-default", IPP_TAG_NAME, IPP_TAG_PRINTER }, + { 0, "job-uuid", IPP_TAG_URI, IPP_TAG_JOB }, + { 0, "landscape", IPP_TAG_BOOLEAN, IPP_TAG_JOB }, + { 1, "marker-change-time", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 1, "marker-colors", IPP_TAG_NAME, IPP_TAG_PRINTER }, + { 1, "marker-high-levels", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 1, "marker-levels", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 1, "marker-low-levels", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 0, "marker-message", IPP_TAG_TEXT, IPP_TAG_PRINTER }, + { 1, "marker-names", IPP_TAG_NAME, IPP_TAG_PRINTER }, + { 1, "marker-types", IPP_TAG_KEYWORD, IPP_TAG_PRINTER }, + { 1, "media", IPP_TAG_KEYWORD, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 0, "media-col", IPP_TAG_BEGIN_COLLECTION, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 0, "media-col-default", IPP_TAG_BEGIN_COLLECTION, IPP_TAG_PRINTER }, + { 0, "media-color", IPP_TAG_KEYWORD, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 1, "media-default", IPP_TAG_KEYWORD, IPP_TAG_PRINTER }, + { 0, "media-key", IPP_TAG_KEYWORD, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 0, "media-size", IPP_TAG_BEGIN_COLLECTION, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 0, "media-type", IPP_TAG_KEYWORD, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 0, "mirror", IPP_TAG_BOOLEAN, IPP_TAG_JOB }, + { 0, "mirror-default", IPP_TAG_BOOLEAN, IPP_TAG_PRINTER }, + { 0, "natural-scaling", IPP_TAG_INTEGER, IPP_TAG_JOB }, + { 0, "natural-scaling-default", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 0, "notify-charset", IPP_TAG_CHARSET, IPP_TAG_SUBSCRIPTION }, + { 1, "notify-events", IPP_TAG_KEYWORD, IPP_TAG_SUBSCRIPTION }, + { 1, "notify-events-default", IPP_TAG_KEYWORD, IPP_TAG_PRINTER }, + { 0, "notify-lease-duration", IPP_TAG_INTEGER, IPP_TAG_SUBSCRIPTION }, + { 0, "notify-lease-duration-default", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 0, "notify-natural-language", IPP_TAG_LANGUAGE, IPP_TAG_SUBSCRIPTION }, + { 0, "notify-pull-method", IPP_TAG_KEYWORD, IPP_TAG_SUBSCRIPTION }, + { 0, "notify-recipient-uri", IPP_TAG_URI, IPP_TAG_SUBSCRIPTION }, + { 0, "notify-time-interval", IPP_TAG_INTEGER, IPP_TAG_SUBSCRIPTION }, + { 0, "notify-user-data", IPP_TAG_STRING, IPP_TAG_SUBSCRIPTION }, + { 0, "number-up", IPP_TAG_INTEGER, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 0, "number-up-default", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 0, "orientation-requested", IPP_TAG_ENUM, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 0, "orientation-requested-default", IPP_TAG_ENUM, IPP_TAG_PRINTER }, + { 1, "overrides", IPP_TAG_BEGIN_COLLECTION, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 0, "page-bottom", IPP_TAG_INTEGER, IPP_TAG_JOB }, + { 0, "page-bottom-default", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 0, "page-left", IPP_TAG_INTEGER, IPP_TAG_JOB }, + { 0, "page-left-default", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 1, "page-ranges", IPP_TAG_RANGE, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 1, "page-ranges-default", IPP_TAG_RANGE, IPP_TAG_PRINTER }, + { 0, "page-right", IPP_TAG_INTEGER, IPP_TAG_JOB }, + { 0, "page-right-default", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 0, "page-top", IPP_TAG_INTEGER, IPP_TAG_JOB }, + { 0, "page-top-default", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 1, "pages", IPP_TAG_RANGE, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 0, "penwidth", IPP_TAG_INTEGER, IPP_TAG_JOB }, + { 0, "penwidth-default", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 0, "port-monitor", IPP_TAG_NAME, IPP_TAG_PRINTER }, + { 0, "ppd-name", IPP_TAG_NAME, IPP_TAG_PRINTER }, + { 0, "ppi", IPP_TAG_INTEGER, IPP_TAG_JOB }, + { 0, "ppi-default", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 0, "prettyprint", IPP_TAG_BOOLEAN, IPP_TAG_JOB }, + { 0, "prettyprint-default", IPP_TAG_BOOLEAN, IPP_TAG_PRINTER }, + { 0, "print-quality", IPP_TAG_ENUM, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 0, "print-quality-default", IPP_TAG_ENUM, IPP_TAG_PRINTER }, + { 1, "printer-commands", IPP_TAG_KEYWORD, IPP_TAG_PRINTER }, + { 0, "printer-error-policy", IPP_TAG_NAME, IPP_TAG_PRINTER }, + { 0, "printer-info", IPP_TAG_TEXT, IPP_TAG_PRINTER }, + { 0, "printer-is-accepting-jobs", IPP_TAG_BOOLEAN, IPP_TAG_PRINTER }, + { 0, "printer-is-shared", IPP_TAG_BOOLEAN, IPP_TAG_PRINTER }, + { 0, "printer-location", IPP_TAG_TEXT, IPP_TAG_PRINTER }, + { 0, "printer-make-and-model", IPP_TAG_TEXT, IPP_TAG_PRINTER }, + { 0, "printer-more-info", IPP_TAG_URI, IPP_TAG_PRINTER }, + { 0, "printer-op-policy", IPP_TAG_NAME, IPP_TAG_PRINTER }, + { 0, "printer-resolution", IPP_TAG_RESOLUTION, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 0, "printer-state", IPP_TAG_ENUM, IPP_TAG_PRINTER }, + { 0, "printer-state-change-time", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 1, "printer-state-reasons", IPP_TAG_KEYWORD, IPP_TAG_PRINTER }, + { 0, "printer-type", IPP_TAG_ENUM, IPP_TAG_PRINTER }, + { 0, "printer-uri", IPP_TAG_URI, IPP_TAG_OPERATION }, + { 1, "printer-uri-supported", IPP_TAG_URI, IPP_TAG_PRINTER }, + { 0, "queued-job-count", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 0, "raw", IPP_TAG_MIMETYPE, IPP_TAG_OPERATION }, + { 1, "requested-attributes", IPP_TAG_NAME, IPP_TAG_OPERATION }, + { 1, "requesting-user-name-allowed", IPP_TAG_NAME, IPP_TAG_PRINTER }, + { 1, "requesting-user-name-denied", IPP_TAG_NAME, IPP_TAG_PRINTER }, + { 0, "resolution", IPP_TAG_RESOLUTION, IPP_TAG_JOB }, + { 0, "resolution-default", IPP_TAG_RESOLUTION, IPP_TAG_PRINTER }, + { 0, "saturation", IPP_TAG_INTEGER, IPP_TAG_JOB }, + { 0, "saturation-default", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 0, "scaling", IPP_TAG_INTEGER, IPP_TAG_JOB }, + { 0, "scaling-default", IPP_TAG_INTEGER, IPP_TAG_PRINTER }, + { 0, "sides", IPP_TAG_KEYWORD, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 0, "sides-default", IPP_TAG_KEYWORD, IPP_TAG_PRINTER }, + { 0, "wrap", IPP_TAG_BOOLEAN, IPP_TAG_JOB }, + { 0, "wrap-default", IPP_TAG_BOOLEAN, IPP_TAG_PRINTER }, + { 0, "x-dimension", IPP_TAG_INTEGER, IPP_TAG_JOB, + IPP_TAG_DOCUMENT }, + { 0, "y-dimension", IPP_TAG_INTEGER, IPP_TAG_JOB, + IPP_TAG_DOCUMENT } +}; + + +/* + * Local functions... + */ + +static int compare_ipp_options(_ipp_option_t *a, _ipp_option_t *b); + + +/* + * 'cupsEncodeOptions()' - Encode printer options into IPP attributes. + * + * This function adds operation, job, and then subscription attributes, + * in that order. Use the cupsEncodeOptions2() function to add attributes + * for a single group. + */ + +void +cupsEncodeOptions(ipp_t *ipp, /* I - Request to add to */ + int num_options, /* I - Number of options */ + cups_option_t *options) /* I - Options */ +{ + DEBUG_printf(("cupsEncodeOptions(%p, %d, %p)", ipp, num_options, options)); + + /* + * Add the options in the proper groups & order... + */ + + cupsEncodeOptions2(ipp, num_options, options, IPP_TAG_OPERATION); + cupsEncodeOptions2(ipp, num_options, options, IPP_TAG_JOB); + cupsEncodeOptions2(ipp, num_options, options, IPP_TAG_SUBSCRIPTION); +} + + +/* + * 'cupsEncodeOptions2()' - Encode printer options into IPP attributes for a group. + * + * This function only adds attributes for a single group. Call this + * function multiple times for each group, or use cupsEncodeOptions() + * to add the standard groups. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void +cupsEncodeOptions2( + ipp_t *ipp, /* I - Request to add to */ + int num_options, /* I - Number of options */ + cups_option_t *options, /* I - Options */ + ipp_tag_t group_tag) /* I - Group to encode */ +{ + int i, j; /* Looping vars */ + int count; /* Number of values */ + char *s, /* Pointer into option value */ + *val, /* Pointer to option value */ + *copy, /* Copy of option value */ + *sep, /* Option separator */ + quote; /* Quote character */ + ipp_attribute_t *attr; /* IPP attribute */ + ipp_tag_t value_tag; /* IPP value tag */ + cups_option_t *option; /* Current option */ + ipp_t *collection; /* Collection value */ + int num_cols; /* Number of collection values */ + cups_option_t *cols; /* Collection values */ + + + DEBUG_printf(("cupsEncodeOptions2(ipp=%p, num_options=%d, options=%p, " + "group_tag=%x)", ipp, num_options, options, group_tag)); + + /* + * Range check input... + */ + + if (!ipp || num_options < 1 || !options) + return; + + /* + * Do special handling for the document-format/raw options... + */ + + if (group_tag == IPP_TAG_OPERATION) + { + /* + * Handle the document format stuff first... + */ + + if ((val = (char *)cupsGetOption("document-format", num_options, options)) != NULL) + ippAddString(ipp, IPP_TAG_OPERATION, IPP_TAG_MIMETYPE, "document-format", + NULL, val); + else if (cupsGetOption("raw", num_options, options)) + ippAddString(ipp, IPP_TAG_OPERATION, IPP_TAG_MIMETYPE, "document-format", + NULL, "application/vnd.cups-raw"); + else + ippAddString(ipp, IPP_TAG_OPERATION, IPP_TAG_MIMETYPE, "document-format", + NULL, "application/octet-stream"); + } + + /* + * Then loop through the options... + */ + + for (i = num_options, option = options; i > 0; i --, option ++) + { + _ipp_option_t *match; /* Matching attribute */ + + + /* + * Skip document format options that are handled above... + */ + + if (!_cups_strcasecmp(option->name, "raw") || + !_cups_strcasecmp(option->name, "document-format") || + !option->name[0]) + continue; + + /* + * Figure out the proper value and group tags for this option... + */ + + if ((match = _ippFindOption(option->name)) != NULL) + { + if (match->group_tag != group_tag && match->alt_group_tag != group_tag) + continue; + + value_tag = match->value_tag; + } + else + { + int namelen; /* Length of name */ + + + namelen = (int)strlen(option->name); + + if (namelen < 10 || + (strcmp(option->name + namelen - 8, "-default") && + strcmp(option->name + namelen - 10, "-supported"))) + { + if (group_tag != IPP_TAG_JOB && group_tag != IPP_TAG_DOCUMENT) + continue; + } + else if (group_tag != IPP_TAG_PRINTER) + continue; + + if (!_cups_strcasecmp(option->value, "true") || + !_cups_strcasecmp(option->value, "false")) + value_tag = IPP_TAG_BOOLEAN; + else + value_tag = IPP_TAG_NAME; + } + + /* + * Count the number of values... + */ + + if (match && match->multivalue) + { + for (count = 1, sep = option->value, quote = 0; *sep; sep ++) + { + if (*sep == quote) + quote = 0; + else if (!quote && (*sep == '\'' || *sep == '\"')) + { + /* + * Skip quoted option value... + */ + + quote = *sep++; + } + else if (*sep == ',' && !quote) + count ++; + else if (*sep == '\\' && sep[1]) + sep ++; + } + } + else + count = 1; + + DEBUG_printf(("2cupsEncodeOptions2: option=\"%s\", count=%d", + option->name, count)); + + /* + * Allocate memory for the attribute values... + */ + + if ((attr = ippAddStrings(ipp, group_tag, value_tag, option->name, count, + NULL, NULL)) == NULL) + { + /* + * Ran out of memory! + */ + + DEBUG_puts("1cupsEncodeOptions2: Ran out of memory for attributes!"); + return; + } + + if (count > 1) + { + /* + * Make a copy of the value we can fiddle with... + */ + + if ((copy = strdup(option->value)) == NULL) + { + /* + * Ran out of memory! + */ + + DEBUG_puts("1cupsEncodeOptions2: Ran out of memory for value copy!"); + ippDeleteAttribute(ipp, attr); + return; + } + + val = copy; + } + else + { + /* + * Since we have a single value, use the value directly... + */ + + val = option->value; + copy = NULL; + } + + /* + * Scan the value string for values... + */ + + for (j = 0, sep = val; j < count; val = sep, j ++) + { + /* + * Find the end of this value and mark it if needed... + */ + + if (count > 1) + { + for (quote = 0; *sep; sep ++) + { + if (*sep == quote) + { + /* + * Finish quoted value... + */ + + quote = 0; + } + else if (!quote && (*sep == '\'' || *sep == '\"')) + { + /* + * Handle quoted option value... + */ + + quote = *sep; + } + else if (*sep == ',' && count > 1) + break; + else if (*sep == '\\' && sep[1]) + { + /* + * Skip quoted character... + */ + + sep ++; + } + } + + if (*sep == ',') + *sep++ = '\0'; + } + + /* + * Copy the option value(s) over as needed by the type... + */ + + switch (attr->value_tag) + { + case IPP_TAG_INTEGER : + case IPP_TAG_ENUM : + /* + * Integer/enumeration value... + */ + + attr->values[j].integer = strtol(val, &s, 10); + + DEBUG_printf(("2cupsEncodeOptions2: Added integer option value " + "%d...", attr->values[j].integer)); + break; + + case IPP_TAG_BOOLEAN : + if (!_cups_strcasecmp(val, "true") || + !_cups_strcasecmp(val, "on") || + !_cups_strcasecmp(val, "yes")) + { + /* + * Boolean value - true... + */ + + attr->values[j].boolean = 1; + + DEBUG_puts("2cupsEncodeOptions2: Added boolean true value..."); + } + else + { + /* + * Boolean value - false... + */ + + attr->values[j].boolean = 0; + + DEBUG_puts("2cupsEncodeOptions2: Added boolean false value..."); + } + break; + + case IPP_TAG_RANGE : + /* + * Range... + */ + + if (*val == '-') + { + attr->values[j].range.lower = 1; + s = val; + } + else + attr->values[j].range.lower = strtol(val, &s, 10); + + if (*s == '-') + { + if (s[1]) + attr->values[j].range.upper = strtol(s + 1, NULL, 10); + else + attr->values[j].range.upper = 2147483647; + } + else + attr->values[j].range.upper = attr->values[j].range.lower; + + DEBUG_printf(("2cupsEncodeOptions2: Added range option value " + "%d-%d...", attr->values[j].range.lower, + attr->values[j].range.upper)); + break; + + case IPP_TAG_RESOLUTION : + /* + * Resolution... + */ + + attr->values[j].resolution.xres = strtol(val, &s, 10); + + if (*s == 'x') + attr->values[j].resolution.yres = strtol(s + 1, &s, 10); + else + attr->values[j].resolution.yres = attr->values[j].resolution.xres; + + if (!_cups_strcasecmp(s, "dpc")) + attr->values[j].resolution.units = IPP_RES_PER_CM; + else + attr->values[j].resolution.units = IPP_RES_PER_INCH; + + DEBUG_printf(("2cupsEncodeOptions2: Added resolution option value " + "%s...", val)); + break; + + case IPP_TAG_STRING : + /* + * octet-string + */ + + attr->values[j].unknown.length = (int)strlen(val); + attr->values[j].unknown.data = strdup(val); + + DEBUG_printf(("2cupsEncodeOptions2: Added octet-string value " + "\"%s\"...", (char *)attr->values[j].unknown.data)); + break; + + case IPP_TAG_BEGIN_COLLECTION : + /* + * Collection value + */ + + num_cols = cupsParseOptions(val, 0, &cols); + if ((collection = ippNew()) == NULL) + { + cupsFreeOptions(num_cols, cols); + + if (copy) + free(copy); + + ippDeleteAttribute(ipp, attr); + return; + } + + attr->values[j].collection = collection; + cupsEncodeOptions2(collection, num_cols, cols, IPP_TAG_JOB); + cupsFreeOptions(num_cols, cols); + break; + + default : + if ((attr->values[j].string.text = _cupsStrAlloc(val)) == NULL) + { + /* + * Ran out of memory! + */ + + DEBUG_puts("1cupsEncodeOptions2: Ran out of memory for string!"); + + if (copy) + free(copy); + + ippDeleteAttribute(ipp, attr); + return; + } + + DEBUG_printf(("2cupsEncodeOptions2: Added string value \"%s\"...", + val)); + break; + } + } + + if (copy) + free(copy); + } +} + + +/* + * '_ippFindOption()' - Find the attribute information for an option. + */ + +_ipp_option_t * /* O - Attribute information */ +_ippFindOption(const char *name) /* I - Option/attribute name */ +{ + _ipp_option_t key; /* Search key */ + + + /* + * Lookup the proper value and group tags for this option... + */ + + key.name = name; + + return ((_ipp_option_t *)bsearch(&key, ipp_options, + sizeof(ipp_options) / sizeof(ipp_options[0]), + sizeof(ipp_options[0]), + (int (*)(const void *, const void *)) + compare_ipp_options)); +} + + +/* + * 'compare_ipp_options()' - Compare two IPP options. + */ + +static int /* O - Result of comparison */ +compare_ipp_options(_ipp_option_t *a, /* I - First option */ + _ipp_option_t *b) /* I - Second option */ +{ + return (strcmp(a->name, b->name)); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/file-private.h b/cups/file-private.h new file mode 100644 index 0000000000..60a2501488 --- /dev/null +++ b/cups/file-private.h @@ -0,0 +1,137 @@ +/* + * "$Id$" + * + * Private file definitions for CUPS. + * + * Since stdio files max out at 256 files on many systems, we have to + * write similar functions without this limit. At the same time, using + * our own file functions allows us to provide transparent support of + * gzip'd print files, PPD files, etc. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#ifndef _CUPS_FILE_PRIVATE_H_ +# define _CUPS_FILE_PRIVATE_H_ + +/* + * Include necessary headers... + */ + +# include "cups-private.h" +# include +# include +# include +# include + +# ifdef HAVE_LIBZ +# include +# endif /* HAVE_LIBZ */ +# ifdef WIN32 +# include +# include +# endif /* WIN32 */ + + +/* + * Some operating systems support large files via open flag O_LARGEFILE... + */ + +# ifndef O_LARGEFILE +# define O_LARGEFILE 0 +# endif /* !O_LARGEFILE */ + + +/* + * Some operating systems don't define O_BINARY, which is used by Microsoft + * and IBM to flag binary files... + */ + +# ifndef O_BINARY +# define O_BINARY 0 +# endif /* !O_BINARY */ + + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Types and structures... + */ + +typedef enum /**** _cupsFileCheck return values ****/ +{ + _CUPS_FILE_CHECK_OK = 0, /* Everything OK */ + _CUPS_FILE_CHECK_MISSING = 1, /* File is missing */ + _CUPS_FILE_CHECK_PERMISSIONS = 2, /* File (or parent dir) has bad perms */ + _CUPS_FILE_CHECK_WRONG_TYPE = 3, /* File has wrong type */ + _CUPS_FILE_CHECK_RELATIVE_PATH = 4 /* File contains a relative path */ +} _cups_fc_result_t; + +typedef enum /**** _cupsFileCheck file type values ****/ +{ + _CUPS_FILE_CHECK_FILE = 0, /* Check the file and parent directory */ + _CUPS_FILE_CHECK_PROGRAM = 1, /* Check the program and parent directory */ + _CUPS_FILE_CHECK_FILE_ONLY = 2, /* Check the file only */ + _CUPS_FILE_CHECK_DIRECTORY = 3 /* Check the directory */ +} _cups_fc_filetype_t; + +typedef void (*_cups_fc_func_t)(void *context, _cups_fc_result_t result, + const char *message); + +struct _cups_file_s /**** CUPS file structure... ****/ + +{ + int fd; /* File descriptor */ + char mode, /* Mode ('r' or 'w') */ + compressed, /* Compression used? */ + is_stdio, /* stdin/out/err? */ + eof, /* End of file? */ + buf[4096], /* Buffer */ + *ptr, /* Pointer into buffer */ + *end; /* End of buffer data */ + off_t pos, /* Position in file */ + bufpos; /* File position for start of buffer */ + +#ifdef HAVE_LIBZ + z_stream stream; /* (De)compression stream */ + Bytef cbuf[4096]; /* (De)compression buffer */ + uLong crc; /* (De)compression CRC */ +#endif /* HAVE_LIBZ */ + + char *printf_buffer; /* cupsFilePrintf buffer */ + size_t printf_size; /* Size of cupsFilePrintf buffer */ +}; + + +/* + * Prototypes... + */ + +extern _cups_fc_result_t _cupsFileCheck(const char *filename, + _cups_fc_filetype_t filetype, + int dorootchecks, + _cups_fc_func_t cb, + void *context); +extern void _cupsFileCheckFilter(void *context, + _cups_fc_result_t result, + const char *message); + +# ifdef __cplusplus +} +# endif /* __cplusplus */ + +#endif /* !_CUPS_FILE_PRIVATE_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/file.c b/cups/file.c new file mode 100644 index 0000000000..a5fae103a4 --- /dev/null +++ b/cups/file.c @@ -0,0 +1,2714 @@ +/* + * "$Id$" + * + * File functions for CUPS. + * + * Since stdio files max out at 256 files on many systems, we have to + * write similar functions without this limit. At the same time, using + * our own file functions allows us to provide transparent support of + * gzip'd print files, PPD files, etc. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * _cupsFileCheck() - Check the permissions of the given filename. + * _cupsFileCheckFilter() - Report file check results as CUPS filter messages. + * cupsFileClose() - Close a CUPS file. + * cupsFileCompression() - Return whether a file is compressed. + * cupsFileEOF() - Return the end-of-file status. + * cupsFileFind() - Find a file using the specified path. + * cupsFileFlush() - Flush pending output. + * cupsFileGetChar() - Get a single character from a file. + * cupsFileGetConf() - Get a line from a configuration file. + * cupsFileGetLine() - Get a CR and/or LF-terminated line that may + * contain binary data. + * cupsFileGets() - Get a CR and/or LF-terminated line. + * cupsFileLock() - Temporarily lock access to a file. + * cupsFileNumber() - Return the file descriptor associated with a CUPS + * file. + * cupsFileOpen() - Open a CUPS file. + * cupsFileOpenFd() - Open a CUPS file using a file descriptor. + * cupsFilePeekChar() - Peek at the next character from a file. + * cupsFilePrintf() - Write a formatted string. + * cupsFilePutChar() - Write a character. + * cupsFilePutConf() - Write a configuration line. + * cupsFilePuts() - Write a string. + * cupsFileRead() - Read from a file. + * cupsFileRewind() - Set the current file position to the beginning of + * the file. + * cupsFileSeek() - Seek in a file. + * cupsFileStderr() - Return a CUPS file associated with stderr. + * cupsFileStdin() - Return a CUPS file associated with stdin. + * cupsFileStdout() - Return a CUPS file associated with stdout. + * cupsFileTell() - Return the current file position. + * cupsFileUnlock() - Unlock access to a file. + * cupsFileWrite() - Write to a file. + * cups_compress() - Compress a buffer of data. + * cups_fill() - Fill the input buffer. + * cups_open() - Safely open a file for writing. + * cups_read() - Read from a file descriptor. + * cups_write() - Write to a file descriptor. + */ + +/* + * Include necessary headers... + */ + +#include "file-private.h" +#include +#include + + +/* + * Local functions... + */ + +#ifdef HAVE_LIBZ +static ssize_t cups_compress(cups_file_t *fp, const char *buf, size_t bytes); +#endif /* HAVE_LIBZ */ +static ssize_t cups_fill(cups_file_t *fp); +static int cups_open(const char *filename, int mode); +static ssize_t cups_read(cups_file_t *fp, char *buf, size_t bytes); +static ssize_t cups_write(cups_file_t *fp, const char *buf, size_t bytes); + + +#ifndef WIN32 +/* + * '_cupsFileCheck()' - Check the permissions of the given filename. + */ + +_cups_fc_result_t /* O - Check result */ +_cupsFileCheck( + const char *filename, /* I - Filename to check */ + _cups_fc_filetype_t filetype, /* I - Type of file checks? */ + int dorootchecks, /* I - Check for root permissions? */ + _cups_fc_func_t cb, /* I - Callback function */ + void *context) /* I - Context pointer for callback */ + +{ + struct stat fileinfo; /* File information */ + char message[1024], /* Message string */ + temp[1024], /* Parent directory filename */ + *ptr; /* Pointer into parent directory */ + _cups_fc_result_t result; /* Check result */ + + + /* + * Does the filename contain a relative path ("../")? + */ + + if (strstr(filename, "../")) + { + /* + * Yes, fail it! + */ + + result = _CUPS_FILE_CHECK_RELATIVE_PATH; + goto finishup; + } + + /* + * Does the program even exist and is it accessible? + */ + + if (stat(filename, &fileinfo)) + { + /* + * Nope... + */ + + result = _CUPS_FILE_CHECK_MISSING; + goto finishup; + } + + /* + * Check the execute bit... + */ + + result = _CUPS_FILE_CHECK_OK; + + switch (filetype) + { + case _CUPS_FILE_CHECK_DIRECTORY : + if (!S_ISDIR(fileinfo.st_mode)) + result = _CUPS_FILE_CHECK_WRONG_TYPE; + break; + + default : + if (!S_ISREG(fileinfo.st_mode)) + result = _CUPS_FILE_CHECK_WRONG_TYPE; + break; + } + + if (result) + goto finishup; + + /* + * Are we doing root checks? + */ + + if (!dorootchecks) + { + /* + * Nope, so anything (else) goes... + */ + + goto finishup; + } + + /* + * Verify permission of the file itself: + * + * 1. Must be owned by root + * 2. Must not be writable by group + * 3. Must not be setuid + * 4. Must not be writable by others + */ + + if (fileinfo.st_uid || /* 1. Must be owned by root */ + (fileinfo.st_mode & S_IWGRP) || /* 2. Must not be writable by group */ + (fileinfo.st_mode & S_ISUID) || /* 3. Must not be setuid */ + (fileinfo.st_mode & S_IWOTH)) /* 4. Must not be writable by others */ + { + result = _CUPS_FILE_CHECK_PERMISSIONS; + goto finishup; + } + + if (filetype == _CUPS_FILE_CHECK_DIRECTORY || + filetype == _CUPS_FILE_CHECK_FILE_ONLY) + goto finishup; + + /* + * Now check the containing directory... + */ + + strlcpy(temp, filename, sizeof(temp)); + if ((ptr = strrchr(temp, '/')) != NULL) + { + if (ptr == temp) + ptr[1] = '\0'; + else + *ptr = '\0'; + } + + if (stat(temp, &fileinfo)) + { + /* + * Doesn't exist?!? + */ + + result = _CUPS_FILE_CHECK_MISSING; + filetype = _CUPS_FILE_CHECK_DIRECTORY; + filename = temp; + + goto finishup; + } + + if (fileinfo.st_uid || /* 1. Must be owned by root */ + (fileinfo.st_mode & S_IWGRP) || /* 2. Must not be writable by group */ + (fileinfo.st_mode & S_ISUID) || /* 3. Must not be setuid */ + (fileinfo.st_mode & S_IWOTH)) /* 4. Must not be writable by others */ + { + result = _CUPS_FILE_CHECK_PERMISSIONS; + filetype = _CUPS_FILE_CHECK_DIRECTORY; + filename = temp; + } + + /* + * Common return point... + */ + + finishup: + + if (cb) + { + cups_lang_t *lang = cupsLangDefault(); + /* Localization information */ + + switch (result) + { + case _CUPS_FILE_CHECK_OK : + if (filetype == _CUPS_FILE_CHECK_DIRECTORY) + snprintf(message, sizeof(message), + _cupsLangString(lang, _("Directory \"%s\" permissions OK " + "(0%o/uid=%d/gid=%d).")), + filename, fileinfo.st_mode, (int)fileinfo.st_uid, + (int)fileinfo.st_gid); + else + snprintf(message, sizeof(message), + _cupsLangString(lang, _("File \"%s\" permissions OK " + "(0%o/uid=%d/gid=%d).")), + filename, fileinfo.st_mode, (int)fileinfo.st_uid, + (int)fileinfo.st_gid); + break; + + case _CUPS_FILE_CHECK_MISSING : + if (filetype == _CUPS_FILE_CHECK_DIRECTORY) + snprintf(message, sizeof(message), + _cupsLangString(lang, _("Directory \"%s\" not available: " + "%s")), + filename, strerror(errno)); + else + snprintf(message, sizeof(message), + _cupsLangString(lang, _("File \"%s\" not available: %s")), + filename, strerror(errno)); + break; + + case _CUPS_FILE_CHECK_PERMISSIONS : + if (filetype == _CUPS_FILE_CHECK_DIRECTORY) + snprintf(message, sizeof(message), + _cupsLangString(lang, _("Directory \"%s\" has insecure " + "permissions " + "(0%o/uid=%d/gid=%d).")), + filename, fileinfo.st_mode, (int)fileinfo.st_uid, + (int)fileinfo.st_gid); + else + snprintf(message, sizeof(message), + _cupsLangString(lang, _("File \"%s\" has insecure " + "permissions " + "(0%o/uid=%d/gid=%d).")), + filename, fileinfo.st_mode, (int)fileinfo.st_uid, + (int)fileinfo.st_gid); + break; + + case _CUPS_FILE_CHECK_WRONG_TYPE : + if (filetype == _CUPS_FILE_CHECK_DIRECTORY) + snprintf(message, sizeof(message), + _cupsLangString(lang, _("Directory \"%s\" is a file.")), + filename); + else + snprintf(message, sizeof(message), + _cupsLangString(lang, _("File \"%s\" is a directory.")), + filename); + break; + + case _CUPS_FILE_CHECK_RELATIVE_PATH : + if (filetype == _CUPS_FILE_CHECK_DIRECTORY) + snprintf(message, sizeof(message), + _cupsLangString(lang, _("Directory \"%s\" contains a " + "relative path.")), filename); + else + snprintf(message, sizeof(message), + _cupsLangString(lang, _("File \"%s\" contains a relative " + "path.")), filename); + break; + } + + (*cb)(context, result, message); + } + + return (result); +} + + +/* + * '_cupsFileCheckFilter()' - Report file check results as CUPS filter messages. + */ + +void +_cupsFileCheckFilter( + void *context, /* I - Context pointer (unused) */ + _cups_fc_result_t result, /* I - Result code */ + const char *message) /* I - Message text */ +{ + const char *prefix; /* Messaging prefix */ + + + (void)context; + + switch (result) + { + default : + case _CUPS_FILE_CHECK_OK : + prefix = "DEBUG2"; + break; + + case _CUPS_FILE_CHECK_MISSING : + case _CUPS_FILE_CHECK_WRONG_TYPE : + prefix = "ERROR"; + fputs("STATE: +cups-missing-filter-warning\n", stderr); + break; + + case _CUPS_FILE_CHECK_PERMISSIONS : + case _CUPS_FILE_CHECK_RELATIVE_PATH : + prefix = "ERROR"; + fputs("STATE: +cups-insecure-filter-warning\n", stderr); + break; + } + + fprintf(stderr, "%s: %s\n", prefix, message); +} +#endif /* !WIN32 */ + + +/* + * 'cupsFileClose()' - Close a CUPS file. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 0 on success, -1 on error */ +cupsFileClose(cups_file_t *fp) /* I - CUPS file */ +{ + int fd; /* File descriptor */ + char mode; /* Open mode */ + int status; /* Return status */ + int is_stdio; /* Is a stdio file? */ + + + DEBUG_printf(("cupsFileClose(fp=%p)", fp)); + + /* + * Range check... + */ + + if (!fp) + return (-1); + + /* + * Flush pending write data... + */ + + if (fp->mode == 'w') + status = cupsFileFlush(fp); + else + status = 0; + +#ifdef HAVE_LIBZ + if (fp->compressed && status >= 0) + { + if (fp->mode == 'r') + { + /* + * Free decompression data... + */ + + inflateEnd(&fp->stream); + } + else + { + /* + * Flush any remaining compressed data... + */ + + unsigned char trailer[8]; /* Trailer CRC and length */ + int done; /* Done writing... */ + + + fp->stream.avail_in = 0; + + for (done = 0;;) + { + if (fp->stream.next_out > fp->cbuf) + { + if (cups_write(fp, (char *)fp->cbuf, + fp->stream.next_out - fp->cbuf) < 0) + status = -1; + + fp->stream.next_out = fp->cbuf; + fp->stream.avail_out = sizeof(fp->cbuf); + } + + if (done || status < 0) + break; + + done = deflate(&fp->stream, Z_FINISH) == Z_STREAM_END && + fp->stream.next_out == fp->cbuf; + } + + /* + * Write the CRC and length... + */ + + trailer[0] = fp->crc; + trailer[1] = fp->crc >> 8; + trailer[2] = fp->crc >> 16; + trailer[3] = fp->crc >> 24; + trailer[4] = fp->pos; + trailer[5] = fp->pos >> 8; + trailer[6] = fp->pos >> 16; + trailer[7] = fp->pos >> 24; + + if (cups_write(fp, (char *)trailer, 8) < 0) + status = -1; + + /* + * Free all memory used by the compression stream... + */ + + deflateEnd(&(fp->stream)); + } + } +#endif /* HAVE_LIBZ */ + + /* + * Save the file descriptor we used and free memory... + */ + + fd = fp->fd; + mode = fp->mode; + is_stdio = fp->is_stdio; + + if (fp->printf_buffer) + free(fp->printf_buffer); + + free(fp); + + /* + * Close the file, returning the close status... + */ + + if (mode == 's') + { + if (closesocket(fd) < 0) + status = -1; + } + else if (!is_stdio) + { + if (close(fd) < 0) + status = -1; + } + + return (status); +} + + +/* + * 'cupsFileCompression()' - Return whether a file is compressed. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - @code CUPS_FILE_NONE@ or @code CUPS_FILE_GZIP@ */ +cupsFileCompression(cups_file_t *fp) /* I - CUPS file */ +{ + return (fp ? fp->compressed : CUPS_FILE_NONE); +} + + +/* + * 'cupsFileEOF()' - Return the end-of-file status. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 1 on end of file, 0 otherwise */ +cupsFileEOF(cups_file_t *fp) /* I - CUPS file */ +{ + return (fp ? fp->eof : 1); +} + + +/* + * 'cupsFileFind()' - Find a file using the specified path. + * + * This function allows the paths in the path string to be separated by + * colons (UNIX standard) or semicolons (Windows standard) and stores the + * result in the buffer supplied. If the file cannot be found in any of + * the supplied paths, @code NULL@ is returned. A @code NULL@ path only + * matches the current directory. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +const char * /* O - Full path to file or @code NULL@ if not found */ +cupsFileFind(const char *filename, /* I - File to find */ + const char *path, /* I - Colon/semicolon-separated path */ + int executable, /* I - 1 = executable files, 0 = any file/dir */ + char *buffer, /* I - Filename buffer */ + int bufsize) /* I - Size of filename buffer */ +{ + char *bufptr, /* Current position in buffer */ + *bufend; /* End of buffer */ + + + /* + * Range check input... + */ + + DEBUG_printf(("cupsFileFind(filename=\"%s\", path=\"%s\", executable=%d, " + "buffer=%p, bufsize=%d)", filename, path, executable, buffer, + bufsize)); + + if (!filename || !buffer || bufsize < 2) + return (NULL); + + if (!path) + { + /* + * No path, so check current directory... + */ + + if (!access(filename, 0)) + { + strlcpy(buffer, filename, bufsize); + return (buffer); + } + else + return (NULL); + } + + /* + * Now check each path and return the first match... + */ + + bufend = buffer + bufsize - 1; + bufptr = buffer; + + while (*path) + { +#ifdef WIN32 + if (*path == ';' || (*path == ':' && ((bufptr - buffer) > 1 || !isalpha(buffer[0] & 255)))) +#else + if (*path == ';' || *path == ':') +#endif /* WIN32 */ + { + if (bufptr > buffer && bufptr[-1] != '/' && bufptr < bufend) + *bufptr++ = '/'; + + strlcpy(bufptr, filename, bufend - bufptr); + +#ifdef WIN32 + if (!access(buffer, 0)) +#else + if (!access(buffer, executable ? X_OK : 0)) +#endif /* WIN32 */ + { + DEBUG_printf(("1cupsFileFind: Returning \"%s\"", buffer)); + return (buffer); + } + + bufptr = buffer; + } + else if (bufptr < bufend) + *bufptr++ = *path; + + path ++; + } + + /* + * Check the last path... + */ + + if (bufptr > buffer && bufptr[-1] != '/' && bufptr < bufend) + *bufptr++ = '/'; + + strlcpy(bufptr, filename, bufend - bufptr); + + if (!access(buffer, 0)) + { + DEBUG_printf(("1cupsFileFind: Returning \"%s\"", buffer)); + return (buffer); + } + else + { + DEBUG_puts("1cupsFileFind: Returning NULL"); + return (NULL); + } +} + + +/* + * 'cupsFileFlush()' - Flush pending output. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 0 on success, -1 on error */ +cupsFileFlush(cups_file_t *fp) /* I - CUPS file */ +{ + ssize_t bytes; /* Bytes to write */ + + + DEBUG_printf(("cupsFileFlush(fp=%p)", fp)); + + /* + * Range check input... + */ + + if (!fp || fp->mode != 'w') + { + DEBUG_puts("1cupsFileFlush: Attempt to flush a read-only file..."); + return (-1); + } + + bytes = (ssize_t)(fp->ptr - fp->buf); + + DEBUG_printf(("2cupsFileFlush: Flushing " CUPS_LLFMT " bytes...", + CUPS_LLCAST bytes)); + + if (bytes > 0) + { +#ifdef HAVE_LIBZ + if (fp->compressed) + bytes = cups_compress(fp, fp->buf, bytes); + else +#endif /* HAVE_LIBZ */ + bytes = cups_write(fp, fp->buf, bytes); + + if (bytes < 0) + return (-1); + + fp->ptr = fp->buf; + } + + return (0); +} + + +/* + * 'cupsFileGetChar()' - Get a single character from a file. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - Character or -1 on end of file */ +cupsFileGetChar(cups_file_t *fp) /* I - CUPS file */ +{ + /* + * Range check input... + */ + + if (!fp || (fp->mode != 'r' && fp->mode != 's')) + { + DEBUG_puts("5cupsFileGetChar: Bad arguments!"); + return (-1); + } + + /* + * If the input buffer is empty, try to read more data... + */ + + if (fp->ptr >= fp->end) + if (cups_fill(fp) < 0) + { + DEBUG_puts("5cupsFileGetChar: Unable to fill buffer!"); + return (-1); + } + + /* + * Return the next character in the buffer... + */ + + DEBUG_printf(("5cupsFileGetChar: Returning %d...", *(fp->ptr) & 255)); + + fp->pos ++; + + DEBUG_printf(("6cupsFileGetChar: pos=" CUPS_LLFMT, CUPS_LLCAST fp->pos)); + + return (*(fp->ptr)++ & 255); +} + + +/* + * 'cupsFileGetConf()' - Get a line from a configuration file. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +char * /* O - Line read or @code NULL@ on end of file or error */ +cupsFileGetConf(cups_file_t *fp, /* I - CUPS file */ + char *buf, /* O - String buffer */ + size_t buflen, /* I - Size of string buffer */ + char **value, /* O - Pointer to value */ + int *linenum) /* IO - Current line number */ +{ + char *ptr; /* Pointer into line */ + + + /* + * Range check input... + */ + + DEBUG_printf(("2cupsFileGetConf(fp=%p, buf=%p, buflen=" CUPS_LLFMT + ", value=%p, linenum=%p)", fp, buf, CUPS_LLCAST buflen, + value, linenum)); + + if (!fp || (fp->mode != 'r' && fp->mode != 's') || + !buf || buflen < 2 || !value) + { + if (value) + *value = NULL; + + return (NULL); + } + + /* + * Read the next non-comment line... + */ + + *value = NULL; + + while (cupsFileGets(fp, buf, buflen)) + { + (*linenum) ++; + + /* + * Strip any comments... + */ + + if ((ptr = strchr(buf, '#')) != NULL) + { + if (ptr > buf && ptr[-1] == '\\') + { + // Unquote the #... + _cups_strcpy(ptr - 1, ptr); + } + else + { + // Strip the comment and any trailing whitespace... + while (ptr > buf) + { + if (!_cups_isspace(ptr[-1])) + break; + + ptr --; + } + + *ptr = '\0'; + } + } + + /* + * Strip leading whitespace... + */ + + for (ptr = buf; _cups_isspace(*ptr); ptr ++); + + if (ptr > buf) + _cups_strcpy(buf, ptr); + + /* + * See if there is anything left... + */ + + if (buf[0]) + { + /* + * Yes, grab any value and return... + */ + + for (ptr = buf; *ptr; ptr ++) + if (_cups_isspace(*ptr)) + break; + + if (*ptr) + { + /* + * Have a value, skip any other spaces... + */ + + while (_cups_isspace(*ptr)) + *ptr++ = '\0'; + + if (*ptr) + *value = ptr; + + /* + * Strip trailing whitespace and > for lines that begin with <... + */ + + ptr += strlen(ptr) - 1; + + if (buf[0] == '<' && *ptr == '>') + *ptr-- = '\0'; + else if (buf[0] == '<' && *ptr != '>') + { + /* + * Syntax error... + */ + + *value = NULL; + return (buf); + } + + while (ptr > *value && _cups_isspace(*ptr)) + *ptr-- = '\0'; + } + + /* + * Return the line... + */ + + return (buf); + } + } + + return (NULL); +} + + +/* + * 'cupsFileGetLine()' - Get a CR and/or LF-terminated line that may + * contain binary data. + * + * This function differs from @link cupsFileGets@ in that the trailing CR + * and LF are preserved, as is any binary data on the line. The buffer is + * nul-terminated, however you should use the returned length to determine + * the number of bytes on the line. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +size_t /* O - Number of bytes on line or 0 on end of file */ +cupsFileGetLine(cups_file_t *fp, /* I - File to read from */ + char *buf, /* I - Buffer */ + size_t buflen) /* I - Size of buffer */ +{ + int ch; /* Character from file */ + char *ptr, /* Current position in line buffer */ + *end; /* End of line buffer */ + + + /* + * Range check input... + */ + + DEBUG_printf(("2cupsFileGetLine(fp=%p, buf=%p, buflen=" CUPS_LLFMT ")", + fp, buf, CUPS_LLCAST buflen)); + + if (!fp || (fp->mode != 'r' && fp->mode != 's') || !buf || buflen < 3) + return (0); + + /* + * Now loop until we have a valid line... + */ + + for (ptr = buf, end = buf + buflen - 2; ptr < end ;) + { + if (fp->ptr >= fp->end) + if (cups_fill(fp) <= 0) + break; + + *ptr++ = ch = *(fp->ptr)++; + fp->pos ++; + + if (ch == '\r') + { + /* + * Check for CR LF... + */ + + if (fp->ptr >= fp->end) + if (cups_fill(fp) <= 0) + break; + + if (*(fp->ptr) == '\n') + { + *ptr++ = *(fp->ptr)++; + fp->pos ++; + } + + break; + } + else if (ch == '\n') + { + /* + * Line feed ends a line... + */ + + break; + } + } + + *ptr = '\0'; + + DEBUG_printf(("4cupsFileGetLine: pos=" CUPS_LLFMT, CUPS_LLCAST fp->pos)); + + return (ptr - buf); +} + + +/* + * 'cupsFileGets()' - Get a CR and/or LF-terminated line. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +char * /* O - Line read or @code NULL@ on end of file or error */ +cupsFileGets(cups_file_t *fp, /* I - CUPS file */ + char *buf, /* O - String buffer */ + size_t buflen) /* I - Size of string buffer */ +{ + int ch; /* Character from file */ + char *ptr, /* Current position in line buffer */ + *end; /* End of line buffer */ + + + /* + * Range check input... + */ + + DEBUG_printf(("2cupsFileGets(fp=%p, buf=%p, buflen=" CUPS_LLFMT ")", fp, buf, + CUPS_LLCAST buflen)); + + if (!fp || (fp->mode != 'r' && fp->mode != 's') || !buf || buflen < 2) + return (NULL); + + /* + * Now loop until we have a valid line... + */ + + for (ptr = buf, end = buf + buflen - 1; ptr < end ;) + { + if (fp->ptr >= fp->end) + if (cups_fill(fp) <= 0) + { + if (ptr == buf) + return (NULL); + else + break; + } + + ch = *(fp->ptr)++; + fp->pos ++; + + if (ch == '\r') + { + /* + * Check for CR LF... + */ + + if (fp->ptr >= fp->end) + if (cups_fill(fp) <= 0) + break; + + if (*(fp->ptr) == '\n') + { + fp->ptr ++; + fp->pos ++; + } + + break; + } + else if (ch == '\n') + { + /* + * Line feed ends a line... + */ + + break; + } + else + *ptr++ = ch; + } + + *ptr = '\0'; + + DEBUG_printf(("4cupsFileGets: pos=" CUPS_LLFMT, CUPS_LLCAST fp->pos)); + + return (buf); +} + + +/* + * 'cupsFileLock()' - Temporarily lock access to a file. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 0 on success, -1 on error */ +cupsFileLock(cups_file_t *fp, /* I - CUPS file */ + int block) /* I - 1 to wait for the lock, 0 to fail right away */ +{ + /* + * Range check... + */ + + if (!fp || fp->mode == 's') + return (-1); + + /* + * Try the lock... + */ + +#ifdef WIN32 + return (_locking(fp->fd, block ? _LK_LOCK : _LK_NBLCK, 0)); +#else + return (lockf(fp->fd, block ? F_LOCK : F_TLOCK, 0)); +#endif /* WIN32 */ +} + + +/* + * 'cupsFileNumber()' - Return the file descriptor associated with a CUPS file. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - File descriptor */ +cupsFileNumber(cups_file_t *fp) /* I - CUPS file */ +{ + if (fp) + return (fp->fd); + else + return (-1); +} + + +/* + * 'cupsFileOpen()' - Open a CUPS file. + * + * The "mode" parameter can be "r" to read, "w" to write, overwriting any + * existing file, "a" to append to an existing file or create a new file, + * or "s" to open a socket connection. + * + * When opening for writing ("w"), an optional number from 1 to 9 can be + * supplied which enables Flate compression of the file. Compression is + * not supported for the "a" (append) mode. + * + * When opening a socket connection, the filename is a string of the form + * "address:port" or "hostname:port". The socket will make an IPv4 or IPv6 + * connection as needed, generally preferring IPv6 connections when there is + * a choice. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +cups_file_t * /* O - CUPS file or @code NULL@ if the file or socket cannot be opened */ +cupsFileOpen(const char *filename, /* I - Name of file */ + const char *mode) /* I - Open mode */ +{ + cups_file_t *fp; /* New CUPS file */ + int fd; /* File descriptor */ + char hostname[1024], /* Hostname */ + *portname; /* Port "name" (number or service) */ + http_addrlist_t *addrlist; /* Host address list */ + + + DEBUG_printf(("cupsFileOpen(filename=\"%s\", mode=\"%s\")", filename, + mode)); + + /* + * Range check input... + */ + + if (!filename || !mode || + (*mode != 'r' && *mode != 'w' && *mode != 'a' && *mode != 's') || + (*mode == 'a' && isdigit(mode[1] & 255))) + return (NULL); + + /* + * Open the file... + */ + + switch (*mode) + { + case 'a' : /* Append file */ + fd = cups_open(filename, + O_RDWR | O_CREAT | O_APPEND | O_LARGEFILE | O_BINARY); + break; + + case 'r' : /* Read file */ + fd = open(filename, O_RDONLY | O_LARGEFILE | O_BINARY, 0); + break; + + case 'w' : /* Write file */ + fd = cups_open(filename, O_WRONLY | O_LARGEFILE | O_BINARY); + if (fd < 0 && errno == ENOENT) + { + fd = cups_open(filename, + O_WRONLY | O_CREAT | O_EXCL | O_LARGEFILE | O_BINARY); + if (fd < 0 && errno == EEXIST) + fd = cups_open(filename, O_WRONLY | O_LARGEFILE | O_BINARY); + } + + if (fd >= 0) +#ifdef WIN32 + _chsize(fd, 0); +#else + ftruncate(fd, 0); +#endif /* WIN32 */ + break; + + case 's' : /* Read/write socket */ + strlcpy(hostname, filename, sizeof(hostname)); + if ((portname = strrchr(hostname, ':')) != NULL) + *portname++ = '\0'; + else + return (NULL); + + /* + * Lookup the hostname and service... + */ + + if ((addrlist = httpAddrGetList(hostname, AF_UNSPEC, portname)) == NULL) + return (NULL); + + /* + * Connect to the server... + */ + + if (!httpAddrConnect(addrlist, &fd)) + { + httpAddrFreeList(addrlist); + return (NULL); + } + + httpAddrFreeList(addrlist); + break; + + default : /* Remove bogus compiler warning... */ + return (NULL); + } + + if (fd < 0) + return (NULL); + + /* + * Create the CUPS file structure... + */ + + if ((fp = cupsFileOpenFd(fd, mode)) == NULL) + { + if (*mode == 's') + closesocket(fd); + else + close(fd); + } + + /* + * Return it... + */ + + return (fp); +} + +/* + * 'cupsFileOpenFd()' - Open a CUPS file using a file descriptor. + * + * The "mode" parameter can be "r" to read, "w" to write, "a" to append, + * or "s" to treat the file descriptor as a bidirectional socket connection. + * + * When opening for writing ("w"), an optional number from 1 to 9 can be + * supplied which enables Flate compression of the file. Compression is + * not supported for the "a" (append) mode. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +cups_file_t * /* O - CUPS file or @code NULL@ if the file could not be opened */ +cupsFileOpenFd(int fd, /* I - File descriptor */ + const char *mode) /* I - Open mode */ +{ + cups_file_t *fp; /* New CUPS file */ + + + DEBUG_printf(("cupsFileOpenFd(fd=%d, mode=\"%s\")", fd, mode)); + + /* + * Range check input... + */ + + if (fd < 0 || !mode || + (*mode != 'r' && *mode != 'w' && *mode != 'a' && *mode != 's') || + (*mode == 'a' && isdigit(mode[1] & 255))) + return (NULL); + + /* + * Allocate memory... + */ + + if ((fp = calloc(1, sizeof(cups_file_t))) == NULL) + return (NULL); + + /* + * Open the file... + */ + + fp->fd = fd; + + switch (*mode) + { + case 'a' : + fp->pos = lseek(fd, 0, SEEK_END); + + case 'w' : + fp->mode = 'w'; + fp->ptr = fp->buf; + fp->end = fp->buf + sizeof(fp->buf); + +#ifdef HAVE_LIBZ + if (mode[1] >= '1' && mode[1] <= '9') + { + /* + * Open a compressed stream, so write the standard gzip file + * header... + */ + + unsigned char header[10]; /* gzip file header */ + time_t curtime; /* Current time */ + + + curtime = time(NULL); + header[0] = 0x1f; + header[1] = 0x8b; + header[2] = Z_DEFLATED; + header[3] = 0; + header[4] = curtime; + header[5] = curtime >> 8; + header[6] = curtime >> 16; + header[7] = curtime >> 24; + header[8] = 0; + header[9] = 0x03; + + cups_write(fp, (char *)header, 10); + + /* + * Initialize the compressor... + */ + + deflateInit2(&(fp->stream), mode[1] - '0', Z_DEFLATED, -15, 8, + Z_DEFAULT_STRATEGY); + + fp->stream.next_out = fp->cbuf; + fp->stream.avail_out = sizeof(fp->cbuf); + fp->compressed = 1; + fp->crc = crc32(0L, Z_NULL, 0); + } +#endif /* HAVE_LIBZ */ + break; + + case 'r' : + fp->mode = 'r'; + break; + + case 's' : + fp->mode = 's'; + break; + + default : /* Remove bogus compiler warning... */ + return (NULL); + } + + /* + * Don't pass this file to child processes... + */ + +#ifndef WIN32 + fcntl(fp->fd, F_SETFD, fcntl(fp->fd, F_GETFD) | FD_CLOEXEC); +#endif /* !WIN32 */ + + return (fp); +} + + +/* + * 'cupsFilePeekChar()' - Peek at the next character from a file. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - Character or -1 on end of file */ +cupsFilePeekChar(cups_file_t *fp) /* I - CUPS file */ +{ + /* + * Range check input... + */ + + if (!fp || (fp->mode != 'r' && fp->mode != 's')) + return (-1); + + /* + * If the input buffer is empty, try to read more data... + */ + + if (fp->ptr >= fp->end) + if (cups_fill(fp) < 0) + return (-1); + + /* + * Return the next character in the buffer... + */ + + return (*(fp->ptr) & 255); +} + + +/* + * 'cupsFilePrintf()' - Write a formatted string. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - Number of bytes written or -1 on error */ +cupsFilePrintf(cups_file_t *fp, /* I - CUPS file */ + const char *format, /* I - Printf-style format string */ + ...) /* I - Additional args as necessary */ +{ + va_list ap; /* Argument list */ + ssize_t bytes; /* Formatted size */ + + + DEBUG_printf(("2cupsFilePrintf(fp=%p, format=\"%s\", ...)", fp, format)); + + if (!fp || !format || (fp->mode != 'w' && fp->mode != 's')) + return (-1); + + if (!fp->printf_buffer) + { + /* + * Start with an 1k printf buffer... + */ + + if ((fp->printf_buffer = malloc(1024)) == NULL) + return (-1); + + fp->printf_size = 1024; + } + + va_start(ap, format); + bytes = vsnprintf(fp->printf_buffer, fp->printf_size, format, ap); + va_end(ap); + + if (bytes >= (ssize_t)fp->printf_size) + { + /* + * Expand the printf buffer... + */ + + char *temp; /* Temporary buffer pointer */ + + + if (bytes > 65535) + return (-1); + + if ((temp = realloc(fp->printf_buffer, bytes + 1)) == NULL) + return (-1); + + fp->printf_buffer = temp; + fp->printf_size = bytes + 1; + + va_start(ap, format); + bytes = vsnprintf(fp->printf_buffer, fp->printf_size, format, ap); + va_end(ap); + } + + if (fp->mode == 's') + { + if (cups_write(fp, fp->printf_buffer, bytes) < 0) + return (-1); + + fp->pos += bytes; + + DEBUG_printf(("4cupsFilePrintf: pos=" CUPS_LLFMT, CUPS_LLCAST fp->pos)); + + return (bytes); + } + + if ((fp->ptr + bytes) > fp->end) + if (cupsFileFlush(fp)) + return (-1); + + fp->pos += bytes; + + DEBUG_printf(("4cupsFilePrintf: pos=" CUPS_LLFMT, CUPS_LLCAST fp->pos)); + + if (bytes > sizeof(fp->buf)) + { +#ifdef HAVE_LIBZ + if (fp->compressed) + return (cups_compress(fp, fp->printf_buffer, bytes)); + else +#endif /* HAVE_LIBZ */ + return (cups_write(fp, fp->printf_buffer, bytes)); + } + else + { + memcpy(fp->ptr, fp->printf_buffer, bytes); + fp->ptr += bytes; + return (bytes); + } +} + + +/* + * 'cupsFilePutChar()' - Write a character. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 0 on success, -1 on error */ +cupsFilePutChar(cups_file_t *fp, /* I - CUPS file */ + int c) /* I - Character to write */ +{ + /* + * Range check input... + */ + + if (!fp || (fp->mode != 'w' && fp->mode != 's')) + return (-1); + + if (fp->mode == 's') + { + /* + * Send character immediately over socket... + */ + + char ch; /* Output character */ + + + ch = c; + + if (send(fp->fd, &ch, 1, 0) < 1) + return (-1); + } + else + { + /* + * Buffer it up... + */ + + if (fp->ptr >= fp->end) + if (cupsFileFlush(fp)) + return (-1); + + *(fp->ptr) ++ = c; + } + + fp->pos ++; + + DEBUG_printf(("4cupsFilePutChar: pos=" CUPS_LLFMT, CUPS_LLCAST fp->pos)); + + return (0); +} + + +/* + * 'cupsFilePutConf()' - Write a configuration line. + * + * This function handles any comment escaping of the value. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +ssize_t /* O - Number of bytes written or -1 on error */ +cupsFilePutConf(cups_file_t *fp, /* I - CUPS file */ + const char *directive, /* I - Directive */ + const char *value) /* I - Value */ +{ + ssize_t bytes, /* Number of bytes written */ + temp; /* Temporary byte count */ + const char *ptr; /* Pointer into value */ + + + if (!fp || !directive || !*directive) + return (-1); + + if ((bytes = cupsFilePuts(fp, directive)) < 0) + return (-1); + + if (cupsFilePutChar(fp, ' ') < 0) + return (-1); + bytes ++; + + if (value && *value) + { + if ((ptr = strchr(value, '#')) != NULL) + { + /* + * Need to quote the first # in the info string... + */ + + if ((temp = cupsFileWrite(fp, value, ptr - value)) < 0) + return (-1); + bytes += temp; + + if (cupsFilePutChar(fp, '\\') < 0) + return (-1); + bytes ++; + + if ((temp = cupsFilePuts(fp, ptr)) < 0) + return (-1); + bytes += temp; + } + else if ((temp = cupsFilePuts(fp, value)) < 0) + return (-1); + else + bytes += temp; + } + + if (cupsFilePutChar(fp, '\n') < 0) + return (-1); + else + return (bytes + 1); +} + + +/* + * 'cupsFilePuts()' - Write a string. + * + * Like the @code fputs@ function, no newline is appended to the string. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - Number of bytes written or -1 on error */ +cupsFilePuts(cups_file_t *fp, /* I - CUPS file */ + const char *s) /* I - String to write */ +{ + ssize_t bytes; /* Bytes to write */ + + + /* + * Range check input... + */ + + if (!fp || !s || (fp->mode != 'w' && fp->mode != 's')) + return (-1); + + /* + * Write the string... + */ + + bytes = (int)strlen(s); + + if (fp->mode == 's') + { + if (cups_write(fp, s, bytes) < 0) + return (-1); + + fp->pos += bytes; + + DEBUG_printf(("4cupsFilePuts: pos=" CUPS_LLFMT, CUPS_LLCAST fp->pos)); + + return (bytes); + } + + if ((fp->ptr + bytes) > fp->end) + if (cupsFileFlush(fp)) + return (-1); + + fp->pos += bytes; + + DEBUG_printf(("4cupsFilePuts: pos=" CUPS_LLFMT, CUPS_LLCAST fp->pos)); + + if (bytes > sizeof(fp->buf)) + { +#ifdef HAVE_LIBZ + if (fp->compressed) + return (cups_compress(fp, s, bytes)); + else +#endif /* HAVE_LIBZ */ + return (cups_write(fp, s, bytes)); + } + else + { + memcpy(fp->ptr, s, bytes); + fp->ptr += bytes; + return (bytes); + } +} + + +/* + * 'cupsFileRead()' - Read from a file. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ssize_t /* O - Number of bytes read or -1 on error */ +cupsFileRead(cups_file_t *fp, /* I - CUPS file */ + char *buf, /* O - Buffer */ + size_t bytes) /* I - Number of bytes to read */ +{ + size_t total; /* Total bytes read */ + ssize_t count; /* Bytes read */ + + + DEBUG_printf(("2cupsFileRead(fp=%p, buf=%p, bytes=" CUPS_LLFMT ")", fp, buf, + CUPS_LLCAST bytes)); + + /* + * Range check input... + */ + + if (!fp || !buf || (fp->mode != 'r' && fp->mode != 's')) + return (-1); + + if (bytes == 0) + return (0); + + /* + * Loop until all bytes are read... + */ + + total = 0; + while (bytes > 0) + { + if (fp->ptr >= fp->end) + if (cups_fill(fp) <= 0) + { + DEBUG_printf(("4cupsFileRead: cups_fill() returned -1, total=" + CUPS_LLFMT, CUPS_LLCAST total)); + + if (total > 0) + return ((ssize_t)total); + else + return (-1); + } + + count = (ssize_t)(fp->end - fp->ptr); + if (count > (ssize_t)bytes) + count = (ssize_t)bytes; + + memcpy(buf, fp->ptr, count); + fp->ptr += count; + fp->pos += count; + + DEBUG_printf(("4cupsFileRead: pos=" CUPS_LLFMT, CUPS_LLCAST fp->pos)); + + /* + * Update the counts for the last read... + */ + + bytes -= count; + total += count; + buf += count; + } + + /* + * Return the total number of bytes read... + */ + + DEBUG_printf(("3cupsFileRead: total=" CUPS_LLFMT, CUPS_LLCAST total)); + + return ((ssize_t)total); +} + + +/* + * 'cupsFileRewind()' - Set the current file position to the beginning of the + * file. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +off_t /* O - New file position or -1 on error */ +cupsFileRewind(cups_file_t *fp) /* I - CUPS file */ +{ + /* + * Range check input... + */ + + DEBUG_printf(("cupsFileRewind(fp=%p)", fp)); + DEBUG_printf(("2cupsFileRewind: pos=" CUPS_LLFMT, CUPS_LLCAST fp->pos)); + + if (!fp || fp->mode != 'r') + return (-1); + + /* + * Handle special cases... + */ + + if (fp->bufpos == 0) + { + /* + * No seeking necessary... + */ + + fp->pos = 0; + + if (fp->ptr) + { + fp->ptr = fp->buf; + fp->eof = 0; + } + + DEBUG_printf(("2cupsFileRewind: pos=" CUPS_LLFMT, CUPS_LLCAST fp->pos)); + + return (0); + } + + /* + * Otherwise, seek in the file and cleanup any compression buffers... + */ + +#ifdef HAVE_LIBZ + if (fp->compressed) + { + inflateEnd(&fp->stream); + fp->compressed = 0; + } +#endif /* HAVE_LIBZ */ + + if (lseek(fp->fd, 0, SEEK_SET)) + { + DEBUG_printf(("1cupsFileRewind: lseek failed: %s", strerror(errno))); + return (-1); + } + + fp->bufpos = 0; + fp->pos = 0; + fp->ptr = NULL; + fp->end = NULL; + fp->eof = 0; + + DEBUG_printf(("2cupsFileRewind: pos=" CUPS_LLFMT, CUPS_LLCAST fp->pos)); + + return (0); +} + + +/* + * 'cupsFileSeek()' - Seek in a file. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +off_t /* O - New file position or -1 on error */ +cupsFileSeek(cups_file_t *fp, /* I - CUPS file */ + off_t pos) /* I - Position in file */ +{ + ssize_t bytes; /* Number bytes in buffer */ + + + DEBUG_printf(("cupsFileSeek(fp=%p, pos=" CUPS_LLFMT ")", fp, + CUPS_LLCAST pos)); + DEBUG_printf(("2cupsFileSeek: fp->pos=" CUPS_LLFMT, CUPS_LLCAST fp->pos)); + DEBUG_printf(("2cupsFileSeek: fp->ptr=%p, fp->end=%p", fp->ptr, fp->end)); + + /* + * Range check input... + */ + + if (!fp || pos < 0 || fp->mode != 'r') + return (-1); + + /* + * Handle special cases... + */ + + if (pos == 0) + return (cupsFileRewind(fp)); + + if (fp->ptr) + { + bytes = (ssize_t)(fp->end - fp->buf); + + DEBUG_printf(("2cupsFileSeek: bytes=" CUPS_LLFMT, CUPS_LLCAST bytes)); + + if (pos >= fp->bufpos && pos < (fp->bufpos + bytes)) + { + /* + * No seeking necessary... + */ + + fp->pos = pos; + fp->ptr = fp->buf + pos - fp->bufpos; + fp->eof = 0; + + return (pos); + } + } + +#ifdef HAVE_LIBZ + if (!fp->compressed && !fp->ptr) + { + /* + * Preload a buffer to determine whether the file is compressed... + */ + + if (cups_fill(fp) < 0) + return (-1); + } +#endif /* HAVE_LIBZ */ + + /* + * Seek forwards or backwards... + */ + + fp->eof = 0; + + if (pos < fp->bufpos) + { + /* + * Need to seek backwards... + */ + + DEBUG_puts("2cupsFileSeek: SEEK BACKWARDS"); + +#ifdef HAVE_LIBZ + if (fp->compressed) + { + inflateEnd(&fp->stream); + + lseek(fp->fd, 0, SEEK_SET); + fp->bufpos = 0; + fp->pos = 0; + fp->ptr = NULL; + fp->end = NULL; + + while ((bytes = cups_fill(fp)) > 0) + if (pos >= fp->bufpos && pos < (fp->bufpos + bytes)) + break; + + if (bytes <= 0) + return (-1); + + fp->ptr = fp->buf + pos - fp->bufpos; + fp->pos = pos; + } + else +#endif /* HAVE_LIBZ */ + { + fp->bufpos = lseek(fp->fd, pos, SEEK_SET); + fp->pos = fp->bufpos; + fp->ptr = NULL; + fp->end = NULL; + + DEBUG_printf(("2cupsFileSeek: lseek() returned " CUPS_LLFMT, + CUPS_LLCAST fp->pos)); + } + } + else + { + /* + * Need to seek forwards... + */ + + DEBUG_puts("2cupsFileSeek: SEEK FORWARDS"); + +#ifdef HAVE_LIBZ + if (fp->compressed) + { + while ((bytes = cups_fill(fp)) > 0) + { + if (pos >= fp->bufpos && pos < (fp->bufpos + bytes)) + break; + } + + if (bytes <= 0) + return (-1); + + fp->ptr = fp->buf + pos - fp->bufpos; + fp->pos = pos; + } + else +#endif /* HAVE_LIBZ */ + { + fp->bufpos = lseek(fp->fd, pos, SEEK_SET); + fp->pos = fp->bufpos; + fp->ptr = NULL; + fp->end = NULL; + + DEBUG_printf(("2cupsFileSeek: lseek() returned " CUPS_LLFMT, + CUPS_LLCAST fp->pos)); + } + } + + DEBUG_printf(("2cupsFileSeek: pos=" CUPS_LLFMT, CUPS_LLCAST fp->pos)); + + return (fp->pos); +} + + +/* + * 'cupsFileStderr()' - Return a CUPS file associated with stderr. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +cups_file_t * /* O - CUPS file */ +cupsFileStderr(void) +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals... */ + + + /* + * Open file descriptor 2 as needed... + */ + + if (!cg->stdio_files[2]) + { + /* + * Flush any pending output on the stdio file... + */ + + fflush(stderr); + + /* + * Open file descriptor 2... + */ + + if ((cg->stdio_files[2] = cupsFileOpenFd(2, "w")) != NULL) + cg->stdio_files[2]->is_stdio = 1; + } + + return (cg->stdio_files[2]); +} + + +/* + * 'cupsFileStdin()' - Return a CUPS file associated with stdin. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +cups_file_t * /* O - CUPS file */ +cupsFileStdin(void) +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals... */ + + + /* + * Open file descriptor 0 as needed... + */ + + if (!cg->stdio_files[0]) + { + /* + * Open file descriptor 0... + */ + + if ((cg->stdio_files[0] = cupsFileOpenFd(0, "r")) != NULL) + cg->stdio_files[0]->is_stdio = 1; + } + + return (cg->stdio_files[0]); +} + + +/* + * 'cupsFileStdout()' - Return a CUPS file associated with stdout. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +cups_file_t * /* O - CUPS file */ +cupsFileStdout(void) +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals... */ + + + /* + * Open file descriptor 1 as needed... + */ + + if (!cg->stdio_files[1]) + { + /* + * Flush any pending output on the stdio file... + */ + + fflush(stdout); + + /* + * Open file descriptor 1... + */ + + if ((cg->stdio_files[1] = cupsFileOpenFd(1, "w")) != NULL) + cg->stdio_files[1]->is_stdio = 1; + } + + return (cg->stdio_files[1]); +} + + +/* + * 'cupsFileTell()' - Return the current file position. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +off_t /* O - File position */ +cupsFileTell(cups_file_t *fp) /* I - CUPS file */ +{ + DEBUG_printf(("2cupsFileTell(fp=%p)", fp)); + DEBUG_printf(("3cupsFileTell: pos=" CUPS_LLFMT, + CUPS_LLCAST (fp ? fp->pos : -1))); + + return (fp ? fp->pos : 0); +} + + +/* + * 'cupsFileUnlock()' - Unlock access to a file. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 0 on success, -1 on error */ +cupsFileUnlock(cups_file_t *fp) /* I - CUPS file */ +{ + /* + * Range check... + */ + + DEBUG_printf(("cupsFileUnlock(fp=%p)", fp)); + + if (!fp || fp->mode == 's') + return (-1); + + /* + * Unlock... + */ + +#ifdef WIN32 + return (_locking(fp->fd, _LK_UNLCK, 0)); +#else + return (lockf(fp->fd, F_ULOCK, 0)); +#endif /* WIN32 */ +} + + +/* + * 'cupsFileWrite()' - Write to a file. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ssize_t /* O - Number of bytes written or -1 on error */ +cupsFileWrite(cups_file_t *fp, /* I - CUPS file */ + const char *buf, /* I - Buffer */ + size_t bytes) /* I - Number of bytes to write */ +{ + /* + * Range check input... + */ + + DEBUG_printf(("2cupsFileWrite(fp=%p, buf=%p, bytes=" CUPS_LLFMT ")", + fp, buf, CUPS_LLCAST bytes)); + + if (!fp || !buf || (fp->mode != 'w' && fp->mode != 's')) + return (-1); + + if (bytes == 0) + return (0); + + /* + * Write the buffer... + */ + + if (fp->mode == 's') + { + if (cups_write(fp, buf, bytes) < 0) + return (-1); + + fp->pos += (off_t)bytes; + + DEBUG_printf(("4cupsFileWrite: pos=" CUPS_LLFMT, CUPS_LLCAST fp->pos)); + + return ((ssize_t)bytes); + } + + if ((fp->ptr + bytes) > fp->end) + if (cupsFileFlush(fp)) + return (-1); + + fp->pos += (off_t)bytes; + + DEBUG_printf(("4cupsFileWrite: pos=" CUPS_LLFMT, CUPS_LLCAST fp->pos)); + + if (bytes > sizeof(fp->buf)) + { +#ifdef HAVE_LIBZ + if (fp->compressed) + return (cups_compress(fp, buf, bytes)); + else +#endif /* HAVE_LIBZ */ + return (cups_write(fp, buf, bytes)); + } + else + { + memcpy(fp->ptr, buf, bytes); + fp->ptr += bytes; + return ((ssize_t)bytes); + } +} + + +#ifdef HAVE_LIBZ +/* + * 'cups_compress()' - Compress a buffer of data. + */ + +static ssize_t /* O - Number of bytes written or -1 */ +cups_compress(cups_file_t *fp, /* I - CUPS file */ + const char *buf, /* I - Buffer */ + size_t bytes) /* I - Number bytes */ +{ + DEBUG_printf(("7cups_compress(fp=%p, buf=%p, bytes=" CUPS_LLFMT ")", fp, buf, + CUPS_LLCAST bytes)); + + /* + * Update the CRC... + */ + + fp->crc = crc32(fp->crc, (const Bytef *)buf, bytes); + + /* + * Deflate the bytes... + */ + + fp->stream.next_in = (Bytef *)buf; + fp->stream.avail_in = bytes; + + while (fp->stream.avail_in > 0) + { + /* + * Flush the current buffer... + */ + + DEBUG_printf(("9cups_compress: avail_in=%d, avail_out=%d", + fp->stream.avail_in, fp->stream.avail_out)); + + if (fp->stream.avail_out < (int)(sizeof(fp->cbuf) / 8)) + { + if (cups_write(fp, (char *)fp->cbuf, fp->stream.next_out - fp->cbuf) < 0) + return (-1); + + fp->stream.next_out = fp->cbuf; + fp->stream.avail_out = sizeof(fp->cbuf); + } + + deflate(&(fp->stream), Z_NO_FLUSH); + } + + return (bytes); +} +#endif /* HAVE_LIBZ */ + + +/* + * 'cups_fill()' - Fill the input buffer. + */ + +static ssize_t /* O - Number of bytes or -1 */ +cups_fill(cups_file_t *fp) /* I - CUPS file */ +{ + ssize_t bytes; /* Number of bytes read */ +#ifdef HAVE_LIBZ + int status; /* Decompression status */ + const unsigned char *ptr, /* Pointer into buffer */ + *end; /* End of buffer */ +#endif /* HAVE_LIBZ */ + + + DEBUG_printf(("7cups_fill(fp=%p)", fp)); + DEBUG_printf(("9cups_fill: fp->ptr=%p, fp->end=%p, fp->buf=%p, " + "fp->bufpos=" CUPS_LLFMT ", fp->eof=%d", + fp->ptr, fp->end, fp->buf, CUPS_LLCAST fp->bufpos, fp->eof)); + + if (fp->ptr && fp->end) + fp->bufpos += fp->end - fp->buf; + +#ifdef HAVE_LIBZ + DEBUG_printf(("9cups_fill: fp->compressed=%d", fp->compressed)); + + while (!fp->ptr || fp->compressed) + { + /* + * Check to see if we have read any data yet; if not, see if we have a + * compressed file... + */ + + if (!fp->ptr) + { + /* + * Reset the file position in case we are seeking... + */ + + fp->compressed = 0; + + /* + * Read the first bytes in the file to determine if we have a gzip'd + * file... + */ + + if ((bytes = cups_read(fp, (char *)fp->buf, sizeof(fp->buf))) < 0) + { + /* + * Can't read from file! + */ + + DEBUG_printf(("9cups_fill: cups_read() returned " CUPS_LLFMT, + CUPS_LLCAST bytes)); + + return (-1); + } + + if (bytes < 10 || fp->buf[0] != 0x1f || + (fp->buf[1] & 255) != 0x8b || + fp->buf[2] != 8 || (fp->buf[3] & 0xe0) != 0) + { + /* + * Not a gzip'd file! + */ + + fp->ptr = fp->buf; + fp->end = fp->buf + bytes; + + DEBUG_printf(("9cups_fill: Returning " CUPS_LLFMT, + CUPS_LLCAST bytes)); + + return (bytes); + } + + /* + * Parse header junk: extra data, original name, and comment... + */ + + ptr = (unsigned char *)fp->buf + 10; + end = (unsigned char *)fp->buf + bytes; + + if (fp->buf[3] & 0x04) + { + /* + * Skip extra data... + */ + + if ((ptr + 2) > end) + { + /* + * Can't read from file! + */ + + return (-1); + } + + bytes = ((unsigned char)ptr[1] << 8) | (unsigned char)ptr[0]; + ptr += 2 + bytes; + + if (ptr > end) + { + /* + * Can't read from file! + */ + + return (-1); + } + } + + if (fp->buf[3] & 0x08) + { + /* + * Skip original name data... + */ + + while (ptr < end && *ptr) + ptr ++; + + if (ptr < end) + ptr ++; + else + { + /* + * Can't read from file! + */ + + return (-1); + } + } + + if (fp->buf[3] & 0x10) + { + /* + * Skip comment data... + */ + + while (ptr < end && *ptr) + ptr ++; + + if (ptr < end) + ptr ++; + else + { + /* + * Can't read from file! + */ + + return (-1); + } + } + + if (fp->buf[3] & 0x02) + { + /* + * Skip header CRC data... + */ + + ptr += 2; + + if (ptr > end) + { + /* + * Can't read from file! + */ + + return (-1); + } + } + + /* + * Copy the flate-compressed data to the compression buffer... + */ + + if ((bytes = end - ptr) > 0) + memcpy(fp->cbuf, ptr, bytes); + + /* + * Setup the decompressor data... + */ + + fp->stream.zalloc = (alloc_func)0; + fp->stream.zfree = (free_func)0; + fp->stream.opaque = (voidpf)0; + fp->stream.next_in = (Bytef *)fp->cbuf; + fp->stream.next_out = NULL; + fp->stream.avail_in = bytes; + fp->stream.avail_out = 0; + fp->crc = crc32(0L, Z_NULL, 0); + + if (inflateInit2(&(fp->stream), -15) != Z_OK) + return (-1); + + fp->compressed = 1; + } + + if (fp->compressed) + { + /* + * If we have reached end-of-file, return immediately... + */ + + if (fp->eof) + return (-1); + + /* + * Fill the decompression buffer as needed... + */ + + if (fp->stream.avail_in == 0) + { + if ((bytes = cups_read(fp, (char *)fp->cbuf, sizeof(fp->cbuf))) <= 0) + return (-1); + + fp->stream.next_in = fp->cbuf; + fp->stream.avail_in = bytes; + } + + /* + * Decompress data from the buffer... + */ + + fp->stream.next_out = (Bytef *)fp->buf; + fp->stream.avail_out = sizeof(fp->buf); + + status = inflate(&(fp->stream), Z_NO_FLUSH); + + if (fp->stream.next_out > (Bytef *)fp->buf) + fp->crc = crc32(fp->crc, (Bytef *)fp->buf, + fp->stream.next_out - (Bytef *)fp->buf); + + if (status == Z_STREAM_END) + { + /* + * Read the CRC and length... + */ + + unsigned char trailer[8]; /* Trailer bytes */ + uLong tcrc; /* Trailer CRC */ + + + if (read(fp->fd, trailer, sizeof(trailer)) < sizeof(trailer)) + { + /* + * Can't get it, so mark end-of-file... + */ + + fp->eof = 1; + } + else + { + tcrc = (((((trailer[3] << 8) | trailer[2]) << 8) | trailer[1]) << 8) | + trailer[0]; + + if (tcrc != fp->crc) + { + /* + * Bad CRC, mark end-of-file... + */ + + DEBUG_printf(("9cups_fill: tcrc=%08x, fp->crc=%08x", + (unsigned int)tcrc, (unsigned int)fp->crc)); + + fp->eof = 1; + + return (-1); + } + + /* + * Otherwise, reset the compressed flag so that we re-read the + * file header... + */ + + fp->compressed = 0; + } + } + + bytes = sizeof(fp->buf) - fp->stream.avail_out; + + /* + * Return the decompressed data... + */ + + fp->ptr = fp->buf; + fp->end = fp->buf + bytes; + + if (bytes) + return (bytes); + } + } +#endif /* HAVE_LIBZ */ + + /* + * Read a buffer's full of data... + */ + + if ((bytes = cups_read(fp, fp->buf, sizeof(fp->buf))) <= 0) + { + /* + * Can't read from file! + */ + + fp->eof = 1; + fp->ptr = fp->buf; + fp->end = fp->buf; + + return (-1); + } + + /* + * Return the bytes we read... + */ + + fp->eof = 0; + fp->ptr = fp->buf; + fp->end = fp->buf + bytes; + + return (bytes); +} + + +/* + * 'cups_open()' - Safely open a file for writing. + * + * We don't allow appending to directories or files that are hard-linked or + * symlinked. + */ + +static int /* O - File descriptor or -1 otherwise */ +cups_open(const char *filename, /* I - Filename */ + int mode) /* I - Open mode */ +{ + int fd; /* File descriptor */ + struct stat fileinfo; /* File information */ +#ifndef WIN32 + struct stat linkinfo; /* Link information */ +#endif /* !WIN32 */ + + + /* + * Open the file... + */ + + if ((fd = open(filename, mode, 0666)) < 0) + return (-1); + + /* + * Then verify that the file descriptor doesn't point to a directory or hard- + * linked file. + */ + + if (fstat(fd, &fileinfo)) + { + close(fd); + return (-1); + } + + if (fileinfo.st_nlink != 1) + { + close(fd); + errno = EPERM; + return (-1); + } + +#ifdef WIN32 + if (fileinfo.st_mode & _S_IFDIR) +#else + if (S_ISDIR(fileinfo.st_mode)) +#endif /* WIN32 */ + { + close(fd); + errno = EISDIR; + return (-1); + } + +#ifndef WIN32 + /* + * Then use lstat to determine whether the filename is a symlink... + */ + + if (lstat(filename, &linkinfo)) + { + close(fd); + return (-1); + } + + if (S_ISLNK(linkinfo.st_mode) || + fileinfo.st_dev != linkinfo.st_dev || + fileinfo.st_ino != linkinfo.st_ino || +#ifdef HAVE_ST_GEN + fileinfo.st_gen != linkinfo.st_gen || +#endif /* HAVE_ST_GEN */ + fileinfo.st_nlink != linkinfo.st_nlink || + fileinfo.st_mode != linkinfo.st_mode) + { + /* + * Yes, don't allow! + */ + + close(fd); + errno = EPERM; + return (-1); + } +#endif /* !WIN32 */ + + return (fd); +} + + +/* + * 'cups_read()' - Read from a file descriptor. + */ + +static ssize_t /* O - Number of bytes read or -1 */ +cups_read(cups_file_t *fp, /* I - CUPS file */ + char *buf, /* I - Buffer */ + size_t bytes) /* I - Number bytes */ +{ + ssize_t total; /* Total bytes read */ + + + DEBUG_printf(("7cups_read(fp=%p, buf=%p, bytes=" CUPS_LLFMT ")", fp, buf, + CUPS_LLCAST bytes)); + + /* + * Loop until we read at least 0 bytes... + */ + + for (;;) + { +#ifdef WIN32 + if (fp->mode == 's') + total = (ssize_t)recv(fp->fd, buf, (unsigned)bytes, 0); + else + total = (ssize_t)read(fp->fd, buf, (unsigned)bytes); +#else + if (fp->mode == 's') + total = recv(fp->fd, buf, bytes, 0); + else + total = read(fp->fd, buf, bytes); +#endif /* WIN32 */ + + DEBUG_printf(("9cups_read: total=" CUPS_LLFMT, CUPS_LLCAST total)); + + if (total >= 0) + break; + + /* + * Reads can be interrupted by signals and unavailable resources... + */ + + if (errno == EAGAIN || errno == EINTR) + continue; + else + return (-1); + } + + /* + * Return the total number of bytes read... + */ + + return (total); +} + + +/* + * 'cups_write()' - Write to a file descriptor. + */ + +static ssize_t /* O - Number of bytes written or -1 */ +cups_write(cups_file_t *fp, /* I - CUPS file */ + const char *buf, /* I - Buffer */ + size_t bytes) /* I - Number bytes */ +{ + size_t total; /* Total bytes written */ + ssize_t count; /* Count this time */ + + + DEBUG_printf(("7cups_write(fp=%p, buf=%p, bytes=" CUPS_LLFMT ")", fp, buf, + CUPS_LLCAST bytes)); + + /* + * Loop until all bytes are written... + */ + + total = 0; + while (bytes > 0) + { +#ifdef WIN32 + if (fp->mode == 's') + count = (ssize_t)send(fp->fd, buf, (unsigned)bytes, 0); + else + count = (ssize_t)write(fp->fd, buf, (unsigned)bytes); +#else + if (fp->mode == 's') + count = send(fp->fd, buf, bytes, 0); + else + count = write(fp->fd, buf, bytes); +#endif /* WIN32 */ + + DEBUG_printf(("9cups_write: count=" CUPS_LLFMT, CUPS_LLCAST count)); + + if (count < 0) + { + /* + * Writes can be interrupted by signals and unavailable resources... + */ + + if (errno == EAGAIN || errno == EINTR) + continue; + else + return (-1); + } + + /* + * Update the counts for the last write call... + */ + + bytes -= count; + total += count; + buf += count; + } + + /* + * Return the total number of bytes written... + */ + + return ((ssize_t)total); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/file.h b/cups/file.h new file mode 100644 index 0000000000..a3defede14 --- /dev/null +++ b/cups/file.h @@ -0,0 +1,116 @@ +/* + * "$Id$" + * + * Public file definitions for CUPS. + * + * Since stdio files max out at 256 files on many systems, we have to + * write similar functions without this limit. At the same time, using + * our own file functions allows us to provide transparent support of + * gzip'd print files, PPD files, etc. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#ifndef _CUPS_FILE_H_ +# define _CUPS_FILE_H_ + + +/* + * Include necessary headers... + */ + +# include "versioning.h" +# include +# include +# if defined(WIN32) && !defined(__CUPS_SSIZE_T_DEFINED) +# define __CUPS_SSIZE_T_DEFINED +/* Windows does not support the ssize_t type, so map it to off_t... */ +typedef off_t ssize_t; /* @private@ */ +# endif /* WIN32 && !__CUPS_SSIZE_T_DEFINED */ + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * CUPS file definitions... + */ + +# define CUPS_FILE_NONE 0 /* No compression */ +# define CUPS_FILE_GZIP 1 /* GZIP compression */ + + +/* + * Types and structures... + */ + +typedef struct _cups_file_s cups_file_t;/**** CUPS file type ****/ + + +/* + * Prototypes... + */ + +extern int cupsFileClose(cups_file_t *fp) _CUPS_API_1_2; +extern int cupsFileCompression(cups_file_t *fp) _CUPS_API_1_2; +extern int cupsFileEOF(cups_file_t *fp) _CUPS_API_1_2; +extern const char *cupsFileFind(const char *filename, const char *path, + int executable, char *buffer, + int bufsize) _CUPS_API_1_2; +extern int cupsFileFlush(cups_file_t *fp) _CUPS_API_1_2; +extern int cupsFileGetChar(cups_file_t *fp) _CUPS_API_1_2; +extern char *cupsFileGetConf(cups_file_t *fp, char *buf, + size_t buflen, char **value, + int *linenum) _CUPS_API_1_2; +extern size_t cupsFileGetLine(cups_file_t *fp, char *buf, + size_t buflen) _CUPS_API_1_2; +extern char *cupsFileGets(cups_file_t *fp, char *buf, size_t buflen) + _CUPS_API_1_2; +extern int cupsFileLock(cups_file_t *fp, int block) _CUPS_API_1_2; +extern int cupsFileNumber(cups_file_t *fp) _CUPS_API_1_2; +extern cups_file_t *cupsFileOpen(const char *filename, const char *mode) + _CUPS_API_1_2; +extern cups_file_t *cupsFileOpenFd(int fd, const char *mode) _CUPS_API_1_2; +extern int cupsFilePeekChar(cups_file_t *fp) _CUPS_API_1_2; +extern int cupsFilePrintf(cups_file_t *fp, const char *format, ...) + __attribute__((__format__ (__printf__, 2, 3))) + _CUPS_API_1_2; +extern int cupsFilePutChar(cups_file_t *fp, int c) _CUPS_API_1_2; +extern ssize_t cupsFilePutConf(cups_file_t *fp, const char *directive, + const char *value) _CUPS_API_1_4; +extern int cupsFilePuts(cups_file_t *fp, const char *s) + _CUPS_API_1_2; +extern ssize_t cupsFileRead(cups_file_t *fp, char *buf, size_t bytes) + _CUPS_API_1_2; +extern off_t cupsFileRewind(cups_file_t *fp) _CUPS_API_1_2; +extern off_t cupsFileSeek(cups_file_t *fp, off_t pos) _CUPS_API_1_2; +extern cups_file_t *cupsFileStderr(void) _CUPS_API_1_2; +extern cups_file_t *cupsFileStdin(void) _CUPS_API_1_2; +extern cups_file_t *cupsFileStdout(void) _CUPS_API_1_2; +extern off_t cupsFileTell(cups_file_t *fp) _CUPS_API_1_2; +extern int cupsFileUnlock(cups_file_t *fp) _CUPS_API_1_2; +extern ssize_t cupsFileWrite(cups_file_t *fp, const char *buf, + size_t bytes) _CUPS_API_1_2; + + +# ifdef __cplusplus +} +# endif /* __cplusplus */ +#endif /* !_CUPS_FILE_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/getdevices.c b/cups/getdevices.c new file mode 100644 index 0000000000..a0744ac894 --- /dev/null +++ b/cups/getdevices.c @@ -0,0 +1,283 @@ +/* + * "$Id$" + * + * cupsGetDevices implementation for CUPS. + * + * Copyright 2008-2011 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsGetDevices() - Get available printer devices. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" + + +/* + * 'cupsGetDevices()' - Get available printer devices. + * + * This function sends a CUPS-Get-Devices request and streams the discovered + * devices to the specified callback function. The "timeout" parameter controls + * how long the request lasts, while the "include_schemes" and "exclude_schemes" + * parameters provide comma-delimited lists of backends to include or omit from + * the request respectively. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +ipp_status_t /* O - Request status - @code IPP_OK@ on success. */ +cupsGetDevices( + http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + int timeout, /* I - Timeout in seconds or @code CUPS_TIMEOUT_DEFAULT@ */ + const char *include_schemes, /* I - Comma-separated URI schemes to include or @code CUPS_INCLUDE_ALL@ */ + const char *exclude_schemes, /* I - Comma-separated URI schemes to exclude or @code CUPS_EXCLUDE_NONE@ */ + cups_device_cb_t callback, /* I - Callback function */ + void *user_data) /* I - User data pointer */ +{ + ipp_t *request, /* CUPS-Get-Devices request */ + *response; /* CUPS-Get-Devices response */ + ipp_attribute_t *attr; /* Current attribute */ + const char *device_class, /* device-class value */ + *device_id, /* device-id value */ + *device_info, /* device-info value */ + *device_location, /* device-location value */ + *device_make_and_model, /* device-make-and-model value */ + *device_uri; /* device-uri value */ + int blocking; /* Current blocking-IO mode */ + cups_option_t option; /* in/exclude-schemes option */ + http_status_t status; /* HTTP status of request */ + ipp_state_t state; /* IPP response state */ + + + /* + * Range check input... + */ + + DEBUG_printf(("cupsGetDevices(http=%p, timeout=%d, include_schemes=\"%s\", " + "exclude_schemes=\"%s\", callback=%p, user_data=%p)", http, + timeout, include_schemes, exclude_schemes, callback, + user_data)); + + if (!callback) + return (IPP_INTERNAL_ERROR); + + if (!http) + http = _cupsConnect(); + + if (!http) + return (IPP_SERVICE_UNAVAILABLE); + + /* + * Create a CUPS-Get-Devices request... + */ + + request = ippNewRequest(CUPS_GET_DEVICES); + + if (timeout > 0) + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_INTEGER, "timeout", + timeout); + + if (include_schemes) + { + option.name = "include-schemes"; + option.value = (char *)include_schemes; + + cupsEncodeOptions2(request, 1, &option, IPP_TAG_OPERATION); + } + + if (exclude_schemes) + { + option.name = "exclude-schemes"; + option.value = (char *)exclude_schemes; + + cupsEncodeOptions2(request, 1, &option, IPP_TAG_OPERATION); + } + + /* + * Send the request and do any necessary authentication... + */ + + do + { + DEBUG_puts("2cupsGetDevices: Sending request..."); + status = cupsSendRequest(http, request, "/", ippLength(request)); + + DEBUG_puts("2cupsGetDevices: Waiting for response status..."); + while (status == HTTP_CONTINUE) + status = httpUpdate(http); + + if (status != HTTP_OK) + { + httpFlush(http); + + if (status == HTTP_UNAUTHORIZED) + { + /* + * See if we can do authentication... + */ + + DEBUG_puts("2cupsGetDevices: Need authorization..."); + + if (!cupsDoAuthentication(http, "POST", "/")) + httpReconnect(http); + else + { + status = HTTP_AUTHORIZATION_CANCELED; + break; + } + } + +#ifdef HAVE_SSL + else if (status == HTTP_UPGRADE_REQUIRED) + { + /* + * Force a reconnect with encryption... + */ + + DEBUG_puts("2cupsGetDevices: Need encryption..."); + + if (!httpReconnect(http)) + httpEncryption(http, HTTP_ENCRYPT_REQUIRED); + } +#endif /* HAVE_SSL */ + } + } + while (status == HTTP_UNAUTHORIZED || status == HTTP_UPGRADE_REQUIRED); + + DEBUG_printf(("2cupsGetDevices: status=%d", status)); + + ippDelete(request); + + if (status != HTTP_OK) + { + _cupsSetHTTPError(status); + return (cupsLastError()); + } + + /* + * Read the response in non-blocking mode... + */ + + blocking = httpGetBlocking(http); + httpBlocking(http, 0); + + response = ippNew(); + device_class = NULL; + device_id = NULL; + device_info = NULL; + device_location = ""; + device_make_and_model = NULL; + device_uri = NULL; + attr = NULL; + + DEBUG_puts("2cupsGetDevices: Reading response..."); + + do + { + if ((state = ippRead(http, response)) == IPP_ERROR) + break; + + DEBUG_printf(("2cupsGetDevices: state=%d, response->last=%p", state, + response->last)); + + if (!response->attrs) + continue; + + while (attr != response->last) + { + if (!attr) + attr = response->attrs; + else + attr = attr->next; + + DEBUG_printf(("2cupsGetDevices: attr->name=\"%s\", attr->value_tag=%d", + attr->name, attr->value_tag)); + + if (!attr->name) + { + if (device_class && device_id && device_info && device_make_and_model && + device_uri) + (*callback)(device_class, device_id, device_info, + device_make_and_model, device_uri, device_location, + user_data); + + device_class = NULL; + device_id = NULL; + device_info = NULL; + device_location = ""; + device_make_and_model = NULL; + device_uri = NULL; + } + else if (!strcmp(attr->name, "device-class") && + attr->value_tag == IPP_TAG_KEYWORD) + device_class = attr->values[0].string.text; + else if (!strcmp(attr->name, "device-id") && + attr->value_tag == IPP_TAG_TEXT) + device_id = attr->values[0].string.text; + else if (!strcmp(attr->name, "device-info") && + attr->value_tag == IPP_TAG_TEXT) + device_info = attr->values[0].string.text; + else if (!strcmp(attr->name, "device-location") && + attr->value_tag == IPP_TAG_TEXT) + device_location = attr->values[0].string.text; + else if (!strcmp(attr->name, "device-make-and-model") && + attr->value_tag == IPP_TAG_TEXT) + device_make_and_model = attr->values[0].string.text; + else if (!strcmp(attr->name, "device-uri") && + attr->value_tag == IPP_TAG_URI) + device_uri = attr->values[0].string.text; + } + } + while (state != IPP_DATA); + + DEBUG_printf(("2cupsGetDevices: state=%d, response->last=%p", state, + response->last)); + + if (device_class && device_id && device_info && device_make_and_model && + device_uri) + (*callback)(device_class, device_id, device_info, + device_make_and_model, device_uri, device_location, user_data); + + /* + * Set the IPP status and return... + */ + + httpBlocking(http, blocking); + httpFlush(http); + + if (status == HTTP_ERROR) + _cupsSetError(IPP_INTERNAL_ERROR, strerror(http->error), 0); + else + { + attr = ippFindAttribute(response, "status-message", IPP_TAG_TEXT); + + DEBUG_printf(("cupsGetDevices: status-code=%s, status-message=\"%s\"", + ippErrorString(response->request.status.status_code), + attr ? attr->values[0].string.text : "")); + + _cupsSetError(response->request.status.status_code, + attr ? attr->values[0].string.text : + ippErrorString(response->request.status.status_code), 0); + } + + ippDelete(response); + + return (cupsLastError()); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/getifaddrs.c b/cups/getifaddrs.c new file mode 100644 index 0000000000..8111e224db --- /dev/null +++ b/cups/getifaddrs.c @@ -0,0 +1,266 @@ +/* + * "$Id$" + * + * Network interface functions for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * _cups_getifaddrs() - Get a list of network interfaces on the system. + * _cups_freeifaddrs() - Free an interface list... + */ + +/* + * Include necessary headers. + */ + +#include "http-private.h" + + +#ifndef HAVE_GETIFADDRS +/* + * '_cups_getifaddrs()' - Get a list of network interfaces on the system. + */ + +int /* O - 0 on success, -1 on error */ +_cups_getifaddrs(struct ifaddrs **addrs)/* O - List of interfaces */ +{ + int sock; /* Socket */ + char buffer[65536], /* Buffer for address info */ + *bufptr, /* Pointer into buffer */ + *bufend; /* End of buffer */ + struct ifconf conf; /* Interface configurations */ + struct sockaddr addr; /* Address data */ + struct ifreq *ifp; /* Interface data */ + int ifpsize; /* Size of interface data */ + struct ifaddrs *temp; /* Pointer to current interface */ + struct ifreq request; /* Interface request */ + + + /* + * Start with an empty list... + */ + + if (addrs == NULL) + return (-1); + + *addrs = NULL; + + /* + * Create a UDP socket to get the interface data... + */ + + memset (&addr, 0, sizeof(addr)); + if ((sock = socket(AF_INET, SOCK_DGRAM, 0)) < 0) + return (-1); + + /* + * Try to get the list of interfaces... + */ + + conf.ifc_len = sizeof(buffer); + conf.ifc_buf = buffer; + + if (ioctl(sock, SIOCGIFCONF, &conf) < 0) + { + /* + * Couldn't get the list of interfaces... + */ + + close(sock); + return (-1); + } + + /* + * OK, got the list of interfaces, now lets step through the + * buffer to pull them out... + */ + +# ifdef HAVE_STRUCT_SOCKADDR_SA_LEN +# define sockaddr_len(a) ((a)->sa_len) +# else +# define sockaddr_len(a) (sizeof(struct sockaddr)) +# endif /* HAVE_STRUCT_SOCKADDR_SA_LEN */ + + for (bufptr = buffer, bufend = buffer + conf.ifc_len; + bufptr < bufend; + bufptr += ifpsize) + { + /* + * Get the current interface information... + */ + + ifp = (struct ifreq *)bufptr; + ifpsize = sizeof(ifp->ifr_name) + sockaddr_len(&(ifp->ifr_addr)); + + if (ifpsize < sizeof(struct ifreq)) + ifpsize = sizeof(struct ifreq); + + memset(&request, 0, sizeof(request)); + memcpy(request.ifr_name, ifp->ifr_name, sizeof(ifp->ifr_name)); + + /* + * Check the status of the interface... + */ + + if (ioctl(sock, SIOCGIFFLAGS, &request) < 0) + continue; + + /* + * Allocate memory for a single interface record... + */ + + if ((temp = calloc(1, sizeof(struct ifaddrs))) == NULL) + { + /* + * Unable to allocate memory... + */ + + close(sock); + return (-1); + } + + /* + * Add this record to the front of the list and copy the name, flags, + * and network address... + */ + + temp->ifa_next = *addrs; + *addrs = temp; + temp->ifa_name = strdup(ifp->ifr_name); + temp->ifa_flags = request.ifr_flags; + if ((temp->ifa_addr = calloc(1, sockaddr_len(&(ifp->ifr_addr)))) != NULL) + memcpy(temp->ifa_addr, &(ifp->ifr_addr), sockaddr_len(&(ifp->ifr_addr))); + + /* + * Try to get the netmask for the interface... + */ + + if (!ioctl(sock, SIOCGIFNETMASK, &request)) + { + /* + * Got it, make a copy... + */ + + if ((temp->ifa_netmask = calloc(1, sizeof(request.ifr_netmask))) != NULL) + memcpy(temp->ifa_netmask, &(request.ifr_netmask), + sizeof(request.ifr_netmask)); + } + + /* + * Then get the broadcast or point-to-point (destination) address, + * if applicable... + */ + + if (temp->ifa_flags & IFF_BROADCAST) + { + /* + * Have a broadcast address, so get it! + */ + + if (!ioctl(sock, SIOCGIFBRDADDR, &request)) + { + /* + * Got it, make a copy... + */ + + if ((temp->ifa_broadaddr = + calloc(1, sizeof(request.ifr_broadaddr))) != NULL) + memcpy(temp->ifa_broadaddr, &(request.ifr_broadaddr), + sizeof(request.ifr_broadaddr)); + } + } + else if (temp->ifa_flags & IFF_POINTOPOINT) + { + /* + * Point-to-point interface; grab the remote address... + */ + + if (!ioctl(sock, SIOCGIFDSTADDR, &request)) + { + temp->ifa_dstaddr = malloc(sizeof(request.ifr_dstaddr)); + memcpy(temp->ifa_dstaddr, &(request.ifr_dstaddr), + sizeof(request.ifr_dstaddr)); + } + } + } + + /* + * OK, we're done with the socket, close it and return 0... + */ + + close(sock); + + return (0); +} + + +/* + * '_cups_freeifaddrs()' - Free an interface list... + */ + +void +_cups_freeifaddrs(struct ifaddrs *addrs)/* I - Interface list to free */ +{ + struct ifaddrs *next; /* Next interface in list */ + + + while (addrs != NULL) + { + /* + * Make a copy of the next interface pointer... + */ + + next = addrs->ifa_next; + + /* + * Free data values as needed... + */ + + if (addrs->ifa_name) + { + free(addrs->ifa_name); + addrs->ifa_name = NULL; + } + + if (addrs->ifa_addr) + { + free(addrs->ifa_addr); + addrs->ifa_addr = NULL; + } + + if (addrs->ifa_netmask) + { + free(addrs->ifa_netmask); + addrs->ifa_netmask = NULL; + } + + if (addrs->ifa_dstaddr) + { + free(addrs->ifa_dstaddr); + addrs->ifa_dstaddr = NULL; + } + + /* + * Free this node and continue to the next... + */ + + free(addrs); + + addrs = next; + } +} +#endif /* !HAVE_GETIFADDRS */ + + +/* + * End of "$Id$". + */ diff --git a/cups/getputfile.c b/cups/getputfile.c new file mode 100644 index 0000000000..80f6e7be45 --- /dev/null +++ b/cups/getputfile.c @@ -0,0 +1,502 @@ +/* + * "$Id$" + * + * Get/put file functions for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsGetFd() - Get a file from the server. + * cupsGetFile() - Get a file from the server. + * cupsPutFd() - Put a file on the server. + * cupsPutFile() - Put a file on the server. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include +#include +#if defined(WIN32) || defined(__EMX__) +# include +#else +# include +#endif /* WIN32 || __EMX__ */ + + +/* + * 'cupsGetFd()' - Get a file from the server. + * + * This function returns @code HTTP_OK@ when the file is successfully retrieved. + * + * @since CUPS 1.1.20/Mac OS X 10.4@ + */ + +http_status_t /* O - HTTP status */ +cupsGetFd(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + const char *resource, /* I - Resource name */ + int fd) /* I - File descriptor */ +{ + int bytes; /* Number of bytes read */ + char buffer[8192]; /* Buffer for file */ + http_status_t status; /* HTTP status from server */ + char if_modified_since[HTTP_MAX_VALUE]; + /* If-Modified-Since header */ + + + /* + * Range check input... + */ + + DEBUG_printf(("cupsGetFd(http=%p, resource=\"%s\", fd=%d)", http, + resource, fd)); + + if (!resource || fd < 0) + { + if (http) + http->error = EINVAL; + + return (HTTP_ERROR); + } + + if (!http) + if ((http = _cupsConnect()) == NULL) + return (HTTP_SERVICE_UNAVAILABLE); + + /* + * Then send GET requests to the HTTP server... + */ + + strlcpy(if_modified_since, httpGetField(http, HTTP_FIELD_IF_MODIFIED_SINCE), + sizeof(if_modified_since)); + + do + { + httpClearFields(http); + httpSetField(http, HTTP_FIELD_AUTHORIZATION, http->authstring); + httpSetField(http, HTTP_FIELD_IF_MODIFIED_SINCE, if_modified_since); + + if (httpGet(http, resource)) + { + if (httpReconnect(http)) + { + status = HTTP_ERROR; + break; + } + else + { + status = HTTP_UNAUTHORIZED; + continue; + } + } + + while ((status = httpUpdate(http)) == HTTP_CONTINUE); + + if (status == HTTP_UNAUTHORIZED) + { + /* + * Flush any error message... + */ + + httpFlush(http); + + /* + * See if we can do authentication... + */ + + if (cupsDoAuthentication(http, "GET", resource)) + { + status = HTTP_AUTHORIZATION_CANCELED; + break; + } + + if (httpReconnect(http)) + { + status = HTTP_ERROR; + break; + } + + continue; + } +#ifdef HAVE_SSL + else if (status == HTTP_UPGRADE_REQUIRED) + { + /* Flush any error message... */ + httpFlush(http); + + /* Reconnect... */ + if (httpReconnect(http)) + { + status = HTTP_ERROR; + break; + } + + /* Upgrade with encryption... */ + httpEncryption(http, HTTP_ENCRYPT_REQUIRED); + + /* Try again, this time with encryption enabled... */ + continue; + } +#endif /* HAVE_SSL */ + } + while (status == HTTP_UNAUTHORIZED || status == HTTP_UPGRADE_REQUIRED); + + /* + * See if we actually got the file or an error... + */ + + if (status == HTTP_OK) + { + /* + * Yes, copy the file... + */ + + while ((bytes = httpRead2(http, buffer, sizeof(buffer))) > 0) + write(fd, buffer, bytes); + } + else + { + _cupsSetHTTPError(status); + httpFlush(http); + } + + /* + * Return the request status... + */ + + DEBUG_printf(("1cupsGetFd: Returning %d...", status)); + + return (status); +} + + +/* + * 'cupsGetFile()' - Get a file from the server. + * + * This function returns @code HTTP_OK@ when the file is successfully retrieved. + * + * @since CUPS 1.1.20/Mac OS X 10.4@ + */ + +http_status_t /* O - HTTP status */ +cupsGetFile(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + const char *resource, /* I - Resource name */ + const char *filename) /* I - Filename */ +{ + int fd; /* File descriptor */ + http_status_t status; /* Status */ + + + /* + * Range check input... + */ + + if (!http || !resource || !filename) + { + if (http) + http->error = EINVAL; + + return (HTTP_ERROR); + } + + /* + * Create the file... + */ + + if ((fd = open(filename, O_WRONLY | O_EXCL | O_TRUNC)) < 0) + { + /* + * Couldn't open the file! + */ + + http->error = errno; + + return (HTTP_ERROR); + } + + /* + * Get the file... + */ + + status = cupsGetFd(http, resource, fd); + + /* + * If the file couldn't be gotten, then remove the file... + */ + + close(fd); + + if (status != HTTP_OK) + unlink(filename); + + /* + * Return the HTTP status code... + */ + + return (status); +} + + +/* + * 'cupsPutFd()' - Put a file on the server. + * + * This function returns @code HTTP_CREATED@ when the file is stored + * successfully. + * + * @since CUPS 1.1.20/Mac OS X 10.4@ + */ + +http_status_t /* O - HTTP status */ +cupsPutFd(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + const char *resource, /* I - Resource name */ + int fd) /* I - File descriptor */ +{ + int bytes, /* Number of bytes read */ + retries; /* Number of retries */ + char buffer[8192]; /* Buffer for file */ + http_status_t status; /* HTTP status from server */ + + + /* + * Range check input... + */ + + DEBUG_printf(("cupsPutFd(http=%p, resource=\"%s\", fd=%d)", http, + resource, fd)); + + if (!resource || fd < 0) + { + if (http) + http->error = EINVAL; + + return (HTTP_ERROR); + } + + if (!http) + if ((http = _cupsConnect()) == NULL) + return (HTTP_SERVICE_UNAVAILABLE); + + /* + * Then send PUT requests to the HTTP server... + */ + + retries = 0; + + do + { + DEBUG_printf(("2cupsPutFd: starting attempt, authstring=\"%s\"...", + http->authstring)); + + httpClearFields(http); + httpSetField(http, HTTP_FIELD_AUTHORIZATION, http->authstring); + httpSetField(http, HTTP_FIELD_TRANSFER_ENCODING, "chunked"); + httpSetExpect(http, HTTP_CONTINUE); + + if (httpPut(http, resource)) + { + if (httpReconnect(http)) + { + status = HTTP_ERROR; + break; + } + else + { + status = HTTP_UNAUTHORIZED; + continue; + } + } + + /* + * Wait up to 1 second for a 100-continue response... + */ + + if (httpWait(http, 1000)) + status = httpUpdate(http); + else + status = HTTP_CONTINUE; + + if (status == HTTP_CONTINUE) + { + /* + * Copy the file... + */ + + lseek(fd, 0, SEEK_SET); + + while ((bytes = read(fd, buffer, sizeof(buffer))) > 0) + if (httpCheck(http)) + { + if ((status = httpUpdate(http)) != HTTP_CONTINUE) + break; + } + else + httpWrite2(http, buffer, bytes); + } + + if (status == HTTP_CONTINUE) + { + httpWrite2(http, buffer, 0); + + while ((status = httpUpdate(http)) == HTTP_CONTINUE); + } + + if (status == HTTP_ERROR && !retries) + { + DEBUG_printf(("2cupsPutFd: retry on status %d", status)); + + retries ++; + + /* Flush any error message... */ + httpFlush(http); + + /* Reconnect... */ + if (httpReconnect(http)) + { + status = HTTP_ERROR; + break; + } + + /* Try again... */ + continue; + } + + DEBUG_printf(("2cupsPutFd: status=%d", status)); + + if (status == HTTP_UNAUTHORIZED) + { + /* + * Flush any error message... + */ + + httpFlush(http); + + /* + * See if we can do authentication... + */ + + if (cupsDoAuthentication(http, "PUT", resource)) + { + status = HTTP_AUTHORIZATION_CANCELED; + break; + } + + if (httpReconnect(http)) + { + status = HTTP_ERROR; + break; + } + + continue; + } +#ifdef HAVE_SSL + else if (status == HTTP_UPGRADE_REQUIRED) + { + /* Flush any error message... */ + httpFlush(http); + + /* Reconnect... */ + if (httpReconnect(http)) + { + status = HTTP_ERROR; + break; + } + + /* Upgrade with encryption... */ + httpEncryption(http, HTTP_ENCRYPT_REQUIRED); + + /* Try again, this time with encryption enabled... */ + continue; + } +#endif /* HAVE_SSL */ + } + while (status == HTTP_UNAUTHORIZED || status == HTTP_UPGRADE_REQUIRED || + (status == HTTP_ERROR && retries < 2)); + + /* + * See if we actually put the file or an error... + */ + + if (status != HTTP_CREATED) + { + _cupsSetHTTPError(status); + httpFlush(http); + } + + DEBUG_printf(("1cupsPutFd: Returning %d...", status)); + + return (status); +} + + +/* + * 'cupsPutFile()' - Put a file on the server. + * + * This function returns @code HTTP_CREATED@ when the file is stored + * successfully. + * + * @since CUPS 1.1.20/Mac OS X 10.4@ + */ + +http_status_t /* O - HTTP status */ +cupsPutFile(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + const char *resource, /* I - Resource name */ + const char *filename) /* I - Filename */ +{ + int fd; /* File descriptor */ + http_status_t status; /* Status */ + + + /* + * Range check input... + */ + + if (!http || !resource || !filename) + { + if (http) + http->error = EINVAL; + + return (HTTP_ERROR); + } + + /* + * Open the local file... + */ + + if ((fd = open(filename, O_RDONLY)) < 0) + { + /* + * Couldn't open the file! + */ + + http->error = errno; + + return (HTTP_ERROR); + } + + /* + * Put the file... + */ + + status = cupsPutFd(http, resource, fd); + + close(fd); + + return (status); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/globals.c b/cups/globals.c new file mode 100644 index 0000000000..6085e05700 --- /dev/null +++ b/cups/globals.c @@ -0,0 +1,354 @@ +/* + * "$Id$" + * + * Global variable access routines for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * _cupsGlobalLock() - Lock the global mutex. + * _cupsGlobals() - Return a pointer to thread local storage + * _cupsGlobalUnlock() - Unlock the global mutex. + * DllMain() - Main entry for library. + * cups_globals_alloc() - Allocate and initialize global data. + * cups_globals_free() - Free global data. + * cups_globals_init() - Initialize environment variables. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" + + +/* + * Local globals... + */ + + +static _cups_threadkey_t cups_globals_key = _CUPS_THREADKEY_INITIALIZER; + /* Thread local storage key */ +#ifdef HAVE_PTHREAD_H +static pthread_once_t cups_globals_key_once = PTHREAD_ONCE_INIT; + /* One-time initialization object */ +#endif /* HAVE_PTHREAD_H */ +static _cups_mutex_t cups_global_mutex = _CUPS_MUTEX_INITIALIZER; + /* Global critical section */ + + +/* + * Local functions... + */ + +static _cups_globals_t *cups_globals_alloc(void); +static void cups_globals_free(_cups_globals_t *g); +#ifdef HAVE_PTHREAD_H +static void cups_globals_init(void); +#endif /* HAVE_PTHREAD_H */ + + +/* + * '_cupsGlobalLock()' - Lock the global mutex. + */ + +void +_cupsGlobalLock(void) +{ +#ifdef HAVE_PTHREAD_H + pthread_mutex_lock(&cups_global_mutex); +#elif defined(WIN32) + EnterCriticalSection(&cups_global_mutex.m_criticalSection); +#endif /* HAVE_PTHREAD_H */ +} + + +/* + * '_cupsGlobals()' - Return a pointer to thread local storage + */ + +_cups_globals_t * /* O - Pointer to global data */ +_cupsGlobals(void) +{ + _cups_globals_t *cg; /* Pointer to global data */ + + +#ifdef HAVE_PTHREAD_H + /* + * Initialize the global data exactly once... + */ + + pthread_once(&cups_globals_key_once, cups_globals_init); +#endif /* HAVE_PTHREAD_H */ + + /* + * See if we have allocated the data yet... + */ + + if ((cg = (_cups_globals_t *)_cupsThreadGetData(cups_globals_key)) == NULL) + { + /* + * No, allocate memory as set the pointer for the key... + */ + + if ((cg = cups_globals_alloc()) != NULL) + _cupsThreadSetData(cups_globals_key, cg); + } + + /* + * Return the pointer to the data... + */ + + return (cg); +} + + +/* + * '_cupsGlobalUnlock()' - Unlock the global mutex. + */ + +void +_cupsGlobalUnlock(void) +{ +#ifdef HAVE_PTHREAD_H + pthread_mutex_unlock(&cups_global_mutex); +#elif defined(WIN32) + LeaveCriticalSection(&cups_global_mutex.m_criticalSection); +#endif /* HAVE_PTHREAD_H */ +} + + +#ifdef WIN32 +/* + * 'DllMain()' - Main entry for library. + */ + +BOOL WINAPI /* O - Success/failure */ +DllMain(HINSTANCE hinst, /* I - DLL module handle */ + DWORD reason, /* I - Reason */ + LPVOID reserved) /* I - Unused */ +{ + _cups_globals_t *cg; /* Global data */ + + + (void)hinst; + (void)reserved; + + switch (reason) + { + case DLL_PROCESS_ATTACH : /* Called on library initialization */ + InitializeCriticalSection(&cups_global_mutex.m_criticalSection); + + if ((cups_globals_key = TlsAlloc()) == TLS_OUT_OF_INDEXES) + return (FALSE); + break; + + case DLL_THREAD_DETACH : /* Called when a thread terminates */ + if ((cg = (_cups_globals_t *)TlsGetValue(cups_globals_key)) != NULL) + cups_globals_free(cg); + break; + + case DLL_PROCESS_DETACH : /* Called when library is unloaded */ + if ((cg = (_cups_globals_t *)TlsGetValue(cups_globals_key)) != NULL) + cups_globals_free(cg); + + TlsFree(cups_globals_key); + DeleteCriticalSection(&cups_global_mutex.m_criticalSection); + break; + + default: + break; + } + + return (TRUE); +} +#endif /* WIN32 */ + + +/* + * 'cups_globals_alloc()' - Allocate and initialize global data. + */ + +static _cups_globals_t * /* O - Pointer to global data */ +cups_globals_alloc(void) +{ + _cups_globals_t *cg = malloc(sizeof(_cups_globals_t)); + /* Pointer to global data */ +#ifdef WIN32 + HKEY key; /* Registry key */ + DWORD size; /* Size of string */ + static char installdir[1024], /* Install directory */ + confdir[1024], /* Server root directory */ + localedir[1024]; /* Locale directory */ +#endif /* WIN32 */ + + + if (!cg) + return (NULL); + + /* + * Clear the global storage and set the default encryption and password + * callback values... + */ + + memset(cg, 0, sizeof(_cups_globals_t)); + cg->encryption = (http_encryption_t)-1; + cg->password_cb = (cups_password_cb2_t)_cupsGetPassword; + cg->any_root = 1; + cg->expired_certs = 1; + cg->expired_root = 1; + + /* + * Then set directories as appropriate... + */ + +#ifdef WIN32 + /* + * Open the registry... + */ + + strcpy(installdir, "C:/Program Files/cups.org"); + + if (!RegOpenKeyEx(HKEY_LOCAL_MACHINE, "SOFTWARE\\cups.org", 0, KEY_READ, + &key)) + { + /* + * Grab the installation directory... + */ + + size = sizeof(installdir); + RegQueryValueEx(key, "installdir", NULL, NULL, installdir, &size); + RegCloseKey(key); + } + + snprintf(confdir, sizeof(confdir), "%s/conf", installdir); + snprintf(localedir, sizeof(localedir), "%s/locale", installdir); + + if ((cg->cups_datadir = getenv("CUPS_DATADIR")) == NULL) + cg->cups_datadir = installdir; + + if ((cg->cups_serverbin = getenv("CUPS_SERVERBIN")) == NULL) + cg->cups_serverbin = installdir; + + if ((cg->cups_serverroot = getenv("CUPS_SERVERROOT")) == NULL) + cg->cups_serverroot = confdir; + + if ((cg->cups_statedir = getenv("CUPS_STATEDIR")) == NULL) + cg->cups_statedir = confdir; + + if ((cg->localedir = getenv("LOCALEDIR")) == NULL) + cg->localedir = localedir; + +#else +# ifdef HAVE_GETEUID + if ((geteuid() != getuid() && getuid()) || getegid() != getgid()) +# else + if (!getuid()) +# endif /* HAVE_GETEUID */ + { + /* + * When running setuid/setgid, don't allow environment variables to override + * the directories... + */ + + cg->cups_datadir = CUPS_DATADIR; + cg->cups_serverbin = CUPS_SERVERBIN; + cg->cups_serverroot = CUPS_SERVERROOT; + cg->cups_statedir = CUPS_STATEDIR; + cg->localedir = CUPS_LOCALEDIR; + } + else + { + /* + * Allow directories to be overridden by environment variables. + */ + + if ((cg->cups_datadir = getenv("CUPS_DATADIR")) == NULL) + cg->cups_datadir = CUPS_DATADIR; + + if ((cg->cups_serverbin = getenv("CUPS_SERVERBIN")) == NULL) + cg->cups_serverbin = CUPS_SERVERBIN; + + if ((cg->cups_serverroot = getenv("CUPS_SERVERROOT")) == NULL) + cg->cups_serverroot = CUPS_SERVERROOT; + + if ((cg->cups_statedir = getenv("CUPS_STATEDIR")) == NULL) + cg->cups_statedir = CUPS_STATEDIR; + + if ((cg->localedir = getenv("LOCALEDIR")) == NULL) + cg->localedir = CUPS_LOCALEDIR; + } +#endif /* WIN32 */ + + return (cg); +} + + +/* + * 'cups_globals_free()' - Free global data. + */ + +static void +cups_globals_free(_cups_globals_t *cg) /* I - Pointer to global data */ +{ + _cups_buffer_t *buffer, /* Current read/write buffer */ + *next; /* Next buffer */ + + + if (cg->last_status_message) + _cupsStrFree(cg->last_status_message); + + for (buffer = cg->cups_buffers; buffer; buffer = next) + { + next = buffer->next; + free(buffer); + } + + cupsArrayDelete(cg->leg_size_lut); + cupsArrayDelete(cg->ppd_size_lut); + cupsArrayDelete(cg->pwg_size_lut); + + httpClose(cg->http); + + _httpFreeCredentials(cg->tls_credentials); + + cupsFileClose(cg->stdio_files[0]); + cupsFileClose(cg->stdio_files[1]); + cupsFileClose(cg->stdio_files[2]); + + cupsFreeOptions(cg->cupsd_num_settings, cg->cupsd_settings); + + free(cg); +} + + +#ifdef HAVE_PTHREAD_H +/* + * 'cups_globals_init()' - Initialize environment variables. + */ + +static void +cups_globals_init(void) +{ + /* + * Register the global data for this thread... + */ + + pthread_key_create(&cups_globals_key, (void (*)(void *))cups_globals_free); +} +#endif /* HAVE_PTHREAD_H */ + + +/* + * End of "$Id$". + */ diff --git a/cups/http-addr.c b/cups/http-addr.c new file mode 100644 index 0000000000..41919bcef2 --- /dev/null +++ b/cups/http-addr.c @@ -0,0 +1,705 @@ +/* + * "$Id$" + * + * HTTP address routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * httpAddrAny() - Check for the "any" address. + * httpAddrEqual() - Compare two addresses. + * httpAddrLocalhost() - Check for the local loopback address. + * httpAddrLookup() - Lookup the hostname associated with the address. + * _httpAddrPort() - Get the port number associated with an address. + * _httpAddrSetPort() - Set the port number associated with an address. + * httpAddrString() - Convert an IP address to a dotted string. + * httpGetHostByName() - Lookup a hostname or IP address, and return + * address records for the specified name. + * httpGetHostname() - Get the FQDN for the local system. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#ifdef HAVE_RESOLV_H +# include +#endif /* HAVE_RESOLV_H */ +#ifdef HAVE_COREFOUNDATION +# include +#endif /* HAVE_COREFOUNDATION */ +#ifdef HAVE_SYSTEMCONFIGURATION +# include +#endif /* HAVE_SYSTEMCONFIGURATION */ + + +/* + * 'httpAddrAny()' - Check for the "any" address. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 1 if "any", 0 otherwise */ +httpAddrAny(const http_addr_t *addr) /* I - Address to check */ +{ + if (!addr) + return (0); + +#ifdef AF_INET6 + if (addr->addr.sa_family == AF_INET6 && + IN6_IS_ADDR_UNSPECIFIED(&(addr->ipv6.sin6_addr))) + return (1); +#endif /* AF_INET6 */ + + if (addr->addr.sa_family == AF_INET && + ntohl(addr->ipv4.sin_addr.s_addr) == 0x00000000) + return (1); + + return (0); +} + + +/* + * 'httpAddrEqual()' - Compare two addresses. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 1 if equal, 0 if not */ +httpAddrEqual(const http_addr_t *addr1, /* I - First address */ + const http_addr_t *addr2) /* I - Second address */ +{ + if (!addr1 && !addr2) + return (1); + + if (!addr1 || !addr2) + return (0); + + if (addr1->addr.sa_family != addr2->addr.sa_family) + return (0); + +#ifdef AF_LOCAL + if (addr1->addr.sa_family == AF_LOCAL) + return (!strcmp(addr1->un.sun_path, addr2->un.sun_path)); +#endif /* AF_LOCAL */ + +#ifdef AF_INET6 + if (addr1->addr.sa_family == AF_INET6) + return (!memcmp(&(addr1->ipv6.sin6_addr), &(addr2->ipv6.sin6_addr), 16)); +#endif /* AF_INET6 */ + + return (addr1->ipv4.sin_addr.s_addr == addr2->ipv4.sin_addr.s_addr); +} + + +/* + * 'httpAddrLength()' - Return the length of the address in bytes. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - Length in bytes */ +httpAddrLength(const http_addr_t *addr) /* I - Address */ +{ + if (!addr) + return (0); + +#ifdef AF_INET6 + if (addr->addr.sa_family == AF_INET6) + return (sizeof(addr->ipv6)); + else +#endif /* AF_INET6 */ +#ifdef AF_LOCAL + if (addr->addr.sa_family == AF_LOCAL) + return (offsetof(struct sockaddr_un, sun_path) + + strlen(addr->un.sun_path) + 1); + else +#endif /* AF_LOCAL */ + if (addr->addr.sa_family == AF_INET) + return (sizeof(addr->ipv4)); + else + return (0); + +} + + +/* + * 'httpAddrLocalhost()' - Check for the local loopback address. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 1 if local host, 0 otherwise */ +httpAddrLocalhost( + const http_addr_t *addr) /* I - Address to check */ +{ + if (!addr) + return (1); + +#ifdef AF_INET6 + if (addr->addr.sa_family == AF_INET6 && + IN6_IS_ADDR_LOOPBACK(&(addr->ipv6.sin6_addr))) + return (1); +#endif /* AF_INET6 */ + +#ifdef AF_LOCAL + if (addr->addr.sa_family == AF_LOCAL) + return (1); +#endif /* AF_LOCAL */ + + if (addr->addr.sa_family == AF_INET && + (ntohl(addr->ipv4.sin_addr.s_addr) & 0xff000000) == 0x7f000000) + return (1); + + return (0); +} + + +#ifdef __sgi +# define ADDR_CAST (struct sockaddr *) +#else +# define ADDR_CAST (char *) +#endif /* __sgi */ + + +/* + * 'httpAddrLookup()' - Lookup the hostname associated with the address. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +char * /* O - Host name */ +httpAddrLookup( + const http_addr_t *addr, /* I - Address to lookup */ + char *name, /* I - Host name buffer */ + int namelen) /* I - Size of name buffer */ +{ + _cups_globals_t *cg = _cupsGlobals(); + /* Global data */ + + + DEBUG_printf(("httpAddrLookup(addr=%p, name=%p, namelen=%d)", addr, name, + namelen)); + + /* + * Range check input... + */ + + if (!addr || !name || namelen <= 2) + { + if (name && namelen >= 1) + *name = '\0'; + + return (NULL); + } + +#ifdef AF_LOCAL + if (addr->addr.sa_family == AF_LOCAL) + { + strlcpy(name, addr->un.sun_path, namelen); + return (name); + } +#endif /* AF_LOCAL */ + + /* + * Optimize lookups for localhost/loopback addresses... + */ + + if (httpAddrLocalhost(addr)) + { + strlcpy(name, "localhost", namelen); + return (name); + } + +#ifdef HAVE_RES_INIT + /* + * STR #2920: Initialize resolver after failure in cups-polld + * + * If the previous lookup failed, re-initialize the resolver to prevent + * temporary network errors from persisting. This *should* be handled by + * the resolver libraries, but apparently the glibc folks do not agree. + * + * We set a flag at the end of this function if we encounter an error that + * requires reinitialization of the resolver functions. We then call + * res_init() if the flag is set on the next call here or in httpAddrLookup(). + */ + + if (cg->need_res_init) + { + res_init(); + + cg->need_res_init = 0; + } +#endif /* HAVE_RES_INIT */ + +#ifdef HAVE_GETNAMEINFO + { + /* + * STR #2486: httpAddrLookup() fails when getnameinfo() returns EAI_AGAIN + * + * FWIW, I think this is really a bug in the implementation of + * getnameinfo(), but falling back on httpAddrString() is easy to + * do... + */ + + int error = getnameinfo(&addr->addr, httpAddrLength(addr), name, namelen, + NULL, 0, 0); + + if (error) + { + if (error == EAI_FAIL) + cg->need_res_init = 1; + + return (httpAddrString(addr, name, namelen)); + } + } +#else + { + struct hostent *host; /* Host from name service */ + + +# ifdef AF_INET6 + if (addr->addr.sa_family == AF_INET6) + host = gethostbyaddr(ADDR_CAST &(addr->ipv6.sin6_addr), + sizeof(struct in_addr), AF_INET6); + else +# endif /* AF_INET6 */ + host = gethostbyaddr(ADDR_CAST &(addr->ipv4.sin_addr), + sizeof(struct in_addr), AF_INET); + + if (host == NULL) + { + /* + * No hostname, so return the raw address... + */ + + if (h_errno == NO_RECOVERY) + cg->need_res_init = 1; + + return (httpAddrString(addr, name, namelen)); + } + + strlcpy(name, host->h_name, namelen); + } +#endif /* HAVE_GETNAMEINFO */ + + DEBUG_printf(("1httpAddrLookup: returning \"%s\"...", name)); + + return (name); +} + + +/* + * '_httpAddrPort()' - Get the port number associated with an address. + */ + +int /* O - Port number */ +_httpAddrPort(http_addr_t *addr) /* I - Address */ +{ + if (!addr) + return (ippPort()); +#ifdef AF_INET6 + else if (addr->addr.sa_family == AF_INET6) + return (ntohs(addr->ipv6.sin6_port)); +#endif /* AF_INET6 */ + else if (addr->addr.sa_family == AF_INET) + return (ntohs(addr->ipv4.sin_port)); + else + return (ippPort()); +} + + +/* + * '_httpAddrSetPort()' - Set the port number associated with an address. + */ + +void +_httpAddrSetPort(http_addr_t *addr, /* I - Address */ + int port) /* I - Port */ +{ + if (!addr || port <= 0) + return; + +#ifdef AF_INET6 + if (addr->addr.sa_family == AF_INET6) + addr->ipv6.sin6_port = htons(port); + else +#endif /* AF_INET6 */ + if (addr->addr.sa_family == AF_INET) + addr->ipv4.sin_port = htons(port); +} + + +/* + * 'httpAddrString()' - Convert an address to a numeric string. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +char * /* O - Numeric address string */ +httpAddrString(const http_addr_t *addr, /* I - Address to convert */ + char *s, /* I - String buffer */ + int slen) /* I - Length of string */ +{ + DEBUG_printf(("httpAddrString(addr=%p, s=%p, slen=%d)", addr, s, slen)); + + /* + * Range check input... + */ + + if (!addr || !s || slen <= 2) + { + if (s && slen >= 1) + *s = '\0'; + + return (NULL); + } + +#ifdef AF_LOCAL + if (addr->addr.sa_family == AF_LOCAL) + { + if (addr->un.sun_path[0] == '/') + strlcpy(s, addr->un.sun_path, slen); + else + strlcpy(s, "localhost", slen); + } + else +#endif /* AF_LOCAL */ + if (addr->addr.sa_family == AF_INET) + { + unsigned temp; /* Temporary address */ + + + temp = ntohl(addr->ipv4.sin_addr.s_addr); + + snprintf(s, slen, "%d.%d.%d.%d", (temp >> 24) & 255, + (temp >> 16) & 255, (temp >> 8) & 255, temp & 255); + } +#ifdef AF_INET6 + else if (addr->addr.sa_family == AF_INET6) + { + char *sptr, /* Pointer into string */ + temps[64]; /* Temporary string for address */ + +# ifdef HAVE_GETNAMEINFO + if (getnameinfo(&addr->addr, httpAddrLength(addr), temps, sizeof(temps), + NULL, 0, NI_NUMERICHOST)) + { + /* + * If we get an error back, then the address type is not supported + * and we should zero out the buffer... + */ + + s[0] = '\0'; + + return (NULL); + } + else if ((sptr = strchr(temps, '%')) != NULL) + { + /* + * Convert "%zone" to "+zone" to match URI form... + */ + + *sptr = '+'; + } + +# else + int i; /* Looping var */ + unsigned temp; /* Current value */ + const char *prefix; /* Prefix for address */ + + + prefix = ""; + for (sptr = temps, i = 0; i < 4 && addr->ipv6.sin6_addr.s6_addr32[i]; i ++) + { + temp = ntohl(addr->ipv6.sin6_addr.s6_addr32[i]); + + snprintf(sptr, sizeof(temps) - (sptr - temps), "%s%x", prefix, + (temp >> 16) & 0xffff); + prefix = ":"; + sptr += strlen(sptr); + + temp &= 0xffff; + + if (temp || i == 3 || addr->ipv6.sin6_addr.s6_addr32[i + 1]) + { + snprintf(sptr, sizeof(temps) - (sptr - temps), "%s%x", prefix, temp); + sptr += strlen(sptr); + } + } + + if (i < 4) + { + while (i < 4 && !addr->ipv6.sin6_addr.s6_addr32[i]) + i ++; + + if (i < 4) + { + snprintf(sptr, sizeof(temps) - (sptr - temps), "%s:", prefix); + prefix = ":"; + sptr += strlen(sptr); + + for (; i < 4; i ++) + { + temp = ntohl(addr->ipv6.sin6_addr.s6_addr32[i]); + + if ((temp & 0xffff0000) || + (i > 0 && addr->ipv6.sin6_addr.s6_addr32[i - 1])) + { + snprintf(sptr, sizeof(temps) - (sptr - temps), "%s%x", prefix, + (temp >> 16) & 0xffff); + sptr += strlen(sptr); + } + + snprintf(sptr, sizeof(temps) - (sptr - temps), "%s%x", prefix, + temp & 0xffff); + sptr += strlen(sptr); + } + } + else if (sptr == s) + { + /* + * Empty address... + */ + + strlcpy(temps, "::", sizeof(temps)); + } + else + { + /* + * Empty at end... + */ + + strlcpy(sptr, "::", sizeof(temps) - (sptr - temps)); + } + } +# endif /* HAVE_GETNAMEINFO */ + + /* + * Add "[v1." and "]" around IPv6 address to convert to URI form. + */ + + snprintf(s, slen, "[v1.%s]", temps); + } +#endif /* AF_INET6 */ + else + strlcpy(s, "UNKNOWN", slen); + + DEBUG_printf(("1httpAddrString: returning \"%s\"...", s)); + + return (s); +} + + +/* + * 'httpGetHostByName()' - Lookup a hostname or IPv4 address, and return + * address records for the specified name. + * + * @deprecated@ + */ + +struct hostent * /* O - Host entry */ +httpGetHostByName(const char *name) /* I - Hostname or IP address */ +{ + const char *nameptr; /* Pointer into name */ + unsigned ip[4]; /* IP address components */ + _cups_globals_t *cg = _cupsGlobals(); + /* Pointer to library globals */ + + + DEBUG_printf(("httpGetHostByName(name=\"%s\")", name)); + + /* + * Avoid lookup delays and configuration problems when connecting + * to the localhost address... + */ + + if (!strcmp(name, "localhost")) + name = "127.0.0.1"; + + /* + * This function is needed because some operating systems have a + * buggy implementation of gethostbyname() that does not support + * IP addresses. If the first character of the name string is a + * number, then sscanf() is used to extract the IP components. + * We then pack the components into an IPv4 address manually, + * since the inet_aton() function is deprecated. We use the + * htonl() macro to get the right byte order for the address. + * + * We also support domain sockets when supported by the underlying + * OS... + */ + +#ifdef AF_LOCAL + if (name[0] == '/') + { + /* + * A domain socket address, so make an AF_LOCAL entry and return it... + */ + + cg->hostent.h_name = (char *)name; + cg->hostent.h_aliases = NULL; + cg->hostent.h_addrtype = AF_LOCAL; + cg->hostent.h_length = strlen(name) + 1; + cg->hostent.h_addr_list = cg->ip_ptrs; + cg->ip_ptrs[0] = (char *)name; + cg->ip_ptrs[1] = NULL; + + DEBUG_puts("1httpGetHostByName: returning domain socket address..."); + + return (&cg->hostent); + } +#endif /* AF_LOCAL */ + + for (nameptr = name; isdigit(*nameptr & 255) || *nameptr == '.'; nameptr ++); + + if (!*nameptr) + { + /* + * We have an IPv4 address; break it up and provide the host entry + * to the caller. + */ + + if (sscanf(name, "%u.%u.%u.%u", ip, ip + 1, ip + 2, ip + 3) != 4) + return (NULL); /* Must have 4 numbers */ + + if (ip[0] > 255 || ip[1] > 255 || ip[2] > 255 || ip[3] > 255) + return (NULL); /* Invalid byte ranges! */ + + cg->ip_addr = htonl(((((((ip[0] << 8) | ip[1]) << 8) | ip[2]) << 8) | + ip[3])); + + /* + * Fill in the host entry and return it... + */ + + cg->hostent.h_name = (char *)name; + cg->hostent.h_aliases = NULL; + cg->hostent.h_addrtype = AF_INET; + cg->hostent.h_length = 4; + cg->hostent.h_addr_list = cg->ip_ptrs; + cg->ip_ptrs[0] = (char *)&(cg->ip_addr); + cg->ip_ptrs[1] = NULL; + + DEBUG_puts("1httpGetHostByName: returning IPv4 address..."); + + return (&cg->hostent); + } + else + { + /* + * Use the gethostbyname() function to get the IPv4 address for + * the name... + */ + + DEBUG_puts("1httpGetHostByName: returning domain lookup address(es)..."); + + return (gethostbyname(name)); + } +} + + +/* + * 'httpGetHostname()' - Get the FQDN for the connection or local system. + * + * When "http" points to a connected socket, return the hostname or + * address that was used in the call to httpConnect() or httpConnectEncrypt(). + * Otherwise, return the FQDN for the local system using both gethostname() + * and gethostbyname() to get the local hostname with domain. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +const char * /* O - FQDN for connection or system */ +httpGetHostname(http_t *http, /* I - HTTP connection or NULL */ + char *s, /* I - String buffer for name */ + int slen) /* I - Size of buffer */ +{ + if (!s || slen <= 1) + return (NULL); + + if (http) + { + if (http->hostname[0] == '/') + strlcpy(s, "localhost", slen); + else + strlcpy(s, http->hostname, slen); + } + else + { + /* + * Get the hostname... + */ + + if (gethostname(s, slen) < 0) + strlcpy(s, "localhost", slen); + + if (!strchr(s, '.')) + { +#ifdef HAVE_SCDYNAMICSTORECOPYCOMPUTERNAME + /* + * The hostname is not a FQDN, so use the local hostname from the + * SystemConfiguration framework... + */ + + SCDynamicStoreRef sc = SCDynamicStoreCreate(kCFAllocatorDefault, + CFSTR("libcups"), NULL, NULL); + /* System configuration data */ + CFStringRef local = sc ? SCDynamicStoreCopyLocalHostName(sc) : NULL; + /* Local host name */ + char localStr[1024]; /* Local host name C string */ + + if (local && CFStringGetCString(local, localStr, sizeof(localStr), + kCFStringEncodingUTF8)) + { + /* + * Append ".local." to the hostname we get... + */ + + snprintf(s, slen, "%s.local.", localStr); + } + + if (local) + CFRelease(local); + if (sc) + CFRelease(sc); + +#else + /* + * The hostname is not a FQDN, so look it up... + */ + + struct hostent *host; /* Host entry to get FQDN */ + + if ((host = gethostbyname(s)) != NULL && host->h_name) + { + /* + * Use the resolved hostname... + */ + + strlcpy(s, host->h_name, slen); + } +#endif /* HAVE_SCDYNAMICSTORECOPYCOMPUTERNAME */ + } + } + + /* + * Return the hostname with as much domain info as we have... + */ + + return (s); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/http-addrlist.c b/cups/http-addrlist.c new file mode 100644 index 0000000000..88bd885a29 --- /dev/null +++ b/cups/http-addrlist.c @@ -0,0 +1,832 @@ +/* + * "$Id$" + * + * HTTP address list routines for CUPS. + * + * Copyright 2007-2012 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * httpAddrConnect() - Connect to any of the addresses in the list. + * httpAddrConnect2() - Connect to any of the addresses in the list with a + * timeout and optional cancel. + * httpAddrFreeList() - Free an address list. + * httpAddrGetList() - Get a list of addresses for a hostname. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#ifdef HAVE_RESOLV_H +# include +#endif /* HAVE_RESOLV_H */ +#ifdef HAVE_POLL +# include +#endif /* HAVE_POLL */ + + +/* + * 'httpAddrConnect()' - Connect to any of the addresses in the list. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +http_addrlist_t * /* O - Connected address or NULL on failure */ +httpAddrConnect( + http_addrlist_t *addrlist, /* I - List of potential addresses */ + int *sock) /* O - Socket */ +{ + DEBUG_printf(("httpAddrConnect(addrlist=%p, sock=%p)", addrlist, sock)); + + return (httpAddrConnect2(addrlist, sock, 30000, NULL)); +} + + +/* + * 'httpAddrConnect2()' - Connect to any of the addresses in the list with a + * timeout and optional cancel. + * + * @since CUPS 1.6@ + */ + +http_addrlist_t * /* O - Connected address or NULL on failure */ +httpAddrConnect2( + http_addrlist_t *addrlist, /* I - List of potential addresses */ + int *sock, /* O - Socket */ + int msec, /* I - Timeout in milliseconds */ + int *cancel) /* I - Pointer to "cancel" variable */ +{ + int val; /* Socket option value */ +#ifdef O_NONBLOCK + socklen_t len; /* Length of value */ + int flags, /* Socket flags */ + remaining; /* Remaining timeout */ +# ifdef HAVE_POLL + struct pollfd pfd; /* Polled file descriptor */ +# else + fd_set input_set, /* select() input set */ + output_set; /* select() output set */ + struct timeval timeout; /* Timeout */ +# endif /* HAVE_POLL */ + int nfds; /* Result from select()/poll() */ +#endif /* O_NONBLOCK */ +#ifdef DEBUG + char temp[256]; /* Temporary address string */ +#endif /* DEBUG */ + + + DEBUG_printf(("httpAddrConnect2(addrlist=%p, sock=%p, msec=%d, cancel=%p)", + addrlist, sock, msec, cancel)); + + if (!sock) + { + errno = EINVAL; + _cupsSetError(IPP_INTERNAL_ERROR, strerror(errno), 0); + return (NULL); + } + + if (cancel && *cancel) + return (NULL); + + if (msec <= 0) + msec = INT_MAX; + + /* + * Loop through each address until we connect or run out of addresses... + */ + + while (addrlist) + { + /* + * Create the socket... + */ + + DEBUG_printf(("2httpAddrConnect2: Trying %s:%d...", + httpAddrString(&(addrlist->addr), temp, sizeof(temp)), + _httpAddrPort(&(addrlist->addr)))); + + if ((*sock = (int)socket(_httpAddrFamily(&(addrlist->addr)), SOCK_STREAM, + 0)) < 0) + { + /* + * Don't abort yet, as this could just be an issue with the local + * system not being configured with IPv4/IPv6/domain socket enabled... + */ + + addrlist = addrlist->next; + continue; + } + + /* + * Set options... + */ + + val = 1; +#ifdef WIN32 + setsockopt(*sock, SOL_SOCKET, SO_REUSEADDR, (const char *)&val, + sizeof(val)); +#else + setsockopt(*sock, SOL_SOCKET, SO_REUSEADDR, &val, sizeof(val)); +#endif /* WIN32 */ + +#ifdef SO_REUSEPORT + val = 1; + setsockopt(*sock, SOL_SOCKET, SO_REUSEPORT, &val, sizeof(val)); +#endif /* SO_REUSEPORT */ + +#ifdef SO_NOSIGPIPE + val = 1; + setsockopt(*sock, SOL_SOCKET, SO_NOSIGPIPE, &val, sizeof(val)); +#endif /* SO_NOSIGPIPE */ + + /* + * Using TCP_NODELAY improves responsiveness, especially on systems + * with a slow loopback interface... + */ + + val = 1; +#ifdef WIN32 + setsockopt(*sock, IPPROTO_TCP, TCP_NODELAY, (const char *)&val, + sizeof(val)); +#else + setsockopt(*sock, IPPROTO_TCP, TCP_NODELAY, &val, sizeof(val)); +#endif /* WIN32 */ + +#ifdef FD_CLOEXEC + /* + * Close this socket when starting another process... + */ + + fcntl(*sock, F_SETFD, FD_CLOEXEC); +#endif /* FD_CLOEXEC */ + +#ifdef O_NONBLOCK + /* + * Do an asynchronous connect by setting the socket non-blocking... + */ + + DEBUG_printf(("httpAddrConnect2: Setting non-blocking connect()")); + + flags = fcntl(*sock, F_GETFL, 0); + if (msec > 0) + fcntl(*sock, F_SETFL, flags | O_NONBLOCK); +#endif /* O_NONBLOCK */ + + /* + * Then connect... + */ + + if (!connect(*sock, &(addrlist->addr.addr), + httpAddrLength(&(addrlist->addr)))) + { + DEBUG_printf(("1httpAddrConnect2: Connected to %s:%d...", + httpAddrString(&(addrlist->addr), temp, sizeof(temp)), + _httpAddrPort(&(addrlist->addr)))); + +#ifdef O_NONBLOCK + fcntl(*sock, F_SETFL, flags); +#endif /* O_NONBLOCK */ + + return (addrlist); + } + +#ifdef O_NONBLOCK +# ifdef WIN32 + if (errno == WSAEINPROGRESS) +# else + if (errno == EINPROGRESS) +# endif /* WIN32 */ + { + DEBUG_puts("1httpAddrConnect2: Finishing async connect()"); + + for (remaining = msec; remaining > 0; remaining -= 250) + { + do + { + if (cancel && *cancel) + { + /* + * Close this socket and return... + */ + + DEBUG_puts("1httpAddrConnect2: Canceled connect()"); + +# ifdef WIN32 + closesocket(*sock); +# else + close(*sock); +# endif /* WIN32 */ + + *sock = -1; + + return (NULL); + } + +# ifdef HAVE_POLL + pfd.fd = *sock; + pfd.events = POLLIN | POLLOUT; + + nfds = poll(&pfd, 1, remaining > 250 ? 250 : remaining); + + DEBUG_printf(("1httpAddrConnect2: poll() returned %d (%d)", nfds, + errno)); + +# else + FD_ZERO(&input_set); + FD_SET(*sock, &input_set); + output_set = input_set; + + timeout.tv_sec = 0; + timeout.tv_usec = (remaining > 250 ? 250 : remaining) * 1000; + + nfds = select(*sock + 1, &input_set, &output_set, NULL, &timeout); + + DEBUG_printf(("1httpAddrConnect2: select() returned %d (%d)", nfds, + errno)); +# endif /* HAVE_POLL */ + } +# ifdef WIN32 + while (nfds < 0 && (WSAGetLastError() == WSAEINTR || + WSAGetLastError() == WSAEWOULDBLOCK)); +# else + while (nfds < 0 && (errno == EINTR || errno == EAGAIN)); +# endif /* WIN32 */ + + if (nfds > 0) + { + len = sizeof(val); + if (getsockopt(*sock, SOL_SOCKET, SO_ERROR, &val, &len) >= 0) + { + DEBUG_printf(("1httpAddrConnect2: Connected to %s:%d...", + httpAddrString(&(addrlist->addr), temp, sizeof(temp)), + _httpAddrPort(&(addrlist->addr)))); + + fcntl(*sock, F_SETFL, flags); + return (addrlist); + } + + break; + } + } + } +#endif /* O_NONBLOCK */ + + DEBUG_printf(("1httpAddrConnect2: Unable to connect to %s:%d: %s", + httpAddrString(&(addrlist->addr), temp, sizeof(temp)), + _httpAddrPort(&(addrlist->addr)), strerror(errno))); + +#ifndef WIN32 + if (errno == EINPROGRESS) + errno = ETIMEDOUT; +#endif /* !WIN32 */ + + /* + * Close this socket and move to the next address... + */ + +#ifdef WIN32 + closesocket(*sock); +#else + close(*sock); +#endif /* WIN32 */ + + *sock = -1; + addrlist = addrlist->next; + } + + if (!addrlist) +#ifdef WIN32 + _cupsSetError(IPP_SERVICE_UNAVAILABLE, "Connection failed", 0); +#else + _cupsSetError(IPP_SERVICE_UNAVAILABLE, strerror(errno), 0); +#endif /* WIN32 */ + + return (addrlist); +} + + +/* + * 'httpAddrFreeList()' - Free an address list. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void +httpAddrFreeList( + http_addrlist_t *addrlist) /* I - Address list to free */ +{ + http_addrlist_t *next; /* Next address in list */ + + + /* + * Free each address in the list... + */ + + while (addrlist) + { + next = addrlist->next; + + free(addrlist); + + addrlist = next; + } +} + + +/* + * 'httpAddrGetList()' - Get a list of addresses for a hostname. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +http_addrlist_t * /* O - List of addresses or NULL */ +httpAddrGetList(const char *hostname, /* I - Hostname, IP address, or NULL for passive listen address */ + int family, /* I - Address family or AF_UNSPEC */ + const char *service) /* I - Service name or port number */ +{ + http_addrlist_t *first, /* First address in list */ + *addr, /* Current address in list */ + *temp; /* New address */ + _cups_globals_t *cg = _cupsGlobals(); + /* Global data */ + + +#ifdef DEBUG + _cups_debug_printf("httpAddrGetList(hostname=\"%s\", family=AF_%s, " + "service=\"%s\")\n", + hostname ? hostname : "(nil)", + family == AF_UNSPEC ? "UNSPEC" : +# ifdef AF_LOCAL + family == AF_LOCAL ? "LOCAL" : +# endif /* AF_LOCAL */ +# ifdef AF_INET6 + family == AF_INET6 ? "INET6" : +# endif /* AF_INET6 */ + family == AF_INET ? "INET" : "???", service); +#endif /* DEBUG */ + +#ifdef HAVE_RES_INIT + /* + * STR #2920: Initialize resolver after failure in cups-polld + * + * If the previous lookup failed, re-initialize the resolver to prevent + * temporary network errors from persisting. This *should* be handled by + * the resolver libraries, but apparently the glibc folks do not agree. + * + * We set a flag at the end of this function if we encounter an error that + * requires reinitialization of the resolver functions. We then call + * res_init() if the flag is set on the next call here or in httpAddrLookup(). + */ + + if (cg->need_res_init) + { + res_init(); + + cg->need_res_init = 0; + } +#endif /* HAVE_RES_INIT */ + + /* + * Lookup the address the best way we can... + */ + + first = addr = NULL; + +#ifdef AF_LOCAL + if (hostname && hostname[0] == '/') + { + /* + * Domain socket address... + */ + + if ((first = (http_addrlist_t *)calloc(1, sizeof(http_addrlist_t))) != NULL) + { + first->addr.un.sun_family = AF_LOCAL; + strlcpy(first->addr.un.sun_path, hostname, sizeof(first->addr.un.sun_path)); + } + } + else +#endif /* AF_LOCAL */ + if (!hostname || _cups_strcasecmp(hostname, "localhost")) + { +#ifdef HAVE_GETADDRINFO + struct addrinfo hints, /* Address lookup hints */ + *results, /* Address lookup results */ + *current; /* Current result */ + char ipv6[64], /* IPv6 address */ + *ipv6zone; /* Pointer to zone separator */ + int ipv6len; /* Length of IPv6 address */ + int error; /* getaddrinfo() error */ + + + /* + * Lookup the address as needed... + */ + + memset(&hints, 0, sizeof(hints)); + hints.ai_family = family; + hints.ai_flags = hostname ? 0 : AI_PASSIVE; + hints.ai_socktype = SOCK_STREAM; + + if (hostname && *hostname == '[') + { + /* + * Remove brackets from numeric IPv6 address... + */ + + if (!strncmp(hostname, "[v1.", 4)) + { + /* + * Copy the newer address format which supports link-local addresses... + */ + + strlcpy(ipv6, hostname + 4, sizeof(ipv6)); + if ((ipv6len = (int)strlen(ipv6) - 1) >= 0 && ipv6[ipv6len] == ']') + { + ipv6[ipv6len] = '\0'; + hostname = ipv6; + + /* + * Convert "+zone" in address to "%zone"... + */ + + if ((ipv6zone = strrchr(ipv6, '+')) != NULL) + *ipv6zone = '%'; + } + } + else + { + /* + * Copy the regular non-link-local IPv6 address... + */ + + strlcpy(ipv6, hostname + 1, sizeof(ipv6)); + if ((ipv6len = (int)strlen(ipv6) - 1) >= 0 && ipv6[ipv6len] == ']') + { + ipv6[ipv6len] = '\0'; + hostname = ipv6; + } + } + } + + if ((error = getaddrinfo(hostname, service, &hints, &results)) == 0) + { + /* + * Copy the results to our own address list structure... + */ + + for (current = results; current; current = current->ai_next) + if (current->ai_family == AF_INET || current->ai_family == AF_INET6) + { + /* + * Copy the address over... + */ + + temp = (http_addrlist_t *)calloc(1, sizeof(http_addrlist_t)); + if (!temp) + { + httpAddrFreeList(first); + _cupsSetError(IPP_INTERNAL_ERROR, strerror(errno), 0); + return (NULL); + } + + if (current->ai_family == AF_INET6) + memcpy(&(temp->addr.ipv6), current->ai_addr, + sizeof(temp->addr.ipv6)); + else + memcpy(&(temp->addr.ipv4), current->ai_addr, + sizeof(temp->addr.ipv4)); + + /* + * Append the address to the list... + */ + + if (!first) + first = temp; + + if (addr) + addr->next = temp; + + addr = temp; + } + + /* + * Free the results from getaddrinfo()... + */ + + freeaddrinfo(results); + } + else + { + if (error == EAI_FAIL) + cg->need_res_init = 1; + + _cupsSetError(IPP_INTERNAL_ERROR, gai_strerror(error), 0); + } + +#else + if (hostname) + { + int i; /* Looping vars */ + unsigned ip[4]; /* IPv4 address components */ + const char *ptr; /* Pointer into hostname */ + struct hostent *host; /* Result of lookup */ + struct servent *port; /* Port number for service */ + int portnum; /* Port number */ + + + /* + * Lookup the service... + */ + + if (!service) + portnum = 0; + else if (isdigit(*service & 255)) + portnum = atoi(service); + else if ((port = getservbyname(service, NULL)) != NULL) + portnum = ntohs(port->s_port); + else if (!strcmp(service, "http")) + portnum = 80; + else if (!strcmp(service, "https")) + portnum = 443; + else if (!strcmp(service, "ipp") || !strcmp(service, "ipps")) + portnum = 631; + else if (!strcmp(service, "lpd")) + portnum = 515; + else if (!strcmp(service, "socket")) + portnum = 9100; + else + return (NULL); + + /* + * This code is needed because some operating systems have a + * buggy implementation of gethostbyname() that does not support + * IPv4 addresses. If the hostname string is an IPv4 address, then + * sscanf() is used to extract the IPv4 components. We then pack + * the components into an IPv4 address manually, since the + * inet_aton() function is deprecated. We use the htonl() macro + * to get the right byte order for the address. + */ + + for (ptr = hostname; isdigit(*ptr & 255) || *ptr == '.'; ptr ++); + + if (!*ptr) + { + /* + * We have an IPv4 address; break it up and create an IPv4 address... + */ + + if (sscanf(hostname, "%u.%u.%u.%u", ip, ip + 1, ip + 2, ip + 3) == 4 && + ip[0] <= 255 && ip[1] <= 255 && ip[2] <= 255 && ip[3] <= 255) + { + first = (http_addrlist_t *)calloc(1, sizeof(http_addrlist_t)); + if (!first) + return (NULL); + + first->addr.ipv4.sin_family = AF_INET; + first->addr.ipv4.sin_addr.s_addr = htonl(((((((ip[0] << 8) | + ip[1]) << 8) | + ip[2]) << 8) | ip[3])); + first->addr.ipv4.sin_port = htons(portnum); + } + } + else if ((host = gethostbyname(hostname)) != NULL && +# ifdef AF_INET6 + (host->h_addrtype == AF_INET || host->h_addrtype == AF_INET6)) +# else + host->h_addrtype == AF_INET) +# endif /* AF_INET6 */ + { + for (i = 0; host->h_addr_list[i]; i ++) + { + /* + * Copy the address over... + */ + + temp = (http_addrlist_t *)calloc(1, sizeof(http_addrlist_t)); + if (!temp) + { + httpAddrFreeList(first); + return (NULL); + } + +# ifdef AF_INET6 + if (host->h_addrtype == AF_INET6) + { + temp->addr.ipv6.sin6_family = AF_INET6; + memcpy(&(temp->addr.ipv6.sin6_addr), host->h_addr_list[i], + sizeof(temp->addr.ipv6)); + temp->addr.ipv6.sin6_port = htons(portnum); + } + else +# endif /* AF_INET6 */ + { + temp->addr.ipv4.sin_family = AF_INET; + memcpy(&(temp->addr.ipv4.sin_addr), host->h_addr_list[i], + sizeof(temp->addr.ipv4)); + temp->addr.ipv4.sin_port = htons(portnum); + } + + /* + * Append the address to the list... + */ + + if (!first) + first = temp; + + if (addr) + addr->next = temp; + + addr = temp; + } + } + else + { + if (h_errno == NO_RECOVERY) + cg->need_res_init = 1; + + _cupsSetError(IPP_INTERNAL_ERROR, hstrerror(h_errno), 0); + } + } +#endif /* HAVE_GETADDRINFO */ + } + + /* + * Detect some common errors and handle them sanely... + */ + + if (!addr && (!hostname || !_cups_strcasecmp(hostname, "localhost"))) + { + struct servent *port; /* Port number for service */ + int portnum; /* Port number */ + + + /* + * Lookup the service... + */ + + if (!service) + portnum = 0; + else if (isdigit(*service & 255)) + portnum = atoi(service); + else if ((port = getservbyname(service, NULL)) != NULL) + portnum = ntohs(port->s_port); + else if (!strcmp(service, "http")) + portnum = 80; + else if (!strcmp(service, "https")) + portnum = 443; + else if (!strcmp(service, "ipp") || !strcmp(service, "ipps")) + portnum = 631; + else if (!strcmp(service, "lpd")) + portnum = 515; + else if (!strcmp(service, "socket")) + portnum = 9100; + else + { + httpAddrFreeList(first); + + _cupsSetError(IPP_INTERNAL_ERROR, _("Unknown service name."), 1); + return (NULL); + } + + if (hostname && !_cups_strcasecmp(hostname, "localhost")) + { + /* + * Unfortunately, some users ignore all of the warnings in the + * /etc/hosts file and delete "localhost" from it. If we get here + * then we were unable to resolve the name, so use the IPv6 and/or + * IPv4 loopback interface addresses... + */ + +#ifdef AF_INET6 + if (family != AF_INET) + { + /* + * Add [::1] to the address list... + */ + + temp = (http_addrlist_t *)calloc(1, sizeof(http_addrlist_t)); + if (!temp) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(errno), 0); + httpAddrFreeList(first); + return (NULL); + } + + temp->addr.ipv6.sin6_family = AF_INET6; + temp->addr.ipv6.sin6_port = htons(portnum); +# ifdef WIN32 + temp->addr.ipv6.sin6_addr.u.Byte[15] = 1; +# else + temp->addr.ipv6.sin6_addr.s6_addr32[3] = htonl(1); +# endif /* WIN32 */ + + if (!first) + first = temp; + + addr = temp; + } + + if (family != AF_INET6) +#endif /* AF_INET6 */ + { + /* + * Add 127.0.0.1 to the address list... + */ + + temp = (http_addrlist_t *)calloc(1, sizeof(http_addrlist_t)); + if (!temp) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(errno), 0); + httpAddrFreeList(first); + return (NULL); + } + + temp->addr.ipv4.sin_family = AF_INET; + temp->addr.ipv4.sin_port = htons(portnum); + temp->addr.ipv4.sin_addr.s_addr = htonl(0x7f000001); + + if (!first) + first = temp; + + if (addr) + addr->next = temp; + } + } + else if (!hostname) + { + /* + * Provide one or more passive listening addresses... + */ + +#ifdef AF_INET6 + if (family != AF_INET) + { + /* + * Add [::] to the address list... + */ + + temp = (http_addrlist_t *)calloc(1, sizeof(http_addrlist_t)); + if (!temp) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(errno), 0); + httpAddrFreeList(first); + return (NULL); + } + + temp->addr.ipv6.sin6_family = AF_INET6; + temp->addr.ipv6.sin6_port = htons(portnum); + + if (!first) + first = temp; + + addr = temp; + } + + if (family != AF_INET6) +#endif /* AF_INET6 */ + { + /* + * Add 0.0.0.0 to the address list... + */ + + temp = (http_addrlist_t *)calloc(1, sizeof(http_addrlist_t)); + if (!temp) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(errno), 0); + httpAddrFreeList(first); + return (NULL); + } + + temp->addr.ipv4.sin_family = AF_INET; + temp->addr.ipv4.sin_port = htons(portnum); + + if (!first) + first = temp; + + if (addr) + addr->next = temp; + } + } + } + + /* + * Return the address list... + */ + + return (first); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/http-private.h b/cups/http-private.h new file mode 100644 index 0000000000..8fea8d3a1b --- /dev/null +++ b/cups/http-private.h @@ -0,0 +1,408 @@ +/* + * "$Id$" + * + * Private HTTP definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_HTTP_PRIVATE_H_ +# define _CUPS_HTTP_PRIVATE_H_ + +/* + * Include necessary headers... + */ + +# include "config.h" +# include +# include + +# ifdef __sun +# include +# endif /* __sun */ + +# include +# ifdef WIN32 +# include +# include +# else +# include +# include +# include +# define closesocket(f) close(f) +# endif /* WIN32 */ + +# ifdef HAVE_GSSAPI +# ifdef HAVE_GSS_GSSAPI_H +# include +# ifdef HAVE_GSSAPI_GENERIC_H +# include +# endif /* HAVE_GSSAPI_GENERIC_H */ +# ifdef HAVE_GSSAPI_KRB5_H +# include +# endif /* HAVE_GSSAPI_KRB5_H */ +# elif defined(HAVE_GSSAPI_GSSAPI_H) +# include +# ifdef HAVE_GSSAPI_GENERIC_H +# include +# endif /* HAVE_GSSAPI_GENERIC_H */ +# ifdef HAVE_GSSAPI_KRB5_H +# include +# endif /* HAVE_GSSAPI_KRB5_H */ +# elif defined(HAVE_GSSAPI_H) +# include +# endif /* HAVE_GSS_GSSAPI_H */ +# ifndef HAVE_GSS_C_NT_HOSTBASED_SERVICE +# define GSS_C_NT_HOSTBASED_SERVICE gss_nt_service_name +# endif /* !HAVE_GSS_C_NT_HOSTBASED_SERVICE */ +# ifdef HAVE_KRB5_H +# include +# endif /* HAVE_KRB5_H */ +# endif /* HAVE_GSSAPI */ + +# ifdef HAVE_AUTHORIZATION_H +# include +# endif /* HAVE_AUTHORIZATION_H */ + +# if defined(__sgi) || (defined(__APPLE__) && !defined(_SOCKLEN_T)) +/* + * IRIX and MacOS X 10.2.x do not define socklen_t, and in fact use an int instead of + * unsigned type for length values... + */ + +typedef int socklen_t; +# endif /* __sgi || (__APPLE__ && !_SOCKLEN_T) */ + +# include +# include "md5-private.h" +# include "ipp-private.h" + +# if defined HAVE_LIBSSL +# include +# include +# include +# elif defined HAVE_GNUTLS +# include +# include +# include +# elif defined(HAVE_CDSASSL) +# include +# include +# include +# ifdef HAVE_SECURETRANSPORTPRIV_H +# include +# endif /* HAVE_SECURETRANSPORTPRIV_H */ +# ifdef HAVE_SECITEM_H +# include +# endif /* HAVE_SECITEM_H */ +# ifdef HAVE_SECBASEPRIV_H +# include +# endif /* HAVE_SECBASEPRIV_H */ +# ifdef HAVE_SECCERTIFICATE_H +# include +# include +# endif /* HAVE_SECCERTIFICATE_H */ +# ifdef HAVE_SECITEMPRIV_H +# include +# endif /* HAVE_SECITEMPRIV_H */ +# ifdef HAVE_SECIDENTITYSEARCHPRIV_H +# include +# endif /* HAVE_SECIDENTITYSEARCHPRIV_H */ +# ifdef HAVE_SECPOLICYPRIV_H +# include +# endif /* HAVE_SECPOLICYPRIV_H */ +# elif defined(HAVE_SSPISSL) +# include "sspi-private.h" +# endif /* HAVE_LIBSSL */ + +# ifndef WIN32 +# include +# ifdef HAVE_GETIFADDRS +# include +# else +# include +# ifdef HAVE_SYS_SOCKIO_H +# include +# endif /* HAVE_SYS_SOCKIO_H */ +# endif /* HAVE_GETIFADDRS */ +# endif /* !WIN32 */ + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Constants... + */ + + +#define _HTTP_RESOLVE_DEFAULT 0 /* Just resolve with default options */ +#define _HTTP_RESOLVE_STDERR 1 /* Log resolve progress to stderr */ +#define _HTTP_RESOLVE_FQDN 2 /* Resolve to a FQDN */ + + +/* + * Types and functions for SSL support... + */ + +# if defined HAVE_LIBSSL +/* + * The OpenSSL library provides its own SSL/TLS context structure for its + * IO and protocol management. However, we need to provide our own BIO + * (basic IO) implementation to do timeouts... + */ + +typedef SSL *http_tls_t; +typedef void *http_tls_credentials_t; + +extern BIO_METHOD *_httpBIOMethods(void); + +# elif defined HAVE_GNUTLS +/* + * The GNU TLS library is more of a "bare metal" SSL/TLS library... + */ + +typedef gnutls_session http_tls_t; +typedef void *http_tls_credentials_t; + +extern ssize_t _httpReadGNUTLS(gnutls_transport_ptr ptr, void *data, + size_t length); +extern ssize_t _httpWriteGNUTLS(gnutls_transport_ptr ptr, const void *data, + size_t length); + +# elif defined(HAVE_CDSASSL) +/* + * Darwin's Security framework provides its own SSL/TLS context structure + * for its IO and protocol management... + */ + +# if !defined(HAVE_SECBASEPRIV_H) && defined(HAVE_CSSMERRORSTRING) /* Declare prototype for function in that header... */ +extern const char *cssmErrorString(int error); +# endif /* !HAVE_SECBASEPRIV_H && HAVE_CSSMERRORSTRING */ +# ifndef HAVE_SECITEMPRIV_H /* Declare constants from that header... */ +extern const CFTypeRef kSecClassCertificate; +extern const CFTypeRef kSecClassIdentity; +# endif /* !HAVE_SECITEMPRIV_H */ +# if !defined(HAVE_SECIDENTITYSEARCHPRIV_H) && defined(HAVE_SECIDENTITYSEARCHCREATEWITHPOLICY) /* Declare prototype for function in that header... */ +extern OSStatus SecIdentitySearchCreateWithPolicy(SecPolicyRef policy, + CFStringRef idString, CSSM_KEYUSE keyUsage, + CFTypeRef keychainOrArray, + Boolean returnOnlyValidIdentities, + SecIdentitySearchRef* searchRef); +# endif /* !HAVE_SECIDENTITYSEARCHPRIV_H && HAVE_SECIDENTITYSEARCHCREATEWITHPOLICY */ +# if !defined(HAVE_SECPOLICYPRIV_H) && defined(HAVE_SECPOLICYSETVALUE) /* Declare prototype for function in that header... */ +extern OSStatus SecPolicySetValue(SecPolicyRef policyRef, + const CSSM_DATA *value); +# endif /* !HAVE_SECPOLICYPRIV_H && HAVE_SECPOLICYSETVALUE */ + +typedef SSLContextRef http_tls_t; +typedef CFArrayRef http_tls_credentials_t; + +extern OSStatus _httpReadCDSA(SSLConnectionRef connection, void *data, + size_t *dataLength); +extern OSStatus _httpWriteCDSA(SSLConnectionRef connection, const void *data, + size_t *dataLength); + +# elif defined(HAVE_SSPISSL) +/* + * Windows' SSPI library gets a CUPS wrapper... + */ + +typedef _sspi_struct_t * http_tls_t; +typedef void *http_tls_credentials_t; + +# else +/* + * Otherwise define stub types since we have no SSL support... + */ + +typedef void *http_tls_t; +typedef void *http_tls_credentials_t; +# endif /* HAVE_LIBSSL */ + +struct _http_s /**** HTTP connection structure. ****/ +{ + int fd; /* File descriptor for this socket */ + int blocking; /* To block or not to block */ + int error; /* Last error on read */ + time_t activity; /* Time since last read/write */ + http_state_t state; /* State of client */ + http_status_t status; /* Status of last request */ + http_version_t version; /* Protocol version */ + http_keepalive_t keep_alive; /* Keep-alive supported? */ + struct sockaddr_in _hostaddr; /* Address of connected host (deprecated) */ + char hostname[HTTP_MAX_HOST], + /* Name of connected host */ + fields[HTTP_FIELD_MAX][HTTP_MAX_VALUE]; + /* Field values */ + char *data; /* Pointer to data buffer */ + http_encoding_t data_encoding; /* Chunked or not */ + int _data_remaining;/* Number of bytes left (deprecated) */ + int used; /* Number of bytes used in buffer */ + char buffer[HTTP_MAX_BUFFER]; + /* Buffer for incoming data */ + int auth_type; /* Authentication in use */ + _cups_md5_state_t md5_state; /* MD5 state */ + char nonce[HTTP_MAX_VALUE]; + /* Nonce value */ + int nonce_count; /* Nonce count */ + http_tls_t tls; /* TLS state information */ + http_encryption_t encryption; /* Encryption requirements */ + /**** New in CUPS 1.1.19 ****/ + fd_set *input_set; /* select() set for httpWait() (deprecated) */ + http_status_t expect; /* Expect: header */ + char *cookie; /* Cookie value(s) */ + /**** New in CUPS 1.1.20 ****/ + char _authstring[HTTP_MAX_VALUE], + /* Current Authentication value (deprecated) */ + userpass[HTTP_MAX_VALUE]; + /* Username:password string */ + int digest_tries; /* Number of tries for digest auth */ + /**** New in CUPS 1.2 ****/ + off_t data_remaining; /* Number of bytes left */ + http_addr_t *hostaddr; /* Current host address and port */ + http_addrlist_t *addrlist; /* List of valid addresses */ + char wbuffer[HTTP_MAX_BUFFER]; + /* Buffer for outgoing data */ + int wused; /* Write buffer bytes used */ + /**** New in CUPS 1.3 ****/ + char *field_authorization; + /* Authorization field */ + char *authstring; /* Current authorization field */ +# ifdef HAVE_GSSAPI + gss_OID gssmech; /* Authentication mechanism */ + gss_ctx_id_t gssctx; /* Authentication context */ + gss_name_t gssname; /* Authentication server name */ +# endif /* HAVE_GSSAPI */ +# ifdef HAVE_AUTHORIZATION_H + AuthorizationRef auth_ref; /* Authorization ref */ +# endif /* HAVE_AUTHORIZATION_H */ + /**** New in CUPS 1.5 ****/ + http_tls_credentials_t tls_credentials; + /* TLS credentials */ + http_timeout_cb_t timeout_cb; /* Timeout callback */ + void *timeout_data; /* User data pointer */ + double timeout_value; /* Timeout in seconds */ + int wait_value; /* httpWait value for timeout */ +# ifdef HAVE_GSSAPI + char gsshost[256]; /* Hostname for Kerberos */ +# endif /* HAVE_GSSAPI */ +}; + + +/* + * Some OS's don't have hstrerror(), most notably Solaris... + */ + +# ifndef HAVE_HSTRERROR +extern const char *_cups_hstrerror(int error); +# define hstrerror _cups_hstrerror +# elif defined(_AIX) || defined(__osf__) +/* + * AIX and Tru64 UNIX don't provide a prototype but do provide the function... + */ +extern const char *hstrerror(int error); +# endif /* !HAVE_HSTRERROR */ + + +/* + * Some OS's don't have getifaddrs() and freeifaddrs()... + */ + +# if !defined(WIN32) && !defined(HAVE_GETIFADDRS) +# ifdef ifa_dstaddr +# undef ifa_dstaddr +# endif /* ifa_dstaddr */ +# ifndef ifr_netmask +# define ifr_netmask ifr_addr +# endif /* !ifr_netmask */ + +struct ifaddrs /**** Interface Structure ****/ +{ + struct ifaddrs *ifa_next; /* Next interface in list */ + char *ifa_name; /* Name of interface */ + unsigned int ifa_flags; /* Flags (up, point-to-point, etc.) */ + struct sockaddr *ifa_addr, /* Network address */ + *ifa_netmask; /* Address mask */ + union + { + struct sockaddr *ifu_broadaddr; /* Broadcast address of this interface. */ + struct sockaddr *ifu_dstaddr; /* Point-to-point destination address. */ + } ifa_ifu; + + void *ifa_data; /* Interface statistics */ +}; + +# ifndef ifa_broadaddr +# define ifa_broadaddr ifa_ifu.ifu_broadaddr +# endif /* !ifa_broadaddr */ +# ifndef ifa_dstaddr +# define ifa_dstaddr ifa_ifu.ifu_dstaddr +# endif /* !ifa_dstaddr */ + +extern int _cups_getifaddrs(struct ifaddrs **addrs); +# define getifaddrs _cups_getifaddrs +extern void _cups_freeifaddrs(struct ifaddrs *addrs); +# define freeifaddrs _cups_freeifaddrs +# endif /* !WIN32 && !HAVE_GETIFADDRS */ + + +/* + * Prototypes... + */ + +#define _httpAddrFamily(addrp) (addrp)->addr.sa_family +extern int _httpAddrPort(http_addr_t *addr); +extern void _httpAddrSetPort(http_addr_t *addr, int port); +extern char *_httpAssembleUUID(const char *server, int port, + const char *name, int number, + char *buffer, size_t bufsize); +extern http_t *_httpCreate(const char *host, int port, + http_addrlist_t *addrlist, + http_encryption_t encryption, + int family); +extern http_tls_credentials_t + _httpCreateCredentials(cups_array_t *credentials); +extern char *_httpDecodeURI(char *dst, const char *src, + size_t dstsize); +extern void _httpDisconnect(http_t *http); +extern char *_httpEncodeURI(char *dst, const char *src, + size_t dstsize); +extern void _httpFreeCredentials(http_tls_credentials_t credentials); +extern ssize_t _httpPeek(http_t *http, char *buffer, size_t length); +extern const char *_httpResolveURI(const char *uri, char *resolved_uri, + size_t resolved_size, int options, + int (*cb)(void *context), + void *context); +extern int _httpUpdate(http_t *http, http_status_t *status); +extern int _httpWait(http_t *http, int msec, int usessl); + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +} +# endif /* __cplusplus */ + +#endif /* !_CUPS_HTTP_PRIVATE_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/http-support.c b/cups/http-support.c new file mode 100644 index 0000000000..877d177b32 --- /dev/null +++ b/cups/http-support.c @@ -0,0 +1,1919 @@ +/* + * "$Id$" + * + * HTTP support routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * httpAssembleURI() - Assemble a uniform resource identifier from its + * components. + * httpAssembleURIf() - Assemble a uniform resource identifier from its + * components with a formatted resource. + * _httpAssembleUUID() - Make a UUID URI conforming to RFC 4122. + * httpDecode64() - Base64-decode a string. + * httpDecode64_2() - Base64-decode a string. + * httpEncode64() - Base64-encode a string. + * httpEncode64_2() - Base64-encode a string. + * httpGetDateString() - Get a formatted date/time string from a time value. + * httpGetDateString2() - Get a formatted date/time string from a time value. + * httpGetDateTime() - Get a time value from a formatted date/time string. + * httpSeparate() - Separate a Universal Resource Identifier into its + * components. + * httpSeparate2() - Separate a Universal Resource Identifier into its + * components. + * httpSeparateURI() - Separate a Universal Resource Identifier into its + * components. + * httpStatus() - Return a short string describing a HTTP status code. + * _cups_hstrerror() - hstrerror() emulation function for Solaris and + * others. + * _httpDecodeURI() - Percent-decode a HTTP request URI. + * _httpEncodeURI() - Percent-encode a HTTP request URI. + * _httpResolveURI() - Resolve a DNS-SD URI. + * http_copy_decode() - Copy and decode a URI. + * http_copy_encode() - Copy and encode a URI. + * http_resolve_cb() - Build a device URI for the given service name. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#ifdef HAVE_DNSSD +# include +# ifdef WIN32 +# include +# elif defined(HAVE_POLL) +# include +# else +# include +# endif /* WIN32 */ +#endif /* HAVE_DNSSD */ + + +/* + * Local types... + */ + +typedef struct _http_uribuf_s /* URI buffer */ +{ + char *buffer; /* Pointer to buffer */ + size_t bufsize; /* Size of buffer */ + int options; /* Options passed to _httpResolveURI */ +} _http_uribuf_t; + + +/* + * Local globals... + */ + +static const char * const http_days[7] = + { + "Sun", + "Mon", + "Tue", + "Wed", + "Thu", + "Fri", + "Sat" + }; +static const char * const http_months[12] = + { + "Jan", + "Feb", + "Mar", + "Apr", + "May", + "Jun", + "Jul", + "Aug", + "Sep", + "Oct", + "Nov", + "Dec" + }; + + +/* + * Local functions... + */ + +static const char *http_copy_decode(char *dst, const char *src, + int dstsize, const char *term, + int decode); +static char *http_copy_encode(char *dst, const char *src, + char *dstend, const char *reserved, + const char *term, int encode); +#ifdef HAVE_DNSSD +static void DNSSD_API http_resolve_cb(DNSServiceRef sdRef, + DNSServiceFlags flags, + uint32_t interfaceIndex, + DNSServiceErrorType errorCode, + const char *fullName, + const char *hostTarget, + uint16_t port, uint16_t txtLen, + const unsigned char *txtRecord, + void *context); +#endif /* HAVE_DNSSD */ + + +/* + * 'httpAssembleURI()' - Assemble a uniform resource identifier from its + * components. + * + * This function escapes reserved characters in the URI depending on the + * value of the "encoding" argument. You should use this function in + * place of traditional string functions whenever you need to create a + * URI string. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +http_uri_status_t /* O - URI status */ +httpAssembleURI( + http_uri_coding_t encoding, /* I - Encoding flags */ + char *uri, /* I - URI buffer */ + int urilen, /* I - Size of URI buffer */ + const char *scheme, /* I - Scheme name */ + const char *username, /* I - Username */ + const char *host, /* I - Hostname or address */ + int port, /* I - Port number */ + const char *resource) /* I - Resource */ +{ + char *ptr, /* Pointer into URI buffer */ + *end; /* End of URI buffer */ + + + /* + * Range check input... + */ + + if (!uri || urilen < 1 || !scheme || port < 0) + { + if (uri) + *uri = '\0'; + + return (HTTP_URI_BAD_ARGUMENTS); + } + + /* + * Assemble the URI starting with the scheme... + */ + + end = uri + urilen - 1; + ptr = http_copy_encode(uri, scheme, end, NULL, NULL, 0); + + if (!ptr) + goto assemble_overflow; + + if (!strcmp(scheme, "mailto")) + { + /* + * mailto: only has :, no //... + */ + + if (ptr < end) + *ptr++ = ':'; + else + goto assemble_overflow; + } + else + { + /* + * Schemes other than mailto: all have //... + */ + + if ((ptr + 2) < end) + { + *ptr++ = ':'; + *ptr++ = '/'; + *ptr++ = '/'; + } + else + goto assemble_overflow; + } + + /* + * Next the username and hostname, if any... + */ + + if (host) + { + if (username && *username) + { + /* + * Add username@ first... + */ + + ptr = http_copy_encode(ptr, username, end, "/?#[]@", NULL, + encoding & HTTP_URI_CODING_USERNAME); + + if (!ptr) + goto assemble_overflow; + + if (ptr < end) + *ptr++ = '@'; + else + goto assemble_overflow; + } + + /* + * Then add the hostname. Since IPv6 is a particular pain to deal + * with, we have several special cases to deal with. If we get + * an IPv6 address with brackets around it, assume it is already in + * URI format. Since DNS-SD service names can sometimes look like + * raw IPv6 addresses, we specifically look for "._tcp" in the name, + * too... + */ + + if (host[0] != '[' && strchr(host, ':') && !strstr(host, "._tcp")) + { + /* + * We have a raw IPv6 address... + */ + + if (strchr(host, '%')) + { + /* + * We have a link-local address, add "[v1." prefix... + */ + + if ((ptr + 4) < end) + { + *ptr++ = '['; + *ptr++ = 'v'; + *ptr++ = '1'; + *ptr++ = '.'; + } + else + goto assemble_overflow; + } + else + { + /* + * We have a normal address, add "[" prefix... + */ + + if (ptr < end) + *ptr++ = '['; + else + goto assemble_overflow; + } + + /* + * Copy the rest of the IPv6 address, and terminate with "]". + */ + + while (ptr < end && *host) + { + if (*host == '%') + { + *ptr++ = '+'; /* Convert zone separator */ + host ++; + } + else + *ptr++ = *host++; + } + + if (*host) + goto assemble_overflow; + + if (ptr < end) + *ptr++ = ']'; + else + goto assemble_overflow; + } + else + { + /* + * Otherwise, just copy the host string... + */ + + ptr = http_copy_encode(ptr, host, end, ":/?#[]@\\\"", NULL, + encoding & HTTP_URI_CODING_HOSTNAME); + + if (!ptr) + goto assemble_overflow; + } + + /* + * Finish things off with the port number... + */ + + if (port > 0) + { + snprintf(ptr, end - ptr + 1, ":%d", port); + ptr += strlen(ptr); + + if (ptr >= end) + goto assemble_overflow; + } + } + + /* + * Last but not least, add the resource string... + */ + + if (resource) + { + char *query; /* Pointer to query string */ + + + /* + * Copy the resource string up to the query string if present... + */ + + query = strchr(resource, '?'); + ptr = http_copy_encode(ptr, resource, end, NULL, "?", + encoding & HTTP_URI_CODING_RESOURCE); + if (!ptr) + goto assemble_overflow; + + if (query) + { + /* + * Copy query string without encoding... + */ + + ptr = http_copy_encode(ptr, query, end, NULL, NULL, + encoding & HTTP_URI_CODING_QUERY); + if (!ptr) + goto assemble_overflow; + } + } + else if (ptr < end) + *ptr++ = '/'; + else + goto assemble_overflow; + + /* + * Nul-terminate the URI buffer and return with no errors... + */ + + *ptr = '\0'; + + return (HTTP_URI_OK); + + /* + * Clear the URI string and return an overflow error; I don't usually + * like goto's, but in this case it makes sense... + */ + + assemble_overflow: + + *uri = '\0'; + return (HTTP_URI_OVERFLOW); +} + + +/* + * 'httpAssembleURIf()' - Assemble a uniform resource identifier from its + * components with a formatted resource. + * + * This function creates a formatted version of the resource string + * argument "resourcef" and escapes reserved characters in the URI + * depending on the value of the "encoding" argument. You should use + * this function in place of traditional string functions whenever + * you need to create a URI string. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +http_uri_status_t /* O - URI status */ +httpAssembleURIf( + http_uri_coding_t encoding, /* I - Encoding flags */ + char *uri, /* I - URI buffer */ + int urilen, /* I - Size of URI buffer */ + const char *scheme, /* I - Scheme name */ + const char *username, /* I - Username */ + const char *host, /* I - Hostname or address */ + int port, /* I - Port number */ + const char *resourcef, /* I - Printf-style resource */ + ...) /* I - Additional arguments as needed */ +{ + va_list ap; /* Pointer to additional arguments */ + char resource[1024]; /* Formatted resource string */ + int bytes; /* Bytes in formatted string */ + + + /* + * Range check input... + */ + + if (!uri || urilen < 1 || !scheme || port < 0 || !resourcef) + { + if (uri) + *uri = '\0'; + + return (HTTP_URI_BAD_ARGUMENTS); + } + + /* + * Format the resource string and assemble the URI... + */ + + va_start(ap, resourcef); + bytes = vsnprintf(resource, sizeof(resource), resourcef, ap); + va_end(ap); + + if (bytes >= sizeof(resource)) + { + *uri = '\0'; + return (HTTP_URI_OVERFLOW); + } + else + return (httpAssembleURI(encoding, uri, urilen, scheme, username, host, + port, resource)); +} + + +/* + * '_httpAssembleUUID()' - Make a UUID URI conforming to RFC 4122. + * + * The buffer needs to be at least 46 bytes in size. + */ + +char * /* I - UUID string */ +_httpAssembleUUID(const char *server, /* I - Server name */ + int port, /* I - Port number */ + const char *name, /* I - Object name or NULL */ + int number, /* I - Object number or 0 */ + char *buffer, /* I - String buffer */ + size_t bufsize) /* I - Size of buffer */ +{ + char data[1024]; /* Source string for MD5 */ + _cups_md5_state_t md5state; /* MD5 state */ + unsigned char md5sum[16]; /* MD5 digest/sum */ + + + /* + * Build a version 3 UUID conforming to RFC 4122. + * + * Start with the MD5 sum of the server, port, object name and + * number, and some random data on the end. + */ + + snprintf(data, sizeof(data), "%s:%d:%s:%d:%04x:%04x", server, + port, name ? name : server, number, + (unsigned)CUPS_RAND() & 0xffff, (unsigned)CUPS_RAND() & 0xffff); + + _cupsMD5Init(&md5state); + _cupsMD5Append(&md5state, (unsigned char *)data, strlen(data)); + _cupsMD5Finish(&md5state, md5sum); + + /* + * Generate the UUID from the MD5... + */ + + snprintf(buffer, bufsize, + "urn:uuid:%02x%02x%02x%02x-%02x%02x-%02x%02x-%02x%02x-" + "%02x%02x%02x%02x%02x%02x", + md5sum[0], md5sum[1], md5sum[2], md5sum[3], md5sum[4], md5sum[5], + (md5sum[6] & 15) | 0x30, md5sum[7], (md5sum[8] & 0x3f) | 0x40, + md5sum[9], md5sum[10], md5sum[11], md5sum[12], md5sum[13], + md5sum[14], md5sum[15]); + + return (buffer); +} + + +/* + * 'httpDecode64()' - Base64-decode a string. + * + * This function is deprecated. Use the httpDecode64_2() function instead + * which provides buffer length arguments. + * + * @deprecated@ + */ + +char * /* O - Decoded string */ +httpDecode64(char *out, /* I - String to write to */ + const char *in) /* I - String to read from */ +{ + int outlen; /* Output buffer length */ + + + /* + * Use the old maximum buffer size for binary compatibility... + */ + + outlen = 512; + + return (httpDecode64_2(out, &outlen, in)); +} + + +/* + * 'httpDecode64_2()' - Base64-decode a string. + * + * @since CUPS 1.1.21/Mac OS X 10.4@ + */ + +char * /* O - Decoded string */ +httpDecode64_2(char *out, /* I - String to write to */ + int *outlen, /* IO - Size of output string */ + const char *in) /* I - String to read from */ +{ + int pos, /* Bit position */ + base64; /* Value of this character */ + char *outptr, /* Output pointer */ + *outend; /* End of output buffer */ + + + /* + * Range check input... + */ + + if (!out || !outlen || *outlen < 1 || !in) + return (NULL); + + if (!*in) + { + *out = '\0'; + *outlen = 0; + + return (out); + } + + /* + * Convert from base-64 to bytes... + */ + + for (outptr = out, outend = out + *outlen - 1, pos = 0; *in != '\0'; in ++) + { + /* + * Decode this character into a number from 0 to 63... + */ + + if (*in >= 'A' && *in <= 'Z') + base64 = *in - 'A'; + else if (*in >= 'a' && *in <= 'z') + base64 = *in - 'a' + 26; + else if (*in >= '0' && *in <= '9') + base64 = *in - '0' + 52; + else if (*in == '+') + base64 = 62; + else if (*in == '/') + base64 = 63; + else if (*in == '=') + break; + else + continue; + + /* + * Store the result in the appropriate chars... + */ + + switch (pos) + { + case 0 : + if (outptr < outend) + *outptr = base64 << 2; + pos ++; + break; + case 1 : + if (outptr < outend) + *outptr++ |= (base64 >> 4) & 3; + if (outptr < outend) + *outptr = (base64 << 4) & 255; + pos ++; + break; + case 2 : + if (outptr < outend) + *outptr++ |= (base64 >> 2) & 15; + if (outptr < outend) + *outptr = (base64 << 6) & 255; + pos ++; + break; + case 3 : + if (outptr < outend) + *outptr++ |= base64; + pos = 0; + break; + } + } + + *outptr = '\0'; + + /* + * Return the decoded string and size... + */ + + *outlen = (int)(outptr - out); + + return (out); +} + + +/* + * 'httpEncode64()' - Base64-encode a string. + * + * This function is deprecated. Use the httpEncode64_2() function instead + * which provides buffer length arguments. + * + * @deprecated@ + */ + +char * /* O - Encoded string */ +httpEncode64(char *out, /* I - String to write to */ + const char *in) /* I - String to read from */ +{ + return (httpEncode64_2(out, 512, in, (int)strlen(in))); +} + + +/* + * 'httpEncode64_2()' - Base64-encode a string. + * + * @since CUPS 1.1.21/Mac OS X 10.4@ + */ + +char * /* O - Encoded string */ +httpEncode64_2(char *out, /* I - String to write to */ + int outlen, /* I - Size of output string */ + const char *in, /* I - String to read from */ + int inlen) /* I - Size of input string */ +{ + char *outptr, /* Output pointer */ + *outend; /* End of output buffer */ + static const char base64[] = /* Base64 characters... */ + { + "ABCDEFGHIJKLMNOPQRSTUVWXYZ" + "abcdefghijklmnopqrstuvwxyz" + "0123456789" + "+/" + }; + + + /* + * Range check input... + */ + + if (!out || outlen < 1 || !in) + return (NULL); + + /* + * Convert bytes to base-64... + */ + + for (outptr = out, outend = out + outlen - 1; inlen > 0; in ++, inlen --) + { + /* + * Encode the up to 3 characters as 4 Base64 numbers... + */ + + if (outptr < outend) + *outptr ++ = base64[(in[0] & 255) >> 2]; + + if (outptr < outend) + { + if (inlen > 1) + *outptr ++ = base64[(((in[0] & 255) << 4) | ((in[1] & 255) >> 4)) & 63]; + else + *outptr ++ = base64[((in[0] & 255) << 4) & 63]; + } + + in ++; + inlen --; + if (inlen <= 0) + { + if (outptr < outend) + *outptr ++ = '='; + if (outptr < outend) + *outptr ++ = '='; + break; + } + + if (outptr < outend) + { + if (inlen > 1) + *outptr ++ = base64[(((in[0] & 255) << 2) | ((in[1] & 255) >> 6)) & 63]; + else + *outptr ++ = base64[((in[0] & 255) << 2) & 63]; + } + + in ++; + inlen --; + if (inlen <= 0) + { + if (outptr < outend) + *outptr ++ = '='; + break; + } + + if (outptr < outend) + *outptr ++ = base64[in[0] & 63]; + } + + *outptr = '\0'; + + /* + * Return the encoded string... + */ + + return (out); +} + + +/* + * 'httpGetDateString()' - Get a formatted date/time string from a time value. + * + * @deprecated@ + */ + +const char * /* O - Date/time string */ +httpGetDateString(time_t t) /* I - UNIX time */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + return (httpGetDateString2(t, cg->http_date, sizeof(cg->http_date))); +} + + +/* + * 'httpGetDateString2()' - Get a formatted date/time string from a time value. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +const char * /* O - Date/time string */ +httpGetDateString2(time_t t, /* I - UNIX time */ + char *s, /* I - String buffer */ + int slen) /* I - Size of string buffer */ +{ + struct tm *tdate; /* UNIX date/time data */ + + + tdate = gmtime(&t); + snprintf(s, slen, "%s, %02d %s %d %02d:%02d:%02d GMT", + http_days[tdate->tm_wday], tdate->tm_mday, + http_months[tdate->tm_mon], tdate->tm_year + 1900, + tdate->tm_hour, tdate->tm_min, tdate->tm_sec); + + return (s); +} + + +/* + * 'httpGetDateTime()' - Get a time value from a formatted date/time string. + */ + +time_t /* O - UNIX time */ +httpGetDateTime(const char *s) /* I - Date/time string */ +{ + int i; /* Looping var */ + char mon[16]; /* Abbreviated month name */ + int day, year; /* Day of month and year */ + int hour, min, sec; /* Time */ + int days; /* Number of days since 1970 */ + static const int normal_days[] = /* Days to a month, normal years */ + { 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365 }; + static const int leap_days[] = /* Days to a month, leap years */ + { 0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335, 366 }; + + + DEBUG_printf(("2httpGetDateTime(s=\"%s\")", s)); + + /* + * Extract the date and time from the formatted string... + */ + + if (sscanf(s, "%*s%d%15s%d%d:%d:%d", &day, mon, &year, &hour, &min, &sec) < 6) + return (0); + + DEBUG_printf(("4httpGetDateTime: day=%d, mon=\"%s\", year=%d, hour=%d, " + "min=%d, sec=%d", day, mon, year, hour, min, sec)); + + /* + * Convert the month name to a number from 0 to 11. + */ + + for (i = 0; i < 12; i ++) + if (!_cups_strcasecmp(mon, http_months[i])) + break; + + if (i >= 12) + return (0); + + DEBUG_printf(("4httpGetDateTime: i=%d", i)); + + /* + * Now convert the date and time to a UNIX time value in seconds since + * 1970. We can't use mktime() since the timezone may not be UTC but + * the date/time string *is* UTC. + */ + + if ((year & 3) == 0 && ((year % 100) != 0 || (year % 400) == 0)) + days = leap_days[i] + day - 1; + else + days = normal_days[i] + day - 1; + + DEBUG_printf(("4httpGetDateTime: days=%d", days)); + + days += (year - 1970) * 365 + /* 365 days per year (normally) */ + ((year - 1) / 4 - 492) - /* + leap days */ + ((year - 1) / 100 - 19) + /* - 100 year days */ + ((year - 1) / 400 - 4); /* + 400 year days */ + + DEBUG_printf(("4httpGetDateTime: days=%d\n", days)); + + return (days * 86400 + hour * 3600 + min * 60 + sec); +} + + +/* + * 'httpSeparate()' - Separate a Universal Resource Identifier into its + * components. + * + * This function is deprecated; use the httpSeparateURI() function instead. + * + * @deprecated@ + */ + +void +httpSeparate(const char *uri, /* I - Universal Resource Identifier */ + char *scheme, /* O - Scheme [32] (http, https, etc.) */ + char *username, /* O - Username [1024] */ + char *host, /* O - Hostname [1024] */ + int *port, /* O - Port number to use */ + char *resource) /* O - Resource/filename [1024] */ +{ + httpSeparateURI(HTTP_URI_CODING_ALL, uri, scheme, 32, username, + HTTP_MAX_URI, host, HTTP_MAX_URI, port, resource, + HTTP_MAX_URI); +} + + +/* + * 'httpSeparate2()' - Separate a Universal Resource Identifier into its + * components. + * + * This function is deprecated; use the httpSeparateURI() function instead. + * + * @since CUPS 1.1.21/Mac OS X 10.4@ + * @deprecated@ + */ + +void +httpSeparate2(const char *uri, /* I - Universal Resource Identifier */ + char *scheme, /* O - Scheme (http, https, etc.) */ + int schemelen, /* I - Size of scheme buffer */ + char *username, /* O - Username */ + int usernamelen, /* I - Size of username buffer */ + char *host, /* O - Hostname */ + int hostlen, /* I - Size of hostname buffer */ + int *port, /* O - Port number to use */ + char *resource, /* O - Resource/filename */ + int resourcelen) /* I - Size of resource buffer */ +{ + httpSeparateURI(HTTP_URI_CODING_ALL, uri, scheme, schemelen, username, + usernamelen, host, hostlen, port, resource, resourcelen); +} + + +/* + * 'httpSeparateURI()' - Separate a Universal Resource Identifier into its + * components. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +http_uri_status_t /* O - Result of separation */ +httpSeparateURI( + http_uri_coding_t decoding, /* I - Decoding flags */ + const char *uri, /* I - Universal Resource Identifier */ + char *scheme, /* O - Scheme (http, https, etc.) */ + int schemelen, /* I - Size of scheme buffer */ + char *username, /* O - Username */ + int usernamelen, /* I - Size of username buffer */ + char *host, /* O - Hostname */ + int hostlen, /* I - Size of hostname buffer */ + int *port, /* O - Port number to use */ + char *resource, /* O - Resource/filename */ + int resourcelen) /* I - Size of resource buffer */ +{ + char *ptr, /* Pointer into string... */ + *end; /* End of string */ + const char *sep; /* Separator character */ + http_uri_status_t status; /* Result of separation */ + + + /* + * Initialize everything to blank... + */ + + if (scheme && schemelen > 0) + *scheme = '\0'; + + if (username && usernamelen > 0) + *username = '\0'; + + if (host && hostlen > 0) + *host = '\0'; + + if (port) + *port = 0; + + if (resource && resourcelen > 0) + *resource = '\0'; + + /* + * Range check input... + */ + + if (!uri || !port || !scheme || schemelen <= 0 || !username || + usernamelen <= 0 || !host || hostlen <= 0 || !resource || + resourcelen <= 0) + return (HTTP_URI_BAD_ARGUMENTS); + + if (!*uri) + return (HTTP_URI_BAD_URI); + + /* + * Grab the scheme portion of the URI... + */ + + status = HTTP_URI_OK; + + if (!strncmp(uri, "//", 2)) + { + /* + * Workaround for HP IPP client bug... + */ + + strlcpy(scheme, "ipp", schemelen); + status = HTTP_URI_MISSING_SCHEME; + } + else if (*uri == '/') + { + /* + * Filename... + */ + + strlcpy(scheme, "file", schemelen); + status = HTTP_URI_MISSING_SCHEME; + } + else + { + /* + * Standard URI with scheme... + */ + + for (ptr = scheme, end = scheme + schemelen - 1; + *uri && *uri != ':' && ptr < end;) + if (strchr("ABCDEFGHIJKLMNOPQRSTUVWXYZ" + "abcdefghijklmnopqrstuvwxyz" + "0123456789-+.", *uri) != NULL) + *ptr++ = *uri++; + else + break; + + *ptr = '\0'; + + if (*uri != ':') + { + *scheme = '\0'; + return (HTTP_URI_BAD_SCHEME); + } + + uri ++; + } + + /* + * Set the default port number... + */ + + if (!strcmp(scheme, "http")) + *port = 80; + else if (!strcmp(scheme, "https")) + *port = 443; + else if (!strcmp(scheme, "ipp") || !strcmp(scheme, "ipps")) + *port = 631; + else if (!_cups_strcasecmp(scheme, "lpd")) + *port = 515; + else if (!strcmp(scheme, "socket")) /* Not yet registered with IANA... */ + *port = 9100; + else if (strcmp(scheme, "file") && strcmp(scheme, "mailto")) + status = HTTP_URI_UNKNOWN_SCHEME; + + /* + * Now see if we have a hostname... + */ + + if (!strncmp(uri, "//", 2)) + { + /* + * Yes, extract it... + */ + + uri += 2; + + /* + * Grab the username, if any... + */ + + if ((sep = strpbrk(uri, "@/")) != NULL && *sep == '@') + { + /* + * Get a username:password combo... + */ + + uri = http_copy_decode(username, uri, usernamelen, "@", + decoding & HTTP_URI_CODING_USERNAME); + + if (!uri) + { + *username = '\0'; + return (HTTP_URI_BAD_USERNAME); + } + + uri ++; + } + + /* + * Then the hostname/IP address... + */ + + if (*uri == '[') + { + /* + * Grab IPv6 address... + */ + + uri ++; + if (!strncmp(uri, "v1.", 3)) + uri += 3; /* Skip IPvN leader... */ + + uri = http_copy_decode(host, uri, hostlen, "]", + decoding & HTTP_URI_CODING_HOSTNAME); + + if (!uri) + { + *host = '\0'; + return (HTTP_URI_BAD_HOSTNAME); + } + + /* + * Validate value... + */ + + if (*uri != ']') + { + *host = '\0'; + return (HTTP_URI_BAD_HOSTNAME); + } + + uri ++; + + for (ptr = host; *ptr; ptr ++) + if (*ptr == '+') + { + /* + * Convert zone separator to % and stop here... + */ + + *ptr = '%'; + break; + } + else if (*ptr != ':' && *ptr != '.' && !isxdigit(*ptr & 255)) + { + *host = '\0'; + return (HTTP_URI_BAD_HOSTNAME); + } + } + else + { + /* + * Validate the hostname or IPv4 address first... + */ + + for (ptr = (char *)uri; *ptr; ptr ++) + if (strchr(":?/", *ptr)) + break; + else if (!strchr("abcdefghijklmnopqrstuvwxyz" + "ABCDEFGHIJKLMNOPQRSTUVWXYZ" + "0123456789" + "-._~" + "%" + "!$&'()*+,;=\\", *ptr)) + { + *host = '\0'; + return (HTTP_URI_BAD_HOSTNAME); + } + + /* + * Then copy the hostname or IPv4 address to the buffer... + */ + + uri = http_copy_decode(host, uri, hostlen, ":?/", + decoding & HTTP_URI_CODING_HOSTNAME); + + if (!uri) + { + *host = '\0'; + return (HTTP_URI_BAD_HOSTNAME); + } + } + + /* + * Validate hostname for file scheme - only empty and localhost are + * acceptable. + */ + + if (!strcmp(scheme, "file") && strcmp(host, "localhost") && host[0]) + { + *host = '\0'; + return (HTTP_URI_BAD_HOSTNAME); + } + + /* + * See if we have a port number... + */ + + if (*uri == ':') + { + /* + * Yes, collect the port number... + */ + + if (!isdigit(uri[1] & 255)) + { + *port = 0; + return (HTTP_URI_BAD_PORT); + } + + *port = strtol(uri + 1, (char **)&uri, 10); + + if (*uri != '/' && *uri) + { + *port = 0; + return (HTTP_URI_BAD_PORT); + } + } + } + + /* + * The remaining portion is the resource string... + */ + + if (*uri == '?' || !*uri) + { + /* + * Hostname but no path... + */ + + status = HTTP_URI_MISSING_RESOURCE; + *resource = '/'; + + /* + * Copy any query string... + */ + + if (*uri == '?') + uri = http_copy_decode(resource + 1, uri, resourcelen - 1, NULL, + decoding & HTTP_URI_CODING_QUERY); + else + resource[1] = '\0'; + } + else + { + uri = http_copy_decode(resource, uri, resourcelen, "?", + decoding & HTTP_URI_CODING_RESOURCE); + + if (uri && *uri == '?') + { + /* + * Concatenate any query string... + */ + + char *resptr = resource + strlen(resource); + + uri = http_copy_decode(resptr, uri, resourcelen - (int)(resptr - resource), + NULL, decoding & HTTP_URI_CODING_QUERY); + } + } + + if (!uri) + { + *resource = '\0'; + return (HTTP_URI_BAD_RESOURCE); + } + + /* + * Return the URI separation status... + */ + + return (status); +} + + +/* + * 'httpStatus()' - Return a short string describing a HTTP status code. + * + * The returned string is localized to the current POSIX locale and is based + * on the status strings defined in RFC 2616. + */ + +const char * /* O - Localized status string */ +httpStatus(http_status_t status) /* I - HTTP status code */ +{ + const char *s; /* Status string */ + _cups_globals_t *cg = _cupsGlobals(); /* Global data */ + + + if (!cg->lang_default) + cg->lang_default = cupsLangDefault(); + + switch (status) + { + case HTTP_CONTINUE : + s = _("Continue"); + break; + case HTTP_SWITCHING_PROTOCOLS : + s = _("Switching Protocols"); + break; + case HTTP_OK : + s = _("OK"); + break; + case HTTP_CREATED : + s = _("Created"); + break; + case HTTP_ACCEPTED : + s = _("Accepted"); + break; + case HTTP_NO_CONTENT : + s = _("No Content"); + break; + case HTTP_MOVED_PERMANENTLY : + s = _("Moved Permanently"); + break; + case HTTP_SEE_OTHER : + s = _("See Other"); + break; + case HTTP_NOT_MODIFIED : + s = _("Not Modified"); + break; + case HTTP_BAD_REQUEST : + s = _("Bad Request"); + break; + case HTTP_UNAUTHORIZED : + case HTTP_AUTHORIZATION_CANCELED : + s = _("Unauthorized"); + break; + case HTTP_FORBIDDEN : + s = _("Forbidden"); + break; + case HTTP_NOT_FOUND : + s = _("Not Found"); + break; + case HTTP_REQUEST_TOO_LARGE : + s = _("Request Entity Too Large"); + break; + case HTTP_URI_TOO_LONG : + s = _("URI Too Long"); + break; + case HTTP_UPGRADE_REQUIRED : + s = _("Upgrade Required"); + break; + case HTTP_NOT_IMPLEMENTED : + s = _("Not Implemented"); + break; + case HTTP_NOT_SUPPORTED : + s = _("Not Supported"); + break; + case HTTP_EXPECTATION_FAILED : + s = _("Expectation Failed"); + break; + case HTTP_SERVICE_UNAVAILABLE : + s = _("Service Unavailable"); + break; + case HTTP_SERVER_ERROR : + s = _("Internal Server Error"); + break; + case HTTP_PKI_ERROR : + s = _("SSL/TLS Negotiation Error"); + break; + case HTTP_WEBIF_DISABLED : + s = _("Web Interface is Disabled"); + break; + + default : + s = _("Unknown"); + break; + } + + return (_cupsLangString(cg->lang_default, s)); +} + + +#ifndef HAVE_HSTRERROR +/* + * '_cups_hstrerror()' - hstrerror() emulation function for Solaris and others. + */ + +const char * /* O - Error string */ +_cups_hstrerror(int error) /* I - Error number */ +{ + static const char * const errors[] = /* Error strings */ + { + "OK", + "Host not found.", + "Try again.", + "Unrecoverable lookup error.", + "No data associated with name." + }; + + + if (error < 0 || error > 4) + return ("Unknown hostname lookup error."); + else + return (errors[error]); +} +#endif /* !HAVE_HSTRERROR */ + + +/* + * '_httpDecodeURI()' - Percent-decode a HTTP request URI. + */ + +char * /* O - Decoded URI or NULL on error */ +_httpDecodeURI(char *dst, /* I - Destination buffer */ + const char *src, /* I - Source URI */ + size_t dstsize) /* I - Size of destination buffer */ +{ + if (http_copy_decode(dst, src, (int)dstsize, NULL, 1)) + return (dst); + else + return (NULL); +} + + +/* + * '_httpEncodeURI()' - Percent-encode a HTTP request URI. + */ + +char * /* O - Encoded URI */ +_httpEncodeURI(char *dst, /* I - Destination buffer */ + const char *src, /* I - Source URI */ + size_t dstsize) /* I - Size of destination buffer */ +{ + http_copy_encode(dst, src, dst + dstsize - 1, NULL, NULL, 1); + return (dst); +} + + +/* + * '_httpResolveURI()' - Resolve a DNS-SD URI. + */ + +const char * /* O - Resolved URI */ +_httpResolveURI( + const char *uri, /* I - DNS-SD URI */ + char *resolved_uri, /* I - Buffer for resolved URI */ + size_t resolved_size, /* I - Size of URI buffer */ + int options, /* I - Resolve options */ + int (*cb)(void *context), /* I - Continue callback function */ + void *context) /* I - Context pointer for callback */ +{ + char scheme[32], /* URI components... */ + userpass[256], + hostname[1024], + resource[1024]; + int port; +#ifdef DEBUG + http_uri_status_t status; /* URI decode status */ +#endif /* DEBUG */ + + + DEBUG_printf(("4_httpResolveURI(uri=\"%s\", resolved_uri=%p, " + "resolved_size=" CUPS_LLFMT ")", uri, resolved_uri, + CUPS_LLCAST resolved_size)); + + /* + * Get the device URI... + */ + +#ifdef DEBUG + if ((status = httpSeparateURI(HTTP_URI_CODING_ALL, uri, scheme, + sizeof(scheme), userpass, sizeof(userpass), + hostname, sizeof(hostname), &port, resource, + sizeof(resource))) < HTTP_URI_OK) +#else + if (httpSeparateURI(HTTP_URI_CODING_ALL, uri, scheme, + sizeof(scheme), userpass, sizeof(userpass), + hostname, sizeof(hostname), &port, resource, + sizeof(resource)) < HTTP_URI_OK) +#endif /* DEBUG */ + { + if (options & _HTTP_RESOLVE_STDERR) + _cupsLangPrintFilter(stderr, "ERROR", _("Bad device-uri \"%s\"."), uri); + + DEBUG_printf(("6_httpResolveURI: httpSeparateURI returned %d!", status)); + DEBUG_puts("5_httpResolveURI: Returning NULL"); + return (NULL); + } + + /* + * Resolve it as needed... + */ + + if (strstr(hostname, "._tcp")) + { +#ifdef HAVE_DNSSD +# ifdef WIN32 +# pragma comment(lib, "dnssd.lib") +# endif /* WIN32 */ + DNSServiceRef ref, /* DNS-SD master service reference */ + domainref, /* DNS-SD service reference for domain */ + localref; /* DNS-SD service reference for .local */ + int domainsent = 0, /* Send the domain resolve? */ + offline = 0; /* offline-report state set? */ + char *regtype, /* Pointer to type in hostname */ + *domain; /* Pointer to domain in hostname */ + _http_uribuf_t uribuf; /* URI buffer */ +#ifdef HAVE_POLL + struct pollfd polldata; /* Polling data */ +#else /* select() */ + fd_set input_set; /* Input set for select() */ + struct timeval stimeout; /* Timeout value for select() */ +#endif /* HAVE_POLL */ + + if (options & _HTTP_RESOLVE_STDERR) + fprintf(stderr, "DEBUG: Resolving \"%s\"...\n", hostname); + + /* + * Separate the hostname into service name, registration type, and domain... + */ + + for (regtype = strstr(hostname, "._tcp") - 2; + regtype > hostname; + regtype --) + if (regtype[0] == '.' && regtype[1] == '_') + { + /* + * Found ._servicetype in front of ._tcp... + */ + + *regtype++ = '\0'; + break; + } + + if (regtype <= hostname) + { + DEBUG_puts("5_httpResolveURI: Bad hostname, returning NULL"); + return (NULL); + } + + for (domain = strchr(regtype, '.'); + domain; + domain = strchr(domain + 1, '.')) + if (domain[1] != '_') + break; + + if (domain) + *domain++ = '\0'; + + uribuf.buffer = resolved_uri; + uribuf.bufsize = resolved_size; + uribuf.options = options; + resolved_uri[0] = '\0'; + + DEBUG_printf(("6_httpResolveURI: Resolving hostname=\"%s\", regtype=\"%s\", " + "domain=\"%s\"\n", hostname, regtype, domain)); + if (options & _HTTP_RESOLVE_STDERR) + { + fputs("STATE: +connecting-to-device\n", stderr); + fprintf(stderr, "DEBUG: Resolving \"%s\", regtype=\"%s\", " + "domain=\"local.\"...\n", hostname, regtype); + } + + uri = NULL; + + if (DNSServiceCreateConnection(&ref) == kDNSServiceErr_NoError) + { + localref = ref; + if (DNSServiceResolve(&localref, kDNSServiceFlagsShareConnection, 0, + hostname, regtype, "local.", http_resolve_cb, + &uribuf) == kDNSServiceErr_NoError) + { + int fds; /* Number of ready descriptors */ + time_t timeout, /* Poll timeout */ + start_time = time(NULL);/* Start time */ + + for (;;) + { + if (options & _HTTP_RESOLVE_STDERR) + _cupsLangPrintFilter(stderr, "INFO", _("Looking for printer.")); + + if (cb && !(*cb)(context)) + { + DEBUG_puts("5_httpResolveURI: callback returned 0 (stop)"); + break; + } + + /* + * For the first minute (or forever if we have a callback), wakeup + * every 2 seconds to emit a "looking for printer" message... + */ + + timeout = (time(NULL) < (start_time + 60) || cb) ? 2000 : -1; + +#ifdef HAVE_POLL + polldata.fd = DNSServiceRefSockFD(ref); + polldata.events = POLLIN; + + fds = poll(&polldata, 1, timeout); + +#else /* select() */ + FD_ZERO(&input_set); + FD_SET(DNSServiceRefSockFD(ref), &input_set); + + stimeout.tv_sec = ((int)timeout) / 1000; + stimeout.tv_usec = ((int)(timeout) * 1000) % 1000000; + + fds = select(DNSServiceRefSockFD(ref)+1, &input_set, NULL, NULL, + timeout < 0.0 ? NULL : &stimeout); +#endif /* HAVE_POLL */ + + if (fds < 0) + { + if (errno != EINTR && errno != EAGAIN) + { + DEBUG_printf(("5_httpResolveURI: poll error: %s", strerror(errno))); + break; + } + } + else if (fds == 0) + { + /* + * Wait 2 seconds for a response to the local resolve; if nothing + * comes in, do an additional domain resolution... + */ + + if (domainsent == 0 && (domain && _cups_strcasecmp(domain, "local."))) + { + if (options & _HTTP_RESOLVE_STDERR) + fprintf(stderr, + "DEBUG: Resolving \"%s\", regtype=\"%s\", " + "domain=\"%s\"...\n", hostname, regtype, + domain ? domain : ""); + + domainref = ref; + if (DNSServiceResolve(&domainref, kDNSServiceFlagsShareConnection, + 0, hostname, regtype, domain, + http_resolve_cb, &uribuf) + == kDNSServiceErr_NoError) + domainsent = 1; + } + + /* + * If it hasn't resolved within 5 seconds set the offline-report + * printer-state-reason... + */ + + if ((options & _HTTP_RESOLVE_STDERR) && offline == 0 && + time(NULL) > (start_time + 5)) + { + fputs("STATE: +offline-report\n", stderr); + offline = 1; + } + } + else + { + if (DNSServiceProcessResult(ref) == kDNSServiceErr_NoError) + { + uri = resolved_uri; + break; + } + } + } + + if (domainsent) + DNSServiceRefDeallocate(domainref); + + DNSServiceRefDeallocate(localref); + } + + DNSServiceRefDeallocate(ref); + } + + if (options & _HTTP_RESOLVE_STDERR) + { + if (uri) + fprintf(stderr, "DEBUG: Resolved as \"%s\"...\n", uri); + else + fputs("DEBUG: Unable to resolve URI\n", stderr); + + fputs("STATE: -connecting-to-device,offline-report\n", stderr); + } + +#else + /* + * No DNS-SD support... + */ + + uri = NULL; +#endif /* HAVE_DNSSD */ + + if ((options & _HTTP_RESOLVE_STDERR) && !uri) + _cupsLangPrintFilter(stderr, "ERROR", _("Unable to find printer.")); + } + else + { + /* + * Nothing more to do... + */ + + strlcpy(resolved_uri, uri, resolved_size); + uri = resolved_uri; + } + + DEBUG_printf(("5_httpResolveURI: Returning \"%s\"", uri)); + + return (uri); +} + + +/* + * 'http_copy_decode()' - Copy and decode a URI. + */ + +static const char * /* O - New source pointer or NULL on error */ +http_copy_decode(char *dst, /* O - Destination buffer */ + const char *src, /* I - Source pointer */ + int dstsize, /* I - Destination size */ + const char *term, /* I - Terminating characters */ + int decode) /* I - Decode %-encoded values */ +{ + char *ptr, /* Pointer into buffer */ + *end; /* End of buffer */ + int quoted; /* Quoted character */ + + + /* + * Copy the src to the destination until we hit a terminating character + * or the end of the string. + */ + + for (ptr = dst, end = dst + dstsize - 1; + *src && (!term || !strchr(term, *src)); + src ++) + if (ptr < end) + { + if (*src == '%' && decode) + { + if (isxdigit(src[1] & 255) && isxdigit(src[2] & 255)) + { + /* + * Grab a hex-encoded character... + */ + + src ++; + if (isalpha(*src)) + quoted = (tolower(*src) - 'a' + 10) << 4; + else + quoted = (*src - '0') << 4; + + src ++; + if (isalpha(*src)) + quoted |= tolower(*src) - 'a' + 10; + else + quoted |= *src - '0'; + + *ptr++ = quoted; + } + else + { + /* + * Bad hex-encoded character... + */ + + *ptr = '\0'; + return (NULL); + } + } + else + *ptr++ = *src; + } + + *ptr = '\0'; + + return (src); +} + + +/* + * 'http_copy_encode()' - Copy and encode a URI. + */ + +static char * /* O - End of current URI */ +http_copy_encode(char *dst, /* O - Destination buffer */ + const char *src, /* I - Source pointer */ + char *dstend, /* I - End of destination buffer */ + const char *reserved, /* I - Extra reserved characters */ + const char *term, /* I - Terminating characters */ + int encode) /* I - %-encode reserved chars? */ +{ + static const char hex[] = "0123456789ABCDEF"; + + + while (*src && dst < dstend) + { + if (term && *src == *term) + return (dst); + + if (encode && (*src == '%' || *src <= ' ' || *src & 128 || + (reserved && strchr(reserved, *src)))) + { + /* + * Hex encode reserved characters... + */ + + if ((dst + 2) >= dstend) + break; + + *dst++ = '%'; + *dst++ = hex[(*src >> 4) & 15]; + *dst++ = hex[*src & 15]; + + src ++; + } + else + *dst++ = *src++; + } + + *dst = '\0'; + + if (*src) + return (NULL); + else + return (dst); +} + + +#ifdef HAVE_DNSSD +/* + * 'http_resolve_cb()' - Build a device URI for the given service name. + */ + +static void DNSSD_API +http_resolve_cb( + DNSServiceRef sdRef, /* I - Service reference */ + DNSServiceFlags flags, /* I - Results flags */ + uint32_t interfaceIndex, /* I - Interface number */ + DNSServiceErrorType errorCode, /* I - Error, if any */ + const char *fullName, /* I - Full service name */ + const char *hostTarget, /* I - Hostname */ + uint16_t port, /* I - Port number */ + uint16_t txtLen, /* I - Length of TXT record */ + const unsigned char *txtRecord, /* I - TXT record data */ + void *context) /* I - Pointer to URI buffer */ +{ + const char *scheme, /* URI scheme */ + *hostptr; /* Pointer into hostTarget */ + char rp[257], /* Remote printer */ + fqdn[256]; /* FQDN of the .local name */ + const void *value; /* Value from TXT record */ + uint8_t valueLen; /* Length of value */ + _http_uribuf_t *uribuf; /* URI buffer */ + + + DEBUG_printf(("7http_resolve_cb(sdRef=%p, flags=%x, interfaceIndex=%u, " + "errorCode=%d, fullName=\"%s\", hostTarget=\"%s\", port=%u, " + "txtLen=%u, txtRecord=%p, context=%p)", sdRef, flags, + interfaceIndex, errorCode, fullName, hostTarget, port, txtLen, + txtRecord, context)); + + uribuf = (_http_uribuf_t *)context; + + /* + * Figure out the scheme from the full name... + */ + + if (strstr(fullName, "._ipps") || strstr(fullName, "._ipp-tls")) + scheme = "ipps"; + else if (strstr(fullName, "._ipp") || strstr(fullName, "._fax-ipp")) + scheme = "ipp"; + else if (strstr(fullName, "._http.")) + scheme = "http"; + else if (strstr(fullName, "._https.")) + scheme = "https"; + else if (strstr(fullName, "._printer.")) + scheme = "lpd"; + else if (strstr(fullName, "._pdl-datastream.")) + scheme = "socket"; + else + scheme = "riousbprint"; + + /* + * Extract the "remote printer" key from the TXT record... + */ + + if ((value = TXTRecordGetValuePtr(txtLen, txtRecord, "rp", + &valueLen)) != NULL) + { + if (((char *)value)[0] == '/') + { + /* + * "rp" value (incorrectly) has a leading slash already... + */ + + memcpy(rp, value, valueLen); + rp[valueLen] = '\0'; + } + else + { + /* + * Convert to resource by concatenating with a leading "/"... + */ + + rp[0] = '/'; + memcpy(rp + 1, value, valueLen); + rp[valueLen + 1] = '\0'; + } + } + else + { + /* + * Default "rp" value is blank, mapping to a path of "/"... + */ + + rp[0] = '/'; + rp[1] = '\0'; + } + + /* + * Lookup the FQDN if needed... + */ + + if ((uribuf->options & _HTTP_RESOLVE_FQDN) && + (hostptr = hostTarget + strlen(hostTarget) - 7) > hostTarget && + !_cups_strcasecmp(hostptr, ".local.")) + { + /* + * OK, we got a .local name but the caller needs a real domain. Start by + * getting the IP address of the .local name and then do reverse-lookups... + */ + + http_addrlist_t *addrlist, /* List of addresses */ + *addr; /* Current address */ + + DEBUG_printf(("8http_resolve_cb: Looking up \"%s\".", hostTarget)); + + snprintf(fqdn, sizeof(fqdn), "%d", ntohs(port)); + if ((addrlist = httpAddrGetList(hostTarget, AF_UNSPEC, fqdn)) != NULL) + { + for (addr = addrlist; addr; addr = addr->next) + { + int error = getnameinfo(&(addr->addr.addr), + httpAddrLength(&(addr->addr)), + fqdn, sizeof(fqdn), NULL, 0, NI_NAMEREQD); + + if (!error) + { + DEBUG_printf(("8http_resolve_cb: Found \"%s\".", fqdn)); + + if ((hostptr = fqdn + strlen(fqdn) - 6) <= fqdn || + _cups_strcasecmp(hostptr, ".local")) + { + hostTarget = fqdn; + break; + } + } +#ifdef DEBUG + else + DEBUG_printf(("8http_resolve_cb: \"%s\" did not resolve: %d", + httpAddrString(&(addr->addr), fqdn, sizeof(fqdn)), + error)); +#endif /* DEBUG */ + } + } + } + + /* + * Assemble the final device URI... + */ + + httpAssembleURI(HTTP_URI_CODING_ALL, uribuf->buffer, uribuf->bufsize, scheme, + NULL, hostTarget, ntohs(port), rp); + + DEBUG_printf(("8http_resolve_cb: Resolved URI is \"%s\"...", uribuf->buffer)); +} +#endif /* HAVE_DNSSD */ + + +/* + * End of "$Id$". + */ diff --git a/cups/http.c b/cups/http.c new file mode 100644 index 0000000000..1e413c0d46 --- /dev/null +++ b/cups/http.c @@ -0,0 +1,4740 @@ +/* + * "$Id$" + * + * HTTP routines for CUPS. + * + * Copyright 2007-2012 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * This file contains Kerberos support code, copyright 2006 by + * Jelmer Vernooij. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * httpAddCredential() - Allocates and adds a single credential to an + * array. + * _httpBIOMethods() - Get the OpenSSL BIO methods for HTTP + * connections. + * httpBlocking() - Set blocking/non-blocking behavior on a + * connection. + * httpCheck() - Check to see if there is a pending response + * from the server. + * httpClearCookie() - Clear the cookie value(s). + * httpClearFields() - Clear HTTP request fields. + * httpClose() - Close an HTTP connection. + * httpConnect() - Connect to a HTTP server. + * httpConnectEncrypt() - Connect to a HTTP server using encryption. + * httpCopyCredentials() - Copy the credentials associated with an + * encrypted connection. + * _httpCreate() - Create an unconnected HTTP connection. + * _httpCreateCredentials() - Create credentials in the internal format. + * httpDelete() - Send a DELETE request to the server. + * _httpDisconnect() - Disconnect a HTTP connection. + * httpEncryption() - Set the required encryption on the link. + * httpError() - Get the last error on a connection. + * httpFlush() - Flush data from a HTTP connection. + * httpFlushWrite() - Flush data in write buffer. + * _httpFreeCredentials() - Free internal credentials. + * httpFreeCredentials() - Free an array of credentials. + * httpGet() - Send a GET request to the server. + * httpGetAuthString() - Get the current authorization string. + * httpGetBlocking() - Get the blocking/non-block state of a + * connection. + * httpGetCookie() - Get any cookie data from the response. + * httpGetFd() - Get the file descriptor associated with a + * connection. + * httpGetField() - Get a field value from a request/response. + * httpGetLength() - Get the amount of data remaining from the + * content-length or transfer-encoding fields. + * httpGetLength2() - Get the amount of data remaining from the + * content-length or transfer-encoding fields. + * httpGets() - Get a line of text from a HTTP connection. + * httpGetState() - Get the current state of the HTTP request. + * httpGetStatus() - Get the status of the last HTTP request. + * httpGetSubField() - Get a sub-field value. + * httpGetSubField2() - Get a sub-field value. + * httpGetVersion() - Get the HTTP version at the other end. + * httpHead() - Send a HEAD request to the server. + * httpInitialize() - Initialize the HTTP interface library and set + * the default HTTP proxy (if any). + * httpOptions() - Send an OPTIONS request to the server. + * _httpPeek() - Peek at data from a HTTP connection. + * httpPost() - Send a POST request to the server. + * httpPrintf() - Print a formatted string to a HTTP connection. + * httpPut() - Send a PUT request to the server. + * httpRead() - Read data from a HTTP connection. + * httpRead2() - Read data from a HTTP connection. + * _httpReadCDSA() - Read function for the CDSA library. + * _httpReadGNUTLS() - Read function for the GNU TLS library. + * httpReconnect() - Reconnect to a HTTP server. + * httpReconnect2() - Reconnect to a HTTP server with timeout and + * optional cancel. + * httpSetAuthString() - Set the current authorization string. + * httpSetCredentials() - Set the credentials associated with an + * encrypted connection. + * httpSetCookie() - Set the cookie value(s). + * httpSetExpect() - Set the Expect: header in a request. + * httpSetField() - Set the value of an HTTP header. + * httpSetLength() - Set the content-length and content-encoding. + * httpSetTimeout() - Set read/write timeouts and an optional + * callback. + * httpTrace() - Send an TRACE request to the server. + * _httpUpdate() - Update the current HTTP status for incoming + * data. + * httpUpdate() - Update the current HTTP state for incoming + * data. + * _httpWait() - Wait for data available on a connection (no + * flush). + * httpWait() - Wait for data available on a connection. + * httpWrite() - Write data to a HTTP connection. + * httpWrite2() - Write data to a HTTP connection. + * _httpWriteCDSA() - Write function for the CDSA library. + * _httpWriteGNUTLS() - Write function for the GNU TLS library. + * http_bio_ctrl() - Control the HTTP connection. + * http_bio_free() - Free OpenSSL data. + * http_bio_new() - Initialize an OpenSSL BIO structure. + * http_bio_puts() - Send a string for OpenSSL. + * http_bio_read() - Read data for OpenSSL. + * http_bio_write() - Write data for OpenSSL. + * http_debug_hex() - Do a hex dump of a buffer. + * http_field() - Return the field index for a field name. + * http_read_ssl() - Read from a SSL/TLS connection. + * http_send() - Send a request with all fields and the trailing + * blank line. + * http_set_credentials() - Set the SSL/TLS credentials. + * http_set_timeout() - Set the socket timeout values. + * http_set_wait() - Set the default wait value for reads. + * http_setup_ssl() - Set up SSL/TLS support on a connection. + * http_shutdown_ssl() - Shut down SSL/TLS on a connection. + * http_upgrade() - Force upgrade to TLS encryption. + * http_write() - Write a buffer to a HTTP connection. + * http_write_chunk() - Write a chunked buffer. + * http_write_ssl() - Write to a SSL/TLS connection. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include +#include +#ifdef WIN32 +# include +#else +# include +# include +# include +#endif /* WIN32 */ +#ifdef HAVE_POLL +# include +#endif /* HAVE_POLL */ + + +/* + * Local functions... + */ + +#ifdef DEBUG +static void http_debug_hex(const char *prefix, const char *buffer, + int bytes); +#endif /* DEBUG */ +static http_field_t http_field(const char *name); +static int http_send(http_t *http, http_state_t request, + const char *uri); +static int http_write(http_t *http, const char *buffer, + int length); +static int http_write_chunk(http_t *http, const char *buffer, + int length); +#ifdef HAVE_SSL +static int http_read_ssl(http_t *http, char *buf, int len); +# if defined(HAVE_CDSASSL) && defined(HAVE_SECCERTIFICATECOPYDATA) +static int http_set_credentials(http_t *http); +# endif /* HAVE_CDSASSL ** HAVE_SECCERTIFICATECOPYDATA */ +#endif /* HAVE_SSL */ +static void http_set_timeout(int fd, double timeout); +static void http_set_wait(http_t *http); +#ifdef HAVE_SSL +static int http_setup_ssl(http_t *http); +static void http_shutdown_ssl(http_t *http); +static int http_upgrade(http_t *http); +static int http_write_ssl(http_t *http, const char *buf, int len); +#endif /* HAVE_SSL */ + + +/* + * Local globals... + */ + +static const char * const http_fields[] = + { + "Accept-Language", + "Accept-Ranges", + "Authorization", + "Connection", + "Content-Encoding", + "Content-Language", + "Content-Length", + "Content-Location", + "Content-MD5", + "Content-Range", + "Content-Type", + "Content-Version", + "Date", + "Host", + "If-Modified-Since", + "If-Unmodified-since", + "Keep-Alive", + "Last-Modified", + "Link", + "Location", + "Range", + "Referer", + "Retry-After", + "Transfer-Encoding", + "Upgrade", + "User-Agent", + "WWW-Authenticate" + }; +#ifdef DEBUG +static const char * const http_states[] = + { + "HTTP_WAITING", + "HTTP_OPTIONS", + "HTTP_GET", + "HTTP_GET_SEND", + "HTTP_HEAD", + "HTTP_POST", + "HTTP_POST_RECV", + "HTTP_POST_SEND", + "HTTP_PUT", + "HTTP_PUT_RECV", + "HTTP_DELETE", + "HTTP_TRACE", + "HTTP_CLOSE", + "HTTP_STATUS" + }; +#endif /* DEBUG */ + + +#if defined(HAVE_SSL) && defined(HAVE_LIBSSL) +/* + * BIO methods for OpenSSL... + */ + +static int http_bio_write(BIO *h, const char *buf, int num); +static int http_bio_read(BIO *h, char *buf, int size); +static int http_bio_puts(BIO *h, const char *str); +static long http_bio_ctrl(BIO *h, int cmd, long arg1, void *arg2); +static int http_bio_new(BIO *h); +static int http_bio_free(BIO *data); + +static BIO_METHOD http_bio_methods = + { + BIO_TYPE_SOCKET, + "http", + http_bio_write, + http_bio_read, + http_bio_puts, + NULL, /* http_bio_gets, */ + http_bio_ctrl, + http_bio_new, + http_bio_free, + NULL, + }; +#endif /* HAVE_SSL && HAVE_LIBSSL */ + + +/* + * 'httpAddCredential()' - Allocates and adds a single credential to an array. + * + * Use @code cupsArrayNew(NULL, NULL)@ to create a credentials array. + * + * @since CUPS 1.5/Mac OS X 10.7@ + */ + +int /* O - 0 on success, -1 on error */ +httpAddCredential( + cups_array_t *credentials, /* I - Credentials array */ + const void *data, /* I - PEM-encoded X.509 data */ + size_t datalen) /* I - Length of data */ +{ + http_credential_t *credential; /* Credential data */ + + + if ((credential = malloc(sizeof(http_credential_t))) != NULL) + { + credential->datalen = datalen; + + if ((credential->data = malloc(datalen)) != NULL) + { + memcpy(credential->data, data, datalen); + cupsArrayAdd(credentials, credential); + return (0); + } + + free(credential); + } + + return (-1); +} + + +#if defined(HAVE_SSL) && defined(HAVE_LIBSSL) +/* + * '_httpBIOMethods()' - Get the OpenSSL BIO methods for HTTP connections. + */ + +BIO_METHOD * /* O - BIO methods for OpenSSL */ +_httpBIOMethods(void) +{ + return (&http_bio_methods); +} +#endif /* HAVE_SSL && HAVE_LIBSSL */ + + +/* + * 'httpBlocking()' - Set blocking/non-blocking behavior on a connection. + */ + +void +httpBlocking(http_t *http, /* I - Connection to server */ + int b) /* I - 1 = blocking, 0 = non-blocking */ +{ + if (http) + { + http->blocking = b; + http_set_wait(http); + } +} + + +/* + * 'httpCheck()' - Check to see if there is a pending response from the server. + */ + +int /* O - 0 = no data, 1 = data available */ +httpCheck(http_t *http) /* I - Connection to server */ +{ + return (httpWait(http, 0)); +} + + +/* + * 'httpClearCookie()' - Clear the cookie value(s). + * + * @since CUPS 1.1.19/Mac OS X 10.3@ + */ + +void +httpClearCookie(http_t *http) /* I - Connection to server */ +{ + if (!http) + return; + + if (http->cookie) + { + free(http->cookie); + http->cookie = NULL; + } +} + + +/* + * 'httpClearFields()' - Clear HTTP request fields. + */ + +void +httpClearFields(http_t *http) /* I - Connection to server */ +{ + if (http) + { + memset(http->fields, 0, sizeof(http->fields)); + if (http->hostname[0] == '/') + httpSetField(http, HTTP_FIELD_HOST, "localhost"); + else + httpSetField(http, HTTP_FIELD_HOST, http->hostname); + + if (http->field_authorization) + { + free(http->field_authorization); + http->field_authorization = NULL; + } + + http->expect = (http_status_t)0; + } +} + + +/* + * 'httpClose()' - Close an HTTP connection. + */ + +void +httpClose(http_t *http) /* I - Connection to server */ +{ +#ifdef HAVE_GSSAPI + OM_uint32 minor_status; /* Minor status code */ +#endif /* HAVE_GSSAPI */ + + + DEBUG_printf(("httpClose(http=%p)", http)); + + /* + * Range check input... + */ + + if (!http) + return; + + /* + * Close any open connection... + */ + + _httpDisconnect(http); + + /* + * Free memory used... + */ + + httpAddrFreeList(http->addrlist); + + if (http->cookie) + free(http->cookie); + +#ifdef HAVE_GSSAPI + if (http->gssctx != GSS_C_NO_CONTEXT) + gss_delete_sec_context(&minor_status, &http->gssctx, GSS_C_NO_BUFFER); + + if (http->gssname != GSS_C_NO_NAME) + gss_release_name(&minor_status, &http->gssname); +#endif /* HAVE_GSSAPI */ + +#ifdef HAVE_AUTHORIZATION_H + if (http->auth_ref) + AuthorizationFree(http->auth_ref, kAuthorizationFlagDefaults); +#endif /* HAVE_AUTHORIZATION_H */ + + httpClearFields(http); + + if (http->authstring && http->authstring != http->_authstring) + free(http->authstring); + + free(http); +} + + +/* + * 'httpConnect()' - Connect to a HTTP server. + * + * This function is deprecated - use @link httpConnectEncrypt@ instead. + * + * @deprecated@ + */ + +http_t * /* O - New HTTP connection */ +httpConnect(const char *host, /* I - Host to connect to */ + int port) /* I - Port number */ +{ + return (httpConnectEncrypt(host, port, HTTP_ENCRYPT_IF_REQUESTED)); +} + + +/* + * 'httpConnectEncrypt()' - Connect to a HTTP server using encryption. + */ + +http_t * /* O - New HTTP connection */ +httpConnectEncrypt( + const char *host, /* I - Host to connect to */ + int port, /* I - Port number */ + http_encryption_t encryption) /* I - Type of encryption to use */ +{ + http_t *http; /* New HTTP connection */ + + + DEBUG_printf(("httpConnectEncrypt(host=\"%s\", port=%d, encryption=%d)", + host, port, encryption)); + + /* + * Create the HTTP structure... + */ + + if ((http = _httpCreate(host, port, NULL, encryption, AF_UNSPEC)) == NULL) + return (NULL); + + /* + * Connect to the remote system... + */ + + if (!httpReconnect(http)) + return (http); + + /* + * Could not connect to any known address - bail out! + */ + + httpAddrFreeList(http->addrlist); + + free(http); + + return (NULL); +} + + +/* + * 'httpCopyCredentials()' - Copy the credentials associated with an encrypted + * connection. + * + * @since CUPS 1.5/Mac OS X 10.7@ + */ + +int /* O - Status of call (0 = success) */ +httpCopyCredentials( + http_t *http, /* I - Connection to server */ + cups_array_t **credentials) /* O - Array of credentials */ +{ +# ifdef HAVE_LIBSSL +# elif defined(HAVE_GNUTLS) +# elif defined(HAVE_CDSASSL) && defined(HAVE_SECCERTIFICATECOPYDATA) + OSStatus error; /* Error code */ + CFIndex count; /* Number of credentials */ + CFArrayRef peerCerts; /* Peer certificates */ + SecCertificateRef secCert; /* Certificate reference */ + CFDataRef data; /* Certificate data */ + int i; /* Looping var */ +# elif defined(HAVE_SSPISSL) +# endif /* HAVE_LIBSSL */ + + + if (credentials) + *credentials = NULL; + + if (!http || !http->tls || !credentials) + return (-1); + +# ifdef HAVE_LIBSSL + return (-1); + +# elif defined(HAVE_GNUTLS) + return (-1); + +# elif defined(HAVE_CDSASSL) && defined(HAVE_SECCERTIFICATECOPYDATA) + if (!(error = SSLCopyPeerCertificates(http->tls, &peerCerts)) && peerCerts) + { + if ((*credentials = cupsArrayNew(NULL, NULL)) != NULL) + { + for (i = 0, count = CFArrayGetCount(peerCerts); i < count; i++) + { + secCert = (SecCertificateRef)CFArrayGetValueAtIndex(peerCerts, i); + if ((data = SecCertificateCopyData(secCert))) + { + httpAddCredential(*credentials, CFDataGetBytePtr(data), + CFDataGetLength(data)); + CFRelease(data); + } + } + } + + CFRelease(peerCerts); + } + + return (error); + +# elif defined(HAVE_SSPISSL) + return (-1); + +# else + return (-1); +# endif /* HAVE_LIBSSL */ +} + + +/* + * '_httpCreate()' - Create an unconnected HTTP connection. + */ + +http_t * /* O - HTTP connection */ +_httpCreate( + const char *host, /* I - Hostname */ + int port, /* I - Port number */ + http_addrlist_t *addrlist, /* I - Address list or NULL */ + http_encryption_t encryption, /* I - Encryption to use */ + int family) /* I - Address family or AF_UNSPEC */ +{ + http_t *http; /* New HTTP connection */ + char service[255]; /* Service name */ + + + DEBUG_printf(("4_httpCreate(host=\"%s\", port=%d, encryption=%d)", + host, port, encryption)); + + if (!host) + return (NULL); + + httpInitialize(); + + /* + * Lookup the host... + */ + + sprintf(service, "%d", port); + + if (!addrlist) + if ((addrlist = httpAddrGetList(host, family, service)) == NULL) + return (NULL); + + /* + * Allocate memory for the structure... + */ + + if ((http = calloc(sizeof(http_t), 1)) == NULL) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(errno), 0); + httpAddrFreeList(addrlist); + return (NULL); + } + + /* + * Initialize the HTTP data... + */ + + http->activity = time(NULL); + http->addrlist = addrlist; + http->blocking = 1; + http->fd = -1; +#ifdef HAVE_GSSAPI + http->gssctx = GSS_C_NO_CONTEXT; + http->gssname = GSS_C_NO_NAME; +#endif /* HAVE_GSSAPI */ + http->version = HTTP_1_1; + + strlcpy(http->hostname, host, sizeof(http->hostname)); + + if (port == 443) /* Always use encryption for https */ + http->encryption = HTTP_ENCRYPT_ALWAYS; + else + http->encryption = encryption; + + http_set_wait(http); + + /* + * Return the new structure... + */ + + return (http); +} + + +/* + * '_httpCreateCredentials()' - Create credentials in the internal format. + */ + +http_tls_credentials_t /* O - Internal credentials */ +_httpCreateCredentials( + cups_array_t *credentials) /* I - Array of credentials */ +{ + if (!credentials) + return (NULL); + +# ifdef HAVE_LIBSSL + return (NULL); + +# elif defined(HAVE_GNUTLS) + return (NULL); + +# elif defined(HAVE_CDSASSL) && defined(HAVE_SECCERTIFICATECOPYDATA) + CFMutableArrayRef peerCerts; /* Peer credentials reference */ + SecCertificateRef secCert; /* Certificate reference */ + CFDataRef data; /* Credential data reference */ + http_credential_t *credential; /* Credential data */ + + + if ((peerCerts = CFArrayCreateMutable(kCFAllocatorDefault, + cupsArrayCount(credentials), + &kCFTypeArrayCallBacks)) == NULL) + return (NULL); + + for (credential = (http_credential_t *)cupsArrayFirst(credentials); + credential; + credential = (http_credential_t *)cupsArrayNext(credentials)) + { + if ((data = CFDataCreate(kCFAllocatorDefault, credential->data, + credential->datalen))) + { + if ((secCert = SecCertificateCreateWithData(kCFAllocatorDefault, data)) + != NULL) + { + CFArrayAppendValue(peerCerts, secCert); + CFRelease(secCert); + } + + CFRelease(data); + } + } + + return (peerCerts); + +# elif defined(HAVE_SSPISSL) + return (NULL); + +# else + return (NULL); +# endif /* HAVE_LIBSSL */ +} + + +/* + * 'httpDelete()' - Send a DELETE request to the server. + */ + +int /* O - Status of call (0 = success) */ +httpDelete(http_t *http, /* I - Connection to server */ + const char *uri) /* I - URI to delete */ +{ + return (http_send(http, HTTP_DELETE, uri)); +} + + +/* + * '_httpDisconnect()' - Disconnect a HTTP connection. + */ + +void +_httpDisconnect(http_t *http) /* I - Connection to server */ +{ +#ifdef HAVE_SSL + if (http->tls) + http_shutdown_ssl(http); +#endif /* HAVE_SSL */ + +#ifdef WIN32 + closesocket(http->fd); +#else + close(http->fd); +#endif /* WIN32 */ + + http->fd = -1; +} + + +/* + * 'httpEncryption()' - Set the required encryption on the link. + */ + +int /* O - -1 on error, 0 on success */ +httpEncryption(http_t *http, /* I - Connection to server */ + http_encryption_t e) /* I - New encryption preference */ +{ + DEBUG_printf(("httpEncryption(http=%p, e=%d)", http, e)); + +#ifdef HAVE_SSL + if (!http) + return (0); + + http->encryption = e; + + if ((http->encryption == HTTP_ENCRYPT_ALWAYS && !http->tls) || + (http->encryption == HTTP_ENCRYPT_NEVER && http->tls)) + return (httpReconnect(http)); + else if (http->encryption == HTTP_ENCRYPT_REQUIRED && !http->tls) + return (http_upgrade(http)); + else + return (0); +#else + if (e == HTTP_ENCRYPT_ALWAYS || e == HTTP_ENCRYPT_REQUIRED) + return (-1); + else + return (0); +#endif /* HAVE_SSL */ +} + + +/* + * 'httpError()' - Get the last error on a connection. + */ + +int /* O - Error code (errno) value */ +httpError(http_t *http) /* I - Connection to server */ +{ + if (http) + return (http->error); + else + return (EINVAL); +} + + +/* + * 'httpFlush()' - Flush data from a HTTP connection. + */ + +void +httpFlush(http_t *http) /* I - Connection to server */ +{ + char buffer[8192]; /* Junk buffer */ + int blocking; /* To block or not to block */ + http_state_t oldstate; /* Old state */ + + + DEBUG_printf(("httpFlush(http=%p), state=%s", http, + http_states[http->state])); + + /* + * Temporarily set non-blocking mode so we don't get stuck in httpRead()... + */ + + blocking = http->blocking; + http->blocking = 0; + + /* + * Read any data we can... + */ + + oldstate = http->state; + while (httpRead2(http, buffer, sizeof(buffer)) > 0); + + /* + * Restore blocking and reset the connection if we didn't get all of + * the remaining data... + */ + + http->blocking = blocking; + + if (http->state == oldstate && http->state != HTTP_WAITING && http->fd >= 0) + { + /* + * Didn't get the data back, so close the current connection. + */ + + http->state = HTTP_WAITING; + +#ifdef HAVE_SSL + if (http->tls) + http_shutdown_ssl(http); +#endif /* HAVE_SSL */ + +#ifdef WIN32 + closesocket(http->fd); +#else + close(http->fd); +#endif /* WIN32 */ + + http->fd = -1; + } +} + + +/* + * 'httpFlushWrite()' - Flush data in write buffer. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - Bytes written or -1 on error */ +httpFlushWrite(http_t *http) /* I - Connection to server */ +{ + int bytes; /* Bytes written */ + + + DEBUG_printf(("httpFlushWrite(http=%p)", http)); + + if (!http || !http->wused) + { + DEBUG_puts(http ? "1httpFlushWrite: Write buffer is empty." : + "1httpFlushWrite: No connection."); + return (0); + } + + if (http->data_encoding == HTTP_ENCODE_CHUNKED) + bytes = http_write_chunk(http, http->wbuffer, http->wused); + else + bytes = http_write(http, http->wbuffer, http->wused); + + http->wused = 0; + + DEBUG_printf(("1httpFlushWrite: Returning %d, errno=%d.", bytes, errno)); + + return (bytes); +} + + +/* + * '_httpFreeCredentials()' - Free internal credentials. + */ + +void +_httpFreeCredentials( + http_tls_credentials_t credentials) /* I - Internal credentials */ +{ + if (!credentials) + return; + +#ifdef HAVE_LIBSSL + (void)credentials; + +#elif defined(HAVE_GNUTLS) + (void)credentials; + +#elif defined(HAVE_CDSASSL) + CFRelease(credentials); + +#elif defined(HAVE_SSPISSL) + (void)credentials; + +#endif /* HAVE_LIBSSL */ +} + + +/* + * 'httpFreeCredentials()' - Free an array of credentials. + */ + +void +httpFreeCredentials( + cups_array_t *credentials) /* I - Array of credentials */ +{ + http_credential_t *credential; /* Credential */ + + + for (credential = (http_credential_t *)cupsArrayFirst(credentials); + credential; + credential = (http_credential_t *)cupsArrayNext(credentials)) + { + cupsArrayRemove(credentials, credential); + free((void *)credential->data); + free(credential); + } + + cupsArrayDelete(credentials); +} + + +/* + * 'httpGet()' - Send a GET request to the server. + */ + +int /* O - Status of call (0 = success) */ +httpGet(http_t *http, /* I - Connection to server */ + const char *uri) /* I - URI to get */ +{ + return (http_send(http, HTTP_GET, uri)); +} + + +/* + * 'httpGetAuthString()' - Get the current authorization string. + * + * The authorization string is set by cupsDoAuthentication() and + * httpSetAuthString(). Use httpGetAuthString() to retrieve the + * string to use with httpSetField() for the HTTP_FIELD_AUTHORIZATION + * value. + * + * @since CUPS 1.3/Mac OS X 10.5@ + */ + +char * /* O - Authorization string */ +httpGetAuthString(http_t *http) /* I - Connection to server */ +{ + if (http) + return (http->authstring); + else + return (NULL); +} + + +/* + * 'httpGetBlocking()' - Get the blocking/non-block state of a connection. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 1 if blocking, 0 if non-blocking */ +httpGetBlocking(http_t *http) /* I - Connection to server */ +{ + return (http ? http->blocking : 0); +} + + +/* + * 'httpGetCookie()' - Get any cookie data from the response. + * + * @since CUPS 1.1.19/Mac OS X 10.3@ + */ + +const char * /* O - Cookie data or NULL */ +httpGetCookie(http_t *http) /* I - HTTP connecion */ +{ + return (http ? http->cookie : NULL); +} + + +/* + * 'httpGetFd()' - Get the file descriptor associated with a connection. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - File descriptor or -1 if none */ +httpGetFd(http_t *http) /* I - Connection to server */ +{ + return (http ? http->fd : -1); +} + + +/* + * 'httpGetField()' - Get a field value from a request/response. + */ + +const char * /* O - Field value */ +httpGetField(http_t *http, /* I - Connection to server */ + http_field_t field) /* I - Field to get */ +{ + if (!http || field <= HTTP_FIELD_UNKNOWN || field >= HTTP_FIELD_MAX) + return (NULL); + else if (field == HTTP_FIELD_AUTHORIZATION && + http->field_authorization) + { + /* + * Special case for WWW-Authenticate: as its contents can be + * longer than HTTP_MAX_VALUE... + */ + + return (http->field_authorization); + } + else + return (http->fields[field]); +} + + +/* + * 'httpGetLength()' - Get the amount of data remaining from the + * content-length or transfer-encoding fields. + * + * This function is deprecated and will not return lengths larger than + * 2^31 - 1; use httpGetLength2() instead. + * + * @deprecated@ + */ + +int /* O - Content length */ +httpGetLength(http_t *http) /* I - Connection to server */ +{ + /* + * Get the read content length and return the 32-bit value. + */ + + if (http) + { + httpGetLength2(http); + + return (http->_data_remaining); + } + else + return (-1); +} + + +/* + * 'httpGetLength2()' - Get the amount of data remaining from the + * content-length or transfer-encoding fields. + * + * This function returns the complete content length, even for + * content larger than 2^31 - 1. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +off_t /* O - Content length */ +httpGetLength2(http_t *http) /* I - Connection to server */ +{ + DEBUG_printf(("2httpGetLength2(http=%p), state=%s", http, + http_states[http->state])); + + if (!http) + return (-1); + + if (!_cups_strcasecmp(http->fields[HTTP_FIELD_TRANSFER_ENCODING], "chunked")) + { + DEBUG_puts("4httpGetLength2: chunked request!"); + + http->data_encoding = HTTP_ENCODE_CHUNKED; + http->data_remaining = 0; + } + else + { + http->data_encoding = HTTP_ENCODE_LENGTH; + + /* + * The following is a hack for HTTP servers that don't send a + * content-length or transfer-encoding field... + * + * If there is no content-length then the connection must close + * after the transfer is complete... + */ + + if (!http->fields[HTTP_FIELD_CONTENT_LENGTH][0]) + { + /* + * Default content length is 0 for errors and 2^31-1 for other + * successful requests... + */ + + if (http->status >= HTTP_MULTIPLE_CHOICES) + http->data_remaining = 0; + else + http->data_remaining = 2147483647; + } + else + http->data_remaining = strtoll(http->fields[HTTP_FIELD_CONTENT_LENGTH], + NULL, 10); + + DEBUG_printf(("4httpGetLength2: content_length=" CUPS_LLFMT, + CUPS_LLCAST http->data_remaining)); + } + + if (http->data_remaining <= INT_MAX) + http->_data_remaining = (int)http->data_remaining; + else + http->_data_remaining = INT_MAX; + + return (http->data_remaining); +} + + +/* + * 'httpGets()' - Get a line of text from a HTTP connection. + */ + +char * /* O - Line or NULL */ +httpGets(char *line, /* I - Line to read into */ + int length, /* I - Max length of buffer */ + http_t *http) /* I - Connection to server */ +{ + char *lineptr, /* Pointer into line */ + *lineend, /* End of line */ + *bufptr, /* Pointer into input buffer */ + *bufend; /* Pointer to end of buffer */ + int bytes, /* Number of bytes read */ + eol; /* End-of-line? */ + + + DEBUG_printf(("2httpGets(line=%p, length=%d, http=%p)", line, length, http)); + + if (http == NULL || line == NULL) + return (NULL); + + /* + * Read a line from the buffer... + */ + + http->error = 0; + lineptr = line; + lineend = line + length - 1; + eol = 0; + + while (lineptr < lineend) + { + /* + * Pre-load the buffer as needed... + */ + +#ifdef WIN32 + WSASetLastError(0); +#else + errno = 0; +#endif /* WIN32 */ + + while (http->used == 0) + { + /* + * No newline; see if there is more data to be read... + */ + + while (!_httpWait(http, http->wait_value, 1)) + { + if (http->timeout_cb && (*http->timeout_cb)(http, http->timeout_data)) + continue; + + DEBUG_puts("3httpGets: Timed out!"); +#ifdef WIN32 + http->error = WSAETIMEDOUT; +#else + http->error = ETIMEDOUT; +#endif /* WIN32 */ + return (NULL); + } + +#ifdef HAVE_SSL + if (http->tls) + bytes = http_read_ssl(http, http->buffer + http->used, + HTTP_MAX_BUFFER - http->used); + else +#endif /* HAVE_SSL */ + bytes = recv(http->fd, http->buffer + http->used, + HTTP_MAX_BUFFER - http->used, 0); + + DEBUG_printf(("4httpGets: read %d bytes...", bytes)); + + if (bytes < 0) + { + /* + * Nope, can't get a line this time... + */ + +#ifdef WIN32 + DEBUG_printf(("3httpGets: recv() error %d!", WSAGetLastError())); + + if (WSAGetLastError() == WSAEINTR) + continue; + else if (WSAGetLastError() == WSAEWOULDBLOCK) + { + if (http->timeout_cb && (*http->timeout_cb)(http, http->timeout_data)) + continue; + + http->error = WSAGetLastError(); + } + else if (WSAGetLastError() != http->error) + { + http->error = WSAGetLastError(); + continue; + } + +#else + DEBUG_printf(("3httpGets: recv() error %d!", errno)); + + if (errno == EINTR) + continue; + else if (errno == EWOULDBLOCK || errno == EAGAIN) + { + if (http->timeout_cb && (*http->timeout_cb)(http, http->timeout_data)) + continue; + else if (!http->timeout_cb && errno == EAGAIN) + continue; + + http->error = errno; + } + else if (errno != http->error) + { + http->error = errno; + continue; + } +#endif /* WIN32 */ + + return (NULL); + } + else if (bytes == 0) + { + http->error = EPIPE; + + return (NULL); + } + + /* + * Yup, update the amount used... + */ + + http->used += bytes; + } + + /* + * Now copy as much of the current line as possible... + */ + + for (bufptr = http->buffer, bufend = http->buffer + http->used; + lineptr < lineend && bufptr < bufend;) + { + if (*bufptr == 0x0a) + { + eol = 1; + bufptr ++; + break; + } + else if (*bufptr == 0x0d) + bufptr ++; + else + *lineptr++ = *bufptr++; + } + + http->used -= (int)(bufptr - http->buffer); + if (http->used > 0) + memmove(http->buffer, bufptr, http->used); + + if (eol) + { + /* + * End of line... + */ + + http->activity = time(NULL); + + *lineptr = '\0'; + + DEBUG_printf(("3httpGets: Returning \"%s\"", line)); + + return (line); + } + } + + DEBUG_puts("3httpGets: No new line available!"); + + return (NULL); +} + + +/* + * 'httpGetState()' - Get the current state of the HTTP request. + */ + +http_state_t /* O - HTTP state */ +httpGetState(http_t *http) /* I - Connection to server */ +{ + return (http ? http->state : HTTP_ERROR); +} + + +/* + * 'httpGetStatus()' - Get the status of the last HTTP request. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +http_status_t /* O - HTTP status */ +httpGetStatus(http_t *http) /* I - Connection to server */ +{ + return (http ? http->status : HTTP_ERROR); +} + + +/* + * 'httpGetSubField()' - Get a sub-field value. + * + * @deprecated@ + */ + +char * /* O - Value or NULL */ +httpGetSubField(http_t *http, /* I - Connection to server */ + http_field_t field, /* I - Field index */ + const char *name, /* I - Name of sub-field */ + char *value) /* O - Value string */ +{ + return (httpGetSubField2(http, field, name, value, HTTP_MAX_VALUE)); +} + + +/* + * 'httpGetSubField2()' - Get a sub-field value. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +char * /* O - Value or NULL */ +httpGetSubField2(http_t *http, /* I - Connection to server */ + http_field_t field, /* I - Field index */ + const char *name, /* I - Name of sub-field */ + char *value, /* O - Value string */ + int valuelen) /* I - Size of value buffer */ +{ + const char *fptr; /* Pointer into field */ + char temp[HTTP_MAX_VALUE], /* Temporary buffer for name */ + *ptr, /* Pointer into string buffer */ + *end; /* End of value buffer */ + + DEBUG_printf(("2httpGetSubField2(http=%p, field=%d, name=\"%s\", value=%p, " + "valuelen=%d)", http, field, name, value, valuelen)); + + if (!http || !name || !value || valuelen < 2 || + field <= HTTP_FIELD_UNKNOWN || field >= HTTP_FIELD_MAX) + return (NULL); + + end = value + valuelen - 1; + + for (fptr = http->fields[field]; *fptr;) + { + /* + * Skip leading whitespace... + */ + + while (_cups_isspace(*fptr)) + fptr ++; + + if (*fptr == ',') + { + fptr ++; + continue; + } + + /* + * Get the sub-field name... + */ + + for (ptr = temp; + *fptr && *fptr != '=' && !_cups_isspace(*fptr) && + ptr < (temp + sizeof(temp) - 1); + *ptr++ = *fptr++); + + *ptr = '\0'; + + DEBUG_printf(("4httpGetSubField2: name=\"%s\"", temp)); + + /* + * Skip trailing chars up to the '='... + */ + + while (_cups_isspace(*fptr)) + fptr ++; + + if (!*fptr) + break; + + if (*fptr != '=') + continue; + + /* + * Skip = and leading whitespace... + */ + + fptr ++; + + while (_cups_isspace(*fptr)) + fptr ++; + + if (*fptr == '\"') + { + /* + * Read quoted string... + */ + + for (ptr = value, fptr ++; + *fptr && *fptr != '\"' && ptr < end; + *ptr++ = *fptr++); + + *ptr = '\0'; + + while (*fptr && *fptr != '\"') + fptr ++; + + if (*fptr) + fptr ++; + } + else + { + /* + * Read unquoted string... + */ + + for (ptr = value; + *fptr && !_cups_isspace(*fptr) && *fptr != ',' && ptr < end; + *ptr++ = *fptr++); + + *ptr = '\0'; + + while (*fptr && !_cups_isspace(*fptr) && *fptr != ',') + fptr ++; + } + + DEBUG_printf(("4httpGetSubField2: value=\"%s\"", value)); + + /* + * See if this is the one... + */ + + if (!strcmp(name, temp)) + { + DEBUG_printf(("3httpGetSubField2: Returning \"%s\"", value)); + return (value); + } + } + + value[0] = '\0'; + + DEBUG_puts("3httpGetSubField2: Returning NULL"); + + return (NULL); +} + + +/* + * 'httpGetVersion()' - Get the HTTP version at the other end. + */ + +http_version_t /* O - Version number */ +httpGetVersion(http_t *http) /* I - Connection to server */ +{ + return (http ? http->version : HTTP_1_0); +} + + +/* + * 'httpHead()' - Send a HEAD request to the server. + */ + +int /* O - Status of call (0 = success) */ +httpHead(http_t *http, /* I - Connection to server */ + const char *uri) /* I - URI for head */ +{ + DEBUG_printf(("httpHead(http=%p, uri=\"%s\")", http, uri)); + return (http_send(http, HTTP_HEAD, uri)); +} + + +/* + * 'httpInitialize()' - Initialize the HTTP interface library and set the + * default HTTP proxy (if any). + */ + +void +httpInitialize(void) +{ + static int initialized = 0; /* Have we been called before? */ +#ifdef WIN32 + WSADATA winsockdata; /* WinSock data */ +#endif /* WIN32 */ +#ifdef HAVE_LIBSSL + int i; /* Looping var */ + unsigned char data[1024]; /* Seed data */ +#endif /* HAVE_LIBSSL */ + + + _cupsGlobalLock(); + if (initialized) + { + _cupsGlobalUnlock(); + return; + } + +#ifdef WIN32 + WSAStartup(MAKEWORD(2,2), &winsockdata); + +#elif !defined(SO_NOSIGPIPE) + /* + * Ignore SIGPIPE signals... + */ + +# ifdef HAVE_SIGSET + sigset(SIGPIPE, SIG_IGN); + +# elif defined(HAVE_SIGACTION) + struct sigaction action; /* POSIX sigaction data */ + + + memset(&action, 0, sizeof(action)); + action.sa_handler = SIG_IGN; + sigaction(SIGPIPE, &action, NULL); + +# else + signal(SIGPIPE, SIG_IGN); +# endif /* !SO_NOSIGPIPE */ +#endif /* WIN32 */ + +#ifdef HAVE_GNUTLS + /* + * Initialize GNU TLS... + */ + + gnutls_global_init(); + +#elif defined(HAVE_LIBSSL) + /* + * Initialize OpenSSL... + */ + + SSL_load_error_strings(); + SSL_library_init(); + + /* + * Using the current time is a dubious random seed, but on some systems + * it is the best we can do (on others, this seed isn't even used...) + */ + + CUPS_SRAND(time(NULL)); + + for (i = 0; i < sizeof(data); i ++) + data[i] = CUPS_RAND(); + + RAND_seed(data, sizeof(data)); +#endif /* HAVE_GNUTLS */ + + initialized = 1; + _cupsGlobalUnlock(); +} + + +/* + * 'httpOptions()' - Send an OPTIONS request to the server. + */ + +int /* O - Status of call (0 = success) */ +httpOptions(http_t *http, /* I - Connection to server */ + const char *uri) /* I - URI for options */ +{ + return (http_send(http, HTTP_OPTIONS, uri)); +} + + +/* + * '_httpPeek()' - Peek at data from a HTTP connection. + * + * This function copies available data from the given HTTP connection, reading + * a buffer as needed. The data is still available for reading using + * @link httpRead@ or @link httpRead2@. + * + * For non-blocking connections the usual timeouts apply. + */ + +ssize_t /* O - Number of bytes copied */ +_httpPeek(http_t *http, /* I - Connection to server */ + char *buffer, /* I - Buffer for data */ + size_t length) /* I - Maximum number of bytes */ +{ + ssize_t bytes; /* Bytes read */ + char len[32]; /* Length string */ + + + DEBUG_printf(("_httpPeek(http=%p, buffer=%p, length=" CUPS_LLFMT ")", + http, buffer, CUPS_LLCAST length)); + + if (http == NULL || buffer == NULL) + return (-1); + + http->activity = time(NULL); + http->error = 0; + + if (length <= 0) + return (0); + + if (http->data_encoding == HTTP_ENCODE_CHUNKED && + http->data_remaining <= 0) + { + DEBUG_puts("2_httpPeek: Getting chunk length..."); + + if (httpGets(len, sizeof(len), http) == NULL) + { + DEBUG_puts("1_httpPeek: Could not get length!"); + return (0); + } + + http->data_remaining = strtoll(len, NULL, 16); + if (http->data_remaining < 0) + { + DEBUG_puts("1_httpPeek: Negative chunk length!"); + return (0); + } + } + + DEBUG_printf(("2_httpPeek: data_remaining=" CUPS_LLFMT, + CUPS_LLCAST http->data_remaining)); + + if (http->data_remaining <= 0) + { + /* + * A zero-length chunk ends a transfer; unless we are reading POST + * data, go idle... + */ + + if (http->data_encoding == HTTP_ENCODE_CHUNKED) + httpGets(len, sizeof(len), http); + + if (http->state == HTTP_POST_RECV) + http->state ++; + else + http->state = HTTP_WAITING; + + /* + * Prevent future reads for this request... + */ + + http->data_encoding = HTTP_ENCODE_LENGTH; + + return (0); + } + else if (length > (size_t)http->data_remaining) + length = (size_t)http->data_remaining; + + if (http->used == 0) + { + /* + * Buffer small reads for better performance... + */ + + if (!http->blocking) + { + while (!httpWait(http, http->wait_value)) + { + if (http->timeout_cb && (*http->timeout_cb)(http, http->timeout_data)) + continue; + + return (0); + } + } + + if (http->data_remaining > sizeof(http->buffer)) + bytes = sizeof(http->buffer); + else + bytes = http->data_remaining; + +#ifdef HAVE_SSL + if (http->tls) + bytes = http_read_ssl(http, http->buffer, bytes); + else +#endif /* HAVE_SSL */ + { + DEBUG_printf(("2_httpPeek: reading %d bytes from socket into buffer...", + (int)bytes)); + + bytes = recv(http->fd, http->buffer, bytes, 0); + + DEBUG_printf(("2_httpPeek: read %d bytes from socket into buffer...", + (int)bytes)); + } + + if (bytes > 0) + http->used = bytes; + else if (bytes < 0) + { +#ifdef WIN32 + if (WSAGetLastError() != WSAEINTR && WSAGetLastError() != WSAEWOULDBLOCK) + { + http->error = WSAGetLastError(); + return (-1); + } +#else + if (errno != EINTR && errno != EAGAIN) + { + http->error = errno; + return (-1); + } +#endif /* WIN32 */ + } + else + { + http->error = EPIPE; + return (0); + } + } + + if (http->used > 0) + { + if (length > (size_t)http->used) + length = (size_t)http->used; + + bytes = (ssize_t)length; + + DEBUG_printf(("2_httpPeek: grabbing %d bytes from input buffer...", + (int)bytes)); + + memcpy(buffer, http->buffer, length); + } + else + bytes = 0; + + if (bytes < 0) + { +#ifdef WIN32 + if (WSAGetLastError() == WSAEINTR || WSAGetLastError() == WSAEWOULDBLOCK) + bytes = 0; + else + http->error = WSAGetLastError(); +#else + if (errno == EINTR || errno == EAGAIN) + bytes = 0; + else + http->error = errno; +#endif /* WIN32 */ + } + else if (bytes == 0) + { + http->error = EPIPE; + return (0); + } + +#ifdef DEBUG + http_debug_hex("_httpPeek", buffer, (int)bytes); +#endif /* DEBUG */ + + return (bytes); +} + + +/* + * 'httpPost()' - Send a POST request to the server. + */ + +int /* O - Status of call (0 = success) */ +httpPost(http_t *http, /* I - Connection to server */ + const char *uri) /* I - URI for post */ +{ + return (http_send(http, HTTP_POST, uri)); +} + + +/* + * 'httpPrintf()' - Print a formatted string to a HTTP connection. + * + * @private@ + */ + +int /* O - Number of bytes written */ +httpPrintf(http_t *http, /* I - Connection to server */ + const char *format, /* I - printf-style format string */ + ...) /* I - Additional args as needed */ +{ + int bytes; /* Number of bytes to write */ + char buf[16384]; /* Buffer for formatted string */ + va_list ap; /* Variable argument pointer */ + + + DEBUG_printf(("2httpPrintf(http=%p, format=\"%s\", ...)", http, format)); + + va_start(ap, format); + bytes = vsnprintf(buf, sizeof(buf), format, ap); + va_end(ap); + + DEBUG_printf(("3httpPrintf: %s", buf)); + + if (http->data_encoding == HTTP_ENCODE_FIELDS) + return (httpWrite2(http, buf, bytes)); + else + { + if (http->wused) + { + DEBUG_puts("4httpPrintf: flushing existing data..."); + + if (httpFlushWrite(http) < 0) + return (-1); + } + + return (http_write(http, buf, bytes)); + } +} + + +/* + * 'httpPut()' - Send a PUT request to the server. + */ + +int /* O - Status of call (0 = success) */ +httpPut(http_t *http, /* I - Connection to server */ + const char *uri) /* I - URI to put */ +{ + DEBUG_printf(("httpPut(http=%p, uri=\"%s\")", http, uri)); + return (http_send(http, HTTP_PUT, uri)); +} + + +/* + * 'httpRead()' - Read data from a HTTP connection. + * + * This function is deprecated. Use the httpRead2() function which can + * read more than 2GB of data. + * + * @deprecated@ + */ + +int /* O - Number of bytes read */ +httpRead(http_t *http, /* I - Connection to server */ + char *buffer, /* I - Buffer for data */ + int length) /* I - Maximum number of bytes */ +{ + return ((int)httpRead2(http, buffer, length)); +} + + +/* + * 'httpRead2()' - Read data from a HTTP connection. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ssize_t /* O - Number of bytes read */ +httpRead2(http_t *http, /* I - Connection to server */ + char *buffer, /* I - Buffer for data */ + size_t length) /* I - Maximum number of bytes */ +{ + ssize_t bytes; /* Bytes read */ + char len[32]; /* Length string */ + + + DEBUG_printf(("httpRead2(http=%p, buffer=%p, length=" CUPS_LLFMT ")", + http, buffer, CUPS_LLCAST length)); + + if (http == NULL || buffer == NULL) + return (-1); + + http->activity = time(NULL); + http->error = 0; + + if (length <= 0) + return (0); + + if (http->data_encoding == HTTP_ENCODE_CHUNKED && + http->data_remaining <= 0) + { + DEBUG_puts("2httpRead2: Getting chunk length..."); + + if (httpGets(len, sizeof(len), http) == NULL) + { + DEBUG_puts("1httpRead2: Could not get length!"); + return (0); + } + + http->data_remaining = strtoll(len, NULL, 16); + if (http->data_remaining < 0) + { + DEBUG_puts("1httpRead2: Negative chunk length!"); + return (0); + } + } + + DEBUG_printf(("2httpRead2: data_remaining=" CUPS_LLFMT, + CUPS_LLCAST http->data_remaining)); + + if (http->data_remaining <= 0) + { + /* + * A zero-length chunk ends a transfer; unless we are reading POST + * data, go idle... + */ + + if (http->data_encoding == HTTP_ENCODE_CHUNKED) + httpGets(len, sizeof(len), http); + + if (http->state == HTTP_POST_RECV) + http->state ++; + else + http->state = HTTP_WAITING; + + /* + * Prevent future reads for this request... + */ + + http->data_encoding = HTTP_ENCODE_LENGTH; + + return (0); + } + else if (length > (size_t)http->data_remaining) + length = (size_t)http->data_remaining; + + if (http->used == 0 && length <= 256) + { + /* + * Buffer small reads for better performance... + */ + + ssize_t buflen; /* Length of read for buffer */ + + if (!http->blocking) + { + while (!httpWait(http, http->wait_value)) + { + if (http->timeout_cb && (*http->timeout_cb)(http, http->timeout_data)) + continue; + + return (0); + } + } + + if (http->data_remaining > sizeof(http->buffer)) + buflen = sizeof(http->buffer); + else + buflen = http->data_remaining; + + DEBUG_printf(("2httpRead2: Reading %d bytes into buffer.", (int)buflen)); + + do + { +#ifdef HAVE_SSL + if (http->tls) + bytes = http_read_ssl(http, http->buffer, buflen); + else +#endif /* HAVE_SSL */ + bytes = recv(http->fd, http->buffer, buflen, 0); + + if (bytes < 0) + { +#ifdef WIN32 + if (WSAGetLastError() != WSAEINTR) + { + http->error = WSAGetLastError(); + return (-1); + } + else if (WSAGetLastError() == WSAEWOULDBLOCK) + { + if (!http->timeout_cb || + !(*http->timeout_cb)(http, http->timeout_data)) + { + http->error = WSAEWOULDBLOCK; + return (-1); + } + } +#else + if (errno == EWOULDBLOCK || errno == EAGAIN) + { + if (http->timeout_cb && !(*http->timeout_cb)(http, http->timeout_data)) + { + http->error = errno; + return (-1); + } + else if (!http->timeout_cb && errno != EAGAIN) + { + http->error = errno; + return (-1); + } + } + else if (errno != EINTR) + { + http->error = errno; + return (-1); + } +#endif /* WIN32 */ + } + } + while (bytes < 0); + + DEBUG_printf(("2httpRead2: Read %d bytes into buffer.", (int)bytes)); + + http->used = bytes; + } + + if (http->used > 0) + { + if (length > (size_t)http->used) + length = (size_t)http->used; + + bytes = (ssize_t)length; + + DEBUG_printf(("2httpRead2: grabbing %d bytes from input buffer...", + (int)bytes)); + + memcpy(buffer, http->buffer, length); + http->used -= (int)length; + + if (http->used > 0) + memmove(http->buffer, http->buffer + length, http->used); + } +#ifdef HAVE_SSL + else if (http->tls) + { + if (!http->blocking) + { + while (!httpWait(http, http->wait_value)) + { + if (http->timeout_cb && (*http->timeout_cb)(http, http->timeout_data)) + continue; + + return (0); + } + } + + while ((bytes = (ssize_t)http_read_ssl(http, buffer, (int)length)) < 0) + { +#ifdef WIN32 + if (WSAGetLastError() == WSAEWOULDBLOCK) + { + if (!http->timeout_cb || !(*http->timeout_cb)(http, http->timeout_data)) + break; + } + else if (WSAGetLastError() != WSAEINTR) + break; +#else + if (errno == EWOULDBLOCK || errno == EAGAIN) + { + if (http->timeout_cb && !(*http->timeout_cb)(http, http->timeout_data)) + break; + else if (!http->timeout_cb && errno != EAGAIN) + break; + } + else if (errno != EINTR) + break; +#endif /* WIN32 */ + } + } +#endif /* HAVE_SSL */ + else + { + if (!http->blocking) + { + while (!httpWait(http, http->wait_value)) + { + if (http->timeout_cb && (*http->timeout_cb)(http, http->timeout_data)) + continue; + + return (0); + } + } + + DEBUG_printf(("2httpRead2: reading " CUPS_LLFMT " bytes from socket...", + CUPS_LLCAST length)); + +#ifdef WIN32 + while ((bytes = (ssize_t)recv(http->fd, buffer, (int)length, 0)) < 0) + { + if (WSAGetLastError() == WSAEWOULDBLOCK) + { + if (!http->timeout_cb || !(*http->timeout_cb)(http, http->timeout_data)) + break; + } + else if (WSAGetLastError() != WSAEINTR) + break; + } +#else + while ((bytes = recv(http->fd, buffer, length, 0)) < 0) + { + if (errno == EWOULDBLOCK || errno == EAGAIN) + { + if (http->timeout_cb && !(*http->timeout_cb)(http, http->timeout_data)) + break; + else if (!http->timeout_cb && errno != EAGAIN) + break; + } + else if (errno != EINTR) + break; + } +#endif /* WIN32 */ + + DEBUG_printf(("2httpRead2: read " CUPS_LLFMT " bytes from socket...", + CUPS_LLCAST bytes)); + } + + if (bytes > 0) + { + http->data_remaining -= bytes; + + if (http->data_remaining <= INT_MAX) + http->_data_remaining = (int)http->data_remaining; + else + http->_data_remaining = INT_MAX; + } + else if (bytes < 0) + { +#ifdef WIN32 + if (WSAGetLastError() == WSAEINTR) + bytes = 0; + else + http->error = WSAGetLastError(); +#else + if (errno == EINTR || (errno == EAGAIN && !http->timeout_cb)) + bytes = 0; + else + http->error = errno; +#endif /* WIN32 */ + } + else + { + http->error = EPIPE; + return (0); + } + + if (http->data_remaining == 0) + { + if (http->data_encoding == HTTP_ENCODE_CHUNKED) + httpGets(len, sizeof(len), http); + + if (http->data_encoding != HTTP_ENCODE_CHUNKED) + { + if (http->state == HTTP_POST_RECV) + http->state ++; + else + http->state = HTTP_WAITING; + } + } + +#ifdef DEBUG + http_debug_hex("httpRead2", buffer, (int)bytes); +#endif /* DEBUG */ + + return (bytes); +} + + +#if defined(HAVE_SSL) && defined(HAVE_CDSASSL) +/* + * '_httpReadCDSA()' - Read function for the CDSA library. + */ + +OSStatus /* O - -1 on error, 0 on success */ +_httpReadCDSA( + SSLConnectionRef connection, /* I - SSL/TLS connection */ + void *data, /* I - Data buffer */ + size_t *dataLength) /* IO - Number of bytes */ +{ + OSStatus result; /* Return value */ + ssize_t bytes; /* Number of bytes read */ + http_t *http; /* HTTP connection */ + + + http = (http_t *)connection; + + if (!http->blocking) + { + /* + * Make sure we have data before we read... + */ + + while (!_httpWait(http, http->wait_value, 0)) + { + if (http->timeout_cb && (*http->timeout_cb)(http, http->timeout_data)) + continue; + + http->error = ETIMEDOUT; + return (-1); + } + } + + do + { + bytes = recv(http->fd, data, *dataLength, 0); + } + while (bytes == -1 && (errno == EINTR || errno == EAGAIN)); + + if (bytes == *dataLength) + { + result = 0; + } + else if (bytes > 0) + { + *dataLength = bytes; + result = errSSLWouldBlock; + } + else + { + *dataLength = 0; + + if (bytes == 0) + result = errSSLClosedGraceful; + else if (errno == EAGAIN) + result = errSSLWouldBlock; + else + result = errSSLClosedAbort; + } + + return (result); +} +#endif /* HAVE_SSL && HAVE_CDSASSL */ + + +#if defined(HAVE_SSL) && defined(HAVE_GNUTLS) +/* + * '_httpReadGNUTLS()' - Read function for the GNU TLS library. + */ + +ssize_t /* O - Number of bytes read or -1 on error */ +_httpReadGNUTLS( + gnutls_transport_ptr ptr, /* I - Connection to server */ + void *data, /* I - Buffer */ + size_t length) /* I - Number of bytes to read */ +{ + http_t *http; /* HTTP connection */ + ssize_t bytes; /* Bytes read */ + + + DEBUG_printf(("6_httpReadGNUTLS(ptr=%p, data=%p, length=%d)", ptr, data, (int)length)); + + http = (http_t *)ptr; + + if (!http->blocking) + { + /* + * Make sure we have data before we read... + */ + + while (!_httpWait(http, http->wait_value, 0)) + { + if (http->timeout_cb && (*http->timeout_cb)(http, http->timeout_data)) + continue; + + http->error = ETIMEDOUT; + return (-1); + } + } + + bytes = recv(http->fd, data, length, 0); + DEBUG_printf(("6_httpReadGNUTLS: bytes=%d", (int)bytes)); + return (bytes); +} +#endif /* HAVE_SSL && HAVE_GNUTLS */ + + +/* + * 'httpReconnect()' - Reconnect to a HTTP server. + */ + +int /* O - 0 on success, non-zero on failure */ +httpReconnect(http_t *http) /* I - Connection to server */ +{ + DEBUG_printf(("httpReconnect(http=%p)", http)); + + return (httpReconnect2(http, 30000, NULL)); +} + + +/* + * 'httpReconnect2()' - Reconnect to a HTTP server with timeout and optional + * cancel. + */ + +int /* O - 0 on success, non-zero on failure */ +httpReconnect2(http_t *http, /* I - Connection to server */ + int msec, /* I - Timeout in milliseconds */ + int *cancel) /* I - Pointer to "cancel" variable */ +{ + http_addrlist_t *addr; /* Connected address */ +#ifdef DEBUG + http_addrlist_t *current; /* Current address */ + char temp[256]; /* Temporary address string */ +#endif /* DEBUG */ + + + DEBUG_printf(("httpReconnect2(http=%p, msec=%d, cancel=%p)", http, msec, + cancel)); + + if (!http) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + return (-1); + } + +#ifdef HAVE_SSL + if (http->tls) + { + DEBUG_puts("2httpReconnect2: Shutting down SSL/TLS..."); + http_shutdown_ssl(http); + } +#endif /* HAVE_SSL */ + + /* + * Close any previously open socket... + */ + + if (http->fd >= 0) + { + DEBUG_printf(("2httpReconnect2: Closing socket %d...", http->fd)); + +#ifdef WIN32 + closesocket(http->fd); +#else + close(http->fd); +#endif /* WIN32 */ + + http->fd = -1; + } + + /* + * Connect to the server... + */ + +#ifdef DEBUG + for (current = http->addrlist; current; current = current->next) + DEBUG_printf(("2httpReconnect2: Address %s:%d", + httpAddrString(&(current->addr), temp, sizeof(temp)), + _httpAddrPort(&(current->addr)))); +#endif /* DEBUG */ + + if ((addr = httpAddrConnect2(http->addrlist, &(http->fd), msec, + cancel)) == NULL) + { + /* + * Unable to connect... + */ + +#ifdef WIN32 + http->error = WSAGetLastError(); +#else + http->error = errno; +#endif /* WIN32 */ + http->status = HTTP_ERROR; + + DEBUG_printf(("1httpReconnect2: httpAddrConnect failed: %s", + strerror(http->error))); + + return (-1); + } + + DEBUG_printf(("2httpReconnect2: New socket=%d", http->fd)); + + if (http->timeout_value > 0) + http_set_timeout(http->fd, http->timeout_value); + + http->hostaddr = &(addr->addr); + http->error = 0; + http->status = HTTP_CONTINUE; + http->state = HTTP_WAITING; + +#ifdef HAVE_SSL + if (http->encryption == HTTP_ENCRYPT_ALWAYS) + { + /* + * Always do encryption via SSL. + */ + + if (http_setup_ssl(http) != 0) + { +# ifdef WIN32 + closesocket(http->fd); +# else + close(http->fd); +# endif /* WIN32 */ + + return (-1); + } + } + else if (http->encryption == HTTP_ENCRYPT_REQUIRED) + return (http_upgrade(http)); +#endif /* HAVE_SSL */ + + DEBUG_printf(("1httpReconnect2: Connected to %s:%d...", + httpAddrString(http->hostaddr, temp, sizeof(temp)), + _httpAddrPort(http->hostaddr))); + + return (0); +} + + +/* + * 'httpSetAuthString()' - Set the current authorization string. + * + * This function just stores a copy of the current authorization string in + * the HTTP connection object. You must still call httpSetField() to set + * HTTP_FIELD_AUTHORIZATION prior to issuing a HTTP request using httpGet(), + * httpHead(), httpOptions(), httpPost, or httpPut(). + * + * @since CUPS 1.3/Mac OS X 10.5@ + */ + +void +httpSetAuthString(http_t *http, /* I - Connection to server */ + const char *scheme, /* I - Auth scheme (NULL to clear it) */ + const char *data) /* I - Auth data (NULL for none) */ +{ + /* + * Range check input... + */ + + if (!http) + return; + + if (http->authstring && http->authstring != http->_authstring) + free(http->authstring); + + http->authstring = http->_authstring; + + if (scheme) + { + /* + * Set the current authorization string... + */ + + int len = (int)strlen(scheme) + (data ? (int)strlen(data) + 1 : 0) + 1; + char *temp; + + if (len > (int)sizeof(http->_authstring)) + { + if ((temp = malloc(len)) == NULL) + len = sizeof(http->_authstring); + else + http->authstring = temp; + } + + if (data) + snprintf(http->authstring, len, "%s %s", scheme, data); + else + strlcpy(http->authstring, scheme, len); + } + else + { + /* + * Clear the current authorization string... + */ + + http->_authstring[0] = '\0'; + } +} + + +/* + * 'httpSetCredentials()' - Set the credentials associated with an encrypted + * connection. + * + * @since CUPS 1.5/Mac OS X 10.7@ + */ + +int /* O - Status of call (0 = success) */ +httpSetCredentials(http_t *http, /* I - Connection to server */ + cups_array_t *credentials) /* I - Array of credentials */ +{ + if (!http || cupsArrayCount(credentials) < 1) + return (-1); + + _httpFreeCredentials(http->tls_credentials); + + http->tls_credentials = _httpCreateCredentials(credentials); + + return (http->tls_credentials ? 0 : -1); +} + + +/* + * 'httpSetCookie()' - Set the cookie value(s). + * + * @since CUPS 1.1.19/Mac OS X 10.3@ + */ + +void +httpSetCookie(http_t *http, /* I - Connection */ + const char *cookie) /* I - Cookie string */ +{ + if (!http) + return; + + if (http->cookie) + free(http->cookie); + + if (cookie) + http->cookie = strdup(cookie); + else + http->cookie = NULL; +} + + +/* + * 'httpSetExpect()' - Set the Expect: header in a request. + * + * Currently only HTTP_CONTINUE is supported for the "expect" argument. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void +httpSetExpect(http_t *http, /* I - Connection to server */ + http_status_t expect) /* I - HTTP status to expect (HTTP_CONTINUE) */ +{ + if (http) + http->expect = expect; +} + + +/* + * 'httpSetField()' - Set the value of an HTTP header. + */ + +void +httpSetField(http_t *http, /* I - Connection to server */ + http_field_t field, /* I - Field index */ + const char *value) /* I - Value */ +{ + if (http == NULL || + field < HTTP_FIELD_ACCEPT_LANGUAGE || + field > HTTP_FIELD_WWW_AUTHENTICATE || + value == NULL) + return; + + strlcpy(http->fields[field], value, HTTP_MAX_VALUE); + + if (field == HTTP_FIELD_AUTHORIZATION) + { + /* + * Special case for Authorization: as its contents can be + * longer than HTTP_MAX_VALUE + */ + + if (http->field_authorization) + free(http->field_authorization); + + http->field_authorization = strdup(value); + } + else if (field == HTTP_FIELD_HOST) + { + /* + * Special-case for Host: as we don't want a trailing "." on the hostname and + * need to bracket IPv6 numeric addresses. + */ + + char *ptr = strchr(value, ':'); + + if (value[0] != '[' && ptr && strchr(ptr + 1, ':')) + { + /* + * Bracket IPv6 numeric addresses... + * + * This is slightly inefficient (basically copying twice), but is an edge + * case and not worth optimizing... + */ + + snprintf(http->fields[HTTP_FIELD_HOST], + sizeof(http->fields[HTTP_FIELD_HOST]), "[%s]", value); + } + else + { + /* + * Check for a trailing dot on the hostname... + */ + + ptr = http->fields[HTTP_FIELD_HOST]; + + if (*ptr) + { + ptr += strlen(ptr) - 1; + + if (*ptr == '.') + *ptr = '\0'; + } + } + } +} + + +/* + * 'httpSetLength()' - Set the content-length and content-encoding. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +void +httpSetLength(http_t *http, /* I - Connection to server */ + size_t length) /* I - Length (0 for chunked) */ +{ + if (!http) + return; + + if (!length) + { + strcpy(http->fields[HTTP_FIELD_TRANSFER_ENCODING], "chunked"); + http->fields[HTTP_FIELD_CONTENT_LENGTH][0] = '\0'; + } + else + { + http->fields[HTTP_FIELD_TRANSFER_ENCODING][0] = '\0'; + snprintf(http->fields[HTTP_FIELD_CONTENT_LENGTH], HTTP_MAX_VALUE, + CUPS_LLFMT, CUPS_LLCAST length); + } +} + + +/* + * 'httpSetTimeout()' - Set read/write timeouts and an optional callback. + * + * The optional timeout callback receives both the HTTP connection and a user + * data pointer and must return 1 to continue or 0 to error (time) out. + * + * @since CUPS 1.5/Mac OS X 10.7@ + */ + +void +httpSetTimeout( + http_t *http, /* I - Connection to server */ + double timeout, /* I - Number of seconds for timeout, + must be greater than 0 */ + http_timeout_cb_t cb, /* I - Callback function or NULL */ + void *user_data) /* I - User data pointer */ +{ + if (!http || timeout <= 0.0) + return; + + http->timeout_cb = cb; + http->timeout_data = user_data; + http->timeout_value = timeout; + + if (http->fd >= 0) + http_set_timeout(http->fd, timeout); + + http_set_wait(http); +} + + +/* + * 'httpTrace()' - Send an TRACE request to the server. + */ + +int /* O - Status of call (0 = success) */ +httpTrace(http_t *http, /* I - Connection to server */ + const char *uri) /* I - URI for trace */ +{ + return (http_send(http, HTTP_TRACE, uri)); +} + + +/* + * '_httpUpdate()' - Update the current HTTP status for incoming data. + * + * Note: Unlike httpUpdate(), this function does not flush pending write data + * and only retrieves a single status line from the HTTP connection. + */ + +int /* O - 1 to continue, 0 to stop */ +_httpUpdate(http_t *http, /* I - Connection to server */ + http_status_t *status) /* O - Current HTTP status */ +{ + char line[32768], /* Line from connection... */ + *value; /* Pointer to value on line */ + http_field_t field; /* Field index */ + int major, minor; /* HTTP version numbers */ + + + DEBUG_printf(("_httpUpdate(http=%p, status=%p), state=%s", http, status, + http_states[http->state])); + + /* + * Grab a single line from the connection... + */ + + if (!httpGets(line, sizeof(line), http)) + { + *status = HTTP_ERROR; + return (0); + } + + DEBUG_printf(("2_httpUpdate: Got \"%s\"", line)); + + if (line[0] == '\0') + { + /* + * Blank line means the start of the data section (if any). Return + * the result code, too... + * + * If we get status 100 (HTTP_CONTINUE), then we *don't* change states. + * Instead, we just return HTTP_CONTINUE to the caller and keep on + * tryin'... + */ + + if (http->status == HTTP_CONTINUE) + { + *status = http->status; + return (0); + } + + if (http->status < HTTP_BAD_REQUEST) + http->digest_tries = 0; + +#ifdef HAVE_SSL + if (http->status == HTTP_SWITCHING_PROTOCOLS && !http->tls) + { + if (http_setup_ssl(http) != 0) + { +# ifdef WIN32 + closesocket(http->fd); +# else + close(http->fd); +# endif /* WIN32 */ + + *status = http->status = HTTP_ERROR; + return (0); + } + + *status = HTTP_CONTINUE; + return (0); + } +#endif /* HAVE_SSL */ + + httpGetLength2(http); + + switch (http->state) + { + case HTTP_GET : + case HTTP_POST : + case HTTP_POST_RECV : + case HTTP_PUT : + http->state ++; + case HTTP_POST_SEND : + case HTTP_HEAD : + break; + + default : + http->state = HTTP_WAITING; + break; + } + + *status = http->status; + return (0); + } + else if (!strncmp(line, "HTTP/", 5)) + { + /* + * Got the beginning of a response... + */ + + int intstatus; /* Status value as an integer */ + + if (sscanf(line, "HTTP/%d.%d%d", &major, &minor, &intstatus) != 3) + { + *status = http->status = HTTP_ERROR; + return (0); + } + + http->version = (http_version_t)(major * 100 + minor); + *status = http->status = (http_status_t)intstatus; + } + else if ((value = strchr(line, ':')) != NULL) + { + /* + * Got a value... + */ + + *value++ = '\0'; + while (_cups_isspace(*value)) + value ++; + + /* + * Be tolerants of servers that send unknown attribute fields... + */ + + if (!_cups_strcasecmp(line, "expect")) + { + /* + * "Expect: 100-continue" or similar... + */ + + http->expect = (http_status_t)atoi(value); + } + else if (!_cups_strcasecmp(line, "cookie")) + { + /* + * "Cookie: name=value[; name=value ...]" - replaces previous cookies... + */ + + httpSetCookie(http, value); + } + else if ((field = http_field(line)) != HTTP_FIELD_UNKNOWN) + httpSetField(http, field, value); +#ifdef DEBUG + else + DEBUG_printf(("1_httpUpdate: unknown field %s seen!", line)); +#endif /* DEBUG */ + } + else + { + DEBUG_printf(("1_httpUpdate: Bad response line \"%s\"!", line)); + *status = http->status = HTTP_ERROR; + return (0); + } + + return (1); +} + + +/* + * 'httpUpdate()' - Update the current HTTP state for incoming data. + */ + +http_status_t /* O - HTTP status */ +httpUpdate(http_t *http) /* I - Connection to server */ +{ + http_status_t status; /* Request status */ + + + DEBUG_printf(("httpUpdate(http=%p), state=%s", http, + http_states[http->state])); + + /* + * Flush pending data, if any... + */ + + if (http->wused) + { + DEBUG_puts("2httpUpdate: flushing buffer..."); + + if (httpFlushWrite(http) < 0) + return (HTTP_ERROR); + } + + /* + * If we haven't issued any commands, then there is nothing to "update"... + */ + + if (http->state == HTTP_WAITING) + return (HTTP_CONTINUE); + + /* + * Grab all of the lines we can from the connection... + */ + + while (_httpUpdate(http, &status)); + + /* + * See if there was an error... + */ + + if (http->error == EPIPE && http->status > HTTP_CONTINUE) + { + DEBUG_printf(("1httpUpdate: Returning status %d...", http->status)); + return (http->status); + } + + if (http->error) + { + DEBUG_printf(("1httpUpdate: socket error %d - %s", http->error, + strerror(http->error))); + http->status = HTTP_ERROR; + return (HTTP_ERROR); + } + + /* + * Return the current status... + */ + + return (status); +} + + +/* + * '_httpWait()' - Wait for data available on a connection (no flush). + */ + +int /* O - 1 if data is available, 0 otherwise */ +_httpWait(http_t *http, /* I - Connection to server */ + int msec, /* I - Milliseconds to wait */ + int usessl) /* I - Use SSL context? */ +{ +#ifdef HAVE_POLL + struct pollfd pfd; /* Polled file descriptor */ +#else + fd_set input_set; /* select() input set */ + struct timeval timeout; /* Timeout */ +#endif /* HAVE_POLL */ + int nfds; /* Result from select()/poll() */ + + + DEBUG_printf(("4_httpWait(http=%p, msec=%d, usessl=%d)", http, msec, usessl)); + + if (http->fd < 0) + { + DEBUG_printf(("5_httpWait: Returning 0 since fd=%d", http->fd)); + return (0); + } + + /* + * Check the SSL/TLS buffers for data first... + */ + +#ifdef HAVE_SSL + if (http->tls && usessl) + { +# ifdef HAVE_LIBSSL + if (SSL_pending(http->tls)) + { + DEBUG_puts("5_httpWait: Return 1 since there is pending SSL data."); + return (1); + } + +# elif defined(HAVE_GNUTLS) + if (gnutls_record_check_pending(http->tls)) + { + DEBUG_puts("5_httpWait: Return 1 since there is pending SSL data."); + return (1); + } + +# elif defined(HAVE_CDSASSL) + size_t bytes; /* Bytes that are available */ + + if (!SSLGetBufferedReadSize(http->tls, &bytes) && + bytes > 0) + { + DEBUG_puts("5_httpWait: Return 1 since there is pending SSL data."); + return (1); + } +# endif /* HAVE_LIBSSL */ + } +#endif /* HAVE_SSL */ + + /* + * Then try doing a select() or poll() to poll the socket... + */ + +#ifdef HAVE_POLL + pfd.fd = http->fd; + pfd.events = POLLIN; + + while ((nfds = poll(&pfd, 1, msec)) < 0 && + (errno == EINTR || errno == EAGAIN)); + +#else + do + { + FD_ZERO(&input_set); + FD_SET(http->fd, &input_set); + + DEBUG_printf(("6_httpWait: msec=%d, http->fd=%d", msec, http->fd)); + + if (msec >= 0) + { + timeout.tv_sec = msec / 1000; + timeout.tv_usec = (msec % 1000) * 1000; + + nfds = select(http->fd + 1, &input_set, NULL, NULL, &timeout); + } + else + nfds = select(http->fd + 1, &input_set, NULL, NULL, NULL); + + DEBUG_printf(("6_httpWait: select() returned %d...", nfds)); + } +# ifdef WIN32 + while (nfds < 0 && (WSAGetLastError() == WSAEINTR || + WSAGetLastError() == WSAEWOULDBLOCK)); +# else + while (nfds < 0 && (errno == EINTR || errno == EAGAIN)); +# endif /* WIN32 */ +#endif /* HAVE_POLL */ + + DEBUG_printf(("5_httpWait: returning with nfds=%d, errno=%d...", nfds, + errno)); + + return (nfds > 0); +} + + +/* + * 'httpWait()' - Wait for data available on a connection. + * + * @since CUPS 1.1.19/Mac OS X 10.3@ + */ + +int /* O - 1 if data is available, 0 otherwise */ +httpWait(http_t *http, /* I - Connection to server */ + int msec) /* I - Milliseconds to wait */ +{ + /* + * First see if there is data in the buffer... + */ + + DEBUG_printf(("2httpWait(http=%p, msec=%d)", http, msec)); + + if (http == NULL) + return (0); + + if (http->used) + { + DEBUG_puts("3httpWait: Returning 1 since there is buffered data ready."); + return (1); + } + + /* + * Flush pending data, if any... + */ + + if (http->wused) + { + DEBUG_puts("3httpWait: Flushing write buffer."); + + if (httpFlushWrite(http) < 0) + return (0); + } + + /* + * If not, check the SSL/TLS buffers and do a select() on the connection... + */ + + return (_httpWait(http, msec, 1)); +} + + +/* + * 'httpWrite()' - Write data to a HTTP connection. + * + * This function is deprecated. Use the httpWrite2() function which can + * write more than 2GB of data. + * + * @deprecated@ + */ + +int /* O - Number of bytes written */ +httpWrite(http_t *http, /* I - Connection to server */ + const char *buffer, /* I - Buffer for data */ + int length) /* I - Number of bytes to write */ +{ + return ((int)httpWrite2(http, buffer, length)); +} + + +/* + * 'httpWrite2()' - Write data to a HTTP connection. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ssize_t /* O - Number of bytes written */ +httpWrite2(http_t *http, /* I - Connection to server */ + const char *buffer, /* I - Buffer for data */ + size_t length) /* I - Number of bytes to write */ +{ + ssize_t bytes; /* Bytes written */ + + + DEBUG_printf(("httpWrite2(http=%p, buffer=%p, length=" CUPS_LLFMT ")", http, + buffer, CUPS_LLCAST length)); + + /* + * Range check input... + */ + + if (http == NULL || buffer == NULL) + return (-1); + + /* + * Mark activity on the connection... + */ + + http->activity = time(NULL); + + /* + * Buffer small writes for better performance... + */ + + if (length > 0) + { + if (http->wused && (length + http->wused) > sizeof(http->wbuffer)) + { + DEBUG_printf(("2httpWrite2: Flushing buffer (wused=%d, length=" + CUPS_LLFMT ")", http->wused, CUPS_LLCAST length)); + + httpFlushWrite(http); + } + + if ((length + http->wused) <= sizeof(http->wbuffer) && + length < sizeof(http->wbuffer)) + { + /* + * Write to buffer... + */ + + DEBUG_printf(("2httpWrite2: Copying " CUPS_LLFMT " bytes to wbuffer...", + CUPS_LLCAST length)); + + memcpy(http->wbuffer + http->wused, buffer, length); + http->wused += (int)length; + bytes = (ssize_t)length; + } + else + { + /* + * Otherwise write the data directly... + */ + + DEBUG_printf(("2httpWrite2: Writing " CUPS_LLFMT " bytes to socket...", + CUPS_LLCAST length)); + + if (http->data_encoding == HTTP_ENCODE_CHUNKED) + bytes = (ssize_t)http_write_chunk(http, buffer, (int)length); + else + bytes = (ssize_t)http_write(http, buffer, (int)length); + + DEBUG_printf(("2httpWrite2: Wrote " CUPS_LLFMT " bytes...", + CUPS_LLCAST bytes)); + } + + if (http->data_encoding == HTTP_ENCODE_LENGTH) + http->data_remaining -= bytes; + } + else + bytes = 0; + + /* + * Handle end-of-request processing... + */ + + if ((http->data_encoding == HTTP_ENCODE_CHUNKED && length == 0) || + (http->data_encoding == HTTP_ENCODE_LENGTH && http->data_remaining == 0)) + { + /* + * Finished with the transfer; unless we are sending POST or PUT + * data, go idle... + */ + + DEBUG_puts("2httpWrite: changing states..."); + + if (http->wused) + httpFlushWrite(http); + + if (http->data_encoding == HTTP_ENCODE_CHUNKED) + { + /* + * Send a 0-length chunk at the end of the request... + */ + + http_write(http, "0\r\n\r\n", 5); + + /* + * Reset the data state... + */ + + http->data_encoding = HTTP_ENCODE_LENGTH; + http->data_remaining = 0; + } + } + + return (bytes); +} + + +#if defined(HAVE_SSL) && defined(HAVE_CDSASSL) +/* + * '_httpWriteCDSA()' - Write function for the CDSA library. + */ + +OSStatus /* O - -1 on error, 0 on success */ +_httpWriteCDSA( + SSLConnectionRef connection, /* I - SSL/TLS connection */ + const void *data, /* I - Data buffer */ + size_t *dataLength) /* IO - Number of bytes */ +{ + OSStatus result; /* Return value */ + ssize_t bytes; /* Number of bytes read */ + http_t *http; /* HTTP connection */ + + + http = (http_t *)connection; + + do + { + bytes = write(http->fd, data, *dataLength); + } + while (bytes == -1 && (errno == EINTR || errno == EAGAIN)); + + if (bytes == *dataLength) + { + result = 0; + } + else if (bytes >= 0) + { + *dataLength = bytes; + result = errSSLWouldBlock; + } + else + { + *dataLength = 0; + + if (errno == EAGAIN) + result = errSSLWouldBlock; + else + result = errSSLClosedAbort; + } + + return (result); +} +#endif /* HAVE_SSL && HAVE_CDSASSL */ + + +#if defined(HAVE_SSL) && defined(HAVE_GNUTLS) +/* + * '_httpWriteGNUTLS()' - Write function for the GNU TLS library. + */ + +ssize_t /* O - Number of bytes written or -1 on error */ +_httpWriteGNUTLS( + gnutls_transport_ptr ptr, /* I - Connection to server */ + const void *data, /* I - Data buffer */ + size_t length) /* I - Number of bytes to write */ +{ + ssize_t bytes; /* Bytes written */ + + + DEBUG_printf(("6_httpWriteGNUTLS(ptr=%p, data=%p, length=%d)", ptr, data, + (int)length)); +#ifdef DEBUG + http_debug_hex("_httpWriteGNUTLS", data, (int)length); +#endif /* DEBUG */ + + bytes = send(((http_t *)ptr)->fd, data, length, 0); + DEBUG_printf(("_httpWriteGNUTLS: bytes=%d", (int)bytes)); + + return (bytes); +} +#endif /* HAVE_SSL && HAVE_GNUTLS */ + + +#if defined(HAVE_SSL) && defined(HAVE_LIBSSL) +/* + * 'http_bio_ctrl()' - Control the HTTP connection. + */ + +static long /* O - Result/data */ +http_bio_ctrl(BIO *h, /* I - BIO data */ + int cmd, /* I - Control command */ + long arg1, /* I - First argument */ + void *arg2) /* I - Second argument */ +{ + switch (cmd) + { + default : + return (0); + + case BIO_CTRL_RESET : + h->ptr = NULL; + return (0); + + case BIO_C_SET_FILE_PTR : + h->ptr = arg2; + h->init = 1; + return (1); + + case BIO_C_GET_FILE_PTR : + if (arg2) + { + *((void **)arg2) = h->ptr; + return (1); + } + else + return (0); + + case BIO_CTRL_DUP : + case BIO_CTRL_FLUSH : + return (1); + } +} + + +/* + * 'http_bio_free()' - Free OpenSSL data. + */ + +static int /* O - 1 on success, 0 on failure */ +http_bio_free(BIO *h) /* I - BIO data */ +{ + if (!h) + return (0); + + if (h->shutdown) + { + h->init = 0; + h->flags = 0; + } + + return (1); +} + + +/* + * 'http_bio_new()' - Initialize an OpenSSL BIO structure. + */ + +static int /* O - 1 on success, 0 on failure */ +http_bio_new(BIO *h) /* I - BIO data */ +{ + if (!h) + return (0); + + h->init = 0; + h->num = 0; + h->ptr = NULL; + h->flags = 0; + + return (1); +} + + +/* + * 'http_bio_puts()' - Send a string for OpenSSL. + */ + +static int /* O - Bytes written */ +http_bio_puts(BIO *h, /* I - BIO data */ + const char *str) /* I - String to write */ +{ +#ifdef WIN32 + return (send(((http_t *)h->ptr)->fd, str, (int)strlen(str), 0)); +#else + return (send(((http_t *)h->ptr)->fd, str, strlen(str), 0)); +#endif /* WIN32 */ +} + + +/* + * 'http_bio_read()' - Read data for OpenSSL. + */ + +static int /* O - Bytes read */ +http_bio_read(BIO *h, /* I - BIO data */ + char *buf, /* I - Buffer */ + int size) /* I - Number of bytes to read */ +{ + http_t *http; /* HTTP connection */ + + + http = (http_t *)h->ptr; + + if (!http->blocking) + { + /* + * Make sure we have data before we read... + */ + + while (!_httpWait(http, http->wait_value, 0)) + { + if (http->timeout_cb && (*http->timeout_cb)(http, http->timeout_data)) + continue; + +#ifdef WIN32 + http->error = WSAETIMEDOUT; +#else + http->error = ETIMEDOUT; +#endif /* WIN32 */ + + return (-1); + } + } + + return (recv(http->fd, buf, size, 0)); +} + + +/* + * 'http_bio_write()' - Write data for OpenSSL. + */ + +static int /* O - Bytes written */ +http_bio_write(BIO *h, /* I - BIO data */ + const char *buf, /* I - Buffer to write */ + int num) /* I - Number of bytes to write */ +{ + return (send(((http_t *)h->ptr)->fd, buf, num, 0)); +} +#endif /* HAVE_SSL && HAVE_LIBSSL */ + + +#ifdef DEBUG +/* + * 'http_debug_hex()' - Do a hex dump of a buffer. + */ + +static void +http_debug_hex(const char *prefix, /* I - Prefix for line */ + const char *buffer, /* I - Buffer to dump */ + int bytes) /* I - Bytes to dump */ +{ + int i, j, /* Looping vars */ + ch; /* Current character */ + char line[255], /* Line buffer */ + *start, /* Start of line after prefix */ + *ptr; /* Pointer into line */ + + + if (_cups_debug_fd < 0 || _cups_debug_level < 6) + return; + + DEBUG_printf(("6%s: %d bytes:", prefix, bytes)); + + snprintf(line, sizeof(line), "6%s: ", prefix); + start = line + strlen(line); + + for (i = 0; i < bytes; i += 16) + { + for (j = 0, ptr = start; j < 16 && (i + j) < bytes; j ++, ptr += 2) + sprintf(ptr, "%02X", buffer[i + j] & 255); + + while (j < 16) + { + strcpy(ptr, " "); + ptr += 2; + j ++; + } + + strcpy(ptr, " "); + ptr += 2; + + for (j = 0; j < 16 && (i + j) < bytes; j ++) + { + ch = buffer[i + j] & 255; + + if (ch < ' ' || ch >= 127) + ch = '.'; + + *ptr++ = ch; + } + + *ptr = '\0'; + DEBUG_puts(line); + } +} +#endif /* DEBUG */ + + +/* + * 'http_field()' - Return the field index for a field name. + */ + +static http_field_t /* O - Field index */ +http_field(const char *name) /* I - String name */ +{ + int i; /* Looping var */ + + + for (i = 0; i < HTTP_FIELD_MAX; i ++) + if (_cups_strcasecmp(name, http_fields[i]) == 0) + return ((http_field_t)i); + + return (HTTP_FIELD_UNKNOWN); +} + + +#ifdef HAVE_SSL +/* + * 'http_read_ssl()' - Read from a SSL/TLS connection. + */ + +static int /* O - Bytes read */ +http_read_ssl(http_t *http, /* I - Connection to server */ + char *buf, /* I - Buffer to store data */ + int len) /* I - Length of buffer */ +{ +# if defined(HAVE_LIBSSL) + return (SSL_read((SSL *)(http->tls), buf, len)); + +# elif defined(HAVE_GNUTLS) + ssize_t result; /* Return value */ + + + result = gnutls_record_recv(http->tls, buf, len); + + if (result < 0 && !errno) + { + /* + * Convert GNU TLS error to errno value... + */ + + switch (result) + { + case GNUTLS_E_INTERRUPTED : + errno = EINTR; + break; + + case GNUTLS_E_AGAIN : + errno = EAGAIN; + break; + + default : + errno = EPIPE; + break; + } + + result = -1; + } + + return ((int)result); + +# elif defined(HAVE_CDSASSL) + int result; /* Return value */ + OSStatus error; /* Error info */ + size_t processed; /* Number of bytes processed */ + + + error = SSLRead(http->tls, buf, len, &processed); + DEBUG_printf(("6http_read_ssl: error=%d, processed=%d", (int)error, + (int)processed)); + switch (error) + { + case 0 : + result = (int)processed; + break; + + case errSSLWouldBlock : + if (processed) + result = (int)processed; + else + { + result = -1; + errno = EINTR; + } + break; + + case errSSLClosedGraceful : + default : + if (processed) + result = (int)processed; + else + { + result = -1; + errno = EPIPE; + } + break; + } + + return (result); + +# elif defined(HAVE_SSPISSL) + return _sspiRead((_sspi_struct_t*) http->tls, buf, len); +# endif /* HAVE_LIBSSL */ +} +#endif /* HAVE_SSL */ + + +/* + * 'http_send()' - Send a request with all fields and the trailing blank line. + */ + +static int /* O - 0 on success, non-zero on error */ +http_send(http_t *http, /* I - Connection to server */ + http_state_t request, /* I - Request code */ + const char *uri) /* I - URI */ +{ + int i; /* Looping var */ + char buf[1024]; /* Encoded URI buffer */ + static const char * const codes[] = + { /* Request code strings */ + NULL, + "OPTIONS", + "GET", + NULL, + "HEAD", + "POST", + NULL, + NULL, + "PUT", + NULL, + "DELETE", + "TRACE", + "CLOSE" + }; + + + DEBUG_printf(("7http_send(http=%p, request=HTTP_%s, uri=\"%s\")", + http, codes[request], uri)); + + if (http == NULL || uri == NULL) + return (-1); + + /* + * Set the User-Agent field if it isn't already... + */ + + if (!http->fields[HTTP_FIELD_USER_AGENT][0]) + httpSetField(http, HTTP_FIELD_USER_AGENT, CUPS_MINIMAL); + + /* + * Encode the URI as needed... + */ + + _httpEncodeURI(buf, uri, sizeof(buf)); + + /* + * See if we had an error the last time around; if so, reconnect... + */ + + if (http->status == HTTP_ERROR || http->status >= HTTP_BAD_REQUEST) + if (httpReconnect(http)) + return (-1); + + /* + * Flush any written data that is pending... + */ + + if (http->wused) + { + if (httpFlushWrite(http) < 0) + if (httpReconnect(http)) + return (-1); + } + + /* + * Send the request header... + */ + + http->state = request; + http->data_encoding = HTTP_ENCODE_FIELDS; + + if (request == HTTP_POST || request == HTTP_PUT) + http->state ++; + + http->status = HTTP_CONTINUE; + +#ifdef HAVE_SSL + if (http->encryption == HTTP_ENCRYPT_REQUIRED && !http->tls) + { + httpSetField(http, HTTP_FIELD_CONNECTION, "Upgrade"); + httpSetField(http, HTTP_FIELD_UPGRADE, "TLS/1.0,SSL/2.0,SSL/3.0"); + } +#endif /* HAVE_SSL */ + + if (httpPrintf(http, "%s %s HTTP/1.1\r\n", codes[request], buf) < 1) + { + http->status = HTTP_ERROR; + return (-1); + } + + for (i = 0; i < HTTP_FIELD_MAX; i ++) + if (http->fields[i][0] != '\0') + { + DEBUG_printf(("9http_send: %s: %s", http_fields[i], + httpGetField(http, i))); + + if (httpPrintf(http, "%s: %s\r\n", http_fields[i], + httpGetField(http, i)) < 1) + { + http->status = HTTP_ERROR; + return (-1); + } + } + + if (http->cookie) + if (httpPrintf(http, "Cookie: $Version=0; %s\r\n", http->cookie) < 1) + { + http->status = HTTP_ERROR; + return (-1); + } + + if (http->expect == HTTP_CONTINUE && + (http->state == HTTP_POST_RECV || http->state == HTTP_PUT_RECV)) + if (httpPrintf(http, "Expect: 100-continue\r\n") < 1) + { + http->status = HTTP_ERROR; + return (-1); + } + + if (httpPrintf(http, "\r\n") < 1) + { + http->status = HTTP_ERROR; + return (-1); + } + + if (httpFlushWrite(http) < 0) + return (-1); + + httpGetLength2(http); + httpClearFields(http); + + /* + * The Kerberos and AuthRef authentication strings can only be used once... + */ + + if (http->field_authorization && http->authstring && + (!strncmp(http->authstring, "Negotiate", 9) || + !strncmp(http->authstring, "AuthRef", 7))) + { + http->_authstring[0] = '\0'; + + if (http->authstring != http->_authstring) + free(http->authstring); + + http->authstring = http->_authstring; + } + + return (0); +} + + +#ifdef HAVE_SSL +# if defined(HAVE_CDSASSL) && defined(HAVE_SECCERTIFICATECOPYDATA) +/* + * 'http_set_credentials()' - Set the SSL/TLS credentials. + */ + +static int /* O - Status of connection */ +http_set_credentials(http_t *http) /* I - Connection to server */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + OSStatus error = 0; /* Error code */ + http_tls_credentials_t credentials = NULL; + /* TLS credentials */ + + + DEBUG_printf(("7http_set_credentials(%p)", http)); + + /* + * Prefer connection specific credentials... + */ + + if ((credentials = http->tls_credentials) == NULL) + credentials = cg->tls_credentials; + +# if HAVE_SECPOLICYCREATESSL + /* + * Otherwise root around in the user's keychain to see if one can be found... + */ + + if (!credentials) + { + CFDictionaryRef query; /* Query dictionary */ + CFTypeRef matches = NULL; /* Matching credentials */ + CFArrayRef dn_array = NULL;/* Distinguished names array */ + CFTypeRef keys[] = { kSecClass, + kSecMatchLimit, + kSecReturnRef }; + /* Keys for dictionary */ + CFTypeRef values[] = { kSecClassCertificate, + kSecMatchLimitOne, + kCFBooleanTrue }; + /* Values for dictionary */ + + /* + * Get the names associated with the server. + */ + + if ((error = SSLCopyDistinguishedNames(http->tls, &dn_array)) != noErr) + { + DEBUG_printf(("4http_set_credentials: SSLCopyDistinguishedNames, error=%d", + (int)error)); + return (error); + } + + /* + * Create a query which will return all identities that can sign and match + * the passed in policy. + */ + + query = CFDictionaryCreate(NULL, + (const void**)(&keys[0]), + (const void**)(&values[0]), + sizeof(keys) / sizeof(keys[0]), + &kCFTypeDictionaryKeyCallBacks, + &kCFTypeDictionaryValueCallBacks); + if (query) + { + error = SecItemCopyMatching(query, &matches); + DEBUG_printf(("4http_set_credentials: SecItemCopyMatching, error=%d", + (int)error)); + CFRelease(query); + } + + if (matches) + CFRelease(matches); + + if (dn_array) + CFRelease(dn_array); + } +# endif /* HAVE_SECPOLICYCREATESSL */ + + if (credentials) + { + error = SSLSetCertificate(http->tls, credentials); + DEBUG_printf(("4http_set_credentials: SSLSetCertificate, error=%d", + (int)error)); + } + else + DEBUG_puts("4http_set_credentials: No credentials to set."); + + return (error); +} +# endif /* HAVE_CDSASSL && HAVE_SECCERTIFICATECOPYDATA */ +#endif /* HAVE_SSL */ + + +/* + * 'http_set_timeout()' - Set the socket timeout values. + */ + +static void +http_set_timeout(int fd, /* I - File descriptor */ + double timeout) /* I - Timeout in seconds */ +{ +#ifdef WIN32 + DWORD tv = (DWORD)(timeout * 1000); + /* Timeout in milliseconds */ + + setsockopt(fd, SOL_SOCKET, SO_RCVTIMEO, (char *)&tv, sizeof(tv)); + setsockopt(fd, SOL_SOCKET, SO_SNDTIMEO, (char *)&tv, sizeof(tv)); + +#else + struct timeval tv; /* Timeout in secs and usecs */ + + tv.tv_sec = (int)timeout; + tv.tv_usec = (int)(1000000 * fmod(timeout, 1.0)); + + setsockopt(fd, SOL_SOCKET, SO_RCVTIMEO, &tv, sizeof(tv)); + setsockopt(fd, SOL_SOCKET, SO_SNDTIMEO, &tv, sizeof(tv)); +#endif /* WIN32 */ +} + + +/* + * 'http_set_wait()' - Set the default wait value for reads. + */ + +static void +http_set_wait(http_t *http) /* I - Connection to server */ +{ + if (http->blocking) + { + http->wait_value = (int)(http->timeout_value * 1000); + + if (http->wait_value <= 0) + http->wait_value = 60000; + } + else + http->wait_value = 10000; +} + + +#ifdef HAVE_SSL +/* + * 'http_setup_ssl()' - Set up SSL/TLS support on a connection. + */ + +static int /* O - 0 on success, -1 on failure */ +http_setup_ssl(http_t *http) /* I - Connection to server */ +{ + _cups_globals_t *cg = _cupsGlobals(); + /* Pointer to library globals */ + int any_root; /* Allow any root */ + char hostname[256], /* Hostname */ + *hostptr; /* Pointer into hostname */ + +# ifdef HAVE_LIBSSL + SSL_CTX *context; /* Context for encryption */ + BIO *bio; /* BIO data */ + const char *message = NULL;/* Error message */ +# elif defined(HAVE_GNUTLS) + int status; /* Status of handshake */ + gnutls_certificate_client_credentials *credentials; + /* TLS credentials */ +# elif defined(HAVE_CDSASSL) + OSStatus error; /* Error code */ + const char *message = NULL;/* Error message */ +# ifdef HAVE_SECCERTIFICATECOPYDATA + cups_array_t *credentials; /* Credentials array */ + cups_array_t *names; /* CUPS distinguished names */ + CFArrayRef dn_array; /* CF distinguished names array */ + CFIndex count; /* Number of credentials */ + CFDataRef data; /* Certificate data */ + int i; /* Looping var */ + http_credential_t *credential; /* Credential data */ +# endif /* HAVE_SECCERTIFICATECOPYDATA */ +# elif defined(HAVE_SSPISSL) + TCHAR username[256]; /* Username returned from GetUserName() */ + TCHAR commonName[256];/* Common name for certificate */ + DWORD dwSize; /* 32 bit size */ +# endif /* HAVE_LIBSSL */ + + + DEBUG_printf(("7http_setup_ssl(http=%p)", http)); + + /* + * Always allow self-signed certificates for the local loopback address... + */ + + if (httpAddrLocalhost(http->hostaddr)) + { + any_root = 1; + strlcpy(hostname, "localhost", sizeof(hostname)); + } + else + { + /* + * Otherwise use the system-wide setting and make sure the hostname we have + * does not end in a trailing dot. + */ + + any_root = cg->any_root; + + strlcpy(hostname, http->hostname, sizeof(hostname)); + if ((hostptr = hostname + strlen(hostname) - 1) >= hostname && + *hostptr == '.') + *hostptr = '\0'; + } + +# ifdef HAVE_LIBSSL + (void)any_root; + + context = SSL_CTX_new(SSLv23_client_method()); + + SSL_CTX_set_options(context, SSL_OP_NO_SSLv2); /* Only use SSLv3 or TLS */ + + bio = BIO_new(_httpBIOMethods()); + BIO_ctrl(bio, BIO_C_SET_FILE_PTR, 0, (char *)http); + + http->tls = SSL_new(context); + SSL_set_bio(http->tls, bio, bio); + + SSL_set_tlsext_host_name(http->tls, hostname); + + if (SSL_connect(http->tls) != 1) + { + unsigned long error; /* Error code */ + + while ((error = ERR_get_error()) != 0) + { + message = ERR_error_string(error, NULL); + DEBUG_printf(("8http_setup_ssl: %s", message)); + } + + SSL_CTX_free(context); + SSL_free(http->tls); + http->tls = NULL; + +# ifdef WIN32 + http->error = WSAGetLastError(); +# else + http->error = errno; +# endif /* WIN32 */ + http->status = HTTP_ERROR; + + if (!message) + message = _("Unable to establish a secure connection to host."); + + _cupsSetError(IPP_PKI_ERROR, message, 1); + + return (-1); + } + +# elif defined(HAVE_GNUTLS) + (void)any_root; + + credentials = (gnutls_certificate_client_credentials *) + malloc(sizeof(gnutls_certificate_client_credentials)); + if (credentials == NULL) + { + DEBUG_printf(("8http_setup_ssl: Unable to allocate credentials: %s", + strerror(errno))); + http->error = errno; + http->status = HTTP_ERROR; + _cupsSetHTTPError(HTTP_ERROR); + + return (-1); + } + + gnutls_certificate_allocate_credentials(credentials); + + gnutls_init(&http->tls, GNUTLS_CLIENT); + gnutls_set_default_priority(http->tls); + gnutls_server_name_set(http->tls, GNUTLS_NAME_DNS, hostname, + strlen(hostname)); + gnutls_credentials_set(http->tls, GNUTLS_CRD_CERTIFICATE, *credentials); + gnutls_transport_set_ptr(http->tls, (gnutls_transport_ptr)http); + gnutls_transport_set_pull_function(http->tls, _httpReadGNUTLS); + gnutls_transport_set_push_function(http->tls, _httpWriteGNUTLS); + + while ((status = gnutls_handshake(http->tls)) != GNUTLS_E_SUCCESS) + { + DEBUG_printf(("8http_setup_ssl: gnutls_handshake returned %d (%s)", + status, gnutls_strerror(status))); + + if (gnutls_error_is_fatal(status)) + { + http->error = EIO; + http->status = HTTP_ERROR; + + _cupsSetError(IPP_PKI_ERROR, gnutls_strerror(status), 0); + + gnutls_deinit(http->tls); + gnutls_certificate_free_credentials(*credentials); + free(credentials); + http->tls = NULL; + + return (-1); + } + } + + http->tls_credentials = credentials; + +# elif defined(HAVE_CDSASSL) + if ((error = SSLNewContext(false, &http->tls))) + { + http->error = errno; + http->status = HTTP_ERROR; + _cupsSetHTTPError(HTTP_ERROR); + + return (-1); + } + + error = SSLSetConnection(http->tls, http); + DEBUG_printf(("4http_setup_ssl: SSLSetConnection, error=%d", (int)error)); + + if (!error) + { + error = SSLSetIOFuncs(http->tls, _httpReadCDSA, _httpWriteCDSA); + DEBUG_printf(("4http_setup_ssl: SSLSetIOFuncs, error=%d", (int)error)); + } + + if (!error) + { + error = SSLSetAllowsAnyRoot(http->tls, any_root); + DEBUG_printf(("4http_setup_ssl: SSLSetAllowsAnyRoot(%d), error=%d", + any_root, (int)error)); + } + + if (!error) + { + error = SSLSetAllowsExpiredCerts(http->tls, cg->expired_certs); + DEBUG_printf(("4http_setup_ssl: SSLSetAllowsExpiredCerts(%d), error=%d", + cg->expired_certs, (int)error)); + } + + if (!error) + { + error = SSLSetAllowsExpiredRoots(http->tls, cg->expired_root); + DEBUG_printf(("4http_setup_ssl: SSLSetAllowsExpiredRoots(%d), error=%d", + cg->expired_root, (int)error)); + } + +# ifdef HAVE_SSLSETPROTOCOLVERSIONMAX + if (!error) + { + error = SSLSetProtocolVersionMax(http->tls, kTLSProtocol1); + DEBUG_printf(("4http_setup_ssl: SSLSetProtocolVersionMax(kTLSProtocol1), " + "error=%d", (int)error)); + } +# endif /* HAVE_SSLSETPROTOCOLVERSIONMAX */ + + /* + * In general, don't verify certificates since things like the common name + * often do not match... + */ + + if (!error) + { + error = SSLSetEnableCertVerify(http->tls, false); + DEBUG_printf(("4http_setup_ssl: SSLSetEnableCertVerify, error=%d", + (int)error)); + } + +# ifdef HAVE_SECCERTIFICATECOPYDATA + if (!error) + { + if (cg->client_cert_cb) + { + error = SSLSetSessionOption(http->tls, + kSSLSessionOptionBreakOnCertRequested, true); + DEBUG_printf(("4http_setup_ssl: kSSLSessionOptionBreakOnCertRequested, " + "error=%d", (int)error)); + } + else + { + error = http_set_credentials(http); + DEBUG_printf(("4http_setup_ssl: http_set_credentials, error=%d", + (int)error)); + } + } + + /* + * If there's a server certificate callback installed let it evaluate the + * certificate(s) during the handshake... + */ + + if (!error && cg->server_cert_cb != NULL) + { + error = SSLSetSessionOption(http->tls, + kSSLSessionOptionBreakOnServerAuth, true); + DEBUG_printf(("4http_setup_ssl: kSSLSessionOptionBreakOnServerAuth, " + "error=%d", (int)error)); + } +# endif /* HAVE_SECCERTIFICATECOPYDATA */ + + /* + * Let the server know which hostname/domain we are trying to connect to + * in case it wants to serve up a certificate with a matching common name. + */ + + if (!error) + { + error = SSLSetPeerDomainName(http->tls, hostname, strlen(hostname)); + + DEBUG_printf(("4http_setup_ssl: SSLSetPeerDomainName, error=%d", + (int)error)); + } + + if (!error) + { + int done = 0; /* Are we done yet? */ + + while (!error && !done) + { + error = SSLHandshake(http->tls); + + DEBUG_printf(("4http_setup_ssl: SSLHandshake returned %d.", (int)error)); + + switch (error) + { + case noErr : + done = 1; + break; + + case errSSLWouldBlock : + usleep(1000); + break; + +# ifdef HAVE_SECCERTIFICATECOPYDATA + case errSSLServerAuthCompleted : + error = 0; + if (cg->server_cert_cb) + { + error = httpCopyCredentials(http, &credentials); + if (!error) + { + error = (cg->server_cert_cb)(http, http->tls, credentials, + cg->server_cert_data); + httpFreeCredentials(credentials); + } + + DEBUG_printf(("4http_setup_ssl: Server certificate callback " + "returned %d.", (int)error)); + } + break; + + case errSSLClientCertRequested : + error = 0; + + if (cg->client_cert_cb) + { + names = NULL; + if (!(error = SSLCopyDistinguishedNames(http->tls, &dn_array)) && + dn_array) + { + if ((names = cupsArrayNew(NULL, NULL)) != NULL) + { + for (i = 0, count = CFArrayGetCount(dn_array); i < count; i++) + { + data = (CFDataRef)CFArrayGetValueAtIndex(dn_array, i); + + if ((credential = malloc(sizeof(*credential)))) + { + credential->datalen = CFDataGetLength(data); + if ((credential->data = malloc(credential->datalen))) + { + memcpy((void *)credential->data, CFDataGetBytePtr(data), + credential->datalen); + cupsArrayAdd(names, credential); + } + } + } + } + + CFRelease(dn_array); + } + + if (!error) + { + error = (cg->client_cert_cb)(http, http->tls, names, + cg->client_cert_data); + + DEBUG_printf(("4http_setup_ssl: Client certificate callback " + "returned %d.", (int)error)); + } + + httpFreeCredentials(names); + } + break; +# endif /* HAVE_SECCERTIFICATECOPYDATA */ + + case errSSLUnknownRootCert : + message = _("Unable to establish a secure connection to host " + "(untrusted certificate)."); + break; + + case errSSLNoRootCert : + message = _("Unable to establish a secure connection to host " + "(self-signed certificate)."); + break; + + case errSSLCertExpired : + message = _("Unable to establish a secure connection to host " + "(expired certificate)."); + break; + + case errSSLCertNotYetValid : + message = _("Unable to establish a secure connection to host " + "(certificate not yet valid)."); + break; + + case errSSLHostNameMismatch : + message = _("Unable to establish a secure connection to host " + "(host name mismatch)."); + break; + + case errSSLXCertChainInvalid : + message = _("Unable to establish a secure connection to host " + "(certificate chain invalid)."); + break; + + case errSSLConnectionRefused : + message = _("Unable to establish a secure connection to host " + "(peer dropped connection before responding)."); + break; + + default : + break; + } + } + } + + if (error) + { + http->error = error; + http->status = HTTP_ERROR; + errno = ECONNREFUSED; + + SSLDisposeContext(http->tls); + http->tls = NULL; + + /* + * If an error string wasn't set by the callbacks use a generic one... + */ + + if (!message) +#ifdef HAVE_CSSMERRORSTRING + message = cssmErrorString(error); +#else + message = _("Unable to establish a secure connection to host."); +#endif /* HAVE_CSSMERRORSTRING */ + + _cupsSetError(IPP_PKI_ERROR, message, 1); + + return (-1); + } + +# elif defined(HAVE_SSPISSL) + http->tls = _sspiAlloc(); + + if (!http->tls) + { + _cupsSetHTTPError(HTTP_ERROR); + return (-1); + } + + http->tls->sock = http->fd; + dwSize = sizeof(username) / sizeof(TCHAR); + GetUserName(username, &dwSize); + _sntprintf_s(commonName, sizeof(commonName) / sizeof(TCHAR), + sizeof(commonName) / sizeof(TCHAR), TEXT("CN=%s"), username); + + if (!_sspiGetCredentials(http->tls_credentials, L"ClientContainer", + commonName, FALSE)) + { + _sspiFree(http->tls_credentials); + http->tls_credentials = NULL; + + http->error = EIO; + http->status = HTTP_ERROR; + + _cupsSetError(IPP_PKI_ERROR, + _("Unable to establish a secure connection to host."), 1); + + return (-1); + } + + _sspiSetAllowsAnyRoot(http->tls_credentials, any_root); + _sspiSetAllowsExpiredCerts(http->tls_credentials, TRUE); + + if (!_sspiConnect(http->tls_credentials, hostname)) + { + _sspiFree(http->tls_credentials); + http->tls_credentials = NULL; + + http->error = EIO; + http->status = HTTP_ERROR; + + _cupsSetError(IPP_PKI_ERROR, + _("Unable to establish a secure connection to host."), 1); + + return (-1); + } +# endif /* HAVE_CDSASSL */ + + return (0); +} + + +/* + * 'http_shutdown_ssl()' - Shut down SSL/TLS on a connection. + */ + +static void +http_shutdown_ssl(http_t *http) /* I - Connection to server */ +{ +# ifdef HAVE_LIBSSL + SSL_CTX *context; /* Context for encryption */ + + context = SSL_get_SSL_CTX(http->tls); + + SSL_shutdown(http->tls); + SSL_CTX_free(context); + SSL_free(http->tls); + +# elif defined(HAVE_GNUTLS) + gnutls_certificate_client_credentials *credentials; + /* TLS credentials */ + + credentials = (gnutls_certificate_client_credentials *)(http->tls_credentials); + + gnutls_bye(http->tls, GNUTLS_SHUT_RDWR); + gnutls_deinit(http->tls); + gnutls_certificate_free_credentials(*credentials); + free(credentials); + +# elif defined(HAVE_CDSASSL) + while (SSLClose(http->tls) == errSSLWouldBlock) + usleep(1000); + + SSLDisposeContext(http->tls); + + if (http->tls_credentials) + CFRelease(http->tls_credentials); + +# elif defined(HAVE_SSPISSL) + _sspiFree(http->tls_credentials); +# endif /* HAVE_LIBSSL */ + + http->tls = NULL; + http->tls_credentials = NULL; +} +#endif /* HAVE_SSL */ + + +#ifdef HAVE_SSL +/* + * 'http_upgrade()' - Force upgrade to TLS encryption. + */ + +static int /* O - Status of connection */ +http_upgrade(http_t *http) /* I - Connection to server */ +{ + int ret; /* Return value */ + http_t myhttp; /* Local copy of HTTP data */ + + + DEBUG_printf(("7http_upgrade(%p)", http)); + + /* + * Flush the connection to make sure any previous "Upgrade" message + * has been read. + */ + + httpFlush(http); + + /* + * Copy the HTTP data to a local variable so we can do the OPTIONS + * request without interfering with the existing request data... + */ + + memcpy(&myhttp, http, sizeof(myhttp)); + + /* + * Send an OPTIONS request to the server, requiring SSL or TLS + * encryption on the link... + */ + + http->field_authorization = NULL; /* Don't free the auth string */ + + httpClearFields(http); + httpSetField(http, HTTP_FIELD_CONNECTION, "upgrade"); + httpSetField(http, HTTP_FIELD_UPGRADE, "TLS/1.2, TLS/1.1, TLS/1.0, SSL/3.0"); + + if ((ret = httpOptions(http, "*")) == 0) + { + /* + * Wait for the secure connection... + */ + + while (httpUpdate(http) == HTTP_CONTINUE); + } + + /* + * Restore the HTTP request data... + */ + + memcpy(http->fields, myhttp.fields, sizeof(http->fields)); + http->data_encoding = myhttp.data_encoding; + http->data_remaining = myhttp.data_remaining; + http->_data_remaining = myhttp._data_remaining; + http->expect = myhttp.expect; + http->field_authorization = myhttp.field_authorization; + http->digest_tries = myhttp.digest_tries; + + /* + * See if we actually went secure... + */ + + if (!http->tls) + { + /* + * Server does not support HTTP upgrade... + */ + + DEBUG_puts("8http_upgrade: Server does not support HTTP upgrade!"); + +# ifdef WIN32 + closesocket(http->fd); +# else + close(http->fd); +# endif + + http->fd = -1; + + return (-1); + } + else + return (ret); +} +#endif /* HAVE_SSL */ + + +/* + * 'http_write()' - Write a buffer to a HTTP connection. + */ + +static int /* O - Number of bytes written */ +http_write(http_t *http, /* I - Connection to server */ + const char *buffer, /* I - Buffer for data */ + int length) /* I - Number of bytes to write */ +{ + int tbytes, /* Total bytes sent */ + bytes; /* Bytes sent */ + + + DEBUG_printf(("2http_write(http=%p, buffer=%p, length=%d)", http, buffer, + length)); + http->error = 0; + tbytes = 0; + + while (length > 0) + { + if (http->timeout_cb) + { +#ifdef HAVE_POLL + struct pollfd pfd; /* Polled file descriptor */ +#else + fd_set output_set; /* Output ready for write? */ + struct timeval timeout; /* Timeout value */ +#endif /* HAVE_POLL */ + int nfds; /* Result from select()/poll() */ + + do + { +#ifdef HAVE_POLL + pfd.fd = http->fd; + pfd.events = POLLOUT; + + while ((nfds = poll(&pfd, 1, http->wait_value)) < 0 && + (errno == EINTR || errno == EAGAIN)); + +#else + do + { + FD_ZERO(&output_set); + FD_SET(http->fd, &output_set); + + timeout.tv_sec = http->wait_value / 1000; + timeout.tv_usec = 1000 * (http->wait_value % 1000); + + nfds = select(http->fd + 1, NULL, &output_set, NULL, &timeout); + } +# ifdef WIN32 + while (nfds < 0 && (WSAGetLastError() == WSAEINTR || + WSAGetLastError() == WSAEWOULDBLOCK)); +# else + while (nfds < 0 && (errno == EINTR || errno == EAGAIN)); +# endif /* WIN32 */ +#endif /* HAVE_POLL */ + + if (nfds < 0) + { + http->error = errno; + return (-1); + } + else if (nfds == 0 && !(*http->timeout_cb)(http, http->timeout_data)) + { +#ifdef WIN32 + http->error = WSAEWOULDBLOCK; +#else + http->error = EWOULDBLOCK; +#endif /* WIN32 */ + return (-1); + } + } + while (nfds <= 0); + } + +#ifdef HAVE_SSL + if (http->tls) + bytes = http_write_ssl(http, buffer, length); + else +#endif /* HAVE_SSL */ + bytes = send(http->fd, buffer, length, 0); + + if (bytes < 0) + { +#ifdef WIN32 + if (WSAGetLastError() == WSAEINTR) + continue; + else if (WSAGetLastError() == WSAEWOULDBLOCK) + { + if (http->timeout_cb && (*http->timeout_cb)(http, http->timeout_data)) + continue; + + http->error = WSAGetLastError(); + } + else if (WSAGetLastError() != http->error && + WSAGetLastError() != WSAECONNRESET) + { + http->error = WSAGetLastError(); + continue; + } + +#else + if (errno == EINTR) + continue; + else if (errno == EWOULDBLOCK || errno == EAGAIN) + { + if (http->timeout_cb && (*http->timeout_cb)(http, http->timeout_data)) + continue; + else if (!http->timeout_cb && errno == EAGAIN) + continue; + + http->error = errno; + } + else if (errno != http->error && errno != ECONNRESET) + { + http->error = errno; + continue; + } +#endif /* WIN32 */ + + DEBUG_printf(("3http_write: error writing data (%s).", + strerror(http->error))); + + return (-1); + } + + buffer += bytes; + tbytes += bytes; + length -= bytes; + } + +#ifdef DEBUG + http_debug_hex("http_write", buffer - tbytes, tbytes); +#endif /* DEBUG */ + + DEBUG_printf(("3http_write: Returning %d.", tbytes)); + + return (tbytes); +} + + +/* + * 'http_write_chunk()' - Write a chunked buffer. + */ + +static int /* O - Number bytes written */ +http_write_chunk(http_t *http, /* I - Connection to server */ + const char *buffer, /* I - Buffer to write */ + int length) /* I - Length of buffer */ +{ + char header[255]; /* Chunk header */ + int bytes; /* Bytes written */ + + + DEBUG_printf(("7http_write_chunk(http=%p, buffer=%p, length=%d)", + http, buffer, length)); + + /* + * Write the chunk header, data, and trailer. + */ + + sprintf(header, "%x\r\n", length); + if (http_write(http, header, (int)strlen(header)) < 0) + { + DEBUG_puts("8http_write_chunk: http_write of length failed!"); + return (-1); + } + + if ((bytes = http_write(http, buffer, length)) < 0) + { + DEBUG_puts("8http_write_chunk: http_write of buffer failed!"); + return (-1); + } + + if (http_write(http, "\r\n", 2) < 0) + { + DEBUG_puts("8http_write_chunk: http_write of CR LF failed!"); + return (-1); + } + + return (bytes); +} + + +#ifdef HAVE_SSL +/* + * 'http_write_ssl()' - Write to a SSL/TLS connection. + */ + +static int /* O - Bytes written */ +http_write_ssl(http_t *http, /* I - Connection to server */ + const char *buf, /* I - Buffer holding data */ + int len) /* I - Length of buffer */ +{ + ssize_t result; /* Return value */ + + + DEBUG_printf(("2http_write_ssl(http=%p, buf=%p, len=%d)", http, buf, len)); + +# if defined(HAVE_LIBSSL) + result = SSL_write((SSL *)(http->tls), buf, len); + +# elif defined(HAVE_GNUTLS) + result = gnutls_record_send(http->tls, buf, len); + + if (result < 0 && !errno) + { + /* + * Convert GNU TLS error to errno value... + */ + + switch (result) + { + case GNUTLS_E_INTERRUPTED : + errno = EINTR; + break; + + case GNUTLS_E_AGAIN : + errno = EAGAIN; + break; + + default : + errno = EPIPE; + break; + } + + result = -1; + } + +# elif defined(HAVE_CDSASSL) + OSStatus error; /* Error info */ + size_t processed; /* Number of bytes processed */ + + + error = SSLWrite(http->tls, buf, len, &processed); + + switch (error) + { + case 0 : + result = (int)processed; + break; + + case errSSLWouldBlock : + if (processed) + result = (int)processed; + else + { + result = -1; + errno = EINTR; + } + break; + + case errSSLClosedGraceful : + default : + if (processed) + result = (int)processed; + else + { + result = -1; + errno = EPIPE; + } + break; + } +# elif defined(HAVE_SSPISSL) + return _sspiWrite((_sspi_struct_t *)http->tls, (void *)buf, len); +# endif /* HAVE_LIBSSL */ + + DEBUG_printf(("3http_write_ssl: Returning %d.", (int)result)); + + return ((int)result); +} +#endif /* HAVE_SSL */ + + +/* + * End of "$Id$". + */ diff --git a/cups/http.h b/cups/http.h new file mode 100644 index 0000000000..16f0846efc --- /dev/null +++ b/cups/http.h @@ -0,0 +1,486 @@ +/* + * "$Id$" + * + * Hyper-Text Transport Protocol definitions for CUPS. + * + * Copyright 2007-2012 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_HTTP_H_ +# define _CUPS_HTTP_H_ + +/* + * Include necessary headers... + */ + +# include "versioning.h" +# include "array.h" +# include +# include +# include +# ifdef WIN32 +# ifndef __CUPS_SSIZE_T_DEFINED +# define __CUPS_SSIZE_T_DEFINED +/* Windows does not support the ssize_t type, so map it to off_t... */ +typedef off_t ssize_t; /* @private@ */ +# endif /* !__CUPS_SSIZE_T_DEFINED */ +# include +# include +# else +# ifdef __sgi +# define INET6 /* IRIX IPv6 support... */ +# endif /* __sgi */ +# include +# include +# include +# include +# include +# include +# include +# include +# if !defined(__APPLE__) || !defined(TCP_NODELAY) +# include +# endif /* !__APPLE__ || !TCP_NODELAY */ +# if defined(AF_UNIX) && !defined(AF_LOCAL) +# define AF_LOCAL AF_UNIX /* Older UNIX's have old names... */ +# endif /* AF_UNIX && !AF_LOCAL */ +# ifdef AF_LOCAL +# include +# endif /* AF_LOCAL */ +# if defined(LOCAL_PEERCRED) && !defined(SO_PEERCRED) +# define SO_PEERCRED LOCAL_PEERCRED +# endif /* LOCAL_PEERCRED && !SO_PEERCRED */ +# endif /* WIN32 */ + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Oh, the wonderful world of IPv6 compatibility. Apparently some + * implementations expose the (more logical) 32-bit address parts + * to everyone, while others only expose it to kernel code... To + * make supporting IPv6 even easier, each vendor chose different + * core structure and union names, so the same defines or code + * can't be used on all platforms. + * + * The following will likely need tweaking on new platforms that + * support IPv6 - the "s6_addr32" define maps to the 32-bit integer + * array in the in6_addr union, which is named differently on various + * platforms. + */ + +#if defined(AF_INET6) && !defined(s6_addr32) +# if defined(__sun) +# define s6_addr32 _S6_un._S6_u32 +# elif defined(__FreeBSD__) || defined(__NetBSD__) || defined(__OpenBSD__) || defined(__APPLE__)|| defined(__DragonFly__) +# define s6_addr32 __u6_addr.__u6_addr32 +# elif defined(__osf__) +# define s6_addr32 s6_un.sa6_laddr +# elif defined(WIN32) +/* + * Windows only defines byte and 16-bit word members of the union and + * requires special casing of all raw address code... + */ +# define s6_addr32 error_need_win32_specific_code +# endif /* __sun */ +#endif /* AF_INET6 && !s6_addr32 */ + + +/* + * Limits... + */ + +# define HTTP_MAX_URI 1024 /* Max length of URI string */ +# define HTTP_MAX_HOST 256 /* Max length of hostname string */ +# define HTTP_MAX_BUFFER 2048 /* Max length of data buffer */ +# define HTTP_MAX_VALUE 256 /* Max header field value length */ + + +/* + * Types and structures... + */ + +typedef enum http_auth_e /**** HTTP authentication types ****/ +{ + HTTP_AUTH_NONE, /* No authentication in use */ + HTTP_AUTH_BASIC, /* Basic authentication in use */ + HTTP_AUTH_MD5, /* Digest authentication in use */ + HTTP_AUTH_MD5_SESS, /* MD5-session authentication in use */ + HTTP_AUTH_MD5_INT, /* Digest authentication in use for body */ + HTTP_AUTH_MD5_SESS_INT, /* MD5-session authentication in use for body */ + HTTP_AUTH_NEGOTIATE /* GSSAPI authentication in use @since CUPS 1.3/Mac OS X 10.5@ */ +} http_auth_t; + +typedef enum http_encoding_e /**** HTTP transfer encoding values ****/ +{ + HTTP_ENCODE_LENGTH, /* Data is sent with Content-Length */ + HTTP_ENCODE_CHUNKED, /* Data is chunked */ + HTTP_ENCODE_FIELDS /* Sending HTTP fields */ +} http_encoding_t; + +typedef enum http_encryption_e /**** HTTP encryption values ****/ +{ + HTTP_ENCRYPT_IF_REQUESTED, /* Encrypt if requested (TLS upgrade) */ + HTTP_ENCRYPT_NEVER, /* Never encrypt */ + HTTP_ENCRYPT_REQUIRED, /* Encryption is required (TLS upgrade) */ + HTTP_ENCRYPT_ALWAYS /* Always encrypt (SSL) */ +} http_encryption_t; + +typedef enum http_field_e /**** HTTP field names ****/ +{ + HTTP_FIELD_UNKNOWN = -1, /* Unknown field */ + HTTP_FIELD_ACCEPT_LANGUAGE, /* Accept-Language field */ + HTTP_FIELD_ACCEPT_RANGES, /* Accept-Ranges field */ + HTTP_FIELD_AUTHORIZATION, /* Authorization field */ + HTTP_FIELD_CONNECTION, /* Connection field */ + HTTP_FIELD_CONTENT_ENCODING, /* Content-Encoding field */ + HTTP_FIELD_CONTENT_LANGUAGE, /* Content-Language field */ + HTTP_FIELD_CONTENT_LENGTH, /* Content-Length field */ + HTTP_FIELD_CONTENT_LOCATION, /* Content-Location field */ + HTTP_FIELD_CONTENT_MD5, /* Content-MD5 field */ + HTTP_FIELD_CONTENT_RANGE, /* Content-Range field */ + HTTP_FIELD_CONTENT_TYPE, /* Content-Type field */ + HTTP_FIELD_CONTENT_VERSION, /* Content-Version field */ + HTTP_FIELD_DATE, /* Date field */ + HTTP_FIELD_HOST, /* Host field */ + HTTP_FIELD_IF_MODIFIED_SINCE, /* If-Modified-Since field */ + HTTP_FIELD_IF_UNMODIFIED_SINCE, /* If-Unmodified-Since field */ + HTTP_FIELD_KEEP_ALIVE, /* Keep-Alive field */ + HTTP_FIELD_LAST_MODIFIED, /* Last-Modified field */ + HTTP_FIELD_LINK, /* Link field */ + HTTP_FIELD_LOCATION, /* Location field */ + HTTP_FIELD_RANGE, /* Range field */ + HTTP_FIELD_REFERER, /* Referer field */ + HTTP_FIELD_RETRY_AFTER, /* Retry-After field */ + HTTP_FIELD_TRANSFER_ENCODING, /* Transfer-Encoding field */ + HTTP_FIELD_UPGRADE, /* Upgrade field */ + HTTP_FIELD_USER_AGENT, /* User-Agent field */ + HTTP_FIELD_WWW_AUTHENTICATE, /* WWW-Authenticate field */ + HTTP_FIELD_MAX /* Maximum field index */ +} http_field_t; + +typedef enum http_keepalive_e /**** HTTP keep-alive values ****/ +{ + HTTP_KEEPALIVE_OFF = 0, /* No keep alive support */ + HTTP_KEEPALIVE_ON /* Use keep alive */ +} http_keepalive_t; + +typedef enum http_state_e /**** HTTP state values; states + **** are server-oriented... + ****/ +{ + HTTP_WAITING, /* Waiting for command */ + HTTP_OPTIONS, /* OPTIONS command, waiting for blank line */ + HTTP_GET, /* GET command, waiting for blank line */ + HTTP_GET_SEND, /* GET command, sending data */ + HTTP_HEAD, /* HEAD command, waiting for blank line */ + HTTP_POST, /* POST command, waiting for blank line */ + HTTP_POST_RECV, /* POST command, receiving data */ + HTTP_POST_SEND, /* POST command, sending data */ + HTTP_PUT, /* PUT command, waiting for blank line */ + HTTP_PUT_RECV, /* PUT command, receiving data */ + HTTP_DELETE, /* DELETE command, waiting for blank line */ + HTTP_TRACE, /* TRACE command, waiting for blank line */ + HTTP_CLOSE, /* CLOSE command, waiting for blank line */ + HTTP_STATUS /* Command complete, sending status */ +} http_state_t; + +typedef enum http_status_e /**** HTTP status codes ****/ +{ + HTTP_ERROR = -1, /* An error response from httpXxxx() */ + + HTTP_CONTINUE = 100, /* Everything OK, keep going... */ + HTTP_SWITCHING_PROTOCOLS, /* HTTP upgrade to TLS/SSL */ + + HTTP_OK = 200, /* OPTIONS/GET/HEAD/POST/TRACE command was successful */ + HTTP_CREATED, /* PUT command was successful */ + HTTP_ACCEPTED, /* DELETE command was successful */ + HTTP_NOT_AUTHORITATIVE, /* Information isn't authoritative */ + HTTP_NO_CONTENT, /* Successful command, no new data */ + HTTP_RESET_CONTENT, /* Content was reset/recreated */ + HTTP_PARTIAL_CONTENT, /* Only a partial file was recieved/sent */ + + HTTP_MULTIPLE_CHOICES = 300, /* Multiple files match request */ + HTTP_MOVED_PERMANENTLY, /* Document has moved permanently */ + HTTP_MOVED_TEMPORARILY, /* Document has moved temporarily */ + HTTP_SEE_OTHER, /* See this other link... */ + HTTP_NOT_MODIFIED, /* File not modified */ + HTTP_USE_PROXY, /* Must use a proxy to access this URI */ + + HTTP_BAD_REQUEST = 400, /* Bad request */ + HTTP_UNAUTHORIZED, /* Unauthorized to access host */ + HTTP_PAYMENT_REQUIRED, /* Payment required */ + HTTP_FORBIDDEN, /* Forbidden to access this URI */ + HTTP_NOT_FOUND, /* URI was not found */ + HTTP_METHOD_NOT_ALLOWED, /* Method is not allowed */ + HTTP_NOT_ACCEPTABLE, /* Not Acceptable */ + HTTP_PROXY_AUTHENTICATION, /* Proxy Authentication is Required */ + HTTP_REQUEST_TIMEOUT, /* Request timed out */ + HTTP_CONFLICT, /* Request is self-conflicting */ + HTTP_GONE, /* Server has gone away */ + HTTP_LENGTH_REQUIRED, /* A content length or encoding is required */ + HTTP_PRECONDITION, /* Precondition failed */ + HTTP_REQUEST_TOO_LARGE, /* Request entity too large */ + HTTP_URI_TOO_LONG, /* URI too long */ + HTTP_UNSUPPORTED_MEDIATYPE, /* The requested media type is unsupported */ + HTTP_REQUESTED_RANGE, /* The requested range is not satisfiable */ + HTTP_EXPECTATION_FAILED, /* The expectation given in an Expect header field was not met */ + HTTP_UPGRADE_REQUIRED = 426, /* Upgrade to SSL/TLS required */ + + HTTP_SERVER_ERROR = 500, /* Internal server error */ + HTTP_NOT_IMPLEMENTED, /* Feature not implemented */ + HTTP_BAD_GATEWAY, /* Bad gateway */ + HTTP_SERVICE_UNAVAILABLE, /* Service is unavailable */ + HTTP_GATEWAY_TIMEOUT, /* Gateway connection timed out */ + HTTP_NOT_SUPPORTED, /* HTTP version not supported */ + + HTTP_AUTHORIZATION_CANCELED = 1000, /* User canceled authorization @since CUPS 1.4@ */ + HTTP_PKI_ERROR, /* Error negotiating a secure connection @since CUPS 1.5/Mac OS X 10.7@ */ + HTTP_WEBIF_DISABLED /* Web interface is disabled @private@ */ +} http_status_t; + +typedef enum http_uri_status_e /**** URI separation status @since CUPS 1.2@ ****/ +{ + HTTP_URI_OVERFLOW = -8, /* URI buffer for httpAssembleURI is too small */ + HTTP_URI_BAD_ARGUMENTS = -7, /* Bad arguments to function (error) */ + HTTP_URI_BAD_RESOURCE = -6, /* Bad resource in URI (error) */ + HTTP_URI_BAD_PORT = -5, /* Bad port number in URI (error) */ + HTTP_URI_BAD_HOSTNAME = -4, /* Bad hostname in URI (error) */ + HTTP_URI_BAD_USERNAME = -3, /* Bad username in URI (error) */ + HTTP_URI_BAD_SCHEME = -2, /* Bad scheme in URI (error) */ + HTTP_URI_BAD_URI = -1, /* Bad/empty URI (error) */ + HTTP_URI_OK = 0, /* URI decoded OK */ + HTTP_URI_MISSING_SCHEME, /* Missing scheme in URI (warning) */ + HTTP_URI_UNKNOWN_SCHEME, /* Unknown scheme in URI (warning) */ + HTTP_URI_MISSING_RESOURCE /* Missing resource in URI (warning) */ +} http_uri_status_t; + +typedef enum http_uri_coding_e /**** URI en/decode flags ****/ +{ + HTTP_URI_CODING_NONE = 0, /* Don't en/decode anything */ + HTTP_URI_CODING_USERNAME = 1, /* En/decode the username portion */ + HTTP_URI_CODING_HOSTNAME = 2, /* En/decode the hostname portion */ + HTTP_URI_CODING_RESOURCE = 4, /* En/decode the resource portion */ + HTTP_URI_CODING_MOST = 7, /* En/decode all but the query */ + HTTP_URI_CODING_QUERY = 8, /* En/decode the query portion */ + HTTP_URI_CODING_ALL = 15 /* En/decode everything */ +} http_uri_coding_t; + +typedef enum http_version_e /**** HTTP version numbers ****/ +{ + HTTP_0_9 = 9, /* HTTP/0.9 */ + HTTP_1_0 = 100, /* HTTP/1.0 */ + HTTP_1_1 = 101 /* HTTP/1.1 */ +} http_version_t; + +typedef union _http_addr_u /**** Socket address union, which + **** makes using IPv6 and other + **** address types easier and + **** more portable. @since CUPS 1.2/Mac OS X 10.5@ + ****/ +{ + struct sockaddr addr; /* Base structure for family value */ + struct sockaddr_in ipv4; /* IPv4 address */ +#ifdef AF_INET6 + struct sockaddr_in6 ipv6; /* IPv6 address */ +#endif /* AF_INET6 */ +#ifdef AF_LOCAL + struct sockaddr_un un; /* Domain socket file */ +#endif /* AF_LOCAL */ + char pad[256]; /* Padding to ensure binary compatibility */ +} http_addr_t; + +typedef struct http_addrlist_s /**** Socket address list, which is + **** used to enumerate all of the + **** addresses that are associated + **** with a hostname. @since CUPS 1.2/Mac OS X 10.5@ + ****/ +{ + struct http_addrlist_s *next; /* Pointer to next address in list */ + http_addr_t addr; /* Address */ +} http_addrlist_t; + +typedef struct _http_s http_t; /**** HTTP connection type ****/ + +typedef struct http_credential_s /**** HTTP credential data @since CUPS 1.5/Mac OS X 10.7@ ****/ +{ + void *data; /* Pointer to credential data */ + size_t datalen; /* Credential length */ +} http_credential_t; + +typedef int (*http_timeout_cb_t)(http_t *http, void *user_data); + /**** HTTP timeout callback @since CUPS 1.5/Mac OS X 10.7@ ****/ + + + +/* + * Prototypes... + */ + +extern void httpBlocking(http_t *http, int b); +extern int httpCheck(http_t *http); +extern void httpClearFields(http_t *http); +extern void httpClose(http_t *http); +extern http_t *httpConnect(const char *host, int port); +extern http_t *httpConnectEncrypt(const char *host, int port, + http_encryption_t encryption); +extern int httpDelete(http_t *http, const char *uri); +extern int httpEncryption(http_t *http, http_encryption_t e); +extern int httpError(http_t *http); +extern void httpFlush(http_t *http); +extern int httpGet(http_t *http, const char *uri); +extern char *httpGets(char *line, int length, http_t *http); +extern const char *httpGetDateString(time_t t); +extern time_t httpGetDateTime(const char *s); +extern const char *httpGetField(http_t *http, http_field_t field); +extern struct hostent *httpGetHostByName(const char *name); +extern char *httpGetSubField(http_t *http, http_field_t field, + const char *name, char *value); +extern int httpHead(http_t *http, const char *uri); +extern void httpInitialize(void); +extern int httpOptions(http_t *http, const char *uri); +extern int httpPost(http_t *http, const char *uri); +extern int httpPrintf(http_t *http, const char *format, ...) + __attribute__ ((__format__ (__printf__, 2, 3))); +extern int httpPut(http_t *http, const char *uri); +extern int httpRead(http_t *http, char *buffer, int length) _CUPS_DEPRECATED; +extern int httpReconnect(http_t *http); +extern void httpSeparate(const char *uri, char *method, + char *username, char *host, int *port, + char *resource) _CUPS_DEPRECATED; +extern void httpSetField(http_t *http, http_field_t field, + const char *value); +extern const char *httpStatus(http_status_t status); +extern int httpTrace(http_t *http, const char *uri); +extern http_status_t httpUpdate(http_t *http); +extern int httpWrite(http_t *http, const char *buffer, int length) _CUPS_DEPRECATED; +extern char *httpEncode64(char *out, const char *in) _CUPS_DEPRECATED; +extern char *httpDecode64(char *out, const char *in) _CUPS_DEPRECATED; +extern int httpGetLength(http_t *http) _CUPS_DEPRECATED; +extern char *httpMD5(const char *, const char *, const char *, + char [33]); +extern char *httpMD5Final(const char *, const char *, const char *, + char [33]); +extern char *httpMD5String(const unsigned char *, char [33]); + +/**** New in CUPS 1.1.19 ****/ +extern void httpClearCookie(http_t *http) _CUPS_API_1_1_19; +extern const char *httpGetCookie(http_t *http) _CUPS_API_1_1_19; +extern void httpSetCookie(http_t *http, const char *cookie) _CUPS_API_1_1_19; +extern int httpWait(http_t *http, int msec) _CUPS_API_1_1_19; + +/**** New in CUPS 1.1.21 ****/ +extern char *httpDecode64_2(char *out, int *outlen, const char *in) _CUPS_API_1_1_21; +extern char *httpEncode64_2(char *out, int outlen, const char *in, + int inlen) _CUPS_API_1_1_21; +extern void httpSeparate2(const char *uri, + char *method, int methodlen, + char *username, int usernamelen, + char *host, int hostlen, int *port, + char *resource, int resourcelen) _CUPS_DEPRECATED; + +/**** New in CUPS 1.2/Mac OS X 10.5 ****/ +extern int httpAddrAny(const http_addr_t *addr) _CUPS_API_1_2; +extern http_addrlist_t *httpAddrConnect(http_addrlist_t *addrlist, int *sock) _CUPS_API_1_2; +extern int httpAddrEqual(const http_addr_t *addr1, + const http_addr_t *addr2) _CUPS_API_1_2; +extern void httpAddrFreeList(http_addrlist_t *addrlist) _CUPS_API_1_2; +extern http_addrlist_t *httpAddrGetList(const char *hostname, int family, + const char *service) _CUPS_API_1_2; +extern int httpAddrLength(const http_addr_t *addr) _CUPS_API_1_2; +extern int httpAddrLocalhost(const http_addr_t *addr) _CUPS_API_1_2; +extern char *httpAddrLookup(const http_addr_t *addr, + char *name, int namelen) _CUPS_API_1_2; +extern char *httpAddrString(const http_addr_t *addr, + char *s, int slen) _CUPS_API_1_2; +extern http_uri_status_t httpAssembleURI(http_uri_coding_t encoding, + char *uri, int urilen, + const char *scheme, + const char *username, + const char *host, int port, + const char *resource) _CUPS_API_1_2; +extern http_uri_status_t httpAssembleURIf(http_uri_coding_t encoding, + char *uri, int urilen, + const char *scheme, + const char *username, + const char *host, int port, + const char *resourcef, ...) _CUPS_API_1_2; +extern int httpFlushWrite(http_t *http) _CUPS_API_1_2; +extern int httpGetBlocking(http_t *http) _CUPS_API_1_2; +extern const char *httpGetDateString2(time_t t, char *s, int slen) _CUPS_API_1_2; +extern int httpGetFd(http_t *http) _CUPS_API_1_2; +extern const char *httpGetHostname(http_t *http, char *s, int slen) _CUPS_API_1_2; +extern off_t httpGetLength2(http_t *http) _CUPS_API_1_2; +extern http_status_t httpGetStatus(http_t *http) _CUPS_API_1_2; +extern char *httpGetSubField2(http_t *http, http_field_t field, + const char *name, char *value, + int valuelen) _CUPS_API_1_2; +extern ssize_t httpRead2(http_t *http, char *buffer, size_t length) _CUPS_API_1_2; +extern http_uri_status_t httpSeparateURI(http_uri_coding_t decoding, + const char *uri, + char *scheme, int schemelen, + char *username, int usernamelen, + char *host, int hostlen, int *port, + char *resource, int resourcelen) _CUPS_API_1_2; +extern void httpSetExpect(http_t *http, http_status_t expect) _CUPS_API_1_2; +extern void httpSetLength(http_t *http, size_t length) _CUPS_API_1_2; +extern ssize_t httpWrite2(http_t *http, const char *buffer, + size_t length) _CUPS_API_1_2; + +/**** New in CUPS 1.3/Mac OS X 10.5 ****/ +extern char *httpGetAuthString(http_t *http) _CUPS_API_1_3; +extern void httpSetAuthString(http_t *http, const char *scheme, + const char *data) _CUPS_API_1_3; + +/**** New in CUPS 1.5/Mac OS X 10.7 ****/ +extern int httpAddCredential(cups_array_t *credentials, + const void *data, size_t datalen) + _CUPS_API_1_5; +extern int httpCopyCredentials(http_t *http, + cups_array_t **credentials) + _CUPS_API_1_5; +extern void httpFreeCredentials(cups_array_t *certs) _CUPS_API_1_5; +extern int httpSetCredentials(http_t *http, cups_array_t *certs) + _CUPS_API_1_5; +extern void httpSetTimeout(http_t *http, double timeout, + http_timeout_cb_t cb, void *user_data) + _CUPS_API_1_5; + +/**** New in CUPS 1.6 ****/ +extern http_addrlist_t *httpAddrConnect2(http_addrlist_t *addrlist, int *sock, + int msec, int *cancel) + _CUPS_API_1_6; +extern http_state_t httpGetState(http_t *http) _CUPS_API_1_6; +extern http_version_t httpGetVersion(http_t *http) _CUPS_API_1_6; +extern int httpReconnect2(http_t *http, int msec, int *cancel) + _CUPS_API_1_6; + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +} +# endif /* __cplusplus */ +#endif /* !_CUPS_HTTP_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/ipp-private.h b/cups/ipp-private.h new file mode 100644 index 0000000000..07e65d19ce --- /dev/null +++ b/cups/ipp-private.h @@ -0,0 +1,175 @@ +/* + * "$Id$" + * + * Private IPP definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_IPP_PRIVATE_H_ +# define _CUPS_IPP_PRIVATE_H_ + +/* + * Include necessary headers... + */ + +# include + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Constants... + */ + +# define IPP_BUF_SIZE (IPP_MAX_LENGTH + 2) + /* Size of buffer */ + + +/* + * Structures... + */ + +typedef union _ipp_request_u /**** Request Header ****/ +{ + struct /* Any Header */ + { + ipp_uchar_t version[2]; /* Protocol version number */ + int op_status; /* Operation ID or status code*/ + int request_id; /* Request ID */ + } any; + + struct /* Operation Header */ + { + ipp_uchar_t version[2]; /* Protocol version number */ + ipp_op_t operation_id; /* Operation ID */ + int request_id; /* Request ID */ + } op; + + struct /* Status Header */ + { + ipp_uchar_t version[2]; /* Protocol version number */ + ipp_status_t status_code; /* Status code */ + int request_id; /* Request ID */ + } status; + + /**** New in CUPS 1.1.19 ****/ + struct /* Event Header @since CUPS 1.1.19/Mac OS X 10.3@ */ + { + ipp_uchar_t version[2]; /* Protocol version number */ + ipp_status_t status_code; /* Status code */ + int request_id; /* Request ID */ + } event; +} _ipp_request_t; + +/**** New in CUPS 1.1.19 ****/ + +typedef union _ipp_value_u /**** Attribute Value ****/ +{ + int integer; /* Integer/enumerated value */ + + char boolean; /* Boolean value */ + + ipp_uchar_t date[11]; /* Date/time value */ + + struct + { + int xres, /* Horizontal resolution */ + yres; /* Vertical resolution */ + ipp_res_t units; /* Resolution units */ + } resolution; /* Resolution value */ + + struct + { + int lower, /* Lower value */ + upper; /* Upper value */ + } range; /* Range of integers value */ + + struct + { + char *language; /* Language code */ + char *text; /* String */ + } string; /* String with language value */ + + struct + { + int length; /* Length of attribute */ + void *data; /* Data in attribute */ + } unknown; /* Unknown attribute type */ + +/**** New in CUPS 1.1.19 ****/ + ipp_t *collection; /* Collection value @since CUPS 1.1.19/Mac OS X 10.3@ */ +} _ipp_value_t; + +struct _ipp_attribute_s /**** Attribute ****/ +{ + ipp_attribute_t *next; /* Next attribute in list */ + ipp_tag_t group_tag, /* Job/Printer/Operation group tag */ + value_tag; /* What type of value is it? */ + char *name; /* Name of attribute */ + int num_values; /* Number of values */ + _ipp_value_t values[1]; /* Values */ +}; + +struct _ipp_s /**** IPP Request/Response/Notification ****/ +{ + ipp_state_t state; /* State of request */ + _ipp_request_t request; /* Request header */ + ipp_attribute_t *attrs; /* Attributes */ + ipp_attribute_t *last; /* Last attribute in list */ + ipp_attribute_t *current; /* Current attribute (for read/write) */ + ipp_tag_t curtag; /* Current attribute group tag */ + +/**** New in CUPS 1.2 ****/ + ipp_attribute_t *prev; /* Previous attribute (for read) @since CUPS 1.2/Mac OS X 10.5@ */ + +/**** New in CUPS 1.4.4 ****/ + int use; /* Use count @since CUPS 1.4.4/Mac OS X 10.6.?@ */ +}; + +typedef struct /**** Attribute mapping data ****/ +{ + int multivalue; /* Option has multiple values? */ + const char *name; /* Option/attribute name */ + ipp_tag_t value_tag; /* Value tag for this attribute */ + ipp_tag_t group_tag; /* Group tag for this attribute */ + ipp_tag_t alt_group_tag; /* Alternate group tag for this + * attribute */ +} _ipp_option_t; + + +/* + * Prototypes for private functions... + */ + +extern _ipp_option_t *_ippFindOption(const char *name); + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +} +# endif /* __cplusplus */ +#endif /* !_CUPS_IPP_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/ipp-support.c b/cups/ipp-support.c new file mode 100644 index 0000000000..223a8efa34 --- /dev/null +++ b/cups/ipp-support.c @@ -0,0 +1,1085 @@ +/* + * "$Id$" + * + * Internet Printing Protocol support functions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * ippAttributeString() - Convert the attribute's value to a string. + * ippEnumString() - Return a string corresponding to the enum value. + * ippEnumValue() - Return the value associated with a given enum + * string. + * ippErrorString() - Return a name for the given status code. + * ippErrorValue() - Return a status code for the given name. + * ippOpString() - Return a name for the given operation id. + * ippOpValue() - Return an operation id for the given name. + * ippPort() - Return the default IPP port number. + * ippSetPort() - Set the default port number. + * ippTagString() - Return the tag name corresponding to a tag value. + * ippTagValue() - Return the tag value corresponding to a tag name. + * ipp_col_string() - Convert a collection to a string. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" + + +/* + * Local globals... + */ + +static const char * const ipp_status_oks[] = /* "OK" status codes */ + { /* (name) = abandoned standard value */ + "successful-ok", + "successful-ok-ignored-or-substituted-attributes", + "successful-ok-conflicting-attributes", + "successful-ok-ignored-subscriptions", + "(successful-ok-ignored-notifications)", + "successful-ok-too-many-events", + "(successful-ok-but-cancel-subscription)", + "successful-ok-events-complete" + }, + * const ipp_status_400s[] = /* Client errors */ + { /* (name) = abandoned standard value */ + "client-error-bad-request", + "client-error-forbidden", + "client-error-not-authenticated", + "client-error-not-authorized", + "client-error-not-possible", + "client-error-timeout", + "client-error-not-found", + "client-error-gone", + "client-error-request-entity-too-large", + "client-error-request-value-too-long", + "client-error-document-format-not-supported", + "client-error-attributes-or-values-not-supported", + "client-error-uri-scheme-not-supported", + "client-error-charset-not-supported", + "client-error-conflicting-attributes", + "client-error-compression-not-supported", + "client-error-compression-error", + "client-error-document-format-error", + "client-error-document-access-error", + "client-error-attributes-not-settable", + "client-error-ignored-all-subscriptions", + "client-error-too-many-subscriptions", + "(client-error-ignored-all-notifications)", + "(client-error-client-print-support-file-not-found)" + }, + * const ipp_status_500s[] = /* Server errors */ + { + "server-error-internal-error", + "server-error-operation-not-supported", + "server-error-service-unavailable", + "server-error-version-not-supported", + "server-error-device-error", + "server-error-temporary-error", + "server-error-not-accepting-jobs", + "server-error-busy", + "server-error-job-canceled", + "server-error-multiple-document-jobs-not-supported", + "server-error-printer-is-deactivated", + "server-error-too-many-jobs", + "server-error-too-many-documents" + }, + * const ipp_status_1000s[] = /* CUPS internal */ + { + "cups-authorization-canceled", + "cups-pki-error", + "cups-upgrade-required" + }; +static const char * const ipp_std_ops[] = + { + /* 0x0000 - 0x000f */ + "0x00", + "0x01", + "Print-Job", + "Print-URI", + "Validate-Job", + "Create-Job", + "Send-Document", + "Send-URI", + "Cancel-Job", + "Get-Job-Attributes", + "Get-Jobs", + "Get-Printer-Attributes", + "Hold-Job", + "Release-Job", + "Restart-Job", + "0x0f", + + /* 0x0010 - 0x001f */ + "Pause-Printer", + "Resume-Printer", + "Purge-Jobs", + "Set-Printer-Attributes", + "Set-Job-Attributes", + "Get-Printer-Supported-Values", + "Create-Printer-Subscription", + "Create-Job-Subscription", + "Get-Subscription-Attributes", + "Get-Subscriptions", + "Renew-Subscription", + "Cancel-Subscription", + "Get-Notifications", + "(Send-Notifications)", + "(Get-Resource-Attributes)", + "(Get-Resource-Data)", + + /* 0x0020 - 0x002f */ + "(Get-Resources)", + "(Get-Printer-Support-Files)", + "Enable-Printer", + "Disable-Printer", + "Pause-Printer-After-Current-Job", + "Hold-New-Jobs", + "Release-Held-New-Jobs", + "Deactivate-Printer", + "Activate-Printer", + "Restart-Printer", + "Shutdown-Printer", + "Startup-Printer", + "Reprocess-Job", + "Cancel-Current-Job", + "Suspend-Current-Job", + "Resume-Job", + + /* 0x0030 - 0x003b */ + "Promote-Job", + "Schedule-Job-After", + "0x32", + "Cancel-Document", + "Get-Document-Attributes", + "Get-Documents", + "Delete-Document", + "Set-Document-Attributes", + "Cancel-Jobs", + "Cancel-My-Jobs", + "Resubmit-Job", + "Close-Job", + "Identify-Printer" + }, + * const ipp_cups_ops[] = + { + "CUPS-Get-Default", + "CUPS-Get-Printers", + "CUPS-Add-Modify-Printer", + "CUPS-Delete-Printer", + "CUPS-Get-Classes", + "CUPS-Add-Modify-Class", + "CUPS-Delete-Class", + "CUPS-Accept-Jobs", + "CUPS-Reject-Jobs", + "CUPS-Set-Default", + "CUPS-Get-Devices", + "CUPS-Get-PPDs", + "CUPS-Move-Job", + "CUPS-Authenticate-Job", + "CUPS-Get-PPD" + }, + * const ipp_cups_ops2[] = + { + "CUPS-Get-Document" + }, + * const ipp_tag_names[] = + { /* Value/group tag names */ + "zero", /* 0x00 */ + "operation-attributes-tag", + /* 0x01 */ + "job-attributes-tag", /* 0x02 */ + "end-of-attributes-tag", + /* 0x03 */ + "printer-attributes-tag", + /* 0x04 */ + "unsupported-attributes-tag", + /* 0x05 */ + "subscription-attributes-tag", + /* 0x06 */ + "event-notification-attributes-tag", + /* 0x07 */ + "(resource-attributes-tag)", + /* 0x08 */ + "document-attributes-tag", + /* 0x09 */ + "0x0a", /* 0x0a */ + "0x0b", /* 0x0b */ + "0x0c", /* 0x0c */ + "0x0d", /* 0x0d */ + "0x0e", /* 0x0e */ + "0x0f", /* 0x0f */ + "unsupported", /* 0x10 */ + "default", /* 0x11 */ + "unknown", /* 0x12 */ + "no-value", /* 0x13 */ + "0x14", /* 0x14 */ + "not-settable", /* 0x15 */ + "delete-attribute", /* 0x16 */ + "admin-define", /* 0x17 */ + "0x18", /* 0x18 */ + "0x19", /* 0x19 */ + "0x1a", /* 0x1a */ + "0x1b", /* 0x1b */ + "0x1c", /* 0x1c */ + "0x1d", /* 0x1d */ + "0x1e", /* 0x1e */ + "0x1f", /* 0x1f */ + "0x20", /* 0x20 */ + "integer", /* 0x21 */ + "boolean", /* 0x22 */ + "enum", /* 0x23 */ + "0x24", /* 0x24 */ + "0x25", /* 0x25 */ + "0x26", /* 0x26 */ + "0x27", /* 0x27 */ + "0x28", /* 0x28 */ + "0x29", /* 0x29 */ + "0x2a", /* 0x2a */ + "0x2b", /* 0x2b */ + "0x2c", /* 0x2c */ + "0x2d", /* 0x2d */ + "0x2e", /* 0x2e */ + "0x2f", /* 0x2f */ + "octetString", /* 0x30 */ + "dateTime", /* 0x31 */ + "resolution", /* 0x32 */ + "rangeOfInteger", /* 0x33 */ + "collection", /* 0x34 */ + "textWithLanguage", /* 0x35 */ + "nameWithLanguage", /* 0x36 */ + "endCollection", /* 0x37 */ + "0x38", /* 0x38 */ + "0x39", /* 0x39 */ + "0x3a", /* 0x3a */ + "0x3b", /* 0x3b */ + "0x3c", /* 0x3c */ + "0x3d", /* 0x3d */ + "0x3e", /* 0x3e */ + "0x3f", /* 0x3f */ + "0x40", /* 0x40 */ + "textWithoutLanguage",/* 0x41 */ + "nameWithoutLanguage",/* 0x42 */ + "0x43", /* 0x43 */ + "keyword", /* 0x44 */ + "uri", /* 0x45 */ + "uriScheme", /* 0x46 */ + "charset", /* 0x47 */ + "naturalLanguage", /* 0x48 */ + "mimeMediaType", /* 0x49 */ + "memberAttrName" /* 0x4a */ + }; +static const char * const ipp_document_states[] = + { /* document-state-enums */ + "pending", + "4", + "processing", + "6", + "canceled", + "aborted", + "completed" + }, + * const ipp_finishings[] = + { /* finishings enums */ + "none", + "staple", + "punch", + "cover", + "bind", + "saddle-stitch", + "edge-stitch", + "fold", + "trim", + "bale", + "booklet-maker", + "jog-offset", + "15", + "16", + "17", + "18", + "19", + "staple-top-left", + "staple-bottom-left", + "staple-top-right", + "staple-bottom-right", + "edge-stitch-left", + "edge-stitch-top", + "edge-stitch-right", + "edge-stitch-bottom", + "staple-dual-left", + "staple-dual-top", + "staple-dual-right", + "staple-dual-bottom", + "32", + "33", + "34", + "35", + "36", + "37", + "38", + "39", + "40", + "41", + "42", + "43", + "44", + "45", + "46", + "47", + "48", + "49", + "bind-left", + "bind-top", + "bind-right", + "bind-bottom", + "54", + "55", + "56", + "57", + "58", + "59", + "trim-after-pages", + "trim-after-documents", + "trim-after-copies", + "trim-after-job" + }, + * const ipp_job_collation_types[] = + { /* job-collation-type enums */ + "uncollated-sheets", + "collated-documents", + "uncollated-documents" + }, + * const ipp_job_states[] = + { /* job-state enums */ + "pending", + "pending-held", + "processing", + "processing-stopped", + "canceled", + "aborted", + "completed" + }, + * const ipp_orientation_requesteds[] = + { /* orientation-requested enums */ + "portrait", + "landscape", + "reverse-landscape", + "reverse-portrait" + }, + * const ipp_print_qualities[] = + { /* print-quality enums */ + "draft", + "normal", + "high" + }, + * const ipp_printer_states[] = + { /* printer-state enums */ + "idle", + "processing", + "stopped", + }; + + +/* + * Local functions... + */ + +static size_t ipp_col_string(ipp_t *col, char *buffer, size_t bufsize); + + +/* + * 'ippAttributeString()' - Convert the attribute's value to a string. + * + * Returns the number of bytes that would be written, not including the + * trailing nul. The buffer pointer can be NULL to get the required length, + * just like (v)snprintf. + * + * @since CUPS 1.6@ + */ + +size_t /* O - Number of bytes less nul */ +ippAttributeString( + ipp_attribute_t *attr, /* I - Attribute */ + char *buffer, /* I - String buffer or NULL */ + size_t bufsize) /* I - Size of string buffer */ +{ + int i; /* Looping var */ + char *bufptr, /* Pointer into buffer */ + *bufend, /* End of buffer */ + temp[256]; /* Temporary string */ + const char *ptr; /* Pointer into string */ + _ipp_value_t *val; /* Current value */ + + + if (!attr || !attr->name) + { + if (buffer) + *buffer = '\0'; + + return (0); + } + + bufptr = buffer; + if (buffer) + bufend = buffer + bufsize - 1; + else + bufend = NULL; + + for (i = attr->num_values, val = attr->values; i > 0; i --, val ++) + { + if (val > attr->values) + { + if (buffer && bufptr < bufend) + *bufptr++ = ','; + else + bufptr ++; + } + + switch (attr->value_tag & ~IPP_TAG_COPY) + { + case IPP_TAG_ENUM : + ptr = ippEnumString(attr->name, val->integer); + + if (buffer && bufptr < bufend) + strlcpy(bufptr, ptr, bufend - bufptr + 1); + + bufptr += strlen(ptr); + break; + + case IPP_TAG_INTEGER : + if (buffer && bufptr < bufend) + bufptr += snprintf(bufptr, bufend - bufptr + 1, "%d", val->integer); + else + bufptr += snprintf(temp, sizeof(temp), "%d", val->integer); + break; + + case IPP_TAG_BOOLEAN : + if (buffer && bufptr < bufend) + strlcpy(bufptr, val->boolean ? "true" : "false", + bufend - bufptr + 1); + + bufptr += val->boolean ? 4 : 5; + break; + + case IPP_TAG_RANGE : + if (buffer && bufptr < bufend) + bufptr += snprintf(bufptr, bufend - bufptr + 1, "%d-%d", + val->range.lower, val->range.upper); + else + bufptr += snprintf(temp, sizeof(temp), "%d-%d", val->range.lower, + val->range.upper); + break; + + case IPP_TAG_RESOLUTION : + if (buffer && bufptr < bufend) + bufptr += snprintf(bufptr, bufend - bufptr + 1, "%dx%d%s", + val->resolution.xres, val->resolution.yres, + val->resolution.units == IPP_RES_PER_INCH ? + "dpi" : "dpc"); + else + bufptr += snprintf(temp, sizeof(temp), "%dx%d%s", + val->resolution.xres, val->resolution.yres, + val->resolution.units == IPP_RES_PER_INCH ? + "dpi" : "dpc"); + break; + + case IPP_TAG_DATE : + { + unsigned year; /* Year */ + + year = (val->date[0] << 8) + val->date[1]; + + if (val->date[9] == 0 && val->date[10] == 0) + snprintf(temp, sizeof(temp), "%04u-%02u-%02uT%02u:%02u:%02uZ", + year, val->date[2], val->date[3], val->date[4], + val->date[5], val->date[6]); + else + snprintf(temp, sizeof(temp), + "%04u-%02u-%02uT%02u:%02u:%02u%c%02u%02u", + year, val->date[2], val->date[3], val->date[4], + val->date[5], val->date[6], val->date[8], val->date[9], + val->date[10]); + + if (buffer && bufptr < bufend) + strlcpy(bufptr, temp, bufend - bufptr + 1); + + bufptr += strlen(temp); + } + break; + + case IPP_TAG_TEXT : + case IPP_TAG_NAME : + case IPP_TAG_KEYWORD : + case IPP_TAG_CHARSET : + case IPP_TAG_URI : + case IPP_TAG_URISCHEME : + case IPP_TAG_MIMETYPE : + case IPP_TAG_LANGUAGE : + case IPP_TAG_TEXTLANG : + case IPP_TAG_NAMELANG : + if (!val->string.text) + break; + + for (ptr = val->string.text; *ptr; ptr ++) + { + if (*ptr == '\\' || *ptr == '\"' || *ptr == '[') + { + if (buffer && bufptr < bufend) + *bufptr = '\\'; + bufptr ++; + } + + if (buffer && bufptr < bufend) + *bufptr = *ptr; + bufptr ++; + } + + if (val->string.language) + { + /* + * Add "[language]" to end of string... + */ + + if (buffer && bufptr < bufend) + *bufptr = '['; + bufptr ++; + + if (buffer && bufptr < bufend) + strlcpy(bufptr, val->string.language, bufend - bufptr); + bufptr += strlen(val->string.language); + + if (buffer && bufptr < bufend) + *bufptr = ']'; + bufptr ++; + } + break; + + case IPP_TAG_BEGIN_COLLECTION : + if (buffer && bufptr < bufend) + bufptr += ipp_col_string(val->collection, bufptr, + bufend - bufptr + 1); + else + bufptr += ipp_col_string(val->collection, NULL, 0); + break; + + case IPP_TAG_STRING : + for (ptr = val->string.text; *ptr; ptr ++) + { + if (*ptr == '\\' || _cups_isspace(*ptr)) + { + if (buffer && bufptr < bufend) + *bufptr = '\\'; + bufptr ++; + + if (buffer && bufptr < bufend) + *bufptr = *ptr; + bufptr ++; + } + else if (!isprint(*ptr & 255)) + { + if (buffer && bufptr < bufend) + bufptr += snprintf(bufptr, bufend - bufptr + 1, "\\%03o", + *ptr & 255); + else + bufptr += snprintf(temp, sizeof(temp), "\\%03o", + *ptr & 255); + } + else + { + if (buffer && bufptr < bufend) + *bufptr = *ptr; + bufptr ++; + } + } + break; + + default : + ptr = ippTagString(attr->value_tag); + if (buffer && bufptr < bufend) + strlcpy(bufptr, ptr, bufend - bufptr + 1); + bufptr += strlen(ptr); + break; + } + } + + if (buffer && bufptr < bufend) + *bufptr = '\0'; + else if (bufend) + *bufend = '\0'; + + return (bufptr - buffer); +} + + +/* + * 'ippEnumString()' - Return a string corresponding to the enum value. + */ + +const char * /* O - Enum string */ +ippEnumString(const char *attrname, /* I - Attribute name */ + int enumvalue) /* I - Enum value */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + /* + * Check for standard enum values... + */ + + if (!strcmp(attrname, "document-state") && + enumvalue >= 3 && + enumvalue <= (3 + (int)(sizeof(ipp_document_states) / + sizeof(ipp_document_states[0])))) + return (ipp_document_states[enumvalue - 3]); + else if ((!strcmp(attrname, "finishings") || + !strcmp(attrname, "finishings-actual") || + !strcmp(attrname, "finishings-default") || + !strcmp(attrname, "finishings-ready") || + !strcmp(attrname, "finishings-supported")) && + enumvalue >= 3 && + enumvalue <= (3 + (int)(sizeof(ipp_finishings) / sizeof(ipp_finishings[0])))) + return (ipp_finishings[enumvalue - 3]); + else if ((!strcmp(attrname, "job-collation-type") || + !strcmp(attrname, "job-collation-type-actual")) && + enumvalue >= 3 && + enumvalue <= (3 + (int)(sizeof(ipp_job_collation_types) / + sizeof(ipp_job_collation_types[0])))) + return (ipp_job_collation_types[enumvalue - 3]); + else if (!strcmp(attrname, "job-state") && + enumvalue >= IPP_JOB_PENDING && enumvalue <= IPP_JOB_COMPLETED) + return (ipp_job_states[enumvalue - IPP_JOB_PENDING]); + else if (!strcmp(attrname, "operations-supported")) + return (ippOpString((ipp_op_t)enumvalue)); + else if ((!strcmp(attrname, "orientation-requested") || + !strcmp(attrname, "orientation-requested-actual") || + !strcmp(attrname, "orientation-requested-default") || + !strcmp(attrname, "orientation-requested-supported")) && + enumvalue >= 3 && + enumvalue <= (3 + (int)(sizeof(ipp_orientation_requesteds) / + sizeof(ipp_orientation_requesteds[0])))) + return (ipp_orientation_requesteds[enumvalue - 3]); + else if ((!strcmp(attrname, "print-quality") || + !strcmp(attrname, "print-quality-actual") || + !strcmp(attrname, "print-quality-default") || + !strcmp(attrname, "print-quality-supported")) && + enumvalue >= 3 && + enumvalue <= (3 + (int)(sizeof(ipp_print_qualities) / + sizeof(ipp_print_qualities[0])))) + return (ipp_print_qualities[enumvalue - 3]); + else if (!strcmp(attrname, "printer-state") && + enumvalue >= IPP_PRINTER_IDLE && enumvalue <= IPP_PRINTER_STOPPED) + return (ipp_printer_states[enumvalue - IPP_PRINTER_IDLE]); + + /* + * Not a standard enum value, just return the decimal equivalent... + */ + + snprintf(cg->ipp_unknown, sizeof(cg->ipp_unknown), "%d", enumvalue); + return (cg->ipp_unknown); +} + + +/* + * 'ippEnumValue()' - Return the value associated with a given enum string. + */ + +int /* O - Enum value or -1 if unknown */ +ippEnumValue(const char *attrname, /* I - Attribute name */ + const char *enumstring) /* I - Enum string */ +{ + int i, /* Looping var */ + num_strings; /* Number of strings to compare */ + const char * const *strings; /* Strings to compare */ + + + /* + * If the string is just a number, return it... + */ + + if (isdigit(*enumstring & 255)) + return (strtol(enumstring, NULL, 0)); + + /* + * Otherwise look up the string... + */ + + if (!strcmp(attrname, "document-state")) + { + num_strings = (int)(sizeof(ipp_document_states) / sizeof(ipp_document_states[0])); + strings = ipp_document_states; + } + else if (!strcmp(attrname, "finishings") || + !strcmp(attrname, "finishings-actual") || + !strcmp(attrname, "finishings-default") || + !strcmp(attrname, "finishings-ready") || + !strcmp(attrname, "finishings-supported")) + { + num_strings = (int)(sizeof(ipp_finishings) / sizeof(ipp_finishings[0])); + strings = ipp_finishings; + } + else if (!strcmp(attrname, "job-collation-type") || + !strcmp(attrname, "job-collation-type-actual")) + { + num_strings = (int)(sizeof(ipp_job_collation_types) / + sizeof(ipp_job_collation_types[0])); + strings = ipp_job_collation_types; + } + else if (!strcmp(attrname, "job-state")) + { + num_strings = (int)(sizeof(ipp_job_states) / sizeof(ipp_job_states[0])); + strings = ipp_job_states; + } + else if (!strcmp(attrname, "operations-supported")) + return (ippOpValue(enumstring)); + else if (!strcmp(attrname, "orientation-requested") || + !strcmp(attrname, "orientation-requested-actual") || + !strcmp(attrname, "orientation-requested-default") || + !strcmp(attrname, "orientation-requested-supported")) + { + num_strings = (int)(sizeof(ipp_orientation_requesteds) / + sizeof(ipp_orientation_requesteds[0])); + strings = ipp_orientation_requesteds; + } + else if (!strcmp(attrname, "print-quality") || + !strcmp(attrname, "print-quality-actual") || + !strcmp(attrname, "print-quality-default") || + !strcmp(attrname, "print-quality-supported")) + { + num_strings = (int)(sizeof(ipp_print_qualities) / sizeof(ipp_print_qualities[0])); + strings = ipp_print_qualities; + } + else if (!strcmp(attrname, "printer-state")) + { + num_strings = (int)(sizeof(ipp_printer_states) / sizeof(ipp_printer_states[0])); + strings = ipp_printer_states; + } + else + return (-1); + + for (i = 0; i < num_strings; i ++) + if (!strcmp(enumstring, strings[i])) + return (i + 3); + + return (-1); +} + + +/* + * 'ippErrorString()' - Return a name for the given status code. + */ + +const char * /* O - Text string */ +ippErrorString(ipp_status_t error) /* I - Error status */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + /* + * See if the error code is a known value... + */ + + if (error >= IPP_OK && error <= IPP_OK_EVENTS_COMPLETE) + return (ipp_status_oks[error]); + else if (error == IPP_REDIRECTION_OTHER_SITE) + return ("redirection-other-site"); + else if (error == CUPS_SEE_OTHER) + return ("cups-see-other"); + else if (error >= IPP_BAD_REQUEST && error <= IPP_PRINT_SUPPORT_FILE_NOT_FOUND) + return (ipp_status_400s[error - IPP_BAD_REQUEST]); + else if (error >= IPP_INTERNAL_ERROR && error <= IPP_PRINTER_IS_DEACTIVATED) + return (ipp_status_500s[error - IPP_INTERNAL_ERROR]); + else if (error >= IPP_AUTHENTICATION_CANCELED && error <= IPP_UPGRADE_REQUIRED) + return (ipp_status_1000s[error - IPP_AUTHENTICATION_CANCELED]); + + /* + * No, build an "0xxxxx" error string... + */ + + sprintf(cg->ipp_unknown, "0x%04x", error); + + return (cg->ipp_unknown); +} + + +/* + * 'ippErrorValue()' - Return a status code for the given name. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ipp_status_t /* O - IPP status code */ +ippErrorValue(const char *name) /* I - Name */ +{ + int i; + + + for (i = 0; i < (sizeof(ipp_status_oks) / sizeof(ipp_status_oks[0])); i ++) + if (!_cups_strcasecmp(name, ipp_status_oks[i])) + return ((ipp_status_t)i); + + if (!_cups_strcasecmp(name, "redirection-other-site")) + return (IPP_REDIRECTION_OTHER_SITE); + + if (!_cups_strcasecmp(name, "cups-see-other")) + return (CUPS_SEE_OTHER); + + for (i = 0; i < (sizeof(ipp_status_400s) / sizeof(ipp_status_400s[0])); i ++) + if (!_cups_strcasecmp(name, ipp_status_400s[i])) + return ((ipp_status_t)(i + 0x400)); + + for (i = 0; i < (sizeof(ipp_status_500s) / sizeof(ipp_status_500s[0])); i ++) + if (!_cups_strcasecmp(name, ipp_status_500s[i])) + return ((ipp_status_t)(i + 0x500)); + + for (i = 0; i < (sizeof(ipp_status_1000s) / sizeof(ipp_status_1000s[0])); i ++) + if (!_cups_strcasecmp(name, ipp_status_1000s[i])) + return ((ipp_status_t)(i + 0x1000)); + + return ((ipp_status_t)-1); +} + + +/* + * 'ippOpString()' - Return a name for the given operation id. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +const char * /* O - Name */ +ippOpString(ipp_op_t op) /* I - Operation ID */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + /* + * See if the operation ID is a known value... + */ + + if (op >= IPP_PRINT_JOB && op <= IPP_CLOSE_JOB) + return (ipp_std_ops[op]); + else if (op == IPP_PRIVATE) + return ("windows-ext"); + else if (op >= CUPS_GET_DEFAULT && op <= CUPS_GET_PPD) + return (ipp_cups_ops[op - CUPS_GET_DEFAULT]); + else if (op == CUPS_GET_DOCUMENT) + return (ipp_cups_ops2[0]); + + /* + * No, build an "0xxxxx" operation string... + */ + + sprintf(cg->ipp_unknown, "0x%04x", op); + + return (cg->ipp_unknown); +} + + +/* + * 'ippOpValue()' - Return an operation id for the given name. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ipp_op_t /* O - Operation ID */ +ippOpValue(const char *name) /* I - Textual name */ +{ + int i; + + + if (!strncmp(name, "0x", 2)) + return ((ipp_op_t)strtol(name + 2, NULL, 16)); + + for (i = 0; i < (sizeof(ipp_std_ops) / sizeof(ipp_std_ops[0])); i ++) + if (!_cups_strcasecmp(name, ipp_std_ops[i])) + return ((ipp_op_t)i); + + if (!_cups_strcasecmp(name, "windows-ext")) + return (IPP_PRIVATE); + + for (i = 0; i < (sizeof(ipp_cups_ops) / sizeof(ipp_cups_ops[0])); i ++) + if (!_cups_strcasecmp(name, ipp_cups_ops[i])) + return ((ipp_op_t)(i + 0x4001)); + + for (i = 0; i < (sizeof(ipp_cups_ops2) / sizeof(ipp_cups_ops2[0])); i ++) + if (!_cups_strcasecmp(name, ipp_cups_ops2[i])) + return ((ipp_op_t)(i + 0x4027)); + + if (!_cups_strcasecmp(name, "CUPS-Add-Class")) + return (CUPS_ADD_MODIFY_CLASS); + + if (!_cups_strcasecmp(name, "CUPS-Add-Printer")) + return (CUPS_ADD_MODIFY_PRINTER); + + return ((ipp_op_t)-1); +} + + +/* + * 'ippPort()' - Return the default IPP port number. + */ + +int /* O - Port number */ +ippPort(void) +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + DEBUG_puts("ippPort()"); + + if (!cg->ipp_port) + _cupsSetDefaults(); + + DEBUG_printf(("1ippPort: Returning %d...", cg->ipp_port)); + + return (cg->ipp_port); +} + + +/* + * 'ippSetPort()' - Set the default port number. + */ + +void +ippSetPort(int p) /* I - Port number to use */ +{ + DEBUG_printf(("ippSetPort(p=%d)", p)); + + _cupsGlobals()->ipp_port = p; +} + + +/* + * 'ippTagString()' - Return the tag name corresponding to a tag value. + * + * The returned names are defined in RFC 2911 and 3382. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +const char * /* O - Tag name */ +ippTagString(ipp_tag_t tag) /* I - Tag value */ +{ + tag &= IPP_TAG_MASK; + + if (tag < (ipp_tag_t)(sizeof(ipp_tag_names) / sizeof(ipp_tag_names[0]))) + return (ipp_tag_names[tag]); + else + return ("UNKNOWN"); +} + + +/* + * 'ippTagValue()' - Return the tag value corresponding to a tag name. + * + * The tag names are defined in RFC 2911 and 3382. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +ipp_tag_t /* O - Tag value */ +ippTagValue(const char *name) /* I - Tag name */ +{ + int i; /* Looping var */ + + + for (i = 0; i < (sizeof(ipp_tag_names) / sizeof(ipp_tag_names[0])); i ++) + if (!_cups_strcasecmp(name, ipp_tag_names[i])) + return ((ipp_tag_t)i); + + if (!_cups_strcasecmp(name, "operation")) + return (IPP_TAG_OPERATION); + else if (!_cups_strcasecmp(name, "job")) + return (IPP_TAG_JOB); + else if (!_cups_strcasecmp(name, "printer")) + return (IPP_TAG_PRINTER); + else if (!_cups_strcasecmp(name, "unsupported")) + return (IPP_TAG_UNSUPPORTED_GROUP); + else if (!_cups_strcasecmp(name, "subscription")) + return (IPP_TAG_SUBSCRIPTION); + else if (!_cups_strcasecmp(name, "event")) + return (IPP_TAG_EVENT_NOTIFICATION); + else if (!_cups_strcasecmp(name, "language")) + return (IPP_TAG_LANGUAGE); + else if (!_cups_strcasecmp(name, "mimetype")) + return (IPP_TAG_MIMETYPE); + else if (!_cups_strcasecmp(name, "name")) + return (IPP_TAG_NAME); + else if (!_cups_strcasecmp(name, "text")) + return (IPP_TAG_TEXT); + else if (!_cups_strcasecmp(name, "begCollection")) + return (IPP_TAG_BEGIN_COLLECTION); + else + return (IPP_TAG_ZERO); +} + + +/* + * 'ipp_col_string()' - Convert a collection to a string. + */ + +static size_t /* O - Number of bytes */ +ipp_col_string(ipp_t *col, /* I - Collection attribute */ + char *buffer, /* I - Buffer or NULL */ + size_t bufsize) /* I - Size of buffer */ +{ + char *bufptr, /* Position in buffer */ + *bufend, /* End of buffer */ + prefix = '{', /* Prefix character */ + temp[256]; /* Temporary string */ + ipp_attribute_t *attr; /* Current member attribute */ + + + bufptr = buffer; + bufend = buffer + bufsize - 1; + + for (attr = col->attrs; attr; attr = attr->next) + { + if (!attr->name) + continue; + + if (buffer && bufptr < bufend) + *bufptr = prefix; + bufptr ++; + prefix = ' '; + + if (buffer && bufptr < bufend) + bufptr += snprintf(bufptr, bufend - bufptr + 1, "%s=", attr->name); + else + bufptr += strlen(attr->name) + 1; + + if (buffer && bufptr < bufend) + bufptr += ippAttributeString(attr, bufptr, bufend - bufptr + 1); + else + bufptr += ippAttributeString(attr, temp, sizeof(temp)); + } + + if (prefix == '{') + { + if (buffer && bufptr < bufend) + *bufptr = prefix; + bufptr ++; + } + + if (buffer && bufptr < bufend) + *bufptr = '}'; + bufptr ++; + + return (bufptr - buffer); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/ipp.c b/cups/ipp.c new file mode 100644 index 0000000000..a9f4d13c88 --- /dev/null +++ b/cups/ipp.c @@ -0,0 +1,5565 @@ +/* + * "$Id$" + * + * Internet Printing Protocol functions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * _cupsBufferGet() - Get a read/write buffer. + * _cupsBufferRelease() - Release a read/write buffer. + * ippAddBoolean() - Add a boolean attribute to an IPP message. + * ippAddBooleans() - Add an array of boolean values. + * ippAddCollection() - Add a collection value. + * ippAddCollections() - Add an array of collection values. + * ippAddDate() - Add a date attribute to an IPP message. + * ippAddInteger() - Add a integer attribute to an IPP message. + * ippAddIntegers() - Add an array of integer values. + * ippAddOctetString() - Add an octetString value to an IPP message. + * ippAddOutOfBand() - Add an out-of-band value to an IPP message. + * ippAddRange() - Add a range of values to an IPP message. + * ippAddRanges() - Add ranges of values to an IPP message. + * ippAddResolution() - Add a resolution value to an IPP message. + * ippAddResolutions() - Add resolution values to an IPP message. + * ippAddSeparator() - Add a group separator to an IPP message. + * ippAddString() - Add a language-encoded string to an IPP message. + * ippAddStrings() - Add language-encoded strings to an IPP message. + * ippCopyAttribute() - Copy an attribute. + * ippCopyAttributes() - Copy attributes from one IPP message to another. + * ippDateToTime() - Convert from RFC 1903 Date/Time format to UNIX + * time in seconds. + * ippDelete() - Delete an IPP message. + * ippDeleteAttribute() - Delete a single attribute in an IPP message. + * ippDeleteValues() - Delete values in an attribute. + * ippFindAttribute() - Find a named attribute in a request. + * ippFindNextAttribute() - Find the next named attribute in a request. + * ippFirstAttribute() - Return the first attribute in the message. + * ippGetBoolean() - Get a boolean value for an attribute. + * ippGetCollection() - Get a collection value for an attribute. + * ippGetCount() - Get the number of values in an attribute. + * ippGetDate() - Get a date value for an attribute. + * ippGetGroupTag() - Get the group associated with an attribute. + * ippGetInteger() - Get the integer/enum value for an attribute. + * ippGetName() - Get the attribute name. + * ippGetOperation() - Get the operation ID in an IPP message. + * ippGetRange() - Get a rangeOfInteger value from an attribute. + * ippGetRequestId() - Get the request ID from an IPP message. + * ippGetResolution() - Get a resolution value for an attribute. + * ippGetStatusCode() - Get the status code from an IPP response or event + * message. + * ippGetString() - Get the string and optionally the language code + * for an attribute. + * ippGetValueTag() - Get the value tag for an attribute. + * ippGetVersion() - Get the major and minor version number from an + * IPP message. + * ippLength() - Compute the length of an IPP message. + * ippNextAttribute() - Return the next attribute in the message. + * ippNew() - Allocate a new IPP message. + * ippNewRequest() - Allocate a new IPP request message. + * ippRead() - Read data for an IPP message from a HTTP + * connection. + * ippReadFile() - Read data for an IPP message from a file. + * ippReadIO() - Read data for an IPP message. + * ippSetBoolean() - Set a boolean value in an attribute. + * ippSetCollection() - Set a collection value in an attribute. + * ippSetDate() - Set a date value in an attribute. + * ippSetGroupTag() - Set the group tag of an attribute. + * ippSetInteger() - Set an integer or enum value in an attribute. + * ippSetName() - Set the name of an attribute. + * ippSetOperation() - Set the operation ID in an IPP request message. + * ippSetRange() - Set a rangeOfInteger value in an attribute. + * ippSetRequestId() - Set the request ID in an IPP message. + * ippSetResolution() - Set a resolution value in an attribute. + * ippSetState() - Set the current state of the IPP message. + * ippSetStatusCode() - Set the status code in an IPP response or event + * message. + * ippSetString() - Set a string value in an attribute. + * ippSetValueTag() - Set the value tag of an attribute. + * ippSetVersion() - Set the version number in an IPP message. + * ippTimeToDate() - Convert from UNIX time to RFC 1903 format. + * ippWrite() - Write data for an IPP message to a HTTP + * connection. + * ippWriteFile() - Write data for an IPP message to a file. + * ippWriteIO() - Write data for an IPP message. + * ipp_add_attr() - Add a new attribute to the message. + * ipp_free_values() - Free attribute values. + * ipp_get_code() - Convert a C locale/charset name into an IPP + * language/charset code. + * ipp_lang_code() - Convert a C locale name into an IPP language + * code. + * ipp_length() - Compute the length of an IPP message or + * collection value. + * ipp_read_http() - Semi-blocking read on a HTTP connection... + * ipp_read_file() - Read IPP data from a file. + * ipp_set_value() - Get the value element from an attribute, + * expanding it as needed. + * ipp_write_file() - Write IPP data to a file. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#ifdef WIN32 +# include +#endif /* WIN32 */ + + +/* + * Local functions... + */ + +static ipp_attribute_t *ipp_add_attr(ipp_t *ipp, const char *name, ipp_tag_t group_tag, + ipp_tag_t value_tag, int num_values); +static void ipp_free_values(ipp_attribute_t *attr, int element, int count); +static char *ipp_get_code(const char *locale, char *buffer, size_t bufsize); +static char *ipp_lang_code(const char *locale, char *buffer, size_t bufsize); +static size_t ipp_length(ipp_t *ipp, int collection); +static ssize_t ipp_read_http(http_t *http, ipp_uchar_t *buffer, + size_t length); +static ssize_t ipp_read_file(int *fd, ipp_uchar_t *buffer, + size_t length); +static _ipp_value_t *ipp_set_value(ipp_t *ipp, ipp_attribute_t **attr, int element); +static ssize_t ipp_write_file(int *fd, ipp_uchar_t *buffer, + size_t length); + + +/* + * '_cupsBufferGet()' - Get a read/write buffer. + */ + +char * /* O - Buffer */ +_cupsBufferGet(size_t size) /* I - Size required */ +{ + _cups_buffer_t *buffer; /* Current buffer */ + _cups_globals_t *cg = _cupsGlobals(); + /* Global data */ + + + for (buffer = cg->cups_buffers; buffer; buffer = buffer->next) + if (!buffer->used && buffer->size >= size) + break; + + if (!buffer) + { + if ((buffer = malloc(sizeof(_cups_buffer_t) + size - 1)) == NULL) + return (NULL); + + buffer->next = cg->cups_buffers; + buffer->size = size; + cg->cups_buffers = buffer; + } + + buffer->used = 1; + + return (buffer->d); +} + + +/* + * '_cupsBufferRelease()' - Release a read/write buffer. + */ + +void +_cupsBufferRelease(char *b) /* I - Buffer to release */ +{ + _cups_buffer_t *buffer; /* Buffer */ + + + /* + * Mark this buffer as unused... + */ + + buffer = (_cups_buffer_t *)(b - offsetof(_cups_buffer_t, d)); + buffer->used = 0; +} + + +/* + * 'ippAddBoolean()' - Add a boolean attribute to an IPP message. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code group@ parameter specifies the IPP attribute group tag: none + * (@code IPP_TAG_ZERO@, for member attributes), document (@code IPP_TAG_DOCUMENT@), + * event notification (@code IPP_TAG_EVENT_NOTIFICATION@), operation + * (@code IPP_TAG_OPERATION@), printer (@code IPP_TAG_PRINTER@), subscription + * (@code IPP_TAG_SUBSCRIPTION@), or unsupported (@code IPP_TAG_UNSUPPORTED_GROUP@). + */ + +ipp_attribute_t * /* O - New attribute */ +ippAddBoolean(ipp_t *ipp, /* I - IPP message */ + ipp_tag_t group, /* I - IPP group */ + const char *name, /* I - Name of attribute */ + char value) /* I - Value of attribute */ +{ + ipp_attribute_t *attr; /* New attribute */ + + + DEBUG_printf(("ippAddBoolean(ipp=%p, group=%02x(%s), name=\"%s\", value=%d)", + ipp, group, ippTagString(group), name, value)); + + /* + * Range check input... + */ + + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE) + return (NULL); + + /* + * Create the attribute... + */ + + if ((attr = ipp_add_attr(ipp, name, group, IPP_TAG_BOOLEAN, 1)) == NULL) + return (NULL); + + attr->values[0].boolean = value; + + return (attr); +} + + +/* + * 'ippAddBooleans()' - Add an array of boolean values. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code group@ parameter specifies the IPP attribute group tag: none + * (@code IPP_TAG_ZERO@, for member attributes), document (@code IPP_TAG_DOCUMENT@), + * event notification (@code IPP_TAG_EVENT_NOTIFICATION@), operation + * (@code IPP_TAG_OPERATION@), printer (@code IPP_TAG_PRINTER@), subscription + * (@code IPP_TAG_SUBSCRIPTION@), or unsupported (@code IPP_TAG_UNSUPPORTED_GROUP@). + */ + +ipp_attribute_t * /* O - New attribute */ +ippAddBooleans(ipp_t *ipp, /* I - IPP message */ + ipp_tag_t group, /* I - IPP group */ + const char *name, /* I - Name of attribute */ + int num_values, /* I - Number of values */ + const char *values) /* I - Values */ +{ + int i; /* Looping var */ + ipp_attribute_t *attr; /* New attribute */ + _ipp_value_t *value; /* Current value */ + + + DEBUG_printf(("ippAddBooleans(ipp=%p, group=%02x(%s), name=\"%s\", " + "num_values=%d, values=%p)", ipp, group, ippTagString(group), + name, num_values, values)); + + /* + * Range check input... + */ + + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE || + num_values < 1) + return (NULL); + + /* + * Create the attribute... + */ + + if ((attr = ipp_add_attr(ipp, name, group, IPP_TAG_BOOLEAN, num_values)) == NULL) + return (NULL); + + if (values) + { + for (i = num_values, value = attr->values; + i > 0; + i --, value ++) + value->boolean = *values++; + } + + return (attr); +} + + +/* + * 'ippAddCollection()' - Add a collection value. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code group@ parameter specifies the IPP attribute group tag: none + * (@code IPP_TAG_ZERO@, for member attributes), document (@code IPP_TAG_DOCUMENT@), + * event notification (@code IPP_TAG_EVENT_NOTIFICATION@), operation + * (@code IPP_TAG_OPERATION@), printer (@code IPP_TAG_PRINTER@), subscription + * (@code IPP_TAG_SUBSCRIPTION@), or unsupported (@code IPP_TAG_UNSUPPORTED_GROUP@). + * + * @since CUPS 1.1.19/Mac OS X 10.3@ + */ + +ipp_attribute_t * /* O - New attribute */ +ippAddCollection(ipp_t *ipp, /* I - IPP message */ + ipp_tag_t group, /* I - IPP group */ + const char *name, /* I - Name of attribute */ + ipp_t *value) /* I - Value */ +{ + ipp_attribute_t *attr; /* New attribute */ + + + DEBUG_printf(("ippAddCollection(ipp=%p, group=%02x(%s), name=\"%s\", " + "value=%p)", ipp, group, ippTagString(group), name, value)); + + /* + * Range check input... + */ + + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE) + return (NULL); + + /* + * Create the attribute... + */ + + if ((attr = ipp_add_attr(ipp, name, group, IPP_TAG_BEGIN_COLLECTION, 1)) == NULL) + return (NULL); + + attr->values[0].collection = value; + + if (value) + value->use ++; + + return (attr); +} + + +/* + * 'ippAddCollections()' - Add an array of collection values. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code group@ parameter specifies the IPP attribute group tag: none + * (@code IPP_TAG_ZERO@, for member attributes), document (@code IPP_TAG_DOCUMENT@), + * event notification (@code IPP_TAG_EVENT_NOTIFICATION@), operation + * (@code IPP_TAG_OPERATION@), printer (@code IPP_TAG_PRINTER@), subscription + * (@code IPP_TAG_SUBSCRIPTION@), or unsupported (@code IPP_TAG_UNSUPPORTED_GROUP@). + * + * @since CUPS 1.1.19/Mac OS X 10.3@ + */ + +ipp_attribute_t * /* O - New attribute */ +ippAddCollections( + ipp_t *ipp, /* I - IPP message */ + ipp_tag_t group, /* I - IPP group */ + const char *name, /* I - Name of attribute */ + int num_values, /* I - Number of values */ + const ipp_t **values) /* I - Values */ +{ + int i; /* Looping var */ + ipp_attribute_t *attr; /* New attribute */ + _ipp_value_t *value; /* Current value */ + + + DEBUG_printf(("ippAddCollections(ipp=%p, group=%02x(%s), name=\"%s\", " + "num_values=%d, values=%p)", ipp, group, ippTagString(group), + name, num_values, values)); + + /* + * Range check input... + */ + + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE || + num_values < 1) + return (NULL); + + /* + * Create the attribute... + */ + + if ((attr = ipp_add_attr(ipp, name, group, IPP_TAG_BEGIN_COLLECTION, + num_values)) == NULL) + return (NULL); + + if (values) + { + for (i = num_values, value = attr->values; + i > 0; + i --, value ++) + { + value->collection = (ipp_t *)*values++; + value->collection->use ++; + } + } + + return (attr); +} + + +/* + * 'ippAddDate()' - Add a date attribute to an IPP message. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code group@ parameter specifies the IPP attribute group tag: none + * (@code IPP_TAG_ZERO@, for member attributes), document (@code IPP_TAG_DOCUMENT@), + * event notification (@code IPP_TAG_EVENT_NOTIFICATION@), operation + * (@code IPP_TAG_OPERATION@), printer (@code IPP_TAG_PRINTER@), subscription + * (@code IPP_TAG_SUBSCRIPTION@), or unsupported (@code IPP_TAG_UNSUPPORTED_GROUP@). + */ + +ipp_attribute_t * /* O - New attribute */ +ippAddDate(ipp_t *ipp, /* I - IPP message */ + ipp_tag_t group, /* I - IPP group */ + const char *name, /* I - Name of attribute */ + const ipp_uchar_t *value) /* I - Value */ +{ + ipp_attribute_t *attr; /* New attribute */ + + + DEBUG_printf(("ippAddDate(ipp=%p, group=%02x(%s), name=\"%s\", value=%p)", + ipp, group, ippTagString(group), name, value)); + + /* + * Range check input... + */ + + if (!ipp || !name || !value || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE) + return (NULL); + + /* + * Create the attribute... + */ + + if ((attr = ipp_add_attr(ipp, name, group, IPP_TAG_DATE, 1)) == NULL) + return (NULL); + + memcpy(attr->values[0].date, value, 11); + + return (attr); +} + + +/* + * 'ippAddInteger()' - Add a integer attribute to an IPP message. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code group@ parameter specifies the IPP attribute group tag: none + * (@code IPP_TAG_ZERO@, for member attributes), document (@code IPP_TAG_DOCUMENT@), + * event notification (@code IPP_TAG_EVENT_NOTIFICATION@), operation + * (@code IPP_TAG_OPERATION@), printer (@code IPP_TAG_PRINTER@), subscription + * (@code IPP_TAG_SUBSCRIPTION@), or unsupported (@code IPP_TAG_UNSUPPORTED_GROUP@). + * + * Supported values include enum (@code IPP_TAG_ENUM@) and integer + * (@code IPP_TAG_INTEGER@). + */ + +ipp_attribute_t * /* O - New attribute */ +ippAddInteger(ipp_t *ipp, /* I - IPP message */ + ipp_tag_t group, /* I - IPP group */ + ipp_tag_t value_tag, /* I - Type of attribute */ + const char *name, /* I - Name of attribute */ + int value) /* I - Value of attribute */ +{ + ipp_attribute_t *attr; /* New attribute */ + + + DEBUG_printf(("ippAddInteger(ipp=%p, group=%02x(%s), type=%02x(%s), " + "name=\"%s\", value=%d)", ipp, group, ippTagString(group), + value_tag, ippTagString(value_tag), name, value)); + + value_tag &= IPP_TAG_MASK; + + /* + * Special-case for legacy usage: map out-of-band attributes to new ippAddOutOfBand + * function... + */ + + if (value_tag >= IPP_TAG_UNSUPPORTED_VALUE && value_tag <= IPP_TAG_ADMINDEFINE) + return (ippAddOutOfBand(ipp, group, value_tag, name)); + + /* + * Range check input... + */ + +#if 0 + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE || + (value_tag != IPP_TAG_INTEGER && value_tag != IPP_TAG_ENUM)) + return (NULL); +#else + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE) + return (NULL); +#endif /* 0 */ + + /* + * Create the attribute... + */ + + if ((attr = ipp_add_attr(ipp, name, group, value_tag, 1)) == NULL) + return (NULL); + + attr->values[0].integer = value; + + return (attr); +} + + +/* + * 'ippAddIntegers()' - Add an array of integer values. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code group@ parameter specifies the IPP attribute group tag: none + * (@code IPP_TAG_ZERO@, for member attributes), document (@code IPP_TAG_DOCUMENT@), + * event notification (@code IPP_TAG_EVENT_NOTIFICATION@), operation + * (@code IPP_TAG_OPERATION@), printer (@code IPP_TAG_PRINTER@), subscription + * (@code IPP_TAG_SUBSCRIPTION@), or unsupported (@code IPP_TAG_UNSUPPORTED_GROUP@). + * + * Supported values include enum (@code IPP_TAG_ENUM@) and integer + * (@code IPP_TAG_INTEGER@). + */ + +ipp_attribute_t * /* O - New attribute */ +ippAddIntegers(ipp_t *ipp, /* I - IPP message */ + ipp_tag_t group, /* I - IPP group */ + ipp_tag_t value_tag, /* I - Type of attribute */ + const char *name, /* I - Name of attribute */ + int num_values, /* I - Number of values */ + const int *values) /* I - Values */ +{ + int i; /* Looping var */ + ipp_attribute_t *attr; /* New attribute */ + _ipp_value_t *value; /* Current value */ + + + DEBUG_printf(("ippAddIntegers(ipp=%p, group=%02x(%s), type=%02x(%s), " + "name=\"%s\", num_values=%d, values=%p)", ipp, + group, ippTagString(group), value_tag, ippTagString(value_tag), name, + num_values, values)); + + value_tag &= IPP_TAG_MASK; + + /* + * Range check input... + */ + +#if 0 + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE || + (value_tag != IPP_TAG_INTEGER && value_tag != IPP_TAG_ENUM) || + num_values < 1) + return (NULL); +#else + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE || + num_values < 1) + return (NULL); +#endif /* 0 */ + + /* + * Create the attribute... + */ + + if ((attr = ipp_add_attr(ipp, name, group, value_tag, num_values)) == NULL) + return (NULL); + + if (values) + { + for (i = num_values, value = attr->values; + i > 0; + i --, value ++) + value->integer = *values++; + } + + return (attr); +} + + +/* + * 'ippAddOctetString()' - Add an octetString value to an IPP message. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code group@ parameter specifies the IPP attribute group tag: none + * (@code IPP_TAG_ZERO@, for member attributes), document (@code IPP_TAG_DOCUMENT@), + * event notification (@code IPP_TAG_EVENT_NOTIFICATION@), operation + * (@code IPP_TAG_OPERATION@), printer (@code IPP_TAG_PRINTER@), subscription + * (@code IPP_TAG_SUBSCRIPTION@), or unsupported (@code IPP_TAG_UNSUPPORTED_GROUP@). + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ipp_attribute_t * /* O - New attribute */ +ippAddOctetString(ipp_t *ipp, /* I - IPP message */ + ipp_tag_t group, /* I - IPP group */ + const char *name, /* I - Name of attribute */ + const void *data, /* I - octetString data */ + int datalen) /* I - Length of data in bytes */ +{ + ipp_attribute_t *attr; /* New attribute */ + + + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE) + return (NULL); + + if ((attr = ipp_add_attr(ipp, name, group, IPP_TAG_STRING, 1)) == NULL) + return (NULL); + + /* + * Initialize the attribute data... + */ + + attr->values[0].unknown.length = datalen; + + if (data) + { + if ((attr->values[0].unknown.data = malloc(datalen)) == NULL) + { + ippDeleteAttribute(ipp, attr); + return (NULL); + } + + memcpy(attr->values[0].unknown.data, data, datalen); + } + + /* + * Return the new attribute... + */ + + return (attr); +} + + +/* + * 'ippAddOutOfBand()' - Add an out-of-band value to an IPP message. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code group@ parameter specifies the IPP attribute group tag: none + * (@code IPP_TAG_ZERO@, for member attributes), document (@code IPP_TAG_DOCUMENT@), + * event notification (@code IPP_TAG_EVENT_NOTIFICATION@), operation + * (@code IPP_TAG_OPERATION@), printer (@code IPP_TAG_PRINTER@), subscription + * (@code IPP_TAG_SUBSCRIPTION@), or unsupported (@code IPP_TAG_UNSUPPORTED_GROUP@). + * + * Supported out-of-band values include unsupported-value + * (@code IPP_TAG_UNSUPPORTED_VALUE@), default (@code IPP_TAG_DEFAULT@), unknown + * (@code IPP_TAG_UNKNOWN@), no-value (@code IPP_TAG_NOVALUE@), not-settable + * (@code IPP_TAG_NOTSETTABLE@), delete-attribute (@code IPP_TAG_DELETEATTR@), and + * admin-define (@code IPP_TAG_ADMINDEFINE@). + * + * @since CUPS 1.6@ + */ + +ipp_attribute_t * /* O - New attribute */ +ippAddOutOfBand(ipp_t *ipp, /* I - IPP message */ + ipp_tag_t group, /* I - IPP group */ + ipp_tag_t value_tag, /* I - Type of attribute */ + const char *name) /* I - Name of attribute */ +{ + DEBUG_printf(("ippAddOutOfBand(ipp=%p, group=%02x(%s), value_tag=%02x(%s), " + "name=\"%s\")", ipp, group, ippTagString(group), value_tag, + ippTagString(value_tag), name)); + + value_tag &= IPP_TAG_MASK; + + /* + * Range check input... + */ + + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE || + (value_tag != IPP_TAG_UNSUPPORTED_VALUE && + value_tag != IPP_TAG_DEFAULT && + value_tag != IPP_TAG_UNKNOWN && + value_tag != IPP_TAG_NOVALUE && + value_tag != IPP_TAG_NOTSETTABLE && + value_tag != IPP_TAG_DELETEATTR && + value_tag != IPP_TAG_ADMINDEFINE)) + return (NULL); + + /* + * Create the attribute... + */ + + return (ipp_add_attr(ipp, name, group, value_tag, 1)); +} + + +/* + * 'ippAddRange()' - Add a range of values to an IPP message. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code group@ parameter specifies the IPP attribute group tag: none + * (@code IPP_TAG_ZERO@, for member attributes), document (@code IPP_TAG_DOCUMENT@), + * event notification (@code IPP_TAG_EVENT_NOTIFICATION@), operation + * (@code IPP_TAG_OPERATION@), printer (@code IPP_TAG_PRINTER@), subscription + * (@code IPP_TAG_SUBSCRIPTION@), or unsupported (@code IPP_TAG_UNSUPPORTED_GROUP@). + * + * The @code lower@ parameter must be less than or equal to the @code upper@ parameter. + */ + +ipp_attribute_t * /* O - New attribute */ +ippAddRange(ipp_t *ipp, /* I - IPP message */ + ipp_tag_t group, /* I - IPP group */ + const char *name, /* I - Name of attribute */ + int lower, /* I - Lower value */ + int upper) /* I - Upper value */ +{ + ipp_attribute_t *attr; /* New attribute */ + + + DEBUG_printf(("ippAddRange(ipp=%p, group=%02x(%s), name=\"%s\", lower=%d, " + "upper=%d)", ipp, group, ippTagString(group), name, lower, + upper)); + + /* + * Range check input... + */ + + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE) + return (NULL); + + /* + * Create the attribute... + */ + + if ((attr = ipp_add_attr(ipp, name, group, IPP_TAG_RANGE, 1)) == NULL) + return (NULL); + + attr->values[0].range.lower = lower; + attr->values[0].range.upper = upper; + + return (attr); +} + + +/* + * 'ippAddRanges()' - Add ranges of values to an IPP message. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code group@ parameter specifies the IPP attribute group tag: none + * (@code IPP_TAG_ZERO@, for member attributes), document (@code IPP_TAG_DOCUMENT@), + * event notification (@code IPP_TAG_EVENT_NOTIFICATION@), operation + * (@code IPP_TAG_OPERATION@), printer (@code IPP_TAG_PRINTER@), subscription + * (@code IPP_TAG_SUBSCRIPTION@), or unsupported (@code IPP_TAG_UNSUPPORTED_GROUP@). + */ + +ipp_attribute_t * /* O - New attribute */ +ippAddRanges(ipp_t *ipp, /* I - IPP message */ + ipp_tag_t group, /* I - IPP group */ + const char *name, /* I - Name of attribute */ + int num_values, /* I - Number of values */ + const int *lower, /* I - Lower values */ + const int *upper) /* I - Upper values */ +{ + int i; /* Looping var */ + ipp_attribute_t *attr; /* New attribute */ + _ipp_value_t *value; /* Current value */ + + + DEBUG_printf(("ippAddRanges(ipp=%p, group=%02x(%s), name=\"%s\", " + "num_values=%d, lower=%p, upper=%p)", ipp, group, + ippTagString(group), name, num_values, lower, upper)); + + /* + * Range check input... + */ + + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE || + num_values < 1) + return (NULL); + + /* + * Create the attribute... + */ + + if ((attr = ipp_add_attr(ipp, name, group, IPP_TAG_RANGE, num_values)) == NULL) + return (NULL); + + if (lower && upper) + { + for (i = num_values, value = attr->values; + i > 0; + i --, value ++) + { + value->range.lower = *lower++; + value->range.upper = *upper++; + } + } + + return (attr); +} + + +/* + * 'ippAddResolution()' - Add a resolution value to an IPP message. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code group@ parameter specifies the IPP attribute group tag: none + * (@code IPP_TAG_ZERO@, for member attributes), document (@code IPP_TAG_DOCUMENT@), + * event notification (@code IPP_TAG_EVENT_NOTIFICATION@), operation + * (@code IPP_TAG_OPERATION@), printer (@code IPP_TAG_PRINTER@), subscription + * (@code IPP_TAG_SUBSCRIPTION@), or unsupported (@code IPP_TAG_UNSUPPORTED_GROUP@). + */ + +ipp_attribute_t * /* O - New attribute */ +ippAddResolution(ipp_t *ipp, /* I - IPP message */ + ipp_tag_t group, /* I - IPP group */ + const char *name, /* I - Name of attribute */ + ipp_res_t units, /* I - Units for resolution */ + int xres, /* I - X resolution */ + int yres) /* I - Y resolution */ +{ + ipp_attribute_t *attr; /* New attribute */ + + + DEBUG_printf(("ippAddResolution(ipp=%p, group=%02x(%s), name=\"%s\", " + "units=%d, xres=%d, yres=%d)", ipp, group, + ippTagString(group), name, units, xres, yres)); + + /* + * Range check input... + */ + + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE || + units < IPP_RES_PER_INCH || units > IPP_RES_PER_CM || + xres < 0 || yres < 0) + return (NULL); + + /* + * Create the attribute... + */ + + if ((attr = ipp_add_attr(ipp, name, group, IPP_TAG_RESOLUTION, 1)) == NULL) + return (NULL); + + attr->values[0].resolution.xres = xres; + attr->values[0].resolution.yres = yres; + attr->values[0].resolution.units = units; + + return (attr); +} + + +/* + * 'ippAddResolutions()' - Add resolution values to an IPP message. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code group@ parameter specifies the IPP attribute group tag: none + * (@code IPP_TAG_ZERO@, for member attributes), document (@code IPP_TAG_DOCUMENT@), + * event notification (@code IPP_TAG_EVENT_NOTIFICATION@), operation + * (@code IPP_TAG_OPERATION@), printer (@code IPP_TAG_PRINTER@), subscription + * (@code IPP_TAG_SUBSCRIPTION@), or unsupported (@code IPP_TAG_UNSUPPORTED_GROUP@). + */ + +ipp_attribute_t * /* O - New attribute */ +ippAddResolutions(ipp_t *ipp, /* I - IPP message */ + ipp_tag_t group, /* I - IPP group */ + const char *name, /* I - Name of attribute */ + int num_values,/* I - Number of values */ + ipp_res_t units, /* I - Units for resolution */ + const int *xres, /* I - X resolutions */ + const int *yres) /* I - Y resolutions */ +{ + int i; /* Looping var */ + ipp_attribute_t *attr; /* New attribute */ + _ipp_value_t *value; /* Current value */ + + + DEBUG_printf(("ippAddResolutions(ipp=%p, group=%02x(%s), name=\"%s\", " + "num_value=%d, units=%d, xres=%p, yres=%p)", ipp, group, + ippTagString(group), name, num_values, units, xres, yres)); + + /* + * Range check input... + */ + + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE || + num_values < 1 || + units < IPP_RES_PER_INCH || units > IPP_RES_PER_CM) + return (NULL); + + /* + * Create the attribute... + */ + + if ((attr = ipp_add_attr(ipp, name, group, IPP_TAG_RESOLUTION, num_values)) == NULL) + return (NULL); + + if (xres && yres) + { + for (i = num_values, value = attr->values; + i > 0; + i --, value ++) + { + value->resolution.xres = *xres++; + value->resolution.yres = *yres++; + value->resolution.units = units; + } + } + + return (attr); +} + + +/* + * 'ippAddSeparator()' - Add a group separator to an IPP message. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + */ + +ipp_attribute_t * /* O - New attribute */ +ippAddSeparator(ipp_t *ipp) /* I - IPP message */ +{ + DEBUG_printf(("ippAddSeparator(ipp=%p)", ipp)); + + /* + * Range check input... + */ + + if (!ipp) + return (NULL); + + /* + * Create the attribute... + */ + + return (ipp_add_attr(ipp, NULL, IPP_TAG_ZERO, IPP_TAG_ZERO, 0)); +} + + +/* + * 'ippAddString()' - Add a language-encoded string to an IPP message. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code group@ parameter specifies the IPP attribute group tag: none + * (@code IPP_TAG_ZERO@, for member attributes), document (@code IPP_TAG_DOCUMENT@), + * event notification (@code IPP_TAG_EVENT_NOTIFICATION@), operation + * (@code IPP_TAG_OPERATION@), printer (@code IPP_TAG_PRINTER@), subscription + * (@code IPP_TAG_SUBSCRIPTION@), or unsupported (@code IPP_TAG_UNSUPPORTED_GROUP@). + * + * Supported string values include charset (@code IPP_TAG_CHARSET@), keyword + * (@code IPP_TAG_KEYWORD@), language (@code IPP_TAG_LANGUAGE@), mimeMediaType + * (@code IPP_TAG_MIMETYPE@), name (@code IPP_TAG_NAME@), nameWithLanguage + * (@code IPP_TAG_NAMELANG), text (@code IPP_TAG_TEXT@), textWithLanguage + * (@code IPP_TAG_TEXTLANG@), uri (@code IPP_TAG_URI@), and uriScheme + * (@code IPP_TAG_URISCHEME@). + * + * The @code language@ parameter must be non-@code NULL@ for nameWithLanguage and + * textWithLanguage string values and must be @code NULL@ for all other string values. + */ + +ipp_attribute_t * /* O - New attribute */ +ippAddString(ipp_t *ipp, /* I - IPP message */ + ipp_tag_t group, /* I - IPP group */ + ipp_tag_t value_tag, /* I - Type of attribute */ + const char *name, /* I - Name of attribute */ + const char *language, /* I - Language code */ + const char *value) /* I - Value */ +{ + ipp_tag_t temp_tag; /* Temporary value tag (masked) */ + ipp_attribute_t *attr; /* New attribute */ + char code[32]; /* Charset/language code buffer */ + + + DEBUG_printf(("ippAddString(ipp=%p, group=%02x(%s), value_tag=%02x(%s), " + "name=\"%s\", language=\"%s\", value=\"%s\")", ipp, + group, ippTagString(group), value_tag, ippTagString(value_tag), name, + language, value)); + + /* + * Range check input... + */ + + temp_tag = (ipp_tag_t)((int)value_tag & IPP_TAG_MASK); + +#if 0 + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE || + (temp_tag < IPP_TAG_TEXT && temp_tag != IPP_TAG_TEXTLANG && + temp_tag != IPP_TAG_NAMELANG) || temp_tag > IPP_TAG_MIMETYPE) + return (NULL); + + if ((temp_tag == IPP_TAG_TEXTLANG || temp_tag == IPP_TAG_NAMELANG) + != (language != NULL)) + return (NULL); +#else + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE) + return (NULL); +#endif /* 0 */ + + /* + * See if we need to map charset, language, or locale values... + */ + + if (language && ((int)value_tag & IPP_TAG_COPY) && + strcmp(language, ipp_lang_code(language, code, sizeof(code)))) + value_tag = temp_tag; /* Don't do a fast copy */ + else if (value && value_tag == (ipp_tag_t)(IPP_TAG_CHARSET | IPP_TAG_COPY) && + strcmp(value, ipp_get_code(value, code, sizeof(code)))) + value_tag = temp_tag; /* Don't do a fast copy */ + else if (value && value_tag == (ipp_tag_t)(IPP_TAG_LANGUAGE | IPP_TAG_COPY) && + strcmp(value, ipp_lang_code(value, code, sizeof(code)))) + value_tag = temp_tag; /* Don't do a fast copy */ + + /* + * Create the attribute... + */ + + if ((attr = ipp_add_attr(ipp, name, group, value_tag, 1)) == NULL) + return (NULL); + + /* + * Initialize the attribute data... + */ + + if ((int)value_tag & IPP_TAG_COPY) + { + attr->values[0].string.language = (char *)language; + attr->values[0].string.text = (char *)value; + } + else + { + if (language) + attr->values[0].string.language = _cupsStrAlloc(ipp_lang_code(language, code, + sizeof(code))); + + if (value_tag == IPP_TAG_CHARSET) + attr->values[0].string.text = _cupsStrAlloc(ipp_get_code(value, code, + sizeof(code))); + else if (value_tag == IPP_TAG_LANGUAGE) + attr->values[0].string.text = _cupsStrAlloc(ipp_lang_code(value, code, + sizeof(code))); + else + attr->values[0].string.text = _cupsStrAlloc(value); + } + + return (attr); +} + + +/* + * 'ippAddStrings()' - Add language-encoded strings to an IPP message. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code group@ parameter specifies the IPP attribute group tag: none + * (@code IPP_TAG_ZERO@, for member attributes), document (@code IPP_TAG_DOCUMENT@), + * event notification (@code IPP_TAG_EVENT_NOTIFICATION@), operation + * (@code IPP_TAG_OPERATION@), printer (@code IPP_TAG_PRINTER@), subscription + * (@code IPP_TAG_SUBSCRIPTION@), or unsupported (@code IPP_TAG_UNSUPPORTED_GROUP@). + * + * Supported string values include charset (@code IPP_TAG_CHARSET@), keyword + * (@code IPP_TAG_KEYWORD@), language (@code IPP_TAG_LANGUAGE@), mimeMediaType + * (@code IPP_TAG_MIMETYPE@), name (@code IPP_TAG_NAME@), nameWithLanguage + * (@code IPP_TAG_NAMELANG), text (@code IPP_TAG_TEXT@), textWithLanguage + * (@code IPP_TAG_TEXTLANG@), uri (@code IPP_TAG_URI@), and uriScheme + * (@code IPP_TAG_URISCHEME@). + * + * The @code language@ parameter must be non-@code NULL@ for nameWithLanguage and + * textWithLanguage string values and must be @code NULL@ for all other string values. + */ + +ipp_attribute_t * /* O - New attribute */ +ippAddStrings( + ipp_t *ipp, /* I - IPP message */ + ipp_tag_t group, /* I - IPP group */ + ipp_tag_t value_tag, /* I - Type of attribute */ + const char *name, /* I - Name of attribute */ + int num_values, /* I - Number of values */ + const char *language, /* I - Language code (@code NULL@ for default) */ + const char * const *values) /* I - Values */ +{ + int i; /* Looping var */ + ipp_tag_t temp_tag; /* Temporary value tag (masked) */ + ipp_attribute_t *attr; /* New attribute */ + _ipp_value_t *value; /* Current value */ + char code[32]; /* Language/charset value buffer */ + + + DEBUG_printf(("ippAddStrings(ipp=%p, group=%02x(%s), value_tag=%02x(%s), " + "name=\"%s\", num_values=%d, language=\"%s\", values=%p)", ipp, + group, ippTagString(group), value_tag, ippTagString(value_tag), name, + num_values, language, values)); + + /* + * Range check input... + */ + + temp_tag = (ipp_tag_t)((int)value_tag & IPP_TAG_MASK); + +#if 0 + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE || + (temp_tag < IPP_TAG_TEXT && temp_tag != IPP_TAG_TEXTLANG && + temp_tag != IPP_TAG_NAMELANG) || temp_tag > IPP_TAG_MIMETYPE || + num_values < 1) + return (NULL); + + if ((temp_tag == IPP_TAG_TEXTLANG || temp_tag == IPP_TAG_NAMELANG) + != (language != NULL)) + return (NULL); +#else + if (!ipp || !name || group < IPP_TAG_ZERO || + group == IPP_TAG_END || group >= IPP_TAG_UNSUPPORTED_VALUE || + num_values < 1) + return (NULL); +#endif /* 0 */ + + /* + * See if we need to map charset, language, or locale values... + */ + + if (language && ((int)value_tag & IPP_TAG_COPY) && + strcmp(language, ipp_lang_code(language, code, sizeof(code)))) + value_tag = temp_tag; /* Don't do a fast copy */ + else if (values && value_tag == (ipp_tag_t)(IPP_TAG_CHARSET | IPP_TAG_COPY)) + { + for (i = 0; i < num_values; i ++) + if (strcmp(values[i], ipp_get_code(values[i], code, sizeof(code)))) + { + value_tag = temp_tag; /* Don't do a fast copy */ + break; + } + } + else if (values && value_tag == (ipp_tag_t)(IPP_TAG_LANGUAGE | IPP_TAG_COPY)) + { + for (i = 0; i < num_values; i ++) + if (strcmp(values[i], ipp_lang_code(values[i], code, sizeof(code)))) + { + value_tag = temp_tag; /* Don't do a fast copy */ + break; + } + } + + /* + * Create the attribute... + */ + + if ((attr = ipp_add_attr(ipp, name, group, value_tag, num_values)) == NULL) + return (NULL); + + /* + * Initialize the attribute data... + */ + + for (i = num_values, value = attr->values; + i > 0; + i --, value ++) + { + if (language) + { + if (value == attr->values) + { + if ((int)value_tag & IPP_TAG_COPY) + value->string.language = (char *)language; + else + value->string.language = _cupsStrAlloc(ipp_lang_code(language, code, + sizeof(code))); + } + else + value->string.language = attr->values[0].string.language; + } + + if (values) + { + if ((int)value_tag & IPP_TAG_COPY) + value->string.text = (char *)*values++; + else if (value_tag == IPP_TAG_CHARSET) + value->string.text = _cupsStrAlloc(ipp_get_code(*values++, code, sizeof(code))); + else if (value_tag == IPP_TAG_LANGUAGE) + value->string.text = _cupsStrAlloc(ipp_lang_code(*values++, code, sizeof(code))); + else + value->string.text = _cupsStrAlloc(*values++); + } + } + + return (attr); +} + + +/* + * 'ippCopyAttribute()' - Copy an attribute. + * + * The specified attribute, @code attr@, is copied to the destination IPP message. + * When @code quickcopy@ is non-zero, a "shallow" reference copy of the attribute is + * created - this should only be done as long as the original source IPP message will + * not be freed for the life of the destination. + * + * @since CUPS 1.6@ + */ + + +ipp_attribute_t * /* O - New attribute */ +ippCopyAttribute( + ipp_t *dst, /* I - Destination IPP message */ + ipp_attribute_t *srcattr, /* I - Attribute to copy */ + int quickcopy) /* I - 1 for a referenced copy, 0 for normal */ +{ + int i; /* Looping var */ + ipp_attribute_t *dstattr; /* Destination attribute */ + _ipp_value_t *srcval, /* Source value */ + *dstval; /* Destination value */ + + + DEBUG_printf(("ippCopyAttribute(dst=%p, srcattr=%p, quickcopy=%d)", dst, srcattr, + quickcopy)); + + /* + * Range check input... + */ + + if (!dst || !srcattr) + return (NULL); + + /* + * Copy it... + */ + + quickcopy = quickcopy ? IPP_TAG_COPY : 0; + + switch (srcattr->value_tag & ~IPP_TAG_COPY) + { + case IPP_TAG_ZERO : + dstattr = ippAddSeparator(dst); + break; + + case IPP_TAG_INTEGER : + case IPP_TAG_ENUM : + dstattr = ippAddIntegers(dst, srcattr->group_tag, srcattr->value_tag, + srcattr->name, srcattr->num_values, NULL); + if (!dstattr) + break; + + for (i = srcattr->num_values, srcval = srcattr->values, dstval = dstattr->values; + i > 0; + i --, srcval ++, dstval ++) + dstval->integer = srcval->integer; + break; + + case IPP_TAG_BOOLEAN : + dstattr = ippAddBooleans(dst, srcattr->group_tag, srcattr->name, + srcattr->num_values, NULL); + if (!dstattr) + break; + + for (i = srcattr->num_values, srcval = srcattr->values, dstval = dstattr->values; + i > 0; + i --, srcval ++, dstval ++) + dstval->boolean = srcval->boolean; + break; + + case IPP_TAG_TEXT : + case IPP_TAG_NAME : + case IPP_TAG_KEYWORD : + case IPP_TAG_URI : + case IPP_TAG_URISCHEME : + case IPP_TAG_CHARSET : + case IPP_TAG_LANGUAGE : + case IPP_TAG_MIMETYPE : + dstattr = ippAddStrings(dst, srcattr->group_tag, + (ipp_tag_t)(srcattr->value_tag | quickcopy), + srcattr->name, srcattr->num_values, NULL, NULL); + if (!dstattr) + break; + + if (quickcopy) + { + for (i = srcattr->num_values, srcval = srcattr->values, + dstval = dstattr->values; + i > 0; + i --, srcval ++, dstval ++) + dstval->string.text = srcval->string.text; + } + else if (srcattr->value_tag & IPP_TAG_COPY) + { + for (i = srcattr->num_values, srcval = srcattr->values, + dstval = dstattr->values; + i > 0; + i --, srcval ++, dstval ++) + dstval->string.text = _cupsStrAlloc(srcval->string.text); + } + else + { + for (i = srcattr->num_values, srcval = srcattr->values, + dstval = dstattr->values; + i > 0; + i --, srcval ++, dstval ++) + dstval->string.text = _cupsStrRetain(srcval->string.text); + } + break; + + case IPP_TAG_DATE : + if (srcattr->num_values != 1) + return (NULL); + + dstattr = ippAddDate(dst, srcattr->group_tag, srcattr->name, + srcattr->values[0].date); + break; + + case IPP_TAG_RESOLUTION : + dstattr = ippAddResolutions(dst, srcattr->group_tag, srcattr->name, + srcattr->num_values, IPP_RES_PER_INCH, + NULL, NULL); + if (!dstattr) + break; + + for (i = srcattr->num_values, srcval = srcattr->values, dstval = dstattr->values; + i > 0; + i --, srcval ++, dstval ++) + { + dstval->resolution.xres = srcval->resolution.xres; + dstval->resolution.yres = srcval->resolution.yres; + dstval->resolution.units = srcval->resolution.units; + } + break; + + case IPP_TAG_RANGE : + dstattr = ippAddRanges(dst, srcattr->group_tag, srcattr->name, + srcattr->num_values, NULL, NULL); + if (!dstattr) + break; + + for (i = srcattr->num_values, srcval = srcattr->values, dstval = dstattr->values; + i > 0; + i --, srcval ++, dstval ++) + { + dstval->range.lower = srcval->range.lower; + dstval->range.upper = srcval->range.upper; + } + break; + + case IPP_TAG_TEXTLANG : + case IPP_TAG_NAMELANG : + dstattr = ippAddStrings(dst, srcattr->group_tag, + (ipp_tag_t)(srcattr->value_tag | quickcopy), + srcattr->name, srcattr->num_values, NULL, NULL); + if (!dstattr) + break; + + if (quickcopy) + { + for (i = srcattr->num_values, srcval = srcattr->values, + dstval = dstattr->values; + i > 0; + i --, srcval ++, dstval ++) + { + dstval->string.language = srcval->string.language; + dstval->string.text = srcval->string.text; + } + } + else if (srcattr->value_tag & IPP_TAG_COPY) + { + for (i = srcattr->num_values, srcval = srcattr->values, + dstval = dstattr->values; + i > 0; + i --, srcval ++, dstval ++) + { + if (srcval == srcattr->values) + dstval->string.language = _cupsStrAlloc(srcval->string.language); + else + dstval->string.language = dstattr->values[0].string.language; + + dstval->string.text = _cupsStrAlloc(srcval->string.text); + } + } + else + { + for (i = srcattr->num_values, srcval = srcattr->values, + dstval = dstattr->values; + i > 0; + i --, srcval ++, dstval ++) + { + if (srcval == srcattr->values) + dstval->string.language = _cupsStrRetain(srcval->string.language); + else + dstval->string.language = dstattr->values[0].string.language; + + dstval->string.text = _cupsStrRetain(srcval->string.text); + } + } + break; + + case IPP_TAG_BEGIN_COLLECTION : + dstattr = ippAddCollections(dst, srcattr->group_tag, srcattr->name, + srcattr->num_values, NULL); + if (!dstattr) + break; + + for (i = srcattr->num_values, srcval = srcattr->values, dstval = dstattr->values; + i > 0; + i --, srcval ++, dstval ++) + { + dstval->collection = srcval->collection; + srcval->collection->use ++; + } + break; + + case IPP_TAG_STRING : + default : + /* TODO: Implement quick copy for unknown/octetString values */ + dstattr = ippAddIntegers(dst, srcattr->group_tag, srcattr->value_tag, + srcattr->name, srcattr->num_values, NULL); + if (!dstattr) + break; + + for (i = srcattr->num_values, srcval = srcattr->values, dstval = dstattr->values; + i > 0; + i --, srcval ++, dstval ++) + { + dstval->unknown.length = srcval->unknown.length; + + if (dstval->unknown.length > 0) + { + if ((dstval->unknown.data = malloc(dstval->unknown.length)) == NULL) + dstval->unknown.length = 0; + else + memcpy(dstval->unknown.data, srcval->unknown.data, dstval->unknown.length); + } + } + break; /* anti-compiler-warning-code */ + } + + return (dstattr); +} + + +/* + * 'ippCopyAttributes()' - Copy attributes from one IPP message to another. + * + * Zero or more attributes are copied from the source IPP message, @code@ src, to the + * destination IPP message, @code dst@. When @code quickcopy@ is non-zero, a "shallow" + * reference copy of the attribute is created - this should only be done as long as the + * original source IPP message will not be freed for the life of the destination. + * + * The @code cb@ and @code context@ parameters provide a generic way to "filter" the + * attributes that are copied - the function must return 1 to copy the attribute or + * 0 to skip it. The function may also choose to do a partial copy of the source attribute + * itself. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on error */ +ippCopyAttributes( + ipp_t *dst, /* I - Destination IPP message */ + ipp_t *src, /* I - Source IPP message */ + int quickcopy, /* I - 1 for a referenced copy, 0 for normal */ + ipp_copycb_t cb, /* I - Copy callback or @code NULL@ for none */ + void *context) /* I - Context pointer */ +{ + ipp_attribute_t *srcattr; /* Source attribute */ + + + DEBUG_printf(("ippCopyAttributes(dst=%p, src=%p, quickcopy=%d, cb=%p, context=%p)", + dst, src, quickcopy, cb, context)); + + /* + * Range check input... + */ + + if (!dst || !src) + return (0); + + /* + * Loop through source attributes and copy as needed... + */ + + for (srcattr = src->attrs; srcattr; srcattr = srcattr->next) + if (!cb || (*cb)(context, dst, srcattr)) + if (!ippCopyAttribute(dst, srcattr, quickcopy)) + return (0); + + return (1); +} + + +/* + * 'ippDateToTime()' - Convert from RFC 1903 Date/Time format to UNIX time + * in seconds. + */ + +time_t /* O - UNIX time value */ +ippDateToTime(const ipp_uchar_t *date) /* I - RFC 1903 date info */ +{ + struct tm unixdate; /* UNIX date/time info */ + time_t t; /* Computed time */ + + + if (!date) + return (0); + + memset(&unixdate, 0, sizeof(unixdate)); + + /* + * RFC-1903 date/time format is: + * + * Byte(s) Description + * ------- ----------- + * 0-1 Year (0 to 65535) + * 2 Month (1 to 12) + * 3 Day (1 to 31) + * 4 Hours (0 to 23) + * 5 Minutes (0 to 59) + * 6 Seconds (0 to 60, 60 = "leap second") + * 7 Deciseconds (0 to 9) + * 8 +/- UTC + * 9 UTC hours (0 to 11) + * 10 UTC minutes (0 to 59) + */ + + unixdate.tm_year = ((date[0] << 8) | date[1]) - 1900; + unixdate.tm_mon = date[2] - 1; + unixdate.tm_mday = date[3]; + unixdate.tm_hour = date[4]; + unixdate.tm_min = date[5]; + unixdate.tm_sec = date[6]; + + t = mktime(&unixdate); + + if (date[8] == '-') + t += date[9] * 3600 + date[10] * 60; + else + t -= date[9] * 3600 + date[10] * 60; + + return (t); +} + + +/* + * 'ippDelete()' - Delete an IPP message. + */ + +void +ippDelete(ipp_t *ipp) /* I - IPP message */ +{ + ipp_attribute_t *attr, /* Current attribute */ + *next; /* Next attribute */ + + + DEBUG_printf(("ippDelete(ipp=%p)", ipp)); + + if (!ipp) + return; + + ipp->use --; + if (ipp->use > 0) + return; + + for (attr = ipp->attrs; attr != NULL; attr = next) + { + next = attr->next; + + ipp_free_values(attr, 0, attr->num_values); + + if (attr->name) + _cupsStrFree(attr->name); + + free(attr); + } + + free(ipp); +} + + +/* + * 'ippDeleteAttribute()' - Delete a single attribute in an IPP message. + * + * @since CUPS 1.1.19/Mac OS X 10.3@ + */ + +void +ippDeleteAttribute( + ipp_t *ipp, /* I - IPP message */ + ipp_attribute_t *attr) /* I - Attribute to delete */ +{ + ipp_attribute_t *current, /* Current attribute */ + *prev; /* Previous attribute */ + + + DEBUG_printf(("ippDeleteAttribute(ipp=%p, attr=%p(%s))", ipp, attr, + attr ? attr->name : "(null)")); + + /* + * Range check input... + */ + + if (!attr) + return; + + /* + * Find the attribute in the list... + */ + + if (ipp) + { + for (current = ipp->attrs, prev = NULL; + current; + prev = current, current = current->next) + if (current == attr) + { + /* + * Found it, remove the attribute from the list... + */ + + if (prev) + prev->next = current->next; + else + ipp->attrs = current->next; + + if (current == ipp->last) + ipp->last = prev; + + break; + } + + if (!current) + return; + } + + /* + * Free memory used by the attribute... + */ + + ipp_free_values(attr, 0, attr->num_values); + + if (attr->name) + _cupsStrFree(attr->name); + + free(attr); +} + + +/* + * 'ippDeleteValues()' - Delete values in an attribute. + * + * The @code element@ parameter specifies the first value to delete, starting at + * 0. It must be less than the number of values returned by @link ippGetCount@. + * + * The @code attr@ parameter may be modified as a result of setting the value. + * + * Deleting all values in an attribute deletes the attribute. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +ippDeleteValues( + ipp_t *ipp, /* I - IPP message */ + ipp_attribute_t **attr, /* IO - Attribute */ + int element, /* I - Index of first value to delete (0-based) */ + int count) /* I - Number of values to delete */ +{ + /* + * Range check input... + */ + + if (!ipp || !attr || !*attr || + element < 0 || element >= (*attr)->num_values || count <= 0 || + (element + count) >= (*attr)->num_values) + return (0); + + /* + * If we are deleting all values, just delete the attribute entirely. + */ + + if (count == (*attr)->num_values) + { + ippDeleteAttribute(ipp, *attr); + *attr = NULL; + return (1); + } + + /* + * Otherwise free the values in question and return. + */ + + ipp_free_values(*attr, element, count); + + return (1); +} + + +/* + * 'ippFindAttribute()' - Find a named attribute in a request. + */ + +ipp_attribute_t * /* O - Matching attribute */ +ippFindAttribute(ipp_t *ipp, /* I - IPP message */ + const char *name, /* I - Name of attribute */ + ipp_tag_t type) /* I - Type of attribute */ +{ + DEBUG_printf(("2ippFindAttribute(ipp=%p, name=\"%s\", type=%02x(%s))", ipp, + name, type, ippTagString(type))); + + if (!ipp || !name) + return (NULL); + + /* + * Reset the current pointer... + */ + + ipp->current = NULL; + + /* + * Search for the attribute... + */ + + return (ippFindNextAttribute(ipp, name, type)); +} + + +/* + * 'ippFindNextAttribute()' - Find the next named attribute in a request. + */ + +ipp_attribute_t * /* O - Matching attribute */ +ippFindNextAttribute(ipp_t *ipp, /* I - IPP message */ + const char *name, /* I - Name of attribute */ + ipp_tag_t type) /* I - Type of attribute */ +{ + ipp_attribute_t *attr; /* Current atttribute */ + ipp_tag_t value_tag; /* Value tag */ + + + DEBUG_printf(("2ippFindNextAttribute(ipp=%p, name=\"%s\", type=%02x(%s))", + ipp, name, type, ippTagString(type))); + + if (!ipp || !name) + return (NULL); + + if (ipp->current) + { + ipp->prev = ipp->current; + attr = ipp->current->next; + } + else + { + ipp->prev = NULL; + attr = ipp->attrs; + } + + for (; attr != NULL; ipp->prev = attr, attr = attr->next) + { + DEBUG_printf(("4ippFindAttribute: attr=%p, name=\"%s\"", attr, + attr->name)); + + value_tag = (ipp_tag_t)(attr->value_tag & IPP_TAG_MASK); + + if (attr->name != NULL && _cups_strcasecmp(attr->name, name) == 0 && + (value_tag == type || type == IPP_TAG_ZERO || + (value_tag == IPP_TAG_TEXTLANG && type == IPP_TAG_TEXT) || + (value_tag == IPP_TAG_NAMELANG && type == IPP_TAG_NAME))) + { + ipp->current = attr; + + return (attr); + } + } + + ipp->current = NULL; + ipp->prev = NULL; + + return (NULL); +} + + +/* + * 'ippFirstAttribute()' - Return the first attribute in the message. + * + * @since CUPS 1.6@ + */ + +ipp_attribute_t * /* O - First attribute or @code NULL@ if none */ +ippFirstAttribute(ipp_t *ipp) /* I - IPP message */ +{ + /* + * Range check input... + */ + + if (!ipp) + return (NULL); + + /* + * Return the first attribute... + */ + + return (ipp->current = ipp->attrs); +} + + +/* + * 'ippGetBoolean()' - Get a boolean value for an attribute. + * + * The @code element@ parameter specifies which value to get from 0 to + * @link ippGetCount(attr)@ - 1. + * + * @since CUPS 1.6@ + */ + +int /* O - Boolean value or -1 on error */ +ippGetBoolean(ipp_attribute_t *attr, /* I - IPP attribute */ + int element) /* I - Value number (0-based) */ +{ + /* + * Range check input... + */ + + if (!attr || attr->value_tag != IPP_TAG_BOOLEAN || + element < 0 || element >= attr->num_values) + return (-1); + + /* + * Return the value... + */ + + return (attr->values[element].boolean); +} + + +/* + * 'ippGetCollection()' - Get a collection value for an attribute. + * + * The @code element@ parameter specifies which value to get from 0 to + * @link ippGetCount(attr)@ - 1. + * + * @since CUPS 1.6@ + */ + +ipp_t * /* O - Collection value or @code NULL@ on error */ +ippGetCollection( + ipp_attribute_t *attr, /* I - IPP attribute */ + int element) /* I - Value number (0-based) */ +{ + /* + * Range check input... + */ + + if (!attr || attr->value_tag != IPP_TAG_BEGIN_COLLECTION || + element < 0 || element >= attr->num_values) + return (NULL); + + /* + * Return the value... + */ + + return (attr->values[element].collection); +} + + +/* + * 'ippGetCount()' - Get the number of values in an attribute. + * + * @since CUPS 1.6@ + */ + +int /* O - Number of values or -1 on error */ +ippGetCount(ipp_attribute_t *attr) /* I - IPP attribute */ +{ + /* + * Range check input... + */ + + if (!attr) + return (-1); + + /* + * Return the number of values... + */ + + return (attr->num_values); +} + + +/* + * 'ippGetDate()' - Get a date value for an attribute. + * + * The @code element@ parameter specifies which value to get from 0 to + * @link ippGetCount(attr)@ - 1. + * + * @since CUPS 1.6@ + */ + +const ipp_uchar_t * /* O - Date value or @code NULL@ */ +ippGetDate(ipp_attribute_t *attr, /* I - IPP attribute */ + int element) /* I - Value number (0-based) */ +{ + /* + * Range check input... + */ + + if (!attr || attr->value_tag != IPP_TAG_DATE || + element < 0 || element >= attr->num_values) + return (NULL); + + /* + * Return the value... + */ + + return (attr->values[element].date); +} + + +/* + * 'ippGetGroupTag()' - Get the group associated with an attribute. + * + * @since CUPS 1.6@ + */ + +ipp_tag_t /* O - Group tag or @code IPP_TAG_ZERO@ on error */ +ippGetGroupTag(ipp_attribute_t *attr) /* I - IPP attribute */ +{ + /* + * Range check input... + */ + + if (!attr) + return (IPP_TAG_ZERO); + + /* + * Return the group... + */ + + return (attr->group_tag); +} + + +/* + * 'ippGetInteger()' - Get the integer/enum value for an attribute. + * + * The @code element@ parameter specifies which value to get from 0 to + * @link ippGetCount(attr)@ - 1. + * + * @since CUPS 1.6@ + */ + +int /* O - Value or -1 on error */ +ippGetInteger(ipp_attribute_t *attr, /* I - IPP attribute */ + int element) /* I - Value number (0-based) */ +{ + /* + * Range check input... + */ + + if (!attr || (attr->value_tag != IPP_TAG_INTEGER && attr->value_tag != IPP_TAG_ENUM) || + element < 0 || element >= attr->num_values) + return (-1); + + /* + * Return the value... + */ + + return (attr->values[element].integer); +} + + +/* + * 'ippGetName()' - Get the attribute name. + * + * @since CUPS 1.6@ + */ + +const char * /* O - Attribute name or @code NULL@ for separators */ +ippGetName(ipp_attribute_t *attr) /* I - IPP attribute */ +{ + /* + * Range check input... + */ + + if (!attr) + return (NULL); + + /* + * Return the name... + */ + + return (attr->name); +} + + +/* + * 'ippGetOperation()' - Get the operation ID in an IPP message. + * + * @since CUPS 1.6@ + */ + +ipp_op_t /* O - Operation ID or -1 on error */ +ippGetOperation(ipp_t *ipp) /* I - IPP request message */ +{ + /* + * Range check input... + */ + + if (!ipp) + return ((ipp_op_t)-1); + + /* + * Return the value... + */ + + return (ipp->request.op.operation_id); +} + + +/* + * 'ippGetRange()' - Get a rangeOfInteger value from an attribute. + * + * The @code element@ parameter specifies which value to get from 0 to + * @link ippGetCount(attr)@ - 1. + * + * @since CUPS 1.6@ + */ + +int /* O - Lower value of range or -1 */ +ippGetRange(ipp_attribute_t *attr, /* I - IPP attribute */ + int element, /* I - Value number (0-based) */ + int *uppervalue)/* O - Upper value of range */ +{ + /* + * Range check input... + */ + + if (!attr || attr->value_tag != IPP_TAG_RANGE || + element < 0 || element >= attr->num_values) + { + if (uppervalue) + *uppervalue = -1; + + return (-1); + } + + /* + * Return the values... + */ + + if (uppervalue) + *uppervalue = attr->values[element].range.upper; + + return (attr->values[element].range.lower); +} + + +/* + * 'ippGetRequestId()' - Get the request ID from an IPP message. + * + * @since CUPS 1.6@ + */ + +int /* O - Request ID or -1 on error */ +ippGetRequestId(ipp_t *ipp) /* I - IPP message */ +{ + /* + * Range check input... + */ + + if (!ipp) + return (-1); + + /* + * Return the request ID... + */ + + return (ipp->request.any.request_id); +} + + +/* + * 'ippGetResolution()' - Get a resolution value for an attribute. + * + * The @code element@ parameter specifies which value to get from 0 to + * @link ippGetCount(attr)@ - 1. + * + * @since CUPS 1.6@ + */ + +int /* O - Horizontal/cross feed resolution or -1 */ +ippGetResolution( + ipp_attribute_t *attr, /* I - IPP attribute */ + int element, /* I - Value number (0-based) */ + int *yres, /* O - Vertical/feed resolution */ + ipp_res_t *units) /* O - Units for resolution */ +{ + /* + * Range check input... + */ + + if (!attr || attr->value_tag != IPP_TAG_RESOLUTION || + element < 0 || element >= attr->num_values) + return (-1); + + /* + * Return the value... + */ + + if (yres) + *yres = attr->values[element].resolution.yres; + + if (units) + *units = attr->values[element].resolution.units; + + return (attr->values[element].resolution.xres); +} + + +/* + * 'ippGetState()' - Get the IPP message state. + * + * @since CUPS 1.6@ + */ + +ipp_state_t /* O - IPP message state value */ +ippGetState(ipp_t *ipp) /* I - IPP message */ +{ + /* + * Range check input... + */ + + if (!ipp) + return (IPP_IDLE); + + /* + * Return the value... + */ + + return (ipp->state); +} + + +/* + * 'ippGetStatusCode()' - Get the status code from an IPP response or event message. + * + * @since CUPS 1.6@ + */ + +ipp_status_t /* O - Status code in IPP message */ +ippGetStatusCode(ipp_t *ipp) /* I - IPP response or event message */ +{ + /* + * Range check input... + */ + + if (!ipp) + return (IPP_INTERNAL_ERROR); + + /* + * Return the value... + */ + + return (ipp->request.status.status_code); +} + + +/* + * 'ippGetString()' - Get the string and optionally the language code for an attribute. + * + * The @code element@ parameter specifies which value to get from 0 to + * @link ippGetCount(attr)@ - 1. + * + * @since CUPS 1.6@ + */ + +const char * +ippGetString(ipp_attribute_t *attr, /* I - IPP attribute */ + int element, /* I - Value number (0-based) */ + const char **language)/* O - Language code (@code NULL@ for don't care) */ +{ + /* + * Range check input... + */ + + if (!attr || element < 0 || element >= attr->num_values || + (attr->value_tag != IPP_TAG_TEXTLANG && attr->value_tag != IPP_TAG_NAMELANG && + (attr->value_tag < IPP_TAG_TEXT || attr->value_tag > IPP_TAG_MIMETYPE))) + return (NULL); + + /* + * Return the value... + */ + + if (language) + *language = attr->values[element].string.language; + + return (attr->values[element].string.text); +} + + +/* + * 'ippGetValueTag()' - Get the value tag for an attribute. + * + * @since CUPS 1.6@ + */ + +ipp_tag_t /* O - Value tag or @code IPP_TAG_ZERO@ on error */ +ippGetValueTag(ipp_attribute_t *attr) /* I - IPP attribute */ +{ + /* + * Range check input... + */ + + if (!attr) + return (IPP_TAG_ZERO); + + /* + * Return the value... + */ + + return (attr->value_tag); +} + + +/* + * 'ippGetVersion()' - Get the major and minor version number from an IPP message. + * + * @since CUPS 1.6@ + */ + +int /* O - Major version number or -1 on error */ +ippGetVersion(ipp_t *ipp, /* I - IPP message */ + int *minor) /* O - Minor version number or @code NULL@ */ +{ + /* + * Range check input... + */ + + if (!ipp) + { + if (minor) + *minor = -1; + + return (-1); + } + + /* + * Return the value... + */ + + if (minor) + *minor = ipp->request.any.version[1]; + + return (ipp->request.any.version[0]); +} + + +/* + * 'ippLength()' - Compute the length of an IPP message. + */ + +size_t /* O - Size of IPP message */ +ippLength(ipp_t *ipp) /* I - IPP message */ +{ + return (ipp_length(ipp, 0)); +} + + +/* + * 'ippNextAttribute()' - Return the next attribute in the message. + * + * @since CUPS 1.6@ + */ + +ipp_attribute_t * /* O - Next attribute or @code NULL@ if none */ +ippNextAttribute(ipp_t *ipp) /* I - IPP message */ +{ + /* + * Range check input... + */ + + if (!ipp || !ipp->current) + return (NULL); + + /* + * Return the next attribute... + */ + + return (ipp->current = ipp->current->next); +} + + +/* + * 'ippNew()' - Allocate a new IPP message. + */ + +ipp_t * /* O - New IPP message */ +ippNew(void) +{ + ipp_t *temp; /* New IPP message */ + + + DEBUG_puts("ippNew()"); + + if ((temp = (ipp_t *)calloc(1, sizeof(ipp_t))) != NULL) + { + /* + * Default to IPP 2.0... + */ + + temp->request.any.version[0] = 2; + temp->request.any.version[1] = 0; + temp->use = 1; + } + + DEBUG_printf(("1ippNew: Returning %p", temp)); + + return (temp); +} + + +/* + * 'ippNewRequest()' - Allocate a new IPP request message. + * + * The new request message is initialized with the attributes-charset and + * attributes-natural-language attributes added. The + * attributes-natural-language value is derived from the current locale. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ipp_t * /* O - IPP request message */ +ippNewRequest(ipp_op_t op) /* I - Operation code */ +{ + ipp_t *request; /* IPP request message */ + cups_lang_t *language; /* Current language localization */ + static int request_id = 0; /* Current request ID */ + static _cups_mutex_t request_mutex = _CUPS_MUTEX_INITIALIZER; + /* Mutex for request ID */ + + + DEBUG_printf(("ippNewRequest(op=%02x(%s))", op, ippOpString(op))); + + /* + * Create a new IPP message... + */ + + if ((request = ippNew()) == NULL) + return (NULL); + + /* + * Set the operation and request ID... + */ + + _cupsMutexLock(&request_mutex); + + request->request.op.operation_id = op; + request->request.op.request_id = ++request_id; + + _cupsMutexUnlock(&request_mutex); + + /* + * Use UTF-8 as the character set... + */ + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_CHARSET, + "attributes-charset", NULL, "utf-8"); + + /* + * Get the language from the current locale... + */ + + language = cupsLangDefault(); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_LANGUAGE, + "attributes-natural-language", NULL, language->language); + + /* + * Return the new request... + */ + + return (request); +} + + +/* + * 'ippRead()' - Read data for an IPP message from a HTTP connection. + */ + +ipp_state_t /* O - Current state */ +ippRead(http_t *http, /* I - HTTP connection */ + ipp_t *ipp) /* I - IPP data */ +{ + DEBUG_printf(("ippRead(http=%p, ipp=%p), data_remaining=" CUPS_LLFMT, + http, ipp, CUPS_LLCAST (http ? http->data_remaining : -1))); + + if (!http) + return (IPP_ERROR); + + DEBUG_printf(("2ippRead: http->state=%d, http->used=%d", http->state, + http->used)); + + return (ippReadIO(http, (ipp_iocb_t)ipp_read_http, http->blocking, NULL, + ipp)); +} + + +/* + * 'ippReadFile()' - Read data for an IPP message from a file. + * + * @since CUPS 1.1.19/Mac OS X 10.3@ + */ + +ipp_state_t /* O - Current state */ +ippReadFile(int fd, /* I - HTTP data */ + ipp_t *ipp) /* I - IPP data */ +{ + DEBUG_printf(("ippReadFile(fd=%d, ipp=%p)", fd, ipp)); + + return (ippReadIO(&fd, (ipp_iocb_t)ipp_read_file, 1, NULL, ipp)); +} + + +/* + * 'ippReadIO()' - Read data for an IPP message. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ipp_state_t /* O - Current state */ +ippReadIO(void *src, /* I - Data source */ + ipp_iocb_t cb, /* I - Read callback function */ + int blocking, /* I - Use blocking IO? */ + ipp_t *parent, /* I - Parent request, if any */ + ipp_t *ipp) /* I - IPP data */ +{ + int n; /* Length of data */ + unsigned char *buffer, /* Data buffer */ + string[IPP_MAX_NAME], + /* Small string buffer */ + *bufptr; /* Pointer into buffer */ + ipp_attribute_t *attr; /* Current attribute */ + ipp_tag_t tag; /* Current tag */ + ipp_tag_t value_tag; /* Current value tag */ + _ipp_value_t *value; /* Current value */ + + + DEBUG_printf(("ippReadIO(src=%p, cb=%p, blocking=%d, parent=%p, ipp=%p)", + src, cb, blocking, parent, ipp)); + DEBUG_printf(("2ippReadIO: ipp->state=%d", ipp->state)); + + if (!src || !ipp) + return (IPP_ERROR); + + if ((buffer = (unsigned char *)_cupsBufferGet(IPP_BUF_SIZE)) == NULL) + { + DEBUG_puts("1ippReadIO: Unable to get read buffer."); + return (IPP_ERROR); + } + + switch (ipp->state) + { + case IPP_IDLE : + ipp->state ++; /* Avoid common problem... */ + + case IPP_HEADER : + if (parent == NULL) + { + /* + * Get the request header... + */ + + if ((*cb)(src, buffer, 8) < 8) + { + DEBUG_puts("1ippReadIO: Unable to read header."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + /* + * Then copy the request header over... + */ + + ipp->request.any.version[0] = buffer[0]; + ipp->request.any.version[1] = buffer[1]; + ipp->request.any.op_status = (buffer[2] << 8) | buffer[3]; + ipp->request.any.request_id = (((((buffer[4] << 8) | buffer[5]) << 8) | + buffer[6]) << 8) | buffer[7]; + + DEBUG_printf(("2ippReadIO: version=%d.%d", buffer[0], buffer[1])); + DEBUG_printf(("2ippReadIO: op_status=%04x", + ipp->request.any.op_status)); + DEBUG_printf(("2ippReadIO: request_id=%d", + ipp->request.any.request_id)); + } + + ipp->state = IPP_ATTRIBUTE; + ipp->current = NULL; + ipp->curtag = IPP_TAG_ZERO; + ipp->prev = ipp->last; + + /* + * If blocking is disabled, stop here... + */ + + if (!blocking) + break; + + case IPP_ATTRIBUTE : + for (;;) + { + if ((*cb)(src, buffer, 1) < 1) + { + DEBUG_puts("1ippReadIO: Callback returned EOF/error"); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + DEBUG_printf(("2ippReadIO: ipp->current=%p, ipp->prev=%p", + ipp->current, ipp->prev)); + + /* + * Read this attribute... + */ + + tag = (ipp_tag_t)buffer[0]; + if (tag == IPP_TAG_EXTENSION) + { + /* + * Read 32-bit "extension" tag... + */ + + if ((*cb)(src, buffer, 4) < 1) + { + DEBUG_puts("1ippReadIO: Callback returned EOF/error"); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + tag = (ipp_tag_t)((((((buffer[0] << 8) | buffer[1]) << 8) | + buffer[2]) << 8) | buffer[3]); + + if (tag & IPP_TAG_COPY) + { + /* + * Fail if the high bit is set in the tag... + */ + + _cupsSetError(IPP_INTERNAL_ERROR, _("IPP extension tag larger than 0x7FFFFFFF."), 1); + DEBUG_printf(("1ippReadIO: bad name length %d.", n)); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + } + + if (tag == IPP_TAG_END) + { + /* + * No more attributes left... + */ + + DEBUG_puts("2ippReadIO: IPP_TAG_END."); + + ipp->state = IPP_DATA; + break; + } + else if (tag < IPP_TAG_UNSUPPORTED_VALUE) + { + /* + * Group tag... Set the current group and continue... + */ + + if (ipp->curtag == tag) + ipp->prev = ippAddSeparator(ipp); + else if (ipp->current) + ipp->prev = ipp->current; + + ipp->curtag = tag; + ipp->current = NULL; + DEBUG_printf(("2ippReadIO: group tag=%x(%s), ipp->prev=%p", tag, + ippTagString(tag), ipp->prev)); + continue; + } + + DEBUG_printf(("2ippReadIO: value tag=%x(%s)", tag, + ippTagString(tag))); + + /* + * Get the name... + */ + + if ((*cb)(src, buffer, 2) < 2) + { + DEBUG_puts("1ippReadIO: unable to read name length."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + n = (buffer[0] << 8) | buffer[1]; + + if (n >= IPP_BUF_SIZE) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("IPP name larger than 32767 bytes."), 1); + DEBUG_printf(("1ippReadIO: bad name length %d.", n)); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + DEBUG_printf(("2ippReadIO: name length=%d", n)); + + if (n == 0 && tag != IPP_TAG_MEMBERNAME && + tag != IPP_TAG_END_COLLECTION) + { + /* + * More values for current attribute... + */ + + if (ipp->current == NULL) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("IPP attribute has no name."), 1); + DEBUG_puts("1ippReadIO: Attribute without name and no current."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + attr = ipp->current; + value_tag = (ipp_tag_t)(attr->value_tag & IPP_TAG_MASK); + + /* + * Make sure we aren't adding a new value of a different + * type... + */ + + if (value_tag == IPP_TAG_ZERO) + { + /* + * Setting the value of a collection member... + */ + + attr->value_tag = tag; + } + else if (value_tag == IPP_TAG_TEXTLANG || + value_tag == IPP_TAG_NAMELANG || + (value_tag >= IPP_TAG_TEXT && + value_tag <= IPP_TAG_MIMETYPE)) + { + /* + * String values can sometimes come across in different + * forms; accept sets of differing values... + */ + + if (tag != IPP_TAG_TEXTLANG && tag != IPP_TAG_NAMELANG && + (tag < IPP_TAG_TEXT || tag > IPP_TAG_MIMETYPE) && + tag != IPP_TAG_NOVALUE) + { + _cupsSetError(IPP_INTERNAL_ERROR, + _("IPP 1setOf attribute with incompatible value " + "tags."), 1); + DEBUG_printf(("1ippReadIO: 1setOf value tag %x(%s) != %x(%s)", + value_tag, ippTagString(value_tag), tag, + ippTagString(tag))); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + if (value_tag != tag) + { + DEBUG_printf(("1ippReadIO: Converting %s attribute from %s to %s.", + attr->name, ippTagString(value_tag), ippTagString(tag))); + ippSetValueTag(ipp, &attr, tag); + } + } + else if (value_tag == IPP_TAG_INTEGER || + value_tag == IPP_TAG_RANGE) + { + /* + * Integer and rangeOfInteger values can sometimes be mixed; accept + * sets of differing values... + */ + + if (tag != IPP_TAG_INTEGER && tag != IPP_TAG_RANGE) + { + _cupsSetError(IPP_INTERNAL_ERROR, + _("IPP 1setOf attribute with incompatible value " + "tags."), 1); + DEBUG_printf(("1ippReadIO: 1setOf value tag %x(%s) != %x(%s)", + value_tag, ippTagString(value_tag), tag, + ippTagString(tag))); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + if (value_tag == IPP_TAG_INTEGER && tag == IPP_TAG_RANGE) + { + /* + * Convert integer values to rangeOfInteger values... + */ + + DEBUG_printf(("1ippReadIO: Converting %s attribute to " + "rangeOfInteger.", attr->name)); + ippSetValueTag(ipp, &attr, IPP_TAG_RANGE); + } + } + else if (value_tag != tag) + { + _cupsSetError(IPP_INTERNAL_ERROR, + _("IPP 1setOf attribute with incompatible value " + "tags."), 1); + DEBUG_printf(("1ippReadIO: value tag %x(%s) != %x(%s)", + value_tag, ippTagString(value_tag), tag, + ippTagString(tag))); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + /* + * Finally, reallocate the attribute array as needed... + */ + + if ((value = ipp_set_value(ipp, &attr, attr->num_values)) == NULL) + { + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + } + else if (tag == IPP_TAG_MEMBERNAME) + { + /* + * Name must be length 0! + */ + + if (n) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("IPP member name is not empty."), 1); + DEBUG_puts("1ippReadIO: member name not empty."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + if (ipp->current) + ipp->prev = ipp->current; + + attr = ipp->current = ipp_add_attr(ipp, NULL, ipp->curtag, IPP_TAG_ZERO, 1); + + DEBUG_printf(("2ippReadIO: membername, ipp->current=%p, ipp->prev=%p", + ipp->current, ipp->prev)); + + value = attr->values; + } + else if (tag != IPP_TAG_END_COLLECTION) + { + /* + * New attribute; read the name and add it... + */ + + if ((*cb)(src, buffer, n) < n) + { + DEBUG_puts("1ippReadIO: unable to read name."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + buffer[n] = '\0'; + + if (ipp->current) + ipp->prev = ipp->current; + + if ((attr = ipp->current = ipp_add_attr(ipp, (char *)buffer, ipp->curtag, tag, + 1)) == NULL) + { + _cupsSetHTTPError(HTTP_ERROR); + DEBUG_puts("1ippReadIO: unable to allocate attribute."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + DEBUG_printf(("2ippReadIO: name=\"%s\", ipp->current=%p, " + "ipp->prev=%p", buffer, ipp->current, ipp->prev)); + + value = attr->values; + } + else + { + attr = NULL; + value = NULL; + } + + if ((*cb)(src, buffer, 2) < 2) + { + DEBUG_puts("1ippReadIO: unable to read value length."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + n = (buffer[0] << 8) | buffer[1]; + DEBUG_printf(("2ippReadIO: value length=%d", n)); + + if (n >= IPP_BUF_SIZE) + { + _cupsSetError(IPP_INTERNAL_ERROR, + _("IPP value larger than 32767 bytes."), 1); + DEBUG_printf(("1ippReadIO: bad value length %d.", n)); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + switch (tag) + { + case IPP_TAG_INTEGER : + case IPP_TAG_ENUM : + if (n != 4) + { + if (tag == IPP_TAG_INTEGER) + _cupsSetError(IPP_INTERNAL_ERROR, + _("IPP integer value not 4 bytes."), 1); + else + _cupsSetError(IPP_INTERNAL_ERROR, + _("IPP enum value not 4 bytes."), 1); + DEBUG_printf(("1ippReadIO: bad value length %d.", n)); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + if ((*cb)(src, buffer, 4) < 4) + { + DEBUG_puts("1ippReadIO: Unable to read integer value."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + n = (((((buffer[0] << 8) | buffer[1]) << 8) | buffer[2]) << 8) | + buffer[3]; + + if (attr->value_tag == IPP_TAG_RANGE) + value->range.lower = value->range.upper = n; + else + value->integer = n; + break; + + case IPP_TAG_BOOLEAN : + if (n != 1) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("IPP boolean value not 1 byte."), + 1); + DEBUG_printf(("1ippReadIO: bad value length %d.", n)); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + if ((*cb)(src, buffer, 1) < 1) + { + DEBUG_puts("1ippReadIO: Unable to read boolean value."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + value->boolean = buffer[0]; + break; + + case IPP_TAG_NOVALUE : + case IPP_TAG_NOTSETTABLE : + case IPP_TAG_DELETEATTR : + case IPP_TAG_ADMINDEFINE : + /* + * These value types are not supposed to have values, however + * some vendors (Brother) do not implement IPP correctly and so + * we need to map non-empty values to text... + */ + + if (attr->value_tag == tag) + { + if (n == 0) + break; + + attr->value_tag = IPP_TAG_TEXT; + } + + case IPP_TAG_TEXT : + case IPP_TAG_NAME : + case IPP_TAG_KEYWORD : + case IPP_TAG_URI : + case IPP_TAG_URISCHEME : + case IPP_TAG_CHARSET : + case IPP_TAG_LANGUAGE : + case IPP_TAG_MIMETYPE : + if ((*cb)(src, buffer, n) < n) + { + DEBUG_puts("1ippReadIO: unable to read string value."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + buffer[n] = '\0'; + value->string.text = _cupsStrAlloc((char *)buffer); + DEBUG_printf(("2ippReadIO: value=\"%s\"", value->string.text)); + break; + + case IPP_TAG_DATE : + if (n != 11) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("IPP date value not 11 bytes."), 1); + DEBUG_printf(("1ippReadIO: bad value length %d.", n)); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + if ((*cb)(src, value->date, 11) < 11) + { + DEBUG_puts("1ippReadIO: Unable to read date value."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + break; + + case IPP_TAG_RESOLUTION : + if (n != 9) + { + _cupsSetError(IPP_INTERNAL_ERROR, + _("IPP resolution value not 9 bytes."), 1); + DEBUG_printf(("1ippReadIO: bad value length %d.", n)); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + if ((*cb)(src, buffer, 9) < 9) + { + DEBUG_puts("1ippReadIO: Unable to read resolution value."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + value->resolution.xres = + (((((buffer[0] << 8) | buffer[1]) << 8) | buffer[2]) << 8) | + buffer[3]; + value->resolution.yres = + (((((buffer[4] << 8) | buffer[5]) << 8) | buffer[6]) << 8) | + buffer[7]; + value->resolution.units = + (ipp_res_t)buffer[8]; + break; + + case IPP_TAG_RANGE : + if (n != 8) + { + _cupsSetError(IPP_INTERNAL_ERROR, + _("IPP rangeOfInteger value not 8 bytes."), 1); + DEBUG_printf(("1ippReadIO: bad value length %d.", n)); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + if ((*cb)(src, buffer, 8) < 8) + { + DEBUG_puts("1ippReadIO: Unable to read range value."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + value->range.lower = + (((((buffer[0] << 8) | buffer[1]) << 8) | buffer[2]) << 8) | + buffer[3]; + value->range.upper = + (((((buffer[4] << 8) | buffer[5]) << 8) | buffer[6]) << 8) | + buffer[7]; + break; + + case IPP_TAG_TEXTLANG : + case IPP_TAG_NAMELANG : + if (n < 4) + { + if (tag == IPP_TAG_TEXTLANG) + _cupsSetError(IPP_INTERNAL_ERROR, + _("IPP textWithLanguage value less than " + "minimum 4 bytes."), 1); + else + _cupsSetError(IPP_INTERNAL_ERROR, + _("IPP nameWithLanguage value less than " + "minimum 4 bytes."), 1); + DEBUG_printf(("1ippReadIO: bad value length %d.", n)); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + if ((*cb)(src, buffer, n) < n) + { + DEBUG_puts("1ippReadIO: Unable to read string w/language " + "value."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + bufptr = buffer; + + /* + * text-with-language and name-with-language are composite + * values: + * + * language-length + * language + * text-length + * text + */ + + n = (bufptr[0] << 8) | bufptr[1]; + + if ((bufptr + 2 + n) >= (buffer + IPP_BUF_SIZE) || + n >= sizeof(string)) + { + _cupsSetError(IPP_INTERNAL_ERROR, + _("IPP language length overflows value."), 1); + DEBUG_printf(("1ippReadIO: bad value length %d.", n)); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + memcpy(string, bufptr + 2, n); + string[n] = '\0'; + + value->string.language = _cupsStrAlloc((char *)string); + + bufptr += 2 + n; + n = (bufptr[0] << 8) | bufptr[1]; + + if ((bufptr + 2 + n) >= (buffer + IPP_BUF_SIZE)) + { + _cupsSetError(IPP_INTERNAL_ERROR, + _("IPP string length overflows value."), 1); + DEBUG_printf(("1ippReadIO: bad value length %d.", n)); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + bufptr[2 + n] = '\0'; + value->string.text = _cupsStrAlloc((char *)bufptr + 2); + break; + + case IPP_TAG_BEGIN_COLLECTION : + /* + * Oh, boy, here comes a collection value, so read it... + */ + + value->collection = ippNew(); + + if (n > 0) + { + _cupsSetError(IPP_INTERNAL_ERROR, + _("IPP begCollection value not 0 bytes."), 1); + DEBUG_puts("1ippReadIO: begCollection tag with value length " + "> 0."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + if (ippReadIO(src, cb, 1, ipp, value->collection) == IPP_ERROR) + { + DEBUG_puts("1ippReadIO: Unable to read collection value."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + break; + + case IPP_TAG_END_COLLECTION : + _cupsBufferRelease((char *)buffer); + + if (n > 0) + { + _cupsSetError(IPP_INTERNAL_ERROR, + _("IPP endCollection value not 0 bytes."), 1); + DEBUG_puts("1ippReadIO: endCollection tag with value length " + "> 0."); + return (IPP_ERROR); + } + + DEBUG_puts("1ippReadIO: endCollection tag..."); + return (ipp->state = IPP_DATA); + + case IPP_TAG_MEMBERNAME : + /* + * The value the name of the member in the collection, which + * we need to carry over... + */ + + if ((*cb)(src, buffer, n) < n) + { + DEBUG_puts("1ippReadIO: Unable to read member name value."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + buffer[n] = '\0'; + attr->name = _cupsStrAlloc((char *)buffer); + + /* + * Since collection members are encoded differently than + * regular attributes, make sure we don't start with an + * empty value... + */ + + attr->num_values --; + + DEBUG_printf(("2ippReadIO: member name=\"%s\"", attr->name)); + break; + + default : /* Other unsupported values */ + value->unknown.length = n; + if (n > 0) + { + if ((value->unknown.data = malloc(n)) == NULL) + { + _cupsSetHTTPError(HTTP_ERROR); + DEBUG_puts("1ippReadIO: Unable to allocate value"); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + if ((*cb)(src, value->unknown.data, n) < n) + { + DEBUG_puts("1ippReadIO: Unable to read unsupported value."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + } + else + value->unknown.data = NULL; + break; + } + + /* + * If blocking is disabled, stop here... + */ + + if (!blocking) + break; + } + break; + + case IPP_DATA : + break; + + default : + break; /* anti-compiler-warning-code */ + } + + DEBUG_printf(("1ippReadIO: returning ipp->state=%d.", ipp->state)); + _cupsBufferRelease((char *)buffer); + + return (ipp->state); +} + + +/* + * 'ippSetBoolean()' - Set a boolean value in an attribute. + * + * The @code ipp@ parameter refers to the IPP message containing the attribute that was + * previously created using the @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code attr@ parameter may be modified as a result of setting the value. + * + * The @code element@ parameter specifies which value to set from 0 to + * @link ippGetCount(attr)@. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +ippSetBoolean(ipp_t *ipp, /* IO - IPP message */ + ipp_attribute_t **attr, /* IO - IPP attribute */ + int element, /* I - Value number (0-based) */ + int boolvalue)/* I - Boolean value */ +{ + _ipp_value_t *value; /* Current value */ + + + /* + * Range check input... + */ + + if (!ipp || !attr || !*attr || (*attr)->value_tag != IPP_TAG_BOOLEAN || + element < 0 || element > (*attr)->num_values) + return (0); + + /* + * Set the value and return... + */ + + if ((value = ipp_set_value(ipp, attr, element)) != NULL) + value->boolean = boolvalue; + + return (value != NULL); +} + + +/* + * 'ippSetCollection()' - Set a collection value in an attribute. + * + * The @code ipp@ parameter refers to the IPP message containing the attribute that was + * previously created using the @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code attr@ parameter may be modified as a result of setting the value. + * + * The @code element@ parameter specifies which value to set from 0 to + * @link ippGetCount(attr)@. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +ippSetCollection( + ipp_t *ipp, /* IO - IPP message */ + ipp_attribute_t **attr, /* IO - IPP attribute */ + int element, /* I - Value number (0-based) */ + ipp_t *colvalue) /* I - Collection value */ +{ + _ipp_value_t *value; /* Current value */ + + + /* + * Range check input... + */ + + if (!ipp || !attr || !*attr || (*attr)->value_tag != IPP_TAG_BEGIN_COLLECTION || + element < 0 || element > (*attr)->num_values || !colvalue) + return (0); + + /* + * Set the value and return... + */ + + if ((value = ipp_set_value(ipp, attr, element)) != NULL) + { + if (value->collection) + ippDelete(value->collection); + + value->collection = colvalue; + colvalue->use ++; + } + + return (value != NULL); +} + + +/* + * 'ippSetDate()' - Set a date value in an attribute. + * + * The @code ipp@ parameter refers to the IPP message containing the attribute that was + * previously created using the @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code attr@ parameter may be modified as a result of setting the value. + * + * The @code element@ parameter specifies which value to set from 0 to + * @link ippGetCount(attr)@. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +ippSetDate(ipp_t *ipp, /* IO - IPP message */ + ipp_attribute_t **attr, /* IO - IPP attribute */ + int element, /* I - Value number (0-based) */ + const ipp_uchar_t *datevalue)/* I - Date value */ +{ + _ipp_value_t *value; /* Current value */ + + + /* + * Range check input... + */ + + if (!ipp || !attr || !*attr || (*attr)->value_tag != IPP_TAG_DATE || + element < 0 || element > (*attr)->num_values || !datevalue) + return (0); + + /* + * Set the value and return... + */ + + if ((value = ipp_set_value(ipp, attr, element)) != NULL) + memcpy(value->date, datevalue, sizeof(value->date)); + + return (value != NULL); +} + + +/* + * 'ippSetGroupTag()' - Set the group tag of an attribute. + * + * The @code ipp@ parameter refers to the IPP message containing the attribute that was + * previously created using the @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code attr@ parameter may be modified as a result of setting the value. + * + * The @code group@ parameter specifies the IPP attribute group tag: none + * (@code IPP_TAG_ZERO@, for member attributes), document (@code IPP_TAG_DOCUMENT@), + * event notification (@code IPP_TAG_EVENT_NOTIFICATION@), operation + * (@code IPP_TAG_OPERATION@), printer (@code IPP_TAG_PRINTER@), subscription + * (@code IPP_TAG_SUBSCRIPTION@), or unsupported (@code IPP_TAG_UNSUPPORTED_GROUP@). + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +ippSetGroupTag( + ipp_t *ipp, /* IO - IPP message */ + ipp_attribute_t **attr, /* IO - Attribute */ + ipp_tag_t group_tag) /* I - Group tag */ +{ + /* + * Range check input - group tag must be 0x01 to 0x0F, per RFC 2911... + */ + + if (!ipp || !attr || group_tag < IPP_TAG_ZERO || group_tag == IPP_TAG_END || + group_tag >= IPP_TAG_UNSUPPORTED_VALUE) + return (0); + + /* + * Set the group tag and return... + */ + + (*attr)->group_tag = group_tag; + + return (1); +} + + +/* + * 'ippSetInteger()' - Set an integer or enum value in an attribute. + * + * The @code ipp@ parameter refers to the IPP message containing the attribute that was + * previously created using the @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code attr@ parameter may be modified as a result of setting the value. + * + * The @code element@ parameter specifies which value to set from 0 to + * @link ippGetCount(attr)@. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +ippSetInteger(ipp_t *ipp, /* IO - IPP message */ + ipp_attribute_t **attr, /* IO - IPP attribute */ + int element, /* I - Value number (0-based) */ + int intvalue) /* I - Integer/enum value */ +{ + _ipp_value_t *value; /* Current value */ + + + /* + * Range check input... + */ + + if (!ipp || !attr || !*attr || + ((*attr)->value_tag != IPP_TAG_INTEGER && (*attr)->value_tag != IPP_TAG_ENUM) || + element < 0 || element > (*attr)->num_values) + return (0); + + /* + * Set the value and return... + */ + + if ((value = ipp_set_value(ipp, attr, element)) != NULL) + value->integer = intvalue; + + return (value != NULL); +} + + +/* + * 'ippSetName()' - Set the name of an attribute. + * + * The @code ipp@ parameter refers to the IPP message containing the attribute that was + * previously created using the @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code attr@ parameter may be modified as a result of setting the value. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +ippSetName(ipp_t *ipp, /* IO - IPP message */ + ipp_attribute_t **attr, /* IO - IPP attribute */ + const char *name) /* I - Attribute name */ +{ + char *temp; /* Temporary name value */ + + + /* + * Range check input... + */ + + if (!ipp || !attr || !*attr) + return (0); + + /* + * Set the value and return... + */ + + if ((temp = _cupsStrAlloc(name)) != NULL) + { + if ((*attr)->name) + _cupsStrFree((*attr)->name); + + (*attr)->name = temp; + } + + return (temp != NULL); +} + + +/* + * 'ippSetOperation()' - Set the operation ID in an IPP request message. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +ippSetOperation(ipp_t *ipp, /* I - IPP request message */ + ipp_op_t op) /* I - Operation ID */ +{ + /* + * Range check input... + */ + + if (!ipp) + return (0); + + /* + * Set the operation and return... + */ + + ipp->request.op.operation_id = op; + + return (1); +} + + +/* + * 'ippSetRange()' - Set a rangeOfInteger value in an attribute. + * + * The @code ipp@ parameter refers to the IPP message containing the attribute that was + * previously created using the @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code attr@ parameter may be modified as a result of setting the value. + * + * The @code element@ parameter specifies which value to set from 0 to + * @link ippGetCount(attr)@. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +ippSetRange(ipp_t *ipp, /* IO - IPP message */ + ipp_attribute_t **attr, /* IO - IPP attribute */ + int element, /* I - Value number (0-based) */ + int lowervalue, /* I - Lower bound for range */ + int uppervalue) /* I - Upper bound for range */ +{ + _ipp_value_t *value; /* Current value */ + + + /* + * Range check input... + */ + + if (!ipp || !attr || !*attr || (*attr)->value_tag != IPP_TAG_RANGE || + element < 0 || element > (*attr)->num_values || lowervalue > uppervalue) + return (0); + + /* + * Set the value and return... + */ + + if ((value = ipp_set_value(ipp, attr, element)) != NULL) + { + value->range.lower = lowervalue; + value->range.upper = uppervalue; + } + + return (value != NULL); +} + + +/* + * 'ippSetRequestId()' - Set the request ID in an IPP message. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code request_id@ parameter must be greater than 0. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +ippSetRequestId(ipp_t *ipp, /* I - IPP message */ + int request_id) /* I - Request ID */ +{ + /* + * Range check input; not checking request_id values since ipptool wants to send + * invalid values for conformance testing and a bad request_id does not affect the + * encoding of a message... + */ + + if (!ipp) + return (0); + + /* + * Set the request ID and return... + */ + + ipp->request.any.request_id = request_id; + + return (1); +} + + +/* + * 'ippSetResolution()' - Set a resolution value in an attribute. + * + * The @code ipp@ parameter refers to the IPP message containing the attribute that was + * previously created using the @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code attr@ parameter may be modified as a result of setting the value. + * + * The @code element@ parameter specifies which value to set from 0 to + * @link ippGetCount(attr)@. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +ippSetResolution( + ipp_t *ipp, /* IO - IPP message */ + ipp_attribute_t **attr, /* IO - IPP attribute */ + int element, /* I - Value number (0-based) */ + ipp_res_t unitsvalue, /* I - Resolution units */ + int xresvalue, /* I - Horizontal/cross feed resolution */ + int yresvalue) /* I - Vertical/feed resolution */ +{ + _ipp_value_t *value; /* Current value */ + + + /* + * Range check input... + */ + + if (!ipp || !attr || !*attr || (*attr)->value_tag != IPP_TAG_RESOLUTION || + element < 0 || element > (*attr)->num_values || xresvalue <= 0 || yresvalue <= 0 || + unitsvalue < IPP_RES_PER_INCH || unitsvalue > IPP_RES_PER_CM) + return (0); + + /* + * Set the value and return... + */ + + if ((value = ipp_set_value(ipp, attr, element)) != NULL) + { + value->resolution.units = unitsvalue; + value->resolution.xres = xresvalue; + value->resolution.yres = yresvalue; + } + + return (value != NULL); +} + + +/* + * 'ippSetState()' - Set the current state of the IPP message. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +ippSetState(ipp_t *ipp, /* I - IPP message */ + ipp_state_t state) /* I - IPP state value */ +{ + /* + * Range check input... + */ + + if (!ipp) + return (0); + + /* + * Set the state and return... + */ + + ipp->state = state; + ipp->current = NULL; + + return (1); +} + + +/* + * 'ippSetStatusCode()' - Set the status code in an IPP response or event message. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +ippSetStatusCode(ipp_t *ipp, /* I - IPP response or event message */ + ipp_status_t status) /* I - Status code */ +{ + /* + * Range check input... + */ + + if (!ipp) + return (0); + + /* + * Set the status code and return... + */ + + ipp->request.status.status_code = status; + + return (1); +} + + +/* + * 'ippSetString()' - Set a string value in an attribute. + * + * The @code ipp@ parameter refers to the IPP message containing the attribute that was + * previously created using the @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code attr@ parameter may be modified as a result of setting the value. + * + * The @code element@ parameter specifies which value to set from 0 to + * @link ippGetCount(attr)@. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +ippSetString(ipp_t *ipp, /* IO - IPP message */ + ipp_attribute_t **attr, /* IO - IPP attribute */ + int element, /* I - Value number (0-based) */ + const char *strvalue) /* I - String value */ +{ + char *temp; /* Temporary string */ + _ipp_value_t *value; /* Current value */ + + + /* + * Range check input... + */ + + if (!ipp || !attr || !*attr || (*attr)->value_tag != IPP_TAG_INTEGER || + element < 0 || element > (*attr)->num_values || !strvalue) + return (0); + + /* + * Set the value and return... + */ + + if ((value = ipp_set_value(ipp, attr, element)) != NULL) + { + if (element > 0) + value->string.language = (*attr)->values[0].string.language; + + if ((int)((*attr)->value_tag) & IPP_TAG_COPY) + value->string.text = (char *)strvalue; + else if ((temp = _cupsStrAlloc(strvalue)) != NULL) + { + if (value->string.text) + _cupsStrFree(value->string.text); + + value->string.text = temp; + } + else + return (0); + } + + return (value != NULL); +} + + +/* + * 'ippSetValueTag()' - Set the value tag of an attribute. + * + * The @code ipp@ parameter refers to the IPP message containing the attribute that was + * previously created using the @link ippNew@ or @link ippNewRequest@ functions. + * + * The @code attr@ parameter may be modified as a result of setting the value. + * + * Integer (@code IPP_TAG_INTEGER@) values can be promoted to rangeOfInteger + * (@code IPP_TAG_RANGE@) values, the various string tags can be promoted to name + * (@code IPP_TAG_NAME@) or nameWithLanguage (@code IPP_TAG_NAMELANG@) values, text + * (@code IPP_TAG_TEXT@) values can be promoted to textWithLanguage + * (@code IPP_TAG_TEXTLANG@) values, and all values can be demoted to the various + * out-of-band value tags such as no-value (@code IPP_TAG_NOVALUE@). All other changes + * will be rejected. + * + * Promoting a string attribute to nameWithLanguage or textWithLanguage adds the language + * code in the "attributes-natural-language" attribute or, if not present, the language + * code for the current locale. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +ippSetValueTag( + ipp_t *ipp, /* IO - IPP message */ + ipp_attribute_t **attr, /* IO - IPP attribute */ + ipp_tag_t value_tag) /* I - Value tag */ +{ + int i; /* Looping var */ + _ipp_value_t *value; /* Current value */ + int integer; /* Current integer value */ + cups_lang_t *language; /* Current language */ + char code[32]; /* Language code */ + ipp_tag_t temp_tag; /* Temporary value tag */ + + + /* + * Range check input... + */ + + if (!ipp || !attr) + return (0); + + /* + * If there is no change, return immediately... + */ + + if (value_tag == (*attr)->value_tag) + return (1); + + /* + * Otherwise implement changes as needed... + */ + + temp_tag = (ipp_tag_t)((int)((*attr)->value_tag) & IPP_TAG_MASK); + + switch (value_tag) + { + case IPP_TAG_UNSUPPORTED_VALUE : + case IPP_TAG_DEFAULT : + case IPP_TAG_UNKNOWN : + case IPP_TAG_NOVALUE : + case IPP_TAG_NOTSETTABLE : + case IPP_TAG_DELETEATTR : + case IPP_TAG_ADMINDEFINE : + /* + * Free any existing values... + */ + + if ((*attr)->num_values > 0) + ipp_free_values(*attr, 0, (*attr)->num_values); + + /* + * Set out-of-band value... + */ + + (*attr)->value_tag = value_tag; + break; + + case IPP_TAG_RANGE : + if (temp_tag != IPP_TAG_INTEGER) + return (0); + + for (i = (*attr)->num_values, value = (*attr)->values; + i > 0; + i --, value ++) + { + integer = value->integer; + value->range.lower = value->range.upper = integer; + } + + (*attr)->value_tag = IPP_TAG_RANGE; + break; + + case IPP_TAG_NAME : + if (temp_tag != IPP_TAG_KEYWORD && temp_tag != IPP_TAG_URI && + temp_tag != IPP_TAG_URISCHEME && temp_tag != IPP_TAG_LANGUAGE && + temp_tag != IPP_TAG_MIMETYPE) + return (0); + + (*attr)->value_tag = (ipp_tag_t)(IPP_TAG_NAME | ((*attr)->value_tag & IPP_TAG_COPY)); + break; + + case IPP_TAG_NAMELANG : + case IPP_TAG_TEXTLANG : + if (value_tag == IPP_TAG_NAMELANG && + (temp_tag != IPP_TAG_NAME && temp_tag != IPP_TAG_KEYWORD && + temp_tag != IPP_TAG_URI && temp_tag != IPP_TAG_URISCHEME && + temp_tag != IPP_TAG_LANGUAGE && temp_tag != IPP_TAG_MIMETYPE)) + return (0); + + if (value_tag == IPP_TAG_TEXTLANG && temp_tag != IPP_TAG_TEXT) + return (0); + + if (ipp->attrs && ipp->attrs->next && ipp->attrs->next->name && + !strcmp(ipp->attrs->next->name, "attributes-natural-language")) + { + /* + * Use the language code from the IPP message... + */ + + (*attr)->values[0].string.language = + _cupsStrAlloc(ipp->attrs->next->values[0].string.text); + } + else + { + /* + * Otherwise, use the language code corresponding to the locale... + */ + + language = cupsLangDefault(); + (*attr)->values[0].string.language = _cupsStrAlloc(ipp_lang_code(language->language, + code, + sizeof(code))); + } + + for (i = (*attr)->num_values - 1, value = (*attr)->values + 1; + i > 0; + i --, value ++) + value->string.language = (*attr)->values[0].string.language; + + if ((int)(*attr)->value_tag & IPP_TAG_COPY) + { + /* + * Make copies of all values... + */ + + for (i = (*attr)->num_values, value = (*attr)->values; + i > 0; + i --, value ++) + value->string.text = _cupsStrAlloc(value->string.text); + } + + (*attr)->value_tag = IPP_TAG_NAMELANG; + break; + + case IPP_TAG_KEYWORD : + if (temp_tag == IPP_TAG_NAME || temp_tag == IPP_TAG_NAMELANG) + break; /* Silently "allow" name -> keyword */ + + default : + return (0); + } + + return (1); +} + + +/* + * 'ippSetVersion()' - Set the version number in an IPP message. + * + * The @code ipp@ parameter refers to an IPP message previously created using the + * @link ippNew@ or @link ippNewRequest@ functions. + * + * The valid version numbers are currently 1.0, 1.1, 2.0, 2.1, and 2.2. + * + * @since CUPS 1.6@ + */ + +int /* O - 1 on success, 0 on failure */ +ippSetVersion(ipp_t *ipp, /* I - IPP message */ + int major, /* I - Major version number (major.minor) */ + int minor) /* I - Minor version number (major.minor) */ +{ + /* + * Range check input... + */ + + if (!ipp || major < 0 || minor < 0) + return (0); + + /* + * Set the version number... + */ + + ipp->request.any.version[0] = major; + ipp->request.any.version[1] = minor; + + return (1); +} + + +/* + * 'ippTimeToDate()' - Convert from UNIX time to RFC 1903 format. + */ + +const ipp_uchar_t * /* O - RFC-1903 date/time data */ +ippTimeToDate(time_t t) /* I - UNIX time value */ +{ + struct tm *unixdate; /* UNIX unixdate/time info */ + ipp_uchar_t *date = _cupsGlobals()->ipp_date; + /* RFC-1903 date/time data */ + + + /* + * RFC-1903 date/time format is: + * + * Byte(s) Description + * ------- ----------- + * 0-1 Year (0 to 65535) + * 2 Month (1 to 12) + * 3 Day (1 to 31) + * 4 Hours (0 to 23) + * 5 Minutes (0 to 59) + * 6 Seconds (0 to 60, 60 = "leap second") + * 7 Deciseconds (0 to 9) + * 8 +/- UTC + * 9 UTC hours (0 to 11) + * 10 UTC minutes (0 to 59) + */ + + unixdate = gmtime(&t); + unixdate->tm_year += 1900; + + date[0] = unixdate->tm_year >> 8; + date[1] = unixdate->tm_year; + date[2] = unixdate->tm_mon + 1; + date[3] = unixdate->tm_mday; + date[4] = unixdate->tm_hour; + date[5] = unixdate->tm_min; + date[6] = unixdate->tm_sec; + date[7] = 0; + date[8] = '+'; + date[9] = 0; + date[10] = 0; + + return (date); +} + + +/* + * 'ippWrite()' - Write data for an IPP message to a HTTP connection. + */ + +ipp_state_t /* O - Current state */ +ippWrite(http_t *http, /* I - HTTP connection */ + ipp_t *ipp) /* I - IPP data */ +{ + DEBUG_printf(("ippWrite(http=%p, ipp=%p)", http, ipp)); + + if (!http) + return (IPP_ERROR); + + return (ippWriteIO(http, (ipp_iocb_t)httpWrite2, http->blocking, NULL, ipp)); +} + + +/* + * 'ippWriteFile()' - Write data for an IPP message to a file. + * + * @since CUPS 1.1.19/Mac OS X 10.3@ + */ + +ipp_state_t /* O - Current state */ +ippWriteFile(int fd, /* I - HTTP data */ + ipp_t *ipp) /* I - IPP data */ +{ + DEBUG_printf(("ippWriteFile(fd=%d, ipp=%p)", fd, ipp)); + + ipp->state = IPP_IDLE; + + return (ippWriteIO(&fd, (ipp_iocb_t)ipp_write_file, 1, NULL, ipp)); +} + + +/* + * 'ippWriteIO()' - Write data for an IPP message. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ipp_state_t /* O - Current state */ +ippWriteIO(void *dst, /* I - Destination */ + ipp_iocb_t cb, /* I - Write callback function */ + int blocking, /* I - Use blocking IO? */ + ipp_t *parent, /* I - Parent IPP message */ + ipp_t *ipp) /* I - IPP data */ +{ + int i; /* Looping var */ + int n; /* Length of data */ + unsigned char *buffer, /* Data buffer */ + *bufptr; /* Pointer into buffer */ + ipp_attribute_t *attr; /* Current attribute */ + _ipp_value_t *value; /* Current value */ + + + DEBUG_printf(("ippWriteIO(dst=%p, cb=%p, blocking=%d, parent=%p, ipp=%p)", + dst, cb, blocking, parent, ipp)); + + if (!dst || !ipp) + return (IPP_ERROR); + + if ((buffer = (unsigned char *)_cupsBufferGet(IPP_BUF_SIZE)) == NULL) + { + DEBUG_puts("1ippWriteIO: Unable to get write buffer"); + return (IPP_ERROR); + } + + switch (ipp->state) + { + case IPP_IDLE : + ipp->state ++; /* Avoid common problem... */ + + case IPP_HEADER : + if (parent == NULL) + { + /* + * Send the request header: + * + * Version = 2 bytes + * Operation/Status Code = 2 bytes + * Request ID = 4 bytes + * Total = 8 bytes + */ + + bufptr = buffer; + + *bufptr++ = ipp->request.any.version[0]; + *bufptr++ = ipp->request.any.version[1]; + *bufptr++ = ipp->request.any.op_status >> 8; + *bufptr++ = ipp->request.any.op_status; + *bufptr++ = ipp->request.any.request_id >> 24; + *bufptr++ = ipp->request.any.request_id >> 16; + *bufptr++ = ipp->request.any.request_id >> 8; + *bufptr++ = ipp->request.any.request_id; + + DEBUG_printf(("2ippWriteIO: version=%d.%d", buffer[0], buffer[1])); + DEBUG_printf(("2ippWriteIO: op_status=%04x", + ipp->request.any.op_status)); + DEBUG_printf(("2ippWriteIO: request_id=%d", + ipp->request.any.request_id)); + + if ((*cb)(dst, buffer, (int)(bufptr - buffer)) < 0) + { + DEBUG_puts("1ippWriteIO: Could not write IPP header..."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + } + + /* + * Reset the state engine to point to the first attribute + * in the request/response, with no current group. + */ + + ipp->state = IPP_ATTRIBUTE; + ipp->current = ipp->attrs; + ipp->curtag = IPP_TAG_ZERO; + + DEBUG_printf(("1ippWriteIO: ipp->current=%p", ipp->current)); + + /* + * If blocking is disabled, stop here... + */ + + if (!blocking) + break; + + case IPP_ATTRIBUTE : + while (ipp->current != NULL) + { + /* + * Write this attribute... + */ + + bufptr = buffer; + attr = ipp->current; + + ipp->current = ipp->current->next; + + if (!parent) + { + if (ipp->curtag != attr->group_tag) + { + /* + * Send a group tag byte... + */ + + ipp->curtag = attr->group_tag; + + if (attr->group_tag == IPP_TAG_ZERO) + continue; + + DEBUG_printf(("2ippWriteIO: wrote group tag=%x(%s)", + attr->group_tag, ippTagString(attr->group_tag))); + *bufptr++ = attr->group_tag; + } + else if (attr->group_tag == IPP_TAG_ZERO) + continue; + } + + DEBUG_printf(("1ippWriteIO: %s (%s%s)", attr->name, + attr->num_values > 1 ? "1setOf " : "", + ippTagString(attr->value_tag))); + + /* + * Write the attribute tag and name. + * + * The attribute name length does not include the trailing nul + * character in the source string. + * + * Collection values (parent != NULL) are written differently... + */ + + if (parent == NULL) + { + /* + * Get the length of the attribute name, and make sure it won't + * overflow the buffer... + */ + + if ((n = (int)strlen(attr->name)) > (IPP_BUF_SIZE - 8)) + { + DEBUG_printf(("1ippWriteIO: Attribute name too long (%d)", n)); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + /* + * Write the value tag, name length, and name string... + */ + + DEBUG_printf(("2ippWriteIO: writing value tag=%x(%s)", + attr->value_tag, ippTagString(attr->value_tag))); + DEBUG_printf(("2ippWriteIO: writing name=%d,\"%s\"", n, + attr->name)); + + if (attr->value_tag > 0xff) + { + *bufptr++ = IPP_TAG_EXTENSION; + *bufptr++ = attr->value_tag >> 24; + *bufptr++ = attr->value_tag >> 16; + *bufptr++ = attr->value_tag >> 8; + *bufptr++ = attr->value_tag; + } + else + *bufptr++ = attr->value_tag; + + *bufptr++ = n >> 8; + *bufptr++ = n; + memcpy(bufptr, attr->name, n); + bufptr += n; + } + else + { + /* + * Get the length of the attribute name, and make sure it won't + * overflow the buffer... + */ + + if ((n = (int)strlen(attr->name)) > (IPP_BUF_SIZE - 12)) + { + DEBUG_printf(("1ippWriteIO: Attribute name too long (%d)", n)); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + /* + * Write the member name tag, name length, name string, value tag, + * and empty name for the collection member attribute... + */ + + DEBUG_printf(("2ippWriteIO: writing value tag=%x(memberName)", + IPP_TAG_MEMBERNAME)); + DEBUG_printf(("2ippWriteIO: writing name=%d,\"%s\"", n, + attr->name)); + DEBUG_printf(("2ippWriteIO: writing value tag=%x(%s)", + attr->value_tag, ippTagString(attr->value_tag))); + DEBUG_puts("2ippWriteIO: writing name=0,\"\""); + + *bufptr++ = IPP_TAG_MEMBERNAME; + *bufptr++ = 0; + *bufptr++ = 0; + *bufptr++ = n >> 8; + *bufptr++ = n; + memcpy(bufptr, attr->name, n); + bufptr += n; + + if (attr->value_tag > 0xff) + { + *bufptr++ = IPP_TAG_EXTENSION; + *bufptr++ = attr->value_tag >> 24; + *bufptr++ = attr->value_tag >> 16; + *bufptr++ = attr->value_tag >> 8; + *bufptr++ = attr->value_tag; + } + else + *bufptr++ = attr->value_tag; + + *bufptr++ = 0; + *bufptr++ = 0; + } + + /* + * Now write the attribute value(s)... + */ + + switch (attr->value_tag & ~IPP_TAG_COPY) + { + case IPP_TAG_UNSUPPORTED_VALUE : + case IPP_TAG_DEFAULT : + case IPP_TAG_UNKNOWN : + case IPP_TAG_NOVALUE : + case IPP_TAG_NOTSETTABLE : + case IPP_TAG_DELETEATTR : + case IPP_TAG_ADMINDEFINE : + *bufptr++ = 0; + *bufptr++ = 0; + break; + + case IPP_TAG_INTEGER : + case IPP_TAG_ENUM : + for (i = 0, value = attr->values; + i < attr->num_values; + i ++, value ++) + { + if ((IPP_BUF_SIZE - (bufptr - buffer)) < 9) + { + if ((*cb)(dst, buffer, (int)(bufptr - buffer)) < 0) + { + DEBUG_puts("1ippWriteIO: Could not write IPP " + "attribute..."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + bufptr = buffer; + } + + if (i) + { + /* + * Arrays and sets are done by sending additional + * values with a zero-length name... + */ + + *bufptr++ = attr->value_tag; + *bufptr++ = 0; + *bufptr++ = 0; + } + + /* + * Integers and enumerations are both 4-byte signed + * (twos-complement) values. + * + * Put the 2-byte length and 4-byte value into the buffer... + */ + + *bufptr++ = 0; + *bufptr++ = 4; + *bufptr++ = value->integer >> 24; + *bufptr++ = value->integer >> 16; + *bufptr++ = value->integer >> 8; + *bufptr++ = value->integer; + } + break; + + case IPP_TAG_BOOLEAN : + for (i = 0, value = attr->values; + i < attr->num_values; + i ++, value ++) + { + if ((IPP_BUF_SIZE - (bufptr - buffer)) < 6) + { + if ((*cb)(dst, buffer, (int)(bufptr - buffer)) < 0) + { + DEBUG_puts("1ippWriteIO: Could not write IPP " + "attribute..."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + bufptr = buffer; + } + + if (i) + { + /* + * Arrays and sets are done by sending additional + * values with a zero-length name... + */ + + *bufptr++ = attr->value_tag; + *bufptr++ = 0; + *bufptr++ = 0; + } + + /* + * Boolean values are 1-byte; 0 = false, 1 = true. + * + * Put the 2-byte length and 1-byte value into the buffer... + */ + + *bufptr++ = 0; + *bufptr++ = 1; + *bufptr++ = value->boolean; + } + break; + + case IPP_TAG_TEXT : + case IPP_TAG_NAME : + case IPP_TAG_KEYWORD : + case IPP_TAG_URI : + case IPP_TAG_URISCHEME : + case IPP_TAG_CHARSET : + case IPP_TAG_LANGUAGE : + case IPP_TAG_MIMETYPE : + for (i = 0, value = attr->values; + i < attr->num_values; + i ++, value ++) + { + if (i) + { + /* + * Arrays and sets are done by sending additional + * values with a zero-length name... + */ + + DEBUG_printf(("2ippWriteIO: writing value tag=%x(%s)", + attr->value_tag, + ippTagString(attr->value_tag))); + DEBUG_printf(("2ippWriteIO: writing name=0,\"\"")); + + if ((IPP_BUF_SIZE - (bufptr - buffer)) < 3) + { + if ((*cb)(dst, buffer, (int)(bufptr - buffer)) < 0) + { + DEBUG_puts("1ippWriteIO: Could not write IPP " + "attribute..."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + bufptr = buffer; + } + + *bufptr++ = attr->value_tag; + *bufptr++ = 0; + *bufptr++ = 0; + } + + if (value->string.text != NULL) + n = (int)strlen(value->string.text); + else + n = 0; + + if (n > (IPP_BUF_SIZE - 2)) + { + DEBUG_printf(("1ippWriteIO: String too long (%d)", n)); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + DEBUG_printf(("2ippWriteIO: writing string=%d,\"%s\"", n, + value->string.text)); + + if ((int)(IPP_BUF_SIZE - (bufptr - buffer)) < (n + 2)) + { + if ((*cb)(dst, buffer, (int)(bufptr - buffer)) < 0) + { + DEBUG_puts("1ippWriteIO: Could not write IPP " + "attribute..."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + bufptr = buffer; + } + + /* + * All simple strings consist of the 2-byte length and + * character data without the trailing nul normally found + * in C strings. Also, strings cannot be longer than IPP_MAX_LENGTH + * bytes since the 2-byte length is a signed (twos-complement) + * value. + * + * Put the 2-byte length and string characters in the buffer. + */ + + *bufptr++ = n >> 8; + *bufptr++ = n; + + if (n > 0) + { + memcpy(bufptr, value->string.text, n); + bufptr += n; + } + } + break; + + case IPP_TAG_DATE : + for (i = 0, value = attr->values; + i < attr->num_values; + i ++, value ++) + { + if ((IPP_BUF_SIZE - (bufptr - buffer)) < 16) + { + if ((*cb)(dst, buffer, (int)(bufptr - buffer)) < 0) + { + DEBUG_puts("1ippWriteIO: Could not write IPP " + "attribute..."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + bufptr = buffer; + } + + if (i) + { + /* + * Arrays and sets are done by sending additional + * values with a zero-length name... + */ + + *bufptr++ = attr->value_tag; + *bufptr++ = 0; + *bufptr++ = 0; + } + + /* + * Date values consist of a 2-byte length and an + * 11-byte date/time structure defined by RFC 1903. + * + * Put the 2-byte length and 11-byte date/time + * structure in the buffer. + */ + + *bufptr++ = 0; + *bufptr++ = 11; + memcpy(bufptr, value->date, 11); + bufptr += 11; + } + break; + + case IPP_TAG_RESOLUTION : + for (i = 0, value = attr->values; + i < attr->num_values; + i ++, value ++) + { + if ((IPP_BUF_SIZE - (bufptr - buffer)) < 14) + { + if ((*cb)(dst, buffer, (int)(bufptr - buffer)) < 0) + { + DEBUG_puts("1ippWriteIO: Could not write IPP " + "attribute..."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + bufptr = buffer; + } + + if (i) + { + /* + * Arrays and sets are done by sending additional + * values with a zero-length name... + */ + + *bufptr++ = attr->value_tag; + *bufptr++ = 0; + *bufptr++ = 0; + } + + /* + * Resolution values consist of a 2-byte length, + * 4-byte horizontal resolution value, 4-byte vertical + * resolution value, and a 1-byte units value. + * + * Put the 2-byte length and resolution value data + * into the buffer. + */ + + *bufptr++ = 0; + *bufptr++ = 9; + *bufptr++ = value->resolution.xres >> 24; + *bufptr++ = value->resolution.xres >> 16; + *bufptr++ = value->resolution.xres >> 8; + *bufptr++ = value->resolution.xres; + *bufptr++ = value->resolution.yres >> 24; + *bufptr++ = value->resolution.yres >> 16; + *bufptr++ = value->resolution.yres >> 8; + *bufptr++ = value->resolution.yres; + *bufptr++ = value->resolution.units; + } + break; + + case IPP_TAG_RANGE : + for (i = 0, value = attr->values; + i < attr->num_values; + i ++, value ++) + { + if ((IPP_BUF_SIZE - (bufptr - buffer)) < 13) + { + if ((*cb)(dst, buffer, (int)(bufptr - buffer)) < 0) + { + DEBUG_puts("1ippWriteIO: Could not write IPP " + "attribute..."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + bufptr = buffer; + } + + if (i) + { + /* + * Arrays and sets are done by sending additional + * values with a zero-length name... + */ + + *bufptr++ = attr->value_tag; + *bufptr++ = 0; + *bufptr++ = 0; + } + + /* + * Range values consist of a 2-byte length, + * 4-byte lower value, and 4-byte upper value. + * + * Put the 2-byte length and range value data + * into the buffer. + */ + + *bufptr++ = 0; + *bufptr++ = 8; + *bufptr++ = value->range.lower >> 24; + *bufptr++ = value->range.lower >> 16; + *bufptr++ = value->range.lower >> 8; + *bufptr++ = value->range.lower; + *bufptr++ = value->range.upper >> 24; + *bufptr++ = value->range.upper >> 16; + *bufptr++ = value->range.upper >> 8; + *bufptr++ = value->range.upper; + } + break; + + case IPP_TAG_TEXTLANG : + case IPP_TAG_NAMELANG : + for (i = 0, value = attr->values; + i < attr->num_values; + i ++, value ++) + { + if (i) + { + /* + * Arrays and sets are done by sending additional + * values with a zero-length name... + */ + + if ((IPP_BUF_SIZE - (bufptr - buffer)) < 3) + { + if ((*cb)(dst, buffer, (int)(bufptr - buffer)) < 0) + { + DEBUG_puts("1ippWriteIO: Could not write IPP " + "attribute..."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + bufptr = buffer; + } + + *bufptr++ = attr->value_tag; + *bufptr++ = 0; + *bufptr++ = 0; + } + + /* + * textWithLanguage and nameWithLanguage values consist + * of a 2-byte length for both strings and their + * individual lengths, a 2-byte length for the + * character string, the character string without the + * trailing nul, a 2-byte length for the character + * set string, and the character set string without + * the trailing nul. + */ + + n = 4; + + if (value->string.language != NULL) + n += (int)strlen(value->string.language); + + if (value->string.text != NULL) + n += (int)strlen(value->string.text); + + if (n > (IPP_BUF_SIZE - 2)) + { + DEBUG_printf(("1ippWriteIO: text/nameWithLanguage value " + "too long (%d)", n)); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + if ((int)(IPP_BUF_SIZE - (bufptr - buffer)) < (n + 2)) + { + if ((*cb)(dst, buffer, (int)(bufptr - buffer)) < 0) + { + DEBUG_puts("1ippWriteIO: Could not write IPP " + "attribute..."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + bufptr = buffer; + } + + /* Length of entire value */ + *bufptr++ = n >> 8; + *bufptr++ = n; + + /* Length of language */ + if (value->string.language != NULL) + n = (int)strlen(value->string.language); + else + n = 0; + + *bufptr++ = n >> 8; + *bufptr++ = n; + + /* Language */ + if (n > 0) + { + memcpy(bufptr, value->string.language, n); + bufptr += n; + } + + /* Length of text */ + if (value->string.text != NULL) + n = (int)strlen(value->string.text); + else + n = 0; + + *bufptr++ = n >> 8; + *bufptr++ = n; + + /* Text */ + if (n > 0) + { + memcpy(bufptr, value->string.text, n); + bufptr += n; + } + } + break; + + case IPP_TAG_BEGIN_COLLECTION : + for (i = 0, value = attr->values; + i < attr->num_values; + i ++, value ++) + { + /* + * Collections are written with the begin-collection + * tag first with a value of 0 length, followed by the + * attributes in the collection, then the end-collection + * value... + */ + + if ((IPP_BUF_SIZE - (bufptr - buffer)) < 5) + { + if ((*cb)(dst, buffer, (int)(bufptr - buffer)) < 0) + { + DEBUG_puts("1ippWriteIO: Could not write IPP " + "attribute..."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + bufptr = buffer; + } + + if (i) + { + /* + * Arrays and sets are done by sending additional + * values with a zero-length name... + */ + + *bufptr++ = attr->value_tag; + *bufptr++ = 0; + *bufptr++ = 0; + } + + /* + * Write a data length of 0 and flush the buffer... + */ + + *bufptr++ = 0; + *bufptr++ = 0; + + if ((*cb)(dst, buffer, (int)(bufptr - buffer)) < 0) + { + DEBUG_puts("1ippWriteIO: Could not write IPP " + "attribute..."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + bufptr = buffer; + + /* + * Then write the collection attribute... + */ + + value->collection->state = IPP_IDLE; + + if (ippWriteIO(dst, cb, 1, ipp, + value->collection) == IPP_ERROR) + { + DEBUG_puts("1ippWriteIO: Unable to write collection value"); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + } + break; + + default : + for (i = 0, value = attr->values; + i < attr->num_values; + i ++, value ++) + { + if (i) + { + /* + * Arrays and sets are done by sending additional + * values with a zero-length name... + */ + + if ((IPP_BUF_SIZE - (bufptr - buffer)) < 3) + { + if ((*cb)(dst, buffer, (int)(bufptr - buffer)) < 0) + { + DEBUG_puts("1ippWriteIO: Could not write IPP " + "attribute..."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + bufptr = buffer; + } + + *bufptr++ = attr->value_tag; + *bufptr++ = 0; + *bufptr++ = 0; + } + + /* + * An unknown value might some new value that a + * vendor has come up with. It consists of a + * 2-byte length and the bytes in the unknown + * value buffer. + */ + + n = value->unknown.length; + + if (n > (IPP_BUF_SIZE - 2)) + { + DEBUG_printf(("1ippWriteIO: Data length too long (%d)", + n)); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + if ((int)(IPP_BUF_SIZE - (bufptr - buffer)) < (n + 2)) + { + if ((*cb)(dst, buffer, (int)(bufptr - buffer)) < 0) + { + DEBUG_puts("1ippWriteIO: Could not write IPP " + "attribute..."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + bufptr = buffer; + } + + /* Length of unknown value */ + *bufptr++ = n >> 8; + *bufptr++ = n; + + /* Value */ + if (n > 0) + { + memcpy(bufptr, value->unknown.data, n); + bufptr += n; + } + } + break; + } + + /* + * Write the data out... + */ + + if (bufptr > buffer) + { + if ((*cb)(dst, buffer, (int)(bufptr - buffer)) < 0) + { + DEBUG_puts("1ippWriteIO: Could not write IPP attribute..."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + DEBUG_printf(("2ippWriteIO: wrote %d bytes", + (int)(bufptr - buffer))); + } + + /* + * If blocking is disabled, stop here... + */ + + if (!blocking) + break; + } + + if (ipp->current == NULL) + { + /* + * Done with all of the attributes; add the end-of-attributes + * tag or end-collection attribute... + */ + + if (parent == NULL) + { + buffer[0] = IPP_TAG_END; + n = 1; + } + else + { + buffer[0] = IPP_TAG_END_COLLECTION; + buffer[1] = 0; /* empty name */ + buffer[2] = 0; + buffer[3] = 0; /* empty value */ + buffer[4] = 0; + n = 5; + } + + if ((*cb)(dst, buffer, n) < 0) + { + DEBUG_puts("1ippWriteIO: Could not write IPP end-tag..."); + _cupsBufferRelease((char *)buffer); + return (IPP_ERROR); + } + + ipp->state = IPP_DATA; + } + break; + + case IPP_DATA : + break; + + default : + break; /* anti-compiler-warning-code */ + } + + _cupsBufferRelease((char *)buffer); + + return (ipp->state); +} + + +/* + * 'ipp_add_attr()' - Add a new attribute to the message. + */ + +static ipp_attribute_t * /* O - New attribute */ +ipp_add_attr(ipp_t *ipp, /* I - IPP message */ + const char *name, /* I - Attribute name or NULL */ + ipp_tag_t group_tag, /* I - Group tag or IPP_TAG_ZERO */ + ipp_tag_t value_tag, /* I - Value tag or IPP_TAG_ZERO */ + int num_values) /* I - Number of values */ +{ + int alloc_values; /* Number of values to allocate */ + ipp_attribute_t *attr; /* New attribute */ + + + DEBUG_printf(("4ipp_add_attr(ipp=%p, name=\"%s\", group_tag=0x%x, value_tag=0x%x, " + "num_values=%d)", ipp, name, group_tag, value_tag, num_values)); + + /* + * Range check input... + */ + + if (!ipp || num_values < 0) + return (NULL); + + /* + * Allocate memory, rounding the allocation up as needed... + */ + + if (num_values <= 1) + alloc_values = 1; + else + alloc_values = (num_values + IPP_MAX_VALUES - 1) & ~(IPP_MAX_VALUES - 1); + + attr = calloc(sizeof(ipp_attribute_t) + + (alloc_values - 1) * sizeof(_ipp_value_t), 1); + + if (attr) + { + /* + * Initialize attribute... + */ + + if (name) + attr->name = _cupsStrAlloc(name); + + attr->group_tag = group_tag; + attr->value_tag = value_tag; + attr->num_values = num_values; + + /* + * Add it to the end of the linked list... + */ + + if (ipp->last) + ipp->last->next = attr; + else + ipp->attrs = attr; + + ipp->prev = ipp->last; + ipp->last = ipp->current = attr; + } + + DEBUG_printf(("5ipp_add_attr: Returning %p", attr)); + + return (attr); +} + + +/* + * 'ipp_free_values()' - Free attribute values. + */ + +static void +ipp_free_values(ipp_attribute_t *attr, /* I - Attribute to free values from */ + int element,/* I - First value to free */ + int count) /* I - Number of values to free */ +{ + int i; /* Looping var */ + _ipp_value_t *value; /* Current value */ + + + DEBUG_printf(("4ipp_free_values(attr=%p, element=%d, count=%d)", attr, element, count)); + + if (!(attr->value_tag & IPP_TAG_COPY)) + { + /* + * Free values as needed... + */ + + switch (attr->value_tag) + { + case IPP_TAG_TEXTLANG : + case IPP_TAG_NAMELANG : + if (element == 0 && count == attr->num_values && attr->values[0].string.language) + _cupsStrFree(attr->values[0].string.language); + + case IPP_TAG_TEXT : + case IPP_TAG_NAME : + case IPP_TAG_RESERVED_STRING : + case IPP_TAG_KEYWORD : + case IPP_TAG_URI : + case IPP_TAG_URISCHEME : + case IPP_TAG_CHARSET : + case IPP_TAG_LANGUAGE : + case IPP_TAG_MIMETYPE : + for (i = count, value = attr->values + element; + i > 0; + i --, value ++) + _cupsStrFree(value->string.text); + break; + + case IPP_TAG_DEFAULT : + case IPP_TAG_UNKNOWN : + case IPP_TAG_NOVALUE : + case IPP_TAG_NOTSETTABLE : + case IPP_TAG_DELETEATTR : + case IPP_TAG_ADMINDEFINE : + case IPP_TAG_INTEGER : + case IPP_TAG_ENUM : + case IPP_TAG_BOOLEAN : + case IPP_TAG_DATE : + case IPP_TAG_RESOLUTION : + case IPP_TAG_RANGE : + break; + + case IPP_TAG_BEGIN_COLLECTION : + for (i = count, value = attr->values + element; + i > 0; + i --, value ++) + ippDelete(value->collection); + break; + + case IPP_TAG_STRING : + default : + for (i = count, value = attr->values + element; + i > 0; + i --, value ++) + if (value->unknown.data) + free(value->unknown.data); + break; + } + } + + /* + * If we are not freeing values from the end, move the remaining values up... + */ + + if ((element + count) < attr->num_values) + memmove(attr->values + element, attr->values + element + count, + (attr->num_values - count - element) * sizeof(_ipp_value_t)); + + attr->num_values -= count; +} + + +/* + * 'ipp_get_code()' - Convert a C locale/charset name into an IPP language/charset code. + * + * This typically converts strings of the form "ll_CC", "ll-REGION", and "CHARSET_NUMBER" + * to "ll-cc", "ll-region", and "charset-number", respectively. + */ + +static char * /* O - Language code string */ +ipp_get_code(const char *value, /* I - Locale/charset string */ + char *buffer, /* I - String buffer */ + size_t bufsize) /* I - Size of string buffer */ +{ + char *bufptr, /* Pointer into buffer */ + *bufend; /* End of buffer */ + + + /* + * Convert values to lowercase and change _ to - as needed... + */ + + for (bufptr = buffer, bufend = buffer + bufsize - 1; + *value && bufptr < bufend; + value ++) + if (*value == '_') + *bufptr++ = '-'; + else + *bufptr++ = _cups_tolower(*value); + + *bufptr = '\0'; + + /* + * Return the converted string... + */ + + return (buffer); +} + + +/* + * 'ipp_lang_code()' - Convert a C locale name into an IPP language code. + * + * This typically converts strings of the form "ll_CC" and "ll-REGION" to "ll-cc" and + * "ll-region", respectively. It also converts the "C" (POSIX) locale to "en". + */ + +static char * /* O - Language code string */ +ipp_lang_code(const char *locale, /* I - Locale string */ + char *buffer, /* I - String buffer */ + size_t bufsize) /* I - Size of string buffer */ +{ + /* + * Map POSIX ("C") locale to generic English, otherwise convert the locale string as-is. + */ + + if (!_cups_strcasecmp(locale, "c")) + { + strlcpy(buffer, "en", bufsize); + return (buffer); + } + else + return (ipp_get_code(locale, buffer, bufsize)); +} + + +/* + * 'ipp_length()' - Compute the length of an IPP message or collection value. + */ + +static size_t /* O - Size of IPP message */ +ipp_length(ipp_t *ipp, /* I - IPP message or collection */ + int collection) /* I - 1 if a collection, 0 otherwise */ +{ + int i; /* Looping var */ + size_t bytes; /* Number of bytes */ + ipp_attribute_t *attr; /* Current attribute */ + ipp_tag_t group; /* Current group */ + _ipp_value_t *value; /* Current value */ + + + DEBUG_printf(("3ipp_length(ipp=%p, collection=%d)", ipp, collection)); + + if (!ipp) + { + DEBUG_puts("4ipp_length: Returning 0 bytes"); + return (0); + } + + /* + * Start with 8 bytes for the IPP message header... + */ + + bytes = collection ? 0 : 8; + + /* + * Then add the lengths of each attribute... + */ + + group = IPP_TAG_ZERO; + + for (attr = ipp->attrs; attr != NULL; attr = attr->next) + { + if (attr->group_tag != group && !collection) + { + group = attr->group_tag; + if (group == IPP_TAG_ZERO) + continue; + + bytes ++; /* Group tag */ + } + + if (!attr->name) + continue; + + DEBUG_printf(("5ipp_length: attr->name=\"%s\", attr->num_values=%d, " + "bytes=" CUPS_LLFMT, attr->name, attr->num_values, CUPS_LLCAST bytes)); + + if (attr->value_tag < IPP_TAG_EXTENSION) + bytes += attr->num_values; /* Value tag for each value */ + else + bytes += 5 * attr->num_values; /* Value tag for each value */ + bytes += 2 * attr->num_values; /* Name lengths */ + bytes += (int)strlen(attr->name); /* Name */ + bytes += 2 * attr->num_values; /* Value lengths */ + + if (collection) + bytes += 5; /* Add membername overhead */ + + switch (attr->value_tag & ~IPP_TAG_COPY) + { + case IPP_TAG_UNSUPPORTED_VALUE : + case IPP_TAG_DEFAULT : + case IPP_TAG_UNKNOWN : + case IPP_TAG_NOVALUE : + case IPP_TAG_NOTSETTABLE : + case IPP_TAG_DELETEATTR : + case IPP_TAG_ADMINDEFINE : + break; + + case IPP_TAG_INTEGER : + case IPP_TAG_ENUM : + bytes += 4 * attr->num_values; + break; + + case IPP_TAG_BOOLEAN : + bytes += attr->num_values; + break; + + case IPP_TAG_TEXT : + case IPP_TAG_NAME : + case IPP_TAG_KEYWORD : + case IPP_TAG_URI : + case IPP_TAG_URISCHEME : + case IPP_TAG_CHARSET : + case IPP_TAG_LANGUAGE : + case IPP_TAG_MIMETYPE : + for (i = 0, value = attr->values; + i < attr->num_values; + i ++, value ++) + if (value->string.text) + bytes += strlen(value->string.text); + break; + + case IPP_TAG_DATE : + bytes += 11 * attr->num_values; + break; + + case IPP_TAG_RESOLUTION : + bytes += 9 * attr->num_values; + break; + + case IPP_TAG_RANGE : + bytes += 8 * attr->num_values; + break; + + case IPP_TAG_TEXTLANG : + case IPP_TAG_NAMELANG : + bytes += 4 * attr->num_values;/* Charset + text length */ + + for (i = 0, value = attr->values; + i < attr->num_values; + i ++, value ++) + { + if (value->string.language) + bytes += strlen(value->string.language); + + if (value->string.text) + bytes += strlen(value->string.text); + } + break; + + case IPP_TAG_BEGIN_COLLECTION : + for (i = 0, value = attr->values; + i < attr->num_values; + i ++, value ++) + bytes += ipp_length(value->collection, 1); + break; + + default : + for (i = 0, value = attr->values; + i < attr->num_values; + i ++, value ++) + bytes += value->unknown.length; + break; + } + } + + /* + * Finally, add 1 byte for the "end of attributes" tag or 5 bytes + * for the "end of collection" tag and return... + */ + + if (collection) + bytes += 5; + else + bytes ++; + + DEBUG_printf(("4ipp_length: Returning " CUPS_LLFMT " bytes", CUPS_LLCAST bytes)); + + return (bytes); +} + + +/* + * 'ipp_read_http()' - Semi-blocking read on a HTTP connection... + */ + +static ssize_t /* O - Number of bytes read */ +ipp_read_http(http_t *http, /* I - Client connection */ + ipp_uchar_t *buffer, /* O - Buffer for data */ + size_t length) /* I - Total length */ +{ + int tbytes, /* Total bytes read */ + bytes; /* Bytes read this pass */ + char len[32]; /* Length string */ + + + DEBUG_printf(("7ipp_read_http(http=%p, buffer=%p, length=%d)", + http, buffer, (int)length)); + + /* + * Loop until all bytes are read... + */ + + for (tbytes = 0, bytes = 0; + tbytes < (int)length; + tbytes += bytes, buffer += bytes) + { + DEBUG_printf(("9ipp_read_http: tbytes=%d, http->state=%d", tbytes, + http->state)); + + if (http->state == HTTP_WAITING) + break; + + if (http->used > 0 && http->data_encoding == HTTP_ENCODE_LENGTH) + { + /* + * Do "fast read" from HTTP buffer directly... + */ + + if (http->used > (int)(length - tbytes)) + bytes = (int)(length - tbytes); + else + bytes = http->used; + + if (bytes == 1) + buffer[0] = http->buffer[0]; + else + memcpy(buffer, http->buffer, bytes); + + http->used -= bytes; + http->data_remaining -= bytes; + + if (http->data_remaining <= INT_MAX) + http->_data_remaining = (int)http->data_remaining; + else + http->_data_remaining = INT_MAX; + + if (http->used > 0) + memmove(http->buffer, http->buffer + bytes, http->used); + + if (http->data_remaining == 0) + { + if (http->data_encoding == HTTP_ENCODE_CHUNKED) + { + /* + * Get the trailing CR LF after the chunk... + */ + + if (!httpGets(len, sizeof(len), http)) + return (-1); + } + + if (http->data_encoding != HTTP_ENCODE_CHUNKED) + { + if (http->state == HTTP_POST_RECV) + http->state ++; + else + http->state = HTTP_WAITING; + } + } + } + else + { + /* + * Wait a maximum of 1 second for data... + */ + + if (!http->blocking) + { + /* + * Wait up to 10 seconds for more data on non-blocking sockets... + */ + + if (!httpWait(http, 10000)) + { + /* + * Signal no data... + */ + + bytes = -1; + break; + } + } + + if ((bytes = httpRead2(http, (char *)buffer, length - tbytes)) < 0) + { +#ifdef WIN32 + break; +#else + if (errno != EAGAIN && errno != EINTR) + break; + + bytes = 0; +#endif /* WIN32 */ + } + else if (bytes == 0) + break; + } + } + + /* + * Return the number of bytes read... + */ + + if (tbytes == 0 && bytes < 0) + tbytes = -1; + + DEBUG_printf(("8ipp_read_http: Returning %d bytes", tbytes)); + + return (tbytes); +} + + +/* + * 'ipp_read_file()' - Read IPP data from a file. + */ + +static ssize_t /* O - Number of bytes read */ +ipp_read_file(int *fd, /* I - File descriptor */ + ipp_uchar_t *buffer, /* O - Read buffer */ + size_t length) /* I - Number of bytes to read */ +{ +#ifdef WIN32 + return ((ssize_t)read(*fd, buffer, (unsigned)length)); +#else + return (read(*fd, buffer, length)); +#endif /* WIN32 */ +} + + +/* + * 'ipp_set_value()' - Get the value element from an attribute, expanding it as + * needed. + */ + +static _ipp_value_t * /* O - IPP value element or NULL on error */ +ipp_set_value(ipp_t *ipp, /* IO - IPP message */ + ipp_attribute_t **attr, /* IO - IPP attribute */ + int element) /* I - Value number (0-based) */ +{ + ipp_attribute_t *temp, /* New attribute pointer */ + *current, /* Current attribute in list */ + *prev; /* Previous attribute in list */ + int alloc_values; /* Allocated values */ + + + /* + * If we are setting an existing value element, return it... + */ + + temp = *attr; + + if (temp->num_values <= 1) + alloc_values = 1; + else + alloc_values = (temp->num_values + IPP_MAX_VALUES - 1) & + ~(IPP_MAX_VALUES - 1); + + if (element < alloc_values) + { + if (element >= temp->num_values) + temp->num_values = element + 1; + + return (temp->values + element); + } + + /* + * Otherwise re-allocate the attribute - we allocate in groups of IPP_MAX_VALUE + * values when num_values > 1. + */ + + if (alloc_values < IPP_MAX_VALUES) + alloc_values = IPP_MAX_VALUES; + else + alloc_values += IPP_MAX_VALUES; + + DEBUG_printf(("4ipp_set_value: Reallocating for up to %d values.", + alloc_values)); + + /* + * Reallocate memory... + */ + + if ((temp = realloc(temp, sizeof(ipp_attribute_t) + + (alloc_values - 1) * sizeof(_ipp_value_t))) == NULL) + { + _cupsSetHTTPError(HTTP_ERROR); + DEBUG_puts("4ipp_set_value: Unable to resize attribute."); + return (NULL); + } + + /* + * Zero the new memory... + */ + + memset(temp->values + temp->num_values, 0, + (alloc_values - temp->num_values) * sizeof(_ipp_value_t)); + + if (temp != *attr) + { + /* + * Reset pointers in the list... + */ + + if (ipp->current == *attr && ipp->prev) + { + /* + * Use current "previous" pointer... + */ + + prev = ipp->prev; + } + else + { + /* + * Find this attribute in the linked list... + */ + + for (prev = NULL, current = ipp->attrs; + current && current != *attr; + prev = current, current = current->next); + + if (!current) + { + /* + * This is a serious error! + */ + + *attr = temp; + _cupsSetError(IPP_INTERNAL_ERROR, + _("IPP attribute is not a member of the message."), 1); + DEBUG_puts("4ipp_set_value: Unable to find attribute in message."); + return (NULL); + } + } + + if (prev) + prev->next = temp; + else + ipp->attrs = temp; + + ipp->current = temp; + ipp->prev = prev; + + if (ipp->last == *attr) + ipp->last = temp; + + *attr = temp; + } + + /* + * Return the value element... + */ + + if (element >= temp->num_values) + temp->num_values = element + 1; + + return (temp->values + element); +} + + +/* + * 'ipp_write_file()' - Write IPP data to a file. + */ + +static ssize_t /* O - Number of bytes written */ +ipp_write_file(int *fd, /* I - File descriptor */ + ipp_uchar_t *buffer, /* I - Data to write */ + size_t length) /* I - Number of bytes to write */ +{ +#ifdef WIN32 + return ((ssize_t)write(*fd, buffer, (unsigned)length)); +#else + return (write(*fd, buffer, length)); +#endif /* WIN32 */ +} + + +/* + * End of "$Id$". + */ diff --git a/cups/ipp.h b/cups/ipp.h new file mode 100644 index 0000000000..051cc38b9c --- /dev/null +++ b/cups/ipp.h @@ -0,0 +1,546 @@ +/* + * "$Id$" + * + * Internet Printing Protocol definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_IPP_H_ +# define _CUPS_IPP_H_ + +/* + * Include necessary headers... + */ + +# include "http.h" + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * IPP version string... + */ + +# define IPP_VERSION "\002\001" + +/* + * IPP registered port number... + * + * Note: Applications should never use IPP_PORT, but instead use the + * ippPort() function to allow overrides via the IPP_PORT environment + * variable and services file if needed! + */ + +# define IPP_PORT 631 + +/* + * Common limits... + */ + +# define IPP_MAX_LENGTH 32767 /* Maximum size of any single value */ +# define IPP_MAX_NAME 256 /* Maximum length of common name values */ +# define IPP_MAX_VALUES 8 /* Power-of-2 allocation increment */ + + +/* + * Types and structures... + */ + +typedef enum ipp_dstate_e /**** Document states ****/ +{ + IPP_DOCUMENT_PENDING = 3, + IPP_DOCUMENT_PROCESSING = 5, + IPP_DOCUMENT_CANCELED = 7, + IPP_DOCUMENT_ABORTED, + IPP_DOCUMENT_COMPLETED +} ipp_dstate_t; + +typedef enum ipp_finish_e /**** Finishings ****/ +{ + IPP_FINISHINGS_NONE = 3, /* No finishing */ + IPP_FINISHINGS_STAPLE, /* Staple (any location) */ + IPP_FINISHINGS_PUNCH, /* Punch (any location/count) */ + IPP_FINISHINGS_COVER, /* Add cover */ + IPP_FINISHINGS_BIND, /* Bind */ + IPP_FINISHINGS_SADDLE_STITCH, /* Staple interior */ + IPP_FINISHINGS_EDGE_STITCH, /* Stitch along any side */ + IPP_FINISHINGS_FOLD, /* Fold (any type) */ + IPP_FINISHINGS_TRIM, /* Trim (any type) */ + IPP_FINISHINGS_BALE, /* Bale (any type) */ + IPP_FINISHINGS_BOOKLET_MAKER, /* Fold to make booklet */ + IPP_FINISHINGS_JOB_OFFSET, /* Offset for binding (any type) */ + IPP_FINISHINGS_STAPLE_TOP_LEFT = 20, /* Staple top left corner */ + IPP_FINISHINGS_STAPLE_BOTTOM_LEFT, /* Staple bottom left corner */ + IPP_FINISHINGS_STAPLE_TOP_RIGHT, /* Staple top right corner */ + IPP_FINISHINGS_STAPLE_BOTTOM_RIGHT, /* Staple bottom right corner */ + IPP_FINISHINGS_EDGE_STITCH_LEFT, /* Stitch along left side */ + IPP_FINISHINGS_EDGE_STITCH_TOP, /* Stitch along top edge */ + IPP_FINISHINGS_EDGE_STITCH_RIGHT, /* Stitch along right side */ + IPP_FINISHINGS_EDGE_STITCH_BOTTOM, /* Stitch along bottom edge */ + IPP_FINISHINGS_STAPLE_DUAL_LEFT, /* Two staples on left */ + IPP_FINISHINGS_STAPLE_DUAL_TOP, /* Two staples on top */ + IPP_FINISHINGS_STAPLE_DUAL_RIGHT, /* Two staples on right */ + IPP_FINISHINGS_STAPLE_DUAL_BOTTOM, /* Two staples on bottom */ + IPP_FINISHINGS_BIND_LEFT = 50, /* Bind on left */ + IPP_FINISHINGS_BIND_TOP, /* Bind on top */ + IPP_FINISHINGS_BIND_RIGHT, /* Bind on right */ + IPP_FINISHINGS_BIND_BOTTOM, /* Bind on bottom */ + IPP_FINISHINGS_TRIM_AFTER_PAGES = 60, /* Trim output after each page */ + IPP_FINISHINGS_TRIM_AFTER_DOCUMENTS, /* Trim output after each document */ + IPP_FINISHINGS_TRIM_AFTER_COPIES, /* Trim output after each copy */ + IPP_FINISHINGS_TRIM_AFTER_JOB /* Trim output after job */ +} ipp_finish_t; + +typedef enum ipp_jcollate_e /**** Job collation types ****/ +{ + IPP_JOB_UNCOLLATED_SHEETS = 3, + IPP_JOB_COLLATED_DOCUMENTS, + IPP_JOB_UNCOLLATED_DOCUMENTS +} ipp_jcollate_t; + +typedef enum ipp_jstate_e /**** Job states ****/ +{ + IPP_JOB_PENDING = 3, /* Job is waiting to be printed */ + IPP_JOB_HELD, /* Job is held for printing */ + IPP_JOB_PROCESSING, /* Job is currently printing */ + IPP_JOB_STOPPED, /* Job has been stopped */ + IPP_JOB_CANCELED, /* Job has been canceled */ + IPP_JOB_ABORTED, /* Job has aborted due to error */ + IPP_JOB_COMPLETED /* Job has completed successfully */ + /* Legacy name for canceled state */ +#define IPP_JOB_CANCELLED IPP_JOB_CANCELED +} ipp_jstate_t; + +typedef enum ipp_op_e /**** IPP operations ****/ +{ + IPP_OP_CUPS_INVALID = -1, /* Invalid operation name for @link ippOpValue@ */ + IPP_PRINT_JOB = 0x0002, /* Print a single file */ + IPP_PRINT_URI, /* Print a single URL @private@ */ + IPP_VALIDATE_JOB, /* Validate job options */ + IPP_CREATE_JOB, /* Create an empty print job */ + IPP_SEND_DOCUMENT, /* Add a file to a job */ + IPP_SEND_URI, /* Add a URL to a job @private@ */ + IPP_CANCEL_JOB, /* Cancel a job */ + IPP_GET_JOB_ATTRIBUTES, /* Get job attributes */ + IPP_GET_JOBS, /* Get a list of jobs */ + IPP_GET_PRINTER_ATTRIBUTES, /* Get printer attributes */ + IPP_HOLD_JOB, /* Hold a job for printing */ + IPP_RELEASE_JOB, /* Release a job for printing */ + IPP_RESTART_JOB, /* Reprint a job */ + IPP_PAUSE_PRINTER = 0x0010, /* Stop a printer */ + IPP_RESUME_PRINTER, /* Start a printer */ + IPP_PURGE_JOBS, /* Cancel all jobs */ + IPP_SET_PRINTER_ATTRIBUTES, /* Set printer attributes @private@ */ + IPP_SET_JOB_ATTRIBUTES, /* Set job attributes */ + IPP_GET_PRINTER_SUPPORTED_VALUES, /* Get supported attribute values */ + IPP_CREATE_PRINTER_SUBSCRIPTION, /* Create a printer subscription @since CUPS 1.2/Mac OS X 10.5@ */ + IPP_CREATE_JOB_SUBSCRIPTION, /* Create a job subscription @since CUPS 1.2/Mac OS X 10.5@ */ + IPP_GET_SUBSCRIPTION_ATTRIBUTES, /* Get subscription attributes @since CUPS 1.2/Mac OS X 10.5@ */ + IPP_GET_SUBSCRIPTIONS, /* Get list of subscriptions @since CUPS 1.2/Mac OS X 10.5@ */ + IPP_RENEW_SUBSCRIPTION, /* Renew a printer subscription @since CUPS 1.2/Mac OS X 10.5@ */ + IPP_CANCEL_SUBSCRIPTION, /* Cancel a subscription @since CUPS 1.2/Mac OS X 10.5@ */ + IPP_GET_NOTIFICATIONS, /* Get notification events @since CUPS 1.2/Mac OS X 10.5@ */ + IPP_SEND_NOTIFICATIONS, /* Send notification events @private@ */ + IPP_GET_RESOURCE_ATTRIBUTES, /* Get resource attributes @private@ */ + IPP_GET_RESOURCE_DATA, /* Get resource data @private@ */ + IPP_GET_RESOURCES, /* Get list of resources @private@ */ + IPP_GET_PRINT_SUPPORT_FILES, /* Get printer support files @private@ */ + IPP_ENABLE_PRINTER, /* Start a printer */ + IPP_DISABLE_PRINTER, /* Stop a printer */ + IPP_PAUSE_PRINTER_AFTER_CURRENT_JOB, /* Stop printer after the current job @private@ */ + IPP_HOLD_NEW_JOBS, /* Hold new jobs @private@ */ + IPP_RELEASE_HELD_NEW_JOBS, /* Release new jobs @private@ */ + IPP_DEACTIVATE_PRINTER, /* Stop a printer @private@ */ + IPP_ACTIVATE_PRINTER, /* Start a printer @private@ */ + IPP_RESTART_PRINTER, /* Restart a printer @private@ */ + IPP_SHUTDOWN_PRINTER, /* Turn a printer off @private@ */ + IPP_STARTUP_PRINTER, /* Turn a printer on @private@ */ + IPP_REPROCESS_JOB, /* Reprint a job @private@ */ + IPP_CANCEL_CURRENT_JOB, /* Cancel the current job @private@ */ + IPP_SUSPEND_CURRENT_JOB, /* Suspend the current job @private@ */ + IPP_RESUME_JOB, /* Resume the current job @private@ */ + IPP_PROMOTE_JOB, /* Promote a job to print sooner @private@ */ + IPP_SCHEDULE_JOB_AFTER, /* Schedule a job to print after another @private@ */ + IPP_CANCEL_DOCUMENT = 0x0033, /* Cancel-Document @private@ */ + IPP_GET_DOCUMENT_ATTRIBUTES, /* Get-Document-Attributes @private@ */ + IPP_GET_DOCUMENTS, /* Get-Documents @private@ */ + IPP_DELETE_DOCUMENT, /* Delete-Document @private@ */ + IPP_SET_DOCUMENT_ATTRIBUTES, /* Set-Document-Attributes @private@ */ + IPP_CANCEL_JOBS, /* Cancel-Jobs */ + IPP_CANCEL_MY_JOBS, /* Cancel-My-Jobs */ + IPP_RESUBMIT_JOB, /* Resubmit-Job */ + IPP_CLOSE_JOB, /* Close-Job */ + IPP_IDENTIFY_PRINTER, /* Identify-Printer (proposed IPP JPS3) */ + IPP_PRIVATE = 0x4000, /* Reserved @private@ */ + CUPS_GET_DEFAULT, /* Get the default printer */ + CUPS_GET_PRINTERS, /* Get a list of printers and/or classes */ + CUPS_ADD_MODIFY_PRINTER, /* Add or modify a printer */ + CUPS_DELETE_PRINTER, /* Delete a printer */ + CUPS_GET_CLASSES, /* Get a list of classes @deprecated@ */ + CUPS_ADD_MODIFY_CLASS, /* Add or modify a class */ + CUPS_DELETE_CLASS, /* Delete a class */ + CUPS_ACCEPT_JOBS, /* Accept new jobs on a printer */ + CUPS_REJECT_JOBS, /* Reject new jobs on a printer */ + CUPS_SET_DEFAULT, /* Set the default printer */ + CUPS_GET_DEVICES, /* Get a list of supported devices */ + CUPS_GET_PPDS, /* Get a list of supported drivers */ + CUPS_MOVE_JOB, /* Move a job to a different printer */ + CUPS_AUTHENTICATE_JOB, /* Authenticate a job @since CUPS 1.2/Mac OS X 10.5@ */ + CUPS_GET_PPD, /* Get a PPD file @since CUPS 1.3/Mac OS X 10.5@ */ + CUPS_GET_DOCUMENT = 0x4027 /* Get a document file @since CUPS 1.4/Mac OS X 10.6@ */ + + /* Legacy names for the add operations */ +#define CUPS_ADD_PRINTER CUPS_ADD_MODIFY_PRINTER +#define CUPS_ADD_CLASS CUPS_ADD_MODIFY_CLASS +} ipp_op_t; + +typedef enum ipp_orient_e /**** Orientation values ****/ +{ + IPP_PORTRAIT = 3, /* No rotation */ + IPP_LANDSCAPE, /* 90 degrees counter-clockwise */ + IPP_REVERSE_LANDSCAPE, /* 90 degrees clockwise */ + IPP_REVERSE_PORTRAIT /* 180 degrees */ +} ipp_orient_t; + +typedef enum ipp_pstate_e /**** Printer states ****/ +{ + IPP_PRINTER_IDLE = 3, /* Printer is idle */ + IPP_PRINTER_PROCESSING, /* Printer is working */ + IPP_PRINTER_STOPPED /* Printer is stopped */ +} ipp_pstate_t; + +typedef enum ipp_quality_e /**** Qualities ****/ +{ + IPP_QUALITY_DRAFT = 3, /* Draft quality */ + IPP_QUALITY_NORMAL, /* Normal quality */ + IPP_QUALITY_HIGH /* High quality */ +} ipp_quality_t; + +typedef enum ipp_res_e /**** Resolution units ****/ +{ + IPP_RES_PER_INCH = 3, /* Pixels per inch */ + IPP_RES_PER_CM /* Pixels per centimeter */ +} ipp_res_t; + +typedef enum ipp_state_e /**** IPP states ****/ +{ + IPP_ERROR = -1, /* An error occurred */ + IPP_IDLE, /* Nothing is happening/request completed */ + IPP_HEADER, /* The request header needs to be sent/received */ + IPP_ATTRIBUTE, /* One or more attributes need to be sent/received */ + IPP_DATA /* IPP request data needs to be sent/received */ +} ipp_state_t; + +typedef enum ipp_status_e /**** IPP status codes ****/ +{ + IPP_STATUS_CUPS_INVALID = -1, /* Invalid status name for @link ippErrorValue@ */ + IPP_OK = 0x0000, /* successful-ok */ + IPP_OK_SUBST, /* successful-ok-ignored-or-substituted-attributes */ + IPP_OK_CONFLICT, /* successful-ok-conflicting-attributes */ + IPP_OK_IGNORED_SUBSCRIPTIONS, /* successful-ok-ignored-subscriptions */ + IPP_OK_IGNORED_NOTIFICATIONS, /* successful-ok-ignored-notifications @private@ */ + IPP_OK_TOO_MANY_EVENTS, /* successful-ok-too-many-events */ + IPP_OK_BUT_CANCEL_SUBSCRIPTION, /* successful-ok-but-cancel-subscription @private@ */ + IPP_OK_EVENTS_COMPLETE, /* successful-ok-events-complete */ + IPP_REDIRECTION_OTHER_SITE = 0x200, /* redirection-other-site @private@ */ + CUPS_SEE_OTHER = 0x280, /* cups-see-other */ + IPP_BAD_REQUEST = 0x0400, /* client-error-bad-request */ + IPP_FORBIDDEN, /* client-error-forbidden */ + IPP_NOT_AUTHENTICATED, /* client-error-not-authenticated */ + IPP_NOT_AUTHORIZED, /* client-error-not-authorized */ + IPP_NOT_POSSIBLE, /* client-error-not-possible */ + IPP_TIMEOUT, /* client-error-timeout */ + IPP_NOT_FOUND, /* client-error-not-found */ + IPP_GONE, /* client-error-gone */ + IPP_REQUEST_ENTITY, /* client-error-request-entity-too-large */ + IPP_REQUEST_VALUE, /* client-error-request-value-too-long */ + IPP_DOCUMENT_FORMAT, /* client-error-document-format-not-supported */ + IPP_ATTRIBUTES, /* client-error-attributes-or-values-not-supported */ + IPP_URI_SCHEME, /* client-error-uri-scheme-not-supported */ + IPP_CHARSET, /* client-error-charset-not-supported */ + IPP_CONFLICT, /* client-error-conflicting-attributes */ + IPP_COMPRESSION_NOT_SUPPORTED, /* client-error-compression-not-supported */ + IPP_COMPRESSION_ERROR, /* client-error-compression-error */ + IPP_DOCUMENT_FORMAT_ERROR, /* client-error-document-format-error */ + IPP_DOCUMENT_ACCESS_ERROR, /* client-error-document-access-error */ + IPP_ATTRIBUTES_NOT_SETTABLE, /* client-error-attributes-not-settable */ + IPP_IGNORED_ALL_SUBSCRIPTIONS, /* client-error-ignored-all-subscriptions */ + IPP_TOO_MANY_SUBSCRIPTIONS, /* client-error-too-many-subscriptions */ + IPP_IGNORED_ALL_NOTIFICATIONS, /* client-error-ignored-all-notifications @private@ */ + IPP_PRINT_SUPPORT_FILE_NOT_FOUND, /* client-error-print-support-file-not-found @private@ */ + IPP_DOCUMENT_PASSWORD_ERROR, /* client-error-document-password-error */ + IPP_DOCUMENT_PERMISSION_ERROR, /* client-error-document-permission-error */ + IPP_DOCUMENT_SECURITY_ERROR, /* client-error-document-security-error */ + IPP_DOCUMENT_UNPRINTABLE_ERROR, /* client-error-document-unprintable-error */ + + IPP_INTERNAL_ERROR = 0x0500, /* server-error-internal-error */ + IPP_OPERATION_NOT_SUPPORTED, /* server-error-operation-not-supported */ + IPP_SERVICE_UNAVAILABLE, /* server-error-service-unavailable */ + IPP_VERSION_NOT_SUPPORTED, /* server-error-version-not-supported */ + IPP_DEVICE_ERROR, /* server-error-device-error */ + IPP_TEMPORARY_ERROR, /* server-error-temporary-error */ + IPP_NOT_ACCEPTING, /* server-error-not-accepting-jobs */ + IPP_PRINTER_BUSY, /* server-error-busy */ + IPP_ERROR_JOB_CANCELED, /* server-error-job-canceled */ + IPP_MULTIPLE_JOBS_NOT_SUPPORTED, /* server-error-multiple-document-jobs-not-supported */ + IPP_PRINTER_IS_DEACTIVATED, /* server-error-printer-is-deactivated */ + IPP_TOO_MANY_JOBS, /* server-error-too-many-jobs */ + IPP_TOO_MANY_DOCUMENTS, /* server-error-too-many-documents */ + + IPP_AUTHENTICATION_CANCELED = 0x1000, /* Authentication canceled by user @since CUPS 1.5/Mac OS X 10.7@ */ + IPP_PKI_ERROR, /* Error negotiating a secure connection @since CUPS 1.5/Mac OS X 10.7@ */ + IPP_UPGRADE_REQUIRED /* TLS upgrade required */ + + /* Legacy name for canceled status */ +#define IPP_ERROR_JOB_CANCELLED IPP_ERROR_JOB_CANCELED + +} ipp_status_t; + +typedef enum ipp_tag_e /**** Format tags for attributes ****/ +{ + IPP_TAG_CUPS_INVALID = -1, /* Invalid tag name for @link ippTagValue@ */ + IPP_TAG_ZERO = 0x00, /* Zero tag - used for separators */ + IPP_TAG_OPERATION, /* Operation group */ + IPP_TAG_JOB, /* Job group */ + IPP_TAG_END, /* End-of-attributes */ + IPP_TAG_PRINTER, /* Printer group */ + IPP_TAG_UNSUPPORTED_GROUP, /* Unsupported attributes group */ + IPP_TAG_SUBSCRIPTION, /* Subscription group */ + IPP_TAG_EVENT_NOTIFICATION, /* Event group */ + IPP_TAG_RESOURCE, /* Resource group @private@ */ + IPP_TAG_DOCUMENT, /* Document group */ + IPP_TAG_UNSUPPORTED_VALUE = 0x10, /* Unsupported value */ + IPP_TAG_DEFAULT, /* Default value */ + IPP_TAG_UNKNOWN, /* Unknown value */ + IPP_TAG_NOVALUE, /* No-value value */ + IPP_TAG_NOTSETTABLE = 0x15, /* Not-settable value */ + IPP_TAG_DELETEATTR, /* Delete-attribute value */ + IPP_TAG_ADMINDEFINE, /* Admin-defined value */ + IPP_TAG_INTEGER = 0x21, /* Integer value */ + IPP_TAG_BOOLEAN, /* Boolean value */ + IPP_TAG_ENUM, /* Enumeration value */ + IPP_TAG_STRING = 0x30, /* Octet string value */ + IPP_TAG_DATE, /* Date/time value */ + IPP_TAG_RESOLUTION, /* Resolution value */ + IPP_TAG_RANGE, /* Range value */ + IPP_TAG_BEGIN_COLLECTION, /* Beginning of collection value */ + IPP_TAG_TEXTLANG, /* Text-with-language value */ + IPP_TAG_NAMELANG, /* Name-with-language value */ + IPP_TAG_END_COLLECTION, /* End of collection value */ + IPP_TAG_TEXT = 0x41, /* Text value */ + IPP_TAG_NAME, /* Name value */ + IPP_TAG_RESERVED_STRING, /* Reserved for future string value @private@ */ + IPP_TAG_KEYWORD, /* Keyword value */ + IPP_TAG_URI, /* URI value */ + IPP_TAG_URISCHEME, /* URI scheme value */ + IPP_TAG_CHARSET, /* Character set value */ + IPP_TAG_LANGUAGE, /* Language value */ + IPP_TAG_MIMETYPE, /* MIME media type value */ + IPP_TAG_MEMBERNAME, /* Collection member name value */ + IPP_TAG_EXTENSION = 0x7f, /* Extension point for 32-bit tags */ + IPP_TAG_MASK = 0x7fffffff, /* Mask for copied attribute values @private@ */ + /* The following expression is used to avoid compiler warnings with +/-0x80000000 */ + IPP_TAG_COPY = -0x7fffffff-1 /* Bitflag for copied attribute values @private@ */ +} ipp_tag_t; + +typedef unsigned char ipp_uchar_t; /**** Unsigned 8-bit integer/character ****/ +typedef struct _ipp_s ipp_t; /**** IPP request/response data ****/ +typedef struct _ipp_attribute_s ipp_attribute_t; + /**** IPP attribute ****/ + +/**** New in CUPS 1.2 ****/ +typedef ssize_t (*ipp_iocb_t)(void *context, ipp_uchar_t *buffer, size_t bytes); + /**** IPP IO Callback Function @since CUPS 1.2/Mac OS X 10.5@ ****/ + +/**** New in CUPS 1.6 ****/ +typedef int (*ipp_copycb_t)(void *context, ipp_t *dst, ipp_attribute_t *attr); + + +/* + * Prototypes... + */ + +extern ipp_attribute_t *ippAddBoolean(ipp_t *ipp, ipp_tag_t group, + const char *name, char value); +extern ipp_attribute_t *ippAddBooleans(ipp_t *ipp, ipp_tag_t group, + const char *name, int num_values, + const char *values); +extern ipp_attribute_t *ippAddDate(ipp_t *ipp, ipp_tag_t group, + const char *name, const ipp_uchar_t *value); +extern ipp_attribute_t *ippAddInteger(ipp_t *ipp, ipp_tag_t group, + ipp_tag_t value_tag, const char *name, + int value); +extern ipp_attribute_t *ippAddIntegers(ipp_t *ipp, ipp_tag_t group, + ipp_tag_t value_tag, const char *name, + int num_values, const int *values); +extern ipp_attribute_t *ippAddRange(ipp_t *ipp, ipp_tag_t group, + const char *name, int lower, int upper); +extern ipp_attribute_t *ippAddRanges(ipp_t *ipp, ipp_tag_t group, + const char *name, int num_values, + const int *lower, const int *upper); +extern ipp_attribute_t *ippAddResolution(ipp_t *ipp, ipp_tag_t group, + const char *name, ipp_res_t units, + int xres, int yres); +extern ipp_attribute_t *ippAddResolutions(ipp_t *ipp, ipp_tag_t group, + const char *name, int num_values, + ipp_res_t units, const int *xres, + const int *yres); +extern ipp_attribute_t *ippAddSeparator(ipp_t *ipp); +extern ipp_attribute_t *ippAddString(ipp_t *ipp, ipp_tag_t group, + ipp_tag_t value_tag, const char *name, + const char *language, const char *value); +extern ipp_attribute_t *ippAddStrings(ipp_t *ipp, ipp_tag_t group, + ipp_tag_t value_tag, const char *name, + int num_values, const char *language, + const char * const *values); +extern time_t ippDateToTime(const ipp_uchar_t *date); +extern void ippDelete(ipp_t *ipp); +extern const char *ippErrorString(ipp_status_t error); +extern ipp_attribute_t *ippFindAttribute(ipp_t *ipp, const char *name, + ipp_tag_t value_tag); +extern ipp_attribute_t *ippFindNextAttribute(ipp_t *ipp, const char *name, + ipp_tag_t value_tag); +extern size_t ippLength(ipp_t *ipp); +extern ipp_t *ippNew(void); +extern ipp_state_t ippRead(http_t *http, ipp_t *ipp); +extern const ipp_uchar_t *ippTimeToDate(time_t t); +extern ipp_state_t ippWrite(http_t *http, ipp_t *ipp); +extern int ippPort(void); +extern void ippSetPort(int p); + +/**** New in CUPS 1.1.19 ****/ +extern ipp_attribute_t *ippAddCollection(ipp_t *ipp, ipp_tag_t group, + const char *name, ipp_t *value) _CUPS_API_1_1_19; +extern ipp_attribute_t *ippAddCollections(ipp_t *ipp, ipp_tag_t group, + const char *name, int num_values, + const ipp_t **values) _CUPS_API_1_1_19; +extern void ippDeleteAttribute(ipp_t *ipp, ipp_attribute_t *attr) _CUPS_API_1_1_19; +extern ipp_state_t ippReadFile(int fd, ipp_t *ipp) _CUPS_API_1_1_19; +extern ipp_state_t ippWriteFile(int fd, ipp_t *ipp) _CUPS_API_1_1_19; + +/**** New in CUPS 1.2/Mac OS X 10.5 ****/ +extern ipp_attribute_t *ippAddOctetString(ipp_t *ipp, ipp_tag_t group, + const char *name, + const void *data, int datalen) _CUPS_API_1_2; +extern ipp_status_t ippErrorValue(const char *name) _CUPS_API_1_2; +extern ipp_t *ippNewRequest(ipp_op_t op) _CUPS_API_1_2; +extern const char *ippOpString(ipp_op_t op) _CUPS_API_1_2; +extern ipp_op_t ippOpValue(const char *name) _CUPS_API_1_2; +extern ipp_state_t ippReadIO(void *src, ipp_iocb_t cb, int blocking, + ipp_t *parent, ipp_t *ipp) _CUPS_API_1_2; +extern ipp_state_t ippWriteIO(void *dst, ipp_iocb_t cb, int blocking, + ipp_t *parent, ipp_t *ipp) _CUPS_API_1_2; + +/**** New in CUPS 1.4/Mac OS X 10.6 ****/ +extern const char *ippTagString(ipp_tag_t tag) _CUPS_API_1_4; +extern ipp_tag_t ippTagValue(const char *name) _CUPS_API_1_4; + +/**** New in CUPS 1.6 ****/ +extern ipp_attribute_t *ippAddOutOfBand(ipp_t *ipp, ipp_tag_t group, + ipp_tag_t value_tag, const char *name) + _CUPS_API_1_6; +extern size_t ippAttributeString(ipp_attribute_t *attr, char *buffer, + size_t bufsize) _CUPS_API_1_6; +extern ipp_attribute_t *ippCopyAttribute(ipp_t *dst, ipp_attribute_t *attr, + int quickcopy) _CUPS_API_1_6; +extern int ippCopyAttributes(ipp_t *dst, ipp_t *src, + int quickcopy, ipp_copycb_t cb, + void *context) _CUPS_API_1_6; +extern int ippDeleteValues(ipp_t *ipp, ipp_attribute_t **attr, + int element, int count) _CUPS_API_1_6; +extern const char *ippEnumString(const char *attrname, int enumvalue) + _CUPS_API_1_6; +extern int ippEnumValue(const char *attrname, + const char *enumstring) _CUPS_API_1_6; +extern ipp_attribute_t *ippFirstAttribute(ipp_t *ipp) _CUPS_API_1_6; +extern int ippGetBoolean(ipp_attribute_t *attr, int element) + _CUPS_API_1_6; +extern ipp_t *ippGetCollection(ipp_attribute_t *attr, + int element) _CUPS_API_1_6; +extern int ippGetCount(ipp_attribute_t *attr) _CUPS_API_1_6; +extern const ipp_uchar_t *ippGetDate(ipp_attribute_t *attr, int element) + _CUPS_API_1_6; +extern ipp_tag_t ippGetGroupTag(ipp_attribute_t *attr) _CUPS_API_1_6; +extern int ippGetInteger(ipp_attribute_t *attr, int element) + _CUPS_API_1_6; +extern const char *ippGetName(ipp_attribute_t *attr) _CUPS_API_1_6; +extern ipp_op_t ippGetOperation(ipp_t *ipp) _CUPS_API_1_6; +extern int ippGetRange(ipp_attribute_t *attr, int element, + int *upper) _CUPS_API_1_6; +extern int ippGetRequestId(ipp_t *ipp) _CUPS_API_1_6; +extern int ippGetResolution(ipp_attribute_t *attr, int element, + int *yres, ipp_res_t *units) + _CUPS_API_1_6; +extern ipp_state_t ippGetState(ipp_t *ipp) _CUPS_API_1_6; +extern ipp_status_t ippGetStatusCode(ipp_t *ipp) _CUPS_API_1_6; +extern const char *ippGetString(ipp_attribute_t *attr, int element, + const char **language) _CUPS_API_1_6; +extern ipp_tag_t ippGetValueTag(ipp_attribute_t *attr) _CUPS_API_1_6; +extern int ippGetVersion(ipp_t *ipp, int *minor) _CUPS_API_1_6; +extern ipp_attribute_t *ippNextAttribute(ipp_t *ipp) _CUPS_API_1_6; +extern int ippSetBoolean(ipp_t *ipp, ipp_attribute_t **attr, + int element, int boolvalue) _CUPS_API_1_6; +extern int ippSetCollection(ipp_t *ipp, ipp_attribute_t **attr, + int element, ipp_t *colvalue) + _CUPS_API_1_6; +extern int ippSetDate(ipp_t *ipp, ipp_attribute_t **attr, + int element, const ipp_uchar_t *datevalue) + _CUPS_API_1_6; +extern int ippSetGroupTag(ipp_t *ipp, ipp_attribute_t **attr, + ipp_tag_t group_tag) _CUPS_API_1_6; +extern int ippSetInteger(ipp_t *ipp, ipp_attribute_t **attr, + int element, int intvalue) _CUPS_API_1_6; +extern int ippSetName(ipp_t *ipp, ipp_attribute_t **attr, + const char *name) _CUPS_API_1_6; +extern int ippSetOperation(ipp_t *ipp, ipp_op_t op) _CUPS_API_1_6; +extern int ippSetRange(ipp_t *ipp, ipp_attribute_t **attr, + int element, int lowervalue, int uppervalue) + _CUPS_API_1_6; +extern int ippSetRequestId(ipp_t *ipp, int request_id) + _CUPS_API_1_6; +extern int ippSetResolution(ipp_t *ipp, ipp_attribute_t **attr, + int element, ipp_res_t unitsvalue, + int xresvalue, int yresvalue) + _CUPS_API_1_6; +extern int ippSetState(ipp_t *ipp, ipp_state_t state) + _CUPS_API_1_6; +extern int ippSetStatusCode(ipp_t *ipp, ipp_status_t status) + _CUPS_API_1_6; +extern int ippSetString(ipp_t *ipp, ipp_attribute_t **attr, + int element, const char *strvalue) + _CUPS_API_1_6; +extern int ippSetValueTag(ipp_t *ipp, ipp_attribute_t **attr, + ipp_tag_t value_tag) _CUPS_API_1_6; +extern int ippSetVersion(ipp_t *ipp, int major, int minor) + _CUPS_API_1_6; + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +} +# endif /* __cplusplus */ +#endif /* !_CUPS_IPP_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/langprintf.c b/cups/langprintf.c new file mode 100644 index 0000000000..9a308dbf2d --- /dev/null +++ b/cups/langprintf.c @@ -0,0 +1,352 @@ +/* + * "$Id$" + * + * Localized printf/puts functions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 2002-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * _cupsLangPrintError() - Print a message followed by a standard error. + * _cupsLangPrintFilter() - Print a formatted filter message string to a file. + * _cupsLangPrintf() - Print a formatted message string to a file. + * _cupsLangPuts() - Print a static message string to a file. + * _cupsSetLocale() - Set the current locale and transcode the + * command-line. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" + + +/* + * '_cupsLangPrintError()' - Print a message followed by a standard error. + */ + +void +_cupsLangPrintError(const char *prefix, /* I - Non-localized message prefix */ + const char *message)/* I - Message */ +{ + int bytes; /* Number of bytes formatted */ + int last_errno; /* Last error */ + char buffer[2048], /* Message buffer */ + *bufptr, /* Pointer into buffer */ + output[8192]; /* Output buffer */ + _cups_globals_t *cg; /* Global data */ + + + /* + * Range check... + */ + + if (!message) + return; + + /* + * Save the errno value... + */ + + last_errno = errno; + + /* + * Get the message catalog... + */ + + cg = _cupsGlobals(); + + if (!cg->lang_default) + cg->lang_default = cupsLangDefault(); + + /* + * Format the message... + */ + + if (prefix) + { + snprintf(buffer, sizeof(buffer), "%s:", prefix); + bufptr = buffer + strlen(buffer); + } + else + bufptr = buffer; + + snprintf(bufptr, sizeof(buffer) - (bufptr - buffer), + /* TRANSLATORS: Message is "subject: error" */ + _cupsLangString(cg->lang_default, _("%s: %s")), + _cupsLangString(cg->lang_default, message), strerror(last_errno)); + strlcat(buffer, "\n", sizeof(buffer)); + + /* + * Convert and write to stderr... + */ + + bytes = cupsUTF8ToCharset(output, (cups_utf8_t *)buffer, sizeof(output), + cg->lang_default->encoding); + + if (bytes > 0) + fwrite(output, 1, bytes, stderr); +} + + +/* + * '_cupsLangPrintFilter()' - Print a formatted filter message string to a file. + */ + +int /* O - Number of bytes written */ +_cupsLangPrintFilter( + FILE *fp, /* I - File to write to */ + const char *prefix, /* I - Non-localized message prefix */ + const char *message, /* I - Message string to use */ + ...) /* I - Additional arguments as needed */ +{ + int bytes; /* Number of bytes formatted */ + char temp[2048], /* Temporary format buffer */ + buffer[2048], /* Message buffer */ + output[8192]; /* Output buffer */ + va_list ap; /* Pointer to additional arguments */ + _cups_globals_t *cg; /* Global data */ + + + /* + * Range check... + */ + + if (!fp || !message) + return (-1); + + cg = _cupsGlobals(); + + if (!cg->lang_default) + cg->lang_default = cupsLangDefault(); + + /* + * Format the string... + */ + + va_start(ap, message); + snprintf(temp, sizeof(temp), "%s: %s\n", prefix, + _cupsLangString(cg->lang_default, message)); + vsnprintf(buffer, sizeof(buffer), temp, ap); + va_end(ap); + + /* + * Transcode to the destination charset... + */ + + bytes = cupsUTF8ToCharset(output, (cups_utf8_t *)buffer, sizeof(output), + cg->lang_default->encoding); + + /* + * Write the string and return the number of bytes written... + */ + + if (bytes > 0) + return ((int)fwrite(output, 1, bytes, fp)); + else + return (bytes); +} + + +/* + * '_cupsLangPrintf()' - Print a formatted message string to a file. + */ + +int /* O - Number of bytes written */ +_cupsLangPrintf(FILE *fp, /* I - File to write to */ + const char *message, /* I - Message string to use */ + ...) /* I - Additional arguments as needed */ +{ + int bytes; /* Number of bytes formatted */ + char buffer[2048], /* Message buffer */ + output[8192]; /* Output buffer */ + va_list ap; /* Pointer to additional arguments */ + _cups_globals_t *cg; /* Global data */ + + + /* + * Range check... + */ + + if (!fp || !message) + return (-1); + + cg = _cupsGlobals(); + + if (!cg->lang_default) + cg->lang_default = cupsLangDefault(); + + /* + * Format the string... + */ + + va_start(ap, message); + vsnprintf(buffer, sizeof(buffer) - 1, + _cupsLangString(cg->lang_default, message), ap); + va_end(ap); + + strlcat(buffer, "\n", sizeof(buffer)); + + /* + * Transcode to the destination charset... + */ + + bytes = cupsUTF8ToCharset(output, (cups_utf8_t *)buffer, sizeof(output), + cg->lang_default->encoding); + + /* + * Write the string and return the number of bytes written... + */ + + if (bytes > 0) + return ((int)fwrite(output, 1, bytes, fp)); + else + return (bytes); +} + + +/* + * '_cupsLangPuts()' - Print a static message string to a file. + */ + +int /* O - Number of bytes written */ +_cupsLangPuts(FILE *fp, /* I - File to write to */ + const char *message) /* I - Message string to use */ +{ + int bytes; /* Number of bytes formatted */ + char output[8192]; /* Message buffer */ + _cups_globals_t *cg; /* Global data */ + + + /* + * Range check... + */ + + if (!fp || !message) + return (-1); + + cg = _cupsGlobals(); + + if (!cg->lang_default) + cg->lang_default = cupsLangDefault(); + + /* + * Transcode to the destination charset... + */ + + bytes = cupsUTF8ToCharset(output, + (cups_utf8_t *)_cupsLangString(cg->lang_default, + message), + sizeof(output) - 4, cg->lang_default->encoding); + bytes += cupsUTF8ToCharset(output + bytes, (cups_utf8_t *)"\n", + sizeof(output) - bytes, + cg->lang_default->encoding); + + /* + * Write the string and return the number of bytes written... + */ + + if (bytes > 0) + return ((int)fwrite(output, 1, bytes, fp)); + else + return (bytes); +} + + +/* + * '_cupsSetLocale()' - Set the current locale and transcode the command-line. + */ + +void +_cupsSetLocale(char *argv[]) /* IO - Command-line arguments */ +{ + int i; /* Looping var */ + char buffer[8192]; /* Command-line argument buffer */ + _cups_globals_t *cg; /* Global data */ +#ifdef LC_TIME + const char *lc_time; /* Current LC_TIME value */ + char new_lc_time[255], /* New LC_TIME value */ + *charset; /* Pointer to character set */ +#endif /* LC_TIME */ + + + /* + * Set the locale so that times, etc. are displayed properly. + * + * Unfortunately, while we need the localized time value, we *don't* + * want to use the localized charset for the time value, so we need + * to set LC_TIME to the locale name with .UTF-8 on the end (if + * the locale includes a character set specifier...) + */ + + setlocale(LC_ALL, ""); + +#ifdef LC_TIME + if ((lc_time = setlocale(LC_TIME, NULL)) == NULL) + lc_time = setlocale(LC_ALL, NULL); + + if (lc_time) + { + strlcpy(new_lc_time, lc_time, sizeof(new_lc_time)); + if ((charset = strchr(new_lc_time, '.')) == NULL) + charset = new_lc_time + strlen(new_lc_time); + + strlcpy(charset, ".UTF-8", sizeof(new_lc_time) - (charset - new_lc_time)); + } + else + strcpy(new_lc_time, "C"); + + setlocale(LC_TIME, new_lc_time); +#endif /* LC_TIME */ + + /* + * Initialize the default language info... + */ + + cg = _cupsGlobals(); + + if (!cg->lang_default) + cg->lang_default = cupsLangDefault(); + + /* + * Transcode the command-line arguments from the locale charset to + * UTF-8... + */ + + if (cg->lang_default->encoding != CUPS_US_ASCII && + cg->lang_default->encoding != CUPS_UTF8) + { + for (i = 1; argv[i]; i ++) + { + /* + * Try converting from the locale charset to UTF-8... + */ + + if (cupsCharsetToUTF8((cups_utf8_t *)buffer, argv[i], sizeof(buffer), + cg->lang_default->encoding) < 0) + continue; + + /* + * Save the new string if it differs from the original... + */ + + if (strcmp(buffer, argv[i])) + argv[i] = strdup(buffer); + } + } +} + + +/* + * End of "$Id$". + */ diff --git a/cups/language-private.h b/cups/language-private.h new file mode 100644 index 0000000000..6a2a886b13 --- /dev/null +++ b/cups/language-private.h @@ -0,0 +1,85 @@ +/* + * "$Id$" + * + * Private localization support for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_LANGUAGE_PRIVATE_H_ +# define _CUPS_LANGUAGE_PRIVATE_H_ + +/* + * Include necessary headers... + */ + +# include +# include + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Macro for localized text... + */ + +# define _(x) x + + +/* + * Types... + */ + +typedef struct _cups_message_s /**** Message catalog entry ****/ +{ + char *id, /* Original string */ + *str; /* Localized string */ +} _cups_message_t; + + +/* + * Prototypes... + */ + +# ifdef __APPLE__ +extern const char *_cupsAppleLanguage(const char *locale, char *language, + size_t langsize); +# endif /* __APPLE__ */ +extern void _cupsCharmapFlush(void); +extern const char *_cupsEncodingName(cups_encoding_t encoding); +extern void _cupsLangPrintError(const char *prefix, + const char *message); +extern int _cupsLangPrintFilter(FILE *fp, const char *prefix, + const char *message, ...) + __attribute__ ((__format__ (__printf__, 3, 4))); +extern int _cupsLangPrintf(FILE *fp, const char *message, ...) + __attribute__ ((__format__ (__printf__, 2, 3))); +extern int _cupsLangPuts(FILE *fp, const char *message); +extern const char *_cupsLangString(cups_lang_t *lang, + const char *message); +extern void _cupsMessageFree(cups_array_t *a); +extern cups_array_t *_cupsMessageLoad(const char *filename, int unquote); +extern const char *_cupsMessageLookup(cups_array_t *a, const char *m); +extern void _cupsSetLocale(char *argv[]); + + +# ifdef __cplusplus +} +# endif /* __cplusplus */ + +#endif /* !_CUPS_LANGUAGE_PRIVATE_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/language.c b/cups/language.c new file mode 100644 index 0000000000..fac7a53093 --- /dev/null +++ b/cups/language.c @@ -0,0 +1,1518 @@ +/* + * "$Id$" + * + * I18N/language support for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * _cupsAppleLanguage() - Get the Apple language identifier associated with + * a locale ID. + * _cupsEncodingName() - Return the character encoding name string for the + * given encoding enumeration. + * cupsLangDefault() - Return the default language. + * cupsLangEncoding() - Return the character encoding (us-ascii, etc.) for + * the given language. + * cupsLangFlush() - Flush all language data out of the cache. + * cupsLangFree() - Free language data. + * cupsLangGet() - Get a language. + * _cupsLangString() - Get a message string. + * _cupsMessageFree() - Free a messages array. + * _cupsMessageLoad() - Load a .po file into a messages array. + * _cupsMessageLookup() - Lookup a message string. + * appleLangDefault() - Get the default locale string. + * appleMessageLoad() - Load a message catalog from a localizable bundle. + * cups_cache_lookup() - Lookup a language in the cache... + * cups_message_compare() - Compare two messages. + * cups_message_free() - Free a message. + * cups_message_load() - Load the message catalog for a language. + * cups_unquote() - Unquote characters in strings... + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#ifdef HAVE_LANGINFO_H +# include +#endif /* HAVE_LANGINFO_H */ +#ifdef WIN32 +# include +#else +# include +#endif /* WIN32 */ +#ifdef HAVE_COREFOUNDATION_H +# include +#endif /* HAVE_COREFOUNDATION_H */ + + +/* + * Local globals... + */ + +static _cups_mutex_t lang_mutex = _CUPS_MUTEX_INITIALIZER; + /* Mutex to control access to cache */ +static cups_lang_t *lang_cache = NULL; + /* Language string cache */ +static const char * const lang_encodings[] = + { /* Encoding strings */ + "us-ascii", "iso-8859-1", + "iso-8859-2", "iso-8859-3", + "iso-8859-4", "iso-8859-5", + "iso-8859-6", "iso-8859-7", + "iso-8859-8", "iso-8859-9", + "iso-8859-10", "utf-8", + "iso-8859-13", "iso-8859-14", + "iso-8859-15", "cp874", + "cp1250", "cp1251", + "cp1252", "cp1253", + "cp1254", "cp1255", + "cp1256", "cp1257", + "cp1258", "koi8-r", + "koi8-u", "iso-8859-11", + "iso-8859-16", "mac", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "cp932", "cp936", + "cp949", "cp950", + "cp1361", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "euc-cn", "euc-jp", + "euc-kr", "euc-tw", + "jis-x0213" + }; + +#ifdef __APPLE__ +typedef struct +{ + const char * const language; /* Language ID */ + const char * const locale; /* Locale ID */ +} _apple_language_locale_t; + +static const _apple_language_locale_t apple_language_locale[] = +{ /* Locale to language ID LUT */ + { "en", "en_US" }, + { "nb", "no" }, + { "zh-Hans", "zh_CN" }, + { "zh-Hant", "zh_TW" } +}; +#endif /* __APPLE__ */ + + +/* + * Local functions... + */ + + +#ifdef __APPLE__ +static const char *appleLangDefault(void); +# ifdef CUPS_BUNDLEDIR +# ifndef CF_RETURNS_RETAINED +# if __has_feature(attribute_cf_returns_retained) +# define CF_RETURNS_RETAINED __attribute__((cf_returns_retained)) +# else +# define CF_RETURNS_RETAINED +# endif /* __has_feature(attribute_cf_returns_retained) */ +# endif /* !CF_RETURNED_RETAINED */ +static cups_array_t *appleMessageLoad(const char *locale) + CF_RETURNS_RETAINED; +# endif /* CUPS_BUNDLEDIR */ +#endif /* __APPLE__ */ +static cups_lang_t *cups_cache_lookup(const char *name, + cups_encoding_t encoding); +static int cups_message_compare(_cups_message_t *m1, + _cups_message_t *m2); +static void cups_message_free(_cups_message_t *m); +static void cups_message_load(cups_lang_t *lang); +static void cups_unquote(char *d, const char *s); + + +#ifdef __APPLE__ +/* + * '_cupsAppleLanguage()' - Get the Apple language identifier associated with a + * locale ID. + */ + +const char * /* O - Language ID */ +_cupsAppleLanguage(const char *locale, /* I - Locale ID */ + char *language,/* I - Language ID buffer */ + size_t langsize) /* I - Size of language ID buffer */ +{ + int i; /* Looping var */ + CFStringRef localeid, /* CF locale identifier */ + langid; /* CF language identifier */ + + + /* + * Copy the locale name and convert, as needed, to the Apple-specific + * locale identifier... + */ + + switch (strlen(locale)) + { + default : + /* + * Invalid locale... + */ + + strlcpy(language, "en", langsize); + break; + + case 2 : + strlcpy(language, locale, langsize); + break; + + case 5 : + strlcpy(language, locale, langsize); + + if (language[2] == '-') + { + /* + * Convert ll-cc to ll_CC... + */ + + language[2] = '_'; + language[3] = toupper(language[3] & 255); + language[4] = toupper(language[4] & 255); + } + break; + } + + for (i = 0; + i < (int)(sizeof(apple_language_locale) / + sizeof(apple_language_locale[0])); + i ++) + if (!strcmp(locale, apple_language_locale[i].locale)) + { + strlcpy(language, apple_language_locale[i].language, sizeof(language)); + break; + } + + /* + * Attempt to map the locale ID to a language ID... + */ + + if ((localeid = CFStringCreateWithCString(kCFAllocatorDefault, language, + kCFStringEncodingASCII)) != NULL) + { + if ((langid = CFLocaleCreateCanonicalLanguageIdentifierFromString( + kCFAllocatorDefault, localeid)) != NULL) + { + CFStringGetCString(langid, language, langsize, kCFStringEncodingASCII); + CFRelease(langid); + } + + CFRelease(localeid); + } + + /* + * Return what we got... + */ + + return (language); +} +#endif /* __APPLE__ */ + + +/* + * '_cupsEncodingName()' - Return the character encoding name string + * for the given encoding enumeration. + */ + +const char * /* O - Character encoding */ +_cupsEncodingName( + cups_encoding_t encoding) /* I - Encoding value */ +{ + if (encoding < 0 || + encoding >= (sizeof(lang_encodings) / sizeof(const char *))) + { + DEBUG_printf(("1_cupsEncodingName(encoding=%d) = out of range (\"%s\")", + encoding, lang_encodings[0])); + return (lang_encodings[0]); + } + else + { + DEBUG_printf(("1_cupsEncodingName(encoding=%d) = \"%s\"", + encoding, lang_encodings[encoding])); + return (lang_encodings[encoding]); + } +} + + +/* + * 'cupsLangDefault()' - Return the default language. + */ + +cups_lang_t * /* O - Language data */ +cupsLangDefault(void) +{ + return (cupsLangGet(NULL)); +} + + +/* + * 'cupsLangEncoding()' - Return the character encoding (us-ascii, etc.) + * for the given language. + */ + +const char * /* O - Character encoding */ +cupsLangEncoding(cups_lang_t *lang) /* I - Language data */ +{ + if (lang == NULL) + return ((char*)lang_encodings[0]); + else + return ((char*)lang_encodings[lang->encoding]); +} + + +/* + * 'cupsLangFlush()' - Flush all language data out of the cache. + */ + +void +cupsLangFlush(void) +{ + cups_lang_t *lang, /* Current language */ + *next; /* Next language */ + + + /* + * Free all languages in the cache... + */ + + _cupsMutexLock(&lang_mutex); + + for (lang = lang_cache; lang != NULL; lang = next) + { + /* + * Free all messages... + */ + + _cupsMessageFree(lang->strings); + + /* + * Then free the language structure itself... + */ + + next = lang->next; + free(lang); + } + + lang_cache = NULL; + + _cupsMutexUnlock(&lang_mutex); +} + + +/* + * 'cupsLangFree()' - Free language data. + * + * This does not actually free anything; use @link cupsLangFlush@ for that. + */ + +void +cupsLangFree(cups_lang_t *lang) /* I - Language to free */ +{ + _cupsMutexLock(&lang_mutex); + + if (lang != NULL && lang->used > 0) + lang->used --; + + _cupsMutexUnlock(&lang_mutex); +} + + +/* + * 'cupsLangGet()' - Get a language. + */ + +cups_lang_t * /* O - Language data */ +cupsLangGet(const char *language) /* I - Language or locale */ +{ + int i; /* Looping var */ +#ifndef __APPLE__ + char locale[255]; /* Copy of locale name */ +#endif /* !__APPLE__ */ + char langname[16], /* Requested language name */ + country[16], /* Country code */ + charset[16], /* Character set */ + *csptr, /* Pointer to CODESET string */ + *ptr, /* Pointer into language/charset */ + real[48]; /* Real language name */ + cups_encoding_t encoding; /* Encoding to use */ + cups_lang_t *lang; /* Current language... */ + static const char * const locale_encodings[] = + { /* Locale charset names */ + "ASCII", "ISO88591", "ISO88592", "ISO88593", + "ISO88594", "ISO88595", "ISO88596", "ISO88597", + "ISO88598", "ISO88599", "ISO885910", "UTF8", + "ISO885913", "ISO885914", "ISO885915", "CP874", + "CP1250", "CP1251", "CP1252", "CP1253", + "CP1254", "CP1255", "CP1256", "CP1257", + "CP1258", "KOI8R", "KOI8U", "ISO885911", + "ISO885916", "MACROMAN", "", "", + + "", "", "", "", + "", "", "", "", + "", "", "", "", + "", "", "", "", + "", "", "", "", + "", "", "", "", + "", "", "", "", + "", "", "", "", + + "CP932", "CP936", "CP949", "CP950", + "CP1361", "", "", "", + "", "", "", "", + "", "", "", "", + "", "", "", "", + "", "", "", "", + "", "", "", "", + "", "", "", "", + + "", "", "", "", + "", "", "", "", + "", "", "", "", + "", "", "", "", + "", "", "", "", + "", "", "", "", + "", "", "", "", + "", "", "", "", + + "EUCCN", "EUCJP", "EUCKR", "EUCTW", + "SHIFT_JISX0213" + }; + + + DEBUG_printf(("2cupsLangGet(language=\"%s\")", language)); + +#ifdef __APPLE__ + /* + * Set the character set to UTF-8... + */ + + strcpy(charset, "UTF8"); + + /* + * Apple's setlocale doesn't give us the user's localization + * preference so we have to look it up this way... + */ + + if (!language) + { + if (!getenv("SOFTWARE") || (language = getenv("LANG")) == NULL) + language = appleLangDefault(); + + DEBUG_printf(("4cupsLangGet: language=\"%s\"", language)); + } + +#else + /* + * Set the charset to "unknown"... + */ + + charset[0] = '\0'; + + /* + * Use setlocale() to determine the currently set locale, and then + * fallback to environment variables to avoid setting the locale, + * since setlocale() is not thread-safe! + */ + + if (!language) + { + /* + * First see if the locale has been set; if it is still "C" or + * "POSIX", use the environment to get the default... + */ + +# ifdef LC_MESSAGES + ptr = setlocale(LC_MESSAGES, NULL); +# else + ptr = setlocale(LC_ALL, NULL); +# endif /* LC_MESSAGES */ + + DEBUG_printf(("4cupsLangGet: current locale is \"%s\"", ptr)); + + if (!ptr || !strcmp(ptr, "C") || !strcmp(ptr, "POSIX")) + { + /* + * Get the character set from the LC_CTYPE locale setting... + */ + + if ((ptr = getenv("LC_CTYPE")) == NULL) + if ((ptr = getenv("LC_ALL")) == NULL) + if ((ptr = getenv("LANG")) == NULL) + ptr = "en_US"; + + if ((csptr = strchr(ptr, '.')) != NULL) + { + /* + * Extract the character set from the environment... + */ + + for (ptr = charset, csptr ++; *csptr; csptr ++) + if (ptr < (charset + sizeof(charset) - 1) && _cups_isalnum(*csptr)) + *ptr++ = *csptr; + + *ptr = '\0'; + } + + /* + * Get the locale for messages from the LC_MESSAGES locale setting... + */ + + if ((ptr = getenv("LC_MESSAGES")) == NULL) + if ((ptr = getenv("LC_ALL")) == NULL) + if ((ptr = getenv("LANG")) == NULL) + ptr = "en_US"; + } + + if (ptr) + { + strlcpy(locale, ptr, sizeof(locale)); + language = locale; + + /* + * CUPS STR #2575: Map "nb" to "no" for back-compatibility... + */ + + if (!strncmp(locale, "nb", 2)) + locale[1] = 'o'; + + DEBUG_printf(("4cupsLangGet: new language value is \"%s\"", language)); + } + } +#endif /* __APPLE__ */ + + /* + * If "language" is NULL at this point, then chances are we are using + * a language that is not installed for the base OS. + */ + + if (!language) + { + /* + * Switch to the POSIX ("C") locale... + */ + + language = "C"; + } + +#ifdef CODESET + /* + * On systems that support the nl_langinfo(CODESET) call, use + * this value as the character set... + */ + + if (!charset[0] && (csptr = nl_langinfo(CODESET)) != NULL) + { + /* + * Copy all of the letters and numbers in the CODESET string... + */ + + for (ptr = charset; *csptr; csptr ++) + if (_cups_isalnum(*csptr) && ptr < (charset + sizeof(charset) - 1)) + *ptr++ = *csptr; + + *ptr = '\0'; + + DEBUG_printf(("4cupsLangGet: charset set to \"%s\" via " + "nl_langinfo(CODESET)...", charset)); + } +#endif /* CODESET */ + + /* + * If we don't have a character set by now, default to UTF-8... + */ + + if (!charset[0]) + strcpy(charset, "UTF8"); + + /* + * Parse the language string passed in to a locale string. "C" is the + * standard POSIX locale and is copied unchanged. Otherwise the + * language string is converted from ll-cc[.charset] (language-country) + * to ll_CC[.CHARSET] to match the file naming convention used by all + * POSIX-compliant operating systems. Invalid language names are mapped + * to the POSIX locale. + */ + + country[0] = '\0'; + + if (language == NULL || !language[0] || + !strcmp(language, "POSIX")) + strcpy(langname, "C"); + else + { + /* + * Copy the parts of the locale string over safely... + */ + + for (ptr = langname; *language; language ++) + if (*language == '_' || *language == '-' || *language == '.') + break; + else if (ptr < (langname + sizeof(langname) - 1)) + *ptr++ = tolower(*language & 255); + + *ptr = '\0'; + + if (*language == '_' || *language == '-') + { + /* + * Copy the country code... + */ + + for (language ++, ptr = country; *language; language ++) + if (*language == '.') + break; + else if (ptr < (country + sizeof(country) - 1)) + *ptr++ = toupper(*language & 255); + + *ptr = '\0'; + } + + if (*language == '.' && !charset[0]) + { + /* + * Copy the encoding... + */ + + for (language ++, ptr = charset; *language; language ++) + if (_cups_isalnum(*language) && ptr < (charset + sizeof(charset) - 1)) + *ptr++ = toupper(*language & 255); + + *ptr = '\0'; + } + + /* + * Force a POSIX locale for an invalid language name... + */ + + if (strlen(langname) != 2) + { + strcpy(langname, "C"); + country[0] = '\0'; + charset[0] = '\0'; + } + } + + DEBUG_printf(("4cupsLangGet: langname=\"%s\", country=\"%s\", charset=\"%s\"", + langname, country, charset)); + + /* + * Figure out the desired encoding... + */ + + encoding = CUPS_AUTO_ENCODING; + + if (charset[0]) + { + for (i = 0; + i < (int)(sizeof(locale_encodings) / sizeof(locale_encodings[0])); + i ++) + if (!_cups_strcasecmp(charset, locale_encodings[i])) + { + encoding = (cups_encoding_t)i; + break; + } + + if (encoding == CUPS_AUTO_ENCODING) + { + /* + * Map alternate names for various character sets... + */ + + if (!_cups_strcasecmp(charset, "iso-2022-jp") || + !_cups_strcasecmp(charset, "sjis")) + encoding = CUPS_WINDOWS_932; + else if (!_cups_strcasecmp(charset, "iso-2022-cn")) + encoding = CUPS_WINDOWS_936; + else if (!_cups_strcasecmp(charset, "iso-2022-kr")) + encoding = CUPS_WINDOWS_949; + else if (!_cups_strcasecmp(charset, "big5")) + encoding = CUPS_WINDOWS_950; + } + } + + DEBUG_printf(("4cupsLangGet: encoding=%d(%s)", encoding, + encoding == CUPS_AUTO_ENCODING ? "auto" : + lang_encodings[encoding])); + + /* + * See if we already have this language/country loaded... + */ + + if (country[0]) + snprintf(real, sizeof(real), "%s_%s", langname, country); + else + strcpy(real, langname); + + _cupsMutexLock(&lang_mutex); + + if ((lang = cups_cache_lookup(real, encoding)) != NULL) + { + _cupsMutexUnlock(&lang_mutex); + + DEBUG_printf(("3cupsLangGet: Using cached copy of \"%s\"...", real)); + + return (lang); + } + + /* + * See if there is a free language available; if so, use that + * record... + */ + + for (lang = lang_cache; lang != NULL; lang = lang->next) + if (lang->used == 0) + break; + + if (lang == NULL) + { + /* + * Allocate memory for the language and add it to the cache. + */ + + if ((lang = calloc(sizeof(cups_lang_t), 1)) == NULL) + { + _cupsMutexUnlock(&lang_mutex); + + return (NULL); + } + + lang->next = lang_cache; + lang_cache = lang; + } + else + { + /* + * Free all old strings as needed... + */ + + _cupsMessageFree(lang->strings); + lang->strings = NULL; + } + + /* + * Then assign the language and encoding fields... + */ + + lang->used ++; + strlcpy(lang->language, real, sizeof(lang->language)); + + if (encoding != CUPS_AUTO_ENCODING) + lang->encoding = encoding; + else + lang->encoding = CUPS_UTF8; + + /* + * Return... + */ + + _cupsMutexUnlock(&lang_mutex); + + return (lang); +} + + +/* + * '_cupsLangString()' - Get a message string. + * + * The returned string is UTF-8 encoded; use cupsUTF8ToCharset() to + * convert the string to the language encoding. + */ + +const char * /* O - Localized message */ +_cupsLangString(cups_lang_t *lang, /* I - Language */ + const char *message) /* I - Message */ +{ + const char *s; /* Localized message */ + + /* + * Range check input... + */ + + if (!lang || !message) + return (message); + + _cupsMutexLock(&lang_mutex); + + /* + * Load the message catalog if needed... + */ + + if (!lang->strings) + cups_message_load(lang); + + s = _cupsMessageLookup(lang->strings, message); + + _cupsMutexUnlock(&lang_mutex); + + return (s); +} + + +/* + * '_cupsMessageFree()' - Free a messages array. + */ + +void +_cupsMessageFree(cups_array_t *a) /* I - Message array */ +{ +#if defined(__APPLE__) && defined(CUPS_BUNDLEDIR) + /* + * Release the cups.strings dictionary as needed... + */ + + if (cupsArrayUserData(a)) + CFRelease((CFDictionaryRef)cupsArrayUserData(a)); +#endif /* __APPLE__ && CUPS_BUNDLEDIR */ + + /* + * Free the array... + */ + + cupsArrayDelete(a); +} + + +/* + * '_cupsMessageLoad()' - Load a .po file into a messages array. + */ + +cups_array_t * /* O - New message array */ +_cupsMessageLoad(const char *filename, /* I - Message catalog to load */ + int unquote) /* I - Unescape \foo in strings? */ +{ + cups_file_t *fp; /* Message file */ + cups_array_t *a; /* Message array */ + _cups_message_t *m; /* Current message */ + char s[4096], /* String buffer */ + *ptr, /* Pointer into buffer */ + *temp; /* New string */ + int length; /* Length of combined strings */ + + + DEBUG_printf(("4_cupsMessageLoad(filename=\"%s\")", filename)); + + /* + * Create an array to hold the messages... + */ + + if ((a = cupsArrayNew3((cups_array_func_t)cups_message_compare, NULL, + (cups_ahash_func_t)NULL, 0, + (cups_acopy_func_t)NULL, + (cups_afree_func_t)cups_message_free)) == NULL) + { + DEBUG_puts("5_cupsMessageLoad: Unable to allocate array!"); + return (NULL); + } + + /* + * Open the message catalog file... + */ + + if ((fp = cupsFileOpen(filename, "r")) == NULL) + { + DEBUG_printf(("5_cupsMessageLoad: Unable to open file: %s", + strerror(errno))); + return (a); + } + + /* + * Read messages from the catalog file until EOF... + * + * The format is the GNU gettext .po format, which is fairly simple: + * + * msgid "some text" + * msgstr "localized text" + * + * The ID and localized text can span multiple lines using the form: + * + * msgid "" + * "some long text" + * msgstr "" + * "localized text spanning " + * "multiple lines" + */ + + m = NULL; + + while (cupsFileGets(fp, s, sizeof(s)) != NULL) + { + /* + * Skip blank and comment lines... + */ + + if (s[0] == '#' || !s[0]) + continue; + + /* + * Strip the trailing quote... + */ + + if ((ptr = strrchr(s, '\"')) == NULL) + continue; + + *ptr = '\0'; + + /* + * Find start of value... + */ + + if ((ptr = strchr(s, '\"')) == NULL) + continue; + + ptr ++; + + /* + * Unquote the text... + */ + + if (unquote) + cups_unquote(ptr, ptr); + + /* + * Create or add to a message... + */ + + if (!strncmp(s, "msgid", 5)) + { + /* + * Add previous message as needed... + */ + + if (m) + cupsArrayAdd(a, m); + + /* + * Create a new message with the given msgid string... + */ + + if ((m = (_cups_message_t *)calloc(1, sizeof(_cups_message_t))) == NULL) + { + cupsFileClose(fp); + return (a); + } + + if ((m->id = strdup(ptr)) == NULL) + { + free(m); + cupsFileClose(fp); + return (a); + } + } + else if (s[0] == '\"' && m) + { + /* + * Append to current string... + */ + + length = (int)strlen(m->str ? m->str : m->id); + + if ((temp = realloc(m->str ? m->str : m->id, + length + strlen(ptr) + 1)) == NULL) + { + cupsFileClose(fp); + return (a); + } + + if (m->str) + { + /* + * Copy the new portion to the end of the msgstr string - safe + * to use strcpy because the buffer is allocated to the correct + * size... + */ + + m->str = temp; + + strcpy(m->str + length, ptr); + } + else + { + /* + * Copy the new portion to the end of the msgid string - safe + * to use strcpy because the buffer is allocated to the correct + * size... + */ + + m->id = temp; + + strcpy(m->id + length, ptr); + } + } + else if (!strncmp(s, "msgstr", 6) && m) + { + /* + * Set the string... + */ + + if ((m->str = strdup(ptr)) == NULL) + { + cupsFileClose(fp); + return (a); + } + } + } + + /* + * Add the last message string to the array as needed... + */ + + if (m) + cupsArrayAdd(a, m); + + /* + * Close the message catalog file and return the new array... + */ + + cupsFileClose(fp); + + DEBUG_printf(("5_cupsMessageLoad: Returning %d messages...", + cupsArrayCount(a))); + + return (a); +} + + +/* + * '_cupsMessageLookup()' - Lookup a message string. + */ + +const char * /* O - Localized message */ +_cupsMessageLookup(cups_array_t *a, /* I - Message array */ + const char *m) /* I - Message */ +{ + _cups_message_t key, /* Search key */ + *match; /* Matching message */ + + + /* + * Lookup the message string; if it doesn't exist in the catalog, + * then return the message that was passed to us... + */ + + key.id = (char *)m; + match = (_cups_message_t *)cupsArrayFind(a, &key); + +#if defined(__APPLE__) && defined(CUPS_BUNDLEDIR) + if (!match && cupsArrayUserData(a)) + { + /* + * Try looking the string up in the cups.strings dictionary... + */ + + CFDictionaryRef dict; /* cups.strings dictionary */ + CFStringRef cfm, /* Message as a CF string */ + cfstr; /* Localized text as a CF string */ + + dict = (CFDictionaryRef)cupsArrayUserData(a); + cfm = CFStringCreateWithCString(kCFAllocatorDefault, m, + kCFStringEncodingUTF8); + match = calloc(1, sizeof(_cups_message_t)); + match->id = strdup(m); + cfstr = cfm ? CFDictionaryGetValue(dict, cfm) : NULL; + + if (cfstr) + { + char buffer[1024]; /* Message buffer */ + + CFStringGetCString(cfstr, buffer, sizeof(buffer), kCFStringEncodingUTF8); + match->str = strdup(buffer); + + DEBUG_printf(("1_cupsMessageLookup: Found \"%s\" as \"%s\"...", + m, buffer)); + } + else + { + match->str = strdup(m); + + DEBUG_printf(("1_cupsMessageLookup: Did not find \"%s\"...", m)); + } + + cupsArrayAdd(a, match); + + if (cfm) + CFRelease(cfm); + } +#endif /* __APPLE__ && CUPS_BUNDLEDIR */ + + if (match && match->str) + return (match->str); + else + return (m); +} + + +#ifdef __APPLE__ +/* + * 'appleLangDefault()' - Get the default locale string. + */ + +static const char * /* O - Locale string */ +appleLangDefault(void) +{ + int i; /* Looping var */ + CFBundleRef bundle; /* Main bundle (if any) */ + CFArrayRef bundleList; /* List of localizations in bundle */ + CFPropertyListRef localizationList; + /* List of localization data */ + CFStringRef languageName; /* Current name */ + CFStringRef localeName; /* Canonical from of name */ + char *lang; /* LANG environment variable */ + _cups_globals_t *cg = _cupsGlobals(); + /* Pointer to library globals */ + + + DEBUG_puts("2appleLangDefault()"); + + /* + * Only do the lookup and translation the first time. + */ + + if (!cg->language[0]) + { + if (getenv("SOFTWARE") != NULL && (lang = getenv("LANG")) != NULL) + { + strlcpy(cg->language, lang, sizeof(cg->language)); + return (cg->language); + } + else if ((bundle = CFBundleGetMainBundle()) != NULL && + (bundleList = CFBundleCopyBundleLocalizations(bundle)) != NULL) + { + localizationList = + CFBundleCopyPreferredLocalizationsFromArray(bundleList); + + CFRelease(bundleList); + } + else + localizationList = + CFPreferencesCopyAppValue(CFSTR("AppleLanguages"), + kCFPreferencesCurrentApplication); + + if (localizationList) + { + if (CFGetTypeID(localizationList) == CFArrayGetTypeID() && + CFArrayGetCount(localizationList) > 0) + { + languageName = CFArrayGetValueAtIndex(localizationList, 0); + + if (languageName && + CFGetTypeID(languageName) == CFStringGetTypeID()) + { + localeName = CFLocaleCreateCanonicalLocaleIdentifierFromString( + kCFAllocatorDefault, languageName); + + if (localeName) + { + CFStringGetCString(localeName, cg->language, sizeof(cg->language), + kCFStringEncodingASCII); + CFRelease(localeName); + + DEBUG_printf(("9appleLangDefault: cg->language=\"%s\"", + cg->language)); + + /* + * Map new language identifiers to locales... + */ + + for (i = 0; + i < (int)(sizeof(apple_language_locale) / + sizeof(apple_language_locale[0])); + i ++) + { + if (!strcmp(cg->language, apple_language_locale[i].language)) + { + DEBUG_printf(("9appleLangDefault: mapping \"%s\" to \"%s\"...", + cg->language, apple_language_locale[i].locale)); + strlcpy(cg->language, apple_language_locale[i].locale, + sizeof(cg->language)); + break; + } + } + + /* + * Convert language subtag into region subtag... + */ + + if (cg->language[2] == '-') + cg->language[2] = '_'; + + if (!strchr(cg->language, '.')) + strlcat(cg->language, ".UTF-8", sizeof(cg->language)); + } + } + } + + CFRelease(localizationList); + } + + /* + * If we didn't find the language, default to en_US... + */ + + if (!cg->language[0]) + strlcpy(cg->language, "en_US.UTF-8", sizeof(cg->language)); + } + + /* + * Return the cached locale... + */ + + return (cg->language); +} + + +# ifdef CUPS_BUNDLEDIR +/* + * 'appleMessageLoad()' - Load a message catalog from a localizable bundle. + */ + +static cups_array_t * /* O - Message catalog */ +appleMessageLoad(const char *locale) /* I - Locale ID */ +{ + char filename[1024], /* Path to cups.strings file */ + applelang[256]; /* Apple language ID */ + CFURLRef url; /* URL to cups.strings file */ + CFReadStreamRef stream = NULL; /* File stream */ + CFPropertyListRef plist = NULL; /* Localization file */ +#ifdef DEBUG + CFErrorRef error = NULL; /* Error when opening file */ +#endif /* DEBUG */ + + + DEBUG_printf(("appleMessageLoad(locale=\"%s\")", locale)); + + /* + * Load the cups.strings file... + */ + + snprintf(filename, sizeof(filename), + CUPS_BUNDLEDIR "/Resources/%s.lproj/cups.strings", + _cupsAppleLanguage(locale, applelang, sizeof(applelang))); + DEBUG_printf(("1appleMessageLoad: filename=\"%s\"", filename)); + + if (access(filename, 0)) + { + /* + * Try alternate lproj directory names... + */ + + if (!strncmp(locale, "en", 2)) + locale = "English"; + else if (!strncmp(locale, "nb", 2) || !strncmp(locale, "nl", 2)) + locale = "Dutch"; + else if (!strncmp(locale, "fr", 2)) + locale = "French"; + else if (!strncmp(locale, "de", 2)) + locale = "German"; + else if (!strncmp(locale, "it", 2)) + locale = "Italian"; + else if (!strncmp(locale, "ja", 2)) + locale = "Japanese"; + else if (!strncmp(locale, "es", 2)) + locale = "Spanish"; + + snprintf(filename, sizeof(filename), + CUPS_BUNDLEDIR "/Resources/%s.lproj/cups.strings", locale); + DEBUG_printf(("1appleMessageLoad: alternate filename=\"%s\"", filename)); + } + + url = CFURLCreateFromFileSystemRepresentation(kCFAllocatorDefault, + (UInt8 *)filename, + strlen(filename), false); + if (url) + { + stream = CFReadStreamCreateWithFile(kCFAllocatorDefault, url); + if (stream) + { + /* + * Read the property list containing the localization data. + * + * NOTE: This code currently generates a clang "potential leak" + * warning, but the object is released in _cupsMessageFree(). + */ + + CFReadStreamOpen(stream); + +#ifdef DEBUG + plist = CFPropertyListCreateWithStream(kCFAllocatorDefault, stream, 0, + kCFPropertyListImmutable, NULL, + &error); + if (error) + { + CFStringRef msg = CFErrorCopyDescription(error); + /* Error message */ + + CFStringGetCString(msg, filename, sizeof(filename), + kCFStringEncodingUTF8); + DEBUG_printf(("1appleMessageLoad: %s", filename)); + + CFRelease(msg); + CFRelease(error); + } + +#else + plist = CFPropertyListCreateWithStream(kCFAllocatorDefault, stream, 0, + kCFPropertyListImmutable, NULL, + NULL); +#endif /* DEBUG */ + + if (plist && CFGetTypeID(plist) != CFDictionaryGetTypeID()) + { + CFRelease(plist); + plist = NULL; + } + + CFRelease(stream); + } + + CFRelease(url); + } + + DEBUG_printf(("1appleMessageLoad: url=%p, stream=%p, plist=%p", url, stream, + plist)); + + /* + * Create and return an empty array to act as a cache for messages, passing the + * plist as the user data. + */ + + return (cupsArrayNew3((cups_array_func_t)cups_message_compare, (void *)plist, + (cups_ahash_func_t)NULL, 0, + (cups_acopy_func_t)NULL, + (cups_afree_func_t)cups_message_free)); +} +# endif /* CUPS_BUNDLEDIR */ +#endif /* __APPLE__ */ + + +/* + * 'cups_cache_lookup()' - Lookup a language in the cache... + */ + +static cups_lang_t * /* O - Language data or NULL */ +cups_cache_lookup( + const char *name, /* I - Name of locale */ + cups_encoding_t encoding) /* I - Encoding of locale */ +{ + cups_lang_t *lang; /* Current language */ + + + DEBUG_printf(("7cups_cache_lookup(name=\"%s\", encoding=%d(%s))", name, + encoding, encoding == CUPS_AUTO_ENCODING ? "auto" : + lang_encodings[encoding])); + + /* + * Loop through the cache and return a match if found... + */ + + for (lang = lang_cache; lang != NULL; lang = lang->next) + { + DEBUG_printf(("9cups_cache_lookup: lang=%p, language=\"%s\", " + "encoding=%d(%s)", lang, lang->language, lang->encoding, + lang_encodings[lang->encoding])); + + if (!strcmp(lang->language, name) && + (encoding == CUPS_AUTO_ENCODING || encoding == lang->encoding)) + { + lang->used ++; + + DEBUG_puts("8cups_cache_lookup: returning match!"); + + return (lang); + } + } + + DEBUG_puts("8cups_cache_lookup: returning NULL!"); + + return (NULL); +} + + +/* + * 'cups_message_compare()' - Compare two messages. + */ + +static int /* O - Result of comparison */ +cups_message_compare( + _cups_message_t *m1, /* I - First message */ + _cups_message_t *m2) /* I - Second message */ +{ + return (strcmp(m1->id, m2->id)); +} + + +/* + * 'cups_message_free()' - Free a message. + */ + +static void +cups_message_free(_cups_message_t *m) /* I - Message */ +{ + if (m->id) + free(m->id); + + if (m->str) + free(m->str); + + free(m); +} + + +/* + * 'cups_message_load()' - Load the message catalog for a language. + */ + +static void +cups_message_load(cups_lang_t *lang) /* I - Language */ +{ +#if defined(__APPLE__) && defined(CUPS_BUNDLEDIR) + lang->strings = appleMessageLoad(lang->language); + +#else + char filename[1024]; /* Filename for language locale file */ + _cups_globals_t *cg = _cupsGlobals(); + /* Pointer to library globals */ + + + snprintf(filename, sizeof(filename), "%s/%s/cups_%s.po", cg->localedir, + lang->language, lang->language); + + if (strchr(lang->language, '_') && access(filename, 0)) + { + /* + * Country localization not available, look for generic localization... + */ + + snprintf(filename, sizeof(filename), "%s/%.2s/cups_%.2s.po", cg->localedir, + lang->language, lang->language); + + if (access(filename, 0)) + { + /* + * No generic localization, so use POSIX... + */ + + DEBUG_printf(("4cups_message_load: access(\"%s\", 0): %s", filename, + strerror(errno))); + + snprintf(filename, sizeof(filename), "%s/C/cups_C.po", cg->localedir); + } + } + + /* + * Read the strings from the file... + */ + + lang->strings = _cupsMessageLoad(filename, 1); +#endif /* __APPLE__ && CUPS_BUNDLEDIR */ +} + + +/* + * 'cups_unquote()' - Unquote characters in strings... + */ + +static void +cups_unquote(char *d, /* O - Unquoted string */ + const char *s) /* I - Original string */ +{ + while (*s) + { + if (*s == '\\') + { + s ++; + if (isdigit(*s)) + { + *d = 0; + + while (isdigit(*s)) + { + *d = *d * 8 + *s - '0'; + s ++; + } + + d ++; + } + else + { + if (*s == 'n') + *d ++ = '\n'; + else if (*s == 'r') + *d ++ = '\r'; + else if (*s == 't') + *d ++ = '\t'; + else + *d++ = *s; + + s ++; + } + } + else + *d++ = *s++; + } + + *d = '\0'; +} + + +/* + * End of "$Id$". + */ diff --git a/cups/language.h b/cups/language.h new file mode 100644 index 0000000000..0b8acdee42 --- /dev/null +++ b/cups/language.h @@ -0,0 +1,115 @@ +/* + * "$Id$" + * + * Multi-language support for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_LANGUAGE_H_ +# define _CUPS_LANGUAGE_H_ + +/* + * Include necessary headers... + */ + +# include +# include "array.h" + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Types... + */ + +typedef enum cups_encoding_e /**** Language Encodings ****/ +{ + CUPS_AUTO_ENCODING = -1, /* Auto-detect the encoding @private@ */ + CUPS_US_ASCII, /* US ASCII */ + CUPS_ISO8859_1, /* ISO-8859-1 */ + CUPS_ISO8859_2, /* ISO-8859-2 */ + CUPS_ISO8859_3, /* ISO-8859-3 */ + CUPS_ISO8859_4, /* ISO-8859-4 */ + CUPS_ISO8859_5, /* ISO-8859-5 */ + CUPS_ISO8859_6, /* ISO-8859-6 */ + CUPS_ISO8859_7, /* ISO-8859-7 */ + CUPS_ISO8859_8, /* ISO-8859-8 */ + CUPS_ISO8859_9, /* ISO-8859-9 */ + CUPS_ISO8859_10, /* ISO-8859-10 */ + CUPS_UTF8, /* UTF-8 */ + CUPS_ISO8859_13, /* ISO-8859-13 */ + CUPS_ISO8859_14, /* ISO-8859-14 */ + CUPS_ISO8859_15, /* ISO-8859-15 */ + CUPS_WINDOWS_874, /* CP-874 */ + CUPS_WINDOWS_1250, /* CP-1250 */ + CUPS_WINDOWS_1251, /* CP-1251 */ + CUPS_WINDOWS_1252, /* CP-1252 */ + CUPS_WINDOWS_1253, /* CP-1253 */ + CUPS_WINDOWS_1254, /* CP-1254 */ + CUPS_WINDOWS_1255, /* CP-1255 */ + CUPS_WINDOWS_1256, /* CP-1256 */ + CUPS_WINDOWS_1257, /* CP-1257 */ + CUPS_WINDOWS_1258, /* CP-1258 */ + CUPS_KOI8_R, /* KOI-8-R */ + CUPS_KOI8_U, /* KOI-8-U */ + CUPS_ISO8859_11, /* ISO-8859-11 */ + CUPS_ISO8859_16, /* ISO-8859-16 */ + CUPS_MAC_ROMAN, /* MacRoman */ + CUPS_ENCODING_SBCS_END = 63, /* End of single-byte encodings @private@ */ + + CUPS_WINDOWS_932, /* Japanese JIS X0208-1990 */ + CUPS_WINDOWS_936, /* Simplified Chinese GB 2312-80 */ + CUPS_WINDOWS_949, /* Korean KS C5601-1992 */ + CUPS_WINDOWS_950, /* Traditional Chinese Big Five */ + CUPS_WINDOWS_1361, /* Korean Johab */ + CUPS_ENCODING_DBCS_END = 127, /* End of double-byte encodings @private@ */ + + CUPS_EUC_CN, /* EUC Simplified Chinese */ + CUPS_EUC_JP, /* EUC Japanese */ + CUPS_EUC_KR, /* EUC Korean */ + CUPS_EUC_TW, /* EUC Traditional Chinese */ + CUPS_JIS_X0213, /* JIS X0213 aka Shift JIS */ + CUPS_ENCODING_VBCS_END = 191 /* End of variable-length encodings @private@ */ +} cups_encoding_t; + +typedef struct cups_lang_s /**** Language Cache Structure ****/ +{ + struct cups_lang_s *next; /* Next language in cache */ + int used; /* Number of times this entry has been used. */ + cups_encoding_t encoding; /* Text encoding */ + char language[16]; /* Language/locale name */ + cups_array_t *strings; /* Message strings @private@ */ +} cups_lang_t; + + +/* + * Prototypes... + */ + +extern cups_lang_t *cupsLangDefault(void); +extern const char *cupsLangEncoding(cups_lang_t *lang); +extern void cupsLangFlush(void); +extern void cupsLangFree(cups_lang_t *lang); +extern cups_lang_t *cupsLangGet(const char *language); + +# ifdef __cplusplus +} +# endif /* __cplusplus */ + +#endif /* !_CUPS_LANGUAGE_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/libcups2.def b/cups/libcups2.def new file mode 100644 index 0000000000..ef22daa4f3 --- /dev/null +++ b/cups/libcups2.def @@ -0,0 +1,382 @@ +LIBRARY libcups2 +VERSION 2.9 +EXPORTS +_cupsBufferGet +_cupsBufferRelease +_cupsGet1284Values +_cupsGetDests +_cupsGetPassword +_cupsGlobals +_cupsLangPrintError +_cupsLangPrintf +_cupsLangPuts +_cupsLangString +_cupsMD5Append +_cupsMD5Finish +_cupsMD5Init +_cupsMessageFree +_cupsMessageLoad +_cupsMessageLookup +_cupsMutexLock +_cupsMutexUnlock +_cupsNextDelay +_cupsSetError +_cupsSetLocale +_cupsStrAlloc +_cupsStrFlush +_cupsStrFormatd +_cupsStrFree +_cupsStrRetain +_cupsStrScand +_cupsStrStatistics +_cups_strcasecmp +_cups_strncasecmp +_cups_strcpy +_cups_strlcat +_cups_strlcpy +_httpAddrPort +_httpAddrSetPort +_httpAssembleUUID +_httpCreate +_httpEncodeURI +_httpPeek +_httpResolveURI +_httpWait +_ippFindOption +_ppdCacheCreateWithFile +_ppdCacheCreateWithPPD +_ppdCacheDestroy +_ppdCacheGetBin +_ppdCacheGetInputSlot +_ppdCacheGetMediaType +_ppdCacheGetOutputBin +_ppdCacheGetPageSize +_ppdCacheGetSize +_ppdCacheGetSource +_ppdCacheGetType +_ppdCacheWriteFile +_ppdFreeLanguages +_ppdGetEncoding +_ppdGetLanguages +_ppdHashName +_ppdLocalizedAttr +_ppdNormalizeMakeAndModel +_ppdOpen +_ppdOpenFile +_ppdParseOptions +_pwgGenerateSize +_pwgInitSize +_pwgMediaForLegacy +_pwgMediaForPPD +_pwgMediaForPWG +_pwgMediaForSize +_pwgMediaTypeForType +_pwgPageSizeForMedia +cupsAddDest +cupsAddOption +cupsAdminCreateWindowsPPD +cupsAdminExportSamba +cupsArrayAdd +cupsArrayClear +cupsArrayCount +cupsArrayCurrent +cupsArrayDelete +cupsArrayDup +cupsArrayFind +cupsArrayFirst +cupsArrayGetIndex +cupsArrayGetInsert +cupsArrayIndex +cupsArrayInsert +cupsArrayLast +cupsArrayNew +cupsArrayNew2 +cupsArrayNew3 +cupsArrayNext +cupsArrayPrev +cupsArrayRemove +cupsArrayRestore +cupsArraySave +cupsArrayUserData +cupsCancelJob +cupsCharsetToUTF8 +cupsDirClose +cupsDirOpen +cupsDirRead +cupsDirRewind +cupsDoAuthentication +cupsDoFileRequest +cupsDoIORequest +cupsDoRequest +cupsEncodeOptions +cupsEncodeOptions2 +cupsEncryption +cupsFileClose +cupsFileCompression +cupsFileEOF +cupsFileFind +cupsFileFlush +cupsFileGetChar +cupsFileGetConf +cupsFileGetLine +cupsFileGets +cupsFileLock +cupsFileNumber +cupsFileOpen +cupsFileOpenFd +cupsFilePeekChar +cupsFilePrintf +cupsFilePutChar +cupsFilePuts +cupsFileRead +cupsFileRewind +cupsFileSeek +cupsFileStderr +cupsFileStdin +cupsFileStdout +cupsFileTell +cupsFileUnlock +cupsFileWrite +cupsFreeDests +cupsFreeJobs +cupsFreeOptions +cupsGetClasses +cupsGetDefault +cupsGetDefault2 +cupsGetDest +cupsGetDests +cupsGetDests2 +cupsGetFd +cupsGetFile +cupsGetJobs +cupsGetJobs2 +cupsGetOption +cupsGetPPD +cupsGetPPD2 +cupsGetPassword +cupsGetPrinters +cupsGetResponse +cupsLangDefault +cupsLangEncoding +cupsLangFlush +cupsLangFree +cupsLangGet +cupsLastError +cupsLastErrorString +cupsMarkOptions +cupsNotifySubject +cupsNotifyText +cupsParseOptions +cupsPrintFile +cupsPrintFile2 +cupsPrintFiles +cupsPrintFiles2 +cupsPutFd +cupsPutFile +cupsRemoveOption +cupsResolveConflicts +cupsSendRequest +cupsServer +cupsSetClientCertCB +cupsSetCredentials +cupsSetDests +cupsSetDests2 +cupsSetEncryption +cupsSetPasswordCB +cupsSetServer +cupsSetServerCertCB +cupsSetUser +cupsTempFd +cupsTempFile +cupsTempFile2 +cupsUTF32ToUTF8 +cupsUTF8ToCharset +cupsUTF8ToUTF32 +cupsUser +cupsWriteRequestData +httpAddCredential +httpAddrAny +httpAddrConnect +httpAddrEqual +httpAddrFreeList +httpAddrGetList +httpAddrLength +httpAddrLocalhost +httpAddrLookup +httpAddrString +httpAssembleURI +httpAssembleURIf +httpBlocking +httpCheck +httpClearCookie +httpClearFields +httpClose +httpConnect +httpConnectEncrypt +httpCopyCredentials +httpDecode64 +httpDecode64_2 +httpDelete +httpEncode64 +httpEncode64_2 +httpEncryption +httpError +httpFlush +httpFlushWrite +httpFreeCredentials +httpGet +httpGetBlocking +httpGetCookie +httpGetDateString +httpGetDateString2 +httpGetDateTime +httpGetFd +httpGetField +httpGetHostByName +httpGetHostname +httpGetLength +httpGetLength2 +httpGetStatus +httpGetSubField +httpGetSubField2 +httpGets +httpHead +httpInitialize +httpMD5 +httpMD5Final +httpMD5String +httpOptions +httpPost +httpPrintf +httpPut +httpRead +httpRead2 +httpReconnect +httpSeparate +httpSeparate2 +httpSeparateURI +httpSetCookie +httpSetCredentials +httpSetExpect +httpSetField +httpSetLength +httpSetTimeout +httpStatus +httpTrace +httpUpdate +httpWait +httpWrite +httpWrite2 +ippAddBoolean +ippAddBooleans +ippAddCollection +ippAddCollections +ippAddDate +ippAddInteger +ippAddIntegers +ippAddOctetString +ippAddOutOfBand +ippAddRange +ippAddRanges +ippAddResolution +ippAddResolutions +ippAddSeparator +ippAddString +ippAddStrings +ippAttributeString +ippCopyAttribute +ippCopyAttributes +ippDateToTime +ippDelete +ippDeleteAttribute +ippDeleteValues +ippEnumString +ippEnumValue +ippErrorString +ippErrorValue +ippFindAttribute +ippFindNextAttribute +ippFirstAttribute +ippGetBoolean +ippGetCollection +ippGetCount +ippGetDate +ippGetGroupTag +ippGetInteger +ippGetName +ippGetOperation +ippGetRange +ippGetRequestId +ippGetResolution +ippGetState +ippGetStatusCode +ippGetString +ippGetValueTag +ippGetVersion +ippLength +ippNew +ippNewRequest +ippNextAttribute +ippOpString +ippOpValue +ippPort +ippRead +ippReadFile +ippReadIO +ippSetPort +ippSetBoolean +ippSetCollection +ippSetDate +ippSetGroupTag +ippSetInteger +ippSetName +ippSetOperation +ippSetRange +ippSetRequestId +ippSetResolution +ippSetState +ippSetStatusCode +ippSetString +ippSetValueTag +ippSetVersion +ippTagString +ippTagValue +ippTimeToDate +ippWrite +ippWriteFile +ippWriteIO +ppdClose +ppdCollect +ppdCollect2 +ppdConflicts +ppdEmit +ppdEmitAfterOrder +ppdEmitFd +ppdEmitJCL +ppdEmitJCLEnd +ppdEmitString +ppdErrorString +ppdFindAttr +ppdFindChoice +ppdFindCustomOption +ppdFindCustomParam +ppdFindMarkedChoice +ppdFindNextAttr +ppdFindOption +ppdFirstCustomParam +ppdFirstOption +ppdIsMarked +ppdLastError +ppdLocalize +ppdMarkDefaults +ppdMarkOption +ppdNextCustomParam +ppdNextOption +ppdOpen +ppdOpen2 +ppdOpenFd +ppdOpenFile +ppdPageLength +ppdPageSize +ppdPageWidth +ppdSetConformance diff --git a/cups/libcups_s.exp b/cups/libcups_s.exp new file mode 100644 index 0000000000..b8b22072d9 --- /dev/null +++ b/cups/libcups_s.exp @@ -0,0 +1,85 @@ +_cups_debug_fd +_cupsBufferGet +_cupsBufferRelease +_cupsGet1284Values +_cupsGetDests +_cupsGetPassword +_cupsGlobals +_cupsLangPrintError +_cupsLangPrintf +_cupsLangPuts +_cupsLangString +_cupsMD5Append +_cupsMD5Finish +_cupsMD5Init +_cupsMessageFree +_cupsMessageLoad +_cupsMessageLookup +_cupsNextDelay +_cupsSetError +_cupsSetLocale +_cupsSNMPClose +_cupsSNMPCopyOID +_cupsSNMPDefaultCommunity +_cupsSNMPIsOID +_cupsSNMPIsOIDPrefixed +_cupsSNMPOIDToString +_cupsSNMPOpen +_cupsSNMPRead +_cupsSNMPSetDebug +_cupsSNMPStringToOID +_cupsSNMPWalk +_cupsSNMPWrite +_cupsStrAlloc +_cupsStrFlush +_cupsStrFormatd +_cupsStrFree +_cupsStrRetain +_cupsStrScand +_cupsStrStatistics +_cups_getifaddrs +_cups_freeifaddrs +_cups_strcpy +_cups_strlcat +_cups_strlcpy +_httpAddrPort +_httpAddrSetPort +_httpAssembleUUID +_httpBIOMethods +_httpCreate +_httpEncodeURI +_httpPeek +_httpResolveURI +_httpSetTimeout +_httpWait +_ippFindOption +_ppdFreeLanguages +_ppdGetEncoding +_ppdGetLanguages +_ppdHashName +_ppdLocalizedAttr +_ppdNormalizeMakeAndModel +_ppdOpen +_ppdOpenFile +_ppdParseOptions +_pwgCreateWithFile +_pwgDestroy +_pwgWriteFile +_pwgGenerateSize +_pwgInitSize +_pwgMediaForLegacy +_pwgMediaForPPD +_pwgMediaForPWG +_pwgMediaForSize +_pwgCreateWithPPD +_pwgGetBin +_pwgGetInputSlot +_pwgGetMediaType +_pwgGetOutputBin +_pwgGetPageSize +_pwgGetSize +_pwgGetSource +_pwgGetType +_pwgInputSlotForSource +_pwgMediaTypeForType +_pwgPageSizeForMedia diff --git a/cups/localize.c b/cups/localize.c new file mode 100644 index 0000000000..d41d757150 --- /dev/null +++ b/cups/localize.c @@ -0,0 +1,779 @@ +/* + * "$Id$" + * + * PPD localization routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * PostScript is a trademark of Adobe Systems, Inc. + * + * This code and any derivative of it may be used and distributed + * freely under the terms of the GNU General Public License when + * used with GNU Ghostscript or its derivatives. Use of the code + * (or any derivative of it) with software other than GNU + * GhostScript (or its derivatives) is governed by the CUPS license + * agreement. + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * ppdLocalize() - Localize the PPD file to the current locale. + * ppdLocalizeAttr() - Localize an attribute. + * ppdLocalizeIPPReason() - Get the localized version of a cupsIPPReason + * attribute. + * ppdLocalizeMarkerName() - Get the localized version of a marker-names + * attribute value. + * _ppdFreeLanguages() - Free an array of languages from _ppdGetLanguages. + * _ppdGetLanguages() - Get an array of languages from a PPD file. + * _ppdHashName() - Generate a hash value for a device or profile + * name. + * _ppdLocalizedAttr() - Find a localized attribute. + * ppd_ll_CC() - Get the current locale names. + */ + +/* + * Include necessary headers. + */ + +#include "cups-private.h" +#include "ppd-private.h" + + +/* + * Local functions... + */ + +static cups_lang_t *ppd_ll_CC(char *ll_CC, int ll_CC_size); + + +/* + * 'ppdLocalize()' - Localize the PPD file to the current locale. + * + * All groups, options, and choices are localized, as are ICC profile + * descriptions, printer presets, and custom option parameters. Each + * localized string uses the UTF-8 character encoding. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 0 on success, -1 on error */ +ppdLocalize(ppd_file_t *ppd) /* I - PPD file */ +{ + int i, j, k; /* Looping vars */ + ppd_group_t *group; /* Current group */ + ppd_option_t *option; /* Current option */ + ppd_choice_t *choice; /* Current choice */ + ppd_coption_t *coption; /* Current custom option */ + ppd_cparam_t *cparam; /* Current custom parameter */ + ppd_attr_t *attr, /* Current attribute */ + *locattr; /* Localized attribute */ + char ckeyword[PPD_MAX_NAME], /* Custom keyword */ + ll_CC[6]; /* Language + country locale */ + + + /* + * Range check input... + */ + + DEBUG_printf(("ppdLocalize(ppd=%p)", ppd)); + + if (!ppd) + return (-1); + + /* + * Get the default language... + */ + + ppd_ll_CC(ll_CC, sizeof(ll_CC)); + + /* + * Now lookup all of the groups, options, choices, etc. + */ + + for (i = ppd->num_groups, group = ppd->groups; i > 0; i --, group ++) + { + if ((locattr = _ppdLocalizedAttr(ppd, "Translation", group->name, + ll_CC)) != NULL) + strlcpy(group->text, locattr->text, sizeof(group->text)); + + for (j = group->num_options, option = group->options; j > 0; j --, option ++) + { + if ((locattr = _ppdLocalizedAttr(ppd, "Translation", option->keyword, + ll_CC)) != NULL) + strlcpy(option->text, locattr->text, sizeof(option->text)); + + for (k = option->num_choices, choice = option->choices; + k > 0; + k --, choice ++) + { + if (strcmp(choice->choice, "Custom") || + !ppdFindCustomOption(ppd, option->keyword)) + locattr = _ppdLocalizedAttr(ppd, option->keyword, choice->choice, + ll_CC); + else + { + snprintf(ckeyword, sizeof(ckeyword), "Custom%s", option->keyword); + + locattr = _ppdLocalizedAttr(ppd, ckeyword, "True", ll_CC); + } + + if (locattr) + strlcpy(choice->text, locattr->text, sizeof(choice->text)); + } + } + } + + /* + * Translate any custom parameters... + */ + + for (coption = (ppd_coption_t *)cupsArrayFirst(ppd->coptions); + coption; + coption = (ppd_coption_t *)cupsArrayNext(ppd->coptions)) + { + for (cparam = (ppd_cparam_t *)cupsArrayFirst(coption->params); + cparam; + cparam = (ppd_cparam_t *)cupsArrayNext(coption->params)) + { + snprintf(ckeyword, sizeof(ckeyword), "ParamCustom%s", coption->keyword); + + if ((locattr = _ppdLocalizedAttr(ppd, ckeyword, cparam->name, + ll_CC)) != NULL) + strlcpy(cparam->text, locattr->text, sizeof(cparam->text)); + } + } + + /* + * Translate ICC profile names... + */ + + if ((attr = ppdFindAttr(ppd, "APCustomColorMatchingName", NULL)) != NULL) + { + if ((locattr = _ppdLocalizedAttr(ppd, "APCustomColorMatchingName", + attr->spec, ll_CC)) != NULL) + strlcpy(attr->text, locattr->text, sizeof(attr->text)); + } + + for (attr = ppdFindAttr(ppd, "cupsICCProfile", NULL); + attr; + attr = ppdFindNextAttr(ppd, "cupsICCProfile", NULL)) + { + cupsArraySave(ppd->sorted_attrs); + + if ((locattr = _ppdLocalizedAttr(ppd, "cupsICCProfile", attr->spec, + ll_CC)) != NULL) + strlcpy(attr->text, locattr->text, sizeof(attr->text)); + + cupsArrayRestore(ppd->sorted_attrs); + } + + /* + * Translate printer presets... + */ + + for (attr = ppdFindAttr(ppd, "APPrinterPreset", NULL); + attr; + attr = ppdFindNextAttr(ppd, "APPrinterPreset", NULL)) + { + cupsArraySave(ppd->sorted_attrs); + + if ((locattr = _ppdLocalizedAttr(ppd, "APPrinterPreset", attr->spec, + ll_CC)) != NULL) + strlcpy(attr->text, locattr->text, sizeof(attr->text)); + + cupsArrayRestore(ppd->sorted_attrs); + } + + return (0); +} + + +/* + * 'ppdLocalizeAttr()' - Localize an attribute. + * + * This function uses the current locale to find the localized attribute for + * the given main and option keywords. If no localized version of the + * attribute exists for the current locale, the unlocalized version is returned. + */ + +ppd_attr_t * /* O - Localized attribute or @code NULL@ if none exists */ +ppdLocalizeAttr(ppd_file_t *ppd, /* I - PPD file */ + const char *keyword, /* I - Main keyword */ + const char *spec) /* I - Option keyword or @code NULL@ for none */ +{ + ppd_attr_t *locattr; /* Localized attribute */ + char ll_CC[6]; /* Language + country locale */ + + + /* + * Get the default language... + */ + + ppd_ll_CC(ll_CC, sizeof(ll_CC)); + + /* + * Find the localized attribute... + */ + + if (spec) + locattr = _ppdLocalizedAttr(ppd, keyword, spec, ll_CC); + else + locattr = _ppdLocalizedAttr(ppd, "Translation", keyword, ll_CC); + + if (!locattr) + locattr = ppdFindAttr(ppd, keyword, spec); + + return (locattr); +} + + +/* + * 'ppdLocalizeIPPReason()' - Get the localized version of a cupsIPPReason + * attribute. + * + * This function uses the current locale to find the corresponding reason + * text or URI from the attribute value. If "scheme" is NULL or "text", + * the returned value contains human-readable (UTF-8) text from the translation + * string or attribute value. Otherwise the corresponding URI is returned. + * + * If no value of the requested scheme can be found, NULL is returned. + * + * @since CUPS 1.3/Mac OS X 10.5@ + */ + +const char * /* O - Value or NULL if not found */ +ppdLocalizeIPPReason( + ppd_file_t *ppd, /* I - PPD file */ + const char *reason, /* I - IPP reason keyword to look up */ + const char *scheme, /* I - URI scheme or NULL for text */ + char *buffer, /* I - Value buffer */ + size_t bufsize) /* I - Size of value buffer */ +{ + cups_lang_t *lang; /* Current language */ + ppd_attr_t *locattr; /* Localized attribute */ + char ll_CC[6], /* Language + country locale */ + *bufptr, /* Pointer into buffer */ + *bufend, /* Pointer to end of buffer */ + *valptr; /* Pointer into value */ + int ch, /* Hex-encoded character */ + schemelen; /* Length of scheme name */ + + + /* + * Range check input... + */ + + if (buffer) + *buffer = '\0'; + + if (!ppd || !reason || (scheme && !*scheme) || + !buffer || bufsize < PPD_MAX_TEXT) + return (NULL); + + /* + * Get the default language... + */ + + lang = ppd_ll_CC(ll_CC, sizeof(ll_CC)); + + /* + * Find the localized attribute... + */ + + if ((locattr = _ppdLocalizedAttr(ppd, "cupsIPPReason", reason, + ll_CC)) == NULL) + locattr = ppdFindAttr(ppd, "cupsIPPReason", reason); + + if (!locattr) + { + if (lang && (!scheme || !strcmp(scheme, "text"))) + { + /* + * Try to localize a standard printer-state-reason keyword... + */ + + const char *message = NULL; /* Localized message */ + + if (!strncmp(reason, "media-needed", 12)) + message = _("The paper tray needs to be filled."); + else if (!strncmp(reason, "media-jam", 9)) + message = _("There is a paper jam."); + else if (!strncmp(reason, "offline", 7) || + !strncmp(reason, "shutdown", 8)) + message = _("The printer is not connected."); + else if (!strncmp(reason, "toner-low", 9)) + message = _("The printer is low on toner."); + else if (!strncmp(reason, "toner-empty", 11)) + message = _("The printer is out of toner."); + else if (!strncmp(reason, "cover-open", 10)) + message = _("The printer's cover is open."); + else if (!strncmp(reason, "interlock-open", 14)) + message = _("The printer's interlock is open."); + else if (!strncmp(reason, "door-open", 9)) + message = _("The printer's door is open."); + else if (!strncmp(reason, "input-tray-missing", 18)) + message = _("The paper tray is missing."); + else if (!strncmp(reason, "media-low", 9)) + message = _("The paper tray is almost empty."); + else if (!strncmp(reason, "media-empty", 11)) + message = _("The paper tray is empty."); + else if (!strncmp(reason, "output-tray-missing", 19)) + message = _("The output bin is missing."); + else if (!strncmp(reason, "output-area-almost-full", 23)) + message = _("The output bin is almost full."); + else if (!strncmp(reason, "output-area-full", 16)) + message = _("The output bin is full."); + else if (!strncmp(reason, "marker-supply-low", 17)) + message = _("The printer is low on ink."); + else if (!strncmp(reason, "marker-supply-empty", 19)) + message = _("The printer may be out of ink."); + else if (!strncmp(reason, "marker-waste-almost-full", 24)) + message = _("The printer's waste bin is almost full."); + else if (!strncmp(reason, "marker-waste-full", 17)) + message = _("The printer's waste bin is full."); + else if (!strncmp(reason, "fuser-over-temp", 15)) + message = _("The fuser's temperature is high."); + else if (!strncmp(reason, "fuser-under-temp", 16)) + message = _("The fuser's temperature is low."); + else if (!strncmp(reason, "opc-near-eol", 12)) + message = _("The optical photoconductor will need to be replaced soon."); + else if (!strncmp(reason, "opc-life-over", 13)) + message = _("The optical photoconductor needs to be replaced."); + else if (!strncmp(reason, "developer-low", 13)) + message = _("The developer unit will need to be replaced soon."); + else if (!strncmp(reason, "developer-empty", 15)) + message = _("The developer unit needs to be replaced."); + + if (message) + { + strlcpy(buffer, _cupsLangString(lang, message), bufsize); + return (buffer); + } + } + + return (NULL); + } + + /* + * Now find the value we need... + */ + + bufend = buffer + bufsize - 1; + + if (!scheme || !strcmp(scheme, "text")) + { + /* + * Copy a text value (either the translation text or text:... URIs from + * the value... + */ + + strlcpy(buffer, locattr->text, bufsize); + + for (valptr = locattr->value, bufptr = buffer; *valptr && bufptr < bufend;) + { + if (!strncmp(valptr, "text:", 5)) + { + /* + * Decode text: URI and add to the buffer... + */ + + valptr += 5; + + while (*valptr && !_cups_isspace(*valptr) && bufptr < bufend) + { + if (*valptr == '%' && isxdigit(valptr[1] & 255) && + isxdigit(valptr[2] & 255)) + { + /* + * Pull a hex-encoded character from the URI... + */ + + valptr ++; + + if (isdigit(*valptr & 255)) + ch = (*valptr - '0') << 4; + else + ch = (tolower(*valptr) - 'a' + 10) << 4; + valptr ++; + + if (isdigit(*valptr & 255)) + *bufptr++ = ch | (*valptr - '0'); + else + *bufptr++ = ch | (tolower(*valptr) - 'a' + 10); + valptr ++; + } + else if (*valptr == '+') + { + *bufptr++ = ' '; + valptr ++; + } + else + *bufptr++ = *valptr++; + } + } + else + { + /* + * Skip this URI... + */ + + while (*valptr && !_cups_isspace(*valptr)) + valptr++; + } + + /* + * Skip whitespace... + */ + + while (_cups_isspace(*valptr)) + valptr ++; + } + + if (bufptr > buffer) + *bufptr = '\0'; + + return (buffer); + } + else + { + /* + * Copy a URI... + */ + + schemelen = strlen(scheme); + if (scheme[schemelen - 1] == ':') /* Force scheme to be just the name */ + schemelen --; + + for (valptr = locattr->value, bufptr = buffer; *valptr && bufptr < bufend;) + { + if ((!strncmp(valptr, scheme, schemelen) && valptr[schemelen] == ':') || + (*valptr == '/' && !strcmp(scheme, "file"))) + { + /* + * Copy URI... + */ + + while (*valptr && !_cups_isspace(*valptr) && bufptr < bufend) + *bufptr++ = *valptr++; + + *bufptr = '\0'; + + return (buffer); + } + else + { + /* + * Skip this URI... + */ + + while (*valptr && !_cups_isspace(*valptr)) + valptr++; + } + + /* + * Skip whitespace... + */ + + while (_cups_isspace(*valptr)) + valptr ++; + } + + return (NULL); + } +} + + +/* + * 'ppdLocalizeMarkerName()' - Get the localized version of a marker-names + * attribute value. + * + * This function uses the current locale to find the corresponding name + * text from the attribute value. If no localized text for the requested + * name can be found, @code NULL@ is returned. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +const char * /* O - Value or @code NULL@ if not found */ +ppdLocalizeMarkerName( + ppd_file_t *ppd, /* I - PPD file */ + const char *name) /* I - Marker name to look up */ +{ + ppd_attr_t *locattr; /* Localized attribute */ + char ll_CC[6]; /* Language + country locale */ + + + /* + * Range check input... + */ + + if (!ppd || !name) + return (NULL); + + /* + * Get the default language... + */ + + ppd_ll_CC(ll_CC, sizeof(ll_CC)); + + /* + * Find the localized attribute... + */ + + if ((locattr = _ppdLocalizedAttr(ppd, "cupsMarkerName", name, + ll_CC)) == NULL) + locattr = ppdFindAttr(ppd, "cupsMarkerName", name); + + return (locattr ? locattr->text : NULL); +} + + +/* + * '_ppdFreeLanguages()' - Free an array of languages from _ppdGetLanguages. + */ + +void +_ppdFreeLanguages( + cups_array_t *languages) /* I - Languages array */ +{ + char *language; /* Current language */ + + + for (language = (char *)cupsArrayFirst(languages); + language; + language = (char *)cupsArrayNext(languages)) + free(language); + + cupsArrayDelete(languages); +} + + +/* + * '_ppdGetLanguages()' - Get an array of languages from a PPD file. + */ + +cups_array_t * /* O - Languages array */ +_ppdGetLanguages(ppd_file_t *ppd) /* I - PPD file */ +{ + cups_array_t *languages; /* Languages array */ + ppd_attr_t *attr; /* cupsLanguages attribute */ + char *value, /* Copy of attribute value */ + *start, /* Start of current language */ + *ptr; /* Pointer into languages */ + + + /* + * See if we have a cupsLanguages attribute... + */ + + if ((attr = ppdFindAttr(ppd, "cupsLanguages", NULL)) == NULL || !attr->value) + return (NULL); + + /* + * Yes, load the list... + */ + + if ((languages = cupsArrayNew((cups_array_func_t)strcmp, NULL)) == NULL) + return (NULL); + + if ((value = strdup(attr->value)) == NULL) + { + cupsArrayDelete(languages); + return (NULL); + } + + for (ptr = value; *ptr;) + { + /* + * Skip leading whitespace... + */ + + while (_cups_isspace(*ptr)) + ptr ++; + + if (!*ptr) + break; + + /* + * Find the end of this language name... + */ + + for (start = ptr; *ptr && !_cups_isspace(*ptr); ptr ++); + + if (*ptr) + *ptr++ = '\0'; + + if (!strcmp(start, "en")) + continue; + + cupsArrayAdd(languages, strdup(start)); + } + + /* + * Free the temporary string and return either an array with one or more + * values or a NULL pointer... + */ + + free(value); + + if (cupsArrayCount(languages) == 0) + { + cupsArrayDelete(languages); + return (NULL); + } + else + return (languages); +} + + +/* + * '_ppdHashName()' - Generate a hash value for a device or profile name. + * + * This function is primarily used on Mac OS X, but is generally accessible + * since cupstestppd needs to check for profile name collisions in PPD files... + */ + +unsigned /* O - Hash value */ +_ppdHashName(const char *name) /* I - Name to hash */ +{ + int mult; /* Multiplier */ + unsigned hash = 0; /* Hash value */ + + + for (mult = 1; *name && mult <= 128; mult ++, name ++) + hash += (*name & 255) * mult; + + return (hash); +} + + +/* + * '_ppdLocalizedAttr()' - Find a localized attribute. + */ + +ppd_attr_t * /* O - Localized attribute or NULL */ +_ppdLocalizedAttr(ppd_file_t *ppd, /* I - PPD file */ + const char *keyword, /* I - Main keyword */ + const char *spec, /* I - Option keyword */ + const char *ll_CC) /* I - Language + country locale */ +{ + char lkeyword[PPD_MAX_NAME]; /* Localization keyword */ + ppd_attr_t *attr; /* Current attribute */ + + + DEBUG_printf(("4_ppdLocalizedAttr(ppd=%p, keyword=\"%s\", spec=\"%s\", " + "ll_CC=\"%s\")", ppd, keyword, spec, ll_CC)); + + /* + * Look for Keyword.ll_CC, then Keyword.ll... + */ + + snprintf(lkeyword, sizeof(lkeyword), "%s.%s", ll_CC, keyword); + if ((attr = ppdFindAttr(ppd, lkeyword, spec)) == NULL) + { + snprintf(lkeyword, sizeof(lkeyword), "%2.2s.%s", ll_CC, keyword); + attr = ppdFindAttr(ppd, lkeyword, spec); + + if (!attr) + { + if (!strncmp(ll_CC, "ja", 2)) + { + /* + * Due to a bug in the CUPS DDK 1.1.0 ppdmerge program, Japanese + * PPD files were incorrectly assigned "jp" as the locale name + * instead of "ja". Support both the old (incorrect) and new + * locale names for Japanese... + */ + + snprintf(lkeyword, sizeof(lkeyword), "jp.%s", keyword); + attr = ppdFindAttr(ppd, lkeyword, spec); + } + else if (!strncmp(ll_CC, "no", 2)) + { + /* + * Norway has two languages, "Bokmal" (the primary one) + * and "Nynorsk" (new Norwegian); we map "no" to "nb" here as + * recommended by the locale folks... + */ + + snprintf(lkeyword, sizeof(lkeyword), "nb.%s", keyword); + attr = ppdFindAttr(ppd, lkeyword, spec); + } + } + } + +#ifdef DEBUG + if (attr) + DEBUG_printf(("5_ppdLocalizedAttr: *%s %s/%s: \"%s\"\n", attr->name, + attr->spec, attr->text, attr->value ? attr->value : "")); + else + DEBUG_puts("5_ppdLocalizedAttr: NOT FOUND"); +#endif /* DEBUG */ + + return (attr); +} + + +/* + * 'ppd_ll_CC()' - Get the current locale names. + */ + +static cups_lang_t * /* O - Current language */ +ppd_ll_CC(char *ll_CC, /* O - Country-specific locale name */ + int ll_CC_size) /* I - Size of country-specific name */ +{ + cups_lang_t *lang; /* Current language */ + + + /* + * Get the current locale... + */ + + if ((lang = cupsLangDefault()) == NULL) + { + strlcpy(ll_CC, "en_US", ll_CC_size); + return (NULL); + } + + /* + * Copy the locale name... + */ + + strlcpy(ll_CC, lang->language, ll_CC_size); + + if (strlen(ll_CC) == 2) + { + /* + * Map "ll" to primary/origin country locales to have the best + * chance of finding a match... + */ + + if (!strcmp(ll_CC, "cs")) + strlcpy(ll_CC, "cs_CZ", ll_CC_size); + else if (!strcmp(ll_CC, "en")) + strlcpy(ll_CC, "en_US", ll_CC_size); + else if (!strcmp(ll_CC, "ja")) + strlcpy(ll_CC, "ja_JP", ll_CC_size); + else if (!strcmp(ll_CC, "sv")) + strlcpy(ll_CC, "sv_SE", ll_CC_size); + else if (!strcmp(ll_CC, "zh")) /* Simplified Chinese */ + strlcpy(ll_CC, "zh_CN", ll_CC_size); + } + + DEBUG_printf(("8ppd_ll_CC: lang->language=\"%s\", ll_CC=\"%s\"...", + lang->language, ll_CC)); + return (lang); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/mark.c b/cups/mark.c new file mode 100644 index 0000000000..214be77f37 --- /dev/null +++ b/cups/mark.c @@ -0,0 +1,1101 @@ +/* + * "$Id$" + * + * Option marking routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * PostScript is a trademark of Adobe Systems, Inc. + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsMarkOptions() - Mark command-line options in a PPD file. + * ppdFindChoice() - Return a pointer to an option choice. + * ppdFindMarkedChoice() - Return the marked choice for the specified option. + * ppdFindOption() - Return a pointer to the specified option. + * ppdIsMarked() - Check to see if an option is marked. + * ppdMarkDefaults() - Mark all default options in the PPD file. + * ppdMarkOption() - Mark an option in a PPD file and return the number + * of conflicts. + * ppdFirstOption() - Return the first option in the PPD file. + * ppdNextOption() - Return the next option in the PPD file. + * _ppdParseOptions() - Parse options from a PPD file. + * ppd_debug_marked() - Output the marked array to stdout... + * ppd_defaults() - Set the defaults for this group and all sub-groups. + * ppd_mark_choices() - Mark one or more option choices from a string. + * ppd_mark_option() - Quickly mark an option without checking for + * conflicts. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" + + +/* + * Local functions... + */ + +#ifdef DEBUG +static void ppd_debug_marked(ppd_file_t *ppd, const char *title); +#else +# define ppd_debug_marked(ppd,title) +#endif /* DEBUG */ +static void ppd_defaults(ppd_file_t *ppd, ppd_group_t *g); +static void ppd_mark_choices(ppd_file_t *ppd, const char *s); +static void ppd_mark_option(ppd_file_t *ppd, const char *option, + const char *choice); + + +/* + * 'cupsMarkOptions()' - Mark command-line options in a PPD file. + * + * This function maps the IPP "finishings", "media", "mirror", + * "multiple-document-handling", "output-bin", "print-color-mode", + * "print-quality", "printer-resolution", and "sides" attributes to their + * corresponding PPD options and choices. + */ + +int /* O - 1 if conflicts exist, 0 otherwise */ +cupsMarkOptions( + ppd_file_t *ppd, /* I - PPD file */ + int num_options, /* I - Number of options */ + cups_option_t *options) /* I - Options */ +{ + int i, j; /* Looping vars */ + char *ptr, /* Pointer into string */ + s[255]; /* Temporary string */ + const char *val, /* Pointer into value */ + *media, /* media option */ + *output_bin, /* output-bin option */ + *page_size, /* PageSize option */ + *ppd_keyword, /* PPD keyword */ + *print_color_mode, /* print-color-mode option */ + *print_quality, /* print-quality option */ + *sides; /* sides option */ + cups_option_t *optptr; /* Current option */ + ppd_attr_t *attr; /* PPD attribute */ + _ppd_cache_t *cache; /* PPD cache and mapping data */ + + + /* + * Check arguments... + */ + + if (!ppd || num_options <= 0 || !options) + return (0); + + ppd_debug_marked(ppd, "Before..."); + + /* + * Do special handling for finishings, media, output-bin, output-mode, + * print-color-mode, print-quality, and PageSize... + */ + + media = cupsGetOption("media", num_options, options); + output_bin = cupsGetOption("output-bin", num_options, options); + page_size = cupsGetOption("PageSize", num_options, options); + print_quality = cupsGetOption("print-quality", num_options, options); + sides = cupsGetOption("sides", num_options, options); + + if ((print_color_mode = cupsGetOption("print-color-mode", num_options, + options)) == NULL) + print_color_mode = cupsGetOption("output-mode", num_options, options); + + if ((media || output_bin || print_color_mode || print_quality || sides) && + !ppd->cache) + { + /* + * Load PPD cache and mapping data as needed... + */ + + ppd->cache = _ppdCacheCreateWithPPD(ppd); + } + + cache = ppd->cache; + + if (media) + { + /* + * Loop through the option string, separating it at commas and marking each + * individual option as long as the corresponding PPD option (PageSize, + * InputSlot, etc.) is not also set. + * + * For PageSize, we also check for an empty option value since some versions + * of MacOS X use it to specify auto-selection of the media based solely on + * the size. + */ + + for (val = media; *val;) + { + /* + * Extract the sub-option from the string... + */ + + for (ptr = s; *val && *val != ',' && (ptr - s) < (sizeof(s) - 1);) + *ptr++ = *val++; + *ptr++ = '\0'; + + if (*val == ',') + val ++; + + /* + * Mark it... + */ + + if (!page_size || !page_size[0]) + { + if (!_cups_strncasecmp(s, "Custom.", 7) || ppdPageSize(ppd, s)) + ppd_mark_option(ppd, "PageSize", s); + else if ((ppd_keyword = _ppdCacheGetPageSize(cache, NULL, s, NULL)) != NULL) + ppd_mark_option(ppd, "PageSize", ppd_keyword); + } + + if (cache && cache->source_option && + !cupsGetOption(cache->source_option, num_options, options) && + (ppd_keyword = _ppdCacheGetInputSlot(cache, NULL, s)) != NULL) + ppd_mark_option(ppd, cache->source_option, ppd_keyword); + + if (!cupsGetOption("MediaType", num_options, options) && + (ppd_keyword = _ppdCacheGetMediaType(cache, NULL, s)) != NULL) + ppd_mark_option(ppd, "MediaType", ppd_keyword); + } + } + + if (cache) + { + if (!cupsGetOption("com.apple.print.DocumentTicket.PMSpoolFormat", + num_options, options) && + !cupsGetOption("APPrinterPreset", num_options, options) && + (print_color_mode || print_quality)) + { + /* + * Map output-mode and print-quality to a preset... + */ + + _pwg_print_color_mode_t pwg_pcm;/* print-color-mode index */ + _pwg_print_quality_t pwg_pq; /* print-quality index */ + cups_option_t *preset;/* Current preset option */ + + if (print_color_mode && !strcmp(print_color_mode, "monochrome")) + pwg_pcm = _PWG_PRINT_COLOR_MODE_MONOCHROME; + else + pwg_pcm = _PWG_PRINT_COLOR_MODE_COLOR; + + if (print_quality) + { + pwg_pq = atoi(print_quality) - IPP_QUALITY_DRAFT; + if (pwg_pq < _PWG_PRINT_QUALITY_DRAFT) + pwg_pq = _PWG_PRINT_QUALITY_DRAFT; + else if (pwg_pq > _PWG_PRINT_QUALITY_HIGH) + pwg_pq = _PWG_PRINT_QUALITY_HIGH; + } + else + pwg_pq = _PWG_PRINT_QUALITY_NORMAL; + + if (cache->num_presets[pwg_pcm][pwg_pq] == 0) + { + /* + * Try to find a preset that works so that we maximize the chances of us + * getting a good print using IPP attributes. + */ + + if (cache->num_presets[pwg_pcm][_PWG_PRINT_QUALITY_NORMAL] > 0) + pwg_pq = _PWG_PRINT_QUALITY_NORMAL; + else if (cache->num_presets[_PWG_PRINT_COLOR_MODE_COLOR][pwg_pq] > 0) + pwg_pcm = _PWG_PRINT_COLOR_MODE_COLOR; + else + { + pwg_pq = _PWG_PRINT_QUALITY_NORMAL; + pwg_pcm = _PWG_PRINT_COLOR_MODE_COLOR; + } + } + + if (cache->num_presets[pwg_pcm][pwg_pq] > 0) + { + /* + * Copy the preset options as long as the corresponding names are not + * already defined in the IPP request... + */ + + for (i = cache->num_presets[pwg_pcm][pwg_pq], + preset = cache->presets[pwg_pcm][pwg_pq]; + i > 0; + i --, preset ++) + { + if (!cupsGetOption(preset->name, num_options, options)) + ppd_mark_option(ppd, preset->name, preset->value); + } + } + } + + if (output_bin && !cupsGetOption("OutputBin", num_options, options) && + (ppd_keyword = _ppdCacheGetOutputBin(cache, output_bin)) != NULL) + { + /* + * Map output-bin to OutputBin... + */ + + ppd_mark_option(ppd, "OutputBin", ppd_keyword); + } + + if (sides && cache->sides_option && + !cupsGetOption(cache->sides_option, num_options, options)) + { + /* + * Map sides to duplex option... + */ + + if (!strcmp(sides, "one-sided") && cache->sides_1sided) + ppd_mark_option(ppd, cache->sides_option, cache->sides_1sided); + else if (!strcmp(sides, "two-sided-long-edge") && + cache->sides_2sided_long) + ppd_mark_option(ppd, cache->sides_option, cache->sides_2sided_long); + else if (!strcmp(sides, "two-sided-short-edge") && + cache->sides_2sided_short) + ppd_mark_option(ppd, cache->sides_option, cache->sides_2sided_short); + } + } + + /* + * Mark other options... + */ + + for (i = num_options, optptr = options; i > 0; i --, optptr ++) + if (!_cups_strcasecmp(optptr->name, "media") || + !_cups_strcasecmp(optptr->name, "output-bin") || + !_cups_strcasecmp(optptr->name, "output-mode") || + !_cups_strcasecmp(optptr->name, "print-quality") || + !_cups_strcasecmp(optptr->name, "sides")) + continue; + else if (!_cups_strcasecmp(optptr->name, "resolution") || + !_cups_strcasecmp(optptr->name, "printer-resolution")) + { + ppd_mark_option(ppd, "Resolution", optptr->value); + ppd_mark_option(ppd, "SetResolution", optptr->value); + /* Calcomp, Linotype, QMS, Summagraphics, Tektronix, Varityper */ + ppd_mark_option(ppd, "JCLResolution", optptr->value); + /* HP */ + ppd_mark_option(ppd, "CNRes_PGP", optptr->value); + /* Canon */ + } + else if (!_cups_strcasecmp(optptr->name, "multiple-document-handling")) + { + if (!cupsGetOption("Collate", num_options, options) && + ppdFindOption(ppd, "Collate")) + { + if (_cups_strcasecmp(optptr->value, "separate-documents-uncollated-copies")) + ppd_mark_option(ppd, "Collate", "True"); + else + ppd_mark_option(ppd, "Collate", "False"); + } + } + else if (!_cups_strcasecmp(optptr->name, "finishings")) + { + /* + * Lookup cupsIPPFinishings attributes for each value... + */ + + for (ptr = optptr->value; *ptr;) + { + /* + * Get the next finishings number... + */ + + if (!isdigit(*ptr & 255)) + break; + + if ((j = strtol(ptr, &ptr, 10)) < 3) + break; + + /* + * Skip separator as needed... + */ + + if (*ptr == ',') + ptr ++; + + /* + * Look it up in the PPD file... + */ + + sprintf(s, "%d", j); + + if ((attr = ppdFindAttr(ppd, "cupsIPPFinishings", s)) == NULL) + continue; + + /* + * Apply "*Option Choice" settings from the attribute value... + */ + + ppd_mark_choices(ppd, attr->value); + } + } + else if (!_cups_strcasecmp(optptr->name, "APPrinterPreset")) + { + /* + * Lookup APPrinterPreset value... + */ + + if ((attr = ppdFindAttr(ppd, "APPrinterPreset", optptr->value)) != NULL) + { + /* + * Apply "*Option Choice" settings from the attribute value... + */ + + ppd_mark_choices(ppd, attr->value); + } + } + else if (!_cups_strcasecmp(optptr->name, "mirror")) + ppd_mark_option(ppd, "MirrorPrint", optptr->value); + else + ppd_mark_option(ppd, optptr->name, optptr->value); + + ppd_debug_marked(ppd, "After..."); + + return (ppdConflicts(ppd) > 0); +} + + +/* + * 'ppdFindChoice()' - Return a pointer to an option choice. + */ + +ppd_choice_t * /* O - Choice pointer or @code NULL@ */ +ppdFindChoice(ppd_option_t *o, /* I - Pointer to option */ + const char *choice) /* I - Name of choice */ +{ + int i; /* Looping var */ + ppd_choice_t *c; /* Current choice */ + + + if (!o || !choice) + return (NULL); + + if (choice[0] == '{' || !_cups_strncasecmp(choice, "Custom.", 7)) + choice = "Custom"; + + for (i = o->num_choices, c = o->choices; i > 0; i --, c ++) + if (!_cups_strcasecmp(c->choice, choice)) + return (c); + + return (NULL); +} + + +/* + * 'ppdFindMarkedChoice()' - Return the marked choice for the specified option. + */ + +ppd_choice_t * /* O - Pointer to choice or @code NULL@ */ +ppdFindMarkedChoice(ppd_file_t *ppd, /* I - PPD file */ + const char *option) /* I - Keyword/option name */ +{ + ppd_choice_t key, /* Search key for choice */ + *marked; /* Marked choice */ + + + DEBUG_printf(("2ppdFindMarkedChoice(ppd=%p, option=\"%s\")", ppd, option)); + + if ((key.option = ppdFindOption(ppd, option)) == NULL) + { + DEBUG_puts("3ppdFindMarkedChoice: Option not found, returning NULL"); + return (NULL); + } + + marked = (ppd_choice_t *)cupsArrayFind(ppd->marked, &key); + + DEBUG_printf(("3ppdFindMarkedChoice: Returning %p(%s)...", marked, + marked ? marked->choice : "NULL")); + + return (marked); +} + + +/* + * 'ppdFindOption()' - Return a pointer to the specified option. + */ + +ppd_option_t * /* O - Pointer to option or @code NULL@ */ +ppdFindOption(ppd_file_t *ppd, /* I - PPD file data */ + const char *option) /* I - Option/Keyword name */ +{ + /* + * Range check input... + */ + + if (!ppd || !option) + return (NULL); + + if (ppd->options) + { + /* + * Search in the array... + */ + + ppd_option_t key; /* Option search key */ + + + strlcpy(key.keyword, option, sizeof(key.keyword)); + + return ((ppd_option_t *)cupsArrayFind(ppd->options, &key)); + } + else + { + /* + * Search in each group... + */ + + int i, j; /* Looping vars */ + ppd_group_t *group; /* Current group */ + ppd_option_t *optptr; /* Current option */ + + + for (i = ppd->num_groups, group = ppd->groups; i > 0; i --, group ++) + for (j = group->num_options, optptr = group->options; + j > 0; + j --, optptr ++) + if (!_cups_strcasecmp(optptr->keyword, option)) + return (optptr); + + return (NULL); + } +} + + +/* + * 'ppdIsMarked()' - Check to see if an option is marked. + */ + +int /* O - Non-zero if option is marked */ +ppdIsMarked(ppd_file_t *ppd, /* I - PPD file data */ + const char *option, /* I - Option/Keyword name */ + const char *choice) /* I - Choice name */ +{ + ppd_choice_t key, /* Search key */ + *c; /* Choice pointer */ + + + if (!ppd) + return (0); + + if ((key.option = ppdFindOption(ppd, option)) == NULL) + return (0); + + if ((c = (ppd_choice_t *)cupsArrayFind(ppd->marked, &key)) == NULL) + return (0); + + return (!strcmp(c->choice, choice)); +} + + +/* + * 'ppdMarkDefaults()' - Mark all default options in the PPD file. + */ + +void +ppdMarkDefaults(ppd_file_t *ppd) /* I - PPD file record */ +{ + int i; /* Looping variables */ + ppd_group_t *g; /* Current group */ + ppd_choice_t *c; /* Current choice */ + + + if (!ppd) + return; + + /* + * Clean out the marked array... + */ + + for (c = (ppd_choice_t *)cupsArrayFirst(ppd->marked); + c; + c = (ppd_choice_t *)cupsArrayNext(ppd->marked)) + { + cupsArrayRemove(ppd->marked, c); + c->marked = 0; + } + + /* + * Then repopulate it with the defaults... + */ + + for (i = ppd->num_groups, g = ppd->groups; i > 0; i --, g ++) + ppd_defaults(ppd, g); +} + + +/* + * 'ppdMarkOption()' - Mark an option in a PPD file and return the number of + * conflicts. + */ + +int /* O - Number of conflicts */ +ppdMarkOption(ppd_file_t *ppd, /* I - PPD file record */ + const char *option, /* I - Keyword */ + const char *choice) /* I - Option name */ +{ + DEBUG_printf(("ppdMarkOption(ppd=%p, option=\"%s\", choice=\"%s\")", + ppd, option, choice)); + + /* + * Range check input... + */ + + if (!ppd || !option || !choice) + return (0); + + /* + * Mark the option... + */ + + ppd_mark_option(ppd, option, choice); + + /* + * Return the number of conflicts... + */ + + return (ppdConflicts(ppd)); +} + + +/* + * 'ppdFirstOption()' - Return the first option in the PPD file. + * + * Options are returned from all groups in ascending alphanumeric order. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ppd_option_t * /* O - First option or @code NULL@ */ +ppdFirstOption(ppd_file_t *ppd) /* I - PPD file */ +{ + if (!ppd) + return (NULL); + else + return ((ppd_option_t *)cupsArrayFirst(ppd->options)); +} + + +/* + * 'ppdNextOption()' - Return the next option in the PPD file. + * + * Options are returned from all groups in ascending alphanumeric order. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ppd_option_t * /* O - Next option or @code NULL@ */ +ppdNextOption(ppd_file_t *ppd) /* I - PPD file */ +{ + if (!ppd) + return (NULL); + else + return ((ppd_option_t *)cupsArrayNext(ppd->options)); +} + + +/* + * '_ppdParseOptions()' - Parse options from a PPD file. + * + * This function looks for strings of the form: + * + * *option choice ... *optionN choiceN + * property value ... propertyN valueN + * + * It stops when it finds a string that doesn't match this format. + */ + +int /* O - Number of options */ +_ppdParseOptions( + const char *s, /* I - String to parse */ + int num_options, /* I - Number of options */ + cups_option_t **options, /* IO - Options */ + _ppd_parse_t which) /* I - What to parse */ +{ + char option[PPD_MAX_NAME * 2 + 1], /* Current option/property */ + choice[PPD_MAX_NAME], /* Current choice/value */ + *ptr; /* Pointer into option or choice */ + + + if (!s) + return (num_options); + + /* + * Read all of the "*Option Choice" and "property value" pairs from the + * string, add them to an options array as we go... + */ + + while (*s) + { + /* + * Skip leading whitespace... + */ + + while (_cups_isspace(*s)) + s ++; + + /* + * Get the option/property name... + */ + + ptr = option; + while (*s && !_cups_isspace(*s) && ptr < (option + sizeof(option) - 1)) + *ptr++ = *s++; + + if (ptr == s || !_cups_isspace(*s)) + break; + + *ptr = '\0'; + + /* + * Get the choice... + */ + + while (_cups_isspace(*s)) + s ++; + + if (!*s) + break; + + ptr = choice; + while (*s && !_cups_isspace(*s) && ptr < (choice + sizeof(choice) - 1)) + *ptr++ = *s++; + + if (*s && !_cups_isspace(*s)) + break; + + *ptr = '\0'; + + /* + * Add it to the options array... + */ + + if (option[0] == '*' && which != _PPD_PARSE_PROPERTIES) + num_options = cupsAddOption(option + 1, choice, num_options, options); + else if (option[0] != '*' && which != _PPD_PARSE_OPTIONS) + num_options = cupsAddOption(option, choice, num_options, options); + } + + return (num_options); +} + + +#ifdef DEBUG +/* + * 'ppd_debug_marked()' - Output the marked array to stdout... + */ + +static void +ppd_debug_marked(ppd_file_t *ppd, /* I - PPD file data */ + const char *title) /* I - Title for list */ +{ + ppd_choice_t *c; /* Current choice */ + + + DEBUG_printf(("2cupsMarkOptions: %s", title)); + + for (c = (ppd_choice_t *)cupsArrayFirst(ppd->marked); + c; + c = (ppd_choice_t *)cupsArrayNext(ppd->marked)) + DEBUG_printf(("2cupsMarkOptions: %s=%s", c->option->keyword, c->choice)); +} +#endif /* DEBUG */ + + +/* + * 'ppd_defaults()' - Set the defaults for this group and all sub-groups. + */ + +static void +ppd_defaults(ppd_file_t *ppd, /* I - PPD file */ + ppd_group_t *g) /* I - Group to default */ +{ + int i; /* Looping var */ + ppd_option_t *o; /* Current option */ + ppd_group_t *sg; /* Current sub-group */ + + + for (i = g->num_options, o = g->options; i > 0; i --, o ++) + if (_cups_strcasecmp(o->keyword, "PageRegion") != 0) + ppdMarkOption(ppd, o->keyword, o->defchoice); + + for (i = g->num_subgroups, sg = g->subgroups; i > 0; i --, sg ++) + ppd_defaults(ppd, sg); +} + + +/* + * 'ppd_mark_choices()' - Mark one or more option choices from a string. + */ + +static void +ppd_mark_choices(ppd_file_t *ppd, /* I - PPD file */ + const char *s) /* I - "*Option Choice ..." string */ +{ + int i, /* Looping var */ + num_options; /* Number of options */ + cups_option_t *options, /* Options */ + *option; /* Current option */ + + + if (!s) + return; + + options = NULL; + num_options = _ppdParseOptions(s, 0, &options, 0); + + for (i = num_options, option = options; i > 0; i --, option ++) + ppd_mark_option(ppd, option->name, option->value); + + cupsFreeOptions(num_options, options); +} + + +/* + * 'ppd_mark_option()' - Quick mark an option without checking for conflicts. + */ + +static void +ppd_mark_option(ppd_file_t *ppd, /* I - PPD file */ + const char *option, /* I - Option name */ + const char *choice) /* I - Choice name */ +{ + int i, j; /* Looping vars */ + ppd_option_t *o; /* Option pointer */ + ppd_choice_t *c, /* Choice pointer */ + *oldc, /* Old choice pointer */ + key; /* Search key for choice */ + struct lconv *loc; /* Locale data */ + + + DEBUG_printf(("7ppd_mark_option(ppd=%p, option=\"%s\", choice=\"%s\")", + ppd, option, choice)); + + /* + * AP_D_InputSlot is the "default input slot" on MacOS X, and setting + * it clears the regular InputSlot choices... + */ + + if (!_cups_strcasecmp(option, "AP_D_InputSlot")) + { + cupsArraySave(ppd->options); + + if ((o = ppdFindOption(ppd, "InputSlot")) != NULL) + { + key.option = o; + if ((oldc = (ppd_choice_t *)cupsArrayFind(ppd->marked, &key)) != NULL) + { + oldc->marked = 0; + cupsArrayRemove(ppd->marked, oldc); + } + } + + cupsArrayRestore(ppd->options); + } + + /* + * Check for custom options... + */ + + cupsArraySave(ppd->options); + + o = ppdFindOption(ppd, option); + + cupsArrayRestore(ppd->options); + + if (!o) + return; + + loc = localeconv(); + + if (!_cups_strncasecmp(choice, "Custom.", 7)) + { + /* + * Handle a custom option... + */ + + if ((c = ppdFindChoice(o, "Custom")) == NULL) + return; + + if (!_cups_strcasecmp(option, "PageSize")) + { + /* + * Handle custom page sizes... + */ + + ppdPageSize(ppd, choice); + } + else + { + /* + * Handle other custom options... + */ + + ppd_coption_t *coption; /* Custom option */ + ppd_cparam_t *cparam; /* Custom parameter */ + char *units; /* Custom points units */ + + + if ((coption = ppdFindCustomOption(ppd, option)) != NULL) + { + if ((cparam = (ppd_cparam_t *)cupsArrayFirst(coption->params)) == NULL) + return; + + switch (cparam->type) + { + case PPD_CUSTOM_CURVE : + case PPD_CUSTOM_INVCURVE : + case PPD_CUSTOM_REAL : + cparam->current.custom_real = (float)_cupsStrScand(choice + 7, + NULL, loc); + break; + + case PPD_CUSTOM_POINTS : + cparam->current.custom_points = (float)_cupsStrScand(choice + 7, + &units, + loc); + + if (units) + { + if (!_cups_strcasecmp(units, "cm")) + cparam->current.custom_points *= 72.0f / 2.54f; + else if (!_cups_strcasecmp(units, "mm")) + cparam->current.custom_points *= 72.0f / 25.4f; + else if (!_cups_strcasecmp(units, "m")) + cparam->current.custom_points *= 72.0f / 0.0254f; + else if (!_cups_strcasecmp(units, "in")) + cparam->current.custom_points *= 72.0f; + else if (!_cups_strcasecmp(units, "ft")) + cparam->current.custom_points *= 12.0f * 72.0f; + } + break; + + case PPD_CUSTOM_INT : + cparam->current.custom_int = atoi(choice + 7); + break; + + case PPD_CUSTOM_PASSCODE : + case PPD_CUSTOM_PASSWORD : + case PPD_CUSTOM_STRING : + if (cparam->current.custom_string) + _cupsStrFree(cparam->current.custom_string); + + cparam->current.custom_string = _cupsStrAlloc(choice + 7); + break; + } + } + } + + /* + * Make sure that we keep the option marked below... + */ + + choice = "Custom"; + } + else if (choice[0] == '{') + { + /* + * Handle multi-value custom options... + */ + + ppd_coption_t *coption; /* Custom option */ + ppd_cparam_t *cparam; /* Custom parameter */ + char *units; /* Custom points units */ + int num_vals; /* Number of values */ + cups_option_t *vals, /* Values */ + *val; /* Value */ + + + if ((c = ppdFindChoice(o, "Custom")) == NULL) + return; + + if ((coption = ppdFindCustomOption(ppd, option)) != NULL) + { + num_vals = cupsParseOptions(choice, 0, &vals); + + for (i = 0, val = vals; i < num_vals; i ++, val ++) + { + if ((cparam = ppdFindCustomParam(coption, val->name)) == NULL) + continue; + + switch (cparam->type) + { + case PPD_CUSTOM_CURVE : + case PPD_CUSTOM_INVCURVE : + case PPD_CUSTOM_REAL : + cparam->current.custom_real = (float)_cupsStrScand(val->value, + NULL, loc); + break; + + case PPD_CUSTOM_POINTS : + cparam->current.custom_points = (float)_cupsStrScand(val->value, + &units, + loc); + + if (units) + { + if (!_cups_strcasecmp(units, "cm")) + cparam->current.custom_points *= 72.0f / 2.54f; + else if (!_cups_strcasecmp(units, "mm")) + cparam->current.custom_points *= 72.0f / 25.4f; + else if (!_cups_strcasecmp(units, "m")) + cparam->current.custom_points *= 72.0f / 0.0254f; + else if (!_cups_strcasecmp(units, "in")) + cparam->current.custom_points *= 72.0f; + else if (!_cups_strcasecmp(units, "ft")) + cparam->current.custom_points *= 12.0f * 72.0f; + } + break; + + case PPD_CUSTOM_INT : + cparam->current.custom_int = atoi(val->value); + break; + + case PPD_CUSTOM_PASSCODE : + case PPD_CUSTOM_PASSWORD : + case PPD_CUSTOM_STRING : + if (cparam->current.custom_string) + _cupsStrFree(cparam->current.custom_string); + + cparam->current.custom_string = _cupsStrRetain(val->value); + break; + } + } + + cupsFreeOptions(num_vals, vals); + } + } + else + { + for (i = o->num_choices, c = o->choices; i > 0; i --, c ++) + if (!_cups_strcasecmp(c->choice, choice)) + break; + + if (!i) + return; + } + + /* + * Option found; mark it and then handle unmarking any other options. + */ + + if (o->ui != PPD_UI_PICKMANY) + { + /* + * Unmark all other choices... + */ + + if ((oldc = (ppd_choice_t *)cupsArrayFind(ppd->marked, c)) != NULL) + { + oldc->marked = 0; + cupsArrayRemove(ppd->marked, oldc); + } + + if (!_cups_strcasecmp(option, "PageSize") || !_cups_strcasecmp(option, "PageRegion")) + { + /* + * Mark current page size... + */ + + for (j = 0; j < ppd->num_sizes; j ++) + ppd->sizes[j].marked = !_cups_strcasecmp(ppd->sizes[j].name, + choice); + + /* + * Unmark the current PageSize or PageRegion setting, as + * appropriate... + */ + + cupsArraySave(ppd->options); + + if (!_cups_strcasecmp(option, "PageSize")) + { + if ((o = ppdFindOption(ppd, "PageRegion")) != NULL) + { + key.option = o; + if ((oldc = (ppd_choice_t *)cupsArrayFind(ppd->marked, &key)) != NULL) + { + oldc->marked = 0; + cupsArrayRemove(ppd->marked, oldc); + } + } + } + else + { + if ((o = ppdFindOption(ppd, "PageSize")) != NULL) + { + key.option = o; + if ((oldc = (ppd_choice_t *)cupsArrayFind(ppd->marked, &key)) != NULL) + { + oldc->marked = 0; + cupsArrayRemove(ppd->marked, oldc); + } + } + } + + cupsArrayRestore(ppd->options); + } + else if (!_cups_strcasecmp(option, "InputSlot")) + { + /* + * Unmark ManualFeed option... + */ + + cupsArraySave(ppd->options); + + if ((o = ppdFindOption(ppd, "ManualFeed")) != NULL) + { + key.option = o; + if ((oldc = (ppd_choice_t *)cupsArrayFind(ppd->marked, &key)) != NULL) + { + oldc->marked = 0; + cupsArrayRemove(ppd->marked, oldc); + } + } + + cupsArrayRestore(ppd->options); + } + else if (!_cups_strcasecmp(option, "ManualFeed") && + !_cups_strcasecmp(choice, "True")) + { + /* + * Unmark InputSlot option... + */ + + cupsArraySave(ppd->options); + + if ((o = ppdFindOption(ppd, "InputSlot")) != NULL) + { + key.option = o; + if ((oldc = (ppd_choice_t *)cupsArrayFind(ppd->marked, &key)) != NULL) + { + oldc->marked = 0; + cupsArrayRemove(ppd->marked, oldc); + } + } + + cupsArrayRestore(ppd->options); + } + } + + c->marked = 1; + + cupsArrayAdd(ppd->marked, c); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/md5-private.h b/cups/md5-private.h new file mode 100644 index 0000000000..3667cf0b33 --- /dev/null +++ b/cups/md5-private.h @@ -0,0 +1,79 @@ +/* + * "$Id$" + * + * Private MD5 definitions for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 2005 by Easy Software Products + * + * Copyright (C) 1999 Aladdin Enterprises. All rights reserved. + * + * This software is provided 'as-is', without any express or implied + * warranty. In no event will the authors be held liable for any damages + * arising from the use of this software. + * + * Permission is granted to anyone to use this software for any purpose, + * including commercial applications, and to alter it and redistribute it + * freely, subject to the following restrictions: + * + * 1. The origin of this software must not be misrepresented; you must not + * claim that you wrote the original software. If you use this software + * in a product, an acknowledgment in the product documentation would be + * appreciated but is not required. + * 2. Altered source versions must be plainly marked as such, and must not be + * misrepresented as being the original software. + * 3. This notice may not be removed or altered from any source distribution. + * + * L. Peter Deutsch + * ghost@aladdin.com + */ + +/* + Independent implementation of MD5 (RFC 1321). + + This code implements the MD5 Algorithm defined in RFC 1321. + It is derived directly from the text of the RFC and not from the + reference implementation. + + The original and principal author of md5.h is L. Peter Deutsch + . Other authors are noted in the change history + that follows (in reverse chronological order): + + 1999-11-04 lpd Edited comments slightly for automatic TOC extraction. + 1999-10-18 lpd Fixed typo in header comment (ansi2knr rather than md5); + added conditionalization for C++ compilation from Martin + Purschke . + 1999-05-03 lpd Original version. + */ + +#ifndef _CUPS_MD5_PRIVATE_H_ +# define _CUPS_MD5_PRIVATE_H_ + +/* Define the state of the MD5 Algorithm. */ +typedef struct _cups_md5_state_s { + unsigned int count[2]; /* message length in bits, lsw first */ + unsigned int abcd[4]; /* digest buffer */ + unsigned char buf[64]; /* accumulate block */ +} _cups_md5_state_t; + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + +/* Initialize the algorithm. */ +void _cupsMD5Init(_cups_md5_state_t *pms); + +/* Append a string to the message. */ +void _cupsMD5Append(_cups_md5_state_t *pms, const unsigned char *data, int nbytes); + +/* Finish the message and return the digest. */ +void _cupsMD5Finish(_cups_md5_state_t *pms, unsigned char digest[16]); + +# ifdef __cplusplus +} /* end extern "C" */ +# endif /* __cplusplus */ +#endif /* !_CUPS_MD5_PRIVATE_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/md5.c b/cups/md5.c new file mode 100644 index 0000000000..1f46a957cd --- /dev/null +++ b/cups/md5.c @@ -0,0 +1,346 @@ +/* + * "$Id$" + * + * Private MD5 implementation for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 2005 by Easy Software Products + * + * Copyright (C) 1999 Aladdin Enterprises. All rights reserved. + * + * This software is provided 'as-is', without any express or implied + * warranty. In no event will the authors be held liable for any damages + * arising from the use of this software. + * + * Permission is granted to anyone to use this software for any purpose, + * including commercial applications, and to alter it and redistribute it + * freely, subject to the following restrictions: + * + * 1. The origin of this software must not be misrepresented; you must not + * claim that you wrote the original software. If you use this software + * in a product, an acknowledgment in the product documentation would be + * appreciated but is not required. + * 2. Altered source versions must be plainly marked as such, and must not be + * misrepresented as being the original software. + * 3. This notice may not be removed or altered from any source distribution. + * + * L. Peter Deutsch + * ghost@aladdin.com + */ +/* + Independent implementation of MD5 (RFC 1321). + + This code implements the MD5 Algorithm defined in RFC 1321. + It is derived directly from the text of the RFC and not from the + reference implementation. + + The original and principal author of md5.c is L. Peter Deutsch + . Other authors are noted in the change history + that follows (in reverse chronological order): + + 1999-11-04 lpd Edited comments slightly for automatic TOC extraction. + 1999-10-18 lpd Fixed typo in header comment (ansi2knr rather than md5). + 1999-05-03 lpd Original version. + */ + +#include "md5-private.h" +#include "string-private.h" + +#define T1 0xd76aa478 +#define T2 0xe8c7b756 +#define T3 0x242070db +#define T4 0xc1bdceee +#define T5 0xf57c0faf +#define T6 0x4787c62a +#define T7 0xa8304613 +#define T8 0xfd469501 +#define T9 0x698098d8 +#define T10 0x8b44f7af +#define T11 0xffff5bb1 +#define T12 0x895cd7be +#define T13 0x6b901122 +#define T14 0xfd987193 +#define T15 0xa679438e +#define T16 0x49b40821 +#define T17 0xf61e2562 +#define T18 0xc040b340 +#define T19 0x265e5a51 +#define T20 0xe9b6c7aa +#define T21 0xd62f105d +#define T22 0x02441453 +#define T23 0xd8a1e681 +#define T24 0xe7d3fbc8 +#define T25 0x21e1cde6 +#define T26 0xc33707d6 +#define T27 0xf4d50d87 +#define T28 0x455a14ed +#define T29 0xa9e3e905 +#define T30 0xfcefa3f8 +#define T31 0x676f02d9 +#define T32 0x8d2a4c8a +#define T33 0xfffa3942 +#define T34 0x8771f681 +#define T35 0x6d9d6122 +#define T36 0xfde5380c +#define T37 0xa4beea44 +#define T38 0x4bdecfa9 +#define T39 0xf6bb4b60 +#define T40 0xbebfbc70 +#define T41 0x289b7ec6 +#define T42 0xeaa127fa +#define T43 0xd4ef3085 +#define T44 0x04881d05 +#define T45 0xd9d4d039 +#define T46 0xe6db99e5 +#define T47 0x1fa27cf8 +#define T48 0xc4ac5665 +#define T49 0xf4292244 +#define T50 0x432aff97 +#define T51 0xab9423a7 +#define T52 0xfc93a039 +#define T53 0x655b59c3 +#define T54 0x8f0ccc92 +#define T55 0xffeff47d +#define T56 0x85845dd1 +#define T57 0x6fa87e4f +#define T58 0xfe2ce6e0 +#define T59 0xa3014314 +#define T60 0x4e0811a1 +#define T61 0xf7537e82 +#define T62 0xbd3af235 +#define T63 0x2ad7d2bb +#define T64 0xeb86d391 + +static void +_cups_md5_process(_cups_md5_state_t *pms, const unsigned char *data /*[64]*/) +{ + unsigned int + a = pms->abcd[0], b = pms->abcd[1], + c = pms->abcd[2], d = pms->abcd[3]; + unsigned int t; + +#ifndef ARCH_IS_BIG_ENDIAN +# define ARCH_IS_BIG_ENDIAN 1 /* slower, default implementation */ +#endif +#if ARCH_IS_BIG_ENDIAN + + /* + * On big-endian machines, we must arrange the bytes in the right + * order. (This also works on machines of unknown byte order.) + */ + unsigned int X[16]; + const unsigned char *xp = data; + int i; + + for (i = 0; i < 16; ++i, xp += 4) + X[i] = xp[0] + (xp[1] << 8) + (xp[2] << 16) + (xp[3] << 24); + +#else /* !ARCH_IS_BIG_ENDIAN */ + + /* + * On little-endian machines, we can process properly aligned data + * without copying it. + */ + unsigned int xbuf[16]; + const unsigned int *X; + + if (!((data - (const unsigned char *)0) & 3)) { + /* data are properly aligned */ + X = (const unsigned int *)data; + } else { + /* not aligned */ + memcpy(xbuf, data, 64); + X = xbuf; + } +#endif + +#define ROTATE_LEFT(x, n) (((x) << (n)) | ((x) >> (32 - (n)))) + + /* Round 1. */ + /* Let [abcd k s i] denote the operation + a = b + ((a + F(b,c,d) + X[k] + T[i]) <<< s). */ +#define F(x, y, z) (((x) & (y)) | (~(x) & (z))) +#define SET(a, b, c, d, k, s, Ti)\ + t = a + F(b,c,d) + X[k] + Ti;\ + a = ROTATE_LEFT(t, s) + b + /* Do the following 16 operations. */ + SET(a, b, c, d, 0, 7, T1); + SET(d, a, b, c, 1, 12, T2); + SET(c, d, a, b, 2, 17, T3); + SET(b, c, d, a, 3, 22, T4); + SET(a, b, c, d, 4, 7, T5); + SET(d, a, b, c, 5, 12, T6); + SET(c, d, a, b, 6, 17, T7); + SET(b, c, d, a, 7, 22, T8); + SET(a, b, c, d, 8, 7, T9); + SET(d, a, b, c, 9, 12, T10); + SET(c, d, a, b, 10, 17, T11); + SET(b, c, d, a, 11, 22, T12); + SET(a, b, c, d, 12, 7, T13); + SET(d, a, b, c, 13, 12, T14); + SET(c, d, a, b, 14, 17, T15); + SET(b, c, d, a, 15, 22, T16); +#undef SET + + /* Round 2. */ + /* Let [abcd k s i] denote the operation + a = b + ((a + G(b,c,d) + X[k] + T[i]) <<< s). */ +#define G(x, y, z) (((x) & (z)) | ((y) & ~(z))) +#define SET(a, b, c, d, k, s, Ti)\ + t = a + G(b,c,d) + X[k] + Ti;\ + a = ROTATE_LEFT(t, s) + b + /* Do the following 16 operations. */ + SET(a, b, c, d, 1, 5, T17); + SET(d, a, b, c, 6, 9, T18); + SET(c, d, a, b, 11, 14, T19); + SET(b, c, d, a, 0, 20, T20); + SET(a, b, c, d, 5, 5, T21); + SET(d, a, b, c, 10, 9, T22); + SET(c, d, a, b, 15, 14, T23); + SET(b, c, d, a, 4, 20, T24); + SET(a, b, c, d, 9, 5, T25); + SET(d, a, b, c, 14, 9, T26); + SET(c, d, a, b, 3, 14, T27); + SET(b, c, d, a, 8, 20, T28); + SET(a, b, c, d, 13, 5, T29); + SET(d, a, b, c, 2, 9, T30); + SET(c, d, a, b, 7, 14, T31); + SET(b, c, d, a, 12, 20, T32); +#undef SET + + /* Round 3. */ + /* Let [abcd k s t] denote the operation + a = b + ((a + H(b,c,d) + X[k] + T[i]) <<< s). */ +#define H(x, y, z) ((x) ^ (y) ^ (z)) +#define SET(a, b, c, d, k, s, Ti)\ + t = a + H(b,c,d) + X[k] + Ti;\ + a = ROTATE_LEFT(t, s) + b + /* Do the following 16 operations. */ + SET(a, b, c, d, 5, 4, T33); + SET(d, a, b, c, 8, 11, T34); + SET(c, d, a, b, 11, 16, T35); + SET(b, c, d, a, 14, 23, T36); + SET(a, b, c, d, 1, 4, T37); + SET(d, a, b, c, 4, 11, T38); + SET(c, d, a, b, 7, 16, T39); + SET(b, c, d, a, 10, 23, T40); + SET(a, b, c, d, 13, 4, T41); + SET(d, a, b, c, 0, 11, T42); + SET(c, d, a, b, 3, 16, T43); + SET(b, c, d, a, 6, 23, T44); + SET(a, b, c, d, 9, 4, T45); + SET(d, a, b, c, 12, 11, T46); + SET(c, d, a, b, 15, 16, T47); + SET(b, c, d, a, 2, 23, T48); +#undef SET + + /* Round 4. */ + /* Let [abcd k s t] denote the operation + a = b + ((a + I(b,c,d) + X[k] + T[i]) <<< s). */ +#define I(x, y, z) ((y) ^ ((x) | ~(z))) +#define SET(a, b, c, d, k, s, Ti)\ + t = a + I(b,c,d) + X[k] + Ti;\ + a = ROTATE_LEFT(t, s) + b + /* Do the following 16 operations. */ + SET(a, b, c, d, 0, 6, T49); + SET(d, a, b, c, 7, 10, T50); + SET(c, d, a, b, 14, 15, T51); + SET(b, c, d, a, 5, 21, T52); + SET(a, b, c, d, 12, 6, T53); + SET(d, a, b, c, 3, 10, T54); + SET(c, d, a, b, 10, 15, T55); + SET(b, c, d, a, 1, 21, T56); + SET(a, b, c, d, 8, 6, T57); + SET(d, a, b, c, 15, 10, T58); + SET(c, d, a, b, 6, 15, T59); + SET(b, c, d, a, 13, 21, T60); + SET(a, b, c, d, 4, 6, T61); + SET(d, a, b, c, 11, 10, T62); + SET(c, d, a, b, 2, 15, T63); + SET(b, c, d, a, 9, 21, T64); +#undef SET + + /* Then perform the following additions. (That is increment each + of the four registers by the value it had before this block + was started.) */ + pms->abcd[0] += a; + pms->abcd[1] += b; + pms->abcd[2] += c; + pms->abcd[3] += d; +} + +void +_cupsMD5Init(_cups_md5_state_t *pms) +{ + pms->count[0] = pms->count[1] = 0; + pms->abcd[0] = 0x67452301; + pms->abcd[1] = 0xefcdab89; + pms->abcd[2] = 0x98badcfe; + pms->abcd[3] = 0x10325476; +} + +void +_cupsMD5Append(_cups_md5_state_t *pms, const unsigned char *data, int nbytes) +{ + const unsigned char *p = data; + int left = nbytes; + int offset = (pms->count[0] >> 3) & 63; + unsigned int nbits = (unsigned int)(nbytes << 3); + + if (nbytes <= 0) + return; + + /* Update the message length. */ + pms->count[1] += nbytes >> 29; + pms->count[0] += nbits; + if (pms->count[0] < nbits) + pms->count[1]++; + + /* Process an initial partial block. */ + if (offset) { + int copy = (offset + nbytes > 64 ? 64 - offset : nbytes); + + memcpy(pms->buf + offset, p, copy); + if (offset + copy < 64) + return; + p += copy; + left -= copy; + _cups_md5_process(pms, pms->buf); + } + + /* Process full blocks. */ + for (; left >= 64; p += 64, left -= 64) + _cups_md5_process(pms, p); + + /* Process a final partial block. */ + if (left) + memcpy(pms->buf, p, left); +} + +void +_cupsMD5Finish(_cups_md5_state_t *pms, unsigned char digest[16]) +{ + static const unsigned char pad[64] = { + 0x80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 + }; + unsigned char data[8]; + int i; + + /* Save the length before padding. */ + for (i = 0; i < 8; ++i) + data[i] = (unsigned char)(pms->count[i >> 2] >> ((i & 3) << 3)); + /* Pad to 56 bytes mod 64. */ + _cupsMD5Append(pms, pad, ((55 - (pms->count[0] >> 3)) & 63) + 1); + /* Append the length. */ + _cupsMD5Append(pms, data, 8); + for (i = 0; i < 16; ++i) + digest[i] = (unsigned char)(pms->abcd[i >> 2] >> ((i & 3) << 3)); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/md5passwd.c b/cups/md5passwd.c new file mode 100644 index 0000000000..a0dc2cf5bd --- /dev/null +++ b/cups/md5passwd.c @@ -0,0 +1,142 @@ +/* + * "$Id$" + * + * MD5 password support for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * httpMD5() - Compute the MD5 sum of the username:group:password. + * httpMD5Nonce() - Combine the MD5 sum of the username, group, and password + * with the server-supplied nonce value. + * httpMD5String() - Convert an MD5 sum to a character string. + */ + +/* + * Include necessary headers... + */ + +#include "http-private.h" +#include "string-private.h" + + +/* + * 'httpMD5()' - Compute the MD5 sum of the username:group:password. + */ + +char * /* O - MD5 sum */ +httpMD5(const char *username, /* I - User name */ + const char *realm, /* I - Realm name */ + const char *passwd, /* I - Password string */ + char md5[33]) /* O - MD5 string */ +{ + _cups_md5_state_t state; /* MD5 state info */ + unsigned char sum[16]; /* Sum data */ + char line[256]; /* Line to sum */ + + + /* + * Compute the MD5 sum of the user name, group name, and password. + */ + + snprintf(line, sizeof(line), "%s:%s:%s", username, realm, passwd); + _cupsMD5Init(&state); + _cupsMD5Append(&state, (unsigned char *)line, (int)strlen(line)); + _cupsMD5Finish(&state, sum); + + /* + * Return the sum... + */ + + return (httpMD5String(sum, md5)); +} + + +/* + * 'httpMD5Final()' - Combine the MD5 sum of the username, group, and password + * with the server-supplied nonce value, method, and + * request-uri. + */ + +char * /* O - New sum */ +httpMD5Final(const char *nonce, /* I - Server nonce value */ + const char *method, /* I - METHOD (GET, POST, etc.) */ + const char *resource, /* I - Resource path */ + char md5[33]) /* IO - MD5 sum */ +{ + _cups_md5_state_t state; /* MD5 state info */ + unsigned char sum[16]; /* Sum data */ + char line[1024]; /* Line of data */ + char a2[33]; /* Hash of method and resource */ + + + /* + * First compute the MD5 sum of the method and resource... + */ + + snprintf(line, sizeof(line), "%s:%s", method, resource); + _cupsMD5Init(&state); + _cupsMD5Append(&state, (unsigned char *)line, (int)strlen(line)); + _cupsMD5Finish(&state, sum); + httpMD5String(sum, a2); + + /* + * Then combine A1 (MD5 of username, realm, and password) with the nonce + * and A2 (method + resource) values to get the final MD5 sum for the + * request... + */ + + snprintf(line, sizeof(line), "%s:%s:%s", md5, nonce, a2); + + _cupsMD5Init(&state); + _cupsMD5Append(&state, (unsigned char *)line, (int)strlen(line)); + _cupsMD5Finish(&state, sum); + + return (httpMD5String(sum, md5)); +} + + +/* + * 'httpMD5String()' - Convert an MD5 sum to a character string. + */ + +char * /* O - MD5 sum in hex */ +httpMD5String(const unsigned char *sum, /* I - MD5 sum data */ + char md5[33]) + /* O - MD5 sum in hex */ +{ + int i; /* Looping var */ + char *md5ptr; /* Pointer into MD5 string */ + static const char hex[] = "0123456789abcdef"; + /* Hex digits */ + + + /* + * Convert the MD5 sum to hexadecimal... + */ + + for (i = 16, md5ptr = md5; i > 0; i --, sum ++) + { + *md5ptr++ = hex[*sum >> 4]; + *md5ptr++ = hex[*sum & 15]; + } + + *md5ptr = '\0'; + + return (md5); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/notify.c b/cups/notify.c new file mode 100644 index 0000000000..82913cb9b9 --- /dev/null +++ b/cups/notify.c @@ -0,0 +1,202 @@ +/* + * "$Id$" + * + * Notification routines for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 2005-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsNotifySubject() - Return the subject for the given notification + * message. + * cupsNotifyText() - Return the text for the given notification message. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" + + +/* + * 'cupsNotifySubject()' - Return the subject for the given notification message. + * + * The returned string must be freed by the caller using @code free@. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +char * /* O - Subject string or @code NULL@ */ +cupsNotifySubject(cups_lang_t *lang, /* I - Language data */ + ipp_t *event) /* I - Event data */ +{ + char buffer[1024]; /* Subject buffer */ + const char *prefix, /* Prefix on subject */ + *state; /* Printer/job state string */ + ipp_attribute_t *job_id, /* notify-job-id */ + *job_name, /* job-name */ + *job_state, /* job-state */ + *printer_name, /* printer-name */ + *printer_state, /* printer-state */ + *printer_uri, /* notify-printer-uri */ + *subscribed; /* notify-subscribed-event */ + + + /* + * Range check input... + */ + + if (!event || !lang) + return (NULL); + + /* + * Get the required attributes... + */ + + job_id = ippFindAttribute(event, "notify-job-id", IPP_TAG_INTEGER); + job_name = ippFindAttribute(event, "job-name", IPP_TAG_NAME); + job_state = ippFindAttribute(event, "job-state", IPP_TAG_ENUM); + printer_name = ippFindAttribute(event, "printer-name", IPP_TAG_NAME); + printer_state = ippFindAttribute(event, "printer-state", IPP_TAG_ENUM); + printer_uri = ippFindAttribute(event, "notify-printer-uri", IPP_TAG_URI); + subscribed = ippFindAttribute(event, "notify-subscribed-event", + IPP_TAG_KEYWORD); + + + if (job_id && printer_name && printer_uri && job_state) + { + /* + * Job event... + */ + + prefix = _cupsLangString(lang, _("Print Job:")); + + switch (job_state->values[0].integer) + { + case IPP_JOB_PENDING : + state = _cupsLangString(lang, _("pending")); + break; + case IPP_JOB_HELD : + state = _cupsLangString(lang, _("held")); + break; + case IPP_JOB_PROCESSING : + state = _cupsLangString(lang, _("processing")); + break; + case IPP_JOB_STOPPED : + state = _cupsLangString(lang, _("stopped")); + break; + case IPP_JOB_CANCELED : + state = _cupsLangString(lang, _("canceled")); + break; + case IPP_JOB_ABORTED : + state = _cupsLangString(lang, _("aborted")); + break; + case IPP_JOB_COMPLETED : + state = _cupsLangString(lang, _("completed")); + break; + default : + state = _cupsLangString(lang, _("unknown")); + break; + } + + snprintf(buffer, sizeof(buffer), "%s %s-%d (%s) %s", + prefix, + printer_name->values[0].string.text, + job_id->values[0].integer, + job_name ? job_name->values[0].string.text : + _cupsLangString(lang, _("untitled")), + state); + } + else if (printer_uri && printer_name && printer_state) + { + /* + * Printer event... + */ + + prefix = _cupsLangString(lang, _("Printer:")); + + switch (printer_state->values[0].integer) + { + case IPP_PRINTER_IDLE : + state = _cupsLangString(lang, _("idle")); + break; + case IPP_PRINTER_PROCESSING : + state = _cupsLangString(lang, _("processing")); + break; + case IPP_PRINTER_STOPPED : + state = _cupsLangString(lang, _("stopped")); + break; + default : + state = _cupsLangString(lang, _("unknown")); + break; + } + + snprintf(buffer, sizeof(buffer), "%s %s %s", + prefix, + printer_name->values[0].string.text, + state); + } + else if (subscribed) + strlcpy(buffer, subscribed->values[0].string.text, sizeof(buffer)); + else + return (NULL); + + /* + * Duplicate and return the subject string... + */ + + return (strdup(buffer)); +} + + +/* + * 'cupsNotifyText()' - Return the text for the given notification message. + * + * The returned string must be freed by the caller using @code free@. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +char * /* O - Message text or @code NULL@ */ +cupsNotifyText(cups_lang_t *lang, /* I - Language data */ + ipp_t *event) /* I - Event data */ +{ + ipp_attribute_t *notify_text; /* notify-text */ + + + /* + * Range check input... + */ + + if (!event || !lang) + return (NULL); + + /* + * Get the notify-text attribute from the server... + */ + + if ((notify_text = ippFindAttribute(event, "notify-text", + IPP_TAG_TEXT)) == NULL) + return (NULL); + + /* + * Return a copy... + */ + + return (strdup(notify_text->values[0].string.text)); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/options.c b/cups/options.c new file mode 100644 index 0000000000..4cff43c0f9 --- /dev/null +++ b/cups/options.c @@ -0,0 +1,711 @@ +/* + * "$Id$" + * + * Option routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsAddOption() - Add an option to an option array. + * cupsFreeOptions() - Free all memory used by options. + * cupsGetOption() - Get an option value. + * cupsParseOptions() - Parse options from a command-line argument. + * cupsRemoveOption() - Remove an option from an option array. + * _cupsGet1284Values() - Get 1284 device ID keys and values. + * cups_compare_options() - Compare two options. + * cups_find_option() - Find an option using a binary search. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" + + +/* + * Local functions... + */ + +static int cups_compare_options(cups_option_t *a, cups_option_t *b); +static int cups_find_option(const char *name, int num_options, + cups_option_t *option, int prev, int *rdiff); + + +/* + * 'cupsAddOption()' - Add an option to an option array. + * + * New option arrays can be initialized simply by passing 0 for the + * "num_options" parameter. + */ + +int /* O - Number of options */ +cupsAddOption(const char *name, /* I - Name of option */ + const char *value, /* I - Value of option */ + int num_options,/* I - Number of options */ + cups_option_t **options) /* IO - Pointer to options */ +{ + cups_option_t *temp; /* Pointer to new option */ + int insert, /* Insertion point */ + diff; /* Result of search */ + + + DEBUG_printf(("2cupsAddOption(name=\"%s\", value=\"%s\", num_options=%d, " + "options=%p)", name, value, num_options, options)); + + if (!name || !name[0] || !value || !options || num_options < 0) + { + DEBUG_printf(("3cupsAddOption: Returning %d", num_options)); + return (num_options); + } + + /* + * Look for an existing option with the same name... + */ + + if (num_options == 0) + { + insert = 0; + diff = 1; + } + else + { + insert = cups_find_option(name, num_options, *options, num_options - 1, + &diff); + + if (diff > 0) + insert ++; + } + + if (diff) + { + /* + * No matching option name... + */ + + DEBUG_printf(("4cupsAddOption: New option inserted at index %d...", + insert)); + + if (num_options == 0) + temp = (cups_option_t *)malloc(sizeof(cups_option_t)); + else + temp = (cups_option_t *)realloc(*options, sizeof(cups_option_t) * + (num_options + 1)); + + if (temp == NULL) + { + DEBUG_puts("3cupsAddOption: Unable to expand option array, returning 0"); + return (0); + } + + *options = temp; + + if (insert < num_options) + { + DEBUG_printf(("4cupsAddOption: Shifting %d options...", + (int)(num_options - insert))); + memmove(temp + insert + 1, temp + insert, + (num_options - insert) * sizeof(cups_option_t)); + } + + temp += insert; + temp->name = _cupsStrAlloc(name); + num_options ++; + } + else + { + /* + * Match found; free the old value... + */ + + DEBUG_printf(("4cupsAddOption: Option already exists at index %d...", + insert)); + + temp = *options + insert; + _cupsStrFree(temp->value); + } + + temp->value = _cupsStrAlloc(value); + + DEBUG_printf(("3cupsAddOption: Returning %d", num_options)); + + return (num_options); +} + + +/* + * 'cupsFreeOptions()' - Free all memory used by options. + */ + +void +cupsFreeOptions( + int num_options, /* I - Number of options */ + cups_option_t *options) /* I - Pointer to options */ +{ + int i; /* Looping var */ + + + DEBUG_printf(("cupsFreeOptions(num_options=%d, options=%p)", num_options, + options)); + + if (num_options <= 0 || !options) + return; + + for (i = 0; i < num_options; i ++) + { + _cupsStrFree(options[i].name); + _cupsStrFree(options[i].value); + } + + free(options); +} + + +/* + * 'cupsGetOption()' - Get an option value. + */ + +const char * /* O - Option value or @code NULL@ */ +cupsGetOption(const char *name, /* I - Name of option */ + int num_options,/* I - Number of options */ + cups_option_t *options) /* I - Options */ +{ + int diff, /* Result of comparison */ + match; /* Matching index */ + + + DEBUG_printf(("2cupsGetOption(name=\"%s\", num_options=%d, options=%p)", + name, num_options, options)); + + if (!name || num_options <= 0 || !options) + { + DEBUG_puts("3cupsGetOption: Returning NULL"); + return (NULL); + } + + match = cups_find_option(name, num_options, options, -1, &diff); + + if (!diff) + { + DEBUG_printf(("3cupsGetOption: Returning \"%s\"", options[match].value)); + return (options[match].value); + } + + DEBUG_puts("3cupsGetOption: Returning NULL"); + return (NULL); +} + + +/* + * 'cupsParseOptions()' - Parse options from a command-line argument. + * + * This function converts space-delimited name/value pairs according + * to the PAPI text option ABNF specification. Collection values + * ("name={a=... b=... c=...}") are stored with the curley brackets + * intact - use @code cupsParseOptions@ on the value to extract the + * collection attributes. + */ + +int /* O - Number of options found */ +cupsParseOptions( + const char *arg, /* I - Argument to parse */ + int num_options, /* I - Number of options */ + cups_option_t **options) /* O - Options found */ +{ + char *copyarg, /* Copy of input string */ + *ptr, /* Pointer into string */ + *name, /* Pointer to name */ + *value, /* Pointer to value */ + sep, /* Separator character */ + quote; /* Quote character */ + + + DEBUG_printf(("cupsParseOptions(arg=\"%s\", num_options=%d, options=%p)", + arg, num_options, options)); + + /* + * Range check input... + */ + + if (!arg) + { + DEBUG_printf(("1cupsParseOptions: Returning %d", num_options)); + return (num_options); + } + + if (!options || num_options < 0) + { + DEBUG_puts("1cupsParseOptions: Returning 0"); + return (0); + } + + /* + * Make a copy of the argument string and then divide it up... + */ + + if ((copyarg = strdup(arg)) == NULL) + { + DEBUG_puts("1cupsParseOptions: Unable to copy arg string"); + DEBUG_printf(("1cupsParseOptions: Returning %d", num_options)); + return (num_options); + } + + if (*copyarg == '{') + { + /* + * Remove surrounding {} so we can parse "{name=value ... name=value}"... + */ + + if ((ptr = copyarg + strlen(copyarg) - 1) > copyarg && *ptr == '}') + { + *ptr = '\0'; + ptr = copyarg + 1; + } + else + ptr = copyarg; + } + else + ptr = copyarg; + + /* + * Skip leading spaces... + */ + + while (_cups_isspace(*ptr)) + ptr ++; + + /* + * Loop through the string... + */ + + while (*ptr != '\0') + { + /* + * Get the name up to a SPACE, =, or end-of-string... + */ + + name = ptr; + while (!strchr("\f\n\r\t\v =", *ptr) && *ptr) + ptr ++; + + /* + * Avoid an empty name... + */ + + if (ptr == name) + break; + + /* + * Skip trailing spaces... + */ + + while (_cups_isspace(*ptr)) + *ptr++ = '\0'; + + if ((sep = *ptr) == '=') + *ptr++ = '\0'; + + DEBUG_printf(("2cupsParseOptions: name=\"%s\"", name)); + + if (sep != '=') + { + /* + * Boolean option... + */ + + if (!_cups_strncasecmp(name, "no", 2)) + num_options = cupsAddOption(name + 2, "false", num_options, + options); + else + num_options = cupsAddOption(name, "true", num_options, options); + + continue; + } + + /* + * Remove = and parse the value... + */ + + value = ptr; + + while (*ptr && !_cups_isspace(*ptr)) + { + if (*ptr == ',') + ptr ++; + else if (*ptr == '\'' || *ptr == '\"') + { + /* + * Quoted string constant... + */ + + quote = *ptr; + _cups_strcpy(ptr, ptr + 1); + + while (*ptr != quote && *ptr) + { + if (*ptr == '\\' && ptr[1]) + _cups_strcpy(ptr, ptr + 1); + + ptr ++; + } + + if (*ptr) + _cups_strcpy(ptr, ptr + 1); + } + else if (*ptr == '{') + { + /* + * Collection value... + */ + + int depth; + + for (depth = 0; *ptr; ptr ++) + { + if (*ptr == '{') + depth ++; + else if (*ptr == '}') + { + depth --; + if (!depth) + { + ptr ++; + break; + } + } + else if (*ptr == '\\' && ptr[1]) + _cups_strcpy(ptr, ptr + 1); + } + } + else + { + /* + * Normal space-delimited string... + */ + + while (*ptr && !_cups_isspace(*ptr)) + { + if (*ptr == '\\' && ptr[1]) + _cups_strcpy(ptr, ptr + 1); + + ptr ++; + } + } + } + + if (*ptr != '\0') + *ptr++ = '\0'; + + DEBUG_printf(("2cupsParseOptions: value=\"%s\"", value)); + + /* + * Skip trailing whitespace... + */ + + while (_cups_isspace(*ptr)) + ptr ++; + + /* + * Add the string value... + */ + + num_options = cupsAddOption(name, value, num_options, options); + } + + /* + * Free the copy of the argument we made and return the number of options + * found. + */ + + free(copyarg); + + DEBUG_printf(("1cupsParseOptions: Returning %d", num_options)); + + return (num_options); +} + + +/* + * 'cupsRemoveOption()' - Remove an option from an option array. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - New number of options */ +cupsRemoveOption( + const char *name, /* I - Option name */ + int num_options, /* I - Current number of options */ + cups_option_t **options) /* IO - Options */ +{ + int i; /* Looping var */ + cups_option_t *option; /* Current option */ + + + DEBUG_printf(("2cupsRemoveOption(name=\"%s\", num_options=%d, options=%p)", + name, num_options, options)); + + /* + * Range check input... + */ + + if (!name || num_options < 1 || !options) + { + DEBUG_printf(("3cupsRemoveOption: Returning %d", num_options)); + return (num_options); + } + + /* + * Loop for the option... + */ + + for (i = num_options, option = *options; i > 0; i --, option ++) + if (!_cups_strcasecmp(name, option->name)) + break; + + if (i) + { + /* + * Remove this option from the array... + */ + + DEBUG_puts("4cupsRemoveOption: Found option, removing it..."); + + num_options --; + i --; + + _cupsStrFree(option->name); + _cupsStrFree(option->value); + + if (i > 0) + memmove(option, option + 1, i * sizeof(cups_option_t)); + } + + /* + * Return the new number of options... + */ + + DEBUG_printf(("3cupsRemoveOption: Returning %d", num_options)); + return (num_options); +} + + +/* + * '_cupsGet1284Values()' - Get 1284 device ID keys and values. + * + * The returned dictionary is a CUPS option array that can be queried with + * cupsGetOption and freed with cupsFreeOptions. + */ + +int /* O - Number of key/value pairs */ +_cupsGet1284Values( + const char *device_id, /* I - IEEE-1284 device ID string */ + cups_option_t **values) /* O - Array of key/value pairs */ +{ + int num_values; /* Number of values */ + char key[256], /* Key string */ + value[256], /* Value string */ + *ptr; /* Pointer into key/value */ + + + /* + * Range check input... + */ + + if (values) + *values = NULL; + + if (!device_id || !values) + return (0); + + /* + * Parse the 1284 device ID value into keys and values. The format is + * repeating sequences of: + * + * [whitespace]key:value[whitespace]; + */ + + num_values = 0; + while (*device_id) + { + while (_cups_isspace(*device_id)) + device_id ++; + + if (!*device_id) + break; + + for (ptr = key; *device_id && *device_id != ':'; device_id ++) + if (ptr < (key + sizeof(key) - 1)) + *ptr++ = *device_id; + + if (!*device_id) + break; + + while (ptr > key && _cups_isspace(ptr[-1])) + ptr --; + + *ptr = '\0'; + device_id ++; + + while (_cups_isspace(*device_id)) + device_id ++; + + if (!*device_id) + break; + + for (ptr = value; *device_id && *device_id != ';'; device_id ++) + if (ptr < (value + sizeof(value) - 1)) + *ptr++ = *device_id; + + if (!*device_id) + break; + + while (ptr > value && _cups_isspace(ptr[-1])) + ptr --; + + *ptr = '\0'; + device_id ++; + + num_values = cupsAddOption(key, value, num_values, values); + } + + return (num_values); +} + + +/* + * 'cups_compare_options()' - Compare two options. + */ + +static int /* O - Result of comparison */ +cups_compare_options(cups_option_t *a, /* I - First option */ + cups_option_t *b) /* I - Second option */ +{ + return (_cups_strcasecmp(a->name, b->name)); +} + + +/* + * 'cups_find_option()' - Find an option using a binary search. + */ + +static int /* O - Index of match */ +cups_find_option( + const char *name, /* I - Option name */ + int num_options, /* I - Number of options */ + cups_option_t *options, /* I - Options */ + int prev, /* I - Previous index */ + int *rdiff) /* O - Difference of match */ +{ + int left, /* Low mark for binary search */ + right, /* High mark for binary search */ + current, /* Current index */ + diff; /* Result of comparison */ + cups_option_t key; /* Search key */ + + + DEBUG_printf(("7cups_find_option(name=\"%s\", num_options=%d, options=%p, " + "prev=%d, rdiff=%p)", name, num_options, options, prev, + rdiff)); + +#ifdef DEBUG + for (left = 0; left < num_options; left ++) + DEBUG_printf(("9cups_find_option: options[%d].name=\"%s\", .value=\"%s\"", + left, options[left].name, options[left].value)); +#endif /* DEBUG */ + + key.name = (char *)name; + + if (prev >= 0) + { + /* + * Start search on either side of previous... + */ + + if ((diff = cups_compare_options(&key, options + prev)) == 0 || + (diff < 0 && prev == 0) || + (diff > 0 && prev == (num_options - 1))) + { + *rdiff = diff; + return (prev); + } + else if (diff < 0) + { + /* + * Start with previous on right side... + */ + + left = 0; + right = prev; + } + else + { + /* + * Start wih previous on left side... + */ + + left = prev; + right = num_options - 1; + } + } + else + { + /* + * Start search in the middle... + */ + + left = 0; + right = num_options - 1; + } + + do + { + current = (left + right) / 2; + diff = cups_compare_options(&key, options + current); + + if (diff == 0) + break; + else if (diff < 0) + right = current; + else + left = current; + } + while ((right - left) > 1); + + if (diff != 0) + { + /* + * Check the last 1 or 2 elements... + */ + + if ((diff = cups_compare_options(&key, options + left)) <= 0) + current = left; + else + { + diff = cups_compare_options(&key, options + right); + current = right; + } + } + + /* + * Return the closest destination and the difference... + */ + + *rdiff = diff; + + return (current); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/page.c b/cups/page.c new file mode 100644 index 0000000000..b14713fb09 --- /dev/null +++ b/cups/page.c @@ -0,0 +1,396 @@ +/* + * "$Id$" + * + * Page size functions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * PostScript is a trademark of Adobe Systems, Inc. + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * ppdPageSize() - Get the page size record for the given size. + * ppdPageSizeLimits() - Return the custom page size limits. + * ppdPageWidth() - Get the page width for the given size. + * ppdPageLength() - Get the page length for the given size. + */ + +/* + * Include necessary headers... + */ + +#include "string-private.h" +#include "debug-private.h" +#include "ppd.h" + + +/* + * 'ppdPageSize()' - Get the page size record for the given size. + */ + +ppd_size_t * /* O - Size record for page or NULL */ +ppdPageSize(ppd_file_t *ppd, /* I - PPD file record */ + const char *name) /* I - Size name */ +{ + int i; /* Looping var */ + ppd_size_t *size; /* Current page size */ + double w, l; /* Width and length of page */ + char *nameptr; /* Pointer into name */ + struct lconv *loc; /* Locale data */ + ppd_coption_t *coption; /* Custom option for page size */ + ppd_cparam_t *cparam; /* Custom option parameter */ + + + DEBUG_printf(("2ppdPageSize(ppd=%p, name=\"%s\")", ppd, name)); + + if (!ppd) + { + DEBUG_puts("3ppdPageSize: Bad PPD pointer, returning NULL..."); + return (NULL); + } + + if (name) + { + if (!strncmp(name, "Custom.", 7) && ppd->variable_sizes) + { + /* + * Find the custom page size... + */ + + for (i = ppd->num_sizes, size = ppd->sizes; i > 0; i --, size ++) + if (!strcmp("Custom", size->name)) + break; + + if (!i) + { + DEBUG_puts("3ppdPageSize: No custom sizes, returning NULL..."); + return (NULL); + } + + /* + * Variable size; size name can be one of the following: + * + * Custom.WIDTHxLENGTHin - Size in inches + * Custom.WIDTHxLENGTHft - Size in feet + * Custom.WIDTHxLENGTHcm - Size in centimeters + * Custom.WIDTHxLENGTHmm - Size in millimeters + * Custom.WIDTHxLENGTHm - Size in meters + * Custom.WIDTHxLENGTH[pt] - Size in points + */ + + loc = localeconv(); + w = _cupsStrScand(name + 7, &nameptr, loc); + if (!nameptr || *nameptr != 'x') + return (NULL); + + l = _cupsStrScand(nameptr + 1, &nameptr, loc); + if (!nameptr) + return (NULL); + + if (!_cups_strcasecmp(nameptr, "in")) + { + w *= 72.0; + l *= 72.0; + } + else if (!_cups_strcasecmp(nameptr, "ft")) + { + w *= 12.0 * 72.0; + l *= 12.0 * 72.0; + } + else if (!_cups_strcasecmp(nameptr, "mm")) + { + w *= 72.0 / 25.4; + l *= 72.0 / 25.4; + } + else if (!_cups_strcasecmp(nameptr, "cm")) + { + w *= 72.0 / 2.54; + l *= 72.0 / 2.54; + } + else if (!_cups_strcasecmp(nameptr, "m")) + { + w *= 72.0 / 0.0254; + l *= 72.0 / 0.0254; + } + + size->width = (float)w; + size->length = (float)l; + size->left = ppd->custom_margins[0]; + size->bottom = ppd->custom_margins[1]; + size->right = (float)(w - ppd->custom_margins[2]); + size->top = (float)(l - ppd->custom_margins[3]); + + /* + * Update the custom option records for the page size, too... + */ + + if ((coption = ppdFindCustomOption(ppd, "PageSize")) != NULL) + { + if ((cparam = ppdFindCustomParam(coption, "Width")) != NULL) + cparam->current.custom_points = (float)w; + + if ((cparam = ppdFindCustomParam(coption, "Height")) != NULL) + cparam->current.custom_points = (float)l; + } + + /* + * Return the page size... + */ + + DEBUG_printf(("3ppdPageSize: Returning %p (\"%s\", %gx%g)", size, + size->name, size->width, size->length)); + + return (size); + } + else + { + /* + * Lookup by name... + */ + + for (i = ppd->num_sizes, size = ppd->sizes; i > 0; i --, size ++) + if (!_cups_strcasecmp(name, size->name)) + { + DEBUG_printf(("3ppdPageSize: Returning %p (\"%s\", %gx%g)", size, + size->name, size->width, size->length)); + + return (size); + } + } + } + else + { + /* + * Find default... + */ + + for (i = ppd->num_sizes, size = ppd->sizes; i > 0; i --, size ++) + if (size->marked) + { + DEBUG_printf(("3ppdPageSize: Returning %p (\"%s\", %gx%g)", size, + size->name, size->width, size->length)); + + return (size); + } + } + + DEBUG_puts("3ppdPageSize: Size not found, returning NULL"); + + return (NULL); +} + + +/* + * 'ppdPageSizeLimits()' - Return the custom page size limits. + * + * This function returns the minimum and maximum custom page sizes and printable + * areas based on the currently-marked (selected) options. + * + * If the specified PPD file does not support custom page sizes, both + * "minimum" and "maximum" are filled with zeroes. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +int /* O - 1 if custom sizes are supported, 0 otherwise */ +ppdPageSizeLimits(ppd_file_t *ppd, /* I - PPD file record */ + ppd_size_t *minimum, /* O - Minimum custom size */ + ppd_size_t *maximum) /* O - Maximum custom size */ +{ + ppd_choice_t *qualifier2, /* Second media qualifier */ + *qualifier3; /* Third media qualifier */ + ppd_attr_t *attr; /* Attribute */ + float width, /* Min/max width */ + length; /* Min/max length */ + char spec[PPD_MAX_NAME]; /* Selector for min/max */ + + + /* + * Range check input... + */ + + if (!ppd || !ppd->variable_sizes || !minimum || !maximum) + { + if (minimum) + memset(minimum, 0, sizeof(ppd_size_t)); + + if (maximum) + memset(maximum, 0, sizeof(ppd_size_t)); + + return (0); + } + + /* + * See if we have the cupsMediaQualifier2 and cupsMediaQualifier3 attributes... + */ + + cupsArraySave(ppd->sorted_attrs); + + if ((attr = ppdFindAttr(ppd, "cupsMediaQualifier2", NULL)) != NULL && + attr->value) + qualifier2 = ppdFindMarkedChoice(ppd, attr->value); + else + qualifier2 = NULL; + + if ((attr = ppdFindAttr(ppd, "cupsMediaQualifier3", NULL)) != NULL && + attr->value) + qualifier3 = ppdFindMarkedChoice(ppd, attr->value); + else + qualifier3 = NULL; + + /* + * Figure out the current minimum width and length... + */ + + width = ppd->custom_min[0]; + length = ppd->custom_min[1]; + + if (qualifier2) + { + /* + * Try getting cupsMinSize... + */ + + if (qualifier3) + { + snprintf(spec, sizeof(spec), ".%s.%s", qualifier2->choice, + qualifier3->choice); + attr = ppdFindAttr(ppd, "cupsMinSize", spec); + } + else + attr = NULL; + + if (!attr) + { + snprintf(spec, sizeof(spec), ".%s.", qualifier2->choice); + attr = ppdFindAttr(ppd, "cupsMinSize", spec); + } + + if (!attr && qualifier3) + { + snprintf(spec, sizeof(spec), "..%s", qualifier3->choice); + attr = ppdFindAttr(ppd, "cupsMinSize", spec); + } + + if ((attr && attr->value && + sscanf(attr->value, "%f%f", &width, &length) != 2) || !attr) + { + width = ppd->custom_min[0]; + length = ppd->custom_min[1]; + } + } + + minimum->width = width; + minimum->length = length; + minimum->left = ppd->custom_margins[0]; + minimum->bottom = ppd->custom_margins[1]; + minimum->right = width - ppd->custom_margins[2]; + minimum->top = length - ppd->custom_margins[3]; + + /* + * Figure out the current maximum width and length... + */ + + width = ppd->custom_max[0]; + length = ppd->custom_max[1]; + + if (qualifier2) + { + /* + * Try getting cupsMaxSize... + */ + + if (qualifier3) + { + snprintf(spec, sizeof(spec), ".%s.%s", qualifier2->choice, + qualifier3->choice); + attr = ppdFindAttr(ppd, "cupsMaxSize", spec); + } + else + attr = NULL; + + if (!attr) + { + snprintf(spec, sizeof(spec), ".%s.", qualifier2->choice); + attr = ppdFindAttr(ppd, "cupsMaxSize", spec); + } + + if (!attr && qualifier3) + { + snprintf(spec, sizeof(spec), "..%s", qualifier3->choice); + attr = ppdFindAttr(ppd, "cupsMaxSize", spec); + } + + if (!attr || + (attr->value && sscanf(attr->value, "%f%f", &width, &length) != 2)) + { + width = ppd->custom_max[0]; + length = ppd->custom_max[1]; + } + } + + maximum->width = width; + maximum->length = length; + maximum->left = ppd->custom_margins[0]; + maximum->bottom = ppd->custom_margins[1]; + maximum->right = width - ppd->custom_margins[2]; + maximum->top = length - ppd->custom_margins[3]; + + /* + * Return the min and max... + */ + + cupsArrayRestore(ppd->sorted_attrs); + + return (1); +} + + +/* + * 'ppdPageWidth()' - Get the page width for the given size. + */ + +float /* O - Width of page in points or 0.0 */ +ppdPageWidth(ppd_file_t *ppd, /* I - PPD file record */ + const char *name) /* I - Size name */ +{ + ppd_size_t *size; /* Page size */ + + + if ((size = ppdPageSize(ppd, name)) == NULL) + return (0.0); + else + return (size->width); +} + + +/* + * 'ppdPageLength()' - Get the page length for the given size. + */ + +float /* O - Length of page in points or 0.0 */ +ppdPageLength(ppd_file_t *ppd, /* I - PPD file */ + const char *name) /* I - Size name */ +{ + ppd_size_t *size; /* Page size */ + + + if ((size = ppdPageSize(ppd, name)) == NULL) + return (0.0); + else + return (size->length); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/ppd-cache.c b/cups/ppd-cache.c new file mode 100644 index 0000000000..9bddf1c9ae --- /dev/null +++ b/cups/ppd-cache.c @@ -0,0 +1,2605 @@ +/* + * "$Id$" + * + * PPD cache implementation for CUPS. + * + * Copyright 2010-2011 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * _ppdCacheCreateWithFile() - Create PPD cache and mapping data from a + * written file. + * _ppdCacheCreateWithPPD() - Create PWG mapping data from a PPD file. + * _ppdCacheDestroy() - Free all memory used for PWG mapping data. + * _ppdCacheGetBin() - Get the PWG output-bin keyword associated with + * a PPD OutputBin. + * _ppdCacheGetInputSlot() - Get the PPD InputSlot associated with the job + * attributes or a keyword string. + * _ppdCacheGetMediaType() - Get the PPD MediaType associated with the job + * attributes or a keyword string. + * _ppdCacheGetOutputBin() - Get the PPD OutputBin associated with the + * keyword string. + * _ppdCacheGetPageSize() - Get the PPD PageSize associated with the job + * attributes or a keyword string. + * _ppdCacheGetSize() - Get the PWG size associated with a PPD + * PageSize. + * _ppdCacheGetSource() - Get the PWG media-source associated with a PPD + * InputSlot. + * _ppdCacheGetType() - Get the PWG media-type associated with a PPD + * MediaType. + * _ppdCacheWriteFile() - Write PWG mapping data to a file. + * _pwgInputSlotForSource() - Get the InputSlot name for the given PWG + * media-source. + * _pwgMediaTypeForType() - Get the MediaType name for the given PWG + * media-type. + * _pwgPageSizeForMedia() - Get the PageSize name for the given media. + * pwg_ppdize_name() - Convert an IPP keyword to a PPD keyword. + * pwg_unppdize_name() - Convert a PPD keyword to a lowercase IPP + * keyword. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include + + +/* + * Macro to test for two almost-equal PWG measurements. + */ + +#define _PWG_EQUIVALENT(x, y) (abs((x)-(y)) < 2) + + +/* + * Local functions... + */ + +static int pwg_compare_finishings(_pwg_finishings_t *a, + _pwg_finishings_t *b); +static void pwg_free_finishings(_pwg_finishings_t *f); +static void pwg_ppdize_name(const char *ipp, char *name, size_t namesize); +static void pwg_unppdize_name(const char *ppd, char *name, size_t namesize); + + +/* + * '_ppdCacheCreateWithFile()' - Create PPD cache and mapping data from a + * written file. + * + * Use the @link _ppdCacheWriteFile@ function to write PWG mapping data to a + * file. + */ + +_ppd_cache_t * /* O - PPD cache and mapping data */ +_ppdCacheCreateWithFile( + const char *filename, /* I - File to read */ + ipp_t **attrs) /* IO - IPP attributes, if any */ +{ + cups_file_t *fp; /* File */ + _ppd_cache_t *pc; /* PWG mapping data */ + _pwg_size_t *size; /* Current size */ + _pwg_map_t *map; /* Current map */ + _pwg_finishings_t *finishings; /* Current finishings option */ + int linenum, /* Current line number */ + num_bins, /* Number of bins in file */ + num_sizes, /* Number of sizes in file */ + num_sources, /* Number of sources in file */ + num_types; /* Number of types in file */ + char line[2048], /* Current line */ + *value, /* Pointer to value in line */ + *valueptr, /* Pointer into value */ + pwg_keyword[128], /* PWG keyword */ + ppd_keyword[PPD_MAX_NAME]; + /* PPD keyword */ + _pwg_print_color_mode_t print_color_mode; + /* Print color mode for preset */ + _pwg_print_quality_t print_quality; /* Print quality for preset */ + + + DEBUG_printf(("_ppdCacheCreateWithFile(filename=\"%s\")", filename)); + + /* + * Range check input... + */ + + if (attrs) + *attrs = NULL; + + if (!filename) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + return (NULL); + } + + /* + * Open the file... + */ + + if ((fp = cupsFileOpen(filename, "r")) == NULL) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(errno), 0); + return (NULL); + } + + /* + * Read the first line and make sure it has "#CUPS-PPD-CACHE-version" in it... + */ + + if (!cupsFileGets(fp, line, sizeof(line))) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(errno), 0); + DEBUG_puts("_ppdCacheCreateWithFile: Unable to read first line."); + cupsFileClose(fp); + return (NULL); + } + + if (strncmp(line, "#CUPS-PPD-CACHE-", 16)) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + DEBUG_printf(("_ppdCacheCreateWithFile: Wrong first line \"%s\".", line)); + cupsFileClose(fp); + return (NULL); + } + + if (atoi(line + 16) != _PPD_CACHE_VERSION) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("Out of date PPD cache file."), 1); + DEBUG_printf(("_ppdCacheCreateWithFile: Cache file has version %s, " + "expected %d.", line + 16, _PPD_CACHE_VERSION)); + cupsFileClose(fp); + return (NULL); + } + + /* + * Allocate the mapping data structure... + */ + + if ((pc = calloc(1, sizeof(_ppd_cache_t))) == NULL) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(errno), 0); + DEBUG_puts("_ppdCacheCreateWithFile: Unable to allocate _ppd_cache_t."); + goto create_error; + } + + /* + * Read the file... + */ + + linenum = 0; + num_bins = 0; + num_sizes = 0; + num_sources = 0; + num_types = 0; + + while (cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + { + DEBUG_printf(("_ppdCacheCreateWithFile: line=\"%s\", value=\"%s\", " + "linenum=%d", line, value, linenum)); + + if (!value) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Missing value on line %d.", + linenum)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + else if (!_cups_strcasecmp(line, "Filter")) + { + if (!pc->filters) + pc->filters = cupsArrayNew3(NULL, NULL, NULL, 0, + (cups_acopy_func_t)_cupsStrAlloc, + (cups_afree_func_t)_cupsStrFree); + + cupsArrayAdd(pc->filters, value); + } + else if (!_cups_strcasecmp(line, "PreFilter")) + { + if (!pc->prefilters) + pc->prefilters = cupsArrayNew3(NULL, NULL, NULL, 0, + (cups_acopy_func_t)_cupsStrAlloc, + (cups_afree_func_t)_cupsStrFree); + + cupsArrayAdd(pc->prefilters, value); + } + else if (!_cups_strcasecmp(line, "Product")) + { + pc->product = _cupsStrAlloc(value); + } + else if (!_cups_strcasecmp(line, "SingleFile")) + { + pc->single_file = !_cups_strcasecmp(value, "true"); + } + else if (!_cups_strcasecmp(line, "IPP")) + { + off_t pos = cupsFileTell(fp), /* Position in file */ + length = strtol(value, NULL, 10); + /* Length of IPP attributes */ + + if (attrs && *attrs) + { + DEBUG_puts("_ppdCacheCreateWithFile: IPP listed multiple times."); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + else if (length <= 0) + { + DEBUG_puts("_ppdCacheCreateWithFile: Bad IPP length."); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + if (attrs) + { + /* + * Read IPP attributes into the provided variable... + */ + + *attrs = ippNew(); + + if (ippReadIO(fp, (ipp_iocb_t)cupsFileRead, 1, NULL, + *attrs) != IPP_DATA) + { + DEBUG_puts("_ppdCacheCreateWithFile: Bad IPP data."); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + } + else + { + /* + * Skip the IPP data entirely... + */ + + cupsFileSeek(fp, pos + length); + } + + if (cupsFileTell(fp) != (pos + length)) + { + DEBUG_puts("_ppdCacheCreateWithFile: Bad IPP data."); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + } + else if (!_cups_strcasecmp(line, "NumBins")) + { + if (num_bins > 0) + { + DEBUG_puts("_ppdCacheCreateWithFile: NumBins listed multiple times."); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + if ((num_bins = atoi(value)) <= 0 || num_bins > 65536) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Bad NumBins value %d on line " + "%d.", num_sizes, linenum)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + if ((pc->bins = calloc(num_bins, sizeof(_pwg_map_t))) == NULL) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Unable to allocate %d bins.", + num_sizes)); + _cupsSetError(IPP_INTERNAL_ERROR, strerror(errno), 0); + goto create_error; + } + } + else if (!_cups_strcasecmp(line, "Bin")) + { + if (sscanf(value, "%127s%40s", pwg_keyword, ppd_keyword) != 2) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Bad Bin on line %d.", linenum)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + if (pc->num_bins >= num_bins) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Too many Bin's on line %d.", + linenum)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + map = pc->bins + pc->num_bins; + map->pwg = _cupsStrAlloc(pwg_keyword); + map->ppd = _cupsStrAlloc(ppd_keyword); + + pc->num_bins ++; + } + else if (!_cups_strcasecmp(line, "NumSizes")) + { + if (num_sizes > 0) + { + DEBUG_puts("_ppdCacheCreateWithFile: NumSizes listed multiple times."); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + if ((num_sizes = atoi(value)) <= 0 || num_sizes > 65536) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Bad NumSizes value %d on line " + "%d.", num_sizes, linenum)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + if ((pc->sizes = calloc(num_sizes, sizeof(_pwg_size_t))) == NULL) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Unable to allocate %d sizes.", + num_sizes)); + _cupsSetError(IPP_INTERNAL_ERROR, strerror(errno), 0); + goto create_error; + } + } + else if (!_cups_strcasecmp(line, "Size")) + { + if (pc->num_sizes >= num_sizes) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Too many Size's on line %d.", + linenum)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + size = pc->sizes + pc->num_sizes; + + if (sscanf(value, "%127s%40s%d%d%d%d%d%d", pwg_keyword, ppd_keyword, + &(size->width), &(size->length), &(size->left), + &(size->bottom), &(size->right), &(size->top)) != 8) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Bad Size on line %d.", + linenum)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + size->map.pwg = _cupsStrAlloc(pwg_keyword); + size->map.ppd = _cupsStrAlloc(ppd_keyword); + + pc->num_sizes ++; + } + else if (!_cups_strcasecmp(line, "CustomSize")) + { + if (pc->custom_max_width > 0) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Too many CustomSize's on line " + "%d.", linenum)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + if (sscanf(value, "%d%d%d%d%d%d%d%d", &(pc->custom_max_width), + &(pc->custom_max_length), &(pc->custom_min_width), + &(pc->custom_min_length), &(pc->custom_size.left), + &(pc->custom_size.bottom), &(pc->custom_size.right), + &(pc->custom_size.top)) != 8) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Bad CustomSize on line %d.", + linenum)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + _pwgGenerateSize(pwg_keyword, sizeof(pwg_keyword), "custom", "max", + pc->custom_max_width, pc->custom_max_length); + pc->custom_max_keyword = _cupsStrAlloc(pwg_keyword); + + _pwgGenerateSize(pwg_keyword, sizeof(pwg_keyword), "custom", "min", + pc->custom_min_width, pc->custom_min_length); + pc->custom_min_keyword = _cupsStrAlloc(pwg_keyword); + } + else if (!_cups_strcasecmp(line, "SourceOption")) + { + pc->source_option = _cupsStrAlloc(value); + } + else if (!_cups_strcasecmp(line, "NumSources")) + { + if (num_sources > 0) + { + DEBUG_puts("_ppdCacheCreateWithFile: NumSources listed multiple " + "times."); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + if ((num_sources = atoi(value)) <= 0 || num_sources > 65536) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Bad NumSources value %d on " + "line %d.", num_sources, linenum)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + if ((pc->sources = calloc(num_sources, sizeof(_pwg_map_t))) == NULL) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Unable to allocate %d sources.", + num_sources)); + _cupsSetError(IPP_INTERNAL_ERROR, strerror(errno), 0); + goto create_error; + } + } + else if (!_cups_strcasecmp(line, "Source")) + { + if (sscanf(value, "%127s%40s", pwg_keyword, ppd_keyword) != 2) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Bad Source on line %d.", + linenum)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + if (pc->num_sources >= num_sources) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Too many Source's on line %d.", + linenum)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + map = pc->sources + pc->num_sources; + map->pwg = _cupsStrAlloc(pwg_keyword); + map->ppd = _cupsStrAlloc(ppd_keyword); + + pc->num_sources ++; + } + else if (!_cups_strcasecmp(line, "NumTypes")) + { + if (num_types > 0) + { + DEBUG_puts("_ppdCacheCreateWithFile: NumTypes listed multiple times."); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + if ((num_types = atoi(value)) <= 0 || num_types > 65536) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Bad NumTypes value %d on " + "line %d.", num_types, linenum)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + if ((pc->types = calloc(num_types, sizeof(_pwg_map_t))) == NULL) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Unable to allocate %d types.", + num_types)); + _cupsSetError(IPP_INTERNAL_ERROR, strerror(errno), 0); + goto create_error; + } + } + else if (!_cups_strcasecmp(line, "Type")) + { + if (sscanf(value, "%127s%40s", pwg_keyword, ppd_keyword) != 2) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Bad Type on line %d.", + linenum)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + if (pc->num_types >= num_types) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Too many Type's on line %d.", + linenum)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + map = pc->types + pc->num_types; + map->pwg = _cupsStrAlloc(pwg_keyword); + map->ppd = _cupsStrAlloc(ppd_keyword); + + pc->num_types ++; + } + else if (!_cups_strcasecmp(line, "Preset")) + { + /* + * Preset output-mode print-quality name=value ... + */ + + print_color_mode = (_pwg_print_color_mode_t)strtol(value, &valueptr, 10); + print_quality = (_pwg_print_quality_t)strtol(valueptr, &valueptr, 10); + + if (print_color_mode < _PWG_PRINT_COLOR_MODE_MONOCHROME || + print_color_mode >= _PWG_PRINT_COLOR_MODE_MAX || + print_quality < _PWG_PRINT_QUALITY_DRAFT || + print_quality >= _PWG_PRINT_QUALITY_MAX || + valueptr == value || !*valueptr) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Bad Preset on line %d.", + linenum)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + pc->num_presets[print_color_mode][print_quality] = + cupsParseOptions(valueptr, 0, + pc->presets[print_color_mode] + print_quality); + } + else if (!_cups_strcasecmp(line, "SidesOption")) + pc->sides_option = _cupsStrAlloc(value); + else if (!_cups_strcasecmp(line, "Sides1Sided")) + pc->sides_1sided = _cupsStrAlloc(value); + else if (!_cups_strcasecmp(line, "Sides2SidedLong")) + pc->sides_2sided_long = _cupsStrAlloc(value); + else if (!_cups_strcasecmp(line, "Sides2SidedShort")) + pc->sides_2sided_short = _cupsStrAlloc(value); + else if (!_cups_strcasecmp(line, "Finishings")) + { + if (!pc->finishings) + pc->finishings = + cupsArrayNew3((cups_array_func_t)pwg_compare_finishings, + NULL, NULL, 0, NULL, + (cups_afree_func_t)pwg_free_finishings); + + if ((finishings = calloc(1, sizeof(_pwg_finishings_t))) == NULL) + goto create_error; + + finishings->value = strtol(value, &valueptr, 10); + finishings->num_options = cupsParseOptions(valueptr, 0, + &(finishings->options)); + + cupsArrayAdd(pc->finishings, finishings); + } + else + { + DEBUG_printf(("_ppdCacheCreateWithFile: Unknown %s on line %d.", line, + linenum)); + } + } + + if (pc->num_sizes < num_sizes) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Not enough sizes (%d < %d).", + pc->num_sizes, num_sizes)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + if (pc->num_sources < num_sources) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Not enough sources (%d < %d).", + pc->num_sources, num_sources)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + if (pc->num_types < num_types) + { + DEBUG_printf(("_ppdCacheCreateWithFile: Not enough types (%d < %d).", + pc->num_types, num_types)); + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad PPD cache file."), 1); + goto create_error; + } + + cupsFileClose(fp); + + return (pc); + + /* + * If we get here the file was bad - free any data and return... + */ + + create_error: + + cupsFileClose(fp); + _ppdCacheDestroy(pc); + + if (attrs) + { + ippDelete(*attrs); + *attrs = NULL; + } + + return (NULL); +} + + +/* + * '_ppdCacheCreateWithPPD()' - Create PWG mapping data from a PPD file. + */ + +_ppd_cache_t * /* O - PPD cache and mapping data */ +_ppdCacheCreateWithPPD(ppd_file_t *ppd) /* I - PPD file */ +{ + int i, j, k; /* Looping vars */ + _ppd_cache_t *pc; /* PWG mapping data */ + ppd_option_t *input_slot, /* InputSlot option */ + *media_type, /* MediaType option */ + *output_bin, /* OutputBin option */ + *color_model, /* ColorModel option */ + *duplex; /* Duplex option */ + ppd_choice_t *choice; /* Current InputSlot/MediaType */ + _pwg_map_t *map; /* Current source/type map */ + ppd_attr_t *ppd_attr; /* Current PPD preset attribute */ + int num_options; /* Number of preset options and props */ + cups_option_t *options; /* Preset options and properties */ + ppd_size_t *ppd_size; /* Current PPD size */ + _pwg_size_t *pwg_size; /* Current PWG size */ + char pwg_keyword[3 + PPD_MAX_NAME + 1 + 12 + 1 + 12 + 3], + /* PWG keyword string */ + ppd_name[PPD_MAX_NAME]; + /* Normalized PPD name */ + const char *pwg_name; /* Standard PWG media name */ + _pwg_media_t *pwg_media; /* PWG media data */ + _pwg_print_color_mode_t pwg_print_color_mode; + /* print-color-mode index */ + _pwg_print_quality_t pwg_print_quality; + /* print-quality index */ + int similar; /* Are the old and new size similar? */ + _pwg_size_t *old_size; /* Current old size */ + int old_imageable, /* Old imageable length in 2540ths */ + old_borderless, /* Old borderless state */ + old_known_pwg; /* Old PWG name is well-known */ + int new_width, /* New width in 2540ths */ + new_length, /* New length in 2540ths */ + new_left, /* New left margin in 2540ths */ + new_bottom, /* New bottom margin in 2540ths */ + new_right, /* New right margin in 2540ths */ + new_top, /* New top margin in 2540ths */ + new_imageable, /* New imageable length in 2540ths */ + new_borderless, /* New borderless state */ + new_known_pwg; /* New PWG name is well-known */ + _pwg_size_t *new_size; /* New size to add, if any */ + const char *filter; /* Current filter */ + _pwg_finishings_t *finishings; /* Current finishings value */ + + + DEBUG_printf(("_ppdCacheCreateWithPPD(ppd=%p)", ppd)); + + /* + * Range check input... + */ + + if (!ppd) + return (NULL); + + /* + * Allocate memory... + */ + + if ((pc = calloc(1, sizeof(_ppd_cache_t))) == NULL) + { + DEBUG_puts("_ppdCacheCreateWithPPD: Unable to allocate _ppd_cache_t."); + goto create_error; + } + + /* + * Copy and convert size data... + */ + + if (ppd->num_sizes == 0) + { + DEBUG_puts("_ppdCacheCreateWithPPD: No page sizes in PPD."); + goto create_error; + } + + if ((pc->sizes = calloc(ppd->num_sizes, sizeof(_pwg_size_t))) == NULL) + { + DEBUG_printf(("_ppdCacheCreateWithPPD: Unable to allocate %d " + "_pwg_size_t's.", ppd->num_sizes)); + goto create_error; + } + + for (i = ppd->num_sizes, pwg_size = pc->sizes, ppd_size = ppd->sizes; + i > 0; + i --, ppd_size ++) + { + /* + * Don't copy over custom size... + */ + + if (!_cups_strcasecmp(ppd_size->name, "Custom")) + continue; + + /* + * Convert the PPD size name to the corresponding PWG keyword name. + */ + + if ((pwg_media = _pwgMediaForPPD(ppd_size->name)) != NULL) + { + /* + * Standard name, do we have conflicts? + */ + + for (j = 0; j < pc->num_sizes; j ++) + if (!strcmp(pc->sizes[j].map.pwg, pwg_media->pwg)) + { + pwg_media = NULL; + break; + } + } + + if (pwg_media) + { + /* + * Standard name and no conflicts, use it! + */ + + pwg_name = pwg_media->pwg; + new_known_pwg = 1; + } + else + { + /* + * Not a standard name; convert it to a PWG vendor name of the form: + * + * pp_lowerppd_WIDTHxHEIGHTuu + */ + + pwg_name = pwg_keyword; + new_known_pwg = 0; + + pwg_unppdize_name(ppd_size->name, ppd_name, sizeof(ppd_name)); + _pwgGenerateSize(pwg_keyword, sizeof(pwg_keyword), NULL, ppd_name, + _PWG_FROMPTS(ppd_size->width), + _PWG_FROMPTS(ppd_size->length)); + } + + /* + * If we have a similar paper with non-zero margins then we only + * want to keep it if it has a larger imageable area length. + */ + + pwg_media = _pwgMediaForSize(_PWG_FROMPTS(ppd_size->width), + _PWG_FROMPTS(ppd_size->length)); + new_width = pwg_media->width; + new_length = pwg_media->length; + new_left = _PWG_FROMPTS(ppd_size->left); + new_bottom = _PWG_FROMPTS(ppd_size->bottom); + new_right = _PWG_FROMPTS(ppd_size->width - ppd_size->right); + new_top = _PWG_FROMPTS(ppd_size->length - ppd_size->top); + new_imageable = new_length - new_top - new_bottom; + new_borderless = new_bottom == 0 && new_top == 0 && + new_left == 0 && new_right == 0; + + for (k = pc->num_sizes, similar = 0, old_size = pc->sizes, new_size = NULL; + k > 0 && !similar; + k --, old_size ++) + { + old_imageable = old_size->length - old_size->top - old_size->bottom; + old_borderless = old_size->left == 0 && old_size->bottom == 0 && + old_size->right == 0 && old_size->top == 0; + old_known_pwg = strncmp(old_size->map.pwg, "oe_", 3) && + strncmp(old_size->map.pwg, "om_", 3); + + similar = old_borderless == new_borderless && + _PWG_EQUIVALENT(old_size->width, new_width) && + _PWG_EQUIVALENT(old_size->length, new_length); + + if (similar && + (new_known_pwg || (!old_known_pwg && new_imageable > old_imageable))) + { + /* + * The new paper has a larger imageable area so it could replace + * the older paper. Regardless of the imageable area, we always + * prefer the size with a well-known PWG name. + */ + + new_size = old_size; + _cupsStrFree(old_size->map.ppd); + _cupsStrFree(old_size->map.pwg); + } + } + + if (!similar) + { + /* + * The paper was unique enough to deserve its own entry so add it to the + * end. + */ + + new_size = pwg_size ++; + pc->num_sizes ++; + } + + if (new_size) + { + /* + * Save this size... + */ + + new_size->map.ppd = _cupsStrAlloc(ppd_size->name); + new_size->map.pwg = _cupsStrAlloc(pwg_name); + new_size->width = new_width; + new_size->length = new_length; + new_size->left = new_left; + new_size->bottom = new_bottom; + new_size->right = new_right; + new_size->top = new_top; + } + } + + if (ppd->variable_sizes) + { + /* + * Generate custom size data... + */ + + _pwgGenerateSize(pwg_keyword, sizeof(pwg_keyword), "custom", "max", + _PWG_FROMPTS(ppd->custom_max[0]), + _PWG_FROMPTS(ppd->custom_max[1])); + pc->custom_max_keyword = _cupsStrAlloc(pwg_keyword); + pc->custom_max_width = _PWG_FROMPTS(ppd->custom_max[0]); + pc->custom_max_length = _PWG_FROMPTS(ppd->custom_max[1]); + + _pwgGenerateSize(pwg_keyword, sizeof(pwg_keyword), "custom", "min", + _PWG_FROMPTS(ppd->custom_min[0]), + _PWG_FROMPTS(ppd->custom_min[1])); + pc->custom_min_keyword = _cupsStrAlloc(pwg_keyword); + pc->custom_min_width = _PWG_FROMPTS(ppd->custom_min[0]); + pc->custom_min_length = _PWG_FROMPTS(ppd->custom_min[1]); + + pc->custom_size.left = _PWG_FROMPTS(ppd->custom_margins[0]); + pc->custom_size.bottom = _PWG_FROMPTS(ppd->custom_margins[1]); + pc->custom_size.right = _PWG_FROMPTS(ppd->custom_margins[2]); + pc->custom_size.top = _PWG_FROMPTS(ppd->custom_margins[3]); + } + + /* + * Copy and convert InputSlot data... + */ + + if ((input_slot = ppdFindOption(ppd, "InputSlot")) == NULL) + input_slot = ppdFindOption(ppd, "HPPaperSource"); + + if (input_slot) + { + pc->source_option = _cupsStrAlloc(input_slot->keyword); + + if ((pc->sources = calloc(input_slot->num_choices, + sizeof(_pwg_map_t))) == NULL) + { + DEBUG_printf(("_ppdCacheCreateWithPPD: Unable to allocate %d " + "_pwg_map_t's for InputSlot.", input_slot->num_choices)); + goto create_error; + } + + pc->num_sources = input_slot->num_choices; + + for (i = input_slot->num_choices, choice = input_slot->choices, + map = pc->sources; + i > 0; + i --, choice ++, map ++) + { + if (!_cups_strncasecmp(choice->choice, "Auto", 4) || + !_cups_strcasecmp(choice->choice, "Default")) + pwg_name = "auto"; + else if (!_cups_strcasecmp(choice->choice, "Cassette")) + pwg_name = "main"; + else if (!_cups_strcasecmp(choice->choice, "PhotoTray")) + pwg_name = "photo"; + else if (!_cups_strcasecmp(choice->choice, "CDTray")) + pwg_name = "disc"; + else if (!_cups_strncasecmp(choice->choice, "Multipurpose", 12) || + !_cups_strcasecmp(choice->choice, "MP") || + !_cups_strcasecmp(choice->choice, "MPTray")) + pwg_name = "by-pass-tray"; + else if (!_cups_strcasecmp(choice->choice, "LargeCapacity")) + pwg_name = "large-capacity"; + else if (!_cups_strncasecmp(choice->choice, "Lower", 5)) + pwg_name = "bottom"; + else if (!_cups_strncasecmp(choice->choice, "Middle", 6)) + pwg_name = "middle"; + else if (!_cups_strncasecmp(choice->choice, "Upper", 5)) + pwg_name = "top"; + else if (!_cups_strncasecmp(choice->choice, "Side", 4)) + pwg_name = "side"; + else if (!_cups_strcasecmp(choice->choice, "Roll")) + pwg_name = "main-roll"; + else + { + /* + * Convert PPD name to lowercase... + */ + + pwg_name = pwg_keyword; + pwg_unppdize_name(choice->choice, pwg_keyword, sizeof(pwg_keyword)); + } + + map->pwg = _cupsStrAlloc(pwg_name); + map->ppd = _cupsStrAlloc(choice->choice); + } + } + + /* + * Copy and convert MediaType data... + */ + + if ((media_type = ppdFindOption(ppd, "MediaType")) != NULL) + { + if ((pc->types = calloc(media_type->num_choices, + sizeof(_pwg_map_t))) == NULL) + { + DEBUG_printf(("_ppdCacheCreateWithPPD: Unable to allocate %d " + "_pwg_map_t's for MediaType.", media_type->num_choices)); + goto create_error; + } + + pc->num_types = media_type->num_choices; + + for (i = media_type->num_choices, choice = media_type->choices, + map = pc->types; + i > 0; + i --, choice ++, map ++) + { + if (!_cups_strncasecmp(choice->choice, "Auto", 4) || + !_cups_strcasecmp(choice->choice, "Any") || + !_cups_strcasecmp(choice->choice, "Default")) + pwg_name = "auto"; + else if (!_cups_strncasecmp(choice->choice, "Card", 4)) + pwg_name = "cardstock"; + else if (!_cups_strncasecmp(choice->choice, "Env", 3)) + pwg_name = "envelope"; + else if (!_cups_strncasecmp(choice->choice, "Gloss", 5)) + pwg_name = "photographic-glossy"; + else if (!_cups_strcasecmp(choice->choice, "HighGloss")) + pwg_name = "photographic-high-gloss"; + else if (!_cups_strcasecmp(choice->choice, "Matte")) + pwg_name = "photographic-matte"; + else if (!_cups_strncasecmp(choice->choice, "Plain", 5)) + pwg_name = "stationery"; + else if (!_cups_strncasecmp(choice->choice, "Coated", 6)) + pwg_name = "stationery-coated"; + else if (!_cups_strcasecmp(choice->choice, "Inkjet")) + pwg_name = "stationery-inkjet"; + else if (!_cups_strcasecmp(choice->choice, "Letterhead")) + pwg_name = "stationery-letterhead"; + else if (!_cups_strncasecmp(choice->choice, "Preprint", 8)) + pwg_name = "stationery-preprinted"; + else if (!_cups_strcasecmp(choice->choice, "Recycled")) + pwg_name = "stationery-recycled"; + else if (!_cups_strncasecmp(choice->choice, "Transparen", 10)) + pwg_name = "transparency"; + else + { + /* + * Convert PPD name to lowercase... + */ + + pwg_name = pwg_keyword; + pwg_unppdize_name(choice->choice, pwg_keyword, sizeof(pwg_keyword)); + } + + map->pwg = _cupsStrAlloc(pwg_name); + map->ppd = _cupsStrAlloc(choice->choice); + } + } + + /* + * Copy and convert OutputBin data... + */ + + if ((output_bin = ppdFindOption(ppd, "OutputBin")) != NULL) + { + if ((pc->bins = calloc(output_bin->num_choices, + sizeof(_pwg_map_t))) == NULL) + { + DEBUG_printf(("_ppdCacheCreateWithPPD: Unable to allocate %d " + "_pwg_map_t's for OutputBin.", output_bin->num_choices)); + goto create_error; + } + + pc->num_bins = output_bin->num_choices; + + for (i = output_bin->num_choices, choice = output_bin->choices, + map = pc->bins; + i > 0; + i --, choice ++, map ++) + { + pwg_unppdize_name(choice->choice, pwg_keyword, sizeof(pwg_keyword)); + + map->pwg = _cupsStrAlloc(pwg_keyword); + map->ppd = _cupsStrAlloc(choice->choice); + } + } + + if ((ppd_attr = ppdFindAttr(ppd, "APPrinterPreset", NULL)) != NULL) + { + /* + * Copy and convert APPrinterPreset (output-mode + print-quality) data... + */ + + const char *quality, /* com.apple.print.preset.quality value */ + *output_mode, /* com.apple.print.preset.output-mode value */ + *color_model_val, /* ColorModel choice */ + *graphicsType, /* com.apple.print.preset.graphicsType value */ + *media_front_coating; /* com.apple.print.preset.media-front-coating value */ + + do + { + num_options = _ppdParseOptions(ppd_attr->value, 0, &options, + _PPD_PARSE_ALL); + + if ((quality = cupsGetOption("com.apple.print.preset.quality", + num_options, options)) != NULL) + { + /* + * Get the print-quality for this preset... + */ + + if (!strcmp(quality, "low")) + pwg_print_quality = _PWG_PRINT_QUALITY_DRAFT; + else if (!strcmp(quality, "high")) + pwg_print_quality = _PWG_PRINT_QUALITY_HIGH; + else + pwg_print_quality = _PWG_PRINT_QUALITY_NORMAL; + + /* + * Ignore graphicsType "Photo" presets that are not high quality. + */ + + graphicsType = cupsGetOption("com.apple.print.preset.graphicsType", + num_options, options); + + if (pwg_print_quality != _PWG_PRINT_QUALITY_HIGH && graphicsType && + !strcmp(graphicsType, "Photo")) + continue; + + /* + * Ignore presets for normal and draft quality where the coating + * isn't "none" or "autodetect". + */ + + media_front_coating = cupsGetOption( + "com.apple.print.preset.media-front-coating", + num_options, options); + + if (pwg_print_quality != _PWG_PRINT_QUALITY_HIGH && + media_front_coating && + strcmp(media_front_coating, "none") && + strcmp(media_front_coating, "autodetect")) + continue; + + /* + * Get the output mode for this preset... + */ + + output_mode = cupsGetOption("com.apple.print.preset.output-mode", + num_options, options); + color_model_val = cupsGetOption("ColorModel", num_options, options); + + if (output_mode) + { + if (!strcmp(output_mode, "monochrome")) + pwg_print_color_mode = _PWG_PRINT_COLOR_MODE_MONOCHROME; + else + pwg_print_color_mode = _PWG_PRINT_COLOR_MODE_COLOR; + } + else if (color_model_val) + { + if (!_cups_strcasecmp(color_model_val, "Gray")) + pwg_print_color_mode = _PWG_PRINT_COLOR_MODE_MONOCHROME; + else + pwg_print_color_mode = _PWG_PRINT_COLOR_MODE_COLOR; + } + else + pwg_print_color_mode = _PWG_PRINT_COLOR_MODE_COLOR; + + /* + * Save the options for this combination as needed... + */ + + if (!pc->num_presets[pwg_print_color_mode][pwg_print_quality]) + pc->num_presets[pwg_print_color_mode][pwg_print_quality] = + _ppdParseOptions(ppd_attr->value, 0, + pc->presets[pwg_print_color_mode] + + pwg_print_quality, _PPD_PARSE_OPTIONS); + } + + cupsFreeOptions(num_options, options); + } + while ((ppd_attr = ppdFindNextAttr(ppd, "APPrinterPreset", NULL)) != NULL); + } + + if (!pc->num_presets[_PWG_PRINT_COLOR_MODE_MONOCHROME][_PWG_PRINT_QUALITY_DRAFT] && + !pc->num_presets[_PWG_PRINT_COLOR_MODE_MONOCHROME][_PWG_PRINT_QUALITY_NORMAL] && + !pc->num_presets[_PWG_PRINT_COLOR_MODE_MONOCHROME][_PWG_PRINT_QUALITY_HIGH]) + { + /* + * Try adding some common color options to create grayscale presets. These + * are listed in order of popularity... + */ + + const char *color_option = NULL, /* Color control option */ + *gray_choice = NULL; /* Choice to select grayscale */ + + if ((color_model = ppdFindOption(ppd, "ColorModel")) != NULL && + ppdFindChoice(color_model, "Gray")) + { + color_option = "ColorModel"; + gray_choice = "Gray"; + } + else if ((color_model = ppdFindOption(ppd, "HPColorMode")) != NULL && + ppdFindChoice(color_model, "grayscale")) + { + color_option = "HPColorMode"; + gray_choice = "grayscale"; + } + else if ((color_model = ppdFindOption(ppd, "BRMonoColor")) != NULL && + ppdFindChoice(color_model, "Mono")) + { + color_option = "BRMonoColor"; + gray_choice = "Mono"; + } + else if ((color_model = ppdFindOption(ppd, "CNIJSGrayScale")) != NULL && + ppdFindChoice(color_model, "1")) + { + color_option = "CNIJSGrayScale"; + gray_choice = "1"; + } + else if ((color_model = ppdFindOption(ppd, "HPColorAsGray")) != NULL && + ppdFindChoice(color_model, "True")) + { + color_option = "HPColorAsGray"; + gray_choice = "True"; + } + + if (color_option && gray_choice) + { + /* + * Copy and convert ColorModel (output-mode) data... + */ + + cups_option_t *coption, /* Color option */ + *moption; /* Monochrome option */ + + for (pwg_print_quality = _PWG_PRINT_QUALITY_DRAFT; + pwg_print_quality < _PWG_PRINT_QUALITY_MAX; + pwg_print_quality ++) + { + if (pc->num_presets[_PWG_PRINT_COLOR_MODE_COLOR][pwg_print_quality]) + { + /* + * Copy the color options... + */ + + num_options = pc->num_presets[_PWG_PRINT_COLOR_MODE_COLOR] + [pwg_print_quality]; + options = calloc(sizeof(cups_option_t), num_options); + + if (options) + { + for (i = num_options, moption = options, + coption = pc->presets[_PWG_PRINT_COLOR_MODE_COLOR] + [pwg_print_quality]; + i > 0; + i --, moption ++, coption ++) + { + moption->name = _cupsStrRetain(coption->name); + moption->value = _cupsStrRetain(coption->value); + } + + pc->num_presets[_PWG_PRINT_COLOR_MODE_MONOCHROME][pwg_print_quality] = + num_options; + pc->presets[_PWG_PRINT_COLOR_MODE_MONOCHROME][pwg_print_quality] = + options; + } + } + else if (pwg_print_quality != _PWG_PRINT_QUALITY_NORMAL) + continue; + + /* + * Add the grayscale option to the preset... + */ + + pc->num_presets[_PWG_PRINT_COLOR_MODE_MONOCHROME][pwg_print_quality] = + cupsAddOption(color_option, gray_choice, + pc->num_presets[_PWG_PRINT_COLOR_MODE_MONOCHROME] + [pwg_print_quality], + pc->presets[_PWG_PRINT_COLOR_MODE_MONOCHROME] + + pwg_print_quality); + } + } + } + + /* + * Copy and convert Duplex (sides) data... + */ + + if ((duplex = ppdFindOption(ppd, "Duplex")) == NULL) + if ((duplex = ppdFindOption(ppd, "JCLDuplex")) == NULL) + if ((duplex = ppdFindOption(ppd, "EFDuplex")) == NULL) + if ((duplex = ppdFindOption(ppd, "EFDuplexing")) == NULL) + duplex = ppdFindOption(ppd, "KD03Duplex"); + + if (duplex) + { + pc->sides_option = _cupsStrAlloc(duplex->keyword); + + for (i = duplex->num_choices, choice = duplex->choices; + i > 0; + i --, choice ++) + { + if ((!_cups_strcasecmp(choice->choice, "None") || + !_cups_strcasecmp(choice->choice, "False")) && !pc->sides_1sided) + pc->sides_1sided = _cupsStrAlloc(choice->choice); + else if ((!_cups_strcasecmp(choice->choice, "DuplexNoTumble") || + !_cups_strcasecmp(choice->choice, "LongEdge") || + !_cups_strcasecmp(choice->choice, "Top")) && !pc->sides_2sided_long) + pc->sides_2sided_long = _cupsStrAlloc(choice->choice); + else if ((!_cups_strcasecmp(choice->choice, "DuplexTumble") || + !_cups_strcasecmp(choice->choice, "ShortEdge") || + !_cups_strcasecmp(choice->choice, "Bottom")) && + !pc->sides_2sided_short) + pc->sides_2sided_short = _cupsStrAlloc(choice->choice); + } + } + + /* + * Copy filters and pre-filters... + */ + + pc->filters = cupsArrayNew3(NULL, NULL, NULL, 0, + (cups_acopy_func_t)_cupsStrAlloc, + (cups_afree_func_t)_cupsStrFree); + + cupsArrayAdd(pc->filters, + "application/vnd.cups-raw application/octet-stream 0 -"); + + if ((ppd_attr = ppdFindAttr(ppd, "cupsFilter2", NULL)) != NULL) + { + do + { + cupsArrayAdd(pc->filters, ppd_attr->value); + } + while ((ppd_attr = ppdFindNextAttr(ppd, "cupsFilter2", NULL)) != NULL); + } + else if (ppd->num_filters > 0) + { + for (i = 0; i < ppd->num_filters; i ++) + cupsArrayAdd(pc->filters, ppd->filters[i]); + } + else + cupsArrayAdd(pc->filters, "application/vnd.cups-postscript 0 -"); + + /* + * See if we have a command filter... + */ + + for (filter = (const char *)cupsArrayFirst(pc->filters); + filter; + filter = (const char *)cupsArrayNext(pc->filters)) + if (!_cups_strncasecmp(filter, "application/vnd.cups-command", 28) && + _cups_isspace(filter[28])) + break; + + if (!filter && + ((ppd_attr = ppdFindAttr(ppd, "cupsCommands", NULL)) == NULL || + _cups_strcasecmp(ppd_attr->value, "none"))) + { + /* + * No command filter and no cupsCommands keyword telling us not to use one. + * See if this is a PostScript printer, and if so add a PostScript command + * filter... + */ + + for (filter = (const char *)cupsArrayFirst(pc->filters); + filter; + filter = (const char *)cupsArrayNext(pc->filters)) + if (!_cups_strncasecmp(filter, "application/vnd.cups-postscript", 31) && + _cups_isspace(filter[31])) + break; + + if (filter) + cupsArrayAdd(pc->filters, + "application/vnd.cups-command application/postscript 100 " + "commandtops"); + } + + if ((ppd_attr = ppdFindAttr(ppd, "cupsPreFilter", NULL)) != NULL) + { + pc->prefilters = cupsArrayNew3(NULL, NULL, NULL, 0, + (cups_acopy_func_t)_cupsStrAlloc, + (cups_afree_func_t)_cupsStrFree); + + do + { + cupsArrayAdd(pc->prefilters, ppd_attr->value); + } + while ((ppd_attr = ppdFindNextAttr(ppd, "cupsPreFilter", NULL)) != NULL); + } + + if ((ppd_attr = ppdFindAttr(ppd, "cupsSingleFile", NULL)) != NULL) + pc->single_file = !_cups_strcasecmp(ppd_attr->value, "true"); + + /* + * Copy the product string, if any... + */ + + if (ppd->product) + pc->product = _cupsStrAlloc(ppd->product); + + /* + * Copy finishings mapping data... + */ + + if ((ppd_attr = ppdFindAttr(ppd, "cupsIPPFinishings", NULL)) != NULL) + { + pc->finishings = cupsArrayNew3((cups_array_func_t)pwg_compare_finishings, + NULL, NULL, 0, NULL, + (cups_afree_func_t)pwg_free_finishings); + + do + { + if ((finishings = calloc(1, sizeof(_pwg_finishings_t))) == NULL) + goto create_error; + + finishings->value = atoi(ppd_attr->spec); + finishings->num_options = _ppdParseOptions(ppd_attr->value, 0, + &(finishings->options), + _PPD_PARSE_OPTIONS); + + cupsArrayAdd(pc->finishings, finishings); + } + while ((ppd_attr = ppdFindNextAttr(ppd, "cupsIPPFinishings", + NULL)) != NULL); + } + + /* + * Return the cache data... + */ + + return (pc); + + /* + * If we get here we need to destroy the PWG mapping data and return NULL... + */ + + create_error: + + _cupsSetError(IPP_INTERNAL_ERROR, _("Out of memory."), 1); + _ppdCacheDestroy(pc); + + return (NULL); +} + + +/* + * '_ppdCacheDestroy()' - Free all memory used for PWG mapping data. + */ + +void +_ppdCacheDestroy(_ppd_cache_t *pc) /* I - PPD cache and mapping data */ +{ + int i; /* Looping var */ + _pwg_map_t *map; /* Current map */ + _pwg_size_t *size; /* Current size */ + + + /* + * Range check input... + */ + + if (!pc) + return; + + /* + * Free memory as needed... + */ + + if (pc->bins) + { + for (i = pc->num_bins, map = pc->bins; i > 0; i --, map ++) + { + _cupsStrFree(map->pwg); + _cupsStrFree(map->ppd); + } + + free(pc->bins); + } + + if (pc->sizes) + { + for (i = pc->num_sizes, size = pc->sizes; i > 0; i --, size ++) + { + _cupsStrFree(size->map.pwg); + _cupsStrFree(size->map.ppd); + } + + free(pc->sizes); + } + + if (pc->source_option) + _cupsStrFree(pc->source_option); + + if (pc->sources) + { + for (i = pc->num_sources, map = pc->sources; i > 0; i --, map ++) + { + _cupsStrFree(map->pwg); + _cupsStrFree(map->ppd); + } + + free(pc->sources); + } + + if (pc->types) + { + for (i = pc->num_types, map = pc->types; i > 0; i --, map ++) + { + _cupsStrFree(map->pwg); + _cupsStrFree(map->ppd); + } + + free(pc->types); + } + + if (pc->custom_max_keyword) + _cupsStrFree(pc->custom_max_keyword); + + if (pc->custom_min_keyword) + _cupsStrFree(pc->custom_min_keyword); + + _cupsStrFree(pc->product); + cupsArrayDelete(pc->filters); + cupsArrayDelete(pc->prefilters); + cupsArrayDelete(pc->finishings); + + free(pc); +} + + +/* + * '_ppdCacheGetBin()' - Get the PWG output-bin keyword associated with a PPD + * OutputBin. + */ + +const char * /* O - output-bin or NULL */ +_ppdCacheGetBin( + _ppd_cache_t *pc, /* I - PPD cache and mapping data */ + const char *output_bin) /* I - PPD OutputBin string */ +{ + int i; /* Looping var */ + + + /* + * Range check input... + */ + + if (!pc || !output_bin) + return (NULL); + + /* + * Look up the OutputBin string... + */ + + + for (i = 0; i < pc->num_bins; i ++) + if (!_cups_strcasecmp(output_bin, pc->bins[i].ppd)) + return (pc->bins[i].pwg); + + return (NULL); +} + + +/* + * '_ppdCacheGetFinishingOptions()' - Get PPD finishing options for the given + * IPP finishings value(s). + */ + +int /* O - New number of options */ +_ppdCacheGetFinishingOptions( + _ppd_cache_t *pc, /* I - PPD cache and mapping data */ + ipp_t *job, /* I - Job attributes or NULL */ + ipp_finish_t value, /* I - IPP finishings value of IPP_FINISHINGS_NONE */ + int num_options, /* I - Number of options */ + cups_option_t **options) /* IO - Options */ +{ + int i; /* Looping var */ + _pwg_finishings_t *f, /* PWG finishings options */ + key; /* Search key */ + ipp_attribute_t *attr; /* Finishings attribute */ + cups_option_t *option; /* Current finishings option */ + + + /* + * Range check input... + */ + + if (!pc || cupsArrayCount(pc->finishings) == 0 || !options || + (!job && value == IPP_FINISHINGS_NONE)) + return (num_options); + + /* + * Apply finishing options... + */ + + if (job && (attr = ippFindAttribute(job, "finishings", IPP_TAG_ENUM)) != NULL) + { + int num_values = ippGetCount(attr); /* Number of values */ + + for (i = 0; i < num_values; i ++) + { + key.value = ippGetInteger(attr, i); + + if ((f = cupsArrayFind(pc->finishings, &key)) != NULL) + { + int j; /* Another looping var */ + + for (j = f->num_options, option = f->options; j > 0; j --, option ++) + num_options = cupsAddOption(option->name, option->value, + num_options, options); + } + } + } + else if (value != IPP_FINISHINGS_NONE) + { + key.value = value; + + if ((f = cupsArrayFind(pc->finishings, &key)) != NULL) + { + int j; /* Another looping var */ + + for (j = f->num_options, option = f->options; j > 0; j --, option ++) + num_options = cupsAddOption(option->name, option->value, + num_options, options); + } + } + + return (num_options); +} + + +/* + * '_ppdCacheGetFinishingValues()' - Get IPP finishings value(s) from the given + * PPD options. + */ + +int /* O - Number of finishings values */ +_ppdCacheGetFinishingValues( + _ppd_cache_t *pc, /* I - PPD cache and mapping data */ + int num_options, /* I - Number of options */ + cups_option_t *options, /* I - Options */ + int max_values, /* I - Maximum number of finishings values */ + int *values) /* O - Finishings values */ +{ + int i, /* Looping var */ + num_values = 0; /* Number of values */ + _pwg_finishings_t *f; /* Current finishings option */ + cups_option_t *option; /* Current option */ + const char *val; /* Value for option */ + + + /* + * Range check input... + */ + + if (!pc || !pc->finishings || num_options < 1 || max_values < 1 || !values) + return (0); + + /* + * Go through the finishings options and see what is set... + */ + + for (f = (_pwg_finishings_t *)cupsArrayFirst(pc->finishings); + f; + f = (_pwg_finishings_t *)cupsArrayNext(pc->finishings)) + { + for (i = f->num_options, option = f->options; i > 0; i --, option ++) + if ((val = cupsGetOption(option->name, num_options, options)) == NULL || + _cups_strcasecmp(option->value, val)) + break; + + if (i == 0) + { + values[num_values ++] = f->value; + + if (num_values >= max_values) + break; + } + } + + return (num_values); +} + + +/* + * '_ppdCacheGetInputSlot()' - Get the PPD InputSlot associated with the job + * attributes or a keyword string. + */ + +const char * /* O - PPD InputSlot or NULL */ +_ppdCacheGetInputSlot( + _ppd_cache_t *pc, /* I - PPD cache and mapping data */ + ipp_t *job, /* I - Job attributes or NULL */ + const char *keyword) /* I - Keyword string or NULL */ +{ + /* + * Range check input... + */ + + if (!pc || pc->num_sources == 0 || (!job && !keyword)) + return (NULL); + + if (job && !keyword) + { + /* + * Lookup the media-col attribute and any media-source found there... + */ + + ipp_attribute_t *media_col, /* media-col attribute */ + *media_source; /* media-source attribute */ + _pwg_size_t size; /* Dimensional size */ + int margins_set; /* Were the margins set? */ + + media_col = ippFindAttribute(job, "media-col", IPP_TAG_BEGIN_COLLECTION); + if (media_col && + (media_source = ippFindAttribute(ippGetCollection(media_col, 0), + "media-source", + IPP_TAG_KEYWORD)) != NULL) + { + /* + * Use the media-source value from media-col... + */ + + keyword = ippGetString(media_source, 0, NULL); + } + else if (_pwgInitSize(&size, job, &margins_set)) + { + /* + * For media <= 5x7, look for a photo tray... + */ + + if (size.width <= (5 * 2540) && size.length <= (7 * 2540)) + keyword = "photo"; + } + } + + if (keyword) + { + int i; /* Looping var */ + + for (i = 0; i < pc->num_sources; i ++) + if (!_cups_strcasecmp(keyword, pc->sources[i].pwg)) + return (pc->sources[i].ppd); + } + + return (NULL); +} + + +/* + * '_ppdCacheGetMediaType()' - Get the PPD MediaType associated with the job + * attributes or a keyword string. + */ + +const char * /* O - PPD MediaType or NULL */ +_ppdCacheGetMediaType( + _ppd_cache_t *pc, /* I - PPD cache and mapping data */ + ipp_t *job, /* I - Job attributes or NULL */ + const char *keyword) /* I - Keyword string or NULL */ +{ + /* + * Range check input... + */ + + if (!pc || pc->num_types == 0 || (!job && !keyword)) + return (NULL); + + if (job && !keyword) + { + /* + * Lookup the media-col attribute and any media-source found there... + */ + + ipp_attribute_t *media_col, /* media-col attribute */ + *media_type; /* media-type attribute */ + + media_col = ippFindAttribute(job, "media-col", IPP_TAG_BEGIN_COLLECTION); + if (media_col) + { + if ((media_type = ippFindAttribute(media_col->values[0].collection, + "media-type", + IPP_TAG_KEYWORD)) == NULL) + media_type = ippFindAttribute(media_col->values[0].collection, + "media-type", IPP_TAG_NAME); + + if (media_type) + keyword = media_type->values[0].string.text; + } + } + + if (keyword) + { + int i; /* Looping var */ + + for (i = 0; i < pc->num_types; i ++) + if (!_cups_strcasecmp(keyword, pc->types[i].pwg)) + return (pc->types[i].ppd); + } + + return (NULL); +} + + +/* + * '_ppdCacheGetOutputBin()' - Get the PPD OutputBin associated with the keyword + * string. + */ + +const char * /* O - PPD OutputBin or NULL */ +_ppdCacheGetOutputBin( + _ppd_cache_t *pc, /* I - PPD cache and mapping data */ + const char *output_bin) /* I - Keyword string */ +{ + int i; /* Looping var */ + + + /* + * Range check input... + */ + + if (!pc || !output_bin) + return (NULL); + + /* + * Look up the OutputBin string... + */ + + + for (i = 0; i < pc->num_bins; i ++) + if (!_cups_strcasecmp(output_bin, pc->bins[i].pwg)) + return (pc->bins[i].ppd); + + return (NULL); +} + + +/* + * '_ppdCacheGetPageSize()' - Get the PPD PageSize associated with the job + * attributes or a keyword string. + */ + +const char * /* O - PPD PageSize or NULL */ +_ppdCacheGetPageSize( + _ppd_cache_t *pc, /* I - PPD cache and mapping data */ + ipp_t *job, /* I - Job attributes or NULL */ + const char *keyword, /* I - Keyword string or NULL */ + int *exact) /* O - 1 if exact match, 0 otherwise */ +{ + int i; /* Looping var */ + _pwg_size_t *size, /* Current size */ + *closest, /* Closest size */ + jobsize; /* Size data from job */ + int margins_set, /* Were the margins set? */ + dwidth, /* Difference in width */ + dlength, /* Difference in length */ + dleft, /* Difference in left margins */ + dright, /* Difference in right margins */ + dbottom, /* Difference in bottom margins */ + dtop, /* Difference in top margins */ + dmin, /* Minimum difference */ + dclosest; /* Closest difference */ + const char *ppd_name; /* PPD media name */ + + + DEBUG_printf(("_ppdCacheGetPageSize(pc=%p, job=%p, keyword=\"%s\", exact=%p)", + pc, job, keyword, exact)); + + /* + * Range check input... + */ + + if (!pc || (!job && !keyword)) + return (NULL); + + if (exact) + *exact = 0; + + ppd_name = keyword; + + if (job) + { + /* + * Try getting the PPD media name from the job attributes... + */ + + ipp_attribute_t *attr; /* Job attribute */ + + if ((attr = ippFindAttribute(job, "PageSize", IPP_TAG_ZERO)) == NULL) + if ((attr = ippFindAttribute(job, "PageRegion", IPP_TAG_ZERO)) == NULL) + attr = ippFindAttribute(job, "media", IPP_TAG_ZERO); + +#ifdef DEBUG + if (attr) + DEBUG_printf(("1_ppdCacheGetPageSize: Found attribute %s (%s)", + attr->name, ippTagString(attr->value_tag))); + else + DEBUG_puts("1_ppdCacheGetPageSize: Did not find media attribute."); +#endif /* DEBUG */ + + if (attr && (attr->value_tag == IPP_TAG_NAME || + attr->value_tag == IPP_TAG_KEYWORD)) + ppd_name = attr->values[0].string.text; + } + + DEBUG_printf(("1_ppdCacheGetPageSize: ppd_name=\"%s\"", ppd_name)); + + if (ppd_name) + { + /* + * Try looking up the named PPD size first... + */ + + for (i = pc->num_sizes, size = pc->sizes; i > 0; i --, size ++) + { + DEBUG_printf(("2_ppdCacheGetPageSize: size[%d]=[\"%s\" \"%s\"]", + (int)(size - pc->sizes), size->map.pwg, size->map.ppd)); + + if (!_cups_strcasecmp(ppd_name, size->map.ppd) || + !_cups_strcasecmp(ppd_name, size->map.pwg)) + { + if (exact) + *exact = 1; + + DEBUG_printf(("1_ppdCacheGetPageSize: Returning \"%s\"", ppd_name)); + + return (size->map.ppd); + } + } + } + + if (job && !keyword) + { + /* + * Get the size using media-col or media, with the preference being + * media-col. + */ + + if (!_pwgInitSize(&jobsize, job, &margins_set)) + return (NULL); + } + else + { + /* + * Get the size using a media keyword... + */ + + _pwg_media_t *media; /* Media definition */ + + + if ((media = _pwgMediaForPWG(keyword)) == NULL) + if ((media = _pwgMediaForLegacy(keyword)) == NULL) + if ((media = _pwgMediaForPPD(keyword)) == NULL) + return (NULL); + + jobsize.width = media->width; + jobsize.length = media->length; + margins_set = 0; + } + + /* + * Now that we have the dimensions and possibly the margins, look at the + * available sizes and find the match... + */ + + closest = NULL; + dclosest = 999999999; + + if (!ppd_name || _cups_strncasecmp(ppd_name, "Custom.", 7) || + _cups_strncasecmp(ppd_name, "custom_", 7)) + { + for (i = pc->num_sizes, size = pc->sizes; i > 0; i --, size ++) + { + /* + * Adobe uses a size matching algorithm with an epsilon of 5 points, which + * is just about 176/2540ths... + */ + + dwidth = size->width - jobsize.width; + dlength = size->length - jobsize.length; + + if (dwidth <= -176 || dwidth >= 176 || dlength <= -176 || dlength >= 176) + continue; + + if (margins_set) + { + /* + * Use a tighter epsilon of 1 point (35/2540ths) for margins... + */ + + dleft = size->left - jobsize.left; + dright = size->right - jobsize.right; + dtop = size->top - jobsize.top; + dbottom = size->bottom - jobsize.bottom; + + if (dleft <= -35 || dleft >= 35 || dright <= -35 || dright >= 35 || + dtop <= -35 || dtop >= 35 || dbottom <= -35 || dbottom >= 35) + { + dleft = dleft < 0 ? -dleft : dleft; + dright = dright < 0 ? -dright : dright; + dbottom = dbottom < 0 ? -dbottom : dbottom; + dtop = dtop < 0 ? -dtop : dtop; + dmin = dleft + dright + dbottom + dtop; + + if (dmin < dclosest) + { + dclosest = dmin; + closest = size; + } + + continue; + } + } + + if (exact) + *exact = 1; + + DEBUG_printf(("1_ppdCacheGetPageSize: Returning \"%s\"", size->map.ppd)); + + return (size->map.ppd); + } + } + + if (closest) + { + DEBUG_printf(("1_ppdCacheGetPageSize: Returning \"%s\" (closest)", + closest->map.ppd)); + + return (closest->map.ppd); + } + + /* + * If we get here we need to check for custom page size support... + */ + + if (jobsize.width >= pc->custom_min_width && + jobsize.width <= pc->custom_max_width && + jobsize.length >= pc->custom_min_length && + jobsize.length <= pc->custom_max_length) + { + /* + * In range, format as Custom.WWWWxLLLL (points). + */ + + snprintf(pc->custom_ppd_size, sizeof(pc->custom_ppd_size), "Custom.%dx%d", + (int)_PWG_TOPTS(jobsize.width), (int)_PWG_TOPTS(jobsize.length)); + + if (margins_set && exact) + { + dleft = pc->custom_size.left - jobsize.left; + dright = pc->custom_size.right - jobsize.right; + dtop = pc->custom_size.top - jobsize.top; + dbottom = pc->custom_size.bottom - jobsize.bottom; + + if (dleft > -35 && dleft < 35 && dright > -35 && dright < 35 && + dtop > -35 && dtop < 35 && dbottom > -35 && dbottom < 35) + *exact = 1; + } + else if (exact) + *exact = 1; + + DEBUG_printf(("1_ppdCacheGetPageSize: Returning \"%s\" (custom)", + pc->custom_ppd_size)); + + return (pc->custom_ppd_size); + } + + /* + * No custom page size support or the size is out of range - return NULL. + */ + + DEBUG_puts("1_ppdCacheGetPageSize: Returning NULL"); + + return (NULL); +} + + +/* + * '_ppdCacheGetSize()' - Get the PWG size associated with a PPD PageSize. + */ + +_pwg_size_t * /* O - PWG size or NULL */ +_ppdCacheGetSize( + _ppd_cache_t *pc, /* I - PPD cache and mapping data */ + const char *page_size) /* I - PPD PageSize */ +{ + int i; /* Looping var */ + _pwg_media_t *media; /* Media */ + _pwg_size_t *size; /* Current size */ + + + /* + * Range check input... + */ + + if (!pc || !page_size) + return (NULL); + + if (!_cups_strncasecmp(page_size, "Custom.", 7)) + { + /* + * Custom size; size name can be one of the following: + * + * Custom.WIDTHxLENGTHin - Size in inches + * Custom.WIDTHxLENGTHft - Size in feet + * Custom.WIDTHxLENGTHcm - Size in centimeters + * Custom.WIDTHxLENGTHmm - Size in millimeters + * Custom.WIDTHxLENGTHm - Size in meters + * Custom.WIDTHxLENGTH[pt] - Size in points + */ + + double w, l; /* Width and length of page */ + char *ptr; /* Pointer into PageSize */ + struct lconv *loc; /* Locale data */ + + loc = localeconv(); + w = (float)_cupsStrScand(page_size + 7, &ptr, loc); + if (!ptr || *ptr != 'x') + return (NULL); + + l = (float)_cupsStrScand(ptr + 1, &ptr, loc); + if (!ptr) + return (NULL); + + if (!_cups_strcasecmp(ptr, "in")) + { + w *= 2540.0; + l *= 2540.0; + } + else if (!_cups_strcasecmp(ptr, "ft")) + { + w *= 12.0 * 2540.0; + l *= 12.0 * 2540.0; + } + else if (!_cups_strcasecmp(ptr, "mm")) + { + w *= 100.0; + l *= 100.0; + } + else if (!_cups_strcasecmp(ptr, "cm")) + { + w *= 1000.0; + l *= 1000.0; + } + else if (!_cups_strcasecmp(ptr, "m")) + { + w *= 100000.0; + l *= 100000.0; + } + else + { + w *= 2540.0 / 72.0; + l *= 2540.0 / 72.0; + } + + pc->custom_size.width = (int)w; + pc->custom_size.length = (int)l; + + return (&(pc->custom_size)); + } + + /* + * Not a custom size - look it up... + */ + + for (i = pc->num_sizes, size = pc->sizes; i > 0; i --, size ++) + if (!_cups_strcasecmp(page_size, size->map.ppd) || + !_cups_strcasecmp(page_size, size->map.pwg)) + return (size); + + /* + * Look up standard sizes... + */ + + if ((media = _pwgMediaForPPD(page_size)) == NULL) + if ((media = _pwgMediaForLegacy(page_size)) == NULL) + media = _pwgMediaForPWG(page_size); + + if (media) + { + pc->custom_size.width = media->width; + pc->custom_size.length = media->length; + + return (&(pc->custom_size)); + } + + return (NULL); +} + + +/* + * '_ppdCacheGetSource()' - Get the PWG media-source associated with a PPD + * InputSlot. + */ + +const char * /* O - PWG media-source keyword */ +_ppdCacheGetSource( + _ppd_cache_t *pc, /* I - PPD cache and mapping data */ + const char *input_slot) /* I - PPD InputSlot */ +{ + int i; /* Looping var */ + _pwg_map_t *source; /* Current source */ + + + /* + * Range check input... + */ + + if (!pc || !input_slot) + return (NULL); + + for (i = pc->num_sources, source = pc->sources; i > 0; i --, source ++) + if (!_cups_strcasecmp(input_slot, source->ppd)) + return (source->pwg); + + return (NULL); +} + + +/* + * '_ppdCacheGetType()' - Get the PWG media-type associated with a PPD + * MediaType. + */ + +const char * /* O - PWG media-type keyword */ +_ppdCacheGetType( + _ppd_cache_t *pc, /* I - PPD cache and mapping data */ + const char *media_type) /* I - PPD MediaType */ +{ + int i; /* Looping var */ + _pwg_map_t *type; /* Current type */ + + + /* + * Range check input... + */ + + if (!pc || !media_type) + return (NULL); + + for (i = pc->num_types, type = pc->types; i > 0; i --, type ++) + if (!_cups_strcasecmp(media_type, type->ppd)) + return (type->pwg); + + return (NULL); +} + + +/* + * '_ppdCacheWriteFile()' - Write PWG mapping data to a file. + */ + +int /* O - 1 on success, 0 on failure */ +_ppdCacheWriteFile( + _ppd_cache_t *pc, /* I - PPD cache and mapping data */ + const char *filename, /* I - File to write */ + ipp_t *attrs) /* I - Attributes to write, if any */ +{ + int i, j, k; /* Looping vars */ + cups_file_t *fp; /* Output file */ + _pwg_size_t *size; /* Current size */ + _pwg_map_t *map; /* Current map */ + _pwg_finishings_t *f; /* Current finishing option */ + cups_option_t *option; /* Current option */ + const char *value; /* Filter/pre-filter value */ + char newfile[1024]; /* New filename */ + + + /* + * Range check input... + */ + + if (!pc || !filename) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + return (0); + } + + /* + * Open the file and write with compression... + */ + + snprintf(newfile, sizeof(newfile), "%s.N", filename); + if ((fp = cupsFileOpen(newfile, "w9")) == NULL) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(errno), 0); + return (0); + } + + /* + * Standard header... + */ + + cupsFilePrintf(fp, "#CUPS-PPD-CACHE-%d\n", _PPD_CACHE_VERSION); + + /* + * Output bins... + */ + + if (pc->num_bins > 0) + { + cupsFilePrintf(fp, "NumBins %d\n", pc->num_bins); + for (i = pc->num_bins, map = pc->bins; i > 0; i --, map ++) + cupsFilePrintf(fp, "Bin %s %s\n", map->pwg, map->ppd); + } + + /* + * Media sizes... + */ + + cupsFilePrintf(fp, "NumSizes %d\n", pc->num_sizes); + for (i = pc->num_sizes, size = pc->sizes; i > 0; i --, size ++) + cupsFilePrintf(fp, "Size %s %s %d %d %d %d %d %d\n", size->map.pwg, + size->map.ppd, size->width, size->length, size->left, + size->bottom, size->right, size->top); + if (pc->custom_max_width > 0) + cupsFilePrintf(fp, "CustomSize %d %d %d %d %d %d %d %d\n", + pc->custom_max_width, pc->custom_max_length, + pc->custom_min_width, pc->custom_min_length, + pc->custom_size.left, pc->custom_size.bottom, + pc->custom_size.right, pc->custom_size.top); + + /* + * Media sources... + */ + + if (pc->source_option) + cupsFilePrintf(fp, "SourceOption %s\n", pc->source_option); + + if (pc->num_sources > 0) + { + cupsFilePrintf(fp, "NumSources %d\n", pc->num_sources); + for (i = pc->num_sources, map = pc->sources; i > 0; i --, map ++) + cupsFilePrintf(fp, "Source %s %s\n", map->pwg, map->ppd); + } + + /* + * Media types... + */ + + if (pc->num_types > 0) + { + cupsFilePrintf(fp, "NumTypes %d\n", pc->num_types); + for (i = pc->num_types, map = pc->types; i > 0; i --, map ++) + cupsFilePrintf(fp, "Type %s %s\n", map->pwg, map->ppd); + } + + /* + * Presets... + */ + + for (i = _PWG_PRINT_COLOR_MODE_MONOCHROME; i < _PWG_PRINT_COLOR_MODE_MAX; i ++) + for (j = _PWG_PRINT_QUALITY_DRAFT; j < _PWG_PRINT_QUALITY_MAX; j ++) + if (pc->num_presets[i][j]) + { + cupsFilePrintf(fp, "Preset %d %d", i, j); + for (k = pc->num_presets[i][j], option = pc->presets[i][j]; + k > 0; + k --, option ++) + cupsFilePrintf(fp, " %s=%s", option->name, option->value); + cupsFilePutChar(fp, '\n'); + } + + /* + * Duplex/sides... + */ + + if (pc->sides_option) + cupsFilePrintf(fp, "SidesOption %s\n", pc->sides_option); + + if (pc->sides_1sided) + cupsFilePrintf(fp, "Sides1Sided %s\n", pc->sides_1sided); + + if (pc->sides_2sided_long) + cupsFilePrintf(fp, "Sides2SidedLong %s\n", pc->sides_2sided_long); + + if (pc->sides_2sided_short) + cupsFilePrintf(fp, "Sides2SidedShort %s\n", pc->sides_2sided_short); + + /* + * Product, cupsFilter, cupsFilter2, and cupsPreFilter... + */ + + if (pc->product) + cupsFilePutConf(fp, "Product", pc->product); + + for (value = (const char *)cupsArrayFirst(pc->filters); + value; + value = (const char *)cupsArrayNext(pc->filters)) + cupsFilePutConf(fp, "Filter", value); + + for (value = (const char *)cupsArrayFirst(pc->prefilters); + value; + value = (const char *)cupsArrayNext(pc->prefilters)) + cupsFilePutConf(fp, "PreFilter", value); + + cupsFilePrintf(fp, "SingleFile %s\n", pc->single_file ? "true" : "false"); + + /* + * Finishing options... + */ + + for (f = (_pwg_finishings_t *)cupsArrayFirst(pc->finishings); + f; + f = (_pwg_finishings_t *)cupsArrayNext(pc->finishings)) + { + cupsFilePrintf(fp, "Finishings %d", f->value); + for (i = f->num_options, option = f->options; i > 0; i --, option ++) + cupsFilePrintf(fp, " %s=%s", option->name, option->value); + cupsFilePutChar(fp, '\n'); + } + + /* + * IPP attributes, if any... + */ + + if (attrs) + { + cupsFilePrintf(fp, "IPP " CUPS_LLFMT "\n", CUPS_LLCAST ippLength(attrs)); + + attrs->state = IPP_IDLE; + ippWriteIO(fp, (ipp_iocb_t)cupsFileWrite, 1, NULL, attrs); + } + + /* + * Close and return... + */ + + if (cupsFileClose(fp)) + { + unlink(newfile); + return (0); + } + + unlink(filename); + return (!rename(newfile, filename)); +} + + +/* + * '_pwgInputSlotForSource()' - Get the InputSlot name for the given PWG + * media-source. + */ + +const char * /* O - InputSlot name */ +_pwgInputSlotForSource( + const char *media_source, /* I - PWG media-source */ + char *name, /* I - Name buffer */ + size_t namesize) /* I - Size of name buffer */ +{ + /* + * Range check input... + */ + + if (!media_source || !name || namesize < PPD_MAX_NAME) + return (NULL); + + if (_cups_strcasecmp(media_source, "main")) + strlcpy(name, "Cassette", namesize); + else if (_cups_strcasecmp(media_source, "alternate")) + strlcpy(name, "Multipurpose", namesize); + else if (_cups_strcasecmp(media_source, "large-capacity")) + strlcpy(name, "LargeCapacity", namesize); + else if (_cups_strcasecmp(media_source, "bottom")) + strlcpy(name, "Lower", namesize); + else if (_cups_strcasecmp(media_source, "middle")) + strlcpy(name, "Middle", namesize); + else if (_cups_strcasecmp(media_source, "top")) + strlcpy(name, "Upper", namesize); + else if (_cups_strcasecmp(media_source, "rear")) + strlcpy(name, "Rear", namesize); + else if (_cups_strcasecmp(media_source, "side")) + strlcpy(name, "Side", namesize); + else if (_cups_strcasecmp(media_source, "envelope")) + strlcpy(name, "Envelope", namesize); + else if (_cups_strcasecmp(media_source, "main-roll")) + strlcpy(name, "Roll", namesize); + else if (_cups_strcasecmp(media_source, "alternate-roll")) + strlcpy(name, "Roll2", namesize); + else + pwg_ppdize_name(media_source, name, namesize); + + return (name); +} + + +/* + * '_pwgMediaTypeForType()' - Get the MediaType name for the given PWG + * media-type. + */ + +const char * /* O - MediaType name */ +_pwgMediaTypeForType( + const char *media_type, /* I - PWG media-type */ + char *name, /* I - Name buffer */ + size_t namesize) /* I - Size of name buffer */ +{ + /* + * Range check input... + */ + + if (!media_type || !name || namesize < PPD_MAX_NAME) + return (NULL); + + if (_cups_strcasecmp(media_type, "auto")) + strlcpy(name, "Auto", namesize); + else if (_cups_strcasecmp(media_type, "cardstock")) + strlcpy(name, "Cardstock", namesize); + else if (_cups_strcasecmp(media_type, "envelope")) + strlcpy(name, "Envelope", namesize); + else if (_cups_strcasecmp(media_type, "photographic-glossy")) + strlcpy(name, "Glossy", namesize); + else if (_cups_strcasecmp(media_type, "photographic-high-gloss")) + strlcpy(name, "HighGloss", namesize); + else if (_cups_strcasecmp(media_type, "photographic-matte")) + strlcpy(name, "Matte", namesize); + else if (_cups_strcasecmp(media_type, "stationery")) + strlcpy(name, "Plain", namesize); + else if (_cups_strcasecmp(media_type, "stationery-coated")) + strlcpy(name, "Coated", namesize); + else if (_cups_strcasecmp(media_type, "stationery-inkjet")) + strlcpy(name, "Inkjet", namesize); + else if (_cups_strcasecmp(media_type, "stationery-letterhead")) + strlcpy(name, "Letterhead", namesize); + else if (_cups_strcasecmp(media_type, "stationery-preprinted")) + strlcpy(name, "Preprinted", namesize); + else if (_cups_strcasecmp(media_type, "transparency")) + strlcpy(name, "Transparency", namesize); + else + pwg_ppdize_name(media_type, name, namesize); + + return (name); +} + + +/* + * '_pwgPageSizeForMedia()' - Get the PageSize name for the given media. + */ + +const char * /* O - PageSize name */ +_pwgPageSizeForMedia( + _pwg_media_t *media, /* I - Media */ + char *name, /* I - PageSize name buffer */ + size_t namesize) /* I - Size of name buffer */ +{ + const char *sizeptr, /* Pointer to size in PWG name */ + *dimptr; /* Pointer to dimensions in PWG name */ + + + /* + * Range check input... + */ + + if (!media || !name || namesize < PPD_MAX_NAME) + return (NULL); + + /* + * Copy or generate a PageSize name... + */ + + if (media->ppd) + { + /* + * Use a standard Adobe name... + */ + + strlcpy(name, media->ppd, namesize); + } + else if (!media->pwg || !strncmp(media->pwg, "custom_", 7) || + (sizeptr = strchr(media->pwg, '_')) == NULL || + (dimptr = strchr(sizeptr + 1, '_')) == NULL || + (size_t)(dimptr - sizeptr) > namesize) + { + /* + * Use a name of the form "wNNNhNNN"... + */ + + snprintf(name, namesize, "w%dh%d", (int)_PWG_TOPTS(media->width), + (int)_PWG_TOPTS(media->length)); + } + else + { + /* + * Copy the size name from class_sizename_dimensions... + */ + + memcpy(name, sizeptr + 1, dimptr - sizeptr - 1); + name[dimptr - sizeptr - 1] = '\0'; + } + + return (name); +} + + +/* + * 'pwg_compare_finishings()' - Compare two finishings values. + */ + +static int /* O- Result of comparison */ +pwg_compare_finishings( + _pwg_finishings_t *a, /* I - First finishings value */ + _pwg_finishings_t *b) /* I - Second finishings value */ +{ + return (b->value - a->value); +} + + +/* + * 'pwg_free_finishings()' - Free a finishings value. + */ + +static void +pwg_free_finishings( + _pwg_finishings_t *f) /* I - Finishings value */ +{ + cupsFreeOptions(f->num_options, f->options); + free(f); +} + + +/* + * 'pwg_ppdize_name()' - Convert an IPP keyword to a PPD keyword. + */ + +static void +pwg_ppdize_name(const char *ipp, /* I - IPP keyword */ + char *name, /* I - Name buffer */ + size_t namesize) /* I - Size of name buffer */ +{ + char *ptr, /* Pointer into name buffer */ + *end; /* End of name buffer */ + + + *name = toupper(*ipp++); + + for (ptr = name + 1, end = name + namesize - 1; *ipp && ptr < end;) + { + if (*ipp == '-' && _cups_isalpha(ipp[1])) + { + ipp ++; + *ptr++ = toupper(*ipp++ & 255); + } + else + *ptr++ = *ipp++; + } + + *ptr = '\0'; +} + + +/* + * 'pwg_unppdize_name()' - Convert a PPD keyword to a lowercase IPP keyword. + */ + +static void +pwg_unppdize_name(const char *ppd, /* I - PPD keyword */ + char *name, /* I - Name buffer */ + size_t namesize) /* I - Size of name buffer */ +{ + char *ptr, /* Pointer into name buffer */ + *end; /* End of name buffer */ + + + for (ptr = name, end = name + namesize - 1; *ppd && ptr < end; ppd ++) + { + if (_cups_isalnum(*ppd) || *ppd == '-') + *ptr++ = tolower(*ppd & 255); + else if (*ppd == '_' || *ppd == '.') + *ptr++ = '-'; + + if (!_cups_isupper(*ppd) && _cups_isalnum(*ppd) && + _cups_isupper(ppd[1]) && ptr < end) + *ptr++ = '-'; + } + + *ptr = '\0'; +} + + +/* + * End of "$Id$". + */ diff --git a/cups/ppd-private.h b/cups/ppd-private.h new file mode 100644 index 0000000000..bc6a0f8b91 --- /dev/null +++ b/cups/ppd-private.h @@ -0,0 +1,214 @@ +/* + * "$Id$" + * + * Private PPD definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * PostScript is a trademark of Adobe Systems, Inc. + * + * This code and any derivative of it may be used and distributed + * freely under the terms of the GNU General Public License when + * used with GNU Ghostscript or its derivatives. Use of the code + * (or any derivative of it) with software other than GNU + * GhostScript (or its derivatives) is governed by the CUPS license + * agreement. + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_PPD_PRIVATE_H_ +# define _CUPS_PPD_PRIVATE_H_ + +/* + * Include necessary headers... + */ + +# include +# include +# include "pwg-private.h" + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Constants... + */ + +# define _PPD_CACHE_VERSION 2 /* Version number in cache file */ + + +/* + * Types and structures... + */ + +typedef enum _ppd_localization_e /**** Selector for _ppdOpen ****/ +{ + _PPD_LOCALIZATION_DEFAULT, /* Load only the default localization */ + _PPD_LOCALIZATION_ICC_PROFILES, /* Load only the color profile localization */ + _PPD_LOCALIZATION_NONE, /* Load no localizations */ + _PPD_LOCALIZATION_ALL /* Load all localizations */ +} _ppd_localization_t; + +typedef enum _ppd_parse_e /**** Selector for _ppdParseOptions ****/ +{ + _PPD_PARSE_OPTIONS, /* Parse only the options */ + _PPD_PARSE_PROPERTIES, /* Parse only the properties */ + _PPD_PARSE_ALL /* Parse everything */ +} _ppd_parse_t; + +typedef struct _ppd_cups_uiconst_s /**** Constraint from cupsUIConstraints ****/ +{ + ppd_option_t *option; /* Constrained option */ + ppd_choice_t *choice; /* Constrained choice or @code NULL@ */ + int installable; /* Installable option? */ +} _ppd_cups_uiconst_t; + +typedef struct _ppd_cups_uiconsts_s /**** cupsUIConstraints ****/ +{ + char resolver[PPD_MAX_NAME]; /* Resolver name */ + int installable, /* Constrained against any installable options? */ + num_constraints; /* Number of constraints */ + _ppd_cups_uiconst_t *constraints; /* Constraints */ +} _ppd_cups_uiconsts_t; + +typedef enum _pwg_print_color_mode_e /**** PWG print-color-mode indices ****/ +{ + _PWG_PRINT_COLOR_MODE_MONOCHROME = 0, /* print-color-mode=monochrome */ + _PWG_PRINT_COLOR_MODE_COLOR, /* print-color-mode=color */ + /* Other proposed values are not supported by CUPS yet. */ + _PWG_PRINT_COLOR_MODE_MAX +} _pwg_print_color_mode_t; + +typedef enum _pwg_print_quality_e /**** PWG print-quality values ****/ +{ + _PWG_PRINT_QUALITY_DRAFT = 0, /* print-quality=3 */ + _PWG_PRINT_QUALITY_NORMAL, /* print-quality=4 */ + _PWG_PRINT_QUALITY_HIGH, /* print-quality=5 */ + _PWG_PRINT_QUALITY_MAX +} _pwg_print_quality_t; + +typedef struct _pwg_finishings_s /**** PWG finishings mapping data ****/ +{ + ipp_finish_t value; /* finishings value */ + int num_options; /* Number of options to apply */ + cups_option_t *options; /* Options to apply */ +} _pwg_finishings_t; + +struct _ppd_cache_s /**** PPD cache and PWG conversion data ****/ +{ + int num_bins; /* Number of output bins */ + _pwg_map_t *bins; /* Output bins */ + int num_sizes; /* Number of media sizes */ + _pwg_size_t *sizes; /* Media sizes */ + int custom_max_width, /* Maximum custom width in 2540ths */ + custom_max_length, /* Maximum custom length in 2540ths */ + custom_min_width, /* Minimum custom width in 2540ths */ + custom_min_length; /* Minimum custom length in 2540ths */ + char *custom_max_keyword, /* Maximum custom size PWG keyword */ + *custom_min_keyword, /* Minimum custom size PWG keyword */ + custom_ppd_size[41]; /* Custom PPD size name */ + _pwg_size_t custom_size; /* Custom size record */ + char *source_option; /* PPD option for media source */ + int num_sources; /* Number of media sources */ + _pwg_map_t *sources; /* Media sources */ + int num_types; /* Number of media types */ + _pwg_map_t *types; /* Media types */ + int num_presets[_PWG_PRINT_COLOR_MODE_MAX][_PWG_PRINT_QUALITY_MAX]; + /* Number of print-color-mode/print-quality options */ + cups_option_t *presets[_PWG_PRINT_COLOR_MODE_MAX][_PWG_PRINT_QUALITY_MAX]; + /* print-color-mode/print-quality options */ + char *sides_option, /* PPD option for sides */ + *sides_1sided, /* Choice for one-sided */ + *sides_2sided_long, /* Choice for two-sided-long-edge */ + *sides_2sided_short; /* Choice for two-sided-short-edge */ + char *product; /* Product value */ + cups_array_t *filters, /* cupsFilter/cupsFilter2 values */ + *prefilters; /* cupsPreFilter values */ + int single_file; /* cupsSingleFile value */ + cups_array_t *finishings; /* cupsIPPFinishings values */ +}; + + +/* + * Prototypes... + */ + +extern _ppd_cache_t *_ppdCacheCreateWithFile(const char *filename, + ipp_t **attrs); +extern _ppd_cache_t *_ppdCacheCreateWithPPD(ppd_file_t *ppd); +extern void _ppdCacheDestroy(_ppd_cache_t *pc); +extern const char *_ppdCacheGetBin(_ppd_cache_t *pc, + const char *output_bin); +extern int _ppdCacheGetFinishingOptions(_ppd_cache_t *pc, ipp_t *job, + ipp_finish_t value, int num_options, + cups_option_t **options); +extern int _ppdCacheGetFinishingValues(_ppd_cache_t *pc, int num_options, + cups_option_t *options, + int max_values, int *values); +extern const char *_ppdCacheGetInputSlot(_ppd_cache_t *pc, ipp_t *job, + const char *keyword); +extern const char *_ppdCacheGetMediaType(_ppd_cache_t *pc, ipp_t *job, + const char *keyword); +extern const char *_ppdCacheGetOutputBin(_ppd_cache_t *pc, + const char *keyword); +extern const char *_ppdCacheGetPageSize(_ppd_cache_t *pc, ipp_t *job, + const char *keyword, int *exact); +extern _pwg_size_t *_ppdCacheGetSize(_ppd_cache_t *pc, + const char *page_size); +extern const char *_ppdCacheGetSource(_ppd_cache_t *pc, + const char *input_slot); +extern const char *_ppdCacheGetType(_ppd_cache_t *pc, + const char *media_type); +extern int _ppdCacheWriteFile(_ppd_cache_t *pc, + const char *filename, ipp_t *attrs); +extern void _ppdFreeLanguages(cups_array_t *languages); +extern cups_encoding_t _ppdGetEncoding(const char *name); +extern cups_array_t *_ppdGetLanguages(ppd_file_t *ppd); +extern unsigned _ppdHashName(const char *name); +extern ppd_attr_t *_ppdLocalizedAttr(ppd_file_t *ppd, const char *keyword, + const char *spec, const char *ll_CC); +extern char *_ppdNormalizeMakeAndModel(const char *make_and_model, + char *buffer, + size_t bufsize); +extern ppd_file_t *_ppdOpen(cups_file_t *fp, + _ppd_localization_t localization); +extern ppd_file_t *_ppdOpenFile(const char *filename, + _ppd_localization_t localization); +extern int _ppdParseOptions(const char *s, int num_options, + cups_option_t **options, + _ppd_parse_t which); +extern const char *_pwgInputSlotForSource(const char *media_source, + char *name, size_t namesize); +extern const char *_pwgMediaTypeForType(const char *media_type, + char *name, size_t namesize); +extern const char *_pwgPageSizeForMedia(_pwg_media_t *media, + char *name, size_t namesize); + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +} +# endif /* __cplusplus */ +#endif /* !_CUPS_PPD_PRIVATE_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/ppd.c b/cups/ppd.c new file mode 100644 index 0000000000..cfb2d42c33 --- /dev/null +++ b/cups/ppd.c @@ -0,0 +1,3398 @@ +/* + * "$Id$" + * + * PPD file routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * PostScript is a trademark of Adobe Systems, Inc. + * + * This code and any derivative of it may be used and distributed + * freely under the terms of the GNU General Public License when + * used with GNU Ghostscript or its derivatives. Use of the code + * (or any derivative of it) with software other than GNU + * GhostScript (or its derivatives) is governed by the CUPS license + * agreement. + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * ppdClose() - Free all memory used by the PPD file. + * ppdErrorString() - Returns the text assocated with a status. + * _ppdGetEncoding() - Get the CUPS encoding value for the given + * LanguageEncoding. + * ppdLastError() - Return the status from the last ppdOpen*(). + * ppdOpen() - Read a PPD file into memory. + * _ppdOpen() - Read a PPD file into memory. + * ppdOpen2() - Read a PPD file into memory. + * ppdOpenFd() - Read a PPD file into memory. + * _ppdOpenFile() - Read a PPD file into memory. + * ppdOpenFile() - Read a PPD file into memory. + * ppdSetConformance() - Set the conformance level for PPD files. + * ppd_add_attr() - Add an attribute to the PPD data. + * ppd_add_choice() - Add a choice to an option. + * ppd_add_size() - Add a page size. + * ppd_compare_attrs() - Compare two attributes. + * ppd_compare_choices() - Compare two choices... + * ppd_compare_coptions() - Compare two custom options. + * ppd_compare_options() - Compare two options. + * ppd_decode() - Decode a string value... + * ppd_free_filters() - Free the filters array. + * ppd_free_group() - Free a single UI group. + * ppd_free_option() - Free a single option. + * ppd_get_coption() - Get a custom option record. + * ppd_get_cparam() - Get a custom parameter record. + * ppd_get_group() - Find or create the named group as needed. + * ppd_get_option() - Find or create the named option as needed. + * ppd_hash_option() - Generate a hash of the option name... + * ppd_read() - Read a line from a PPD file, skipping comment + * lines as necessary. + * ppd_update_filters() - Update the filters array as needed. + */ + +/* + * Include necessary headers. + */ + +#include "cups-private.h" +#include "ppd-private.h" + + +/* + * Definitions... + */ + +#if defined(WIN32) || defined(__EMX__) +# define READ_BINARY "rb" /* Open a binary file for reading */ +# define WRITE_BINARY "wb" /* Open a binary file for writing */ +#else +# define READ_BINARY "r" /* Open a binary file for reading */ +# define WRITE_BINARY "w" /* Open a binary file for writing */ +#endif /* WIN32 || __EMX__ */ + +#define ppd_free(p) if (p) free(p) /* Safe free macro */ + +#define PPD_KEYWORD 1 /* Line contained a keyword */ +#define PPD_OPTION 2 /* Line contained an option name */ +#define PPD_TEXT 4 /* Line contained human-readable text */ +#define PPD_STRING 8 /* Line contained a string or code */ + +#define PPD_HASHSIZE 512 /* Size of hash */ + + +/* + * Line buffer structure... + */ + +typedef struct _ppd_line_s +{ + char *buffer; /* Pointer to buffer */ + size_t bufsize; /* Size of the buffer */ +} _ppd_line_t; + + +/* + * Local functions... + */ + +static ppd_attr_t *ppd_add_attr(ppd_file_t *ppd, const char *name, + const char *spec, const char *text, + const char *value); +static ppd_choice_t *ppd_add_choice(ppd_option_t *option, const char *name); +static ppd_size_t *ppd_add_size(ppd_file_t *ppd, const char *name); +static int ppd_compare_attrs(ppd_attr_t *a, ppd_attr_t *b); +static int ppd_compare_choices(ppd_choice_t *a, ppd_choice_t *b); +static int ppd_compare_coptions(ppd_coption_t *a, + ppd_coption_t *b); +static int ppd_compare_options(ppd_option_t *a, ppd_option_t *b); +static int ppd_decode(char *string); +static void ppd_free_filters(ppd_file_t *ppd); +static void ppd_free_group(ppd_group_t *group); +static void ppd_free_option(ppd_option_t *option); +static ppd_coption_t *ppd_get_coption(ppd_file_t *ppd, const char *name); +static ppd_cparam_t *ppd_get_cparam(ppd_coption_t *opt, + const char *param, + const char *text); +static ppd_group_t *ppd_get_group(ppd_file_t *ppd, const char *name, + const char *text, _cups_globals_t *cg, + cups_encoding_t encoding); +static ppd_option_t *ppd_get_option(ppd_group_t *group, const char *name); +static int ppd_hash_option(ppd_option_t *option); +static int ppd_read(cups_file_t *fp, _ppd_line_t *line, + char *keyword, char *option, char *text, + char **string, int ignoreblank, + _cups_globals_t *cg); +static int ppd_update_filters(ppd_file_t *ppd, + _cups_globals_t *cg); + + +/* + * 'ppdClose()' - Free all memory used by the PPD file. + */ + +void +ppdClose(ppd_file_t *ppd) /* I - PPD file record */ +{ + int i; /* Looping var */ + ppd_emul_t *emul; /* Current emulation */ + ppd_group_t *group; /* Current group */ + char **font; /* Current font */ + ppd_attr_t **attr; /* Current attribute */ + ppd_coption_t *coption; /* Current custom option */ + ppd_cparam_t *cparam; /* Current custom parameter */ + + + /* + * Range check arguments... + */ + + if (!ppd) + return; + + /* + * Free all strings at the top level... + */ + + _cupsStrFree(ppd->lang_encoding); + _cupsStrFree(ppd->nickname); + if (ppd->patches) + free(ppd->patches); + _cupsStrFree(ppd->jcl_begin); + _cupsStrFree(ppd->jcl_end); + _cupsStrFree(ppd->jcl_ps); + + /* + * Free any emulations... + */ + + if (ppd->num_emulations > 0) + { + for (i = ppd->num_emulations, emul = ppd->emulations; i > 0; i --, emul ++) + { + _cupsStrFree(emul->start); + _cupsStrFree(emul->stop); + } + + ppd_free(ppd->emulations); + } + + /* + * Free any UI groups, subgroups, and options... + */ + + if (ppd->num_groups > 0) + { + for (i = ppd->num_groups, group = ppd->groups; i > 0; i --, group ++) + ppd_free_group(group); + + ppd_free(ppd->groups); + } + + cupsArrayDelete(ppd->options); + cupsArrayDelete(ppd->marked); + + /* + * Free any page sizes... + */ + + if (ppd->num_sizes > 0) + ppd_free(ppd->sizes); + + /* + * Free any constraints... + */ + + if (ppd->num_consts > 0) + ppd_free(ppd->consts); + + /* + * Free any filters... + */ + + ppd_free_filters(ppd); + + /* + * Free any fonts... + */ + + if (ppd->num_fonts > 0) + { + for (i = ppd->num_fonts, font = ppd->fonts; i > 0; i --, font ++) + _cupsStrFree(*font); + + ppd_free(ppd->fonts); + } + + /* + * Free any profiles... + */ + + if (ppd->num_profiles > 0) + ppd_free(ppd->profiles); + + /* + * Free any attributes... + */ + + if (ppd->num_attrs > 0) + { + for (i = ppd->num_attrs, attr = ppd->attrs; i > 0; i --, attr ++) + { + _cupsStrFree((*attr)->value); + ppd_free(*attr); + } + + ppd_free(ppd->attrs); + } + + cupsArrayDelete(ppd->sorted_attrs); + + /* + * Free custom options... + */ + + for (coption = (ppd_coption_t *)cupsArrayFirst(ppd->coptions); + coption; + coption = (ppd_coption_t *)cupsArrayNext(ppd->coptions)) + { + for (cparam = (ppd_cparam_t *)cupsArrayFirst(coption->params); + cparam; + cparam = (ppd_cparam_t *)cupsArrayNext(coption->params)) + { + switch (cparam->type) + { + case PPD_CUSTOM_PASSCODE : + case PPD_CUSTOM_PASSWORD : + case PPD_CUSTOM_STRING : + _cupsStrFree(cparam->current.custom_string); + break; + + default : + break; + } + + free(cparam); + } + + cupsArrayDelete(coption->params); + + free(coption); + } + + cupsArrayDelete(ppd->coptions); + + /* + * Free constraints... + */ + + if (ppd->cups_uiconstraints) + { + _ppd_cups_uiconsts_t *consts; /* Current constraints */ + + + for (consts = (_ppd_cups_uiconsts_t *)cupsArrayFirst(ppd->cups_uiconstraints); + consts; + consts = (_ppd_cups_uiconsts_t *)cupsArrayNext(ppd->cups_uiconstraints)) + { + free(consts->constraints); + free(consts); + } + + cupsArrayDelete(ppd->cups_uiconstraints); + } + + /* + * Free any PPD cache/mapping data... + */ + + if (ppd->cache) + _ppdCacheDestroy(ppd->cache); + + /* + * Free the whole record... + */ + + ppd_free(ppd); +} + + +/* + * 'ppdErrorString()' - Returns the text assocated with a status. + * + * @since CUPS 1.1.19/Mac OS X 10.3@ + */ + +const char * /* O - Status string */ +ppdErrorString(ppd_status_t status) /* I - PPD status */ +{ + static const char * const messages[] =/* Status messages */ + { + _("OK"), + _("Unable to open PPD file"), + _("NULL PPD file pointer"), + _("Memory allocation error"), + _("Missing PPD-Adobe-4.x header"), + _("Missing value string"), + _("Internal error"), + _("Bad OpenGroup"), + _("OpenGroup without a CloseGroup first"), + _("Bad OpenUI/JCLOpenUI"), + _("OpenUI/JCLOpenUI without a CloseUI/JCLCloseUI first"), + _("Bad OrderDependency"), + _("Bad UIConstraints"), + _("Missing asterisk in column 1"), + _("Line longer than the maximum allowed (255 characters)"), + _("Illegal control character"), + _("Illegal main keyword string"), + _("Illegal option keyword string"), + _("Illegal translation string"), + _("Illegal whitespace character"), + _("Bad custom parameter"), + _("Missing option keyword"), + _("Bad value string"), + _("Missing CloseGroup") + }; + + + if (status < PPD_OK || status >= PPD_MAX_STATUS) + return (_cupsLangString(cupsLangDefault(), _("Unknown"))); + else + return (_cupsLangString(cupsLangDefault(), messages[status])); +} + + +/* + * '_ppdGetEncoding()' - Get the CUPS encoding value for the given + * LanguageEncoding. + */ + +cups_encoding_t /* O - CUPS encoding value */ +_ppdGetEncoding(const char *name) /* I - LanguageEncoding string */ +{ + if (!_cups_strcasecmp(name, "ISOLatin1")) + return (CUPS_ISO8859_1); + else if (!_cups_strcasecmp(name, "ISOLatin2")) + return (CUPS_ISO8859_2); + else if (!_cups_strcasecmp(name, "ISOLatin5")) + return (CUPS_ISO8859_5); + else if (!_cups_strcasecmp(name, "JIS83-RKSJ")) + return (CUPS_JIS_X0213); + else if (!_cups_strcasecmp(name, "MacStandard")) + return (CUPS_MAC_ROMAN); + else if (!_cups_strcasecmp(name, "WindowsANSI")) + return (CUPS_WINDOWS_1252); + else + return (CUPS_UTF8); +} + + +/* + * 'ppdLastError()' - Return the status from the last ppdOpen*(). + * + * @since CUPS 1.1.19/Mac OS X 10.3@ + */ + +ppd_status_t /* O - Status code */ +ppdLastError(int *line) /* O - Line number */ +{ + _cups_globals_t *cg = _cupsGlobals(); + /* Global data */ + + + if (line) + *line = cg->ppd_line; + + return (cg->ppd_status); +} + + +/* + * '_ppdOpen()' - Read a PPD file into memory. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ppd_file_t * /* O - PPD file record or @code NULL@ if the PPD file could not be opened. */ +_ppdOpen( + cups_file_t *fp, /* I - File to read from */ + _ppd_localization_t localization) /* I - Localization to load */ +{ + int i, j, k; /* Looping vars */ + int count; /* Temporary count */ + _ppd_line_t line; /* Line buffer */ + ppd_file_t *ppd; /* PPD file record */ + ppd_group_t *group, /* Current group */ + *subgroup; /* Current sub-group */ + ppd_option_t *option; /* Current option */ + ppd_choice_t *choice; /* Current choice */ + ppd_const_t *constraint; /* Current constraint */ + ppd_size_t *size; /* Current page size */ + int mask; /* Line data mask */ + char keyword[PPD_MAX_NAME], + /* Keyword from file */ + name[PPD_MAX_NAME], + /* Option from file */ + text[PPD_MAX_LINE], + /* Human-readable text from file */ + *string, /* Code/text from file */ + *sptr, /* Pointer into string */ + *nameptr, /* Pointer into name */ + *temp, /* Temporary string pointer */ + **tempfonts; /* Temporary fonts pointer */ + float order; /* Order dependency number */ + ppd_section_t section; /* Order dependency section */ + ppd_profile_t *profile; /* Pointer to color profile */ + char **filter; /* Pointer to filter */ + struct lconv *loc; /* Locale data */ + int ui_keyword; /* Is this line a UI keyword? */ + cups_lang_t *lang; /* Language data */ + cups_encoding_t encoding; /* Encoding of PPD file */ + _cups_globals_t *cg = _cupsGlobals(); + /* Global data */ + char custom_name[PPD_MAX_NAME]; + /* CustomFoo attribute name */ + ppd_attr_t *custom_attr; /* CustomFoo attribute */ + char ll[4], /* Language + '.' */ + ll_CC[7]; /* Language + country + '.' */ + size_t ll_len, /* Language length */ + ll_CC_len; /* Language + country length */ + static const char * const ui_keywords[] = + { +#ifdef CUPS_USE_FULL_UI_KEYWORDS_LIST + /* + * Adobe defines some 41 keywords as "UI", meaning that they are + * user interface elements and that they should be treated as such + * even if the PPD creator doesn't use Open/CloseUI around them. + * + * Since this can cause previously invisible options to appear and + * confuse users, the default is to only treat the PageSize and + * PageRegion keywords this way. + */ + /* Boolean keywords */ + "BlackSubstitution", + "Booklet", + "Collate", + "ManualFeed", + "MirrorPrint", + "NegativePrint", + "Sorter", + "TraySwitch", + + /* PickOne keywords */ + "AdvanceMedia", + "BindColor", + "BindEdge", + "BindType", + "BindWhen", + "BitsPerPixel", + "ColorModel", + "CutMedia", + "Duplex", + "FoldType", + "FoldWhen", + "InputSlot", + "JCLFrameBufferSize", + "JCLResolution", + "Jog", + "MediaColor", + "MediaType", + "MediaWeight", + "OutputBin", + "OutputMode", + "OutputOrder", + "PageRegion", + "PageSize", + "Resolution", + "Separations", + "Signature", + "Slipsheet", + "Smoothing", + "StapleLocation", + "StapleOrientation", + "StapleWhen", + "StapleX", + "StapleY" +#else /* !CUPS_USE_FULL_UI_KEYWORDS_LIST */ + "PageRegion", + "PageSize" +#endif /* CUPS_USE_FULL_UI_KEYWORDS_LIST */ + }; + static const char * const color_keywords[] = /* Keywords associated with color profiles */ + { + ".cupsICCProfile", + ".ColorModel", + }; + + + DEBUG_printf(("_ppdOpen(fp=%p)", fp)); + + /* + * Default to "OK" status... + */ + + cg->ppd_status = PPD_OK; + cg->ppd_line = 0; + + /* + * Range check input... + */ + + if (fp == NULL) + { + cg->ppd_status = PPD_NULL_FILE; + return (NULL); + } + + /* + * If only loading a single localization set up the strings to match... + */ + + if (localization == _PPD_LOCALIZATION_DEFAULT) + { + if ((lang = cupsLangDefault()) == NULL) + return (NULL); + + snprintf(ll_CC, sizeof(ll_CC), "%s.", lang->language); + snprintf(ll, sizeof(ll), "%2.2s.", lang->language); + + ll_CC_len = strlen(ll_CC); + ll_len = strlen(ll); + + DEBUG_printf(("2_ppdOpen: Loading localizations matching \"%s\" and \"%s\"", + ll_CC, ll)); + } + + /* + * Grab the first line and make sure it reads '*PPD-Adobe: "major.minor"'... + */ + + line.buffer = NULL; + line.bufsize = 0; + + mask = ppd_read(fp, &line, keyword, name, text, &string, 0, cg); + + DEBUG_printf(("2_ppdOpen: mask=%x, keyword=\"%s\"...", mask, keyword)); + + if (mask == 0 || + strcmp(keyword, "PPD-Adobe") || + string == NULL || string[0] != '4') + { + /* + * Either this is not a PPD file, or it is not a 4.x PPD file. + */ + + if (cg->ppd_status == PPD_OK) + cg->ppd_status = PPD_MISSING_PPDADOBE4; + + _cupsStrFree(string); + ppd_free(line.buffer); + + return (NULL); + } + + DEBUG_printf(("2_ppdOpen: keyword=%s, string=%p", keyword, string)); + + _cupsStrFree(string); + + /* + * Allocate memory for the PPD file record... + */ + + if ((ppd = calloc(1, sizeof(ppd_file_t))) == NULL) + { + cg->ppd_status = PPD_ALLOC_ERROR; + + _cupsStrFree(string); + ppd_free(line.buffer); + + return (NULL); + } + + ppd->language_level = 2; + ppd->color_device = 0; + ppd->colorspace = PPD_CS_N; + ppd->landscape = -90; + ppd->coptions = cupsArrayNew((cups_array_func_t)ppd_compare_coptions, + NULL); + + /* + * Read lines from the PPD file and add them to the file record... + */ + + group = NULL; + subgroup = NULL; + option = NULL; + choice = NULL; + ui_keyword = 0; + encoding = CUPS_ISO8859_1; + loc = localeconv(); + + while ((mask = ppd_read(fp, &line, keyword, name, text, &string, 1, cg)) != 0) + { + DEBUG_printf(("2_ppdOpen: mask=%x, keyword=\"%s\", name=\"%s\", " + "text=\"%s\", string=%d chars...", mask, keyword, name, text, + string ? (int)strlen(string) : 0)); + + if (strncmp(keyword, "Default", 7) && !string && + cg->ppd_conform != PPD_CONFORM_RELAXED) + { + /* + * Need a string value! + */ + + cg->ppd_status = PPD_MISSING_VALUE; + + goto error; + } + else if (!string) + continue; + + /* + * Certain main keywords (as defined by the PPD spec) may be used + * without the usual OpenUI/CloseUI stuff. Presumably this is just + * so that Adobe wouldn't completely break compatibility with PPD + * files prior to v4.0 of the spec, but it is hopelessly + * inconsistent... Catch these main keywords and automatically + * create the corresponding option, as needed... + */ + + if (ui_keyword) + { + /* + * Previous line was a UI keyword... + */ + + option = NULL; + ui_keyword = 0; + } + + /* + * If we are filtering out keyword localizations, see if this line needs to + * be used... + */ + + if (localization != _PPD_LOCALIZATION_ALL && + (temp = strchr(keyword, '.')) != NULL && + ((temp - keyword) == 2 || (temp - keyword) == 5) && + _cups_isalpha(keyword[0]) && + _cups_isalpha(keyword[1]) && + (keyword[2] == '.' || + (keyword[2] == '_' && _cups_isalpha(keyword[3]) && + _cups_isalpha(keyword[4]) && keyword[5] == '.'))) + { + if (localization == _PPD_LOCALIZATION_NONE || + (localization == _PPD_LOCALIZATION_DEFAULT && + strncmp(ll_CC, keyword, ll_CC_len) && + strncmp(ll, keyword, ll_len))) + { + DEBUG_printf(("2_ppdOpen: Ignoring localization: \"%s\"\n", keyword)); + continue; + } + else if (localization == _PPD_LOCALIZATION_ICC_PROFILES) + { + /* + * Only load localizations for the color profile related keywords... + */ + + for (i = 0; + i < (int)(sizeof(color_keywords) / sizeof(color_keywords[0])); + i ++) + { + if (!_cups_strcasecmp(temp, color_keywords[i])) + break; + } + + if (i >= (int)(sizeof(color_keywords) / sizeof(color_keywords[0]))) + { + DEBUG_printf(("2_ppdOpen: Ignoring localization: \"%s\"\n", keyword)); + continue; + } + } + } + + if (option == NULL && + (mask & (PPD_KEYWORD | PPD_OPTION | PPD_STRING)) == + (PPD_KEYWORD | PPD_OPTION | PPD_STRING)) + { + for (i = 0; i < (int)(sizeof(ui_keywords) / sizeof(ui_keywords[0])); i ++) + if (!strcmp(keyword, ui_keywords[i])) + break; + + if (i < (int)(sizeof(ui_keywords) / sizeof(ui_keywords[0]))) + { + /* + * Create the option in the appropriate group... + */ + + ui_keyword = 1; + + DEBUG_printf(("2_ppdOpen: FOUND ADOBE UI KEYWORD %s WITHOUT OPENUI!", + keyword)); + + if (!group) + { + if ((group = ppd_get_group(ppd, "General", _("General"), cg, + encoding)) == NULL) + goto error; + + DEBUG_printf(("2_ppdOpen: Adding to group %s...", group->text)); + option = ppd_get_option(group, keyword); + group = NULL; + } + else + option = ppd_get_option(group, keyword); + + if (option == NULL) + { + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + /* + * Now fill in the initial information for the option... + */ + + if (!strncmp(keyword, "JCL", 3)) + option->section = PPD_ORDER_JCL; + else + option->section = PPD_ORDER_ANY; + + option->order = 10.0f; + + if (i < 8) + option->ui = PPD_UI_BOOLEAN; + else + option->ui = PPD_UI_PICKONE; + + for (j = 0; j < ppd->num_attrs; j ++) + if (!strncmp(ppd->attrs[j]->name, "Default", 7) && + !strcmp(ppd->attrs[j]->name + 7, keyword) && + ppd->attrs[j]->value) + { + DEBUG_printf(("2_ppdOpen: Setting Default%s to %s via attribute...", + option->keyword, ppd->attrs[j]->value)); + strlcpy(option->defchoice, ppd->attrs[j]->value, + sizeof(option->defchoice)); + break; + } + + if (!strcmp(keyword, "PageSize")) + strlcpy(option->text, _("Media Size"), sizeof(option->text)); + else if (!strcmp(keyword, "MediaType")) + strlcpy(option->text, _("Media Type"), sizeof(option->text)); + else if (!strcmp(keyword, "InputSlot")) + strlcpy(option->text, _("Media Source"), sizeof(option->text)); + else if (!strcmp(keyword, "ColorModel")) + strlcpy(option->text, _("Output Mode"), sizeof(option->text)); + else if (!strcmp(keyword, "Resolution")) + strlcpy(option->text, _("Resolution"), sizeof(option->text)); + else + strlcpy(option->text, keyword, sizeof(option->text)); + } + } + + if (!strcmp(keyword, "LanguageLevel")) + ppd->language_level = atoi(string); + else if (!strcmp(keyword, "LanguageEncoding")) + { + /* + * Say all PPD files are UTF-8, since we convert to UTF-8... + */ + + ppd->lang_encoding = _cupsStrAlloc("UTF-8"); + encoding = _ppdGetEncoding(string); + } + else if (!strcmp(keyword, "LanguageVersion")) + ppd->lang_version = string; + else if (!strcmp(keyword, "Manufacturer")) + ppd->manufacturer = string; + else if (!strcmp(keyword, "ModelName")) + ppd->modelname = string; + else if (!strcmp(keyword, "Protocols")) + ppd->protocols = string; + else if (!strcmp(keyword, "PCFileName")) + ppd->pcfilename = string; + else if (!strcmp(keyword, "NickName")) + { + if (encoding != CUPS_UTF8) + { + cups_utf8_t utf8[256]; /* UTF-8 version of NickName */ + + + cupsCharsetToUTF8(utf8, string, sizeof(utf8), encoding); + ppd->nickname = _cupsStrAlloc((char *)utf8); + } + else + ppd->nickname = _cupsStrAlloc(string); + } + else if (!strcmp(keyword, "Product")) + ppd->product = string; + else if (!strcmp(keyword, "ShortNickName")) + ppd->shortnickname = string; + else if (!strcmp(keyword, "TTRasterizer")) + ppd->ttrasterizer = string; + else if (!strcmp(keyword, "JCLBegin")) + { + ppd->jcl_begin = _cupsStrAlloc(string); + ppd_decode(ppd->jcl_begin); /* Decode quoted string */ + } + else if (!strcmp(keyword, "JCLEnd")) + { + ppd->jcl_end = _cupsStrAlloc(string); + ppd_decode(ppd->jcl_end); /* Decode quoted string */ + } + else if (!strcmp(keyword, "JCLToPSInterpreter")) + { + ppd->jcl_ps = _cupsStrAlloc(string); + ppd_decode(ppd->jcl_ps); /* Decode quoted string */ + } + else if (!strcmp(keyword, "AccurateScreensSupport")) + ppd->accurate_screens = !strcmp(string, "True"); + else if (!strcmp(keyword, "ColorDevice")) + ppd->color_device = !strcmp(string, "True"); + else if (!strcmp(keyword, "ContoneOnly")) + ppd->contone_only = !strcmp(string, "True"); + else if (!strcmp(keyword, "cupsFlipDuplex")) + ppd->flip_duplex = !strcmp(string, "True"); + else if (!strcmp(keyword, "cupsManualCopies")) + ppd->manual_copies = !strcmp(string, "True"); + else if (!strcmp(keyword, "cupsModelNumber")) + ppd->model_number = atoi(string); + else if (!strcmp(keyword, "cupsColorProfile")) + { + if (ppd->num_profiles == 0) + profile = malloc(sizeof(ppd_profile_t)); + else + profile = realloc(ppd->profiles, sizeof(ppd_profile_t) * + (ppd->num_profiles + 1)); + + if (!profile) + { + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + ppd->profiles = profile; + profile += ppd->num_profiles; + ppd->num_profiles ++; + + memset(profile, 0, sizeof(ppd_profile_t)); + strlcpy(profile->resolution, name, sizeof(profile->resolution)); + strlcpy(profile->media_type, text, sizeof(profile->media_type)); + + profile->density = (float)_cupsStrScand(string, &sptr, loc); + profile->gamma = (float)_cupsStrScand(sptr, &sptr, loc); + profile->matrix[0][0] = (float)_cupsStrScand(sptr, &sptr, loc); + profile->matrix[0][1] = (float)_cupsStrScand(sptr, &sptr, loc); + profile->matrix[0][2] = (float)_cupsStrScand(sptr, &sptr, loc); + profile->matrix[1][0] = (float)_cupsStrScand(sptr, &sptr, loc); + profile->matrix[1][1] = (float)_cupsStrScand(sptr, &sptr, loc); + profile->matrix[1][2] = (float)_cupsStrScand(sptr, &sptr, loc); + profile->matrix[2][0] = (float)_cupsStrScand(sptr, &sptr, loc); + profile->matrix[2][1] = (float)_cupsStrScand(sptr, &sptr, loc); + profile->matrix[2][2] = (float)_cupsStrScand(sptr, &sptr, loc); + } + else if (!strcmp(keyword, "cupsFilter")) + { + if (ppd->num_filters == 0) + filter = malloc(sizeof(char *)); + else + filter = realloc(ppd->filters, sizeof(char *) * (ppd->num_filters + 1)); + + if (filter == NULL) + { + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + ppd->filters = filter; + filter += ppd->num_filters; + ppd->num_filters ++; + + /* + * Retain a copy of the filter string... + */ + + *filter = _cupsStrRetain(string); + } + else if (!strcmp(keyword, "Throughput")) + ppd->throughput = atoi(string); + else if (!strcmp(keyword, "Font")) + { + /* + * Add this font to the list of available fonts... + */ + + if (ppd->num_fonts == 0) + tempfonts = (char **)malloc(sizeof(char *)); + else + tempfonts = (char **)realloc(ppd->fonts, + sizeof(char *) * (ppd->num_fonts + 1)); + + if (tempfonts == NULL) + { + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + ppd->fonts = tempfonts; + ppd->fonts[ppd->num_fonts] = _cupsStrAlloc(name); + ppd->num_fonts ++; + } + else if (!strncmp(keyword, "ParamCustom", 11)) + { + ppd_coption_t *coption; /* Custom option */ + ppd_cparam_t *cparam; /* Custom parameter */ + int corder; /* Order number */ + char ctype[33], /* Data type */ + cminimum[65], /* Minimum value */ + cmaximum[65]; /* Maximum value */ + + + /* + * Get the custom option and parameter... + */ + + if ((coption = ppd_get_coption(ppd, keyword + 11)) == NULL) + { + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + if ((cparam = ppd_get_cparam(coption, name, text)) == NULL) + { + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + /* + * Get the parameter data... + */ + + if (!string || + sscanf(string, "%d%32s%64s%64s", &corder, ctype, cminimum, + cmaximum) != 4) + { + cg->ppd_status = PPD_BAD_CUSTOM_PARAM; + + goto error; + } + + cparam->order = corder; + + if (!strcmp(ctype, "curve")) + { + cparam->type = PPD_CUSTOM_CURVE; + cparam->minimum.custom_curve = (float)_cupsStrScand(cminimum, NULL, loc); + cparam->maximum.custom_curve = (float)_cupsStrScand(cmaximum, NULL, loc); + } + else if (!strcmp(ctype, "int")) + { + cparam->type = PPD_CUSTOM_INT; + cparam->minimum.custom_int = atoi(cminimum); + cparam->maximum.custom_int = atoi(cmaximum); + } + else if (!strcmp(ctype, "invcurve")) + { + cparam->type = PPD_CUSTOM_INVCURVE; + cparam->minimum.custom_invcurve = (float)_cupsStrScand(cminimum, NULL, loc); + cparam->maximum.custom_invcurve = (float)_cupsStrScand(cmaximum, NULL, loc); + } + else if (!strcmp(ctype, "passcode")) + { + cparam->type = PPD_CUSTOM_PASSCODE; + cparam->minimum.custom_passcode = atoi(cminimum); + cparam->maximum.custom_passcode = atoi(cmaximum); + } + else if (!strcmp(ctype, "password")) + { + cparam->type = PPD_CUSTOM_PASSWORD; + cparam->minimum.custom_password = atoi(cminimum); + cparam->maximum.custom_password = atoi(cmaximum); + } + else if (!strcmp(ctype, "points")) + { + cparam->type = PPD_CUSTOM_POINTS; + cparam->minimum.custom_points = (float)_cupsStrScand(cminimum, NULL, loc); + cparam->maximum.custom_points = (float)_cupsStrScand(cmaximum, NULL, loc); + } + else if (!strcmp(ctype, "real")) + { + cparam->type = PPD_CUSTOM_REAL; + cparam->minimum.custom_real = (float)_cupsStrScand(cminimum, NULL, loc); + cparam->maximum.custom_real = (float)_cupsStrScand(cmaximum, NULL, loc); + } + else if (!strcmp(ctype, "string")) + { + cparam->type = PPD_CUSTOM_STRING; + cparam->minimum.custom_string = atoi(cminimum); + cparam->maximum.custom_string = atoi(cmaximum); + } + else + { + cg->ppd_status = PPD_BAD_CUSTOM_PARAM; + + goto error; + } + + /* + * Now special-case for CustomPageSize... + */ + + if (!strcmp(coption->keyword, "PageSize")) + { + if (!strcmp(name, "Width")) + { + ppd->custom_min[0] = cparam->minimum.custom_points; + ppd->custom_max[0] = cparam->maximum.custom_points; + } + else if (!strcmp(name, "Height")) + { + ppd->custom_min[1] = cparam->minimum.custom_points; + ppd->custom_max[1] = cparam->maximum.custom_points; + } + } + } + else if (!strcmp(keyword, "HWMargins")) + { + for (i = 0, sptr = string; i < 4; i ++) + ppd->custom_margins[i] = (float)_cupsStrScand(sptr, &sptr, loc); + } + else if (!strncmp(keyword, "Custom", 6) && !strcmp(name, "True") && !option) + { + ppd_option_t *custom_option; /* Custom option */ + + DEBUG_puts("2_ppdOpen: Processing Custom option..."); + + /* + * Get the option and custom option... + */ + + if (!ppd_get_coption(ppd, keyword + 6)) + { + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + if (option && !_cups_strcasecmp(option->keyword, keyword + 6)) + custom_option = option; + else + custom_option = ppdFindOption(ppd, keyword + 6); + + if (custom_option) + { + /* + * Add the "custom" option... + */ + + if ((choice = ppdFindChoice(custom_option, "Custom")) == NULL) + if ((choice = ppd_add_choice(custom_option, "Custom")) == NULL) + { + DEBUG_puts("1_ppdOpen: Unable to add Custom choice!"); + + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + strlcpy(choice->text, text[0] ? text : _("Custom"), + sizeof(choice->text)); + + choice->code = _cupsStrAlloc(string); + + if (custom_option->section == PPD_ORDER_JCL) + ppd_decode(choice->code); + } + + /* + * Now process custom page sizes specially... + */ + + if (!strcmp(keyword, "CustomPageSize")) + { + /* + * Add a "Custom" page size entry... + */ + + ppd->variable_sizes = 1; + + ppd_add_size(ppd, "Custom"); + + if (option && !_cups_strcasecmp(option->keyword, "PageRegion")) + custom_option = option; + else + custom_option = ppdFindOption(ppd, "PageRegion"); + + if (custom_option) + { + if ((choice = ppdFindChoice(custom_option, "Custom")) == NULL) + if ((choice = ppd_add_choice(custom_option, "Custom")) == NULL) + { + DEBUG_puts("1_ppdOpen: Unable to add Custom choice!"); + + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + strlcpy(choice->text, text[0] ? text : _("Custom"), + sizeof(choice->text)); + } + } + } + else if (!strcmp(keyword, "LandscapeOrientation")) + { + if (!strcmp(string, "Minus90")) + ppd->landscape = -90; + else if (!strcmp(string, "Plus90")) + ppd->landscape = 90; + } + else if (!strcmp(keyword, "Emulators") && string) + { + for (count = 1, sptr = string; sptr != NULL;) + if ((sptr = strchr(sptr, ' ')) != NULL) + { + count ++; + while (*sptr == ' ') + sptr ++; + } + + ppd->num_emulations = count; + if ((ppd->emulations = calloc(count, sizeof(ppd_emul_t))) == NULL) + { + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + for (i = 0, sptr = string; i < count; i ++) + { + for (nameptr = ppd->emulations[i].name; + *sptr != '\0' && *sptr != ' '; + sptr ++) + if (nameptr < (ppd->emulations[i].name + sizeof(ppd->emulations[i].name) - 1)) + *nameptr++ = *sptr; + + *nameptr = '\0'; + + while (*sptr == ' ') + sptr ++; + } + } + else if (!strncmp(keyword, "StartEmulator_", 14)) + { + ppd_decode(string); + + for (i = 0; i < ppd->num_emulations; i ++) + if (!strcmp(keyword + 14, ppd->emulations[i].name)) + { + ppd->emulations[i].start = string; + string = NULL; + } + } + else if (!strncmp(keyword, "StopEmulator_", 13)) + { + ppd_decode(string); + + for (i = 0; i < ppd->num_emulations; i ++) + if (!strcmp(keyword + 13, ppd->emulations[i].name)) + { + ppd->emulations[i].stop = string; + string = NULL; + } + } + else if (!strcmp(keyword, "JobPatchFile")) + { + /* + * CUPS STR #3421: Check for "*JobPatchFile: int: string" + */ + + if (isdigit(*string & 255)) + { + for (sptr = string + 1; isdigit(*sptr & 255); sptr ++); + + if (*sptr == ':') + { + /* + * Found "*JobPatchFile: int: string"... + */ + + cg->ppd_status = PPD_BAD_VALUE; + + goto error; + } + } + + if (!name[0] && cg->ppd_conform == PPD_CONFORM_STRICT) + { + /* + * Found "*JobPatchFile: string"... + */ + + cg->ppd_status = PPD_MISSING_OPTION_KEYWORD; + + goto error; + } + + if (ppd->patches == NULL) + ppd->patches = strdup(string); + else + { + temp = realloc(ppd->patches, strlen(ppd->patches) + + strlen(string) + 1); + if (temp == NULL) + { + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + ppd->patches = temp; + + strcpy(ppd->patches + strlen(ppd->patches), string); + } + } + else if (!strcmp(keyword, "OpenUI")) + { + /* + * Don't allow nesting of options... + */ + + if (option && cg->ppd_conform == PPD_CONFORM_STRICT) + { + cg->ppd_status = PPD_NESTED_OPEN_UI; + + goto error; + } + + /* + * Add an option record to the current sub-group, group, or file... + */ + + DEBUG_printf(("2_ppdOpen: name=\"%s\" (%d)", name, (int)strlen(name))); + + if (name[0] == '*') + _cups_strcpy(name, name + 1); /* Eliminate leading asterisk */ + + for (i = (int)strlen(name) - 1; i > 0 && _cups_isspace(name[i]); i --) + name[i] = '\0'; /* Eliminate trailing spaces */ + + DEBUG_printf(("2_ppdOpen: OpenUI of %s in group %s...", name, + group ? group->text : "(null)")); + + if (subgroup != NULL) + option = ppd_get_option(subgroup, name); + else if (group == NULL) + { + if ((group = ppd_get_group(ppd, "General", _("General"), cg, + encoding)) == NULL) + goto error; + + DEBUG_printf(("2_ppdOpen: Adding to group %s...", group->text)); + option = ppd_get_option(group, name); + group = NULL; + } + else + option = ppd_get_option(group, name); + + if (option == NULL) + { + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + /* + * Now fill in the initial information for the option... + */ + + if (string && !strcmp(string, "PickMany")) + option->ui = PPD_UI_PICKMANY; + else if (string && !strcmp(string, "Boolean")) + option->ui = PPD_UI_BOOLEAN; + else if (string && !strcmp(string, "PickOne")) + option->ui = PPD_UI_PICKONE; + else if (cg->ppd_conform == PPD_CONFORM_STRICT) + { + cg->ppd_status = PPD_BAD_OPEN_UI; + + goto error; + } + else + option->ui = PPD_UI_PICKONE; + + for (j = 0; j < ppd->num_attrs; j ++) + if (!strncmp(ppd->attrs[j]->name, "Default", 7) && + !strcmp(ppd->attrs[j]->name + 7, name) && + ppd->attrs[j]->value) + { + DEBUG_printf(("2_ppdOpen: Setting Default%s to %s via attribute...", + option->keyword, ppd->attrs[j]->value)); + strlcpy(option->defchoice, ppd->attrs[j]->value, + sizeof(option->defchoice)); + break; + } + + if (text[0]) + cupsCharsetToUTF8((cups_utf8_t *)option->text, text, + sizeof(option->text), encoding); + else + { + if (!strcmp(name, "PageSize")) + strlcpy(option->text, _("Media Size"), sizeof(option->text)); + else if (!strcmp(name, "MediaType")) + strlcpy(option->text, _("Media Type"), sizeof(option->text)); + else if (!strcmp(name, "InputSlot")) + strlcpy(option->text, _("Media Source"), sizeof(option->text)); + else if (!strcmp(name, "ColorModel")) + strlcpy(option->text, _("Output Mode"), sizeof(option->text)); + else if (!strcmp(name, "Resolution")) + strlcpy(option->text, _("Resolution"), sizeof(option->text)); + else + strlcpy(option->text, name, sizeof(option->text)); + } + + option->section = PPD_ORDER_ANY; + + _cupsStrFree(string); + string = NULL; + + /* + * Add a custom option choice if we have already seen a CustomFoo + * attribute... + */ + + if (!_cups_strcasecmp(name, "PageRegion")) + strcpy(custom_name, "CustomPageSize"); + else + snprintf(custom_name, sizeof(custom_name), "Custom%s", name); + + if ((custom_attr = ppdFindAttr(ppd, custom_name, "True")) != NULL) + { + if ((choice = ppdFindChoice(option, "Custom")) == NULL) + if ((choice = ppd_add_choice(option, "Custom")) == NULL) + { + DEBUG_puts("1_ppdOpen: Unable to add Custom choice!"); + + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + strlcpy(choice->text, + custom_attr->text[0] ? custom_attr->text : _("Custom"), + sizeof(choice->text)); + choice->code = _cupsStrRetain(custom_attr->value); + } + } + else if (!strcmp(keyword, "JCLOpenUI")) + { + /* + * Don't allow nesting of options... + */ + + if (option && cg->ppd_conform == PPD_CONFORM_STRICT) + { + cg->ppd_status = PPD_NESTED_OPEN_UI; + + goto error; + } + + /* + * Find the JCL group, and add if needed... + */ + + group = ppd_get_group(ppd, "JCL", _("JCL"), cg, encoding); + + if (group == NULL) + goto error; + + /* + * Add an option record to the current JCLs... + */ + + if (name[0] == '*') + _cups_strcpy(name, name + 1); + + option = ppd_get_option(group, name); + + if (option == NULL) + { + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + /* + * Now fill in the initial information for the option... + */ + + if (string && !strcmp(string, "PickMany")) + option->ui = PPD_UI_PICKMANY; + else if (string && !strcmp(string, "Boolean")) + option->ui = PPD_UI_BOOLEAN; + else if (string && !strcmp(string, "PickOne")) + option->ui = PPD_UI_PICKONE; + else + { + cg->ppd_status = PPD_BAD_OPEN_UI; + + goto error; + } + + for (j = 0; j < ppd->num_attrs; j ++) + if (!strncmp(ppd->attrs[j]->name, "Default", 7) && + !strcmp(ppd->attrs[j]->name + 7, name) && + ppd->attrs[j]->value) + { + DEBUG_printf(("2_ppdOpen: Setting Default%s to %s via attribute...", + option->keyword, ppd->attrs[j]->value)); + strlcpy(option->defchoice, ppd->attrs[j]->value, + sizeof(option->defchoice)); + break; + } + + if (text[0]) + cupsCharsetToUTF8((cups_utf8_t *)option->text, text, + sizeof(option->text), encoding); + else + strlcpy(option->text, name, sizeof(option->text)); + + option->section = PPD_ORDER_JCL; + group = NULL; + + _cupsStrFree(string); + string = NULL; + + /* + * Add a custom option choice if we have already seen a CustomFoo + * attribute... + */ + + snprintf(custom_name, sizeof(custom_name), "Custom%s", name); + + if ((custom_attr = ppdFindAttr(ppd, custom_name, "True")) != NULL) + { + if ((choice = ppd_add_choice(option, "Custom")) == NULL) + { + DEBUG_puts("1_ppdOpen: Unable to add Custom choice!"); + + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + strlcpy(choice->text, + custom_attr->text[0] ? custom_attr->text : _("Custom"), + sizeof(choice->text)); + choice->code = _cupsStrRetain(custom_attr->value); + } + } + else if (!strcmp(keyword, "CloseUI") || !strcmp(keyword, "JCLCloseUI")) + { + option = NULL; + + _cupsStrFree(string); + string = NULL; + } + else if (!strcmp(keyword, "OpenGroup")) + { + /* + * Open a new group... + */ + + if (group != NULL) + { + cg->ppd_status = PPD_NESTED_OPEN_GROUP; + + goto error; + } + + if (!string) + { + cg->ppd_status = PPD_BAD_OPEN_GROUP; + + goto error; + } + + /* + * Separate the group name from the text (name/text)... + */ + + if ((sptr = strchr(string, '/')) != NULL) + *sptr++ = '\0'; + else + sptr = string; + + /* + * Fix up the text... + */ + + ppd_decode(sptr); + + /* + * Find/add the group... + */ + + group = ppd_get_group(ppd, string, sptr, cg, encoding); + + if (group == NULL) + goto error; + + _cupsStrFree(string); + string = NULL; + } + else if (!strcmp(keyword, "CloseGroup")) + { + group = NULL; + + _cupsStrFree(string); + string = NULL; + } + else if (!strcmp(keyword, "OrderDependency")) + { + order = (float)_cupsStrScand(string, &sptr, loc); + + if (!sptr || sscanf(sptr, "%40s%40s", name, keyword) != 2) + { + cg->ppd_status = PPD_BAD_ORDER_DEPENDENCY; + + goto error; + } + + if (keyword[0] == '*') + _cups_strcpy(keyword, keyword + 1); + + if (!strcmp(name, "ExitServer")) + section = PPD_ORDER_EXIT; + else if (!strcmp(name, "Prolog")) + section = PPD_ORDER_PROLOG; + else if (!strcmp(name, "DocumentSetup")) + section = PPD_ORDER_DOCUMENT; + else if (!strcmp(name, "PageSetup")) + section = PPD_ORDER_PAGE; + else if (!strcmp(name, "JCLSetup")) + section = PPD_ORDER_JCL; + else + section = PPD_ORDER_ANY; + + if (option == NULL) + { + ppd_group_t *gtemp; + + + /* + * Only valid for Non-UI options... + */ + + for (i = ppd->num_groups, gtemp = ppd->groups; i > 0; i --, gtemp ++) + if (gtemp->text[0] == '\0') + break; + + if (i > 0) + for (i = 0; i < gtemp->num_options; i ++) + if (!strcmp(keyword, gtemp->options[i].keyword)) + { + gtemp->options[i].section = section; + gtemp->options[i].order = order; + break; + } + } + else + { + option->section = section; + option->order = order; + } + + _cupsStrFree(string); + string = NULL; + } + else if (!strncmp(keyword, "Default", 7)) + { + if (string == NULL) + continue; + + /* + * Drop UI text, if any, from value... + */ + + if (strchr(string, '/') != NULL) + *strchr(string, '/') = '\0'; + + /* + * Assign the default value as appropriate... + */ + + if (!strcmp(keyword, "DefaultColorSpace")) + { + /* + * Set default colorspace... + */ + + if (!strcmp(string, "CMY")) + ppd->colorspace = PPD_CS_CMY; + else if (!strcmp(string, "CMYK")) + ppd->colorspace = PPD_CS_CMYK; + else if (!strcmp(string, "RGB")) + ppd->colorspace = PPD_CS_RGB; + else if (!strcmp(string, "RGBK")) + ppd->colorspace = PPD_CS_RGBK; + else if (!strcmp(string, "N")) + ppd->colorspace = PPD_CS_N; + else + ppd->colorspace = PPD_CS_GRAY; + } + else if (option && !strcmp(keyword + 7, option->keyword)) + { + /* + * Set the default as part of the current option... + */ + + DEBUG_printf(("2_ppdOpen: Setting %s to %s...", keyword, string)); + + strlcpy(option->defchoice, string, sizeof(option->defchoice)); + + DEBUG_printf(("2_ppdOpen: %s is now %s...", keyword, option->defchoice)); + } + else + { + /* + * Lookup option and set if it has been defined... + */ + + ppd_option_t *toption; /* Temporary option */ + + + if ((toption = ppdFindOption(ppd, keyword + 7)) != NULL) + { + DEBUG_printf(("2_ppdOpen: Setting %s to %s...", keyword, string)); + strlcpy(toption->defchoice, string, sizeof(toption->defchoice)); + } + } + } + else if (!strcmp(keyword, "UIConstraints") || + !strcmp(keyword, "NonUIConstraints")) + { + if (!string) + { + cg->ppd_status = PPD_BAD_UI_CONSTRAINTS; + goto error; + } + + if (ppd->num_consts == 0) + constraint = calloc(2, sizeof(ppd_const_t)); + else + constraint = realloc(ppd->consts, + (ppd->num_consts + 2) * sizeof(ppd_const_t)); + + if (constraint == NULL) + { + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + ppd->consts = constraint; + constraint += ppd->num_consts; + ppd->num_consts ++; + + switch (sscanf(string, "%40s%40s%40s%40s", constraint->option1, + constraint->choice1, constraint->option2, + constraint->choice2)) + { + case 0 : /* Error */ + case 1 : /* Error */ + cg->ppd_status = PPD_BAD_UI_CONSTRAINTS; + goto error; + + case 2 : /* Two options... */ + /* + * Check for broken constraints like "* Option"... + */ + + if (cg->ppd_conform == PPD_CONFORM_STRICT && + (!strcmp(constraint->option1, "*") || + !strcmp(constraint->choice1, "*"))) + { + cg->ppd_status = PPD_BAD_UI_CONSTRAINTS; + goto error; + } + + /* + * The following strcpy's are safe, as optionN and + * choiceN are all the same size (size defined by PPD spec...) + */ + + if (constraint->option1[0] == '*') + _cups_strcpy(constraint->option1, constraint->option1 + 1); + else if (cg->ppd_conform == PPD_CONFORM_STRICT) + { + cg->ppd_status = PPD_BAD_UI_CONSTRAINTS; + goto error; + } + + if (constraint->choice1[0] == '*') + _cups_strcpy(constraint->option2, constraint->choice1 + 1); + else if (cg->ppd_conform == PPD_CONFORM_STRICT) + { + cg->ppd_status = PPD_BAD_UI_CONSTRAINTS; + goto error; + } + + constraint->choice1[0] = '\0'; + constraint->choice2[0] = '\0'; + break; + + case 3 : /* Two options, one choice... */ + /* + * Check for broken constraints like "* Option"... + */ + + if (cg->ppd_conform == PPD_CONFORM_STRICT && + (!strcmp(constraint->option1, "*") || + !strcmp(constraint->choice1, "*") || + !strcmp(constraint->option2, "*"))) + { + cg->ppd_status = PPD_BAD_UI_CONSTRAINTS; + goto error; + } + + /* + * The following _cups_strcpy's are safe, as optionN and + * choiceN are all the same size (size defined by PPD spec...) + */ + + if (constraint->option1[0] == '*') + _cups_strcpy(constraint->option1, constraint->option1 + 1); + else if (cg->ppd_conform == PPD_CONFORM_STRICT) + { + cg->ppd_status = PPD_BAD_UI_CONSTRAINTS; + goto error; + } + + if (constraint->choice1[0] == '*') + { + if (cg->ppd_conform == PPD_CONFORM_STRICT && + constraint->option2[0] == '*') + { + cg->ppd_status = PPD_BAD_UI_CONSTRAINTS; + goto error; + } + + _cups_strcpy(constraint->choice2, constraint->option2); + _cups_strcpy(constraint->option2, constraint->choice1 + 1); + constraint->choice1[0] = '\0'; + } + else + { + if (constraint->option2[0] == '*') + _cups_strcpy(constraint->option2, constraint->option2 + 1); + else if (cg->ppd_conform == PPD_CONFORM_STRICT) + { + cg->ppd_status = PPD_BAD_UI_CONSTRAINTS; + goto error; + } + + constraint->choice2[0] = '\0'; + } + break; + + case 4 : /* Two options, two choices... */ + /* + * Check for broken constraints like "* Option"... + */ + + if (cg->ppd_conform == PPD_CONFORM_STRICT && + (!strcmp(constraint->option1, "*") || + !strcmp(constraint->choice1, "*") || + !strcmp(constraint->option2, "*") || + !strcmp(constraint->choice2, "*"))) + { + cg->ppd_status = PPD_BAD_UI_CONSTRAINTS; + goto error; + } + + if (constraint->option1[0] == '*') + _cups_strcpy(constraint->option1, constraint->option1 + 1); + else if (cg->ppd_conform == PPD_CONFORM_STRICT) + { + cg->ppd_status = PPD_BAD_UI_CONSTRAINTS; + goto error; + } + + if (cg->ppd_conform == PPD_CONFORM_STRICT && + constraint->choice1[0] == '*') + { + cg->ppd_status = PPD_BAD_UI_CONSTRAINTS; + goto error; + } + + if (constraint->option2[0] == '*') + _cups_strcpy(constraint->option2, constraint->option2 + 1); + else if (cg->ppd_conform == PPD_CONFORM_STRICT) + { + cg->ppd_status = PPD_BAD_UI_CONSTRAINTS; + goto error; + } + + if (cg->ppd_conform == PPD_CONFORM_STRICT && + constraint->choice2[0] == '*') + { + cg->ppd_status = PPD_BAD_UI_CONSTRAINTS; + goto error; + } + break; + } + + /* + * Don't add this one as an attribute... + */ + + _cupsStrFree(string); + string = NULL; + } + else if (!strcmp(keyword, "PaperDimension")) + { + if ((size = ppdPageSize(ppd, name)) == NULL) + size = ppd_add_size(ppd, name); + + if (size == NULL) + { + /* + * Unable to add or find size! + */ + + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + size->width = (float)_cupsStrScand(string, &sptr, loc); + size->length = (float)_cupsStrScand(sptr, NULL, loc); + + _cupsStrFree(string); + string = NULL; + } + else if (!strcmp(keyword, "ImageableArea")) + { + if ((size = ppdPageSize(ppd, name)) == NULL) + size = ppd_add_size(ppd, name); + + if (size == NULL) + { + /* + * Unable to add or find size! + */ + + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + size->left = (float)_cupsStrScand(string, &sptr, loc); + size->bottom = (float)_cupsStrScand(sptr, &sptr, loc); + size->right = (float)_cupsStrScand(sptr, &sptr, loc); + size->top = (float)_cupsStrScand(sptr, NULL, loc); + + _cupsStrFree(string); + string = NULL; + } + else if (option != NULL && + (mask & (PPD_KEYWORD | PPD_OPTION | PPD_STRING)) == + (PPD_KEYWORD | PPD_OPTION | PPD_STRING) && + !strcmp(keyword, option->keyword)) + { + DEBUG_printf(("2_ppdOpen: group=%p, subgroup=%p", group, subgroup)); + + if (!strcmp(keyword, "PageSize")) + { + /* + * Add a page size... + */ + + if (ppdPageSize(ppd, name) == NULL) + ppd_add_size(ppd, name); + } + + /* + * Add the option choice... + */ + + if ((choice = ppd_add_choice(option, name)) == NULL) + { + cg->ppd_status = PPD_ALLOC_ERROR; + + goto error; + } + + if (text[0]) + cupsCharsetToUTF8((cups_utf8_t *)choice->text, text, + sizeof(choice->text), encoding); + else if (!strcmp(name, "True")) + strcpy(choice->text, _("Yes")); + else if (!strcmp(name, "False")) + strcpy(choice->text, _("No")); + else + strlcpy(choice->text, name, sizeof(choice->text)); + + if (option->section == PPD_ORDER_JCL) + ppd_decode(string); /* Decode quoted string */ + + choice->code = string; + string = NULL; /* Don't add as an attribute below */ + } + + /* + * Add remaining lines with keywords and string values as attributes... + */ + + if (string && + (mask & (PPD_KEYWORD | PPD_STRING)) == (PPD_KEYWORD | PPD_STRING)) + ppd_add_attr(ppd, keyword, name, text, string); + else + _cupsStrFree(string); + } + + /* + * Check for a missing CloseGroup... + */ + + if (group && cg->ppd_conform == PPD_CONFORM_STRICT) + { + cg->ppd_status = PPD_MISSING_CLOSE_GROUP; + goto error; + } + + ppd_free(line.buffer); + + /* + * Reset language preferences... + */ + +#ifdef DEBUG + if (!cupsFileEOF(fp)) + DEBUG_printf(("1_ppdOpen: Premature EOF at %lu...\n", + (unsigned long)cupsFileTell(fp))); +#endif /* DEBUG */ + + if (cg->ppd_status != PPD_OK) + { + /* + * Had an error reading the PPD file, cannot continue! + */ + + ppdClose(ppd); + + return (NULL); + } + + /* + * Update the filters array as needed... + */ + + if (!ppd_update_filters(ppd, cg)) + { + ppdClose(ppd); + + return (NULL); + } + + /* + * Create the sorted options array and set the option back-pointer for + * each choice and custom option... + */ + + ppd->options = cupsArrayNew2((cups_array_func_t)ppd_compare_options, NULL, + (cups_ahash_func_t)ppd_hash_option, + PPD_HASHSIZE); + + for (i = ppd->num_groups, group = ppd->groups; + i > 0; + i --, group ++) + { + for (j = group->num_options, option = group->options; + j > 0; + j --, option ++) + { + ppd_coption_t *coption; /* Custom option */ + + + cupsArrayAdd(ppd->options, option); + + for (k = 0; k < option->num_choices; k ++) + option->choices[k].option = option; + + if ((coption = ppdFindCustomOption(ppd, option->keyword)) != NULL) + coption->option = option; + } + } + + /* + * Create an array to track the marked choices... + */ + + ppd->marked = cupsArrayNew((cups_array_func_t)ppd_compare_choices, NULL); + + /* + * Return the PPD file structure... + */ + + return (ppd); + + /* + * Common exit point for errors to save code size... + */ + + error: + + _cupsStrFree(string); + ppd_free(line.buffer); + + ppdClose(ppd); + + return (NULL); +} + + +/* + * 'ppdOpen()' - Read a PPD file into memory. + */ + +ppd_file_t * /* O - PPD file record */ +ppdOpen(FILE *fp) /* I - File to read from */ +{ + ppd_file_t *ppd; /* PPD file record */ + cups_file_t *cf; /* CUPS file */ + + + /* + * Reopen the stdio file as a CUPS file... + */ + + if ((cf = cupsFileOpenFd(fileno(fp), "r")) == NULL) + return (NULL); + + /* + * Load the PPD file using the newer API... + */ + + ppd = _ppdOpen(cf, _PPD_LOCALIZATION_DEFAULT); + + /* + * Close the CUPS file and return the PPD... + */ + + cupsFileClose(cf); + + return (ppd); +} + + +/* + * 'ppdOpen2()' - Read a PPD file into memory. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +ppd_file_t * /* O - PPD file record or @code NULL@ if the PPD file could not be opened. */ +ppdOpen2(cups_file_t *fp) /* I - File to read from */ +{ + return _ppdOpen(fp, _PPD_LOCALIZATION_DEFAULT); +} + + +/* + * 'ppdOpenFd()' - Read a PPD file into memory. + */ + +ppd_file_t * /* O - PPD file record or @code NULL@ if the PPD file could not be opened. */ +ppdOpenFd(int fd) /* I - File to read from */ +{ + cups_file_t *fp; /* CUPS file pointer */ + ppd_file_t *ppd; /* PPD file record */ + _cups_globals_t *cg = _cupsGlobals(); + /* Global data */ + + + /* + * Set the line number to 0... + */ + + cg->ppd_line = 0; + + /* + * Range check input... + */ + + if (fd < 0) + { + cg->ppd_status = PPD_NULL_FILE; + + return (NULL); + } + + /* + * Try to open the file and parse it... + */ + + if ((fp = cupsFileOpenFd(fd, "r")) != NULL) + { + ppd = ppdOpen2(fp); + + cupsFileClose(fp); + } + else + { + cg->ppd_status = PPD_FILE_OPEN_ERROR; + ppd = NULL; + } + + return (ppd); +} + + +/* + * '_ppdOpenFile()' - Read a PPD file into memory. + */ + +ppd_file_t * /* O - PPD file record or @code NULL@ if the PPD file could not be opened. */ +_ppdOpenFile(const char *filename, /* I - File to read from */ + _ppd_localization_t localization) /* I - Localization to load */ +{ + cups_file_t *fp; /* File pointer */ + ppd_file_t *ppd; /* PPD file record */ + _cups_globals_t *cg = _cupsGlobals(); + /* Global data */ + + + /* + * Set the line number to 0... + */ + + cg->ppd_line = 0; + + /* + * Range check input... + */ + + if (filename == NULL) + { + cg->ppd_status = PPD_NULL_FILE; + + return (NULL); + } + + /* + * Try to open the file and parse it... + */ + + if ((fp = cupsFileOpen(filename, "r")) != NULL) + { + ppd = _ppdOpen(fp, localization); + + cupsFileClose(fp); + } + else + { + cg->ppd_status = PPD_FILE_OPEN_ERROR; + ppd = NULL; + } + + return (ppd); +} + + +/* + * 'ppdOpenFile()' - Read a PPD file into memory. + */ + +ppd_file_t * /* O - PPD file record or @code NULL@ if the PPD file could not be opened. */ +ppdOpenFile(const char *filename) /* I - File to read from */ +{ + return _ppdOpenFile(filename, _PPD_LOCALIZATION_DEFAULT); +} + + +/* + * 'ppdSetConformance()' - Set the conformance level for PPD files. + * + * @since CUPS 1.1.20/Mac OS X 10.4@ + */ + +void +ppdSetConformance(ppd_conform_t c) /* I - Conformance level */ +{ + _cups_globals_t *cg = _cupsGlobals(); + /* Global data */ + + + cg->ppd_conform = c; +} + + +/* + * 'ppd_add_attr()' - Add an attribute to the PPD data. + */ + +static ppd_attr_t * /* O - New attribute */ +ppd_add_attr(ppd_file_t *ppd, /* I - PPD file data */ + const char *name, /* I - Attribute name */ + const char *spec, /* I - Specifier string, if any */ + const char *text, /* I - Text string, if any */ + const char *value) /* I - Value of attribute */ +{ + ppd_attr_t **ptr, /* New array */ + *temp; /* New attribute */ + + + /* + * Range check input... + */ + + if (ppd == NULL || name == NULL || spec == NULL) + return (NULL); + + /* + * Create the array as needed... + */ + + if (!ppd->sorted_attrs) + ppd->sorted_attrs = cupsArrayNew((cups_array_func_t)ppd_compare_attrs, + NULL); + + /* + * Allocate memory for the new attribute... + */ + + if (ppd->num_attrs == 0) + ptr = malloc(sizeof(ppd_attr_t *)); + else + ptr = realloc(ppd->attrs, (ppd->num_attrs + 1) * sizeof(ppd_attr_t *)); + + if (ptr == NULL) + return (NULL); + + ppd->attrs = ptr; + ptr += ppd->num_attrs; + + if ((temp = calloc(1, sizeof(ppd_attr_t))) == NULL) + return (NULL); + + *ptr = temp; + + ppd->num_attrs ++; + + /* + * Copy data over... + */ + + strlcpy(temp->name, name, sizeof(temp->name)); + strlcpy(temp->spec, spec, sizeof(temp->spec)); + strlcpy(temp->text, text, sizeof(temp->text)); + temp->value = (char *)value; + + /* + * Add the attribute to the sorted array... + */ + + cupsArrayAdd(ppd->sorted_attrs, temp); + + /* + * Return the attribute... + */ + + return (temp); +} + + +/* + * 'ppd_add_choice()' - Add a choice to an option. + */ + +static ppd_choice_t * /* O - Named choice */ +ppd_add_choice(ppd_option_t *option, /* I - Option */ + const char *name) /* I - Name of choice */ +{ + ppd_choice_t *choice; /* Choice */ + + + if (option->num_choices == 0) + choice = malloc(sizeof(ppd_choice_t)); + else + choice = realloc(option->choices, + sizeof(ppd_choice_t) * (option->num_choices + 1)); + + if (choice == NULL) + return (NULL); + + option->choices = choice; + choice += option->num_choices; + option->num_choices ++; + + memset(choice, 0, sizeof(ppd_choice_t)); + strlcpy(choice->choice, name, sizeof(choice->choice)); + + return (choice); +} + + +/* + * 'ppd_add_size()' - Add a page size. + */ + +static ppd_size_t * /* O - Named size */ +ppd_add_size(ppd_file_t *ppd, /* I - PPD file */ + const char *name) /* I - Name of size */ +{ + ppd_size_t *size; /* Size */ + + + if (ppd->num_sizes == 0) + size = malloc(sizeof(ppd_size_t)); + else + size = realloc(ppd->sizes, sizeof(ppd_size_t) * (ppd->num_sizes + 1)); + + if (size == NULL) + return (NULL); + + ppd->sizes = size; + size += ppd->num_sizes; + ppd->num_sizes ++; + + memset(size, 0, sizeof(ppd_size_t)); + strlcpy(size->name, name, sizeof(size->name)); + + return (size); +} + + +/* + * 'ppd_compare_attrs()' - Compare two attributes. + */ + +static int /* O - Result of comparison */ +ppd_compare_attrs(ppd_attr_t *a, /* I - First attribute */ + ppd_attr_t *b) /* I - Second attribute */ +{ + return (_cups_strcasecmp(a->name, b->name)); +} + + +/* + * 'ppd_compare_choices()' - Compare two choices... + */ + +static int /* O - Result of comparison */ +ppd_compare_choices(ppd_choice_t *a, /* I - First choice */ + ppd_choice_t *b) /* I - Second choice */ +{ + return (strcmp(a->option->keyword, b->option->keyword)); +} + + +/* + * 'ppd_compare_coptions()' - Compare two custom options. + */ + +static int /* O - Result of comparison */ +ppd_compare_coptions(ppd_coption_t *a, /* I - First option */ + ppd_coption_t *b) /* I - Second option */ +{ + return (_cups_strcasecmp(a->keyword, b->keyword)); +} + + +/* + * 'ppd_compare_options()' - Compare two options. + */ + +static int /* O - Result of comparison */ +ppd_compare_options(ppd_option_t *a, /* I - First option */ + ppd_option_t *b) /* I - Second option */ +{ + return (_cups_strcasecmp(a->keyword, b->keyword)); +} + + +/* + * 'ppd_decode()' - Decode a string value... + */ + +static int /* O - Length of decoded string */ +ppd_decode(char *string) /* I - String to decode */ +{ + char *inptr, /* Input pointer */ + *outptr; /* Output pointer */ + + + inptr = string; + outptr = string; + + while (*inptr != '\0') + if (*inptr == '<' && isxdigit(inptr[1] & 255)) + { + /* + * Convert hex to 8-bit values... + */ + + inptr ++; + while (isxdigit(*inptr & 255)) + { + if (_cups_isalpha(*inptr)) + *outptr = (tolower(*inptr) - 'a' + 10) << 4; + else + *outptr = (*inptr - '0') << 4; + + inptr ++; + + if (!isxdigit(*inptr & 255)) + break; + + if (_cups_isalpha(*inptr)) + *outptr |= tolower(*inptr) - 'a' + 10; + else + *outptr |= *inptr - '0'; + + inptr ++; + outptr ++; + } + + while (*inptr != '>' && *inptr != '\0') + inptr ++; + while (*inptr == '>') + inptr ++; + } + else + *outptr++ = *inptr++; + + *outptr = '\0'; + + return ((int)(outptr - string)); +} + + +/* + * 'ppd_free_filters()' - Free the filters array. + */ + +static void +ppd_free_filters(ppd_file_t *ppd) /* I - PPD file */ +{ + int i; /* Looping var */ + char **filter; /* Current filter */ + + + if (ppd->num_filters > 0) + { + for (i = ppd->num_filters, filter = ppd->filters; i > 0; i --, filter ++) + _cupsStrFree(*filter); + + ppd_free(ppd->filters); + + ppd->num_filters = 0; + ppd->filters = NULL; + } +} + + +/* + * 'ppd_free_group()' - Free a single UI group. + */ + +static void +ppd_free_group(ppd_group_t *group) /* I - Group to free */ +{ + int i; /* Looping var */ + ppd_option_t *option; /* Current option */ + ppd_group_t *subgroup; /* Current sub-group */ + + + if (group->num_options > 0) + { + for (i = group->num_options, option = group->options; + i > 0; + i --, option ++) + ppd_free_option(option); + + ppd_free(group->options); + } + + if (group->num_subgroups > 0) + { + for (i = group->num_subgroups, subgroup = group->subgroups; + i > 0; + i --, subgroup ++) + ppd_free_group(subgroup); + + ppd_free(group->subgroups); + } +} + + +/* + * 'ppd_free_option()' - Free a single option. + */ + +static void +ppd_free_option(ppd_option_t *option) /* I - Option to free */ +{ + int i; /* Looping var */ + ppd_choice_t *choice; /* Current choice */ + + + if (option->num_choices > 0) + { + for (i = option->num_choices, choice = option->choices; + i > 0; + i --, choice ++) + { + _cupsStrFree(choice->code); + } + + ppd_free(option->choices); + } +} + + +/* + * 'ppd_get_coption()' - Get a custom option record. + */ + +static ppd_coption_t * /* O - Custom option... */ +ppd_get_coption(ppd_file_t *ppd, /* I - PPD file */ + const char *name) /* I - Name of option */ +{ + ppd_coption_t *copt; /* New custom option */ + + + /* + * See if the option already exists... + */ + + if ((copt = ppdFindCustomOption(ppd, name)) != NULL) + return (copt); + + /* + * Not found, so create the custom option record... + */ + + if ((copt = calloc(1, sizeof(ppd_coption_t))) == NULL) + return (NULL); + + strlcpy(copt->keyword, name, sizeof(copt->keyword)); + + copt->params = cupsArrayNew((cups_array_func_t)NULL, NULL); + + cupsArrayAdd(ppd->coptions, copt); + + /* + * Return the new record... + */ + + return (copt); +} + + +/* + * 'ppd_get_cparam()' - Get a custom parameter record. + */ + +static ppd_cparam_t * /* O - Extended option... */ +ppd_get_cparam(ppd_coption_t *opt, /* I - PPD file */ + const char *param, /* I - Name of parameter */ + const char *text) /* I - Human-readable text */ +{ + ppd_cparam_t *cparam; /* New custom parameter */ + + + /* + * See if the parameter already exists... + */ + + if ((cparam = ppdFindCustomParam(opt, param)) != NULL) + return (cparam); + + /* + * Not found, so create the custom parameter record... + */ + + if ((cparam = calloc(1, sizeof(ppd_cparam_t))) == NULL) + return (NULL); + + strlcpy(cparam->name, param, sizeof(cparam->name)); + strlcpy(cparam->text, text[0] ? text : param, sizeof(cparam->text)); + + /* + * Add this record to the array... + */ + + cupsArrayAdd(opt->params, cparam); + + /* + * Return the new record... + */ + + return (cparam); +} + + +/* + * 'ppd_get_group()' - Find or create the named group as needed. + */ + +static ppd_group_t * /* O - Named group */ +ppd_get_group(ppd_file_t *ppd, /* I - PPD file */ + const char *name, /* I - Name of group */ + const char *text, /* I - Text for group */ + _cups_globals_t *cg, /* I - Global data */ + cups_encoding_t encoding) /* I - Encoding of text */ +{ + int i; /* Looping var */ + ppd_group_t *group; /* Group */ + + + DEBUG_printf(("7ppd_get_group(ppd=%p, name=\"%s\", text=\"%s\", cg=%p)", + ppd, name, text, cg)); + + for (i = ppd->num_groups, group = ppd->groups; i > 0; i --, group ++) + if (!strcmp(group->name, name)) + break; + + if (i == 0) + { + DEBUG_printf(("8ppd_get_group: Adding group %s...", name)); + + if (cg->ppd_conform == PPD_CONFORM_STRICT && strlen(text) >= sizeof(group->text)) + { + cg->ppd_status = PPD_ILLEGAL_TRANSLATION; + + return (NULL); + } + + if (ppd->num_groups == 0) + group = malloc(sizeof(ppd_group_t)); + else + group = realloc(ppd->groups, + (ppd->num_groups + 1) * sizeof(ppd_group_t)); + + if (group == NULL) + { + cg->ppd_status = PPD_ALLOC_ERROR; + + return (NULL); + } + + ppd->groups = group; + group += ppd->num_groups; + ppd->num_groups ++; + + memset(group, 0, sizeof(ppd_group_t)); + strlcpy(group->name, name, sizeof(group->name)); + + cupsCharsetToUTF8((cups_utf8_t *)group->text, text, + sizeof(group->text), encoding); + } + + return (group); +} + + +/* + * 'ppd_get_option()' - Find or create the named option as needed. + */ + +static ppd_option_t * /* O - Named option */ +ppd_get_option(ppd_group_t *group, /* I - Group */ + const char *name) /* I - Name of option */ +{ + int i; /* Looping var */ + ppd_option_t *option; /* Option */ + + + DEBUG_printf(("7ppd_get_option(group=%p(\"%s\"), name=\"%s\")", + group, group->name, name)); + + for (i = group->num_options, option = group->options; i > 0; i --, option ++) + if (!strcmp(option->keyword, name)) + break; + + if (i == 0) + { + if (group->num_options == 0) + option = malloc(sizeof(ppd_option_t)); + else + option = realloc(group->options, + (group->num_options + 1) * sizeof(ppd_option_t)); + + if (option == NULL) + return (NULL); + + group->options = option; + option += group->num_options; + group->num_options ++; + + memset(option, 0, sizeof(ppd_option_t)); + strlcpy(option->keyword, name, sizeof(option->keyword)); + } + + return (option); +} + + +/* + * 'ppd_hash_option()' - Generate a hash of the option name... + */ + +static int /* O - Hash index */ +ppd_hash_option(ppd_option_t *option) /* I - Option */ +{ + int hash = 0; /* Hash index */ + const char *k; /* Pointer into keyword */ + + + for (hash = option->keyword[0], k = option->keyword + 1; *k;) + hash = 33 * hash + *k++; + + return (hash & 511); +} + + +/* + * 'ppd_read()' - Read a line from a PPD file, skipping comment lines as + * necessary. + */ + +static int /* O - Bitmask of fields read */ +ppd_read(cups_file_t *fp, /* I - File to read from */ + _ppd_line_t *line, /* I - Line buffer */ + char *keyword, /* O - Keyword from line */ + char *option, /* O - Option from line */ + char *text, /* O - Human-readable text from line */ + char **string, /* O - Code/string data */ + int ignoreblank, /* I - Ignore blank lines? */ + _cups_globals_t *cg) /* I - Global data */ +{ + int ch, /* Character from file */ + col, /* Column in line */ + colon, /* Colon seen? */ + endquote, /* Waiting for an end quote */ + mask, /* Mask to be returned */ + startline, /* Start line */ + textlen; /* Length of text */ + char *keyptr, /* Keyword pointer */ + *optptr, /* Option pointer */ + *textptr, /* Text pointer */ + *strptr, /* Pointer into string */ + *lineptr; /* Current position in line buffer */ + + + /* + * Now loop until we have a valid line... + */ + + *string = NULL; + col = 0; + startline = cg->ppd_line + 1; + + if (!line->buffer) + { + line->bufsize = 1024; + line->buffer = malloc(1024); + + if (!line->buffer) + return (0); + } + + do + { + /* + * Read the line... + */ + + lineptr = line->buffer; + endquote = 0; + colon = 0; + + while ((ch = cupsFileGetChar(fp)) != EOF) + { + if (lineptr >= (line->buffer + line->bufsize - 1)) + { + /* + * Expand the line buffer... + */ + + char *temp; /* Temporary line pointer */ + + + line->bufsize += 1024; + if (line->bufsize > 262144) + { + /* + * Don't allow lines longer than 256k! + */ + + cg->ppd_line = startline; + cg->ppd_status = PPD_LINE_TOO_LONG; + + return (0); + } + + temp = realloc(line->buffer, line->bufsize); + if (!temp) + { + cg->ppd_line = startline; + cg->ppd_status = PPD_LINE_TOO_LONG; + + return (0); + } + + lineptr = temp + (lineptr - line->buffer); + line->buffer = temp; + } + + if (ch == '\r' || ch == '\n') + { + /* + * Line feed or carriage return... + */ + + cg->ppd_line ++; + col = 0; + + if (ch == '\r') + { + /* + * Check for a trailing line feed... + */ + + if ((ch = cupsFilePeekChar(fp)) == EOF) + { + ch = '\n'; + break; + } + + if (ch == 0x0a) + cupsFileGetChar(fp); + } + + if (lineptr == line->buffer && ignoreblank) + continue; /* Skip blank lines */ + + ch = '\n'; + + if (!endquote) /* Continue for multi-line text */ + break; + + *lineptr++ = '\n'; + } + else if (ch < ' ' && ch != '\t' && cg->ppd_conform == PPD_CONFORM_STRICT) + { + /* + * Other control characters... + */ + + cg->ppd_line = startline; + cg->ppd_status = PPD_ILLEGAL_CHARACTER; + + return (0); + } + else if (ch != 0x1a) + { + /* + * Any other character... + */ + + *lineptr++ = ch; + col ++; + + if (col > (PPD_MAX_LINE - 1)) + { + /* + * Line is too long... + */ + + cg->ppd_line = startline; + cg->ppd_status = PPD_LINE_TOO_LONG; + + return (0); + } + + if (ch == ':' && strncmp(line->buffer, "*%", 2) != 0) + colon = 1; + + if (ch == '\"' && colon) + endquote = !endquote; + } + } + + if (endquote) + { + /* + * Didn't finish this quoted string... + */ + + while ((ch = cupsFileGetChar(fp)) != EOF) + if (ch == '\"') + break; + else if (ch == '\r' || ch == '\n') + { + cg->ppd_line ++; + col = 0; + + if (ch == '\r') + { + /* + * Check for a trailing line feed... + */ + + if ((ch = cupsFilePeekChar(fp)) == EOF) + break; + if (ch == 0x0a) + cupsFileGetChar(fp); + } + } + else if (ch < ' ' && ch != '\t' && cg->ppd_conform == PPD_CONFORM_STRICT) + { + /* + * Other control characters... + */ + + cg->ppd_line = startline; + cg->ppd_status = PPD_ILLEGAL_CHARACTER; + + return (0); + } + else if (ch != 0x1a) + { + col ++; + + if (col > (PPD_MAX_LINE - 1)) + { + /* + * Line is too long... + */ + + cg->ppd_line = startline; + cg->ppd_status = PPD_LINE_TOO_LONG; + + return (0); + } + } + } + + if (ch != '\n') + { + /* + * Didn't finish this line... + */ + + while ((ch = cupsFileGetChar(fp)) != EOF) + if (ch == '\r' || ch == '\n') + { + /* + * Line feed or carriage return... + */ + + cg->ppd_line ++; + col = 0; + + if (ch == '\r') + { + /* + * Check for a trailing line feed... + */ + + if ((ch = cupsFilePeekChar(fp)) == EOF) + break; + if (ch == 0x0a) + cupsFileGetChar(fp); + } + + break; + } + else if (ch < ' ' && ch != '\t' && cg->ppd_conform == PPD_CONFORM_STRICT) + { + /* + * Other control characters... + */ + + cg->ppd_line = startline; + cg->ppd_status = PPD_ILLEGAL_CHARACTER; + + return (0); + } + else if (ch != 0x1a) + { + col ++; + + if (col > (PPD_MAX_LINE - 1)) + { + /* + * Line is too long... + */ + + cg->ppd_line = startline; + cg->ppd_status = PPD_LINE_TOO_LONG; + + return (0); + } + } + } + + if (lineptr > line->buffer && lineptr[-1] == '\n') + lineptr --; + + *lineptr = '\0'; + + DEBUG_printf(("9ppd_read: LINE=\"%s\"", line->buffer)); + + /* + * The dynamically created PPDs for older style Mac OS X + * drivers include a large blob of data inserted as comments + * at the end of the file. As an optimization we can stop + * reading the PPD when we get to the start of this data. + */ + + if (!strcmp(line->buffer, "*%APLWORKSET START")) + return (0); + + if (ch == EOF && lineptr == line->buffer) + return (0); + + /* + * Now parse it... + */ + + mask = 0; + lineptr = line->buffer + 1; + + keyword[0] = '\0'; + option[0] = '\0'; + text[0] = '\0'; + *string = NULL; + + if ((!line->buffer[0] || /* Blank line */ + !strncmp(line->buffer, "*%", 2) || /* Comment line */ + !strcmp(line->buffer, "*End")) && /* End of multi-line string */ + ignoreblank) /* Ignore these? */ + { + startline = cg->ppd_line + 1; + continue; + } + + if (!strcmp(line->buffer, "*")) /* (Bad) comment line */ + { + if (cg->ppd_conform == PPD_CONFORM_RELAXED) + { + startline = cg->ppd_line + 1; + continue; + } + else + { + cg->ppd_line = startline; + cg->ppd_status = PPD_ILLEGAL_MAIN_KEYWORD; + + return (0); + } + } + + if (line->buffer[0] != '*') /* All lines start with an asterisk */ + { + /* + * Allow lines consisting of just whitespace... + */ + + for (lineptr = line->buffer; *lineptr; lineptr ++) + if (*lineptr && !_cups_isspace(*lineptr)) + break; + + if (*lineptr) + { + cg->ppd_status = PPD_MISSING_ASTERISK; + return (0); + } + else if (ignoreblank) + continue; + else + return (0); + } + + /* + * Get a keyword... + */ + + keyptr = keyword; + + while (*lineptr && *lineptr != ':' && !_cups_isspace(*lineptr)) + { + if (*lineptr <= ' ' || *lineptr > 126 || *lineptr == '/' || + (keyptr - keyword) >= (PPD_MAX_NAME - 1)) + { + cg->ppd_status = PPD_ILLEGAL_MAIN_KEYWORD; + return (0); + } + + *keyptr++ = *lineptr++; + } + + *keyptr = '\0'; + + if (!strcmp(keyword, "End")) + continue; + + mask |= PPD_KEYWORD; + + if (_cups_isspace(*lineptr)) + { + /* + * Get an option name... + */ + + while (_cups_isspace(*lineptr)) + lineptr ++; + + optptr = option; + + while (*lineptr && !_cups_isspace(*lineptr) && *lineptr != ':' && + *lineptr != '/') + { + if (*lineptr <= ' ' || *lineptr > 126 || + (optptr - option) >= (PPD_MAX_NAME - 1)) + { + cg->ppd_status = PPD_ILLEGAL_OPTION_KEYWORD; + return (0); + } + + *optptr++ = *lineptr++; + } + + *optptr = '\0'; + + if (_cups_isspace(*lineptr) && cg->ppd_conform == PPD_CONFORM_STRICT) + { + cg->ppd_status = PPD_ILLEGAL_WHITESPACE; + return (0); + } + + while (_cups_isspace(*lineptr)) + lineptr ++; + + mask |= PPD_OPTION; + + if (*lineptr == '/') + { + /* + * Get human-readable text... + */ + + lineptr ++; + + textptr = text; + + while (*lineptr != '\0' && *lineptr != '\n' && *lineptr != ':') + { + if (((unsigned char)*lineptr < ' ' && *lineptr != '\t') || + (textptr - text) >= (PPD_MAX_LINE - 1)) + { + cg->ppd_status = PPD_ILLEGAL_TRANSLATION; + return (0); + } + + *textptr++ = *lineptr++; + } + + *textptr = '\0'; + textlen = ppd_decode(text); + + if (textlen > PPD_MAX_TEXT && cg->ppd_conform == PPD_CONFORM_STRICT) + { + cg->ppd_status = PPD_ILLEGAL_TRANSLATION; + return (0); + } + + mask |= PPD_TEXT; + } + } + + if (_cups_isspace(*lineptr) && cg->ppd_conform == PPD_CONFORM_STRICT) + { + cg->ppd_status = PPD_ILLEGAL_WHITESPACE; + return (0); + } + + while (_cups_isspace(*lineptr)) + lineptr ++; + + if (*lineptr == ':') + { + /* + * Get string after triming leading and trailing whitespace... + */ + + lineptr ++; + while (_cups_isspace(*lineptr)) + lineptr ++; + + strptr = lineptr + strlen(lineptr) - 1; + while (strptr >= lineptr && _cups_isspace(*strptr)) + *strptr-- = '\0'; + + if (*strptr == '\"') + { + /* + * Quoted string by itself, remove quotes... + */ + + *strptr = '\0'; + lineptr ++; + } + + *string = _cupsStrAlloc(lineptr); + + mask |= PPD_STRING; + } + } + while (mask == 0); + + return (mask); +} + + +/* + * 'ppd_update_filters()' - Update the filters array as needed. + * + * This function re-populates the filters array with cupsFilter2 entries that + * have been stripped of the destination MIME media types and any maxsize hints. + * + * (All for backwards-compatibility) + */ + +static int /* O - 1 on success, 0 on failure */ +ppd_update_filters(ppd_file_t *ppd,/* I - PPD file */ + _cups_globals_t *cg) /* I - Global data */ +{ + ppd_attr_t *attr; /* Current cupsFilter2 value */ + char srcsuper[16], /* Source MIME media type */ + srctype[256], + dstsuper[16], /* Destination MIME media type */ + dsttype[256], + program[1024], /* Command to run */ + *ptr, /* Pointer into command to run */ + buffer[1024], /* Re-written cupsFilter value */ + **filter; /* Current filter */ + int cost; /* Cost of filter */ + + + DEBUG_printf(("4ppd_update_filters(ppd=%p, cg=%p)", ppd, cg)); + + /* + * See if we have any cupsFilter2 lines... + */ + + if ((attr = ppdFindAttr(ppd, "cupsFilter2", NULL)) == NULL) + { + DEBUG_puts("5ppd_update_filters: No cupsFilter2 keywords present."); + return (1); + } + + /* + * Yes, free the cupsFilter-defined filters and re-build... + */ + + ppd_free_filters(ppd); + + do + { + /* + * Parse the cupsFilter2 string: + * + * src/type dst/type cost program + * src/type dst/type cost maxsize(n) program + */ + + DEBUG_printf(("5ppd_update_filters: cupsFilter2=\"%s\"", attr->value)); + + if (sscanf(attr->value, "%15[^/]/%255s%*[ \t]%15[^/]/%255s%d%*[ \t]%1023[^\n]", + srcsuper, srctype, dstsuper, dsttype, &cost, program) != 6) + { + DEBUG_puts("5ppd_update_filters: Bad cupsFilter2 line."); + cg->ppd_status = PPD_BAD_VALUE; + + return (0); + } + + DEBUG_printf(("5ppd_update_filters: srcsuper=\"%s\", srctype=\"%s\", " + "dstsuper=\"%s\", dsttype=\"%s\", cost=%d, program=\"%s\"", + srcsuper, srctype, dstsuper, dsttype, cost, program)); + + if (!strncmp(program, "maxsize(", 8) && + (ptr = strchr(program + 8, ')')) != NULL) + { + DEBUG_puts("5ppd_update_filters: Found maxsize(nnn)."); + + ptr ++; + while (_cups_isspace(*ptr)) + ptr ++; + + _cups_strcpy(program, ptr); + DEBUG_printf(("5ppd_update_filters: New program=\"%s\"", program)); + } + + /* + * Convert to cupsFilter format: + * + * src/type cost program + */ + + snprintf(buffer, sizeof(buffer), "%s/%s %d %s", srcsuper, srctype, cost, + program); + DEBUG_printf(("5ppd_update_filters: Adding \"%s\".", buffer)); + + /* + * Add a cupsFilter-compatible string to the filters array. + */ + + if (ppd->num_filters == 0) + filter = malloc(sizeof(char *)); + else + filter = realloc(ppd->filters, sizeof(char *) * (ppd->num_filters + 1)); + + if (filter == NULL) + { + DEBUG_puts("5ppd_update_filters: Out of memory."); + cg->ppd_status = PPD_ALLOC_ERROR; + + return (0); + } + + ppd->filters = filter; + filter += ppd->num_filters; + ppd->num_filters ++; + + *filter = _cupsStrAlloc(buffer); + } + while ((attr = ppdFindNextAttr(ppd, "cupsFilter2", NULL)) != NULL); + + DEBUG_puts("5ppd_update_filters: Completed OK."); + return (1); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/ppd.h b/cups/ppd.h new file mode 100644 index 0000000000..049b69a80e --- /dev/null +++ b/cups/ppd.h @@ -0,0 +1,468 @@ +/* + * "$Id$" + * + * PostScript Printer Description definitions for CUPS. + * + * THESE APIS ARE DEPRECATED. TO COMPILE WITHOUT WARNINGS ADD + * -D_PPD_DEPRECATED="" TO YOUR COMPILE OPTIONS. THIS HEADER AND THESE + * FUNCTIONS WILL BE REMOVED IN A FUTURE RELEASE OF CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * PostScript is a trademark of Adobe Systems, Inc. + * + * This code and any derivative of it may be used and distributed + * freely under the terms of the GNU General Public License when + * used with GNU Ghostscript or its derivatives. Use of the code + * (or any derivative of it) with software other than GNU + * GhostScript (or its derivatives) is governed by the CUPS license + * agreement. + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_PPD_H_ +# define _CUPS_PPD_H_ + +/* + * Include necessary headers... + */ + +# include +# include "cups.h" +# include "array.h" +# include "file.h" + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Define _PPD_DEPRECATED to silence the warnings about PPD functions being + * deprecated... + */ + +# ifndef _PPD_DEPRECATED +# define _PPD_DEPRECATED _CUPS_DEPRECATED +# endif /* !_PPD_DEPRECATED */ + + +/* + * PPD version... + */ + +# define PPD_VERSION 4.3 /* Kept in sync with Adobe version number */ + + +/* + * PPD size limits (defined in Adobe spec) + */ + +# define PPD_MAX_NAME 41 /* Maximum size of name + 1 for nul */ +# define PPD_MAX_TEXT 81 /* Maximum size of text + 1 for nul */ +# define PPD_MAX_LINE 256 /* Maximum size of line + 1 for nul */ + + +/* + * Types and structures... + */ + +typedef enum ppd_ui_e /**** UI Types ****/ +{ + PPD_UI_BOOLEAN, /* True or False option */ + PPD_UI_PICKONE, /* Pick one from a list */ + PPD_UI_PICKMANY /* Pick zero or more from a list */ +} ppd_ui_t; + +typedef enum ppd_section_e /**** Order dependency sections ****/ +{ + PPD_ORDER_ANY, /* Option code can be anywhere in the file */ + PPD_ORDER_DOCUMENT, /* ... must be in the DocumentSetup section */ + PPD_ORDER_EXIT, /* ... must be sent prior to the document */ + PPD_ORDER_JCL, /* ... must be sent as a JCL command */ + PPD_ORDER_PAGE, /* ... must be in the PageSetup section */ + PPD_ORDER_PROLOG /* ... must be in the Prolog section */ +} ppd_section_t; + +typedef enum ppd_cs_e /**** Colorspaces ****/ +{ + PPD_CS_CMYK = -4, /* CMYK colorspace */ + PPD_CS_CMY, /* CMY colorspace */ + PPD_CS_GRAY = 1, /* Grayscale colorspace */ + PPD_CS_RGB = 3, /* RGB colorspace */ + PPD_CS_RGBK, /* RGBK (K = gray) colorspace */ + PPD_CS_N /* DeviceN colorspace */ +} ppd_cs_t; + +typedef enum ppd_status_e /**** Status Codes @since CUPS 1.1.19/Mac OS X 10.3@ ****/ +{ + PPD_OK = 0, /* OK */ + PPD_FILE_OPEN_ERROR, /* Unable to open PPD file */ + PPD_NULL_FILE, /* NULL PPD file pointer */ + PPD_ALLOC_ERROR, /* Memory allocation error */ + PPD_MISSING_PPDADOBE4, /* Missing PPD-Adobe-4.x header */ + PPD_MISSING_VALUE, /* Missing value string */ + PPD_INTERNAL_ERROR, /* Internal error */ + PPD_BAD_OPEN_GROUP, /* Bad OpenGroup */ + PPD_NESTED_OPEN_GROUP, /* OpenGroup without a CloseGroup first */ + PPD_BAD_OPEN_UI, /* Bad OpenUI/JCLOpenUI */ + PPD_NESTED_OPEN_UI, /* OpenUI/JCLOpenUI without a CloseUI/JCLCloseUI first */ + PPD_BAD_ORDER_DEPENDENCY, /* Bad OrderDependency */ + PPD_BAD_UI_CONSTRAINTS, /* Bad UIConstraints */ + PPD_MISSING_ASTERISK, /* Missing asterisk in column 0 */ + PPD_LINE_TOO_LONG, /* Line longer than 255 chars */ + PPD_ILLEGAL_CHARACTER, /* Illegal control character */ + PPD_ILLEGAL_MAIN_KEYWORD, /* Illegal main keyword string */ + PPD_ILLEGAL_OPTION_KEYWORD, /* Illegal option keyword string */ + PPD_ILLEGAL_TRANSLATION, /* Illegal translation string */ + PPD_ILLEGAL_WHITESPACE, /* Illegal whitespace character */ + PPD_BAD_CUSTOM_PARAM, /* Bad custom parameter */ + PPD_MISSING_OPTION_KEYWORD, /* Missing option keyword */ + PPD_BAD_VALUE, /* Bad value string */ + PPD_MISSING_CLOSE_GROUP, /* Missing CloseGroup */ + PPD_MAX_STATUS /* @private@ */ +} ppd_status_t; + +enum ppd_conform_e /**** Conformance Levels @since CUPS 1.1.19/Mac OS X 10.3@ ****/ +{ + PPD_CONFORM_RELAXED, /* Relax whitespace and control char */ + PPD_CONFORM_STRICT /* Require strict conformance */ +}; + +typedef enum ppd_conform_e ppd_conform_t; + /**** Conformance Levels @since CUPS 1.1.19/Mac OS X 10.3@ ****/ + +typedef struct ppd_attr_s /**** PPD Attribute Structure @since CUPS 1.1.19/Mac OS X 10.3@ ****/ +{ + char name[PPD_MAX_NAME]; /* Name of attribute (cupsXYZ) */ + char spec[PPD_MAX_NAME]; /* Specifier string, if any */ + char text[PPD_MAX_TEXT]; /* Human-readable text, if any */ + char *value; /* Value string */ +} ppd_attr_t; + +typedef struct ppd_option_s ppd_option_t; + /**** Options ****/ + +typedef struct ppd_choice_s /**** Option choices ****/ +{ + char marked; /* 0 if not selected, 1 otherwise */ + char choice[PPD_MAX_NAME]; /* Computer-readable option name */ + char text[PPD_MAX_TEXT]; /* Human-readable option name */ + char *code; /* Code to send for this option */ + ppd_option_t *option; /* Pointer to parent option structure */ +} ppd_choice_t; + +struct ppd_option_s /**** Options ****/ +{ + char conflicted; /* 0 if no conflicts exist, 1 otherwise */ + char keyword[PPD_MAX_NAME]; /* Option keyword name ("PageSize", etc.) */ + char defchoice[PPD_MAX_NAME];/* Default option choice */ + char text[PPD_MAX_TEXT]; /* Human-readable text */ + ppd_ui_t ui; /* Type of UI option */ + ppd_section_t section; /* Section for command */ + float order; /* Order number */ + int num_choices; /* Number of option choices */ + ppd_choice_t *choices; /* Option choices */ +}; + +typedef struct ppd_group_s /**** Groups ****/ +{ + /**** Group text strings are limited to 39 chars + nul in order to + **** preserve binary compatibility and allow applications to get + **** the group's keyword name. + ****/ + char text[PPD_MAX_TEXT - PPD_MAX_NAME]; + /* Human-readable group name */ + char name[PPD_MAX_NAME]; /* Group name @since CUPS 1.1.18/Mac OS X 10.3@ */ + int num_options; /* Number of options */ + ppd_option_t *options; /* Options */ + int num_subgroups; /* Number of sub-groups */ + struct ppd_group_s *subgroups; /* Sub-groups (max depth = 1) */ +} ppd_group_t; + +typedef struct ppd_const_s /**** Constraints ****/ +{ + char option1[PPD_MAX_NAME]; /* First keyword */ + char choice1[PPD_MAX_NAME]; /* First option/choice (blank for all) */ + char option2[PPD_MAX_NAME]; /* Second keyword */ + char choice2[PPD_MAX_NAME]; /* Second option/choice (blank for all) */ +} ppd_const_t; + +typedef struct ppd_size_s /**** Page Sizes ****/ +{ + int marked; /* Page size selected? */ + char name[PPD_MAX_NAME]; /* Media size option */ + float width; /* Width of media in points */ + float length; /* Length of media in points */ + float left; /* Left printable margin in points */ + float bottom; /* Bottom printable margin in points */ + float right; /* Right printable margin in points */ + float top; /* Top printable margin in points */ +} ppd_size_t; + +typedef struct ppd_emul_s /**** Emulators ****/ +{ + char name[PPD_MAX_NAME]; /* Emulator name */ + char *start; /* Code to switch to this emulation */ + char *stop; /* Code to stop this emulation */ +} ppd_emul_t; + +typedef struct ppd_profile_s /**** sRGB Color Profiles ****/ +{ + char resolution[PPD_MAX_NAME]; + /* Resolution or "-" */ + char media_type[PPD_MAX_NAME]; + /* Media type or "-" */ + float density; /* Ink density to use */ + float gamma; /* Gamma correction to use */ + float matrix[3][3]; /* Transform matrix */ +} ppd_profile_t; + +/**** New in CUPS 1.2/Mac OS X 10.5 ****/ +typedef enum ppd_cptype_e /**** Custom Parameter Type @since CUPS 1.2/Mac OS X 10.5@ ****/ +{ + PPD_CUSTOM_CURVE, /* Curve value for f(x) = x^value */ + PPD_CUSTOM_INT, /* Integer number value */ + PPD_CUSTOM_INVCURVE, /* Curve value for f(x) = x^(1/value) */ + PPD_CUSTOM_PASSCODE, /* String of (hidden) numbers */ + PPD_CUSTOM_PASSWORD, /* String of (hidden) characters */ + PPD_CUSTOM_POINTS, /* Measurement value in points */ + PPD_CUSTOM_REAL, /* Real number value */ + PPD_CUSTOM_STRING /* String of characters */ +} ppd_cptype_t; + +typedef union ppd_cplimit_u /**** Custom Parameter Limit @since CUPS 1.2/Mac OS X 10.5@ ****/ +{ + float custom_curve; /* Gamma value */ + int custom_int; /* Integer value */ + float custom_invcurve; /* Gamma value */ + int custom_passcode; /* Passcode length */ + int custom_password; /* Password length */ + float custom_points; /* Measurement value */ + float custom_real; /* Real value */ + int custom_string; /* String length */ +} ppd_cplimit_t; + +typedef union ppd_cpvalue_u /**** Custom Parameter Value @since CUPS 1.2/Mac OS X 10.5@ ****/ +{ + float custom_curve; /* Gamma value */ + int custom_int; /* Integer value */ + float custom_invcurve; /* Gamma value */ + char *custom_passcode; /* Passcode value */ + char *custom_password; /* Password value */ + float custom_points; /* Measurement value */ + float custom_real; /* Real value */ + char *custom_string; /* String value */ +} ppd_cpvalue_t; + +typedef struct ppd_cparam_s /**** Custom Parameter @since CUPS 1.2/Mac OS X 10.5@ ****/ +{ + char name[PPD_MAX_NAME]; /* Parameter name */ + char text[PPD_MAX_TEXT]; /* Human-readable text */ + int order; /* Order (0 to N) */ + ppd_cptype_t type; /* Parameter type */ + ppd_cplimit_t minimum, /* Minimum value */ + maximum; /* Maximum value */ + ppd_cpvalue_t current; /* Current value */ +} ppd_cparam_t; + +typedef struct ppd_coption_s /**** Custom Option @since CUPS 1.2/Mac OS X 10.5@ ****/ +{ + char keyword[PPD_MAX_NAME]; /* Name of option that is being extended... */ + ppd_option_t *option; /* Option that is being extended... */ + int marked; /* Extended option is marked */ + cups_array_t *params; /* Parameters */ +} ppd_coption_t; + +typedef struct _ppd_cache_s _ppd_cache_t; + /**** PPD cache and mapping data @since CUPS 1.5/Mac OS X 10.7@ @private@ ****/ + +typedef struct ppd_file_s /**** PPD File ****/ +{ + int language_level; /* Language level of device */ + int color_device; /* 1 = color device, 0 = grayscale */ + int variable_sizes; /* 1 = supports variable sizes, 0 = doesn't */ + int accurate_screens; /* 1 = supports accurate screens, 0 = not */ + int contone_only; /* 1 = continuous tone only, 0 = not */ + int landscape; /* -90 or 90 */ + int model_number; /* Device-specific model number */ + int manual_copies; /* 1 = Copies done manually, 0 = hardware */ + int throughput; /* Pages per minute */ + ppd_cs_t colorspace; /* Default colorspace */ + char *patches; /* Patch commands to be sent to printer */ + int num_emulations; /* Number of emulations supported */ + ppd_emul_t *emulations; /* Emulations and the code to invoke them */ + char *jcl_begin; /* Start JCL commands */ + char *jcl_ps; /* Enter PostScript interpreter */ + char *jcl_end; /* End JCL commands */ + char *lang_encoding; /* Language encoding */ + char *lang_version; /* Language version (English, Spanish, etc.) */ + char *modelname; /* Model name (general) */ + char *ttrasterizer; /* Truetype rasterizer */ + char *manufacturer; /* Manufacturer name */ + char *product; /* Product name (from PS RIP/interpreter) */ + char *nickname; /* Nickname (specific) */ + char *shortnickname; /* Short version of nickname */ + int num_groups; /* Number of UI groups */ + ppd_group_t *groups; /* UI groups */ + int num_sizes; /* Number of page sizes */ + ppd_size_t *sizes; /* Page sizes */ + float custom_min[2]; /* Minimum variable page size */ + float custom_max[2]; /* Maximum variable page size */ + float custom_margins[4]; /* Margins around page */ + int num_consts; /* Number of UI/Non-UI constraints */ + ppd_const_t *consts; /* UI/Non-UI constraints */ + int num_fonts; /* Number of pre-loaded fonts */ + char **fonts; /* Pre-loaded fonts */ + int num_profiles; /* Number of sRGB color profiles @deprecated@ */ + ppd_profile_t *profiles; /* sRGB color profiles @deprecated@ */ + int num_filters; /* Number of filters */ + char **filters; /* Filter strings... */ + + /**** New in CUPS 1.1 ****/ + int flip_duplex; /* 1 = Flip page for back sides @deprecated@ */ + + /**** New in CUPS 1.1.19 ****/ + char *protocols; /* Protocols (BCP, TBCP) string @since CUPS 1.1.19/Mac OS X 10.3@ */ + char *pcfilename; /* PCFileName string @since CUPS 1.1.19/Mac OS X 10.3@ */ + int num_attrs; /* Number of attributes @since CUPS 1.1.19/Mac OS X 10.3@ @private@ */ + int cur_attr; /* Current attribute @since CUPS 1.1.19/Mac OS X 10.3@ @private@ */ + ppd_attr_t **attrs; /* Attributes @since CUPS 1.1.19/Mac OS X 10.3@ @private@ */ + + /**** New in CUPS 1.2/Mac OS X 10.5 ****/ + cups_array_t *sorted_attrs; /* Attribute lookup array @since CUPS 1.2/Mac OS X 10.5@ @private@ */ + cups_array_t *options; /* Option lookup array @since CUPS 1.2/Mac OS X 10.5@ @private@ */ + cups_array_t *coptions; /* Custom options array @since CUPS 1.2/Mac OS X 10.5@ @private@ */ + + /**** New in CUPS 1.3/Mac OS X 10.5 ****/ + cups_array_t *marked; /* Marked choices @since CUPS 1.3/Mac OS X 10.5@ @private@ */ + + /**** New in CUPS 1.4/Mac OS X 10.6 ****/ + cups_array_t *cups_uiconstraints; /* cupsUIConstraints @since CUPS 1.4/Mac OS X 10.6@ @private@ */ + + /**** New in CUPS 1.5 ****/ + _ppd_cache_t *cache; /* PPD cache and mapping data @since CUPS 1.5/Mac OS X 10.7@ @private@ */ +} ppd_file_t; + + +/* + * Prototypes... + */ + +extern int cupsMarkOptions(ppd_file_t *ppd, int num_options, + cups_option_t *options); +extern void ppdClose(ppd_file_t *ppd); +extern int ppdCollect(ppd_file_t *ppd, ppd_section_t section, + ppd_choice_t ***choices); +extern int ppdConflicts(ppd_file_t *ppd); +extern int ppdEmit(ppd_file_t *ppd, FILE *fp, + ppd_section_t section); +extern int ppdEmitFd(ppd_file_t *ppd, int fd, + ppd_section_t section); +extern int ppdEmitJCL(ppd_file_t *ppd, FILE *fp, int job_id, + const char *user, const char *title); +extern ppd_choice_t *ppdFindChoice(ppd_option_t *o, const char *option); +extern ppd_choice_t *ppdFindMarkedChoice(ppd_file_t *ppd, + const char *keyword); +extern ppd_option_t *ppdFindOption(ppd_file_t *ppd, const char *keyword); +extern int ppdIsMarked(ppd_file_t *ppd, const char *keyword, + const char *option); +extern void ppdMarkDefaults(ppd_file_t *ppd); +extern int ppdMarkOption(ppd_file_t *ppd, const char *keyword, + const char *option); +extern ppd_file_t *ppdOpen(FILE *fp); +extern ppd_file_t *ppdOpenFd(int fd); +extern ppd_file_t *ppdOpenFile(const char *filename); +extern float ppdPageLength(ppd_file_t *ppd, const char *name); +extern ppd_size_t *ppdPageSize(ppd_file_t *ppd, const char *name); +extern float ppdPageWidth(ppd_file_t *ppd, const char *name); + +/**** New in CUPS 1.1.19 ****/ +extern const char *ppdErrorString(ppd_status_t status) _PPD_DEPRECATED; +extern ppd_attr_t *ppdFindAttr(ppd_file_t *ppd, const char *name, + const char *spec) _PPD_DEPRECATED; +extern ppd_attr_t *ppdFindNextAttr(ppd_file_t *ppd, const char *name, + const char *spec) _PPD_DEPRECATED; +extern ppd_status_t ppdLastError(int *line) _PPD_DEPRECATED; + +/**** New in CUPS 1.1.20 ****/ +extern void ppdSetConformance(ppd_conform_t c) _PPD_DEPRECATED; + +/**** New in CUPS 1.2 ****/ +extern int ppdCollect2(ppd_file_t *ppd, ppd_section_t section, + float min_order, ppd_choice_t ***choices) + _PPD_DEPRECATED; +extern int ppdEmitAfterOrder(ppd_file_t *ppd, FILE *fp, + ppd_section_t section, int limit, + float min_order) _PPD_DEPRECATED; +extern int ppdEmitJCLEnd(ppd_file_t *ppd, FILE *fp) _PPD_DEPRECATED; +extern char *ppdEmitString(ppd_file_t *ppd, ppd_section_t section, + float min_order) _PPD_DEPRECATED; +extern ppd_coption_t *ppdFindCustomOption(ppd_file_t *ppd, + const char *keyword) _PPD_DEPRECATED; +extern ppd_cparam_t *ppdFindCustomParam(ppd_coption_t *opt, + const char *name) _PPD_DEPRECATED; +extern ppd_cparam_t *ppdFirstCustomParam(ppd_coption_t *opt) _PPD_DEPRECATED; +extern ppd_option_t *ppdFirstOption(ppd_file_t *ppd) _PPD_DEPRECATED; +extern ppd_cparam_t *ppdNextCustomParam(ppd_coption_t *opt) _PPD_DEPRECATED; +extern ppd_option_t *ppdNextOption(ppd_file_t *ppd) _PPD_DEPRECATED; +extern int ppdLocalize(ppd_file_t *ppd) _PPD_DEPRECATED; +extern ppd_file_t *ppdOpen2(cups_file_t *fp) _PPD_DEPRECATED; + +/**** New in CUPS 1.3/Mac OS X 10.5 ****/ +extern const char *ppdLocalizeIPPReason(ppd_file_t *ppd, + const char *reason, + const char *scheme, + char *buffer, + size_t bufsize) _PPD_DEPRECATED; + +/**** New in CUPS 1.4/Mac OS X 10.6 ****/ +extern int cupsGetConflicts(ppd_file_t *ppd, const char *option, + const char *choice, + cups_option_t **options) + _PPD_DEPRECATED; +extern int cupsResolveConflicts(ppd_file_t *ppd, + const char *option, + const char *choice, + int *num_options, + cups_option_t **options) + _PPD_DEPRECATED; +extern int ppdInstallableConflict(ppd_file_t *ppd, + const char *option, + const char *choice) + _PPD_DEPRECATED; +extern ppd_attr_t *ppdLocalizeAttr(ppd_file_t *ppd, const char *keyword, + const char *spec) _PPD_DEPRECATED; +extern const char *ppdLocalizeMarkerName(ppd_file_t *ppd, + const char *name) + _PPD_DEPRECATED; +extern int ppdPageSizeLimits(ppd_file_t *ppd, + ppd_size_t *minimum, + ppd_size_t *maximum) _PPD_DEPRECATED; + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +} +# endif /* __cplusplus */ +#endif /* !_CUPS_PPD_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/pwg-media.c b/cups/pwg-media.c new file mode 100644 index 0000000000..db2e635db6 --- /dev/null +++ b/cups/pwg-media.c @@ -0,0 +1,851 @@ +/* + * "$Id$" + * + * PWG media name API implementation for CUPS. + * + * Copyright 2009-2011 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * _pwgGenerateSize() - Generate a PWG size keyword. + * _pwgInitSize() - Initialize a PWG size using IPP job template + * attributes. + * _pwgMediaForLegacy() - Find a PWG media size by ISO/IPP legacy name. + * _pwgMediaForPPD() - Find a PWG media size by Adobe PPD name. + * _pwgMediaForPWG() - Find a PWG media size by 5101.1 self-describing + * name. + * _pwgMediaForSize() - Get the PWG media name for a given size. + * pwg_compare_legacy() - Compare two sizes using the legacy names. + * pwg_compare_ppd() - Compare two sizes using the PPD names. + * pwg_compare_pwg() - Compare two sizes using the PWG names. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include + + +/* + * Local macros... + */ + +#define _PWG_MEDIA_IN(p,l,a,x,y) {p, l, a, (int)(x * 2540), (int)(y * 2540)} +#define _PWG_MEDIA_MM(p,l,a,x,y) {p, l, a, (int)(x * 100), (int)(y * 100)} + + +/* + * Local functions... + */ + +static int pwg_compare_legacy(_pwg_media_t *a, _pwg_media_t *b); +static int pwg_compare_pwg(_pwg_media_t *a, _pwg_media_t *b); +static int pwg_compare_ppd(_pwg_media_t *a, _pwg_media_t *b); + + +/* + * Local globals... + */ + +static _pwg_media_t const cups_pwg_media[] = +{ /* Media size lookup table */ + /* North American Standard Sheet Media Sizes */ + _PWG_MEDIA_IN("na_index-3x5_3x5in", NULL, "3x5", 3, 5), + _PWG_MEDIA_IN("na_personal_3.625x6.5in", NULL, "EnvPersonal", 3.625, 6.5), + _PWG_MEDIA_IN("na_monarch_3.875x7.5in", "monarch-envelope", "EnvMonarch", 3.875, 7.5), + _PWG_MEDIA_IN("na_number-9_3.875x8.875in", "na-number-9-envelope", "Env9", 3.875, 8.875), + _PWG_MEDIA_IN("na_index-4x6_4x6in", NULL, "4x6", 4, 6), + _PWG_MEDIA_IN("na_number-10_4.125x9.5in", "na-number-10-envelope", "Env10", 4.125, 9.5), + _PWG_MEDIA_IN("na_a2_4.375x5.75in", NULL, "EnvA2", 4.375, 5.75), + _PWG_MEDIA_IN("na_number-11_4.5x10.375in", NULL, "Env11", 4.5, 10.375), + _PWG_MEDIA_IN("na_number-12_4.75x11in", NULL, "Env12", 4.75, 11), + _PWG_MEDIA_IN("na_5x7_5x7in", NULL, "5x7", 5, 7), + _PWG_MEDIA_IN("na_index-5x8_5x8in", NULL, "5x8", 5, 8), + _PWG_MEDIA_IN("na_number-14_5x11.5in", NULL, "Env14", 5, 11.5), + _PWG_MEDIA_IN("na_invoice_5.5x8.5in", "invoice", "Statement", 5.5, 8.5), + _PWG_MEDIA_IN("na_index-4x6-ext_6x8in", NULL, NULL, 6, 8), + _PWG_MEDIA_IN("na_6x9_6x9in", "na-6x9-envelope", "6x9", 6, 9), + _PWG_MEDIA_IN("na_c5_6.5x9.5in", NULL, "6.5x9.5", 6.5, 9.5), + _PWG_MEDIA_IN("na_7x9_7x9in", "na-7x9-envelope", "7x9", 7, 9), + _PWG_MEDIA_IN("na_executive_7.25x10.5in", "executive", "Executive", 7.25, 10.5), + _PWG_MEDIA_IN("na_govt-letter_8x10in", "na-8x10", "8x10", 8, 10), + _PWG_MEDIA_IN("na_govt-legal_8x13in", NULL, "8x13", 8, 13), + _PWG_MEDIA_IN("na_quarto_8.5x10.83in", "quarto", "Quarto", 8.5, 10.83), + _PWG_MEDIA_IN("na_letter_8.5x11in", "na-letter", "Letter", 8.5, 11), + _PWG_MEDIA_IN("na_fanfold-eur_8.5x12in", NULL, "FanFoldGerman", 8.5, 12), + _PWG_MEDIA_IN("na_letter-plus_8.5x12.69in", NULL, "LetterPlus", 8.5, 12.69), + _PWG_MEDIA_IN("na_foolscap_8.5x13in", NULL, "FanFoldGermanLegal", 8.5, 13), + _PWG_MEDIA_IN("na_legal_8.5x14in", "na-legal", "Legal", 8.5, 14), + _PWG_MEDIA_IN("na_super-a_8.94x14in", NULL, "SuperA", 8.94, 14), + _PWG_MEDIA_IN("na_9x11_9x11in", "na-9x11-envelope", "9x11", 9, 11), + _PWG_MEDIA_IN("na_arch-a_9x12in", "arch-a", "ARCHA", 9, 12), + _PWG_MEDIA_IN("na_letter-extra_9.5x12in", NULL, "LetterExtra", 9.5, 12), + _PWG_MEDIA_IN("na_legal-extra_9.5x15in", NULL, "LegalExtra", 9.5, 15), + _PWG_MEDIA_IN("na_10x11_10x11in", NULL, "10x11", 10, 11), + _PWG_MEDIA_IN("na_10x13_10x13in", "na-10x13-envelope", "10x13", 10, 13), + _PWG_MEDIA_IN("na_10x14_10x14in", "na-10x14-envelope", "10x14", 10, 14), + _PWG_MEDIA_IN("na_10x15_10x15in", "na-10x15-envelope", "10x15", 10, 15), + _PWG_MEDIA_IN("na_11x12_11x12in", NULL, "11x12", 11, 12), + _PWG_MEDIA_IN("na_edp_11x14in", NULL, "11x14", 11, 14), + _PWG_MEDIA_IN("na_fanfold-us_11x14.875in", NULL, NULL, 11, 14.875), + _PWG_MEDIA_IN("na_11x15_11x15in", NULL, "11x15", 11, 15), + _PWG_MEDIA_IN("na_ledger_11x17in", "tabloid", "Tabloid", 11, 17), + _PWG_MEDIA_IN("na_eur-edp_12x14in", NULL, NULL, 12, 14), + _PWG_MEDIA_IN("na_arch-b_12x18in", "arch-b", "ARCHB", 12, 18), + _PWG_MEDIA_IN("na_12x19_12x19in", NULL, "12x19", 12, 19), + _PWG_MEDIA_IN("na_b-plus_12x19.17in", NULL, "SuperB", 12, 19.17), + _PWG_MEDIA_IN("na_super-b_13x19in", "super-b", "13x19", 13, 19), + _PWG_MEDIA_IN("na_c_17x22in", "c", "AnsiC", 17, 22), + _PWG_MEDIA_IN("na_arch-c_18x24in", "arch-c", "ARCHC", 18, 24), + _PWG_MEDIA_IN("na_d_22x34in", "d", "AnsiD", 22, 34), + _PWG_MEDIA_IN("na_arch-d_24x36in", "arch-d", "ARCHD", 24, 36), + _PWG_MEDIA_IN("asme_f_28x40in", "f", NULL, 28, 40), + _PWG_MEDIA_IN("na_wide-format_30x42in", NULL, NULL, 30, 42), + _PWG_MEDIA_IN("na_e_34x44in", "e", "AnsiE", 34, 44), + _PWG_MEDIA_IN("na_arch-e_36x48in", "arch-e", "ARCHE", 36, 48), + _PWG_MEDIA_IN("na_f_44x68in", NULL, "AnsiF", 44, 68), + + /* Chinese Standard Sheet Media Inch Sizes */ + _PWG_MEDIA_IN("roc_16k_7.75x10.75in", NULL, "roc16k", 7.75, 10.75), + _PWG_MEDIA_IN("roc_8k_10.75x15.5in", NULL, NULL, 10.75, 15.5), + + /* ISO Standard Sheet Media Sizes */ + _PWG_MEDIA_MM("iso_a10_26x37mm", "iso-a10", "A10", 26, 37), + _PWG_MEDIA_MM("iso_a9_37x52mm", "iso-a9", "A9", 37, 52), + _PWG_MEDIA_MM("iso_a8_52x74mm", "iso-a8", "A8", 52, 74), + _PWG_MEDIA_MM("iso_a7_74x105mm", "iso-a7", "A7", 74, 105), + _PWG_MEDIA_MM("iso_a6_105x148mm", "iso-a6", "A6", 105, 148), + _PWG_MEDIA_MM("iso_a5_148x210mm", "iso-a5", "A5", 148, 210), + _PWG_MEDIA_MM("iso_a5-extra_174x235mm", NULL, "A5Extra", 174, 235), + _PWG_MEDIA_MM("iso_a4_210x297mm", "iso-a4", "A4", 210, 297), + _PWG_MEDIA_MM("iso_a4-tab_225x297mm", NULL, NULL, 225, 297), + _PWG_MEDIA_MM("iso_a4-extra_235.5x322.3mm", NULL, "A4Extra", 235.5, 322.3), + _PWG_MEDIA_MM("iso_a3_297x420mm", "iso-a3", "A3", 297, 420), + _PWG_MEDIA_MM("iso_a4x3_297x630mm", "iso-a4x3", NULL, 297, 630), + _PWG_MEDIA_MM("iso_a4x4_297x841mm", "iso-a4x4", NULL, 297, 841), + _PWG_MEDIA_MM("iso_a4x5_297x1051mm", "iso-a4x5", NULL, 297, 1051), + _PWG_MEDIA_MM("iso_a4x6_297x1261mm", "iso-a4x6", NULL, 297, 1261), + _PWG_MEDIA_MM("iso_a4x7_297x1471mm", "iso-a4x7", NULL, 297, 1471), + _PWG_MEDIA_MM("iso_a4x8_297x1682mm", "iso-a4x8", NULL, 297, 1682), + _PWG_MEDIA_MM("iso_a4x9_297x1892mm", "iso-a4x9", NULL, 297, 1892), + _PWG_MEDIA_MM("iso_a3-extra_322x445mm", "iso-a3-extra", "A3Extra", 322, 445), + _PWG_MEDIA_MM("iso_a2_420x594mm", "iso-a2", "A2", 420, 594), + _PWG_MEDIA_MM("iso_a3x3_420x891mm", "iso-a3x3", NULL, 420, 891), + _PWG_MEDIA_MM("iso_a3x4_420x1189mm", "iso-a3x4", NULL, 420, 1189), + _PWG_MEDIA_MM("iso_a3x5_420x1486mm", "iso-a3x5", NULL, 420, 1486), + _PWG_MEDIA_MM("iso_a3x6_420x1783mm", "iso-a3x6", NULL, 420, 1783), + _PWG_MEDIA_MM("iso_a3x7_420x2080mm", "iso-a3x7", NULL, 420, 2080), + _PWG_MEDIA_MM("iso_a1_594x841mm", "iso-a1", "A1", 594, 841), + _PWG_MEDIA_MM("iso_a2x3_594x1261mm", "iso-a2x3", NULL, 594, 1261), + _PWG_MEDIA_MM("iso_a2x4_594x1682mm", "iso-a2x4", NULL, 594, 1682), + _PWG_MEDIA_MM("iso_a2x5_594x2102mm", "iso-a2x5", NULL, 594, 2102), + _PWG_MEDIA_MM("iso_a0_841x1189mm", "iso-a0", "A0", 841, 1189), + _PWG_MEDIA_MM("iso_a1x3_841x1783mm", "iso-a1x3", NULL, 841, 1783), + _PWG_MEDIA_MM("iso_a1x4_841x2378mm", "iso-a1x4", NULL, 841, 2378), + _PWG_MEDIA_MM("iso_2a0_1189x1682mm", NULL, NULL, 1189, 1682), + _PWG_MEDIA_MM("iso_a0x3_1189x2523mm", NULL, NULL, 1189, 2523), + _PWG_MEDIA_MM("iso_b10_31x44mm", "iso-b10", "ISOB10", 31, 44), + _PWG_MEDIA_MM("iso_b9_44x62mm", "iso-b9", "ISOB9", 44, 62), + _PWG_MEDIA_MM("iso_b8_62x88mm", "iso-b8", "ISOB8", 62, 88), + _PWG_MEDIA_MM("iso_b7_88x125mm", "iso-b7", "ISOB7", 88, 125), + _PWG_MEDIA_MM("iso_b6_125x176mm", "iso-b6", "ISOB6", 125, 176), + _PWG_MEDIA_MM("iso_b6c4_125x324mm", NULL, NULL, 125, 324), + _PWG_MEDIA_MM("iso_b5_176x250mm", "iso-b5", "ISOB5", 176, 250), + _PWG_MEDIA_MM("iso_b5-extra_201x276mm", NULL, "ISOB5Extra", 201, 276), + _PWG_MEDIA_MM("iso_b4_250x353mm", "iso-b4", "ISOB4", 250, 353), + _PWG_MEDIA_MM("iso_b3_353x500mm", "iso-b3", "ISOB3", 353, 500), + _PWG_MEDIA_MM("iso_b2_500x707mm", "iso-b2", "ISOB2", 500, 707), + _PWG_MEDIA_MM("iso_b1_707x1000mm", "iso-b1", "ISOB1", 707, 1000), + _PWG_MEDIA_MM("iso_b0_1000x1414mm", "iso-b0", "ISOB0", 1000, 1414), + _PWG_MEDIA_MM("iso_c10_28x40mm", "iso-c10", NULL, 28, 40), + _PWG_MEDIA_MM("iso_c9_40x57mm", "iso-c9", NULL, 40, 57), + _PWG_MEDIA_MM("iso_c8_57x81mm", "iso-c8", NULL, 57, 81), + _PWG_MEDIA_MM("iso_c7_81x114mm", "iso-c7", "EnvC7", 81, 114), + _PWG_MEDIA_MM("iso_c7c6_81x162mm", NULL, NULL, 81, 162), + _PWG_MEDIA_MM("iso_c6_114x162mm", "iso-c6", "EnvC6", 114, 162), + _PWG_MEDIA_MM("iso_c6c5_114x229mm", NULL, "EnvC65", 114, 229), + _PWG_MEDIA_MM("iso_c5_162x229mm", "iso-c5", "EnvC5", 162, 229), + _PWG_MEDIA_MM("iso_c4_229x324mm", "iso-c4", "EnvC4", 229, 324), + _PWG_MEDIA_MM("iso_c3_324x458mm", "iso-c3", "EnvC3", 324, 458), + _PWG_MEDIA_MM("iso_c2_458x648mm", "iso-c2", "EnvC2", 458, 648), + _PWG_MEDIA_MM("iso_c1_648x917mm", "iso-c1", "EnvC1", 648, 917), + _PWG_MEDIA_MM("iso_c0_917x1297mm", "iso-c0", "EnvC0", 917, 1297), + _PWG_MEDIA_MM("iso_dl_110x220mm", "iso-designated", "EnvDL", 110, 220), + _PWG_MEDIA_MM("iso_ra2_430x610mm", "iso-ra2", NULL, 430, 610), + _PWG_MEDIA_MM("iso_sra2_450x640mm", "iso-sra2", NULL, 450, 640), + _PWG_MEDIA_MM("iso_ra1_610x860mm", "iso-ra1", NULL, 610, 860), + _PWG_MEDIA_MM("iso_sra1_640x900mm", "iso-sra1", NULL, 640, 900), + _PWG_MEDIA_MM("iso_ra0_860x1220mm", "iso-ra0", NULL, 860, 1220), + _PWG_MEDIA_MM("iso_sra0_900x1280mm", "iso-sra0", NULL, 900, 1280), + + /* Japanese Standard Sheet Media Sizes */ + _PWG_MEDIA_MM("jis_b10_32x45mm", "jis-b10", "B10", 32, 45), + _PWG_MEDIA_MM("jis_b9_45x64mm", "jis-b9", "B9", 45, 64), + _PWG_MEDIA_MM("jis_b8_64x91mm", "jis-b8", "B8", 64, 91), + _PWG_MEDIA_MM("jis_b7_91x128mm", "jis-b7", "B7", 91, 128), + _PWG_MEDIA_MM("jis_b6_128x182mm", "jis-b6", "B6", 128, 182), + _PWG_MEDIA_MM("jis_b5_182x257mm", "jis-b5", "B5", 182, 257), + _PWG_MEDIA_MM("jis_b4_257x364mm", "jis-b4", "B4", 257, 364), + _PWG_MEDIA_MM("jis_b3_364x515mm", "jis-b3", "B3", 364, 515), + _PWG_MEDIA_MM("jis_b2_515x728mm", "jis-b2", "B2", 515, 728), + _PWG_MEDIA_MM("jis_b1_728x1030mm", "jis-b1", "B1", 728, 1030), + _PWG_MEDIA_MM("jis_b0_1030x1456mm", "jis-b0", "B0", 1030, 1456), + _PWG_MEDIA_MM("jis_exec_216x330mm", NULL, NULL, 216, 330), + _PWG_MEDIA_MM("jpn_chou4_90x205mm", NULL, "EnvChou4", 90, 205), + _PWG_MEDIA_MM("jpn_hagaki_100x148mm", NULL, "Postcard", 100, 148), + _PWG_MEDIA_MM("jpn_you4_105x235mm", NULL, "EnvYou4", 105, 235), + _PWG_MEDIA_MM("jpn_chou2_111.1x146mm", NULL, NULL, 111.1, 146), + _PWG_MEDIA_MM("jpn_chou3_120x235mm", NULL, "EnvChou3", 120, 235), + _PWG_MEDIA_MM("jpn_oufuku_148x200mm", NULL, "DoublePostcardRotated", 148, 200), + _PWG_MEDIA_MM("jpn_kahu_240x322.1mm", NULL, NULL, 240, 322.1), + _PWG_MEDIA_MM("jpn_kaku2_240x332mm", NULL, "EnvKaku2", 240, 332), + + /* Chinese Standard Sheet Media Sizes */ + _PWG_MEDIA_MM("prc_32k_97x151mm", NULL, "PRC32K", 97, 151), + _PWG_MEDIA_MM("prc_1_102x165mm", NULL, "EnvPRC1", 102, 165), + _PWG_MEDIA_MM("prc_2_102x176mm", NULL, "EnvPRC2", 102, 176), + _PWG_MEDIA_MM("prc_4_110x208mm", NULL, "EnvPRC4", 110, 208), + _PWG_MEDIA_MM("prc_5_110x220mm", NULL, "EnvPRC5", 110, 220), + _PWG_MEDIA_MM("prc_8_120x309mm", NULL, "EnvPRC8", 120, 309), + _PWG_MEDIA_MM("prc_6_120x320mm", NULL, NULL, 120, 320), + _PWG_MEDIA_MM("prc_3_125x176mm", NULL, "EnvPRC3", 125, 176), + _PWG_MEDIA_MM("prc_16k_146x215mm", NULL, "PRC16K", 146, 215), + _PWG_MEDIA_MM("prc_7_160x230mm", NULL, "EnvPRC7", 160, 230), + _PWG_MEDIA_MM("om_juuro-ku-kai_198x275mm", NULL, NULL, 198, 275), + _PWG_MEDIA_MM("om_pa-kai_267x389mm", NULL, NULL, 267, 389), + _PWG_MEDIA_MM("om_dai-pa-kai_275x395mm", NULL, NULL, 275, 395), + _PWG_MEDIA_MM("prc_10_324x458mm", NULL, "EnvPRC10", 324, 458), + + /* Other English Standard Sheet Media Sizes */ + _PWG_MEDIA_IN("oe_photo-l_3.5x5in", NULL, "3.5x5", 3.5, 5), + + /* Other Metric Standard Sheet Media Sizes */ + _PWG_MEDIA_MM("om_small-photo_100x150mm", NULL, "om_small-photo", 100, 150), + _PWG_MEDIA_MM("om_italian_110x230mm", NULL, "EnvItalian", 110, 230), + _PWG_MEDIA_MM("om_postfix_114x229mm", NULL, NULL, 114, 229), + _PWG_MEDIA_MM("om_large-photo_200x300", NULL, NULL, 200, 300), + _PWG_MEDIA_MM("om_folio_210x330mm", "folio", "Folio", 210, 330), + _PWG_MEDIA_MM("om_folio-sp_215x315mm", NULL, NULL, 215, 315), + _PWG_MEDIA_MM("om_invite_220x220mm", NULL, "EnvInvite", 220, 220) +}; + + +/* + * '_pwgGenerateSize()' - Generate a PWG size keyword. + */ + +void +_pwgGenerateSize(char *keyword, /* I - Keyword buffer */ + size_t keysize, /* I - Size of keyword buffer */ + const char *prefix, /* I - Prefix for PWG size or NULL */ + const char *name, /* I - Size name or NULL */ + int width, /* I - Width of page in 2540ths */ + int length) /* I - Length of page in 2540ths */ +{ + struct lconv *loc; /* Locale conversion data */ + double uwidth, /* Width in inches or millimeters */ + ulength; /* Height in inches or millimeters */ + const char *units; /* Units to report */ + char usize[12 + 1 + 12 + 3], /* Unit size: NNNNNNNNNNNNxNNNNNNNNNNNNuu */ + *uptr; /* Pointer into unit size */ + + + loc = localeconv(); + + if ((width % 635) == 0 && (length % 635) == 0) + { + /* + * Use inches since the size is a multiple of 1/4 inch. + */ + + uwidth = width / 2540.0; + ulength = length / 2540.0; + units = "in"; + + if (!prefix) + prefix = "oe"; + } + else + { + /* + * Use millimeters since the size is not a multiple of 1/4 inch. + */ + + uwidth = width * 0.01; + ulength = length * 0.01; + units = "mm"; + + if (!prefix) + prefix = "om"; + } + + uptr = usize; + _cupsStrFormatd(uptr, uptr + 12, uwidth, loc); + uptr += strlen(uptr); + *uptr++ = 'x'; + _cupsStrFormatd(uptr, uptr + 12, ulength, loc); + uptr += strlen(uptr); + + /* + * Safe because usize can hold up to 12 + 1 + 12 + 4 bytes. + */ + + strcpy(uptr, units); + + if (!name) + name = usize; + + /* + * Format the name... + */ + + snprintf(keyword, keysize, "%s_%s_%s", prefix, name, usize); +} + + +/* + * '_pwgInitSize()' - Initialize a PWG size using IPP job template attributes. + */ + +int /* O - 1 if size was initialize, 0 otherwise */ +_pwgInitSize(_pwg_size_t *size, /* I - Size to initialize */ + ipp_t *job, /* I - Job template attributes */ + int *margins_set) /* O - 1 if margins were set, 0 otherwise */ +{ + ipp_attribute_t *media, /* media attribute */ + *media_bottom_margin, /* media-bottom-margin member attribute */ + *media_col, /* media-col attribute */ + *media_left_margin, /* media-left-margin member attribute */ + *media_right_margin, /* media-right-margin member attribute */ + *media_size, /* media-size member attribute */ + *media_top_margin, /* media-top-margin member attribute */ + *x_dimension, /* x-dimension member attribute */ + *y_dimension; /* y-dimension member attribute */ + _pwg_media_t *pwg; /* PWG media value */ + + + /* + * Range check input... + */ + + if (!size || !job || !margins_set) + return (0); + + /* + * Look for media-col and then media... + */ + + memset(size, 0, sizeof(_pwg_size_t)); + *margins_set = 0; + + if ((media_col = ippFindAttribute(job, "media-col", + IPP_TAG_BEGIN_COLLECTION)) != NULL) + { + /* + * Got media-col, look for media-size member attribute... + */ + + if ((media_size = ippFindAttribute(media_col->values[0].collection, + "media-size", + IPP_TAG_BEGIN_COLLECTION)) != NULL) + { + /* + * Got media-size, look for x-dimension and y-dimension member + * attributes... + */ + + x_dimension = ippFindAttribute(media_size->values[0].collection, + "x-dimension", IPP_TAG_INTEGER); + y_dimension = ippFindAttribute(media_size->values[0].collection, + "y-dimension", IPP_TAG_INTEGER); + + if (x_dimension && y_dimension) + { + size->width = x_dimension->values[0].integer; + size->length = y_dimension->values[0].integer; + } + else if (!x_dimension) + { + _cupsSetError(IPP_INTERNAL_ERROR, + _("Missing x-dimension in media-size."), 1); + return (0); + } + else if (!y_dimension) + { + _cupsSetError(IPP_INTERNAL_ERROR, + _("Missing y-dimension in media-size."), 1); + return (0); + } + } + else + { + _cupsSetError(IPP_INTERNAL_ERROR, _("Missing media-size in media-col."), + 1); + return (0); + } + + /* media-*-margin */ + media_bottom_margin = ippFindAttribute(media_col->values[0].collection, + "media-bottom-margin", + IPP_TAG_INTEGER); + media_left_margin = ippFindAttribute(media_col->values[0].collection, + "media-left-margin", + IPP_TAG_INTEGER); + media_right_margin = ippFindAttribute(media_col->values[0].collection, + "media-right-margin", + IPP_TAG_INTEGER); + media_top_margin = ippFindAttribute(media_col->values[0].collection, + "media-top-margin", + IPP_TAG_INTEGER); + if (media_bottom_margin && media_left_margin && media_right_margin && + media_top_margin) + { + *margins_set = 1; + size->bottom = media_bottom_margin->values[0].integer; + size->left = media_left_margin->values[0].integer; + size->right = media_right_margin->values[0].integer; + size->top = media_top_margin->values[0].integer; + } + } + else + { + if ((media = ippFindAttribute(job, "media", IPP_TAG_NAME)) == NULL) + if ((media = ippFindAttribute(job, "media", IPP_TAG_KEYWORD)) == NULL) + if ((media = ippFindAttribute(job, "PageSize", IPP_TAG_NAME)) == NULL) + media = ippFindAttribute(job, "PageRegion", IPP_TAG_NAME); + + if (media) + { + const char *name = media->values[0].string.text; + /* Name string */ + + if ((pwg = _pwgMediaForPWG(name)) == NULL) + { + /* + * Not a PWG name, try a legacy name... + */ + + if ((pwg = _pwgMediaForLegacy(name)) == NULL) + { + /* + * Not a legacy name, try a PPD name... + */ + + const char *suffix; /* Suffix on media string */ + + pwg = _pwgMediaForPPD(name); + if (pwg && + (suffix = name + strlen(name) - 10 /* .FullBleed */) > name && + !_cups_strcasecmp(suffix, ".FullBleed")) + { + /* + * Indicate that margins are set with the default values of 0. + */ + + *margins_set = 1; + } + } + } + + if (pwg) + { + size->width = pwg->width; + size->length = pwg->length; + } + else + { + _cupsSetError(IPP_INTERNAL_ERROR, _("Unsupported media value."), 1); + return (0); + } + } + else + { + _cupsSetError(IPP_INTERNAL_ERROR, _("Missing media or media-col."), 1); + return (0); + } + } + + return (1); +} + + +/* + * '_pwgMediaForLegacy()' - Find a PWG media size by ISO/IPP legacy name. + */ + +_pwg_media_t * /* O - Matching size or NULL */ +_pwgMediaForLegacy( + const char *legacy) /* I - Legacy size name */ +{ + _pwg_media_t key; /* Search key */ + _cups_globals_t *cg = _cupsGlobals(); /* Global data */ + + + /* + * Range check input... + */ + + if (!legacy) + return (NULL); + + /* + * Build the lookup table for PWG names as needed... + */ + + if (!cg->leg_size_lut) + { + int i; /* Looping var */ + _pwg_media_t *size; /* Current size */ + + cg->leg_size_lut = cupsArrayNew((cups_array_func_t)pwg_compare_legacy, + NULL); + + for (i = (int)(sizeof(cups_pwg_media) / sizeof(cups_pwg_media[0])), + size = (_pwg_media_t *)cups_pwg_media; + i > 0; + i --, size ++) + if (size->legacy) + cupsArrayAdd(cg->leg_size_lut, size); + } + + /* + * Lookup the name... + */ + + key.legacy = legacy; + return ((_pwg_media_t *)cupsArrayFind(cg->leg_size_lut, &key)); +} + + +/* + * '_pwgMediaForPPD()' - Find a PWG media size by Adobe PPD name. + */ + +_pwg_media_t * /* O - Matching size or NULL */ +_pwgMediaForPPD(const char *ppd) /* I - PPD size name */ +{ + _pwg_media_t key, /* Search key */ + *size; /* Matching size */ + _cups_globals_t *cg = _cupsGlobals(); /* Global data */ + + + /* + * Range check input... + */ + + if (!ppd) + return (NULL); + + /* + * Build the lookup table for PWG names as needed... + */ + + if (!cg->ppd_size_lut) + { + int i; /* Looping var */ + + cg->ppd_size_lut = cupsArrayNew((cups_array_func_t)pwg_compare_ppd, NULL); + + for (i = (int)(sizeof(cups_pwg_media) / sizeof(cups_pwg_media[0])), + size = (_pwg_media_t *)cups_pwg_media; + i > 0; + i --, size ++) + if (size->ppd) + cupsArrayAdd(cg->ppd_size_lut, size); + } + + /* + * Lookup the name... + */ + + key.ppd = ppd; + if ((size = (_pwg_media_t *)cupsArrayFind(cg->ppd_size_lut, &key)) == NULL) + { + /* + * See if the name is of the form: + * + * [Custom.]WIDTHxLENGTH[.FullBleed] - Size in points/inches [borderless] + * [Custom.]WIDTHxLENGTHcm[.FullBleed] - Size in centimeters [borderless] + * [Custom.]WIDTHxLENGTHft[.FullBleed] - Size in feet [borderless] + * [Custom.]WIDTHxLENGTHin[.FullBleed] - Size in inches [borderless] + * [Custom.]WIDTHxLENGTHm[.FullBleed] - Size in meters [borderless] + * [Custom.]WIDTHxLENGTHmm[.FullBleed] - Size in millimeters [borderless] + * [Custom.]WIDTHxLENGTHpt[.FullBleed] - Size in points [borderless] + */ + + double w, l, /* Width and length of page */ + factor; /* Unit scaling factor */ + char *ptr; /* Pointer into name */ + struct lconv *loc; /* Locale data */ + int custom; /* Custom page size? */ + + if (!_cups_strncasecmp(ppd, "Custom.", 7)) + { + custom = 1; + factor = 2540.0 / 72.0; + ptr = (char *)ppd + 7; + } + else + { + custom = 0; + factor = 2540.0; + ptr = (char *)ppd; + } + + loc = localeconv(); + w = _cupsStrScand(ptr, &ptr, loc); + + if (ptr && ptr > ppd && *ptr == 'x') + { + l = _cupsStrScand(ptr + 1, &ptr, loc); + + if (ptr && + (!*ptr || + !_cups_strcasecmp(ptr, "FullBleed") || + !_cups_strcasecmp(ptr, ".FullBleed") || + !_cups_strcasecmp(ptr, "cm") || + !_cups_strcasecmp(ptr, "cm.FullBleed") || + !_cups_strcasecmp(ptr, "ft") || + !_cups_strcasecmp(ptr, "ft.FullBleed") || + !_cups_strcasecmp(ptr, "in") || + !_cups_strcasecmp(ptr, "in.FullBleed") || + !_cups_strcasecmp(ptr, "m") || + !_cups_strcasecmp(ptr, "m.FullBleed") || + !_cups_strcasecmp(ptr, "mm") || + !_cups_strcasecmp(ptr, "mm.FullBleed") || + !_cups_strcasecmp(ptr, "pt") || + !_cups_strcasecmp(ptr, "pt.FullBleed"))) + { + size = &(cg->pwg_media); + + if (!_cups_strncasecmp(ptr, "cm", 2)) + factor = 1000.0; + else if (!_cups_strncasecmp(ptr, "ft", 2)) + factor = 2540.0 * 12.0; + else if (!_cups_strncasecmp(ptr, "in", 2)) + factor = 2540.0; + else if (!_cups_strncasecmp(ptr, "mm", 2)) + factor = 100.0; + else if (*ptr == 'm' || *ptr == 'M') + factor = 100000.0; + else if (!_cups_strncasecmp(ptr, "pt", 2)) + factor = 2540.0 / 72.0; + + /* + * Not a standard size; convert it to a PWG custom name of the form: + * + * [oe|om]_WIDTHxHEIGHTuu_WIDTHxHEIGHTuu + */ + + size->width = (int)(w * factor); + size->length = (int)(l * factor); + size->pwg = cg->pwg_name; + + _pwgGenerateSize(cg->pwg_name, sizeof(cg->pwg_name), + custom ? "custom" : NULL, custom ? ppd + 7 : NULL, + size->width, size->length); + } + } + } + + return (size); +} + + +/* + * '_pwgMediaForPWG()' - Find a PWG media size by 5101.1 self-describing name. + */ + +_pwg_media_t * /* O - Matching size or NULL */ +_pwgMediaForPWG(const char *pwg) /* I - PWG size name */ +{ + char *ptr; /* Pointer into name */ + _pwg_media_t key, /* Search key */ + *size; /* Matching size */ + _cups_globals_t *cg = _cupsGlobals(); /* Global data */ + + + /* + * Range check input... + */ + + if (!pwg) + return (NULL); + + /* + * Build the lookup table for PWG names as needed... + */ + + if (!cg->pwg_size_lut) + { + int i; /* Looping var */ + + cg->pwg_size_lut = cupsArrayNew((cups_array_func_t)pwg_compare_pwg, NULL); + + for (i = (int)(sizeof(cups_pwg_media) / sizeof(cups_pwg_media[0])), + size = (_pwg_media_t *)cups_pwg_media; + i > 0; + i --, size ++) + cupsArrayAdd(cg->pwg_size_lut, size); + } + + /* + * Lookup the name... + */ + + key.pwg = pwg; + if ((size = (_pwg_media_t *)cupsArrayFind(cg->pwg_size_lut, &key)) == NULL && + (ptr = (char *)strchr(pwg, '_')) != NULL && + (ptr = (char *)strchr(ptr + 1, '_')) != NULL) + { + /* + * Try decoding the self-describing name of the form: + * + * class_name_WWWxHHHin + * class_name_WWWxHHHmm + */ + + double w, l; /* Width and length of page */ + struct lconv *loc; /* Locale data */ + + ptr ++; + loc = localeconv(); + w = _cupsStrScand(ptr, &ptr, loc); + + if (ptr && *ptr == 'x') + { + l = _cupsStrScand(ptr + 1, &ptr, loc); + + if (ptr && (!strcmp(ptr, "in") || !strcmp(ptr, "mm"))) + { + size = &(cg->pwg_media); + + if (!strcmp(ptr, "mm")) + { + size->width = (int)(w * 100); + size->length = (int)(l * 100); + } + else + { + size->width = (int)(w * 2540); + size->length = (int)(l * 2540); + } + + strlcpy(cg->pwg_name, pwg, sizeof(cg->pwg_name)); + size->pwg = cg->pwg_name; + } + } + } + + return (size); +} + + +/* + * '_pwgMediaForSize()' - Get the PWG media name for a given size. + */ + +_pwg_media_t * /* O - PWG media name */ +_pwgMediaForSize(int width, /* I - Width in 2540ths */ + int length) /* I - Length in 2540ths */ +{ + int i; /* Looping var */ + _pwg_media_t *media; /* Current media */ + int dw, dl; /* Difference in width and length */ + _cups_globals_t *cg = _cupsGlobals(); /* Global data */ + + + /* + * Range check input... + */ + + if (width <= 0 || length <= 0) + return (NULL); + + /* + * Look for a standard size... + */ + + for (i = (int)(sizeof(cups_pwg_media) / sizeof(cups_pwg_media[0])), + media = (_pwg_media_t *)cups_pwg_media; + i > 0; + i --, media ++) + { + /* + * Adobe uses a size matching algorithm with an epsilon of 5 points, which + * is just about 176/2540ths... + */ + + dw = media->width - width; + dl = media->length - length; + + if (dw > -176 && dw < 176 && dl > -176 && dl < 176) + return (media); + } + + /* + * Not a standard size; convert it to a PWG custom name of the form: + * + * custom_WIDTHxHEIGHTuu_WIDTHxHEIGHTuu + */ + + _pwgGenerateSize(cg->pwg_name, sizeof(cg->pwg_name), "custom", NULL, width, + length); + + cg->pwg_media.pwg = cg->pwg_name; + cg->pwg_media.width = width; + cg->pwg_media.length = length; + + return (&(cg->pwg_media)); +} + + +/* + * 'pwg_compare_legacy()' - Compare two sizes using the legacy names. + */ + +static int /* O - Result of comparison */ +pwg_compare_legacy(_pwg_media_t *a, /* I - First size */ + _pwg_media_t *b) /* I - Second size */ +{ + return (strcmp(a->legacy, b->legacy)); +} + + +/* + * 'pwg_compare_ppd()' - Compare two sizes using the PPD names. + */ + +static int /* O - Result of comparison */ +pwg_compare_ppd(_pwg_media_t *a, /* I - First size */ + _pwg_media_t *b) /* I - Second size */ +{ + return (strcmp(a->ppd, b->ppd)); +} + + +/* + * 'pwg_compare_pwg()' - Compare two sizes using the PWG names. + */ + +static int /* O - Result of comparison */ +pwg_compare_pwg(_pwg_media_t *a, /* I - First size */ + _pwg_media_t *b) /* I - Second size */ +{ + return (strcmp(a->pwg, b->pwg)); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/pwg-private.h b/cups/pwg-private.h new file mode 100644 index 0000000000..13eedf62d7 --- /dev/null +++ b/cups/pwg-private.h @@ -0,0 +1,102 @@ +/* + * "$Id$" + * + * Private PWG media API definitions for CUPS. + * + * Copyright 2009-2012 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_PWG_PRIVATE_H_ +# define _CUPS_PWG_PRIVATE_H_ + + +/* + * Include necessary headers... + */ + +# include + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Macros... + */ + +/* Convert from points to 2540ths */ +# define _PWG_FROMPTS(n) (int)(((n) * 2540 + 36) / 72) +/* Convert from 2540ths to points */ +# define _PWG_TOPTS(n) ((n) * 72.0 / 2540.0) + + +/* + * Types and structures... + */ + +typedef struct _pwg_map_s /**** Map element - PPD to/from PWG */ +{ + char *pwg, /* PWG media keyword */ + *ppd; /* PPD option keyword */ +} _pwg_map_t; + +typedef struct _pwg_media_s /**** Common media size data ****/ +{ + const char *pwg, /* PWG 5101.1 "self describing" name */ + *legacy, /* IPP/ISO legacy name */ + *ppd; /* Standard Adobe PPD name */ + int width, /* Width in 2540ths */ + length; /* Length in 2540ths */ +} _pwg_media_t; + +typedef struct _pwg_size_s /**** Size element - PPD to/from PWG */ +{ + _pwg_map_t map; /* Map element */ + int width, /* Width in 2540ths */ + length, /* Length in 2540ths */ + left, /* Left margin in 2540ths */ + bottom, /* Bottom margin in 2540ths */ + right, /* Right margin in 2540ths */ + top; /* Top margin in 2540ths */ +} _pwg_size_t; + + +/* + * Functions... + */ + +extern void _pwgGenerateSize(char *keyword, size_t keysize, + const char *prefix, + const char *name, + int width, int length); +extern int _pwgInitSize(_pwg_size_t *size, ipp_t *job, + int *margins_set); +extern _pwg_media_t *_pwgMediaForLegacy(const char *legacy); +extern _pwg_media_t *_pwgMediaForPPD(const char *ppd); +extern _pwg_media_t *_pwgMediaForPWG(const char *pwg); +extern _pwg_media_t *_pwgMediaForSize(int width, int length); + + +# ifdef __cplusplus +} +# endif /* __cplusplus */ + +#endif /* !_CUPS_PWG_PRIVATE_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/raster-private.h b/cups/raster-private.h new file mode 100644 index 0000000000..169631e683 --- /dev/null +++ b/cups/raster-private.h @@ -0,0 +1,69 @@ +/* + * "$Id$" + * + * Private image library definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1993-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_RASTER_PRIVATE_H_ +# define _CUPS_RASTER_PRIVATE_H_ + +/* + * Include necessary headers... + */ + +# include "raster.h" +# include +# include +# include +# ifdef WIN32 +# include +# include /* for htonl() definition */ +# else +# include +# include +# endif /* WIN32 */ + + +/* + * min/max/abs macros... + */ + +# ifndef max +# define max(a,b) ((a) > (b) ? (a) : (b)) +# endif /* !max */ +# ifndef min +# define min(a,b) ((a) < (b) ? (a) : (b)) +# endif /* !min */ +# ifndef abs +# define abs(a) ((a) < 0 ? -(a) : (a)) +# endif /* !abs */ + + +/* + * Prototypes... + */ + +extern int _cupsRasterExecPS(cups_page_header2_t *h, + int *preferred_bits, + const char *code) + __attribute__((nonnull(3))); +extern void _cupsRasterAddError(const char *f, ...) + __attribute__((__format__(__printf__, 1, 2))); +extern void _cupsRasterClearError(void); + +#endif /* !_CUPS_RASTER_PRIVATE_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/raster.h b/cups/raster.h new file mode 100644 index 0000000000..1510df2d6e --- /dev/null +++ b/cups/raster.h @@ -0,0 +1,405 @@ +/* + * "$Id$" + * + * Raster file definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * This file is part of the CUPS Imaging library. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_RASTER_H_ +# define _CUPS_RASTER_H_ + +/* + * Include necessary headers... + */ + +# include "cups.h" +# include "ppd.h" + + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + +/* + * Every non-PostScript printer driver that supports raster images + * should use the application/vnd.cups-raster image file format. + * Since both the PostScript RIP (pstoraster, based on GNU/GPL + * Ghostscript) and Image RIP (imagetoraster, located in the filter + * directory) use it, using this format saves you a lot of work. + * Also, the PostScript RIP passes any printer options that are in + * a PS file to your driver this way as well... + */ + +/* + * Constants... + */ + +# define CUPS_RASTER_SYNC 0x52615333 /* RaS3 */ +# define CUPS_RASTER_REVSYNC 0x33536152 /* 3SaR */ + +# define CUPS_RASTER_SYNCv1 0x52615374 /* RaSt */ +# define CUPS_RASTER_REVSYNCv1 0x74536152 /* tSaR */ + +# define CUPS_RASTER_SYNCv2 0x52615332 /* RaS2 */ +# define CUPS_RASTER_REVSYNCv2 0x32536152 /* 2SaR */ + +# define CUPS_RASTER_SYNC_PWG CUPS_RASTER_SYNCv2 + + +/* + * The following definition can be used to determine if the + * colorimetric colorspaces (CIEXYZ, CIELAB, and ICCn) are + * defined... + */ + +# define CUPS_RASTER_HAVE_COLORIMETRIC 1 + +/* + * The following definition can be used to determine if the + * device colorspaces (DEVICEn) are defined... + */ + +# define CUPS_RASTER_HAVE_DEVICE 1 + +/* + * The following definition can be used to determine if PWG Raster is supported. + */ + +# define CUPS_RASTER_HAVE_PWGRASTER 1 + + +/* + * Types... + */ + +typedef enum cups_adv_e /**** AdvanceMedia attribute values ****/ +{ + CUPS_ADVANCE_NONE = 0, /* Never advance the roll */ + CUPS_ADVANCE_FILE = 1, /* Advance the roll after this file */ + CUPS_ADVANCE_JOB = 2, /* Advance the roll after this job */ + CUPS_ADVANCE_SET = 3, /* Advance the roll after this set */ + CUPS_ADVANCE_PAGE = 4 /* Advance the roll after this page */ +} cups_adv_t; + +typedef enum cups_bool_e /**** Boolean type ****/ +{ + CUPS_FALSE = 0, /* Logical false */ + CUPS_TRUE = 1 /* Logical true */ +} cups_bool_t; + +typedef enum cups_cspace_e /**** cupsColorSpace attribute values ****/ +{ + CUPS_CSPACE_W = 0, /* Luminance (DeviceGray, gamma 2.2 by default) */ + CUPS_CSPACE_RGB = 1, /* Red, green, blue (DeviceRGB, sRGB by default) */ + CUPS_CSPACE_RGBA = 2, /* Red, green, blue, alpha (DeviceRGB, sRGB by default) */ + CUPS_CSPACE_K = 3, /* Black (DeviceK) */ + CUPS_CSPACE_CMY = 4, /* Cyan, magenta, yellow (DeviceCMY) */ + CUPS_CSPACE_YMC = 5, /* Yellow, magenta, cyan @deprecated@ */ + CUPS_CSPACE_CMYK = 6, /* Cyan, magenta, yellow, black (DeviceCMYK) */ + CUPS_CSPACE_YMCK = 7, /* Yellow, magenta, cyan, black @deprecated@ */ + CUPS_CSPACE_KCMY = 8, /* Black, cyan, magenta, yellow @deprecated@ */ + CUPS_CSPACE_KCMYcm = 9, /* Black, cyan, magenta, yellow, light-cyan, light-magenta @deprecated@ */ + CUPS_CSPACE_GMCK = 10, /* Gold, magenta, yellow, black @deprecated@ */ + CUPS_CSPACE_GMCS = 11, /* Gold, magenta, yellow, silver @deprecated@ */ + CUPS_CSPACE_WHITE = 12, /* White ink (as black) @deprecated@ */ + CUPS_CSPACE_GOLD = 13, /* Gold foil @deprecated@ */ + CUPS_CSPACE_SILVER = 14, /* Silver foil @deprecated@ */ + + CUPS_CSPACE_CIEXYZ = 15, /* CIE XYZ @since CUPS 1.1.19/Mac OS X 10.3@ */ + CUPS_CSPACE_CIELab = 16, /* CIE Lab @since CUPS 1.1.19/Mac OS X 10.3@ */ + CUPS_CSPACE_RGBW = 17, /* Red, green, blue, white (DeviceRGB, sRGB by default) @since CUPS 1.2/Mac OS X 10.5@ */ + CUPS_CSPACE_SW = 18, /* Luminance (gamma 2.2) @since CUPS 1.4.5@ */ + CUPS_CSPACE_SRGB = 19, /* Red, green, blue (sRGB) @since CUPS 1.4.5@ */ + CUPS_CSPACE_ADOBERGB = 20, /* Red, green, blue (Adobe RGB) @since CUPS 1.4.5@ */ + + CUPS_CSPACE_ICC1 = 32, /* ICC-based, 1 color @since CUPS 1.1.19/Mac OS X 10.3@ */ + CUPS_CSPACE_ICC2 = 33, /* ICC-based, 2 colors @since CUPS 1.1.19/Mac OS X 10.3@ */ + CUPS_CSPACE_ICC3 = 34, /* ICC-based, 3 colors @since CUPS 1.1.19/Mac OS X 10.3@ */ + CUPS_CSPACE_ICC4 = 35, /* ICC-based, 4 colors @since CUPS 1.1.19/Mac OS X 10.3@ */ + CUPS_CSPACE_ICC5 = 36, /* ICC-based, 5 colors @since CUPS 1.1.19/Mac OS X 10.3@ */ + CUPS_CSPACE_ICC6 = 37, /* ICC-based, 6 colors @since CUPS 1.1.19/Mac OS X 10.3@ */ + CUPS_CSPACE_ICC7 = 38, /* ICC-based, 7 colors @since CUPS 1.1.19/Mac OS X 10.3@ */ + CUPS_CSPACE_ICC8 = 39, /* ICC-based, 8 colors @since CUPS 1.1.19/Mac OS X 10.3@ */ + CUPS_CSPACE_ICC9 = 40, /* ICC-based, 9 colors @since CUPS 1.1.19/Mac OS X 10.3@ */ + CUPS_CSPACE_ICCA = 41, /* ICC-based, 10 colors @since CUPS 1.1.19/Mac OS X 10.3@ */ + CUPS_CSPACE_ICCB = 42, /* ICC-based, 11 colors @since CUPS 1.1.19/Mac OS X 10.3@ */ + CUPS_CSPACE_ICCC = 43, /* ICC-based, 12 colors @since CUPS 1.1.19/Mac OS X 10.3@ */ + CUPS_CSPACE_ICCD = 44, /* ICC-based, 13 colors @since CUPS 1.1.19/Mac OS X 10.3@ */ + CUPS_CSPACE_ICCE = 45, /* ICC-based, 14 colors @since CUPS 1.1.19/Mac OS X 10.3@ */ + CUPS_CSPACE_ICCF = 46, /* ICC-based, 15 colors @since CUPS 1.1.19/Mac OS X 10.3@ */ + + CUPS_CSPACE_DEVICE1 = 48, /* DeviceN, 1 color @since CUPS 1.4.5@ */ + CUPS_CSPACE_DEVICE2 = 49, /* DeviceN, 2 colors @since CUPS 1.4.5@ */ + CUPS_CSPACE_DEVICE3 = 50, /* DeviceN, 3 colors @since CUPS 1.4.5@ */ + CUPS_CSPACE_DEVICE4 = 51, /* DeviceN, 4 colors @since CUPS 1.4.5@ */ + CUPS_CSPACE_DEVICE5 = 52, /* DeviceN, 5 colors @since CUPS 1.4.5@ */ + CUPS_CSPACE_DEVICE6 = 53, /* DeviceN, 6 colors @since CUPS 1.4.5@ */ + CUPS_CSPACE_DEVICE7 = 54, /* DeviceN, 7 colors @since CUPS 1.4.5@ */ + CUPS_CSPACE_DEVICE8 = 55, /* DeviceN, 8 colors @since CUPS 1.4.5@ */ + CUPS_CSPACE_DEVICE9 = 56, /* DeviceN, 9 colors @since CUPS 1.4.5@ */ + CUPS_CSPACE_DEVICEA = 57, /* DeviceN, 10 colors @since CUPS 1.4.5@ */ + CUPS_CSPACE_DEVICEB = 58, /* DeviceN, 11 colors @since CUPS 1.4.5@ */ + CUPS_CSPACE_DEVICEC = 59, /* DeviceN, 12 colors @since CUPS 1.4.5@ */ + CUPS_CSPACE_DEVICED = 60, /* DeviceN, 13 colors @since CUPS 1.4.5@ */ + CUPS_CSPACE_DEVICEE = 61, /* DeviceN, 14 colors @since CUPS 1.4.5@ */ + CUPS_CSPACE_DEVICEF = 62 /* DeviceN, 15 colors @since CUPS 1.4.5@ */ +} cups_cspace_t; + +typedef enum cups_cut_e /**** CutMedia attribute values ****/ +{ + CUPS_CUT_NONE = 0, /* Never cut the roll */ + CUPS_CUT_FILE = 1, /* Cut the roll after this file */ + CUPS_CUT_JOB = 2, /* Cut the roll after this job */ + CUPS_CUT_SET = 3, /* Cut the roll after this set */ + CUPS_CUT_PAGE = 4 /* Cut the roll after this page */ +} cups_cut_t; + +typedef enum cups_edge_e /**** LeadingEdge attribute values ****/ +{ + CUPS_EDGE_TOP = 0, /* Leading edge is the top of the page */ + CUPS_EDGE_RIGHT = 1, /* Leading edge is the right of the page */ + CUPS_EDGE_BOTTOM = 2, /* Leading edge is the bottom of the page */ + CUPS_EDGE_LEFT = 3 /* Leading edge is the left of the page */ +} cups_edge_t; + +typedef enum cups_jog_e /**** Jog attribute values ****/ +{ + CUPS_JOG_NONE = 0, /* Never move pages */ + CUPS_JOG_FILE = 1, /* Move pages after this file */ + CUPS_JOG_JOB = 2, /* Move pages after this job */ + CUPS_JOG_SET = 3 /* Move pages after this set */ +} cups_jog_t; + +enum cups_mode_e /**** cupsRasterOpen modes ****/ +{ + CUPS_RASTER_READ = 0, /* Open stream for reading */ + CUPS_RASTER_WRITE = 1, /* Open stream for writing */ + CUPS_RASTER_WRITE_COMPRESSED = 2, /* Open stream for compressed writing @since CUPS 1.3/Mac OS X 10.5@ */ + CUPS_RASTER_WRITE_PWG = 3 /* Open stream for compressed writing in PWG mode @since CUPS 1.5/Mac OS X 10.7@ */ +}; + +typedef enum cups_mode_e cups_mode_t; /**** cupsRasterOpen modes ****/ + +typedef enum cups_order_e /**** cupsColorOrder attribute values ****/ +{ + CUPS_ORDER_CHUNKED = 0, /* CMYK CMYK CMYK ... */ + CUPS_ORDER_BANDED = 1, /* CCC MMM YYY KKK ... */ + CUPS_ORDER_PLANAR = 2 /* CCC ... MMM ... YYY ... KKK ... */ +} cups_order_t; + +typedef enum cups_orient_e /**** Orientation attribute values ****/ +{ + CUPS_ORIENT_0 = 0, /* Don't rotate the page */ + CUPS_ORIENT_90 = 1, /* Rotate the page counter-clockwise */ + CUPS_ORIENT_180 = 2, /* Turn the page upside down */ + CUPS_ORIENT_270 = 3 /* Rotate the page clockwise */ +} cups_orient_t; + + +/* + * The page header structure contains the standard PostScript page device + * dictionary, along with some CUPS-specific parameters that are provided + * by the RIPs... + * + * The API supports a "version 1" (from CUPS 1.0 and 1.1) and a "version 2" + * (from CUPS 1.2 and higher) page header, for binary compatibility. + */ + +typedef struct cups_page_header_s /**** Version 1 page header @deprecated@ ****/ +{ + /**** Standard Page Device Dictionary String Values ****/ + char MediaClass[64]; /* MediaClass string */ + char MediaColor[64]; /* MediaColor string */ + char MediaType[64]; /* MediaType string */ + char OutputType[64]; /* OutputType string */ + + /**** Standard Page Device Dictionary Integer Values ****/ + unsigned AdvanceDistance; /* AdvanceDistance value in points */ + cups_adv_t AdvanceMedia; /* AdvanceMedia value (@link cups_adv_t@) */ + cups_bool_t Collate; /* Collated copies value */ + cups_cut_t CutMedia; /* CutMedia value (@link cups_cut_t@) */ + cups_bool_t Duplex; /* Duplexed (double-sided) value */ + unsigned HWResolution[2]; /* Resolution in dots-per-inch */ + unsigned ImagingBoundingBox[4]; /* Pixel region that is painted (points, left, bottom, right, top) */ + cups_bool_t InsertSheet; /* InsertSheet value */ + cups_jog_t Jog; /* Jog value (@link cups_jog_t@) */ + cups_edge_t LeadingEdge; /* LeadingEdge value (@link cups_edge_t@) */ + unsigned Margins[2]; /* Lower-lefthand margins in points */ + cups_bool_t ManualFeed; /* ManualFeed value */ + unsigned MediaPosition; /* MediaPosition value */ + unsigned MediaWeight; /* MediaWeight value in grams/m^2 */ + cups_bool_t MirrorPrint; /* MirrorPrint value */ + cups_bool_t NegativePrint; /* NegativePrint value */ + unsigned NumCopies; /* Number of copies to produce */ + cups_orient_t Orientation; /* Orientation value (@link cups_orient_t@) */ + cups_bool_t OutputFaceUp; /* OutputFaceUp value */ + unsigned PageSize[2]; /* Width and length of page in points */ + cups_bool_t Separations; /* Separations value */ + cups_bool_t TraySwitch; /* TraySwitch value */ + cups_bool_t Tumble; /* Tumble value */ + + /**** CUPS Page Device Dictionary Values ****/ + unsigned cupsWidth; /* Width of page image in pixels */ + unsigned cupsHeight; /* Height of page image in pixels */ + unsigned cupsMediaType; /* Media type code */ + unsigned cupsBitsPerColor; /* Number of bits for each color */ + unsigned cupsBitsPerPixel; /* Number of bits for each pixel */ + unsigned cupsBytesPerLine; /* Number of bytes per line */ + cups_order_t cupsColorOrder; /* Order of colors */ + cups_cspace_t cupsColorSpace; /* True colorspace */ + unsigned cupsCompression; /* Device compression to use */ + unsigned cupsRowCount; /* Rows per band */ + unsigned cupsRowFeed; /* Feed between bands */ + unsigned cupsRowStep; /* Spacing between lines */ +} cups_page_header_t; + +/**** New in CUPS 1.2 ****/ +typedef struct cups_page_header2_s /**** Version 2 page header @since CUPS 1.2/Mac OS X 10.5@ ****/ +{ + /**** Standard Page Device Dictionary String Values ****/ + char MediaClass[64]; /* MediaClass string */ + char MediaColor[64]; /* MediaColor string */ + char MediaType[64]; /* MediaType string */ + char OutputType[64]; /* OutputType string */ + + /**** Standard Page Device Dictionary Integer Values ****/ + unsigned AdvanceDistance; /* AdvanceDistance value in points */ + cups_adv_t AdvanceMedia; /* AdvanceMedia value (@link cups_adv_t@) */ + cups_bool_t Collate; /* Collated copies value */ + cups_cut_t CutMedia; /* CutMedia value (@link cups_cut_t@) */ + cups_bool_t Duplex; /* Duplexed (double-sided) value */ + unsigned HWResolution[2]; /* Resolution in dots-per-inch */ + unsigned ImagingBoundingBox[4]; /* Pixel region that is painted (points, left, bottom, right, top) */ + cups_bool_t InsertSheet; /* InsertSheet value */ + cups_jog_t Jog; /* Jog value (@link cups_jog_t@) */ + cups_edge_t LeadingEdge; /* LeadingEdge value (@link cups_edge_t@) */ + unsigned Margins[2]; /* Lower-lefthand margins in points */ + cups_bool_t ManualFeed; /* ManualFeed value */ + unsigned MediaPosition; /* MediaPosition value */ + unsigned MediaWeight; /* MediaWeight value in grams/m^2 */ + cups_bool_t MirrorPrint; /* MirrorPrint value */ + cups_bool_t NegativePrint; /* NegativePrint value */ + unsigned NumCopies; /* Number of copies to produce */ + cups_orient_t Orientation; /* Orientation value (@link cups_orient_t@) */ + cups_bool_t OutputFaceUp; /* OutputFaceUp value */ + unsigned PageSize[2]; /* Width and length of page in points */ + cups_bool_t Separations; /* Separations value */ + cups_bool_t TraySwitch; /* TraySwitch value */ + cups_bool_t Tumble; /* Tumble value */ + + /**** CUPS Page Device Dictionary Values ****/ + unsigned cupsWidth; /* Width of page image in pixels */ + unsigned cupsHeight; /* Height of page image in pixels */ + unsigned cupsMediaType; /* Media type code */ + unsigned cupsBitsPerColor; /* Number of bits for each color */ + unsigned cupsBitsPerPixel; /* Number of bits for each pixel */ + unsigned cupsBytesPerLine; /* Number of bytes per line */ + cups_order_t cupsColorOrder; /* Order of colors */ + cups_cspace_t cupsColorSpace; /* True colorspace */ + unsigned cupsCompression; /* Device compression to use */ + unsigned cupsRowCount; /* Rows per band */ + unsigned cupsRowFeed; /* Feed between bands */ + unsigned cupsRowStep; /* Spacing between lines */ + + /**** Version 2 Dictionary Values ****/ + unsigned cupsNumColors; /* Number of color compoents @since CUPS 1.2/Mac OS X 10.5@ */ + float cupsBorderlessScalingFactor; + /* Scaling that was applied to page data @since CUPS 1.2/Mac OS X 10.5@ */ + float cupsPageSize[2]; /* Floating point PageSize (scaling * + * factor not applied) @since CUPS 1.2/Mac OS X 10.5@ */ + float cupsImagingBBox[4]; /* Floating point ImagingBoundingBox + * (scaling factor not applied, left, + * bottom, right, top) @since CUPS 1.2/Mac OS X 10.5@ */ + unsigned cupsInteger[16]; /* User-defined integer values @since CUPS 1.2/Mac OS X 10.5@ */ + float cupsReal[16]; /* User-defined floating-point values @since CUPS 1.2/Mac OS X 10.5@ */ + char cupsString[16][64]; /* User-defined string values @since CUPS 1.2/Mac OS X 10.5@ */ + char cupsMarkerType[64]; /* Ink/toner type @since CUPS 1.2/Mac OS X 10.5@ */ + char cupsRenderingIntent[64];/* Color rendering intent @since CUPS 1.2/Mac OS X 10.5@ */ + char cupsPageSizeName[64]; /* PageSize name @since CUPS 1.2/Mac OS X 10.5@ */ +} cups_page_header2_t; + +typedef struct _cups_raster_s cups_raster_t; + /**** Raster stream data ****/ + +typedef int (*cups_interpret_cb_t)(cups_page_header2_t *header, int preferred_bits); + /**** cupsRasterInterpretPPD callback function + * + * This function is called by + * @link cupsRasterInterpretPPD@ to + * validate (and update, as needed) + * the page header attributes. The + * "preferred_bits" argument provides + * the value of the + * @code cupsPreferredBitsPerColor@ + * key from the PostScript page device + * dictionary and is 0 if undefined. + ****/ + +/**** New in CUPS 1.5 ****/ +typedef ssize_t (*cups_raster_iocb_t)(void *ctx, unsigned char *buffer, size_t length); + /**** cupsRasterOpenIO callback function + * + * This function is specified when + * creating a raster stream with + * @link cupsRasterOpenIO@ and handles + * generic reading and writing of raster + * data. It must return -1 on error or + * the number of bytes specified by + * "length" on success. + ****/ + + +/* + * Prototypes... + */ + +extern void cupsRasterClose(cups_raster_t *r); +extern cups_raster_t *cupsRasterOpen(int fd, cups_mode_t mode); +extern unsigned cupsRasterReadHeader(cups_raster_t *r, + cups_page_header_t *h) _CUPS_DEPRECATED; +extern unsigned cupsRasterReadPixels(cups_raster_t *r, + unsigned char *p, unsigned len); +extern unsigned cupsRasterWriteHeader(cups_raster_t *r, + cups_page_header_t *h) _CUPS_DEPRECATED; +extern unsigned cupsRasterWritePixels(cups_raster_t *r, + unsigned char *p, unsigned len); + +/**** New in CUPS 1.2 ****/ +extern int cupsRasterInterpretPPD(cups_page_header2_t *h, + ppd_file_t *ppd, + int num_options, + cups_option_t *options, + cups_interpret_cb_t func) _CUPS_API_1_2; +extern unsigned cupsRasterReadHeader2(cups_raster_t *r, + cups_page_header2_t *h) _CUPS_API_1_2; +extern unsigned cupsRasterWriteHeader2(cups_raster_t *r, + cups_page_header2_t *h) _CUPS_API_1_2; + +/**** New in CUPS 1.3 ****/ +extern const char *cupsRasterErrorString(void) _CUPS_API_1_3; + +/**** New in CUPS 1.5 ****/ +extern cups_raster_t *cupsRasterOpenIO(cups_raster_iocb_t iocb, void *ctx, + cups_mode_t mode); + +# ifdef __cplusplus +} +# endif /* __cplusplus */ + +#endif /* !_CUPS_RASTER_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/request.c b/cups/request.c new file mode 100644 index 0000000000..ba81cebbb1 --- /dev/null +++ b/cups/request.c @@ -0,0 +1,1149 @@ +/* + * "$Id$" + * + * IPP utilities for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsDoFileRequest() - Do an IPP request with a file. + * cupsDoIORequest() - Do an IPP request with file descriptors. + * cupsDoRequest() - Do an IPP request. + * cupsGetResponse() - Get a response to an IPP request. + * cupsLastError() - Return the last IPP status code. + * cupsLastErrorString() - Return the last IPP status-message. + * _cupsNextDelay() - Return the next retry delay value. + * cupsReadResponseData() - Read additional data after the IPP response. + * cupsSendRequest() - Send an IPP request. + * cupsWriteRequestData() - Write additional data after an IPP request. + * _cupsConnect() - Get the default server connection... + * _cupsSetError() - Set the last IPP status code and status-message. + * _cupsSetHTTPError() - Set the last error using the HTTP status. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include +#include +#if defined(WIN32) || defined(__EMX__) +# include +#else +# include +#endif /* WIN32 || __EMX__ */ +#ifndef O_BINARY +# define O_BINARY 0 +#endif /* O_BINARY */ + + +/* + * 'cupsDoFileRequest()' - Do an IPP request with a file. + * + * This function sends the IPP request to the specified server, retrying + * and authenticating as necessary. The request is freed with @link ippDelete@ + * after receiving a valid IPP response. + */ + +ipp_t * /* O - Response data */ +cupsDoFileRequest(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + ipp_t *request, /* I - IPP request */ + const char *resource, /* I - HTTP resource for POST */ + const char *filename) /* I - File to send or @code NULL@ for none */ +{ + ipp_t *response; /* IPP response data */ + int infile; /* Input file */ + + + DEBUG_printf(("cupsDoFileRequest(http=%p, request=%p(%s), resource=\"%s\", " + "filename=\"%s\")", http, request, + request ? ippOpString(request->request.op.operation_id) : "?", + resource, filename)); + + if (filename) + { + if ((infile = open(filename, O_RDONLY | O_BINARY)) < 0) + { + /* + * Can't get file information! + */ + + _cupsSetError(errno == ENOENT ? IPP_NOT_FOUND : IPP_NOT_AUTHORIZED, + NULL, 0); + + ippDelete(request); + + return (NULL); + } + } + else + infile = -1; + + response = cupsDoIORequest(http, request, resource, infile, -1); + + if (infile >= 0) + close(infile); + + return (response); +} + + +/* + * 'cupsDoIORequest()' - Do an IPP request with file descriptors. + * + * This function sends the IPP request to the specified server, retrying + * and authenticating as necessary. The request is freed with ippDelete() + * after receiving a valid IPP response. + * + * If "infile" is a valid file descriptor, cupsDoIORequest() copies + * all of the data from the file after the IPP request message. + * + * If "outfile" is a valid file descriptor, cupsDoIORequest() copies + * all of the data after the IPP response message to the file. + * + * @since CUPS 1.3/Mac OS X 10.5@ + */ + +ipp_t * /* O - Response data */ +cupsDoIORequest(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + ipp_t *request, /* I - IPP request */ + const char *resource, /* I - HTTP resource for POST */ + int infile, /* I - File to read from or -1 for none */ + int outfile) /* I - File to write to or -1 for none */ +{ + ipp_t *response = NULL; /* IPP response data */ + size_t length = 0; /* Content-Length value */ + http_status_t status; /* Status of HTTP request */ + struct stat fileinfo; /* File information */ + int bytes; /* Number of bytes read/written */ + char buffer[32768]; /* Output buffer */ + + + DEBUG_printf(("cupsDoIORequest(http=%p, request=%p(%s), resource=\"%s\", " + "infile=%d, outfile=%d)", http, request, + request ? ippOpString(request->request.op.operation_id) : "?", + resource, infile, outfile)); + + /* + * Range check input... + */ + + if (!request || !resource) + { + ippDelete(request); + + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + + return (NULL); + } + + /* + * Get the default connection as needed... + */ + + if (!http) + if ((http = _cupsConnect()) == NULL) + { + ippDelete(request); + + return (NULL); + } + + /* + * See if we have a file to send... + */ + + if (infile >= 0) + { + if (fstat(infile, &fileinfo)) + { + /* + * Can't get file information! + */ + + _cupsSetError(errno == EBADF ? IPP_NOT_FOUND : IPP_NOT_AUTHORIZED, + NULL, 0); + + ippDelete(request); + + return (NULL); + } + +#ifdef WIN32 + if (fileinfo.st_mode & _S_IFDIR) +#else + if (S_ISDIR(fileinfo.st_mode)) +#endif /* WIN32 */ + { + /* + * Can't send a directory... + */ + + ippDelete(request); + + _cupsSetError(IPP_NOT_POSSIBLE, strerror(EISDIR), 0); + + return (NULL); + } + +#ifndef WIN32 + if (!S_ISREG(fileinfo.st_mode)) + length = 0; /* Chunk when piping */ + else +#endif /* !WIN32 */ + length = ippLength(request) + fileinfo.st_size; + } + else + length = ippLength(request); + + DEBUG_printf(("2cupsDoIORequest: Request length=%ld, total length=%ld", + (long)ippLength(request), (long)length)); + + /* + * Clear any "Local" authentication data since it is probably stale... + */ + + if (http->authstring && !strncmp(http->authstring, "Local ", 6)) + httpSetAuthString(http, NULL, NULL); + + /* + * Loop until we can send the request without authorization problems. + */ + + while (response == NULL) + { + DEBUG_puts("2cupsDoIORequest: setup..."); + + /* + * Send the request... + */ + + status = cupsSendRequest(http, request, resource, length); + + DEBUG_printf(("2cupsDoIORequest: status=%d", status)); + + if (status == HTTP_CONTINUE && request->state == IPP_DATA && infile >= 0) + { + DEBUG_puts("2cupsDoIORequest: file write..."); + + /* + * Send the file with the request... + */ + +#ifndef WIN32 + if (S_ISREG(fileinfo.st_mode)) +#endif /* WIN32 */ + lseek(infile, 0, SEEK_SET); + + while ((bytes = (int)read(infile, buffer, sizeof(buffer))) > 0) + { + if ((status = cupsWriteRequestData(http, buffer, bytes)) + != HTTP_CONTINUE) + break; + } + } + + /* + * Get the server's response... + */ + + if (status != HTTP_ERROR) + { + response = cupsGetResponse(http, resource); + status = httpGetStatus(http); + } + + DEBUG_printf(("2cupsDoIORequest: status=%d", status)); + + if (status == HTTP_ERROR || + (status >= HTTP_BAD_REQUEST && status != HTTP_UNAUTHORIZED && + status != HTTP_UPGRADE_REQUIRED)) + { + _cupsSetHTTPError(status); + break; + } + + if (response && outfile >= 0) + { + /* + * Write trailing data to file... + */ + + while ((bytes = (int)httpRead2(http, buffer, sizeof(buffer))) > 0) + if (write(outfile, buffer, bytes) < bytes) + break; + } + + if (http->state != HTTP_WAITING) + { + /* + * Flush any remaining data... + */ + + httpFlush(http); + } + } + + /* + * Delete the original request and return the response... + */ + + ippDelete(request); + + return (response); +} + + +/* + * 'cupsDoRequest()' - Do an IPP request. + * + * This function sends the IPP request to the specified server, retrying + * and authenticating as necessary. The request is freed with ippDelete() + * after receiving a valid IPP response. + */ + +ipp_t * /* O - Response data */ +cupsDoRequest(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + ipp_t *request, /* I - IPP request */ + const char *resource) /* I - HTTP resource for POST */ +{ + DEBUG_printf(("cupsDoRequest(http=%p, request=%p(%s), resource=\"%s\")", + http, request, + request ? ippOpString(request->request.op.operation_id) : "?", + resource)); + + return (cupsDoIORequest(http, request, resource, -1, -1)); +} + + +/* + * 'cupsGetResponse()' - Get a response to an IPP request. + * + * Use this function to get the response for an IPP request sent using + * cupsSendDocument() or cupsSendRequest(). For requests that return + * additional data, use httpRead() after getting a successful response, + * otherwise call httpFlush() to complete the response processing. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +ipp_t * /* O - Response or @code NULL@ on HTTP error */ +cupsGetResponse(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + const char *resource) /* I - HTTP resource for POST */ +{ + http_status_t status; /* HTTP status */ + ipp_state_t state; /* IPP read state */ + ipp_t *response = NULL; /* IPP response */ + + + DEBUG_printf(("cupsGetResponse(http=%p, resource=\"%s\")", http, resource)); + + /* + * Connect to the default server as needed... + */ + + if (!http) + http = _cupsConnect(); + + if (!http || (http->state != HTTP_POST_RECV && http->state != HTTP_POST_SEND)) + return (NULL); + + /* + * Check for an unfinished chunked request... + */ + + if (http->data_encoding == HTTP_ENCODE_CHUNKED) + { + /* + * Send a 0-length chunk to finish off the request... + */ + + DEBUG_puts("2cupsGetResponse: Finishing chunked POST..."); + + if (httpWrite2(http, "", 0) < 0) + return (NULL); + } + + /* + * Wait for a response from the server... + */ + + DEBUG_printf(("2cupsGetResponse: Update loop, http->status=%d...", + http->status)); + + do + { + status = httpUpdate(http); + } + while (status != HTTP_ERROR && http->state == HTTP_POST_RECV); + + DEBUG_printf(("2cupsGetResponse: status=%d", status)); + + if (status == HTTP_OK) + { + /* + * Get the IPP response... + */ + + response = ippNew(); + + while ((state = ippRead(http, response)) != IPP_DATA) + if (state == IPP_ERROR) + break; + + if (state == IPP_ERROR) + { + /* + * Flush remaining data and delete the response... + */ + + DEBUG_puts("1cupsGetResponse: IPP read error!"); + + httpFlush(http); + + ippDelete(response); + response = NULL; + + _cupsSetError(IPP_SERVICE_UNAVAILABLE, NULL, 0); + http->status = status = HTTP_ERROR; + http->error = EIO; + } + } + else if (status != HTTP_ERROR) + { + /* + * Flush any error message... + */ + + httpFlush(http); + + /* + * Then handle encryption and authentication... + */ + + if (status == HTTP_UNAUTHORIZED) + { + /* + * See if we can do authentication... + */ + + DEBUG_puts("2cupsGetResponse: Need authorization..."); + + if (!cupsDoAuthentication(http, "POST", resource)) + httpReconnect(http); + else + http->status = status = HTTP_AUTHORIZATION_CANCELED; + } + +#ifdef HAVE_SSL + else if (status == HTTP_UPGRADE_REQUIRED) + { + /* + * Force a reconnect with encryption... + */ + + DEBUG_puts("2cupsGetResponse: Need encryption..."); + + if (!httpReconnect(http)) + httpEncryption(http, HTTP_ENCRYPT_REQUIRED); + } +#endif /* HAVE_SSL */ + } + + if (response) + { + ipp_attribute_t *attr; /* status-message attribute */ + + + attr = ippFindAttribute(response, "status-message", IPP_TAG_TEXT); + + DEBUG_printf(("1cupsGetResponse: status-code=%s, status-message=\"%s\"", + ippErrorString(response->request.status.status_code), + attr ? attr->values[0].string.text : "")); + + _cupsSetError(response->request.status.status_code, + attr ? attr->values[0].string.text : + ippErrorString(response->request.status.status_code), 0); + } + else if (status == HTTP_ERROR) + _cupsSetError(IPP_INTERNAL_ERROR, strerror(http->error), 0); + else if (status != HTTP_OK) + _cupsSetHTTPError(status); + + return (response); +} + + +/* + * 'cupsLastError()' - Return the last IPP status code. + */ + +ipp_status_t /* O - IPP status code from last request */ +cupsLastError(void) +{ + return (_cupsGlobals()->last_error); +} + + +/* + * 'cupsLastErrorString()' - Return the last IPP status-message. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +const char * /* O - status-message text from last request */ +cupsLastErrorString(void) +{ + return (_cupsGlobals()->last_status_message); +} + + +/* + * '_cupsNextDelay()' - Return the next retry delay value. + * + * This function currently returns the Fibonacci sequence 1 1 2 3 5 8. + * + * Pass 0 for the current delay value to initialize the sequence. + */ + +int /* O - Next delay value */ +_cupsNextDelay(int current, /* I - Current delay value or 0 */ + int *previous) /* IO - Previous delay value */ +{ + int next; /* Next delay value */ + + + if (current > 0) + { + next = (current + *previous) % 12; + *previous = next < current ? 0 : current; + } + else + { + next = 1; + *previous = 0; + } + + return (next); +} + + +/* + * 'cupsReadResponseData()' - Read additional data after the IPP response. + * + * This function is used after cupsGetResponse() to read the PPD or document + * files for CUPS_GET_PPD and CUPS_GET_DOCUMENT requests, respectively. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +ssize_t /* O - Bytes read, 0 on EOF, -1 on error */ +cupsReadResponseData( + http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + char *buffer, /* I - Buffer to use */ + size_t length) /* I - Number of bytes to read */ +{ + /* + * Get the default connection as needed... + */ + + DEBUG_printf(("cupsReadResponseData(http=%p, buffer=%p, " + "length=" CUPS_LLFMT ")", http, buffer, CUPS_LLCAST length)); + + if (!http) + { + _cups_globals_t *cg = _cupsGlobals(); + /* Pointer to library globals */ + + if ((http = cg->http) == NULL) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("No active connection"), 1); + return (-1); + } + } + + /* + * Then read from the HTTP connection... + */ + + return (httpRead2(http, buffer, length)); +} + + +/* + * 'cupsSendRequest()' - Send an IPP request. + * + * Use httpWrite() to write any additional data (document, PPD file, etc.) + * for the request, cupsGetResponse() to get the IPP response, and httpRead() + * to read any additional data following the response. Only one request can be + * sent/queued at a time. + * + * Unlike cupsDoFileRequest(), cupsDoIORequest(), and cupsDoRequest(), the + * request is not freed. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +http_status_t /* O - Initial HTTP status */ +cupsSendRequest(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + ipp_t *request, /* I - IPP request */ + const char *resource, /* I - Resource path */ + size_t length) /* I - Length of data to follow or @code CUPS_LENGTH_VARIABLE@ */ +{ + http_status_t status; /* Status of HTTP request */ + int got_status; /* Did we get the status? */ + ipp_state_t state; /* State of IPP processing */ + http_status_t expect; /* Expect: header to use */ + + + DEBUG_printf(("cupsSendRequest(http=%p, request=%p(%s), resource=\"%s\", " + "length=" CUPS_LLFMT ")", http, request, + request ? ippOpString(request->request.op.operation_id) : "?", + resource, CUPS_LLCAST length)); + + /* + * Range check input... + */ + + if (!request || !resource) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + + return (HTTP_ERROR); + } + + /* + * Get the default connection as needed... + */ + + if (!http) + if ((http = _cupsConnect()) == NULL) + return (HTTP_SERVICE_UNAVAILABLE); + + /* + * If the prior request was not flushed out, do so now... + */ + + if (http->state == HTTP_GET_SEND || + http->state == HTTP_POST_SEND) + { + DEBUG_puts("2cupsSendRequest: Flush prior response."); + httpFlush(http); + } + else if (http->state != HTTP_WAITING) + { + DEBUG_printf(("1cupsSendRequest: Unknown HTTP state (%d), bailing.", + http->state)); + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + + return (HTTP_ERROR); + } + +#ifdef HAVE_SSL + /* + * See if we have an auth-info attribute and are communicating over + * a non-local link. If so, encrypt the link so that we can pass + * the authentication information securely... + */ + + if (ippFindAttribute(request, "auth-info", IPP_TAG_TEXT) && + !httpAddrLocalhost(http->hostaddr) && !http->tls && + httpEncryption(http, HTTP_ENCRYPT_REQUIRED)) + { + DEBUG_puts("1cupsSendRequest: Unable to encrypt connection."); + return (HTTP_SERVICE_UNAVAILABLE); + } +#endif /* HAVE_SSL */ + + /* + * Reconnect if the last response had a "Connection: close"... + */ + + if (!_cups_strcasecmp(http->fields[HTTP_FIELD_CONNECTION], "close")) + { + DEBUG_puts("2cupsSendRequest: Connection: close"); + httpClearFields(http); + if (httpReconnect(http)) + { + DEBUG_puts("1cupsSendRequest: Unable to reconnect."); + return (HTTP_SERVICE_UNAVAILABLE); + } + } + + /* + * Loop until we can send the request without authorization problems. + */ + + expect = HTTP_CONTINUE; + + for (;;) + { + DEBUG_puts("2cupsSendRequest: Setup..."); + + /* + * Setup the HTTP variables needed... + */ + + httpClearFields(http); + httpSetExpect(http, expect); + httpSetField(http, HTTP_FIELD_CONTENT_TYPE, "application/ipp"); + httpSetLength(http, length); + +#ifdef HAVE_GSSAPI + if (http->authstring && !strncmp(http->authstring, "Negotiate", 9)) + { + /* + * Do not use cached Kerberos credentials since they will look like a + * "replay" attack... + */ + + _cupsSetNegotiateAuthString(http, "POST", resource); + } +#endif /* HAVE_GSSAPI */ + + httpSetField(http, HTTP_FIELD_AUTHORIZATION, http->authstring); + + DEBUG_printf(("2cupsSendRequest: authstring=\"%s\"", http->authstring)); + + /* + * Try the request... + */ + + DEBUG_puts("2cupsSendRequest: Sending HTTP POST..."); + + if (httpPost(http, resource)) + { + DEBUG_puts("2cupsSendRequest: POST failed, reconnecting."); + if (httpReconnect(http)) + { + DEBUG_puts("1cupsSendRequest: Unable to reconnect."); + return (HTTP_SERVICE_UNAVAILABLE); + } + else + continue; + } + + /* + * Send the IPP data... + */ + + DEBUG_puts("2cupsSendRequest: Writing IPP request..."); + + request->state = IPP_IDLE; + status = HTTP_CONTINUE; + got_status = 0; + + while ((state = ippWrite(http, request)) != IPP_DATA) + if (state == IPP_ERROR) + break; + else if (httpCheck(http)) + { + got_status = 1; + + _httpUpdate(http, &status); + if (status >= HTTP_MULTIPLE_CHOICES) + break; + } + + if (state == IPP_ERROR) + { + DEBUG_puts("1cupsSendRequest: Unable to send IPP request."); + + http->status = HTTP_ERROR; + http->state = HTTP_WAITING; + + return (HTTP_ERROR); + } + + /* + * Wait up to 1 second to get the 100-continue response as needed... + */ + + if (!got_status) + { + if (expect == HTTP_CONTINUE) + { + DEBUG_puts("2cupsSendRequest: Waiting for 100-continue..."); + + if (httpWait(http, 1000)) + _httpUpdate(http, &status); + } + else if (httpCheck(http)) + _httpUpdate(http, &status); + } + + DEBUG_printf(("2cupsSendRequest: status=%d", status)); + + /* + * Process the current HTTP status... + */ + + if (status >= HTTP_MULTIPLE_CHOICES) + { + _cupsSetHTTPError(status); + + do + { + status = httpUpdate(http); + } + while (status != HTTP_ERROR && http->state == HTTP_POST_RECV); + + httpFlush(http); + } + + switch (status) + { + case HTTP_ERROR : + case HTTP_CONTINUE : + case HTTP_OK : + DEBUG_printf(("1cupsSendRequest: Returning %d.", status)); + return (status); + + case HTTP_UNAUTHORIZED : + if (cupsDoAuthentication(http, "POST", resource)) + { + DEBUG_puts("1cupsSendRequest: Returning HTTP_AUTHORIZATION_CANCELED."); + return (HTTP_AUTHORIZATION_CANCELED); + } + + DEBUG_puts("2cupsSendRequest: Reconnecting after HTTP_UNAUTHORIZED."); + + if (httpReconnect(http)) + { + DEBUG_puts("1cupsSendRequest: Unable to reconnect."); + return (HTTP_SERVICE_UNAVAILABLE); + } + break; + +#ifdef HAVE_SSL + case HTTP_UPGRADE_REQUIRED : + /* + * Flush any error message, reconnect, and then upgrade with + * encryption... + */ + + DEBUG_puts("2cupsSendRequest: Reconnecting after " + "HTTP_UPGRADE_REQUIRED."); + + if (httpReconnect(http)) + { + DEBUG_puts("1cupsSendRequest: Unable to reconnect."); + return (HTTP_SERVICE_UNAVAILABLE); + } + + DEBUG_puts("2cupsSendRequest: Upgrading to TLS."); + if (httpEncryption(http, HTTP_ENCRYPT_REQUIRED)) + { + DEBUG_puts("1cupsSendRequest: Unable to encrypt connection."); + return (HTTP_SERVICE_UNAVAILABLE); + } + break; +#endif /* HAVE_SSL */ + + case HTTP_EXPECTATION_FAILED : + /* + * Don't try using the Expect: header the next time around... + */ + + expect = (http_status_t)0; + + DEBUG_puts("2cupsSendRequest: Reconnecting after " + "HTTP_EXPECTATION_FAILED."); + + if (httpReconnect(http)) + { + DEBUG_puts("1cupsSendRequest: Unable to reconnect."); + return (HTTP_SERVICE_UNAVAILABLE); + } + break; + + default : + /* + * Some other error... + */ + + return (status); + } + } +} + + +/* + * 'cupsWriteRequestData()' - Write additional data after an IPP request. + * + * This function is used after @link cupsSendRequest@ to provide a PPD and + * after @link cupsStartDocument@ to provide a document file. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +http_status_t /* O - @code HTTP_CONTINUE@ if OK or HTTP status on error */ +cupsWriteRequestData( + http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + const char *buffer, /* I - Bytes to write */ + size_t length) /* I - Number of bytes to write */ +{ + int wused; /* Previous bytes in buffer */ + + + /* + * Get the default connection as needed... + */ + + DEBUG_printf(("cupsWriteRequestData(http=%p, buffer=%p, " + "length=" CUPS_LLFMT ")", http, buffer, CUPS_LLCAST length)); + + if (!http) + { + _cups_globals_t *cg = _cupsGlobals(); + /* Pointer to library globals */ + + if ((http = cg->http) == NULL) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("No active connection"), 1); + DEBUG_puts("1cupsWriteRequestData: Returning HTTP_ERROR."); + return (HTTP_ERROR); + } + } + + /* + * Then write to the HTTP connection... + */ + + wused = http->wused; + + if (httpWrite2(http, buffer, length) < 0) + { + DEBUG_puts("1cupsWriteRequestData: Returning HTTP_ERROR."); + _cupsSetError(IPP_INTERNAL_ERROR, strerror(http->error), 0); + return (HTTP_ERROR); + } + + /* + * Finally, check if we have any pending data from the server... + */ + + if (length >= HTTP_MAX_BUFFER || + http->wused < wused || + (wused > 0 && http->wused == length)) + { + /* + * We've written something to the server, so check for response data... + */ + + if (_httpWait(http, 0, 1)) + { + http_status_t status; /* Status from _httpUpdate */ + + _httpUpdate(http, &status); + if (status >= HTTP_MULTIPLE_CHOICES) + { + _cupsSetHTTPError(status); + + do + { + status = httpUpdate(http); + } + while (status != HTTP_ERROR && http->state == HTTP_POST_RECV); + + httpFlush(http); + } + + DEBUG_printf(("1cupsWriteRequestData: Returning %d.\n", status)); + return (status); + } + } + + DEBUG_puts("1cupsWriteRequestData: Returning HTTP_CONTINUE."); + return (HTTP_CONTINUE); +} + + +/* + * '_cupsConnect()' - Get the default server connection... + */ + +http_t * /* O - HTTP connection */ +_cupsConnect(void) +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + /* + * See if we are connected to the same server... + */ + + if (cg->http) + { + /* + * Compare the connection hostname, port, and encryption settings to + * the cached defaults; these were initialized the first time we + * connected... + */ + + if (strcmp(cg->http->hostname, cg->server) || + cg->ipp_port != _httpAddrPort(cg->http->hostaddr) || + (cg->http->encryption != cg->encryption && + cg->http->encryption == HTTP_ENCRYPT_NEVER)) + { + /* + * Need to close the current connection because something has changed... + */ + + httpClose(cg->http); + cg->http = NULL; + } + } + + /* + * (Re)connect as needed... + */ + + if (!cg->http) + { + if ((cg->http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption())) == NULL) + { + if (errno) + _cupsSetError(IPP_SERVICE_UNAVAILABLE, NULL, 0); + else + _cupsSetError(IPP_SERVICE_UNAVAILABLE, + _("Unable to connect to host."), 1); + } + } + + /* + * Return the cached connection... + */ + + return (cg->http); +} + + +/* + * '_cupsSetError()' - Set the last IPP status code and status-message. + */ + +void +_cupsSetError(ipp_status_t status, /* I - IPP status code */ + const char *message, /* I - status-message value */ + int localize) /* I - Localize the message? */ +{ + _cups_globals_t *cg; /* Global data */ + + + if (!message && errno) + { + message = strerror(errno); + localize = 0; + } + + cg = _cupsGlobals(); + cg->last_error = status; + + if (cg->last_status_message) + { + _cupsStrFree(cg->last_status_message); + + cg->last_status_message = NULL; + } + + if (message) + { + if (localize) + { + /* + * Get the message catalog... + */ + + if (!cg->lang_default) + cg->lang_default = cupsLangDefault(); + + cg->last_status_message = _cupsStrAlloc(_cupsLangString(cg->lang_default, + message)); + } + else + cg->last_status_message = _cupsStrAlloc(message); + } + + DEBUG_printf(("4_cupsSetError: last_error=%s, last_status_message=\"%s\"", + ippErrorString(cg->last_error), cg->last_status_message)); +} + + +/* + * '_cupsSetHTTPError()' - Set the last error using the HTTP status. + */ + +void +_cupsSetHTTPError(http_status_t status) /* I - HTTP status code */ +{ + switch (status) + { + case HTTP_NOT_FOUND : + _cupsSetError(IPP_NOT_FOUND, httpStatus(status), 0); + break; + + case HTTP_UNAUTHORIZED : + _cupsSetError(IPP_NOT_AUTHENTICATED, httpStatus(status), 0); + break; + + case HTTP_AUTHORIZATION_CANCELED : + _cupsSetError(IPP_AUTHENTICATION_CANCELED, httpStatus(status), 0); + break; + + case HTTP_FORBIDDEN : + _cupsSetError(IPP_FORBIDDEN, httpStatus(status), 0); + break; + + case HTTP_BAD_REQUEST : + _cupsSetError(IPP_BAD_REQUEST, httpStatus(status), 0); + break; + + case HTTP_REQUEST_TOO_LARGE : + _cupsSetError(IPP_REQUEST_VALUE, httpStatus(status), 0); + break; + + case HTTP_NOT_IMPLEMENTED : + _cupsSetError(IPP_OPERATION_NOT_SUPPORTED, httpStatus(status), 0); + break; + + case HTTP_NOT_SUPPORTED : + _cupsSetError(IPP_VERSION_NOT_SUPPORTED, httpStatus(status), 0); + break; + + case HTTP_UPGRADE_REQUIRED : + _cupsSetError(IPP_UPGRADE_REQUIRED, httpStatus(status), 0); + break; + + case HTTP_PKI_ERROR : + _cupsSetError(IPP_PKI_ERROR, httpStatus(status), 0); + break; + + case HTTP_ERROR : + _cupsSetError(IPP_INTERNAL_ERROR, strerror(errno), 0); + break; + + default : + DEBUG_printf(("4_cupsSetHTTPError: HTTP error %d mapped to " + "IPP_SERVICE_UNAVAILABLE!", status)); + _cupsSetError(IPP_SERVICE_UNAVAILABLE, httpStatus(status), 0); + break; + } +} + + +/* + * End of "$Id$". + */ diff --git a/cups/sidechannel.c b/cups/sidechannel.c new file mode 100644 index 0000000000..339ef181a3 --- /dev/null +++ b/cups/sidechannel.c @@ -0,0 +1,642 @@ +/* + * "$Id$" + * + * Side-channel API code for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsSideChannelDoRequest() - Send a side-channel command to a backend and + * wait for a response. + * cupsSideChannelRead() - Read a side-channel message. + * cupsSideChannelSNMPGet() - Query a SNMP OID's value. + * cupsSideChannelSNMPWalk() - Query multiple SNMP OID values. + * cupsSideChannelWrite() - Write a side-channel message. + */ + +/* + * Include necessary headers... + */ + +#include "sidechannel.h" +#include "cups-private.h" +#ifdef WIN32 +# include +#else +# include +#endif /* WIN32 */ +#ifdef __hpux +# include +#elif !defined(WIN32) +# include +#endif /* __hpux */ +#ifndef WIN32 +# include +#endif /* !WIN32 */ +#ifdef HAVE_POLL +# include +#endif /* HAVE_POLL */ + + +/* + * Buffer size for side-channel requests... + */ + +#define _CUPS_SC_MAX_DATA 65535 +#define _CUPS_SC_MAX_BUFFER 65540 + + +/* + * 'cupsSideChannelDoRequest()' - Send a side-channel command to a backend and wait for a response. + * + * This function is normally only called by filters, drivers, or port + * monitors in order to communicate with the backend used by the current + * printer. Programs must be prepared to handle timeout or "not + * implemented" status codes, which indicate that the backend or device + * do not support the specified side-channel command. + * + * The "datalen" parameter must be initialized to the size of the buffer + * pointed to by the "data" parameter. cupsSideChannelDoRequest() will + * update the value to contain the number of data bytes in the buffer. + * + * @since CUPS 1.3/Mac OS X 10.5@ + */ + +cups_sc_status_t /* O - Status of command */ +cupsSideChannelDoRequest( + cups_sc_command_t command, /* I - Command to send */ + char *data, /* O - Response data buffer pointer */ + int *datalen, /* IO - Size of data buffer on entry, number of bytes in buffer on return */ + double timeout) /* I - Timeout in seconds */ +{ + cups_sc_status_t status; /* Status of command */ + cups_sc_command_t rcommand; /* Response command */ + + + if (cupsSideChannelWrite(command, CUPS_SC_STATUS_NONE, NULL, 0, timeout)) + return (CUPS_SC_STATUS_TIMEOUT); + + if (cupsSideChannelRead(&rcommand, &status, data, datalen, timeout)) + return (CUPS_SC_STATUS_TIMEOUT); + + if (rcommand != command) + return (CUPS_SC_STATUS_BAD_MESSAGE); + + return (status); +} + + +/* + * 'cupsSideChannelRead()' - Read a side-channel message. + * + * This function is normally only called by backend programs to read + * commands from a filter, driver, or port monitor program. The + * caller must be prepared to handle incomplete or invalid messages + * and return the corresponding status codes. + * + * The "datalen" parameter must be initialized to the size of the buffer + * pointed to by the "data" parameter. cupsSideChannelDoRequest() will + * update the value to contain the number of data bytes in the buffer. + * + * @since CUPS 1.3/Mac OS X 10.5@ + */ + +int /* O - 0 on success, -1 on error */ +cupsSideChannelRead( + cups_sc_command_t *command, /* O - Command code */ + cups_sc_status_t *status, /* O - Status code */ + char *data, /* O - Data buffer pointer */ + int *datalen, /* IO - Size of data buffer on entry, number of bytes in buffer on return */ + double timeout) /* I - Timeout in seconds */ +{ + char *buffer; /* Message buffer */ + int bytes; /* Bytes read */ + int templen; /* Data length from message */ + int nfds; /* Number of file descriptors */ +#ifdef HAVE_POLL + struct pollfd pfd; /* Poll structure for poll() */ +#else /* select() */ + fd_set input_set; /* Input set for select() */ + struct timeval stimeout; /* Timeout value for select() */ +#endif /* HAVE_POLL */ + + + DEBUG_printf(("cupsSideChannelRead(command=%p, status=%p, data=%p, " + "datalen=%p(%d), timeout=%.3f)", command, status, data, + datalen, datalen ? *datalen : -1, timeout)); + + /* + * Range check input... + */ + + if (!command || !status) + return (-1); + + /* + * See if we have pending data on the side-channel socket... + */ + +#ifdef HAVE_POLL + pfd.fd = CUPS_SC_FD; + pfd.events = POLLIN; + + while ((nfds = poll(&pfd, 1, + timeout < 0.0 ? -1 : (long)(timeout * 1000))) < 0 && + (errno == EINTR || errno == EAGAIN)) + ; + +#else /* select() */ + FD_ZERO(&input_set); + FD_SET(CUPS_SC_FD, &input_set); + + stimeout.tv_sec = (int)timeout; + stimeout.tv_usec = (int)(timeout * 1000000) % 1000000; + + while ((nfds = select(CUPS_SC_FD + 1, &input_set, NULL, NULL, + timeout < 0.0 ? NULL : &stimeout)) < 0 && + (errno == EINTR || errno == EAGAIN)) + ; + +#endif /* HAVE_POLL */ + + if (nfds < 1) + { + *command = CUPS_SC_CMD_NONE; + *status = nfds==0 ? CUPS_SC_STATUS_TIMEOUT : CUPS_SC_STATUS_IO_ERROR; + return (-1); + } + + /* + * Read a side-channel message for the format: + * + * Byte(s) Description + * ------- ------------------------------------------- + * 0 Command code + * 1 Status code + * 2-3 Data length (network byte order) + * 4-N Data + */ + + if ((buffer = _cupsBufferGet(_CUPS_SC_MAX_BUFFER)) == NULL) + { + *command = CUPS_SC_CMD_NONE; + *status = CUPS_SC_STATUS_TOO_BIG; + + return (-1); + } + + while ((bytes = read(CUPS_SC_FD, buffer, _CUPS_SC_MAX_BUFFER)) < 0) + if (errno != EINTR && errno != EAGAIN) + { + DEBUG_printf(("1cupsSideChannelRead: Read error: %s", strerror(errno))); + + _cupsBufferRelease(buffer); + + *command = CUPS_SC_CMD_NONE; + *status = CUPS_SC_STATUS_IO_ERROR; + + return (-1); + } + + /* + * Watch for EOF or too few bytes... + */ + + if (bytes < 4) + { + DEBUG_printf(("1cupsSideChannelRead: Short read of %d bytes", bytes)); + + _cupsBufferRelease(buffer); + + *command = CUPS_SC_CMD_NONE; + *status = CUPS_SC_STATUS_BAD_MESSAGE; + + return (-1); + } + + /* + * Validate the command code in the message... + */ + + if (buffer[0] < CUPS_SC_CMD_SOFT_RESET || + buffer[0] >= CUPS_SC_CMD_MAX) + { + DEBUG_printf(("1cupsSideChannelRead: Bad command %d!", buffer[0])); + + _cupsBufferRelease(buffer); + + *command = CUPS_SC_CMD_NONE; + *status = CUPS_SC_STATUS_BAD_MESSAGE; + + return (-1); + } + + *command = (cups_sc_command_t)buffer[0]; + + /* + * Validate the data length in the message... + */ + + templen = ((buffer[2] & 255) << 8) | (buffer[3] & 255); + + if (templen > 0 && (!data || !datalen)) + { + /* + * Either the response is bigger than the provided buffer or the + * response is bigger than we've read... + */ + + *status = CUPS_SC_STATUS_TOO_BIG; + } + else if (!datalen || templen > *datalen || templen > (bytes - 4)) + { + /* + * Either the response is bigger than the provided buffer or the + * response is bigger than we've read... + */ + + *status = CUPS_SC_STATUS_TOO_BIG; + } + else + { + /* + * The response data will fit, copy it over and provide the actual + * length... + */ + + *status = (cups_sc_status_t)buffer[1]; + *datalen = templen; + + memcpy(data, buffer + 4, templen); + } + + _cupsBufferRelease(buffer); + + DEBUG_printf(("1cupsSideChannelRead: Returning status=%d", *status)); + + return (0); +} + + +/* + * 'cupsSideChannelSNMPGet()' - Query a SNMP OID's value. + * + * This function asks the backend to do a SNMP OID query on behalf of the + * filter, port monitor, or backend using the default community name. + * + * "oid" contains a numeric OID consisting of integers separated by periods, + * for example ".1.3.6.1.2.1.43". Symbolic names from SNMP MIBs are not + * supported and must be converted to their numeric forms. + * + * On input, "data" and "datalen" provide the location and size of the + * buffer to hold the OID value as a string. HEX-String (binary) values are + * converted to hexadecimal strings representing the binary data, while + * NULL-Value and unknown OID types are returned as the empty string. + * The returned "datalen" does not include the trailing nul. + * + * @code CUPS_SC_STATUS_NOT_IMPLEMENTED@ is returned by backends that do not + * support SNMP queries. @code CUPS_SC_STATUS_NO_RESPONSE@ is returned when + * the printer does not respond to the SNMP query. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +cups_sc_status_t /* O - Query status */ +cupsSideChannelSNMPGet( + const char *oid, /* I - OID to query */ + char *data, /* I - Buffer for OID value */ + int *datalen, /* IO - Size of OID buffer on entry, size of value on return */ + double timeout) /* I - Timeout in seconds */ +{ + cups_sc_status_t status; /* Status of command */ + cups_sc_command_t rcommand; /* Response command */ + char *real_data; /* Real data buffer for response */ + int real_datalen, /* Real length of data buffer */ + real_oidlen; /* Length of returned OID string */ + + + DEBUG_printf(("cupsSideChannelSNMPGet(oid=\"%s\", data=%p, datalen=%p(%d), " + "timeout=%.3f)", oid, data, datalen, datalen ? *datalen : -1, + timeout)); + + /* + * Range check input... + */ + + if (!oid || !*oid || !data || !datalen || *datalen < 2) + return (CUPS_SC_STATUS_BAD_MESSAGE); + + *data = '\0'; + + /* + * Send the request to the backend and wait for a response... + */ + + if (cupsSideChannelWrite(CUPS_SC_CMD_SNMP_GET, CUPS_SC_STATUS_NONE, oid, + (int)strlen(oid) + 1, timeout)) + return (CUPS_SC_STATUS_TIMEOUT); + + if ((real_data = _cupsBufferGet(_CUPS_SC_MAX_BUFFER)) == NULL) + return (CUPS_SC_STATUS_TOO_BIG); + + real_datalen = _CUPS_SC_MAX_BUFFER; + if (cupsSideChannelRead(&rcommand, &status, real_data, &real_datalen, timeout)) + { + _cupsBufferRelease(real_data); + return (CUPS_SC_STATUS_TIMEOUT); + } + + if (rcommand != CUPS_SC_CMD_SNMP_GET) + { + _cupsBufferRelease(real_data); + return (CUPS_SC_STATUS_BAD_MESSAGE); + } + + if (status == CUPS_SC_STATUS_OK) + { + /* + * Parse the response of the form "oid\0value"... + */ + + real_oidlen = strlen(real_data) + 1; + real_datalen -= real_oidlen; + + if ((real_datalen + 1) > *datalen) + { + _cupsBufferRelease(real_data); + return (CUPS_SC_STATUS_TOO_BIG); + } + + memcpy(data, real_data + real_oidlen, real_datalen); + data[real_datalen] = '\0'; + + *datalen = real_datalen; + } + + _cupsBufferRelease(real_data); + + return (status); +} + + +/* + * 'cupsSideChannelSNMPWalk()' - Query multiple SNMP OID values. + * + * This function asks the backend to do multiple SNMP OID queries on behalf + * of the filter, port monitor, or backend using the default community name. + * All OIDs under the "parent" OID are queried and the results are sent to + * the callback function you provide. + * + * "oid" contains a numeric OID consisting of integers separated by periods, + * for example ".1.3.6.1.2.1.43". Symbolic names from SNMP MIBs are not + * supported and must be converted to their numeric forms. + * + * "timeout" specifies the timeout for each OID query. The total amount of + * time will depend on the number of OID values found and the time required + * for each query. + * + * "cb" provides a function to call for every value that is found. "context" + * is an application-defined pointer that is sent to the callback function + * along with the OID and current data. The data passed to the callback is the + * same as returned by @link cupsSideChannelSNMPGet@. + * + * @code CUPS_SC_STATUS_NOT_IMPLEMENTED@ is returned by backends that do not + * support SNMP queries. @code CUPS_SC_STATUS_NO_RESPONSE@ is returned when + * the printer does not respond to the first SNMP query. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +cups_sc_status_t /* O - Status of first query of @code CUPS_SC_STATUS_OK@ on success */ +cupsSideChannelSNMPWalk( + const char *oid, /* I - First numeric OID to query */ + double timeout, /* I - Timeout for each query in seconds */ + cups_sc_walk_func_t cb, /* I - Function to call with each value */ + void *context) /* I - Application-defined pointer to send to callback */ +{ + cups_sc_status_t status; /* Status of command */ + cups_sc_command_t rcommand; /* Response command */ + char *real_data; /* Real data buffer for response */ + int real_datalen, /* Real length of data buffer */ + real_oidlen, /* Length of returned OID string */ + oidlen; /* Length of first OID */ + const char *current_oid; /* Current OID */ + char last_oid[2048]; /* Last OID */ + + + DEBUG_printf(("cupsSideChannelSNMPWalk(oid=\"%s\", timeout=%.3f, cb=%p, " + "context=%p)", oid, timeout, cb, context)); + + /* + * Range check input... + */ + + if (!oid || !*oid || !cb) + return (CUPS_SC_STATUS_BAD_MESSAGE); + + if ((real_data = _cupsBufferGet(_CUPS_SC_MAX_BUFFER)) == NULL) + return (CUPS_SC_STATUS_TOO_BIG); + + /* + * Loop until the OIDs don't match... + */ + + current_oid = oid; + oidlen = (int)strlen(oid); + last_oid[0] = '\0'; + + do + { + /* + * Send the request to the backend and wait for a response... + */ + + if (cupsSideChannelWrite(CUPS_SC_CMD_SNMP_GET_NEXT, CUPS_SC_STATUS_NONE, + current_oid, (int)strlen(current_oid) + 1, timeout)) + { + _cupsBufferRelease(real_data); + return (CUPS_SC_STATUS_TIMEOUT); + } + + real_datalen = _CUPS_SC_MAX_BUFFER; + if (cupsSideChannelRead(&rcommand, &status, real_data, &real_datalen, + timeout)) + { + _cupsBufferRelease(real_data); + return (CUPS_SC_STATUS_TIMEOUT); + } + + if (rcommand != CUPS_SC_CMD_SNMP_GET_NEXT) + { + _cupsBufferRelease(real_data); + return (CUPS_SC_STATUS_BAD_MESSAGE); + } + + if (status == CUPS_SC_STATUS_OK) + { + /* + * Parse the response of the form "oid\0value"... + */ + + if (strncmp(real_data, oid, oidlen) || real_data[oidlen] != '.' || + !strcmp(real_data, last_oid)) + { + /* + * Done with this set of OIDs... + */ + + _cupsBufferRelease(real_data); + return (CUPS_SC_STATUS_OK); + } + + if (real_datalen < sizeof(real_data)) + real_data[real_datalen] = '\0'; + + real_oidlen = strlen(real_data) + 1; + real_datalen -= real_oidlen; + + /* + * Call the callback with the OID and data... + */ + + (*cb)(real_data, real_data + real_oidlen, real_datalen, context); + + /* + * Update the current OID... + */ + + current_oid = real_data; + strlcpy(last_oid, current_oid, sizeof(last_oid)); + } + } + while (status == CUPS_SC_STATUS_OK); + + _cupsBufferRelease(real_data); + + return (status); +} + + +/* + * 'cupsSideChannelWrite()' - Write a side-channel message. + * + * This function is normally only called by backend programs to send + * responses to a filter, driver, or port monitor program. + * + * @since CUPS 1.3/Mac OS X 10.5@ + */ + +int /* O - 0 on success, -1 on error */ +cupsSideChannelWrite( + cups_sc_command_t command, /* I - Command code */ + cups_sc_status_t status, /* I - Status code */ + const char *data, /* I - Data buffer pointer */ + int datalen, /* I - Number of bytes of data */ + double timeout) /* I - Timeout in seconds */ +{ + char *buffer; /* Message buffer */ + int bytes; /* Bytes written */ +#ifdef HAVE_POLL + struct pollfd pfd; /* Poll structure for poll() */ +#else /* select() */ + fd_set output_set; /* Output set for select() */ + struct timeval stimeout; /* Timeout value for select() */ +#endif /* HAVE_POLL */ + + + /* + * Range check input... + */ + + if (command < CUPS_SC_CMD_SOFT_RESET || command >= CUPS_SC_CMD_MAX || + datalen < 0 || datalen > _CUPS_SC_MAX_DATA || (datalen > 0 && !data)) + return (-1); + + /* + * See if we can safely write to the side-channel socket... + */ + +#ifdef HAVE_POLL + pfd.fd = CUPS_SC_FD; + pfd.events = POLLOUT; + + if (timeout < 0.0) + { + if (poll(&pfd, 1, -1) < 1) + return (-1); + } + else if (poll(&pfd, 1, (long)(timeout * 1000)) < 1) + return (-1); + +#else /* select() */ + FD_ZERO(&output_set); + FD_SET(CUPS_SC_FD, &output_set); + + if (timeout < 0.0) + { + if (select(CUPS_SC_FD + 1, NULL, &output_set, NULL, NULL) < 1) + return (-1); + } + else + { + stimeout.tv_sec = (int)timeout; + stimeout.tv_usec = (int)(timeout * 1000000) % 1000000; + + if (select(CUPS_SC_FD + 1, NULL, &output_set, NULL, &stimeout) < 1) + return (-1); + } +#endif /* HAVE_POLL */ + + /* + * Write a side-channel message in the format: + * + * Byte(s) Description + * ------- ------------------------------------------- + * 0 Command code + * 1 Status code + * 2-3 Data length (network byte order) <= 16384 + * 4-N Data + */ + + if ((buffer = _cupsBufferGet(datalen + 4)) == NULL) + return (-1); + + buffer[0] = command; + buffer[1] = status; + buffer[2] = datalen >> 8; + buffer[3] = datalen & 255; + + bytes = 4; + + if (datalen > 0) + { + memcpy(buffer + 4, data, datalen); + bytes += datalen; + } + + while (write(CUPS_SC_FD, buffer, bytes) < 0) + if (errno != EINTR && errno != EAGAIN) + { + _cupsBufferRelease(buffer); + return (-1); + } + + _cupsBufferRelease(buffer); + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/sidechannel.h b/cups/sidechannel.h new file mode 100644 index 0000000000..ac91a5292c --- /dev/null +++ b/cups/sidechannel.h @@ -0,0 +1,147 @@ +/* + * "$Id$" + * + * Side-channel API definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_SIDECHANNEL_H_ +# define _CUPS_SIDECHANNEL_H_ + +/* + * Include necessary headers... + */ + +# include "versioning.h" + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Constants... + */ + +#define CUPS_SC_FD 4 /* File descriptor for select/poll */ + + +/* + * Enumerations... + */ + +enum cups_sc_bidi_e /**** Bidirectional capability values ****/ +{ + CUPS_SC_BIDI_NOT_SUPPORTED = 0, /* Bidirectional I/O is not supported */ + CUPS_SC_BIDI_SUPPORTED = 1 /* Bidirectional I/O is supported */ +}; +typedef enum cups_sc_bidi_e cups_sc_bidi_t; + /**** Bidirectional capabilities ****/ + +enum cups_sc_command_e /**** Request command codes ****/ +{ + CUPS_SC_CMD_NONE = 0, /* No command @private@ */ + CUPS_SC_CMD_SOFT_RESET = 1, /* Do a soft reset */ + CUPS_SC_CMD_DRAIN_OUTPUT = 2, /* Drain all pending output */ + CUPS_SC_CMD_GET_BIDI = 3, /* Return bidirectional capabilities */ + CUPS_SC_CMD_GET_DEVICE_ID = 4, /* Return the IEEE-1284 device ID */ + CUPS_SC_CMD_GET_STATE = 5, /* Return the device state */ + CUPS_SC_CMD_SNMP_GET = 6, /* Query an SNMP OID @since CUPS 1.4/Mac OS X 10.6@ */ + CUPS_SC_CMD_SNMP_GET_NEXT = 7, /* Query the next SNMP OID @since CUPS 1.4/Mac OS X 10.6@ */ + CUPS_SC_CMD_GET_CONNECTED = 8, /* Return whether the backend is "connected" to the printer @since CUPS 1.5/Mac OS X 10.7@ */ + CUPS_SC_CMD_MAX /* End of valid values @private@ */ +}; +typedef enum cups_sc_command_e cups_sc_command_t; + /**** Request command codes ****/ + +enum cups_sc_connected_e /**** Connectivity values ****/ +{ + CUPS_SC_NOT_CONNECTED = 0, /* Backend is not "connected" to printer */ + CUPS_SC_CONNECTED = 1 /* Backend is "connected" to printer */ +}; +typedef enum cups_sc_connected_e cups_sc_connected_t; + /**** Connectivity values ****/ + + +enum cups_sc_state_e /**** Printer state bits ****/ +{ + CUPS_SC_STATE_OFFLINE = 0, /* Device is offline */ + CUPS_SC_STATE_ONLINE = 1, /* Device is online */ + CUPS_SC_STATE_BUSY = 2, /* Device is busy */ + CUPS_SC_STATE_ERROR = 4, /* Other error condition */ + CUPS_SC_STATE_MEDIA_LOW = 16, /* Paper low condition */ + CUPS_SC_STATE_MEDIA_EMPTY = 32, /* Paper out condition */ + CUPS_SC_STATE_MARKER_LOW = 64, /* Toner/ink low condition */ + CUPS_SC_STATE_MARKER_EMPTY = 128 /* Toner/ink out condition */ +}; +typedef enum cups_sc_state_e cups_sc_state_t; + /**** Printer state bits ****/ + +enum cups_sc_status_e /**** Response status codes ****/ +{ + CUPS_SC_STATUS_NONE, /* No status */ + CUPS_SC_STATUS_OK, /* Operation succeeded */ + CUPS_SC_STATUS_IO_ERROR, /* An I/O error occurred */ + CUPS_SC_STATUS_TIMEOUT, /* The backend did not respond */ + CUPS_SC_STATUS_NO_RESPONSE, /* The device did not respond */ + CUPS_SC_STATUS_BAD_MESSAGE, /* The command/response message was invalid */ + CUPS_SC_STATUS_TOO_BIG, /* Response too big */ + CUPS_SC_STATUS_NOT_IMPLEMENTED /* Command not implemented */ +}; +typedef enum cups_sc_status_e cups_sc_status_t; + /**** Response status codes ****/ + +typedef void (*cups_sc_walk_func_t)(const char *oid, const char *data, + int datalen, void *context); + /**** SNMP walk callback ****/ + + +/* + * Prototypes... + */ + +extern cups_sc_status_t cupsSideChannelDoRequest(cups_sc_command_t command, + char *data, int *datalen, + double timeout) _CUPS_API_1_3; +extern int cupsSideChannelRead(cups_sc_command_t *command, + cups_sc_status_t *status, + char *data, int *datalen, + double timeout) _CUPS_API_1_3; +extern int cupsSideChannelWrite(cups_sc_command_t command, + cups_sc_status_t status, + const char *data, int datalen, + double timeout) _CUPS_API_1_3; + +/**** New in CUPS 1.4 ****/ +extern cups_sc_status_t cupsSideChannelSNMPGet(const char *oid, char *data, + int *datalen, double timeout) + _CUPS_API_1_4; +extern cups_sc_status_t cupsSideChannelSNMPWalk(const char *oid, double timeout, + cups_sc_walk_func_t cb, + void *context) _CUPS_API_1_4; + + +# ifdef __cplusplus +} +# endif /* __cplusplus */ + +#endif /* !_CUPS_SIDECHANNEL_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/snmp-private.h b/cups/snmp-private.h new file mode 100644 index 0000000000..2c6d953719 --- /dev/null +++ b/cups/snmp-private.h @@ -0,0 +1,145 @@ +/* + * "$Id$" + * + * Private SNMP definitions for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 2006-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_SNMP_PRIVATE_H_ +# define _CUPS_SNMP_PRIVATE_H_ + + +/* + * Include necessary headers. + */ + +#include + + +/* + * Constants... + */ + +#define CUPS_SNMP_PORT 161 /* SNMP well-known port */ +#define CUPS_SNMP_MAX_OID 128 /* Maximum number of OID numbers */ +#define CUPS_SNMP_MAX_PACKET 1472 /* Maximum size of SNMP packet */ +#define CUPS_SNMP_MAX_STRING 512 /* Maximum size of string */ +#define CUPS_SNMP_VERSION_1 0 /* SNMPv1 */ + + +/* + * Types... + */ + +enum cups_asn1_e /**** ASN1 request/object types ****/ +{ + CUPS_ASN1_END_OF_CONTENTS = 0x00, /* End-of-contents */ + CUPS_ASN1_BOOLEAN = 0x01, /* BOOLEAN */ + CUPS_ASN1_INTEGER = 0x02, /* INTEGER or ENUMERATION */ + CUPS_ASN1_BIT_STRING = 0x03, /* BIT STRING */ + CUPS_ASN1_OCTET_STRING = 0x04, /* OCTET STRING */ + CUPS_ASN1_NULL_VALUE = 0x05, /* NULL VALUE */ + CUPS_ASN1_OID = 0x06, /* OBJECT IDENTIFIER */ + CUPS_ASN1_SEQUENCE = 0x30, /* SEQUENCE */ + CUPS_ASN1_HEX_STRING = 0x40, /* Binary string aka Hex-STRING */ + CUPS_ASN1_COUNTER = 0x41, /* 32-bit unsigned aka Counter32 */ + CUPS_ASN1_GAUGE = 0x42, /* 32-bit unsigned aka Gauge32 */ + CUPS_ASN1_TIMETICKS = 0x43, /* 32-bit unsigned aka Timeticks32 */ + CUPS_ASN1_GET_REQUEST = 0xa0, /* GetRequest-PDU */ + CUPS_ASN1_GET_NEXT_REQUEST = 0xa1, /* GetNextRequest-PDU */ + CUPS_ASN1_GET_RESPONSE = 0xa2 /* GetResponse-PDU */ +}; +typedef enum cups_asn1_e cups_asn1_t; /**** ASN1 request/object types ****/ + +typedef struct cups_snmp_string_s /**** String value ****/ +{ + unsigned char bytes[CUPS_SNMP_MAX_STRING]; + /* Bytes in string */ + int num_bytes; /* Number of bytes */ +} cups_snmp_string_t; + +union cups_snmp_value_u /**** Object value ****/ +{ + int boolean; /* Boolean value */ + int integer; /* Integer value */ + unsigned counter; /* Counter value */ + unsigned gauge; /* Gauge value */ + unsigned timeticks; /* Timeticks value */ + int oid[CUPS_SNMP_MAX_OID]; /* OID value */ + cups_snmp_string_t string; /* String value */ +}; + +typedef struct cups_snmp_s /**** SNMP data packet ****/ +{ + const char *error; /* Encode/decode error */ + http_addr_t address; /* Source address */ + int version; /* Version number */ + char community[CUPS_SNMP_MAX_STRING]; + /* Community name */ + cups_asn1_t request_type; /* Request type */ + int request_id; /* request-id value */ + int error_status; /* error-status value */ + int error_index; /* error-index value */ + int object_name[CUPS_SNMP_MAX_OID]; + /* object-name value */ + cups_asn1_t object_type; /* object-value type */ + union cups_snmp_value_u + object_value; /* object-value value */ +} cups_snmp_t; + +typedef void (*cups_snmp_cb_t)(cups_snmp_t *packet, void *data); + +/* + * Prototypes... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + +extern void _cupsSNMPClose(int fd) _CUPS_API_1_4; +extern int *_cupsSNMPCopyOID(int *dst, const int *src, int dstsize) + _CUPS_API_1_4; +extern const char *_cupsSNMPDefaultCommunity(void) _CUPS_API_1_4; +extern int _cupsSNMPIsOID(cups_snmp_t *packet, const int *oid) + _CUPS_API_1_4; +extern int _cupsSNMPIsOIDPrefixed(cups_snmp_t *packet, + const int *prefix) _CUPS_API_1_4; +extern char *_cupsSNMPOIDToString(const int *src, char *dst, + size_t dstsize) _CUPS_API_1_4; +extern int _cupsSNMPOpen(int family) _CUPS_API_1_4; +extern cups_snmp_t *_cupsSNMPRead(int fd, cups_snmp_t *packet, + double timeout) _CUPS_API_1_4; +extern void _cupsSNMPSetDebug(int level) _CUPS_API_1_4; +extern int *_cupsSNMPStringToOID(const char *src, + int *dst, int dstsize) + _CUPS_API_1_4; +extern int _cupsSNMPWalk(int fd, http_addr_t *address, int version, + const char *community, const int *prefix, + double timeout, cups_snmp_cb_t cb, + void *data) _CUPS_API_1_4; +extern int _cupsSNMPWrite(int fd, http_addr_t *address, int version, + const char *community, + cups_asn1_t request_type, + const unsigned request_id, + const int *oid) _CUPS_API_1_4; + +# ifdef __cplusplus +} +# endif /* __cplusplus */ +#endif /* !_CUPS_SNMP_PRIVATE_H_ */ + + +/* + * End of "$Id$". + */ diff --git a/cups/snmp.c b/cups/snmp.c new file mode 100644 index 0000000000..29b47d77ee --- /dev/null +++ b/cups/snmp.c @@ -0,0 +1,1737 @@ +/* + * "$Id$" + * + * SNMP functions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 2006-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * _cupsSNMPClose() - Close a SNMP socket. + * _cupsSNMPCopyOID() - Copy an OID. + * _cupsSNMPDefaultCommunity() - Get the default SNMP community name. + * _cupsSNMPIsOID() - Test whether a SNMP response contains the + * specified OID. + * _cupsSNMPIsOIDPrefixed() - Test whether a SNMP response uses the + * specified OID prefix. + * _cupsSNMPOIDToString() - Convert an OID to a string. + * _cupsSNMPOpen() - Open a SNMP socket. + * _cupsSNMPRead() - Read and parse a SNMP response. + * _cupsSNMPSetDebug() - Enable/disable debug logging to stderr. + * _cupsSNMPStringToOID() - Convert a numeric OID string to an OID array. + * _cupsSNMPWalk() - Enumerate a group of OIDs. + * _cupsSNMPWrite() - Send an SNMP query packet. + * asn1_debug() - Decode an ASN1-encoded message. + * asn1_decode_snmp() - Decode a SNMP packet. + * asn1_encode_snmp() - Encode a SNMP packet. + * asn1_get_integer() - Get an integer value. + * asn1_get_length() - Get a value length. + * asn1_get_oid() - Get an OID value. + * asn1_get_packed() - Get a packed integer value. + * asn1_get_string() - Get a string value. + * asn1_get_type() - Get a value type. + * asn1_set_integer() - Set an integer value. + * asn1_set_length() - Set a value length. + * asn1_set_oid() - Set an OID value. + * asn1_set_packed() - Set a packed integer value. + * asn1_size_integer() - Figure out the number of bytes needed for an + * integer value. + * asn1_size_length() - Figure out the number of bytes needed for a + * length value. + * asn1_size_oid() - Figure out the numebr of bytes needed for an + * OID value. + * asn1_size_packed() - Figure out the number of bytes needed for a + * packed integer value. + * snmp_set_error() - Set the localized error for a packet. + */ + +/* + * Include necessary headers. + */ + +#include "cups-private.h" +#include "snmp-private.h" +#ifdef HAVE_POLL +# include +#endif /* HAVE_POLL */ + + +/* + * Local functions... + */ + +static void asn1_debug(const char *prefix, unsigned char *buffer, + size_t len, int indent); +static int asn1_decode_snmp(unsigned char *buffer, size_t len, + cups_snmp_t *packet); +static int asn1_encode_snmp(unsigned char *buffer, size_t len, + cups_snmp_t *packet); +static int asn1_get_integer(unsigned char **buffer, + unsigned char *bufend, + int length); +static int asn1_get_oid(unsigned char **buffer, + unsigned char *bufend, + int length, int *oid, int oidsize); +static int asn1_get_packed(unsigned char **buffer, + unsigned char *bufend); +static char *asn1_get_string(unsigned char **buffer, + unsigned char *bufend, + int length, char *string, + int strsize); +static unsigned asn1_get_length(unsigned char **buffer, + unsigned char *bufend); +static int asn1_get_type(unsigned char **buffer, + unsigned char *bufend); +static void asn1_set_integer(unsigned char **buffer, + int integer); +static void asn1_set_length(unsigned char **buffer, + unsigned length); +static void asn1_set_oid(unsigned char **buffer, + const int *oid); +static void asn1_set_packed(unsigned char **buffer, + int integer); +static int asn1_size_integer(int integer); +static int asn1_size_length(int length); +static int asn1_size_oid(const int *oid); +static int asn1_size_packed(int integer); +static void snmp_set_error(cups_snmp_t *packet, + const char *message); + + +/* + * '_cupsSNMPClose()' - Close a SNMP socket. + */ + +void +_cupsSNMPClose(int fd) /* I - SNMP socket file descriptor */ +{ + DEBUG_printf(("4_cupsSNMPClose(fd=%d)", fd)); + +#ifdef WIN32 + closesocket(fd); +#else + close(fd); +#endif /* WIN32 */ +} + + +/* + * '_cupsSNMPCopyOID()' - Copy an OID. + * + * The array pointed to by "src" is terminated by the value -1. + */ + +int * /* O - New OID */ +_cupsSNMPCopyOID(int *dst, /* I - Destination OID */ + const int *src, /* I - Source OID */ + int dstsize) /* I - Number of integers in dst */ +{ + int i; /* Looping var */ + + + DEBUG_printf(("4_cupsSNMPCopyOID(dst=%p, src=%p, dstsize=%d)", dst, src, + dstsize)); + + for (i = 0, dstsize --; src[i] >= 0 && i < dstsize; i ++) + dst[i] = src[i]; + + dst[i] = -1; + + return (dst); +} + + +/* + * '_cupsSNMPDefaultCommunity()' - Get the default SNMP community name. + * + * The default community name is the first community name found in the + * snmp.conf file. If no community name is defined there, "public" is used. + */ + +const char * /* O - Default community name */ +_cupsSNMPDefaultCommunity(void) +{ + cups_file_t *fp; /* snmp.conf file */ + char line[1024], /* Line from file */ + *value; /* Value from file */ + int linenum; /* Line number in file */ + _cups_globals_t *cg = _cupsGlobals(); /* Global data */ + + + DEBUG_puts("4_cupsSNMPDefaultCommunity()"); + + if (!cg->snmp_community[0]) + { + strlcpy(cg->snmp_community, "public", sizeof(cg->snmp_community)); + + snprintf(line, sizeof(line), "%s/snmp.conf", cg->cups_serverroot); + if ((fp = cupsFileOpen(line, "r")) != NULL) + { + linenum = 0; + while (cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + if (!_cups_strcasecmp(line, "Community") && value) + { + strlcpy(cg->snmp_community, value, sizeof(cg->snmp_community)); + break; + } + + cupsFileClose(fp); + } + } + + DEBUG_printf(("5_cupsSNMPDefaultCommunity: Returning \"%s\"", + cg->snmp_community)); + + return (cg->snmp_community); +} + + +/* + * '_cupsSNMPIsOID()' - Test whether a SNMP response contains the specified OID. + * + * The array pointed to by "oid" is terminated by the value -1. + */ + +int /* O - 1 if equal, 0 if not equal */ +_cupsSNMPIsOID(cups_snmp_t *packet, /* I - Response packet */ + const int *oid) /* I - OID */ +{ + int i; /* Looping var */ + + + /* + * Range check input... + */ + + DEBUG_printf(("4_cupsSNMPIsOID(packet=%p, oid=%p)", packet, oid)); + + if (!packet || !oid) + { + DEBUG_puts("5_cupsSNMPIsOID: Returning 0"); + + return (0); + } + + /* + * Compare OIDs... + */ + + for (i = 0; + i < CUPS_SNMP_MAX_OID && oid[i] >= 0 && packet->object_name[i] >= 0; + i ++) + if (oid[i] != packet->object_name[i]) + { + DEBUG_puts("5_cupsSNMPIsOID: Returning 0"); + + return (0); + } + + DEBUG_printf(("5_cupsSNMPIsOID: Returning %d", + i < CUPS_SNMP_MAX_OID && oid[i] == packet->object_name[i])); + + return (i < CUPS_SNMP_MAX_OID && oid[i] == packet->object_name[i]); +} + + +/* + * '_cupsSNMPIsOIDPrefixed()' - Test whether a SNMP response uses the specified + * OID prefix. + * + * The array pointed to by "prefix" is terminated by the value -1. + */ + +int /* O - 1 if prefixed, 0 if not prefixed */ +_cupsSNMPIsOIDPrefixed( + cups_snmp_t *packet, /* I - Response packet */ + const int *prefix) /* I - OID prefix */ +{ + int i; /* Looping var */ + + + /* + * Range check input... + */ + + DEBUG_printf(("4_cupsSNMPIsOIDPrefixed(packet=%p, prefix=%p)", packet, + prefix)); + + if (!packet || !prefix) + { + DEBUG_puts("5_cupsSNMPIsOIDPrefixed: Returning 0"); + + return (0); + } + + /* + * Compare OIDs... + */ + + for (i = 0; + i < CUPS_SNMP_MAX_OID && prefix[i] >= 0 && packet->object_name[i] >= 0; + i ++) + if (prefix[i] != packet->object_name[i]) + { + DEBUG_puts("5_cupsSNMPIsOIDPrefixed: Returning 0"); + + return (0); + } + + DEBUG_printf(("5_cupsSNMPIsOIDPrefixed: Returning %d", + i < CUPS_SNMP_MAX_OID)); + + return (i < CUPS_SNMP_MAX_OID); +} + + +/* + * '_cupsSNMPOIDToString()' - Convert an OID to a string. + */ + + +char * /* O - New string or @code NULL@ on error */ +_cupsSNMPOIDToString(const int *src, /* I - OID */ + char *dst, /* I - String buffer */ + size_t dstsize) /* I - Size of string buffer */ +{ + char *dstptr, /* Pointer into string buffer */ + *dstend; /* End of string buffer */ + + + DEBUG_printf(("4_cupsSNMPOIDToString(src=%p, dst=%p, dstsize=" CUPS_LLFMT ")", + src, dst, CUPS_LLCAST dstsize)); + + /* + * Range check input... + */ + + if (!src || !dst || dstsize < 4) + return (NULL); + + /* + * Loop through the OID array and build a string... + */ + + for (dstptr = dst, dstend = dstptr + dstsize - 1; + *src >= 0 && dstptr < dstend; + src ++, dstptr += strlen(dstptr)) + snprintf(dstptr, dstend - dstptr + 1, ".%d", *src); + + if (*src >= 0) + return (NULL); + else + return (dst); +} + + +/* + * '_cupsSNMPOpen()' - Open a SNMP socket. + */ + +int /* O - SNMP socket file descriptor */ +_cupsSNMPOpen(int family) /* I - Address family - @code AF_INET@ or @code AF_INET6@ */ +{ + int fd; /* SNMP socket file descriptor */ + int val; /* Socket option value */ + + + /* + * Create the SNMP socket... + */ + + DEBUG_printf(("4_cupsSNMPOpen(family=%d)", family)); + + if ((fd = socket(family, SOCK_DGRAM, 0)) < 0) + { + DEBUG_printf(("5_cupsSNMPOpen: Returning -1 (%s)", strerror(errno))); + + return (-1); + } + + /* + * Set the "broadcast" flag... + */ + + val = 1; + +#ifdef WIN32 + if (setsockopt(fd, SOL_SOCKET, SO_BROADCAST, (char *)&val, sizeof(val))) +#else + if (setsockopt(fd, SOL_SOCKET, SO_BROADCAST, &val, sizeof(val))) +#endif /* WIN32 */ + { + DEBUG_printf(("5_cupsSNMPOpen: Returning -1 (%s)", strerror(errno))); + + close(fd); + + return (-1); + } + + DEBUG_printf(("5_cupsSNMPOpen: Returning %d", fd)); + + return (fd); +} + + +/* + * '_cupsSNMPRead()' - Read and parse a SNMP response. + * + * If "timeout" is negative, @code _cupsSNMPRead@ will wait for a response + * indefinitely. + */ + +cups_snmp_t * /* O - SNMP packet or @code NULL@ if none */ +_cupsSNMPRead(int fd, /* I - SNMP socket file descriptor */ + cups_snmp_t *packet, /* I - SNMP packet buffer */ + double timeout) /* I - Timeout in seconds */ +{ + unsigned char buffer[CUPS_SNMP_MAX_PACKET]; + /* Data packet */ + int bytes; /* Number of bytes received */ + socklen_t addrlen; /* Source address length */ + http_addr_t address; /* Source address */ + + + /* + * Range check input... + */ + + DEBUG_printf(("4_cupsSNMPRead(fd=%d, packet=%p, timeout=%.1f)", fd, packet, + timeout)); + + if (fd < 0 || !packet) + { + DEBUG_puts("5_cupsSNMPRead: Returning NULL"); + + return (NULL); + } + + /* + * Optionally wait for a response... + */ + + if (timeout >= 0.0) + { + int ready; /* Data ready on socket? */ +#ifdef HAVE_POLL + struct pollfd pfd; /* Polled file descriptor */ + + pfd.fd = fd; + pfd.events = POLLIN; + + while ((ready = poll(&pfd, 1, (int)(timeout * 1000.0))) < 0 && + (errno == EINTR || errno == EAGAIN)); + +#else + fd_set input_set; /* select() input set */ + struct timeval stimeout; /* select() timeout */ + + do + { + FD_ZERO(&input_set); + FD_SET(fd, &input_set); + + stimeout.tv_sec = (int)timeout; + stimeout.tv_usec = (int)((timeout - stimeout.tv_sec) * 1000000); + + ready = select(fd + 1, &input_set, NULL, NULL, &stimeout); + } +# ifdef WIN32 + while (ready < 0 && WSAGetLastError() == WSAEINTR); +# else + while (ready < 0 && (errno == EINTR || errno == EAGAIN)); +# endif /* WIN32 */ +#endif /* HAVE_POLL */ + + /* + * If we don't have any data ready, return right away... + */ + + if (ready <= 0) + { + DEBUG_puts("5_cupsSNMPRead: Returning NULL (timeout)"); + + return (NULL); + } + } + + /* + * Read the response data... + */ + + addrlen = sizeof(address); + + if ((bytes = recvfrom(fd, buffer, sizeof(buffer), 0, (void *)&address, + &addrlen)) < 0) + { + DEBUG_printf(("5_cupsSNMPRead: Returning NULL (%s)", strerror(errno))); + + return (NULL); + } + + /* + * Look for the response status code in the SNMP message header... + */ + + asn1_debug("DEBUG: IN ", buffer, bytes, 0); + + asn1_decode_snmp(buffer, bytes, packet); + + memcpy(&(packet->address), &address, sizeof(packet->address)); + + /* + * Return decoded data packet... + */ + + DEBUG_puts("5_cupsSNMPRead: Returning packet"); + + return (packet); +} + + +/* + * '_cupsSNMPSetDebug()' - Enable/disable debug logging to stderr. + */ + +void +_cupsSNMPSetDebug(int level) /* I - 1 to enable debug output, 0 otherwise */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Global data */ + + + DEBUG_printf(("4_cupsSNMPSetDebug(level=%d)", level)); + + cg->snmp_debug = level; +} + + +/* + * '_cupsSNMPStringToOID()' - Convert a numeric OID string to an OID array. + * + * This function converts a string of the form ".N.N.N.N.N" to the + * corresponding OID array terminated by -1. + * + * @code NULL@ is returned if the array is not large enough or the string is + * not a valid OID number. + */ + +int * /* O - Pointer to OID array or @code NULL@ on error */ +_cupsSNMPStringToOID(const char *src, /* I - OID string */ + int *dst, /* I - OID array */ + int dstsize)/* I - Number of integers in OID array */ +{ + int *dstptr, /* Pointer into OID array */ + *dstend; /* End of OID array */ + + + DEBUG_printf(("4_cupsSNMPStringToOID(src=\"%s\", dst=%p, dstsize=%d)", + src, dst, dstsize)); + + /* + * Range check input... + */ + + if (!src || !dst || dstsize < 2) + return (NULL); + + /* + * Skip leading "."... + */ + + if (*src == '.') + src ++; + + /* + * Loop to the end of the string... + */ + + for (dstend = dst + dstsize - 1, dstptr = dst, *dstptr = 0; + *src && dstptr < dstend; + src ++) + { + if (*src == '.') + { + dstptr ++; + *dstptr = 0; + } + else if (isdigit(*src & 255)) + *dstptr = *dstptr * 10 + *src - '0'; + else + break; + } + + if (*src) + return (NULL); + + /* + * Terminate the end of the OID array and return... + */ + + dstptr[1] = -1; + + return (dst); +} + + +/* + * '_cupsSNMPWalk()' - Enumerate a group of OIDs. + * + * This function queries all of the OIDs with the specified OID prefix, + * calling the "cb" function for every response that is received. + * + * The array pointed to by "prefix" is terminated by the value -1. + * + * If "timeout" is negative, @code _cupsSNMPWalk@ will wait for a response + * indefinitely. + */ + +int /* O - Number of OIDs found or -1 on error */ +_cupsSNMPWalk(int fd, /* I - SNMP socket */ + http_addr_t *address, /* I - Address to query */ + int version, /* I - SNMP version */ + const char *community,/* I - Community name */ + const int *prefix, /* I - OID prefix */ + double timeout, /* I - Timeout for each response in seconds */ + cups_snmp_cb_t cb, /* I - Function to call for each response */ + void *data) /* I - User data pointer that is passed to the callback function */ +{ + int count = 0; /* Number of OIDs found */ + int request_id = 0; /* Current request ID */ + cups_snmp_t packet; /* Current response packet */ + int lastoid[CUPS_SNMP_MAX_OID]; + /* Last OID we got */ + + + /* + * Range check input... + */ + + DEBUG_printf(("4_cupsSNMPWalk(fd=%d, address=%p, version=%d, " + "community=\"%s\", prefix=%p, timeout=%.1f, cb=%p, data=%p)", + fd, address, version, community, prefix, timeout, cb, data)); + + if (fd < 0 || !address || version != CUPS_SNMP_VERSION_1 || !community || + !prefix || !cb) + { + DEBUG_puts("5_cupsSNMPWalk: Returning -1"); + + return (-1); + } + + /* + * Copy the OID prefix and then loop until we have no more OIDs... + */ + + _cupsSNMPCopyOID(packet.object_name, prefix, CUPS_SNMP_MAX_OID); + lastoid[0] = -1; + + for (;;) + { + request_id ++; + + if (!_cupsSNMPWrite(fd, address, version, community, + CUPS_ASN1_GET_NEXT_REQUEST, request_id, + packet.object_name)) + { + DEBUG_puts("5_cupsSNMPWalk: Returning -1"); + + return (-1); + } + + if (!_cupsSNMPRead(fd, &packet, timeout)) + { + DEBUG_puts("5_cupsSNMPWalk: Returning -1"); + + return (-1); + } + + if (!_cupsSNMPIsOIDPrefixed(&packet, prefix) || + _cupsSNMPIsOID(&packet, lastoid)) + { + DEBUG_printf(("5_cupsSNMPWalk: Returning %d", count)); + + return (count); + } + + if (packet.error || packet.error_status) + { + DEBUG_printf(("5_cupsSNMPWalk: Returning %d", count > 0 ? count : -1)); + + return (count > 0 ? count : -1); + } + + _cupsSNMPCopyOID(lastoid, packet.object_name, CUPS_SNMP_MAX_OID); + + count ++; + + (*cb)(&packet, data); + } +} + + +/* + * '_cupsSNMPWrite()' - Send an SNMP query packet. + * + * The array pointed to by "oid" is terminated by the value -1. + */ + +int /* O - 1 on success, 0 on error */ +_cupsSNMPWrite( + int fd, /* I - SNMP socket */ + http_addr_t *address, /* I - Address to send to */ + int version, /* I - SNMP version */ + const char *community, /* I - Community name */ + cups_asn1_t request_type, /* I - Request type */ + const unsigned request_id, /* I - Request ID */ + const int *oid) /* I - OID */ +{ + int i; /* Looping var */ + cups_snmp_t packet; /* SNMP message packet */ + unsigned char buffer[CUPS_SNMP_MAX_PACKET]; + /* SNMP message buffer */ + int bytes; /* Size of message */ + http_addr_t temp; /* Copy of address */ + + + /* + * Range check input... + */ + + DEBUG_printf(("4_cupsSNMPWrite(fd=%d, address=%p, version=%d, " + "community=\"%s\", request_type=%d, request_id=%u, oid=%p)", + fd, address, version, community, request_type, request_id, oid)); + + if (fd < 0 || !address || version != CUPS_SNMP_VERSION_1 || !community || + (request_type != CUPS_ASN1_GET_REQUEST && + request_type != CUPS_ASN1_GET_NEXT_REQUEST) || request_id < 1 || !oid) + { + DEBUG_puts("5_cupsSNMPWrite: Returning 0 (bad arguments)"); + + return (0); + } + + /* + * Create the SNMP message... + */ + + memset(&packet, 0, sizeof(packet)); + + packet.version = version; + packet.request_type = request_type; + packet.request_id = request_id; + packet.object_type = CUPS_ASN1_NULL_VALUE; + + strlcpy(packet.community, community, sizeof(packet.community)); + + for (i = 0; oid[i] >= 0 && i < (CUPS_SNMP_MAX_OID - 1); i ++) + packet.object_name[i] = oid[i]; + packet.object_name[i] = -1; + + if (oid[i] >= 0) + { + DEBUG_puts("5_cupsSNMPWrite: Returning 0 (OID too big)"); + + errno = E2BIG; + return (0); + } + + bytes = asn1_encode_snmp(buffer, sizeof(buffer), &packet); + + if (bytes < 0) + { + DEBUG_puts("5_cupsSNMPWrite: Returning 0 (request too big)"); + + errno = E2BIG; + return (0); + } + + asn1_debug("DEBUG: OUT ", buffer, bytes, 0); + + /* + * Send the message... + */ + + temp = *address; + + _httpAddrSetPort(&temp, CUPS_SNMP_PORT); + + return (sendto(fd, buffer, bytes, 0, (void *)&temp, + httpAddrLength(&temp)) == bytes); +} + + +/* + * 'asn1_debug()' - Decode an ASN1-encoded message. + */ + +static void +asn1_debug(const char *prefix, /* I - Prefix string */ + unsigned char *buffer, /* I - Buffer */ + size_t len, /* I - Length of buffer */ + int indent) /* I - Indentation */ +{ + int i; /* Looping var */ + unsigned char *bufend; /* End of buffer */ + int integer; /* Number value */ + int oid[CUPS_SNMP_MAX_OID]; /* OID value */ + char string[CUPS_SNMP_MAX_STRING]; + /* String value */ + unsigned char value_type; /* Type of value */ + int value_length; /* Length of value */ + _cups_globals_t *cg = _cupsGlobals(); /* Global data */ + + + if (cg->snmp_debug <= 0) + return; + + if (cg->snmp_debug > 1 && indent == 0) + { + /* + * Do a hex dump of the packet... + */ + + int j; + + fprintf(stderr, "%sHex Dump (%d bytes):\n", prefix, (int)len); + + for (i = 0; i < (int)len; i += 16) + { + fprintf(stderr, "%s%04x:", prefix, i); + + for (j = 0; j < 16 && (i + j) < (int)len; j ++) + { + if (j && !(j & 3)) + fprintf(stderr, " %02x", buffer[i + j]); + else + fprintf(stderr, " %02x", buffer[i + j]); + } + + while (j < 16) + { + if (j && !(j & 3)) + fputs(" ", stderr); + else + fputs(" ", stderr); + + j ++; + } + + fputs(" ", stderr); + + for (j = 0; j < 16 && (i + j) < (int)len; j ++) + if (buffer[i + j] < ' ' || buffer[i + j] >= 0x7f) + putc('.', stderr); + else + putc(buffer[i + j], stderr); + + putc('\n', stderr); + } + } + + if (indent == 0) + fprintf(stderr, "%sMessage:\n", prefix); + + bufend = buffer + len; + + while (buffer < bufend) + { + /* + * Get value type... + */ + + value_type = asn1_get_type(&buffer, bufend); + value_length = asn1_get_length(&buffer, bufend); + + switch (value_type) + { + case CUPS_ASN1_BOOLEAN : + integer = asn1_get_integer(&buffer, bufend, value_length); + + fprintf(stderr, "%s%*sBOOLEAN %d bytes %d\n", prefix, indent, "", + value_length, integer); + break; + + case CUPS_ASN1_INTEGER : + integer = asn1_get_integer(&buffer, bufend, value_length); + + fprintf(stderr, "%s%*sINTEGER %d bytes %d\n", prefix, indent, "", + value_length, integer); + break; + + case CUPS_ASN1_COUNTER : + integer = asn1_get_integer(&buffer, bufend, value_length); + + fprintf(stderr, "%s%*sCOUNTER %d bytes %u\n", prefix, indent, "", + value_length, (unsigned)integer); + break; + + case CUPS_ASN1_GAUGE : + integer = asn1_get_integer(&buffer, bufend, value_length); + + fprintf(stderr, "%s%*sGAUGE %d bytes %u\n", prefix, indent, "", + value_length, (unsigned)integer); + break; + + case CUPS_ASN1_TIMETICKS : + integer = asn1_get_integer(&buffer, bufend, value_length); + + fprintf(stderr, "%s%*sTIMETICKS %d bytes %u\n", prefix, indent, "", + value_length, (unsigned)integer); + break; + + case CUPS_ASN1_OCTET_STRING : + fprintf(stderr, "%s%*sOCTET STRING %d bytes \"%s\"\n", prefix, + indent, "", value_length, + asn1_get_string(&buffer, bufend, value_length, string, + sizeof(string))); + break; + + case CUPS_ASN1_HEX_STRING : + asn1_get_string(&buffer, bufend, value_length, string, + sizeof(string)); + fprintf(stderr, "%s%*sHex-STRING %d bytes", prefix, + indent, "", value_length); + for (i = 0; i < value_length; i ++) + fprintf(stderr, " %02X", string[i] & 255); + putc('\n', stderr); + break; + + case CUPS_ASN1_NULL_VALUE : + fprintf(stderr, "%s%*sNULL VALUE %d bytes\n", prefix, indent, "", + value_length); + + buffer += value_length; + break; + + case CUPS_ASN1_OID : + integer = asn1_get_oid(&buffer, bufend, value_length, oid, + CUPS_SNMP_MAX_OID); + + fprintf(stderr, "%s%*sOID %d bytes ", prefix, indent, "", + value_length); + for (i = 0; i < integer; i ++) + fprintf(stderr, ".%d", oid[i]); + putc('\n', stderr); + break; + + case CUPS_ASN1_SEQUENCE : + fprintf(stderr, "%s%*sSEQUENCE %d bytes\n", prefix, indent, "", + value_length); + asn1_debug(prefix, buffer, value_length, indent + 4); + + buffer += value_length; + break; + + case CUPS_ASN1_GET_NEXT_REQUEST : + fprintf(stderr, "%s%*sGet-Next-Request-PDU %d bytes\n", prefix, + indent, "", value_length); + asn1_debug(prefix, buffer, value_length, indent + 4); + + buffer += value_length; + break; + + case CUPS_ASN1_GET_REQUEST : + fprintf(stderr, "%s%*sGet-Request-PDU %d bytes\n", prefix, indent, "", + value_length); + asn1_debug(prefix, buffer, value_length, indent + 4); + + buffer += value_length; + break; + + case CUPS_ASN1_GET_RESPONSE : + fprintf(stderr, "%s%*sGet-Response-PDU %d bytes\n", prefix, indent, + "", value_length); + asn1_debug(prefix, buffer, value_length, indent + 4); + + buffer += value_length; + break; + + default : + fprintf(stderr, "%s%*sUNKNOWN(%x) %d bytes\n", prefix, indent, "", + value_type, value_length); + + buffer += value_length; + break; + } + } +} + + +/* + * 'asn1_decode_snmp()' - Decode a SNMP packet. + */ + +static int /* O - 0 on success, -1 on error */ +asn1_decode_snmp(unsigned char *buffer, /* I - Buffer */ + size_t len, /* I - Size of buffer */ + cups_snmp_t *packet) /* I - SNMP packet */ +{ + unsigned char *bufptr, /* Pointer into the data */ + *bufend; /* End of data */ + int length; /* Length of value */ + + + /* + * Initialize the decoding... + */ + + memset(packet, 0, sizeof(cups_snmp_t)); + packet->object_name[0] = -1; + + bufptr = buffer; + bufend = buffer + len; + + if (asn1_get_type(&bufptr, bufend) != CUPS_ASN1_SEQUENCE) + snmp_set_error(packet, _("Packet does not start with SEQUENCE")); + else if (asn1_get_length(&bufptr, bufend) == 0) + snmp_set_error(packet, _("SEQUENCE uses indefinite length")); + else if (asn1_get_type(&bufptr, bufend) != CUPS_ASN1_INTEGER) + snmp_set_error(packet, _("No version number")); + else if ((length = asn1_get_length(&bufptr, bufend)) == 0) + snmp_set_error(packet, _("Version uses indefinite length")); + else if ((packet->version = asn1_get_integer(&bufptr, bufend, length)) + != CUPS_SNMP_VERSION_1) + snmp_set_error(packet, _("Bad SNMP version number")); + else if (asn1_get_type(&bufptr, bufend) != CUPS_ASN1_OCTET_STRING) + snmp_set_error(packet, _("No community name")); + else if ((length = asn1_get_length(&bufptr, bufend)) == 0) + snmp_set_error(packet, _("Community name uses indefinite length")); + else + { + asn1_get_string(&bufptr, bufend, length, packet->community, + sizeof(packet->community)); + + if ((packet->request_type = asn1_get_type(&bufptr, bufend)) + != CUPS_ASN1_GET_RESPONSE) + snmp_set_error(packet, _("Packet does not contain a Get-Response-PDU")); + else if (asn1_get_length(&bufptr, bufend) == 0) + snmp_set_error(packet, _("Get-Response-PDU uses indefinite length")); + else if (asn1_get_type(&bufptr, bufend) != CUPS_ASN1_INTEGER) + snmp_set_error(packet, _("No request-id")); + else if ((length = asn1_get_length(&bufptr, bufend)) == 0) + snmp_set_error(packet, _("request-id uses indefinite length")); + else + { + packet->request_id = asn1_get_integer(&bufptr, bufend, length); + + if (asn1_get_type(&bufptr, bufend) != CUPS_ASN1_INTEGER) + snmp_set_error(packet, _("No error-status")); + else if ((length = asn1_get_length(&bufptr, bufend)) == 0) + snmp_set_error(packet, _("error-status uses indefinite length")); + else + { + packet->error_status = asn1_get_integer(&bufptr, bufend, length); + + if (asn1_get_type(&bufptr, bufend) != CUPS_ASN1_INTEGER) + snmp_set_error(packet, _("No error-index")); + else if ((length = asn1_get_length(&bufptr, bufend)) == 0) + snmp_set_error(packet, _("error-index uses indefinite length")); + else + { + packet->error_index = asn1_get_integer(&bufptr, bufend, length); + + if (asn1_get_type(&bufptr, bufend) != CUPS_ASN1_SEQUENCE) + snmp_set_error(packet, _("No variable-bindings SEQUENCE")); + else if (asn1_get_length(&bufptr, bufend) == 0) + snmp_set_error(packet, + _("variable-bindings uses indefinite length")); + else if (asn1_get_type(&bufptr, bufend) != CUPS_ASN1_SEQUENCE) + snmp_set_error(packet, _("No VarBind SEQUENCE")); + else if (asn1_get_length(&bufptr, bufend) == 0) + snmp_set_error(packet, _("VarBind uses indefinite length")); + else if (asn1_get_type(&bufptr, bufend) != CUPS_ASN1_OID) + snmp_set_error(packet, _("No name OID")); + else if ((length = asn1_get_length(&bufptr, bufend)) == 0) + snmp_set_error(packet, _("Name OID uses indefinite length")); + else + { + asn1_get_oid(&bufptr, bufend, length, packet->object_name, + CUPS_SNMP_MAX_OID); + + packet->object_type = asn1_get_type(&bufptr, bufend); + + if ((length = asn1_get_length(&bufptr, bufend)) == 0 && + packet->object_type != CUPS_ASN1_NULL_VALUE && + packet->object_type != CUPS_ASN1_OCTET_STRING) + snmp_set_error(packet, _("Value uses indefinite length")); + else + { + switch (packet->object_type) + { + case CUPS_ASN1_BOOLEAN : + packet->object_value.boolean = + asn1_get_integer(&bufptr, bufend, length); + break; + + case CUPS_ASN1_INTEGER : + packet->object_value.integer = + asn1_get_integer(&bufptr, bufend, length); + break; + + case CUPS_ASN1_NULL_VALUE : + break; + + case CUPS_ASN1_OCTET_STRING : + case CUPS_ASN1_BIT_STRING : + case CUPS_ASN1_HEX_STRING : + packet->object_value.string.num_bytes = length; + asn1_get_string(&bufptr, bufend, length, + (char *)packet->object_value.string.bytes, + CUPS_SNMP_MAX_STRING); + break; + + case CUPS_ASN1_OID : + asn1_get_oid(&bufptr, bufend, length, + packet->object_value.oid, CUPS_SNMP_MAX_OID); + break; + + case CUPS_ASN1_COUNTER : + packet->object_value.counter = + asn1_get_integer(&bufptr, bufend, length); + break; + + case CUPS_ASN1_GAUGE : + packet->object_value.gauge = + asn1_get_integer(&bufptr, bufend, length); + break; + + case CUPS_ASN1_TIMETICKS : + packet->object_value.timeticks = + asn1_get_integer(&bufptr, bufend, length); + break; + + default : + snmp_set_error(packet, _("Unsupported value type")); + break; + } + } + } + } + } + } + } + + return (packet->error ? -1 : 0); +} + + +/* + * 'asn1_encode_snmp()' - Encode a SNMP packet. + */ + +static int /* O - Length on success, -1 on error */ +asn1_encode_snmp(unsigned char *buffer, /* I - Buffer */ + size_t bufsize, /* I - Size of buffer */ + cups_snmp_t *packet) /* I - SNMP packet */ +{ + unsigned char *bufptr; /* Pointer into buffer */ + int total, /* Total length */ + msglen, /* Length of entire message */ + commlen, /* Length of community string */ + reqlen, /* Length of request */ + listlen, /* Length of variable list */ + varlen, /* Length of variable */ + namelen, /* Length of object name OID */ + valuelen; /* Length of object value */ + + + /* + * Get the lengths of the community string, OID, and message... + */ + + + namelen = asn1_size_oid(packet->object_name); + + switch (packet->object_type) + { + case CUPS_ASN1_NULL_VALUE : + valuelen = 0; + break; + + case CUPS_ASN1_BOOLEAN : + valuelen = asn1_size_integer(packet->object_value.boolean); + break; + + case CUPS_ASN1_INTEGER : + valuelen = asn1_size_integer(packet->object_value.integer); + break; + + case CUPS_ASN1_OCTET_STRING : + valuelen = packet->object_value.string.num_bytes; + break; + + case CUPS_ASN1_OID : + valuelen = asn1_size_oid(packet->object_value.oid); + break; + + default : + packet->error = "Unknown object type"; + return (-1); + } + + varlen = 1 + asn1_size_length(namelen) + namelen + + 1 + asn1_size_length(valuelen) + valuelen; + listlen = 1 + asn1_size_length(varlen) + varlen; + reqlen = 2 + asn1_size_integer(packet->request_id) + + 2 + asn1_size_integer(packet->error_status) + + 2 + asn1_size_integer(packet->error_index) + + 1 + asn1_size_length(listlen) + listlen; + commlen = strlen(packet->community); + msglen = 2 + asn1_size_integer(packet->version) + + 1 + asn1_size_length(commlen) + commlen + + 1 + asn1_size_length(reqlen) + reqlen; + total = 1 + asn1_size_length(msglen) + msglen; + + if (total > (int)bufsize) + { + packet->error = "Message too large for buffer"; + return (-1); + } + + /* + * Then format the message... + */ + + bufptr = buffer; + + *bufptr++ = CUPS_ASN1_SEQUENCE; /* SNMPv1 message header */ + asn1_set_length(&bufptr, msglen); + + asn1_set_integer(&bufptr, packet->version); + /* version */ + + *bufptr++ = CUPS_ASN1_OCTET_STRING; /* community */ + asn1_set_length(&bufptr, commlen); + memcpy(bufptr, packet->community, commlen); + bufptr += commlen; + + *bufptr++ = packet->request_type; /* Get-Request-PDU/Get-Next-Request-PDU */ + asn1_set_length(&bufptr, reqlen); + + asn1_set_integer(&bufptr, packet->request_id); + + asn1_set_integer(&bufptr, packet->error_status); + + asn1_set_integer(&bufptr, packet->error_index); + + *bufptr++ = CUPS_ASN1_SEQUENCE; /* variable-bindings */ + asn1_set_length(&bufptr, listlen); + + *bufptr++ = CUPS_ASN1_SEQUENCE; /* variable */ + asn1_set_length(&bufptr, varlen); + + asn1_set_oid(&bufptr, packet->object_name); + /* ObjectName */ + + switch (packet->object_type) + { + case CUPS_ASN1_NULL_VALUE : + *bufptr++ = CUPS_ASN1_NULL_VALUE; + /* ObjectValue */ + *bufptr++ = 0; /* Length */ + break; + + case CUPS_ASN1_BOOLEAN : + asn1_set_integer(&bufptr, packet->object_value.boolean); + break; + + case CUPS_ASN1_INTEGER : + asn1_set_integer(&bufptr, packet->object_value.integer); + break; + + case CUPS_ASN1_OCTET_STRING : + *bufptr++ = CUPS_ASN1_OCTET_STRING; + asn1_set_length(&bufptr, valuelen); + memcpy(bufptr, packet->object_value.string.bytes, valuelen); + bufptr += valuelen; + break; + + case CUPS_ASN1_OID : + asn1_set_oid(&bufptr, packet->object_value.oid); + break; + + default : + break; + } + + return (bufptr - buffer); +} + + +/* + * 'asn1_get_integer()' - Get an integer value. + */ + +static int /* O - Integer value */ +asn1_get_integer( + unsigned char **buffer, /* IO - Pointer in buffer */ + unsigned char *bufend, /* I - End of buffer */ + int length) /* I - Length of value */ +{ + int value; /* Integer value */ + + + if (length > sizeof(int)) + { + (*buffer) += length; + return (0); + } + + for (value = (**buffer & 0x80) ? -1 : 0; + length > 0 && *buffer < bufend; + length --, (*buffer) ++) + value = (value << 8) | **buffer; + + return (value); +} + + +/* + * 'asn1_get_length()' - Get a value length. + */ + +static unsigned /* O - Length */ +asn1_get_length(unsigned char **buffer, /* IO - Pointer in buffer */ + unsigned char *bufend) /* I - End of buffer */ +{ + unsigned length; /* Length */ + + + length = **buffer; + (*buffer) ++; + + if (length & 128) + { + int count; /* Number of bytes for length */ + + + if ((count = length & 127) > sizeof(unsigned)) + { + (*buffer) += count; + return (0); + } + + for (length = 0; + count > 0 && *buffer < bufend; + count --, (*buffer) ++) + length = (length << 8) | **buffer; + } + + return (length); +} + + +/* + * 'asn1_get_oid()' - Get an OID value. + */ + +static int /* O - Number of OIDs */ +asn1_get_oid( + unsigned char **buffer, /* IO - Pointer in buffer */ + unsigned char *bufend, /* I - End of buffer */ + int length, /* I - Length of value */ + int *oid, /* I - OID buffer */ + int oidsize) /* I - Size of OID buffer */ +{ + unsigned char *valend; /* End of value */ + int *oidptr, /* Current OID */ + *oidend; /* End of OID buffer */ + int number; /* OID number */ + + + valend = *buffer + length; + oidptr = oid; + oidend = oid + oidsize - 1; + + if (valend > bufend) + valend = bufend; + + number = asn1_get_packed(buffer, bufend); + + if (number < 80) + { + *oidptr++ = number / 40; + number = number % 40; + *oidptr++ = number; + } + else + { + *oidptr++ = 2; + number -= 80; + *oidptr++ = number; + } + + while (*buffer < valend) + { + number = asn1_get_packed(buffer, bufend); + + if (oidptr < oidend) + *oidptr++ = number; + } + + *oidptr = -1; + + return (oidptr - oid); +} + + +/* + * 'asn1_get_packed()' - Get a packed integer value. + */ + +static int /* O - Value */ +asn1_get_packed( + unsigned char **buffer, /* IO - Pointer in buffer */ + unsigned char *bufend) /* I - End of buffer */ +{ + int value; /* Value */ + + + value = 0; + + while ((**buffer & 128) && *buffer < bufend) + { + value = (value << 7) | (**buffer & 127); + (*buffer) ++; + } + + if (*buffer < bufend) + { + value = (value << 7) | **buffer; + (*buffer) ++; + } + + return (value); +} + + +/* + * 'asn1_get_string()' - Get a string value. + */ + +static char * /* O - String */ +asn1_get_string( + unsigned char **buffer, /* IO - Pointer in buffer */ + unsigned char *bufend, /* I - End of buffer */ + int length, /* I - Value length */ + char *string, /* I - String buffer */ + int strsize) /* I - String buffer size */ +{ + if (length > (bufend - *buffer)) + length = bufend - *buffer; + + if (length < 0) + { + /* + * Disallow negative lengths! + */ + + *string = '\0'; + } + else if (length < strsize) + { + /* + * String is smaller than the buffer... + */ + + if (length > 0) + memcpy(string, *buffer, length); + + string[length] = '\0'; + } + else + { + /* + * String is larger than the buffer... + */ + + memcpy(string, *buffer, strsize - 1); + string[strsize - 1] = '\0'; + } + + if (length > 0) + (*buffer) += length; + + return (length < 0 ? NULL : string); +} + + +/* + * 'asn1_get_type()' - Get a value type. + */ + +static int /* O - Type */ +asn1_get_type(unsigned char **buffer, /* IO - Pointer in buffer */ + unsigned char *bufend) /* I - End of buffer */ +{ + int type; /* Type */ + + + type = **buffer; + (*buffer) ++; + + if ((type & 31) == 31) + type = asn1_get_packed(buffer, bufend); + + return (type); +} + + +/* + * 'asn1_set_integer()' - Set an integer value. + */ + +static void +asn1_set_integer(unsigned char **buffer,/* IO - Pointer in buffer */ + int integer) /* I - Integer value */ +{ + **buffer = CUPS_ASN1_INTEGER; + (*buffer) ++; + + if (integer > 0x7fffff || integer < -0x800000) + { + **buffer = 4; + (*buffer) ++; + **buffer = integer >> 24; + (*buffer) ++; + **buffer = integer >> 16; + (*buffer) ++; + **buffer = integer >> 8; + (*buffer) ++; + **buffer = integer; + (*buffer) ++; + } + else if (integer > 0x7fff || integer < -0x8000) + { + **buffer = 3; + (*buffer) ++; + **buffer = integer >> 16; + (*buffer) ++; + **buffer = integer >> 8; + (*buffer) ++; + **buffer = integer; + (*buffer) ++; + } + else if (integer > 0x7f || integer < -0x80) + { + **buffer = 2; + (*buffer) ++; + **buffer = integer >> 8; + (*buffer) ++; + **buffer = integer; + (*buffer) ++; + } + else + { + **buffer = 1; + (*buffer) ++; + **buffer = integer; + (*buffer) ++; + } +} + + +/* + * 'asn1_set_length()' - Set a value length. + */ + +static void +asn1_set_length(unsigned char **buffer, /* IO - Pointer in buffer */ + unsigned length) /* I - Length value */ +{ + if (length > 255) + { + **buffer = 0x82; /* 2-byte length */ + (*buffer) ++; + **buffer = length >> 8; + (*buffer) ++; + **buffer = length; + (*buffer) ++; + } + else if (length > 127) + { + **buffer = 0x81; /* 1-byte length */ + (*buffer) ++; + **buffer = length; + (*buffer) ++; + } + else + { + **buffer = length; /* Length */ + (*buffer) ++; + } +} + + +/* + * 'asn1_set_oid()' - Set an OID value. + */ + +static void +asn1_set_oid(unsigned char **buffer, /* IO - Pointer in buffer */ + const int *oid) /* I - OID value */ +{ + **buffer = CUPS_ASN1_OID; + (*buffer) ++; + + asn1_set_length(buffer, asn1_size_oid(oid)); + + if (oid[1] < 0) + { + asn1_set_packed(buffer, oid[0] * 40); + return; + } + + asn1_set_packed(buffer, oid[0] * 40 + oid[1]); + + for (oid += 2; *oid >= 0; oid ++) + asn1_set_packed(buffer, *oid); +} + + +/* + * 'asn1_set_packed()' - Set a packed integer value. + */ + +static void +asn1_set_packed(unsigned char **buffer, /* IO - Pointer in buffer */ + int integer) /* I - Integer value */ +{ + if (integer > 0xfffffff) + { + **buffer = ((integer >> 28) & 0x7f) | 0x80; + (*buffer) ++; + } + + if (integer > 0x1fffff) + { + **buffer = ((integer >> 21) & 0x7f) | 0x80; + (*buffer) ++; + } + + if (integer > 0x3fff) + { + **buffer = ((integer >> 14) & 0x7f) | 0x80; + (*buffer) ++; + } + + if (integer > 0x7f) + { + **buffer = ((integer >> 7) & 0x7f) | 0x80; + (*buffer) ++; + } + + **buffer = integer & 0x7f; + (*buffer) ++; +} + + +/* + * 'asn1_size_integer()' - Figure out the number of bytes needed for an + * integer value. + */ + +static int /* O - Size in bytes */ +asn1_size_integer(int integer) /* I - Integer value */ +{ + if (integer > 0x7fffff || integer < -0x800000) + return (4); + else if (integer > 0x7fff || integer < -0x8000) + return (3); + else if (integer > 0x7f || integer < -0x80) + return (2); + else + return (1); +} + + +/* + * 'asn1_size_length()' - Figure out the number of bytes needed for a + * length value. + */ + +static int /* O - Size in bytes */ +asn1_size_length(int length) /* I - Length value */ +{ + if (length > 0xff) + return (3); + else if (length > 0x7f) + return (2); + else + return (1); +} + + +/* + * 'asn1_size_oid()' - Figure out the numebr of bytes needed for an + * OID value. + */ + +static int /* O - Size in bytes */ +asn1_size_oid(const int *oid) /* I - OID value */ +{ + int length; /* Length of value */ + + + if (oid[1] < 0) + return (asn1_size_packed(oid[0] * 40)); + + for (length = asn1_size_packed(oid[0] * 40 + oid[1]), oid += 2; + *oid >= 0; + oid ++) + length += asn1_size_packed(*oid); + + return (length); +} + + +/* + * 'asn1_size_packed()' - Figure out the number of bytes needed for a + * packed integer value. + */ + +static int /* O - Size in bytes */ +asn1_size_packed(int integer) /* I - Integer value */ +{ + if (integer > 0xfffffff) + return (5); + else if (integer > 0x1fffff) + return (4); + else if (integer > 0x3fff) + return (3); + else if (integer > 0x7f) + return (2); + else + return (1); +} + + +/* + * 'snmp_set_error()' - Set the localized error for a packet. + */ + +static void +snmp_set_error(cups_snmp_t *packet, /* I - Packet */ + const char *message) /* I - Error message */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Global data */ + + + if (!cg->lang_default) + cg->lang_default = cupsLangDefault(); + + packet->error = _cupsLangString(cg->lang_default, message); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/snprintf.c b/cups/snprintf.c new file mode 100644 index 0000000000..595be6594d --- /dev/null +++ b/cups/snprintf.c @@ -0,0 +1,362 @@ +/* + * "$Id$" + * + * snprintf functions for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * _cups_vsnprintf() - Format a string into a fixed size buffer. + * _cups_snprintf() - Format a string into a fixed size buffer. + */ + +/* + * Include necessary headers... + */ + +#include "string-private.h" + + +#ifndef HAVE_VSNPRINTF +/* + * '_cups_vsnprintf()' - Format a string into a fixed size buffer. + */ + +int /* O - Number of bytes formatted */ +_cups_vsnprintf(char *buffer, /* O - Output buffer */ + size_t bufsize, /* O - Size of output buffer */ + const char *format, /* I - printf-style format string */ + va_list ap) /* I - Pointer to additional arguments */ +{ + char *bufptr, /* Pointer to position in buffer */ + *bufend, /* Pointer to end of buffer */ + sign, /* Sign of format width */ + size, /* Size character (h, l, L) */ + type; /* Format type character */ + int width, /* Width of field */ + prec; /* Number of characters of precision */ + char tformat[100], /* Temporary format string for sprintf() */ + *tptr, /* Pointer into temporary format */ + temp[1024]; /* Buffer for formatted numbers */ + char *s; /* Pointer to string */ + int slen; /* Length of string */ + int bytes; /* Total number of bytes needed */ + + + /* + * Loop through the format string, formatting as needed... + */ + + bufptr = buffer; + bufend = buffer + bufsize - 1; + bytes = 0; + + while (*format) + { + if (*format == '%') + { + tptr = tformat; + *tptr++ = *format++; + + if (*format == '%') + { + if (bufptr && bufptr < bufend) *bufptr++ = *format; + bytes ++; + format ++; + continue; + } + else if (strchr(" -+#\'", *format)) + { + *tptr++ = *format; + sign = *format++; + } + else + sign = 0; + + if (*format == '*') + { + /* + * Get width from argument... + */ + + format ++; + width = va_arg(ap, int); + + snprintf(tptr, sizeof(tformat) - (tptr - tformat), "%d", width); + tptr += strlen(tptr); + } + else + { + width = 0; + + while (isdigit(*format & 255)) + { + if (tptr < (tformat + sizeof(tformat) - 1)) + *tptr++ = *format; + + width = width * 10 + *format++ - '0'; + } + } + + if (*format == '.') + { + if (tptr < (tformat + sizeof(tformat) - 1)) + *tptr++ = *format; + + format ++; + + if (*format == '*') + { + /* + * Get precision from argument... + */ + + format ++; + prec = va_arg(ap, int); + + snprintf(tptr, sizeof(tformat) - (tptr - tformat), "%d", prec); + tptr += strlen(tptr); + } + else + { + prec = 0; + + while (isdigit(*format & 255)) + { + if (tptr < (tformat + sizeof(tformat) - 1)) + *tptr++ = *format; + + prec = prec * 10 + *format++ - '0'; + } + } + } + else + prec = -1; + + if (*format == 'l' && format[1] == 'l') + { + size = 'L'; + + if (tptr < (tformat + sizeof(tformat) - 2)) + { + *tptr++ = 'l'; + *tptr++ = 'l'; + } + + format += 2; + } + else if (*format == 'h' || *format == 'l' || *format == 'L') + { + if (tptr < (tformat + sizeof(tformat) - 1)) + *tptr++ = *format; + + size = *format++; + } + + if (!*format) + break; + + if (tptr < (tformat + sizeof(tformat) - 1)) + *tptr++ = *format; + + type = *format++; + *tptr = '\0'; + + switch (type) + { + case 'E' : /* Floating point formats */ + case 'G' : + case 'e' : + case 'f' : + case 'g' : + if ((width + 2) > sizeof(temp)) + break; + + sprintf(temp, tformat, va_arg(ap, double)); + + bytes += (int)strlen(temp); + + if (bufptr) + { + if ((bufptr + strlen(temp)) > bufend) + { + strncpy(bufptr, temp, (size_t)(bufend - bufptr)); + bufptr = bufend; + } + else + { + strcpy(bufptr, temp); + bufptr += strlen(temp); + } + } + break; + + case 'B' : /* Integer formats */ + case 'X' : + case 'b' : + case 'd' : + case 'i' : + case 'o' : + case 'u' : + case 'x' : + if ((width + 2) > sizeof(temp)) + break; + + sprintf(temp, tformat, va_arg(ap, int)); + + bytes += (int)strlen(temp); + + if (bufptr) + { + if ((bufptr + strlen(temp)) > bufend) + { + strncpy(bufptr, temp, (size_t)(bufend - bufptr)); + bufptr = bufend; + } + else + { + strcpy(bufptr, temp); + bufptr += strlen(temp); + } + } + break; + + case 'p' : /* Pointer value */ + if ((width + 2) > sizeof(temp)) + break; + + sprintf(temp, tformat, va_arg(ap, void *)); + + bytes += (int)strlen(temp); + + if (bufptr) + { + if ((bufptr + strlen(temp)) > bufend) + { + strncpy(bufptr, temp, (size_t)(bufend - bufptr)); + bufptr = bufend; + } + else + { + strcpy(bufptr, temp); + bufptr += strlen(temp); + } + } + break; + + case 'c' : /* Character or character array */ + bytes += width; + + if (bufptr) + { + if (width <= 1) + *bufptr++ = va_arg(ap, int); + else + { + if ((bufptr + width) > bufend) + width = (int)(bufend - bufptr); + + memcpy(bufptr, va_arg(ap, char *), (size_t)width); + bufptr += width; + } + } + break; + + case 's' : /* String */ + if ((s = va_arg(ap, char *)) == NULL) + s = "(null)"; + + slen = (int)strlen(s); + if (slen > width && prec != width) + width = slen; + + bytes += width; + + if (bufptr) + { + if ((bufptr + width) > bufend) + width = (int)(bufend - bufptr); + + if (slen > width) + slen = width; + + if (sign == '-') + { + strncpy(bufptr, s, (size_t)slen); + memset(bufptr + slen, ' ', (size_t)(width - slen)); + } + else + { + memset(bufptr, ' ', (size_t)(width - slen)); + strncpy(bufptr + width - slen, s, (size_t)slen); + } + + bufptr += width; + } + break; + + case 'n' : /* Output number of chars so far */ + *(va_arg(ap, int *)) = bytes; + break; + } + } + else + { + bytes ++; + + if (bufptr && bufptr < bufend) + *bufptr++ = *format; + + format ++; + } + } + + /* + * Nul-terminate the string and return the number of characters needed. + */ + + *bufptr = '\0'; + + return (bytes); +} +#endif /* !HAVE_VSNPRINT */ + + +#ifndef HAVE_SNPRINTF +/* + * '_cups_snprintf()' - Format a string into a fixed size buffer. + */ + +int /* O - Number of bytes formatted */ +_cups_snprintf(char *buffer, /* O - Output buffer */ + size_t bufsize, /* O - Size of output buffer */ + const char *format, /* I - printf-style format string */ + ...) /* I - Additional arguments as needed */ +{ + int bytes; /* Number of bytes formatted */ + va_list ap; /* Pointer to additional arguments */ + + + va_start(ap, format); + bytes = vsnprintf(buffer, bufsize, format, ap); + va_end(ap); + + return (bytes); +} +#endif /* !HAVE_SNPRINTF */ + + +/* + * End of "$Id$". + */ + diff --git a/cups/sspi-private.h b/cups/sspi-private.h new file mode 100644 index 0000000000..e8f36c2d14 --- /dev/null +++ b/cups/sspi-private.h @@ -0,0 +1,82 @@ +/* + * Private SSPI definitions for CUPS. + * + * Copyright 2010 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#ifndef _CUPS_SSPI_PRIVATE_H_ +# define _CUPS_SSPI_PRIVATE_H_ + +/* + * Include necessary headers... + */ + +# include +# include +# include +# include +# include +# include +# define SECURITY_WIN32 +# include +# include + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +typedef struct /**** SSPI/SSL data structure ****/ +{ + SOCKET sock; /* TCP/IP socket */ + CredHandle creds; /* Credentials */ + CtxtHandle context; /* SSL context */ + BOOL contextInitialized; /* Is context init'd? */ + SecPkgContext_StreamSizes streamSizes; /* SSL data stream sizes */ + BYTE *decryptBuffer; /* Data pre-decryption*/ + size_t decryptBufferLength; /* Length of decrypt buffer */ + size_t decryptBufferUsed; /* Bytes used in buffer */ + BYTE *readBuffer; /* Data post-decryption */ + size_t readBufferLength; /* Length of read buffer */ + size_t readBufferUsed; /* Bytes used in buffer */ + DWORD certFlags; /* Cert verification flags */ +} _sspi_struct_t; + + +/* + * Prototypes... + */ +_sspi_struct_t *_sspiAlloc(void); +BOOL _sspiAccept(_sspi_struct_t *conn); +BOOL _sspiConnect(_sspi_struct_t *conn, + const CHAR *hostname); +void _sspiFree(_sspi_struct_t *conn); +BOOL _sspiGetCredentials(_sspi_struct_t *conn, + const LPWSTR containerName, + const TCHAR *commonName, + BOOL server); +int _sspiPending(_sspi_struct_t *conn); +int _sspiRead(_sspi_struct_t *conn, + void *buf, size_t len); +void _sspiSetAllowsAnyRoot(_sspi_struct_t *conn, + BOOL allow); +void _sspiSetAllowsExpiredCerts(_sspi_struct_t *conn, + BOOL allow); +int _sspiWrite(_sspi_struct_t *conn, + void *buf, size_t len); + + +# ifdef __cplusplus +} +# endif /* __cplusplus */ +#endif /* !_CUPS_SSPI_PRIVATE_H_ */ diff --git a/cups/sspi.c b/cups/sspi.c new file mode 100644 index 0000000000..0efcb56793 --- /dev/null +++ b/cups/sspi.c @@ -0,0 +1,1485 @@ +/* + * "$Id$" + * + * Windows SSPI SSL implementation for CUPS. + * + * Copyright 2010-2011 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * sspi_alloc() - Allocate SSPI ssl object + * _sspiGetCredentials() - Retrieve an SSL/TLS certificate from the + * system store If one cannot be found, one is + * created. + * _sspiConnect() - Make an SSL connection. This function + * assumes a TCP/IP connection has already been + * successfully made + * _sspiAccept() - Accept an SSL/TLS connection + * _sspiSetAllowsAnyRoot() - Set the client cert policy for untrusted + * root certs + * _sspiSetAllowsExpiredCerts() - Set the client cert policy for expired root + * certs + * _sspiWrite() - Write a buffer to an ssl socket + * _sspiRead() - Read a buffer from an ssl socket + * _sspiPending() - Returns the number of available bytes + * _sspiFree() - Close a connection and free resources + * sspi_verify_certificate() - Verify a server certificate + */ + +/* + * Include necessary headers... + */ + +#include "sspi-private.h" +#include "debug-private.h" + + +/* required to link this library for certificate functions */ +#pragma comment(lib, "Crypt32.lib") +#pragma comment(lib, "Secur32.lib") +#pragma comment(lib, "Ws2_32.lib") + + +#if !defined(SECURITY_FLAG_IGNORE_UNKNOWN_CA) +# define SECURITY_FLAG_IGNORE_UNKNOWN_CA 0x00000100 /* Untrusted root */ +#endif + +#if !defined(SECURITY_FLAG_IGNORE_CERT_DATE_INVALID) +# define SECURITY_FLAG_IGNORE_CERT_DATE_INVALID 0x00002000 /* Expired X509 Cert. */ +#endif + +static DWORD sspi_verify_certificate(PCCERT_CONTEXT serverCert, + const CHAR *serverName, + DWORD dwCertFlags); + + +/* + * 'sspi_alloc()' - Allocate SSPI ssl object + */ +_sspi_struct_t* /* O - New SSPI/SSL object */ +_sspiAlloc(void) +{ + _sspi_struct_t *conn = calloc(sizeof(_sspi_struct_t), 1); + + if (conn) + conn->sock = INVALID_SOCKET; + + return (conn); +} + + +/* + * '_sspiGetCredentials()' - Retrieve an SSL/TLS certificate from the system store + * If one cannot be found, one is created. + */ +BOOL /* O - 1 on success, 0 on failure */ +_sspiGetCredentials(_sspi_struct_t *conn, + /* I - Client connection */ + const LPWSTR container, + /* I - Cert container name */ + const TCHAR *cn, /* I - Common name of certificate */ + BOOL isServer) + /* I - Is caller a server? */ +{ + HCERTSTORE store = NULL; /* Certificate store */ + PCCERT_CONTEXT storedContext = NULL; + /* Context created from the store */ + PCCERT_CONTEXT createdContext = NULL; + /* Context created by us */ + DWORD dwSize = 0; /* 32 bit size */ + PBYTE p = NULL; /* Temporary storage */ + HCRYPTPROV hProv = (HCRYPTPROV) NULL; + /* Handle to a CSP */ + CERT_NAME_BLOB sib; /* Arbitrary array of bytes */ + SCHANNEL_CRED SchannelCred; /* Schannel credential data */ + TimeStamp tsExpiry; /* Time stamp */ + SECURITY_STATUS Status; /* Status */ + HCRYPTKEY hKey = (HCRYPTKEY) NULL; + /* Handle to crypto key */ + CRYPT_KEY_PROV_INFO kpi; /* Key container info */ + SYSTEMTIME et; /* System time */ + CERT_EXTENSIONS exts; /* Array of cert extensions */ + CRYPT_KEY_PROV_INFO ckp; /* Handle to crypto key */ + BOOL ok = TRUE; /* Return value */ + + if (!conn) + return (FALSE); + if (!cn) + return (FALSE); + + if (!CryptAcquireContextW(&hProv, (LPWSTR) container, MS_DEF_PROV_W, + PROV_RSA_FULL, + CRYPT_NEWKEYSET | CRYPT_MACHINE_KEYSET)) + { + if (GetLastError() == NTE_EXISTS) + { + if (!CryptAcquireContextW(&hProv, (LPWSTR) container, MS_DEF_PROV_W, + PROV_RSA_FULL, CRYPT_MACHINE_KEYSET)) + { + DEBUG_printf(("_sspiGetCredentials: CryptAcquireContext failed: %x\n", + GetLastError())); + ok = FALSE; + goto cleanup; + } + } + } + + store = CertOpenStore(CERT_STORE_PROV_SYSTEM, + X509_ASN_ENCODING|PKCS_7_ASN_ENCODING, + hProv, + CERT_SYSTEM_STORE_LOCAL_MACHINE | + CERT_STORE_NO_CRYPT_RELEASE_FLAG | + CERT_STORE_OPEN_EXISTING_FLAG, + L"MY"); + + if (!store) + { + DEBUG_printf(("_sspiGetCredentials: CertOpenSystemStore failed: %x\n", + GetLastError())); + ok = FALSE; + goto cleanup; + } + + dwSize = 0; + + if (!CertStrToName(X509_ASN_ENCODING, cn, CERT_OID_NAME_STR, + NULL, NULL, &dwSize, NULL)) + { + DEBUG_printf(("_sspiGetCredentials: CertStrToName failed: %x\n", + GetLastError())); + ok = FALSE; + goto cleanup; + } + + p = (PBYTE) malloc(dwSize); + + if (!p) + { + DEBUG_printf(("_sspiGetCredentials: malloc failed for %d bytes", dwSize)); + ok = FALSE; + goto cleanup; + } + + if (!CertStrToName(X509_ASN_ENCODING, cn, CERT_OID_NAME_STR, NULL, + p, &dwSize, NULL)) + { + DEBUG_printf(("_sspiGetCredentials: CertStrToName failed: %x", + GetLastError())); + ok = FALSE; + goto cleanup; + } + + sib.cbData = dwSize; + sib.pbData = p; + + storedContext = CertFindCertificateInStore(store, X509_ASN_ENCODING|PKCS_7_ASN_ENCODING, + 0, CERT_FIND_SUBJECT_NAME, &sib, NULL); + + if (!storedContext) + { + /* + * If we couldn't find the context, then we'll + * create a new one + */ + if (!CryptGenKey(hProv, AT_KEYEXCHANGE, CRYPT_EXPORTABLE, &hKey)) + { + DEBUG_printf(("_sspiGetCredentials: CryptGenKey failed: %x", + GetLastError())); + ok = FALSE; + goto cleanup; + } + + ZeroMemory(&kpi, sizeof(kpi)); + kpi.pwszContainerName = (LPWSTR) container; + kpi.pwszProvName = MS_DEF_PROV_W; + kpi.dwProvType = PROV_RSA_FULL; + kpi.dwFlags = CERT_SET_KEY_CONTEXT_PROP_ID; + kpi.dwKeySpec = AT_KEYEXCHANGE; + + GetSystemTime(&et); + et.wYear += 10; + + ZeroMemory(&exts, sizeof(exts)); + + createdContext = CertCreateSelfSignCertificate(hProv, &sib, 0, &kpi, NULL, NULL, + &et, &exts); + + if (!createdContext) + { + DEBUG_printf(("_sspiGetCredentials: CertCreateSelfSignCertificate failed: %x", + GetLastError())); + ok = FALSE; + goto cleanup; + } + + if (!CertAddCertificateContextToStore(store, createdContext, + CERT_STORE_ADD_REPLACE_EXISTING, + &storedContext)) + { + DEBUG_printf(("_sspiGetCredentials: CertAddCertificateContextToStore failed: %x", + GetLastError())); + ok = FALSE; + goto cleanup; + } + + ZeroMemory(&ckp, sizeof(ckp)); + ckp.pwszContainerName = (LPWSTR) container; + ckp.pwszProvName = MS_DEF_PROV_W; + ckp.dwProvType = PROV_RSA_FULL; + ckp.dwFlags = CRYPT_MACHINE_KEYSET; + ckp.dwKeySpec = AT_KEYEXCHANGE; + + if (!CertSetCertificateContextProperty(storedContext, + CERT_KEY_PROV_INFO_PROP_ID, + 0, &ckp)) + { + DEBUG_printf(("_sspiGetCredentials: CertSetCertificateContextProperty failed: %x", + GetLastError())); + ok = FALSE; + goto cleanup; + } + } + + ZeroMemory(&SchannelCred, sizeof(SchannelCred)); + + SchannelCred.dwVersion = SCHANNEL_CRED_VERSION; + SchannelCred.cCreds = 1; + SchannelCred.paCred = &storedContext; + + /* + * SSPI doesn't seem to like it if grbitEnabledProtocols + * is set for a client + */ + if (isServer) + SchannelCred.grbitEnabledProtocols = SP_PROT_SSL3TLS1; + + /* + * Create an SSPI credential. + */ + Status = AcquireCredentialsHandle(NULL, UNISP_NAME, + isServer ? SECPKG_CRED_INBOUND:SECPKG_CRED_OUTBOUND, + NULL, &SchannelCred, NULL, NULL, &conn->creds, + &tsExpiry); + if (Status != SEC_E_OK) + { + DEBUG_printf(("_sspiGetCredentials: AcquireCredentialsHandle failed: %x", Status)); + ok = FALSE; + goto cleanup; + } + +cleanup: + + /* + * Cleanup + */ + if (hKey) + CryptDestroyKey(hKey); + + if (createdContext) + CertFreeCertificateContext(createdContext); + + if (storedContext) + CertFreeCertificateContext(storedContext); + + if (p) + free(p); + + if (store) + CertCloseStore(store, 0); + + if (hProv) + CryptReleaseContext(hProv, 0); + + return (ok); +} + + +/* + * '_sspiConnect()' - Make an SSL connection. This function + * assumes a TCP/IP connection has already + * been successfully made + */ +BOOL /* O - 1 on success, 0 on failure */ +_sspiConnect(_sspi_struct_t *conn, /* I - Client connection */ + const CHAR *hostname) /* I - Server hostname */ +{ + PCCERT_CONTEXT serverCert; /* Server certificate */ + DWORD dwSSPIFlags; /* SSL connection attributes we want */ + DWORD dwSSPIOutFlags; /* SSL connection attributes we got */ + TimeStamp tsExpiry; /* Time stamp */ + SECURITY_STATUS scRet; /* Status */ + DWORD cbData; /* Data count */ + SecBufferDesc inBuffer; /* Array of SecBuffer structs */ + SecBuffer inBuffers[2]; /* Security package buffer */ + SecBufferDesc outBuffer; /* Array of SecBuffer structs */ + SecBuffer outBuffers[1]; /* Security package buffer */ + BOOL ok = TRUE; /* Return value */ + + serverCert = NULL; + + dwSSPIFlags = ISC_REQ_SEQUENCE_DETECT | + ISC_REQ_REPLAY_DETECT | + ISC_REQ_CONFIDENTIALITY | + ISC_RET_EXTENDED_ERROR | + ISC_REQ_ALLOCATE_MEMORY | + ISC_REQ_STREAM; + + /* + * Initiate a ClientHello message and generate a token. + */ + outBuffers[0].pvBuffer = NULL; + outBuffers[0].BufferType = SECBUFFER_TOKEN; + outBuffers[0].cbBuffer = 0; + + outBuffer.cBuffers = 1; + outBuffer.pBuffers = outBuffers; + outBuffer.ulVersion = SECBUFFER_VERSION; + + scRet = InitializeSecurityContext(&conn->creds, NULL, TEXT(""), dwSSPIFlags, + 0, SECURITY_NATIVE_DREP, NULL, 0, &conn->context, + &outBuffer, &dwSSPIOutFlags, &tsExpiry); + + if (scRet != SEC_I_CONTINUE_NEEDED) + { + DEBUG_printf(("_sspiConnect: InitializeSecurityContext(1) failed: %x", scRet)); + ok = FALSE; + goto cleanup; + } + + /* + * Send response to server if there is one. + */ + if (outBuffers[0].cbBuffer && outBuffers[0].pvBuffer) + { + cbData = send(conn->sock, outBuffers[0].pvBuffer, outBuffers[0].cbBuffer, 0); + + if ((cbData == SOCKET_ERROR) || !cbData) + { + DEBUG_printf(("_sspiConnect: send failed: %d", WSAGetLastError())); + FreeContextBuffer(outBuffers[0].pvBuffer); + DeleteSecurityContext(&conn->context); + ok = FALSE; + goto cleanup; + } + + DEBUG_printf(("_sspiConnect: %d bytes of handshake data sent", cbData)); + + /* + * Free output buffer. + */ + FreeContextBuffer(outBuffers[0].pvBuffer); + outBuffers[0].pvBuffer = NULL; + } + + dwSSPIFlags = ISC_REQ_MANUAL_CRED_VALIDATION | + ISC_REQ_SEQUENCE_DETECT | + ISC_REQ_REPLAY_DETECT | + ISC_REQ_CONFIDENTIALITY | + ISC_RET_EXTENDED_ERROR | + ISC_REQ_ALLOCATE_MEMORY | + ISC_REQ_STREAM; + + conn->decryptBufferUsed = 0; + + /* + * Loop until the handshake is finished or an error occurs. + */ + scRet = SEC_I_CONTINUE_NEEDED; + + while(scRet == SEC_I_CONTINUE_NEEDED || + scRet == SEC_E_INCOMPLETE_MESSAGE || + scRet == SEC_I_INCOMPLETE_CREDENTIALS) + { + if ((conn->decryptBufferUsed == 0) || (scRet == SEC_E_INCOMPLETE_MESSAGE)) + { + if (conn->decryptBufferLength <= conn->decryptBufferUsed) + { + conn->decryptBufferLength += 4096; + conn->decryptBuffer = (BYTE*) realloc(conn->decryptBuffer, conn->decryptBufferLength); + + if (!conn->decryptBuffer) + { + DEBUG_printf(("_sspiConnect: unable to allocate %d byte decrypt buffer", + conn->decryptBufferLength)); + SetLastError(E_OUTOFMEMORY); + ok = FALSE; + goto cleanup; + } + } + + cbData = recv(conn->sock, conn->decryptBuffer + conn->decryptBufferUsed, + (int) (conn->decryptBufferLength - conn->decryptBufferUsed), 0); + + if (cbData == SOCKET_ERROR) + { + DEBUG_printf(("_sspiConnect: recv failed: %d", WSAGetLastError())); + ok = FALSE; + goto cleanup; + } + else if (cbData == 0) + { + DEBUG_printf(("_sspiConnect: server unexpectedly disconnected")); + ok = FALSE; + goto cleanup; + } + + DEBUG_printf(("_sspiConnect: %d bytes of handshake data received", + cbData)); + + conn->decryptBufferUsed += cbData; + } + + /* + * Set up the input buffers. Buffer 0 is used to pass in data + * received from the server. Schannel will consume some or all + * of this. Leftover data (if any) will be placed in buffer 1 and + * given a buffer type of SECBUFFER_EXTRA. + */ + inBuffers[0].pvBuffer = conn->decryptBuffer; + inBuffers[0].cbBuffer = (unsigned long) conn->decryptBufferUsed; + inBuffers[0].BufferType = SECBUFFER_TOKEN; + + inBuffers[1].pvBuffer = NULL; + inBuffers[1].cbBuffer = 0; + inBuffers[1].BufferType = SECBUFFER_EMPTY; + + inBuffer.cBuffers = 2; + inBuffer.pBuffers = inBuffers; + inBuffer.ulVersion = SECBUFFER_VERSION; + + /* + * Set up the output buffers. These are initialized to NULL + * so as to make it less likely we'll attempt to free random + * garbage later. + */ + outBuffers[0].pvBuffer = NULL; + outBuffers[0].BufferType= SECBUFFER_TOKEN; + outBuffers[0].cbBuffer = 0; + + outBuffer.cBuffers = 1; + outBuffer.pBuffers = outBuffers; + outBuffer.ulVersion = SECBUFFER_VERSION; + + /* + * Call InitializeSecurityContext. + */ + scRet = InitializeSecurityContext(&conn->creds, &conn->context, NULL, dwSSPIFlags, + 0, SECURITY_NATIVE_DREP, &inBuffer, 0, NULL, + &outBuffer, &dwSSPIOutFlags, &tsExpiry); + + /* + * If InitializeSecurityContext was successful (or if the error was + * one of the special extended ones), send the contends of the output + * buffer to the server. + */ + if (scRet == SEC_E_OK || + scRet == SEC_I_CONTINUE_NEEDED || + FAILED(scRet) && (dwSSPIOutFlags & ISC_RET_EXTENDED_ERROR)) + { + if (outBuffers[0].cbBuffer && outBuffers[0].pvBuffer) + { + cbData = send(conn->sock, outBuffers[0].pvBuffer, outBuffers[0].cbBuffer, 0); + + if ((cbData == SOCKET_ERROR) || !cbData) + { + DEBUG_printf(("_sspiConnect: send failed: %d", WSAGetLastError())); + FreeContextBuffer(outBuffers[0].pvBuffer); + DeleteSecurityContext(&conn->context); + ok = FALSE; + goto cleanup; + } + + DEBUG_printf(("_sspiConnect: %d bytes of handshake data sent", cbData)); + + /* + * Free output buffer. + */ + FreeContextBuffer(outBuffers[0].pvBuffer); + outBuffers[0].pvBuffer = NULL; + } + } + + /* + * If InitializeSecurityContext returned SEC_E_INCOMPLETE_MESSAGE, + * then we need to read more data from the server and try again. + */ + if (scRet == SEC_E_INCOMPLETE_MESSAGE) + continue; + + /* + * If InitializeSecurityContext returned SEC_E_OK, then the + * handshake completed successfully. + */ + if (scRet == SEC_E_OK) + { + /* + * If the "extra" buffer contains data, this is encrypted application + * protocol layer stuff. It needs to be saved. The application layer + * will later decrypt it with DecryptMessage. + */ + DEBUG_printf(("_sspiConnect: Handshake was successful")); + + if (inBuffers[1].BufferType == SECBUFFER_EXTRA) + { + if (conn->decryptBufferLength < inBuffers[1].cbBuffer) + { + conn->decryptBuffer = realloc(conn->decryptBuffer, inBuffers[1].cbBuffer); + + if (!conn->decryptBuffer) + { + DEBUG_printf(("_sspiConnect: unable to allocate %d bytes for decrypt buffer", + inBuffers[1].cbBuffer)); + SetLastError(E_OUTOFMEMORY); + ok = FALSE; + goto cleanup; + } + } + + memmove(conn->decryptBuffer, + conn->decryptBuffer + (conn->decryptBufferUsed - inBuffers[1].cbBuffer), + inBuffers[1].cbBuffer); + + conn->decryptBufferUsed = inBuffers[1].cbBuffer; + + DEBUG_printf(("_sspiConnect: %d bytes of app data was bundled with handshake data", + conn->decryptBufferUsed)); + } + else + conn->decryptBufferUsed = 0; + + /* + * Bail out to quit + */ + break; + } + + /* + * Check for fatal error. + */ + if (FAILED(scRet)) + { + DEBUG_printf(("_sspiConnect: InitializeSecurityContext(2) failed: %x", scRet)); + ok = FALSE; + break; + } + + /* + * If InitializeSecurityContext returned SEC_I_INCOMPLETE_CREDENTIALS, + * then the server just requested client authentication. + */ + if (scRet == SEC_I_INCOMPLETE_CREDENTIALS) + { + /* + * Unimplemented + */ + DEBUG_printf(("_sspiConnect: server requested client credentials")); + ok = FALSE; + break; + } + + /* + * Copy any leftover data from the "extra" buffer, and go around + * again. + */ + if (inBuffers[1].BufferType == SECBUFFER_EXTRA) + { + memmove(conn->decryptBuffer, + conn->decryptBuffer + (conn->decryptBufferUsed - inBuffers[1].cbBuffer), + inBuffers[1].cbBuffer); + + conn->decryptBufferUsed = inBuffers[1].cbBuffer; + } + else + { + conn->decryptBufferUsed = 0; + } + } + + if (ok) + { + conn->contextInitialized = TRUE; + + /* + * Get the server cert + */ + scRet = QueryContextAttributes(&conn->context, SECPKG_ATTR_REMOTE_CERT_CONTEXT, (VOID*) &serverCert ); + + if (scRet != SEC_E_OK) + { + DEBUG_printf(("_sspiConnect: QueryContextAttributes failed(SECPKG_ATTR_REMOTE_CERT_CONTEXT): %x", scRet)); + ok = FALSE; + goto cleanup; + } + + scRet = sspi_verify_certificate(serverCert, hostname, conn->certFlags); + + if (scRet != SEC_E_OK) + { + DEBUG_printf(("_sspiConnect: sspi_verify_certificate failed: %x", scRet)); + ok = FALSE; + goto cleanup; + } + + /* + * Find out how big the header/trailer will be: + */ + scRet = QueryContextAttributes(&conn->context, SECPKG_ATTR_STREAM_SIZES, &conn->streamSizes); + + if (scRet != SEC_E_OK) + { + DEBUG_printf(("_sspiConnect: QueryContextAttributes failed(SECPKG_ATTR_STREAM_SIZES): %x", scRet)); + ok = FALSE; + } + } + +cleanup: + + if (serverCert) + CertFreeCertificateContext(serverCert); + + return (ok); +} + + +/* + * '_sspiAccept()' - Accept an SSL/TLS connection + */ +BOOL /* O - 1 on success, 0 on failure */ +_sspiAccept(_sspi_struct_t *conn) /* I - Client connection */ +{ + DWORD dwSSPIFlags; /* SSL connection attributes we want */ + DWORD dwSSPIOutFlags; /* SSL connection attributes we got */ + TimeStamp tsExpiry; /* Time stamp */ + SECURITY_STATUS scRet; /* SSPI Status */ + SecBufferDesc inBuffer; /* Array of SecBuffer structs */ + SecBuffer inBuffers[2]; /* Security package buffer */ + SecBufferDesc outBuffer; /* Array of SecBuffer structs */ + SecBuffer outBuffers[1]; /* Security package buffer */ + DWORD num = 0; /* 32 bit status value */ + BOOL fInitContext = TRUE; + /* Has the context been init'd? */ + BOOL ok = TRUE; /* Return value */ + + if (!conn) + return (FALSE); + + dwSSPIFlags = ASC_REQ_SEQUENCE_DETECT | + ASC_REQ_REPLAY_DETECT | + ASC_REQ_CONFIDENTIALITY | + ASC_REQ_EXTENDED_ERROR | + ASC_REQ_ALLOCATE_MEMORY | + ASC_REQ_STREAM; + + conn->decryptBufferUsed = 0; + + /* + * Set OutBuffer for AcceptSecurityContext call + */ + outBuffer.cBuffers = 1; + outBuffer.pBuffers = outBuffers; + outBuffer.ulVersion = SECBUFFER_VERSION; + + scRet = SEC_I_CONTINUE_NEEDED; + + while (scRet == SEC_I_CONTINUE_NEEDED || + scRet == SEC_E_INCOMPLETE_MESSAGE || + scRet == SEC_I_INCOMPLETE_CREDENTIALS) + { + if ((conn->decryptBufferUsed == 0) || (scRet == SEC_E_INCOMPLETE_MESSAGE)) + { + if (conn->decryptBufferLength <= conn->decryptBufferUsed) + { + conn->decryptBufferLength += 4096; + conn->decryptBuffer = (BYTE*) realloc(conn->decryptBuffer, + conn->decryptBufferLength); + + if (!conn->decryptBuffer) + { + DEBUG_printf(("_sspiAccept: unable to allocate %d byte decrypt buffer", + conn->decryptBufferLength)); + ok = FALSE; + goto cleanup; + } + } + + for (;;) + { + num = recv(conn->sock, + conn->decryptBuffer + conn->decryptBufferUsed, + (int)(conn->decryptBufferLength - conn->decryptBufferUsed), + 0); + + if ((num == SOCKET_ERROR) && (WSAGetLastError() == WSAEWOULDBLOCK)) + Sleep(1); + else + break; + } + + if (num == SOCKET_ERROR) + { + DEBUG_printf(("_sspiAccept: recv failed: %d", WSAGetLastError())); + ok = FALSE; + goto cleanup; + } + else if (num == 0) + { + DEBUG_printf(("_sspiAccept: client disconnected")); + ok = FALSE; + goto cleanup; + } + + DEBUG_printf(("_sspiAccept: received %d (handshake) bytes from client", + num)); + conn->decryptBufferUsed += num; + } + + /* + * InBuffers[1] is for getting extra data that + * SSPI/SCHANNEL doesn't proccess on this + * run around the loop. + */ + inBuffers[0].pvBuffer = conn->decryptBuffer; + inBuffers[0].cbBuffer = (unsigned long) conn->decryptBufferUsed; + inBuffers[0].BufferType = SECBUFFER_TOKEN; + + inBuffers[1].pvBuffer = NULL; + inBuffers[1].cbBuffer = 0; + inBuffers[1].BufferType = SECBUFFER_EMPTY; + + inBuffer.cBuffers = 2; + inBuffer.pBuffers = inBuffers; + inBuffer.ulVersion = SECBUFFER_VERSION; + + /* + * Initialize these so if we fail, pvBuffer contains NULL, + * so we don't try to free random garbage at the quit + */ + outBuffers[0].pvBuffer = NULL; + outBuffers[0].BufferType = SECBUFFER_TOKEN; + outBuffers[0].cbBuffer = 0; + + scRet = AcceptSecurityContext(&conn->creds, (fInitContext?NULL:&conn->context), + &inBuffer, dwSSPIFlags, SECURITY_NATIVE_DREP, + (fInitContext?&conn->context:NULL), &outBuffer, + &dwSSPIOutFlags, &tsExpiry); + + fInitContext = FALSE; + + if (scRet == SEC_E_OK || + scRet == SEC_I_CONTINUE_NEEDED || + (FAILED(scRet) && ((dwSSPIOutFlags & ISC_RET_EXTENDED_ERROR) != 0))) + { + if (outBuffers[0].cbBuffer && outBuffers[0].pvBuffer) + { + /* + * Send response to server if there is one + */ + num = send(conn->sock, outBuffers[0].pvBuffer, outBuffers[0].cbBuffer, 0); + + if ((num == SOCKET_ERROR) || (num == 0)) + { + DEBUG_printf(("_sspiAccept: handshake send failed: %d", WSAGetLastError())); + ok = FALSE; + goto cleanup; + } + + DEBUG_printf(("_sspiAccept: send %d handshake bytes to client", + outBuffers[0].cbBuffer)); + + FreeContextBuffer(outBuffers[0].pvBuffer); + outBuffers[0].pvBuffer = NULL; + } + } + + if (scRet == SEC_E_OK) + { + /* + * If there's extra data then save it for + * next time we go to decrypt + */ + if (inBuffers[1].BufferType == SECBUFFER_EXTRA) + { + memcpy(conn->decryptBuffer, + (LPBYTE) (conn->decryptBuffer + (conn->decryptBufferUsed - inBuffers[1].cbBuffer)), + inBuffers[1].cbBuffer); + conn->decryptBufferUsed = inBuffers[1].cbBuffer; + } + else + { + conn->decryptBufferUsed = 0; + } + + ok = TRUE; + break; + } + else if (FAILED(scRet) && (scRet != SEC_E_INCOMPLETE_MESSAGE)) + { + DEBUG_printf(("_sspiAccept: AcceptSecurityContext failed: %x", scRet)); + ok = FALSE; + break; + } + + if (scRet != SEC_E_INCOMPLETE_MESSAGE && + scRet != SEC_I_INCOMPLETE_CREDENTIALS) + { + if (inBuffers[1].BufferType == SECBUFFER_EXTRA) + { + memcpy(conn->decryptBuffer, + (LPBYTE) (conn->decryptBuffer + (conn->decryptBufferUsed - inBuffers[1].cbBuffer)), + inBuffers[1].cbBuffer); + conn->decryptBufferUsed = inBuffers[1].cbBuffer; + } + else + { + conn->decryptBufferUsed = 0; + } + } + } + + if (ok) + { + conn->contextInitialized = TRUE; + + /* + * Find out how big the header will be: + */ + scRet = QueryContextAttributes(&conn->context, SECPKG_ATTR_STREAM_SIZES, &conn->streamSizes); + + if (scRet != SEC_E_OK) + { + DEBUG_printf(("_sspiAccept: QueryContextAttributes failed: %x", scRet)); + ok = FALSE; + } + } + +cleanup: + + return (ok); +} + + +/* + * '_sspiSetAllowsAnyRoot()' - Set the client cert policy for untrusted root certs + */ +void +_sspiSetAllowsAnyRoot(_sspi_struct_t *conn, + /* I - Client connection */ + BOOL allow) + /* I - Allow any root */ +{ + conn->certFlags = (allow) ? conn->certFlags | SECURITY_FLAG_IGNORE_UNKNOWN_CA : + conn->certFlags & ~SECURITY_FLAG_IGNORE_UNKNOWN_CA; +} + + +/* + * '_sspiSetAllowsExpiredCerts()' - Set the client cert policy for expired root certs + */ +void +_sspiSetAllowsExpiredCerts(_sspi_struct_t *conn, + /* I - Client connection */ + BOOL allow) + /* I - Allow expired certs */ +{ + conn->certFlags = (allow) ? conn->certFlags | SECURITY_FLAG_IGNORE_CERT_DATE_INVALID : + conn->certFlags & ~SECURITY_FLAG_IGNORE_CERT_DATE_INVALID; +} + + +/* + * '_sspiWrite()' - Write a buffer to an ssl socket + */ +int /* O - Bytes written or SOCKET_ERROR */ +_sspiWrite(_sspi_struct_t *conn, /* I - Client connection */ + void *buf, /* I - Buffer */ + size_t len) /* I - Buffer length */ +{ + SecBufferDesc message; /* Array of SecBuffer struct */ + SecBuffer buffers[4] = { 0 }; /* Security package buffer */ + BYTE *buffer = NULL; /* Scratch buffer */ + int bufferLen; /* Buffer length */ + size_t bytesLeft; /* Bytes left to write */ + int index = 0; /* Index into buffer */ + int num = 0; /* Return value */ + + if (!conn || !buf || !len) + { + WSASetLastError(WSAEINVAL); + num = SOCKET_ERROR; + goto cleanup; + } + + bufferLen = conn->streamSizes.cbMaximumMessage + + conn->streamSizes.cbHeader + + conn->streamSizes.cbTrailer; + + buffer = (BYTE*) malloc(bufferLen); + + if (!buffer) + { + DEBUG_printf(("_sspiWrite: buffer alloc of %d bytes failed", bufferLen)); + WSASetLastError(E_OUTOFMEMORY); + num = SOCKET_ERROR; + goto cleanup; + } + + bytesLeft = len; + + while (bytesLeft) + { + size_t chunk = min(conn->streamSizes.cbMaximumMessage, /* Size of data to write */ + bytesLeft); + SECURITY_STATUS scRet; /* SSPI status */ + + /* + * Copy user data into the buffer, starting + * just past the header + */ + memcpy(buffer + conn->streamSizes.cbHeader, + ((BYTE*) buf) + index, + chunk); + + /* + * Setup the SSPI buffers + */ + message.ulVersion = SECBUFFER_VERSION; + message.cBuffers = 4; + message.pBuffers = buffers; + buffers[0].pvBuffer = buffer; + buffers[0].cbBuffer = conn->streamSizes.cbHeader; + buffers[0].BufferType = SECBUFFER_STREAM_HEADER; + buffers[1].pvBuffer = buffer + conn->streamSizes.cbHeader; + buffers[1].cbBuffer = (unsigned long) chunk; + buffers[1].BufferType = SECBUFFER_DATA; + buffers[2].pvBuffer = buffer + conn->streamSizes.cbHeader + chunk; + buffers[2].cbBuffer = conn->streamSizes.cbTrailer; + buffers[2].BufferType = SECBUFFER_STREAM_TRAILER; + buffers[3].BufferType = SECBUFFER_EMPTY; + + /* + * Encrypt the data + */ + scRet = EncryptMessage(&conn->context, 0, &message, 0); + + if (FAILED(scRet)) + { + DEBUG_printf(("_sspiWrite: EncryptMessage failed: %x", scRet)); + WSASetLastError(WSASYSCALLFAILURE); + num = SOCKET_ERROR; + goto cleanup; + } + + /* + * Send the data. Remember the size of + * the total data to send is the size + * of the header, the size of the data + * the caller passed in and the size + * of the trailer + */ + num = send(conn->sock, + buffer, + buffers[0].cbBuffer + buffers[1].cbBuffer + buffers[2].cbBuffer, + 0); + + if ((num == SOCKET_ERROR) || (num == 0)) + { + DEBUG_printf(("_sspiWrite: send failed: %ld", WSAGetLastError())); + goto cleanup; + } + + bytesLeft -= (int) chunk; + index += (int) chunk; + } + + num = (int) len; + +cleanup: + + if (buffer) + free(buffer); + + return (num); +} + + +/* + * '_sspiRead()' - Read a buffer from an ssl socket + */ +int /* O - Bytes read or SOCKET_ERROR */ +_sspiRead(_sspi_struct_t *conn, /* I - Client connection */ + void *buf, /* I - Buffer */ + size_t len) /* I - Buffer length */ +{ + SecBufferDesc message; /* Array of SecBuffer struct */ + SecBuffer buffers[4] = { 0 }; /* Security package buffer */ + int num = 0; /* Return value */ + + if (!conn) + { + WSASetLastError(WSAEINVAL); + num = SOCKET_ERROR; + goto cleanup; + } + + /* + * If there are bytes that have already been + * decrypted and have not yet been read, return + * those + */ + if (buf && (conn->readBufferUsed > 0)) + { + int bytesToCopy = (int) min(conn->readBufferUsed, len); /* Amount of bytes to copy */ + /* from read buffer */ + + memcpy(buf, conn->readBuffer, bytesToCopy); + conn->readBufferUsed -= bytesToCopy; + + if (conn->readBufferUsed > 0) + /* + * If the caller didn't request all the bytes + * we have in the buffer, then move the unread + * bytes down + */ + memmove(conn->readBuffer, + conn->readBuffer + bytesToCopy, + conn->readBufferUsed); + + num = bytesToCopy; + } + else + { + PSecBuffer pDataBuffer; /* Data buffer */ + PSecBuffer pExtraBuffer; /* Excess data buffer */ + SECURITY_STATUS scRet; /* SSPI status */ + int i; /* Loop control variable */ + + /* + * Initialize security buffer structs + */ + message.ulVersion = SECBUFFER_VERSION; + message.cBuffers = 4; + message.pBuffers = buffers; + + do + { + /* + * If there is not enough space in the + * buffer, then increase it's size + */ + if (conn->decryptBufferLength <= conn->decryptBufferUsed) + { + conn->decryptBufferLength += 4096; + conn->decryptBuffer = (BYTE*) realloc(conn->decryptBuffer, + conn->decryptBufferLength); + + if (!conn->decryptBuffer) + { + DEBUG_printf(("_sspiRead: unable to allocate %d byte buffer", + conn->decryptBufferLength)); + WSASetLastError(E_OUTOFMEMORY); + num = SOCKET_ERROR; + goto cleanup; + } + } + + buffers[0].pvBuffer = conn->decryptBuffer; + buffers[0].cbBuffer = (unsigned long) conn->decryptBufferUsed; + buffers[0].BufferType = SECBUFFER_DATA; + buffers[1].BufferType = SECBUFFER_EMPTY; + buffers[2].BufferType = SECBUFFER_EMPTY; + buffers[3].BufferType = SECBUFFER_EMPTY; + + scRet = DecryptMessage(&conn->context, &message, 0, NULL); + + if (scRet == SEC_E_INCOMPLETE_MESSAGE) + { + if (buf) + { + num = recv(conn->sock, + conn->decryptBuffer + conn->decryptBufferUsed, + (int)(conn->decryptBufferLength - conn->decryptBufferUsed), + 0); + if (num == SOCKET_ERROR) + { + DEBUG_printf(("_sspiRead: recv failed: %d", WSAGetLastError())); + goto cleanup; + } + else if (num == 0) + { + DEBUG_printf(("_sspiRead: server disconnected")); + goto cleanup; + } + + conn->decryptBufferUsed += num; + } + else + { + num = (int) conn->readBufferUsed; + goto cleanup; + } + } + } + while (scRet == SEC_E_INCOMPLETE_MESSAGE); + + if (scRet == SEC_I_CONTEXT_EXPIRED) + { + DEBUG_printf(("_sspiRead: context expired")); + WSASetLastError(WSAECONNRESET); + num = SOCKET_ERROR; + goto cleanup; + } + else if (scRet != SEC_E_OK) + { + DEBUG_printf(("_sspiRead: DecryptMessage failed: %lx", scRet)); + WSASetLastError(WSASYSCALLFAILURE); + num = SOCKET_ERROR; + goto cleanup; + } + + /* + * The decryption worked. Now, locate data buffer. + */ + pDataBuffer = NULL; + pExtraBuffer = NULL; + for (i = 1; i < 4; i++) + { + if (buffers[i].BufferType == SECBUFFER_DATA) + pDataBuffer = &buffers[i]; + else if (!pExtraBuffer && (buffers[i].BufferType == SECBUFFER_EXTRA)) + pExtraBuffer = &buffers[i]; + } + + /* + * If a data buffer is found, then copy + * the decrypted bytes to the passed-in + * buffer + */ + if (pDataBuffer) + { + int bytesToCopy = min(pDataBuffer->cbBuffer, (int) len); + /* Number of bytes to copy into buf */ + int bytesToSave = pDataBuffer->cbBuffer - bytesToCopy; + /* Number of bytes to save in our read buffer */ + + if (bytesToCopy) + memcpy(buf, pDataBuffer->pvBuffer, bytesToCopy); + + /* + * If there are more decrypted bytes than can be + * copied to the passed in buffer, then save them + */ + if (bytesToSave) + { + if ((int)(conn->readBufferLength - conn->readBufferUsed) < bytesToSave) + { + conn->readBufferLength = conn->readBufferUsed + bytesToSave; + conn->readBuffer = realloc(conn->readBuffer, + conn->readBufferLength); + + if (!conn->readBuffer) + { + DEBUG_printf(("_sspiRead: unable to allocate %d bytes", conn->readBufferLength)); + WSASetLastError(E_OUTOFMEMORY); + num = SOCKET_ERROR; + goto cleanup; + } + } + + memcpy(((BYTE*) conn->readBuffer) + conn->readBufferUsed, + ((BYTE*) pDataBuffer->pvBuffer) + bytesToCopy, + bytesToSave); + + conn->readBufferUsed += bytesToSave; + } + + num = (buf) ? bytesToCopy : (int) conn->readBufferUsed; + } + else + { + DEBUG_printf(("_sspiRead: unable to find data buffer")); + WSASetLastError(WSASYSCALLFAILURE); + num = SOCKET_ERROR; + goto cleanup; + } + + /* + * If the decryption process left extra bytes, + * then save those back in decryptBuffer. They will + * be processed the next time through the loop. + */ + if (pExtraBuffer) + { + memmove(conn->decryptBuffer, pExtraBuffer->pvBuffer, pExtraBuffer->cbBuffer); + conn->decryptBufferUsed = pExtraBuffer->cbBuffer; + } + else + { + conn->decryptBufferUsed = 0; + } + } + +cleanup: + + return (num); +} + + +/* + * '_sspiPending()' - Returns the number of available bytes + */ +int /* O - Number of available bytes */ +_sspiPending(_sspi_struct_t *conn) /* I - Client connection */ +{ + return (_sspiRead(conn, NULL, 0)); +} + + +/* + * '_sspiFree()' - Close a connection and free resources + */ +void +_sspiFree(_sspi_struct_t *conn) /* I - Client connection */ +{ + if (!conn) + return; + + if (conn->contextInitialized) + { + SecBufferDesc message; /* Array of SecBuffer struct */ + SecBuffer buffers[1] = { 0 }; + /* Security package buffer */ + DWORD dwType; /* Type */ + DWORD status; /* Status */ + + /* + * Notify schannel that we are about to close the connection. + */ + dwType = SCHANNEL_SHUTDOWN; + + buffers[0].pvBuffer = &dwType; + buffers[0].BufferType = SECBUFFER_TOKEN; + buffers[0].cbBuffer = sizeof(dwType); + + message.cBuffers = 1; + message.pBuffers = buffers; + message.ulVersion = SECBUFFER_VERSION; + + status = ApplyControlToken(&conn->context, &message); + + if (SUCCEEDED(status)) + { + PBYTE pbMessage; /* Message buffer */ + DWORD cbMessage; /* Message buffer count */ + DWORD cbData; /* Data count */ + DWORD dwSSPIFlags; /* SSL attributes we requested */ + DWORD dwSSPIOutFlags; /* SSL attributes we received */ + TimeStamp tsExpiry; /* Time stamp */ + + dwSSPIFlags = ASC_REQ_SEQUENCE_DETECT | + ASC_REQ_REPLAY_DETECT | + ASC_REQ_CONFIDENTIALITY | + ASC_REQ_EXTENDED_ERROR | + ASC_REQ_ALLOCATE_MEMORY | + ASC_REQ_STREAM; + + buffers[0].pvBuffer = NULL; + buffers[0].BufferType = SECBUFFER_TOKEN; + buffers[0].cbBuffer = 0; + + message.cBuffers = 1; + message.pBuffers = buffers; + message.ulVersion = SECBUFFER_VERSION; + + status = AcceptSecurityContext(&conn->creds, &conn->context, NULL, + dwSSPIFlags, SECURITY_NATIVE_DREP, NULL, + &message, &dwSSPIOutFlags, &tsExpiry); + + if (SUCCEEDED(status)) + { + pbMessage = buffers[0].pvBuffer; + cbMessage = buffers[0].cbBuffer; + + /* + * Send the close notify message to the client. + */ + if (pbMessage && cbMessage) + { + cbData = send(conn->sock, pbMessage, cbMessage, 0); + if ((cbData == SOCKET_ERROR) || (cbData == 0)) + { + status = WSAGetLastError(); + DEBUG_printf(("_sspiFree: sending close notify failed: %d", status)); + } + else + { + FreeContextBuffer(pbMessage); + } + } + } + else + { + DEBUG_printf(("_sspiFree: AcceptSecurityContext failed: %x", status)); + } + } + else + { + DEBUG_printf(("_sspiFree: ApplyControlToken failed: %x", status)); + } + + DeleteSecurityContext(&conn->context); + conn->contextInitialized = FALSE; + } + + if (conn->decryptBuffer) + { + free(conn->decryptBuffer); + conn->decryptBuffer = NULL; + } + + if (conn->readBuffer) + { + free(conn->readBuffer); + conn->readBuffer = NULL; + } + + if (conn->sock != INVALID_SOCKET) + { + closesocket(conn->sock); + conn->sock = INVALID_SOCKET; + } + + free(conn); +} + + +/* + * 'sspi_verify_certificate()' - Verify a server certificate + */ +static DWORD /* 0 - Error code (0 == No error) */ +sspi_verify_certificate(PCCERT_CONTEXT serverCert, + /* I - Server certificate */ + const CHAR *serverName, + /* I - Server name */ + DWORD dwCertFlags) + /* I - Verification flags */ +{ + HTTPSPolicyCallbackData httpsPolicy; + /* HTTPS Policy Struct */ + CERT_CHAIN_POLICY_PARA policyPara; + /* Cert chain policy parameters */ + CERT_CHAIN_POLICY_STATUS policyStatus; + /* Cert chain policy status */ + CERT_CHAIN_PARA chainPara; + /* Used for searching and matching criteria */ + PCCERT_CHAIN_CONTEXT chainContext = NULL; + /* Certificate chain */ + PWSTR serverNameUnicode = NULL; + /* Unicode server name */ + LPSTR rgszUsages[] = { szOID_PKIX_KP_SERVER_AUTH, + szOID_SERVER_GATED_CRYPTO, + szOID_SGC_NETSCAPE }; + /* How are we using this certificate? */ + DWORD cUsages = sizeof(rgszUsages) / sizeof(LPSTR); + /* Number of ites in rgszUsages */ + DWORD count; /* 32 bit count variable */ + DWORD status; /* Return value */ + + if (!serverCert) + { + status = SEC_E_WRONG_PRINCIPAL; + goto cleanup; + } + + /* + * Convert server name to unicode. + */ + if (!serverName || (strlen(serverName) == 0)) + { + status = SEC_E_WRONG_PRINCIPAL; + goto cleanup; + } + + count = MultiByteToWideChar(CP_ACP, 0, serverName, -1, NULL, 0); + serverNameUnicode = LocalAlloc(LMEM_FIXED, count * sizeof(WCHAR)); + if (!serverNameUnicode) + { + status = SEC_E_INSUFFICIENT_MEMORY; + goto cleanup; + } + count = MultiByteToWideChar(CP_ACP, 0, serverName, -1, serverNameUnicode, count); + if (count == 0) + { + status = SEC_E_WRONG_PRINCIPAL; + goto cleanup; + } + + /* + * Build certificate chain. + */ + ZeroMemory(&chainPara, sizeof(chainPara)); + chainPara.cbSize = sizeof(chainPara); + chainPara.RequestedUsage.dwType = USAGE_MATCH_TYPE_OR; + chainPara.RequestedUsage.Usage.cUsageIdentifier = cUsages; + chainPara.RequestedUsage.Usage.rgpszUsageIdentifier = rgszUsages; + + if (!CertGetCertificateChain(NULL, serverCert, NULL, serverCert->hCertStore, + &chainPara, 0, NULL, &chainContext)) + { + status = GetLastError(); + DEBUG_printf(("CertGetCertificateChain returned 0x%x\n", status)); + goto cleanup; + } + + /* + * Validate certificate chain. + */ + ZeroMemory(&httpsPolicy, sizeof(HTTPSPolicyCallbackData)); + httpsPolicy.cbStruct = sizeof(HTTPSPolicyCallbackData); + httpsPolicy.dwAuthType = AUTHTYPE_SERVER; + httpsPolicy.fdwChecks = dwCertFlags; + httpsPolicy.pwszServerName = serverNameUnicode; + + memset(&policyPara, 0, sizeof(policyPara)); + policyPara.cbSize = sizeof(policyPara); + policyPara.pvExtraPolicyPara = &httpsPolicy; + + memset(&policyStatus, 0, sizeof(policyStatus)); + policyStatus.cbSize = sizeof(policyStatus); + + if (!CertVerifyCertificateChainPolicy(CERT_CHAIN_POLICY_SSL, chainContext, + &policyPara, &policyStatus)) + { + status = GetLastError(); + DEBUG_printf(("CertVerifyCertificateChainPolicy returned %d", status)); + goto cleanup; + } + + if (policyStatus.dwError) + { + status = policyStatus.dwError; + goto cleanup; + } + + status = SEC_E_OK; + +cleanup: + + if (chainContext) + CertFreeCertificateChain(chainContext); + + if (serverNameUnicode) + LocalFree(serverNameUnicode); + + return (status); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/string-private.h b/cups/string-private.h new file mode 100644 index 0000000000..bef1aea803 --- /dev/null +++ b/cups/string-private.h @@ -0,0 +1,200 @@ +/* + * "$Id$" + * + * Private string definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_STRING_PRIVATE_H_ +# define _CUPS_STRING_PRIVATE_H_ + +/* + * Include necessary headers... + */ + +# include +# include +# include +# include +# include +# include + +# include "config.h" + +# ifdef HAVE_STRING_H +# include +# endif /* HAVE_STRING_H */ + +# ifdef HAVE_STRINGS_H +# include +# endif /* HAVE_STRINGS_H */ + +# ifdef HAVE_BSTRING_H +# include +# endif /* HAVE_BSTRING_H */ + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * String pool structures... + */ + +# define _CUPS_STR_GUARD 0x12344321 + +typedef struct _cups_sp_item_s /**** String Pool Item ****/ +{ +# ifdef DEBUG_GUARDS + unsigned int guard; /* Guard word */ +# endif /* DEBUG_GUARDS */ + unsigned int ref_count; /* Reference count */ + char str[1]; /* String */ +} _cups_sp_item_t; + + +/* + * Replacements for the ctype macros that are not affected by locale, since we + * really only care about testing for ASCII characters when parsing files, etc. + * + * The _CUPS_INLINE definition controls whether we get an inline function body, + * and external function body, or an external definition. + */ + +# if defined(__GNUC__) || __STDC_VERSION__ >= 199901L +# define _CUPS_INLINE static inline +# elif defined(_MSC_VER) +# define _CUPS_INLINE static __inline +# elif defined(_CUPS_STRING_C_) +# define _CUPS_INLINE +# endif /* __GNUC__ || __STDC_VERSION__ */ + +# ifdef _CUPS_INLINE +_CUPS_INLINE int /* O - 1 on match, 0 otherwise */ +_cups_isalnum(int ch) /* I - Character to test */ +{ + return ((ch >= '0' && ch <= '9') || + (ch >= 'A' && ch <= 'Z') || + (ch >= 'a' && ch <= 'z')); +} + +_CUPS_INLINE int /* O - 1 on match, 0 otherwise */ +_cups_isalpha(int ch) /* I - Character to test */ +{ + return ((ch >= 'A' && ch <= 'Z') || + (ch >= 'a' && ch <= 'z')); +} + +_CUPS_INLINE int /* O - 1 on match, 0 otherwise */ +_cups_isspace(int ch) /* I - Character to test */ +{ + return (ch == ' ' || ch == '\f' || ch == '\n' || ch == '\r' || ch == '\t' || + ch == '\v'); +} + +_CUPS_INLINE int /* O - 1 on match, 0 otherwise */ +_cups_isupper(int ch) /* I - Character to test */ +{ + return (ch >= 'A' && ch <= 'Z'); +} + +_CUPS_INLINE int /* O - Converted character */ +_cups_tolower(int ch) /* I - Character to convert */ +{ + return (_cups_isupper(ch) ? ch - 'A' + 'a' : ch); +} +# else +extern int _cups_isalnum(int ch); +extern int _cups_isalpha(int ch); +extern int _cups_isspace(int ch); +extern int _cups_isupper(int ch); +extern int _cups_tolower(int ch); +# endif /* _CUPS_INLINE */ + + +/* + * Prototypes... + */ + +extern void _cups_strcpy(char *dst, const char *src); + +# ifndef HAVE_STRDUP +extern char *_cups_strdup(const char *); +# define strdup _cups_strdup +# endif /* !HAVE_STRDUP */ + +extern int _cups_strcasecmp(const char *, const char *); + +extern int _cups_strncasecmp(const char *, const char *, size_t n); + +# ifndef HAVE_STRLCAT +extern size_t _cups_strlcat(char *, const char *, size_t); +# define strlcat _cups_strlcat +# endif /* !HAVE_STRLCAT */ + +# ifndef HAVE_STRLCPY +extern size_t _cups_strlcpy(char *, const char *, size_t); +# define strlcpy _cups_strlcpy +# endif /* !HAVE_STRLCPY */ + +# ifndef HAVE_SNPRINTF +extern int _cups_snprintf(char *, size_t, const char *, ...) + __attribute__ ((__format__ (__printf__, 3, 4))); +# define snprintf _cups_snprintf +# endif /* !HAVE_SNPRINTF */ + +# ifndef HAVE_VSNPRINTF +extern int _cups_vsnprintf(char *, size_t, const char *, va_list); +# define vsnprintf _cups_vsnprintf +# endif /* !HAVE_VSNPRINTF */ + +/* + * String pool functions... + */ + +extern char *_cupsStrAlloc(const char *s); +extern void _cupsStrFlush(void); +extern void _cupsStrFree(const char *s); +extern char *_cupsStrRetain(const char *s); +extern size_t _cupsStrStatistics(size_t *alloc_bytes, size_t *total_bytes); + + +/* + * Floating point number functions... + */ + +extern char *_cupsStrFormatd(char *buf, char *bufend, double number, + struct lconv *loc); +extern double _cupsStrScand(const char *buf, char **bufptr, + struct lconv *loc); + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +} +# endif /* __cplusplus */ + +#endif /* !_CUPS_STRING_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/string.c b/cups/string.c new file mode 100644 index 0000000000..60e346453f --- /dev/null +++ b/cups/string.c @@ -0,0 +1,759 @@ +/* + * "$Id$" + * + * String functions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * _cupsStrAlloc() - Allocate/reference a string. + * _cupsStrFlush() - Flush the string pool. + * _cupsStrFormatd() - Format a floating-point number. + * _cupsStrFree() - Free/dereference a string. + * _cupsStrRetain() - Increment the reference count of a string. + * _cupsStrScand() - Scan a string for a floating-point number. + * _cupsStrStatistics() - Return allocation statistics for string pool. + * _cups_strcpy() - Copy a string allowing for overlapping strings. + * _cups_strdup() - Duplicate a string. + * _cups_strcasecmp() - Do a case-insensitive comparison. + * _cups_strncasecmp() - Do a case-insensitive comparison on up to N chars. + * _cups_strlcat() - Safely concatenate two strings. + * _cups_strlcpy() - Safely copy two strings. + * compare_sp_items() - Compare two string pool items... + */ + +/* + * Include necessary headers... + */ + +#define _CUPS_STRING_C_ +#include "string-private.h" +#include "debug-private.h" +#include "thread-private.h" +#include "array.h" +#include +#include + + +/* + * Local globals... + */ + +static _cups_mutex_t sp_mutex = _CUPS_MUTEX_INITIALIZER; + /* Mutex to control access to pool */ +static cups_array_t *stringpool = NULL; + /* Global string pool */ + + +/* + * Local functions... + */ + +static int compare_sp_items(_cups_sp_item_t *a, _cups_sp_item_t *b); + + +/* + * '_cupsStrAlloc()' - Allocate/reference a string. + */ + +char * /* O - String pointer */ +_cupsStrAlloc(const char *s) /* I - String */ +{ + _cups_sp_item_t *item, /* String pool item */ + *key; /* Search key */ + + + /* + * Range check input... + */ + + if (!s) + return (NULL); + + /* + * Get the string pool... + */ + + _cupsMutexLock(&sp_mutex); + + if (!stringpool) + stringpool = cupsArrayNew((cups_array_func_t)compare_sp_items, NULL); + + if (!stringpool) + { + _cupsMutexUnlock(&sp_mutex); + + return (NULL); + } + + /* + * See if the string is already in the pool... + */ + + key = (_cups_sp_item_t *)(s - offsetof(_cups_sp_item_t, str)); + + if ((item = (_cups_sp_item_t *)cupsArrayFind(stringpool, key)) != NULL) + { + /* + * Found it, return the cached string... + */ + + item->ref_count ++; + +#ifdef DEBUG_GUARDS + DEBUG_printf(("5_cupsStrAlloc: Using string %p(%s) for \"%s\", guard=%08x, " + "ref_count=%d", item, item->str, s, item->guard, + item->ref_count)); + + if (item->guard != _CUPS_STR_GUARD) + abort(); +#endif /* DEBUG_GUARDS */ + + _cupsMutexUnlock(&sp_mutex); + + return (item->str); + } + + /* + * Not found, so allocate a new one... + */ + + item = (_cups_sp_item_t *)calloc(1, sizeof(_cups_sp_item_t) + strlen(s)); + if (!item) + { + _cupsMutexUnlock(&sp_mutex); + + return (NULL); + } + + item->ref_count = 1; + strcpy(item->str, s); + +#ifdef DEBUG_GUARDS + item->guard = _CUPS_STR_GUARD; + + DEBUG_printf(("5_cupsStrAlloc: Created string %p(%s) for \"%s\", guard=%08x, " + "ref_count=%d", item, item->str, s, item->guard, + item->ref_count)); +#endif /* DEBUG_GUARDS */ + + /* + * Add the string to the pool and return it... + */ + + cupsArrayAdd(stringpool, item); + + _cupsMutexUnlock(&sp_mutex); + + return (item->str); +} + + +/* + * '_cupsStrFlush()' - Flush the string pool. + */ + +void +_cupsStrFlush(void) +{ + _cups_sp_item_t *item; /* Current item */ + + + DEBUG_printf(("4_cupsStrFlush: %d strings in array", + cupsArrayCount(stringpool))); + + _cupsMutexLock(&sp_mutex); + + for (item = (_cups_sp_item_t *)cupsArrayFirst(stringpool); + item; + item = (_cups_sp_item_t *)cupsArrayNext(stringpool)) + free(item); + + cupsArrayDelete(stringpool); + stringpool = NULL; + + _cupsMutexUnlock(&sp_mutex); +} + + +/* + * '_cupsStrFormatd()' - Format a floating-point number. + */ + +char * /* O - Pointer to end of string */ +_cupsStrFormatd(char *buf, /* I - String */ + char *bufend, /* I - End of string buffer */ + double number, /* I - Number to format */ + struct lconv *loc) /* I - Locale data */ +{ + char *bufptr, /* Pointer into buffer */ + temp[1024], /* Temporary string */ + *tempdec, /* Pointer to decimal point */ + *tempptr; /* Pointer into temporary string */ + const char *dec; /* Decimal point */ + int declen; /* Length of decimal point */ + + + /* + * Format the number using the "%.12f" format and then eliminate + * unnecessary trailing 0's. + */ + + snprintf(temp, sizeof(temp), "%.12f", number); + for (tempptr = temp + strlen(temp) - 1; + tempptr > temp && *tempptr == '0'; + *tempptr-- = '\0'); + + /* + * Next, find the decimal point... + */ + + if (loc && loc->decimal_point) + { + dec = loc->decimal_point; + declen = (int)strlen(dec); + } + else + { + dec = "."; + declen = 1; + } + + if (declen == 1) + tempdec = strchr(temp, *dec); + else + tempdec = strstr(temp, dec); + + /* + * Copy everything up to the decimal point... + */ + + if (tempdec) + { + for (tempptr = temp, bufptr = buf; + tempptr < tempdec && bufptr < bufend; + *bufptr++ = *tempptr++); + + tempptr += declen; + + if (*tempptr && bufptr < bufend) + { + *bufptr++ = '.'; + + while (*tempptr && bufptr < bufend) + *bufptr++ = *tempptr++; + } + + *bufptr = '\0'; + } + else + { + strlcpy(buf, temp, bufend - buf + 1); + bufptr = buf + strlen(buf); + } + + return (bufptr); +} + + +/* + * '_cupsStrFree()' - Free/dereference a string. + */ + +void +_cupsStrFree(const char *s) /* I - String to free */ +{ + _cups_sp_item_t *item, /* String pool item */ + *key; /* Search key */ + + + /* + * Range check input... + */ + + if (!s) + return; + + /* + * Check the string pool... + * + * We don't need to lock the mutex yet, as we only want to know if + * the stringpool is initialized. The rest of the code will still + * work if it is initialized before we lock... + */ + + if (!stringpool) + return; + + /* + * See if the string is already in the pool... + */ + + _cupsMutexLock(&sp_mutex); + + key = (_cups_sp_item_t *)(s - offsetof(_cups_sp_item_t, str)); + +#ifdef DEBUG_GUARDS + if (key->guard != _CUPS_STR_GUARD) + { + DEBUG_printf(("5_cupsStrFree: Freeing string %p(%s), guard=%08x, " + "ref_count=%d", key, key->str, key->guard, key->ref_count)); + abort(); + } +#endif /* DEBUG_GUARDS */ + + if ((item = (_cups_sp_item_t *)cupsArrayFind(stringpool, key)) != NULL && + item == key) + { + /* + * Found it, dereference... + */ + + item->ref_count --; + + if (!item->ref_count) + { + /* + * Remove and free... + */ + + cupsArrayRemove(stringpool, item); + + free(item); + } + } + + _cupsMutexUnlock(&sp_mutex); +} + + +/* + * '_cupsStrRetain()' - Increment the reference count of a string. + * + * Note: This function does not verify that the passed pointer is in the + * string pool, so any calls to it MUST know they are passing in a + * good pointer. + */ + +char * /* O - Pointer to string */ +_cupsStrRetain(const char *s) /* I - String to retain */ +{ + _cups_sp_item_t *item; /* Pointer to string pool item */ + + + if (s) + { + item = (_cups_sp_item_t *)(s - offsetof(_cups_sp_item_t, str)); + +#ifdef DEBUG_GUARDS + if (item->guard != _CUPS_STR_GUARD) + { + DEBUG_printf(("5_cupsStrRetain: Retaining string %p(%s), guard=%08x, " + "ref_count=%d", item, s, item->guard, item->ref_count)); + abort(); + } +#endif /* DEBUG_GUARDS */ + + _cupsMutexLock(&sp_mutex); + + item->ref_count ++; + + _cupsMutexUnlock(&sp_mutex); + } + + return ((char *)s); +} + + +/* + * '_cupsStrScand()' - Scan a string for a floating-point number. + * + * This function handles the locale-specific BS so that a decimal + * point is always the period (".")... + */ + +double /* O - Number */ +_cupsStrScand(const char *buf, /* I - Pointer to number */ + char **bufptr, /* O - New pointer or NULL on error */ + struct lconv *loc) /* I - Locale data */ +{ + char temp[1024], /* Temporary buffer */ + *tempptr; /* Pointer into temporary buffer */ + + + /* + * Range check input... + */ + + if (!buf) + return (0.0); + + /* + * Skip leading whitespace... + */ + + while (_cups_isspace(*buf)) + buf ++; + + /* + * Copy leading sign, numbers, period, and then numbers... + */ + + tempptr = temp; + if (*buf == '-' || *buf == '+') + *tempptr++ = *buf++; + + while (isdigit(*buf & 255)) + if (tempptr < (temp + sizeof(temp) - 1)) + *tempptr++ = *buf++; + else + { + if (bufptr) + *bufptr = NULL; + + return (0.0); + } + + if (*buf == '.') + { + /* + * Read fractional portion of number... + */ + + buf ++; + + if (loc && loc->decimal_point) + { + strlcpy(tempptr, loc->decimal_point, sizeof(temp) - (tempptr - temp)); + tempptr += strlen(tempptr); + } + else if (tempptr < (temp + sizeof(temp) - 1)) + *tempptr++ = '.'; + else + { + if (bufptr) + *bufptr = NULL; + + return (0.0); + } + + while (isdigit(*buf & 255)) + if (tempptr < (temp + sizeof(temp) - 1)) + *tempptr++ = *buf++; + else + { + if (bufptr) + *bufptr = NULL; + + return (0.0); + } + } + + if (*buf == 'e' || *buf == 'E') + { + /* + * Read exponent... + */ + + if (tempptr < (temp + sizeof(temp) - 1)) + *tempptr++ = *buf++; + else + { + if (bufptr) + *bufptr = NULL; + + return (0.0); + } + + if (*buf == '+' || *buf == '-') + { + if (tempptr < (temp + sizeof(temp) - 1)) + *tempptr++ = *buf++; + else + { + if (bufptr) + *bufptr = NULL; + + return (0.0); + } + } + + while (isdigit(*buf & 255)) + if (tempptr < (temp + sizeof(temp) - 1)) + *tempptr++ = *buf++; + else + { + if (bufptr) + *bufptr = NULL; + + return (0.0); + } + } + + /* + * Nul-terminate the temporary string and return the value... + */ + + if (bufptr) + *bufptr = (char *)buf; + + *tempptr = '\0'; + + return (strtod(temp, NULL)); +} + + +/* + * '_cupsStrStatistics()' - Return allocation statistics for string pool. + */ + +size_t /* O - Number of strings */ +_cupsStrStatistics(size_t *alloc_bytes, /* O - Allocated bytes */ + size_t *total_bytes) /* O - Total string bytes */ +{ + size_t count, /* Number of strings */ + abytes, /* Allocated string bytes */ + tbytes, /* Total string bytes */ + len; /* Length of string */ + _cups_sp_item_t *item; /* Current item */ + + + /* + * Loop through strings in pool, counting everything up... + */ + + _cupsMutexLock(&sp_mutex); + + for (count = 0, abytes = 0, tbytes = 0, + item = (_cups_sp_item_t *)cupsArrayFirst(stringpool); + item; + item = (_cups_sp_item_t *)cupsArrayNext(stringpool)) + { + /* + * Count allocated memory, using a 64-bit aligned buffer as a basis. + */ + + count += item->ref_count; + len = (strlen(item->str) + 8) & ~7; + abytes += sizeof(_cups_sp_item_t) + len; + tbytes += item->ref_count * len; + } + + _cupsMutexUnlock(&sp_mutex); + + /* + * Return values... + */ + + if (alloc_bytes) + *alloc_bytes = abytes; + + if (total_bytes) + *total_bytes = tbytes; + + return (count); +} + + +/* + * '_cups_strcpy()' - Copy a string allowing for overlapping strings. + */ + +void +_cups_strcpy(char *dst, /* I - Destination string */ + const char *src) /* I - Source string */ +{ + while (*src) + *dst++ = *src++; + + *dst = '\0'; +} + + +/* + * '_cups_strdup()' - Duplicate a string. + */ + +#ifndef HAVE_STRDUP +char * /* O - New string pointer */ +_cups_strdup(const char *s) /* I - String to duplicate */ +{ + char *t; /* New string pointer */ + + + if (s == NULL) + return (NULL); + + if ((t = malloc(strlen(s) + 1)) == NULL) + return (NULL); + + return (strcpy(t, s)); +} +#endif /* !HAVE_STRDUP */ + + +/* + * '_cups_strcasecmp()' - Do a case-insensitive comparison. + */ + +int /* O - Result of comparison (-1, 0, or 1) */ +_cups_strcasecmp(const char *s, /* I - First string */ + const char *t) /* I - Second string */ +{ + while (*s != '\0' && *t != '\0') + { + if (_cups_tolower(*s) < _cups_tolower(*t)) + return (-1); + else if (_cups_tolower(*s) > _cups_tolower(*t)) + return (1); + + s ++; + t ++; + } + + if (*s == '\0' && *t == '\0') + return (0); + else if (*s != '\0') + return (1); + else + return (-1); +} + +/* + * '_cups_strncasecmp()' - Do a case-insensitive comparison on up to N chars. + */ + +int /* O - Result of comparison (-1, 0, or 1) */ +_cups_strncasecmp(const char *s, /* I - First string */ + const char *t, /* I - Second string */ + size_t n) /* I - Maximum number of characters to compare */ +{ + while (*s != '\0' && *t != '\0' && n > 0) + { + if (_cups_tolower(*s) < _cups_tolower(*t)) + return (-1); + else if (_cups_tolower(*s) > _cups_tolower(*t)) + return (1); + + s ++; + t ++; + n --; + } + + if (n == 0) + return (0); + else if (*s == '\0' && *t == '\0') + return (0); + else if (*s != '\0') + return (1); + else + return (-1); +} + + +#ifndef HAVE_STRLCAT +/* + * '_cups_strlcat()' - Safely concatenate two strings. + */ + +size_t /* O - Length of string */ +_cups_strlcat(char *dst, /* O - Destination string */ + const char *src, /* I - Source string */ + size_t size) /* I - Size of destination string buffer */ +{ + size_t srclen; /* Length of source string */ + size_t dstlen; /* Length of destination string */ + + + /* + * Figure out how much room is left... + */ + + dstlen = strlen(dst); + size -= dstlen + 1; + + if (!size) + return (dstlen); /* No room, return immediately... */ + + /* + * Figure out how much room is needed... + */ + + srclen = strlen(src); + + /* + * Copy the appropriate amount... + */ + + if (srclen > size) + srclen = size; + + memcpy(dst + dstlen, src, srclen); + dst[dstlen + srclen] = '\0'; + + return (dstlen + srclen); +} +#endif /* !HAVE_STRLCAT */ + + +#ifndef HAVE_STRLCPY +/* + * '_cups_strlcpy()' - Safely copy two strings. + */ + +size_t /* O - Length of string */ +_cups_strlcpy(char *dst, /* O - Destination string */ + const char *src, /* I - Source string */ + size_t size) /* I - Size of destination string buffer */ +{ + size_t srclen; /* Length of source string */ + + + /* + * Figure out how much room is needed... + */ + + size --; + + srclen = strlen(src); + + /* + * Copy the appropriate amount... + */ + + if (srclen > size) + srclen = size; + + memcpy(dst, src, srclen); + dst[srclen] = '\0'; + + return (srclen); +} +#endif /* !HAVE_STRLCPY */ + + +/* + * 'compare_sp_items()' - Compare two string pool items... + */ + +static int /* O - Result of comparison */ +compare_sp_items(_cups_sp_item_t *a, /* I - First item */ + _cups_sp_item_t *b) /* I - Second item */ +{ + return (strcmp(a->str, b->str)); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/tempfile.c b/cups/tempfile.c new file mode 100644 index 0000000000..0eec464712 --- /dev/null +++ b/cups/tempfile.c @@ -0,0 +1,233 @@ +/* + * "$Id$" + * + * Temp file utilities for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsTempFd() - Creates a temporary file. + * cupsTempFile() - Generates a temporary filename. + * cupsTempFile2() - Creates a temporary CUPS file. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include +#include +#include +#if defined(WIN32) || defined(__EMX__) +# include +#else +# include +#endif /* WIN32 || __EMX__ */ + + +/* + * 'cupsTempFd()' - Creates a temporary file. + * + * The temporary filename is returned in the filename buffer. + * The temporary file is opened for reading and writing. + */ + +int /* O - New file descriptor or -1 on error */ +cupsTempFd(char *filename, /* I - Pointer to buffer */ + int len) /* I - Size of buffer */ +{ + int fd; /* File descriptor for temp file */ + int tries; /* Number of tries */ + const char *tmpdir; /* TMPDIR environment var */ +#ifdef WIN32 + char tmppath[1024]; /* Windows temporary directory */ + DWORD curtime; /* Current time */ +#else + struct timeval curtime; /* Current time */ +#endif /* WIN32 */ + + + /* + * See if TMPDIR is defined... + */ + +#ifdef WIN32 + if ((tmpdir = getenv("TEMP")) == NULL) + { + GetTempPath(sizeof(tmppath), tmppath); + tmpdir = tmppath; + } +#else + /* + * Previously we put root temporary files in the default CUPS temporary + * directory under /var/spool/cups. However, since the scheduler cleans + * out temporary files there and runs independently of the user apps, we + * don't want to use it unless specifically told to by cupsd. + */ + + if ((tmpdir = getenv("TMPDIR")) == NULL) +# ifdef __APPLE__ + tmpdir = "/private/tmp"; /* /tmp is a symlink to /private/tmp */ +# else + tmpdir = "/tmp"; +# endif /* __APPLE__ */ +#endif /* WIN32 */ + + /* + * Make the temporary name using the specified directory... + */ + + tries = 0; + + do + { +#ifdef WIN32 + /* + * Get the current time of day... + */ + + curtime = GetTickCount() + tries; + + /* + * Format a string using the hex time values... + */ + + snprintf(filename, len - 1, "%s/%05lx%08lx", tmpdir, + GetCurrentProcessId(), curtime); +#else + /* + * Get the current time of day... + */ + + gettimeofday(&curtime, NULL); + + /* + * Format a string using the hex time values... + */ + + snprintf(filename, len - 1, "%s/%05x%08x", tmpdir, (unsigned)getpid(), + (unsigned)(curtime.tv_sec + curtime.tv_usec + tries)); +#endif /* WIN32 */ + + /* + * Open the file in "exclusive" mode, making sure that we don't + * stomp on an existing file or someone's symlink crack... + */ + +#ifdef WIN32 + fd = open(filename, _O_CREAT | _O_RDWR | _O_TRUNC | _O_BINARY, + _S_IREAD | _S_IWRITE); +#elif defined(O_NOFOLLOW) + fd = open(filename, O_RDWR | O_CREAT | O_EXCL | O_NOFOLLOW, 0600); +#else + fd = open(filename, O_RDWR | O_CREAT | O_EXCL, 0600); +#endif /* WIN32 */ + + if (fd < 0 && errno != EEXIST) + break; + + tries ++; + } + while (fd < 0 && tries < 1000); + + /* + * Return the file descriptor... + */ + + return (fd); +} + + +/* + * 'cupsTempFile()' - Generates a temporary filename. + * + * The temporary filename is returned in the filename buffer. + * This function is deprecated - use @link cupsTempFd@ or + * @link cupsTempFile2@ instead. + * + * @deprecated@ + */ + +char * /* O - Filename or @code NULL@ on error */ +cupsTempFile(char *filename, /* I - Pointer to buffer */ + int len) /* I - Size of buffer */ +{ + int fd; /* File descriptor for temp file */ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + /* + * See if a filename was specified... + */ + + if (filename == NULL) + { + filename = cg->tempfile; + len = sizeof(cg->tempfile); + } + + /* + * Create the temporary file... + */ + + if ((fd = cupsTempFd(filename, len)) < 0) + return (NULL); + + /* + * Close the temp file - it'll be reopened later as needed... + */ + + close(fd); + + /* + * Return the temp filename... + */ + + return (filename); +} + + +/* + * 'cupsTempFile2()' - Creates a temporary CUPS file. + * + * The temporary filename is returned in the filename buffer. + * The temporary file is opened for writing. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +cups_file_t * /* O - CUPS file or @code NULL@ on error */ +cupsTempFile2(char *filename, /* I - Pointer to buffer */ + int len) /* I - Size of buffer */ +{ + cups_file_t *file; /* CUPS file */ + int fd; /* File descriptor */ + + + if ((fd = cupsTempFd(filename, len)) < 0) + return (NULL); + else if ((file = cupsFileOpenFd(fd, "w")) == NULL) + { + close(fd); + unlink(filename); + return (NULL); + } + else + return (file); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/test.ppd b/cups/test.ppd new file mode 100644 index 0000000000..ed7fe1d1d2 --- /dev/null +++ b/cups/test.ppd @@ -0,0 +1,262 @@ +*PPD-Adobe: "4.3" +*% +*% "$Id$" +*% +*% Test PPD file for CUPS. +*% +*% This file is used to test the CUPS PPD API functions and cannot be +*% used with any known printers. Look on the CUPS web site for working PPD +*% files. +*% +*% If you are a PPD file developer, consider using the PPD compiler (ppdc) +*% to create your PPD files - not only will it save you time, it produces +*% consistently high-quality files. +*% +*% Copyright 2007-2010 by Apple Inc. +*% Copyright 2002-2006 by Easy Software Products. +*% +*% These coded instructions, statements, and computer programs are the +*% property of Apple Inc. and are protected by Federal copyright +*% law. Distribution and use rights are outlined in the file "LICENSE.txt" +*% which should have been included with this file. If this file is +*% file is missing or damaged, see the license at "http://www.cups.org/". +*FormatVersion: "4.3" +*FileVersion: "1.3" +*LanguageVersion: English +*LanguageEncoding: ISOLatin1 +*PCFileName: "TEST.PPD" +*Manufacturer: "ESP" +*Product: "(Test)" +*cupsVersion: 1.4 +*ModelName: "Test" +*ShortNickName: "Test" +*NickName: "Test for CUPS" +*PSVersion: "(3010.000) 0" +*LanguageLevel: "3" +*ColorDevice: True +*DefaultColorSpace: RGB +*FileSystem: False +*Throughput: "1" +*LandscapeOrientation: Plus90 +*TTRasterizer: Type42 +*cupsFilter: "application/vnd.cups-raster 0 -" +*RequiresPageRegion All: True + +*% These constraints are used to test ppdConflicts() and cupsResolveConflicts() +*UIConstraints: *PageSize Letter *InputSlot Envelope +*UIConstraints: *InputSlot Envelope *PageSize Letter +*UIConstraints: *PageRegion Letter *InputSlot Envelope +*UIConstraints: *InputSlot Envelope *PageRegion Letter + +*% These constraints are used to test ppdInstallableConflict() +*UIConstraints: "*Duplex *InstalledDuplexer False" +*UIConstraints: "*InstalledDuplexer False *Duplex" + +*% These attributes test ppdFindAttr/ppdFindNext... +*cupsTest Foo/I Love Foo: "" +*cupsTest Bar/I Love Bar: "" + +*% For PageSize, we have put all of the translations in-line... +*OpenUI *PageSize/Page Size: PickOne +*fr.Translation PageSize/French Page Size: "" +*fr_CA.Translation PageSize/French Canadian Page Size: "" +*OrderDependency: 10 AnySetup *PageSize +*DefaultPageSize: Letter +*PageSize Letter/US Letter: "PageSize=Letter" +*fr.PageSize Letter/French US Letter: "" +*fr_CA.PageSize Letter/French Canadian US Letter: "" +*PageSize Letter.Banner/US Letter Banner: "PageSize=Letter.Banner" +*fr.PageSize Letter.Banner/French US Letter Banner: "" +*fr_CA.PageSize Letter.Banner/French Canadian US Letter Banner: "" +*PageSize Letter.Fullbleed/US Letter Borderless: "PageSize=Letter.Fullbleed" +*fr.PageSize Letter.Fullbleed/French US Letter Borderless: "" +*fr_CA.PageSize Letter.Fullbleed/French Canadian US Letter Borderless: "" +*PageSize A4/A4: "PageSize=A4" +*fr.PageSize A4/French A4: "" +*fr_CA.PageSize A4/French Canadian A4: "" +*PageSize Env10/#10 Envelope: "PageSize=Env10" +*fr.PageSize Env10/French #10 Envelope: "" +*fr_CA.PageSize Env10/French Canadian #10 Envelope: "" +*CloseUI: *PageSize + +*% For PageRegion, we have separated the translations... +*OpenUI *PageRegion/Page Region: PickOne +*OrderDependency: 10 AnySetup *PageRegion +*DefaultPageRegion: Letter +*PageRegion Letter/US Letter: "PageRegion=Letter" +*PageRegion Letter.Banner/US Letter Banner: "PageRegion=Letter.Fullbleed" +*PageRegion Letter.Fullbleed/US Letter Borderless: "PageRegion=Letter.Fullbleed" +*PageRegion A4/A4: "PageRegion=A4" +*PageRegion Env10/#10 Envelope: "PageRegion=Env10" +*CloseUI: *PageRegion + +*fr.Translation PageRegion/French Page Region: "" +*fr.PageRegion Letter/French US Letter: "" +*fr.PageRegion Letter.Banner/French US Letter Banner: "" +*fr.PageRegion Letter.Fullbleed/French US Letter Borderless: "" +*fr.PageRegion A4/French A4: "" +*fr.PageRegion Env10/French #10 Envelope: "" + +*fr_CA.Translation PageRegion/French Canadian Page Region: "" +*fr_CA.PageRegion Letter/French Canadian US Letter: "" +*fr_CA.PageRegion Letter.Banner/French Canadian US Letter Banner: "" +*fr_CA.PageRegion Letter.Fullbleed/French Canadian US Letter Borderless: "" +*fr_CA.PageRegion A4/French Canadian A4: "" +*fr_CA.PageRegion Env10/French Canadian #10 Envelope: "" + +*DefaultImageableArea: Letter +*ImageableArea Letter: "18 36 594 756" +*ImageableArea Letter.Banner: "18 0 594 792" +*ImageableArea Letter.Fullbleed: "0 0 612 792" +*ImageableArea A4: "18 36 577 806" +*ImageableArea Env10: "18 36 279 648" + +*DefaultPaperDimension: Letter +*PaperDimension Letter: "612 792" +*PaperDimension Letter.Banner: "612 792" +*PaperDimension Letter.Fullbleed: "612 792" +*PaperDimension A4: "595 842" +*PaperDimension Env10: "297 684" + +*% Custom page size support +*HWMargins: 0 0 0 0 +*NonUIOrderDependency: 100 AnySetup *CustomPageSize True +*CustomPageSize True/Custom Page Size: "PageSize=Custom" +*ParamCustomPageSize Width: 1 points 36 1080 +*ParamCustomPageSize Height: 2 points 36 86400 +*ParamCustomPageSize WidthOffset/Width Offset: 3 points 0 0 +*ParamCustomPageSize HeightOffset/Height Offset: 4 points 0 0 +*ParamCustomPageSize Orientation: 5 int 0 0 + +*OpenUI *InputSlot/Input Slot: PickOne +*OrderDependency: 20 AnySetup *InputSlot +*DefaultInputSlot: Tray +*InputSlot Tray/Tray: "InputSlot=Tray" +*InputSlot Manual/Manual Feed: "InputSlot=Manual" +*InputSlot Envelope/Envelope Feed: "InputSlot=Envelope" +*CloseUI: *InputSlot + +*OpenUI *MediaType/Media Type: PickOne +*OrderDependency: 25 AnySetup *MediaType +*DefaultMediaType: Plain +*MediaType Plain/Plain Paper: "MediaType=Plain" +*MediaType Matte/Matte Photo: "MediaType=Matte" +*MediaType Glossy/Glossy Photo: "MediaType=Glossy" +*MediaType Transparency/Transparency Film: "MediaType=Transparency" +*CloseUI: *MediaType + +*OpenUI *Duplex/2-Sided Printing: PickOne +*OrderDependency: 10 DocumentSetup *Duplex +*DefaultDuplex: None +*Duplex None/Off: "Duplex=None" +*Duplex DuplexNoTumble/Long Edge: "Duplex=DuplexNoTumble" +*Duplex DuplexTumble/Short Edge: "Duplex=DuplexTumble" +*CloseUI: *Duplex + +*% Installable option... +*OpenGroup: InstallableOptions/Installable Options +*OpenUI InstalledDuplexer/Duplexer Installed: Boolean +*DefaultInstalledDuplexer: False +*InstalledDuplexer False: "" +*InstalledDuplexer True: "" +*CloseUI: *InstalledDuplexer +*CloseGroup: InstallableOptions + +*% Custom options... +*OpenGroup: Extended/Extended Options + +*OpenUI IntOption/Integer: PickOne +*OrderDependency: 30 AnySetup *IntOption +*DefaultIntOption: None +*IntOption None: "" +*IntOption 1: "IntOption=1" +*IntOption 2: "IntOption=2" +*IntOption 3: "IntOption=3" +*CloseUI: *IntOption + +*CustomIntOption True/Custom Integer: "IntOption=Custom" +*ParamCustomIntOption Integer: 1 int -100 100 + +*OpenUI StringOption/String: PickOne +*OrderDependency: 40 AnySetup *StringOption +*DefaultStringOption: None +*StringOption None: "" +*StringOption foo: "StringOption=foo" +*StringOption bar: "StringOption=bar" +*CloseUI: *StringOption + +*CustomStringOption True/Custom String: "StringOption=Custom" +*ParamCustomStringOption String1: 2 string 1 10 +*ParamCustomStringOption String2: 1 string 1 10 + +*CloseGroup: Extended + +*% IPP reasons for ppdLocalizeIPPReason tests +*cupsIPPReason foo/Foo Reason: "http://foo/bar.html +help:anchor='foo'%20bookID=Vendor%20Help +/help/foo/bar.html" +*End +*fr.cupsIPPReason foo/La Foo Reason: "text:La%20Long%20 +text:Foo%20Reason +http://foo/fr/bar.html +help:anchor='foo'%20bookID=Vendor%20Help +/help/fr/foo/bar.html" +*End +*zh_TW.cupsIPPReason foo/Number 1 Foo Reason: "text:Number%201%20 +text:Foo%20Reason +http://foo/zh_TW/bar.html +help:anchor='foo'%20bookID=Vendor%20Help +/help/zh_TW/foo/bar.html" +*End +*zh.cupsIPPReason foo/Number 2 Foo Reason: "text:Number%202%20 +text:Foo%20Reason +http://foo/zh/bar.html +help:anchor='foo'%20bookID=Vendor%20Help +/help/zh/foo/bar.html" +*End + +*% Marker names for ppdLocalizeMarkerName tests +*cupsMarkerName cyan/Cyan Toner: "" +*fr.cupsMarkerName cyan/La Toner Cyan: "" +*zh_TW.cupsMarkerName cyan/Number 1 Cyan Toner: "" +*zh.cupsMarkerName cyan/Number 2 Cyan Toner: "" + +*DefaultFont: Courier +*Font AvantGarde-Book: Standard "(001.006S)" Standard ROM +*Font AvantGarde-BookOblique: Standard "(001.006S)" Standard ROM +*Font AvantGarde-Demi: Standard "(001.007S)" Standard ROM +*Font AvantGarde-DemiOblique: Standard "(001.007S)" Standard ROM +*Font Bookman-Demi: Standard "(001.004S)" Standard ROM +*Font Bookman-DemiItalic: Standard "(001.004S)" Standard ROM +*Font Bookman-Light: Standard "(001.004S)" Standard ROM +*Font Bookman-LightItalic: Standard "(001.004S)" Standard ROM +*Font Courier: Standard "(002.004S)" Standard ROM +*Font Courier-Bold: Standard "(002.004S)" Standard ROM +*Font Courier-BoldOblique: Standard "(002.004S)" Standard ROM +*Font Courier-Oblique: Standard "(002.004S)" Standard ROM +*Font Helvetica: Standard "(001.006S)" Standard ROM +*Font Helvetica-Bold: Standard "(001.007S)" Standard ROM +*Font Helvetica-BoldOblique: Standard "(001.007S)" Standard ROM +*Font Helvetica-Narrow: Standard "(001.006S)" Standard ROM +*Font Helvetica-Narrow-Bold: Standard "(001.007S)" Standard ROM +*Font Helvetica-Narrow-BoldOblique: Standard "(001.007S)" Standard ROM +*Font Helvetica-Narrow-Oblique: Standard "(001.006S)" Standard ROM +*Font Helvetica-Oblique: Standard "(001.006S)" Standard ROM +*Font NewCenturySchlbk-Bold: Standard "(001.009S)" Standard ROM +*Font NewCenturySchlbk-BoldItalic: Standard "(001.007S)" Standard ROM +*Font NewCenturySchlbk-Italic: Standard "(001.006S)" Standard ROM +*Font NewCenturySchlbk-Roman: Standard "(001.007S)" Standard ROM +*Font Palatino-Bold: Standard "(001.005S)" Standard ROM +*Font Palatino-BoldItalic: Standard "(001.005S)" Standard ROM +*Font Palatino-Italic: Standard "(001.005S)" Standard ROM +*Font Palatino-Roman: Standard "(001.005S)" Standard ROM +*Font Symbol: Special "(001.007S)" Special ROM +*Font Times-Bold: Standard "(001.007S)" Standard ROM +*Font Times-BoldItalic: Standard "(001.009S)" Standard ROM +*Font Times-Italic: Standard "(001.007S)" Standard ROM +*Font Times-Roman: Standard "(001.007S)" Standard ROM +*Font ZapfChancery-MediumItalic: Standard "(001.007S)" Standard ROM +*Font ZapfDingbats: Special "(001.004S)" Standard ROM +*% +*% End of "$Id$". +*% diff --git a/cups/test2.ppd b/cups/test2.ppd new file mode 100644 index 0000000000..78ff2f7f01 --- /dev/null +++ b/cups/test2.ppd @@ -0,0 +1,252 @@ +*PPD-Adobe: "4.3" +*% +*% "$Id$" +*% +*% Test PPD file #2 for CUPS. +*% +*% This file is used to test the CUPS PPD API functions and cannot be +*% used with any known printers. Look on the CUPS web site for working PPD +*% files. +*% +*% If you are a PPD file developer, consider using the PPD compiler (ppdc) +*% to create your PPD files - not only will it save you time, it produces +*% consistently high-quality files. +*% +*% Copyright 2007-2011 by Apple Inc. +*% Copyright 2002-2006 by Easy Software Products. +*% +*% These coded instructions, statements, and computer programs are the +*% property of Apple Inc. and are protected by Federal copyright +*% law. Distribution and use rights are outlined in the file "LICENSE.txt" +*% which should have been included with this file. If this file is +*% file is missing or damaged, see the license at "http://www.cups.org/". +*FormatVersion: "4.3" +*FileVersion: "1.3" +*LanguageVersion: English +*LanguageEncoding: ISOLatin1 +*PCFileName: "TEST.PPD" +*Manufacturer: "ESP" +*Product: "(Test2)" +*cupsVersion: 1.4 +*ModelName: "Test2" +*ShortNickName: "Test2" +*NickName: "Test2 for CUPS" +*PSVersion: "(3010.000) 0" +*LanguageLevel: "3" +*ColorDevice: True +*DefaultColorSpace: RGB +*FileSystem: False +*Throughput: "1" +*LandscapeOrientation: Plus90 +*TTRasterizer: Type42 + +*% These constraints are used to test ppdConflicts() and cupsResolveConflicts() +*cupsUIConstraints envelope: "*PageSize Letter *InputSlot Envelope" +*cupsUIConstraints envelope: "*PageSize A4 *InputSlot Envelope" +*cupsUIResolver envelope: "*InputSlot Manual *PageSize Env10" + +*cupsUIConstraints envphoto: "*PageSize Env10 *InputSlot Envelope *Quality Photo" +*cupsUIResolver envphoto: "*Quality Normal" + +*% This constraint is used to test ppdInstallableConflict() +*cupsUIConstraints: "*Duplex *InstalledDuplexer False" + +*% These constraints are used to test the loop detection code in cupsResolveConflicts() +*cupsUIConstraints loop1: "*PageSize A4 *Quality Photo" +*cupsUIResolver loop1: "*Quality Normal" +*cupsUIConstraints loop2: "*PageSize A4 *Quality Normal" +*cupsUIResolver loop2: "*Quality Photo" + +*% For PageSize, we have put all of the translations in-line... +*OpenUI *PageSize/Page Size: PickOne +*fr.Translation PageSize/French Page Size: "" +*fr_CA.Translation PageSize/French Canadian Page Size: "" +*OrderDependency: 10 AnySetup *PageSize +*DefaultPageSize: Letter +*PageSize Letter/US Letter: "PageSize=Letter" +*fr.PageSize Letter/French US Letter: "" +*fr_CA.PageSize Letter/French Canadian US Letter: "" +*PageSize A4/A4: "PageSize=A4" +*fr.PageSize A4/French A4: "" +*fr_CA.PageSize A4/French Canadian A4: "" +*PageSize Env10/#10 Envelope: "PageSize=Env10" +*fr.PageSize Env10/French #10 Envelope: "" +*fr_CA.PageSize Env10/French Canadian #10 Envelope: "" +*CloseUI: *PageSize + +*% For PageRegion, we have separated the translations... +*OpenUI *PageRegion/Page Region: PickOne +*OrderDependency: 10 AnySetup *PageRegion +*DefaultPageRegion: Letter +*PageRegion Letter/US Letter: "PageRegion=Letter" +*PageRegion A4/A4: "PageRegion=A4" +*PageRegion Env10/#10 Envelope: "PageRegion=Env10" +*CloseUI: *PageRegion + +*fr.Translation PageRegion/French Page Region: "" +*fr.PageRegion Letter/French US Letter: "" +*fr.PageRegion A4/French A4: "" +*fr.PageRegion Env10/French #10 Envelope: "" + +*fr_CA.Translation PageRegion/French Canadian Page Region: "" +*fr_CA.PageRegion Letter/French Canadian US Letter: "" +*fr_CA.PageRegion A4/French Canadian A4: "" +*fr_CA.PageRegion Env10/French Canadian #10 Envelope: "" + +*DefaultImageableArea: Letter +*ImageableArea Letter: "18 36 594 756" +*ImageableArea A4: "18 36 577 806" +*ImageableArea Env10: "18 36 279 648" + +*DefaultPaperDimension: Letter +*PaperDimension Letter: "612 792" +*PaperDimension A4: "595 842" +*PaperDimension Env10: "297 684" + +*% Custom page size support +*HWMargins: 0 0 0 0 +*NonUIOrderDependency: 100 AnySetup *CustomPageSize True +*CustomPageSize True/Custom Page Size: "PageSize=Custom" +*ParamCustomPageSize Width: 1 points 36 1080 +*ParamCustomPageSize Height: 2 points 36 86400 +*ParamCustomPageSize WidthOffset/Width Offset: 3 points 0 0 +*ParamCustomPageSize HeightOffset/Height Offset: 4 points 0 0 +*ParamCustomPageSize Orientation: 5 int 0 0 + +*cupsMediaQualifier2: InputSlot +*cupsMediaQualifier3: Quality +*cupsMaxSize .Manual.: "1000 1000" +*cupsMinSize .Manual.: "100 100" +*cupsMinSize .Manual.Photo: "200 200" +*cupsMinSize ..Photo: "300 300" + +*OpenUI *InputSlot/Input Slot: PickOne +*OrderDependency: 20 AnySetup *InputSlot +*DefaultInputSlot: Tray +*InputSlot Tray/Tray: "InputSlot=Tray" +*InputSlot Manual/Manual Feed: "InputSlot=Manual" +*InputSlot Envelope/Envelope Feed: "InputSlot=Envelope" +*CloseUI: *InputSlot + +*OpenUI *Quality/Output Mode: PickOne +*OrderDependency: 20 AnySetup *Quality +*DefaultQuality: Normal +*Quality Draft: "Quality=Draft" +*Quality Normal: "Quality=Normal" +*Quality Photo: "Quality=Photo" +*CloseUI: *Quality + +*OpenUI *Duplex/2-Sided Printing: PickOne +*OrderDependency: 10 DocumentSetup *Duplex +*DefaultDuplex: None +*Duplex None/Off: "Duplex=None" +*Duplex DuplexNoTumble/Long Edge: "Duplex=DuplexNoTumble" +*Duplex DuplexTumble/Short Edge: "Duplex=DuplexTumble" +*CloseUI: *Duplex + +*% Installable option... +*OpenGroup: InstallableOptions/Installable Options +*OpenUI InstalledDuplexer/Duplexer Installed: Boolean +*DefaultInstalledDuplexer: False +*InstalledDuplexer False: "" +*InstalledDuplexer True: "" +*CloseUI: *InstalledDuplexer +*CloseGroup: InstallableOptions + +*% Custom options... +*OpenGroup: Extended/Extended Options + +*OpenUI IntOption/Integer: PickOne +*OrderDependency: 30 AnySetup *IntOption +*DefaultIntOption: None +*IntOption None: "" +*IntOption 1: "IntOption=1" +*IntOption 2: "IntOption=2" +*IntOption 3: "IntOption=3" +*CloseUI: *IntOption + +*CustomIntOption True/Custom Integer: "IntOption=Custom" +*ParamCustomIntOption Integer: 1 int -100 100 + +*OpenUI StringOption/String: PickOne +*OrderDependency: 40 AnySetup *StringOption +*DefaultStringOption: None +*StringOption None: "" +*StringOption foo: "StringOption=foo" +*StringOption bar: "StringOption=bar" +*CloseUI: *StringOption + +*CustomStringOption True/Custom String: "StringOption=Custom" +*ParamCustomStringOption String: 1 string 1 10 + +*CloseGroup: Extended + +*% IPP reasons for ppdLocalizeIPPReason tests +*cupsIPPReason foo/Foo Reason: "http://foo/bar.html +help:anchor='foo'%20bookID=Vendor%20Help +/help/foo/bar.html" +*End +*fr.cupsIPPReason foo/La Foo Reason: "text:La%20Long%20 +text:Foo%20Reason +http://foo/fr/bar.html +help:anchor='foo'%20bookID=Vendor%20Help +/help/fr/foo/bar.html" +*End +*zh_TW.cupsIPPReason foo/Number 1 Foo Reason: "text:Number%201%20 +text:Foo%20Reason +http://foo/zh_TW/bar.html +help:anchor='foo'%20bookID=Vendor%20Help +/help/zh_TW/foo/bar.html" +*End +*zh.cupsIPPReason foo/Number 2 Foo Reason: "text:Number%202%20 +text:Foo%20Reason +http://foo/zh/bar.html +help:anchor='foo'%20bookID=Vendor%20Help +/help/zh/foo/bar.html" +*End + +*% Marker names for ppdLocalizeMarkerName tests +*cupsMarkerName cyan/Cyan Toner: "" +*fr.cupsMarkerName cyan/La Toner Cyan: "" +*zh_TW.cupsMarkerName cyan/Number 1 Cyan Toner: "" +*zh.cupsMarkerName cyan/Number 2 Cyan Toner: "" + +*DefaultFont: Courier +*Font AvantGarde-Book: Standard "(001.006S)" Standard ROM +*Font AvantGarde-BookOblique: Standard "(001.006S)" Standard ROM +*Font AvantGarde-Demi: Standard "(001.007S)" Standard ROM +*Font AvantGarde-DemiOblique: Standard "(001.007S)" Standard ROM +*Font Bookman-Demi: Standard "(001.004S)" Standard ROM +*Font Bookman-DemiItalic: Standard "(001.004S)" Standard ROM +*Font Bookman-Light: Standard "(001.004S)" Standard ROM +*Font Bookman-LightItalic: Standard "(001.004S)" Standard ROM +*Font Courier: Standard "(002.004S)" Standard ROM +*Font Courier-Bold: Standard "(002.004S)" Standard ROM +*Font Courier-BoldOblique: Standard "(002.004S)" Standard ROM +*Font Courier-Oblique: Standard "(002.004S)" Standard ROM +*Font Helvetica: Standard "(001.006S)" Standard ROM +*Font Helvetica-Bold: Standard "(001.007S)" Standard ROM +*Font Helvetica-BoldOblique: Standard "(001.007S)" Standard ROM +*Font Helvetica-Narrow: Standard "(001.006S)" Standard ROM +*Font Helvetica-Narrow-Bold: Standard "(001.007S)" Standard ROM +*Font Helvetica-Narrow-BoldOblique: Standard "(001.007S)" Standard ROM +*Font Helvetica-Narrow-Oblique: Standard "(001.006S)" Standard ROM +*Font Helvetica-Oblique: Standard "(001.006S)" Standard ROM +*Font NewCenturySchlbk-Bold: Standard "(001.009S)" Standard ROM +*Font NewCenturySchlbk-BoldItalic: Standard "(001.007S)" Standard ROM +*Font NewCenturySchlbk-Italic: Standard "(001.006S)" Standard ROM +*Font NewCenturySchlbk-Roman: Standard "(001.007S)" Standard ROM +*Font Palatino-Bold: Standard "(001.005S)" Standard ROM +*Font Palatino-BoldItalic: Standard "(001.005S)" Standard ROM +*Font Palatino-Italic: Standard "(001.005S)" Standard ROM +*Font Palatino-Roman: Standard "(001.005S)" Standard ROM +*Font Symbol: Special "(001.007S)" Special ROM +*Font Times-Bold: Standard "(001.007S)" Standard ROM +*Font Times-BoldItalic: Standard "(001.009S)" Standard ROM +*Font Times-Italic: Standard "(001.007S)" Standard ROM +*Font Times-Roman: Standard "(001.007S)" Standard ROM +*Font ZapfChancery-MediumItalic: Standard "(001.007S)" Standard ROM +*Font ZapfDingbats: Special "(001.004S)" Standard ROM +*% +*% End of "$Id$". +*% diff --git a/cups/testadmin.c b/cups/testadmin.c new file mode 100644 index 0000000000..84772ffed3 --- /dev/null +++ b/cups/testadmin.c @@ -0,0 +1,120 @@ +/* + * "$Id$" + * + * Admin function test program for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Main entry. + * show_settings() - Show settings in the array... + */ + +/* + * Include necessary headers... + */ + +#include "adminutil.h" +#include "string-private.h" + + +/* + * Local functions... + */ + +static void show_settings(int num_settings, cups_option_t *settings); + + +/* + * 'main()' - Main entry. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int i, /* Looping var */ + num_settings; /* Number of settings */ + cups_option_t *settings; /* Settings */ + http_t *http; /* Connection to server */ + + + /* + * Connect to the server using the defaults... + */ + + http = httpConnectEncrypt(cupsServer(), ippPort(), cupsEncryption()); + + /* + * Set the current configuration if we have anything on the command-line... + */ + + if (argc > 1) + { + for (i = 1, num_settings = 0, settings = NULL; i < argc; i ++) + num_settings = cupsParseOptions(argv[i], num_settings, &settings); + + if (cupsAdminSetServerSettings(http, num_settings, settings)) + { + puts("New server settings:"); + cupsFreeOptions(num_settings, settings); + } + else + { + printf("Server settings not changed: %s\n", cupsLastErrorString()); + return (1); + } + } + else + puts("Current server settings:"); + + /* + * Get the current configuration... + */ + + if (cupsAdminGetServerSettings(http, &num_settings, &settings)) + { + show_settings(num_settings, settings); + cupsFreeOptions(num_settings, settings); + return (0); + } + else + { + printf(" %s\n", cupsLastErrorString()); + return (1); + } +} + + +/* + * 'show_settings()' - Show settings in the array... + */ + +static void +show_settings( + int num_settings, /* I - Number of settings */ + cups_option_t *settings) /* I - Settings */ +{ + while (num_settings > 0) + { + printf(" %s=%s\n", settings->name, settings->value); + + settings ++; + num_settings --; + } +} + + +/* + * End of "$Id$". + */ diff --git a/cups/testarray.c b/cups/testarray.c new file mode 100644 index 0000000000..8371840fa0 --- /dev/null +++ b/cups/testarray.c @@ -0,0 +1,480 @@ +/* + * "$Id$" + * + * Array test program for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Main entry. + * get_seconds() - Get the current time in seconds... + * load_words() - Load words from a file. + */ + +/* + * Include necessary headers... + */ + +#include "string-private.h" +#include "debug-private.h" +#include "array.h" +#include "dir.h" + + +/* + * Local functions... + */ + +static double get_seconds(void); +static int load_words(const char *filename, cups_array_t *array); + + +/* + * 'main()' - Main entry. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + cups_array_t *array, /* Test array */ + *dup_array; /* Duplicate array */ + int status; /* Exit status */ + char *text; /* Text from array */ + char word[256]; /* Word from file */ + double start, /* Start time */ + end; /* End time */ + cups_dir_t *dir; /* Current directory */ + cups_dentry_t *dent; /* Directory entry */ + char *saved[32]; /* Saved entries */ + void *data; /* User data for arrays */ + + + /* + * No errors so far... + */ + + status = 0; + + /* + * cupsArrayNew() + */ + + fputs("cupsArrayNew: ", stdout); + + data = (void *)"testarray"; + array = cupsArrayNew((cups_array_func_t)strcmp, data); + + if (array) + puts("PASS"); + else + { + puts("FAIL (returned NULL, expected pointer)"); + status ++; + } + + /* + * cupsArrayUserData() + */ + + fputs("cupsArrayUserData: ", stdout); + if (cupsArrayUserData(array) == data) + puts("PASS"); + else + { + printf("FAIL (returned %p instead of %p!)\n", cupsArrayUserData(array), + data); + status ++; + } + + /* + * cupsArrayAdd() + */ + + fputs("cupsArrayAdd: ", stdout); + + if (!cupsArrayAdd(array, strdup("One Fish"))) + { + puts("FAIL (\"One Fish\")"); + status ++; + } + else + { + if (!cupsArrayAdd(array, strdup("Two Fish"))) + { + puts("FAIL (\"Two Fish\")"); + status ++; + } + else + { + if (!cupsArrayAdd(array, strdup("Red Fish"))) + { + puts("FAIL (\"Red Fish\")"); + status ++; + } + else + { + if (!cupsArrayAdd(array, strdup("Blue Fish"))) + { + puts("FAIL (\"Blue Fish\")"); + status ++; + } + else + puts("PASS"); + } + } + } + + /* + * cupsArrayCount() + */ + + fputs("cupsArrayCount: ", stdout); + if (cupsArrayCount(array) == 4) + puts("PASS"); + else + { + printf("FAIL (returned %d, expected 4)\n", cupsArrayCount(array)); + status ++; + } + + /* + * cupsArrayFirst() + */ + + fputs("cupsArrayFirst: ", stdout); + if ((text = (char *)cupsArrayFirst(array)) != NULL && + !strcmp(text, "Blue Fish")) + puts("PASS"); + else + { + printf("FAIL (returned \"%s\", expected \"Blue Fish\")\n", text); + status ++; + } + + /* + * cupsArrayNext() + */ + + fputs("cupsArrayNext: ", stdout); + if ((text = (char *)cupsArrayNext(array)) != NULL && + !strcmp(text, "One Fish")) + puts("PASS"); + else + { + printf("FAIL (returned \"%s\", expected \"One Fish\")\n", text); + status ++; + } + + /* + * cupsArrayLast() + */ + + fputs("cupsArrayLast: ", stdout); + if ((text = (char *)cupsArrayLast(array)) != NULL && + !strcmp(text, "Two Fish")) + puts("PASS"); + else + { + printf("FAIL (returned \"%s\", expected \"Two Fish\")\n", text); + status ++; + } + + /* + * cupsArrayPrev() + */ + + fputs("cupsArrayPrev: ", stdout); + if ((text = (char *)cupsArrayPrev(array)) != NULL && + !strcmp(text, "Red Fish")) + puts("PASS"); + else + { + printf("FAIL (returned \"%s\", expected \"Red Fish\")\n", text); + status ++; + } + + /* + * cupsArrayFind() + */ + + fputs("cupsArrayFind: ", stdout); + if ((text = (char *)cupsArrayFind(array, (void *)"One Fish")) != NULL && + !strcmp(text, "One Fish")) + puts("PASS"); + else + { + printf("FAIL (returned \"%s\", expected \"One Fish\")\n", text); + status ++; + } + + /* + * cupsArrayCurrent() + */ + + fputs("cupsArrayCurrent: ", stdout); + if ((text = (char *)cupsArrayCurrent(array)) != NULL && + !strcmp(text, "One Fish")) + puts("PASS"); + else + { + printf("FAIL (returned \"%s\", expected \"One Fish\")\n", text); + status ++; + } + + /* + * cupsArrayDup() + */ + + fputs("cupsArrayDup: ", stdout); + if ((dup_array = cupsArrayDup(array)) != NULL && + cupsArrayCount(dup_array) == 4) + puts("PASS"); + else + { + printf("FAIL (returned %p with %d elements, expected pointer with 4 elements)\n", + dup_array, cupsArrayCount(dup_array)); + status ++; + } + + /* + * cupsArrayRemove() + */ + + fputs("cupsArrayRemove: ", stdout); + if (cupsArrayRemove(array, (void *)"One Fish") && + cupsArrayCount(array) == 3) + puts("PASS"); + else + { + printf("FAIL (returned 0 with %d elements, expected 1 with 4 elements)\n", + cupsArrayCount(array)); + status ++; + } + + /* + * cupsArrayClear() + */ + + fputs("cupsArrayClear: ", stdout); + cupsArrayClear(array); + if (cupsArrayCount(array) == 0) + puts("PASS"); + else + { + printf("FAIL (%d elements, expected 0 elements)\n", + cupsArrayCount(array)); + status ++; + } + + /* + * Now load this source file and grab all of the unique words... + */ + + fputs("Load unique words: ", stdout); + fflush(stdout); + + start = get_seconds(); + + if ((dir = cupsDirOpen(".")) == NULL) + { + puts("FAIL (cupsDirOpen failed)"); + status ++; + } + else + { + while ((dent = cupsDirRead(dir)) != NULL) + { + i = strlen(dent->filename) - 2; + + if (i > 0 && dent->filename[i] == '.' && + (dent->filename[i + 1] == 'c' || + dent->filename[i + 1] == 'h')) + load_words(dent->filename, array); + } + + cupsDirClose(dir); + + end = get_seconds(); + + printf("%d words in %.3f seconds (%.0f words/sec), ", cupsArrayCount(array), + end - start, cupsArrayCount(array) / (end - start)); + fflush(stdout); + + for (text = (char *)cupsArrayFirst(array); text;) + { + /* + * Copy this word to the word buffer (safe because we strdup'd from + * the same buffer in the first place... :) + */ + + strcpy(word, text); + + /* + * Grab the next word and compare... + */ + + if ((text = (char *)cupsArrayNext(array)) == NULL) + break; + + if (strcmp(word, text) >= 0) + break; + } + + if (text) + { + printf("FAIL (\"%s\" >= \"%s\"!)\n", word, text); + status ++; + } + else + puts("PASS"); + } + + /* + * Test deleting with iteration... + */ + + fputs("Delete While Iterating: ", stdout); + + text = (char *)cupsArrayFirst(array); + cupsArrayRemove(array, text); + free(text); + + text = (char *)cupsArrayNext(array); + if (!text) + { + puts("FAIL (cupsArrayNext returned NULL!)"); + status ++; + } + else + puts("PASS"); + + /* + * Test save/restore... + */ + + fputs("cupsArraySave: ", stdout); + + for (i = 0, text = (char *)cupsArrayFirst(array); + i < 32; + i ++, text = (char *)cupsArrayNext(array)) + { + saved[i] = text; + + if (!cupsArraySave(array)) + break; + } + + if (i < 32) + printf("FAIL (depth = %d)\n", i); + else + puts("PASS"); + + fputs("cupsArrayRestore: ", stdout); + + while (i > 0) + { + i --; + + text = cupsArrayRestore(array); + if (text != saved[i]) + break; + } + + if (i) + printf("FAIL (depth = %d)\n", i); + else + puts("PASS"); + + /* + * Delete the arrays... + */ + + cupsArrayDelete(array); + cupsArrayDelete(dup_array); + + /* + * Summarize the results and return... + */ + + if (!status) + puts("\nALL TESTS PASSED!"); + else + printf("\n%d TEST(S) FAILED!\n", status); + + return (status); +} + + +/* + * 'get_seconds()' - Get the current time in seconds... + */ + +#ifdef WIN32 +# include + + +static double +get_seconds(void) +{ +} +#else +# include + + +static double +get_seconds(void) +{ + struct timeval curtime; /* Current time */ + + + gettimeofday(&curtime, NULL); + return (curtime.tv_sec + 0.000001 * curtime.tv_usec); +} +#endif /* WIN32 */ + + +/* + * 'load_words()' - Load words from a file. + */ + +static int /* O - 1 on success, 0 on failure */ +load_words(const char *filename, /* I - File to load */ + cups_array_t *array) /* I - Array to add to */ +{ + FILE *fp; /* Test file */ + char word[256]; /* Word from file */ + + + if ((fp = fopen(filename, "r")) == NULL) + { + perror(filename); + return (0); + } + + while (fscanf(fp, "%255s", word) == 1) + { + if (!cupsArrayFind(array, word)) + cupsArrayAdd(array, strdup(word)); + } + + fclose(fp); + + return (1); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/testconflicts.c b/cups/testconflicts.c new file mode 100644 index 0000000000..6f55f0ff42 --- /dev/null +++ b/cups/testconflicts.c @@ -0,0 +1,127 @@ +/* + * "$Id$" + * + * PPD constraint test program for CUPS. + * + * Copyright 2008-2011 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Main entry. + */ + +/* + * Include necessary headers... + */ + +#include "cups.h" +#include "ppd.h" +#include "string-private.h" + + +/* + * 'main()' - Main entry. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + ppd_file_t *ppd; /* PPD file loaded from disk */ + char line[256], /* Input buffer */ + *ptr, /* Pointer into buffer */ + *optr, /* Pointer to first option name */ + *cptr; /* Pointer to first choice */ + int num_options; /* Number of options */ + cups_option_t *options; /* Options */ + char *option, /* Current option */ + *choice; /* Current choice */ + + + if (argc != 2) + { + puts("Usage: testconflicts filename.ppd"); + return (1); + } + + if ((ppd = ppdOpenFile(argv[1])) == NULL) + { + ppd_status_t err; /* Last error in file */ + int linenum; /* Line number in file */ + + err = ppdLastError(&linenum); + + printf("Unable to open PPD file \"%s\": %s on line %d\n", argv[1], + ppdErrorString(err), linenum); + return (1); + } + + ppdMarkDefaults(ppd); + + option = NULL; + choice = NULL; + + for (;;) + { + num_options = 0; + options = NULL; + + if (!cupsResolveConflicts(ppd, option, choice, &num_options, &options)) + puts("Unable to resolve conflicts!"); + else if ((!option && num_options > 0) || (option && num_options > 1)) + { + fputs("Resolved conflicts with the following options:\n ", stdout); + for (i = 0; i < num_options; i ++) + if (!option || _cups_strcasecmp(option, options[i].name)) + printf(" %s=%s", options[i].name, options[i].value); + putchar('\n'); + + cupsFreeOptions(num_options, options); + } + + if (option) + { + free(option); + free(choice); + } + + printf("\nNew Option(s): "); + fflush(stdout); + if (!fgets(line, sizeof(line), stdin) || line[0] == '\n') + break; + + for (ptr = line; isspace(*ptr & 255); ptr ++); + for (optr = ptr; *ptr && *ptr != '='; ptr ++); + if (!*ptr) + break; + for (*ptr++ = '\0', cptr = ptr; *ptr && !isspace(*ptr & 255); ptr ++); + if (!*ptr) + break; + *ptr++ = '\0'; + + option = strdup(optr); + choice = strdup(cptr); + num_options = cupsParseOptions(ptr, 0, &options); + + ppdMarkOption(ppd, option, choice); + if (cupsMarkOptions(ppd, num_options, options)) + puts("Options Conflict!"); + cupsFreeOptions(num_options, options); + } + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/testcups.c b/cups/testcups.c new file mode 100644 index 0000000000..d25a78e55c --- /dev/null +++ b/cups/testcups.c @@ -0,0 +1,472 @@ +/* + * "$Id$" + * + * CUPS API test program for CUPS. + * + * Copyright 2007-2012 by Apple Inc. + * Copyright 2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Main entry. + * dests_equal() - Determine whether two destinations are equal. + */ + +/* + * Include necessary headers... + */ + +#include "string-private.h" +#include "cups.h" +#include "ppd.h" +#include + + +/* + * Local functions... + */ + +static int dests_equal(cups_dest_t *a, cups_dest_t *b); +static int enum_cb(void *user_data, unsigned flags, cups_dest_t *dest); +static void show_diffs(cups_dest_t *a, cups_dest_t *b); + + +/* + * 'main()' - Main entry. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int status = 0, /* Exit status */ + i, /* Looping var */ + num_dests; /* Number of destinations */ + cups_dest_t *dests, /* Destinations */ + *dest, /* Current destination */ + *named_dest; /* Current named destination */ + const char *ppdfile; /* PPD file */ + ppd_file_t *ppd; /* PPD file data */ + int num_jobs; /* Number of jobs for queue */ + cups_job_t *jobs; /* Jobs for queue */ + + if (argc > 1) + { + if (!strcmp(argv[1], "enum")) + { + int msec; /* Timeout in milliseconds */ + + if (argc >= 3) + msec = atoi(argv[2]) * 1000; + else + msec = 0; + + cupsEnumDests(CUPS_DEST_FLAGS_NONE, msec, NULL, 0, 0, enum_cb, NULL); + } + else if (!strcmp(argv[1], "password")) + { + const char *pass = cupsGetPassword("Password:"); + /* Password string */ + + if (pass) + printf("Password entered: %s\n", pass); + else + puts("No password entered."); + } + else if (!strcmp(argv[1], "print") && argc == 5) + { + /* + * ./testcups printer file interval + */ + + int interval, /* Interval between writes */ + job_id; /* Job ID */ + cups_file_t *fp; /* Print file */ + char buffer[16384]; /* Read/write buffer */ + ssize_t bytes; /* Bytes read/written */ + + if ((fp = cupsFileOpen(argv[3], "r")) == NULL) + { + printf("Unable to open \"%s\": %s\n", argv[2], strerror(errno)); + return (1); + } + + if ((job_id = cupsCreateJob(CUPS_HTTP_DEFAULT, argv[2], "testcups", 0, + NULL)) <= 0) + { + printf("Unable to create print job on %s: %s\n", argv[1], + cupsLastErrorString()); + return (1); + } + + interval = atoi(argv[4]); + + if (cupsStartDocument(CUPS_HTTP_DEFAULT, argv[1], job_id, argv[2], + CUPS_FORMAT_AUTO, 1) != HTTP_CONTINUE) + { + puts("Unable to start document!"); + return (1); + } + + while ((bytes = cupsFileRead(fp, buffer, sizeof(buffer))) > 0) + { + printf("Writing %d bytes...\n", (int)bytes); + + if (cupsWriteRequestData(CUPS_HTTP_DEFAULT, buffer, + bytes) != HTTP_CONTINUE) + { + puts("Unable to write bytes!"); + return (1); + } + + if (interval > 0) + sleep(interval); + } + + cupsFileClose(fp); + + if (cupsFinishDocument(CUPS_HTTP_DEFAULT, argv[1]) > IPP_OK_SUBST) + { + puts("Unable to finish document!"); + return (1); + } + } + else + { + puts("Usage:"); + puts(""); + puts("Run basic unit tests:"); + puts(""); + puts(" ./testcups"); + puts(""); + puts("Enumerate printers (for N seconds, -1 for indefinitely):"); + puts(""); + puts(" ./testcups enum [seconds]"); + puts(""); + puts("Ask for a password:"); + puts(""); + puts(" ./testcups password"); + puts(""); + puts("Print a file (interval controls delay between buffers in seconds):"); + puts(""); + puts(" ./testcups print printer file interval"); + return (1); + } + + return (0); + } + + /* + * cupsGetDests() + */ + + fputs("cupsGetDests: ", stdout); + fflush(stdout); + + num_dests = cupsGetDests(&dests); + + if (num_dests == 0) + { + puts("FAIL"); + return (1); + } + else + { + printf("PASS (%d dests)\n", num_dests); + + for (i = num_dests, dest = dests; i > 0; i --, dest ++) + { + printf(" %s", dest->name); + + if (dest->instance) + printf(" /%s", dest->instance); + + if (dest->is_default) + puts(" ***DEFAULT***"); + else + putchar('\n'); + } + } + + /* + * cupsGetDest(NULL) + */ + + fputs("cupsGetDest(NULL): ", stdout); + fflush(stdout); + + if ((dest = cupsGetDest(NULL, NULL, num_dests, dests)) == NULL) + { + for (i = num_dests, dest = dests; i > 0; i --, dest ++) + if (dest->is_default) + break; + + if (i) + { + status = 1; + puts("FAIL"); + } + else + puts("PASS (no default)"); + + dest = NULL; + } + else + printf("PASS (%s)\n", dest->name); + + /* + * cupsGetNamedDest(NULL, NULL, NULL) + */ + + fputs("cupsGetNamedDest(NULL, NULL, NULL): ", stdout); + fflush(stdout); + + if ((named_dest = cupsGetNamedDest(NULL, NULL, NULL)) == NULL || + !dests_equal(dest, named_dest)) + { + if (!dest) + puts("PASS (no default)"); + else if (named_dest) + { + puts("FAIL (different values)"); + show_diffs(dest, named_dest); + status = 1; + } + else + { + puts("FAIL (no default)"); + status = 1; + } + } + else + printf("PASS (%s)\n", named_dest->name); + + if (named_dest) + cupsFreeDests(1, named_dest); + + /* + * cupsGetDest(printer) + */ + + printf("cupsGetDest(\"%s\"): ", dests[num_dests / 2].name); + fflush(stdout); + + if ((dest = cupsGetDest(dests[num_dests / 2].name, NULL, num_dests, + dests)) == NULL) + { + puts("FAIL"); + return (1); + } + else + puts("PASS"); + + /* + * cupsGetNamedDest(NULL, printer, instance) + */ + + printf("cupsGetNamedDest(NULL, \"%s\", \"%s\"): ", dest->name, + dest->instance ? dest->instance : "(null)"); + fflush(stdout); + + if ((named_dest = cupsGetNamedDest(NULL, dest->name, + dest->instance)) == NULL || + !dests_equal(dest, named_dest)) + { + if (named_dest) + { + puts("FAIL (different values)"); + show_diffs(dest, named_dest); + } + else + puts("FAIL (no destination)"); + + + status = 1; + } + else + puts("PASS"); + + if (named_dest) + cupsFreeDests(1, named_dest); + + /* + * cupsPrintFile() + */ + + fputs("cupsPrintFile: ", stdout); + fflush(stdout); + + if (cupsPrintFile(dest->name, "../data/testprint", "Test Page", + dest->num_options, dest->options) <= 0) + { + printf("FAIL (%s)\n", cupsLastErrorString()); + return (1); + } + else + puts("PASS"); + + /* + * cupsGetPPD(printer) + */ + + fputs("cupsGetPPD(): ", stdout); + fflush(stdout); + + if ((ppdfile = cupsGetPPD(dest->name)) == NULL) + { + puts("FAIL"); + } + else + { + puts("PASS"); + + /* + * ppdOpenFile() + */ + + fputs("ppdOpenFile(): ", stdout); + fflush(stdout); + + if ((ppd = ppdOpenFile(ppdfile)) == NULL) + { + puts("FAIL"); + return (1); + } + else + puts("PASS"); + + ppdClose(ppd); + unlink(ppdfile); + } + + /* + * cupsGetJobs() + */ + + fputs("cupsGetJobs: ", stdout); + fflush(stdout); + + num_jobs = cupsGetJobs(&jobs, NULL, 0, -1); + + if (num_jobs == 0) + { + puts("FAIL"); + return (1); + } + else + puts("PASS"); + + cupsFreeJobs(num_jobs, jobs); + cupsFreeDests(num_dests, dests); + + return (status); +} + + +/* + * 'dests_equal()' - Determine whether two destinations are equal. + */ + +static int /* O - 1 if equal, 0 if not equal */ +dests_equal(cups_dest_t *a, /* I - First destination */ + cups_dest_t *b) /* I - Second destination */ +{ + int i; /* Looping var */ + cups_option_t *aoption; /* Current option */ + const char *bval; /* Option value */ + + + if (a == b) + return (1); + + if (!a || !b) + return (0); + + if (_cups_strcasecmp(a->name, b->name) || + (a->instance && !b->instance) || + (!a->instance && b->instance) || + (a->instance && _cups_strcasecmp(a->instance, b->instance)) || + a->num_options != b->num_options) + return (0); + + for (i = a->num_options, aoption = a->options; i > 0; i --, aoption ++) + if ((bval = cupsGetOption(aoption->name, b->num_options, + b->options)) == NULL || + strcmp(aoption->value, bval)) + return (0); + + return (1); +} + + +/* + * 'enum_cb()' - Report additions and removals. + */ + +static int /* O - 1 to continue, 0 to stop */ +enum_cb(void *user_data, /* I - User data (unused) */ + unsigned flags, /* I - Destination flags */ + cups_dest_t *dest) /* I - Destination */ +{ + if (flags & CUPS_DEST_FLAGS_REMOVED) + printf("Removed '%s'.\n", dest->name); + else + printf("Added '%s'.\n", dest->name); + + return (1); +} + + +/* + * 'show_diffs()' - Show differences between two destinations. + */ + +static void +show_diffs(cups_dest_t *a, /* I - First destination */ + cups_dest_t *b) /* I - Second destination */ +{ + int i; /* Looping var */ + cups_option_t *aoption; /* Current option */ + const char *bval; /* Option value */ + + + if (!a || !b) + return; + + puts(" Item cupsGetDest cupsGetNamedDest"); + puts(" -------------------- -------------------- --------------------"); + + if (_cups_strcasecmp(a->name, b->name)) + printf(" name %-20.20s %-20.20s\n", a->name, b->name); + + if ((a->instance && !b->instance) || + (!a->instance && b->instance) || + (a->instance && _cups_strcasecmp(a->instance, b->instance))) + printf(" instance %-20.20s %-20.20s\n", + a->instance ? a->instance : "(null)", + b->instance ? b->instance : "(null)"); + + if (a->num_options != b->num_options) + printf(" num_options %-20d %-20d\n", a->num_options, + b->num_options); + + for (i = a->num_options, aoption = a->options; i > 0; i --, aoption ++) + if ((bval = cupsGetOption(aoption->name, b->num_options, + b->options)) == NULL || + strcmp(aoption->value, bval)) + printf(" %-20.20s %-20.20s %-20.20s\n", aoption->name, + aoption->value, bval ? bval : "(null)"); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/testfile.c b/cups/testfile.c new file mode 100644 index 0000000000..e3b44638bd --- /dev/null +++ b/cups/testfile.c @@ -0,0 +1,821 @@ +/* + * "$Id$" + * + * File test program for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Main entry. + * count_lines() - Count the number of lines in a file. + * random_tests() - Do random access tests. + * read_write_tests() - Perform read/write tests. + */ + +/* + * Include necessary headers... + */ + +#include "string-private.h" +#include "debug-private.h" +#include "file.h" +#include +#include +#ifdef HAVE_LIBZ +# include +#endif /* HAVE_LIBZ */ +#ifdef WIN32 +# include +#else +# include +#endif /* WIN32 */ +#include + + +/* + * Local functions... + */ + +static int count_lines(cups_file_t *fp); +static int random_tests(void); +static int read_write_tests(int compression); + + +/* + * 'main()' - Main entry. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int status; /* Exit status */ + char filename[1024]; /* Filename buffer */ + cups_file_t *fp; /* File pointer */ +#ifndef WIN32 + int fds[2]; /* Open file descriptors */ + cups_file_t *fdfile; /* File opened with cupsFileOpenFd() */ +#endif /* !WIN32 */ + int count; /* Number of lines in file */ + + + if (argc == 1) + { + /* + * Do uncompressed file tests... + */ + + status = read_write_tests(0); + +#ifdef HAVE_LIBZ + /* + * Do compressed file tests... + */ + + putchar('\n'); + + status += read_write_tests(1); +#endif /* HAVE_LIBZ */ + + /* + * Do uncompressed random I/O tests... + */ + + status += random_tests(); + +#ifndef WIN32 + /* + * Test fdopen and close without reading... + */ + + pipe(fds); + close(fds[1]); + + fputs("\ncupsFileOpenFd(fd, \"r\"): ", stdout); + fflush(stdout); + + if ((fdfile = cupsFileOpenFd(fds[0], "r")) == NULL) + { + puts("FAIL"); + status ++; + } + else + { + /* + * Able to open file, now close without reading. If we don't return + * before the alarm fires, that is a failure and we will crash on the + * alarm signal... + */ + + puts("PASS"); + fputs("cupsFileClose(no read): ", stdout); + fflush(stdout); + + alarm(5); + cupsFileClose(fdfile); + alarm(0); + + puts("PASS"); + } +#endif /* !WIN32 */ + + /* + * Count lines in psglyphs, rewind, then count again. + */ + + fputs("\ncupsFileOpen(\"../data/media.defs\", \"r\"): ", stdout); + + if ((fp = cupsFileOpen("../data/media.defs", "r")) == NULL) + { + puts("FAIL"); + status ++; + } + else + { + puts("PASS"); + fputs("cupsFileGets: ", stdout); + + if ((count = count_lines(fp)) != 208) + { + printf("FAIL (got %d lines, expected 208)\n", count); + status ++; + } + else + { + puts("PASS"); + fputs("cupsFileRewind: ", stdout); + + if (cupsFileRewind(fp) != 0) + { + puts("FAIL"); + status ++; + } + else + { + puts("PASS"); + fputs("cupsFileGets: ", stdout); + + if ((count = count_lines(fp)) != 208) + { + printf("FAIL (got %d lines, expected 208)\n", count); + status ++; + } + else + puts("PASS"); + } + } + + cupsFileClose(fp); + } + + /* + * Test path functions... + */ + + fputs("\ncupsFileFind: ", stdout); +#ifdef WIN32 + if (cupsFileFind("notepad.exe", "C:/WINDOWS", 1, filename, sizeof(filename)) && + cupsFileFind("notepad.exe", "C:/WINDOWS;C:/WINDOWS/SYSTEM32", 1, filename, sizeof(filename))) +#else + if (cupsFileFind("cat", "/bin", 1, filename, sizeof(filename)) && + cupsFileFind("cat", "/bin:/usr/bin", 1, filename, sizeof(filename))) +#endif /* WIN32 */ + printf("PASS (%s)\n", filename); + else + { + puts("FAIL"); + status ++; + } + + /* + * Summarize the results and return... + */ + + if (!status) + puts("\nALL TESTS PASSED!"); + else + printf("\n%d TEST(S) FAILED!\n", status); + } + else + { + /* + * Cat the filename on the command-line... + */ + + char line[1024]; /* Line from file */ + + if ((fp = cupsFileOpen(argv[1], "r")) == NULL) + { + perror(argv[1]); + status = 1; + } + else + { + status = 0; + + while (cupsFileGets(fp, line, sizeof(line))) + puts(line); + + if (!cupsFileEOF(fp)) + perror(argv[1]); + + cupsFileClose(fp); + } + } + + return (status); +} + + +/* + * 'count_lines()' - Count the number of lines in a file. + */ + +static int /* O - Number of lines */ +count_lines(cups_file_t *fp) /* I - File to read from */ +{ + int count; /* Number of lines */ + char line[1024]; /* Line buffer */ + + + for (count = 0; cupsFileGets(fp, line, sizeof(line)); count ++); + + return (count); +} + + +/* + * 'random_tests()' - Do random access tests. + */ + +static int /* O - Status */ +random_tests(void) +{ + int status, /* Status of tests */ + pass, /* Current pass */ + count, /* Number of records read */ + record, /* Current record */ + num_records; /* Number of records */ + ssize_t pos, /* Position in file */ + expected; /* Expected position in file */ + cups_file_t *fp; /* File */ + char buffer[512]; /* Data buffer */ + + + /* + * Run 4 passes, each time appending to a data file and then reopening the + * file for reading to validate random records in the file. + */ + + for (status = 0, pass = 0; pass < 4; pass ++) + { + /* + * cupsFileOpen(append) + */ + + printf("\ncupsFileOpen(append %d): ", pass); + + if ((fp = cupsFileOpen("testfile.dat", "a")) == NULL) + { + printf("FAIL (%s)\n", strerror(errno)); + status ++; + break; + } + else + puts("PASS"); + + /* + * cupsFileTell() + */ + + expected = 256 * sizeof(buffer) * pass; + + fputs("cupsFileTell(): ", stdout); + if ((pos = cupsFileTell(fp)) != expected) + { + printf("FAIL (" CUPS_LLFMT " instead of " CUPS_LLFMT ")\n", + CUPS_LLCAST pos, CUPS_LLCAST expected); + status ++; + break; + } + else + puts("PASS"); + + /* + * cupsFileWrite() + */ + + fputs("cupsFileWrite(256 512-byte records): ", stdout); + for (record = 0; record < 256; record ++) + { + memset(buffer, record, sizeof(buffer)); + if (cupsFileWrite(fp, buffer, sizeof(buffer)) < sizeof(buffer)) + break; + } + + if (record < 256) + { + printf("FAIL (%d: %s)\n", record, strerror(errno)); + status ++; + break; + } + else + puts("PASS"); + + /* + * cupsFileTell() + */ + + expected += 256 * sizeof(buffer); + + fputs("cupsFileTell(): ", stdout); + if ((pos = cupsFileTell(fp)) != expected) + { + printf("FAIL (" CUPS_LLFMT " instead of " CUPS_LLFMT ")\n", + CUPS_LLCAST pos, CUPS_LLCAST expected); + status ++; + break; + } + else + puts("PASS"); + + cupsFileClose(fp); + + /* + * cupsFileOpen(read) + */ + + printf("\ncupsFileOpen(read %d): ", pass); + + if ((fp = cupsFileOpen("testfile.dat", "r")) == NULL) + { + printf("FAIL (%s)\n", strerror(errno)); + status ++; + break; + } + else + puts("PASS"); + + /* + * cupsFileSeek, cupsFileRead + */ + + fputs("cupsFileSeek(), cupsFileRead(): ", stdout); + + for (num_records = (pass + 1) * 256, count = (pass + 1) * 256, + record = CUPS_RAND() % num_records; + count > 0; + count --, record = (record + (CUPS_RAND() & 31) - 16 + num_records) % + num_records) + { + /* + * The last record is always the first... + */ + + if (count == 1) + record = 0; + + /* + * Try reading the data for the specified record, and validate the + * contents... + */ + + expected = sizeof(buffer) * record; + + if ((pos = cupsFileSeek(fp, expected)) != expected) + { + printf("FAIL (" CUPS_LLFMT " instead of " CUPS_LLFMT ")\n", + CUPS_LLCAST pos, CUPS_LLCAST expected); + status ++; + break; + } + else + { + if (cupsFileRead(fp, buffer, sizeof(buffer)) != sizeof(buffer)) + { + printf("FAIL (%s)\n", strerror(errno)); + status ++; + break; + } + else if ((buffer[0] & 255) != (record & 255) || + memcmp(buffer, buffer + 1, sizeof(buffer) - 1)) + { + printf("FAIL (Bad Data - %d instead of %d)\n", buffer[0] & 255, + record & 255); + status ++; + break; + } + } + } + + if (count == 0) + puts("PASS"); + + cupsFileClose(fp); + } + + /* + * Remove the test file... + */ + + unlink("testfile.dat"); + + /* + * Return the test status... + */ + + return (status); +} + + +/* + * 'read_write_tests()' - Perform read/write tests. + */ + +static int /* O - Status */ +read_write_tests(int compression) /* I - Use compression? */ +{ + int i; /* Looping var */ + cups_file_t *fp; /* File */ + int status; /* Exit status */ + char line[1024], /* Line from file */ + *value; /* Directive value from line */ + int linenum; /* Line number */ + unsigned char readbuf[8192], /* Read buffer */ + writebuf[8192]; /* Write buffer */ + int byte; /* Byte from file */ + off_t length; /* Length of file */ + static const char *partial_line = "partial line"; + /* Partial line */ + + + /* + * No errors so far... + */ + + status = 0; + + /* + * Initialize the write buffer with random data... + */ + + CUPS_SRAND((unsigned)time(NULL)); + + for (i = 0; i < (int)sizeof(writebuf); i ++) + writebuf[i] = CUPS_RAND(); + + /* + * cupsFileOpen(write) + */ + + printf("cupsFileOpen(write%s): ", compression ? " compressed" : ""); + + fp = cupsFileOpen(compression ? "testfile.dat.gz" : "testfile.dat", + compression ? "w9" : "w"); + if (fp) + { + puts("PASS"); + + /* + * cupsFileCompression() + */ + + fputs("cupsFileCompression(): ", stdout); + + if (cupsFileCompression(fp) == compression) + puts("PASS"); + else + { + printf("FAIL (Got %d, expected %d)\n", cupsFileCompression(fp), + compression); + status ++; + } + + /* + * cupsFilePuts() + */ + + fputs("cupsFilePuts(): ", stdout); + + if (cupsFilePuts(fp, "# Hello, World\n") > 0) + puts("PASS"); + else + { + printf("FAIL (%s)\n", strerror(errno)); + status ++; + } + + /* + * cupsFilePrintf() + */ + + fputs("cupsFilePrintf(): ", stdout); + + for (i = 0; i < 1000; i ++) + if (cupsFilePrintf(fp, "TestLine %03d\n", i) < 0) + break; + + if (i >= 1000) + puts("PASS"); + else + { + printf("FAIL (%s)\n", strerror(errno)); + status ++; + } + + /* + * cupsFilePutChar() + */ + + fputs("cupsFilePutChar(): ", stdout); + + for (i = 0; i < 256; i ++) + if (cupsFilePutChar(fp, i) < 0) + break; + + if (i >= 256) + puts("PASS"); + else + { + printf("FAIL (%s)\n", strerror(errno)); + status ++; + } + + /* + * cupsFileWrite() + */ + + fputs("cupsFileWrite(): ", stdout); + + for (i = 0; i < 10000; i ++) + if (cupsFileWrite(fp, (char *)writebuf, sizeof(writebuf)) < 0) + break; + + if (i >= 10000) + puts("PASS"); + else + { + printf("FAIL (%s)\n", strerror(errno)); + status ++; + } + + /* + * cupsFilePuts() with partial line... + */ + + fputs("cupsFilePuts(\"partial line\"): ", stdout); + + if (cupsFilePuts(fp, partial_line) > 0) + puts("PASS"); + else + { + printf("FAIL (%s)\n", strerror(errno)); + status ++; + } + + /* + * cupsFileTell() + */ + + fputs("cupsFileTell(): ", stdout); + + if ((length = cupsFileTell(fp)) == 81933283) + puts("PASS"); + else + { + printf("FAIL (" CUPS_LLFMT " instead of 81933283)\n", CUPS_LLCAST length); + status ++; + } + + /* + * cupsFileClose() + */ + + fputs("cupsFileClose(): ", stdout); + + if (!cupsFileClose(fp)) + puts("PASS"); + else + { + printf("FAIL (%s)\n", strerror(errno)); + status ++; + } + } + else + { + printf("FAIL (%s)\n", strerror(errno)); + status ++; + } + + /* + * cupsFileOpen(read) + */ + + fputs("\ncupsFileOpen(read): ", stdout); + + fp = cupsFileOpen(compression ? "testfile.dat.gz" : "testfile.dat", "r"); + if (fp) + { + puts("PASS"); + + /* + * cupsFileGets() + */ + + fputs("cupsFileGets(): ", stdout); + + if (cupsFileGets(fp, line, sizeof(line))) + { + if (line[0] == '#') + puts("PASS"); + else + { + printf("FAIL (Got line \"%s\", expected comment line)\n", line); + status ++; + } + } + else + { + printf("FAIL (%s)\n", strerror(errno)); + status ++; + } + + /* + * cupsFileCompression() + */ + + fputs("cupsFileCompression(): ", stdout); + + if (cupsFileCompression(fp) == compression) + puts("PASS"); + else + { + printf("FAIL (Got %d, expected %d)\n", cupsFileCompression(fp), + compression); + status ++; + } + + /* + * cupsFileGetConf() + */ + + linenum = 1; + + fputs("cupsFileGetConf(): ", stdout); + + for (i = 0; i < 1000; i ++) + if (!cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + break; + else if (_cups_strcasecmp(line, "TestLine") || !value || atoi(value) != i || + linenum != (i + 2)) + break; + + if (i >= 1000) + puts("PASS"); + else if (line[0]) + { + printf("FAIL (Line %d, directive \"%s\", value \"%s\")\n", linenum, + line, value ? value : "(null)"); + status ++; + } + else + { + printf("FAIL (%s)\n", strerror(errno)); + status ++; + } + + /* + * cupsFileGetChar() + */ + + fputs("cupsFileGetChar(): ", stdout); + + for (i = 0; i < 256; i ++) + if ((byte = cupsFileGetChar(fp)) != i) + break; + + if (i >= 256) + puts("PASS"); + else if (byte >= 0) + { + printf("FAIL (Got %d, expected %d)\n", byte, i); + status ++; + } + else + { + printf("FAIL (%s)\n", strerror(errno)); + status ++; + } + + /* + * cupsFileRead() + */ + + fputs("cupsFileRead(): ", stdout); + + for (i = 0; i < 10000; i ++) + if ((byte = cupsFileRead(fp, (char *)readbuf, sizeof(readbuf))) < 0) + break; + else if (memcmp(readbuf, writebuf, sizeof(readbuf))) + break; + + if (i >= 10000) + puts("PASS"); + else if (byte > 0) + { + printf("FAIL (Pass %d, ", i); + + for (i = 0; i < (int)sizeof(readbuf); i ++) + if (readbuf[i] != writebuf[i]) + break; + + printf("match failed at offset %d - got %02X, expected %02X)\n", + i, readbuf[i], writebuf[i]); + } + else + { + printf("FAIL (%s)\n", strerror(errno)); + status ++; + } + + /* + * cupsFileGetChar() with partial line... + */ + + fputs("cupsFileGetChar(partial line): ", stdout); + + for (i = 0; i < (int)strlen(partial_line); i ++) + if ((byte = cupsFileGetChar(fp)) < 0) + break; + else if (byte != partial_line[i]) + break; + + if (!partial_line[i]) + puts("PASS"); + else + { + printf("FAIL (got '%c', expected '%c')\n", byte, partial_line[i]); + status ++; + } + + /* + * cupsFileTell() + */ + + fputs("cupsFileTell(): ", stdout); + + if ((length = cupsFileTell(fp)) == 81933283) + puts("PASS"); + else + { + printf("FAIL (" CUPS_LLFMT " instead of 81933283)\n", CUPS_LLCAST length); + status ++; + } + + /* + * cupsFileClose() + */ + + fputs("cupsFileClose(): ", stdout); + + if (!cupsFileClose(fp)) + puts("PASS"); + else + { + printf("FAIL (%s)\n", strerror(errno)); + status ++; + } + } + else + { + printf("FAIL (%s)\n", strerror(errno)); + status ++; + } + + /* + * Remove the test file... + */ + + unlink(compression ? "testfile.dat.gz" : "testfile.dat"); + + /* + * Return the test status... + */ + + return (status); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/testhttp.c b/cups/testhttp.c new file mode 100644 index 0000000000..bca5f9a772 --- /dev/null +++ b/cups/testhttp.c @@ -0,0 +1,600 @@ +/* + * "$Id$" + * + * HTTP test program for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Main entry. + */ + +/* + * Include necessary headers... + */ + +#include "string-private.h" +#include "http-private.h" + + +/* + * Types and structures... + */ + +typedef struct uri_test_s /**** URI test cases ****/ +{ + http_uri_status_t result; /* Expected return value */ + const char *uri, /* URI */ + *scheme, /* Scheme string */ + *username, /* Username:password string */ + *hostname, /* Hostname string */ + *resource; /* Resource string */ + int port, /* Port number */ + assemble_port; /* Port number for httpAssembleURI() */ +} uri_test_t; + + +/* + * Local globals... + */ + +static uri_test_t uri_tests[] = /* URI test data */ + { + /* Start with valid URIs */ + { HTTP_URI_OK, "file:/filename", + "file", "", "", "/filename", 0, 0 }, + { HTTP_URI_OK, "file:/filename%20with%20spaces", + "file", "", "", "/filename with spaces", 0, 0 }, + { HTTP_URI_OK, "file:///filename", + "file", "", "", "/filename", 0, 0 }, + { HTTP_URI_OK, "file:///filename%20with%20spaces", + "file", "", "", "/filename with spaces", 0, 0 }, + { HTTP_URI_OK, "file://localhost/filename", + "file", "", "localhost", "/filename", 0, 0 }, + { HTTP_URI_OK, "file://localhost/filename%20with%20spaces", + "file", "", "localhost", "/filename with spaces", 0, 0 }, + { HTTP_URI_OK, "http://server/", + "http", "", "server", "/", 80, 0 }, + { HTTP_URI_OK, "http://username@server/", + "http", "username", "server", "/", 80, 0 }, + { HTTP_URI_OK, "http://username:passwor%64@server/", + "http", "username:password", "server", "/", 80, 0 }, + { HTTP_URI_OK, "http://username:passwor%64@server:8080/", + "http", "username:password", "server", "/", 8080, 8080 }, + { HTTP_URI_OK, "http://username:passwor%64@server:8080/directory/filename", + "http", "username:password", "server", "/directory/filename", 8080, 8080 }, + { HTTP_URI_OK, "http://[2000::10:100]:631/ipp", + "http", "", "2000::10:100", "/ipp", 631, 631 }, + { HTTP_URI_OK, "https://username:passwor%64@server/directory/filename", + "https", "username:password", "server", "/directory/filename", 443, 0 }, + { HTTP_URI_OK, "ipp://username:passwor%64@[::1]/ipp", + "ipp", "username:password", "::1", "/ipp", 631, 0 }, + { HTTP_URI_OK, "lpd://server/queue?reserve=yes", + "lpd", "", "server", "/queue?reserve=yes", 515, 0 }, + { HTTP_URI_OK, "mailto:user@domain.com", + "mailto", "", "", "user@domain.com", 0, 0 }, + { HTTP_URI_OK, "socket://server/", + "socket", "", "server", "/", 9100, 0 }, + { HTTP_URI_OK, "socket://192.168.1.1:9101/", + "socket", "", "192.168.1.1", "/", 9101, 9101 }, + { HTTP_URI_OK, "ipp://username:password@[v1.fe80::200:1234:5678:9abc+eth0]:999/ipp", + "ipp", "username:password", "fe80::200:1234:5678:9abc%eth0", "/ipp", 999, 999 }, + { HTTP_URI_OK, "http://server/admin?DEVICE_URI=usb://HP/Photosmart%25202600%2520series?serial=MY53OK70V10400", + "http", "", "server", "/admin?DEVICE_URI=usb://HP/Photosmart%25202600%2520series?serial=MY53OK70V10400", 80, 0 }, + { HTTP_URI_OK, "lpd://Acme%20Laser%20(01%3A23%3A45).local._tcp._printer/", + "lpd", "", "Acme Laser (01:23:45).local._tcp._printer", "/", 515, 0 }, + { HTTP_URI_OK, "ipp://HP%20Officejet%204500%20G510n-z%20%40%20Will's%20MacBook%20Pro%2015%22._ipp._tcp.local./", + "ipp", "", "HP Officejet 4500 G510n-z @ Will's MacBook Pro 15\"._ipp._tcp.local.", "/", 631, 0 }, + + /* Missing scheme */ + { HTTP_URI_MISSING_SCHEME, "/path/to/file/index.html", + "file", "", "", "/path/to/file/index.html", 0, 0 }, + { HTTP_URI_MISSING_SCHEME, "//server/ipp", + "ipp", "", "server", "/ipp", 631, 0 }, + + /* Unknown scheme */ + { HTTP_URI_UNKNOWN_SCHEME, "vendor://server/resource", + "vendor", "", "server", "/resource", 0, 0 }, + + /* Missing resource */ + { HTTP_URI_MISSING_RESOURCE, "socket://[::192.168.2.1]", + "socket", "", "::192.168.2.1", "/", 9100, 0 }, + { HTTP_URI_MISSING_RESOURCE, "socket://192.168.1.1:9101", + "socket", "", "192.168.1.1", "/", 9101 }, + + /* Bad URI */ + { HTTP_URI_BAD_URI, "", + "", "", "", "", 0, 0 }, + + /* Bad scheme */ + { HTTP_URI_BAD_SCHEME, "bad_scheme://server/resource", + "", "", "", "", 0, 0 }, + + /* Bad username */ + { HTTP_URI_BAD_USERNAME, "http://username:passwor%6@server/resource", + "http", "", "", "", 80, 0 }, + + /* Bad hostname */ + { HTTP_URI_BAD_HOSTNAME, "http://[/::1]/index.html", + "http", "", "", "", 80, 0 }, + { HTTP_URI_BAD_HOSTNAME, "http://[", + "http", "", "", "", 80, 0 }, + { HTTP_URI_BAD_HOSTNAME, "http://serve%7/index.html", + "http", "", "", "", 80, 0 }, + + /* Bad port number */ + { HTTP_URI_BAD_PORT, "http://127.0.0.1:9999a/index.html", + "http", "", "127.0.0.1", "", 0, 0 }, + + /* Bad resource */ + { HTTP_URI_BAD_RESOURCE, "http://server/index.html%", + "http", "", "server", "", 80, 0 } + }; +static const char * const base64_tests[][2] = + { + { "A", "QQ==" }, + /* 010000 01 */ + { "AB", "QUI=" }, + /* 010000 010100 0010 */ + { "ABC", "QUJD" }, + /* 010000 010100 001001 000011 */ + { "ABCD", "QUJDRA==" }, + /* 010000 010100 001001 000011 010001 00 */ + { "ABCDE", "QUJDREU=" }, + /* 010000 010100 001001 000011 010001 000100 0101 */ + { "ABCDEF", "QUJDREVG" }, + /* 010000 010100 001001 000011 010001 000100 010101 000110 */ + }; + + +/* + * 'main()' - Main entry. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i, j, k; /* Looping vars */ + http_t *http; /* HTTP connection */ + http_encryption_t encryption; /* Encryption type */ + http_status_t status; /* Status of GET command */ + int failures; /* Number of test failures */ + char buffer[8192]; /* Input buffer */ + long bytes; /* Number of bytes read */ + FILE *out; /* Output file */ + char encode[256], /* Base64-encoded string */ + decode[256]; /* Base64-decoded string */ + int decodelen; /* Length of decoded string */ + char scheme[HTTP_MAX_URI], /* Scheme from URI */ + hostname[HTTP_MAX_URI], /* Hostname from URI */ + username[HTTP_MAX_URI], /* Username:password from URI */ + resource[HTTP_MAX_URI]; /* Resource from URI */ + int port; /* Port number from URI */ + http_uri_status_t uri_status; /* Status of URI separation */ + http_addrlist_t *addrlist, /* Address list */ + *addr; /* Current address */ + off_t length, total; /* Length and total bytes */ + time_t start, current; /* Start and end time */ + static const char * const uri_status_strings[] = + { + "HTTP_URI_OVERFLOW", + "HTTP_URI_BAD_ARGUMENTS", + "HTTP_URI_BAD_RESOURCE", + "HTTP_URI_BAD_PORT", + "HTTP_URI_BAD_HOSTNAME", + "HTTP_URI_BAD_USERNAME", + "HTTP_URI_BAD_SCHEME", + "HTTP_URI_BAD_URI", + "HTTP_URI_OK", + "HTTP_URI_MISSING_SCHEME", + "HTTP_URI_UNKNOWN_SCHEME", + "HTTP_URI_MISSING_RESOURCE" + }; + + + /* + * Do API tests if we don't have a URL on the command-line... + */ + + if (argc == 1) + { + failures = 0; + + /* + * httpGetDateString()/httpGetDateTime() + */ + + fputs("httpGetDateString()/httpGetDateTime(): ", stdout); + + start = time(NULL); + strcpy(buffer, httpGetDateString(start)); + current = httpGetDateTime(buffer); + + i = (int)(current - start); + if (i < 0) + i = -i; + + if (!i) + puts("PASS"); + else + { + failures ++; + puts("FAIL"); + printf(" Difference is %d seconds, %02d:%02d:%02d...\n", i, i / 3600, + (i / 60) % 60, i % 60); + printf(" httpGetDateString(%d) returned \"%s\"\n", (int)start, buffer); + printf(" httpGetDateTime(\"%s\") returned %d\n", buffer, (int)current); + printf(" httpGetDateString(%d) returned \"%s\"\n", (int)current, + httpGetDateString(current)); + } + + /* + * httpDecode64_2()/httpEncode64_2() + */ + + fputs("httpDecode64_2()/httpEncode64_2(): ", stdout); + + for (i = 0, j = 0; i < (int)(sizeof(base64_tests) / sizeof(base64_tests[0])); i ++) + { + httpEncode64_2(encode, sizeof(encode), base64_tests[i][0], + (int)strlen(base64_tests[i][0])); + decodelen = (int)sizeof(decode); + httpDecode64_2(decode, &decodelen, base64_tests[i][1]); + + if (strcmp(decode, base64_tests[i][0])) + { + failures ++; + + if (j) + { + puts("FAIL"); + j = 1; + } + + printf(" httpDecode64_2() returned \"%s\", expected \"%s\"...\n", + decode, base64_tests[i][0]); + } + + if (strcmp(encode, base64_tests[i][1])) + { + failures ++; + + if (j) + { + puts("FAIL"); + j = 1; + } + + printf(" httpEncode64_2() returned \"%s\", expected \"%s\"...\n", + encode, base64_tests[i][1]); + } + } + + if (!j) + puts("PASS"); + + /* + * httpGetHostname() + */ + + fputs("httpGetHostname(): ", stdout); + + if (httpGetHostname(NULL, hostname, sizeof(hostname))) + printf("PASS (%s)\n", hostname); + else + { + failures ++; + puts("FAIL"); + } + + /* + * httpAddrGetList() + */ + + printf("httpAddrGetList(%s): ", hostname); + + addrlist = httpAddrGetList(hostname, AF_UNSPEC, NULL); + if (addrlist) + { + for (i = 0, addr = addrlist; addr; i ++, addr = addr->next) + { + char numeric[1024]; /* Numeric IP address */ + + + httpAddrString(&(addr->addr), numeric, sizeof(numeric)); + if (!strcmp(numeric, "UNKNOWN")) + break; + } + + if (addr) + printf("FAIL (bad address for %s)\n", hostname); + else + printf("PASS (%d address(es) for %s)\n", i, hostname); + + httpAddrFreeList(addrlist); + } + else if (isdigit(hostname[0] & 255)) + { + puts("FAIL (ignored because hostname is numeric)"); + } + else + { + failures ++; + puts("FAIL"); + } + + /* + * Test httpSeparateURI()... + */ + + fputs("httpSeparateURI(): ", stdout); + for (i = 0, j = 0; i < (int)(sizeof(uri_tests) / sizeof(uri_tests[0])); i ++) + { + uri_status = httpSeparateURI(HTTP_URI_CODING_MOST, + uri_tests[i].uri, scheme, sizeof(scheme), + username, sizeof(username), + hostname, sizeof(hostname), &port, + resource, sizeof(resource)); + if (uri_status != uri_tests[i].result || + strcmp(scheme, uri_tests[i].scheme) || + strcmp(username, uri_tests[i].username) || + strcmp(hostname, uri_tests[i].hostname) || + port != uri_tests[i].port || + strcmp(resource, uri_tests[i].resource)) + { + failures ++; + + if (!j) + { + puts("FAIL"); + j = 1; + } + + printf(" \"%s\":\n", uri_tests[i].uri); + + if (uri_status != uri_tests[i].result) + printf(" Returned %s instead of %s\n", + uri_status_strings[uri_status + 8], + uri_status_strings[uri_tests[i].result + 8]); + + if (strcmp(scheme, uri_tests[i].scheme)) + printf(" Scheme \"%s\" instead of \"%s\"\n", + scheme, uri_tests[i].scheme); + + if (strcmp(username, uri_tests[i].username)) + printf(" Username \"%s\" instead of \"%s\"\n", + username, uri_tests[i].username); + + if (strcmp(hostname, uri_tests[i].hostname)) + printf(" Hostname \"%s\" instead of \"%s\"\n", + hostname, uri_tests[i].hostname); + + if (port != uri_tests[i].port) + printf(" Port %d instead of %d\n", + port, uri_tests[i].port); + + if (strcmp(resource, uri_tests[i].resource)) + printf(" Resource \"%s\" instead of \"%s\"\n", + resource, uri_tests[i].resource); + } + } + + if (!j) + printf("PASS (%d URIs tested)\n", + (int)(sizeof(uri_tests) / sizeof(uri_tests[0]))); + + /* + * Test httpAssembleURI()... + */ + + fputs("httpAssembleURI(): ", stdout); + for (i = 0, j = 0, k = 0; + i < (int)(sizeof(uri_tests) / sizeof(uri_tests[0])); + i ++) + if (uri_tests[i].result == HTTP_URI_OK && + !strstr(uri_tests[i].uri, "%64") && + strstr(uri_tests[i].uri, "//")) + { + k ++; + uri_status = httpAssembleURI(HTTP_URI_CODING_MOST, + buffer, sizeof(buffer), + uri_tests[i].scheme, + uri_tests[i].username, + uri_tests[i].hostname, + uri_tests[i].assemble_port, + uri_tests[i].resource); + + if (uri_status != HTTP_URI_OK) + { + failures ++; + + if (!j) + { + puts("FAIL"); + j = 1; + } + + printf(" \"%s\": %s\n", uri_tests[i].uri, + uri_status_strings[uri_status + 8]); + } + else if (strcmp(buffer, uri_tests[i].uri)) + { + failures ++; + + if (!j) + { + puts("FAIL"); + j = 1; + } + + printf(" \"%s\": assembled = \"%s\"\n", uri_tests[i].uri, + buffer); + } + } + + if (!j) + printf("PASS (%d URIs tested)\n", k); + + /* + * Show a summary and return... + */ + + if (failures) + printf("\n%d TESTS FAILED!\n", failures); + else + puts("\nALL TESTS PASSED!"); + + return (failures); + } + else if (strstr(argv[1], "._tcp")) + { + /* + * Test resolving an mDNS name. + */ + + char resolved[1024]; /* Resolved URI */ + + + printf("_httpResolveURI(%s, _HTTP_RESOLVE_DEFAULT): ", argv[1]); + fflush(stdout); + + if (!_httpResolveURI(argv[1], resolved, sizeof(resolved), + _HTTP_RESOLVE_DEFAULT, NULL, NULL)) + { + puts("FAIL"); + return (1); + } + else + printf("PASS (%s)\n", resolved); + + printf("_httpResolveURI(%s, _HTTP_RESOLVE_FQDN): ", argv[1]); + fflush(stdout); + + if (!_httpResolveURI(argv[1], resolved, sizeof(resolved), + _HTTP_RESOLVE_FQDN, NULL, NULL)) + { + puts("FAIL"); + return (1); + } + else if (strstr(resolved, ".local:")) + { + printf("FAIL (%s)\n", resolved); + return (1); + } + else + { + printf("PASS (%s)\n", resolved); + return (0); + } + } + else if (!strcmp(argv[1], "-u") && argc == 3) + { + /* + * Test URI separation... + */ + + uri_status = httpSeparateURI(HTTP_URI_CODING_ALL, argv[2], scheme, + sizeof(scheme), username, sizeof(username), + hostname, sizeof(hostname), &port, + resource, sizeof(resource)); + printf("uri_status = %s\n", uri_status_strings[uri_status + 8]); + printf("scheme = \"%s\"\n", scheme); + printf("username = \"%s\"\n", username); + printf("hostname = \"%s\"\n", hostname); + printf("port = %d\n", port); + printf("resource = \"%s\"\n", resource); + + return (0); + } + + /* + * Test HTTP GET requests... + */ + + http = NULL; + out = stdout; + + for (i = 1; i < argc; i ++) + { + if (!strcmp(argv[i], "-o")) + { + i ++; + if (i >= argc) + break; + + out = fopen(argv[i], "wb"); + continue; + } + + httpSeparateURI(HTTP_URI_CODING_MOST, argv[i], scheme, sizeof(scheme), + username, sizeof(username), + hostname, sizeof(hostname), &port, + resource, sizeof(resource)); + + if (!_cups_strcasecmp(scheme, "https") || !_cups_strcasecmp(scheme, "ipps") || + port == 443) + encryption = HTTP_ENCRYPT_ALWAYS; + else + encryption = HTTP_ENCRYPT_IF_REQUESTED; + + http = httpConnectEncrypt(hostname, port, encryption); + if (http == NULL) + { + perror(hostname); + continue; + } + printf("Requesting file \"%s\"...\n", resource); + httpClearFields(http); + httpSetField(http, HTTP_FIELD_ACCEPT_LANGUAGE, "en"); + httpGet(http, resource); + while ((status = httpUpdate(http)) == HTTP_CONTINUE); + + if (status == HTTP_OK) + puts("GET OK:"); + else + printf("GET failed with status %d...\n", status); + + start = time(NULL); + length = httpGetLength2(http); + total = 0; + + while ((bytes = httpRead2(http, buffer, sizeof(buffer))) > 0) + { + total += bytes; + fwrite(buffer, bytes, 1, out); + if (out != stdout) + { + current = time(NULL); + if (current == start) current ++; + printf("\r" CUPS_LLFMT "/" CUPS_LLFMT " bytes (" + CUPS_LLFMT " bytes/sec) ", CUPS_LLCAST total, + CUPS_LLCAST length, CUPS_LLCAST (total / (current - start))); + fflush(stdout); + } + } + } + + puts("Closing connection to server..."); + httpClose(http); + + if (out != stdout) + fclose(out); + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/testi18n.c b/cups/testi18n.c new file mode 100644 index 0000000000..2e626715f7 --- /dev/null +++ b/cups/testi18n.c @@ -0,0 +1,619 @@ +/* + * "$Id$" + * + * Internationalization test for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Main entry for internationalization test module. + * print_utf8() - Print UTF-8 string with (optional) message. + */ + +/* + * Include necessary headers... + */ + +#include "string-private.h" +#include "language-private.h" +#include +#include +#include + + +/* + * Local globals... + */ + +static const char * const lang_encodings[] = + { /* Encoding strings */ + "us-ascii", "iso-8859-1", + "iso-8859-2", "iso-8859-3", + "iso-8859-4", "iso-8859-5", + "iso-8859-6", "iso-8859-7", + "iso-8859-8", "iso-8859-9", + "iso-8859-10", "utf-8", + "iso-8859-13", "iso-8859-14", + "iso-8859-15", "windows-874", + "windows-1250", "windows-1251", + "windows-1252", "windows-1253", + "windows-1254", "windows-1255", + "windows-1256", "windows-1257", + "windows-1258", "koi8-r", + "koi8-u", "iso-8859-11", + "iso-8859-16", "mac-roman", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "windows-932", "windows-936", + "windows-949", "windows-950", + "windows-1361", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "unknown", "unknown", + "euc-cn", "euc-jp", + "euc-kr", "euc-tw", + "jis-x0213" + }; + + +/* + * Local functions... + */ + +static void print_utf8(const char *msg, const cups_utf8_t *src); + + +/* + * 'main()' - Main entry for internationalization test module. + */ + +int /* O - Exit code */ +main(int argc, /* I - Argument Count */ + char *argv[]) /* I - Arguments */ +{ + FILE *fp; /* File pointer */ + int count; /* File line counter */ + int status, /* Status of current test */ + errors; /* Error count */ + char line[1024]; /* File line source string */ + int len; /* Length (count) of string */ + char legsrc[1024], /* Legacy source string */ + legdest[1024], /* Legacy destination string */ + *legptr; /* Pointer into legacy string */ + cups_utf8_t utf8latin[] = /* UTF-8 Latin-1 source */ + { 0x41, 0x20, 0x21, 0x3D, 0x20, 0xC3, 0x84, 0x2E, 0x00 }; + /* "A != ." - use ISO 8859-1 */ + cups_utf8_t utf8repla[] = /* UTF-8 Latin-1 replacement */ + { 0x41, 0x20, 0xE2, 0x89, 0xA2, 0x20, 0xC3, 0x84, 0x2E, 0x00 }; + /* "A ." */ + cups_utf8_t utf8greek[] = /* UTF-8 Greek source string */ + { 0x41, 0x20, 0x21, 0x3D, 0x20, 0xCE, 0x91, 0x2E, 0x00 }; + /* "A != ." - use ISO 8859-7 */ + cups_utf8_t utf8japan[] = /* UTF-8 Japanese source */ + { 0x41, 0x20, 0x21, 0x3D, 0x20, 0xEE, 0x9C, 0x80, 0x2E, 0x00 }; + /* "A != ." - use Windows 932 or EUC-JP */ + cups_utf8_t utf8taiwan[] = /* UTF-8 Chinese source */ + { 0x41, 0x20, 0x21, 0x3D, 0x20, 0xE4, 0xB9, 0x82, 0x2E, 0x00 }; + /* "A != ." - use Windows 950 (Big5) or EUC-TW */ + cups_utf8_t utf8dest[1024]; /* UTF-8 destination string */ + cups_utf32_t utf32dest[1024]; /* UTF-32 destination string */ + + + if (argc > 1) + { + int i; /* Looping var */ + cups_encoding_t encoding; /* Source encoding */ + + + if (argc != 3) + { + puts("Usage: ./testi18n [filename charset]"); + return (1); + } + + if ((fp = fopen(argv[1], "rb")) == NULL) + { + perror(argv[1]); + return (1); + } + + for (i = 0, encoding = CUPS_AUTO_ENCODING; + i < (int)(sizeof(lang_encodings) / sizeof(lang_encodings[0])); + i ++) + if (!_cups_strcasecmp(lang_encodings[i], argv[2])) + { + encoding = (cups_encoding_t)i; + break; + } + + if (encoding == CUPS_AUTO_ENCODING) + { + fprintf(stderr, "%s: Unknown character set!\n", argv[2]); + return (1); + } + + while (fgets(line, sizeof(line), fp)) + { + if (cupsCharsetToUTF8(utf8dest, line, sizeof(utf8dest), encoding) < 0) + { + fprintf(stderr, "%s: Unable to convert line: %s", argv[1], line); + return (1); + } + + fputs((char *)utf8dest, stdout); + } + + fclose(fp); + return (0); + } + + /* + * Start with some conversion tests from a UTF-8 test file. + */ + + errors = 0; + + if ((fp = fopen("utf8demo.txt", "rb")) == NULL) + { + perror("utf8demo.txt"); + return (1); + } + + /* + * cupsUTF8ToUTF32 + */ + + fputs("cupsUTF8ToUTF32 of utfdemo.txt: ", stdout); + + for (count = 0, status = 0; fgets(line, sizeof(line), fp);) + { + count ++; + + if (cupsUTF8ToUTF32(utf32dest, (cups_utf8_t *)line, 1024) < 0) + { + printf("FAIL (UTF-8 to UTF-32 on line %d)\n", count); + errors ++; + status = 1; + break; + } + } + + if (!status) + puts("PASS"); + + /* + * cupsUTF8ToCharset(CUPS_EUC_JP) + */ + + fputs("cupsUTF8ToCharset(CUPS_EUC_JP) of utfdemo.txt: ", stdout); + + rewind(fp); + + for (count = 0, status = 0; fgets(line, sizeof(line), fp);) + { + count ++; + + len = cupsUTF8ToCharset(legdest, (cups_utf8_t *)line, 1024, CUPS_EUC_JP); + if (len < 0) + { + printf("FAIL (UTF-8 to EUC-JP on line %d)\n", count); + errors ++; + status = 1; + break; + } + } + + if (!status) + puts("PASS"); + + fclose(fp); + + /* + * Test UTF-8 to legacy charset (ISO 8859-1)... + */ + + fputs("cupsUTF8ToCharset(CUPS_ISO8859_1): ", stdout); + + legdest[0] = 0; + + len = cupsUTF8ToCharset(legdest, utf8latin, 1024, CUPS_ISO8859_1); + if (len < 0) + { + printf("FAIL (len=%d)\n", len); + errors ++; + } + else + puts("PASS"); + + /* + * cupsCharsetToUTF8 + */ + + fputs("cupsCharsetToUTF8(CUPS_ISO8859_1): ", stdout); + + strcpy(legsrc, legdest); + + len = cupsCharsetToUTF8(utf8dest, legsrc, 1024, CUPS_ISO8859_1); + if (len != strlen((char *)utf8latin)) + { + printf("FAIL (len=%d, expected %d)\n", len, (int)strlen((char *)utf8latin)); + print_utf8(" utf8latin", utf8latin); + print_utf8(" utf8dest", utf8dest); + errors ++; + } + else if (memcmp(utf8latin, utf8dest, len)) + { + puts("FAIL (results do not match)"); + print_utf8(" utf8latin", utf8latin); + print_utf8(" utf8dest", utf8dest); + errors ++; + } + else if (cupsUTF8ToCharset(legdest, utf8repla, 1024, CUPS_ISO8859_1) < 0) + { + puts("FAIL (replacement characters do not work!)"); + errors ++; + } + else + puts("PASS"); + + /* + * Test UTF-8 to/from legacy charset (ISO 8859-7)... + */ + + fputs("cupsUTF8ToCharset(CUPS_ISO8859_7): ", stdout); + + if (cupsUTF8ToCharset(legdest, utf8greek, 1024, CUPS_ISO8859_7) < 0) + { + puts("FAIL"); + errors ++; + } + else + { + for (legptr = legdest; *legptr && *legptr != '?'; legptr ++); + + if (*legptr) + { + puts("FAIL (unknown character)"); + errors ++; + } + else + puts("PASS"); + } + + fputs("cupsCharsetToUTF8(CUPS_ISO8859_7): ", stdout); + + strcpy(legsrc, legdest); + + len = cupsCharsetToUTF8(utf8dest, legsrc, 1024, CUPS_ISO8859_7); + if (len != strlen((char *)utf8greek)) + { + printf("FAIL (len=%d, expected %d)\n", len, (int)strlen((char *)utf8greek)); + print_utf8(" utf8greek", utf8greek); + print_utf8(" utf8dest", utf8dest); + errors ++; + } + else if (memcmp(utf8greek, utf8dest, len)) + { + puts("FAIL (results do not match)"); + print_utf8(" utf8greek", utf8greek); + print_utf8(" utf8dest", utf8dest); + errors ++; + } + else + puts("PASS"); + + /* + * Test UTF-8 to/from legacy charset (Windows 932)... + */ + + fputs("cupsUTF8ToCharset(CUPS_WINDOWS_932): ", stdout); + + if (cupsUTF8ToCharset(legdest, utf8japan, 1024, CUPS_WINDOWS_932) < 0) + { + puts("FAIL"); + errors ++; + } + else + { + for (legptr = legdest; *legptr && *legptr != '?'; legptr ++); + + if (*legptr) + { + puts("FAIL (unknown character)"); + errors ++; + } + else + puts("PASS"); + } + + fputs("cupsCharsetToUTF8(CUPS_WINDOWS_932): ", stdout); + + strcpy(legsrc, legdest); + + len = cupsCharsetToUTF8(utf8dest, legsrc, 1024, CUPS_WINDOWS_932); + if (len != strlen((char *)utf8japan)) + { + printf("FAIL (len=%d, expected %d)\n", len, (int)strlen((char *)utf8japan)); + print_utf8(" utf8japan", utf8japan); + print_utf8(" utf8dest", utf8dest); + errors ++; + } + else if (memcmp(utf8japan, utf8dest, len)) + { + puts("FAIL (results do not match)"); + print_utf8(" utf8japan", utf8japan); + print_utf8(" utf8dest", utf8dest); + errors ++; + } + else + puts("PASS"); + + /* + * Test UTF-8 to/from legacy charset (EUC-JP)... + */ + + fputs("cupsUTF8ToCharset(CUPS_EUC_JP): ", stdout); + + if (cupsUTF8ToCharset(legdest, utf8japan, 1024, CUPS_EUC_JP) < 0) + { + puts("FAIL"); + errors ++; + } + else + { + for (legptr = legdest; *legptr && *legptr != '?'; legptr ++); + + if (*legptr) + { + puts("FAIL (unknown character)"); + errors ++; + } + else + puts("PASS"); + } + +#ifndef __linux + fputs("cupsCharsetToUTF8(CUPS_EUC_JP): ", stdout); + + strcpy(legsrc, legdest); + + len = cupsCharsetToUTF8(utf8dest, legsrc, 1024, CUPS_EUC_JP); + if (len != strlen((char *)utf8japan)) + { + printf("FAIL (len=%d, expected %d)\n", len, (int)strlen((char *)utf8japan)); + print_utf8(" utf8japan", utf8japan); + print_utf8(" utf8dest", utf8dest); + errors ++; + } + else if (memcmp(utf8japan, utf8dest, len)) + { + puts("FAIL (results do not match)"); + print_utf8(" utf8japan", utf8japan); + print_utf8(" utf8dest", utf8dest); + errors ++; + } + else + puts("PASS"); +#endif /* !__linux */ + + /* + * Test UTF-8 to/from legacy charset (Windows 950)... + */ + + fputs("cupsUTF8ToCharset(CUPS_WINDOWS_950): ", stdout); + + if (cupsUTF8ToCharset(legdest, utf8taiwan, 1024, CUPS_WINDOWS_950) < 0) + { + puts("FAIL"); + errors ++; + } + else + { + for (legptr = legdest; *legptr && *legptr != '?'; legptr ++); + + if (*legptr) + { + puts("FAIL (unknown character)"); + errors ++; + } + else + puts("PASS"); + } + + fputs("cupsCharsetToUTF8(CUPS_WINDOWS_950): ", stdout); + + strcpy(legsrc, legdest); + + len = cupsCharsetToUTF8(utf8dest, legsrc, 1024, CUPS_WINDOWS_950); + if (len != strlen((char *)utf8taiwan)) + { + printf("FAIL (len=%d, expected %d)\n", len, (int)strlen((char *)utf8taiwan)); + print_utf8(" utf8taiwan", utf8taiwan); + print_utf8(" utf8dest", utf8dest); + errors ++; + } + else if (memcmp(utf8taiwan, utf8dest, len)) + { + puts("FAIL (results do not match)"); + print_utf8(" utf8taiwan", utf8taiwan); + print_utf8(" utf8dest", utf8dest); + errors ++; + } + else + puts("PASS"); + + /* + * Test UTF-8 to/from legacy charset (EUC-TW)... + */ + + fputs("cupsUTF8ToCharset(CUPS_EUC_TW): ", stdout); + + if (cupsUTF8ToCharset(legdest, utf8taiwan, 1024, CUPS_EUC_TW) < 0) + { + puts("FAIL"); + errors ++; + } + else + { + for (legptr = legdest; *legptr && *legptr != '?'; legptr ++); + + if (*legptr) + { + puts("FAIL (unknown character)"); + errors ++; + } + else + puts("PASS"); + } + + fputs("cupsCharsetToUTF8(CUPS_EUC_TW): ", stdout); + + strcpy(legsrc, legdest); + + len = cupsCharsetToUTF8(utf8dest, legsrc, 1024, CUPS_EUC_TW); + if (len != strlen((char *)utf8taiwan)) + { + printf("FAIL (len=%d, expected %d)\n", len, (int)strlen((char *)utf8taiwan)); + print_utf8(" utf8taiwan", utf8taiwan); + print_utf8(" utf8dest", utf8dest); + errors ++; + } + else if (memcmp(utf8taiwan, utf8dest, len)) + { + puts("FAIL (results do not match)"); + print_utf8(" utf8taiwan", utf8taiwan); + print_utf8(" utf8dest", utf8dest); + errors ++; + } + else + puts("PASS"); + +#if 0 + /* + * Test UTF-8 (16-bit) to UTF-32 (w/ BOM)... + */ + if (verbose) + printf("\ntesti18n: Testing UTF-8 to UTF-32 (w/ BOM)...\n"); + len = cupsUTF8ToUTF32(utf32dest, utf8good, 1024); + if (len < 0) + return (1); + if (verbose) + { + print_utf8(" utf8good ", utf8good); + print_utf32(" utf32dest", utf32dest); + } + memcpy (utf32src, utf32dest, (len + 1) * sizeof(cups_utf32_t)); + len = cupsUTF32ToUTF8(utf8dest, utf32src, 1024); + if (len < 0) + return (1); + if (len != strlen ((char *) utf8good)) + return (1); + if (memcmp(utf8good, utf8dest, len) != 0) + return (1); + + /* + * Test invalid UTF-8 (16-bit) to UTF-32 (w/ BOM)... + */ + if (verbose) + printf("\ntesti18n: Testing UTF-8 bad 16-bit source string...\n"); + len = cupsUTF8ToUTF32(utf32dest, utf8bad, 1024); + if (len >= 0) + return (1); + if (verbose) + print_utf8(" utf8bad ", utf8bad); + + /* + * Test _cupsCharmapFlush()... + */ + if (verbose) + printf("\ntesti18n: Testing _cupsCharmapFlush()...\n"); + _cupsCharmapFlush(); + return (0); +#endif /* 0 */ + + return (errors > 0); +} + + +/* + * 'print_utf8()' - Print UTF-8 string with (optional) message. + */ + +static void +print_utf8(const char *msg, /* I - Message String */ + const cups_utf8_t *src) /* I - UTF-8 Source String */ +{ + const char *prefix; /* Prefix string */ + + + if (msg) + printf("%s:", msg); + + for (prefix = " "; *src; src ++) + { + printf("%s%02x", prefix, *src); + + if ((src[0] & 0x80) && (src[1] & 0x80)) + prefix = ""; + else + prefix = " "; + } + + putchar('\n'); +} + + +/* + * End of "$Id$" + */ diff --git a/cups/testipp.c b/cups/testipp.c new file mode 100644 index 0000000000..bdab874247 --- /dev/null +++ b/cups/testipp.c @@ -0,0 +1,1005 @@ +/* + * "$Id$" + * + * IPP test program for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Main entry. + * hex_dump() - Produce a hex dump of a buffer. + * print_attributes() - Print the attributes in a request... + * read_cb() - Read data from a buffer. + * write_cb() - Write data into a buffer. + */ + +/* + * Include necessary headers... + */ + +#include "file.h" +#include "string-private.h" +#include "ipp-private.h" +#ifdef WIN32 +# include +#else +# include +# include +#endif /* WIN32 */ + + +/* + * Local types... + */ + +typedef struct _ippdata_t +{ + size_t rpos, /* Read position */ + wused, /* Bytes used */ + wsize; /* Max size of buffer */ + ipp_uchar_t *wbuffer; /* Buffer */ +} _ippdata_t; + + +/* + * Local globals... + */ + +ipp_uchar_t collection[] = /* Collection buffer */ + { + 0x01, 0x01, /* IPP version */ + 0x00, 0x02, /* Print-Job operation */ + 0x00, 0x00, 0x00, 0x01, + /* Request ID */ + + IPP_TAG_OPERATION, + + IPP_TAG_CHARSET, + 0x00, 0x12, /* Name length + name */ + 'a','t','t','r','i','b','u','t','e','s','-', + 'c','h','a','r','s','e','t', + 0x00, 0x05, /* Value length + value */ + 'u','t','f','-','8', + + IPP_TAG_LANGUAGE, + 0x00, 0x1b, /* Name length + name */ + 'a','t','t','r','i','b','u','t','e','s','-', + 'n','a','t','u','r','a','l','-','l','a','n', + 'g','u','a','g','e', + 0x00, 0x02, /* Value length + value */ + 'e','n', + + IPP_TAG_URI, + 0x00, 0x0b, /* Name length + name */ + 'p','r','i','n','t','e','r','-','u','r','i', + 0x00, 0x1c, /* Value length + value */ + 'i','p','p',':','/','/','l','o','c','a','l', + 'h','o','s','t','/','p','r','i','n','t','e', + 'r','s','/','f','o','o', + + IPP_TAG_JOB, /* job group tag */ + + IPP_TAG_BEGIN_COLLECTION, + /* begCollection tag */ + 0x00, 0x09, /* Name length + name */ + 'm', 'e', 'd', 'i', 'a', '-', 'c', 'o', 'l', + 0x00, 0x00, /* No value */ + IPP_TAG_MEMBERNAME, /* memberAttrName tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x0a, /* Value length + value */ + 'm', 'e', 'd', 'i', 'a', '-', 's', 'i', 'z', 'e', + IPP_TAG_BEGIN_COLLECTION, + /* begCollection tag */ + 0x00, 0x00, /* Name length + name */ + 0x00, 0x00, /* No value */ + IPP_TAG_MEMBERNAME, + /* memberAttrName tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x0b, /* Value length + value */ + 'x', '-', 'd', 'i', 'm', 'e', 'n', 's', 'i', 'o', 'n', + IPP_TAG_INTEGER, /* integer tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x04, /* Value length + value */ + 0x00, 0x00, 0x54, 0x56, + IPP_TAG_MEMBERNAME, + /* memberAttrName tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x0b, /* Value length + value */ + 'y', '-', 'd', 'i', 'm', 'e', 'n', 's', 'i', 'o', 'n', + IPP_TAG_INTEGER, /* integer tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x04, /* Value length + value */ + 0x00, 0x00, 0x6d, 0x24, + IPP_TAG_END_COLLECTION, + /* endCollection tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x00, /* No value */ + IPP_TAG_MEMBERNAME, /* memberAttrName tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x0b, /* Value length + value */ + 'm', 'e', 'd', 'i', 'a', '-', 'c', 'o', 'l', 'o', 'r', + IPP_TAG_KEYWORD, /* keyword tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x04, /* Value length + value */ + 'b', 'l', 'u', 'e', + + IPP_TAG_MEMBERNAME, /* memberAttrName tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x0a, /* Value length + value */ + 'm', 'e', 'd', 'i', 'a', '-', 't', 'y', 'p', 'e', + IPP_TAG_KEYWORD, /* keyword tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x05, /* Value length + value */ + 'p', 'l', 'a', 'i', 'n', + IPP_TAG_END_COLLECTION, + /* endCollection tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x00, /* No value */ + + IPP_TAG_BEGIN_COLLECTION, + /* begCollection tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x00, /* No value */ + IPP_TAG_MEMBERNAME, /* memberAttrName tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x0a, /* Value length + value */ + 'm', 'e', 'd', 'i', 'a', '-', 's', 'i', 'z', 'e', + IPP_TAG_BEGIN_COLLECTION, + /* begCollection tag */ + 0x00, 0x00, /* Name length + name */ + 0x00, 0x00, /* No value */ + IPP_TAG_MEMBERNAME, + /* memberAttrName tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x0b, /* Value length + value */ + 'x', '-', 'd', 'i', 'm', 'e', 'n', 's', 'i', 'o', 'n', + IPP_TAG_INTEGER, /* integer tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x04, /* Value length + value */ + 0x00, 0x00, 0x52, 0x08, + IPP_TAG_MEMBERNAME, + /* memberAttrName tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x0b, /* Value length + value */ + 'y', '-', 'd', 'i', 'm', 'e', 'n', 's', 'i', 'o', 'n', + IPP_TAG_INTEGER, /* integer tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x04, /* Value length + value */ + 0x00, 0x00, 0x74, 0x04, + IPP_TAG_END_COLLECTION, + /* endCollection tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x00, /* No value */ + IPP_TAG_MEMBERNAME, /* memberAttrName tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x0b, /* Value length + value */ + 'm', 'e', 'd', 'i', 'a', '-', 'c', 'o', 'l', 'o', 'r', + IPP_TAG_KEYWORD, /* keyword tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x05, /* Value length + value */ + 'p', 'l', 'a', 'i', 'd', + + IPP_TAG_MEMBERNAME, /* memberAttrName tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x0a, /* Value length + value */ + 'm', 'e', 'd', 'i', 'a', '-', 't', 'y', 'p', 'e', + IPP_TAG_KEYWORD, /* keyword tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x06, /* Value length + value */ + 'g', 'l', 'o', 's', 's', 'y', + IPP_TAG_END_COLLECTION, + /* endCollection tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x00, /* No value */ + + IPP_TAG_END /* end tag */ + }; + +ipp_uchar_t mixed[] = /* Mixed value buffer */ + { + 0x01, 0x01, /* IPP version */ + 0x00, 0x02, /* Print-Job operation */ + 0x00, 0x00, 0x00, 0x01, + /* Request ID */ + + IPP_TAG_OPERATION, + + IPP_TAG_INTEGER, /* integer tag */ + 0x00, 0x1f, /* Name length + name */ + 'n', 'o', 't', 'i', 'f', 'y', '-', 'l', 'e', 'a', 's', 'e', + '-', 'd', 'u', 'r', 'a', 't', 'i', 'o', 'n', '-', 's', 'u', + 'p', 'p', 'o', 'r', 't', 'e', 'd', + 0x00, 0x04, /* Value length + value */ + 0x00, 0x00, 0x00, 0x01, + + IPP_TAG_RANGE, /* rangeOfInteger tag */ + 0x00, 0x00, /* No name */ + 0x00, 0x08, /* Value length + value */ + 0x00, 0x00, 0x00, 0x10, + 0x00, 0x00, 0x00, 0x20, + + IPP_TAG_END /* end tag */ + }; + + +/* + * Local functions... + */ + +void hex_dump(const char *title, ipp_uchar_t *buffer, int bytes); +void print_attributes(ipp_t *ipp, int indent); +ssize_t read_cb(_ippdata_t *data, ipp_uchar_t *buffer, size_t bytes); +ssize_t write_cb(_ippdata_t *data, ipp_uchar_t *buffer, size_t bytes); + + +/* + * 'main()' - Main entry. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + _ippdata_t data; /* IPP buffer */ + ipp_uchar_t buffer[8192]; /* Write buffer data */ + ipp_t *cols[2], /* Collections */ + *size; /* media-size collection */ + ipp_t *request; /* Request */ + ipp_attribute_t *media_col, /* media-col attribute */ + *media_size, /* media-size attribute */ + *attr; /* Other attribute */ + ipp_state_t state; /* State */ + int length; /* Length of data */ + cups_file_t *fp; /* File pointer */ + int i; /* Looping var */ + int status; /* Status of tests (0 = success, 1 = fail) */ + + + status = 0; + + if (argc == 1) + { + /* + * Test request generation code... + */ + + printf("Create Sample Request: "); + + request = ippNew(); + request->request.op.version[0] = 0x01; + request->request.op.version[1] = 0x01; + request->request.op.operation_id = IPP_PRINT_JOB; + request->request.op.request_id = 1; + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_CHARSET, + "attributes-charset", NULL, "utf-8"); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_LANGUAGE, + "attributes-natural-language", NULL, "en"); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, "ipp://localhost/printers/foo"); + + cols[0] = ippNew(); + size = ippNew(); + ippAddInteger(size, IPP_TAG_ZERO, IPP_TAG_INTEGER, "x-dimension", 21590); + ippAddInteger(size, IPP_TAG_ZERO, IPP_TAG_INTEGER, "y-dimension", 27940); + ippAddCollection(cols[0], IPP_TAG_JOB, "media-size", size); + ippDelete(size); + ippAddString(cols[0], IPP_TAG_JOB, IPP_TAG_KEYWORD, "media-color", NULL, + "blue"); + ippAddString(cols[0], IPP_TAG_JOB, IPP_TAG_KEYWORD, "media-type", NULL, + "plain"); + + cols[1] = ippNew(); + size = ippNew(); + ippAddInteger(size, IPP_TAG_ZERO, IPP_TAG_INTEGER, "x-dimension", 21000); + ippAddInteger(size, IPP_TAG_ZERO, IPP_TAG_INTEGER, "y-dimension", 29700); + ippAddCollection(cols[1], IPP_TAG_JOB, "media-size", size); + ippDelete(size); + ippAddString(cols[1], IPP_TAG_JOB, IPP_TAG_KEYWORD, "media-color", NULL, + "plaid"); + ippAddString(cols[1], IPP_TAG_JOB, IPP_TAG_KEYWORD, "media-type", NULL, + "glossy"); + + ippAddCollections(request, IPP_TAG_JOB, "media-col", 2, + (const ipp_t **)cols); + ippDelete(cols[0]); + ippDelete(cols[1]); + + length = ippLength(request); + if (length != sizeof(collection)) + { + printf("FAIL - wrong ippLength(), %d instead of %d bytes!\n", + length, (int)sizeof(collection)); + status = 1; + } + else + puts("PASS"); + + /* + * Write test #1... + */ + + printf("Write Sample to Memory: "); + + data.wused = 0; + data.wsize = sizeof(buffer); + data.wbuffer = buffer; + + while ((state = ippWriteIO(&data, (ipp_iocb_t)write_cb, 1, NULL, + request)) != IPP_DATA) + if (state == IPP_ERROR) + break; + + if (state != IPP_DATA) + { + printf("FAIL - %d bytes written.\n", (int)data.wused); + status = 1; + } + else if (data.wused != sizeof(collection)) + { + printf("FAIL - wrote %d bytes, expected %d bytes!\n", (int)data.wused, + (int)sizeof(collection)); + hex_dump("Bytes Written", data.wbuffer, data.wused); + hex_dump("Baseline", collection, sizeof(collection)); + status = 1; + } + else if (memcmp(data.wbuffer, collection, data.wused)) + { + for (i = 0; i < data.wused; i ++) + if (data.wbuffer[i] != collection[i]) + break; + + printf("FAIL - output does not match baseline at 0x%04x!\n", i); + hex_dump("Bytes Written", data.wbuffer, data.wused); + hex_dump("Baseline", collection, sizeof(collection)); + status = 1; + } + else + puts("PASS"); + + ippDelete(request); + + /* + * Read the data back in and confirm... + */ + + printf("Read Sample from Memory: "); + + request = ippNew(); + data.rpos = 0; + + while ((state = ippReadIO(&data, (ipp_iocb_t)read_cb, 1, NULL, + request)) != IPP_DATA) + if (state == IPP_ERROR) + break; + + length = ippLength(request); + + if (state != IPP_DATA) + { + printf("FAIL - %d bytes read.\n", (int)data.rpos); + status = 1; + } + else if (data.rpos != data.wused) + { + printf("FAIL - read %d bytes, expected %d bytes!\n", (int)data.rpos, + (int)data.wused); + print_attributes(request, 8); + status = 1; + } + else if (length != sizeof(collection)) + { + printf("FAIL - wrong ippLength(), %d instead of %d bytes!\n", + length, (int)sizeof(collection)); + print_attributes(request, 8); + status = 1; + } + else + puts("PASS"); + + fputs("ippFindAttribute(media-col): ", stdout); + if ((media_col = ippFindAttribute(request, "media-col", + IPP_TAG_BEGIN_COLLECTION)) == NULL) + { + if ((media_col = ippFindAttribute(request, "media-col", + IPP_TAG_ZERO)) == NULL) + puts("FAIL (not found)"); + else + printf("FAIL (wrong type - %s)\n", ippTagString(media_col->value_tag)); + + status = 1; + } + else if (media_col->num_values != 2) + { + printf("FAIL (wrong count - %d)\n", media_col->num_values); + status = 1; + } + else + puts("PASS"); + + if (media_col) + { + fputs("ippFindAttribute(media-size 1): ", stdout); + if ((media_size = ippFindAttribute(media_col->values[0].collection, + "media-size", + IPP_TAG_BEGIN_COLLECTION)) == NULL) + { + if ((media_size = ippFindAttribute(media_col->values[0].collection, + "media-col", + IPP_TAG_ZERO)) == NULL) + puts("FAIL (not found)"); + else + printf("FAIL (wrong type - %s)\n", + ippTagString(media_size->value_tag)); + + status = 1; + } + else + { + if ((attr = ippFindAttribute(media_size->values[0].collection, + "x-dimension", IPP_TAG_INTEGER)) == NULL) + { + if ((attr = ippFindAttribute(media_size->values[0].collection, + "x-dimension", IPP_TAG_ZERO)) == NULL) + puts("FAIL (missing x-dimension)"); + else + printf("FAIL (wrong type for x-dimension - %s)\n", + ippTagString(attr->value_tag)); + + status = 1; + } + else if (attr->values[0].integer != 21590) + { + printf("FAIL (wrong value for x-dimension - %d)\n", + attr->values[0].integer); + status = 1; + } + else if ((attr = ippFindAttribute(media_size->values[0].collection, + "y-dimension", + IPP_TAG_INTEGER)) == NULL) + { + if ((attr = ippFindAttribute(media_size->values[0].collection, + "y-dimension", IPP_TAG_ZERO)) == NULL) + puts("FAIL (missing y-dimension)"); + else + printf("FAIL (wrong type for y-dimension - %s)\n", + ippTagString(attr->value_tag)); + + status = 1; + } + else if (attr->values[0].integer != 27940) + { + printf("FAIL (wrong value for y-dimension - %d)\n", + attr->values[0].integer); + status = 1; + } + else + puts("PASS"); + } + + fputs("ippFindAttribute(media-size 2): ", stdout); + if ((media_size = ippFindAttribute(media_col->values[1].collection, + "media-size", + IPP_TAG_BEGIN_COLLECTION)) == NULL) + { + if ((media_size = ippFindAttribute(media_col->values[1].collection, + "media-col", + IPP_TAG_ZERO)) == NULL) + puts("FAIL (not found)"); + else + printf("FAIL (wrong type - %s)\n", + ippTagString(media_size->value_tag)); + + status = 1; + } + else + { + if ((attr = ippFindAttribute(media_size->values[0].collection, + "x-dimension", + IPP_TAG_INTEGER)) == NULL) + { + if ((attr = ippFindAttribute(media_size->values[0].collection, + "x-dimension", IPP_TAG_ZERO)) == NULL) + puts("FAIL (missing x-dimension)"); + else + printf("FAIL (wrong type for x-dimension - %s)\n", + ippTagString(attr->value_tag)); + + status = 1; + } + else if (attr->values[0].integer != 21000) + { + printf("FAIL (wrong value for x-dimension - %d)\n", + attr->values[0].integer); + status = 1; + } + else if ((attr = ippFindAttribute(media_size->values[0].collection, + "y-dimension", + IPP_TAG_INTEGER)) == NULL) + { + if ((attr = ippFindAttribute(media_size->values[0].collection, + "y-dimension", IPP_TAG_ZERO)) == NULL) + puts("FAIL (missing y-dimension)"); + else + printf("FAIL (wrong type for y-dimension - %s)\n", + ippTagString(attr->value_tag)); + + status = 1; + } + else if (attr->values[0].integer != 29700) + { + printf("FAIL (wrong value for y-dimension - %d)\n", + attr->values[0].integer); + status = 1; + } + else + puts("PASS"); + } + } + + ippDelete(request); + + /* + * Read the mixed data and confirm we converted everything to rangeOfInteger + * values... + */ + + printf("Read Mixed integer/rangeOfInteger from Memory: "); + + request = ippNew(); + data.rpos = 0; + data.wused = sizeof(mixed); + data.wsize = sizeof(mixed); + data.wbuffer = mixed; + + while ((state = ippReadIO(&data, (ipp_iocb_t)read_cb, 1, NULL, + request)) != IPP_DATA) + if (state == IPP_ERROR) + break; + + length = ippLength(request); + + if (state != IPP_DATA) + { + printf("FAIL - %d bytes read.\n", (int)data.rpos); + status = 1; + } + else if (data.rpos != sizeof(mixed)) + { + printf("FAIL - read %d bytes, expected %d bytes!\n", (int)data.rpos, + (int)sizeof(mixed)); + print_attributes(request, 8); + status = 1; + } + else if (length != (sizeof(mixed) + 4)) + { + printf("FAIL - wrong ippLength(), %d instead of %d bytes!\n", + length, (int)sizeof(mixed) + 4); + print_attributes(request, 8); + status = 1; + } + else + puts("PASS"); + + fputs("ippFindAttribute(notify-lease-duration-supported): ", stdout); + if ((attr = ippFindAttribute(request, "notify-lease-duration-supported", + IPP_TAG_ZERO)) == NULL) + { + puts("FAIL (not found)"); + status = 1; + } + else if (attr->value_tag != IPP_TAG_RANGE) + { + printf("FAIL (wrong type - %s)\n", ippTagString(attr->value_tag)); + status = 1; + } + else if (attr->num_values != 2) + { + printf("FAIL (wrong count - %d)\n", attr->num_values); + status = 1; + } + else if (attr->values[0].range.lower != 1 || + attr->values[0].range.upper != 1 || + attr->values[1].range.lower != 16 || + attr->values[1].range.upper != 32) + { + printf("FAIL (wrong values - %d,%d and %d,%d)\n", + attr->values[0].range.lower, + attr->values[0].range.upper, + attr->values[1].range.lower, + attr->values[1].range.upper); + status = 1; + } + else + puts("PASS"); + + ippDelete(request); + + /* + * Test _ippFindOption() private API... + */ + + fputs("_ippFindOption(printer-type): ", stdout); + if (_ippFindOption("printer-type")) + puts("PASS"); + else + { + puts("FAIL"); + status = 1; + } + + /* + * Summarize... + */ + + putchar('\n'); + + if (status) + puts("Core IPP tests failed."); + else + puts("Core IPP tests passed."); + } + else + { + /* + * Read IPP files... + */ + + for (i = 1; i < argc; i ++) + { + if ((fp = cupsFileOpen(argv[i], "r")) == NULL) + { + printf("Unable to open \"%s\" - %s\n", argv[i], strerror(errno)); + status = 1; + continue; + } + + request = ippNew(); + while ((state = ippReadIO(fp, (ipp_iocb_t)cupsFileRead, 1, NULL, + request)) == IPP_ATTRIBUTE); + + if (state != IPP_DATA) + { + printf("Error reading IPP message from \"%s\"!\n", argv[i]); + status = 1; + } + else + { + printf("\n%s:\n", argv[i]); + print_attributes(request, 4); + } + + ippDelete(request); + cupsFileClose(fp); + } + } + + return (status); +} + + +/* + * 'hex_dump()' - Produce a hex dump of a buffer. + */ + +void +hex_dump(const char *title, /* I - Title */ + ipp_uchar_t *buffer, /* I - Buffer to dump */ + int bytes) /* I - Number of bytes */ +{ + int i, j; /* Looping vars */ + int ch; /* Current ASCII char */ + + + /* + * Show lines of 16 bytes at a time... + */ + + printf(" %s:\n", title); + + for (i = 0; i < bytes; i += 16) + { + /* + * Show the offset... + */ + + printf(" %04x ", i); + + /* + * Then up to 16 bytes in hex... + */ + + for (j = 0; j < 16; j ++) + if ((i + j) < bytes) + printf(" %02x", buffer[i + j]); + else + printf(" "); + + /* + * Then the ASCII representation of the bytes... + */ + + putchar(' '); + putchar(' '); + + for (j = 0; j < 16 && (i + j) < bytes; j ++) + { + ch = buffer[i + j] & 127; + + if (ch < ' ' || ch == 127) + putchar('.'); + else + putchar(ch); + } + + putchar('\n'); + } +} + + +/* + * 'print_attributes()' - Print the attributes in a request... + */ + +void +print_attributes(ipp_t *ipp, /* I - IPP request */ + int indent) /* I - Indentation */ +{ + int i; /* Looping var */ + ipp_tag_t group; /* Current group */ + ipp_attribute_t *attr; /* Current attribute */ + _ipp_value_t *val; /* Current value */ + static const char * const tags[] = /* Value/group tag strings */ + { + "reserved-00", + "operation-attributes-tag", + "job-attributes-tag", + "end-of-attributes-tag", + "printer-attributes-tag", + "unsupported-attributes-tag", + "subscription-attributes-tag", + "event-attributes-tag", + "reserved-08", + "reserved-09", + "reserved-0A", + "reserved-0B", + "reserved-0C", + "reserved-0D", + "reserved-0E", + "reserved-0F", + "unsupported", + "default", + "unknown", + "no-value", + "reserved-14", + "not-settable", + "delete-attr", + "admin-define", + "reserved-18", + "reserved-19", + "reserved-1A", + "reserved-1B", + "reserved-1C", + "reserved-1D", + "reserved-1E", + "reserved-1F", + "reserved-20", + "integer", + "boolean", + "enum", + "reserved-24", + "reserved-25", + "reserved-26", + "reserved-27", + "reserved-28", + "reserved-29", + "reserved-2a", + "reserved-2b", + "reserved-2c", + "reserved-2d", + "reserved-2e", + "reserved-2f", + "octetString", + "dateTime", + "resolution", + "rangeOfInteger", + "begCollection", + "textWithLanguage", + "nameWithLanguage", + "endCollection", + "reserved-38", + "reserved-39", + "reserved-3a", + "reserved-3b", + "reserved-3c", + "reserved-3d", + "reserved-3e", + "reserved-3f", + "reserved-40", + "textWithoutLanguage", + "nameWithoutLanguage", + "reserved-43", + "keyword", + "uri", + "uriScheme", + "charset", + "naturalLanguage", + "mimeMediaType", + "memberName" + }; + + + for (group = IPP_TAG_ZERO, attr = ipp->attrs; attr; attr = attr->next) + { + if (!attr->name && indent == 4) + { + group = IPP_TAG_ZERO; + putchar('\n'); + continue; + } + + if (group != attr->group_tag) + { + group = attr->group_tag; + + printf("\n%*s%s:\n\n", indent - 4, "", tags[group]); + } + + printf("%*s%s (", indent, "", attr->name ? attr->name : "(null)"); + if (attr->num_values > 1) + printf("1setOf "); + printf("%s):", tags[attr->value_tag]); + + switch (attr->value_tag) + { + case IPP_TAG_ENUM : + case IPP_TAG_INTEGER : + for (i = 0, val = attr->values; i < attr->num_values; i ++, val ++) + printf(" %d", val->integer); + putchar('\n'); + break; + + case IPP_TAG_BOOLEAN : + for (i = 0, val = attr->values; i < attr->num_values; i ++, val ++) + printf(" %s", val->boolean ? "true" : "false"); + putchar('\n'); + break; + + case IPP_TAG_RANGE : + for (i = 0, val = attr->values; i < attr->num_values; i ++, val ++) + printf(" %d-%d", val->range.lower, val->range.upper); + putchar('\n'); + break; + + case IPP_TAG_DATE : + { + time_t vtime; /* Date/Time value */ + struct tm *vdate; /* Date info */ + char vstring[256]; /* Formatted time */ + + for (i = 0, val = attr->values; i < attr->num_values; i ++, val ++) + { + vtime = ippDateToTime(val->date); + vdate = localtime(&vtime); + strftime(vstring, sizeof(vstring), "%c", vdate); + printf(" (%s)", vstring); + } + } + putchar('\n'); + break; + + case IPP_TAG_RESOLUTION : + for (i = 0, val = attr->values; i < attr->num_values; i ++, val ++) + printf(" %dx%d%s", val->resolution.xres, val->resolution.yres, + val->resolution.units == IPP_RES_PER_INCH ? "dpi" : "dpc"); + putchar('\n'); + break; + + case IPP_TAG_STRING : + case IPP_TAG_TEXTLANG : + case IPP_TAG_NAMELANG : + case IPP_TAG_TEXT : + case IPP_TAG_NAME : + case IPP_TAG_KEYWORD : + case IPP_TAG_URI : + case IPP_TAG_URISCHEME : + case IPP_TAG_CHARSET : + case IPP_TAG_LANGUAGE : + case IPP_TAG_MIMETYPE : + for (i = 0, val = attr->values; i < attr->num_values; i ++, val ++) + printf(" \"%s\"", val->string.text); + putchar('\n'); + break; + + case IPP_TAG_BEGIN_COLLECTION : + putchar('\n'); + + for (i = 0, val = attr->values; i < attr->num_values; i ++, val ++) + { + if (i) + putchar('\n'); + print_attributes(val->collection, indent + 4); + } + break; + + default : + printf("UNKNOWN (%d values)\n", attr->num_values); + break; + } + } +} + + +/* + * 'read_cb()' - Read data from a buffer. + */ + +ssize_t /* O - Number of bytes read */ +read_cb(_ippdata_t *data, /* I - Data */ + ipp_uchar_t *buffer, /* O - Buffer to read */ + size_t bytes) /* I - Number of bytes to read */ +{ + size_t count; /* Number of bytes */ + + + /* + * Copy bytes from the data buffer to the read buffer... + */ + + if ((count = data->wsize - data->rpos) > bytes) + count = bytes; + + memcpy(buffer, data->wbuffer + data->rpos, count); + data->rpos += count; + + /* + * Return the number of bytes read... + */ + + return (count); +} + + +/* + * 'write_cb()' - Write data into a buffer. + */ + +ssize_t /* O - Number of bytes written */ +write_cb(_ippdata_t *data, /* I - Data */ + ipp_uchar_t *buffer, /* I - Buffer to write */ + size_t bytes) /* I - Number of bytes to write */ +{ + size_t count; /* Number of bytes */ + + + /* + * Loop until all bytes are written... + */ + + if ((count = data->wsize - data->wused) > bytes) + count = bytes; + + memcpy(data->wbuffer + data->wused, buffer, count); + data->wused += count; + + /* + * Return the number of bytes written... + */ + + return (count); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/testlang.c b/cups/testlang.c new file mode 100644 index 0000000000..701c14db40 --- /dev/null +++ b/cups/testlang.c @@ -0,0 +1,114 @@ +/* + * "$Id$" + * + * Localization test program for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Load the specified language and show the strings for yes and no. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" + + +/* + * 'main()' - Load the specified language and show the strings for yes and no. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + int errors = 0; /* Number of errors */ + cups_lang_t *language; /* Message catalog */ + cups_lang_t *language2; /* Message catalog */ + struct lconv *loc; /* Locale data */ + char buffer[1024]; /* String buffer */ + double number; /* Number */ + static const char * const tests[] = /* Test strings */ + { + "1", + "-1", + "3", + "5.125" + }; + + + _cupsSetLocale(argv); + + if (argc == 1) + { + language = cupsLangDefault(); + language2 = cupsLangDefault(); + } + else + { + language = cupsLangGet(argv[1]); + language2 = cupsLangGet(argv[1]); + } + + if (language != language2) + { + errors ++; + + puts("**** ERROR: Language cache did not work! ****"); + puts("First result from cupsLangGet:"); + } + + printf("Language = \"%s\"\n", language->language); + printf("Encoding = \"%s\"\n", _cupsEncodingName(language->encoding)); + printf("No = \"%s\"\n", _cupsLangString(language, "No")); + printf("Yes = \"%s\"\n", _cupsLangString(language, "Yes")); + + if (language != language2) + { + puts("Second result from cupsLangGet:"); + + printf("Language = \"%s\"\n", language2->language); + printf("Encoding = \"%s\"\n", _cupsEncodingName(language2->encoding)); + printf("No = \"%s\"\n", _cupsLangString(language2, "No")); + printf("Yes = \"%s\"\n", _cupsLangString(language2, "Yes")); + } + + loc = localeconv(); + + for (i = 0; i < (int)(sizeof(tests) / sizeof(tests[0])); i ++) + { + number = _cupsStrScand(tests[i], NULL, loc); + + printf("_cupsStrScand(\"%s\") number=%f\n", tests[i], number); + + _cupsStrFormatd(buffer, buffer + sizeof(buffer), number, loc); + + printf("_cupsStrFormatd(%f) buffer=\"%s\"\n", number, buffer); + + if (strcmp(buffer, tests[i])) + { + errors ++; + puts("**** ERROR: Bad formatted number! ****"); + } + } + + return (errors > 0); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/testoptions.c b/cups/testoptions.c new file mode 100644 index 0000000000..1cd72794fa --- /dev/null +++ b/cups/testoptions.c @@ -0,0 +1,116 @@ +/* + * "$Id$" + * + * Option test program for CUPS. + * + * Copyright 2008-2010 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Test option processing functions. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" + + +/* + * 'main()' - Test option processing functions. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int status = 0, /* Exit status */ + num_options; /* Number of options */ + cups_option_t *options; /* Options */ + const char *value; /* Value of an option */ + + + if (argc == 1) + { + /* + * cupsParseOptions() + */ + + fputs("cupsParseOptions: ", stdout); + + num_options = cupsParseOptions("foo=1234 " + "bar=\"One Fish\",\"Two Fish\",\"Red Fish\"," + "\"Blue Fish\" " + "baz={param1=1 param2=2} " + "foobar=FOO\\ BAR " + "barfoo=barfoo " + "barfoo=\"\'BAR FOO\'\"", 0, &options); + + if (num_options != 5) + { + printf("FAIL (num_options=%d, expected 5)\n", num_options); + status ++; + } + else if ((value = cupsGetOption("foo", num_options, options)) == NULL || + strcmp(value, "1234")) + { + printf("FAIL (foo=\"%s\", expected \"1234\")\n", value); + status ++; + } + else if ((value = cupsGetOption("bar", num_options, options)) == NULL || + strcmp(value, "One Fish,Two Fish,Red Fish,Blue Fish")) + { + printf("FAIL (bar=\"%s\", expected \"One Fish,Two Fish,Red Fish,Blue " + "Fish\")\n", value); + status ++; + } + else if ((value = cupsGetOption("baz", num_options, options)) == NULL || + strcmp(value, "{param1=1 param2=2}")) + { + printf("FAIL (baz=\"%s\", expected \"{param1=1 param2=2}\")\n", value); + status ++; + } + else if ((value = cupsGetOption("foobar", num_options, options)) == NULL || + strcmp(value, "FOO BAR")) + { + printf("FAIL (foobar=\"%s\", expected \"FOO BAR\")\n", value); + status ++; + } + else if ((value = cupsGetOption("barfoo", num_options, options)) == NULL || + strcmp(value, "\'BAR FOO\'")) + { + printf("FAIL (barfoo=\"%s\", expected \"\'BAR FOO\'\")\n", value); + status ++; + } + else + puts("PASS"); + } + else + { + int i; /* Looping var */ + cups_option_t *option; /* Current option */ + + + num_options = cupsParseOptions(argv[1], 0, &options); + + for (i = 0, option = options; i < num_options; i ++, option ++) + printf("options[%d].name=\"%s\", value=\"%s\"\n", i, option->name, + option->value); + } + + exit (status); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/testppd.c b/cups/testppd.c new file mode 100644 index 0000000000..3de6475117 --- /dev/null +++ b/cups/testppd.c @@ -0,0 +1,1102 @@ +/* + * "$Id$" + * + * PPD test program for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Main entry. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include +#ifdef WIN32 +# include +#else +# include +# include +#endif /* WIN32 */ + + +/* + * Test data... + */ + +static const char *default_code = + "[{\n" + "%%BeginFeature: *InstalledDuplexer False\n" + "%%EndFeature\n" + "} stopped cleartomark\n" + "[{\n" + "%%BeginFeature: *PageRegion Letter\n" + "PageRegion=Letter\n" + "%%EndFeature\n" + "} stopped cleartomark\n" + "[{\n" + "%%BeginFeature: *InputSlot Tray\n" + "InputSlot=Tray\n" + "%%EndFeature\n" + "} stopped cleartomark\n" + "[{\n" + "%%BeginFeature: *MediaType Plain\n" + "MediaType=Plain\n" + "%%EndFeature\n" + "} stopped cleartomark\n" + "[{\n" + "%%BeginFeature: *IntOption None\n" + "%%EndFeature\n" + "} stopped cleartomark\n" + "[{\n" + "%%BeginFeature: *StringOption None\n" + "%%EndFeature\n" + "} stopped cleartomark\n"; + +static const char *custom_code = + "[{\n" + "%%BeginFeature: *InstalledDuplexer False\n" + "%%EndFeature\n" + "} stopped cleartomark\n" + "[{\n" + "%%BeginFeature: *InputSlot Tray\n" + "InputSlot=Tray\n" + "%%EndFeature\n" + "} stopped cleartomark\n" + "[{\n" + "%%BeginFeature: *MediaType Plain\n" + "MediaType=Plain\n" + "%%EndFeature\n" + "} stopped cleartomark\n" + "[{\n" + "%%BeginFeature: *IntOption None\n" + "%%EndFeature\n" + "} stopped cleartomark\n" + "[{\n" + "%%BeginFeature: *CustomStringOption True\n" + "(value\\0502\\051)\n" + "(value 1)\n" + "StringOption=Custom\n" + "%%EndFeature\n" + "} stopped cleartomark\n" + "[{\n" + "%%BeginFeature: *CustomPageSize True\n" + "400\n" + "500\n" + "0\n" + "0\n" + "0\n" + "PageSize=Custom\n" + "%%EndFeature\n" + "} stopped cleartomark\n"; + +static const char *default2_code = + "[{\n" + "%%BeginFeature: *InstalledDuplexer False\n" + "%%EndFeature\n" + "} stopped cleartomark\n" + "[{\n" + "%%BeginFeature: *InputSlot Tray\n" + "InputSlot=Tray\n" + "%%EndFeature\n" + "} stopped cleartomark\n" + "[{\n" + "%%BeginFeature: *Quality Normal\n" + "Quality=Normal\n" + "%%EndFeature\n" + "} stopped cleartomark\n" + "[{\n" + "%%BeginFeature: *IntOption None\n" + "%%EndFeature\n" + "} stopped cleartomark\n" + "[{\n" + "%%BeginFeature: *StringOption None\n" + "%%EndFeature\n" + "} stopped cleartomark\n"; + + +/* + * 'main()' - Main entry. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + ppd_file_t *ppd; /* PPD file loaded from disk */ + int status; /* Status of tests (0 = success, 1 = fail) */ + int conflicts; /* Number of conflicts */ + char *s; /* String */ + char buffer[8192]; /* String buffer */ + const char *text, /* Localized text */ + *val; /* Option value */ + int num_options; /* Number of options */ + cups_option_t *options; /* Options */ + ppd_size_t minsize, /* Minimum size */ + maxsize, /* Maximum size */ + *size; /* Current size */ + ppd_attr_t *attr; /* Current attribute */ + + + status = 0; + + if (argc == 1) + { + /* + * Setup directories for locale stuff... + */ + + if (access("locale", 0)) + { + mkdir("locale", 0777); + mkdir("locale/fr", 0777); + symlink("../../../locale/cups_fr.po", "locale/fr/cups_fr.po"); + mkdir("locale/zh_TW", 0777); + symlink("../../../locale/cups_zh_TW.po", "locale/zh_TW/cups_zh_TW.po"); + } + + putenv("LOCALEDIR=locale"); + putenv("SOFTWARE=CUPS"); + + /* + * Do tests with test.ppd... + */ + + fputs("ppdOpenFile(test.ppd): ", stdout); + + if ((ppd = _ppdOpenFile("test.ppd", _PPD_LOCALIZATION_ALL)) != NULL) + puts("PASS"); + else + { + ppd_status_t err; /* Last error in file */ + int line; /* Line number in file */ + + + status ++; + err = ppdLastError(&line); + + printf("FAIL (%s on line %d)\n", ppdErrorString(err), line); + } + + fputs("ppdFindAttr(wildcard): ", stdout); + if ((attr = ppdFindAttr(ppd, "cupsTest", NULL)) == NULL) + { + status ++; + puts("FAIL (not found)"); + } + else if (strcmp(attr->name, "cupsTest") || strcmp(attr->spec, "Foo")) + { + status ++; + printf("FAIL (got \"%s %s\")\n", attr->name, attr->spec); + } + else + puts("PASS"); + + fputs("ppdFindNextAttr(wildcard): ", stdout); + if ((attr = ppdFindNextAttr(ppd, "cupsTest", NULL)) == NULL) + { + status ++; + puts("FAIL (not found)"); + } + else if (strcmp(attr->name, "cupsTest") || strcmp(attr->spec, "Bar")) + { + status ++; + printf("FAIL (got \"%s %s\")\n", attr->name, attr->spec); + } + else + puts("PASS"); + + fputs("ppdFindAttr(Foo): ", stdout); + if ((attr = ppdFindAttr(ppd, "cupsTest", "Foo")) == NULL) + { + status ++; + puts("FAIL (not found)"); + } + else if (strcmp(attr->name, "cupsTest") || strcmp(attr->spec, "Foo")) + { + status ++; + printf("FAIL (got \"%s %s\")\n", attr->name, attr->spec); + } + else + puts("PASS"); + + fputs("ppdFindNextAttr(Foo): ", stdout); + if ((attr = ppdFindNextAttr(ppd, "cupsTest", "Foo")) != NULL) + { + status ++; + printf("FAIL (got \"%s %s\")\n", attr->name, attr->spec); + } + else + puts("PASS"); + + fputs("ppdMarkDefaults: ", stdout); + ppdMarkDefaults(ppd); + + if ((conflicts = ppdConflicts(ppd)) == 0) + puts("PASS"); + else + { + status ++; + printf("FAIL (%d conflicts)\n", conflicts); + } + + fputs("ppdEmitString (defaults): ", stdout); + if ((s = ppdEmitString(ppd, PPD_ORDER_ANY, 0.0)) != NULL && + !strcmp(s, default_code)) + puts("PASS"); + else + { + status ++; + printf("FAIL (%d bytes instead of %d)\n", s ? (int)strlen(s) : 0, + (int)strlen(default_code)); + + if (s) + puts(s); + } + + if (s) + free(s); + + fputs("ppdEmitString (custom size and string): ", stdout); + ppdMarkOption(ppd, "PageSize", "Custom.400x500"); + ppdMarkOption(ppd, "StringOption", "{String1=\"value 1\" String2=value(2)}"); + + if ((s = ppdEmitString(ppd, PPD_ORDER_ANY, 0.0)) != NULL && + !strcmp(s, custom_code)) + puts("PASS"); + else + { + status ++; + printf("FAIL (%d bytes instead of %d)\n", s ? (int)strlen(s) : 0, + (int)strlen(custom_code)); + + if (s) + puts(s); + } + + if (s) + free(s); + + /* + * Test constraints... + */ + + fputs("cupsGetConflicts(InputSlot=Envelope): ", stdout); + ppdMarkOption(ppd, "PageSize", "Letter"); + + num_options = cupsGetConflicts(ppd, "InputSlot", "Envelope", &options); + if (num_options != 2 || + (val = cupsGetOption("PageRegion", num_options, options)) == NULL || + _cups_strcasecmp(val, "Letter") || + (val = cupsGetOption("PageSize", num_options, options)) == NULL || + _cups_strcasecmp(val, "Letter")) + { + printf("FAIL (%d options:", num_options); + for (i = 0; i < num_options; i ++) + printf(" %s=%s", options[i].name, options[i].value); + puts(")"); + status ++; + } + else + puts("PASS"); + + fputs("ppdConflicts(): ", stdout); + ppdMarkOption(ppd, "InputSlot", "Envelope"); + + if ((conflicts = ppdConflicts(ppd)) == 2) + puts("PASS (2)"); + else + { + printf("FAIL (%d)\n", conflicts); + status ++; + } + + fputs("cupsResolveConflicts(InputSlot=Envelope): ", stdout); + num_options = 0; + options = NULL; + if (!cupsResolveConflicts(ppd, "InputSlot", "Envelope", &num_options, + &options)) + { + puts("FAIL (Unable to resolve)"); + status ++; + } + else if (num_options != 2 || + !cupsGetOption("PageSize", num_options, options)) + { + printf("FAIL (%d options:", num_options); + for (i = 0; i < num_options; i ++) + printf(" %s=%s", options[i].name, options[i].value); + puts(")"); + status ++; + } + else + puts("PASS (Resolved by changing PageSize)"); + + cupsFreeOptions(num_options, options); + + fputs("cupsResolveConflicts(No option/choice): ", stdout); + num_options = 0; + options = NULL; + if (cupsResolveConflicts(ppd, NULL, NULL, &num_options, &options) && + num_options == 1 && !_cups_strcasecmp(options[0].name, "InputSlot") && + !_cups_strcasecmp(options[0].value, "Tray")) + puts("PASS (Resolved by changing InputSlot)"); + else if (num_options > 0) + { + printf("FAIL (%d options:", num_options); + for (i = 0; i < num_options; i ++) + printf(" %s=%s", options[i].name, options[i].value); + puts(")"); + status ++; + } + else + { + puts("FAIL (Unable to resolve)"); + status ++; + } + cupsFreeOptions(num_options, options); + + fputs("ppdInstallableConflict(): ", stdout); + if (ppdInstallableConflict(ppd, "Duplex", "DuplexNoTumble") && + !ppdInstallableConflict(ppd, "Duplex", "None")) + puts("PASS"); + else if (!ppdInstallableConflict(ppd, "Duplex", "DuplexNoTumble")) + { + puts("FAIL (Duplex=DuplexNoTumble did not conflict)"); + status ++; + } + else + { + puts("FAIL (Duplex=None conflicted)"); + status ++; + } + + /* + * ppdPageSizeLimits + */ + + fputs("ppdPageSizeLimits: ", stdout); + if (ppdPageSizeLimits(ppd, &minsize, &maxsize)) + { + if (minsize.width != 36 || minsize.length != 36 || + maxsize.width != 1080 || maxsize.length != 86400) + { + printf("FAIL (got min=%.0fx%.0f, max=%.0fx%.0f, " + "expected min=36x36, max=1080x86400)\n", minsize.width, + minsize.length, maxsize.width, maxsize.length); + status ++; + } + else + puts("PASS"); + } + else + { + puts("FAIL (returned 0)"); + status ++; + } + + /* + * cupsMarkOptions with PWG and IPP size names. + */ + + fputs("cupsMarkOptions(media=iso-a4): ", stdout); + num_options = cupsAddOption("media", "iso-a4", 0, &options); + cupsMarkOptions(ppd, num_options, options); + cupsFreeOptions(num_options, options); + + size = ppdPageSize(ppd, NULL); + if (!size || strcmp(size->name, "A4")) + { + printf("FAIL (%s)\n", size ? size->name : "unknown"); + status ++; + } + else + puts("PASS"); + + fputs("cupsMarkOptions(media=na_letter_8.5x11in): ", stdout); + num_options = cupsAddOption("media", "na_letter_8.5x11in", 0, &options); + cupsMarkOptions(ppd, num_options, options); + cupsFreeOptions(num_options, options); + + size = ppdPageSize(ppd, NULL); + if (!size || strcmp(size->name, "Letter")) + { + printf("FAIL (%s)\n", size ? size->name : "unknown"); + status ++; + } + else + puts("PASS"); + + fputs("cupsMarkOptions(media=oe_letter-fullbleed_8.5x11in): ", stdout); + num_options = cupsAddOption("media", "oe_letter-fullbleed_8.5x11in", 0, + &options); + cupsMarkOptions(ppd, num_options, options); + cupsFreeOptions(num_options, options); + + size = ppdPageSize(ppd, NULL); + if (!size || strcmp(size->name, "Letter.Fullbleed")) + { + printf("FAIL (%s)\n", size ? size->name : "unknown"); + status ++; + } + else + puts("PASS"); + + fputs("cupsMarkOptions(media=A4): ", stdout); + num_options = cupsAddOption("media", "A4", 0, &options); + cupsMarkOptions(ppd, num_options, options); + cupsFreeOptions(num_options, options); + + size = ppdPageSize(ppd, NULL); + if (!size || strcmp(size->name, "A4")) + { + printf("FAIL (%s)\n", size ? size->name : "unknown"); + status ++; + } + else + puts("PASS"); + + /* + * Custom sizes... + */ + + fputs("cupsMarkOptions(media=Custom.8x10in): ", stdout); + num_options = cupsAddOption("media", "Custom.8x10in", 0, &options); + cupsMarkOptions(ppd, num_options, options); + cupsFreeOptions(num_options, options); + + size = ppdPageSize(ppd, NULL); + if (!size || strcmp(size->name, "Custom") || + size->width != 576 || size->length != 720) + { + printf("FAIL (%s - %gx%g)\n", size ? size->name : "unknown", + size ? size->width : 0.0, size ? size->length : 0.0); + status ++; + } + else + puts("PASS"); + + /* + * Test localization... + */ + + fputs("ppdLocalizeIPPReason(text): ", stdout); + if (ppdLocalizeIPPReason(ppd, "foo", NULL, buffer, sizeof(buffer)) && + !strcmp(buffer, "Foo Reason")) + puts("PASS"); + else + { + status ++; + printf("FAIL (\"%s\" instead of \"Foo Reason\")\n", buffer); + } + + fputs("ppdLocalizeIPPReason(http): ", stdout); + if (ppdLocalizeIPPReason(ppd, "foo", "http", buffer, sizeof(buffer)) && + !strcmp(buffer, "http://foo/bar.html")) + puts("PASS"); + else + { + status ++; + printf("FAIL (\"%s\" instead of \"http://foo/bar.html\")\n", buffer); + } + + fputs("ppdLocalizeIPPReason(help): ", stdout); + if (ppdLocalizeIPPReason(ppd, "foo", "help", buffer, sizeof(buffer)) && + !strcmp(buffer, "help:anchor='foo'%20bookID=Vendor%20Help")) + puts("PASS"); + else + { + status ++; + printf("FAIL (\"%s\" instead of \"help:anchor='foo'%%20bookID=Vendor%%20Help\")\n", buffer); + } + + fputs("ppdLocalizeIPPReason(file): ", stdout); + if (ppdLocalizeIPPReason(ppd, "foo", "file", buffer, sizeof(buffer)) && + !strcmp(buffer, "/help/foo/bar.html")) + puts("PASS"); + else + { + status ++; + printf("FAIL (\"%s\" instead of \"/help/foo/bar.html\")\n", buffer); + } + + putenv("LANG=fr"); + putenv("LC_ALL=fr"); + putenv("LC_CTYPE=fr"); + putenv("LC_MESSAGES=fr"); + + fputs("ppdLocalizeIPPReason(fr text): ", stdout); + if (ppdLocalizeIPPReason(ppd, "foo", NULL, buffer, sizeof(buffer)) && + !strcmp(buffer, "La Long Foo Reason")) + puts("PASS"); + else + { + status ++; + printf("FAIL (\"%s\" instead of \"La Long Foo Reason\")\n", buffer); + } + + putenv("LANG=zh_TW"); + putenv("LC_ALL=zh_TW"); + putenv("LC_CTYPE=zh_TW"); + putenv("LC_MESSAGES=zh_TW"); + + fputs("ppdLocalizeIPPReason(zh_TW text): ", stdout); + if (ppdLocalizeIPPReason(ppd, "foo", NULL, buffer, sizeof(buffer)) && + !strcmp(buffer, "Number 1 Foo Reason")) + puts("PASS"); + else + { + status ++; + printf("FAIL (\"%s\" instead of \"Number 1 Foo Reason\")\n", buffer); + } + + /* + * cupsMarkerName localization... + */ + + putenv("LANG=en"); + putenv("LC_ALL=en"); + putenv("LC_CTYPE=en"); + putenv("LC_MESSAGES=en"); + + fputs("ppdLocalizeMarkerName(bogus): ", stdout); + + if ((text = ppdLocalizeMarkerName(ppd, "bogus")) != NULL) + { + status ++; + printf("FAIL (\"%s\" instead of NULL)\n", text); + } + else + puts("PASS"); + + fputs("ppdLocalizeMarkerName(cyan): ", stdout); + + if ((text = ppdLocalizeMarkerName(ppd, "cyan")) != NULL && + !strcmp(text, "Cyan Toner")) + puts("PASS"); + else + { + status ++; + printf("FAIL (\"%s\" instead of \"Cyan Toner\")\n", + text ? text : "(null)"); + } + + putenv("LANG=fr"); + putenv("LC_ALL=fr"); + putenv("LC_CTYPE=fr"); + putenv("LC_MESSAGES=fr"); + + fputs("ppdLocalizeMarkerName(fr cyan): ", stdout); + if ((text = ppdLocalizeMarkerName(ppd, "cyan")) != NULL && + !strcmp(text, "La Toner Cyan")) + puts("PASS"); + else + { + status ++; + printf("FAIL (\"%s\" instead of \"La Toner Cyan\")\n", + text ? text : "(null)"); + } + + putenv("LANG=zh_TW"); + putenv("LC_ALL=zh_TW"); + putenv("LC_CTYPE=zh_TW"); + putenv("LC_MESSAGES=zh_TW"); + + fputs("ppdLocalizeMarkerName(zh_TW cyan): ", stdout); + if ((text = ppdLocalizeMarkerName(ppd, "cyan")) != NULL && + !strcmp(text, "Number 1 Cyan Toner")) + puts("PASS"); + else + { + status ++; + printf("FAIL (\"%s\" instead of \"Number 1 Cyan Toner\")\n", + text ? text : "(null)"); + } + + ppdClose(ppd); + + /* + * Test new constraints... + */ + + fputs("ppdOpenFile(test2.ppd): ", stdout); + + if ((ppd = ppdOpenFile("test2.ppd")) != NULL) + puts("PASS"); + else + { + ppd_status_t err; /* Last error in file */ + int line; /* Line number in file */ + + + status ++; + err = ppdLastError(&line); + + printf("FAIL (%s on line %d)\n", ppdErrorString(err), line); + } + + fputs("ppdMarkDefaults: ", stdout); + ppdMarkDefaults(ppd); + + if ((conflicts = ppdConflicts(ppd)) == 0) + puts("PASS"); + else + { + status ++; + printf("FAIL (%d conflicts)\n", conflicts); + } + + fputs("ppdEmitString (defaults): ", stdout); + if ((s = ppdEmitString(ppd, PPD_ORDER_ANY, 0.0)) != NULL && + !strcmp(s, default2_code)) + puts("PASS"); + else + { + status ++; + printf("FAIL (%d bytes instead of %d)\n", s ? (int)strlen(s) : 0, + (int)strlen(default2_code)); + + if (s) + puts(s); + } + + if (s) + free(s); + + fputs("ppdConflicts(): ", stdout); + ppdMarkOption(ppd, "PageSize", "Env10"); + ppdMarkOption(ppd, "InputSlot", "Envelope"); + ppdMarkOption(ppd, "Quality", "Photo"); + + if ((conflicts = ppdConflicts(ppd)) == 1) + puts("PASS (1)"); + else + { + printf("FAIL (%d)\n", conflicts); + status ++; + } + + fputs("cupsResolveConflicts(Quality=Photo): ", stdout); + num_options = 0; + options = NULL; + if (cupsResolveConflicts(ppd, "Quality", "Photo", &num_options, + &options)) + { + printf("FAIL (%d options:", num_options); + for (i = 0; i < num_options; i ++) + printf(" %s=%s", options[i].name, options[i].value); + puts(")"); + status ++; + } + else + puts("PASS (Unable to resolve)"); + cupsFreeOptions(num_options, options); + + fputs("cupsResolveConflicts(No option/choice): ", stdout); + num_options = 0; + options = NULL; + if (cupsResolveConflicts(ppd, NULL, NULL, &num_options, &options) && + num_options == 1 && !_cups_strcasecmp(options->name, "Quality") && + !_cups_strcasecmp(options->value, "Normal")) + puts("PASS"); + else if (num_options > 0) + { + printf("FAIL (%d options:", num_options); + for (i = 0; i < num_options; i ++) + printf(" %s=%s", options[i].name, options[i].value); + puts(")"); + status ++; + } + else + { + puts("FAIL (Unable to resolve!)"); + status ++; + } + cupsFreeOptions(num_options, options); + + fputs("cupsResolveConflicts(loop test): ", stdout); + ppdMarkOption(ppd, "PageSize", "A4"); + ppdMarkOption(ppd, "InputSlot", "Tray"); + ppdMarkOption(ppd, "Quality", "Photo"); + num_options = 0; + options = NULL; + if (!cupsResolveConflicts(ppd, NULL, NULL, &num_options, &options)) + puts("PASS"); + else if (num_options > 0) + { + printf("FAIL (%d options:", num_options); + for (i = 0; i < num_options; i ++) + printf(" %s=%s", options[i].name, options[i].value); + puts(")"); + } + else + puts("FAIL (No conflicts!)"); + + fputs("ppdInstallableConflict(): ", stdout); + if (ppdInstallableConflict(ppd, "Duplex", "DuplexNoTumble") && + !ppdInstallableConflict(ppd, "Duplex", "None")) + puts("PASS"); + else if (!ppdInstallableConflict(ppd, "Duplex", "DuplexNoTumble")) + { + puts("FAIL (Duplex=DuplexNoTumble did not conflict)"); + status ++; + } + else + { + puts("FAIL (Duplex=None conflicted)"); + status ++; + } + + /* + * ppdPageSizeLimits + */ + + ppdMarkDefaults(ppd); + + fputs("ppdPageSizeLimits(default): ", stdout); + if (ppdPageSizeLimits(ppd, &minsize, &maxsize)) + { + if (minsize.width != 36 || minsize.length != 36 || + maxsize.width != 1080 || maxsize.length != 86400) + { + printf("FAIL (got min=%.0fx%.0f, max=%.0fx%.0f, " + "expected min=36x36, max=1080x86400)\n", minsize.width, + minsize.length, maxsize.width, maxsize.length); + status ++; + } + else + puts("PASS"); + } + else + { + puts("FAIL (returned 0)"); + status ++; + } + + ppdMarkOption(ppd, "InputSlot", "Manual"); + + fputs("ppdPageSizeLimits(InputSlot=Manual): ", stdout); + if (ppdPageSizeLimits(ppd, &minsize, &maxsize)) + { + if (minsize.width != 100 || minsize.length != 100 || + maxsize.width != 1000 || maxsize.length != 1000) + { + printf("FAIL (got min=%.0fx%.0f, max=%.0fx%.0f, " + "expected min=100x100, max=1000x1000)\n", minsize.width, + minsize.length, maxsize.width, maxsize.length); + status ++; + } + else + puts("PASS"); + } + else + { + puts("FAIL (returned 0)"); + status ++; + } + + ppdMarkOption(ppd, "Quality", "Photo"); + + fputs("ppdPageSizeLimits(Quality=Photo): ", stdout); + if (ppdPageSizeLimits(ppd, &minsize, &maxsize)) + { + if (minsize.width != 200 || minsize.length != 200 || + maxsize.width != 1000 || maxsize.length != 1000) + { + printf("FAIL (got min=%.0fx%.0f, max=%.0fx%.0f, " + "expected min=200x200, max=1000x1000)\n", minsize.width, + minsize.length, maxsize.width, maxsize.length); + status ++; + } + else + puts("PASS"); + } + else + { + puts("FAIL (returned 0)"); + status ++; + } + + ppdMarkOption(ppd, "InputSlot", "Tray"); + + fputs("ppdPageSizeLimits(Quality=Photo): ", stdout); + if (ppdPageSizeLimits(ppd, &minsize, &maxsize)) + { + if (minsize.width != 300 || minsize.length != 300 || + maxsize.width != 1080 || maxsize.length != 86400) + { + printf("FAIL (got min=%.0fx%.0f, max=%.0fx%.0f, " + "expected min=300x300, max=1080x86400)\n", minsize.width, + minsize.length, maxsize.width, maxsize.length); + status ++; + } + else + puts("PASS"); + } + else + { + puts("FAIL (returned 0)"); + status ++; + } + } + else + { + const char *filename; /* PPD filename */ + struct stat fileinfo; /* File information */ + + + if (!strncmp(argv[1], "-d", 2)) + { + const char *printer; /* Printer name */ + + if (argv[1][2]) + printer = argv[1] + 2; + else if (argv[2]) + printer = argv[2]; + else + { + puts("Usage: ./testppd -d printer"); + return (1); + } + + filename = cupsGetPPD(printer); + + if (!filename) + { + printf("%s: %s\n", printer, cupsLastErrorString()); + return (1); + } + } + else + filename = argv[1]; + + if (lstat(filename, &fileinfo)) + { + printf("%s: %s\n", filename, strerror(errno)); + return (1); + } + + if (S_ISLNK(fileinfo.st_mode)) + { + char realfile[1024]; /* Real file path */ + ssize_t realsize; /* Size of real file path */ + + + if ((realsize = readlink(filename, realfile, sizeof(realfile) - 1)) < 0) + strcpy(realfile, "Unknown"); + else + realfile[realsize] = '\0'; + + if (stat(realfile, &fileinfo)) + printf("%s: symlink to \"%s\", %s\n", filename, realfile, + strerror(errno)); + else + printf("%s: symlink to \"%s\", %ld bytes\n", filename, realfile, + (long)fileinfo.st_size); + } + else + printf("%s: regular file, %ld bytes\n", filename, (long)fileinfo.st_size); + + if ((ppd = ppdOpenFile(filename)) == NULL) + { + ppd_status_t err; /* Last error in file */ + int line; /* Line number in file */ + + + status ++; + err = ppdLastError(&line); + + printf("%s: %s on line %d\n", argv[1], ppdErrorString(err), line); + } + else + { + int j, k; /* Looping vars */ + ppd_group_t *group; /* Option group */ + ppd_option_t *option; /* Option */ + ppd_coption_t *coption; /* Custom option */ + ppd_cparam_t *cparam; /* Custom parameter */ + ppd_const_t *c; /* UIConstraints */ + char lang[255], /* LANG environment variable */ + lc_all[255], /* LC_ALL environment variable */ + lc_ctype[255], /* LC_CTYPE environment variable */ + lc_messages[255];/* LC_MESSAGES environment variable */ + + + if (argc > 2) + { + snprintf(lang, sizeof(lang), "LANG=%s", argv[2]); + putenv(lang); + snprintf(lc_all, sizeof(lc_all), "LC_ALL=%s", argv[2]); + putenv(lc_all); + snprintf(lc_ctype, sizeof(lc_ctype), "LC_CTYPE=%s", argv[2]); + putenv(lc_ctype); + snprintf(lc_messages, sizeof(lc_messages), "LC_MESSAGES=%s", argv[2]); + putenv(lc_messages); + } + + ppdLocalize(ppd); + ppdMarkDefaults(ppd); + + if (argc > 3) + { + text = ppdLocalizeIPPReason(ppd, argv[3], NULL, buffer, sizeof(buffer)); + printf("ppdLocalizeIPPReason(%s)=%s\n", argv[3], + text ? text : "(null)"); + return (text == NULL); + } + + for (i = ppd->num_groups, group = ppd->groups; + i > 0; + i --, group ++) + { + printf("%s (%s):\n", group->name, group->text); + + for (j = group->num_options, option = group->options; + j > 0; + j --, option ++) + { + printf(" %s (%s):\n", option->keyword, option->text); + + for (k = 0; k < option->num_choices; k ++) + printf(" - %s%s (%s)\n", + option->choices[k].marked ? "*" : "", + option->choices[k].choice, option->choices[k].text); + + if ((coption = ppdFindCustomOption(ppd, option->keyword)) != NULL) + { + for (cparam = (ppd_cparam_t *)cupsArrayFirst(coption->params); + cparam; + cparam = (ppd_cparam_t *)cupsArrayNext(coption->params)) + { + switch (cparam->type) + { + case PPD_CUSTOM_CURVE : + printf(" %s(%s): PPD_CUSTOM_CURVE (%g to %g)\n", + cparam->name, cparam->text, + cparam->minimum.custom_curve, + cparam->maximum.custom_curve); + break; + + case PPD_CUSTOM_INT : + printf(" %s(%s): PPD_CUSTOM_INT (%d to %d)\n", + cparam->name, cparam->text, + cparam->minimum.custom_int, + cparam->maximum.custom_int); + break; + + case PPD_CUSTOM_INVCURVE : + printf(" %s(%s): PPD_CUSTOM_INVCURVE (%g to %g)\n", + cparam->name, cparam->text, + cparam->minimum.custom_invcurve, + cparam->maximum.custom_invcurve); + break; + + case PPD_CUSTOM_PASSCODE : + printf(" %s(%s): PPD_CUSTOM_PASSCODE (%d to %d)\n", + cparam->name, cparam->text, + cparam->minimum.custom_passcode, + cparam->maximum.custom_passcode); + break; + + case PPD_CUSTOM_PASSWORD : + printf(" %s(%s): PPD_CUSTOM_PASSWORD (%d to %d)\n", + cparam->name, cparam->text, + cparam->minimum.custom_password, + cparam->maximum.custom_password); + break; + + case PPD_CUSTOM_POINTS : + printf(" %s(%s): PPD_CUSTOM_POINTS (%g to %g)\n", + cparam->name, cparam->text, + cparam->minimum.custom_points, + cparam->maximum.custom_points); + break; + + case PPD_CUSTOM_REAL : + printf(" %s(%s): PPD_CUSTOM_REAL (%g to %g)\n", + cparam->name, cparam->text, + cparam->minimum.custom_real, + cparam->maximum.custom_real); + break; + + case PPD_CUSTOM_STRING : + printf(" %s(%s): PPD_CUSTOM_STRING (%d to %d)\n", + cparam->name, cparam->text, + cparam->minimum.custom_string, + cparam->maximum.custom_string); + break; + } + } + } + } + } + + puts("\nSizes:"); + for (i = ppd->num_sizes, size = ppd->sizes; i > 0; i --, size ++) + printf(" %s = %gx%g, [%g %g %g %g]\n", size->name, size->width, + size->length, size->left, size->bottom, size->right, size->top); + + puts("\nConstraints:"); + + for (i = ppd->num_consts, c = ppd->consts; i > 0; i --, c ++) + printf(" *UIConstraints: *%s %s *%s %s\n", c->option1, c->choice1, + c->option2, c->choice2); + if (ppd->num_consts == 0) + puts(" NO CONSTRAINTS"); + + puts("\nFilters:"); + + for (i = 0; i < ppd->num_filters; i ++) + printf(" %s\n", ppd->filters[i]); + + if (ppd->num_filters == 0) + puts(" NO FILTERS"); + + puts("\nAttributes:"); + + for (attr = (ppd_attr_t *)cupsArrayFirst(ppd->sorted_attrs); + attr; + attr = (ppd_attr_t *)cupsArrayNext(ppd->sorted_attrs)) + printf(" *%s %s/%s: \"%s\"\n", attr->name, attr->spec, + attr->text, attr->value ? attr->value : ""); + } + + if (!strncmp(argv[1], "-d", 2)) + unlink(filename); + } + +#ifdef __APPLE__ + if (getenv("MallocStackLogging") && getenv("MallocStackLoggingNoCompact")) + { + char command[1024]; /* malloc_history command */ + + snprintf(command, sizeof(command), "malloc_history %d -all_by_size", + getpid()); + fflush(stdout); + system(command); + } +#endif /* __APPLE__ */ + + ppdClose(ppd); + + return (status); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/testpwg.c b/cups/testpwg.c new file mode 100644 index 0000000000..756baf0d86 --- /dev/null +++ b/cups/testpwg.c @@ -0,0 +1,497 @@ +/* + * "$Id$" + * + * PWG test program for CUPS. + * + * Copyright 2009-2011 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Main entry. + * test_pagesize() - Test the PWG mapping functions. + * test_ppd_cache() - Test the PPD cache functions. + */ + +/* + * Include necessary headers... + */ + +#include "ppd-private.h" +#include "file-private.h" + + +/* + * Local functions... + */ + +static int test_pagesize(_ppd_cache_t *pc, ppd_file_t *ppd, + const char *ppdsize); +static int test_ppd_cache(_ppd_cache_t *pc, ppd_file_t *ppd); + + +/* + * 'main()' - Main entry. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int status; /* Status of tests (0 = success, 1 = fail) */ + const char *ppdfile; /* PPD filename */ + ppd_file_t *ppd; /* PPD file */ + _ppd_cache_t *pc; /* PPD cache and PWG mapping data */ + _pwg_media_t *pwgmedia; /* PWG media size */ + + + status = 0; + + if (argc < 2 || argc > 3) + { + puts("Usage: ./testpwg filename.ppd [jobfile]"); + return (1); + } + + ppdfile = argv[1]; + + printf("ppdOpenFile(%s): ", ppdfile); + if ((ppd = ppdOpenFile(ppdfile)) == NULL) + { + ppd_status_t err; /* Last error in file */ + int line; /* Line number in file */ + + + err = ppdLastError(&line); + + printf("FAIL (%s on line %d)\n", ppdErrorString(err), line); + + return (1); + } + else + puts("PASS"); + + fputs("_ppdCacheCreateWithPPD(ppd): ", stdout); + if ((pc = _ppdCacheCreateWithPPD(ppd)) == NULL) + { + puts("FAIL"); + status ++; + } + else + { + puts("PASS"); + status += test_ppd_cache(pc, ppd); + + if (argc == 3) + { + /* + * Test PageSize mapping code. + */ + + int fd; /* Job file descriptor */ + const char *pagesize; /* PageSize value */ + ipp_t *job; /* Job attributes */ + ipp_attribute_t *media; /* Media attribute */ + + if ((fd = open(argv[2], O_RDONLY)) >= 0) + { + job = ippNew(); + ippReadFile(fd, job); + close(fd); + + if ((media = ippFindAttribute(job, "media", IPP_TAG_ZERO)) != NULL && + media->value_tag != IPP_TAG_NAME && + media->value_tag != IPP_TAG_KEYWORD) + media = NULL; + + if (media) + printf("_ppdCacheGetPageSize(media=%s): ", + media->values[0].string.text); + else + fputs("_ppdCacheGetPageSize(media-col): ", stdout); + + fflush(stdout); + + if ((pagesize = _ppdCacheGetPageSize(pc, job, NULL, NULL)) == NULL) + { + puts("FAIL (Not Found)"); + status = 1; + } + else if (media && _cups_strcasecmp(pagesize, media->values[0].string.text)) + { + printf("FAIL (Got \"%s\", Expected \"%s\")\n", pagesize, + media->values[0].string.text); + status = 1; + } + else + printf("PASS (%s)\n", pagesize); + + ippDelete(job); + } + else + { + perror(argv[2]); + status = 1; + } + } + + /* + * _ppdCacheDestroy should never fail... + */ + + fputs("_ppdCacheDestroy(pc): ", stdout); + _ppdCacheDestroy(pc); + puts("PASS"); + } + + fputs("_pwgMediaForPWG(\"iso_a4_210x297mm\"): ", stdout); + if ((pwgmedia = _pwgMediaForPWG("iso_a4_210x297mm")) == NULL) + { + puts("FAIL (not found)"); + status ++; + } + else if (strcmp(pwgmedia->pwg, "iso_a4_210x297mm")) + { + printf("FAIL (%s)\n", pwgmedia->pwg); + status ++; + } + else if (pwgmedia->width != 21000 || pwgmedia->length != 29700) + { + printf("FAIL (%dx%d)\n", pwgmedia->width, pwgmedia->length); + status ++; + } + else + puts("PASS"); + + fputs("_pwgMediaForLegacy(\"na-letter\"): ", stdout); + if ((pwgmedia = _pwgMediaForLegacy("na-letter")) == NULL) + { + puts("FAIL (not found)"); + status ++; + } + else if (strcmp(pwgmedia->pwg, "na_letter_8.5x11in")) + { + printf("FAIL (%s)\n", pwgmedia->pwg); + status ++; + } + else if (pwgmedia->width != 21590 || pwgmedia->length != 27940) + { + printf("FAIL (%dx%d)\n", pwgmedia->width, pwgmedia->length); + status ++; + } + else + puts("PASS"); + + fputs("_pwgMediaForPPD(\"4x6\"): ", stdout); + if ((pwgmedia = _pwgMediaForPPD("4x6")) == NULL) + { + puts("FAIL (not found)"); + status ++; + } + else if (strcmp(pwgmedia->pwg, "na_index-4x6_4x6in")) + { + printf("FAIL (%s)\n", pwgmedia->pwg); + status ++; + } + else if (pwgmedia->width != 10160 || pwgmedia->length != 15240) + { + printf("FAIL (%dx%d)\n", pwgmedia->width, pwgmedia->length); + status ++; + } + else + puts("PASS"); + + fputs("_pwgMediaForPPD(\"10x15cm\"): ", stdout); + if ((pwgmedia = _pwgMediaForPPD("10x15cm")) == NULL) + { + puts("FAIL (not found)"); + status ++; + } + else if (strcmp(pwgmedia->pwg, "om_100x150mm_100x150mm")) + { + printf("FAIL (%s)\n", pwgmedia->pwg); + status ++; + } + else if (pwgmedia->width != 10000 || pwgmedia->length != 15000) + { + printf("FAIL (%dx%d)\n", pwgmedia->width, pwgmedia->length); + status ++; + } + else + puts("PASS"); + + fputs("_pwgMediaForPPD(\"Custom.10x15cm\"): ", stdout); + if ((pwgmedia = _pwgMediaForPPD("Custom.10x15cm")) == NULL) + { + puts("FAIL (not found)"); + status ++; + } + else if (strcmp(pwgmedia->pwg, "custom_10x15cm_100x150mm")) + { + printf("FAIL (%s)\n", pwgmedia->pwg); + status ++; + } + else if (pwgmedia->width != 10000 || pwgmedia->length != 15000) + { + printf("FAIL (%dx%d)\n", pwgmedia->width, pwgmedia->length); + status ++; + } + else + puts("PASS"); + + fputs("_pwgMediaForSize(29700, 42000): ", stdout); + if ((pwgmedia = _pwgMediaForSize(29700, 42000)) == NULL) + { + puts("FAIL (not found)"); + status ++; + } + else if (strcmp(pwgmedia->pwg, "iso_a3_297x420mm")) + { + printf("FAIL (%s)\n", pwgmedia->pwg); + status ++; + } + else + puts("PASS"); + + return (status); +} + + +/* + * 'test_pagesize()' - Test the PWG mapping functions. + */ + +static int /* O - 1 on failure, 0 on success */ +test_pagesize(_ppd_cache_t *pc, /* I - PWG mapping data */ + ppd_file_t *ppd, /* I - PPD file */ + const char *ppdsize) /* I - PPD page size */ +{ + int status = 0; /* Return status */ + ipp_t *job; /* Job attributes */ + const char *pagesize; /* PageSize value */ + + + if (ppdPageSize(ppd, ppdsize)) + { + printf("_ppdCacheGetPageSize(keyword=%s): ", ppdsize); + fflush(stdout); + + if ((pagesize = _ppdCacheGetPageSize(pc, NULL, ppdsize, NULL)) == NULL) + { + puts("FAIL (Not Found)"); + status = 1; + } + else if (_cups_strcasecmp(pagesize, ppdsize)) + { + printf("FAIL (Got \"%s\", Expected \"%s\")\n", pagesize, ppdsize); + status = 1; + } + else + puts("PASS"); + + job = ippNew(); + ippAddString(job, IPP_TAG_JOB, IPP_TAG_KEYWORD, "media", NULL, ppdsize); + + printf("_ppdCacheGetPageSize(media=%s): ", ppdsize); + fflush(stdout); + + if ((pagesize = _ppdCacheGetPageSize(pc, job, NULL, NULL)) == NULL) + { + puts("FAIL (Not Found)"); + status = 1; + } + else if (_cups_strcasecmp(pagesize, ppdsize)) + { + printf("FAIL (Got \"%s\", Expected \"%s\")\n", pagesize, ppdsize); + status = 1; + } + else + puts("PASS"); + + ippDelete(job); + } + + return (status); +} + + +/* + * 'test_ppd_cache()' - Test the PPD cache functions. + */ + +static int /* O - 1 on failure, 0 on success */ +test_ppd_cache(_ppd_cache_t *pc, /* I - PWG mapping data */ + ppd_file_t *ppd) /* I - PPD file */ +{ + int i, /* Looping var */ + status = 0; /* Return status */ + _ppd_cache_t *pc2; /* Loaded data */ + _pwg_size_t *size, /* Size from original */ + *size2; /* Size from saved */ + _pwg_map_t *map, /* Map from original */ + *map2; /* Map from saved */ + + + /* + * Verify that we can write and read back the same data... + */ + + fputs("_ppdCacheWriteFile(test.pwg): ", stdout); + if (!_ppdCacheWriteFile(pc, "test.pwg", NULL)) + { + puts("FAIL"); + status ++; + } + else + puts("PASS"); + + fputs("_ppdCacheCreateWithFile(test.pwg): ", stdout); + if ((pc2 = _ppdCacheCreateWithFile("test.pwg", NULL)) == NULL) + { + puts("FAIL"); + status ++; + } + else + { + // TODO: FINISH ADDING ALL VALUES IN STRUCTURE + if (pc2->num_sizes != pc->num_sizes) + { + if (!status) + puts("FAIL"); + + printf(" SAVED num_sizes=%d, ORIG num_sizes=%d\n", pc2->num_sizes, + pc->num_sizes); + + status ++; + } + else + { + for (i = pc->num_sizes, size = pc->sizes, size2 = pc2->sizes; + i > 0; + i --, size ++, size2 ++) + { + if (strcmp(size2->map.pwg, size->map.pwg) || + strcmp(size2->map.ppd, size->map.ppd) || + size2->width != size->width || + size2->length != size->length || + size2->left != size->left || + size2->bottom != size->bottom || + size2->right != size->right || + size2->top != size->top) + { + if (!status) + puts("FAIL"); + + if (strcmp(size->map.pwg, size2->map.pwg)) + printf(" SAVED size->map.pwg=\"%s\", ORIG " + "size->map.pwg=\"%s\"\n", size2->map.pwg, size->map.pwg); + + if (strcmp(size2->map.ppd, size->map.ppd)) + printf(" SAVED size->map.ppd=\"%s\", ORIG " + "size->map.ppd=\"%s\"\n", size2->map.ppd, size->map.ppd); + + if (size2->width != size->width) + printf(" SAVED size->width=%d, ORIG size->width=%d\n", + size2->width, size->width); + + if (size2->length != size->length) + printf(" SAVED size->length=%d, ORIG size->length=%d\n", + size2->length, size->length); + + if (size2->left != size->left) + printf(" SAVED size->left=%d, ORIG size->left=%d\n", + size2->left, size->left); + + if (size2->bottom != size->bottom) + printf(" SAVED size->bottom=%d, ORIG size->bottom=%d\n", + size2->bottom, size->bottom); + + if (size2->right != size->right) + printf(" SAVED size->right=%d, ORIG size->right=%d\n", + size2->right, size->right); + + if (size2->top != size->top) + printf(" SAVED size->top=%d, ORIG size->top=%d\n", + size2->top, size->top); + + status ++; + break; + } + } + + for (i = pc->num_sources, map = pc->sources, map2 = pc2->sources; + i > 0; + i --, map ++, map2 ++) + { + if (strcmp(map2->pwg, map->pwg) || + strcmp(map2->ppd, map->ppd)) + { + if (!status) + puts("FAIL"); + + if (strcmp(map->pwg, map2->pwg)) + printf(" SAVED source->pwg=\"%s\", ORIG source->pwg=\"%s\"\n", + map2->pwg, map->pwg); + + if (strcmp(map2->ppd, map->ppd)) + printf(" SAVED source->ppd=\"%s\", ORIG source->ppd=\"%s\"\n", + map2->ppd, map->ppd); + + status ++; + break; + } + } + + for (i = pc->num_types, map = pc->types, map2 = pc2->types; + i > 0; + i --, map ++, map2 ++) + { + if (strcmp(map2->pwg, map->pwg) || + strcmp(map2->ppd, map->ppd)) + { + if (!status) + puts("FAIL"); + + if (strcmp(map->pwg, map2->pwg)) + printf(" SAVED type->pwg=\"%s\", ORIG type->pwg=\"%s\"\n", + map2->pwg, map->pwg); + + if (strcmp(map2->ppd, map->ppd)) + printf(" SAVED type->ppd=\"%s\", ORIG type->ppd=\"%s\"\n", + map2->ppd, map->ppd); + + status ++; + break; + } + } + } + + if (!status) + puts("PASS"); + + _ppdCacheDestroy(pc2); + } + + /* + * Test PageSize mapping code... + */ + + status += test_pagesize(pc, ppd, "Letter"); + status += test_pagesize(pc, ppd, "na-letter"); + status += test_pagesize(pc, ppd, "A4"); + status += test_pagesize(pc, ppd, "iso-a4"); + + return (status); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/testsnmp.c b/cups/testsnmp.c new file mode 100644 index 0000000000..f1498c8349 --- /dev/null +++ b/cups/testsnmp.c @@ -0,0 +1,304 @@ +/* + * "$Id$" + * + * SNMP test program for CUPS. + * + * Copyright 2008-2010 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Main entry. + * scan_oid() - Scan an OID value. + * show_oid() - Show the specified OID. + * usage() - Show program usage and exit. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include "snmp-private.h" + + +/* + * Local functions... + */ + +static void print_packet(cups_snmp_t *packet, void *data); +static int show_oid(int fd, const char *community, + http_addr_t *addr, const char *s, int walk); +static void usage(void) __attribute__((noreturn)); + + +/* + * 'main()' - Main entry. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + int fd = -1; /* SNMP socket */ + http_addrlist_t *host = NULL; /* Address of host */ + int walk = 0; /* Walk OIDs? */ + char *oid = NULL; /* Last OID shown */ + const char *community; /* Community name */ + + + fputs("_cupsSNMPDefaultCommunity: ", stdout); + + if ((community = _cupsSNMPDefaultCommunity()) == NULL) + { + puts("FAIL (NULL community name)"); + return (1); + } + + printf("PASS (%s)\n", community); + + /* + * Query OIDs from the command-line... + */ + + for (i = 1; i < argc; i ++) + if (!strcmp(argv[i], "-c")) + { + i ++; + + if (i >= argc) + usage(); + else + community = argv[i]; + } + else if (!strcmp(argv[i], "-d")) + _cupsSNMPSetDebug(10); + else if (!strcmp(argv[i], "-w")) + walk = 1; + else if (!host) + { + if ((host = httpAddrGetList(argv[i], AF_UNSPEC, "161")) == NULL) + { + printf("testsnmp: Unable to find \"%s\"!\n", argv[1]); + return (1); + } + + if (fd < 0) + { + fputs("_cupsSNMPOpen: ", stdout); + + if ((fd = _cupsSNMPOpen(host->addr.addr.sa_family)) < 0) + { + printf("FAIL (%s)\n", strerror(errno)); + return (1); + } + + puts("PASS"); + } + } + else if (!show_oid(fd, community, &(host->addr), argv[i], walk)) + return (1); + else + oid = argv[i]; + + if (!host) + usage(); + + if (!oid) + { + if (!show_oid(fd, community, &(host->addr), + walk ? ".1.3.6.1.2.1.43" : + ".1.3.6.1.2.1.43.10.2.1.4.1.1", walk)) + return (1); + } + + return (0); +} + + +/* + * 'print_packet()' - Print the contents of the response packet. + */ + +static void +print_packet(cups_snmp_t *packet, /* I - SNMP response packet */ + void *data) /* I - User data pointer (not used) */ +{ + int i; /* Looping var */ + char temp[1024]; /* Temporary OID string */ + + + (void)data; + + printf("%s = ", _cupsSNMPOIDToString(packet->object_name, temp, sizeof(temp))); + + switch (packet->object_type) + { + case CUPS_ASN1_BOOLEAN : + printf("BOOLEAN %s\n", + packet->object_value.boolean ? "TRUE" : "FALSE"); + break; + + case CUPS_ASN1_INTEGER : + printf("INTEGER %d\n", packet->object_value.integer); + break; + + case CUPS_ASN1_BIT_STRING : + printf("BIT-STRING \"%s\"\n", + (char *)packet->object_value.string.bytes); + break; + + case CUPS_ASN1_OCTET_STRING : + printf("OCTET-STRING \"%s\"\n", + (char *)packet->object_value.string.bytes); + break; + + case CUPS_ASN1_NULL_VALUE : + puts("NULL-VALUE"); + break; + + case CUPS_ASN1_OID : + printf("OID %s\n", _cupsSNMPOIDToString(packet->object_value.oid, + temp, sizeof(temp))); + break; + + case CUPS_ASN1_HEX_STRING : + fputs("Hex-STRING", stdout); + for (i = 0; i < packet->object_value.string.num_bytes; i ++) + printf(" %02X", packet->object_value.string.bytes[i]); + putchar('\n'); + break; + + case CUPS_ASN1_COUNTER : + printf("Counter %d\n", packet->object_value.counter); + break; + + case CUPS_ASN1_GAUGE : + printf("Gauge %u\n", packet->object_value.gauge); + break; + + case CUPS_ASN1_TIMETICKS : + printf("Timeticks %u days, %u:%02u:%02u.%02u\n", + packet->object_value.timeticks / 8640000, + (packet->object_value.timeticks / 360000) % 24, + (packet->object_value.timeticks / 6000) % 60, + (packet->object_value.timeticks / 100) % 60, + packet->object_value.timeticks % 100); + break; + + default : + printf("Unknown-%X\n", packet->object_type); + break; + } +} + + +/* + * 'show_oid()' - Show the specified OID. + */ + +static int /* O - 1 on success, 0 on error */ +show_oid(int fd, /* I - SNMP socket */ + const char *community, /* I - Community name */ + http_addr_t *addr, /* I - Address to query */ + const char *s, /* I - OID to query */ + int walk) /* I - Walk OIDs? */ +{ + int i; /* Looping var */ + int oid[CUPS_SNMP_MAX_OID]; /* OID */ + cups_snmp_t packet; /* SNMP packet */ + char temp[1024]; /* Temporary OID string */ + + + if (!_cupsSNMPStringToOID(s, oid, sizeof(oid) / sizeof(oid[0]))) + { + puts("testsnmp: Bad OID"); + return (0); + } + + if (walk) + { + printf("_cupsSNMPWalk(%s): ", _cupsSNMPOIDToString(oid, temp, sizeof(temp))); + + if (_cupsSNMPWalk(fd, addr, CUPS_SNMP_VERSION_1, community, oid, 5.0, + print_packet, NULL) < 0) + { + printf("FAIL (%s)\n", strerror(errno)); + return (0); + } + } + else + { + printf("_cupsSNMPWrite(%s): ", _cupsSNMPOIDToString(oid, temp, sizeof(temp))); + + if (!_cupsSNMPWrite(fd, addr, CUPS_SNMP_VERSION_1, community, + CUPS_ASN1_GET_REQUEST, 1, oid)) + { + printf("FAIL (%s)\n", strerror(errno)); + return (0); + } + + puts("PASS"); + + fputs("_cupsSNMPRead(5.0): ", stdout); + + if (!_cupsSNMPRead(fd, &packet, 5.0)) + { + puts("FAIL (timeout)"); + return (0); + } + + if (!_cupsSNMPIsOID(&packet, oid)) + { + printf("FAIL (bad OID %d", packet.object_name[0]); + for (i = 1; packet.object_name[i] >= 0; i ++) + printf(".%d", packet.object_name[i]); + puts(")"); + return (0); + } + + if (packet.error) + { + printf("FAIL (%s)\n", packet.error); + return (0); + } + + puts("PASS"); + + print_packet(&packet, NULL); + } + + return (1); +} + + +/* + * 'usage()' - Show program usage and exit. + */ + +static void +usage(void) +{ + puts("Usage: testsnmp [options] host-or-ip [oid ...]"); + puts(""); + puts("Options:"); + puts(""); + puts(" -c community Set community name"); + puts(" -d Enable debugging"); + puts(" -w Walk all OIDs under the specified one"); + + exit (1); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/thread-private.h b/cups/thread-private.h new file mode 100644 index 0000000000..48f8a39a64 --- /dev/null +++ b/cups/thread-private.h @@ -0,0 +1,97 @@ +/* + * "$Id$" + * + * Private threading definitions for CUPS. + * + * Copyright 2009-2010 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#ifndef _CUPS_THREAD_PRIVATE_H_ +# define _CUPS_THREAD_PRIVATE_H_ + +/* + * Include necessary headers... + */ + +# include "config.h" + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +# ifdef HAVE_PTHREAD_H +# include +typedef void *(*_cups_thread_func_t)(void *arg); +typedef pthread_mutex_t _cups_mutex_t; +typedef pthread_rwlock_t _cups_rwlock_t; +typedef pthread_key_t _cups_threadkey_t; +# define _CUPS_MUTEX_INITIALIZER PTHREAD_MUTEX_INITIALIZER +# define _CUPS_RWLOCK_INITIALIZER PTHREAD_RWLOCK_INITIALIZER +# define _CUPS_THREADKEY_INITIALIZER -1 +# define _cupsThreadGetData(k) pthread_getspecific(k) +# define _cupsThreadSetData(k,p) pthread_setspecific(k,p) + +# elif defined(WIN32) +# include +# include +typedef void *(__stdcall *_cups_thread_func_t)(void *arg); +typedef struct _cups_mutex_s +{ + int m_init; /* Flag for on-demand initialization */ + CRITICAL_SECTION m_criticalSection; + /* Win32 Critical Section */ +} _cups_mutex_t; +typedef _cups_mutex_t _cups_rwlock_t; /* TODO: Implement Win32 reader/writer lock */ +typedef DWORD _cups_threadkey_t; +# define _CUPS_MUTEX_INITIALIZER { 0, 0 } +# define _CUPS_RWLOCK_INITIALIZER { 0, 0 } +# define _CUPS_THREADKEY_INITIALIZER 0 +# define _cupsThreadGetData(k) TlsGetValue(k) +# define _cupsThreadSetData(k,p) TlsSetValue(k,p) + +# else +typedef char _cups_mutex_t; +typedef char _cups_rwlock_t; +typedef void *_cups_threadkey_t; +# define _CUPS_MUTEX_INITIALIZER 0 +# define _CUPS_RWLOCK_INITIALIZER 0 +# define _CUPS_THREADKEY_INITIALIZER (void *)0 +# define _cupsThreadGetData(k) k +# define _cupsThreadSetData(k,p) k=p +# endif /* HAVE_PTHREAD_H */ + + +/* + * Functions... + */ + +extern void _cupsMutexInit(_cups_mutex_t *mutex); +extern void _cupsMutexLock(_cups_mutex_t *mutex); +extern void _cupsMutexUnlock(_cups_mutex_t *mutex); +extern void _cupsRWInit(_cups_rwlock_t *rwlock); +extern void _cupsRWLockRead(_cups_rwlock_t *rwlock); +extern void _cupsRWLockWrite(_cups_rwlock_t *rwlock); +extern void _cupsRWUnlock(_cups_rwlock_t *rwlock); +extern int _cupsThreadCreate(_cups_thread_func_t func, void *arg); + + +# ifdef __cplusplus +} +# endif /* __cplusplus */ +#endif /* !_CUPS_THREAD_PRIVATE_H_ */ + +/* + * End of "$Id$". + */ diff --git a/cups/thread.c b/cups/thread.c new file mode 100644 index 0000000000..7574298faf --- /dev/null +++ b/cups/thread.c @@ -0,0 +1,317 @@ +/* + * "$Id$" + * + * Threading primitives for CUPS. + * + * Copyright 2009-2010 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * _cupsMutexInit() - Initialize a mutex. + * _cupsMutexLock() - Lock a mutex. + * _cupsMutexUnlock() - Unlock a mutex. + * _cupsRWInit() - Initialize a reader/writer lock. + * _cupsRWLockRead() - Acquire a reader/writer lock for reading. + * _cupsRWLockWrite() - Acquire a reader/writer lock for writing. + * _cupsRWUnlock() - Release a reader/writer lock. + * _cupsThreadCreate() - Create a thread. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include "thread-private.h" + + +#if defined(HAVE_PTHREAD_H) +/* + * '_cupsMutexInit()' - Initialize a mutex. + */ + +void +_cupsMutexInit(_cups_mutex_t *mutex) /* I - Mutex */ +{ + pthread_mutex_init(mutex, NULL); +} + + +/* + * '_cupsMutexLock()' - Lock a mutex. + */ + +void +_cupsMutexLock(_cups_mutex_t *mutex) /* I - Mutex */ +{ + pthread_mutex_lock(mutex); +} + + +/* + * '_cupsMutexUnlock()' - Unlock a mutex. + */ + +void +_cupsMutexUnlock(_cups_mutex_t *mutex) /* I - Mutex */ +{ + pthread_mutex_unlock(mutex); +} + + +/* + * '_cupsRWInit()' - Initialize a reader/writer lock. + */ + +void +_cupsRWInit(_cups_rwlock_t *rwlock) /* I - Reader/writer lock */ +{ + pthread_rwlock_init(rwlock, NULL); +} + + +/* + * '_cupsRWLockRead()' - Acquire a reader/writer lock for reading. + */ + +void +_cupsRWLockRead(_cups_rwlock_t *rwlock) /* I - Reader/writer lock */ +{ + pthread_rwlock_rdlock(rwlock); +} + + +/* + * '_cupsRWLockWrite()' - Acquire a reader/writer lock for writing. + */ + +void +_cupsRWLockWrite(_cups_rwlock_t *rwlock)/* I - Reader/writer lock */ +{ + pthread_rwlock_wrlock(rwlock); +} + + +/* + * '_cupsRWUnlock()' - Release a reader/writer lock. + */ + +void +_cupsRWUnlock(_cups_rwlock_t *rwlock) /* I - Reader/writer lock */ +{ + pthread_rwlock_unlock(rwlock); +} + + +/* + * '_cupsThreadCreate()' - Create a thread. + */ + +int /* O - 0 on failure, 1 on success */ +_cupsThreadCreate( + _cups_thread_func_t func, /* I - Entry point */ + void *arg) /* I - Entry point context */ +{ + pthread_t thread; + + return (pthread_create(&thread, NULL, (void *(*)(void *))func, arg) == 0); +} + + +#elif defined(WIN32) +# include + + +/* + * '_cupsMutexInit()' - Initialize a mutex. + */ + +void +_cupsMutexInit(_cups_mutex_t *mutex) /* I - Mutex */ +{ + InitializeCriticalSection(&mutex->m_criticalSection); + mutex->m_init = 1; +} + + +/* + * '_cupsMutexLock()' - Lock a mutex. + */ + +void +_cupsMutexLock(_cups_mutex_t *mutex) /* I - Mutex */ +{ + if (!mutex->m_init) + { + _cupsGlobalLock(); + + if (!mutex->m_init) + { + InitializeCriticalSection(&mutex->m_criticalSection); + mutex->m_init = 1; + } + + _cupsGlobalUnlock(); + } + + EnterCriticalSection(&mutex->m_criticalSection); +} + + +/* + * '_cupsMutexUnlock()' - Unlock a mutex. + */ + +void +_cupsMutexUnlock(_cups_mutex_t *mutex) /* I - Mutex */ +{ + LeaveCriticalSection(&mutex->m_criticalSection); +} + + +/* + * '_cupsRWInit()' - Initialize a reader/writer lock. + */ + +void +_cupsRWInit(_cups_rwlock_t *rwlock) /* I - Reader/writer lock */ +{ + _cupsMutexInit((_cups_mutex_t *)rwlock); +} + + +/* + * '_cupsRWLockRead()' - Acquire a reader/writer lock for reading. + */ + +void +_cupsRWLockRead(_cups_rwlock_t *rwlock) /* I - Reader/writer lock */ +{ + _cupsMutexLock((_cups_mutex_t *)rwlock); +} + + +/* + * '_cupsRWLockWrite()' - Acquire a reader/writer lock for writing. + */ + +void +_cupsRWLockWrite(_cups_rwlock_t *rwlock)/* I - Reader/writer lock */ +{ + _cupsMutexLock((_cups_mutex_t *)rwlock); +} + + +/* + * '_cupsRWUnlock()' - Release a reader/writer lock. + */ + +void +_cupsRWUnlock(_cups_rwlock_t *rwlock) /* I - Reader/writer lock */ +{ + _cupsMutexUnlock((_cups_mutex_t *)rwlock); +} + + +/* + * '_cupsThreadCreate()' - Create a thread. + */ + +int /* O - 0 on failure, 1 on success */ +_cupsThreadCreate( + _cups_thread_func_t func, /* I - Entry point */ + void *arg) /* I - Entry point context */ +{ + return (_beginthreadex(NULL, 0, (LPTHREAD_START_ROUTINE) func, arg, 0, NULL) + != 0); +} + + +#else +/* + * '_cupsMutexInit()' - Initialize a mutex. + */ + +void +_cupsMutexInit(_cups_mutex_t *mutex) /* I - Mutex */ +{ + (void)mutex; +} + + +/* + * '_cupsMutexLock()' - Lock a mutex. + */ + +void +_cupsMutexLock(_cups_mutex_t *mutex) /* I - Mutex */ +{ + (void)mutex; +} + + +/* + * '_cupsMutexUnlock()' - Unlock a mutex. + */ + +void +_cupsMutexUnlock(_cups_mutex_t *mutex) /* I - Mutex */ +{ + (void)mutex; +} + + +/* + * '_cupsRWInit()' - Initialize a reader/writer lock. + */ + +void +_cupsRWInit(_cups_rwlock_t *rwlock) /* I - Reader/writer lock */ +{ + (void)rwlock; +} + + +/* + * '_cupsRWLockRead()' - Acquire a reader/writer lock for reading. + */ + +void +_cupsRWLockRead(_cups_rwlock_t *rwlock) /* I - Reader/writer lock */ +{ + (void)rwlock; +} + + +/* + * '_cupsRWLockWrite()' - Acquire a reader/writer lock for writing. + */ + +void +_cupsRWLockWrite(_cups_rwlock_t *rwlock)/* I - Reader/writer lock */ +{ + (void)rwlock; +} + + +/* + * '_cupsRWUnlock()' - Release a reader/writer lock. + */ + +void +_cupsRWUnlock(_cups_rwlock_t *rwlock) /* I - Reader/writer lock */ +{ + (void)rwlock; +} +#endif /* HAVE_PTHREAD_H */ + + +/* + * End of "$Id$". + */ diff --git a/cups/transcode.c b/cups/transcode.c new file mode 100644 index 0000000000..380807f950 --- /dev/null +++ b/cups/transcode.c @@ -0,0 +1,720 @@ +/* + * "$Id$" + * + * Transcoding support for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * _cupsCharmapFlush() - Flush all character set maps out of cache. + * cupsCharsetToUTF8() - Convert legacy character set to UTF-8. + * cupsUTF8ToCharset() - Convert UTF-8 to legacy character set. + * cupsUTF8ToUTF32() - Convert UTF-8 to UTF-32. + * cupsUTF32ToUTF8() - Convert UTF-32 to UTF-8. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include +#include +#ifdef HAVE_ICONV_H +# include +#endif /* HAVE_ICONV_H */ + + +/* + * Local globals... + */ + +#ifdef HAVE_ICONV_H +static _cups_mutex_t map_mutex = _CUPS_MUTEX_INITIALIZER; + /* Mutex to control access to maps */ +static iconv_t map_from_utf8 = (iconv_t)-1; + /* Convert from UTF-8 to charset */ +static iconv_t map_to_utf8 = (iconv_t)-1; + /* Convert from charset to UTF-8 */ +static cups_encoding_t map_encoding = CUPS_AUTO_ENCODING; + /* Which charset is cached */ +#endif /* HAVE_ICONV_H */ + + +/* + * '_cupsCharmapFlush()' - Flush all character set maps out of cache. + */ + +void +_cupsCharmapFlush(void) +{ +#ifdef HAVE_ICONV_H + if (map_from_utf8 != (iconv_t)-1) + { + iconv_close(map_from_utf8); + map_from_utf8 = (iconv_t)-1; + } + + if (map_to_utf8 != (iconv_t)-1) + { + iconv_close(map_to_utf8); + map_to_utf8 = (iconv_t)-1; + } + + map_encoding = CUPS_AUTO_ENCODING; +#endif /* HAVE_ICONV_H */ +} + + +/* + * 'cupsCharsetToUTF8()' - Convert legacy character set to UTF-8. + */ + +int /* O - Count or -1 on error */ +cupsCharsetToUTF8( + cups_utf8_t *dest, /* O - Target string */ + const char *src, /* I - Source string */ + const int maxout, /* I - Max output */ + const cups_encoding_t encoding) /* I - Encoding */ +{ + cups_utf8_t *destptr; /* Pointer into UTF-8 buffer */ +#ifdef HAVE_ICONV_H + size_t srclen, /* Length of source string */ + outBytesLeft; /* Bytes remaining in output buffer */ +#endif /* HAVE_ICONV_H */ + + + /* + * Check for valid arguments... + */ + + DEBUG_printf(("2cupsCharsetToUTF8(dest=%p, src=\"%s\", maxout=%d, encoding=%d)", + dest, src, maxout, encoding)); + + if (!dest || !src || maxout < 1) + { + if (dest) + *dest = '\0'; + + DEBUG_puts("3cupsCharsetToUTF8: Bad arguments, returning -1"); + return (-1); + } + + /* + * Handle identity conversions... + */ + + if (encoding == CUPS_UTF8 || encoding <= CUPS_US_ASCII || + encoding >= CUPS_ENCODING_VBCS_END) + { + strlcpy((char *)dest, src, maxout); + return ((int)strlen((char *)dest)); + } + + /* + * Handle ISO-8859-1 to UTF-8 directly... + */ + + destptr = dest; + + if (encoding == CUPS_ISO8859_1) + { + int ch; /* Character from string */ + cups_utf8_t *destend; /* End of UTF-8 buffer */ + + + destend = dest + maxout - 2; + + while (*src && destptr < destend) + { + ch = *src++ & 255; + + if (ch & 128) + { + *destptr++ = 0xc0 | (ch >> 6); + *destptr++ = 0x80 | (ch & 0x3f); + } + else + *destptr++ = ch; + } + + *destptr = '\0'; + + return ((int)(destptr - dest)); + } + + /* + * Convert input legacy charset to UTF-8... + */ + +#ifdef HAVE_ICONV_H + _cupsMutexLock(&map_mutex); + + if (map_encoding != encoding) + { + _cupsCharmapFlush(); + + map_from_utf8 = iconv_open(_cupsEncodingName(encoding), "UTF-8"); + map_to_utf8 = iconv_open("UTF-8", _cupsEncodingName(encoding)); + map_encoding = encoding; + } + + if (map_to_utf8 != (iconv_t)-1) + { + char *altdestptr = (char *)dest; /* Silence bogus GCC type-punned */ + + srclen = strlen(src); + outBytesLeft = maxout - 1; + + iconv(map_to_utf8, (char **)&src, &srclen, &altdestptr, &outBytesLeft); + *altdestptr = '\0'; + + _cupsMutexUnlock(&map_mutex); + + return ((int)(altdestptr - (char *)dest)); + } + + _cupsMutexUnlock(&map_mutex); +#endif /* HAVE_ICONV_H */ + + /* + * No iconv() support, so error out... + */ + + *destptr = '\0'; + + return (-1); +} + + +/* + * 'cupsUTF8ToCharset()' - Convert UTF-8 to legacy character set. + */ + +int /* O - Count or -1 on error */ +cupsUTF8ToCharset( + char *dest, /* O - Target string */ + const cups_utf8_t *src, /* I - Source string */ + const int maxout, /* I - Max output */ + const cups_encoding_t encoding) /* I - Encoding */ +{ + char *destptr; /* Pointer into destination */ +#ifdef HAVE_ICONV_H + size_t srclen, /* Length of source string */ + outBytesLeft; /* Bytes remaining in output buffer */ +#endif /* HAVE_ICONV_H */ + + + /* + * Check for valid arguments... + */ + + if (!dest || !src || maxout < 1) + { + if (dest) + *dest = '\0'; + + return (-1); + } + + /* + * Handle identity conversions... + */ + + if (encoding == CUPS_UTF8 || + encoding >= CUPS_ENCODING_VBCS_END) + { + strlcpy(dest, (char *)src, maxout); + return ((int)strlen(dest)); + } + + /* + * Handle UTF-8 to ISO-8859-1 directly... + */ + + destptr = dest; + + if (encoding == CUPS_ISO8859_1 || encoding <= CUPS_US_ASCII) + { + int ch, /* Character from string */ + maxch; /* Maximum character for charset */ + char *destend; /* End of ISO-8859-1 buffer */ + + maxch = encoding == CUPS_ISO8859_1 ? 256 : 128; + destend = dest + maxout - 1; + + while (*src && destptr < destend) + { + ch = *src++; + + if ((ch & 0xe0) == 0xc0) + { + ch = ((ch & 0x1f) << 6) | (*src++ & 0x3f); + + if (ch < maxch) + *destptr++ = ch; + else + *destptr++ = '?'; + } + else if ((ch & 0xf0) == 0xe0 || + (ch & 0xf8) == 0xf0) + *destptr++ = '?'; + else if (!(ch & 0x80)) + *destptr++ = ch; + } + + *destptr = '\0'; + + return ((int)(destptr - dest)); + } + +#ifdef HAVE_ICONV_H + /* + * Convert input UTF-8 to legacy charset... + */ + + _cupsMutexLock(&map_mutex); + + if (map_encoding != encoding) + { + _cupsCharmapFlush(); + + map_from_utf8 = iconv_open(_cupsEncodingName(encoding), "UTF-8"); + map_to_utf8 = iconv_open("UTF-8", _cupsEncodingName(encoding)); + map_encoding = encoding; + } + + if (map_from_utf8 != (iconv_t)-1) + { + char *altsrc = (char *)src; /* Silence bogus GCC type-punned */ + + srclen = strlen((char *)src); + outBytesLeft = maxout - 1; + + iconv(map_from_utf8, &altsrc, &srclen, &destptr, &outBytesLeft); + *destptr = '\0'; + + _cupsMutexUnlock(&map_mutex); + + return ((int)(destptr - dest)); + } + + _cupsMutexUnlock(&map_mutex); +#endif /* HAVE_ICONV_H */ + + /* + * No iconv() support, so error out... + */ + + *destptr = '\0'; + + return (-1); +} + + +/* + * 'cupsUTF8ToUTF32()' - Convert UTF-8 to UTF-32. + * + * 32-bit UTF-32 (actually 21-bit) maps to UTF-8 as follows... + * + * UTF-32 char UTF-8 char(s) + * -------------------------------------------------- + * 0 to 127 = 0xxxxxxx (US-ASCII) + * 128 to 2047 = 110xxxxx 10yyyyyy + * 2048 to 65535 = 1110xxxx 10yyyyyy 10zzzzzz + * > 65535 = 11110xxx 10yyyyyy 10zzzzzz 10xxxxxx + * + * UTF-32 prohibits chars beyond Plane 16 (> 0x10ffff) in UCS-4, + * which would convert to five- or six-octet UTF-8 sequences... + */ + +int /* O - Count or -1 on error */ +cupsUTF8ToUTF32( + cups_utf32_t *dest, /* O - Target string */ + const cups_utf8_t *src, /* I - Source string */ + const int maxout) /* I - Max output */ +{ + int i; /* Looping variable */ + cups_utf8_t ch; /* Character value */ + cups_utf8_t next; /* Next character value */ + cups_utf32_t ch32; /* UTF-32 character value */ + + + /* + * Check for valid arguments and clear output... + */ + + DEBUG_printf(("2cupsUTF8ToUTF32(dest=%p, src=\"%s\", maxout=%d)", dest, + src, maxout)); + + if (dest) + *dest = 0; + + if (!dest || !src || maxout < 1 || maxout > CUPS_MAX_USTRING) + { + DEBUG_puts("3cupsUTF8ToUTF32: Returning -1 (bad arguments)"); + + return (-1); + } + + /* + * Convert input UTF-8 to output UTF-32... + */ + + for (i = maxout - 1; *src && i > 0; i --) + { + ch = *src++; + + /* + * Convert UTF-8 character(s) to UTF-32 character... + */ + + if (!(ch & 0x80)) + { + /* + * One-octet UTF-8 <= 127 (US-ASCII)... + */ + + *dest++ = ch; + + DEBUG_printf(("4cupsUTF8ToUTF32: %02x => %08X", src[-1], ch)); + continue; + } + else if ((ch & 0xe0) == 0xc0) + { + /* + * Two-octet UTF-8 <= 2047 (Latin-x)... + */ + + next = *src++; + if ((next & 0xc0) != 0x80) + { + DEBUG_puts("3cupsUTF8ToUTF32: Returning -1 (bad UTF-8 sequence)"); + + return (-1); + } + + ch32 = ((ch & 0x1f) << 6) | (next & 0x3f); + + /* + * Check for non-shortest form (invalid UTF-8)... + */ + + if (ch32 < 0x80) + { + DEBUG_puts("3cupsUTF8ToUTF32: Returning -1 (bad UTF-8 sequence)"); + + return (-1); + } + + *dest++ = ch32; + + DEBUG_printf(("4cupsUTF8ToUTF32: %02x %02x => %08X", + src[-2], src[-1], (unsigned)ch32)); + } + else if ((ch & 0xf0) == 0xe0) + { + /* + * Three-octet UTF-8 <= 65535 (Plane 0 - BMP)... + */ + + next = *src++; + if ((next & 0xc0) != 0x80) + { + DEBUG_puts("3cupsUTF8ToUTF32: Returning -1 (bad UTF-8 sequence)"); + + return (-1); + } + + ch32 = ((ch & 0x0f) << 6) | (next & 0x3f); + + next = *src++; + if ((next & 0xc0) != 0x80) + { + DEBUG_puts("3cupsUTF8ToUTF32: Returning -1 (bad UTF-8 sequence)"); + + return (-1); + } + + ch32 = (ch32 << 6) | (next & 0x3f); + + /* + * Check for non-shortest form (invalid UTF-8)... + */ + + if (ch32 < 0x800) + { + DEBUG_puts("3cupsUTF8ToUTF32: Returning -1 (bad UTF-8 sequence)"); + + return (-1); + } + + *dest++ = ch32; + + DEBUG_printf(("4cupsUTF8ToUTF32: %02x %02x %02x => %08X", + src[-3], src[-2], src[-1], (unsigned)ch32)); + } + else if ((ch & 0xf8) == 0xf0) + { + /* + * Four-octet UTF-8... + */ + + next = *src++; + if ((next & 0xc0) != 0x80) + { + DEBUG_puts("3cupsUTF8ToUTF32: Returning -1 (bad UTF-8 sequence)"); + + return (-1); + } + + ch32 = ((ch & 0x07) << 6) | (next & 0x3f); + + next = *src++; + if ((next & 0xc0) != 0x80) + { + DEBUG_puts("3cupsUTF8ToUTF32: Returning -1 (bad UTF-8 sequence)"); + + return (-1); + } + + ch32 = (ch32 << 6) | (next & 0x3f); + + next = *src++; + if ((next & 0xc0) != 0x80) + { + DEBUG_puts("3cupsUTF8ToUTF32: Returning -1 (bad UTF-8 sequence)"); + + return (-1); + } + + ch32 = (ch32 << 6) | (next & 0x3f); + + /* + * Check for non-shortest form (invalid UTF-8)... + */ + + if (ch32 < 0x10000) + { + DEBUG_puts("3cupsUTF8ToUTF32: Returning -1 (bad UTF-8 sequence)"); + + return (-1); + } + + *dest++ = ch32; + + DEBUG_printf(("4cupsUTF8ToUTF32: %02x %02x %02x %02x => %08X", + src[-4], src[-3], src[-2], src[-1], (unsigned)ch32)); + } + else + { + /* + * More than 4-octet (invalid UTF-8 sequence)... + */ + + DEBUG_puts("3cupsUTF8ToUTF32: Returning -1 (bad UTF-8 sequence)"); + + return (-1); + } + + /* + * Check for UTF-16 surrogate (illegal UTF-8)... + */ + + if (ch32 >= 0xd800 && ch32 <= 0xdfff) + return (-1); + } + + *dest = 0; + + DEBUG_printf(("3cupsUTF8ToUTF32: Returning %d characters", maxout - 1 - i)); + + return (maxout - 1 - i); +} + + +/* + * 'cupsUTF32ToUTF8()' - Convert UTF-32 to UTF-8. + * + * 32-bit UTF-32 (actually 21-bit) maps to UTF-8 as follows... + * + * UTF-32 char UTF-8 char(s) + * -------------------------------------------------- + * 0 to 127 = 0xxxxxxx (US-ASCII) + * 128 to 2047 = 110xxxxx 10yyyyyy + * 2048 to 65535 = 1110xxxx 10yyyyyy 10zzzzzz + * > 65535 = 11110xxx 10yyyyyy 10zzzzzz 10xxxxxx + * + * UTF-32 prohibits chars beyond Plane 16 (> 0x10ffff) in UCS-4, + * which would convert to five- or six-octet UTF-8 sequences... + */ + +int /* O - Count or -1 on error */ +cupsUTF32ToUTF8( + cups_utf8_t *dest, /* O - Target string */ + const cups_utf32_t *src, /* I - Source string */ + const int maxout) /* I - Max output */ +{ + cups_utf8_t *start; /* Start of destination string */ + int i; /* Looping variable */ + int swap; /* Byte-swap input to output */ + cups_utf32_t ch; /* Character value */ + + + /* + * Check for valid arguments and clear output... + */ + + DEBUG_printf(("2cupsUTF32ToUTF8(dest=%p, src=%p, maxout=%d)", dest, src, + maxout)); + + if (dest) + *dest = '\0'; + + if (!dest || !src || maxout < 1) + { + DEBUG_puts("3cupsUTF32ToUTF8: Returning -1 (bad args)"); + + return (-1); + } + + /* + * Check for leading BOM in UTF-32 and inverted BOM... + */ + + start = dest; + swap = *src == 0xfffe0000; + + DEBUG_printf(("4cupsUTF32ToUTF8: swap=%d", swap)); + + if (*src == 0xfffe0000 || *src == 0xfeff) + src ++; + + /* + * Convert input UTF-32 to output UTF-8... + */ + + for (i = maxout - 1; *src && i > 0;) + { + ch = *src++; + + /* + * Byte swap input UTF-32, if necessary... + * (only byte-swapping 24 of 32 bits) + */ + + if (swap) + ch = ((ch >> 24) | ((ch >> 8) & 0xff00) | ((ch << 8) & 0xff0000)); + + /* + * Check for beyond Plane 16 (invalid UTF-32)... + */ + + if (ch > 0x10ffff) + { + DEBUG_puts("3cupsUTF32ToUTF8: Returning -1 (character out of range)"); + + return (-1); + } + + /* + * Convert UTF-32 character to UTF-8 character(s)... + */ + + if (ch < 0x80) + { + /* + * One-octet UTF-8 <= 127 (US-ASCII)... + */ + + *dest++ = (cups_utf8_t)ch; + i --; + + DEBUG_printf(("4cupsUTF32ToUTF8: %08x => %02x", (unsigned)ch, dest[-1])); + } + else if (ch < 0x800) + { + /* + * Two-octet UTF-8 <= 2047 (Latin-x)... + */ + + if (i < 2) + { + DEBUG_puts("3cupsUTF32ToUTF8: Returning -1 (too long 2)"); + + return (-1); + } + + *dest++ = (cups_utf8_t)(0xc0 | ((ch >> 6) & 0x1f)); + *dest++ = (cups_utf8_t)(0x80 | (ch & 0x3f)); + i -= 2; + + DEBUG_printf(("4cupsUTF32ToUTF8: %08x => %02x %02x", (unsigned)ch, + dest[-2], dest[-1])); + } + else if (ch < 0x10000) + { + /* + * Three-octet UTF-8 <= 65535 (Plane 0 - BMP)... + */ + + if (i < 3) + { + DEBUG_puts("3cupsUTF32ToUTF8: Returning -1 (too long 3)"); + + return (-1); + } + + *dest++ = (cups_utf8_t)(0xe0 | ((ch >> 12) & 0x0f)); + *dest++ = (cups_utf8_t)(0x80 | ((ch >> 6) & 0x3f)); + *dest++ = (cups_utf8_t)(0x80 | (ch & 0x3f)); + i -= 3; + + DEBUG_printf(("4cupsUTF32ToUTF8: %08x => %02x %02x %02x", (unsigned)ch, + dest[-3], dest[-2], dest[-1])); + } + else + { + /* + * Four-octet UTF-8... + */ + + if (i < 4) + { + DEBUG_puts("3cupsUTF32ToUTF8: Returning -1 (too long 4)"); + + return (-1); + } + + *dest++ = (cups_utf8_t)(0xf0 | ((ch >> 18) & 0x07)); + *dest++ = (cups_utf8_t)(0x80 | ((ch >> 12) & 0x3f)); + *dest++ = (cups_utf8_t)(0x80 | ((ch >> 6) & 0x3f)); + *dest++ = (cups_utf8_t)(0x80 | (ch & 0x3f)); + i -= 4; + + DEBUG_printf(("4cupsUTF32ToUTF8: %08x => %02x %02x %02x %02x", + (unsigned)ch, dest[-4], dest[-3], dest[-2], dest[-1])); + } + } + + *dest = '\0'; + + DEBUG_printf(("3cupsUTF32ToUTF8: Returning %d", (int)(dest - start))); + + return ((int)(dest - start)); +} + + +/* + * End of "$Id$" + */ diff --git a/cups/transcode.h b/cups/transcode.h new file mode 100644 index 0000000000..cafea33c56 --- /dev/null +++ b/cups/transcode.h @@ -0,0 +1,81 @@ +/* + * "$Id$" + * + * Transcoding definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_TRANSCODE_H_ +# define _CUPS_TRANSCODE_H_ + +/* + * Include necessary headers... + */ + +# include "language.h" + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Constants... + */ + +# define CUPS_MAX_USTRING 8192 /* Max size of Unicode string */ + + +/* + * Types... + */ + +typedef unsigned char cups_utf8_t; /* UTF-8 Unicode/ISO-10646 unit */ +typedef unsigned long cups_utf32_t; /* UTF-32 Unicode/ISO-10646 unit */ +typedef unsigned short cups_ucs2_t; /* UCS-2 Unicode/ISO-10646 unit */ +typedef unsigned long cups_ucs4_t; /* UCS-4 Unicode/ISO-10646 unit */ +typedef unsigned char cups_sbcs_t; /* SBCS Legacy 8-bit unit */ +typedef unsigned short cups_dbcs_t; /* DBCS Legacy 16-bit unit */ +typedef unsigned long cups_vbcs_t; /* VBCS Legacy 32-bit unit */ + /* EUC uses 8, 16, 24, 32-bit */ + + +/* + * Prototypes... + */ + +extern int cupsCharsetToUTF8(cups_utf8_t *dest, + const char *src, + const int maxout, + const cups_encoding_t encoding) _CUPS_API_1_2; +extern int cupsUTF8ToCharset(char *dest, + const cups_utf8_t *src, + const int maxout, + const cups_encoding_t encoding) _CUPS_API_1_2; +extern int cupsUTF8ToUTF32(cups_utf32_t *dest, + const cups_utf8_t *src, + const int maxout) _CUPS_API_1_2; +extern int cupsUTF32ToUTF8(cups_utf8_t *dest, + const cups_utf32_t *src, + const int maxout) _CUPS_API_1_2; + +# ifdef __cplusplus +} +# endif /* __cplusplus */ + +#endif /* !_CUPS_TRANSCODE_H_ */ + + +/* + * End of "$Id$" + */ diff --git a/cups/usersys.c b/cups/usersys.c new file mode 100644 index 0000000000..4567013860 --- /dev/null +++ b/cups/usersys.c @@ -0,0 +1,1059 @@ +/* + * "$Id$" + * + * User, system, and password routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsEncryption() - Get the current encryption settings. + * cupsGetPassword() - Get a password from the user. + * cupsGetPassword2() - Get a password from the user using the advanced + * password callback. + * cupsServer() - Return the hostname/address of the current + * server. + * cupsSetClientCertCB() - Set the client certificate callback. + * cupsSetEncryption() - Set the encryption preference. + * cupsSetPasswordCB() - Set the password callback for CUPS. + * cupsSetPasswordCB2() - Set the advanced password callback for CUPS. + * cupsSetServer() - Set the default server name and port. + * cupsSetServerCertCB() - Set the server certificate callback. + * cupsSetUser() - Set the default user name. + * cupsUser() - Return the current user's name. + * _cupsGetPassword() - Get a password from the user. + * _cupsGSSServiceName() - Get the GSS (Kerberos) service name. + * _cupsSetDefaults() - Set the default server, port, and encryption. + * cups_read_client_conf() - Read a client.conf file. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include +#include +#ifdef WIN32 +# include +#else +# include +# include +#endif /* WIN32 */ + + +/* + * Local constants... + */ + +#define _CUPS_PASSCHAR '*' /* Character that is echoed for password */ + + +/* + * Local functions... + */ + +static void cups_read_client_conf(cups_file_t *fp, + _cups_globals_t *cg, + const char *cups_encryption, + const char *cups_server, +#ifdef HAVE_GSSAPI + const char *cups_gssservicename, +#endif /* HAVE_GSSAPI */ + const char *cups_anyroot, + const char *cups_expiredroot, + const char *cups_expiredcerts); + + +/* + * 'cupsEncryption()' - Get the current encryption settings. + * + * The default encryption setting comes from the CUPS_ENCRYPTION + * environment variable, then the ~/.cups/client.conf file, and finally the + * /etc/cups/client.conf file. If not set, the default is + * @code HTTP_ENCRYPT_IF_REQUESTED@. + * + * Note: The current encryption setting is tracked separately for each thread + * in a program. Multi-threaded programs that override the setting via the + * @link cupsSetEncryption@ function need to do so in each thread for the same + * setting to be used. + */ + +http_encryption_t /* O - Encryption settings */ +cupsEncryption(void) +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + if (cg->encryption == (http_encryption_t)-1) + _cupsSetDefaults(); + + return (cg->encryption); +} + + +/* + * 'cupsGetPassword()' - Get a password from the user. + * + * Uses the current password callback function. Returns @code NULL@ if the + * user does not provide a password. + * + * Note: The current password callback function is tracked separately for each + * thread in a program. Multi-threaded programs that override the setting via + * the @link cupsSetPasswordCB@ or @link cupsSetPasswordCB2@ functions need to + * do so in each thread for the same function to be used. + */ + +const char * /* O - Password */ +cupsGetPassword(const char *prompt) /* I - Prompt string */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + return ((cg->password_cb)(prompt, NULL, NULL, NULL, cg->password_data)); +} + + +/* + * 'cupsGetPassword2()' - Get a password from the user using the advanced + * password callback. + * + * Uses the current password callback function. Returns @code NULL@ if the + * user does not provide a password. + * + * Note: The current password callback function is tracked separately for each + * thread in a program. Multi-threaded programs that override the setting via + * the @link cupsSetPasswordCB@ or @link cupsSetPasswordCB2@ functions need to + * do so in each thread for the same function to be used. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +const char * /* O - Password */ +cupsGetPassword2(const char *prompt, /* I - Prompt string */ + http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + const char *method, /* I - Request method ("GET", "POST", "PUT") */ + const char *resource) /* I - Resource path */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + if (!http) + http = _cupsConnect(); + + return ((cg->password_cb)(prompt, http, method, resource, cg->password_data)); +} + + +/* + * 'cupsServer()' - Return the hostname/address of the current server. + * + * The default server comes from the CUPS_SERVER environment variable, then the + * ~/.cups/client.conf file, and finally the /etc/cups/client.conf file. If not + * set, the default is the local system - either "localhost" or a domain socket + * path. + * + * The returned value can be a fully-qualified hostname, a numeric IPv4 or IPv6 + * address, or a domain socket pathname. + * + * Note: The current server is tracked separately for each thread in a program. + * Multi-threaded programs that override the server via the + * @link cupsSetServer@ function need to do so in each thread for the same + * server to be used. + */ + +const char * /* O - Server name */ +cupsServer(void) +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + if (!cg->server[0]) + _cupsSetDefaults(); + + return (cg->server); +} + + +/* + * 'cupsSetClientCertCB()' - Set the client certificate callback. + * + * Pass @code NULL@ to restore the default callback. + * + * Note: The current certificate callback is tracked separately for each thread + * in a program. Multi-threaded programs that override the callback need to do + * so in each thread for the same callback to be used. + * + * @since CUPS 1.5/Mac OS X 10.7@ + */ + +void +cupsSetClientCertCB( + cups_client_cert_cb_t cb, /* I - Callback function */ + void *user_data) /* I - User data pointer */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + cg->client_cert_cb = cb; + cg->client_cert_data = user_data; +} + + +/* + * 'cupsSetCredentials()' - Set the default credentials to be used for SSL/TLS + * connections. + * + * Note: The default credentials are tracked separately for each thread in a + * program. Multi-threaded programs that override the setting need to do so in + * each thread for the same setting to be used. + * + * @since CUPS 1.5/Mac OS X 10.7@ + */ + +int /* O - Status of call (0 = success) */ +cupsSetCredentials( + cups_array_t *credentials) /* I - Array of credentials */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + if (cupsArrayCount(credentials) < 1) + return (-1); + + _httpFreeCredentials(cg->tls_credentials); + cg->tls_credentials = _httpCreateCredentials(credentials); + + return (cg->tls_credentials ? 0 : -1); +} + + +/* + * 'cupsSetEncryption()' - Set the encryption preference. + * + * The default encryption setting comes from the CUPS_ENCRYPTION + * environment variable, then the ~/.cups/client.conf file, and finally the + * /etc/cups/client.conf file. If not set, the default is + * @code HTTP_ENCRYPT_IF_REQUESTED@. + * + * Note: The current encryption setting is tracked separately for each thread + * in a program. Multi-threaded programs that override the setting need to do + * so in each thread for the same setting to be used. + */ + +void +cupsSetEncryption(http_encryption_t e) /* I - New encryption preference */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + cg->encryption = e; + + if (cg->http) + httpEncryption(cg->http, e); +} + + +/* + * 'cupsSetPasswordCB()' - Set the password callback for CUPS. + * + * Pass @code NULL@ to restore the default (console) password callback, which + * reads the password from the console. Programs should call either this + * function or @link cupsSetPasswordCB2@, as only one callback can be registered + * by a program per thread. + * + * Note: The current password callback is tracked separately for each thread + * in a program. Multi-threaded programs that override the callback need to do + * so in each thread for the same callback to be used. + */ + +void +cupsSetPasswordCB(cups_password_cb_t cb)/* I - Callback function */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + if (cb == (cups_password_cb_t)0) + cg->password_cb = (cups_password_cb2_t)_cupsGetPassword; + else + cg->password_cb = (cups_password_cb2_t)cb; + + cg->password_data = NULL; +} + + +/* + * 'cupsSetPasswordCB2()' - Set the advanced password callback for CUPS. + * + * Pass @code NULL@ to restore the default (console) password callback, which + * reads the password from the console. Programs should call either this + * function or @link cupsSetPasswordCB2@, as only one callback can be registered + * by a program per thread. + * + * Note: The current password callback is tracked separately for each thread + * in a program. Multi-threaded programs that override the callback need to do + * so in each thread for the same callback to be used. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +void +cupsSetPasswordCB2( + cups_password_cb2_t cb, /* I - Callback function */ + void *user_data) /* I - User data pointer */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + if (cb == (cups_password_cb2_t)0) + cg->password_cb = (cups_password_cb2_t)_cupsGetPassword; + else + cg->password_cb = cb; + + cg->password_data = user_data; +} + + +/* + * 'cupsSetServer()' - Set the default server name and port. + * + * The "server" string can be a fully-qualified hostname, a numeric + * IPv4 or IPv6 address, or a domain socket pathname. Hostnames and numeric IP + * addresses can be optionally followed by a colon and port number to override + * the default port 631, e.g. "hostname:8631". Pass @code NULL@ to restore the + * default server name and port. + * + * Note: The current server is tracked separately for each thread in a program. + * Multi-threaded programs that override the server need to do so in each + * thread for the same server to be used. + */ + +void +cupsSetServer(const char *server) /* I - Server name */ +{ + char *port; /* Pointer to port */ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + if (server) + { + strlcpy(cg->server, server, sizeof(cg->server)); + + if (cg->server[0] != '/' && (port = strrchr(cg->server, ':')) != NULL && + !strchr(port, ']') && isdigit(port[1] & 255)) + { + *port++ = '\0'; + + cg->ipp_port = atoi(port); + } + + if (cg->server[0] == '/') + strcpy(cg->servername, "localhost"); + else + strlcpy(cg->servername, cg->server, sizeof(cg->servername)); + } + else + { + cg->server[0] = '\0'; + cg->servername[0] = '\0'; + } + + if (cg->http) + { + httpClose(cg->http); + cg->http = NULL; + } +} + + +/* + * 'cupsSetServerCertCB()' - Set the server certificate callback. + * + * Pass @code NULL@ to restore the default callback. + * + * Note: The current credentials callback is tracked separately for each thread + * in a program. Multi-threaded programs that override the callback need to do + * so in each thread for the same callback to be used. + * + * @since CUPS 1.5/Mac OS X 10.7@ + */ + +void +cupsSetServerCertCB( + cups_server_cert_cb_t cb, /* I - Callback function */ + void *user_data) /* I - User data pointer */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + cg->server_cert_cb = cb; + cg->server_cert_data = user_data; +} + + +/* + * 'cupsSetUser()' - Set the default user name. + * + * Pass @code NULL@ to restore the default user name. + * + * Note: The current user name is tracked separately for each thread in a + * program. Multi-threaded programs that override the user name need to do so + * in each thread for the same user name to be used. + */ + +void +cupsSetUser(const char *user) /* I - User name */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + if (user) + strlcpy(cg->user, user, sizeof(cg->user)); + else + cg->user[0] = '\0'; +} + + +/* + * 'cupsUser()' - Return the current user's name. + * + * Note: The current user name is tracked separately for each thread in a + * program. Multi-threaded programs that override the user name with the + * @link cupsSetUser@ function need to do so in each thread for the same user + * name to be used. + */ + +const char * /* O - User name */ +cupsUser(void) +{ + const char *user; /* USER environment variable */ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + if (!cg->user[0]) + { +#ifdef WIN32 + /* + * Get the current user name from the OS... + */ + + DWORD size; /* Size of string */ + + size = sizeof(cg->user); + if (!GetUserName(cg->user, &size)) +#else + /* + * Get the user name corresponding to the current UID... + */ + + struct passwd *pwd; /* User/password entry */ + + setpwent(); + if ((pwd = getpwuid(getuid())) != NULL) + { + /* + * Found a match! + */ + + strlcpy(cg->user, pwd->pw_name, sizeof(cg->user)); + } + else +#endif /* WIN32 */ + if ((user = getenv("USER")) != NULL) + { + /* + * Use the username from the "USER" environment variable... + */ + strlcpy(cg->user, user, sizeof(cg->user)); + } + else + { + /* + * Use the default "unknown" user name... + */ + + strcpy(cg->user, "unknown"); + } + } + + return (cg->user); +} + + +/* + * '_cupsGetPassword()' - Get a password from the user. + */ + +const char * /* O - Password or @code NULL@ if none */ +_cupsGetPassword(const char *prompt) /* I - Prompt string */ +{ +#ifdef WIN32 + HANDLE tty; /* Console handle */ + DWORD mode; /* Console mode */ + char passch, /* Current key press */ + *passptr, /* Pointer into password string */ + *passend; /* End of password string */ + DWORD passbytes; /* Bytes read */ + _cups_globals_t *cg = _cupsGlobals(); + /* Thread globals */ + + + /* + * Disable input echo and set raw input... + */ + + if ((tty = GetStdHandle(STD_INPUT_HANDLE)) == INVALID_HANDLE_VALUE) + return (NULL); + + if (!GetConsoleMode(tty, &mode)) + return (NULL); + + if (!SetConsoleMode(tty, 0)) + return (NULL); + + /* + * Display the prompt... + */ + + printf("%s ", prompt); + fflush(stdout); + + /* + * Read the password string from /dev/tty until we get interrupted or get a + * carriage return or newline... + */ + + passptr = cg->password; + passend = cg->password + sizeof(cg->password) - 1; + + while (ReadFile(tty, &passch, 1, &passbytes, NULL)) + { + if (passch == 0x0A || passch == 0x0D) + { + /* + * Enter/return... + */ + + break; + } + else if (passch == 0x08 || passch == 0x7F) + { + /* + * Backspace/delete (erase character)... + */ + + if (passptr > cg->password) + { + passptr --; + fputs("\010 \010", stdout); + } + else + putchar(0x07); + } + else if (passch == 0x15) + { + /* + * CTRL+U (erase line) + */ + + if (passptr > cg->password) + { + while (passptr > cg->password) + { + passptr --; + fputs("\010 \010", stdout); + } + } + else + putchar(0x07); + } + else if (passch == 0x03) + { + /* + * CTRL+C... + */ + + passptr = cg->password; + break; + } + else if ((passch & 255) < 0x20 || passptr >= passend) + putchar(0x07); + else + { + *passptr++ = passch; + putchar(_CUPS_PASSCHAR); + } + + fflush(stdout); + } + + putchar('\n'); + fflush(stdout); + + /* + * Cleanup... + */ + + SetConsoleMode(tty, mode); + + /* + * Return the proper value... + */ + + if (passbytes == 1 && passptr > cg->password) + { + *passptr = '\0'; + return (cg->password); + } + else + { + memset(cg->password, 0, sizeof(cg->password)); + return (NULL); + } + +#else + int tty; /* /dev/tty - never read from stdin */ + struct termios original, /* Original input mode */ + noecho; /* No echo input mode */ + char passch, /* Current key press */ + *passptr, /* Pointer into password string */ + *passend; /* End of password string */ + ssize_t passbytes; /* Bytes read */ + _cups_globals_t *cg = _cupsGlobals(); + /* Thread globals */ + + + /* + * Disable input echo and set raw input... + */ + + if ((tty = open("/dev/tty", O_RDONLY)) < 0) + return (NULL); + + if (tcgetattr(tty, &original)) + { + close(tty); + return (NULL); + } + + noecho = original; + noecho.c_lflag &= ~(ICANON | ECHO | ECHOE | ISIG); + + if (tcsetattr(tty, TCSAFLUSH, &noecho)) + { + close(tty); + return (NULL); + } + + /* + * Display the prompt... + */ + + printf("%s ", prompt); + fflush(stdout); + + /* + * Read the password string from /dev/tty until we get interrupted or get a + * carriage return or newline... + */ + + passptr = cg->password; + passend = cg->password + sizeof(cg->password) - 1; + + while ((passbytes = read(tty, &passch, 1)) == 1) + { + if (passch == noecho.c_cc[VEOL] || passch == noecho.c_cc[VEOL2] || + passch == 0x0A || passch == 0x0D) + { + /* + * Enter/return... + */ + + break; + } + else if (passch == noecho.c_cc[VERASE] || + passch == 0x08 || passch == 0x7F) + { + /* + * Backspace/delete (erase character)... + */ + + if (passptr > cg->password) + { + passptr --; + fputs("\010 \010", stdout); + } + else + putchar(0x07); + } + else if (passch == noecho.c_cc[VKILL]) + { + /* + * CTRL+U (erase line) + */ + + if (passptr > cg->password) + { + while (passptr > cg->password) + { + passptr --; + fputs("\010 \010", stdout); + } + } + else + putchar(0x07); + } + else if (passch == noecho.c_cc[VINTR] || passch == noecho.c_cc[VQUIT] || + passch == noecho.c_cc[VEOF]) + { + /* + * CTRL+C, CTRL+D, or CTRL+Z... + */ + + passptr = cg->password; + break; + } + else if ((passch & 255) < 0x20 || passptr >= passend) + putchar(0x07); + else + { + *passptr++ = passch; + putchar(_CUPS_PASSCHAR); + } + + fflush(stdout); + } + + putchar('\n'); + fflush(stdout); + + /* + * Cleanup... + */ + + tcsetattr(tty, TCSAFLUSH, &original); + close(tty); + + /* + * Return the proper value... + */ + + if (passbytes == 1 && passptr > cg->password) + { + *passptr = '\0'; + return (cg->password); + } + else + { + memset(cg->password, 0, sizeof(cg->password)); + return (NULL); + } +#endif /* WIN32 */ +} + + +#ifdef HAVE_GSSAPI +/* + * '_cupsGSSServiceName()' - Get the GSS (Kerberos) service name. + */ + +const char * +_cupsGSSServiceName(void) +{ + _cups_globals_t *cg = _cupsGlobals(); /* Thread globals */ + + + if (!cg->gss_service_name[0]) + _cupsSetDefaults(); + + return (cg->gss_service_name); +} +#endif /* HAVE_GSSAPI */ + + +/* + * '_cupsSetDefaults()' - Set the default server, port, and encryption. + */ + +void +_cupsSetDefaults(void) +{ + cups_file_t *fp; /* File */ + const char *home, /* Home directory of user */ + *cups_encryption, /* CUPS_ENCRYPTION env var */ + *cups_server, /* CUPS_SERVER env var */ +#ifdef HAVE_GSSAPI + *cups_gssservicename, /* CUPS_GSSSERVICENAME env var */ +#endif /* HAVE_GSSAPI */ + *cups_anyroot, /* CUPS_ANYROOT env var */ + *cups_expiredroot, /* CUPS_EXPIREDROOT env var */ + *cups_expiredcerts; /* CUPS_EXPIREDCERTS env var */ + char filename[1024]; /* Filename */ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + DEBUG_puts("_cupsSetDefaults()"); + + /* + * First collect environment variables... + */ + + cups_encryption = getenv("CUPS_ENCRYPTION"); + cups_server = getenv("CUPS_SERVER"); +#ifdef HAVE_GSSAPI + cups_gssservicename = getenv("CUPS_GSSSERVICENAME"); +#endif /* HAVE_GSSAPI */ + cups_anyroot = getenv("CUPS_ANYROOT"); + cups_expiredroot = getenv("CUPS_EXPIREDROOT"); + cups_expiredcerts = getenv("CUPS_EXPIREDCERTS"); + + /* + * Then, if needed, read the ~/.cups/client.conf or /etc/cups/client.conf + * files to get the default values... + */ + + if (cg->encryption == (http_encryption_t)-1 || !cg->server[0] || + !cg->ipp_port) + { + if ((home = getenv("HOME")) != NULL) + { + /* + * Look for ~/.cups/client.conf... + */ + + snprintf(filename, sizeof(filename), "%s/.cups/client.conf", home); + fp = cupsFileOpen(filename, "r"); + } + else + fp = NULL; + + if (!fp) + { + /* + * Look for CUPS_SERVERROOT/client.conf... + */ + + snprintf(filename, sizeof(filename), "%s/client.conf", + cg->cups_serverroot); + fp = cupsFileOpen(filename, "r"); + } + + /* + * Read the configuration file and apply any environment variables; both + * functions handle NULL cups_file_t pointers... + */ + + cups_read_client_conf(fp, cg, cups_encryption, cups_server, +#ifdef HAVE_GSSAPI + cups_gssservicename, +#endif /* HAVE_GSSAPI */ + cups_anyroot, cups_expiredroot, + cups_expiredcerts); + cupsFileClose(fp); + } +} + + +/* + * 'cups_read_client_conf()' - Read a client.conf file. + */ + +static void +cups_read_client_conf( + cups_file_t *fp, /* I - File to read */ + _cups_globals_t *cg, /* I - Global data */ + const char *cups_encryption, /* I - CUPS_ENCRYPTION env var */ + const char *cups_server, /* I - CUPS_SERVER env var */ +#ifdef HAVE_GSSAPI + const char *cups_gssservicename, + /* I - CUPS_GSSSERVICENAME env var */ +#endif /* HAVE_GSSAPI */ + const char *cups_anyroot, /* I - CUPS_ANYROOT env var */ + const char *cups_expiredroot, /* I - CUPS_EXPIREDROOT env var */ + const char *cups_expiredcerts) /* I - CUPS_EXPIREDCERTS env var */ +{ + int linenum; /* Current line number */ + char line[1024], /* Line from file */ + *value, /* Pointer into line */ + encryption[1024], /* Encryption value */ +#ifndef __APPLE__ + server_name[1024], /* ServerName value */ +#endif /* !__APPLE__ */ + any_root[1024], /* AllowAnyRoot value */ + expired_root[1024], /* AllowExpiredRoot value */ + expired_certs[1024]; /* AllowExpiredCerts value */ +#ifdef HAVE_GSSAPI + char gss_service_name[32]; /* GSSServiceName value */ +#endif /* HAVE_GSSAPI */ + + + /* + * Read from the file... + */ + + linenum = 0; + while (cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + { + if (!cups_encryption && cg->encryption == (http_encryption_t)-1 && + !_cups_strcasecmp(line, "Encryption") && value) + { + strlcpy(encryption, value, sizeof(encryption)); + cups_encryption = encryption; + } +#ifndef __APPLE__ + /* + * The Server directive is not supported on Mac OS X due to app sandboxing + * restrictions, i.e. not all apps request network access. + */ + else if (!cups_server && (!cg->server[0] || !cg->ipp_port) && + !_cups_strcasecmp(line, "ServerName") && value) + { + strlcpy(server_name, value, sizeof(server_name)); + cups_server = server_name; + } +#endif /* !__APPLE__ */ + else if (!cups_anyroot && !_cups_strcasecmp(line, "AllowAnyRoot") && value) + { + strlcpy(any_root, value, sizeof(any_root)); + cups_anyroot = any_root; + } + else if (!cups_expiredroot && !_cups_strcasecmp(line, "AllowExpiredRoot") && + value) + { + strlcpy(expired_root, value, sizeof(expired_root)); + cups_expiredroot = expired_root; + } + else if (!cups_expiredcerts && !_cups_strcasecmp(line, "AllowExpiredCerts") && + value) + { + strlcpy(expired_certs, value, sizeof(expired_certs)); + cups_expiredcerts = expired_certs; + } +#ifdef HAVE_GSSAPI + else if (!cups_gssservicename && !_cups_strcasecmp(line, "GSSServiceName") && + value) + { + strlcpy(gss_service_name, value, sizeof(gss_service_name)); + cups_gssservicename = gss_service_name; + } +#endif /* HAVE_GSSAPI */ + } + + /* + * Set values... + */ + + if (cg->encryption == (http_encryption_t)-1 && cups_encryption) + { + if (!_cups_strcasecmp(cups_encryption, "never")) + cg->encryption = HTTP_ENCRYPT_NEVER; + else if (!_cups_strcasecmp(cups_encryption, "always")) + cg->encryption = HTTP_ENCRYPT_ALWAYS; + else if (!_cups_strcasecmp(cups_encryption, "required")) + cg->encryption = HTTP_ENCRYPT_REQUIRED; + else + cg->encryption = HTTP_ENCRYPT_IF_REQUESTED; + } + + if ((!cg->server[0] || !cg->ipp_port) && cups_server) + { + if (!cg->server[0]) + { + /* + * Copy server name... + */ + + strlcpy(cg->server, cups_server, sizeof(cg->server)); + + if (cg->server[0] != '/' && (value = strrchr(cg->server, ':')) != NULL && + !strchr(value, ']') && isdigit(value[1] & 255)) + *value++ = '\0'; + else + value = NULL; + + if (cg->server[0] == '/') + strcpy(cg->servername, "localhost"); + else + strlcpy(cg->servername, cg->server, sizeof(cg->servername)); + } + else if (cups_server[0] != '/' && + (value = strrchr(cups_server, ':')) != NULL && + !strchr(value, ']') && isdigit(value[1] & 255)) + value ++; + else + value = NULL; + + if (!cg->ipp_port && value) + cg->ipp_port = atoi(value); + } + + if (!cg->server[0]) + { +#ifdef CUPS_DEFAULT_DOMAINSOCKET + /* + * If we are compiled with domain socket support, only use the + * domain socket if it exists and has the right permissions... + */ + + struct stat sockinfo; /* Domain socket information */ + + if (!stat(CUPS_DEFAULT_DOMAINSOCKET, &sockinfo) && + (sockinfo.st_mode & S_IRWXO) == S_IRWXO) + cups_server = CUPS_DEFAULT_DOMAINSOCKET; + else +#endif /* CUPS_DEFAULT_DOMAINSOCKET */ + cups_server = "localhost"; + + cupsSetServer(cups_server); + } + + if (!cg->ipp_port) + { + const char *ipp_port; /* IPP_PORT environment variable */ + + if ((ipp_port = getenv("IPP_PORT")) != NULL) + { + if ((cg->ipp_port = atoi(ipp_port)) <= 0) + cg->ipp_port = CUPS_DEFAULT_IPP_PORT; + } + else + cg->ipp_port = CUPS_DEFAULT_IPP_PORT; + } + +#ifdef HAVE_GSSAPI + if (!cups_gssservicename) + cups_gssservicename = CUPS_DEFAULT_GSSSERVICENAME; + + strlcpy(cg->gss_service_name, cups_gssservicename, + sizeof(cg->gss_service_name)); +#endif /* HAVE_GSSAPI */ + + if (cups_anyroot) + cg->any_root = !_cups_strcasecmp(cups_anyroot, "yes") || + !_cups_strcasecmp(cups_anyroot, "on") || + !_cups_strcasecmp(cups_anyroot, "true"); + + if (cups_expiredroot) + cg->expired_root = !_cups_strcasecmp(cups_expiredroot, "yes") || + !_cups_strcasecmp(cups_expiredroot, "on") || + !_cups_strcasecmp(cups_expiredroot, "true"); + + if (cups_expiredcerts) + cg->expired_certs = !_cups_strcasecmp(cups_expiredcerts, "yes") || + !_cups_strcasecmp(cups_expiredcerts, "on") || + !_cups_strcasecmp(cups_expiredcerts, "true"); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/utf8demo.txt b/cups/utf8demo.txt new file mode 100644 index 0000000000..03802e4d7a --- /dev/null +++ b/cups/utf8demo.txt @@ -0,0 +1,213 @@ +UTF-8 encoded sample plain-text file +‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾ + +Markus Kuhn [ˈmaʳkʊs kuːn] — 2002-07-25 + + +The ASCII compatible UTF-8 encoding used in this plain-text file +is defined in Unicode, ISO 10646-1, and RFC 2279. + + +Using Unicode/UTF-8, you can write in emails and source code things such as + +Mathematics and sciences: + + ∮ E⋅da = Q, n → ∞, ∑ f(i) = ∏ g(i), ⎧⎡⎛┌─────┐⎞⎤⎫ + ⎪⎢⎜│a²+b³ ⎟⎥⎪ + ∀x∈ℝ: ⌈x⌉ = −⌊−x⌋, α ∧ ¬β = ¬(¬α ∨ β), ⎪⎢⎜│───── ⎟⎥⎪ + ⎪⎢⎜⎷ c₈ ⎟⎥⎪ + ℕ ⊆ ℕ₀ ⊂ ℤ ⊂ ℚ ⊂ ℝ ⊂ ℂ, ⎨⎢⎜ ⎟⎥⎬ + ⎪⎢⎜ ∞ ⎟⎥⎪ + ⊥ < a ≠ b ≡ c ≤ d ≪ ⊤ ⇒ (⟦A⟧ ⇔ ⟪B⟫), ⎪⎢⎜ ⎲ ⎟⎥⎪ + ⎪⎢⎜ ⎳aⁱ-bⁱ⎟⎥⎪ + 2H₂ + O₂ ⇌ 2H₂O, R = 4.7 kΩ, ⌀ 200 mm ⎩⎣⎝i=1 ⎠⎦⎭ + +Linguistics and dictionaries: + + ði ıntəˈnæʃənəl fəˈnɛtık əsoʊsiˈeıʃn + Y [ˈʏpsilɔn], Yen [jɛn], Yoga [ˈjoːgɑ] + +APL: + + ((V⍳V)=⍳⍴V)/V←,V ⌷←⍳→⍴∆∇⊃‾⍎⍕⌈ + +Nicer typography in plain text files: + + ╔══════════════════════════════════════════╗ + ║ ║ + ║ • ‘single’ and “double” quotes ║ + ║ ║ + ║ • Curly apostrophes: “We’ve been here” ║ + ║ ║ + ║ • Latin-1 apostrophe and accents: '´` ║ + ║ ║ + ║ • ‚deutsche‘ „Anführungszeichen“ ║ + ║ ║ + ║ • †, ‡, ‰, •, 3–4, —, −5/+5, ™, … ║ + ║ ║ + ║ • ASCII safety test: 1lI|, 0OD, 8B ║ + ║ ╭─────────╮ ║ + ║ • the euro symbol: │ 14.95 € │ ║ + ║ ╰─────────╯ ║ + ╚══════════════════════════════════════════╝ + +Combining characters: + + STARGΛ̊TE SG-1, a = v̇ = r̈, a⃑ ⊥ b⃑ + +Greek (in Polytonic): + + The Greek anthem: + + Σὲ γνωρίζω ἀπὸ τὴν κόψη + τοῦ σπαθιοῦ τὴν τρομερή, + σὲ γνωρίζω ἀπὸ τὴν ὄψη + ποὺ μὲ βία μετράει τὴ γῆ. + + ᾿Απ᾿ τὰ κόκκαλα βγαλμένη + τῶν ῾Ελλήνων τὰ ἱερά + καὶ σὰν πρῶτα ἀνδρειωμένη + χαῖρε, ὦ χαῖρε, ᾿Ελευθεριά! + + From a speech of Demosthenes in the 4th century BC: + + Οὐχὶ ταὐτὰ παρίσταταί μοι γιγνώσκειν, ὦ ἄνδρες ᾿Αθηναῖοι, + ὅταν τ᾿ εἰς τὰ πράγματα ἀποβλέψω καὶ ὅταν πρὸς τοὺς + λόγους οὓς ἀκούω· τοὺς μὲν γὰρ λόγους περὶ τοῦ + τιμωρήσασθαι Φίλιππον ὁρῶ γιγνομένους, τὰ δὲ πράγματ᾿ + εἰς τοῦτο προήκοντα, ὥσθ᾿ ὅπως μὴ πεισόμεθ᾿ αὐτοὶ + πρότερον κακῶς σκέψασθαι δέον. οὐδέν οὖν ἄλλο μοι δοκοῦσιν + οἱ τὰ τοιαῦτα λέγοντες ἢ τὴν ὑπόθεσιν, περὶ ἧς βουλεύεσθαι, + οὐχὶ τὴν οὖσαν παριστάντες ὑμῖν ἁμαρτάνειν. ἐγὼ δέ, ὅτι μέν + ποτ᾿ ἐξῆν τῇ πόλει καὶ τὰ αὑτῆς ἔχειν ἀσφαλῶς καὶ Φίλιππον + τιμωρήσασθαι, καὶ μάλ᾿ ἀκριβῶς οἶδα· ἐπ᾿ ἐμοῦ γάρ, οὐ πάλαι + γέγονεν ταῦτ᾿ ἀμφότερα· νῦν μέντοι πέπεισμαι τοῦθ᾿ ἱκανὸν + προλαβεῖν ἡμῖν εἶναι τὴν πρώτην, ὅπως τοὺς συμμάχους + σώσομεν. ἐὰν γὰρ τοῦτο βεβαίως ὑπάρξῃ, τότε καὶ περὶ τοῦ + τίνα τιμωρήσεταί τις καὶ ὃν τρόπον ἐξέσται σκοπεῖν· πρὶν δὲ + τὴν ἀρχὴν ὀρθῶς ὑποθέσθαι, μάταιον ἡγοῦμαι περὶ τῆς + τελευτῆς ὁντινοῦν ποιεῖσθαι λόγον. + + Δημοσθένους, Γ´ ᾿Ολυνθιακὸς + +Georgian: + + From a Unicode conference invitation: + + გთხოვთ ახლავე გაიაროთ რეგისტრაცია Unicode-ის მეათე საერთაშორისო + კონფერენციაზე დასასწრებად, რომელიც გაიმართება 10-12 მარტს, + ქ. მაინცში, გერმანიაში. კონფერენცია შეჰკრებს ერთად მსოფლიოს + ექსპერტებს ისეთ დარგებში როგორიცაა ინტერნეტი და Unicode-ი, + ინტერნაციონალიზაცია და ლოკალიზაცია, Unicode-ის გამოყენება + ოპერაციულ სისტემებსა, და გამოყენებით პროგრამებში, შრიფტებში, + ტექსტების დამუშავებასა და მრავალენოვან კომპიუტერულ სისტემებში. + +Russian: + + From a Unicode conference invitation: + + Зарегистрируйтесь сейчас на Десятую Международную Конференцию по + Unicode, которая состоится 10-12 марта 1997 года в Майнце в Германии. + Конференция соберет широкий круг экспертов по вопросам глобального + Интернета и Unicode, локализации и интернационализации, воплощению и + применению Unicode в различных операционных системах и программных + приложениях, шрифтах, верстке и многоязычных компьютерных системах. + +Thai (UCS Level 2): + + Excerpt from a poetry on The Romance of The Three Kingdoms (a Chinese + classic 'San Gua'): + + [----------------------------|------------------------] + ๏ แผ่นดินฮั่นเสื่อมโทรมแสนสังเวช พระปกเกศกองบู๊กู้ขึ้นใหม่ + สิบสองกษัตริย์ก่อนหน้าแลถัดไป สององค์ไซร้โง่เขลาเบาปัญญา + ทรงนับถือขันทีเป็นที่พึ่ง บ้านเมืองจึงวิปริตเป็นนักหนา + โฮจิ๋นเรียกทัพทั่วหัวเมืองมา หมายจะฆ่ามดชั่วตัวสำคัญ + เหมือนขับไสไล่เสือจากเคหา รับหมาป่าเข้ามาเลยอาสัญ + ฝ่ายอ้องอุ้นยุแยกให้แตกกัน ใช้สาวนั้นเป็นชนวนชื่นชวนใจ + พลันลิฉุยกุยกีกลับก่อเหตุ ช่างอาเพศจริงหนาฟ้าร้องไห้ + ต้องรบราฆ่าฟันจนบรรลัย ฤๅหาใครค้ำชูกู้บรรลังก์ ฯ + + (The above is a two-column text. If combining characters are handled + correctly, the lines of the second column should be aligned with the + | character above.) + +Ethiopian: + + Proverbs in the Amharic language: + + ሰማይ አይታረስ ንጉሥ አይከሰስ። + ብላ ካለኝ እንደአባቴ በቆመጠኝ። + ጌጥ ያለቤቱ ቁምጥና ነው። + ደሀ በሕልሙ ቅቤ ባይጠጣ ንጣት በገደለው። + የአፍ ወለምታ በቅቤ አይታሽም። + አይጥ በበላ ዳዋ ተመታ። + ሲተረጉሙ ይደረግሙ። + ቀስ በቀስ፥ ዕንቁላል በእግሩ ይሄዳል። + ድር ቢያብር አንበሳ ያስር። + ሰው እንደቤቱ እንጅ እንደ ጉረቤቱ አይተዳደርም። + እግዜር የከፈተውን ጉሮሮ ሳይዘጋው አይድርም። + የጎረቤት ሌባ፥ ቢያዩት ይስቅ ባያዩት ያጠልቅ። + ሥራ ከመፍታት ልጄን ላፋታት። + ዓባይ ማደሪያ የለው፥ ግንድ ይዞ ይዞራል። + የእስላም አገሩ መካ የአሞራ አገሩ ዋርካ። + ተንጋሎ ቢተፉ ተመልሶ ባፉ። + ወዳጅህ ማር ቢሆን ጨርስህ አትላሰው። + እግርህን በፍራሽህ ልክ ዘርጋ። + +Runes: + + ᚻᛖ ᚳᚹᚫᚦ ᚦᚫᛏ ᚻᛖ ᛒᚢᛞᛖ ᚩᚾ ᚦᚫᛗ ᛚᚪᚾᛞᛖ ᚾᚩᚱᚦᚹᛖᚪᚱᛞᚢᛗ ᚹᛁᚦ ᚦᚪ ᚹᛖᛥᚫ + + (Old English, which transcribed into Latin reads 'He cwaeth that he + bude thaem lande northweardum with tha Westsae.' and means 'He said + that he lived in the northern land near the Western Sea.') + +Braille: + + ⡌⠁⠧⠑ ⠼⠁⠒ ⡍⠜⠇⠑⠹⠰⠎ ⡣⠕⠌ + + ⡍⠜⠇⠑⠹ ⠺⠁⠎ ⠙⠑⠁⠙⠒ ⠞⠕ ⠃⠑⠛⠔ ⠺⠊⠹⠲ ⡹⠻⠑ ⠊⠎ ⠝⠕ ⠙⠳⠃⠞ + ⠱⠁⠞⠑⠧⠻ ⠁⠃⠳⠞ ⠹⠁⠞⠲ ⡹⠑ ⠗⠑⠛⠊⠌⠻ ⠕⠋ ⠙⠊⠎ ⠃⠥⠗⠊⠁⠇ ⠺⠁⠎ + ⠎⠊⠛⠝⠫ ⠃⠹ ⠹⠑ ⠊⠇⠻⠛⠹⠍⠁⠝⠂ ⠹⠑ ⠊⠇⠻⠅⠂ ⠹⠑ ⠥⠝⠙⠻⠞⠁⠅⠻⠂ + ⠁⠝⠙ ⠹⠑ ⠡⠊⠑⠋ ⠍⠳⠗⠝⠻⠲ ⡎⠊⠗⠕⠕⠛⠑ ⠎⠊⠛⠝⠫ ⠊⠞⠲ ⡁⠝⠙ + ⡎⠊⠗⠕⠕⠛⠑⠰⠎ ⠝⠁⠍⠑ ⠺⠁⠎ ⠛⠕⠕⠙ ⠥⠏⠕⠝ ⠰⡡⠁⠝⠛⠑⠂ ⠋⠕⠗ ⠁⠝⠹⠹⠔⠛ ⠙⠑ + ⠡⠕⠎⠑ ⠞⠕ ⠏⠥⠞ ⠙⠊⠎ ⠙⠁⠝⠙ ⠞⠕⠲ + + ⡕⠇⠙ ⡍⠜⠇⠑⠹ ⠺⠁⠎ ⠁⠎ ⠙⠑⠁⠙ ⠁⠎ ⠁ ⠙⠕⠕⠗⠤⠝⠁⠊⠇⠲ + + ⡍⠔⠙⠖ ⡊ ⠙⠕⠝⠰⠞ ⠍⠑⠁⠝ ⠞⠕ ⠎⠁⠹ ⠹⠁⠞ ⡊ ⠅⠝⠪⠂ ⠕⠋ ⠍⠹ + ⠪⠝ ⠅⠝⠪⠇⠫⠛⠑⠂ ⠱⠁⠞ ⠹⠻⠑ ⠊⠎ ⠏⠜⠞⠊⠊⠥⠇⠜⠇⠹ ⠙⠑⠁⠙ ⠁⠃⠳⠞ + ⠁ ⠙⠕⠕⠗⠤⠝⠁⠊⠇⠲ ⡊ ⠍⠊⠣⠞ ⠙⠁⠧⠑ ⠃⠑⠲ ⠔⠊⠇⠔⠫⠂ ⠍⠹⠎⠑⠇⠋⠂ ⠞⠕ + ⠗⠑⠛⠜⠙ ⠁ ⠊⠕⠋⠋⠔⠤⠝⠁⠊⠇ ⠁⠎ ⠹⠑ ⠙⠑⠁⠙⠑⠌ ⠏⠊⠑⠊⠑ ⠕⠋ ⠊⠗⠕⠝⠍⠕⠝⠛⠻⠹ + ⠔ ⠹⠑ ⠞⠗⠁⠙⠑⠲ ⡃⠥⠞ ⠹⠑ ⠺⠊⠎⠙⠕⠍ ⠕⠋ ⠳⠗ ⠁⠝⠊⠑⠌⠕⠗⠎ + ⠊⠎ ⠔ ⠹⠑ ⠎⠊⠍⠊⠇⠑⠆ ⠁⠝⠙ ⠍⠹ ⠥⠝⠙⠁⠇⠇⠪⠫ ⠙⠁⠝⠙⠎ + ⠩⠁⠇⠇ ⠝⠕⠞ ⠙⠊⠌⠥⠗⠃ ⠊⠞⠂ ⠕⠗ ⠹⠑ ⡊⠳⠝⠞⠗⠹⠰⠎ ⠙⠕⠝⠑ ⠋⠕⠗⠲ ⡹⠳ + ⠺⠊⠇⠇ ⠹⠻⠑⠋⠕⠗⠑ ⠏⠻⠍⠊⠞ ⠍⠑ ⠞⠕ ⠗⠑⠏⠑⠁⠞⠂ ⠑⠍⠏⠙⠁⠞⠊⠊⠁⠇⠇⠹⠂ ⠹⠁⠞ + ⡍⠜⠇⠑⠹ ⠺⠁⠎ ⠁⠎ ⠙⠑⠁⠙ ⠁⠎ ⠁ ⠙⠕⠕⠗⠤⠝⠁⠊⠇⠲ + + (The first couple of paragraphs of "A Christmas Carol" by Dickens) + +Compact font selection example text: + + ABCDEFGHIJKLMNOPQRSTUVWXYZ /0123456789 + abcdefghijklmnopqrstuvwxyz £©µÀÆÖÞßéöÿ + –—‘“”„†•…‰™œŠŸž€ ΑΒΓΔΩαβγδω АБВГДабвгд + ∀∂∈ℝ∧∪≡∞ ↑↗↨↻⇣ ┐┼╔╘░►☺♀ fi?⑀₂ἠḂӥẄɐː⍎אԱა + +Greetings in various languages: + + Hello world, Καλημέρα κόσμε, コンニチハ + +Box drawing alignment tests: █ + ▉ + ╔══╦══╗ ┌──┬──┐ ╭──┬──╮ ╭──┬──╮ ┏━━┳━━┓ ┎┒┏┑ ╷ ╻ ┏┯┓ ┌┰┐ ▊ ╱╲╱╲╳╳╳ + ║┌─╨─┐║ │╔═╧═╗│ │╒═╪═╕│ │╓─╁─╖│ ┃┌─╂─┐┃ ┗╃╄┙ ╶┼╴╺╋╸┠┼┨ ┝╋┥ ▋ ╲╱╲╱╳╳╳ + ║│╲ ╱│║ │║ ║│ ││ │ ││ │║ ┃ ║│ ┃│ ╿ │┃ ┍╅╆┓ ╵ ╹ ┗┷┛ └┸┘ ▌ ╱╲╱╲╳╳╳ + ╠╡ ╳ ╞╣ ├╢ ╟┤ ├┼─┼─┼┤ ├╫─╂─╫┤ ┣┿╾┼╼┿┫ ┕┛┖┚ ┌┄┄┐ ╎ ┏┅┅┓ ┋ ▍ ╲╱╲╱╳╳╳ + ║│╱ ╲│║ │║ ║│ ││ │ ││ │║ ┃ ║│ ┃│ ╽ │┃ ░░▒▒▓▓██ ┊ ┆ ╎ ╏ ┇ ┋ ▎ + ║└─╥─┘║ │╚═╤═╝│ │╘═╪═╛│ │╙─╀─╜│ ┃└─╂─┘┃ ░░▒▒▓▓██ ┊ ┆ ╎ ╏ ┇ ┋ ▏ + ╚══╩══╝ └──┴──┘ ╰──┴──╯ ╰──┴──╯ ┗━━┻━━┛ ▗▄▖▛▀▜ └╌╌┘ ╎ ┗╍╍┛ ┋ ▁▂▃▄▅▆▇█ + ▝▀▘▙▄▟ + + diff --git a/cups/util.c b/cups/util.c new file mode 100644 index 0000000000..5fd3fe197d --- /dev/null +++ b/cups/util.c @@ -0,0 +1,1800 @@ +/* + * "$Id$" + * + * Printing utilities for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsCancelJob() - Cancel a print job on the default server. + * cupsCancelJob2() - Cancel or purge a print job. + * cupsCreateJob() - Create an empty job for streaming. + * cupsFinishDocument() - Finish sending a document. + * cupsFreeJobs() - Free memory used by job data. + * cupsGetClasses() - Get a list of printer classes from the default + * server. + * cupsGetDefault() - Get the default printer or class for the default + * server. + * cupsGetDefault2() - Get the default printer or class for the specified + * server. + * cupsGetJobs() - Get the jobs from the default server. + * cupsGetJobs2() - Get the jobs from the specified server. + * cupsGetPPD() - Get the PPD file for a printer on the default + * server. + * cupsGetPPD2() - Get the PPD file for a printer from the specified + * server. + * cupsGetPPD3() - Get the PPD file for a printer on the specified + * server if it has changed. + * cupsGetPrinters() - Get a list of printers from the default server. + * cupsGetServerPPD() - Get an available PPD file from the server. + * cupsPrintFile() - Print a file to a printer or class on the default + * server. + * cupsPrintFile2() - Print a file to a printer or class on the + * specified server. + * cupsPrintFiles() - Print one or more files to a printer or class on + * the default server. + * cupsPrintFiles2() - Print one or more files to a printer or class on + * the specified server. + * cupsStartDocument() - Add a document to a job created with + * cupsCreateJob(). + * cups_get_printer_uri() - Get the printer-uri-supported attribute for the + * first printer in a class. + */ + +/* + * Include necessary headers... + */ + +#include "cups-private.h" +#include +#include +#if defined(WIN32) || defined(__EMX__) +# include +#else +# include +#endif /* WIN32 || __EMX__ */ + + +/* + * Local functions... + */ + +static int cups_get_printer_uri(http_t *http, const char *name, + char *host, int hostsize, int *port, + char *resource, int resourcesize, + int depth); + + +/* + * 'cupsCancelJob()' - Cancel a print job on the default server. + * + * Pass @code CUPS_JOBID_ALL@ to cancel all jobs or @code CUPS_JOBID_CURRENT@ + * to cancel the current job on the named destination. + * + * Use the @link cupsLastError@ and @link cupsLastErrorString@ functions to get + * the cause of any failure. + */ + +int /* O - 1 on success, 0 on failure */ +cupsCancelJob(const char *name, /* I - Name of printer or class */ + int job_id) /* I - Job ID, @code CUPS_JOBID_CURRENT@ for the current job, or @code CUPS_JOBID_ALL@ for all jobs */ +{ + return (cupsCancelJob2(CUPS_HTTP_DEFAULT, name, job_id, 0) + < IPP_REDIRECTION_OTHER_SITE); +} + + +/* + * 'cupsCancelJob2()' - Cancel or purge a print job. + * + * Canceled jobs remain in the job history while purged jobs are removed + * from the job history. + * + * Pass @code CUPS_JOBID_ALL@ to cancel all jobs or @code CUPS_JOBID_CURRENT@ + * to cancel the current job on the named destination. + * + * Use the @link cupsLastError@ and @link cupsLastErrorString@ functions to get + * the cause of any failure. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +ipp_status_t /* O - IPP status */ +cupsCancelJob2(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + const char *name, /* I - Name of printer or class */ + int job_id, /* I - Job ID, @code CUPS_JOBID_CURRENT@ for the current job, or @code CUPS_JOBID_ALL@ for all jobs */ + int purge) /* I - 1 to purge, 0 to cancel */ +{ + char uri[HTTP_MAX_URI]; /* Job/printer URI */ + ipp_t *request; /* IPP request */ + + + /* + * Range check input... + */ + + if (job_id < -1 || (!name && job_id == 0)) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + return (0); + } + + /* + * Connect to the default server as needed... + */ + + if (!http) + if ((http = _cupsConnect()) == NULL) + return (IPP_SERVICE_UNAVAILABLE); + + /* + * Build an IPP_CANCEL_JOB or IPP_PURGE_JOBS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * job-uri or printer-uri + job-id + * requesting-user-name + * [purge-job] or [purge-jobs] + */ + + request = ippNewRequest(job_id < 0 ? IPP_PURGE_JOBS : IPP_CANCEL_JOB); + + if (name) + { + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", ippPort(), "/printers/%s", name); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", NULL, + uri); + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_INTEGER, "job-id", + job_id); + } + else if (job_id > 0) + { + snprintf(uri, sizeof(uri), "ipp://localhost/jobs/%d", job_id); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-uri", NULL, uri); + } + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, cupsUser()); + + if (purge && job_id >= 0) + ippAddBoolean(request, IPP_TAG_OPERATION, "purge-job", 1); + else if (!purge && job_id < 0) + ippAddBoolean(request, IPP_TAG_OPERATION, "purge-jobs", 0); + + /* + * Do the request... + */ + + ippDelete(cupsDoRequest(http, request, "/jobs/")); + + return (cupsLastError()); +} + + +/* + * 'cupsCreateJob()' - Create an empty job for streaming. + * + * Use this function when you want to stream print data using the + * @link cupsStartDocument@, @link cupsWriteRequestData@, and + * @link cupsFinishDocument@ functions. If you have one or more files to + * print, use the @link cupsPrintFile2@ or @link cupsPrintFiles2@ function + * instead. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +int /* O - Job ID or 0 on error */ +cupsCreateJob( + http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + const char *name, /* I - Destination name */ + const char *title, /* I - Title of job */ + int num_options, /* I - Number of options */ + cups_option_t *options) /* I - Options */ +{ + char printer_uri[1024], /* Printer URI */ + resource[1024]; /* Printer resource */ + ipp_t *request, /* Create-Job request */ + *response; /* Create-Job response */ + ipp_attribute_t *attr; /* job-id attribute */ + int job_id = 0; /* job-id value */ + + + DEBUG_printf(("cupsCreateJob(http=%p, name=\"%s\", title=\"%s\", " + "num_options=%d, options=%p)", + http, name, title, num_options, options)); + + /* + * Range check input... + */ + + if (!name) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + return (0); + } + + /* + * Build a Create-Job request... + */ + + if ((request = ippNewRequest(IPP_CREATE_JOB)) == NULL) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(ENOMEM), 0); + return (0); + } + + httpAssembleURIf(HTTP_URI_CODING_ALL, printer_uri, sizeof(printer_uri), "ipp", + NULL, "localhost", ippPort(), "/printers/%s", name); + snprintf(resource, sizeof(resource), "/printers/%s", name); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, printer_uri); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, cupsUser()); + if (title) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "job-name", NULL, + title); + cupsEncodeOptions(request, num_options, options); + + /* + * Send the request and get the job-id... + */ + + response = cupsDoRequest(http, request, resource); + + if ((attr = ippFindAttribute(response, "job-id", IPP_TAG_INTEGER)) != NULL) + job_id = attr->values[0].integer; + + ippDelete(response); + + /* + * Return it... + */ + + return (job_id); +} + + +/* + * 'cupsFinishDocument()' - Finish sending a document. + * + * The document must have been started using @link cupsStartDocument@. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +ipp_status_t /* O - Status of document submission */ +cupsFinishDocument(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + const char *name) /* I - Destination name */ +{ + char resource[1024]; /* Printer resource */ + + + snprintf(resource, sizeof(resource), "/printers/%s", name); + + ippDelete(cupsGetResponse(http, resource)); + + return (cupsLastError()); +} + + +/* + * 'cupsFreeJobs()' - Free memory used by job data. + */ + +void +cupsFreeJobs(int num_jobs, /* I - Number of jobs */ + cups_job_t *jobs) /* I - Jobs */ +{ + int i; /* Looping var */ + cups_job_t *job; /* Current job */ + + + if (num_jobs <= 0 || !jobs) + return; + + for (i = num_jobs, job = jobs; i > 0; i --, job ++) + { + _cupsStrFree(job->dest); + _cupsStrFree(job->user); + _cupsStrFree(job->format); + _cupsStrFree(job->title); + } + + free(jobs); +} + + +/* + * 'cupsGetClasses()' - Get a list of printer classes from the default server. + * + * This function is deprecated - use @link cupsGetDests@ instead. + * + * @deprecated@ + */ + +int /* O - Number of classes */ +cupsGetClasses(char ***classes) /* O - Classes */ +{ + int n; /* Number of classes */ + ipp_t *request, /* IPP Request */ + *response; /* IPP Response */ + ipp_attribute_t *attr; /* Current attribute */ + char **temp; /* Temporary pointer */ + http_t *http; /* Connection to server */ + + + if (!classes) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + + return (0); + } + + *classes = NULL; + + if ((http = _cupsConnect()) == NULL) + return (0); + + /* + * Build a CUPS_GET_CLASSES request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * requested-attributes + */ + + request = ippNewRequest(CUPS_GET_CLASSES); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", NULL, "printer-name"); + + /* + * Do the request and get back a response... + */ + + n = 0; + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + for (attr = response->attrs; attr != NULL; attr = attr->next) + if (attr->name != NULL && + _cups_strcasecmp(attr->name, "printer-name") == 0 && + attr->value_tag == IPP_TAG_NAME) + { + if (n == 0) + temp = malloc(sizeof(char *)); + else + temp = realloc(*classes, sizeof(char *) * (n + 1)); + + if (temp == NULL) + { + /* + * Ran out of memory! + */ + + while (n > 0) + { + n --; + free((*classes)[n]); + } + + free(*classes); + ippDelete(response); + return (0); + } + + *classes = temp; + temp[n] = strdup(attr->values[0].string.text); + n ++; + } + + ippDelete(response); + } + + return (n); +} + + +/* + * 'cupsGetDefault()' - Get the default printer or class for the default server. + * + * This function returns the default printer or class as defined by + * the LPDEST or PRINTER environment variables. If these environment + * variables are not set, the server default destination is returned. + * Applications should use the @link cupsGetDests@ and @link cupsGetDest@ + * functions to get the user-defined default printer, as this function does + * not support the lpoptions-defined default printer. + */ + +const char * /* O - Default printer or @code NULL@ */ +cupsGetDefault(void) +{ + /* + * Return the default printer... + */ + + return (cupsGetDefault2(CUPS_HTTP_DEFAULT)); +} + + +/* + * 'cupsGetDefault2()' - Get the default printer or class for the specified server. + * + * This function returns the default printer or class as defined by + * the LPDEST or PRINTER environment variables. If these environment + * variables are not set, the server default destination is returned. + * Applications should use the @link cupsGetDests@ and @link cupsGetDest@ + * functions to get the user-defined default printer, as this function does + * not support the lpoptions-defined default printer. + * + * @since CUPS 1.1.21/Mac OS X 10.4@ + */ + +const char * /* O - Default printer or @code NULL@ */ +cupsGetDefault2(http_t *http) /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ +{ + ipp_t *request, /* IPP Request */ + *response; /* IPP Response */ + ipp_attribute_t *attr; /* Current attribute */ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + /* + * See if we have a user default printer set... + */ + + if (_cupsUserDefault(cg->def_printer, sizeof(cg->def_printer))) + return (cg->def_printer); + + /* + * Connect to the server as needed... + */ + + if (!http) + if ((http = _cupsConnect()) == NULL) + return (NULL); + + /* + * Build a CUPS_GET_DEFAULT request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + */ + + request = ippNewRequest(CUPS_GET_DEFAULT); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + if ((attr = ippFindAttribute(response, "printer-name", + IPP_TAG_NAME)) != NULL) + { + strlcpy(cg->def_printer, attr->values[0].string.text, + sizeof(cg->def_printer)); + ippDelete(response); + return (cg->def_printer); + } + + ippDelete(response); + } + + return (NULL); +} + + +/* + * 'cupsGetJobs()' - Get the jobs from the default server. + * + * A "whichjobs" value of @code CUPS_WHICHJOBS_ALL@ returns all jobs regardless + * of state, while @code CUPS_WHICHJOBS_ACTIVE@ returns jobs that are + * pending, processing, or held and @code CUPS_WHICHJOBS_COMPLETED@ returns + * jobs that are stopped, canceled, aborted, or completed. + */ + +int /* O - Number of jobs */ +cupsGetJobs(cups_job_t **jobs, /* O - Job data */ + const char *name, /* I - @code NULL@ = all destinations, otherwise show jobs for named destination */ + int myjobs, /* I - 0 = all users, 1 = mine */ + int whichjobs) /* I - @code CUPS_WHICHJOBS_ALL@, @code CUPS_WHICHJOBS_ACTIVE@, or @code CUPS_WHICHJOBS_COMPLETED@ */ +{ + /* + * Return the jobs... + */ + + return (cupsGetJobs2(CUPS_HTTP_DEFAULT, jobs, name, myjobs, whichjobs)); +} + + + +/* + * 'cupsGetJobs2()' - Get the jobs from the specified server. + * + * A "whichjobs" value of @code CUPS_WHICHJOBS_ALL@ returns all jobs regardless + * of state, while @code CUPS_WHICHJOBS_ACTIVE@ returns jobs that are + * pending, processing, or held and @code CUPS_WHICHJOBS_COMPLETED@ returns + * jobs that are stopped, canceled, aborted, or completed. + * + * @since CUPS 1.1.21/Mac OS X 10.4@ + */ + +int /* O - Number of jobs */ +cupsGetJobs2(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + cups_job_t **jobs, /* O - Job data */ + const char *name, /* I - @code NULL@ = all destinations, otherwise show jobs for named destination */ + int myjobs, /* I - 0 = all users, 1 = mine */ + int whichjobs) /* I - @code CUPS_WHICHJOBS_ALL@, @code CUPS_WHICHJOBS_ACTIVE@, or @code CUPS_WHICHJOBS_COMPLETED@ */ +{ + int n; /* Number of jobs */ + ipp_t *request, /* IPP Request */ + *response; /* IPP Response */ + ipp_attribute_t *attr; /* Current attribute */ + cups_job_t *temp; /* Temporary pointer */ + int id, /* job-id */ + priority, /* job-priority */ + size; /* job-k-octets */ + ipp_jstate_t state; /* job-state */ + time_t completed_time, /* time-at-completed */ + creation_time, /* time-at-creation */ + processing_time; /* time-at-processing */ + const char *dest, /* job-printer-uri */ + *format, /* document-format */ + *title, /* job-name */ + *user; /* job-originating-user-name */ + char uri[HTTP_MAX_URI]; /* URI for jobs */ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + static const char * const attrs[] = /* Requested attributes */ + { + "document-format", + "job-id", + "job-k-octets", + "job-name", + "job-originating-user-name", + "job-printer-uri", + "job-priority", + "job-state", + "time-at-completed", + "time-at-creation", + "time-at-processing" + }; + + + /* + * Range check input... + */ + + if (!jobs) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + + return (-1); + } + + /* + * Get the right URI... + */ + + if (name) + { + if (httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", name) != HTTP_URI_OK) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("Unable to create printer-uri"), 1); + + return (-1); + } + } + else + strcpy(uri, "ipp://localhost/"); + + if (!http) + if ((http = _cupsConnect()) == NULL) + return (-1); + + /* + * Build an IPP_GET_JOBS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requesting-user-name + * which-jobs + * my-jobs + * requested-attributes + */ + + request = ippNewRequest(IPP_GET_JOBS); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, uri); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, cupsUser()); + + if (myjobs) + ippAddBoolean(request, IPP_TAG_OPERATION, "my-jobs", 1); + + if (whichjobs == CUPS_WHICHJOBS_COMPLETED) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "which-jobs", NULL, "completed"); + else if (whichjobs == CUPS_WHICHJOBS_ALL) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "which-jobs", NULL, "all"); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", sizeof(attrs) / sizeof(attrs[0]), + NULL, attrs); + + /* + * Do the request and get back a response... + */ + + n = 0; + *jobs = NULL; + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + for (attr = response->attrs; attr; attr = attr->next) + { + /* + * Skip leading attributes until we hit a job... + */ + + while (attr && attr->group_tag != IPP_TAG_JOB) + attr = attr->next; + + if (!attr) + break; + + /* + * Pull the needed attributes from this job... + */ + + id = 0; + size = 0; + priority = 50; + state = IPP_JOB_PENDING; + user = "unknown"; + dest = NULL; + format = "application/octet-stream"; + title = "untitled"; + creation_time = 0; + completed_time = 0; + processing_time = 0; + + while (attr && attr->group_tag == IPP_TAG_JOB) + { + if (!strcmp(attr->name, "job-id") && + attr->value_tag == IPP_TAG_INTEGER) + id = attr->values[0].integer; + else if (!strcmp(attr->name, "job-state") && + attr->value_tag == IPP_TAG_ENUM) + state = (ipp_jstate_t)attr->values[0].integer; + else if (!strcmp(attr->name, "job-priority") && + attr->value_tag == IPP_TAG_INTEGER) + priority = attr->values[0].integer; + else if (!strcmp(attr->name, "job-k-octets") && + attr->value_tag == IPP_TAG_INTEGER) + size = attr->values[0].integer; + else if (!strcmp(attr->name, "time-at-completed") && + attr->value_tag == IPP_TAG_INTEGER) + completed_time = attr->values[0].integer; + else if (!strcmp(attr->name, "time-at-creation") && + attr->value_tag == IPP_TAG_INTEGER) + creation_time = attr->values[0].integer; + else if (!strcmp(attr->name, "time-at-processing") && + attr->value_tag == IPP_TAG_INTEGER) + processing_time = attr->values[0].integer; + else if (!strcmp(attr->name, "job-printer-uri") && + attr->value_tag == IPP_TAG_URI) + { + if ((dest = strrchr(attr->values[0].string.text, '/')) != NULL) + dest ++; + } + else if (!strcmp(attr->name, "job-originating-user-name") && + attr->value_tag == IPP_TAG_NAME) + user = attr->values[0].string.text; + else if (!strcmp(attr->name, "document-format") && + attr->value_tag == IPP_TAG_MIMETYPE) + format = attr->values[0].string.text; + else if (!strcmp(attr->name, "job-name") && + (attr->value_tag == IPP_TAG_TEXT || + attr->value_tag == IPP_TAG_NAME)) + title = attr->values[0].string.text; + + attr = attr->next; + } + + /* + * See if we have everything needed... + */ + + if (!dest || !id) + { + if (!attr) + break; + else + continue; + } + + /* + * Allocate memory for the job... + */ + + if (n == 0) + temp = malloc(sizeof(cups_job_t)); + else + temp = realloc(*jobs, sizeof(cups_job_t) * (n + 1)); + + if (!temp) + { + /* + * Ran out of memory! + */ + + _cupsSetError(IPP_INTERNAL_ERROR, NULL, 0); + + cupsFreeJobs(n, *jobs); + *jobs = NULL; + + ippDelete(response); + + return (-1); + } + + *jobs = temp; + temp += n; + n ++; + + /* + * Copy the data over... + */ + + temp->dest = _cupsStrAlloc(dest); + temp->user = _cupsStrAlloc(user); + temp->format = _cupsStrAlloc(format); + temp->title = _cupsStrAlloc(title); + temp->id = id; + temp->priority = priority; + temp->state = state; + temp->size = size; + temp->completed_time = completed_time; + temp->creation_time = creation_time; + temp->processing_time = processing_time; + + if (!attr) + break; + } + + ippDelete(response); + } + + if (n == 0 && cg->last_error >= IPP_BAD_REQUEST) + return (-1); + else + return (n); +} + + +/* + * 'cupsGetPPD()' - Get the PPD file for a printer on the default server. + * + * For classes, @code cupsGetPPD@ returns the PPD file for the first printer + * in the class. + * + * The returned filename is stored in a static buffer and is overwritten with + * each call to @code cupsGetPPD@ or @link cupsGetPPD2@. The caller "owns" the + * file that is created and must @code unlink@ the returned filename. + */ + +const char * /* O - Filename for PPD file */ +cupsGetPPD(const char *name) /* I - Destination name */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + time_t modtime = 0; /* Modification time */ + + + /* + * Return the PPD file... + */ + + cg->ppd_filename[0] = '\0'; + + if (cupsGetPPD3(CUPS_HTTP_DEFAULT, name, &modtime, cg->ppd_filename, + sizeof(cg->ppd_filename)) == HTTP_OK) + return (cg->ppd_filename); + else + return (NULL); +} + + +/* + * 'cupsGetPPD2()' - Get the PPD file for a printer from the specified server. + * + * For classes, @code cupsGetPPD2@ returns the PPD file for the first printer + * in the class. + * + * The returned filename is stored in a static buffer and is overwritten with + * each call to @link cupsGetPPD@ or @code cupsGetPPD2@. The caller "owns" the + * file that is created and must @code unlink@ the returned filename. + * + * @since CUPS 1.1.21/Mac OS X 10.4@ + */ + +const char * /* O - Filename for PPD file */ +cupsGetPPD2(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + const char *name) /* I - Destination name */ +{ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + time_t modtime = 0; /* Modification time */ + + + cg->ppd_filename[0] = '\0'; + + if (cupsGetPPD3(http, name, &modtime, cg->ppd_filename, + sizeof(cg->ppd_filename)) == HTTP_OK) + return (cg->ppd_filename); + else + return (NULL); +} + + +/* + * 'cupsGetPPD3()' - Get the PPD file for a printer on the specified + * server if it has changed. + * + * The "modtime" parameter contains the modification time of any + * locally-cached content and is updated with the time from the PPD file on + * the server. + * + * The "buffer" parameter contains the local PPD filename. If it contains + * the empty string, a new temporary file is created, otherwise the existing + * file will be overwritten as needed. The caller "owns" the file that is + * created and must @code unlink@ the returned filename. + * + * On success, @code HTTP_OK@ is returned for a new PPD file and + * @code HTTP_NOT_MODIFIED@ if the existing PPD file is up-to-date. Any other + * status is an error. + * + * For classes, @code cupsGetPPD3@ returns the PPD file for the first printer + * in the class. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +http_status_t /* O - HTTP status */ +cupsGetPPD3(http_t *http, /* I - HTTP connection or @code CUPS_HTTP_DEFAULT@ */ + const char *name, /* I - Destination name */ + time_t *modtime, /* IO - Modification time */ + char *buffer, /* I - Filename buffer */ + size_t bufsize) /* I - Size of filename buffer */ +{ + int http_port; /* Port number */ + char http_hostname[HTTP_MAX_HOST]; + /* Hostname associated with connection */ + http_t *http2; /* Alternate HTTP connection */ + int fd; /* PPD file */ + char localhost[HTTP_MAX_URI],/* Local hostname */ + hostname[HTTP_MAX_URI], /* Hostname */ + resource[HTTP_MAX_URI]; /* Resource name */ + int port; /* Port number */ + http_status_t status; /* HTTP status from server */ + _cups_globals_t *cg = _cupsGlobals(); /* Pointer to library globals */ + + + /* + * Range check input... + */ + + DEBUG_printf(("cupsGetPPD3(http=%p, name=\"%s\", modtime=%p(%d), buffer=%p, " + "bufsize=%d)", http, name, modtime, + modtime ? (int)*modtime : 0, buffer, (int)bufsize)); + + if (!name) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("No printer name"), 1); + return (HTTP_NOT_ACCEPTABLE); + } + + if (!modtime) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("No modification time"), 1); + return (HTTP_NOT_ACCEPTABLE); + } + + if (!buffer || bufsize <= 1) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("Bad filename buffer"), 1); + return (HTTP_NOT_ACCEPTABLE); + } + +#ifndef WIN32 + /* + * See if the PPD file is available locally... + */ + + if (!cg->servername[0]) + cupsServer(); + + if (!_cups_strcasecmp(cg->servername, "localhost")) + { + char ppdname[1024]; /* PPD filename */ + struct stat ppdinfo; /* PPD file information */ + + + snprintf(ppdname, sizeof(ppdname), "%s/ppd/%s.ppd", cg->cups_serverroot, + name); + if (!stat(ppdname, &ppdinfo)) + { + /* + * OK, the file exists, use it! + */ + + if (buffer[0]) + { + unlink(buffer); + + if (symlink(ppdname, buffer) && errno != EEXIST) + { + _cupsSetError(IPP_INTERNAL_ERROR, NULL, 0); + + return (HTTP_SERVER_ERROR); + } + } + else + { + int tries; /* Number of tries */ + const char *tmpdir; /* TMPDIR environment variable */ + struct timeval curtime; /* Current time */ + + /* + * Previously we put root temporary files in the default CUPS temporary + * directory under /var/spool/cups. However, since the scheduler cleans + * out temporary files there and runs independently of the user apps, we + * don't want to use it unless specifically told to by cupsd. + */ + + if ((tmpdir = getenv("TMPDIR")) == NULL) +# ifdef __APPLE__ + tmpdir = "/private/tmp"; /* /tmp is a symlink to /private/tmp */ +# else + tmpdir = "/tmp"; +# endif /* __APPLE__ */ + + /* + * Make the temporary name using the specified directory... + */ + + tries = 0; + + do + { + /* + * Get the current time of day... + */ + + gettimeofday(&curtime, NULL); + + /* + * Format a string using the hex time values... + */ + + snprintf(buffer, bufsize, "%s/%08lx%05lx", tmpdir, + (unsigned long)curtime.tv_sec, + (unsigned long)curtime.tv_usec); + + /* + * Try to make a symlink... + */ + + if (!symlink(ppdname, buffer)) + break; + + tries ++; + } + while (tries < 1000); + + if (tries >= 1000) + { + _cupsSetError(IPP_INTERNAL_ERROR, NULL, 0); + + return (HTTP_SERVER_ERROR); + } + } + + if (*modtime >= ppdinfo.st_mtime) + return (HTTP_NOT_MODIFIED); + else + { + *modtime = ppdinfo.st_mtime; + return (HTTP_OK); + } + } + } +#endif /* !WIN32 */ + + /* + * Try finding a printer URI for this printer... + */ + + if (!http) + if ((http = _cupsConnect()) == NULL) + return (HTTP_SERVICE_UNAVAILABLE); + + if (!cups_get_printer_uri(http, name, hostname, sizeof(hostname), &port, + resource, sizeof(resource), 0)) + return (HTTP_NOT_FOUND); + + DEBUG_printf(("2cupsGetPPD3: Printer hostname=\"%s\", port=%d", hostname, + port)); + + /* + * Remap local hostname to localhost... + */ + + httpGetHostname(NULL, localhost, sizeof(localhost)); + + DEBUG_printf(("2cupsGetPPD3: Local hostname=\"%s\"", localhost)); + + if (!_cups_strcasecmp(localhost, hostname)) + strcpy(hostname, "localhost"); + + /* + * Get the hostname and port number we are connected to... + */ + + httpGetHostname(http, http_hostname, sizeof(http_hostname)); + http_port = _httpAddrPort(http->hostaddr); + + DEBUG_printf(("2cupsGetPPD3: Connection hostname=\"%s\", port=%d", + http_hostname, http_port)); + + /* + * Reconnect to the correct server as needed... + */ + + if (!_cups_strcasecmp(http_hostname, hostname) && port == http_port) + http2 = http; + else if ((http2 = httpConnectEncrypt(hostname, port, + cupsEncryption())) == NULL) + { + DEBUG_puts("1cupsGetPPD3: Unable to connect to server"); + + return (HTTP_SERVICE_UNAVAILABLE); + } + + /* + * Get a temp file... + */ + + if (buffer[0]) + fd = open(buffer, O_CREAT | O_TRUNC | O_WRONLY, 0600); + else + fd = cupsTempFd(buffer, bufsize); + + if (fd < 0) + { + /* + * Can't open file; close the server connection and return NULL... + */ + + _cupsSetError(IPP_INTERNAL_ERROR, NULL, 0); + + if (http2 != http) + httpClose(http2); + + return (HTTP_SERVER_ERROR); + } + + /* + * And send a request to the HTTP server... + */ + + strlcat(resource, ".ppd", sizeof(resource)); + + if (*modtime > 0) + httpSetField(http2, HTTP_FIELD_IF_MODIFIED_SINCE, + httpGetDateString(*modtime)); + + status = cupsGetFd(http2, resource, fd); + + close(fd); + + /* + * See if we actually got the file or an error... + */ + + if (status == HTTP_OK) + *modtime = httpGetDateTime(httpGetField(http2, HTTP_FIELD_DATE)); + else if (status != HTTP_NOT_MODIFIED) + { + _cupsSetHTTPError(status); + + unlink(cg->ppd_filename); + } + + if (http2 != http) + httpClose(http2); + + /* + * Return the PPD file... + */ + + DEBUG_printf(("1cupsGetPPD3: Returning status %d", status)); + + return (status); +} + + +/* + * 'cupsGetPrinters()' - Get a list of printers from the default server. + * + * This function is deprecated - use @link cupsGetDests@ instead. + * + * @deprecated@ + */ + +int /* O - Number of printers */ +cupsGetPrinters(char ***printers) /* O - Printers */ +{ + int n; /* Number of printers */ + ipp_t *request, /* IPP Request */ + *response; /* IPP Response */ + ipp_attribute_t *attr; /* Current attribute */ + char **temp; /* Temporary pointer */ + http_t *http; /* Connection to server */ + + + /* + * Range check input... + */ + + if (!printers) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + + return (0); + } + + *printers = NULL; + + /* + * Try to connect to the server... + */ + + if ((http = _cupsConnect()) == NULL) + return (0); + + /* + * Build a CUPS_GET_PRINTERS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * requested-attributes + */ + + request = ippNewRequest(CUPS_GET_PRINTERS); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", NULL, "printer-name"); + + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_ENUM, + "printer-type", 0); + + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_ENUM, + "printer-type-mask", CUPS_PRINTER_CLASS); + + /* + * Do the request and get back a response... + */ + + n = 0; + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + for (attr = response->attrs; attr != NULL; attr = attr->next) + if (attr->name != NULL && + _cups_strcasecmp(attr->name, "printer-name") == 0 && + attr->value_tag == IPP_TAG_NAME) + { + if (n == 0) + temp = malloc(sizeof(char *)); + else + temp = realloc(*printers, sizeof(char *) * (n + 1)); + + if (temp == NULL) + { + /* + * Ran out of memory! + */ + + while (n > 0) + { + n --; + free((*printers)[n]); + } + + free(*printers); + ippDelete(response); + return (0); + } + + *printers = temp; + temp[n] = strdup(attr->values[0].string.text); + n ++; + } + + ippDelete(response); + } + + return (n); +} + + +/* + * 'cupsGetServerPPD()' - Get an available PPD file from the server. + * + * This function returns the named PPD file from the server. The + * list of available PPDs is provided by the IPP @code CUPS_GET_PPDS@ + * operation. + * + * You must remove (unlink) the PPD file when you are finished with + * it. The PPD filename is stored in a static location that will be + * overwritten on the next call to @link cupsGetPPD@, @link cupsGetPPD2@, + * or @link cupsGetServerPPD@. + * + * @since CUPS 1.3/Mac OS X 10.5@ + */ + +char * /* O - Name of PPD file or @code NULL@ on error */ +cupsGetServerPPD(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + const char *name) /* I - Name of PPD file ("ppd-name") */ +{ + int fd; /* PPD file descriptor */ + ipp_t *request; /* IPP request */ + _cups_globals_t *cg = _cupsGlobals(); + /* Pointer to library globals */ + + + /* + * Range check input... + */ + + if (!name) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("No PPD name"), 1); + + return (NULL); + } + + if (!http) + if ((http = _cupsConnect()) == NULL) + return (NULL); + + /* + * Get a temp file... + */ + + if ((fd = cupsTempFd(cg->ppd_filename, sizeof(cg->ppd_filename))) < 0) + { + /* + * Can't open file; close the server connection and return NULL... + */ + + _cupsSetError(IPP_INTERNAL_ERROR, NULL, 0); + + return (NULL); + } + + /* + * Get the PPD file... + */ + + request = ippNewRequest(CUPS_GET_PPD); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "ppd-name", NULL, + name); + + ippDelete(cupsDoIORequest(http, request, "/", -1, fd)); + + close(fd); + + if (cupsLastError() != IPP_OK) + { + unlink(cg->ppd_filename); + return (NULL); + } + else + return (cg->ppd_filename); +} + + +/* + * 'cupsPrintFile()' - Print a file to a printer or class on the default server. + */ + +int /* O - Job ID or 0 on error */ +cupsPrintFile(const char *name, /* I - Destination name */ + const char *filename, /* I - File to print */ + const char *title, /* I - Title of job */ + int num_options,/* I - Number of options */ + cups_option_t *options) /* I - Options */ +{ + DEBUG_printf(("cupsPrintFile(name=\"%s\", filename=\"%s\", " + "title=\"%s\", num_options=%d, options=%p)", + name, filename, title, num_options, options)); + + return (cupsPrintFiles2(CUPS_HTTP_DEFAULT, name, 1, &filename, title, + num_options, options)); +} + + +/* + * 'cupsPrintFile2()' - Print a file to a printer or class on the specified + * server. + * + * @since CUPS 1.1.21/Mac OS X 10.4@ + */ + +int /* O - Job ID or 0 on error */ +cupsPrintFile2( + http_t *http, /* I - Connection to server */ + const char *name, /* I - Destination name */ + const char *filename, /* I - File to print */ + const char *title, /* I - Title of job */ + int num_options, /* I - Number of options */ + cups_option_t *options) /* I - Options */ +{ + DEBUG_printf(("cupsPrintFile2(http=%p, name=\"%s\", filename=\"%s\", " + "title=\"%s\", num_options=%d, options=%p)", + http, name, filename, title, num_options, options)); + + return (cupsPrintFiles2(http, name, 1, &filename, title, num_options, + options)); +} + + +/* + * 'cupsPrintFiles()' - Print one or more files to a printer or class on the + * default server. + */ + +int /* O - Job ID or 0 on error */ +cupsPrintFiles( + const char *name, /* I - Destination name */ + int num_files, /* I - Number of files */ + const char **files, /* I - File(s) to print */ + const char *title, /* I - Title of job */ + int num_options, /* I - Number of options */ + cups_option_t *options) /* I - Options */ +{ + DEBUG_printf(("cupsPrintFiles(name=\"%s\", num_files=%d, " + "files=%p, title=\"%s\", num_options=%d, options=%p)", + name, num_files, files, title, num_options, options)); + + /* + * Print the file(s)... + */ + + return (cupsPrintFiles2(CUPS_HTTP_DEFAULT, name, num_files, files, title, + num_options, options)); +} + + +/* + * 'cupsPrintFiles2()' - Print one or more files to a printer or class on the + * specified server. + * + * @since CUPS 1.1.21/Mac OS X 10.4@ + */ + +int /* O - Job ID or 0 on error */ +cupsPrintFiles2( + http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + const char *name, /* I - Destination name */ + int num_files, /* I - Number of files */ + const char **files, /* I - File(s) to print */ + const char *title, /* I - Title of job */ + int num_options, /* I - Number of options */ + cups_option_t *options) /* I - Options */ +{ + int i; /* Looping var */ + int job_id; /* New job ID */ + const char *docname; /* Basename of current filename */ + const char *format; /* Document format */ + cups_file_t *fp; /* Current file */ + char buffer[8192]; /* Copy buffer */ + ssize_t bytes; /* Bytes in buffer */ + http_status_t status; /* Status of write */ + _cups_globals_t *cg = _cupsGlobals(); /* Global data */ + ipp_status_t cancel_status; /* Status code to preserve */ + char *cancel_message; /* Error message to preserve */ + + + DEBUG_printf(("cupsPrintFiles2(http=%p, name=\"%s\", num_files=%d, " + "files=%p, title=\"%s\", num_options=%d, options=%p)", + http, name, num_files, files, title, num_options, options)); + + /* + * Range check input... + */ + + if (!name || num_files < 1 || !files) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(EINVAL), 0); + + return (0); + } + + /* + * Create the print job... + */ + + if ((job_id = cupsCreateJob(http, name, title, num_options, options)) == 0) + return (0); + + /* + * Send each of the files... + */ + + if (cupsGetOption("raw", num_options, options)) + format = CUPS_FORMAT_RAW; + else if ((format = cupsGetOption("document-format", num_options, + options)) == NULL) + format = CUPS_FORMAT_AUTO; + + for (i = 0; i < num_files; i ++) + { + /* + * Start the next file... + */ + + if ((docname = strrchr(files[i], '/')) != NULL) + docname ++; + else + docname = files[i]; + + if ((fp = cupsFileOpen(files[i], "rb")) == NULL) + { + /* + * Unable to open print file, cancel the job and return... + */ + + _cupsSetError(IPP_DOCUMENT_ACCESS_ERROR, NULL, 0); + goto cancel_job; + } + + status = cupsStartDocument(http, name, job_id, docname, format, + i == (num_files - 1)); + + while (status == HTTP_CONTINUE && + (bytes = cupsFileRead(fp, buffer, sizeof(buffer))) > 0) + status = cupsWriteRequestData(http, buffer, bytes); + + cupsFileClose(fp); + + if (status != HTTP_CONTINUE || cupsFinishDocument(http, name) != IPP_OK) + { + /* + * Unable to queue, cancel the job and return... + */ + + goto cancel_job; + } + } + + return (job_id); + + /* + * If we get here, something happened while sending the print job so we need + * to cancel the job without setting the last error (since we need to preserve + * the current error... + */ + + cancel_job: + + cancel_status = cg->last_error; + cancel_message = cg->last_status_message ? + _cupsStrRetain(cg->last_status_message) : NULL; + + cupsCancelJob2(http, name, job_id, 0); + + cg->last_error = cancel_status; + cg->last_status_message = cancel_message; + + return (0); +} + + +/* + * 'cupsStartDocument()' - Add a document to a job created with cupsCreateJob(). + * + * Use @link cupsWriteRequestData@ to write data for the document and + * @link cupsFinishDocument@ to finish the document and get the submission status. + * + * The MIME type constants @code CUPS_FORMAT_AUTO@, @code CUPS_FORMAT_PDF@, + * @code CUPS_FORMAT_POSTSCRIPT@, @code CUPS_FORMAT_RAW@, and + * @code CUPS_FORMAT_TEXT@ are provided for the "format" argument, although + * any supported MIME type string can be supplied. + * + * @since CUPS 1.4/Mac OS X 10.6@ + */ + +http_status_t /* O - HTTP status of request */ +cupsStartDocument( + http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */ + const char *name, /* I - Destination name */ + int job_id, /* I - Job ID from @link cupsCreateJob@ */ + const char *docname, /* I - Name of document */ + const char *format, /* I - MIME type or @code CUPS_FORMAT_foo@ */ + int last_document) /* I - 1 for last document in job, 0 otherwise */ +{ + char resource[1024], /* Resource for destinatio */ + printer_uri[1024]; /* Printer URI */ + ipp_t *request; /* Send-Document request */ + http_status_t status; /* HTTP status */ + + + /* + * Create a Send-Document request... + */ + + if ((request = ippNewRequest(IPP_SEND_DOCUMENT)) == NULL) + { + _cupsSetError(IPP_INTERNAL_ERROR, strerror(ENOMEM), 0); + return (0); + } + + httpAssembleURIf(HTTP_URI_CODING_ALL, printer_uri, sizeof(printer_uri), "ipp", + NULL, "localhost", ippPort(), "/printers/%s", name); + snprintf(resource, sizeof(resource), "/printers/%s", name); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, printer_uri); + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_INTEGER, "job-id", job_id); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, cupsUser()); + if (docname) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "document-name", + NULL, docname); + if (format) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_MIMETYPE, + "document-format", NULL, format); + ippAddBoolean(request, IPP_TAG_OPERATION, "last-document", last_document); + + /* + * Send and delete the request, then return the status... + */ + + status = cupsSendRequest(http, request, resource, CUPS_LENGTH_VARIABLE); + + ippDelete(request); + + return (status); +} + + +/* + * 'cups_get_printer_uri()' - Get the printer-uri-supported attribute for the + * first printer in a class. + */ + +static int /* O - 1 on success, 0 on failure */ +cups_get_printer_uri( + http_t *http, /* I - Connection to server */ + const char *name, /* I - Name of printer or class */ + char *host, /* I - Hostname buffer */ + int hostsize, /* I - Size of hostname buffer */ + int *port, /* O - Port number */ + char *resource, /* I - Resource buffer */ + int resourcesize, /* I - Size of resource buffer */ + int depth) /* I - Depth of query */ +{ + int i; /* Looping var */ + int http_port; /* Port number */ + http_t *http2; /* Alternate HTTP connection */ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + ipp_attribute_t *attr; /* Current attribute */ + char uri[HTTP_MAX_URI], /* printer-uri attribute */ + scheme[HTTP_MAX_URI], /* Scheme name */ + username[HTTP_MAX_URI], /* Username:password */ + classname[255], /* Temporary class name */ + http_hostname[HTTP_MAX_HOST]; + /* Hostname associated with connection */ + static const char * const requested_attrs[] = + { /* Requested attributes */ + "member-uris", + "printer-uri-supported", + "printer-type" + }; + + + DEBUG_printf(("7cups_get_printer_uri(http=%p, name=\"%s\", host=%p, " + "hostsize=%d, resource=%p, resourcesize=%d, depth=%d)", + http, name, host, hostsize, resource, resourcesize, depth)); + + /* + * Setup the printer URI... + */ + + if (httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", name) != HTTP_URI_OK) + { + _cupsSetError(IPP_INTERNAL_ERROR, _("Unable to create printer-uri"), 1); + + *host = '\0'; + *resource = '\0'; + + return (0); + } + + DEBUG_printf(("9cups_get_printer_uri: printer-uri=\"%s\"", uri)); + + /* + * Get the hostname and port number we are connected to... + */ + + httpGetHostname(http, http_hostname, sizeof(http_hostname)); + http_port = _httpAddrPort(http->hostaddr); + + /* + * Build an IPP_GET_PRINTER_ATTRIBUTES request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requested-attributes + */ + + request = ippNewRequest(IPP_GET_PRINTER_ATTRIBUTES); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requested-attributes", + sizeof(requested_attrs) / sizeof(requested_attrs[0]), + NULL, requested_attrs); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(http, request, "/")) != NULL) + { + if ((attr = ippFindAttribute(response, "member-uris", IPP_TAG_URI)) != NULL) + { + /* + * Get the first actual printer name in the class... + */ + + for (i = 0; i < attr->num_values; i ++) + { + httpSeparateURI(HTTP_URI_CODING_ALL, attr->values[i].string.text, + scheme, sizeof(scheme), username, sizeof(username), + host, hostsize, port, resource, resourcesize); + if (!strncmp(resource, "/printers/", 10)) + { + /* + * Found a printer! + */ + + ippDelete(response); + + return (1); + } + } + + /* + * No printers in this class - try recursively looking for a printer, + * but not more than 3 levels deep... + */ + + if (depth < 3) + { + for (i = 0; i < attr->num_values; i ++) + { + httpSeparateURI(HTTP_URI_CODING_ALL, attr->values[i].string.text, + scheme, sizeof(scheme), username, sizeof(username), + host, hostsize, port, resource, resourcesize); + if (!strncmp(resource, "/classes/", 9)) + { + /* + * Found a class! Connect to the right server... + */ + + if (!_cups_strcasecmp(http_hostname, host) && *port == http_port) + http2 = http; + else if ((http2 = httpConnectEncrypt(host, *port, + cupsEncryption())) == NULL) + { + DEBUG_puts("8cups_get_printer_uri: Unable to connect to server"); + + continue; + } + + /* + * Look up printers on that server... + */ + + strlcpy(classname, resource + 9, sizeof(classname)); + + cups_get_printer_uri(http2, classname, host, hostsize, port, + resource, resourcesize, depth + 1); + + /* + * Close the connection as needed... + */ + + if (http2 != http) + httpClose(http2); + + if (*host) + return (1); + } + } + } + } + else if ((attr = ippFindAttribute(response, "printer-uri-supported", + IPP_TAG_URI)) != NULL) + { + httpSeparateURI(HTTP_URI_CODING_ALL, + _httpResolveURI(attr->values[0].string.text, uri, + sizeof(uri), _HTTP_RESOLVE_DEFAULT, + NULL, NULL), + scheme, sizeof(scheme), username, sizeof(username), + host, hostsize, port, resource, resourcesize); + ippDelete(response); + + if (!strncmp(resource, "/classes/", 9)) + { + _cupsSetError(IPP_INTERNAL_ERROR, + _("No printer-uri found for class"), 1); + + *host = '\0'; + *resource = '\0'; + + return (0); + } + + return (1); + } + + ippDelete(response); + } + + if (cupsLastError() != IPP_NOT_FOUND) + _cupsSetError(IPP_INTERNAL_ERROR, _("No printer-uri found"), 1); + + *host = '\0'; + *resource = '\0'; + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/cups/versioning.h b/cups/versioning.h new file mode 100644 index 0000000000..2e6447af54 --- /dev/null +++ b/cups/versioning.h @@ -0,0 +1,92 @@ +/* + * "$Id$" + * + * API versioning definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +#ifndef _CUPS_VERSIONING_H_ +# define _CUPS_VERSIONING_H_ + +/* + * This header defines several constants - _CUPS_DEPRECATED, + * _CUPS_API_1_1, _CUPS_API_1_1_19, _CUPS_API_1_1_20, _CUPS_API_1_1_21, + * _CUPS_API_1_2, _CUPS_API_1_3, _CUPS_API_1_4, _CUPS_API_1_5, _CUPS_API_1_6 - + * which add compiler-specific attributes that flag functions that are + * deprecated or added in particular releases. + * + * On Mac OS X, the _CUPS_API_* constants are defined based on the values of + * the MAC_OS_X_VERSION_MIN_ALLOWED and MAC_OS_X_VERSION_MAX_ALLOWED constants + * provided by the compiler. + * + * + */ + +# if defined(__APPLE__) && !defined(_CUPS_SOURCE) +# include +# ifndef AVAILABLE_MAC_OS_X_VERSION_10_3_AND_LATER +# define AVAILABLE_MAC_OS_X_VERSION_10_3_AND_LATER __attribute__((unavailable)) +# endif /* !AVAILABLE_MAC_OS_X_VERSION_10_3_AND_LATER */ +# ifndef AVAILABLE_MAC_OS_X_VERSION_10_4_AND_LATER +# define AVAILABLE_MAC_OS_X_VERSION_10_4_AND_LATER __attribute__((unavailable)) +# endif /* !AVAILABLE_MAC_OS_X_VERSION_10_4_AND_LATER */ +# ifndef AVAILABLE_MAC_OS_X_VERSION_10_5_AND_LATER +# define AVAILABLE_MAC_OS_X_VERSION_10_5_AND_LATER __attribute__((unavailable)) +# endif /* !AVAILABLE_MAC_OS_X_VERSION_10_5_AND_LATER */ +# ifndef AVAILABLE_MAC_OS_X_VERSION_10_6_AND_LATER +# define AVAILABLE_MAC_OS_X_VERSION_10_6_AND_LATER __attribute__((unavailable)) +# endif /* !AVAILABLE_MAC_OS_X_VERSION_10_6_AND_LATER */ +# ifndef AVAILABLE_MAC_OS_X_VERSION_10_7_AND_LATER +# define AVAILABLE_MAC_OS_X_VERSION_10_7_AND_LATER __attribute__((unavailable)) +# endif /* !AVAILABLE_MAC_OS_X_VERSION_10_7_AND_LATER */ +# ifndef AVAILABLE_MAC_OS_X_VERSION_10_8_AND_LATER +# define AVAILABLE_MAC_OS_X_VERSION_10_8_AND_LATER __attribute__((unavailable)) +# endif /* !AVAILABLE_MAC_OS_X_VERSION_10_8_AND_LATER */ +# define _CUPS_API_1_1_19 AVAILABLE_MAC_OS_X_VERSION_10_3_AND_LATER +# define _CUPS_API_1_1_20 AVAILABLE_MAC_OS_X_VERSION_10_4_AND_LATER +# define _CUPS_API_1_1_21 AVAILABLE_MAC_OS_X_VERSION_10_4_AND_LATER +# define _CUPS_API_1_2 AVAILABLE_MAC_OS_X_VERSION_10_5_AND_LATER +# define _CUPS_API_1_3 AVAILABLE_MAC_OS_X_VERSION_10_5_AND_LATER +# define _CUPS_API_1_4 AVAILABLE_MAC_OS_X_VERSION_10_6_AND_LATER +# define _CUPS_API_1_5 AVAILABLE_MAC_OS_X_VERSION_10_7_AND_LATER +# define _CUPS_API_1_6 +# else +# define _CUPS_API_1_1_19 +# define _CUPS_API_1_1_20 +# define _CUPS_API_1_1_21 +# define _CUPS_API_1_2 +# define _CUPS_API_1_3 +# define _CUPS_API_1_4 +# define _CUPS_API_1_5 +# define _CUPS_API_1_6 +# endif /* __APPLE__ && !_CUPS_SOURCE */ + +/* + * With GCC 3.0 and higher, we can mark old APIs "deprecated" so you get + * a warning at compile-time. + */ + +# if defined(__GNUC__) && __GNUC__ > 2 && !defined(_CUPS_SOURCE) +# define _CUPS_DEPRECATED __attribute__ ((__deprecated__)) +# else +# define _CUPS_DEPRECATED +# endif /* __GNUC__ && __GNUC__ > 2 && !_CUPS_SOURCE */ + +# ifndef __GNUC__ +# define __attribute__(x) +# endif /* !__GNUC__ */ + +#endif /* !_CUPS_VERSIONING_H_ */ + +/* + * End of "$Id$". + */ diff --git a/data/Makefile b/data/Makefile new file mode 100644 index 0000000000..4f7f68034e --- /dev/null +++ b/data/Makefile @@ -0,0 +1,150 @@ +# +# "$Id$" +# +# Datafile makefile for CUPS. +# +# Copyright 2007-2011 by Apple Inc. +# Copyright 1993-2006 by Easy Software Products. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +include ../Makedefs + +# +# Data files... +# + +BANNERS = \ + classified \ + confidential \ + secret \ + standard \ + topsecret \ + unclassified + +DATAFILES = \ + testprint + +PPDCFILES = \ + epson.h \ + font.defs \ + hp.h \ + label.h \ + media.defs \ + raster.defs + + +# +# Make everything... +# + +all: + + +# +# Make library targets... +# + +libs: + + +# +# Make unit tests... +# + +unittests: + + +# +# Clean all config and object files... +# + +clean: + + +# +# Dummy depend... +# + +depend: + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: + $(INSTALL_DIR) -m 755 $(DATADIR)/banners + for file in $(BANNERS); do \ + $(INSTALL_DATA) $$file $(DATADIR)/banners; \ + done + $(INSTALL_DIR) -m 755 $(DATADIR)/data + for file in $(DATAFILES); do \ + $(INSTALL_DATA) $$file $(DATADIR)/data; \ + done + $(INSTALL_DIR) -m 755 $(DATADIR)/model + $(INSTALL_DIR) -m 755 $(DATADIR)/ppdc + for file in $(PPDCFILES); do \ + $(INSTALL_DATA) $$file $(DATADIR)/ppdc; \ + done + $(INSTALL_DIR) -m 755 $(DATADIR)/profiles + + +# +# Install programs... +# + +install-exec: + + +# +# Install headers... +# + +install-headers: + + +# +# Install libraries... +# + +install-libs: + + +# +# Uninstall files... +# + +uninstall: + for file in $(BANNERS); do \ + $(RM) $(DATADIR)/banners/$$file; \ + done + for file in $(DATAFILES); do \ + $(RM) $(DATADIR)/data/$$file; \ + done + for file in $(PPDCFILES); do \ + $(RM) $(DATADIR)/ppdc/$$file; \ + done + -$(RMDIR) $(DATADIR)/profiles + -$(RMDIR) $(DATADIR)/ppdc + -$(RMDIR) $(DATADIR)/model + -$(RMDIR) $(DATADIR)/data + -$(RMDIR) $(DATADIR)/banners + -$(RMDIR) $(DATADIR) + + +# +# End of "$Id$". +# diff --git a/data/classified b/data/classified new file mode 100644 index 0000000000..ca01a13b7d --- /dev/null +++ b/data/classified @@ -0,0 +1,6 @@ +#CUPS-BANNER +Show job-id job-name job-originating-user-name job-originating-host-name job-billing +Header Classified +Footer Classified +Image images/cups.png + diff --git a/data/confidential b/data/confidential new file mode 100644 index 0000000000..f1f0d854db --- /dev/null +++ b/data/confidential @@ -0,0 +1,6 @@ +#CUPS-BANNER +Show job-id job-name job-originating-user-name job-originating-host-name job-billing +Header Confidential +Footer Confidential +Image images/cups.png + diff --git a/data/cups.irix b/data/cups.irix new file mode 100644 index 0000000000..476383acb5 --- /dev/null +++ b/data/cups.irix @@ -0,0 +1,3 @@ +#%PAM-1.0 +auth required pam_unix.so shadow nodelay nullok +account required pam_unix.so diff --git a/data/cups.pam b/data/cups.pam new file mode 100644 index 0000000000..f38e70184a --- /dev/null +++ b/data/cups.pam @@ -0,0 +1,2 @@ +auth required /lib/security/pam_pwdb.so nullok shadow +account required /lib/security/pam_pwdb.so diff --git a/data/cups.suse b/data/cups.suse new file mode 100644 index 0000000000..a9369e1619 --- /dev/null +++ b/data/cups.suse @@ -0,0 +1,2 @@ +auth required pam_unix2.so nullok shadow +account required pam_unix2.so diff --git a/data/epson.h b/data/epson.h new file mode 100644 index 0000000000..13cb1a3464 --- /dev/null +++ b/data/epson.h @@ -0,0 +1,27 @@ +/* + * "$Id$" + * + * This file contains model number definitions for the CUPS sample + * ESC/P driver. + * + * Copyright 2007 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#define EPSON_9PIN 0 /* 9-pin dot matrix */ +#define EPSON_24PIN 1 /* 24-pin dot matrix */ +#define EPSON_COLOR 2 /* Epson Stylus Color with ESC . */ +#define EPSON_PHOTO 3 /* Epson Stylus Photo with ESC . */ +#define EPSON_ICOLOR 4 /* Epson Stylus Color with ESC i */ +#define EPSON_IPHOTO 5 /* Epson Stylus Photo with ESC i */ + + +/* + * End of "$Id$". + */ diff --git a/data/font.defs b/data/font.defs new file mode 100644 index 0000000000..ee4559e82e --- /dev/null +++ b/data/font.defs @@ -0,0 +1,55 @@ +/* + * "$Id: font.defs 343 2007-07-13 19:52:48Z mike $" + * + * Standard ESP Ghostscript font definitions for the CUPS PPD file + * compiler. + * + * Copyright 2007 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#font AvantGarde-Book Standard "(1.05)" Standard ROM +#font AvantGarde-BookOblique Standard "(1.05)" Standard ROM +#font AvantGarde-Demi Standard "(1.05)" Standard ROM +#font AvantGarde-DemiOblique Standard "(1.05)" Standard ROM +#font Bookman-Demi Standard "(1.05)" Standard ROM +#font Bookman-DemiItalic Standard "(1.05)" Standard ROM +#font Bookman-Light Standard "(1.05)" Standard ROM +#font Bookman-LightItalic Standard "(1.05)" Standard ROM +#font Courier Standard "(1.05)" Standard ROM +#font Courier-Bold Standard "(1.05)" Standard ROM +#font Courier-BoldOblique Standard "(1.05)" Standard ROM +#font Courier-Oblique Standard "(1.05)" Standard ROM +#font Helvetica Standard "(1.05)" Standard ROM +#font Helvetica-Bold Standard "(1.05)" Standard ROM +#font Helvetica-BoldOblique Standard "(1.05)" Standard ROM +#font Helvetica-Narrow Standard "(1.05)" Standard ROM +#font Helvetica-Narrow-Bold Standard "(1.05)" Standard ROM +#font Helvetica-Narrow-BoldOblique Standard "(1.05)" Standard ROM +#font Helvetica-Narrow-Oblique Standard "(1.05)" Standard ROM +#font Helvetica-Oblique Standard "(1.05)" Standard ROM +#font NewCenturySchlbk-Bold Standard "(1.05)" Standard ROM +#font NewCenturySchlbk-BoldItalic Standard "(1.05)" Standard ROM +#font NewCenturySchlbk-Italic Standard "(1.05)" Standard ROM +#font NewCenturySchlbk-Roman Standard "(1.05)" Standard ROM +#font Palatino-Bold Standard "(1.05)" Standard ROM +#font Palatino-BoldItalic Standard "(1.05)" Standard ROM +#font Palatino-Italic Standard "(1.05)" Standard ROM +#font Palatino-Roman Standard "(1.05)" Standard ROM +#font Symbol Special "(001.005)" Special ROM +#font Times-Bold Standard "(1.05)" Standard ROM +#font Times-BoldItalic Standard "(1.05)" Standard ROM +#font Times-Italic Standard "(1.05)" Standard ROM +#font Times-Roman Standard "(1.05)" Standard ROM +#font ZapfChancery-MediumItalic Standard "(1.05)" Standard ROM +#font ZapfDingbats Special "(001.005)" Special ROM + +/* + * End of "$Id: font.defs 343 2007-07-13 19:52:48Z mike $". + */ diff --git a/data/hp.h b/data/hp.h new file mode 100644 index 0000000000..c700343163 --- /dev/null +++ b/data/hp.h @@ -0,0 +1,24 @@ +/* + * "$Id$" + * + * This file contains model number definitions for the CUPS sample + * HP driver. + * + * Copyright 2007 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#define HP_LASERJET 0 /* HP LaserJet */ +#define HP_DESKJET 1 /* HP DeskJet with simple color */ +#define HP_DESKJET2 2 /* HP DeskJet with CRet color */ + + +/* + * End of "$Id$". + */ diff --git a/data/label.h b/data/label.h new file mode 100644 index 0000000000..4b6fba64ff --- /dev/null +++ b/data/label.h @@ -0,0 +1,28 @@ +/* + * "$Id$" + * + * This file contains model number definitions for the CUPS sample + * label printer driver. + * + * Copyright 2007 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#define DYMO_3x0 0 /* Dymo Labelwriter 300/330/330 Turbo */ + +#define ZEBRA_EPL_LINE 0x10 /* Zebra EPL line mode printers */ +#define ZEBRA_EPL_PAGE 0x11 /* Zebra EPL page mode printers */ +#define ZEBRA_ZPL 0x12 /* Zebra ZPL-based printers */ +#define ZEBRA_CPCL 0x13 /* Zebra CPCL-based printers */ + +#define INTELLITECH_PCL 0x20 /* Intellitech PCL-based printers */ + +/* + * End of "$Id$". + */ diff --git a/data/media.defs b/data/media.defs new file mode 100644 index 0000000000..43b48c352d --- /dev/null +++ b/data/media.defs @@ -0,0 +1,208 @@ +/* + * "$Id: media.defs 343 2007-07-13 19:52:48Z mike $" + * + * Adobe standard media size definitions for the CUPS PPD file compiler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These codedinstructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * copyright law. Distribution and use rights are outlinedin the file + * "LICENSE.txt" which should have beenincluded with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#media "3x5/3 x 5" 216 360 +#media "3.5x5/3.5 x 5" 252 360 +#media "5x7/5 x 7" 360 504 +#media "10x11/10 x 11" 720 792 +#media "10x13/10 x 13" 720 936 +#media "10x14/10 x 14" 720 1008 +#media "12x11/12 x 11" 864 792 +#media "15x11/15 x 11" 1080 792 +#media "7x9/7 x 9" 504 648 +#media "8x10/8 x 10" 576 720 +#media "9x11/9 x 11" 648 792 +#media "9x12/9 x 12" 648 864 +#media "A0/A0" 2384 3370 +#media "A0.Transverse/A0 Long Edge" 3370 2384 +#media "A1/A1" 1684 2384 +#media "A1.Transverse/A1 Long Edge" 2384 1684 +#media "A2/A2" 1191 1684 +#media "A2.Transverse/A2 Long Edge" 1684 1191 +#media "A3/A3" 842 1191 +#media "A3.Transverse/A3 Long Edge" 1191 842 +#media "A3Extra/A3 Oversize" 913 1262 +#media "A3Extra.Transverse/A3 Oversize Long Edge" 913 1262 +#media "A3Rotated/A3 Long Edge" 1191 842 +#media "A4/A4" 595 842 +#media "A4Extra/A4 Oversize" 667 914 +#media "A4Plus/A4 Oversize" 595 936 +#media "A4Rotated/A4 Long Edge" 842 595 +#media "A4Small/A4 Small" 595 842 +#media "A4.Transverse/A4 Long Edge" 842 595 +#media "A5/A5" 420 595 +#media "A5Extra/A5 Oversize" 492 668 +#media "A5Rotated/A5 Long Edge" 595 420 +#media "A5.Transverse/A5 Long Edge" 595 420 +#media "A6/A6" 297 420 +#media "A6Rotated/A6 Long Edge" 420 297 +#media "A7/A7" 210 297 +#media "A8/A8" 148 210 +#media "A9/A9" 105 148 +#media "A10/A10" 73 105 +#media "AnsiA/ANSI A" 612 792 +#media "AnsiB/ANSI B" 792 1224 +#media "AnsiC/ANSI C" 1224 1584 +#media "AnsiD/ANSI D" 1584 2448 +#media "AnsiE/ANSI E" 2448 3168 +#media "ARCHA/Letter Oversize" 648 864 +#media "ARCHA.Transverse/Letter Oversize Long Edge" 864 648 +#media "ARCHB/Tabloid Oversize" 864 1296 +#media "ARCHB.Transverse/Tabloid Oversize Long Edge" 1296 864 +#media "ARCHC/ARCH C" 1296 1728 +#media "ARCHC.Transverse/ARCH C Long Edge" 1728 1296 +#media "ARCHD/ARCH D" 1728 2592 +#media "ARCHD.Transverse/ARCH D Long Edge" 2592 1728 +#media "ARCHE/ARCH E" 2592 3456 +#media "ARCHE.Transverse/ARCH E Long Edge" 3456 2592 +#media "B0/JIS B0" 2920 4127 +#media "B10/JIS B10" 91 127 +#media "B1/JIS B1" 2064 2918 +#media "B1/JIS B1" 2064 2920 +#media "B2/JIS B2" 1460 2064 +#media "B3/JIS B3" 1032 1460 +#media "B4/JIS B4" 729 1032 +#media "B4Rotated/JIS B4 Long Edge" 1032 729 +#media "B5/JIS B5" 516 729 +#media "B5Rotated/JIS B5 Long Edge" 729 516 +#media "B5.Transverse/JIS B5 Long Edge" 516 729 +#media "B6/JIS B6" 363 516 +#media "B6Rotated/JIS B6 Long Edge" 516 363 +#media "B7/JIS B7" 258 363 +#media "B8/JIS B8" 181 258 +#media "B9/JIS B9" 127 181 +#media "C4/Envelope C4" 649 918 +#media "C5/Envelope C5" 459 649 +#media "C6/Envelope C6" 323 459 +#media "DL/Envelope DL" 312 624 +#media "DoublePostcard/Postcard Double " 567 420 +#media "DoublePostcardRotated/Postcard Double Long Edge" 420 567 +#media "Env10/Envelope #10 " 297 684 +#media "Env11/Envelope #11" 324 747 +#media "Env12/Envelope #12" 342 792 +#media "Env14/Envelope #14" 360 828 +#media "Env9/Envelope #9" 279 639 +#media "EnvC0/Envelope C0" 2599 3676 +#media "EnvC1/Envelope C1" 1837 2599 +#media "EnvC2/Envelope C2" 1298 1837 +#media "EnvC3/Envelope C3" 918 1296 +#media "EnvC4/Envelope C4" 649 918 +#media "EnvC5/Envelope C5" 459 649 +#media "EnvC65/Envelope C65" 324 648 +#media "EnvC6/Envelope C6" 323 459 +#media "EnvC7/Envelope C7" 230 323 +#media "EnvChou3/Envelope Choukei 3" 340 666 +#media "EnvChou3Rotated/Envelope Choukei 3 Long Edge" 666 340 +#media "EnvChou4/Envelope Choukei 4" 255 581 +#media "EnvChou4Rotated/Envelope Choukei 4 Long Edge" 581 255 +#media "EnvDL/Envelope DL" 312 624 +#media "EnvInvite/Envelope Invite" 624 624 +#media "EnvISOB4/Envelope B4" 708 1001 +#media "EnvISOB5/Envelope B5" 499 709 +#media "EnvISOB6/Envelope B6" 499 354 +#media "EnvItalian/Envelope Italian" 312 652 +#media "EnvKaku2/Envelope Kaku2" 680 941 +#media "EnvKaku2Rotated/Envelope Kaku2 Long Edge" 941 680 +#media "EnvKaku3/Envelope Kaku3" 612 785 +#media "EnvKaku3Rotated/Envelope Kaku3 Long Edge" 785 612 +#media "EnvMonarch/Envelope Monarch" 279 540 +#media "EnvPersonal/Envelope Personal" 261 468 +#media "EnvPRC1/Envelope PRC1 " 289 468 +#media "EnvPRC1Rotated/Envelope PRC1 Long Edge" 468 289 +#media "EnvPRC2/Envelope PRC2" 289 499 +#media "EnvPRC2Rotated/Envelope PRC2 Long Edge" 499 289 +#media "EnvPRC3/Envelope PRC3" 354 499 +#media "EnvPRC3Rotated/Envelope PRC3 Long Edge" 499 354 +#media "EnvPRC4/Envelope PRC4" 312 590 +#media "EnvPRC4Rotated/Envelope PRC4 Long Edge" 590 312 +#media "EnvPRC5/Envelope PRC5PRC5" 312 624 +#media "EnvPRC5Rotated/Envelope PRC5 Long Edge" 624 312 +#media "EnvPRC6/Envelope PRC6" 340 652 +#media "EnvPRC6Rotated/Envelope PRC6 Long Edge" 652 340 +#media "EnvPRC7/Envelope PRC7" 454 652 +#media "EnvPRC7Rotated/Envelope PRC7 Long Edge" 652 454 +#media "EnvPRC8/Envelope PRC8" 340 876 +#media "EnvPRC8Rotated/Envelope PRC8 Long Edge" 876 340 +#media "EnvPRC9/Envelope PRC9" 649 918 +#media "EnvPRC9Rotated/Envelope PRC9 Long Edge" 918 649 +#media "EnvPRC10/Envelope PRC10" 918 1298 +#media "EnvPRC10Rotated/Envelope PRC10 Long Edge" 1298 918 +#media "EnvYou4/Envelope You4" 298 666 +#media "EnvYou4Rotated/Envelope You4 Long Edge" 666 298 +#media "Executive/Executive" 522 756 +#media "FanFoldGerman/FanFold German" 612 864 +#media "FanFoldGermanLegal/FanFold Legal German" 612 936 +#media "FanFoldUS/Fanfold US" 1071 792 +#media "Folio/Folio" 595 935 +#media "ISOB0/B0" 2835 4008 +#media "ISOB1/B1" 2004 2835 +#media "ISOB2/B2" 1417 2004 +#media "ISOB3/B3" 1001 1417 +#media "ISOB4/B4" 709 1001 +#media "ISOB5/B5" 499 709 +#media "ISOB5Extra/B5 Oversize" 570 782 +#media "ISOB6/B6" 354 499 +#media "ISOB7/B7" 249 354 +#media "ISOB8/B8" 176 249 +#media "ISOB9/B9" 125 176 +#media "ISOB10/B10" 88 125 +#media "Ledger/US Ledger" 1224 792 +#media "Legal/US Legal" 612 1008 +#media "LegalExtra/US Legal Oversize" 684 1080 +#media "Letter/US Letter" 612 792 +#media "Letter.Transverse/US Letter Long Edge" 792 612 +#media "LetterExtra/US Letter Oversize" 684 864 +#media "LetterExtra.Transverse/US Letter Oversize Long Edge" 864 684 +#media "LetterPlus/US Letter Oversize" 612 914 +#media "LetterRotated/US Letter Long Edge" 792 612 +#media "LetterSmall/US Letter Small" 612 792 +#media "Monarch/Envelope Monarch" 279 540 +#media "Note/Note" 612 792 +#media "Postcard/Postcard" 284 419 +#media "PostcardRotated/Postcard Long Edge" 419 284 +#media "PRC16K/PRC16K" 414 610 +#media "PRC16KRotated/PRC16K Long Edge" 610 414 +#media "PRC32K/PRC32K" 275 428 +#media "PRC32KBig/PRC32K Oversize" 275 428 +#media "PRC32KBigRotated/PRC32K Oversize Long Edge" 428 275 +#media "PRC32KRotated/PRC32K Long Edge" 428 275 +#media "Quarto/Quarto" 610 780 +#media "Statement/Statement" 396 612 +#media "SuperA/Super A" 643 1009 +#media "SuperB/Super B" 864 1380 +#media "Tabloid/Tabloid" 792 1224 +#media "TabloidExtra/Tabloid Oversize" 864 1296 + +/* + * Non-standard sizes... + */ + +#media "Photo4x6/Photo" 288 432 +#media "PhotoLabel/Photo Labels" 288 468 +#media "w936h1368/Super B/A3" 936 1368 +#media "w81h252/Address" 81 252 +#media "w101h252/Large Address" 101 252 +#media "w54h144/Return Address" 54 144 +#media "w167h288/Shipping Address" 167 288 +#media "w162h540/Internet Postage 2-Part" 162 540 +#media "w162h504/Internet Postage 3-Part" 162 504 +#media "w41h248/File Folder" 41 248 +#media "w41h144/Hanging Folder" 41 144 +#media "w153h198/3.5\" Disk" 153 198 + + +/* + * End of "$Id: media.defs 343 2007-07-13 19:52:48Z mike $". + */ diff --git a/data/raster.defs b/data/raster.defs new file mode 100644 index 0000000000..486165f012 --- /dev/null +++ b/data/raster.defs @@ -0,0 +1,94 @@ +/* + * "$Id: raster.defs 343 2007-07-13 19:52:48Z mike $" + * + * This file contains the standard definitions for enumerated attributes + * in the CUPS raster page device dictionary. + * + * Copyright 2007 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +/* Jog values */ +#define CUPS_JOG_NONE 0 /* Never move pages */ +#define CUPS_JOG_FILE 1 /* Move pages after this file */ +#define CUPS_JOG_JOB 2 /* Move pages after this job */ +#define CUPS_JOG_SET 3 /* Move pages after this set */ + +/* Orientation values */ +#define CUPS_ORIENT_0 0 /* Don't rotate the page */ +#define CUPS_ORIENT_90 1 /* Rotate the page counter-clockwise */ +#define CUPS_ORIENT_180 2 /* Turn the page upside down */ +#define CUPS_ORIENT_270 3 /* Rotate the page clockwise */ + +/* CutMedia values */ +#define CUPS_CUT_NONE 0 /* Never cut the roll */ +#define CUPS_CUT_FILE 1 /* Cut the roll after this file */ +#define CUPS_CUT_JOB 2 /* Cut the roll after this job */ +#define CUPS_CUT_SET 3 /* Cut the roll after this set */ +#define CUPS_CUT_PAGE 4 /* Cut the roll after this page */ + +/* AdvanceMedia values */ +#define CUPS_ADVANCE_NONE 0 /* Never advance the roll */ +#define CUPS_ADVANCE_FILE 1 /* Advance the roll after this file */ +#define CUPS_ADVANCE_JOB 2 /* Advance the roll after this job */ +#define CUPS_ADVANCE_SET 3 /* Advance the roll after this set */ +#define CUPS_ADVANCE_PAGE 4 /* Advance the roll after this page */ + +/* LeadingEdge values */ +#define CUPS_EDGE_TOP 0 /* Leading edge is the top of the page */ +#define CUPS_EDGE_RIGHT 1 /* Leading edge is the right of the page */ +#define CUPS_EDGE_BOTTOM 2 /* Leading edge is the bottom of the page */ +#define CUPS_EDGE_LEFT 3 /* Leading edge is the left of the page */ + +/* cupsColorOrder values */ +#define CUPS_ORDER_CHUNKED 0 /* CMYK CMYK CMYK ... */ +#define CUPS_ORDER_BANDED 1 /* CCC MMM YYY KKK ... */ +#define CUPS_ORDER_PLANAR 2 /* CCC ... MMM ... YYY ... KKK ... */ + +/* cupsColorSpace values */ +#define CUPS_CSPACE_W 0 /* Luminance */ +#define CUPS_CSPACE_RGB 1 /* Red, green, blue */ +#define CUPS_CSPACE_RGBA 2 /* Red, green, blue, alpha */ +#define CUPS_CSPACE_K 3 /* Black */ +#define CUPS_CSPACE_CMY 4 /* Cyan, magenta, yellow */ +#define CUPS_CSPACE_YMC 5 /* Yellow, magenta, cyan */ +#define CUPS_CSPACE_CMYK 6 /* Cyan, magenta, yellow, black */ +#define CUPS_CSPACE_YMCK 7 /* Yellow, magenta, cyan, black */ +#define CUPS_CSPACE_KCMY 8 /* Black, cyan, magenta, yellow */ +#define CUPS_CSPACE_KCMYcm 9 /* Black, cyan, magenta, yellow, * + * light-cyan, light-magenta */ +#define CUPS_CSPACE_GMCK 10 /* Gold, magenta, yellow, black */ +#define CUPS_CSPACE_GMCS 11 /* Gold, magenta, yellow, silver */ +#define CUPS_CSPACE_WHITE 12 /* White ink (as black) */ +#define CUPS_CSPACE_GOLD 13 /* Gold foil */ +#define CUPS_CSPACE_SILVER 14 /* Silver foil */ + +#define CUPS_CSPACE_CIEXYZ 15 /* CIE XYZ */ +#define CUPS_CSPACE_CIELab 16 /* CIE Lab */ + +#define CUPS_CSPACE_ICC1 32 /* ICC-based, 1 color */ +#define CUPS_CSPACE_ICC2 33 /* ICC-based, 2 colors */ +#define CUPS_CSPACE_ICC3 34 /* ICC-based, 3 colors */ +#define CUPS_CSPACE_ICC4 35 /* ICC-based, 4 colors */ +#define CUPS_CSPACE_ICC5 36 /* ICC-based, 5 colors */ +#define CUPS_CSPACE_ICC6 37 /* ICC-based, 6 colors */ +#define CUPS_CSPACE_ICC7 38 /* ICC-based, 7 colors */ +#define CUPS_CSPACE_ICC8 39 /* ICC-based, 8 colors */ +#define CUPS_CSPACE_ICC9 40 /* ICC-based, 9 colors */ +#define CUPS_CSPACE_ICCA 41 /* ICC-based, 10 colors */ +#define CUPS_CSPACE_ICCB 42 /* ICC-based, 11 colors */ +#define CUPS_CSPACE_ICCC 43 /* ICC-based, 12 colors */ +#define CUPS_CSPACE_ICCD 44 /* ICC-based, 13 colors */ +#define CUPS_CSPACE_ICCE 45 /* ICC-based, 14 colors */ +#define CUPS_CSPACE_ICCF 46 /* ICC-based, 15 colors */ + + +/* + * End of "$Id: raster.defs 343 2007-07-13 19:52:48Z mike $". + */ diff --git a/data/secret b/data/secret new file mode 100644 index 0000000000..66a0e8a919 --- /dev/null +++ b/data/secret @@ -0,0 +1,6 @@ +#CUPS-BANNER +Show job-id job-name job-originating-user-name job-originating-host-name job-billing +Header Secret +Footer Secret +Image images/cups.png + diff --git a/data/smiley.ps b/data/smiley.ps new file mode 100644 index 0000000000..6ff31fdfbe --- /dev/null +++ b/data/smiley.ps @@ -0,0 +1,28 @@ +%!PS-Adobe-3.0 +%%BoundingBox: 36 36 576 756 +%%Pages: 1 +%%LanguageLevel: 2 +%%EndComments +%%Page: (1) 1 +% Draw a black box around the page +0 setgray +1 setlinewidth +36 36 540 720 rectstroke + +% Draw a two inch blue circle in the middle of the page +0 0 1 setrgbcolor +306 396 144 0 360 arc closepath fill + +% Draw two half inch yellow circles for eyes +1 1 0 setrgbcolor +252 432 36 0 360 arc closepath fill +360 432 36 0 360 arc closepath fill + +% Draw the smile +1 setlinecap +18 setlinewidth +306 396 99 200 340 arc stroke + +% Print it! +showpage +%%EOF diff --git a/data/standard b/data/standard new file mode 100644 index 0000000000..31c68a08ed --- /dev/null +++ b/data/standard @@ -0,0 +1,6 @@ +#CUPS-BANNER +Show job-id job-name job-originating-user-name job-originating-host-name job-billing +Header Cover Page +Footer Cover Page +Image images/cups.png + diff --git a/data/testprint.in b/data/testprint.in new file mode 100644 index 0000000000..8cee3ab4f9 --- /dev/null +++ b/data/testprint.in @@ -0,0 +1,7 @@ +#CUPS-BANNER +Show printer-name printer-info printer-location printer-make-and-model printer-driver-name printer-driver-version paper-size imageable-area +Header Printer Test Page +Footer Printer Test Page +Notice CUPS @CUPS_VERSION@. +Image images/cups.png +Image images/color-wheel.png diff --git a/data/topsecret b/data/topsecret new file mode 100644 index 0000000000..d2b87fabb9 --- /dev/null +++ b/data/topsecret @@ -0,0 +1,6 @@ +#CUPS-BANNER +Show job-id job-name job-originating-user-name job-originating-host-name job-billing +Header Top Secret +Footer Top Secret +Image images/cups.png + diff --git a/data/unclassified b/data/unclassified new file mode 100644 index 0000000000..2d1a7d0f97 --- /dev/null +++ b/data/unclassified @@ -0,0 +1,6 @@ +#CUPS-BANNER +Show job-id job-name job-originating-user-name job-originating-host-name job-billing +Header Unclassified +Footer Unclassified +Image images/cups.png + diff --git a/desktop/Makefile b/desktop/Makefile new file mode 100644 index 0000000000..9d786f6f07 --- /dev/null +++ b/desktop/Makefile @@ -0,0 +1,135 @@ +# +# "$Id$" +# +# Desktop makefile for CUPS. +# +# Copyright 2007-2009 by Apple Inc. +# Copyright 1993-2006 by Easy Software Products. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +include ../Makedefs + + +# +# Make everything... +# + +all: + + +# +# Make library targets... +# + +libs: + + +# +# Make unit tests... +# + +unittests: + + +# +# Clean all config and object files... +# + +clean: + + +# +# Dummy depend... +# + +depend: + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: + if test "x$(DBUSDIR)" != x; then \ + echo Installing cups.conf in $(DBUSDIR)...;\ + $(INSTALL_DIR) -m 755 $(BUILDROOT)$(DBUSDIR)/system.d; \ + $(INSTALL_DATA) cups.conf $(BUILDROOT)$(DBUSDIR)/system.d/cups.conf; \ + fi + if test "x$(MENUDIR)" != x; then \ + echo Installing desktop menu...; \ + $(INSTALL_DIR) -m 755 $(BUILDROOT)$(MENUDIR); \ + $(INSTALL_DATA) cups.desktop $(BUILDROOT)$(MENUDIR); \ + fi + if test "x$(ICONDIR)" != x; then \ + echo Installing desktop icons...; \ + $(INSTALL_DIR) -m 755 $(BUILDROOT)$(ICONDIR)/hicolor/16x16/apps; \ + $(INSTALL_DATA) cups-16.png $(BUILDROOT)$(ICONDIR)/hicolor/16x16/apps/cups.png; \ + $(INSTALL_DIR) -m 755 $(BUILDROOT)$(ICONDIR)/hicolor/32x32/apps; \ + $(INSTALL_DATA) cups-32.png $(BUILDROOT)$(ICONDIR)/hicolor/32x32/apps/cups.png; \ + $(INSTALL_DIR) -m 755 $(BUILDROOT)$(ICONDIR)/hicolor/64x64/apps; \ + $(INSTALL_DATA) cups-64.png $(BUILDROOT)$(ICONDIR)/hicolor/64x64/apps/cups.png; \ + $(INSTALL_DIR) -m 755 $(BUILDROOT)$(ICONDIR)/hicolor/128x128/apps; \ + $(INSTALL_DATA) cups-128.png $(BUILDROOT)$(ICONDIR)/hicolor/128x128/apps/cups.png; \ + fi + + +# +# Install programs... +# + +install-exec: + + +# +# Install headers... +# + +install-headers: + + +# +# Install libraries... +# + +install-libs: + + +# +# Uninstall files... +# + +uninstall: + if test "x$(DBUSDIR)" != x; then \ + echo Uninstalling cups.conf in $(DBUSDIR)...;\ + $(RM) $(BUILDROOT)$(DBUSDIR)/cups.conf; \ + $(RMDIR) $(BUILDROOT)$(DBUSDIR); \ + fi + if test "x$(MENUDIR)" != x; then \ + echo Uninstalling desktop menu...; \ + $(RM) $(BUILDROOT)$(MENUDIR)/cups.desktop; \ + fi + if test "x$(ICONDIR)" != x; then \ + echo Uninstalling desktop icons...; \ + $(RM) $(BUILDROOT)$(ICONDIR)/hicolor/16x16/apps/cups.png; \ + $(RM) $(BUILDROOT)$(ICONDIR)/hicolor/32x32/apps/cups.png; \ + $(RM) $(BUILDROOT)$(ICONDIR)/hicolor/64x64/apps/cups.png; \ + $(RM) $(BUILDROOT)$(ICONDIR)/hicolor/128x128/apps/cups.png; \ + fi + + +# +# End of "$Id$". +# diff --git a/desktop/cups-128.png b/desktop/cups-128.png new file mode 100644 index 0000000000000000000000000000000000000000..5c0f6ff20b6a79d23874eb72563f4fdc32b6ae1e GIT binary patch literal 4888 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 tmSx{M?l$pXH=6ELQ^z=ND%=A1SEn= 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 I71!nlAN6Mw77Tjc^{wG zdEP1Lz=iWX_ul{cpO1U)O%dV$l-V=clv0|Di!qGFID|2bN5o`LU=h)3pdXv?I*ub6 z#$ScV1DMwex25qLu(ylRZsQ0ZoQ}wJh7%s|V&EV?!5jD(zjj&oAqIOUfLYkq0Jt3^ zNDX^dU<;1b_PmDKy%M;$266)Tv`pDjjMw&WX-xpDYF_+-4Q&gyU@XHA6^uzNY)jyU z8v8piuMoQumt$ekuf{rDge5hcjWwJ*>$Xz85v7!t6y|4@Uq6-HuE4d$vk45AwDgxk zyAiiW#K{U)KlWpO`So^0tgqT@YD60=#^KS3IG$2Eh}9U9kK`WQR0uAWd&3ZZo7xx= z6Dg&;aas9wBBj(95hqj@)x%${{NFLBGIxR8`OZV19QLIpI_F|hFLe?TjS;w{qVI}` zqa~ad;y!#Y*F(amh0t=jC!CW~T90?i{$=~X%l}%P$L99h{OXG5gGFCcfLF_@SYGrcm?NiXpluEG*S-fT z(OD_qay#U+-zTSNs6=E`Zqu9O8W@&O#5z3eR9$pPy#~f>-^(ld1m2VL`ji~%kFZaE z8+?Oz^<4B3dW}Mvy_q0~rMp8=W%a4m;`6~VvyCdS~p0Q>^;J=i=0UucMa*88~Q~&?~ M07*qoM6N<$f~`{K-T(jq literal 0 Hc-jL100001 diff --git a/desktop/cups-512.png b/desktop/cups-512.png new file mode 100644 index 0000000000000000000000000000000000000000..54a1b3bdcf22b97a1c286199f221c056f9f44452 GIT binary patch literal 21029 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 + + + + + + + + + + + diff --git a/desktop/cups.desktop.in b/desktop/cups.desktop.in new file mode 100644 index 0000000000..44bb115f82 --- /dev/null +++ b/desktop/cups.desktop.in @@ -0,0 +1,39 @@ +[Desktop Entry] +Categories=System;Printing;HardwareSettings;X-Red-Hat-Base; +Exec=@CUPS_HTMLVIEW@ http://localhost:631/ +Icon=cups +StartupNotify=false +Terminal=false +Type=Application +Name=Manage Printing +Comment=CUPS Web Interface +Name[de]=Druckerverwaltung +Comment[de]=CUPS Webinterface +Name[en_US]=Manage Printing +Comment[en_US]=CUPS Web Interface +Name[es]=Administrar impresión +Comment[es]=Interfaz Web de CUPS +Name[et]=Trükkimise haldur +Comment[et]=CUPS-i veebiliides +Name[eu]=Kudeatu inprimaketak +Comment[eu]=CUPSen web interfazea +Name[fr]=Gestionnaire d'impression +Comment[fr]=Interface Web de CUPS +Name[he]=נהל הדפסות +Comment[he]=ממשק דפדפן של CUPS +Name[id]=Manajemen Pencetakan +Comment[id]=Antarmuka Web CUPS +Name[it]=Gestione stampa +Comment[it]=Interfaccia web di CUPS +Name[ja]=印刷の管理 +Comment[ja]=CUPS Web インタフェース +Name[pl]=Zarządzanie drukowaniem +Comment[pl]=Interfejs WWW CUPS +Name[ru]=Настройка печати +Comment[ru]=Настройка CUPS +Name[sv]=Hantera skrivare +Comment[sv]=CUPS webb-gränssnitt +Name[zh]=打印机管理 +Comment[zh]=CUPS网页界面 +Name[zh_TW]=印表管理 +Comment[zh_TW]=CUPS 網頁介面 diff --git a/desktop/cups.icns b/desktop/cups.icns new file mode 100644 index 0000000000000000000000000000000000000000..0fd7a35d57afe1926e4b58fcc874e133f345c8c3 GIT binary patch literal 107501 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

FMfP4|Tn$}2 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 MRsaA13_|Vz09H$3z5oCK literal 0 Hc-jL100001 diff --git a/desktop/cups.svg b/desktop/cups.svg new file mode 100644 index 0000000000..8d19c358cb --- /dev/null +++ b/desktop/cups.svg @@ -0,0 +1,533 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + image/svg+xml + + + CUPS Icon + + + Michael Sweet + + + + + Apple Inc. + + + + + + + + + + + + + + + + + + + + diff --git a/doc/Makefile b/doc/Makefile new file mode 100644 index 0000000000..ec7414eb2b --- /dev/null +++ b/doc/Makefile @@ -0,0 +1,266 @@ +# +# "$Id$" +# +# Documentation makefile for CUPS. +# +# Copyright 2007-2011 by Apple Inc. +# Copyright 1997-2007 by Easy Software Products. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +include ../Makedefs + +# +# Document files... +# + +WEBPAGES = \ + cups.css \ + cups-printable.css \ + index.html \ + robots.txt +WEBIMAGES = \ + images/color-wheel.png \ + images/cups.png \ + images/cups-icon.png \ + images/generic.png \ + images/left.gif \ + images/right.gif \ + images/sel.gif \ + images/unsel.gif \ + images/wait.gif +HELPIMAGES = \ + images/cups-block-diagram.png \ + images/cups-command-chain.png \ + images/cups-postscript-chain.png \ + images/cups-raster-chain.png \ + images/raster.png \ + images/raster-organization.png \ + images/sample-image.png \ + images/smiley.jpg +HELPFILES = \ + help/accounting.html \ + help/api-array.html \ + help/api-cgi.html \ + help/api-cups.html \ + help/api-driver.html \ + help/api-filedir.html \ + help/api-filter.html \ + help/api-httpipp.html \ + help/api-mime.html \ + help/api-overview.html \ + help/api-ppd.html \ + help/api-ppdc.html \ + help/api-raster.html \ + help/cgi.html \ + help/glossary.html \ + help/kerberos.html \ + help/license.html \ + help/man-cupsaccept.html \ + help/man-backend.html \ + help/man-cancel.html \ + help/man-cups-config.html \ + help/man-cups-lpd.html \ + help/man-cups-polld.html \ + help/man-cupsaddsmb.html \ + help/man-cupsd.html \ + help/man-cupsenable.html \ + help/man-cupstestdsc.html \ + help/man-cupstestppd.html \ + help/man-filter.html \ + help/man-ipptool.html \ + help/man-ipptoolfile.html \ + help/man-lp.html \ + help/man-lpadmin.html \ + help/man-lpc.html \ + help/man-lpinfo.html \ + help/man-lpmove.html \ + help/man-lpoptions.html \ + help/man-lppasswd.html \ + help/man-lpq.html \ + help/man-lpr.html \ + help/man-lprm.html \ + help/man-lpstat.html \ + help/man-mime.convs.html \ + help/man-mime.types.html \ + help/man-notifier.html \ + help/man-ppdc.html \ + help/man-ppdhtml.html \ + help/man-ppdi.html \ + help/man-ppdmerge.html \ + help/man-ppdpo.html \ + help/network.html \ + help/options.html \ + help/overview.html \ + help/policies.html \ + help/postscript-driver.html \ + help/ppd-compiler.html \ + help/raster-driver.html \ + help/ref-access_log.html \ + help/ref-classes-conf.html \ + help/ref-client-conf.html \ + help/ref-cupsd-conf.html \ + help/ref-error_log.html \ + help/ref-mailto-conf.html \ + help/ref-page_log.html \ + help/ref-ppdcfile.html \ + help/ref-printers-conf.html \ + help/ref-snmp-conf.html \ + help/ref-subscriptions-conf.html \ + help/security.html \ + help/sharing.html \ + help/spec-banner.html \ + help/spec-browsing.html \ + help/spec-cmp.html \ + help/spec-command.html \ + help/spec-design.html \ + help/spec-ipp.html \ + help/spec-pdf.html \ + help/spec-postscript.html \ + help/spec-ppd.html \ + help/spec-raster.html \ + help/spec-stp.html \ + help/standard.html \ + help/translation.html \ + help/whatsnew.html + + +# +# Make all documents... +# + +all: + + +# +# Make library targets... +# + +libs: + + +# +# Make unit tests... +# + +unittests: + + +# +# Remove all generated files... +# + +clean: + + +# +# Dummy depend target... +# + +depend: + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: $(INSTALL_LANGUAGES) + $(INSTALL_DIR) -m 755 $(DOCDIR) + for file in $(WEBPAGES); do \ + $(INSTALL_MAN) $$file $(DOCDIR); \ + done + $(INSTALL_DIR) -m 755 $(DOCDIR)/help + for file in $(HELPFILES); do \ + $(INSTALL_MAN) $$file $(DOCDIR)/help; \ + done + $(INSTALL_DIR) -m 755 $(DOCDIR)/images + for file in $(WEBIMAGES) $(HELPIMAGES); do \ + $(INSTALL_MAN) $$file $(DOCDIR)/images; \ + done + +install-languages: + for lang in $(LANGUAGES); do \ + if test -d $$lang; then \ + $(INSTALL_DIR) -m 755 $(DOCDIR)/$$lang; \ + $(INSTALL_DATA) $$lang/index.html $(DOCDIR)/$$lang; \ + $(INSTALL_DATA) $$lang/cups.css $(DOCDIR)/$$lang >/dev/null 2>&1 || true; \ + fi; \ + done + +install-langbundle: + + +# +# Install programs... +# + +install-exec: + + +# +# Install headers... +# + +install-headers: + + +# +# Install libraries... +# + +install-libs: + + +# +# Uninstall all documentation files... +# + +uninstall: $(UNINSTALL_LANGUAGES) + for file in $(WEBPAGES); do \ + $(RM) $(DOCDIR)/$$file; \ + done + for file in $(HELPFILES); do \ + $(RM) $(DOCDIR)/$$file; \ + done + for file in $(WEBIMAGES); do \ + $(RM) $(DOCDIR)/$$file; \ + done + -$(RMDIR) $(DOCDIR)/images + -$(RMDIR) $(DOCDIR)/help + -$(RMDIR) $(DOCDIR) + +uninstall-languages: + -for lang in $(LANGUAGES); do \ + $(RM) $(DOCDIR)/$$lang/index.html; \ + $(RM) $(DOCDIR)/$$lang/cups.css; \ + $(RMDIR) $(DOCDIR)/$$lang; \ + done + +install-langbundle: + + +# +# Install the docset bits locally... +# + +docset: + cp $(HELPFILES) ../org.cups.docset/Contents/Resources/Documentation/help + cp cups-printable.css ../org.cups.docset/Contents/Resources/Documentation + cp $(HELPIMAGES) ../org.cups.docset/Contents/Resources/Documentation/images + + +# +# End of Makefile. +# diff --git a/doc/cups-printable.css b/doc/cups-printable.css new file mode 100644 index 0000000000..42ea2bee15 --- /dev/null +++ b/doc/cups-printable.css @@ -0,0 +1,329 @@ +BODY { + font-family: lucida grande, geneva, helvetica, arial, sans-serif; +} + +H1, H2, H3, H4, H5, H6, P, TD, TH { + font-family: lucida grande, geneva, helvetica, arial, sans-serif; +} + +KBD { + font-family: monaco, courier, monospace; + font-weight: bold; +} + +PRE { + font-family: monaco, courier, monospace; +} + +PRE.command { + border: dotted thin #7f7f7f; + margin-left: 36pt; + padding: 10px; +} + +P.compact { + margin: 0; +} + +P.example { + font-style: italic; + margin-left: 36pt; +} + +PRE.example { + background: #eeeeee; + border: dotted thin #999999; + margin-left: 36pt; + padding: 10pt; +} + +PRE.command EM, PRE.example EM { + font-family: lucida grande, geneva, helvetica, arial, sans-serif; +} + +P.command { + font-family: monaco, courier, monospace; + margin-left: 36pt; +} + +P.formula { + font-style: italic; + margin-left: 36pt; +} + +BLOCKQUOTE { + background: #eeeeee; + border: solid thin #999999; + padding: 10pt; +} + +A IMG { + border: none; +} + +A:link:hover IMG { + background: #f0f0f0; + border-radius: 10px; + -moz-border-radius: 10px; +} + +A:link, A:visited { + font-weight: normal; + text-decoration: none; +} + +A:link:hover, A:visited:hover, A:active { + text-decoration: underline; +} + +SUB, SUP { + font-size: 50%; +} + +TR.data, TD.data, TR.data TD { + margin-top: 10pt; + padding: 5pt; + border-bottom: solid 1pt #999999; +} + +TR.data TH { + border-bottom: solid 1pt #999999; + padding-top: 10pt; + padding-left: 5pt; + text-align: left; +} + +DIV.table TABLE { + border: solid thin #999999; + border-collapse: collapse; + border-spacing: 0; + margin-left: auto; + margin-right: auto; +} + +DIV.table CAPTION { + caption-side: top; + font-size: 120%; + font-style: italic; + font-weight: bold; + margin-left: auto; + margin-right: auto; +} + +DIV.table TABLE TD { + border: solid thin #cccccc; + padding-top: 5pt; +} + +DIV.table TABLE TH { + background: #cccccc; + border: none; + border-bottom: solid thin #999999; +} + +DIV.figure TABLE { + margin-left: auto; + margin-right: auto; +} + +DIV.figure CAPTION { + caption-side: bottom; + font-size: 120%; + font-style: italic; + font-weight: bold; + margin-left: auto; + margin-right: auto; +} + +TH.label { + text-align: right; + vertical-align: top; +} + +TH.sublabel { + text-align: right; + font-weight: normal; +} + +HR { + border: solid thin; +} + +SPAN.info { + background: black; + border: thin solid black; + color: white; + font-size: 80%; + font-style: italic; + font-weight: bold; + white-space: nowrap; +} + +H2 SPAN.info, H3 SPAN.info, H4 SPAN.info { + float: right; + font-size: 100%; +} + +H1.title { +} + +H2.title, H3.title { + border-bottom: solid 2pt #000000; +} + +DIV.indent, TABLE.indent { + margin-top: 2em; + margin-left: auto; + margin-right: auto; + width: 90%; +} + +TABLE.indent { + border-collapse: collapse; +} + +TABLE.indent TD, TABLE.indent TH { + padding: 0; +} + +TABLE.list { + border-collapse: collapse; + margin-left: auto; + margin-right: auto; + width: 90%; +} + +TABLE.list TH { + background: white; + border-bottom: solid thin #cccccc; + color: #444444; + padding-top: 10pt; + padding-left: 5pt; + text-align: left; + vertical-align: bottom; + white-space: nowrap; +} + +TABLE.list TH A { + color: #4444cc; +} + +TABLE.list TD { + border-bottom: solid thin #eeeeee; + padding-top: 5pt; + padding-left: 5pt; +} + +TABLE.list TR:nth-child(even) { + background: #f8f8f8; +} + +TABLE.list TR:nth-child(odd) { + background: #f4f4f4; +} + +DT { + margin-left: 36pt; + margin-top: 12pt; +} + +DD { + margin-left: 54pt; +} + +DL.category DT { + font-weight: bold; +} + +P.summary { + margin-left: 36pt; + font-family: monaco, courier, monospace; +} + +DIV.summary TABLE { + border: solid thin #999999; + border-collapse: collapse; + border-spacing: 0; + margin: 10px; +} + +DIV.summary TABLE TD, DIV.summary TABLE TH { + border: solid thin #999999; + padding: 5px; + text-align: left; + vertical-align: top; +} + +DIV.summary TABLE THEAD TH { + background: #eeeeee; +} + +/* API documentation styles... */ +div.body h1 { + margin: 0; +} +div.body h2 { + margin-top: 1.5em; +} +div.body h3, div.body h4, div.body h5 { + margin-bottom: 0.5em; + margin-top: 1.5em; +} +.class, .enumeration, .function, .struct, .typedef, .union { + border-bottom: solid thin #999999; + margin-bottom: 0; + margin-top: 2em; +} +.description { + margin-top: 0.5em; +} +code, p.code, pre, ul.code li { + font-family: monaco, courier, monospace; + font-size: 90%; +} +ul.code, ul.contents, ul.subcontents { + list-style-type: none; + margin: 0; + padding-left: 0; +} +ul.code li { + margin: 0; +} +ul.contents > li { + margin-top: 1em; +} +ul.contents li ul.code, ul.contents li ul.subcontents { + padding-left: 2em; +} +div.body dl { + margin-left: 0; + margin-top: 0; +} +div.body dt { + font-style: italic; + margin-left: 0; + margin-top: 0; +} +div.body dd { + margin-bottom: 0.5em; +} + +/* This is just for the HTML files generated with the framedhelp target */ +div.contents { + background: #e8e8e8; + border: solid thin black; + padding: 10px; +} +div.contents h1 { + font-size: 110%; +} +div.contents h2 { + font-size: 100%; +} +div.contents ul.contents { + font-size: 80%; +} +div.contents ul.subcontents li { + margin-left: 1em; + text-indent: -1em; +} diff --git a/doc/cups.css b/doc/cups.css new file mode 100644 index 0000000000..815c193c3f --- /dev/null +++ b/doc/cups.css @@ -0,0 +1,462 @@ +BODY { + background: white; + color: black; + font-family: lucida grande, geneva, helvetica, arial, sans-serif; +} + +H1, H2, H3, H4, H5, H6, P, TD, TH { + font-family: lucida grande, geneva, helvetica, arial, sans-serif; +} + +KBD { + color: #006600; + font-family: monaco, courier, monospace; + font-weight: bold; +} + +PRE { + font-family: monaco, courier, monospace; +} + +PRE.command { + background: #f0f0f0; + border: dotted thin #7f7f7f; + color: #7f0000; + margin-left: 36pt; + padding: 10px; +} + +P.example { + font-style: italic; + margin-left: 36pt; +} + +PRE.example { + background: white; + border: dotted thin #999999; + margin-left: 36pt; + padding: 10px; +} + +PRE.command EM, PRE.example EM { + color: #3f0000; + font-family: lucida grande, geneva, helvetica, arial, sans-serif; +} + +P.command { + color: #7f0000; + font-family: monaco, courier, monospace; + margin-left: 36pt; +} + +P.formula { + font-style: italic; + margin-left: 36pt; +} + +BLOCKQUOTE { + background: #f0f0f0; + border: inset 1px #eeeeee; + padding: 10px; + /* These are not implemented by all browsers, but that's OK */ + border-radius: 5px; + -moz-border-radius: 5px; +} + +A IMG { + border: none; +} + +A:link:hover IMG { + background: #f0f0f0; + border-radius: 10px; + -moz-border-radius: 10px; +} + +A:link, A:visited { + font-weight: inherit; + text-decoration: none; + color: #000099; +} + +A:link:hover, A:visited:hover, A:active { + text-decoration: underline; + color: #990099; +} + +TABLE.page { + border: none; + border-collapse: collapse; + height: 100%; + margin: 0; + padding: 0; + width: 100%; +} + +TD.body { + height: 100%; + vertical-align: top; +} + +TD.sel, TD.unsel { + border-left: thin solid #cccccc; + padding: 0px 5px; + text-align: center; + vertical-align: middle; + width: 14%; +} + +TD.sel { + background: url(images/sel.gif); +} + +TD.unsel { + background: url(images/unsel.gif); +} + +TD.sel A, TD.sel A:hover, TD.unsel A:link:hover, TD.unsel A:visited:hover, +TD.unsel A:active, TD.unsel A, TD.unsel A:visited { + color: #666666; + display: block; + font-weight: normal; + padding: 8px; + text-decoration: none; +} + +TD.trailer { + background: #f0f0f0; + border: solid thin #e0e0e0; + color: #666666; + font-size: 80%; + padding: 5px; +} + +TD.trailer A { + color: #666699; +} + +FORM { + display: inline; +} + +INPUT[TYPE="TEXT"], TEXTAREA { + font-family: monaco, courier, monospace; +} + +INPUT[TYPE="IMAGE"] { + border: none; + padding: 2pt; + vertical-align: bottom; +} + +SUB, SUP { + font-size: 50%; +} + +TR.data, TD.data, TR.data TD { + margin-top: 10pt; + padding: 5pt; + border-bottom: solid 1pt #999999; +} + +TR.data TH { + border-bottom: solid 1pt #999999; + padding-top: 10pt; + padding-left: 5pt; + text-align: left; +} + +DIV.table TABLE { + border: solid thin #999999; + border-collapse: collapse; + border-spacing: 0; + margin-left: auto; + margin-right: auto; +} + +DIV.table CAPTION { + caption-side: top; + font-size: 120%; + font-style: italic; + font-weight: bold; + margin-left: auto; + margin-right: auto; +} + +DIV.table TABLE TD { + background: white; + border: solid thin #bbbbbb; + padding-top: 5pt; +} + +DIV.table TABLE TH { + background: #f0f0f0; + border: none; + border-bottom: solid thin #999999; +} + +DIV.figure TABLE { + margin-left: auto; + margin-right: auto; +} + +DIV.figure CAPTION { + caption-side: bottom; + font-size: 120%; + font-style: italic; + font-weight: bold; + margin-left: auto; + margin-right: auto; +} + +TH.label { + text-align: right; + vertical-align: top; +} + +TH.sublabel { + text-align: right; + font-weight: normal; +} + +HR { + border: solid thin; +} + +SPAN.info { + background: black; + border: thin solid black; + color: white; + font-size: 80%; + font-style: italic; + font-weight: bold; + white-space: nowrap; +} + +H2 SPAN.info, H3 SPAN.info, H4 SPAN.info { + float: right; + font-size: 100%; +} + +.conflict { + background: red; + color: white; +} + +TH.conflict { + text-align: right; +} + +H1.title { + display: none; +} + +H2.title, H3.title { + border-bottom: solid 1pt #999999; +} + +DIV.indent, TABLE.indent { + margin-top: 2em; + margin-left: auto; + margin-right: auto; + width: 90%; +} + +TABLE.indent { + border-collapse: collapse; +} + +TABLE.indent TD, TABLE.indent TH { + padding: 0; +} + +TABLE.list { + border-collapse: collapse; + margin-left: auto; + margin-right: auto; + width: 90%; +} + +TABLE.list TH { + background: white; + border-bottom: solid thin #cccccc; + color: #444444; + padding-top: 10pt; + padding-left: 5pt; + text-align: left; + vertical-align: bottom; + white-space: nowrap; +} + +TABLE.list TH A { + color: #4444cc; +} + +TABLE.list TD { + border-bottom: solid thin #eeeeee; + padding-top: 5pt; + padding-left: 5pt; +} + +TABLE.list TR:nth-child(even) { + background: #f8f8f8; +} + +TABLE.list TR:nth-child(odd) { + background: #f4f4f4; +} + +DIV.sidebar { + float: right; + min-width: 25%; + margin-left: 10px; + max-width: 33%; +} + +DIV.sidebar P.l0 { + margin-bottom: 0; + margin-left: 0; + margin-right: 0; + margin-top: 12pt; +} + +DIV.sidebar P.l1 { + margin-bottom: 0; + margin-left: 36pt; + margin-right: 0; + margin-top: 0; + text-indent: -18pt; +} + +DIV.sidebar P.l2 { + font-style: italic; + margin-bottom: 0; + margin-left: 54pt; + margin-right: 0; + margin-top: 0; + text-indent: -18pt; +} + +TABLE.inset { + background: #f0f0f0; + border: thin solid #e0e0e0; + margin-top: 1em; + padding: 0; + width: 100%; + /* These are not implemented by all browsers, but that's OK */ + border-radius: 5px; + -moz-border-radius: 5px; +} + +TABLE.inset CAPTION { + caption-side: top; + color: #666666; + font-size: 80%; + margin-left: 10px; + margin-bottom: 2px; + text-align: left; +} + +TABLE.inset TD { + padding: 2px; +} + +DT { + margin-left: 36pt; + margin-top: 12pt; +} + +DD { + margin-left: 54pt; +} + +DL.category DT { + font-weight: bold; +} + +P.summary { + margin-left: 36pt; + font-family: monaco, courier, monospace; +} + +DIV.summary TABLE { + border: solid thin #999999; + border-collapse: collapse; + border-spacing: 0; + margin: 10px; +} + +DIV.summary TABLE TD, DIV.summary TABLE TH { + background: white; + border: solid thin #999999; + border-spacing: 0; + padding: 5px; + text-align: left; + vertical-align: top; +} + +DIV.summary TABLE THEAD TH { + background: #f0f0f0; +} + +DIV.tabs { + height: 480px; + overflow: hidden; +} + +DIV.tab { + float: left; + height: 100%; + overflow-y: auto; + width: 100%; +} + +/* API documentation styles... */ +div.body h1 { + margin: 0; +} +div.body h2 { + margin-top: 1.5em; +} +div.body h3, div.body h4, div.body h5 { + margin-bottom: 0.5em; + margin-top: 1.5em; +} +.class, .enumeration, .function, .struct, .typedef, .union { + border-bottom: solid thin #999999; + margin-bottom: 0; + margin-top: 2em; +} +.description { + margin-top: 0.5em; +} +code, p.code, pre, ul.code li { + font-family: monaco, courier, monospace; + font-size: 90%; +} +ul.code, ul.contents, ul.subcontents { + list-style-type: none; + margin: 0; + padding-left: 0; +} +ul.code li { + margin: 0; +} +ul.contents > li { + margin-top: 1em; +} +ul.contents li ul.code, ul.contents li ul.subcontents { + padding-left: 2em; +} +div.body dl { + margin-left: 0; + margin-top: 0; +} +div.body dt { + font-style: italic; + margin-left: 0; + margin-top: 0; +} +div.body dd { + margin-bottom: 0.5em; +} diff --git a/doc/help/accounting.html b/doc/help/accounting.html new file mode 100644 index 0000000000..66c34ec1f3 --- /dev/null +++ b/doc/help/accounting.html @@ -0,0 +1,62 @@ + + + + Printer Accounting Basics + + + + +

Printer Accounting Basics

+ +

CUPS supports a variety of printer accounting schemes. Aside from the +built-in quota and page logging +support, there are several third-party solutions that can be found on +www.cups.org.

+ + +

Quota Support

+ +

CUPS supports page and size-based quotas for each printer. +The quotas are tracked individually for each user, but a single set of +limits applies to all users for a particular printer. For example, you +can limit every user to 5 pages per day on an expensive printer, but +you cannot limit every user except Johnny.

+ +

The job-k-limit, job-page-limit, and job-quota-period +options determine whether and how quotas are enforced for a printer. +The job-quota-period option determines the time interval for +quota tracking. The interval is expressed in seconds, so a day is +86,400, a week is 604,800, and a month is 2,592,000 seconds. The +job-k-limit option specifies the job size limit in killobytes. The +job-page-limit option specifies the number of pages limit.

+ +

For quotas to be enforced, the period and at least one of the limits +must be set to a non-zero value. The following options will enable +weekly quotas with the given size and page count limits:

+ +
+/usr/sbin/lpadmin -p printer -o job-quota-period=604800 \
+    -o job-k-limit=1024 ENTER
+/usr/sbin/lpadmin -p printer -o job-quota-period=604800 \
+    -o job-page-limit=100 ENTER
+
+ +

Or, you can combine all three options on the same line.

+ +

While there is no way to query the current quota state for a particular +user, any application can request a list of jobs for a user and printer that +can be used to easily determine that information.

+ + +

Page Logging

+ +

CUPS logs every page that is printed on a system to the +page_log file. Page logging +is only available for drivers that provide page accounting information, +typically all PostScript and CUPS raster devices. Raw queues and queues +using third-party solutions such as Foomatic generally do not have +useful page accounting information available.

+ + + + diff --git a/doc/help/api-array.html b/doc/help/api-array.html new file mode 100644 index 0000000000..7f22251b81 --- /dev/null +++ b/doc/help/api-array.html @@ -0,0 +1,1062 @@ + + + + + Array API + + + + + + +
+ + +

Array API

+ +
+ + + + + + + + + + + + + + + + +
Headercups/array.h
Library-lcups
See AlsoProgramming: Introduction to CUPS Programming
+

Contents

+ + + +

Overview

+ +

The CUPS array API provides a high-performance generic array container. +The contents of the array container can be sorted and the container itself is +designed for optimal speed and memory usage under a wide variety of conditions. +Sorted arrays use a binary search algorithm from the last found or inserted +element to quickly find matching elements in the array. Arrays created with the +optional hash function can often find elements with a single lookup. The +cups_array_t type is used when +referring to a CUPS array.

+ +

The CUPS scheduler (cupsd) and many of the CUPS API +functions use the array API to efficiently manage large lists of +data.

+ +

Managing Arrays

+ +

Arrays are created using either the +cupsArrayNew, +cupsArrayNew2, or +cupsArrayNew3 functions. The +first function creates a new array with the specified callback function +and user data pointer:

+ +
+#include <cups/array.h>
+
+static int compare_func(void *first, void *second, void *user_data);
+
+void *user_data;
+cups_array_t *array = cupsArrayNew(compare_func, user_data);
+
+ +

The comparison function (type +cups_arrayfunc_t) is called +whenever an element is added to the array and can be NULL to +create an unsorted array. The function returns -1 if the first element should +come before the second, 0 if the first and second elements should have the same +ordering, and 1 if the first element should come after the second.

+ +

The "user_data" pointer is passed to your comparison function. Pass +NULL if you do not need to associate the elements in your array +with additional information.

+ +

The cupsArrayNew2 function adds +two more arguments to support hashed lookups, which can potentially provide +instantaneous ("O(1)") lookups in your array:

+ +
+#include <cups/array.h>
+
+#define HASH_SIZE 512 /* Size of hash table */
+
+static int compare_func(void *first, void *second, void *user_data);
+static int hash_func(void *element, void *user_data);
+
+void *user_data;
+cups_array_t *hash_array = cupsArrayNew2(compare_func, user_data, hash_func, HASH_SIZE);
+
+ +

The hash function (type +cups_ahash_func_t) should return a +number from 0 to (hash_size-1) that (hopefully) uniquely identifies the +element and is called whenever you look up an element in the array with +cupsArrayFind. The hash size is +only limited by available memory, but generally should not be larger than +16384 to realize any performance improvement.

+ +

The cupsArrayNew3 function adds +copy and free callbacks to support basic memory management of elements:

+ +
+#include <cups/array.h>
+
+#define HASH_SIZE 512 /* Size of hash table */
+
+static int compare_func(void *first, void *second, void *user_data);
+static void *copy_func(void *element, void *user_data);
+static void free_func(void *element, void *user_data);
+static int hash_func(void *element, void *user_data);
+
+void *user_data;
+cups_array_t *array = cupsArrayNew3(compare_func, user_data, NULL, 0, copy_func, free_func);
+
+cups_array_t *hash_array = cupsArrayNew3(compare_func, user_data, hash_func, HASH_SIZE, copy_func, free_func);
+
+ +

Once you have created the array, you add elements using the +cupsArrayAdd +cupsArrayInsert functions. +The first function adds an element to the array, adding the new element +after any elements that have the same order, while the second inserts the +element before others with the same order. For unsorted arrays, +cupsArrayAdd appends the element to +the end of the array while +cupsArrayInsert inserts the +element at the beginning of the array. For example, the following code +creates a sorted array of character strings:

+ +
+#include <cups/array.h>
+
+/* Use strcmp() to compare strings - it will ignore the user_data pointer */
+cups_array_t *array = cupsArrayNew((cups_array_func_t)strcmp, NULL);
+
+/* Add four strings to the array */
+cupsArrayAdd(array, "One Fish");
+cupsArrayAdd(array, "Two Fish");
+cupsArrayAdd(array, "Red Fish");
+cupsArrayAdd(array, "Blue Fish");
+
+ +

Elements are removed using the +cupsArrayRemove function, for +example:

+ +
+#include <cups/array.h>
+
+/* Use strcmp() to compare strings - it will ignore the user_data pointer */
+cups_array_t *array = cupsArrayNew((cups_array_func_t)strcmp, NULL);
+
+/* Add four strings to the array */
+cupsArrayAdd(array, "One Fish");
+cupsArrayAdd(array, "Two Fish");
+cupsArrayAdd(array, "Red Fish");
+cupsArrayAdd(array, "Blue Fish");
+
+/* Remove "Red Fish" */
+cupsArrayRemove(array, "Red Fish");
+
+ +

Finally, you free the memory used by the array using the +cupsArrayDelete function. All +of the memory for the array and hash table (if any) is freed, however CUPS +does not free the elements unless you provide copy and free functions.

+ +

Finding and Enumerating Elements

+ +

CUPS provides several functions to find and enumerate elements in an +array. Each one sets or updates a "current index" into the array, such that +future lookups will start where the last one left off:

+ +
+
cupsArrayFind
+
Returns the first matching element.
+
cupsArrayFirst
+
Returns the first element in the array.
+
cupsArrayIndex
+
Returns the Nth element in the array, starting at 0.
+
cupsArrayLast
+
Returns the last element in the array.
+
cupsArrayNext
+
Returns the next element in the array.
+
cupsArrayPrev
+
Returns the previous element in the array.
+
+ +

Each of these functions returns NULL when there is no +corresponding element. For example, a simple for loop using the +cupsArrayFirst and +cupsArrayNext functions will +enumerate all of the strings in our previous example:

+ +
+#include <cups/array.h>
+
+/* Use strcmp() to compare strings - it will ignore the user_data pointer */
+cups_array_t *array = cupsArrayNew((cups_array_func_t)strcmp, NULL);
+
+/* Add four strings to the array */
+cupsArrayAdd(array, "One Fish");
+cupsArrayAdd(array, "Two Fish");
+cupsArrayAdd(array, "Red Fish");
+cupsArrayAdd(array, "Blue Fish");
+
+/* Show all of the strings in the array */
+char *s;
+for (s = (char *)cupsArrayFirst(array); s != NULL; s = (char *)cupsArrayNext(array))
+  puts(s);
+
+

Functions

+

 CUPS 1.2/Mac OS X 10.5 cupsArrayAdd

+

Add an element to the array.

+

+int cupsArrayAdd (
+    cups_array_t *a,
+    void *e
+);

+

Parameters

+
+
a
+
Array
+
e
+
Element
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

When adding an element to a sorted array, non-unique elements are +appended at the end of the run of identical elements. For unsorted arrays, +the element is appended to the end of the array. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsArrayClear

+

Clear the array.

+

+void cupsArrayClear (
+    cups_array_t *a
+);

+

Parameters

+
+
a
+
Array
+
+

Discussion

+

This function is equivalent to removing all elements in the array. +The caller is responsible for freeing the memory used by the +elements themselves. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsArrayCount

+

Get the number of elements in the array.

+

+int cupsArrayCount (
+    cups_array_t *a
+);

+

Parameters

+
+
a
+
Array
+
+

Return Value

+

Number of elements

+

 CUPS 1.2/Mac OS X 10.5 cupsArrayCurrent

+

Return the current element in the array.

+

+void *cupsArrayCurrent (
+    cups_array_t *a
+);

+

Parameters

+
+
a
+
Array
+
+

Return Value

+

Element

+

Discussion

+

The current element is undefined until you call cupsArrayFind, +cupsArrayFirst, or cupsArrayIndex, or cupsArrayLast. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsArrayDelete

+

Free all memory used by the array.

+

+void cupsArrayDelete (
+    cups_array_t *a
+);

+

Parameters

+
+
a
+
Array
+
+

Discussion

+

The caller is responsible for freeing the memory used by the +elements themselves. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsArrayDup

+

Duplicate the array.

+

+cups_array_t *cupsArrayDup (
+    cups_array_t *a
+);

+

Parameters

+
+
a
+
Array
+
+

Return Value

+

Duplicate array

+

 CUPS 1.2/Mac OS X 10.5 cupsArrayFind

+

Find an element in the array.

+

+void *cupsArrayFind (
+    cups_array_t *a,
+    void *e
+);

+

Parameters

+
+
a
+
Array
+
e
+
Element
+
+

Return Value

+

Element found or NULL

+

 CUPS 1.2/Mac OS X 10.5 cupsArrayFirst

+

Get the first element in the array.

+

+void *cupsArrayFirst (
+    cups_array_t *a
+);

+

Parameters

+
+
a
+
Array
+
+

Return Value

+

First element or NULL if the array is empty

+

 CUPS 1.3/Mac OS X 10.5 cupsArrayGetIndex

+

Get the index of the current element.

+

+int cupsArrayGetIndex (
+    cups_array_t *a
+);

+

Parameters

+
+
a
+
Array
+
+

Return Value

+

Index of the current element, starting at 0

+

Discussion

+

The current element is undefined until you call cupsArrayFind, +cupsArrayFirst, or cupsArrayIndex, or cupsArrayLast. + +

+

 CUPS 1.3/Mac OS X 10.5 cupsArrayGetInsert

+

Get the index of the last inserted element.

+

+int cupsArrayGetInsert (
+    cups_array_t *a
+);

+

Parameters

+
+
a
+
Array
+
+

Return Value

+

Index of the last inserted element, starting at 0

+

 CUPS 1.2/Mac OS X 10.5 cupsArrayIndex

+

Get the N-th element in the array.

+

+void *cupsArrayIndex (
+    cups_array_t *a,
+    int n
+);

+

Parameters

+
+
a
+
Array
+
n
+
Index into array, starting at 0
+
+

Return Value

+

N-th element or NULL

+

 CUPS 1.2/Mac OS X 10.5 cupsArrayInsert

+

Insert an element in the array.

+

+int cupsArrayInsert (
+    cups_array_t *a,
+    void *e
+);

+

Parameters

+
+
a
+
Array
+
e
+
Element
+
+

Return Value

+

0 on failure, 1 on success

+

Discussion

+

When inserting an element in a sorted array, non-unique elements are +inserted at the beginning of the run of identical elements. For unsorted +arrays, the element is inserted at the beginning of the array. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsArrayLast

+

Get the last element in the array.

+

+void *cupsArrayLast (
+    cups_array_t *a
+);

+

Parameters

+
+
a
+
Array
+
+

Return Value

+

Last element or NULL if the array is empty

+

 CUPS 1.2/Mac OS X 10.5 cupsArrayNew

+

Create a new array.

+

+cups_array_t *cupsArrayNew (
+    cups_array_func_t f,
+    void *d
+);

+

Parameters

+
+
f
+
Comparison function or NULL for an unsorted array
+
d
+
User data pointer or NULL
+
+

Return Value

+

Array

+

Discussion

+

The comparison function ("f") is used to create a sorted array. The function +receives pointers to two elements and the user data pointer ("d") - the user +data pointer argument can safely be omitted when not required so functions +like strcmp can be used for sorted string arrays. + +

+

 CUPS 1.3/Mac OS X 10.5 cupsArrayNew2

+

Create a new array with hash.

+

+cups_array_t *cupsArrayNew2 (
+    cups_array_func_t f,
+    void *d,
+    cups_ahash_func_t h,
+    int hsize
+);

+

Parameters

+
+
f
+
Comparison function or NULL for an unsorted array
+
d
+
User data or NULL
+
h
+
Hash function or NULL for unhashed lookups
+
hsize
+
Hash size (>= 0)
+
+

Return Value

+

Array

+

Discussion

+

The comparison function ("f") is used to create a sorted array. The function +receives pointers to two elements and the user data pointer ("d") - the user +data pointer argument can safely be omitted when not required so functions +like strcmp can be used for sorted string arrays.
+
+The hash function ("h") is used to implement cached lookups with the +specified hash size ("hsize"). + +

+

 CUPS 1.5/Mac OS X 10.7 cupsArrayNew3

+

Create a new array with hash and/or free function.

+

+cups_array_t *cupsArrayNew3 (
+    cups_array_func_t f,
+    void *d,
+    cups_ahash_func_t h,
+    int hsize,
+    cups_acopy_func_t cf,
+    cups_afree_func_t ff
+);

+

Parameters

+
+
f
+
Comparison function or NULL for an unsorted array
+
d
+
User data or NULL
+
h
+
Hash function or NULL for unhashed lookups
+
hsize
+
Hash size (>= 0)
+
cf
+
Copy function
+
ff
+
Free function
+
+

Return Value

+

Array

+

Discussion

+

The comparison function ("f") is used to create a sorted array. The function +receives pointers to two elements and the user data pointer ("d") - the user +data pointer argument can safely be omitted when not required so functions +like strcmp can be used for sorted string arrays.
+
+The hash function ("h") is used to implement cached lookups with the +specified hash size ("hsize").
+
+The copy function ("cf") is used to automatically copy/retain elements when +added or the array is copied.
+
+The free function ("cf") is used to automatically free/release elements when +removed or the array is deleted. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsArrayNext

+

Get the next element in the array.

+

+void *cupsArrayNext (
+    cups_array_t *a
+);

+

Parameters

+
+
a
+
Array
+
+

Return Value

+

Next element or NULL

+

Discussion

+

This function is equivalent to "cupsArrayIndex(a, cupsArrayGetIndex(a) + 1)".
+
+The next element is undefined until you call cupsArrayFind, +cupsArrayFirst, or cupsArrayIndex, or cupsArrayLast +to set the current element. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsArrayPrev

+

Get the previous element in the array.

+

+void *cupsArrayPrev (
+    cups_array_t *a
+);

+

Parameters

+
+
a
+
Array
+
+

Return Value

+

Previous element or NULL

+

Discussion

+

This function is equivalent to "cupsArrayIndex(a, cupsArrayGetIndex(a) - 1)".
+
+The previous element is undefined until you call cupsArrayFind, +cupsArrayFirst, or cupsArrayIndex, or cupsArrayLast +to set the current element. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsArrayRemove

+

Remove an element from the array.

+

+int cupsArrayRemove (
+    cups_array_t *a,
+    void *e
+);

+

Parameters

+
+
a
+
Array
+
e
+
Element
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

If more than one element matches "e", only the first matching element is +removed.
+
+The caller is responsible for freeing the memory used by the +removed element. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsArrayRestore

+

Reset the current element to the last cupsArraySave.

+

+void *cupsArrayRestore (
+    cups_array_t *a
+);

+

Parameters

+
+
a
+
Array
+
+

Return Value

+

New current element

+

 CUPS 1.2/Mac OS X 10.5 cupsArraySave

+

Mark the current element for a later cupsArrayRestore.

+

+int cupsArraySave (
+    cups_array_t *a
+);

+

Parameters

+
+
a
+
Array
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

The current element is undefined until you call cupsArrayFind, +cupsArrayFirst, or cupsArrayIndex, or cupsArrayLast +to set the current element.
+
+The save/restore stack is guaranteed to be at least 32 elements deep. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsArrayUserData

+

Return the user data for an array.

+

+void *cupsArrayUserData (
+    cups_array_t *a
+);

+

Parameters

+
+
a
+
Array
+
+

Return Value

+

User data

+

Data Types

+

cups_acopy_func_t

+

Array element copy function

+

+typedef void *(*cups_acopy_func_t)(void *element, void *data); +

+

cups_afree_func_t

+

Array element free function

+

+typedef void (*cups_afree_func_t)(void *element, void *data); +

+

cups_ahash_func_t

+

Array hash function

+

+typedef int (*cups_ahash_func_t)(void *element, void *data); +

+

cups_array_func_t

+

Array comparison function

+

+typedef int (*cups_array_func_t)(void *first, void *second, void *data); +

+

cups_array_t

+

CUPS array type

+

+typedef struct _cups_array_s cups_array_t; +

+
+ + diff --git a/doc/help/api-cgi.html b/doc/help/api-cgi.html new file mode 100644 index 0000000000..5c07e58a37 --- /dev/null +++ b/doc/help/api-cgi.html @@ -0,0 +1,1126 @@ + + + + + CGI API + + + + + + +
+ + +

CGI API

+ +
+ + + + + + + + + + + + + + + + +
Headercups/cgi.h
Library-lcupscgi
See AlsoProgramming: Introduction to CUPS Programming
+

Contents

+ + + +

Overview

+ +

The CGI API provides Common Gateway Interface functions for CUPS.

+

Functions

+

cgiCheckVariables

+

Check for the presence of "required" variables.

+

+int cgiCheckVariables (
+    const char *names
+);

+

Parameters

+
+
names
+
Variables to look for
+
+

Return Value

+

1 if all variables present, 0 otherwise

+

Discussion

+

Names may be separated by spaces and/or commas.

+

cgiClearVariables

+

Clear all form variables.

+

+void cgiClearVariables (void);

+

cgiCompileSearch

+

Compile a search string.

+

+void *cgiCompileSearch (
+    const char *query
+);

+

Parameters

+
+
query
+
Query string
+
+

Return Value

+

Search context

+

cgiCopyTemplateFile

+

Copy a template file and replace all the +'{variable}' strings with the variable value.

+

+void cgiCopyTemplateFile (
+    FILE *out,
+    const char *tmpl
+);

+

Parameters

+
+
out
+
Output file
+
tmpl
+
Template file to read
+
+

cgiCopyTemplateLang

+

Copy a template file using a language...

+

+void cgiCopyTemplateLang (
+    const char *tmpl
+);

+

Parameters

+
+
tmpl
+
Base filename
+
+

cgiDoSearch

+

Do a search of some text.

+

+int cgiDoSearch (
+    void *search,
+    const char *text
+);

+

Parameters

+
+
search
+
Search context
+
text
+
Text to search
+
+

Return Value

+

Number of matches

+

cgiEndHTML

+

End a HTML page.

+

+void cgiEndHTML (void);

+

cgiEndMultipart

+

End the delivery of a multipart web page.

+

+void cgiEndMultipart (void);

+

cgiFormEncode

+

Encode a string as a form variable.

+

+char *cgiFormEncode (
+    char *dst,
+    const char *src,
+    size_t dstsize
+);

+

Parameters

+
+
dst
+
Destination string
+
src
+
Source string
+
dstsize
+
Size of destination string
+
+

Return Value

+

Destination string

+

cgiFreeSearch

+

Free a compiled search context.

+

+void cgiFreeSearch (
+    void *search
+);

+

Parameters

+
+
search
+
Search context
+
+

cgiGetArray

+

Get an element from a form array.

+

+const char *cgiGetArray (
+    const char *name,
+    int element
+);

+

Parameters

+
+
name
+
Name of array variable
+
element
+
Element number (0 to N)
+
+

Return Value

+

Element value or NULL

+

cgiGetAttributes

+

Get the list of attributes that are needed +by the template file.

+

+void cgiGetAttributes (
+    ipp_t *request,
+    const char *tmpl
+);

+

Parameters

+
+
request
+
IPP request
+
tmpl
+
Base filename
+
+

cgiGetCookie

+

Get a cookie value.

+

+const char *cgiGetCookie (
+    const char *name
+);

+

Parameters

+
+
name
+
Name of cookie
+
+

Return Value

+

Value or NULL

+

cgiGetFile

+

Get the file (if any) that was submitted in the form.

+

+const cgi_file_t *cgiGetFile (void);

+

Return Value

+

Attached file or NULL

+

cgiGetIPPObjects

+

Get the objects in an IPP response.

+

+cups_array_t *cgiGetIPPObjects (
+    ipp_t *response,
+    void *search
+);

+

Parameters

+
+
response
+
IPP response
+
search
+
Search filter
+
+

Return Value

+

Array of objects

+

cgiGetSize

+

Get the size of a form array value.

+

+int cgiGetSize (
+    const char *name
+);

+

Parameters

+
+
name
+
Name of variable
+
+

Return Value

+

Number of elements

+

cgiGetTemplateDir

+

Get the templates directory...

+

+char *cgiGetTemplateDir (void);

+

Return Value

+

Template directory

+

cgiGetVariable

+

Get a CGI variable from the database.

+

+const char *cgiGetVariable (
+    const char *name
+);

+

Parameters

+
+
name
+
Name of variable
+
+

Return Value

+

Value of variable

+

Discussion

+

Returns NULL if the variable doesn't exist. If the variable is an +array of values, returns the last element.

+

cgiInitialize

+

Initialize the CGI variable "database".

+

+int cgiInitialize (void);

+

Return Value

+

Non-zero if there was form data

+

cgiIsPOST

+

Determine whether this page was POSTed.

+

+int cgiIsPOST (void);

+

Return Value

+

1 if POST, 0 if GET

+

cgiMoveJobs

+

Move one or more jobs.

+

+void cgiMoveJobs (
+    http_t *http,
+    const char *dest,
+    int job_id
+);

+

Parameters

+
+
http
+
Connection to server
+
dest
+
Destination or NULL
+
job_id
+
Job ID or 0 for all
+
+

Discussion

+

At least one of dest or job_id must be non-zero/NULL.

+

cgiPrintCommand

+

Print a CUPS command job.

+

+void cgiPrintCommand (
+    http_t *http,
+    const char *dest,
+    const char *command,
+    const char *title
+);

+

Parameters

+
+
http
+
Connection to server
+
dest
+
Destination printer
+
command
+
Command to send
+
title
+
Page/job title
+
+

cgiPrintTestPage

+

Print a test page.

+

+void cgiPrintTestPage (
+    http_t *http,
+    const char *dest
+);

+

Parameters

+
+
http
+
Connection to server
+
dest
+
Destination printer/class
+
+

cgiRewriteURL

+

Rewrite a printer URI into a web browser URL...

+

+char *cgiRewriteURL (
+    const char *uri,
+    char *url,
+    int urlsize,
+    const char *newresource
+);

+

Parameters

+
+
uri
+
Current URI
+
url
+
New URL
+
urlsize
+
Size of URL buffer
+
newresource
+
Replacement resource
+
+

Return Value

+

New URL

+

cgiSetArray

+

Set array element N to the specified string.

+

+void cgiSetArray (
+    const char *name,
+    int element,
+    const char *value
+);

+

Parameters

+
+
name
+
Name of variable
+
element
+
Element number (0 to N)
+
value
+
Value of variable
+
+

Discussion

+

If the variable array is smaller than (element + 1), the intervening +elements are set to NULL.

+

cgiSetCookie

+

Set a cookie value.

+

+void cgiSetCookie (
+    const char *name,
+    const char *value,
+    const char *path,
+    const char *domain,
+    time_t expires,
+    int secure
+);

+

Parameters

+
+
name
+
Name
+
value
+
Value
+
path
+
Path (typically "/")
+
domain
+
Domain name
+
expires
+
Expiration date (0 for session)
+
secure
+
Require SSL
+
+

cgiSetIPPObjectVars

+

Set CGI variables from an IPP object.

+

+ipp_attribute_t *cgiSetIPPObjectVars (
+    ipp_attribute_t *obj,
+    const char *prefix,
+    int element
+);

+

Parameters

+
+
obj
+
Response data to be copied...
+
prefix
+
Prefix for name or NULL
+
element
+
Parent element number
+
+

Return Value

+

Next object

+

cgiSetIPPVars

+

Set CGI variables from an IPP response.

+

+int cgiSetIPPVars (
+    ipp_t *response,
+    const char *filter_name,
+    const char *filter_value,
+    const char *prefix,
+    int parent_el
+);

+

Parameters

+
+
response
+
Response data to be copied...
+
filter_name
+
Filter name
+
filter_value
+
Filter value
+
prefix
+
Prefix for name or NULL
+
parent_el
+
Parent element number
+
+

Return Value

+

Maximum number of elements

+

cgiSetServerVersion

+

Set the server name and CUPS version...

+

+void cgiSetServerVersion (void);

+

cgiSetSize

+

Set the array size.

+

+void cgiSetSize (
+    const char *name,
+    int size
+);

+

Parameters

+
+
name
+
Name of variable
+
size
+
Number of elements (0 to N)
+
+

cgiSetVariable

+

Set a CGI variable in the database.

+

+void cgiSetVariable (
+    const char *name,
+    const char *value
+);

+

Parameters

+
+
name
+
Name of variable
+
value
+
Value of variable
+
+

Discussion

+

If the variable is an array, this truncates the array to a single element.

+

cgiShowIPPError

+

Show the last IPP error message.

+

+void cgiShowIPPError (
+    const char *message
+);

+

Parameters

+
+
message
+
Contextual message
+
+

Discussion

+

The caller must still call cgiStartHTML() and cgiEndHTML().

+

cgiShowJobs

+

Show print jobs.

+

+void cgiShowJobs (
+    http_t *http,
+    const char *dest
+);

+

Parameters

+
+
http
+
Connection to server
+
dest
+
Destination name or NULL
+
+

cgiStartHTML

+

Start a HTML page.

+

+void cgiStartHTML (
+    const char *title
+);

+

Parameters

+
+
title
+
Title of page
+
+

cgiStartMultipart

+

Start a multipart delivery of a web page.

+

+void cgiStartMultipart (void);

+

cgiSupportsMultipart

+

Does the browser support multi-part documents?

+

+int cgiSupportsMultipart (void);

+

Return Value

+

1 if multi-part supported, 0 otherwise

+

cgiText

+

Return localized text.

+

+const char *cgiText (
+    const char *message
+);

+

Parameters

+
+
message
+
Message
+
+

Return Value

+

Localized message

+

helpDeleteIndex

+

Delete an index, freeing all memory used.

+

+void helpDeleteIndex (
+    help_index_t *hi
+);

+

Parameters

+
+
hi
+
Help index
+
+

helpFindNode

+

Find a node in an index.

+

+help_node_t *helpFindNode (
+    help_index_t *hi,
+    const char *filename,
+    const char *anchor
+);

+

Parameters

+
+
hi
+
Index
+
filename
+
Filename
+
anchor
+
Anchor
+
+

Return Value

+

Node pointer or NULL

+

helpLoadIndex

+

Load a help index from disk.

+

+help_index_t *helpLoadIndex (
+    const char *hifile,
+    const char *directory
+);

+

Parameters

+
+
hifile
+
Index filename
+
directory
+
Directory that is indexed
+
+

Return Value

+

Index pointer or NULL

+

helpSaveIndex

+

Save a help index to disk.

+

+int helpSaveIndex (
+    help_index_t *hi,
+    const char *hifile
+);

+

Parameters

+
+
hi
+
Index
+
hifile
+
Index filename
+
+

Return Value

+

0 on success, -1 on error

+

helpSearchIndex

+

Search an index.

+

+help_index_t *helpSearchIndex (
+    help_index_t *hi,
+    const char *query,
+    const char *section,
+    const char *filename
+);

+

Parameters

+
+
hi
+
Index
+
query
+
Query string
+
section
+
Limit search to this section
+
filename
+
Limit search to this file
+
+

Return Value

+

Search index

+

Data Types

+

cgi_file_t

+

Uploaded file data

+

+typedef struct cgi_file_s cgi_file_t; +

+

help_index_t

+

Help index structure

+

+typedef struct help_index_s help_index_t; +

+

help_node_t

+

Help node structure...

+

+typedef struct help_node_s help_node_t; +

+

help_word_t

+

Help word structure...

+

+typedef struct help_word_s help_word_t; +

+

Structures

+

cgi_file_s

+

Uploaded file data

+

struct cgi_file_s {
+    size_t filesize;
+    char tempfile[1024], *name, *filename, *mimetype;
+};

+

Members

+
+
filesize
+
Size of uploaded file
+
mimetype
+
MIME media type
+
+

help_index_s

+

Help index structure

+

struct help_index_s {
+    cups_array_t *nodes;
+    int search;
+    cups_array_t *sorted;
+};

+

Members

+
+
nodes
+
Nodes sorted by filename
+
search
+
1 = search index, 0 = normal
+
sorted
+
Nodes sorted by score + text
+
+

help_node_s

+

Help node structure...

+

struct help_node_s {
+    char *anchor;
+    char *filename;
+    size_t length;
+    time_t mtime;
+    off_t offset;
+    int score;
+    char *section;
+    char *text;
+    cups_array_t *words;
+};

+

Members

+
+
anchor
+
Anchor name (NULL if none)
+
filename
+
Filename, relative to help dir
+
length
+
Length in bytes
+
mtime
+
Last modification time
+
offset
+
Offset in file
+
score
+
Search score
+
section
+
Section name (NULL if none)
+
text
+
Text in anchor
+
words
+
Words after this node
+
+

help_word_s

+

Help word structure...

+

struct help_word_s {
+    int count;
+    char *text;
+};

+

Members

+
+
count
+
Number of occurrences
+
text
+
Word text
+
+
+ + diff --git a/doc/help/api-cups.html b/doc/help/api-cups.html new file mode 100644 index 0000000000..2dd408308e --- /dev/null +++ b/doc/help/api-cups.html @@ -0,0 +1,2521 @@ + + + + + CUPS API + + + + + + +
+ + +

CUPS API

+ +
+ + + + + + + + + + + + + + + + +
Headercups/cups.h
Library-lcups
See AlsoProgramming: Introduction to CUPS Programming
+ Programming: Array API
+ Programming: File and Directory APIs
+ Programming: Filter and Backend Programming
+ Programming: HTTP and IPP APIs
+ Programming: PPD API
+ Programming: Raster API
+

Contents

+ + + +

Overview

+ +

The CUPS API provides the convenience functions needed to support +applications, filters, printer drivers, and backends that need to interface +with the CUPS scheduler.

+ +

Clients and Servers

+ +

CUPS is based on the Internet Printing Protocol ("IPP"), which allows +clients (applications) to communicate with a server (the scheduler) to get a +list of printers, send print jobs, and so forth. You identify which server +you want to communicate with using a pointer to the opaque structure +http_t. All of the examples in this document use the +CUPS_HTTP_DEFAULT constant, referring to the default connection +to the scheduler. The HTTP and IPP +APIs document provides more information on server connections.

+ +

Printers and Classes

+ +

Printers and classes (collections of printers) are accessed through +the cups_dest_t structure which +includes the name (name), instance (instance - +a way of selecting certain saved options/settings), and the options and +attributes associated with that destination (num_options and +options). Destinations are created using the +cupsGetDests function and freed +using the cupsFreeDests function. +The cupsGetDest function finds a +specific destination for printing:

+ +
+#include <cups/cups.h>
+
+cups_dest_t *dests;
+int num_dests = cupsGetDests(&dests);
+cups_dest_t *dest = cupsGetDest("name", NULL, num_dests, dests);
+
+/* do something with dest */
+
+cupsFreeDests(num_dests, dests);
+
+ +

Passing NULL to +cupsGetDest for the destination name +will return the default destination. Similarly, passing a NULL +instance will return the default instance for that destination.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 1: Printer Attributes
Attribute NameDescription
"auth-info-required"The type of authentication required for printing to this + destination: "none", "username,password", "domain,username,password", + or "negotiate" (Kerberos)
"printer-info"The human-readable description of the destination such as "My + Laser Printer".
"printer-is-accepting-jobs""true" if the destination is accepting new jobs, "false" if + not.
"printer-is-shared""true" if the destination is being shared with other computers, + "false" if not.
"printer-location"The human-readable location of the destination such as "Lab 4".
"printer-make-and-model"The human-readable make and model of the destination such as "HP + LaserJet 4000 Series".
"printer-state""3" if the destination is idle, "4" if the destination is printing + a job, and "5" if the destination is stopped.
"printer-state-change-time"The UNIX time when the destination entered the current state.
"printer-state-reasons"Additional comma-delimited state keywords for the destination + such as "media-tray-empty-error" and "toner-low-warning".
"printer-type"The cups_printer_t + value associated with the destination.
+ +

Options

+ +

Options are stored in arrays of +cups_option_t structures. Each +option has a name (name) and value (value) +associated with it. The cups_dest_t +num_options and options members contain the +default options for a particular destination, along with several informational +attributes about the destination as shown in Table 1. +The cupsGetOption function gets +the value for the named option. For example, the following code lists the +available destinations and their human-readable descriptions:

+ +
+#include <cups/cups.h>
+
+cups_dest_t *dests;
+int num_dests = cupsGetDests(&dests);
+cups_dest_t *dest;
+int i;
+const char *value;
+
+for (i = num_dests, dest = dests; i > 0; i --, dest ++)
+  if (dest->instance == NULL)
+  {
+    value = cupsGetOption("printer-info", dest->num_options, dest->options);
+    printf("%s (%s)\n", dest->name, value ? value : "no description");
+  }
+
+cupsFreeDests(num_dests, dests);
+
+ +

You can create your own option arrays using the +cupsAddOption function, which +adds a single named option to an array:

+ +
+#include <cups/cups.h>
+
+int num_options = 0;
+cups_option_t *options = NULL;
+
+/* The returned num_options value is updated as needed */
+num_options = cupsAddOption("first", "value", num_options, &options);
+
+/* This adds a second option value */
+num_options = cupsAddOption("second", "value", num_options, &options);
+
+/* This replaces the first option we added */
+num_options = cupsAddOption("first", "new value", num_options, &options);
+
+ +

Use a for loop to copy the options from a destination:

+ +
+#include <cups/cups.h>
+
+int i;
+int num_options = 0;
+cups_option_t *options = NULL;
+cups_dest_t *dest;
+
+for (i = 0; i < dest->num_options; i ++)
+  num_options = cupsAddOption(dest->options[i].name, dest->options[i].value,
+                              num_options, &options);
+
+ +

Use the cupsFreeOptions +function to free the options array when you are done using it:

+ +
+cupsFreeOptions(num_options, options);
+
+ +

Print Jobs

+ +

Print jobs are identified by a locally-unique job ID number from 1 to +231-1 and have options and one or more files for printing to a +single destination. The cupsPrintFile +function creates a new job with one file. The following code prints the CUPS +test page file:

+ +
+#include <cups/cups.h>
+
+cups_dest_t *dest;
+int num_options;
+cups_option_t *options;
+int job_id;
+
+/* Print a single file */
+job_id = cupsPrintFile(dest->name, "/usr/share/cups/data/testprint.ps",
+                        "Test Print", num_options, options);
+
+ +

The cupsPrintFiles function +creates a job with multiple files. The files are provided in a +char * array:

+ +
+#include <cups/cups.h>
+
+cups_dest_t *dest;
+int num_options;
+cups_option_t *options;
+int job_id;
+char *files[3] = { "file1.pdf", "file2.pdf", "file3.pdf" };
+
+/* Print three files */
+job_id = cupsPrintFiles(dest->name, 3, files, "Test Print", num_options, options);
+
+ +

Finally, the cupsCreateJob +function creates a new job with no files in it. Files are added using the +cupsStartDocument, +cupsWriteRequestData, +and cupsFinishDocument functions. +The following example creates a job with 10 text files for printing:

+ +
+#include <cups/cups.h>
+
+cups_dest_t *dest;
+int num_options;
+cups_option_t *options;
+int job_id;
+int i;
+char buffer[1024];
+
+/* Create the job */
+job_id = cupsCreateJob(CUPS_HTTP_DEFAULT, dest->name, "10 Text Files",
+                       num_options, options);
+
+/* If the job is created, add 10 files */
+if (job_id > 0)
+{
+  for (i = 1; i <= 10; i ++)
+  {
+    snprintf(buffer, sizeof(buffer), "file%d.txt", i);
+
+    cupsStartDocument(CUPS_HTTP_DEFAULT, dest->name, job_id, buffer,
+                      CUPS_FORMAT_TEXT, i == 10);
+
+    snprintf(buffer, sizeof(buffer),
+             "File %d\n"
+             "\n"
+             "One fish,\n"
+             "Two fish,\n
+             "Red fish,\n
+             "Blue fish\n", i);
+
+    /* cupsWriteRequestData can be called as many times as needed */
+    cupsWriteRequestData(CUPS_HTTP_DEFAULT, buffer, strlen(buffer));
+
+    cupsFinishDocument(CUPS_HTTP_DEFAULT, dest->name);
+  }
+}
+
+ +

Once you have created a job, you can monitor its status using the +cupsGetJobs function, which returns +an array of cups_job_t structures. +Each contains the job ID (id), destination name +(dest), title (title), and other information +associated with the job. The job array is freed using the +cupsFreeJobs function. The following +example monitors a specific job ID, showing the current job state once every +5 seconds until the job is completed:

+ +
+#include <cups/cups.h>
+
+cups_dest_t *dest;
+int job_id;
+int num_jobs;
+cups_job_t *jobs;
+int i;
+ipp_jstate_t job_state = IPP_JOB_PENDING;
+ 
+while (job_state < IPP_JOB_STOPPED)
+{
+  /* Get my jobs (1) with any state (-1) */
+  num_jobs = cupsGetJobs(&jobs, dest->name, 1, -1);
+
+  /* Loop to find my job */
+  job_state = IPP_JOB_COMPLETED;
+
+  for (i = 0; i < num_jobs; i ++)
+    if (jobs[i].id == job_id)
+    {
+      job_state = jobs[i].state;
+      break;
+    }
+
+  /* Free the job array */
+  cupsFreeJobs(num_jobs, jobs);
+
+  /* Show the current state */
+  switch (job_state)
+  {
+    case IPP_JOB_PENDING :
+        printf("Job %d is pending.\n", job_id);
+        break;
+    case IPP_JOB_HELD :
+        printf("Job %d is held.\n", job_id);
+        break;
+    case IPP_JOB_PROCESSING :
+        printf("Job %d is processing.\n", job_id);
+        break;
+    case IPP_JOB_STOPPED :
+        printf("Job %d is stopped.\n", job_id);
+        break;
+    case IPP_JOB_CANCELED :
+        printf("Job %d is canceled.\n", job_id);
+        break;
+    case IPP_JOB_ABORTED :
+        printf("Job %d is aborted.\n", job_id);
+        break;
+    case IPP_JOB_COMPLETED :
+        printf("Job %d is completed.\n", job_id);
+        break;
+  }
+
+  /* Sleep if the job is not finished */
+  if (job_state < IPP_JOB_STOPPED)
+    sleep(5);
+}
+
+ +

To cancel a job, use the +cupsCancelJob function with the +job ID:

+ +
+#include <cups/cups.h>
+
+cups_dest_t *dest;
+int job_id;
+
+cupsCancelJob(dest->name, job_id);
+
+ +

Error Handling

+ +

If any of the CUPS API printing functions returns an error, the reason for +that error can be found by calling the +cupsLastError and +cupsLastErrorString functions. +cupsLastError returns the last IPP +error code +(ipp_status_t) +that was encountered, while +cupsLastErrorString returns +a (localized) human-readable string that can be shown to the user. For example, +if any of the job creation functions returns a job ID of 0, you can use +cupsLastErrorString to show +the reason why the job could not be created:

+ +
+#include <cups/cups.h>
+
+int job_id;
+
+if (job_id == 0)
+  puts(cupsLastErrorString());
+
+ +

Passwords and Authentication

+ +

CUPS supports authentication of any request, including submission of print +jobs. The default mechanism for getting the username and password is to use the +login user and a password from the console.

+ +

To support other types of applications, in particular Graphical User +Interfaces ("GUIs"), the CUPS API provides functions to set the default +username and to register a callback function that returns a password string.

+ +

The cupsSetPasswordCB +function is used to set a password callback in your program. Only one +function can be used at any time.

+ +

The cupsSetUser function sets the +current username for authentication. This function can be called by your +password callback function to change the current username as needed.

+ +

The following example shows a simple password callback that gets a +username and password from the user:

+ +
+#include <cups/cups.h>
+
+const char *
+my_password_cb(const char *prompt)
+{
+  char	user[65];
+
+
+  puts(prompt);
+
+  /* Get a username from the user */
+  printf("Username: ");
+  if (fgets(user, sizeof(user), stdin) == NULL)
+    return (NULL);
+
+  /* Strip the newline from the string and set the user */
+  user[strlen(user) - 1] = '\0';
+
+  cupsSetUser(user);
+
+  /* Use getpass() to ask for the password... */
+  return (getpass("Password: "));
+}
+
+cupsSetPasswordCB(my_password_cb);
+
+ +

Similarly, a GUI could display the prompt string in a window with input +fields for the username and password. The username should default to the +string returned by the cupsUser +function.

+

Functions

+

appleGetPaperSize

+

Get the default paper size.

+

+char *appleGetPaperSize (
+    char *name,
+    int namesize
+);

+

Parameters

+
+
name
+
Paper size name buffer
+
namesize
+
Size of buffer
+
+

Return Value

+

Default paper size

+

cupsAddDest

+

Add a destination to the list of destinations.

+

+int cupsAddDest (
+    const char *name,
+    const char *instance,
+    int num_dests,
+    cups_dest_t **dests
+);

+

Parameters

+
+
name
+
Destination name
+
instance
+
Instance name or NULL for none/primary
+
num_dests
+
Number of destinations
+
dests
+
Destinations
+
+

Return Value

+

New number of destinations

+

Discussion

+

This function cannot be used to add a new class or printer queue, +it only adds a new container of saved options for the named +destination or instance.
+
+If the named destination already exists, the destination list is +returned unchanged. Adding a new instance of a destination creates +a copy of that destination's options.
+
+Use the cupsSaveDests function to save the updated list of +destinations to the user's lpoptions file.

+

cupsAddOption

+

Add an option to an option array.

+

+int cupsAddOption (
+    const char *name,
+    const char *value,
+    int num_options,
+    cups_option_t **options
+);

+

Parameters

+
+
name
+
Name of option
+
value
+
Value of option
+
num_options
+
Number of options
+
options
+
Pointer to options
+
+

Return Value

+

Number of options

+

Discussion

+

New option arrays can be initialized simply by passing 0 for the +"num_options" parameter.

+

 CUPS 1.2/Mac OS X 10.5 cupsAdminCreateWindowsPPD

+

Create the Windows PPD file for a printer.

+

+char *cupsAdminCreateWindowsPPD (
+    http_t *http,
+    const char *dest,
+    char *buffer,
+    int bufsize
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
dest
+
Printer or class
+
buffer
+
Filename buffer
+
bufsize
+
Size of filename buffer
+
+

Return Value

+

PPD file or NULL

+

 CUPS 1.2/Mac OS X 10.5 cupsAdminExportSamba

+

Export a printer to Samba.

+

+int cupsAdminExportSamba (
+    const char *dest,
+    const char *ppd,
+    const char *samba_server,
+    const char *samba_user,
+    const char *samba_password,
+    FILE *logfile
+);

+

Parameters

+
+
dest
+
Destination to export
+
ppd
+
PPD file
+
samba_server
+
Samba server
+
samba_user
+
Samba username
+
samba_password
+
Samba password
+
logfile
+
Log file, if any
+
+

Return Value

+

1 on success, 0 on failure

+

 CUPS 1.3/Mac OS X 10.5 cupsAdminGetServerSettings

+

Get settings from the server.

+

+int cupsAdminGetServerSettings (
+    http_t *http,
+    int *num_settings,
+    cups_option_t **settings
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
num_settings
+
Number of settings
+
settings
+
Settings
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

The returned settings should be freed with cupsFreeOptions() when +you are done with them. + +

+

 CUPS 1.3/Mac OS X 10.5 cupsAdminSetServerSettings

+

Set settings on the server.

+

+int cupsAdminSetServerSettings (
+    http_t *http,
+    int num_settings,
+    cups_option_t *settings
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
num_settings
+
Number of settings
+
settings
+
Settings
+
+

Return Value

+

1 on success, 0 on failure

+

cupsCancelJob

+

Cancel a print job on the default server.

+

+int cupsCancelJob (
+    const char *name,
+    int job_id
+);

+

Parameters

+
+
name
+
Name of printer or class
+
job_id
+
Job ID, CUPS_JOBID_CURRENT for the current job, or CUPS_JOBID_ALL for all jobs
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

Pass CUPS_JOBID_ALL to cancel all jobs or CUPS_JOBID_CURRENT +to cancel the current job on the named destination.
+
+Use the cupsLastError and cupsLastErrorString functions to get +the cause of any failure.

+

 CUPS 1.4/Mac OS X 10.6 cupsCancelJob2

+

Cancel or purge a print job.

+

+ipp_status_t cupsCancelJob2 (
+    http_t *http,
+    const char *name,
+    int job_id,
+    int purge
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
name
+
Name of printer or class
+
job_id
+
Job ID, CUPS_JOBID_CURRENT for the current job, or CUPS_JOBID_ALL for all jobs
+
purge
+
1 to purge, 0 to cancel
+
+

Return Value

+

IPP status

+

Discussion

+

Canceled jobs remain in the job history while purged jobs are removed +from the job history.
+
+Pass CUPS_JOBID_ALL to cancel all jobs or CUPS_JOBID_CURRENT +to cancel the current job on the named destination.
+
+Use the cupsLastError and cupsLastErrorString functions to get +the cause of any failure. + +

+

 CUPS 1.4/Mac OS X 10.6 cupsCreateJob

+

Create an empty job for streaming.

+

+int cupsCreateJob (
+    http_t *http,
+    const char *name,
+    const char *title,
+    int num_options,
+    cups_option_t *options
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
name
+
Destination name
+
title
+
Title of job
+
num_options
+
Number of options
+
options
+
Options
+
+

Return Value

+

Job ID or 0 on error

+

Discussion

+

Use this function when you want to stream print data using the +cupsStartDocument, cupsWriteRequestData, and +cupsFinishDocument functions. If you have one or more files to +print, use the cupsPrintFile2 or cupsPrintFiles2 function +instead. + +

+

cupsEncryption

+

Get the current encryption settings.

+

+http_encryption_t cupsEncryption (void);

+

Return Value

+

Encryption settings

+

Discussion

+

The default encryption setting comes from the CUPS_ENCRYPTION +environment variable, then the ~/.cups/client.conf file, and finally the +/etc/cups/client.conf file. If not set, the default is +HTTP_ENCRYPT_IF_REQUESTED.
+
+Note: The current encryption setting is tracked separately for each thread +in a program. Multi-threaded programs that override the setting via the +cupsSetEncryption function need to do so in each thread for the same +setting to be used.

+

 CUPS 1.4/Mac OS X 10.6 cupsFinishDocument

+

Finish sending a document.

+

+ipp_status_t cupsFinishDocument (
+    http_t *http,
+    const char *name
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
name
+
Destination name
+
+

Return Value

+

Status of document submission

+

Discussion

+

The document must have been started using cupsStartDocument. + +

+

cupsFreeDests

+

Free the memory used by the list of destinations.

+

+void cupsFreeDests (
+    int num_dests,
+    cups_dest_t *dests
+);

+

Parameters

+
+
num_dests
+
Number of destinations
+
dests
+
Destinations
+
+

cupsFreeJobs

+

Free memory used by job data.

+

+void cupsFreeJobs (
+    int num_jobs,
+    cups_job_t *jobs
+);

+

Parameters

+
+
num_jobs
+
Number of jobs
+
jobs
+
Jobs
+
+

cupsFreeOptions

+

Free all memory used by options.

+

+void cupsFreeOptions (
+    int num_options,
+    cups_option_t *options
+);

+

Parameters

+
+
num_options
+
Number of options
+
options
+
Pointer to options
+
+

 DEPRECATED cupsGetClasses

+

Get a list of printer classes from the default server.

+

+int cupsGetClasses (
+    char ***classes
+);

+

Parameters

+
+
classes
+
Classes
+
+

Return Value

+

Number of classes

+

Discussion

+

This function is deprecated - use cupsGetDests instead. + +

+

cupsGetDefault

+

Get the default printer or class for the default server.

+

+const char *cupsGetDefault (void);

+

Return Value

+

Default printer or NULL

+

Discussion

+

This function returns the default printer or class as defined by +the LPDEST or PRINTER environment variables. If these environment +variables are not set, the server default destination is returned. +Applications should use the cupsGetDests and cupsGetDest +functions to get the user-defined default printer, as this function does +not support the lpoptions-defined default printer.

+

 CUPS 1.1.21/Mac OS X 10.4 cupsGetDefault2

+

Get the default printer or class for the specified server.

+

+const char *cupsGetDefault2 (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
+

Return Value

+

Default printer or NULL

+

Discussion

+

This function returns the default printer or class as defined by +the LPDEST or PRINTER environment variables. If these environment +variables are not set, the server default destination is returned. +Applications should use the cupsGetDests and cupsGetDest +functions to get the user-defined default printer, as this function does +not support the lpoptions-defined default printer. + +

+

cupsGetDest

+

Get the named destination from the list.

+

+cups_dest_t *cupsGetDest (
+    const char *name,
+    const char *instance,
+    int num_dests,
+    cups_dest_t *dests
+);

+

Parameters

+
+
name
+
Destination name or NULL for the default destination
+
instance
+
Instance name or NULL
+
num_dests
+
Number of destinations
+
dests
+
Destinations
+
+

Return Value

+

Destination pointer or NULL

+

Discussion

+

Use the cupsGetDests or cupsGetDests2 functions to get a +list of supported destinations for the current user.

+

cupsGetDests

+

Get the list of destinations from the default server.

+

+int cupsGetDests (
+    cups_dest_t **dests
+);

+

Parameters

+
+
dests
+
Destinations
+
+

Return Value

+

Number of destinations

+

Discussion

+

Starting with CUPS 1.2, the returned list of destinations include the +printer-info, printer-is-accepting-jobs, printer-is-shared, +printer-make-and-model, printer-state, printer-state-change-time, +printer-state-reasons, and printer-type attributes as options. CUPS 1.4 +adds the marker-change-time, marker-colors, marker-high-levels, +marker-levels, marker-low-levels, marker-message, marker-names, +marker-types, and printer-commands attributes as well.
+
+Use the cupsFreeDests function to free the destination list and +the cupsGetDest function to find a particular destination.

+

 CUPS 1.1.21/Mac OS X 10.4 cupsGetDests2

+

Get the list of destinations from the specified server.

+

+int cupsGetDests2 (
+    http_t *http,
+    cups_dest_t **dests
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
dests
+
Destinations
+
+

Return Value

+

Number of destinations

+

Discussion

+

Starting with CUPS 1.2, the returned list of destinations include the +printer-info, printer-is-accepting-jobs, printer-is-shared, +printer-make-and-model, printer-state, printer-state-change-time, +printer-state-reasons, and printer-type attributes as options. CUPS 1.4 +adds the marker-change-time, marker-colors, marker-high-levels, +marker-levels, marker-low-levels, marker-message, marker-names, +marker-types, and printer-commands attributes as well.
+
+Use the cupsFreeDests function to free the destination list and +the cupsGetDest function to find a particular destination. + +

+

cupsGetJobs

+

Get the jobs from the default server.

+

+int cupsGetJobs (
+    cups_job_t **jobs,
+    const char *name,
+    int myjobs,
+    int whichjobs
+);

+

Parameters

+
+
jobs
+
Job data
+
name
+
NULL = all destinations, otherwise show jobs for named destination
+
myjobs
+
0 = all users, 1 = mine
+
whichjobs
+
CUPS_WHICHJOBS_ALL, CUPS_WHICHJOBS_ACTIVE, or CUPS_WHICHJOBS_COMPLETED
+
+

Return Value

+

Number of jobs

+

Discussion

+

A "whichjobs" value of CUPS_WHICHJOBS_ALL returns all jobs regardless +of state, while CUPS_WHICHJOBS_ACTIVE returns jobs that are +pending, processing, or held and CUPS_WHICHJOBS_COMPLETED returns +jobs that are stopped, canceled, aborted, or completed.

+

 CUPS 1.1.21/Mac OS X 10.4 cupsGetJobs2

+

Get the jobs from the specified server.

+

+int cupsGetJobs2 (
+    http_t *http,
+    cups_job_t **jobs,
+    const char *name,
+    int myjobs,
+    int whichjobs
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
jobs
+
Job data
+
name
+
NULL = all destinations, otherwise show jobs for named destination
+
myjobs
+
0 = all users, 1 = mine
+
whichjobs
+
CUPS_WHICHJOBS_ALL, CUPS_WHICHJOBS_ACTIVE, or CUPS_WHICHJOBS_COMPLETED
+
+

Return Value

+

Number of jobs

+

Discussion

+

A "whichjobs" value of CUPS_WHICHJOBS_ALL returns all jobs regardless +of state, while CUPS_WHICHJOBS_ACTIVE returns jobs that are +pending, processing, or held and CUPS_WHICHJOBS_COMPLETED returns +jobs that are stopped, canceled, aborted, or completed. + +

+

 CUPS 1.4/Mac OS X 10.6 cupsGetNamedDest

+

Get options for the named destination.

+

+cups_dest_t *cupsGetNamedDest (
+    http_t *http,
+    const char *name,
+    const char *instance
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
name
+
Destination name or NULL for the default destination
+
instance
+
Instance name or NULL
+
+

Return Value

+

Destination or NULL

+

Discussion

+

This function is optimized for retrieving a single destination and should +be used instead of cupsGetDests and cupsGetDest when you either +know the name of the destination or want to print to the default destination. +If NULL is returned, the destination does not exist or there is no +default destination.
+
+If "http" is CUPS_HTTP_DEFAULT, the connection to the default print +server will be used.
+
+If "name" is NULL, the default printer for the current user will be +returned.
+
+The returned destination must be freed using cupsFreeDests with a +"num_dests" value of 1. + +

+

cupsGetOption

+

Get an option value.

+

+const char *cupsGetOption (
+    const char *name,
+    int num_options,
+    cups_option_t *options
+);

+

Parameters

+
+
name
+
Name of option
+
num_options
+
Number of options
+
options
+
Options
+
+

Return Value

+

Option value or NULL

+

cupsGetPPD

+

Get the PPD file for a printer on the default server.

+

+const char *cupsGetPPD (
+    const char *name
+);

+

Parameters

+
+
name
+
Destination name
+
+

Return Value

+

Filename for PPD file

+

Discussion

+

For classes, cupsGetPPD returns the PPD file for the first printer +in the class.
+
+The returned filename is stored in a static buffer and is overwritten with +each call to cupsGetPPD or cupsGetPPD2. The caller "owns" the +file that is created and must unlink the returned filename.

+

 CUPS 1.1.21/Mac OS X 10.4 cupsGetPPD2

+

Get the PPD file for a printer from the specified server.

+

+const char *cupsGetPPD2 (
+    http_t *http,
+    const char *name
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
name
+
Destination name
+
+

Return Value

+

Filename for PPD file

+

Discussion

+

For classes, cupsGetPPD2 returns the PPD file for the first printer +in the class.
+
+The returned filename is stored in a static buffer and is overwritten with +each call to cupsGetPPD or cupsGetPPD2. The caller "owns" the +file that is created and must unlink the returned filename. + +

+

 CUPS 1.4/Mac OS X 10.6 cupsGetPPD3

+

Get the PPD file for a printer on the specified +server if it has changed.

+

+http_status_t cupsGetPPD3 (
+    http_t *http,
+    const char *name,
+    time_t *modtime,
+    char *buffer,
+    size_t bufsize
+);

+

Parameters

+
+
http
+
HTTP connection or CUPS_HTTP_DEFAULT
+
name
+
Destination name
+
modtime
+
Modification time
+
buffer
+
Filename buffer
+
bufsize
+
Size of filename buffer
+
+

Return Value

+

HTTP status

+

Discussion

+

The "modtime" parameter contains the modification time of any +locally-cached content and is updated with the time from the PPD file on +the server.
+
+The "buffer" parameter contains the local PPD filename. If it contains +the empty string, a new temporary file is created, otherwise the existing +file will be overwritten as needed. The caller "owns" the file that is +created and must unlink the returned filename.
+
+On success, HTTP_OK is returned for a new PPD file and +HTTP_NOT_MODIFIED if the existing PPD file is up-to-date. Any other +status is an error.
+
+For classes, cupsGetPPD3 returns the PPD file for the first printer +in the class. + +

+

cupsGetPassword

+

Get a password from the user.

+

+const char *cupsGetPassword (
+    const char *prompt
+);

+

Parameters

+
+
prompt
+
Prompt string
+
+

Return Value

+

Password

+

Discussion

+

Uses the current password callback function. Returns NULL if the +user does not provide a password.
+
+Note: The current password callback function is tracked separately for each +thread in a program. Multi-threaded programs that override the setting via +the cupsSetPasswordCB or cupsSetPasswordCB2 functions need to +do so in each thread for the same function to be used.

+

 CUPS 1.4/Mac OS X 10.6 cupsGetPassword2

+

Get a password from the user using the advanced +password callback.

+

+const char *cupsGetPassword2 (
+    const char *prompt,
+    http_t *http,
+    const char *method,
+    const char *resource
+);

+

Parameters

+
+
prompt
+
Prompt string
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
method
+
Request method ("GET", "POST", "PUT")
+
resource
+
Resource path
+
+

Return Value

+

Password

+

Discussion

+

Uses the current password callback function. Returns NULL if the +user does not provide a password.
+
+Note: The current password callback function is tracked separately for each +thread in a program. Multi-threaded programs that override the setting via +the cupsSetPasswordCB or cupsSetPasswordCB2 functions need to +do so in each thread for the same function to be used. + +

+

 DEPRECATED cupsGetPrinters

+

Get a list of printers from the default server.

+

+int cupsGetPrinters (
+    char ***printers
+);

+

Parameters

+
+
printers
+
Printers
+
+

Return Value

+

Number of printers

+

Discussion

+

This function is deprecated - use cupsGetDests instead. + +

+

 CUPS 1.3/Mac OS X 10.5 cupsGetServerPPD

+

Get an available PPD file from the server.

+

+char *cupsGetServerPPD (
+    http_t *http,
+    const char *name
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
name
+
Name of PPD file ("ppd-name")
+
+

Return Value

+

Name of PPD file or NULL on error

+

Discussion

+

This function returns the named PPD file from the server. The +list of available PPDs is provided by the IPP CUPS_GET_PPDS +operation.
+
+You must remove (unlink) the PPD file when you are finished with +it. The PPD filename is stored in a static location that will be +overwritten on the next call to cupsGetPPD, cupsGetPPD2, +or cupsGetServerPPD. + +

+

cupsLangDefault

+

Return the default language.

+

+cups_lang_t *cupsLangDefault (void);

+

Return Value

+

Language data

+

cupsLangEncoding

+

Return the character encoding (us-ascii, etc.) +for the given language.

+

+const char *cupsLangEncoding (
+    cups_lang_t *lang
+);

+

Parameters

+
+
lang
+
Language data
+
+

Return Value

+

Character encoding

+

cupsLangFlush

+

Flush all language data out of the cache.

+

+void cupsLangFlush (void);

+

cupsLangFree

+

Free language data.

+

+void cupsLangFree (
+    cups_lang_t *lang
+);

+

Parameters

+
+
lang
+
Language to free
+
+

Discussion

+

This does not actually free anything; use cupsLangFlush for that.

+

cupsLangGet

+

Get a language.

+

+cups_lang_t *cupsLangGet (
+    const char *language
+);

+

Parameters

+
+
language
+
Language or locale
+
+

Return Value

+

Language data

+

 CUPS 1.2/Mac OS X 10.5 cupsNotifySubject

+

Return the subject for the given notification message.

+

+char *cupsNotifySubject (
+    cups_lang_t *lang,
+    ipp_t *event
+);

+

Parameters

+
+
lang
+
Language data
+
event
+
Event data
+
+

Return Value

+

Subject string or NULL

+

Discussion

+

The returned string must be freed by the caller using free. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsNotifyText

+

Return the text for the given notification message.

+

+char *cupsNotifyText (
+    cups_lang_t *lang,
+    ipp_t *event
+);

+

Parameters

+
+
lang
+
Language data
+
event
+
Event data
+
+

Return Value

+

Message text or NULL

+

Discussion

+

The returned string must be freed by the caller using free. + +

+

cupsParseOptions

+

Parse options from a command-line argument.

+

+int cupsParseOptions (
+    const char *arg,
+    int num_options,
+    cups_option_t **options
+);

+

Parameters

+
+
arg
+
Argument to parse
+
num_options
+
Number of options
+
options
+
Options found
+
+

Return Value

+

Number of options found

+

Discussion

+

This function converts space-delimited name/value pairs according +to the PAPI text option ABNF specification. Collection values +("name={a=... b=... c=...}") are stored with the curley brackets +intact - use cupsParseOptions on the value to extract the +collection attributes.

+

cupsPrintFile

+

Print a file to a printer or class on the default server.

+

+int cupsPrintFile (
+    const char *name,
+    const char *filename,
+    const char *title,
+    int num_options,
+    cups_option_t *options
+);

+

Parameters

+
+
name
+
Destination name
+
filename
+
File to print
+
title
+
Title of job
+
num_options
+
Number of options
+
options
+
Options
+
+

Return Value

+

Job ID or 0 on error

+

 CUPS 1.1.21/Mac OS X 10.4 cupsPrintFile2

+

Print a file to a printer or class on the specified +server.

+

+int cupsPrintFile2 (
+    http_t *http,
+    const char *name,
+    const char *filename,
+    const char *title,
+    int num_options,
+    cups_option_t *options
+);

+

Parameters

+
+
http
+
Connection to server
+
name
+
Destination name
+
filename
+
File to print
+
title
+
Title of job
+
num_options
+
Number of options
+
options
+
Options
+
+

Return Value

+

Job ID or 0 on error

+

cupsPrintFiles

+

Print one or more files to a printer or class on the +default server.

+

+int cupsPrintFiles (
+    const char *name,
+    int num_files,
+    const char **files,
+    const char *title,
+    int num_options,
+    cups_option_t *options
+);

+

Parameters

+
+
name
+
Destination name
+
num_files
+
Number of files
+
files
+
File(s) to print
+
title
+
Title of job
+
num_options
+
Number of options
+
options
+
Options
+
+

Return Value

+

Job ID or 0 on error

+

 CUPS 1.1.21/Mac OS X 10.4 cupsPrintFiles2

+

Print one or more files to a printer or class on the +specified server.

+

+int cupsPrintFiles2 (
+    http_t *http,
+    const char *name,
+    int num_files,
+    const char **files,
+    const char *title,
+    int num_options,
+    cups_option_t *options
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
name
+
Destination name
+
num_files
+
Number of files
+
files
+
File(s) to print
+
title
+
Title of job
+
num_options
+
Number of options
+
options
+
Options
+
+

Return Value

+

Job ID or 0 on error

+

 CUPS 1.3/Mac OS X 10.5 cupsRemoveDest

+

Remove a destination from the destination list.

+

+int cupsRemoveDest (
+    const char *name,
+    const char *instance,
+    int num_dests,
+    cups_dest_t **dests
+);

+

Parameters

+
+
name
+
Destination name
+
instance
+
Instance name or NULL
+
num_dests
+
Number of destinations
+
dests
+
Destinations
+
+

Return Value

+

New number of destinations

+

Discussion

+

Removing a destination/instance does not delete the class or printer +queue, merely the lpoptions for that destination/instance. Use the +cupsSetDests or cupsSetDests2 functions to save the new +options for the user. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsRemoveOption

+

Remove an option from an option array.

+

+int cupsRemoveOption (
+    const char *name,
+    int num_options,
+    cups_option_t **options
+);

+

Parameters

+
+
name
+
Option name
+
num_options
+
Current number of options
+
options
+
Options
+
+

Return Value

+

New number of options

+

cupsServer

+

Return the hostname/address of the current server.

+

+const char *cupsServer (void);

+

Return Value

+

Server name

+

Discussion

+

The default server comes from the CUPS_SERVER environment variable, then the +~/.cups/client.conf file, and finally the /etc/cups/client.conf file. If not +set, the default is the local system - either "localhost" or a domain socket +path.
+
+The returned value can be a fully-qualified hostname, a numeric IPv4 or IPv6 +address, or a domain socket pathname.
+
+Note: The current server is tracked separately for each thread in a program. +Multi-threaded programs that override the server via the +cupsSetServer function need to do so in each thread for the same +server to be used.

+

 CUPS 1.5/Mac OS X 10.7 cupsSetClientCertCB

+

Set the client certificate callback.

+

+void cupsSetClientCertCB (
+    cups_client_cert_cb_t cb,
+    void *user_data
+);

+

Parameters

+
+
cb
+
Callback function
+
user_data
+
User data pointer
+
+

Discussion

+

Pass NULL to restore the default callback.
+
+Note: The current certificate callback is tracked separately for each thread +in a program. Multi-threaded programs that override the callback need to do +so in each thread for the same callback to be used. + +

+

 CUPS 1.5/Mac OS X 10.7 cupsSetCredentials

+

Set the default credentials to be used for SSL/TLS +connections.

+

+int cupsSetCredentials (
+    cups_array_t *credentials
+);

+

Parameters

+
+
credentials
+
Array of credentials
+
+

Return Value

+

Status of call (0 = success)

+

Discussion

+

Note: The default credentials are tracked separately for each thread in a +program. Multi-threaded programs that override the setting need to do so in +each thread for the same setting to be used. + +

+

 CUPS 1.3/Mac OS X 10.5 cupsSetDefaultDest

+

Set the default destination.

+

+void cupsSetDefaultDest (
+    const char *name,
+    const char *instance,
+    int num_dests,
+    cups_dest_t *dests
+);

+

Parameters

+
+
name
+
Destination name
+
instance
+
Instance name or NULL
+
num_dests
+
Number of destinations
+
dests
+
Destinations
+
+

cupsSetDests

+

Save the list of destinations for the default server.

+

+void cupsSetDests (
+    int num_dests,
+    cups_dest_t *dests
+);

+

Parameters

+
+
num_dests
+
Number of destinations
+
dests
+
Destinations
+
+

Discussion

+

This function saves the destinations to /etc/cups/lpoptions when run +as root and ~/.cups/lpoptions when run as a normal user.

+

 CUPS 1.1.21/Mac OS X 10.4 cupsSetDests2

+

Save the list of destinations for the specified server.

+

+int cupsSetDests2 (
+    http_t *http,
+    int num_dests,
+    cups_dest_t *dests
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
num_dests
+
Number of destinations
+
dests
+
Destinations
+
+

Return Value

+

0 on success, -1 on error

+

Discussion

+

This function saves the destinations to /etc/cups/lpoptions when run +as root and ~/.cups/lpoptions when run as a normal user. + +

+

cupsSetEncryption

+

Set the encryption preference.

+

+void cupsSetEncryption (
+    http_encryption_t e
+);

+

Parameters

+
+
e
+
New encryption preference
+
+

Discussion

+

The default encryption setting comes from the CUPS_ENCRYPTION +environment variable, then the ~/.cups/client.conf file, and finally the +/etc/cups/client.conf file. If not set, the default is +HTTP_ENCRYPT_IF_REQUESTED.
+
+Note: The current encryption setting is tracked separately for each thread +in a program. Multi-threaded programs that override the setting need to do +so in each thread for the same setting to be used.

+

cupsSetPasswordCB

+

Set the password callback for CUPS.

+

+void cupsSetPasswordCB (
+    cups_password_cb_t cb
+);

+

Parameters

+
+
cb
+
Callback function
+
+

Discussion

+

Pass NULL to restore the default (console) password callback, which +reads the password from the console. Programs should call either this +function or cupsSetPasswordCB2, as only one callback can be registered +by a program per thread.
+
+Note: The current password callback is tracked separately for each thread +in a program. Multi-threaded programs that override the callback need to do +so in each thread for the same callback to be used.

+

 CUPS 1.4/Mac OS X 10.6 cupsSetPasswordCB2

+

Set the advanced password callback for CUPS.

+

+void cupsSetPasswordCB2 (
+    cups_password_cb2_t cb,
+    void *user_data
+);

+

Parameters

+
+
cb
+
Callback function
+
user_data
+
User data pointer
+
+

Discussion

+

Pass NULL to restore the default (console) password callback, which +reads the password from the console. Programs should call either this +function or cupsSetPasswordCB2, as only one callback can be registered +by a program per thread.
+
+Note: The current password callback is tracked separately for each thread +in a program. Multi-threaded programs that override the callback need to do +so in each thread for the same callback to be used. + +

+

cupsSetServer

+

Set the default server name and port.

+

+void cupsSetServer (
+    const char *server
+);

+

Parameters

+
+
server
+
Server name
+
+

Discussion

+

The "server" string can be a fully-qualified hostname, a numeric +IPv4 or IPv6 address, or a domain socket pathname. Hostnames and numeric IP +addresses can be optionally followed by a colon and port number to override +the default port 631, e.g. "hostname:8631". Pass NULL to restore the +default server name and port.
+
+Note: The current server is tracked separately for each thread in a program. +Multi-threaded programs that override the server need to do so in each +thread for the same server to be used.

+

 CUPS 1.5/Mac OS X 10.7 cupsSetServerCertCB

+

Set the server certificate callback.

+

+void cupsSetServerCertCB (
+    cups_server_cert_cb_t cb,
+    void *user_data
+);

+

Parameters

+
+
cb
+
Callback function
+
user_data
+
User data pointer
+
+

Discussion

+

Pass NULL to restore the default callback.
+
+Note: The current credentials callback is tracked separately for each thread +in a program. Multi-threaded programs that override the callback need to do +so in each thread for the same callback to be used. + +

+

cupsSetUser

+

Set the default user name.

+

+void cupsSetUser (
+    const char *user
+);

+

Parameters

+
+
user
+
User name
+
+

Discussion

+

Pass NULL to restore the default user name.
+
+Note: The current user name is tracked separately for each thread in a +program. Multi-threaded programs that override the user name need to do so +in each thread for the same user name to be used.

+

 CUPS 1.4/Mac OS X 10.6 cupsStartDocument

+

Add a document to a job created with cupsCreateJob().

+

+http_status_t cupsStartDocument (
+    http_t *http,
+    const char *name,
+    int job_id,
+    const char *docname,
+    const char *format,
+    int last_document
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
name
+
Destination name
+
job_id
+
Job ID from cupsCreateJob
+
docname
+
Name of document
+
format
+
MIME type or CUPS_FORMAT_foo
+
last_document
+
1 for last document in job, 0 otherwise
+
+

Return Value

+

HTTP status of request

+

Discussion

+

Use cupsWriteRequestData to write data for the document and +cupsFinishDocument to finish the document and get the submission status.
+
+The MIME type constants CUPS_FORMAT_AUTO, CUPS_FORMAT_PDF, +CUPS_FORMAT_POSTSCRIPT, CUPS_FORMAT_RAW, and +CUPS_FORMAT_TEXT are provided for the "format" argument, although +any supported MIME type string can be supplied. + +

+

cupsTempFd

+

Creates a temporary file.

+

+int cupsTempFd (
+    char *filename,
+    int len
+);

+

Parameters

+
+
filename
+
Pointer to buffer
+
len
+
Size of buffer
+
+

Return Value

+

New file descriptor or -1 on error

+

Discussion

+

The temporary filename is returned in the filename buffer. +The temporary file is opened for reading and writing.

+

 DEPRECATED cupsTempFile

+

Generates a temporary filename.

+

+char *cupsTempFile (
+    char *filename,
+    int len
+);

+

Parameters

+
+
filename
+
Pointer to buffer
+
len
+
Size of buffer
+
+

Return Value

+

Filename or NULL on error

+

Discussion

+

The temporary filename is returned in the filename buffer. +This function is deprecated - use cupsTempFd or +cupsTempFile2 instead. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsTempFile2

+

Creates a temporary CUPS file.

+

+cups_file_t *cupsTempFile2 (
+    char *filename,
+    int len
+);

+

Parameters

+
+
filename
+
Pointer to buffer
+
len
+
Size of buffer
+
+

Return Value

+

CUPS file or NULL on error

+

Discussion

+

The temporary filename is returned in the filename buffer. +The temporary file is opened for writing. + +

+

cupsUser

+

Return the current user's name.

+

+const char *cupsUser (void);

+

Return Value

+

User name

+

Discussion

+

Note: The current user name is tracked separately for each thread in a +program. Multi-threaded programs that override the user name with the +cupsSetUser function need to do so in each thread for the same user +name to be used.

+

Data Types

+

 CUPS 1.5/Mac OS X 10.7 cups_client_cert_cb_t

+

Client credentials callback

+

+typedef int (*cups_client_cert_cb_t)(http_t *http, void *tls, cups_array_t *distinguished_names, void *user_data); +

+

cups_dest_t

+

Destination

+

+typedef struct cups_dest_s cups_dest_t; +

+

 CUPS 1.4/Mac OS X 10.6 cups_device_cb_t

+

Device callback

+

+typedef void (*cups_device_cb_t)(const char *device_class, const char *device_id, const char *device_info, const char *device_make_and_model, const char *device_uri, const char *device_location, void *user_data); +

+

cups_job_t

+

Job

+

+typedef struct cups_job_s cups_job_t; +

+

cups_option_t

+

Printer Options

+

+typedef struct cups_option_s cups_option_t; +

+

 CUPS 1.4/Mac OS X 10.6 cups_password_cb2_t

+

New password callback

+

+typedef const char *(*cups_password_cb2_t)(const char *prompt, http_t *http, const char *method, const char *resource, void *user_data); +

+

cups_password_cb_t

+

Password callback

+

+typedef const char *(*cups_password_cb_t)(const char *prompt); +

+

cups_ptype_t

+

Printer type/capability bits

+

+typedef unsigned cups_ptype_t; +

+

 CUPS 1.5/Mac OS X 10.7 cups_server_cert_cb_t

+

Server credentials callback

+

+typedef int (*cups_server_cert_cb_t)(http_t *http, void *tls, cups_array_t *certs, void *user_data); +

+

Structures

+

cups_dest_s

+

Destination

+

struct cups_dest_s {
+    char *name, *instance;
+    int is_default;
+    int num_options;
+    cups_option_t *options;
+};

+

Members

+
+
instance
+
Local instance name or NULL
+
is_default
+
Is this printer the default?
+
num_options
+
Number of options
+
options
+
Options
+
+

cups_job_s

+

Job

+

struct cups_job_s {
+    time_t completed_time;
+    time_t creation_time;
+    char *dest;
+    char *format;
+    int id;
+    int priority;
+    time_t processing_time;
+    int size;
+    ipp_jstate_t state;
+    char *title;
+    char *user;
+};

+

Members

+
+
completed_time
+
Time the job was completed
+
creation_time
+
Time the job was created
+
dest
+
Printer or class name
+
format
+
Document format
+
id
+
The job ID
+
priority
+
Priority (1-100)
+
processing_time
+
Time the job was processed
+
size
+
Size in kilobytes
+
state
+
Job state
+
title
+
Title/job name
+
user
+
User the submitted the job
+
+

cups_option_s

+

Printer Options

+

struct cups_option_s {
+    char *name;
+    char *value;
+};

+

Members

+
+
name
+
Name of option
+
value
+
Value of option
+
+

Variables

+

CF_RETURNS_RETAINED

+

Get the Apple language identifier associated with a +locale ID.

+

const char *locale) CF_RETURNS_RETAINED;

+

Constants

+

cups_ptype_e

+

Printer type/capability bit constants

+

Constants

+
+
CUPS_PRINTER_AUTHENTICATED  CUPS 1.2/Mac OS X 10.5 
+
Printer requires authentication
+
CUPS_PRINTER_BIND
+
Can bind output
+
CUPS_PRINTER_BW
+
Can do B&W printing
+
CUPS_PRINTER_CLASS
+
Printer class
+
CUPS_PRINTER_COLLATE
+
Can collage copies
+
CUPS_PRINTER_COLOR
+
Can do color printing
+
CUPS_PRINTER_COMMANDS  CUPS 1.2/Mac OS X 10.5 
+
Printer supports maintenance commands
+
CUPS_PRINTER_COPIES
+
Can do copies
+
CUPS_PRINTER_COVER
+
Can cover output
+
CUPS_PRINTER_DEFAULT
+
Default printer on network
+
CUPS_PRINTER_DELETE  CUPS 1.2/Mac OS X 10.5 
+
Delete printer
+
CUPS_PRINTER_DISCOVERED  CUPS 1.3/Mac OS X 10.5 
+
Printer was automatically discovered and added
+
CUPS_PRINTER_DUPLEX
+
Can do duplexing
+
CUPS_PRINTER_FAX
+
Fax queue
+
CUPS_PRINTER_IMPLICIT
+
Implicit class
+
CUPS_PRINTER_LARGE
+
Can do D/E/A1/A0
+
CUPS_PRINTER_LOCAL
+
Local printer or class
+
CUPS_PRINTER_MEDIUM
+
Can do Tabloid/B/C/A3/A2
+
CUPS_PRINTER_MFP  CUPS 1.4/Mac OS X 10.6 
+
Printer with scanning capabilities
+
CUPS_PRINTER_NOT_SHARED  CUPS 1.2/Mac OS X 10.5 
+
Printer is not shared
+
CUPS_PRINTER_PUNCH
+
Can punch output
+
CUPS_PRINTER_REJECTING
+
Printer is rejecting jobs
+
CUPS_PRINTER_REMOTE
+
Remote printer or class
+
CUPS_PRINTER_SCANNER  CUPS 1.4/Mac OS X 10.6 
+
Scanner-only device
+
CUPS_PRINTER_SMALL
+
Can do Letter/Legal/A4
+
CUPS_PRINTER_SORT
+
Can sort output
+
CUPS_PRINTER_STAPLE
+
Can staple output
+
CUPS_PRINTER_VARIABLE
+
Can do variable sizes
+
+
+ + diff --git a/doc/help/api-driver.html b/doc/help/api-driver.html new file mode 100644 index 0000000000..1a3ee3aed7 --- /dev/null +++ b/doc/help/api-driver.html @@ -0,0 +1,1156 @@ + + + + + Printer Driver API + + + + + + +
+ + +

Driver API

+ +
+ + + + + + + + + + + + + + + + +
Headercups/driver.h
Library-lcupsdriver
See AlsoProgramming: Introduction to CUPS Programming
+

Contents

+ + + +

Overview

+ +

The driver API provides common dithering, color conversion, and utility +functions for CUPS drivers.

+

Functions

+

cupsCMYKDelete

+

Delete a color separation.

+

+void cupsCMYKDelete (
+    cups_cmyk_t *cmyk
+);

+

Parameters

+
+
cmyk
+
Color separation
+
+

cupsCMYKDoBlack

+

Do a black separation...

+

+void cupsCMYKDoBlack (
+    const cups_cmyk_t *cmyk,
+    const unsigned char *input,
+    short *output,
+    int num_pixels
+);

+

Parameters

+
+
cmyk
+
Color separation
+
input
+
Input grayscale pixels
+
output
+
Output Device-N pixels
+
num_pixels
+
Number of pixels
+
+

cupsCMYKDoCMYK

+

Do a CMYK separation...

+

+void cupsCMYKDoCMYK (
+    const cups_cmyk_t *cmyk,
+    const unsigned char *input,
+    short *output,
+    int num_pixels
+);

+

Parameters

+
+
cmyk
+
Color separation
+
input
+
Input grayscale pixels
+
output
+
Output Device-N pixels
+
num_pixels
+
Number of pixels
+
+

cupsCMYKDoGray

+

Do a grayscale separation...

+

+void cupsCMYKDoGray (
+    const cups_cmyk_t *cmyk,
+    const unsigned char *input,
+    short *output,
+    int num_pixels
+);

+

Parameters

+
+
cmyk
+
Color separation
+
input
+
Input grayscale pixels
+
output
+
Output Device-N pixels
+
num_pixels
+
Number of pixels
+
+

cupsCMYKDoRGB

+

Do an sRGB separation...

+

+void cupsCMYKDoRGB (
+    const cups_cmyk_t *cmyk,
+    const unsigned char *input,
+    short *output,
+    int num_pixels
+);

+

Parameters

+
+
cmyk
+
Color separation
+
input
+
Input grayscale pixels
+
output
+
Output Device-N pixels
+
num_pixels
+
Number of pixels
+
+

cupsCMYKLoad

+

Load a CMYK color profile from PPD attributes.

+

+cups_cmyk_t *cupsCMYKLoad (
+    ppd_file_t *ppd,
+    const char *colormodel,
+    const char *media,
+    const char *resolution
+);

+

Parameters

+
+
ppd
+
PPD file
+
colormodel
+
ColorModel value
+
media
+
MediaType value
+
resolution
+
Resolution value
+
+

Return Value

+

CMYK color separation

+

cupsCMYKNew

+

Create a new CMYK color separation.

+

+cups_cmyk_t *cupsCMYKNew (
+    int num_channels
+);

+

Parameters

+
+
num_channels
+
Number of color components
+
+

Return Value

+

New CMYK separation or NULL

+

cupsCMYKSetBlack

+

Set the transition range for CMY to black.

+

+void cupsCMYKSetBlack (
+    cups_cmyk_t *cmyk,
+    float lower,
+    float upper
+);

+

Parameters

+
+
cmyk
+
CMYK color separation
+
lower
+
No black ink
+
upper
+
Only black ink
+
+

cupsCMYKSetCurve

+

Set a color transform curve using points.

+

+void cupsCMYKSetCurve (
+    cups_cmyk_t *cmyk,
+    int channel,
+    int num_xypoints,
+    const float *xypoints
+);

+

Parameters

+
+
cmyk
+
CMYK color separation
+
channel
+
Color channel
+
num_xypoints
+
Number of X,Y points
+
xypoints
+
X,Y points
+
+

cupsCMYKSetGamma

+

Set a color transform curve using gamma and density.

+

+void cupsCMYKSetGamma (
+    cups_cmyk_t *cmyk,
+    int channel,
+    float gamval,
+    float density
+);

+

Parameters

+
+
cmyk
+
CMYK color separation
+
channel
+
Ink channel
+
gamval
+
Gamma correction
+
density
+
Maximum density
+
+

cupsCMYKSetInkLimit

+

Set the limit on the amount of ink.

+

+void cupsCMYKSetInkLimit (
+    cups_cmyk_t *cmyk,
+    float limit
+);

+

Parameters

+
+
cmyk
+
CMYK color separation
+
limit
+
Limit of ink
+
+

cupsCMYKSetLtDk

+

Set light/dark ink transforms.

+

+void cupsCMYKSetLtDk (
+    cups_cmyk_t *cmyk,
+    int channel,
+    float light,
+    float dark
+);

+

Parameters

+
+
cmyk
+
CMYK color separation
+
channel
+
Dark ink channel (+1 for light)
+
light
+
Light ink only level
+
dark
+
Dark ink only level
+
+

cupsCheckBytes

+

Check to see if all bytes are zero.

+

+int cupsCheckBytes (
+    const unsigned char *bytes,
+    int length
+);

+

Parameters

+
+
bytes
+
Bytes to check
+
length
+
Number of bytes to check
+
+

Return Value

+

1 if they match

+

cupsCheckValue

+

Check to see if all bytes match the given value.

+

+int cupsCheckValue (
+    const unsigned char *bytes,
+    int length,
+    const unsigned char value
+);

+

Parameters

+
+
bytes
+
Bytes to check
+
length
+
Number of bytes to check
+
value
+
Value to check
+
+

Return Value

+

1 if they match

+

cupsDitherDelete

+

Free a dithering buffer.

+

+void cupsDitherDelete (
+    cups_dither_t *d
+);

+

Parameters

+
+
d
+
Dithering buffer
+
+

Discussion

+

Returns 0 on success, -1 on failure.

+

cupsDitherLine

+

Dither a line of pixels...

+

+void cupsDitherLine (
+    cups_dither_t *d,
+    const cups_lut_t *lut,
+    const short *data,
+    int num_channels,
+    unsigned char *p
+);

+

Parameters

+
+
d
+
Dither data
+
lut
+
Lookup table
+
data
+
Separation data
+
num_channels
+
Number of components
+
p
+
Pixels
+
+

cupsDitherNew

+

Create an error-diffusion dithering buffer.

+

+cups_dither_t *cupsDitherNew (
+    int width
+);

+

Parameters

+
+
width
+
Width of output in pixels
+
+

Return Value

+

New state array

+

cupsFindAttr

+

Find a PPD attribute based on the colormodel, +media, and resolution.

+

+ppd_attr_t *cupsFindAttr (
+    ppd_file_t *ppd,
+    const char *name,
+    const char *colormodel,
+    const char *media,
+    const char *resolution,
+    char *spec,
+    int specsize
+);

+

Parameters

+
+
ppd
+
PPD file
+
name
+
Attribute name
+
colormodel
+
Color model
+
media
+
Media type
+
resolution
+
Resolution
+
spec
+
Final selection string
+
specsize
+
Size of string buffer
+
+

Return Value

+

Matching attribute or NULL

+

cupsLutDelete

+

Free the memory used by a lookup table.

+

+void cupsLutDelete (
+    cups_lut_t *lut
+);

+

Parameters

+
+
lut
+
Lookup table to free
+
+

cupsLutLoad

+

Load a LUT from a PPD file.

+

+cups_lut_t *cupsLutLoad (
+    ppd_file_t *ppd,
+    const char *colormodel,
+    const char *media,
+    const char *resolution,
+    const char *ink
+);

+

Parameters

+
+
ppd
+
PPD file
+
colormodel
+
Color model
+
media
+
Media type
+
resolution
+
Resolution
+
ink
+
Ink name
+
+

Return Value

+

New lookup table

+

cupsLutNew

+

Make a lookup table from a list of pixel values.

+

+cups_lut_t *cupsLutNew (
+    int num_values,
+    const float *values
+);

+

Parameters

+
+
num_values
+
Number of values
+
values
+
Lookup table values
+
+

Return Value

+

New lookup table

+

Discussion

+

Returns a pointer to the lookup table on success, NULL on failure.

+

cupsPackHorizontal

+

Pack pixels horizontally...

+

+void cupsPackHorizontal (
+    const unsigned char *ipixels,
+    unsigned char *obytes,
+    int width,
+    const unsigned char clearto,
+    const int step
+);

+

Parameters

+
+
ipixels
+
Input pixels
+
obytes
+
Output bytes
+
width
+
Number of pixels
+
clearto
+
Initial value of bytes
+
step
+
Step value between pixels
+
+

cupsPackHorizontal2

+

Pack 2-bit pixels horizontally...

+

+void cupsPackHorizontal2 (
+    const unsigned char *ipixels,
+    unsigned char *obytes,
+    int width,
+    const int step
+);

+

Parameters

+
+
ipixels
+
Input pixels
+
obytes
+
Output bytes
+
width
+
Number of pixels
+
step
+
Stepping value
+
+

cupsPackHorizontalBit

+

Pack pixels horizontally by bit...

+

+void cupsPackHorizontalBit (
+    const unsigned char *ipixels,
+    unsigned char *obytes,
+    int width,
+    const unsigned char clearto,
+    const unsigned char bit
+);

+

Parameters

+
+
ipixels
+
Input pixels
+
obytes
+
Output bytes
+
width
+
Number of pixels
+
clearto
+
Initial value of bytes
+
bit
+
Bit to check
+
+

cupsPackVertical

+

Pack pixels vertically...

+

+void cupsPackVertical (
+    const unsigned char *ipixels,
+    unsigned char *obytes,
+    int width,
+    const unsigned char bit,
+    const int step
+);

+

Parameters

+
+
ipixels
+
Input pixels
+
obytes
+
Output bytes
+
width
+
Number of input pixels
+
bit
+
Output bit
+
step
+
Number of bytes between columns
+
+

cupsRGBDelete

+

Delete a color separation.

+

+void cupsRGBDelete (
+    cups_rgb_t *rgbptr
+);

+

Parameters

+
+
rgbptr
+
Color separation
+
+

cupsRGBDoGray

+

Do a grayscale separation...

+

+void cupsRGBDoGray (
+    cups_rgb_t *rgbptr,
+    const unsigned char *input,
+    unsigned char *output,
+    int num_pixels
+);

+

Parameters

+
+
rgbptr
+
Color separation
+
input
+
Input grayscale pixels
+
output
+
Output Device-N pixels
+
num_pixels
+
Number of pixels
+
+

cupsRGBDoRGB

+

Do a RGB separation...

+

+void cupsRGBDoRGB (
+    cups_rgb_t *rgbptr,
+    const unsigned char *input,
+    unsigned char *output,
+    int num_pixels
+);

+

Parameters

+
+
rgbptr
+
Color separation
+
input
+
Input RGB pixels
+
output
+
Output Device-N pixels
+
num_pixels
+
Number of pixels
+
+

cupsRGBLoad

+

Load a RGB color profile from a PPD file.

+

+cups_rgb_t *cupsRGBLoad (
+    ppd_file_t *ppd,
+    const char *colormodel,
+    const char *media,
+    const char *resolution
+);

+

Parameters

+
+
ppd
+
PPD file
+
colormodel
+
Color model
+
media
+
Media type
+
resolution
+
Resolution
+
+

Return Value

+

New color profile

+

cupsRGBNew

+

Create a new RGB color separation.

+

+cups_rgb_t *cupsRGBNew (
+    int num_samples,
+    cups_sample_t *samples,
+    int cube_size,
+    int num_channels
+);

+

Parameters

+
+
num_samples
+
Number of samples
+
samples
+
Samples
+
cube_size
+
Size of LUT cube
+
num_channels
+
Number of color components
+
+

Return Value

+

New color separation or NULL

+

Data Types

+

cups_cmyk_t

+

Simple CMYK lookup table

+

+typedef struct cups_cmyk_s cups_cmyk_t; +

+

cups_dither_t

+

Dithering State

+

+typedef struct cups_dither_s cups_dither_t; +

+

cups_lut_t

+

Lookup Table for Dithering

+

+typedef struct cups_lut_s cups_lut_t; +

+

cups_rgb_t

+

Color separation lookup table

+

+typedef struct cups_rgb_s cups_rgb_t; +

+

cups_sample_t

+

Color sample point

+

+typedef struct cups_sample_s cups_sample_t; +

+

Structures

+

cups_cmyk_s

+

Simple CMYK lookup table

+

struct cups_cmyk_s {
+    unsigned char black_lut[256];
+    short *channels[CUPS_MAX_CHAN];
+    unsigned char color_lut[256];
+    int ink_limit;
+    int num_channels;
+};

+

Members

+
+
black_lut[256]
+
Black generation LUT
+
channels[CUPS_MAX_CHAN]
+
Lookup tables
+
color_lut[256]
+
Color removal LUT
+
ink_limit
+
Ink limit
+
num_channels
+
Number of components
+
+

cups_dither_s

+

Dithering State

+

struct cups_dither_s {
+    int errors[96];
+    int row;
+    int width;
+};

+

Members

+
+
errors[96]
+
Error values
+
row
+
Current row
+
width
+
Width of buffer
+
+

cups_lut_s

+

Lookup Table for Dithering

+

struct cups_lut_s {
+    int error;
+    short intensity;
+    short pixel;
+};

+

Members

+
+
error
+
Error from desired value
+
intensity
+
Adjusted intensity
+
pixel
+
Output pixel value
+
+

cups_rgb_s

+

Color separation lookup table

+

struct cups_rgb_s {
+    unsigned char black[CUPS_MAX_RGB];
+    int cache_init;
+    unsigned char ****colors;
+    int cube_index[256];
+    int cube_mult[256];
+    int cube_size;
+    int num_channels;
+    unsigned char white[CUPS_MAX_RGB];
+};

+

Members

+
+
black[CUPS_MAX_RGB]
+
Cached black (sRGB = 0,0,0)
+
cache_init
+
Are cached values initialized?
+
colors
+
4-D array of sample values
+
cube_index[256]
+
Index into cube for a given sRGB value
+
cube_mult[256]
+
Multiplier value for a given sRGB value
+
cube_size
+
Size of color cube (2-N) on a side
+
num_channels
+
Number of colors per sample
+
white[CUPS_MAX_RGB]
+
Cached white (sRGB = 255,255,255)
+
+

cups_sample_s

+

Color sample point

+

struct cups_sample_s {
+    unsigned char colors[CUPS_MAX_RGB];
+    unsigned char rgb[3];
+};

+

Members

+
+
colors[CUPS_MAX_RGB]
+
Color values
+
rgb[3]
+
sRGB values
+
+

Variables

+

cups_scmy_lut[256]

+

sRGB gamma lookup table (inverted)

+

extern const unsigned char cups_scmy_lut[256];

+

cups_srgb_lut[256]

+

sRGB gamma lookup table

+

extern const unsigned char cups_srgb_lut[256];

+
+ + diff --git a/doc/help/api-filedir.html b/doc/help/api-filedir.html new file mode 100644 index 0000000000..430494a75c --- /dev/null +++ b/doc/help/api-filedir.html @@ -0,0 +1,995 @@ + + + + + File and Directory APIs + + + + + + +
+ + +

File and Directory APIs

+ +
+ + + + + + + + + + + + + + + + +
Headerscups/file.h
+ cups/dir.h
Library-lcups
See AlsoProgramming: Introduction to CUPS Programming
+ Programming: CUPS API
+

Contents

+ + + +

Overview

+ +

The CUPS file and directory APIs provide portable interfaces +for manipulating files and listing files and directories. Unlike +stdio FILE streams, the cupsFile functions +allow you to open more than 256 files at any given time. They +also manage the platform-specific details of locking, large file +support, line endings (CR, LF, or CR LF), and reading and writing +files using Flate ("gzip") compression. Finally, you can also +connect, read from, and write to network connections using the +cupsFile functions.

+ +

The cupsDir functions manage the platform-specific +details of directory access/listing and provide a convenient way +to get both a list of files and the information (permissions, +size, timestamp, etc.) for each of those files.

+

Functions

+

 CUPS 1.2/Mac OS X 10.5 cupsDirClose

+

Close a directory.

+

+void cupsDirClose (
+    cups_dir_t *dp
+);

+

Parameters

+
+
dp
+
Directory pointer
+
+

 CUPS 1.2/Mac OS X 10.5 cupsDirOpen

+

Open a directory.

+

+cups_dir_t *cupsDirOpen (
+    const char *directory
+);

+

Parameters

+
+
directory
+
Directory name
+
+

Return Value

+

Directory pointer or NULL if the directory could not be opened.

+

 CUPS 1.2/Mac OS X 10.5 cupsDirRead

+

Read the next directory entry.

+

+cups_dentry_t *cupsDirRead (
+    cups_dir_t *dp
+);

+

Parameters

+
+
dp
+
Directory pointer
+
+

Return Value

+

Directory entry or NULL when there are no more

+

 CUPS 1.2/Mac OS X 10.5 cupsDirRewind

+

Rewind to the start of the directory.

+

+void cupsDirRewind (
+    cups_dir_t *dp
+);

+

Parameters

+
+
dp
+
Directory pointer
+
+

 CUPS 1.2/Mac OS X 10.5 cupsFileClose

+

Close a CUPS file.

+

+int cupsFileClose (
+    cups_file_t *fp
+);

+

Parameters

+
+
fp
+
CUPS file
+
+

Return Value

+

0 on success, -1 on error

+

 CUPS 1.2/Mac OS X 10.5 cupsFileCompression

+

Return whether a file is compressed.

+

+int cupsFileCompression (
+    cups_file_t *fp
+);

+

Parameters

+
+
fp
+
CUPS file
+
+

Return Value

+

CUPS_FILE_NONE or CUPS_FILE_GZIP

+

 CUPS 1.2/Mac OS X 10.5 cupsFileEOF

+

Return the end-of-file status.

+

+int cupsFileEOF (
+    cups_file_t *fp
+);

+

Parameters

+
+
fp
+
CUPS file
+
+

Return Value

+

1 on end of file, 0 otherwise

+

 CUPS 1.2/Mac OS X 10.5 cupsFileFind

+

Find a file using the specified path.

+

+const char *cupsFileFind (
+    const char *filename,
+    const char *path,
+    int executable,
+    char *buffer,
+    int bufsize
+);

+

Parameters

+
+
filename
+
File to find
+
path
+
Colon/semicolon-separated path
+
executable
+
1 = executable files, 0 = any file/dir
+
buffer
+
Filename buffer
+
bufsize
+
Size of filename buffer
+
+

Return Value

+

Full path to file or NULL if not found

+

Discussion

+

This function allows the paths in the path string to be separated by +colons (UNIX standard) or semicolons (Windows standard) and stores the +result in the buffer supplied. If the file cannot be found in any of +the supplied paths, NULL is returned. A NULL path only +matches the current directory. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsFileFlush

+

Flush pending output.

+

+int cupsFileFlush (
+    cups_file_t *fp
+);

+

Parameters

+
+
fp
+
CUPS file
+
+

Return Value

+

0 on success, -1 on error

+

 CUPS 1.2/Mac OS X 10.5 cupsFileGetChar

+

Get a single character from a file.

+

+int cupsFileGetChar (
+    cups_file_t *fp
+);

+

Parameters

+
+
fp
+
CUPS file
+
+

Return Value

+

Character or -1 on end of file

+

 CUPS 1.2/Mac OS X 10.5 cupsFileGetConf

+

Get a line from a configuration file.

+

+char *cupsFileGetConf (
+    cups_file_t *fp,
+    char *buf,
+    size_t buflen,
+    char **value,
+    int *linenum
+);

+

Parameters

+
+
fp
+
CUPS file
+
buf
+
String buffer
+
buflen
+
Size of string buffer
+
value
+
Pointer to value
+
linenum
+
Current line number
+
+

Return Value

+

Line read or NULL on end of file or error

+

 CUPS 1.2/Mac OS X 10.5 cupsFileGetLine

+

Get a CR and/or LF-terminated line that may +contain binary data.

+

+size_t cupsFileGetLine (
+    cups_file_t *fp,
+    char *buf,
+    size_t buflen
+);

+

Parameters

+
+
fp
+
File to read from
+
buf
+
Buffer
+
buflen
+
Size of buffer
+
+

Return Value

+

Number of bytes on line or 0 on end of file

+

Discussion

+

This function differs from cupsFileGets in that the trailing CR +and LF are preserved, as is any binary data on the line. The buffer is +nul-terminated, however you should use the returned length to determine +the number of bytes on the line. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsFileGets

+

Get a CR and/or LF-terminated line.

+

+char *cupsFileGets (
+    cups_file_t *fp,
+    char *buf,
+    size_t buflen
+);

+

Parameters

+
+
fp
+
CUPS file
+
buf
+
String buffer
+
buflen
+
Size of string buffer
+
+

Return Value

+

Line read or NULL on end of file or error

+

 CUPS 1.2/Mac OS X 10.5 cupsFileLock

+

Temporarily lock access to a file.

+

+int cupsFileLock (
+    cups_file_t *fp,
+    int block
+);

+

Parameters

+
+
fp
+
CUPS file
+
block
+
1 to wait for the lock, 0 to fail right away
+
+

Return Value

+

0 on success, -1 on error

+

 CUPS 1.2/Mac OS X 10.5 cupsFileNumber

+

Return the file descriptor associated with a CUPS file.

+

+int cupsFileNumber (
+    cups_file_t *fp
+);

+

Parameters

+
+
fp
+
CUPS file
+
+

Return Value

+

File descriptor

+

 CUPS 1.2/Mac OS X 10.5 cupsFileOpen

+

Open a CUPS file.

+

+cups_file_t *cupsFileOpen (
+    const char *filename,
+    const char *mode
+);

+

Parameters

+
+
filename
+
Name of file
+
mode
+
Open mode
+
+

Return Value

+

CUPS file or NULL if the file or socket cannot be opened

+

Discussion

+

The "mode" parameter can be "r" to read, "w" to write, overwriting any +existing file, "a" to append to an existing file or create a new file, +or "s" to open a socket connection.
+
+When opening for writing ("w"), an optional number from 1 to 9 can be +supplied which enables Flate compression of the file. Compression is +not supported for the "a" (append) mode.
+
+When opening a socket connection, the filename is a string of the form +"address:port" or "hostname:port". The socket will make an IPv4 or IPv6 +connection as needed, generally preferring IPv6 connections when there is +a choice. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsFileOpenFd

+

Open a CUPS file using a file descriptor.

+

+cups_file_t *cupsFileOpenFd (
+    int fd,
+    const char *mode
+);

+

Parameters

+
+
fd
+
File descriptor
+
mode
+
Open mode
+
+

Return Value

+

CUPS file or NULL if the file could not be opened

+

Discussion

+

The "mode" parameter can be "r" to read, "w" to write, "a" to append, +or "s" to treat the file descriptor as a bidirectional socket connection.
+
+When opening for writing ("w"), an optional number from 1 to 9 can be +supplied which enables Flate compression of the file. Compression is +not supported for the "a" (append) mode. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsFilePeekChar

+

Peek at the next character from a file.

+

+int cupsFilePeekChar (
+    cups_file_t *fp
+);

+

Parameters

+
+
fp
+
CUPS file
+
+

Return Value

+

Character or -1 on end of file

+

 CUPS 1.2/Mac OS X 10.5 cupsFilePrintf

+

Write a formatted string.

+

+int cupsFilePrintf (
+    cups_file_t *fp,
+    const char *format,
+    ...
+);

+

Parameters

+
+
fp
+
CUPS file
+
format
+
Printf-style format string
+
...
+
Additional args as necessary
+
+

Return Value

+

Number of bytes written or -1 on error

+

 CUPS 1.2/Mac OS X 10.5 cupsFilePutChar

+

Write a character.

+

+int cupsFilePutChar (
+    cups_file_t *fp,
+    int c
+);

+

Parameters

+
+
fp
+
CUPS file
+
c
+
Character to write
+
+

Return Value

+

0 on success, -1 on error

+

 CUPS 1.4/Mac OS X 10.6 cupsFilePutConf

+

Write a configuration line.

+

+ssize_t cupsFilePutConf (
+    cups_file_t *fp,
+    const char *directive,
+    const char *value
+);

+

Parameters

+
+
fp
+
CUPS file
+
directive
+
Directive
+
value
+
Value
+
+

Return Value

+

Number of bytes written or -1 on error

+

Discussion

+

This function handles any comment escaping of the value. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsFilePuts

+

Write a string.

+

+int cupsFilePuts (
+    cups_file_t *fp,
+    const char *s
+);

+

Parameters

+
+
fp
+
CUPS file
+
s
+
String to write
+
+

Return Value

+

Number of bytes written or -1 on error

+

Discussion

+

Like the fputs function, no newline is appended to the string. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsFileRead

+

Read from a file.

+

+ssize_t cupsFileRead (
+    cups_file_t *fp,
+    char *buf,
+    size_t bytes
+);

+

Parameters

+
+
fp
+
CUPS file
+
buf
+
Buffer
+
bytes
+
Number of bytes to read
+
+

Return Value

+

Number of bytes read or -1 on error

+

 CUPS 1.2/Mac OS X 10.5 cupsFileRewind

+

Set the current file position to the beginning of the +file.

+

+off_t cupsFileRewind (
+    cups_file_t *fp
+);

+

Parameters

+
+
fp
+
CUPS file
+
+

Return Value

+

New file position or -1 on error

+

 CUPS 1.2/Mac OS X 10.5 cupsFileSeek

+

Seek in a file.

+

+off_t cupsFileSeek (
+    cups_file_t *fp,
+    off_t pos
+);

+

Parameters

+
+
fp
+
CUPS file
+
pos
+
Position in file
+
+

Return Value

+

New file position or -1 on error

+

 CUPS 1.2/Mac OS X 10.5 cupsFileStderr

+

Return a CUPS file associated with stderr.

+

+cups_file_t *cupsFileStderr (void);

+

Return Value

+

CUPS file

+

 CUPS 1.2/Mac OS X 10.5 cupsFileStdin

+

Return a CUPS file associated with stdin.

+

+cups_file_t *cupsFileStdin (void);

+

Return Value

+

CUPS file

+

 CUPS 1.2/Mac OS X 10.5 cupsFileStdout

+

Return a CUPS file associated with stdout.

+

+cups_file_t *cupsFileStdout (void);

+

Return Value

+

CUPS file

+

 CUPS 1.2/Mac OS X 10.5 cupsFileTell

+

Return the current file position.

+

+off_t cupsFileTell (
+    cups_file_t *fp
+);

+

Parameters

+
+
fp
+
CUPS file
+
+

Return Value

+

File position

+

 CUPS 1.2/Mac OS X 10.5 cupsFileUnlock

+

Unlock access to a file.

+

+int cupsFileUnlock (
+    cups_file_t *fp
+);

+

Parameters

+
+
fp
+
CUPS file
+
+

Return Value

+

0 on success, -1 on error

+

 CUPS 1.2/Mac OS X 10.5 cupsFileWrite

+

Write to a file.

+

+ssize_t cupsFileWrite (
+    cups_file_t *fp,
+    const char *buf,
+    size_t bytes
+);

+

Parameters

+
+
fp
+
CUPS file
+
buf
+
Buffer
+
bytes
+
Number of bytes to write
+
+

Return Value

+

Number of bytes written or -1 on error

+

Data Types

+

cups_dentry_t

+

Directory entry type

+

+typedef struct cups_dentry_s cups_dentry_t; +

+

cups_dir_t

+

Directory type

+

+typedef struct _cups_dir_s cups_dir_t; +

+

cups_file_t

+

CUPS file type

+

+typedef struct _cups_file_s cups_file_t; +

+

Structures

+

cups_dentry_s

+

Directory entry type

+

struct cups_dentry_s {
+    struct stat fileinfo;
+    char filename[260];
+};

+

Members

+
+
fileinfo
+
File information
+
filename[260]
+
File name
+
+
+ + diff --git a/doc/help/api-filter.html b/doc/help/api-filter.html new file mode 100644 index 0000000000..3a9d1288eb --- /dev/null +++ b/doc/help/api-filter.html @@ -0,0 +1,1625 @@ + + + + + Filter and Backend Programming + + + + + + +
+ + +

Filter and Backend Programming

+ +
+ + + + + + + + + + + + + + + + +
Headerscups/backend.h
+ cups/sidechannel.h
Library-lcups
See AlsoProgramming: Introduction to CUPS Programming
+ Programming: CUPS API
+ Programming: PPD API
+ Programming: Raster API
+ Programming: Developing PostScript Printer Drivers
+ Programming: Developing Raster Printer Drivers
+ Specifications: CUPS Design Description
+

Contents

+ + + +

Overview

+ +

Filters (which include printer drivers and port monitors) and backends +are used to convert job files to a printable format and send that data to the +printer itself. All of these programs use a common interface for processing +print jobs and communicating status information to the scheduler. Each is run +with a standard set of command-line arguments:

+ +

+ +
argv[1]
+
The job ID
+ +
argv[2]
+
The user printing the job
+ +
argv[3]
+
The job name/title
+ +
argv[4]
+
The number of copies to print
+ +
argv[5]
+
The options that were provided when the job was submitted
+ +
argv[6]
+
The file to print (first program only)
+
+ +

The scheduler runs one or more of these programs to print any given job. The +first filter reads from the print file and writes to the standard output, while +the remaining filters read from the standard input and write to the standard +output. The backend is the last filter in the chain and writes to the +device.

+ +

Filters are always run as a non-privileged user, typically "lp", with no +connection to the user's desktop. Backends are run either as a non-privileged +user or as root if the file permissions do not allow user or group execution. +The file permissions section talks about this in +more detail.

+ +

Security Considerations

+ +

It is always important to use security programming practices. Filters and +most backends are run as a non-privileged user, so the major security +consideration is resource utilization - filters should not depend on unlimited +amounts of CPU, memory, or disk space, and should protect against conditions +that could lead to excess usage of any resource like infinite loops and +unbounded recursion. In addition, filters must never allow the user to +specify an arbitrary file path to a separator page, template, or other file +used by the filter since that can lead to an unauthorized disclosure of +information. Always treat input as suspect and validate it!

+ +

If you are developing a backend that runs as root, make sure to check for +potential buffer overflows, integer under/overflow conditions, and file +accesses since these can lead to privilege escalations. When writing files, +always validate the file path and never allow a user to determine +where to store a file.

+ +
Note: + +

Never write files to a user's home directory. Aside from the +security implications, CUPS is a network print service and as such the network +user may not be the same as the local user and/or there may not be a local home +directory to write to.

+ +

In addition, some operating systems provide additional security mechanisms +that further limit file system access, even for backends running as root. On +Mac OS X, for example, no backend may write to a user's home directory.

+
+ +

Canceled Jobs and Signal Handling

+ +

The scheduler sends SIGTERM when a printing job is canceled or +held. Filters, backends, and port monitors must catch +SIGTERM and perform any cleanup necessary to produce a valid output +file or return the printer to a known good state. The recommended behavior is to +end the output on the current page, preferably on the current line or object +being printed.

+ +

Filters and backends may also receive SIGPIPE when an upstream or downstream filter/backend exits with a non-zero status. Developers should generally ignore SIGPIPE at the beginning of main() with the following function call:

+ +
+#include <signal.h>>
+
+...
+
+int
+main(int argc, char *argv[])
+{
+  signal(SIGPIPE, SIG_IGN);
+
+  ...
+}
+
+ +

File Permissions

+ +

For security reasons, CUPS will only run filters and backends that are owned +by root and do not have world or group write permissions. The recommended +permissions for filters and backends are 0555 - read and execute but no write. +Backends that must run as root should use permissions of 0500 - read and execute +by root, no access for other users. Write permissions can be enabled for the +root user only.

+ +

To avoid a warning message, the directory containing your filter(s) must also +be owned by root and have world and group write disabled - permissions of 0755 +or 0555 are strongly encouraged.

+ +

Temporary Files

+ +

Temporary files should be created in the directory specified by the +"TMPDIR" environment variable. The +cupsTempFile2 function can be +used to safely create temporary files in this directory.

+ +

Copy Generation

+ +

The argv[4] argument specifies the number of copies to produce +of the input file. In general, you should only generate copies if the +filename argument is supplied. The only exception to this are +filters that produce device-independent PostScript output, since the PostScript +filter pstops is responsible for generating copies of PostScript +files.

+ +

Exit Codes

+ +

Filters must exit with status 0 when they successfully generate print data +or 1 when they encounter an error. Backends can return any of the +cups_backend_t constants.

+ +

Environment Variables

+ +

The following environment variables are defined by the printing system +when running print filters and backends:

+ +
+ +
APPLE_LANGUAGE
+
The Apple language identifier associated with the job + (Mac OS X only).
+ +
CHARSET
+
The job character set, typically "utf-8".
+ +
CLASS
+
When a job is submitted to a printer class, contains the name of + the destination printer class. Otherwise this environment + variable will not be set.
+ +
CONTENT_TYPE
+
The MIME type associated with the file (e.g. + application/postscript).
+ +
CUPS_CACHEDIR
+
The directory where cache files can be stored. Cache files can be + used to retain information between jobs or files in a job.
+ +
CUPS_DATADIR
+
The directory where (read-only) CUPS data files can be found.
+ +
CUPS_FILETYPE
+
The type of file being printed: "job-sheet" for a banner page and + "document" for a regular print file.
+ +
CUPS_SERVERROOT
+
The root directory of the server.
+ +
DEVICE_URI
+
The device-uri associated with the printer.
+ +
FINAL_CONTENT_TYPE
+
The MIME type associated with the printer (e.g. + application/vnd.cups-postscript).
+ +
LANG
+
The language locale associated with the job.
+ +
PPD
+
The full pathname of the PostScript Printer Description (PPD) + file for this printer.
+ +
PRINTER
+
The queue name of the class or printer.
+ +
RIP_CACHE
+
The recommended amount of memory to use for Raster Image + Processors (RIPs).
+ +
TMPDIR
+
The directory where temporary files should be created.
+ +
+ +

Communicating with the Scheduler

+ +

Filters and backends communicate with the scheduler by writing messages +to the standard error file. The scheduler reads messages from all filters in +a job and processes the message based on its prefix. For example, the following +code sets the current printer state message to "Printing page 5":

+ +
+int page = 5;
+
+fprintf(stderr, "INFO: Printing page %d\n", page);
+
+ +

Each message is a single line of text starting with one of the following +prefix strings:

+ +
+ +
ALERT: message
+
Sets the printer-state-message attribute and adds the specified + message to the current error log file using the "alert" log level.
+ +
ATTR: attribute=value [attribute=value]
+
Sets the named printer or job attribute(s). Typically this is used + to set the marker-colors, marker-high-levels, + marker-levels, marker-low-levels, + marker-message, marker-names, + marker-types, printer-alert, and + printer-alert-description printer attributes. Standard + marker-types values are listed in Table + 1.
+ +
CRIT: message
+
Sets the printer-state-message attribute and adds the specified + message to the current error log file using the "critical" log + level.
+ +
DEBUG: message
+
Sets the printer-state-message attribute and adds the specified + message to the current error log file using the "debug" log level.
+ +
DEBUG2: message
+
Sets the printer-state-message attribute and adds the specified + message to the current error log file using the "debug2" log level.
+ +
EMERG: message
+
Sets the printer-state-message attribute and adds the specified + message to the current error log file using the "emergency" log + level.
+ +
ERROR: message
+
Sets the printer-state-message attribute and adds the specified + message to the current error log file using the "error" log level. + Use "ERROR:" messages for non-persistent processing errors.
+ +
INFO: message
+
Sets the printer-state-message attribute. If the current log level + is set to "debug2", also adds the specified message to the current error + log file using the "info" log level.
+ +
NOTICE: message
+
Sets the printer-state-message attribute and adds the specified + message to the current error log file using the "notice" log level.
+ +
PAGE: page-number #-copies
+
PAGE: total #-pages
+
Adds an entry to the current page log file. The first form adds + #-copies to the job-media-sheets-completed attribute. The second + form sets the job-media-sheets-completed attribute to #-pages.
+ +
PPD: keyword=value [keyword=value ...]
+
Changes or adds keywords to the printer's PPD file. Typically + this is used to update installable options or default media settings + based on the printer configuration.
+ +
STATE: + printer-state-reason [printer-state-reason ...]
+
STATE: - printer-state-reason [printer-state-reason ...]
+
Sets or clears printer-state-reason keywords for the current queue. + Typically this is used to indicate persistent media, ink, toner, and + configuration conditions or errors on a printer. + Table 2 lists the standard state keywords - + use vendor-prefixed ("com.example.foo") keywords for custom states. See + Managing Printer State in a Filter for more + information. + +
WARNING: message
+
Sets the printer-state-message attribute and adds the specified + message to the current error log file using the "warning" log + level.
+ +
+ +

Messages without one of these prefixes are treated as if they began with +the "DEBUG:" prefix string.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 1: Standard marker-types Values
marker-typeDescription
developerDeveloper unit
fuserFuser unit
fuserCleaningPadFuser cleaning pad
fuserOilFuser oil
inkInk supply
opcPhoto conductor
solidWaxWax supply
staplesStaple supply
tonerToner supply
transferUnitTransfer unit
wasteInkWaste ink tank
wasteTonerWaste toner tank
wasteWaxWaste wax tank
+ +
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 2: Standard State Keywords
KeywordDescription
connecting-to-deviceConnecting to printer but not printing yet.
cover-openThe printer's cover is open.
input-tray-missingThe paper tray is missing.
marker-supply-emptyThe printer is out of ink.
marker-supply-lowThe printer is almost out of ink.
marker-waste-almost-fullThe printer's waste bin is almost full.
marker-waste-fullThe printer's waste bin is full.
media-emptyThe paper tray (any paper tray) is empty.
media-jamThere is a paper jam.
media-lowThe paper tray (any paper tray) is almost empty.
media-neededThe paper tray needs to be filled (for a job that is printing).
pausedStop the printer.
timed-outUnable to connect to printer.
toner-emptyThe printer is out of toner.
toner-lowThe printer is low on toner.
+ +

Managing Printer State in a Filter

+ +

Filters are responsible for managing the state keywords they set using +"STATE:" messages. Typically you will update all of the keywords that +are used by the filter at startup, for example:

+ +
+if (foo_condition != 0)
+  fputs("STATE: +com.example.foo\n", stderr);
+else
+  fputs("STATE: -com.example.foo\n", stderr);
+
+if (bar_condition != 0)
+  fputs("STATE: +com.example.bar\n", stderr);
+else
+  fputs("STATE: -com.example.bar\n", stderr);
+
+ +

Then as conditions change, your filter sends "STATE: +keyword" or "STATE: +-keyword" messages as necessary to set or clear the corresponding keyword, +respectively.

+ +

State keywords are often used to notify the user of issues that span across +jobs, for example "media-empty-warning" that indicates one or more paper trays +are empty. These keywords should not be cleared unless the corresponding issue +no longer exists.

+ +

Filters should clear job-related keywords on startup and exit so that they +do not remain set between jobs. For example, "connecting-to-device" is a job +sub-state and not an issue that applies when a job is not printing.

+ +
Note: + +

"STATE:" messages often provide visible alerts to the user. For example, +on Mac OS X setting a printer-state-reason value with an "-error" or +"-warning" suffix will cause the printer's dock item to bounce if the +corresponding reason is localized with a cupsIPPReason keyword in the +printer's PPD file.

+ +

When providing a vendor-prefixed keyword, always provide the +corresponding standard keyword (if any) to allow clients to respond to the +condition correctly. For example, if you provide a vendor-prefixed keyword +for a low cyan ink condition ("com.example.cyan-ink-low") you must also set the +"marker-supply-low-warning" keyword. In such cases you should also refrain +from localizing the vendor-prefixed keyword in the PPD file - otherwise both +the generic and vendor-specific keyword will be shown in the user +interface.

+ +
+ +

Reporting Supply Levels

+ +

CUPS tracks several "marker-*" attributes for ink/toner supply level +reporting. These attributes allow applications to display the current supply +levels for a printer without printer-specific software. Table 3 lists the marker attributes and what they represent.

+ +

Filters set marker attributes by sending "ATTR:" messages to stderr. For +example, a filter supporting an inkjet printer with black and tri-color ink +cartridges would use the following to initialize the supply attributes:

+ +
+fputs("ATTR: marker-colors=#000000,#00FFFF#FF00FF#FFFF00\n", stderr);
+fputs("ATTR: marker-low-levels=5,10\n", stderr);
+fputs("ATTR: marker-names=Black,Tri-Color\n", stderr);
+fputs("ATTR: marker-types=ink,ink\n", stderr);
+
+ +

Then periodically the filter queries the printer for its current supply +levels and updates them with a separate "ATTR:" message:

+ +
+int black_level, tri_level;
+...
+fprintf(stderr, "ATTR: marker-levels=%d,%d\n", black_level, tri_level);
+
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 3: Supply Level Attributes
AttributeDescription
marker-colorsA list of comma-separated colors; each color is either "none" or one or + more hex-encoded sRGB colors of the form "#RRGGBB".
marker-high-levelsA list of comma-separated "almost full" level values from 0 to 100; a + value of 100 should be used for supplies that are consumed/emptied like ink + cartridges.
marker-levelsA list of comma-separated level values for each supply. A value of -1 + indicates the level is unavailable, -2 indicates unknown, and -3 indicates + the level is unknown but has not yet reached capacity. Values from 0 to 100 + indicate the corresponding percentage.
marker-low-levelsA list of comma-separated "almost empty" level values from 0 to 100; a + value of 0 should be used for supplies that are filled like waste ink + tanks.
marker-messageA human-readable supply status message for the user like "12 pages of + ink remaining."
marker-namesA list of comma-separated supply names like "Cyan Ink", "Fuser", + etc.
marker-typesA list of comma-separated supply types; the types are listed in + Table 1.
+ +

Communicating with the Backend

+ +

Filters can communicate with the backend via the +cupsBackChannelRead and +cupsSideChannelDoRequest +functions. The +cupsBackChannelRead function +reads data that has been sent back from the device and is typically used to +obtain status and configuration information. For example, the following code +polls the backend for back-channel data:

+ +
+#include <cups/cups.h>
+
+char buffer[8192];
+ssize_t bytes;
+
+/* Use a timeout of 0.0 seconds to poll for back-channel data */
+bytes = cupsBackChannelRead(buffer, sizeof(buffer), 0.0);
+
+ +

Filters can also use select() or poll() on the +back-channel file descriptor (3 or CUPS_BC_FD) to read data only +when it is available.

+ +

The +cupsSideChannelDoRequest +function allows you to get out-of-band status information and do synchronization +with the device. For example, the following code gets the current IEEE-1284 +device ID string from the backend:

+ +
+#include <cups/sidechannel.h>
+
+char data[2049];
+int datalen;
+cups_sc_status_t status;
+
+/* Tell cupsSideChannelDoRequest() how big our buffer is, less 1 byte for
+   nul-termination... */
+datalen = sizeof(data) - 1;
+
+/* Get the IEEE-1284 device ID, waiting for up to 1 second */
+status = cupsSideChannelDoRequest(CUPS_SC_CMD_GET_DEVICE_ID, data, &datalen, 1.0);
+
+/* Use the returned value if OK was returned and the length is non-zero */
+if (status == CUPS_SC_STATUS_OK && datalen > 0)
+  data[datalen] = '\0';
+else
+  data[0] = '\0';
+
+ +

Forcing All Output to a Printer

+ +

The +cupsSideChannelDoRequest +function allows you to tell the backend to send all pending data to the printer. +This is most often needed when sending query commands to the printer. For example:

+ +
+#include <cups/cups.h>
+#include <cups/sidechannel.h>
+
+char data[1024];
+int datalen = sizeof(data);
+cups_sc_status_t status;
+
+/* Flush pending output to stdout */
+fflush(stdout);
+
+/* Drain output to backend, waiting for up to 30 seconds */
+status = cupsSideChannelDoRequest(CUPS_SC_CMD_DRAIN_OUTPUT, data, &datalen, 30.0);
+
+/* Read the response if the output was sent */
+if (status == CUPS_SC_STATUS_OK)
+{
+  ssize_t bytes;
+
+  /* Wait up to 10.0 seconds for back-channel data */
+  bytes = cupsBackChannelRead(data, sizeof(data), 10.0);
+  /* do something with the data from the printer */
+}
+
+ +

Communicating with Filters

+ +

Backends communicate with filters using the reciprocal functions +cupsBackChannelWrite, +cupsSideChannelRead, and +cupsSideChannelWrite. We +recommend writing back-channel data using a timeout of 1.0 seconds:

+ +
+#include <cups/cups.h>
+
+char buffer[8192];
+ssize_t bytes;
+
+/* Obtain data from printer/device */
+...
+
+/* Use a timeout of 1.0 seconds to give filters a chance to read */
+cupsBackChannelWrite(buffer, bytes, 1.0);
+
+ +

The cupsSideChannelRead +function reads a side-channel command from a filter, driver, or port monitor. +Backends can either poll for commands using a timeout of 0.0, wait +indefinitely for commands using a timeout of -1.0 (probably in a +separate thread for that purpose), or use select or +poll on the CUPS_SC_FD file descriptor (4) to handle +input and output on several file descriptors at the same time.

+ +

Once a command is processed, the backend uses the +cupsSideChannelWrite function +to send its response. For example, the following code shows how to poll for a +side-channel command and respond to it:

+ +
+#include <cups/sidechannel.h>
+
+cups_sc_command_t command;
+cups_sc_status_t status;
+char data[2048];
+int datalen = sizeof(data);
+
+/* Poll for a command... */
+if (!cupsSideChannelRead(&command, &status, data, &datalen, 0.0))
+{
+  switch (command)
+  {
+    /* handle supported commands, fill data/datalen/status with values as needed */
+
+    default :
+        status  = CUPS_SC_STATUS_NOT_IMPLEMENTED;
+	datalen = 0;
+	break;
+  }
+
+  /* Send a response... */
+  cupsSideChannelWrite(command, status, data, datalen, 1.0);
+}
+
+ +

Doing SNMP Queries with Network Printers

+ +

The Simple Network Management Protocol (SNMP) allows you to get the current +status, page counter, and supply levels from most network printers. Every +piece of information is associated with an Object Identifier (OID), and +every printer has a community name associated with it. OIDs can be +queried directly or by "walking" over a range of OIDs with a common prefix.

+ +

The two CUPS SNMP functions provide a simple API for querying network +printers through the side-channel interface. Each accepts a string containing +an OID like ".1.3.6.1.2.1.43.10.2.1.4.1.1" (the standard page counter OID) +along with a timeout for the query.

+ +

The cupsSideChannelSNMPGet +function queries a single OID and returns the value as a string in a buffer +you supply:

+ +
+#include <cups/sidechannel.h>
+
+char data[512];
+int datalen = sizeof(data);
+
+if (cupsSideChannelSNMPGet(".1.3.6.1.2.1.43.10.2.1.4.1.1", data, &datalen, 5.0)
+        == CUPS_SC_STATUS_OK)
+{
+  /* Do something with the value */
+  printf("Page counter is: %s\n", data);
+}
+
+ +

The +cupsSideChannelSNMPWalk +function allows you to query a whole group of OIDs, calling a function of your +choice for each OID that is found:

+ +
+#include <cups/sidechannel.h>
+
+void
+my_callback(const char *oid, const char *data, int datalen, void *context)
+{
+  /* Do something with the value */
+  printf("%s=%s\n", oid, data);
+}
+
+...
+
+void *my_data;
+
+cupsSNMPSideChannelWalk(".1.3.6.1.2.1.43", 5.0, my_callback, my_data);
+
+

Functions

+

 CUPS 1.2/Mac OS X 10.5 cupsBackChannelRead

+

Read data from the backchannel.

+

+ssize_t cupsBackChannelRead (
+    char *buffer,
+    size_t bytes,
+    double timeout
+);

+

Parameters

+
+
buffer
+
Buffer to read into
+
bytes
+
Bytes to read
+
timeout
+
Timeout in seconds, typically 0.0 to poll
+
+

Return Value

+

Bytes read or -1 on error

+

Discussion

+

Reads up to "bytes" bytes from the backchannel/backend. The "timeout" +parameter controls how many seconds to wait for the data - use 0.0 to +return immediately if there is no data, -1.0 to wait for data indefinitely. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsBackChannelWrite

+

Write data to the backchannel.

+

+ssize_t cupsBackChannelWrite (
+    const char *buffer,
+    size_t bytes,
+    double timeout
+);

+

Parameters

+
+
buffer
+
Buffer to write
+
bytes
+
Bytes to write
+
timeout
+
Timeout in seconds, typically 1.0
+
+

Return Value

+

Bytes written or -1 on error

+

Discussion

+

Writes "bytes" bytes to the backchannel/filter. The "timeout" parameter +controls how many seconds to wait for the data to be written - use +0.0 to return immediately if the data cannot be written, -1.0 to wait +indefinitely. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsBackendDeviceURI

+

Get the device URI for a backend.

+

+const char *cupsBackendDeviceURI (
+    char **argv
+);

+

Parameters

+
+
argv
+
Command-line arguments
+
+

Return Value

+

Device URI or NULL

+

Discussion

+

The "argv" argument is the argv argument passed to main(). This +function returns the device URI passed in the DEVICE_URI environment +variable or the device URI passed in argv[0], whichever is found +first. + +

+

 CUPS 1.4/Mac OS X 10.6 cupsBackendReport

+

Write a device line from a backend.

+

+void cupsBackendReport (
+    const char *device_scheme,
+    const char *device_uri,
+    const char *device_make_and_model,
+    const char *device_info,
+    const char *device_id,
+    const char *device_location
+);

+

Parameters

+
+
device_scheme
+
device-scheme string
+
device_uri
+
device-uri string
+
device_make_and_model
+
device-make-and-model string or NULL
+
device_info
+
device-info string or NULL
+
device_id
+
device-id string or NULL
+
device_location
+
device-location string or NULL
+
+

Discussion

+

This function writes a single device line to stdout for a backend. +It handles quoting of special characters in the device-make-and-model, +device-info, device-id, and device-location strings. + +

+

 CUPS 1.3/Mac OS X 10.5 cupsSideChannelDoRequest

+

Send a side-channel command to a backend and wait for a response.

+

+cups_sc_status_t cupsSideChannelDoRequest (
+    cups_sc_command_t command,
+    char *data,
+    int *datalen,
+    double timeout
+);

+

Parameters

+
+
command
+
Command to send
+
data
+
Response data buffer pointer
+
datalen
+
Size of data buffer on entry, number of bytes in buffer on return
+
timeout
+
Timeout in seconds
+
+

Return Value

+

Status of command

+

Discussion

+

This function is normally only called by filters, drivers, or port +monitors in order to communicate with the backend used by the current +printer. Programs must be prepared to handle timeout or "not +implemented" status codes, which indicate that the backend or device +do not support the specified side-channel command.
+
+The "datalen" parameter must be initialized to the size of the buffer +pointed to by the "data" parameter. cupsSideChannelDoRequest() will +update the value to contain the number of data bytes in the buffer. + +

+

 CUPS 1.3/Mac OS X 10.5 cupsSideChannelRead

+

Read a side-channel message.

+

+int cupsSideChannelRead (
+    cups_sc_command_t *command,
+    cups_sc_status_t *status,
+    char *data,
+    int *datalen,
+    double timeout
+);

+

Parameters

+
+
command
+
Command code
+
status
+
Status code
+
data
+
Data buffer pointer
+
datalen
+
Size of data buffer on entry, number of bytes in buffer on return
+
timeout
+
Timeout in seconds
+
+

Return Value

+

0 on success, -1 on error

+

Discussion

+

This function is normally only called by backend programs to read +commands from a filter, driver, or port monitor program. The +caller must be prepared to handle incomplete or invalid messages +and return the corresponding status codes.
+
+The "datalen" parameter must be initialized to the size of the buffer +pointed to by the "data" parameter. cupsSideChannelDoRequest() will +update the value to contain the number of data bytes in the buffer. + +

+

 CUPS 1.4/Mac OS X 10.6 cupsSideChannelSNMPGet

+

Query a SNMP OID's value.

+

+cups_sc_status_t cupsSideChannelSNMPGet (
+    const char *oid,
+    char *data,
+    int *datalen,
+    double timeout
+);

+

Parameters

+
+
oid
+
OID to query
+
data
+
Buffer for OID value
+
datalen
+
Size of OID buffer on entry, size of value on return
+
timeout
+
Timeout in seconds
+
+

Return Value

+

Query status

+

Discussion

+

This function asks the backend to do a SNMP OID query on behalf of the +filter, port monitor, or backend using the default community name.
+
+"oid" contains a numeric OID consisting of integers separated by periods, +for example ".1.3.6.1.2.1.43". Symbolic names from SNMP MIBs are not +supported and must be converted to their numeric forms.
+
+On input, "data" and "datalen" provide the location and size of the +buffer to hold the OID value as a string. HEX-String (binary) values are +converted to hexadecimal strings representing the binary data, while +NULL-Value and unknown OID types are returned as the empty string. +The returned "datalen" does not include the trailing nul. + +CUPS_SC_STATUS_NOT_IMPLEMENTED is returned by backends that do not +support SNMP queries. CUPS_SC_STATUS_NO_RESPONSE is returned when +the printer does not respond to the SNMP query. + +

+

 CUPS 1.4/Mac OS X 10.6 cupsSideChannelSNMPWalk

+

Query multiple SNMP OID values.

+

+cups_sc_status_t cupsSideChannelSNMPWalk (
+    const char *oid,
+    double timeout,
+    cups_sc_walk_func_t cb,
+    void *context
+);

+

Parameters

+
+
oid
+
First numeric OID to query
+
timeout
+
Timeout for each query in seconds
+
cb
+
Function to call with each value
+
context
+
Application-defined pointer to send to callback
+
+

Return Value

+

Status of first query of CUPS_SC_STATUS_OK on success

+

Discussion

+

This function asks the backend to do multiple SNMP OID queries on behalf +of the filter, port monitor, or backend using the default community name. +All OIDs under the "parent" OID are queried and the results are sent to +the callback function you provide.
+
+"oid" contains a numeric OID consisting of integers separated by periods, +for example ".1.3.6.1.2.1.43". Symbolic names from SNMP MIBs are not +supported and must be converted to their numeric forms.
+
+"timeout" specifies the timeout for each OID query. The total amount of +time will depend on the number of OID values found and the time required +for each query.
+
+"cb" provides a function to call for every value that is found. "context" +is an application-defined pointer that is sent to the callback function +along with the OID and current data. The data passed to the callback is the +same as returned by cupsSideChannelSNMPGet. + +CUPS_SC_STATUS_NOT_IMPLEMENTED is returned by backends that do not +support SNMP queries. CUPS_SC_STATUS_NO_RESPONSE is returned when +the printer does not respond to the first SNMP query. + +

+

 CUPS 1.3/Mac OS X 10.5 cupsSideChannelWrite

+

Write a side-channel message.

+

+int cupsSideChannelWrite (
+    cups_sc_command_t command,
+    cups_sc_status_t status,
+    const char *data,
+    int datalen,
+    double timeout
+);

+

Parameters

+
+
command
+
Command code
+
status
+
Status code
+
data
+
Data buffer pointer
+
datalen
+
Number of bytes of data
+
timeout
+
Timeout in seconds
+
+

Return Value

+

0 on success, -1 on error

+

Discussion

+

This function is normally only called by backend programs to send +responses to a filter, driver, or port monitor program. + +

+

Data Types

+

cups_backend_t

+

Backend exit codes

+

+typedef enum cups_backend_e cups_backend_t; +

+

cups_sc_bidi_t

+

Bidirectional capabilities

+

+typedef enum cups_sc_bidi_e cups_sc_bidi_t; +

+

cups_sc_command_t

+

Request command codes

+

+typedef enum cups_sc_command_e cups_sc_command_t; +

+

cups_sc_connected_t

+

Connectivity values

+

+typedef enum cups_sc_connected_e cups_sc_connected_t; +

+

cups_sc_state_t

+

Printer state bits

+

+typedef enum cups_sc_state_e cups_sc_state_t; +

+

cups_sc_status_t

+

Response status codes

+

+typedef enum cups_sc_status_e cups_sc_status_t; +

+

cups_sc_walk_func_t

+

SNMP walk callback

+

+typedef void (*cups_sc_walk_func_t)(const char *oid, const char *data, int datalen, void *context); +

+

Constants

+

cups_backend_e

+

Backend exit codes

+

Constants

+
+
CUPS_BACKEND_AUTH_REQUIRED
+
Job failed, authentication required
+
CUPS_BACKEND_CANCEL
+
Job failed, cancel job
+
CUPS_BACKEND_FAILED
+
Job failed, use error-policy
+
CUPS_BACKEND_HOLD
+
Job failed, hold job
+
CUPS_BACKEND_OK
+
Job completed successfully
+
CUPS_BACKEND_RETRY
+
Job failed, retry this job later
+
CUPS_BACKEND_RETRY_CURRENT
+
Job failed, retry this job immediately
+
CUPS_BACKEND_STOP
+
Job failed, stop queue
+
+

cups_sc_bidi_e

+

Bidirectional capability values

+

Constants

+
+
CUPS_SC_BIDI_NOT_SUPPORTED
+
Bidirectional I/O is not supported
+
CUPS_SC_BIDI_SUPPORTED
+
Bidirectional I/O is supported
+
+

cups_sc_command_e

+

Request command codes

+

Constants

+
+
CUPS_SC_CMD_DRAIN_OUTPUT
+
Drain all pending output
+
CUPS_SC_CMD_GET_BIDI
+
Return bidirectional capabilities
+
CUPS_SC_CMD_GET_CONNECTED  CUPS 1.5/Mac OS X 10.7 
+
Return whether the backend is "connected" to the printer
+
CUPS_SC_CMD_GET_DEVICE_ID
+
Return the IEEE-1284 device ID
+
CUPS_SC_CMD_GET_STATE
+
Return the device state
+
CUPS_SC_CMD_SNMP_GET  CUPS 1.4/Mac OS X 10.6 
+
Query an SNMP OID
+
CUPS_SC_CMD_SNMP_GET_NEXT  CUPS 1.4/Mac OS X 10.6 
+
Query the next SNMP OID
+
CUPS_SC_CMD_SOFT_RESET
+
Do a soft reset
+
+

cups_sc_connected_e

+

Connectivity values

+

Constants

+
+
CUPS_SC_CONNECTED
+
Backend is "connected" to printer
+
CUPS_SC_NOT_CONNECTED
+
Backend is not "connected" to printer
+
+

cups_sc_state_e

+

Printer state bits

+

Constants

+
+
CUPS_SC_STATE_BUSY
+
Device is busy
+
CUPS_SC_STATE_ERROR
+
Other error condition
+
CUPS_SC_STATE_MARKER_EMPTY
+
Toner/ink out condition
+
CUPS_SC_STATE_MARKER_LOW
+
Toner/ink low condition
+
CUPS_SC_STATE_MEDIA_EMPTY
+
Paper out condition
+
CUPS_SC_STATE_MEDIA_LOW
+
Paper low condition
+
CUPS_SC_STATE_OFFLINE
+
Device is offline
+
CUPS_SC_STATE_ONLINE
+
Device is online
+
+

cups_sc_status_e

+

Response status codes

+

Constants

+
+
CUPS_SC_STATUS_BAD_MESSAGE
+
The command/response message was invalid
+
CUPS_SC_STATUS_IO_ERROR
+
An I/O error occurred
+
CUPS_SC_STATUS_NONE
+
No status
+
CUPS_SC_STATUS_NOT_IMPLEMENTED
+
Command not implemented
+
CUPS_SC_STATUS_NO_RESPONSE
+
The device did not respond
+
CUPS_SC_STATUS_OK
+
Operation succeeded
+
CUPS_SC_STATUS_TIMEOUT
+
The backend did not respond
+
CUPS_SC_STATUS_TOO_BIG
+
Response too big
+
+
+ + diff --git a/doc/help/api-httpipp.html b/doc/help/api-httpipp.html new file mode 100644 index 0000000000..857abbdc20 --- /dev/null +++ b/doc/help/api-httpipp.html @@ -0,0 +1,5358 @@ + + + + + HTTP and IPP APIs + + + + + + +
+ + +

HTTP and IPP APIs

+ +
+ + + + + + + + + + + + + + + + +
Headercups/cups.h
Library-lcups
See AlsoProgramming: Introduction to CUPS Programming
+ Programming: CUPS API
+ References: CUPS Implementation of IPP
+

Contents

+ + + +

Overview

+ +

The CUPS HTTP and IPP APIs provide low-level access to the HTTP and IPP +protocols and CUPS scheduler. They are typically used by monitoring and +administration programs to perform specific functions not supported by the +high-level CUPS API functions.

+ +

The HTTP APIs use an opaque structure called +http_t to manage connections to +a particular HTTP or IPP server. The +httpConnectEncrypt function is +used to create an instance of this structure for a particular server. +The constant CUPS_HTTP_DEFAULT can be used with all of the +cups functions to refer to the default CUPS server - the functions +create a per-thread http_t as needed.

+ +

The IPP APIs use two structures for requests (messages sent to the CUPS +scheduler) and responses (messages sent back to your application from the +scheduler). The ipp_t structure holds a +complete request or response and is allocated using the +ippNew or +ippNewRequest functions and +freed using the ippDelete function.

+ +

The second structure is called +ipp_attribute_t and holds a +single IPP attribute which consists of a group tag (group_tag), a +value type tag (value_tag), the attribute name (name), +and 1 or more values (values[]). Attributes are added to an +ipp_t structure using one of the +ippAdd functions. For example, use +ippAddString to add a +"requesting-user-name" string attribute to a request:

+ +
+ipp_t *request = ippNewRequest(IPP_GET_JOBS);
+
+ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name",
+             NULL, cupsUser());
+
+ +

Once you have created an IPP request, use the cups +functions to send the request to and read the response from the server. +For example, the cupsDoRequest +function can be used for simple query operations that do not involve files:

+ +
+#include <cups/cups.h>
+
+
+ipp_t *get_jobs(void)
+{
+  ipp_t *request = ippNewRequest(IPP_GET_JOBS);
+
+  ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name",
+               NULL, cupsUser());
+
+  return (cupsDoRequest(CUPS_HTTP_DEFAULT, request, "/"));
+}
+
+ +

The cupsDoRequest function frees +the request structure and returns an IPP response structure or NULL pointer if +the request could not be sent to the server. Once you have a response from +the server, you can either use the +ippFindAttribute and +ippFindNextAttribute functions +to find specific attributes, for example:

+ +
+ipp_t *response;
+ipp_attribute_t *attr;
+
+attr = ippFindAttribute(response, "printer-state", IPP_TAG_ENUM);
+
+ +

You can also walk the list of attributes with a simple for loop +like this:

+ +
+ipp_t *response;
+ipp_attribute_t *attr;
+
+for (attr = response->attrs; attr != NULL; attr = attr->next)
+  if (attr->name == NULL)
+    puts("--SEPARATOR--");
+  else
+    puts(attr->name);
+
+ +

The for loop approach is normally used when collecting +attributes for multiple objects (jobs, printers, etc.) in a response. Attributes +with NULL names indicate a separator between the attributes of +each object. For example, the following code will list the jobs returned from +our previous get_jobs example code:

+ +
+ipp_t *response = get_jobs();
+
+if (response != NULL)
+{
+  ipp_attribute_t *attr;
+  int job_id = 0;
+  char *job_name = NULL;
+  char *job_originating_user_name = NULL;
+
+  puts("Job ID  Owner             Title");
+  puts("------  ----------------  ---------------------------------");
+
+  for (attr = response->attrs; attr != NULL; attr = attr->next)
+  {
+   /* Attributes without names are separators between jobs */
+    if (attr->name == NULL)
+    {
+      if (job_id > 0 && job_name != NULL && job_originating_user_name != NULL)
+        printf("%5d  %-16s  %s\n", job_id, job_originating_user_name, job_name);
+
+      job_id = 0;
+      job_name = NULL;
+      job_originating_user_name = NULL;
+      continue;
+    }
+    else if (!strcmp(attr->name, "job-id") && attr->value_tag == IPP_TAG_INTEGER)
+      job_id = attr->values[0].integer;
+    else if (!strcmp(attr->name, "job-name") && attr->value_tag == IPP_TAG_NAME)
+      job_name = attr->values[0].string.text;
+    else if (!strcmp(attr->name, "job-originating-user-name") &&
+             attr->value_tag == IPP_TAG_NAME)
+      job_originating_user_name = attr->values[0].string.text;
+  }
+
+  if (job_id > 0 && job_name != NULL && job_originating_user_name != NULL)
+    printf("%5d  %-16s  %s\n", job_id, job_originating_user_name, job_name);
+}
+
+ +

Creating URI Strings

+ +

To ensure proper encoding, the +httpAssembleURIf function must be +used to format a "printer-uri" string for all printer-based requests:

+ +
+const char *name = "Foo";
+char uri[1024];
+ipp_t *request;
+
+httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, cupsServer(),
+                 ippPort(), "/printers/%s", name);
+ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", NULL, uri);
+
+ +

Sending Requests with Files

+ +

The cupsDoFileRequest and +cupsDoIORequest functions are +used for requests involving files. The +cupsDoFileRequest function +attaches the named file to a request and is typically used when sending a print +file or changing a printer's PPD file:

+ +
+const char *filename = "/usr/share/cups/data/testprint.ps";
+const char *name = "Foo";
+char uri[1024];
+char resource[1024];
+ipp_t *request = ippNewRequest(IPP_PRINT_JOB);
+ipp_t *response;
+
+/* Use httpAssembleURIf for the printer-uri string */
+httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, cupsServer(),
+                 ippPort(), "/printers/%s", name);
+ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", NULL, uri);
+ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name",
+             NULL, cupsUser());
+ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "job-name",
+             NULL, "testprint.ps");
+
+/* Use snprintf for the resource path */
+snprintf(resource, sizeof(resource), "/printers/%s", name);
+
+response = cupsDoFileRequest(CUPS_HTTP_DEFAULT, request, resource, filename);
+
+ +

The cupsDoIORequest function +optionally attaches a file to the request and optionally saves a file in the +response from the server. It is used when using a pipe for the request +attachment or when using a request that returns a file, currently only +CUPS_GET_DOCUMENT and CUPS_GET_PPD. For example, +the following code will download the PPD file for the sample HP LaserJet +printer driver:

+ +
+char tempfile[1024];
+int tempfd;
+ipp_t *request = ippNewRequest(CUPS_GET_PPD);
+ipp_t *response;
+
+ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "ppd-name",
+             NULL, "laserjet.ppd");
+
+tempfd = cupsTempFd(tempfile, sizeof(tempfile));
+
+response = cupsDoIORequest(CUPS_HTTP_DEFAULT, request, "/", -1, tempfd);
+
+ +

The example passes -1 for the input file descriptor to specify +that no file is to be attached to the request. The PPD file attached to the +response is written to the temporary file descriptor we created using the +cupsTempFd function.

+ +

Asynchronous Request Processing

+ +

The cupsSendRequest and +cupsGetResponse support +asynchronous communications with the server. Unlike the other request +functions, the IPP request is not automatically freed, so remember to +free your request with the ippDelete +function.

+ +

File data is attached to the request using the +cupsWriteRequestData +function, while file data returned from the server is read using the +cupsReadResponseData +function. We can rewrite the previous CUPS_GET_PPD example +to use the asynchronous functions quite easily:

+ +
+char tempfile[1024];
+int tempfd;
+ipp_t *request = ippNewRequest(CUPS_GET_PPD);
+ipp_t *response;
+
+ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "ppd-name",
+             NULL, "laserjet.ppd");
+
+tempfd = cupsTempFd(tempfile, sizeof(tempfile));
+
+if (cupsSendRequest(CUPS_HTTP_DEFAULT, request, "/") == HTTP_CONTINUE)
+{
+  response = cupsGetResponse(CUPS_HTTP_DEFAULT, "/");
+
+  if (response != NULL)
+  {
+    ssize_t bytes;
+    char buffer[8192];
+
+    while ((bytes = cupsReadResponseData(CUPS_HTTP_DEFAULT, buffer, sizeof(buffer))) > 0)
+      write(tempfd, buffer, bytes);
+  }
+}
+
+/* Free the request! */
+ippDelete(request);
+
+ +

The cupsSendRequest function +returns the initial HTTP request status, typically either +HTTP_CONTINUE or HTTP_UNAUTHORIZED. The latter status +is returned when the request requires authentication of some sort. The +cupsDoAuthentication function +must be called when your see HTTP_UNAUTHORIZED and the request +re-sent. We can add authentication support to our example code by using a +do ... while loop:

+ +
+char tempfile[1024];
+int tempfd;
+ipp_t *request = ippNewRequest(CUPS_GET_PPD);
+ipp_t *response;
+http_status_t status;
+
+ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "ppd-name",
+             NULL, "laserjet.ppd");
+
+tempfd = cupsTempFd(tempfile, sizeof(tempfile));
+
+/* Loop for authentication */
+do
+{
+  status = cupsSendRequest(CUPS_HTTP_DEFAULT, request, "/");
+
+  if (status == HTTP_UNAUTHORIZED)
+  {
+    /* Try to authenticate, break out of the loop if that fails */
+    if (cupsDoAuthentication(CUPS_HTTP_DEFAULT, "POST", "/"))
+      break;
+  }
+}
+while (status != HTTP_CONTINUE && status != HTTP_UNAUTHORIZED);
+
+if (status == HTTP_CONTINUE)
+{
+  response = cupsGetResponse(CUPS_HTTP_DEFAULT, "/");
+
+  if (response != NULL)
+  {
+    ssize_t bytes;
+    char buffer[8192];
+
+    while ((bytes = cupsReadResponseData(CUPS_HTTP_DEFAULT, buffer, sizeof(buffer))) > 0)
+      write(tempfd, buffer, bytes);
+  }
+}
+
+/* Free the request! */
+ippDelete(request);
+
+

Functions

+

 CUPS 1.1.20/Mac OS X 10.4 cupsDoAuthentication

+

Authenticate a request.

+

+int cupsDoAuthentication (
+    http_t *http,
+    const char *method,
+    const char *resource
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
method
+
Request method ("GET", "POST", "PUT")
+
resource
+
Resource path
+
+

Return Value

+

0 on success, -1 on error

+

Discussion

+

This function should be called in response to a HTTP_UNAUTHORIZED +status, prior to resubmitting your request. + +

+

cupsDoFileRequest

+

Do an IPP request with a file.

+

+ipp_t *cupsDoFileRequest (
+    http_t *http,
+    ipp_t *request,
+    const char *resource,
+    const char *filename
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
request
+
IPP request
+
resource
+
HTTP resource for POST
+
filename
+
File to send or NULL for none
+
+

Return Value

+

Response data

+

Discussion

+

This function sends the IPP request to the specified server, retrying +and authenticating as necessary. The request is freed with ippDelete +after receiving a valid IPP response.

+

 CUPS 1.3/Mac OS X 10.5 cupsDoIORequest

+

Do an IPP request with file descriptors.

+

+ipp_t *cupsDoIORequest (
+    http_t *http,
+    ipp_t *request,
+    const char *resource,
+    int infile,
+    int outfile
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
request
+
IPP request
+
resource
+
HTTP resource for POST
+
infile
+
File to read from or -1 for none
+
outfile
+
File to write to or -1 for none
+
+

Return Value

+

Response data

+

Discussion

+

This function sends the IPP request to the specified server, retrying +and authenticating as necessary. The request is freed with ippDelete() +after receiving a valid IPP response.
+
+If "infile" is a valid file descriptor, cupsDoIORequest() copies +all of the data from the file after the IPP request message.
+
+If "outfile" is a valid file descriptor, cupsDoIORequest() copies +all of the data after the IPP response message to the file. + +

+

cupsDoRequest

+

Do an IPP request.

+

+ipp_t *cupsDoRequest (
+    http_t *http,
+    ipp_t *request,
+    const char *resource
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
request
+
IPP request
+
resource
+
HTTP resource for POST
+
+

Return Value

+

Response data

+

Discussion

+

This function sends the IPP request to the specified server, retrying +and authenticating as necessary. The request is freed with ippDelete() +after receiving a valid IPP response.

+

cupsEncodeOptions

+

Encode printer options into IPP attributes.

+

+void cupsEncodeOptions (
+    ipp_t *ipp,
+    int num_options,
+    cups_option_t *options
+);

+

Parameters

+
+
ipp
+
Request to add to
+
num_options
+
Number of options
+
options
+
Options
+
+

Discussion

+

This function adds operation, job, and then subscription attributes, +in that order. Use the cupsEncodeOptions2() function to add attributes +for a single group.

+

 CUPS 1.2/Mac OS X 10.5 cupsEncodeOptions2

+

Encode printer options into IPP attributes for a group.

+

+void cupsEncodeOptions2 (
+    ipp_t *ipp,
+    int num_options,
+    cups_option_t *options,
+    ipp_tag_t group_tag
+);

+

Parameters

+
+
ipp
+
Request to add to
+
num_options
+
Number of options
+
options
+
Options
+
group_tag
+
Group to encode
+
+

Discussion

+

This function only adds attributes for a single group. Call this +function multiple times for each group, or use cupsEncodeOptions() +to add the standard groups. + +

+

 CUPS 1.4/Mac OS X 10.6 cupsGetDevices

+

Get available printer devices.

+

+ipp_status_t cupsGetDevices (
+    http_t *http,
+    int timeout,
+    const char *include_schemes,
+    const char *exclude_schemes,
+    cups_device_cb_t callback,
+    void *user_data
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
timeout
+
Timeout in seconds or CUPS_TIMEOUT_DEFAULT
+
include_schemes
+
Comma-separated URI schemes to include or CUPS_INCLUDE_ALL
+
exclude_schemes
+
Comma-separated URI schemes to exclude or CUPS_EXCLUDE_NONE
+
callback
+
Callback function
+
user_data
+
User data pointer
+
+

Return Value

+

Request status - IPP_OK on success.

+

Discussion

+

This function sends a CUPS-Get-Devices request and streams the discovered +devices to the specified callback function. The "timeout" parameter controls +how long the request lasts, while the "include_schemes" and "exclude_schemes" +parameters provide comma-delimited lists of backends to include or omit from +the request respectively. + +

+

 CUPS 1.1.20/Mac OS X 10.4 cupsGetFd

+

Get a file from the server.

+

+http_status_t cupsGetFd (
+    http_t *http,
+    const char *resource,
+    int fd
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
resource
+
Resource name
+
fd
+
File descriptor
+
+

Return Value

+

HTTP status

+

Discussion

+

This function returns HTTP_OK when the file is successfully retrieved. + +

+

 CUPS 1.1.20/Mac OS X 10.4 cupsGetFile

+

Get a file from the server.

+

+http_status_t cupsGetFile (
+    http_t *http,
+    const char *resource,
+    const char *filename
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
resource
+
Resource name
+
filename
+
Filename
+
+

Return Value

+

HTTP status

+

Discussion

+

This function returns HTTP_OK when the file is successfully retrieved. + +

+

 CUPS 1.4/Mac OS X 10.6 cupsGetResponse

+

Get a response to an IPP request.

+

+ipp_t *cupsGetResponse (
+    http_t *http,
+    const char *resource
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
resource
+
HTTP resource for POST
+
+

Return Value

+

Response or NULL on HTTP error

+

Discussion

+

Use this function to get the response for an IPP request sent using +cupsSendDocument() or cupsSendRequest(). For requests that return +additional data, use httpRead() after getting a successful response, +otherwise call httpFlush() to complete the response processing. + +

+

cupsLastError

+

Return the last IPP status code.

+

+ipp_status_t cupsLastError (void);

+

Return Value

+

IPP status code from last request

+

 CUPS 1.2/Mac OS X 10.5 cupsLastErrorString

+

Return the last IPP status-message.

+

+const char *cupsLastErrorString (void);

+

Return Value

+

status-message text from last request

+

 CUPS 1.1.20/Mac OS X 10.4 cupsPutFd

+

Put a file on the server.

+

+http_status_t cupsPutFd (
+    http_t *http,
+    const char *resource,
+    int fd
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
resource
+
Resource name
+
fd
+
File descriptor
+
+

Return Value

+

HTTP status

+

Discussion

+

This function returns HTTP_CREATED when the file is stored +successfully. + +

+

 CUPS 1.1.20/Mac OS X 10.4 cupsPutFile

+

Put a file on the server.

+

+http_status_t cupsPutFile (
+    http_t *http,
+    const char *resource,
+    const char *filename
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
resource
+
Resource name
+
filename
+
Filename
+
+

Return Value

+

HTTP status

+

Discussion

+

This function returns HTTP_CREATED when the file is stored +successfully. + +

+

 CUPS 1.4/Mac OS X 10.6 cupsReadResponseData

+

Read additional data after the IPP response.

+

+ssize_t cupsReadResponseData (
+    http_t *http,
+    char *buffer,
+    size_t length
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
buffer
+
Buffer to use
+
length
+
Number of bytes to read
+
+

Return Value

+

Bytes read, 0 on EOF, -1 on error

+

Discussion

+

This function is used after cupsGetResponse() to read the PPD or document +files for CUPS_GET_PPD and CUPS_GET_DOCUMENT requests, respectively. + +

+

 CUPS 1.4/Mac OS X 10.6 cupsSendRequest

+

Send an IPP request.

+

+http_status_t cupsSendRequest (
+    http_t *http,
+    ipp_t *request,
+    const char *resource,
+    size_t length
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
request
+
IPP request
+
resource
+
Resource path
+
length
+
Length of data to follow or CUPS_LENGTH_VARIABLE
+
+

Return Value

+

Initial HTTP status

+

Discussion

+

Use httpWrite() to write any additional data (document, PPD file, etc.) +for the request, cupsGetResponse() to get the IPP response, and httpRead() +to read any additional data following the response. Only one request can be +sent/queued at a time.
+
+Unlike cupsDoFileRequest(), cupsDoIORequest(), and cupsDoRequest(), the +request is not freed. + +

+

 CUPS 1.4/Mac OS X 10.6 cupsWriteRequestData

+

Write additional data after an IPP request.

+

+http_status_t cupsWriteRequestData (
+    http_t *http,
+    const char *buffer,
+    size_t length
+);

+

Parameters

+
+
http
+
Connection to server or CUPS_HTTP_DEFAULT
+
buffer
+
Bytes to write
+
length
+
Number of bytes to write
+
+

Return Value

+

HTTP_CONTINUE if OK or HTTP status on error

+

Discussion

+

This function is used after cupsSendRequest to provide a PPD and +after cupsStartDocument to provide a document file. + +

+

 CUPS 1.5/Mac OS X 10.7 httpAddCredential

+

Allocates and adds a single credential to an array.

+

+int httpAddCredential (
+    cups_array_t *credentials,
+    const void *data,
+    size_t datalen
+);

+

Parameters

+
+
credentials
+
Credentials array
+
data
+
PEM-encoded X.509 data
+
datalen
+
Length of data
+
+

Return Value

+

0 on success, -1 on error

+

Discussion

+

Use cupsArrayNew(NULL, NULL) to create a credentials array. + +

+

 CUPS 1.2/Mac OS X 10.5 httpAddrAny

+

Check for the "any" address.

+

+int httpAddrAny (
+    const http_addr_t *addr
+);

+

Parameters

+
+
addr
+
Address to check
+
+

Return Value

+

1 if "any", 0 otherwise

+

 CUPS 1.2/Mac OS X 10.5 httpAddrEqual

+

Compare two addresses.

+

+int httpAddrEqual (
+    const http_addr_t *addr1,
+    const http_addr_t *addr2
+);

+

Parameters

+
+
addr1
+
First address
+
addr2
+
Second address
+
+

Return Value

+

1 if equal, 0 if not

+

 CUPS 1.2/Mac OS X 10.5 httpAddrLength

+

Return the length of the address in bytes.

+

+int httpAddrLength (
+    const http_addr_t *addr
+);

+

Parameters

+
+
addr
+
Address
+
+

Return Value

+

Length in bytes

+

 CUPS 1.2/Mac OS X 10.5 httpAddrLocalhost

+

Check for the local loopback address.

+

+int httpAddrLocalhost (
+    const http_addr_t *addr
+);

+

Parameters

+
+
addr
+
Address to check
+
+

Return Value

+

1 if local host, 0 otherwise

+

 CUPS 1.2/Mac OS X 10.5 httpAddrLookup

+

Lookup the hostname associated with the address.

+

+char *httpAddrLookup (
+    const http_addr_t *addr,
+    char *name,
+    int namelen
+);

+

Parameters

+
+
addr
+
Address to lookup
+
name
+
Host name buffer
+
namelen
+
Size of name buffer
+
+

Return Value

+

Host name

+

 CUPS 1.2/Mac OS X 10.5 httpAddrString

+

Convert an address to a numeric string.

+

+char *httpAddrString (
+    const http_addr_t *addr,
+    char *s,
+    int slen
+);

+

Parameters

+
+
addr
+
Address to convert
+
s
+
String buffer
+
slen
+
Length of string
+
+

Return Value

+

Numeric address string

+

 CUPS 1.2/Mac OS X 10.5 httpAssembleURI

+

Assemble a uniform resource identifier from its +components.

+

+http_uri_status_t httpAssembleURI (
+    http_uri_coding_t encoding,
+    char *uri,
+    int urilen,
+    const char *scheme,
+    const char *username,
+    const char *host,
+    int port,
+    const char *resource
+);

+

Parameters

+
+
encoding
+
Encoding flags
+
uri
+
URI buffer
+
urilen
+
Size of URI buffer
+
scheme
+
Scheme name
+
username
+
Username
+
host
+
Hostname or address
+
port
+
Port number
+
resource
+
Resource
+
+

Return Value

+

URI status

+

Discussion

+

This function escapes reserved characters in the URI depending on the +value of the "encoding" argument. You should use this function in +place of traditional string functions whenever you need to create a +URI string. + +

+

 CUPS 1.2/Mac OS X 10.5 httpAssembleURIf

+

Assemble a uniform resource identifier from its +components with a formatted resource.

+

+http_uri_status_t httpAssembleURIf (
+    http_uri_coding_t encoding,
+    char *uri,
+    int urilen,
+    const char *scheme,
+    const char *username,
+    const char *host,
+    int port,
+    const char *resourcef,
+    ...
+);

+

Parameters

+
+
encoding
+
Encoding flags
+
uri
+
URI buffer
+
urilen
+
Size of URI buffer
+
scheme
+
Scheme name
+
username
+
Username
+
host
+
Hostname or address
+
port
+
Port number
+
resourcef
+
Printf-style resource
+
...
+
Additional arguments as needed
+
+

Return Value

+

URI status

+

Discussion

+

This function creates a formatted version of the resource string +argument "resourcef" and escapes reserved characters in the URI +depending on the value of the "encoding" argument. You should use +this function in place of traditional string functions whenever +you need to create a URI string. + +

+

httpBlocking

+

Set blocking/non-blocking behavior on a connection.

+

+void httpBlocking (
+    http_t *http,
+    int b
+);

+

Parameters

+
+
http
+
Connection to server
+
b
+
1 = blocking, 0 = non-blocking
+
+

httpCheck

+

Check to see if there is a pending response from the server.

+

+int httpCheck (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server
+
+

Return Value

+

0 = no data, 1 = data available

+

 CUPS 1.1.19/Mac OS X 10.3 httpClearCookie

+

Clear the cookie value(s).

+

+void httpClearCookie (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server
+
+

httpClearFields

+

Clear HTTP request fields.

+

+void httpClearFields (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server
+
+

httpClose

+

Close an HTTP connection.

+

+void httpClose (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server
+
+

 DEPRECATED httpConnect

+

Connect to a HTTP server.

+

+http_t *httpConnect (
+    const char *host,
+    int port
+);

+

Parameters

+
+
host
+
Host to connect to
+
port
+
Port number
+
+

Return Value

+

New HTTP connection

+

Discussion

+

This function is deprecated - use httpConnectEncrypt instead. + +

+

httpConnectEncrypt

+

Connect to a HTTP server using encryption.

+

+http_t *httpConnectEncrypt (
+    const char *host,
+    int port,
+    http_encryption_t encryption
+);

+

Parameters

+
+
host
+
Host to connect to
+
port
+
Port number
+
encryption
+
Type of encryption to use
+
+

Return Value

+

New HTTP connection

+

 CUPS 1.5/Mac OS X 10.7 httpCopyCredentials

+

Copy the credentials associated with an encrypted +connection.

+

+int httpCopyCredentials (
+    http_t *http,
+    cups_array_t **credentials
+);

+

Parameters

+
+
http
+
Connection to server
+
credentials
+
Array of credentials
+
+

Return Value

+

Status of call (0 = success)

+

 DEPRECATED httpDecode64

+

Base64-decode a string.

+

+char *httpDecode64 (
+    char *out,
+    const char *in
+);

+

Parameters

+
+
out
+
String to write to
+
in
+
String to read from
+
+

Return Value

+

Decoded string

+

Discussion

+

This function is deprecated. Use the httpDecode64_2() function instead +which provides buffer length arguments. + +

+

 CUPS 1.1.21/Mac OS X 10.4 httpDecode64_2

+

Base64-decode a string.

+

+char *httpDecode64_2 (
+    char *out,
+    int *outlen,
+    const char *in
+);

+

Parameters

+
+
out
+
String to write to
+
outlen
+
Size of output string
+
in
+
String to read from
+
+

Return Value

+

Decoded string

+

httpDelete

+

Send a DELETE request to the server.

+

+int httpDelete (
+    http_t *http,
+    const char *uri
+);

+

Parameters

+
+
http
+
Connection to server
+
uri
+
URI to delete
+
+

Return Value

+

Status of call (0 = success)

+

 DEPRECATED httpEncode64

+

Base64-encode a string.

+

+char *httpEncode64 (
+    char *out,
+    const char *in
+);

+

Parameters

+
+
out
+
String to write to
+
in
+
String to read from
+
+

Return Value

+

Encoded string

+

Discussion

+

This function is deprecated. Use the httpEncode64_2() function instead +which provides buffer length arguments. + +

+

 CUPS 1.1.21/Mac OS X 10.4 httpEncode64_2

+

Base64-encode a string.

+

+char *httpEncode64_2 (
+    char *out,
+    int outlen,
+    const char *in,
+    int inlen
+);

+

Parameters

+
+
out
+
String to write to
+
outlen
+
Size of output string
+
in
+
String to read from
+
inlen
+
Size of input string
+
+

Return Value

+

Encoded string

+

httpEncryption

+

Set the required encryption on the link.

+

+int httpEncryption (
+    http_t *http,
+    http_encryption_t e
+);

+

Parameters

+
+
http
+
Connection to server
+
e
+
New encryption preference
+
+

Return Value

+

-1 on error, 0 on success

+

httpError

+

Get the last error on a connection.

+

+int httpError (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server
+
+

Return Value

+

Error code (errno) value

+

httpFlush

+

Flush data from a HTTP connection.

+

+void httpFlush (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server
+
+

 CUPS 1.2/Mac OS X 10.5 httpFlushWrite

+

Flush data in write buffer.

+

+int httpFlushWrite (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server
+
+

Return Value

+

Bytes written or -1 on error

+

httpFreeCredentials

+

Free an array of credentials.

+

+void httpFreeCredentials (
+    cups_array_t *credentials
+);

+

Parameters

+
+
credentials
+
Array of credentials
+
+

httpGet

+

Send a GET request to the server.

+

+int httpGet (
+    http_t *http,
+    const char *uri
+);

+

Parameters

+
+
http
+
Connection to server
+
uri
+
URI to get
+
+

Return Value

+

Status of call (0 = success)

+

 CUPS 1.3/Mac OS X 10.5 httpGetAuthString

+

Get the current authorization string.

+

+char *httpGetAuthString (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server
+
+

Return Value

+

Authorization string

+

Discussion

+

The authorization string is set by cupsDoAuthentication() and +httpSetAuthString(). Use httpGetAuthString() to retrieve the +string to use with httpSetField() for the HTTP_FIELD_AUTHORIZATION +value. + +

+

 CUPS 1.2/Mac OS X 10.5 httpGetBlocking

+

Get the blocking/non-block state of a connection.

+

+int httpGetBlocking (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server
+
+

Return Value

+

1 if blocking, 0 if non-blocking

+

 CUPS 1.1.19/Mac OS X 10.3 httpGetCookie

+

Get any cookie data from the response.

+

+const char *httpGetCookie (
+    http_t *http
+);

+

Parameters

+
+
http
+
HTTP connecion
+
+

Return Value

+

Cookie data or NULL

+

 DEPRECATED httpGetDateString

+

Get a formatted date/time string from a time value.

+

+const char *httpGetDateString (
+    time_t t
+);

+

Parameters

+
+
t
+
UNIX time
+
+

Return Value

+

Date/time string

+

 CUPS 1.2/Mac OS X 10.5 httpGetDateString2

+

Get a formatted date/time string from a time value.

+

+const char *httpGetDateString2 (
+    time_t t,
+    char *s,
+    int slen
+);

+

Parameters

+
+
t
+
UNIX time
+
s
+
String buffer
+
slen
+
Size of string buffer
+
+

Return Value

+

Date/time string

+

httpGetDateTime

+

Get a time value from a formatted date/time string.

+

+time_t httpGetDateTime (
+    const char *s
+);

+

Parameters

+
+
s
+
Date/time string
+
+

Return Value

+

UNIX time

+

 CUPS 1.2/Mac OS X 10.5 httpGetFd

+

Get the file descriptor associated with a connection.

+

+int httpGetFd (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server
+
+

Return Value

+

File descriptor or -1 if none

+

httpGetField

+

Get a field value from a request/response.

+

+const char *httpGetField (
+    http_t *http,
+    http_field_t field
+);

+

Parameters

+
+
http
+
Connection to server
+
field
+
Field to get
+
+

Return Value

+

Field value

+

 DEPRECATED httpGetHostByName

+

Lookup a hostname or IPv4 address, and return +address records for the specified name.

+

+struct hostent *httpGetHostByName (
+    const char *name
+);

+

Parameters

+
+
name
+
Hostname or IP address
+
+

Return Value

+

Host entry

+

 CUPS 1.2/Mac OS X 10.5 httpGetHostname

+

Get the FQDN for the connection or local system.

+

+const char *httpGetHostname (
+    http_t *http,
+    char *s,
+    int slen
+);

+

Parameters

+
+
http
+
HTTP connection or NULL
+
s
+
String buffer for name
+
slen
+
Size of buffer
+
+

Return Value

+

FQDN for connection or system

+

Discussion

+

When "http" points to a connected socket, return the hostname or +address that was used in the call to httpConnect() or httpConnectEncrypt(). +Otherwise, return the FQDN for the local system using both gethostname() +and gethostbyname() to get the local hostname with domain. + +

+

 DEPRECATED httpGetLength

+

Get the amount of data remaining from the +content-length or transfer-encoding fields.

+

+int httpGetLength (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server
+
+

Return Value

+

Content length

+

Discussion

+

This function is deprecated and will not return lengths larger than +2^31 - 1; use httpGetLength2() instead. + +

+

 CUPS 1.2/Mac OS X 10.5 httpGetLength2

+

Get the amount of data remaining from the +content-length or transfer-encoding fields.

+

+off_t httpGetLength2 (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server
+
+

Return Value

+

Content length

+

Discussion

+

This function returns the complete content length, even for +content larger than 2^31 - 1. + +

+

httpGetState

+

Get the current state of the HTTP request.

+

+http_state_t httpGetState (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server
+
+

Return Value

+

HTTP state

+

 CUPS 1.2/Mac OS X 10.5 httpGetStatus

+

Get the status of the last HTTP request.

+

+http_status_t httpGetStatus (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server
+
+

Return Value

+

HTTP status

+

 DEPRECATED httpGetSubField

+

Get a sub-field value.

+

+char *httpGetSubField (
+    http_t *http,
+    http_field_t field,
+    const char *name,
+    char *value
+);

+

Parameters

+
+
http
+
Connection to server
+
field
+
Field index
+
name
+
Name of sub-field
+
value
+
Value string
+
+

Return Value

+

Value or NULL

+

 CUPS 1.2/Mac OS X 10.5 httpGetSubField2

+

Get a sub-field value.

+

+char *httpGetSubField2 (
+    http_t *http,
+    http_field_t field,
+    const char *name,
+    char *value,
+    int valuelen
+);

+

Parameters

+
+
http
+
Connection to server
+
field
+
Field index
+
name
+
Name of sub-field
+
value
+
Value string
+
valuelen
+
Size of value buffer
+
+

Return Value

+

Value or NULL

+

httpGetVersion

+

Get the HTTP version at the other end.

+

+http_version_t httpGetVersion (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server
+
+

Return Value

+

Version number

+

httpGets

+

Get a line of text from a HTTP connection.

+

+char *httpGets (
+    char *line,
+    int length,
+    http_t *http
+);

+

Parameters

+
+
line
+
Line to read into
+
length
+
Max length of buffer
+
http
+
Connection to server
+
+

Return Value

+

Line or NULL

+

httpHead

+

Send a HEAD request to the server.

+

+int httpHead (
+    http_t *http,
+    const char *uri
+);

+

Parameters

+
+
http
+
Connection to server
+
uri
+
URI for head
+
+

Return Value

+

Status of call (0 = success)

+

httpInitialize

+

Initialize the HTTP interface library and set the +default HTTP proxy (if any).

+

+void httpInitialize (void);

+

httpMD5

+

Compute the MD5 sum of the username:group:password.

+

+char *httpMD5 (
+    const char *username,
+    const char *realm,
+    const char *passwd,
+    char md5[33]
+);

+

Parameters

+
+
username
+
User name
+
realm
+
Realm name
+
passwd
+
Password string
+
md5[33]
+
MD5 string
+
+

Return Value

+

MD5 sum

+

httpMD5Final

+

Combine the MD5 sum of the username, group, and password +with the server-supplied nonce value, method, and +request-uri.

+

+char *httpMD5Final (
+    const char *nonce,
+    const char *method,
+    const char *resource,
+    char md5[33]
+);

+

Parameters

+
+
nonce
+
Server nonce value
+
method
+
METHOD (GET, POST, etc.)
+
resource
+
Resource path
+
md5[33]
+
MD5 sum
+
+

Return Value

+

New sum

+

httpMD5String

+

Convert an MD5 sum to a character string.

+

+char *httpMD5String (
+    const unsigned char *sum,
+    char md5[33]
+);

+

Parameters

+
+
sum
+
MD5 sum data
+
md5[33]
+
MD5 sum in hex
+
+

Return Value

+

MD5 sum in hex

+

httpOptions

+

Send an OPTIONS request to the server.

+

+int httpOptions (
+    http_t *http,
+    const char *uri
+);

+

Parameters

+
+
http
+
Connection to server
+
uri
+
URI for options
+
+

Return Value

+

Status of call (0 = success)

+

httpPost

+

Send a POST request to the server.

+

+int httpPost (
+    http_t *http,
+    const char *uri
+);

+

Parameters

+
+
http
+
Connection to server
+
uri
+
URI for post
+
+

Return Value

+

Status of call (0 = success)

+

httpPut

+

Send a PUT request to the server.

+

+int httpPut (
+    http_t *http,
+    const char *uri
+);

+

Parameters

+
+
http
+
Connection to server
+
uri
+
URI to put
+
+

Return Value

+

Status of call (0 = success)

+

 DEPRECATED httpRead

+

Read data from a HTTP connection.

+

+int httpRead (
+    http_t *http,
+    char *buffer,
+    int length
+);

+

Parameters

+
+
http
+
Connection to server
+
buffer
+
Buffer for data
+
length
+
Maximum number of bytes
+
+

Return Value

+

Number of bytes read

+

Discussion

+

This function is deprecated. Use the httpRead2() function which can +read more than 2GB of data. + +

+

 CUPS 1.2/Mac OS X 10.5 httpRead2

+

Read data from a HTTP connection.

+

+ssize_t httpRead2 (
+    http_t *http,
+    char *buffer,
+    size_t length
+);

+

Parameters

+
+
http
+
Connection to server
+
buffer
+
Buffer for data
+
length
+
Maximum number of bytes
+
+

Return Value

+

Number of bytes read

+

httpReconnect

+

Reconnect to a HTTP server.

+

+int httpReconnect (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server
+
+

Return Value

+

0 on success, non-zero on failure

+

 DEPRECATED httpSeparate

+

Separate a Universal Resource Identifier into its +components.

+

+void httpSeparate (
+    const char *uri,
+    char *scheme,
+    char *username,
+    char *host,
+    int *port,
+    char *resource
+);

+

Parameters

+
+
uri
+
Universal Resource Identifier
+
scheme
+
Scheme [32] (http, https, etc.)
+
username
+
Username [1024]
+
host
+
Hostname [1024]
+
port
+
Port number to use
+
resource
+
Resource/filename [1024]
+
+

Discussion

+

This function is deprecated; use the httpSeparateURI() function instead. + +

+

 CUPS 1.1.21/Mac OS X 10.4 httpSeparate2

+

Separate a Universal Resource Identifier into its +components.

+

+void httpSeparate2 (
+    const char *uri,
+    char *scheme,
+    int schemelen,
+    char *username,
+    int usernamelen,
+    char *host,
+    int hostlen,
+    int *port,
+    char *resource,
+    int resourcelen
+);

+

Parameters

+
+
uri
+
Universal Resource Identifier
+
scheme
+
Scheme (http, https, etc.)
+
schemelen
+
Size of scheme buffer
+
username
+
Username
+
usernamelen
+
Size of username buffer
+
host
+
Hostname
+
hostlen
+
Size of hostname buffer
+
port
+
Port number to use
+
resource
+
Resource/filename
+
resourcelen
+
Size of resource buffer
+
+

Discussion

+

This function is deprecated; use the httpSeparateURI() function instead. + + +

+

 CUPS 1.2/Mac OS X 10.5 httpSeparateURI

+

Separate a Universal Resource Identifier into its +components.

+

+http_uri_status_t httpSeparateURI (
+    http_uri_coding_t decoding,
+    const char *uri,
+    char *scheme,
+    int schemelen,
+    char *username,
+    int usernamelen,
+    char *host,
+    int hostlen,
+    int *port,
+    char *resource,
+    int resourcelen
+);

+

Parameters

+
+
decoding
+
Decoding flags
+
uri
+
Universal Resource Identifier
+
scheme
+
Scheme (http, https, etc.)
+
schemelen
+
Size of scheme buffer
+
username
+
Username
+
usernamelen
+
Size of username buffer
+
host
+
Hostname
+
hostlen
+
Size of hostname buffer
+
port
+
Port number to use
+
resource
+
Resource/filename
+
resourcelen
+
Size of resource buffer
+
+

Return Value

+

Result of separation

+

 CUPS 1.3/Mac OS X 10.5 httpSetAuthString

+

Set the current authorization string.

+

+void httpSetAuthString (
+    http_t *http,
+    const char *scheme,
+    const char *data
+);

+

Parameters

+
+
http
+
Connection to server
+
scheme
+
Auth scheme (NULL to clear it)
+
data
+
Auth data (NULL for none)
+
+

Discussion

+

This function just stores a copy of the current authorization string in +the HTTP connection object. You must still call httpSetField() to set +HTTP_FIELD_AUTHORIZATION prior to issuing a HTTP request using httpGet(), +httpHead(), httpOptions(), httpPost, or httpPut(). + +

+

 CUPS 1.1.19/Mac OS X 10.3 httpSetCookie

+

Set the cookie value(s).

+

+void httpSetCookie (
+    http_t *http,
+    const char *cookie
+);

+

Parameters

+
+
http
+
Connection
+
cookie
+
Cookie string
+
+

 CUPS 1.5/Mac OS X 10.7 httpSetCredentials

+

Set the credentials associated with an encrypted +connection.

+

+int httpSetCredentials (
+    http_t *http,
+    cups_array_t *credentials
+);

+

Parameters

+
+
http
+
Connection to server
+
credentials
+
Array of credentials
+
+

Return Value

+

Status of call (0 = success)

+

 CUPS 1.2/Mac OS X 10.5 httpSetExpect

+

Set the Expect: header in a request.

+

+void httpSetExpect (
+    http_t *http,
+    http_status_t expect
+);

+

Parameters

+
+
http
+
Connection to server
+
expect
+
HTTP status to expect (HTTP_CONTINUE)
+
+

Discussion

+

Currently only HTTP_CONTINUE is supported for the "expect" argument. + +

+

httpSetField

+

Set the value of an HTTP header.

+

+void httpSetField (
+    http_t *http,
+    http_field_t field,
+    const char *value
+);

+

Parameters

+
+
http
+
Connection to server
+
field
+
Field index
+
value
+
Value
+
+

 CUPS 1.2/Mac OS X 10.5 httpSetLength

+

Set the content-length and content-encoding.

+

+void httpSetLength (
+    http_t *http,
+    size_t length
+);

+

Parameters

+
+
http
+
Connection to server
+
length
+
Length (0 for chunked)
+
+

 CUPS 1.5/Mac OS X 10.7 httpSetTimeout

+

Set read/write timeouts and an optional callback.

+

+void httpSetTimeout (
+    http_t *http,
+    double timeout,
+    http_timeout_cb_t cb,
+    void *user_data
+);

+

Parameters

+
+
http
+
Connection to server
+
timeout
+
Number of seconds for timeout, +must be greater than 0
+
cb
+
Callback function or NULL
+
user_data
+
User data pointer
+
+

Discussion

+

The optional timeout callback receives both the HTTP connection and a user +data pointer and must return 1 to continue or 0 to error (time) out. + +

+

httpStatus

+

Return a short string describing a HTTP status code.

+

+const char *httpStatus (
+    http_status_t status
+);

+

Parameters

+
+
status
+
HTTP status code
+
+

Return Value

+

Localized status string

+

Discussion

+

The returned string is localized to the current POSIX locale and is based +on the status strings defined in RFC 2616.

+

httpTrace

+

Send an TRACE request to the server.

+

+int httpTrace (
+    http_t *http,
+    const char *uri
+);

+

Parameters

+
+
http
+
Connection to server
+
uri
+
URI for trace
+
+

Return Value

+

Status of call (0 = success)

+

httpUpdate

+

Update the current HTTP state for incoming data.

+

+http_status_t httpUpdate (
+    http_t *http
+);

+

Parameters

+
+
http
+
Connection to server
+
+

Return Value

+

HTTP status

+

 CUPS 1.1.19/Mac OS X 10.3 httpWait

+

Wait for data available on a connection.

+

+int httpWait (
+    http_t *http,
+    int msec
+);

+

Parameters

+
+
http
+
Connection to server
+
msec
+
Milliseconds to wait
+
+

Return Value

+

1 if data is available, 0 otherwise

+

 DEPRECATED httpWrite

+

Write data to a HTTP connection.

+

+int httpWrite (
+    http_t *http,
+    const char *buffer,
+    int length
+);

+

Parameters

+
+
http
+
Connection to server
+
buffer
+
Buffer for data
+
length
+
Number of bytes to write
+
+

Return Value

+

Number of bytes written

+

Discussion

+

This function is deprecated. Use the httpWrite2() function which can +write more than 2GB of data. + +

+

 CUPS 1.2/Mac OS X 10.5 httpWrite2

+

Write data to a HTTP connection.

+

+ssize_t httpWrite2 (
+    http_t *http,
+    const char *buffer,
+    size_t length
+);

+

Parameters

+
+
http
+
Connection to server
+
buffer
+
Buffer for data
+
length
+
Number of bytes to write
+
+

Return Value

+

Number of bytes written

+

ippAddBoolean

+

Add a boolean attribute to an IPP message.

+

+ipp_attribute_t *ippAddBoolean (
+    ipp_t *ipp,
+    ipp_tag_t group,
+    const char *name,
+    char value
+);

+

Parameters

+
+
ipp
+
IPP message
+
group
+
IPP group
+
name
+
Name of attribute
+
value
+
Value of attribute
+
+

Return Value

+

New attribute

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.
+
+The group parameter specifies the IPP attribute group tag: none +(IPP_TAG_ZERO, for member attributes), document (IPP_TAG_DOCUMENT), +event notification (IPP_TAG_EVENT_NOTIFICATION), operation +(IPP_TAG_OPERATION), printer (IPP_TAG_PRINTER), subscription +(IPP_TAG_SUBSCRIPTION), or unsupported (IPP_TAG_UNSUPPORTED_GROUP).

+

ippAddBooleans

+

Add an array of boolean values.

+

+ipp_attribute_t *ippAddBooleans (
+    ipp_t *ipp,
+    ipp_tag_t group,
+    const char *name,
+    int num_values,
+    const char *values
+);

+

Parameters

+
+
ipp
+
IPP message
+
group
+
IPP group
+
name
+
Name of attribute
+
num_values
+
Number of values
+
values
+
Values
+
+

Return Value

+

New attribute

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.
+
+The group parameter specifies the IPP attribute group tag: none +(IPP_TAG_ZERO, for member attributes), document (IPP_TAG_DOCUMENT), +event notification (IPP_TAG_EVENT_NOTIFICATION), operation +(IPP_TAG_OPERATION), printer (IPP_TAG_PRINTER), subscription +(IPP_TAG_SUBSCRIPTION), or unsupported (IPP_TAG_UNSUPPORTED_GROUP).

+

 CUPS 1.1.19/Mac OS X 10.3 ippAddCollection

+

Add a collection value.

+

+ipp_attribute_t *ippAddCollection (
+    ipp_t *ipp,
+    ipp_tag_t group,
+    const char *name,
+    ipp_t *value
+);

+

Parameters

+
+
ipp
+
IPP message
+
group
+
IPP group
+
name
+
Name of attribute
+
value
+
Value
+
+

Return Value

+

New attribute

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.
+
+The group parameter specifies the IPP attribute group tag: none +(IPP_TAG_ZERO, for member attributes), document (IPP_TAG_DOCUMENT), +event notification (IPP_TAG_EVENT_NOTIFICATION), operation +(IPP_TAG_OPERATION), printer (IPP_TAG_PRINTER), subscription +(IPP_TAG_SUBSCRIPTION), or unsupported (IPP_TAG_UNSUPPORTED_GROUP). + +

+

 CUPS 1.1.19/Mac OS X 10.3 ippAddCollections

+

Add an array of collection values.

+

+ipp_attribute_t *ippAddCollections (
+    ipp_t *ipp,
+    ipp_tag_t group,
+    const char *name,
+    int num_values,
+    const ipp_t **values
+);

+

Parameters

+
+
ipp
+
IPP message
+
group
+
IPP group
+
name
+
Name of attribute
+
num_values
+
Number of values
+
values
+
Values
+
+

Return Value

+

New attribute

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.
+
+The group parameter specifies the IPP attribute group tag: none +(IPP_TAG_ZERO, for member attributes), document (IPP_TAG_DOCUMENT), +event notification (IPP_TAG_EVENT_NOTIFICATION), operation +(IPP_TAG_OPERATION), printer (IPP_TAG_PRINTER), subscription +(IPP_TAG_SUBSCRIPTION), or unsupported (IPP_TAG_UNSUPPORTED_GROUP). + +

+

ippAddDate

+

Add a date attribute to an IPP message.

+

+ipp_attribute_t *ippAddDate (
+    ipp_t *ipp,
+    ipp_tag_t group,
+    const char *name,
+    const ipp_uchar_t *value
+);

+

Parameters

+
+
ipp
+
IPP message
+
group
+
IPP group
+
name
+
Name of attribute
+
value
+
Value
+
+

Return Value

+

New attribute

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.
+
+The group parameter specifies the IPP attribute group tag: none +(IPP_TAG_ZERO, for member attributes), document (IPP_TAG_DOCUMENT), +event notification (IPP_TAG_EVENT_NOTIFICATION), operation +(IPP_TAG_OPERATION), printer (IPP_TAG_PRINTER), subscription +(IPP_TAG_SUBSCRIPTION), or unsupported (IPP_TAG_UNSUPPORTED_GROUP).

+

ippAddInteger

+

Add a integer attribute to an IPP message.

+

+ipp_attribute_t *ippAddInteger (
+    ipp_t *ipp,
+    ipp_tag_t group,
+    ipp_tag_t value_tag,
+    const char *name,
+    int value
+);

+

Parameters

+
+
ipp
+
IPP message
+
group
+
IPP group
+
value_tag
+
Type of attribute
+
name
+
Name of attribute
+
value
+
Value of attribute
+
+

Return Value

+

New attribute

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.
+
+The group parameter specifies the IPP attribute group tag: none +(IPP_TAG_ZERO, for member attributes), document (IPP_TAG_DOCUMENT), +event notification (IPP_TAG_EVENT_NOTIFICATION), operation +(IPP_TAG_OPERATION), printer (IPP_TAG_PRINTER), subscription +(IPP_TAG_SUBSCRIPTION), or unsupported (IPP_TAG_UNSUPPORTED_GROUP).
+
+Supported values include enum (IPP_TAG_ENUM) and integer +(IPP_TAG_INTEGER).

+

ippAddIntegers

+

Add an array of integer values.

+

+ipp_attribute_t *ippAddIntegers (
+    ipp_t *ipp,
+    ipp_tag_t group,
+    ipp_tag_t value_tag,
+    const char *name,
+    int num_values,
+    const int *values
+);

+

Parameters

+
+
ipp
+
IPP message
+
group
+
IPP group
+
value_tag
+
Type of attribute
+
name
+
Name of attribute
+
num_values
+
Number of values
+
values
+
Values
+
+

Return Value

+

New attribute

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.
+
+The group parameter specifies the IPP attribute group tag: none +(IPP_TAG_ZERO, for member attributes), document (IPP_TAG_DOCUMENT), +event notification (IPP_TAG_EVENT_NOTIFICATION), operation +(IPP_TAG_OPERATION), printer (IPP_TAG_PRINTER), subscription +(IPP_TAG_SUBSCRIPTION), or unsupported (IPP_TAG_UNSUPPORTED_GROUP).
+
+Supported values include enum (IPP_TAG_ENUM) and integer +(IPP_TAG_INTEGER).

+

 CUPS 1.2/Mac OS X 10.5 ippAddOctetString

+

Add an octetString value to an IPP message.

+

+ipp_attribute_t *ippAddOctetString (
+    ipp_t *ipp,
+    ipp_tag_t group,
+    const char *name,
+    const void *data,
+    int datalen
+);

+

Parameters

+
+
ipp
+
IPP message
+
group
+
IPP group
+
name
+
Name of attribute
+
data
+
octetString data
+
datalen
+
Length of data in bytes
+
+

Return Value

+

New attribute

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.
+
+The group parameter specifies the IPP attribute group tag: none +(IPP_TAG_ZERO, for member attributes), document (IPP_TAG_DOCUMENT), +event notification (IPP_TAG_EVENT_NOTIFICATION), operation +(IPP_TAG_OPERATION), printer (IPP_TAG_PRINTER), subscription +(IPP_TAG_SUBSCRIPTION), or unsupported (IPP_TAG_UNSUPPORTED_GROUP). + +

+

 CUPS 1.6 ippAddOutOfBand

+

Add an out-of-band value to an IPP message.

+

+ipp_attribute_t *ippAddOutOfBand (
+    ipp_t *ipp,
+    ipp_tag_t group,
+    ipp_tag_t value_tag,
+    const char *name
+);

+

Parameters

+
+
ipp
+
IPP message
+
group
+
IPP group
+
value_tag
+
Type of attribute
+
name
+
Name of attribute
+
+

Return Value

+

New attribute

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.
+
+The group parameter specifies the IPP attribute group tag: none +(IPP_TAG_ZERO, for member attributes), document (IPP_TAG_DOCUMENT), +event notification (IPP_TAG_EVENT_NOTIFICATION), operation +(IPP_TAG_OPERATION), printer (IPP_TAG_PRINTER), subscription +(IPP_TAG_SUBSCRIPTION), or unsupported (IPP_TAG_UNSUPPORTED_GROUP).
+
+Supported out-of-band values include unsupported-value +(IPP_TAG_UNSUPPORTED_VALUE), default (IPP_TAG_DEFAULT), unknown +(IPP_TAG_UNKNOWN), no-value (IPP_TAG_NOVALUE), not-settable +(IPP_TAG_NOTSETTABLE), delete-attribute (IPP_TAG_DELETEATTR), and +admin-define (IPP_TAG_ADMINDEFINE). + +

+

ippAddRange

+

Add a range of values to an IPP message.

+

+ipp_attribute_t *ippAddRange (
+    ipp_t *ipp,
+    ipp_tag_t group,
+    const char *name,
+    int lower,
+    int upper
+);

+

Parameters

+
+
ipp
+
IPP message
+
group
+
IPP group
+
name
+
Name of attribute
+
lower
+
Lower value
+
upper
+
Upper value
+
+

Return Value

+

New attribute

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.
+
+The group parameter specifies the IPP attribute group tag: none +(IPP_TAG_ZERO, for member attributes), document (IPP_TAG_DOCUMENT), +event notification (IPP_TAG_EVENT_NOTIFICATION), operation +(IPP_TAG_OPERATION), printer (IPP_TAG_PRINTER), subscription +(IPP_TAG_SUBSCRIPTION), or unsupported (IPP_TAG_UNSUPPORTED_GROUP).
+
+The lower parameter must be less than or equal to the upper parameter.

+

ippAddRanges

+

Add ranges of values to an IPP message.

+

+ipp_attribute_t *ippAddRanges (
+    ipp_t *ipp,
+    ipp_tag_t group,
+    const char *name,
+    int num_values,
+    const int *lower,
+    const int *upper
+);

+

Parameters

+
+
ipp
+
IPP message
+
group
+
IPP group
+
name
+
Name of attribute
+
num_values
+
Number of values
+
lower
+
Lower values
+
upper
+
Upper values
+
+

Return Value

+

New attribute

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.
+
+The group parameter specifies the IPP attribute group tag: none +(IPP_TAG_ZERO, for member attributes), document (IPP_TAG_DOCUMENT), +event notification (IPP_TAG_EVENT_NOTIFICATION), operation +(IPP_TAG_OPERATION), printer (IPP_TAG_PRINTER), subscription +(IPP_TAG_SUBSCRIPTION), or unsupported (IPP_TAG_UNSUPPORTED_GROUP).

+

ippAddResolution

+

Add a resolution value to an IPP message.

+

+ipp_attribute_t *ippAddResolution (
+    ipp_t *ipp,
+    ipp_tag_t group,
+    const char *name,
+    ipp_res_t units,
+    int xres,
+    int yres
+);

+

Parameters

+
+
ipp
+
IPP message
+
group
+
IPP group
+
name
+
Name of attribute
+
units
+
Units for resolution
+
xres
+
X resolution
+
yres
+
Y resolution
+
+

Return Value

+

New attribute

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.
+
+The group parameter specifies the IPP attribute group tag: none +(IPP_TAG_ZERO, for member attributes), document (IPP_TAG_DOCUMENT), +event notification (IPP_TAG_EVENT_NOTIFICATION), operation +(IPP_TAG_OPERATION), printer (IPP_TAG_PRINTER), subscription +(IPP_TAG_SUBSCRIPTION), or unsupported (IPP_TAG_UNSUPPORTED_GROUP).

+

ippAddResolutions

+

Add resolution values to an IPP message.

+

+ipp_attribute_t *ippAddResolutions (
+    ipp_t *ipp,
+    ipp_tag_t group,
+    const char *name,
+    int num_values,
+    ipp_res_t units,
+    const int *xres,
+    const int *yres
+);

+

Parameters

+
+
ipp
+
IPP message
+
group
+
IPP group
+
name
+
Name of attribute
+
num_values
+
Number of values
+
units
+
Units for resolution
+
xres
+
X resolutions
+
yres
+
Y resolutions
+
+

Return Value

+

New attribute

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.
+
+The group parameter specifies the IPP attribute group tag: none +(IPP_TAG_ZERO, for member attributes), document (IPP_TAG_DOCUMENT), +event notification (IPP_TAG_EVENT_NOTIFICATION), operation +(IPP_TAG_OPERATION), printer (IPP_TAG_PRINTER), subscription +(IPP_TAG_SUBSCRIPTION), or unsupported (IPP_TAG_UNSUPPORTED_GROUP).

+

ippAddSeparator

+

Add a group separator to an IPP message.

+

+ipp_attribute_t *ippAddSeparator (
+    ipp_t *ipp
+);

+

Parameters

+
+
ipp
+
IPP message
+
+

Return Value

+

New attribute

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.

+

ippAddString

+

Add a language-encoded string to an IPP message.

+

+ipp_attribute_t *ippAddString (
+    ipp_t *ipp,
+    ipp_tag_t group,
+    ipp_tag_t value_tag,
+    const char *name,
+    const char *language,
+    const char *value
+);

+

Parameters

+
+
ipp
+
IPP message
+
group
+
IPP group
+
value_tag
+
Type of attribute
+
name
+
Name of attribute
+
language
+
Language code
+
value
+
Value
+
+

Return Value

+

New attribute

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.
+
+The group parameter specifies the IPP attribute group tag: none +(IPP_TAG_ZERO, for member attributes), document (IPP_TAG_DOCUMENT), +event notification (IPP_TAG_EVENT_NOTIFICATION), operation +(IPP_TAG_OPERATION), printer (IPP_TAG_PRINTER), subscription +(IPP_TAG_SUBSCRIPTION), or unsupported (IPP_TAG_UNSUPPORTED_GROUP).
+
+Supported string values include charset (IPP_TAG_CHARSET), keyword +(IPP_TAG_KEYWORD), language (IPP_TAG_LANGUAGE), mimeMediaType +(IPP_TAG_MIMETYPE), name (IPP_TAG_NAME), nameWithLanguage +(IPP_TAG_NAMELANG), text (code IPP_TAG_TEXT@), textWithLanguage +(IPP_TAG_TEXTLANG), uri (IPP_TAG_URI), and uriScheme +(IPP_TAG_URISCHEME).
+
+The language parameter must be non-NULL for nameWithLanguage and +textWithLanguage string values and must be NULL for all other string values.

+

ippAddStrings

+

Add language-encoded strings to an IPP message.

+

+ipp_attribute_t *ippAddStrings (
+    ipp_t *ipp,
+    ipp_tag_t group,
+    ipp_tag_t value_tag,
+    const char *name,
+    int num_values,
+    const char *language,
+    const char *const *values
+);

+

Parameters

+
+
ipp
+
IPP message
+
group
+
IPP group
+
value_tag
+
Type of attribute
+
name
+
Name of attribute
+
num_values
+
Number of values
+
language
+
Language code (NULL for default)
+
values
+
Values
+
+

Return Value

+

New attribute

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.
+
+The group parameter specifies the IPP attribute group tag: none +(IPP_TAG_ZERO, for member attributes), document (IPP_TAG_DOCUMENT), +event notification (IPP_TAG_EVENT_NOTIFICATION), operation +(IPP_TAG_OPERATION), printer (IPP_TAG_PRINTER), subscription +(IPP_TAG_SUBSCRIPTION), or unsupported (IPP_TAG_UNSUPPORTED_GROUP).
+
+Supported string values include charset (IPP_TAG_CHARSET), keyword +(IPP_TAG_KEYWORD), language (IPP_TAG_LANGUAGE), mimeMediaType +(IPP_TAG_MIMETYPE), name (IPP_TAG_NAME), nameWithLanguage +(IPP_TAG_NAMELANG), text (code IPP_TAG_TEXT@), textWithLanguage +(IPP_TAG_TEXTLANG), uri (IPP_TAG_URI), and uriScheme +(IPP_TAG_URISCHEME).
+
+The language parameter must be non-NULL for nameWithLanguage and +textWithLanguage string values and must be NULL for all other string values.

+

 CUPS 1.6 ippAttributeString

+

Convert the attribute's value to a string.

+

+size_t ippAttributeString (
+    ipp_attribute_t *attr,
+    char *buffer,
+    size_t bufsize
+);

+

Parameters

+
+
attr
+
Attribute
+
buffer
+
String buffer or NULL
+
bufsize
+
Size of string buffer
+
+

Return Value

+

Number of bytes less nul

+

Discussion

+

Returns the number of bytes that would be written, not including the +trailing nul. The buffer pointer can be NULL to get the required length, +just like (v)snprintf. + +

+

 CUPS 1.6 ippCopyAttribute

+

Copy an attribute.

+

+ipp_attribute_t *ippCopyAttribute (
+    ipp_t *dst,
+    ipp_attribute_t *srcattr,
+    int quickcopy
+);

+

Parameters

+
+
dst
+
Destination IPP message
+
srcattr
+
Attribute to copy
+
quickcopy
+
1 for a referenced copy, 0 for normal
+
+

Return Value

+

New attribute

+

Discussion

+

The specified attribute, attr, is copied to the destination IPP message. +When quickcopy is non-zero, a "shallow" reference copy of the attribute is +created - this should only be done as long as the original source IPP message will +not be freed for the life of the destination. + +

+

 CUPS 1.6 ippCopyAttributes

+

Copy attributes from one IPP message to another.

+

+int ippCopyAttributes (
+    ipp_t *dst,
+    ipp_t *src,
+    int quickcopy,
+    ipp_copycb_t cb,
+    void *context
+);

+

Parameters

+
+
dst
+
Destination IPP message
+
src
+
Source IPP message
+
quickcopy
+
1 for a referenced copy, 0 for normal
+
cb
+
Copy callback or NULL for none
+
context
+
Context pointer
+
+

Return Value

+

1 on success, 0 on error

+

Discussion

+

Zero or more attributes are copied from the source IPP message, @code@ src, to the +destination IPP message, dst. When quickcopy is non-zero, a "shallow" +reference copy of the attribute is created - this should only be done as long as the +original source IPP message will not be freed for the life of the destination.
+
+The cb and context parameters provide a generic way to "filter" the +attributes that are copied - the function must return 1 to copy the attribute or +0 to skip it. The function may also choose to do a partial copy of the source attribute +itself. + +

+

ippDateToTime

+

Convert from RFC 1903 Date/Time format to UNIX time +in seconds.

+

+time_t ippDateToTime (
+    const ipp_uchar_t *date
+);

+

Parameters

+
+
date
+
RFC 1903 date info
+
+

Return Value

+

UNIX time value

+

ippDelete

+

Delete an IPP message.

+

+void ippDelete (
+    ipp_t *ipp
+);

+

Parameters

+
+
ipp
+
IPP message
+
+

 CUPS 1.1.19/Mac OS X 10.3 ippDeleteAttribute

+

Delete a single attribute in an IPP message.

+

+void ippDeleteAttribute (
+    ipp_t *ipp,
+    ipp_attribute_t *attr
+);

+

Parameters

+
+
ipp
+
IPP message
+
attr
+
Attribute to delete
+
+

 CUPS 1.6 ippDeleteValues

+

Delete values in an attribute.

+

+int ippDeleteValues (
+    ipp_t *ipp,
+    ipp_attribute_t *attr,
+    int element,
+    int count
+);

+

Parameters

+
+
ipp
+
IPP message
+
attr
+
Attribute
+
element
+
Index of first value to delete (0-based)
+
count
+
Number of values to delete
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

The element parameter specifies the first value to delete, starting at 0. It +must be less than the number of values returned by ippGetCount.
+
+Deleting all values in an attribute deletes the attribute. + +

+

ippEnumString

+

Return a string corresponding to the enum value.

+

+const char *ippEnumString (
+    const char *attrname,
+    int enumvalue
+);

+

Parameters

+
+
attrname
+
Attribute name
+
enumvalue
+
Enum value
+
+

Return Value

+

Enum string

+

ippEnumValue

+

Return the value associated with a given enum string.

+

+int ippEnumValue (
+    const char *attrname,
+    const char *enumstring
+);

+

Parameters

+
+
attrname
+
Attribute name
+
enumstring
+
Enum string
+
+

Return Value

+

Enum value or -1 if unknown

+

ippErrorString

+

Return a name for the given status code.

+

+const char *ippErrorString (
+    ipp_status_t error
+);

+

Parameters

+
+
error
+
Error status
+
+

Return Value

+

Text string

+

 CUPS 1.2/Mac OS X 10.5 ippErrorValue

+

Return a status code for the given name.

+

+ipp_status_t ippErrorValue (
+    const char *name
+);

+

Parameters

+
+
name
+
Name
+
+

Return Value

+

IPP status code

+

ippFindAttribute

+

Find a named attribute in a request...

+

+ipp_attribute_t *ippFindAttribute (
+    ipp_t *ipp,
+    const char *name,
+    ipp_tag_t type
+);

+

Parameters

+
+
ipp
+
IPP message
+
name
+
Name of attribute
+
type
+
Type of attribute
+
+

Return Value

+

Matching attribute

+

ippFindNextAttribute

+

Find the next named attribute in a request...

+

+ipp_attribute_t *ippFindNextAttribute (
+    ipp_t *ipp,
+    const char *name,
+    ipp_tag_t type
+);

+

Parameters

+
+
ipp
+
IPP message
+
name
+
Name of attribute
+
type
+
Type of attribute
+
+

Return Value

+

Matching attribute

+

 CUPS 1.6 ippFirstAttribute

+

Return the first attribute in the message.

+

+ipp_attribute_t *ippFirstAttribute (
+    ipp_t *ipp
+);

+

Parameters

+
+
ipp
+
IPP message
+
+

Return Value

+

First attribute or NULL if none

+

 CUPS 1.6 ippGetBoolean

+

Get a boolean value for an attribute.

+

+int ippGetBoolean (
+    ipp_attribute_t *attr,
+    int element
+);

+

Parameters

+
+
attr
+
IPP attribute
+
element
+
Value number (0-based)
+
+

Return Value

+

Boolean value or -1 on error

+

Discussion

+

The element parameter specifies which value to get from 0 to +ippGetCount(attr) - 1. + +

+

 CUPS 1.6 ippGetCollection

+

Get a collection value for an attribute.

+

+ipp_t *ippGetCollection (
+    ipp_attribute_t *attr,
+    int element
+);

+

Parameters

+
+
attr
+
IPP attribute
+
element
+
Value number (0-based)
+
+

Return Value

+

Collection value or NULL on error

+

Discussion

+

The element parameter specifies which value to get from 0 to +ippGetCount(attr) - 1. + +

+

 CUPS 1.6 ippGetCount

+

Get the number of values in an attribute.

+

+int ippGetCount (
+    ipp_attribute_t *attr
+);

+

Parameters

+
+
attr
+
IPP attribute
+
+

Return Value

+

Number of values or -1 on error

+

 CUPS 1.6 ippGetGroupTag

+

Get the group associated with an attribute.

+

+ipp_tag_t ippGetGroupTag (
+    ipp_attribute_t *attr
+);

+

Parameters

+
+
attr
+
IPP attribute
+
+

Return Value

+

Group tag or IPP_TAG_ZERO on error

+

 CUPS 1.6 ippGetInteger

+

Get the integer/enum value for an attribute.

+

+int ippGetInteger (
+    ipp_attribute_t *attr,
+    int element
+);

+

Parameters

+
+
attr
+
IPP attribute
+
element
+
Value number (0-based)
+
+

Return Value

+

Value or -1 on error

+

Discussion

+

The element parameter specifies which value to get from 0 to +ippGetCount(attr) - 1. + +

+

 CUPS 1.6 ippGetName

+

Get the attribute name.

+

+const char *ippGetName (
+    ipp_attribute_t *attr
+);

+

Parameters

+
+
attr
+
IPP attribute
+
+

Return Value

+

Attribute name or NULL for separators

+

 CUPS 1.6 ippGetOperation

+

Get the operation ID in an IPP message.

+

+ipp_op_t ippGetOperation (
+    ipp_t *ipp
+);

+

Parameters

+
+
ipp
+
IPP request message
+
+

Return Value

+

Operation ID or -1 on error

+

 CUPS 1.6 ippGetRequestId

+

Get the request ID from an IPP message.

+

+int ippGetRequestId (
+    ipp_t *ipp
+);

+

Parameters

+
+
ipp
+
IPP message
+
+

Return Value

+

Request ID or -1 on error

+

 CUPS 1.6 ippGetResolution

+

Get a resolution value for an attribute.

+

+int ippGetResolution (
+    ipp_attribute_t *attr,
+    int element,
+    int *yres,
+    ipp_res_t *units
+);

+

Parameters

+
+
attr
+
IPP attribute
+
element
+
Value number (0-based)
+
yres
+
Vertical/feed resolution
+
units
+
Units for resolution
+
+

Return Value

+

Horizontal/cross feed resolution or -1

+

Discussion

+

The element parameter specifies which value to get from 0 to +ippGetCount(attr) - 1. + +

+

 CUPS 1.6 ippGetStatusCode

+

Get the status code from an IPP response or event message.

+

+ipp_status_t ippGetStatusCode (
+    ipp_t *ipp
+);

+

Parameters

+
+
ipp
+
IPP response or event message
+
+

Return Value

+

Status code in IPP message

+

ippGetString

+

Return the value...

+

+const char *ippGetString (
+    ipp_attribute_t *attr,
+    int element,
+    const char **language
+);

+

Parameters

+
+
attr
+
IPP attribute
+
element
+
Value number (0-based)
+
language
+
Language code (NULL for don't care)
+
+

Return Value

+

Get the string and optionally the language code for an attribute.

+

The element parameter specifies which value to get from 0 to +ippGetCount(attr) - 1. + +

+

 CUPS 1.6 ippGetValueTag

+

Get the value tag for an attribute.

+

+ipp_tag_t ippGetValueTag (
+    ipp_attribute_t *attr
+);

+

Parameters

+
+
attr
+
IPP attribute
+
+

Return Value

+

Value tag or IPP_TAG_ZERO on error

+

 CUPS 1.6 ippGetVersion

+

Get the major and minor version number from an IPP message.

+

+int ippGetVersion (
+    ipp_t *ipp,
+    int *minor
+);

+

Parameters

+
+
ipp
+
IPP message
+
minor
+
Minor version number or NULL
+
+

Return Value

+

Major version number or -1 on error

+

ippLength

+

Compute the length of an IPP message.

+

+size_t ippLength (
+    ipp_t *ipp
+);

+

Parameters

+
+
ipp
+
IPP message
+
+

Return Value

+

Size of IPP message

+

ippNew

+

Allocate a new IPP message.

+

+ipp_t *ippNew (void);

+

Return Value

+

New IPP message

+

 CUPS 1.2/Mac OS X 10.5 ippNewRequest

+

Allocate a new IPP request message.

+

+ipp_t *ippNewRequest (
+    ipp_op_t op
+);

+

Parameters

+
+
op
+
Operation code
+
+

Return Value

+

IPP request message

+

Discussion

+

The new request message is initialized with the attributes-charset and +attributes-natural-language attributes added. The +attributes-natural-language value is derived from the current locale. + +

+

 CUPS 1.6 ippNextAttribute

+

Return the next attribute in the message.

+

+ipp_attribute_t *ippNextAttribute (
+    ipp_t *ipp
+);

+

Parameters

+
+
ipp
+
IPP message
+
+

Return Value

+

Next attribute or NULL if none

+

 CUPS 1.2/Mac OS X 10.5 ippOpString

+

Return a name for the given operation id.

+

+const char *ippOpString (
+    ipp_op_t op
+);

+

Parameters

+
+
op
+
Operation ID
+
+

Return Value

+

Name

+

 CUPS 1.2/Mac OS X 10.5 ippOpValue

+

Return an operation id for the given name.

+

+ipp_op_t ippOpValue (
+    const char *name
+);

+

Parameters

+
+
name
+
Textual name
+
+

Return Value

+

Operation ID

+

ippPort

+

Return the default IPP port number.

+

+int ippPort (void);

+

Return Value

+

Port number

+

ippRead

+

Read data for an IPP message from a HTTP connection.

+

+ipp_state_t ippRead (
+    http_t *http,
+    ipp_t *ipp
+);

+

Parameters

+
+
http
+
HTTP connection
+
ipp
+
IPP data
+
+

Return Value

+

Current state

+

 CUPS 1.1.19/Mac OS X 10.3 ippReadFile

+

Read data for an IPP message from a file.

+

+ipp_state_t ippReadFile (
+    int fd,
+    ipp_t *ipp
+);

+

Parameters

+
+
fd
+
HTTP data
+
ipp
+
IPP data
+
+

Return Value

+

Current state

+

 CUPS 1.2/Mac OS X 10.5 ippReadIO

+

Read data for an IPP message.

+

+ipp_state_t ippReadIO (
+    void *src,
+    ipp_iocb_t cb,
+    int blocking,
+    ipp_t *parent,
+    ipp_t *ipp
+);

+

Parameters

+
+
src
+
Data source
+
cb
+
Read callback function
+
blocking
+
Use blocking IO?
+
parent
+
Parent request, if any
+
ipp
+
IPP data
+
+

Return Value

+

Current state

+

 CUPS 1.6 ippSetBoolean

+

Set a boolean value in an attribute.

+

+int ippSetBoolean (
+    ipp_t *ipp,
+    ipp_attribute_t **attr,
+    int element,
+    int boolvalue
+);

+

Parameters

+
+
ipp
+
IPP message
+
attr
+
IPP attribute
+
element
+
Value number (0-based)
+
boolvalue
+
Boolean value
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

The ipp parameter refers to the IPP message containing the attribute that was +previously created using the ippNew or ippNewRequest functions.
+
+The attr parameter may be modified as a result of setting the value.
+
+The element parameter specifies which value to set from 0 to +ippGetCount(attr). + +

+

 CUPS 1.6 ippSetCollection

+

Set a collection value in an attribute.

+

+int ippSetCollection (
+    ipp_t *ipp,
+    ipp_attribute_t **attr,
+    int element,
+    ipp_t *colvalue
+);

+

Parameters

+
+
ipp
+
IPP message
+
attr
+
IPP attribute
+
element
+
Value number (0-based)
+
colvalue
+
Collection value
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

The ipp parameter refers to the IPP message containing the attribute that was +previously created using the ippNew or ippNewRequest functions.
+
+The attr parameter may be modified as a result of setting the value.
+
+The element parameter specifies which value to set from 0 to +ippGetCount(attr). + +

+

 CUPS 1.6 ippSetGroupTag

+

Set the group tag of an attribute.

+

+int ippSetGroupTag (
+    ipp_t *ipp,
+    ipp_attribute_t **attr,
+    ipp_tag_t group_tag
+);

+

Parameters

+
+
ipp
+
IPP message
+
attr
+
Attribute
+
group_tag
+
Group tag
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

The ipp parameter refers to the IPP message containing the attribute that was +previously created using the ippNew or ippNewRequest functions.
+
+The attr parameter may be modified as a result of setting the value.
+
+The group parameter specifies the IPP attribute group tag: none +(IPP_TAG_ZERO, for member attributes), document (IPP_TAG_DOCUMENT), +event notification (IPP_TAG_EVENT_NOTIFICATION), operation +(IPP_TAG_OPERATION), printer (IPP_TAG_PRINTER), subscription +(IPP_TAG_SUBSCRIPTION), or unsupported (IPP_TAG_UNSUPPORTED_GROUP). + +

+

 CUPS 1.6 ippSetInteger

+

Set an integer or enum value in an attribute.

+

+int ippSetInteger (
+    ipp_t *ipp,
+    ipp_attribute_t **attr,
+    int element,
+    int intvalue
+);

+

Parameters

+
+
ipp
+
IPP message
+
attr
+
IPP attribute
+
element
+
Value number (0-based)
+
intvalue
+
Integer/enum value
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

The ipp parameter refers to the IPP message containing the attribute that was +previously created using the ippNew or ippNewRequest functions.
+
+The attr parameter may be modified as a result of setting the value.
+
+The element parameter specifies which value to set from 0 to +ippGetCount(attr). + +

+

 CUPS 1.6 ippSetName

+

Set the name of an attribute.

+

+int ippSetName (
+    ipp_t *ipp,
+    ipp_attribute_t **attr,
+    const char *name
+);

+

Parameters

+
+
ipp
+
IPP message
+
attr
+
IPP attribute
+
name
+
Attribute name
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

The ipp parameter refers to the IPP message containing the attribute that was +previously created using the ippNew or ippNewRequest functions.
+
+The attr parameter may be modified as a result of setting the value. + +

+

 CUPS 1.6 ippSetOperation

+

Set the operation ID in an IPP request message.

+

+int ippSetOperation (
+    ipp_t *ipp,
+    ipp_op_t op
+);

+

Parameters

+
+
ipp
+
IPP request message
+
op
+
Operation ID
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions. + +

+

ippSetPort

+

Set the default port number.

+

+void ippSetPort (
+    int p
+);

+

Parameters

+
+
p
+
Port number to use
+
+

 CUPS 1.6 ippSetRange

+

Set a rangeOfInteger value in an attribute.

+

+int ippSetRange (
+    ipp_t *ipp,
+    ipp_attribute_t **attr,
+    int element,
+    int lowervalue,
+    int uppervalue
+);

+

Parameters

+
+
ipp
+
IPP message
+
attr
+
IPP attribute
+
element
+
Value number (0-based)
+
lowervalue
+
Lower bound for range
+
uppervalue
+
Upper bound for range
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

The ipp parameter refers to the IPP message containing the attribute that was +previously created using the ippNew or ippNewRequest functions.
+
+The attr parameter may be modified as a result of setting the value.
+
+The element parameter specifies which value to set from 0 to +ippGetCount(attr). + +

+

 CUPS 1.6 ippSetRequestId

+

Set the request ID in an IPP message.

+

+int ippSetRequestId (
+    ipp_t *ipp,
+    int request_id
+);

+

Parameters

+
+
ipp
+
IPP message
+
request_id
+
Request ID
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.
+
+The request_id parameter must be greater than 0. + +

+

 CUPS 1.6 ippSetResolution

+

Set a resolution value in an attribute.

+

+int ippSetResolution (
+    ipp_t *ipp,
+    ipp_attribute_t **attr,
+    int element,
+    ipp_res_t unitsvalue,
+    int xresvalue,
+    int yresvalue
+);

+

Parameters

+
+
ipp
+
IPP message
+
attr
+
IPP attribute
+
element
+
Value number (0-based)
+
unitsvalue
+
Resolution units
+
xresvalue
+
Horizontal/cross feed resolution
+
yresvalue
+
Vertical/feed resolution
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

The ipp parameter refers to the IPP message containing the attribute that was +previously created using the ippNew or ippNewRequest functions.
+
+The attr parameter may be modified as a result of setting the value.
+
+The element parameter specifies which value to set from 0 to +ippGetCount(attr). + +

+

 CUPS 1.6 ippSetStatusCode

+

Set the status code in an IPP response or event message.

+

+int ippSetStatusCode (
+    ipp_t *ipp,
+    ipp_status_t status
+);

+

Parameters

+
+
ipp
+
IPP response or event message
+
status
+
Status code
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions. + +

+

 CUPS 1.6 ippSetString

+

Set a string value in an attribute.

+

+int ippSetString (
+    ipp_t *ipp,
+    ipp_attribute_t **attr,
+    int element,
+    const char *strvalue
+);

+

Parameters

+
+
ipp
+
IPP message
+
attr
+
IPP attribute
+
element
+
Value number (0-based)
+
strvalue
+
String value
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

The ipp parameter refers to the IPP message containing the attribute that was +previously created using the ippNew or ippNewRequest functions.
+
+The attr parameter may be modified as a result of setting the value.
+
+The element parameter specifies which value to set from 0 to +ippGetCount(attr). + +

+

 CUPS 1.6 ippSetValueTag

+

Set the value tag of an attribute.

+

+int ippSetValueTag (
+    ipp_t *ipp,
+    ipp_attribute_t **attr,
+    ipp_tag_t value_tag
+);

+

Parameters

+
+
ipp
+
IPP message
+
attr
+
IPP attribute
+
value_tag
+
Value tag
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

The ipp parameter refers to the IPP message containing the attribute that was +previously created using the ippNew or ippNewRequest functions.
+
+The attr parameter may be modified as a result of setting the value.
+
+Integer (IPP_TAG_INTEGER) values can be promoted to rangeOfInteger +(IPP_TAG_RANGE) values, the various string tags can be promoted to name +(IPP_TAG_NAME) or nameWithLanguage (IPP_TAG_NAMELANG) values, text +(IPP_TAG_TEXT) values can be promoted to textWithLanguage +(IPP_TAG_TEXTLANG) values, and all values can be demoted to the various +out-of-band value tags such as no-value (IPP_TAG_NOVALUE). All other changes +will be rejected.
+
+Promoting a string attribute to nameWithLanguage or textWithLanguage adds the language +code in the "attributes-natural-language" attribute or, if not present, the language +code for the current locale. + +

+

 CUPS 1.6 ippSetVersion

+

Set the version number in an IPP message.

+

+int ippSetVersion (
+    ipp_t *ipp,
+    int major,
+    int minor
+);

+

Parameters

+
+
ipp
+
IPP message
+
major
+
Major version number (major.minor)
+
minor
+
Minor version number (major.minor)
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

The ipp parameter refers to an IPP message previously created using the +ippNew or ippNewRequest functions.
+
+The valid version numbers are currently 1.0, 1.1, 2.0, 2.1, and 2.2. + +

+

 CUPS 1.4/Mac OS X 10.6 ippTagString

+

Return the tag name corresponding to a tag value.

+

+const char *ippTagString (
+    ipp_tag_t tag
+);

+

Parameters

+
+
tag
+
Tag value
+
+

Return Value

+

Tag name

+

Discussion

+

The returned names are defined in RFC 2911 and 3382. + +

+

 CUPS 1.4/Mac OS X 10.6 ippTagValue

+

Return the tag value corresponding to a tag name.

+

+ipp_tag_t ippTagValue (
+    const char *name
+);

+

Parameters

+
+
name
+
Tag name
+
+

Return Value

+

Tag value

+

Discussion

+

The tag names are defined in RFC 2911 and 3382. + +

+

ippTimeToDate

+

Convert from UNIX time to RFC 1903 format.

+

+const ipp_uchar_t *ippTimeToDate (
+    time_t t
+);

+

Parameters

+
+
t
+
UNIX time value
+
+

Return Value

+

RFC-1903 date/time data

+

ippWrite

+

Write data for an IPP message to a HTTP connection.

+

+ipp_state_t ippWrite (
+    http_t *http,
+    ipp_t *ipp
+);

+

Parameters

+
+
http
+
HTTP connection
+
ipp
+
IPP data
+
+

Return Value

+

Current state

+

 CUPS 1.1.19/Mac OS X 10.3 ippWriteFile

+

Write data for an IPP message to a file.

+

+ipp_state_t ippWriteFile (
+    int fd,
+    ipp_t *ipp
+);

+

Parameters

+
+
fd
+
HTTP data
+
ipp
+
IPP data
+
+

Return Value

+

Current state

+

 CUPS 1.2/Mac OS X 10.5 ippWriteIO

+

Write data for an IPP message.

+

+ipp_state_t ippWriteIO (
+    void *dst,
+    ipp_iocb_t cb,
+    int blocking,
+    ipp_t *parent,
+    ipp_t *ipp
+);

+

Parameters

+
+
dst
+
Destination
+
cb
+
Write callback function
+
blocking
+
Use blocking IO?
+
parent
+
Parent IPP message
+
ipp
+
IPP data
+
+

Return Value

+

Current state

+

Data Types

+

gss_auth_identity_desc

+

Local functions...

+

+typedef struct gss_auth_identity gss_auth_identity_desc; +

+

 CUPS 1.2/Mac OS X 10.5 http_addr_t

+

Socket address union, which +makes using IPv6 and other +address types easier and +more portable.

+

+typedef union _http_addr_u / http_addr_t; +

+

 CUPS 1.2/Mac OS X 10.5 http_addrlist_t

+

Socket address list, which is +used to enumerate all of the +addresses that are associated +with a hostname.

+

+typedef struct http_addrlist_s / http_addrlist_t; +

+

http_auth_t

+

HTTP authentication types

+

+typedef enum http_auth_e http_auth_t; +

+

 CUPS 1.5/Mac OS X 10.7 http_credential_t

+

HTTP credential data

+

+typedef struct http_credential_s http_credential_t; +

+

http_encoding_t

+

HTTP transfer encoding values

+

+typedef enum http_encoding_e http_encoding_t; +

+

http_encryption_t

+

HTTP encryption values

+

+typedef enum http_encryption_e http_encryption_t; +

+

http_field_t

+

HTTP field names

+

+typedef enum http_field_e http_field_t; +

+

http_keepalive_t

+

HTTP keep-alive values

+

+typedef enum http_keepalive_e http_keepalive_t; +

+

http_state_t

+

HTTP state values; states +are server-oriented...

+

+typedef enum http_state_e / http_state_t; +

+

http_status_t

+

HTTP status codes

+

+typedef enum http_status_e http_status_t; +

+

http_t

+

HTTP connection type

+

+typedef struct _http_s http_t; +

+

 CUPS 1.5/Mac OS X 10.7 http_timeout_cb_t

+

HTTP timeout callback

+

+typedef int (*http_timeout_cb_t)(http_t *http, void *user_data); +

+

http_uri_coding_t

+

URI en/decode flags

+

+typedef enum http_uri_coding_e http_uri_coding_t; +

+

 CUPS 1.2 http_uri_status_t

+

URI separation status

+

+typedef enum http_uri_status_e http_uri_status_t; +

+

http_version_t

+

HTTP version numbers

+

+typedef enum http_version_e http_version_t; +

+

ipp_attribute_t

+

IPP attribute

+

+typedef struct _ipp_attribute_s ipp_attribute_t; +

+

ipp_copycb_t

+

Prototypes...

+

+typedef int (*ipp_copycb_t)(void *context, ipp_t *dst, ipp_attribute_t *attr); +

+

ipp_dstate_t

+

Document states

+

+typedef enum ipp_dstate_e ipp_dstate_t; +

+

ipp_finish_t

+

Finishings

+

+typedef enum ipp_finish_e ipp_finish_t; +

+

 CUPS 1.2/Mac OS X 10.5 ipp_iocb_t

+

IPP IO Callback Function

+

+typedef ssize_t (*ipp_iocb_t)(void *context, ipp_uchar_t *buffer, size_t bytes); +

+

ipp_jcollate_t

+

Job collation types

+

+typedef enum ipp_jcollate_e ipp_jcollate_t; +

+

ipp_orient_t

+

Orientation values

+

+typedef enum ipp_orient_e ipp_orient_t; +

+

ipp_pstate_t

+

Printer states

+

+typedef enum ipp_pstate_e ipp_pstate_t; +

+

ipp_quality_t

+

Qualities

+

+typedef enum ipp_quality_e ipp_quality_t; +

+

ipp_res_t

+

Resolution units

+

+typedef enum ipp_res_e ipp_res_t; +

+

ipp_state_t

+

IPP states

+

+typedef enum ipp_state_e ipp_state_t; +

+

ipp_t

+

IPP request/response data

+

+typedef struct _ipp_s ipp_t; +

+

ipp_uchar_t

+

Unsigned 8-bit integer/character

+

+typedef unsigned char ipp_uchar_t; +

+

Structures

+

gss_auth_identity

+

Local functions...

+

struct gss_auth_identity {
+    gss_buffer_t *credentialsRef;
+    uint32_t flags;
+    char *password;
+    char *realm;
+    uint32_t type;
+    char *username;
+};

+

Members

+
+
credentialsRef
+
flags
+
password
+
realm
+
type
+
username
+
+

 CUPS 1.2/Mac OS X 10.5 http_addrlist_s

+

Socket address list, which is +used to enumerate all of the +addresses that are associated +with a hostname.

+

struct http_addrlist_s {
+    http_addr_t addr;
+    struct http_addrlist_s *next;
+};

+

Members

+
+
addr
+
Address
+
next
+
Pointer to next address in list
+
+

 CUPS 1.5/Mac OS X 10.7 http_credential_s

+

HTTP credential data

+

struct http_credential_s {
+    void *data;
+    size_t datalen;
+};

+

Members

+
+
data
+
Pointer to credential data
+
datalen
+
Credential length
+
+

Constants

+

http_auth_e

+

HTTP authentication types

+

Constants

+
+
HTTP_AUTH_BASIC
+
Basic authentication in use
+
HTTP_AUTH_MD5
+
Digest authentication in use
+
HTTP_AUTH_MD5_INT
+
Digest authentication in use for body
+
HTTP_AUTH_MD5_SESS
+
MD5-session authentication in use
+
HTTP_AUTH_MD5_SESS_INT
+
MD5-session authentication in use for body
+
HTTP_AUTH_NEGOTIATE  CUPS 1.3/Mac OS X 10.5 
+
GSSAPI authentication in use
+
HTTP_AUTH_NONE
+
No authentication in use
+
+

http_encoding_e

+

HTTP transfer encoding values

+

Constants

+
+
HTTP_ENCODE_CHUNKED
+
Data is chunked
+
HTTP_ENCODE_FIELDS
+
Sending HTTP fields
+
HTTP_ENCODE_LENGTH
+
Data is sent with Content-Length
+
+

http_encryption_e

+

HTTP encryption values

+

Constants

+
+
HTTP_ENCRYPT_ALWAYS
+
Always encrypt (SSL)
+
HTTP_ENCRYPT_IF_REQUESTED
+
Encrypt if requested (TLS upgrade)
+
HTTP_ENCRYPT_NEVER
+
Never encrypt
+
HTTP_ENCRYPT_REQUIRED
+
Encryption is required (TLS upgrade)
+
+

http_field_e

+

HTTP field names

+

Constants

+
+
HTTP_FIELD_ACCEPT_LANGUAGE
+
Accept-Language field
+
HTTP_FIELD_ACCEPT_RANGES
+
Accept-Ranges field
+
HTTP_FIELD_AUTHORIZATION
+
Authorization field
+
HTTP_FIELD_CONNECTION
+
Connection field
+
HTTP_FIELD_CONTENT_ENCODING
+
Content-Encoding field
+
HTTP_FIELD_CONTENT_LANGUAGE
+
Content-Language field
+
HTTP_FIELD_CONTENT_LENGTH
+
Content-Length field
+
HTTP_FIELD_CONTENT_LOCATION
+
Content-Location field
+
HTTP_FIELD_CONTENT_MD5
+
Content-MD5 field
+
HTTP_FIELD_CONTENT_RANGE
+
Content-Range field
+
HTTP_FIELD_CONTENT_TYPE
+
Content-Type field
+
HTTP_FIELD_CONTENT_VERSION
+
Content-Version field
+
HTTP_FIELD_DATE
+
Date field
+
HTTP_FIELD_HOST
+
Host field
+
HTTP_FIELD_IF_MODIFIED_SINCE
+
If-Modified-Since field
+
HTTP_FIELD_IF_UNMODIFIED_SINCE
+
If-Unmodified-Since field
+
HTTP_FIELD_KEEP_ALIVE
+
Keep-Alive field
+
HTTP_FIELD_LAST_MODIFIED
+
Last-Modified field
+
HTTP_FIELD_LINK
+
Link field
+
HTTP_FIELD_LOCATION
+
Location field
+
HTTP_FIELD_MAX
+
Maximum field index
+
HTTP_FIELD_RANGE
+
Range field
+
HTTP_FIELD_REFERER
+
Referer field
+
HTTP_FIELD_RETRY_AFTER
+
Retry-After field
+
HTTP_FIELD_TRANSFER_ENCODING
+
Transfer-Encoding field
+
HTTP_FIELD_UNKNOWN
+
Unknown field
+
HTTP_FIELD_UPGRADE
+
Upgrade field
+
HTTP_FIELD_USER_AGENT
+
User-Agent field
+
HTTP_FIELD_WWW_AUTHENTICATE
+
WWW-Authenticate field
+
+

http_keepalive_e

+

HTTP keep-alive values

+

Constants

+
+
HTTP_KEEPALIVE_OFF
+
No keep alive support
+
HTTP_KEEPALIVE_ON
+
Use keep alive
+
+

http_state_e

+

HTTP state values; states +are server-oriented...

+

Constants

+
+
HTTP_CLOSE
+
CLOSE command, waiting for blank line
+
HTTP_DELETE
+
DELETE command, waiting for blank line
+
HTTP_GET
+
GET command, waiting for blank line
+
HTTP_GET_SEND
+
GET command, sending data
+
HTTP_HEAD
+
HEAD command, waiting for blank line
+
HTTP_OPTIONS
+
OPTIONS command, waiting for blank line
+
HTTP_POST
+
POST command, waiting for blank line
+
HTTP_POST_RECV
+
POST command, receiving data
+
HTTP_POST_SEND
+
POST command, sending data
+
HTTP_PUT
+
PUT command, waiting for blank line
+
HTTP_PUT_RECV
+
PUT command, receiving data
+
HTTP_STATUS
+
Command complete, sending status
+
HTTP_TRACE
+
TRACE command, waiting for blank line
+
HTTP_WAITING
+
Waiting for command
+
+

http_status_e

+

HTTP status codes

+

Constants

+
+
HTTP_ACCEPTED
+
DELETE command was successful
+
HTTP_AUTHORIZATION_CANCELED  CUPS 1.4 
+
User canceled authorization
+
HTTP_BAD_GATEWAY
+
Bad gateway
+
HTTP_BAD_REQUEST
+
Bad request
+
HTTP_CONFLICT
+
Request is self-conflicting
+
HTTP_CONTINUE
+
Everything OK, keep going...
+
HTTP_CREATED
+
PUT command was successful
+
HTTP_ERROR
+
An error response from httpXxxx()
+
HTTP_EXPECTATION_FAILED
+
The expectation given in an Expect header field was not met
+
HTTP_FORBIDDEN
+
Forbidden to access this URI
+
HTTP_GATEWAY_TIMEOUT
+
Gateway connection timed out
+
HTTP_GONE
+
Server has gone away
+
HTTP_LENGTH_REQUIRED
+
A content length or encoding is required
+
HTTP_METHOD_NOT_ALLOWED
+
Method is not allowed
+
HTTP_MOVED_PERMANENTLY
+
Document has moved permanently
+
HTTP_MOVED_TEMPORARILY
+
Document has moved temporarily
+
HTTP_MULTIPLE_CHOICES
+
Multiple files match request
+
HTTP_NOT_ACCEPTABLE
+
Not Acceptable
+
HTTP_NOT_AUTHORITATIVE
+
Information isn't authoritative
+
HTTP_NOT_FOUND
+
URI was not found
+
HTTP_NOT_IMPLEMENTED
+
Feature not implemented
+
HTTP_NOT_MODIFIED
+
File not modified
+
HTTP_NOT_SUPPORTED
+
HTTP version not supported
+
HTTP_NO_CONTENT
+
Successful command, no new data
+
HTTP_OK
+
OPTIONS/GET/HEAD/POST/TRACE command was successful
+
HTTP_PARTIAL_CONTENT
+
Only a partial file was recieved/sent
+
HTTP_PAYMENT_REQUIRED
+
Payment required
+
HTTP_PKI_ERROR  CUPS 1.5/Mac OS X 10.7 
+
Error negotiating a secure connection
+
HTTP_PRECONDITION
+
Precondition failed
+
HTTP_PROXY_AUTHENTICATION
+
Proxy Authentication is Required
+
HTTP_REQUESTED_RANGE
+
The requested range is not satisfiable
+
HTTP_REQUEST_TIMEOUT
+
Request timed out
+
HTTP_REQUEST_TOO_LARGE
+
Request entity too large
+
HTTP_RESET_CONTENT
+
Content was reset/recreated
+
HTTP_SEE_OTHER
+
See this other link...
+
HTTP_SERVER_ERROR
+
Internal server error
+
HTTP_SERVICE_UNAVAILABLE
+
Service is unavailable
+
HTTP_SWITCHING_PROTOCOLS
+
HTTP upgrade to TLS/SSL
+
HTTP_UNAUTHORIZED
+
Unauthorized to access host
+
HTTP_UNSUPPORTED_MEDIATYPE
+
The requested media type is unsupported
+
HTTP_UPGRADE_REQUIRED
+
Upgrade to SSL/TLS required
+
HTTP_URI_TOO_LONG
+
URI too long
+
HTTP_USE_PROXY
+
Must use a proxy to access this URI
+
+

http_uri_coding_e

+

URI en/decode flags

+

Constants

+
+
HTTP_URI_CODING_ALL
+
En/decode everything
+
HTTP_URI_CODING_HOSTNAME
+
En/decode the hostname portion
+
HTTP_URI_CODING_MOST
+
En/decode all but the query
+
HTTP_URI_CODING_NONE
+
Don't en/decode anything
+
HTTP_URI_CODING_QUERY
+
En/decode the query portion
+
HTTP_URI_CODING_RESOURCE
+
En/decode the resource portion
+
HTTP_URI_CODING_USERNAME
+
En/decode the username portion
+
+

 CUPS 1.2 http_uri_status_e

+

URI separation status

+

Constants

+
+
HTTP_URI_BAD_ARGUMENTS
+
Bad arguments to function (error)
+
HTTP_URI_BAD_HOSTNAME
+
Bad hostname in URI (error)
+
HTTP_URI_BAD_PORT
+
Bad port number in URI (error)
+
HTTP_URI_BAD_RESOURCE
+
Bad resource in URI (error)
+
HTTP_URI_BAD_SCHEME
+
Bad scheme in URI (error)
+
HTTP_URI_BAD_URI
+
Bad/empty URI (error)
+
HTTP_URI_BAD_USERNAME
+
Bad username in URI (error)
+
HTTP_URI_MISSING_RESOURCE
+
Missing resource in URI (warning)
+
HTTP_URI_MISSING_SCHEME
+
Missing scheme in URI (warning)
+
HTTP_URI_OK
+
URI decoded OK
+
HTTP_URI_OVERFLOW
+
URI buffer for httpAssembleURI is too small
+
HTTP_URI_UNKNOWN_SCHEME
+
Unknown scheme in URI (warning)
+
+

http_version_e

+

HTTP version numbers

+

Constants

+
+
HTTP_0_9
+
HTTP/0.9
+
HTTP_1_0
+
HTTP/1.0
+
HTTP_1_1
+
HTTP/1.1
+
+

ipp_dstate_e

+

Document states

+

Constants

+
+
IPP_DOCUMENT_ABORTED
+
IPP_DOCUMENT_CANCELED
+
IPP_DOCUMENT_COMPLETED
+
IPP_DOCUMENT_PENDING
+
IPP_DOCUMENT_PROCESSING
+
+

ipp_finish_e

+

Finishings

+

Constants

+
+
IPP_FINISHINGS_BALE
+
Bale (any type)
+
IPP_FINISHINGS_BIND
+
Bind
+
IPP_FINISHINGS_BIND_BOTTOM
+
Bind on bottom
+
IPP_FINISHINGS_BIND_LEFT
+
Bind on left
+
IPP_FINISHINGS_BIND_RIGHT
+
Bind on right
+
IPP_FINISHINGS_BIND_TOP
+
Bind on top
+
IPP_FINISHINGS_BOOKLET_MAKER
+
Fold to make booklet
+
IPP_FINISHINGS_COVER
+
Add cover
+
IPP_FINISHINGS_EDGE_STITCH
+
Stitch along any side
+
IPP_FINISHINGS_EDGE_STITCH_BOTTOM
+
Stitch along bottom edge
+
IPP_FINISHINGS_EDGE_STITCH_LEFT
+
Stitch along left side
+
IPP_FINISHINGS_EDGE_STITCH_RIGHT
+
Stitch along right side
+
IPP_FINISHINGS_EDGE_STITCH_TOP
+
Stitch along top edge
+
IPP_FINISHINGS_FOLD
+
Fold (any type)
+
IPP_FINISHINGS_JOB_OFFSET
+
Offset for binding (any type)
+
IPP_FINISHINGS_NONE
+
No finishing
+
IPP_FINISHINGS_PUNCH
+
Punch (any location/count)
+
IPP_FINISHINGS_SADDLE_STITCH
+
Staple interior
+
IPP_FINISHINGS_STAPLE
+
Staple (any location)
+
IPP_FINISHINGS_STAPLE_BOTTOM_LEFT
+
Staple bottom left corner
+
IPP_FINISHINGS_STAPLE_BOTTOM_RIGHT
+
Staple bottom right corner
+
IPP_FINISHINGS_STAPLE_DUAL_BOTTOM
+
Two staples on bottom
+
IPP_FINISHINGS_STAPLE_DUAL_LEFT
+
Two staples on left
+
IPP_FINISHINGS_STAPLE_DUAL_RIGHT
+
Two staples on right
+
IPP_FINISHINGS_STAPLE_DUAL_TOP
+
Two staples on top
+
IPP_FINISHINGS_STAPLE_TOP_LEFT
+
Staple top left corner
+
IPP_FINISHINGS_STAPLE_TOP_RIGHT
+
Staple top right corner
+
IPP_FINISHINGS_TRIM
+
Trim (any type)
+
IPP_FINISHINGS_TRIM_AFTER_COPIES
+
Trim output after each copy
+
IPP_FINISHINGS_TRIM_AFTER_DOCUMENTS
+
Trim output after each document
+
IPP_FINISHINGS_TRIM_AFTER_JOB
+
Trim output after job
+
IPP_FINISHINGS_TRIM_AFTER_PAGES
+
Trim output after each page
+
+

ipp_jcollate_e

+

Job collation types

+

Constants

+
+
IPP_JOB_COLLATED_DOCUMENTS
+
IPP_JOB_UNCOLLATED_DOCUMENTS
+
IPP_JOB_UNCOLLATED_SHEETS
+
+

ipp_jstate_e

+

Job states

+

Constants

+
+
IPP_JOB_ABORTED
+
Job has aborted due to error
+
IPP_JOB_CANCELED
+
Job has been canceled
+
IPP_JOB_COMPLETED
+
Job has completed successfully
+
IPP_JOB_HELD
+
Job is held for printing
+
IPP_JOB_PENDING
+
Job is waiting to be printed
+
IPP_JOB_PROCESSING
+
Job is currently printing
+
IPP_JOB_STOPPED
+
Job has been stopped
+
+

ipp_op_e

+

IPP operations

+

Constants

+
+
CUPS_ACCEPT_JOBS
+
Accept new jobs on a printer
+
CUPS_ADD_MODIFY_CLASS
+
Add or modify a class
+
CUPS_ADD_MODIFY_PRINTER
+
Add or modify a printer
+
CUPS_AUTHENTICATE_JOB  CUPS 1.2/Mac OS X 10.5 
+
Authenticate a job
+
CUPS_DELETE_CLASS
+
Delete a class
+
CUPS_DELETE_PRINTER
+
Delete a printer
+
CUPS_GET_CLASSES  DEPRECATED 
+
Get a list of classes
+
CUPS_GET_DEFAULT
+
Get the default printer
+
CUPS_GET_DEVICES
+
Get a list of supported devices
+
CUPS_GET_DOCUMENT  CUPS 1.4/Mac OS X 10.6 
+
Get a document file
+
CUPS_GET_PPD  CUPS 1.3/Mac OS X 10.5 
+
Get a PPD file
+
CUPS_GET_PPDS
+
Get a list of supported drivers
+
CUPS_GET_PRINTERS
+
Get a list of printers and/or classes
+
CUPS_MOVE_JOB
+
Move a job to a different printer
+
CUPS_REJECT_JOBS
+
Reject new jobs on a printer
+
CUPS_SET_DEFAULT
+
Set the default printer
+
IPP_CANCEL_JOB
+
Cancel a job
+
IPP_CANCEL_JOBS
+
Cancel-Jobs
+
IPP_CANCEL_MY_JOBS
+
Cancel-My-Jobs
+
IPP_CANCEL_SUBSCRIPTION  CUPS 1.2/Mac OS X 10.5 
+
Cancel a subscription
+
IPP_CLOSE_JOB
+
Close-Job
+
IPP_CREATE_JOB
+
Create an empty print job
+
IPP_CREATE_JOB_SUBSCRIPTION  CUPS 1.2/Mac OS X 10.5 
+
Create a job subscription
+
IPP_CREATE_PRINTER_SUBSCRIPTION  CUPS 1.2/Mac OS X 10.5 
+
Create a printer subscription
+
IPP_DISABLE_PRINTER
+
Stop a printer
+
IPP_ENABLE_PRINTER
+
Start a printer
+
IPP_GET_JOBS
+
Get a list of jobs
+
IPP_GET_JOB_ATTRIBUTES
+
Get job attributes
+
IPP_GET_NOTIFICATIONS  CUPS 1.2/Mac OS X 10.5 
+
Get notification events
+
IPP_GET_PRINTER_ATTRIBUTES
+
Get printer attributes
+
IPP_GET_PRINTER_SUPPORTED_VALUES
+
Get supported attribute values
+
IPP_GET_SUBSCRIPTIONS  CUPS 1.2/Mac OS X 10.5 
+
Get list of subscriptions
+
IPP_GET_SUBSCRIPTION_ATTRIBUTES  CUPS 1.2/Mac OS X 10.5 
+
Get subscription attributes
+
IPP_HOLD_JOB
+
Hold a job for printing
+
IPP_IDENTIFY_PRINTER
+
Identify-Printer (proposed IPP JPS3)
+
IPP_PAUSE_PRINTER
+
Stop a printer
+
IPP_PRINT_JOB
+
Print a single file
+
IPP_PURGE_JOBS
+
Cancel all jobs
+
IPP_RELEASE_JOB
+
Release a job for printing
+
IPP_RENEW_SUBSCRIPTION  CUPS 1.2/Mac OS X 10.5 
+
Renew a printer subscription
+
IPP_RESTART_JOB
+
Reprint a job
+
IPP_RESUBMIT_JOB
+
Resubmit-Job
+
IPP_RESUME_PRINTER
+
Start a printer
+
IPP_SEND_DOCUMENT
+
Add a file to a job
+
IPP_SET_JOB_ATTRIBUTES
+
Set job attributes
+
IPP_VALIDATE_JOB
+
Validate job options
+
+

ipp_orient_e

+

Orientation values

+

Constants

+
+
IPP_LANDSCAPE
+
90 degrees counter-clockwise
+
IPP_PORTRAIT
+
No rotation
+
IPP_REVERSE_LANDSCAPE
+
90 degrees clockwise
+
IPP_REVERSE_PORTRAIT
+
180 degrees
+
+

ipp_pstate_e

+

Printer states

+

Constants

+
+
IPP_PRINTER_IDLE
+
Printer is idle
+
IPP_PRINTER_PROCESSING
+
Printer is working
+
IPP_PRINTER_STOPPED
+
Printer is stopped
+
+

ipp_quality_e

+

Qualities

+

Constants

+
+
IPP_QUALITY_DRAFT
+
Draft quality
+
IPP_QUALITY_HIGH
+
High quality
+
IPP_QUALITY_NORMAL
+
Normal quality
+
+

ipp_res_e

+

Resolution units

+

Constants

+
+
IPP_RES_PER_CM
+
Pixels per centimeter
+
IPP_RES_PER_INCH
+
Pixels per inch
+
+

ipp_state_e

+

IPP states

+

Constants

+
+
IPP_ATTRIBUTE
+
One or more attributes need to be sent/received
+
IPP_DATA
+
IPP request data needs to be sent/received
+
IPP_ERROR
+
An error occurred
+
IPP_HEADER
+
The request header needs to be sent/received
+
IPP_IDLE
+
Nothing is happening/request completed
+
+

ipp_status_e

+

IPP status codes

+

Constants

+
+
CUPS_SEE_OTHER
+
cups-see-other
+
IPP_ATTRIBUTES
+
client-error-attributes-or-values-not-supported
+
IPP_ATTRIBUTES_NOT_SETTABLE
+
client-error-attributes-not-settable
+
IPP_AUTHENTICATION_CANCELED  CUPS 1.5/Mac OS X 10.7 
+
Authentication canceled by user
+
IPP_BAD_REQUEST
+
client-error-bad-request
+
IPP_CHARSET
+
client-error-charset-not-supported
+
IPP_COMPRESSION_ERROR
+
client-error-compression-error
+
IPP_COMPRESSION_NOT_SUPPORTED
+
client-error-compression-not-supported
+
IPP_CONFLICT
+
client-error-conflicting-attributes
+
IPP_DEVICE_ERROR
+
server-error-device-error
+
IPP_DOCUMENT_ACCESS_ERROR
+
client-error-document-access-error
+
IPP_DOCUMENT_FORMAT
+
client-error-document-format-not-supported
+
IPP_DOCUMENT_FORMAT_ERROR
+
client-error-document-format-error
+
IPP_DOCUMENT_PASSWORD_ERROR
+
client-error-document-password-error
+
IPP_DOCUMENT_PERMISSION_ERROR
+
client-error-document-permission-error
+
IPP_DOCUMENT_SECURITY_ERROR
+
client-error-document-security-error
+
IPP_DOCUMENT_UNPRINTABLE_ERROR
+
client-error-document-unprintable-error
+
IPP_ERROR_JOB_CANCELED
+
server-error-job-canceled
+
IPP_FORBIDDEN
+
client-error-forbidden
+
IPP_GONE
+
client-error-gone
+
IPP_IGNORED_ALL_SUBSCRIPTIONS
+
client-error-ignored-all-subscriptions
+
IPP_INTERNAL_ERROR
+
server-error-internal-error
+
IPP_MULTIPLE_JOBS_NOT_SUPPORTED
+
server-error-multiple-document-jobs-not-supported
+
IPP_NOT_ACCEPTING
+
server-error-not-accepting-jobs
+
IPP_NOT_AUTHENTICATED
+
client-error-not-authenticated
+
IPP_NOT_AUTHORIZED
+
client-error-not-authorized
+
IPP_NOT_FOUND
+
client-error-not-found
+
IPP_NOT_POSSIBLE
+
client-error-not-possible
+
IPP_OK
+
successful-ok
+
IPP_OK_CONFLICT
+
successful-ok-conflicting-attributes
+
IPP_OK_EVENTS_COMPLETE
+
successful-ok-events-complete
+
IPP_OK_IGNORED_SUBSCRIPTIONS
+
successful-ok-ignored-subscriptions
+
IPP_OK_SUBST
+
successful-ok-ignored-or-substituted-attributes
+
IPP_OK_TOO_MANY_EVENTS
+
successful-ok-too-many-events
+
IPP_OPERATION_NOT_SUPPORTED
+
server-error-operation-not-supported
+
IPP_PKI_ERROR  CUPS 1.5/Mac OS X 10.7 
+
Error negotiating a secure connection
+
IPP_PRINTER_BUSY
+
server-error-busy
+
IPP_PRINTER_IS_DEACTIVATED
+
server-error-printer-is-deactivated
+
IPP_REQUEST_ENTITY
+
client-error-request-entity-too-large
+
IPP_REQUEST_VALUE
+
client-error-request-value-too-long
+
IPP_SERVICE_UNAVAILABLE
+
server-error-service-unavailable
+
IPP_TEMPORARY_ERROR
+
server-error-temporary-error
+
IPP_TIMEOUT
+
client-error-timeout
+
IPP_TOO_MANY_SUBSCRIPTIONS
+
client-error-too-many-subscriptions
+
IPP_UPGRADE_REQUIRED
+
TLS upgrade required
+
IPP_URI_SCHEME
+
client-error-uri-scheme-not-supported
+
IPP_VERSION_NOT_SUPPORTED
+
server-error-version-not-supported
+
+

ipp_tag_e

+

Format tags for attributes

+

Constants

+
+
IPP_TAG_ADMINDEFINE
+
Admin-defined value
+
IPP_TAG_BEGIN_COLLECTION
+
Beginning of collection value
+
IPP_TAG_BOOLEAN
+
Boolean value
+
IPP_TAG_CHARSET
+
Character set value
+
IPP_TAG_DATE
+
Date/time value
+
IPP_TAG_DEFAULT
+
Default value
+
IPP_TAG_DELETEATTR
+
Delete-attribute value
+
IPP_TAG_DOCUMENT
+
Document group
+
IPP_TAG_END
+
End-of-attributes
+
IPP_TAG_END_COLLECTION
+
End of collection value
+
IPP_TAG_ENUM
+
Enumeration value
+
IPP_TAG_EVENT_NOTIFICATION
+
Event group
+
IPP_TAG_EXTENSION
+
Extension point for 32-bit tags
+
IPP_TAG_INTEGER
+
Integer value
+
IPP_TAG_JOB
+
Job group
+
IPP_TAG_KEYWORD
+
Keyword value
+
IPP_TAG_LANGUAGE
+
Language value
+
IPP_TAG_MEMBERNAME
+
Collection member name value
+
IPP_TAG_MIMETYPE
+
MIME media type value
+
IPP_TAG_NAME
+
Name value
+
IPP_TAG_NAMELANG
+
Name-with-language value
+
IPP_TAG_NOTSETTABLE
+
Not-settable value
+
IPP_TAG_NOVALUE
+
No-value value
+
IPP_TAG_OPERATION
+
Operation group
+
IPP_TAG_PRINTER
+
Printer group
+
IPP_TAG_RANGE
+
Range value
+
IPP_TAG_RESOLUTION
+
Resolution value
+
IPP_TAG_STRING
+
Octet string value
+
IPP_TAG_SUBSCRIPTION
+
Subscription group
+
IPP_TAG_TEXT
+
Text value
+
IPP_TAG_TEXTLANG
+
Text-with-language value
+
IPP_TAG_UNKNOWN
+
Unknown value
+
IPP_TAG_UNSUPPORTED_GROUP
+
Unsupported attributes group
+
IPP_TAG_UNSUPPORTED_VALUE
+
Unsupported value
+
IPP_TAG_URI
+
URI value
+
IPP_TAG_URISCHEME
+
URI scheme value
+
IPP_TAG_ZERO
+
Zero tag - used for separators
+
+
+ + diff --git a/doc/help/api-mime.html b/doc/help/api-mime.html new file mode 100644 index 0000000000..c27337640e --- /dev/null +++ b/doc/help/api-mime.html @@ -0,0 +1,833 @@ + + + + + MIME API + + + + + + +
+ + +

MIME API

+ +
+ + + + + + + + + + + + + + + + +
Headercups/mime.h
Library-lcupsmime
See AlsoProgramming: Introduction to CUPS Programming
+

Contents

+ + + +

Overview

+ +

The MIME API provides file typing and conversion services for CUPS.

+

Functions

+

mimeAddFilter

+

Add a filter to the current MIME database.

+

+mime_filter_t *mimeAddFilter (
+    mime_t *mime,
+    mime_type_t *src,
+    mime_type_t *dst,
+    int cost,
+    const char *filter
+);

+

Parameters

+
+
mime
+
MIME database
+
src
+
Source type
+
dst
+
Destination type
+
cost
+
Relative time/resource cost
+
filter
+
Filter program to run
+
+

Return Value

+

New filter

+

mimeAddType

+

Add a MIME type to a database.

+

+mime_type_t *mimeAddType (
+    mime_t *mime,
+    const char *super,
+    const char *type
+);

+

Parameters

+
+
mime
+
MIME database
+
super
+
Super-type name
+
type
+
Type name
+
+

Return Value

+

New (or existing) MIME type

+

mimeAddTypeRule

+

Add a detection rule for a file type.

+

+int mimeAddTypeRule (
+    mime_type_t *mt,
+    const char *rule
+);

+

Parameters

+
+
mt
+
Type to add to
+
rule
+
Rule to add
+
+

Return Value

+

0 on success, -1 on failure

+

mimeDelete

+

Delete (free) a MIME database.

+

+void mimeDelete (
+    mime_t *mime
+);

+

Parameters

+
+
mime
+
MIME database
+
+

mimeDeleteFilter

+

Delete a filter from the MIME database.

+

+void mimeDeleteFilter (
+    mime_t *mime,
+    mime_filter_t *filter
+);

+

Parameters

+
+
mime
+
MIME database
+
filter
+
Filter
+
+

mimeDeleteType

+

Delete a type from the MIME database.

+

+void mimeDeleteType (
+    mime_t *mime,
+    mime_type_t *mt
+);

+

Parameters

+
+
mime
+
MIME database
+
mt
+
Type
+
+

mimeFileType

+

Determine the type of a file.

+

+mime_type_t *mimeFileType (
+    mime_t *mime,
+    const char *pathname,
+    const char *filename,
+    int *compression
+);

+

Parameters

+
+
mime
+
MIME database
+
pathname
+
Name of file to check on disk
+
filename
+
Original filename or NULL
+
compression
+
Is the file compressed?
+
+

Return Value

+

Type of file

+

mimeFilter

+

Find the fastest way to convert from one type to another.

+

+cups_array_t *mimeFilter (
+    mime_t *mime,
+    mime_type_t *src,
+    mime_type_t *dst,
+    int *cost
+);

+

Parameters

+
+
mime
+
MIME database
+
src
+
Source file type
+
dst
+
Destination file type
+
cost
+
Cost of filters
+
+

Return Value

+

Array of filters to run

+

mimeFilter2

+

Find the fastest way to convert from one type to another, +including file size.

+

+cups_array_t *mimeFilter2 (
+    mime_t *mime,
+    mime_type_t *src,
+    size_t srcsize,
+    mime_type_t *dst,
+    int *cost
+);

+

Parameters

+
+
mime
+
MIME database
+
src
+
Source file type
+
srcsize
+
Size of source file
+
dst
+
Destination file type
+
cost
+
Cost of filters
+
+

Return Value

+

Array of filters to run

+

mimeFilterLookup

+

Lookup a filter.

+

+mime_filter_t *mimeFilterLookup (
+    mime_t *mime,
+    mime_type_t *src,
+    mime_type_t *dst
+);

+

Parameters

+
+
mime
+
MIME database
+
src
+
Source type
+
dst
+
Destination type
+
+

Return Value

+

Filter for src->dst

+

mimeFirstFilter

+

Get the first filter in the MIME database.

+

+mime_filter_t *mimeFirstFilter (
+    mime_t *mime
+);

+

Parameters

+
+
mime
+
MIME database
+
+

Return Value

+

Filter or NULL

+

mimeFirstType

+

Get the first type in the MIME database.

+

+mime_type_t *mimeFirstType (
+    mime_t *mime
+);

+

Parameters

+
+
mime
+
MIME database
+
+

Return Value

+

Type or NULL

+

mimeLoad

+

Create a new MIME database from disk.

+

+mime_t *mimeLoad (
+    const char *pathname,
+    const char *filterpath
+);

+

Parameters

+
+
pathname
+
Directory to load
+
filterpath
+
Directory to load
+
+

Return Value

+

New MIME database

+

Discussion

+

This function uses mimeLoadFilters and mimeLoadTypes to +create a MIME database from a single directory.

+

mimeLoadFilters

+

Load filter definitions from disk.

+

+mime_t *mimeLoadFilters (
+    mime_t *mime,
+    const char *pathname,
+    const char *filterpath
+);

+

Parameters

+
+
mime
+
MIME database
+
pathname
+
Directory to load from
+
filterpath
+
Default filter program directory
+
+

Return Value

+

MIME database

+

Discussion

+

This function loads all of the .convs files from the specified directory. +Use mimeLoadTypes to load all types before you load the filters.

+

mimeLoadTypes

+

Load type definitions from disk.

+

+mime_t *mimeLoadTypes (
+    mime_t *mime,
+    const char *pathname
+);

+

Parameters

+
+
mime
+
MIME database or NULL to create a new one
+
pathname
+
Directory to load from
+
+

Return Value

+

MIME database

+

Discussion

+

This function loads all of the .types files from the specified directory. +Use mimeLoadFilters to load all filters after you load the types.

+

mimeNew

+

Create a new, empty MIME database.

+

+mime_t *mimeNew (void);

+

Return Value

+

MIME database

+

mimeNextFilter

+

Get the next filter in the MIME database.

+

+mime_filter_t *mimeNextFilter (
+    mime_t *mime
+);

+

Parameters

+
+
mime
+
MIME database
+
+

Return Value

+

Filter or NULL

+

mimeNextType

+

Get the next type in the MIME database.

+

+mime_type_t *mimeNextType (
+    mime_t *mime
+);

+

Parameters

+
+
mime
+
MIME database
+
+

Return Value

+

Type or NULL

+

mimeNumFilters

+

Next type

+

+int mimeNumFilters (
+    mime_t *mime
+);

+

Parameters

+
+
mime
+
MIME database
+
+

Return Value

+

Get the number of filters in a MIME database.

+

mimeNumTypes

+

MIME database

+

+int mimeNumTypes (
+    mime_t *mime
+);

+

Parameters

+
+
mime
+
MIME database
+
+

Return Value

+

Get the number of types in a MIME database.

+

mimeSetErrorCallback

+

Set the callback for error messages.

+

+void mimeSetErrorCallback (
+    mime_t *mime,
+    mime_error_cb_t cb,
+    void *ctx
+);

+

Parameters

+
+
mime
+
MIME database
+
cb
+
Callback function
+
ctx
+
Context pointer for callback
+
+

mimeType

+

Lookup a file type.

+

+mime_type_t *mimeType (
+    mime_t *mime,
+    const char *super,
+    const char *type
+);

+

Parameters

+
+
mime
+
MIME database
+
super
+
Super-type name
+
type
+
Type name
+
+

Return Value

+

Matching file type definition

+

Data Types

+

const

+

MIME Database

+

+typedef void (*constmime_error_cb_t)(void *; +

+

mime_filter_t

+

MIME Conversion Filter Data

+

+typedef struct _mime_filter_s mime_filter_t; +

+

mime_magic_t

+

MIME Magic Data

+

+typedef struct _mime_magic_s mime_magic_t; +

+

mime_op_t

+

Types/structures...

+

+typedef enum mime_op_t; +

+

mime_t

+

MIME Database

+

+typedef struct _mime_s mime_t; +

+

mime_type_t

+

MIME Type Data

+

+typedef struct _mime_type_s mime_type_t; +

+
+ + diff --git a/doc/help/api-overview.html b/doc/help/api-overview.html new file mode 100644 index 0000000000..278df7ca16 --- /dev/null +++ b/doc/help/api-overview.html @@ -0,0 +1,501 @@ + + + + + Introduction to CUPS Programming + + + + + + +
+ + +

Introduction to CUPS Programming

+ +
+ + + + + + + + + + + + + + + + +
Headerscups/cups.h
+ cups/array.h
+ cups/backend.h
+ cups/dir.h
+ cups/file.h
+ cups/ppd.h
+ cups/raster.h
+ cups/sidechannel.h
Libraries-lcups
+ -lcupsimage
See AlsoProgramming: Developing Raster Printer Drivers
+ Programming: Developing PostScript Printer Drivers
+ Programming: Filter and Backend Programming
+ Programming: Introduction to the PPD Compiler
+ Programming: Array API
+ Programming: CUPS API
+ Programming: File and Directory APIs
+ Programming: HTTP and IPP APIs
+ Programming: PPD API
+ Programming: Raster API
+ References: PPD Compiler Driver Information File Reference
+ Specifications: CUPS PPD Extensions
+

Contents

+ + + +

Overview

+ +

CUPS provides two libraries that interface with the different parts of the +printing system. The "cups" library provides all of the common application and +filter functions while the "cupsimage" library provides all of the imaging +functions used in raster printer drivers. The "cups" library functions are +accessed by including the <cups/cups.h> header, while +"cupsimage" functions are found in the <cups/raster.h> +header.

+ +

Compiling Programs

+ +

The CUPS libraries can be used from any C, C++, or Objective C program. +The method of compiling against the libraries varies depending on the +operating system and installation of CUPS. The following sections show how +to compile a simple program (shown below) in two common environments.

+ +

The following simple program lists the available printers on the system:

+ +
+#include <stdio.h>
+#include <cups/cups.h>
+
+int main(void)
+{
+  int i;
+  cups_dest_t *dests, *dest;
+  int num_dests = cupsGetDests(&dests);
+
+  for (i = num_dests, dest = dests; i > 0; i --, dest ++)
+  {
+    if (dest->instance)
+      printf("%s/%s\n", dest->name, dest->instance);
+    else
+      puts(dest->name);
+  }
+
+  return (0);
+}
+
+ +

Compiling with Xcode

+ +

In Xcode, choose New Project... from the File menu, +then select the Standard Tool project type under Command Line +Utility. Click Next and choose a project directory. Click +Next to create the project.

+ +

In the project window, double-click on the Targets group and +control-click on the simple target to show the context menu. Choose +Existing Framework... from the Add submenu. When the file +chooser sheet appears, press the / key and enter "/usr/lib". Scroll +down the file list and select the libcups.dylib file. Click the +Add button in the file chooser and attributes sheets.

+ +

In the project window, double-click on the main.c source file. +Replace the template source code with the listing above and save it. Click the +Build and Go button to build the sample program and run it.

+ +

Compiling with GCC

+ +

From the command-line, create a file called sample.c using your +favorite editor and then run the following command to compile it with GCC and +run it:

+ +
+gcc -o simple `cups-config --cflags` simple.c `cups-config --libs`
+./simple
+
+ +

The cups-config command provides the compiler flags +("cups-config --cflags") and libraries ("cups-config --libs") needed for the +local system.

+ +

Where to Go Next

+ +

If you are developing a print filter, driver, or backend, see the +Filter and Backend Programming +guide. Raster printer driver developers should also read the +Raster API reference.

+
+ + diff --git a/doc/help/api-ppd.html b/doc/help/api-ppd.html new file mode 100644 index 0000000000..b2978bce34 --- /dev/null +++ b/doc/help/api-ppd.html @@ -0,0 +1,2193 @@ + + + + + PPD API (DEPRECATED) + + + + + + +
+ + +

PPD API (DEPRECATED)

+ +
The PPD API is deprecated starting in CUPS 1.6. Please use the new Job Ticket APIs in the CUPS API documentation. These functions will be removed in a future release of CUPS.
+ +
+ + + + + + + + + + + + + + + + +
Headercups/ppd.h
Library-lcups
See AlsoProgramming: Introduction to CUPS Programming
+ Programming: CUPS API
+ Specifications: CUPS PPD Extensions
+

Contents

+ + + +

Overview

+ +
The PPD API is deprecated starting in CUPS 1.6. Please use the new Job Ticket APIs in the CUPS API documentation. These functions will be removed in a future release of CUPS.
+ +

The CUPS PPD API provides read-only access the data in PostScript Printer +Description ("PPD") files which are used for all printers with a driver. With +it you can obtain the data necessary to display printer options to users, mark +option choices and check for conflicting choices, and output marked choices in +PostScript output. The ppd_file_t +structure contains all of the information in a PPD file.

+ +
Note: + +

The CUPS PPD API uses the terms "option" and "choice" instead of the Adobe +terms "MainKeyword" and "OptionKeyword" to refer to specific printer options and +features. CUPS also treats option ("MainKeyword") and choice ("OptionKeyword") +values as case-insensitive strings, so option "InputSlot" and choice "Upper" +are equivalent to "inputslot" and "upper", respectively.

+
+ +

Loading a PPD File

+ +

The ppdOpenFile function "opens" a +PPD file and loads it into memory. For example, the following code opens the +current printer's PPD file in a CUPS filter:

+ +
+#include <cups/ppd.h>
+
+ppd_file_t *ppd = ppdOpenFile(getenv("PPD"));
+
+ +

The return value is a pointer to a new +ppd_file_t structure or NULL +if the PPD file does not exist or cannot be loaded. The +ppdClose function frees the memory used +by the structure:

+ +
+#include <cups/ppd.h>
+
+ppd_file_t *ppd;
+
+ppdClose(ppd);
+
+ +

Once closed, pointers to the ppd_file_t +structure and any data in it will no longer be valid.

+ +

Options and Groups

+ +

PPD files support multiple options, which are stored in arrays of +ppd_option_t and +ppd_choice_t structures.

+ +

Each option in turn is associated with a group stored in a +ppd_group_t structure. Groups can be +specified in the PPD file; if an option is not associated with a group +then it is put in an automatically-generated "General" group. Groups can also +have sub-groups, however CUPS currently ignores sub-groups because of past +abuses of this functionality.

+ +

Option choices are selected by marking them using one of three functions. The +first is ppdMarkDefaults which +selects all of the default options in the PPD file:

+ +
+#include <cups/ppd.h>
+
+ppd_file_t *ppd;
+
+ppdMarkDefaults(ppd);
+
+ +

The second is ppdMarkOption +which selects a single option choice in the PPD file. For example, the following +code selects the upper paper tray:

+ +
+#include <cups/ppd.h>
+
+ppd_file_t *ppd;
+
+ppdMarkOption(ppd, "InputSlot", "Upper");
+
+ +

The last function is +cupsMarkOptions which selects +multiple option choices in the PPD file from an array of CUPS options, mapping +IPP attributes like "media" and "sides" to their corresponding PPD options. You +typically use this function in a print filter with +cupsParseOptions and +ppdMarkDefaults to select all of +the option choices needed for the job, for example:

+ +
+#include <cups/ppd.h>
+
+ppd_file_t *ppd = ppdOpenFile(getenv("PPD"));
+cups_option_t *options = NULL;
+int num_options = cupsParseOptions(argv[5], 0, &options);
+
+ppdMarkDefaults(ppd);
+cupsMarkOptions(ppd, num_options, options);
+cupsFreeOptions(num_options, options);
+
+ +

Constraints

+ +

PPD files support specification of conflict conditions, called +constraints, between different options. Constraints are stored in an array of +ppd_const_t structures which specify +the options and choices that conflict with each other. The +ppdConflicts function tells you +how many of the selected options are incompatible. Since constraints are +normally specified in pairs, the returned value is typically an even number.

+ +

Page Sizes

+ +

Page sizes are special options which have physical dimensions and margins +associated with them. The size information is stored in +ppd_size_t structures and is available +by looking up the named size with the +ppdPageSize function. The page size and +margins are returned in units called points; there are 72 points per inch. If +you pass NULL for the size, the currently selected size is +returned:

+ +
+#include <cups/ppd.h>
+
+ppd_file_t *ppd;
+ppd_size_t *size = ppdPageSize(ppd, NULL);
+
+ +

Besides the standard page sizes listed in a PPD file, some printers +support variable or custom page sizes. Custom page sizes are supported if the +variables_sizes member of the +ppd_file_t structure is non-zero. +The custom_min, custom_max, and +custom_margins members of the +ppd_file_t structure define the limits +of the printable area. To get the resulting media size, use a page size string +of the form "Custom.widthxlength", where "width" and "length" are +in points. Custom page size names can also be specified in inches +("Custom.widthxheightin"), centimeters +("Custom.widthxheightcm"), or millimeters +("Custom.widthxheightmm"):

+ +
+#include <cups/ppd.h>
+
+ppd_file_t *ppd;
+
+/* Get an 576x720 point custom page size */
+ppd_size_t *size = ppdPageSize(ppd, "Custom.576x720");
+
+/* Get an 8x10 inch custom page size */
+ppd_size_t *size = ppdPageSize(ppd, "Custom.8x10in");
+
+/* Get a 100x200 millimeter custom page size */
+ppd_size_t *size = ppdPageSize(ppd, "Custom.100x200mm");
+
+/* Get a 12.7x34.5 centimeter custom page size */
+ppd_size_t *size = ppdPageSize(ppd, "Custom.12.7x34.5cm");
+
+ +

If the PPD does not support variable page sizes, the +ppdPageSize function will return +NULL.

+ +

Attributes

+ +

Every PPD file is composed of one or more attributes. Most of these +attributes are used to define groups, options, choices, and page sizes, +however several informational attributes may be present which you can access +in your program or filter. Attributes normally look like one of the following +examples in a PPD file:

+ +
+*name: "value"
+*name spec: "value"
+*name spec/text: "value"
+
+ +

The ppdFindAttr and +ppdFindNextAttr functions find the +first and next instances, respectively, of the named attribute with the given +"spec" string and return a ppd_attr_t +structure. If you provide a NULL specifier string, all attributes with the +given name will be returned. For example, the following code lists all of the +Product attributes in a PPD file:

+ +
+#include <cups/ppd.h>
+
+ppd_file_t *ppd;
+ppd_attr_t *attr;
+
+for (attr = ppdFindAttr(ppd, "Product", NULL);
+     attr != NULL;
+     attr = ppdFindNextAttr(ppd, "Product", NULL))
+  puts(attr->value);
+
+

Functions

+

 CUPS 1.4/Mac OS X 10.6 cupsGetConflicts

+

Get a list of conflicting options in a marked PPD.

+

+int cupsGetConflicts (
+    ppd_file_t *ppd,
+    const char *option,
+    const char *choice,
+    cups_option_t **options
+);

+

Parameters

+
+
ppd
+
PPD file
+
option
+
Option to test
+
choice
+
Choice to test
+
options
+
Conflicting options
+
+

Return Value

+

Number of conflicting options

+

Discussion

+

This function gets a list of options that would conflict if "option" and +"choice" were marked in the PPD. You would typically call this function +after marking the currently selected options in the PPD in order to +determine whether a new option selection would cause a conflict.
+
+The number of conflicting options are returned with "options" pointing to +the conflicting options. The returned option array must be freed using +cupsFreeOptions. + +

+

cupsMarkOptions

+

Mark command-line options in a PPD file.

+

+int cupsMarkOptions (
+    ppd_file_t *ppd,
+    int num_options,
+    cups_option_t *options
+);

+

Parameters

+
+
ppd
+
PPD file
+
num_options
+
Number of options
+
options
+
Options
+
+

Return Value

+

1 if conflicts exist, 0 otherwise

+

Discussion

+

This function maps the IPP "finishings", "media", "mirror", +"multiple-document-handling", "output-bin", "print-color-mode", +"print-quality", "printer-resolution", and "sides" attributes to their +corresponding PPD options and choices.

+

 CUPS 1.4/Mac OS X 10.6 cupsResolveConflicts

+

Resolve conflicts in a marked PPD.

+

+int cupsResolveConflicts (
+    ppd_file_t *ppd,
+    const char *option,
+    const char *choice,
+    int *num_options,
+    cups_option_t **options
+);

+

Parameters

+
+
ppd
+
PPD file
+
option
+
Newly selected option or NULL for none
+
choice
+
Newly selected choice or NULL for none
+
num_options
+
Number of additional selected options
+
options
+
Additional selected options
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

This function attempts to resolve any conflicts in a marked PPD, returning +a list of option changes that are required to resolve them. On input, +"num_options" and "options" contain any pending option changes that have +not yet been marked, while "option" and "choice" contain the most recent +selection which may or may not be in "num_options" or "options".
+
+On successful return, "num_options" and "options" are updated to contain +"option" and "choice" along with any changes required to resolve conflicts +specified in the PPD file and 1 is returned.
+
+If option conflicts cannot be resolved, "num_options" and "options" are not +changed and 0 is returned.
+
+When resolving conflicts, cupsResolveConflicts does not consider +changes to the current page size (media, PageSize, and +PageRegion) or to the most recent option specified in "option". +Thus, if the only way to resolve a conflict is to change the page size +or the option the user most recently changed, cupsResolveConflicts +will return 0 to indicate it was unable to resolve the conflicts.
+
+The cupsResolveConflicts function uses one of two sources of option +constraint information. The preferred constraint information is defined by +cupsUIConstraints and cupsUIResolver attributes - in this +case, the PPD file provides constraint resolution actions.
+
+The backup constraint information is defined by the +UIConstraints and NonUIConstraints attributes. These +constraints are resolved algorithmically by first selecting the default +choice for the conflicting option, then iterating over all possible choices +until a non-conflicting option choice is found. + +

+

ppdClose

+

Free all memory used by the PPD file.

+

+void ppdClose (
+    ppd_file_t *ppd
+);

+

Parameters

+
+
ppd
+
PPD file record
+
+

ppdCollect

+

Collect all marked options that reside in the specified +section.

+

+int ppdCollect (
+    ppd_file_t *ppd,
+    ppd_section_t section,
+    ppd_choice_t ***choices
+);

+

Parameters

+
+
ppd
+
PPD file data
+
section
+
Section to collect
+
choices
+
Pointers to choices
+
+

Return Value

+

Number of options marked

+

Discussion

+

The choices array should be freed using free when you are +finished with it.

+

 CUPS 1.2/Mac OS X 10.5 ppdCollect2

+

Collect all marked options that reside in the +specified section and minimum order.

+

+int ppdCollect2 (
+    ppd_file_t *ppd,
+    ppd_section_t section,
+    float min_order,
+    ppd_choice_t ***choices
+);

+

Parameters

+
+
ppd
+
PPD file data
+
section
+
Section to collect
+
min_order
+
Minimum OrderDependency value
+
choices
+
Pointers to choices
+
+

Return Value

+

Number of options marked

+

Discussion

+

The choices array should be freed using free when you are +finished with it. + +

+

ppdConflicts

+

Check to see if there are any conflicts among the +marked option choices.

+

+int ppdConflicts (
+    ppd_file_t *ppd
+);

+

Parameters

+
+
ppd
+
PPD to check
+
+

Return Value

+

Number of conflicts found

+

Discussion

+

The returned value is the same as returned by ppdMarkOption.

+

ppdEmit

+

Emit code for marked options to a file.

+

+int ppdEmit (
+    ppd_file_t *ppd,
+    FILE *fp,
+    ppd_section_t section
+);

+

Parameters

+
+
ppd
+
PPD file record
+
fp
+
File to write to
+
section
+
Section to write
+
+

Return Value

+

0 on success, -1 on failure

+

 CUPS 1.2/Mac OS X 10.5 ppdEmitAfterOrder

+

Emit a subset of the code for marked options to a file.

+

+int ppdEmitAfterOrder (
+    ppd_file_t *ppd,
+    FILE *fp,
+    ppd_section_t section,
+    int limit,
+    float min_order
+);

+

Parameters

+
+
ppd
+
PPD file record
+
fp
+
File to write to
+
section
+
Section to write
+
limit
+
Non-zero to use min_order
+
min_order
+
Lowest OrderDependency
+
+

Return Value

+

0 on success, -1 on failure

+

Discussion

+

When "limit" is non-zero, this function only emits options whose +OrderDependency value is greater than or equal to "min_order".
+
+When "limit" is zero, this function is identical to ppdEmit(). + +

+

ppdEmitFd

+

Emit code for marked options to a file.

+

+int ppdEmitFd (
+    ppd_file_t *ppd,
+    int fd,
+    ppd_section_t section
+);

+

Parameters

+
+
ppd
+
PPD file record
+
fd
+
File to write to
+
section
+
Section to write
+
+

Return Value

+

0 on success, -1 on failure

+

ppdEmitJCL

+

Emit code for JCL options to a file.

+

+int ppdEmitJCL (
+    ppd_file_t *ppd,
+    FILE *fp,
+    int job_id,
+    const char *user,
+    const char *title
+);

+

Parameters

+
+
ppd
+
PPD file record
+
fp
+
File to write to
+
job_id
+
Job ID
+
user
+
Username
+
title
+
Title
+
+

Return Value

+

0 on success, -1 on failure

+

 CUPS 1.2/Mac OS X 10.5 ppdEmitJCLEnd

+

Emit JCLEnd code to a file.

+

+int ppdEmitJCLEnd (
+    ppd_file_t *ppd,
+    FILE *fp
+);

+

Parameters

+
+
ppd
+
PPD file record
+
fp
+
File to write to
+
+

Return Value

+

0 on success, -1 on failure

+

 CUPS 1.2/Mac OS X 10.5 ppdEmitString

+

Get a string containing the code for marked options.

+

+char *ppdEmitString (
+    ppd_file_t *ppd,
+    ppd_section_t section,
+    float min_order
+);

+

Parameters

+
+
ppd
+
PPD file record
+
section
+
Section to write
+
min_order
+
Lowest OrderDependency
+
+

Return Value

+

String containing option code or NULL if there is no option code

+

Discussion

+

When "min_order" is greater than zero, this function only includes options +whose OrderDependency value is greater than or equal to "min_order". +Otherwise, all options in the specified section are included in the +returned string.
+
+The return string is allocated on the heap and should be freed using +free when you are done with it. + +

+

 CUPS 1.1.19/Mac OS X 10.3 ppdErrorString

+

Returns the text assocated with a status.

+

+const char *ppdErrorString (
+    ppd_status_t status
+);

+

Parameters

+
+
status
+
PPD status
+
+

Return Value

+

Status string

+

 CUPS 1.1.19/Mac OS X 10.3 ppdFindAttr

+

Find the first matching attribute.

+

+ppd_attr_t *ppdFindAttr (
+    ppd_file_t *ppd,
+    const char *name,
+    const char *spec
+);

+

Parameters

+
+
ppd
+
PPD file data
+
name
+
Attribute name
+
spec
+
Specifier string or NULL
+
+

Return Value

+

Attribute or NULL if not found

+

ppdFindChoice

+

Return a pointer to an option choice.

+

+ppd_choice_t *ppdFindChoice (
+    ppd_option_t *o,
+    const char *choice
+);

+

Parameters

+
+
o
+
Pointer to option
+
choice
+
Name of choice
+
+

Return Value

+

Choice pointer or NULL

+

 CUPS 1.2/Mac OS X 10.5 ppdFindCustomOption

+

Find a custom option.

+

+ppd_coption_t *ppdFindCustomOption (
+    ppd_file_t *ppd,
+    const char *keyword
+);

+

Parameters

+
+
ppd
+
PPD file
+
keyword
+
Custom option name
+
+

Return Value

+

Custom option or NULL

+

 CUPS 1.2/Mac OS X 10.5 ppdFindCustomParam

+

Find a parameter for a custom option.

+

+ppd_cparam_t *ppdFindCustomParam (
+    ppd_coption_t *opt,
+    const char *name
+);

+

Parameters

+
+
opt
+
Custom option
+
name
+
Parameter name
+
+

Return Value

+

Custom parameter or NULL

+

ppdFindMarkedChoice

+

Return the marked choice for the specified option.

+

+ppd_choice_t *ppdFindMarkedChoice (
+    ppd_file_t *ppd,
+    const char *option
+);

+

Parameters

+
+
ppd
+
PPD file
+
option
+
Keyword/option name
+
+

Return Value

+

Pointer to choice or NULL

+

 CUPS 1.1.19/Mac OS X 10.3 ppdFindNextAttr

+

Find the next matching attribute.

+

+ppd_attr_t *ppdFindNextAttr (
+    ppd_file_t *ppd,
+    const char *name,
+    const char *spec
+);

+

Parameters

+
+
ppd
+
PPD file data
+
name
+
Attribute name
+
spec
+
Specifier string or NULL
+
+

Return Value

+

Attribute or NULL if not found

+

ppdFindOption

+

Return a pointer to the specified option.

+

+ppd_option_t *ppdFindOption (
+    ppd_file_t *ppd,
+    const char *option
+);

+

Parameters

+
+
ppd
+
PPD file data
+
option
+
Option/Keyword name
+
+

Return Value

+

Pointer to option or NULL

+

 CUPS 1.2/Mac OS X 10.5 ppdFirstCustomParam

+

Return the first parameter for a custom option.

+

+ppd_cparam_t *ppdFirstCustomParam (
+    ppd_coption_t *opt
+);

+

Parameters

+
+
opt
+
Custom option
+
+

Return Value

+

Custom parameter or NULL

+

 CUPS 1.2/Mac OS X 10.5 ppdFirstOption

+

Return the first option in the PPD file.

+

+ppd_option_t *ppdFirstOption (
+    ppd_file_t *ppd
+);

+

Parameters

+
+
ppd
+
PPD file
+
+

Return Value

+

First option or NULL

+

Discussion

+

Options are returned from all groups in ascending alphanumeric order. + +

+

 CUPS 1.4/Mac OS X 10.6 ppdInstallableConflict

+

Test whether an option choice conflicts with +an installable option.

+

+int ppdInstallableConflict (
+    ppd_file_t *ppd,
+    const char *option,
+    const char *choice
+);

+

Parameters

+
+
ppd
+
PPD file
+
option
+
Option
+
choice
+
Choice
+
+

Return Value

+

1 if conflicting, 0 if not conflicting

+

Discussion

+

This function tests whether a particular option choice is available based +on constraints against options in the "InstallableOptions" group. + +

+

ppdIsMarked

+

Check to see if an option is marked.

+

+int ppdIsMarked (
+    ppd_file_t *ppd,
+    const char *option,
+    const char *choice
+);

+

Parameters

+
+
ppd
+
PPD file data
+
option
+
Option/Keyword name
+
choice
+
Choice name
+
+

Return Value

+

Non-zero if option is marked

+

 CUPS 1.1.19/Mac OS X 10.3 ppdLastError

+

Return the status from the last ppdOpen*().

+

+ppd_status_t ppdLastError (
+    int *line
+);

+

Parameters

+
+
line
+
Line number
+
+

Return Value

+

Status code

+

 CUPS 1.2/Mac OS X 10.5 ppdLocalize

+

Localize the PPD file to the current locale.

+

+int ppdLocalize (
+    ppd_file_t *ppd
+);

+

Parameters

+
+
ppd
+
PPD file
+
+

Return Value

+

0 on success, -1 on error

+

Discussion

+

All groups, options, and choices are localized, as are ICC profile +descriptions, printer presets, and custom option parameters. Each +localized string uses the UTF-8 character encoding. + +

+

ppdLocalizeAttr

+

Localize an attribute.

+

+ppd_attr_t *ppdLocalizeAttr (
+    ppd_file_t *ppd,
+    const char *keyword,
+    const char *spec
+);

+

Parameters

+
+
ppd
+
PPD file
+
keyword
+
Main keyword
+
spec
+
Option keyword or NULL for none
+
+

Return Value

+

Localized attribute or NULL if none exists

+

Discussion

+

This function uses the current locale to find the localized attribute for +the given main and option keywords. If no localized version of the +attribute exists for the current locale, the unlocalized version is returned.

+

 CUPS 1.3/Mac OS X 10.5 ppdLocalizeIPPReason

+

Get the localized version of a cupsIPPReason +attribute.

+

+const char *ppdLocalizeIPPReason (
+    ppd_file_t *ppd,
+    const char *reason,
+    const char *scheme,
+    char *buffer,
+    size_t bufsize
+);

+

Parameters

+
+
ppd
+
PPD file
+
reason
+
IPP reason keyword to look up
+
scheme
+
URI scheme or NULL for text
+
buffer
+
Value buffer
+
bufsize
+
Size of value buffer
+
+

Return Value

+

Value or NULL if not found

+

Discussion

+

This function uses the current locale to find the corresponding reason +text or URI from the attribute value. If "scheme" is NULL or "text", +the returned value contains human-readable (UTF-8) text from the translation +string or attribute value. Otherwise the corresponding URI is returned.
+
+If no value of the requested scheme can be found, NULL is returned. + +

+

 CUPS 1.4/Mac OS X 10.6 ppdLocalizeMarkerName

+

Get the localized version of a marker-names +attribute value.

+

+const char *ppdLocalizeMarkerName (
+    ppd_file_t *ppd,
+    const char *name
+);

+

Parameters

+
+
ppd
+
PPD file
+
name
+
Marker name to look up
+
+

Return Value

+

Value or NULL if not found

+

Discussion

+

This function uses the current locale to find the corresponding name +text from the attribute value. If no localized text for the requested +name can be found, NULL is returned. + +

+

ppdMarkDefaults

+

Mark all default options in the PPD file.

+

+void ppdMarkDefaults (
+    ppd_file_t *ppd
+);

+

Parameters

+
+
ppd
+
PPD file record
+
+

ppdMarkOption

+

Mark an option in a PPD file and return the number of +conflicts.

+

+int ppdMarkOption (
+    ppd_file_t *ppd,
+    const char *option,
+    const char *choice
+);

+

Parameters

+
+
ppd
+
PPD file record
+
option
+
Keyword
+
choice
+
Option name
+
+

Return Value

+

Number of conflicts

+

 CUPS 1.2/Mac OS X 10.5 ppdNextCustomParam

+

Return the next parameter for a custom option.

+

+ppd_cparam_t *ppdNextCustomParam (
+    ppd_coption_t *opt
+);

+

Parameters

+
+
opt
+
Custom option
+
+

Return Value

+

Custom parameter or NULL

+

 CUPS 1.2/Mac OS X 10.5 ppdNextOption

+

Return the next option in the PPD file.

+

+ppd_option_t *ppdNextOption (
+    ppd_file_t *ppd
+);

+

Parameters

+
+
ppd
+
PPD file
+
+

Return Value

+

Next option or NULL

+

Discussion

+

Options are returned from all groups in ascending alphanumeric order. + +

+

ppdOpen

+

Read a PPD file into memory.

+

+ppd_file_t *ppdOpen (
+    FILE *fp
+);

+

Parameters

+
+
fp
+
File to read from
+
+

Return Value

+

PPD file record

+

 CUPS 1.2/Mac OS X 10.5 ppdOpen2

+

Read a PPD file into memory.

+

+ppd_file_t *ppdOpen2 (
+    cups_file_t *fp
+);

+

Parameters

+
+
fp
+
File to read from
+
+

Return Value

+

PPD file record or NULL if the PPD file could not be opened.

+

ppdOpenFd

+

Read a PPD file into memory.

+

+ppd_file_t *ppdOpenFd (
+    int fd
+);

+

Parameters

+
+
fd
+
File to read from
+
+

Return Value

+

PPD file record or NULL if the PPD file could not be opened.

+

ppdOpenFile

+

Read a PPD file into memory.

+

+ppd_file_t *ppdOpenFile (
+    const char *filename
+);

+

Parameters

+
+
filename
+
File to read from
+
+

Return Value

+

PPD file record or NULL if the PPD file could not be opened.

+

ppdPageLength

+

Get the page length for the given size.

+

+float ppdPageLength (
+    ppd_file_t *ppd,
+    const char *name
+);

+

Parameters

+
+
ppd
+
PPD file
+
name
+
Size name
+
+

Return Value

+

Length of page in points or 0.0

+

ppdPageSize

+

Get the page size record for the given size.

+

+ppd_size_t *ppdPageSize (
+    ppd_file_t *ppd,
+    const char *name
+);

+

Parameters

+
+
ppd
+
PPD file record
+
name
+
Size name
+
+

Return Value

+

Size record for page or NULL

+

 CUPS 1.4/Mac OS X 10.6 ppdPageSizeLimits

+

Return the custom page size limits.

+

+int ppdPageSizeLimits (
+    ppd_file_t *ppd,
+    ppd_size_t *minimum,
+    ppd_size_t *maximum
+);

+

Parameters

+
+
ppd
+
PPD file record
+
minimum
+
Minimum custom size
+
maximum
+
Maximum custom size
+
+

Return Value

+

1 if custom sizes are supported, 0 otherwise

+

Discussion

+

This function returns the minimum and maximum custom page sizes and printable +areas based on the currently-marked (selected) options.
+
+If the specified PPD file does not support custom page sizes, both +"minimum" and "maximum" are filled with zeroes. + +

+

ppdPageWidth

+

Get the page width for the given size.

+

+float ppdPageWidth (
+    ppd_file_t *ppd,
+    const char *name
+);

+

Parameters

+
+
ppd
+
PPD file record
+
name
+
Size name
+
+

Return Value

+

Width of page in points or 0.0

+

 CUPS 1.1.20/Mac OS X 10.4 ppdSetConformance

+

Set the conformance level for PPD files.

+

+void ppdSetConformance (
+    ppd_conform_t c
+);

+

Parameters

+
+
c
+
Conformance level
+
+

Data Types

+

 CUPS 1.1.19/Mac OS X 10.3 ppd_attr_t

+

PPD Attribute Structure

+

+typedef struct ppd_attr_s ppd_attr_t; +

+

ppd_choice_t

+

Option choices

+

+typedef struct ppd_choice_s ppd_choice_t; +

+

 CUPS 1.1.19/Mac OS X 10.3 ppd_conform_t

+

Conformance Levels

+

+typedef enum ppd_conform_e ppd_conform_t; +

+

ppd_const_t

+

Constraints

+

+typedef struct ppd_const_s ppd_const_t; +

+

 CUPS 1.2/Mac OS X 10.5 ppd_coption_t

+

Custom Option

+

+typedef struct ppd_coption_s ppd_coption_t; +

+

 CUPS 1.2/Mac OS X 10.5 ppd_cparam_t

+

Custom Parameter

+

+typedef struct ppd_cparam_s ppd_cparam_t; +

+

 CUPS 1.2/Mac OS X 10.5 ppd_cplimit_t

+

Custom Parameter Limit

+

+typedef union ppd_cplimit_u ppd_cplimit_t; +

+

 CUPS 1.2/Mac OS X 10.5 ppd_cptype_t

+

Custom Parameter Type

+

+typedef enum ppd_cptype_e ppd_cptype_t; +

+

 CUPS 1.2/Mac OS X 10.5 ppd_cpvalue_t

+

Custom Parameter Value

+

+typedef union ppd_cpvalue_u ppd_cpvalue_t; +

+

ppd_cs_t

+

Colorspaces

+

+typedef enum ppd_cs_e ppd_cs_t; +

+

ppd_emul_t

+

Emulators

+

+typedef struct ppd_emul_s ppd_emul_t; +

+

ppd_file_t

+

PPD File

+

+typedef struct ppd_file_s ppd_file_t; +

+

ppd_group_t

+

Groups

+

+typedef struct ppd_group_s ppd_group_t; +

+

ppd_option_t

+

Options

+

+typedef struct ppd_option_s ppd_option_t; +

+

ppd_profile_t

+

sRGB Color Profiles

+

+typedef struct ppd_profile_s ppd_profile_t; +

+

ppd_section_t

+

Order dependency sections

+

+typedef enum ppd_section_e ppd_section_t; +

+

ppd_size_t

+

Page Sizes

+

+typedef struct ppd_size_s ppd_size_t; +

+

 CUPS 1.1.19/Mac OS X 10.3 ppd_status_t

+

Status Codes

+

+typedef enum ppd_status_e ppd_status_t; +

+

ppd_ui_t

+

UI Types

+

+typedef enum ppd_ui_e ppd_ui_t; +

+

Structures

+

 CUPS 1.1.19/Mac OS X 10.3 ppd_attr_s

+

PPD Attribute Structure

+

struct ppd_attr_s {
+    char name[PPD_MAX_NAME];
+    char spec[PPD_MAX_NAME];
+    char text[PPD_MAX_TEXT];
+    char *value;
+};

+

Members

+
+
name[PPD_MAX_NAME]
+
Name of attribute (cupsXYZ)
+
spec[PPD_MAX_NAME]
+
Specifier string, if any
+
text[PPD_MAX_TEXT]
+
Human-readable text, if any
+
value
+
Value string
+
+

ppd_choice_s

+

Option choices

+

struct ppd_choice_s {
+    char choice[PPD_MAX_NAME];
+    char *code;
+    char marked;
+    ppd_option_t *option;
+    char text[PPD_MAX_TEXT];
+};

+

Members

+
+
choice[PPD_MAX_NAME]
+
Computer-readable option name
+
code
+
Code to send for this option
+
marked
+
0 if not selected, 1 otherwise
+
option
+
Pointer to parent option structure
+
text[PPD_MAX_TEXT]
+
Human-readable option name
+
+

ppd_const_s

+

Constraints

+

struct ppd_const_s {
+    char choice1[PPD_MAX_NAME];
+    char choice2[PPD_MAX_NAME];
+    char option1[PPD_MAX_NAME];
+    char option2[PPD_MAX_NAME];
+};

+

Members

+
+
choice1[PPD_MAX_NAME]
+
First option/choice (blank for all)
+
choice2[PPD_MAX_NAME]
+
Second option/choice (blank for all)
+
option1[PPD_MAX_NAME]
+
First keyword
+
option2[PPD_MAX_NAME]
+
Second keyword
+
+

 CUPS 1.2/Mac OS X 10.5 ppd_coption_s

+

Custom Option

+

struct ppd_coption_s {
+    char keyword[PPD_MAX_NAME];
+    int marked;
+    ppd_option_t *option;
+    cups_array_t *params;
+};

+

Members

+
+
keyword[PPD_MAX_NAME]
+
Name of option that is being extended...
+
marked
+
Extended option is marked
+
option
+
Option that is being extended...
+
params
+
Parameters
+
+

 CUPS 1.2/Mac OS X 10.5 ppd_cparam_s

+

Custom Parameter

+

struct ppd_cparam_s {
+    ppd_cpvalue_t current;
+    ppd_cplimit_t minimum, maximum;
+    char name[PPD_MAX_NAME];
+    int order;
+    char text[PPD_MAX_TEXT];
+    ppd_cptype_t type;
+};

+

Members

+
+
current
+
Current value
+
maximum
+
Maximum value
+
name[PPD_MAX_NAME]
+
Parameter name
+
order
+
Order (0 to N)
+
text[PPD_MAX_TEXT]
+
Human-readable text
+
type
+
Parameter type
+
+

ppd_emul_s

+

Emulators

+

struct ppd_emul_s {
+    char name[PPD_MAX_NAME];
+    char *start;
+    char *stop;
+};

+

Members

+
+
name[PPD_MAX_NAME]
+
Emulator name
+
start
+
Code to switch to this emulation
+
stop
+
Code to stop this emulation
+
+

ppd_file_s

+

PPD File

+

struct ppd_file_s {
+    int accurate_screens;
+    int color_device;
+    ppd_cs_t colorspace;
+    ppd_const_t *consts;
+    int contone_only;
+    float custom_margins[4];
+    float custom_max[2];
+    float custom_min[2];
+    ppd_emul_t *emulations;
+    char **filters;
+    int flip_duplex;
+    char **fonts;
+    ppd_group_t *groups;
+    char *jcl_begin;
+    char *jcl_end;
+    char *jcl_ps;
+    int landscape;
+    char *lang_encoding;
+    char *lang_version;
+    int language_level;
+    int manual_copies;
+    char *manufacturer;
+    int model_number;
+    char *modelname;
+    char *nickname;
+    int num_consts;
+    int num_emulations;
+    int num_filters;
+    int num_fonts;
+    int num_groups;
+    int num_profiles;
+    int num_sizes;
+    char *patches;
+    char *pcfilename;
+    char *product;
+    ppd_profile_t *profiles;
+    char *protocols;
+    char *shortnickname;
+    ppd_size_t *sizes;
+    int throughput;
+    char *ttrasterizer;
+    int variable_sizes;
+};

+

Members

+
+
accurate_screens
+
1 = supports accurate screens, 0 = not
+
color_device
+
1 = color device, 0 = grayscale
+
colorspace
+
Default colorspace
+
consts
+
UI/Non-UI constraints
+
contone_only
+
1 = continuous tone only, 0 = not
+
custom_margins[4]
+
Margins around page
+
custom_max[2]
+
Maximum variable page size
+
custom_min[2]
+
Minimum variable page size
+
emulations
+
Emulations and the code to invoke them
+
filters
+
Filter strings...
+
flip_duplex  DEPRECATED 
+
1 = Flip page for back sides
+
fonts
+
Pre-loaded fonts
+
groups
+
UI groups
+
jcl_begin
+
Start JCL commands
+
jcl_end
+
End JCL commands
+
jcl_ps
+
Enter PostScript interpreter
+
landscape
+
-90 or 90
+
lang_encoding
+
Language encoding
+
lang_version
+
Language version (English, Spanish, etc.)
+
language_level
+
Language level of device
+
manual_copies
+
1 = Copies done manually, 0 = hardware
+
manufacturer
+
Manufacturer name
+
model_number
+
Device-specific model number
+
modelname
+
Model name (general)
+
nickname
+
Nickname (specific)
+
num_consts
+
Number of UI/Non-UI constraints
+
num_emulations
+
Number of emulations supported
+
num_filters
+
Number of filters
+
num_fonts
+
Number of pre-loaded fonts
+
num_groups
+
Number of UI groups
+
num_profiles  DEPRECATED 
+
Number of sRGB color profiles
+
num_sizes
+
Number of page sizes
+
patches
+
Patch commands to be sent to printer
+
pcfilename  CUPS 1.1.19/Mac OS X 10.3 
+
PCFileName string
+
product
+
Product name (from PS RIP/interpreter)
+
profiles  DEPRECATED 
+
sRGB color profiles
+
protocols  CUPS 1.1.19/Mac OS X 10.3 
+
Protocols (BCP, TBCP) string
+
shortnickname
+
Short version of nickname
+
sizes
+
Page sizes
+
throughput
+
Pages per minute
+
ttrasterizer
+
Truetype rasterizer
+
variable_sizes
+
1 = supports variable sizes, 0 = doesn't
+
+

ppd_group_s

+

Groups

+

struct ppd_group_s {
+    char text[PPD_MAX_TEXT - PPD_MAX_NAME];
+    char name[PPD_MAX_NAME];
+    int num_options;
+    int num_subgroups;
+    ppd_option_t *options;
+    struct ppd_group_s *subgroups;
+};

+

Members

+
+
PPD_MAX_NAME]
+
Human-readable group name
+
name[PPD_MAX_NAME]  CUPS 1.1.18/Mac OS X 10.3 
+
Group name
+
num_options
+
Number of options
+
num_subgroups
+
Number of sub-groups
+
options
+
Options
+
subgroups
+
Sub-groups (max depth = 1)
+
+

ppd_option_s

+

Options

+

struct ppd_option_s {
+    ppd_choice_t *choices;
+    char conflicted;
+    char defchoice[PPD_MAX_NAME];
+    char keyword[PPD_MAX_NAME];
+    int num_choices;
+    float order;
+    ppd_section_t section;
+    char text[PPD_MAX_TEXT];
+    ppd_ui_t ui;
+};

+

Members

+
+
choices
+
Option choices
+
conflicted
+
0 if no conflicts exist, 1 otherwise
+
defchoice[PPD_MAX_NAME]
+
Default option choice
+
keyword[PPD_MAX_NAME]
+
Option keyword name ("PageSize", etc.)
+
num_choices
+
Number of option choices
+
order
+
Order number
+
section
+
Section for command
+
text[PPD_MAX_TEXT]
+
Human-readable text
+
ui
+
Type of UI option
+
+

ppd_profile_s

+

sRGB Color Profiles

+

struct ppd_profile_s {
+    float density;
+    float gamma;
+    float matrix[3][3];
+    char media_type[PPD_MAX_NAME];
+    char resolution[PPD_MAX_NAME];
+};

+

Members

+
+
density
+
Ink density to use
+
gamma
+
Gamma correction to use
+
matrix[3][3]
+
Transform matrix
+
media_type[PPD_MAX_NAME]
+
Media type or "-"
+
resolution[PPD_MAX_NAME]
+
Resolution or "-"
+
+

ppd_size_s

+

Page Sizes

+

struct ppd_size_s {
+    float bottom;
+    float left;
+    float length;
+    int marked;
+    char name[PPD_MAX_NAME];
+    float right;
+    float top;
+    float width;
+};

+

Members

+
+
bottom
+
Bottom printable margin in points
+
left
+
Left printable margin in points
+
length
+
Length of media in points
+
marked
+
Page size selected?
+
name[PPD_MAX_NAME]
+
Media size option
+
right
+
Right printable margin in points
+
top
+
Top printable margin in points
+
width
+
Width of media in points
+
+

Unions

+

 CUPS 1.2/Mac OS X 10.5 ppd_cplimit_u

+

Custom Parameter Limit

+

union ppd_cplimit_u {
+    float custom_curve;
+    int custom_int;
+    float custom_invcurve;
+    int custom_passcode;
+    int custom_password;
+    float custom_points;
+    float custom_real;
+    int custom_string;
+};

+

Members

+
+
custom_curve
+
Gamma value
+
custom_int
+
Integer value
+
custom_invcurve
+
Gamma value
+
custom_passcode
+
Passcode length
+
custom_password
+
Password length
+
custom_points
+
Measurement value
+
custom_real
+
Real value
+
custom_string
+
String length
+
+

 CUPS 1.2/Mac OS X 10.5 ppd_cpvalue_u

+

Custom Parameter Value

+

union ppd_cpvalue_u {
+    float custom_curve;
+    int custom_int;
+    float custom_invcurve;
+    char *custom_passcode;
+    char *custom_password;
+    float custom_points;
+    float custom_real;
+    char *custom_string;
+};

+

Members

+
+
custom_curve
+
Gamma value
+
custom_int
+
Integer value
+
custom_invcurve
+
Gamma value
+
custom_passcode
+
Passcode value
+
custom_password
+
Password value
+
custom_points
+
Measurement value
+
custom_real
+
Real value
+
custom_string
+
String value
+
+

Constants

+

 CUPS 1.1.19/Mac OS X 10.3 ppd_conform_e

+

Conformance Levels

+

Constants

+
+
PPD_CONFORM_RELAXED
+
Relax whitespace and control char
+
PPD_CONFORM_STRICT
+
Require strict conformance
+
+

ppd_cs_e

+

Colorspaces

+

Constants

+
+
PPD_CS_CMY
+
CMY colorspace
+
PPD_CS_CMYK
+
CMYK colorspace
+
PPD_CS_GRAY
+
Grayscale colorspace
+
PPD_CS_N
+
DeviceN colorspace
+
PPD_CS_RGB
+
RGB colorspace
+
PPD_CS_RGBK
+
RGBK (K = gray) colorspace
+
+

ppd_section_e

+

Order dependency sections

+

Constants

+
+
PPD_ORDER_ANY
+
Option code can be anywhere in the file
+
PPD_ORDER_DOCUMENT
+
... must be in the DocumentSetup section
+
PPD_ORDER_EXIT
+
... must be sent prior to the document
+
PPD_ORDER_JCL
+
... must be sent as a JCL command
+
PPD_ORDER_PAGE
+
... must be in the PageSetup section
+
PPD_ORDER_PROLOG
+
... must be in the Prolog section
+
+

 CUPS 1.1.19/Mac OS X 10.3 ppd_status_e

+

Status Codes

+

Constants

+
+
PPD_ALLOC_ERROR
+
Memory allocation error
+
PPD_BAD_CUSTOM_PARAM
+
Bad custom parameter
+
PPD_BAD_OPEN_GROUP
+
Bad OpenGroup
+
PPD_BAD_OPEN_UI
+
Bad OpenUI/JCLOpenUI
+
PPD_BAD_ORDER_DEPENDENCY
+
Bad OrderDependency
+
PPD_BAD_UI_CONSTRAINTS
+
Bad UIConstraints
+
PPD_BAD_VALUE
+
Bad value string
+
PPD_FILE_OPEN_ERROR
+
Unable to open PPD file
+
PPD_ILLEGAL_CHARACTER
+
Illegal control character
+
PPD_ILLEGAL_MAIN_KEYWORD
+
Illegal main keyword string
+
PPD_ILLEGAL_OPTION_KEYWORD
+
Illegal option keyword string
+
PPD_ILLEGAL_TRANSLATION
+
Illegal translation string
+
PPD_ILLEGAL_WHITESPACE
+
Illegal whitespace character
+
PPD_INTERNAL_ERROR
+
Internal error
+
PPD_LINE_TOO_LONG
+
Line longer than 255 chars
+
PPD_MISSING_ASTERISK
+
Missing asterisk in column 0
+
PPD_MISSING_CLOSE_GROUP
+
Missing CloseGroup
+
PPD_MISSING_OPTION_KEYWORD
+
Missing option keyword
+
PPD_MISSING_PPDADOBE4
+
Missing PPD-Adobe-4.x header
+
PPD_MISSING_VALUE
+
Missing value string
+
PPD_NESTED_OPEN_GROUP
+
OpenGroup without a CloseGroup first
+
PPD_NESTED_OPEN_UI
+
OpenUI/JCLOpenUI without a CloseUI/JCLCloseUI first
+
PPD_NULL_FILE
+
NULL PPD file pointer
+
PPD_OK
+
OK
+
+

ppd_ui_e

+

UI Types

+

Constants

+
+
PPD_UI_BOOLEAN
+
True or False option
+
PPD_UI_PICKMANY
+
Pick zero or more from a list
+
PPD_UI_PICKONE
+
Pick one from a list
+
+
+ + diff --git a/doc/help/api-ppdc.html b/doc/help/api-ppdc.html new file mode 100644 index 0000000000..cb15f61e65 --- /dev/null +++ b/doc/help/api-ppdc.html @@ -0,0 +1,2197 @@ + + + + + PPD Compiler API + + + + + + +
+ + +

PPD Compiler API

+ +
+ + + + + + + + + + + + + + + + +
Headercups/ppdc.h
Library-lcupsppdc
See AlsoProgramming: Introduction to CUPS Programming
+

Contents

+ + + +

Overview

+ +

The PPD Compiler API provides access to CUPS driver information files and +methods for generating and importing PPD files.

+

Classes

+

ppdcArray

+

// Shared Array

+

class ppdcArray : public ppdcShared {
+  public:
+    int count, alloc, current;
+    ppdcShared **data;
+};

+

Members

+
+
current
+
Current element
+
data
+
Elements
+
+

PPDC_NAME

+

+

+void PPDC_NAME (
+    void add(ppdcShared *d
+);

+
Parameters
+
+
d
+
+

add

+

+

+void add (
+    ppdcShared *d
+);

+
Parameters
+
+
d
+
+

first

+

Return the first element in the array.

+

+ppdcShared *first (void);

+
Return Value
+

+

next

+

Return the next element in the array.

+

+ppdcShared *next (void);

+
Return Value
+

+

ppdcArray

+

+

+void ppdcArray (
+    ppdcArray *a
+);

+
Parameters
+
+
a
+
+

remove

+

+

+void remove (
+    ppdcShared *d
+);

+
Parameters
+
+
d
+
Data element
+
+

~ppdcArray

+

+

+void ~ppdcArray (void);

+

ppdcAttr

+

// Attribute

+

class ppdcAttr : public ppdcShared {
+  public:
+    bool localizable;
+    ppdcString *name, *selector, *text, *value;
+};

+

Members

+
+
localizable
+
Should this attribute be localized?
+
value
+
Value string
+
+

ppdcAttr

+

+

+void ppdcAttr (
+    const char *n,
+    const char *s,
+    const char *t,
+    const char *v,
+    bool loc
+);

+
Parameters
+
+
n
+
Name
+
s
+
Spec string
+
t
+
Human-readable text
+
v
+
Value
+
loc
+
Localize this attribute?
+
+

~ppdcAttr

+

+

+void ~ppdcAttr (void);

+

ppdcCatalog

+

// Translation catalog

+

class ppdcCatalog : public ppdcShared {
+  public:
+    ppdcString *filename;
+    ppdcString *locale;
+    ppdcArray *messages;
+};

+

Members

+
+
filename
+
Name of translation file
+
locale
+
Name of locale
+
messages
+
Array of translation messages
+
+

PPDC_NAME

+

+

+void PPDC_NAME (
+    void add_message(const char *id,
+    const char *string = NULL
+);

+
Parameters
+
+
id
+
string
+
+

add_message

+

+

+void add_message (
+    const char *id,
+    const char *string
+);

+
Parameters
+
+
id
+
Message ID to add
+
string
+
Translation string
+
+

find_message

+

+

+const char *find_message (
+    const char *id
+);

+
Parameters
+
+
id
+
Message ID
+
+
Return Value
+

Message text

+

load_messages

+

+

+int load_messages (
+    const char *f
+);

+
Parameters
+
+
f
+
Message catalog file
+
+
Return Value
+

0 on success, -1 on failure

+

ppdcCatalog

+

+

+void ppdcCatalog (
+    const char *l,
+    const char *f
+);

+
Parameters
+
+
l
+
Locale
+
f
+
Message catalog file
+
+

save_messages

+

+

+int save_messages (
+    const char *f
+);

+
Parameters
+
+
f
+
File to save to
+
+
Return Value
+

0 on success, -1 on error

+

~ppdcCatalog

+

+

+void ~ppdcCatalog (void);

+

ppdcChoice

+

// Option Choice

+

class ppdcChoice : public ppdcShared {
+  public:
+    ppdcString *name, *text, *code;
+};

+

Members

+
+
code
+
PS code of choice
+
+

ppdcChoice

+

+

+void ppdcChoice (
+    const char *n,
+    const char *t,
+    const char *c
+);

+
Parameters
+
+
n
+
Name of choice
+
t
+
Text of choice
+
c
+
Code of choice
+
+

~ppdcChoice

+

+

+void ~ppdcChoice (void);

+

ppdcConstraint

+

// Constraint

+

class ppdcConstraint : public ppdcShared {
+  public:
+    ppdcString *option1, *choice1, *option2, *choice2;
+};

+

Members

+
+
choice2
+
Second choice
+
+

ppdcConstraint

+

+

+void ppdcConstraint (
+    const char *o1,
+    const char *c1,
+    const char *o2,
+    const char *c2
+);

+
Parameters
+
+
o1
+
First option
+
c1
+
First choice
+
o2
+
Second option
+
c2
+
Second choice
+
+

~ppdcConstraint

+

+

+void ~ppdcConstraint (void);

+

ppdcDriver

+

// Printer Driver Data

+

class ppdcDriver : public ppdcShared {
+  public:
+    ppdcArray *copyright;
+    ppdcString *custom_size_code;
+    ppdcString *default_font, *default_size;
+    float left_margin, bottom_margin, right_margin, top_margin, max_width, max_length, min_width, min_length;
+    ppdcArray *attrs, *constraints, *filters, *fonts, *groups, *profiles, *sizes;
+    int model_number, manual_copies, color_device, throughput;
+    ppdcDrvType type;
+    int variable_paper_size;
+    ppdcString *manufacturer, *model_name, *file_name, *pc_file_name, *version;
+};

+

Members

+
+
copyright
+
Copyright strings
+
custom_size_code
+
Custom page size code, if any
+
default_size
+
Default size option
+
min_length
+
Minimum length (points)
+
sizes
+
Fixed sizes
+
throughput
+
Throughput in pages per minute
+
type
+
Driver type
+
variable_paper_size
+
Support variable sizes?
+
version
+
Version number
+
+

PPDC_NAME

+

+

+void PPDC_NAME (
+    void add_attr(ppdcAttr *a
+);

+
Parameters
+
+
a
+
+

add_constraint

+

+

+void add_constraint (
+    ppdcConstraint *c
+);

+
Parameters
+
+
c
+
+

add_copyright

+

+

+void add_copyright (
+    const char *c
+);

+
Parameters
+
+
c
+
+

add_filter

+

+

+void add_filter (
+    ppdcFilter *f
+);

+
Parameters
+
+
f
+
+

add_font

+

+

+void add_font (
+    ppdcFont *f
+);

+
Parameters
+
+
f
+
+

add_group

+

+

+void add_group (
+    ppdcGroup *g
+);

+
Parameters
+
+
g
+
+

add_profile

+

+

+void add_profile (
+    ppdcProfile *p
+);

+
Parameters
+
+
p
+
+

add_size

+

+

+void add_size (
+    ppdcMediaSize *m
+);

+
Parameters
+
+
m
+
+

find_attr

+

+

+ppdcAttr *find_attr (
+    const char *k,
+    const char *s
+);

+
Parameters
+
+
k
+
Keyword string
+
s
+
Spec string
+
+
Return Value
+

Attribute or NULL

+

find_group

+

+

+ppdcGroup *find_group (
+    const char *n
+);

+
Parameters
+
+
n
+
Group name
+
+
Return Value
+

Matching group or NULL

+

find_option

+

+

+ppdcOption *find_option (
+    const char *n
+);

+
Parameters
+
+
n
+
Option name
+
+
Return Value
+

Matching option or NULL

+

find_option_group

+

+

+ppdcOption *find_option_group (
+    const char *n,
+    ppdcGroup **mg
+);

+
Parameters
+
+
n
+
Option name
+
mg
+
Matching group or NULL
+
+
Return Value
+

Matching option or NULL

+

ppdcDriver

+

+

+void ppdcDriver (
+    ppdcDriver *d
+);

+
Parameters
+
+
d
+
Printer driver template
+
+

set_custom_size_code

+

+

+void set_custom_size_code (
+    const char *c
+);

+
Parameters
+
+
c
+
CustomPageSize code
+
+

set_default_font

+

+

+void set_default_font (
+    ppdcFont *f
+);

+
Parameters
+
+
f
+
Font
+
+

set_default_size

+

+

+void set_default_size (
+    ppdcMediaSize *m
+);

+
Parameters
+
+
m
+
Media size
+
+

set_file_name

+

+

+void set_file_name (
+    const char *f
+);

+
Parameters
+
+
f
+
Filename
+
+

set_manufacturer

+

+

+void set_manufacturer (
+    const char *m
+);

+
Parameters
+
+
m
+
Model name
+
+

set_model_name

+

+

+void set_model_name (
+    const char *m
+);

+
Parameters
+
+
m
+
Model name
+
+

set_pc_file_name

+

+

+void set_pc_file_name (
+    const char *f
+);

+
Parameters
+
+
f
+
Filename
+
+

set_version

+

+

+void set_version (
+    const char *v
+);

+
Parameters
+
+
v
+
Version
+
+

write_ppd_file

+

+

+int write_ppd_file (
+    cups_file_t *fp,
+    ppdcCatalog *catalog,
+    ppdcArray *locales,
+    ppdcSource *src,
+    ppdcLineEnding le
+);

+
Parameters
+
+
fp
+
PPD file
+
catalog
+
Message catalog
+
locales
+
Additional languages to add
+
src
+
Driver source
+
le
+
Line endings to use
+
+
Return Value
+

0 on success, -1 on failure

+

~ppdcDriver

+

+

+void ~ppdcDriver (void);

+

ppdcFile

+

// File

+

class ppdcFile {
+  public:
+    const char *filename;
+    cups_file_t *fp;
+    int line;
+};

+

Members

+
+
filename
+
Filename
+
fp
+
File pointer
+
line
+
Line in file
+
+

get

+

Get a character from a file.

+

+int get (void);

+
Return Value
+

+

peek

+

+

+int peek (void);

+
Return Value
+

Next character in file

+

ppdcFile

+

+

+void ppdcFile (
+    const char *f,
+    cups_file_t *ffp
+);

+
Parameters
+
+
f
+
File to open
+
ffp
+
File pointer to use
+
+

~ppdcFile

+

+

+void ~ppdcFile (void);

+

ppdcFilter

+

// Filter Program

+

class ppdcFilter : public ppdcShared {
+  public:
+    int cost;
+    ppdcString *mime_type, *program;
+};

+

Members

+
+
cost
+
Relative cost of filter
+
program
+
Filter program
+
+

ppdcFilter

+

+

+void ppdcFilter (
+    const char *t,
+    const char *p,
+    int c
+);

+
Parameters
+
+
t
+
MIME type
+
p
+
Filter program
+
c
+
Relative cost
+
+

~ppdcFilter

+

+

+void ~ppdcFilter (void);

+

ppdcFont

+

// Shared Font

+

class ppdcFont : public ppdcShared {
+  public:
+    ppdcString *name, *encoding, *version, *charset;
+    ppdcFontStatus status;
+};

+

Members

+
+
charset
+
Font charset
+
status
+
Font status (ROM or Disk)
+
+

ppdcFont

+

+

+void ppdcFont (
+    const char *n,
+    const char *e,
+    const char *v,
+    const char *c,
+    ppdcFontStatus s
+);

+
Parameters
+
+
n
+
Name of font
+
e
+
Font encoding
+
v
+
Font version
+
c
+
Font charset
+
s
+
Font status
+
+

~ppdcFont

+

+

+void ~ppdcFont (void);

+

ppdcGroup

+

// Group of Options

+

class ppdcGroup : public ppdcShared {
+  public:
+    ppdcArray *options;
+    ppdcString *name, *text;
+};

+

Members

+
+
options
+
Options
+
text
+
Human-readable text of option
+
+

PPDC_NAME

+

+

+void PPDC_NAME (
+    void add_option(ppdcOption *o
+);

+
Parameters
+
+
o
+
+

find_option

+

Find an option in a group.

+

+ppdcOption *find_option (
+    const char *n
+);

+
Parameters
+
+
n
+
Name of option
+
+
Return Value
+

+

ppdcGroup

+

+

+void ppdcGroup (
+    ppdcGroup *g
+);

+
Parameters
+
+
g
+
Group template
+
+

~ppdcGroup

+

+

+void ~ppdcGroup (void);

+

ppdcInteger

+

// Shared integer

+

class ppdcInteger : public ppdcShared {
+  public:
+    int *value;
+};

+

Members

+
+
value
+
Integer value
+
+

ppdcInteger

+

Integer value

+

+void ppdcInteger (
+    int *v
+);

+
Parameters
+
+
v
+
+

ppdcMediaSize

+

// Media Size

+

class ppdcMediaSize : public ppdcShared {
+  public:
+    ppdcString *size_code, *region_code;
+    ppdcString *name, *text;
+    float width, length, left, bottom, right, top;
+};

+

Members

+
+
region_code
+
PageRegion code, if any
+
text
+
Human-readable text
+
top
+
Top limit in points
+
+

ppdcMediaSize

+

+

+void ppdcMediaSize (
+    const char *n,
+    const char *t,
+    float w,
+    float l,
+    float lm,
+    float bm,
+    float rm,
+    float tm,
+    const char *sc,
+    const char *rc
+);

+
Parameters
+
+
n
+
Name of media size
+
t
+
Text of media size
+
w
+
Width in points
+
l
+
Length in points
+
lm
+
Left margin in points
+
bm
+
Bottom margin in points
+
rm
+
Right margin in points
+
tm
+
Top margin in points
+
sc
+
PageSize code, if any
+
rc
+
PageRegion code, if any
+
+

~ppdcMediaSize

+

+

+void ~ppdcMediaSize (void);

+

ppdcMessage

+

// Translation message

+

class ppdcMessage : public ppdcShared {
+  public:
+    ppdcString *id, *string;
+};

+

Members

+
+
string
+
Translation string
+
+

ppdcMessage

+

+

+void ppdcMessage (
+    const char *i,
+    const char *s
+);

+
Parameters
+
+
i
+
ID
+
s
+
Text
+
+

~ppdcMessage

+

+

+void ~ppdcMessage (void);

+

ppdcOption

+

// Option

+

class ppdcOption : public ppdcShared {
+  public:
+    ppdcArray *choices;
+    ppdcString *defchoice;
+    float order;
+    ppdcOptSection section;
+    ppdcString *name, *text;
+    ppdcOptType type;
+};

+

Members

+
+
choices
+
Choices
+
defchoice
+
Default choice
+
order
+
Order number
+
section
+
Section for option code
+
text
+
Human-readable text of option
+
type
+
Type of option
+
+

PPDC_NAME

+

+

+void PPDC_NAME (
+    void add_choice(ppdcChoice *c
+);

+
Parameters
+
+
c
+
+

find_choice

+

+

+ppdcChoice *find_choice (
+    const char *n
+);

+
Parameters
+
+
n
+
Name of choice
+
+
Return Value
+

Choice or NULL

+

ppdcOption

+

+

+void ppdcOption (
+    ppdcOption *o
+);

+
Parameters
+
+
o
+
Template option
+
+

set_defchoice

+

+

+void set_defchoice (
+    ppdcChoice *c
+);

+
Parameters
+
+
c
+
Choice
+
+

~ppdcOption

+

+

+void ~ppdcOption (void);

+

ppdcProfile

+

// Color Profile

+

class ppdcProfile : public ppdcShared {
+  public:
+    ppdcString *resolution, *media_type;
+    float density, gamma, profile[9];
+};

+

Members

+
+
media_type
+
Media type name
+
profile[9]
+
Color profile matrix
+
+

ppdcProfile

+

+

+void ppdcProfile (
+    const char *r,
+    const char *m,
+    float d,
+    float g,
+    const float *p
+);

+
Parameters
+
+
r
+
Resolution name
+
m
+
Media type name
+
d
+
Density
+
g
+
Gamma
+
p
+
3x3 transform matrix
+
+

~ppdcProfile

+

+

+void ~ppdcProfile (void);

+

ppdcShared

+

// Shared Data Value

+

class ppdcShared {
+  private:
+    int use;
+};

+

Members

+
+
use
+
Use count (delete when 0)
+
+

class_name

+

+

+virtual const char *class_name (void);

+
Return Value
+

+

ppdcShared

+

+

+void ppdcShared (void);

+

release

+

+

+void release (void);

+

retain

+

+

+void retain (void);

+

~ppdcShared

+

+

+void ~ppdcShared (void);

+

ppdcSource

+

// Source File

+

class ppdcSource : public ppdcShared {
+  public:
+    int cond_state, *cond_current, cond_stack[101];
+    static const char *driver_types[];
+    ppdcString *filename;
+    static ppdcArray *includes;
+    ppdcArray *base_fonts, *drivers, *po_files, *sizes, *vars;
+};

+

Members

+
+
cond_stack[101]
+
#if state stack
+
driver_types[]
+
Driver types
+
filename
+
Filename
+
includes
+
Include directories
+
vars
+
Defined variables
+
+

PPDC_NAME

+

+

+void PPDC_NAME (
+    static void add_include(const char *d
+);

+
Parameters
+
+
d
+
+

add_include

+

+

+void add_include (
+    const char *d
+);

+
Parameters
+
+
d
+
Include directory
+
+

find_driver

+

+

+ppdcDriver *find_driver (
+    const char *f
+);

+
Parameters
+
+
f
+
Driver file name
+
+
Return Value
+

Driver

+

find_include

+

+

+char *find_include (
+    const char *f,
+    const char *base,
+    char *n,
+    int nlen
+);

+
Parameters
+
+
f
+
Include filename
+
base
+
Current directory
+
n
+
Path buffer
+
nlen
+
Path buffer length
+
+
Return Value
+

Found path or NULL

+

find_po

+

+

+ppdcCatalog *find_po (
+    const char *l
+);

+
Parameters
+
+
l
+
Locale name
+
+
Return Value
+

Message catalog or NULL

+

find_size

+

+

+ppdcMediaSize *find_size (
+    const char *s
+);

+
Parameters
+
+
s
+
Size name
+
+
Return Value
+

Size

+

find_variable

+

+

+ppdcVariable *find_variable (
+    const char *n
+);

+
Parameters
+
+
n
+
Variable name
+
+
Return Value
+

Variable

+

get_attr

+

+

+ppdcAttr *get_attr (
+    ppdcFile *fp,
+    bool loc
+);

+
Parameters
+
+
fp
+
File to read
+
loc
+
Localize this attribute?
+
+
Return Value
+

Attribute

+

get_boolean

+

+

+int get_boolean (
+    ppdcFile *fp
+);

+
Parameters
+
+
fp
+
File to read
+
+
Return Value
+

Boolean value

+

get_choice

+

+

+ppdcChoice *get_choice (
+    ppdcFile *fp
+);

+
Parameters
+
+
fp
+
File to read
+
+
Return Value
+

Choice data

+

get_color_model

+

+

+ppdcChoice *get_color_model (
+    ppdcFile *fp
+);

+
Parameters
+
+
fp
+
File to read
+
+
Return Value
+

Choice data

+

get_color_order

+

+

+int get_color_order (
+    const char *co
+);

+
Parameters
+
+
co
+
Color order string
+
+
Return Value
+

Color order value

+

get_color_profile

+

+

+ppdcProfile *get_color_profile (
+    ppdcFile *fp
+);

+
Parameters
+
+
fp
+
File to read
+
+
Return Value
+

Color profile

+

get_color_space

+

+

+int get_color_space (
+    const char *cs
+);

+
Parameters
+
+
cs
+
Colorspace string
+
+
Return Value
+

Colorspace value

+

get_constraint

+

+

+ppdcConstraint *get_constraint (
+    ppdcFile *fp
+);

+
Parameters
+
+
fp
+
File to read
+
+
Return Value
+

Constraint

+

get_custom_size

+

+

+ppdcMediaSize *get_custom_size (
+    ppdcFile *fp
+);

+
Parameters
+
+
fp
+
File to read
+
+
Return Value
+

Media size

+

get_duplex

+

+

+void get_duplex (
+    ppdcFile *fp,
+    ppdcDriver *d
+);

+
Parameters
+
+
fp
+
File to read from
+
d
+
Current driver
+
+

get_filter

+

+

+ppdcFilter *get_filter (
+    ppdcFile *fp
+);

+
Parameters
+
+
fp
+
File to read
+
+
Return Value
+

Filter

+

get_float

+

+

+float get_float (
+    ppdcFile *fp
+);

+
Parameters
+
+
fp
+
File to read
+
+
Return Value
+

Number

+

get_font

+

+

+ppdcFont *get_font (
+    ppdcFile *fp
+);

+
Parameters
+
+
fp
+
File to read
+
+
Return Value
+

Font data

+

get_generic

+

+

+ppdcChoice *get_generic (
+    ppdcFile *fp,
+    const char *keyword,
+    const char *tattr,
+    const char *nattr
+);

+
Parameters
+
+
fp
+
File to read
+
keyword
+
Keyword name
+
tattr
+
Text attribute
+
nattr
+
Numeric attribute
+
+
Return Value
+

Choice data

+

get_group

+

+

+ppdcGroup *get_group (
+    ppdcFile *fp,
+    ppdcDriver *d
+);

+
Parameters
+
+
fp
+
File to read
+
d
+
Printer driver
+
+
Return Value
+

Group

+

get_installable

+

+

+ppdcOption *get_installable (
+    ppdcFile *fp
+);

+
Parameters
+
+
fp
+
File to read
+
+
Return Value
+

Option

+

get_integer

+

+

+int get_integer (
+    ppdcFile *fp
+);

+
Parameters
+
+
fp
+
File to read
+
+
Return Value
+

Integer value

+

get_measurement

+

+

+float get_measurement (
+    ppdcFile *fp
+);

+
Parameters
+
+
fp
+
File to read
+
+
Return Value
+

Measurement value in points

+

get_option

+

+

+ppdcOption *get_option (
+    ppdcFile *fp,
+    ppdcDriver *d,
+    ppdcGroup *g
+);

+
Parameters
+
+
fp
+
File to read
+
d
+
Printer driver
+
g
+
Current group
+
+
Return Value
+

Option

+

get_po

+

+

+ppdcCatalog *get_po (
+    ppdcFile *fp
+);

+
Parameters
+
+
fp
+
File to read
+
+
Return Value
+

Message catalog

+

get_resolution

+

+

+ppdcChoice *get_resolution (
+    ppdcFile *fp
+);

+
Parameters
+
+
fp
+
File to read
+
+
Return Value
+

Choice data

+

get_simple_profile

+

+

+ppdcProfile *get_simple_profile (
+    ppdcFile *fp
+);

+
Parameters
+
+
fp
+
File to read
+
+
Return Value
+

Color profile

+

get_size

+

+

+ppdcMediaSize *get_size (
+    ppdcFile *fp
+);

+
Parameters
+
+
fp
+
File to read
+
+
Return Value
+

Media size

+

get_token

+

+

+char *get_token (
+    ppdcFile *fp,
+    char *buffer,
+    int buflen
+);

+
Parameters
+
+
fp
+
File to read
+
buffer
+
Buffer
+
buflen
+
Length of buffer
+
+
Return Value
+

Token string or NULL

+

get_variable

+

+

+ppdcVariable *get_variable (
+    ppdcFile *fp
+);

+
Parameters
+
+
fp
+
File to read
+
+
Return Value
+

Variable

+

import_ppd

+

+

+int import_ppd (
+    const char *f
+);

+
Parameters
+
+
f
+
Filename
+
+
Return Value
+

1 on success, 0 on failure

+

ppdcSource

+

+

+void ppdcSource (
+    const char *f,
+    cups_file_t *ffp
+);

+
Parameters
+
+
f
+
File to read
+
ffp
+
File pointer to use
+
+

quotef

+

+

+int quotef (
+    cups_file_t *fp,
+    const char *format,
+    ...
+);

+
Parameters
+
+
fp
+
File to write to
+
format
+
Printf-style format string
+
...
+
Additional args as needed
+
+
Return Value
+

Number bytes on success, -1 on failure

+

read_file

+

+

+void read_file (
+    const char *f,
+    cups_file_t *ffp
+);

+
Parameters
+
+
f
+
File to read
+
ffp
+
File pointer to use
+
+

scan_file

+

+

+void scan_file (
+    ppdcFile *fp,
+    ppdcDriver *td,
+    bool inc
+);

+
Parameters
+
+
fp
+
File to read
+
td
+
Driver template
+
inc
+
Including?
+
+

set_variable

+

+

+ppdcVariable *set_variable (
+    const char *name,
+    const char *value
+);

+
Parameters
+
+
name
+
Name
+
value
+
Value
+
+
Return Value
+

Variable

+

write_file

+

+

+int write_file (
+    const char *f
+);

+
Parameters
+
+
f
+
File to write
+
+
Return Value
+

0 on success, -1 on error

+

~ppdcSource

+

+

+void ~ppdcSource (void);

+

ppdcString

+

// Shared String

+

class ppdcString : public ppdcShared {
+  public:
+    char *value;
+};

+

Members

+
+
value
+
String value
+
+

ppdcString

+

+

+void ppdcString (
+    const char *v
+);

+
Parameters
+
+
v
+
String
+
+

~ppdcString

+

+

+void ~ppdcString (void);

+

ppdcVariable

+

// Variable Definition

+

class ppdcVariable : public ppdcShared {
+  public:
+    ppdcString *name, *value;
+};

+

Members

+
+
value
+
Value of variable
+
+

PPDC_NAME

+

+

+void PPDC_NAME (
+    void set_value(const char *v
+);

+
Parameters
+
+
v
+
+

ppdcVariable

+

+

+void ppdcVariable (
+    const char *n,
+    const char *v
+);

+
Parameters
+
+
n
+
Name of variable
+
v
+
Value of variable
+
+

set_value

+

+

+void set_value (
+    const char *v
+);

+
Parameters
+
+
v
+
+

~ppdcVariable

+

+

+void ~ppdcVariable (void);

+

Data Types

+

ppdc_cs_t

+

+

+typedef enum ppdc_cs_t; +

+

Variables

+

ppdcSource

+

// Printer Driver Data

+

class ppdcSource;

+

Constants

+

ppdcCondFlags

+

// Condition flags

+

Constants

+
+
PPDC_COND_NORMAL
+
Normal state
+
PPDC_COND_SATISFIED
+
At least one condition satisfied
+
PPDC_COND_SKIP
+
Skip state
+
+

ppdcDrvType

+

// Driver type

+

Constants

+
+
PPDC_DRIVER_CUSTOM
+
Custom driver
+
PPDC_DRIVER_EPSON
+
rastertoepson driver
+
PPDC_DRIVER_ESCP
+
rastertoescpx driver
+
PPDC_DRIVER_HP
+
rastertohp driver
+
PPDC_DRIVER_LABEL
+
rastertolabel/rastertodymo driver
+
PPDC_DRIVER_MAX
+
Number of driver types defined
+
PPDC_DRIVER_PCL
+
rastertopclx driver
+
PPDC_DRIVER_PS
+
PostScript driver
+
+

ppdcFontStatus

+

// Load status of font

+

Constants

+
+
PPDC_FONT_DISK
+
Font is on disk
+
PPDC_FONT_ROM
+
Font is in ROM
+
+

ppdcLineEnding

+

// Line endings

+

Constants

+
+
PPDC_CRLF
+
CR + LF
+
PPDC_CRONLY
+
CR only
+
PPDC_LFONLY
+
LF only
+
+

ppdcOptSection

+

// Option section

+

Constants

+
+
PPDC_SECTION_ANY
+
AnySetup
+
PPDC_SECTION_DOCUMENT
+
DocumentSetup
+
PPDC_SECTION_EXIT
+
ExitServer
+
PPDC_SECTION_JCL
+
JCLSetup
+
PPDC_SECTION_PAGE
+
PageSetup
+
PPDC_SECTION_PROLOG
+
Prolog
+
+

ppdcOptType

+

// Option type

+

Constants

+
+
PPDC_BOOLEAN
+
True/false option
+
PPDC_PICKMANY
+
Multiple choices from list
+
PPDC_PICKONE
+
Single choice from list
+
+
+ + diff --git a/doc/help/api-raster.html b/doc/help/api-raster.html new file mode 100644 index 0000000000..f193d07e21 --- /dev/null +++ b/doc/help/api-raster.html @@ -0,0 +1,1416 @@ + + + + + Raster API + + + + + + +
+ + +

Raster API

+ +
+ + + + + + + + + + + + + + + + +
Headercups/raster.h
Library-lcupsimage
See AlsoProgramming: Introduction to CUPS Programming
+ Programming: CUPS API
+ Programming: PPD API
+ References: CUPS PPD Specification
+

Contents

+ + + +

Overview

+ +

The CUPS raster API provides a standard interface for reading and writing +CUPS raster streams which are used for printing to raster printers. Because the +raster format is updated from time to time, it is important to use this API to +avoid incompatibilities with newer versions of CUPS.

+ +

Two kinds of CUPS filters use the CUPS raster API - raster image processor +(RIP) filters such as pstoraster and cgpdftoraster +(Mac OS X) that produce CUPS raster files and printer driver filters that +convert CUPS raster files into a format usable by the printer. Printer +driver filters are by far the most common.

+ +

CUPS raster files (application/vnd.cups-raster) consists of +a stream of raster page descriptions produced by one of the RIP filters such as +pstoraster, imagetoraster, or +cgpdftoraster. CUPS raster files are referred to using the +cups_raster_t type and are +opened using the cupsRasterOpen +function. For example, to read raster data from the standard input, open +file descriptor 0:

+ +
+#include <cups/raster.h>>
+
+cups_raster_t *ras = cupsRasterOpen(0, CUPS_RASTER_READ);
+
+ +

Each page of data begins with a page dictionary structure called +cups_page_header2_t. This +structure contains the colorspace, bits per color, media size, media type, +hardware resolution, and so forth used for the page.

+ +
Note: + +

Do not confuse the colorspace in the page header with the PPD + ColorModel keyword. ColorModel refers to the general type of + color used for a device (Gray, RGB, CMYK, DeviceN) and is often used to + select a particular colorspace for the page header along with the associate + color profile. The page header colorspace (cupsColorSpace) describes + both the type and organization of the color data, for example KCMY (black + first) instead of CMYK and RGBA (RGB + alpha) instead of RGB.

+ +
+ +

You read the page header using the +cupsRasterReadHeader2 +function:

+ +
+#include <cups/raster.h>>
+
+cups_raster_t *ras = cupsRasterOpen(0, CUPS_RASTER_READ);
+cups_page_header2_t header;
+
+while (cupsRasterReadHeader2(ras, &header))
+{
+  /* setup this page */
+
+  /* read raster data */
+
+  /* finish this page */
+}
+
+ +

After the page dictionary comes the page data which is a full-resolution, +possibly compressed bitmap representing the page in the printer's output +colorspace. You read uncompressed raster data using the +cupsRasterReadPixels +function. A for loop is normally used to read the page one line +at a time:

+ +
+#include <cups/raster.h>>
+
+cups_raster_t *ras = cupsRasterOpen(0, CUPS_RASTER_READ);
+cups_page_header2_t header;
+int page = 0;
+int y;
+char *buffer;
+
+while (cupsRasterReadHeader2(ras, &header))
+{
+  /* setup this page */
+  page ++;
+  fprintf(stderr, "PAGE: %d %d\n", page, header.NumCopies);
+
+  /* allocate memory for 1 line */
+  buffer = malloc(header.cupsBytesPerLine);
+
+  /* read raster data */
+  for (y = 0; y < header.cupsHeight; y ++)
+  {
+    if (cupsRasterReadPixels(ras, buffer, header.cupsBytesPerLine) == 0)
+      break;
+
+    /* write raster data to printer on stdout */
+  }
+
+  /* finish this page */
+}
+
+ +

When you are done reading the raster data, call the +cupsRasterClose function to free +the memory used to read the raster file:

+ +
+cups_raster_t *ras;
+
+cupsRasterClose(ras);
+
+ + +

Functions by Task

+ +

Opening and Closing Raster Streams

+ + + +

Reading Raster Streams

+ + + +

Writing Raster Streams

+ + +

Functions

+

cupsRasterClose

+

Close a raster stream.

+

+void cupsRasterClose (
+    cups_raster_t *r
+);

+

Parameters

+
+
r
+
Stream to close
+
+

Discussion

+

The file descriptor associated with the raster stream must be closed +separately as needed.

+

 CUPS 1.2/Mac OS X 10.5 cupsRasterInterpretPPD

+

Interpret PPD commands to create a page header.

+

+int cupsRasterInterpretPPD (
+    cups_page_header2_t *h,
+    ppd_file_t *ppd,
+    int num_options,
+    cups_option_t *options,
+    cups_interpret_cb_t func
+);

+

Parameters

+
+
h
+
Page header to create
+
ppd
+
PPD file
+
num_options
+
Number of options
+
options
+
Options
+
func
+
Optional page header callback (NULL for none)
+
+

Return Value

+

0 on success, -1 on failure

+

Discussion

+

This function is used by raster image processing (RIP) filters like +cgpdftoraster and imagetoraster when writing CUPS raster data for a page. +It is not used by raster printer driver filters which only read CUPS +raster data.
+
+ +cupsRasterInterpretPPD does not mark the options in the PPD using +the "num_options" and "options" arguments. Instead, mark the options with +cupsMarkOptions and ppdMarkOption prior to calling it - +this allows for per-page options without manipulating the options array.
+
+The "func" argument specifies an optional callback function that is +called prior to the computation of the final raster data. The function +can make changes to the cups_page_header2_t data as needed to use a +supported raster format and then returns 0 on success and -1 if the +requested attributes cannot be supported.
+
+ +cupsRasterInterpretPPD supports a subset of the PostScript language. +Currently only the [, ], <<, >>, {, +}, cleartomark, copy, dup, index, +pop, roll, setpagedevice, and stopped operators +are supported. + +

+

cupsRasterOpen

+

Open a raster stream using a file descriptor.

+

+cups_raster_t *cupsRasterOpen (
+    int fd,
+    cups_mode_t mode
+);

+

Parameters

+
+
fd
+
File descriptor
+
mode
+
Mode - CUPS_RASTER_READ, +CUPS_RASTER_WRITE, +CUPS_RASTER_WRITE_COMPRESSED, +or CUPS_RASTER_WRITE_PWG
+
+

Return Value

+

New stream

+

Discussion

+

This function associates a raster stream with the given file descriptor. +For most printer driver filters, "fd" will be 0 (stdin). For most raster +image processor (RIP) filters that generate raster data, "fd" will be 1 +(stdout).
+
+When writing raster data, the CUPS_RASTER_WRITE, +CUPS_RASTER_WRITE_COMPRESS, or CUPS_RASTER_WRITE_PWG mode can +be used - compressed and PWG output is generally 25-50% smaller but adds a +100-300% execution time overhead.

+

cupsRasterOpenIO

+

Open a raster stream using a callback function.

+

+cups_raster_t *cupsRasterOpenIO (
+    cups_raster_iocb_t iocb,
+    void *ctx,
+    cups_mode_t mode
+);

+

Parameters

+
+
iocb
+
Read/write callback
+
ctx
+
Context pointer for callback
+
mode
+
Mode - CUPS_RASTER_READ, +CUPS_RASTER_WRITE, +CUPS_RASTER_WRITE_COMPRESSED, +or CUPS_RASTER_WRITE_PWG
+
+

Return Value

+

New stream

+

Discussion

+

This function associates a raster stream with the given callback function and +context pointer.
+
+When writing raster data, the CUPS_RASTER_WRITE, +CUPS_RASTER_WRITE_COMPRESS, or CUPS_RASTER_WRITE_PWG mode can +be used - compressed and PWG output is generally 25-50% smaller but adds a +100-300% execution time overhead.

+

 DEPRECATED cupsRasterReadHeader

+

Read a raster page header and store it in a +version 1 page header structure.

+

+unsigned cupsRasterReadHeader (
+    cups_raster_t *r,
+    cups_page_header_t *h
+);

+

Parameters

+
+
r
+
Raster stream
+
h
+
Pointer to header data
+
+

Return Value

+

1 on success, 0 on failure/end-of-file

+

Discussion

+

This function is deprecated. Use cupsRasterReadHeader2 instead.
+
+Version 1 page headers were used in CUPS 1.0 and 1.1 and contain a subset +of the version 2 page header data. This function handles reading version 2 +page headers and copying only the version 1 data into the provided buffer. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsRasterReadHeader2

+

Read a raster page header and store it in a +version 2 page header structure.

+

+unsigned cupsRasterReadHeader2 (
+    cups_raster_t *r,
+    cups_page_header2_t *h
+);

+

Parameters

+
+
r
+
Raster stream
+
h
+
Pointer to header data
+
+

Return Value

+

1 on success, 0 on failure/end-of-file

+

cupsRasterReadPixels

+

Read raster pixels.

+

+unsigned cupsRasterReadPixels (
+    cups_raster_t *r,
+    unsigned char *p,
+    unsigned len
+);

+

Parameters

+
+
r
+
Raster stream
+
p
+
Pointer to pixel buffer
+
len
+
Number of bytes to read
+
+

Return Value

+

Number of bytes read

+

Discussion

+

For best performance, filters should read one or more whole lines. +The "cupsBytesPerLine" value from the page header can be used to allocate +the line buffer and as the number of bytes to read.

+

 DEPRECATED cupsRasterWriteHeader

+

Write a raster page header from a version 1 page +header structure.

+

+unsigned cupsRasterWriteHeader (
+    cups_raster_t *r,
+    cups_page_header_t *h
+);

+

Parameters

+
+
r
+
Raster stream
+
h
+
Raster page header
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

This function is deprecated. Use cupsRasterWriteHeader2 instead. + +

+

 CUPS 1.2/Mac OS X 10.5 cupsRasterWriteHeader2

+

Write a raster page header from a version 2 +page header structure.

+

+unsigned cupsRasterWriteHeader2 (
+    cups_raster_t *r,
+    cups_page_header2_t *h
+);

+

Parameters

+
+
r
+
Raster stream
+
h
+
Raster page header
+
+

Return Value

+

1 on success, 0 on failure

+

Discussion

+

The page header can be initialized using cupsRasterInterpretPPD. + +

+

cupsRasterWritePixels

+

Write raster pixels.

+

+unsigned cupsRasterWritePixels (
+    cups_raster_t *r,
+    unsigned char *p,
+    unsigned len
+);

+

Parameters

+
+
r
+
Raster stream
+
p
+
Bytes to write
+
len
+
Number of bytes to write
+
+

Return Value

+

Number of bytes written

+

Discussion

+

For best performance, filters should write one or more whole lines. +The "cupsBytesPerLine" value from the page header can be used to allocate +the line buffer and as the number of bytes to write.

+

Data Types

+

cups_adv_t

+

AdvanceMedia attribute values

+

+typedef enum cups_adv_e cups_adv_t; +

+

cups_bool_t

+

Boolean type

+

+typedef enum cups_bool_e cups_bool_t; +

+

cups_cspace_t

+

cupsColorSpace attribute values

+

+typedef enum cups_cspace_e cups_cspace_t; +

+

cups_cut_t

+

CutMedia attribute values

+

+typedef enum cups_cut_e cups_cut_t; +

+

cups_edge_t

+

LeadingEdge attribute values

+

+typedef enum cups_edge_e cups_edge_t; +

+

cups_interpret_cb_t

+

cupsRasterInterpretPPD callback function

+

+typedef int (*cups_interpret_cb_t)(cups_page_header2_t *header, int preferred_bits); +

+

cups_jog_t

+

Jog attribute values

+

+typedef enum cups_jog_e cups_jog_t; +

+

cups_mode_t

+

cupsRasterOpen modes

+

+typedef enum cups_mode_e cups_mode_t; +

+

cups_order_t

+

cupsColorOrder attribute values

+

+typedef enum cups_order_e cups_order_t; +

+

cups_orient_t

+

Orientation attribute values

+

+typedef enum cups_orient_e cups_orient_t; +

+

 CUPS 1.2/Mac OS X 10.5 cups_page_header2_t

+

Version 2 page header

+

+typedef struct cups_page_header2_s cups_page_header2_t; +

+

 DEPRECATED cups_page_header_t

+

Version 1 page header

+

+typedef struct cups_page_header_s cups_page_header_t; +

+

cups_raster_iocb_t

+

cupsRasterOpenIO callback function

+

+typedef ssize_t (*cups_raster_iocb_t)(void *ctx, unsigned char *buffer, size_t length); +

+

cups_raster_t

+

Raster stream data

+

+typedef struct _cups_raster_s cups_raster_t; +

+

Structures

+

 CUPS 1.2/Mac OS X 10.5 cups_page_header2_s

+

Version 2 page header

+

struct cups_page_header2_s {
+    unsigned AdvanceDistance;
+    cups_adv_t AdvanceMedia;
+    cups_bool_t Collate;
+    cups_cut_t CutMedia;
+    cups_bool_t Duplex;
+    unsigned HWResolution[2];
+    unsigned ImagingBoundingBox[4];
+    cups_bool_t InsertSheet;
+    cups_jog_t Jog;
+    cups_edge_t LeadingEdge;
+    cups_bool_t ManualFeed;
+    unsigned Margins[2];
+    char MediaClass[64];
+    char MediaColor[64];
+    unsigned MediaPosition;
+    char MediaType[64];
+    unsigned MediaWeight;
+    cups_bool_t MirrorPrint;
+    cups_bool_t NegativePrint;
+    unsigned NumCopies;
+    cups_orient_t Orientation;
+    cups_bool_t OutputFaceUp;
+    char OutputType[64];
+    unsigned PageSize[2];
+    cups_bool_t Separations;
+    cups_bool_t TraySwitch;
+    cups_bool_t Tumble;
+    unsigned cupsBitsPerColor;
+    unsigned cupsBitsPerPixel;
+    float cupsBorderlessScalingFactor;
+    unsigned cupsBytesPerLine;
+    cups_order_t cupsColorOrder;
+    cups_cspace_t cupsColorSpace;
+    unsigned cupsCompression;
+    unsigned cupsHeight;
+    float cupsImagingBBox[4];
+    unsigned cupsInteger[16];
+    char cupsMarkerType[64];
+    unsigned cupsMediaType;
+    unsigned cupsNumColors;
+    char cupsPageSizeName[64];
+    float cupsPageSize[2];
+    float cupsReal[16];
+    char cupsRenderingIntent[64];
+    unsigned cupsRowCount;
+    unsigned cupsRowFeed;
+    unsigned cupsRowStep;
+    char cupsString[16][64];
+    unsigned cupsWidth;
+};

+

Members

+
+
AdvanceDistance
+
AdvanceDistance value in points
+
AdvanceMedia
+
AdvanceMedia value (cups_adv_t)
+
Collate
+
Collated copies value
+
CutMedia
+
CutMedia value (cups_cut_t)
+
Duplex
+
Duplexed (double-sided) value
+
HWResolution[2]
+
Resolution in dots-per-inch
+
ImagingBoundingBox[4]
+
Pixel region that is painted (points, left, bottom, right, top)
+
InsertSheet
+
InsertSheet value
+
Jog
+
Jog value (cups_jog_t)
+
LeadingEdge
+
LeadingEdge value (cups_edge_t)
+
ManualFeed
+
ManualFeed value
+
Margins[2]
+
Lower-lefthand margins in points
+
MediaClass[64]
+
MediaClass string
+
MediaColor[64]
+
MediaColor string
+
MediaPosition
+
MediaPosition value
+
MediaType[64]
+
MediaType string
+
MediaWeight
+
MediaWeight value in grams/m^2
+
MirrorPrint
+
MirrorPrint value
+
NegativePrint
+
NegativePrint value
+
NumCopies
+
Number of copies to produce
+
Orientation
+
Orientation value (cups_orient_t)
+
OutputFaceUp
+
OutputFaceUp value
+
OutputType[64]
+
OutputType string
+
PageSize[2]
+
Width and length of page in points
+
Separations
+
Separations value
+
TraySwitch
+
TraySwitch value
+
Tumble
+
Tumble value
+
cupsBitsPerColor
+
Number of bits for each color
+
cupsBitsPerPixel
+
Number of bits for each pixel
+
cupsBorderlessScalingFactor  CUPS 1.2/Mac OS X 10.5 
+
Scaling that was applied to page data
+
cupsBytesPerLine
+
Number of bytes per line
+
cupsColorOrder
+
Order of colors
+
cupsColorSpace
+
True colorspace
+
cupsCompression
+
Device compression to use
+
cupsHeight
+
Height of page image in pixels
+
cupsImagingBBox[4]  CUPS 1.2/Mac OS X 10.5 
+
Floating point ImagingBoundingBox +(scaling factor not applied, left, +bottom, right, top)
+
cupsInteger[16]  CUPS 1.2/Mac OS X 10.5 
+
User-defined integer values
+
cupsMarkerType[64]  CUPS 1.2/Mac OS X 10.5 
+
Ink/toner type
+
cupsMediaType
+
Media type code
+
cupsNumColors  CUPS 1.2/Mac OS X 10.5 
+
Number of color compoents
+
cupsPageSizeName[64]  CUPS 1.2/Mac OS X 10.5 
+
PageSize name
+
cupsPageSize[2]  CUPS 1.2/Mac OS X 10.5 
+
Floating point PageSize (scaling * +factor not applied)
+
cupsReal[16]  CUPS 1.2/Mac OS X 10.5 
+
User-defined floating-point values
+
cupsRenderingIntent[64]  CUPS 1.2/Mac OS X 10.5 
+
Color rendering intent
+
cupsRowCount
+
Rows per band
+
cupsRowFeed
+
Feed between bands
+
cupsRowStep
+
Spacing between lines
+
cupsString[16][64]  CUPS 1.2/Mac OS X 10.5 
+
User-defined string values
+
cupsWidth
+
Width of page image in pixels
+
+

 DEPRECATED cups_page_header_s

+

Version 1 page header

+

struct cups_page_header_s {
+    unsigned AdvanceDistance;
+    cups_adv_t AdvanceMedia;
+    cups_bool_t Collate;
+    cups_cut_t CutMedia;
+    cups_bool_t Duplex;
+    unsigned HWResolution[2];
+    unsigned ImagingBoundingBox[4];
+    cups_bool_t InsertSheet;
+    cups_jog_t Jog;
+    cups_edge_t LeadingEdge;
+    cups_bool_t ManualFeed;
+    unsigned Margins[2];
+    char MediaClass[64];
+    char MediaColor[64];
+    unsigned MediaPosition;
+    char MediaType[64];
+    unsigned MediaWeight;
+    cups_bool_t MirrorPrint;
+    cups_bool_t NegativePrint;
+    unsigned NumCopies;
+    cups_orient_t Orientation;
+    cups_bool_t OutputFaceUp;
+    char OutputType[64];
+    unsigned PageSize[2];
+    cups_bool_t Separations;
+    cups_bool_t TraySwitch;
+    cups_bool_t Tumble;
+    unsigned cupsBitsPerColor;
+    unsigned cupsBitsPerPixel;
+    unsigned cupsBytesPerLine;
+    cups_order_t cupsColorOrder;
+    cups_cspace_t cupsColorSpace;
+    unsigned cupsCompression;
+    unsigned cupsHeight;
+    unsigned cupsMediaType;
+    unsigned cupsRowCount;
+    unsigned cupsRowFeed;
+    unsigned cupsRowStep;
+    unsigned cupsWidth;
+};

+

Members

+
+
AdvanceDistance
+
AdvanceDistance value in points
+
AdvanceMedia
+
AdvanceMedia value (cups_adv_t)
+
Collate
+
Collated copies value
+
CutMedia
+
CutMedia value (cups_cut_t)
+
Duplex
+
Duplexed (double-sided) value
+
HWResolution[2]
+
Resolution in dots-per-inch
+
ImagingBoundingBox[4]
+
Pixel region that is painted (points, left, bottom, right, top)
+
InsertSheet
+
InsertSheet value
+
Jog
+
Jog value (cups_jog_t)
+
LeadingEdge
+
LeadingEdge value (cups_edge_t)
+
ManualFeed
+
ManualFeed value
+
Margins[2]
+
Lower-lefthand margins in points
+
MediaClass[64]
+
MediaClass string
+
MediaColor[64]
+
MediaColor string
+
MediaPosition
+
MediaPosition value
+
MediaType[64]
+
MediaType string
+
MediaWeight
+
MediaWeight value in grams/m^2
+
MirrorPrint
+
MirrorPrint value
+
NegativePrint
+
NegativePrint value
+
NumCopies
+
Number of copies to produce
+
Orientation
+
Orientation value (cups_orient_t)
+
OutputFaceUp
+
OutputFaceUp value
+
OutputType[64]
+
OutputType string
+
PageSize[2]
+
Width and length of page in points
+
Separations
+
Separations value
+
TraySwitch
+
TraySwitch value
+
Tumble
+
Tumble value
+
cupsBitsPerColor
+
Number of bits for each color
+
cupsBitsPerPixel
+
Number of bits for each pixel
+
cupsBytesPerLine
+
Number of bytes per line
+
cupsColorOrder
+
Order of colors
+
cupsColorSpace
+
True colorspace
+
cupsCompression
+
Device compression to use
+
cupsHeight
+
Height of page image in pixels
+
cupsMediaType
+
Media type code
+
cupsRowCount
+
Rows per band
+
cupsRowFeed
+
Feed between bands
+
cupsRowStep
+
Spacing between lines
+
cupsWidth
+
Width of page image in pixels
+
+

Constants

+

cups_adv_e

+

AdvanceMedia attribute values

+

Constants

+
+
CUPS_ADVANCE_FILE
+
Advance the roll after this file
+
CUPS_ADVANCE_JOB
+
Advance the roll after this job
+
CUPS_ADVANCE_NONE
+
Never advance the roll
+
CUPS_ADVANCE_PAGE
+
Advance the roll after this page
+
CUPS_ADVANCE_SET
+
Advance the roll after this set
+
+

cups_bool_e

+

Boolean type

+

Constants

+
+
CUPS_FALSE
+
Logical false
+
CUPS_TRUE
+
Logical true
+
+

cups_cspace_e

+

cupsColorSpace attribute values

+

Constants

+
+
CUPS_CSPACE_ADOBERGB  CUPS 1.4.5 
+
Red, green, blue (Adobe RGB)
+
CUPS_CSPACE_CIELab  CUPS 1.1.19/Mac OS X 10.3 
+
CIE Lab
+
CUPS_CSPACE_CIEXYZ  CUPS 1.1.19/Mac OS X 10.3 
+
CIE XYZ
+
CUPS_CSPACE_CMY
+
Cyan, magenta, yellow (DeviceCMY)
+
CUPS_CSPACE_CMYK
+
Cyan, magenta, yellow, black (DeviceCMYK)
+
CUPS_CSPACE_DEVICE1  CUPS 1.4.5 
+
DeviceN, 1 color
+
CUPS_CSPACE_DEVICE2  CUPS 1.4.5 
+
DeviceN, 2 colors
+
CUPS_CSPACE_DEVICE3  CUPS 1.4.5 
+
DeviceN, 3 colors
+
CUPS_CSPACE_DEVICE4  CUPS 1.4.5 
+
DeviceN, 4 colors
+
CUPS_CSPACE_DEVICE5  CUPS 1.4.5 
+
DeviceN, 5 colors
+
CUPS_CSPACE_DEVICE6  CUPS 1.4.5 
+
DeviceN, 6 colors
+
CUPS_CSPACE_DEVICE7  CUPS 1.4.5 
+
DeviceN, 7 colors
+
CUPS_CSPACE_DEVICE8  CUPS 1.4.5 
+
DeviceN, 8 colors
+
CUPS_CSPACE_DEVICE9  CUPS 1.4.5 
+
DeviceN, 9 colors
+
CUPS_CSPACE_DEVICEA  CUPS 1.4.5 
+
DeviceN, 10 colors
+
CUPS_CSPACE_DEVICEB  CUPS 1.4.5 
+
DeviceN, 11 colors
+
CUPS_CSPACE_DEVICEC  CUPS 1.4.5 
+
DeviceN, 12 colors
+
CUPS_CSPACE_DEVICED  CUPS 1.4.5 
+
DeviceN, 13 colors
+
CUPS_CSPACE_DEVICEE  CUPS 1.4.5 
+
DeviceN, 14 colors
+
CUPS_CSPACE_DEVICEF  CUPS 1.4.5 
+
DeviceN, 15 colors
+
CUPS_CSPACE_GMCK  DEPRECATED 
+
Gold, magenta, yellow, black
+
CUPS_CSPACE_GMCS  DEPRECATED 
+
Gold, magenta, yellow, silver
+
CUPS_CSPACE_GOLD  DEPRECATED 
+
Gold foil
+
CUPS_CSPACE_ICC1  CUPS 1.1.19/Mac OS X 10.3 
+
ICC-based, 1 color
+
CUPS_CSPACE_ICC2  CUPS 1.1.19/Mac OS X 10.3 
+
ICC-based, 2 colors
+
CUPS_CSPACE_ICC3  CUPS 1.1.19/Mac OS X 10.3 
+
ICC-based, 3 colors
+
CUPS_CSPACE_ICC4  CUPS 1.1.19/Mac OS X 10.3 
+
ICC-based, 4 colors
+
CUPS_CSPACE_ICC5  CUPS 1.1.19/Mac OS X 10.3 
+
ICC-based, 5 colors
+
CUPS_CSPACE_ICC6  CUPS 1.1.19/Mac OS X 10.3 
+
ICC-based, 6 colors
+
CUPS_CSPACE_ICC7  CUPS 1.1.19/Mac OS X 10.3 
+
ICC-based, 7 colors
+
CUPS_CSPACE_ICC8  CUPS 1.1.19/Mac OS X 10.3 
+
ICC-based, 8 colors
+
CUPS_CSPACE_ICC9  CUPS 1.1.19/Mac OS X 10.3 
+
ICC-based, 9 colors
+
CUPS_CSPACE_ICCA  CUPS 1.1.19/Mac OS X 10.3 
+
ICC-based, 10 colors
+
CUPS_CSPACE_ICCB  CUPS 1.1.19/Mac OS X 10.3 
+
ICC-based, 11 colors
+
CUPS_CSPACE_ICCC  CUPS 1.1.19/Mac OS X 10.3 
+
ICC-based, 12 colors
+
CUPS_CSPACE_ICCD  CUPS 1.1.19/Mac OS X 10.3 
+
ICC-based, 13 colors
+
CUPS_CSPACE_ICCE  CUPS 1.1.19/Mac OS X 10.3 
+
ICC-based, 14 colors
+
CUPS_CSPACE_ICCF  CUPS 1.1.19/Mac OS X 10.3 
+
ICC-based, 15 colors
+
CUPS_CSPACE_K
+
Black (DeviceK)
+
CUPS_CSPACE_KCMY  DEPRECATED 
+
Black, cyan, magenta, yellow
+
CUPS_CSPACE_KCMYcm  DEPRECATED 
+
Black, cyan, magenta, yellow, light-cyan, light-magenta
+
CUPS_CSPACE_RGB
+
Red, green, blue (DeviceRGB, sRGB by default)
+
CUPS_CSPACE_RGBA
+
Red, green, blue, alpha (DeviceRGB, sRGB by default)
+
CUPS_CSPACE_RGBW  CUPS 1.2/Mac OS X 10.5 
+
Red, green, blue, white (DeviceRGB, sRGB by default)
+
CUPS_CSPACE_SILVER  DEPRECATED 
+
Silver foil
+
CUPS_CSPACE_SRGB  CUPS 1.4.5 
+
Red, green, blue (sRGB)
+
CUPS_CSPACE_SW  CUPS 1.4.5 
+
Luminance (gamma 2.2)
+
CUPS_CSPACE_W
+
Luminance (DeviceGray, gamma 2.2 by default)
+
CUPS_CSPACE_WHITE  DEPRECATED 
+
White ink (as black)
+
CUPS_CSPACE_YMC  DEPRECATED 
+
Yellow, magenta, cyan
+
CUPS_CSPACE_YMCK  DEPRECATED 
+
Yellow, magenta, cyan, black
+
+

cups_cut_e

+

CutMedia attribute values

+

Constants

+
+
CUPS_CUT_FILE
+
Cut the roll after this file
+
CUPS_CUT_JOB
+
Cut the roll after this job
+
CUPS_CUT_NONE
+
Never cut the roll
+
CUPS_CUT_PAGE
+
Cut the roll after this page
+
CUPS_CUT_SET
+
Cut the roll after this set
+
+

cups_edge_e

+

LeadingEdge attribute values

+

Constants

+
+
CUPS_EDGE_BOTTOM
+
Leading edge is the bottom of the page
+
CUPS_EDGE_LEFT
+
Leading edge is the left of the page
+
CUPS_EDGE_RIGHT
+
Leading edge is the right of the page
+
CUPS_EDGE_TOP
+
Leading edge is the top of the page
+
+

cups_jog_e

+

Jog attribute values

+

Constants

+
+
CUPS_JOG_FILE
+
Move pages after this file
+
CUPS_JOG_JOB
+
Move pages after this job
+
CUPS_JOG_NONE
+
Never move pages
+
CUPS_JOG_SET
+
Move pages after this set
+
+

cups_mode_e

+

cupsRasterOpen modes

+

Constants

+
+
CUPS_RASTER_READ
+
Open stream for reading
+
CUPS_RASTER_WRITE
+
Open stream for writing
+
CUPS_RASTER_WRITE_COMPRESSED  CUPS 1.3/Mac OS X 10.5 
+
Open stream for compressed writing
+
CUPS_RASTER_WRITE_PWG  CUPS 1.5/Mac OS X 10.7 
+
Open stream for compressed writing in PWG mode
+
+

cups_order_e

+

cupsColorOrder attribute values

+

Constants

+
+
CUPS_ORDER_BANDED
+
CCC MMM YYY KKK ...
+
CUPS_ORDER_CHUNKED
+
CMYK CMYK CMYK ...
+
CUPS_ORDER_PLANAR
+
CCC ... MMM ... YYY ... KKK ...
+
+

cups_orient_e

+

Orientation attribute values

+

Constants

+
+
CUPS_ORIENT_0
+
Don't rotate the page
+
CUPS_ORIENT_180
+
Turn the page upside down
+
CUPS_ORIENT_270
+
Rotate the page clockwise
+
CUPS_ORIENT_90
+
Rotate the page counter-clockwise
+
+
+ + diff --git a/doc/help/cgi.html b/doc/help/cgi.html new file mode 100644 index 0000000000..c50a2a5014 --- /dev/null +++ b/doc/help/cgi.html @@ -0,0 +1,86 @@ + + + + Using CGI Programs + + + + +

Using CGI Programs

+ +

CUPS provides a dynamic web interface through dedicated CGI programs that +are executed when users open special directories on the CUPS server. Each CGI +performs administration, class, help, job, and printer functions as directed by +the user, but the actual programs that are run and functions that are available +are limited to those that were originally designed into the scheduler.

+ +

CUPS also supports CGI programs and specific scripting languages (Java, Perl, +PHP, and Python) for pages you want to provide. The interpreters for these +languages are currently configured at compile time and are associated with +MIME media types. Table 1 shows the MIME media types that +are reserved for each type of page and are the same as those used by the Apache +web server.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + +
Table 1: CGI MIME Media Types
MIME Media TypeDescription
application/x-httpd-cgiCGI script/program
application/x-httpd-javaJava program
application/x-httpd-perlPerl script
application/x-httpd-phpPHP script
application/x-httpd-pythonPython script
+ +

Configuring the Server

+ +

In order to enable the corresponding type, you must create a +new /etc/cups/cgi.types file which maps the filename +extensions to the appropriate MIME types, for example:

+ +
+application/x-httpd-cgi cgi
+application/x-httpd-java class
+application/x-httpd-perl pl
+application/x-httpd-php php
+application/x-httpd-python py
+
+ +

CGI scripts/programs (application/x-httpd-cgi) also must have execution +permissions to be treated as a CGI script or program.

+ +

Limitations

+ +

CUPS implements most of the CGI/1.1 specification, with the +following exceptions:

+ +
    + +
  • No PATH_INFO or PATH_TRANSLATED support
  • + +
  • Limited HTTP field support; only the Content-Length + (CONTENT_LENGTH), Cookie (HTTP_COOKIE), and User-Agent + (HTTP_USER_AGENT) fields are placed in environment + variables at this time
  • + +
+ + + diff --git a/doc/help/glossary.html b/doc/help/glossary.html new file mode 100644 index 0000000000..c6898dc7df --- /dev/null +++ b/doc/help/glossary.html @@ -0,0 +1,219 @@ + + + + Glossary + + + + +

Glossary

+ +

A

+ +
+ +
ASCII +
American Standard Code for Information Interchange + +
+ + +

C

+ +
+ +
C +
A computer language + +
Character Set +
The association of numbers with specific printed or + displayed characters or symbols + +
+ + +

E

+ +
+ +
ESC/P +
EPSON Standard Code for Printers + +
+ + +

F

+ +
+ +
FTP +
File Transfer Protocol + +
+ + +

G

+ +
+ +
GIF +
Graphics Interchange Format + +
+ + +

H

+ +
+ +
HP-GL +
Hewlett-Packard Graphics Language + +
HP-PCL +
Hewlett-Packard Page Control Language + +
HP-PJL +
Hewlett-Packard Printer Job Language + +
+ + +

I

+ +
+ +
IETF +
Internet Engineering Task Force + +
IP +
Internet Protocol + +
IPv4 +
Internet Protocol, version 4; IPv4 addresses are 32-bits in + length and often look like "nnn.nnn.nnn.nnn" and "127.0.0.1" + +
IPv6 +
Internet Protocol, version 6: IPv6 addresses are 128-bits in + length and look like "xxxx::xxxx:xxxx:xxxx:xxxx" and "::1" + +
IPP +
Internet Printing Protocol + +
ISO +
International Standards Organization + +
+ + +

J

+ +
+ +
JFIF +
JPEG File Interchange Format + +
JPEG +
Joint Photographic Experts Group + +
+ + +

L

+ +
+ +
LPD +
Line Printer Daemon + +
+ + +

M

+ +
+ +
MIME +
Multimedia Internet Mail Exchange + +
+ + +

P

+ +
+ +
parallel +
Sending or receiving data more than 1 bit at a time + +
PDF +
Portable Document Format + +
pipe +
A one-way communications channel between two programs + +
PNG +
Portable Network Graphics + +
PostScript +
A page description language that is most often used + for printing + +
PPD +
PostScript Printer Description + +
+ + +

S

+ +
+ +
SCSI +
Small Computer Systems Interface + +
serial +
Sending or receiving data 1 bit at a time + +
SMB +
Server Message Block + +
socket +
A two-way network communications channel + +
+ + +

T

+ +
+ +
TCP +
Transmission Control Protocol + +
TFTP +
Trivial File Transfer Protocol + +
TIFF +
Tagged Image File Format + +
+ + +

U

+ +
+ +
UDP +
Unicast Datagram Protocol + +
Unicode +
A universal character set for all languages of the + world + +
UTF-8 +
Unicode Transfer Format 8-Bit + +
+ + + diff --git a/doc/help/kerberos.html b/doc/help/kerberos.html new file mode 100644 index 0000000000..cb8b88c29d --- /dev/null +++ b/doc/help/kerberos.html @@ -0,0 +1,89 @@ + + + + Using Kerberos Authentication + + + + +

Using Kerberos Authentication

+ +

CUPS allows you to use a Key Distribution Center (KDC) for authentication on your local CUPS server and when printing to a remote authenticated queue. This document describes how to configure CUPS to use Kerberos authentication and provides links to the MIT help pages for configuring Kerberos on your systems and network.

+ + +

System Requirements

+ +

The following are required to use Kerberos with CUPS:

+ +
    + +
  1. Heimdal Kerberos (any version) or MIT Kerberos (1.6.3 or newer)
  2. + +
  3. Properly configured Domain Name System (DNS) infrastructure (for your servers):
      +
    1. DNS server(s) with static IP addresses for all CUPS servers or configured to allow DHCP updates to the host addresses and
    2. +
    3. All CUPS clients and servers configured to use the same +DNS server(s).
    4. +
  4. + +
  5. Properly configured Kerberos infrastructure:
      +
    1. KDC configured to allow CUPS servers to obtain Service Granting Tickets (SGTs) for the "host" service,
    2. +
    3. LDAP-based user accounts - both OpenDirectory and ActiveDirectory provide this with the KDC, and
    4. +
    5. CUPS clients and servers bound to the same KDC and LDAP + server(s).
    6. +
  6. + +
  7. A "host" Service Granting Ticket (SGT) for every CUPS server
  8. + +
+ + +

Configuring Kerberos on Your System

+ +

Before you can use Kerberos with CUPS, you will need to configure Kerberos on your system and setup a system as a KDC. Because this configuration is highly system and site-specific, please consult the following on-line resources provided by the creators of Kerberos at the Massachusetts Institute of Technology (MIT):

+ + + +

The Linux Documentation Project also has a HOWTO on Kerberos:

+ + + + +

Configuring CUPS to Use Kerberos

+ +

Once youhave configured Kerberos on your system(s), you can then enable Kerberos authentication by selecting the Negotiate authentication type. The simplest way to do this is using the cupsctl(8) command on your server(s):

+ +
+cupsctl DefaultAuthType=Negotiate
+
+ +

You can also enable Kerberos from the web interface by checking the Use Kerberos Authentication box and clicking Change Settings:

+ +
+http://server.example.com:631/admin
+
+ +

After you have enabled Kerberos authentication, use the built-in "authenticated" policy or your own custom policies with the printers you will be sharing. See Managing Operation Policies for more information.

+ + +

Implementation Information

+ +

CUPS implements Kerberos over HTTP using GSSAPI and the service name "host". Because of limitations in the HTTP GSSAPI protocol extension, only a single domain/KDC is supported for authentication. The HTTP extension is described in RFC 4559.

+ +

When doing printing tasks that require authentication, CUPS requests single-use "tickets" from your login session to authenticate who you are. These tickets give CUPS a username of the form "user@REALM", which is then converted to just "user" for purposes of user and group checks.

+ +

In order to support printing to a shared printer, CUPS runs the IPP backend as the owner of the print job so it can obtain the necessary credentials.

+ + + diff --git a/doc/help/license.html b/doc/help/license.html new file mode 100644 index 0000000000..2b7c2460f7 --- /dev/null +++ b/doc/help/license.html @@ -0,0 +1,1076 @@ + + + + Software License Agreement + + + + +

Software License Agreement

+ +

Copyright 2007-2011 by Apple Inc.
+1 Infinite Loop
+Cupertino, CA 95014 USA
+
+WWW: http://www.cups.org/ + +

Introduction

+ +

CUPSTM is provided under the GNU General Public License ("GPL") +and GNU Library General Public License ("LGPL"), Version 2, with exceptions for +Apple operating systems and the OpenSSL toolkit. A copy of the exceptions and +licenses follow this introduction.

+ +

The GNU LGPL applies to the CUPS and CUPS Imaging libraries located in the +"cups" and "filter" subdirectories of the CUPS source distribution and the files +in the "test" subdirectory. The GNU GPL applies to the remainder of the CUPS +distribution.

+ +

For those not familiar with the GNU GPL, the license basically +allows you to:

+ +
    + +
  • Use the CUPS software at no charge.
  • + +
  • Distribute verbatim copies of the software in source + or binary form.
  • + +
  • Sell verbatim copies of the software for a media + fee, or sell support for the software.
  • + +
+ +

What this license does not allow you to do is make +changes or add features to CUPS and then sell a binary +distribution without source code. You must provide source for any +changes or additions to the software, and all code must be +provided under the GPL or LGPL as appropriate. The only +exceptions to this are the portions of the CUPS software covered +by the Apple operating system license exceptions outlined later +in this license agreement.

+ +

The GNU LGPL relaxes the "link-to" restriction, allowing you +to develop applications that use the CUPS and CUPS Imaging +libraries under other licenses and/or conditions as appropriate +for your application, driver, or filter.

+ + +

License Exceptions

+ +

In addition, as the copyright holder of CUPS, Apple Inc. grants +the following special exceptions:

+ +
    + +
  1. Apple Operating System Development License + Exception; + +
      + +
    1. Software that is developed by any person or + entity for an Apple Operating System ("Apple + OS-Developed Software"), including but not + limited to Apple and third party printer + drivers, filters, and backends for an Apple + Operating System, that is linked to the CUPS + imaging library or based on any sample filters + or backends provided with CUPS shall not be + considered to be a derivative work or collective + work based on the CUPS program and is exempt + from the mandatory source code release clauses + of the GNU GPL. You may therefore distribute + linked combinations of the CUPS imaging library + with Apple OS-Developed Software without + releasing the source code of the Apple + OS-Developed Software. You may also use sample + filters and backends provided with CUPS to + develop Apple OS-Developed Software without + releasing the source code of the Apple + OS-Developed Software.
    2. + +
    3. An Apple Operating System means any + operating system software developed and/or + marketed by Apple Computer, Inc., including but + not limited to all existing releases and + versions of Apple's Darwin, Mac OS X, and Mac OS + X Server products and all follow-on releases and + future versions thereof.
    4. + +
    5. This exception is only available for Apple + OS-Developed Software and does not apply to + software that is distributed for use on other + operating systems.
    6. + +
    7. All CUPS software that falls under this + license exception have the following text at the + top of each source file: + +
      This file is subject to the Apple + OS-Developed Software + exception.
    8. + +
    + +
  2. OpenSSL Toolkit License Exception; + +
      + +
    1. Apple Inc. explicitly allows the + compilation and distribution of the CUPS + software with the OpenSSL Toolkit.
    2. + +
    + +
+ +

No developer is required to provide these exceptions in a +derived work.

+ + +

Kerberos Support Code

+ +

The Kerberos support code ("KSC") is copyright 2006 by Jelmer +Vernooij and is provided 'as-is', without any express or implied +warranty. In no event will the author or Apple Inc. be held liable +for any damages arising from the use of the KSC.

+ +

Sources files containing KSC have the following text at the top +of each source file:

+ +
This file contains Kerberos support code, copyright +2006 by Jelmer Vernooij.
+ +

The KSC copyright and license apply only to +Kerberos-related feature code in CUPS. Such code is typically +conditionally compiled based on the present of the +HAVE_GSSAPI preprocessor definition.

+ +

Permission is granted to anyone to use the KSC for any +purpose, including commercial applications, and to alter it and +redistribute it freely, subject to the following +restrictions:

+ +
    + +
  1. The origin of the KSC must not be misrepresented; you + must not claim that you wrote the original software. If + you use the KSC in a product, an acknowledgment in the + product documentation would be appreciated but is not + required.
  2. + +
  3. Altered source versions must be plainly marked as + such, and must not be misrepresented as being the + original software.
  4. + +
  5. This notice may not be removed or altered from any + source distribution.
  6. + +
+ + +

Trademarks

+ +

CUPS and the CUPS logo (the "CUPS Marks") are trademarks of Apple Inc. Apple +grants you a non-exclusive and non-transferable right to use the CUPS Marks in +any direct port or binary distribution incorporating CUPS software and in any +promotional material therefor. You agree that your products will meet the +highest levels of quality and integrity for similar goods, not be unlawful, and +be developed, manufactured, and distributed in compliance with this license. +You will not interfere with Apple's rights in the CUPS Marks, and all use of the +CUPS Marks shall inure to the benefit of Apple. This license does not apply to +use of the CUPS Marks in a derivative products, which requires prior written +permission from Apple Inc.

+ + +

GNU GENERAL PUBLIC LICENSE

+ +

Version 2, June 1991 + +

+Copyright 1989, 1991 Free Software Foundation, Inc.
+59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+
+Everyone is permitted to copy and distribute verbatim
+copies of this license document, but changing it is not allowed.
+
+ +

Preamble

+ +

The licenses for most software are designed to take away your +freedom to share and change it. By contrast, the GNU General Public +License is intended to guarantee your freedom to share and change free +software--to make sure the software is free for all its users. This +General Public License applies to most of the Free Software +Foundation's software and to any other program whose authors commit to +using it. (Some other Free Software Foundation software is covered by +the GNU Library General Public License instead.) You can apply it to +your programs, too. + +

When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +this service if you wish), that you receive source code or can get it +if you want it, that you can change the software or use pieces of it +in new free programs; and that you know you can do these things. + +

To protect your rights, we need to make restrictions that forbid +anyone to deny you these rights or to ask you to surrender the rights. +These restrictions translate to certain responsibilities for you if you +distribute copies of the software, or if you modify it. + +

For example, if you distribute copies of such a program, whether +gratis or for a fee, you must give the recipients all the rights that +you have. You must make sure that they, too, receive or can get the +source code. And you must show them these terms so they know their +rights. + +

We protect your rights with two steps: (1) copyright the software, and +(2) offer you this license which gives you legal permission to copy, +distribute and/or modify the software. + +

Also, for each author's protection and ours, we want to make certain +that everyone understands that there is no warranty for this free +software. If the software is modified by someone else and passed on, we +want its recipients to know that what they have is not the original, so +that any problems introduced by others will not reflect on the original +authors' reputations. + +

Finally, any free program is threatened constantly by software +patents. We wish to avoid the danger that redistributors of a free +program will individually obtain patent licenses, in effect making the +program proprietary. To prevent this, we have made it clear that any +patent must be licensed for everyone's free use or not licensed at all. + +

The precise terms and conditions for copying, distribution and +modification follow. + +

GNU GENERAL PUBLIC LICENSE
+TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION

+ +
    + +
  1. This License applies to any program or other work which contains +a notice placed by the copyright holder saying it may be distributed +under the terms of this General Public License. The "Program", below, +refers to any such program or work, and a "work based on the Program" +means either the Program or any derivative work under copyright law: +that is to say, a work containing the Program or a portion of it, +either verbatim or with modifications and/or translated into another +language. (Hereinafter, translation is included without limitation in +the term "modification".) Each licensee is addressed as "you". + +

    Activities other than copying, distribution and modification are not +covered by this License; they are outside its scope. The act of +running the Program is not restricted, and the output from the Program +is covered only if its contents constitute a work based on the +Program (independent of having been made by running the Program). +Whether that is true depends on what the Program does. + +

  2. You may copy and distribute verbatim copies of the Program's +source code as you receive it, in any medium, provided that you +conspicuously and appropriately publish on each copy an appropriate +copyright notice and disclaimer of warranty; keep intact all the +notices that refer to this License and to the absence of any warranty; +and give any other recipients of the Program a copy of this License +along with the Program. + +

    You may charge a fee for the physical act of transferring a copy, and +you may at your option offer warranty protection in exchange for a fee. + +

  3. You may modify your copy or copies of the Program or any portion +of it, thus forming a work based on the Program, and copy and +distribute such modifications or work under the terms of Section 1 +above, provided that you also meet all of these conditions: + +
      + +
    1. You must cause the modified files to carry prominent notices +stating that you changed the files and the date of any change. + +
    2. You must cause any work that you distribute or publish, that in +whole or in part contains or is derived from the Program or any +part thereof, to be licensed as a whole at no charge to all third +parties under the terms of this License. + +
    3. if the modified program normally reads commands interactively +when run, you must cause it, when started running for such +interactive use in the most ordinary way, to print or display an +announcement including an appropriate copyright notice and a +notice that there is no warranty (or else, saying that you provide +a warranty) and that users may redistribute the program under +these conditions, and telling the user how to view a copy of this +License. (Exception: if the Program itself is interactive but +does not normally print such an announcement, your work based on +the Program is not required to print an announcement.) + +
    + +

    These requirements apply to the modified work as a whole. If +identifiable sections of that work are not derived from the Program, +and can be reasonably considered independent and separate works in +themselves, then this License, and its terms, do not apply to those +sections when you distribute them as separate works. But when you +distribute the same sections as part of a whole which is a work based +on the Program, the distribution of the whole must be on the terms of +this License, whose permissions for other licensees extend to the +entire whole, and thus to each and every part regardless of who wrote it. + +

    Thus, it is not the intent of this section to claim rights or contest +your rights to work written entirely by you; rather, the intent is to +exercise the right to control the distribution of derivative or +collective works based on the Program. + +

    In addition, mere aggregation of another work not based on the Program +with the Program (or with a work based on the Program) on a volume of +a storage or distribution medium does not bring the other work under +the scope of this License. + +

  4. You may copy and distribute the Program (or a work based on it, +under Section 2) in object code or executable form under the terms of +Sections 1 and 2 above provided that you also do one of the following: + +
      + +
    1. Accompany it with the complete corresponding machine-readable +source code, which must be distributed under the terms of Sections +1 and 2 above on a medium customarily used for software interchange; or, + +
    2. Accompany it with a written offer, valid for at least three +years, to give any third party, for a charge no more than your +cost of physically performing source distribution, a complete +machine-readable copy of the corresponding source code, to be +distributed under the terms of Sections 1 and 2 above on a medium +customarily used for software interchange; or, + +
    3. Accompany it with the information you received as to the offer +to distribute corresponding source code. (This alternative is +allowed only for noncommercial distribution and only if you +received the program in object code or executable form with such +an offer, in accord with Subsection b above.) + +
    + +

    The source code for a work means the preferred form of the work for +making modifications to it. For an executable work, complete source +code means all the source code for all modules it contains, plus any +associated interface definition files, plus the scripts used to +control compilation and installation of the executable. However, as a +special exception, the source code distributed need not include +anything that is normally distributed (in either source or binary +form) with the major components (compiler, kernel, and so on) of the +operating system on which the executable runs, unless that component +itself accompanies the executable. + +

    If distribution of executable or object code is made by offering +access to copy from a designated place, then offering equivalent +access to copy the source code from the same place counts as +distribution of the source code, even though third parties are not +compelled to copy the source along with the object code. + +

  5. You may not copy, modify, sublicense, or distribute the Program +except as expressly provided under this License. Any attempt +otherwise to copy, modify, sublicense or distribute the Program is +void, and will automatically terminate your rights under this License. +However, parties who have received copies, or rights, from you under +this License will not have their licenses terminated so long as such +parties remain in full compliance. + +
  6. You are not required to accept this License, since you have not +signed it. However, nothing else grants you permission to modify or +distribute the Program or its derivative works. These actions are +prohibited by law if you do not accept this License. Therefore, by +modifying or distributing the Program (or any work based on the +Program), you indicate your acceptance of this License to do so, and +all its terms and conditions for copying, distributing or modifying +the Program or works based on it. + +
  7. Each time you redistribute the Program (or any work based on the +Program), the recipient automatically receives a license from the +original licensor to copy, distribute or modify the Program subject to +these terms and conditions. You may not impose any further +restrictions on the recipients' exercise of the rights granted herein. +You are not responsible for enforcing compliance by third parties to +this License. + +
  8. If, as a consequence of a court judgment or allegation of patent +infringement or for any other reason (not limited to patent issues), +conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot +distribute so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you +may not distribute the Program at all. For example, if a patent +license would not permit royalty-free redistribution of the Program by +all those who receive copies directly or indirectly through you, then +the only way you could satisfy both it and this License would be to +refrain entirely from distribution of the Program. + +

    If any portion of this section is held invalid or unenforceable under +any particular circumstance, the balance of the section is intended to +apply and the section as a whole is intended to apply in other +circumstances. + +

    It is not the purpose of this section to induce you to infringe any +patents or other property right claims or to contest validity of any +such claims; this section has the sole purpose of protecting the +integrity of the free software distribution system, which is +implemented by public license practices. Many people have made +generous contributions to the wide range of software distributed +through that system in reliance on consistent application of that +system; it is up to the author/donor to decide if he or she is willing +to distribute software through any other system and a licensee cannot +impose that choice. + +

    This section is intended to make thoroughly clear what is believed to +be a consequence of the rest of this License. + +

  9. If the distribution and/or use of the Program is restricted in +certain countries either by patents or by copyrighted interfaces, the +original copyright holder who places the Program under this License +may add an explicit geographical distribution limitation excluding +those countries, so that distribution is permitted only in or among +countries not thus excluded. In such case, this License incorporates +the limitation as if written in the body of this License. + +
  10. The Free Software Foundation may publish revised and/or new versions +of the General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + +

    Each version is given a distinguishing version number. If the Program +specifies a version number of this License which applies to it and "any +later version", you have the option of following the terms and conditions +either of that version or of any later version published by the Free +Software Foundation. If the Program does not specify a version number of +this License, you may choose any version ever published by the Free Software +Foundation. + +

  11. If you wish to incorporate parts of the Program into other free +programs whose distribution conditions are different, write to the author +to ask for permission. For software which is copyrighted by the Free +Software Foundation, write to the Free Software Foundation; we sometimes +make exceptions for this. Our decision will be guided by the two goals +of preserving the free status of all derivatives of our free software and +of promoting the sharing and reuse of software generally. + +
+ +

NO WARRANTY

+ +
    + +
  1. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY +FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN +OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES +PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED +OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF +MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS +TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE +PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, +REPAIR OR CORRECTION. + +
  2. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR +REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, +INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING +OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED +TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY +YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER +PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE +POSSIBILITY OF SUCH DAMAGES. + +
+ +

END OF TERMS AND CONDITIONS

+ +

How to Apply These Terms to Your New Programs

+ +

If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + +

To do so, attach the following notices to the program. It is safest +to attach them to the start of each source file to most effectively +convey the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + +

+one line to give the program's name and an idea of what it does.
+Copyright (C) yyyy  name of author
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
+
+ +

Also add information on how to contact you by electronic and paper mail. + +

If the program is interactive, make it output a short notice like this +when it starts in an interactive mode: + +

+Gnomovision version 69, Copyright (C) year name of author
+Gnomovision comes with ABSOLUTELY NO WARRANTY; for details
+type `show w'.  This is free software, and you are welcome
+to redistribute it under certain conditions; type `show c'
+for details.
+
+ +

The hypothetical commands `show w' and `show c' should show +the appropriate parts of the General Public License. Of course, the +commands you use may be called something other than `show w' and +`show c'; they could even be mouse-clicks or menu items--whatever +suits your program. + +

You should also get your employer (if you work as a programmer) or your +school, if any, to sign a "copyright disclaimer" for the program, if +necessary. Here is a sample; alter the names: + +

+Yoyodyne, Inc., hereby disclaims all copyright
+interest in the program `Gnomovision'
+(which makes passes at compilers) written
+by James Hacker.
+
+signature of Ty Coon, 1 April 1989
+Ty Coon, President of Vice
+
+ +

GNU LIBRARY GENERAL PUBLIC LICENSE

+ +

Version 2, June 1991 + +

+Copyright (C) 1991 Free Software Foundation, Inc.
+59 Temple Place - Suite 330, Boston, MA  02111-1307, USA
+Everyone is permitted to copy and distribute verbatim copies
+of this license document, but changing it is not allowed.
+
+[This is the first released version of the library GPL.  It is
+ numbered 2 because it goes with version 2 of the ordinary GPL.]
+
+ +

Preamble

+ +

The licenses for most software are designed to take away your +freedom to share and change it. By contrast, the GNU General Public +Licenses are intended to guarantee your freedom to share and change +free software--to make sure the software is free for all its users. + +

This license, the Library General Public License, applies to some +specially designated Free Software Foundation software, and to any +other libraries whose authors decide to use it. You can use it for +your libraries, too. + +

When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +this service if you wish), that you receive source code or can get it +if you want it, that you can change the software or use pieces of it +in new free programs; and that you know you can do these things. + +

To protect your rights, we need to make restrictions that forbid +anyone to deny you these rights or to ask you to surrender the rights. +These restrictions translate to certain responsibilities for you if +you distribute copies of the library, or if you modify it. + +

For example, if you distribute copies of the library, whether gratis +or for a fee, you must give the recipients all the rights that we gave +you. You must make sure that they, too, receive or can get the source +code. If you link a program with the library, you must provide +complete object files to the recipients so that they can relink them +with the library, after making changes to the library and recompiling +it. And you must show them these terms so they know their rights. + +

Our method of protecting your rights has two steps: (1) copyright +the library, and (2) offer you this license which gives you legal +permission to copy, distribute and/or modify the library. + +

Also, for each distributor's protection, we want to make certain +that everyone understands that there is no warranty for this free +library. If the library is modified by someone else and passed on, we +want its recipients to know that what they have is not the original +version, so that any problems introduced by others will not reflect on +the original authors' reputations. + +

Finally, any free program is threatened constantly by software +patents. We wish to avoid the danger that companies distributing free +software will individually obtain patent licenses, thus in effect +transforming the program into proprietary software. To prevent this, +we have made it clear that any patent must be licensed for everyone's +free use or not licensed at all. + +

Most GNU software, including some libraries, is covered by the ordinary +GNU General Public License, which was designed for utility programs. This +license, the GNU Library General Public License, applies to certain +designated libraries. This license is quite different from the ordinary +one; be sure to read it in full, and don't assume that anything in it is +the same as in the ordinary license. + +

The reason we have a separate public license for some libraries is that +they blur the distinction we usually make between modifying or adding to a +program and simply using it. Linking a program with a library, without +changing the library, is in some sense simply using the library, and is +analogous to running a utility program or application program. However, in +a textual and legal sense, the linked executable is a combined work, a +derivative of the original library, and the ordinary General Public License +treats it as such. + +

Because of this blurred distinction, using the ordinary General +Public License for libraries did not effectively promote software +sharing, because most developers did not use the libraries. We +concluded that weaker conditions might promote sharing better. + +

However, unrestricted linking of non-free programs would deprive the +users of those programs of all benefit from the free status of the +libraries themselves. This Library General Public License is intended to +permit developers of non-free programs to use free libraries, while +preserving your freedom as a user of such programs to change the free +libraries that are incorporated in them. (We have not seen how to achieve +this as regards changes in header files, but we have achieved it as regards +changes in the actual functions of the Library.) The hope is that this +will lead to faster development of free libraries. + +

The precise terms and conditions for copying, distribution and +modification follow. Pay close attention to the difference between a +"work based on the library" and a "work that uses the library". The +former contains code derived from the library, while the latter only +works together with the library. + +

Note that it is possible for a library to be covered by the ordinary +General Public License rather than by this special one. + +

TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION

+ +

0. +This License Agreement applies to any software library which +contains a notice placed by the copyright holder or other authorized +party saying it may be distributed under the terms of this Library +General Public License (also called "this License"). Each licensee is +addressed as "you". + +

A "library" means a collection of software functions and/or data +prepared so as to be conveniently linked with application programs +(which use some of those functions and data) to form executables. + +

The "Library", below, refers to any such software library or work +which has been distributed under these terms. A "work based on the +Library" means either the Library or any derivative work under +copyright law: that is to say, a work containing the Library or a +portion of it, either verbatim or with modifications and/or translated +straightforwardly into another language. (Hereinafter, translation is +included without limitation in the term "modification".) + +

"Source code" for a work means the preferred form of the work for +making modifications to it. For a library, complete source code means +all the source code for all modules it contains, plus any associated +interface definition files, plus the scripts used to control compilation +and installation of the library. + +

Activities other than copying, distribution and modification are not +covered by this License; they are outside its scope. The act of +running a program using the Library is not restricted, and output from +such a program is covered only if its contents constitute a work based +on the Library (independent of the use of the Library in a tool for +writing it). Whether that is true depends on what the Library does +and what the program that uses the Library does. + +

1. +You may copy and distribute verbatim copies of the Library's +complete source code as you receive it, in any medium, provided that +you conspicuously and appropriately publish on each copy an +appropriate copyright notice and disclaimer of warranty; keep intact +all the notices that refer to this License and to the absence of any +warranty; and distribute a copy of this License along with the +Library. + +

You may charge a fee for the physical act of transferring a copy, +and you may at your option offer warranty protection in exchange for a +fee. + +

2. +You may modify your copy or copies of the Library or any portion +of it, thus forming a work based on the Library, and copy and +distribute such modifications or work under the terms of Section 1 +above, provided that you also meet all of these conditions: + +

    + +
  1. The modified work must itself be a software library. + +

    +

  2. You must cause the files modified to carry prominent notices + stating that you changed the files and the date of any change. + +

    +

  3. You must cause the whole of the work to be licensed at no + charge to all third parties under the terms of this License. + +

    +

  4. If a facility in the modified Library refers to a function or a + table of data to be supplied by an application program that uses + the facility, other than as an argument passed when the facility + is invoked, then you must make a good faith effort to ensure that, + in the event an application does not supply such function or + table, the facility still operates, and performs whatever part of + its purpose remains meaningful. + +

    (For example, a function in a library to compute square roots has + a purpose that is entirely well-defined independent of the + application. Therefore, Subsection 2d requires that any + application-supplied function or table used by this function must + be optional: if the application does not supply it, the square + root function must still compute square roots.) + +

+ +

These requirements apply to the modified work as a whole. If +identifiable sections of that work are not derived from the Library, +and can be reasonably considered independent and separate works in +themselves, then this License, and its terms, do not apply to those +sections when you distribute them as separate works. But when you +distribute the same sections as part of a whole which is a work based +on the Library, the distribution of the whole must be on the terms of +this License, whose permissions for other licensees extend to the +entire whole, and thus to each and every part regardless of who wrote +it. + +

Thus, it is not the intent of this section to claim rights or contest +your rights to work written entirely by you; rather, the intent is to +exercise the right to control the distribution of derivative or +collective works based on the Library. + +

In addition, mere aggregation of another work not based on the Library +with the Library (or with a work based on the Library) on a volume of +a storage or distribution medium does not bring the other work under +the scope of this License. + +

3. +You may opt to apply the terms of the ordinary GNU General Public +License instead of this License to a given copy of the Library. To do +this, you must alter all the notices that refer to this License, so +that they refer to the ordinary GNU General Public License, version 2, +instead of to this License. (If a newer version than version 2 of the +ordinary GNU General Public License has appeared, then you can specify +that version instead if you wish.) Do not make any other change in +these notices. + +

Once this change is made in a given copy, it is irreversible for +that copy, so the ordinary GNU General Public License applies to all +subsequent copies and derivative works made from that copy. + +

This option is useful when you wish to copy part of the code of +the Library into a program that is not a library. + +

4. +You may copy and distribute the Library (or a portion or +derivative of it, under Section 2) in object code or executable form +under the terms of Sections 1 and 2 above provided that you accompany +it with the complete corresponding machine-readable source code, which +must be distributed under the terms of Sections 1 and 2 above on a +medium customarily used for software interchange. + +

If distribution of object code is made by offering access to copy +from a designated place, then offering equivalent access to copy the +source code from the same place satisfies the requirement to +distribute the source code, even though third parties are not +compelled to copy the source along with the object code. + +

5. +A program that contains no derivative of any portion of the +Library, but is designed to work with the Library by being compiled or +linked with it, is called a "work that uses the Library". Such a +work, in isolation, is not a derivative work of the Library, and +therefore falls outside the scope of this License. + +

However, linking a "work that uses the Library" with the Library +creates an executable that is a derivative of the Library (because it +contains portions of the Library), rather than a "work that uses the +library". The executable is therefore covered by this License. +Section 6 states terms for distribution of such executables. + +

When a "work that uses the Library" uses material from a header file +that is part of the Library, the object code for the work may be a +derivative work of the Library even though the source code is not. +Whether this is true is especially significant if the work can be +linked without the Library, or if the work is itself a library. The +threshold for this to be true is not precisely defined by law. + +

If such an object file uses only numerical parameters, data +structure layouts and accessors, and small macros and small inline +functions (ten lines or less in length), then the use of the object +file is unrestricted, regardless of whether it is legally a derivative +work. (Executables containing this object code plus portions of the +Library will still fall under Section 6.) + +

Otherwise, if the work is a derivative of the Library, you may +distribute the object code for the work under the terms of Section 6. +Any executables containing that work also fall under Section 6, +whether or not they are linked directly with the Library itself. + +

6. +As an exception to the Sections above, you may also compile or +link a "work that uses the Library" with the Library to produce a +work containing portions of the Library, and distribute that work +under terms of your choice, provided that the terms permit +modification of the work for the customer's own use and reverse +engineering for debugging such modifications. + +

You must give prominent notice with each copy of the work that the +Library is used in it and that the Library and its use are covered by +this License. You must supply a copy of this License. If the work +during execution displays copyright notices, you must include the +copyright notice for the Library among them, as well as a reference +directing the user to the copy of this License. Also, you must do one +of these things: + +

    + +
  1. Accompany the work with the complete corresponding + machine-readable source code for the Library including whatever + changes were used in the work (which must be distributed under + Sections 1 and 2 above); and, if the work is an executable linked + with the Library, with the complete machine-readable "work that + uses the Library", as object code and/or source code, so that the + user can modify the Library and then relink to produce a modified + executable containing the modified Library. (It is understood + that the user who changes the contents of definitions files in the + Library will not necessarily be able to recompile the application + to use the modified definitions.) + +

    +

  2. Accompany the work with a written offer, valid for at + least three years, to give the same user the materials + specified in Subsection 6a, above, for a charge no more + than the cost of performing this distribution. + +

    +

  3. If distribution of the work is made by offering access to copy + from a designated place, offer equivalent access to copy the above + specified materials from the same place. + +

    +

  4. Verify that the user has already received a copy of these + materials or that you have already sent this user a copy. + +
+ +

For an executable, the required form of the "work that uses the +Library" must include any data and utility programs needed for +reproducing the executable from it. However, as a special exception, +the source code distributed need not include anything that is normally +distributed (in either source or binary form) with the major +components (compiler, kernel, and so on) of the operating system on +which the executable runs, unless that component itself accompanies +the executable. + +

It may happen that this requirement contradicts the license +restrictions of other proprietary libraries that do not normally +accompany the operating system. Such a contradiction means you cannot +use both them and the Library together in an executable that you +distribute. + +

7. +You may place library facilities that are a work based on the +Library side-by-side in a single library together with other library +facilities not covered by this License, and distribute such a combined +library, provided that the separate distribution of the work based on +the Library and of the other library facilities is otherwise +permitted, and provided that you do these two things: + +

    + +
  1. Accompany the combined library with a copy of the same work + based on the Library, uncombined with any other library + facilities. This must be distributed under the terms of the + Sections above. + +

    +

  2. Give prominent notice with the combined library of the fact + that part of it is a work based on the Library, and explaining + where to find the accompanying uncombined form of the same work. + +
+ +

8. +You may not copy, modify, sublicense, link with, or distribute +the Library except as expressly provided under this License. Any +attempt otherwise to copy, modify, sublicense, link with, or +distribute the Library is void, and will automatically terminate your +rights under this License. However, parties who have received copies, +or rights, from you under this License will not have their licenses +terminated so long as such parties remain in full compliance. + +

9. +You are not required to accept this License, since you have not +signed it. However, nothing else grants you permission to modify or +distribute the Library or its derivative works. These actions are +prohibited by law if you do not accept this License. Therefore, by +modifying or distributing the Library (or any work based on the +Library), you indicate your acceptance of this License to do so, and +all its terms and conditions for copying, distributing or modifying +the Library or works based on it. + +

10. +Each time you redistribute the Library (or any work based on the +Library), the recipient automatically receives a license from the +original licensor to copy, distribute, link with or modify the Library +subject to these terms and conditions. You may not impose any further +restrictions on the recipients' exercise of the rights granted herein. +You are not responsible for enforcing compliance by third parties to +this License. + +

11. +If, as a consequence of a court judgment or allegation of patent +infringement or for any other reason (not limited to patent issues), +conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot +distribute so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you +may not distribute the Library at all. For example, if a patent +license would not permit royalty-free redistribution of the Library by +all those who receive copies directly or indirectly through you, then +the only way you could satisfy both it and this License would be to +refrain entirely from distribution of the Library. + +

If any portion of this section is held invalid or unenforceable under any +particular circumstance, the balance of the section is intended to apply, +and the section as a whole is intended to apply in other circumstances. + +

It is not the purpose of this section to induce you to infringe any +patents or other property right claims or to contest validity of any +such claims; this section has the sole purpose of protecting the +integrity of the free software distribution system which is +implemented by public license practices. Many people have made +generous contributions to the wide range of software distributed +through that system in reliance on consistent application of that +system; it is up to the author/donor to decide if he or she is willing +to distribute software through any other system and a licensee cannot +impose that choice. + +

This section is intended to make thoroughly clear what is believed to +be a consequence of the rest of this License. + +

12. +If the distribution and/or use of the Library is restricted in +certain countries either by patents or by copyrighted interfaces, the +original copyright holder who places the Library under this License may add +an explicit geographical distribution limitation excluding those countries, +so that distribution is permitted only in or among countries not thus +excluded. In such case, this License incorporates the limitation as if +written in the body of this License. + +

13. +The Free Software Foundation may publish revised and/or new +versions of the Library General Public License from time to time. +Such new versions will be similar in spirit to the present version, +but may differ in detail to address new problems or concerns. + +

Each version is given a distinguishing version number. If the Library +specifies a version number of this License which applies to it and +"any later version", you have the option of following the terms and +conditions either of that version or of any later version published by +the Free Software Foundation. If the Library does not specify a +license version number, you may choose any version ever published by +the Free Software Foundation. + +

14. +If you wish to incorporate parts of the Library into other free +programs whose distribution conditions are incompatible with these, +write to the author to ask for permission. For software which is +copyrighted by the Free Software Foundation, write to the Free +Software Foundation; we sometimes make exceptions for this. Our +decision will be guided by the two goals of preserving the free status +of all derivatives of our free software and of promoting the sharing +and reuse of software generally. + +

NO WARRANTY + +

15. +BECAUSE THE LIBRARY IS LICENSED FREE OF CHARGE, THERE IS NO +WARRANTY FOR THE LIBRARY, TO THE EXTENT PERMITTED BY APPLICABLE LAW. +EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR +OTHER PARTIES PROVIDE THE LIBRARY "AS IS" WITHOUT WARRANTY OF ANY +KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE +IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE +LIBRARY IS WITH YOU. SHOULD THE LIBRARY PROVE DEFECTIVE, YOU ASSUME +THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + +

16. +IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN +WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY +AND/OR REDISTRIBUTE THE LIBRARY AS PERMITTED ABOVE, BE LIABLE TO YOU +FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR +CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE +LIBRARY (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING +RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A +FAILURE OF THE LIBRARY TO OPERATE WITH ANY OTHER SOFTWARE), EVEN IF +SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH +DAMAGES. + +

END OF TERMS AND CONDITIONS

+ +

How to Apply These Terms to Your New Libraries

+ +

If you develop a new library, and you want it to be of the greatest +possible use to the public, we recommend making it free software that +everyone can redistribute and change. You can do so by permitting +redistribution under these terms (or, alternatively, under the terms of the +ordinary General Public License). + +

To apply these terms, attach the following notices to the library. It is +safest to attach them to the start of each source file to most effectively +convey the exclusion of warranty; and each file should have at least the +"copyright" line and a pointer to where the full notice is found. + +

+one line to give the library's name and an idea of what it does.
+Copyright (C) year  name of author
+
+This library is free software; you can redistribute it and/or
+modify it under the terms of the GNU Lesser General Public
+License as published by the Free Software Foundation; either
+version 2.1 of the License, or (at your option) any later version.
+
+This library is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+Lesser General Public License for more details.
+
+You should have received a copy of the GNU Lesser General Public
+License along with this library; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+
+ +

Also add information on how to contact you by electronic and paper mail. + +

You should also get your employer (if you work as a programmer) or your +school, if any, to sign a "copyright disclaimer" for the library, if +necessary. Here is a sample; alter the names: + +

+Yoyodyne, Inc., hereby disclaims all copyright interest in
+the library `Frob' (a library for tweaking knobs) written
+by James Random Hacker.
+
+signature of Ty Coon, 1 April 1990
+Ty Coon, President of Vice
+
+ +

That's all there is to it! + + + diff --git a/doc/help/man-ipptool.html b/doc/help/man-ipptool.html new file mode 100644 index 0000000000..ce090951d6 --- /dev/null +++ b/doc/help/man-ipptool.html @@ -0,0 +1,144 @@ + + + + + + ipptool(1) + + +

ipptool(1)

+

Name

+ipptool - perform internet printing protocol requests +

Synopsis

+ipptool +[ -4 ] [ -6 ] [ -C ] [ -E ] [ -I ] [ -L ] [ -S ] [ -T +seconds +] [ -V +version +] [ -X ] [ -c ] [ -d +name=value +] [ -f +filename +] [ -i +seconds +] [ -n +repeat-count +] [ -q ] [ -t ] [ -v ] +URI +filename +[ +... filenameN +] +

Description

+ipptool sends IPP requests to the specified URI and tests and/or displays the results. Each named file defines one or more requests, including the expected response status, attributes, and values. Output is either a plain text, formatted text, CSV, or XML report on the standard output, with a non-zero exit status indicating that one or more tests have failed. The file format is described in ipptoolfile(5). +

Options

+The following options are recognized by ipptool: +
+
-4 +
+
Specifies that ipptool must connect to the printer or server using IPv4. +
+
-6 +
+
Specifies that ipptool must connect to the printer or server using IPv6. +
+
-C +
+
Specifies that requests should be sent using the HTTP/1.1 "Transfer-Encoding: chunked" header, which is required for conformance by all versions of IPP. The default is to use "Transfer-Encoding: chunked" for requests with attached files and "Content-Length:" for requests without attached files. +
+
-E +
+
Forces TLS encryption when connecting to the server using the HTTP "Upgrade" header. +
+
-I +
+
Specifies that ipptool will continue past errors. +
+
-L +
+
Specifies that requests should be sent using the HTTP/1.0 "Content-Length:" header, which is required for conformance by all versions of IPP. The default is to use "Transfer-Encoding: chunked" for requests with attached files and "Content-Length:" for requests without attached files. +
+
-S +
+
Forces (dedicated) SSL encryption when connecting to the server. +
+
-T seconds +
+
Specifies a timeout for IPP requests in seconds. +
+
-V version +
+
Specifies the default IPP version to use: 1.0, 1.1, 2.0, 2.1, or 2.2. If not specified, version 1.1 is used. +
+
-X +
+
Specifies that XML (Apple plist) output is desired instead of the plain text report. This option is incompatible with the -i (interval) and -n (repeat-count) options. +
+
-c +
+
Specifies that CSV (comma-separated values) output is desired instead of the plain text output. +
+
-d name=value +
+
Defines the named variable. +
+
-f filename +
+
Defines the default request filename for tests. +
+
-i seconds +
+
Specifies that the (last) file should be repeated at the specified interval. This option is incompatible with the -X (XML plist output) option. +
+
-l +
+
Specifies that plain text output is desired. +
+
-n repeat-count +
+
Specifies that the (last) file should be repeated the specified number of times. This option is incompatible with the -X (XML plist output) option. +
+
-t +
+
Specifies that CUPS test report output is desired instead of the plain text output. +
+
-v +
+
Specifies that all request and response attributes should be output in CUPS test mode (-t). This is the default for XML output. +
+
+

Compatibility

+The ipptool program is unique to CUPS. +

Examples

+Get a list of completed jobs for "myprinter": +
+    ipptool ipp://localhost/printers/myprinter get-completed-jobs.test
+
+

Send email notifications to "user@example.com" when "myprinter" changes: +

+    ipptool -d recipient=mailto:user@example.com 
+        ipp://localhost/printers/myprinter create-printer-subscription.test
+
+

Standard Files

+The following standard files are available: +
+    create-printer-subscription.test
+    get-completed-jobs.test
+    get-jobs.test
+    ipp-1.1.test
+    ipp-2.0.test
+    ipp-2.1.test
+    testfile.jpg
+    testfile.pdf
+    testfile.ps
+    testfile.txt
+
+

See Also

+ipptoolfile(5), +
+http://localhost:631/help +

Copyright

+Copyright 2007-2010 by Apple Inc. + + + diff --git a/doc/help/man-ipptoolfile.html b/doc/help/man-ipptoolfile.html new file mode 100644 index 0000000000..baab6fefba --- /dev/null +++ b/doc/help/man-ipptoolfile.html @@ -0,0 +1,607 @@ + + + + + + ipptoolfile(5) + + +

ipptoolfile(5)

+

Name

+ipptoolfile - ipptool file format + +

Description

+The ipptool(1) program accepts free-form plain text files that describe one or more IPP requests. Comments start with the "#" character and continue to the end of the line. Each request is enclosed by curley braces, for example: +
+
+    # This is a comment
+    {
+      # The name of the test
+      NAME "Print PostScript Job"
+
+      # The request to send
+      OPERATION Print-Job
+      GROUP operation-attributes-tag
+      ATTR charset attributes-charset utf-8
+      ATTR language attributes-natural-language en
+      ATTR uri printer-uri $uri
+      ATTR name requesting-user-name $user
+      FILE testfile.ps
+
+      # The response to expect
+      STATUS successful-ok
+      EXPECT attributes-charset OF-TYPE charset
+      EXPECT attributes-natural-language OF-TYPE naturalLanguage
+      EXPECT job-id OF-TYPE integer
+      EXPECT job-uri OF-TYPE uri
+    }
+    {
+      # The name of the test
+      NAME "Get Attributes of PostScript Job"
+
+      # The request to send
+      OPERATION Get-Job-Attributes
+      GROUP operation-attributes-tag
+      ATTR charset attributes-charset utf-8
+      ATTR language attributes-natural-language en
+      ATTR uri printer-uri $uri
+      ATTR integer job-id $job-id
+      ATTR name requesting-user-name $user
+
+      # The response to expect
+      STATUS successful-ok
+      EXPECT attributes-charset OF-TYPE charset
+      EXPECT attributes-natural-language OF-TYPE naturalLanguage
+      EXPECT job-id OF-TYPE integer
+      EXPECT job-uri OF-TYPE uri
+      EXPECT job-state OF-TYPE enum
+      EXPECT job-originating-user-name OF-TYPE name WITH-VALUE "$user"
+    }
+
+ +

Top-level Directives

+The following directives can be used outside of a test: +
+
{ test } +
+
Defines a test. +
+
DEFINE variable-name value +
+
Defines the named variable to the given value. This is equivalent to specifying +"-d variable-name=value" on the ipptool command-line. +
+
DEFINE-DEFAULT variable-name value +
+
Defines the named variable to the given value if it does not already have a +value. +
+
IGNORE-ERRORS yes +
+
+
IGNORE-ERRORS no +
+
Specifies whether, by default, ipptool will ignore errors and continue with +subsequent tests. +
+
INCLUDE "filename" +
+
+
INCLUDE <filename> +
+
Includes another test file. The first form includes a file relative to the +current test file, while the second form includes a file from the ipptool +include directory. +
+
INCLUDE-IF-DEFINED name "filename" +
+
+
INCLUDE-IF-DEFINED name <filename> +
+
Includes another test file if the named variable is defined. The first form +includes a file relative to the current test file, while the second form +includes a file from the ipptool include directory. +
+
INCLUDE-IF-NOT-DEFINED name "filename" +
+
+
INCLUDE-IF-NOT-DEFINED name <filename> +
+
Includes another test file if the named variable is not defined. The first form +includes a file relative to the current test file, while the second form +includes a file from the ipptool include directory. +
+
SKIP-IF-DEFINED variable-name +
+
+
SKIP-IF-NOT-DEFINED variable-name +
+
Specifies that the remainder of the test file should be skipped when the +variable is or is not defined. +
+
TRANSFER auto +
+
Specifies that tests will, by default, use "Transfer-Encoding: chunked" for +requests with attached files and "Content-Length:" for requests without attached +files. +
+
TRANSFER chunked +
+
Specifies that tests will, by default, use the HTTP/1.1 "Transfer-Encoding: +chunked" header. This is the default and is equivalent to specifying "-c" on the +ipptool command-line. Support for chunked requests is required for +conformance with all versions of IPP. +
+
TRANSFER length +
+
Specifies that tests will, by default, use the HTTP/1.0 "Content-Length:" +header. This is equivalent to specifying "-l" on the ipptool command-line. +Support for content length requests is required for conformance with all +versions of IPP. +
+
VERSION 1.0 +
+
+
VERSION 1.1 +
+
+
VERSION 2.0 +
+
+
VERSION 2.1 +
+
+
VERSION 2.2 +
+
Specifies the default IPP version number to use for the tests that follow. + +
+
+

Test Directives

+The following directives are understood in a test: +
+
ATTR tag attribute-name value(s) +
+
Adds an attribute to the test request. Values are separated by the comma (",") +character - escape commas using the "" character. +
+
ATTR collection attribute-name { MEMBER tag member-name value(s) ... } [ ... { ... } ] +
+
Adds a collection attribute to the test request. Member attributes follow the +same syntax as regular attributes and can themselves be nested collections. +Multiple collection values can be supplied as needed. +
+
DELAY seconds +
+
Specifies a delay before this test will be run. +
+
DISPLAY attribute-name +
+
Specifies that value of the named attribute should be output as part of the +test report. +
+
EXPECT attribute-name [ predicate(s) ] +
+
+
EXPECT ?attribute-name predicate(s) +
+
+
EXPECT !attribute-name +
+
Specifies that the response must/may/must not include the named attribute. +Additional requirements can be added as predicates - see the "EXPECT PREDICATES" +section for more information on predicates. +
+
FILE filename +
+
Specifies a file to include at the end of the request. This is typically used +when sending a test print file. +
+
GROUP tag +
+
Specifies the group tag for subsequent attributes in the request. +
+
IGNORE-ERRORS yes +
+
+
IGNORE-ERRORS no +
+
Specifies whether ipptool will ignore errors and continue with subsequent +tests. +
+
NAME "literal string" +
+
Specifies the human-readable name of the test. +
+
OPERATION operation-code +
+
Specifies the operation to be performed. +
+
REQUEST-ID number +
+
+
REQUEST-ID random +
+
Specifies the request-id value to use in the request, either an integer or the +word "random" to use a randomly generated value (the default). +
+
RESOURCE path +
+
Specifies an alternate resource path that is used for the HTTP POST request. +The default is the resource from the URI provided to the ipptool program. +
+
SKIP-IF-DEFINED variable-name +
+
+
SKIP-IF-NOT-DEFINED variable-name +
+
Specifies that the current test should be skipped when the variable is or is not +defined. +
+
SKIP-PREVIOUS-ERROR yes +
+
+
SKIP-PREVIOUS-ERROR no +
+
Specifies whether ipptool will skip the current test if the previous test +resulted in an error/failure. +
+
STATUS status-code [ predicate ] +
+
Specifies an expected response status-code value. Additional requirements can be +added as predicates - see the "STATUS PREDICATES" section for more information +on predicates. +
+
TRANSFER auto +
+
Specifies that this test will use "Transfer-Encoding: chunked" if it has an +attached file or "Content-Length:" otherwise. +
+
TRANSFER chunked +
+
Specifies that this test will use the HTTP/1.1 "Transfer-Encoding: chunked" +header. +
+
TRANSFER length +
+
Specifies that this test will use the HTTP/1.0 "Content-Length:" header. +
+
VERSION 1.0 +
+
+
VERSION 1.1 +
+
+
VERSION 2.0 +
+
+
VERSION 2.1 +
+
+
VERSION 2.2 +
+
Specifies the IPP version number to use for this test. + +
+
+

Expect Predicates

+The following predicates are understood following the EXPECT test directive: +
+
COUNT number +
+
Requires the EXPECT attribute to have the specified number of values. +
+
DEFINE-MATCH variable-name +
+
Defines the variable to "1" when the EXPECT condition matches. A side-effect of +this predicate is that this EXPECT will never fail a test. +
+
DEFINE-NO-MATCH variable-name +
+
Defines the variable to "1" when the EXPECT condition does not match. A side- +effect of this predicate is that this EXPECT will never fail a test. +
+
DEFINE-VALUE variable-name +
+
Defines the variable to the value of the attribute when the EXPECT condition +matches. A side-effect of this predicate is that this EXPECT will never fail a test. +
+
IF-DEFINED variable-name +
+
Makes the EXPECT conditions apply only if the specified variable is defined. +
+
IF-NOT-DEFINED variable-name +
+
Makes the EXPECT conditions apply only if the specified variable is not +defined. +
+
IN-GROUP tag +
+
Requires the EXPECT attribute to be in the specified group tag. +
+
OF-TYPE tag[,tag,...] +
+
Requires the EXPECT attribute to use the specified value tag(s). +
+
REPEAT-MATCH +
+
+
REPEAT-NO-MATCH +
+
Specifies that the current test should be repeated when the EXPECT condition +matches or does not match. +
+
SAME-COUNT-AS attribute-name +
+
Requires the EXPECT attribute to have the same number of values as the specified +parallel attribute. +
+
WITH-VALUE "literal string" +
+
Requires at least one value of the EXPECT attribute to match the literal string. +Comparisons are case-sensitive. +
+
WITH-VALUE "/regular expression/" +
+
Requires that all values of the EXPECT attribute match the regular expression, +which must conform to the POSIX regular expression syntax. +Comparisons are case-sensitive. + +
+
+

Status Predicates

+The following predicates are understood following the STATUS test directive: +
+
IF-DEFINED variable-name +
+
Makes the STATUS apply only if the specified variable is defined. +
+
IF-NOT-DEFINED variable-name +
+
Makes the STATUS apply only if the specified variable is not defined. +
+
REPEAT-MATCH +
+
+
REPEAT-NO-MATCH +
+
Specifies that the current test should be repeated when the response status-code +matches or does not match the value specified by the STATUS directive. + +
+
+

Operation Codes

+Operation codes correspond to the hexadecimal numbers (0xHHHH) and names from +RFC 2911 and other IPP extension specifications. Here is a complete list: +
+    Activate-Printer
+    CUPS-Accept-Jobs
+    CUPS-Add-Modify-Class
+    CUPS-Add-Modify-Printer
+    CUPS-Authenticate-Job
+    CUPS-Delete-Class
+    CUPS-Delete-Printer
+    CUPS-Get-Classes
+    CUPS-Get-Default
+    CUPS-Get-Devices
+    CUPS-Get-Document
+    CUPS-Get-PPD
+    CUPS-Get-PPDs
+    CUPS-Get-Printers
+    CUPS-Move-Job
+    CUPS-Reject-Jobs
+    CUPS-Set-Default
+    Cancel-Current-Job
+    Cancel-Job
+    Cancel-Jobs
+    Cancel-My-Jobs
+    Cancel-Subscription
+    Close-Job
+    Create-Job
+    Create-Job-Subscription
+    Create-Printer-Subscription
+    Deactivate-Printer
+    Disable-Printer
+    Enable-Printer
+    Get-Job-Attributes
+    Get-Jobs
+    Get-Notifications
+    Get-Printer-Attributes
+    Get-Printer-Support-Files
+    Get-Printer-Supported-Values
+    Get-Subscription-Attributes
+    Get-Subscriptions
+    Hold-Job
+    Hold-New-Jobs
+    Pause-Printer
+    Pause-Printer-After-Current-Job
+    Print-Job
+    Print-URI
+    Promote-Job
+    Purge-Jobs
+    Release-Held-New-Jobs
+    Release-Job
+    Renew-Subscription
+    Reprocess-Job
+    Restart-Job
+    Restart-Printer
+    Resubmit-Job
+    Resume-Job
+    Resume-Printer
+    Schedule-Job-After
+    Send-Document
+    Send-Notifications
+    Send-URI
+    Set-Job-Attributes
+    Set-Printer-Attributes
+    Shutdown-Printer
+    Startup-Printer
+    Suspend-Current-Job
+    Validate-Job
+
+ +

Status Codes

+Status codes correspond to the hexadecimal numbers (0xHHHH) and names from RFC +2911 and other IPP extension specifications. Here is a complete list: +
+    client-error-attributes-not-settable
+    client-error-attributes-or-values-not-supported
+    client-error-bad-request
+    client-error-charset-not-supported
+    client-error-compression-error
+    client-error-compression-not-supported
+    client-error-conflicting-attributes
+    client-error-document-access-error
+    client-error-document-format-error
+    client-error-document-format-not-supported
+    client-error-forbidden
+    client-error-gone
+    client-error-ignored-all-notifications
+    client-error-ignored-all-subscriptions
+    client-error-not-authenticated
+    client-error-not-authorized
+    client-error-not-found
+    client-error-not-possible
+    client-error-print-support-file-not-found
+    client-error-request-entity-too-large
+    client-error-request-value-too-long
+    client-error-timeout
+    client-error-too-many-subscriptions
+    client-error-uri-scheme-not-supported
+    cups-see-other
+    redirection-other-site
+    server-error-busy
+    server-error-device-error
+    server-error-internal-error
+    server-error-job-canceled
+    server-error-multiple-document-jobs-not-supported
+    server-error-not-accepting-jobs
+    server-error-operation-not-supported
+    server-error-printer-is-deactivated
+    server-error-service-unavailable
+    server-error-temporary-error
+    server-error-version-not-supported
+    successful-ok
+    successful-ok-but-cancel-subscription
+    successful-ok-conflicting-attributes
+    successful-ok-events-complete
+    successful-ok-ignored-notifications
+    successful-ok-ignored-or-substituted-attributes
+    successful-ok-ignored-subscriptions
+    successful-ok-too-many-events
+
+ +

Tags

+Value and group tags correspond to the names from RFC 2911 and other IPP +extension specifications. Here are the group tags: +
+    event-notification-attributes-tag
+    job-attributes-tag
+    operation-attributes-tag
+    printer-attributes-tag
+    subscription-attributes-tag
+    unsupported-attributes-tag
+
+

Here are the value tags: +

+    admin-define
+    boolean
+    charset
+    collection
+    dateTime
+    default
+    delete-attribute
+    enum
+    integer
+    keyword
+    mimeMediaType
+    nameWithLanguage
+    nameWithoutLanguage
+    naturalLanguage
+    no-value
+    not-settable
+    octetString
+    rangeOfInteger
+    resolution
+    textWithLanguage
+    textWithoutLanguage
+    unknown
+    unsupported
+    uri
+    uriScheme
+
+ +

Variables

+The ipptool program maintains a list of variables that can be used in any +literal string or attribute value by specifying "$variable-name". Aside from +variables defined using the "-d" option or "DEFINE" directive, the following +pre-defined variables are available: +
+
$$ +
+
Inserts a single "$" character. +
+
$ENV[name] +
+
Inserts the value of the named environment variable, or an empty string if the +environment variable is not defined. +
+
$filename +
+
Inserts the filename provided to ipptool with the "-f" option. +
+
$hostname +
+
Inserts the hostname from the URI provided to ipptool. +
+
$job-id +
+
Inserts the last job-id value returned in a test response or 0 if no job-id has +been seen. +
+
$job-uri +
+
Inserts the last job-uri value returned in a test response or an empty string if +no job-uri has been seen. +
+
$scheme +
+
Inserts the scheme from the URI provided to ipptool. +
+
$notify-subscription-id +
+
Inserts the last notify-subscription-id value returnd in a test response or 0 if +no notify-subscription-id has been seen. +
+
$port +
+
Inserts the port number from the URI provided to ipptool. +
+
$resource +
+
Inserts the resource path from the URI provided to ipptool. +
+
$uri +
+
Inserts the URI provided to ipptool. +
+
$user +
+
Inserts the current user's login name. +
+
$username +
+
Inserts the username from the URI provided to ipptool, if any. + +
+
+

See Also

+ipptool(1), +
+http://localhost:631/help + +

Copyright

+Copyright 2007-2011 by Apple Inc. + + + diff --git a/doc/help/network.html b/doc/help/network.html new file mode 100644 index 0000000000..d02d4f4036 --- /dev/null +++ b/doc/help/network.html @@ -0,0 +1,684 @@ + + + + Using Network Printers + + + + +

Using Network Printers

+ +

This help document describes how to discover, configure, and use TCP/IP network printers with CUPS.

+ +

Getting the IP Address

+ +

Every network printer or print server has a unique Internet Protocol (IP) address associated with it. This address is either configured manually or set using an automatic network protocol such as the Boot Protocol (BOOTP), Dynamic Host Control Protocol (DHCP), Reverse Address Resolution Protocol (RARP), or ZeroConf.

+ +

You can normally find the IP address of a printer on the printer's control panel or by printing the configuration or status page. The Simple Network Management Protocol (SNMP) can also be used to get the IP address remotely, which happens automatically when you visit the CUPS administration web page or choose an available CUPS device when adding a printer.

+ + +

Configuring the IP Address

+ +

When you first install a network printer or print server on your LAN, you need to set the Internet Protocol ("IP") address. Most higher-end "workgroup" printers allow you to set the address through the printer control panel. However, if you have many printers you will want to assign the addresses remotely - this makes administration a bit easier and avoids assigning duplicate addresses accidentally.

+ +

To setup your printer or print server for remote address assignment, you'll need the Ethernet Media Access Control ("MAC") address, also sometimes called a node address, and the IP address you want to use for the device. The Ethernet MAC address can often be found on the printer test page or bottom of the print server.

+ +

Configuring the IP Address Using DHCP

+ +

The DHCP protocol is the usual way of setting the IP address of a printer on a managed network. Using the standard dhcpd(8) program supplied with UNIX you simply need to add a line to the /etc/dhcpd.conf file:

+ +
+host hostname {
+  hardware ethernet mac-address;
+  fixed-address ip-address;
+}
+
+ +

Make sure that the hostname you use is also listed in the /etc/hosts file or is registered with your DNS server.

+ +

Configuring the IP Address Using BOOTP

+ +

The BOOTP protocol is used when you need to provide additional information such as the location of a configuration file to the network interface. Using the standard bootpd(8) program supplied with UNIX you simply need to add a line to the /etc/bootptab file; for IRIX:

+ +
+myprinter 08:00:69:00:12:34 192.0.2.2 myprinter.boot
+
+ +

Newer versions of bootpd use a different format:

+ +
+myprinter:ha=080069001234:ip=192.0.2.2:t144=myprinter.boot
+
+ +

The myprinter.boot file resides in the /usr/local/boot directory by default. If you do not need to provide a boot file you may leave the last part of the line blank.

+ +
Note: Some versions of UNIX do not enable the BOOTP service by default. The /etc/inetd.conf or /etc/xinetd.d/bootp file usually contains a line for the BOOTP service that can be uncommented if needed.
+ + +

Verifying the Printer Connection

+ +

To test that the IP address has been successfully assigned and +that the printer is properly connected to your LAN, type:

+ +
+ping ip-address
+
+ +

If the connection is working properly you will see something +like:

+ +
+ping myprinter
+PING myprinter (192.0.2.2): 56 data bytes
+64 bytes from 192.0.2.2: icmp_seq=0 ttl=15 time=5 ms
+64 bytes from 192.0.2.2: icmp_seq=1 ttl=15 time=3 ms
+64 bytes from 192.0.2.2: icmp_seq=2 ttl=15 time=3 ms
+64 bytes from 192.0.2.2: icmp_seq=3 ttl=15 time=3 ms
+
+ +

If not, verify that the printer or print server is connected +to the LAN, it is powered on, the LAN cabling is good, and the IP +address is set correctly. You can usually see the current IP +address and network status by printing a configuration or test +page on the device.

+ + +

Network Protocols Supported by CUPS

+ +

CUPS supports most network printers using one of three TCP/IP-based protocols. Printer discovery is currently accomplished using the SNMP protocol, however future versions of CUPS will also include support for multicast DNS service discovery as well.

+ +

AppSocket Protocol

+ +

The AppSocket protocol (sometimes also called the JetDirect protocol, owing to its origins with the HP JetDirect network interfaces) is the simplest, fastest, and generally the most reliable network protocol used for printers. AppSocket printing normally happens over port 9100 and uses the socket URI scheme:

+ +
+socket://ip-address-or-hostname
+socket://ip-address-or-hostname/?waiteof=false
+socket://ip-address-or-hostname:port-number
+socket://ip-address-or-hostname:port-number/?waiteof=false
+
+ +

The "waiteof" option controls whether the socket backend waits for the printer to complete the printing of the job. The default is to wait.

+ +

Internet Printing Protocol (IPP)

+ +

IPP is the only protocol that CUPS supports natively and is supported by some network printers and print servers. However, since many printers do not implement IPP properly, only use IPP when the vendor actually documents official support for it. IPP printing normally happens over port 631 and uses the http and ipp URI schemes:

+ +
+http://ip-address-or-hostname:port-number/resource
+http://ip-address-or-hostname:port-number/resource?option=value
+http://ip-address-or-hostname:port-number/resource?option=value&option=value
+ipp://ip-address-or-hostname/resource
+ipp://ip-address-or-hostname/resource?option=value
+ipp://ip-address-or-hostname/resource?option=value&option=value
+ipp://ip-address-or-hostname:port-number/resource
+ipp://ip-address-or-hostname:port-number/resource?option=value
+ipp://ip-address-or-hostname:port-number/resource?option=value&option=value
+
+ +

The ipp backend supports many options, which are summarized in Table 2.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 2: IPP URI Options
OptionDescription
compression=gzipSpecifies that print data should be compressed before sending.
encryption=alwaysSpecifies that the connection to the IPP server should be encrypted using SSL.
encryption=ifrequestedSpecifies that the connection to the IPP server should only be encrypted if the server requests it.
encryption=neverSpecifies that the connection to the IPP server should not be encrypted.
encryption=requiredSpecifies that the connection to the IPP server should be encrypted using TLS.
version=1.0Specifies that version 1.0 of the IPP protocol should be used instead of the default version 1.1.
version=2.0Specifies that version 2.0 of the IPP protocol should be used instead of the default version 1.1.
version=2.1Specifies that version 2.1 of the IPP protocol should be used instead of the default version 1.1.
waitjob=falseSpecifies that the IPP backend should not wait for the job to complete.
waitprinter=falseSpecifies that the IPP backend should not wait for the printer to become idle before sending the print job.
+ +

Line Printer Daemon (LPD) Protocol

+ +

LPD is the original network printing protocol and is supported by many network printers. Due to limitations in the LPD protocol, we do not recommend using it if the printer or server supports one of the other protocols. LPD printing normally happens over port 515 and uses the lpd URI scheme:

+ +
+lpd://ip-address-or-hostname/queue
+lpd://username@ip-address-or-hostname/queue
+lpd://ip-address-or-hostname/queue?option=value
+lpd://username@ip-address-or-hostname/queue?option=value
+lpd://ip-address-or-hostname/queue?option=value&option=value
+lpd://username@ip-address-or-hostname/queue?option=value&option=value
+
+ +

Table 3 summarizes the options supported by the lpd backend.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 3: LPD URI Options
OptionDescription
banner=onSpecifies that a banner page should be printed by the server.
contimeout=secondsSpecifies the number of seconds to wait for the connection to the server to complete.
format=cSpecifies that the print data is a CIF file.
format=dSpecifies that the print data is a DVI file.
format=fSpecifies that the print data is a plain text file.
format=gSpecifies that the print data is a Berkeley plot file.
format=lSpecifies that the print data is a raw (preformatted) print file.
format=nSpecifies that the print data is a ditroff file.
format=oSpecifies that the print data is a PostScript file.
format=pSpecifies that the print data is a plain text file that should be "pretty" printed with a header and footer.
format=rSpecifies that the print data is a FORTRAN carriage control file.
format=tSpecifies that the print data is a troff Graphic Systems C/A/T phototypesetter file.
format=vSpecifies that the print data is a Sun raster file.
order=data,controlSpecifies that the print data files should be sent before the control file.
reserve=noneSpecifies that the backend should not reserve a source port.
reserve=rfc1179Specifies that the backend should reserve a source port from 721 to 731 as required by RFC 1179.
sanitize_title=noSpecifies that the job title string should not be restricted to ASCII characters.
sanitize_title=yesSpecifies that the job title string should be restricted to ASCII characters.
timeout=secondsSpecifies the number of seconds to wait for LPD commands to complete.
+ + +

Common Network Printer URIs

+ +

Once you have set the IP address you can access the printer or print server using the ipp, lpd, or socket backends. Table 1 shows a list of common network interfaces and printer servers and the settings you should use with CUPS:

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 1: Common Device URIs
Model/ManufacturerDevice URI(s)
Apple LaserWriterlpd://address/PASSTHRU
Axis w/o IPP
+ Axis OfficeBasic
+ (see directions)
socket://address:9100
+ socket://address:9101
+ socket://address:9102
Axis w/IPPipp://address/LPT1
+ ipp://address/LPT2
+ ipp://address/COM1
Castelle LANpressTMlpd://address/pr1
+ lpd://address/pr2
+ lpd://address/pr3
DPI NETPrintlpd://address/pr1
+ lpd://address/pr2
+ lpd://address/pr3
DLink DP-301P+socket://address
EFI® Fiery® RIPlpd://address/print
EPSON® Multiprotocol Ethernet Interface Boardsocket://address
Extended System ExtendNETlpd://address/pr1
+ lpd://address/pr2
+ lpd://address/pr3
Hewlett Packard JetDirectsocket://address:9100
+ socket://address:9101
+ socket://address:9102
Intel® NetportExpress XL, PRO/100lpd://address/LPT1_PASSTHRU
+ lpd://address/LPT2_PASSTHRU
+ lpd://address/COM1_PASSTHRU
LexmarkTM MarkNetlpd://address/ps
Linksys EtherFast®
+ (see directions)
socket://address:4010
+ socket://address:4020
+ socket://address:4030
Linksys PSUS4lpd://address/lp
Kodak®lpd://address/ps
Netgear WGPS606lpd://address/L1
+ lpd://address/L2
QMS® CrownNetTMlpd://address/ps
Tektronix® PhaserShareTMsocket://address:9100
XEROX® 4512 NIClpd://address/PORT1
XEROX® XNIClpd://address/PASSTHRU
XEROX® (most others)socket://address:5503
+ + +

Troubleshooting SNMP Discovery Problems

+ +

Whenever you view the administration web page or a list of supported device URIs, the snmp backend will probe the local network(s) using Simple Network Management Protocol (SNMP) broadcasts. Printers that respond to these broadcasts are then interrogated for the make and model and supported protocols, yielding a device URI that can be used to add the printer.

+ +

That said, the SNMP requests sometimes expose problems in vendor SNMP or IPP implementations. If you are experiencing long delays in loading the CUPS web interface administration page, or if you don't see your printer listed, the following instructions will help you to diagnose those problems and/or provide important feedback to the CUPS developers so that we can correct problems and improve the SNMP backend in future releases.

+ +

Quick Fixes

+ +

If you don't use "public" as your community name, create a text file called /etc/cups/snmp.conf and put the following line in it:

+ +
+Community your community name
+
+ +

If you have more than one community name, list them all on separate lines.

+ +

If you don't support SNMP v1 on your network, you are currently "out of luck". That said, we will be adding v2, v2c, and v3 support in future CUPS releases once we have a handle on the actual requirements people have for such things. Please file or update an SNMP enhancement request with specific requirements you have - what you need supported, why you need it supported, and how you would like to see the functionality provided/exposed - so that we can do it "right" the first time.

+ +

Basic Debugging

+ +

The SNMP backend supports a debugging mode that is activated by running it from a shell prompt. If you are using Bash (/bin/bash), Bourne shell (/bin/sh), Korn shell (/bin/ksh), or Z shell (/bin/zsh), you can run the following command to get a verbose log of the SNMP backend:

+ +
+CUPS_DEBUG_LEVEL=2 /usr/lib/cups/backend/snmp 2>&1 | tee snmp.log
+
+ +

For C shell (/bin/csh) and TCsh (/bin/tcsh), use the following command instead:

+ +
+(setenv CUPS_DEBUG_LEVEL 2; /usr/lib/cups/backend/snmp) |& tee snmp.log
+
+ +

On MacOS X you'll find the SNMP backend in /usr/libexec/cups/backend instead:

+ +
+CUPS_DEBUG_LEVEL=2 /usr/libexec/cups/backend/snmp 2>&1 | tee snmp.log
+
+ +

The output will look something like this:

+ +
+ 1  INFO: Using default SNMP Address @LOCAL
+ 2  INFO: Using default SNMP Community public
+ 3  DEBUG: Scanning for devices in "public" via "@LOCAL"...
+ 4  DEBUG: 0.000 Sending 46 bytes to 192.168.2.255...
+ 5  DEBUG: SEQUENCE 44 bytes
+ 6  DEBUG:     INTEGER 1 bytes 0
+ 7  DEBUG:     OCTET STRING 6 bytes "public"
+ 8  DEBUG:     Get-Request-PDU 31 bytes
+ 9  DEBUG:         INTEGER 4 bytes 1149539174
+10  DEBUG:         INTEGER 1 bytes 0
+11  DEBUG:         INTEGER 1 bytes 0
+12  DEBUG:         SEQUENCE 17 bytes
+13  DEBUG:             SEQUENCE 15 bytes
+14  DEBUG:                 OID 11 bytes .1.3.6.1.2.1.25.3.2.1.2.1
+15  DEBUG:                 NULL VALUE 0 bytes
+16  DEBUG: 0.001 Received 55 bytes from 192.168.2.229...
+17  DEBUG: community="public"
+18  DEBUG: request-id=1149539174
+19  DEBUG: error-status=0
+20  DEBUG: SEQUENCE 53 bytes
+21  DEBUG:     INTEGER 1 bytes 0
+22  DEBUG:     OCTET STRING 6 bytes "public"
+23  DEBUG:     Get-Response-PDU 40 bytes
+24  DEBUG:         INTEGER 4 bytes 1149539174
+25  DEBUG:         INTEGER 1 bytes 0
+26  DEBUG:         INTEGER 1 bytes 0
+27  DEBUG:         SEQUENCE 26 bytes
+28  DEBUG:             SEQUENCE 24 bytes
+29  DEBUG:                 OID 11 bytes .1.3.6.1.2.1.25.3.2.1.2.1
+30  DEBUG:                 OID 9 bytes .1.3.6.1.2.1.25.3.1.5
+31  DEBUG: add_cache(addr=0xbfffe170, addrname="192.168.2.229",
+    uri="(null)", id="(null)", make_and_model="(null)")
+32  DEBUG: 0.002 Sending 46 bytes to 192.168.2.229...
+33  DEBUG: SEQUENCE 44 bytes
+34  DEBUG:     INTEGER 1 bytes 0
+35  DEBUG:     OCTET STRING 6 bytes "public"
+36  DEBUG:     Get-Request-PDU 31 bytes
+37  DEBUG:         INTEGER 4 bytes 1149539175
+38  DEBUG:         INTEGER 1 bytes 0
+39  DEBUG:         INTEGER 1 bytes 0
+40  DEBUG:         SEQUENCE 17 bytes
+41  DEBUG:             SEQUENCE 15 bytes
+42  DEBUG:                 OID 11 bytes .1.3.6.1.2.1.25.3.2.1.3.1
+43  DEBUG:                 NULL VALUE 0 bytes
+44  DEBUG: 0.003 Received 69 bytes from 192.168.2.229...
+45  DEBUG: community="public"
+46  DEBUG: request-id=1149539175
+47  DEBUG: error-status=0
+48  DEBUG: SEQUENCE 67 bytes
+49  DEBUG:     INTEGER 1 bytes 0
+50  DEBUG:     OCTET STRING 6 bytes "public"
+51  DEBUG:     Get-Response-PDU 54 bytes
+52  DEBUG:         INTEGER 4 bytes 1149539175
+53  DEBUG:         INTEGER 1 bytes 0
+54  DEBUG:         INTEGER 1 bytes 0
+55  DEBUG:         SEQUENCE 40 bytes
+56  DEBUG:             SEQUENCE 38 bytes
+57  DEBUG:                 OID 11 bytes .1.3.6.1.2.1.25.3.2.1.3.1
+58  DEBUG:                 OCTET STRING 23 bytes "HP LaserJet 4000
+    Series"
+59  DEBUG: 1.001 Probing 192.168.2.229...
+60  DEBUG: 1.001 Trying socket://192.168.2.229:9100...
+61  DEBUG: 192.168.2.229 supports AppSocket!
+62  DEBUG: 1.002 Scan complete!
+63  network socket://192.168.2.229 "HP LaserJet 4000 Series"
+    "HP LaserJet 4000 Series 192.168.2.229" ""
+
+ +

Dissecting the Output

+ +

The first two lines are just informational and let you know that the default community name and address are being used. Lines 3-15 contain the initial SNMP query for the device type OID (.1.3.6.1.2.1.25.3.2.1.2.1) from the Host MIB.

+ +

Lines 16-31 show the response we got from an HP LaserJet 4000 network printer. At this point we discover that it is a printer device and then send another SNMP query (lines 32-43) for the device description OID (.1.3.6.1.2.1.25.3.2.1.3.1) from the Host MIB as well.

+ +

Lines 44-58 show the response to the device description query, which tells us that this is an HP LaserJet 4000 Series printer.

+ +

On line 59 we start our active connection probe and discover that this print server supports the AppSocket (JetDirect) protocol on port 9100.

+ +

Finally, line 63 shows the device information line for the print server that is sent to CUPS.

+ +

Reporting Problems

+ +If you don't see your printer listed, or the wrong information is listed, then you need to gather more information on the printer. The easiest way to do this is to run the snmpwalk command: + +
+snmpwalk -Cc -v 1 -c public ip-address | tee snmpwalk.log
+
+ +

where "ip-address" is the IP address of the printer or print server. You should see a lot of values stream by - the ones you want to see are:

+ +
+HOST-RESOURCES-MIB::hrDeviceType.1 = OID: HOST-RESOURCES-TYPES::hrDevicePrinter
+HOST-RESOURCES-MIB::hrDeviceDescr.1 = STRING: HP LaserJet 4000 Series
+
+ +

The hrDeviceType line should show hrDevicePrinter; if not, then your printer or print server doesn't identify itself as a printer. The hrDeviceDescr line should provide a human-readable string for the make and model of the printer, although in some cases you'll just see something less useful like "Axis OfficeBASIC Parallel Print Server".

+ +

Once you have collected the snmpwalk output, you should go to the CUPS Bugs & Features page to submit a feature request to support your printer or print server. Be sure to attach those two log files you created - they will help us to identify the SNMP values we need to look for.

+ + +

Configuring Print Servers

+ +

Configuring Axis Print Servers

+ +

The Axis print servers can be configured using BOOTP or DHCP. However, on models that do not provide IPP support an additional step must be performed to configure the TCP/IP portion of the print server for use with CUPS.

+ +

Each print server contains a configuration file named config that contains a list of network parameters used by the server. To modify this file you must first download it from the print server using the ftp(1) program:

+ +
+ftp ip-address
+Connected to ip-address.
+220 Axis NPS ### FTP Printer Server V#.## MON DD YEAR ready.
+ftp> user root
+331 User name ok, need password
+Password: pass (this is not echoed)
+230 User logged in
+ftp> get config
+local: config remote: config
+200 PORT command successful.
+150 Opening data connection for config (192,0,2,2),
+(mode ascii).
+226 Transfer complete.
+##### bytes received in #.## seconds (##### Kbytes/s)
+ftp> quit
+221 Goodbye.
+
+ +

Next, edit the file with your favorite text editor and locate the lines beginning with:

+ +
+RTN_OPT.     : YES
+RTEL_PR1.    : 0
+RTEL_PR2.    : 0
+RTEL_PR3.    : 0
+RTEL_PR4.    : 0
+RTEL_PR5.    : 0
+RTEL_PR6.    : 0
+RTEL_PR7.    : 0
+RTEL_PR8.    : 0
+
+ +

Change the RTN_OPT line to read:

+ +
+RTN_OPT.     : NO
+
+ +

This disables the Reverse TELNET protocol and enables the standard TELNET protocol on the print server. Next, assign a port number for each parallel and serial port on the server as follows:

+ +
+RTEL_PR1.    : 9100
+RTEL_PR2.    : 9101
+RTEL_PR3.    : 9102
+RTEL_PR4.    : 9103
+RTEL_PR5.    : 9104
+RTEL_PR6.    : 9105
+RTEL_PR7.    : 9106
+RTEL_PR8.    : 9107
+
+ +

This essentially makes the Axis print server look like a Hewlett Packard JetDirect EX print server. Save the file and then upload the new config file using the ftp command:

+ +
+ftp ip-address
+Connected to ip-address.
+220 Axis NPS ### FTP Printer Server V#.## MON DD YEAR ready.
+ftp> user root
+331 User name ok, need password
+Password: pass (this is not echoed)
+230 User logged in
+ftp> put config CONFIG
+local: config remote: CONFIG
+200 PORT command successful.
+150 Opening data connection for config (192,0,2,2), (mode ascii).
+226 Transfer complete.
+##### bytes received in #.## seconds (##### Kbytes/s)
+ftp> get hardreset
+local: hardreset remote: hardreset
+200 PORT command successful.
+421 Axis NPS ### hard reset, closing connection.
+ftp> quit
+221 Goodbye.
+
+ +

Your Axis print server is now ready for use!

+ +

Configuring Linksys Print Servers

+ +

The Linksys print servers can be configured using BOOTP or DHCP. Like older Axis print servers, an additional step must be performed to configure the TCP/IP portion of the print server for use with CUPS.

+ +

Each print server contains a configuration file named CONFIG that contains a list of network parameters used by the server. To modify this file you must first download it from the print server using the ftp(1) program:

+ +
+ftp -n ip-address
+Connected to ip-address.
+220 Print Server Ready.
+Remote system type is Print.
+ftp> get CONFIG
+local: CONFIG remote: CONFIG
+200 Command OK.
+150 Open ASCII Mode Connection.
+WARNING! 68 bare linefeeds received in ASCII mode
+File may not have transferred correctly.
+226 Transfer complete.
+##### bytes received in #.## seconds (##### Kbytes/s)
+ftp> quit
+221 Goodbye.
+
+ +

Next, edit the file with your favorite text editor and locate the lines beginning with:

+ +
+0100 L1_PROUT:P1
+0120 L2_PROUT:P1
+0140 L3_PROUT:P1
+
+ +

Change the port number for each parallel and serial port on the server as follows:

+ +
+0100 L1_PROUT:P1
+0120 L2_PROUT:P2
+0140 L3_PROUT:P3
+
+ +

This maps each virtual printer with a physical port. Save the file and then upload the new CONFIG file using the ftp command:

+ +
+ftp -n ip-address
+Connected to ip-address.
+220 Print Server Ready.
+Remote system type is Print.
+ftp> put CONFIG
+local: CONFIG remote: CONFIG
+200 Command OK.
+150 Open ASCII Mode Connection.
+226 Transfer complete.
+##### bytes received in #.## seconds (##### Kbytes/s)
+ftp> quit
+221 Goodbye.
+
+ +

Your Linksys print server is now ready for use!

+ + + diff --git a/doc/help/options.html b/doc/help/options.html new file mode 100644 index 0000000000..0cfa741354 --- /dev/null +++ b/doc/help/options.html @@ -0,0 +1,778 @@ + + + + Command-Line Printing and Options + + + + +

Command-Line Printing and Options

+ +

CUPS provides both the System V (lp(1)) and Berkeley (lpr(1)) printing commands for printing +files. In addition, it supported a large number of standard and +printer-specific options that allow you to control how and where +files are printed.

+ + +

Printing Files

+ +

CUPS understands many different types of files directly, +including text, PostScript, PDF, and image files. This allows you +to print from inside your applications or at the command-line, +whichever is most convenient! Type either of the following +commands to print a file to the default (or only) printer on the +system:

+ +
+lp filename
+lpr filename
+
+ +

Choosing a Printer

+ +

Many systems will have more than one printer available to the +user. These printers can be attached to the local system via a +parallel, serial, or USB port, or available over the network. Use +the lpstat(1) command to see a list +of available printers:

+ +
+lpstat -p -d
+
+ +

The -p option specifies that you want to see a +list of printers, and the -d option reports the +current default printer or class.

+ +

Use the -d option with the lp command to +print to a specific printer:

+ +
+lp -d printer filename
+
+ +

or the -P option with the lpr command:

+ +
+lpr -P printer filename
+
+ +

Setting the Default Printer

+ +

If you normally use a particular printer, you can tell CUPS to +use it by default using the lpoptions(1) command:

+ +
+lpoptions -d printer
+
+ +

Printing the Output of a Program

+ +

Both the lp and lpr commands support printing +from the standard input:

+ +
+program | lp
+program | lp -d printer
+program | lpr
+program | lpr -P printer
+
+ +

If the program does not provide any output, then nothing will +be queued for printing.

+ +

Specifying Printer Options

+ +

For many types of files, the default printer options may be +sufficient for your needs. However, there may be times when you +need to change the options for a particular file you are +printing.

+ +

The lp and lpr commands allow you to pass +printer options using the -o option:

+ +
+lp -o landscape -o scaling=75 -o media=A4 filename.jpg
+lpr -o landscape -o scaling=75 -o media=A4 filename.jpg
+
+ +

The available printer options vary depending on the printer. +The standard options are described in the "Standard Printing Options" section +below. Printer-specific options are also available and can be +listed using the lpoptions command:

+ +
+lpoptions -p printer -l
+
+ +

Creating Saved Options

+ +

Saved options are supported in CUPS through printer +instances. Printer instances are, as their name implies, copies +of a printer that have certain options associated with them. Use the +lpoptions command to create a printer instance:

+ +
+lpoptions -p printer/instance -o name=value ...
+
+ +

The -p printer/instance option provides the name of +the instance, which is always the printer name, a slash, and the +instance name which can contain any printable characters except +space and slash. The remaining options are then associated with the +instance instead of the main queue. For example, the following +command creates a duplex instance of the LaserJet queue:

+ +
+lpoptions -p LaserJet/duplex -o sides=two-sided-long-edge
+
+ +

Instances do not inherit lpoptions from the main +queue.

+ +

Printing Multiple Copies

+ +

Both the lp and lpr commands have options for +printing more than one copy of a file:

+ +
+lp -n num-copies filename
+lpr -#num-copies filename
+
+ +

Copies are normally not collated for you. Use the +-o Collate=True option to get collated copies:

+ +
+lp -n num-copies -o Collate=True filename
+lpr -#num-copies -o Collate=True filename
+
+ + +

Canceling a Print Job

+ +

The cancel(1) and lprm(1) commands cancel a print job:

+ +
+cancel job-id
+lprm job-id
+
+ +

The job-id is the number that was reported to you by +the lp command. You can also get the job ID using the lpq(1) or lpstat commands:

+ +
+lpq
+lpstat
+
+ + +

Moving a Print Job

+ +

The lpmove(8) command moves a print +job to a new printer or class:

+ +
+lpmove job-id destination
+
+ +

The job-id is the number that was reported to you by +the lp or lpstat commands. Destination is the +name of a printer or class that you want to actually print the job. + +

Note: + +

The lpmove command is located in the system command +directory (typically /usr/sbin or /usr/local/sbin), +and so may not be in your command path. Specify the full path to the +command if you get a "command not found" error, for example: + +

+/usr/sbin/lpmove foo-123 bar
+
+ +
+ + +

Standard Printing Options

+ +

The following options apply when printing all types of +files.

+ +

Selecting the Media Size, Type, and Source

+ +

The -o media=xyz option sets the media size, +type, and/or source:

+ +
+lp -o media=Letter filename
+lp -o media=Letter,MultiPurpose filename
+lpr -o media=Letter,Transparency filename
+lpr -o media=Letter,MultiPurpose,Transparency filename
+
+ +

The available media sizes, types, and sources depend on the +printer, but most support the following options (case is not +significant):

+ +
    + +
  • Letter - US Letter (8.5x11 inches, or 216x279mm) + +
  • Legal - US Legal (8.5x14 inches, or 216x356mm) + +
  • A4 - ISO A4 (8.27x11.69 inches, or 210x297mm) + +
  • COM10 - US #10 Envelope (9.5x4.125 inches, or + 241x105mm) + +
  • DL - ISO DL Envelope (8.66x4.33 inches, or 220x110mm) + +
  • Transparency - Transparency media type or source + +
  • Upper - Upper paper tray + +
  • Lower - Lower paper tray + +
  • MultiPurpose - Multi-purpose paper tray + +
  • LargeCapacity - Large capacity paper tray + +
+ +

The actual options supported are defined in the printer's PPD +file in the PageSize, InputSlot, and +MediaType options. You can list them using the +lpoptions(1) command:

+ +
+lpoptions -p printer -l
+
+ +

When Custom is listed for the PageSize option, you can specify custom media sizes using one of the following forms:

+ +
+lp -o media=Custom.WIDTHxLENGTH filename
+lp -o media=Custom.WIDTHxLENGTHin filename
+lp -o media=Custom.WIDTHxLENGTHcm filename
+lp -o media=Custom.WIDTHxLENGTHmm filename
+
+ +

where "WIDTH" and "LENGTH" are the width and length of the media in points, inches, centimeters, or millimeters, respectively.

+ + +

Setting the Orientation

+ +

The -o landscape option will rotate the page 90 +degrees to print in landscape orientation:

+ +
+lp -o landscape filename
+lpr -o landscape filename
+
+ +

The -o orientation-requested=N option rotates the +page depending on the value of N:

+ +
    + +
  • -o orientation-requested=3 - portrait + orientation (no rotation)
  • + +
  • -o orientation-requested=4 - landscape + orientation (90 degrees)
  • + +
  • -o orientation-requested=5 - reverse + landscape or seascape orientation (270 degrees)
  • + +
  • -o orientation-requested=6 - reverse + portrait or upside-down orientation (180 degrees)
  • + +
+ + +

Printing On Both Sides of the Paper

+ +

The -o sides=two-sided-short-edge and -o +sides=two-sided-long-edge options will enable two-sided +printing on the printer if the printer supports it. The -o +sides=two-sided-short-edge option is suitable for +landscape pages, while the -o +sides=two-sided-long-edge option is suitable for portrait +pages:

+ +
+lp -o sides=two-sided-short-edge filename
+lp -o sides=two-sided-long-edge filename
+lpr -o sides=two-sided-long-edge filename
+
+ +

The default is to print single-sided:

+ +
+lp -o sides=one-sided filename
+lpr -o sides=one-sided filename
+
+ + +

Selecting the Banner Page(s)

+ +

The -o job-sheets=start,end option sets the banner +page(s) to use for a job:

+ +
+lp -o job-sheets=none filename
+lp -o job-sheets=standard filename
+lpr -o job-sheets=classified,classified filename
+
+ +

If only one banner file is specified, it will be printed +before the files in the job. If a second banner file is +specified, it is printed after the files in the job.

+ +

The available banner pages depend on the local system +configuration; CUPS includes the following banner files:

+ +
    + +
  • none - Do not produce a banner page. + +
  • classified - A banner page with a "classified" + label at the top and bottom. + +
  • confidential - A banner page with a + "confidential" label at the top and bottom. + +
  • secret - A banner page with a "secret" label + at the top and bottom. + +
  • standard - A banner page with no label at the + top and bottom. + +
  • topsecret - A banner page with a "top secret" + label at the top and bottom. + +
  • unclassified - A banner page with an + "unclassified" label at the top and bottom. + +
+ + +

Holding Jobs for Later Printing

+ +

The -o job-hold-until=when option tells CUPS to +delay printing until the "when" time, which can be one of the +following:

+ +
    + +
  • -o job-hold-until=indefinite; print only + after released by the user or an administrator
  • + +
  • -o job-hold-until=day-time; print from + 6am to 6pm local time
  • + +
  • -o job-hold-until=night; print from + 6pm to 6am local time
  • + +
  • -o job-hold-until=second-shift; print from + 4pm to 12am local time
  • + +
  • -o job-hold-until=third-shift; print from + 12am to 8am local time
  • + +
  • -o job-hold-until=weekend; print on Saturday + or Sunday
  • + +
  • -o job-hold-until=HH:MM; print at the specified + UTC time
  • + +
+ +

Releasing Held Jobs

+ +

Aside from the web interface, you can use the lp command +to release a held job:

+ +
+lp -i job-id -H resume
+
+ +

where "job-id" is the job ID reported by the lpstat +command.

+ + +

Setting the Job Priority

+ +

The -o job-priority=NNN option tells CUPS to +assign a priority to your job from 1 (lowest) to 100 (highest), +which influences where the job appears in the print queue. Higher +priority jobs are printed before lower priority jobs, however +submitting a new job with a high priority will not interrupt an +already printing job.

+ + +

Specifying the Output Order

+ +

The -o outputorder=normal and -o outputorder=reverse options specify the order of the pages. Normal order prints page 1 first, page 2 second, and so forth. Reverse order prints page 1 last.

+ + +

Selecting a Range of Pages

+ +

The -o page-ranges=pages option selects a range +of pages for printing:

+ +
+lp -o page-ranges=1 filename
+lp -o page-ranges=1-4 filename
+lp -o page-ranges=1-4,7,9-12 filename
+lpr -o page-ranges=1-4,7,9-12 filename
+
+ +

As shown above, the pages value can be a single page, a +range of pages, or a collection of page numbers and ranges separated by +commas. The pages will always be printed in ascending order, regardless +of the order of the pages in the page-ranges option. + +

The default is to print all pages. + +

Note: + +

The page numbers used by page-ranges refer to the output +pages and not the document's page numbers. Options like number-up +can make the output page numbering not match the document page numbers.

+ +
+ + +

Selecting Even or Odd Pages

+ +

Use the -o page-set=set option to select the even or odd pages:

+ +
+lp -o page-set=odd filename
+lp -o page-set=even filename
+lpr -o page-set=even filename
+
+ +

The default is to print all pages. + +

N-Up Printing

+ +

The -o number-up=value option selects N-Up +printing. N-Up printing places multiple document pages on a +single printed page. CUPS supports 1, 2, 4, 6, 9, and 16-Up +formats; the default format is 1-Up:

+ +
+lp -o number-up=1 filename
+lp -o number-up=2 filename
+lp -o number-up=4 filename
+lpr -o number-up=16 filename
+
+ +

The -o page-border=value option chooses the +border to draw around each page:

+ +
    +
  • -o page-border=double; draw two hairline borders around each page
  • +
  • -o page-border=double-thick; draw two 1pt borders around each page
  • +
  • -o page-border=none; do not draw a border (default)
  • +
  • -o page-border=single; draw one hairline border around each page
  • +
  • -o page-border=single-thick; draw one 1pt border around each page
  • +
+ +

The -o number-up-layout=value option chooses the +layout of the pages on each output page:

+ +
    +
  • -o number-up-layout=btlr; Bottom to top, left to right
  • +
  • -o number-up-layout=btrl; Bottom to top, right to left
  • +
  • -o number-up-layout=lrbt; Left to right, bottom to top
  • +
  • -o number-up-layout=lrtb; Left to right, top to bottom (default)
  • +
  • -o number-up-layout=rlbt; Right to left, bottom to top
  • +
  • -o number-up-layout=rltb; Right to left, top to bottom
  • +
  • -o number-up-layout=tblr; Top to bottom, left to right
  • +
  • -o number-up-layout=tbrl; Top to bottom, right to left
  • +
+ +

Scaling to Fit

+ +

The -o fit-to-page option specifies that the document +should be scaled to fit on the page:

+ +
+lp -o fit-to-page filename
+lpr -o fit-to-page filename
+
+ +

The default is to use the size specified in the file.

+ +
Note: + +

This feature depends upon an accurate size in +the print file. If no size is given in the file, the page may be +scaled incorrectly! + +

+ +

Printing in Reverse Order

+ +

The -o outputorder=reverse option will print the +pages in reverse order:

+ +
+lp -o outputorder=reverse filename
+lpr -o outputorder=reverse filename
+
+ +

Similarly, the -o outputorder=normal option will +print starting with page 1:

+ +
+lp -o outputorder=normal filename
+lpr -o outputorder=normal filename
+
+ +

The default is -o outputorder=normal for +printers that print face down and -o outputorder=reverse +for printers that print face up. + +

Printing Mirrored Pages

+ +

The -o mirror option flips each page along the +vertical axis to produce a mirrored image:

+ +
+lp -o mirror filename
+lpr -o mirror filename
+
+ +

This is typically used when printing on T-shirt transfer +media or sometimes on transparencies.

+ +

Raw or Unfiltered Output

+ +

The -o raw option allows you to send files +directly to a printer without filtering. This is sometimes +required when printing from applications that provide their own +"printer drivers" for your printer:

+ +
+lp -o raw filename
+lpr -o raw filename
+
+ +

The -l option can also be used with the +lpr command to send files directly to a printer:

+ +
+lpr -l filename
+
+ + +

Text Options

+ +

CUPS supports several options that are only used when printing +plain text files. These options have absolutely no effect on +PostScript, PDF, HP-GL/2, or image files.

+ +

Setting the Number of Characters Per Inch

+ +

The -o cpi=value option sets the number of +characters per inch:

+ +
+lp -o cpi=10 filename
+lp -o cpi=12 filename
+lpr -o cpi=17 filename
+
+ +

The default characters per inch is 10.

+ +

Setting the Number of Lines Per Inch

+ +

The -o lpi=value option sets the number of lines +per inch:

+ +
+lp -o lpi=6 filename
+lpr -o lpi=8 filename
+
+ +

The default lines per inch is 6.

+ +

Setting the Number of Columns

+ +

The -o columns=value option sets the number of +text columns:

+ +
+lp -o columns=2 filename
+lpr -o columns=3 filename
+
+ +

The default number of columns is 1.

+ +

Setting the Page Margins

+ +

Normally the page margins are set to the hard limits of the +printer. Use the -o page-left=value, -o +page-right=value, -o page-top=value, and +-o page-bottom=value options to adjust the page +margins:

+ +
+lp -o page-left=value filename
+lp -o page-right=value filename
+lp -o page-top=value filename
+lp -o page-bottom=value filename
+lpr -o page-left=value -o page-right=value -o page-top=value -o page-bottom=value filename
+
+ +

The value argument is the margin in points; each +point is 1/72 inch or 0.35mm.

+ +

Pretty Printing

+ +

The -o prettyprint option puts a header at the +top of each page with the page number, job title (usually the +filename), and the date. Also, C and C++ keywords are +highlighted, and comment lines are italicized:

+ +
+lp -o prettyprint filename
+lpr -o prettyprint filename
+
+ +

Turning Off Text Wrapping

+ +

The -o nowrap option disables wrapping of long lines:

+ +
+lp -o nowrap filename
+lpr -o nowrap filename
+
+ + +

Not Supported on Mac OS XImage Options

+ +

CUPS supports several options that are only used when printing +image files. These options have absolutely no effect on PostScript, PDF, +HP-GL/2, or text files.

+ +

Positioning Images

+ +

The -o position=name option specifies the position of the +image on the page: + +

    + +
  • center - Center the image on the page (default) + +
  • top - Print the image centered at the top of the page + +
  • left - Print the image centered on the left of page + +
  • right - Print the image centered on the right of the page + +
  • top-left - Print the image at the top left corner of + the page + +
  • top-right - Print the image at the top right corner of + the page + +
  • bottom - Print the image centered at the bottom of + the page + +
  • bottom-left - Print the image at the bottom left + corner of the page + +
  • bottom-right - Print the image at the bottom right + corner of the page + +
+ +

Scaling Images

+ +

The -o scaling=percent, -o +ppi=value, and -o natural-scaling=percent +options change the size of a printed image: + +

+lp -o scaling=percent filename 
+lp -o ppi=value filename
+lpr -o natural-scaling=percent filename
+
+ +

The scaling=percent value is a number from 1 to 800 +specifying the size in relation to the page (not the image.) A +scaling of 100 percent will fill the page as completely as the image +aspect ratio allows. A scaling of 200 percent will print on up to 4 +pages. + +

The ppi=value value is a number from 1 to 1200 specifying the +resolution of the image in pixels per inch. An image that is 3000x2400 +pixels will print 10x8 inches at 300 pixels per inch, for example. If +the specified resolution makes the image larger than the page, multiple +pages will be printed to satisfy the request. + +

The natural-scaling=percent value is a number +from 1 to 800 specifying the size in relation to the natural +image size. A scaling of 100 percent will print the image at its +natural size, while a scaling of 50 percent will print the image +at half its natural size. If the specified scaling makes the +image larger than the page, multiple pages will be printed to +satisfy the request. + + +

HP-GL/2 Options

+ +

CUPS supports several options that are only used when printing +HP-GL/2 files. These options have absolutely no effect on PostScript, PDF, +image, or text files.

+ +

Printing in Black

+ +

The -o blackplot option specifies that all pens should +plot in black:

+ +
+lp -o blackplot filename
+lpr -o blackplot filename
+
+ +

The default is to use the colors defined in the plot file or the +standard pen colors defined in the HP-GL/2 reference manual from +Hewlett Packard. + +

Setting the Default Pen Width

+ +

The -o penwidth=value option specifies the default pen +width for HP-GL/2 files:

+ +
+lp -o penwidth=value filename
+lpr -o penwidth=value filename
+
+ +

The pen width value specifies the pen width in micrometers. +The default value of 1000 produces lines that are 1 millimeter in width. +Specifying a pen width of 0 produces lines that are exactly 1 pixel wide.

+ +
Note: + +

This option is ignored when the pen widths are set in the plot +file. + +

+ + + diff --git a/doc/help/overview.html b/doc/help/overview.html new file mode 100644 index 0000000000..59a2ac0677 --- /dev/null +++ b/doc/help/overview.html @@ -0,0 +1,70 @@ + + + + Overview of CUPS + + + + +

Overview of CUPS

+ +

CUPS is the software you use to print from applications like the web browser +you are using to read this page. It converts the page descriptions +produced by your application (put a paragraph here, draw a line there, and so +forth) into something your printer can understand and then sends the information +to the printer for printing.

+ +

Now, since every printer manufacturer does things differently, printing can +be very complicated. CUPS does its best to hide this from you and your +application so that you can concentrate on printing and less on how to +print. Generally, the only time you need to know anything about your printer is +when you use it for the first time, and even then CUPS can often figure things +out on its own.

+ +

How Does It Work?

+ +

The first time you print to a printer, CUPS creates a queue to keep +track of the current status of the printer (everything OK, out of paper, etc.) +and any pages you have printed. Most of the time the queue points to a printer +connected directly to your computer via a USB or parallel port, however it can +also point to a printer on your network, a printer on the Internet, or multiple +printers depending on the configuration. Regardless of where the queue +points, it will look like any other printer to you and your applications.

+ +

Every time you print something, CUPS creates a job which contains +the queue you are sending the print to, the name of the document you are +printing, and the page descriptions. Job are numbered (queue-1, queue-2, and so +forth) so you can monitor the job as it is printed or cancel it if you see a +mistake. When CUPS gets a job for printing, it determines the best programs +(filters, printer drivers, port monitors, and +backends) to convert the pages into a printable format and then runs +them to actually print the job.

+ +

When the print job is completely printed, CUPS removes the job from the queue +and moves on to any other jobs you have submitted. You can also be notified when +the job is finished, or if there are any errors during printing, in several +different ways.

+ +

Where Do I Begin?

+ +

Click on the Administration tab at +the top of this page. Click on the Add Printer button and follow the +prompts.

+ +
When you are asked for a username and password, enter your login +username and password or the "root" username and password. On MacOS X, the login +username (or "short name") is typically your first and last name in +lowercase.
+ +

After the printer is added, CUPS will ask you to set the default printer +options (paper size, output mode, etc.) for the printer. Make any changes as +needed and then click on the Set Default Options button to save +them. Some printers also support auto-configuration - click on the Query +Printer for Default Options button to update the options automatically.

+ +

Once you have added the printer, you can print to it from any application. +You can also choose Print Test Page from the maintenance menu to print +a simple test page and verify that everything is working properly.

+ + + diff --git a/doc/help/policies.html b/doc/help/policies.html new file mode 100644 index 0000000000..14a5b4b870 --- /dev/null +++ b/doc/help/policies.html @@ -0,0 +1,601 @@ + + + + Managing Operation Policies + + + + +

Managing Operation Policies

+ +

Operation policies are the rules used for each IPP operation in CUPS. These rules include things like "user must provide a password", "user must be in the system group", "allow only from the local system", and so forth. Until CUPS 1.2, these rules were largely hardcoded and could only be customized at a very basic level.

+ +

CUPS 1.2 and later provides a fine-grained policy layer which allows you to completely redefine the rules for each operation and/or printer. Each policy is named and defines access control rules for each IPP operation. This document describes how to manage policies and their rules.

+ +

The Basics

+ +

Operation policies are used for all IPP requests sent to the scheduler and are evaluated after the Location based access control rules. This means that operation policies can only add additional security restrictions to a request, never relax them. Use Location based access control rules for server-wide limits and operation policies for limits on individual printers, tasks, or services.

+ +

Policies are stored in the cupsd.conf file in Policy sections. Each policy has an alphanumeric name that is used to select it. Inside the policy section are one or more Limit subsections which list the operations that are affected by the rules inside it. Listing 1 shows the default operation policy, appropriately called "default", that is shipped with CUPS.

+ +

The easiest way to add a policy to the cupsd.conf file is to use the web interface. Click on the Administration tab and then the Edit Configuration File button to edit the current cupsd.conf file. Click on the Save Changes button to save the changes and restart the scheduler. If you edit the cupsd.conf file from the console, make sure to restart the cupsd process before trying to use the new policy.

+ +
+Listing 1: Default Operation Policy
+
+ 1    <Policy default>
+ 2      # Job-related operations must be done by the owner or an
+      administrator...
+ 3      <Limit Send-Document Send-URI Hold-Job Release-Job
+      Restart-Job Purge-Jobs Set-Job-Attributes
+      Create-Job-Subscription Renew-Subscription
+      Cancel-Subscription Get-Notifications Reprocess-Job
+      Cancel-Current-Job Suspend-Current-Job Resume-Job
+      CUPS-Move-Job CUPS-Get-Document>
+ 4        Require user @OWNER @SYSTEM
+ 5        Order deny,allow
+ 6      </Limit>
+ 7    
+ 8      # All administration operations require an administrator
+      to authenticate...
+ 9      <Limit CUPS-Add-Printer CUPS-Delete-Printer CUPS-Add-Class
+      CUPS-Delete-Class CUPS-Set-Default>
+10        AuthType Default
+11        Require user @SYSTEM
+12        Order deny,allow
+13      </Limit>
+14    
+15      # All printer operations require a printer operator
+      to authenticate...
+16      <Limit Pause-Printer Resume-Printer
+      Set-Printer-Attributes Enable-Printer Disable-Printer
+      Pause-Printer-After-Current-Job Hold-New-Jobs
+      Release-Held-New-Jobs Deactivate-Printer Activate-Printer
+      Restart-Printer Shutdown-Printer Startup-Printer
+      Promote-Job Schedule-Job-After CUPS-Accept-Jobs
+      CUPS-Reject-Jobs>
+17        AuthType Default
+18        Require user varies by OS
+19        Order deny,allow
+20      </Limit>
+21    
+22      # Only the owner or an administrator can cancel or
+      authenticate a job...
+23      <Limit Cancel-Job CUPS-Authenticate-Job>
+24        Require user @OWNER @SYSTEM
+25        Order deny,allow
+26      </Limit>
+27    
+28      <Limit All>
+29        Order deny,allow
+30      </Limit>
+31    </Policy>
+
+ +

The Default CUPS Operation Policy

+ +

The policy definition starts with an opening Policy directive:

+ +
+ 1    <Policy default>
+
+ +

The first Limit subsection defines the rules for IPP job operations:

+ +
+ 3      <Limit Send-Document Send-URI Hold-Job Release-Job
+      Restart-Job Purge-Jobs Set-Job-Attributes
+      Create-Job-Subscription Renew-Subscription
+      Cancel-Subscription Get-Notifications Reprocess-Job
+      Cancel-Current-Job Suspend-Current-Job Resume-Job
+      CUPS-Move-Job CUPS-Get-Document>
+ 4        Require user @OWNER @SYSTEM
+ 5        Order deny,allow
+ 6      </Limit>
+
+ +

The operation names are listed on a single line with spaces separating them. Each name corresponds to the IPP operation described in any of the IETF or PWG standards documents for the Internet Printing Protocol. Table 1 lists all of the operations that have been defined along with their usage in CUPS.

+ +

The access control rules are listed after the Limit line and are the same as those used for Location sections. In this case, we require the owner of the job ("@OWNER") or a member of the SystemGroup ("@SYSTEM") to do the operation. Because we do not include an AuthType directive here, the user information can come from the IPP request itself or the authenticated username from the HTTP request. The administrative operations starting on line 9, however, do use the AuthType directive, and so administrative operations need to be authenticated:

+ +
+ 9      <Limit CUPS-Add-Printer CUPS-Delete-Printer CUPS-Add-Class
+      CUPS-Delete-Class CUPS-Set-Default>
+10        AuthType Default
+11        Require user @SYSTEM
+12        Order deny,allow
+13      </Limit>
+14    
+15      # All printer operations require a printer operator
+      to authenticate...
+16      <Limit Pause-Printer Resume-Printer
+      Set-Printer-Attributes Enable-Printer Disable-Printer
+      Pause-Printer-After-Current-Job Hold-New-Jobs
+      Release-Held-New-Jobs Deactivate-Printer Activate-Printer
+      Restart-Printer Shutdown-Printer Startup-Printer
+      Promote-Job Schedule-Job-After CUPS-Accept-Jobs
+      CUPS-Reject-Jobs>
+17        AuthType Default
+18        Require user varies by OS
+19        Order deny,allow
+20      </Limit>
+
+ +

The "Order deny,allow" line at the end of both Limit subsections allows the request to come from any system allowed by the Location sections elsewhere in the cupsd.conf file.

+ +

The Cancel-Job and CUPS-Authenticate-Job operations are listed separately to allow the web interface to more easily edit their policy without disturbing the rest. Like the rest of the job operations, we want the job's owner ("@OWNER") or an administrator ("@SYSTEM") to do it:

+ +
+16      <Limit Cancel-Job CUPS-Authenticate-Job>
+17        Require user @OWNER @SYSTEM
+18        Order deny,allow
+19      </Limit>
+
+ +

The last Limit subsection in any policy uses the special operation name All. CUPS will use the rules in this subsection for any operation you don't list specifically in the policy. In this case, all other operations are allowed without a username or authentication:

+ +
+21      <Limit All>
+22        Order deny,allow
+23      </Limit>
+24    </Policy>
+
+ + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 1: IPP Operation Names
NameUsed by CUPS?Description
Print-JobYesCreates a print job with a single file.
Print-URINoCreate a print job with a single URI.
Validate-JobYesValidates a print request before printing.
Create-JobYesCreates a print job with no files or URIs.
Send-DocumentYesAdds a file to a print job.
Send-URINoAdds a URI to a print job.
Cancel-JobYesCancels a print job.
Get-Job-AttributesYesGets information and options associated with a job.
Get-JobsYesGets a list of jobs.
Get-Printer-AttributesYesGets information and options associated with a printer or class.
Hold-JobYesHolds a print job for printing.
Release-JobYesReleases a print job for printing.
Restart-JobYesReprints a print job.
Pause-PrinterYesStops a printer or class.
Resume-PrinterYesStarts a printer or class.
Purge-JobsYesCancels all jobs on the server or a printer or class + and removes the job history information.
Set-Printer-AttributesNoSets printer or class information; CUPS uses + CUPS-Add-Modify-Printer and CUPS-Add-Modify-Class + instead.
Set-Job-AttributesYesChanges job options.
Get-Printer-Supported-ValuesNoGets -supported attributes for a printer based on job + options.
Create-Printer-SubscriptionYesCreates an event subscription for a printer or the server.
Create-Job-SubscriptionYesCreates an event subscription for a job.
Get-Subscription-AttributesYesGets information for an event subscription.
Get-SubscriptionsYesGets a list of event subscriptions.
Renew-SubscriptionYesRenews an event subscription that is about to expire.
Cancel-SubscriptionYesCancels an event subscription.
Get-NotificationsYesGets (pending) events for an event subscription.
Send-NotificationsNoSends events for an event subscription.
Get-Printer-Support-FilesNoGets printer driver files for a Novell client.
Enable-PrinterYesStarts a printer or class.
Disable-PrinterYesStops a printer or class.
Pause-Printer-After-Current-JobNoStops a printer or class after the current job is finished.
Hold-New-JobsNoHolds new jobs submitted to a printer or class.
Release-Held-New-JobsNoReleases jobs that were held because of the + Hold-New-Jobs operation.
Deactivate-PrinterNoDeactivates a printer or class.
Activate-PrinterNoActivates a printer or class.
Restart-PrinterNoRestarts a printer or class, resuming print jobs as needed.
Shutdown-PrinterNoPowers a printer or class off.
Startup-PrinterNoPowers a printer or class on.
Reprocess-JobNoReprints a job on a different printer or class; CUPS has the + CUPS-Move-Job operation instead.
Cancel-Current-JobNoCancels the current job on a printer or class.
Suspend-Current-JobNoStops the current job on a printer or class.
Resume-JobNoResumes printing of a stopped job.
Promote-JobNoPrints a job before others.
Schedule-Job-AfterNoPrints a job after others.
CUPS-Get-DefaultYesGets the server/network default printer or class.
CUPS-Get-PrintersYesGets a list of printers and/or classes.
CUPS-Add-Modify-PrinterYesAdds or modifies a printer.
CUPS-Delete-PrinterYesRemoves a printer.
CUPS-Get-ClassesYesGets a list of classes.
CUPS-Add-Modify-ClassYesAdds or modifies a class.
CUPS-Delete-ClassYesRemoves a class.
CUPS-Accept-JobsYesSets a printer's or class' printer-is-accepting-jobs + attribute to true.
CUPS-Reject-JobsYesSets a printer's or class' printer-is-accepting-jobs + attribute to false.
CUPS-Set-DefaultYesSets the server/network default printer or class.
CUPS-Get-DevicesYesGets a list of printer devices.
CUPS-Get-PPDsYesGets a list of printer drivers or manufacturers.
CUPS-Move-JobYesMoves a job to a different printer or class.
CUPS-Authenticate-JobYesAuthenticates a job for printing.
CUPS-Get-DocumentYesRetrieves a document file from a job.
+ + +

Creating Your Own Policies

+ +

The easiest way to create a new policy is to start with the default policy and then make changes to the copy. The first change you'll make is to give the policy a new name. Policy names can use the same characters as a printer name, specifically all printable characters except space, slash (/), and pound (#):

+ +
+<Policy mypolicy>
+
+ +

Then you need to decide exactly what limits you want for the policy. For example, if you want to allow any user to cancel any other users' jobs, you can change the Cancel-Job limits to:

+ +
+<Limit Cancel-Job>
+  Order deny,allow
+</Limit>
+
+ +

The directives inside the Limit subsection can use any of the normal limiting directives: Allow, AuthType, Deny, Encryption, Require, and Satisfy. Table 2 lists some basic "recipes" for different access control rules.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 2: Access Control Recipes
Access LevelDirectives to Use
Allow Everyone
Order deny,allow
+Allow from all
Allow Everyone on the Local Network
Order deny,allow
+Allow from @LOCAL
Deny Everyone/Disable Operation(s)
Order deny,allow
Require Login (System) Password
AuthType Basic
Require CUPS (lppasswd) Password
AuthType BasicDigest
Require Kerberos
AuthType Negotiate
Require the Owner of a Job or Subscription
Require user @OWNER
Require an Administrative User
Require user @SYSTEM
Require Member of Group "foogroup"
Require user @foogroup
Require "john" or "mary"
Require user john mary
Require Encryption
Encryption Required
+ + +

Creating a Policy for a Computer Lab

+ +

One common operating scenario is a computer lab. The lab is managed by one or more technicians that assist the users of the lab and handle the basic administration tasks. Listing 2 shows an operation policy that only allows access from the lab's subnet, 10.0.2.x, and allows the lab technicians, who are members of a special UNIX group for that lab called "lab999", to do job, printer, and subscription management operations.

+ +
+Listing 2: Operation Policy for a Lab
+
+ 1    <Policy lab999>
+ 2      # Job- and subscription-related operations must be done
+      by the owner, a lab technician, or an administrator...
+ 3      <Limit Send-Document Send-URI Hold-Job Release-Job
+      Restart-Job Purge-Jobs Set-Job-Attributes
+      Create-Job-Subscription Renew-Subscription
+      Cancel-Subscription Get-Notifications Reprocess-Job
+      Cancel-Current-Job Suspend-Current-Job Resume-Job
+      CUPS-Move-Job Cancel-Job CUPS-Authenticate-Job CUPS-Get-Document>
+ 4        Require user @OWNER @lab999 @SYSTEM
+ 5        Order allow,deny
+ 6        Allow from 10.0.2.0/24
+ 7      </Limit>
+ 8    
+ 9      # All administration operations require a lab technician
+      or an administrator to authenticate...
+10      <Limit Pause-Printer Resume-Printer
+      Set-Printer-Attributes Enable-Printer Disable-Printer
+      Pause-Printer-After-Current-Job Hold-New-Jobs
+      Release-Held-New-Jobs Deactivate-Printer Activate-Printer
+      Restart-Printer Shutdown-Printer Startup-Printer
+      Promote-Job Schedule-Job-After CUPS-Accept-Jobs
+      CUPS-Reject-Jobs CUPS-Set-Default>
+11        AuthType Default
+12        Require user @lab999 @SYSTEM
+13        Order allow,deny
+14        Allow from 10.0.2.0/24
+15      </Limit>
+16
+17      # All other operations are allowed from the lab network...
+18      <Limit All>
+19        Order allow,deny
+20        Allow from 10.0.2.0/24
+21      </Limit>
+22    </Policy>
+
+ + +

Using Policies

+ +

Once you have created a policy, you can use it in two ways. The first way is to assign it as the default policy for the system using the DefaultPolicy directive in the cupsd.conf file. For example, add the following line to the cupsd.conf file to use the "lab999" policy from the previous section:

+ +
+DefaultPolicy lab999
+
+ +

To associate the policy with one or more printers, use either the lpadmin(8) command or the web interface to change the operation policy for each printer. When using the lpadmin command, the -o printer-op-policy=name option sets the operation policy for a printer. For example, enter the following command to use the "lab999" policy from the previous section with a printer named "LaserJet4000":

+ +
+lpadmin -p LaserJet4000 -o printer-op-policy=lab999
+
+ +

To make the same change in the web interface, go to the printer's web page, for example "http://localhost:631/printers/LaserJet4000", and choose Set Default Options from the Administration menu button. Click on the Policies link and choose the desired policy from the pull-down list. Click on Set Default Options to change the policy for the printer.

+ + + diff --git a/doc/help/postscript-driver.html b/doc/help/postscript-driver.html new file mode 100644 index 0000000000..501043c78f --- /dev/null +++ b/doc/help/postscript-driver.html @@ -0,0 +1,663 @@ + + + + + Developing PostScript Printer Drivers + + + + + + +
+ + +

Developing PostScript Printer Drivers

+ +

This document describes how to develop printer drivers for PostScript printers. Topics include: printer driver basics, creating new PPD files, importing existing PPD files, using custom filters, implementing color management, and adding Mac OS X features.

+ +
+ + + + + + +
See AlsoProgramming: Developing Raster Printer Drivers
+ Programming: Filter and Backend Programming
+ Programming: Introduction to the PPD Compiler
+ Programming: Raster API
+ References: PPD Compiler Driver Information File Reference
+ Specifications: CUPS PPD Extensions
+

Contents

+ +

Printer Driver Basics

+ +

A CUPS PostScript printer driver consists of a PostScript Printer Description (PPD) file that describes the features and capabilities of the device, zero or more filter programs that prepare print data for the device, and zero or more support files for color management, online help, and so forth. The PPD file includes references to all of the filters and support files used by the driver.

+ +

Every time a user prints something the scheduler program, cupsd(8), determines the format of the print job and the programs required to convert that job into something the printer understands. CUPS includes filter programs for many common formats, for example to convert Portable Document Format (PDF) files into device-independent PostScript, and then from device-independent PostScript to device-dependent PostScript. Figure 1 shows the data flow of a typical print job.

+ +
+ + +
Figure 1: PostScript Filter Chain
PostScript Filter Chain
+ +

The optional PostScript filter can be provided to add printer-specific commands to the PostScript output that cannot be represented in the PPD file or to reorganize the output for special printer features. Typically this is used to support advanced job management or finishing functions on the printer. CUPS includes a generic PostScript filter that handles all PPD-defined commands.

+ +

The optional port monitor handles interface-specific protocol or encoding issues. For example, many PostScript printers support the Binary Communications Protocol (BCP) and Tagged Binary Communications Protocol (TBCP) to allow applications to print 8-bit ("binary") PostScript jobs. CUPS includes port monitors for BCP and TBCP, and you can supply your own port monitors as needed.

+ +

The backend handles communications with the printer, sending print data from the last filter to the printer and relaying back-channel data from the printer to the upstream filters. CUPS includes backend programs for common direct-connect interfaces and network protocols, and you can provide your own backend to support custom interfaces and protocols.

+ +

The scheduler also supports a special "command" file format for sending maintenance commands and status queries to a printer or printer driver. Command print jobs typically use a single command filter program defined in the PPD file to generate the appropriate printer commands and handle any responses from the printer. Figure 2 shows the data flow of a typical command job.

+ +
+ + +
Figure 2: Command Filter Chain
Command Filter Chain
+ +

PostScript printer drivers typically do not require their own command filter since CUPS includes a generic PostScript command filter that supports all of the standard functions using PPD-defined commands.

+ + +

Creating New PPD Files

+ +

We recommend using the CUPS PPD compiler, ppdc(1), to create new PPD files since it manages many of the tedious (and error-prone!) details of paper sizes and localization for you. It also allows you to easily support multiple devices from a single source file. For more information see the "Introduction to the PPD Compiler" document. Listing 1 shows a driver information file for a black-and-white PostScript printer.

+ +

Listing 1: "examples/postscript.drv"

+ +
+// Include standard font and media definitions
+#include <font.defs>
+#include <media.defs>
+
+// Specify this is a PostScript printer driver
+DriverType ps
+
+// List the fonts that are supported, in this case all standard fonts
+Font *
+
+// Manufacturer, model name, and version
+Manufacturer "Foo"
+ModelName "Foo LaserProofer 2000"
+Version 1.0
+
+// PostScript printer attributes
+Attribute DefaultColorSpace "" Gray
+Attribute LandscapeOrientation "" Minus90
+Attribute LanguageLevel "" "3"
+Attribute Product "" "(Foo LaserProofer 2000)"
+Attribute PSVersion "" "(3010) 0"
+Attribute TTRasterizer "" Type42
+
+// Supported page sizes
+*MediaSize Letter
+MediaSize Legal
+MediaSize A4
+
+// Query command for page size
+Attribute "?PageSize" "" "
+      save
+      currentpagedevice /PageSize get aload pop
+      2 copy gt {exch} if (Unknown)
+      23 dict
+              dup [612 792] (Letter) put
+              dup [612 1008] (Legal) put
+              dup [595 842] (A4) put
+              {exch aload pop 4 index sub abs 5 le exch 
+               5 index sub abs 5 le and
+              {exch pop exit} {pop} ifelse
+      } bind forall = flush pop pop
+      restore"
+
+// Specify the name of the PPD file we want to generate
+PCFileName "fooproof.ppd"
+
+ +

Required Attributes

+ +

PostScript drivers require the attributes listed in Table 1. If not specified, the defaults for CUPS drivers are used. A typical PostScript driver information file would include the following attributes:

+ +
+Attribute DefaultColorSpace "" Gray
+Attribute LandscapeOrientation "" Minus90
+Attribute LanguageLevel "" "3"
+Attribute Product "" "(Foo LaserProofer 2000)"
+Attribute PSVersion "" "(3010) 0"
+Attribute TTRasterizer "" Type42
+
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 1: Required PostScript Printer Driver Attributes
AttributeDescription
DefaultColorSpaceThe default colorspace: + Gray, RGB, CMY, or + CMYK. If not specified, then RGB is + assumed.
LandscapeOrientationThe preferred landscape + orientation: Plus90, Minus90, or + Any. If not specified, Plus90 is + assumed.
LanguageLevelThe PostScript language + level supported by the device: 1, 2, or 3. If not + specified, 2 is assumed.
ProductThe string returned by + the PostScript product operator, which + must include parenthesis to conform with + PostScript syntax rules for strings. Multiple + Product attributes may be specified to support + multiple products with the same PPD file. If not + specified, "(ESP Ghostscript)" and "(GNU Ghostscript)" + are assumed.
PSVersionThe PostScript + interpreter version numbers as returned by the + version and revision operators. The + required format is "(version) revision". Multiple + PSVersion attributes may be specified to + support multiple interpreter version numbers. If not + specified, "(3010) 705" and "(3010) 707" are + assumed.
TTRasterizerThe type of TrueType + font rasterizer supported by the device, if any. The + supported values are None, Accept68k, + Type42, and TrueImage. If not + specified, None is assumed.
+ +

Query Commands

+ +

Most PostScript printer PPD files include query commands (?PageSize, etc.) that allow applications to query the printer for its current settings and configuration. Query commands are included in driver information files as attributes. For example, the example in Listing 1 uses the following definition for the PageSize query command:

+ +
+Attribute "?PageSize" "" "
+      save
+      currentpagedevice /PageSize get aload pop
+      2 copy gt {exch} if (Unknown)
+      23 dict
+              dup [612 792] (Letter) put
+              dup [612 1008] (Legal) put
+              dup [595 842] (A4) put
+              {exch aload pop 4 index sub abs 5 le exch 
+               5 index sub abs 5 le and
+              {exch pop exit} {pop} ifelse
+      } bind forall = flush pop pop
+      restore"
+
+ +

Query commands can span multiple lines, however no single line may contain more than 255 characters.

+ +

Importing Existing PPD Files

+ +

CUPS includes a utility called ppdi(1) +which allows you to import existing PPD files into the driver information file +format used by the PPD compiler ppdc(1). Once +imported, you can modify, localize, and regenerate the PPD files easily. Type +the following command to import the PPD file mydevice.ppd into the +driver information file mydevice.drv:

+ +
+ppdi -o mydevice.drv mydevice.ppd
+
+ +

If you have a whole directory of PPD files that you would like to import, +you can list multiple filenames or use shell wildcards to import more than one +PPD file on the command-line:

+ +
+ppdi -o mydevice.drv mydevice1.ppd mydevice2.ppd
+ppdi -o mydevice.drv *.ppd
+
+ +

If the driver information file already exists, the new PPD +file entries are appended to the end of the file. Each PPD file +is placed in its own group of curly braces within the driver +information file.

+ + +

Using Custom Filters

+ +

Normally a PostScript printer driver will not utilize any additional print filters. For drivers that provide additional filters such as a CUPS command file filter for doing printer maintenance, you must also list the following Filter directive to handle printing PostScript files:

+ +
+Filter application/vnd.cups-postscript 0 -
+
+ +

Custom Command Filters

+ +

The application/vnd.cups-command file type is used for CUPS command files. Use the following Filter directive to handle CUPS command files:

+ +
+Filter application/vnd.cups-command 100 /path/to/command/filter
+
+ +

To use the standard PostScript command filter, specify commandtops as the path to the command filter.

+ +

Custom PDF Filters

+ +

The application/pdf file type is used for unfiltered PDF files while the application/vnd.cups-pdf file type is used for filtered PDF files. Use the following Filter directive to handle filtered PDF files:

+ +
+Filter application/vnd.cups-pdf 100 /path/to/pdf/filter
+
+ +

For unfiltered PDF files, use:

+ +
+Filter application/pdf 100 /path/to/pdf/filter
+
+ +

Custom PDF filters that accept filtered data do not need to perform number-up processing and other types of page imposition, while those that accept unfiltered data MUST do the number-up processing themselves.

+ +

Custom PostScript Filters

+ +

The application/vnd.cups-postscript file type is used for filtered PostScript files. Use the following Filter directive to handle PostScript files:

+ +
+Filter application/vnd.cups-postscript 100 /path/to/postscript/filter
+
+ + +

Implementing Color Management

+ +

CUPS uses ICC color profiles to provide more accurate color reproduction. The cupsICCProfile attribute defines the color profiles that are available for a given printer, for example:

+ +
+Attribute cupsICCProfile "ColorModel.MediaType.Resolution/Description" /path/to/ICC/profile
+
+ +

where "ColorModel.MediaType.Resolution" defines a selector based on the corresponding option selections. A simple driver might only define profiles for the color models that are supported, for example a printer supporting Gray and RGB might use:

+ +
+Attribute cupsICCProfile "Gray../Grayscale Profile" /path/to/ICC/gray-profile
+Attribute cupsICCProfile "RGB../Full Color Profile" /path/to/ICC/rgb-profile
+
+ +

The options used for profile selection can be customized using the cupsICCQualifier2 and cupsICCQualifier3 attributes.

+ + +

Adding Mac OS X Features

+ +

Mac OS X printer drivers can provide additional attributes to specify additional option panes in the print dialog, an image of the printer, a help book, and option presets for the driver software:

+ +
+Attribute APDialogExtension "" /Library/Printers/Vendor/filename.plugin
+Attribute APHelpBook "" /Library/Printers/Vendor/filename.bundle
+Attribute APPrinterIconPath "" /Library/Printers/Vendor/filename.icns
+Attribute APPrinterPreset "name/text" "*option choice ..."
+
+
+ + diff --git a/doc/help/ppd-compiler.html b/doc/help/ppd-compiler.html new file mode 100644 index 0000000000..db891f17dc --- /dev/null +++ b/doc/help/ppd-compiler.html @@ -0,0 +1,1279 @@ + + + + + Introduction to the PPD Compiler + + + + + + +
+ + +

Introduction to the PPD Compiler

+ +

This document describes how to use the CUPS PostScript Printer Description +(PPD) file compiler. The PPD compiler generates PPD files from simple text files +that describe the features and capabilities of one or more printers.

+ +
+ + + + + + +
See AlsoProgramming: Developing Raster Printer Drivers
+ Programming: Developing PostScript Printer Drivers
+ Programming: Filter and Backend Programming
+ Programming: Raster API
+ References: PPD Compiler Driver Information File Reference
+ Specifications: CUPS PPD Extensions
+

Contents

+ +

The Basics

+ +

The PPD compiler, ppdc(1), is a +simple command-line tool that takes a single driver information file, +which by convention uses the extension .drv, and produces one or more +PPD files that may be distributed with your printer drivers for use with CUPS. +For example, you would run the following command to create the English language +PPD files defined by the driver information file mydrivers.drv:

+ +
+ppdc mydrivers.drv
+
+ +

The PPD files are placed in a subdirectory called +ppd. The -d option is used to put the PPD +files in a different location, for example:

+ +
+ppdc -d myppds mydrivers.drv
+
+ +

places the PPD files in a subdirectory named +myppds. Finally, use the -l option to +specify the language localization for the PPD files that are +created, for example:

+ +
+ppdc -d myppds/de -l de mydrivers.drv
+ppdc -d myppds/en -l en mydrivers.drv
+ppdc -d myppds/es -l es mydrivers.drv
+ppdc -d myppds/fr -l fr mydrivers.drv
+ppdc -d myppds/it -l it mydrivers.drv
+
+ +

creates PPD files in German (de), English (en), Spanish (es), +French (fr), and Italian (it) in the corresponding +subdirectories. Specify multiple languages (separated by commas) to produce +"globalized" PPD files:

+ +
+ppdc -d myppds -l de,en,es,fr,it mydrivers.drv
+
+ + +

Driver Information Files

+ +

The driver information files accepted by the PPD compiler are +plain text files that define the various attributes and options +that are included in the PPD files that are generated. A driver +information file can define the information for one or more printers and +their corresponding PPD files.

+ +

Listing 1: "examples/minimum.drv"

+ +
+// Include standard font and media definitions
+#include <font.defs>
+#include <media.defs>
+
+// List the fonts that are supported, in this case all standard fonts...
+Font *
+
+// Manufacturer, model name, and version
+Manufacturer "Foo"
+ModelName "FooJet 2000"
+Version 1.0
+
+// Each filter provided by the driver...
+Filter application/vnd.cups-raster 100 rastertofoo
+
+// Supported page sizes
+*MediaSize Letter
+MediaSize A4
+
+// Supported resolutions
+*Resolution k 8 0 0 0 "600dpi/600 DPI"
+
+// Specify the name of the PPD file we want to generate...
+PCFileName "foojet2k.ppd"
+
+ + +

A Simple Example

+ +

The example in Listing 1 shows a driver information +file which defines the minimum required attributes to provide a valid PPD file. +The first part of the file includes standard definition files for fonts and +media sizes:

+ +
+#include <font.defs>
+#include <media.defs>
+
+ +

The #include directive works just like the C/C++ include directive; +files included using the angle brackets (<filename>) are found +in any of the standard include directories and files included using quotes +("filename") are found in the same directory as the source or include +file. The <font.defs> include file defines the standard fonts +which are included with GPL Ghostscript and the Apple PDF RIP, while the +<media.defs> include file defines the standard media sizes +listed in Appendix B of the Adobe PostScript Printer Description File Format +Specification.

+ +

CUPS provides several other standard include files:

+ +
    + +
  • <epson.h> - Defines all of the rastertoepson driver + constants.
  • + +
  • <escp.h> - Defines all of the rastertoescpx driver + constants.
  • + +
  • <hp.h> - Defines all of the rastertohp driver + constants.
  • + +
  • <label.h> - Defines all of the rastertolabel driver + constants.
  • + +
  • <pcl.h> - Defines all of the rastertopclx driver + constants.
  • + +
  • <raster.defs> - Defines all of the CUPS raster format + constants.
  • + +
+ +

Next we list all of the fonts that are available in the driver; for CUPS +raster drivers, the following line is all that is usually supplied:

+ +
+Font *
+
+ +

The Font directive specifies the name of a single font or the +asterisk to specify all fonts. For example, you would use the following line to +define an additional bar code font that you are supplying with your printer +driver:

+ +
+//   name         encoding  version  charset  status
+Font Barcode-Foo  Special   "(1.0)"  Special  ROM
+
+ +

The name of the font is Barcode-Foo. Since it is not a standard +text font, the encoding and charset name Special is used. The version +number is 1.0 and the status (where the font is located) is +ROM to indicate that the font does not need to be embedded in +documents that use the font for this printer.

+ +

Third comes the manufacturer, model name, and version number information +strings:

+ +
+Manufacturer "Foo"
+ModelName "FooJet 2000"
+Version 1.0
+
+ +

These strings are used when the user (or auto-configuration program) selects +the printer driver for a newly connected device.

+ +

The list of filters comes after the information strings; for the example in +Listing 1, we have a single filter that takes CUPS +raster data:

+ +
+Filter application/vnd.cups-raster 100 rastertofoo
+
+ +

Each filter specified in the driver information file is the equivalent of a +printer driver for that format; if a user submits a print job in a different +format, CUPS figures out the sequence of commands that will produce a supported +format for the least relative cost.

+ +

Once we have defined the driver information we specify the supported options. +For the example driver we support a single resolution of 600 dots per inch and +two media sizes, A4 and Letter:

+ +
+*MediaSize Letter
+MediaSize A4
+
+*Resolution k 8 0 0 0 "600dpi/600 DPI"
+
+ +

The asterisk in front of the MediaSize and Resolution +directives specify that those option choices are the default. The +MediaSize directive is followed by a media size name which is normally +defined in the <media.defs> file and corresponds to a standard +Adobe media size name. If the default media size is Letter, the PPD +compiler will override it to be A4 for non-English localizations for +you automatically.

+ +

The Resolution directive accepts several values after it as +follows:

+ +
    + +
  1. Colorspace for this resolution, if any. In the example file, the + colorspace k is used which corresponds to black. For printer + drivers that support color printing, this field is usually specified as + "-" for "no change".
  2. + +
  3. Bits per color. In the example file, we define 8 bits per color, for + a continuous-tone grayscale output. All versions of CUPS support 1 and + 8 bits per color. CUPS 1.2 and higher (Mac OS X 10.5 and higher) also + supports 16 bits per color.
  4. + +
  5. Rows per band. In the example file, we define 0 rows per band to + indicate that our printer driver does not process the page in + bands.
  6. + +
  7. Row feed. In the example, we define the feed value to be 0 to + indicate that our printer driver does not interleave the output.
  8. + +
  9. Row step. In the example, we define the step value to be 0 to + indicate that our printer driver does not interleave the output. This + value normally indicates the spacing between the nozzles of an inkjet + printer - when combined with the previous two values, it informs the + driver how to stagger the output on the page to produce interleaved + lines on the page for higher-resolution output.
  10. + +
  11. Choice name and text. In the example, we define the choice name and + text to be "600dpi/600 DPI". The name and text are separated by + slash (/) character; if no text is specified, then the name is + used as the text. The PPD compiler parses the name to determine the + actual resolution; the name can be of the form + RESOLUTIONdpi for resolutions that are equal + horizontally and vertically or HRESxVRESdpi for + isometric resolutions. Only integer resolution values are supported, so + a resolution name of 300dpi is valid while 300.1dpi is + not.
  12. + +
+ +

Finally, the PCFileName directive specifies that the named PPD file +should be written for the current driver definitions:

+ +
+PCFileName "foojet2k.ppd"
+
+ +

The filename follows the directive and must conform to the Adobe +filename requirements in the Adobe Postscript Printer Description File Format +Specification. Specifically, the filename may not exceed 8 characters followed +by the extension .ppd. The FileName directive can be used to +specify longer filenames:

+ +
+FileName "FooJet 2000"
+
+ + +

Grouping and Inheritance

+ +

The previous example created a single PPD file. Driver information files can +also define multiple printers by using the PPD compiler grouping functionality. +Directives are grouped using the curly braces ({ and }) and +every group that uses the PCFileName or FileName directives +produces a PPD file with that name. Listing 2 shows a +variation of the original example that uses two groups to define two printers +that share the same printer driver filter but provide two different resolution +options.

+ +

Listing 2: "examples/grouping.drv"

+ +
+
+// Include standard font and media definitions
+#include <font.defs>
+#include <media.defs>
+
+// List the fonts that are supported, in this case all standard fonts...
+Font *
+
+// Manufacturer and version
+Manufacturer "Foo"
+Version 1.0
+
+// Each filter provided by the driver...
+Filter application/vnd.cups-raster 100 rastertofoo
+
+// Supported page sizes
+*MediaSize Letter
+MediaSize A4
+
+{
+  // Supported resolutions
+  *Resolution k 8 0 0 0 "600dpi/600 DPI"
+
+  // Specify the model name and filename...
+  ModelName "FooJet 2000"
+  PCFileName "foojet2k.ppd"
+}
+
+{
+  // Supported resolutions
+  *Resolution k 8 0 0 0 "1200dpi/1200 DPI"
+
+  // Specify the model name and filename...
+  ModelName "FooJet 2001"
+  PCFileName "foojt2k1.ppd"
+}
+
+ +

The second example is essentially the same as the first, except that each +printer model is defined inside of a pair of curly braces. For example, the +first printer is defined using:

+ +
+{
+  // Supported resolutions
+  *Resolution k 8 0 0 0 "600dpi/600 DPI"
+
+  // Specify the model name and filename...
+  ModelName "FooJet 2000"
+  PCFileName "foojet2k.ppd"
+}
+
+ +

The printer inherits all of the definitions from the parent group (the +top part of the file) and adds the additional definitions inside the curly +braces for that printer driver. When we define the second group, it also +inherits the same definitions from the parent group but none of the +definitions from the first driver. Groups can be nested to any number of levels +to support variations of similar models without duplication of information.

+ + +

Color Support

+ +

For printer drivers that support color printing, the +ColorDevice and ColorModel directives should be +used to tell the printing system that color output is desired +and in what formats. Listing 3 shows a +variation of the previous example which includes a color printer +that supports printing at 300 and 600 DPI.

+ +

The key changes are the addition of the ColorDevice +directive:

+ +
+ColorDevice true
+
+ +

which tells the printing system that the printer supports +color printing, and the ColorModel directives:

+ +
+ColorModel Gray/Grayscale w chunky 0
+*ColorModel RGB/Color rgb chunky 0
+
+ +

which tell the printing system which colorspaces are supported by the printer +driver for color printing. Each of the ColorModel directives is +followed by the option name and text (Gray/Grayscale and +RGB/Color), the colorspace name (w and rgb), the +color organization (chunky), and the compression mode number +(0) to be passed to the driver. The option name can be any of the +standard Adobe ColorModel names:

+ +
    + +
  • Gray - Grayscale output. + +
  • RGB - Color output, typically using the RGB + colorspace, but without a separate black channel. + +
  • CMYK - Color output with a separate black + channel. + +
+ +

Custom names can be used, however it is recommended that you use your vendor +prefix for any custom names, for example "fooName".

+ +

The colorspace name can be any of the following universally supported +colorspaces:

+ +
    +
  • w - Luminance
  • + +
  • rgb - Red, green, blue
  • + +
  • k - Black
  • + +
  • cmy - Cyan, magenta, yellow
  • + +
  • cmyk - Cyan, magenta, yellow, black
  • + +
+ +

The color organization can be any of the following values:

+ +
    + +
  • chunky - Color values are passed together on a line + as RGB RGB RGB RGB
  • + +
  • banded - Color values are passed separately + on a line as RRRR GGGG BBBB; not supported by the Apple + RIP filters
  • + +
  • planar - Color values are passed separately + on a page as RRRR RRRR RRRR ... GGGG GGGG GGGG ... BBBB + BBBB BBBB; not supported by the Apple RIP filters
  • + +
+ +

The compression mode value is passed to the driver in the +cupsCompression attribute. It is traditionally used to select an +appropriate compression mode for the color model but can be used for any +purpose, such as specifying a photo mode vs. standard mode.

+ +

Listing 3: "examples/color.drv"

+ +
+
+// Include standard font and media definitions
+#include <font.defs>
+#include <media.defs>
+
+// List the fonts that are supported, in this case all standard fonts...
+Font *
+
+// Manufacturer and version
+Manufacturer "Foo"
+Version 1.0
+
+// Each filter provided by the driver...
+Filter application/vnd.cups-raster 100 rastertofoo
+
+// Supported page sizes
+*MediaSize Letter
+MediaSize A4
+
+{
+  // Supported resolutions
+  *Resolution k 8 0 0 0 "600dpi/600 DPI"
+
+  // Specify the model name and filename...
+  ModelName "FooJet 2000"
+  PCFileName "foojet2k.ppd"
+}
+
+{
+  // Supports color printing
+  ColorDevice true
+
+  // Supported colorspaces
+  ColorModel Gray/Grayscale w chunky 0
+  *ColorModel RGB/Color rgb chunky 0
+
+  // Supported resolutions
+  *Resolution - 8 0 0 0 "300dpi/300 DPI"
+  Resolution - 8 0 0 0 "600dpi/600 DPI"
+
+  // Specify the model name and filename...
+  ModelName "FooJet Color"
+  PCFileName "foojetco.ppd"
+}
+
+ + +

Defining Custom Options and Option Groups

+ +

The Group, Option, and Choice +directives are used to define or select a group, option, or +choice. Listing 4 shows a variation of +the first example that provides two custom options in a group +named "Footasm".

+ +

Listing 4: "examples/custom.drv"

+ +
+
+// Include standard font and media definitions
+#include <font.defs>
+#include <media.defs>
+
+// List the fonts that are supported, in this case all standard fonts...
+Font *
+
+// Manufacturer, model name, and version
+Manufacturer "Foo"
+ModelName "FooJet 2000"
+Version 1.0
+
+// Each filter provided by the driver...
+Filter application/vnd.cups-raster 100 rastertofoo
+
+// Supported page sizes
+*MediaSize Letter
+MediaSize A4
+
+// Supported resolutions
+*Resolution k 8 0 0 0 "600dpi/600 DPI"
+
+// Option Group
+Group "Footasm"
+
+  // Boolean option
+  Option "fooEnhance/Resolution Enhancement" Boolean AnySetup 10
+    *Choice True/Yes "<</cupsCompression 1>>setpagedevice"
+    Choice False/No "<</cupsCompression 0>>setpagedevice"
+
+  // Multiple choice option
+  Option "fooOutputType/Output Quality" PickOne AnySetup 10
+    *Choice "Auto/Automatic Selection"
+            "<</OutputType(Auto)>>setpagedevice""
+    Choice "Text/Optimize for Text"
+            "<</OutputType(Text)>>setpagedevice""
+    Choice "Graph/Optimize for Graphics"
+            "<</OutputType(Graph)>>setpagedevice""
+    Choice "Photo/Optimize for Photos"
+            "<</OutputType(Photo)>>setpagedevice""
+
+// Specify the name of the PPD file we want to generate...
+PCFileName "foojet2k.ppd"
+
+ +

The custom group is introduced by the Group +directive which is followed by the name and optionally text for +the user:

+ +
+Group "Footasm/Footastic Options"
+
+ +

The group name must conform to the PPD specification and +cannot exceed 40 characters in length. If you specify user text, +it cannot exceed 80 characters in length. The groups +General, Extra, and +InstallableOptions are predefined by CUPS; the general +and extra groups are filled by the UI options defined by the PPD +specification. The InstallableOptions group is reserved +for options that define whether accessories for the printer +(duplexer unit, finisher, stapler, etc.) are installed.

+ +

Once the group is specified, the Option directive is +used to introduce a new option:

+ +
+Option "fooEnhance/Resolution Enhancement" Boolean AnySetup 10
+
+ +

The directive is followed by the name of the option and any +optional user text, the option type, the PostScript document group, and +the sort order number. The option name must conform to the PPD specification +and cannot exceed 40 characters in length. If you specify user text, it +cannot exceed 80 characters in length.

+ +

The option type can be Boolean for true/false +selections, PickOne for picking one of many choices, or +PickMany for picking zero or more choices. Boolean +options can have at most two choices with the names +False and True. Pick options can have any +number of choices, although for Windows compatibility reasons +the number of choices should not exceed 255.

+ +

The PostScript document group is typically AnySetup, +meaning that the option can be introduced at any point in the +PostScript document. Other values include PageSetup to +include the option before each page and DocumentSetup +to include the option once at the beginning of the document.

+ +

The sort order number is used to sort the printer commands +associated with each option choice within the PostScript +document. This allows you to setup certain options before others +as required by the printer. For most CUPS raster printer +drivers, the value 10 can be used for all options.

+ +

Once the option is specified, each option choice can be +listed using the Choice directive:

+ +
+*Choice True/Yes "<</cupsCompression 1>>setpagedevice"
+Choice False/No "<</cupsCompression 0>>setpagedevice"
+
+ +

The directive is followed by the choice name and optionally +user text, and the PostScript commands that should be inserted +when printing a file to this printer. The option name must +conform to the PPD specification and cannot exceed 40 characters +in length. If you specify user text, it cannot exceed 80 +characters in length.

+ +

The PostScript commands are also interpreted by any RIP +filters, so these commands typically must be present for all +option choices. Most commands take the form:

+ +
+<</name value>>setpagedevice
+
+ +

where name is the name of the PostScript page device +attribute and value is the numeric or string value for +that attribute.

+ + +

Defining Constants

+ +

Sometimes you will want to define constants for your drivers +so that you can share values in different groups within the same +driver information file, or to share values between different +driver information files using the #include directive. +The #define directive is used to define constants for +use in your printer definitions:

+ +
+#define NAME value
+
+ +

The NAME is any sequence of letters, numbers, and +the underscore. The value is a number or string; if the +value contains spaces you must put double quotes around it, for +example:

+ +
+#define FOO "My String Value"
+
+ +

Constants can also be defined on the command-line using the -D +option:

+ +
+ppdc -DNAME="value" filename.drv
+
+ +

Once defined, you use the notation $NAME to substitute the value of +the constant in the file, for example:

+ +
+#define MANUFACTURER "Foo"
+#define FOO_600      0
+#define FOO_1200     1
+
+{
+  Manufacturer "$MANUFACTURER"
+  ModelNumber $FOO_600
+  ModelName "FooJet 2000"
+  ...
+}
+
+{
+  Manufacturer "$MANUFACTURER"
+  ModelNumber $FOO_1200
+  ModelName "FooJet 2001"
+  ...
+}
+
+ +

Numeric constants can be bitwise OR'd together by placing the constants +inside parenthesis, for example:

+ +
+// ModelNumber capability bits
+#define DUPLEX 1
+#define COLOR  2
+
+...
+
+{
+  // Define a model number specifying the capabilities of the printer...
+  ModelNumber ($DUPLEX $COLOR)
+  ...
+}
+
+ + +

Conditional Statements

+ +

The PPD compiler supports conditional compilation using the #if, +#elif, #else, and #endif directives. The #if +and #elif directives are followed by a constant name or an expression. +For example, to include a group of options when "ADVANCED" is defined:

+ +
+#if ADVANCED
+Group "Advanced/Advanced Options"
+  Option "fooCyanAdjust/Cyan Adjustment"
+    Choice "plus10/+10%" ""
+    Choice "plus5/+5%" ""
+    *Choice "none/No Adjustment" ""
+    Choice "minus5/-5%" ""
+    Choice "minus10/-10%" ""
+  Option "fooMagentaAdjust/Magenta Adjustment"
+    Choice "plus10/+10%" ""
+    Choice "plus5/+5%" ""
+    *Choice "none/No Adjustment" ""
+    Choice "minus5/-5%" ""
+    Choice "minus10/-10%" ""
+  Option "fooYellowAdjust/Yellow Adjustment"
+    Choice "plus10/+10%" ""
+    Choice "plus5/+5%" ""
+    *Choice "none/No Adjustment" ""
+    Choice "minus5/-5%" ""
+    Choice "minus10/-10%" ""
+  Option "fooBlackAdjust/Black Adjustment"
+    Choice "plus10/+10%" ""
+    Choice "plus5/+5%" ""
+    *Choice "none/No Adjustment" ""
+    Choice "minus5/-5%" ""
+    Choice "minus10/-10%" ""
+#endif
+
+ + +

Defining Constraints

+ +

Constraints are strings that are used to specify that one or more option +choices are incompatible, for example two-sided printing on transparency media. +Constraints are also used to prevent the use of uninstalled features such as the +duplexer unit, additional media trays, and so forth.

+ +

The UIConstraints directive is used to specify a constraint that is +placed in the PPD file. The directive is followed by a string using one of the +following formats:

+ +
+UIConstraints "*Option1 *Option2"
+UIConstraints "*Option1 Choice1 *Option2"
+UIConstraints "*Option1 *Option2 Choice2"
+UIConstraints "*Option1 Choice1 *Option2 Choice2"
+
+ +

Each option name is preceded by the asterisk (*). If no choice is +given for an option, then all choices except False and +None will conflict with the other option and choice(s). Since the PPD +compiler automatically adds reciprocal constraints (option A conflicts with +option B, so therefore option B conflicts with option A), you need only specify +the constraint once.

+ +

Listing 5: "examples/constraint.drv"

+ +
+
+// Include standard font and media definitions
+#include <font.defs>
+#include <media.defs>
+
+// List the fonts that are supported, in this case all standard fonts...
+Font *
+
+// Manufacturer, model name, and version
+Manufacturer "Foo"
+ModelName "FooJet 2000"
+Version 1.0
+
+// Each filter provided by the driver...
+Filter application/vnd.cups-raster 100 rastertofoo
+
+// Supported page sizes
+*MediaSize Letter
+MediaSize A4
+
+// Supported resolutions
+*Resolution k 8 0 0 0 "600dpi/600 DPI"
+
+// Installable Option Group
+Group "InstallableOptions/Options Installed"
+
+  // Duplexing unit option
+  Option "OptionDuplexer/Duplexing Unit" Boolean AnySetup 10
+    Choice True/Installed ""
+    *Choice "False/Not Installed" ""
+
+// General Option Group
+Group General
+
+  // Duplexing option
+  Option "Duplex/Two-Sided Printing" PickOne AnySetup 10
+    *Choice "None/No" "<</Duplex false>>setpagedevice""
+    Choice "DuplexNoTumble/Long Edge Binding"
+           "<</Duplex true/Tumble false>>setpagedevice""
+    Choice "DuplexTumble/Short Edge Binding"
+           "<</Duplex true/Tumble true>>setpagedevice""
+
+// Only allow duplexing if the duplexer is installed
+UIConstraints "*Duplex *OptionDuplexer False"
+
+// Specify the name of the PPD file we want to generate...
+PCFileName "foojet2k.ppd"
+
+ +

Listing 5 shows a variation of the first example with +an added Duplex option and installable option for the duplexer, +OptionDuplex. A constraint is added at the end to specify that any +choice of the Duplex option that is not None is incompatible +with the "Duplexer Installed" option set to "Not Installed" +(False):

+ +
+UIConstraints "*Duplex *OptionDuplexer False"
+
+ +

Enhanced Constraints

+ +

CUPS 1.4 supports constraints between 2 or more options using the +Attribute directive. cupsUIConstraints attributes define +the constraints, while cupsUIResolver attributes define option changes +to resolve constraints. For example, we can specify the previous duplex +constraint with a resolver that turns off duplexing with the following two +lines:

+ +
+Attribute cupsUIConstraints DuplexOff "*Duplex *OptionDuplexer False"
+Attribute cupsUIResolver DuplexOff "*Duplex None"
+
+ +

Localization

+ +

The PPD compiler provides localization of PPD files in different languages +through message catalog files in the GNU gettext or Apple .strings +formats. Each user text string and several key PPD attribute values such as +LanguageVersion and LanguageEncoding are looked up in the +corresponding message catalog and the translated text is substituted in the +generated PPD files. One message catalog file can be used by multiple driver +information files, and each file contains a single language translation.

+ +

The ppdpo Utility

+ +

While CUPS includes localizations of all standard media sizes and options in +several languages, your driver information files may provide their own media +sizes and options that need to be localized. CUPS provides a utility program to +aid in the localization of drivers called ppdpo(1). The ppdpo program creates +or updates a message catalog file based upon one or more driver information +files. New messages are added with the word "TRANSLATE" added to the front of +the translation string to make locating new strings for translation easier. The +program accepts the message catalog filename and one or more driver information +files.

+ +

For example, run the following command to create a new German message catalog +called de.po for all of the driver information files in the current +directory:

+ +
+ppdpo -o de.po *.drv
+
+ +

If the file de.po already exists, ppdpo will update the +contents of the file with any new messages that need to be translated. To create +an Apple .strings file instead, specify the output filename with a .strings +extension, for example:

+ +
+ppdpo -o de.strings *.drv
+
+ +

Using Message Catalogs with the PPD Compiler

+ +

Once you have created a message catalog, use the #po directive to declare it in each +driver information file. For example, to declare the German message catalog for +a driver use:

+ +
+#po de "de.po"  // German
+
+ +

In fact, you can use the #po directive as many times as needed:

+ +
+#po de "de.po"  // German
+#po es "es.po"  // Spanish
+#po fr "fr.po"  // French
+#po it "it.po"  // Italian
+#po ja "ja.po"  // Japanese
+
+ +

The filename ("de.po", etc.) can be relative to the location of the driver +information file or an absolute path. Once defined, the PPD compiler will +automatically generate a globalized PPD for every language declared in your +driver information file. To generate a single-language PPD file, simply use the +-l option to list the corresponding locale, for example:

+ +
+ppdc -l de -d ppd/de mydrivers.drv
+
+ +

to generate German PPD files.

+
+ + diff --git a/doc/help/raster-driver.html b/doc/help/raster-driver.html new file mode 100644 index 0000000000..2ec3f4e5a8 --- /dev/null +++ b/doc/help/raster-driver.html @@ -0,0 +1,579 @@ + + + + + Developing Raster Printer Drivers + + + + + + +
+ + +

Developing Raster Printer Drivers

+ +

This document describes how to develop printer drivers for raster printers. Topics include: printer driver basics, creating new PPD files, using filters, implementing color management, and adding Mac OS X features.

+ +
+ + + + + + +
See AlsoProgramming: Developing PostScript Printer Drivers
+ Programming: Filter and Backend Programming
+ Programming: Introduction to the PPD Compiler
+ Programming: Raster API
+ References: PPD Compiler Driver Information File Reference
+ Specifications: CUPS PPD Extensions
+

Contents

+ +

Printer Driver Basics

+ +

A CUPS raster printer driver consists of a PostScript Printer Description (PPD) file that describes the features and capabilities of the device, one or more filter programs that prepare print data for the device, and zero or more support files for color management, online help, and so forth. The PPD file includes references to all of the filters and support files used by the driver.

+ +

Every time a user prints something the scheduler program, cupsd(8), determines the format of the print job and the programs required to convert that job into something the printer understands. CUPS includes filter programs for many common formats, for example to convert Portable Document Format (PDF) files into CUPS raster data. Figure 1 shows the data flow of a typical print job.

+ +
+ + +
Figure 1: Raster Filter Chain
Raster Filter Chain
+ +

The raster filter converts CUPS raster data into a format the printer understands, for example HP-PCL. CUPS includes several sample raster filters supporting standard page description languages (PDLs). Table 1 shows the raster filters that are bundled with CUPS and the languages they support.

+ +
+ + + + + + + + + + + +
Table 1: Standard CUPS Raster Filters
FilterPDLsppdc DriverTypeppdc #include file
rastertoepsonESC/P, ESC/P2epsonepson.h
rastertoescpxESC/P, ESC/P2, EPSON Remote Modeescpescp.h
rastertohpHP-PCL3, HP-PCL5hphp.h
rastertolabelCPCL, Dymo, EPL1, EPL2, Intellitech PCL, ZPLlabellabel.h
rastertopclxHP-RTL, HP-PCL3, HP-PCL3GUI, HP-PCL5, HP-PCL5c, HP-PCL5epclpcl.h
+ +

The optional port monitor handles interface-specific protocol or encoding issues. For example, some raster printers use the 1284.4 communications protocol.

+ +

The backend handles communications with the printer, sending print data from the last filter to the printer and relaying back-channel data from the printer to the upstream filters. CUPS includes backend programs for common direct-connect interfaces and network protocols, and you can provide your own backend to support custom interfaces and protocols.

+ +

The scheduler also supports a special "command" file format for sending maintenance commands and status queries to a printer or printer driver. Command print jobs typically use a single command filter program defined in the PPD file to generate the appropriate printer commands and handle any responses from the printer. Figure 2 shows the data flow of a typical command job.

+ +
+ + +
Figure 2: Command Filter Chain
Command Filter Chain
+ +

Raster printer drivers must provide their own command filter.

+ + +

Creating New PPD Files

+ +

We recommend using the CUPS PPD compiler, ppdc(1), to create new PPD files since it manages many of the tedious (and error-prone!) details of paper sizes and localization for you. It also allows you to easily support multiple devices from a single source file. For more information see the "Introduction to the PPD Compiler" document. Listing 1 shows a driver information file for several similar black-and-white HP-PCL5 laser printers.

+ +

Listing 1: "examples/laserjet-basic.drv"

+ +
+// Include standard font and media definitions
+#include <font.defs>
+#include <media.defs>
+
+// Include HP-PCL driver definitions
+#include <pcl.h>
+
+// Specify that this driver uses the HP-PCL driver...
+DriverType pcl
+
+// Specify the driver options via the model number...
+ModelNumber ($PCL_PAPER_SIZE $PCL_PJL $PCL_PJL_RESOLUTION)
+
+// List the fonts that are supported, in this case all standard fonts...
+Font *
+
+// Manufacturer and driver version
+Manufacturer "HP"
+Version 1.0
+
+// Supported page sizes and their margins
+HWMargins 18 12 18 12
+*MediaSize Letter
+MediaSize Legal
+MediaSize Executive
+MediaSize Monarch
+MediaSize Statement
+MediaSize FanFoldGermanLegal
+
+HWMargins 18 12.72 18 12.72
+MediaSize Env10
+
+HWMargins 9.72 12 9.72 12
+MediaSize A4
+MediaSize A5
+MediaSize B5
+MediaSize EnvC5
+MediaSize EnvDL
+MediaSize EnvISOB5
+MediaSize Postcard
+MediaSize DoublePostcard
+
+// Only black-and-white output with mode 3 compression...
+ColorModel Gray k chunky 3
+
+// Supported resolutions
+Resolution - 1 0 0 0 "300dpi/300 DPI"
+*Resolution - 8 0 0 0 "600dpi/600 DPI"
+
+// Supported input slots
+*InputSlot 7 "Auto/Automatic Selection"
+InputSlot 2 "Manual/Tray 1 - Manual Feed"
+InputSlot 4 "Upper/Tray 1"
+InputSlot 1 "Lower/Tray 2"
+InputSlot 5 "LargeCapacity/Tray 3"
+
+// Tray 3 is an option...
+Installable "OptionLargeCapacity/Tray 3 Installed"
+UIConstraints "*OptionLargeCapacity False *InputSlot LargeCapacity"
+
+{
+  // HP LaserJet 2100 Series
+  Throughput 10
+  ModelName "LaserJet 2100 Series"
+  PCFileName "hpljt211.ppd"
+}
+
+{
+  // LaserJet 2200 and 2300 series have duplexer option...
+  Duplex normal
+  Installable "OptionDuplex/Duplexer Installed"
+  UIConstraints "*OptionDuplex False *Duplex"
+
+  {
+    // HP LaserJet 2200 Series
+    Throughput 19
+    ModelName "LaserJet 2200 Series"
+    PCFileName "hpljt221.ppd"
+  }
+
+  {
+    // HP LaserJet 2300 Series
+    Throughput 25
+    ModelName "LaserJet 2300 Series"
+    PCFileName "hpljt231.ppd"
+  }
+}
+
+ + +

Using Filters

+ +

The standard CUPS raster filters can be specified using the +DriverType directive, for example:

+ +
+// Specify that this driver uses the HP-PCL driver...
+DriverType pcl
+
+ +

Table 1 shows the driver types for each of the standard CUPS raster filters. For drivers that do not use the standard raster filters, the "custom" type is used with Filter directives:

+ +
+DriverType custom
+Filter application/vnd.cups-raster 100 /path/to/raster/filter
+Filter application/vnd.cups-command 100 /path/to/command/filter
+
+ + +

Implementing Color Management

+ +

CUPS uses ICC color profiles to provide more accurate color reproduction. The cupsICCProfile attribute defines the color profiles that are available for a given printer, for example:

+ +
+Attribute cupsICCProfile "ColorModel.MediaType.Resolution/Description" /path/to/ICC/profile
+
+ +

where "ColorModel.MediaType.Resolution" defines a selector based on the corresponding option selections. A simple driver might only define profiles for the color models that are supported, for example a printer supporting Gray and RGB might use:

+ +
+Attribute cupsICCProfile "Gray../Grayscale Profile" /path/to/ICC/gray-profile
+Attribute cupsICCProfile "RGB../Full Color Profile" /path/to/ICC/rgb-profile
+
+ +

The options used for profile selection can be customized using the cupsICCQualifier2 and cupsICCQualifier3 attributes.

+ +

Since Mac OS X 10.5Custom Color Matching Support

+ +

Mac OS X printer drivers that are based on an existing standard RGB colorspace can tell the system to use the corresponding colorspace instead of an arbitrary ICC color profile when doing color management. The APSupportsCustomColorMatching and APDefaultCustomColorMatchingProfile attributes can be used to enable this mode:

+ +
+Attribute APSupportsCustomColorMatching "" true
+Attribute APDefaultCustomColorMatchingProfile "" sRGB
+
+ + +

Adding Mac OS X Features

+ +

Mac OS X printer drivers can provide additional attributes to specify additional option panes in the print dialog, an image of the printer, a help book, and option presets for the driver software:

+ +
+Attribute APDialogExtension "" /Library/Printers/Vendor/filename.plugin
+Attribute APHelpBook "" /Library/Printers/Vendor/filename.bundle
+Attribute APPrinterIconPath "" /Library/Printers/Vendor/filename.icns
+Attribute APPrinterPreset "name/text" "*option choice ..."
+
+
+ + diff --git a/doc/help/ref-access_log.html b/doc/help/ref-access_log.html new file mode 100644 index 0000000000..015ea17bdb --- /dev/null +++ b/doc/help/ref-access_log.html @@ -0,0 +1,140 @@ + + + + access_log + + + + +

access_log

+ +

The access_log file lists each HTTP resource that +is accessed by a web browser or client. Each line is in an +extended version of the so-called "Common Log Format" used by +many web servers and web reporting tools:

+ +

+ +host group user date-time \"method resource version\" status +bytes ipp-operation ipp-status
+ +
+ +10.0.1.2 - - [01/Dec/2005:21:50:28 +0000] "POST / HTTP/1.1" 200 +317 CUPS-Get-Printers +successful-ok-ignored-or-substituted-attributes
+ +localhost - - [01/Dec/2005:21:50:32 +0000] "GET /admin HTTP/1.1" +200 0 - -
+ +localhost - - [01/Dec/2005:21:50:32 +0000] "POST / HTTP/1.1" 200 +157 CUPS-Get-Printers +successful-ok-ignored-or-substituted-attributes
+ +localhost - - [01/Dec/2005:21:50:32 +0000] "POST / HTTP/1.1" 200 +1411 CUPS-Get-Devices -
+ +localhost - - [01/Dec/2005:21:50:32 +0000] "GET /admin HTTP/1.1" +200 6667 - -
+ +

+ +

The host field will normally only be an IP address +unless you have enabled the HostNameLookups +directive in the cupsd.conf file or if the IP address +corresponds to your local machine.

+ +

The group field always contains "-".

+ +

The user field is the authenticated username of the +requesting user. If no username and password is supplied for the +request then this field contains "-".

+ +

The date-time field is the date and time of the request +in local time and is in the format:

+ +

+[DD/MON/YYYY:HH:MM:SS +ZZZZ] +

+ +

The method field is the HTTP method used: "GET", +"OPTIONS", "PUT", or "POST". "GET" requests are used to get files +from the server, both for the web interface and to get +configuration and log files. "OPTIONS" requests are used to +upgrade connections to TLS encryption. "PUT" requests are used to +upload configuration files. "POST" requests are used for web +interface forms and IPP requests.

+ +

The resource field is the filename of the requested +resource.

+ +

The version field is the HTTP specification version used by the +client. For CUPS clients this will always be "HTTP/1.1".

+ +

The status field contains the HTTP result status of the +request, as follows:

+ +
    + +
  • 200 - Successful operation.
  • + +
  • 201 - File created/modified successfully.
  • + +
  • 304 - The requested file has not changed.
  • + +
  • 400 - Bad HTTP request; typically this + means that you have a malicious program trying to access + your server.
  • + +
  • 401 - Unauthorized, authentication + (username + password) is required.
  • + +
  • 403 - Access is forbidden; typically + this means that a client tried to access a file or + resource they do not have permission to access.
  • + +
  • 404 - The file or resource does not + exist.
  • + +
  • 405 - URL access method is not allowed; + typically this means you have a web browser using your + server as a proxy.
  • + +
  • 413 - Request too large; typically this + means that a client tried to print a file larger than the + MaxRequestSize + allows.
  • + +
  • 426 - Upgrading to TLS-encrypted + connection.
  • + +
  • 500 - Server error; typically this + happens when the server is unable to open/create a file - + consult the error_log file for details.
  • + +
  • 501 - The client requested encryption + but encryption support is not enabled/compiled in.
  • + +
  • 505 - HTTP version number not supported; + typically this means that you have a malicious program + trying to access your server.
  • + +
+ +

The bytes field contains the number of bytes in the +request. For POST requests the bytes field contains the +number of bytes of non-IPP data that is received from the +client.

+ +

The ipp-operation field contains either "-" for non-IPP +requests or the IPP operation name for POST requests containing +an IPP request.

+ +

The ipp-status field contains either "-" for non-IPP +requests or the IPP status code name for POST requests containing +an IPP response.

+ + + diff --git a/doc/help/ref-classes-conf.html b/doc/help/ref-classes-conf.html new file mode 100644 index 0000000000..40bc3d5e27 --- /dev/null +++ b/doc/help/ref-classes-conf.html @@ -0,0 +1,566 @@ + + + + classes.conf + + + + +

classes.conf

+ +

The CUPS scheduler (cupsd) uses the +/etc/cups/classes.conf file to store the list of +available classes. This file contains only locally defined +classes, not remote classes that are created automatically via +browsing. Each directive is listed on a line by itself followed +by its value. Comments are introduced using the number sign ("#") +character at the beginning of a line.

+ +

While the class configuration file consists of plain text and +can be modified using your favorite text editor, you should +normally use the lpadmin(8) +command, web interface, or any of the available GUIs to manage +your classes instead. If you do choose to edit this file +manually, you will need to stop the scheduler first, make your +changes, and then start the scheduler to make them active.

+ + +

Accepting

+ +

Examples

+ +
+<Class name>
+  ...
+  Accepting yes
+</Class>
+
+ +

Description

+ +

The Accepting directive defines the initial state +of the printer-is-accepting-jobs attribute. This state +is also set by the accept(8) and +reject(8) commands:

+ +
+/usr/sbin/accept classname
+/usr/sbin/reject classname
+
+ +

This directive must appear inside a Class or DefaultClass +section.

+ + +

AllowUser

+ +

Examples

+ +
+<Class name>
+  ...
+  AllowUser foo_user
+  AllowUser @bar_group
+</Class>
+
+ +

Description

+ +

The AllowUser directive adds a username or group +name to the requesting-user-name-allowed attribute +which can be set by the lpadmin(8) command:

+ +
+/usr/sbin/lpadmin -p classname -u allow:foo_user,@bar_group
+
+ +

This directive must appear inside a Class or DefaultClass directive. +This directive cannot be used with DenyUser.

+ + +

Class

+ +

Examples

+ +
+<Class name>
+  ...
+</Class>
+
+ +

Description

+ +

The Class directive begins a class definition. +Classes are added using the lpadmin(8) command: + +

+/usr/sbin/lpadmin -p printername -c classname
+
+ + +

DefaultClass

+ +

Examples

+ +
+<DefaultClass name>
+  ...
+</Class>
+
+ +

Description

+ +

The DefaultClass directive begins a class +definition as the default server destination. The default server +destination can be set using the lpadmin(8) +command:

+ +
+/usr/sbin/lpadmin -d classname
+
+ +

Note that the server default destination settings can be +overridden by the user's default destination settings which are +normally set using the lpoptions(1) command.

+ + +

DenyUser

+ +

Examples

+ +
+<Class name>
+  ...
+  DenyUser foo_user
+  DenyUser @bar_group
+</Class>
+
+ +

Description

+ +

The DenyUser directive adds a username or group +name to the requesting-user-name-denied attribute +which can be set by the lpadmin(8) command:

+ +
+/usr/sbin/lpadmin -p classname -u deny:foo_user,@bar_group
+
+ +

This directive must appear inside a Class or DefaultClass directive. +This directive cannot be used with AllowUser

+ + +

Info

+ +

Examples

+ +
+<Class name>
+  ...
+  Info My Class
+</Class>
+
+ +

Description

+ +

The Info directive defines the string for the +printer-info attribute. It is normally set using the +lpadmin(8) command:

+ +
+/usr/sbin/lpadmin -p classname -D "My Class"
+
+ +

This directive must appear inside a Class or DefaultClass +section.

+ + +

JobSheets

+ +

Examples

+ +
+<Class name>
+  ...
+  JobSheets none,standard
+</Class>
+
+ +

Description

+ +

The JobSheets directive specifies the default +banner pages to print before and after a print job. In the above +example, only a standard banner will print after each +job. The lpadmin(8) command is normally used to set +the default banners: + +

+/usr/sbin/lpadmin -p classname -o job-sheets-default=none,standard
+
+ +

If only one banner file is specified, it will be printed +before the files in the job. If a second banner file is +specified, it is printed after the files in the job.

+ +

The available banner pages depend on the local system +configuration; CUPS includes the following standard banner +files:

+ +
    + +
  • none - Do not produce a banner + page.
  • + +
  • classified - A banner page with a + "classified" label at the top and bottom.
  • + +
  • confidential - A banner page with a + "confidential" label at the top and bottom.
  • + +
  • secret - A banner page with a + "secret" label at the top and bottom.
  • + +
  • standard - A banner page with no label + at the top and bottom.
  • + +
  • topsecret - A banner page with a + "top secret" label at the top and bottom.
  • + +
  • unclassified - A banner page with an + "unclassified" label at the top and bottom.
  • + +
+ +

This directive must appear inside a Class or DefaultClass +section.

+ + +

KLimit

+ +

Examples

+ +
+<Class name>
+  ...
+  KLimit 1234
+</Class>
+
+ +

Description

+ +

The KLimit directive defines the value of the +job-k-limit attribute. It is normally set using the +lpadmin(8) command:

+ +
+/usr/sbin/lpadmin -p classname -o job-k-limit=1234
+
+ +

This directive must appear inside a Class or DefaultClass +section.

+ + +

Location

+ +

Examples

+ +
+<Class name>
+  ...
+  Location Building 3321
+</Class>
+
+ +

Description

+ +

The Location directive defines the string for the +printer-location attribute. It is normally set using the +lpadmin(8) command:

+ +
+/usr/sbin/lpadmin -p classname -L "Building 3321"
+
+ +

This directive must appear inside a Class or DefaultClass +section.

+ + +

CUPS 1.2OpPolicy

+ +

Examples

+ +
+<Class name>
+  ...
+  OpPolicy default
+</Class>
+
+ +

Description

+ +

The OpPolicy directive sets the operation policy +that is used for the printer. The lpadmin(8) command +sets the current operation policy:

+ +
+/usr/sbin/lpadmin -p classname -o printer-op-policy=default
+
+ +

The default policy is named "default". All policies correspond +to those defined using the cupsd.conf Policy +section.

+ +

This directive must appear inside a Class or DefaultClass +section.

+ + +

CUPS 1.2Option

+ +

Examples

+ +
+<Class name>
+  ...
+  Option name value
+  Option scaling 100
+  Option page-left 72
+</Class>
+
+ +

Description

+ +

The Option directive specifies a default job +template attribute value. It is mapped to +name-default in the printer attributes and applied +to jobs as name.

+ +

This directive must appear inside a Class or DefaultClass +section.

+ + +

PageLimit

+ +

Examples

+ +
+<Class name>
+  ...
+  PageLimit 1234
+</Class>
+
+ +

Description

+ +

The PageLimit directive defines the value of the +job-page-limit attribute. It can be set using the +lpadmin(8) command:

+ +
+/usr/sbin/lpadmin -p classname -o job-page-limit=1234
+
+ +

This directive must appear inside a Class or DefaultClass +section.

+ + +

Printer

+ +

Examples

+ +
+<Class name>
+  ...
+  Printer foo
+  Printer bar
+  Printer bleep
+</Class>
+
+ +

Description

+ +

The Printer directive adds a printer to a class. +Printers are added to a class using the lpadmin(8) +command:

+ +
+/usr/sbin/lpadmin -p printername -c classname
+
+ +

This directive must appear inside a Class or DefaultClass +section.

+ + +

QuotaPeriod

+ +

Examples

+ +
+<Class name>
+  ...
+  QuotaPeriod 604800
+</Class>
+
+ +

Description

+ +

The QuotaPeriod directive defines the value of +the job-quota-period attribute. Typical values are +86400 (1 day), 604800 (1 week), 2592000 (1 month), and 31536000 +(1 year). It is set using the lpadmin(8) +command:

+ +
+/usr/sbin/lpadmin -p classname -o job-quota-period=604800
+
+ +

This directive must appear inside a Class or DefaultClass +section.

+ + +

CUPS 1.2Shared

+ +

Examples

+ +
+<Class name>
+  ...
+  Shared yes
+</Class>
+
+ +

Description

+ +

The Shared directive defines the initial value of +the printer-is-shared attribute. The strings +yes and no correspond to the true and false +values, respectively. The lpadmin(8) command sets +the current state:

+ +
+/usr/sbin/lpadmin -p classname -o printer-is-shared=true
+
+ +

This directive must appear inside a Class or DefaultClass +section.

+ + +

State

+ +

Examples

+ +
+<Class name>
+  ...
+  State idle
+</Class>
+
+ +

Description

+ +

The State directive defines the initial value of +the printer-state attribute. The strings +idle and stopped correspond to the IPP +enumeration values 3 and 5, respectively. The +cupsenable(8) and cupsdisable(8) +commands set the current state:

+ +
+/usr/sbin/cupsenable classname
+/usr/sbin/cupsdisable classname
+
+ +

This directive must appear inside a Class or DefaultClass +section.

+ + +

StateMessage

+ +

Examples

+ +
+<Class name>
+  ...
+  StateMessage Ready to print.
+</Class>
+
+ +

Description

+ +

The StateMessage directive defines the initial +string for the printer-state-message attribute. The +following are some example messages:

+ +
+StateMessage Connected to host_name...
+StateMessage Connecting to printer_queue on port port_number...
+StateMessage Network host host_name is busy; will retry in 30 seconds...
+StateMessage Class busy; will retry in 10 seconds...
+StateMessage Class is busy; retrying print job...
+StateMessage Print file accepted - job ID id_number.
+StateMessage Ready to print.
+StateMessage Waiting for job to complete
+
+ +

This directive must appear inside a Class or DefaultClass +section.

+ + +

CUPS 1.2StateTime

+ +

Examples

+ +
+<Class name>
+  ...
+  StateTime 1133542425
+</Class>
+
+ +

Description

+ +

The StateTime directive defines the UNIX time +(seconds since Jan 1, 1970) for the last state change of the +queue. It is mapped to the printer-state-change-time +attribute.

+ +

This directive must appear inside a Class or DefaultClass +section.

+ + + + diff --git a/doc/help/ref-client-conf.html b/doc/help/ref-client-conf.html new file mode 100644 index 0000000000..5b4e7e772c --- /dev/null +++ b/doc/help/ref-client-conf.html @@ -0,0 +1,75 @@ + + + + client.conf + + + + +

client.conf

+ +

The /etc/cups/client.conf and +~/.cups/client.conf files contain up to two directives +that determine how the client behaves. Each directive is listed +on a line by itself followed by its value. Comments are +introduced using the number sign ("#") character at the beginning +of a line.

+ +

Encryption

+ +

Examples

+ +
+Encryption Never
+Encryption IfRequested
+Encryption Required
+Encryption Always
+
+ +

Description

+ +

The Encryption directive specifies the default +encryption settings for the client.

+ +

The default setting is IfRequested.

+ + +

CUPS 1.6GSSServiceName

+ +

Examples

+ +
+GSSServiceName http
+GSSServiceName ipp
+
+ +

Description

+ +

The GSSServiceName directive sets the Kerberos service name to use. The default is http for compatibility with Microsoft Windows.

+ + +

ServerName

+ +

Examples

+ +
+ServerName foo.bar.com
+ServerName 11.22.33.44
+ServerName foo.bar.com:8631
+
+ +

Description

+ +

The ServerName directive specifies sets the remote server that is to be used for all client operations. That is, it redirects all client requests to the remote server.

+ +

The default port number is 631 but can be overridden by adding a colon followed by the desired port number to the value.

+ +

The default is to use the local server ("localhost") or domain socket, if so configured.

+ +
Note: +

Only one ServerName directive may appear. If multiple names are +present, only the last one is used. This directive is not supported on Mac OS X 10.7 or later.

+
+ + + diff --git a/doc/help/ref-cupsd-conf.html.in b/doc/help/ref-cupsd-conf.html.in new file mode 100644 index 0000000000..d14d0d7c07 --- /dev/null +++ b/doc/help/ref-cupsd-conf.html.in @@ -0,0 +1,2709 @@ + + + + cupsd.conf + + + + +

cupsd.conf

+ +

The /etc/cups/cupsd.conf file contains +configuration directives that control how the server +functions. Each directive is listed on a line by itself followed +by its value. Comments are introduced using the number sign ("#") +character at the beginning of a line.

+ +

Since the server configuration file consists of plain text, +you can use your favorite text editor to make changes to it. +After making any changes, restart the cupsd(8) +process using the startup script for your operating system:

+ +
    + +
  • AIX, IRIX, Linux, Solaris: +
    +/etc/init.d/cups restart
    +	
  • + +
  • HP-UX: +
    +/sbin/init.d/cups restart
    +	
  • + +
  • MacOS X: +
    +sudo launchctl unload /System/Library/LaunchDaemons/org.cups.cupsd.plist
    +sudo launchctl load /System/Library/LaunchDaemons/org.cups.cupsd.plist
    +	
  • + +
+ +

You can also edit this file from the CUPS web interface, which +automatically handles restarting the scheduler.

+ + +

AccessLog

+ +

Examples

+ +
+AccessLog /var/log/cups/access_log
+AccessLog /var/log/cups/access_log-%s
+AccessLog syslog
+
+ +

Description

+ +

The AccessLog directive sets the name of the +access log file. If the filename is not absolute then it is +assumed to be relative to the ServerRoot directory. The +access log file is stored in "common log format" and can be used +by any web access reporting tool to generate a report on CUPS +server activity.

+ +

The server name can be included in the filename by using +%s in the name.

+ +

The special name "syslog" can be used to send the access +information to the system log instead of a plain file.

+ +

The default access log file is +@CUPS_LOGDIR@/access_log.

+ + +

AccessLogLevel

+ +

Examples

+ +
+AccessLogLevel config
+AccessLogLevel actions
+AccessLogLevel all
+
+ +

Description

+ +

The AccessLogLevel directive controls which requests are logged +to the access log file. The following levels are defined:

+ +
    + +
  • config; Log when printers and classes are added, + deleted, or modified and when configuration files are accessed or + updated.
  • + +
  • actions; Log when print jobs are submitted, + held, released, modified, or canceled, and any of the conditions + for config.
  • + +
  • all; Log all requests.
  • + +
+ +

The default access log level is @CUPS_ACCESS_LOG_LEVEL@.

+ + +

Allow

+ +

Examples

+ +
+<Location /path>
+  ...
+  Allow from All
+  Allow from None
+  Allow from *.example.com
+  Allow from .example.com
+  Allow from host.example.com
+  Allow from nnn.*
+  Allow from nnn.nnn.*
+  Allow from nnn.nnn.nnn.*
+  Allow from nnn.nnn.nnn.nnn
+  Allow from nnn.nnn.nnn.nnn/mm
+  Allow from nnn.nnn.nnn.nnn/mmm.mmm.mmm.mmm
+  Allow from [xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:xxxx]
+  Allow from [xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:xxxx]/mmm
+  Allow from @LOCAL
+  Allow from @IF(name)
+</Location>
+
+ +

Description

+ +

The Allow directive specifies a hostname, IP +address, or network that is allowed access to the server. +Allow directives are cumulative, so multiple +Allow directives can be used to allow access for +multiple hosts or networks.

+ +

Host and domain name matching require that you enable the HostNameLookups +directive.

+ +

The /mm notation specifies a CIDR netmask, as shown in +Table 1.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 1: CIDR Netmasks
mmnetmaskmmnetmask
00.0.0.08255.0.0.0
1128.0.0.016255.255.0.0
2192.0.0.024255.255.255.0
......32255.255.255.255
+ +

The @LOCAL name will allow access from all local +interfaces. The @IF(name) name will allow access +from the named interface. In both cases, CUPS only allows access +from the network that the interface(s) are configured for - +requests arriving on the interface from a foreign network will +not be accepted.

+ +

The Allow directive must appear inside a Location or Limit section.

+ + +

DeprecatedAuthClass

+ +

Examples

+ +
+<Location /path>
+  ...
+  AuthClass Anonymous
+  AuthClass User
+  AuthClass System
+  AuthClass Group
+</Location>
+
+ +

Description

+ +

The AuthClass directive defines what level of +authentication is required:

+ +
    + +
  • Anonymous - No authentication should be + performed (default)
  • + +
  • User - A valid username and password is + required
  • + +
  • System - A valid username and password + is required, and the username must belong to the "sys" + group; this can be changed using the SystemGroup + directive
  • + +
  • Group - A valid username and password is + required, and the username must belong to the group named + by the AuthGroupName + directive
  • + +
+ +

The AuthClass directive must appear inside a Location or Limit section.

+ +

This directive is deprecated and will be removed from a +future release of CUPS. Consider using the more flexible Require directive instead.

+ + +

DeprecatedAuthGroupName

+ +

Examples

+ +
+<Location /path>
+  ...
+  AuthGroupName mygroup
+  AuthGroupName lp
+</Location>
+
+ +

Description

+ +

The AuthGroupName directive sets the group to use +for Group authentication.

+ +

The AuthGroupName directive must appear inside a +Location or Limit section.

+ +

This directive is deprecated and will be removed from a +future release of CUPS. Consider using the more flexible Require directive instead.

+ + +

AuthType

+ +

Examples

+ +
+<Location /path>
+  ...
+  AuthType None
+  AuthType Basic
+  AuthType Digest
+  AuthType BasicDigest
+  AuthType Negotiate
+</Location>
+
+ +

Description

+ +

The AuthType directive defines the type of +authentication to perform:

+ +
    + +
  • None - No authentication should be + performed (default)
  • + +
  • Basic - Basic authentication should be + performed using the UNIX password and group files
  • + +
  • Digest - Digest authentication should be + performed using the /etc/cups/passwd.md5 + file
  • + +
  • BasicDigest - Basic authentication + should be performed using the + /etc/cups/passwd.md5 file
  • + +
  • Negotiate - Kerberos authentication + should be performed
  • + +
+ +

When using Basic, Digest, +BasicDigest, or Negotiate authentication, +clients connecting through the localhost interface can +also authenticate using certificates.

+ +

The AuthType directive must appear inside a Location or Limit section.

+ + +

AutoPurgeJobs

+ +

Examples

+ +
+AutoPurgeJobs Yes
+AutoPurgeJobs No
+
+ +

Description

+ +

The AutoPurgeJobs directive specifies whether or +not to purge completed jobs once they are no longer required for +quotas. This option has no effect if quotas are not enabled. The +default setting is No.

+ + +

CUPS 1.2/Mac OS X 10.5BrowseLocalProtocols

+ +

Examples

+ +
+BrowseLocalProtocols all
+BrowseLocalProtocols none
+BrowseLocalProtocols dnssd
+
+ +

Description

+ +

The BrowseLocalProtocols directive specifies the +protocols to use when advertising local shared printers on the +network. Multiple protocols can be specified by separating them +with spaces. The default is "@CUPS_BROWSE_LOCAL_PROTOCOLS@".

+ + +

BrowseWebIF

+ +

Examples

+ +
+BrowseWebIF On
+BrowseWebIF Off
+
+ +

Description

+ +

The BrowseWebIF directive controls whether the CUPS web +interface is advertised via DNS-SD. The default setting is +Off.

+ + +

Browsing

+ +

Examples

+ +
+Browsing On
+Browsing Off
+
+ +

Description

+ +

The Browsing directive controls whether or not +printer sharing is enabled. The default setting is +@CUPS_BROWSING@.

+ + +

CUPS 1.1.7Classification

+ +

Examples

+ +
+Classification
+Classification classified
+Classification confidential
+Classification secret
+Classification topsecret
+Classification unclassified
+
+ +

Description

+ +

The Classification directive sets the +classification level on the server. When this option is set, at +least one of the banner pages is forced to the classification +level, and the classification is placed on each page of output. +The default is no classification level.

+ + +

CUPS 1.1.10ClassifyOverride

+ +

Examples

+ +
+ClassifyOverride Yes
+ClassifyOverride No
+
+ +

Description

+ +

The ClassifyOverride directive specifies whether +users can override the default classification level on the +server. When the server classification is set, users can change +the classification using the job-sheets option and +can choose to only print one security banner before or after the +job. If the job-sheets option is set to +none then the server default classification is +used.

+ +

The default is to not allow classification overrides.

+ + +

CUPS 1.1.15ConfigFilePerm

+ +

Examples

+ +
+ConfigFilePerm 0644
+ConfigFilePerm 0640
+
+ +

Description

+ +

The ConfigFilePerm directive specifies the permissions to use when the scheduler writes configuration and cache files, typically in response to IPP or HTTP requests. The default is @CUPS_CONFIG_FILE_PERM@.

+ +
Note: + +

The permissions for the printers.conf file are always masked to only allow access from the scheduler user (typically root). This is done because printer device URIs sometimes contain sensitive authentication information that should not be generally known on the system. There is no way to disable this security feature.

+ +
+ + +

DataDir

+ +

Examples

+ +
+DataDir /usr/share/cups
+
+ +

Description

+ +

The DataDir directive sets the directory to use +for data files.

+ + +

CUPS 1.2/Mac OS X 10.5DefaultAuthType

+ +

Examples

+ +
+DefaultAuthType Basic
+DefaultAuthType BasicDigest
+DefaultAuthType Digest
+DefaultAuthType Negotiate
+
+ +

Description

+ +

The DefaultAuthType directive specifies the type +of authentication to use for IPP operations that require a +username. The default is Basic.

+ + +

CUPS 1.2/Mac OS X 10.5DefaultEncryption

+ +

Examples

+ +
+DefaultEncryption Never
+DefaultEncryption IfRequested
+DefaultEncryption Required
+
+ +

Description

+ +

The DefaultEncryption directive specifies the +type of encryption to use when performing authentication. The +default is Required.

+ + +

DefaultLanguage

+ +

Examples

+ +
+DefaultLanguage de
+DefaultLanguage en
+DefaultLanguage es
+DefaultLanguage fr
+DefaultLanguage it
+
+ +

Description

+ +

The DefaultLanguage directive specifies the +default language to use for client connections. Setting the +default language also sets the default character set if a +language localization file exists for it. The default language +is "en" for English.

+ + +

CUPS 1.4/Mac OS X 10.6DefaultPaperSize

+ +

Examples

+ +
+DefaultPaperSize Letter
+DefaultPaperSize A4
+DefaultPaperSize Auto
+DefaultPaperSize None
+
+ +

Description

+ +

The DefaultPaperSize directive specifies the default paper +size to use when creating new printers. The default is Auto +which uses a paper size appropriate for the system default locale. A value +of None tells the scheduler to not set the default paper +size.

+ + +

CUPS 1.2/Mac OS X 10.5DefaultPolicy

+ +

Examples

+ +
+DefaultPolicy default
+DefaultPolicy authenticated
+DefaultPolicy foo
+
+ +

Description

+ +

The DefaultPolicy directive specifies the default +policy to use for IPP operation. The default is +default. CUPS also includes a policy called +authenticated that requires a username and password for printing +and other job operations.

+ + +

CUPS 1.2/Mac OS X 10.5DefaultShared

+ +

Examples

+ +
+DefaultShared yes
+DefaultShared no
+
+ +

Description

+ +

The DefaultShared directive specifies whether +printers are shared (published) by default. The default is +@CUPS_DEFAULT_SHARED@.

+ + +

Deny

+ +

Examples

+ +
+<Location /path>
+  ..
+  Deny from All
+  Deny from None
+  Deny from *.example.com
+  Deny from .example.com
+  Deny from host.example.com
+  Deny from nnn.*
+  Deny from nnn.nnn.*
+  Deny from nnn.nnn.nnn.*
+  Deny from nnn.nnn.nnn.nnn
+  Deny from nnn.nnn.nnn.nnn/mm
+  Deny from nnn.nnn.nnn.nnn/mmm.mmm.mmm.mmm
+  Deny from [xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:xxxx]
+  Deny from [xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:xxxx]/mmm
+  Deny from @LOCAL
+  Deny from @IF(name)
+</Location>
+
+ +

Description

+ +

The Deny directive specifies a hostname, IP +address, or network that is denied access to the server. +Deny directives are cumulative, so multiple +Deny directives can be used to deny access for +multiple hosts or networks.

+ +

Host and domain name matching require that you enable the HostNameLookups +directive.

+ +

The /mm notation specifies a CIDR netmask, a shown in +Table 1.

+ +

The @LOCAL name will deny access from all local +interfaces. The @IF(name) name will deny access from +the named interface. In both cases, CUPS only denies access from +the network that the interface(s) are configured for - requests +arriving on the interface from a foreign network will +not be denied.

+ +

The Deny directive must appear inside a Location or Limit section.

+ + +

CUPS 1.4/Mac OS X 10.6DirtyCleanInterval

+ +

Examples

+ +
+DirtyCleanInterval 30
+DirtyCleanInterval 0
+
+ +

Description

+ +

The DirtyCleanInterval directive specifies the number of +seconds to wait before updating configuration and state files for printers, +classes, subscriptions, and jobs. The default is 30 seconds. A value of 0 +causes the update to occur as soon as possible, typically within a few +milliseconds.

+ + +

DocumentRoot

+ +

Examples

+ +
+DocumentRoot /usr/share/doc/cups
+DocumentRoot /foo/bar/doc/cups
+
+ +

Description

+ +

The DocumentRoot directive specifies the location +of web content for the HTTP server in CUPS. If an absolute path +is not specified then it is assumed to be relative to the ServerRoot directory. The +default directory is @CUPS_DOCROOT@.

+ +

Documents are first looked up in a sub-directory for the +primary language requested by the client (e.g. +@CUPS_DOCROOT@/fr/...) and then directly under +the DocumentRoot directory (e.g. +@CUPS_DOCROOT@/...), so it is possible to +localize the web content by providing subdirectories for each +language needed.

+ + +

Encryption

+ +

Examples

+ +
+<Location /path>
+  ...
+  Encryption Never
+  Encryption IfRequested
+  Encryption Required
+</Location>
+
+ +

Description

+ +

The Encryption directive must appear instead a Location or Limit section and specifies the +encryption settings for that location. The default setting is +IfRequested for all locations.

+ + +

ErrorLog

+ +

Examples

+ +
+ErrorLog /var/log/cups/error_log
+ErrorLog /var/log/cups/error_log-%s
+ErrorLog syslog
+
+ +

Description

+ +

The ErrorLog directive sets the name of the error +log file. If the filename is not absolute then it is assumed to +be relative to the ServerRoot directory. The +default error log file is @CUPS_LOGDIR@/error_log.

+ +

The server name can be included in the filename by using +%s in the name.

+ +

The special name "syslog" can be used to send the error +information to the system log instead of a plain file.

+ + +

CUPS 1.3/Mac OS X 10.5ErrorPolicy

+ +

Examples

+ +
+ErrorPolicy abort-job
+ErrorPolicy retry-job
+ErrorPolicy stop-printer
+
+ +

Description

+ +

The ErrorPolicy directive defines the default policy that +is used when a backend is unable to send a print job to the +printer.

+ +

The following values are supported:

+ +
    + +
  • abort-job - Abort the job and proceed + with the next job in the queue
  • + +
  • retry-job - Retry the job after waiting + for N seconds; the cupsd.conf JobRetryInterval + directive controls the value of N
  • + +
  • retry-this-job - Retry the current job immediately + and indefinitely.
  • + +
  • stop-printer - Stop the printer and keep + the job for future printing; this is the default + value
  • + +
+ + + +

CUPS 1.4/Mac OS X 10.6FatalErrors

+ +

Examples

+ +
+FatalErrors none
+FatalErrors all
+FatalErrors browse
+FatalErrors config
+FatalErrors listen
+FatalErrors log
+FatalErrors permissions
+FatalErrors all -permissions
+FatalErrors config permissions log
+
+ +

Description

+ +

The FatalErrors directive determines whether certain kinds of +errors are fatal. The following kinds of errors are currently recognized:

+ +
    + +
  • none - No errors are fatal
  • + +
  • all - All of the errors below are fatal
  • + +
  • browse - Browsing initialization errors are fatal, + for example failed binding to the CUPS browse port or failed connections + to LDAP servers
  • + +
  • config - Configuration file syntax errors are + fatal
  • + +
  • listen - Listen or Port errors are fatal, except for + IPv6 failures on the loopback or "any" addresses
  • + +
  • log - Log file creation or write errors are fatal
  • + +
  • permissions - Bad startup file permissions are + fatal, for example shared SSL certificate and key files with world- + read permissions
  • + +
+ +

Multiple errors can be listed, and the form "-kind" can be used with +all to remove specific kinds of errors. The default setting is +@CUPS_FATAL_ERRORS@.

+ + +

CUPS 1.1.18FileDevice

+ +

Examples

+ +
+FileDevice Yes
+FileDevice No
+
+ +

Description

+ +

The FileDevice directive determines whether the +scheduler allows new printers to be added using device URIs of +the form file:/filename. File devices are most often +used to test new printer drivers and do not support raw file +printing.

+ +

The default setting is No.

+ +
Note: + +

File devices are managed by the scheduler. Since the +scheduler normally runs as the root user, file devices +can be used to overwrite system files and potentially +gain unauthorized access to the system. If you must +create printers using file devices, we recommend that +you set the FileDevice directive to +Yes for only as long as you need to add the +printers to the system, and then reset the directive to +No.

+ +
+ + +

CUPS 1.1.3FilterLimit

+ +

Examples

+ +
+FilterLimit 0
+FilterLimit 200
+FilterLimit 1000
+
+ +

Description

+ +

The FilterLimit directive sets the maximum cost +of all running job filters. It can be used to limit the number of +filter programs that are run on a server to minimize disk, +memory, and CPU resource problems. A limit of 0 disables filter +limiting.

+ +

An average print to a non-PostScript printer needs a filter +limit of about 200. A PostScript printer needs about half that +(100). Setting the limit below these thresholds will effectively +limit the scheduler to printing a single job at any time.

+ +

The default limit is 0.

+ + +

CUPS 1.1.16FilterNice

+ +

Examples

+ +
+FilterNice 0
+FilterNice 10
+FilterNice 19
+
+ +

Description

+ +

The FilterNice directive sets the nice(1) +value to assign to filter processes. The nice value ranges from +0, the highest priority, to 19, the lowest priority. The default +is 0.

+ + +

CUPS 1.1.3FontPath

+ +

Examples

+ +
+FontPath /foo/bar/fonts
+FontPath /usr/share/cups/fonts:/foo/bar/fonts
+
+ +

Description

+ +

The FontPath directive specifies the font path to +use when searching for fonts. The default font path is +/usr/share/cups/fonts.

+ + +

Group

+ +

Examples

+ +
+Group lp
+Group nobody
+
+ +

Description

+ +

The Group directive specifies the UNIX group that +filter and CGI programs run as. The default group is +system-specific but is usually lp or +nobody.

+ + +

CUPS 1.6GSSServiceName

+ +

Examples

+ +
+GSSServiceName http
+GSSServiceName ipp
+
+ +

Description

+ +

The GSSServiceName directive sets the Kerberos service name to use. The default is @CUPS_DEFAULT_GSSSERVICE_NAME@ for compatibility with Microsoft Windows.

+ + +

HostNameLookups

+ +

Examples

+ +
+HostNameLookups On
+HostNameLookups Off
+HostNameLookups Double
+
+ +

Description

+ +

The HostNameLookups directive controls whether or +not CUPS looks up the hostname for connecting clients. The +Double setting causes CUPS to verify that the +hostname resolved from the address matches one of the addresses +returned for that hostname. Double lookups also +prevent clients with unregistered addresses from connecting to +your server.

+ +

The default is Off to avoid the potential server +performance problems with hostname lookups. Set this option to +On or Double only if absolutely +required.

+ + +

CUPS 1.1.9Include

+ +

Examples

+ +
+Include filename
+Include /foo/bar/filename
+
+ +

Description

+ +

The Include directive includes the named file in +the cupsd.conf file. If no leading path is provided, +the file is assumed to be relative to the ServerRoot directory.

+ + +

CUPS 1.5JobPrivateAccess

+ +

Examples

+ +
+JobPrivateAccess all
+JobPrivateAccess default
+JobPrivateAccess {user|@group|@ACL|@OWNER|@SYSTEM}+
+
+ +

Description

+ +

The JobPrivateAccess directive specifies the access list for a +job's private values. The "default" access list is "@OWNER @SYSTEM". "@ACL" maps +to the printer's requesting-user-name-allowed or requesting-user-name-denied +values.

+ +

The JobPrivateAccess directive must appear inside a Policy section.

+ + +

CUPS 1.5JobPrivateValues

+ +

Examples

+ +
+JobPrivateValues all
+JobPrivateValues default
+JobPrivateValues none
+JobPrivateValues attribute-name-1 [ ... attribute-name-N ]
+
+ +

Description

+ +

The JobPrivateValues directive specifies the list of job values +to make private. The "default" values are "job-name", +"job-originating-host-name", and "job-originating-user-name".

+ +

The JobPrivateValues directive must appear inside a Policy section.

+ + +

CUPS 1.2/Mac OS X 10.5JobRetryInterval

+ +

Examples

+ +
+JobRetryInterval 30
+JobRetryInterval 120
+
+ +

Description

+ +

The JobRetryInterval directive specifies the +number of seconds to wait before retrying a job. This is +typically used for fax queues but can also be used with normal +print queues whose error policy is retry-job. The +default is 30 seconds.

+ + +

CUPS 1.4/Mac OS X 10.6JobKillDelay

+ +

Examples

+ +
+JobKillDelay 30
+JobKillDelay 120
+
+ +

Description

+ +

The JobKillDelay directive specifies the number of seconds to +wait before killing the filters and backend associated with a canceled or held +job. The default is 30 seconds.

+ + +

CUPS 1.2/Mac OS X 10.5JobRetryLimit

+ +

Examples

+ +
+JobRetryLimit 5
+JobRetryLimit 50
+
+ +

Description

+ +

The JobRetryLimit directive specifies the maximum +number of times the scheduler will try to print a job. This is +typically used for fax queues but can also be used with normal +print queues whose error policy is retry-job. The +default is 5 times.

+ + +

KeepAlive

+ +

Examples

+ +
+KeepAlive On
+KeepAlive Off
+
+ +

Description

+ +

The KeepAlive directive controls whether or not +to support persistent HTTP connections. The default is +On.

+ +

HTTP/1.1 clients automatically support persistent connections, +while HTTP/1.0 clients must specifically request them using the +Keep-Alive attribute in the Connection: +field of each request.

+ + +

KeepAliveTimeout

+ +

Examples

+ +
+KeepAliveTimeout 60
+KeepAliveTimeout 30
+
+ +

Description

+ +

The KeepAliveTimeout directive controls how long +a persistent HTTP connection will remain open after the last +request. The default is 30 seconds.

+ + +

CUPS 1.1.7Limit (Location)

+ +

Examples

+ +
+<Location /path>
+  <Limit GET POST>
+  ...
+  </Limit>
+
+  <Limit ALL>
+  ...
+  </Limit>
+</Location>
+
+ +

Description

+ +

The Limit directive groups access control +directives for specific types of HTTP requests and must appear +inside a Location section. +Access can be limited for individual request types +(DELETE, GET, HEAD, +OPTIONS, POST, PUT, and +TRACE) or for all request types (ALL). +The request type names are case-sensitive for compatibility with +Apache.

+ + +

CUPS 1.2/Mac OS X 10.5Limit (Policy)

+ +

Examples

+ +
+<Policy name>
+  <Limit CUPS-Add-Modify-Printer CUPS-Delete-Printer>
+  ...
+  </Limit>
+
+  <Limit All>
+  ...
+  </Limit>
+</Policy>
+
+ +

Description

+ +

When included in Policy +sections, the Limit directive groups access control +directives for specific IPP operations. Multiple operations can +be listed, separated by spaces. Table 2 lists the supported +operations.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 2: Supported IPP Operations
Operation NameDescription
AllAll operations - used as the default limit for + operations that are not listed
Cancel-JobCancel a job
Cancel-SubscriptionCancel a subscription
Create-JobCreate a new, empty job
Create-Job-SubscriptionCreates a notification subscription on a job
Create-Printer-SubscriptionCreates a notification subscription on a printer
CUPS-Accept-JobsSets the printer-is-accepting-jobs value for a printer to true
CUPS-Add-Modify-ClassAdds or modifies a class
CUPS-Add-Modify-PrinterAdds or modifies a printer
CUPS-Authenticate-JobAuthenticates a job for printing
CUPS-Delete-ClassDeletes a class
CUPS-Delete-PrinterDeletes a printer
CUPS-Get-ClassesGets a list of classes
CUPS-Get-DefaultGets the (network/server) default printer or class
CUPS-Get-DevicesGets a list of available devices
CUPS-Get-PPDsGets a list of available manufacturers or drivers
CUPS-Get-PrintersGets a list of printers and/or classes
CUPS-Move-JobMoves a job to a new destination
CUPS-Reject-JobsSets the printer-is-accepting-jobs value for a printer to false
CUPS-Set-DefaultSets the network/server default printer or class
Disable-PrinterSets the printer-state value for a printer to stopped
Enable-PrinterSets the printer-state value for a printer to idle/processing
Get-Job-AttributesGets information about a job
Get-JobsGets a list of jobs
Get-NotificationsGets a list of events
Get-Printer-AttributesGets information about a printer or class
Get-Subscription-AttributesGets information about a notification subscription
Get-SubscriptionsGets a list of notification subscriptions
Hold-JobHolds a job for printing
Pause-PrinterSets the printer-state value for a printer to stopped
Print-JobCreates a job with a single file for printing
Purge-JobsRemoves all jobs from a printer
Release-JobReleases a previously held job for printing
Renew-SubscriptionRenews a notification subscription
Restart-JobReprints a job
Resume-PrinterSets the printer-state value for a printer to idle/processing
Send-DocumentAdds a file to an job created with Create-Job
Set-Job-AttributesChanges job options
Validate-JobValidates job options prior to printing
+ + +

CUPS 1.1.7LimitExcept

+ +

Examples

+ +
+<Location /path>
+  <LimitExcept GET POST>
+  ...
+  </LimitExcept>
+</Location>
+
+ +

Description

+ +

The LimitExcept directive groups access control +directives for specific types of HTTP requests and must appear +inside a Location section. +Unlike the Limit directive, +LimitExcept restricts access for all requests +except those listed on the LimitExcept +line.

+ + +

LimitRequestBody

+ +

Examples

+ +
+LimitRequestBody 10485760
+LimitRequestBody 10m
+LimitRequestBody 0
+
+ +

Description

+ +

The LimitRequestBody directive controls the +maximum size of print files, IPP requests, and HTML form data in +HTTP POST requests. The default limit is 0 which disables the +limit check.

+ + +

Listen

+ +

Examples

+ +
+Listen 127.0.0.1:631
+Listen 192.0.2.1:631
+Listen [::1]:631
+Listen *:631
+
+ +

Description

+ +

The Listen directive specifies a network address +and port to listen for connections. Multiple Listen +directives can be provided to listen on multiple addresses.

+ +

The Listen directive is similar to the Port directive but allows you to +restrict access to specific interfaces or networks.

+ + +

CUPS 1.1.7ListenBackLog

+ +

Examples

+ +
+ListenBackLog 5
+ListenBackLog 10
+
+ +

Description

+ +

The ListenBackLog directive sets the maximum +number of pending connections the scheduler will allow. This +normally only affects very busy servers that have reached the MaxClients limit, but can +also be triggered by large numbers of simultaneous connections. +When the limit is reached, the operating system will refuse +additional connections until the scheduler can accept the pending +ones. The default is the OS-defined default limit, typically +either 5 for older operating systems or 128 for newer operating +systems.

+ + +

Location

+ +

Examples

+ +
+<Location />
+...
+</Location>
+
+<Location /admin>
+...
+</Location>
+
+<Location /admin/conf>
+...
+</Location>
+
+<Location /admin/log>
+...
+</Location>
+
+<Location /classes>
+...
+</Location>
+
+<Location /classes/name>
+...
+</Location>
+
+<Location /jobs>
+...
+</Location>
+
+<Location /printers>
+...
+</Location>
+
+<Location /printers/name>
+...
+</Location>
+
+
+ +

Description

+ +

The Location directive specifies access control +and authentication options for the specified HTTP resource or +path. The Allow, AuthType, Deny, Encryption, Limit, LimitExcept, Order, Require, and Satisfy directives may all +appear inside a location.

+ +

Note that more specific resources override the less specific +ones. So the directives inside the /printers/name +location will override ones from /printers. +Directives inside /printers will override ones from +/. None of the directives are inherited.

+ +
+ + + + + + + + + + + + + + + + + +
Table 3: Common Locations on the Server
LocationDescription
/The path for all get operations (get-printers, get-jobs, etc.)
/adminThe path for all administration operations (add-printer, delete-printer, start-printer, etc.)
/admin/confThe path for access to the CUPS configuration files (cupsd.conf, client.conf, etc.)
/admin/logThe path for access to the CUPS log files (access_log, error_log, page_log)
/classesThe path for all classes
/classes/nameThe resource for class name
/jobsThe path for all jobs (hold-job, release-job, etc.)
/jobs/idThe resource for job id
/printersThe path for all printers
/printers/nameThe path for printer name
/printers/name.ppdThe PPD file path for printer name
+ + +

LogDebugHistory

+ +

Examples

+ +
+LogDebugHistory 0
+LogDebugHistory 200
+
+ +

Description

+ +

When LogLevel is not set to +debug or debug2, the LogDebugHistory +directive specifies the number of debugging messages that are logged when an +error occurs during printing. The default is 200 messages. A value of 0 +disables debugging history entirely and is not recommended.

+ + +

CUPS 1.1.15LogFilePerm

+ +

Examples

+ +
+LogFilePerm 0644
+LogFilePerm 0600
+
+ +

Description

+ +

The LogFilePerm directive specifies the +permissions to use when writing log files. The default +is @CUPS_LOG_FILE_PERM@.

+ + +

LogLevel

+ +

Examples

+ +
+LogLevel none
+LogLevel emerg
+LogLevel alert
+LogLevel crit
+LogLevel error
+LogLevel warn
+LogLevel notice
+LogLevel info
+LogLevel debug
+LogLevel debug2
+
+ +

Description

+ +

The LogLevel directive specifies the level of +logging for the ErrorLog +file. The following values are recognized (each level logs +everything under the preceding levels):

+ +
    + +
  • none - Log nothing
  • + +
  • emerg - Log emergency conditions that + prevent the server from running
  • + +
  • alert - Log alerts that must be handled + immediately
  • + +
  • crit - Log critical errors that don't + prevent the server from running
  • + +
  • error - Log general errors
  • + +
  • warn - Log errors and warnings
  • + +
  • notice - Log temporary error conditions
  • + +
  • info - Log all requests and state + changes
  • + +
  • debug - Log basic debugging + information
  • + +
  • debug2 - Log all debugging + information
  • + +
+ +

The default LogLevel is @CUPS_LOG_LEVEL@.

+ + +

LogTimeFormat

+ +

Examples

+ +
+LogTimeFormat standard
+LogTimeFormat usecs
+
+ +

Description

+ +

The LogTimeFormat directive specifies the format used for the +date and time in the log files. Standard uses the standard Apache +Common Log Format date and time while usecs adds microseconds. +The default is standard.

+ + +

MaxClients

+ +

Examples

+ +
+MaxClients 100
+MaxClients 1024
+
+ +

Description

+ +

The MaxClients directive controls the maximum +number of simultaneous clients that will be allowed by the +server. The default is 100 clients.

+ +
Note: + +

Since each print job requires a file descriptor for the status +pipe, the scheduler internally limits the MaxClients +value to 1/3 of the available file descriptors to avoid possible +problems when printing large numbers of jobs.

+ +
+ + +

CUPS 1.1.18MaxClientsPerHost

+ +

Examples

+ +
+MaxClientsPerHost 10
+
+ +

Description

+ +

The MaxClientsPerHost directive controls the +maximum number of simultaneous clients that will be allowed from +a single host by the server. The default is the +MaxClients value.

+ +

This directive provides a small measure of protection against +Denial of Service attacks from a single host.

+ + +

CUPS 1.1.16MaxCopies

+ +

Examples

+ +
+MaxCopies 100
+MaxCopies 65535
+
+ +

Description

+ +

The MaxCopies directive controls the maximum +number of copies that a user can print of a job. The default is +@CUPS_MAX_COPIES@ copies.

+ +
Note: + +

Most HP PCL laser printers internally limit the number of +copies to 100.

+ +
+ + + +

MaxJobs

+ +

Examples

+ +
+MaxJobs 100
+MaxJobs 9999
+MaxJobs 0
+
+ +

Description

+ +

The MaxJobs directive controls the maximum number +of jobs that are kept in memory. Once the number of jobs reaches +the limit, the oldest completed job is automatically purged from +the system to make room for the new one. If all of the known jobs +are still pending or active then the new job will be +rejected.

+ +

Setting the maximum size to 0 disables this functionality. The +default setting is 500.

+ + +

CUPS 1.1.7MaxJobsPerPrinter

+ +

Examples

+ +
+MaxJobsPerPrinter 100
+MaxJobsPerPrinter 9999
+MaxJobsPerPrinter 0
+
+ +

Description

+ +

The MaxJobsPerPrinter directive controls the +maximum number of active jobs that are allowed for each printer +or class. Once a printer or class reaches the limit, new jobs +will be rejected until one of the active jobs is completed, +stopped, aborted, or canceled.

+ +

Setting the maximum to 0 disables this functionality. The +default setting is 0.

+ + +

CUPS 1.1.7MaxJobsPerUser

+ +

Examples

+ +
+MaxJobsPerUser 100
+MaxJobsPerUser 9999
+MaxJobsPerUser 0
+
+ +

Description

+ +

The MaxJobsPerUser directive controls the maximum +number of active jobs that are allowed for each user. Once a user +reaches the limit, new jobs will be rejected until one of the +active jobs is completed, stopped, aborted, or canceled.

+ +

Setting the maximum to 0 disables this functionality. The +default setting is 0.

+ + +

CUPS 1.6MaxJobTime

+ +

Examples

+ +
+MaxJobTime 10800
+MaxJobTime 0
+
+ +

Description

+ +

The MaxJobTime directive controls the maximum number of +seconds allowed for a job to complete printing before it is considered "stuck". +The job is canceled automatically if it takes longer than the specified number of seconds to complete.

+ +

Setting the maximum time to 0 disables this functionality. The default +setting is 10800 seconds (3 hours).

+ + +

MaxLogSize

+ +

Examples

+ +
+MaxLogSize 1048576
+MaxLogSize 1m
+MaxLogSize 0
+
+ +

Description

+ +

The MaxLogSize directive controls the maximum +size of each log file. Once a log file reaches or exceeds the +maximum size it is closed and renamed to filename.O. +This allows you to rotate the logs automatically. The default +size is 1048576 bytes (1MB).

+ +

Setting the maximum size to 0 disables log rotation.

+ + +

DeprecatedMaxRequestSize

+ +

Examples

+ +
+MaxRequestSize 10485760
+MaxRequestSize 10m
+MaxRequestSize 0
+
+ +

Description

+ +

The MaxRequestSize directive controls the maximum +size of print files, IPP requests, and HTML form data in HTTP +POST requests. The default limit is 0 which disables the limit +check.

+ +

This directive is deprecated and will be removed in a +future CUPS release. Use the LimitRequestBody +directive instead.

+ + +

CUPS 1.4/Mac OS X 10.6MultipleOperationTimeout

+ +

Examples

+ +
+MultipleOperationTimeout 60
+MultipleOperationTimeout 300
+MultipleOperationTimeout 86400
+
+ +

Description

+ +

The MultipleOperationTimeout directive sets the maximum amount +of time between files in a multi-file print job. The default is 300 seconds.

+ + +

Order

+ +

Examples

+ +
+<Location /path>
+  ...
+  Order Allow,Deny
+  Order Deny,Allow
+</Location>
+
+ +

Description

+ +

The Order directive defines the default access +control. The following values are supported:

+ +
    + +
  • allow,deny - Deny requests by default, + then check the Allow + lines followed by the Deny lines
  • + +
  • deny,allow - Allow requests by default, + then check the Deny + lines followed by the Allow lines
  • + +
+ +

The Order directive must appear inside a Location or Limit section.

+ + +

PageLog

+ +

Examples

+ +
+PageLog /var/log/cups/page_log
+PageLog /var/log/cups/page_log-%s
+PageLog syslog
+
+ +

Description

+ +

The PageLog directive sets the name of the page +log file. If the filename is not absolute then it is assumed to +be relative to the ServerRoot directory. The +default page log file is @CUPS_LOGDIR@/page_log.

+ +

The server name can be included in the filename by using +%s in the name.

+ +

The special name "syslog" can be used to send the page +information to the system log instead of a plain file.

+ + +

PageLogFormat

+ +

Examples

+ +
+PageLogFormat %p %u %j %T %P %C %{job-billing} %{job-originating-host-name} %{job-name} %{media} %{sides}
+PageLogFormat PAGE %p %u %j %P %C %{job-billing} %{job-originating-host-name}
+
+ +

Description

+ +

The PageLogFormat directive sets the format of lines +that are logged to the page log file. Sequences beginning with percent (%) +characters are replaced with the corresponding information, while all other +characters are copied literally. The following percent sequences are +recognized:

+ +
    + +
  • %%: Inserts a single percent character.
  • + +
  • %{name}: Inserts the value of the specified IPP + attribute.
  • + +
  • %C: Inserts the number of copies for the current page.
  • + +
  • %P: Inserts the current page number.
  • + +
  • %T: Inserts the current date and time in common log + format.
  • + +
  • %j: Inserts the job ID.
  • + +
  • %p: Inserts the printer name.
  • + +
  • %u: Inserts the username.
  • + +
+ +

The default is "%p %u %j %T %P %C %{job-billing} %{job-originating-host-name} %{job-name} %{media} %{sides}".

+ + +

CUPS 1.2/Mac OS X 10.5PassEnv

+ +

Examples

+ +
+PassEnv MY_ENV_VARIABLE
+
+ +

Description

+ +

The PassEnv directive specifies an environment +variable that should be passed to child processes. Normally, the +scheduler only passes the DYLD_LIBRARY_PATH, +LD_ASSUME_KERNEL, LD_LIBRARY_PATH, +LD_PRELOAD, NLSPATH, +SHLIB_PATH, TZ, and VGARGS +environment variables to child processes.

+ + +

CUPS 1.2/Mac OS X 10.5Policy

+ +

Examples

+ +
+<Policy name>
+  <Limit operation ... operation>
+  ...
+  </Limit>
+  <Limit operation ... operation>
+  ...
+  </Limit>
+  <Limit All>
+  ...
+  </Limit>
+</Policy>
+
+ +

Description

+ +

The Policy directive specifies IPP operation +access control limits. Each policy contains 1 or more Limit sections to set the +access control limits for specific operations - user limits, +authentication, encryption, and allowed/denied addresses, +domains, or hosts. The <Limit All> section +specifies the default access control limits for operations that +are not listed.

+ +

Policies are named and associated with printers via the +printer's operation policy setting +(printer-op-policy). The default policy for the +scheduler is specified using the DefaultPolicy +directive.

+ + +

Port

+ +

Examples

+ +
+Port 631
+Port 80
+
+ +

Description

+ +

The Port directive specifies a port to listen on. +Multiple Port lines can be specified to listen on +multiple ports. The Port directive is equivalent to +"Listen *:nnn". The default port is 631.

+ +
Note: + +

On systems that support IPv6, this directive will bind to both +the IPv4 and IPv6 wildcard address.

+ +
+ + +

PreserveJobHistory

+ +

Examples

+ +
+PreserveJobHistory On
+PreserveJobHistory Off
+
+ +

Description

+ +

The PreserveJobHistory directive controls whether +the history of completed, canceled, or aborted print jobs is +stored on disk.

+ +

A value of On (the default) preserves job +information until the administrator purges it with the +cancel command.

+ +

A value of Off removes the job information as +soon as each job is completed, canceled, or aborted.

+ + +

PreserveJobFiles

+ +

Examples

+ +
+PreserveJobFiles On
+PreserveJobFiles Off
+
+ +

Description

+ +

The PreserveJobFiles directive controls whether +the document files of completed, canceled, or aborted print jobs +are stored on disk.

+ +

A value of On preserves job files until the +administrator purges them with the cancel command. +Jobs can be restarted (and reprinted) as desired until they are +purged.

+ +

A value of Off (the default) removes the job +files as soon as each job is completed, canceled, or aborted.

+ + +

Printcap

+ +

Examples

+ +
+Printcap
+Printcap /etc/printcap
+Printcap /etc/printers.conf
+Printcap /Library/Preferences/org.cups.printers.plist
+
+ +

Description

+ +

The Printcap directive controls whether or not a +printcap file is automatically generated and updated with a list +of available printers. If specified with no value, then no +printcap file will be generated. The default is to generate a +file named @CUPS_DEFAUL_PRINTCAP@.

+ +

When a filename is specified (e.g. @CUPS_DEFAULT_PRINTCAP@), +the printcap file is written whenever a printer is added or +removed. The printcap file can then be used by applications that +are hardcoded to look at the printcap file for the available +printers.

+ + +

PrintcapFormat

+ +

Examples

+ +
+PrintcapFormat BSD
+PrintcapFormat Solaris
+PrintcapFormat plist
+
+ +

Description

+ +

The PrintcapFormat directive controls the output format of the +printcap file. The default is to generate the plist format on Mac OS X, the +Solaris format on Solaris, and the BSD format on other operating systems.

+ + +

CUPS 1.1.13PrintcapGUI

+ +

Examples

+ +
+PrintGUI /usr/bin/glpoptions
+
+ +

Description

+ +

The PrintcapGUI directive sets the program to +associate with the IRIX printer GUI interface script which is +used by IRIX applications to display printer-specific options. +There is no default program.

+ + +

CUPS 1.1.21ReloadTimeout

+ +

Examples

+ +
+ReloadTimeout 0
+ReloadTimeout 30
+
+ +

Description

+ +

The ReloadTimeout directive specifies the number +of seconds the scheduler will wait for active jobs to complete +before doing a restart. The default is 30 seconds.

+ + +

CUPS 1.1.3RemoteRoot

+ +

Examples

+ +
+RemoteRoot remroot
+RemoteRoot root
+
+ +

Description

+ +

The RemoteRoot directive sets the username for +unauthenticated root requests from remote hosts. The default +username is remroot. Setting RemoteRoot +to root effectively disables this security +mechanism.

+ + +

RequestRoot

+ +

Examples

+ +
+RequestRoot /var/spool/cups
+RequestRoot /foo/bar/spool/cups
+
+ +

Description

+ +

The RequestRoot directive sets the directory for +incoming IPP requests and HTML forms. If an absolute path is not +provided then it is assumed to be relative to the ServerRoot directory. The +default request directory is @CUPS_REQUESTS@.

+ + +

CUPS 1.1.7Require

+ +

Examples

+ +
+<Location /path>
+  ...
+  Require group foo bar
+  Require user john mary
+  Require valid-user
+  Require user @groupname
+  Require user @SYSTEM
+  Require user @OWNER
+</Location>
+
+ +

Description

+ +

The Require directive specifies that +authentication is required for the resource. The +group keyword specifies that the authenticated user +must be a member of one or more of the named groups that +follow.

+ +

The user keyword specifies that the +authenticated user must be one of the named users or groups that +follow. Group names are specified using the "@" prefix.

+ +

The valid-user keyword specifies that any +authenticated user may access the resource.

+ +

The default is to do no authentication. This directive must +appear inside a Location or +Limit section.

+ + +

RIPCache

+ +

Examples

+ +
+RIPCache 128m
+RIPCache 1g
+RIPCache 2048k
+
+ +

Description

+ +

The RIPCache directive sets the size of the +memory cache used by Raster Image Processor ("RIP") filters such +as imagetoraster and pstoraster. The +size can be suffixed with a "k" for kilobytes, "m" for megabytes, +or "g" for gigabytes. The default cache size is "128m", or 128 +megabytes.

+ + +

CUPS 1.1.16RootCertDuration

+ +

Examples

+ +
+RootCertDuration 0
+RootCertDuration 300
+
+ +

Description

+ +

The RootCertDuration directive specifies the +number of seconds the root certificate remains valid. +The scheduler will generate a new certificate as needed when the +number of seconds has expired. If set to 0, the root certificate +is generated only once on startup or on a restart. The default is +300 seconds.

+ + +

CUPS 1.1.7Satisfy

+ +

Examples

+ +
+<Location /path>
+  ...
+  Satisfy all
+  Satisfy any
+</Location>
+
+ +

Description

+ +

The Satisfy directive specifies whether all +conditions must be satisfied to allow access to the resource. If +set to all, then all authentication and access +control conditions must be satisfied to allow access.

+ +

Setting Satisfy to any allows a user +to gain access if the authentication or access control +requirements are satisfied. For example, you might require +authentication for remote access, but allow local access without +authentication.

+ +

The default is all. This directive must appear +inside a Location or Limit section.

+ + +

ServerAdmin

+ +

Examples

+ +
+ServerAdmin user@host
+ServerAdmin root@foo.bar.com
+
+ +

Description

+ +

The ServerAdmin directive identifies the email +address for the administrator on the system. By default the +administrator email address is root@server, where +server is the ServerName.

+ + +

CUPS 1.3.10ServerAlias

+ +

Examples

+ +
+ServerAlias althost
+ServerAlias foo.example.com
+ServerAlias bar.example.com
+ServerAlias one.example.com two.example.com
+ServerAlias *
+
+ +

Description

+ +

The ServerAlias directive specifies alternate names that the server is known by. By default it contains a list of all aliases associated with the ServerName. The special name "*" can be used to allow any hostname when accessing CUPS via an external network interfaces.

+ +
Note + +

The ServerAlias directive is used for HTTP Host header +validation when clients connect to the scheduler from external interfaces. +Using the special name "*" can expose your system to known browser-based +DNS rebinding attacks, even when accessing sites through a firewall. If the +auto-discovery of alternate names does not work, we recommend listing each +alternate name with a ServerAlias directive instead of using "*".

+ +
+ + +

ServerBin

+ +

Examples

+ +
+ServerBin /usr/lib/cups
+ServerBin /foo/bar/lib/cups
+
+ +

Description

+ +

The ServerBin directive sets the directory for +server-run executables. If an absolute path is not provided then +it is assumed to be relative to the ServerRoot directory. The +default executable directory is /usr/lib/cups, +/usr/lib32/cups, or /usr/libexec/cups +depending on the operating system.

+ + +

ServerCertificate

+ +

Examples

+ +
+ServerCertificate /etc/cups/ssl/server.crt
+
+ +

Description

+ +

The ServerCertificate directive specifies the +location of the SSL certificate file used by the server when +negotiating encrypted connections. The certificate must not be +encrypted (password protected) since the scheduler normally runs +in the background and will be unable to ask for a password.

+ +

The default certificate file is +/etc/cups/ssl/server.crt.

+ + +

ServerKey

+ +

Examples

+ +
+ServerKey /etc/cups/ssl/server.key
+
+ +

Description

+ +

The ServerKey directive specifies the location of +the SSL private key file used by the server when negotiating +encrypted connections.

+ +

The default key file is +/etc/cups/ssl/server.crt.

+ + +

ServerName

+ +

Examples

+ +
+ServerName foo.example.com
+ServerName myserver.example.com
+
+ +

Description

+ +

The ServerName directive specifies the hostname +that is reported to clients. By default the server name is the +hostname.

+ + +

ServerRoot

+ +

Examples

+ +
+ServerRoot /etc/cups
+ServerRoot /foo/bar/cups
+
+ +

Description

+ +

The ServerRoot directive specifies the absolute +path to the server configuration and state files. It is also used +to resolve relative paths in the cupsd.conf file. The +default server directory is /etc/cups.

+ + +

CUPS 1.1.21ServerTokens

+ +

Examples

+ +
+ServerTokens None
+ServerTokens ProductOnly
+ServerTokens Major
+ServerTokens Minor
+ServerTokens Minimal
+ServerTokens OS
+ServerTokens Full
+
+ +

Description

+ +

The ServerTokens directive specifies the +information that is included in the Server: header +of all HTTP responses. Table 4 lists the token name along with +the text that is returned. The default is +Minimal.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 4: ServerToken Names and Values
NameValue
NoneNo Server: header is returned
ProductOnly"CUPS"
Major"CUPS 1"
Minor"CUPS 1.2"
Minimal"CUPS 1.2.N" where N is the patch release
OS"CUPS 1.2.N (UNAME)" where N is the patch release and + UNAME is the output of the uname(1) command
Full"CUPS 1.2.N (UNAME) IPP/1.1" where N is the patch + release and UNAME is the output of the uname(1) + command
+ + +

CUPS 1.2/Mac OS X 10.5SetEnv

+ +

Examples

+ +
+SetEnv PATH /usr/lib/cups/filter:/bin:/usr/bin:/usr/local/bin
+SetEnv MY_ENV_VAR foo
+
+ +

Description

+ +

The SetEnv directive specifies an environment +variable that should be passed to child processes.

+ + +

SSLListen

+ +

Examples

+ +
+SSLListen 127.0.0.1:443
+SSLListen 192.0.2.1:443
+
+ +

Description

+ +

The SSLListen directive specifies a network +address and port to listen for secure connections. Multiple +SSLListen directives can be provided to listen on +multiple addresses.

+ +

The SSLListen directive is similar to the SSLPort directive but allows you +to restrict access to specific interfaces or networks.

+ + +

SSLOptions

+ +

Examples

+ +
+SSLOptions None
+SSLOptions NoEmptyFragments
+
+ +

Description

+ +

The SSLOptions directive specifies additional SSL/TLS +protocol options to use for encrypted connected. Currently only two +options are supported - None (the default) for the most +secure mode and NoEmptyFragments to allow CUPS to work with +Microsoft Windows with the FIPS conformance mode enabled.

+ + +

SSLPort

+ +

Examples

+ +
+SSLPort 443
+
+ +

Description

+ +

The SSLPort directive specifies a port to listen +on for secure connections. Multiple SSLPort lines +can be specified to listen on multiple ports.

+ + +

CUPS 1.5SubscriptionPrivateAccess

+ +

Examples

+ +
+SubscriptionPrivateAccess all
+SubscriptionPrivateAccess default
+SubscriptionPrivateAccess {user|@group|@ACL|@OWNER|@SYSTEM}+
+
+ +

Description

+ +

The SubscriptionPrivateAccess directive specifies the access list for a +subscription's private values. The "default" access list is "@OWNER @SYSTEM". +"@ACL" maps to the printer's requesting-user-name-allowed or +requesting-user-name-denied values.

+ +

The SubscriptionPrivateAccess directive must appear inside a Policy section.

+ + +

CUPS 1.5SubscriptionPrivateValues

+ +

Examples

+ +
+SubscriptionPrivateValues all
+SubscriptionPrivateValues default
+SubscriptionPrivateValues none
+SubscriptionPrivateValues attribute-name-1 [ ... attribute-name-N ]
+
+ +

Description

+ +

The SubscriptionPrivateValues directive specifies the list of +subscription values to make private. The "default" values are "notify-events", +"notify-pull-method", "notify-recipient-uri", "notify-subscriber-user-name", and +"notify-user-data".

+ +

The SubscriptionPrivateValues directive must appear inside a Policy section.

+ + +

SystemGroup

+ +

Examples

+ +
+SystemGroup lpadmin
+SystemGroup sys
+SystemGroup system
+SystemGroup root
+SystemGroup root lpadmin
+
+ +

Description

+ +

The SystemGroup directive specifies the system +administration group for System authentication. +Multiple groups can be listed, separated with spaces. The default +group list is @CUPS_SYSTEM_GROUPS@.

+ + +

TempDir

+ +

Examples

+ +
+TempDir /var/tmp
+TempDir /foo/bar/tmp
+
+ +

Description

+ +

The TempDir directive specifies an absolute path +for the directory to use for temporary files. The default +directory is @CUPS_REQUESTS@/tmp.

+ +

Temporary directories must be world-writable and should have +the "sticky" permission bit enabled so that other users cannot +delete filter temporary files. The following commands will create +an appropriate temporary directory called +/foo/bar/tmp:

+ +
+mkdir /foo/bar/tmp
+chmod a+rwxt /foo/bar/tmp
+
+ + +

Timeout

+ +

Examples

+ +
+Timeout 300
+Timeout 90
+
+ +

Description

+ +

The Timeout directive controls the amount of time +to wait before an active HTTP or IPP request times out. The +default timeout is 300 seconds.

+ + +

CUPS 1.2/Mac OS X 10.5UseNetworkDefault

+ +

Examples

+ +
+UseNetworkDefault yes
+UseNetworkDefault no
+
+ +

Description

+ +

The UseNetworkDefault directive controls whether +the client will use a network/remote printer as a default +printer. If enabled, the default printer of a server is used as +the default printer on a client. When multiple servers are +advertising a default printer, the client's default printer is +set to the first discovered printer, or to the implicit class for +the same printer available from multiple servers.

+ +

The default is @CUPS_USE_NETWORK_DEFAULT@.

+ + +

User

+ +

Examples

+ +
+User lp
+User guest
+
+ +

Description

+ +

The User directive specifies the UNIX user that +filter and CGI programs run as. The default user is +@CUPS_USER@.

+ +
Note: + +

You may not use user root, as that would expose +the system to unacceptable security risks. The scheduler will +automatically choose user nobody if you specify a +user whose ID is 0.

+ +
+ + +

CUPS 1.5WebInterface

+ +

Examples

+ +
+WebInterface yes
+WebInterface no
+
+ +

Description

+ +

The WebInterface directive specifies whether the web interface is enabled. The default value is @CUPS_WEBIF@.

+ + + diff --git a/doc/help/ref-error_log.html b/doc/help/ref-error_log.html new file mode 100644 index 0000000000..0109278022 --- /dev/null +++ b/doc/help/ref-error_log.html @@ -0,0 +1,55 @@ + + + + error_log + + + + +

error_log

+ +

The error_log file lists messages from the +scheduler - errors, warnings, etc. The LogLevel +directive controls which messages are logged:

+ +

+ +level date-time message
+ +
+ +I [20/May/1999:19:18:28 +0000] Job 1 queued on 'DeskJet' by 'mike'.
+ +I [20/May/1999:19:21:02 +0000] Job 2 queued on 'DeskJet' by 'mike'.
+ +I [20/May/1999:19:22:24 +0000] Job 2 was cancelled by 'mike'.
+ +

+ +

The level field contains the type of message:

+ +
    + +
  • A - Alert message (LogLevel alert) +
  • C - Critical error message (LogLevel crit) +
  • D - Debugging message (LogLevel debug) +
  • d - Detailed debugging message (LogLevel debug2) +
  • E - Normal error message (LogLevel error) +
  • I - Informational message (LogLevel info) +
  • N - Notice message (LogLevel notice) +
  • W - Warning message (LogLevel warn) +
  • X - Emergency error message (LogLevel emerg) + +
+ +

The date-time field contains the date and time of when +the page started printing. The format of this field is identical +to the data-time field in the access_log file.

+ +

The message fields contains a free-form textual +message. Messages from job filters are prefixed with "[Job +NNN]".

+ + + diff --git a/doc/help/ref-mailto-conf.html b/doc/help/ref-mailto-conf.html new file mode 100644 index 0000000000..0b9513220b --- /dev/null +++ b/doc/help/ref-mailto-conf.html @@ -0,0 +1,108 @@ + + + + mailto.conf + + + + +

mailto.conf

+ +

The /etc/cups/mailto.conf file contains several +directives that defines the local mail server and email +notification preferences for CUPS. Each directive is listed on a +line by itself followed by its value. Comments are introduced +using the number sign ("#") character at the beginning of a +line.

+ + +

Cc

+ +

Examples

+ +
+Cc bigbrother@example.com
+Cc John Doe <jd@example.com>
+
+ +

Description

+ +

The Cc directive specifies an additional +recipient ("carbon copy") for all email notifications. The +default is to not send a copy to anyone but the subscriber.

+ + +

From

+ +

Examples

+ +
+From printserver@example.com
+From Your Happy Printer <printserver@example.com>
+
+ +

Description

+ +

The From directive specifies the sender of email +notifications. The default is the ServerAdmin +address defined in the cupsd.conf file.

+ + +

Sendmail

+ +

Examples

+ +
+Sendmail /usr/sbin/sendmail
+Sendmail /usr/lib/sendmail -bm -i
+
+ +

Description

+ +

The Sendmail directive specifies the command to +run to deliver an email locally. This directive cannot be used +with the SMTPServer directive, and if both +Sendmail and SMTPServer lines appear in the +mailto.conf file, only the last line is used. The +default is /usr/sbin/sendmail.

+ + +

SMTPServer

+ +

Examples

+ +
+SMTPServer mail.example.com
+SMTPServer 192.168.2.1
+
+ +

Description

+ +

The SMTPServer directive specifies a hostname or +IP address of a (possibly remote) SMTP mail server. This +directive cannot be used with the Sendmail directive, +and if both Sendmail and SMTPServer lines +appear in the mailto.conf file, only the last line is +used. The default is to use the Sendmail command +instead.

+ + +

Subject

+ +

Examples

+ +
+Subject [CUPS]
+Subject URGENT EMAIL NOTIFICATION
+
+ +

Description

+ +

The Subject directive specifies a prefix string to +add to the subject of each email notification. The default is to +not add a prefix string.

+ + + + diff --git a/doc/help/ref-page_log.html b/doc/help/ref-page_log.html new file mode 100644 index 0000000000..46453eb2e9 --- /dev/null +++ b/doc/help/ref-page_log.html @@ -0,0 +1,77 @@ + + + + page_log + + + + +

page_log

+ +

The page_log file lists each page that is sent to a +printer. By default, each line contains the following information:

+ +

+ +printer user job-id date-time page-number num-copies job-billing job-originating-host-name job-name media sides
+ +
+ +DeskJet root 2 [20/May/1999:19:21:05 +0000] 1 1 acme-123 localhost myjob letter one-sided
+ +DeskJet root 2 [20/May/1999:19:21:05 +0000] 2 1 acme-123 localhost myjob letter one-sided
+ +

+ +

The PageLogFormat +directive can be used to change this information.

+ +

The printer field contains the name of the printer that +printed the page. If you send a job to a printer class, this +field will contain the name of the printer that was assigned the +job.

+ +

The user field contains the name of the user (the IPP +requesting-user-name attribute) that submitted this +file for printing.

+ +

The job-id field contains the job number of the page +being printed. Job numbers are reset to 1 whenever the spool +directory (/var/spool/cups) is cleared, so don't +depend on this number being unique!

+ +

The date-time field contains the date and time of when +the page started printing. The format of this field is identical +to the data-time field in the access_log +file.

+ +

The page-number and num-copies fields contain the +page number and number of copies being printed of that page. For +printers that cannot produce copies on their own, the +num-copies field will always be 1.

+ +

The job-billing field contains a copy of the +job-billing attribute provided with the IPP +Create-Job or Print-Job requests or "-" +if none was provided. + +

The job-originating-host-name field contains the hostname or IP +address of the client that printed the job.

+ +

The job-name field contains a copy of the +job-name attribute provided with the IPP +Create-Job or Print-Job requests or "-" +if none was provided. + +

The media field contains a copy of the +media attribute provided with the IPP +Create-Job or Print-Job requests or "-" +if none was provided. + +

The sides field contains a copy of the +sides attribute provided with the IPP +Create-Job or Print-Job requests or "-" +if none was provided. + + + diff --git a/doc/help/ref-ppdcfile.html b/doc/help/ref-ppdcfile.html new file mode 100644 index 0000000000..c3967f1c23 --- /dev/null +++ b/doc/help/ref-ppdcfile.html @@ -0,0 +1,2449 @@ + + + + PPD Compiler Driver Information File Reference + + + + +

PPD Compiler Driver Information File Reference

+ +

The CUPS PPD compiler reads meta files that contain descriptions +of one or more PPD files to be generated by +ppdc(1) or the corresponding driver interface +program drv(1). The source file format is plain +ASCII text that can be edited using your favorite text editor.

+ +

Directives may be placed anywhere on a line and are followed by +zero or more values.

+ +

Comments are supported using the C (/* ... */) and C++ (// ...) comment +mechanisms.

+ +

Directives that accept expressions look for sequences of the form:

+ +
+ +
NAME
+
Evaluates to 1 if NAME is defined, otherwise 0.
+ +
number
+ +
Evaluates to the specified integer; the number can be preceded by + a leading sign (+/-) followed by a decimal number (1234), octal number + (01234), or hexadecimal number (0x1234) using the same rules as C and + C++.
+ +
(NAME NAME ... number number ...)
+
Evaluates to the bitwise OR of each named #define constant or + number.
+ +
(NAME == OTHERNAME)
+
(NAME == number)
+
Evaluates to 1 if NAME is equal to the other named constant or + number, otherwise 0.
+ +
(NAME != OTHERNAME)
+
(NAME != number)
+
Evaluates to 1 if NAME is not equal to the other named constant or + number, otherwise 0.
+ +
(NAME < OTHERNAME)
+
(NAME < number)
+
Evaluates to 1 if NAME is less than to the other named constant or + number, otherwise 0.
+ +
(NAME <= OTHERNAME)
+
(NAME <= number)
+
Evaluates to 1 if NAME is less than or equal to the other named + constant or number, otherwise 0.
+ +
(NAME > OTHERNAME)
+
(NAME > number)
+
Evaluates to 1 if NAME is greater than to the other named constant + or number, otherwise 0.
+ +
(NAME >= OTHERNAME)
+
(NAME >= number)
+
Evaluates to 1 if NAME is greater than or equal to the other named + constant or number, otherwise 0.
+ +
+ +

Printer driver information can be grouped and shared using +curly braces ({ ... }); PPD files are written when a close +brace or end-of-file is seen and a PCFileName +directive has been defined.

+ + +

#define

+ +

Syntax

+ +
+#define name expression
+
+ +

Examples

+ +
+#define FOO 100
+#define BAR "Bar, Inc."
+
+ +

Description

+ +

The #define directive assigns a value to a name +which can be later referenced using $name. The name is +case-insensitive and can be any sequence of letters, numbers, +and the underscore. The value can be any valid expression.

+ +

Predefined Names

+ +

The following #define names are set by the PPD compiler:

+ +
    + +
  • CUPS_VERSION - The full CUPS version string, e.g. + "1.4.0"
  • + +
  • CUPS_VERSION_MAJOR - The major version number, e.g. + "1"
  • + +
  • CUPS_VERSION_MINOR - The minor version number, e.g. + "4"
  • + +
  • CUPS_VERSION_PATCH - The patch version number, e.g. + "0"
  • + +
  • PLATFORM_NAME - The operating system name used by the + current system as reported by "uname" ("Windows" on Microsoft + Windows)
  • + +
  • PLATFORM_ARCH - The processor architecture used by the + current system as reported by "uname -m" ("X86" or "X64" on Microsoft + Windows)
  • + +
+ +

See Also

+ +

#include

+ + +

#elif

+ +

Syntax

+ +
+#elif expression
+
+ +

Examples

+ +
+#if HAVE_FOO
+...
+#elif (HAVE_BAR >= 999)
+...
+#else
+...
+#endif 
+
+ +

Description

+ +

The #elif directive allows portions of a driver information file +to be used conditionally. #elif directives must appear after a +corresponding #if directive.

+ +

See Also

+ +

#else, +#endif, +#if

+ + +

#else

+ +

Syntax

+ +
+#else
+
+ +

Examples

+ +
+#if HAVE_FOO
+...
+#elif (HAVE_BAR >= 999)
+...
+#else
+...
+#endif 
+
+ +

Description

+ +

The #else directive allows portions of a driver information file +to be used conditionally when the corresponding +#if and #elif +expressions are non-zero.

+ +

See Also

+ +

#elif, +#endif, +#if

+ + +

#endif

+ +

Syntax

+ +
+#endif
+
+ +

Examples

+ +
+#if HAVE_FOO
+...
+#elif (HAVE_BAR >= 999)
+...
+#else
+...
+#endif 
+
+ +

Description

+ +

The #endif directive ends a conditional block of a driver +information file. It must appear after all of the +#if, #elif, +and #else directives for the current +conditional block.

+ +

See Also

+ +

#elif, +#else, +#if

+ + +

#font

+ +

Syntax

+ +
+#font name encoding "version" charset status
+
+ +

Examples

+ +
+#font Courier Standard "(1.05)" Standard ROM
+#font Symbol Special "(001.005)" Special ROM
+#font Barcode-Foo Special "(1.0)" Special Disk
+#font Unicode-Foo Expert "(2.0)" Adobe-Identity ROM
+
+ +

Description

+ +

The #font directive defines a "base font" for all +printer drivers. The name is the PostScript font name.

+ +

The encoding is the default encoding of the font, usually +Standard, Expert, or Special, as +defined in the Adobe PPD file specification.

+ +

The version is the PostScript string definition that +corresponds to the font version number.

+ +

The charset defines the available characters in the font, +usually Standard or Special, as defined in the +Adobe PPD file specification.

+ +

The status is the installation status of the font and must be +either the word ROM or Disk. + +

Base fonts differ from fonts defined using the Font directive in that they are not +automatically associated with all drivers - you must use the +special Font * directive to include them in a +driver.

+ +

Currently the #font directive is used mainly for +defining the standard raster fonts in the +<font.defs> include file.

+ +

See Also

+ +

#include, +Font

+ + +

#if

+ +

Syntax

+ +
+#if name or expression
+
+ +

Examples

+ +
+#if HAVE_FOO
+...
+#elif (HAVE_BAR >= 999)
+...
+#else
+...
+#endif 
+
+ +

Description

+ +

The #if directive allows portions of a driver information file +to be used conditionally. When followed by a name, the data that follows is +used only when the name is defined, otherwise the data is ignored. +#if directives can be nested up to 100 times.

+ +

See Also

+ +

#elif, +#else, +#endif

+ + +

#include

+ +

Syntax

+ +
+#include <filename>
+#include "filename"
+
+ +

Examples

+ +
+#include <font.defs>
+#include "myfile.h"
+
+ +

Description

+ +

The #include directive reads the named driver +information file. If the filename is included inside angle +brackets (<filename>), then the PPD compiler will +look for the file in all of the include directories it knows +about. Otherwise, the file is opened in the current directory +relative to the current driver information file, and if that +fails then it looks in the include directories for the file.

+ +

The #include directive can be nested to as many +files as are allowed by the host operating system, typically at +least 100 files.

+ +

See Also

+ +

#define, +#font, +#media

+ + +

#media

+ +

Syntax

+ +
+#media name width length
+#media "name/text" width length
+
+ +

Examples

+ +
+#media "Letter/Letter - 8.5x11in" 8.5in 11in
+#media "A4/A4 - 210x297mm" 210mm 297mm
+#media "w936h1368/Super B/A3 - 13x19in" 936 1368
+#media Photo 4in 6in
+
+ +

Description

+ +

The #media directive defines a named media size for +inclusion in a driver. The name with optional user text defines +the name for the media size and is used with the MediaSize directive to associate +the media size with the driver. The name may contain up to 40 ASCII +characters within the range of decimal 33 to decimal 126 inclusive, +except for the characters comma (44), slash (47) and colon (58). +The user text, if supplied, may not exceed 80 bytes in length.

+ +

The width and length define the dimensions of the media. Each +number is optionally followed by one of the following unit +suffixes:

+ +
    + +
  • cm - centimeters
  • + +
  • ft - feet
  • + +
  • in - inches
  • + +
  • m - meters
  • + +
  • mm - millimeters
  • + +
  • pt - points (72 points = 1 inch)
  • + +
+ +

Points are assumed if no units are specified. + +

See Also

+ +

#include, +CustomMedia, +MediaSize

+ + +

#po

+ +

Syntax

+ +
+#po locale filename
+
+ +

Examples

+ +
+#po es "es.po"
+#po fr_CA "mydriver-fr_CA.po"
+
+ +

Description

+ +

The #po directive defines a message catalog to use for the +given POSIX language abbreviation. Multiple #po directives can be +specified to list multiple catalogs. The filename can be an absolute path or +relative to the driver information file. GNU gettext and Mac OS X .strings +files are supported.

+ + +

Attribute

+ +

Syntax

+ +
+Attribute name "" value
+Attribute name keyword value
+Attribute name "keyword/text" value
+
+ +

Examples

+ +
+Attribute cupsInkChannels "" 1
+Attribute cupsAllDither 600dpi "1.0"
+Attribute fooProfile "Photo/Photographic Profile" "photopro.icc"
+
+ +

Description

+ +

The Attribute directive creates a PPD attribute. The +name may contain up to 40 ASCII characters within the range of decimal +33 to decimal 126 inclusive, except for the characters comma (44), +slash (47) and colon (58).

+ +

The selector can be the empty string ("") or text of up +to 80 bytes.

+ +

The value is any string or number; the string may contain multiple +lines, however no one line may exceed 255 bytes.

+ +

See Also

+ +

LocAttribute

+ + +

Choice

+ +

Syntax

+ +
+Choice name "code"
+Choice "name/text" "code"
+
+ +

Examples

+ +
+Choice None "<</MediaType (None)>>setpagedevice"
+Choice "False/No" "<</cupsCompression 0>>setpagedevice"
+
+ +

Description

+ +

The Choice directive adds a single choice to the +current option. The name may contain up to 40 ASCII characters within +the range of decimal 33 to decimal 126 inclusive, except for the +characters comma (44), slash (47) and colon (58).

+ +

If provided, the text can be any string up to 80 bytes +in length. If no text is provided, the name is used.

+ +

The code is any string and may contain multiple lines, +however no one line may exceed 255 bytes.

+ +

See Also

+ +

ColorModel, +Cutter, +Darkness, +Duplex, +Finishing, +Group, +InputSlot, +Installable, +MediaType, +Option, +Resolution, +UIConstraints

+ + +

ColorDevice

+ +

Syntax

+ +
+ColorDevice boolean-value
+
+ +

Examples

+ +
+ColorDevice no
+ColorDevice yes
+
+ +

Description

+ +

The ColorDevice directive tells the application if +the printer supports color. It is typically used in conjunction +with the ColorModel directive +to provide color printing support.

+ +

See Also

+ +

ColorModel

+ + +

DeprecatedColorModel

+ +

Syntax

+ +
+ColorModel name colorspace colororder compression
+ColorModel "name/text" colorspace colororder compression
+
+ +

Examples

+ +
+ColorModel Gray/Grayscale w chunky 0
+ColorModel RGB/Color rgb chunky 0
+ColorModel CMYK cmyk chunky 0
+
+ +

Description

+ +

The ColorModel directive is a convenience directive +which creates a ColorModel option and choice for the current +printer driver. The name may contain up to 40 ASCII characters within +the range of decimal 33 to decimal 126 inclusive, except for the +characters comma (44), slash (47) and colon (58).

+ +

If provided, the text can be any string up to 80 bytes in length. +If no text is provided, the name is used.

+ +

The colorspace argument is one of the standard colorspace +keywords defined later in this appendix in the section titled, +"Colorspace Keywords".

+ +

The colororder argument is one of the standard color order +keywords defined later in this appendix in the section titled, +"Color Order Keywords".

+ +

The compression argument is any number and is assigned to the +cupsCompression attribute in the PostScript page device +dictionary.

+ +

See Also

+ +

Choice, +ColorDevice, +Cutter, +Darkness, +Duplex, +Finishing, +Group, +InputSlot, +Installable, +MediaType, +Option, +Resolution, +UIConstraints

+ + +

DeprecatedColorProfile

+ +

Syntax

+ +
+ColorProfile resolution/mediatype gamma density matrix
+
+ +

Examples

+ +
+ColorProfile -/- 1.7 1.0
+     1.0    0.0    0.0
+     0.0    1.0    0.0
+     0.0    0.0    1.0
+
+ColorProfile 360dpi/- 1.6 1.0
+     1.0   -0.05  -0.3
+    -0.35   1.0   -0.15
+    -0.095 -0.238  0.95
+
+ColorProfile 720dpi/Special 1.5 1.0
+     1.0    0.0   -0.38
+    -0.4    1.0    0.0
+     0.0   -0.38   0.9
+
+ +

Description

+ +

The ColorProfile directive defines a CMY +transform-based color profile. The resolution and mediatype +arguments specify the Resolution and MediaType +choices which use the profile; the hyphen (-) is used to +specify that any resolution or mediatype can be used with the +profile.

+ +

The gamma argument specifies the gamma correction to apply to +the color values (P = pg) and is a real number +greater than 0. Values larger than 1 cause a general lightening +of the print while values smaller than 1 cause a general +darkening of the print. A value of 1 disables gamma +correction.

+ +

The density argument specifies the linear density correction +to apply to the color values (P = d * pg) and is a +real number greater than 0 and less than or equal to 1. A value +1 of disables density correction while lower values produce +proportionately lighter output.

+ +

The matrix argument specifies a 3x3 linear transformation +matrix in row-major order. The matrix is applied only to the CMY +component of a RGB to CMYK transformation and is not used when +printing in grayscale or CMYK mode unless the printer only +supports printing with 3 colors.

+ +

See Also

+ +

SimpleColorProfile

+ + +

Copyright

+ +

Syntax

+ +
+Copyright "text"
+
+ +

Examples

+ +
+Copyright "Copyright 2008 by Foo Enterprises"
+
+Copyright
+"This software is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License as
+published by the Free Software Foundation; either version 2 of
+the License, or (at your option) any later version.
+
+This software is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public
+License along with this software; if not, write to the Free
+Software Foundation, Inc., 59 Temple Place, Suite 330, Boston,
+MA 02111 USA"
+
+ +

Description

+ +

The Copyright directive adds text comments to the +top of a PPD file, typically for use in copyright notices. The +text argument can contain multiple lines of text, but no line +may exceed 255 bytes.

+ + +

CustomMedia

+ +

Syntax

+ +
+CustomMedia name width length left bottom right top
+    "size-code" "region-code"
+
+CustomMedia "name/text" width length left bottom right top
+    "size-code" "region-code"
+
+ +

Examples

+ +
+CustomMedia Letter 8.5in 11in 0.25in 0.46in 0.25in 0.04in
+    "<</PageSize[612 792]/ImagingBBox null/ManualFeed false>>
+     setpagedevice"
+    "<</PageSize[612 792]/ImagingBBox null/ManualFeed true>>
+     setpagedevice"
+
+CustomMedia "A4/A4 - 210x297mm" 210mm 297mm 12 12 12 12
+    "<</PageSize[595 842]/ImagingBBox null>>setpagedevice"
+    "<</PageSize[595 842]/ImagingBBox null>>setpagedevice"
+
+ +

Description

+ +

The CustomMedia directive adds a custom media size to +the driver. The name may contain up to 40 ASCII characters within the +range of decimal 33 to decimal 126 inclusive, except for the characters +comma (44), slash (47) and colon (58).

+ +

If provided, the text can be any string up to 80 bytes in length. +If no text is provided, the name is used.

+ +

The width and length arguments specify the dimensions of the +media as defined for the #media +directive.

+ +

The left, bottom, right, and top arguments specify the +printable margins of the media.

+ +

The size-code and region-code arguments specify the +PostScript commands to run for the PageSize and +PageRegion options, respectively. The commands can +contain multiple lines, however no line may be more than 255 +bytes in length.

+ +

See Also

+ +

#media, +MediaSize

+ + +

Cutter

+ +

Syntax

+ +
+Cutter boolean-value
+
+ +

Examples

+ +
+Cutter yes
+Cutter no
+
+ +

Description

+ +

The Cutter directive specifies whether the printer +has a built-in media cutter. When a cutter is present, the +printer's PPD file will contain a CutMedia option that +allows the user to control whether the media is cut at the end +of the job.

+ +

See Also

+ +

Choice, +ColorModel, +Darkness, +Duplex, +Finishing, +Group, +InputSlot, +Installable, +MediaType, +Option, +Resolution, +UIConstraints

+ + +

DeprecatedDarkness

+ +

Syntax

+ +
+Darkness temperature name
+Darkness temperature "name/text"
+
+ +

Examples

+ +
+Darkness 0 Light
+Darkness 2 "Normal/Standard"
+
+ +

Description

+ +

The Darkness directive defines a choice for the +cupsDarkness option which sets the +cupsCompression attribute in the PostScript page device +dictionary. It is used with the CUPS rastertolabel +sample driver to control the print head temperature and +therefore the darkness of the print.

+ +

The temperature argument specifies a temperature value for +the Dymo driver from 0 (lowest) to 3 (highest), with 2 +representing the normal setting.

+ +

The name may contain up to 40 ASCII characters within the range of +decimal 33 to decimal 126 inclusive, except for the characters comma +(44), slash (47) and colon (58).

+ +

If provided, the text can be any string up to 80 bytes in length. +If no text is provided, the name is used.

+ +

See Also

+ +

Choice, +ColorModel, +Cutter, +Duplex, +Finishing, +Group, +InputSlot, +Installable, +MediaType, +Option, +Resolution, +UIConstraints

+ + +

DriverType

+ +

Syntax

+ +
+DriverType type
+
+ +

Examples

+ +
+DriverType custom
+DriverType escp
+DriverType pcl
+DriverType ps
+
+ +

Description

+ +

The DriverType directive tells the PPD compiler +which DDK filters to include in the PPD file. The following +types are supported:

+ +
    + +
  • custom - Use only those filters that are + defined in the driver information file
  • + +
  • epson - Use the CUPS sample Epson driver filter + rastertoepson
  • + +
  • escp - Use the ESC/P DDK driver filters + commandtoescpx and + rastertoescpx
  • + +
  • hp - Use the CUPS sample HP driver filter + rastertohp
  • + +
  • label - Use the CUPS sample label driver filter rastertolabel
  • + +
  • pcl - Use the HP-PCL DDK driver filters + commandtopclx and + rastertopclx
  • + +
  • ps - Use no filters; this driver is for a + standard PostScript device
  • + +
+ +

See Also

+ +

Filter, +ModelNumber

+ + +

Duplex

+ +

Syntax

+ +
+Duplex type
+
+ +

Examples

+ +
+Duplex none
+Duplex normal
+Duplex flip
+Duplex rotated
+Duplex manualtumble
+
+ +

Description

+ +

The Duplex directive determines whether double-sided printing +is supported in the current driver. The type argument specifies the type +of duplexing that is supported:

+ +
    + +
  • none - double-sided printing is not + supported
  • + +
  • normal - double-sided printing is + supported
  • + +
  • flip - double-sided printing is supported, + but the back side image needs to be flipped vertically + (used primarily with inkjet printers)
  • + +
  • rotated - double-sided printing is supported, + but the back side image needs to be rotated 180 degrees for + DuplexNoTumble
  • + +
  • manualtumble - double-sided printing is supported, + but the back side image needs to be rotated 180 degrees for + DuplexTumble
  • + +
+ +

See Also

+ +

Choice, +ColorModel, +Cutter, +Darkness, +Finishing, +Group, +InputSlot, +Installable, +MediaType, +Option, +Resolution, +UIConstraints

+ + +

FileName

+ +

Syntax

+ +
+FileName "filename"
+
+ +

Examples

+ +
+FileName "Acme Laser Printer 2000"
+FileName "Acme Ink Waster 1000"
+
+ +

Description

+ +

The FileName attribute specifies the "long" name of the +PPD file for the current driver.

+ +

See Also

+ +

Manufacturer, +ModelName, +PCFileName, +Version

+ + +

Filter

+ +

Syntax

+ +
+Filter mime-type cost program
+
+ +

Examples

+ +
+Filter application/vnd.cups-raster 50 rastertofoo
+Filter application/vnd.hp-HPGL 25 /usr/foo/filter/hpgltofoo
+
+ +

Description

+ +

The Filter directive adds a filter for the current +driver. The mime-type argument is a valid MIME media type name +as defined in a CUPS mime.types file.

+ +

The cost argument specifies the relative cost of the filter. +In general, use a number representing the average percentage of +CPU time that is used when printing the specified MIME media +type.

+ +

The program argument specifies the program to run; if the +program is not an absolute filename, then CUPS will look for the +program in the CUPS filter directory.

+ +

See Also

+ +

DriverType

+ + +

DeprecatedFinishing

+ +

Syntax

+ +
+Finishing name
+Finishing "name/text"
+
+ +

Examples

+ +
+Finishing None
+Finishing "Glossy/Photo Overcoat"
+
+ +

Description

+ +

The Finishing directive adds a choice to the +cupsFinishing option. The name may contain up to 40 ASCII +characters within the range of decimal 33 to decimal 126 inclusive, +except for the characters comma (44), slash (47) and colon (58).

+ +

If provided, the text can be any string up to 80 bytes in length. +If no text is provided, the name is used.

+ +

The name is stored in the OutputType attribute in the +PostScript page device dictionary.

+ +

See Also

+ +

Choice, +ColorModel, +Cutter, +Darkness, +Duplex, +Group, +InputSlot, +Installable, +MediaType, +Option, +Resolution, +UIConstraints

+ + +

Font

+ +

Syntax

+ +
+Font name encoding "version" charset status
+Font *
+
+ +

Examples

+ +
+Font *
+Font Courier Standard "(1.05)" Standard ROM
+Font Symbol Special "(001.005)" Special ROM
+Font Barcode-Foo Special "(1.0)" Special Disk
+Font Unicode-Foo Expert "(2.0)" Adobe-Identity ROM
+
+ +

Description

+ +

The Font directive defines a "device font" for the +current printer driver. The name is the PostScript font name.

+ +

The encoding is the default encoding of the font, usually +Standard, Expert, or Special, as +defined in the Adobe PPD file specification.

+ +

The version is the PostScript string definition that +corresponds to the font version number.

+ +

The charset defines the available characters in the font, +usually Standard or Special, as defined in the +Adobe PPD file specification.

+ +

The status is the installation status of the font and must be +either the word ROM or Disk.

+ +

Device fonts differ from fonts defined using the #font directive in that they are +automatically associated with the current driver. Fonts defined +using #font may be imported into the current driver +using the Font * form of this directive.

+ +

See Also

+ +

#font

+ + +

Group

+ +

Syntax

+ +
+Group name
+Group "name/text"
+
+ +

Examples

+ +
+Group General
+Group "InstallableOptions/Options Installed"
+Group "Special/Vendor Options"
+
+ +

Description

+ +

The Group directive specifies the group for new +Option directives. The name may contain up to 40 ASCII +characters within the range of decimal 33 to decimal 126 inclusive, +except for the characters comma (44), slash (47) and colon (58).

+ +

If provided, the text can be any string up to 40 bytes in length. +If no text is provided, the name is used.

+ +

The names General and InstallableOptions +are predefined for the standard Adobe UI keywords and for installable +options, respectively.

+ +
+ + + +
Note: + +

Because of certain API binary compatibility issues, + CUPS limits the length of PPD group translation strings + (text) to 40 bytes, while the PPD specification + allows for up to 80 bytes.

+ +
+ +

See Also

+ +

Choice, +ColorModel, +Cutter, +Darkness, +Duplex, +Finishing, +InputSlot, +Installable, +MediaType, +Option, +Resolution, +UIConstraints

+ + +

HWMargins

+ +

Syntax

+ +
+HWMargins left bottom right top
+
+ +

Examples

+ +
+HWMargins 18 36 18 36
+HWMargins 0.25in 0.5in 0.25in 0.5in
+HWMargins 0.6cm 1.2cm 0.6cm 1.2cm
+
+ +

Description

+ +

The HWMargins directive specifies the current +margins for MediaSize that +follow. The left, bottom, right, and top margin values specify +the printable margins.

+ +

See Also

+ +

MediaSize

+ + +

InputSlot

+ +

Syntax

+ +
+InputSlot position name
+InputSlot position "name/text"
+
+ +

Examples

+ +
+InputSlot 0 Auto
+InputSlot 1 "Upper/Tray 1"
+
+ +

Description

+ +

The InputSlot directive adds a new choice to the +InputSlot option. The position argument is a number +from 0 to 232-1 specifying the value that is placed +in the MediaPosition attribute in the PostScript page +device dictionary.

+ +

The name may contain up to 40 ASCII characters within the range of +decimal 33 to decimal 126 inclusive, except for the characters comma +(44), slash (47) and colon (58).

+ +

If provided, the text can be any string up to 80 bytes in length. +If no text is provided, the name is used.

+ +

See Also

+ +

Choice, +ColorModel, +Cutter, +Darkness, +Duplex, +Finishing, +Group, +Installable, +MediaType, +Option, +Resolution, +UIConstraints

+ + +

Installable

+ +

Syntax

+ +
+Installable name
+Installable "name/text"
+
+ +

Examples

+ +
+Installable EnvTray
+Installable "Option1/Duplexer Installed"
+
+ +

Description

+ +

The Installable directive adds a new boolean option +to the InstallableOptions group with a default value of +False. The name may contain up to 40 ASCII characters +within the range of decimal 33 to decimal 126 inclusive, except for +the characters comma (44), slash (47) and colon (58).

+ +

If provided, the text can be any string up to 80 bytes in length. +If no text is provided, the name is used.

+ + +

LocAttribute

+ +

Syntax

+ +
+LocAttribute name "keyword/text" value
+
+ +

Examples

+ +
+LocAttribute fooProfile "Photo/Photographic Profile" "photopro.icc"
+
+ +

Description

+ +

The LocAttribute directive creates a localized PPD +attribute. The name may contain up to 40 ASCII characters within the +range of decimal 33 to decimal 126 inclusive, except for the characters +comma (44), slash (47) and colon (58).

+ +

The selector can be the empty string ("") or text of up +to 80 bytes.

+ +

The value is any string or number; the string may contain multiple +lines, however no one line may exceed 255 bytes.

+ +

See Also

+ +

Attribute

+ + +

ManualCopies

+ +

Syntax

+ +
+ManualCopies boolean-value
+
+ +

Examples

+ +
+ManualCopies no
+ManualCopies yes
+
+ +

Description

+ +

The ManualCopies directive specifies whether copies +need to be produced by the RIP filters. The default is +no.

+ +

See Also

+ +

Choice, +ColorModel, +Cutter, +Darkness, +Duplex, +Finishing, +Group, +InputSlot, +MediaType, +Option, +Resolution, +UIConstraints

+ + +

Manufacturer

+ +

Syntax

+ +
+Manufacturer "name"
+
+ +

Examples

+ +
+Manufacturer "Foo"
+Manufacturer "HP"
+
+ +

Description

+ +

The Manufacturer directive specifies the +manufacturer name for the current driver. The name argument must +conform to the manufacturer name requirements in the Adobe PPD +file specification.

+ +

See Also

+ +

FileName, +ModelName, +PCFileName, +Version

+ + +

MaxSize

+ +

Syntax

+ +
+MaxSize width length
+
+ +

Examples

+ +
+MaxSize 36in 100ft
+MaxSize 300cm 30m
+
+ +

Description

+ +

The MaxSize directive specifies the maximum width +and length that is supported for custom page sizes.

+ +

See Also

+ +

MinSize, +VariablePaperSize

+ + +

MediaSize

+ +

Syntax

+ +
+MediaSize name
+
+ +

Examples

+ +
+MediaSize Letter
+MediaSize A4
+
+ +

Description

+ +

The MediaSize directive adds the named size to the +current printer driver using the current margins defined with +the HWMargins directive. The +name argument must match a media size defined using the #media directive.

+ +

See Also

+ +

#media, +HWMargins

+ + +

MediaType

+ +

Syntax

+ +
+MediaType type name
+MediaType type "name/text"
+
+ +

Examples

+ +
+MediaType 0 Auto
+MediaType 1 "Plain/Plain Paper"
+
+ +

Description

+ +

The MediaType directive adds a new choice to the +MediaType option. The type argument is a number +from 0 to 232-1 specifying the value that is placed +in the cupsMediaType attribute in the PostScript page +device dictionary.

+ +

The name may contain up to 40 ASCII characters within the range of +decimal 33 to decimal 126 inclusive, except for the characters comma +(44), slash (47) and colon (58).

+ +

If provided, the text can be any string up to 80 bytes in length. +If no text is provided, the name is used.

+ +

The name is placed in the MediaType attribute in the +PostScript page device dictionary.

+ +

See Also

+ +

Choice, +ColorModel, +Cutter, +Darkness, +Duplex, +Finishing, +Group, +InputSlot, +Installable, +Option, +Resolution, +UIConstraints

+ + +

MinSize

+ +

Syntax

+ +
+MinSize width length
+
+ +

Examples

+ +
+MinSize 4in 8in
+MinSize 10cm 20cm
+
+ +

Description

+ +

The MinSize directive specifies the minimum width +and length that is supported for custom page sizes.

+ +

See Also

+ +

MaxSize, +VariablePaperSize

+ + +

ModelName

+ +

Syntax

+ +
+ModelName "name"
+
+ +

Examples

+ +
+ModelName "Foo Laser Printer 2000"
+ModelName "Colorific 123"
+
+ +

Description

+ +

The ModelName directive sets the printer name for +the ModelName, NickName, and +ShortNickName attributes for the printer driver. The +name is any string of letters, numbers, spaces, and the +characters ".", "/", "-", and "+" and should not begin with the +manufacturer name since the PPD compiler will add this +automatically for you. The maximum length of the name string is +31 bytes to conform to the Adobe limits on the length of +ShortNickName.

+ +

See Also

+ +

FileName, +Manufacturer, +PCFileName, +Version

+ + +

ModelNumber

+ +

Syntax

+ +
+ModelNumber expression
+
+ +

Examples

+ +
+ModelNumber 123
+ModelNumber ($PCL_PAPER_SIZE $PCL_PJL)
+
+ +

Description

+ +

The ModelNumber directive sets the +cupsModelNumber attribute for the printer driver, which +is often used by the printer driver filter to tailor its output +for the current device. The number is any integer or bitwise OR +of integers and constants that is appropriate for the printer +driver filters.

+ +

A complete list of printer driver model number constants is +available later in this appendix in the section titled, "Printer Driver ModelNumber +Constants".

+ +

See Also

+ +

DriverType, +Filter

+ + +

Option

+ +

Syntax

+ +
+Option name type section order
+Option "name/text" type section order
+
+ +

Examples

+ +
+Option Punch Boolean AnySetup 10
+Option "fooFinish/Finishing Option" PickOne DocumentSetup 10 
+
+ +

Description

+ +

The Option directive creates a new option in the +current group, by default the General group. The name +may contain up to 40 ASCII characters within the range of decimal 33 +to decimal 126 inclusive, except for the characters comma (44), slash +(47) and colon (58).

+ +

If provided, the text can be any string up to 80 bytes in length. +If no text is provided, the name is used.

+ +

The type argument is one of the following keywords:

+ +
    + +
  • Boolean - a true/false option
  • + +
  • PickOne - allows the user to pick one + choice from a list
  • + +
  • PickMany - allows the user to pick zero or + more choices from a list
  • + +
+ +

The section argument is one of the following keywords:

+ +
    + +
  • AnySetup - The option can be placed in + either the DocumentSetup or PageSetup sections of the + PostScript document
  • + +
  • DocumentSetup - The option must be placed + in the DocumentSetup section of the PostScript document; + this does not allow the option to be overridden on + individual pages
  • + +
  • ExitServer - The option must be placed in a + separate initialization job prior to the document (not + used for raster printer drivers)
  • + +
  • JCLSetup - The option contains job control + language commands and must be sent prior to the document + using the JCLBegin and + JCLToPSInterpreter attributes (not used for + raster printer drivers)
  • + +
  • PageSetup - The option must be placed at the + beginning of each page in the PostScript document
  • + +
  • Prolog - The option must be placed in the + prolog section of the PostScript document; this is + typically used to add special comments for high-end + typesetters, but can also be used to add CUPS PostScript + job ticket comments.
  • + +
+ +

The order argument is a real number greater than or equal to +0.0 and is used to sort the printer commands from many options +before sending them to the printer or RIP filter.

+ +

See Also

+ +

Choice, +ColorModel, +Cutter, +Darkness, +Duplex, +Finishing, +Group, +InputSlot, +Installable, +MediaType, +Resolution, +UIConstraints

+ + +

PCFileName

+ +

Syntax

+ +
+PCFileName "filename.ppd"
+
+ +

Examples

+ +
+PCFileName "foljt2k1.ppd"
+PCFileName "deskjet.ppd"
+
+ +

Description

+ +

The PCFileName attribute specifies the name of the +PPD file for the current driver. The filename argument must +conform to the Adobe PPD file specification and can be no more +than 8 filename characters plus the extension ".ppd".

+ +

See Also

+ +

FileName, +Manufacturer, +ModelName, +Version

+ + +

DeprecatedResolution

+ +

Syntax

+ +
+Resolution colorspace bits-per-color row-count row-feed row-step name
+Resolution colorspace bits-per-color row-count row-feed row-step "name/text"
+
+ +

Examples

+ +
+Resolution - 8 0 0 0 300dpi
+Resolution k 8 0 0 0 "600x300dpi/600 DPI Grayscale"
+
+ +

Description

+ +

The Resolution directive creates a new +Resolution option choice which sets the +HWResolution, cupsBitsPerColor, +cupsRowCount, cupsRowFeed, +cupsRowStep, and optionally the cupsColorSpace +page device dictionary attributes. The colorspace argument +specifies a colorspace to use for the specified resolution and +can be the hyphen (-) character to make no change to +the selected color model or any keyword listed in the section +titled, "Colorspace Keywords", to +force the named colorspace.

+ +

The bits-per-color argument specifies the number of bits per +color to generate when RIP'ing a job. The values 1, 2, 4, and 8 +are currently supported by CUPS.

+ +

The row-count, row-feed, and row-step argument specify the +driver-dependent values for the cupsRowCount, +cupsRowFeed, and cupsRowStep attributes, +respectively. Most drivers leave these attributes set to 0, but +any number from 0 to 232-1 is allowed.

+ +

The name argument must conform to the resolution naming +conventions in the Adobe PPD file specification, either +HHHdpi for symmetric resolutions or HHHxVVVdpi +for asymmetric resolutions. The HHH and VVV in +the examples represent the horizontal and vertical resolutions +which must be positive integer values.

+ +

If provided, the text can be any string up to 80 bytes in length. +If no text is provided, the name is used.

+ +

See Also

+ +

Choice, +ColorModel, +Cutter, +Darkness, +Duplex, +Finishing, +Group, +InputSlot, +Installable, +MediaType, +Option, +UIConstraints

+ + +

DeprecatedSimpleColorProfile

+ +

Syntax

+ +
+SimpleColorProfile resolution/mediatype density
+    yellow-density red-density gamma
+    red-adjust green-adjust blue-adjust
+
+ +

Examples

+ +
+SimpleColorProfile -/- 100 100 200 1.0 0 0 0
+
+SimpleColorProfile 360dpi/- 100 95 150 1.2 5 10 15
+
+SimpleColorProfile 720dpi/Glossy 100 90 120 1.5 -5 5 10
+
+ +

Description

+ +

The SimpleColorProfile directive creates a +matrix-based ColorProfile. +The resolution and mediatype arguments specify the +Resolution and MediaType choices which use the +profile; the hyphen (-) is used to specify that any +resolution or mediatype can be used with the profile.

+ +

The density argument specifies the linear density correction +to apply to the color values (P = d * 0.01 * pg) and +is an integer greater than 0 and less than or equal to 100. A +value 100 of disables density correction while lower values +produce proportionately lighter output. The density value +adjusts all color channels equally in all color modes.

+ +

The yellow-density argument specifies the density of the +yellow channel when printing in grayscale or RGB mode and is an +integer greater than 0 and less then or equal to 100. A value of +100 disables yellow density correction while lower values +produce proportionately lighter output.

+ +

The red-density argument specifies the two-color density +limit (e.g. C + M, C + Y, M + Y) when printing in grayscale or +RGB mode and is an integer greater than 0 and less then or equal +to 200. A value of 200 disables two-color density correction +while lower values produce proportionately lighter output.

+ +

The gamma argument specifies the gamma correction to apply to +the color values (P = pg) and is a real number +greater than 0. Values larger than 1 cause a general lightening +of the print while values smaller than 1 cause a general +darkening of the print. A value of 1 disables gamma +correction.

+ +

The red-adjust, green-adjust, blue-adjust arguments specify +the percentage of color to add or remove. Positive red-adjust +values add magenta and negative values add yellow. Positive +green-adjust values add cyan and negative values add yellow. +Positive blue-adjust values add cyan and negative values add +magenta. Values of 0 disable color adjustments.

+ +

See Also

+ +

ColorProfile

+ + +

Throughput

+ +

Syntax

+ +
+Throughput pages-per-minute
+
+ +

Examples

+ +
+Throughput 1
+Throughput 10
+
+ +

Description

+ +

The Throughput directive sets the Throughput +attribute for the current printer driver. The pages-per-minute +argument is a positive integer representing the peak number of +pages per minute that the printer is capable of producing. Use a +value of 1 for printers that produce less than 1 page per +minute.

+ + +

UIConstraints

+ +

Syntax

+ +
+UIConstraints "*Option1 *Option2"
+UIConstraints "*Option1 Choice1 *Option2"
+UIConstraints "*Option1 *Option2 Choice2"
+UIConstraints "*Option1 Choice1 *Option2 Choice2"
+
+ +

Examples

+ +
+UIConstraints "*Finishing *MediaType"
+UIConstraints "*Option1 False *Duplex"
+UIConstraints "*Duplex *MediaType Transparency"
+UIConstraints "*Resolution 600dpi *ColorModel RGB"
+
+ +

Description

+ +

The UIConstraints directive adds a constraint +between two options. Constraints inform the application when a +user has chosen incompatible options. Each option name is +preceded by the asterisk (*). If no choice is given for +an option, then all choices except False and +None will conflict with the other option and choice(s). +Since the PPD compiler automatically adds reciprocal constraints +(option A conflicts with option B, so therefore option B +conflicts with option A), you need only specify the constraint +once.

+ +

See Also

+ +

Choice, +ColorModel, +Cutter, +Darkness, +Duplex, +Finishing, +Group, +InputSlot, +Installable, +MediaType, +Option, +Resolution

+ + +

VariablePaperSize

+ +

Syntax

+ +
+VariablePaperSize boolean-value
+
+ +

Examples

+ +
+VariablePaperSize yes
+VariablePaperSize no
+
+ +

Description

+ +

The VariablePaperSize directive specifies whether +the current printer supports variable (custom) page sizes. When +yes is specified, the PPD compiler will include the +standard PPD attributes required to support custom page +sizes.

+ +

See Also

+ +

MaxSize, +MinSize

+ + +

Version

+ +

Syntax

+ +
+Version number
+
+ +

Examples

+ +
+Version 1.0
+Version 3.7
+
+ +

Description

+ +

The Version directive sets the FileVersion +attribute in the PPD file and is also used for the +NickName attribute. The number argument is a positive +real number.

+ +

See Also

+ +

Manufacturer, +ModelName, +PCFileName

+ + +

Standard Include Files

+ +

Table B-1 shows the standard include +files which are provided with the DDK.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table B-1, +Standard Include Files
Include FileDescription
<font.defs>Defines all of the + standard fonts which are included with ESP Ghostscript + and the Apple PDF RIP.
<epson.h>Defines all of the + CUPS ESC/P sample driver constants.
<escp.h>Defines all of the + DDK ESC/P driver constants.
<hp.h>Defines all of the + CUPS HP-PCL sample driver constants.
<label.h>Defines all of the + CUPS label sample driver constants.
<media.defs>Defines all of the + standard media sizes listed in Appendix B of the Adobe + PostScript Printer Description File Format + Specification.
<pcl.h>Defines all of the + DDK HP-PCL driver constants.
<raster.defs>Defines all of the CUPS + raster format constants.
+ +

Printer Driver ModelNumber Constants

+ +

The CUPS DDK and sample drivers use the +cupsModelNumber attribute in the PPD file to tailor +their output to the printer. The following sections describe the +constants for each driver.

+ +

The CUPS ESC/P Sample Driver (epson)

+ +

The epson driver supports Epson and Okidata +dot-matrix, Epson Stylus Color, and Epson Stylus Photo printers. +Table B-2 lists the constants for the ModelNumber directive. +ModelNumber values should be inserted by referencing +only one of these constants.

+ + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table B-2, epson driver +constants
ConstantDescription
EPSON_9PINEpson and Okidata 9-pin + dot-matrix printers
EPSON_24PINEpson and Okidata 24-pin + dot-matrix printers
EPSON_COLOROlder Epson Stylus Color + printers that use the ESC . graphics command
EPSON_PHOTOOlder Epson Stylus Photo + printers that use the ESC . graphics command
EPSON_ICOLORNewer Epson Stylus Color + printers that use the ESC i graphics command
EPSON_IPHOTONewer Epson Stylus Photo + printers that use the ESC i graphics command
+ +

The CUPS HP-PCL Sample Driver (hp)

+ +

The hp driver supports HP LaserJet and DeskJet +printers. Table B-3 lists the constants +for the ModelNumber +directive. ModelNumber values should be inserted by +referencing only one of these constants.

+ +
+ + + + + + + + + + + + + + + + + +
Table B-3, hp driver +constants
ConstantDescription
HP_LASERJETHP LaserJet printers supporting + PCL 3, 4, or 5
HP_DESKJETHP DeskJet printers + supporting PCL 3 and using the simple color graphics + command (ESC * r # U)
HP_DESKJET2HP DeskJet printers + supporting PCL3GUI and using the configure raster graphics + command (ESC * g # W)
+ +

The CUPS Label Sample Driver (label)

+ +

The label driver supports the Dymo Labelwriter, Zebra CPCL, Zebra EPL, and Zebra ZPL, and Intellitech PCL label printers. Table B-4 +lists the constants for the ModelNumber directive. +ModelNumber values should be inserted by referencing +only one of these constants.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table B-4, label driver +constants
ConstantDescription
DYMO_3x0Format output for the + Dymo Labelwriter 300, 330, or 330 Turbo.
INTELLITECH_PCLFormat output for the Intellitech PCL printers.
ZEBRA_CPCLFormat output for the Zebra CPCL printers.
ZEBRA_EPL_LINEFormat output for the Zebra EPL line mode (EPL 1) printers.
ZEBRA_EPL_PAGEFormat output for the Zebra EPL page mode (EPL 2) printers.
ZEBRA_ZPLFormat output for the Zebra ZPL printers.
+ +

The DDK ESC/P Driver (escp)

+ +

The escp driver supports all Epson inkjet printers. +Table B-6 lists the constants for the ModelNumber directive. +ModelNumber values should be specified as the bitwise +OR of one or more of these constants.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table B-6, escp driver +constants
ConstantDescription
ESCP_MICROWEAVEUse microweave command?
ESCP_STAGGERAre color jets staggered?
ESCP_ESCKUse print mode command?
ESCP_EXT_UNITSUse extended unit commands?
ESCP_EXT_MARGINSUse extended margin command?
ESCP_USBSend USB packet mode escape
ESCP_PAGE_SIZEUse page size command
ESCP_RASTER_ESCIUse ESC i graphics command
ESCP_REMOTEUse remote mode commands
ESCP_REMOTE_ACUse auto-cutter command
ESCP_REMOTE_COUse cutter-operation command
ESCP_REMOTE_EXUse media-position command
ESCP_REMOTE_MSUse media-size command
ESCP_REMOTE_MTUse media-type command
ESCP_REMOTE_PCUse paper-check command
ESCP_REMOTE_PHUse paper-thickness command
ESCP_REMOTE_PPUse paper-path command
ESCP_REMOTE_SN0Use feed-sequence-0 command
ESCP_REMOTE_SN1Use platten-gap command
ESCP_REMOTE_SN2Use feed-sequence-2 command
ESCP_REMOTE_SN6Use eject-delay command
ESCP_REMOTE_FPUse print-position command
+ +

The DDK HP-PCL Driver (pcl)

+ +

The pcl driver supports all HP LaserJet, DeskJet, +and DesignJet printers. Table B-5 lists +the constants for the ModelNumber directive. +ModelNumber values should be specified as the bitwise +OR of one or more of these constants.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table B-5, pcl driver +constants
ConstantDescription
PCL_PAPER_SIZEUse paper size command (ESC & l # A)
PCL_INKJETUse inkjet commands
PCL_RASTER_END_COLORUse new end-raster command (ESC * r C)
PCL_RASTER_CIDUse configure-image-data command (ESC * v # W)
PCL_RASTER_CRDUse configure-raster-data command (ESC * g # W)
PCL_RASTER_SIMPLEUse simple-raster-color command (ESC * r # U)
PCL_RASTER_RGB24Use 24-bit RGB mode
PCL_PJLUse PJL commands
PCL_PJL_PAPERWIDTHUse PJL PAPERWIDTH/LENGTH commands
PCL_PJL_HPGL2Use PJL ENTER HPGL2 command
PCL_PJL_PCL3GUIUse PJL ENTER PCL3GUI command
PCL_PJL_RESOLUTIONUse PJL SET RESOLUTION command
+ +

Color Keywords

+ +

The PPD compiler defines two types of color keywords: +colorspace and color order. The following sections list the +supported keywords for each type.

+ +

Colorspace Keywords

+ +

The following colorspace keywords are recognized:

+ +
    + +
  • cielab - CIE Lab **
  • + +
  • ciexyz - CIE XYZ **
  • + +
  • cmy - Cyan, magenta, yellow
  • + +
  • cmyk - Cyan, magenta, yellow, black
  • + +
  • gmck - Gold, magenta, yellow, black **
  • + +
  • gmcs - Gold, magenta, yellow, silver **
  • + +
  • gold - Gold foil **
  • + +
  • icc1 - ICC-based, 1 color **
  • + +
  • icc2 - ICC-based, 2 colors **
  • + +
  • icc3 - ICC-based, 3 colors **
  • + +
  • icc4 - ICC-based, 4 colors **
  • + +
  • icc5 - ICC-based, 5 colors **
  • + +
  • icc6 - ICC-based, 6 colors **
  • + +
  • icc7 - ICC-based, 7 colors **
  • + +
  • icc8 - ICC-based, 8 colors **
  • + +
  • icc9 - ICC-based, 9 colors **
  • + +
  • icca - ICC-based, 10 colors **
  • + +
  • iccb - ICC-based, 11 colors **
  • + +
  • iccc - ICC-based, 12 colors **
  • + +
  • iccd - ICC-based, 13 colors **
  • + +
  • icce - ICC-based, 14 colors **
  • + +
  • iccf - ICC-based, 15 colors **
  • + +
  • k - Black
  • + +
  • kcmy - Black, cyan, magenta, yellow *
  • + +
  • kcmycm - Black, cyan, magenta, yellow, light-cyan, light-magenta *
  • + +
  • rgb - Red, green, blue
  • + +
  • rgba - Red, green, blue, alpha
  • + +
  • rgbw - Red, green, blue, luminance *
  • + +
  • silver - Silver foil **
  • + +
  • w - Luminance
  • + +
  • white - White ink (as black) **
  • + +
  • ymc - Yellow, magenta, cyan *
  • + +
  • ymck - Yellow, magenta, cyan, black * + +
      + +
    * = This colorspace is not supported on Mac OS X prior to 10.4. +
    ** = This colorspace is not supported on Mac OS X.
  • + +
+ +

Color Order Keywords

+ +

The following color order keywords are recognized:

+ +
    + +
  • chunked or chunky - Color values + are passed together on a line as RGB RGB RGB RGB
  • + +
  • banded - Color values are passed separately + on a line as RRRR GGGG BBBB *
  • + +
  • planar - Color values are passed separately + on a page as RRRR RRRR RRRR ... GGGG GGGG GGGG ... BBBB + BBBB BBBB * + +
      + +
    * = This color order + is not supported by the current Apple RIP filters and + should not be used when developing printer drivers for + Mac OS X.
  • + +
+ + + diff --git a/doc/help/ref-printers-conf.html b/doc/help/ref-printers-conf.html new file mode 100644 index 0000000000..9568fc7c83 --- /dev/null +++ b/doc/help/ref-printers-conf.html @@ -0,0 +1,720 @@ + + + + printers.conf + + + + +

printers.conf

+ +

The CUPS scheduler (cupsd) uses the +/etc/cups/printers.conf file to store the list of +available printers. This file contains only locally defined +printers, not remote printers that are created automatically via +browsing. Each directive is listed on a line by itself followed +by its value. Comments are introduced using the number sign ("#") +character at the beginning of a line.

+ +

While the printer configuration file consists of plain text +and can be modified using your favorite text editor, you should +normally use the lpadmin(8) +command, web interface, or any of the available GUIs to manage +your printers instead. If you do choose to edit this file +manually, you will need to stop the scheduler first, make your +changes, and then start the scheduler to make them active.

+ + +

Accepting

+ +

Examples

+ +
+<Printer name>
+  ...
+  Accepting yes
+</Printer>
+
+ +

Description

+ +

The Accepting directive defines the initial state +of the printer-is-accepting-jobs attribute. This state +is also set by the cupsaccept(8) and +cupsreject(8) commands:

+ +
+/usr/sbin/cupsaccept printername
+/usr/sbin/cupsreject printername
+
+ +

This directive must appear inside a Printer or DefaultPrinter +section.

+ + +

AllowUser

+ +

Examples

+ +
+<Printer name>
+  ...
+  AllowUser foo_user
+  AllowUser @bar_group
+</Printer>
+
+ +

Description

+ +

The AllowUser directive adds a username or group +name to the requesting-user-name-allowed attribute +which can be set by the lpadmin(8) command:

+ +
+/usr/sbin/lpadmin -p printername -u allow:foo_user,@bar_group
+
+ +

This directive must appear inside a Printer or DefaultPrinter directive. +This directive cannot be used with DenyUser.

+ + +

DefaultPrinter

+ +

Examples

+ +
+<DefaultPrinter name>
+  ...
+</Printer>
+
+ +

Description

+ +

The DefaultPrinter directive begins a printer +definition as the default server destination. The default server +destination can be set using the lpadmin(8) +command:

+ +
+/usr/sbin/lpadmin -d printername
+
+ +

Note that the server default destination settings can be +overridden by the user's default destination settings which are +normally set using the lpoptions(1) command.

+ + +

DenyUser

+ +

Examples

+ +
+<Printer name>
+  ...
+  DenyUser foo_user
+  DenyUser @bar_group
+</Printer>
+
+ +

Description

+ +

The DenyUser directive adds a username or group +name to the requesting-user-name-denied attribute +which can be set by the lpadmin(8) command:

+ +
+/usr/sbin/lpadmin -p printername -u deny:foo_user,@bar_group
+
+ +

This directive must appear inside a Printer or DefaultPrinter directive. +This directive cannot be used with AllowUser

+ + +

DeviceURI

+ +

Examples

+ +
+<Printer name>
+  ...
+  DeviceURI socket://foo.bar.com:9100
+</Printer>
+
+ +

Description

+ +

The DeviceURI directive defines the value of the +device-uri-attribute attribute. It is normally set +using the lpadmin(8) command:

+ +
+/usr/sbin/lpadmin -p printername -v device-uri
+
+ +

This directive must appear inside a Printer or DefaultPrinter +section.

+ + +

CUPS 1.2ErrorPolicy

+ +

Examples

+ +
+<Printer name>
+  ...
+  ErrorPolicy abort-job
+</Printer>
+
+ +

Description

+ +

The ErrorPolicy directive defines the policy that +is used when a backend is unable to send a print job to the +printer. The lpadmin(8) command sets the current +error policy:

+ +
+/usr/sbin/lpadmin -p printername -o printer-error-policy=stop-printer
+
+ +

The following values are supported:

+ +
    + +
  • abort-job - Abort the job and proceed + with the next job in the queue
  • + +
  • retry-current-job - Retry the current job + immediately
  • + +
  • retry-job - Retry the job after waiting + for N seconds; the cupsd.conf JobRetryInterval + directive controls the value of N
  • + +
  • stop-printer - Stop the printer and keep + the job for future printing; this is the default + value
  • + +
+ +

This directive must appear inside a Printer or DefaultPrinter +section.

+ + +

CUPS 1.4Filter

+ +

Examples

+ +
+<Printer name>
+  ...
+  Filter mime/type 100 program
+</Printer>
+
+ +

Description

+ +

The Filter directive lists a single filter program as defined +in the printer's PPD file.

+ +

This directive must appear inside a +Printer or +DefaultPrinter section.

+ + +

Info

+ +

Examples

+ +
+<Printer name>
+  ...
+  Info My Printer
+</Printer>
+
+ +

Description

+ +

The Info directive defines the string for the +printer-info attribute. It is normally set using the +lpadmin(8) command:

+ +
+/usr/sbin/lpadmin -p printername -D "My Printer"
+
+ +

This directive must appear inside a Printer or DefaultPrinter +section.

+ + +

JobSheets

+ +

Examples

+ +
+<Printer name>
+  ...
+  JobSheets none,standard
+</Printer>
+
+ +

Description

+ +

The JobSheets directive specifies the default +banner pages to print before and after a print job. In the above +example, only a standard banner will print after each +job. The lpadmin(8) command is normally used to set +the default banners: + +

+/usr/sbin/lpadmin -p printername -o job-sheets-default=none,standard
+
+ +

If only one banner file is specified, it will be printed +before the files in the job. If a second banner file is +specified, it is printed after the files in the job.

+ +

The available banner pages depend on the local system +configuration; CUPS includes the following standard banner +files:

+ +
    + +
  • none - Do not produce a banner + page.
  • + +
  • classified - A banner page with a + "classified" label at the top and bottom.
  • + +
  • confidential - A banner page with a + "confidential" label at the top and bottom.
  • + +
  • secret - A banner page with a + "secret" label at the top and bottom.
  • + +
  • standard - A banner page with no label + at the top and bottom.
  • + +
  • topsecret - A banner page with a + "top secret" label at the top and bottom.
  • + +
  • unclassified - A banner page with an + "unclassified" label at the top and bottom.
  • + +
+ +

This directive must appear inside a Printer or DefaultPrinter +section.

+ + +

KLimit

+ +

Examples

+ +
+<Printer name>
+  ...
+  KLimit 1234
+</Printer>
+
+ +

Description

+ +

The KLimit directive defines the value of the +job-k-limit attribute. It is normally set using the +lpadmin(8) command:

+ +
+/usr/sbin/lpadmin -p printername -o job-k-limit=1234
+
+ +

This directive must appear inside a Printer or DefaultPrinter +section.

+ + +

Location

+ +

Examples

+ +
+<Printer name>
+  ...
+  Location Building 3321
+</Printer>
+
+ +

Description

+ +

The Location directive defines the string for the +printer-location attribute. It is normally set using the +lpadmin(8) command:

+ +
+/usr/sbin/lpadmin -p printername -L "Building 3321"
+
+ +

This directive must appear inside a Printer or DefaultPrinter +section.

+ + +

CUPS 1.2OpPolicy

+ +

Examples

+ +
+<Printer name>
+  ...
+  OpPolicy default
+</Printer>
+
+ +

Description

+ +

The OpPolicy directive sets the operation policy +that is used for the printer. The lpadmin(8) command +sets the current operation policy:

+ +
+/usr/sbin/lpadmin -p printername -o printer-op-policy=default
+
+ +

The default policy is named "default". All policies correspond +to those defined using the cupsd.conf Policy +section.

+ +

This directive must appear inside a Printer or DefaultPrinter +section.

+ + +

CUPS 1.2Option

+ +

Examples

+ +
+<Printer name>
+  ...
+  Option name value
+  Option scaling 100
+  Option page-left 72
+</Printer>
+
+ +

Description

+ +

The Option directive specifies a default job +template attribute value. It is mapped to +name-default in the printer attributes and applied +to jobs as name.

+ +

This directive must appear inside a Printer or DefaultPrinter +section.

+ + +

PageLimit

+ +

Examples

+ +
+<Printer name>
+  ...
+  PageLimit 1234
+</Printer>
+
+ +

Description

+ +

The PageLimit directive defines the value of the +job-page-limit attribute. It can be set using the +lpadmin(8) command:

+ +
+/usr/sbin/lpadmin -p printername -o job-page-limit=1234
+
+ +

This directive must appear inside a Printer or DefaultPrinter +section.

+ + +

CUPS 1.2PortMonitor

+ +

Examples

+ +
+<Printer name>
+  ...
+  PortMonitor bcp
+</Printer>
+
+ +

Description

+ +

The PortMonitor directive sets the filter program that +is used for every print job, typically to encode or package the print +data in a format acceptable for a particular printer interface. It is +set using the lpadmin(8) command:

+ +
+/usr/sbin/lpadmin -p printername -o port-monitor=tbcp
+
+ +

The following standard port monitors are included with CUPS:

+ +
    + +
  • bcp - Encode PostScript print data using + the Adobe Binary Control Protocol (BCP)
  • + +
  • none - Do not use a port monitor
  • + +
  • tbcp - Encode PostScript print data + using the Adobe Tagged Binary Control Protocol + (TBCP)
  • + +
+ +

This directive must appear inside a Printer or DefaultPrinter +section.

+ + +

CUPS 1.4PreFilter

+ +

Examples

+ +
+<Printer name>
+  ...
+  PreFilter mime/type 100 program
+</Printer>
+
+ +

Description

+ +

The PreFilter directive lists a single pre-filter program as +defined in the printer's PPD file.

+ +

This directive must appear inside a +Printer or +DefaultPrinter section.

+ + +

Printer

+ +

Examples

+ +
+<Printer name>
+  ...
+</Printer>
+
+ +

Description

+ +

The Printer directive begins a printer +definition. Printers are added using the lpadmin(8) +command: + +

+/usr/sbin/lpadmin -p printername ...
+
+ + +

CUPS 1.4Product

+ +

Examples

+ +
+<Printer name>
+  ...
+  Product Acme PaperWriter
+</Printer>
+
+ +

Description

+ +

The Product directive defines the main product string from the +printer's PPD file and is used when advertising the queue via DNS-SD.

+ +

This directive must appear inside a +Printer or +DefaultPrinter section.

+ + +

QuotaPeriod

+ +

Examples

+ +
+<Printer name>
+  ...
+  QuotaPeriod 604800
+</Printer>
+
+ +

Description

+ +

The QuotaPeriod directive defines the value of +the job-quota-period attribute. Typical values are +86400 (1 day), 604800 (1 week), 2592000 (1 month), and 31536000 +(1 year). It is set using the lpadmin(8) +command:

+ +
+/usr/sbin/lpadmin -p printername -o job-quota-period=604800
+
+ +

This directive must appear inside a Printer or DefaultPrinter +section.

+ + +

CUPS 1.2Shared

+ +

Examples

+ +
+<Printer name>
+  ...
+  Shared yes
+</Printer>
+
+ +

Description

+ +

The Shared directive defines the initial value of +the printer-is-shared attribute. The strings +yes and no correspond to the true and false +values, respectively. The lpadmin(8) command sets +the current state:

+ +
+/usr/sbin/lpadmin -p printername -o printer-is-shared=true
+
+ +

This directive must appear inside a Printer or DefaultPrinter +section.

+ + +

State

+ +

Examples

+ +
+<Printer name>
+  ...
+  State idle
+</Printer>
+
+ +

Description

+ +

The State directive defines the initial value of +the printer-state attribute. The strings +idle and stopped correspond to the IPP +enumeration values 3 and 5, respectively. The +cupsenable(8) and cupsdisable(8) +commands set the current state:

+ +
+/usr/sbin/cupsenable printername
+/usr/sbin/cupsdisable printername
+
+ +

This directive must appear inside a Printer or DefaultPrinter +section.

+ + +

StateMessage

+ +

Examples

+ +
+<Printer name>
+  ...
+  StateMessage Ready to print.
+</Printer>
+
+ +

Description

+ +

The StateMessage directive defines the initial +string for the printer-state-message attribute. The +following are some example messages:

+ +
+StateMessage Connected to host_name...
+StateMessage Connecting to printer_queue on port port_number...
+StateMessage Network host host_name is busy; will retry in 30 seconds...
+StateMessage Printer busy; will retry in 10 seconds...
+StateMessage Printer is busy; retrying print job...
+StateMessage Print file accepted - job ID id_number.
+StateMessage Ready to print.
+StateMessage Waiting for job to complete
+
+ +

This directive must appear inside a Printer or DefaultPrinter +section.

+ + +

CUPS 1.2StateTime

+ +

Examples

+ +
+<Printer name>
+  ...
+  StateTime 1133542425
+</Printer>
+
+ +

Description

+ +

The StateTime directive defines the UNIX time +(seconds since Jan 1, 1970) for the last state change of the +queue. It is mapped to the printer-state-change-time +attribute.

+ +

This directive must appear inside a Printer or DefaultPrinter +section.

+ + + + diff --git a/doc/help/ref-snmp-conf.html b/doc/help/ref-snmp-conf.html new file mode 100644 index 0000000000..db4aacee53 --- /dev/null +++ b/doc/help/ref-snmp-conf.html @@ -0,0 +1,146 @@ + + + + snmp.conf + + + + +

snmp.conf

+ +

The /etc/cups/snmp.conf file contains several +directives that determine how the SNMP printer discovery backend +behaves. Each directive is listed on a line by itself followed +by its value. Comments are introduced using the number sign ("#") +character at the beginning of a line.

+ +

The SNMP backend uses the SNMPv1 protocol to discover network +printers, collecting information from the Host MIB along with +intelligent port probes to determine the correct device URI and +make and model for each printer. Future versions of CUPS will +likely support the new Port Monitor MIB as well.

+ +

Address

+ +

Examples

+ +
+Address @LOCAL
+Address @IF(name)
+Address 255.255.255.255
+Address 192.168.2.255
+
+ +

Description

+ +

The Address directive specifies a broadcast +address to use when discovering printers. Multiple +Address lines can be provided to scan different +subnets.

+ +

The default address is @LOCAL, which broadcasts to +all LANs.

+ + +

Community

+ +

Examples

+ +
+Community public
+Community easysw
+Community BigCorp
+
+ +

Description

+ +

The Community directive specifies a community +name to use when discovering printers. Multiple +Community lines can be provided to scan different +SNMP communities.

+ +

The default community is "public".

+ + +

DebugLevel

+ +

Examples

+ +
+DebugLevel 0
+DebugLevel 1
+DebugLevel 2
+DebugLevel 3
+
+ +

Description

+ +

The DebugLevel directive specifies the debugging +level to use when searching for network printers. Level 0 +produces no debugging information. Level 1 produces basic +debugging information. Level 2 adds printing of the SNMP +messages. Level 3 adds a hex dump of the network data.

+ +

The default setting is 0.

+ +

DeviceURI

+ +

Examples

+ +
+DeviceURI "HP.*JetDirect.*" socket://%s:9100 socket://%s:9101 socket://%s:9102
+DeviceURI "HP.*" socket://%s
+DeviceURI "Acme.*Laser.*" lpd://%s/print
+DeviceURI "Xerox.*"
+
+ +

Description

+ +

The DeviceURI directive specifies a regular expression +(enclosed in double quotes) that is matched against the SNMP device +description OID returned by a printer. If the description matches the +regular expression, each device URI that follows the regular expression +is listed by the backend, with any occurrences of %s +replaced by the device's hostname or IP address. If no URIs are listed, +the device is ignored.

+ +

The DeviceURI directives are processed serially in +the order specified in the snmp.conf file until a match +is found.

+ + +

HostNameLookups

+ +

Examples

+ +
+HostNameLookups on
+HostNameLookups off
+
+ +

Description

+ +

The HostNameLookups directive specifies whether printer +addresses are converted to hostnames or left as numeric IP addresses.

+ +

The default setting is off.

+ +

MaxRunTime

+ +

Examples

+ +
+MaxRunTime 10
+MaxRunTime 300
+
+ +

Description

+ +

The MaxRunTime directive specifies the maximum +number of seconds that the SNMP backend will spend looking for +printer devices on the network.

+ +

The default setting is 10.

+ + + diff --git a/doc/help/ref-subscriptions-conf.html b/doc/help/ref-subscriptions-conf.html new file mode 100644 index 0000000000..d4f4defc33 --- /dev/null +++ b/doc/help/ref-subscriptions-conf.html @@ -0,0 +1,354 @@ + + + + subscriptions.conf + + + + +

subscriptions.conf

+ +

The CUPS scheduler (cupsd) uses the +/etc/cups/subscriptions.conf file to store the list +of active subscriptions. Each directive is listed on a line by +itself followed by its value. Comments are introduced using the +number sign ("#") character at the beginning of a line. + +

While the subscriptions configuration file consists of plain +text and can be modified using your favorite text editor, you +should normally use the command-line programs (lp(1) and lpr(1)) or specific applications via IPP +requests to manage your subscriptions.

+ +

Events

+ +

Examples

+ +
+<Subscription NNN>
+  ...
+  Events job-completed
+</Subscription>
+
+ +

Description

+ +

The Events directive lists the events, separated +by spaces, that the subscriber is interested in. Table 1 lists +the supported event names.

+ +

The Events directive must appear inside a Subscription section.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 1: Event Names
NameDescription
allAll events
job-completedSend notification when the job is completed
job-config-changedSend notification when the job is changed
job-createdSend notification when a job is created
job-progressSend notification for job progress
job-state-changedSend notification when the job-state changes
job-stoppedSend notification when the job is stopped
printer-addedSend notification when a printer is added
printer-changedSend notification when a printer is changed
printer-config-changedSend notification when a printer's configuration is changed
printer-deletedSend notification when a printer is deleted
printer-modifiedSend notification when a printer is modified
printer-state-changedSend notification when the printer-state changes
printer-stoppedSend notification when a printer is stopped
server-auditSend notification when a bad request, security error, or + authentication error occurs
server-restartedSend notification when the server is restarted
server-startedSend notification when the server is initially started
server-stoppedSend notification when the server is shutdown
+ + +

ExpirationTime

+ +

Examples

+ +
+<Subscription NNN>
+  ...
+  ExpirationTime 1012563145
+</Subscription>
+
+ +

Description

+ +

The ExpirationTime directive specifies the +expiration time of the subscription as a UNIX time value. It is 0 +for subscriptions with no predefined expiration time.

+ +

The ExpirationTime directive must appear inside a +Subscription +section.

+ + +

Interval

+ +

Examples

+ +
+<Subscription NNN>
+  ...
+  Interval 30
+</Subscription>
+
+ +

Description

+ +

The Interval directive specifies the preferred +time interval for event notifications in seconds.

+ +

The Interval directive must appear inside a Subscription section.

+ + +

JobId

+ +

Examples

+ +
+<Subscription NNN>
+  ...
+  JobId 123
+</Subscription>
+
+ +

Description

+ +

The JobId directive specifies the +job-id for job subscriptions.

+ +

The JobId directive must appear inside a Subscription section.

+ + +

LeaseDuration

+ +

Examples

+ +
+<Subscription NNN>
+  ...
+  LeaseDuration 
+</Subscription>
+
+ +

Description

+ +

The LeaseDuration directive specifies the number +of seconds that the subscription is valid. A value of 0 means +that the subscription will last forever or the life of the print +job the subscription is attached to.

+ +

The LeaseDuration directive must appear inside a Subscription section.

+ + +

NextEventId

+ +

Examples

+ +
+<Subscription NNN>
+  ...
+  NextEventId 999
+</Subscription>
+
+ +

Description

+ +

The NextEventId directive specifies the +notify-sequence-number value for the next +notification event. It starts at 1 and increases for every event +that is delivered for the subscription.

+ +

The NextEventId directive must appear inside a Subscription section.

+ + +

NextSubscriptionId

+ +

Examples

+ +
+NextSubscriptionId 999
+
+ +

Description

+ +

The NextSubscriptionId directive specifies the +next subscription ID to use. It defaults to 1 more than the +highest subscription number seen.

+ + +

Owner

+ +

Examples

+ +
+<Subscription NNN>
+  ...
+  Owner username
+</Subscription>
+
+ +

Description

+ +

The Owner directive specifies the user that owns +this subscription.

+ +

The Owner directive must appear inside a Subscription section.

+ + +

PrinterName

+ +

Examples

+ +
+<Subscription NNN>
+  ...
+  PrinterName name
+</Subscription>
+
+ +

Description

+ +

The PrinterName directive specifies the name of +the printer or class that is associated with this +subscription.

+ +

The PrinterName directive must appear inside a Subscription section.

+ + +

Recipient

+ +

Examples

+ +
+<Subscription NNN>
+  ...
+  Recipient mailto:foo@domain.com
+</Subscription>
+
+ +

Description

+ +

The Recipient directive specifies the +notify-recipient-uri value for push-type +notifications. The URI scheme name determines which notifier +program is used to send the event(s).

+ +

The Recipient directive must appear inside a Subscription section.

+ + +

Subscription

+ +

Examples

+ +
+<Subscription NNN>
+  ...
+</Subscription>
+
+ +

Description

+ +

The Subscription section defines a single +subscription in the system. Each subscription is assigned a +unique (to the server) number starting at 1.

+ + +

UserData

+ +

Examples

+ +
+<Subscription NNN>
+  ...
+  UserData mailto:bar<40>domain.com
+</Subscription>
+
+ +

Description

+ +

The UserData directive specifies the +notify-user-data value, which is normally the "from" +address used in mailto notifications. Binary values +are introduced by encoding the bytes as hexadecimal values inside +angle brackets, e.g. "<1234>".

+ +

The UserData directive must appear inside a Subscription section.

+ + + diff --git a/doc/help/security.html b/doc/help/security.html new file mode 100644 index 0000000000..c86151d176 --- /dev/null +++ b/doc/help/security.html @@ -0,0 +1,172 @@ + + + + Server Security + + + + +

Server Security

+ +

In the default "standalone" configuration, there are few +potential security risks - the CUPS server does not accept remote +connections, and only accepts shared printer information from the +local subnet. When you share printers and/or enable remote +administration, you expose your system to potential unauthorized +access. This help page provides an analysis of possible CUPS +security concerns and describes how to better secure your +server.

+ +

Authentication Issues

+ +

When you enable remote administration, the server will use +Basic authentication for administration tasks. The current CUPS +server supports Basic, Digest, Kerberos, and local certificate +authentication:

+ +
    + +
  1. Basic authentication essentially places the clear + text of the username and password on the network. + +

    Since CUPS uses the system username and password + account information, the authentication information could + be used to gain access to possibly privileged accounts on + the server.

    + +

    Recommendation: Enable encryption to hide the + username and password information - this is the default on + MacOS X and systems with GNU TLS or OpenSSL installed.

  2. + +
  3. Digest authentication uses an MD5 checksum of the + username, password, and domain ("CUPS"), so the original + username and password is not sent over the network. + +

    The current implementation does not authenticate the + entire message and uses the client's IP address for the + nonce value, making it possible to launch "man in the + middle" and replay attacks from the same client.

    + +

    Recommendation: Enable encryption to hide the + username and password information.

  4. + +
  5. Local certificate authentication passes 128-bit + "certificates" that identify an authenticated user. + Certificates are created on-the-fly from random data and + stored in files under /var/run/cups/certs. + They have restricted read permissions: root + + system-group(s) for the root certificate, and lp + lp + for CGI certificates. + +

    Because certificates are only available on the local + system, the CUPS server does not accept local + authentication unless the client is connected to the + loopback interface (127.0.0.1 or ::1) or domain + socket.

    + +

    Recommendation: Ensure that unauthorized users + are not added to the system group(s).

  6. + +
+ +

Denial of Service Attacks

+ +

When printer sharing or remote administration is enabled, the +CUPS server, like all Internet services, is vulnerable to a +variety of denial of service attacks:

+ +
    + +
  1. Establishing multiple connections to the server until + the server will accept no more. + +

    This cannot be protected against by any known + software. The MaxClientsPerHost directive + can be used to configure CUPS to limit the number of + connections allowed from a single host, however that does + not prevent a distributed attack.

    + +

    Recommendation: Limit access to trusted systems + and networks.

  2. + +
  3. Repeatedly opening and closing connections to the + server as fast as possible. + +

    There is no easy way of protecting against this in the + CUPS software. If the attack is coming from outside the + local network, it may be possible to filter such an + attack. However, once the connection request has been + received by the server it must at least accept the + connection to find out who is connecting.

    + +

    Recommendation: None.

  4. + +
  5. Flooding the network with broadcast packets on port + 631. + +

    It might be possible to disable browsing if this + condition is detected by the CUPS software, however if + there are large numbers of printers available on the + network such an algorithm might think that an attack was + occurring when instead a valid update was being + received.

    + +

    Recommendation: Block browse packets from + foreign or untrusted networks using a router or + firewall.

  6. + +
  7. Sending partial IPP requests; specifically, sending + part of an attribute value and then stopping + transmission. + +

    The current code will wait up to 1 second before + timing out the partial value and closing the connection. + This will slow the server responses to valid requests and + may lead to dropped browsing packets, but will otherwise + not affect the operation of the server.

    + +

    Recommendation: Block IPP packets from foreign + or untrusted networks using a router or + firewall.

  8. + +
  9. Sending large/long print jobs to printers, preventing + other users from printing. + +

    There are limited facilities for protecting against + large print jobs (the MaxRequestSize + attribute), however this will not protect printers from + malicious users and print files that generate hundreds or + thousands of pages.

    + +

    Recommendation: Restrict printer access to + known hosts or networks, and add user-level access + controls as needed for expensive printers.

  10. + +
+ +

Encryption Issues

+ +

CUPS supports 128-bit SSL 3.0 and TLS 1.0 encryption of +network connections via the OpenSSL, GNU TLS, and CDSA encryption +libraries. In additional to the potential security issues posed +by the SSL and TLS protocols, CUPS currently has the following +additional issue:

+ +
    + +
  1. Certification validation/revocation; currently CUPS + does not validate or revoke server or client certificates + when establishing a secure connection. This can + potentially lead to "man in the middle" and + impersonation/spoofing attacks over unsecured networks. + Future versions of CUPS will support both validation and + revocation of server certificates. + +

    Recommendation: Do not depend on encryption for + security when connecting to servers over the Internet or + untrusted WAN links.

  2. + +
+ + + diff --git a/doc/help/sharing.html b/doc/help/sharing.html new file mode 100644 index 0000000000..00ccb6fe87 --- /dev/null +++ b/doc/help/sharing.html @@ -0,0 +1,184 @@ + + + + Printer Sharing + + + + +

Printer Sharing

+ +

This document discusses several ways to configure printer sharing.

+ +

The Basics

+ +

A "server" is any machine that communicates directly to a printer. A "client" +is any machine that sends print jobs to a server for final printing. Clients can +also be servers if they communicate directly with any printers of their own.

+ +

By default, CUPS uses the Internet Printing Protocol (IPP) to send jobs from +a client to a server. When printing to legacy print servers you may also use the +Line Printer Daemon (LPD) when printing to older UNIX-based servers or Server +Message Block (SMB) when printing to Windows® servers.

+ +

Clients can automatically discover and access shared printers via CUPS +browsing, IPP, Service Location Protocol (SLP), and Lightweight Directory Access +Protocol (LDAP). DNS Service Discovery (DNS-SD a.k.a. Bonjour®) +and SMB browsing can also be used to manually discover and access shared +printers.

+ + +

Configuring the Server

+ +

You must enable printer sharing on the server before clients can print +through it. The simplest way to do this is to use the +cupsctl(8) command on the server:

+ +
+cupsctl --share-printers
+
+ +

By default, the above command will allow printing from other clients on the +same subnet as your server. To allow printing from any subnet, use the following +command instead:

+ +
+cupsctl --share-printers --remote-any
+
+ +

Next, you need to choose which protocols to use for printer sharing. The +default is CUPS browsing and DNS-SD on Mac OS X and CUPS browsing alone on +other platforms. To set the sharing protocols, run the cupsctl command +to set the +BrowseLocalProtocols +value. For example, run the following command to allow shared printing via +CUPS, DNS-SD, LPD, and SMB:

+ +
+cupsctl 'BrowseLocalProtocols="cups dnssd lpd smb"'
+
+ + +

Automatic Configuration using CUPS Browsing

+ +

CUPS browsing works by periodically broadcasting information about printers +that are being shared to client systems on the same subnet. Each client +maintains its own list of shared printers, and when more than one server shares +the same printer (or the same kind of printer) the client uses all of the +servers and printers to provide high-availability and failsafe printing.

+ +

To configure printers on the same subnet, do nothing. Each client +should see the available printers within 30 seconds automatically. The printer +and class lists are updated automatically as printers and servers are added or +removed.

+ +
Note: + +

Due to user interface changes in Mac OS X 10.5, CUPS shared printers will not +automatically appear in the print dialog. Instead, you must first run the +following command to enable CUPS browsing on your system:

+ +
+cupsctl BrowseRemoteProtocols=cups
+
+ +

Then choose each of the CUPS shared printers you want to see in the print +dialog by adding them, either from the Add Printer... item in the +print dialog or from the Print & Fax preference pane in the +System Preferences window.

+ +
+ +

Seeing Printers on Other Subnets

+ +

You can automatically access printers on other subnets by adding +BrowsePoll lines +to the cupsd.conf file on your local system. For a single +server you can use the cupsctl command:

+ +
+cupsctl BrowsePoll=server:port
+
+ +

For multiple servers, use the CUPS web interface (http://localhost:631/admin) +to edit the configuration file instead. Enter one BrowsePoll line +per server at the bottom of the file, as follows:

+ +
+BrowsePoll server1:port
+BrowsePoll server2:port
+BrowsePoll server3:port
+
+ +

If you have more than one client on your subnet that wants to see the +printers on those servers, add a +BrowseRely line +to the cupsd.conf file on your local system using the cupsctl +command:

+ +
+cupsctl 'BrowseRelay="127.0.0.1 @LOCAL"'
+
+ +

or CUPS web interface (again, at the bottom of the file):

+ +
+BrowseRelay 127.0.0.1 @LOCAL
+
+ + +

Automatic Configuration using IPP

+ +

CUPS can be configured to run without a local spooler and send all jobs to a +single server. However, if that server goes down then all printing will be +disabled. Use this configuration only as absolutely necessary.

+ +

The default server is normally the local system ("localhost"). To override +the default server create a file named /etc/cups/client.conf with a +line as follows:

+ +
+ServerName server
+
+ +

The server name can be the hostname or IP address of the default +server. If the server is not using the default IPP port (631), you can add the +port number at the end like this:

+ +
+ServerName server:port
+
+ +

The default server can also be customized on a per-user basis. To set a +user-specific server create a file named ~/.cups/client.conf instead. +The user client.conf file takes precedence over the system one.

+ +

Finally, you can set the CUPS_SERVER environment variable to +override the default server for a single process, for example:

+ +
+CUPS_SERVER=server:port firefox http://www.cups.org
+
+ +

will run the Firefox web browser pointed to the specified server and +port. The environment variable overrides both the user and system +client.conf files, if any.

+ + +

Manual Configuration of Print Queues

+ +

The most tedious method of configuring client machines is to configure +each remote queue by hand using the lpadmin(8) +command:

+ +
+lpadmin -p printer -E -v ipp://server/printers/printer
+
+ +

The printer name is the name of the printer on the server machine. +The server name is the hostname or IP address of the server machine. +Repeat the lpadmin command for each remote printer you wish to use.

+ + + + diff --git a/doc/help/spec-banner.html b/doc/help/spec-banner.html new file mode 100644 index 0000000000..2d22d9e8e4 --- /dev/null +++ b/doc/help/spec-banner.html @@ -0,0 +1,156 @@ + + + + + CUPS Banner File Format + + + + + + +

CUPS Banner File Format

+ +

Introduction

+ +

This specification describes the CUPS banner file format +(application/vnd.cups-banner) which is used to generate print job cover pages +and the CUPS test page. The format itself consists of a header followed by +lines of UTF-8 text containing comments or keywords and values:

+ +
+#CUPS-BANNER
+
+# What to show on the cover page
+Show job-id job-name job-originating-user-name time-at-creation
+
+# The header and footer text
+Header Cover Page
+Footer Cover Page
+
+# Arbitrary "notice" text
+Notice All work and no play makes Johnny a dull boy.
+Notice All work and no play makes Johnny a dull boy.
+Notice All work and no play makes Johnny a dull boy.
+Notice All work and no play makes Johnny a dull boy.
+
+# Images to place below the rest
+Image /usr/share/doc/cups/images/cups-icon.png
+Image /usr/share/doc/cups/images/smiley.jpg
+
+ + +

Standard Keywords

+ +

Footer

+ +

+Footer text for footer +

+ +

The Footer key defines the text that is centered at the bottom +of the page. Only one Footer key can be specified.

+ + +

Header

+ +

+Header text for Header +

+ +

The Header key defines the text that is centered at the top +of the page. Only one Header key can be specified.

+ + +

Image

+ +

+Image /path/to/image/filename
+Image relative/path/in/DocumentRoot/filename +

+ +

The Image key defines images that are centered above the footer +text. Multiple images are centered as a group from left to right. Images are +scaled as needed to fit on the page with a nominal size of 1"/25cm.

+ + +

Notice

+ +

+Notice Text to display below the job information.
+Notice More text to display below the job information. +

+ +

The Notice key defines lines of text that are centered below +the job information.

+ + +

Show

+ +

+Show value value ... value +

+ +

The Show key lists the job information that is shown. The +following values are supported:

+ +
    + +
  • imageable-area: The imageable area of the current + page size
  • + +
  • job-billing: Billing information for the job
  • + +
  • job-id: The job ID
  • + +
  • job-name: The title of the job
  • + +
  • job-originating-host-name: The computer that printed + the job
  • + +
  • job-originating-user-name: The user that printed the + job
  • + +
  • job-uuid: The job UUID
  • + +
  • options: The options that were provided with the + job
  • + +
  • paper-name: The name of the paper size used
  • + +
  • paper-size: The dimensions of the paper size used.
  • + +
  • printer-driver-name: The printer driver used
  • + +
  • printer-driver-version: The driver version
  • + +
  • printer-info: The printer description
  • + +
  • printer-location: The location of the printer
  • + +
  • printer-make-and-model: The make and model strings + reported by the printer driver
  • + +
  • printer-name: The printer used
  • + +
  • time-at-creation: When the job was submitted
  • + +
  • time-at-processing: The current date and time
  • + +
+ + + + diff --git a/doc/help/spec-browsing.html b/doc/help/spec-browsing.html new file mode 100644 index 0000000000..4583731d47 --- /dev/null +++ b/doc/help/spec-browsing.html @@ -0,0 +1,118 @@ + + + + + CUPS Browse Protocol + + + + + + +
Note: + +

The CUPS Browse Protocol is deprecated and will no longer be used in a future release of CUPS.

+ +
+ +

CUPS Browse Protocol

+ +

Introduction

+ +

This specification describes the CUPS browsing protocol which is used for +printer sharing. The protocol is a UDP/IP-based broadcast service that +operates on IP service port 631 by default. Each broadcast packet describes +a single printer or class being shared.

+ +

For simple networks with a single subnet, a CUPS system sharing a printer +(the server) will periodically broadcast that printer's availability +and status information to the subnet. Every other CUPS system on the subnet +(the clients) will receive the broadcast and make that printer +available to local users. If a client stops receiving broadcasts from the +server, or if the server sends a special "deleted" broadcast message, the +client will remove its copy of the printer.

+ +

For larger networks with multiple subnets, a relay configuration can be used +where one or more client systems poll the server and then broadcast the +availability and status information for the server's shared printers to the +clients' local subnets.

+ +

A key feature of CUPS printer sharing is support for implicit +classes, which are automatically-created classes for printers that are +shared by multiple servers. These implicit classes provide automatic load +balancing and fail-safe printing functionality transparently to the user.

+ + +

Security Considerations

+ +

Like most discovery protocols, CUPS browse packets are not encrypted or +signed, so it is possible for malicious systems on a network to advertise +or remove printers on that network to cause denial of service or information +disclosure. In order to combat this, CUPS logs incoming browse packets and +provides access controls to limit browse packet reception to known hosts.

+ + +

Browse Packet Format

+ +

Each broadcast packet is an ASCII text string of up to 1450 bytes ending +with a line feed (0x0a). The general format is:

+ +

+printer-type printer-state printer-uri "printer-location" "printer-info" +"printer-make-and-model" name=value name2=value2 ... +

+ +

Each of the fields contains the value of the corresponding IPP attribute. +The trailing "name=value" information is used to convey default job template +attribute values (job-sheets-default, media-default, etc.), authentication +requirements (auth-info-required), and additional IPP URI options that are +requested by the server (ipp-options).

+ +

ABNF Definition

+ +

The following ABNF definition [RFC4234, RFC3986] defines the format of each +browse packet:

+ +
+PACKET         = TYPE WSP STATE WSP URI WSP LOCATION WSP INFO WSP
+                 MAKE-AND-MODEL WSP *[ WSP ATTR-NAME "=" ATTR-VALUE ] LF
+
+TYPE           = 1*HEXDIG
+
+STATE          = "3" / "4" / "5"
+
+URI            = "ipp://" ( 1*NAMECHAR / IP-literal / IPv4address )
+                 [ ":" 1*DIGIT ] ( "/printers/" / "/classes/" ) 1*NAMECHAR
+NAMECHAR       = %x21.22.24.26-2E.30-7E / %x25 HEXDIG HEXDIG
+IP-literal     = See RFC 3986
+IPv4address    = See RFC 3986
+
+LOCATION       = QUOTED-STRING
+INFO           = QUOTED-STRING
+MAKE-AND-MODEL = QUOTED-STRING
+
+ATTR-NAME      = 1*( ALPHA / DIGIT / "-" / "." )
+ATTR-VALUE     = QUOTED-STRING / 1*UNQUOTE-CHAR
+
+QUOTED-STRING  = DQUOTE *QUOTED-CHAR DQUOTE
+QUOTED-CHAR    = %x20.21.23-5B.5D-7E / UTF8-CHAR / %x5C %x5C / %x5C %x22
+UNQUOTE-CHAR   = %x21.23-26.28-5B.5D-7E / UTF8-CHAR
+UTF8-CHAR      = %xC0.DF %x80.BF / %xE0.EF %x80.BF %x80.BF /
+                 %xF0.F7 %x80.BF %x80.BF %x80.BF
+
+ + + diff --git a/doc/help/spec-cmp.html b/doc/help/spec-cmp.html new file mode 100644 index 0000000000..47f56d31d3 --- /dev/null +++ b/doc/help/spec-cmp.html @@ -0,0 +1,1218 @@ + + + + + CUPS Developer Guide + + + + +

CUPS Developer Guide

+ +

This developer guide documents the guidelines and processes we use when developing and maintaining CUPS and related software. Our goal is to provide reliable and efficient software and documentation that addresses the needs of our users.

+ +

Communication

+ +

How to Contact the Developers

+ +

The CUPS +Forums are the primary means of asking questions and +informally discussing issues and feature requests with the CUPS +developers. Table 1 shows the available forums and their +focus:

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + +
Table 1: CUPS Forums
ForumFocus/Purpose
cups.bugsDiscussion of bugs and issues in the CUPS + software
cups.commitReport of all commits to the Subversion repository + (read-only)
cups.ddkUsage and development questions for the CUPS Driver + Development Kit
cups.developmentDevelopment questions and discussion of new features + in the CUPS software
cups.generalUsage questions for the CUPS software
+ +

How to Submit a Bug Report or Feature Request

+ +

The CUPS "Bugs & +Features" page provides access to the CUPS software +trouble report database and is the formal way to submit a +bug report or feature request to the CUPS developers. Please +note, however, that we do not provide answers to usage +questions or resolve problems in third-party software on this +page - use the CUPS Forums for that instead.

+ +

Unlike discussions that occur on the CUPS Forums, formal bug +reports and feature requests must be acted on by the CUPS +developers. This does not mean that every bug report is resolved +or every feature request is implemented, but we do respond and +keep track of them all for posterity.

+ +
Please use the search feature of the Bugs & +Features page before submitting a new bug report or feature +request. If you see an existing report that matches your issue, +please post a message to that report ("I have this issue as +well", "I would also like to see", etc.) rather than submitting a +new report. This helps speed the resolution of your issue by +reducing the CUPS developers' work load.
+ +

How to Prepare a Patch

+ +

When submitting a bug report or feature request, you can +include patch files that resolve the bug or implement the feature +to speed the inclusion of that bug fix or feature in a new CUPS +release. For changes to existing files, we prefer a unified diff +against the current Subversion trunk branch, which can +be generated easily using the following Subversion command:

+ +
+svn diff >filename.patch
+
+ +

If you produce a patch using a released source archive, use +one of the following commands instead:

+ +
+diff -u oldfilename filename >filename.patch
+
+diff -urN olddirectory directory >filename.patch
+
+ +

New files and files with significant changes can be submitted +in their entirety, however that may delay the adoption of your +changes.

+ +
Patches and files must conform to the standards outlined in the +"Coding Guidelines" and "Makefile +Guidelines" sections in this document. In addition, since Apple Inc. +provides CUPS under multiple licenses, we require that you assign the copyright +for your changes and files to us for inclusion in CUPS.
+ + +

Software Development Practices

+ +

Version Numbering

+ +

CUPS uses a three-part version number separated by periods to +represent the major, minor, and patch release numbers. Major +release numbers indicate large design changes or +backwards-incompatible changes to the CUPS API or CUPS Imaging +API. Minor release numbers indicate new features and other +smaller changes which are backwards-compatible with previous CUPS +releases. Patch numbers indicate bug fixes to the previous +release.

+ +
When we talk about compatibility, we are talking +about binary compatibility for public APIs and output format +compatibility for program interfaces. Changes to configuration +file formats or the default behavior of programs are not +generally considered incompatible as the upgrade process can +normally address such changes gracefully.
+ +

Production releases use the plain version numbers:

+ +
+MAJOR.MINOR.PATCH
+1.0.0
+1.0.1
+1.0.2
+...
+1.1.0
+...
+1.1.23
+1.2.0
+1.2.1
+...
+1.3.0
+...
+2.0.0
+
+ +

The first production release in a MAJOR.MINOR series (MAJOR.MINOR.0) is +called a feature release. Feature releases are the only releases that may +contain new features. Subsequent production releases in a MAJOR.MINOR series may +only contain bug fixes.

+ +
We did not hold to this limitation in the CUPS 1.1 series for a +variety of reasons. Starting with CUPS 1.2, the "no new features in a patch +release" policy has been strictly enforced. The policy has also resulted in +fewer new features (and interactions!) to validate/test in the subsequence +feature releases.
+ +

Beta-test releases are identified by appending the letter B to the major and +minor version numbers followed by the beta release number:

+ +
+MAJOR.MINORbNUMBER
+1.2b1
+
+ +

Release candidates are identified by appending the letters RC to the major +and minor version numbers followed by the release candidate number:

+ +
+MAJOR.MINORrcNUMBER
+1.2rc1
+
+ +

Developer snapshots are identified by appending the letters SVN-R to the +major and minor version numbers followed by the revision number:

+ +
+MAJOR.MINORsvn-rREV
+1.2svn-r1234
+
+ +

Beta-test releases, release candidates, and developer snapshots are only +created for new minor releases. Once a production release has been made +(MAJOR.MINOR.0), subsequent patch releases are issues without preliminary beta +or release testing.

+ +

Version Control (Subversion)

+ +

The CUPS source files are managed by the Subversion ("SVN") +software, available at:

+ +
+subversion.tigris.org
+
+ +

Source files are "checked in" with each change so that +modifications can be tracked, and each checkin must reference any +applicable STRs. The following format must be used for +commit log messages:

+ +
+Summary of the change on one line followed by bug number (STR #NNNN)
+
+Detailed list of changes.
+
+ +

Primary development occurs on the trunk branch, +with changes merged back to release branches as needed. Table 2 +shows the URLs developers use for the various CUPS sub-projects +and branches:

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 2: CUPS Subversion URLs
URLPurpose
https://svn.easysw.com/public/cups/trunk/Primary CUPS development branch
https://svn.easysw.com/public/cups/branches/CUPS maintenance branches (merge-only)
https://svn.easysw.com/public/cups/tags/CUPS release tags (read-only)
https://svn.easysw.com/public/windows/trunk/Primary CUPS Windows Driver development branch
https://svn.easysw.com/public/windows/branches/CUPS Windows Driver maintenance branches (merge-only)
https://svn.easysw.com/public/windows/tags/CUPS Windows Driver release tags (read-only)
+ +

The branch for a MAJOR.MINOR release are created when the +first production release (MAJOR.MINOR.0) is made using the name +"branch-MAJOR.MINOR". Release tags are created for every beta, +candidate, and production release using the name +"release-MAJOR.MINOR.PATCHbNUMBER", +"release-MAJOR.MINOR.PATCHrcNUMBER", or +"release-MAJOR.MINOR.PATCH", respectively. No release tags are +created for developer snapshots.

+ + +

Files and Directories

+ +

File and directory names may not exceed 16 characters in +length to ensure compatibility with older UNIX filesystems. In +addition, to avoid problems with case-insensitive filesystems, +you may not use names which differ only by case, for example +"ReadMe" and "README" are not allowed in the same directory.

+ +

Source files must be documented and formatted as described in +"Coding Requirements". Make files must +follow the guidelines in "Makefile +Guidelines".

+ + +

Build System

+ +

The CUPS build system uses GNU autoconf to +tailor the library to the local operating system. Project files +for major IDEs are also provided for Microsoft +Windows®. To improve portability, makefiles must +not make use of the unique features offered by GNU make. See the Makefile Guidelines section for a +description of the allowed make features and makefile +guidelines.

+ +

Additional GNU build programs such as GNU automake and +GNU libtool +must not be used. GNU automake produces non-portable makefiles +which depend on GNU-specific extensions, and GNU libtool is not +portable or reliable enough for CUPS.

+ + +

Packaging

+ +

Source packages are created using the +tools/makesrcdist script in the Subversion repository. +The script optionally uses a version number argument:

+ +
+tools/makesrcdist
+tools/makesrcdist version
+
+ +

When run with no arguments, the script creates a snapshot of +the current working copy and names it using the highest revision +number in the WC, for example +"/tmp/cups-1.3svn-r1234-source.tar.bz2" and +"/tmp/cups-1.3svn-r1234-source.tar.gz". When run with two +arguments, the script creates a release tag in the repository and +exports that tag, creating the files +"/tmp/cups-version-source.tar.bz2" and +"/tmp/cups-version-source.tar.gz".

+ +

Binary packages are not generally distributed by the CUPS +team, however the packaging/cups.spec and +packaging/cups.list files may be used to create binary +packages on Linux, Mac OS X, and UNIX. The +packaging/cups.spec file produces a binary package +using the rpmbuild(8) software:

+ +
+rpmbuild -ta cups-version-source.tar.gz
+
+ +

The cups.list file is generated by the +configure script and produces binary packages for many +platforms using the EPM software. Table 3 shows the targets that +are available for each type of binary package:

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 3: Binary Package Targets
TargetType of Package
aixAIX installp
bsd*BSD pkg_install
debDebian dpkg
depotHP-UX swinstall
epmPortable tarball with install script
instIRIX inst/tardist
osxMac OS X Install
pkgSolaris pkgadd
rpmRPM binary
setldTru64 UNIX setld
slackwareSlackware install
swinstallHP-UX swinstall
tardistIRIX inst/tardist
+ +

Finally, the tools/testrpm and +tools/testosx scripts can be used to create binary +packages from the current working copy for testing on Linux and +Mac OS X, respectively:

+ +
+tools/testrpm
+sudo rpm -U /usr/src/redhat/RPMS/i386/cups*.rpm
+
+sudo tools/testosx
+open cups.pkg
+
+ + +

Testing

+ +

Software testing is conducted according to the CUPS Software Test Plan. This testing is +automated via the top-level makefile test target:

+ +
+make test
+
+ +

The test environment allows for both short-term automated +testing and long-term testing and development without the +automated test script.

+ + +

Trouble Report Processing

+ +

A Software Trouble Report ("STR") must be submitted every time +a user or vendor experiences a problem with the CUPS software. +Trouble reports are maintained on the Bugs & +Features page with one of the following states:

+ +
    + +
  1. STR is closed with complete resolution
  2. + +
  3. STR is closed without resolution
  4. + +
  5. STR is active, waiting on information from submitter
  6. + +
  7. STR is pending with additional information from submitter
  8. + +
  9. STR is newly submitted
  10. + +
+ +

Trouble reports are processed using the following steps.

+ +
    + +
  1. Classification + +

    When a trouble report is received it must be classified at one +of the following priority levels:

    + +
      + +
    1. Request for enhancement, e.g. asking for a + feature + +
    2. Low, e.g. a documentation error or undocumented + side-effect + +
    3. Moderate, e.g. unable to print a file or unable to + compile the software + +
    4. High, e.g. unable to print to a printer or key + functionality not working + +
    5. Critical, e.g. unable to print at all + +
    + +

    Level 4 and 5 trouble reports must be resolved in the next +software release. Level 2 and 3 trouble reports are scheduled for +resolution in a specific release at the discretion of the release +coordinator. Level 1 trouble reports are scheduled for resolution +in a future feature release.

    + +

    The scope of the problem is also determined as:

    + +
      + +
    1. Specific to a machine or printer + +
    2. Specific to an operating system + +
    3. Applies to all machines, printers, and operating systems + +
    + +
  2. Identification + +

    Once the level and scope of the trouble report is determined +the software sub-system(s) involved with the problem are +determined. This may involve additional communication with the +user or vendor to isolate the problem to a specific cause.

    + +

    When the sub-system(s) involved have been identified, an +engineer will then determine the change(s) needed and estimate +the time required for the change(s).

    + +
  3. Correction + +

    Corrections are scheduled based upon the severity and +complexity of the problem. Once all changes have been made, +documented, and tested successfully a new software release +snapshot is generated. Additional tests are added as necessary +for proper testing of the changes.

    + +
  4. Notification + +

    The user or vendor is notified when the fix is available or if +the problem was caused by user error.

    + +
+ + +

Release Management

+ +

When testing has been completed successfully, a new source +package is created using the tools/makesrcdist script. +Three types of releases, beta, candidate, and production, are +created and released to the public using the basic schedule in +Table 4. At least one beta and one release candidate must be +created prior to a production release, and there must be at least +two weeks between the last beta and first candidate and last +candidate and first production release.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table: CUPS Basic Release Schedule
WeekVersionDescription
T-6 weeks1.2b1First beta release
T-5 weeks1.2b2Second beta release
T-3 weeks1.2rc1First release candidate
T-2 weeks1.2rc2Second release candidate
T-0 weeks1.2.0Production (feature) release
+ + +

Coding Guidelines

+ +

These coding guidelines provide detailed information on source +file formatting and documentation content and must be applied to +all C and C++ source files provided with CUPS. Source code for +other languages should conform to these guidelines as allowed by +the language.

+ +

Source Files

+ +

All source files names shall be 16 characters or less in +length to ensure compatibility with older UNIX filesystems. +Source files containing functions shall have an extension of ".c" +for ANSI C and ".cxx" for C++ source files. All other "include" +files shall have an extension of ".h".

+ +

The top of each source file shall contain a header giving the +name of the file, the purpose or nature of the source file, the +copyright and licensing notice, and the functions contained in +the file. The file name and revision information is provided by +the Subversion "$Id$" tag:

+ +
+/*
+ * "$Id$"
+ *
+ *   Description of file contents.
+ *
+ *   Copyright 2010 by Apple Inc.
+ *
+ *   These coded instructions, statements, and computer programs are the
+ *   property of Apple Inc. and are protected by Federal copyright
+ *   law.  Distribution and use rights are outlined in the file "LICENSE.txt"
+ *   which should have been included with this file.  If this file is
+ *   file is missing or damaged, see the license at "http://www.cups.org/".
+ *
+ * Contents:
+ *
+ *   function1() - Description 1.
+ *   function2() - Description 2.
+ *   function3() - Description 3.
+ */
+
+ +

For source files that are subject to the Apple OS-Developed +Software exception, the following additional comment should +appear after the contact information:

+ +
+ *   This file is subject to the Apple OS-Developed Software exception.
+
+ +

The bottom of each source file shall contain a trailer giving +the name of the file using the Subversion "$Id$" tag. The +primary purpose of this is to mark the end of a source file; if +the trailer is missing it is possible that code has been lost +near the end of the file:

+ +
+/*
+ * End of "$Id$".
+ */
+
+ +

Functions

+ +

Functions with a global scope shall have a lowercase prefix +followed by capitalized words ("cupsDoThis", "cupsDoThat", +"cupsDoSomethingElse", etc.) Private global functions shall begin +with a leading underscore ("_cupsDoThis", "_cupsDoThat", +etc.)

+ +

Functions with a local scope shall be declared "static" and be +lowercase with underscores between words ("do_this", "do_that", +"do_something_else", etc.)

+ +

Each function shall begin with a comment header describing +what the function does, the possible input limits (if any), and +the possible output values (if any), and any special information +needed:

+ +
+/*
+ * 'do_this()' - Compute y = this(x).
+ *
+ * Notes: none.
+ */
+
+static float                            /* O - Inverse power value, 0.0 <= y <= 1.1 */
+do_this(float x)                        /* I - Power value (0.0 <= x <= 1.1) */
+{
+  ...
+  return (y);
+}
+
+ +

Return/output values are indicated using an "O" prefix, input +values are indicated using the "I" prefix, and values that are +both input and output use the "IO" prefix for the corresponding +in-line comment.

+ +

The Mini-XML documentation generator also understands the following +special text in the function description comment:

+ +
    + +
  • @deprecated@ - Marks the function as + deprecated (not recommended for new development and + scheduled for removal)
  • + +
  • @since CUPS version@ - Marks the + function as new in the specified version of CUPS.
  • + +
  • @private@ - Marks the function as private.
  • + +
+ +

Variables

+ +

Variables with a global scope shall be capitalized +("ThisVariable", "ThatVariable", "ThisStateVariable", etc.) The +only exception to this rule shall be the CUPS interface library +global variables which must begin with the prefix "cups" +("cupsThisVariable", "cupsThatVariable", etc.) Global variables +shall be replaced by function arguments whenever possible.

+ +

Variables with a local scope shall be lowercase with +underscores between words ("this_variable", "that_variable", +etc.) Any local variables shared by functions within a source +file shall be declared "static".

+ +

Each variable shall be declared on a separate line and shall +be immediately followed by a comment block describing the +variable:

+ +
+int this_variable;   /* The current state of this */
+int that_variable;   /* The current state of that */
+
+ +

Types

+ +

All type names shall be lowercase with underscores between +words and "_t" appended to the end of the name +("cups_this_type_t", "cups_that_type_t", etc.) Type names must +start with a prefix, typically "cups" or the name of the program, +to avoid conflicts with system types. Private type names must +start with an underscore ("_cups_this_t", "_cups_that_t", +etc.)

+ +

Each type shall have a comment block immediately after the +typedef:

+ +
+typedef int cups_this_type_t;           /* This type is for CUPS foobar options. */
+
+ +

Structures

+ +

All structure names shall be lowercase with underscores +between words and "_s" appended to the end of the name +("cups_this_s", "cups_that_s", etc.) Structure names must start +with a prefix, typically "cups" or the name of the program, to +avoid conflicts with system types. Private structure names must +start with an underscore ("_cups_this_s", "_cups_that_s", +etc.)

+ +

Each structure shall have a comment block immediately after +the struct and each member shall be documented in accordance with +the variable naming policy above:

+ +
+struct cups_this_struct_s               /* This structure is for CUPS foobar options. */
+{
+  int this_member;                      /* Current state for this */
+  int that_member;                      /* Current state for that */
+};
+
+ +

Constants

+ +

All constant names shall be uppercase with underscored between +words ("CUPS_THIS_CONSTANT", "CUPS_THAT_CONSTANT", etc.) +Constants must begin with an uppercase prefix, typically "CUPS" +or the program name.

+ +

Typed enumerations shall be used whenever possible to allow +for type checking by the compiler.

+ +

Comment blocks shall immediately follow each constant:

+ +
+enum
+{
+  CUPS_THIS_TRAY,                       /* This tray */
+  CUPS_THAT_TRAY                        /* That tray */
+};
+
+ +

Code

+ +

All source code shall utilize block comments within functions +to describe the operations being performed by a group of +statements; avoid putting a comment per line unless absolutely +necessary, and then consider refactoring the code so that it is +not necessary:

+ +
+/*
+ * Clear the state array before we begin...
+ */
+
+for (i = 0; i < (sizeof(array) / sizeof(sizeof(array[0])); i ++)
+  array[i] = STATE_IDLE;
+
+/*
+ * Wait for state changes...
+ */
+
+do
+{
+  for (i = 0; i < (sizeof(array) / sizeof(sizeof(array[0])); i ++)
+    if (array[i] != STATE_IDLE)
+      break;
+
+  if (i == (sizeof(array) / sizeof(array[0])))
+    sleep(1);
+} while (i == (sizeof(array) / sizeof(array[0])));
+
+ +

Indentation

+ +

All code blocks enclosed by brackets shall begin with the +opening brace on a new line. The code then follows starting on a +new line after the brace and is indented 2 spaces. The closing +brace is then placed on a new line following the code at the +original indentation:

+ +
+{
+  int i; /* Looping var */
+
+ /*
+  * Process foobar values from 0 to 999...
+  */
+
+  for (i = 0; i < 1000; i ++)
+  {
+    do_this(i);
+    do_that(i);
+  }
+}
+
+ +

Single-line statements following "do", "else", "for", "if", +and "while" shall be indented 2 spaces as well. Blocks of code +in a "switch" block shall be indented 4 spaces after each "case" +and "default" case:

+ +
+switch (array[i])
+{
+  case STATE_IDLE :
+      do_this(i);
+      do_that(i);
+      break;
+  default :
+      do_nothing(i);
+      break;
+}
+
+ +

Spacing

+ +

A space shall follow each reserved word ("if", "while", etc.) +Spaces shall not be inserted between a function name and the +arguments in parenthesis.

+ +

Return Values

+ +

Parenthesis shall surround values returned from a function +using "return":

+ +
+return (CUPS_STATE_IDLE);
+
+ +

Loops

+ +

Whenever convenient loops should count downward to zero to +improve program performance:

+ +
+for (i = sizeof(array) / sizeof(array[0]) - 1; i >= 0; i --)
+  array[i] = CUPS_STATE_IDLE;
+
+ +

Makefile Guidelines

+ +

The following is a guide to the makefile-based build system +used by CUPS. These standards have been developed over the years +to allow CUPS to be built on as many systems and environments as +possible.

+ +

General Organization

+ +

The CUPS source code is organized functionally into a +top-level makefile, include file, and subdirectories each with +their own makefile and dependencies files. The ".in" files are +template files for the autoconf software and are +used to generate a static version of the corresponding file.

+ +

Makefile Documentation

+ +

Each make file must start with the standard CUPS header +containing the Subversion "$Id$" keyword, description of the +file, and CUPS copyright and license notice:

+ +
+#
+# "$Id$"
+#
+#   Makefile for ...
+#
+#   Copyright 2007 by Apple Inc.
+#
+#   These coded instructions, statements, and computer programs are the
+#   property of Apple Inc. and are protected by Federal copyright
+#   law.  Distribution and use rights are outlined in the file "LICENSE.txt"
+#   which should have been included with this file.  If this file is
+#   file is missing or damaged, see the license at "http://www.cups.org/".
+#
+
+ +

The end of each makefile must have a comment saying:

+ +
+#
+# End of "$Id$".
+#
+
+ +

The purpose of the trailer is to indicate the end of the +makefile so that truncations are immediately obvious.

+ +

Portable Makefile Construction

+ +

CUPS uses a common subset of make program syntax to ensure +that the software can be compiled "out of the box" on as many +systems as possible. The following is a list of assumptions we +follow when constructing makefiles:

+ +
    + +
  • Targets; we assume that the make program + supports the notion of simple targets of the form + "name:" that perform tab-indented commands that follow + the target, e.g.: +
    +target:
    +→ target commands
  • + +
  • Dependencies; we assume that the make program + supports recursive dependencies on targets, e.g.: +
    +target: foo bar
    +→ target commands
    +
    +foo: bla
    +→ foo commands
    +
    +bar:
    +→ bar commands
    +
    +bla:
    +→ bla commands
  • + +
  • Variable Definition; we assume that the make program + supports variable definition on the command-line or in the makefile + using the following form: +
    +name=value
    + +
  • Variable Substitution; we assume that the make program + supports variable substitution using the following forms: +
      +
    • $(name); substitutes the value of "name",
    • +
    • ($name:.old=.new); substitutes the value of "name" + with the filename extensions ".old" changed to ".new",
    • +
    • $(MAKEFLAGS); substitutes the + command-line options passed to the program + without the leading hyphen (-),
    • +
    • $$; substitutes a single $ character,
    • +
    • $<; substitutes the current source file or dependency, and
    • +
    • $@; substitutes the current target name.
    • +
  • + +
  • Suffixes; we assume that the make program + supports filename suffixes with assumed dependencies, e.g.: +
    +.SUFFIXES: .c .o
    +.c.o:
    +→ $(CC) $(CFLAGS) -o $@ -c $<
  • + +
  • Include Files; we assume that the make program + supports the include directive, e.g.: +
    +include ../Makedefs
    +include Dependencies
  • + +
  • Comments; we assume that comments begin with + a # character and proceed to the end of the + current line.
  • + +
  • Line Length; we assume that there is no + practical limit to the length of lines.
  • + +
  • Continuation of long lines; we assume that + the \ character may be placed at the end of a + line to concatenate two or more lines in a + makefile to form a single long line.
  • + +
  • Shell; we assume a POSIX-compatible shell is + present on the build system.
  • + +
+ +

Standard Variables

+ +

The following variables are defined in the "Makedefs" file +generated by the autoconf software:

+ +
    + +
  • AR; the library archiver command,
  • + +
  • ARFLAGS; options for the library archiver command,
  • + +
  • BUILDROOT; optional installation prefix,
  • + +
  • MAN1EXT; extension for man pages in section 1,
  • + +
  • MAN3EXT; extension for man pages in section 3,
  • + +
  • MAN5EXT; extension for man pages in section 5,
  • + +
  • MAN7EXT; extension for man pages in section 7,
  • + +
  • MAN8DIR; subdirectory for man pages in section 8,
  • + +
  • MAN8EXT; extension for man pages in section 8,
  • + +
  • CC; the C compiler command,
  • + +
  • CFLAGS; options for the C compiler command,
  • + +
  • CXX; the C++ compiler command,
  • + +
  • CXXFLAGS; options for the C++ compiler command,
  • + +
  • DSOCOMMAND; the shared library building command,
  • + +
  • DSOFLAGS; options for the shared library building command,
  • + +
  • INSTALL; the install command,
  • + +
  • INSTALL_BIN; the program installation command,
  • + +
  • INSTALL_DATA; the data file installation command,
  • + +
  • INSTALL_DIR; the directory installation command,
  • + +
  • INSTALL_LIB; the library installation command,
  • + +
  • INSTALL_MAN; the documentation installation command,
  • + +
  • INSTALL_SCRIPT; the shell script installation command,
  • + +
  • LDFLAGS; options for the linker,
  • + +
  • LIBS; libraries for all programs,
  • + +
  • LN; the ln command,
  • + +
  • OPTIM; common compiler optimization options,
  • + +
  • RM; the rm command,
  • + +
  • SHELL; the sh (POSIX shell) command,
  • + +
  • STRIP; the strip command,
  • + +
  • bindir; the binary installation directory,
  • + +
  • datadir; the data file installation directory,
  • + +
  • exec_prefix; the installation prefix for executable files,
  • + +
  • libdir; the library installation directory,
  • + +
  • mandir; the man page installation directory,
  • + +
  • prefix; the installation prefix for non-executable files, and
  • + +
  • srcdir; the source directory.
  • + +
+ +

Standard Targets

+ +

The following standard targets must be defined in each +makefile:

+ +
    + +
  • all; creates all target programs, + libraries, and documentation files,
  • + +
  • clean; removes all target programs, + libraries, documentation files, and object files,
  • + +
  • depend; generates automatic dependencies + for any C or C++ source files (also see "Dependencies"),
  • + +
  • distclean; removes autoconf-generated files + in addition to those removed by the "clean" target,
  • + +
  • install; installs all distribution files in + their corresponding locations (also see "Install/Uninstall Support"),
  • + +
  • uninstall; removes all distribution files from + their corresponding locations (also see "Install/Uninstall Support"), and
  • + +
+ + +

Object Files

+ +

Object files (the result of compiling a C or C++ source file) +have the extension ".o".

+ +

Programs

+ +

Program files are the result of linking object files and +libraries together to form an executable file. A typical +program target looks like:

+ +
+program: $(OBJS)
+→ echo Linking $@...
+→ $(CC) $(LDFLAGS) -o $@ $(OBJS) $(LIBS)
+
+ +

Static Libraries

+ +

Static libraries have a prefix of "lib" and the extension +".a". A typical static library target looks like:

+ +
+libname.a: $(OBJECTS)
+→ echo Creating $@...
+→ $(RM) $@
+→ $(AR) $(ARFLAGS) $@ $(OBJECTS)
+→ $(RANLIB) $@
+
+ +

Shared Libraries

+ +

Shared libraries have a prefix of "lib" and the extension +".dylib", ".sl", ".so", or "_s.a" depending on the operating +system. A typical shared library is composed of several targets +that look like:

+ +
+libname.so: $(OBJECTS)
+→ echo $(DSOCOMMAND) libname.so.$(DSOVERSION) ...
+→ $(DSOCOMMAND) libname.so.$(DSOVERSION) $(OBJECTS)
+→ $(RM) libname.so libname.so.$(DSOMAJOR)
+→ $(LN) libname.so.$(DSOVERSION) libname.so.$(DSOMAJOR)
+→ $(LN) libname.so.$(DSOVERSION) libname.so
+
+libname.sl: $(OBJECTS)
+→ echo $(DSOCOMMAND) libname.sl.$(DSOVERSION) ...
+→ $(DSOCOMMAND) libname.sl.$(DSOVERSION) $(OBJECTS)
+→ $(RM) libname.sl libname.sl.$(DSOMAJOR)
+→ $(LN) libname.sl.$(DSOVERSION) libname.sl.$(DSOMAJOR)
+→ $(LN) libname.sl.$(DSOVERSION) libname.sl
+
+libname.dylib: $(OBJECTS)
+→ echo $(DSOCOMMAND) libname.$(DSOVERSION).dylib ...
+→ $(DSOCOMMAND) libname.$(DSOVERSION).dylib \
+→ → -install_name $(libdir)/libname.$(DSOMAJOR).dylib \
+→ → -current_version libname.$(DSOVERSION).dylib \
+→ → -compatibility_version $(DSOMAJOR).0 \
+→ → $(OBJECTS) $(LIBS)
+→ $(RM) libname.dylib
+→ $(RM) libname.$(DSOMAJOR).dylib
+→ $(LN) libname.$(DSOVERSION).dylib libname.$(DSOMAJOR).dylib
+→ $(LN) libname.$(DSOVERSION).dylib libname.dylib
+
+libname_s.a: $(OBJECTS)
+→ echo $(DSOCOMMAND) libname_s.o ...
+→ $(DSOCOMMAND) libname_s.o $(OBJECTS) $(LIBS)
+→ echo $(LIBCOMMAND) libname_s.a libname_s.o
+→ $(RM) $@
+→ $(LIBCOMMAND) libname_s.a libname_s.o
+→ $(CHMOD) +x libname_s.a
+
+ + +

Dependencies

+ +

Static dependencies are expressed in each makefile following the +target, for example:

+ +
+foo: bar
+
+ +

Static dependencies shall only be used when it is not +possible to automatically generate them. Automatic dependencies +are stored in a file named "Dependencies" and included at the +end of the makefile. The following "depend" target rule shall be +used to create the automatic dependencies: + +

+depend:
+→ $(MAKEDEPEND) -Y -I.. -f Dependencies $(OBJS:.o=.c)
+
+ +

We only regenerate the automatic dependencies on a Linux +system and express any non-Linux dependencies statically in the +makefile.

+ +

Install/Uninstall Support

+ +

All makefiles must contain install and uninstall rules which +install or remove the corresponding software. These rules must +use the $(BUILDROOT) variable as a prefix to any +installation directory so that CUPS can be installed in a +temporary location for packaging by programs like +rpmbuild.

+ +

The $(INSTALL_BIN), $(INSTALL_DATA), +$(INSTALL_DIR), $(INSTALL_LIB), +$(INSTALL_MAN), and $(INSTALL_SCRIPT) +variables must be used when installing files so that the proper +ownership and permissions are set on the installed files.

+ +

The $(RANLIB) command must be run on any static +libraries after installation since the symbol table is +invalidated when the library is copied on some platforms.

+ + + diff --git a/doc/help/spec-command.html b/doc/help/spec-command.html new file mode 100644 index 0000000000..af201cb55c --- /dev/null +++ b/doc/help/spec-command.html @@ -0,0 +1,218 @@ + + + + + CUPS Command File Format + + + + + + +

CUPS Command File Format

+ +

Introduction

+ +

This specification describes the CUPS command file format +(application/vnd.cups-command) which is used to send printer +maintenance commands to a printer in a device-independent way. +The current specification supports basic maintenance functions +such as head cleaning and self-test pages and query functions +such as auto-configure, report supply levels, and report status.

+ +

Printer drivers advertise support for the CUPS command file +format by providing a filter for the +application/vnd.cups-command file type. Applications +can determine if a printer supports printing of CUPS command +files by checking the printer-type attribute for the +CUPS_PRINTER_COMMANDS capability bit.

+ +

In addition, the PPD file for a printer can contain a +cupsCommands keyword that provides a list of supported +commands separated by spaces, for example:

+ +
+*cupsCommand: "AutoConfigure Clean PrintSelfTestPage ReportLevels ReportStatus"
+
+ +

If no cupsCommands keyword is provided, the command filter +must support AutoConfigure, +Clean, +PrintSelfTestPage, +and ReportLevels. The scheduler also +provides the printer-commands attribute containing the list of +supported commands.

+ + +

File Syntax

+ +

CUPS command files are ASCII text files. The first line of a +CUPS command file MUST contain:

+ +
+#CUPS-COMMAND
+
+ +

After that, each line is either a command or a comment. +Comments begin with the # character, e.g.:

+ +
+# This is a comment
+
+ +

Commands are any sequence of letters, numbers, and punctuation characters +optionally followed by parameters separated by whitespace, e.g.:

+ +
+Clean all
+PrintSelfTestPage
+
+ +

Command names are case-insensitive, so "PRINTSELFTESTPAGE", +"printselftestpage", and "PrintSelfTestPage" are equivalent. Vendor-specific +commands should use a domain name prefix, e.g.:

+ +
+com.vendor.foo
+com.vendor.bar param param2 ... paramN
+
+ + +

Standard Commands

+ +

The following are the standard commands supported by the format. The only +required command is +PrintSelfTestPage.

+ + +

AutoConfigure

+ +

AutoConfigure

+ +

The AutoConfigure command updates the printer's PPD file +and driver state information to reflect the current configuration of the +printer. There are no arguments for this command.

+ +

Example:

+ +
+#CUPS-COMMAND
+AutoConfigure
+
+ + +

Clean

+ +

Clean colorname

+ +

The Clean command performs a standard print head cleaning. The +"colorname" parameter specifies which color or head to clean. If a printer does +not support cleaning of individual colors or cartridges, then all colors are +cleaned. Command filters MUST support the "all" colorname. Other standard color +names include "black", "color", "photo", "cyan", "magenta", "yellow", +"light-cyan", "light-magenta", "light-black", "light-gray", and "dark-gray".

+ +

Example:

+ +
+#CUPS-COMMAND
+Clean all
+
+ + +

PrintAlignmentPage

+ +

PrintAlignmentPage pass

+ +

The PrintAlignmentPage command prints a head alignment page on +the printer. The "pass" parameter provides a pass number from 1 to N. The number +of passes is device-dependent.

+ +

Example:

+ +
+#CUPS-COMMAND
+PrintAlignmentPage 1
+
+ + +

PrintSelfTestPage

+ +

PrintSelfTestPage

+ +

The PrintSelfTestPage command prints a self-test page on the +printer. Typically this page shows whether all jets on a print head are +functioning and that the print feed mechanisms are working properly.

+ +

Example:

+ +
+#CUPS-COMMAND
+PrintSelfTestPage
+
+ + +

ReportLevels

+ +

ReportLevels

+ +

The ReportLevels command queries the supply levels on a printer +and reports "marker-colors", "marker-levels", "marker-names", and +"marker-types" attributes using "ATTR:" messages sent to the scheduler. This +command should also report the current printer status using "STATE:" messages +like the ReportStatus command.

+ +

Example:

+ +
+#CUPS-COMMAND
+ReportLevels
+
+ + +

ReportStatus

+ +

ReportStatus

+ +

The ReportStatus command queries the printer for its current +status and reports it using "STATE:" messages sent to the scheduler.

+ +

Example:

+ +
+#CUPS-COMMAND
+ReportLevels
+
+ + +

SetAlignment

+ +

SetAlignment pass value ... valueN

+ +

The SetAlignment command sets print head alignment values. The +"pass" parameter is a number from 1 to N. All parameters are +device-dependent.

+ +

Example:

+ +
+#CUPS-COMMAND
+SetAlignment 1 14
+
+ + + + diff --git a/doc/help/spec-design.html b/doc/help/spec-design.html new file mode 100644 index 0000000000..28a14eb31d --- /dev/null +++ b/doc/help/spec-design.html @@ -0,0 +1,184 @@ + + + + + CUPS Design Description + + + + +

CUPS Design Description

+ +

This design description documents the overall organization of CUPS. The purpose is not to provide a line-by-line description of the CUPS source code, but rather to describe the overall architecture and location of key pieces so that developers can more easily understand the underlying operation of CUPS.

+ +

Introduction

+ +

Like most printing systems, CUPS is designed around a central print scheduling process that dispatches print jobs, processes administrative commands, provides printer status information to local and remote programs, and informs users as needed. Figure 1 shows the basic organization of CUPS.

+ +

Scheduler

+ +

The scheduler is a HTTP/1.1 and IPP/2.1 server application that manages HTTP and IPP requests, printers, classes, jobs, subscriptions, and notifications on the system. HTTP is used for normal web browser services as well as IPP operation messages passed via HTTP POST requests with the application/ipp content type. The scheduler uses a series of helper applications based on the Common Gateway Interface ("CGI") to provide dynamic web interfaces and can be configured to run additional site-specific programs or scripts for the web interface.

+ +

The scheduler is designed as a traditional single-threaded server process which runs external processes to do longer-term operations such as printing, notification, device/driver enumeration, and remote printer monitoring. External processes are normally run as a non-privileged account ("lp") and, on some platforms, with additional restrictions that limit what the processes are allowed to do.

+ +

The maximum number of simultaneous clients and print jobs that can be supported is primarily limited by the available server memory, file descriptors, and CPU - the scheduler itself imposes no hard limits.

+ +
+ + +
Figure 1: CUPS Block Diagram
CUPS Block Diagram
+ +

Config Files

+ +

The scheduler uses several configuration files to store the server settings (cupsd.conf), available classes (classes.conf), available printers (printers.conf), current notification subscriptions (subscriptions.conf), and supported file types and filters (mime.types, mime.convs). In addition, PostScript Printer Description ("PPD") files or interface scripts are associated with each printer, and the scheduler has cache files for remote printers, PPD files, and current jobs to optimize the scheduler's startup speed and availability.

+ +

Job Files

+ +

The scheduler stores job files in a spool directory, typically /var/spool/cups. Two types of files will be found in the spool directory: control files starting with the letter "c" ("c00001", "c99999", "c100000", etc.) and data files starting with the letter "d" ("d00001-001", "d99999-001", "d100000-001", etc.) Control files are IPP messages based on the original IPP Print-Job or Create-Job messages, while data files are the original print files that were submitted for printing. There is one control file for every job known to the system and 0 or more data files for each job. + +

Control files are normally cleaned out after the 500th job is submitted, while data files are removed immediately after a job has successfully printed. Both behaviors can be configured.

+ +

Log Files

+ +

The scheduler keeps three kinds of log files which are normally stored in the /var/log/cups directory. The access_log file lists every HTTP and IPP request that is processed by the scheduler. The error_log file contains messages from the scheduler and its helper applications that can be used +to track down problems. The page_log file lists every page that is printed, allowing for simple print accounting.

+ +

Log files are rotated automatically by the scheduler when they reach the configured size limit, by default 1MB. If the limit is set to 0 then no rotation is performed in the scheduler - this mode is often used by Linux distributions so they can use the logrotated(8) program to rotate them instead.

+ +

Berkeley Commands

+ +

CUPS provides the Berkeley lpc(8), lpq(1), lpr(1), and lprm(1) commands. In general, they function identically to the original Berkeley commands with the following exceptions:

+ +
    + +
  1. The lpc command currently only supports the "status" sub-command.
  2. + +
  3. The lpr command does not support the format modifier options "1" (TROFF font set 1), "2" (TROFF font set 2), "3" (TROFF font set 3), "4" (TROFF font set 4), "c" (CIFPLOT), "d" (DVI), "f" (FORTRAN), "g" (GNU plot), "i" (indentation), "n" (Ditroff), "r" (Sun raster), "t" (Troff), or "w" (width), as they do not map to the IPP MIME media type based document formats.
  4. + +
+ +

System V Commands

+ +

CUPS provides the System V cancel(1), lp(1), lpadmin(8), lpmove(8), and lpstat(1) commands. In general, they function identically to the original System V commands with the following exceptions:

+ +
    + +
  1. All commands may ask for a password; the System V print spooler requires root access to perform administration tasks, while CUPS allows for more flexible configurations.
  2. + +
  3. The lpadmin command does not implement the Solaris "-A" (alert), "-F" (fault recovery), "-M" (mount form/wheel), "-P" (paper list), "-S" (print wheels), "-T" (type list), "-U" (dialer info), "-W" (wait), "-f" (form name), "-l" (content-type list), "-s" (remote printer), or "-t" (number of trays) options.
  4. + +
+ +

CUPS Commands

+ +

CUPS provides the cupsaccept(8), cupsaddsmb(8), cupsdisable(8), cupsenable(8), cupsreject(8), cupstestppd(1), lpinfo(8), and lppasswd(1) commands. The cupsaccept, cupsdisable, cupsenable, and cupsreject commands correspond to the System V accept, disable, enable, and reject commands but have been renamed to avoid confusion and conflicts with the bash(1) internal enable command of the same name.

+ +

LPD Support

+ +

LPD client support is provided via the cups-lpd(8) program. Incoming LPD requests are accepted on TCP port 515 by the local inetd(8), launchd(8), or xinetd(8) process and forwarded to the cups-lpd program for conversion to the corresponding IPP request(s).

+ +

The cups-lpd program conforms, for the most part, to RFC 1179: Line Printer Daemon Protocol, but does not enforce the privileged source port restriction specified in that document. In addition, the banner page and output format options are usually overridden via command-line options to the cups-lpd program when it is invoked by the corresponding super-daemon program.

+ +

Web Interface

+ +

The web interface is supported by five CGI programs. Table 1 describes the purpose of each of the programs.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 1: CGI Programs
ProgramLocationDescription
admin.cgi/adminProvides all of the administrative functions
classes.cgi/classesLists classes and provides class management functions
help.cgi/helpProvides access to online help documents
jobs.cgi/jobsLists jobs and provides job management functions
printers.cgi/printersLists printers and provides printer management functions
+ +

Notifiers

+ +

Notifiers (notifier(7)) provide the means for sending asynchronous event notifications from the scheduler. Notifiers are executed with the recipient information on the command-line and the event data on the standard input. For example:

+ +
+CUPS_SERVERBIN/notifier/foo recipient user-data
+
+ +

CUPS includes two notifiers: mailto to provide SMTP-based email notifications and rss to provide Really Simple Syndication ("RSS") notifications from the scheduler. Additional notifiers can be installed in the notifier directory as needed to support other methods.

+ +

Filters

+ +

Filters (filter(7)) convert job files into a printable format. Multiple filters are run, as needed, to convert from the job file format to the printable format. A filter program reads from the standard input or from a file if a filename is supplied. All filters must support a common set of options including printer name, job ID, username, job title, number of copies, and job options. All output is sent to the standard output.

+ +

CUPS provides filters for printing text, PostScript, PDF, HP-GL/2, and many types of image files. CUPS also provides printer driver filters for HP-PCL, ESC/P, and several types of label printers. Additional filters can be registered with CUPS via mime.convs and PPD files.

+ +

Port Monitors

+ +

Port monitors handle the device- and channel-specific data formatting for a printer. Port monitors use the same interface as filters.

+ +

CUPS includes two port monitors: the bcp port monitor which supports the PostScript Binary Communications Protocol ("BCP") and the tbcp port monitor which supports the PostScript Tagged Binary Communications Protocol ("TBCP"). Additional port monitors can be registered in PPD files.

+ +

Backends

+ +

Backends (backend(7)) send print data to the printer and enumerate available printers/devices as needed. Backends use the same interface as filters.

+ +

CUPS includes backends for AppSocket (JetDirect), IPP, LPD, parallel, SCSI, serial, and USB connections. Additional backends can be added as needed without additional configuration.

+ + +

Programming Interfaces

+ +

CUPS makes use of several general-purpose libraries to provide its printing services. Unlike the rest of CUPS, the libraries are provided under the terms of the GNU LGPL so they may be used by non-GPL applications.

+ +

CUPS Library (libcups)

+ +

The CUPS library contains all of the core HTTP and IPP communications code as well as convenience functions for queuing print jobs, getting printer information, accessing resources via HTTP and IPP, and manipulating PPD files. The scheduler and all commands, filters, and backends use this library.

+ +

CUPS CGI Library (libcupscgi)

+ +

The CUPS CGI library provides all of the web interface support functions. It is used by the CGI programs to provide the CUPS web interface.

+ +

CUPS Driver Library (libcupsdriver)

+ +

The CUPS driver library provides access to the dithering, color conversion, and helper functions used by the CUPS sample printer drivers.

+ +

CUPS Imaging Library (libcupsimage)

+ +

The CUPS imaging library provides functions for managing large images, doing colorspace conversion and color management, scaling images for printing, and managing raster page streams. It is used by the CUPS image file filters, the PostScript RIP, and all raster printers drivers.

+ +

CUPS MIME Library (libcupsmime)

+ +

The CUPS MIME library provides file typing and conversion functions and is used by the scheduler and cupsfilter(8) command to auto-type and convert print files to a printable format.

+ +

CUPS PPD Compiler Library (libcupsppdc)

+ +

The CUPS PPD compiler library provides access to driver information files and is used by the PPD compiler tools as well as the cups-driverd(8) helper program to generate PPD files and message catalogs for localization.

+ + + + diff --git a/doc/help/spec-ipp.html b/doc/help/spec-ipp.html new file mode 100644 index 0000000000..fbb1271b23 --- /dev/null +++ b/doc/help/spec-ipp.html @@ -0,0 +1,2744 @@ + + + + + CUPS Implementation of IPP + + + + + + +

CUPS Implementation of IPP

+ +

Introduction

+ +

CUPS implements IPP/2.1 and the operations and attributes +defined in the "IPP: Job and Printer Set Operations", "IPP/1.1: +Output-bin Attribute Extension", and "IPP/1.1: finishings +'fold',' trim', and 'bale' attribute values extension" +specifications.

+ +

CUPS also provides 16 new operations and many new attributes +to support multiple IPP printers and printer classes on a single +host.

+ +

IPP URIs

+ +

CUPS supports the "http", "https", and "ipp" schemes. The +following resource names are used:

+ +
+ +
scheme://hostname:port/
+ +
Can be used for all "get" operations and for server + subscriptions.
+ +
scheme://hostname:port/admin/
+ +
Used for all administrative operations.
+ +
scheme://hostname:port/classes/name
+ +
Specifies a printer class.
+ +
scheme://hostname:port/jobs/id
+ +
Specifies a job.
+ +
scheme://hostname:port/printers/name
+ +
Specifies a printer.
+ +
+ +

So a typical printer URI would be +"ipp://foo.bar.com/printers/LaserJet". In addition, the CUPS +server also supports normal browser access via +"http://hostname:port/" and "https://hostname:port/".

+ +

CUPS IPP Operations

+ +

CUPS provides 16 extension operations in addition to most of the +standard IPP and registered extension operations: + +

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Operation NameCUPSCodeBrief Description
Print-Job1.00x0002Print a file.
Validate-Job1.00x0004Validate job attributes.
Create-Job1.10x0005Create a print job.
Send-Document1.10x0006Send a file for a print job.
Cancel-Job1.00x0008Cancel a print job.
Get-Job-Attributes1.00x0009Get job attributes.
Get-Jobs1.00x000AGet all jobs.
Get-Printer-Attributes1.00x000BGet printer attributes.
Hold-Job1.10x000CHold a job for printing.
Release-Job1.10x000DRelease a job for printing.
Restart-Job1.10x000ERestarts a print job.
Pause-Printer1.00x0010Pause printing on a printer.
Resume-Printer1.00x0011Resume printing on a printer.
Purge-Jobs1.00x0012Purge all jobs.
Set-Job-Attributes1.10x0014Set attributes for a pending or held job.
Create-Printer-Subscription1.20x0016Creates a subscription associated with a printer or the server.
Create-Job-Subscription1.20x0017Creates a subscription associated with a job.
Get-Subscription-Attributes1.20x0018Gets the attributes for a subscription.
Get-Subscriptions1.20x0019Gets the attributes for zero or more subscriptions.
Renew-Subscription1.20x001ARenews a subscription.
Cancel-Subscription1.20x001BCancels a subscription.
Get-Notifications1.20x001CGet notification events for ippget subscriptions.
Enable-Printer1.20x0022Accepts jobs on a printer.
Disable-Printer1.20x0023Rejects jobs on a printer.
CUPS-Get-Default1.00x4001Get the default destination.
CUPS-Get-Printers1.00x4002Get all of the available printers.
CUPS-Add-Modify-Printer1.00x4003Add or modify a printer.
CUPS-Delete-Printer1.00x4004Delete a printer.
CUPS-Get-Classes1.00x4005Get all of the available printer classes.
CUPS-Add-Modify-Class1.00x4006Add or modify a printer class.
CUPS-Delete-Class1.00x4007Delete a printer class.
CUPS-Accept-Jobs1.00x4008Accept jobs on a printer or printer class.
CUPS-Reject-Jobs1.00x4009Reject jobs on a printer or printer class.
CUPS-Set-Default1.00x400ASet the default destination.
CUPS-Get-Devices1.10x400BGet all of the available devices.
CUPS-Get-PPDs1.10x400CGet all of the available PPDs.
CUPS-Move-Job1.10x400DMove a job to a different printer.
CUPS-Authenticate-Job1.20x400EAuthenticate a job for printing.
CUPS-Get-PPD1.30x400FGet a PPD file.
CUPS-Get-Document1.40x4027Get a document file from a job.
+ +

Operations

+ +

The following sections describe the operations supported by CUPS. +In the interest of brevity, operations which use only the standard +IPP attributes are not described. + +

Print-Job Operation

+ +

The Print-Job operation (0x0002) prints a file. + +

Print-Job Request

+ +

The following groups of attributes are supplied as part of the +Print-Job request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"printer-uri" (uri): + +
The client MUST supply a URI for the specified printer. + +
+ +

Group 2: Job Template Attributes + +

+ +
"auth-info" (1setOf text(MAX)):CUPS 1.3/Mac OS X 10.5 + +
The client OPTIONALLY supplies one or more authentication values as specified by the "auth-info-required" attribute. + +
"job-billing" (text(MAX)):CUPS 1.1 + +
The client OPTIONALLY supplies a billing string that is logged + with the page accounting information. + +
"job-sheets" (1setof type3 keyword | name(MAX)):CUPS 1.1 + +
The client OPTIONALLY supplies one or two banner pages that + are printed before and after any files in the print job. The + name of "none" is reserved to indicate that no banner page + should be printed. If the client does not specify this + attribute then the value of the "job-sheets-default" printer + object attribute is used. + +
Note: Standard IPP only allows + specification of a single job-sheets attribute + value.
+ +
"media" (1setof type3 keyword | name(MAX)): + +
The client OPTIONALLY supplies one or more media attributes + specifying the size, type, source, and color of the output + media. If the client does not specify this attribute then the + value of the "media-default" printer object attribute is used. + +
Note: Standard IPP only allows + specification of a single media attribute + value.
+ +
Other Job Template Attributes + +
+ +

The Print-Job request is followed by a file to be printed. + +

Print-Job Response

+ +

The following groups of attributes are send as part of the Print-Job +Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

Group 2: Job Attributes + +

+ +
Standard Job Attributes + +
+ +

Group 3: Unsupported Attributes (status=client-eror-attributes-or-values-not-supported) + +

+ +
auth-info-required (1setOf Type2 keyword) + +
The required authentication information. + +
+ +

CUPS 1.1Create-Job Operation

+ +

The Create-Job operation (0x0005) creates a new, empty print job. + +

Create-Job Request

+ +

The following groups of attributes are supplied as part of the +Create-Job request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"printer-uri" (uri): + +
The client MUST supply a URI for the specified printer. + +
+ +

Group 2: Job Template Attributes + +

+ +
"auth-info" (1setOf text(MAX)):CUPS 1.3/Mac OS X 10.5 + +
The client OPTIONALLY supplies one or more authentication values as specified by the "auth-info-required" attribute. + +
"job-billing" (text(MAX)):CUPS 1.1 + +
The client OPTIONALLY supplies a billing string that is logged + with the page accounting information. + +
"job-sheets" (1setof type3 keyword | name(MAX)):CUPS 1.1 + +
The client OPTIONALLY supplies one or two banner pages that + are printed before and after any files in the print job. The + name of "none" is reserved to indicate that no banner page + should be printed. If the client does not specify this + attribute then the value of the "job-sheets-default" printer + object attribute is used. + +
Note: Standard IPP only allows + specification of a single job-sheets attribute + value.
+ +
"media" (1setof type3 keyword | name(MAX)): + +
The client OPTIONALLY supplies one or more media attributes + specifying the size, type, source, and color of the output + media. If the client does not specify this attribute then the + value of the "media-default" printer object attribute is used. + +
Note: Standard IPP only allows + specification of a single media attribute + value.
+ +
Standard Job Template Attributes + +
+ +

Create-Job Response

+ +

The following groups of attributes are send as part of the +Create-Job Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

Group 2: Job Attributes + +

+ +
Standard Job Attributes + +
+ +

Group 3: Unsupported Attributes (status=client-eror-attributes-or-values-not-supported) + +

+ +
auth-info-required (1setOf Type2 keyword) + +
The required authentication information. + +
+ +

Cancel Job Operation

+ +

The Cancel-Job operation (0x0008) cancels the specified job. CUPS 1.4 adds +a new purge-job (boolean) attribute that allows you to purge both +active and completed jobs, removing all history and document files for the +job as well. + +

Cancel-Job Request

+ +

The following groups of attributes are supplied as part of the +Cancel-Job request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"printer-uri" (uri) and "job-id" (integer) +
OR +
"job-uri": + +
The client MUST supply a URI for the specified printer and + a job ID number, or the job URI. + +
CUPS 1.4/Mac OS X 10.6"purge-job" (boolean): + +
The client OPTIONALLY supplies this attribute. When true, + all job files (history and document) are purged. The default + is false, leading to the standard IPP behavior. + +
+ +

Cancel-Job Response

+ +

The following groups of attributes are send as part of the Cancel-Job +Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

Purge-Jobs Operation

+ +

The Purge-Jobs operation (0x0012) cancels all of the jobs on a +given destination and optionally removes all history and document +files for the jobs as well. + +

Purge-Jobs Request

+ +

The following groups of attributes are supplied as part of the +Purge-Jobs request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"printer-uri" (uri): + +
The client MUST supply a URI for the specified printer or + "ipp://.../printers" for all printers and classes. + +
CUPS 1.2/Mac OS X 10.5"requesting-user-name" (name(MAX)): + +
The client OPTIONALLY supplies this attribute to specify whose jobs + jobs are purged or canceled. + +
CUPS 1.2/Mac OS X 10.5"my-jobs" (boolean): + +
The client OPTIONALLY supplies this attribute to specify that only + the jobs owned by the requesting user are purged or canceled. The + default is false. + +
CUPS 1.2/Mac OS X 10.5"purge-jobs" (boolean): + +
The client OPTIONALLY supplies this attribute to specify + whether the jobs are purged (true) or just canceled (false). + The default is true. + +
+ +

Purge-Jobs Response

+ +

The following groups of attributes are send as part of the Purge-Jobs +Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

CUPS 1.1Set-Job-Attributes Operation

+ +

The Set-Job-Attributes operation (0x0014) changes the attributes of +an active (not completed) job. + +

Set-Job-Attributes Request

+ +

The following groups of attributes are supplied as part of the +Set-Job-Attributes request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"printer-uri" (uri) and "job-id" (integer) +
OR +
"job-uri": + +
The client MUST supply a URI for the specified printer and + a job ID number, or the job URI. + +
+ +

Group 2: Job Template Attributes + +

+ +
"job-sheets" (1setof type3 keyword | name(MAX)):CUPS 1.1 + +
The client OPTIONALLY supplies one or two banner pages that + are printed before and after any files in the print job. The + name of "none" is reserved to indicate that no banner page + should be printed. If the client does not specify this + attribute then the value of the "job-sheets-default" printer + object attribute is used. + +
Note: Standard IPP only allows + specification of a single job-sheets attribute + value.
+ +
"media" (1setof type3 keyword | name(MAX)): + +
The client OPTIONALLY supplies one or more media attributes + specifying the size, type, source, and color of the output + media. If the client does not specify this attribute then the + value of the "media-default" printer object attribute is used. + +
Note: Standard IPP only allows + specification of a single media attribute + value.
+ +
Other Job Template Attributes + +
+ +

Set-Job-Attributes Response

+ +

The following groups of attributes are send as part of the Set-Job-Attributes +Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

CUPS 1.2/Mac OS X 10.5Create-Printer-Subscription

+ +

The Create-Printer-Subscription operation (0x0016) creates a +subscription for printer or server event notifications. CUPS +provides several additional events in addition to the standard +events in the IPP notifications specification.

+ +

Create-Printer-Subscription Request

+ +

The following groups of attributes are supplied as part of the +request:

+ +

Group 1: Operation Attributes

+ +
+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"printer-uri" (uri): + +
The printer + (ipp://server/printers/printername), class + (ipp://server/classes/classname), or server + (ipp://server/) URI for event notifications. + +
"notify-events" (1setOf keyword): + +
The events to monitor. In addition to the standard + events, CUPS adds the following keywords: +
    + +
  • printer-added - Get notified + whenever a printer or class is added
  • + +
  • printer-deleted - Get notified + whenever a printer or class is deleted
  • + +
  • printer-modified - Get notified + whenever a printer or class is modified
  • + +
  • server-audit - Get notified when a + security condition occurs
  • + +
  • server-restarted - Get notified when + the server is restarted
  • + +
  • server-started - Get notified when + the server is started
  • + +
  • server-stopped - Get notified when + the server is stopped
  • + +
+ +
+ +

Create-Printer-Subscription Response

+ +

The following groups of attributes are send as part of the +response:

+ +

Group 1: Operation Attributes

+ +
+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

Group 2: Subscription Object Attributes

+ +
+ +
"subscription-id" (integer): + +
The subscription number. + +
+ +

CUPS-Get-Default Operation

+ +

The CUPS-Get-Default operation (0x4001) returns the default printer +URI and attributes. + +

CUPS-Get-Default Request

+ +

The following groups of attributes are supplied as part of the +CUPS-Get-Default request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"requested-attributes" (1setOf keyword) : + +
The client OPTIONALLY supplies a set of attribute names + and/or attribute group names in whose values the requester is + interested. If the client omits this attribute, the server + responds as if this attribute had been supplied with a value of + 'all'. + +
+ +

CUPS-Get-Default Response

+ +

The following groups of attributes are send as part of the +CUPS-Get-Default Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

Group 2: Printer Object Attributes + +

+ +
The set of requested attributes and their current values. + +
+ +

CUPS-Get-Printers Operation

+ +

The CUPS-Get-Printers operation (0x4002) returns the printer +attributes for every printer known to the system. This may include +printers that are not served directly by the server. + +

CUPS-Get-Printers Request

+ +

The following groups of attributes are supplied as part of the +CUPS-Get-Printers request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"first-printer-name" (name(127)):CUPS 1.2/Mac OS X 10.5 + +
The client OPTIONALLY supplies this attribute to + select the first printer that is returned. + +
"limit" (integer (1:MAX)): + +
The client OPTIONALLY supplies this attribute limiting the + number of printers that are returned. + +
"printer-location" (text(127)): CUPS 1.1.7 + +
The client OPTIONALLY supplies this attribute to + select which printers are returned. + +
"printer-type" (type2 enum): CUPS 1.1.7 + +
The client OPTIONALLY supplies a printer type enumeration to + select which printers are returned. + +
"printer-type-mask" (type2 enum): CUPS 1.1.7 + +
The client OPTIONALLY supplies a printer type mask + enumeration to select which bits are used in the "printer-type" + attribute. + +
"requested-attributes" (1setOf keyword) : + +
The client OPTIONALLY supplies a set of attribute names + and/or attribute group names in whose values the requester is + interested. If the client omits this attribute, the server + responds as if this attribute had been supplied with a value of + 'all'. + +
"requested-user-name" (name(127)) : CUPS 1.2/Mac OS X 10.5 + +
The client OPTIONALLY supplies a user name that is used to filter + the returned printers. + +
+ +

CUPS-Get-Printers Response

+ +

The following groups of attributes are send as part of the +CUPS-Get-Printers Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

Group 2: Printer Object Attributes + +

+ +
The set of requested attributes and their current values for + each printer. + +
+ +

CUPS-Add-Modify-Printer Operation

+ +

The CUPS-Add-Modify-Printer operation (0x4003) adds a new printer or +modifies an existing printer on the system. + +

CUPS-Add-Modify-Printer Request

+ +

The following groups of attributes are supplied as part of the +CUPS-Add-Modify-Printer request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"printer-uri" (uri): + +
The client MUST supply a URI for the specified printer. + +
+ +

Group 2: Printer Object Attributes + +

+ +
"auth-info-required" (1setOf type2 keyword):CUPS 1.3/Mac OS X 10.5 + +
The client OPTIONALLY supplies one or more authentication keywords that are required to communicate with the printer/remote queue. + +
"job-sheets-default" (1setOf name(127)): CUPS 1.1.7 + +
The client OPTIONALLY supplies one or two banner page + names that are printed before and after files in a job. + The reserved name "none" is used to specify that no + banner page should be printed. + +
"device-uri" (uri): + +
The client OPTIONALLY supplies a device URI for the + specified printer. + +
"port-monitor" (name(127)): + +
The client OPTIONALLY supplies a port monitor name for the + specified printer. + +
"ppd-name" (name(127)): + +
The client OPTIONALLY supplies a PPD name for the specified + printer. + +
"printer-is-accepting-jobs" (boolean): + +
The client OPTIONALLY supplies this boolean attribute + indicating whether or not the printer object should accept new jobs. + +
"printer-info" (text(127)): + +
The client OPTIONALLY supplies this attribute indicating the + printer information string. + +
"printer-location" (text(127)): + +
The client OPTIONALLY supplies this attribute indicating a + textual location of the printer. + +
"printer-more-info" (uri): + +
The client OPTIONALLY supplies this attribute indicating a + URI for additional printer information. + +
"printer-state" (type2 enum): + +
The client OPTIONALLY supplies this attribute indicating the + initial/current state of the printer. Only the "idle" and "stopped" + enumerations are recognized. + +
"printer-state-message" (text(MAX)): + +
The client OPTIONALLY supplies this attribute indicating a + textual reason for the current printer state. + +
"requesting-user-name-allowed" (1setof name(127) | delete) +
OR +
"requesting-user-name-denied" (1setof name(127) | delete): + +
The client OPTIONALLY supplies one of these attributes to + specify an access control list for incoming print jobs. To allow + all users access to a printer, use the delete tag for the + attribute value. + +
+ +

The CUPS-Add-Modify-Printer request can optionally be followed by a PPD +file or System V interface script to be used for the printer. The +"ppd-name" attribute overrides any file that is attached to the end of +the request with a local CUPS PPD file. + +

CUPS-Add-Modify-Printer Response

+ +

The following groups of attributes are send as part of the +CUPS-Add-Modify-Printer Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

CUPS-Delete-Printer Operation

+ +

The CUPS-Delete-Printer operation (0x4004) removes an existing +printer from the system. + +

CUPS-Delete-Printer Request

+ +

The following groups of attributes are supplied as part of the +CUPS-Delete-Printer request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"printer-uri" (uri): + +
The client MUST supply a URI for the specified printer. + +
+ +

CUPS-Delete-Printer Response

+ +

The following groups of attributes are send as part of the +CUPS-Delete-Printer Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

CUPS-Get-Classes Operation

+ +

The CUPS-Get-Classes operation (0x4005) returns the printer +attributes for every printer class known to the system. This may +include printer classes that are not served directly by the server. + +

CUPS-Get-Classes Request

+ +

The following groups of attributes are supplied as part of the +CUPS-Get-Classes request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"first-printer-name" (name(127)):CUPS 1.2/Mac OS X 10.5 + +
The client OPTIONALLY supplies this attribute to + select the first printer that is returned. + +
"limit" (integer (1:MAX)): + +
The client OPTIONALLY supplies this attribute limiting the + number of printer classes that are returned. + +
"printer-location" (text(127)): CUPS 1.1.7 +
The client OPTIONALLY supplies this attribute to + select which printer classes are returned. + +
"printer-type" (type2 enum): CUPS 1.1.7 +
The client OPTIONALLY supplies a printer type enumeration to + select which printer classes are returned. + +
"printer-type-mask" (type2 enum): CUPS 1.1.7 +
The client OPTIONALLY supplies a printer type mask + enumeration to select which bits are used in the "printer-type" + attribute. + +
"requested-attributes" (1setOf keyword) : + +
The client OPTIONALLY supplies a set of attribute names + and/or attribute group names in whose values the requester is + interested. If the client omits this attribute, the server responds as + if this attribute had been supplied with a value of 'all'. + +
"requested-user-name" (name(127)) : CUPS 1.2/Mac OS X 10.5 + +
The client OPTIONALLY supplies a user name that is used to filter + the returned printers. + +
+ +

CUPS-Get-Classes Response

+ +

The following groups of attributes are send as part of the +CUPS-Get-Classes Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

Group 2: Printer Class Object Attributes + +

+ +
The set of requested attributes and their current values for + each printer class. + +
+ +

CUPS-Add-Modify-Class Operation

+ +

The CUPS-Add-Modify-Class operation (0x4006) adds a new printer class or +modifies and existing printer class on the system. + +

CUPS-Add-Modify-Class Request

+ +

The following groups of attributes are supplied as part of the +CUPS-Add-Modify-Class request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"printer-uri" (uri): + +
The client MUST supply a URI for the specified printer class. + +
+ +

Group 2: Printer Object Attributes + +

+ +
"auth-info-required" (1setOf type2 keyword):CUPS 1.3/Mac OS X 10.5 + +
The client OPTIONALLY supplies one or more authentication keywords that are required to communicate with the printer/remote queue. + +
"member-uris" (1setof uri): + +
The client OPTIONALLY supplies the "member-uris" set + specifying the printers and printer classes that are part of the class. + +
"printer-is-accepting-jobs" (boolean): + +
The client OPTIONALLY supplies this boolean attribute + indicating whether or not the class object should accept new jobs. + +
"printer-info" (text(127)): + +
The client OPTIONALLY supplies this attribute indicating the + printer information string. + +
"printer-location" (text(127)): + +
The client OPTIONALLY supplies this attribute indicating a + textual location of the class. + +
"printer-more-info" (uri): + +
The client OPTIONALLY supplies this attribute indicating a + URI for additional class information. + +
"printer-state" (type2 enum): + +
The client OPTIONALLY supplies this attribute indicating the + initial/current state of the class. Only the "idle" and "stopped" + enumerations are recognized. + +
"printer-state-message" (text(MAX)): + +
The client OPTIONALLY supplies this attribute indicating a + textual reason for the current class state. + +
"requesting-user-name-allowed" (1setof name(127)) +
OR +
"requesting-user-name-denied" (1setof name(127)): + +
The client OPTIONALLY supplies one of these attributes to + specify an access control list for incoming print jobs. To allow + all users access to a class, use the delete tag for the + attribute value. + +
+ +

CUPS-Add-Modify-Class Response

+ +

The following groups of attributes are send as part of the CUPS-Add-Modify-Class Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

CUPS-Delete-Class Operation

+ +

The CUPS-Delete-Class operation (0x4007) removes an existing printer +class from the system. + +

CUPS-Delete-Class Request

+ +

The following groups of attributes are supplied as part of the +CUPS-Delete-Class request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"printer-uri" (uri): + +
The client MUST supply a URI for the specified printer class. + +
+ +

CUPS-Delete-Class Response

+ +

The following groups of attributes are send as part of the +CUPS-Delete-Class Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

CUPS-Accept-Jobs Operation

+ +

The CUPS-Accept-Jobs operation (0x4008) sets the +"printer-is-accepting-jobs" attribute to true for the specified printer +or printer class. + +

CUPS-Accept-Jobs Request

+ +

The following groups of attributes are supplied as part of the +CUPS-Accept-Jobs request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"printer-uri" (uri): + +
The client MUST supply a URI for the specified printer or printer class. + +
+ +

CUPS-Accept-Jobs Response

+ +

The following groups of attributes are send as part of the +CUPS-Accept-Jobs Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

CUPS-Reject-Jobs Operation

+ +

The CUPS-Reject-Jobs operation (0x4009) sets +the"printer-is-accepting-jobs" attribute to false for the specified +printer or printer class. + +

CUPS-Reject-Jobs Request

+ +

The following groups of attributes are supplied as part of the +CUPS-Reject-Jobs request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"printer-uri" (uri): + +
The client MUST supply a URI for the specified printer or printer class. + +
+ +

Group 2: Printer Object Attributes + +

+ +
"printer-state-message" (text(MAX)): + +
The client OPTIONALLY supplies this attribute indicating a + textual reason for the current printer state. + +
+ +

CUPS-Reject-Jobs Response

+ +

The following groups of attributes are send as part of the +CUPS-Reject-Jobs Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

CUPS-Set-Default Operation

+ +

The CUPS-Set-Default operation (0x400A) sets the default printer +destination for all clients when a resource name of "/printers" is +specified. + +

CUPS-Set-Default Request

+ +

The following groups of attributes are supplied as part of the +CUPS-Set-Default request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"printer-uri" (uri): + +
The client MUST supply a URI for the specified printer or + printer class. + +
+ +

CUPS-Set-Default Response

+ +

The following groups of attributes are send as part of the +CUPS-Set-Default Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

CUPS 1.1CUPS-Get-Devices Operation

+ +

The CUPS-Get-Devices operation (0x400B) returns all of the +supported device-uri's for the server.

+ +

CUPS-Get-Devices Request

+ +

The following groups of attributes are supplied as part of the +CUPS-Get-Devices request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"device-class" (type1 keyword): + +
The client OPTIONALLY supplies a device class keyword to select + which devices are returned. + +
"exclude-schemes" (1setOf name) :CUPS 1.4/Mac OS X 10.6 + +
The client OPTIONALLY supplies a set of scheme names that the + requestor does not want to discover. If the client omits this attribute, + the server responds with devices of all schemes specified by + the "include-schemes" attribute. + +
"include-schemes" (1setOf name) :CUPS 1.4/Mac OS X 10.6 + +
The client OPTIONALLY supplies a set of scheme names that the + requestor wants to discover. If the client omits this attribute, + the server responds with devices of all schemes except those specified + by the "exclude-schemes" attribute. + +
"limit" (integer (1:MAX)): + +
The client OPTIONALLY supplies this attribute limiting the number of + devices that are returned. + +
"requested-attributes" (1setOf keyword) : + +
The client OPTIONALLY supplies a set of attribute names and/or + attribute group names in whose values the requester is interested. If + the client omits this attribute, the server responds as if this + attribute had been supplied with a value of 'all'. + +
"timeout" (integer (1:MAX)) :CUPS 1.4/Mac OS X 10.6 + +
The client OPTIONALLY supplies this attribute to limit the duration + of the lookup. The default timeout is 15 seconds. + +
+ +

CUPS-Get-Devices Response

+ +

The following groups of attributes are send as part of the +CUPS-Get-Devices Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

Group 2: Device Object Attributes + +

+ +
The set of requested attributes and their current values for + each device. + +
+ +

CUPS 1.1CUPS-Get-PPDs Operation

+ +

The CUPS-Get-PPDs operation (0x400C) returns all of the +locally available PPD files on the system.

+ +

CUPS-Get-PPDs Request

+ +

The following groups of attributes are supplied as part of the +CUPS-Get-PPDs request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"exclude-schemes" (1setOf name) :CUPS 1.4/Mac OS X 10.6 + +
The client OPTIONALLY supplies a set of scheme names that the + requestor does not want to list. If the client omits this attribute, + the server responds with PPDs of all schemes specified by the + "include-schemes" attribute. + +
"include-schemes" (1setOf name) :CUPS 1.4/Mac OS X 10.6 + +
The client OPTIONALLY supplies a set of scheme names that the + requestor wants to list. If the client omits this attribute, the server + responds with PPDs of all schemes except those specified by the + "exclude-schemes" attribute. + +
"limit" (integer (1:MAX)): + +
The client OPTIONALLY supplies this attribute limiting the number of PPDs that are returned. + +
"ppd-make" (text(127)): + +
The client OPTIONALLY supplies a printer manufacturer to select which PPDs are returned. + +
"ppd-make-and-model" (text(127)):CUPS 1.3/Mac OS X 10.5 + +
The client OPTIONALLY supplies a make and model to select which PPDs are returned. + +
"ppd-model-number" (integer):CUPS 1.3/Mac OS X 10.5 + +
The client OPTIONALLY supplies a model number to select which PPDs are returned. + +
"ppd-natural-language" (naturalLanguage):CUPS 1.3/Mac OS X 10.5 + +
The client OPTIONALLY supplies a language to select which PPDs are returned. + +
"ppd-product" (text(127)):CUPS 1.3/Mac OS X 10.5 + +
The client OPTIONALLY supplies a PostScript product string to select which PPDs are returned. + +
"ppd-psversion" (text(127)):CUPS 1.3/Mac OS X 10.5 + +
The client OPTIONALLY supplies a PostScript version string to select which PPDs are returned. + +
"ppd-type" (type1 keyword):CUPS 1.3/Mac OS X 10.5 + +
The client OPTIONALLY supplies a driver type to select which PPDs are returned. + +
"requested-attributes" (1setOf keyword) : + +
The client OPTIONALLY supplies a set of attribute names and/or attribute group names in whose values the requester is interested. If the client omits this attribute, the server responds as if this attribute had been supplied with a value of 'all'. Specify "ppd-make" to get a list of manufacturers. + +
+ +

CUPS-Get-PPDs Response

+ +

The following groups of attributes are send as part of the +CUPS-Get-PPDs Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

Group 2: PPD Attributes + +

+ +
The set of requested attributes and their current values for each + PPD file. + +
+ +

CUPS 1.1CUPS-Move-Job Operation

+ +

The CUPS-Move-Job operation (0x400D) moves an active print job or all print +jobs for a printer to a different printer.

+ +

CUPS-Move-Job Request

+ +

The following groups of attributes are supplied as part of the +CUPS-Move-Job request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"printer-uri" (uri) +
OR +
"printer-uri" (uri) and "job-id" (integer) +
OR +
"job-uri": + +
The client MUST supply a URI for the specified printer, the URI for + the specified printer and a job ID number, or the job URI. + +
+ +

Group 2: Job Template Attributes + +

+ +
"job-printer-uri" (uri) + +
The client MUST supply a URI for a printer on the same server. + +
+ +

CUPS-Move-Job Response

+ +

The following groups of attributes are send as part of the +CUPS-Move-Job Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

CUPS 1.2/Mac OS X 10.5CUPS-Authenticate-Job Operation

+ +

The CUPS-Authenticate-Job operation (0x400E) authenticates a print job for +printing, releasing the job if it is held. Typically this is used when printing +to a remote server. The authentication information is passed in the HTTP +request; the HTTP connection is normally encrypted for this type of request.

+ +

CUPS-Authenticate-Job Request

+ +

The following groups of attributes are supplied as part of the +CUPS-Authenticate-Job request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"printer-uri" (uri) and "job-id" (integer) +
OR +
"job-uri": + +
The client MUST supply a URI for the specified printer and + a job ID number, or the job URI. + +
+ +

Group 2: Job Attributes + +

+ +
"auth-info" (1setOf text(MAX)):CUPS 1.3/Mac OS X 10.5 + +
The client OPTIONALLY supplies one or more authentication values as specified by the "auth-info-required" attribute. + +
"job-hold-until" (keyword | name(MAX)):CUPS 1.3/Mac OS X 10.5 + +
The client OPTIONALLY supplies a new job-hold-until value for the + job. If specified and not the "no-hold" value, the job is held instead + of released for printing. + +
+ +

CUPS-Authenticate-Job Response

+ +

The following groups of attributes are send as part of the +CUPS-Authenticate-Job Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
+ +

Group 2: Unsupported Attributes (status=client-eror-attributes-or-values-not-supported) + +

+ +
auth-info-required (1setOf Type2 keyword) + +
The required authentication information. + +
+ +

CUPS 1.3/Mac OS X 10.5CUPS-Get-PPD Operation

+ +

The CUPS-Get-PPD operation (0x400F) gets a PPD file from the +server. The PPD file can be specified using a ppd-name +returned by CUPS-Get-PPDs +or using the printer-uri for a queue.

+ +

If the PPD file is found, successful-ok is returned with +the PPD file following the response data.

+ +

If the PPD file cannot be served by the local server because +the printer-uri attribute points to an external printer, +a cups-see-other status is returned with the correct +URI to use.

+ +

If the PPD file does not exist, client-error-not-found is +returned.

+ +

CUPS-Get-PPD Request

+ +

The following group of attributes is supplied as part of the +CUPS-Get-PPD request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"printer-uri" (uri) +
OR +
"ppd-name" (name(255)): + +
The client MUST supply a printer URI or PPD name. + +
+ +

CUPS-Get-PPD Response

+ +

The following group of attributes is sent as part of the +CUPS-Get-PPD Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
"printer-uri" (uri): + +
The printer that provides the actual PPD file when + the status code is cups-see-other (0x280) + +
+ +

If the status code is successful-ok, the PPD file follows +the end of the IPP response.

+ +

CUPS 1.4/Mac OS X 10.6CUPS-Get-Document Operation

+ +

The CUPS-Get-Document operation (0x4027) gets a document file from a +job on the server. The document file is specified using the +document-number and either the job-uri or printer-uri +and job-id identifying the job.

+ +

If the document file is found, successful-ok is returned with +the document file following the response data.

+ +

If the document file does not exist, client-error-not-found is +returned.

+ +

If the requesting user does not have access to the document file, +client-error-not-authorized is returned. + +

CUPS-Get-Document Request

+ +

The following group of attributes is supplied as part of the +CUPS-Get-Document request: + +

Group 1: Operation Attributes + +

+ +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.1 of the IPP Model and + Semantics document. + +
"printer-uri" (uri) and "job-id" (integer) +
OR +
"job-uri" (uri): + +
The client MUST supply a printer URI and job ID or job URI. + +
"document-number" (integer(1:MAX)): + +
The client MUST supply a document number to retrieve. The + document-count attribute for the job defines the maximum + document number that can be specified. In the case of jobs with + banners (job-sheets is not "none"), document number 1 + will typically contain the start banner and document number N + will typically contain the end banner. + +
+ +

CUPS-Get-Document Response

+ +

The following group of attributes is sent as part of the +CUPS-Get-Document Response: + +

Group 1: Operation Attributes + +

+ +
Status Message: + +
The standard response status message. + +
Natural Language and Character Set: + +
The "attributes-charset" and "attributes-natural-language" + attributes as described in section 3.1.4.2 of the IPP Model and + Semantics document. + +
"document-format" (mimeType): + +
The format of the document file. + +
"document-number" (integer(1:MAX)): + +
The requested document number. + +
"document-name" (name(MAX)): + +
The name that was supplied with the document, if any. + +
+ +

If the status code is successful-ok, the document file follows +the end of the IPP response.

+ + +

Attributes

+ +

CUPS provides many extension attributes to support multiple +devices, PPD files, standard job filters, printers, and printer +classes.

+ +

Device Attributes

+ +

Device attributes are returned by the CUPS-Get-Devices +operation and enumerate all of the available hardware devices and +network protocols that are supported by the server.

+ +

device-class (type2 keyword)

+ +

The device-class attribute specifies the class of device and can be +one of the following: + +

    + +
  • "file" - a disk file. + +
  • "direct" - a parallel or fixed-rate serial data port, + currently used for Centronics, IEEE-1284, and USB printer + ports. + +
  • "serial" - a variable-rate serial port. + +
  • "network" - a network connection, typically via AppSocket, + HTTP, IPP, LPD, or SMB/CIFS protocols. + +
+ +

device-id (text(127))CUPS 1.2/Mac OS X 10.5

+ +

The device-id attribute specifies the IEEE-1284 device ID +string for the device.

+ +

device-info (text(127))

+ +

The device-info attribute specifies a human-readable string describing +the device, e.g. "Parallel Port #1". + +

device-location (text(127))CUPS 1.4/Mac OS X 10.6

+ +

The device-location attribute specifies the physical location of the +printer. + +

device-make-and-model (text(127))

+ +

The device-make-and-model attribute specifies a device +identification string provided by the printer connected to the device. +If the device or printer does not support identification then this +attribute contains the string "unknown". + +

device-uri (uri)

+ +

The device-uri attribute specifies a unique identifier for the +device. The actual format of the device-uri string depends on the value +of the device-class attribute: + +

    + +
  • "file" - The device-uri will be of the form + "file:///path/to/filename". + +
  • "direct" - The device-uri will be of the form + "scheme:/dev/filename" or "scheme://vendor/identifier", + where scheme may be "parallel" or "usb" in the current + implementation. + +
  • "serial" - The device-uri will be of the form + "serial:/dev/filename?baud=value+parity=value+flow=value". + The baud value is the data rate in bits per second; the + supported values depend on the underlying hardware. + The parity value can be one of "none", "even", or "odd". + The flow value can be one of "none", "soft" (XON/XOFF + handshaking), "hard" or "rts/cts" (RTS/CTS handshaking), + or "dtrdsr" (DTR/DSR handshaking). + +

    The URI returned by CUPS-Get-Devices will contain the + maximum baud rate supported by the device and the best + type of flow control available ("soft" or "hard"). + +

  • "network" - The device-uri will be of the form + "scheme://[username:password@]hostname[:port]/[resource]", + where scheme may be "http", "https", "ipp", "lpd", "smb", or + "socket" in the current implementation. + +

    The URI returned by CUPS-Get-Devices will only contain + the scheme name ("scheme"). It is up to the client + application to add the appropriate host and other + information when adding a new printer. + +

    The URI returned by Get-Printer-Attributes and + CUPS-Get-Printers has any username and password information + stripped; the information is still stored and used by the + server internally to perform any needed authentication. + +

+ +

Job Template Attributes

+ +

auth-info (1setOf text(MAX))CUPS 1.3/Mac OS X 10.5

+ +

The auth-info attribute specifies the authentication information to use when printing to a remote device. The order and content of each text value is specifed by the auth-info-required printer attribute. + +

cpi (type2 enum)

+ +

The cpi attribute specifies the number of characters per inch when +printing text files. Only the values 10, 12, and 17 are currently +supported. The default value is 10. + +

fit-to-page (boolean)CUPS 1.4/Mac OS X 10.6

+ +

The fit-to-page attribute specifies whether to scale documents to fit on the +selected media (fit-to-page=true) or use the physical size specified in the +document (fit-to-page=false). The default value is false. + +

job-billing (text(MAX))CUPS 1.1

+ +

The job-billing attribute provides a text value to associate with a job +for billing purposes. + +

job-hold-until (keyword | name(MAX))CUPS 1.1

+ +

The job-hold-until attribute specifies a hold time. In addition to the +standard IPP/1.1 keyword names, CUPS supports name values of the form +"HH:MM" and "HH:MM:SS" that specify a hold time. The hold time is in +Universal Coordinated Time (UTC) and not in the local time zone. If the +specified time is less than the current time, the job is held until the +next day. + +

job-media-progress (integer(0:100))CUPS 1.4/Mac OS X 10.6

+ +

The job-media-progress attribute specifies the percentage of completion of +the current page. It is only valid when the job-state attribute has the +"processing" value (5).

+ +

job-printer-state-message (text(MAX))CUPS 1.3/Mac OS X 10.5

+ +

The job-printer-state-message attribute provides the last known value of the printer-state-message attribute for the printer that processed (or is processing) the job.

+ +

job-printer-state-reasons (1setOf type2 keyword)CUPS 1.3/Mac OS X 10.5

+ +

The job-printer-state-reasons attribute provides the last known value of the printer-state-reasons attribute for the printer that processed (or is processing) the job.

+ +

job-sheets (1setof type3 keyword | name(MAX))CUPS 1.1

+ +

The job-sheets attribute specifies one or two banner files that are printed +before and after a job. The reserved value of "none" disables banner printing. +The default value is stored in the job-sheets-default attribute. + +

If only one value is supplied, the banner file is printed before the job. +If two values are supplied, the first value is used as the starting banner +file and the second as the ending banner file. + +

job-originating-host-name (name(MAX))

+ +

(CUPS 1.1.5 and higher) + +

The job-originating-host-name attribute specifies the host +from which the job was queued. The value will be the hostname or +IP address of the client depending on whether hostname +resolution is enabled. The localhost address (127.0.0.1) is +always resolved to the name "localhost". + +

This attribute is read-only. + +

lpi (type2 enum)

+ +

The lpi attribute specifies the number of lines per inch when +printing text files. Only the values 6 and 8 are currently supported. +The default value is 6. + +

mirror (boolean)

+ +

The mirror attribute specifies whether pages are mirrored on +their X axis, which is useful for printing transfer images on +special media. The default value is false. + +

number-up-layout (type2 keyword)Deprecated/Introduced in CUPS 1.1.15

+ +

The number-up-layout attribute specifies the order each input +page is placed on each output page. The following keywords are +presently defined: + +

    + +
  • btlr - Bottom to top, left to right
  • + +
  • btrl - Bottom to top, right to left
  • + +
  • lrbt - Left to right, bottom to top
  • + +
  • lrtb - Left to right, top to bottom (default)
  • + +
  • rlbt - Right to left, bottom to top
  • + +
  • rltb - Right to left, top to bottom
  • + +
  • tblr - Top to bottom, left to right
  • + +
  • tbrl - Top to bottom, right to left
  • + +
+ +
Note: + +

This attribute is deprecated in favor of the PWG presentation-direction-number-up attribute and will be removed in a future release.

+ +
+ +

page-border (type2 keyword)CUPS 1.1.15

+

The page-border attribute specifies whether a border is +draw around each page. The following keywords are presently +defined: + +

    + +
  • double - Two hairline borders are drawn
  • + +
  • double-thick - Two 1pt borders are drawn
  • + +
  • none - No border is drawn (default)
  • + +
  • single - A single hairline border is drawn
  • + +
  • single-thick - A single 1pt border is drawn
  • + +
+ +

page-bottom (integer(0:MAX))

+ +

The page-bottom attribute specifies the bottom margin in points (72 points +equals 1 inch). The default value is the device physical margin. + +

page-label (text(MAX))CUPS 1.1.7

+

The page-label attribute provides a text value to place in +the header and footer on each page. If a classification level is +set on the server, then this classification is printed before +the page label. + +

page-left (integer(0:MAX))

+ +

The page-left attribute specifies the left margin in points (72 points +equals 1 inch). The default value is the device physical margin. + +

page-right (integer(0:MAX))

+ +

The page-right attribute specifies the right margin in points (72 points +equals 1 inch). The default value is the device physical margin. + +

page-set (type2 keyword)

+ +

The page-set attribute specifies which pages to print in a file. The +supported keywords are "all", "even", and "odd". The default value is +"all". + +

page-top (integer(0:MAX))

+ +

The page-top attribute specifies the top margin in points (72 points +equals 1 inch). The default value is the device physical margin. + +

prettyprint (boolean)

+ +

The prettyprint attribute specifies whether text files should be printed +with a shaded header and keyword highlighting (prettyprint=true) or without +additional formatting (prettyprint=false). The default value is false. + +

wrap (boolean)

+ +

The wrap attribute specifies whether long lines should be wrapped +(wrap=true) or not (wrap=false) when printing text files. The default +value is true. + +

PPD Attributes

+ +

ppd-device-id (text(127))

+ +

The ppd-device-id attribute specifies the IEEE-1284 device ID +string for the device described by the PPD file.

+ +

ppd-make (text(127))

+ +

The ppd-make attribute specifies the manufacturer of the printer +(the Manufacturer attribute in the PPD file). If the manufacturer +is not specified in the PPD file then an educated guess is made using +the NickName attribute in the PPD file. + +

ppd-make-and-model (text(127))

+ +

The ppd-make-and-model attribute specifies the manufacturer and model +name of the PPD file (the NickName attribute in the PPD file). If the +make and model is not specified in the PPD file then the ModelName or +ShortNickName attributes are used instead. + +

ppd-model-number (integer)CUPS 1.3/Mac OS X 10.5

+ +

The ppd-model-number attribute provides the cupsModelNumber value from the PPD file. + +

ppd-name (name(255))

+ +

The ppd-name attribute specifies either the PPD filename on the server relative to the model directory or a URI that maps to a specific driver interface in the driver directory. The forward slash (/) is used to delineate directories. + +

ppd-natural-language (1setOf naturalLanguage)

+ +

The ppd-natural-language attribute specifies the language encoding +of the PPD file (the LanguageVersion attribute in the PPD file). If the +language is unknown or undefined then "en" (English) is assumed. + +

ppd-product (1setOf text(127))

+ +

The ppd-product attribute specifies the Product attribute values in the PPD file. + +

ppd-psversion (1setOf text(127))CUPS 1.3/Mac OS X 10.5

+ +

The ppd-product attribute specifies the PSVersion attribute values in the PPD file. + +

ppd-type (type1 keyword)CUPS 1.3/Mac OS X 10.5

+ +

The ppd-type attribute specifies the type of driver described by the PPD file:

+ +
    + +
  • fax - A facsimile or multi-function device
  • + +
  • pdf - A PDF printer
  • + +
  • postscript - A PostScript printer (no filters)
  • + +
  • raster - A CUPS raster driver
  • + +
  • unknown - An unknown or hybrid driver
  • + +
+ + +

Printer Attributes

+ +

auth-info-required (1setOf type2 keyword)CUPS 1.3/Mac OS X 10.5

+ +

The auth-info-required attribute specifies the authentication information that is required for printing a job. The following keywords are recognized:

+ +
    + +
  • domain - A domain name is required.
  • + +
  • none - No authentication is required - this keyword can only appear by itself.
  • + +
  • password - A password is required.
  • + +
  • username - A username is required.
  • + +
+ +

job-k-limit (integer)CUPS 1.1

+ +

The job-k-limit attribute specifies the maximum number of kilobytes that +may be printed by a user, including banner files. The default value of 0 +specifies that there is no limit. + +

job-page-limit (integer)CUPS 1.1

+ +

The job-page-limit attribute specifies the maximum number of pages that +may be printed by a user, including banner files. The default value of 0 +specifies that there is no limit. + +

job-quota-period (integer)CUPS 1.1

+ +

The job-quota-period attribute specifies the time period used for quota +calculations, in seconds. The default value of 0 specifies that the limits +apply to all jobs that have been printed by a user that are still known to +the system. + +

job-sheets-supported (1setof type3 keyword | name(MAX))CUPS 1.1

+ +

The job-sheets-supported attribute specifies the available banner files. +There will always be at least one banner file available called "none". + +

marker-change-time (integer)CUPS 1.3/Mac OS X 10.5

+ +

The marker-change-time attribute specifies the printer-up-time value when +the last change to the marker-colors, marker-levels, marker-message, +marker-names, or marker-types attributes was made.

+ +

marker-colors (1setof name(MAX))CUPS 1.3/Mac OS X 10.5

+ +

The marker-colors attribute specifies the color(s) for each supply in the +printer. It is only available when the driver provides supply levels. The +color is either "none" or one or more hex-encoded sRGB colors of the form +"#RRGGBB".

+ +

marker-high-levels (1setof integer(0:100))CUPS 1.4/Mac OS X 10.6

+ +

The marker-high-levels attribute specifies the supply levels that indicate +a near-full condition. A value of 100 should be used for supplies that are +consumed/emptied, e.g. ink cartridges.

+ +

marker-levels (1setof integer(-3:100))CUPS 1.3/Mac OS X 10.5

+ +

The marker-levels attribute specifies the current supply levels for the +printer. It is only available when the driver provides supply levels. A +value of -1 indicates the level is unavailable, -2 indicates unknown, and -3 +indicates the level is unknown but has not yet reached capacity. Values from 0 +to 100 indicate the corresponding percentage.

+ +

marker-low-levels (1setof integer(0:100))CUPS 1.4/Mac OS X 10.6

+ +

The marker-low-levels attribute specifies the supply levels that indicate +a near-empty condition. A value of 0 should be used for supplies that are +filled, e.g. waste ink tanks.

+ +

marker-message (text(MAX))CUPS 1.4/Mac OS X 10.6

+ +

The marker-message attribute provides a human-readable status message +for the current supply levels, e.g. "12 pages of ink remaining." It is only +available when the driver provides supply levels.

+ +

marker-names (1setof name(MAX))CUPS 1.3/Mac OS X 10.5

+ +

The marker-names attribute specifies the name(s) for each supply in the +printer. It is only available when the driver provides supply levels.

+ +

marker-types (1setof type3 keyword)CUPS 1.3/Mac OS X 10.5

+ +

The marker-types attribute specifies the type(s) of each supply in the +printer. It is only available when the driver provides supply levels. The +following (RFC 3805) types are currently supported:

+ +
    + +
  • toner
  • + +
  • wasteToner
  • + +
  • ink
  • + +
  • inkCartridge
  • + +
  • inkRibbon
  • + +
  • wasteInk
  • + +
  • opc
  • + +
  • developer
  • + +
  • fuserOil
  • + +
  • solidWax
  • + +
  • ribbonWax
  • + +
  • wasteWax
  • + +
  • fuser
  • + +
  • coronaWire
  • + +
  • fuserOilWick
  • + +
  • cleanerUnit
  • + +
  • fuserCleaningPad
  • + +
  • transferUnit
  • + +
  • tonerCartridge
  • + +
  • fuserOiler
  • + +
  • water
  • + +
  • wasteWater
  • + +
  • bindingSupply
  • + +
  • bandingSupply
  • + +
  • stichingWire
  • + +
  • shrinkWrap
  • + +
  • paperWrap
  • + +
  • staples
  • + +
  • inserts
  • + +
  • covers
  • + +
+ +

port-monitor" (name(127))

+ +

The port-monitor attribute specifies the port monitor to use when printing +to this printer. The default port monitor is "none". + +

port-monitor-supported" (1setOf name(127))

+ +

The port-monitor-supported attribute specifies the available port monitors. + +

printer-commands (1setOf Type3 keyword)CUPS 1.4/Mac OS X 10.6

+ +

The printer-commands attribute specifies the commands that are supported +by the CUPS command file filter. The keyword "none" indicates that no commands +are supported.

+ +

printer-dns-sd-name (name(MAX) | noValue)CUPS 1.4/Mac OS X 10.6

+ +

The printer-dns-sd-name attribute specifies the registered DNS-SD service +name for the printer. If the printer is not being shared using this protocol, +printer-dns-sd-name will have the noValue value.

+ +

printer-state-reasons (1setOf type2 keyword)

+ +

The printer-state-reasons attribute provides additional persistent state +information for a printer. In addition to the keywords defined in RFC 2911, +CUPS supports vendor-specific keywords with a domain prefix ("com.vendor.foo") +and the following CUPS-specific keywords:

+ +
    + +
  • cups-insecure-filter-warning - a filter or backend (or the + directory containing the filter or backend) has insecure file + permissions. CUPS will not execute programs with world write permissions + or setuid programs. When run as root (the default), CUPS also does not + execute programs that are not owned by root. + CUPS 1.4/Mac OS X 10.6
  • + +
  • cups-missing-filter-warning - a filter or backend is not + installed. CUPS 1.4/Mac OS X 10.6
  • + +
+ +

printer-type (type2 enum)

+ +

The printer-type attribute specifies printer type and +capability bits for the printer or class. The default value is +computed from internal state information and the PPD file for the +printer. The following bits are defined:

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
BitDescription
0x00000001Is a printer class.
0x00000002Is a remote destination.
0x00000004Can print in black.
0x00000008Can print in color.
0x00000010Can print on both sides of the page in hardware.
0x00000020Can staple output.
0x00000040Can do fast copies in hardware.
0x00000080Can do fast copy collation in hardware.
0x00000100Can punch output.
0x00000200Can cover output.
0x00000400Can bind output.
0x00000800Can sort output.
0x00001000Can handle media up to US-Legal/A4.
0x00002000Can handle media from US-Legal/A4 to ISO-C/A2.
0x00004000Can handle media larger than ISO-C/A2.
0x00008000Can handle user-defined media sizes.
0x00010000Is an implicit (server-generated) class.
0x00020000Is the a default printer on the network.
0x00040000Is a facsimile device.
0x00080000Is rejecting jobs.
0x00100000Delete this queue.
0x00200000Queue is not shared.
0x00400000Queue requires authentication.
0x00800000Queue supports CUPS command files.
0x01000000Queue was automatically discovered and added.
0x02000000Queue is a scanner with no printing capabilities.
0x04000000Queue is a printer with scanning capabilities.
+ +

printer-type-mask (type2 enum)CUPS 1.1

+ +

The printer-type-mask attribute is used to choose printers or classes with +the CUPS-Get-Printers and CUPS-Get-Classes operations. The bits are defined +identically to the printer-type attribute and default to all 1's. + +

requesting-user-name-allowed (1setof name(127))CUPS 1.1

+ +

The requesting-user-name-allowed attribute lists all of the users that are +allowed to access a printer or class. Either this attribute or the +requesting-user-name-denied attribute will be defined, but not both. + +

requesting-user-name-denied (1setof name(127))CUPS 1.1

+ +

The requesting-user-name-denied attribute lists all of the users that are +not allowed to access a printer or class. Either this attribute or the +requesting-user-name-allowed attribute will be defined, but not both. + +

Printer Class Attributes

+ +

member-names (1setof name(127))

+ +

The member-names attribute specifies each of the printer-name attributes of +the member printers and classes. Each name corresponds to the same element of +the member-uris attribute. + +

member-uris (1setof uri)

+ +

The member-uris attribute specifies each of the printer-uri attributes of +the member printers and classes. Each URI corresponds to the same element of +the member-names attribute. + + + diff --git a/doc/help/spec-pdf.html b/doc/help/spec-pdf.html new file mode 100644 index 0000000000..407f47887f --- /dev/null +++ b/doc/help/spec-pdf.html @@ -0,0 +1,24 @@ + + + + + CUPS PDF Format + + + + +

CUPS PDF Format

+ +

CUPS PDF files (application/vnd.cups-pdf) are device-dependent +PDF/A files that contain a job ticket information. These files +are typically produced by the CUPS pdftopdf filter +which handles job ticket generation, imposition, page labeling, +scaling, and other formatting options requested by the user. CUPS +PDF files are intended for direct consumption by a PDF-capable +printer, PDF RIP, or the pdftops filter.

+ +

More information will be posted here as the PDF workflow +filters are added to CUPS.

+ + + diff --git a/doc/help/spec-postscript.html b/doc/help/spec-postscript.html new file mode 100644 index 0000000000..3aa99c4126 --- /dev/null +++ b/doc/help/spec-postscript.html @@ -0,0 +1,148 @@ + + + + + Generating PostScript for CUPS + + + + + + +

Generating PostScript for CUPS

+ +

Introduction

+ +

This document describes how to generate PostScript output for +CUPS and is largely based on the +Adobe TechNote #5001: PostScript Language Document Structuring +Conventions Specification Version 3.0. While CUPS can +generally print any PostScript file, following the rules in the +Adobe TechNote and this document will ensure that your PostScript +output will work reliably.

+ +
Note: While PostScript is currently the +de-facto standard print job file format/language for UNIX-based +applications, it is slowly being phased out in favor of Adobe's +Portable Document Format ("PDF") which offers many advantages +over PostScript. Mac OS X uses PDF as the primary print job file +format and Linux is making the transition. Both PostScript and +PDF are complex formats, and we highly recommend using high-level +toolkits whenever possible to create your print jobs.
+ +

Anatomy of a PostScript File

+ +

PostScript files are ASCII text files starting with a header +line (%!PS-Adobe-3.0) followed by a combination of +comment lines starting with the percent sign (%) and +PostScript code lines. The lines themselves should not exceed 255 +characters to conform to the DSC. The following short PostScript +file produces a box with a smiley face in it:

+ +
+%!PS-Adobe-3.0
+%%BoundingBox: 36 36 576 756
+%%Pages: 1
+%%LanguageLevel: 2
+%%EndComments
+%%BeginSetup
+% this is where fonts would be embedded
+%%EndSetup
+%%Page: (1) 1
+%%BeginPageSetup
+% this is where page-specific features would be specified
+%%EndPageSetup
+% Draw a black box around the page
+0 setgray
+1 setlinewidth
+36 36 540 720 rectstroke
+
+% Draw a two inch blue circle in the middle of the page
+0 0 1 setrgbcolor
+306 396 144 0 360 arc closepath fill
+
+% Draw two half inch yellow circles for eyes
+1 1 0 setrgbcolor
+252 432 36 0 360 arc closepath fill
+360 432 36 0 360 arc closepath fill
+
+% Draw the smile
+1 setlinecap
+18 setlinewidth
+306 396 99 200 340 arc stroke
+
+% Print it!
+showpage
+%%EOF
+
+ + + + +

Embedding Printer Options

+ +

There are two main strategies for embedding printer options in PostScript +files. The first is to list CUPS options using the %cupsJobTicket +comment:

+ +
+%!PS-Adobe-3.0
+%cupsJobTicket: media=A4 sides=two-sided-long-edge
+%cupsJobTicket: PrinterOption=foo PrinterOption2=bar
+...
+%%EndComments
+
+ +

CUPS options apply to the entire job. To apply options to individual pages, +use the %%IncludeFeature comment instead:

+ +
+%%Page: label 123
+%%BeginPageSetup
+%%IncludeFeature: *PageSize A4
+%%IncludeFeature: *PrinterOption Foo
+%%IncludeFeature: *PrinterOption2 Bar
+%%EndPageSetup
+...
+
+ + +

Embedding Fonts and Text

+ +

Always embed the fonts used by your print job, and for best performance +embed the fonts and character encodings in the setup section of the PostScript +file. Type 1 and Type 3 fonts are supported by all PostScript printers, while +Type 42 (TrueType) and CID fonts are supported by most level 2 and all level 3 +PostScript printers. Binary font files should always be converted to the +corresponding ASCII (hex) encoding to avoid problems when printing over +interfaces that do not support binary PostScript.

+ + +

Embedding Images

+ +

The image operator should be used to embed images in PostScript +files. Always use ASCII hex or Base-85 encoding for the image data to avoid +problems when printing over interfaces that do not support binary PostScript. +In most cases, the Base-85 encoding and compression filters can be used to +embed images with very little, if any, increase in data size.

+ + + diff --git a/doc/help/spec-ppd.html b/doc/help/spec-ppd.html new file mode 100644 index 0000000000..f92725d810 --- /dev/null +++ b/doc/help/spec-ppd.html @@ -0,0 +1,2340 @@ + + + + + CUPS PPD Extensions + + + + + + +
+ + +

CUPS PPD Extensions

+ +

This specification describes the attributes and extensions that CUPS adds to Adobe TechNote #5003: PostScript Printer Description File Format Specification Version 4.3. PostScript Printer Description ("PPD") files describe the capabilities of each printer and are used by CUPS to support printer-specific features and intelligent filtering.

+ + +

Contents

+ +

PPD File Syntax

+ +

The PPD format is text-based and uses lines of up to 255 characters terminated by a carriage return, linefeed, or combination of carriage return and line feed. The following ABNF definition [RFC5234] defines the general format of lines in a PPD file:

+ +
+PPD-FILE = HEADER +(DATA / COMMENT / LINE-END)
+
+HEADER   = "*PPD-Adobe:" *WSP DQUOTE VERSION DQUOTE LINE-END
+
+VERSION  = "4.0" / "4.1" / "4.2" / "4.3"
+
+COMMENT  = "*%" *TCHAR LINE-END
+
+DATA     = "*" 1*KCHAR [ WSP 1*KCHAR [ "/" 1*TCHAR ] ] ":"
+           1*(*WSP VALUE) LINE-END
+
+VALUE    = 1*TCHAR / DQUOTE 1*SCHAR DQUOTE
+
+KCHAR    = ALPHA / DIGIT / "_" / "." / "-"
+
+SCHAR    = LINE-END / WSP / %x21.23-7E.A0-FF
+
+TCHAR    = %x20-7E.A0-FF
+
+LINE-END = CR / LF / CR LF
+
+ + +

Auto-Configuration

+ +

CUPS supports several methods of auto-configuration via PPD keywords.

+ +

Mac OS X 10.5APAutoSetupTool

+ +

*APAutoSetupTool: "/LibraryPrinters/vendor/filename"

+ +

This Mac OS X keyword defines a program that sets the default option choices. It is run when a printer is added from the Add Printer window or the Nearby Printers list in the Print dialog.

+ +

The program is provided with two arguments: the printer's device URI and the PPD file to be used for the printer. The program must write an updated PPD file to stdout.

+ +

Examples:

+ +
+*% Use our setup tool when adding a printer
+*APAutoSetupTool: "/Library/Printers/vendor/Tools/autosetuptool"
+
+ +

Mac OS X 10.2/CUPS 1.4?MainKeyword

+ +

*?MainKeyword: "
+ PostScript query code that writes a message using the = operator...
+"
+*End

+ +

The ?MainKeyword keyword defines PostScript code that determines the currently selected/enabled option keyword (choice) for the main keyword (option). It is typically used when communicating with USB, serial, Appletalk, and AppSocket (port 9100) printers.

+ +

The PostScript code typically sends its response back using the = operator.

+ +

Example:

+ +
+*OpenUI OptionDuplex/Duplexer Installed: Boolean
+*DuplexOptionDuplex: False
+*OptionDuplex False/Not Installed: ""
+*OptionDuplex True/Installed: ""
+
+*% Query the printer for the presence of the duplexer option...
+*?OptionDuplex: "
+  currentpagedevice /Duplex known
+  {(True)} {(False)} ifelse
+  = flush
+"
+*End
+*CloseUI: OptionDuplex
+
+ +

Mac OS X 10.4/CUPS 1.5OIDMainKeyword

+ +

*?OIDMainKeyword: ".n.n.n..."
+*OIDMainKeyword OptionKeyword1: "value"
+...
+*OIDMainKeyword OptionKeywordN: "value"

+ +

The OIDMainKeyword keyword is used to define SNMP OIDs that map to installable options. The first (query) line defines the OID to lookup on the network device. The second and subsequent keywords define a mapping from OID value to option keyword. Since SNMP is an IP-based network protocol, this method is typically only used to configure AppSocket, IPP, and LPD network printers.

+ +

Examples:

+ +
+*% Get the installed memory on the printer...
+*?OIDInstalledMemory: ".1.3.6.1.2.1.25.2.2.0"
+*OIDInstalledMemory 16MB: "16384 KBytes"
+*OIDInstalledMemory 32MB: "32768 KBytes"
+*OIDInstalledMemory 48MB: "49152 KBytes"
+*OIDInstalledMemory 72MB: "73728 KBytes"
+
+ + +

Color Profiles

+ +

CUPS supports three types of color profiles. The first type is based on sRGB and is used by the standard CUPS raster filters and GPL Ghostscript. The second type is based on ICC profiles and is used by the Quartz-based filters on MacOS X. The final type is based on well-known colorspaces such as sRGB and Adobe RGB.

+ +
Note: + +

At this time, none of the CUPS raster filters support ICC profiles. This will be addressed as time and resources permit.

+ +
+ +

DeprecatedcupsColorProfile

+ +

*cupsColorProfile Resolution/MediaType: "density gamma m00 m01 m02 m10 m11 m12 m20 m21 m22"

+ +

This string keyword specifies an sRGB-based color profile consisting of gamma and density controls and a 3x3 CMY color transform matrix. This keyword is not supported on Mac OS X.

+ +

The Resolution and MediaType values may be "-" to act as a wildcard. Otherwise they must match one of the Resolution or MediaType option keywords defined in the PPD file.

+ +

The density and gamma values define gamma and +density adjustment function such that:

+ +
+f(x) = density * x gamma
+
+ +

The m00 through m22 values define a 3x3 transformation matrix for the CMY color values. The density function is applied after the CMY transformation:

+ +
+| m00 m01 m02 |
+| m10 m11 m12 |
+| m20 m21 m22 |
+
+ +

Examples:

+ +
+*% Specify a profile for printing at 360dpi on all media types
+*cupsColorProfile 360dpi/-: "1.0 1.5 1.0 0.0 -0.2 -0.4 1.0 0.0 -0.2 0.0 1.0"
+
+*% Specify a profile for printing at 720dpi on Glossy media
+*cupsColorProfile 720dpi/Glossy: "1.0 2.5 1.0 0.0 -0.2 -0.4 1.0 0.0 -0.2 0.0 1.0"
+
+*% Specify a default profile for printing at all other resolutions and media types
+*cupsColorProfile -/-: "0.9 2.0 1.0 0.0 -0.2 -0.4 1.0 0.0 -0.2 0.0 1.0"
+
+ + +

Mac OS X 10.3/CUPS 1.2cupsICCProfile

+ +

*cupsICCProfile ColorModel.MediaType.Resolution/Description: "filename"

+ +

This keyword specifies an ICC color profile that is used to convert the document colors to the device colorspace. The ColorModel, MediaType, and Resolution option keywords specify a selector for color profiles. If omitted, the color profile will match any option keyword for the corresponding main keyword.

+ +

The Description specifies human-readable text that is associated with the color profile. The filename portion specifies the ICC color profile to use; if the filename is not absolute, it is loaded relative to the /usr/share/cups/profiles directory.

+ +

Examples:

+ +
+*% Specify a profile for CMYK printing at 360dpi on all media types
+*cupsICCProfile CMYK..360dpi/360dpi CMYK: "/Library/Printers/vendor/Profiles/foo-360-cmyk.icc"
+
+*% Specify a profile for RGB printing at 720dpi on Glossy media
+*cupsColorProfile RGB.Glossy.720dpi/720dpi Glossy: "/Library/Printers/vendor/Profiles/foo-720-glossy-rgb.icc"
+
+*% Specify a default profile for printing at all other resolutions and media types
+*cupsICCProfile ../Default: "/Library/Printers/vendor/Profiles/foo-default.icc"
+
+ +

Customizing the Profile Selection Keywords

+ +

The ColorModel, MediaType, and Resolution main keywords can be reassigned to different main keywords, allowing drivers to do color profile selection based on different parameters. The cupsICCQualifier1, cupsICCQualifier2, and cupsICCQualifier3 keywords define the mapping from selector to main keyword:

+ +
+*cupsICCQualifier1: MainKeyword1
+*cupsICCQualifier2: MainKeyword2
+*cupsICCQualifier3: MainKeyword3
+
+ +

The default mapping is as follows:

+ +
+*cupsICCQualifier1: ColorModel
+*cupsICCQualifier2: MediaType
+*cupsICCQualifier3: Resolution
+
+ +

Mac OS X 10.4Custom Color Matching Support

+ +

*APSupportsCustomColorMatching: true
+*APCustomColorMatchingName name/text: ""
+*APCustomColorMatchingProfile: profile
+*APDefaultCustomColorMatchingProfile: profile

+ +

These keywords tell the Mac OS X raster filters that the printer driver provides its own custom color matching and that generic color profiles should be used when generating 1-, 3-, and 4-component raster data as requested by the driver. The APCustomColorMatchingProfile and APDefaultColorMatchingProfile keywords specify alternate color profiles (sRGB or AdobeRGB) to use for 3-color (RGB) raster data.

+ +
Note: + +

Prior to Mac OS X 10.6, the default RGB color space was Apple's "GenericRGB". The new default in Mac OS X 10.6 and later is "sRGB". For more information, see "Mac OS X v10.6: About gamma 2.2" on Apple's support site.

+ +
+ +

Mac OS X 10.5APCustomColorMatchingName

+ +

*APCustomColorMatchingName name/text: ""

+ +

This keyword defines an alternate name for the color matching provided by a driver in the Color Matching print panel. The default is to use the name "Vendor Matching" or its localized equivalent.

+ +

Examples:

+ +
+*% Define the names for our color matching...
+*APCustomColorMatchingName name/AcmeColor(tm): ""
+*fr.APCustomColorMatchingName name/La AcmeColor(tm): ""
+
+ +

Mac OS X 10.5APCustomColorMatchingProfile

+ +

*APCustomColorMatchingProfile: name

+ +

This keyword defines a supported RGB color profile that can be used when doing custom color matching. Currently only sRGB, AdobeRGB, and GenericRGB are supported. If not specified, RGB data will use the GenericRGB colorspace.

+ +
Note: + +

If you provide multiple APCustomColorMatchingProfile keywords, you are responsible for providing the necessary user interface controls to select the profile in a print dialog pane. Add the named profile to the print settings using the key kPMCustomColorMatchingProfileKey.

+ +
+ +

Examples:

+ +
+*% Use sRGB for RGB color by default, but support both sRGB and AdobeRGB
+*APSupportsCustomColorMatching: true
+*APDefaultCustomColorMatchingProfile: sRGB
+*APCustomColorMatchingProfile: sRGB
+*APCustomColorMatchingProfile: AdobeRGB
+
+ +

Mac OS X 10.5APDefaultCustomColorMatchingProfile

+ +

*APDefaultCustomColorMatchingProfile: name

+ +

This keyword defines the default RGB color profile that will be used when doing custom color matching. Currently only sRGB, AdobeRGB, and GenericRGB are supported.

+ +

Examples:

+ +
+*% Use sRGB for RGB color by default
+*APSupportsCustomColorMatching: true
+*APDefaultCustomColorMatchingProfile: sRGB
+
+ +

Mac OS X 10.4APSupportsCustomColorMatching

+ +

*APSupportsCustomColorMatching: boolean

+ +

This keyword specifies that the driver provides its own custom color matching. When true, the default hand-off colorspace will be GenericGray, GenericRGB, or GenericCMYK depending on the number of components the driver requests. The APDefaultCustomColorMatchingProfile keyword can be used to override the default 3-component (RGB) colorspace.

+ +

The default for APSupportsCustomColorMatching is false.

+ +

Examples:

+ +
+*APSupportsCustomColorMatching: true
+*APDefaultCustomColorMatchingProfile: sRGB
+
+ + +

Constraints

+ +

Constraints are option choices that are not allowed by the driver or device, for example printing 2-sided transparencies. All versions of CUPS support constraints defined by the legacy Adobe UIConstraints and NonUIConstraints keywords which support conflicts between any two option choices, for example:

+ +
+*% Do not allow 2-sided printing on transparency media
+*UIConstraints: "*Duplex *MediaType Transparency"
+*UIConstraints: "*MediaType Transparency *Duplex"
+
+ +

While nearly all constraints can be expressed using these keywords, there are valid scenarios requiring constraints between more than two option choices. In addition, resolution of constraints is problematic since users and software have to guess how a particular constraint is best resolved.

+ +

CUPS 1.4 and higher define two new keywords for constraints, cupsUIConstraints and cupsUIResolver. Each cupsUIConstraints keyword points to a cupsUIResolver keyword which specifies alternate options that resolve the conflict condition. The same cupsUIResolver can be used by multiple cupsUIConstraints.

+ +
Note: + +

When developing PPD files that contain constraints, it is very important to use the cupstestppd(1) program to verify that your constraints are accurate and cannot result in unresolvable option selections.

+ +
+ +

CUPS 1.4/Mac OS X 10.6cupsUIConstraints

+ +

*cupsUIConstraints resolver: "*Keyword1 *Keyword2 ..."
+*cupsUIConstraints resolver: "*Keyword1 OptionKeyword1 *Keyword2 ..."
+*cupsUIConstraints resolver: "*Keyword1 *Keyword2 OptionKeyword2 ..."
+*cupsUIConstraints resolver: "*Keyword1 OptionKeyword1 *Keyword2 OptionKeyword2 ..."
+*cupsUIConstraints: "*InstallableKeyword1 OptionKeyword1 *Keyword2 OptionKeyword2 ..."

+ +

Lists two or more options which conflict. The "resolver" string is a (possibly unique) keyword which specifies which options to change when the constraint exists. When no resolver is provided, CUPS first tries the default choice followed by testing each option choice to resolve the conflict.

+ +

Examples:

+ +
+*% Specify that 2-sided printing cannot happen on transparencies
+*cupsUIConstraints transparency: "*Duplex *MediaType Transparency"
+
+*% Specify that envelope printing cannot happen from the paper trays
+*cupsUIConstraints envelope: "*PageSize Env10 *InputSlot Tray1"
+*cupsUIConstraints envelope: "*PageSize Env10 *InputSlot Tray1"
+*cupsUIConstraints envelope: "*PageSize EnvDL *InputSlot Tray2"
+*cupsUIConstraints envelope: "*PageSize EnvDL *InputSlot Tray2"
+
+*% Specify an installable option constraint for the envelope feeder
+*cupsUIConstraints: "*InputSlot EnvFeeder *InstalledEnvFeeder"
+
+*% Specify that photo printing cannot happen on plain paper or transparencies at 1200dpi
+*cupsUIConstraints photo: "*OutputMode Photo *MediaType Plain *Resolution 1200dpi"
+*cupsUIConstraints photo: "*OutputMode Photo *MediaType Transparency *Resolution 1200dpi"
+
+ +

CUPS 1.4/Mac OS X 10.6cupsUIResolver

+ +

*cupsUIResolver resolver: "*Keyword1 OptionKeyword1 *Keyword2 OptionKeyword2 ..."

+ +

Specifies two or more options to mark/select to resolve a constraint. The "resolver" string identifies a particular action to take for one or more cupsUIConstraints. The same action can be used for multiple constraints. The option keyword pairs are treated as an ordered list of option selections to try - only the first N selections will be used, where N is the minimum number of selections required. Because cupsResolveConflicts() will not change the most recent option selection passed to it, at least two options from the constraints must be listed to avoid situations where conflicts cannot be resolved.

+ +

Examples:

+ +
+*% Specify the options to change for the 2-sided transparency constraint
+*cupsUIResolver transparency: "*Duplex None *MediaType Plain"
+
+*% Specify the options to change for the envelope printing constraints.  Notice
+*% that we try to change the InputSlot to either the envelope feeder or the
+*% manual feed first, then we change the page size...
+*cupsUIResolver envelope: "*InputSlot EnvFeeder *InputSlot ManualFeed *PageSize Letter"
+
+*% Specify the options to change for the photo printing constraints
+*cupsUIResolver photo: "*OutputMode Best *Resolution 600dpi"
+
+ + +

Globalized PPD Support

+ +

CUPS 1.2 and higher adds support for PPD files containing multiple languages by following the following additional rules:

+ +
    + +
  1. The LanguageVersion MUST be English
  2. + +
  3. The LanguageEncoding MUST be ISOLatin1
  4. + +
  5. The cupsLanguages keyword MUST be provided and list each of the supported locales in the PPD file
  6. + +
  7. Main and option keywords MUST NOT exceed 34 (instead of 40) characters to allow room for the locale prefixes in translation keywords
  8. + +
  9. The main keyword "Translation" MUST NOT be used
  10. + +
  11. Translation strings included with the main and option keywords MUST NOT contain characters outside the ASCII subset of ISOLatin1 and UTF-8; developers wishing to use characters outside ASCII MUST provide a separate set of English localization keywords for the affected keywords.
  12. + +
  13. Localizations are specified using a locale prefix of the form "ll" or "ll_CC." where "ll" is the 2-letter ISO language code and "CC" is the 2-letter ISO country code
      +
    • A generic language translation ("ll") SHOULD be provided with country-specific differences ("ll_CC") provided only as needed
    • +
    • For historical reasons, the "zh" and "zh_CN" locales map to Simplified Chinese while the "zh_TW" locale maps to Traditional Chinese
    • +
  14. + +
  15. Locale-specific translation strings MUST be encoded using UTF-8.
  16. + +
  17. Main keywords MUST be localized using one of the following forms: +

    *ll.Translation MainKeyword/translation text: ""
    + *ll_CC.Translation MainKeyword/translation text: ""

  18. + +
  19. Option keywords MUST be localized using one of the following forms: +

    *ll.MainKeyword OptionKeyword/translation text: ""
    + *ll_CC.MainKeyword OptionKeyword/translation text: ""

  20. + +
  21. Localization keywords MAY appear anywhere after the first line of the PPD file
  22. + +
+ +
Note: + +

We use a LanguageEncoding value of ISOLatin1 and limit the allowed base translation strings to ASCII to avoid character coding issues that would otherwise occur. In addition, requiring the base translation strings to be in English allows for easier fallback translation when no localization is provided in the PPD file for a given locale.

+ +
+ +

Examples:

+ +
+*LanguageVersion: English
+*LanguageEncoding: ISOLatin1
+*cupsLanguages: "de fr_CA"
+*ModelName: "Foobar Laser 9999"
+
+*% Localize ModelName for French and German
+*fr_CA.Translation ModelName/La Foobar Laser 9999: ""
+*de.Translation ModelName/Foobar LaserDrucken 9999: ""
+
+*cupsIPPReason com.vendor-error/A serious error occurred: "/help/com.vendor/error.html"
+*% Localize printer-state-reason for French and German
+*fr_CA.cupsIPPReason com.vendor-error/Une erreur sèrieuse s'est produite: "/help/com.vendor/error.html"
+*de.cupsIPPReason com.vendor-error/Eine ernste Störung trat: "/help/com.vendor/error.html"
+
+...
+
+*OpenUI *InputSlot/Paper Source: PickOne
+*OrderDependency: 10 AnySetup *InputSlot
+*DefaultInputSlot: Auto
+*% Localize InputSlot for French and German
+*fr_CA.Translation InputSlot/Papier source: ""
+*de.Translation InputSlot/Papiereinzug: ""
+*InputSlot Auto/Default: "<</ManualFeed false>>setpagedevice"
+*% Localize InputSlot=Auto for French and German
+*fr_CA.InputSlot Auto/Par Defaut: ""
+*de.InputSlot Auto/Standard: ""
+*InputSlot Manual/Manual Feed: "<</ManualFeed true>>setpagedevice"
+*% Localize InputSlot=Manual for French and German
+*fr_CA.InputSlot Manual/Manuel mecanisme de alimentation: ""
+*de.InputSlot Manual/Manueller Einzug: ""
+*CloseUI: *InputSlot
+
+ + +

CUPS 1.3/Mac OS X 10.6Custom Options

+ +

CUPS supports custom options using an extension of the CustomPageSize and ParamCustomPageSize syntax:

+ +
+*CustomFoo True: "command"
+*ParamCustomFoo Name1/Text 1: order type minimum maximum
+*ParamCustomFoo Name2/Text 2: order type minimum maximum
+...
+*ParamCustomFoo NameN/Text N: order type minimum maximum
+
+ +

When the base option is part of the JCLSetup section, the "command" string contains JCL commands with "\order" placeholders for each numbered parameter. The CUPS API handles any necessary value quoting for HP-PJL commands. For example, if the JCL command string is "@PJL SET PASSCODE=\1" and the first +option value is "1234" then CUPS will output the string "@PJL SET PASSCODE=1234".

+ +

For non-JCLSetup options, the "order" value is a number from 1 to N and specifies the order of values as they are placed on the stack before the command. For example, if the PostScript command string is "<</cupsReal1 2 1 roll>>setpagedevice" and the option value is "2.0" then CUPS will output the string "2.0 <</cupsReal1 2 1 roll>>setpagedevice".

+ +

The "type" is one of the following keywords:

+ +
    + +
  • curve - a real value from "minimum" to "maximum" representing a gamma correction curve using the function: f(x) = x value
  • + +
  • int - an integer value from "minimum" to "maximum"
  • + +
  • invcurve - a real value from "minimum" to "maximum" representing a gamma correction curve using the function: f(x) = x 1 / value
  • + +
  • passcode - a string of numbers value with a minimum of "minimum" numbers and a maximum of "maximum" numbers ("minimum" and "maximum" are numbers and passcode strings are not displayed in the user interface)
  • + +
  • password - a string value with a minimum of "minimum" characters and a maximum of "maximum" characters ("minimum" and "maximum" are numbers and password strings are not displayed in the user interface)
  • + +
  • points - a measurement value in points from "minimum" to "maximum"
  • + +
  • real - a real value from "minimum" to "maximum"
  • + +
  • string - a string value with a minimum of "minimum" characters and a maximum of "maximum" characters ("minimum" and "maximum" are numbers)
  • + +
+ +

Examples:

+ +
+*% Base JCL key code option
+*JCLOpenUI JCLPasscode/Key Code: PickOne
+*OrderDependency: 10 JCLSetup *JCLPasscode
+*DefaultJCLPasscode: None
+*JCLPasscode None/No Code: ""
+*JCLPasscode 1111: "@PJL SET PASSCODE = 1111<0A>"
+*JCLPasscode 2222: "@PJL SET PASSCODE = 2222<0A>"
+*JCLPasscode 3333: "@PJL SET PASSCODE = 3333<0A>"
+*JCLCloseUI: *JCLPasscode
+
+*% Custom JCL key code option
+*CustomJCLPasscode True: "@PJL SET PASSCODE = \1<0A>"
+*ParamCustomJCLPasscode Code/Key Code: 1 passcode 4 4
+
+
+*% Base PostScript watermark option
+*OpenUI WatermarkText/Watermark Text: PickOne
+*OrderDependency: 10 AnySetup *WatermarkText
+*DefaultWatermarkText: None
+*WatermarkText None: ""
+*WatermarkText Draft: "<</cupsString1(Draft)>>setpagedevice"
+*CloseUI: *WatermarkText
+
+*% Custom PostScript watermark option
+*CustomWatermarkText True: "<</cupsString1 3 -1 roll>>setpagedevice"
+*ParamCustomWatermarkText Text: 1 string 0 32
+
+
+*% Base PostScript gamma/density option
+*OpenUI GammaDensity/Gamma and Density: PickOne
+*OrderDependency: 10 AnySetup *GammaDensity
+*DefaultGammaDensity: Normal
+*GammaDensity Normal/Normal: "<</cupsReal1 1.0/cupsReal2 1.0>>setpagedevice"
+*GammaDensity Light/Lighter: "<</cupsReal1 0.9/cupsReal2 0.67>>setpagedevice"
+*GammaDensity Dark/Darker: "<</cupsReal1 1.1/cupsReal2 1.5>>setpagedevice"
+*CloseUI: *GammaDensity
+
+*% Custom PostScript gamma/density option
+*CustomGammaDensity True: "<</cupsReal1 3 -1 roll/cupsReal2 5 -1>>setpagedevice"
+*ParamCustomGammaDensity Gamma: 1 curve 0.1 10
+*ParamCustomGammaDensity Density: 2 real 0 2
+
+ + +

Writing PostScript Option Commands for Raster Drivers

+ +

PPD files are used for both PostScript and non-PostScript printers. For CUPS raster drivers, you use a subset of the PostScript language to set page device keywords such as page size, resolution, and so forth. For example, the following code sets the page size to A4 size:

+ +
+*PageSize A4: "<</PageSize[595 842]>>setpagedevice"
+
+ +

Custom options typically use other operators to organize the values into a key/value dictionary for setpagedevice. For example, our previous CustomWatermarkText option code uses the roll operator to move the custom string value into the dictionary for setpagedevice:

+ +
+*CustomWatermarkText True: "<</cupsString1 3 -1 roll>>setpagedevice"
+
+ +

For a custom string value of "My Watermark", CUPS will produce the following PostScript code for the option:

+ +
+(My Watermark)
+<</cupsString1 3 -1 roll>>setpagedevice
+
+ +

The code moves the string value ("My Watermark") from the bottom of the stack to the top, creating a dictionary that looks like:

+ +
+<</cupsString1(My Watermark)>>setpagedevice
+
+ +

The resulting dictionary sets the page device attributes that are sent to your raster driver in the page header.

+ +

Custom Page Size Code

+ +

There are many possible implementations of the CustomPageSize code. For CUPS raster drivers, the following code is recommended:

+ +
+*ParamCustomPageSize Width:        1 points min-width max-width
+*ParamCustomPageSize Height:       2 points min-height max-height
+*ParamCustomPageSize WidthOffset:  3 points 0 0
+*ParamCustomPageSize HeightOffset: 4 points 0 0
+*ParamCustomPageSize Orientation:  5 int 0 0
+*CustomPageSize True: "pop pop pop <</PageSize[5 -2 roll]/ImagingBBox null>>setpagedevice"
+
+ +

Supported PostScript Operators

+ +

CUPS supports the following PostScript operators in addition to the usual PostScript number, string (literal and hex-encoded), boolean, null, and name values:

+ +
    + +
  • << - Start a dictionary.
  • + +
  • >> - End a dictionary.
  • + +
  • [ - Start an array.
  • + +
  • ] - End an array.
  • + +
  • copy - Copy the top N objects on the stack.
  • + +
  • dup - Copy the top object on the stack.
  • + +
  • index - Copy the Nth from the top object on the stack.
  • + +
  • pop - Pop the top object on the stack.
  • + +
  • roll - Shift the top N objects on the stack.
  • + +
  • setpagedevice - Set the page header values according to the key/value dictionary on the stack.
  • + +
+ +
Note: + +

Never use the unsupported dict or put +operators in your option code. These operators are typically used in +option code dating back to Level 1 PostScript printers, which did not +support the simpler << or >> operators. +If you have old option code using dict or put, you can +rewrite it very easily to use the newer << and +>> operators instead. For example, the following code +to set the page size:

+ + + +
+1 dict dup /PageSize [612 792] put setpagedevice
+
+ +

can be rewritten as:

+ +
+<< /PageSize [612 792] >> setpagedevice
+
+ +
+ +

Supported Page Device Attributes

+ +

Table 2 shows the supported page device attributes along with PostScript code examples.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 2: Supported Page Device Attributes
Name(s)TypeDescriptionExample(s)
AdvanceDistanceIntegerSpecifies the number of points to advance roll media after printing.<</AdvanceDistance 18>>setpagedevice
AdvanceMediaIntegerSpecifies when to advance the media: 0 = never, 1 = after the file, 2 = after the job, 3 = after the set, and 4 = after the page.<</AdvanceMedia 4>>setpagedevice
CollateBooleanSpecifies whether collated copies are required.<</Collate true>>setpagedevice
CutMediaIntegerSpecifies when to cut the media: 0 = never, 1 = after the file, 2 = after the job, 3 = after the set, and 4 = after the page.<</CutMedia 1>>setpagedevice
DuplexBooleanSpecifies whether 2-sided printing is required.<</Duplex true>>setpagedevice
HWResolutionInteger ArraySpecifies the resolution of the page image in pixels per inch.<</HWResolution[1200 1200]>>setpagedevice
InsertSheetBooleanSpecifies whether to insert a blank sheet before the job.<</InsertSheet true>>setpagedevice
JogIntegerSpecifies when to shift the media in the output bin: 0 = never, 1 = after the file, 2 = after the job, 3 = after the set, and 4 = after the page.<</Jog 2>>setpagedevice
LeadingEdgeIntegerSpecifies the leading edge of the media: 0 = top, 1 = right, 2 = bottom, 3 = left.<</LeadingEdge 0>>setpagedevice
ManualFeedBooleanSpecifies whether media should be drawn from the manual feed tray. Note: The MediaPosition attribute is preferred over the ManualFeed attribute.<</ManualFeed true>>setpagedevice
MediaClassStringSpecifies a named media.<</MediaClass (Invoices)>>setpagedevice
MediaColorStringSpecifies the color of the media.<</MediaColor >>setpagedevice
MediaPositionIntegerSpecifies the tray or source of the media.<</MediaPosition 12>>setpagedevice
MediaTypeStringSpecifies the general media type.<</MediaType (Glossy)>>setpagedevice
MediaWeightIntegerSpecifies the media weight in grams per meter2.<</MediaWeight 100>>setpagedevice
MirrorPrintBooleanSpecifies whether to flip the output image horizontally.<</MirrorPrint true>>setpagedevice
NegativePrintBooleanSpecifies whether to invert the output image.<</NegativePrint true>>setpagedevice
NumCopiesIntegerSpecifies the number of copies to produce of each page.<</NumCopies 100>>setpagedevice
OrientationIntegerSpecifies the orientation of the output: 0 = portrait, 1 = landscape rotated counter-clockwise, 2 = upside-down, 3 = landscape rotated clockwise.<</Orientation 3>>setpagedevice
OutputFaceUpBooleanSpecifies whether to place the media face-up in the output bin/tray.<</OutputFaceUp true>>setpagedevice
OutputTypeStringSpecifies the output type name.<</OutputType (Photo)>>setpagedevice
PageSizeInteger/Real ArraySpecifies the width and length/height of the page in points.<</PageSize[595 842]>>setpagedevice
SeparationsBooleanSpecifies whether to produce color separations.<</Separations true>>setpagedevice
TraySwitchBooleanSpecifies whether to switch trays automatically.<</TraySwitch true>>setpagedevice
TumbleBooleanSpecifies whether the back sides of pages are rotated 180 degrees.<</Tumble true>>setpagedevice
cupsBorderlessScalingFactorRealSpecifies the amount to scale the page image dimensions.<</cupsBorderlessScalingFactor 1.01>>setpagedevice
cupsColorOrderIntegerSpecifies the order of colors: 0 = chunked, 1 = banded, 2 = planar.<</cupsColorOrder 0>>setpagedevice
cupsColorSpaceIntegerSpecifies the page image colorspace: 0 = W, 1 = RGB, 2 = RGBA, 3 = K, 4 = CMY, 5 = YMC, 6 = CMYK, 7 = YMCK, 8 = KCMY, 9 = KCMYcm, 10 = GMCK, 11 = GMCS, 12 = White, 13 = Gold, 14 = Silver, 15 = CIE XYZ, 16 = CIE Lab, 17 = RGBW, 32 to 46 = CIE Lab (1 to 15 inks)<</cupsColorSpace 1 >>setpagedevice
cupsCompressionIntegerSpecifies a driver compression type/mode.<</cupsCompression 2>>setpagedevice
cupsInteger0
+ ...
+ cupsInteger15
IntegerSpecifies driver integer values.<</cupsInteger11 1234>>setpagedevice
cupsMarkerTypeStringSpecifies the type of ink/toner to use.<</cupsMarkerType (Black+Color)>>setpagedevice
cupsMediaTypeIntegerSpecifies a numeric media type.<</cupsMediaType 999>>setpagedevice
cupsPageSizeNameStringSpecifies the name of the page size.<</cupsPageSizeName (A4.Full)>>setpagedevice
cupsPreferredBitsPerColorIntegerSpecifies the preferred number of bits per color, typically 8 or 16.<</cupsPreferredBitsPerColor 16>>setpagedevice
cupsReal0
+ ...
+ cupsReal15
RealSpecifies driver real number values.<</cupsReal15 1.234>>setpagedevice
cupsRenderingIntentStringSpecifies the color rendering intent.<</cupsRenderingIntent (AbsoluteColorimetric)>>setpagedevice
cupsRowCountIntegerSpecifies the number of rows of raster data to print on each line for some drivers.<</cupsRowCount 24>>setpagedevice
cupsRowFeedIntegerSpecifies the number of rows to feed between passes for some drivers.<</cupsRowFeed 17>>setpagedevice
cupsRowStepIntegerSpecifies the number of lines between columns/rows on the print head for some drivers.<</cupsRowStep 2>>setpagedevice
cupsString0
+ ...
+ cupsString15
StringSpecifies driver string values.<</cupsString0(String Value)>>setpagedevice
+ + +

Media Keywords

+ +

The CUPS media keywords allow drivers to specify alternate custom page +size limits based on up to two options.

+ +

CUPS 1.4/Mac OS X 10.6cupsMediaQualifier2

+ +

*cupsMediaQualifier2: MainKeyword

+ +

This keyword specifies the second option to use for overriding the +custom page size limits.

+ +

Example:

+ +
+*% Specify alternate custom page size limits based on InputSlot and Quality
+*cupsMediaQualifier2: InputSlot
+*cupsMediaQualifier3: Quality
+*cupsMaxSize .Manual.: "1000 1000"
+*cupsMinSize .Manual.: "100 100"
+*cupsMinSize .Manual.Photo: "200 200"
+*cupsMinSize ..Photo: "300 300"
+
+ +

CUPS 1.4/Mac OS X 10.6cupsMediaQualifier3

+ +

*cupsMediaQualifier3: MainKeyword

+ +

This keyword specifies the third option to use for overriding the +custom page size limits.

+ +

Example:

+ +
+*% Specify alternate custom page size limits based on InputSlot and Quality
+*cupsMediaQualifier2: InputSlot
+*cupsMediaQualifier3: Quality
+*cupsMaxSize .Manual.: "1000 1000"
+*cupsMinSize .Manual.: "100 100"
+*cupsMinSize .Manual.Photo: "200 200"
+*cupsMinSize ..Photo: "300 300"
+
+ +

CUPS 1.4/Mac OS X 10.6cupsMinSize

+ +

*cupsMinSize .Qualifier2.Qualifier3: "width length"
+*cupsMinSize .Qualifier2.: "width length"
+*cupsMinSize ..Qualifier3: "width length"

+ +

This keyword specifies alternate minimum custom page sizes in points. +The cupsMediaQualifier2 and +cupsMediaQualifier3 keywords +are used to identify options to use for matching.

+ +

Example:

+ +
+*% Specify alternate custom page size limits based on InputSlot and Quality
+*cupsMediaQualifier2: InputSlot
+*cupsMediaQualifier3: Quality
+*cupsMaxSize .Manual.: "1000 1000"
+*cupsMinSize .Manual.: "100 100"
+*cupsMinSize .Manual.Photo: "200 200"
+*cupsMinSize ..Photo: "300 300"
+
+ +

CUPS 1.4/Mac OS X 10.6cupsMaxSize

+ +

*cupsMaxSize .Qualifier2.Qualifier3: "width length"
+*cupsMaxSize .Qualifier2.: "width length"
+*cupsMaxSize ..Qualifier3: "width length"

+ +

This keyword specifies alternate maximum custom page sizes in points. +The cupsMediaQualifier2 and +cupsMediaQualifier3 keywords +are used to identify options to use for matching.

+ +

Example:

+ +
+*% Specify alternate custom page size limits based on InputSlot and Quality
+*cupsMediaQualifier2: InputSlot
+*cupsMediaQualifier3: Quality
+*cupsMaxSize .Manual.: "1000 1000"
+*cupsMinSize .Manual.: "100 100"
+*cupsMinSize .Manual.Photo: "200 200"
+*cupsMinSize ..Photo: "300 300"
+
+ + +

General Attributes

+ +

CUPS 1.3/Mac OS X 10.5cupsBackSide

+ +

*cupsBackSide: keyword

+ +

This keyword requests special handling of the back side of pages +when doing duplexed (2-sided) output. Table 1 +shows the supported keyword values for this keyword and their effect +on the raster data sent to your driver. For example, when cupsBackSide +is Rotated and Tumble is false, your driver +will receive print data starting at the bottom right corner of the page, with +each line going right-to-left instead of left-to-right. The default value is +Normal.

+ +
Note: + +

cupsBackSide replaces the older cupsFlipDuplex +keyword - if cupsBackSide is specified, cupsFlipDuplex +will be ignored.

+ +
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 1: Back Side Raster Coordinate System
cupsBackSideTumble ValueImage Presentation
NormalfalseLeft-to-right, top-to-bottom
NormaltrueLeft-to-right, top-to-bottom
ManualTumblefalseLeft-to-right, top-to-bottom
ManualTumbletrueRight-to-left, bottom-to-top
RotatedfalseRight-to-left, bottom-to-top
RotatedtrueRight-to-left, top-to-bottom
Flipped *falseLeft-to-right, bottom-to-top
Flipped *trueRight-to-left, top-to-bottom
+
+ +

* - Not supported in Mac OS X 10.5.x and earlier

+ +
+ + +
Figure 1: Back side images
Back side images
+ +

Examples:

+ +
+*% Flip the page image for the back side of duplexed output
+*cupsBackSide: Flipped
+
+*% Rotate the page image for the back side of duplexed output
+*cupsBackSide: Rotated
+
+ +

Also see the related APDuplexRequiresFlippedMargin +keyword.

+ +

CUPS 1.4/Mac OS X 10.6cupsCommands

+ +

*cupsCommands: "name name2 ... nameN"

+ +

This string keyword specifies the commands that are supported by the +CUPS command file filter for this device. The command names are separated +by whitespace.

+ +

Example:

+ +
+*% Specify the list of commands we support
+*cupsCommands: "AutoConfigure Clean PrintSelfTestPage ReportLevels com.vendor.foo"
+
+ + +

CUPS 1.3/Mac OS X 10.5cupsEvenDuplex

+ +

*cupsEvenDuplex: boolean

+ +

This boolean keyword notifies the RIP filters that the +destination printer requires an even number of pages when 2-sided +printing is selected. The default value is false.

+ +

Example:

+ +
+*% Always send an even number of pages when duplexing
+*cupsEvenDuplex: true
+
+ +

cupsFax

+ +

*cupsFax: boolean

+ +

This boolean keyword specifies whether the PPD defines a facsimile device. The default is false.

+ +

Examples:

+ +
+*cupsFax: true
+
+ +

cupsFilter

+ +

*cupsFilter: "source/type cost program"

+ +

This string keyword provides a conversion rule from the +given source type to the printer's native format using the +filter "program". If a printer supports the source type directly, +the special filter program "-" may be specified.

+ +

Examples:

+ +
+*% Standard raster printer driver filter
+*cupsFilter: "application/vnd.cups-raster 100 rastertofoo"
+
+*% Plain text filter
+*cupsFilter: "text/plain 10 texttofoo"
+
+*% Pass-through filter for PostScript printers
+*cupsFilter: "application/vnd.cups-postscript 0 -"
+
+ +

CUPS 1.5cupsFilter2

+ +

*cupsFilter2: "source/type destination/type cost program"

+ +

This string keyword provides a conversion rule from the given source type to the printer's native format using the filter "program". If a printer supports the source type directly, the special filter program "-" may be specified. The destination type is automatically created as needed and is passed to the filters and backend as the FINAL_CONTENT_TYPE value.

+ +
Note: + +

The presence of a single cupsFilter2 keyword in the PPD file will hide any cupsFilter keywords from the CUPS scheduler. When using cupsFilter2 to provide filters specific for CUPS 1.5 and later, provide a cupsFilter2 line for every filter and a cupsFilter line for each filter that is compatible with older versions of CUPS.

+ +
+ +

Examples:

+ +
+*% Standard raster printer driver filter
+*cupsFilter2: "application/vnd.cups-raster application/vnd.foo 100 rastertofoo"
+
+*% Plain text filter
+*cupsFilter2: "text/plain application/vnd.foo 10 texttofoo"
+
+*% Pass-through filter for PostScript printers
+*cupsFilter2: "application/vnd.cups-postscript application/postscript 0 -"
+
+ +

DeprecatedcupsFlipDuplex

+ +

*cupsFlipDuplex: boolean

+ +

Due to implementation differences between Mac OS X and Ghostscript, +the cupsFlipDuplex keyword is deprecated. Instead, use +the cupsBackSide keyword to specify +the coordinate system (pixel layout) of the page data on the back side of +duplex pages.

+ +

The value true maps to a cupsBackSide value +of Rotated on Mac OS X and Flipped with +Ghostscript.

+ +

The default value is false.

+ +
Note: + +

Mac OS X drivers that previously used +cupsFlipDuplex may wish to provide both the old and +new keywords for maximum compatibility, for example:

+ +
+*cupsBackSide: Rotated
+*cupsFlipDuplex: true
+
+ +

Similarly, drivers written for other operating systems using +Ghostscript can use:

+ +
+*cupsBackSide: Flipped
+*cupsFlipDuplex: true
+
+ +

CUPS 1.3/Mac OS X 10.5cupsIPPFinishings

+ +

*cupsIPPFinishings number/text: "*Option Choice ..."

+ +

This keyword defines a mapping from IPP finishings +values to PPD options and choices.

+ +

Examples:

+ +
+*cupsIPPFinishings 4/staple: "*StapleLocation SinglePortrait"
+*cupsIPPFinishings 5/punch: "*PunchMedia Yes *PunchLocation LeftSide"
+*cupsIPPFinishings 20/staple-top-left: "*StapleLocation SinglePortrait"
+*cupsIPPFinishings 21/staple-bottom-left: "*StapleLocation SingleLandscape"
+
+ +

CUPS 1.3/Mac OS X 10.5cupsIPPReason

+ +

*cupsIPPReason reason/Reason Text: "optional URIs"

+ +

This optional keyword maps custom +printer-state-reasons keywords that are generated by +the driver to human readable text. The optional URIs string +contains zero or more URIs separated by a newline. Each URI can +be a CUPS server absolute path to a help file under the +scheduler's DocumentRoot directory, a full HTTP URL +("http://www.domain.com/path/to/help/page.html"), or any other +valid URI which directs the user at additional information +concerning the condition that is being reported.

+ +

Since the reason text is limited to 80 characters by the PPD specification, longer text strings can be included by URI-encoding the text with the "text" scheme, for example "text:some%20text". Multiple text URIs are combined by the ppdLocalizeIPPReason into a single string that can be displayed to the user.

+ +

Examples:

+ +
+*% Map com.vendor-error to text but no page
+*cupsIPPReason com.vendor-error/A serious error occurred: ""
+
+*% Map com.vendor-error to more than 80 characters of text but no page
+*cupsIPPReason com.vendor-error/A serious error occurred: "text:Now%20is%20the%20time
+text:for%20all%20good%20men%20to%20come%20to%20the%20aid%20of%20their%20country."
+
+*% Map com.vendor-error to text and a local page
+*cupsIPPReason com.vendor-error/A serious error occurred: "/help/com.vendor/error.html"
+
+*% Map com.vendor-error to text and a remote page
+*cupsIPPReason com.vendor-error/A serious error occurred: "http://www.vendor.com/help"
+
+*% Map com.vendor-error to text and a local, Apple help book, and remote page
+*APHelpBook: "file:///Library/Printers/vendor/Help.bundle"
+*cupsIPPReason com.vendor-error/A serious error occurred: "/help/com.vendor/error.html
+help:anchor='com.vendor-error'%20bookID=Vendor%20Help
+http://www.vendor.com/help"
+*End
+
+ +

CUPS 1.5cupsIPPSupplies

+ +

*cupsIPPSupplies: boolean

+ +

This keyword tells the IPP backend whether it should report the current marker-xxx supply attribute values. The default value is True. + +

Example:

+ +
+*% Do not use IPP marker-xxx attributes to report supply levels
+*cupsIPPSupplies: False
+
+ +

CUPS 1.2/Mac OS X 10.5cupsLanguages

+ +

*cupsLanguages: "locale list"

+ +

This keyword describes which language localizations are +included in the PPD. The "locale list" string is a space-delimited +list of locale names ("en", "en_US", "fr_CA", etc.)

+ +

Example:

+ +
+*% Specify Canadian, UK, and US English, and Canadian and French French
+*cupsLanguages: "en_CA en_UK en_US fr_CA fr_FR"
+
+ +

cupsManualCopies

+ +

*cupsManualCopies: boolean

+ +

This boolean keyword notifies the RIP filters that the +destination printer does not support copy generation in +hardware. The default value is false.

+ +

Example:

+ +
+*% Tell the RIP filters to generate the copies for us
+*cupsManualCopies: true
+
+ +

CUPS 1.4/Mac OS X 10.6cupsMarkerName

+ +

*cupsMarkerName/Name Text: ""

+ +

This optional keyword maps marker-names strings that are +generated by the driver to human readable text.

+ +

Examples:

+ +
+*% Map cyanToner to "Cyan Toner"
+*cupsMarkerName cyanToner/Cyan Toner: ""
+
+ +

CUPS 1.4/Mac OS X 10.6cupsMarkerNotice

+ +

*cupsMarkerNotice: "disclaimer text"

+ +

This optional keyword provides disclaimer text for the supply level +information provided by the driver, typically something like "supply levels +are approximate".

+ +

Examples:

+ +
+*cupsMarkerNotice: "Supply levels are approximate."
+
+ +

cupsModelNumber

+ +

*cupsModelNumber: number

+ +

This integer keyword specifies a printer-specific model +number. This number can be used by a filter program to adjust +the output for a specific model of printer.

+ +

Example:

+ +
+*% Specify an integer for a driver-specific model number
+*cupsModelNumber: 1234
+
+ +

CUPS 1.3/Mac OS X 10.5cupsPJLCharset

+ +

*cupsPJLCharset: "ISO character set name"

+ +

This string keyword specifies the character set that is used +for strings in PJL commands. If not specified, US-ASCII is +assumed.

+ +

Example:

+ +
+*% Specify UTF-8 is used in PJL strings
+*cupsPJLCharset: "UTF-8"
+
+ +

CUPS 1.4/Mac OS X 10.6cupsPJLDisplay

+ +

*cupsPJLDisplay: "what"

+ +

This optional keyword specifies which command is used to display the +job ID, name, and user on the printer's control panel. "What" is either "none" +to disable this functionality, "job" to use "@PJL JOB DISPLAY", or "rdymsg" +to use "@PJL RDYMSG DISPLAY". The default is "job".

+ +

Examples:

+ +
+*% Display job information using @PJL SET RDYMSG DISPLAY="foo"
+*cupsPJLDisplay: "rdymsg"
+
+*% Display job information display
+*cupsPJLDisplay: "none"
+
+ +

CUPS 1.2/Mac OS X 10.5cupsPortMonitor

+ +

*cupsPortMonitor urischeme/Descriptive Text: "port monitor"

+ +

This string keyword specifies printer-specific "port +monitor" filters that may be used with the printer. The CUPS +scheduler also looks for the Protocols keyword to see +if the BCP or TBCP protocols are supported. If +so, the corresponding port monitor ("bcp" and "tbcp", +respectively) is listed in the printer's +port-monitor-supported keyword.

+ +

The "urischeme" portion of the keyword specifies the URI scheme +that this port monitor should be used for. Typically this is used to +pre-select a particular port monitor for each type of connection that +is supported by the printer. The "port monitor" string can be "none" +to disable the port monitor for the given URI scheme.

+ +

Examples:

+ +
+*% Specify a PostScript printer that supports the TBCP protocol
+*Protocols: TBCP PJL
+
+*% Specify that TBCP should be used for socket connections but not USB
+*cupsPortMonitor socket/AppSocket Printing: "tbcp"
+*cupsPortMonitor usb/USB Printing: "none"
+
+*% Specify a printer-specific port monitor for an Epson USB printer
+*cupsPortMonitor usb/USB Status Monitor: "epson-usb"
+
+ +

CUPS 1.3/Mac OS X 10.5cupsPreFilter

+ +

*cupsPreFilter: "source/type cost program"

+ +

This string keyword provides a pre-filter rule. The pre-filter +program will be inserted in the conversion chain immediately +before the filter that accepts the given MIME type.

+ +

Examples:

+ +
+*% PDF pre-filter
+*cupsPreFilter: "application/pdf 100 mypdfprefilter"
+
+*% PNG pre-filter
+*cupsPreFilter: "image/png 0 mypngprefilter"
+
+ + +

CUPS 1.5cupsPrintQuality

+ +

*cupsPrintQuality keyword/text: "code"

+ +

This UI keyword defines standard print qualities that directly map from the IPP "print-quality" job template keyword. Standard keyword values are "Draft", "Normal", and "High" which are mapped from the IPP "print-quality" values 3, 4, and 5 respectively. Each cupsPrintQuality option typically sets output mode and resolution parameters in the page device dictionary, eliminating the need for separate (and sometimes confusing) output mode and resolution options.

+ +
Note: + +

Unlike all of the other keywords defined in this document, cupsPrintQuality is a UI keyword that MUST be enclosed inside the PPD OpenUI and CloseUI keywords.

+ +
+ +

Examples:

+ +
+*OpenUI *cupsPrintQuality/Print Quality: PickOne
+*OrderDependency: 10 AnySetup *cupsPrintQuality
+*DefaultcupsPrintQuality: Normal
+*cupsPrintQuality Draft/Draft: "code"
+*cupsPrintQuality Normal/Normal: "code"
+*cupsPrintQuality High/Photo: "code"
+*CloseUI: *cupsPrintQuality
+
+ +

CUPS 1.5cupsSingleFile

+ +

*cupsSingleFile: Boolean

+ +

This boolean keyword tells the scheduler whether to print multiple files in a job together or singly. The default is "False" which uses a single instance of the backend for all files in the print job. Setting this keyword to "True" will result in separate instances of the backend for each file in the print job.

+ +

Examples:

+ +
+*% Send all print data to a single backend
+*cupsSingleFile: False
+
+*% Send each file using a separate backend
+*cupsSingleFile: True
+
+ +

CUPS 1.4/Mac OS X 10.6cupsSNMPSupplies

+ +

*cupsSNMPSupplies: boolean

+ +

This keyword tells the standard network backends whether they should query +the standard SNMP Printer MIB OIDs for supply levels. The default value is +True. + +

Example:

+ +
+*% Do not use SNMP queries to report supply levels
+*cupsSNMPSupplies: False
+
+ +

cupsVersion

+ +

*cupsVersion: major.minor

+ +

This required keyword describes which version of the CUPS +PPD file extensions was used. Currently it must be the string +"1.0", "1.1", "1.2", or "1.3".

+ +

Example:

+ +
+*% Specify a CUPS 1.2 driver
+*cupsVersion: "1.2"
+
+ + +

Mac OS X Attributes

+ +

Mac OS X 10.3APDialogExtension

+ +

*APDialogExtension: "/Library/Printers/vendor/filename.plugin"

+ +

This keyword defines additional option panes that are displayed in the +print dialog. Each keyword adds one or more option panes. See the "OutputBinsPDE" +example and Apple +Technical Q&A QA1352 for information on writing your own print dialog +plug-ins.

+ +
Note: + +

Starting with Mac OS X 10.5, each plug-in must be compiled "4-way fat" +(32-bit and 64-bit for both PowerPC and Intel) with garbage collection enabled +in order to be usable with all applications.

+ +
+ +

Examples:

+ +
+*% Add two panes for finishing and driver options
+*APDialogExtension: "/Library/Printers/vendor/finishing.plugin"
+*APDialogExtension: "/Library/Printers/vendor/options.plugin"
+
+ +

Mac OS X 10.4APDuplexRequiresFlippedMargin

+ +

*APDuplexRequiresFlippedMargin: boolean

+ +

This boolean keyword notifies the RIP filters that the +destination printer requires the top and bottom margins of the +ImageableArea to be swapped for the back page. The +default is true when cupsBackSide is Flipped +and false otherwise. Table 2 shows how +APDuplexRequiresFlippedMargin interacts with cupsBackSide +and the Tumble page attribute.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 2: Margin Flipping Modes
APDuplexRequiresFlippedMargincupsBackSideTumble ValueMargins
falseanyanyNormal
anyNormalanyNormal
trueManualDuplexfalseNormal
trueManualDuplextrueFlipped
trueRotatedfalseFlipped
trueRotatedtrueNormal
true or unspecifiedFlippedanyFlipped
+ +

Example:

+ +
+*% Rotate the back side images
+*cupsBackSide: Rotated
+
+*% Don't swap the top and bottom margins for the back side
+*APDuplexRequiresFlippedMargin: false
+
+ +

Also see the related cupsBackSide +keyword.

+ +

APHelpBook

+ +

*APHelpBook: "bundle URL"

+ +

This string keyword specifies the Apple help book bundle to use when +looking up IPP reason codes for this printer driver. The +cupsIPPReason keyword maps +"help" URIs to this file.

+ +

Example:

+ +
+*APHelpBook: "file:///Library/Printers/vendor/Help.bundle"
+
+ +

Mac OS X 10.6APICADriver

+ +

*APICADriver: boolean

+ +

This keyword specifies whether the device has a matching Image Capture +Architecture (ICA) driver for scanning. The default is False.

+ +

Examples:

+ +
+*APICADriver: True
+*APScanAppBundleID: "com.apple.ImageCaptureApp"
+
+ +

Mac OS X 10.3APPrinterIconPath

+ +

*APPrinterIconPath: "/Library/Printers/vendor/filename.icns"

+ +

This keyword defines the location of a printer icon file to use when +displaying the printer. The file must be in the Apple icon format.

+ +

Examples:

+ +
+*% Apple icon file
+*APPrinterIconPath: "/Library/Printers/vendor/Icons/filename.icns"
+
+ +

Mac OS X 10.4APPrinterLowInkTool

+ +

*APPrinterLowInkTool: "/Library/Printers/vendor/program"

+ +

This keyword defines an program that checks the ink/toner/marker levels +on a printer, returning an XML document with those levels. See the "InkTool" +example and +Apple +Technical Note TN2144 for more information.

+ +

Examples:

+ +
+*% Use a vendor monitoring program
+*APPrinterLowInkTool: "/Library/Printers/vendor/Tools/lowinktool"
+
+ +

Mac OS X 10.5APPrinterPreset

+ +

*APPrinterPreset name/text: "*Option Choice ..."

+ +

This keyword defines presets for multiple options that show up +in the print dialog of applications (such as iPhoto) that set the job +style hint to NSPrintPhotoJobStyleHint. Each preset maps to one or +more pairs of PPD options and choices as well as providing key/value data for +the application. The following standard preset names are currently defined:

+ +
    + +
  • General_with_Paper_Auto-Detect; Normal quality general printing with auto-detected media.
  • + +
  • General_with_Paper_Auto-Detect_-_Draft; Draft quality general printing with auto-detected media.
  • + +
  • General_on_Plain_Paper; Normal quality general printing on plain paper.
  • + +
  • General_on_Plain_Paper_-_Draft; Draft quality general printing on plain paper.
  • + +
  • Photo_with_Paper_Auto-Detect; Normal quality photo printing with auto-detected media.
  • + +
  • Photo_with_Paper_Auto-Detect_-_Fine; High quality photo printing with auto-detected media.
  • + +
  • Photo_on_Plain_Paper; Normal quality photo printing on plain paper.
  • + +
  • Photo_on_Plain_Paper_-_Fine; High quality photo printing on plain paper.
  • + +
  • Photo_on_Photo_Paper; Normal quality photo printing on glossy photo paper.
  • + +
  • Photo_on_Photo_Paper_-_Fine; High quality photo printing on glossy photo paper.
  • + +
  • Photo_on_Matte_Paper; Normal quality photo printing on matte paper.
  • + +
  • Photo_on_Matte_Paper_-_Fine; High quality photo printing on matte paper.
  • + +
+ +

The value string consists of pairs of keywords, either an option name and +choice (*MainKeyword OptionKeyword) or a preset identifier and value +(com.apple.print.preset.foo value). The following preset identifiers are currently used:

+ +
    + +
  • com.apple.print.preset.graphicsType; specifies the type of printing used for this printing - "General" for general purpose printing and "Photo" for photo printing.
  • + +
  • com.apple.print.preset.media-front-coating; specifies the media type selected by this preset - "none" (plain paper), "glossy", "high-gloss", "semi-gloss", "satin", "matte", and "autodetect".
  • + +
  • com.apple.print.preset.output-mode; specifies the output mode for this preset - "color" (default for color printers) or "monochrome" (grayscale, default for B&W printers).
  • + +
  • com.apple.print.preset.quality; specifies the overall print quality selected by this preset - "low" (draft), "mid" (normal), or "high".
  • + +
+ +

Presets, like options, can also be localized in multiple languages.

+ +

Examples:

+ +
+*APPrinterPreset Photo_on_Photo_Paper/Photo on Photo Paper: "
+  *MediaType Glossy
+  *ColorModel RGB
+  *Resolution 300dpi
+  com.apple.print.preset.graphicsType Photo
+  com.apple.print.preset.quality mid
+  com.apple.print.preset.media-front-coating glossy"
+*End
+*fr.APPrinterPreset Photo_on_Photo_Paper/Photo sur papier photographique: ""
+
+ +

Mac OS X 10.3APPrinterUtilityPath

+ +

*APPrinterPrinterUtilityPath: "/Library/Printers/vendor/filename.app"

+ +

This keyword defines a GUI application that can be used to do printer +maintenance functions such as cleaning the print head(s). See ... for more +information.

+ +

Examples:

+ +
+*% Define the printer utility application
+*APPrinterPrinterUtilityPath: "/Library/Printers/vendor/Tools/utility.app"
+
+ +

Mac OS X 10.6APScannerOnly

+ +

*APScannerOnly: boolean

+ +

This keyword specifies whether the device has scanning but no printing +capabilities. The default is False.

+ +

Examples:

+ +
+*APICADriver: True
+*APScannerOnly: True
+
+ +

Mac OS X 10.3APScanAppBundleID

+ +

*APScanAppBundleID: "bundle ID"

+ +

This keyword defines the application to use when scanning pages from +the device.

+ +

Examples:

+ +
+*APICADriver: True
+*APScanAppBundleID: "com.apple.ImageCaptureApp"
+
+ + +

Change History

+ +

Changes in CUPS 1.5

+ +
    + +
  • Changes all instances of PPD attributes to PPD keywords, to be consistent with the parent specification from Adobe.
  • + +
+ + +

Changes in CUPS 1.4.5

+ + + + +

Changes in CUPS 1.4

+ + + + +

Changes in CUPS 1.3.1

+ +
    + +
  • Added missing Mac OS X AP keywords.
  • + +
  • Added section on auto-configuration including the + OIDMainKeyword and ?MainKeyword + keywords.
  • + +
  • Minor reorganization.
  • + +
+ + +

Changes in CUPS 1.3

+ + + +

Changes in CUPS 1.2.8

+ +
    + +
  • Added section on supported PostScript commands for raster + drivers
  • + +
+ +

Changes in CUPS 1.2

+ + + +

Changes in CUPS 1.1

+ + +
+ + diff --git a/doc/help/spec-raster.html b/doc/help/spec-raster.html new file mode 100644 index 0000000000..50fca220ac --- /dev/null +++ b/doc/help/spec-raster.html @@ -0,0 +1,720 @@ + + + + + CUPS Raster Format + + + + +

CUPS Raster Format

+ +

CUPS Raster files are device-dependent raster image files that contain a PostScript page device dictionary and device-dependent raster imagery for each page in the document. These files are used to transfer raster data from the PostScript and image file RIPs to device-dependent filters that convert the raster data to a printable format.

+ +

CUPS 1.0 and 1.1 used version 1 of the raster format. CUPS 1.2 and later use version 2 (compressed) and version 3 (uncompressed) that are a superset of the version 1 raster format. All three versions of CUPS Raster are streamable formats, and applications using the CUPS Imaging API (the cupsRaster* functions) can read all formats without code changes.

+ +

The registered MIME media type for CUPS Raster files is application/vnd.cups-raster.

+ + +

Organization of a CUPS Raster File

+ +

Figure 1, "Raster Organization", shows the general organization of all CUPS Raster files. Each file begins with a 32-bit synchronization word followed by zero or more pages. Each page consists of a header (the PostScript page device dictionary and raster-specific values) followed by the bitmap image for the page.

+ + + +

Each page bitmap is stored as described by the cupsBitsPerColor, cupsBytesPerLine, cupsColorOrder, cupsColorSpace, cupsHeight, and cupsWidth values in the page header. Pixels for the front side of a sheet are always stored left-to-right, top-to-bottom. When doing duplex printing, pixels for the back side of a sheet may be stored differently depending on the value of the cupsBackSide keyword ("Normal", "ManualTumble", "Rotated", or "Flipped") in the PPD file and the Tumble value ("true" or "false") in the page header. Figure 2, "Page Bitmaps", shows the pixel order for each combination.

+ + + + +

Version 1 Raster File Format

+ +

A version 1 raster file begins with a 32-bit synchronization word: 0x52615374 ("RaSt") for big-endian architectures or 0x74536152 ("tSaR") for little-endian architectures. The writer of the raster file will use the native word order, and the reader is responsible for detecting a reversed word order file and swapping bytes as needed. The CUPS Imaging API raster functions perform this function automatically.

+ +

Following the synchronization word are a series of raster pages. Each page starts with a page device dictionary header and is followed immediately by the (uncompressed/raw) raster data for that page.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 1: CUPS Version 1 Raster Page Device Dictionary
BytesTypeDescriptionValues
0-63C StringMediaClassMedia class string
64-127C StringMediaColorMedia color string
128-191C StringMediaTypeMedia type string
192-255C StringOutputTypeOutput type string
256-259Unsigned IntegerAdvanceDistance0 to 232 - 1 points
260-263Unsigned IntegerAdvanceMedia0 = Never advance roll
+ 1 = Advance roll after file
+ 2 = Advance roll after job
+ 3 = Advance roll after set
+ 4 = Advance roll after page
264-267Unsigned IntegerCollate0 = do not collate copies
+ 1 = collate copies
268-271Unsigned IntegerCutMedia0 = Never cut media
+ 1 = Cut roll after file
+ 2 = Cut roll after job
+ 3 = Cut roll after set
+ 4 = Cut roll after page
272-275Unsigned IntegerDuplex0 = Print single-sided
+ 1 = Print double-sided
276-283Unsigned Integers (2)HWResolutionHorizontal and vertical resolution in dots-per-inch.
284-299Unsigned Integers (4)ImagingBoundingBoxFour integers giving the left, bottom, right, and top positions + of the page bounding box in points
300-303Unsigned IntegerInsertSheet0 = Do not insert separator sheets
+ 1 = Insert separator sheets
304-307Unsigned IntegerJog0 = Do no jog pages
+ 1 = Jog pages after file
+ 2 = Jog pages after job
+ 3 = Jog pages after set
308-311Unsigned IntegerLeadingEdge0 = Top edge is first
+ 1 = Right edge is first
+ 2 = Bottom edge is first
+ 3 = Left edge is first
312-319Unsigned Integers (2)MarginsLeft and bottom origin of image in points
320-323Unsigned IntegerManualFeed0 = Do not manually feed media
+ 1 = Manually feed media
324-327Unsigned IntegerMediaPositionInput slot position from 0 to N
328-331Unsigned IntegerMediaWeightMedia weight in grams per meter squared, 0 = printer default
332-335Unsigned IntegerMirrorPrint0 = Do not mirror prints
+ 1 = Mirror prints
336-339Unsigned IntegerNegativePrint0 = Do not invert prints
+ 1 = Invert prints
340-343Unsigned IntegerNumCopies0 to 232 - 1, 0 = printer default
344-347Unsigned IntegerOrientation0 = Do not rotate page
+ 1 = Rotate page counter-clockwise
+ 2 = Turn page upside down
+ 3 = Rotate page clockwise
348-351Unsigned IntegerOutputFaceUp0 = Output face down
+ 1 = Output face up
352-359Unsigned Integers (2)PageSizeWidth and length in points
360-363Unsigned IntegerSeparations0 = Print composite image
+ 1 = Print color separations
364-367Unsigned IntegerTraySwitch0 = Do not change trays if selected tray is empty
+ 1 = Change trays if selected tray is empty
368-371Unsigned IntegerTumble0 = Do not rotate even pages when duplexing
+ 1 = Rotate even pages when duplexing
372-375Unsigned IntegercupsWidthWidth of page image in pixels
376-379Unsigned IntegercupsHeightHeight of page image in pixels
380-383Unsigned IntegercupsMediaTypeDriver-specific 0 to 232 - 1
384-387Unsigned IntegercupsBitsPerColor1, 2, 4, 8 bits for version 1 raster files
+ 1, 2, 4, 8, and 16 bits for version 2/3 raster files
388-391Unsigned IntegercupsBitsPerPixel1 to 32 bits for version 1 raster files
+ 1 to 240 bits for version 2/3 raster files
392-395Unsigned IntegercupsBytesPerLine1 to 232 - 1 bytes
396-399Unsigned IntegercupsColorOrder0 = chunky pixels (CMYK CMYK CMYK)
+ 1 = banded pixels (CCC MMM YYY KKK)
+ 2 = planar pixels (CCC... MMM... YYY... KKK...)
400-403Unsigned IntegercupsColorSpace0 = gray (device, typically sRGB-based)
+ 1 = RGB (device, typically sRGB)
+ 2 = RGBA (device, typically sRGB)
+ 3 = black
+ 4 = CMY
+ 5 = YMC
+ 6 = CMYK
+ 7 = YMCK
+ 8 = KCMY
+ 9 = KCMYcm
+ 10 = GMCK
+ 11 = GMCS
+ 12 = WHITE
+ 13 = GOLD
+ 14 = SILVER
+ 15 = CIE XYZ
+ 16 = CIE Lab
+ 17 = RGBW (sRGB)
+ 18 = sGray (gray using sRGB gamma/white point)
+ 19 = sRGB
+ 20 = AdobeRGB
+ 32 = ICC1 (CIE Lab with hint for 1 color)
+ 33 = ICC2 (CIE Lab with hint for 2 colors)
+ 34 = ICC3 (CIE Lab with hint for 3 colors)
+ 35 = ICC4 (CIE Lab with hint for 4 colors)
+ 36 = ICC5 (CIE Lab with hint for 5 colors)
+ 37 = ICC6 (CIE Lab with hint for 6 colors)
+ 38 = ICC7 (CIE Lab with hint for 7 colors)
+ 39 = ICC8 (CIE Lab with hint for 8 colors)
+ 40 = ICC9 (CIE Lab with hint for 9 colors)
+ 41 = ICCA (CIE Lab with hint for 10 colors)
+ 42 = ICCB (CIE Lab with hint for 11 colors)
+ 43 = ICCC (CIE Lab with hint for 12 colors)
+ 44 = ICCD (CIE Lab with hint for 13 colors)
+ 45 = ICCE (CIE Lab with hint for 14 colors)
+ 46 = ICCF (CIE Lab with hint for 15 colors)
+ 48 = Device1 (DeviceN for 1 color)
+ 49 = Device2 (DeviceN for 2 colors)
+ 50 = Device3 (DeviceN for 3 colors)
+ 51 = Device4 (DeviceN for 4 colors)
+ 52 = Device5 (DeviceN for 5 colors)
+ 53 = Device6 (DeviceN for 6 colors)
+ 54 = Device7 (DeviceN for 7 colors)
+ 55 = Device8 (DeviceN for 8 colors)
+ 56 = Device9 (DeviceN for 9 colors)
+ 57 = DeviceA (DeviceN for 10 colors)
+ 58 = DeviceB (DeviceN for 11 colors)
+ 59 = DeviceC (DeviceN for 12 colors)
+ 60 = DeviceD (DeviceN for 13 colors)
+ 61 = DeviceE (DeviceN for 14 colors)
+ 62 = DeviceF (DeviceN for 15 colors) +
404-407Unsigned IntegercupsCompressionDriver-specific 0 to 232 - 1
408-411Unsigned IntegercupsRowCountDriver-specific 0 to 232 - 1
412-415Unsigned IntegercupsRowFeedDriver-specific 0 to 232 - 1
416-419Unsigned IntegercupsRowStepDriver-specific 0 to 232 - 1
+ + +

Version 2 Raster File Format

+ +

A version 2 raster file begins with a 32-bit synchronization word: 0x52615332 ("RaS2") for big-endian architectures or 0x32536152 ("2SaR") for little-endian architectures. The writer of the raster file will use the native word order, and the reader is responsible for detecting a reversed word order file and swapping bytes as needed. The CUPS Imaging API raster functions perform this function automatically.

+ +

Following the synchronization word are a series of raster pages. Each page starts with a version 2 page device dictionary header and is followed immediately by the compressed raster data for that page.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 2: CUPS Version 2 Raster Page Device Dictionary
BytesTypeDescriptionValues
0-419Version 1 header dataSee Table 1
420-423Unsigned IntegercupsNumColors1 to 15 colors
424-427IEEE Single PrecisioncupsBorderlessScalingFactor0.0 or 1.0 or greater
428-435IEEE Single Precision (2)cupsPageSizeWidth and length in points
436-451IEEE Single Precision (4)cupsImagingBBoxFour floating point numbers giving the left, bottom, right, and top positions of the page bounding box in points
452-515Unsigned Integers (16)cupsInteger16 driver-defined integer values
516-579IEEE Single Precision (16)cupsReal16 driver-defined floating point values
580-1603C Strings (16x64)cupsString16 driver-defined strings
1604-1667C StringcupsMarkerTypeInk/toner type string
1668-1731C StringcupsRenderingIntentColor rendering intent string
1732-1795C StringcupsPageSizeNamePage size name/keyword string from PPD
+ +

Compressed Raster Data Format

+ +

The version 2 raster data is compressed using a PackBits-like algorithm. Lines are grouped into an integral number of color values based upon the cupsColorOrder setting:

+ +
+ + + + + + + + + + + + + + + + + +
Table 3: Color Value Sizes
cupsColorOrderBytes per color value
0 (chunky)(cupsBitsPerPixel + 7) / 8
1 (banded)(cupsBitsPerColor + 7) / 8
2 (planar)(cupsBitsPerColor + 7) / 8
+ +

Each line of raster data begins with a repetition count from 1 to 256 that is encoded using a single byte of "count - 1".

+ +

After the repetition count, whole color values for that line are run-length encoded using a PackBits-like run-length encoding algorithm: 1 to 128 repeated colors are encoded using an initial byte of "count - 1" followed by the color value byte(s) while 2 to 128 non-repeating colors are encoded using an initial byte of "257 - count" followed by the color value bytes.

+ +

For example, the 8x8 24-bit sRGB image shown in Figure 3, "Sample Image", would be encoded as the following 89 octets:

+ +
+%x00 %x00.FF.FF.FF %x02.FF.FF.00 %x03.FF.FF.FF
+%x00 %xFE.FF.FF.00.00.00.FF.FF.FF.00 %x02.FF.FF.FF %x00.00.FF.00 %x00.FF.FF.FF
+%x00 %x01.FF.FF.00 %x02.FF.FF.FF %x02.00.FF.00
+%x00 %x02.FF.FF.00 %x02.FF.FF.FF %x00.00.FF.00 %x00.FF.FF.FF
+%x00 %x00.FF.FF.FF %x02.FF.FF.00 %x03.FF.FF.FF
+%x00 %x07.FF.FF.FF
+%x01 %x07.FF.00.00
+
+ +

The first line (%x00) contains 1 white pixel (%x00.FF.FF.FF), 3 yellow pixels (%x02.FF.FF.00), and 4 white pixels (%x03.FF.FF.FF).

+ +

The second line (%x00) contains a sequence of yellow + blue + yellow pixels (%xFE.FF.FF.00.00.00.FF.FF.FF.00), 3 white pixels (%x02.FF.FF.FF), 1 green pixel (%x00.00.FF.00), and 1 white pixel (%x00.FF.FF.FF).

+ +

The third line (%x00) contains 2 yellow pixels (%x01.FF.FF.00), 3 white pixels (%x02.FF.FF.FF), and 3 green pixels (%x02.00.FF.00)

+ +

The fourth line (%x00) contains 3 yellow pixels (%x02.FF.FF.00), 3 white pixels (%x02.FF.FF.FF), 1 green pixel (%x00.00.FF.00), and 1 white pixel (%x00.FF.FF.FF).

+ +

The fifth line (%x00) contains 1 white pixel (%x00.FF.FF.FF), 3 yellow pixels (%x02.FF.FF.00), and 4 white pixels (%x03.FF.FF.FF).

+ +

The sixth line (%x00) contains 8 white pixels (%x07.FF.FF.FF).

+ +

The seventh and eighth lines (%x01) contain 8 red pixels (%x07.FF.00.00).

+ + + + +

Version 3 Raster File Format

+ +

A version 3 raster file begins with a 32-bit synchronization word: 0x52615333 ("RaS3") for big-endian architectures and 0x33536152 ("3SaR") for little-endian architectures. The writer of the raster file will use the native word order, and the reader is responsible for detecting a reversed word order file and swapping bytes as needed. The CUPS Imaging API raster functions perform this function automatically.

+ +

Following the synchronization word are a series of raster pages. Each page starts with a version 2 page device dictionary header and is followed immediately by the uncompressed/raw raster data for that page.

+ + +

Pixel Value Coding

+ +

The following sections describe the encoding and decoding of the color values in a CUPS Raster file. In general, colors are packed into the minimum number of bytes, with special consideration provided for efficiency of encoding and access. Multi-byte values are stored in the native byte order and automatically swapped as needed when reading them using the CUPS imaging API.

+ +

CUPS_ORDER_CHUNKED

+ +

The chunked order provides the pixel value packed in a single place. Pixel values with 8 or more bits per color are stored as an array of colors in order, e.g. for CUPS_CSPACE_RGB you will see 8/16-bits of red, then blue, then green, then red, green, blue, etc. Pixel values with less than 8 bits per color are packed together as shown in Table 4. Multi-byte pixel values are stored in the native word order, just as for 16-bit color values.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 4: Chunked Color Values
Bits1-color3-color4-color6-color
1W/W/W/W/W/W/W/W0RGB/0RGBCMYK/CMYK00KCMYcm
2WW/WW/WW/WW00RRGGBBCCMMYYKKN/A
4WWWW/WWWW0000RRRRGGGGBBBB
+ (multi-byte)
CCCCMMMMYYYYKKKK
+ (multi-byte)
N/A
+ +

CUPS_ORDER_BANDED

+ +

The banded order provides each color as a separate line of data. Each color plane for a line is written in sequence, e.g. for the CUPS_CSPACE_CMYK color space you would see all of the cyan pixels for a line followed by the magenta, yellow, and black pixels for that line. This is repeated for all of the lines on the page. Color values are packed starting with the most-significant bit (MSB) first.

+ +

CUPS_ORDER_PLANAR

+ +

The planar order provides each color as a separate page of data using a shared page header. Each color plane for a page is written in sequence, e.g. for the CUPS_CSPACE_CMYK color space you would see all of the cyan pixels for a page followed by the magenta, yellow, and black pixels for that page. Color values are packed starting with the most-significant bit (MSB) first. Each line starts on an 8-bit boundary.

+ +

CUPS_CSPACE_RGBW

+ +

This color space provides a dedicated black text channel and uses the sRGB color space definition and white point for the RGB color channels. The white channel is 0 for text (or "true") black, otherwise it must contain the maximum color value: 1 for 1-bit, 3 for 2-bit, 15 for 4-bit, 255 for 8-bit, or 65535 for 16-bit.

+ +

CUPS_CSPACE_KCMYcm

+ +

When cupsBitsPerColor is 1, 6 color planes are provided - black, cyan, magenta, yellow, light cyan, and light magenta. When cupsBitsPerColor is greater than 1, 4 color planes are provided using the CUPS_CSPACE_KCMY color space instead.

+ +

When cupsColorOrder is CUPS_ORDER_CHUNKED, bit 5 corresponds to black and bit 0 corresponds to light magenta. For CUPS_ORDER_BANDED and CUPS_ORDER_PLANAR, each color plane is encoded separately.

+ +

CUPS_CSPACE_CIELab and CUPS_CSPACE_ICCn

+ +

These color spaces map a CIE Lab color value with a D65 white point to either a 8- or 16-bit per color chunked (CUPS_ORDER_CHUNKED) format; the banded (CUPS_ORDER_BANDED) and planar (CUPS_ORDER_PLANAR) color orders are not supported.

+ +

The values are encoded and decoded using the following formulas:

+ +
    + +
  • 8-bit Encoding:
    + L8 = 2.55 * L + 0.5
    + a8 = a + 128.5
    + b8 = b + 128.5
    +  
  • + +
  • 8-bit Decoding:
    + L = L8 / 2.55
    + a = a8 - 128
    + b = b8 - 128
    +  
  • + +
  • 16-bit Encoding:
    + L16 = 655.35 * L + 0.5
    + a16 = 256 * (a + 128) + 0.5
    + b16 = 256 * (b + 128) + 0.5
    +  
  • + +
  • 16-bit Decoding:
    + L = L16 / 655.35
    + a = a16 / 256 - 128
    + b = b16 / 256 - 128
    +  
  • + +
+ +

CUPS_CSPACE_CIEXYZ

+ +

These color spaces map a CIE XYZ color value with a D65 white point to either a 8- or 16-bit per color chunked (CUPS_ORDER_CHUNKED) format; the banded (CUPS_ORDER_BANDED) and planar (CUPS_ORDER_PLANAR) color orders are not supported.

+ +

The values are encoded and decoded using the following formulas:

+ +
    + +
  • 8-bit Encoding:
    + X8 = 231.8181 * X + 0.5
    + Y8 = 231.8181 * Y + 0.5
    + Z8 = 231.8181 * Z + 0.5
    +  
  • + +
  • 8-bit Decoding:
    + X = X8 / 231.8181
    + Y = Y8 / 231.8181
    + Z = Z8 / 231.8181
    +  
  • + +
  • 16-bit Encoding:
    + X16 = 59577.2727 * X + 0.5
    + Y16 = 59577.2727 * Y + 0.5
    + Z16 = 59577.2727 * Z + 0.5
    +  
  • + +
  • 16-bit Decoding:
    + X = X16 / 59577.2727
    + Y = Y16 / 59577.2727
    + Z = Z16 / 59577.2727
    +  
  • + +
+ +

The scaling factor for XYZ values is 1/1.1, or 231.8181 for 8-bit values and 59577.2727 for 16-bit values. This allows for a slight overflow of XYZ values when converting from RGB, improving accuracy.

+ + +

Change History

+ +

Changes in CUPS 1.4.7

+ +
    + +
  • Greatly improved the detail and now include an example of the bitmap compression.
  • +
  • Added all missing cupsColorSpace values and a separate description of CUPS_CSPACE_RGBW.
  • + +
+ + +

Changes in CUPS 1.2.2

+ +
    + +
  • Added version 3 (uncompressed) format.
  • + +
+ + +

Changes in CUPS 1.2.1

+ +
    + +
  • Added new sections on coding pixel values.
  • + +
  • Clarified definitions of color spaces.
  • + +
+ + +

Changes in CUPS 1.2

+ +
    + +
  • Bumped raster version to 2
  • + +
  • Added RGBW color space
  • + +
  • Added 16 bit per color support
  • + +
  • Added cupsNumColors, cupsBorderlessScalingFactor, cupsPageSize, cupsImagingBBox, cupsInteger, cupsReal, cupsString, cupsMarkerType, cupsRenderingIntent, and cupsPageSizeName attributes to the page device dictionary
  • + +
  • Added raster data compression
  • + +
  • Added data type column to device dictionary documentation.
  • + +
+ +

Changes in CUPS 1.1.19

+ +
    + +
  • Added ICC and CIE color spaces.
  • + +
+ + + diff --git a/doc/help/spec-stp.html b/doc/help/spec-stp.html new file mode 100644 index 0000000000..06c099b55f --- /dev/null +++ b/doc/help/spec-stp.html @@ -0,0 +1,133 @@ + + + + + CUPS Software Test Plan + + + + +

CUPS Software Test Plan

+ +

This software test plan provides detailed tests that are used +to evaluate the stability and compliance of CUPS.

+ + +

Test Procedure

+ +

The test software and data files are located in the +test subdirectory of the source distribution. A script +is provided to compile the ipptool program and run +all of the tests that follow, producing a success/fail +report.

+ +

The test target of the top-level makefile can be +used to run this script:

+ +
+make test
+
+ +

or you can run the test script directly:

+ +
+cd test
+./run-stp-tests
+
+ +

A Software Test Report is stored in a HTML file in the +test subdirectory at the conclusion of the test.

+ + +

IPP Compliance Tests

+ +

This section describes the tests used to validate the IPP +standards compliance of the CUPS server.

+ +

Request Tests

+ +

These tests verify that the CUPS scheduler only accepts valid +IPP requests that start with the attributes-charset +and attributes-natural-language attributes and also +contain a printer-uri or job-uri +attribute.

+ +

It also verifies that the CUPS scheduler always responds with +attributes-charset and +attributes-natural-language attributes, using +default values if they are not provided by the client.

+ +

CUPS Printer Operation Tests

+ +

These tests verify that the CUPS printer operations are +supported and function properly. Two printers called +Test1 and Test2 are created, one as a +PostScript printer and one as a raster printer.

+ +

Job Operation Tests

+ +

These test verify that the CUPS scheduler accepts print jobs +for all supported file formats and that the +cancel-job, hold-job, and +resume-job operations work.

+ +

Subscription Operation Tests

+ +

These test verify that the CUPS scheduler accepts +subscriptions with print jobs and that all subscription +operations work as required by the IPP notification and mailto +specifications.

+ + +

Command Tests

+ +

This section describes the tests used to validate the Berkeley +and System V commands included with CUPS.

+ +

lpadmin

+ +

This test verifies that printers can be added, modified, and +defaulted using the lpadmin command.

+ +

lpc

+ +

This test verifies that the lpc command can show +the current status of all print queues.

+ +

lpq

+ +

This test verifies that the lpq command lists +any jobs in the queue.

+ +

lpstat

+ +

This test verifies that the lpstat command works +with all reports using the "-t" option.

+ +

lp

+ +

This test verifies that the lp command works with +both the default destination and a specific destination.

+ +

lpr

+ +

This test verifies that the lpr command works +with both the default destination and a specific destination.

+ +

lprm

+ +

This test verifies that the lprm command can +properly cancel a job.

+ +

cancel

+ +

This test verifies that the cancel command can +properly cancel a job or all jobs.

+ +

lpinfo

+ +

This test verifies that the lpinfo command +returns a list of available printer drivers and devices.

+ + + diff --git a/doc/help/standard.html.in b/doc/help/standard.html.in new file mode 100644 index 0000000000..f957fe249c --- /dev/null +++ b/doc/help/standard.html.in @@ -0,0 +1,181 @@ + + + + Standard Configuration + + + + +

Standard Configuration

+ +

This document describes the standard configuration for this CUPS +server.

+ +
Note: + + +

This file reflects the standard CUPS configuration as distributed +by Apple Inc, the developer of CUPS.

+ +
+ + +

cupsd Configuration

+ +

cupsd(8) is configured by default to show +printers shared by other systems and only allow local access to +the system and its printers. Administration operations require +Basic authentication with membership in the group +"@CUPS_GROUP@".

+ +

Connections are accepted via domain socket +(@CUPS_DEFAULT_DOMAINSOCKET@) or "localhost" +(127.0.0.1).

+ +

Settings

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 1: Default Values
DirectiveValue
Browsing@CUPS_BROWSING@
BrowseLocalProtocols@CUPS_BROWSE_LOCAL_PROTOCOLS@
BrowseRemoteProtocols@CUPS_BROWSE_REMOTE_PROTOCOLS@
BrowseShortNames@CUPS_BROWSE_SHORT_NAMES@
ConfigFilePerm@CUPS_CONFIG_FILE_PERM@
DefaultShared@CUPS_DEFAULT_SHARED@
ImplicitClasses@CUPS_IMPLICIT_CLASSES@
LogFilePerm@CUPS_LOG_FILE_PERM@
MaxCopies@CUPS_MAX_COPIES@
UseNetworkDefault@CUPS_USE_NETWORK_DEFAULT@
+ + +

Directories

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 2: Directories
DirectoryPurpose
@CUPS_SERVERROOT@The location of configuration files such as + printers.conf. Overridden by the ServerRoot + directive in cupsd.conf.
@bindir@The location of user programs.
@includedir@The location of CUPS include files.
@libdir@The location of CUPS library files.
@CUPS_SERVERBIN@The location of server programs such as backends and + filters. Overridden by the ServerBin + directive in cupsd.conf.
@sbindir@The location of administrator programs.
@CUPS_DATADIR@The location of data files such as fonts. Overridden + by the DataDir + directive in cupsd.conf.
@CUPS_DOCROOT@The location of documentation files. Overridden by + the DocumentRoot + directive in cupsd.conf.
@CUPS_LOCALEDIR@The location of localization files.
@CUPS_CACHEDIR@The location of cache files such as + ppds.dat and remote.cache. + Overridden by the CacheDir + directive in cupsd.conf.
@CUPS_LOGDIR@The location of the access_log, + error_log, and page_log files. + Overridden by the AccessLog, + ErrorLog, + PageLog, + directive in cupsd.conf.
@CUPS_STATEDIR@The location of the domain socket file and state data + such as authentication certificates. Overridden by the StateDir + directive in cupsd.conf.
@CUPS_REQUESTS@The location of spooled print jobs. Overridden by the + RequestRoot + directive in cupsd.conf.
+ + + diff --git a/doc/help/translation.html b/doc/help/translation.html new file mode 100644 index 0000000000..a4ada4a70e --- /dev/null +++ b/doc/help/translation.html @@ -0,0 +1,852 @@ + + + + Translating and Customizing CUPS + + + + +

Translating and Customizing CUPS

+ +

Thanks to its extensive use of templates, images, and message +catalogs, CUPS can be easily translated (or customized!) to suit +your needs. This help file will guide you through the CUPS +localization files so you can get the most out of it.

+ + +

Getting Started

+ +

Start by downloading the CUPS source code from www.cups.org. After you extract the files +from the source archive, you will want to copy the following +files and directories:

+ +
    + +
  • desktop/cups.desktop - the GNOME/KDE + desktop file pointing to the CUPS web interface
  • + +
  • doc/index.html - the web interface home + page
  • + +
  • locale/cups.pot - the message catalog
  • + +
  • templates/*.tmpl - the web interface + template files
  • + +
+ +

With the exception of the message catalogs and desktop file, +localization files are placed in subdirectories under the +doc and templates using the locale name. +Locale names are either ll or ll_CC, where +"ll" is the 2-letter language code and "CC" is the 2-letter +country code. CUPS does not currently use or support the newer +ll-region syntax for locale names.

+ +

All non-image files must be encoded using the UTF-8 character +set.

+ + +

Submitting a Translation for CUPS

+ +

To submit a translation for inclusion in CUPS, translate the +desktop file, all of the template files, the index.html.in +file, and the message catalog. Place these files in the correct +subdirectories in the CUPS source code archive and run the following +command to create an archive with your files:

+ +
+tar cvf ll_CC.tar.gz desktop/cups.desktop doc/ll_CC \
+    locale/cups_ll_CC.po templates/ll_CC
+
+ +

Replace "ll_CC" with the locale name for your translation. +Once you have created the archive, go to the CUPS Bugs +& Features page and submit a bug report, attaching the +translation to the report.

+ + +

The Desktop File

+ +

The desktop/cups.desktop file provides a link to +the CUPS web interface from desktop environments such as +GNOME and KDE. To translate this file, add two lines to the +bottom with the Name and Comment keys:

+ +
+Name[ll_CC]=Translation of "Manage Printing"
+Comment[ll_CC]=Translation of "CUPS Web Interface"
+
+ + +

The Home Page

+ +

The index.html file is a complete HTML file that is +displayed when the user visits "http://localhost:631/". Edit the +existing doc/index.html and save it in the +doc/ll_CC subdirectory so that the makefile can +install it. Run "make install" in the doc subdirectory +to test the new home page.

+ + +

Message Catalogs

+ +

CUPS message catalogs are GNU gettext ".po" text files that +provide a list of localized message strings for the CUPS +software. Message catalogs are named cups_ll.po or +cups_ll_CC.po, where "ll" is the standard 2-letter +abbreviation for the language and "CC" is the standard 2-letter +abbreviation for the country.

+ +

When translating a new message catalog, copy the +cups.pot message catalog file in the locale +subdirectory of the CUPS source code. For example, to start +translating the message catalog to Canadian French, you would +type the following commands:

+ +
+cd locale
+cp cups.pot cups_fr_CA.po
+
+ +

Alternatively, you can copy the existing cups_fr.po +message catalog and then make any necessary changes.

+ +

Once you have make your copy of the file, edit it using your +favorite text editor or translation program to translate the text +to the desired language.

+ +

Then validate your translation using the locale/checkpo utility:

+ +
+cd locale
+./checkpo cups_ll_CC.po
+
+ +

After fixing any errors in your translation, add your locale to the LANGUAGES +variable in the Makedefs file and run the "make +install" command in the locale subdirectory to test +the translation.

+ + +

Template Files

+ +

The CUPS scheduler provides a web interface that can be used +to do many common printing and administration tasks. The built-in +web server supports localization of web pages through the use of +subdirectories for each locale, e.g. "fr" for French, "de" for +German, "fr_ca" for French in Canada, and so forth.

+ +

Template files are HTML files with special formatting +characters in them that allow substitution of variables and arrays. +The CUPS CGI programs (admin.cgi, +classes.cgi, help.cgi, +jobs.cgi, and printers.cgi) use these +template file to provide dynamic content for the web interface. +Template files are installed in the +/usr/share/cups/templates directory by default. Table +2 lists the various template files and their purpose.

+ +

Translated versions of the template files should be saved in +the templates/ll_CC subdirectory. For example, +Canadian French template files should be saved in the +templates/fr_CA subdirectory. After you have +translated all of the templates, add the locale to the +LANGUAGES variable in the +Makedefs file and run "make install" in the +templates subdirectory to test the translation.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 2: Web Interface Template Files
FilenamePurpose
add-class.tmplThis is the initial form that is shown to add a new + printer class.
add-printer.tmplThis is the initial form that is shown to add a new + printer.
add-rss-subscription.tmplThis is the form that is shown to add a new RSS subscription.
admin.tmplThis is the main administration page.
choose-device.tmplThis is the form that shows the list of available + devices.
choose-make.tmplThis is the form that shows the list of available + manufacturers.
choose-model.tmplThis is the form that shows the list of available + printer models/drivers.
choose-serial.tmplThis is the form that allows the user to choose + a serial port and any options.
choose-uri.tmplThis is the form that allows the user to enter + a device URI for network printers.
class-added.tmplThis template shows the "class added" message.
class-confirm.tmplThis is the template used to confirm the + deletion of a class.
class-deleted.tmplThis template shows the "class deleted" message.
class-jobs-header.tmplThis template shows the "jobs" header for jobs in a + class.
class-modified.tmplThis template shows the "class modified" message.
classes.tmplThis template shows one or more printer classes.
classes-header.tmplThis template shows the "showing N of M classes" header in + the class list.
edit-config.tmplThis is the cupsd.conf editor page.
error.tmplThis template displays a generic error message.
error-op.tmplThis is the "unknown operation" error page.
header.tmplThis template is used as the standard header on all dynamic + content.
help-header.tmplThis is the top part of the help page.
help-printable.tmplThis is the standard page header for the printable + version of help files.
job-cancel.tmplThis template shows "job canceled".
job-hold.tmplThis template shows "job held".
job-move.tmplThis template shows the move-job form.
job-moved.tmplThis template shows "job moved".
job-release.tmplThis template shows "job released".
job-restart.tmplThis template shows "job reprinted".
jobs.tmplThis template is used to list the print jobs on a server, + class, or printer.
jobs-header.tmplThis template shows the "showing N or M jobs" header in the + jobs list.
list-available-printers.tmplThis template shows a list of new printers that have been found.
maintenance.tmplThis template shows "maintenance commands sent".
modify-class.tmplThis template is used as the first form when modifying a + class.
modify-printer.tmplThis template is used as the first form when modifying a + printer.
norestart.tmplThis template shows "server not restarted because no + changes were made to the configuration".
option-boolean.tmplThis template is used to select a boolean PPD option.
option-conflict.tmplThis template shows the conflicting options.
option-header.tmplThis template is used to start a PPD option group.
option-pickmany.tmplThis template is used to select a multi-valued PPD option.
option-pickone.tmplThis template is used to select a single-valued PPD option.
option-trailer.tmplThis template is used to end a PPD option group.
pager.tmplThis template shows the previous/next pager bar.
printer-accept.tmplThis template shows "printer now accepting jobs".
printer-added.tmplThis template shows "printer added".
printer-configured.tmplThis template shows "printer configured".
printer-confirm.tmplThis template asks the user to confirm the deletion + of a printer.
printer-default.tmplThis template shows "default printer set".
printer-deleted.tmplThis template shows "printer deleted".
printer-jobs-header.tmplThis templates shows the "jobs" header for jobs on a + printer.
printer-modified.tmplThis template shows "printer modified".
printer-purge.tmplThis template shows "printer has been purged of all jobs".
printer-reject.tmplThis template shows "printer now rejecting jobs".
printer-start.tmplThis template shows "printer started".
printer-stop.tmplThis template shows "printer stopped".
printers.tmplThis template is used to list information on one or more + printers.
printers-header.tmplThis template shows the "showing printer N of M" header in + the printers list.
restart.tmplThis template shows "server restarting".
samba-export.tmplThis template shows the export printers to Samba form.
samba-exported.tmplThis template shows "printers exported to Samba".
search.tmplThis template shows the search form.
set-printer-options-header.tmplThis template shows the first part of the set printer options + form.
set-printer-options-trailer.tmplThis template shows the last part of the set printer options + form.
subscription-added.tmplThis template shows "subscription ... added".
subscription-canceled.tmplThis template shows "subscription #NNN canceled".
test-page.tmplThis template shows "test page printed".
trailer.tmplThis template is used as the standard trailer on all dynamic + content.
users.tmplThis template shows the set allowed users form.
+ +

Inserting Attributes and Values

+ +

Template files consist of HTML with variable substitutions for +named inside curly braces "{name}". Variable names are generally +the IPP attribute names with the hyphen ("-") replaced by the +underscore ("_") character. For example, the +job-printer-uri attribute is renamed to +job_printer_uri.

+ +

Curley braces ("{" and "}") to indicate substitutions, and the +backslash ("\") character for quoting. To insert any of these +special characters as-is you need to use the HTML +&name; mechanism or prefix each special +character with the backslash ("\".)

+ +

You substitute the value of a variable using +{NAME} in your template file. If the variable is +undefined then the {NAME} string is output +as-is.

+ +

To substitute an empty string if the variable is undefined, +use {?NAME} instead.

+ +

Array Substitutions

+ +

The number of array elements can be inserted using +{#NAME}. If the array is undefined then 0 is output. +The current array element (starting at 1) is inserted with +{#}.

+ +

Arrays are handled using {[NAME] at the beginning +of a section and } at the end. The information +between the closing bracket ("]") and closing brace ("}") is +repeated for as many elements as are in the named array. For +example, the following template will display a list of each job +in the job_id array:

+ +
+<TABLE>
+<TR>
+	<TH>Job ID</TH>
+	<TH>Destination</TH>
+	<TH>Title</TH>
+</TR>
+
+{[job_id]
+<TR>
+	<TD>{?job_id}</TD>
+	<TD>{?job_printer_name}</TD>
+	<TD>{?job_name}</TD>
+</TR>
+}
+</TABLE>
+
+ +

Arrays can be nested, however all elements within the curly +braces ("{" and "}") are indexed using the innermost array.

+ +

Conditional Tests

+ +

Templates can also test variables against specific values and +conditionally include text in the template. The format is:

+ +
+{variable?true:false}
+{variable=value?true:false}
+{variable!value?true:false}
+{variable<value?true:false}
+{variable>value?true:false}
+
+ +

where true is the text that is included if the +condition is true and false is the text that is +included if the condition is false. A value of # is +replaced with the current element number (starting at 1.) The +character after the variable name specifies the condition to +test. Table 3 shows the available test conditions.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + +
Table 3: Template Substitution Conditions
CharCondition
?True if variable exists.
=True if variable is equal to value.
!True if variable is not equal to value.
<True if variable is less than value.
>True if variable is greater than value.
+ +

CGI Programs

+ +

CUPS uses five CGI programs to manage the dynamic web +interfaces:

+ +
    + +
  • admin.cgi
  • +
  • classes.cgi
  • +
  • help.cgi
  • +
  • jobs.cgi
  • +
  • printers.cgi
  • + +
+ +

Each CGI program accepts standard form variables such as +OP for the operation to perform, +PRINTER_NAME for the printer or class name to +operate on, QUERY for any search words, +FIRST for the first class, job, or printer to +display, and ORDER to control the order that +classes, jobs, or printers are displayed.

+ +

In addition, the classes.cgi, +jobs.cgi, and printers.cgi programs +support a WHICH_JOBS variable to control which jobs +are displayed. Table 4 lists the supported values.

+ +
+ + + + + + + + + + + + + + + + + + + +
Table 4: WHICH_JOBS Values
WHICH_JOBS ValueDescription
allShow all jobs
completedShow completed jobs
not-completedShow active jobs
+ +

admin.cgi

+ +

The admin.cgi program handles all of the printer +and class administration functions and is run for all direct +accesses to the /admin resource. For most operations +it uses the PRINTER_NAME and OP form +variables to specify the action requested. Table 5 shows the +supported OP values.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 5: admin.cgi OP Values
OP ValueDescription
accept-jobsAccepts jobs on the named destination.
add-classAdds a new printer class.
add-printerAdds a new printer.
config-serverConfigures the server.
delete-classDeletes a printer class. The form variable CONFIRM + may be set to any value to bypass the confirmation page.
delete-printerDeletes a printer. The form variable CONFIRM + may be set to any value to bypass the confirmation page.
export-sambaExports printers to Samba.
find-new-printersFind new printers that have not yet been added.
modify-classModifies a printer class.
modify-printerModifies a printer.
purge-jobsPurges all jobs on the named destination.
redirectRedirects the web browser to the location referenced by + the URL form variable.
reject-jobsRejects new jobs on the named destination.
set-allowed-usersSets the allowed users for a destination.
set-as-defaultSets the default destination.
set-printer-optionsSets the default options for a printer.
set-sharingSets the printer-is-shared attribute for a destination.
start-classStarts the named class.
start-printerStarts the named printer.
stop-classStops the named class.
stop-printerStops the named printer.
+ +

classes.cgi

+ +

The classes.cgi program is responsible for +listing class information, including jobs destined for that +class. It is for all direct accesses to the /classes +resource and supports the optional form variables OP +and WHICH_JOBS. If no form variables are supplied +then the CGI lists all or a specific class and the active jobs on +each class. Table 6 shows the supported OP +values.

+ +
+ + + + + + + + + + + + + + + + + +
Table 6: classes.cgi OP Values
OP ValueDescription
move-jobsMoves the jobs on this class to another destination.
print-test-pagePrints the standard PostScript test page.
+ + +

help.cgi

+ +

The help.cgi program handles all of the on-line +help functions and is run for all direct accesses to the +/help resource.

+ + +

jobs.cgi

+ +

The jobs.cgi program handles all of the job +functions and is run for all direct accesses to the +/jobs resource. For most operations it uses the +JOB_ID, OP, and WHICH_JOBS +form variables to specify the action requested. Table 7 shows the +supported OP values.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 7: jobs.cgi OP Values
OP ValueDescription
cancel-jobCancels a job.
hold-jobHolds a job indefinitely.
move-jobMoves a job to another destination.
release-jobReleases a job for printing.
restart-jobRestarts/reprints a stopped, canceled, completed, or aborted + print job.
+ + +

printers.cgi

+ +

The printers.cgi program is responsible for +listing printer information, including jobs destined for that +printer. It is for all direct accesses to the +/printers resource and supports the optional form +variables OP and WHICH_JOBS. If no form +variables are supplied then the CGI lists all printers or a +specific printer and the active jobs on that printer. Table 8 +shows the supported OP values.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + +
Table 8: printers.cgi OP Values
OP ValueDescription
clean-print-headsClean the print heads.
move-jobsMove all jobs to a different destination.
print-self-test-pagePrint a printer self-test page.
print-test-pagePrint a PostScript test page.
+ + + diff --git a/doc/help/whatsnew.html b/doc/help/whatsnew.html new file mode 100644 index 0000000000..856f30a94f --- /dev/null +++ b/doc/help/whatsnew.html @@ -0,0 +1,61 @@ + + + + What's New in CUPS 1.5 + + + + +

What's New in CUPS 1.5

+ +

CUPS 1.5 many changes and new features to CUPS 1.4.x. This page provides a high-level outline of these changes. If you have never used CUPS before, read the "Overview of CUPS" document instead.

+ +

Commands

+ +
    + +
  1. User defaults; Starting in CUPS 1.2, user defaults were stored in the ~/.cups directory, and the old ~/.cupsrc and ~/.lpoptions files were deprecated. Support for ~/.cupsrc and ~/.lpoptions have been removed in CUPS 1.5.
  2. + +
  3. IPP test tool; The ipptool IPP test tool is now a first-class user program.
  4. + +
  5. lpadmin; The lpadmin command now supports setting the cupsIPPSupplies and cupsSNMPSupplies keywords in PPDs and deleting options.
  6. +
+ +

Scheduler

+ +
    + +
  1. Filter security; Filters and backends must now have group write permissions disabled.
  2. + +
+ + +

Print Filters

+ +
    + +
  1. HP-GL/2 filter; The HP-GL/2 filter is no longer provided with CUPS.
  2. + +
  3. PDF filter; The PDF filter now supports new Poppler and Ghostscript options.
  4. + +
  5. PWG Raster driver; Added a new PWG Raster printer driver filter.
  6. + +
  7. SCSI backend; The SCSI backend is no longer provided with CUPS.
  8. + +
+ + +

CUPS API

+ +
    + +
  1. CUPS headers; Changes to the main CUPS header may require code changes to applications and toolkits. Users of PPD functions must now include the <cups/ppd.h> header file explicitly since the <cups/cups.h> header no longer does so.
  2. + +
  3. HTTP support: Applications can now set a per-connection timeout interval and callback for HTTP operations as well as query and set SSL/TLS X.509 certificates.
  4. + +
  5. Raster support; The CUPS raster functions now support the creation of PWG Raster files and streams, and a new callback-based context function allows reading and writing from objects other than file descriptors.
  6. + +
+ + + diff --git a/doc/images/color-wheel.png b/doc/images/color-wheel.png new file mode 100644 index 0000000000000000000000000000000000000000..301e3fd59adceacf495738adc24618e7f8a20b51 GIT binary patch literal 13148 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 z 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 + + + + + + + + + + + + + + + + + + + + + + + + + + + + image/svg+xml + + + + + + + + + + + + + + + + + Scheduler (cupsd) + + + + + Filter + PPD Filter + Backend + PortMonitor + + + + + + + JobFiles + Web Interface(CGI) + + + + + BerkeleyCommands + CUPSCommands + System VCommands + + + + + + + ConfigFiles + + + + + + + LogFiles + + Notifiers + EmailRSS + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + LPD Support(cups-lpd) + + + + Printer + + + + + + + + + + + + diff --git a/doc/images/cups-command-chain.png b/doc/images/cups-command-chain.png new file mode 100644 index 0000000000000000000000000000000000000000..a7218e5ce45e57783db68fb655cdd0de11d91749 GIT binary patch literal 14902 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 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + image/svg+xml + + + + + + + + + + + + + + + + + + PPD + + + + OptionalCommandFilter + + + + + + + + + CommandFile + + + + + + Printer + + + + + OptionalPortMonitor + + + + + Backend + + + diff --git a/doc/images/cups-icon.png b/doc/images/cups-icon.png new file mode 100644 index 0000000000000000000000000000000000000000..5c0f6ff20b6a79d23874eb72563f4fdc32b6ae1e GIT binary patch literal 4888 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 B#y9`~ literal 0 Hc-jL100001 diff --git a/doc/images/cups-postscript-chain.svg b/doc/images/cups-postscript-chain.svg new file mode 100644 index 0000000000..d1e2d3e844 --- /dev/null +++ b/doc/images/cups-postscript-chain.svg @@ -0,0 +1,531 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + image/svg+xml + + + + + + + + + + + + + + + + + + + PPD + + + + + Printer + + + + + OptionalPortMonitor + + + + + Backend + + + + OptionalPostScriptFilter + + OptionalPostScriptFilter + + + + + + + + + + + + PrintFile + + + + + + + CUPSFilters + + + CUPSFilters + + + + + diff --git a/doc/images/cups-raster-chain.png b/doc/images/cups-raster-chain.png new file mode 100644 index 0000000000000000000000000000000000000000..5349bd9f573c0ffd8e630deb29a1c24f593da68c GIT binary patch literal 16916 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 literal 0 Hc-jL100001 diff --git a/doc/images/cups-raster-chain.svg b/doc/images/cups-raster-chain.svg new file mode 100644 index 0000000000..5130c81914 --- /dev/null +++ b/doc/images/cups-raster-chain.svg @@ -0,0 +1,534 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + image/svg+xml + + + + + + + + + + + + + + + + + + + PPD + + + + + Printer + + + + + OptionalPortMonitor + + + + + Backend + + + + OptionalPostScriptFilter + + + RequiredRasterFilter + + + + + + + + + + + + + PrintFile + + + + + + + CUPSFilters + + + CUPSFilters + + + + + diff --git a/doc/images/cups.png b/doc/images/cups.png new file mode 100644 index 0000000000000000000000000000000000000000..5c0f6ff20b6a79d23874eb72563f4fdc32b6ae1e GIT binary patch literal 4888 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 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + image/svg+xml + + + CUPS Icon + + + Michael Sweet + + + + + Apple Inc. + + + + + + + + + + + + + + + + + + + + diff --git a/doc/images/generic.png b/doc/images/generic.png new file mode 100644 index 0000000000000000000000000000000000000000..2abe9de9ec6a3f00e2ebb2649aa3e3b0ade4037c GIT binary patch literal 16913 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 literal 0 Hc-jL100001 diff --git a/doc/images/left.gif b/doc/images/left.gif new file mode 100644 index 0000000000000000000000000000000000000000..e82004257b70460672d89924f77a284f09ea1e28 GIT binary patch literal 1492 zc-r05{Z~?V90%~rK?|{|88vO%pyABwY}VY4Eo)|`WY&PJtm8T=F%jiBm1A^7g)5K( zW)P*>>ImUPe8i_rt>i%_rlyISc{D*6_T*z)_Cwpf_+RYx{(Rr>`w!gkqhW#j;(`G@ zU~32B;^N}t<0%vhl}b%WNTAVZiHV7HIz1^ViNRo`q@<*#rlzH(rKhJenaqrgjEfg9 zUb=KCGc%LTW@lw(WoKvSy?uRs zPo6yK@9!TN7#JKJG((2WLqkKu!^0yZBhMhC&qhZ_pF_r;kByCuPmE7YOuT?hzL=bx zv{)=wt95E>YI=J5AIMAS*4S)t!DEBX%-G;SYxn9ET<}6(!w>cbzBl%=(edydXFwjPM#7-aH_EDF?s7rxWgcb_KuLz z)&W2$<_jS*QBX#_`xp0c`3ZAnbOYVzXWMY~x&3-!bzXUtADKdOG(}UQFx!vidHIpM zurje9F)Jd;%@rShrP-wbe9tRAcqTD6SQ|D^4&r>mW&|^*GooJT5&DQ~Lrx?)_ZaR1vgo+-Bq zcQ?;iMgGPe6p@YYZ0Cn(s?FE+M(7l!tWa9e7EI^N}7P6cykmk)B{% zxK7_T6!F540xh#AdVf!at*DGn>bn$KAYw{eH;k&6@OBG=T<^U~5XL ccyVV59=z-oFp|8~8)&Oq>I-_|RqO>aW*Ay*Y#!gJo8an2+e!3wE1DqS&sMam1Md^U z3|9kM1=wS6 zkyy#fmdl_G;@BBWHOg7|1)vk_kz)X}mdD)Bd5*epyu$z=k(;n9Z2byoW9@q%9Q$Nu z0cWe3|9leD0*SU|;Y(88ySM4yea*XDwl|PF>TySS0AQN5e0FN_#gFHP-X1&m*6=gD zKkNc{mfSrs@aW6_cRC*FpMBlw#fQOzCIJRXUC*wL2X^msJlu6pTcv;pDX#(?A-2=A zCxfq@99f(>JBOE#90l-@T31)|{ZDt^6*$t@%^uJ~@(%$lAXX7=^4s}3^t6^J{Di#a zw_bruyhl3=+9B-7_W+#14_HX6KoV9}mS|+U~ zRfk%LNmRDWT7)L^IeShM)!aWEv)PPlS2dc==$KA3EYh@)c3_xFr+JtXQVs-lKju_C z(Y}U7nh+AKHIdNMiFll+<3gNmQ3S;yYgCCT?E5sP#5hJ{I!2;V#3GFd5!M!tgn2p~ zj_4?}xPYyB}WFleaCC)>{xWdy0 zAu&bagU5sz4-FooLe#-&(dUnZOK9SuNag*QlVpk4WJ%&%C7ll{ZQ?^@UE-B%oR66J z=kH7QB`pbOk<>>|l#ojU z#b6*(!Y5D+QRtlf2N{M&iZm>Q9h<0-WfKK!Oh}XybY0{MK`N+NP8tRkGt~tC1zFMU I?_CH00G^a!ApigX literal 0 Hc-jL100001 diff --git a/doc/images/raster-organization.png b/doc/images/raster-organization.png new file mode 100644 index 0000000000000000000000000000000000000000..c390f4144deee31bf5abb3c7f9a1c47a06ba4091 GIT binary patch literal 20974 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 A<^TWy literal 0 Hc-jL100001 diff --git a/doc/images/raster-organization.svg b/doc/images/raster-organization.svg new file mode 100644 index 0000000000..442032f2e9 --- /dev/null +++ b/doc/images/raster-organization.svg @@ -0,0 +1,189 @@ + + + + + + + + + + + + image/svg+xml + + + + + + + + Synchronization Word + + + Page Header 1 Page Bitmap 1 + + + + Page Header N Page Bitmap N + + + + + + diff --git a/doc/images/raster.png b/doc/images/raster.png new file mode 100644 index 0000000000000000000000000000000000000000..17ba3b230885d209f95cf1b36a6e03e296af8fd8 GIT binary patch literal 37656 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 literal 0 Hc-jL100001 diff --git a/doc/images/raster.svg b/doc/images/raster.svg new file mode 100644 index 0000000000..58277e70c6 --- /dev/null +++ b/doc/images/raster.svg @@ -0,0 +1,386 @@ + + + + + + + + + image/svg+xml + + + + + + + + BackSide + + + BackSide + + + BackSide + + + BackSide + + + BackSide + + + BackSide + + + BackSide + + + BackSide + Normalfalse + Normaltrue + ManualTumblefalse + ManualTumbletrue + Rotatedfalse + Rotatedtrue + Flippedfalse + Flippedtrue + + diff --git a/doc/images/right.gif b/doc/images/right.gif new file mode 100644 index 0000000000000000000000000000000000000000..9ebe464332e62bea0ed1ac92853bcbf479a094db GIT binary patch literal 341 zc-mFZzb^w(0KoBUl_D5uqC)%@P0*mIzAh}9P7;GfB4L%7G!ltFKs9 zNkk;3s&G=hd++Yv&wKCAS*X!x{BGY|erZ0v5z&UVv#bUoa$O;%q?8(_p`_xRyERgp z&?J-?hM|;XjMkEa9n&;}@4$rc?fnf|2oq9zDODyI$Fh(R zoHC|==$_{b&IzT}YPI}YmadC2K@jP>E`(5?60UHi>-#<-1Y>O5HbRIoW?7cwI7%r9 z;m?2LTFu|fLc`u|=fm0kR5LhP$!%FX2la=sL$rQzv|T)hiOb{U>Hg>Abu3)8<1OHM zp?Fs+T-~&3PmALnksdJHJQ?qfHYJm*nfeR}w}nA86ao_iBl)>5&|B9Eri*K-RAF}} ZJp=mFo3mrf%k7<`Lx~9J8*E(EvcD`S-G~4H literal 0 Hc-jL100001 diff --git a/doc/images/sample-image.png b/doc/images/sample-image.png new file mode 100644 index 0000000000000000000000000000000000000000..f78760e071d5b75a28ca417ba47fea8d91671d25 GIT binary patch literal 3541 zc-rNcYfw{X8po4ENC_9irYe+H6I>&>ZV9?BV7VkNs9apkf?CU@B1|UT7Cl5kqi;fC5b|2;~wk0TKi;fgt2~PR`jAr?VfNo$2iCbY|WC z@P0VwopgMLV&0UO6L8qXijK>@G zbRX3aNSRRPC5G3E6OFCSPPB$AE}LBToM9AgXi(IUYNn~~Z&qDg>GLape=kCR|0L6q zlis?8b0ab?piD(<@pGnRkUypq{|0^M{bU0}Ogm*GwfF&9K#jpAQuJiI5;@Al^_>)|8ewz{%p_Cen;({38eBQQrU90XSpJw z5a!(WBzZe;^{!6(gHU*Zva)Io@#EJD)8!6#$2xq)Wbgd1d>gLUxIQ@W`%Q-gjFuf` z8+V+P-38Y>ICRiYl|``}G{7n{8NKAR@z>#GGEqv-mmVQ#YPljq-Q;7OAmziz_l^(< zuh-fa@qposhCp!L85Oa0w}d#W(o~`?7c7nR)g!LFTZ+3R!ROij5sO5mxm2mA;u930DVZZ}QVD-S6rl>gt^Sy(a#u z6KD@y%I<e6Pd-V`DuIhD}3Kvi#&Cy+^G76+C_DoS!`zV*vA?(HtVvO5hCKozNkbLp68J|~ew zndoBDe3=ST^i_>NXXi2VMqA9+zxEX&}*_~WkdwI-S zZZwBdl2DN3G~rmn8Ro5##R1WZo_Acs87u{9-aw=Llo2@W$SkqAcM+C`NE>yFos+5# zZ%i11AxH|8*PPLCPK?d_%+aYp5r2nf=@`~U1xU-{6d>)EZLuRY=Xvy7qjxndSZL-q zEr{Vq8df>BdJdoZueRLFel|z#~5AF{FPROLmEm9S%r|A5zlU%dYLcES`dDRp~>+ zCjv2q{lI=2Z zbY#!MN?3x{iN47;%nwgqtOXl*u^IB5M&0lo)Ip0)sXD~Zd@%%E9JjC^p>28B$NL_d zr8>}(N08+$pOTZ(s7w7uB?LdA=EF}eFVzn8K=HZ)er=y^v-N^;#Ou7eZ{$83q*Jvc z5-)^n!E8&SL4M9sZEuVqAUN%GFn-iSBVEL0Xyx_y_Wumy3y>hZBG)zU;xn=Oz%zs8 zCWBnQY%`j;mbSbr|H9Fj-H{o0WF8mJhilnewg@dy2uirt zH|I(_%BG2{~3&U<5-x4YOBR%5l?hs)}@|Zv4ir&(u_Kq`$E8-h3u?aM z`xW%5i&)@`wkR`#3&RH+vr0ge4L^qD?O;e`aLy{Xkv>}>+sQpBQiete{t8dcJi9G# zZ3-PKkjo7w4cqgnrbzja8VXL^I@oqqiJV5=HTgOd)BlllVQ#!Vx!&t(9sTCL3Pfr> zJr;)S1Li+S{d!*u{qLBGk1j{;qbQf{+De!H&sD|C5ft>bp~1l^uq|wLlP>K=BP)I( zq+4nY*Yf?8IyDl7wB>#Ea$VrC4Pl6V%Jf4Pxz;!b49tpcn=@Im{!XcNJ+6S>w)gU$ zaT8RohX#L`6vOKOi%B6C2i%t~%!&v3=I^9gC@e+B*b7EcNa4$ucl4@9(SQi70B!V; zhBT`D`Z~GV$ih%~qOmszJf_Q{3}ubgR5X>)^%i3wlkvl@`EohUTyXb}>Q(;``q2pA zQqQ8oTds)q19nR{Hh1)if<8dUHrP4X$cWZb-ty&`eCFt_5P&r8w6F^hsU$r^1zOf* zo=+ftF|*NX2_~J_bqVJaft!gz$xk)Lm{B{*<`GNaJ=mnFO^a@(lPxCct48MEj(}IX zE;Kb23IZ=6wIjp4?WNigODTwjW(72135}T^hK?O;h`=@y`da1tt-#*h#O?Uc1|e!& LY{a$S2u}V3{-81j literal 0 Hc-jL100001 diff --git a/doc/images/sel.gif b/doc/images/sel.gif new file mode 100644 index 0000000000000000000000000000000000000000..36b16bf69c82311944f29eb457db67bada16d82c GIT binary patch literal 362 zc-jGK0hRtoNk%v~VITk`0E8L<($dn|+1cRW;O6G$>FMd}>gwz3>+J08?d|RE?(XmJ z@9^;O@$vEU^78ZZ^Yrxe_4W1k_V)Mp_xSku`T6TbqLDK%+1cv%nH)e)YaD4)L7cvRSn+X;Njxq;1A~K z=;`X~=wR;eT@v#1^!4`l^c4F0{Qds_{D24?*yDo1g9sBUT!;{j!-r*1Nt{@UmBot~ IF@^vDJGY4K7XSbN literal 0 Hc-jL100001 diff --git a/doc/images/smiley.jpg b/doc/images/smiley.jpg new file mode 100644 index 0000000000000000000000000000000000000000..0076fae2d11978f27ec632236d0c3ce2de771b2f GIT binary patch literal 14120 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 literal 0 Hc-jL100001 diff --git a/doc/images/unsel.gif b/doc/images/unsel.gif new file mode 100644 index 0000000000000000000000000000000000000000..10477fe55faa5da33d2f7a87dd3f18515139a98a GIT binary patch literal 127 zc-nLKbh9u|RA5kHIK;|u_1e{E&!4|~_3HEI&p&_u{0{><3_t)<T4%}S)%A3T9(7t)Q0Jw0b~O5<>kq2y``70_e|+wB z?{D(DB*Ur_h(N@NAaiqb2!hPa%(&g|si~=?q@?=#dfz55FHbI)4-E}@JRY%F+}GC! z0MP68<>lpb|L29w(HV&w*2fz&6C+pZFhAevkVgQ)kvT-|n~`}2g4ar=D9R7{b^uK< zkynjfgDJtTscYUL(V5F7!`QnCb}#&nlRkG|Y`mBEuq(5ln|)8V(P2N670Unkw#il) zclZWD?VvN}>8v;*S+C2vqi0U=GXz+N1Rl01Zc9yG-OUeo~@y*-Ae%R=TY zemL^v-z_`lEf%h9RY!u6yl#YTqwa~>ae=>fFwkr*mrL5?OTsuQ-gfn#HRU)-9)iPl zV@asZTeIFJh7P~F&}N(kz`OF&H_qLg#9KQ2K{{G2!*I$XcHoqY(48W&ln(& z_(A05AZgefkRnn`Rg-sg63x5pOl4B8JGBx8B>miAX$Ri@lx zt!o#`a>a6&y)muY7Hh1I{pT)UMEfw#5He*}_K`}|xrDx%)v4tiBdr<%6xpJiAi%F2 zx58*aTXy(w7d0Z{@ZEFPljOBy=E+ROem&{9edJKk%tq~FDW9ZOYT#4?OcAavl2tt0 ztlh0LZtLE)&rPTGw7H9nje7kN{eqN=Kfg#a-$h1Von7Xmv@D^SF%N=iJ~QZ2fXF`5 z^i)RNQRe1{{u`W$_iw_nfK!Wqs#*+#?)LvUQuZeE{l=HlB8zkLfBB{@$8|Ph>kGM{ zjC;8zMXRkfKi%xw(ipnewO`+}FLXgb<2>MfwGUvIk&fA*2^FjaoK`J!&a|(Eazm93t*-@o*W=#Yj!$i zr38VOWLUHeQA99GWxIwTk@T{PU3`MBwCqFx!B&XWuCMS`bb_G`OL5S36*z1 z7MpluzthGgD-`L=Z&|Cha{0~2T#qjJOBv_SVcbao@!Fw$r4%}?6_|qRY_Z)s*+G(3 z4{B740tg~|c_|O_2{R#7DisWY5!-har*?tXwZ(-3Vuc{NGL~$UFRZEkOM|M^d!&j9 zl}GW|6jLCIkob*C-^m~SbX`e#2rv1-iLyFL{pkHvU2EAT{kWIQY1lZ3gW2c{=MySN zo5SRS8*ltzl`s^)kie4>td7rTcOqOkXaEd@3W=w}DI#K@hM!mxgH|a2rHry~df?3! z#=f-Udrwb&*{*);{O=?+R|Fp ze`wTcJ*r3NC(u0uy2EtI!MM-&+#L4XWUy^EN8fv~e;Mr^Fl|Qcn?I-v_TcJ~%eVXk zJ>5q}J__*MSTsGU;ERZBW5Y|OfF^li)LM*SNt|Y=NEIN2s2k-8Eh?I?#qrg!H(ZWk zI~LZ|{H3WVI3d4Hji$n?g15a@OjNHqtKgv^W|8gT>Og_}wHdH3FGM$9ZA>sonD`xo zUDw6<*D0A>EdHKp3zF3XIvvJL&bkD~^VqNigw5nZAk2!^Gq?}KT1qV_Br~?^YFVc; z^xwOzWU4#^lsXxuPo+YnT5&o!wadVeE%7tcDt~v&Zqr$HZ%{|`_sb<~sAgN+*{=dc z^xIX|UO);JT literal 0 Hc-jL100001 diff --git a/doc/images/webinterface.png b/doc/images/webinterface.png new file mode 100644 index 0000000000000000000000000000000000000000..feca5d66abd4e70a33567b88f9491a2558b04628 GIT binary patch literal 150003 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 axVBZc@wF9zru|_H|6_Ow_7}mxCHg;iCHN}< literal 0 Hc-jL100001 diff --git a/doc/index.html.in b/doc/index.html.in new file mode 100644 index 0000000000..62bee6bdbe --- /dev/null +++ b/doc/index.html.in @@ -0,0 +1,107 @@ + + + + + Home - CUPS @CUPS_VERSION@@CUPS_REVISION@ + + + + + + + + +
+ + + + + + + + + + + + +
  Home    Administration    Classes    Online Help    Jobs    Printers  
+ + + + + +
+ +

CUPS @CUPS_VERSION@

+ +

CUPS is the standards-based, open source printing system developed by +Apple Inc. for Mac OS® X and +other UNIX®-like operating systems.

+ +
CUPS
+ + + +
+ +

CUPS for Users

+ +

Overview of CUPS

+ +

Command-Line Printing and Options

+ +

What's New in CUPS 1.6

+ +

User Forum

+ +
+ +

CUPS for Administrators

+ +

Adding Printers and Classes

+ +

Managing Operation Policies

+ +

Printer Accounting Basics

+ +

Server Security

+ +

Using Kerberos Authentication

+ +

Using Network Printers

+ +

cupsd.conf Reference

+ +

Find Printer Drivers

+ +
+ +

CUPS for Developers

+ +

Introduction to CUPS Programming

+ +

CUPS API

+ +

Filter and Backend Programming

+ +

HTTP and IPP APIs

+ +

PPD API

+ +

Raster API

+ +

PPD Compiler Driver Information File Reference

+ +

Developer Forum

+ +
+ +
 
CUPS and the CUPS logo are trademarks of +Apple Inc. CUPS is copyright 2007-2011 Apple +Inc. All rights reserved.
+ + diff --git a/doc/robots.txt b/doc/robots.txt new file mode 100644 index 0000000000..45dcdc2210 --- /dev/null +++ b/doc/robots.txt @@ -0,0 +1,31 @@ +# +# "$Id: robots.txt 3494 2003-03-19 15:37:44Z mike $" +# +# This file tells search engines not to index your CUPS server. +# +# Copyright 1993-2003 by Easy Software Products. +# +# These coded instructions, statements, and computer programs are the +# property of Easy Software Products and are protected by Federal +# copyright law. Distribution and use rights are outlined in the file +# "LICENSE.txt" which should have been included with this file. If this +# file is missing or damaged please contact Easy Software Products +# at: +# +# Attn: CUPS Licensing Information +# Easy Software Products +# 44141 Airport View Drive, Suite 204 +# Hollywood, Maryland 20636-3111 USA +# +# Voice: (301) 373-9600 +# EMail: cups-info@cups.org +# WWW: http://www.cups.org +# + +User-agent: * +Disallow: / + +# +# End of "$Id: robots.txt 3494 2003-03-19 15:37:44Z mike $". +# + diff --git a/examples/Makefile b/examples/Makefile new file mode 100644 index 0000000000..da08298b38 --- /dev/null +++ b/examples/Makefile @@ -0,0 +1,128 @@ +# +# "$Id$" +# +# PPD compiler example makefile for CUPS. +# +# Copyright 2007-2011 by Apple Inc. +# Copyright 2002-2005 by Easy Software Products. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +# +# Include standard definitions... +# + +include ../Makedefs + + +# +# Examples... +# + +EXAMPLES = \ + color.drv \ + constraint.drv \ + custom.drv \ + grouping.drv \ + laserjet-basic.drv \ + laserjet-pjl.drv \ + minimum.drv \ + postscript.drv \ + r300-basic.drv \ + r300-colorman.drv \ + r300-remote.drv + + +# +# Make everything... +# + +all: + + +# +# Make library targets... +# + +libs: + + +# +# Make unit tests... +# + +unittests: + + +# +# Clean everything... +# + +clean: + + +# +# Dummy depend... +# + +depend: + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: + $(INSTALL_DIR) $(DATADIR)/examples + for file in $(EXAMPLES); do \ + $(INSTALL_DATA) $$file $(DATADIR)/examples; \ + done + + +# +# Install programs... +# + +install-exec: + + +# +# Install headers... +# + +install-headers: + + +# +# Install libraries... +# + +install-libs: + + +# +# Uninstall files... +# + +uninstall: + for file in $(EXAMPLES); do \ + $(RM) $(DATADIR)/examples/$$file; \ + done + -$(RMDIR) $(DATADIR)/examples + + +# +# End of "$Id$". +# diff --git a/examples/color.drv b/examples/color.drv new file mode 100644 index 0000000000..69984c4b9c --- /dev/null +++ b/examples/color.drv @@ -0,0 +1,44 @@ +// Include standard font and media definitions +#include +#include + +// List the fonts that are supported, in this case all standard +// fonts... +Font * + +// Manufacturer and version +Manufacturer "Foo" +Version 1.0 + +// Each filter provided by the driver... +Filter application/vnd.cups-raster 100 rastertofoo + +// Supported page sizes +*MediaSize Letter +MediaSize A4 + +{ + // Supported resolutions + *Resolution k 8 0 0 0 "600dpi/600 DPI" + + // Specify the model name and filename... + ModelName "FooJet 2000" + PCFileName "foojet2k.ppd" +} + +{ + // Supports color printing + ColorDevice true + + // Supported colorspaces + ColorModel Gray/Grayscale w chunky 0 + *ColorModel RGB/Color rgb chunky 0 + + // Supported resolutions + *Resolution - 8 0 0 0 "300dpi/300 DPI" + Resolution - 8 0 0 0 "600dpi/600 DPI" + + // Specify the model name and filename... + ModelName "FooJet Color" + PCFileName "foojetco.ppd" +} diff --git a/examples/constraint.drv b/examples/constraint.drv new file mode 100644 index 0000000000..6acb7f106d --- /dev/null +++ b/examples/constraint.drv @@ -0,0 +1,48 @@ +// Include standard font and media definitions +#include +#include + +// List the fonts that are supported, in this case all standard +// fonts... +Font * + +// Manufacturer, model name, and version +Manufacturer "Foo" +ModelName "FooJet 2000" +Version 1.0 + +// Each filter provided by the driver... +Filter application/vnd.cups-raster 100 rastertofoo + +// Supported page sizes +*MediaSize Letter +MediaSize A4 + +// Supported resolutions +*Resolution k 8 0 0 0 "600dpi/600 DPI" + +// Installable Option Group +Group "InstallableOptions/Options Installed" + + // Duplexing unit option + Option "Option1/Duplexing Unit" Boolean AnySetup 10 + Choice True/Installed "" + *Choice "False/Not Installed" "" + +// General Option Group +Group General + + // Duplexing option + Option "Duplex/Two-Sided Printing" PickOne AnySetup 10 + *Choice "None/No" "<>setpagedevice" + Choice "DuplexNoTumble/Long Edge Binding" + "<>setpagedevice" + Choice "DuplexTumble/Short Edge Binding" + "<>setpagedevice" + +// Only allow duplexing if the duplexer is installed +UIConstraints "*Duplex *Option1 False" + +// Specify the name of the PPD file we want to generate... +PCFileName "foojet2k.ppd" + diff --git a/examples/custom.drv b/examples/custom.drv new file mode 100644 index 0000000000..1001c4fc19 --- /dev/null +++ b/examples/custom.drv @@ -0,0 +1,41 @@ +// Include standard font and media definitions +#include +#include + +// List the fonts that are supported, in this case all standard +// fonts... +Font * + +// Manufacturer, model name, and version +Manufacturer "Foo" +ModelName "FooJet 2000" +Version 1.0 + +// Each filter provided by the driver... +Filter application/vnd.cups-raster 100 rastertofoo + +// Supported page sizes +*MediaSize Letter +MediaSize A4 + +// Supported resolutions +*Resolution k 8 0 0 0 "600dpi/600 DPI" + +// Option Group +Group "Footasm" + + // Boolean option + Option "fooEnhance/Resolution Enhancement" Boolean AnySetup 10 + *Choice True/Yes "<>setpagedevice" + Choice False/No "<>setpagedevice" + + // Multiple choice option + Option "fooOutputType/Output Quality" PickOne AnySetup 10 + *Choice "Auto/Automatic Selection" "<>setpagedevice" + Choice "Text/Optimize for Text" "<>setpagedevice" + Choice "Graph/Optimize for Graphics" "<>setpagedevice" + Choice "Photo/Optimize for Photos" "<>setpagedevice" + +// Specify the name of the PPD file we want to generate... +PCFileName "foojet2k.ppd" + diff --git a/examples/grouping.drv b/examples/grouping.drv new file mode 100644 index 0000000000..da66d74fc9 --- /dev/null +++ b/examples/grouping.drv @@ -0,0 +1,36 @@ +// Include standard font and media definitions +#include +#include + +// List the fonts that are supported, in this case all standard +// fonts... +Font * + +// Manufacturer and version +Manufacturer "Foo" +Version 1.0 + +// Each filter provided by the driver... +Filter application/vnd.cups-raster 100 rastertofoo + +// Supported page sizes +*MediaSize Letter +MediaSize A4 + +{ + // Supported resolutions + *Resolution k 8 0 0 0 "600dpi/600 DPI" + + // Specify the model name and filename... + ModelName "FooJet 2000" + PCFileName "foojet2k.ppd" +} + +{ + // Supported resolutions + *Resolution k 8 0 0 0 "1200dpi/1200 DPI" + + // Specify the model name and filename... + ModelName "FooJet 2001" + PCFileName "foojt2k1.ppd" +} diff --git a/examples/laserjet-basic.drv b/examples/laserjet-basic.drv new file mode 100644 index 0000000000..6924ed7b89 --- /dev/null +++ b/examples/laserjet-basic.drv @@ -0,0 +1,88 @@ +// Include standard font and media definitions +#include +#include + +// Include HP-PCL driver definitions +#include + +// Specify that this driver uses the HP-PCL driver... +DriverType pcl + +// Specify the driver options via the model number... +ModelNumber ($PCL_PAPER_SIZE $PCL_PJL $PCL_PJL_RESOLUTION) + +// List the fonts that are supported, in this case all standard +// fonts... +Font * + +// Manufacturer and driver version +Manufacturer "HP" +Version 1.0 + +// Supported page sizes and their margins +HWMargins 18 12 18 12 +*MediaSize Letter +MediaSize Legal +MediaSize Executive +MediaSize Monarch +MediaSize Statement +MediaSize FanFoldGermanLegal + +HWMargins 18 12.72 18 12.72 +MediaSize Env10 + +HWMargins 9.72 12 9.72 12 +MediaSize A4 +MediaSize A5 +MediaSize B5 +MediaSize EnvC5 +MediaSize EnvDL +MediaSize EnvISOB5 +MediaSize Postcard +MediaSize DoublePostcard + +// Only black-and-white output with mode 3 compression... +ColorModel Gray k chunky 3 + +// Supported resolutions +Resolution - 1 0 0 0 "300dpi/300 DPI" +*Resolution - 8 0 0 0 "600dpi/600 DPI" + +// Supported input slots +*InputSlot 7 "Auto/Automatic Selection" +InputSlot 2 "Manual/Tray 1 - Manual Feed" +InputSlot 4 "Upper/Tray 1" +InputSlot 1 "Lower/Tray 2" +InputSlot 5 "LargeCapacity/Tray 3" + +// Tray 3 is an option... +Installable "OptionLargeCapacity/Tray 3 Installed" +UIConstraints "*OptionLargeCapacity False *InputSlot LargeCapacity" + +{ + // HP LaserJet 2100 Series + Throughput 10 + ModelName "LaserJet 2100 Series" + PCFileName "hpljt211.ppd" +} + +{ + // LaserJet 2200 and 2300 series have duplexer option... + Duplex normal + Installable "OptionDuplex/Duplexer Installed" + UIConstraints "*OptionDuplex False *Duplex" + + { + // HP LaserJet 2200 Series + Throughput 19 + ModelName "LaserJet 2200 Series" + PCFileName "hpljt221.ppd" + } + + { + // HP LaserJet 2300 Series + Throughput 25 + ModelName "LaserJet 2300 Series" + PCFileName "hpljt231.ppd" + } +} diff --git a/examples/laserjet-pjl.drv b/examples/laserjet-pjl.drv new file mode 100644 index 0000000000..32a0bc4f02 --- /dev/null +++ b/examples/laserjet-pjl.drv @@ -0,0 +1,101 @@ +// Include standard font and media definitions +#include +#include + +// Include HP-PCL driver definitions +#include + +// Specify that this driver uses the HP-PCL driver... +DriverType pcl + +// Specify the driver options via the model number... +ModelNumber ($PCL_PAPER_SIZE $PCL_PJL $PCL_PJL_RESOLUTION) + +// List the fonts that are supported, in this case all standard +// fonts... +Font * + +// Manufacturer and driver version +Manufacturer "HP" +Version 2.0 + +// Supported page sizes and their margins +HWMargins 18 12 18 12 +*MediaSize Letter +MediaSize Legal +MediaSize Executive +MediaSize Monarch +MediaSize Statement +MediaSize FanFoldGermanLegal + +HWMargins 18 12.72 18 12.72 +MediaSize Env10 + +HWMargins 9.72 12 9.72 12 +MediaSize A4 +MediaSize A5 +MediaSize B5 +MediaSize EnvC5 +MediaSize EnvDL +MediaSize EnvISOB5 +MediaSize Postcard +MediaSize DoublePostcard + +// Only black-and-white output with mode 3 compression... +ColorModel Gray k chunky 3 + +// Supported resolutions +Resolution - 1 0 0 0 "300dpi/300 DPI" +*Resolution - 8 0 0 0 "600dpi/600 DPI" + +// Supported input slots +*InputSlot 7 "Auto/Automatic Selection" +InputSlot 2 "Manual/Tray 1 - Manual Feed" +InputSlot 4 "Upper/Tray 1" +InputSlot 1 "Lower/Tray 2" +InputSlot 5 "LargeCapacity/Tray 3" + +// Tray 3 is an option... +Installable "OptionLargeCapacity/Tray 3 Installed" +UIConstraints "*OptionLargeCapacity False *InputSlot LargeCapacity" + +// PJL options +Attribute cupsPJL cupsRET "@PJL SET SMOOTHING=%?False:OFF;%?True:ON;%n" + +Option "cupsRET/Smoothing" Boolean DocumentSetup 10 + Choice "False/Off" "" + *Choice "True/On" "" + +Attribute cupsPJL cupsTonerSave "@PJL SET ECONOMODE=%?False:OFF;%?True:ON;%n" + +Option "cupsTonerSave/Save Toner" Boolean DocumentSetup 10 + *Choice "False/No" "" + Choice "True/Yes" "" + +{ + // HP LaserJet 2100 Series + Throughput 10 + ModelName "LaserJet 2100 Series PJL" + PCFileName "hpljt212.ppd" +} + +{ + // LaserJet 2200 and 2300 series have duplexer option... + Duplex normal + Installable "OptionDuplex/Duplexer Installed" + UIConstraints "*OptionDuplex False *Duplex" + + { + // HP LaserJet 2200 Series + Throughput 19 + ModelName "LaserJet 2200 Series PJL" + PCFileName "hpljt222.ppd" + } + + { + // HP LaserJet 2300 Series + Throughput 25 + ModelName "LaserJet 2300 Series PJL" + PCFileName "hpljt232.ppd" + } +} diff --git a/examples/minimum.drv b/examples/minimum.drv new file mode 100644 index 0000000000..ac6e38db08 --- /dev/null +++ b/examples/minimum.drv @@ -0,0 +1,26 @@ +// Include standard font and media definitions +#include +#include + +// List the fonts that are supported, in this case all standard +// fonts... +Font * + +// Manufacturer, model name, and version +Manufacturer "Foo" +ModelName "FooJet 2000" +Version 1.0 + +// Each filter provided by the driver... +Filter application/vnd.cups-raster 100 rastertofoo + +// Supported page sizes +*MediaSize Letter +MediaSize A4 + +// Supported resolutions +*Resolution k 8 0 0 0 "600dpi/600 DPI" + +// Specify the name of the PPD file we want to generate... +PCFileName "foojet2k.ppd" + diff --git a/examples/postscript.drv b/examples/postscript.drv new file mode 100644 index 0000000000..ebb02f8a09 --- /dev/null +++ b/examples/postscript.drv @@ -0,0 +1,46 @@ +// Include standard font and media definitions +#include +#include + +// Specify this is a PostScript printer driver +DriverType ps + +// List the fonts that are supported, in this case all standard fonts +Font * + +// Manufacturer, model name, and version +Manufacturer "Foo" +ModelName "Foo LaserProofer 2000" +Version 1.0 + +// PostScript printer attributes +Attribute DefaultColorSpace "" Gray +Attribute LandscapeOrientation "" Minus90 +Attribute LanguageLevel "" "3" +Attribute Product "" "(Foo LaserProofer 2000)" +Attribute PSVersion "" "(3010) 0" +Attribute TTRasterizer "" Type42 + +// Supported page sizes +*MediaSize Letter +MediaSize Legal +MediaSize A4 + +// Query command for page size +Attribute "?PageSize" "" " + save + currentpagedevice /PageSize get aload pop + 2 copy gt {exch} if (Unknown) + 23 dict + dup [612 792] (Letter) put + dup [612 1008] (Legal) put + dup [595 842] (A4) put + {exch aload pop 4 index sub abs 5 le exch + 5 index sub abs 5 le and + {exch pop exit} {pop} ifelse + } bind forall = flush pop pop + restore" + +// Specify the name of the PPD file we want to generate +PCFileName "fooproof.ppd" + diff --git a/examples/r300-basic.drv b/examples/r300-basic.drv new file mode 100644 index 0000000000..e203d511d8 --- /dev/null +++ b/examples/r300-basic.drv @@ -0,0 +1,75 @@ +// Include standard font and media definitions +#include +#include + +// Include ESC/P driver definitions +#include + +// Specify that this driver uses the ESC/P driver... +DriverType escp + +// Specify the driver options via the model number... +ModelNumber ($ESCP_ESCK $ESCP_EXT_UNITS $ESCP_EXT_MARGINS $ESCP_USB + $ESCP_PAGE_SIZE $ESCP_RASTER_ESCI) + +// List the fonts that are supported, in this case all standard +// fonts... +Font * + +// Manufacturer and driver version +Manufacturer "Epson" +Version 1.0 + +// Supported page sizes and their margins +HWMargins 8.4 0 8.4 0 +*MediaSize Letter +MediaSize Legal +MediaSize Executive +MediaSize Statement +MediaSize A4 +MediaSize A5 +MediaSize A6 +MediaSize B5 +MediaSize Env10 +MediaSize EnvC5 +MediaSize EnvDL +MediaSize EnvISOB5 +MediaSize Postcard +MediaSize DoublePostcard + +VariablePaperSize Yes +MinSize 1in 4in +MaxSize 8.5in 44in + +// Four color modes are supported... +ColorModel Gray/Grayscale w chunky 1 +ColorModel Black k chunky 1 +*ColorModel RGB/Color rgb chunky 1 +ColorModel CMYK cmyk chunky 1 + +// Supported resolutions +Resolution - 8 90 0 103 "360dpi/360 DPI" +*Resolution - 8 90 0 206 "720dpi/720 DPI" +Resolution - 8 90 0 412 "1440dpi/1440 DPI" + +// Very basic dithering settings +Attribute cupsInkChannels "" 6 +Attribute cupsInkLimit "" 2.0 + +Attribute cupsCyanLtDk "" "0.5 1.0" +Attribute cupsMagentaLtDk "" "0.5 1.0" + +Attribute cupsAllDither 360dpi "0.5 0.75 1.0" +Attribute cupsAllDither 720dpi "0.6 0.9 1.2" +Attribute cupsAllDither 1440dpi "0.9 1.35" + +Attribute cupsESCPDotSize 360dpi 16 +Attribute cupsESCPDotSize 720dpi 17 +Attribute cupsESCPDotSize 1440dpi 18 + +{ + // EPSON Stylus Photo R300 Series + Throughput 1 + ModelName "Stylus Photo R300" + PCFileName "epspr301.ppd" +} diff --git a/examples/r300-colorman.drv b/examples/r300-colorman.drv new file mode 100644 index 0000000000..ddaf58ec70 --- /dev/null +++ b/examples/r300-colorman.drv @@ -0,0 +1,85 @@ +// Include standard font and media definitions +#include +#include + +// Include ESC/P driver definitions +#include + +// Specify that this driver uses the ESC/P driver... +DriverType escp + +// Specify the driver options via the model number... +ModelNumber ($ESCP_ESCK $ESCP_EXT_UNITS $ESCP_EXT_MARGINS $ESCP_USB + $ESCP_PAGE_SIZE $ESCP_RASTER_ESCI $ESCP_REMOTE) + +// List the fonts that are supported, in this case all standard +// fonts... +Font * + +// Manufacturer and driver version +Manufacturer "Epson" +Version 3.0 + +// Supported page sizes and their margins +HWMargins 0 0 0 0 +*MediaSize Letter +MediaSize Legal +MediaSize Executive +MediaSize Statement +MediaSize A4 +MediaSize A5 +MediaSize A6 +MediaSize B5 +MediaSize Env10 +MediaSize EnvC5 +MediaSize EnvDL +MediaSize EnvISOB5 +MediaSize Postcard +MediaSize DoublePostcard + +VariablePaperSize Yes +MinSize 1in 4in +MaxSize 8.5in 44in + +// Borderless printing offset... +Attribute cupsESCPFP "" 0 + +// Four color modes are supported... +ColorModel Gray/Grayscale w chunky 1 +ColorModel Black k chunky 1 +*ColorModel RGB/Color rgb chunky 1 +ColorModel CMYK cmyk chunky 1 + +// Supported resolutions +Resolution - 8 90 0 103 "360dpi/360 DPI" +*Resolution - 8 90 0 206 "720dpi/720 DPI" +Resolution - 8 90 0 412 "1440dpi/1440 DPI" + +// Paper trays... +*InputSlot 0 "Auto/Auto Select" +InputSlot 1 "Manual/Manual Feed" + +Attribute cupsESCPPP 0 "1 255" +Attribute cupsESCPPP 1 "2 1" + +// Very basic dithering settings +Attribute cupsInkChannels "" 6 +Attribute cupsInkLimit "" 3.0 + +Attribute cupsCyanLtDk "" "0.5 1.0" +Attribute cupsMagentaLtDk "" "0.5 1.0" + +Attribute cupsAllDither 360dpi "0.5 0.75 1.0" +Attribute cupsAllDither 720dpi "0.6 0.9 1.2" +Attribute cupsAllDither 1440dpi "0.9 1.35" + +Attribute cupsESCPDotSize 360dpi 16 +Attribute cupsESCPDotSize 720dpi 17 +Attribute cupsESCPDotSize 1440dpi 18 + +{ + // EPSON Stylus Photo R300 Series + Throughput 1 + ModelName "Epson Stylus Photo R300" + PCFileName "epspr303.ppd" +} diff --git a/examples/r300-remote.drv b/examples/r300-remote.drv new file mode 100644 index 0000000000..c3065d00c1 --- /dev/null +++ b/examples/r300-remote.drv @@ -0,0 +1,85 @@ +// Include standard font and media definitions +#include +#include + +// Include ESC/P driver definitions +#include + +// Specify that this driver uses the ESC/P driver... +DriverType escp + +// Specify the driver options via the model number... +ModelNumber ($ESCP_ESCK $ESCP_EXT_UNITS $ESCP_EXT_MARGINS $ESCP_USB + $ESCP_PAGE_SIZE $ESCP_RASTER_ESCI $ESCP_REMOTE) + +// List the fonts that are supported, in this case all standard +// fonts... +Font * + +// Manufacturer and driver version +Manufacturer "Epson" +Version 2.0 + +// Supported page sizes and their margins +HWMargins 0 0 0 0 +*MediaSize Letter +MediaSize Legal +MediaSize Executive +MediaSize Statement +MediaSize A4 +MediaSize A5 +MediaSize A6 +MediaSize B5 +MediaSize Env10 +MediaSize EnvC5 +MediaSize EnvDL +MediaSize EnvISOB5 +MediaSize Postcard +MediaSize DoublePostcard + +VariablePaperSize Yes +MinSize 1in 4in +MaxSize 8.5in 44in + +// Borderless printing offset... +Attribute cupsESCPFP "" -80 + +// Four color modes are supported... +ColorModel Gray/Grayscale w chunky 1 +ColorModel Black k chunky 1 +*ColorModel RGB/Color rgb chunky 1 +ColorModel CMYK cmyk chunky 1 + +// Supported resolutions +Resolution - 8 90 0 103 "360dpi/360 DPI" +*Resolution - 8 90 0 206 "720dpi/720 DPI" +Resolution - 8 90 0 412 "1440dpi/1440 DPI" + +// Paper trays... +*InputSlot 0 "Auto/Auto Select" +InputSlot 1 "Manual/Manual Feed" + +Attribute cupsESCPPP 0 "1 255" +Attribute cupsESCPPP 1 "2 1" + +// Very basic dithering settings +Attribute cupsInkChannels "" 6 +Attribute cupsInkLimit "" 2.0 + +Attribute cupsCyanLtDk "" "0.5 1.0" +Attribute cupsMagentaLtDk "" "0.5 1.0" + +Attribute cupsAllDither 360dpi "0.5 0.75 1.0" +Attribute cupsAllDither 720dpi "0.6 0.9 1.2" +Attribute cupsAllDither 1440dpi "0.9 1.35" + +Attribute cupsESCPDotSize 360dpi 16 +Attribute cupsESCPDotSize 720dpi 17 +Attribute cupsESCPDotSize 1440dpi 18 + +{ + // EPSON Stylus Photo R300 Series + Throughput 1 + ModelName "Epson Stylus Photo R300" + PCFileName "epspr302.ppd" +} diff --git a/filter/Dependencies b/filter/Dependencies new file mode 100644 index 0000000000..23a4c185a3 --- /dev/null +++ b/filter/Dependencies @@ -0,0 +1,59 @@ +error.o: error.c ../cups/raster-private.h ../cups/raster.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/ppd.h \ + ../cups/debug-private.h ../cups/string-private.h ../config.h +interpret.o: interpret.c ../cups/raster-private.h ../cups/raster.h \ + ../cups/cups.h ../cups/file.h ../cups/versioning.h ../cups/ipp.h \ + ../cups/http.h ../cups/array.h ../cups/language.h ../cups/ppd.h \ + ../cups/debug-private.h ../cups/string-private.h ../config.h +raster.o: raster.c ../cups/raster-private.h ../cups/raster.h \ + ../cups/cups.h ../cups/file.h ../cups/versioning.h ../cups/ipp.h \ + ../cups/http.h ../cups/array.h ../cups/language.h ../cups/ppd.h \ + ../cups/debug-private.h ../cups/string-private.h ../config.h +commandtops.o: commandtops.c ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/sidechannel.h +gziptoany.o: gziptoany.c ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h +common.o: common.c common.h ../cups/string-private.h ../config.h \ + ../cups/cups.h ../cups/file.h ../cups/versioning.h ../cups/ipp.h \ + ../cups/http.h ../cups/array.h ../cups/language.h ../cups/ppd.h +pstops.o: pstops.c common.h ../cups/string-private.h ../config.h \ + ../cups/cups.h ../cups/file.h ../cups/versioning.h ../cups/ipp.h \ + ../cups/http.h ../cups/array.h ../cups/language.h ../cups/ppd.h \ + ../cups/language-private.h ../cups/transcode.h +rasterbench.o: rasterbench.c ../config.h ../cups/raster.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/ppd.h +rastertoepson.o: rastertoepson.c ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/ppd.h ../cups/string-private.h ../config.h \ + ../cups/language-private.h ../cups/transcode.h ../cups/raster.h +rastertohp.o: rastertohp.c ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/ppd.h ../cups/string-private.h ../config.h \ + ../cups/language-private.h ../cups/transcode.h ../cups/raster.h +rastertolabel.o: rastertolabel.c ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/ppd.h ../cups/string-private.h ../config.h \ + ../cups/language-private.h ../cups/transcode.h ../cups/raster.h +rastertopwg.o: rastertopwg.c ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/raster.h +testraster.o: testraster.c ../cups/raster-private.h ../cups/raster.h \ + ../cups/cups.h ../cups/file.h ../cups/versioning.h ../cups/ipp.h \ + ../cups/http.h ../cups/array.h ../cups/language.h ../cups/ppd.h \ + ../cups/debug-private.h ../cups/string-private.h ../config.h diff --git a/filter/Makefile b/filter/Makefile new file mode 100644 index 0000000000..3d630a42dd --- /dev/null +++ b/filter/Makefile @@ -0,0 +1,400 @@ +# +# "$Id$" +# +# Filter makefile for CUPS. +# +# Copyright 2007-2012 by Apple Inc. +# Copyright 1997-2006 by Easy Software Products. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# +# This file is subject to the Apple OS-Developed Software exception. +# + +include ../Makedefs + + +FILTERS = \ + commandtops \ + gziptoany \ + pstops \ + rastertoepson \ + rastertohp \ + rastertolabel \ + rastertopwg +LIBTARGETS = \ + $(LIBCUPSIMAGE) \ + libcupsimage.a +UNITTARGETS = \ + rasterbench \ + testraster +TARGETS = \ + $(LIBTARGETS) \ + $(FILTERS) + +IMAGEOBJS = error.o interpret.o raster.o +OBJS = $(IMAGEOBJS) \ + commandtops.o gziptoany.o common.o pstops.o \ + rasterbench.o rastertoepson.o rastertohp.o rastertolabel.o \ + rastertopwg.o testraster.o + + +# +# Make all targets... +# + +all: $(TARGETS) + + +# +# Make library targets... +# + +libs: $(LIBTARGETS) + + +# +# Make unit tests... +# + +unittests: $(UNITTARGETS) + + +# +# Clean all object files... +# + +clean: + $(RM) $(OBJS) $(TARGETS) $(UNITTARGETS) + $(RM) libcupsimage.so libcupsimage.sl libcupsimage.dylib + + +# +# Update dependencies (without system header dependencies...) +# + +depend: + $(CC) -MM $(ALL_CFLAGS) $(OBJS:.o=.c) >Dependencies + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: + + +# +# Install programs... +# + +install-exec: + $(INSTALL_DIR) -m 755 $(SERVERBIN)/filter + for file in $(FILTERS); do \ + $(INSTALL_BIN) $$file $(SERVERBIN)/filter; \ + done + $(RM) $(SERVERBIN)/filter/rastertodymo + $(LN) rastertolabel $(SERVERBIN)/filter/rastertodymo + if test "x$(SYMROOT)" != "x"; then \ + $(INSTALL_DIR) $(SYMROOT); \ + for file in $(FILTERS); do \ + cp $$file $(SYMROOT); \ + done \ + fi + + +# +# Install headers... +# + +install-headers: + + +# +# Install libraries... +# + +install-libs: $(INSTALLSTATIC) + $(INSTALL_DIR) -m 755 $(LIBDIR) + $(INSTALL_LIB) $(LIBCUPSIMAGE) $(LIBDIR) + -if test $(LIBCUPSIMAGE) = "libcupsimage.so.2" -o $(LIBCUPSIMAGE) = "libcupsimage.sl.2"; then \ + $(RM) $(LIBDIR)/`basename $(LIBCUPSIMAGE) .2`; \ + $(LN) $(LIBCUPSIMAGE) $(LIBDIR)/`basename $(LIBCUPSIMAGE) .2`; \ + fi + -if test $(LIBCUPSIMAGE) = "libcupsimage.2.dylib"; then \ + $(RM) $(LIBDIR)/libcupsimage.dylib; \ + $(LN) $(LIBCUPSIMAGE) $(LIBDIR)/libcupsimage.dylib; \ + fi + if test "x$(SYMROOT)" != "x"; then \ + $(INSTALL_DIR) $(SYMROOT); \ + cp $(LIBCUPSIMAGE) $(SYMROOT); \ + fi + +installstatic: + $(INSTALL_DIR) -m 755 $(LIBDIR) + $(INSTALL_LIB) -m 755 libcupsimage.a $(LIBDIR) + $(RANLIB) $(LIBDIR)/libcupsimage.a + $(CHMOD) 555 $(LIBDIR)/libcupsimage.a + + +# +# Uninstall all targets... +# + +uninstall: + for file in $(FILTERS); do \ + $(RM) $(SERVERBIN)/filter/$$file; \ + done + $(RM) $(SERVERBIN)/filter/rastertodymo + -$(RMDIR) $(SERVERBIN)/filter + -$(RMDIR) $(SERVERBIN) + $(RM) $(LIBDIR)/libcupsimage.2.dylib + $(RM) $(LIBDIR)/libcupsimage.a + $(RM) $(LIBDIR)/libcupsimage.dylib + $(RM) $(LIBDIR)/libcupsimage_s.a + $(RM) $(LIBDIR)/libcupsimage.sl + $(RM) $(LIBDIR)/libcupsimage.sl.2 + $(RM) $(LIBDIR)/libcupsimage.so + $(RM) $(LIBDIR)/libcupsimage.so.2 + -$(RMDIR) $(LIBDIR) + + +# +# Automatic API help files... +# + +apihelp: + echo Generating CUPS API help files... + mxmldoc --section "Programming" --title "Raster API" \ + --css ../doc/cups-printable.css \ + --header api-raster.header --intro api-raster.shtml \ + api-raster.xml \ + ../cups/raster.h interpret.c raster.c \ + >../doc/help/api-raster.html + mxmldoc --tokens help/api-raster.html api-raster.xml >../doc/help/api-raster.tokens + $(RM) api-raster.xml + mxmldoc --section "Programming" \ + --title "Developing PostScript Printer Drivers" \ + --css ../doc/cups-printable.css \ + --header postscript-driver.header \ + --intro postscript-driver.shtml \ + >../doc/help/postscript-driver.html + mxmldoc --section "Programming" \ + --title "Introduction to the PPD Compiler" \ + --css ../doc/cups-printable.css \ + --header ppd-compiler.header \ + --intro ppd-compiler.shtml \ + >../doc/help/ppd-compiler.html + mxmldoc --section "Programming" \ + --title "Developing Raster Printer Drivers" \ + --css ../doc/cups-printable.css \ + --header raster-driver.header \ + --intro raster-driver.shtml \ + >../doc/help/raster-driver.html + mxmldoc --section "Specifications" \ + --title "CUPS PPD Extensions" \ + --css ../doc/cups-printable.css \ + --header spec-ppd.header \ + --intro spec-ppd.shtml \ + >../doc/help/spec-ppd.html + +framedhelp: + echo Generating CUPS API help files... + mxmldoc --section "Programming" --title "Raster API" \ + --framed ../cups/api-raster \ + --css ../doc/cups-printable.css \ + --header api-raster.header --intro api-raster.shtml \ + ../cups/raster.h interpret.c raster.c + mxmldoc --section "Programming" \ + --title "Developing PostScript Printer Drivers" \ + --framed ../cups/postscript-driver \ + --css ../doc/cups-printable.css \ + --header postscript-driver.header \ + --intro postscript-driver.shtml + mxmldoc --section "Programming" \ + --title "Introduction to the PPD Compiler" \ + --framed ../cups/ppd-compiler \ + --css ../doc/cups-printable.css \ + --header ppd-compiler.header \ + --intro ppd-compiler.shtml + mxmldoc --section "Programming" \ + --title "Developing Raster Printer Drivers" \ + --framed ../cups/raster-driver \ + --css ../doc/cups-printable.css \ + --header raster-driver.header \ + --intro raster-driver.shtml + mxmldoc --section "Specifications" \ + --title "CUPS PPD Extensions" \ + --framed ../cups/spec-ppd \ + --css ../doc/cups-printable.css \ + --header spec-ppd.header \ + --intro spec-ppd.shtml \ + + +# +# commandtops +# + +commandtops: commandtops.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ commandtops.o $(LIBS) + + +# +# gziptoany +# + +gziptoany: gziptoany.o ../Makedefs ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ gziptoany.o $(LIBZ) $(LIBS) + + +# +# libcupsimage.so.2, libcupsimage.sl.2 +# + +libcupsimage.so.2 libcupsimage.sl.2: $(IMAGEOBJS) + echo Linking $@... + $(DSO) $(ARCHFLAGS) $(DSOFLAGS) -o $@ $(IMAGEOBJS) $(DSOLIBS) \ + -L../cups $(LINKCUPS) + $(RM) `basename $@ .2` + $(LN) $@ `basename $@ .2` + + +# +# libcupsimage.2.dylib +# + +libcupsimage.2.dylib: $(IMAGEOBJS) $(LIBCUPSIMAGEORDER) + echo Linking $@... + $(DSO) $(ARCHFLAGS) $(DSOFLAGS) -o $@ \ + -install_name $(libdir)/$@ \ + -current_version 2.3.0 \ + -compatibility_version 2.0.0 \ + $(IMAGEOBJS) $(DSOLIBS) -L../cups $(LINKCUPS) + $(RM) libcupsimage.dylib + $(LN) $@ libcupsimage.dylib + + +# +# libcupsimage_s.a +# + +libcupsimage_s.a: $(IMAGEOBJS) libcupsimage_s.exp + echo Linking $@... + $(DSO) $(DSOFLAGS) -Wl,-berok,-bexport:libcupsimage_s.exp \ + -o libcupsimage_s.o $(IMAGEOBJS) $(DSOLIBS) + $(RM) $@ + $(AR) $(ARFLAGS) $@ libcupsimage_s.o + + +# +# libcupsimage.la +# + +libcupsimage.la: $(IMAGEOBJS) + echo Linking $@... + $(DSO) $(ARCHFLAGS) $(DSOFLAGS) -o $@ $(IMAGEOBJS:.o=.lo) $(DSOLIBS) \ + -L../cups $(LINKCUPS) \ + -rpath $(LIBDIR) -version-info 2:3 + + +# +# libcupsimage.a +# + +libcupsimage.a: $(IMAGEOBJS) + echo Archiving $@... + $(RM) $@ + $(AR) $(ARFLAGS) $@ $(IMAGEOBJS) + $(RANLIB) $@ + + +# +# pstops +# + +pstops: pstops.o common.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ pstops.o common.o $(LIBS) + + +# +# rastertoepson +# + +rastertoepson: rastertoepson.o ../cups/$(LIBCUPS) $(LIBCUPSIMAGE) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ rastertoepson.o $(LINKCUPSIMAGE) $(IMGLIBS) $(LIBS) + + +# +# rastertohp +# + +rastertohp: rastertohp.o ../cups/$(LIBCUPS) $(LIBCUPSIMAGE) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ rastertohp.o $(LINKCUPSIMAGE) $(IMGLIBS) $(LIBS) + + +# +# rastertolabel +# + +rastertolabel: rastertolabel.o ../cups/$(LIBCUPS) $(LIBCUPSIMAGE) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ rastertolabel.o $(LINKCUPSIMAGE) $(IMGLIBS) $(LIBS) + + +# +# rastertopwg +# + +rastertopwg: rastertopwg.o ../cups/$(LIBCUPS) $(LIBCUPSIMAGE) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ rastertopwg.o $(LINKCUPSIMAGE) $(IMGLIBS) $(LIBS) + + +# +# testraster +# + +testraster: testraster.o ../cups/$(LIBCUPSSTATIC) libcupsimage.a + echo Linking $@... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o $@ testraster.o libcupsimage.a \ + ../cups/$(LIBCUPSSTATIC) $(IMGLIBS) $(DSOLIBS) $(COMMONLIBS) \ + $(SSLLIBS) $(DNSSDLIBS) $(LIBGSSAPI) + echo Running raster API tests... + ./testraster + + +# +# rasterbench +# + +rasterbench: rasterbench.o libcupsimage.a + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ rasterbench.o libcupsimage.a $(LIBS) + + +# +# Dependencies... +# + +include Dependencies + + +# +# End of "$Id$". +# diff --git a/filter/api-raster.header b/filter/api-raster.header new file mode 100644 index 0000000000..201c799d3b --- /dev/null +++ b/filter/api-raster.header @@ -0,0 +1,37 @@ + + +

Raster API

+ +
+ + + + + + + + + + + + + + + + +
Headercups/raster.h
Library-lcupsimage
See AlsoProgramming: Introduction to CUPS Programming
+ Programming: CUPS API
+ Programming: PPD API
+ References: CUPS PPD Specification
diff --git a/filter/api-raster.shtml b/filter/api-raster.shtml new file mode 100644 index 0000000000..a309022eae --- /dev/null +++ b/filter/api-raster.shtml @@ -0,0 +1,160 @@ + + +

Overview

+ +

The CUPS raster API provides a standard interface for reading and writing +CUPS raster streams which are used for printing to raster printers. Because the +raster format is updated from time to time, it is important to use this API to +avoid incompatibilities with newer versions of CUPS.

+ +

Two kinds of CUPS filters use the CUPS raster API - raster image processor +(RIP) filters such as pstoraster and cgpdftoraster +(Mac OS X) that produce CUPS raster files and printer driver filters that +convert CUPS raster files into a format usable by the printer. Printer +driver filters are by far the most common.

+ +

CUPS raster files (application/vnd.cups-raster) consists of +a stream of raster page descriptions produced by one of the RIP filters such as +pstoraster, imagetoraster, or +cgpdftoraster. CUPS raster files are referred to using the +cups_raster_t type and are +opened using the cupsRasterOpen +function. For example, to read raster data from the standard input, open +file descriptor 0:

+ +
+#include <cups/raster.h>>
+
+cups_raster_t *ras = cupsRasterOpen(0, CUPS_RASTER_READ);
+
+ +

Each page of data begins with a page dictionary structure called +cups_page_header2_t. This +structure contains the colorspace, bits per color, media size, media type, +hardware resolution, and so forth used for the page.

+ +
Note: + +

Do not confuse the colorspace in the page header with the PPD + ColorModel keyword. ColorModel refers to the general type of + color used for a device (Gray, RGB, CMYK, DeviceN) and is often used to + select a particular colorspace for the page header along with the associate + color profile. The page header colorspace (cupsColorSpace) describes + both the type and organization of the color data, for example KCMY (black + first) instead of CMYK and RGBA (RGB + alpha) instead of RGB.

+ +
+ +

You read the page header using the +cupsRasterReadHeader2 +function:

+ +
+#include <cups/raster.h>>
+
+cups_raster_t *ras = cupsRasterOpen(0, CUPS_RASTER_READ);
+cups_page_header2_t header;
+
+while (cupsRasterReadHeader2(ras, &header))
+{
+  /* setup this page */
+
+  /* read raster data */
+
+  /* finish this page */
+}
+
+ +

After the page dictionary comes the page data which is a full-resolution, +possibly compressed bitmap representing the page in the printer's output +colorspace. You read uncompressed raster data using the +cupsRasterReadPixels +function. A for loop is normally used to read the page one line +at a time:

+ +
+#include <cups/raster.h>>
+
+cups_raster_t *ras = cupsRasterOpen(0, CUPS_RASTER_READ);
+cups_page_header2_t header;
+int page = 0;
+int y;
+char *buffer;
+
+while (cupsRasterReadHeader2(ras, &header))
+{
+  /* setup this page */
+  page ++;
+  fprintf(stderr, "PAGE: %d %d\n", page, header.NumCopies);
+
+  /* allocate memory for 1 line */
+  buffer = malloc(header.cupsBytesPerLine);
+
+  /* read raster data */
+  for (y = 0; y < header.cupsHeight; y ++)
+  {
+    if (cupsRasterReadPixels(ras, buffer, header.cupsBytesPerLine) == 0)
+      break;
+
+    /* write raster data to printer on stdout */
+  }
+
+  /* finish this page */
+}
+
+ +

When you are done reading the raster data, call the +cupsRasterClose function to free +the memory used to read the raster file:

+ +
+cups_raster_t *ras;
+
+cupsRasterClose(ras);
+
+ + +

Functions by Task

+ +

Opening and Closing Raster Streams

+ + + +

Reading Raster Streams

+ + + +

Writing Raster Streams

+ + diff --git a/filter/commandtops.c b/filter/commandtops.c new file mode 100644 index 0000000000..467d3d0925 --- /dev/null +++ b/filter/commandtops.c @@ -0,0 +1,371 @@ +/* + * "$Id$" + * + * PostScript command filter for CUPS. + * + * Copyright 2008-2011 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * + * Contents: + * + * main() - Process a CUPS command file. + * auto_configure() - Automatically configure the printer using + * PostScript query commands and/or SNMP lookups. + * begin_ps() - Send the standard PostScript prolog. + * end_ps() - Send the standard PostScript trailer. + * print_self_test_page() - Print a self-test page. + * report_levels() - Report supply levels. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include + + +/* + * Local functions... + */ + +static void auto_configure(ppd_file_t *ppd, const char *user); +static void begin_ps(ppd_file_t *ppd, const char *user); +static void end_ps(ppd_file_t *ppd); +static void print_self_test_page(ppd_file_t *ppd, const char *user); +static void report_levels(ppd_file_t *ppd, const char *user); + + +/* + * 'main()' - Process a CUPS command file. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + cups_file_t *fp; /* Command file */ + char line[1024], /* Line from file */ + *value; /* Value on line */ + int linenum; /* Line number in file */ + ppd_file_t *ppd; /* PPD file */ + + + /* + * Check for valid arguments... + */ + + if (argc < 6 || argc > 7) + { + /* + * We don't have the correct number of arguments; write an error message + * and return. + */ + + _cupsLangPrintf(stderr, + _("Usage: %s job-id user title copies options file"), + argv[0]); + return (1); + } + + /* + * Open the PPD file... + */ + + if ((ppd = ppdOpenFile(getenv("PPD"))) == NULL) + { + fputs("ERROR: Unable to open PPD file!\n", stderr); + return (1); + } + + /* + * Open the command file as needed... + */ + + if (argc == 7) + { + if ((fp = cupsFileOpen(argv[6], "r")) == NULL) + { + perror("ERROR: Unable to open command file - "); + return (1); + } + } + else + fp = cupsFileStdin(); + + /* + * Read the commands from the file and send the appropriate commands... + */ + + linenum = 0; + + while (cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + { + /* + * Parse the command... + */ + + if (!_cups_strcasecmp(line, "AutoConfigure")) + auto_configure(ppd, argv[2]); + else if (!_cups_strcasecmp(line, "PrintSelfTestPage")) + print_self_test_page(ppd, argv[2]); + else if (!_cups_strcasecmp(line, "ReportLevels")) + report_levels(ppd, argv[2]); + else + fprintf(stderr, "ERROR: Invalid printer command \"%s\"!\n", line); + } + + return (0); +} + + +/* + * 'auto_configure()' - Automatically configure the printer using PostScript + * query commands and/or SNMP lookups. + */ + +static void +auto_configure(ppd_file_t *ppd, /* I - PPD file */ + const char *user) /* I - Printing user */ +{ + ppd_option_t *option; /* Current option in PPD */ + ppd_attr_t *attr; /* Query command attribute */ + char buffer[1024], /* String buffer */ + *bufptr; /* Pointer into buffer */ + ssize_t bytes; /* Number of bytes read */ + int datalen; /* Side-channel data length */ + + + /* + * See if the backend supports bidirectional I/O... + */ + + datalen = 1; + if (cupsSideChannelDoRequest(CUPS_SC_CMD_GET_BIDI, buffer, &datalen, + 30.0) != CUPS_SC_STATUS_OK || + buffer[0] != CUPS_SC_BIDI_SUPPORTED) + { + fputs("DEBUG: Unable to auto-configure PostScript Printer - no " + "bidirectional I/O available!\n", stderr); + return; + } + + /* + * Put the printer in PostScript mode... + */ + + begin_ps(ppd, user); + fflush(stdout); + + /* + * Wait for the printer to become connected... + */ + + do + { + sleep(1); + datalen = 1; + } + while (cupsSideChannelDoRequest(CUPS_SC_CMD_GET_CONNECTED, buffer, &datalen, + 5.0) == CUPS_SC_STATUS_OK && !buffer[0]); + + /* + * Then loop through every option in the PPD file and ask for the current + * value... + */ + + fputs("DEBUG: Auto-configuring PostScript printer...\n", stderr); + + for (option = ppdFirstOption(ppd); option; option = ppdNextOption(ppd)) + { + /* + * See if we have a query command for this option... + */ + + snprintf(buffer, sizeof(buffer), "?%s", option->keyword); + + if ((attr = ppdFindAttr(ppd, buffer, NULL)) == NULL || !attr->value) + { + fprintf(stderr, "DEBUG: Skipping %s option...\n", option->keyword); + continue; + } + + /* + * Send the query code to the printer... + */ + + fprintf(stderr, "DEBUG: Querying %s...\n", option->keyword); + fputs(attr->value, stdout); + fflush(stdout); + + datalen = 0; + cupsSideChannelDoRequest(CUPS_SC_CMD_DRAIN_OUTPUT, buffer, &datalen, 5.0); + + /* + * Read the response data... + */ + + while ((bytes = cupsBackChannelRead(buffer, sizeof(buffer) - 1, 90.0)) > 0) + { + /* + * Trim whitespace from both ends... + */ + + buffer[bytes] = '\0'; + + for (bufptr = buffer + bytes - 1; bufptr >= buffer; bufptr --) + if (isspace(*bufptr & 255)) + *bufptr = '\0'; + else + break; + + for (bufptr = buffer; isspace(*bufptr & 255); bufptr ++); + + fprintf(stderr, "DEBUG: Got \"%s\" (%d bytes)\n", bufptr, (int)bytes); + + /* + * Skip blank lines... + */ + + if (!*bufptr) + continue; + + /* + * Verify the result is a valid option choice... + */ + + if (!ppdFindChoice(option, bufptr)) + continue; + + /* + * Write out the result and move on to the next option... + */ + + fprintf(stderr, "DEBUG: Default%s=%s\n", option->keyword, bufptr); + fprintf(stderr, "PPD: Default%s=%s\n", option->keyword, bufptr); + break; + } + } + + /* + * Finish the job... + */ + + end_ps(ppd); +} + + +/* + * 'begin_ps()' - Send the standard PostScript prolog. + */ + +static void +begin_ps(ppd_file_t *ppd, /* I - PPD file */ + const char *user) /* I - Username */ +{ + (void)user; + + if (ppd->jcl_begin) + { + fputs(ppd->jcl_begin, stdout); + fputs(ppd->jcl_ps, stdout); + } + + puts("%!"); + puts("userdict dup(\\004)cvn{}put (\\004\\004)cvn{}put\n"); + fflush(stdout); +} + + +/* + * 'end_ps()' - Send the standard PostScript trailer. + */ + +static void +end_ps(ppd_file_t *ppd) /* I - PPD file */ +{ + if (ppd->jcl_end) + fputs(ppd->jcl_end, stdout); + else + putchar(0x04); + + fflush(stdout); +} + + +/* + * 'print_self_test_page()' - Print a self-test page. + */ + +static void +print_self_test_page(ppd_file_t *ppd, /* I - PPD file */ + const char *user) /* I - Printing user */ +{ + /* + * Put the printer in PostScript mode... + */ + + begin_ps(ppd, user); + + /* + * Send a simple file the draws a box around the imageable area and shows + * the product/interpreter information... + */ + + puts("% You are using the wrong driver for your printer!\n" + "0 setgray\n" + "2 setlinewidth\n" + "initclip newpath clippath gsave stroke grestore pathbbox\n" + "exch pop exch pop exch 9 add exch 9 sub moveto\n" + "/Courier findfont 12 scalefont setfont\n" + "0 -12 rmoveto gsave product show grestore\n" + "0 -12 rmoveto gsave version show ( ) show revision 20 string cvs show " + "grestore\n" + "0 -12 rmoveto gsave serialnumber 20 string cvs show grestore\n" + "showpage"); + + /* + * Finish the job... + */ + + end_ps(ppd); +} + + +/* + * 'report_levels()' - Report supply levels. + */ + +static void +report_levels(ppd_file_t *ppd, /* I - PPD file */ + const char *user) /* I - Printing user */ +{ + /* + * Put the printer in PostScript mode... + */ + + begin_ps(ppd, user); + + /* + * Don't bother sending any additional PostScript commands, since we just + * want the backend to have enough time to collect the supply info. + */ + + /* + * Finish the job... + */ + + end_ps(ppd); +} + + +/* + * End of "$Id$". + */ diff --git a/filter/common.c b/filter/common.c new file mode 100644 index 0000000000..b80f4d5b56 --- /dev/null +++ b/filter/common.c @@ -0,0 +1,535 @@ +/* + * "$Id$" + * + * Common filter routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * SetCommonOptions() - Set common filter options for media size, + * etc. + * UpdatePageVars() - Update the page variables for the orientation. + * WriteComment() - Write a DSC comment. + * WriteCommon() - Write common procedures... + * WriteLabelProlog() - Write the prolog with the classification + * and page label. + * WriteLabels() - Write the actual page labels. + */ + +/* + * Include necessary headers... + */ + +#include "common.h" +#include + + +/* + * Globals... + */ + +int Orientation = 0, /* 0 = portrait, 1 = landscape, etc. */ + Duplex = 0, /* Duplexed? */ + LanguageLevel = 1, /* Language level of printer */ + ColorDevice = 1; /* Do color text? */ +float PageLeft = 18.0f, /* Left margin */ + PageRight = 594.0f, /* Right margin */ + PageBottom = 36.0f, /* Bottom margin */ + PageTop = 756.0f, /* Top margin */ + PageWidth = 612.0f, /* Total page width */ + PageLength = 792.0f; /* Total page length */ + + +/* + * 'SetCommonOptions()' - Set common filter options for media size, etc. + */ + +ppd_file_t * /* O - PPD file */ +SetCommonOptions( + int num_options, /* I - Number of options */ + cups_option_t *options, /* I - Options */ + int change_size) /* I - Change page size? */ +{ + ppd_file_t *ppd; /* PPD file */ + ppd_size_t *pagesize; /* Current page size */ + const char *val; /* Option value */ + + +#ifdef LC_TIME + setlocale(LC_TIME, ""); +#endif /* LC_TIME */ + + ppd = ppdOpenFile(getenv("PPD")); + + ppdMarkDefaults(ppd); + cupsMarkOptions(ppd, num_options, options); + + if ((pagesize = ppdPageSize(ppd, NULL)) != NULL) + { + PageWidth = pagesize->width; + PageLength = pagesize->length; + PageTop = pagesize->top; + PageBottom = pagesize->bottom; + PageLeft = pagesize->left; + PageRight = pagesize->right; + + fprintf(stderr, "DEBUG: Page = %.0fx%.0f; %.0f,%.0f to %.0f,%.0f\n", + PageWidth, PageLength, PageLeft, PageBottom, PageRight, PageTop); + } + + if (ppd != NULL) + { + ColorDevice = ppd->color_device; + LanguageLevel = ppd->language_level; + } + + if ((val = cupsGetOption("landscape", num_options, options)) != NULL) + { + if (_cups_strcasecmp(val, "no") != 0 && _cups_strcasecmp(val, "off") != 0 && + _cups_strcasecmp(val, "false") != 0) + { + if (ppd && ppd->landscape > 0) + Orientation = 1; + else + Orientation = 3; + } + } + else if ((val = cupsGetOption("orientation-requested", num_options, options)) != NULL) + { + /* + * Map IPP orientation values to 0 to 3: + * + * 3 = 0 degrees = 0 + * 4 = 90 degrees = 1 + * 5 = -90 degrees = 3 + * 6 = 180 degrees = 2 + */ + + Orientation = atoi(val) - 3; + if (Orientation >= 2) + Orientation ^= 1; + } + + if ((val = cupsGetOption("page-left", num_options, options)) != NULL) + { + switch (Orientation & 3) + { + case 0 : + PageLeft = (float)atof(val); + break; + case 1 : + PageBottom = (float)atof(val); + break; + case 2 : + PageRight = PageWidth - (float)atof(val); + break; + case 3 : + PageTop = PageLength - (float)atof(val); + break; + } + } + + if ((val = cupsGetOption("page-right", num_options, options)) != NULL) + { + switch (Orientation & 3) + { + case 0 : + PageRight = PageWidth - (float)atof(val); + break; + case 1 : + PageTop = PageLength - (float)atof(val); + break; + case 2 : + PageLeft = (float)atof(val); + break; + case 3 : + PageBottom = (float)atof(val); + break; + } + } + + if ((val = cupsGetOption("page-bottom", num_options, options)) != NULL) + { + switch (Orientation & 3) + { + case 0 : + PageBottom = (float)atof(val); + break; + case 1 : + PageLeft = (float)atof(val); + break; + case 2 : + PageTop = PageLength - (float)atof(val); + break; + case 3 : + PageRight = PageWidth - (float)atof(val); + break; + } + } + + if ((val = cupsGetOption("page-top", num_options, options)) != NULL) + { + switch (Orientation & 3) + { + case 0 : + PageTop = PageLength - (float)atof(val); + break; + case 1 : + PageRight = PageWidth - (float)atof(val); + break; + case 2 : + PageBottom = (float)atof(val); + break; + case 3 : + PageLeft = (float)atof(val); + break; + } + } + + if (change_size) + UpdatePageVars(); + + if (ppdIsMarked(ppd, "Duplex", "DuplexNoTumble") || + ppdIsMarked(ppd, "Duplex", "DuplexTumble") || + ppdIsMarked(ppd, "JCLDuplex", "DuplexNoTumble") || + ppdIsMarked(ppd, "JCLDuplex", "DuplexTumble") || + ppdIsMarked(ppd, "EFDuplex", "DuplexNoTumble") || + ppdIsMarked(ppd, "EFDuplex", "DuplexTumble") || + ppdIsMarked(ppd, "KD03Duplex", "DuplexNoTumble") || + ppdIsMarked(ppd, "KD03Duplex", "DuplexTumble")) + Duplex = 1; + + return (ppd); +} + + +/* + * 'UpdatePageVars()' - Update the page variables for the orientation. + */ + +void +UpdatePageVars(void) +{ + float temp; /* Swapping variable */ + + + switch (Orientation & 3) + { + case 0 : /* Portait */ + break; + + case 1 : /* Landscape */ + temp = PageLeft; + PageLeft = PageBottom; + PageBottom = temp; + + temp = PageRight; + PageRight = PageTop; + PageTop = temp; + + temp = PageWidth; + PageWidth = PageLength; + PageLength = temp; + break; + + case 2 : /* Reverse Portrait */ + temp = PageWidth - PageLeft; + PageLeft = PageWidth - PageRight; + PageRight = temp; + + temp = PageLength - PageBottom; + PageBottom = PageLength - PageTop; + PageTop = temp; + break; + + case 3 : /* Reverse Landscape */ + temp = PageWidth - PageLeft; + PageLeft = PageWidth - PageRight; + PageRight = temp; + + temp = PageLength - PageBottom; + PageBottom = PageLength - PageTop; + PageTop = temp; + + temp = PageLeft; + PageLeft = PageBottom; + PageBottom = temp; + + temp = PageRight; + PageRight = PageTop; + PageTop = temp; + + temp = PageWidth; + PageWidth = PageLength; + PageLength = temp; + break; + } +} + + +/* + * 'WriteCommon()' - Write common procedures... + */ + +void +WriteCommon(void) +{ + puts("% x y w h ESPrc - Clip to a rectangle.\n" + "userdict/ESPrc/rectclip where{pop/rectclip load}\n" + "{{newpath 4 2 roll moveto 1 index 0 rlineto 0 exch rlineto\n" + "neg 0 rlineto closepath clip newpath}bind}ifelse put"); + puts("% x y w h ESPrf - Fill a rectangle.\n" + "userdict/ESPrf/rectfill where{pop/rectfill load}\n" + "{{gsave newpath 4 2 roll moveto 1 index 0 rlineto 0 exch rlineto\n" + "neg 0 rlineto closepath fill grestore}bind}ifelse put"); + puts("% x y w h ESPrs - Stroke a rectangle.\n" + "userdict/ESPrs/rectstroke where{pop/rectstroke load}\n" + "{{gsave newpath 4 2 roll moveto 1 index 0 rlineto 0 exch rlineto\n" + "neg 0 rlineto closepath stroke grestore}bind}ifelse put"); +} + + +/* + * 'WriteLabelProlog()' - Write the prolog with the classification + * and page label. + */ + +void +WriteLabelProlog(const char *label, /* I - Page label */ + float bottom, /* I - Bottom position in points */ + float top, /* I - Top position in points */ + float width) /* I - Width in points */ +{ + const char *classification; /* CLASSIFICATION environment variable */ + const char *ptr; /* Temporary string pointer */ + + + /* + * First get the current classification... + */ + + if ((classification = getenv("CLASSIFICATION")) == NULL) + classification = ""; + if (strcmp(classification, "none") == 0) + classification = ""; + + /* + * If there is nothing to show, bind an empty 'write labels' procedure + * and return... + */ + + if (!classification[0] && (label == NULL || !label[0])) + { + puts("userdict/ESPwl{}bind put"); + return; + } + + /* + * Set the classification + page label string... + */ + + printf("userdict"); + if (strcmp(classification, "confidential") == 0) + printf("/ESPpl(CONFIDENTIAL"); + else if (strcmp(classification, "classified") == 0) + printf("/ESPpl(CLASSIFIED"); + else if (strcmp(classification, "secret") == 0) + printf("/ESPpl(SECRET"); + else if (strcmp(classification, "topsecret") == 0) + printf("/ESPpl(TOP SECRET"); + else if (strcmp(classification, "unclassified") == 0) + printf("/ESPpl(UNCLASSIFIED"); + else + { + printf("/ESPpl("); + + for (ptr = classification; *ptr; ptr ++) + if (*ptr < 32 || *ptr > 126) + printf("\\%03o", *ptr); + else if (*ptr == '_') + putchar(' '); + else + { + if (*ptr == '(' || *ptr == ')' || *ptr == '\\') + putchar('\\'); + + putchar(*ptr); + } + } + + if (label) + { + if (classification[0]) + printf(" - "); + + /* + * Quote the label string as needed... + */ + + for (ptr = label; *ptr; ptr ++) + if (*ptr < 32 || *ptr > 126) + printf("\\%03o", *ptr); + else + { + if (*ptr == '(' || *ptr == ')' || *ptr == '\\') + putchar('\\'); + + putchar(*ptr); + } + } + + puts(")put"); + + /* + * Then get a 14 point Helvetica-Bold font... + */ + + puts("userdict/ESPpf /Helvetica-Bold findfont 14 scalefont put"); + + /* + * Finally, the procedure to write the labels on the page... + */ + + puts("userdict/ESPwl{"); + puts(" ESPpf setfont"); + printf(" ESPpl stringwidth pop dup 12 add exch -0.5 mul %.0f add\n", + width * 0.5f); + puts(" 1 setgray"); + printf(" dup 6 sub %.0f 3 index 20 ESPrf\n", bottom - 2.0); + printf(" dup 6 sub %.0f 3 index 20 ESPrf\n", top - 18.0); + puts(" 0 setgray"); + printf(" dup 6 sub %.0f 3 index 20 ESPrs\n", bottom - 2.0); + printf(" dup 6 sub %.0f 3 index 20 ESPrs\n", top - 18.0); + printf(" dup %.0f moveto ESPpl show\n", bottom + 2.0); + printf(" %.0f moveto ESPpl show\n", top - 14.0); + puts("pop"); + puts("}bind put"); +} + + +/* + * 'WriteLabels()' - Write the actual page labels. + */ + +void +WriteLabels(int orient) /* I - Orientation of the page */ +{ + float width, /* Width of page */ + length; /* Length of page */ + + + puts("gsave"); + + if ((orient ^ Orientation) & 1) + { + width = PageLength; + length = PageWidth; + } + else + { + width = PageWidth; + length = PageLength; + } + + switch (orient & 3) + { + case 1 : /* Landscape */ + printf("%.1f 0.0 translate 90 rotate\n", length); + break; + case 2 : /* Reverse Portrait */ + printf("%.1f %.1f translate 180 rotate\n", width, length); + break; + case 3 : /* Reverse Landscape */ + printf("0.0 %.1f translate -90 rotate\n", width); + break; + } + + puts("ESPwl"); + puts("grestore"); +} + + +/* + * 'WriteTextComment()' - Write a DSC text comment. + */ + +void +WriteTextComment(const char *name, /* I - Comment name ("Title", etc.) */ + const char *value) /* I - Comment value */ +{ + int len; /* Current line length */ + + + /* + * DSC comments are of the form: + * + * %%name: value + * + * The name and value must be limited to 7-bit ASCII for most printers, + * so we escape all non-ASCII and ASCII control characters as described + * in the Adobe Document Structuring Conventions specification. + */ + + printf("%%%%%s: (", name); + len = 5 + strlen(name); + + while (*value) + { + if (*value < ' ' || *value >= 127) + { + /* + * Escape this character value... + */ + + if (len >= 251) /* Keep line < 254 chars */ + break; + + printf("\\%03o", *value & 255); + len += 4; + } + else if (*value == '\\') + { + /* + * Escape the backslash... + */ + + if (len >= 253) /* Keep line < 254 chars */ + break; + + putchar('\\'); + putchar('\\'); + len += 2; + } + else + { + /* + * Put this character literally... + */ + + if (len >= 254) /* Keep line < 254 chars */ + break; + + putchar(*value); + len ++; + } + + value ++; + } + + puts(")"); +} + + +/* + * End of "$Id$". + */ diff --git a/filter/common.h b/filter/common.h new file mode 100644 index 0000000000..bbaf99eed0 --- /dev/null +++ b/filter/common.h @@ -0,0 +1,78 @@ +/* + * "$Id$" + * + * Common filter definitions for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include + + +/* + * C++ magic... + */ + +#ifdef __cplusplus +extern "C" { +#endif /* __cplusplus */ + + +/* + * Globals... + */ + +extern int Orientation, /* 0 = portrait, 1 = landscape, etc. */ + Duplex, /* Duplexed? */ + LanguageLevel, /* Language level of printer */ + ColorDevice; /* Do color text? */ +extern float PageLeft, /* Left margin */ + PageRight, /* Right margin */ + PageBottom, /* Bottom margin */ + PageTop, /* Top margin */ + PageWidth, /* Total page width */ + PageLength; /* Total page length */ + + +/* + * Prototypes... + */ + +extern ppd_file_t *SetCommonOptions(int num_options, cups_option_t *options, + int change_size); +extern void UpdatePageVars(void); +extern void WriteCommon(void); +extern void WriteLabelProlog(const char *label, float bottom, + float top, float width); +extern void WriteLabels(int orient); +extern void WriteTextComment(const char *name, const char *value); + + +/* + * C++ magic... + */ + +#ifdef __cplusplus +} +#endif /* __cplusplus */ + + +/* + * End of "$Id$". + */ diff --git a/filter/error.c b/filter/error.c new file mode 100644 index 0000000000..3a7afda3fc --- /dev/null +++ b/filter/error.c @@ -0,0 +1,286 @@ +/* + * "$Id$" + * + * Raster error handling for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * _cupsRasterAddError() - Add an error message to the error buffer. + * _cupsRasterClearError() - Clear the error buffer. + * cupsRasterErrorString() - Return the last error from a raster function. + * get_error_buffer() - Return a pointer to thread local storage. + * raster_init() - Initialize error buffer once. + * raster_destructor() - Free memory allocated by get_error_buffer(). + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * Local structures... + */ + +typedef struct _cups_raster_error_s /**** Error buffer structure ****/ +{ + char *start, /* Start of buffer */ + *current, /* Current position in buffer */ + *end; /* End of buffer */ +} _cups_raster_error_t; + + +/* + * Local functions... + */ + +static _cups_raster_error_t *get_error_buffer(void); + + +/* + * '_cupsRasterAddError()' - Add an error message to the error buffer. + */ + +void +_cupsRasterAddError(const char *f, /* I - Printf-style error message */ + ...) /* I - Additional arguments as needed */ +{ + _cups_raster_error_t *buf = get_error_buffer(); + /* Error buffer */ + va_list ap; /* Pointer to additional arguments */ + char s[2048]; /* Message string */ + size_t bytes; /* Bytes in message string */ + + + va_start(ap, f); + bytes = vsnprintf(s, sizeof(s), f, ap); + va_end(ap); + + if (bytes <= 0) + return; + + bytes ++; + + if (bytes >= sizeof(s)) + return; + + if (bytes > (size_t)(buf->end - buf->current)) + { + /* + * Allocate more memory... + */ + + char *temp; /* New buffer */ + size_t size; /* Size of buffer */ + + + size = buf->end - buf->start + 2 * bytes + 1024; + + if (buf->start) + temp = realloc(buf->start, size); + else + temp = malloc(size); + + if (!temp) + return; + + /* + * Update pointers... + */ + + buf->end = temp + size; + buf->current = temp + (buf->current - buf->start); + buf->start = temp; + } + + /* + * Append the message to the end of the current string... + */ + + memcpy(buf->current, s, bytes); + buf->current += bytes - 1; +} + + +/* + * '_cupsRasterClearError()' - Clear the error buffer. + */ + +void +_cupsRasterClearError(void) +{ + _cups_raster_error_t *buf = get_error_buffer(); + /* Error buffer */ + + + buf->current = buf->start; + + if (buf->start) + *(buf->start) = '\0'; +} + + +/* + * 'cupsRasterErrorString()' - Return the last error from a raster function. + * + * If there are no recent errors, NULL is returned. + * + * @since CUPS 1.3/Mac OS X 10.5@ + */ + +const char * /* O - Last error */ +cupsRasterErrorString(void) +{ + _cups_raster_error_t *buf = get_error_buffer(); + /* Error buffer */ + + + if (buf->current == buf->start) + return (NULL); + else + return (buf->start); +} + + +#ifdef HAVE_PTHREAD_H +/* + * Implement per-thread globals... + */ + +# include + + +/* + * Local globals... + */ + +static pthread_key_t raster_key = -1; + /* Thread local storage key */ +static pthread_once_t raster_key_once = PTHREAD_ONCE_INIT; + /* One-time initialization object */ + + +/* + * Local functions... + */ + +static void raster_init(void); +static void raster_destructor(void *value); + + +/* + * 'get_error_buffer()' - Return a pointer to thread local storage. + */ + +_cups_raster_error_t * /* O - Pointer to error buffer */ +get_error_buffer(void) +{ + _cups_raster_error_t *buf; /* Pointer to error buffer */ + + + /* + * Initialize the global data exactly once... + */ + + DEBUG_puts("get_error_buffer()"); + + pthread_once(&raster_key_once, raster_init); + + /* + * See if we have allocated the data yet... + */ + + if ((buf = (_cups_raster_error_t *)pthread_getspecific(raster_key)) + == NULL) + { + DEBUG_puts("get_error_buffer: allocating memory for thread..."); + + /* + * No, allocate memory as set the pointer for the key... + */ + + buf = calloc(1, sizeof(_cups_raster_error_t)); + pthread_setspecific(raster_key, buf); + + DEBUG_printf((" buf=%p\n", buf)); + } + + /* + * Return the pointer to the data... + */ + + return (buf); +} + + +/* + * 'raster_init()' - Initialize error buffer once. + */ + +static void +raster_init(void) +{ + pthread_key_create(&raster_key, raster_destructor); + + DEBUG_printf(("raster_init(): raster_key=%x(%u)\n", (unsigned)raster_key, + (unsigned)raster_key)); +} + + +/* + * 'raster_destructor()' - Free memory allocated by get_error_buffer(). + */ + +static void +raster_destructor(void *value) /* I - Data to free */ +{ + _cups_raster_error_t *buf = (_cups_raster_error_t *)value; + /* Error buffer */ + + + DEBUG_printf(("raster_destructor(value=%p)\n", value)); + + if (buf->start) + free(buf->start); + + free(value); +} + + +#else +/* + * Implement static globals... + */ + +/* + * 'get_error_buffer()' - Return a pointer to thread local storage. + */ + +_cups_raster_error_t * /* O - Pointer to error buffer */ +get_error_buffer(void) +{ + static _cups_raster_error_t buf = { 0, 0, 0 }; + /* Error buffer */ + + + return (&buf); +} +#endif /* HAVE_PTHREAD_H */ + + +/* + * End of "$Id$". + */ diff --git a/filter/gziptoany.c b/filter/gziptoany.c new file mode 100644 index 0000000000..bde1c81a6f --- /dev/null +++ b/filter/gziptoany.c @@ -0,0 +1,112 @@ +/* + * "$Id$" + * + * GZIP/raw pre-filter for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1993-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Copy (and uncompress) files to stdout. + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * 'main()' - Copy (and uncompress) files to stdout. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + cups_file_t *fp; /* File */ + char buffer[8192]; /* Data buffer */ + int bytes; /* Number of bytes read/written */ + int copies; /* Number of copies */ + + + /* + * Check command-line... + */ + + if (argc != 7) + { + _cupsLangPrintf(stderr, + _("Usage: %s job-id user title copies options file"), + argv[0]); + return (1); + } + + /* + * Get the copy count; if we have no final content type, this is a + * raw queue or raw print file, so we need to make copies... + */ + + if (!getenv("FINAL_CONTENT_TYPE")) + copies = atoi(argv[4]); + else + copies = 1; + + /* + * Open the file... + */ + + if ((fp = cupsFileOpen(argv[6], "r")) == NULL) + { + _cupsLangPrintError("ERROR", _("Unable to open print file")); + return (1); + } + + /* + * Copy the file to stdout... + */ + + while (copies > 0) + { + if (!getenv("FINAL_CONTENT_TYPE")) + fputs("PAGE: 1 1\n", stderr); + + cupsFileRewind(fp); + + while ((bytes = cupsFileRead(fp, buffer, sizeof(buffer))) > 0) + if (write(1, buffer, bytes) < bytes) + { + _cupsLangPrintFilter(stderr, "ERROR", + _("Unable to write uncompressed print data: %s"), + strerror(errno)); + cupsFileClose(fp); + + return (1); + } + + copies --; + } + + /* + * Close the file and return... + */ + + cupsFileClose(fp); + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/filter/interpret.c b/filter/interpret.c new file mode 100644 index 0000000000..5d7e327636 --- /dev/null +++ b/filter/interpret.c @@ -0,0 +1,1688 @@ +/* + * "$Id$" + * + * PPD command interpreter for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1993-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsRasterInterpretPPD() - Interpret PPD commands to create a page header. + * _cupsRasterExecPS() - Execute PostScript code to initialize a page + * header. + * cleartomark_stack() - Clear to the last mark ([) on the stack. + * copy_stack() - Copy the top N stack objects. + * delete_stack() - Free memory used by a stack. + * error_object() - Add an object's value to the current error + * message. + * error_stack() - Add a stack to the current error message. + * index_stack() - Copy the Nth value on the stack. + * new_stack() - Create a new stack. + * pop_stock() - Pop the top object off the stack. + * push_stack() - Push an object on the stack. + * roll_stack() - Rotate stack objects. + * scan_ps() - Scan a string for the next PS object. + * setpagedevice() - Simulate the PostScript setpagedevice operator. + * DEBUG_object() - Print an object value. + * DEBUG_stack() - Print a stack. + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * Stack values for the PostScript mini-interpreter... + */ + +typedef enum +{ + CUPS_PS_NAME, + CUPS_PS_NUMBER, + CUPS_PS_STRING, + CUPS_PS_BOOLEAN, + CUPS_PS_NULL, + CUPS_PS_START_ARRAY, + CUPS_PS_END_ARRAY, + CUPS_PS_START_DICT, + CUPS_PS_END_DICT, + CUPS_PS_START_PROC, + CUPS_PS_END_PROC, + CUPS_PS_CLEARTOMARK, + CUPS_PS_COPY, + CUPS_PS_DUP, + CUPS_PS_INDEX, + CUPS_PS_POP, + CUPS_PS_ROLL, + CUPS_PS_SETPAGEDEVICE, + CUPS_PS_STOPPED, + CUPS_PS_OTHER +} _cups_ps_type_t; + +typedef struct +{ + _cups_ps_type_t type; /* Object type */ + union + { + int boolean; /* Boolean value */ + char name[64]; /* Name value */ + double number; /* Number value */ + char other[64]; /* Other operator */ + char string[64]; /* Sring value */ + } value; /* Value */ +} _cups_ps_obj_t; + +typedef struct +{ + int num_objs, /* Number of objects on stack */ + alloc_objs; /* Number of allocated objects */ + _cups_ps_obj_t *objs; /* Objects in stack */ +} _cups_ps_stack_t; + + +/* + * Local functions... + */ + +static int cleartomark_stack(_cups_ps_stack_t *st); +static int copy_stack(_cups_ps_stack_t *st, int count); +static void delete_stack(_cups_ps_stack_t *st); +static void error_object(_cups_ps_obj_t *obj); +static void error_stack(_cups_ps_stack_t *st, const char *title); +static _cups_ps_obj_t *index_stack(_cups_ps_stack_t *st, int n); +static _cups_ps_stack_t *new_stack(void); +static _cups_ps_obj_t *pop_stack(_cups_ps_stack_t *st); +static _cups_ps_obj_t *push_stack(_cups_ps_stack_t *st, + _cups_ps_obj_t *obj); +static int roll_stack(_cups_ps_stack_t *st, int c, int s); +static _cups_ps_obj_t *scan_ps(_cups_ps_stack_t *st, char **ptr); +static int setpagedevice(_cups_ps_stack_t *st, + cups_page_header2_t *h, + int *preferred_bits); +#ifdef DEBUG +static void DEBUG_object(_cups_ps_obj_t *obj); +static void DEBUG_stack(_cups_ps_stack_t *st); +#endif /* DEBUG */ + + +/* + * 'cupsRasterInterpretPPD()' - Interpret PPD commands to create a page header. + * + * This function is used by raster image processing (RIP) filters like + * cgpdftoraster and imagetoraster when writing CUPS raster data for a page. + * It is not used by raster printer driver filters which only read CUPS + * raster data. + * + * + * @code cupsRasterInterpretPPD@ does not mark the options in the PPD using + * the "num_options" and "options" arguments. Instead, mark the options with + * @code cupsMarkOptions@ and @code ppdMarkOption@ prior to calling it - + * this allows for per-page options without manipulating the options array. + * + * The "func" argument specifies an optional callback function that is + * called prior to the computation of the final raster data. The function + * can make changes to the @link cups_page_header2_t@ data as needed to use a + * supported raster format and then returns 0 on success and -1 if the + * requested attributes cannot be supported. + * + * + * @code cupsRasterInterpretPPD@ supports a subset of the PostScript language. + * Currently only the @code [@, @code ]@, @code <<@, @code >>@, @code {@, + * @code }@, @code cleartomark@, @code copy@, @code dup@, @code index@, + * @code pop@, @code roll@, @code setpagedevice@, and @code stopped@ operators + * are supported. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +int /* O - 0 on success, -1 on failure */ +cupsRasterInterpretPPD( + cups_page_header2_t *h, /* O - Page header to create */ + ppd_file_t *ppd, /* I - PPD file */ + int num_options, /* I - Number of options */ + cups_option_t *options, /* I - Options */ + cups_interpret_cb_t func) /* I - Optional page header callback (@code NULL@ for none) */ +{ + int status; /* Cummulative status */ + char *code; /* Code to run */ + const char *val; /* Option value */ + ppd_size_t *size; /* Current size */ + float left, /* Left position */ + bottom, /* Bottom position */ + right, /* Right position */ + top; /* Top position */ + int preferred_bits; /* Preferred bits per color */ + + + /* + * Range check input... + */ + + _cupsRasterClearError(); + + if (!h) + { + _cupsRasterAddError("Page header cannot be NULL!\n"); + return (-1); + } + + /* + * Reset the page header to the defaults... + */ + + memset(h, 0, sizeof(cups_page_header2_t)); + + h->NumCopies = 1; + h->PageSize[0] = 612; + h->PageSize[1] = 792; + h->HWResolution[0] = 100; + h->HWResolution[1] = 100; + h->cupsBitsPerColor = 1; + h->cupsColorOrder = CUPS_ORDER_CHUNKED; + h->cupsColorSpace = CUPS_CSPACE_K; + h->cupsBorderlessScalingFactor = 1.0f; + h->cupsPageSize[0] = 612.0f; + h->cupsPageSize[1] = 792.0f; + h->cupsImagingBBox[0] = 0.0f; + h->cupsImagingBBox[1] = 0.0f; + h->cupsImagingBBox[2] = 612.0f; + h->cupsImagingBBox[3] = 792.0f; + + strcpy(h->cupsPageSizeName, "Letter"); + +#ifdef __APPLE__ + /* + * cupsInteger0 is also used for the total page count on Mac OS X; set an + * uncommon default value so we can tell if the driver is using cupsInteger0. + */ + + h->cupsInteger[0] = 0x80000000; +#endif /* __APPLE__ */ + + /* + * Apply patches and options to the page header... + */ + + status = 0; + preferred_bits = 0; + + if (ppd) + { + /* + * Apply any patch code (used to override the defaults...) + */ + + if (ppd->patches) + status |= _cupsRasterExecPS(h, &preferred_bits, ppd->patches); + + /* + * Then apply printer options in the proper order... + */ + + if ((code = ppdEmitString(ppd, PPD_ORDER_DOCUMENT, 0.0)) != NULL) + { + status |= _cupsRasterExecPS(h, &preferred_bits, code); + free(code); + } + + if ((code = ppdEmitString(ppd, PPD_ORDER_ANY, 0.0)) != NULL) + { + status |= _cupsRasterExecPS(h, &preferred_bits, code); + free(code); + } + + if ((code = ppdEmitString(ppd, PPD_ORDER_PROLOG, 0.0)) != NULL) + { + status |= _cupsRasterExecPS(h, &preferred_bits, code); + free(code); + } + + if ((code = ppdEmitString(ppd, PPD_ORDER_PAGE, 0.0)) != NULL) + { + status |= _cupsRasterExecPS(h, &preferred_bits, code); + free(code); + } + } + + /* + * Allow option override for page scaling... + */ + + if ((val = cupsGetOption("cupsBorderlessScalingFactor", num_options, + options)) != NULL) + { + double sc = atof(val); /* Scale factor */ + + if (sc >= 0.1 && sc <= 2.0) + h->cupsBorderlessScalingFactor = (float)sc; + } + + /* + * Get the margins for the current size... + */ + + if ((size = ppdPageSize(ppd, NULL)) != NULL) + { + /* + * Use the margins from the PPD file... + */ + + left = size->left; + bottom = size->bottom; + right = size->right; + top = size->top; + + strlcpy(h->cupsPageSizeName, size->name, sizeof(h->cupsPageSizeName)); + + h->cupsPageSize[0] = size->width; + h->cupsPageSize[1] = size->length; + } + else + { + /* + * Use the default margins... + */ + + left = 0.0f; + bottom = 0.0f; + right = 612.0f; + top = 792.0f; + } + + h->PageSize[0] = (unsigned)(h->cupsPageSize[0] * + h->cupsBorderlessScalingFactor); + h->PageSize[1] = (unsigned)(h->cupsPageSize[1] * + h->cupsBorderlessScalingFactor); + h->Margins[0] = (unsigned)(left * + h->cupsBorderlessScalingFactor); + h->Margins[1] = (unsigned)(bottom * + h->cupsBorderlessScalingFactor); + h->ImagingBoundingBox[0] = (unsigned)(left * + h->cupsBorderlessScalingFactor); + h->ImagingBoundingBox[1] = (unsigned)(bottom * + h->cupsBorderlessScalingFactor); + h->ImagingBoundingBox[2] = (unsigned)(right * + h->cupsBorderlessScalingFactor); + h->ImagingBoundingBox[3] = (unsigned)(top * + h->cupsBorderlessScalingFactor); + h->cupsImagingBBox[0] = (float)left; + h->cupsImagingBBox[1] = (float)bottom; + h->cupsImagingBBox[2] = (float)right; + h->cupsImagingBBox[3] = (float)top; + + /* + * Use the callback to validate the page header... + */ + + if (func && (*func)(h, preferred_bits)) + { + _cupsRasterAddError("Page header callback returned error.\n"); + return (-1); + } + + /* + * Check parameters... + */ + + if (!h->HWResolution[0] || !h->HWResolution[1] || + !h->PageSize[0] || !h->PageSize[1] || + (h->cupsBitsPerColor != 1 && h->cupsBitsPerColor != 2 && + h->cupsBitsPerColor != 4 && h->cupsBitsPerColor != 8 && + h->cupsBitsPerColor != 16) || + h->cupsBorderlessScalingFactor < 0.1 || + h->cupsBorderlessScalingFactor > 2.0) + { + _cupsRasterAddError("Page header uses unsupported values.\n"); + return (-1); + } + + /* + * Compute the bitmap parameters... + */ + + h->cupsWidth = (int)((right - left) * h->cupsBorderlessScalingFactor * + h->HWResolution[0] / 72.0f + 0.5f); + h->cupsHeight = (int)((top - bottom) * h->cupsBorderlessScalingFactor * + h->HWResolution[1] / 72.0f + 0.5f); + + switch (h->cupsColorSpace) + { + case CUPS_CSPACE_W : + case CUPS_CSPACE_K : + case CUPS_CSPACE_WHITE : + case CUPS_CSPACE_GOLD : + case CUPS_CSPACE_SILVER : + case CUPS_CSPACE_SW : + h->cupsNumColors = 1; + h->cupsBitsPerPixel = h->cupsBitsPerColor; + break; + + default : + /* + * Ensure that colorimetric colorspaces use at least 8 bits per + * component... + */ + + if (h->cupsColorSpace >= CUPS_CSPACE_CIEXYZ && + h->cupsBitsPerColor < 8) + h->cupsBitsPerColor = 8; + + /* + * Figure out the number of bits per pixel... + */ + + if (h->cupsColorOrder == CUPS_ORDER_CHUNKED) + { + if (h->cupsBitsPerColor >= 8) + h->cupsBitsPerPixel = h->cupsBitsPerColor * 3; + else + h->cupsBitsPerPixel = h->cupsBitsPerColor * 4; + } + else + h->cupsBitsPerPixel = h->cupsBitsPerColor; + + h->cupsNumColors = 3; + break; + + case CUPS_CSPACE_KCMYcm : + if (h->cupsBitsPerColor == 1) + { + if (h->cupsColorOrder == CUPS_ORDER_CHUNKED) + h->cupsBitsPerPixel = 8; + else + h->cupsBitsPerPixel = 1; + + h->cupsNumColors = 6; + break; + } + + /* + * Fall through to CMYK code... + */ + + case CUPS_CSPACE_RGBA : + case CUPS_CSPACE_RGBW : + case CUPS_CSPACE_CMYK : + case CUPS_CSPACE_YMCK : + case CUPS_CSPACE_KCMY : + case CUPS_CSPACE_GMCK : + case CUPS_CSPACE_GMCS : + if (h->cupsColorOrder == CUPS_ORDER_CHUNKED) + h->cupsBitsPerPixel = h->cupsBitsPerColor * 4; + else + h->cupsBitsPerPixel = h->cupsBitsPerColor; + + h->cupsNumColors = 4; + break; + + case CUPS_CSPACE_DEVICE1 : + case CUPS_CSPACE_DEVICE2 : + case CUPS_CSPACE_DEVICE3 : + case CUPS_CSPACE_DEVICE4 : + case CUPS_CSPACE_DEVICE5 : + case CUPS_CSPACE_DEVICE6 : + case CUPS_CSPACE_DEVICE7 : + case CUPS_CSPACE_DEVICE8 : + case CUPS_CSPACE_DEVICE9 : + case CUPS_CSPACE_DEVICEA : + case CUPS_CSPACE_DEVICEB : + case CUPS_CSPACE_DEVICEC : + case CUPS_CSPACE_DEVICED : + case CUPS_CSPACE_DEVICEE : + case CUPS_CSPACE_DEVICEF : + h->cupsNumColors = h->cupsColorSpace - CUPS_CSPACE_DEVICE1 + 1; + + if (h->cupsColorOrder == CUPS_ORDER_CHUNKED) + h->cupsBitsPerPixel = h->cupsBitsPerColor * h->cupsNumColors; + else + h->cupsBitsPerPixel = h->cupsBitsPerColor; + break; + } + + h->cupsBytesPerLine = (h->cupsBitsPerPixel * h->cupsWidth + 7) / 8; + + if (h->cupsColorOrder == CUPS_ORDER_BANDED) + h->cupsBytesPerLine *= h->cupsNumColors; + + return (status); +} + + +/* + * '_cupsRasterExecPS()' - Execute PostScript code to initialize a page header. + */ + +int /* O - 0 on success, -1 on error */ +_cupsRasterExecPS( + cups_page_header2_t *h, /* O - Page header */ + int *preferred_bits,/* O - Preferred bits per color */ + const char *code) /* I - PS code to execute */ +{ + _cups_ps_stack_t *st; /* PostScript value stack */ + _cups_ps_obj_t *obj; /* Object from top of stack */ + char *codecopy, /* Copy of code */ + *codeptr; /* Pointer into copy of code */ + + + DEBUG_printf(("_cupsRasterExecPS(h=%p, preferred_bits=%p, code=\"%s\")\n", + h, preferred_bits, code ? code : "(null)")); + + /* + * Copy the PostScript code and create a stack... + */ + + if ((codecopy = strdup(code)) == NULL) + { + _cupsRasterAddError("Unable to duplicate code string.\n"); + return (-1); + } + + if ((st = new_stack()) == NULL) + { + _cupsRasterAddError("Unable to create stack.\n"); + free(codecopy); + return (-1); + } + + /* + * Parse the PS string until we run out of data... + */ + + codeptr = codecopy; + + while ((obj = scan_ps(st, &codeptr)) != NULL) + { +#ifdef DEBUG + DEBUG_printf(("_cupsRasterExecPS: Stack (%d objects)\n", st->num_objs)); + DEBUG_object(obj); +#endif /* DEBUG */ + + switch (obj->type) + { + default : + /* Do nothing for regular values */ + break; + + case CUPS_PS_CLEARTOMARK : + pop_stack(st); + + if (cleartomark_stack(st)) + _cupsRasterAddError("cleartomark: Stack underflow!\n"); + +#ifdef DEBUG + DEBUG_puts(" dup: "); + DEBUG_stack(st); +#endif /* DEBUG */ + break; + + case CUPS_PS_COPY : + pop_stack(st); + if ((obj = pop_stack(st)) != NULL) + { + copy_stack(st, (int)obj->value.number); + +#ifdef DEBUG + DEBUG_puts("_cupsRasterExecPS: copy"); + DEBUG_stack(st); +#endif /* DEBUG */ + } + break; + + case CUPS_PS_DUP : + pop_stack(st); + copy_stack(st, 1); + +#ifdef DEBUG + DEBUG_puts("_cupsRasterExecPS: dup"); + DEBUG_stack(st); +#endif /* DEBUG */ + break; + + case CUPS_PS_INDEX : + pop_stack(st); + if ((obj = pop_stack(st)) != NULL) + { + index_stack(st, (int)obj->value.number); + +#ifdef DEBUG + DEBUG_puts("_cupsRasterExecPS: index"); + DEBUG_stack(st); +#endif /* DEBUG */ + } + break; + + case CUPS_PS_POP : + pop_stack(st); + pop_stack(st); + +#ifdef DEBUG + DEBUG_puts("_cupsRasterExecPS: pop"); + DEBUG_stack(st); +#endif /* DEBUG */ + break; + + case CUPS_PS_ROLL : + pop_stack(st); + if ((obj = pop_stack(st)) != NULL) + { + int c; /* Count */ + + + c = (int)obj->value.number; + + if ((obj = pop_stack(st)) != NULL) + { + roll_stack(st, (int)obj->value.number, c); + +#ifdef DEBUG + DEBUG_puts("_cupsRasterExecPS: roll"); + DEBUG_stack(st); +#endif /* DEBUG */ + } + } + break; + + case CUPS_PS_SETPAGEDEVICE : + pop_stack(st); + setpagedevice(st, h, preferred_bits); + +#ifdef DEBUG + DEBUG_puts("_cupsRasterExecPS: setpagedevice"); + DEBUG_stack(st); +#endif /* DEBUG */ + break; + + case CUPS_PS_START_PROC : + case CUPS_PS_END_PROC : + case CUPS_PS_STOPPED : + pop_stack(st); + break; + + case CUPS_PS_OTHER : + _cupsRasterAddError("Unknown operator \"%s\"!\n", obj->value.other); + DEBUG_printf(("_cupsRasterExecPS: Unknown operator \"%s\"!\n", + obj->value.other)); + break; + } + + if (obj && obj->type == CUPS_PS_OTHER) + break; + } + + /* + * Cleanup... + */ + + free(codecopy); + + if (st->num_objs > 0) + { + error_stack(st, "Stack not empty:"); + +#ifdef DEBUG + DEBUG_puts("_cupsRasterExecPS: Stack not empty:"); + DEBUG_stack(st); +#endif /* DEBUG */ + + delete_stack(st); + + return (-1); + } + + delete_stack(st); + + /* + * Return success... + */ + + return (0); +} + + +/* + * 'cleartomark_stack()' - Clear to the last mark ([) on the stack. + */ + +static int /* O - 0 on success, -1 on error */ +cleartomark_stack(_cups_ps_stack_t *st) /* I - Stack */ +{ + _cups_ps_obj_t *obj; /* Current object on stack */ + + + while ((obj = pop_stack(st)) != NULL) + if (obj->type == CUPS_PS_START_ARRAY) + break; + + return (obj ? 0 : -1); +} + + +/* + * 'copy_stack()' - Copy the top N stack objects. + */ + +static int /* O - 0 on success, -1 on error */ +copy_stack(_cups_ps_stack_t *st, /* I - Stack */ + int c) /* I - Number of objects to copy */ +{ + int n; /* Index */ + + + if (c < 0) + return (-1); + else if (c == 0) + return (0); + + if ((n = st->num_objs - c) < 0) + return (-1); + + while (c > 0) + { + if (!push_stack(st, st->objs + n)) + return (-1); + + n ++; + c --; + } + + return (0); +} + + +/* + * 'delete_stack()' - Free memory used by a stack. + */ + +static void +delete_stack(_cups_ps_stack_t *st) /* I - Stack */ +{ + free(st->objs); + free(st); +} + + +/* + * 'error_object()' - Add an object's value to the current error message. + */ + +static void +error_object(_cups_ps_obj_t *obj) /* I - Object to add */ +{ + switch (obj->type) + { + case CUPS_PS_NAME : + _cupsRasterAddError(" /%s", obj->value.name); + break; + + case CUPS_PS_NUMBER : + _cupsRasterAddError(" %g", obj->value.number); + break; + + case CUPS_PS_STRING : + _cupsRasterAddError(" (%s)", obj->value.string); + break; + + case CUPS_PS_BOOLEAN : + if (obj->value.boolean) + _cupsRasterAddError(" true"); + else + _cupsRasterAddError(" false"); + break; + + case CUPS_PS_NULL : + _cupsRasterAddError(" null"); + break; + + case CUPS_PS_START_ARRAY : + _cupsRasterAddError(" ["); + break; + + case CUPS_PS_END_ARRAY : + _cupsRasterAddError(" ]"); + break; + + case CUPS_PS_START_DICT : + _cupsRasterAddError(" <<"); + break; + + case CUPS_PS_END_DICT : + _cupsRasterAddError(" >>"); + break; + + case CUPS_PS_START_PROC : + _cupsRasterAddError(" {"); + break; + + case CUPS_PS_END_PROC : + _cupsRasterAddError(" }"); + break; + + case CUPS_PS_COPY : + _cupsRasterAddError(" --copy--"); + break; + + case CUPS_PS_CLEARTOMARK : + _cupsRasterAddError(" --cleartomark--"); + break; + + case CUPS_PS_DUP : + _cupsRasterAddError(" --dup--"); + break; + + case CUPS_PS_INDEX : + _cupsRasterAddError(" --index--"); + break; + + case CUPS_PS_POP : + _cupsRasterAddError(" --pop--"); + break; + + case CUPS_PS_ROLL : + _cupsRasterAddError(" --roll--"); + break; + + case CUPS_PS_SETPAGEDEVICE : + _cupsRasterAddError(" --setpagedevice--"); + break; + + case CUPS_PS_STOPPED : + _cupsRasterAddError(" --stopped--"); + break; + + case CUPS_PS_OTHER : + _cupsRasterAddError(" --%s--", obj->value.other); + break; + } +} + + +/* + * 'error_stack()' - Add a stack to the current error message... + */ + +static void +error_stack(_cups_ps_stack_t *st, /* I - Stack */ + const char *title) /* I - Title string */ +{ + int c; /* Looping var */ + _cups_ps_obj_t *obj; /* Current object on stack */ + + + _cupsRasterAddError("%s", title); + + for (obj = st->objs, c = st->num_objs; c > 0; c --, obj ++) + error_object(obj); + + _cupsRasterAddError("\n"); +} + + +/* + * 'index_stack()' - Copy the Nth value on the stack. + */ + +static _cups_ps_obj_t * /* O - New object */ +index_stack(_cups_ps_stack_t *st, /* I - Stack */ + int n) /* I - Object index */ +{ + if (n < 0 || (n = st->num_objs - n - 1) < 0) + return (NULL); + + return (push_stack(st, st->objs + n)); +} + + +/* + * 'new_stack()' - Create a new stack. + */ + +static _cups_ps_stack_t * /* O - New stack */ +new_stack(void) +{ + _cups_ps_stack_t *st; /* New stack */ + + + if ((st = calloc(1, sizeof(_cups_ps_stack_t))) == NULL) + return (NULL); + + st->alloc_objs = 32; + + if ((st->objs = calloc(32, sizeof(_cups_ps_obj_t))) == NULL) + { + free(st); + return (NULL); + } + else + return (st); +} + + +/* + * 'pop_stock()' - Pop the top object off the stack. + */ + +static _cups_ps_obj_t * /* O - Object */ +pop_stack(_cups_ps_stack_t *st) /* I - Stack */ +{ + if (st->num_objs > 0) + { + st->num_objs --; + + return (st->objs + st->num_objs); + } + else + return (NULL); +} + + +/* + * 'push_stack()' - Push an object on the stack. + */ + +static _cups_ps_obj_t * /* O - New object */ +push_stack(_cups_ps_stack_t *st, /* I - Stack */ + _cups_ps_obj_t *obj) /* I - Object */ +{ + _cups_ps_obj_t *temp; /* New object */ + + + if (st->num_objs >= st->alloc_objs) + { + + + st->alloc_objs += 32; + + if ((temp = realloc(st->objs, st->alloc_objs * + sizeof(_cups_ps_obj_t))) == NULL) + return (NULL); + + st->objs = temp; + memset(temp + st->num_objs, 0, 32 * sizeof(_cups_ps_obj_t)); + } + + temp = st->objs + st->num_objs; + st->num_objs ++; + + memcpy(temp, obj, sizeof(_cups_ps_obj_t)); + + return (temp); +} + + +/* + * 'roll_stack()' - Rotate stack objects. + */ + +static int /* O - 0 on success, -1 on error */ +roll_stack(_cups_ps_stack_t *st, /* I - Stack */ + int c, /* I - Number of objects */ + int s) /* I - Amount to shift */ +{ + _cups_ps_obj_t *temp; /* Temporary array of objects */ + int n; /* Index into array */ + + + DEBUG_printf((" roll_stack(st=%p, s=%d, c=%d)\n", st, s, c)); + + /* + * Range check input... + */ + + if (c < 0) + return (-1); + else if (c == 0) + return (0); + + if ((n = st->num_objs - c) < 0) + return (-1); + + s %= c; + + if (s == 0) + return (0); + + /* + * Copy N objects and move things around... + */ + + if (s < 0) + { + /* + * Shift down... + */ + + s = -s; + + if ((temp = calloc(s, sizeof(_cups_ps_obj_t))) == NULL) + return (-1); + + memcpy(temp, st->objs + n, s * sizeof(_cups_ps_obj_t)); + memmove(st->objs + n, st->objs + n + s, (c - s) * sizeof(_cups_ps_obj_t)); + memcpy(st->objs + n + c - s, temp, s * sizeof(_cups_ps_obj_t)); + } + else + { + /* + * Shift up... + */ + + if ((temp = calloc(s, sizeof(_cups_ps_obj_t))) == NULL) + return (-1); + + memcpy(temp, st->objs + n + c - s, s * sizeof(_cups_ps_obj_t)); + memmove(st->objs + n + s, st->objs + n, + (c - s) * sizeof(_cups_ps_obj_t)); + memcpy(st->objs + n, temp, s * sizeof(_cups_ps_obj_t)); + } + + free(temp); + + return (0); +} + + +/* + * 'scan_ps()' - Scan a string for the next PS object. + */ + +static _cups_ps_obj_t * /* O - New object or NULL on EOF */ +scan_ps(_cups_ps_stack_t *st, /* I - Stack */ + char **ptr) /* IO - String pointer */ +{ + _cups_ps_obj_t obj; /* Current object */ + char *start, /* Start of object */ + *cur, /* Current position */ + *valptr, /* Pointer into value string */ + *valend; /* End of value string */ + int parens; /* Parenthesis nesting level */ + + + /* + * Skip leading whitespace... + */ + + for (cur = *ptr; *cur; cur ++) + { + if (*cur == '%') + { + /* + * Comment, skip to end of line... + */ + + for (cur ++; *cur && *cur != '\n' && *cur != '\r'; cur ++); + + if (!*cur) + cur --; + } + else if (!isspace(*cur & 255)) + break; + } + + if (!*cur) + { + *ptr = NULL; + + return (NULL); + } + + /* + * See what we have... + */ + + memset(&obj, 0, sizeof(obj)); + + switch (*cur) + { + case '(' : /* (string) */ + obj.type = CUPS_PS_STRING; + start = cur; + + for (cur ++, parens = 1, valptr = obj.value.string, + valend = obj.value.string + sizeof(obj.value.string) - 1; + *cur; + cur ++) + { + if (*cur == ')' && parens == 1) + break; + + if (*cur == '(') + parens ++; + else if (*cur == ')') + parens --; + + if (valptr >= valend) + { + *ptr = start; + + return (NULL); + } + + if (*cur == '\\') + { + /* + * Decode escaped character... + */ + + cur ++; + + if (*cur == 'b') + *valptr++ = '\b'; + else if (*cur == 'f') + *valptr++ = '\f'; + else if (*cur == 'n') + *valptr++ = '\n'; + else if (*cur == 'r') + *valptr++ = '\r'; + else if (*cur == 't') + *valptr++ = '\t'; + else if (*cur >= '0' && *cur <= '7') + { + int ch = *cur - '0'; + + if (cur[1] >= '0' && cur[1] <= '7') + { + cur ++; + ch = (ch << 3) + *cur - '0'; + } + + if (cur[1] >= '0' && cur[1] <= '7') + { + cur ++; + ch = (ch << 3) + *cur - '0'; + } + + *valptr++ = ch; + } + else if (*cur == '\r') + { + if (cur[1] == '\n') + cur ++; + } + else if (*cur != '\n') + *valptr++ = *cur; + } + else + *valptr++ = *cur; + } + + if (*cur != ')') + { + *ptr = start; + + return (NULL); + } + + cur ++; + break; + + case '[' : /* Start array */ + obj.type = CUPS_PS_START_ARRAY; + cur ++; + break; + + case ']' : /* End array */ + obj.type = CUPS_PS_END_ARRAY; + cur ++; + break; + + case '<' : /* Start dictionary or hex string */ + if (cur[1] == '<') + { + obj.type = CUPS_PS_START_DICT; + cur += 2; + } + else + { + obj.type = CUPS_PS_STRING; + start = cur; + + for (cur ++, valptr = obj.value.string, + valend = obj.value.string + sizeof(obj.value.string) - 1; + *cur; + cur ++) + { + int ch; /* Current character */ + + + + if (*cur == '>') + break; + else if (valptr >= valend || !isxdigit(*cur & 255)) + { + *ptr = start; + return (NULL); + } + + if (*cur >= '0' && *cur <= '9') + ch = (*cur - '0') << 4; + else + ch = (tolower(*cur) - 'a' + 10) << 4; + + if (isxdigit(cur[1] & 255)) + { + cur ++; + + if (*cur >= '0' && *cur <= '9') + ch |= *cur - '0'; + else + ch |= tolower(*cur) - 'a' + 10; + } + + *valptr++ = ch; + } + + if (*cur != '>') + { + *ptr = start; + return (NULL); + } + + cur ++; + } + break; + + case '>' : /* End dictionary? */ + if (cur[1] == '>') + { + obj.type = CUPS_PS_END_DICT; + cur += 2; + } + else + { + obj.type = CUPS_PS_OTHER; + obj.value.other[0] = *cur; + + cur ++; + } + break; + + case '{' : /* Start procedure */ + obj.type = CUPS_PS_START_PROC; + cur ++; + break; + + case '}' : /* End procedure */ + obj.type = CUPS_PS_END_PROC; + cur ++; + break; + + case '-' : /* Possible number */ + case '+' : + if (!isdigit(cur[1] & 255) && cur[1] != '.') + { + obj.type = CUPS_PS_OTHER; + obj.value.other[0] = *cur; + + cur ++; + break; + } + + case '0' : /* Number */ + case '1' : + case '2' : + case '3' : + case '4' : + case '5' : + case '6' : + case '7' : + case '8' : + case '9' : + case '.' : + obj.type = CUPS_PS_NUMBER; + + start = cur; + for (cur ++; *cur; cur ++) + if (!isdigit(*cur & 255)) + break; + + if (*cur == '#') + { + /* + * Integer with radix... + */ + + obj.value.number = strtol(cur + 1, &cur, atoi(start)); + break; + } + else if (strchr(".Ee()<>[]{}/%", *cur) || isspace(*cur & 255)) + { + /* + * Integer or real number... + */ + + obj.value.number = _cupsStrScand(start, &cur, localeconv()); + break; + } + else + cur = start; + + default : /* Operator/variable name */ + start = cur; + + if (*cur == '/') + { + obj.type = CUPS_PS_NAME; + valptr = obj.value.name; + valend = obj.value.name + sizeof(obj.value.name) - 1; + cur ++; + } + else + { + obj.type = CUPS_PS_OTHER; + valptr = obj.value.other; + valend = obj.value.other + sizeof(obj.value.other) - 1; + } + + while (*cur) + { + if (strchr("()<>[]{}/%", *cur) || isspace(*cur & 255)) + break; + else if (valptr < valend) + *valptr++ = *cur++; + else + { + *ptr = start; + return (NULL); + } + } + + if (obj.type == CUPS_PS_OTHER) + { + if (!strcmp(obj.value.other, "true")) + { + obj.type = CUPS_PS_BOOLEAN; + obj.value.boolean = 1; + } + else if (!strcmp(obj.value.other, "false")) + { + obj.type = CUPS_PS_BOOLEAN; + obj.value.boolean = 0; + } + else if (!strcmp(obj.value.other, "null")) + obj.type = CUPS_PS_NULL; + else if (!strcmp(obj.value.other, "cleartomark")) + obj.type = CUPS_PS_CLEARTOMARK; + else if (!strcmp(obj.value.other, "copy")) + obj.type = CUPS_PS_COPY; + else if (!strcmp(obj.value.other, "dup")) + obj.type = CUPS_PS_DUP; + else if (!strcmp(obj.value.other, "index")) + obj.type = CUPS_PS_INDEX; + else if (!strcmp(obj.value.other, "pop")) + obj.type = CUPS_PS_POP; + else if (!strcmp(obj.value.other, "roll")) + obj.type = CUPS_PS_ROLL; + else if (!strcmp(obj.value.other, "setpagedevice")) + obj.type = CUPS_PS_SETPAGEDEVICE; + else if (!strcmp(obj.value.other, "stopped")) + obj.type = CUPS_PS_STOPPED; + } + break; + } + + /* + * Save the current position in the string and return the new object... + */ + + *ptr = cur; + + return (push_stack(st, &obj)); +} + + +/* + * 'setpagedevice()' - Simulate the PostScript setpagedevice operator. + */ + +static int /* O - 0 on success, -1 on error */ +setpagedevice( + _cups_ps_stack_t *st, /* I - Stack */ + cups_page_header2_t *h, /* O - Page header */ + int *preferred_bits)/* O - Preferred bits per color */ +{ + int i; /* Index into array */ + _cups_ps_obj_t *obj, /* Current object */ + *end; /* End of dictionary */ + const char *name; /* Attribute name */ + + + /* + * Make sure we have a dictionary on the stack... + */ + + if (st->num_objs == 0) + return (-1); + + obj = end = st->objs + st->num_objs - 1; + + if (obj->type != CUPS_PS_END_DICT) + return (-1); + + obj --; + + while (obj > st->objs) + { + if (obj->type == CUPS_PS_START_DICT) + break; + + obj --; + } + + if (obj < st->objs) + return (-1); + + /* + * Found the start of the dictionary, empty the stack to this point... + */ + + st->num_objs = (int)(obj - st->objs); + + /* + * Now pull /name and value pairs from the dictionary... + */ + + DEBUG_puts("setpagedevice: Dictionary:"); + + for (obj ++; obj < end; obj ++) + { + /* + * Grab the name... + */ + + if (obj->type != CUPS_PS_NAME) + return (-1); + + name = obj->value.name; + obj ++; + +#ifdef DEBUG + DEBUG_printf(("setpagedevice: /%s ", name)); + DEBUG_object(obj); +#endif /* DEBUG */ + + /* + * Then grab the value... + */ + + if (!strcmp(name, "MediaClass") && obj->type == CUPS_PS_STRING) + strlcpy(h->MediaClass, obj->value.string, sizeof(h->MediaClass)); + else if (!strcmp(name, "MediaColor") && obj->type == CUPS_PS_STRING) + strlcpy(h->MediaColor, obj->value.string, sizeof(h->MediaColor)); + else if (!strcmp(name, "MediaType") && obj->type == CUPS_PS_STRING) + strlcpy(h->MediaType, obj->value.string, sizeof(h->MediaType)); + else if (!strcmp(name, "OutputType") && obj->type == CUPS_PS_STRING) + strlcpy(h->OutputType, obj->value.string, sizeof(h->OutputType)); + else if (!strcmp(name, "AdvanceDistance") && obj->type == CUPS_PS_NUMBER) + h->AdvanceDistance = (unsigned)obj->value.number; + else if (!strcmp(name, "AdvanceMedia") && obj->type == CUPS_PS_NUMBER) + h->AdvanceMedia = (unsigned)obj->value.number; + else if (!strcmp(name, "Collate") && obj->type == CUPS_PS_BOOLEAN) + h->Collate = (unsigned)obj->value.boolean; + else if (!strcmp(name, "CutMedia") && obj->type == CUPS_PS_NUMBER) + h->CutMedia = (cups_cut_t)(unsigned)obj->value.number; + else if (!strcmp(name, "Duplex") && obj->type == CUPS_PS_BOOLEAN) + h->Duplex = (unsigned)obj->value.boolean; + else if (!strcmp(name, "HWResolution") && obj->type == CUPS_PS_START_ARRAY) + { + if (obj[1].type == CUPS_PS_NUMBER && obj[2].type == CUPS_PS_NUMBER && + obj[3].type == CUPS_PS_END_ARRAY) + { + h->HWResolution[0] = (unsigned)obj[1].value.number; + h->HWResolution[1] = (unsigned)obj[2].value.number; + obj += 3; + } + else + return (-1); + } + else if (!strcmp(name, "InsertSheet") && obj->type == CUPS_PS_BOOLEAN) + h->InsertSheet = (unsigned)obj->value.boolean; + else if (!strcmp(name, "Jog") && obj->type == CUPS_PS_NUMBER) + h->Jog = (unsigned)obj->value.number; + else if (!strcmp(name, "LeadingEdge") && obj->type == CUPS_PS_NUMBER) + h->LeadingEdge = (unsigned)obj->value.number; + else if (!strcmp(name, "ManualFeed") && obj->type == CUPS_PS_BOOLEAN) + h->ManualFeed = (unsigned)obj->value.boolean; + else if ((!strcmp(name, "cupsMediaPosition") || + !strcmp(name, "MediaPosition")) && obj->type == CUPS_PS_NUMBER) + { + /* + * cupsMediaPosition is supported for backwards compatibility only. + * We added it back in the Ghostscript 5.50 days to work around a + * bug in Ghostscript WRT handling of MediaPosition and setpagedevice. + * + * All new development should set MediaPosition... + */ + + h->MediaPosition = (unsigned)obj->value.number; + } + else if (!strcmp(name, "MediaWeight") && obj->type == CUPS_PS_NUMBER) + h->MediaWeight = (unsigned)obj->value.number; + else if (!strcmp(name, "MirrorPrint") && obj->type == CUPS_PS_BOOLEAN) + h->MirrorPrint = (unsigned)obj->value.boolean; + else if (!strcmp(name, "NegativePrint") && obj->type == CUPS_PS_BOOLEAN) + h->NegativePrint = (unsigned)obj->value.boolean; + else if (!strcmp(name, "NumCopies") && obj->type == CUPS_PS_NUMBER) + h->NumCopies = (unsigned)obj->value.number; + else if (!strcmp(name, "Orientation") && obj->type == CUPS_PS_NUMBER) + h->Orientation = (unsigned)obj->value.number; + else if (!strcmp(name, "OutputFaceUp") && obj->type == CUPS_PS_BOOLEAN) + h->OutputFaceUp = (unsigned)obj->value.boolean; + else if (!strcmp(name, "PageSize") && obj->type == CUPS_PS_START_ARRAY) + { + if (obj[1].type == CUPS_PS_NUMBER && obj[2].type == CUPS_PS_NUMBER && + obj[3].type == CUPS_PS_END_ARRAY) + { + h->cupsPageSize[0] = (float)obj[1].value.number; + h->cupsPageSize[1] = (float)obj[2].value.number; + + h->PageSize[0] = (unsigned)obj[1].value.number; + h->PageSize[1] = (unsigned)obj[2].value.number; + + obj += 3; + } + else + return (-1); + } + else if (!strcmp(name, "Separations") && obj->type == CUPS_PS_BOOLEAN) + h->Separations = (unsigned)obj->value.boolean; + else if (!strcmp(name, "TraySwitch") && obj->type == CUPS_PS_BOOLEAN) + h->TraySwitch = (unsigned)obj->value.boolean; + else if (!strcmp(name, "Tumble") && obj->type == CUPS_PS_BOOLEAN) + h->Tumble = (unsigned)obj->value.boolean; + else if (!strcmp(name, "cupsMediaType") && obj->type == CUPS_PS_NUMBER) + h->cupsMediaType = (unsigned)obj->value.number; + else if (!strcmp(name, "cupsBitsPerColor") && obj->type == CUPS_PS_NUMBER) + h->cupsBitsPerColor = (unsigned)obj->value.number; + else if (!strcmp(name, "cupsPreferredBitsPerColor") && + obj->type == CUPS_PS_NUMBER) + *preferred_bits = (int)obj->value.number; + else if (!strcmp(name, "cupsColorOrder") && obj->type == CUPS_PS_NUMBER) + h->cupsColorOrder = (cups_order_t)(unsigned)obj->value.number; + else if (!strcmp(name, "cupsColorSpace") && obj->type == CUPS_PS_NUMBER) + h->cupsColorSpace = (cups_cspace_t)(unsigned)obj->value.number; + else if (!strcmp(name, "cupsCompression") && obj->type == CUPS_PS_NUMBER) + h->cupsCompression = (unsigned)obj->value.number; + else if (!strcmp(name, "cupsRowCount") && obj->type == CUPS_PS_NUMBER) + h->cupsRowCount = (unsigned)obj->value.number; + else if (!strcmp(name, "cupsRowFeed") && obj->type == CUPS_PS_NUMBER) + h->cupsRowFeed = (unsigned)obj->value.number; + else if (!strcmp(name, "cupsRowStep") && obj->type == CUPS_PS_NUMBER) + h->cupsRowStep = (unsigned)obj->value.number; + else if (!strcmp(name, "cupsBorderlessScalingFactor") && + obj->type == CUPS_PS_NUMBER) + h->cupsBorderlessScalingFactor = (float)obj->value.number; + else if (!strncmp(name, "cupsInteger", 11) && obj->type == CUPS_PS_NUMBER) + { + if ((i = atoi(name + 11)) < 0 || i > 15) + return (-1); + + h->cupsInteger[i] = (unsigned)obj->value.number; + } + else if (!strncmp(name, "cupsReal", 8) && obj->type == CUPS_PS_NUMBER) + { + if ((i = atoi(name + 8)) < 0 || i > 15) + return (-1); + + h->cupsReal[i] = (float)obj->value.number; + } + else if (!strncmp(name, "cupsString", 10) && obj->type == CUPS_PS_STRING) + { + if ((i = atoi(name + 10)) < 0 || i > 15) + return (-1); + + strlcpy(h->cupsString[i], obj->value.string, sizeof(h->cupsString[i])); + } + else if (!strcmp(name, "cupsMarkerType") && obj->type == CUPS_PS_STRING) + strlcpy(h->cupsMarkerType, obj->value.string, sizeof(h->cupsMarkerType)); + else if (!strcmp(name, "cupsPageSizeName") && obj->type == CUPS_PS_STRING) + strlcpy(h->cupsPageSizeName, obj->value.string, + sizeof(h->cupsPageSizeName)); + else if (!strcmp(name, "cupsRenderingIntent") && + obj->type == CUPS_PS_STRING) + strlcpy(h->cupsRenderingIntent, obj->value.string, + sizeof(h->cupsRenderingIntent)); + else + { + /* + * Ignore unknown name+value... + */ + + DEBUG_printf((" Unknown name (\"%s\") or value...\n", name)); + + while (obj[1].type != CUPS_PS_NAME && obj < end) + obj ++; + } + } + + return (0); +} + + +#ifdef DEBUG +/* + * 'DEBUG_object()' - Print an object's value... + */ + +static void +DEBUG_object(_cups_ps_obj_t *obj) /* I - Object to print */ +{ + switch (obj->type) + { + case CUPS_PS_NAME : + DEBUG_printf(("/%s\n", obj->value.name)); + break; + + case CUPS_PS_NUMBER : + DEBUG_printf(("%g\n", obj->value.number)); + break; + + case CUPS_PS_STRING : + DEBUG_printf(("(%s)\n", obj->value.string)); + break; + + case CUPS_PS_BOOLEAN : + if (obj->value.boolean) + DEBUG_puts("true"); + else + DEBUG_puts("false"); + break; + + case CUPS_PS_NULL : + DEBUG_puts("null"); + break; + + case CUPS_PS_START_ARRAY : + DEBUG_puts("["); + break; + + case CUPS_PS_END_ARRAY : + DEBUG_puts("]"); + break; + + case CUPS_PS_START_DICT : + DEBUG_puts("<<"); + break; + + case CUPS_PS_END_DICT : + DEBUG_puts(">>"); + break; + + case CUPS_PS_START_PROC : + DEBUG_puts("{"); + break; + + case CUPS_PS_END_PROC : + DEBUG_puts("}"); + break; + + case CUPS_PS_CLEARTOMARK : + DEBUG_puts("--cleartomark--"); + break; + + case CUPS_PS_COPY : + DEBUG_puts("--copy--"); + break; + + case CUPS_PS_DUP : + DEBUG_puts("--dup--"); + break; + + case CUPS_PS_INDEX : + DEBUG_puts("--index--"); + break; + + case CUPS_PS_POP : + DEBUG_puts("--pop--"); + break; + + case CUPS_PS_ROLL : + DEBUG_puts("--roll--"); + break; + + case CUPS_PS_SETPAGEDEVICE : + DEBUG_puts("--setpagedevice--"); + break; + + case CUPS_PS_STOPPED : + DEBUG_puts("--stopped--"); + break; + + case CUPS_PS_OTHER : + DEBUG_printf(("--%s--\n", obj->value.other)); + break; + } +} + + +/* + * 'DEBUG_stack()' - Print a stack... + */ + +static void +DEBUG_stack(_cups_ps_stack_t *st) /* I - Stack */ +{ + int c; /* Looping var */ + _cups_ps_obj_t *obj; /* Current object on stack */ + + + for (obj = st->objs, c = st->num_objs; c > 0; c --, obj ++) + DEBUG_object(obj); +} +#endif /* DEBUG */ + + +/* + * End of "$Id$". + */ diff --git a/filter/libcupsimage2.def b/filter/libcupsimage2.def new file mode 100644 index 0000000000..3c20284f5e --- /dev/null +++ b/filter/libcupsimage2.def @@ -0,0 +1,14 @@ +LIBRARY libcupsimage2 +VERSION 2.3 +EXPORTS +cupsRasterClose +cupsRasterErrorString +cupsRasterInterpretPPD +cupsRasterOpen +cupsRasterOpenIO +cupsRasterReadHeader +cupsRasterReadHeader2 +cupsRasterReadPixels +cupsRasterWriteHeader +cupsRasterWriteHeader2 +cupsRasterWritePixels diff --git a/filter/libcupsimage_s.exp b/filter/libcupsimage_s.exp new file mode 100644 index 0000000000..57f4259f99 --- /dev/null +++ b/filter/libcupsimage_s.exp @@ -0,0 +1,16 @@ +_cupsImagePutCol +_cupsImagePutRow +_cupsImageReadBMP +_cupsImageReadGIF +_cupsImageReadJPEG +_cupsImageReadPIX +_cupsImageReadPNG +_cupsImageReadPNM +_cupsImageReadPhotoCD +_cupsImageReadSGI +_cupsImageReadSunRaster +_cupsImageReadTIFF +_cupsImageZoomDelete +_cupsImageZoomFill +_cupsImageZoomNew +_cupsRasterExecPS diff --git a/filter/postscript-driver.header b/filter/postscript-driver.header new file mode 100644 index 0000000000..a0fedcf648 --- /dev/null +++ b/filter/postscript-driver.header @@ -0,0 +1,32 @@ + + +

Developing PostScript Printer Drivers

+ +

This document describes how to develop printer drivers for PostScript printers. Topics include: printer driver basics, creating new PPD files, importing existing PPD files, using custom filters, implementing color management, and adding Mac OS X features.

+ +
+ + + + + + +
See AlsoProgramming: Developing Raster Printer Drivers
+ Programming: Filter and Backend Programming
+ Programming: Introduction to the PPD Compiler
+ Programming: Raster API
+ References: PPD Compiler Driver Information File Reference
+ Specifications: CUPS PPD Extensions
diff --git a/filter/postscript-driver.shtml b/filter/postscript-driver.shtml new file mode 100644 index 0000000000..439744f0fa --- /dev/null +++ b/filter/postscript-driver.shtml @@ -0,0 +1,276 @@ +

Printer Driver Basics

+ +

A CUPS PostScript printer driver consists of a PostScript Printer Description (PPD) file that describes the features and capabilities of the device, zero or more filter programs that prepare print data for the device, and zero or more support files for color management, online help, and so forth. The PPD file includes references to all of the filters and support files used by the driver.

+ +

Every time a user prints something the scheduler program, cupsd(8), determines the format of the print job and the programs required to convert that job into something the printer understands. CUPS includes filter programs for many common formats, for example to convert Portable Document Format (PDF) files into device-independent PostScript, and then from device-independent PostScript to device-dependent PostScript. Figure 1 shows the data flow of a typical print job.

+ +
+ + +
Figure 1: PostScript Filter Chain
PostScript Filter Chain
+ +

The optional PostScript filter can be provided to add printer-specific commands to the PostScript output that cannot be represented in the PPD file or to reorganize the output for special printer features. Typically this is used to support advanced job management or finishing functions on the printer. CUPS includes a generic PostScript filter that handles all PPD-defined commands.

+ +

The optional port monitor handles interface-specific protocol or encoding issues. For example, many PostScript printers support the Binary Communications Protocol (BCP) and Tagged Binary Communications Protocol (TBCP) to allow applications to print 8-bit ("binary") PostScript jobs. CUPS includes port monitors for BCP and TBCP, and you can supply your own port monitors as needed.

+ +

The backend handles communications with the printer, sending print data from the last filter to the printer and relaying back-channel data from the printer to the upstream filters. CUPS includes backend programs for common direct-connect interfaces and network protocols, and you can provide your own backend to support custom interfaces and protocols.

+ +

The scheduler also supports a special "command" file format for sending maintenance commands and status queries to a printer or printer driver. Command print jobs typically use a single command filter program defined in the PPD file to generate the appropriate printer commands and handle any responses from the printer. Figure 2 shows the data flow of a typical command job.

+ +
+ + +
Figure 2: Command Filter Chain
Command Filter Chain
+ +

PostScript printer drivers typically do not require their own command filter since CUPS includes a generic PostScript command filter that supports all of the standard functions using PPD-defined commands.

+ + +

Creating New PPD Files

+ +

We recommend using the CUPS PPD compiler, ppdc(1), to create new PPD files since it manages many of the tedious (and error-prone!) details of paper sizes and localization for you. It also allows you to easily support multiple devices from a single source file. For more information see the "Introduction to the PPD Compiler" document. Listing 1 shows a driver information file for a black-and-white PostScript printer.

+ +

Listing 1: "examples/postscript.drv"

+ +
+// Include standard font and media definitions
+#include <font.defs>
+#include <media.defs>
+
+// Specify this is a PostScript printer driver
+DriverType ps
+
+// List the fonts that are supported, in this case all standard fonts
+Font *
+
+// Manufacturer, model name, and version
+Manufacturer "Foo"
+ModelName "Foo LaserProofer 2000"
+Version 1.0
+
+// PostScript printer attributes
+Attribute DefaultColorSpace "" Gray
+Attribute LandscapeOrientation "" Minus90
+Attribute LanguageLevel "" "3"
+Attribute Product "" "(Foo LaserProofer 2000)"
+Attribute PSVersion "" "(3010) 0"
+Attribute TTRasterizer "" Type42
+
+// Supported page sizes
+*MediaSize Letter
+MediaSize Legal
+MediaSize A4
+
+// Query command for page size
+Attribute "?PageSize" "" "
+      save
+      currentpagedevice /PageSize get aload pop
+      2 copy gt {exch} if (Unknown)
+      23 dict
+              dup [612 792] (Letter) put
+              dup [612 1008] (Legal) put
+              dup [595 842] (A4) put
+              {exch aload pop 4 index sub abs 5 le exch 
+               5 index sub abs 5 le and
+              {exch pop exit} {pop} ifelse
+      } bind forall = flush pop pop
+      restore"
+
+// Specify the name of the PPD file we want to generate
+PCFileName "fooproof.ppd"
+
+ +

Required Attributes

+ +

PostScript drivers require the attributes listed in Table 1. If not specified, the defaults for CUPS drivers are used. A typical PostScript driver information file would include the following attributes:

+ +
+Attribute DefaultColorSpace "" Gray
+Attribute LandscapeOrientation "" Minus90
+Attribute LanguageLevel "" "3"
+Attribute Product "" "(Foo LaserProofer 2000)"
+Attribute PSVersion "" "(3010) 0"
+Attribute TTRasterizer "" Type42
+
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 1: Required PostScript Printer Driver Attributes
AttributeDescription
DefaultColorSpaceThe default colorspace: + Gray, RGB, CMY, or + CMYK. If not specified, then RGB is + assumed.
LandscapeOrientationThe preferred landscape + orientation: Plus90, Minus90, or + Any. If not specified, Plus90 is + assumed.
LanguageLevelThe PostScript language + level supported by the device: 1, 2, or 3. If not + specified, 2 is assumed.
ProductThe string returned by + the PostScript product operator, which + must include parenthesis to conform with + PostScript syntax rules for strings. Multiple + Product attributes may be specified to support + multiple products with the same PPD file. If not + specified, "(ESP Ghostscript)" and "(GNU Ghostscript)" + are assumed.
PSVersionThe PostScript + interpreter version numbers as returned by the + version and revision operators. The + required format is "(version) revision". Multiple + PSVersion attributes may be specified to + support multiple interpreter version numbers. If not + specified, "(3010) 705" and "(3010) 707" are + assumed.
TTRasterizerThe type of TrueType + font rasterizer supported by the device, if any. The + supported values are None, Accept68k, + Type42, and TrueImage. If not + specified, None is assumed.
+ +

Query Commands

+ +

Most PostScript printer PPD files include query commands (?PageSize, etc.) that allow applications to query the printer for its current settings and configuration. Query commands are included in driver information files as attributes. For example, the example in Listing 1 uses the following definition for the PageSize query command:

+ +
+Attribute "?PageSize" "" "
+      save
+      currentpagedevice /PageSize get aload pop
+      2 copy gt {exch} if (Unknown)
+      23 dict
+              dup [612 792] (Letter) put
+              dup [612 1008] (Legal) put
+              dup [595 842] (A4) put
+              {exch aload pop 4 index sub abs 5 le exch 
+               5 index sub abs 5 le and
+              {exch pop exit} {pop} ifelse
+      } bind forall = flush pop pop
+      restore"
+
+ +

Query commands can span multiple lines, however no single line may contain more than 255 characters.

+ +

Importing Existing PPD Files

+ +

CUPS includes a utility called ppdi(1) +which allows you to import existing PPD files into the driver information file +format used by the PPD compiler ppdc(1). Once +imported, you can modify, localize, and regenerate the PPD files easily. Type +the following command to import the PPD file mydevice.ppd into the +driver information file mydevice.drv:

+ +
+ppdi -o mydevice.drv mydevice.ppd
+
+ +

If you have a whole directory of PPD files that you would like to import, +you can list multiple filenames or use shell wildcards to import more than one +PPD file on the command-line:

+ +
+ppdi -o mydevice.drv mydevice1.ppd mydevice2.ppd
+ppdi -o mydevice.drv *.ppd
+
+ +

If the driver information file already exists, the new PPD +file entries are appended to the end of the file. Each PPD file +is placed in its own group of curly braces within the driver +information file.

+ + +

Using Custom Filters

+ +

Normally a PostScript printer driver will not utilize any additional print filters. For drivers that provide additional filters such as a CUPS command file filter for doing printer maintenance, you must also list the following Filter directive to handle printing PostScript files:

+ +
+Filter application/vnd.cups-postscript 0 -
+
+ +

Custom Command Filters

+ +

The application/vnd.cups-command file type is used for CUPS command files. Use the following Filter directive to handle CUPS command files:

+ +
+Filter application/vnd.cups-command 100 /path/to/command/filter
+
+ +

To use the standard PostScript command filter, specify commandtops as the path to the command filter.

+ +

Custom PDF Filters

+ +

The application/pdf file type is used for unfiltered PDF files while the application/vnd.cups-pdf file type is used for filtered PDF files. Use the following Filter directive to handle filtered PDF files:

+ +
+Filter application/vnd.cups-pdf 100 /path/to/pdf/filter
+
+ +

For unfiltered PDF files, use:

+ +
+Filter application/pdf 100 /path/to/pdf/filter
+
+ +

Custom PDF filters that accept filtered data do not need to perform number-up processing and other types of page imposition, while those that accept unfiltered data MUST do the number-up processing themselves.

+ +

Custom PostScript Filters

+ +

The application/vnd.cups-postscript file type is used for filtered PostScript files. Use the following Filter directive to handle PostScript files:

+ +
+Filter application/vnd.cups-postscript 100 /path/to/postscript/filter
+
+ + +

Implementing Color Management

+ +

CUPS uses ICC color profiles to provide more accurate color reproduction. The cupsICCProfile attribute defines the color profiles that are available for a given printer, for example:

+ +
+Attribute cupsICCProfile "ColorModel.MediaType.Resolution/Description" /path/to/ICC/profile
+
+ +

where "ColorModel.MediaType.Resolution" defines a selector based on the corresponding option selections. A simple driver might only define profiles for the color models that are supported, for example a printer supporting Gray and RGB might use:

+ +
+Attribute cupsICCProfile "Gray../Grayscale Profile" /path/to/ICC/gray-profile
+Attribute cupsICCProfile "RGB../Full Color Profile" /path/to/ICC/rgb-profile
+
+ +

The options used for profile selection can be customized using the cupsICCQualifier2 and cupsICCQualifier3 attributes.

+ + +

Adding Mac OS X Features

+ +

Mac OS X printer drivers can provide additional attributes to specify additional option panes in the print dialog, an image of the printer, a help book, and option presets for the driver software:

+ +
+Attribute APDialogExtension "" /Library/Printers/Vendor/filename.plugin
+Attribute APHelpBook "" /Library/Printers/Vendor/filename.bundle
+Attribute APPrinterIconPath "" /Library/Printers/Vendor/filename.icns
+Attribute APPrinterPreset "name/text" "*option choice ..."
+
diff --git a/filter/ppd-compiler.header b/filter/ppd-compiler.header new file mode 100644 index 0000000000..5a36477f9e --- /dev/null +++ b/filter/ppd-compiler.header @@ -0,0 +1,34 @@ + + +

Introduction to the PPD Compiler

+ +

This document describes how to use the CUPS PostScript Printer Description +(PPD) file compiler. The PPD compiler generates PPD files from simple text files +that describe the features and capabilities of one or more printers.

+ +
+ + + + + + +
See AlsoProgramming: Developing Raster Printer Drivers
+ Programming: Developing PostScript Printer Drivers
+ Programming: Filter and Backend Programming
+ Programming: Raster API
+ References: PPD Compiler Driver Information File Reference
+ Specifications: CUPS PPD Extensions
diff --git a/filter/ppd-compiler.shtml b/filter/ppd-compiler.shtml new file mode 100644 index 0000000000..c98b95cee6 --- /dev/null +++ b/filter/ppd-compiler.shtml @@ -0,0 +1,883 @@ +

The Basics

+ +

The PPD compiler, ppdc(1), is a +simple command-line tool that takes a single driver information file, +which by convention uses the extension .drv, and produces one or more +PPD files that may be distributed with your printer drivers for use with CUPS. +For example, you would run the following command to create the English language +PPD files defined by the driver information file mydrivers.drv:

+ +
+ppdc mydrivers.drv
+
+ +

The PPD files are placed in a subdirectory called +ppd. The -d option is used to put the PPD +files in a different location, for example:

+ +
+ppdc -d myppds mydrivers.drv
+
+ +

places the PPD files in a subdirectory named +myppds. Finally, use the -l option to +specify the language localization for the PPD files that are +created, for example:

+ +
+ppdc -d myppds/de -l de mydrivers.drv
+ppdc -d myppds/en -l en mydrivers.drv
+ppdc -d myppds/es -l es mydrivers.drv
+ppdc -d myppds/fr -l fr mydrivers.drv
+ppdc -d myppds/it -l it mydrivers.drv
+
+ +

creates PPD files in German (de), English (en), Spanish (es), +French (fr), and Italian (it) in the corresponding +subdirectories. Specify multiple languages (separated by commas) to produce +"globalized" PPD files:

+ +
+ppdc -d myppds -l de,en,es,fr,it mydrivers.drv
+
+ + +

Driver Information Files

+ +

The driver information files accepted by the PPD compiler are +plain text files that define the various attributes and options +that are included in the PPD files that are generated. A driver +information file can define the information for one or more printers and +their corresponding PPD files.

+ +

Listing 1: "examples/minimum.drv"

+ +
+// Include standard font and media definitions
+#include <font.defs>
+#include <media.defs>
+
+// List the fonts that are supported, in this case all standard fonts...
+Font *
+
+// Manufacturer, model name, and version
+Manufacturer "Foo"
+ModelName "FooJet 2000"
+Version 1.0
+
+// Each filter provided by the driver...
+Filter application/vnd.cups-raster 100 rastertofoo
+
+// Supported page sizes
+*MediaSize Letter
+MediaSize A4
+
+// Supported resolutions
+*Resolution k 8 0 0 0 "600dpi/600 DPI"
+
+// Specify the name of the PPD file we want to generate...
+PCFileName "foojet2k.ppd"
+
+ + +

A Simple Example

+ +

The example in Listing 1 shows a driver information +file which defines the minimum required attributes to provide a valid PPD file. +The first part of the file includes standard definition files for fonts and +media sizes:

+ +
+#include <font.defs>
+#include <media.defs>
+
+ +

The #include directive works just like the C/C++ include directive; +files included using the angle brackets (<filename>) are found +in any of the standard include directories and files included using quotes +("filename") are found in the same directory as the source or include +file. The <font.defs> include file defines the standard fonts +which are included with GPL Ghostscript and the Apple PDF RIP, while the +<media.defs> include file defines the standard media sizes +listed in Appendix B of the Adobe PostScript Printer Description File Format +Specification.

+ +

CUPS provides several other standard include files:

+ +
    + +
  • <epson.h> - Defines all of the rastertoepson driver + constants.
  • + +
  • <escp.h> - Defines all of the rastertoescpx driver + constants.
  • + +
  • <hp.h> - Defines all of the rastertohp driver + constants.
  • + +
  • <label.h> - Defines all of the rastertolabel driver + constants.
  • + +
  • <pcl.h> - Defines all of the rastertopclx driver + constants.
  • + +
  • <raster.defs> - Defines all of the CUPS raster format + constants.
  • + +
+ +

Next we list all of the fonts that are available in the driver; for CUPS +raster drivers, the following line is all that is usually supplied:

+ +
+Font *
+
+ +

The Font directive specifies the name of a single font or the +asterisk to specify all fonts. For example, you would use the following line to +define an additional bar code font that you are supplying with your printer +driver:

+ +
+//   name         encoding  version  charset  status
+Font Barcode-Foo  Special   "(1.0)"  Special  ROM
+
+ +

The name of the font is Barcode-Foo. Since it is not a standard +text font, the encoding and charset name Special is used. The version +number is 1.0 and the status (where the font is located) is +ROM to indicate that the font does not need to be embedded in +documents that use the font for this printer.

+ +

Third comes the manufacturer, model name, and version number information +strings:

+ +
+Manufacturer "Foo"
+ModelName "FooJet 2000"
+Version 1.0
+
+ +

These strings are used when the user (or auto-configuration program) selects +the printer driver for a newly connected device.

+ +

The list of filters comes after the information strings; for the example in +Listing 1, we have a single filter that takes CUPS +raster data:

+ +
+Filter application/vnd.cups-raster 100 rastertofoo
+
+ +

Each filter specified in the driver information file is the equivalent of a +printer driver for that format; if a user submits a print job in a different +format, CUPS figures out the sequence of commands that will produce a supported +format for the least relative cost.

+ +

Once we have defined the driver information we specify the supported options. +For the example driver we support a single resolution of 600 dots per inch and +two media sizes, A4 and Letter:

+ +
+*MediaSize Letter
+MediaSize A4
+
+*Resolution k 8 0 0 0 "600dpi/600 DPI"
+
+ +

The asterisk in front of the MediaSize and Resolution +directives specify that those option choices are the default. The +MediaSize directive is followed by a media size name which is normally +defined in the <media.defs> file and corresponds to a standard +Adobe media size name. If the default media size is Letter, the PPD +compiler will override it to be A4 for non-English localizations for +you automatically.

+ +

The Resolution directive accepts several values after it as +follows:

+ +
    + +
  1. Colorspace for this resolution, if any. In the example file, the + colorspace k is used which corresponds to black. For printer + drivers that support color printing, this field is usually specified as + "-" for "no change".
  2. + +
  3. Bits per color. In the example file, we define 8 bits per color, for + a continuous-tone grayscale output. All versions of CUPS support 1 and + 8 bits per color. CUPS 1.2 and higher (Mac OS X 10.5 and higher) also + supports 16 bits per color.
  4. + +
  5. Rows per band. In the example file, we define 0 rows per band to + indicate that our printer driver does not process the page in + bands.
  6. + +
  7. Row feed. In the example, we define the feed value to be 0 to + indicate that our printer driver does not interleave the output.
  8. + +
  9. Row step. In the example, we define the step value to be 0 to + indicate that our printer driver does not interleave the output. This + value normally indicates the spacing between the nozzles of an inkjet + printer - when combined with the previous two values, it informs the + driver how to stagger the output on the page to produce interleaved + lines on the page for higher-resolution output.
  10. + +
  11. Choice name and text. In the example, we define the choice name and + text to be "600dpi/600 DPI". The name and text are separated by + slash (/) character; if no text is specified, then the name is + used as the text. The PPD compiler parses the name to determine the + actual resolution; the name can be of the form + RESOLUTIONdpi for resolutions that are equal + horizontally and vertically or HRESxVRESdpi for + isometric resolutions. Only integer resolution values are supported, so + a resolution name of 300dpi is valid while 300.1dpi is + not.
  12. + +
+ +

Finally, the PCFileName directive specifies that the named PPD file +should be written for the current driver definitions:

+ +
+PCFileName "foojet2k.ppd"
+
+ +

The filename follows the directive and must conform to the Adobe +filename requirements in the Adobe Postscript Printer Description File Format +Specification. Specifically, the filename may not exceed 8 characters followed +by the extension .ppd. The FileName directive can be used to +specify longer filenames:

+ +
+FileName "FooJet 2000"
+
+ + +

Grouping and Inheritance

+ +

The previous example created a single PPD file. Driver information files can +also define multiple printers by using the PPD compiler grouping functionality. +Directives are grouped using the curly braces ({ and }) and +every group that uses the PCFileName or FileName directives +produces a PPD file with that name. Listing 2 shows a +variation of the original example that uses two groups to define two printers +that share the same printer driver filter but provide two different resolution +options.

+ +

Listing 2: "examples/grouping.drv"

+ +
+
+// Include standard font and media definitions
+#include <font.defs>
+#include <media.defs>
+
+// List the fonts that are supported, in this case all standard fonts...
+Font *
+
+// Manufacturer and version
+Manufacturer "Foo"
+Version 1.0
+
+// Each filter provided by the driver...
+Filter application/vnd.cups-raster 100 rastertofoo
+
+// Supported page sizes
+*MediaSize Letter
+MediaSize A4
+
+{
+  // Supported resolutions
+  *Resolution k 8 0 0 0 "600dpi/600 DPI"
+
+  // Specify the model name and filename...
+  ModelName "FooJet 2000"
+  PCFileName "foojet2k.ppd"
+}
+
+{
+  // Supported resolutions
+  *Resolution k 8 0 0 0 "1200dpi/1200 DPI"
+
+  // Specify the model name and filename...
+  ModelName "FooJet 2001"
+  PCFileName "foojt2k1.ppd"
+}
+
+ +

The second example is essentially the same as the first, except that each +printer model is defined inside of a pair of curly braces. For example, the +first printer is defined using:

+ +
+{
+  // Supported resolutions
+  *Resolution k 8 0 0 0 "600dpi/600 DPI"
+
+  // Specify the model name and filename...
+  ModelName "FooJet 2000"
+  PCFileName "foojet2k.ppd"
+}
+
+ +

The printer inherits all of the definitions from the parent group (the +top part of the file) and adds the additional definitions inside the curly +braces for that printer driver. When we define the second group, it also +inherits the same definitions from the parent group but none of the +definitions from the first driver. Groups can be nested to any number of levels +to support variations of similar models without duplication of information.

+ + +

Color Support

+ +

For printer drivers that support color printing, the +ColorDevice and ColorModel directives should be +used to tell the printing system that color output is desired +and in what formats. Listing 3 shows a +variation of the previous example which includes a color printer +that supports printing at 300 and 600 DPI.

+ +

The key changes are the addition of the ColorDevice +directive:

+ +
+ColorDevice true
+
+ +

which tells the printing system that the printer supports +color printing, and the ColorModel directives:

+ +
+ColorModel Gray/Grayscale w chunky 0
+*ColorModel RGB/Color rgb chunky 0
+
+ +

which tell the printing system which colorspaces are supported by the printer +driver for color printing. Each of the ColorModel directives is +followed by the option name and text (Gray/Grayscale and +RGB/Color), the colorspace name (w and rgb), the +color organization (chunky), and the compression mode number +(0) to be passed to the driver. The option name can be any of the +standard Adobe ColorModel names:

+ +
    + +
  • Gray - Grayscale output. + +
  • RGB - Color output, typically using the RGB + colorspace, but without a separate black channel. + +
  • CMYK - Color output with a separate black + channel. + +
+ +

Custom names can be used, however it is recommended that you use your vendor +prefix for any custom names, for example "fooName".

+ +

The colorspace name can be any of the following universally supported +colorspaces:

+ +
    +
  • w - Luminance
  • + +
  • rgb - Red, green, blue
  • + +
  • k - Black
  • + +
  • cmy - Cyan, magenta, yellow
  • + +
  • cmyk - Cyan, magenta, yellow, black
  • + +
+ +

The color organization can be any of the following values:

+ +
    + +
  • chunky - Color values are passed together on a line + as RGB RGB RGB RGB
  • + +
  • banded - Color values are passed separately + on a line as RRRR GGGG BBBB; not supported by the Apple + RIP filters
  • + +
  • planar - Color values are passed separately + on a page as RRRR RRRR RRRR ... GGGG GGGG GGGG ... BBBB + BBBB BBBB; not supported by the Apple RIP filters
  • + +
+ +

The compression mode value is passed to the driver in the +cupsCompression attribute. It is traditionally used to select an +appropriate compression mode for the color model but can be used for any +purpose, such as specifying a photo mode vs. standard mode.

+ +

Listing 3: "examples/color.drv"

+ +
+
+// Include standard font and media definitions
+#include <font.defs>
+#include <media.defs>
+
+// List the fonts that are supported, in this case all standard fonts...
+Font *
+
+// Manufacturer and version
+Manufacturer "Foo"
+Version 1.0
+
+// Each filter provided by the driver...
+Filter application/vnd.cups-raster 100 rastertofoo
+
+// Supported page sizes
+*MediaSize Letter
+MediaSize A4
+
+{
+  // Supported resolutions
+  *Resolution k 8 0 0 0 "600dpi/600 DPI"
+
+  // Specify the model name and filename...
+  ModelName "FooJet 2000"
+  PCFileName "foojet2k.ppd"
+}
+
+{
+  // Supports color printing
+  ColorDevice true
+
+  // Supported colorspaces
+  ColorModel Gray/Grayscale w chunky 0
+  *ColorModel RGB/Color rgb chunky 0
+
+  // Supported resolutions
+  *Resolution - 8 0 0 0 "300dpi/300 DPI"
+  Resolution - 8 0 0 0 "600dpi/600 DPI"
+
+  // Specify the model name and filename...
+  ModelName "FooJet Color"
+  PCFileName "foojetco.ppd"
+}
+
+ + +

Defining Custom Options and Option Groups

+ +

The Group, Option, and Choice +directives are used to define or select a group, option, or +choice. Listing 4 shows a variation of +the first example that provides two custom options in a group +named "Footasm".

+ +

Listing 4: "examples/custom.drv"

+ +
+
+// Include standard font and media definitions
+#include <font.defs>
+#include <media.defs>
+
+// List the fonts that are supported, in this case all standard fonts...
+Font *
+
+// Manufacturer, model name, and version
+Manufacturer "Foo"
+ModelName "FooJet 2000"
+Version 1.0
+
+// Each filter provided by the driver...
+Filter application/vnd.cups-raster 100 rastertofoo
+
+// Supported page sizes
+*MediaSize Letter
+MediaSize A4
+
+// Supported resolutions
+*Resolution k 8 0 0 0 "600dpi/600 DPI"
+
+// Option Group
+Group "Footasm"
+
+  // Boolean option
+  Option "fooEnhance/Resolution Enhancement" Boolean AnySetup 10
+    *Choice True/Yes "<</cupsCompression 1>>setpagedevice"
+    Choice False/No "<</cupsCompression 0>>setpagedevice"
+
+  // Multiple choice option
+  Option "fooOutputType/Output Quality" PickOne AnySetup 10
+    *Choice "Auto/Automatic Selection"
+            "<</OutputType(Auto)>>setpagedevice""
+    Choice "Text/Optimize for Text"
+            "<</OutputType(Text)>>setpagedevice""
+    Choice "Graph/Optimize for Graphics"
+            "<</OutputType(Graph)>>setpagedevice""
+    Choice "Photo/Optimize for Photos"
+            "<</OutputType(Photo)>>setpagedevice""
+
+// Specify the name of the PPD file we want to generate...
+PCFileName "foojet2k.ppd"
+
+ +

The custom group is introduced by the Group +directive which is followed by the name and optionally text for +the user:

+ +
+Group "Footasm/Footastic Options"
+
+ +

The group name must conform to the PPD specification and +cannot exceed 40 characters in length. If you specify user text, +it cannot exceed 80 characters in length. The groups +General, Extra, and +InstallableOptions are predefined by CUPS; the general +and extra groups are filled by the UI options defined by the PPD +specification. The InstallableOptions group is reserved +for options that define whether accessories for the printer +(duplexer unit, finisher, stapler, etc.) are installed.

+ +

Once the group is specified, the Option directive is +used to introduce a new option:

+ +
+Option "fooEnhance/Resolution Enhancement" Boolean AnySetup 10
+
+ +

The directive is followed by the name of the option and any +optional user text, the option type, the PostScript document group, and +the sort order number. The option name must conform to the PPD specification +and cannot exceed 40 characters in length. If you specify user text, it +cannot exceed 80 characters in length.

+ +

The option type can be Boolean for true/false +selections, PickOne for picking one of many choices, or +PickMany for picking zero or more choices. Boolean +options can have at most two choices with the names +False and True. Pick options can have any +number of choices, although for Windows compatibility reasons +the number of choices should not exceed 255.

+ +

The PostScript document group is typically AnySetup, +meaning that the option can be introduced at any point in the +PostScript document. Other values include PageSetup to +include the option before each page and DocumentSetup +to include the option once at the beginning of the document.

+ +

The sort order number is used to sort the printer commands +associated with each option choice within the PostScript +document. This allows you to setup certain options before others +as required by the printer. For most CUPS raster printer +drivers, the value 10 can be used for all options.

+ +

Once the option is specified, each option choice can be +listed using the Choice directive:

+ +
+*Choice True/Yes "<</cupsCompression 1>>setpagedevice"
+Choice False/No "<</cupsCompression 0>>setpagedevice"
+
+ +

The directive is followed by the choice name and optionally +user text, and the PostScript commands that should be inserted +when printing a file to this printer. The option name must +conform to the PPD specification and cannot exceed 40 characters +in length. If you specify user text, it cannot exceed 80 +characters in length.

+ +

The PostScript commands are also interpreted by any RIP +filters, so these commands typically must be present for all +option choices. Most commands take the form:

+ +
+<</name value>>setpagedevice
+
+ +

where name is the name of the PostScript page device +attribute and value is the numeric or string value for +that attribute.

+ + +

Defining Constants

+ +

Sometimes you will want to define constants for your drivers +so that you can share values in different groups within the same +driver information file, or to share values between different +driver information files using the #include directive. +The #define directive is used to define constants for +use in your printer definitions:

+ +
+#define NAME value
+
+ +

The NAME is any sequence of letters, numbers, and +the underscore. The value is a number or string; if the +value contains spaces you must put double quotes around it, for +example:

+ +
+#define FOO "My String Value"
+
+ +

Constants can also be defined on the command-line using the -D +option:

+ +
+ppdc -DNAME="value" filename.drv
+
+ +

Once defined, you use the notation $NAME to substitute the value of +the constant in the file, for example:

+ +
+#define MANUFACTURER "Foo"
+#define FOO_600      0
+#define FOO_1200     1
+
+{
+  Manufacturer "$MANUFACTURER"
+  ModelNumber $FOO_600
+  ModelName "FooJet 2000"
+  ...
+}
+
+{
+  Manufacturer "$MANUFACTURER"
+  ModelNumber $FOO_1200
+  ModelName "FooJet 2001"
+  ...
+}
+
+ +

Numeric constants can be bitwise OR'd together by placing the constants +inside parenthesis, for example:

+ +
+// ModelNumber capability bits
+#define DUPLEX 1
+#define COLOR  2
+
+...
+
+{
+  // Define a model number specifying the capabilities of the printer...
+  ModelNumber ($DUPLEX $COLOR)
+  ...
+}
+
+ + +

Conditional Statements

+ +

The PPD compiler supports conditional compilation using the #if, +#elif, #else, and #endif directives. The #if +and #elif directives are followed by a constant name or an expression. +For example, to include a group of options when "ADVANCED" is defined:

+ +
+#if ADVANCED
+Group "Advanced/Advanced Options"
+  Option "fooCyanAdjust/Cyan Adjustment"
+    Choice "plus10/+10%" ""
+    Choice "plus5/+5%" ""
+    *Choice "none/No Adjustment" ""
+    Choice "minus5/-5%" ""
+    Choice "minus10/-10%" ""
+  Option "fooMagentaAdjust/Magenta Adjustment"
+    Choice "plus10/+10%" ""
+    Choice "plus5/+5%" ""
+    *Choice "none/No Adjustment" ""
+    Choice "minus5/-5%" ""
+    Choice "minus10/-10%" ""
+  Option "fooYellowAdjust/Yellow Adjustment"
+    Choice "plus10/+10%" ""
+    Choice "plus5/+5%" ""
+    *Choice "none/No Adjustment" ""
+    Choice "minus5/-5%" ""
+    Choice "minus10/-10%" ""
+  Option "fooBlackAdjust/Black Adjustment"
+    Choice "plus10/+10%" ""
+    Choice "plus5/+5%" ""
+    *Choice "none/No Adjustment" ""
+    Choice "minus5/-5%" ""
+    Choice "minus10/-10%" ""
+#endif
+
+ + +

Defining Constraints

+ +

Constraints are strings that are used to specify that one or more option +choices are incompatible, for example two-sided printing on transparency media. +Constraints are also used to prevent the use of uninstalled features such as the +duplexer unit, additional media trays, and so forth.

+ +

The UIConstraints directive is used to specify a constraint that is +placed in the PPD file. The directive is followed by a string using one of the +following formats:

+ +
+UIConstraints "*Option1 *Option2"
+UIConstraints "*Option1 Choice1 *Option2"
+UIConstraints "*Option1 *Option2 Choice2"
+UIConstraints "*Option1 Choice1 *Option2 Choice2"
+
+ +

Each option name is preceded by the asterisk (*). If no choice is +given for an option, then all choices except False and +None will conflict with the other option and choice(s). Since the PPD +compiler automatically adds reciprocal constraints (option A conflicts with +option B, so therefore option B conflicts with option A), you need only specify +the constraint once.

+ +

Listing 5: "examples/constraint.drv"

+ +
+
+// Include standard font and media definitions
+#include <font.defs>
+#include <media.defs>
+
+// List the fonts that are supported, in this case all standard fonts...
+Font *
+
+// Manufacturer, model name, and version
+Manufacturer "Foo"
+ModelName "FooJet 2000"
+Version 1.0
+
+// Each filter provided by the driver...
+Filter application/vnd.cups-raster 100 rastertofoo
+
+// Supported page sizes
+*MediaSize Letter
+MediaSize A4
+
+// Supported resolutions
+*Resolution k 8 0 0 0 "600dpi/600 DPI"
+
+// Installable Option Group
+Group "InstallableOptions/Options Installed"
+
+  // Duplexing unit option
+  Option "OptionDuplexer/Duplexing Unit" Boolean AnySetup 10
+    Choice True/Installed ""
+    *Choice "False/Not Installed" ""
+
+// General Option Group
+Group General
+
+  // Duplexing option
+  Option "Duplex/Two-Sided Printing" PickOne AnySetup 10
+    *Choice "None/No" "<</Duplex false>>setpagedevice""
+    Choice "DuplexNoTumble/Long Edge Binding"
+           "<</Duplex true/Tumble false>>setpagedevice""
+    Choice "DuplexTumble/Short Edge Binding"
+           "<</Duplex true/Tumble true>>setpagedevice""
+
+// Only allow duplexing if the duplexer is installed
+UIConstraints "*Duplex *OptionDuplexer False"
+
+// Specify the name of the PPD file we want to generate...
+PCFileName "foojet2k.ppd"
+
+ +

Listing 5 shows a variation of the first example with +an added Duplex option and installable option for the duplexer, +OptionDuplex. A constraint is added at the end to specify that any +choice of the Duplex option that is not None is incompatible +with the "Duplexer Installed" option set to "Not Installed" +(False):

+ +
+UIConstraints "*Duplex *OptionDuplexer False"
+
+ +

Enhanced Constraints

+ +

CUPS 1.4 supports constraints between 2 or more options using the +Attribute directive. cupsUIConstraints attributes define +the constraints, while cupsUIResolver attributes define option changes +to resolve constraints. For example, we can specify the previous duplex +constraint with a resolver that turns off duplexing with the following two +lines:

+ +
+Attribute cupsUIConstraints DuplexOff "*Duplex *OptionDuplexer False"
+Attribute cupsUIResolver DuplexOff "*Duplex None"
+
+ +

Localization

+ +

The PPD compiler provides localization of PPD files in different languages +through message catalog files in the GNU gettext or Apple .strings +formats. Each user text string and several key PPD attribute values such as +LanguageVersion and LanguageEncoding are looked up in the +corresponding message catalog and the translated text is substituted in the +generated PPD files. One message catalog file can be used by multiple driver +information files, and each file contains a single language translation.

+ +

The ppdpo Utility

+ +

While CUPS includes localizations of all standard media sizes and options in +several languages, your driver information files may provide their own media +sizes and options that need to be localized. CUPS provides a utility program to +aid in the localization of drivers called ppdpo(1). The ppdpo program creates +or updates a message catalog file based upon one or more driver information +files. New messages are added with the word "TRANSLATE" added to the front of +the translation string to make locating new strings for translation easier. The +program accepts the message catalog filename and one or more driver information +files.

+ +

For example, run the following command to create a new German message catalog +called de.po for all of the driver information files in the current +directory:

+ +
+ppdpo -o de.po *.drv
+
+ +

If the file de.po already exists, ppdpo will update the +contents of the file with any new messages that need to be translated. To create +an Apple .strings file instead, specify the output filename with a .strings +extension, for example:

+ +
+ppdpo -o de.strings *.drv
+
+ +

Using Message Catalogs with the PPD Compiler

+ +

Once you have created a message catalog, use the #po directive to declare it in each +driver information file. For example, to declare the German message catalog for +a driver use:

+ +
+#po de "de.po"  // German
+
+ +

In fact, you can use the #po directive as many times as needed:

+ +
+#po de "de.po"  // German
+#po es "es.po"  // Spanish
+#po fr "fr.po"  // French
+#po it "it.po"  // Italian
+#po ja "ja.po"  // Japanese
+
+ +

The filename ("de.po", etc.) can be relative to the location of the driver +information file or an absolute path. Once defined, the PPD compiler will +automatically generate a globalized PPD for every language declared in your +driver information file. To generate a single-language PPD file, simply use the +-l option to list the corresponding locale, for example:

+ +
+ppdc -l de -d ppd/de mydrivers.drv
+
+ +

to generate German PPD files.

diff --git a/filter/pstops.c b/filter/pstops.c new file mode 100644 index 0000000000..77b7de5060 --- /dev/null +++ b/filter/pstops.c @@ -0,0 +1,3425 @@ +/* + * "$Id$" + * + * PostScript filter for CUPS. + * + * Copyright 2007-2012 by Apple Inc. + * Copyright 1993-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Main entry. + * add_page() - Add a page to the pages array. + * cancel_job() - Flag the job as canceled. + * check_range() - Check to see if the current page is selected for + * printing. + * copy_bytes() - Copy bytes from the input file to stdout. + * copy_comments() - Copy all of the comments section. + * copy_dsc() - Copy a DSC-conforming document. + * copy_non_dsc() - Copy a document that does not conform to the DSC. + * copy_page() - Copy a page description. + * copy_prolog() - Copy the document prolog section. + * copy_setup() - Copy the document setup section. + * copy_trailer() - Copy the document trailer. + * do_prolog() - Send the necessary document prolog commands. + * do_setup() - Send the necessary document setup commands. + * doc_printf() - Send a formatted string to stdout and/or the temp + * file. + * doc_puts() - Send a nul-terminated string to stdout and/or the + * temp file. + * doc_write() - Send data to stdout and/or the temp file. + * end_nup() - End processing for N-up printing. + * include_feature() - Include a printer option/feature command. + * parse_text() - Parse a text value in a comment. + * set_pstops_options() - Set pstops options. + * skip_page() - Skip past a page that won't be printed. + * start_nup() - Start processing for N-up printing. + * write_label_prolog() - Write the prolog with the classification and page + * label. + * write_labels() - Write the actual page labels. + * write_options() - Write options provided via %%IncludeFeature. + */ + +/* + * Include necessary headers... + */ + +#include "common.h" +#include +#include +#include +#include +#include +#include + + +/* + * Constants... + */ + +#define PSTOPS_BORDERNONE 0 /* No border or hairline border */ +#define PSTOPS_BORDERTHICK 1 /* Think border */ +#define PSTOPS_BORDERSINGLE 2 /* Single-line hairline border */ +#define PSTOPS_BORDERSINGLE2 3 /* Single-line thick border */ +#define PSTOPS_BORDERDOUBLE 4 /* Double-line hairline border */ +#define PSTOPS_BORDERDOUBLE2 5 /* Double-line thick border */ + +#define PSTOPS_LAYOUT_LRBT 0 /* Left to right, bottom to top */ +#define PSTOPS_LAYOUT_LRTB 1 /* Left to right, top to bottom */ +#define PSTOPS_LAYOUT_RLBT 2 /* Right to left, bottom to top */ +#define PSTOPS_LAYOUT_RLTB 3 /* Right to left, top to bottom */ +#define PSTOPS_LAYOUT_BTLR 4 /* Bottom to top, left to right */ +#define PSTOPS_LAYOUT_TBLR 5 /* Top to bottom, left to right */ +#define PSTOPS_LAYOUT_BTRL 6 /* Bottom to top, right to left */ +#define PSTOPS_LAYOUT_TBRL 7 /* Top to bottom, right to left */ + +#define PSTOPS_LAYOUT_NEGATEY 1 /* The bits for the layout */ +#define PSTOPS_LAYOUT_NEGATEX 2 /* definitions above... */ +#define PSTOPS_LAYOUT_VERTICAL 4 + + +/* + * Types... + */ + +typedef struct /**** Page information ****/ +{ + char *label; /* Page label */ + int bounding_box[4]; /* PageBoundingBox */ + off_t offset; /* Offset to start of page */ + ssize_t length; /* Number of bytes for page */ + int num_options; /* Number of options for this page */ + cups_option_t *options; /* Options for this page */ +} pstops_page_t; + +typedef struct /**** Document information ****/ +{ + int page; /* Current page */ + int bounding_box[4]; /* BoundingBox from header */ + int new_bounding_box[4]; /* New composite bounding box */ + int num_options; /* Number of document-wide options */ + cups_option_t *options; /* Document-wide options */ + int normal_landscape, /* Normal rotation for landscape? */ + saw_eof, /* Saw the %%EOF comment? */ + slow_collate, /* Collate copies by hand? */ + slow_duplex, /* Duplex pages slowly? */ + slow_order, /* Reverse pages slowly? */ + use_ESPshowpage; /* Use ESPshowpage? */ + cups_array_t *pages; /* Pages in document */ + cups_file_t *temp; /* Temporary file, if any */ + char tempfile[1024]; /* Temporary filename */ + int job_id; /* Job ID */ + const char *user, /* User name */ + *title; /* Job name */ + int copies; /* Number of copies */ + const char *ap_input_slot, /* AP_FIRSTPAGE_InputSlot value */ + *ap_manual_feed, /* AP_FIRSTPAGE_ManualFeed value */ + *ap_media_color, /* AP_FIRSTPAGE_MediaColor value */ + *ap_media_type, /* AP_FIRSTPAGE_MediaType value */ + *ap_page_region, /* AP_FIRSTPAGE_PageRegion value */ + *ap_page_size; /* AP_FIRSTPAGE_PageSize value */ + int collate, /* Collate copies? */ + emit_jcl, /* Emit JCL commands? */ + fitplot; /* Fit pages to media */ + const char *input_slot, /* InputSlot value */ + *manual_feed, /* ManualFeed value */ + *media_color, /* MediaColor value */ + *media_type, /* MediaType value */ + *page_region, /* PageRegion value */ + *page_size; /* PageSize value */ + int mirror, /* doc->mirror/mirror pages */ + number_up, /* Number of pages on each sheet */ + number_up_layout, /* doc->number_up_layout of N-up pages */ + output_order, /* Requested reverse output order? */ + page_border; /* doc->page_border around pages */ + const char *page_label, /* page-label option, if any */ + *page_ranges, /* page-ranges option, if any */ + *page_set; /* page-set option, if any */ +} pstops_doc_t; + + +/* + * Convenience macros... + */ + +#define is_first_page(p) (doc->number_up == 1 || \ + ((p) % doc->number_up) == 1) +#define is_last_page(p) (doc->number_up == 1 || \ + ((p) % doc->number_up) == 0) +#define is_not_last_page(p) (doc->number_up > 1 && \ + ((p) % doc->number_up) != 0) + + +/* + * Local globals... + */ + +static int JobCanceled = 0;/* Set to 1 on SIGTERM */ + + +/* + * Local functions... + */ + +static pstops_page_t *add_page(pstops_doc_t *doc, const char *label); +static void cancel_job(int sig); +static int check_range(pstops_doc_t *doc, int page); +static void copy_bytes(cups_file_t *fp, off_t offset, + size_t length); +static ssize_t copy_comments(cups_file_t *fp, pstops_doc_t *doc, + ppd_file_t *ppd, char *line, + ssize_t linelen, size_t linesize); +static void copy_dsc(cups_file_t *fp, pstops_doc_t *doc, + ppd_file_t *ppd, char *line, ssize_t linelen, + size_t linesize); +static void copy_non_dsc(cups_file_t *fp, pstops_doc_t *doc, + ppd_file_t *ppd, char *line, + ssize_t linelen, size_t linesize); +static ssize_t copy_page(cups_file_t *fp, pstops_doc_t *doc, + ppd_file_t *ppd, int number, char *line, + ssize_t linelen, size_t linesize); +static ssize_t copy_prolog(cups_file_t *fp, pstops_doc_t *doc, + ppd_file_t *ppd, char *line, + ssize_t linelen, size_t linesize); +static ssize_t copy_setup(cups_file_t *fp, pstops_doc_t *doc, + ppd_file_t *ppd, char *line, + ssize_t linelen, size_t linesize); +static ssize_t copy_trailer(cups_file_t *fp, pstops_doc_t *doc, + ppd_file_t *ppd, int number, char *line, + ssize_t linelen, size_t linesize); +static void do_prolog(pstops_doc_t *doc, ppd_file_t *ppd); +static void do_setup(pstops_doc_t *doc, ppd_file_t *ppd); +static void doc_printf(pstops_doc_t *doc, const char *format, ...) + __attribute__ ((__format__ (__printf__, 2, 3))); +static void doc_puts(pstops_doc_t *doc, const char *s); +static void doc_write(pstops_doc_t *doc, const char *s, size_t len); +static void end_nup(pstops_doc_t *doc, int number); +static int include_feature(ppd_file_t *ppd, const char *line, + int num_options, + cups_option_t **options); +static char *parse_text(const char *start, char **end, char *buffer, + size_t bufsize); +static void set_pstops_options(pstops_doc_t *doc, ppd_file_t *ppd, + char *argv[], int num_options, + cups_option_t *options); +static ssize_t skip_page(cups_file_t *fp, char *line, ssize_t linelen, + size_t linesize); +static void start_nup(pstops_doc_t *doc, int number, + int show_border, const int *bounding_box); +static void write_label_prolog(pstops_doc_t *doc, const char *label, + float bottom, float top, + float width); +static void write_labels(pstops_doc_t *doc, int orient); +static void write_options(pstops_doc_t *doc, ppd_file_t *ppd, + int num_options, cups_option_t *options); + + +/* + * 'main()' - Main entry. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + pstops_doc_t doc; /* Document information */ + cups_file_t *fp; /* Print file */ + ppd_file_t *ppd; /* PPD file */ + int num_options; /* Number of print options */ + cups_option_t *options; /* Print options */ + char line[8192]; /* Line buffer */ + size_t len; /* Length of line buffer */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* Actions for POSIX signals */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ + + + /* + * Make sure status messages are not buffered... + */ + + setbuf(stderr, NULL); + + /* + * Ignore broken pipe signals... + */ + + signal(SIGPIPE, SIG_IGN); + + /* + * Check command-line... + */ + + if (argc < 6 || argc > 7) + { + _cupsLangPrintf(stderr, + _("Usage: %s job-id user title copies options file"), + argv[0]); + return (1); + } + + /* + * Register a signal handler to cleanly cancel a job. + */ + +#ifdef HAVE_SIGSET /* Use System V signals over POSIX to avoid bugs */ + sigset(SIGTERM, cancel_job); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + + sigemptyset(&action.sa_mask); + action.sa_handler = cancel_job; + sigaction(SIGTERM, &action, NULL); +#else + signal(SIGTERM, cancel_job); +#endif /* HAVE_SIGSET */ + + /* + * If we have 7 arguments, print the file named on the command-line. + * Otherwise, send stdin instead... + */ + + if (argc == 6) + fp = cupsFileStdin(); + else + { + /* + * Try to open the print file... + */ + + if ((fp = cupsFileOpen(argv[6], "r")) == NULL) + { + _cupsLangPrintError("ERROR", _("Unable to open print file")); + return (1); + } + } + + /* + * Read the first line to see if we have DSC comments... + */ + + if ((len = cupsFileGetLine(fp, line, sizeof(line))) == 0) + { + fputs("DEBUG: The print file is empty.\n", stderr); + return (1); + } + + /* + * Process command-line options... + */ + + options = NULL; + num_options = cupsParseOptions(argv[5], 0, &options); + ppd = SetCommonOptions(num_options, options, 1); + + set_pstops_options(&doc, ppd, argv, num_options, options); + + /* + * Write any "exit server" options that have been selected... + */ + + ppdEmit(ppd, stdout, PPD_ORDER_EXIT); + + /* + * Write any JCL commands that are needed to print PostScript code... + */ + + if (doc.emit_jcl) + ppdEmitJCL(ppd, stdout, doc.job_id, doc.user, doc.title); + + /* + * Start with a DSC header... + */ + + puts("%!PS-Adobe-3.0"); + + /* + * Skip leading PJL in the document... + */ + + while (!strncmp(line, "\033%-12345X", 9) || !strncmp(line, "@PJL ", 5)) + { + /* + * Yup, we have leading PJL fun, so skip it until we hit the line + * with "ENTER LANGUAGE"... + */ + + fputs("DEBUG: Skipping PJL header...\n", stderr); + + while (strstr(line, "ENTER LANGUAGE") == NULL && strncmp(line, "%!", 2)) + if ((len = cupsFileGetLine(fp, line, sizeof(line))) == 0) + break; + + if (!strncmp(line, "%!", 2)) + break; + + if ((len = cupsFileGetLine(fp, line, sizeof(line))) == 0) + break; + } + + /* + * Now see if the document conforms to the Adobe Document Structuring + * Conventions... + */ + + if (!strncmp(line, "%!PS-Adobe-", 11)) + { + /* + * Yes, filter the document... + */ + + copy_dsc(fp, &doc, ppd, line, len, sizeof(line)); + } + else + { + /* + * No, display an error message and treat the file as if it contains + * a single page... + */ + + copy_non_dsc(fp, &doc, ppd, line, len, sizeof(line)); + } + + /* + * Send %%EOF as needed... + */ + + if (!doc.saw_eof) + puts("%%EOF"); + + /* + * End the job with the appropriate JCL command or CTRL-D... + */ + + if (doc.emit_jcl) + { + if (ppd && ppd->jcl_end) + ppdEmitJCLEnd(ppd, stdout); + else + putchar(0x04); + } + + /* + * Close files and remove the temporary file if needed... + */ + + if (doc.temp) + { + cupsFileClose(doc.temp); + unlink(doc.tempfile); + } + + ppdClose(ppd); + cupsFreeOptions(num_options, options); + + cupsFileClose(fp); + + return (0); +} + + +/* + * 'add_page()' - Add a page to the pages array. + */ + +static pstops_page_t * /* O - New page info object */ +add_page(pstops_doc_t *doc, /* I - Document information */ + const char *label) /* I - Page label */ +{ + pstops_page_t *pageinfo; /* New page info object */ + + + if (!doc->pages) + doc->pages = cupsArrayNew(NULL, NULL); + + if (!doc->pages) + { + _cupsLangPrintError("EMERG", _("Unable to allocate memory for pages array")); + exit(1); + } + + if ((pageinfo = calloc(1, sizeof(pstops_page_t))) == NULL) + { + _cupsLangPrintError("EMERG", _("Unable to allocate memory for page info")); + exit(1); + } + + pageinfo->label = strdup(label); + pageinfo->offset = cupsFileTell(doc->temp); + + cupsArrayAdd(doc->pages, pageinfo); + + doc->page ++; + + return (pageinfo); +} + + +/* + * 'cancel_job()' - Flag the job as canceled. + */ + +static void +cancel_job(int sig) /* I - Signal number (unused) */ +{ + (void)sig; + + JobCanceled = 1; +} + + +/* + * 'check_range()' - Check to see if the current page is selected for + * printing. + */ + +static int /* O - 1 if selected, 0 otherwise */ +check_range(pstops_doc_t *doc, /* I - Document information */ + int page) /* I - Page number */ +{ + const char *range; /* Pointer into range string */ + int lower, upper; /* Lower and upper page numbers */ + + + if (doc->page_set) + { + /* + * See if we only print even or odd pages... + */ + + if (!_cups_strcasecmp(doc->page_set, "even") && (page & 1)) + return (0); + + if (!_cups_strcasecmp(doc->page_set, "odd") && !(page & 1)) + return (0); + } + + if (!doc->page_ranges) + return (1); /* No range, print all pages... */ + + for (range = doc->page_ranges; *range != '\0';) + { + if (*range == '-') + { + lower = 1; + range ++; + upper = strtol(range, (char **)&range, 10); + } + else + { + lower = strtol(range, (char **)&range, 10); + + if (*range == '-') + { + range ++; + if (!isdigit(*range & 255)) + upper = 65535; + else + upper = strtol(range, (char **)&range, 10); + } + else + upper = lower; + } + + if (page >= lower && page <= upper) + return (1); + + if (*range == ',') + range ++; + else + break; + } + + return (0); +} + + +/* + * 'copy_bytes()' - Copy bytes from the input file to stdout. + */ + +static void +copy_bytes(cups_file_t *fp, /* I - File to read from */ + off_t offset, /* I - Offset to page data */ + size_t length) /* I - Length of page data */ +{ + char buffer[8192]; /* Data buffer */ + ssize_t nbytes; /* Number of bytes read */ + size_t nleft; /* Number of bytes left/remaining */ + + + nleft = length; + + if (cupsFileSeek(fp, offset) < 0) + { + _cupsLangPrintError("ERROR", _("Unable to see in file")); + return; + } + + while (nleft > 0 || length == 0) + { + if (nleft > sizeof(buffer) || length == 0) + nbytes = sizeof(buffer); + else + nbytes = nleft; + + if ((nbytes = cupsFileRead(fp, buffer, nbytes)) < 1) + return; + + nleft -= nbytes; + + fwrite(buffer, 1, nbytes, stdout); + } +} + + +/* + * 'copy_comments()' - Copy all of the comments section. + * + * This function expects "line" to be filled with a comment line. + * On return, "line" will contain the next line in the file, if any. + */ + +static ssize_t /* O - Length of next line */ +copy_comments(cups_file_t *fp, /* I - File to read from */ + pstops_doc_t *doc, /* I - Document info */ + ppd_file_t *ppd, /* I - PPD file */ + char *line, /* I - Line buffer */ + ssize_t linelen, /* I - Length of initial line */ + size_t linesize) /* I - Size of line buffer */ +{ + int saw_bounding_box, /* Saw %%BoundingBox: comment? */ + saw_for, /* Saw %%For: comment? */ + saw_pages, /* Saw %%Pages: comment? */ + saw_title; /* Saw %%Title: comment? */ + + + /* + * Loop until we see %%EndComments or a non-comment line... + */ + + saw_bounding_box = 0; + saw_for = 0; + saw_pages = 0; + saw_title = 0; + + while (line[0] == '%') + { + /* + * Strip trailing whitespace... + */ + + while (linelen > 0) + { + linelen --; + + if (!isspace(line[linelen] & 255)) + break; + else + line[linelen] = '\0'; + } + + /* + * Log the header... + */ + + fprintf(stderr, "DEBUG: %s\n", line); + + /* + * Pull the headers out... + */ + + if (!strncmp(line, "%%Pages:", 8)) + { + int pages; /* Number of pages */ + + if (saw_pages) + fputs("DEBUG: A duplicate %%Pages: comment was seen.\n", stderr); + + saw_pages = 1; + + if (Duplex && (pages = atoi(line + 8)) > 0 && pages <= doc->number_up) + { + /* + * Since we will only be printing on a single page, disable duplexing. + */ + + Duplex = 0; + doc->slow_duplex = 0; + + if (cupsGetOption("sides", doc->num_options, doc->options)) + doc->num_options = cupsAddOption("sides", "one-sided", + doc->num_options, &(doc->options)); + + if (cupsGetOption("Duplex", doc->num_options, doc->options)) + doc->num_options = cupsAddOption("Duplex", "None", + doc->num_options, &(doc->options)); + + if (cupsGetOption("EFDuplex", doc->num_options, doc->options)) + doc->num_options = cupsAddOption("EFDuplex", "None", + doc->num_options, &(doc->options)); + + if (cupsGetOption("EFDuplexing", doc->num_options, doc->options)) + doc->num_options = cupsAddOption("EFDuplexing", "False", + doc->num_options, &(doc->options)); + + if (cupsGetOption("KD03Duplex", doc->num_options, doc->options)) + doc->num_options = cupsAddOption("KD03Duplex", "None", + doc->num_options, &(doc->options)); + + if (cupsGetOption("JCLDuplex", doc->num_options, doc->options)) + doc->num_options = cupsAddOption("JCLDuplex", "None", + doc->num_options, &(doc->options)); + + ppdMarkOption(ppd, "Duplex", "None"); + ppdMarkOption(ppd, "EFDuplex", "None"); + ppdMarkOption(ppd, "EFDuplexing", "False"); + ppdMarkOption(ppd, "KD03Duplex", "None"); + ppdMarkOption(ppd, "JCLDuplex", "None"); + } + } + else if (!strncmp(line, "%%BoundingBox:", 14)) + { + if (saw_bounding_box) + fputs("DEBUG: A duplicate %%BoundingBox: comment was seen.\n", stderr); + else if (strstr(line + 14, "(atend)")) + { + /* + * Do nothing for now but use the default imageable area... + */ + } + else if (sscanf(line + 14, "%d%d%d%d", doc->bounding_box + 0, + doc->bounding_box + 1, doc->bounding_box + 2, + doc->bounding_box + 3) != 4) + { + fputs("DEBUG: A bad %%BoundingBox: comment was seen.\n", stderr); + + doc->bounding_box[0] = (int)PageLeft; + doc->bounding_box[1] = (int)PageBottom; + doc->bounding_box[2] = (int)PageRight; + doc->bounding_box[3] = (int)PageTop; + } + + saw_bounding_box = 1; + } + else if (!strncmp(line, "%%For:", 6)) + { + saw_for = 1; + doc_printf(doc, "%s\n", line); + } + else if (!strncmp(line, "%%Title:", 8)) + { + saw_title = 1; + doc_printf(doc, "%s\n", line); + } + else if (!strncmp(line, "%cupsRotation:", 14)) + { + /* + * Reset orientation of document? + */ + + int orient = (atoi(line + 14) / 90) & 3; + + if (orient != Orientation) + { + /* + * Yes, update things so that the pages come out right... + */ + + Orientation = (4 - Orientation + orient) & 3; + UpdatePageVars(); + Orientation = orient; + } + } + else if (!strcmp(line, "%%EndComments")) + { + linelen = cupsFileGetLine(fp, line, linesize); + break; + } + else if (strncmp(line, "%!", 2) && strncmp(line, "%cups", 5)) + doc_printf(doc, "%s\n", line); + + if ((linelen = cupsFileGetLine(fp, line, linesize)) == 0) + break; + } + + if (!saw_bounding_box) + fputs("DEBUG: There wasn't a %%BoundingBox: comment in the header.\n", + stderr); + + if (!saw_pages) + fputs("DEBUG: There wasn't a %%Pages: comment in the header.\n", stderr); + + if (!saw_for) + WriteTextComment("For", doc->user); + + if (!saw_title) + WriteTextComment("Title", doc->title); + + if (doc->copies != 1 && (!doc->collate || !doc->slow_collate)) + { + /* + * Tell the document processor the copy and duplex options + * that are required... + */ + + doc_printf(doc, "%%%%Requirements: numcopies(%d)%s%s\n", doc->copies, + doc->collate ? " collate" : "", + Duplex ? " duplex" : ""); + + /* + * Apple uses RBI comments for various non-PPD options... + */ + + doc_printf(doc, "%%RBINumCopies: %d\n", doc->copies); + } + else + { + /* + * Tell the document processor the duplex option that is required... + */ + + if (Duplex) + doc_puts(doc, "%%Requirements: duplex\n"); + + /* + * Apple uses RBI comments for various non-PPD options... + */ + + doc_puts(doc, "%RBINumCopies: 1\n"); + } + + doc_puts(doc, "%%Pages: (atend)\n"); + doc_puts(doc, "%%BoundingBox: (atend)\n"); + doc_puts(doc, "%%EndComments\n"); + + return (linelen); +} + + +/* + * 'copy_dsc()' - Copy a DSC-conforming document. + * + * This function expects "line" to be filled with the %!PS-Adobe comment line. + */ + +static void +copy_dsc(cups_file_t *fp, /* I - File to read from */ + pstops_doc_t *doc, /* I - Document info */ + ppd_file_t *ppd, /* I - PPD file */ + char *line, /* I - Line buffer */ + ssize_t linelen, /* I - Length of initial line */ + size_t linesize) /* I - Size of line buffer */ +{ + int number; /* Page number */ + pstops_page_t *pageinfo; /* Page information */ + + + /* + * Make sure we use ESPshowpage for EPS files... + */ + + if (strstr(line, "EPSF")) + { + doc->use_ESPshowpage = 1; + doc->number_up = 1; + } + + /* + * Start sending the document with any commands needed... + */ + + fprintf(stderr, "DEBUG: Before copy_comments - %s", line); + linelen = copy_comments(fp, doc, ppd, line, linelen, linesize); + + /* + * Now find the prolog section, if any... + */ + + fprintf(stderr, "DEBUG: Before copy_prolog - %s", line); + linelen = copy_prolog(fp, doc, ppd, line, linelen, linesize); + + /* + * Then the document setup section... + */ + + fprintf(stderr, "DEBUG: Before copy_setup - %s", line); + linelen = copy_setup(fp, doc, ppd, line, linelen, linesize); + + /* + * Copy until we see %%Page:... + */ + + while (strncmp(line, "%%Page:", 7) && strncmp(line, "%%Trailer", 9)) + { + doc_write(doc, line, linelen); + + if ((linelen = cupsFileGetLine(fp, line, linesize)) == 0) + break; + } + + /* + * Then process pages until we have no more... + */ + + number = 0; + + fprintf(stderr, "DEBUG: Before page loop - %s", line); + while (!strncmp(line, "%%Page:", 7)) + { + if (JobCanceled) + break; + + number ++; + + if (check_range(doc, (number - 1) / doc->number_up + 1)) + { + fprintf(stderr, "DEBUG: Copying page %d...\n", number); + linelen = copy_page(fp, doc, ppd, number, line, linelen, linesize); + } + else + { + fprintf(stderr, "DEBUG: Skipping page %d...\n", number); + linelen = skip_page(fp, line, linelen, linesize); + } + } + + /* + * Finish up the last page(s)... + */ + + if (number && is_not_last_page(number) && cupsArrayLast(doc->pages) && + check_range(doc, (number - 1) / doc->number_up + 1)) + { + pageinfo = (pstops_page_t *)cupsArrayLast(doc->pages); + + start_nup(doc, doc->number_up, 0, doc->bounding_box); + doc_puts(doc, "showpage\n"); + end_nup(doc, doc->number_up); + + pageinfo->length = cupsFileTell(doc->temp) - pageinfo->offset; + } + + if (doc->slow_duplex && (doc->page & 1)) + { + /* + * Make sure we have an even number of pages... + */ + + pageinfo = add_page(doc, "(filler)"); + + if (!doc->slow_order) + { + if (!ppd || !ppd->num_filters) + fprintf(stderr, "PAGE: %d %d\n", doc->page, + doc->slow_collate ? 1 : doc->copies); + + printf("%%%%Page: (filler) %d\n", doc->page); + } + + start_nup(doc, doc->number_up, 0, doc->bounding_box); + doc_puts(doc, "showpage\n"); + end_nup(doc, doc->number_up); + + pageinfo->length = cupsFileTell(doc->temp) - pageinfo->offset; + } + + /* + * Make additional copies as necessary... + */ + + number = doc->slow_order ? 0 : doc->page; + + if (doc->temp && !JobCanceled && cupsArrayCount(doc->pages) > 0) + { + int copy; /* Current copy */ + + + /* + * Reopen the temporary file for reading... + */ + + cupsFileClose(doc->temp); + + doc->temp = cupsFileOpen(doc->tempfile, "r"); + + /* + * Make the copies... + */ + + if (doc->slow_collate) + copy = !doc->slow_order; + else + copy = doc->copies - 1; + + for (; copy < doc->copies; copy ++) + { + if (JobCanceled) + break; + + /* + * Send end-of-job stuff followed by any start-of-job stuff required + * for the JCL options... + */ + + if (number && doc->emit_jcl && ppd && ppd->jcl_end) + { + /* + * Send the trailer... + */ + + puts("%%Trailer"); + printf("%%%%Pages: %d\n", cupsArrayCount(doc->pages)); + if (doc->number_up > 1 || doc->fitplot) + printf("%%%%BoundingBox: %.0f %.0f %.0f %.0f\n", + PageLeft, PageBottom, PageRight, PageTop); + else + printf("%%%%BoundingBox: %d %d %d %d\n", + doc->new_bounding_box[0], doc->new_bounding_box[1], + doc->new_bounding_box[2], doc->new_bounding_box[3]); + puts("%%EOF"); + + /* + * Start a new document... + */ + + ppdEmitJCLEnd(ppd, stdout); + ppdEmitJCL(ppd, stdout, doc->job_id, doc->user, doc->title); + + puts("%!PS-Adobe-3.0"); + + number = 0; + } + + /* + * Copy the prolog as needed... + */ + + if (!number) + { + pageinfo = (pstops_page_t *)cupsArrayFirst(doc->pages); + copy_bytes(doc->temp, 0, pageinfo->offset); + } + + /* + * Then copy all of the pages... + */ + + pageinfo = doc->slow_order ? (pstops_page_t *)cupsArrayLast(doc->pages) : + (pstops_page_t *)cupsArrayFirst(doc->pages); + + while (pageinfo) + { + if (JobCanceled) + break; + + number ++; + + if (!ppd || !ppd->num_filters) + fprintf(stderr, "PAGE: %d %d\n", number, + doc->slow_collate ? 1 : doc->copies); + + if (doc->number_up > 1) + { + printf("%%%%Page: (%d) %d\n", number, number); + printf("%%%%PageBoundingBox: %.0f %.0f %.0f %.0f\n", + PageLeft, PageBottom, PageRight, PageTop); + } + else + { + printf("%%%%Page: %s %d\n", pageinfo->label, number); + printf("%%%%PageBoundingBox: %d %d %d %d\n", + pageinfo->bounding_box[0], pageinfo->bounding_box[1], + pageinfo->bounding_box[2], pageinfo->bounding_box[3]); + } + + copy_bytes(doc->temp, pageinfo->offset, pageinfo->length); + + pageinfo = doc->slow_order ? (pstops_page_t *)cupsArrayPrev(doc->pages) : + (pstops_page_t *)cupsArrayNext(doc->pages); + } + } + } + + /* + * Restore the old showpage operator as needed... + */ + + if (doc->use_ESPshowpage) + puts("userdict/showpage/ESPshowpage load put\n"); + + /* + * Write/copy the trailer... + */ + + if (!JobCanceled) + copy_trailer(fp, doc, ppd, number, line, linelen, linesize); +} + + +/* + * 'copy_non_dsc()' - Copy a document that does not conform to the DSC. + * + * This function expects "line" to be filled with the %! comment line. + */ + +static void +copy_non_dsc(cups_file_t *fp, /* I - File to read from */ + pstops_doc_t *doc, /* I - Document info */ + ppd_file_t *ppd, /* I - PPD file */ + char *line, /* I - Line buffer */ + ssize_t linelen, /* I - Length of initial line */ + size_t linesize) /* I - Size of line buffer */ +{ + int copy; /* Current copy */ + char buffer[8192]; /* Copy buffer */ + int bytes; /* Number of bytes copied */ + + + /* + * First let the user know that they are attempting to print a file + * that may not print correctly... + */ + + fputs("DEBUG: This document does not conform to the Adobe Document " + "Structuring Conventions and may not print correctly.\n", stderr); + + /* + * Then write a standard DSC comment section... + */ + + printf("%%%%BoundingBox: %.0f %.0f %.0f %.0f\n", PageLeft, PageBottom, + PageRight, PageTop); + + if (doc->slow_collate && doc->copies > 1) + printf("%%%%Pages: %d\n", doc->copies); + else + puts("%%Pages: 1"); + + WriteTextComment("For", doc->user); + WriteTextComment("Title", doc->title); + + if (doc->copies != 1 && (!doc->collate || !doc->slow_collate)) + { + /* + * Tell the document processor the copy and duplex options + * that are required... + */ + + printf("%%%%Requirements: numcopies(%d)%s%s\n", doc->copies, + doc->collate ? " collate" : "", + Duplex ? " duplex" : ""); + + /* + * Apple uses RBI comments for various non-PPD options... + */ + + printf("%%RBINumCopies: %d\n", doc->copies); + } + else + { + /* + * Tell the document processor the duplex option that is required... + */ + + if (Duplex) + puts("%%Requirements: duplex"); + + /* + * Apple uses RBI comments for various non-PPD options... + */ + + puts("%RBINumCopies: 1"); + } + + puts("%%EndComments"); + + /* + * Then the prolog... + */ + + puts("%%BeginProlog"); + + do_prolog(doc, ppd); + + puts("%%EndProlog"); + + /* + * Then the setup section... + */ + + puts("%%BeginSetup"); + + do_setup(doc, ppd); + + puts("%%EndSetup"); + + /* + * Finally, embed a copy of the file inside a %%Page... + */ + + if (!ppd || !ppd->num_filters) + fprintf(stderr, "PAGE: 1 %d\n", doc->temp ? 1 : doc->copies); + + puts("%%Page: 1 1"); + puts("%%BeginPageSetup"); + ppdEmit(ppd, stdout, PPD_ORDER_PAGE); + puts("%%EndPageSetup"); + puts("%%BeginDocument: nondsc"); + + fwrite(line, linelen, 1, stdout); + + if (doc->temp) + cupsFileWrite(doc->temp, line, linelen); + + while ((bytes = cupsFileRead(fp, buffer, sizeof(buffer))) > 0) + { + fwrite(buffer, 1, bytes, stdout); + + if (doc->temp) + cupsFileWrite(doc->temp, buffer, bytes); + } + + puts("%%EndDocument"); + + if (doc->use_ESPshowpage) + { + WriteLabels(Orientation); + puts("ESPshowpage"); + } + + if (doc->temp && !JobCanceled) + { + /* + * Reopen the temporary file for reading... + */ + + cupsFileClose(doc->temp); + + doc->temp = cupsFileOpen(doc->tempfile, "r"); + + /* + * Make the additional copies as needed... + */ + + for (copy = 1; copy < doc->copies; copy ++) + { + if (JobCanceled) + break; + + if (!ppd || !ppd->num_filters) + fputs("PAGE: 1 1\n", stderr); + + printf("%%%%Page: %d %d\n", copy + 1, copy + 1); + puts("%%BeginPageSetup"); + ppdEmit(ppd, stdout, PPD_ORDER_PAGE); + puts("%%EndPageSetup"); + puts("%%BeginDocument: nondsc"); + + copy_bytes(doc->temp, 0, 0); + + puts("%%EndDocument"); + + if (doc->use_ESPshowpage) + { + WriteLabels(Orientation); + puts("ESPshowpage"); + } + } + } + + /* + * Restore the old showpage operator as needed... + */ + + if (doc->use_ESPshowpage) + puts("userdict/showpage/ESPshowpage load put\n"); +} + + +/* + * 'copy_page()' - Copy a page description. + * + * This function expects "line" to be filled with a %%Page comment line. + * On return, "line" will contain the next line in the file, if any. + */ + +static ssize_t /* O - Length of next line */ +copy_page(cups_file_t *fp, /* I - File to read from */ + pstops_doc_t *doc, /* I - Document info */ + ppd_file_t *ppd, /* I - PPD file */ + int number, /* I - Current page number */ + char *line, /* I - Line buffer */ + ssize_t linelen, /* I - Length of initial line */ + size_t linesize) /* I - Size of line buffer */ +{ + char label[256], /* Page label string */ + *ptr; /* Pointer into line */ + int level; /* Embedded document level */ + pstops_page_t *pageinfo; /* Page information */ + int first_page; /* First page on N-up output? */ + int has_page_setup = 0; /* Does the page have %%Begin/EndPageSetup? */ + int bounding_box[4]; /* PageBoundingBox */ + + + /* + * Get the page label for this page... + */ + + first_page = is_first_page(number); + + if (!parse_text(line + 7, &ptr, label, sizeof(label))) + { + fputs("DEBUG: There was a bad %%Page: comment in the file.\n", stderr); + label[0] = '\0'; + number = doc->page; + } + else if (strtol(ptr, &ptr, 10) == LONG_MAX || !isspace(*ptr & 255)) + { + fputs("DEBUG: There was a bad %%Page: comment in the file.\n", stderr); + number = doc->page; + } + + /* + * Create or update the current output page... + */ + + if (first_page) + pageinfo = add_page(doc, label); + else + pageinfo = (pstops_page_t *)cupsArrayLast(doc->pages); + + /* + * Handle first page override... + */ + + if (doc->ap_input_slot || doc->ap_manual_feed) + { + if (doc->page == 1) + { + /* + * First page/sheet gets AP_FIRSTPAGE_* options... + */ + + pageinfo->num_options = cupsAddOption("InputSlot", doc->ap_input_slot, + pageinfo->num_options, + &(pageinfo->options)); + pageinfo->num_options = cupsAddOption("ManualFeed", + doc->ap_input_slot ? "False" : + doc->ap_manual_feed, + pageinfo->num_options, + &(pageinfo->options)); + pageinfo->num_options = cupsAddOption("MediaColor", doc->ap_media_color, + pageinfo->num_options, + &(pageinfo->options)); + pageinfo->num_options = cupsAddOption("MediaType", doc->ap_media_type, + pageinfo->num_options, + &(pageinfo->options)); + pageinfo->num_options = cupsAddOption("PageRegion", doc->ap_page_region, + pageinfo->num_options, + &(pageinfo->options)); + pageinfo->num_options = cupsAddOption("PageSize", doc->ap_page_size, + pageinfo->num_options, + &(pageinfo->options)); + } + else if (doc->page == (Duplex + 2)) + { + /* + * Second page/sheet gets default options... + */ + + pageinfo->num_options = cupsAddOption("InputSlot", doc->input_slot, + pageinfo->num_options, + &(pageinfo->options)); + pageinfo->num_options = cupsAddOption("ManualFeed", + doc->input_slot ? "False" : + doc->manual_feed, + pageinfo->num_options, + &(pageinfo->options)); + pageinfo->num_options = cupsAddOption("MediaColor", doc->media_color, + pageinfo->num_options, + &(pageinfo->options)); + pageinfo->num_options = cupsAddOption("MediaType", doc->media_type, + pageinfo->num_options, + &(pageinfo->options)); + pageinfo->num_options = cupsAddOption("PageRegion", doc->page_region, + pageinfo->num_options, + &(pageinfo->options)); + pageinfo->num_options = cupsAddOption("PageSize", doc->page_size, + pageinfo->num_options, + &(pageinfo->options)); + } + } + + /* + * Scan comments until we see something other than %%Page*: or + * %%Include*... + */ + + memcpy(bounding_box, doc->bounding_box, sizeof(bounding_box)); + + while ((linelen = cupsFileGetLine(fp, line, linesize)) > 0) + { + if (!strncmp(line, "%%PageBoundingBox:", 18)) + { + /* + * %%PageBoundingBox: llx lly urx ury + */ + + if (sscanf(line + 18, "%d%d%d%d", bounding_box + 0, + bounding_box + 1, bounding_box + 2, + bounding_box + 3) != 4) + { + fputs("DEBUG: There was a bad %%PageBoundingBox: comment in the file.\n", stderr); + memcpy(bounding_box, doc->bounding_box, + sizeof(bounding_box)); + } + else if (doc->number_up == 1 && !doc->fitplot && Orientation) + { + int temp_bbox[4]; /* Temporary bounding box */ + + + memcpy(temp_bbox, bounding_box, sizeof(temp_bbox)); + + fprintf(stderr, "DEBUG: Orientation = %d\n", Orientation); + fprintf(stderr, "DEBUG: original bounding_box = [ %d %d %d %d ]\n", + bounding_box[0], bounding_box[1], + bounding_box[2], bounding_box[3]); + fprintf(stderr, "DEBUG: PageWidth = %.1f, PageLength = %.1f\n", + PageWidth, PageLength); + + switch (Orientation) + { + case 1 : /* Landscape */ + bounding_box[0] = PageLength - temp_bbox[3]; + bounding_box[1] = temp_bbox[0]; + bounding_box[2] = PageLength - temp_bbox[1]; + bounding_box[3] = temp_bbox[2]; + break; + + case 2 : /* Reverse Portrait */ + bounding_box[0] = PageWidth - temp_bbox[2]; + bounding_box[1] = PageLength - temp_bbox[3]; + bounding_box[2] = PageWidth - temp_bbox[0]; + bounding_box[3] = PageLength - temp_bbox[1]; + break; + + case 3 : /* Reverse Landscape */ + bounding_box[0] = temp_bbox[1]; + bounding_box[1] = PageWidth - temp_bbox[2]; + bounding_box[2] = temp_bbox[3]; + bounding_box[3] = PageWidth - temp_bbox[0]; + break; + } + + fprintf(stderr, "DEBUG: updated bounding_box = [ %d %d %d %d ]\n", + bounding_box[0], bounding_box[1], + bounding_box[2], bounding_box[3]); + } + } +#if 0 + else if (!strncmp(line, "%%PageCustomColors:", 19) || + !strncmp(line, "%%PageMedia:", 12) || + !strncmp(line, "%%PageOrientation:", 18) || + !strncmp(line, "%%PageProcessColors:", 20) || + !strncmp(line, "%%PageRequirements:", 18) || + !strncmp(line, "%%PageResources:", 16)) + { + /* + * Copy literal... + */ + } +#endif /* 0 */ + else if (!strncmp(line, "%%PageCustomColors:", 19)) + { + /* + * %%PageCustomColors: ... + */ + } + else if (!strncmp(line, "%%PageMedia:", 12)) + { + /* + * %%PageMedia: ... + */ + } + else if (!strncmp(line, "%%PageOrientation:", 18)) + { + /* + * %%PageOrientation: ... + */ + } + else if (!strncmp(line, "%%PageProcessColors:", 20)) + { + /* + * %%PageProcessColors: ... + */ + } + else if (!strncmp(line, "%%PageRequirements:", 18)) + { + /* + * %%PageRequirements: ... + */ + } + else if (!strncmp(line, "%%PageResources:", 16)) + { + /* + * %%PageResources: ... + */ + } + else if (!strncmp(line, "%%IncludeFeature:", 17)) + { + /* + * %%IncludeFeature: *MainKeyword OptionKeyword + */ + + if (doc->number_up == 1 &&!doc->fitplot) + pageinfo->num_options = include_feature(ppd, line, + pageinfo->num_options, + &(pageinfo->options)); + } + else if (!strncmp(line, "%%BeginPageSetup", 16)) + { + has_page_setup = 1; + break; + } + else + break; + } + + if (doc->number_up == 1) + { + /* + * Update the document's composite and page bounding box... + */ + + memcpy(pageinfo->bounding_box, bounding_box, + sizeof(pageinfo->bounding_box)); + + if (bounding_box[0] < doc->new_bounding_box[0]) + doc->new_bounding_box[0] = bounding_box[0]; + if (bounding_box[1] < doc->new_bounding_box[1]) + doc->new_bounding_box[1] = bounding_box[1]; + if (bounding_box[2] > doc->new_bounding_box[2]) + doc->new_bounding_box[2] = bounding_box[2]; + if (bounding_box[3] > doc->new_bounding_box[3]) + doc->new_bounding_box[3] = bounding_box[3]; + } + + /* + * Output the page header as needed... + */ + + if (!doc->slow_order && first_page) + { + if (!ppd || !ppd->num_filters) + fprintf(stderr, "PAGE: %d %d\n", doc->page, + doc->slow_collate ? 1 : doc->copies); + + if (doc->number_up > 1) + { + printf("%%%%Page: (%d) %d\n", doc->page, doc->page); + printf("%%%%PageBoundingBox: %.0f %.0f %.0f %.0f\n", + PageLeft, PageBottom, PageRight, PageTop); + } + else + { + printf("%%%%Page: %s %d\n", pageinfo->label, doc->page); + printf("%%%%PageBoundingBox: %d %d %d %d\n", + pageinfo->bounding_box[0], pageinfo->bounding_box[1], + pageinfo->bounding_box[2], pageinfo->bounding_box[3]); + } + } + + /* + * Copy any page setup commands... + */ + + if (first_page) + doc_puts(doc, "%%BeginPageSetup\n"); + + if (has_page_setup) + { + int feature = 0; /* In a Begin/EndFeature block? */ + + while ((linelen = cupsFileGetLine(fp, line, linesize)) > 0) + { + if (!strncmp(line, "%%EndPageSetup", 14)) + break; + else if (!strncmp(line, "%%BeginFeature:", 15)) + { + feature = 1; + + if (doc->number_up > 1 || doc->fitplot) + continue; + } + else if (!strncmp(line, "%%EndFeature", 12)) + { + feature = 0; + + if (doc->number_up > 1 || doc->fitplot) + continue; + } + else if (!strncmp(line, "%%IncludeFeature:", 17)) + { + pageinfo->num_options = include_feature(ppd, line, + pageinfo->num_options, + &(pageinfo->options)); + continue; + } + else if (!strncmp(line, "%%Include", 9)) + continue; + + if (line[0] != '%' && !feature) + break; + + if (!feature || (doc->number_up == 1 && !doc->fitplot)) + doc_write(doc, line, linelen); + } + + /* + * Skip %%EndPageSetup... + */ + + if (linelen > 0 && !strncmp(line, "%%EndPageSetup", 14)) + linelen = cupsFileGetLine(fp, line, linesize); + } + + if (first_page) + { + char *page_setup; /* PageSetup commands to send */ + + + if (pageinfo->num_options > 0) + write_options(doc, ppd, pageinfo->num_options, pageinfo->options); + + /* + * Output commands for the current page... + */ + + page_setup = ppdEmitString(ppd, PPD_ORDER_PAGE, 0); + + if (page_setup) + { + doc_puts(doc, page_setup); + free(page_setup); + } + } + + /* + * Prep for the start of the page description... + */ + + start_nup(doc, number, 1, bounding_box); + + if (first_page) + doc_puts(doc, "%%EndPageSetup\n"); + + /* + * Read the rest of the page description... + */ + + level = 0; + + do + { + if (level == 0 && + (!strncmp(line, "%%Page:", 7) || + !strncmp(line, "%%Trailer", 9) || + !strncmp(line, "%%EOF", 5))) + break; + else if (!strncmp(line, "%%BeginDocument", 15) || + !strncmp(line, "%ADO_BeginApplication", 21)) + { + doc_write(doc, line, linelen); + + level ++; + } + else if ((!strncmp(line, "%%EndDocument", 13) || + !strncmp(line, "%ADO_EndApplication", 19)) && level > 0) + { + doc_write(doc, line, linelen); + + level --; + } + else if (!strncmp(line, "%%BeginBinary:", 14) || + (!strncmp(line, "%%BeginData:", 12) && + !strstr(line, "ASCII") && !strstr(line, "Hex"))) + { + /* + * Copy binary data... + */ + + int bytes; /* Bytes of data */ + + + doc_write(doc, line, linelen); + + bytes = atoi(strchr(line, ':') + 1); + + while (bytes > 0) + { + if (bytes > linesize) + linelen = cupsFileRead(fp, line, linesize); + else + linelen = cupsFileRead(fp, line, bytes); + + if (linelen < 1) + { + line[0] = '\0'; + perror("ERROR: Early end-of-file while reading binary data"); + return (0); + } + + doc_write(doc, line, linelen); + + bytes -= linelen; + } + } + else + doc_write(doc, line, linelen); + } + while ((linelen = cupsFileGetLine(fp, line, linesize)) > 0); + + /* + * Finish up this page and return... + */ + + end_nup(doc, number); + + pageinfo->length = cupsFileTell(doc->temp) - pageinfo->offset; + + return (linelen); +} + + +/* + * 'copy_prolog()' - Copy the document prolog section. + * + * This function expects "line" to be filled with a %%BeginProlog comment line. + * On return, "line" will contain the next line in the file, if any. + */ + +static ssize_t /* O - Length of next line */ +copy_prolog(cups_file_t *fp, /* I - File to read from */ + pstops_doc_t *doc, /* I - Document info */ + ppd_file_t *ppd, /* I - PPD file */ + char *line, /* I - Line buffer */ + ssize_t linelen, /* I - Length of initial line */ + size_t linesize) /* I - Size of line buffer */ +{ + while (strncmp(line, "%%BeginProlog", 13)) + { + if (!strncmp(line, "%%BeginSetup", 12) || !strncmp(line, "%%Page:", 7)) + break; + + doc_write(doc, line, linelen); + + if ((linelen = cupsFileGetLine(fp, line, linesize)) == 0) + break; + } + + doc_puts(doc, "%%BeginProlog\n"); + + do_prolog(doc, ppd); + + if (!strncmp(line, "%%BeginProlog", 13)) + { + while ((linelen = cupsFileGetLine(fp, line, linesize)) > 0) + { + if (!strncmp(line, "%%EndProlog", 11) || + !strncmp(line, "%%BeginSetup", 12) || + !strncmp(line, "%%Page:", 7)) + break; + + doc_write(doc, line, linelen); + } + + if (!strncmp(line, "%%EndProlog", 11)) + linelen = cupsFileGetLine(fp, line, linesize); + else + fputs("DEBUG: The %%EndProlog comment is missing.\n", stderr); + } + + doc_puts(doc, "%%EndProlog\n"); + + return (linelen); +} + + +/* + * 'copy_setup()' - Copy the document setup section. + * + * This function expects "line" to be filled with a %%BeginSetup comment line. + * On return, "line" will contain the next line in the file, if any. + */ + +static ssize_t /* O - Length of next line */ +copy_setup(cups_file_t *fp, /* I - File to read from */ + pstops_doc_t *doc, /* I - Document info */ + ppd_file_t *ppd, /* I - PPD file */ + char *line, /* I - Line buffer */ + ssize_t linelen, /* I - Length of initial line */ + size_t linesize) /* I - Size of line buffer */ +{ + int num_options; /* Number of options */ + cups_option_t *options; /* Options */ + + + while (strncmp(line, "%%BeginSetup", 12)) + { + if (!strncmp(line, "%%Page:", 7)) + break; + + doc_write(doc, line, linelen); + + if ((linelen = cupsFileGetLine(fp, line, linesize)) == 0) + break; + } + + doc_puts(doc, "%%BeginSetup\n"); + + do_setup(doc, ppd); + + num_options = 0; + options = NULL; + + if (!strncmp(line, "%%BeginSetup", 12)) + { + while (strncmp(line, "%%EndSetup", 10)) + { + if (!strncmp(line, "%%Page:", 7)) + break; + else if (!strncmp(line, "%%IncludeFeature:", 17)) + { + /* + * %%IncludeFeature: *MainKeyword OptionKeyword + */ + + if (doc->number_up == 1 && !doc->fitplot) + num_options = include_feature(ppd, line, num_options, &options); + } + else if (strncmp(line, "%%BeginSetup", 12)) + doc_write(doc, line, linelen); + + if ((linelen = cupsFileGetLine(fp, line, linesize)) == 0) + break; + } + + if (!strncmp(line, "%%EndSetup", 10)) + linelen = cupsFileGetLine(fp, line, linesize); + else + fputs("DEBUG: The %%EndSetup comment is missing.\n", stderr); + } + + if (num_options > 0) + { + write_options(doc, ppd, num_options, options); + cupsFreeOptions(num_options, options); + } + + doc_puts(doc, "%%EndSetup\n"); + + return (linelen); +} + + +/* + * 'copy_trailer()' - Copy the document trailer. + * + * This function expects "line" to be filled with a %%Trailer comment line. + * On return, "line" will contain the next line in the file, if any. + */ + +static ssize_t /* O - Length of next line */ +copy_trailer(cups_file_t *fp, /* I - File to read from */ + pstops_doc_t *doc, /* I - Document info */ + ppd_file_t *ppd, /* I - PPD file */ + int number, /* I - Number of pages */ + char *line, /* I - Line buffer */ + ssize_t linelen, /* I - Length of initial line */ + size_t linesize) /* I - Size of line buffer */ +{ + /* + * Write the trailer comments... + */ + + puts("%%Trailer"); + + while (linelen > 0) + { + if (!strncmp(line, "%%EOF", 5)) + break; + else if (strncmp(line, "%%Trailer", 9) && + strncmp(line, "%%Pages:", 8) && + strncmp(line, "%%BoundingBox:", 14)) + fwrite(line, 1, linelen, stdout); + + linelen = cupsFileGetLine(fp, line, linesize); + } + + fprintf(stderr, "DEBUG: Wrote %d pages...\n", number); + + printf("%%%%Pages: %d\n", number); + if (doc->number_up > 1 || doc->fitplot) + printf("%%%%BoundingBox: %.0f %.0f %.0f %.0f\n", + PageLeft, PageBottom, PageRight, PageTop); + else + printf("%%%%BoundingBox: %d %d %d %d\n", + doc->new_bounding_box[0], doc->new_bounding_box[1], + doc->new_bounding_box[2], doc->new_bounding_box[3]); + + return (linelen); +} + + +/* + * 'do_prolog()' - Send the necessary document prolog commands. + */ + +static void +do_prolog(pstops_doc_t *doc, /* I - Document information */ + ppd_file_t *ppd) /* I - PPD file */ +{ + char *ps; /* PS commands */ + + + /* + * Send the document prolog commands... + */ + + if (ppd && ppd->patches) + { + doc_puts(doc, "%%BeginFeature: *JobPatchFile 1\n"); + doc_puts(doc, ppd->patches); + doc_puts(doc, "\n%%EndFeature\n"); + } + + if ((ps = ppdEmitString(ppd, PPD_ORDER_PROLOG, 0.0)) != NULL) + { + doc_puts(doc, ps); + free(ps); + } + + /* + * Define ESPshowpage here so that applications that define their + * own procedure to do a showpage pick it up... + */ + + if (doc->use_ESPshowpage) + doc_puts(doc, "userdict/ESPshowpage/showpage load put\n" + "userdict/showpage{}put\n"); +} + + +/* + * 'do_setup()' - Send the necessary document setup commands. + */ + +static void +do_setup(pstops_doc_t *doc, /* I - Document information */ + ppd_file_t *ppd) /* I - PPD file */ +{ + char *ps; /* PS commands */ + + + /* + * Disable CTRL-D so that embedded files don't cause printing + * errors... + */ + + doc_puts(doc, "% Disable CTRL-D as an end-of-file marker...\n"); + doc_puts(doc, "userdict dup(\\004)cvn{}put (\\004\\004)cvn{}put\n"); + + /* + * Mark job options... + */ + + cupsMarkOptions(ppd, doc->num_options, doc->options); + + /* + * Send all the printer-specific setup commands... + */ + + if ((ps = ppdEmitString(ppd, PPD_ORDER_DOCUMENT, 0.0)) != NULL) + { + doc_puts(doc, ps); + free(ps); + } + + if ((ps = ppdEmitString(ppd, PPD_ORDER_ANY, 0.0)) != NULL) + { + doc_puts(doc, ps); + free(ps); + } + + /* + * Set the number of copies for the job... + */ + + if (doc->copies != 1 && (!doc->collate || !doc->slow_collate)) + { + doc_printf(doc, "%%RBIBeginNonPPDFeature: *NumCopies %d\n", doc->copies); + doc_printf(doc, + "%d/languagelevel where{pop languagelevel 2 ge}{false}ifelse\n" + "{1 dict begin/NumCopies exch def currentdict end " + "setpagedevice}\n" + "{userdict/#copies 3 -1 roll put}ifelse\n", doc->copies); + doc_puts(doc, "%RBIEndNonPPDFeature\n"); + } + + /* + * If we are doing N-up printing, disable setpagedevice... + */ + + if (doc->number_up > 1) + { + doc_puts(doc, "userdict/CUPSsetpagedevice/setpagedevice load put\n"); + doc_puts(doc, "userdict/setpagedevice{pop}bind put\n"); + } + + /* + * Make sure we have rectclip and rectstroke procedures of some sort... + */ + + doc_puts(doc, + "% x y w h ESPrc - Clip to a rectangle.\n" + "userdict/ESPrc/rectclip where{pop/rectclip load}\n" + "{{newpath 4 2 roll moveto 1 index 0 rlineto 0 exch rlineto\n" + "neg 0 rlineto closepath clip newpath}bind}ifelse put\n"); + + doc_puts(doc, + "% x y w h ESPrf - Fill a rectangle.\n" + "userdict/ESPrf/rectfill where{pop/rectfill load}\n" + "{{gsave newpath 4 2 roll moveto 1 index 0 rlineto 0 exch rlineto\n" + "neg 0 rlineto closepath fill grestore}bind}ifelse put\n"); + + doc_puts(doc, + "% x y w h ESPrs - Stroke a rectangle.\n" + "userdict/ESPrs/rectstroke where{pop/rectstroke load}\n" + "{{gsave newpath 4 2 roll moveto 1 index 0 rlineto 0 exch rlineto\n" + "neg 0 rlineto closepath stroke grestore}bind}ifelse put\n"); + + /* + * Write the page and label prologs... + */ + + if (doc->number_up == 2 || doc->number_up == 6) + { + /* + * For 2- and 6-up output, rotate the labels to match the orientation + * of the pages... + */ + + if (Orientation & 1) + write_label_prolog(doc, doc->page_label, PageBottom, + PageWidth - PageLength + PageTop, PageLength); + else + write_label_prolog(doc, doc->page_label, PageLeft, PageRight, + PageLength); + } + else + write_label_prolog(doc, doc->page_label, PageBottom, PageTop, PageWidth); +} + + +/* + * 'doc_printf()' - Send a formatted string to stdout and/or the temp file. + * + * This function should be used for all page-level output that is affected + * by ordering, collation, etc. + */ + +static void +doc_printf(pstops_doc_t *doc, /* I - Document information */ + const char *format, /* I - Printf-style format string */ + ...) /* I - Additional arguments as needed */ +{ + va_list ap; /* Pointer to arguments */ + char buffer[1024]; /* Output buffer */ + size_t bytes; /* Number of bytes to write */ + + + va_start(ap, format); + bytes = vsnprintf(buffer, sizeof(buffer), format, ap); + va_end(ap); + + if (bytes > sizeof(buffer)) + { + _cupsLangPrintFilter(stderr, "ERROR", + _("Buffer overflow detected, aborting.")); + exit(1); + } + + doc_write(doc, buffer, bytes); +} + + +/* + * 'doc_puts()' - Send a nul-terminated string to stdout and/or the temp file. + * + * This function should be used for all page-level output that is affected + * by ordering, collation, etc. + */ + +static void +doc_puts(pstops_doc_t *doc, /* I - Document information */ + const char *s) /* I - String to send */ +{ + doc_write(doc, s, strlen(s)); +} + + +/* + * 'doc_write()' - Send data to stdout and/or the temp file. + */ + +static void +doc_write(pstops_doc_t *doc, /* I - Document information */ + const char *s, /* I - Data to send */ + size_t len) /* I - Number of bytes to send */ +{ + if (!doc->slow_order) + fwrite(s, 1, len, stdout); + + if (doc->temp) + cupsFileWrite(doc->temp, s, len); +} + + +/* + * 'end_nup()' - End processing for N-up printing. + */ + +static void +end_nup(pstops_doc_t *doc, /* I - Document information */ + int number) /* I - Page number */ +{ + if (doc->number_up > 1) + doc_puts(doc, "userdict/ESPsave get restore\n"); + + switch (doc->number_up) + { + case 1 : + if (doc->use_ESPshowpage) + { + write_labels(doc, Orientation); + doc_puts(doc, "ESPshowpage\n"); + } + break; + + case 2 : + case 6 : + if (is_last_page(number) && doc->use_ESPshowpage) + { + if (Orientation & 1) + { + /* + * Rotate the labels back to portrait... + */ + + write_labels(doc, Orientation - 1); + } + else if (Orientation == 0) + { + /* + * Rotate the labels to landscape... + */ + + write_labels(doc, doc->normal_landscape ? 1 : 3); + } + else + { + /* + * Rotate the labels to landscape... + */ + + write_labels(doc, doc->normal_landscape ? 3 : 1); + } + + doc_puts(doc, "ESPshowpage\n"); + } + break; + + default : + if (is_last_page(number) && doc->use_ESPshowpage) + { + write_labels(doc, Orientation); + doc_puts(doc, "ESPshowpage\n"); + } + break; + } + + fflush(stdout); +} + + +/* + * 'include_feature()' - Include a printer option/feature command. + */ + +static int /* O - New number of options */ +include_feature( + ppd_file_t *ppd, /* I - PPD file */ + const char *line, /* I - DSC line */ + int num_options, /* I - Number of options */ + cups_option_t **options) /* IO - Options */ +{ + char name[255], /* Option name */ + value[255]; /* Option value */ + ppd_option_t *option; /* Option in file */ + + + /* + * Get the "%%IncludeFeature: *Keyword OptionKeyword" values... + */ + + if (sscanf(line + 17, "%254s%254s", name, value) != 2) + { + fputs("DEBUG: The %%IncludeFeature: comment is not valid.\n", stderr); + return (num_options); + } + + /* + * Find the option and choice... + */ + + if ((option = ppdFindOption(ppd, name + 1)) == NULL) + { + _cupsLangPrintFilter(stderr, "WARNING", _("Unknown option \"%s\"."), + name + 1); + return (num_options); + } + + if (option->section == PPD_ORDER_EXIT || + option->section == PPD_ORDER_JCL) + { + _cupsLangPrintFilter(stderr, "WARNING", + _("Option \"%s\" cannot be included via " + "%%%%IncludeFeature."), name + 1); + return (num_options); + } + + if (!ppdFindChoice(option, value)) + { + _cupsLangPrintFilter(stderr, "WARNING", + _("Unknown choice \"%s\" for option \"%s\"."), + value, name + 1); + return (num_options); + } + + /* + * Add the option to the option array and return... + */ + + return (cupsAddOption(name + 1, value, num_options, options)); +} + + +/* + * 'parse_text()' - Parse a text value in a comment. + * + * This function parses a DSC text value as defined on page 36 of the + * DSC specification. Text values are either surrounded by parenthesis + * or whitespace-delimited. + * + * The value returned is the literal characters for the entire text + * string, including any parenthesis and escape characters. + */ + +static char * /* O - Value or NULL on error */ +parse_text(const char *start, /* I - Start of text value */ + char **end, /* O - End of text value */ + char *buffer, /* I - Buffer */ + size_t bufsize) /* I - Size of buffer */ +{ + char *bufptr, /* Pointer in buffer */ + *bufend; /* End of buffer */ + int level; /* Parenthesis level */ + + + /* + * Skip leading whitespace... + */ + + while (isspace(*start & 255)) + start ++; + + /* + * Then copy the value... + */ + + level = 0; + bufptr = buffer; + bufend = buffer + bufsize - 1; + + while (bufptr < bufend) + { + if (isspace(*start & 255) && !level) + break; + + *bufptr++ = *start; + + if (*start == '(') + level ++; + else if (*start == ')') + { + if (!level) + { + start ++; + break; + } + else + level --; + } + else if (*start == '\\') + { + /* + * Copy escaped character... + */ + + int i; /* Looping var */ + + + for (i = 1; + i <= 3 && isdigit(start[i] & 255) && bufptr < bufend; + *bufptr++ = start[i], i ++); + } + + start ++; + } + + *bufptr = '\0'; + + /* + * Return the value and new pointer into the line... + */ + + if (end) + *end = (char *)start; + + if (bufptr == bufend) + return (NULL); + else + return (buffer); +} + + +/* + * 'set_pstops_options()' - Set pstops options. + */ + +static void +set_pstops_options( + pstops_doc_t *doc, /* I - Document information */ + ppd_file_t *ppd, /* I - PPD file */ + char *argv[], /* I - Command-line arguments */ + int num_options, /* I - Number of options */ + cups_option_t *options) /* I - Options */ +{ + const char *val; /* Option value */ + int intval; /* Integer option value */ + ppd_attr_t *attr; /* PPD attribute */ + ppd_option_t *option; /* PPD option */ + ppd_choice_t *choice; /* PPD choice */ + const char *content_type; /* Original content type */ + + + /* + * Initialize document information structure... + */ + + memset(doc, 0, sizeof(pstops_doc_t)); + + doc->job_id = atoi(argv[1]); + doc->user = argv[2]; + doc->title = argv[3]; + doc->copies = atoi(argv[4]); + + if (ppd && ppd->landscape > 0) + doc->normal_landscape = 1; + + doc->bounding_box[0] = (int)PageLeft; + doc->bounding_box[1] = (int)PageBottom; + doc->bounding_box[2] = (int)PageRight; + doc->bounding_box[3] = (int)PageTop; + + doc->new_bounding_box[0] = INT_MAX; + doc->new_bounding_box[1] = INT_MAX; + doc->new_bounding_box[2] = INT_MIN; + doc->new_bounding_box[3] = INT_MIN; + + /* + * AP_FIRSTPAGE_* and the corresponding non-first-page options. + */ + + doc->ap_input_slot = cupsGetOption("AP_FIRSTPAGE_InputSlot", num_options, + options); + doc->ap_manual_feed = cupsGetOption("AP_FIRSTPAGE_ManualFeed", num_options, + options); + doc->ap_media_color = cupsGetOption("AP_FIRSTPAGE_MediaColor", num_options, + options); + doc->ap_media_type = cupsGetOption("AP_FIRSTPAGE_MediaType", num_options, + options); + doc->ap_page_region = cupsGetOption("AP_FIRSTPAGE_PageRegion", num_options, + options); + doc->ap_page_size = cupsGetOption("AP_FIRSTPAGE_PageSize", num_options, + options); + + if ((choice = ppdFindMarkedChoice(ppd, "InputSlot")) != NULL) + doc->input_slot = choice->choice; + if ((choice = ppdFindMarkedChoice(ppd, "ManualFeed")) != NULL) + doc->manual_feed = choice->choice; + if ((choice = ppdFindMarkedChoice(ppd, "MediaColor")) != NULL) + doc->media_color = choice->choice; + if ((choice = ppdFindMarkedChoice(ppd, "MediaType")) != NULL) + doc->media_type = choice->choice; + if ((choice = ppdFindMarkedChoice(ppd, "PageRegion")) != NULL) + doc->page_region = choice->choice; + if ((choice = ppdFindMarkedChoice(ppd, "PageSize")) != NULL) + doc->page_size = choice->choice; + + /* + * collate, multiple-document-handling + */ + + if ((val = cupsGetOption("multiple-document-handling", num_options, options)) != NULL) + { + /* + * This IPP attribute is unnecessarily complicated... + * + * single-document, separate-documents-collated-copies, and + * single-document-new-sheet all require collated copies. + * + * separate-documents-uncollated-copies allows for uncollated copies. + */ + + doc->collate = _cups_strcasecmp(val, "separate-documents-uncollated-copies") != 0; + } + + if ((val = cupsGetOption("Collate", num_options, options)) != NULL && + (!_cups_strcasecmp(val, "true") ||!_cups_strcasecmp(val, "on") || + !_cups_strcasecmp(val, "yes"))) + doc->collate = 1; + + /* + * emit-jcl + */ + + if ((val = cupsGetOption("emit-jcl", num_options, options)) != NULL && + (!_cups_strcasecmp(val, "false") || !_cups_strcasecmp(val, "off") || + !_cups_strcasecmp(val, "no") || !strcmp(val, "0"))) + doc->emit_jcl = 0; + else + doc->emit_jcl = 1; + + /* + * fitplot/fit-to-page/ipp-attribute-fidelity + * + * (Only for original PostScript content) + */ + + if ((content_type = getenv("CONTENT_TYPE")) == NULL) + content_type = "application/postscript"; + + if (!_cups_strcasecmp(content_type, "application/postscript")) + { + if ((val = cupsGetOption("fitplot", num_options, options)) != NULL && + !_cups_strcasecmp(val, "true")) + doc->fitplot = 1; + else if ((val = cupsGetOption("fit-to-page", num_options, options)) != NULL && + !_cups_strcasecmp(val, "true")) + doc->fitplot = 1; + else if ((val = cupsGetOption("ipp-attribute-fidelity", num_options, + options)) != NULL && + !_cups_strcasecmp(val, "true")) + doc->fitplot = 1; + } + + /* + * mirror/MirrorPrint + */ + + if ((choice = ppdFindMarkedChoice(ppd, "MirrorPrint")) != NULL) + { + val = choice->choice; + choice->marked = 0; + } + else + val = cupsGetOption("mirror", num_options, options); + + if (val && (!_cups_strcasecmp(val, "true") || !_cups_strcasecmp(val, "on") || + !_cups_strcasecmp(val, "yes"))) + doc->mirror = 1; + + /* + * number-up + */ + + if ((val = cupsGetOption("number-up", num_options, options)) != NULL) + { + switch (intval = atoi(val)) + { + case 1 : + case 2 : + case 4 : + case 6 : + case 9 : + case 16 : + doc->number_up = intval; + break; + default : + _cupsLangPrintFilter(stderr, "ERROR", + _("Unsupported number-up value %d, using " + "number-up=1."), intval); + doc->number_up = 1; + break; + } + } + else + doc->number_up = 1; + + /* + * number-up-layout + */ + + if ((val = cupsGetOption("number-up-layout", num_options, options)) != NULL) + { + if (!_cups_strcasecmp(val, "lrtb")) + doc->number_up_layout = PSTOPS_LAYOUT_LRTB; + else if (!_cups_strcasecmp(val, "lrbt")) + doc->number_up_layout = PSTOPS_LAYOUT_LRBT; + else if (!_cups_strcasecmp(val, "rltb")) + doc->number_up_layout = PSTOPS_LAYOUT_RLTB; + else if (!_cups_strcasecmp(val, "rlbt")) + doc->number_up_layout = PSTOPS_LAYOUT_RLBT; + else if (!_cups_strcasecmp(val, "tblr")) + doc->number_up_layout = PSTOPS_LAYOUT_TBLR; + else if (!_cups_strcasecmp(val, "tbrl")) + doc->number_up_layout = PSTOPS_LAYOUT_TBRL; + else if (!_cups_strcasecmp(val, "btlr")) + doc->number_up_layout = PSTOPS_LAYOUT_BTLR; + else if (!_cups_strcasecmp(val, "btrl")) + doc->number_up_layout = PSTOPS_LAYOUT_BTRL; + else + { + _cupsLangPrintFilter(stderr, "ERROR", + _("Unsupported number-up-layout value %s, using " + "number-up-layout=lrtb."), val); + doc->number_up_layout = PSTOPS_LAYOUT_LRTB; + } + } + else + doc->number_up_layout = PSTOPS_LAYOUT_LRTB; + + /* + * OutputOrder + */ + + if ((val = cupsGetOption("OutputOrder", num_options, options)) != NULL) + { + if (!_cups_strcasecmp(val, "Reverse")) + doc->output_order = 1; + } + else if (ppd) + { + /* + * Figure out the right default output order from the PPD file... + */ + + if ((choice = ppdFindMarkedChoice(ppd, "OutputBin")) != NULL && + (attr = ppdFindAttr(ppd, "PageStackOrder", choice->choice)) != NULL && + attr->value) + doc->output_order = !_cups_strcasecmp(attr->value, "Reverse"); + else if ((attr = ppdFindAttr(ppd, "DefaultOutputOrder", NULL)) != NULL && + attr->value) + doc->output_order = !_cups_strcasecmp(attr->value, "Reverse"); + } + + /* + * page-border + */ + + if ((val = cupsGetOption("page-border", num_options, options)) != NULL) + { + if (!_cups_strcasecmp(val, "none")) + doc->page_border = PSTOPS_BORDERNONE; + else if (!_cups_strcasecmp(val, "single")) + doc->page_border = PSTOPS_BORDERSINGLE; + else if (!_cups_strcasecmp(val, "single-thick")) + doc->page_border = PSTOPS_BORDERSINGLE2; + else if (!_cups_strcasecmp(val, "double")) + doc->page_border = PSTOPS_BORDERDOUBLE; + else if (!_cups_strcasecmp(val, "double-thick")) + doc->page_border = PSTOPS_BORDERDOUBLE2; + else + { + _cupsLangPrintFilter(stderr, "ERROR", + _("Unsupported page-border value %s, using " + "page-border=none."), val); + doc->page_border = PSTOPS_BORDERNONE; + } + } + else + doc->page_border = PSTOPS_BORDERNONE; + + /* + * page-label + */ + + doc->page_label = cupsGetOption("page-label", num_options, options); + + /* + * page-ranges + */ + + doc->page_ranges = cupsGetOption("page-ranges", num_options, options); + + /* + * page-set + */ + + doc->page_set = cupsGetOption("page-set", num_options, options); + + /* + * Now figure out if we have to force collated copies, etc. + */ + + if (ppd && ppd->manual_copies && Duplex && doc->copies > 1) + { + /* + * Force collated copies when printing a duplexed document to + * a non-PS printer that doesn't do hardware copy generation. + * Otherwise the copies will end up on the front/back side of + * each page. + */ + + doc->collate = 1; + } + + /* + * See if we have to filter the fast or slow way... + */ + + if (doc->collate && doc->copies > 1) + { + /* + * See if we need to manually collate the pages... + */ + + doc->slow_collate = 1; + + if ((choice = ppdFindMarkedChoice(ppd, "Collate")) != NULL && + !_cups_strcasecmp(choice->choice, "True")) + { + /* + * Hardware collate option is selected, see if the option is + * conflicting - if not, collate in hardware. Otherwise, + * turn the hardware collate option off... + */ + + if ((option = ppdFindOption(ppd, "Collate")) != NULL && + !option->conflicted) + doc->slow_collate = 0; + else + ppdMarkOption(ppd, "Collate", "False"); + } + } + else + doc->slow_collate = 0; + + if (!ppdFindOption(ppd, "OutputOrder") && doc->output_order) + doc->slow_order = 1; + else + doc->slow_order = 0; + + if (Duplex && + (doc->slow_collate || doc->slow_order || + ((attr = ppdFindAttr(ppd, "cupsEvenDuplex", NULL)) != NULL && + attr->value && !_cups_strcasecmp(attr->value, "true")))) + doc->slow_duplex = 1; + else + doc->slow_duplex = 0; + + /* + * Create a temporary file for page data if we need to filter slowly... + */ + + if (doc->slow_order || doc->slow_collate) + { + if ((doc->temp = cupsTempFile2(doc->tempfile, + sizeof(doc->tempfile))) == NULL) + { + perror("DEBUG: Unable to create temporary file"); + exit(1); + } + } + + /* + * Figure out if we should use ESPshowpage or not... + */ + + if (doc->page_label || getenv("CLASSIFICATION") || doc->number_up > 1 || + doc->page_border) + { + /* + * Yes, use ESPshowpage... + */ + + doc->use_ESPshowpage = 1; + } + + fprintf(stderr, "DEBUG: slow_collate=%d, slow_duplex=%d, slow_order=%d\n", + doc->slow_collate, doc->slow_duplex, doc->slow_order); +} + + +/* + * 'skip_page()' - Skip past a page that won't be printed. + */ + +static ssize_t /* O - Length of next line */ +skip_page(cups_file_t *fp, /* I - File to read from */ + char *line, /* I - Line buffer */ + ssize_t linelen, /* I - Length of initial line */ + size_t linesize) /* I - Size of line buffer */ +{ + int level; /* Embedded document level */ + + + level = 0; + + while ((linelen = cupsFileGetLine(fp, line, linesize)) > 0) + { + if (level == 0 && + (!strncmp(line, "%%Page:", 7) || !strncmp(line, "%%Trailer", 9))) + break; + else if (!strncmp(line, "%%BeginDocument", 15) || + !strncmp(line, "%ADO_BeginApplication", 21)) + level ++; + else if ((!strncmp(line, "%%EndDocument", 13) || + !strncmp(line, "%ADO_EndApplication", 19)) && level > 0) + level --; + else if (!strncmp(line, "%%BeginBinary:", 14) || + (!strncmp(line, "%%BeginData:", 12) && + !strstr(line, "ASCII") && !strstr(line, "Hex"))) + { + /* + * Skip binary data... + */ + + int bytes; /* Bytes of data */ + + + bytes = atoi(strchr(line, ':') + 1); + + while (bytes > 0) + { + if (bytes > linesize) + linelen = cupsFileRead(fp, line, linesize); + else + linelen = cupsFileRead(fp, line, bytes); + + if (linelen < 1) + { + line[0] = '\0'; + perror("ERROR: Early end-of-file while reading binary data"); + return (0); + } + + bytes -= linelen; + } + } + } + + return (linelen); +} + + +/* + * 'start_nup()' - Start processing for N-up printing. + */ + +static void +start_nup(pstops_doc_t *doc, /* I - Document information */ + int number, /* I - Page number */ + int show_border, /* I - Show the border? */ + const int *bounding_box) /* I - BoundingBox value */ +{ + int pos; /* Position on page */ + int x, y; /* Relative position of subpage */ + float w, l, /* Width and length of subpage */ + tx, ty; /* Translation values for subpage */ + float pagew, /* Printable width of page */ + pagel; /* Printable height of page */ + int bboxx, /* BoundingBox X origin */ + bboxy, /* BoundingBox Y origin */ + bboxw, /* BoundingBox width */ + bboxl; /* BoundingBox height */ + float margin = 0; /* Current margin for border */ + + + if (doc->number_up > 1) + doc_puts(doc, "userdict/ESPsave save put\n"); + + pos = (number - 1) % doc->number_up; + pagew = PageRight - PageLeft; + pagel = PageTop - PageBottom; + + if (doc->fitplot) + { + bboxx = bounding_box[0]; + bboxy = bounding_box[1]; + bboxw = bounding_box[2] - bounding_box[0]; + bboxl = bounding_box[3] - bounding_box[1]; + } + else + { + bboxx = 0; + bboxy = 0; + bboxw = PageWidth; + bboxl = PageLength; + } + + fprintf(stderr, "DEBUG: pagew = %.1f, pagel = %.1f\n", pagew, pagel); + fprintf(stderr, "DEBUG: bboxx = %d, bboxy = %d, bboxw = %d, bboxl = %d\n", + bboxx, bboxy, bboxw, bboxl); + fprintf(stderr, "DEBUG: PageLeft = %.1f, PageRight = %.1f\n", + PageLeft, PageRight); + fprintf(stderr, "DEBUG: PageTop = %.1f, PageBottom = %.1f\n", + PageTop, PageBottom); + fprintf(stderr, "DEBUG: PageWidth = %.1f, PageLength = %.1f\n", + PageWidth, PageLength); + + switch (Orientation) + { + case 1 : /* Landscape */ + doc_printf(doc, "%.1f 0.0 translate 90 rotate\n", PageLength); + break; + case 2 : /* Reverse Portrait */ + doc_printf(doc, "%.1f %.1f translate 180 rotate\n", PageWidth, + PageLength); + break; + case 3 : /* Reverse Landscape */ + doc_printf(doc, "0.0 %.1f translate -90 rotate\n", PageWidth); + break; + } + + /* + * Mirror the page as needed... + */ + + if (doc->mirror) + doc_printf(doc, "%.1f 0.0 translate -1 1 scale\n", PageWidth); + + /* + * Offset and scale as necessary for fitplot/fit-to-page/number-up... + */ + + if (Duplex && doc->number_up > 1 && ((number / doc->number_up) & 1)) + doc_printf(doc, "%.1f %.1f translate\n", PageWidth - PageRight, PageBottom); + else if (doc->number_up > 1 || doc->fitplot) + doc_printf(doc, "%.1f %.1f translate\n", PageLeft, PageBottom); + + switch (doc->number_up) + { + default : + if (doc->fitplot) + { + w = pagew; + l = w * bboxl / bboxw; + + if (l > pagel) + { + l = pagel; + w = l * bboxw / bboxl; + } + + tx = 0.5 * (pagew - w); + ty = 0.5 * (pagel - l); + + doc_printf(doc, "%.1f %.1f translate %.3f %.3f scale\n", tx, ty, + w / bboxw, l / bboxl); + } + else + w = PageWidth; + break; + + case 2 : + if (Orientation & 1) + { + x = pos & 1; + + if (doc->number_up_layout & PSTOPS_LAYOUT_NEGATEY) + x = 1 - x; + + w = pagel; + l = w * bboxl / bboxw; + + if (l > (pagew * 0.5)) + { + l = pagew * 0.5; + w = l * bboxw / bboxl; + } + + tx = 0.5 * (pagew * 0.5 - l); + ty = 0.5 * (pagel - w); + + if (doc->normal_landscape) + doc_printf(doc, "0.0 %.1f translate -90 rotate\n", pagel); + else + doc_printf(doc, "%.1f 0.0 translate 90 rotate\n", pagew); + + doc_printf(doc, "%.1f %.1f translate %.3f %.3f scale\n", + ty, tx + pagew * 0.5 * x, w / bboxw, l / bboxl); + } + else + { + x = pos & 1; + + if (doc->number_up_layout & PSTOPS_LAYOUT_NEGATEX) + x = 1 - x; + + l = pagew; + w = l * bboxw / bboxl; + + if (w > (pagel * 0.5)) + { + w = pagel * 0.5; + l = w * bboxl / bboxw; + } + + tx = 0.5 * (pagel * 0.5 - w); + ty = 0.5 * (pagew - l); + + if (doc->normal_landscape) + doc_printf(doc, "%.1f 0.0 translate 90 rotate\n", pagew); + else + doc_printf(doc, "0.0 %.1f translate -90 rotate\n", pagel); + + doc_printf(doc, "%.1f %.1f translate %.3f %.3f scale\n", + tx + pagel * 0.5 * x, ty, w / bboxw, l / bboxl); + } + break; + + case 4 : + if (doc->number_up_layout & PSTOPS_LAYOUT_VERTICAL) + { + x = (pos / 2) & 1; + y = pos & 1; + } + else + { + x = pos & 1; + y = (pos / 2) & 1; + } + + if (doc->number_up_layout & PSTOPS_LAYOUT_NEGATEX) + x = 1 - x; + + if (doc->number_up_layout & PSTOPS_LAYOUT_NEGATEY) + y = 1 - y; + + w = pagew * 0.5; + l = w * bboxl / bboxw; + + if (l > (pagel * 0.5)) + { + l = pagel * 0.5; + w = l * bboxw / bboxl; + } + + tx = 0.5 * (pagew * 0.5 - w); + ty = 0.5 * (pagel * 0.5 - l); + + doc_printf(doc, "%.1f %.1f translate %.3f %.3f scale\n", + tx + x * pagew * 0.5, ty + y * pagel * 0.5, + w / bboxw, l / bboxl); + break; + + case 6 : + if (Orientation & 1) + { + if (doc->number_up_layout & PSTOPS_LAYOUT_VERTICAL) + { + x = pos / 3; + y = pos % 3; + + if (doc->number_up_layout & PSTOPS_LAYOUT_NEGATEX) + x = 1 - x; + + if (doc->number_up_layout & PSTOPS_LAYOUT_NEGATEY) + y = 2 - y; + } + else + { + x = pos & 1; + y = pos / 2; + + if (doc->number_up_layout & PSTOPS_LAYOUT_NEGATEX) + x = 1 - x; + + if (doc->number_up_layout & PSTOPS_LAYOUT_NEGATEY) + y = 2 - y; + } + + w = pagel * 0.5; + l = w * bboxl / bboxw; + + if (l > (pagew * 0.333)) + { + l = pagew * 0.333; + w = l * bboxw / bboxl; + } + + tx = 0.5 * (pagel - 2 * w); + ty = 0.5 * (pagew - 3 * l); + + if (doc->normal_landscape) + doc_printf(doc, "0 %.1f translate -90 rotate\n", pagel); + else + doc_printf(doc, "%.1f 0 translate 90 rotate\n", pagew); + + doc_printf(doc, "%.1f %.1f translate %.3f %.3f scale\n", + tx + x * w, ty + y * l, l / bboxl, w / bboxw); + } + else + { + if (doc->number_up_layout & PSTOPS_LAYOUT_VERTICAL) + { + x = pos / 2; + y = pos & 1; + + if (doc->number_up_layout & PSTOPS_LAYOUT_NEGATEX) + x = 2 - x; + + if (doc->number_up_layout & PSTOPS_LAYOUT_NEGATEY) + y = 1 - y; + } + else + { + x = pos % 3; + y = pos / 3; + + if (doc->number_up_layout & PSTOPS_LAYOUT_NEGATEX) + x = 2 - x; + + if (doc->number_up_layout & PSTOPS_LAYOUT_NEGATEY) + y = 1 - y; + } + + l = pagew * 0.5; + w = l * bboxw / bboxl; + + if (w > (pagel * 0.333)) + { + w = pagel * 0.333; + l = w * bboxl / bboxw; + } + + tx = 0.5 * (pagel - 3 * w); + ty = 0.5 * (pagew - 2 * l); + + if (doc->normal_landscape) + doc_printf(doc, "%.1f 0 translate 90 rotate\n", pagew); + else + doc_printf(doc, "0 %.1f translate -90 rotate\n", pagel); + + doc_printf(doc, "%.1f %.1f translate %.3f %.3f scale\n", + tx + w * x, ty + l * y, w / bboxw, l / bboxl); + + } + break; + + case 9 : + if (doc->number_up_layout & PSTOPS_LAYOUT_VERTICAL) + { + x = (pos / 3) % 3; + y = pos % 3; + } + else + { + x = pos % 3; + y = (pos / 3) % 3; + } + + if (doc->number_up_layout & PSTOPS_LAYOUT_NEGATEX) + x = 2 - x; + + if (doc->number_up_layout & PSTOPS_LAYOUT_NEGATEY) + y = 2 - y; + + w = pagew * 0.333; + l = w * bboxl / bboxw; + + if (l > (pagel * 0.333)) + { + l = pagel * 0.333; + w = l * bboxw / bboxl; + } + + tx = 0.5 * (pagew * 0.333 - w); + ty = 0.5 * (pagel * 0.333 - l); + + doc_printf(doc, "%.1f %.1f translate %.3f %.3f scale\n", + tx + x * pagew * 0.333, ty + y * pagel * 0.333, + w / bboxw, l / bboxl); + break; + + case 16 : + if (doc->number_up_layout & PSTOPS_LAYOUT_VERTICAL) + { + x = (pos / 4) & 3; + y = pos & 3; + } + else + { + x = pos & 3; + y = (pos / 4) & 3; + } + + if (doc->number_up_layout & PSTOPS_LAYOUT_NEGATEX) + x = 3 - x; + + if (doc->number_up_layout & PSTOPS_LAYOUT_NEGATEY) + y = 3 - y; + + w = pagew * 0.25; + l = w * bboxl / bboxw; + + if (l > (pagel * 0.25)) + { + l = pagel * 0.25; + w = l * bboxw / bboxl; + } + + tx = 0.5 * (pagew * 0.25 - w); + ty = 0.5 * (pagel * 0.25 - l); + + doc_printf(doc, "%.1f %.1f translate %.3f %.3f scale\n", + tx + x * pagew * 0.25, ty + y * pagel * 0.25, + w / bboxw, l / bboxl); + break; + } + + /* + * Draw borders as necessary... + */ + + if (doc->page_border && show_border) + { + int rects; /* Number of border rectangles */ + float fscale; /* Scaling value for points */ + + + rects = (doc->page_border & PSTOPS_BORDERDOUBLE) ? 2 : 1; + fscale = PageWidth / w; + margin = 2.25 * fscale; + + /* + * Set the line width and color... + */ + + doc_puts(doc, "gsave\n"); + doc_printf(doc, "%.3f setlinewidth 0 setgray newpath\n", + (doc->page_border & PSTOPS_BORDERTHICK) ? 0.5 * fscale : + 0.24 * fscale); + + /* + * Draw border boxes... + */ + + for (; rects > 0; rects --, margin += 2 * fscale) + if (doc->number_up > 1) + doc_printf(doc, "%.1f %.1f %.1f %.1f ESPrs\n", + margin, + margin, + bboxw - 2 * margin, + bboxl - 2 * margin); + else + doc_printf(doc, "%.1f %.1f %.1f %.1f ESPrs\n", + PageLeft + margin, + PageBottom + margin, + PageRight - PageLeft - 2 * margin, + PageTop - PageBottom - 2 * margin); + + /* + * Restore pen settings... + */ + + doc_puts(doc, "grestore\n"); + } + + if (doc->fitplot) + { + /* + * Offset the page by its bounding box... + */ + + doc_printf(doc, "%d %d translate\n", -bounding_box[0], + -bounding_box[1]); + } + + if (doc->fitplot || doc->number_up > 1) + { + /* + * Clip the page to the page's bounding box... + */ + + doc_printf(doc, "%.1f %.1f %.1f %.1f ESPrc\n", + bboxx + margin, bboxy + margin, + bboxw - 2 * margin, bboxl - 2 * margin); + } +} + + +/* + * 'write_label_prolog()' - Write the prolog with the classification + * and page label. + */ + +static void +write_label_prolog(pstops_doc_t *doc, /* I - Document info */ + const char *label, /* I - Page label */ + float bottom, /* I - Bottom position in points */ + float top, /* I - Top position in points */ + float width) /* I - Width in points */ +{ + const char *classification; /* CLASSIFICATION environment variable */ + const char *ptr; /* Temporary string pointer */ + + + /* + * First get the current classification... + */ + + if ((classification = getenv("CLASSIFICATION")) == NULL) + classification = ""; + if (strcmp(classification, "none") == 0) + classification = ""; + + /* + * If there is nothing to show, bind an empty 'write labels' procedure + * and return... + */ + + if (!classification[0] && (label == NULL || !label[0])) + { + doc_puts(doc, "userdict/ESPwl{}bind put\n"); + return; + } + + /* + * Set the classification + page label string... + */ + + doc_puts(doc, "userdict"); + if (!strcmp(classification, "confidential")) + doc_puts(doc, "/ESPpl(CONFIDENTIAL"); + else if (!strcmp(classification, "classified")) + doc_puts(doc, "/ESPpl(CLASSIFIED"); + else if (!strcmp(classification, "secret")) + doc_puts(doc, "/ESPpl(SECRET"); + else if (!strcmp(classification, "topsecret")) + doc_puts(doc, "/ESPpl(TOP SECRET"); + else if (!strcmp(classification, "unclassified")) + doc_puts(doc, "/ESPpl(UNCLASSIFIED"); + else + { + doc_puts(doc, "/ESPpl("); + + for (ptr = classification; *ptr; ptr ++) + { + if (*ptr < 32 || *ptr > 126) + doc_printf(doc, "\\%03o", *ptr); + else if (*ptr == '_') + doc_puts(doc, " "); + else if (*ptr == '(' || *ptr == ')' || *ptr == '\\') + doc_printf(doc, "\\%c", *ptr); + else + doc_printf(doc, "%c", *ptr); + } + } + + if (label) + { + if (classification[0]) + doc_puts(doc, " - "); + + /* + * Quote the label string as needed... + */ + + for (ptr = label; *ptr; ptr ++) + { + if (*ptr < 32 || *ptr > 126) + doc_printf(doc, "\\%03o", *ptr); + else if (*ptr == '(' || *ptr == ')' || *ptr == '\\') + doc_printf(doc, "\\%c", *ptr); + else + doc_printf(doc, "%c", *ptr); + } + } + + doc_puts(doc, ")put\n"); + + /* + * Then get a 14 point Helvetica-Bold font... + */ + + doc_puts(doc, "userdict/ESPpf /Helvetica-Bold findfont 14 scalefont put\n"); + + /* + * Finally, the procedure to write the labels on the page... + */ + + doc_puts(doc, "userdict/ESPwl{\n"); + doc_puts(doc, " ESPpf setfont\n"); + doc_printf(doc, " ESPpl stringwidth pop dup 12 add exch -0.5 mul %.0f add\n", + width * 0.5f); + doc_puts(doc, " 1 setgray\n"); + doc_printf(doc, " dup 6 sub %.0f 3 index 20 ESPrf\n", bottom - 2.0); + doc_printf(doc, " dup 6 sub %.0f 3 index 20 ESPrf\n", top - 18.0); + doc_puts(doc, " 0 setgray\n"); + doc_printf(doc, " dup 6 sub %.0f 3 index 20 ESPrs\n", bottom - 2.0); + doc_printf(doc, " dup 6 sub %.0f 3 index 20 ESPrs\n", top - 18.0); + doc_printf(doc, " dup %.0f moveto ESPpl show\n", bottom + 2.0); + doc_printf(doc, " %.0f moveto ESPpl show\n", top - 14.0); + doc_puts(doc, "pop\n"); + doc_puts(doc, "}bind put\n"); +} + + +/* + * 'write_labels()' - Write the actual page labels. + * + * This function is a copy of the one in common.c since we need to + * use doc_puts/doc_printf instead of puts/printf... + */ + +static void +write_labels(pstops_doc_t *doc, /* I - Document information */ + int orient) /* I - Orientation of the page */ +{ + float width, /* Width of page */ + length; /* Length of page */ + + + doc_puts(doc, "gsave\n"); + + if ((orient ^ Orientation) & 1) + { + width = PageLength; + length = PageWidth; + } + else + { + width = PageWidth; + length = PageLength; + } + + switch (orient & 3) + { + case 1 : /* Landscape */ + doc_printf(doc, "%.1f 0.0 translate 90 rotate\n", length); + break; + case 2 : /* Reverse Portrait */ + doc_printf(doc, "%.1f %.1f translate 180 rotate\n", width, length); + break; + case 3 : /* Reverse Landscape */ + doc_printf(doc, "0.0 %.1f translate -90 rotate\n", width); + break; + } + + doc_puts(doc, "ESPwl\n"); + doc_puts(doc, "grestore\n"); +} + + +/* + * 'write_options()' - Write options provided via %%IncludeFeature. + */ + +static void +write_options( + pstops_doc_t *doc, /* I - Document */ + ppd_file_t *ppd, /* I - PPD file */ + int num_options, /* I - Number of options */ + cups_option_t *options) /* I - Options */ +{ + int i; /* Looping var */ + ppd_option_t *option; /* PPD option */ + int min_order; /* Minimum OrderDependency value */ + char *doc_setup, /* DocumentSetup commands to send */ + *any_setup; /* AnySetup commands to send */ + + + /* + * Figure out the minimum OrderDependency value... + */ + + if ((option = ppdFindOption(ppd, "PageRegion")) != NULL) + min_order = option->order; + else + min_order = 999.0f; + + for (i = 0; i < num_options; i ++) + if ((option = ppdFindOption(ppd, options[i].name)) != NULL && + option->order < min_order) + min_order = option->order; + + /* + * Mark and extract them... + */ + + cupsMarkOptions(ppd, num_options, options); + + doc_setup = ppdEmitString(ppd, PPD_ORDER_DOCUMENT, min_order); + any_setup = ppdEmitString(ppd, PPD_ORDER_ANY, min_order); + + /* + * Then send them out... + */ + + if (doc->number_up > 1) + { + /* + * Temporarily restore setpagedevice so we can set the options... + */ + + doc_puts(doc, "userdict/setpagedevice/CUPSsetpagedevice load put\n"); + } + + if (doc_setup) + { + doc_puts(doc, doc_setup); + free(doc_setup); + } + + if (any_setup) + { + doc_puts(doc, any_setup); + free(any_setup); + } + + if (doc->number_up > 1) + { + /* + * Disable setpagedevice again... + */ + + doc_puts(doc, "userdict/setpagedevice{pop}bind put\n"); + } +} + + +/* + * End of "$Id$". + */ diff --git a/filter/raster-driver.header b/filter/raster-driver.header new file mode 100644 index 0000000000..e85c2a99a7 --- /dev/null +++ b/filter/raster-driver.header @@ -0,0 +1,32 @@ + + +

Developing Raster Printer Drivers

+ +

This document describes how to develop printer drivers for raster printers. Topics include: printer driver basics, creating new PPD files, using filters, implementing color management, and adding Mac OS X features.

+ +
+ + + + + + +
See AlsoProgramming: Developing PostScript Printer Drivers
+ Programming: Filter and Backend Programming
+ Programming: Introduction to the PPD Compiler
+ Programming: Raster API
+ References: PPD Compiler Driver Information File Reference
+ Specifications: CUPS PPD Extensions
diff --git a/filter/raster-driver.shtml b/filter/raster-driver.shtml new file mode 100644 index 0000000000..1c0c32a9d0 --- /dev/null +++ b/filter/raster-driver.shtml @@ -0,0 +1,194 @@ +

Printer Driver Basics

+ +

A CUPS raster printer driver consists of a PostScript Printer Description (PPD) file that describes the features and capabilities of the device, one or more filter programs that prepare print data for the device, and zero or more support files for color management, online help, and so forth. The PPD file includes references to all of the filters and support files used by the driver.

+ +

Every time a user prints something the scheduler program, cupsd(8), determines the format of the print job and the programs required to convert that job into something the printer understands. CUPS includes filter programs for many common formats, for example to convert Portable Document Format (PDF) files into CUPS raster data. Figure 1 shows the data flow of a typical print job.

+ +
+ + +
Figure 1: Raster Filter Chain
Raster Filter Chain
+ +

The raster filter converts CUPS raster data into a format the printer understands, for example HP-PCL. CUPS includes several sample raster filters supporting standard page description languages (PDLs). Table 1 shows the raster filters that are bundled with CUPS and the languages they support.

+ +
+ + + + + + + + + + + +
Table 1: Standard CUPS Raster Filters
FilterPDLsppdc DriverTypeppdc #include file
rastertoepsonESC/P, ESC/P2epsonepson.h
rastertoescpxESC/P, ESC/P2, EPSON Remote Modeescpescp.h
rastertohpHP-PCL3, HP-PCL5hphp.h
rastertolabelCPCL, Dymo, EPL1, EPL2, Intellitech PCL, ZPLlabellabel.h
rastertopclxHP-RTL, HP-PCL3, HP-PCL3GUI, HP-PCL5, HP-PCL5c, HP-PCL5epclpcl.h
+ +

The optional port monitor handles interface-specific protocol or encoding issues. For example, some raster printers use the 1284.4 communications protocol.

+ +

The backend handles communications with the printer, sending print data from the last filter to the printer and relaying back-channel data from the printer to the upstream filters. CUPS includes backend programs for common direct-connect interfaces and network protocols, and you can provide your own backend to support custom interfaces and protocols.

+ +

The scheduler also supports a special "command" file format for sending maintenance commands and status queries to a printer or printer driver. Command print jobs typically use a single command filter program defined in the PPD file to generate the appropriate printer commands and handle any responses from the printer. Figure 2 shows the data flow of a typical command job.

+ +
+ + +
Figure 2: Command Filter Chain
Command Filter Chain
+ +

Raster printer drivers must provide their own command filter.

+ + +

Creating New PPD Files

+ +

We recommend using the CUPS PPD compiler, ppdc(1), to create new PPD files since it manages many of the tedious (and error-prone!) details of paper sizes and localization for you. It also allows you to easily support multiple devices from a single source file. For more information see the "Introduction to the PPD Compiler" document. Listing 1 shows a driver information file for several similar black-and-white HP-PCL5 laser printers.

+ +

Listing 1: "examples/laserjet-basic.drv"

+ +
+// Include standard font and media definitions
+#include <font.defs>
+#include <media.defs>
+
+// Include HP-PCL driver definitions
+#include <pcl.h>
+
+// Specify that this driver uses the HP-PCL driver...
+DriverType pcl
+
+// Specify the driver options via the model number...
+ModelNumber ($PCL_PAPER_SIZE $PCL_PJL $PCL_PJL_RESOLUTION)
+
+// List the fonts that are supported, in this case all standard fonts...
+Font *
+
+// Manufacturer and driver version
+Manufacturer "HP"
+Version 1.0
+
+// Supported page sizes and their margins
+HWMargins 18 12 18 12
+*MediaSize Letter
+MediaSize Legal
+MediaSize Executive
+MediaSize Monarch
+MediaSize Statement
+MediaSize FanFoldGermanLegal
+
+HWMargins 18 12.72 18 12.72
+MediaSize Env10
+
+HWMargins 9.72 12 9.72 12
+MediaSize A4
+MediaSize A5
+MediaSize B5
+MediaSize EnvC5
+MediaSize EnvDL
+MediaSize EnvISOB5
+MediaSize Postcard
+MediaSize DoublePostcard
+
+// Only black-and-white output with mode 3 compression...
+ColorModel Gray k chunky 3
+
+// Supported resolutions
+Resolution - 1 0 0 0 "300dpi/300 DPI"
+*Resolution - 8 0 0 0 "600dpi/600 DPI"
+
+// Supported input slots
+*InputSlot 7 "Auto/Automatic Selection"
+InputSlot 2 "Manual/Tray 1 - Manual Feed"
+InputSlot 4 "Upper/Tray 1"
+InputSlot 1 "Lower/Tray 2"
+InputSlot 5 "LargeCapacity/Tray 3"
+
+// Tray 3 is an option...
+Installable "OptionLargeCapacity/Tray 3 Installed"
+UIConstraints "*OptionLargeCapacity False *InputSlot LargeCapacity"
+
+{
+  // HP LaserJet 2100 Series
+  Throughput 10
+  ModelName "LaserJet 2100 Series"
+  PCFileName "hpljt211.ppd"
+}
+
+{
+  // LaserJet 2200 and 2300 series have duplexer option...
+  Duplex normal
+  Installable "OptionDuplex/Duplexer Installed"
+  UIConstraints "*OptionDuplex False *Duplex"
+
+  {
+    // HP LaserJet 2200 Series
+    Throughput 19
+    ModelName "LaserJet 2200 Series"
+    PCFileName "hpljt221.ppd"
+  }
+
+  {
+    // HP LaserJet 2300 Series
+    Throughput 25
+    ModelName "LaserJet 2300 Series"
+    PCFileName "hpljt231.ppd"
+  }
+}
+
+ + +

Using Filters

+ +

The standard CUPS raster filters can be specified using the +DriverType directive, for example:

+ +
+// Specify that this driver uses the HP-PCL driver...
+DriverType pcl
+
+ +

Table 1 shows the driver types for each of the standard CUPS raster filters. For drivers that do not use the standard raster filters, the "custom" type is used with Filter directives:

+ +
+DriverType custom
+Filter application/vnd.cups-raster 100 /path/to/raster/filter
+Filter application/vnd.cups-command 100 /path/to/command/filter
+
+ + +

Implementing Color Management

+ +

CUPS uses ICC color profiles to provide more accurate color reproduction. The cupsICCProfile attribute defines the color profiles that are available for a given printer, for example:

+ +
+Attribute cupsICCProfile "ColorModel.MediaType.Resolution/Description" /path/to/ICC/profile
+
+ +

where "ColorModel.MediaType.Resolution" defines a selector based on the corresponding option selections. A simple driver might only define profiles for the color models that are supported, for example a printer supporting Gray and RGB might use:

+ +
+Attribute cupsICCProfile "Gray../Grayscale Profile" /path/to/ICC/gray-profile
+Attribute cupsICCProfile "RGB../Full Color Profile" /path/to/ICC/rgb-profile
+
+ +

The options used for profile selection can be customized using the cupsICCQualifier2 and cupsICCQualifier3 attributes.

+ +

Since Mac OS X 10.5Custom Color Matching Support

+ +

Mac OS X printer drivers that are based on an existing standard RGB colorspace can tell the system to use the corresponding colorspace instead of an arbitrary ICC color profile when doing color management. The APSupportsCustomColorMatching and APDefaultCustomColorMatchingProfile attributes can be used to enable this mode:

+ +
+Attribute APSupportsCustomColorMatching "" true
+Attribute APDefaultCustomColorMatchingProfile "" sRGB
+
+ + +

Adding Mac OS X Features

+ +

Mac OS X printer drivers can provide additional attributes to specify additional option panes in the print dialog, an image of the printer, a help book, and option presets for the driver software:

+ +
+Attribute APDialogExtension "" /Library/Printers/Vendor/filename.plugin
+Attribute APHelpBook "" /Library/Printers/Vendor/filename.bundle
+Attribute APPrinterIconPath "" /Library/Printers/Vendor/filename.icns
+Attribute APPrinterPreset "name/text" "*option choice ..."
+
diff --git a/filter/raster.c b/filter/raster.c new file mode 100644 index 0000000000..ae0e7b9df3 --- /dev/null +++ b/filter/raster.c @@ -0,0 +1,1476 @@ +/* + * "$Id$" + * + * Raster file routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * This file is part of the CUPS Imaging library. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * cupsRasterClose() - Close a raster stream. + * cupsRasterOpen() - Open a raster stream using a file descriptor. + * cupsRasterOpenIO() - Open a raster stream using a callback function. + * cupsRasterReadHeader() - Read a raster page header and store it in a + * version 1 page header structure. + * cupsRasterReadHeader2() - Read a raster page header and store it in a + * version 2 page header structure. + * cupsRasterReadPixels() - Read raster pixels. + * cupsRasterWriteHeader() - Write a raster page header from a version 1 + * page header structure. + * cupsRasterWriteHeader2() - Write a raster page header from a version 2 + * page header structure. + * cupsRasterWritePixels() - Write raster pixels. + * cups_raster_read_header() - Read a raster page header. + * cups_raster_read() - Read through the raster buffer. + * cups_raster_update() - Update the raster header and row count for the + * current page. + * cups_raster_write() - Write a row of compressed raster data... + * cups_read_fd() - Read bytes from a file. + * cups_swap() - Swap bytes in raster data... + * cups_write_fd() - Write bytes to a file. + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * Private structures... + */ + +struct _cups_raster_s /**** Raster stream data ****/ +{ + unsigned sync; /* Sync word from start of stream */ + void *ctx; /* File descriptor */ + cups_raster_iocb_t iocb; /* IO callback */ + cups_mode_t mode; /* Read/write mode */ + cups_page_header2_t header; /* Raster header for current page */ + int count, /* Current row run-length count */ + remaining, /* Remaining rows in page image */ + bpp; /* Bytes per pixel/color */ + unsigned char *pixels, /* Pixels for current row */ + *pend, /* End of pixel buffer */ + *pcurrent; /* Current byte in pixel buffer */ + int compressed, /* Non-zero if data is compressed */ + swapped; /* Non-zero if data is byte-swapped */ + unsigned char *buffer, /* Read/write buffer */ + *bufptr, /* Current (read) position in buffer */ + *bufend; /* End of current (read) buffer */ + size_t bufsize; /* Buffer size */ +}; + + +/* + * Local functions... + */ + +static int cups_raster_io(cups_raster_t *r, unsigned char *buf, int bytes); +static unsigned cups_raster_read_header(cups_raster_t *r); +static int cups_raster_read(cups_raster_t *r, unsigned char *buf, + int bytes); +static void cups_raster_update(cups_raster_t *r); +static int cups_raster_write(cups_raster_t *r, + const unsigned char *pixels); +static ssize_t cups_read_fd(void *ctx, unsigned char *buf, size_t bytes); +static void cups_swap(unsigned char *buf, int bytes); +static ssize_t cups_write_fd(void *ctx, unsigned char *buf, size_t bytes); + + +/* + * 'cupsRasterClose()' - Close a raster stream. + * + * The file descriptor associated with the raster stream must be closed + * separately as needed. + */ + +void +cupsRasterClose(cups_raster_t *r) /* I - Stream to close */ +{ + if (r != NULL) + { + if (r->buffer) + free(r->buffer); + + if (r->pixels) + free(r->pixels); + + free(r); + } +} + + +/* + * 'cupsRasterOpen()' - Open a raster stream using a file descriptor. + * + * This function associates a raster stream with the given file descriptor. + * For most printer driver filters, "fd" will be 0 (stdin). For most raster + * image processor (RIP) filters that generate raster data, "fd" will be 1 + * (stdout). + * + * When writing raster data, the @code CUPS_RASTER_WRITE@, + * @code CUPS_RASTER_WRITE_COMPRESS@, or @code CUPS_RASTER_WRITE_PWG@ mode can + * be used - compressed and PWG output is generally 25-50% smaller but adds a + * 100-300% execution time overhead. + */ + +cups_raster_t * /* O - New stream */ +cupsRasterOpen(int fd, /* I - File descriptor */ + cups_mode_t mode) /* I - Mode - @code CUPS_RASTER_READ@, + @code CUPS_RASTER_WRITE@, + @code CUPS_RASTER_WRITE_COMPRESSED@, + or @code CUPS_RASTER_WRITE_PWG@ */ +{ + if (mode == CUPS_RASTER_READ) + return (cupsRasterOpenIO(cups_read_fd, (void *)((intptr_t)fd), mode)); + else + return (cupsRasterOpenIO(cups_write_fd, (void *)((intptr_t)fd), mode)); +} + + +/* + * 'cupsRasterOpenIO()' - Open a raster stream using a callback function. + * + * This function associates a raster stream with the given callback function and + * context pointer. + * + * When writing raster data, the @code CUPS_RASTER_WRITE@, + * @code CUPS_RASTER_WRITE_COMPRESS@, or @code CUPS_RASTER_WRITE_PWG@ mode can + * be used - compressed and PWG output is generally 25-50% smaller but adds a + * 100-300% execution time overhead. + */ + +cups_raster_t * /* O - New stream */ +cupsRasterOpenIO( + cups_raster_iocb_t iocb, /* I - Read/write callback */ + void *ctx, /* I - Context pointer for callback */ + cups_mode_t mode) /* I - Mode - @code CUPS_RASTER_READ@, + @code CUPS_RASTER_WRITE@, + @code CUPS_RASTER_WRITE_COMPRESSED@, + or @code CUPS_RASTER_WRITE_PWG@ */ +{ + cups_raster_t *r; /* New stream */ + + + _cupsRasterClearError(); + + if ((r = calloc(sizeof(cups_raster_t), 1)) == NULL) + { + _cupsRasterAddError("Unable to allocate memory for raster stream: %s\n", + strerror(errno)); + return (NULL); + } + + r->ctx = ctx; + r->iocb = iocb; + r->mode = mode; + + if (mode == CUPS_RASTER_READ) + { + /* + * Open for read - get sync word... + */ + + if (cups_raster_io(r, (unsigned char *)&(r->sync), sizeof(r->sync)) != + sizeof(r->sync)) + { + _cupsRasterAddError("Unable to read header from raster stream: %s\n", + strerror(errno)); + free(r); + return (NULL); + } + + if (r->sync != CUPS_RASTER_SYNC && + r->sync != CUPS_RASTER_REVSYNC && + r->sync != CUPS_RASTER_SYNCv1 && + r->sync != CUPS_RASTER_REVSYNCv1 && + r->sync != CUPS_RASTER_SYNCv2 && + r->sync != CUPS_RASTER_REVSYNCv2) + { + _cupsRasterAddError("Unknown raster format %08x!\n", r->sync); + free(r); + return (NULL); + } + + if (r->sync == CUPS_RASTER_SYNCv2 || + r->sync == CUPS_RASTER_REVSYNCv2) + r->compressed = 1; + + if (r->sync == CUPS_RASTER_REVSYNC || + r->sync == CUPS_RASTER_REVSYNCv1 || + r->sync == CUPS_RASTER_REVSYNCv2) + r->swapped = 1; + + DEBUG_printf(("r->swapped=%d, r->sync=%08x\n", r->swapped, r->sync)); + } + else + { + /* + * Open for write - put sync word... + */ + + switch (mode) + { + default : + case CUPS_RASTER_WRITE : + r->sync = CUPS_RASTER_SYNC; + break; + + case CUPS_RASTER_WRITE_COMPRESSED : + r->compressed = 1; + r->sync = CUPS_RASTER_SYNCv2; + break; + + case CUPS_RASTER_WRITE_PWG : + r->compressed = 1; + r->sync = htonl(CUPS_RASTER_SYNC_PWG); + r->swapped = r->sync != CUPS_RASTER_SYNC_PWG; + break; + } + + if (cups_raster_io(r, (unsigned char *)&(r->sync), sizeof(r->sync)) + < sizeof(r->sync)) + { + _cupsRasterAddError("Unable to write raster stream header: %s\n", + strerror(errno)); + free(r); + return (NULL); + } + } + + return (r); +} + + +/* + * 'cupsRasterReadHeader()' - Read a raster page header and store it in a + * version 1 page header structure. + * + * This function is deprecated. Use @link cupsRasterReadHeader2@ instead. + * + * Version 1 page headers were used in CUPS 1.0 and 1.1 and contain a subset + * of the version 2 page header data. This function handles reading version 2 + * page headers and copying only the version 1 data into the provided buffer. + * + * @deprecated@ + */ + +unsigned /* O - 1 on success, 0 on failure/end-of-file */ +cupsRasterReadHeader( + cups_raster_t *r, /* I - Raster stream */ + cups_page_header_t *h) /* I - Pointer to header data */ +{ + /* + * Get the raster header... + */ + + if (!cups_raster_read_header(r)) + return (0); + + /* + * Copy the header to the user-supplied buffer... + */ + + memcpy(h, &(r->header), sizeof(cups_page_header_t)); + + return (1); +} + + +/* + * 'cupsRasterReadHeader2()' - Read a raster page header and store it in a + * version 2 page header structure. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +unsigned /* O - 1 on success, 0 on failure/end-of-file */ +cupsRasterReadHeader2( + cups_raster_t *r, /* I - Raster stream */ + cups_page_header2_t *h) /* I - Pointer to header data */ +{ + /* + * Get the raster header... + */ + + if (!cups_raster_read_header(r)) + return (0); + + /* + * Copy the header to the user-supplied buffer... + */ + + memcpy(h, &(r->header), sizeof(cups_page_header2_t)); + + return (1); +} + + +/* + * 'cupsRasterReadPixels()' - Read raster pixels. + * + * For best performance, filters should read one or more whole lines. + * The "cupsBytesPerLine" value from the page header can be used to allocate + * the line buffer and as the number of bytes to read. + */ + +unsigned /* O - Number of bytes read */ +cupsRasterReadPixels(cups_raster_t *r, /* I - Raster stream */ + unsigned char *p, /* I - Pointer to pixel buffer */ + unsigned len) /* I - Number of bytes to read */ +{ + int bytes; /* Bytes read */ + unsigned cupsBytesPerLine; /* cupsBytesPerLine value */ + unsigned remaining; /* Bytes remaining */ + unsigned char *ptr, /* Pointer to read buffer */ + byte, /* Byte from file */ + *temp; /* Pointer into buffer */ + int count; /* Repetition count */ + + + if (r == NULL || r->mode != CUPS_RASTER_READ || r->remaining == 0 || + r->header.cupsBytesPerLine == 0) + return (0); + + if (!r->compressed) + { + /* + * Read without compression... + */ + + r->remaining -= len / r->header.cupsBytesPerLine; + + if (cups_raster_io(r, p, len) < (ssize_t)len) + return (0); + + /* + * Swap bytes as needed... + */ + + if (r->swapped && + (r->header.cupsBitsPerColor == 16 || + r->header.cupsBitsPerPixel == 12 || + r->header.cupsBitsPerPixel == 16)) + cups_swap(p, len); + + /* + * Return... + */ + + return (len); + } + + /* + * Read compressed data... + */ + + remaining = len; + cupsBytesPerLine = r->header.cupsBytesPerLine; + + while (remaining > 0 && r->remaining > 0) + { + if (r->count == 0) + { + /* + * Need to read a new row... + */ + + if (remaining == cupsBytesPerLine) + ptr = p; + else + ptr = r->pixels; + + /* + * Read using a modified PackBits compression... + */ + + if (!cups_raster_read(r, &byte, 1)) + return (0); + + r->count = byte + 1; + + if (r->count > 1) + ptr = r->pixels; + + temp = ptr; + bytes = cupsBytesPerLine; + + while (bytes > 0) + { + /* + * Get a new repeat count... + */ + + if (!cups_raster_read(r, &byte, 1)) + return (0); + + if (byte & 128) + { + /* + * Copy N literal pixels... + */ + + count = (257 - byte) * r->bpp; + + if (count > bytes) + count = bytes; + + if (!cups_raster_read(r, temp, count)) + return (0); + + temp += count; + bytes -= count; + } + else + { + /* + * Repeat the next N bytes... + */ + + count = (byte + 1) * r->bpp; + if (count > bytes) + count = bytes; + + if (count < r->bpp) + break; + + bytes -= count; + + if (!cups_raster_read(r, temp, r->bpp)) + return (0); + + temp += r->bpp; + count -= r->bpp; + + while (count > 0) + { + memcpy(temp, temp - r->bpp, r->bpp); + temp += r->bpp; + count -= r->bpp; + } + } + } + + /* + * Swap bytes as needed... + */ + + if ((r->header.cupsBitsPerColor == 16 || + r->header.cupsBitsPerPixel == 12 || + r->header.cupsBitsPerPixel == 16) && + r->swapped) + cups_swap(ptr, bytes); + + /* + * Update pointers... + */ + + if (remaining >= cupsBytesPerLine) + { + bytes = cupsBytesPerLine; + r->pcurrent = r->pixels; + r->count --; + r->remaining --; + } + else + { + bytes = remaining; + r->pcurrent = r->pixels + bytes; + } + + /* + * Copy data as needed... + */ + + if (ptr != p) + memcpy(p, ptr, bytes); + } + else + { + /* + * Copy fragment from buffer... + */ + + if ((unsigned)(bytes = (int)(r->pend - r->pcurrent)) > remaining) + bytes = remaining; + + memcpy(p, r->pcurrent, bytes); + r->pcurrent += bytes; + + if (r->pcurrent >= r->pend) + { + r->pcurrent = r->pixels; + r->count --; + r->remaining --; + } + } + + remaining -= bytes; + p += bytes; + } + + return (len); +} + + +/* + * 'cupsRasterWriteHeader()' - Write a raster page header from a version 1 page + * header structure. + * + * This function is deprecated. Use @link cupsRasterWriteHeader2@ instead. + * + * @deprecated@ + */ + +unsigned /* O - 1 on success, 0 on failure */ +cupsRasterWriteHeader( + cups_raster_t *r, /* I - Raster stream */ + cups_page_header_t *h) /* I - Raster page header */ +{ + if (r == NULL || r->mode == CUPS_RASTER_READ) + return (0); + + /* + * Make a copy of the header, and compute the number of raster + * lines in the page image... + */ + + memset(&(r->header), 0, sizeof(r->header)); + memcpy(&(r->header), h, sizeof(cups_page_header_t)); + + cups_raster_update(r); + + /* + * Write the raster header... + */ + + if (r->mode == CUPS_RASTER_WRITE_PWG) + { + /* + * PWG raster data is always network byte order with much of the page header + * zeroed. + */ + + cups_page_header2_t fh; /* File page header */ + + memset(&fh, 0, sizeof(fh)); + + strlcpy(fh.MediaClass, "PwgRaster", sizeof(fh.MediaClass)); + /* PwgRaster */ + strlcpy(fh.MediaColor, r->header.MediaColor, sizeof(fh.MediaColor)); + strlcpy(fh.MediaType, r->header.MediaType, sizeof(fh.MediaType)); + strlcpy(fh.OutputType, r->header.OutputType, sizeof(fh.OutputType)); + /* PrintContentType */ + + fh.CutMedia = htonl(r->header.CutMedia); + fh.Duplex = htonl(r->header.Duplex); + fh.HWResolution[0] = htonl(r->header.HWResolution[0]); + fh.HWResolution[1] = htonl(r->header.HWResolution[1]); + fh.ImagingBoundingBox[0] = htonl(r->header.ImagingBoundingBox[0]); + fh.ImagingBoundingBox[1] = htonl(r->header.ImagingBoundingBox[1]); + fh.ImagingBoundingBox[2] = htonl(r->header.ImagingBoundingBox[2]); + fh.ImagingBoundingBox[3] = htonl(r->header.ImagingBoundingBox[3]); + fh.InsertSheet = htonl(r->header.InsertSheet); + fh.Jog = htonl(r->header.Jog); + fh.LeadingEdge = htonl(r->header.LeadingEdge); + fh.ManualFeed = htonl(r->header.ManualFeed); + fh.MediaPosition = htonl(r->header.MediaPosition); + fh.MediaWeight = htonl(r->header.MediaWeight); + fh.NumCopies = htonl(r->header.NumCopies); + fh.Orientation = htonl(r->header.Orientation); + fh.PageSize[0] = htonl(r->header.PageSize[0]); + fh.PageSize[1] = htonl(r->header.PageSize[1]); + fh.Tumble = htonl(r->header.Tumble); + fh.cupsWidth = htonl(r->header.cupsWidth); + fh.cupsHeight = htonl(r->header.cupsHeight); + fh.cupsBitsPerColor = htonl(r->header.cupsBitsPerColor); + fh.cupsBitsPerPixel = htonl(r->header.cupsBitsPerPixel); + fh.cupsBytesPerLine = htonl(r->header.cupsBytesPerLine); + fh.cupsColorOrder = htonl(r->header.cupsColorOrder); + fh.cupsColorSpace = htonl(r->header.cupsColorSpace); + fh.cupsNumColors = htonl(r->header.cupsNumColors); + fh.cupsInteger[0] = htonl(r->header.cupsInteger[0]); + /* TotalPageCount */ + fh.cupsInteger[1] = htonl(r->header.cupsInteger[1]); + /* CrossFeedTransform */ + fh.cupsInteger[2] = htonl(r->header.cupsInteger[2]); + /* FeedTransform */ + fh.cupsInteger[3] = htonl(r->header.cupsInteger[3]); + /* ImageBoxLeft */ + fh.cupsInteger[4] = htonl(r->header.cupsInteger[4]); + /* ImageBoxTop */ + fh.cupsInteger[5] = htonl(r->header.cupsInteger[5]); + /* ImageBoxRight */ + fh.cupsInteger[6] = htonl(r->header.cupsInteger[6]); + /* ImageBoxBottom */ + fh.cupsInteger[7] = htonl(r->header.cupsInteger[7]); + /* BlackPrimary */ + fh.cupsInteger[8] = htonl(r->header.cupsInteger[8]); + /* PrintQuality */ + fh.cupsInteger[14] = htonl(r->header.cupsInteger[14]); + /* VendorIdentifier */ + fh.cupsInteger[15] = htonl(r->header.cupsInteger[15]); + /* VendorLength */ + + memcpy(fh.cupsReal, r->header.cupsReal, + sizeof(fh.cupsReal) + sizeof(fh.cupsString)); + /* VendorData */ + + strlcpy(fh.cupsRenderingIntent, r->header.cupsRenderingIntent, + sizeof(fh.cupsRenderingIntent)); + strlcpy(fh.cupsPageSizeName, r->header.cupsPageSizeName, + sizeof(fh.cupsPageSizeName)); + + return (cups_raster_io(r, (unsigned char *)&fh, sizeof(fh)) == sizeof(fh)); + } + else + return (cups_raster_io(r, (unsigned char *)&(r->header), sizeof(r->header)) + == sizeof(r->header)); +} + + +/* + * 'cupsRasterWriteHeader2()' - Write a raster page header from a version 2 + * page header structure. + * + * The page header can be initialized using @link cupsRasterInterpretPPD@. + * + * @since CUPS 1.2/Mac OS X 10.5@ + */ + +unsigned /* O - 1 on success, 0 on failure */ +cupsRasterWriteHeader2( + cups_raster_t *r, /* I - Raster stream */ + cups_page_header2_t *h) /* I - Raster page header */ +{ + if (r == NULL || r->mode == CUPS_RASTER_READ) + return (0); + + /* + * Make a copy of the header, and compute the number of raster + * lines in the page image... + */ + + memcpy(&(r->header), h, sizeof(cups_page_header2_t)); + + cups_raster_update(r); + + /* + * Write the raster header... + */ + + if (r->mode == CUPS_RASTER_WRITE_PWG) + { + /* + * PWG raster data is always network byte order with most of the page header + * zeroed. + */ + + cups_page_header2_t fh; /* File page header */ + + memset(&fh, 0, sizeof(fh)); + strlcpy(fh.MediaClass, "PwgRaster", sizeof(fh.MediaClass)); + strlcpy(fh.MediaColor, r->header.MediaColor, sizeof(fh.MediaColor)); + strlcpy(fh.MediaType, r->header.MediaType, sizeof(fh.MediaType)); + strlcpy(fh.OutputType, r->header.OutputType, sizeof(fh.OutputType)); + strlcpy(fh.cupsRenderingIntent, r->header.cupsRenderingIntent, + sizeof(fh.cupsRenderingIntent)); + strlcpy(fh.cupsPageSizeName, r->header.cupsPageSizeName, + sizeof(fh.cupsPageSizeName)); + + fh.CutMedia = htonl(r->header.CutMedia); + fh.Duplex = htonl(r->header.Duplex); + fh.HWResolution[0] = htonl(r->header.HWResolution[0]); + fh.HWResolution[1] = htonl(r->header.HWResolution[1]); + fh.ImagingBoundingBox[0] = htonl(r->header.ImagingBoundingBox[0]); + fh.ImagingBoundingBox[1] = htonl(r->header.ImagingBoundingBox[1]); + fh.ImagingBoundingBox[2] = htonl(r->header.ImagingBoundingBox[2]); + fh.ImagingBoundingBox[3] = htonl(r->header.ImagingBoundingBox[3]); + fh.InsertSheet = htonl(r->header.InsertSheet); + fh.Jog = htonl(r->header.Jog); + fh.LeadingEdge = htonl(r->header.LeadingEdge); + fh.ManualFeed = htonl(r->header.ManualFeed); + fh.MediaPosition = htonl(r->header.MediaPosition); + fh.MediaWeight = htonl(r->header.MediaWeight); + fh.NumCopies = htonl(r->header.NumCopies); + fh.Orientation = htonl(r->header.Orientation); + fh.PageSize[0] = htonl(r->header.PageSize[0]); + fh.PageSize[1] = htonl(r->header.PageSize[1]); + fh.Tumble = htonl(r->header.Tumble); + fh.cupsWidth = htonl(r->header.cupsWidth); + fh.cupsHeight = htonl(r->header.cupsHeight); + fh.cupsBitsPerColor = htonl(r->header.cupsBitsPerColor); + fh.cupsBitsPerPixel = htonl(r->header.cupsBitsPerPixel); + fh.cupsBytesPerLine = htonl(r->header.cupsBytesPerLine); + fh.cupsColorOrder = htonl(r->header.cupsColorOrder); + fh.cupsColorSpace = htonl(r->header.cupsColorSpace); + fh.cupsNumColors = htonl(r->header.cupsNumColors); + fh.cupsInteger[0] = htonl(r->header.cupsInteger[0]); + fh.cupsInteger[1] = htonl(r->header.cupsInteger[1]); + fh.cupsInteger[2] = htonl(r->header.cupsInteger[2]); + fh.cupsInteger[3] = htonl((unsigned)(r->header.cupsImagingBBox[0] * + r->header.HWResolution[0])); + fh.cupsInteger[4] = htonl((unsigned)(r->header.cupsImagingBBox[1] * + r->header.HWResolution[1])); + fh.cupsInteger[5] = htonl((unsigned)(r->header.cupsImagingBBox[2] * + r->header.HWResolution[0])); + fh.cupsInteger[6] = htonl((unsigned)(r->header.cupsImagingBBox[3] * + r->header.HWResolution[1])); + fh.cupsInteger[7] = htonl(0xffffff); + + return (cups_raster_io(r, (unsigned char *)&fh, sizeof(fh)) == sizeof(fh)); + } + else + return (cups_raster_io(r, (unsigned char *)&(r->header), sizeof(r->header)) + == sizeof(r->header)); +} + + +/* + * 'cupsRasterWritePixels()' - Write raster pixels. + * + * For best performance, filters should write one or more whole lines. + * The "cupsBytesPerLine" value from the page header can be used to allocate + * the line buffer and as the number of bytes to write. + */ + +unsigned /* O - Number of bytes written */ +cupsRasterWritePixels(cups_raster_t *r, /* I - Raster stream */ + unsigned char *p, /* I - Bytes to write */ + unsigned len)/* I - Number of bytes to write */ +{ + int bytes; /* Bytes read */ + unsigned remaining; /* Bytes remaining */ + + + DEBUG_printf(("cupsRasterWritePixels(r=%p, p=%p, len=%u), remaining=%u\n", + r, p, len, r->remaining)); + + if (r == NULL || r->mode == CUPS_RASTER_READ || r->remaining == 0) + return (0); + + if (!r->compressed) + { + /* + * Without compression, just write the raster data raw unless the data needs + * to be swapped... + */ + + r->remaining -= len / r->header.cupsBytesPerLine; + + if (r->swapped && + (r->header.cupsBitsPerColor == 16 || + r->header.cupsBitsPerPixel == 12 || + r->header.cupsBitsPerPixel == 16)) + { + unsigned char *bufptr; /* Pointer into write buffer */ + unsigned count; /* Remaining count */ + + /* + * Allocate a write buffer as needed... + */ + + if ((size_t)len > r->bufsize) + { + if (r->buffer) + bufptr = realloc(r->buffer, len); + else + bufptr = malloc(len); + + if (!bufptr) + return (0); + + r->buffer = bufptr; + r->bufsize = len; + } + + /* + * Byte swap the pixels... + */ + + for (bufptr = r->buffer, count = len; count > 1; count -= 2, bufptr += 2) + { + bufptr[1] = *p++; + bufptr[0] = *p++; + } + + if (count) /* This should never happen... */ + *bufptr = *p; + + /* + * Write the byte-swapped buffer... + */ + + return (cups_raster_io(r, r->buffer, len)); + } + else + return (cups_raster_io(r, p, len)); + } + + /* + * Otherwise, compress each line... + */ + + for (remaining = len; remaining > 0; remaining -= bytes, p += bytes) + { + /* + * Figure out the number of remaining bytes on the current line... + */ + + if ((bytes = remaining) > (int)(r->pend - r->pcurrent)) + bytes = (int)(r->pend - r->pcurrent); + + if (r->count > 0) + { + /* + * Check to see if this line is the same as the previous line... + */ + + if (memcmp(p, r->pcurrent, bytes)) + { + if (!cups_raster_write(r, r->pixels)) + return (0); + + r->count = 0; + } + else + { + /* + * Mark more bytes as the same... + */ + + r->pcurrent += bytes; + + if (r->pcurrent >= r->pend) + { + /* + * Increase the repeat count... + */ + + r->count ++; + r->pcurrent = r->pixels; + + /* + * Flush out this line if it is the last one... + */ + + r->remaining --; + + if (r->remaining == 0) + return (cups_raster_write(r, r->pixels)); + else if (r->count == 256) + { + if (cups_raster_write(r, r->pixels) == 0) + return (0); + + r->count = 0; + } + } + + continue; + } + } + + if (r->count == 0) + { + /* + * Copy the raster data to the buffer... + */ + + memcpy(r->pcurrent, p, bytes); + + r->pcurrent += bytes; + + if (r->pcurrent >= r->pend) + { + /* + * Increase the repeat count... + */ + + r->count ++; + r->pcurrent = r->pixels; + + /* + * Flush out this line if it is the last one... + */ + + r->remaining --; + + if (r->remaining == 0) + return (cups_raster_write(r, r->pixels)); + } + } + } + + return (len); +} + + +/* + * 'cups_raster_read_header()' - Read a raster page header. + */ + +static unsigned /* O - 1 on success, 0 on fail */ +cups_raster_read_header( + cups_raster_t *r) /* I - Raster stream */ +{ + int len; /* Length for read/swap */ + + + if (r == NULL || r->mode != CUPS_RASTER_READ) + return (0); + + /* + * Get the length of the raster header... + */ + + if (r->sync == CUPS_RASTER_SYNCv1 || r->sync == CUPS_RASTER_REVSYNCv1) + len = sizeof(cups_page_header_t); + else + len = sizeof(cups_page_header2_t); + + /* + * Read the header... + */ + + memset(&(r->header), 0, sizeof(r->header)); + + if (cups_raster_read(r, (unsigned char *)&(r->header), len) < len) + return (0); + + /* + * Swap bytes as needed... + */ + + if (r->swapped) + { + unsigned *s, /* Current word */ + temp; /* Temporary copy */ + + + DEBUG_puts("Swapping header bytes..."); + + for (len = 81, s = &(r->header.AdvanceDistance); + len > 0; + len --, s ++) + { + DEBUG_printf(("%08x =>", *s)); + + temp = *s; + *s = ((temp & 0xff) << 24) | + ((temp & 0xff00) << 8) | + ((temp & 0xff0000) >> 8) | + ((temp & 0xff000000) >> 24); + + DEBUG_printf((" %08x\n", *s)); + } + } + + /* + * Update the header and row count... + */ + + cups_raster_update(r); + + return (r->header.cupsBytesPerLine != 0 && r->header.cupsHeight != 0); +} + + +/* + * 'cups_raster_io()' - Read/write bytes from a context, handling interruptions. + */ + +static int /* O - Bytes read or -1 */ +cups_raster_io(cups_raster_t *r, /* I - Raster stream */ + unsigned char *buf, /* I - Buffer for read/write */ + int bytes) /* I - Number of bytes to read/write */ +{ + ssize_t count; /* Number of bytes read/written */ + size_t total; /* Total bytes read/written */ + + + DEBUG_printf(("4cups_raster_io(r=%p, buf=%p, bytes=%d)", r, buf, bytes)); + + for (total = 0; total < (size_t)bytes; total += count, buf += count) + { + count = (*r->iocb)(r->ctx, buf, bytes - total); + + DEBUG_printf(("5cups_raster_io: count=%d, total=%d", (int)count, + (int)total)); + if (count == 0) + return (0); + else if (count < 0) + return (-1); + } + + return ((int)total); +} + + +/* + * 'cups_raster_read()' - Read through the raster buffer. + */ + +static int /* O - Number of bytes read */ +cups_raster_read(cups_raster_t *r, /* I - Raster stream */ + unsigned char *buf, /* I - Buffer */ + int bytes) /* I - Number of bytes to read */ +{ + int count, /* Number of bytes read */ + remaining, /* Remaining bytes in buffer */ + total; /* Total bytes read */ + + + DEBUG_printf(("cups_raster_read(r=%p, buf=%p, bytes=%d)\n", r, buf, bytes)); + + if (!r->compressed) + return (cups_raster_io(r, buf, bytes)); + + /* + * Allocate a read buffer as needed... + */ + + count = 2 * r->header.cupsBytesPerLine; + + if ((size_t)count > r->bufsize) + { + int offset = (int)(r->bufptr - r->buffer); + /* Offset to current start of buffer */ + int end = (int)(r->bufend - r->buffer); + /* Offset to current end of buffer */ + unsigned char *rptr; /* Pointer in read buffer */ + + if (r->buffer) + rptr = realloc(r->buffer, count); + else + rptr = malloc(count); + + if (!rptr) + return (0); + + r->buffer = rptr; + r->bufptr = rptr + offset; + r->bufend = rptr + end; + r->bufsize = count; + } + + /* + * Loop until we have read everything... + */ + + for (total = 0, remaining = (int)(r->bufend - r->bufptr); + total < bytes; + total += count, buf += count) + { + count = bytes - total; + + DEBUG_printf(("count=%d, remaining=%d, buf=%p, bufptr=%p, bufend=%p...\n", + count, remaining, buf, r->bufptr, r->bufend)); + + if (remaining == 0) + { + if (count < 16) + { + /* + * Read into the raster buffer and then copy... + */ + + remaining = (*r->iocb)(r->ctx, r->buffer, r->bufsize); + if (remaining <= 0) + return (0); + + r->bufptr = r->buffer; + r->bufend = r->buffer + remaining; + } + else + { + /* + * Read directly into "buf"... + */ + + count = (*r->iocb)(r->ctx, buf, count); + + if (count <= 0) + return (0); + + continue; + } + } + + /* + * Copy bytes from raster buffer to "buf"... + */ + + if (count > remaining) + count = remaining; + + if (count == 1) + { + /* + * Copy 1 byte... + */ + + *buf = *(r->bufptr)++; + remaining --; + } + else if (count < 128) + { + /* + * Copy up to 127 bytes without using memcpy(); this is + * faster because it avoids an extra function call and is + * often further optimized by the compiler... + */ + + unsigned char *bufptr; /* Temporary buffer pointer */ + + remaining -= count; + + for (bufptr = r->bufptr; count > 0; count --, total ++) + *buf++ = *bufptr++; + + r->bufptr = bufptr; + } + else + { + /* + * Use memcpy() for a large read... + */ + + memcpy(buf, r->bufptr, count); + r->bufptr += count; + remaining -= count; + } + } + + return (total); +} + + +/* + * 'cups_raster_update()' - Update the raster header and row count for the + * current page. + */ + +static void +cups_raster_update(cups_raster_t *r) /* I - Raster stream */ +{ + if (r->sync == CUPS_RASTER_SYNCv1 || r->sync == CUPS_RASTER_REVSYNCv1 || + r->header.cupsNumColors == 0) + { + /* + * Set the "cupsNumColors" field according to the colorspace... + */ + + switch (r->header.cupsColorSpace) + { + case CUPS_CSPACE_W : + case CUPS_CSPACE_K : + case CUPS_CSPACE_WHITE : + case CUPS_CSPACE_GOLD : + case CUPS_CSPACE_SILVER : + case CUPS_CSPACE_SW : + r->header.cupsNumColors = 1; + break; + + case CUPS_CSPACE_RGB : + case CUPS_CSPACE_CMY : + case CUPS_CSPACE_YMC : + case CUPS_CSPACE_CIEXYZ : + case CUPS_CSPACE_CIELab : + case CUPS_CSPACE_SRGB : + case CUPS_CSPACE_ADOBERGB : + case CUPS_CSPACE_ICC1 : + case CUPS_CSPACE_ICC2 : + case CUPS_CSPACE_ICC3 : + case CUPS_CSPACE_ICC4 : + case CUPS_CSPACE_ICC5 : + case CUPS_CSPACE_ICC6 : + case CUPS_CSPACE_ICC7 : + case CUPS_CSPACE_ICC8 : + case CUPS_CSPACE_ICC9 : + case CUPS_CSPACE_ICCA : + case CUPS_CSPACE_ICCB : + case CUPS_CSPACE_ICCC : + case CUPS_CSPACE_ICCD : + case CUPS_CSPACE_ICCE : + case CUPS_CSPACE_ICCF : + r->header.cupsNumColors = 3; + break; + + case CUPS_CSPACE_RGBA : + case CUPS_CSPACE_RGBW : + case CUPS_CSPACE_CMYK : + case CUPS_CSPACE_YMCK : + case CUPS_CSPACE_KCMY : + case CUPS_CSPACE_GMCK : + case CUPS_CSPACE_GMCS : + r->header.cupsNumColors = 4; + break; + + case CUPS_CSPACE_KCMYcm : + if (r->header.cupsBitsPerPixel < 8) + r->header.cupsNumColors = 6; + else + r->header.cupsNumColors = 4; + break; + + case CUPS_CSPACE_DEVICE1 : + case CUPS_CSPACE_DEVICE2 : + case CUPS_CSPACE_DEVICE3 : + case CUPS_CSPACE_DEVICE4 : + case CUPS_CSPACE_DEVICE5 : + case CUPS_CSPACE_DEVICE6 : + case CUPS_CSPACE_DEVICE7 : + case CUPS_CSPACE_DEVICE8 : + case CUPS_CSPACE_DEVICE9 : + case CUPS_CSPACE_DEVICEA : + case CUPS_CSPACE_DEVICEB : + case CUPS_CSPACE_DEVICEC : + case CUPS_CSPACE_DEVICED : + case CUPS_CSPACE_DEVICEE : + case CUPS_CSPACE_DEVICEF : + r->header.cupsNumColors = r->header.cupsColorSpace - + CUPS_CSPACE_DEVICE1 + 1; + break; + } + } + + /* + * Set the number of bytes per pixel/color... + */ + + if (r->header.cupsColorOrder == CUPS_ORDER_CHUNKED) + r->bpp = (r->header.cupsBitsPerPixel + 7) / 8; + else + r->bpp = (r->header.cupsBitsPerColor + 7) / 8; + + /* + * Set the number of remaining rows... + */ + + if (r->header.cupsColorOrder == CUPS_ORDER_PLANAR) + r->remaining = r->header.cupsHeight * r->header.cupsNumColors; + else + r->remaining = r->header.cupsHeight; + + /* + * Allocate the compression buffer... + */ + + if (r->compressed) + { + if (r->pixels != NULL) + free(r->pixels); + + r->pixels = calloc(r->header.cupsBytesPerLine, 1); + r->pcurrent = r->pixels; + r->pend = r->pixels + r->header.cupsBytesPerLine; + r->count = 0; + } +} + + +/* + * 'cups_raster_write()' - Write a row of compressed raster data... + */ + +static int /* O - Number of bytes written */ +cups_raster_write( + cups_raster_t *r, /* I - Raster stream */ + const unsigned char *pixels) /* I - Pixel data to write */ +{ + const unsigned char *start, /* Start of sequence */ + *ptr, /* Current pointer in sequence */ + *pend, /* End of raster buffer */ + *plast; /* Pointer to last pixel */ + unsigned char *wptr; /* Pointer into write buffer */ + int bpp, /* Bytes per pixel */ + count; /* Count */ + + + DEBUG_printf(("cups_raster_write(r=%p, pixels=%p)\n", r, pixels)); + + /* + * Allocate a write buffer as needed... + */ + + count = r->header.cupsBytesPerLine * 2; + if ((size_t)count > r->bufsize) + { + if (r->buffer) + wptr = realloc(r->buffer, count); + else + wptr = malloc(count); + + if (!wptr) + return (-1); + + r->buffer = wptr; + r->bufsize = count; + } + + /* + * Write the row repeat count... + */ + + bpp = r->bpp; + pend = pixels + r->header.cupsBytesPerLine; + plast = pend - bpp; + wptr = r->buffer; + *wptr++ = r->count - 1; + + /* + * Write using a modified PackBits compression... + */ + + for (ptr = pixels; ptr < pend;) + { + start = ptr; + ptr += bpp; + + if (ptr == pend) + { + /* + * Encode a single pixel at the end... + */ + + *wptr++ = 0; + for (count = bpp; count > 0; count --) + *wptr++ = *start++; + } + else if (!memcmp(start, ptr, bpp)) + { + /* + * Encode a sequence of repeating pixels... + */ + + for (count = 2; count < 128 && ptr < plast; count ++, ptr += bpp) + if (memcmp(ptr, ptr + bpp, bpp)) + break; + + *wptr++ = count - 1; + for (count = bpp; count > 0; count --) + *wptr++ = *ptr++; + } + else + { + /* + * Encode a sequence of non-repeating pixels... + */ + + for (count = 1; count < 128 && ptr < plast; count ++, ptr += bpp) + if (!memcmp(ptr, ptr + bpp, bpp)) + break; + + if (ptr >= plast && count < 128) + { + count ++; + ptr += bpp; + } + + *wptr++ = 257 - count; + + count *= bpp; + memcpy(wptr, start, count); + wptr += count; + } + } + + return (cups_raster_io(r, r->buffer, (int)(wptr - r->buffer))); +} + + +/* + * 'cups_read_fd()' - Read bytes from a file. + */ + +static ssize_t /* O - Bytes read or -1 */ +cups_read_fd(void *ctx, /* I - File descriptor as pointer */ + unsigned char *buf, /* I - Buffer for read */ + size_t bytes) /* I - Maximum number of bytes to read */ +{ + int fd = (int)((intptr_t)ctx); + /* File descriptor */ + ssize_t count; /* Number of bytes read */ + + +#ifdef WIN32 /* Sigh */ + while ((count = read(fd, buf, (unsigned)bytes)) < 0) +#else + while ((count = read(fd, buf, bytes)) < 0) +#endif /* WIN32 */ + if (errno != EINTR && errno != EAGAIN) + return (-1); + + return (count); +} + + +/* + * 'cups_swap()' - Swap bytes in raster data... + */ + +static void +cups_swap(unsigned char *buf, /* I - Buffer to swap */ + int bytes) /* I - Number of bytes to swap */ +{ + unsigned char even, odd; /* Temporary variables */ + + + bytes /= 2; + + while (bytes > 0) + { + even = buf[0]; + odd = buf[1]; + buf[0] = odd; + buf[1] = even; + + buf += 2; + bytes --; + } +} + + +/* + * 'cups_write_fd()' - Write bytes to a file. + */ + +static ssize_t /* O - Bytes written or -1 */ +cups_write_fd(void *ctx, /* I - File descriptor pointer */ + unsigned char *buf, /* I - Bytes to write */ + size_t bytes) /* I - Number of bytes to write */ +{ + int fd = (int)((intptr_t)ctx); + /* File descriptor */ + ssize_t count; /* Number of bytes written */ + + +#ifdef WIN32 /* Sigh */ + while ((count = write(fd, buf, (unsigned)bytes)) < 0) +#else + while ((count = write(fd, buf, bytes)) < 0) +#endif /* WIN32 */ + if (errno != EINTR && errno != EAGAIN) + return (-1); + + return (count); +} + + +/* + * End of "$Id$". + */ diff --git a/filter/rasterbench.c b/filter/rasterbench.c new file mode 100644 index 0000000000..69e139d37d --- /dev/null +++ b/filter/rasterbench.c @@ -0,0 +1,355 @@ +/* + * "$Id$" + * + * Raster benchmark program for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Benchmark the raster read/write functions. + * compute_median() - Compute the median time for a test. + * read_test() - Benchmark the raster read functions. + * write_test() - Benchmark the raster write functions. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include +#include +#include +#include + + +/* + * Constants... + */ + +#define TEST_WIDTH 1024 +#define TEST_HEIGHT 1024 +#define TEST_PAGES 16 +#define TEST_PASSES 20 + + +/* + * Local functions... + */ + +static double compute_median(double *secs); +static double get_time(void); +static void read_test(int fd); +static int run_read_test(void); +static void write_test(int fd, cups_mode_t mode); + + +/* + * 'main()' - Benchmark the raster read/write functions. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + int ras_fd, /* File descriptor for read process */ + status; /* Exit status of read process */ + double start_secs, /* Start time */ + write_secs, /* Write time */ + read_secs, /* Read time */ + pass_secs[TEST_PASSES]; /* Total test times */ + cups_mode_t mode; /* Write mode */ + + + /* + * See if we have anything on the command-line... + */ + + if (argc > 2 || (argc == 2 && strcmp(argv[1], "-z"))) + { + puts("Usage: rasterbench [-z]"); + return (1); + } + + mode = argc > 1 ? CUPS_RASTER_WRITE_COMPRESSED : CUPS_RASTER_WRITE; + + /* + * Ignore SIGPIPE... + */ + + signal(SIGPIPE, SIG_IGN); + + /* + * Run the tests several times to get a good average... + */ + + printf("Test read/write speed of %d pages, %dx%d pixels...\n\n", + TEST_PAGES, TEST_WIDTH, TEST_HEIGHT); + for (i = 0; i < TEST_PASSES; i ++) + { + printf("PASS %2d: ", i + 1); + fflush(stdout); + + ras_fd = run_read_test(); + start_secs = get_time(); + + write_test(ras_fd, mode); + + write_secs = get_time(); + printf(" %.3f write,", write_secs - start_secs); + fflush(stdout); + + close(ras_fd); + wait(&status); + + read_secs = get_time(); + pass_secs[i] = read_secs - start_secs; + printf(" %.3f read, %.3f total\n", read_secs - write_secs, pass_secs[i]); + } + + printf("\nMedian Total Time: %.3f seconds per document\n", + compute_median(pass_secs)); + + return (0); +} + + +/* + * 'compute_median()' - Compute the median time for a test. + */ + +static double /* O - Median time in seconds */ +compute_median(double *secs) /* I - Array of time samples */ +{ + int i, j; /* Looping vars */ + double temp; /* Swap variable */ + + + /* + * Sort the array into ascending order using a quicky bubble sort... + */ + + for (i = 0; i < (TEST_PASSES - 1); i ++) + for (j = i + 1; j < TEST_PASSES; j ++) + if (secs[i] > secs[j]) + { + temp = secs[i]; + secs[i] = secs[j]; + secs[j] = temp; + } + + /* + * Return the average of the middle two samples... + */ + + return (0.5 * (secs[TEST_PASSES / 2 - 1] + secs[TEST_PASSES / 2])); +} + + +/* + * 'get_time()' - Get the current time in seconds. + */ + +static double /* O - Time in seconds */ +get_time(void) +{ + struct timeval curtime; /* Current time */ + + + gettimeofday(&curtime, NULL); + return (curtime.tv_sec + 0.000001 * curtime.tv_usec); +} + + +/* + * 'read_test()' - Benchmark the raster read functions. + */ + +static void +read_test(int fd) /* I - File descriptor to read from */ +{ + int y; /* Looping var */ + cups_raster_t *r; /* Raster stream */ + cups_page_header2_t header; /* Page header */ + unsigned char buffer[8 * TEST_WIDTH]; + /* Read buffer */ + + + /* + * Test read speed... + */ + + if ((r = cupsRasterOpen(fd, CUPS_RASTER_READ)) == NULL) + { + perror("Unable to create raster input stream"); + return; + } + + while (cupsRasterReadHeader2(r, &header)) + { + for (y = 0; y < header.cupsHeight; y ++) + cupsRasterReadPixels(r, buffer, header.cupsBytesPerLine); + } + + cupsRasterClose(r); +} + + +/* + * 'run_read_test()' - Run the read test as a child process via pipes. + */ + +static int /* O - Standard input of child */ +run_read_test(void) +{ + int ras_pipes[2]; /* Raster data pipes */ + int pid; /* Child process ID */ + + + if (pipe(ras_pipes)) + return (-1); + + if ((pid = fork()) < 0) + { + /* + * Fork error - return -1 on error... + */ + + close(ras_pipes[0]); + close(ras_pipes[1]); + + return (-1); + } + else if (pid == 0) + { + /* + * Child comes here - read data from the input pipe... + */ + + close(ras_pipes[1]); + read_test(ras_pipes[0]); + exit(0); + } + else + { + /* + * Parent comes here - return the output pipe... + */ + + close(ras_pipes[0]); + return (ras_pipes[1]); + } +} + + +/* + * 'write_test()' - Benchmark the raster write functions. + */ + +static void +write_test(int fd, /* I - File descriptor to write to */ + cups_mode_t mode) /* I - Write mode */ +{ + int page, x, y; /* Looping vars */ + int count; /* Number of bytes to set */ + cups_raster_t *r; /* Raster stream */ + cups_page_header2_t header; /* Page header */ + unsigned char data[32][8 * TEST_WIDTH]; + /* Raster data to write */ + + + /* + * Create a combination of random data and repeated data to simulate + * text with some whitespace. + */ + + CUPS_SRAND(time(NULL)); + + memset(data, 0, sizeof(data)); + + for (y = 0; y < 28; y ++) + { + for (x = CUPS_RAND() & 127, count = (CUPS_RAND() & 15) + 1; + x < sizeof(data[0]); + x ++, count --) + { + if (count <= 0) + { + x += (CUPS_RAND() & 15) + 1; + count = (CUPS_RAND() & 15) + 1; + + if (x >= sizeof(data[0])) + break; + } + + data[y][x] = CUPS_RAND(); + } + } + + /* + * Test write speed... + */ + + if ((r = cupsRasterOpen(fd, mode)) == NULL) + { + perror("Unable to create raster output stream"); + return; + } + + for (page = 0; page < TEST_PAGES; page ++) + { + memset(&header, 0, sizeof(header)); + header.cupsWidth = TEST_WIDTH; + header.cupsHeight = TEST_HEIGHT; + header.cupsBytesPerLine = TEST_WIDTH; + + if (page & 1) + { + header.cupsBytesPerLine *= 4; + header.cupsColorSpace = CUPS_CSPACE_CMYK; + header.cupsColorOrder = CUPS_ORDER_CHUNKED; + } + else + { + header.cupsColorSpace = CUPS_CSPACE_K; + header.cupsColorOrder = CUPS_ORDER_BANDED; + } + + if (page & 2) + { + header.cupsBytesPerLine *= 2; + header.cupsBitsPerColor = 16; + header.cupsBitsPerPixel = (page & 1) ? 64 : 16; + } + else + { + header.cupsBitsPerColor = 8; + header.cupsBitsPerPixel = (page & 1) ? 32 : 8; + } + + cupsRasterWriteHeader2(r, &header); + + for (y = 0; y < TEST_HEIGHT; y ++) + cupsRasterWritePixels(r, data[y & 31], header.cupsBytesPerLine); + } + + cupsRasterClose(r); +} + + +/* + * End of "$Id$". + */ diff --git a/filter/rastertoepson.c b/filter/rastertoepson.c new file mode 100644 index 0000000000..6045e9add2 --- /dev/null +++ b/filter/rastertoepson.c @@ -0,0 +1,1160 @@ +/* + * "$Id$" + * + * EPSON ESC/P and ESC/P2 filter for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1993-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * Setup() - Prepare the printer for printing. + * StartPage() - Start a page of graphics. + * EndPage() - Finish a page of graphics. + * Shutdown() - Shutdown the printer. + * CompressData() - Compress a line of graphics. + * OutputLine() - Output a line of graphics. + * main() - Main entry and processing of driver. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include +#include +#include +#include +#include + + +/* + * Model numbers... + */ + +#define EPSON_9PIN 0 +#define EPSON_24PIN 1 +#define EPSON_COLOR 2 +#define EPSON_PHOTO 3 +#define EPSON_ICOLOR 4 +#define EPSON_IPHOTO 5 + + +/* + * Macros... + */ + +#define pwrite(s,n) fwrite((s), 1, (n), stdout) + + +/* + * Globals... + */ + +unsigned char *Planes[6], /* Output buffers */ + *CompBuffer, /* Compression buffer */ + *LineBuffers[2]; /* Line bitmap buffers */ +int Model, /* Model number */ + NumPlanes, /* Number of color planes */ + Feed, /* Number of lines to skip */ + EjectPage; /* Eject the page when done? */ +int DotBit, /* Bit in buffers */ + DotBytes, /* # bytes in a dot column */ + DotColumns, /* # columns in 1/60 inch */ + LineCount, /* # of lines processed */ + EvenOffset, /* Offset into 'even' buffers */ + OddOffset, /* Offset into 'odd' buffers */ + Shingling, /* Shingle output? */ + Canceled; /* Has the current job been canceled? */ + + +/* + * Prototypes... + */ + +void Setup(void); +void StartPage(const ppd_file_t *ppd, const cups_page_header2_t *header); +void EndPage(const cups_page_header2_t *header); +void Shutdown(void); + +void CancelJob(int sig); +void CompressData(const unsigned char *line, int length, int plane, + int type, int xstep, int ystep); +void OutputLine(const cups_page_header2_t *header); +void OutputRows(const cups_page_header2_t *header, int row); + + +/* + * 'Setup()' - Prepare the printer for printing. + */ + +void +Setup(void) +{ + const char *device_uri; /* The device for the printer... */ + + + /* + * EPSON USB printers need an additional command issued at the + * beginning of each job to exit from "packet" mode... + */ + + if ((device_uri = getenv("DEVICE_URI")) != NULL && + strncmp(device_uri, "usb:", 4) == 0 && Model >= EPSON_ICOLOR) + pwrite("\000\000\000\033\001@EJL 1284.4\n@EJL \n\033@", 29); +} + + +/* + * 'StartPage()' - Start a page of graphics. + */ + +void +StartPage( + const ppd_file_t *ppd, /* I - PPD file */ + const cups_page_header2_t *header) /* I - Page header */ +{ + int n, t; /* Numbers */ + int plane; /* Looping var */ + + + /* + * Send a reset sequence. + */ + + if (ppd && ppd->nickname && strstr(ppd->nickname, "OKIDATA") != NULL) + printf("\033{A"); /* Set EPSON emulation mode */ + + printf("\033@"); + + /* + * See which type of printer we are using... + */ + + switch (Model) + { + case EPSON_9PIN : + case EPSON_24PIN : + printf("\033P\022"); /* Set 10 CPI */ + + if (header->HWResolution[0] == 360 || header->HWResolution[0] == 240) + { + printf("\033x1"); /* LQ printing */ + printf("\033U1"); /* Unidirectional */ + } + else + { + printf("\033x0"); /* Draft printing */ + printf("\033U0"); /* Bidirectional */ + } + + printf("\033l%c\033Q%c", 0, /* Side margins */ + (int)(10.0 * header->PageSize[0] / 72.0 + 0.5)); + printf("\033\062\033C%c", /* Page length in 1/6th inches */ + (int)(header->PageSize[1] / 12.0 + 0.5)); + printf("\033N%c", 0); /* Bottom margin */ + printf("\033O"); /* No perforation skip */ + + /* + * Setup various buffer limits... + */ + + DotBytes = header->cupsRowCount / 8; + DotColumns = header->HWResolution[0] / 60; + Shingling = 0; + + if (Model == EPSON_9PIN) + printf("\033\063\030"); /* Set line feed */ + else + switch (header->HWResolution[0]) + { + case 60: + case 120 : + case 240 : + printf("\033\063\030"); /* Set line feed */ + break; + + case 180 : + case 360 : + Shingling = 1; + + if (header->HWResolution[1] == 180) + printf("\033\063\010");/* Set line feed */ + else + printf("\033+\010"); /* Set line feed */ + break; + } + break; + + default : + /* + * Set graphics mode... + */ + + pwrite("\033(G\001\000\001", 6); /* Graphics mode */ + + /* + * Set the media size... + */ + + if (Model < EPSON_ICOLOR) + { + pwrite("\033(U\001\000", 5); /* Resolution/units */ + putchar(3600 / header->HWResolution[1]); + } + else + { + pwrite("\033(U\005\000", 5); + putchar(1440 / header->HWResolution[1]); + putchar(1440 / header->HWResolution[1]); + putchar(1440 / header->HWResolution[0]); + putchar(0xa0); /* n/1440ths... */ + putchar(0x05); + } + + n = header->PageSize[1] * header->HWResolution[1] / 72.0; + + pwrite("\033(C\002\000", 5); /* Page length */ + putchar(n); + putchar(n >> 8); + + if (ppd) + t = (ppd->sizes[1].length - ppd->sizes[1].top) * + header->HWResolution[1] / 72.0; + else + t = 0; + + pwrite("\033(c\004\000", 5); /* Top & bottom margins */ + putchar(t); + putchar(t >> 8); + putchar(n); + putchar(n >> 8); + + if (header->HWResolution[1] == 720) + { + pwrite("\033(i\001\000\001", 6); /* Microweave */ + pwrite("\033(e\002\000\000\001", 7); /* Small dots */ + } + + pwrite("\033(V\002\000\000\000", 7); /* Set absolute position 0 */ + + DotBytes = 0; + DotColumns = 0; + Shingling = 0; + break; + } + + /* + * Set other stuff... + */ + + if (header->cupsColorSpace == CUPS_CSPACE_CMY) + NumPlanes = 3; + else if (header->cupsColorSpace == CUPS_CSPACE_KCMY) + NumPlanes = 4; + else if (header->cupsColorSpace == CUPS_CSPACE_KCMYcm) + NumPlanes = 6; + else + NumPlanes = 1; + + Feed = 0; /* No blank lines yet */ + + /* + * Allocate memory for a line/row of graphics... + */ + + if ((Planes[0] = malloc(header->cupsBytesPerLine)) == NULL) + { + fputs("ERROR: Unable to allocate memory\n", stderr); + exit(1); + } + + for (plane = 1; plane < NumPlanes; plane ++) + Planes[plane] = Planes[0] + plane * header->cupsBytesPerLine / NumPlanes; + + if (header->cupsCompression || DotBytes) + { + if ((CompBuffer = calloc(2, header->cupsWidth)) == NULL) + { + fputs("ERROR: Unable to allocate memory\n", stderr); + exit(1); + } + } + else + CompBuffer = NULL; + + if (DotBytes) + { + if ((LineBuffers[0] = calloc(DotBytes, + header->cupsWidth * (Shingling + 1))) == NULL) + { + fputs("ERROR: Unable to allocate memory\n", stderr); + exit(1); + } + + LineBuffers[1] = LineBuffers[0] + DotBytes * header->cupsWidth; + DotBit = 128; + LineCount = 0; + EvenOffset = 0; + OddOffset = 0; + } +} + + +/* + * 'EndPage()' - Finish a page of graphics. + */ + +void +EndPage( + const cups_page_header2_t *header) /* I - Page header */ +{ + if (DotBytes && header) + { + /* + * Flush remaining graphics as needed... + */ + + if (!Shingling) + { + if (DotBit < 128 || EvenOffset) + OutputRows(header, 0); + } + else if (OddOffset > EvenOffset) + { + OutputRows(header, 1); + OutputRows(header, 0); + } + else + { + OutputRows(header, 0); + OutputRows(header, 1); + } + } + + /* + * Eject the current page... + */ + + putchar(12); /* Form feed */ + fflush(stdout); + + /* + * Free memory... + */ + + free(Planes[0]); + + if (CompBuffer) + free(CompBuffer); + + if (DotBytes) + free(LineBuffers[0]); +} + + +/* + * 'Shutdown()' - Shutdown the printer. + */ + +void +Shutdown(void) +{ + /* + * Send a reset sequence. + */ + + printf("\033@"); +} + + +/* + * 'CancelJob()' - Cancel the current job... + */ + +void +CancelJob(int sig) /* I - Signal */ +{ + (void)sig; + + Canceled = 1; +} + + +/* + * 'CompressData()' - Compress a line of graphics. + */ + +void +CompressData(const unsigned char *line, /* I - Data to compress */ + int length,/* I - Number of bytes */ + int plane, /* I - Color plane */ + int type, /* I - Type of compression */ + int xstep, /* I - X resolution */ + int ystep) /* I - Y resolution */ +{ + const unsigned char *line_ptr, /* Current byte pointer */ + *line_end, /* End-of-line byte pointer */ + *start; /* Start of compression sequence */ + unsigned char *comp_ptr, /* Pointer into compression buffer */ + temp; /* Current byte */ + int count; /* Count of bytes for output */ + static int ctable[6] = { 0, 2, 1, 4, 18, 17 }; + /* KCMYcm color values */ + + + /* + * Setup pointers... + */ + + line_ptr = line; + line_end = line + length; + + /* + * Do depletion for 720 DPI printing... + */ + + if (ystep == 5) + { + for (comp_ptr = (unsigned char *)line; comp_ptr < line_end;) + { + /* + * Grab the current byte... + */ + + temp = *comp_ptr; + + /* + * Check adjacent bits... + */ + + if ((temp & 0xc0) == 0xc0) + temp &= 0xbf; + if ((temp & 0x60) == 0x60) + temp &= 0xdf; + if ((temp & 0x30) == 0x30) + temp &= 0xef; + if ((temp & 0x18) == 0x18) + temp &= 0xf7; + if ((temp & 0x0c) == 0x0c) + temp &= 0xfb; + if ((temp & 0x06) == 0x06) + temp &= 0xfd; + if ((temp & 0x03) == 0x03) + temp &= 0xfe; + + *comp_ptr++ = temp; + + /* + * Check the last bit in the current byte and the first bit in the + * next byte... + */ + + if ((temp & 0x01) && comp_ptr < line_end && *comp_ptr & 0x80) + *comp_ptr &= 0x7f; + } + } + + switch (type) + { + case 0 : + /* + * Do no compression... + */ + break; + + case 1 : + /* + * Do TIFF pack-bits encoding... + */ + + comp_ptr = CompBuffer; + + while (line_ptr < line_end) + { + if ((line_ptr + 1) >= line_end) + { + /* + * Single byte on the end... + */ + + *comp_ptr++ = 0x00; + *comp_ptr++ = *line_ptr++; + } + else if (line_ptr[0] == line_ptr[1]) + { + /* + * Repeated sequence... + */ + + line_ptr ++; + count = 2; + + while (line_ptr < (line_end - 1) && + line_ptr[0] == line_ptr[1] && + count < 127) + { + line_ptr ++; + count ++; + } + + *comp_ptr++ = 257 - count; + *comp_ptr++ = *line_ptr++; + } + else + { + /* + * Non-repeated sequence... + */ + + start = line_ptr; + line_ptr ++; + count = 1; + + while (line_ptr < (line_end - 1) && + line_ptr[0] != line_ptr[1] && + count < 127) + { + line_ptr ++; + count ++; + } + + *comp_ptr++ = count - 1; + + memcpy(comp_ptr, start, count); + comp_ptr += count; + } + } + + line_ptr = CompBuffer; + line_end = comp_ptr; + break; + } + + putchar(0x0d); /* Move print head to left margin */ + + if (Model < EPSON_ICOLOR) + { + /* + * Do graphics the "old" way... + */ + + if (NumPlanes > 1) + { + /* + * Set the color... + */ + + if (plane > 3) + printf("\033(r%c%c%c%c", 2, 0, 1, ctable[plane] & 15); + /* Set extended color */ + else if (NumPlanes == 3) + printf("\033r%c", ctable[plane + 1]); + /* Set color */ + else + printf("\033r%c", ctable[plane]); /* Set color */ + } + + /* + * Send a raster plane... + */ + + length *= 8; + printf("\033."); /* Raster graphics */ + putchar(type); + putchar(ystep); + putchar(xstep); + putchar(1); + putchar(length); + putchar(length >> 8); + } + else + { + /* + * Do graphics the "new" way... + */ + + printf("\033i"); + putchar(ctable[plane]); + putchar(type); + putchar(1); + putchar(length & 255); + putchar(length >> 8); + putchar(1); + putchar(0); + } + + pwrite(line_ptr, line_end - line_ptr); + fflush(stdout); +} + + +/* + * 'OutputLine()' - Output a line of graphics. + */ + +void +OutputLine( + const cups_page_header2_t *header) /* I - Page header */ +{ + if (header->cupsRowCount) + { + int width; + unsigned char *tempptr, + *evenptr, + *oddptr; + register int x; + unsigned char bit; + const unsigned char *pixel; + unsigned char *temp; + + + /* + * Collect bitmap data in the line buffers and write after each buffer. + */ + + for (x = header->cupsWidth, bit = 128, pixel = Planes[0], + temp = CompBuffer; + x > 0; + x --, temp ++) + { + if (*pixel & bit) + *temp |= DotBit; + + if (bit > 1) + bit >>= 1; + else + { + bit = 128; + pixel ++; + } + } + + if (DotBit > 1) + DotBit >>= 1; + else + { + /* + * Copy the holding buffer to the output buffer, shingling as necessary... + */ + + if (Shingling && LineCount != 0) + { + /* + * Shingle the output... + */ + + if (LineCount & 1) + { + evenptr = LineBuffers[1] + OddOffset; + oddptr = LineBuffers[0] + EvenOffset + DotBytes; + } + else + { + evenptr = LineBuffers[0] + EvenOffset; + oddptr = LineBuffers[1] + OddOffset + DotBytes; + } + + for (width = header->cupsWidth, tempptr = CompBuffer; + width > 0; + width -= 2, tempptr += 2, oddptr += DotBytes * 2, + evenptr += DotBytes * 2) + { + evenptr[0] = tempptr[0]; + oddptr[0] = tempptr[1]; + } + } + else + { + /* + * Don't shingle the output... + */ + + for (width = header->cupsWidth, tempptr = CompBuffer, + evenptr = LineBuffers[0] + EvenOffset; + width > 0; + width --, tempptr ++, evenptr += DotBytes) + *evenptr = tempptr[0]; + } + + if (Shingling && LineCount != 0) + { + EvenOffset ++; + OddOffset ++; + + if (EvenOffset == DotBytes) + { + EvenOffset = 0; + OutputRows(header, 0); + } + + if (OddOffset == DotBytes) + { + OddOffset = 0; + OutputRows(header, 1); + } + } + else + { + EvenOffset ++; + + if (EvenOffset == DotBytes) + { + EvenOffset = 0; + OutputRows(header, 0); + } + } + + DotBit = 128; + LineCount ++; + + memset(CompBuffer, 0, header->cupsWidth); + } + } + else + { + int plane; /* Current plane */ + int bytes; /* Bytes per plane */ + int xstep, ystep; /* X & Y resolutions */ + + + /* + * Write a single line of bitmap data as needed... + */ + + xstep = 3600 / header->HWResolution[0]; + ystep = 3600 / header->HWResolution[1]; + bytes = header->cupsBytesPerLine / NumPlanes; + + for (plane = 0; plane < NumPlanes; plane ++) + { + /* + * Skip blank data... + */ + + if (!Planes[plane][0] && + memcmp(Planes[plane], Planes[plane] + 1, bytes - 1) == 0) + continue; + + /* + * Output whitespace as needed... + */ + + if (Feed > 0) + { + pwrite("\033(v\002\000", 5); /* Relative vertical position */ + putchar(Feed); + putchar(Feed >> 8); + + Feed = 0; + } + + CompressData(Planes[plane], bytes, plane, header->cupsCompression, xstep, + ystep); + } + + Feed ++; + } +} + + +/* + * 'OutputRows()' - Output 8, 24, or 48 rows. + */ + +void +OutputRows( + const cups_page_header2_t *header, /* I - Page image header */ + int row) /* I - Row number (0 or 1) */ +{ + unsigned i, n; /* Looping vars */ + int dot_count, /* Number of bytes to print */ + dot_min; /* Minimum number of bytes */ + unsigned char *dot_ptr, /* Pointer to print data */ + *ptr; /* Current data */ + + + dot_min = DotBytes * DotColumns; + + if (LineBuffers[row][0] != 0 || + memcmp(LineBuffers[row], LineBuffers[row] + 1, + header->cupsWidth * DotBytes - 1)) + { + /* + * Skip leading space... + */ + + i = 0; + dot_count = header->cupsWidth * DotBytes; + dot_ptr = LineBuffers[row]; + + while (dot_count >= dot_min && dot_ptr[0] == 0 && + memcmp(dot_ptr, dot_ptr + 1, dot_min - 1) == 0) + { + i ++; + dot_ptr += dot_min; + dot_count -= dot_min; + } + + /* + * Skip trailing space... + */ + + while (dot_count >= dot_min && dot_ptr[dot_count - dot_min] == 0 && + memcmp(dot_ptr + dot_count - dot_min, + dot_ptr + dot_count - dot_min + 1, dot_min - 1) == 0) + dot_count -= dot_min; + + /* + * Position print head for printing... + */ + + if (i == 0) + putchar('\r'); + else + { + putchar(0x1b); + putchar('$'); + putchar(i & 255); + putchar(i >> 8); + } + + /* + * Start bitmap graphics for this line... + */ + + printf("\033*"); /* Select bit image */ + switch (header->HWResolution[0]) + { + case 60 : /* 60x60/72 DPI gfx */ + putchar(0); + break; + case 120 : /* 120x60/72 DPI gfx */ + putchar(1); + break; + case 180 : /* 180 DPI gfx */ + putchar(39); + break; + case 240 : /* 240x72 DPI gfx */ + putchar(3); + break; + case 360 : /* 360x180/360 DPI gfx */ + if (header->HWResolution[1] == 180) + { + if (Shingling && LineCount != 0) + putchar(40); /* 360x180 fast */ + else + putchar(41); /* 360x180 slow */ + } + else + { + if (Shingling && LineCount != 0) + putchar(72); /* 360x360 fast */ + else + putchar(73); /* 360x360 slow */ + } + break; + } + + n = (unsigned)dot_count / DotBytes; + putchar(n & 255); + putchar(n / 256); + + /* + * Write the graphics data... + */ + + if (header->HWResolution[0] == 120 || + header->HWResolution[0] == 240) + { + /* + * Need to interleave the dots to avoid hosing the print head... + */ + + for (n = dot_count / 2, ptr = dot_ptr; n > 0; n --, ptr += 2) + { + putchar(*ptr); + putchar(0); + } + + /* + * Move the head back and print the odd bytes... + */ + + if (i == 0) + putchar('\r'); + else + { + putchar(0x1b); + putchar('$'); + putchar(i & 255); + putchar(i >> 8); + } + + if (header->HWResolution[0] == 120) + printf("\033*\001"); /* Select bit image */ + else + printf("\033*\003"); /* Select bit image */ + + n = (unsigned)dot_count / DotBytes; + putchar(n & 255); + putchar(n / 256); + + for (n = dot_count / 2, ptr = dot_ptr + 1; n > 0; n --, ptr += 2) + { + putchar(0); + putchar(*ptr); + } + } + else + pwrite(dot_ptr, dot_count); + } + + /* + * Feed the paper... + */ + + putchar('\n'); + + if (Shingling && row == 1) + { + if (header->HWResolution[1] == 360) + printf("\n\n\n\n"); + else + printf("\n"); + } + + fflush(stdout); + + /* + * Clear the buffer... + */ + + memset(LineBuffers[row], 0, header->cupsWidth * DotBytes); +} + + +/* + * 'main()' - Main entry and processing of driver. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int fd; /* File descriptor */ + cups_raster_t *ras; /* Raster stream for printing */ + cups_page_header2_t header; /* Page header from file */ + ppd_file_t *ppd; /* PPD file */ + int page; /* Current page */ + int y; /* Current line */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* Actions for POSIX signals */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ + + + /* + * Make sure status messages are not buffered... + */ + + setbuf(stderr, NULL); + + /* + * Check command-line... + */ + + if (argc < 6 || argc > 7) + { + /* + * We don't have the correct number of arguments; write an error message + * and return. + */ + + _cupsLangPrintFilter(stderr, "ERROR", + _("%s job-id user title copies options [file]"), + "rastertoepson"); + return (1); + } + + /* + * Open the page stream... + */ + + if (argc == 7) + { + if ((fd = open(argv[6], O_RDONLY)) == -1) + { + _cupsLangPrintError("ERROR", _("Unable to open raster file")); + sleep(1); + return (1); + } + } + else + fd = 0; + + ras = cupsRasterOpen(fd, CUPS_RASTER_READ); + + /* + * Register a signal handler to eject the current page if the + * job is cancelled. + */ + + Canceled = 0; + +#ifdef HAVE_SIGSET /* Use System V signals over POSIX to avoid bugs */ + sigset(SIGTERM, CancelJob); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + + sigemptyset(&action.sa_mask); + action.sa_handler = CancelJob; + sigaction(SIGTERM, &action, NULL); +#else + signal(SIGTERM, CancelJob); +#endif /* HAVE_SIGSET */ + + /* + * Initialize the print device... + */ + + ppd = ppdOpenFile(getenv("PPD")); + if (!ppd) + { + ppd_status_t status; /* PPD error */ + int linenum; /* Line number */ + + _cupsLangPrintFilter(stderr, "ERROR", + _("The PPD file could not be opened.")); + + status = ppdLastError(&linenum); + + fprintf(stderr, "DEBUG: %s on line %d.\n", ppdErrorString(status), linenum); + + return (1); + } + + Model = ppd->model_number; + + Setup(); + + /* + * Process pages as needed... + */ + + page = 0; + + while (cupsRasterReadHeader2(ras, &header)) + { + /* + * Write a status message with the page number and number of copies. + */ + + if (Canceled) + break; + + page ++; + + fprintf(stderr, "PAGE: %d %d\n", page, header.NumCopies); + _cupsLangPrintFilter(stderr, "INFO", _("Starting page %d."), page); + + /* + * Start the page... + */ + + StartPage(ppd, &header); + + /* + * Loop for each line on the page... + */ + + for (y = 0; y < header.cupsHeight; y ++) + { + /* + * Let the user know how far we have progressed... + */ + + if (Canceled) + break; + + if ((y & 127) == 0) + { + _cupsLangPrintFilter(stderr, "INFO", + _("Printing page %d, %d%% complete."), + page, 100 * y / header.cupsHeight); + fprintf(stderr, "ATTR: job-media-progress=%d\n", + 100 * y / header.cupsHeight); + } + + /* + * Read a line of graphics... + */ + + if (cupsRasterReadPixels(ras, Planes[0], header.cupsBytesPerLine) < 1) + break; + + /* + * Write it to the printer... + */ + + OutputLine(&header); + } + + /* + * Eject the page... + */ + + _cupsLangPrintFilter(stderr, "INFO", _("Finished page %d."), page); + + EndPage(&header); + + if (Canceled) + break; + } + + /* + * Shutdown the printer... + */ + + Shutdown(); + + ppdClose(ppd); + + /* + * Close the raster stream... + */ + + cupsRasterClose(ras); + if (fd != 0) + close(fd); + + /* + * If no pages were printed, send an error message... + */ + + if (page == 0) + { + _cupsLangPrintFilter(stderr, "ERROR", _("No pages were found.")); + return (1); + } + else + { + _cupsLangPrintFilter(stderr, "INFO", _("Ready to print.")); + return (0); + } +} + + +/* + * End of "$Id$". + */ diff --git a/filter/rastertohp.c b/filter/rastertohp.c new file mode 100644 index 0000000000..5684c1542d --- /dev/null +++ b/filter/rastertohp.c @@ -0,0 +1,889 @@ +/* + * "$Id$" + * + * Hewlett-Packard Page Control Language filter for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1993-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * Setup() - Prepare the printer for printing. + * StartPage() - Start a page of graphics. + * EndPage() - Finish a page of graphics. + * Shutdown() - Shutdown the printer. + * CancelJob() - Cancel the current job... + * CompressData() - Compress a line of graphics. + * OutputLine() - Output a line of graphics. + * main() - Main entry and processing of driver. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include +#include +#include +#include +#include + + +/* + * Globals... + */ + +unsigned char *Planes[4], /* Output buffers */ + *CompBuffer, /* Compression buffer */ + *BitBuffer; /* Buffer for output bits */ +int NumPlanes, /* Number of color planes */ + ColorBits, /* Number of bits per color */ + Feed, /* Number of lines to skip */ + Duplex, /* Current duplex mode */ + Page, /* Current page number */ + Canceled; /* Has the current job been canceled? */ + + +/* + * Prototypes... + */ + +void Setup(void); +void StartPage(ppd_file_t *ppd, cups_page_header2_t *header); +void EndPage(void); +void Shutdown(void); + +void CancelJob(int sig); +void CompressData(unsigned char *line, int length, int plane, int type); +void OutputLine(cups_page_header2_t *header); + + +/* + * 'Setup()' - Prepare the printer for printing. + */ + +void +Setup(void) +{ + /* + * Send a PCL reset sequence. + */ + + putchar(0x1b); + putchar('E'); +} + + +/* + * 'StartPage()' - Start a page of graphics. + */ + +void +StartPage(ppd_file_t *ppd, /* I - PPD file */ + cups_page_header2_t *header) /* I - Page header */ +{ + int plane; /* Looping var */ + + + /* + * Show page device dictionary... + */ + + fprintf(stderr, "DEBUG: StartPage...\n"); + fprintf(stderr, "DEBUG: MediaClass = \"%s\"\n", header->MediaClass); + fprintf(stderr, "DEBUG: MediaColor = \"%s\"\n", header->MediaColor); + fprintf(stderr, "DEBUG: MediaType = \"%s\"\n", header->MediaType); + fprintf(stderr, "DEBUG: OutputType = \"%s\"\n", header->OutputType); + + fprintf(stderr, "DEBUG: AdvanceDistance = %d\n", header->AdvanceDistance); + fprintf(stderr, "DEBUG: AdvanceMedia = %d\n", header->AdvanceMedia); + fprintf(stderr, "DEBUG: Collate = %d\n", header->Collate); + fprintf(stderr, "DEBUG: CutMedia = %d\n", header->CutMedia); + fprintf(stderr, "DEBUG: Duplex = %d\n", header->Duplex); + fprintf(stderr, "DEBUG: HWResolution = [ %d %d ]\n", header->HWResolution[0], + header->HWResolution[1]); + fprintf(stderr, "DEBUG: ImagingBoundingBox = [ %d %d %d %d ]\n", + header->ImagingBoundingBox[0], header->ImagingBoundingBox[1], + header->ImagingBoundingBox[2], header->ImagingBoundingBox[3]); + fprintf(stderr, "DEBUG: InsertSheet = %d\n", header->InsertSheet); + fprintf(stderr, "DEBUG: Jog = %d\n", header->Jog); + fprintf(stderr, "DEBUG: LeadingEdge = %d\n", header->LeadingEdge); + fprintf(stderr, "DEBUG: Margins = [ %d %d ]\n", header->Margins[0], + header->Margins[1]); + fprintf(stderr, "DEBUG: ManualFeed = %d\n", header->ManualFeed); + fprintf(stderr, "DEBUG: MediaPosition = %d\n", header->MediaPosition); + fprintf(stderr, "DEBUG: MediaWeight = %d\n", header->MediaWeight); + fprintf(stderr, "DEBUG: MirrorPrint = %d\n", header->MirrorPrint); + fprintf(stderr, "DEBUG: NegativePrint = %d\n", header->NegativePrint); + fprintf(stderr, "DEBUG: NumCopies = %d\n", header->NumCopies); + fprintf(stderr, "DEBUG: Orientation = %d\n", header->Orientation); + fprintf(stderr, "DEBUG: OutputFaceUp = %d\n", header->OutputFaceUp); + fprintf(stderr, "DEBUG: PageSize = [ %d %d ]\n", header->PageSize[0], + header->PageSize[1]); + fprintf(stderr, "DEBUG: Separations = %d\n", header->Separations); + fprintf(stderr, "DEBUG: TraySwitch = %d\n", header->TraySwitch); + fprintf(stderr, "DEBUG: Tumble = %d\n", header->Tumble); + fprintf(stderr, "DEBUG: cupsWidth = %d\n", header->cupsWidth); + fprintf(stderr, "DEBUG: cupsHeight = %d\n", header->cupsHeight); + fprintf(stderr, "DEBUG: cupsMediaType = %d\n", header->cupsMediaType); + fprintf(stderr, "DEBUG: cupsBitsPerColor = %d\n", header->cupsBitsPerColor); + fprintf(stderr, "DEBUG: cupsBitsPerPixel = %d\n", header->cupsBitsPerPixel); + fprintf(stderr, "DEBUG: cupsBytesPerLine = %d\n", header->cupsBytesPerLine); + fprintf(stderr, "DEBUG: cupsColorOrder = %d\n", header->cupsColorOrder); + fprintf(stderr, "DEBUG: cupsColorSpace = %d\n", header->cupsColorSpace); + fprintf(stderr, "DEBUG: cupsCompression = %d\n", header->cupsCompression); + + /* + * Setup printer/job attributes... + */ + + Duplex = header->Duplex; + ColorBits = header->cupsBitsPerColor; + + if ((!Duplex || (Page & 1)) && header->MediaPosition) + printf("\033&l%dH", /* Set media position */ + header->MediaPosition); + + if (Duplex && ppd && ppd->model_number == 2) + { + /* + * Handle duplexing on new DeskJet printers... + */ + + printf("\033&l-2H"); /* Load media */ + + if (Page & 1) + printf("\033&l2S"); /* Set duplex mode */ + } + + if (!Duplex || (Page & 1) || (ppd && ppd->model_number == 2)) + { + /* + * Set the media size... + */ + + printf("\033&l6D\033&k12H"); /* Set 6 LPI, 10 CPI */ + printf("\033&l0O"); /* Set portrait orientation */ + + switch (header->PageSize[1]) + { + case 540 : /* Monarch Envelope */ + printf("\033&l80A"); /* Set page size */ + break; + + case 595 : /* A5 */ + printf("\033&l25A"); /* Set page size */ + break; + + case 624 : /* DL Envelope */ + printf("\033&l90A"); /* Set page size */ + break; + + case 649 : /* C5 Envelope */ + printf("\033&l91A"); /* Set page size */ + break; + + case 684 : /* COM-10 Envelope */ + printf("\033&l81A"); /* Set page size */ + break; + + case 709 : /* B5 Envelope */ + printf("\033&l100A"); /* Set page size */ + break; + + case 756 : /* Executive */ + printf("\033&l1A"); /* Set page size */ + break; + + case 792 : /* Letter */ + printf("\033&l2A"); /* Set page size */ + break; + + case 842 : /* A4 */ + printf("\033&l26A"); /* Set page size */ + break; + + case 1008 : /* Legal */ + printf("\033&l3A"); /* Set page size */ + break; + + case 1191 : /* A3 */ + printf("\033&l27A"); /* Set page size */ + break; + + case 1224 : /* Tabloid */ + printf("\033&l6A"); /* Set page size */ + break; + } + + printf("\033&l%dP", /* Set page length */ + header->PageSize[1] / 12); + printf("\033&l0E"); /* Set top margin to 0 */ + } + + if (!Duplex || (Page & 1)) + { + /* + * Set other job options... + */ + + printf("\033&l%dX", header->NumCopies); /* Set number copies */ + + if (header->cupsMediaType && + (!ppd || ppd->model_number != 2 || header->HWResolution[0] == 600)) + printf("\033&l%dM", /* Set media type */ + header->cupsMediaType); + + if (!ppd || ppd->model_number != 2) + { + int mode = Duplex ? 1 + header->Tumble != 0 : 0; + + printf("\033&l%dS", mode); /* Set duplex mode */ + printf("\033&l0L"); /* Turn off perforation skip */ + } + } + else if (!ppd || ppd->model_number != 2) + printf("\033&a2G"); /* Set back side */ + + /* + * Set graphics mode... + */ + + if (ppd && ppd->model_number == 2) + { + /* + * Figure out the number of color planes... + */ + + if (header->cupsColorSpace == CUPS_CSPACE_KCMY) + NumPlanes = 4; + else + NumPlanes = 1; + + /* + * Set the resolution and top-of-form... + */ + + printf("\033&u%dD", header->HWResolution[0]); + /* Resolution */ + printf("\033&l0e0L"); /* Reset top and don't skip */ + printf("\033*p0Y\033*p0X"); /* Set top of form */ + + /* + * Send 26-byte configure image data command with horizontal and + * vertical resolutions as well as a color count... + */ + + printf("\033*g26W"); + putchar(2); /* Format 2 */ + putchar(NumPlanes); /* Output planes */ + + putchar(header->HWResolution[0] >> 8); /* Black resolution */ + putchar(header->HWResolution[0]); + putchar(header->HWResolution[1] >> 8); + putchar(header->HWResolution[1]); + putchar(0); + putchar(1 << ColorBits); /* # of black levels */ + + putchar(header->HWResolution[0] >> 8); /* Cyan resolution */ + putchar(header->HWResolution[0]); + putchar(header->HWResolution[1] >> 8); + putchar(header->HWResolution[1]); + putchar(0); + putchar(1 << ColorBits); /* # of cyan levels */ + + putchar(header->HWResolution[0] >> 8); /* Magenta resolution */ + putchar(header->HWResolution[0]); + putchar(header->HWResolution[1] >> 8); + putchar(header->HWResolution[1]); + putchar(0); + putchar(1 << ColorBits); /* # of magenta levels */ + + putchar(header->HWResolution[0] >> 8); /* Yellow resolution */ + putchar(header->HWResolution[0]); + putchar(header->HWResolution[1] >> 8); + putchar(header->HWResolution[1]); + putchar(0); + putchar(1 << ColorBits); /* # of yellow levels */ + + printf("\033&l0H"); /* Set media position */ + } + else + { + printf("\033*t%dR", header->HWResolution[0]); + /* Set resolution */ + + if (header->cupsColorSpace == CUPS_CSPACE_KCMY) + { + NumPlanes = 4; + printf("\033*r-4U"); /* Set KCMY graphics */ + } + else if (header->cupsColorSpace == CUPS_CSPACE_CMY) + { + NumPlanes = 3; + printf("\033*r-3U"); /* Set CMY graphics */ + } + else + NumPlanes = 1; /* Black&white graphics */ + + /* + * Set size and position of graphics... + */ + + printf("\033*r%dS", header->cupsWidth); /* Set width */ + printf("\033*r%dT", header->cupsHeight); /* Set height */ + + printf("\033&a0H"); /* Set horizontal position */ + + if (ppd) + printf("\033&a%.0fV", /* Set vertical position */ + 10.0 * (ppd->sizes[0].length - ppd->sizes[0].top)); + else + printf("\033&a0V"); /* Set top-of-page */ + } + + printf("\033*r1A"); /* Start graphics */ + + if (header->cupsCompression) + printf("\033*b%dM", /* Set compression */ + header->cupsCompression); + + Feed = 0; /* No blank lines yet */ + + /* + * Allocate memory for a line of graphics... + */ + + if ((Planes[0] = malloc(header->cupsBytesPerLine)) == NULL) + { + fputs("ERROR: Unable to allocate memory\n", stderr); + exit(1); + } + + for (plane = 1; plane < NumPlanes; plane ++) + Planes[plane] = Planes[0] + plane * header->cupsBytesPerLine / NumPlanes; + + if (ColorBits > 1) + BitBuffer = malloc(ColorBits * ((header->cupsWidth + 7) / 8)); + else + BitBuffer = NULL; + + if (header->cupsCompression) + CompBuffer = malloc(header->cupsBytesPerLine * 2); + else + CompBuffer = NULL; +} + + +/* + * 'EndPage()' - Finish a page of graphics. + */ + +void +EndPage(void) +{ + /* + * Eject the current page... + */ + + if (NumPlanes > 1) + { + printf("\033*rC"); /* End color GFX */ + + if (!(Duplex && (Page & 1))) + printf("\033&l0H"); /* Eject current page */ + } + else + { + printf("\033*r0B"); /* End GFX */ + + if (!(Duplex && (Page & 1))) + printf("\014"); /* Eject current page */ + } + + fflush(stdout); + + /* + * Free memory... + */ + + free(Planes[0]); + + if (BitBuffer) + free(BitBuffer); + + if (CompBuffer) + free(CompBuffer); +} + + +/* + * 'Shutdown()' - Shutdown the printer. + */ + +void +Shutdown(void) +{ + /* + * Send a PCL reset sequence. + */ + + putchar(0x1b); + putchar('E'); +} + + +/* + * 'CancelJob()' - Cancel the current job... + */ + +void +CancelJob(int sig) /* I - Signal */ +{ + (void)sig; + + Canceled = 1; +} + + +/* + * 'CompressData()' - Compress a line of graphics. + */ + +void +CompressData(unsigned char *line, /* I - Data to compress */ + int length, /* I - Number of bytes */ + int plane, /* I - Color plane */ + int type) /* I - Type of compression */ +{ + unsigned char *line_ptr, /* Current byte pointer */ + *line_end, /* End-of-line byte pointer */ + *comp_ptr, /* Pointer into compression buffer */ + *start; /* Start of compression sequence */ + int count; /* Count of bytes for output */ + + + switch (type) + { + default : + /* + * Do no compression... + */ + + line_ptr = line; + line_end = line + length; + break; + + case 1 : + /* + * Do run-length encoding... + */ + + line_end = line + length; + for (line_ptr = line, comp_ptr = CompBuffer; + line_ptr < line_end; + comp_ptr += 2, line_ptr += count) + { + for (count = 1; + (line_ptr + count) < line_end && + line_ptr[0] == line_ptr[count] && + count < 256; + count ++); + + comp_ptr[0] = count - 1; + comp_ptr[1] = line_ptr[0]; + } + + line_ptr = CompBuffer; + line_end = comp_ptr; + break; + + case 2 : + /* + * Do TIFF pack-bits encoding... + */ + + line_ptr = line; + line_end = line + length; + comp_ptr = CompBuffer; + + while (line_ptr < line_end) + { + if ((line_ptr + 1) >= line_end) + { + /* + * Single byte on the end... + */ + + *comp_ptr++ = 0x00; + *comp_ptr++ = *line_ptr++; + } + else if (line_ptr[0] == line_ptr[1]) + { + /* + * Repeated sequence... + */ + + line_ptr ++; + count = 2; + + while (line_ptr < (line_end - 1) && + line_ptr[0] == line_ptr[1] && + count < 127) + { + line_ptr ++; + count ++; + } + + *comp_ptr++ = 257 - count; + *comp_ptr++ = *line_ptr++; + } + else + { + /* + * Non-repeated sequence... + */ + + start = line_ptr; + line_ptr ++; + count = 1; + + while (line_ptr < (line_end - 1) && + line_ptr[0] != line_ptr[1] && + count < 127) + { + line_ptr ++; + count ++; + } + + *comp_ptr++ = count - 1; + + memcpy(comp_ptr, start, count); + comp_ptr += count; + } + } + + line_ptr = CompBuffer; + line_end = comp_ptr; + break; + } + + /* + * Set the length of the data and write a raster plane... + */ + + printf("\033*b%d%c", (int)(line_end - line_ptr), plane); + fwrite(line_ptr, line_end - line_ptr, 1, stdout); +} + + +/* + * 'OutputLine()' - Output a line of graphics. + */ + +void +OutputLine(cups_page_header2_t *header) /* I - Page header */ +{ + int plane, /* Current plane */ + bytes, /* Bytes to write */ + count; /* Bytes to convert */ + unsigned char bit, /* Current plane data */ + bit0, /* Current low bit data */ + bit1, /* Current high bit data */ + *plane_ptr, /* Pointer into Planes */ + *bit_ptr; /* Pointer into BitBuffer */ + + + /* + * Output whitespace as needed... + */ + + if (Feed > 0) + { + printf("\033*b%dY", Feed); + Feed = 0; + } + + /* + * Write bitmap data as needed... + */ + + bytes = (header->cupsWidth + 7) / 8; + + for (plane = 0; plane < NumPlanes; plane ++) + if (ColorBits == 1) + { + /* + * Send bits as-is... + */ + + CompressData(Planes[plane], bytes, plane < (NumPlanes - 1) ? 'V' : 'W', + header->cupsCompression); + } + else + { + /* + * Separate low and high bit data into separate buffers. + */ + + for (count = header->cupsBytesPerLine / NumPlanes, + plane_ptr = Planes[plane], bit_ptr = BitBuffer; + count > 0; + count -= 2, plane_ptr += 2, bit_ptr ++) + { + bit = plane_ptr[0]; + + bit0 = ((bit & 64) << 1) | ((bit & 16) << 2) | ((bit & 4) << 3) | ((bit & 1) << 4); + bit1 = (bit & 128) | ((bit & 32) << 1) | ((bit & 8) << 2) | ((bit & 2) << 3); + + if (count > 1) + { + bit = plane_ptr[1]; + + bit0 |= (bit & 1) | ((bit & 4) >> 1) | ((bit & 16) >> 2) | ((bit & 64) >> 3); + bit1 |= ((bit & 2) >> 1) | ((bit & 8) >> 2) | ((bit & 32) >> 3) | ((bit & 128) >> 4); + } + + bit_ptr[0] = bit0; + bit_ptr[bytes] = bit1; + } + + /* + * Send low and high bits... + */ + + CompressData(BitBuffer, bytes, 'V', header->cupsCompression); + CompressData(BitBuffer + bytes, bytes, plane < (NumPlanes - 1) ? 'V' : 'W', + header->cupsCompression); + } + + fflush(stdout); +} + + +/* + * 'main()' - Main entry and processing of driver. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int fd; /* File descriptor */ + cups_raster_t *ras; /* Raster stream for printing */ + cups_page_header2_t header; /* Page header from file */ + int y; /* Current line */ + ppd_file_t *ppd; /* PPD file */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* Actions for POSIX signals */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ + + + /* + * Make sure status messages are not buffered... + */ + + setbuf(stderr, NULL); + + /* + * Check command-line... + */ + + if (argc < 6 || argc > 7) + { + /* + * We don't have the correct number of arguments; write an error message + * and return. + */ + + _cupsLangPrintFilter(stderr, "ERROR", + _("%s job-id user title copies options [file]"), + "rastertohp"); + return (1); + } + + /* + * Open the page stream... + */ + + if (argc == 7) + { + if ((fd = open(argv[6], O_RDONLY)) == -1) + { + _cupsLangPrintError("ERROR", _("Unable to open raster file")); + sleep(1); + return (1); + } + } + else + fd = 0; + + ras = cupsRasterOpen(fd, CUPS_RASTER_READ); + + /* + * Register a signal handler to eject the current page if the + * job is cancelled. + */ + + Canceled = 0; + +#ifdef HAVE_SIGSET /* Use System V signals over POSIX to avoid bugs */ + sigset(SIGTERM, CancelJob); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + + sigemptyset(&action.sa_mask); + action.sa_handler = CancelJob; + sigaction(SIGTERM, &action, NULL); +#else + signal(SIGTERM, CancelJob); +#endif /* HAVE_SIGSET */ + + /* + * Initialize the print device... + */ + + ppd = ppdOpenFile(getenv("PPD")); + if (!ppd) + { + ppd_status_t status; /* PPD error */ + int linenum; /* Line number */ + + _cupsLangPrintFilter(stderr, "ERROR", + _("The PPD file could not be opened.")); + + status = ppdLastError(&linenum); + + fprintf(stderr, "DEBUG: %s on line %d.\n", ppdErrorString(status), linenum); + + return (1); + } + + Setup(); + + /* + * Process pages as needed... + */ + + Page = 0; + + while (cupsRasterReadHeader2(ras, &header)) + { + /* + * Write a status message with the page number and number of copies. + */ + + if (Canceled) + break; + + Page ++; + + fprintf(stderr, "PAGE: %d %d\n", Page, header.NumCopies); + _cupsLangPrintFilter(stderr, "INFO", _("Starting page %d."), Page); + + /* + * Start the page... + */ + + StartPage(ppd, &header); + + /* + * Loop for each line on the page... + */ + + for (y = 0; y < header.cupsHeight; y ++) + { + /* + * Let the user know how far we have progressed... + */ + + if (Canceled) + break; + + if ((y & 127) == 0) + { + _cupsLangPrintFilter(stderr, "INFO", + _("Printing page %d, %d%% complete."), + Page, 100 * y / header.cupsHeight); + fprintf(stderr, "ATTR: job-media-progress=%d\n", + 100 * y / header.cupsHeight); + } + + /* + * Read a line of graphics... + */ + + if (cupsRasterReadPixels(ras, Planes[0], header.cupsBytesPerLine) < 1) + break; + + /* + * See if the line is blank; if not, write it to the printer... + */ + + if (Planes[0][0] || + memcmp(Planes[0], Planes[0] + 1, header.cupsBytesPerLine - 1)) + OutputLine(&header); + else + Feed ++; + } + + /* + * Eject the page... + */ + + _cupsLangPrintFilter(stderr, "INFO", _("Finished page %d."), Page); + + EndPage(); + + if (Canceled) + break; + } + + /* + * Shutdown the printer... + */ + + Shutdown(); + + if (ppd) + ppdClose(ppd); + + /* + * Close the raster stream... + */ + + cupsRasterClose(ras); + if (fd != 0) + close(fd); + + /* + * If no pages were printed, send an error message... + */ + + if (Page == 0) + { + _cupsLangPrintFilter(stderr, "ERROR", _("No pages were found.")); + return (1); + } + else + { + _cupsLangPrintFilter(stderr, "INFO", _("Ready to print.")); + return (0); + } +} + + +/* + * End of "$Id$". + */ diff --git a/filter/rastertolabel.c b/filter/rastertolabel.c new file mode 100644 index 0000000000..aa45a2b11a --- /dev/null +++ b/filter/rastertolabel.c @@ -0,0 +1,1315 @@ +/* + * "$Id$" + * + * Label printer filter for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 2001-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * Setup() - Prepare the printer for printing. + * StartPage() - Start a page of graphics. + * EndPage() - Finish a page of graphics. + * CancelJob() - Cancel the current job... + * OutputLine() - Output a line of graphics. + * PCLCompress() - Output a PCL (mode 3) compressed line. + * ZPLCompress() - Output a run-length compression sequence. + * main() - Main entry and processing of driver. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include +#include +#include +#include +#include + + +/* + * This driver filter currently supports Dymo, Intellitech, and Zebra + * label printers. + * + * The Dymo portion of the driver has been tested with the 300, 330, + * and 330 Turbo label printers; it may also work with other models. + * The Dymo printers support printing at 136, 203, and 300 DPI. + * + * The Intellitech portion of the driver has been tested with the + * Intellibar 408, 412, and 808 and supports their PCL variant. + * + * The Zebra portion of the driver has been tested with the LP-2844, + * LP-2844Z, QL-320, and QL-420 label printers; it may also work with + * other models. The driver supports EPL line mode, EPL page mode, + * ZPL, and CPCL as defined in Zebra's online developer documentation. + */ + +/* + * Model number constants... + */ + +#define DYMO_3x0 0 /* Dymo Labelwriter 300/330/330 Turbo */ + +#define ZEBRA_EPL_LINE 0x10 /* Zebra EPL line mode printers */ +#define ZEBRA_EPL_PAGE 0x11 /* Zebra EPL page mode printers */ +#define ZEBRA_ZPL 0x12 /* Zebra ZPL-based printers */ +#define ZEBRA_CPCL 0x13 /* Zebra CPCL-based printers */ + +#define INTELLITECH_PCL 0x20 /* Intellitech PCL-based printers */ + + +/* + * Globals... + */ + +unsigned char *Buffer; /* Output buffer */ +unsigned char *CompBuffer; /* Compression buffer */ +unsigned char *LastBuffer; /* Last buffer */ +int LastSet; /* Number of repeat characters */ +int ModelNumber, /* cupsModelNumber attribute */ + Page, /* Current page */ + Feed, /* Number of lines to skip */ + Canceled; /* Non-zero if job is canceled */ + + +/* + * Prototypes... + */ + +void Setup(ppd_file_t *ppd); +void StartPage(ppd_file_t *ppd, cups_page_header2_t *header); +void EndPage(ppd_file_t *ppd, cups_page_header2_t *header); +void CancelJob(int sig); +void OutputLine(ppd_file_t *ppd, cups_page_header2_t *header, int y); +void PCLCompress(unsigned char *line, int length); +void ZPLCompress(char repeat_char, int repeat_count); + + +/* + * 'Setup()' - Prepare the printer for printing. + */ + +void +Setup(ppd_file_t *ppd) /* I - PPD file */ +{ + int i; /* Looping var */ + + + /* + * Get the model number from the PPD file... + */ + + if (ppd) + ModelNumber = ppd->model_number; + + /* + * Initialize based on the model number... + */ + + switch (ModelNumber) + { + case DYMO_3x0 : + /* + * Clear any remaining data... + */ + + for (i = 0; i < 100; i ++) + putchar(0x1b); + + /* + * Reset the printer... + */ + + fputs("\033@", stdout); + break; + + case ZEBRA_EPL_LINE : + break; + + case ZEBRA_EPL_PAGE : + break; + + case ZEBRA_ZPL : + break; + + case ZEBRA_CPCL : + break; + + case INTELLITECH_PCL : + /* + * Send a PCL reset sequence. + */ + + putchar(0x1b); + putchar('E'); + break; + } +} + + +/* + * 'StartPage()' - Start a page of graphics. + */ + +void +StartPage(ppd_file_t *ppd, /* I - PPD file */ + cups_page_header2_t *header) /* I - Page header */ +{ + ppd_choice_t *choice; /* Marked choice */ + int length; /* Actual label length */ + + + /* + * Show page device dictionary... + */ + + fprintf(stderr, "DEBUG: StartPage...\n"); + fprintf(stderr, "DEBUG: MediaClass = \"%s\"\n", header->MediaClass); + fprintf(stderr, "DEBUG: MediaColor = \"%s\"\n", header->MediaColor); + fprintf(stderr, "DEBUG: MediaType = \"%s\"\n", header->MediaType); + fprintf(stderr, "DEBUG: OutputType = \"%s\"\n", header->OutputType); + + fprintf(stderr, "DEBUG: AdvanceDistance = %d\n", header->AdvanceDistance); + fprintf(stderr, "DEBUG: AdvanceMedia = %d\n", header->AdvanceMedia); + fprintf(stderr, "DEBUG: Collate = %d\n", header->Collate); + fprintf(stderr, "DEBUG: CutMedia = %d\n", header->CutMedia); + fprintf(stderr, "DEBUG: Duplex = %d\n", header->Duplex); + fprintf(stderr, "DEBUG: HWResolution = [ %d %d ]\n", header->HWResolution[0], + header->HWResolution[1]); + fprintf(stderr, "DEBUG: ImagingBoundingBox = [ %d %d %d %d ]\n", + header->ImagingBoundingBox[0], header->ImagingBoundingBox[1], + header->ImagingBoundingBox[2], header->ImagingBoundingBox[3]); + fprintf(stderr, "DEBUG: InsertSheet = %d\n", header->InsertSheet); + fprintf(stderr, "DEBUG: Jog = %d\n", header->Jog); + fprintf(stderr, "DEBUG: LeadingEdge = %d\n", header->LeadingEdge); + fprintf(stderr, "DEBUG: Margins = [ %d %d ]\n", header->Margins[0], + header->Margins[1]); + fprintf(stderr, "DEBUG: ManualFeed = %d\n", header->ManualFeed); + fprintf(stderr, "DEBUG: MediaPosition = %d\n", header->MediaPosition); + fprintf(stderr, "DEBUG: MediaWeight = %d\n", header->MediaWeight); + fprintf(stderr, "DEBUG: MirrorPrint = %d\n", header->MirrorPrint); + fprintf(stderr, "DEBUG: NegativePrint = %d\n", header->NegativePrint); + fprintf(stderr, "DEBUG: NumCopies = %d\n", header->NumCopies); + fprintf(stderr, "DEBUG: Orientation = %d\n", header->Orientation); + fprintf(stderr, "DEBUG: OutputFaceUp = %d\n", header->OutputFaceUp); + fprintf(stderr, "DEBUG: PageSize = [ %d %d ]\n", header->PageSize[0], + header->PageSize[1]); + fprintf(stderr, "DEBUG: Separations = %d\n", header->Separations); + fprintf(stderr, "DEBUG: TraySwitch = %d\n", header->TraySwitch); + fprintf(stderr, "DEBUG: Tumble = %d\n", header->Tumble); + fprintf(stderr, "DEBUG: cupsWidth = %d\n", header->cupsWidth); + fprintf(stderr, "DEBUG: cupsHeight = %d\n", header->cupsHeight); + fprintf(stderr, "DEBUG: cupsMediaType = %d\n", header->cupsMediaType); + fprintf(stderr, "DEBUG: cupsBitsPerColor = %d\n", header->cupsBitsPerColor); + fprintf(stderr, "DEBUG: cupsBitsPerPixel = %d\n", header->cupsBitsPerPixel); + fprintf(stderr, "DEBUG: cupsBytesPerLine = %d\n", header->cupsBytesPerLine); + fprintf(stderr, "DEBUG: cupsColorOrder = %d\n", header->cupsColorOrder); + fprintf(stderr, "DEBUG: cupsColorSpace = %d\n", header->cupsColorSpace); + fprintf(stderr, "DEBUG: cupsCompression = %d\n", header->cupsCompression); + fprintf(stderr, "DEBUG: cupsRowCount = %d\n", header->cupsRowCount); + fprintf(stderr, "DEBUG: cupsRowFeed = %d\n", header->cupsRowFeed); + fprintf(stderr, "DEBUG: cupsRowStep = %d\n", header->cupsRowStep); + + switch (ModelNumber) + { + case DYMO_3x0 : + /* + * Setup printer/job attributes... + */ + + length = header->PageSize[1] * header->HWResolution[1] / 72; + + printf("\033L%c%c", length >> 8, length); + printf("\033D%c", header->cupsBytesPerLine); + + printf("\033%c", header->cupsCompression + 'c'); /* Darkness */ + break; + + case ZEBRA_EPL_LINE : + /* + * Set print rate... + */ + + if ((choice = ppdFindMarkedChoice(ppd, "zePrintRate")) != NULL && + strcmp(choice->choice, "Default")) + printf("\033S%.0f", atof(choice->choice) * 2.0 - 2.0); + + /* + * Set darkness... + */ + + if (header->cupsCompression > 0 && header->cupsCompression <= 100) + printf("\033D%d", 7 * header->cupsCompression / 100); + + /* + * Set left margin to 0... + */ + + fputs("\033M01", stdout); + + /* + * Start buffered output... + */ + + fputs("\033B", stdout); + break; + + case ZEBRA_EPL_PAGE : + /* + * Start a new label... + */ + + puts(""); + puts("N"); + + /* + * Set hardware options... + */ + + if (!strcmp(header->MediaType, "Direct")) + puts("OD"); + + /* + * Set print rate... + */ + + if ((choice = ppdFindMarkedChoice(ppd, "zePrintRate")) != NULL && + strcmp(choice->choice, "Default")) + { + float val = atof(choice->choice); + + if (val >= 3.0) + printf("S%.0f\n", val); + else + printf("S%.0f\n", val * 2.0 - 2.0); + } + + /* + * Set darkness... + */ + + if (header->cupsCompression > 0 && header->cupsCompression <= 100) + printf("D%d\n", 15 * header->cupsCompression / 100); + + /* + * Set label size... + */ + + printf("q%d\n", (header->cupsWidth + 7) & ~7); + break; + + case ZEBRA_ZPL : + /* + * Set darkness... + */ + + if (header->cupsCompression > 0 && header->cupsCompression <= 100) + printf("~SD%02d\n", 30 * header->cupsCompression / 100); + + /* + * Start bitmap graphics... + */ + + printf("~DGR:CUPS.GRF,%d,%d,\n", + header->cupsHeight * header->cupsBytesPerLine, + header->cupsBytesPerLine); + + /* + * Allocate compression buffers... + */ + + CompBuffer = malloc(2 * header->cupsBytesPerLine + 1); + LastBuffer = malloc(header->cupsBytesPerLine); + LastSet = 0; + break; + + case ZEBRA_CPCL : + /* + * Start label... + */ + + printf("! 0 %u %u %u %u\r\n", header->HWResolution[0], + header->HWResolution[1], header->cupsHeight, + header->NumCopies); + printf("PAGE-WIDTH %d\r\n", header->cupsWidth); + printf("PAGE-HEIGHT %d\r\n", header->cupsWidth); + break; + + case INTELLITECH_PCL : + /* + * Set the media size... + */ + + printf("\033&l6D\033&k12H"); /* Set 6 LPI, 10 CPI */ + printf("\033&l0O"); /* Set portrait orientation */ + + switch (header->PageSize[1]) + { + case 540 : /* Monarch Envelope */ + printf("\033&l80A"); /* Set page size */ + break; + + case 624 : /* DL Envelope */ + printf("\033&l90A"); /* Set page size */ + break; + + case 649 : /* C5 Envelope */ + printf("\033&l91A"); /* Set page size */ + break; + + case 684 : /* COM-10 Envelope */ + printf("\033&l81A"); /* Set page size */ + break; + + case 756 : /* Executive */ + printf("\033&l1A"); /* Set page size */ + break; + + case 792 : /* Letter */ + printf("\033&l2A"); /* Set page size */ + break; + + case 842 : /* A4 */ + printf("\033&l26A"); /* Set page size */ + break; + + case 1008 : /* Legal */ + printf("\033&l3A"); /* Set page size */ + break; + + default : /* Custom size */ + printf("\033!f%dZ", header->PageSize[1] * 300 / 72); + break; + } + + printf("\033&l%dP", /* Set page length */ + header->PageSize[1] / 12); + printf("\033&l0E"); /* Set top margin to 0 */ + printf("\033&l%dX", header->NumCopies); + /* Set number copies */ + printf("\033&l0L"); /* Turn off perforation skip */ + + /* + * Print settings... + */ + + if (Page == 1) + { + if (header->cupsRowFeed) /* inPrintRate */ + printf("\033!p%dS", header->cupsRowFeed); + + if (header->cupsCompression != ~0) + /* inPrintDensity */ + printf("\033&d%dA", 30 * header->cupsCompression / 100 - 15); + + if ((choice = ppdFindMarkedChoice(ppd, "inPrintMode")) != NULL) + { + if (!strcmp(choice->choice, "Standard")) + fputs("\033!p0M", stdout); + else if (!strcmp(choice->choice, "Tear")) + { + fputs("\033!p1M", stdout); + + if (header->cupsRowCount) /* inTearInterval */ + printf("\033!n%dT", header->cupsRowCount); + } + else + { + fputs("\033!p2M", stdout); + + if (header->cupsRowStep) /* inCutInterval */ + printf("\033!n%dC", header->cupsRowStep); + } + } + } + + /* + * Setup graphics... + */ + + printf("\033*t%dR", header->HWResolution[0]); + /* Set resolution */ + + printf("\033*r%dS", header->cupsWidth); + /* Set width */ + printf("\033*r%dT", header->cupsHeight); + /* Set height */ + + printf("\033&a0H"); /* Set horizontal position */ + printf("\033&a0V"); /* Set vertical position */ + printf("\033*r1A"); /* Start graphics */ + printf("\033*b3M"); /* Set compression */ + + /* + * Allocate compression buffers... + */ + + CompBuffer = malloc(2 * header->cupsBytesPerLine + 1); + LastBuffer = malloc(header->cupsBytesPerLine); + LastSet = 0; + break; + } + + /* + * Allocate memory for a line of graphics... + */ + + Buffer = malloc(header->cupsBytesPerLine); + Feed = 0; +} + + +/* + * 'EndPage()' - Finish a page of graphics. + */ + +void +EndPage(ppd_file_t *ppd, /* I - PPD file */ + cups_page_header2_t *header) /* I - Page header */ +{ + int val; /* Option value */ + ppd_choice_t *choice; /* Marked choice */ + + + switch (ModelNumber) + { + case DYMO_3x0 : + /* + * Eject the current page... + */ + + fputs("\033E", stdout); + break; + + case ZEBRA_EPL_LINE : + /* + * End buffered output, eject the label... + */ + + fputs("\033E\014", stdout); + break; + + case ZEBRA_EPL_PAGE : + /* + * Print the label... + */ + + puts("P1"); + break; + + case ZEBRA_ZPL : + if (Canceled) + { + /* + * Cancel bitmap download... + */ + + puts("~DN"); + break; + } + + /* + * Start label... + */ + + puts("^XA"); + + /* + * Set print rate... + */ + + if ((choice = ppdFindMarkedChoice(ppd, "zePrintRate")) != NULL && + strcmp(choice->choice, "Default")) + { + val = atoi(choice->choice); + printf("^PR%d,%d,%d\n", val, val, val); + } + + /* + * Put label home in default position (0,0)... + */ + + printf("^LH0,0\n"); + + /* + * Set media tracking... + */ + + if (ppdIsMarked(ppd, "zeMediaTracking", "Continuous")) + { + /* + * Add label length command for continuous... + */ + + printf("^LL%d\n", header->cupsHeight); + printf("^MNN\n"); + } + else if (ppdIsMarked(ppd, "zeMediaTracking", "Web")) + printf("^MNY\n"); + else if (ppdIsMarked(ppd, "zeMediaTracking", "Mark")) + printf("^MNM\n"); + + /* + * Set label top + */ + + if (header->cupsRowStep != 200) + printf("^LT%u\n", header->cupsRowStep); + + /* + * Set media type... + */ + + if (!strcmp(header->MediaType, "Thermal")) + printf("^MTT\n"); + else if (!strcmp(header->MediaType, "Direct")) + printf("^MTD\n"); + + /* + * Set print mode... + */ + + if ((choice = ppdFindMarkedChoice(ppd, "zePrintMode")) != NULL && + strcmp(choice->choice, "Saved")) + { + printf("^MM"); + + if (!strcmp(choice->choice, "Tear")) + printf("T,Y\n"); + else if (!strcmp(choice->choice, "Peel")) + printf("P,Y\n"); + else if (!strcmp(choice->choice, "Rewind")) + printf("R,Y\n"); + else if (!strcmp(choice->choice, "Applicator")) + printf("A,Y\n"); + else + printf("C,Y\n"); + } + + /* + * Set tear-off adjust position... + */ + + if (header->AdvanceDistance != 1000) + { + if ((int)header->AdvanceDistance < 0) + printf("~TA%04d\n", (int)header->AdvanceDistance); + else + printf("~TA%03d\n", (int)header->AdvanceDistance); + } + + /* + * Allow for reprinting after an error... + */ + + if (ppdIsMarked(ppd, "zeErrorReprint", "Always")) + printf("^JZY\n"); + else if (ppdIsMarked(ppd, "zeErrorReprint", "Never")) + printf("^JZN\n"); + + /* + * Print multiple copies + */ + + if (header->NumCopies > 1) + printf("^PQ%d, 0, 0, N\n", header->NumCopies); + + /* + * Display the label image... + */ + + puts("^FO0,0^XGR:CUPS.GRF,1,1^FS"); + + /* + * End the label and eject... + */ + + puts("^IDR:CUPS.GRF^FS"); + puts("^XZ"); + + /* + * Free compression buffers... + */ + + free(CompBuffer); + free(LastBuffer); + break; + + case ZEBRA_CPCL : + /* + * Set tear-off adjust position... + */ + + if (header->AdvanceDistance != 1000) + printf("PRESENT-AT %d 1\r\n", (int)header->AdvanceDistance); + + /* + * Allow for reprinting after an error... + */ + + if (ppdIsMarked(ppd, "zeErrorReprint", "Always")) + puts("ON-OUT-OF-PAPER WAIT\r"); + else if (ppdIsMarked(ppd, "zeErrorReprint", "Never")) + puts("ON-OUT-OF-PAPER PURGE\r"); + + /* + * Cut label? + */ + + if (header->CutMedia) + puts("CUT\r"); + + /* + * Set darkness... + */ + + if (header->cupsCompression > 0) + printf("TONE %u\r\n", 2 * header->cupsCompression); + + /* + * Set print rate... + */ + + if ((choice = ppdFindMarkedChoice(ppd, "zePrintRate")) != NULL && + strcmp(choice->choice, "Default")) + { + val = atoi(choice->choice); + printf("SPEED %d\r\n", val); + } + + /* + * Print the label... + */ + + if ((choice = ppdFindMarkedChoice(ppd, "zeMediaTracking")) == NULL || + strcmp(choice->choice, "Continuous")) + puts("FORM\r"); + + puts("PRINT\r"); + break; + + case INTELLITECH_PCL : + printf("\033*rB"); /* End GFX */ + printf("\014"); /* Eject current page */ + break; + } + + fflush(stdout); + + /* + * Free memory... + */ + + free(Buffer); +} + + +/* + * 'CancelJob()' - Cancel the current job... + */ + +void +CancelJob(int sig) /* I - Signal */ +{ + /* + * Tell the main loop to stop... + */ + + (void)sig; + + Canceled = 1; +} + + +/* + * 'OutputLine()' - Output a line of graphics... + */ + +void +OutputLine(ppd_file_t *ppd, /* I - PPD file */ + cups_page_header2_t *header, /* I - Page header */ + int y) /* I - Line number */ +{ + int i; /* Looping var */ + unsigned char *ptr; /* Pointer into buffer */ + unsigned char *compptr; /* Pointer into compression buffer */ + char repeat_char; /* Repeated character */ + int repeat_count; /* Number of repeated characters */ + static const char *hex = "0123456789ABCDEF"; + /* Hex digits */ + + + switch (ModelNumber) + { + case DYMO_3x0 : + /* + * See if the line is blank; if not, write it to the printer... + */ + + if (Buffer[0] || + memcmp(Buffer, Buffer + 1, header->cupsBytesPerLine - 1)) + { + if (Feed) + { + while (Feed > 255) + { + printf("\033f\001%c", 255); + Feed -= 255; + } + + printf("\033f\001%c", Feed); + Feed = 0; + } + + putchar(0x16); + fwrite(Buffer, header->cupsBytesPerLine, 1, stdout); + fflush(stdout); + +#ifdef __sgi + /* + * This hack works around a bug in the IRIX serial port driver when + * run at high baud rates (e.g. 115200 baud)... This results in + * slightly slower label printing, but at least the labels come + * out properly. + */ + + sginap(1); +#endif /* __sgi */ + } + else + Feed ++; + break; + + case ZEBRA_EPL_LINE : + printf("\033g%03d", header->cupsBytesPerLine); + fwrite(Buffer, 1, header->cupsBytesPerLine, stdout); + fflush(stdout); + break; + + case ZEBRA_EPL_PAGE : + if (Buffer[0] || memcmp(Buffer, Buffer + 1, header->cupsBytesPerLine)) + { + printf("GW0,%d,%d,1\n", y, header->cupsBytesPerLine); + for (i = header->cupsBytesPerLine, ptr = Buffer; i > 0; i --, ptr ++) + putchar(~*ptr); + putchar('\n'); + fflush(stdout); + } + break; + + case ZEBRA_ZPL : + /* + * Determine if this row is the same as the previous line. + * If so, output a ':' and return... + */ + + if (LastSet) + { + if (!memcmp(Buffer, LastBuffer, header->cupsBytesPerLine)) + { + putchar(':'); + return; + } + } + + /* + * Convert the line to hex digits... + */ + + for (ptr = Buffer, compptr = CompBuffer, i = header->cupsBytesPerLine; + i > 0; + i --, ptr ++) + { + *compptr++ = hex[*ptr >> 4]; + *compptr++ = hex[*ptr & 15]; + } + + *compptr = '\0'; + + /* + * Run-length compress the graphics... + */ + + for (compptr = CompBuffer + 1, repeat_char = CompBuffer[0], repeat_count = 1; + *compptr; + compptr ++) + if (*compptr == repeat_char) + repeat_count ++; + else + { + ZPLCompress(repeat_char, repeat_count); + repeat_char = *compptr; + repeat_count = 1; + } + + if (repeat_char == '0') + { + /* + * Handle 0's on the end of the line... + */ + + if (repeat_count & 1) + { + repeat_count --; + putchar('0'); + } + + if (repeat_count > 0) + putchar(','); + } + else + ZPLCompress(repeat_char, repeat_count); + + fflush(stdout); + + /* + * Save this line for the next round... + */ + + memcpy(LastBuffer, Buffer, header->cupsBytesPerLine); + LastSet = 1; + break; + + case ZEBRA_CPCL : + if (Buffer[0] || memcmp(Buffer, Buffer + 1, header->cupsBytesPerLine)) + { + printf("CG %u 1 0 %d ", header->cupsBytesPerLine, y); + fwrite(Buffer, 1, header->cupsBytesPerLine, stdout); + puts("\r"); + fflush(stdout); + } + break; + + case INTELLITECH_PCL : + if (Buffer[0] || + memcmp(Buffer, Buffer + 1, header->cupsBytesPerLine - 1)) + { + if (Feed) + { + printf("\033*b%dY", Feed); + Feed = 0; + LastSet = 0; + } + + PCLCompress(Buffer, header->cupsBytesPerLine); + } + else + Feed ++; + break; + } +} + + +/* + * 'PCLCompress()' - Output a PCL (mode 3) compressed line. + */ + +void +PCLCompress(unsigned char *line, /* I - Line to compress */ + int length) /* I - Length of line */ +{ + unsigned char *line_ptr, /* Current byte pointer */ + *line_end, /* End-of-line byte pointer */ + *comp_ptr, /* Pointer into compression buffer */ + *start, /* Start of compression sequence */ + *seed; /* Seed buffer pointer */ + int count, /* Count of bytes for output */ + offset; /* Offset of bytes for output */ + + + /* + * Do delta-row compression... + */ + + line_ptr = line; + line_end = line + length; + + comp_ptr = CompBuffer; + seed = LastBuffer; + + while (line_ptr < line_end) + { + /* + * Find the next non-matching sequence... + */ + + start = line_ptr; + + if (!LastSet) + { + /* + * The seed buffer is invalid, so do the next 8 bytes, max... + */ + + offset = 0; + + if ((count = line_end - line_ptr) > 8) + count = 8; + + line_ptr += count; + } + else + { + /* + * The seed buffer is valid, so compare against it... + */ + + while (*line_ptr == *seed && + line_ptr < line_end) + { + line_ptr ++; + seed ++; + } + + if (line_ptr == line_end) + break; + + offset = line_ptr - start; + + /* + * Find up to 8 non-matching bytes... + */ + + start = line_ptr; + count = 0; + while (*line_ptr != *seed && + line_ptr < line_end && + count < 8) + { + line_ptr ++; + seed ++; + count ++; + } + } + + /* + * Place mode 3 compression data in the buffer; see HP manuals + * for details... + */ + + if (offset >= 31) + { + /* + * Output multi-byte offset... + */ + + *comp_ptr++ = ((count - 1) << 5) | 31; + + offset -= 31; + while (offset >= 255) + { + *comp_ptr++ = 255; + offset -= 255; + } + + *comp_ptr++ = offset; + } + else + { + /* + * Output single-byte offset... + */ + + *comp_ptr++ = ((count - 1) << 5) | offset; + } + + memcpy(comp_ptr, start, count); + comp_ptr += count; + } + + /* + * Set the length of the data and write it... + */ + + printf("\033*b%dW", (int)(comp_ptr - CompBuffer)); + fwrite(CompBuffer, comp_ptr - CompBuffer, 1, stdout); + + /* + * Save this line as a "seed" buffer for the next... + */ + + memcpy(LastBuffer, line, length); + LastSet = 1; +} + + +/* + * 'ZPLCompress()' - Output a run-length compression sequence. + */ + +void +ZPLCompress(char repeat_char, /* I - Character to repeat */ + int repeat_count) /* I - Number of repeated characters */ +{ + if (repeat_count > 1) + { + /* + * Print as many z's as possible - they are the largest denomination + * representing 400 characters (zC stands for 400 adjacent C's) + */ + + while (repeat_count >= 400) + { + putchar('z'); + repeat_count -= 400; + } + + /* + * Then print 'g' through 'y' as multiples of 20 characters... + */ + + if (repeat_count >= 20) + { + putchar('f' + repeat_count / 20); + repeat_count %= 20; + } + + /* + * Finally, print 'G' through 'Y' as 1 through 19 characters... + */ + + if (repeat_count > 0) + putchar('F' + repeat_count); + } + + /* + * Then the character to be repeated... + */ + + putchar(repeat_char); +} + + +/* + * 'main()' - Main entry and processing of driver. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int fd; /* File descriptor */ + cups_raster_t *ras; /* Raster stream for printing */ + cups_page_header2_t header; /* Page header from file */ + int y; /* Current line */ + ppd_file_t *ppd; /* PPD file */ + int num_options; /* Number of options */ + cups_option_t *options; /* Options */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* Actions for POSIX signals */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ + + + /* + * Make sure status messages are not buffered... + */ + + setbuf(stderr, NULL); + + /* + * Check command-line... + */ + + if (argc < 6 || argc > 7) + { + /* + * We don't have the correct number of arguments; write an error message + * and return. + */ + + _cupsLangPrintFilter(stderr, "ERROR", + _("%s job-id user title copies options [file]"), + "rastertolabel"); + return (1); + } + + /* + * Open the page stream... + */ + + if (argc == 7) + { + if ((fd = open(argv[6], O_RDONLY)) == -1) + { + _cupsLangPrintError("ERROR", _("Unable to open raster file")); + sleep(1); + return (1); + } + } + else + fd = 0; + + ras = cupsRasterOpen(fd, CUPS_RASTER_READ); + + /* + * Register a signal handler to eject the current page if the + * job is cancelled. + */ + + Canceled = 0; + +#ifdef HAVE_SIGSET /* Use System V signals over POSIX to avoid bugs */ + sigset(SIGTERM, CancelJob); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + + sigemptyset(&action.sa_mask); + action.sa_handler = CancelJob; + sigaction(SIGTERM, &action, NULL); +#else + signal(SIGTERM, CancelJob); +#endif /* HAVE_SIGSET */ + + /* + * Open the PPD file and apply options... + */ + + num_options = cupsParseOptions(argv[5], 0, &options); + + ppd = ppdOpenFile(getenv("PPD")); + if (!ppd) + { + ppd_status_t status; /* PPD error */ + int linenum; /* Line number */ + + _cupsLangPrintFilter(stderr, "ERROR", + _("The PPD file could not be opened.")); + + status = ppdLastError(&linenum); + + fprintf(stderr, "DEBUG: %s on line %d.\n", ppdErrorString(status), linenum); + + return (1); + } + + ppdMarkDefaults(ppd); + cupsMarkOptions(ppd, num_options, options); + + /* + * Initialize the print device... + */ + + Setup(ppd); + + /* + * Process pages as needed... + */ + + Page = 0; + + while (cupsRasterReadHeader2(ras, &header)) + { + /* + * Write a status message with the page number and number of copies. + */ + + if (Canceled) + break; + + Page ++; + + fprintf(stderr, "PAGE: %d 1\n", Page); + _cupsLangPrintFilter(stderr, "INFO", _("Starting page %d."), Page); + + /* + * Start the page... + */ + + StartPage(ppd, &header); + + /* + * Loop for each line on the page... + */ + + for (y = 0; y < header.cupsHeight && !Canceled; y ++) + { + /* + * Let the user know how far we have progressed... + */ + + if (Canceled) + break; + + if ((y & 15) == 0) + { + _cupsLangPrintFilter(stderr, "INFO", + _("Printing page %d, %d%% complete."), + Page, 100 * y / header.cupsHeight); + fprintf(stderr, "ATTR: job-media-progress=%d\n", + 100 * y / header.cupsHeight); + } + + /* + * Read a line of graphics... + */ + + if (cupsRasterReadPixels(ras, Buffer, header.cupsBytesPerLine) < 1) + break; + + /* + * Write it to the printer... + */ + + OutputLine(ppd, &header, y); + } + + /* + * Eject the page... + */ + + _cupsLangPrintFilter(stderr, "INFO", _("Finished page %d."), Page); + + EndPage(ppd, &header); + + if (Canceled) + break; + } + + /* + * Close the raster stream... + */ + + cupsRasterClose(ras); + if (fd != 0) + close(fd); + + /* + * Close the PPD file and free the options... + */ + + ppdClose(ppd); + cupsFreeOptions(num_options, options); + + /* + * If no pages were printed, send an error message... + */ + + if (Page == 0) + { + _cupsLangPrintFilter(stderr, "ERROR", _("No pages were found.")); + return (1); + } + else + { + _cupsLangPrintFilter(stderr, "INFO", _("Ready to print.")); + return (0); + } +} + + +/* + * End of "$Id$". + */ diff --git a/filter/rastertopwg.c b/filter/rastertopwg.c new file mode 100644 index 0000000000..622aa605ab --- /dev/null +++ b/filter/rastertopwg.c @@ -0,0 +1,461 @@ +/* + * "$Id$" + * + * CUPS raster to PWG raster format filter for CUPS. + * + * Copyright 2011 Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright law. + * Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Main entry for filter. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include + + +/* + * 'main()' - Main entry for filter. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int fd; /* Raster file */ + cups_raster_t *inras, /* Input raster stream */ + *outras; /* Output raster stream */ + cups_page_header2_t inheader, /* Input raster page header */ + outheader; /* Output raster page header */ + int y; /* Current line */ + unsigned char *line; /* Line buffer */ + int page = 0, /* Current page */ + page_width, /* Actual page width */ + page_height, /* Actual page height */ + page_top, /* Top margin */ + page_bottom, /* Bottom margin */ + page_left, /* Left margin */ + linesize, /* Bytes per line */ + lineoffset; /* Offset into line */ + unsigned char white; /* White pixel */ + ppd_file_t *ppd; /* PPD file */ + ppd_attr_t *back; /* cupsBackSize attribute */ + _ppd_cache_t *cache; /* PPD cache */ + _pwg_size_t *pwg_size; /* PWG media size */ + _pwg_media_t *pwg_media; /* PWG media name */ + int num_options; /* Number of options */ + cups_option_t *options = NULL;/* Options */ + const char *val; /* Option value */ + + + if (argc < 6 || argc > 7) + { + puts("Usage: rastertopwg job user title copies options [filename]"); + return (1); + } + else if (argc == 7) + { + if ((fd = open(argv[6], O_RDONLY)) < 0) + { + perror("ERROR: Unable to open print file"); + return (1); + } + } + else + fd = 0; + + inras = cupsRasterOpen(fd, CUPS_RASTER_READ); + outras = cupsRasterOpen(1, CUPS_RASTER_WRITE_PWG); + + ppd = ppdOpenFile(getenv("PPD")); + back = ppdFindAttr(ppd, "cupsBackSide", NULL); + + num_options = cupsParseOptions(argv[5], 0, &options); + + ppdMarkDefaults(ppd); + cupsMarkOptions(ppd, num_options, options); + + cache = ppd ? ppd->cache : NULL; + + while (cupsRasterReadHeader2(inras, &inheader)) + { + /* + * Compute the real raster size... + */ + + page ++; + + fprintf(stderr, "PAGE: %d %d\n", page, inheader.NumCopies); + + page_width = (int)(inheader.cupsPageSize[0] * inheader.HWResolution[0] / + 72.0); + page_height = (int)(inheader.cupsPageSize[1] * inheader.HWResolution[1] / + 72.0); + page_left = (int)(inheader.cupsImagingBBox[0] * + inheader.HWResolution[0] / 72.0); + page_bottom = (int)(inheader.cupsImagingBBox[1] * + inheader.HWResolution[1] / 72.0); + page_top = page_height - page_bottom - inheader.cupsHeight; + linesize = (page_width * inheader.cupsBitsPerPixel + 7) / 8; + lineoffset = page_left * inheader.cupsBitsPerPixel / 8; /* Round down */ + + switch (inheader.cupsColorSpace) + { + case CUPS_CSPACE_W : + case CUPS_CSPACE_RGB : + case CUPS_CSPACE_SW : + case CUPS_CSPACE_SRGB : + case CUPS_CSPACE_ADOBERGB : + white = 255; + break; + + case CUPS_CSPACE_K : + case CUPS_CSPACE_CMYK : + case CUPS_CSPACE_DEVICE1 : + case CUPS_CSPACE_DEVICE2 : + case CUPS_CSPACE_DEVICE3 : + case CUPS_CSPACE_DEVICE4 : + case CUPS_CSPACE_DEVICE5 : + case CUPS_CSPACE_DEVICE6 : + case CUPS_CSPACE_DEVICE7 : + case CUPS_CSPACE_DEVICE8 : + case CUPS_CSPACE_DEVICE9 : + case CUPS_CSPACE_DEVICEA : + case CUPS_CSPACE_DEVICEB : + case CUPS_CSPACE_DEVICEC : + case CUPS_CSPACE_DEVICED : + case CUPS_CSPACE_DEVICEE : + case CUPS_CSPACE_DEVICEF : + white = 0; + break; + + default : + _cupsLangPrintFilter(stderr, "ERROR", _("Unsupported raster data.")); + fprintf(stderr, "DEBUG: Unsupported cupsColorSpace %d on page %d.\n", + inheader.cupsColorSpace, page); + return (1); + } + + if (inheader.cupsColorOrder != CUPS_ORDER_CHUNKED) + { + _cupsLangPrintFilter(stderr, "ERROR", _("Unsupported raster data.")); + fprintf(stderr, "DEBUG: Unsupported cupsColorOrder %d on page %d.\n", + inheader.cupsColorOrder, page); + return (1); + } + + if (inheader.cupsBitsPerPixel != 1 && + inheader.cupsBitsPerColor != 8 && inheader.cupsBitsPerColor != 16) + { + _cupsLangPrintFilter(stderr, "ERROR", _("Unsupported raster data.")); + fprintf(stderr, "DEBUG: Unsupported cupsBitsPerColor %d on page %d.\n", + inheader.cupsBitsPerColor, page); + return (1); + } + + memcpy(&outheader, &inheader, sizeof(outheader)); + outheader.cupsWidth = page_width; + outheader.cupsHeight = page_height; + outheader.cupsBytesPerLine = linesize; + + outheader.cupsInteger[14] = 0; /* VendorIdentifier */ + outheader.cupsInteger[15] = 0; /* VendorLength */ + + if ((val = cupsGetOption("print-content-optimize", num_options, + options)) != NULL) + { + if (!strcmp(val, "automatic")) + strlcpy(outheader.OutputType, "Automatic", + sizeof(outheader.OutputType)); + else if (!strcmp(val, "graphics")) + strlcpy(outheader.OutputType, "Graphics", sizeof(outheader.OutputType)); + else if (!strcmp(val, "photo")) + strlcpy(outheader.OutputType, "Photo", sizeof(outheader.OutputType)); + else if (!strcmp(val, "text")) + strlcpy(outheader.OutputType, "Text", sizeof(outheader.OutputType)); + else if (!strcmp(val, "text-and-graphics")) + strlcpy(outheader.OutputType, "TextAndGraphics", + sizeof(outheader.OutputType)); + else + { + fprintf(stderr, "DEBUG: Unsupported print-content-type \"%s\".\n", val); + outheader.OutputType[0] = '\0'; + } + } + + if ((val = cupsGetOption("print-quality", num_options, options)) != NULL) + { + int quality = atoi(val); /* print-quality value */ + + if (quality >= IPP_QUALITY_DRAFT && quality <= IPP_QUALITY_HIGH) + outheader.cupsInteger[8] = quality; + else + { + fprintf(stderr, "DEBUG: Unsupported print-quality %d.\n", quality); + outheader.cupsInteger[8] = 0; + } + } + + if ((val = cupsGetOption("print-rendering-intent", num_options, + options)) != NULL) + { + if (!strcmp(val, "absolute")) + strlcpy(outheader.cupsRenderingIntent, "Absolute", + sizeof(outheader.cupsRenderingIntent)); + else if (!strcmp(val, "automatic")) + strlcpy(outheader.cupsRenderingIntent, "Automatic", + sizeof(outheader.cupsRenderingIntent)); + else if (!strcmp(val, "perceptual")) + strlcpy(outheader.cupsRenderingIntent, "Perceptual", + sizeof(outheader.cupsRenderingIntent)); + else if (!strcmp(val, "relative")) + strlcpy(outheader.cupsRenderingIntent, "Relative", + sizeof(outheader.cupsRenderingIntent)); + else if (!strcmp(val, "relative-bpc")) + strlcpy(outheader.cupsRenderingIntent, "RelativeBpc", + sizeof(outheader.cupsRenderingIntent)); + else if (!strcmp(val, "saturation")) + strlcpy(outheader.cupsRenderingIntent, "Saturation", + sizeof(outheader.cupsRenderingIntent)); + else + { + fprintf(stderr, "DEBUG: Unsupported print-rendering-intent \"%s\".\n", + val); + outheader.cupsRenderingIntent[0] = '\0'; + } + } + + if (inheader.cupsPageSizeName[0] && + (pwg_size = _ppdCacheGetSize(cache, inheader.cupsPageSizeName)) != NULL) + { + strlcpy(outheader.cupsPageSizeName, pwg_size->map.pwg, + sizeof(outheader.cupsPageSizeName)); + } + else + { + pwg_media = _pwgMediaForSize((int)(2540.0 * inheader.cupsPageSize[0] / + 72.0), + (int)(2540.0 * inheader.cupsPageSize[1] / + 72.0)); + + if (pwg_media) + strlcpy(outheader.cupsPageSizeName, pwg_media->pwg, + sizeof(outheader.cupsPageSizeName)); + else + { + fprintf(stderr, "DEBUG: Unsupported PageSize %.2fx%.2f.\n", + inheader.cupsPageSize[0], inheader.cupsPageSize[1]); + outheader.cupsPageSizeName[0] = '\0'; + } + } + + if (inheader.Duplex && !(page & 1) && + back && _cups_strcasecmp(back->value, "Normal")) + { + if (_cups_strcasecmp(back->value, "Flipped")) + { + if (inheader.Tumble) + { + outheader.cupsInteger[1] = -1;/* CrossFeedTransform */ + outheader.cupsInteger[2] = 1; /* FeedTransform */ + + outheader.cupsInteger[3] = page_width - page_left - + inheader.cupsWidth; + /* ImageBoxLeft */ + outheader.cupsInteger[4] = page_top; + /* ImageBoxTop */ + outheader.cupsInteger[5] = page_width - page_left; + /* ImageBoxRight */ + outheader.cupsInteger[6] = page_height - page_bottom; + /* ImageBoxBottom */ + } + else + { + outheader.cupsInteger[1] = 1; /* CrossFeedTransform */ + outheader.cupsInteger[2] = -1;/* FeedTransform */ + + outheader.cupsInteger[3] = page_left; + /* ImageBoxLeft */ + outheader.cupsInteger[4] = page_bottom; + /* ImageBoxTop */ + outheader.cupsInteger[5] = page_left + inheader.cupsWidth; + /* ImageBoxRight */ + outheader.cupsInteger[6] = page_height - page_top; + /* ImageBoxBottom */ + } + } + else if (_cups_strcasecmp(back->value, "ManualTumble")) + { + if (inheader.Tumble) + { + outheader.cupsInteger[1] = -1;/* CrossFeedTransform */ + outheader.cupsInteger[2] = -1;/* FeedTransform */ + + outheader.cupsInteger[3] = page_width - page_left - + inheader.cupsWidth; + /* ImageBoxLeft */ + outheader.cupsInteger[4] = page_bottom; + /* ImageBoxTop */ + outheader.cupsInteger[5] = page_width - page_left; + /* ImageBoxRight */ + outheader.cupsInteger[6] = page_height - page_top; + /* ImageBoxBottom */ + } + else + { + outheader.cupsInteger[1] = 1; /* CrossFeedTransform */ + outheader.cupsInteger[2] = 1; /* FeedTransform */ + + outheader.cupsInteger[3] = page_left; + /* ImageBoxLeft */ + outheader.cupsInteger[4] = page_top; + /* ImageBoxTop */ + outheader.cupsInteger[5] = page_left + inheader.cupsWidth; + /* ImageBoxRight */ + outheader.cupsInteger[6] = page_height - page_bottom; + /* ImageBoxBottom */ + } + } + else if (_cups_strcasecmp(back->value, "Rotated")) + { + if (inheader.Tumble) + { + outheader.cupsInteger[1] = -1;/* CrossFeedTransform */ + outheader.cupsInteger[2] = -1;/* FeedTransform */ + + outheader.cupsInteger[3] = page_width - page_left - + inheader.cupsWidth; + /* ImageBoxLeft */ + outheader.cupsInteger[4] = page_bottom; + /* ImageBoxTop */ + outheader.cupsInteger[5] = page_width - page_left; + /* ImageBoxRight */ + outheader.cupsInteger[6] = page_height - page_top; + /* ImageBoxBottom */ + } + else + { + outheader.cupsInteger[1] = 1; /* CrossFeedTransform */ + outheader.cupsInteger[2] = 1; /* FeedTransform */ + + outheader.cupsInteger[3] = page_left; + /* ImageBoxLeft */ + outheader.cupsInteger[4] = page_top; + /* ImageBoxTop */ + outheader.cupsInteger[5] = page_left + inheader.cupsWidth; + /* ImageBoxRight */ + outheader.cupsInteger[6] = page_height - page_bottom; + /* ImageBoxBottom */ + } + } + else + { + /* + * Unsupported value... + */ + + fprintf(stderr, "DEBUG: Unsupported cupsBackSide \"%s\".\n", back->value); + + outheader.cupsInteger[1] = 1; /* CrossFeedTransform */ + outheader.cupsInteger[2] = 1; /* FeedTransform */ + + outheader.cupsInteger[3] = page_left; + /* ImageBoxLeft */ + outheader.cupsInteger[4] = page_top; + /* ImageBoxTop */ + outheader.cupsInteger[5] = page_left + inheader.cupsWidth; + /* ImageBoxRight */ + outheader.cupsInteger[6] = page_height - page_bottom; + /* ImageBoxBottom */ + } + } + else + { + outheader.cupsInteger[1] = 1; /* CrossFeedTransform */ + outheader.cupsInteger[2] = 1; /* FeedTransform */ + + outheader.cupsInteger[3] = page_left; + /* ImageBoxLeft */ + outheader.cupsInteger[4] = page_top; + /* ImageBoxTop */ + outheader.cupsInteger[5] = page_left + inheader.cupsWidth; + /* ImageBoxRight */ + outheader.cupsInteger[6] = page_height - page_bottom; + /* ImageBoxBottom */ + } + + if (!cupsRasterWriteHeader2(outras, &outheader)) + { + _cupsLangPrintFilter(stderr, "ERROR", _("Error sending raster data.")); + fprintf(stderr, "DEBUG: Unable to write header for page %d.\n", page); + return (1); + } + + /* + * Copy raster data... + */ + + line = malloc(linesize); + + memset(line, white, linesize); + for (y = page_top; y > 0; y --) + if (!cupsRasterWritePixels(outras, line, outheader.cupsBytesPerLine)) + { + _cupsLangPrintFilter(stderr, "ERROR", _("Error sending raster data.")); + fprintf(stderr, "DEBUG: Unable to write line %d for page %d.\n", + page_top - y + 1, page); + return (1); + } + + for (y = inheader.cupsHeight; y > 0; y --) + { + cupsRasterReadPixels(inras, line + lineoffset, inheader.cupsBytesPerLine); + if (!cupsRasterWritePixels(outras, line, outheader.cupsBytesPerLine)) + { + _cupsLangPrintFilter(stderr, "ERROR", _("Error sending raster data.")); + fprintf(stderr, "DEBUG: Unable to write line %d for page %d.\n", + inheader.cupsHeight - y + page_top + 1, page); + return (1); + } + } + + memset(line, white, linesize); + for (y = page_bottom; y > 0; y --) + if (!cupsRasterWritePixels(outras, line, outheader.cupsBytesPerLine)) + { + _cupsLangPrintFilter(stderr, "ERROR", _("Error sending raster data.")); + fprintf(stderr, "DEBUG: Unable to write line %d for page %d.\n", + page_bottom - y + page_top + inheader.cupsHeight + 1, page); + return (1); + } + + free(line); + } + + cupsRasterClose(inras); + if (fd) + close(fd); + + cupsRasterClose(outras); + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/filter/spec-ppd.header b/filter/spec-ppd.header new file mode 100644 index 0000000000..c8793d3785 --- /dev/null +++ b/filter/spec-ppd.header @@ -0,0 +1,32 @@ + + +

CUPS PPD Extensions

+ +

This specification describes the attributes and extensions that CUPS adds to Adobe TechNote #5003: PostScript Printer Description File Format Specification Version 4.3. PostScript Printer Description ("PPD") files describe the capabilities of each printer and are used by CUPS to support printer-specific features and intelligent filtering.

+ + diff --git a/filter/spec-ppd.shtml b/filter/spec-ppd.shtml new file mode 100644 index 0000000000..670e89913c --- /dev/null +++ b/filter/spec-ppd.shtml @@ -0,0 +1,1898 @@ +

PPD File Syntax

+ +

The PPD format is text-based and uses lines of up to 255 characters terminated by a carriage return, linefeed, or combination of carriage return and line feed. The following ABNF definition [RFC5234] defines the general format of lines in a PPD file:

+ +
+PPD-FILE = HEADER +(DATA / COMMENT / LINE-END)
+
+HEADER   = "*PPD-Adobe:" *WSP DQUOTE VERSION DQUOTE LINE-END
+
+VERSION  = "4.0" / "4.1" / "4.2" / "4.3"
+
+COMMENT  = "*%" *TCHAR LINE-END
+
+DATA     = "*" 1*KCHAR [ WSP 1*KCHAR [ "/" 1*TCHAR ] ] ":"
+           1*(*WSP VALUE) LINE-END
+
+VALUE    = 1*TCHAR / DQUOTE 1*SCHAR DQUOTE
+
+KCHAR    = ALPHA / DIGIT / "_" / "." / "-"
+
+SCHAR    = LINE-END / WSP / %x21.23-7E.A0-FF
+
+TCHAR    = %x20-7E.A0-FF
+
+LINE-END = CR / LF / CR LF
+
+ + +

Auto-Configuration

+ +

CUPS supports several methods of auto-configuration via PPD keywords.

+ +

Mac OS X 10.5APAutoSetupTool

+ +

*APAutoSetupTool: "/LibraryPrinters/vendor/filename"

+ +

This Mac OS X keyword defines a program that sets the default option choices. It is run when a printer is added from the Add Printer window or the Nearby Printers list in the Print dialog.

+ +

The program is provided with two arguments: the printer's device URI and the PPD file to be used for the printer. The program must write an updated PPD file to stdout.

+ +

Examples:

+ +
+*% Use our setup tool when adding a printer
+*APAutoSetupTool: "/Library/Printers/vendor/Tools/autosetuptool"
+
+ +

Mac OS X 10.2/CUPS 1.4?MainKeyword

+ +

*?MainKeyword: "
+ PostScript query code that writes a message using the = operator...
+"
+*End

+ +

The ?MainKeyword keyword defines PostScript code that determines the currently selected/enabled option keyword (choice) for the main keyword (option). It is typically used when communicating with USB, serial, Appletalk, and AppSocket (port 9100) printers.

+ +

The PostScript code typically sends its response back using the = operator.

+ +

Example:

+ +
+*OpenUI OptionDuplex/Duplexer Installed: Boolean
+*DuplexOptionDuplex: False
+*OptionDuplex False/Not Installed: ""
+*OptionDuplex True/Installed: ""
+
+*% Query the printer for the presence of the duplexer option...
+*?OptionDuplex: "
+  currentpagedevice /Duplex known
+  {(True)} {(False)} ifelse
+  = flush
+"
+*End
+*CloseUI: OptionDuplex
+
+ +

Mac OS X 10.4/CUPS 1.5OIDMainKeyword

+ +

*?OIDMainKeyword: ".n.n.n..."
+*OIDMainKeyword OptionKeyword1: "value"
+...
+*OIDMainKeyword OptionKeywordN: "value"

+ +

The OIDMainKeyword keyword is used to define SNMP OIDs that map to installable options. The first (query) line defines the OID to lookup on the network device. The second and subsequent keywords define a mapping from OID value to option keyword. Since SNMP is an IP-based network protocol, this method is typically only used to configure AppSocket, IPP, and LPD network printers.

+ +

Examples:

+ +
+*% Get the installed memory on the printer...
+*?OIDInstalledMemory: ".1.3.6.1.2.1.25.2.2.0"
+*OIDInstalledMemory 16MB: "16384 KBytes"
+*OIDInstalledMemory 32MB: "32768 KBytes"
+*OIDInstalledMemory 48MB: "49152 KBytes"
+*OIDInstalledMemory 72MB: "73728 KBytes"
+
+ + +

Color Profiles

+ +

CUPS supports three types of color profiles. The first type is based on sRGB and is used by the standard CUPS raster filters and GPL Ghostscript. The second type is based on ICC profiles and is used by the Quartz-based filters on MacOS X. The final type is based on well-known colorspaces such as sRGB and Adobe RGB.

+ +
Note: + +

At this time, none of the CUPS raster filters support ICC profiles. This will be addressed as time and resources permit.

+ +
+ +

DeprecatedcupsColorProfile

+ +

*cupsColorProfile Resolution/MediaType: "density gamma m00 m01 m02 m10 m11 m12 m20 m21 m22"

+ +

This string keyword specifies an sRGB-based color profile consisting of gamma and density controls and a 3x3 CMY color transform matrix. This keyword is not supported on Mac OS X.

+ +

The Resolution and MediaType values may be "-" to act as a wildcard. Otherwise they must match one of the Resolution or MediaType option keywords defined in the PPD file.

+ +

The density and gamma values define gamma and +density adjustment function such that:

+ +
+f(x) = density * x gamma
+
+ +

The m00 through m22 values define a 3x3 transformation matrix for the CMY color values. The density function is applied after the CMY transformation:

+ +
+| m00 m01 m02 |
+| m10 m11 m12 |
+| m20 m21 m22 |
+
+ +

Examples:

+ +
+*% Specify a profile for printing at 360dpi on all media types
+*cupsColorProfile 360dpi/-: "1.0 1.5 1.0 0.0 -0.2 -0.4 1.0 0.0 -0.2 0.0 1.0"
+
+*% Specify a profile for printing at 720dpi on Glossy media
+*cupsColorProfile 720dpi/Glossy: "1.0 2.5 1.0 0.0 -0.2 -0.4 1.0 0.0 -0.2 0.0 1.0"
+
+*% Specify a default profile for printing at all other resolutions and media types
+*cupsColorProfile -/-: "0.9 2.0 1.0 0.0 -0.2 -0.4 1.0 0.0 -0.2 0.0 1.0"
+
+ + +

Mac OS X 10.3/CUPS 1.2cupsICCProfile

+ +

*cupsICCProfile ColorModel.MediaType.Resolution/Description: "filename"

+ +

This keyword specifies an ICC color profile that is used to convert the document colors to the device colorspace. The ColorModel, MediaType, and Resolution option keywords specify a selector for color profiles. If omitted, the color profile will match any option keyword for the corresponding main keyword.

+ +

The Description specifies human-readable text that is associated with the color profile. The filename portion specifies the ICC color profile to use; if the filename is not absolute, it is loaded relative to the /usr/share/cups/profiles directory.

+ +

Examples:

+ +
+*% Specify a profile for CMYK printing at 360dpi on all media types
+*cupsICCProfile CMYK..360dpi/360dpi CMYK: "/Library/Printers/vendor/Profiles/foo-360-cmyk.icc"
+
+*% Specify a profile for RGB printing at 720dpi on Glossy media
+*cupsColorProfile RGB.Glossy.720dpi/720dpi Glossy: "/Library/Printers/vendor/Profiles/foo-720-glossy-rgb.icc"
+
+*% Specify a default profile for printing at all other resolutions and media types
+*cupsICCProfile ../Default: "/Library/Printers/vendor/Profiles/foo-default.icc"
+
+ +

Customizing the Profile Selection Keywords

+ +

The ColorModel, MediaType, and Resolution main keywords can be reassigned to different main keywords, allowing drivers to do color profile selection based on different parameters. The cupsICCQualifier1, cupsICCQualifier2, and cupsICCQualifier3 keywords define the mapping from selector to main keyword:

+ +
+*cupsICCQualifier1: MainKeyword1
+*cupsICCQualifier2: MainKeyword2
+*cupsICCQualifier3: MainKeyword3
+
+ +

The default mapping is as follows:

+ +
+*cupsICCQualifier1: ColorModel
+*cupsICCQualifier2: MediaType
+*cupsICCQualifier3: Resolution
+
+ +

Mac OS X 10.4Custom Color Matching Support

+ +

*APSupportsCustomColorMatching: true
+*APCustomColorMatchingName name/text: ""
+*APCustomColorMatchingProfile: profile
+*APDefaultCustomColorMatchingProfile: profile

+ +

These keywords tell the Mac OS X raster filters that the printer driver provides its own custom color matching and that generic color profiles should be used when generating 1-, 3-, and 4-component raster data as requested by the driver. The APCustomColorMatchingProfile and APDefaultColorMatchingProfile keywords specify alternate color profiles (sRGB or AdobeRGB) to use for 3-color (RGB) raster data.

+ +
Note: + +

Prior to Mac OS X 10.6, the default RGB color space was Apple's "GenericRGB". The new default in Mac OS X 10.6 and later is "sRGB". For more information, see "Mac OS X v10.6: About gamma 2.2" on Apple's support site.

+ +
+ +

Mac OS X 10.5APCustomColorMatchingName

+ +

*APCustomColorMatchingName name/text: ""

+ +

This keyword defines an alternate name for the color matching provided by a driver in the Color Matching print panel. The default is to use the name "Vendor Matching" or its localized equivalent.

+ +

Examples:

+ +
+*% Define the names for our color matching...
+*APCustomColorMatchingName name/AcmeColor(tm): ""
+*fr.APCustomColorMatchingName name/La AcmeColor(tm): ""
+
+ +

Mac OS X 10.5APCustomColorMatchingProfile

+ +

*APCustomColorMatchingProfile: name

+ +

This keyword defines a supported RGB color profile that can be used when doing custom color matching. Currently only sRGB, AdobeRGB, and GenericRGB are supported. If not specified, RGB data will use the GenericRGB colorspace.

+ +
Note: + +

If you provide multiple APCustomColorMatchingProfile keywords, you are responsible for providing the necessary user interface controls to select the profile in a print dialog pane. Add the named profile to the print settings using the key kPMCustomColorMatchingProfileKey.

+ +
+ +

Examples:

+ +
+*% Use sRGB for RGB color by default, but support both sRGB and AdobeRGB
+*APSupportsCustomColorMatching: true
+*APDefaultCustomColorMatchingProfile: sRGB
+*APCustomColorMatchingProfile: sRGB
+*APCustomColorMatchingProfile: AdobeRGB
+
+ +

Mac OS X 10.5APDefaultCustomColorMatchingProfile

+ +

*APDefaultCustomColorMatchingProfile: name

+ +

This keyword defines the default RGB color profile that will be used when doing custom color matching. Currently only sRGB, AdobeRGB, and GenericRGB are supported.

+ +

Examples:

+ +
+*% Use sRGB for RGB color by default
+*APSupportsCustomColorMatching: true
+*APDefaultCustomColorMatchingProfile: sRGB
+
+ +

Mac OS X 10.4APSupportsCustomColorMatching

+ +

*APSupportsCustomColorMatching: boolean

+ +

This keyword specifies that the driver provides its own custom color matching. When true, the default hand-off colorspace will be GenericGray, GenericRGB, or GenericCMYK depending on the number of components the driver requests. The APDefaultCustomColorMatchingProfile keyword can be used to override the default 3-component (RGB) colorspace.

+ +

The default for APSupportsCustomColorMatching is false.

+ +

Examples:

+ +
+*APSupportsCustomColorMatching: true
+*APDefaultCustomColorMatchingProfile: sRGB
+
+ + +

Constraints

+ +

Constraints are option choices that are not allowed by the driver or device, for example printing 2-sided transparencies. All versions of CUPS support constraints defined by the legacy Adobe UIConstraints and NonUIConstraints keywords which support conflicts between any two option choices, for example:

+ +
+*% Do not allow 2-sided printing on transparency media
+*UIConstraints: "*Duplex *MediaType Transparency"
+*UIConstraints: "*MediaType Transparency *Duplex"
+
+ +

While nearly all constraints can be expressed using these keywords, there are valid scenarios requiring constraints between more than two option choices. In addition, resolution of constraints is problematic since users and software have to guess how a particular constraint is best resolved.

+ +

CUPS 1.4 and higher define two new keywords for constraints, cupsUIConstraints and cupsUIResolver. Each cupsUIConstraints keyword points to a cupsUIResolver keyword which specifies alternate options that resolve the conflict condition. The same cupsUIResolver can be used by multiple cupsUIConstraints.

+ +
Note: + +

When developing PPD files that contain constraints, it is very important to use the cupstestppd(1) program to verify that your constraints are accurate and cannot result in unresolvable option selections.

+ +
+ +

CUPS 1.4/Mac OS X 10.6cupsUIConstraints

+ +

*cupsUIConstraints resolver: "*Keyword1 *Keyword2 ..."
+*cupsUIConstraints resolver: "*Keyword1 OptionKeyword1 *Keyword2 ..."
+*cupsUIConstraints resolver: "*Keyword1 *Keyword2 OptionKeyword2 ..."
+*cupsUIConstraints resolver: "*Keyword1 OptionKeyword1 *Keyword2 OptionKeyword2 ..."
+*cupsUIConstraints: "*InstallableKeyword1 OptionKeyword1 *Keyword2 OptionKeyword2 ..."

+ +

Lists two or more options which conflict. The "resolver" string is a (possibly unique) keyword which specifies which options to change when the constraint exists. When no resolver is provided, CUPS first tries the default choice followed by testing each option choice to resolve the conflict.

+ +

Examples:

+ +
+*% Specify that 2-sided printing cannot happen on transparencies
+*cupsUIConstraints transparency: "*Duplex *MediaType Transparency"
+
+*% Specify that envelope printing cannot happen from the paper trays
+*cupsUIConstraints envelope: "*PageSize Env10 *InputSlot Tray1"
+*cupsUIConstraints envelope: "*PageSize Env10 *InputSlot Tray1"
+*cupsUIConstraints envelope: "*PageSize EnvDL *InputSlot Tray2"
+*cupsUIConstraints envelope: "*PageSize EnvDL *InputSlot Tray2"
+
+*% Specify an installable option constraint for the envelope feeder
+*cupsUIConstraints: "*InputSlot EnvFeeder *InstalledEnvFeeder"
+
+*% Specify that photo printing cannot happen on plain paper or transparencies at 1200dpi
+*cupsUIConstraints photo: "*OutputMode Photo *MediaType Plain *Resolution 1200dpi"
+*cupsUIConstraints photo: "*OutputMode Photo *MediaType Transparency *Resolution 1200dpi"
+
+ +

CUPS 1.4/Mac OS X 10.6cupsUIResolver

+ +

*cupsUIResolver resolver: "*Keyword1 OptionKeyword1 *Keyword2 OptionKeyword2 ..."

+ +

Specifies two or more options to mark/select to resolve a constraint. The "resolver" string identifies a particular action to take for one or more cupsUIConstraints. The same action can be used for multiple constraints. The option keyword pairs are treated as an ordered list of option selections to try - only the first N selections will be used, where N is the minimum number of selections required. Because cupsResolveConflicts() will not change the most recent option selection passed to it, at least two options from the constraints must be listed to avoid situations where conflicts cannot be resolved.

+ +

Examples:

+ +
+*% Specify the options to change for the 2-sided transparency constraint
+*cupsUIResolver transparency: "*Duplex None *MediaType Plain"
+
+*% Specify the options to change for the envelope printing constraints.  Notice
+*% that we try to change the InputSlot to either the envelope feeder or the
+*% manual feed first, then we change the page size...
+*cupsUIResolver envelope: "*InputSlot EnvFeeder *InputSlot ManualFeed *PageSize Letter"
+
+*% Specify the options to change for the photo printing constraints
+*cupsUIResolver photo: "*OutputMode Best *Resolution 600dpi"
+
+ + +

Globalized PPD Support

+ +

CUPS 1.2 and higher adds support for PPD files containing multiple languages by following the following additional rules:

+ +
    + +
  1. The LanguageVersion MUST be English
  2. + +
  3. The LanguageEncoding MUST be ISOLatin1
  4. + +
  5. The cupsLanguages keyword MUST be provided and list each of the supported locales in the PPD file
  6. + +
  7. Main and option keywords MUST NOT exceed 34 (instead of 40) characters to allow room for the locale prefixes in translation keywords
  8. + +
  9. The main keyword "Translation" MUST NOT be used
  10. + +
  11. Translation strings included with the main and option keywords MUST NOT contain characters outside the ASCII subset of ISOLatin1 and UTF-8; developers wishing to use characters outside ASCII MUST provide a separate set of English localization keywords for the affected keywords.
  12. + +
  13. Localizations are specified using a locale prefix of the form "ll" or "ll_CC." where "ll" is the 2-letter ISO language code and "CC" is the 2-letter ISO country code
      +
    • A generic language translation ("ll") SHOULD be provided with country-specific differences ("ll_CC") provided only as needed
    • +
    • For historical reasons, the "zh" and "zh_CN" locales map to Simplified Chinese while the "zh_TW" locale maps to Traditional Chinese
    • +
  14. + +
  15. Locale-specific translation strings MUST be encoded using UTF-8.
  16. + +
  17. Main keywords MUST be localized using one of the following forms: +

    *ll.Translation MainKeyword/translation text: ""
    + *ll_CC.Translation MainKeyword/translation text: ""

  18. + +
  19. Option keywords MUST be localized using one of the following forms: +

    *ll.MainKeyword OptionKeyword/translation text: ""
    + *ll_CC.MainKeyword OptionKeyword/translation text: ""

  20. + +
  21. Localization keywords MAY appear anywhere after the first line of the PPD file
  22. + +
+ +
Note: + +

We use a LanguageEncoding value of ISOLatin1 and limit the allowed base translation strings to ASCII to avoid character coding issues that would otherwise occur. In addition, requiring the base translation strings to be in English allows for easier fallback translation when no localization is provided in the PPD file for a given locale.

+ +
+ +

Examples:

+ +
+*LanguageVersion: English
+*LanguageEncoding: ISOLatin1
+*cupsLanguages: "de fr_CA"
+*ModelName: "Foobar Laser 9999"
+
+*% Localize ModelName for French and German
+*fr_CA.Translation ModelName/La Foobar Laser 9999: ""
+*de.Translation ModelName/Foobar LaserDrucken 9999: ""
+
+*cupsIPPReason com.vendor-error/A serious error occurred: "/help/com.vendor/error.html"
+*% Localize printer-state-reason for French and German
+*fr_CA.cupsIPPReason com.vendor-error/Une erreur sèrieuse s'est produite: "/help/com.vendor/error.html"
+*de.cupsIPPReason com.vendor-error/Eine ernste Störung trat: "/help/com.vendor/error.html"
+
+...
+
+*OpenUI *InputSlot/Paper Source: PickOne
+*OrderDependency: 10 AnySetup *InputSlot
+*DefaultInputSlot: Auto
+*% Localize InputSlot for French and German
+*fr_CA.Translation InputSlot/Papier source: ""
+*de.Translation InputSlot/Papiereinzug: ""
+*InputSlot Auto/Default: "<</ManualFeed false>>setpagedevice"
+*% Localize InputSlot=Auto for French and German
+*fr_CA.InputSlot Auto/Par Defaut: ""
+*de.InputSlot Auto/Standard: ""
+*InputSlot Manual/Manual Feed: "<</ManualFeed true>>setpagedevice"
+*% Localize InputSlot=Manual for French and German
+*fr_CA.InputSlot Manual/Manuel mecanisme de alimentation: ""
+*de.InputSlot Manual/Manueller Einzug: ""
+*CloseUI: *InputSlot
+
+ + +

CUPS 1.3/Mac OS X 10.6Custom Options

+ +

CUPS supports custom options using an extension of the CustomPageSize and ParamCustomPageSize syntax:

+ +
+*CustomFoo True: "command"
+*ParamCustomFoo Name1/Text 1: order type minimum maximum
+*ParamCustomFoo Name2/Text 2: order type minimum maximum
+...
+*ParamCustomFoo NameN/Text N: order type minimum maximum
+
+ +

When the base option is part of the JCLSetup section, the "command" string contains JCL commands with "\order" placeholders for each numbered parameter. The CUPS API handles any necessary value quoting for HP-PJL commands. For example, if the JCL command string is "@PJL SET PASSCODE=\1" and the first +option value is "1234" then CUPS will output the string "@PJL SET PASSCODE=1234".

+ +

For non-JCLSetup options, the "order" value is a number from 1 to N and specifies the order of values as they are placed on the stack before the command. For example, if the PostScript command string is "<</cupsReal1 2 1 roll>>setpagedevice" and the option value is "2.0" then CUPS will output the string "2.0 <</cupsReal1 2 1 roll>>setpagedevice".

+ +

The "type" is one of the following keywords:

+ +
    + +
  • curve - a real value from "minimum" to "maximum" representing a gamma correction curve using the function: f(x) = x value
  • + +
  • int - an integer value from "minimum" to "maximum"
  • + +
  • invcurve - a real value from "minimum" to "maximum" representing a gamma correction curve using the function: f(x) = x 1 / value
  • + +
  • passcode - a string of numbers value with a minimum of "minimum" numbers and a maximum of "maximum" numbers ("minimum" and "maximum" are numbers and passcode strings are not displayed in the user interface)
  • + +
  • password - a string value with a minimum of "minimum" characters and a maximum of "maximum" characters ("minimum" and "maximum" are numbers and password strings are not displayed in the user interface)
  • + +
  • points - a measurement value in points from "minimum" to "maximum"
  • + +
  • real - a real value from "minimum" to "maximum"
  • + +
  • string - a string value with a minimum of "minimum" characters and a maximum of "maximum" characters ("minimum" and "maximum" are numbers)
  • + +
+ +

Examples:

+ +
+*% Base JCL key code option
+*JCLOpenUI JCLPasscode/Key Code: PickOne
+*OrderDependency: 10 JCLSetup *JCLPasscode
+*DefaultJCLPasscode: None
+*JCLPasscode None/No Code: ""
+*JCLPasscode 1111: "@PJL SET PASSCODE = 1111<0A>"
+*JCLPasscode 2222: "@PJL SET PASSCODE = 2222<0A>"
+*JCLPasscode 3333: "@PJL SET PASSCODE = 3333<0A>"
+*JCLCloseUI: *JCLPasscode
+
+*% Custom JCL key code option
+*CustomJCLPasscode True: "@PJL SET PASSCODE = \1<0A>"
+*ParamCustomJCLPasscode Code/Key Code: 1 passcode 4 4
+
+
+*% Base PostScript watermark option
+*OpenUI WatermarkText/Watermark Text: PickOne
+*OrderDependency: 10 AnySetup *WatermarkText
+*DefaultWatermarkText: None
+*WatermarkText None: ""
+*WatermarkText Draft: "<</cupsString1(Draft)>>setpagedevice"
+*CloseUI: *WatermarkText
+
+*% Custom PostScript watermark option
+*CustomWatermarkText True: "<</cupsString1 3 -1 roll>>setpagedevice"
+*ParamCustomWatermarkText Text: 1 string 0 32
+
+
+*% Base PostScript gamma/density option
+*OpenUI GammaDensity/Gamma and Density: PickOne
+*OrderDependency: 10 AnySetup *GammaDensity
+*DefaultGammaDensity: Normal
+*GammaDensity Normal/Normal: "<</cupsReal1 1.0/cupsReal2 1.0>>setpagedevice"
+*GammaDensity Light/Lighter: "<</cupsReal1 0.9/cupsReal2 0.67>>setpagedevice"
+*GammaDensity Dark/Darker: "<</cupsReal1 1.1/cupsReal2 1.5>>setpagedevice"
+*CloseUI: *GammaDensity
+
+*% Custom PostScript gamma/density option
+*CustomGammaDensity True: "<</cupsReal1 3 -1 roll/cupsReal2 5 -1>>setpagedevice"
+*ParamCustomGammaDensity Gamma: 1 curve 0.1 10
+*ParamCustomGammaDensity Density: 2 real 0 2
+
+ + +

Writing PostScript Option Commands for Raster Drivers

+ +

PPD files are used for both PostScript and non-PostScript printers. For CUPS raster drivers, you use a subset of the PostScript language to set page device keywords such as page size, resolution, and so forth. For example, the following code sets the page size to A4 size:

+ +
+*PageSize A4: "<</PageSize[595 842]>>setpagedevice"
+
+ +

Custom options typically use other operators to organize the values into a key/value dictionary for setpagedevice. For example, our previous CustomWatermarkText option code uses the roll operator to move the custom string value into the dictionary for setpagedevice:

+ +
+*CustomWatermarkText True: "<</cupsString1 3 -1 roll>>setpagedevice"
+
+ +

For a custom string value of "My Watermark", CUPS will produce the following PostScript code for the option:

+ +
+(My Watermark)
+<</cupsString1 3 -1 roll>>setpagedevice
+
+ +

The code moves the string value ("My Watermark") from the bottom of the stack to the top, creating a dictionary that looks like:

+ +
+<</cupsString1(My Watermark)>>setpagedevice
+
+ +

The resulting dictionary sets the page device attributes that are sent to your raster driver in the page header.

+ +

Custom Page Size Code

+ +

There are many possible implementations of the CustomPageSize code. For CUPS raster drivers, the following code is recommended:

+ +
+*ParamCustomPageSize Width:        1 points min-width max-width
+*ParamCustomPageSize Height:       2 points min-height max-height
+*ParamCustomPageSize WidthOffset:  3 points 0 0
+*ParamCustomPageSize HeightOffset: 4 points 0 0
+*ParamCustomPageSize Orientation:  5 int 0 0
+*CustomPageSize True: "pop pop pop <</PageSize[5 -2 roll]/ImagingBBox null>>setpagedevice"
+
+ +

Supported PostScript Operators

+ +

CUPS supports the following PostScript operators in addition to the usual PostScript number, string (literal and hex-encoded), boolean, null, and name values:

+ +
    + +
  • << - Start a dictionary.
  • + +
  • >> - End a dictionary.
  • + +
  • [ - Start an array.
  • + +
  • ] - End an array.
  • + +
  • copy - Copy the top N objects on the stack.
  • + +
  • dup - Copy the top object on the stack.
  • + +
  • index - Copy the Nth from the top object on the stack.
  • + +
  • pop - Pop the top object on the stack.
  • + +
  • roll - Shift the top N objects on the stack.
  • + +
  • setpagedevice - Set the page header values according to the key/value dictionary on the stack.
  • + +
+ +
Note: + +

Never use the unsupported dict or put +operators in your option code. These operators are typically used in +option code dating back to Level 1 PostScript printers, which did not +support the simpler << or >> operators. +If you have old option code using dict or put, you can +rewrite it very easily to use the newer << and +>> operators instead. For example, the following code +to set the page size:

+ + + +
+1 dict dup /PageSize [612 792] put setpagedevice
+
+ +

can be rewritten as:

+ +
+<< /PageSize [612 792] >> setpagedevice
+
+ +
+ +

Supported Page Device Attributes

+ +

Table 2 shows the supported page device attributes along with PostScript code examples.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 2: Supported Page Device Attributes
Name(s)TypeDescriptionExample(s)
AdvanceDistanceIntegerSpecifies the number of points to advance roll media after printing.<</AdvanceDistance 18>>setpagedevice
AdvanceMediaIntegerSpecifies when to advance the media: 0 = never, 1 = after the file, 2 = after the job, 3 = after the set, and 4 = after the page.<</AdvanceMedia 4>>setpagedevice
CollateBooleanSpecifies whether collated copies are required.<</Collate true>>setpagedevice
CutMediaIntegerSpecifies when to cut the media: 0 = never, 1 = after the file, 2 = after the job, 3 = after the set, and 4 = after the page.<</CutMedia 1>>setpagedevice
DuplexBooleanSpecifies whether 2-sided printing is required.<</Duplex true>>setpagedevice
HWResolutionInteger ArraySpecifies the resolution of the page image in pixels per inch.<</HWResolution[1200 1200]>>setpagedevice
InsertSheetBooleanSpecifies whether to insert a blank sheet before the job.<</InsertSheet true>>setpagedevice
JogIntegerSpecifies when to shift the media in the output bin: 0 = never, 1 = after the file, 2 = after the job, 3 = after the set, and 4 = after the page.<</Jog 2>>setpagedevice
LeadingEdgeIntegerSpecifies the leading edge of the media: 0 = top, 1 = right, 2 = bottom, 3 = left.<</LeadingEdge 0>>setpagedevice
ManualFeedBooleanSpecifies whether media should be drawn from the manual feed tray. Note: The MediaPosition attribute is preferred over the ManualFeed attribute.<</ManualFeed true>>setpagedevice
MediaClassStringSpecifies a named media.<</MediaClass (Invoices)>>setpagedevice
MediaColorStringSpecifies the color of the media.<</MediaColor >>setpagedevice
MediaPositionIntegerSpecifies the tray or source of the media.<</MediaPosition 12>>setpagedevice
MediaTypeStringSpecifies the general media type.<</MediaType (Glossy)>>setpagedevice
MediaWeightIntegerSpecifies the media weight in grams per meter2.<</MediaWeight 100>>setpagedevice
MirrorPrintBooleanSpecifies whether to flip the output image horizontally.<</MirrorPrint true>>setpagedevice
NegativePrintBooleanSpecifies whether to invert the output image.<</NegativePrint true>>setpagedevice
NumCopiesIntegerSpecifies the number of copies to produce of each page.<</NumCopies 100>>setpagedevice
OrientationIntegerSpecifies the orientation of the output: 0 = portrait, 1 = landscape rotated counter-clockwise, 2 = upside-down, 3 = landscape rotated clockwise.<</Orientation 3>>setpagedevice
OutputFaceUpBooleanSpecifies whether to place the media face-up in the output bin/tray.<</OutputFaceUp true>>setpagedevice
OutputTypeStringSpecifies the output type name.<</OutputType (Photo)>>setpagedevice
PageSizeInteger/Real ArraySpecifies the width and length/height of the page in points.<</PageSize[595 842]>>setpagedevice
SeparationsBooleanSpecifies whether to produce color separations.<</Separations true>>setpagedevice
TraySwitchBooleanSpecifies whether to switch trays automatically.<</TraySwitch true>>setpagedevice
TumbleBooleanSpecifies whether the back sides of pages are rotated 180 degrees.<</Tumble true>>setpagedevice
cupsBorderlessScalingFactorRealSpecifies the amount to scale the page image dimensions.<</cupsBorderlessScalingFactor 1.01>>setpagedevice
cupsColorOrderIntegerSpecifies the order of colors: 0 = chunked, 1 = banded, 2 = planar.<</cupsColorOrder 0>>setpagedevice
cupsColorSpaceIntegerSpecifies the page image colorspace: 0 = W, 1 = RGB, 2 = RGBA, 3 = K, 4 = CMY, 5 = YMC, 6 = CMYK, 7 = YMCK, 8 = KCMY, 9 = KCMYcm, 10 = GMCK, 11 = GMCS, 12 = White, 13 = Gold, 14 = Silver, 15 = CIE XYZ, 16 = CIE Lab, 17 = RGBW, 32 to 46 = CIE Lab (1 to 15 inks)<</cupsColorSpace 1 >>setpagedevice
cupsCompressionIntegerSpecifies a driver compression type/mode.<</cupsCompression 2>>setpagedevice
cupsInteger0
+ ...
+ cupsInteger15
IntegerSpecifies driver integer values.<</cupsInteger11 1234>>setpagedevice
cupsMarkerTypeStringSpecifies the type of ink/toner to use.<</cupsMarkerType (Black+Color)>>setpagedevice
cupsMediaTypeIntegerSpecifies a numeric media type.<</cupsMediaType 999>>setpagedevice
cupsPageSizeNameStringSpecifies the name of the page size.<</cupsPageSizeName (A4.Full)>>setpagedevice
cupsPreferredBitsPerColorIntegerSpecifies the preferred number of bits per color, typically 8 or 16.<</cupsPreferredBitsPerColor 16>>setpagedevice
cupsReal0
+ ...
+ cupsReal15
RealSpecifies driver real number values.<</cupsReal15 1.234>>setpagedevice
cupsRenderingIntentStringSpecifies the color rendering intent.<</cupsRenderingIntent (AbsoluteColorimetric)>>setpagedevice
cupsRowCountIntegerSpecifies the number of rows of raster data to print on each line for some drivers.<</cupsRowCount 24>>setpagedevice
cupsRowFeedIntegerSpecifies the number of rows to feed between passes for some drivers.<</cupsRowFeed 17>>setpagedevice
cupsRowStepIntegerSpecifies the number of lines between columns/rows on the print head for some drivers.<</cupsRowStep 2>>setpagedevice
cupsString0
+ ...
+ cupsString15
StringSpecifies driver string values.<</cupsString0(String Value)>>setpagedevice
+ + +

Media Keywords

+ +

The CUPS media keywords allow drivers to specify alternate custom page +size limits based on up to two options.

+ +

CUPS 1.4/Mac OS X 10.6cupsMediaQualifier2

+ +

*cupsMediaQualifier2: MainKeyword

+ +

This keyword specifies the second option to use for overriding the +custom page size limits.

+ +

Example:

+ +
+*% Specify alternate custom page size limits based on InputSlot and Quality
+*cupsMediaQualifier2: InputSlot
+*cupsMediaQualifier3: Quality
+*cupsMaxSize .Manual.: "1000 1000"
+*cupsMinSize .Manual.: "100 100"
+*cupsMinSize .Manual.Photo: "200 200"
+*cupsMinSize ..Photo: "300 300"
+
+ +

CUPS 1.4/Mac OS X 10.6cupsMediaQualifier3

+ +

*cupsMediaQualifier3: MainKeyword

+ +

This keyword specifies the third option to use for overriding the +custom page size limits.

+ +

Example:

+ +
+*% Specify alternate custom page size limits based on InputSlot and Quality
+*cupsMediaQualifier2: InputSlot
+*cupsMediaQualifier3: Quality
+*cupsMaxSize .Manual.: "1000 1000"
+*cupsMinSize .Manual.: "100 100"
+*cupsMinSize .Manual.Photo: "200 200"
+*cupsMinSize ..Photo: "300 300"
+
+ +

CUPS 1.4/Mac OS X 10.6cupsMinSize

+ +

*cupsMinSize .Qualifier2.Qualifier3: "width length"
+*cupsMinSize .Qualifier2.: "width length"
+*cupsMinSize ..Qualifier3: "width length"

+ +

This keyword specifies alternate minimum custom page sizes in points. +The cupsMediaQualifier2 and +cupsMediaQualifier3 keywords +are used to identify options to use for matching.

+ +

Example:

+ +
+*% Specify alternate custom page size limits based on InputSlot and Quality
+*cupsMediaQualifier2: InputSlot
+*cupsMediaQualifier3: Quality
+*cupsMaxSize .Manual.: "1000 1000"
+*cupsMinSize .Manual.: "100 100"
+*cupsMinSize .Manual.Photo: "200 200"
+*cupsMinSize ..Photo: "300 300"
+
+ +

CUPS 1.4/Mac OS X 10.6cupsMaxSize

+ +

*cupsMaxSize .Qualifier2.Qualifier3: "width length"
+*cupsMaxSize .Qualifier2.: "width length"
+*cupsMaxSize ..Qualifier3: "width length"

+ +

This keyword specifies alternate maximum custom page sizes in points. +The cupsMediaQualifier2 and +cupsMediaQualifier3 keywords +are used to identify options to use for matching.

+ +

Example:

+ +
+*% Specify alternate custom page size limits based on InputSlot and Quality
+*cupsMediaQualifier2: InputSlot
+*cupsMediaQualifier3: Quality
+*cupsMaxSize .Manual.: "1000 1000"
+*cupsMinSize .Manual.: "100 100"
+*cupsMinSize .Manual.Photo: "200 200"
+*cupsMinSize ..Photo: "300 300"
+
+ + +

General Attributes

+ +

CUPS 1.3/Mac OS X 10.5cupsBackSide

+ +

*cupsBackSide: keyword

+ +

This keyword requests special handling of the back side of pages +when doing duplexed (2-sided) output. Table 1 +shows the supported keyword values for this keyword and their effect +on the raster data sent to your driver. For example, when cupsBackSide +is Rotated and Tumble is false, your driver +will receive print data starting at the bottom right corner of the page, with +each line going right-to-left instead of left-to-right. The default value is +Normal.

+ +
Note: + +

cupsBackSide replaces the older cupsFlipDuplex +keyword - if cupsBackSide is specified, cupsFlipDuplex +will be ignored.

+ +
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 1: Back Side Raster Coordinate System
cupsBackSideTumble ValueImage Presentation
NormalfalseLeft-to-right, top-to-bottom
NormaltrueLeft-to-right, top-to-bottom
ManualTumblefalseLeft-to-right, top-to-bottom
ManualTumbletrueRight-to-left, bottom-to-top
RotatedfalseRight-to-left, bottom-to-top
RotatedtrueRight-to-left, top-to-bottom
Flipped *falseLeft-to-right, bottom-to-top
Flipped *trueRight-to-left, top-to-bottom
+
+ +

* - Not supported in Mac OS X 10.5.x and earlier

+ +
+ + +
Figure 1: Back side images
Back side images
+ +

Examples:

+ +
+*% Flip the page image for the back side of duplexed output
+*cupsBackSide: Flipped
+
+*% Rotate the page image for the back side of duplexed output
+*cupsBackSide: Rotated
+
+ +

Also see the related APDuplexRequiresFlippedMargin +keyword.

+ +

CUPS 1.4/Mac OS X 10.6cupsCommands

+ +

*cupsCommands: "name name2 ... nameN"

+ +

This string keyword specifies the commands that are supported by the +CUPS command file filter for this device. The command names are separated +by whitespace.

+ +

Example:

+ +
+*% Specify the list of commands we support
+*cupsCommands: "AutoConfigure Clean PrintSelfTestPage ReportLevels com.vendor.foo"
+
+ + +

CUPS 1.3/Mac OS X 10.5cupsEvenDuplex

+ +

*cupsEvenDuplex: boolean

+ +

This boolean keyword notifies the RIP filters that the +destination printer requires an even number of pages when 2-sided +printing is selected. The default value is false.

+ +

Example:

+ +
+*% Always send an even number of pages when duplexing
+*cupsEvenDuplex: true
+
+ +

cupsFax

+ +

*cupsFax: boolean

+ +

This boolean keyword specifies whether the PPD defines a facsimile device. The default is false.

+ +

Examples:

+ +
+*cupsFax: true
+
+ +

cupsFilter

+ +

*cupsFilter: "source/type cost program"

+ +

This string keyword provides a conversion rule from the +given source type to the printer's native format using the +filter "program". If a printer supports the source type directly, +the special filter program "-" may be specified.

+ +

Examples:

+ +
+*% Standard raster printer driver filter
+*cupsFilter: "application/vnd.cups-raster 100 rastertofoo"
+
+*% Plain text filter
+*cupsFilter: "text/plain 10 texttofoo"
+
+*% Pass-through filter for PostScript printers
+*cupsFilter: "application/vnd.cups-postscript 0 -"
+
+ +

CUPS 1.5cupsFilter2

+ +

*cupsFilter2: "source/type destination/type cost program"

+ +

This string keyword provides a conversion rule from the given source type to the printer's native format using the filter "program". If a printer supports the source type directly, the special filter program "-" may be specified. The destination type is automatically created as needed and is passed to the filters and backend as the FINAL_CONTENT_TYPE value.

+ +
Note: + +

The presence of a single cupsFilter2 keyword in the PPD file will hide any cupsFilter keywords from the CUPS scheduler. When using cupsFilter2 to provide filters specific for CUPS 1.5 and later, provide a cupsFilter2 line for every filter and a cupsFilter line for each filter that is compatible with older versions of CUPS.

+ +
+ +

Examples:

+ +
+*% Standard raster printer driver filter
+*cupsFilter2: "application/vnd.cups-raster application/vnd.foo 100 rastertofoo"
+
+*% Plain text filter
+*cupsFilter2: "text/plain application/vnd.foo 10 texttofoo"
+
+*% Pass-through filter for PostScript printers
+*cupsFilter2: "application/vnd.cups-postscript application/postscript 0 -"
+
+ +

DeprecatedcupsFlipDuplex

+ +

*cupsFlipDuplex: boolean

+ +

Due to implementation differences between Mac OS X and Ghostscript, +the cupsFlipDuplex keyword is deprecated. Instead, use +the cupsBackSide keyword to specify +the coordinate system (pixel layout) of the page data on the back side of +duplex pages.

+ +

The value true maps to a cupsBackSide value +of Rotated on Mac OS X and Flipped with +Ghostscript.

+ +

The default value is false.

+ +
Note: + +

Mac OS X drivers that previously used +cupsFlipDuplex may wish to provide both the old and +new keywords for maximum compatibility, for example:

+ +
+*cupsBackSide: Rotated
+*cupsFlipDuplex: true
+
+ +

Similarly, drivers written for other operating systems using +Ghostscript can use:

+ +
+*cupsBackSide: Flipped
+*cupsFlipDuplex: true
+
+ +

CUPS 1.3/Mac OS X 10.5cupsIPPFinishings

+ +

*cupsIPPFinishings number/text: "*Option Choice ..."

+ +

This keyword defines a mapping from IPP finishings +values to PPD options and choices.

+ +

Examples:

+ +
+*cupsIPPFinishings 4/staple: "*StapleLocation SinglePortrait"
+*cupsIPPFinishings 5/punch: "*PunchMedia Yes *PunchLocation LeftSide"
+*cupsIPPFinishings 20/staple-top-left: "*StapleLocation SinglePortrait"
+*cupsIPPFinishings 21/staple-bottom-left: "*StapleLocation SingleLandscape"
+
+ +

CUPS 1.3/Mac OS X 10.5cupsIPPReason

+ +

*cupsIPPReason reason/Reason Text: "optional URIs"

+ +

This optional keyword maps custom +printer-state-reasons keywords that are generated by +the driver to human readable text. The optional URIs string +contains zero or more URIs separated by a newline. Each URI can +be a CUPS server absolute path to a help file under the +scheduler's DocumentRoot directory, a full HTTP URL +("http://www.domain.com/path/to/help/page.html"), or any other +valid URI which directs the user at additional information +concerning the condition that is being reported.

+ +

Since the reason text is limited to 80 characters by the PPD specification, longer text strings can be included by URI-encoding the text with the "text" scheme, for example "text:some%20text". Multiple text URIs are combined by the ppdLocalizeIPPReason into a single string that can be displayed to the user.

+ +

Examples:

+ +
+*% Map com.vendor-error to text but no page
+*cupsIPPReason com.vendor-error/A serious error occurred: ""
+
+*% Map com.vendor-error to more than 80 characters of text but no page
+*cupsIPPReason com.vendor-error/A serious error occurred: "text:Now%20is%20the%20time
+text:for%20all%20good%20men%20to%20come%20to%20the%20aid%20of%20their%20country."
+
+*% Map com.vendor-error to text and a local page
+*cupsIPPReason com.vendor-error/A serious error occurred: "/help/com.vendor/error.html"
+
+*% Map com.vendor-error to text and a remote page
+*cupsIPPReason com.vendor-error/A serious error occurred: "http://www.vendor.com/help"
+
+*% Map com.vendor-error to text and a local, Apple help book, and remote page
+*APHelpBook: "file:///Library/Printers/vendor/Help.bundle"
+*cupsIPPReason com.vendor-error/A serious error occurred: "/help/com.vendor/error.html
+help:anchor='com.vendor-error'%20bookID=Vendor%20Help
+http://www.vendor.com/help"
+*End
+
+ +

CUPS 1.5cupsIPPSupplies

+ +

*cupsIPPSupplies: boolean

+ +

This keyword tells the IPP backend whether it should report the current marker-xxx supply attribute values. The default value is True. + +

Example:

+ +
+*% Do not use IPP marker-xxx attributes to report supply levels
+*cupsIPPSupplies: False
+
+ +

CUPS 1.2/Mac OS X 10.5cupsLanguages

+ +

*cupsLanguages: "locale list"

+ +

This keyword describes which language localizations are +included in the PPD. The "locale list" string is a space-delimited +list of locale names ("en", "en_US", "fr_CA", etc.)

+ +

Example:

+ +
+*% Specify Canadian, UK, and US English, and Canadian and French French
+*cupsLanguages: "en_CA en_UK en_US fr_CA fr_FR"
+
+ +

cupsManualCopies

+ +

*cupsManualCopies: boolean

+ +

This boolean keyword notifies the RIP filters that the +destination printer does not support copy generation in +hardware. The default value is false.

+ +

Example:

+ +
+*% Tell the RIP filters to generate the copies for us
+*cupsManualCopies: true
+
+ +

CUPS 1.4/Mac OS X 10.6cupsMarkerName

+ +

*cupsMarkerName/Name Text: ""

+ +

This optional keyword maps marker-names strings that are +generated by the driver to human readable text.

+ +

Examples:

+ +
+*% Map cyanToner to "Cyan Toner"
+*cupsMarkerName cyanToner/Cyan Toner: ""
+
+ +

CUPS 1.4/Mac OS X 10.6cupsMarkerNotice

+ +

*cupsMarkerNotice: "disclaimer text"

+ +

This optional keyword provides disclaimer text for the supply level +information provided by the driver, typically something like "supply levels +are approximate".

+ +

Examples:

+ +
+*cupsMarkerNotice: "Supply levels are approximate."
+
+ +

cupsModelNumber

+ +

*cupsModelNumber: number

+ +

This integer keyword specifies a printer-specific model +number. This number can be used by a filter program to adjust +the output for a specific model of printer.

+ +

Example:

+ +
+*% Specify an integer for a driver-specific model number
+*cupsModelNumber: 1234
+
+ +

CUPS 1.3/Mac OS X 10.5cupsPJLCharset

+ +

*cupsPJLCharset: "ISO character set name"

+ +

This string keyword specifies the character set that is used +for strings in PJL commands. If not specified, US-ASCII is +assumed.

+ +

Example:

+ +
+*% Specify UTF-8 is used in PJL strings
+*cupsPJLCharset: "UTF-8"
+
+ +

CUPS 1.4/Mac OS X 10.6cupsPJLDisplay

+ +

*cupsPJLDisplay: "what"

+ +

This optional keyword specifies which command is used to display the +job ID, name, and user on the printer's control panel. "What" is either "none" +to disable this functionality, "job" to use "@PJL JOB DISPLAY", or "rdymsg" +to use "@PJL RDYMSG DISPLAY". The default is "job".

+ +

Examples:

+ +
+*% Display job information using @PJL SET RDYMSG DISPLAY="foo"
+*cupsPJLDisplay: "rdymsg"
+
+*% Display job information display
+*cupsPJLDisplay: "none"
+
+ +

CUPS 1.2/Mac OS X 10.5cupsPortMonitor

+ +

*cupsPortMonitor urischeme/Descriptive Text: "port monitor"

+ +

This string keyword specifies printer-specific "port +monitor" filters that may be used with the printer. The CUPS +scheduler also looks for the Protocols keyword to see +if the BCP or TBCP protocols are supported. If +so, the corresponding port monitor ("bcp" and "tbcp", +respectively) is listed in the printer's +port-monitor-supported keyword.

+ +

The "urischeme" portion of the keyword specifies the URI scheme +that this port monitor should be used for. Typically this is used to +pre-select a particular port monitor for each type of connection that +is supported by the printer. The "port monitor" string can be "none" +to disable the port monitor for the given URI scheme.

+ +

Examples:

+ +
+*% Specify a PostScript printer that supports the TBCP protocol
+*Protocols: TBCP PJL
+
+*% Specify that TBCP should be used for socket connections but not USB
+*cupsPortMonitor socket/AppSocket Printing: "tbcp"
+*cupsPortMonitor usb/USB Printing: "none"
+
+*% Specify a printer-specific port monitor for an Epson USB printer
+*cupsPortMonitor usb/USB Status Monitor: "epson-usb"
+
+ +

CUPS 1.3/Mac OS X 10.5cupsPreFilter

+ +

*cupsPreFilter: "source/type cost program"

+ +

This string keyword provides a pre-filter rule. The pre-filter +program will be inserted in the conversion chain immediately +before the filter that accepts the given MIME type.

+ +

Examples:

+ +
+*% PDF pre-filter
+*cupsPreFilter: "application/pdf 100 mypdfprefilter"
+
+*% PNG pre-filter
+*cupsPreFilter: "image/png 0 mypngprefilter"
+
+ + +

CUPS 1.5cupsPrintQuality

+ +

*cupsPrintQuality keyword/text: "code"

+ +

This UI keyword defines standard print qualities that directly map from the IPP "print-quality" job template keyword. Standard keyword values are "Draft", "Normal", and "High" which are mapped from the IPP "print-quality" values 3, 4, and 5 respectively. Each cupsPrintQuality option typically sets output mode and resolution parameters in the page device dictionary, eliminating the need for separate (and sometimes confusing) output mode and resolution options.

+ +
Note: + +

Unlike all of the other keywords defined in this document, cupsPrintQuality is a UI keyword that MUST be enclosed inside the PPD OpenUI and CloseUI keywords.

+ +
+ +

Examples:

+ +
+*OpenUI *cupsPrintQuality/Print Quality: PickOne
+*OrderDependency: 10 AnySetup *cupsPrintQuality
+*DefaultcupsPrintQuality: Normal
+*cupsPrintQuality Draft/Draft: "code"
+*cupsPrintQuality Normal/Normal: "code"
+*cupsPrintQuality High/Photo: "code"
+*CloseUI: *cupsPrintQuality
+
+ +

CUPS 1.5cupsSingleFile

+ +

*cupsSingleFile: Boolean

+ +

This boolean keyword tells the scheduler whether to print multiple files in a job together or singly. The default is "False" which uses a single instance of the backend for all files in the print job. Setting this keyword to "True" will result in separate instances of the backend for each file in the print job.

+ +

Examples:

+ +
+*% Send all print data to a single backend
+*cupsSingleFile: False
+
+*% Send each file using a separate backend
+*cupsSingleFile: True
+
+ +

CUPS 1.4/Mac OS X 10.6cupsSNMPSupplies

+ +

*cupsSNMPSupplies: boolean

+ +

This keyword tells the standard network backends whether they should query +the standard SNMP Printer MIB OIDs for supply levels. The default value is +True. + +

Example:

+ +
+*% Do not use SNMP queries to report supply levels
+*cupsSNMPSupplies: False
+
+ +

cupsVersion

+ +

*cupsVersion: major.minor

+ +

This required keyword describes which version of the CUPS +PPD file extensions was used. Currently it must be the string +"1.0", "1.1", "1.2", or "1.3".

+ +

Example:

+ +
+*% Specify a CUPS 1.2 driver
+*cupsVersion: "1.2"
+
+ + +

Mac OS X Attributes

+ +

Mac OS X 10.3APDialogExtension

+ +

*APDialogExtension: "/Library/Printers/vendor/filename.plugin"

+ +

This keyword defines additional option panes that are displayed in the +print dialog. Each keyword adds one or more option panes. See the "OutputBinsPDE" +example and Apple +Technical Q&A QA1352 for information on writing your own print dialog +plug-ins.

+ +
Note: + +

Starting with Mac OS X 10.5, each plug-in must be compiled "4-way fat" +(32-bit and 64-bit for both PowerPC and Intel) with garbage collection enabled +in order to be usable with all applications.

+ +
+ +

Examples:

+ +
+*% Add two panes for finishing and driver options
+*APDialogExtension: "/Library/Printers/vendor/finishing.plugin"
+*APDialogExtension: "/Library/Printers/vendor/options.plugin"
+
+ +

Mac OS X 10.4APDuplexRequiresFlippedMargin

+ +

*APDuplexRequiresFlippedMargin: boolean

+ +

This boolean keyword notifies the RIP filters that the +destination printer requires the top and bottom margins of the +ImageableArea to be swapped for the back page. The +default is true when cupsBackSide is Flipped +and false otherwise. Table 2 shows how +APDuplexRequiresFlippedMargin interacts with cupsBackSide +and the Tumble page attribute.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Table 2: Margin Flipping Modes
APDuplexRequiresFlippedMargincupsBackSideTumble ValueMargins
falseanyanyNormal
anyNormalanyNormal
trueManualDuplexfalseNormal
trueManualDuplextrueFlipped
trueRotatedfalseFlipped
trueRotatedtrueNormal
true or unspecifiedFlippedanyFlipped
+ +

Example:

+ +
+*% Rotate the back side images
+*cupsBackSide: Rotated
+
+*% Don't swap the top and bottom margins for the back side
+*APDuplexRequiresFlippedMargin: false
+
+ +

Also see the related cupsBackSide +keyword.

+ +

APHelpBook

+ +

*APHelpBook: "bundle URL"

+ +

This string keyword specifies the Apple help book bundle to use when +looking up IPP reason codes for this printer driver. The +cupsIPPReason keyword maps +"help" URIs to this file.

+ +

Example:

+ +
+*APHelpBook: "file:///Library/Printers/vendor/Help.bundle"
+
+ +

Mac OS X 10.6APICADriver

+ +

*APICADriver: boolean

+ +

This keyword specifies whether the device has a matching Image Capture +Architecture (ICA) driver for scanning. The default is False.

+ +

Examples:

+ +
+*APICADriver: True
+*APScanAppBundleID: "com.apple.ImageCaptureApp"
+
+ +

Mac OS X 10.3APPrinterIconPath

+ +

*APPrinterIconPath: "/Library/Printers/vendor/filename.icns"

+ +

This keyword defines the location of a printer icon file to use when +displaying the printer. The file must be in the Apple icon format.

+ +

Examples:

+ +
+*% Apple icon file
+*APPrinterIconPath: "/Library/Printers/vendor/Icons/filename.icns"
+
+ +

Mac OS X 10.4APPrinterLowInkTool

+ +

*APPrinterLowInkTool: "/Library/Printers/vendor/program"

+ +

This keyword defines an program that checks the ink/toner/marker levels +on a printer, returning an XML document with those levels. See the "InkTool" +example and +Apple +Technical Note TN2144 for more information.

+ +

Examples:

+ +
+*% Use a vendor monitoring program
+*APPrinterLowInkTool: "/Library/Printers/vendor/Tools/lowinktool"
+
+ +

Mac OS X 10.5APPrinterPreset

+ +

*APPrinterPreset name/text: "*Option Choice ..."

+ +

This keyword defines presets for multiple options that show up +in the print dialog of applications (such as iPhoto) that set the job +style hint to NSPrintPhotoJobStyleHint. Each preset maps to one or +more pairs of PPD options and choices as well as providing key/value data for +the application. The following standard preset names are currently defined:

+ +
    + +
  • General_with_Paper_Auto-Detect; Normal quality general printing with auto-detected media.
  • + +
  • General_with_Paper_Auto-Detect_-_Draft; Draft quality general printing with auto-detected media.
  • + +
  • General_on_Plain_Paper; Normal quality general printing on plain paper.
  • + +
  • General_on_Plain_Paper_-_Draft; Draft quality general printing on plain paper.
  • + +
  • Photo_with_Paper_Auto-Detect; Normal quality photo printing with auto-detected media.
  • + +
  • Photo_with_Paper_Auto-Detect_-_Fine; High quality photo printing with auto-detected media.
  • + +
  • Photo_on_Plain_Paper; Normal quality photo printing on plain paper.
  • + +
  • Photo_on_Plain_Paper_-_Fine; High quality photo printing on plain paper.
  • + +
  • Photo_on_Photo_Paper; Normal quality photo printing on glossy photo paper.
  • + +
  • Photo_on_Photo_Paper_-_Fine; High quality photo printing on glossy photo paper.
  • + +
  • Photo_on_Matte_Paper; Normal quality photo printing on matte paper.
  • + +
  • Photo_on_Matte_Paper_-_Fine; High quality photo printing on matte paper.
  • + +
+ +

The value string consists of pairs of keywords, either an option name and +choice (*MainKeyword OptionKeyword) or a preset identifier and value +(com.apple.print.preset.foo value). The following preset identifiers are currently used:

+ +
    + +
  • com.apple.print.preset.graphicsType; specifies the type of printing used for this printing - "General" for general purpose printing and "Photo" for photo printing.
  • + +
  • com.apple.print.preset.media-front-coating; specifies the media type selected by this preset - "none" (plain paper), "glossy", "high-gloss", "semi-gloss", "satin", "matte", and "autodetect".
  • + +
  • com.apple.print.preset.output-mode; specifies the output mode for this preset - "color" (default for color printers) or "monochrome" (grayscale, default for B&W printers).
  • + +
  • com.apple.print.preset.quality; specifies the overall print quality selected by this preset - "low" (draft), "mid" (normal), or "high".
  • + +
+ +

Presets, like options, can also be localized in multiple languages.

+ +

Examples:

+ +
+*APPrinterPreset Photo_on_Photo_Paper/Photo on Photo Paper: "
+  *MediaType Glossy
+  *ColorModel RGB
+  *Resolution 300dpi
+  com.apple.print.preset.graphicsType Photo
+  com.apple.print.preset.quality mid
+  com.apple.print.preset.media-front-coating glossy"
+*End
+*fr.APPrinterPreset Photo_on_Photo_Paper/Photo sur papier photographique: ""
+
+ +

Mac OS X 10.3APPrinterUtilityPath

+ +

*APPrinterPrinterUtilityPath: "/Library/Printers/vendor/filename.app"

+ +

This keyword defines a GUI application that can be used to do printer +maintenance functions such as cleaning the print head(s). See ... for more +information.

+ +

Examples:

+ +
+*% Define the printer utility application
+*APPrinterPrinterUtilityPath: "/Library/Printers/vendor/Tools/utility.app"
+
+ +

Mac OS X 10.6APScannerOnly

+ +

*APScannerOnly: boolean

+ +

This keyword specifies whether the device has scanning but no printing +capabilities. The default is False.

+ +

Examples:

+ +
+*APICADriver: True
+*APScannerOnly: True
+
+ +

Mac OS X 10.3APScanAppBundleID

+ +

*APScanAppBundleID: "bundle ID"

+ +

This keyword defines the application to use when scanning pages from +the device.

+ +

Examples:

+ +
+*APICADriver: True
+*APScanAppBundleID: "com.apple.ImageCaptureApp"
+
+ + +

Change History

+ +

Changes in CUPS 1.5

+ +
    + +
  • Changes all instances of PPD attributes to PPD keywords, to be consistent with the parent specification from Adobe.
  • + +
+ + +

Changes in CUPS 1.4.5

+ + + + +

Changes in CUPS 1.4

+ + + + +

Changes in CUPS 1.3.1

+ +
    + +
  • Added missing Mac OS X AP keywords.
  • + +
  • Added section on auto-configuration including the + OIDMainKeyword and ?MainKeyword + keywords.
  • + +
  • Minor reorganization.
  • + +
+ + +

Changes in CUPS 1.3

+ + + +

Changes in CUPS 1.2.8

+ +
    + +
  • Added section on supported PostScript commands for raster + drivers
  • + +
+ +

Changes in CUPS 1.2

+ + + +

Changes in CUPS 1.1

+ + diff --git a/filter/testraster.c b/filter/testraster.c new file mode 100644 index 0000000000..8920e2e770 --- /dev/null +++ b/filter/testraster.c @@ -0,0 +1,1078 @@ +/* + * "$Id$" + * + * Raster test program routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Test the raster functions. + * do_ppd_tests() - Test the default option commands in a PPD file. + * do_ps_tests() - Test standard PostScript commands. + * do_ras_file() - Test reading of a raster file. + * do_raster_tests() - Test reading and writing of raster data. + * print_changes() - Print differences in the page header. + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * Test PS commands and header... + */ + +static const char *dsc_code = +"[{\n" +"%%BeginFeature: *PageSize Tabloid\n" +"<>setpagedevice\n" +"%%EndFeature\n" +"} stopped cleartomark\n"; +static const char *setpagedevice_code = +"<<" +"/MediaClass(Media Class)" +"/MediaColor((Media Color))" +"/MediaType(Media\\\\Type)" +"/OutputType<416263>" +"/AdvanceDistance 1000" +"/AdvanceMedia 1" +"/Collate false" +"/CutMedia 2" +"/Duplex true" +"/HWResolution[100 200]" +"/InsertSheet true" +"/Jog 3" +"/LeadingEdge 1" +"/ManualFeed true" +"/MediaPosition 8#777" +"/MediaWeight 16#fe01" +"/MirrorPrint true" +"/NegativePrint true" +"/NumCopies 1" +"/Orientation 1" +"/OutputFaceUp true" +"/PageSize[612 792.1]" +"/Separations true" +"/TraySwitch true" +"/Tumble true" +"/cupsMediaType 2" +"/cupsColorOrder 1" +"/cupsColorSpace 1" +"/cupsCompression 1" +"/cupsRowCount 1" +"/cupsRowFeed 1" +"/cupsRowStep 1" +"/cupsBorderlessScalingFactor 1.001" +"/cupsInteger0 1" +"/cupsInteger1 2" +"/cupsInteger2 3" +"/cupsInteger3 4" +"/cupsInteger4 5" +"/cupsInteger5 6" +"/cupsInteger6 7" +"/cupsInteger7 8" +"/cupsInteger8 9" +"/cupsInteger9 10" +"/cupsInteger10 11" +"/cupsInteger11 12" +"/cupsInteger12 13" +"/cupsInteger13 14" +"/cupsInteger14 15" +"/cupsInteger15 16" +"/cupsReal0 1.1" +"/cupsReal1 2.1" +"/cupsReal2 3.1" +"/cupsReal3 4.1" +"/cupsReal4 5.1" +"/cupsReal5 6.1" +"/cupsReal6 7.1" +"/cupsReal7 8.1" +"/cupsReal8 9.1" +"/cupsReal9 10.1" +"/cupsReal10 11.1" +"/cupsReal11 12.1" +"/cupsReal12 13.1" +"/cupsReal13 14.1" +"/cupsReal14 15.1" +"/cupsReal15 16.1" +"/cupsString0(1)" +"/cupsString1(2)" +"/cupsString2(3)" +"/cupsString3(4)" +"/cupsString4(5)" +"/cupsString5(6)" +"/cupsString6(7)" +"/cupsString7(8)" +"/cupsString8(9)" +"/cupsString9(10)" +"/cupsString10(11)" +"/cupsString11(12)" +"/cupsString12(13)" +"/cupsString13(14)" +"/cupsString14(15)" +"/cupsString15(16)" +"/cupsMarkerType(Marker Type)" +"/cupsRenderingIntent(Rendering Intent)" +"/cupsPageSizeName(Letter)" +"/cupsPreferredBitsPerColor 17" +">> setpagedevice"; + +static cups_page_header2_t setpagedevice_header = +{ + "Media Class", /* MediaClass */ + "(Media Color)", /* MediaColor */ + "Media\\Type", /* MediaType */ + "Abc", /* OutputType */ + 1000, /* AdvanceDistance */ + CUPS_ADVANCE_FILE, /* AdvanceMedia */ + CUPS_FALSE, /* Collate */ + CUPS_CUT_JOB, /* CutMedia */ + CUPS_TRUE, /* Duplex */ + { 100, 200 }, /* HWResolution */ + { 0, 0, 0, 0 }, /* ImagingBoundingBox */ + CUPS_TRUE, /* InsertSheet */ + CUPS_JOG_SET, /* Jog */ + CUPS_EDGE_RIGHT, /* LeadingEdge */ + { 0, 0 }, /* Margins */ + CUPS_TRUE, /* ManualFeed */ + 0777, /* MediaPosition */ + 0xfe01, /* MediaWeight */ + CUPS_TRUE, /* MirrorPrint */ + CUPS_TRUE, /* NegativePrint */ + 1, /* NumCopies */ + CUPS_ORIENT_90, /* Orientation */ + CUPS_TRUE, /* OutputFaceUp */ + { 612, 792 }, /* PageSize */ + CUPS_TRUE, /* Separations */ + CUPS_TRUE, /* TraySwitch */ + CUPS_TRUE, /* Tumble */ + 0, /* cupsWidth */ + 0, /* cupsHeight */ + 2, /* cupsMediaType */ + 0, /* cupsBitsPerColor */ + 0, /* cupsBitsPerPixel */ + 0, /* cupsBytesPerLine */ + CUPS_ORDER_BANDED, /* cupsColorOrder */ + CUPS_CSPACE_RGB, /* cupsColorSpace */ + 1, /* cupsCompression */ + 1, /* cupsRowCount */ + 1, /* cupsRowFeed */ + 1, /* cupsRowStep */ + 0, /* cupsNumColors */ + 1.001, /* cupsBorderlessScalingFactor */ + { 612.0, 792.1 }, /* cupsPageSize */ + { 0.0, 0.0, 0.0, 0.0 }, /* cupsImagingBBox */ + { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16 }, + /* cupsInteger[16] */ + { 1.1, 2.1, 3.1, 4.1, 5.1, 6.1, 7.1, 8.1, 9.1, 10.1, 11.1, 12.1, 13.1, + 14.1, 15.1, 16.1 }, /* cupsReal[16] */ + { "1", "2", "3", "4", "5", "6", "7", "8", "9", "10", "11", "12", "13", + "14", "15", "16" }, /* cupsString[16] */ + "Marker Type", /* cupsMarkerType */ + "Rendering Intent", /* cupsRenderingIntent */ + "Letter" /* cupsPageSizeName */ +}; + + +/* + * Local functions... + */ + +static int do_ppd_tests(const char *filename, int num_options, + cups_option_t *options); +static int do_ps_tests(void); +static int do_ras_file(const char *filename); +static int do_raster_tests(cups_mode_t mode); +static void print_changes(cups_page_header2_t *header, + cups_page_header2_t *expected); + + +/* + * 'main()' - Test the raster functions. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int errors; /* Number of errors */ + const char *ext; /* Filename extension */ + + + if (argc == 1) + { + errors = do_ps_tests(); + errors += do_raster_tests(CUPS_RASTER_WRITE); + errors += do_raster_tests(CUPS_RASTER_WRITE_COMPRESSED); + errors += do_raster_tests(CUPS_RASTER_WRITE_PWG); + } + else + { + int i; /* Looping var */ + int num_options; /* Number of options */ + cups_option_t *options; /* Options */ + + + for (errors = 0, num_options = 0, options = NULL, i = 1; i < argc; i ++) + { + if (argv[i][0] == '-') + { + if (argv[i][1] == 'o') + { + if (argv[i][2]) + num_options = cupsParseOptions(argv[i] + 2, num_options, &options); + else + { + i ++; + if (i < argc) + num_options = cupsParseOptions(argv[i], num_options, &options); + else + { + puts("Usage: testraster [-o name=value ...] [filename.ppd ...]"); + puts(" testraster [filename.ras ...]"); + return (1); + } + } + } + else + { + puts("Usage: testraster [-o name=value ...] [filename.ppd ...]"); + puts(" testraster [filename.ras ...]"); + return (1); + } + } + else if ((ext = strrchr(argv[i], '.')) != NULL) + { + if (!strcmp(ext, ".ppd")) + errors += do_ppd_tests(argv[i], num_options, options); + else + errors += do_ras_file(argv[i]); + } + else + { + puts("Usage: testraster [-o name=value ...] [filename.ppd ...]"); + puts(" testraster [filename.ras ...]"); + return (1); + } + } + + cupsFreeOptions(num_options, options); + } + + return (errors); +} + + +/* + * 'do_ppd_tests()' - Test the default option commands in a PPD file. + */ + +static int /* O - Number of errors */ +do_ppd_tests(const char *filename, /* I - PPD file */ + int num_options, /* I - Number of options */ + cups_option_t *options) /* I - Options */ +{ + ppd_file_t *ppd; /* PPD file data */ + cups_page_header2_t header; /* Page header */ + + + printf("\"%s\": ", filename); + fflush(stdout); + + if ((ppd = ppdOpenFile(filename)) == NULL) + { + ppd_status_t status; /* Status from PPD loader */ + int line; /* Line number containing error */ + + + status = ppdLastError(&line); + + puts("FAIL (bad PPD file)"); + printf(" %s on line %d\n", ppdErrorString(status), line); + + return (1); + } + + ppdMarkDefaults(ppd); + cupsMarkOptions(ppd, num_options, options); + + if (cupsRasterInterpretPPD(&header, ppd, 0, NULL, NULL)) + { + puts("FAIL (error from function)"); + puts(cupsRasterErrorString()); + + return (1); + } + else + { + puts("PASS"); + + return (0); + } +} + + +/* + * 'do_ps_tests()' - Test standard PostScript commands. + */ + +static int +do_ps_tests(void) +{ + cups_page_header2_t header; /* Page header */ + int preferred_bits; /* Preferred bits */ + int errors = 0; /* Number of errors */ + + + /* + * Test PS exec code... + */ + + fputs("_cupsRasterExecPS(\"setpagedevice\"): ", stdout); + fflush(stdout); + + memset(&header, 0, sizeof(header)); + header.Collate = CUPS_TRUE; + preferred_bits = 0; + + if (_cupsRasterExecPS(&header, &preferred_bits, setpagedevice_code)) + { + puts("FAIL (error from function)"); + puts(cupsRasterErrorString()); + errors ++; + } + else if (preferred_bits != 17 || + memcmp(&header, &setpagedevice_header, sizeof(header))) + { + puts("FAIL (bad header)"); + + if (preferred_bits != 17) + printf(" cupsPreferredBitsPerColor %d, expected 17\n", + preferred_bits); + + print_changes(&setpagedevice_header, &header); + errors ++; + } + else + puts("PASS"); + + fputs("_cupsRasterExecPS(\"roll\"): ", stdout); + fflush(stdout); + + if (_cupsRasterExecPS(&header, &preferred_bits, + "792 612 0 0 0\n" + "pop pop pop\n" + "<>" + "setpagedevice\n")) + { + puts("FAIL (error from function)"); + puts(cupsRasterErrorString()); + errors ++; + } + else if (header.PageSize[0] != 792 || header.PageSize[1] != 612) + { + printf("FAIL (PageSize [%d %d], expected [792 612])\n", header.PageSize[0], + header.PageSize[1]); + errors ++; + } + else + puts("PASS"); + + fputs("_cupsRasterExecPS(\"dup index\"): ", stdout); + fflush(stdout); + + if (_cupsRasterExecPS(&header, &preferred_bits, + "true false dup\n" + "<>setpagedevice\n" + "pop pop pop")) + { + puts("FAIL (error from function)"); + puts(cupsRasterErrorString()); + errors ++; + } + else + { + if (!header.Collate) + { + printf("FAIL (Collate false, expected true)\n"); + errors ++; + } + + if (header.Duplex) + { + printf("FAIL (Duplex true, expected false)\n"); + errors ++; + } + + if (header.Tumble) + { + printf("FAIL (Tumble true, expected false)\n"); + errors ++; + } + + if(header.Collate && !header.Duplex && !header.Tumble) + puts("PASS"); + } + + fputs("_cupsRasterExecPS(\"%%Begin/EndFeature code\"): ", stdout); + fflush(stdout); + + if (_cupsRasterExecPS(&header, &preferred_bits, dsc_code)) + { + puts("FAIL (error from function)"); + puts(cupsRasterErrorString()); + errors ++; + } + else if (header.PageSize[0] != 792 || header.PageSize[1] != 1224) + { + printf("FAIL (bad PageSize [%d %d], expected [792 1224])\n", + header.PageSize[0], header.PageSize[1]); + errors ++; + } + else + puts("PASS"); + + return (errors); +} + + +/* + * 'do_ras_file()' - Test reading of a raster file. + */ + +static int /* O - Number of errors */ +do_ras_file(const char *filename) /* I - Filename */ +{ + unsigned y; /* Looping vars */ + int fd; /* File descriptor */ + cups_raster_t *ras; /* Raster stream */ + cups_page_header2_t header; /* Page header */ + unsigned char *data; /* Raster data */ + int errors = 0; /* Number of errors */ + unsigned pages = 0; /* Number of pages */ + + + if ((fd = open(filename, O_RDONLY)) < 0) + { + printf("%s: %s\n", filename, strerror(errno)); + return (1); + } + + if ((ras = cupsRasterOpen(fd, CUPS_RASTER_READ)) == NULL) + { + printf("%s: cupsRasterOpen failed.\n", filename); + close(fd); + return (1); + } + + printf("%s:\n", filename); + + while (cupsRasterReadHeader2(ras, &header)) + { + pages ++; + data = malloc(header.cupsBytesPerLine); + + printf(" Page %u: %ux%ux%u@%ux%udpi", pages, + header.cupsWidth, header.cupsHeight, header.cupsBitsPerPixel, + header.HWResolution[0], header.HWResolution[1]); + fflush(stdout); + + for (y = 0; y < header.cupsHeight; y ++) + if (cupsRasterReadPixels(ras, data, header.cupsBytesPerLine) < + header.cupsBytesPerLine) + break; + + if (y < header.cupsHeight) + printf(" ERROR AT LINE %d\n", y); + else + putchar('\n'); + + free(data); + } + + cupsRasterClose(ras); + close(fd); + + return (errors); +} + + +/* + * 'do_raster_tests()' - Test reading and writing of raster data. + */ + +static int /* O - Number of errors */ +do_raster_tests(cups_mode_t mode) /* O - Write mode */ +{ + int page, x, y; /* Looping vars */ + FILE *fp; /* Raster file */ + cups_raster_t *r; /* Raster stream */ + cups_page_header2_t header, /* Page header */ + expected; /* Expected page header */ + unsigned char data[2048]; /* Raster data */ + int errors = 0; /* Number of errors */ + + + /* + * Test writing... + */ + + printf("cupsRasterOpen(%s): ", + mode == CUPS_RASTER_WRITE ? "CUPS_RASTER_WRITE" : + mode == CUPS_RASTER_WRITE ? "CUPS_RASTER_WRITE_COMPRESSED" : + "CUPS_RASTER_WRITE_PWG"); + fflush(stdout); + + if ((fp = fopen("test.raster", "wb")) == NULL) + { + printf("FAIL (%s)\n", strerror(errno)); + return (1); + } + + if ((r = cupsRasterOpen(fileno(fp), mode)) == NULL) + { + printf("FAIL (%s)\n", strerror(errno)); + fclose(fp); + return (1); + } + + puts("PASS"); + + for (page = 0; page < 4; page ++) + { + memset(&header, 0, sizeof(header)); + header.cupsWidth = 256; + header.cupsHeight = 256; + header.cupsBytesPerLine = 256; + + if (page & 1) + { + header.cupsBytesPerLine *= 2; + header.cupsColorSpace = CUPS_CSPACE_CMYK; + header.cupsColorOrder = CUPS_ORDER_CHUNKED; + header.cupsNumColors = 4; + } + else + { + header.cupsColorSpace = CUPS_CSPACE_K; + header.cupsColorOrder = CUPS_ORDER_BANDED; + header.cupsNumColors = 1; + } + + if (page & 2) + { + header.cupsBytesPerLine *= 2; + header.cupsBitsPerColor = 16; + header.cupsBitsPerPixel = (page & 1) ? 64 : 16; + } + else + { + header.cupsBitsPerColor = 8; + header.cupsBitsPerPixel = (page & 1) ? 32 : 8; + } + + if (cupsRasterWriteHeader2(r, &header)) + puts("cupsRasterWriteHeader2: PASS"); + else + { + puts("cupsRasterWriteHeader2: FAIL"); + errors ++; + } + + fputs("cupsRasterWritePixels: ", stdout); + fflush(stdout); + + memset(data, 0, header.cupsBytesPerLine); + for (y = 0; y < 64; y ++) + if (!cupsRasterWritePixels(r, data, header.cupsBytesPerLine)) + break; + + if (y < 64) + { + puts("FAIL"); + errors ++; + } + else + { + for (x = 0; x < header.cupsBytesPerLine; x ++) + data[x] = x; + + for (y = 0; y < 64; y ++) + if (!cupsRasterWritePixels(r, data, header.cupsBytesPerLine)) + break; + + if (y < 64) + { + puts("FAIL"); + errors ++; + } + else + { + memset(data, 255, header.cupsBytesPerLine); + for (y = 0; y < 64; y ++) + if (!cupsRasterWritePixels(r, data, header.cupsBytesPerLine)) + break; + + if (y < 64) + { + puts("FAIL"); + errors ++; + } + else + { + for (x = 0; x < header.cupsBytesPerLine; x ++) + data[x] = x / 4; + + for (y = 0; y < 64; y ++) + if (!cupsRasterWritePixels(r, data, header.cupsBytesPerLine)) + break; + + if (y < 64) + { + puts("FAIL"); + errors ++; + } + else + puts("PASS"); + } + } + } + } + + cupsRasterClose(r); + fclose(fp); + + /* + * Test reading... + */ + + fputs("cupsRasterOpen(CUPS_RASTER_READ): ", stdout); + fflush(stdout); + + if ((fp = fopen("test.raster", "rb")) == NULL) + { + printf("FAIL (%s)\n", strerror(errno)); + return (1); + } + + if ((r = cupsRasterOpen(fileno(fp), CUPS_RASTER_READ)) == NULL) + { + printf("FAIL (%s)\n", strerror(errno)); + fclose(fp); + return (1); + } + + puts("PASS"); + + for (page = 0; page < 4; page ++) + { + memset(&expected, 0, sizeof(expected)); + expected.cupsWidth = 256; + expected.cupsHeight = 256; + expected.cupsBytesPerLine = 256; + + if (mode == CUPS_RASTER_WRITE_PWG) + { + strlcpy(expected.MediaClass, "PwgRaster", sizeof(expected.MediaClass)); + expected.cupsInteger[7] = 0xffffff; + } + + if (page & 1) + { + expected.cupsBytesPerLine *= 2; + expected.cupsColorSpace = CUPS_CSPACE_CMYK; + expected.cupsColorOrder = CUPS_ORDER_CHUNKED; + expected.cupsNumColors = 4; + } + else + { + expected.cupsColorSpace = CUPS_CSPACE_K; + expected.cupsColorOrder = CUPS_ORDER_BANDED; + expected.cupsNumColors = 1; + } + + if (page & 2) + { + expected.cupsBytesPerLine *= 2; + expected.cupsBitsPerColor = 16; + expected.cupsBitsPerPixel = (page & 1) ? 64 : 16; + } + else + { + expected.cupsBitsPerColor = 8; + expected.cupsBitsPerPixel = (page & 1) ? 32 : 8; + } + + fputs("cupsRasterReadHeader2: ", stdout); + fflush(stdout); + + if (!cupsRasterReadHeader2(r, &header)) + { + puts("FAIL (read error)"); + errors ++; + break; + } + + if (memcmp(&header, &expected, sizeof(header))) + { + puts("FAIL (bad page header)"); + errors ++; + print_changes(&header, &expected); + } + + fputs("cupsRasterReadPixels: ", stdout); + fflush(stdout); + + for (y = 0; y < 64; y ++) + { + if (!cupsRasterReadPixels(r, data, header.cupsBytesPerLine)) + { + puts("FAIL (read error)"); + errors ++; + break; + } + + if (data[0] != 0 || memcmp(data, data + 1, header.cupsBytesPerLine - 1)) + { + printf("FAIL (raster line %d corrupt)\n", y); + errors ++; + break; + } + } + + if (y == 64) + { + for (y = 0; y < 64; y ++) + { + if (!cupsRasterReadPixels(r, data, header.cupsBytesPerLine)) + { + puts("FAIL (read error)"); + errors ++; + break; + } + + for (x = 0; x < header.cupsBytesPerLine; x ++) + if (data[x] != (x & 255)) + break; + + if (x < header.cupsBytesPerLine) + { + printf("FAIL (raster line %d corrupt)\n", y + 64); + errors ++; + break; + } + } + + if (y == 64) + { + for (y = 0; y < 64; y ++) + { + if (!cupsRasterReadPixels(r, data, header.cupsBytesPerLine)) + { + puts("FAIL (read error)"); + errors ++; + break; + } + + if (data[0] != 255 || memcmp(data, data + 1, header.cupsBytesPerLine - 1)) + { + printf("fail (raster line %d corrupt)\n", y + 128); + errors ++; + break; + } + } + + if (y == 64) + { + for (y = 0; y < 64; y ++) + { + if (!cupsRasterReadPixels(r, data, header.cupsBytesPerLine)) + { + puts("FAIL (read error)"); + errors ++; + break; + } + + for (x = 0; x < header.cupsBytesPerLine; x ++) + if (data[x] != ((x / 4) & 255)) + break; + + if (x < header.cupsBytesPerLine) + { + printf("FAIL (raster line %d corrupt)\n", y + 192); + errors ++; + break; + } + } + + if (y == 64) + puts("PASS"); + } + } + } + } + + cupsRasterClose(r); + fclose(fp); + + return (errors); +} + + +/* + * 'print_changes()' - Print differences in the page header. + */ + +static void +print_changes( + cups_page_header2_t *header, /* I - Actual page header */ + cups_page_header2_t *expected) /* I - Expected page header */ +{ + int i; /* Looping var */ + + + if (strcmp(header->MediaClass, expected->MediaClass)) + printf(" MediaClass (%s), expected (%s)\n", header->MediaClass, + expected->MediaClass); + + if (strcmp(header->MediaColor, expected->MediaColor)) + printf(" MediaColor (%s), expected (%s)\n", header->MediaColor, + expected->MediaColor); + + if (strcmp(header->MediaType, expected->MediaType)) + printf(" MediaType (%s), expected (%s)\n", header->MediaType, + expected->MediaType); + + if (strcmp(header->OutputType, expected->OutputType)) + printf(" OutputType (%s), expected (%s)\n", header->OutputType, + expected->OutputType); + + if (header->AdvanceDistance != expected->AdvanceDistance) + printf(" AdvanceDistance %d, expected %d\n", header->AdvanceDistance, + expected->AdvanceDistance); + + if (header->AdvanceMedia != expected->AdvanceMedia) + printf(" AdvanceMedia %d, expected %d\n", header->AdvanceMedia, + expected->AdvanceMedia); + + if (header->Collate != expected->Collate) + printf(" Collate %d, expected %d\n", header->Collate, + expected->Collate); + + if (header->CutMedia != expected->CutMedia) + printf(" CutMedia %d, expected %d\n", header->CutMedia, + expected->CutMedia); + + if (header->Duplex != expected->Duplex) + printf(" Duplex %d, expected %d\n", header->Duplex, + expected->Duplex); + + if (header->HWResolution[0] != expected->HWResolution[0] || + header->HWResolution[1] != expected->HWResolution[1]) + printf(" HWResolution [%d %d], expected [%d %d]\n", + header->HWResolution[0], header->HWResolution[1], + expected->HWResolution[0], expected->HWResolution[1]); + + if (memcmp(header->ImagingBoundingBox, expected->ImagingBoundingBox, + sizeof(header->ImagingBoundingBox))) + printf(" ImagingBoundingBox [%d %d %d %d], expected [%d %d %d %d]\n", + header->ImagingBoundingBox[0], + header->ImagingBoundingBox[1], + header->ImagingBoundingBox[2], + header->ImagingBoundingBox[3], + expected->ImagingBoundingBox[0], + expected->ImagingBoundingBox[1], + expected->ImagingBoundingBox[2], + expected->ImagingBoundingBox[3]); + + if (header->InsertSheet != expected->InsertSheet) + printf(" InsertSheet %d, expected %d\n", header->InsertSheet, + expected->InsertSheet); + + if (header->Jog != expected->Jog) + printf(" Jog %d, expected %d\n", header->Jog, + expected->Jog); + + if (header->LeadingEdge != expected->LeadingEdge) + printf(" LeadingEdge %d, expected %d\n", header->LeadingEdge, + expected->LeadingEdge); + + if (header->Margins[0] != expected->Margins[0] || + header->Margins[1] != expected->Margins[1]) + printf(" Margins [%d %d], expected [%d %d]\n", + header->Margins[0], header->Margins[1], + expected->Margins[0], expected->Margins[1]); + + if (header->ManualFeed != expected->ManualFeed) + printf(" ManualFeed %d, expected %d\n", header->ManualFeed, + expected->ManualFeed); + + if (header->MediaPosition != expected->MediaPosition) + printf(" MediaPosition %d, expected %d\n", header->MediaPosition, + expected->MediaPosition); + + if (header->MediaWeight != expected->MediaWeight) + printf(" MediaWeight %d, expected %d\n", header->MediaWeight, + expected->MediaWeight); + + if (header->MirrorPrint != expected->MirrorPrint) + printf(" MirrorPrint %d, expected %d\n", header->MirrorPrint, + expected->MirrorPrint); + + if (header->NegativePrint != expected->NegativePrint) + printf(" NegativePrint %d, expected %d\n", header->NegativePrint, + expected->NegativePrint); + + if (header->NumCopies != expected->NumCopies) + printf(" NumCopies %d, expected %d\n", header->NumCopies, + expected->NumCopies); + + if (header->Orientation != expected->Orientation) + printf(" Orientation %d, expected %d\n", header->Orientation, + expected->Orientation); + + if (header->OutputFaceUp != expected->OutputFaceUp) + printf(" OutputFaceUp %d, expected %d\n", header->OutputFaceUp, + expected->OutputFaceUp); + + if (header->PageSize[0] != expected->PageSize[0] || + header->PageSize[1] != expected->PageSize[1]) + printf(" PageSize [%d %d], expected [%d %d]\n", + header->PageSize[0], header->PageSize[1], + expected->PageSize[0], expected->PageSize[1]); + + if (header->Separations != expected->Separations) + printf(" Separations %d, expected %d\n", header->Separations, + expected->Separations); + + if (header->TraySwitch != expected->TraySwitch) + printf(" TraySwitch %d, expected %d\n", header->TraySwitch, + expected->TraySwitch); + + if (header->Tumble != expected->Tumble) + printf(" Tumble %d, expected %d\n", header->Tumble, + expected->Tumble); + + if (header->cupsWidth != expected->cupsWidth) + printf(" cupsWidth %d, expected %d\n", header->cupsWidth, + expected->cupsWidth); + + if (header->cupsHeight != expected->cupsHeight) + printf(" cupsHeight %d, expected %d\n", header->cupsHeight, + expected->cupsHeight); + + if (header->cupsMediaType != expected->cupsMediaType) + printf(" cupsMediaType %d, expected %d\n", header->cupsMediaType, + expected->cupsMediaType); + + if (header->cupsBitsPerColor != expected->cupsBitsPerColor) + printf(" cupsBitsPerColor %d, expected %d\n", header->cupsBitsPerColor, + expected->cupsBitsPerColor); + + if (header->cupsBitsPerPixel != expected->cupsBitsPerPixel) + printf(" cupsBitsPerPixel %d, expected %d\n", header->cupsBitsPerPixel, + expected->cupsBitsPerPixel); + + if (header->cupsBytesPerLine != expected->cupsBytesPerLine) + printf(" cupsBytesPerLine %d, expected %d\n", header->cupsBytesPerLine, + expected->cupsBytesPerLine); + + if (header->cupsColorOrder != expected->cupsColorOrder) + printf(" cupsColorOrder %d, expected %d\n", header->cupsColorOrder, + expected->cupsColorOrder); + + if (header->cupsColorSpace != expected->cupsColorSpace) + printf(" cupsColorSpace %d, expected %d\n", header->cupsColorSpace, + expected->cupsColorSpace); + + if (header->cupsCompression != expected->cupsCompression) + printf(" cupsCompression %d, expected %d\n", header->cupsCompression, + expected->cupsCompression); + + if (header->cupsRowCount != expected->cupsRowCount) + printf(" cupsRowCount %d, expected %d\n", header->cupsRowCount, + expected->cupsRowCount); + + if (header->cupsRowFeed != expected->cupsRowFeed) + printf(" cupsRowFeed %d, expected %d\n", header->cupsRowFeed, + expected->cupsRowFeed); + + if (header->cupsRowStep != expected->cupsRowStep) + printf(" cupsRowStep %d, expected %d\n", header->cupsRowStep, + expected->cupsRowStep); + + if (header->cupsNumColors != expected->cupsNumColors) + printf(" cupsNumColors %d, expected %d\n", header->cupsNumColors, + expected->cupsNumColors); + + if (header->cupsBorderlessScalingFactor != + expected->cupsBorderlessScalingFactor) + printf(" cupsBorderlessScalingFactor %g, expected %g\n", + header->cupsBorderlessScalingFactor, + expected->cupsBorderlessScalingFactor); + + if (header->cupsPageSize[0] != expected->cupsPageSize[0] || + header->cupsPageSize[1] != expected->cupsPageSize[1]) + printf(" cupsPageSize [%g %g], expected [%g %g]\n", + header->cupsPageSize[0], header->cupsPageSize[1], + expected->cupsPageSize[0], expected->cupsPageSize[1]); + + if (header->cupsImagingBBox[0] != expected->cupsImagingBBox[0] || + header->cupsImagingBBox[1] != expected->cupsImagingBBox[1] || + header->cupsImagingBBox[2] != expected->cupsImagingBBox[2] || + header->cupsImagingBBox[3] != expected->cupsImagingBBox[3]) + printf(" cupsImagingBBox [%g %g %g %g], expected [%g %g %g %g]\n", + header->cupsImagingBBox[0], header->cupsImagingBBox[1], + header->cupsImagingBBox[2], header->cupsImagingBBox[3], + expected->cupsImagingBBox[0], expected->cupsImagingBBox[1], + expected->cupsImagingBBox[2], expected->cupsImagingBBox[3]); + + for (i = 0; i < 16; i ++) + if (header->cupsInteger[i] != expected->cupsInteger[i]) + printf(" cupsInteger%d %d, expected %d\n", i, header->cupsInteger[i], + expected->cupsInteger[i]); + + for (i = 0; i < 16; i ++) + if (header->cupsReal[i] != expected->cupsReal[i]) + printf(" cupsReal%d %g, expected %g\n", i, header->cupsReal[i], + expected->cupsReal[i]); + + for (i = 0; i < 16; i ++) + if (strcmp(header->cupsString[i], expected->cupsString[i])) + printf(" cupsString%d (%s), expected (%s)\n", i, + header->cupsString[i], expected->cupsString[i]); + + if (strcmp(header->cupsMarkerType, expected->cupsMarkerType)) + printf(" cupsMarkerType (%s), expected (%s)\n", header->cupsMarkerType, + expected->cupsMarkerType); + + if (strcmp(header->cupsRenderingIntent, expected->cupsRenderingIntent)) + printf(" cupsRenderingIntent (%s), expected (%s)\n", + header->cupsRenderingIntent, + expected->cupsRenderingIntent); + + if (strcmp(header->cupsPageSizeName, expected->cupsPageSizeName)) + printf(" cupsPageSizeName (%s), expected (%s)\n", + header->cupsPageSizeName, + expected->cupsPageSizeName); +} + + +/* + * End of "$Id$". + */ diff --git a/install-sh b/install-sh new file mode 100755 index 0000000000..05fe96fe44 --- /dev/null +++ b/install-sh @@ -0,0 +1,222 @@ +#!/bin/sh +# +# "$Id: install-sh 8688 2009-05-27 16:42:40Z mike $" +# +# Install a program, script, or datafile. +# +# Copyright 2008-2009 by Apple Inc. +# +# This script is not compatible with BSD (or any other) install program, as it +# allows owner and group changes to fail with a warning and makes sure that the +# destination directory permissions are as specified - BSD install and the +# original X11 install script did not change permissions of existing +# directories. It also does not support the transform options since CUPS does +# not use them... +# +# Original script from X11R5 (mit/util/scripts/install.sh) +# Copyright 1991 by the Massachusetts Institute of Technology +# +# Permission to use, copy, modify, distribute, and sell this software and its +# documentation for any purpose is hereby granted without fee, provided that +# the above copyright notice appear in all copies and that both that +# copyright notice and this permission notice appear in supporting +# documentation, and that the name of M.I.T. not be used in advertising or +# publicity pertaining to distribution of the software without specific, +# written prior permission. M.I.T. makes no representations about the +# suitability of this software for any purpose. It is provided "as is" +# without express or implied warranty. +# +# Calling this script install-sh is preferred over install.sh, to prevent +# `make' implicit rules from creating a file called install from it +# when there is no Makefile. + +# set DOITPROG to echo to test this script +# Don't use :- since 4.3BSD and earlier shells don't like it. +doit="${DOITPROG-}" + +# Force umask to 022... +umask 022 + +# put in absolute paths if you don't have them in your path; or use env. vars. +mvprog="${MVPROG-mv}" +cpprog="${CPPROG-cp}" +chmodprog="${CHMODPROG-chmod}" +chownprog="${CHOWNPROG-chown}" +chgrpprog="${CHGRPPROG-chgrp}" +stripprog="${STRIPPROG-strip}" +rmprog="${RMPROG-rm}" +mkdirprog="${MKDIRPROG-mkdir}" + +transformbasename="" +transform_arg="" +instcmd="$mvprog" +chmodcmd="$chmodprog 0755" +chowncmd="" +chgrpcmd="" +stripcmd="" +rmcmd="$rmprog -f" +mvcmd="$mvprog" +src="" +dst="" +dir_arg="" + +while [ x"$1" != x ]; do + case $1 in + -c) + instcmd="$cpprog" + shift + continue + ;; + + -d) + dir_arg=true + shift + continue + ;; + + -m) + chmodcmd="$chmodprog $2" + shift + shift + continue + ;; + + -o) + chowncmd="$chownprog $2" + shift + shift + continue + ;; + + -g) + chgrpcmd="$chgrpprog $2" + shift + shift + continue + ;; + + -s) + stripcmd="$stripprog" + shift + continue + ;; + + *) + if [ x"$src" = x ]; then + src="$1" + else + dst="$1" + fi + shift + continue + ;; + esac +done + +if [ x"$src" = x ]; then + echo "install-sh: No input file specified" + exit 1 +fi + +if [ x"$dir_arg" != x ]; then + dst="$src" + src="" + + if [ -d "$dst" ]; then + instcmd=: + else + instcmd=$mkdirprog + fi +else + # Waiting for this to be detected by the "$instcmd $src $dsttmp" command + # might cause directories to be created, which would be especially bad + # if $src (and thus $dsttmp) contains '*'. + if [ ! -f "$src" -a ! -d "$src" ]; then + echo "install: $src does not exist" + exit 1 + fi + + if [ x"$dst" = x ]; then + echo "install: No destination specified" + exit 1 + fi + + # If destination is a directory, append the input filename. + if [ -d "$dst" ]; then + dst="$dst/`basename $src`" + fi +fi + +## this sed command emulates the dirname command +dstdir="`echo $dst | sed -e 's,[^/]*$,,;s,/$,,;s,^$,.,'`" + +# Make sure that the destination directory exists. +# This part is taken from Noah Friedman's mkinstalldirs script + +# Skip lots of stat calls in the usual case. +if [ ! -d "$dstdir" ]; then + defaultIFS=' + ' + IFS="${IFS-${defaultIFS}}" + + oIFS="${IFS}" + # Some sh's can't handle IFS=/ for some reason. + IFS='%' + set - `echo ${dstdir} | sed -e 's@/@%@g' -e 's@^%@/@'` + IFS="${oIFS}" + + pathcomp='' + + while [ $# -ne 0 ] ; do + pathcomp="${pathcomp}${1}" + shift + + if [ ! -d "${pathcomp}" ]; then $doit $mkdirprog "${pathcomp}"; fi + + pathcomp="${pathcomp}/" + done +fi + +if [ x"$dir_arg" != x ]; then + # Make a directory... + $doit $instcmd $dst || exit 1 + + # Allow chown/chgrp to fail, but log a warning + if [ x"$chowncmd" != x ]; then $doit $chowncmd $dst || echo "warning: Unable to change owner of $dst!"; fi + if [ x"$chgrpcmd" != x ]; then $doit $chgrpcmd $dst || echo "warning: Unable to change group of $dst!"; fi + if [ x"$chmodcmd" != x ]; then $doit $chmodcmd $dst || exit 1; fi +else + # Install a file... + dstfile="`basename $dst`" + + # Check the destination file - for libraries just use the "-x" option + # to strip... + case "$dstfile" in + *.a | *.dylib | *.sl | *.sl.* | *.so | *.so.*) + stripopt="-x" + ;; + *) + stripopt="" + ;; + esac + + # Make a temp file name in the proper directory. + dsttmp="$dstdir/#inst.$$#" + + # Move or copy the file name to the temp name + $doit $instcmd $src $dsttmp || exit 1 + + # Update permissions and strip as needed, then move to the final name. + # If the chmod, strip, rm, or mv commands fail, remove the installed + # file... + if [ x"$stripcmd" != x ]; then $doit $stripcmd $stripopt "$dsttmp" || echo "warning: Unable to strip $dst!"; fi + if [ x"$chowncmd" != x ]; then $doit $chowncmd "$dsttmp" || echo "warning: Unable to change owner of $dst!"; fi + if [ x"$chgrpcmd" != x ]; then $doit $chgrpcmd "$dsttmp" || echo "warning: Unable to change group of $dst!"; fi + + trap "rm -f ${dsttmp}" 0 && + if [ x"$chmodcmd" != x ]; then $doit $chmodcmd "$dsttmp"; fi && + $doit $rmcmd -f "$dstdir/$dstfile" && + $doit $mvcmd "$dsttmp" "$dstdir/$dstfile" +fi + +exit 0 diff --git a/locale/Dependencies b/locale/Dependencies new file mode 100644 index 0000000000..038521b55e --- /dev/null +++ b/locale/Dependencies @@ -0,0 +1,22 @@ +checkpo.o: checkpo.c ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +po2strings.o: po2strings.c ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h +strings2po.o: strings2po.c +translate.o: translate.c ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h diff --git a/locale/Makefile b/locale/Makefile new file mode 100644 index 0000000000..d9d01bd908 --- /dev/null +++ b/locale/Makefile @@ -0,0 +1,219 @@ +# +# "$Id$" +# +# Locale file makefile for CUPS. +# +# Copyright 2007-2012 by Apple Inc. +# Copyright 1993-2007 by Easy Software Products. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +include ../Makedefs + + +OBJS = checkpo.o po2strings.o strings2po.o translate.o +TARGETS = checkpo po2strings strings2po translate + + +# +# Make everything... +# + +all: $(TARGETS) + + +# +# Make library targets... +# + +libs: + + +# +# Make unit tests... +# + +unittests: + + +# +# Clean all config and object files... +# + +clean: + $(RM) $(TARGETS) $(OBJS) + + +# +# Update dependencies (without system header dependencies...) +# + +depend: + $(CC) -MM $(ALL_CFLAGS) $(OBJS:.o=.c) >Dependencies + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: $(INSTALL_LANGUAGES) + +install-languages: + $(INSTALL_DIR) -m 755 $(LOCALEDIR) + for loc in $(LANGUAGES) ; do \ + if test -f cups_$$loc.po; then \ + $(INSTALL_DIR) -m 755 $(LOCALEDIR)/$$loc ; \ + $(INSTALL_DATA) cups_$$loc.po $(LOCALEDIR)/$$loc/cups_$$loc.po ; \ + fi ; \ + done + +install-langbundle: po2strings + $(INSTALL_DIR) -m 755 "$(BUILDROOT)$(BUNDLEDIR)/Resources/English.lproj" + $(INSTALL_DATA) cups.strings "$(BUILDROOT)$(BUNDLEDIR)/Resources/English.lproj" + + +# +# Install programs... +# + +install-exec: + + +# +# Install headers... +# + +install-headers: + + +# +# Install libraries... +# + +install-libs: + + +# +# Uninstall files... +# + +uninstall: $(UNINSTALL_LANGUAGES) + +uninstall-languages: + -for loc in $(LANGUAGES) ; do \ + $(RM) $(LOCALEDIR)/$$loc/cups_$$loc.po ; \ + done + +uninstall-langbundle: + $(RM) "$(BUILDROOT)$(BUNDLEDIR)/Resources/English.lproj/cups.strings" + + +# +# pot - Creates/updates the cups.pot template file, merges changes into existing +# message catalogs, and updates the cups.strings file. We don't use +# xgettext to update the cups.strings file due to known xgettext bugs. +# + +pot: checkpo po2strings + echo Updating cups.pot... + mv cups.pot cups.pot.bck + touch cups.pot + cd ..; xgettext -o locale/cups.pot -cTRANSLATORS -s \ + --keyword=_ --no-wrap \ + --copyright-holder="Apple Inc." \ + --package-name="CUPS" --package-version="1.5" \ + --msgid-bugs-address="http://www.cups.org/str.php" \ + */*.c */*.cxx + (cat cups.header; tail +6 cups.pot; cat cups.footer) > cups.pot.N + mv cups.pot.N cups.pot + echo Checking cups.pot... + ./checkpo cups.pot + for loc in *.po ; do \ + echo Merging changes into $$loc... ; \ + msgmerge -o $$loc -s -N --no-location $$loc cups.pot ; \ + done + echo Updating cups.strings... + ./po2strings cups.pot cups.strings + + +# +# checkpo - A simple utility to check PO files for correct translation +# strings. Dependency on static library is deliberate. +# +# checkpo filename.po [... filenameN.po] +# + +checkpo: checkpo.o ../cups/$(LIBCUPSSTATIC) + echo Linking $<... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o checkpo checkpo.o \ + ../cups/$(LIBCUPSSTATIC) $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) \ + $(COMMONLIBS) $(LIBZ) + +checkall: checkpo + for file in *.po; do \ + ./checkpo $$file; \ + done + + +# +# po2strings - A simple utility which uses iconv to convert GNU gettext +# message catalogs to Mac OS X .strings files. +# +# po2strings filename.po filename.strings +# + +po2strings: po2strings.o ../cups/$(LIBCUPSSTATIC) + echo Linking $<... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o po2strings po2strings.o \ + ../cups/$(LIBCUPSSTATIC) $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) \ + $(COMMONLIBS) $(LIBZ) + + +# +# strings2po - A simple utility which uses iconv to convert Mac OS X +# .strings files to GNU gettext message catalogs. +# +# strings2po filename.strings filename.po +# + +strings2po: strings2po.o + echo Linking $<... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o strings2po strings2po.o + + +# +# translate - A simple utility which uses Google to translate the cups.pot +# file to one of several languages. +# +# translate outfile language +# + +translate: translate.o ../cups/$(LIBCUPSSTATIC) + echo Linking $<... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o translate translate.o \ + ../cups/$(LIBCUPSSTATIC) $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) \ + $(COMMONLIBS) $(LIBZ) + + +# +# Dependencies... +# + +include Dependencies + + +# +# End of "$Id$". +# diff --git a/locale/checkpo.c b/locale/checkpo.c new file mode 100644 index 0000000000..fa2afbad3c --- /dev/null +++ b/locale/checkpo.c @@ -0,0 +1,413 @@ +/* + * "$Id$" + * + * Verify that translations in the .po file have the same number and type of + * printf-style format strings. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Usage: + * + * checkpo filename.po [... filenameN.po] + * + * Compile with: + * + * gcc -o checkpo checkpo.c `cups-config --libs` + * + * Contents: + * + * main() - Validate .po files. + * abbreviate() - Abbreviate a message string as needed. + * collect_formats() - Collect all of the format strings in the msgid. + * free_formats() - Free all of the format strings. + */ + +#include + + +/* + * Local functions... + */ + +static char *abbreviate(const char *s, char *buf, int bufsize); +static cups_array_t *collect_formats(const char *id); +static void free_formats(cups_array_t *fmts); + + +/* + * 'main()' - Validate .po files. + */ + +int /* O - Exit code */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + cups_array_t *po; /* .po file */ + _cups_message_t *msg; /* Current message */ + cups_array_t *idfmts, /* Format strings in msgid */ + *strfmts; /* Format strings in msgstr */ + char *idfmt, /* Current msgid format string */ + *strfmt; /* Current msgstr format string */ + int fmtidx; /* Format index */ + int status, /* Exit status */ + pass, /* Pass/fail status */ + untranslated; /* Untranslated messages */ + char idbuf[80], /* Abbreviated msgid */ + strbuf[80]; /* Abbreviated msgstr */ + + + if (argc < 2) + { + puts("Usage: checkpo filename.po [... filenameN.po]"); + return (1); + } + + /* + * Check every .po file on the command-line... + */ + + for (i = 1, status = 0; i < argc; i ++) + { + /* + * Use the CUPS .po loader to get the message strings... + */ + + if ((po = _cupsMessageLoad(argv[i], 1)) == NULL) + { + perror(argv[i]); + return (1); + } + + if (i > 1) + putchar('\n'); + printf("%s: ", argv[i]); + fflush(stdout); + + /* + * Scan every message for a % string and then match them up with + * the corresponding string in the translation... + */ + + pass = 1; + untranslated = 0; + + for (msg = (_cups_message_t *)cupsArrayFirst(po); + msg; + msg = (_cups_message_t *)cupsArrayNext(po)) + { + /* + * Make sure filter message prefixes are not translated... + */ + + if (!strncmp(msg->id, "ALERT:", 6) || !strncmp(msg->id, "CRIT:", 5) || + !strncmp(msg->id, "DEBUG:", 6) || !strncmp(msg->id, "DEBUG2:", 7) || + !strncmp(msg->id, "EMERG:", 6) || !strncmp(msg->id, "ERROR:", 6) || + !strncmp(msg->id, "INFO:", 5) || !strncmp(msg->id, "NOTICE:", 7) || + !strncmp(msg->id, "WARNING:", 8)) + { + if (pass) + { + pass = 0; + puts("FAIL"); + } + + printf(" Bad prefix on filter message \"%s\"\n", + abbreviate(msg->id, idbuf, sizeof(idbuf))); + } + + idfmt = msg->id + strlen(msg->id) - 1; + if (idfmt >= msg->id && *idfmt == '\n') + { + if (pass) + { + pass = 0; + puts("FAIL"); + } + + printf(" Trailing newline in message \"%s\"\n", + abbreviate(msg->id, idbuf, sizeof(idbuf))); + } + + for (; idfmt >= msg->id; idfmt --) + if (!isspace(*idfmt & 255)) + break; + + if (idfmt >= msg->id && *idfmt == '!') + { + if (pass) + { + pass = 0; + puts("FAIL"); + } + + printf(" Exclamation in message \"%s\"\n", + abbreviate(msg->id, idbuf, sizeof(idbuf))); + } + + if ((idfmt - 2) >= msg->id && !strncmp(idfmt - 2, "...", 3)) + { + if (pass) + { + pass = 0; + puts("FAIL"); + } + + printf(" Ellipsis in message \"%s\"\n", + abbreviate(msg->id, idbuf, sizeof(idbuf))); + } + + + if (!msg->str || !msg->str[0]) + { + untranslated ++; + continue; + } + else if (strchr(msg->id, '%')) + { + idfmts = collect_formats(msg->id); + strfmts = collect_formats(msg->str); + fmtidx = 0; + + for (strfmt = (char *)cupsArrayFirst(strfmts); + strfmt; + strfmt = (char *)cupsArrayNext(strfmts)) + { + if (isdigit(strfmt[1] & 255) && strfmt[2] == '$') + { + /* + * Handle positioned format stuff... + */ + + fmtidx = strfmt[1] - '1'; + strfmt += 3; + if ((idfmt = (char *)cupsArrayIndex(idfmts, fmtidx)) != NULL) + idfmt ++; + } + else + { + /* + * Compare against the current format... + */ + + idfmt = (char *)cupsArrayIndex(idfmts, fmtidx); + } + + fmtidx ++; + + if (!idfmt || strcmp(strfmt, idfmt)) + break; + } + + if (cupsArrayCount(strfmts) != cupsArrayCount(idfmts) || strfmt) + { + if (pass) + { + pass = 0; + puts("FAIL"); + } + + printf(" Bad translation string \"%s\"\n for \"%s\"\n", + abbreviate(msg->str, strbuf, sizeof(strbuf)), + abbreviate(msg->id, idbuf, sizeof(idbuf))); + fputs(" Translation formats:", stdout); + for (strfmt = (char *)cupsArrayFirst(strfmts); + strfmt; + strfmt = (char *)cupsArrayNext(strfmts)) + printf(" %s", strfmt); + fputs("\n Original formats:", stdout); + for (idfmt = (char *)cupsArrayFirst(idfmts); + idfmt; + idfmt = (char *)cupsArrayNext(idfmts)) + printf(" %s", idfmt); + putchar('\n'); + putchar('\n'); + } + + free_formats(idfmts); + free_formats(strfmts); + } + + /* + * Only allow \\, \n, \r, \t, \", and \### character escapes... + */ + + for (strfmt = msg->str; *strfmt; strfmt ++) + if (*strfmt == '\\' && + strfmt[1] != '\\' && strfmt[1] != 'n' && strfmt[1] != 'r' && + strfmt[1] != 't' && strfmt[1] != '\"' && !isdigit(strfmt[1] & 255)) + { + if (pass) + { + pass = 0; + puts("FAIL"); + } + + printf(" Bad escape \\%c in filter message \"%s\"\n" + " for \"%s\"\n", strfmt[1], + abbreviate(msg->str, strbuf, sizeof(strbuf)), + abbreviate(msg->id, idbuf, sizeof(idbuf))); + break; + } + } + + if (pass) + { + if ((untranslated * 10) >= cupsArrayCount(po) && + strcmp(argv[i], "cups.pot")) + { + /* + * Only allow 10% of messages to be untranslated before we fail... + */ + + pass = 0; + puts("FAIL"); + printf(" Too many untranslated messages (%d of %d)\n", + untranslated, cupsArrayCount(po)); + } + else if (untranslated > 0) + printf("PASS (%d of %d untranslated)\n", untranslated, + cupsArrayCount(po)); + else + puts("PASS"); + } + + if (!pass) + status = 1; + + _cupsMessageFree(po); + } + + return (status); +} + + +/* + * 'abbreviate()' - Abbreviate a message string as needed. + */ + +static char * /* O - Abbreviated string */ +abbreviate(const char *s, /* I - String to abbreviate */ + char *buf, /* I - Buffer */ + int bufsize) /* I - Size of buffer */ +{ + char *bufptr; /* Pointer into buffer */ + + + for (bufptr = buf, bufsize -= 4; *s && bufsize > 0; s ++) + { + if (*s == '\n') + { + if (bufsize < 2) + break; + + *bufptr++ = '\\'; + *bufptr++ = 'n'; + bufsize -= 2; + } + else if (*s == '\t') + { + if (bufsize < 2) + break; + + *bufptr++ = '\\'; + *bufptr++ = 't'; + bufsize -= 2; + } + else if (*s >= 0 && *s < ' ') + { + if (bufsize < 4) + break; + + sprintf(bufptr, "\\%03o", *s); + bufptr += 4; + bufsize -= 4; + } + else + { + *bufptr++ = *s; + bufsize --; + } + } + + if (*s) + strcpy(bufptr, "..."); + else + *bufptr = '\0'; + + return (buf); +} + + +/* + * 'collect_formats()' - Collect all of the format strings in the msgid. + */ + +static cups_array_t * /* O - Array of format strings */ +collect_formats(const char *id) /* I - msgid string */ +{ + cups_array_t *fmts; /* Array of format strings */ + char buf[255], /* Format string buffer */ + *bufptr; /* Pointer into format string */ + + + fmts = cupsArrayNew(NULL, NULL); + + while ((id = strchr(id, '%')) != NULL) + { + if (id[1] == '%') + { + /* + * Skip %%... + */ + + id += 2; + continue; + } + + for (bufptr = buf; *id && bufptr < (buf + sizeof(buf) - 1); id ++) + { + *bufptr++ = *id; + + if (strchr("CDEFGIOSUXcdeifgopsux", *id)) + { + id ++; + break; + } + } + + *bufptr = '\0'; + cupsArrayAdd(fmts, strdup(buf)); + } + + return (fmts); +} + + +/* + * 'free_formats()' - Free all of the format strings. + */ + +static void +free_formats(cups_array_t *fmts) /* I - Array of format strings */ +{ + char *s; /* Current string */ + + + for (s = (char *)cupsArrayFirst(fmts); s; s = (char *)cupsArrayNext(fmts)) + free(s); + + cupsArrayDelete(fmts); +} + + +/* + * End of "$Id$". + */ diff --git a/locale/cups.footer b/locale/cups.footer new file mode 100644 index 0000000000..490c9d4ebd --- /dev/null +++ b/locale/cups.footer @@ -0,0 +1,5 @@ + + +# +# End of "$Id$". +# diff --git a/locale/cups.header b/locale/cups.header new file mode 100644 index 0000000000..a50d50a553 --- /dev/null +++ b/locale/cups.header @@ -0,0 +1,27 @@ +# +# "$Id$" +# +# Message catalog template for CUPS. +# +# Copyright 2007-2010 by Apple Inc. +# Copyright 2005-2007 by Easy Software Products. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +# +# Notes for Translators: +# +# The "checkpo" program located in the "locale" source directory can be used +# to verify that your translations do not introduce formatting errors or other +# problems. Run with: +# +# cd locale +# ./checkpo cups_LL.po +# +# where "LL" is your locale. +# diff --git a/locale/cups.pot b/locale/cups.pot new file mode 100644 index 0000000000..404e2ba319 --- /dev/null +++ b/locale/cups.pot @@ -0,0 +1,6824 @@ +# +# "$Id$" +# +# Message catalog template for CUPS. +# +# Copyright 2007-2010 by Apple Inc. +# Copyright 2005-2007 by Easy Software Products. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +# +# Notes for Translators: +# +# The "checkpo" program located in the "locale" source directory can be used +# to verify that your translations do not introduce formatting errors or other +# problems. Run with: +# +# cd locale +# ./checkpo cups_LL.po +# +# where "LL" is your locale. +# +#, fuzzy +msgid "" +msgstr "" +"Project-Id-Version: CUPS 1.5\n" +"Report-Msgid-Bugs-To: http://www.cups.org/str.php\n" +"POT-Creation-Date: 2012-01-03 20:47-0800\n" +"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n" +"Last-Translator: FULL NAME \n" +"Language-Team: LANGUAGE \n" +"Language: \n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=CHARSET\n" +"Content-Transfer-Encoding: 8bit\n" + +#: systemv/lpstat.c:1876 systemv/lpstat.c:2001 +msgid "\t\t(all)" +msgstr "" + +#: systemv/lpstat.c:1879 systemv/lpstat.c:1882 systemv/lpstat.c:2004 +#: systemv/lpstat.c:2007 +msgid "\t\t(none)" +msgstr "" + +#: berkeley/lpc.c:433 +#, c-format +msgid "\t%d entries" +msgstr "" + +#: systemv/lpstat.c:756 systemv/lpstat.c:772 +#, c-format +msgid "\t%s" +msgstr "" + +#: systemv/lpstat.c:1857 systemv/lpstat.c:1982 +msgid "\tAfter fault: continue" +msgstr "" + +#: systemv/lpstat.c:1481 systemv/lpstat.c:1826 systemv/lpstat.c:1952 +#, c-format +msgid "\tAlerts: %s" +msgstr "" + +#: systemv/lpstat.c:1880 systemv/lpstat.c:2005 +msgid "\tBanner required" +msgstr "" + +#: systemv/lpstat.c:1881 systemv/lpstat.c:2006 +msgid "\tCharset sets:" +msgstr "" + +#: systemv/lpstat.c:1845 systemv/lpstat.c:1970 +msgid "\tConnection: direct" +msgstr "" + +#: systemv/lpstat.c:1836 systemv/lpstat.c:1962 +msgid "\tConnection: remote" +msgstr "" + +#: systemv/lpstat.c:1800 systemv/lpstat.c:1926 +msgid "\tContent types: any" +msgstr "" + +#: systemv/lpstat.c:1884 systemv/lpstat.c:2009 +msgid "\tDefault page size:" +msgstr "" + +#: systemv/lpstat.c:1883 systemv/lpstat.c:2008 +msgid "\tDefault pitch:" +msgstr "" + +#: systemv/lpstat.c:1885 systemv/lpstat.c:2010 +msgid "\tDefault port settings:" +msgstr "" + +#: systemv/lpstat.c:1806 systemv/lpstat.c:1932 +#, c-format +msgid "\tDescription: %s" +msgstr "" + +#: systemv/lpstat.c:1799 systemv/lpstat.c:1925 +msgid "\tForm mounted:" +msgstr "" + +#: systemv/lpstat.c:1878 systemv/lpstat.c:2003 +msgid "\tForms allowed:" +msgstr "" + +#: systemv/lpstat.c:1840 systemv/lpstat.c:1966 +#, c-format +msgid "\tInterface: %s.ppd" +msgstr "" + +#: systemv/lpstat.c:1849 systemv/lpstat.c:1974 +#, c-format +msgid "\tInterface: %s/interfaces/%s" +msgstr "" + +#: systemv/lpstat.c:1853 systemv/lpstat.c:1978 +#, c-format +msgid "\tInterface: %s/ppd/%s.ppd" +msgstr "" + +#: systemv/lpstat.c:1831 systemv/lpstat.c:1957 +#, c-format +msgid "\tLocation: %s" +msgstr "" + +#: systemv/lpstat.c:1856 systemv/lpstat.c:1981 +msgid "\tOn fault: no alert" +msgstr "" + +#: systemv/lpstat.c:1801 systemv/lpstat.c:1927 +msgid "\tPrinter types: unknown" +msgstr "" + +#: systemv/lpstat.c:1462 +#, c-format +msgid "\tStatus: %s" +msgstr "" + +#: systemv/lpstat.c:1861 systemv/lpstat.c:1875 systemv/lpstat.c:1986 +#: systemv/lpstat.c:2000 +msgid "\tUsers allowed:" +msgstr "" + +#: systemv/lpstat.c:1868 systemv/lpstat.c:1993 +msgid "\tUsers denied:" +msgstr "" + +#: berkeley/lpc.c:435 +msgid "\tdaemon present" +msgstr "" + +#: berkeley/lpc.c:431 +msgid "\tno entries" +msgstr "" + +#: berkeley/lpc.c:403 berkeley/lpc.c:415 +#, c-format +msgid "\tprinter is on device '%s' speed -1" +msgstr "" + +#: berkeley/lpc.c:428 +msgid "\tprinting is disabled" +msgstr "" + +#: berkeley/lpc.c:426 +msgid "\tprinting is enabled" +msgstr "" + +#: systemv/lpstat.c:1484 +#, c-format +msgid "\tqueued for %s" +msgstr "" + +#: berkeley/lpc.c:423 +msgid "\tqueuing is disabled" +msgstr "" + +#: berkeley/lpc.c:421 +msgid "\tqueuing is enabled" +msgstr "" + +#: systemv/lpstat.c:1792 systemv/lpstat.c:1918 +msgid "\treason unknown" +msgstr "" + +#: systemv/cupstestppd.c:436 +msgid "" +"\n" +" DETAILED CONFORMANCE TEST RESULTS" +msgstr "" + +#: systemv/cupstestppd.c:3780 +msgid " Ignore specific warnings." +msgstr "" + +#: systemv/cupstestppd.c:3784 +msgid " Issue warnings instead of errors." +msgstr "" + +#: systemv/cupstestppd.c:392 systemv/cupstestppd.c:397 +msgid " REF: Page 15, section 3.1." +msgstr "" + +#: systemv/cupstestppd.c:387 +msgid " REF: Page 15, section 3.2." +msgstr "" + +#: systemv/cupstestppd.c:407 +msgid " REF: Page 19, section 3.3." +msgstr "" + +#: systemv/cupstestppd.c:360 +msgid " REF: Page 20, section 3.4." +msgstr "" + +#: systemv/cupstestppd.c:412 +msgid " REF: Page 27, section 3.5." +msgstr "" + +#: systemv/cupstestppd.c:355 +msgid " REF: Page 42, section 5.2." +msgstr "" + +#: systemv/cupstestppd.c:402 +msgid " REF: Pages 16-17, section 3.2." +msgstr "" + +#: systemv/cupstestppd.c:372 +msgid " REF: Pages 42-45, section 5.2." +msgstr "" + +#: systemv/cupstestppd.c:366 +msgid " REF: Pages 45-46, section 5.2." +msgstr "" + +#: systemv/cupstestppd.c:377 +msgid " REF: Pages 48-49, section 5.2." +msgstr "" + +#: systemv/cupstestppd.c:382 +msgid " REF: Pages 52-54, section 5.2." +msgstr "" + +#: berkeley/lpq.c:554 +#, c-format +msgid " %-39.39s %.0f bytes" +msgstr "" + +#: systemv/cupstestppd.c:571 +#, c-format +msgid " PASS Default%s" +msgstr "" + +#: systemv/cupstestppd.c:506 +msgid " PASS DefaultImageableArea" +msgstr "" + +#: systemv/cupstestppd.c:540 +msgid " PASS DefaultPaperDimension" +msgstr "" + +#: systemv/cupstestppd.c:613 +msgid " PASS FileVersion" +msgstr "" + +#: systemv/cupstestppd.c:657 +msgid " PASS FormatVersion" +msgstr "" + +#: systemv/cupstestppd.c:677 +msgid " PASS LanguageEncoding" +msgstr "" + +#: systemv/cupstestppd.c:697 +msgid " PASS LanguageVersion" +msgstr "" + +#: systemv/cupstestppd.c:749 +msgid " PASS Manufacturer" +msgstr "" + +#: systemv/cupstestppd.c:789 +msgid " PASS ModelName" +msgstr "" + +#: systemv/cupstestppd.c:809 +msgid " PASS NickName" +msgstr "" + +#: systemv/cupstestppd.c:869 +msgid " PASS PCFileName" +msgstr "" + +#: systemv/cupstestppd.c:944 +msgid " PASS PSVersion" +msgstr "" + +#: systemv/cupstestppd.c:849 +msgid " PASS PageRegion" +msgstr "" + +#: systemv/cupstestppd.c:829 +msgid " PASS PageSize" +msgstr "" + +#: systemv/cupstestppd.c:904 +msgid " PASS Product" +msgstr "" + +#: systemv/cupstestppd.c:979 +msgid " PASS ShortNickName" +msgstr "" + +#: systemv/cupstestppd.c:1354 +#, c-format +msgid " WARN %s has no corresponding options." +msgstr "" + +#: systemv/cupstestppd.c:1466 +#, c-format +msgid "" +" WARN %s shares a common prefix with %s\n" +" REF: Page 15, section 3.2." +msgstr "" + +#: systemv/cupstestppd.c:1325 +#, c-format +msgid "" +" WARN Duplex option keyword %s may not work as expected and should be named Duplex.\n" +" REF: Page 122, section 5.17" +msgstr "" + +#: systemv/cupstestppd.c:1724 +msgid " WARN File contains a mix of CR, LF, and CR LF line endings." +msgstr "" + +#: systemv/cupstestppd.c:1370 +msgid "" +" WARN LanguageEncoding required by PPD 4.3 spec.\n" +" REF: Pages 56-57, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:1706 +#, c-format +msgid " WARN Line %d only contains whitespace." +msgstr "" + +#: systemv/cupstestppd.c:1378 +msgid "" +" WARN Manufacturer required by PPD 4.3 spec.\n" +" REF: Pages 58-59, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:1729 +msgid " WARN Non-Windows PPD files should use lines ending with only LF, not CR LF." +msgstr "" + +#: systemv/cupstestppd.c:1362 +#, c-format +msgid "" +" WARN Obsolete PPD version %.1f.\n" +" REF: Page 42, section 5.2." +msgstr "" + +#: systemv/cupstestppd.c:1393 +msgid "" +" WARN PCFileName longer than 8.3 in violation of PPD spec.\n" +" REF: Pages 61-62, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:1401 +msgid "" +" WARN PCFileName should contain a unique filename.\n" +" REF: Pages 61-62, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:1436 +msgid "" +" WARN Protocols contains PJL but JCL attributes are not set.\n" +" REF: Pages 78-79, section 5.7." +msgstr "" + +#: systemv/cupstestppd.c:1427 +msgid "" +" WARN Protocols contains both PJL and BCP; expected TBCP.\n" +" REF: Pages 78-79, section 5.7." +msgstr "" + +#: systemv/cupstestppd.c:1410 +msgid "" +" WARN ShortNickName required by PPD 4.3 spec.\n" +" REF: Pages 64-65, section 5.3." +msgstr "" + +#: systemv/cupsaddsmb.c:282 +msgid " cupsaddsmb [options] -a" +msgstr "" + +#: systemv/cupstestdsc.c:427 +msgid " cupstestdsc [options] -" +msgstr "" + +#: systemv/cupstestppd.c:3775 +msgid " program | cupstestppd [options] -" +msgstr "" + +#: systemv/cupstestppd.c:3707 +#, c-format +msgid "" +" %s \"%s %s\" conflicts with \"%s %s\"\n" +" (constraint=\"%s %s %s %s\")." +msgstr "" + +#: systemv/cupstestppd.c:2228 +#, c-format +msgid " %s %s %s does not exist." +msgstr "" + +#: systemv/cupstestppd.c:3864 +#, c-format +msgid " %s %s file \"%s\" has the wrong capitalization." +msgstr "" + +#: systemv/cupstestppd.c:2298 +#, c-format +msgid "" +" %s Bad %s choice %s.\n" +" REF: Page 122, section 5.17" +msgstr "" + +#: systemv/cupstestppd.c:3467 systemv/cupstestppd.c:3516 +#: systemv/cupstestppd.c:3555 +#, c-format +msgid " %s Bad UTF-8 \"%s\" translation string for option %s, choice %s." +msgstr "" + +#: systemv/cupstestppd.c:3421 +#, c-format +msgid " %s Bad UTF-8 \"%s\" translation string for option %s." +msgstr "" + +#: systemv/cupstestppd.c:2369 +#, c-format +msgid " %s Bad cupsFilter value \"%s\"." +msgstr "" + +#: systemv/cupstestppd.c:2455 +#, c-format +msgid " %s Bad cupsFilter2 value \"%s\"." +msgstr "" + +#: systemv/cupstestppd.c:2944 +#, c-format +msgid " %s Bad cupsICCProfile %s." +msgstr "" + +#: systemv/cupstestppd.c:2551 +#, c-format +msgid " %s Bad cupsPreFilter value \"%s\"." +msgstr "" + +#: systemv/cupstestppd.c:1802 +#, c-format +msgid " %s Bad cupsUIConstraints %s: \"%s\"" +msgstr "" + +#: systemv/cupstestppd.c:3371 +#, c-format +msgid " %s Bad language \"%s\"." +msgstr "" + +#: systemv/cupstestppd.c:2413 systemv/cupstestppd.c:2509 +#: systemv/cupstestppd.c:2595 systemv/cupstestppd.c:2653 +#: systemv/cupstestppd.c:2708 systemv/cupstestppd.c:2763 +#: systemv/cupstestppd.c:2818 systemv/cupstestppd.c:2871 +#: systemv/cupstestppd.c:2993 +#, c-format +msgid " %s Bad permissions on %s file \"%s\"." +msgstr "" + +#: systemv/cupstestppd.c:2353 systemv/cupstestppd.c:2439 +#: systemv/cupstestppd.c:2535 systemv/cupstestppd.c:2622 +#: systemv/cupstestppd.c:2677 systemv/cupstestppd.c:2732 +#: systemv/cupstestppd.c:2787 systemv/cupstestppd.c:2842 +#, c-format +msgid " %s Bad spelling of %s - should be %s." +msgstr "" + +#: systemv/cupstestppd.c:2887 +#, c-format +msgid " %s Cannot provide both APScanAppPath and APScanAppBundleID." +msgstr "" + +#: systemv/cupstestppd.c:2185 +#, c-format +msgid " %s Default choices conflicting." +msgstr "" + +#: systemv/cupstestppd.c:1783 +#, c-format +msgid " %s Empty cupsUIConstraints %s" +msgstr "" + +#: systemv/cupstestppd.c:3499 systemv/cupstestppd.c:3539 +#, c-format +msgid " %s Missing \"%s\" translation string for option %s, choice %s." +msgstr "" + +#: systemv/cupstestppd.c:3407 +#, c-format +msgid " %s Missing \"%s\" translation string for option %s." +msgstr "" + +#: systemv/cupstestppd.c:2398 systemv/cupstestppd.c:2494 +#: systemv/cupstestppd.c:2580 systemv/cupstestppd.c:2638 +#: systemv/cupstestppd.c:2693 systemv/cupstestppd.c:2748 +#: systemv/cupstestppd.c:2803 systemv/cupstestppd.c:2855 +#: systemv/cupstestppd.c:2978 +#, c-format +msgid " %s Missing %s file \"%s\"." +msgstr "" + +#: systemv/cupstestppd.c:3101 +#, c-format +msgid "" +" %s Missing REQUIRED PageRegion option.\n" +" REF: Page 100, section 5.14." +msgstr "" + +#: systemv/cupstestppd.c:3086 +#, c-format +msgid "" +" %s Missing REQUIRED PageSize option.\n" +" REF: Page 99, section 5.14." +msgstr "" + +#: systemv/cupstestppd.c:1993 systemv/cupstestppd.c:2034 +#, c-format +msgid " %s Missing choice *%s %s in UIConstraints \"*%s %s *%s %s\"." +msgstr "" + +#: systemv/cupstestppd.c:1888 +#, c-format +msgid " %s Missing choice *%s %s in cupsUIConstraints %s: \"%s\"" +msgstr "" + +#: systemv/cupstestppd.c:1820 +#, c-format +msgid " %s Missing cupsUIResolver %s" +msgstr "" + +#: systemv/cupstestppd.c:1979 systemv/cupstestppd.c:2020 +#, c-format +msgid " %s Missing option %s in UIConstraints \"*%s %s *%s %s\"." +msgstr "" + +#: systemv/cupstestppd.c:1872 +#, c-format +msgid " %s Missing option %s in cupsUIConstraints %s: \"%s\"" +msgstr "" + +#: systemv/cupstestppd.c:3593 +#, c-format +msgid " %s No base translation \"%s\" is included in file." +msgstr "" + +#: systemv/cupstestppd.c:2274 +#, c-format +msgid "" +" %s REQUIRED %s does not define choice None.\n" +" REF: Page 122, section 5.17" +msgstr "" + +#: systemv/cupstestppd.c:3159 systemv/cupstestppd.c:3173 +#, c-format +msgid " %s Size \"%s\" defined for %s but not for %s." +msgstr "" + +#: systemv/cupstestppd.c:3139 +#, c-format +msgid " %s Size \"%s\" has unexpected dimensions (%gx%g)." +msgstr "" + +#: systemv/cupstestppd.c:3312 +#, c-format +msgid " %s Size \"%s\" should be \"%s\"." +msgstr "" + +#: systemv/cupstestppd.c:3273 +#, c-format +msgid " %s Size \"%s\" should be the Adobe standard name \"%s\"." +msgstr "" + +#: systemv/cupstestppd.c:3021 +#, c-format +msgid " %s cupsICCProfile %s hash value collides with %s." +msgstr "" + +#: systemv/cupstestppd.c:1943 +#, c-format +msgid " %s cupsUIResolver %s causes a loop." +msgstr "" + +#: systemv/cupstestppd.c:1925 +#, c-format +msgid " %s cupsUIResolver %s does not list at least two different options." +msgstr "" + +#: systemv/cupstestppd.c:2143 +#, c-format +msgid " **FAIL** %s choice names %s and %s differ only by case." +msgstr "" + +#: systemv/cupstestppd.c:1148 +#, c-format +msgid "" +" **FAIL** %s must be 1284DeviceID\n" +" REF: Page 72, section 5.5" +msgstr "" + +#: systemv/cupstestppd.c:562 +#, c-format +msgid "" +" **FAIL** BAD Default%s %s\n" +" REF: Page 40, section 4.5." +msgstr "" + +#: systemv/cupstestppd.c:496 +#, c-format +msgid "" +" **FAIL** BAD DefaultImageableArea %s\n" +" REF: Page 102, section 5.15." +msgstr "" + +#: systemv/cupstestppd.c:532 +#, c-format +msgid "" +" **FAIL** BAD DefaultPaperDimension %s\n" +" REF: Page 103, section 5.15." +msgstr "" + +#: systemv/cupstestppd.c:1005 +msgid "" +" **FAIL** BAD JobPatchFile attribute in file\n" +" REF: Page 24, section 3.4." +msgstr "" + +#: systemv/cupstestppd.c:725 +msgid "" +" **FAIL** BAD Manufacturer (should be \"HP\")\n" +" REF: Page 211, table D.1." +msgstr "" + +#: systemv/cupstestppd.c:741 +msgid "" +" **FAIL** BAD Manufacturer (should be \"Oki\")\n" +" REF: Page 211, table D.1." +msgstr "" + +#: systemv/cupstestppd.c:780 +#, c-format +msgid "" +" **FAIL** BAD ModelName - \"%c\" not allowed in string.\n" +" REF: Pages 59-60, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:936 +msgid "" +" **FAIL** BAD PSVersion - not \"(string) int\".\n" +" REF: Pages 62-64, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:897 +msgid "" +" **FAIL** BAD Product - not \"(string)\".\n" +" REF: Page 62, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:971 +msgid "" +" **FAIL** BAD ShortNickName - longer than 31 chars.\n" +" REF: Pages 64-65, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:1129 +#, c-format +msgid "" +" **FAIL** Bad %s choice %s\n" +" REF: Page 84, section 5.9" +msgstr "" + +#: systemv/cupstestppd.c:605 +#, c-format +msgid "" +" **FAIL** Bad FileVersion \"%s\"\n" +" REF: Page 56, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:649 +#, c-format +msgid "" +" **FAIL** Bad FormatVersion \"%s\"\n" +" REF: Page 56, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:1193 +#, c-format +msgid " **FAIL** Bad LanguageEncoding %s - must be ISOLatin1." +msgstr "" + +#: systemv/cupstestppd.c:1207 +#, c-format +msgid " **FAIL** Bad LanguageVersion %s - must be English." +msgstr "" + +#: systemv/cupstestppd.c:3734 systemv/cupstestppd.c:3756 +#, c-format +msgid " **FAIL** Default option code cannot be interpreted: %s" +msgstr "" + +#: systemv/cupstestppd.c:1266 +#, c-format +msgid " **FAIL** Default translation string for option %s choice %s contains 8-bit characters." +msgstr "" + +#: systemv/cupstestppd.c:1239 +#, c-format +msgid " **FAIL** Default translation string for option %s contains 8-bit characters." +msgstr "" + +#: systemv/cupstestppd.c:2081 +#, c-format +msgid " **FAIL** Group names %s and %s differ only by case." +msgstr "" + +#: systemv/cupstestppd.c:2126 +#, c-format +msgid " **FAIL** Multiple occurrences of %s choice name %s." +msgstr "" + +#: systemv/cupstestppd.c:2103 +#, c-format +msgid " **FAIL** Option names %s and %s differ only by case." +msgstr "" + +#: systemv/cupstestppd.c:582 +#, c-format +msgid "" +" **FAIL** REQUIRED Default%s\n" +" REF: Page 40, section 4.5." +msgstr "" + +#: systemv/cupstestppd.c:481 +msgid "" +" **FAIL** REQUIRED DefaultImageableArea\n" +" REF: Page 102, section 5.15." +msgstr "" + +#: systemv/cupstestppd.c:517 +msgid "" +" **FAIL** REQUIRED DefaultPaperDimension\n" +" REF: Page 103, section 5.15." +msgstr "" + +#: systemv/cupstestppd.c:623 +msgid "" +" **FAIL** REQUIRED FileVersion\n" +" REF: Page 56, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:667 +msgid "" +" **FAIL** REQUIRED FormatVersion\n" +" REF: Page 56, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:1056 +#, c-format +msgid "" +" **FAIL** REQUIRED ImageableArea for PageSize %s\n" +" REF: Page 41, section 5.\n" +" REF: Page 102, section 5.15." +msgstr "" + +#: systemv/cupstestppd.c:687 +msgid "" +" **FAIL** REQUIRED LanguageEncoding\n" +" REF: Pages 56-57, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:707 +msgid "" +" **FAIL** REQUIRED LanguageVersion\n" +" REF: Pages 57-58, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:759 +msgid "" +" **FAIL** REQUIRED Manufacturer\n" +" REF: Pages 58-59, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:799 +msgid "" +" **FAIL** REQUIRED ModelName\n" +" REF: Pages 59-60, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:819 +msgid "" +" **FAIL** REQUIRED NickName\n" +" REF: Page 60, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:879 +msgid "" +" **FAIL** REQUIRED PCFileName\n" +" REF: Pages 61-62, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:954 +msgid "" +" **FAIL** REQUIRED PSVersion\n" +" REF: Pages 62-64, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:859 +msgid "" +" **FAIL** REQUIRED PageRegion\n" +" REF: Page 100, section 5.14." +msgstr "" + +#: systemv/cupstestppd.c:1025 +msgid "" +" **FAIL** REQUIRED PageSize\n" +" REF: Page 41, section 5.\n" +" REF: Page 99, section 5.14." +msgstr "" + +#: systemv/cupstestppd.c:839 +msgid "" +" **FAIL** REQUIRED PageSize\n" +" REF: Pages 99-100, section 5.14." +msgstr "" + +#: systemv/cupstestppd.c:1078 +#, c-format +msgid "" +" **FAIL** REQUIRED PaperDimension for PageSize %s\n" +" REF: Page 41, section 5.\n" +" REF: Page 103, section 5.15." +msgstr "" + +#: systemv/cupstestppd.c:914 +msgid "" +" **FAIL** REQUIRED Product\n" +" REF: Page 62, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:989 +msgid "" +" **FAIL** REQUIRED ShortNickName\n" +" REF: Page 64-65, section 5.3." +msgstr "" + +#: systemv/cupstestppd.c:335 +#, c-format +msgid " **FAIL** Unable to open PPD file - %s" +msgstr "" + +#: systemv/cupstestppd.c:347 +#, c-format +msgid " **FAIL** Unable to open PPD file - %s on line %d." +msgstr "" + +#: systemv/cupstestppd.c:1478 +#, c-format +msgid " %d ERRORS FOUND" +msgstr "" + +#: systemv/cupstestdsc.c:431 +msgid " -h Show program usage" +msgstr "" + +#: systemv/cupstestdsc.c:234 systemv/cupstestdsc.c:276 +#, c-format +msgid "" +" Bad %%%%BoundingBox: on line %d.\n" +" REF: Page 39, %%%%BoundingBox:" +msgstr "" + +#: systemv/cupstestdsc.c:305 +#, c-format +msgid "" +" Bad %%%%Page: on line %d.\n" +" REF: Page 53, %%%%Page:" +msgstr "" + +#: systemv/cupstestdsc.c:218 systemv/cupstestdsc.c:258 +#, c-format +msgid "" +" Bad %%%%Pages: on line %d.\n" +" REF: Page 43, %%%%Pages:" +msgstr "" + +#: systemv/cupstestdsc.c:176 +#, c-format +msgid "" +" Line %d is longer than 255 characters (%d).\n" +" REF: Page 25, Line Length" +msgstr "" + +#: systemv/cupstestdsc.c:192 +msgid "" +" Missing %!PS-Adobe-3.0 on first line.\n" +" REF: Page 17, 3.1 Conforming Documents" +msgstr "" + +#: systemv/cupstestdsc.c:362 +#, c-format +msgid " Missing %%EndComments comment. REF: Page 41, %%EndComments" +msgstr "" + +#: systemv/cupstestdsc.c:342 +#, c-format +msgid "" +" Missing or bad %%BoundingBox: comment.\n" +" REF: Page 39, %%BoundingBox:" +msgstr "" + +#: systemv/cupstestdsc.c:372 +#, c-format +msgid "" +" Missing or bad %%Page: comments.\n" +" REF: Page 53, %%Page:" +msgstr "" + +#: systemv/cupstestdsc.c:352 +#, c-format +msgid "" +" Missing or bad %%Pages: comment.\n" +" REF: Page 43, %%Pages:" +msgstr "" + +#: systemv/cupstestppd.c:1480 +msgid " NO ERRORS FOUND" +msgstr "" + +#: systemv/cupstestdsc.c:395 +#, c-format +msgid " Saw %d lines that exceeded 255 characters." +msgstr "" + +#: systemv/cupstestdsc.c:390 +#, c-format +msgid " Too many %%BeginDocument comments." +msgstr "" + +#: systemv/cupstestdsc.c:382 +#, c-format +msgid " Too many %%EndDocument comments." +msgstr "" + +#: systemv/cupstestdsc.c:402 +msgid " Warning: file contains binary data." +msgstr "" + +#: systemv/cupstestdsc.c:410 +#, c-format +msgid " Warning: no %%EndComments comment in file." +msgstr "" + +#: systemv/cupstestdsc.c:406 +#, c-format +msgid " Warning: obsolete DSC version %.1f in file." +msgstr "" + +#: systemv/cupsctl.c:210 +msgid " --[no-]debug-logging Turn debug logging on/off." +msgstr "" + +#: systemv/cupsctl.c:212 +msgid " --[no-]remote-admin Turn remote administration on/off." +msgstr "" + +#: systemv/cupsctl.c:214 +msgid " --[no-]remote-any Allow/prevent access from the Internet." +msgstr "" + +#: systemv/cupsctl.c:216 +msgid " --[no-]share-printers Turn printer sharing on/off." +msgstr "" + +#: systemv/cupsctl.c:218 +msgid " --[no-]user-cancel-any Allow/prevent users to cancel any job." +msgstr "" + +#: ppdc/ppdc.cxx:456 +msgid " --cr End lines with CR (Mac OS 9)." +msgstr "" + +#: ppdc/ppdc.cxx:458 +msgid " --crlf End lines with CR + LF (Windows)." +msgstr "" + +#: ppdc/ppdc.cxx:460 +msgid " --lf End lines with LF (UNIX/Linux/Mac OS X)." +msgstr "" + +#: test/ipptool.c:4382 +msgid " -4 Connect using IPv4." +msgstr "" + +#: test/ipptool.c:4383 +msgid " -6 Connect using IPv6." +msgstr "" + +#: test/ipptool.c:4384 +msgid " -C Send requests using chunking (default)." +msgstr "" + +#: scheduler/cupsfilter.c:1439 scheduler/cupsfilter.c:1466 +msgid " -D Remove the input file when finished." +msgstr "" + +#: ppdc/ppdc.cxx:438 ppdc/ppdhtml.cxx:175 ppdc/ppdpo.cxx:255 +msgid " -D name=value Set named variable to value." +msgstr "" + +#: systemv/cupsctl.c:205 +msgid " -E Enable encryption." +msgstr "" + +#: systemv/cupsaddsmb.c:285 +msgid " -E Encrypt the connection to the server." +msgstr "" + +#: test/ipptool.c:4386 +msgid " -E Test with TLS encryption." +msgstr "" + +#: scheduler/main.c:1977 +msgid " -F Run in the foreground but detach from console." +msgstr "" + +#: systemv/cupsaddsmb.c:287 +msgid " -H samba-server Use the named SAMBA server." +msgstr "" + +#: test/ipptool.c:4388 +msgid " -I Ignore errors." +msgstr "" + +#: ppdc/ppdc.cxx:440 ppdc/ppdhtml.cxx:177 ppdc/ppdi.cxx:131 ppdc/ppdpo.cxx:257 +msgid " -I include-dir Add include directory to search path." +msgstr "" + +#: systemv/cupstestppd.c:3779 +msgid " -I {filename,filters,none,profiles}" +msgstr "" + +#: scheduler/cupsfilter.c:1468 +msgid " -J title Set title." +msgstr "" + +#: test/ipptool.c:4389 +msgid " -L Send requests using content-length." +msgstr "" + +#: scheduler/cupsfilter.c:1441 scheduler/cupsfilter.c:1469 +msgid " -P filename.ppd Set PPD file." +msgstr "" + +#: systemv/cupstestppd.c:3781 +msgid " -R root-directory Set alternate root." +msgstr "" + +#: test/ipptool.c:4391 +msgid " -S Test with SSL encryption." +msgstr "" + +#: test/ipptool.c:4393 +msgid " -T Set the receive/send timeout in seconds." +msgstr "" + +#: systemv/cupsaddsmb.c:289 +msgid " -U samba-user Authenticate using the named SAMBA user." +msgstr "" + +#: scheduler/cupsfilter.c:1442 scheduler/cupsfilter.c:1470 +msgid " -U username Set username for job." +msgstr "" + +#: systemv/cupsctl.c:206 +msgid " -U username Specify username." +msgstr "" + +#: test/ipptool.c:4395 +msgid " -V version Set default IPP version." +msgstr "" + +#: systemv/cupstestppd.c:3782 +msgid " -W {all,none,constraints,defaults,duplex,filters,profiles,sizes,translations}" +msgstr "" + +#: test/ipptool.c:4397 +msgid " -X Produce XML plist instead of plain text." +msgstr "" + +#: systemv/cupsaddsmb.c:291 +msgid " -a Export all printers." +msgstr "" + +#: scheduler/cupsfilter.c:1471 +msgid " -a 'name=value ...' Set option(s)." +msgstr "" + +#: ppdc/ppdc.cxx:442 +msgid " -c catalog.po Load the specified message catalog." +msgstr "" + +#: scheduler/main.c:1974 +msgid " -c config-file Load alternate configuration file." +msgstr "" + +#: scheduler/cupsfilter.c:1472 +msgid " -c copies Set number of copies." +msgstr "" + +#: scheduler/cupsfilter.c:1443 +msgid " -c cupsd.conf Set cupsd.conf file to use." +msgstr "" + +#: test/ipptool.c:4399 +msgid " -d name=value Set named variable to value." +msgstr "" + +#: ppdc/ppdc.cxx:444 +msgid " -d output-dir Specify the output directory." +msgstr "" + +#: scheduler/cupsfilter.c:1445 scheduler/cupsfilter.c:1473 +msgid " -d printer Use the named printer." +msgstr "" + +#: scheduler/cupsfilter.c:1447 scheduler/cupsfilter.c:1475 +msgid " -e Use every filter from the PPD file." +msgstr "" + +#: scheduler/main.c:1976 +msgid " -f Run in the foreground." +msgstr "" + +#: test/ipptool.c:4401 +msgid " -f filename Set default request filename." +msgstr "" + +#: scheduler/cupsfilter.c:1477 +msgid " -f filename Set file to be converted (otherwise stdin)." +msgstr "" + +#: scheduler/main.c:1979 +msgid " -h Show this usage message." +msgstr "" + +#: systemv/cupsaddsmb.c:292 +msgid " -h cups-server Use the named CUPS server." +msgstr "" + +#: systemv/cupsctl.c:207 +msgid " -h server[:port] Specify server address." +msgstr "" + +#: scheduler/cupsfilter.c:1449 scheduler/cupsfilter.c:1479 +msgid " -i mime/type Set input MIME type (otherwise auto-typed)." +msgstr "" + +#: test/ipptool.c:4403 +msgid " -i seconds Repeat the last file with the given time interval." +msgstr "" + +#: scheduler/cupsfilter.c:1451 +msgid " -j job-id[,N] Filter file N from the specified job (default is file 1)." +msgstr "" + +#: scheduler/cupsfilter.c:1481 +msgid " -j mime/type Set output MIME type (otherwise application/pdf)." +msgstr "" + +#: scheduler/main.c:1980 +msgid " -l Run cupsd from launchd(8)." +msgstr "" + +#: ppdc/ppdc.cxx:446 +msgid " -l lang[,lang,...] Specify the output language(s) (locale)." +msgstr "" + +#: ppdc/ppdc.cxx:448 +msgid " -m Use the ModelName value as the filename." +msgstr "" + +#: scheduler/cupsfilter.c:1453 +msgid " -m mime/type Set output MIME type (otherwise application/pdf)." +msgstr "" + +#: scheduler/cupsfilter.c:1455 +msgid " -n copies Set number of copies." +msgstr "" + +#: test/ipptool.c:4405 +msgid " -n count Repeat the last file the given number of times." +msgstr "" + +#: scheduler/cupsfilter.c:1483 +msgid " -o filename Set file to be generated (otherwise stdout)." +msgstr "" + +#: ppdc/ppdi.cxx:133 +msgid " -o filename.drv Set driver information file (otherwise ppdi.drv)." +msgstr "" + +#: ppdc/ppdmerge.cxx:370 +msgid " -o filename.ppd[.gz] Set output file (otherwise stdout)." +msgstr "" + +#: scheduler/cupsfilter.c:1456 +msgid " -o name=value Set option(s)." +msgstr "" + +#: scheduler/cupsfilter.c:1457 +msgid " -p filename.ppd Set PPD file." +msgstr "" + +#: test/ipptool.c:4407 +msgid " -q Be quiet - no output except errors." +msgstr "" + +#: systemv/cupstestppd.c:3786 +msgid " -q Run silently." +msgstr "" + +#: systemv/cupstestppd.c:3787 +msgid " -r Use 'relaxed' open mode." +msgstr "" + +#: test/ipptool.c:4409 +msgid " -t Produce a test report." +msgstr "" + +#: ppdc/ppdc.cxx:450 +msgid " -t Test PPDs instead of generating them." +msgstr "" + +#: scheduler/main.c:1981 +msgid " -t Test the configuration file." +msgstr "" + +#: scheduler/cupsfilter.c:1458 +msgid " -t title Set title." +msgstr "" + +#: scheduler/cupsfilter.c:1459 scheduler/cupsfilter.c:1485 +msgid " -u Remove the PPD file when finished." +msgstr "" + +#: systemv/cupstestppd.c:3788 +msgid " -v Be slightly verbose." +msgstr "" + +#: ppdc/ppdc.cxx:452 ppdc/ppdpo.cxx:259 +msgid " -v Be verbose (more v's for more verbosity)." +msgstr "" + +#: systemv/cupsaddsmb.c:294 +msgid " -v Be verbose (show commands)." +msgstr "" + +#: test/ipptool.c:4410 +msgid " -v Show all attributes sent and received." +msgstr "" + +#: systemv/cupstestppd.c:3789 +msgid " -vv Be very verbose." +msgstr "" + +#: ppdc/ppdc.cxx:454 +msgid " -z Compress PPD files using GNU zip." +msgstr "" + +#: systemv/cupstestppd.c:333 systemv/cupstestppd.c:345 +#: systemv/cupstestppd.c:478 systemv/cupstestppd.c:493 +#: systemv/cupstestppd.c:514 systemv/cupstestppd.c:529 +#: systemv/cupstestppd.c:559 systemv/cupstestppd.c:579 +#: systemv/cupstestppd.c:602 systemv/cupstestppd.c:620 +#: systemv/cupstestppd.c:646 systemv/cupstestppd.c:664 +#: systemv/cupstestppd.c:684 systemv/cupstestppd.c:704 +#: systemv/cupstestppd.c:722 systemv/cupstestppd.c:738 +#: systemv/cupstestppd.c:756 systemv/cupstestppd.c:777 +#: systemv/cupstestppd.c:796 systemv/cupstestppd.c:816 +#: systemv/cupstestppd.c:836 systemv/cupstestppd.c:856 +#: systemv/cupstestppd.c:876 systemv/cupstestppd.c:894 +#: systemv/cupstestppd.c:911 systemv/cupstestppd.c:933 +#: systemv/cupstestppd.c:951 systemv/cupstestppd.c:968 +#: systemv/cupstestppd.c:986 systemv/cupstestppd.c:1002 +#: systemv/cupstestppd.c:1022 systemv/cupstestppd.c:1053 +#: systemv/cupstestppd.c:1075 systemv/cupstestppd.c:1126 +#: systemv/cupstestppd.c:1145 systemv/cupstestppd.c:1189 +#: systemv/cupstestppd.c:1203 systemv/cupstestppd.c:1235 +#: systemv/cupstestppd.c:1262 systemv/cupstestppd.c:1780 +#: systemv/cupstestppd.c:1799 systemv/cupstestppd.c:1817 +#: systemv/cupstestppd.c:1869 systemv/cupstestppd.c:1885 +#: systemv/cupstestppd.c:1922 systemv/cupstestppd.c:1940 +#: systemv/cupstestppd.c:1976 systemv/cupstestppd.c:1990 +#: systemv/cupstestppd.c:2017 systemv/cupstestppd.c:2031 +#: systemv/cupstestppd.c:2077 systemv/cupstestppd.c:2099 +#: systemv/cupstestppd.c:2122 systemv/cupstestppd.c:2139 +#: systemv/cupstestppd.c:2181 systemv/cupstestppd.c:2224 +#: systemv/cupstestppd.c:2271 systemv/cupstestppd.c:2295 +#: systemv/cupstestppd.c:2349 systemv/cupstestppd.c:2365 +#: systemv/cupstestppd.c:2395 systemv/cupstestppd.c:2409 +#: systemv/cupstestppd.c:2435 systemv/cupstestppd.c:2451 +#: systemv/cupstestppd.c:2491 systemv/cupstestppd.c:2505 +#: systemv/cupstestppd.c:2531 systemv/cupstestppd.c:2547 +#: systemv/cupstestppd.c:2577 systemv/cupstestppd.c:2591 +#: systemv/cupstestppd.c:2618 systemv/cupstestppd.c:2635 +#: systemv/cupstestppd.c:2649 systemv/cupstestppd.c:2673 +#: systemv/cupstestppd.c:2690 systemv/cupstestppd.c:2704 +#: systemv/cupstestppd.c:2728 systemv/cupstestppd.c:2745 +#: systemv/cupstestppd.c:2759 systemv/cupstestppd.c:2783 +#: systemv/cupstestppd.c:2800 systemv/cupstestppd.c:2814 +#: systemv/cupstestppd.c:2838 systemv/cupstestppd.c:2852 +#: systemv/cupstestppd.c:2867 systemv/cupstestppd.c:2884 +#: systemv/cupstestppd.c:2940 systemv/cupstestppd.c:2975 +#: systemv/cupstestppd.c:2989 systemv/cupstestppd.c:3017 +#: systemv/cupstestppd.c:3082 systemv/cupstestppd.c:3097 +#: systemv/cupstestppd.c:3135 systemv/cupstestppd.c:3155 +#: systemv/cupstestppd.c:3169 systemv/cupstestppd.c:3367 +#: systemv/cupstestppd.c:3403 systemv/cupstestppd.c:3417 +#: systemv/cupstestppd.c:3463 systemv/cupstestppd.c:3495 +#: systemv/cupstestppd.c:3512 systemv/cupstestppd.c:3535 +#: systemv/cupstestppd.c:3551 systemv/cupstestppd.c:3589 +#: systemv/cupstestppd.c:3730 systemv/cupstestppd.c:3752 +#: systemv/cupstestppd.c:3860 +msgid " FAIL" +msgstr "" + +#: systemv/cupstestppd.c:1286 +msgid " PASS" +msgstr "" + +#: berkeley/lpq.c:560 +#, c-format +msgid "%-6s %-10.10s %-4d %-10d %-27.27s %.0f bytes" +msgstr "" + +#: berkeley/lpq.c:565 +#, c-format +msgid "%-7s %-7.7s %-7d %-31.31s %.0f bytes" +msgstr "" + +#: systemv/lpstat.c:750 +#, c-format +msgid "%s accepting requests since %s" +msgstr "" + +#: scheduler/ipp.c:10902 +#, c-format +msgid "%s cannot be changed." +msgstr "" + +#: berkeley/lpc.c:189 +#, c-format +msgid "%s is not implemented by the CUPS version of lpc." +msgstr "" + +#: berkeley/lpq.c:651 +#, c-format +msgid "%s is not ready" +msgstr "" + +#: berkeley/lpq.c:644 +#, c-format +msgid "%s is ready" +msgstr "" + +#: berkeley/lpq.c:647 +#, c-format +msgid "%s is ready and printing" +msgstr "" + +#: filter/rastertoepson.c:985 filter/rastertohp.c:711 +#: filter/rastertolabel.c:1134 +#, c-format +msgid "%s job-id user title copies options [file]" +msgstr "" + +#: systemv/lpstat.c:754 +#, c-format +msgid "%s not accepting requests since %s -" +msgstr "" + +#: scheduler/ipp.c:712 +#, c-format +msgid "%s not supported." +msgstr "" + +#: systemv/lpstat.c:765 +#, c-format +msgid "%s/%s accepting requests since %s" +msgstr "" + +#: systemv/lpstat.c:770 +#, c-format +msgid "%s/%s not accepting requests since %s -" +msgstr "" + +#: berkeley/lpq.c:552 +#, c-format +msgid "%s: %-33.33s [job %d localhost]" +msgstr "" + +#. TRANSLATORS: Message is "subject: error" +#: cups/langprintf.c:86 scheduler/cupsfilter.c:720 systemv/lpadmin.c:805 +#: systemv/lpadmin.c:856 systemv/lpadmin.c:906 systemv/lpadmin.c:962 +#: systemv/lpadmin.c:1060 systemv/lpadmin.c:1112 systemv/lpadmin.c:1168 +#: systemv/lpadmin.c:1478 +#, c-format +msgid "%s: %s" +msgstr "" + +#: systemv/cancel.c:294 systemv/cancel.c:357 +#, c-format +msgid "%s: %s failed: %s" +msgstr "" + +#: systemv/cupsaccept.c:68 +#, c-format +msgid "%s: Don't know what to do." +msgstr "" + +#: berkeley/lpq.c:236 berkeley/lpr.c:344 systemv/lp.c:584 +#, c-format +msgid "%s: Error - %s environment variable names non-existent destination \"%s\"." +msgstr "" + +#: systemv/lp.c:231 +#, c-format +msgid "%s: Error - bad job ID." +msgstr "" + +#: systemv/lp.c:219 +#, c-format +msgid "%s: Error - cannot print files and alter jobs simultaneously." +msgstr "" + +#: systemv/lp.c:505 +#, c-format +msgid "%s: Error - cannot print from stdin if files or a job ID are provided." +msgstr "" + +#: systemv/lp.c:461 +#, c-format +msgid "%s: Error - expected character set after \"-S\" option." +msgstr "" + +#: systemv/lp.c:480 +#, c-format +msgid "%s: Error - expected content type after \"-T\" option." +msgstr "" + +#: berkeley/lpr.c:240 +#, c-format +msgid "%s: Error - expected copies after \"-#\" option." +msgstr "" + +#: systemv/lp.c:264 +#, c-format +msgid "%s: Error - expected copies after \"-n\" option." +msgstr "" + +#: berkeley/lpr.c:209 +#, c-format +msgid "%s: Error - expected destination after \"-P\" option." +msgstr "" + +#: systemv/lpstat.c:231 +#, c-format +msgid "%s: Error - expected destination after \"-b\" option." +msgstr "" + +#: systemv/lp.c:138 +#, c-format +msgid "%s: Error - expected destination after \"-d\" option." +msgstr "" + +#: systemv/lp.c:168 +#, c-format +msgid "%s: Error - expected form after \"-f\" option." +msgstr "" + +#: systemv/lp.c:391 +#, c-format +msgid "%s: Error - expected hold name after \"-H\" option." +msgstr "" + +#: berkeley/lpr.c:103 +#, c-format +msgid "%s: Error - expected hostname after \"-H\" option." +msgstr "" + +#: berkeley/lpq.c:180 berkeley/lprm.c:123 systemv/cancel.c:124 +#: systemv/cupsaccept.c:123 systemv/lp.c:189 systemv/lpstat.c:291 +#, c-format +msgid "%s: Error - expected hostname after \"-h\" option." +msgstr "" + +#: systemv/lp.c:371 +#, c-format +msgid "%s: Error - expected mode list after \"-y\" option." +msgstr "" + +#: berkeley/lpr.c:263 +#, c-format +msgid "%s: Error - expected name after \"-%c\" option." +msgstr "" + +#: berkeley/lpr.c:153 systemv/lp.c:288 +#, c-format +msgid "%s: Error - expected option=value after \"-o\" option." +msgstr "" + +#: systemv/lp.c:441 +#, c-format +msgid "%s: Error - expected page list after \"-P\" option." +msgstr "" + +#: systemv/lp.c:308 +#, c-format +msgid "%s: Error - expected priority after \"-%c\" option." +msgstr "" + +#: systemv/cupsaccept.c:141 +#, c-format +msgid "%s: Error - expected reason text after \"-r\" option." +msgstr "" + +#: systemv/lp.c:354 +#, c-format +msgid "%s: Error - expected title after \"-t\" option." +msgstr "" + +#: berkeley/lpq.c:111 berkeley/lpr.c:84 berkeley/lprm.c:104 +#: systemv/cancel.c:94 systemv/cupsaccept.c:101 systemv/lp.c:116 +#: systemv/lpadmin.c:438 systemv/lpstat.c:137 +#, c-format +msgid "%s: Error - expected username after \"-U\" option." +msgstr "" + +#: systemv/cancel.c:145 +#, c-format +msgid "%s: Error - expected username after \"-u\" option." +msgstr "" + +#: berkeley/lpr.c:125 +#, c-format +msgid "%s: Error - expected value after \"-%c\" option." +msgstr "" + +#: systemv/lpstat.c:157 systemv/lpstat.c:171 +#, c-format +msgid "%s: Error - need \"completed\", \"not-completed\", or \"all\" after \"-W\" option." +msgstr "" + +#: berkeley/lpq.c:241 berkeley/lpr.c:349 systemv/lp.c:589 +#, c-format +msgid "%s: Error - no default destination available." +msgstr "" + +#: systemv/lp.c:330 +#, c-format +msgid "%s: Error - priority must be between 1 and 100." +msgstr "" + +#: berkeley/lpr.c:352 systemv/lp.c:592 +#, c-format +msgid "%s: Error - scheduler not responding." +msgstr "" + +#: berkeley/lpr.c:305 systemv/lp.c:537 +#, c-format +msgid "%s: Error - too many files - \"%s\"." +msgstr "" + +#: berkeley/lpr.c:287 systemv/lp.c:520 +#, c-format +msgid "%s: Error - unable to access \"%s\" - %s" +msgstr "" + +#: berkeley/lpr.c:396 systemv/lp.c:624 +#, c-format +msgid "%s: Error - unable to queue from stdin - %s." +msgstr "" + +#: berkeley/lprm.c:87 berkeley/lprm.c:172 systemv/cancel.c:214 +#, c-format +msgid "%s: Error - unknown destination \"%s\"." +msgstr "" + +#: berkeley/lpq.c:150 +#, c-format +msgid "%s: Error - unknown destination \"%s/%s\"." +msgstr "" + +#: berkeley/lpr.c:274 berkeley/lprm.c:139 systemv/cancel.c:156 +#: systemv/cupsaccept.c:164 systemv/lp.c:496 systemv/lpstat.c:452 +#, c-format +msgid "%s: Error - unknown option \"%c\"." +msgstr "" + +#: systemv/cupsaccept.c:157 +#, c-format +msgid "%s: Error - unknown option \"%s\"." +msgstr "" + +#: systemv/lp.c:208 +#, c-format +msgid "%s: Expected job ID after \"-i\" option." +msgstr "" + +#: systemv/lpstat.c:504 systemv/lpstat.c:543 +#, c-format +msgid "%s: Invalid destination name in list \"%s\"." +msgstr "" + +#: scheduler/cupsfilter.c:573 +#, c-format +msgid "%s: Invalid filter string \"%s\"." +msgstr "" + +#: systemv/lp.c:418 +#, c-format +msgid "%s: Need job ID (\"-i jobid\") before \"-H restart\"." +msgstr "" + +#: scheduler/cupsfilter.c:464 +#, c-format +msgid "%s: No filter to convert from %s/%s to %s/%s." +msgstr "" + +#: systemv/cupsaccept.c:198 +#, c-format +msgid "%s: Operation failed: %s" +msgstr "" + +#: berkeley/lpq.c:97 berkeley/lpr.c:70 berkeley/lprm.c:67 systemv/cancel.c:81 +#: systemv/cupsaccept.c:88 systemv/cupsaddsmb.c:86 systemv/lp.c:102 +#: systemv/lpadmin.c:239 systemv/lpinfo.c:88 systemv/lpmove.c:73 +#: systemv/lpstat.c:102 test/ipptool.c:294 test/ipptool.c:311 +#, c-format +msgid "%s: Sorry, no encryption support." +msgstr "" + +#: berkeley/lpq.c:295 scheduler/cupsfilter.c:1227 systemv/cancel.c:237 +#: systemv/cupsaddsmb.c:144 systemv/cupsaddsmb.c:171 +#, c-format +msgid "%s: Unable to connect to server." +msgstr "" + +#: systemv/cancel.c:317 +#, c-format +msgid "%s: Unable to contact server." +msgstr "" + +#: scheduler/cupsfilter.c:430 +#, c-format +msgid "%s: Unable to determine MIME type of \"%s\"." +msgstr "" + +#: ppdc/ppdmerge.cxx:96 +#, c-format +msgid "%s: Unable to open %s: %s" +msgstr "" + +#: scheduler/cupsfilter.c:668 ppdc/ppdmerge.cxx:112 +#, c-format +msgid "%s: Unable to open PPD file: %s on line %d." +msgstr "" + +#: scheduler/cupsfilter.c:398 +#, c-format +msgid "%s: Unable to read MIME database from \"%s\" or \"%s\"." +msgstr "" + +#: berkeley/lpq.c:153 systemv/lpstat.c:558 +#, c-format +msgid "%s: Unknown destination \"%s\"." +msgstr "" + +#: scheduler/cupsfilter.c:441 +#, c-format +msgid "%s: Unknown destination MIME type %s/%s." +msgstr "" + +#: scheduler/cupsfilter.c:1433 +#, c-format +msgid "%s: Unknown option \"%c\"." +msgstr "" + +#: scheduler/cupsfilter.c:422 +#, c-format +msgid "%s: Unknown source MIME type %s/%s." +msgstr "" + +#: berkeley/lpr.c:139 +#, c-format +msgid "%s: Warning - \"%c\" format modifier not supported - output may not be correct." +msgstr "" + +#: systemv/lp.c:468 +#, c-format +msgid "%s: Warning - character set option ignored." +msgstr "" + +#: systemv/lp.c:487 +#, c-format +msgid "%s: Warning - content type option ignored." +msgstr "" + +#: systemv/lp.c:175 +#, c-format +msgid "%s: Warning - form option ignored." +msgstr "" + +#: systemv/lp.c:378 +#, c-format +msgid "%s: Warning - mode option ignored." +msgstr "" + +#: ppdc/sample.c:319 +msgid "-1" +msgstr "" + +#: ppdc/sample.c:310 +msgid "-10" +msgstr "" + +#: ppdc/sample.c:402 +msgid "-100" +msgstr "" + +#: ppdc/sample.c:401 +msgid "-105" +msgstr "" + +#: ppdc/sample.c:309 +msgid "-11" +msgstr "" + +#: ppdc/sample.c:400 +msgid "-110" +msgstr "" + +#: ppdc/sample.c:399 +msgid "-115" +msgstr "" + +#: ppdc/sample.c:308 +msgid "-12" +msgstr "" + +#: ppdc/sample.c:398 +msgid "-120" +msgstr "" + +#: ppdc/sample.c:307 +msgid "-13" +msgstr "" + +#: ppdc/sample.c:306 +msgid "-14" +msgstr "" + +#: ppdc/sample.c:305 +msgid "-15" +msgstr "" + +#: ppdc/sample.c:318 +msgid "-2" +msgstr "" + +#: ppdc/sample.c:418 +msgid "-20" +msgstr "" + +#: ppdc/sample.c:417 +msgid "-25" +msgstr "" + +#: ppdc/sample.c:317 +msgid "-3" +msgstr "" + +#: ppdc/sample.c:416 +msgid "-30" +msgstr "" + +#: ppdc/sample.c:415 +msgid "-35" +msgstr "" + +#: ppdc/sample.c:316 +msgid "-4" +msgstr "" + +#: ppdc/sample.c:414 +msgid "-40" +msgstr "" + +#: ppdc/sample.c:413 +msgid "-45" +msgstr "" + +#: ppdc/sample.c:315 +msgid "-5" +msgstr "" + +#: ppdc/sample.c:412 +msgid "-50" +msgstr "" + +#: ppdc/sample.c:411 +msgid "-55" +msgstr "" + +#: ppdc/sample.c:314 +msgid "-6" +msgstr "" + +#: ppdc/sample.c:410 +msgid "-60" +msgstr "" + +#: ppdc/sample.c:409 +msgid "-65" +msgstr "" + +#: ppdc/sample.c:313 +msgid "-7" +msgstr "" + +#: ppdc/sample.c:408 +msgid "-70" +msgstr "" + +#: ppdc/sample.c:407 +msgid "-75" +msgstr "" + +#: ppdc/sample.c:312 +msgid "-8" +msgstr "" + +#: ppdc/sample.c:406 +msgid "-80" +msgstr "" + +#: ppdc/sample.c:405 +msgid "-85" +msgstr "" + +#: ppdc/sample.c:311 +msgid "-9" +msgstr "" + +#: ppdc/sample.c:404 +msgid "-90" +msgstr "" + +#: ppdc/sample.c:403 +msgid "-95" +msgstr "" + +#: ppdc/sample.c:320 +msgid "0" +msgstr "" + +#: ppdc/sample.c:321 +msgid "1" +msgstr "" + +#: ppdc/sample.c:393 +msgid "1 inch/sec." +msgstr "" + +#: ppdc/sample.c:181 +msgid "1.25x0.25\"" +msgstr "" + +#: ppdc/sample.c:182 +msgid "1.25x2.25\"" +msgstr "" + +#: ppdc/sample.c:441 +msgid "1.5 inch/sec." +msgstr "" + +#: ppdc/sample.c:183 +msgid "1.50x0.25\"" +msgstr "" + +#: ppdc/sample.c:184 +msgid "1.50x0.50\"" +msgstr "" + +#: ppdc/sample.c:185 +msgid "1.50x1.00\"" +msgstr "" + +#: ppdc/sample.c:186 +msgid "1.50x2.00\"" +msgstr "" + +#: ppdc/sample.c:330 +msgid "10" +msgstr "" + +#: ppdc/sample.c:452 +msgid "10 inches/sec." +msgstr "" + +#: ppdc/sample.c:6 +msgid "10 x 11" +msgstr "" + +#: ppdc/sample.c:7 +msgid "10 x 13" +msgstr "" + +#: ppdc/sample.c:8 +msgid "10 x 14" +msgstr "" + +#: ppdc/sample.c:432 +msgid "100" +msgstr "" + +#: ppdc/sample.c:343 +msgid "100 mm/sec." +msgstr "" + +#: ppdc/sample.c:433 +msgid "105" +msgstr "" + +#: ppdc/sample.c:331 +msgid "11" +msgstr "" + +#: ppdc/sample.c:453 +msgid "11 inches/sec." +msgstr "" + +#: ppdc/sample.c:434 +msgid "110" +msgstr "" + +#: ppdc/sample.c:435 +msgid "115" +msgstr "" + +#: ppdc/sample.c:332 +msgid "12" +msgstr "" + +#: ppdc/sample.c:454 +msgid "12 inches/sec." +msgstr "" + +#: ppdc/sample.c:9 +msgid "12 x 11" +msgstr "" + +#: ppdc/sample.c:436 +msgid "120" +msgstr "" + +#: ppdc/sample.c:344 +msgid "120 mm/sec." +msgstr "" + +#: ppdc/sample.c:252 +msgid "120x60dpi" +msgstr "" + +#: ppdc/sample.c:258 +msgid "120x72dpi" +msgstr "" + +#: ppdc/sample.c:333 +msgid "13" +msgstr "" + +#: ppdc/sample.c:241 +msgid "136dpi" +msgstr "" + +#: ppdc/sample.c:334 +msgid "14" +msgstr "" + +#: ppdc/sample.c:335 +msgid "15" +msgstr "" + +#: ppdc/sample.c:337 +msgid "15 mm/sec." +msgstr "" + +#: ppdc/sample.c:10 +msgid "15 x 11" +msgstr "" + +#: ppdc/sample.c:345 +msgid "150 mm/sec." +msgstr "" + +#: ppdc/sample.c:292 +msgid "150dpi" +msgstr "" + +#: ppdc/sample.c:377 +msgid "16" +msgstr "" + +#: ppdc/sample.c:378 +msgid "17" +msgstr "" + +#: ppdc/sample.c:379 +msgid "18" +msgstr "" + +#: ppdc/sample.c:253 +msgid "180dpi" +msgstr "" + +#: ppdc/sample.c:380 +msgid "19" +msgstr "" + +#: ppdc/sample.c:322 +msgid "2" +msgstr "" + +#: ppdc/sample.c:394 +msgid "2 inches/sec." +msgstr "" + +#: ppdc/sample.c:279 +msgid "2-Sided Printing" +msgstr "" + +#: ppdc/sample.c:187 +msgid "2.00x0.37\"" +msgstr "" + +#: ppdc/sample.c:188 +msgid "2.00x0.50\"" +msgstr "" + +#: ppdc/sample.c:189 +msgid "2.00x1.00\"" +msgstr "" + +#: ppdc/sample.c:190 +msgid "2.00x1.25\"" +msgstr "" + +#: ppdc/sample.c:191 +msgid "2.00x2.00\"" +msgstr "" + +#: ppdc/sample.c:192 +msgid "2.00x3.00\"" +msgstr "" + +#: ppdc/sample.c:193 +msgid "2.00x4.00\"" +msgstr "" + +#: ppdc/sample.c:194 +msgid "2.00x5.50\"" +msgstr "" + +#: ppdc/sample.c:195 +msgid "2.25x0.50\"" +msgstr "" + +#: ppdc/sample.c:196 +msgid "2.25x1.25\"" +msgstr "" + +#: ppdc/sample.c:197 +msgid "2.25x4.00\"" +msgstr "" + +#: ppdc/sample.c:198 +msgid "2.25x5.50\"" +msgstr "" + +#: ppdc/sample.c:199 +msgid "2.38x5.50\"" +msgstr "" + +#: ppdc/sample.c:442 +msgid "2.5 inches/sec." +msgstr "" + +#: ppdc/sample.c:200 +msgid "2.50x1.00\"" +msgstr "" + +#: ppdc/sample.c:201 +msgid "2.50x2.00\"" +msgstr "" + +#: ppdc/sample.c:202 +msgid "2.75x1.25\"" +msgstr "" + +#: ppdc/sample.c:203 +msgid "2.9 x 1\"" +msgstr "" + +#: ppdc/sample.c:381 +msgid "20" +msgstr "" + +#: ppdc/sample.c:338 +msgid "20 mm/sec." +msgstr "" + +#: ppdc/sample.c:346 +msgid "200 mm/sec." +msgstr "" + +#: ppdc/sample.c:242 +msgid "203dpi" +msgstr "" + +#: ppdc/sample.c:382 +msgid "21" +msgstr "" + +#: ppdc/sample.c:383 +msgid "22" +msgstr "" + +#: ppdc/sample.c:384 +msgid "23" +msgstr "" + +#: ppdc/sample.c:385 +msgid "24" +msgstr "" + +#: ppdc/sample.c:250 +msgid "24-Pin Series" +msgstr "" + +#: ppdc/sample.c:259 +msgid "240x72dpi" +msgstr "" + +#: ppdc/sample.c:386 +msgid "25" +msgstr "" + +#: ppdc/sample.c:347 +msgid "250 mm/sec." +msgstr "" + +#: ppdc/sample.c:387 +msgid "26" +msgstr "" + +#: ppdc/sample.c:388 +msgid "27" +msgstr "" + +#: ppdc/sample.c:389 +msgid "28" +msgstr "" + +#: ppdc/sample.c:390 +msgid "29" +msgstr "" + +#: ppdc/sample.c:323 +msgid "3" +msgstr "" + +#: ppdc/sample.c:395 +msgid "3 inches/sec." +msgstr "" + +#: ppdc/sample.c:3 +msgid "3 x 5" +msgstr "" + +#: ppdc/sample.c:204 +msgid "3.00x1.00\"" +msgstr "" + +#: ppdc/sample.c:205 +msgid "3.00x1.25\"" +msgstr "" + +#: ppdc/sample.c:206 +msgid "3.00x2.00\"" +msgstr "" + +#: ppdc/sample.c:207 +msgid "3.00x3.00\"" +msgstr "" + +#: ppdc/sample.c:208 +msgid "3.00x5.00\"" +msgstr "" + +#: ppdc/sample.c:209 +msgid "3.25x2.00\"" +msgstr "" + +#: ppdc/sample.c:210 +msgid "3.25x5.00\"" +msgstr "" + +#: ppdc/sample.c:211 +msgid "3.25x5.50\"" +msgstr "" + +#: ppdc/sample.c:212 +msgid "3.25x5.83\"" +msgstr "" + +#: ppdc/sample.c:213 +msgid "3.25x7.83\"" +msgstr "" + +#: ppdc/sample.c:4 +msgid "3.5 x 5" +msgstr "" + +#: ppdc/sample.c:171 +msgid "3.5\" Disk" +msgstr "" + +#: ppdc/sample.c:180 +msgid "3.5\" Disk - 2 1/8 x 2 3/4\"" +msgstr "" + +#: ppdc/sample.c:214 +msgid "3.50x1.00\"" +msgstr "" + +#: ppdc/sample.c:391 +msgid "30" +msgstr "" + +#: ppdc/sample.c:339 +msgid "30 mm/sec." +msgstr "" + +#: ppdc/sample.c:348 +msgid "300 mm/sec." +msgstr "" + +#: ppdc/sample.c:243 +msgid "300dpi" +msgstr "" + +#: ppdc/sample.c:419 +msgid "35" +msgstr "" + +#: ppdc/sample.c:255 +msgid "360dpi" +msgstr "" + +#: ppdc/sample.c:254 +msgid "360x180dpi" +msgstr "" + +#: ppdc/sample.c:324 +msgid "4" +msgstr "" + +#: ppdc/sample.c:396 +msgid "4 inches/sec." +msgstr "" + +#: ppdc/sample.c:215 +msgid "4.00x1.00\"" +msgstr "" + +#: ppdc/sample.c:223 +msgid "4.00x13.00\"" +msgstr "" + +#: ppdc/sample.c:216 +msgid "4.00x2.00\"" +msgstr "" + +#: ppdc/sample.c:217 +msgid "4.00x2.50\"" +msgstr "" + +#: ppdc/sample.c:218 +msgid "4.00x3.00\"" +msgstr "" + +#: ppdc/sample.c:219 +msgid "4.00x4.00\"" +msgstr "" + +#: ppdc/sample.c:220 +msgid "4.00x5.00\"" +msgstr "" + +#: ppdc/sample.c:221 +msgid "4.00x6.00\"" +msgstr "" + +#: ppdc/sample.c:222 +msgid "4.00x6.50\"" +msgstr "" + +#: ppdc/sample.c:420 +msgid "40" +msgstr "" + +#: ppdc/sample.c:340 +msgid "40 mm/sec." +msgstr "" + +#: ppdc/sample.c:421 +msgid "45" +msgstr "" + +#: ppdc/sample.c:325 +msgid "5" +msgstr "" + +#: ppdc/sample.c:446 +msgid "5 inches/sec." +msgstr "" + +#: ppdc/sample.c:5 +msgid "5 x 7" +msgstr "" + +#: ppdc/sample.c:422 +msgid "50" +msgstr "" + +#: ppdc/sample.c:423 +msgid "55" +msgstr "" + +#: ppdc/sample.c:326 +msgid "6" +msgstr "" + +#: ppdc/sample.c:447 +msgid "6 inches/sec." +msgstr "" + +#: ppdc/sample.c:224 +msgid "6.00x1.00\"" +msgstr "" + +#: ppdc/sample.c:225 +msgid "6.00x2.00\"" +msgstr "" + +#: ppdc/sample.c:226 +msgid "6.00x3.00\"" +msgstr "" + +#: ppdc/sample.c:227 +msgid "6.00x4.00\"" +msgstr "" + +#: ppdc/sample.c:228 +msgid "6.00x5.00\"" +msgstr "" + +#: ppdc/sample.c:229 +msgid "6.00x6.00\"" +msgstr "" + +#: ppdc/sample.c:230 +msgid "6.00x6.50\"" +msgstr "" + +#: ppdc/sample.c:424 +msgid "60" +msgstr "" + +#: ppdc/sample.c:341 +msgid "60 mm/sec." +msgstr "" + +#: ppdc/sample.c:270 +msgid "600dpi" +msgstr "" + +#: ppdc/sample.c:251 +msgid "60dpi" +msgstr "" + +#: ppdc/sample.c:257 +msgid "60x72dpi" +msgstr "" + +#: ppdc/sample.c:425 +msgid "65" +msgstr "" + +#: ppdc/sample.c:327 +msgid "7" +msgstr "" + +#: ppdc/sample.c:449 +msgid "7 inches/sec." +msgstr "" + +#: ppdc/sample.c:11 +msgid "7 x 9" +msgstr "" + +#: ppdc/sample.c:426 +msgid "70" +msgstr "" + +#: ppdc/sample.c:261 +msgid "720dpi" +msgstr "" + +#: ppdc/sample.c:427 +msgid "75" +msgstr "" + +#: ppdc/sample.c:328 +msgid "8" +msgstr "" + +#: ppdc/sample.c:450 +msgid "8 inches/sec." +msgstr "" + +#: ppdc/sample.c:12 +msgid "8 x 10" +msgstr "" + +#: ppdc/sample.c:231 +msgid "8.00x1.00\"" +msgstr "" + +#: ppdc/sample.c:232 +msgid "8.00x2.00\"" +msgstr "" + +#: ppdc/sample.c:233 +msgid "8.00x3.00\"" +msgstr "" + +#: ppdc/sample.c:234 +msgid "8.00x4.00\"" +msgstr "" + +#: ppdc/sample.c:235 +msgid "8.00x5.00\"" +msgstr "" + +#: ppdc/sample.c:236 +msgid "8.00x6.00\"" +msgstr "" + +#: ppdc/sample.c:237 +msgid "8.00x6.50\"" +msgstr "" + +#: ppdc/sample.c:428 +msgid "80" +msgstr "" + +#: ppdc/sample.c:342 +msgid "80 mm/sec." +msgstr "" + +#: ppdc/sample.c:429 +msgid "85" +msgstr "" + +#: ppdc/sample.c:329 +msgid "9" +msgstr "" + +#: ppdc/sample.c:451 +msgid "9 inches/sec." +msgstr "" + +#: ppdc/sample.c:13 +msgid "9 x 11" +msgstr "" + +#: ppdc/sample.c:14 +msgid "9 x 12" +msgstr "" + +#: ppdc/sample.c:256 +msgid "9-Pin Series" +msgstr "" + +#: ppdc/sample.c:430 +msgid "90" +msgstr "" + +#: ppdc/sample.c:431 +msgid "95" +msgstr "" + +#: berkeley/lpc.c:213 +msgid "?Invalid help command unknown." +msgstr "" + +#: cgi-bin/admin.c:2349 +msgid "A Samba password is required to export printer drivers" +msgstr "" + +#: cgi-bin/admin.c:2345 +msgid "A Samba username is required to export printer drivers" +msgstr "" + +#: scheduler/ipp.c:2367 +#, c-format +msgid "A class named \"%s\" already exists." +msgstr "" + +#: scheduler/ipp.c:1025 +#, c-format +msgid "A printer named \"%s\" already exists." +msgstr "" + +#: ppdc/sample.c:15 +msgid "A0" +msgstr "" + +#: ppdc/sample.c:16 +msgid "A0 Long Edge" +msgstr "" + +#: ppdc/sample.c:17 +msgid "A1" +msgstr "" + +#: ppdc/sample.c:18 +msgid "A1 Long Edge" +msgstr "" + +#: ppdc/sample.c:37 +msgid "A10" +msgstr "" + +#: ppdc/sample.c:19 +msgid "A2" +msgstr "" + +#: ppdc/sample.c:20 +msgid "A2 Long Edge" +msgstr "" + +#: ppdc/sample.c:21 +msgid "A3" +msgstr "" + +#: ppdc/sample.c:22 +msgid "A3 Long Edge" +msgstr "" + +#: ppdc/sample.c:23 +msgid "A3 Oversize" +msgstr "" + +#: ppdc/sample.c:24 +msgid "A3 Oversize Long Edge" +msgstr "" + +#: ppdc/sample.c:25 +msgid "A4" +msgstr "" + +#: ppdc/sample.c:27 +msgid "A4 Long Edge" +msgstr "" + +#: ppdc/sample.c:26 +msgid "A4 Oversize" +msgstr "" + +#: ppdc/sample.c:28 +msgid "A4 Small" +msgstr "" + +#: ppdc/sample.c:29 +msgid "A5" +msgstr "" + +#: ppdc/sample.c:31 +msgid "A5 Long Edge" +msgstr "" + +#: ppdc/sample.c:30 +msgid "A5 Oversize" +msgstr "" + +#: ppdc/sample.c:32 +msgid "A6" +msgstr "" + +#: ppdc/sample.c:33 +msgid "A6 Long Edge" +msgstr "" + +#: ppdc/sample.c:34 +msgid "A7" +msgstr "" + +#: ppdc/sample.c:35 +msgid "A8" +msgstr "" + +#: ppdc/sample.c:36 +msgid "A9" +msgstr "" + +#: ppdc/sample.c:38 +msgid "ANSI A" +msgstr "" + +#: ppdc/sample.c:39 +msgid "ANSI B" +msgstr "" + +#: ppdc/sample.c:40 +msgid "ANSI C" +msgstr "" + +#: ppdc/sample.c:41 +msgid "ANSI D" +msgstr "" + +#: ppdc/sample.c:42 +msgid "ANSI E" +msgstr "" + +#: ppdc/sample.c:47 +msgid "ARCH C" +msgstr "" + +#: ppdc/sample.c:48 +msgid "ARCH C Long Edge" +msgstr "" + +#: ppdc/sample.c:49 +msgid "ARCH D" +msgstr "" + +#: ppdc/sample.c:50 +msgid "ARCH D Long Edge" +msgstr "" + +#: ppdc/sample.c:51 +msgid "ARCH E" +msgstr "" + +#: ppdc/sample.c:52 +msgid "ARCH E Long Edge" +msgstr "" + +#: cgi-bin/classes.c:169 cgi-bin/printers.c:172 +msgid "Accept Jobs" +msgstr "" + +#: cups/http-support.c:1254 +msgid "Accepted" +msgstr "" + +#: cgi-bin/admin.c:570 +msgid "Add Class" +msgstr "" + +#: cgi-bin/admin.c:882 +msgid "Add Printer" +msgstr "" + +#: cgi-bin/admin.c:444 cgi-bin/admin.c:477 cgi-bin/admin.c:525 +#: cgi-bin/admin.c:535 +msgid "Add RSS Subscription" +msgstr "" + +#: ppdc/sample.c:163 +msgid "Address" +msgstr "" + +#: ppdc/sample.c:172 +msgid "Address - 1 1/8 x 3 1/2\"" +msgstr "" + +#: cgi-bin/admin.c:210 cgi-bin/admin.c:284 cgi-bin/admin.c:2723 +msgid "Administration" +msgstr "" + +#: ppdc/sample.c:438 +msgid "Always" +msgstr "" + +#: backend/socket.c:129 +msgid "AppSocket/HP JetDirect" +msgstr "" + +#: ppdc/sample.c:459 +msgid "Applicator" +msgstr "" + +#: scheduler/ipp.c:1100 +#, c-format +msgid "Attempt to set %s printer-state to bad value %d." +msgstr "" + +#: scheduler/ipp.c:346 +#, c-format +msgid "Attribute groups are out of order (%x < %x)." +msgstr "" + +#: ppdc/sample.c:126 +msgid "B0" +msgstr "" + +#: ppdc/sample.c:127 +msgid "B1" +msgstr "" + +#: ppdc/sample.c:137 +msgid "B10" +msgstr "" + +#: ppdc/sample.c:128 +msgid "B2" +msgstr "" + +#: ppdc/sample.c:129 +msgid "B3" +msgstr "" + +#: ppdc/sample.c:130 +msgid "B4" +msgstr "" + +#: ppdc/sample.c:131 +msgid "B5" +msgstr "" + +#: ppdc/sample.c:132 +msgid "B5 Oversize" +msgstr "" + +#: ppdc/sample.c:133 +msgid "B6" +msgstr "" + +#: ppdc/sample.c:134 +msgid "B7" +msgstr "" + +#: ppdc/sample.c:135 +msgid "B8" +msgstr "" + +#: ppdc/sample.c:136 +msgid "B9" +msgstr "" + +#: cups/dest.c:925 +msgid "Bad NULL dests pointer" +msgstr "" + +#: cups/ppd.c:345 +msgid "Bad OpenGroup" +msgstr "" + +#: cups/ppd.c:347 +msgid "Bad OpenUI/JCLOpenUI" +msgstr "" + +#: cups/ppd.c:349 +msgid "Bad OrderDependency" +msgstr "" + +#: cups/ppd-cache.c:148 cups/ppd-cache.c:193 cups/ppd-cache.c:231 +#: cups/ppd-cache.c:237 cups/ppd-cache.c:253 cups/ppd-cache.c:269 +#: cups/ppd-cache.c:278 cups/ppd-cache.c:286 cups/ppd-cache.c:303 +#: cups/ppd-cache.c:311 cups/ppd-cache.c:326 cups/ppd-cache.c:334 +#: cups/ppd-cache.c:352 cups/ppd-cache.c:364 cups/ppd-cache.c:379 +#: cups/ppd-cache.c:391 cups/ppd-cache.c:413 cups/ppd-cache.c:421 +#: cups/ppd-cache.c:439 cups/ppd-cache.c:447 cups/ppd-cache.c:462 +#: cups/ppd-cache.c:470 cups/ppd-cache.c:488 cups/ppd-cache.c:496 +#: cups/ppd-cache.c:523 cups/ppd-cache.c:567 cups/ppd-cache.c:575 +#: cups/ppd-cache.c:583 +msgid "Bad PPD cache file." +msgstr "" + +#: cups/http-support.c:1269 +msgid "Bad Request" +msgstr "" + +#: cups/snmp.c:1002 +msgid "Bad SNMP version number" +msgstr "" + +#: cups/ppd.c:350 +msgid "Bad UIConstraints" +msgstr "" + +#: scheduler/ipp.c:1401 +#, c-format +msgid "Bad copies value %d." +msgstr "" + +#: cups/ppd.c:358 +msgid "Bad custom parameter" +msgstr "" + +#: cups/http-support.c:1421 scheduler/ipp.c:2434 +#, c-format +msgid "Bad device-uri \"%s\"." +msgstr "" + +#: scheduler/ipp.c:2475 +#, c-format +msgid "Bad device-uri scheme \"%s\"." +msgstr "" + +#: scheduler/ipp.c:9113 scheduler/ipp.c:9129 scheduler/ipp.c:10320 +#: scheduler/ipp.c:11825 +#, c-format +msgid "Bad document-format \"%s\"." +msgstr "" + +#: scheduler/ipp.c:10336 +#, c-format +msgid "Bad document-format-default \"%s\"." +msgstr "" + +#: cups/util.c:927 +msgid "Bad filename buffer" +msgstr "" + +#: scheduler/ipp.c:10917 +msgid "Bad job-priority value." +msgstr "" + +#: scheduler/ipp.c:1431 +#, c-format +msgid "Bad job-sheets value \"%s\"." +msgstr "" + +#: scheduler/ipp.c:1415 +msgid "Bad job-sheets value type." +msgstr "" + +#: scheduler/ipp.c:10947 +msgid "Bad job-state value." +msgstr "" + +#: scheduler/ipp.c:3946 scheduler/ipp.c:4399 scheduler/ipp.c:6967 +#: scheduler/ipp.c:7114 scheduler/ipp.c:8547 scheduler/ipp.c:8816 +#: scheduler/ipp.c:9664 scheduler/ipp.c:9889 scheduler/ipp.c:10216 +#: scheduler/ipp.c:10810 +#, c-format +msgid "Bad job-uri \"%s\"." +msgstr "" + +#: scheduler/ipp.c:2133 scheduler/ipp.c:6509 +#, c-format +msgid "Bad notify-pull-method \"%s\"." +msgstr "" + +#: scheduler/ipp.c:2097 scheduler/ipp.c:6473 +#, c-format +msgid "Bad notify-recipient-uri \"%s\"." +msgstr "" + +#: scheduler/ipp.c:1447 +#, c-format +msgid "Bad number-up value %d." +msgstr "" + +#: cups/adminutil.c:292 +#, c-format +msgid "Bad option + choice on line %d." +msgstr "" + +#: scheduler/ipp.c:1464 +#, c-format +msgid "Bad page-ranges values %d-%d." +msgstr "" + +#: scheduler/ipp.c:2518 +#, c-format +msgid "Bad port-monitor \"%s\"." +msgstr "" + +#: scheduler/ipp.c:2579 +#, c-format +msgid "Bad printer-state value %d." +msgstr "" + +#: scheduler/ipp.c:314 +#, c-format +msgid "Bad request ID %d." +msgstr "" + +#: scheduler/ipp.c:299 +#, c-format +msgid "Bad request version number %d.%d." +msgstr "" + +#: cgi-bin/admin.c:1484 +msgid "Bad subscription ID" +msgstr "" + +#: cups/ppd.c:360 +msgid "Bad value string" +msgstr "" + +#: cgi-bin/admin.c:3268 cgi-bin/admin.c:3514 +msgid "Banners" +msgstr "" + +#: ppdc/sample.c:296 +msgid "Bond Paper" +msgstr "" + +#: backend/usb-darwin.c:1846 +#, c-format +msgid "Boolean expected for waiteof option \"%s\"." +msgstr "" + +#: filter/pstops.c:2069 +msgid "Buffer overflow detected, aborting." +msgstr "" + +#: ppdc/sample.c:263 +msgid "CMYK" +msgstr "" + +#: ppdc/sample.c:372 +msgid "CPCL Label Printer" +msgstr "" + +#: cgi-bin/admin.c:1485 cgi-bin/admin.c:1524 cgi-bin/admin.c:1534 +msgid "Cancel RSS Subscription" +msgstr "" + +#: backend/ipp.c:1805 +msgid "Canceling print job." +msgstr "" + +#: scheduler/ipp.c:2559 +msgid "Cannot share a remote Kerberized printer." +msgstr "" + +#: ppdc/sample.c:288 +msgid "Cassette" +msgstr "" + +#: cgi-bin/admin.c:1636 cgi-bin/admin.c:1778 cgi-bin/admin.c:1791 +#: cgi-bin/admin.c:1802 +msgid "Change Settings" +msgstr "" + +#: scheduler/ipp.c:2145 scheduler/ipp.c:6521 +#, c-format +msgid "Character set \"%s\" not supported." +msgstr "" + +#: cgi-bin/classes.c:195 cgi-bin/classes.c:322 +msgid "Classes" +msgstr "" + +#: cgi-bin/printers.c:182 +msgid "Clean Print Heads" +msgstr "" + +#: scheduler/ipp.c:4851 +msgid "Close-Job doesn't support the job-uri attribute." +msgstr "" + +#: ppdc/sample.c:291 +msgid "Color" +msgstr "" + +#: ppdc/sample.c:262 +msgid "Color Mode" +msgstr "" + +#: berkeley/lpc.c:204 +msgid "" +"Commands may be abbreviated. Commands are:\n" +"\n" +"exit help quit status ?" +msgstr "" + +#: cups/snmp.c:1006 +msgid "Community name uses indefinite length" +msgstr "" + +#: backend/ipp.c:765 backend/lpd.c:868 backend/socket.c:395 +msgid "Connected to printer." +msgstr "" + +#: backend/ipp.c:672 backend/lpd.c:708 backend/socket.c:314 +msgid "Connecting to printer." +msgstr "" + +#: cups/http-support.c:1242 +msgid "Continue" +msgstr "" + +#: ppdc/sample.c:374 +msgid "Continuous" +msgstr "" + +#: backend/lpd.c:1019 backend/lpd.c:1161 +msgid "Control file sent successfully." +msgstr "" + +#: backend/ipp.c:1189 backend/lpd.c:462 +msgid "Copying print data." +msgstr "" + +#: cups/http-support.c:1251 +msgid "Created" +msgstr "" + +#: cups/ppd.c:1113 cups/ppd.c:1153 cups/ppd.c:1398 cups/ppd.c:1501 +msgid "Custom" +msgstr "" + +#: ppdc/sample.c:368 +msgid "CustominCutInterval" +msgstr "" + +#: ppdc/sample.c:366 +msgid "CustominTearInterval" +msgstr "" + +#: ppdc/sample.c:352 +msgid "Cut" +msgstr "" + +#: ppdc/sample.c:460 +msgid "Cutter" +msgstr "" + +#: ppdc/sample.c:248 +msgid "Dark" +msgstr "" + +#: ppdc/sample.c:244 +msgid "Darkness" +msgstr "" + +#: backend/lpd.c:1109 +msgid "Data file sent successfully." +msgstr "" + +#: cgi-bin/admin.c:2075 cgi-bin/admin.c:2086 cgi-bin/admin.c:2131 +msgid "Delete Class" +msgstr "" + +#: cgi-bin/admin.c:2160 cgi-bin/admin.c:2171 cgi-bin/admin.c:2216 +msgid "Delete Printer" +msgstr "" + +#: ppdc/sample.c:290 +msgid "DeskJet Series" +msgstr "" + +#: scheduler/ipp.c:1367 +#, c-format +msgid "Destination \"%s\" is not accepting jobs." +msgstr "" + +#: systemv/lpinfo.c:300 +#, c-format +msgid "" +"Device: uri = %s\n" +" class = %s\n" +" info = %s\n" +" make-and-model = %s\n" +" device-id = %s\n" +" location = %s" +msgstr "" + +#: ppdc/sample.c:445 +msgid "Direct Thermal Media" +msgstr "" + +#: cups/file.c:296 +#, c-format +msgid "Directory \"%s\" contains a relative path." +msgstr "" + +#: cups/file.c:268 +#, c-format +msgid "Directory \"%s\" has insecure permissions (0%o/uid=%d/gid=%d)." +msgstr "" + +#: cups/file.c:285 +#, c-format +msgid "Directory \"%s\" is a file." +msgstr "" + +#: cups/file.c:256 +#, c-format +msgid "Directory \"%s\" not available: %s" +msgstr "" + +#: cups/file.c:241 +#, c-format +msgid "Directory \"%s\" permissions OK (0%o/uid=%d/gid=%d)." +msgstr "" + +#: ppdc/sample.c:354 +msgid "Disabled" +msgstr "" + +#: scheduler/ipp.c:7016 +#, c-format +msgid "Document #%d does not exist in job #%d." +msgstr "" + +#: ppdc/sample.c:284 +msgid "Duplexer" +msgstr "" + +#: ppdc/sample.c:238 +msgid "Dymo" +msgstr "" + +#: ppdc/sample.c:440 +msgid "EPL1 Label Printer" +msgstr "" + +#: ppdc/sample.c:443 +msgid "EPL2 Label Printer" +msgstr "" + +#: cgi-bin/admin.c:1830 cgi-bin/admin.c:1842 cgi-bin/admin.c:1896 +#: cgi-bin/admin.c:1903 cgi-bin/admin.c:1938 cgi-bin/admin.c:1951 +#: cgi-bin/admin.c:1975 cgi-bin/admin.c:2048 +msgid "Edit Configuration File" +msgstr "" + +#: cups/adminutil.c:337 +msgid "Empty PPD file." +msgstr "" + +#. TRANSLATORS: Banner/cover sheet after the print job. +#: cgi-bin/admin.c:3539 +msgid "Ending Banner" +msgstr "" + +#: ppdc/sample.c:2 +msgid "English" +msgstr "" + +#: systemv/lppasswd.c:193 +msgid "Enter old password:" +msgstr "" + +#: systemv/lppasswd.c:224 +msgid "Enter password again:" +msgstr "" + +#: systemv/lppasswd.c:212 +msgid "Enter password:" +msgstr "" + +#: scheduler/client.c:2498 +msgid "Enter your username and password or the root username and password to access this page. If you are using Kerberos authentication, make sure you have a valid Kerberos ticket." +msgstr "" + +#: ppdc/sample.c:73 +msgid "Envelope #10 " +msgstr "" + +#: ppdc/sample.c:74 +msgid "Envelope #11" +msgstr "" + +#: ppdc/sample.c:75 +msgid "Envelope #12" +msgstr "" + +#: ppdc/sample.c:76 +msgid "Envelope #14" +msgstr "" + +#: ppdc/sample.c:77 +msgid "Envelope #9" +msgstr "" + +#: ppdc/sample.c:89 +msgid "Envelope B4" +msgstr "" + +#: ppdc/sample.c:90 +msgid "Envelope B5" +msgstr "" + +#: ppdc/sample.c:91 +msgid "Envelope B6" +msgstr "" + +#: ppdc/sample.c:78 +msgid "Envelope C0" +msgstr "" + +#: ppdc/sample.c:79 +msgid "Envelope C1" +msgstr "" + +#: ppdc/sample.c:80 +msgid "Envelope C2" +msgstr "" + +#: ppdc/sample.c:81 +msgid "Envelope C3" +msgstr "" + +#: ppdc/sample.c:67 +msgid "Envelope C4" +msgstr "" + +#: ppdc/sample.c:68 +msgid "Envelope C5" +msgstr "" + +#: ppdc/sample.c:69 +msgid "Envelope C6" +msgstr "" + +#: ppdc/sample.c:82 +msgid "Envelope C65" +msgstr "" + +#: ppdc/sample.c:83 +msgid "Envelope C7" +msgstr "" + +#: ppdc/sample.c:84 +msgid "Envelope Choukei 3" +msgstr "" + +#: ppdc/sample.c:85 +msgid "Envelope Choukei 3 Long Edge" +msgstr "" + +#: ppdc/sample.c:86 +msgid "Envelope Choukei 4" +msgstr "" + +#: ppdc/sample.c:87 +msgid "Envelope Choukei 4 Long Edge" +msgstr "" + +#: ppdc/sample.c:70 +msgid "Envelope DL" +msgstr "" + +#: ppdc/sample.c:278 +msgid "Envelope Feed" +msgstr "" + +#: ppdc/sample.c:88 +msgid "Envelope Invite" +msgstr "" + +#: ppdc/sample.c:92 +msgid "Envelope Italian" +msgstr "" + +#: ppdc/sample.c:93 +msgid "Envelope Kaku2" +msgstr "" + +#: ppdc/sample.c:94 +msgid "Envelope Kaku2 Long Edge" +msgstr "" + +#: ppdc/sample.c:95 +msgid "Envelope Kaku3" +msgstr "" + +#: ppdc/sample.c:96 +msgid "Envelope Kaku3 Long Edge" +msgstr "" + +#: ppdc/sample.c:97 +msgid "Envelope Monarch" +msgstr "" + +#: ppdc/sample.c:99 +msgid "Envelope PRC1 " +msgstr "" + +#: ppdc/sample.c:100 +msgid "Envelope PRC1 Long Edge" +msgstr "" + +#: ppdc/sample.c:117 +msgid "Envelope PRC10" +msgstr "" + +#: ppdc/sample.c:118 +msgid "Envelope PRC10 Long Edge" +msgstr "" + +#: ppdc/sample.c:101 +msgid "Envelope PRC2" +msgstr "" + +#: ppdc/sample.c:102 +msgid "Envelope PRC2 Long Edge" +msgstr "" + +#: ppdc/sample.c:103 +msgid "Envelope PRC3" +msgstr "" + +#: ppdc/sample.c:104 +msgid "Envelope PRC3 Long Edge" +msgstr "" + +#: ppdc/sample.c:105 +msgid "Envelope PRC4" +msgstr "" + +#: ppdc/sample.c:106 +msgid "Envelope PRC4 Long Edge" +msgstr "" + +#: ppdc/sample.c:108 +msgid "Envelope PRC5 Long Edge" +msgstr "" + +#: ppdc/sample.c:107 +msgid "Envelope PRC5PRC5" +msgstr "" + +#: ppdc/sample.c:109 +msgid "Envelope PRC6" +msgstr "" + +#: ppdc/sample.c:110 +msgid "Envelope PRC6 Long Edge" +msgstr "" + +#: ppdc/sample.c:111 +msgid "Envelope PRC7" +msgstr "" + +#: ppdc/sample.c:112 +msgid "Envelope PRC7 Long Edge" +msgstr "" + +#: ppdc/sample.c:113 +msgid "Envelope PRC8" +msgstr "" + +#: ppdc/sample.c:114 +msgid "Envelope PRC8 Long Edge" +msgstr "" + +#: ppdc/sample.c:115 +msgid "Envelope PRC9" +msgstr "" + +#: ppdc/sample.c:116 +msgid "Envelope PRC9 Long Edge" +msgstr "" + +#: ppdc/sample.c:98 +msgid "Envelope Personal" +msgstr "" + +#: ppdc/sample.c:119 +msgid "Envelope You4" +msgstr "" + +#: ppdc/sample.c:120 +msgid "Envelope You4 Long Edge" +msgstr "" + +#: ppdc/sample.c:249 +msgid "Epson" +msgstr "" + +#: cgi-bin/admin.c:3582 +msgid "Error Policy" +msgstr "" + +#: filter/rastertopwg.c:403 filter/rastertopwg.c:418 filter/rastertopwg.c:429 +#: filter/rastertopwg.c:440 +msgid "Error sending raster data." +msgstr "" + +#: systemv/lpinfo.c:103 systemv/lpmove.c:88 +msgid "Error: need hostname after \"-h\" option." +msgstr "" + +#: ppdc/sample.c:364 +msgid "Every 10 Labels" +msgstr "" + +#: ppdc/sample.c:356 +msgid "Every 2 Labels" +msgstr "" + +#: ppdc/sample.c:357 +msgid "Every 3 Labels" +msgstr "" + +#: ppdc/sample.c:358 +msgid "Every 4 Labels" +msgstr "" + +#: ppdc/sample.c:359 +msgid "Every 5 Labels" +msgstr "" + +#: ppdc/sample.c:360 +msgid "Every 6 Labels" +msgstr "" + +#: ppdc/sample.c:361 +msgid "Every 7 Labels" +msgstr "" + +#: ppdc/sample.c:362 +msgid "Every 8 Labels" +msgstr "" + +#: ppdc/sample.c:363 +msgid "Every 9 Labels" +msgstr "" + +#: ppdc/sample.c:355 +msgid "Every Label" +msgstr "" + +#: ppdc/sample.c:121 +msgid "Executive" +msgstr "" + +#: cups/http-support.c:1297 +msgid "Expectation Failed" +msgstr "" + +#: cgi-bin/admin.c:2337 cgi-bin/admin.c:2356 +msgid "Export Printers to Samba" +msgstr "" + +#: systemv/cupstestdsc.c:172 systemv/cupstestdsc.c:189 +#: systemv/cupstestdsc.c:214 systemv/cupstestdsc.c:231 +#: systemv/cupstestdsc.c:255 systemv/cupstestdsc.c:273 +#: systemv/cupstestdsc.c:302 systemv/cupstestdsc.c:339 +#: systemv/cupstestdsc.c:349 systemv/cupstestdsc.c:359 +#: systemv/cupstestdsc.c:369 systemv/cupstestdsc.c:379 +#: systemv/cupstestdsc.c:387 +msgid "FAIL" +msgstr "" + +#: ppdc/sample.c:122 +msgid "FanFold German" +msgstr "" + +#: ppdc/sample.c:123 +msgid "FanFold Legal German" +msgstr "" + +#: ppdc/sample.c:124 +msgid "Fanfold US" +msgstr "" + +#: cups/file.c:300 +#, c-format +msgid "File \"%s\" contains a relative path." +msgstr "" + +#: cups/file.c:275 +#, c-format +msgid "File \"%s\" has insecure permissions (0%o/uid=%d/gid=%d)." +msgstr "" + +#: cups/file.c:289 +#, c-format +msgid "File \"%s\" is a directory." +msgstr "" + +#: cups/file.c:261 +#, c-format +msgid "File \"%s\" not available: %s" +msgstr "" + +#: cups/file.c:247 +#, c-format +msgid "File \"%s\" permissions OK (0%o/uid=%d/gid=%d)." +msgstr "" + +#: ppdc/sample.c:169 +msgid "File Folder" +msgstr "" + +#: ppdc/sample.c:178 +msgid "File Folder - 9/16 x 3 7/16\"" +msgstr "" + +#: scheduler/ipp.c:2454 +#, c-format +msgid "File device URIs have been disabled. To enable, see the FileDevice directive in \"%s/cupsd.conf\"." +msgstr "" + +#: filter/rastertoepson.c:1117 filter/rastertohp.c:845 +#: filter/rastertolabel.c:1273 +#, c-format +msgid "Finished page %d." +msgstr "" + +#: ppdc/sample.c:125 +msgid "Folio" +msgstr "" + +#: cups/http-support.c:1276 +msgid "Forbidden" +msgstr "" + +#: cups/ppd.c:742 cups/ppd.c:1302 +msgid "General" +msgstr "" + +#: ppdc/sample.c:268 +msgid "Generic" +msgstr "" + +#: cups/snmp.c:1016 +msgid "Get-Response-PDU uses indefinite length" +msgstr "" + +#: ppdc/sample.c:299 +msgid "Glossy Paper" +msgstr "" + +#: scheduler/ipp.c:3924 scheduler/ipp.c:4325 scheduler/ipp.c:4863 +#: scheduler/ipp.c:6945 scheduler/ipp.c:7092 scheduler/ipp.c:8524 +#: scheduler/ipp.c:9642 scheduler/ipp.c:9867 scheduler/ipp.c:10194 +#: scheduler/ipp.c:10788 +msgid "Got a printer-uri attribute but no job-id." +msgstr "" + +#: ppdc/sample.c:264 +msgid "Grayscale" +msgstr "" + +#: ppdc/sample.c:289 +msgid "HP" +msgstr "" + +#: ppdc/sample.c:170 +msgid "Hanging Folder" +msgstr "" + +#: ppdc/sample.c:179 +msgid "Hanging Folder - 9/16 x 2\"" +msgstr "" + +#: cups/ipp.c:2677 cups/ipp.c:2704 cups/ipp.c:2727 +msgid "IPP 1setOf attribute with incompatible value tags." +msgstr "" + +#: cups/ipp.c:2640 +msgid "IPP attribute has no name." +msgstr "" + +#: cups/ipp.c:5515 +msgid "IPP attribute is not a member of the message." +msgstr "" + +#: cups/ipp.c:3067 +msgid "IPP begCollection value not 0 bytes." +msgstr "" + +#: cups/ipp.c:2863 +msgid "IPP boolean value not 1 byte." +msgstr "" + +#: cups/ipp.c:2921 +msgid "IPP date value not 11 bytes." +msgstr "" + +#: cups/ipp.c:3088 +msgid "IPP endCollection value not 0 bytes." +msgstr "" + +#: cups/ipp.c:2838 +msgid "IPP enum value not 4 bytes." +msgstr "" + +#: cups/ipp.c:2569 +msgid "IPP extension tag larger than 0x7FFFFFFF." +msgstr "" + +#: cups/ipp.c:2835 +msgid "IPP integer value not 4 bytes." +msgstr "" + +#: cups/ipp.c:3030 +msgid "IPP language length overflows value." +msgstr "" + +#: cups/ipp.c:2754 +msgid "IPP member name is not empty." +msgstr "" + +#: cups/ipp.c:2623 +msgid "IPP name larger than 32767 bytes." +msgstr "" + +#: cups/ipp.c:2997 +msgid "IPP nameWithLanguage value less than minimum 4 bytes." +msgstr "" + +#: cups/ipp.c:2966 +msgid "IPP rangeOfInteger value not 8 bytes." +msgstr "" + +#: cups/ipp.c:2939 +msgid "IPP resolution value not 9 bytes." +msgstr "" + +#: cups/ipp.c:3047 +msgid "IPP string length overflows value." +msgstr "" + +#: cups/ipp.c:2993 +msgid "IPP textWithLanguage value less than minimum 4 bytes." +msgstr "" + +#: cups/ipp.c:2821 +msgid "IPP value larger than 32767 bytes." +msgstr "" + +#: ppdc/sample.c:1 +msgid "ISOLatin1" +msgstr "" + +#: cups/ppd.c:353 +msgid "Illegal control character" +msgstr "" + +#: cups/ppd.c:354 +msgid "Illegal main keyword string" +msgstr "" + +#: cups/ppd.c:355 +msgid "Illegal option keyword string" +msgstr "" + +#: cups/ppd.c:356 +msgid "Illegal translation string" +msgstr "" + +#: cups/ppd.c:357 +msgid "Illegal whitespace character" +msgstr "" + +#: ppdc/sample.c:283 +msgid "Installable Options" +msgstr "" + +#: ppdc/sample.c:286 +msgid "Installed" +msgstr "" + +#: ppdc/sample.c:302 +msgid "IntelliBar Label Printer" +msgstr "" + +#: ppdc/sample.c:301 +msgid "Intellitech" +msgstr "" + +#: cups/http-support.c:1303 +msgid "Internal Server Error" +msgstr "" + +#: cups/ppd.c:344 +msgid "Internal error" +msgstr "" + +#: ppdc/sample.c:167 +msgid "Internet Postage 2-Part" +msgstr "" + +#: ppdc/sample.c:176 +msgid "Internet Postage 2-Part - 2 1/4 x 7 1/2\"" +msgstr "" + +#: ppdc/sample.c:168 +msgid "Internet Postage 3-Part" +msgstr "" + +#: ppdc/sample.c:177 +msgid "Internet Postage 3-Part - 2 1/4 x 7\"" +msgstr "" + +#: backend/ipp.c:294 +msgid "Internet Printing Protocol" +msgstr "" + +#: cups/ppd.c:1420 +msgid "JCL" +msgstr "" + +#: ppdc/sample.c:53 +msgid "JIS B0" +msgstr "" + +#: ppdc/sample.c:55 +msgid "JIS B1" +msgstr "" + +#: ppdc/sample.c:54 +msgid "JIS B10" +msgstr "" + +#: ppdc/sample.c:56 +msgid "JIS B2" +msgstr "" + +#: ppdc/sample.c:57 +msgid "JIS B3" +msgstr "" + +#: ppdc/sample.c:58 +msgid "JIS B4" +msgstr "" + +#: ppdc/sample.c:59 +msgid "JIS B4 Long Edge" +msgstr "" + +#: ppdc/sample.c:60 +msgid "JIS B5" +msgstr "" + +#: ppdc/sample.c:61 +msgid "JIS B5 Long Edge" +msgstr "" + +#: ppdc/sample.c:62 +msgid "JIS B6" +msgstr "" + +#: ppdc/sample.c:63 +msgid "JIS B6 Long Edge" +msgstr "" + +#: ppdc/sample.c:64 +msgid "JIS B7" +msgstr "" + +#: ppdc/sample.c:65 +msgid "JIS B8" +msgstr "" + +#: ppdc/sample.c:66 +msgid "JIS B9" +msgstr "" + +#: scheduler/ipp.c:9939 +#, c-format +msgid "Job #%d cannot be restarted - no files." +msgstr "" + +#: scheduler/ipp.c:3964 scheduler/ipp.c:4195 scheduler/ipp.c:4250 +#: scheduler/ipp.c:4427 scheduler/ipp.c:4873 scheduler/ipp.c:6607 +#: scheduler/ipp.c:6985 scheduler/ipp.c:7132 scheduler/ipp.c:7432 +#: scheduler/ipp.c:8371 scheduler/ipp.c:8393 scheduler/ipp.c:8565 +#: scheduler/ipp.c:8790 scheduler/ipp.c:8833 scheduler/ipp.c:9682 +#: scheduler/ipp.c:9907 scheduler/ipp.c:10234 scheduler/ipp.c:10828 +#, c-format +msgid "Job #%d does not exist." +msgstr "" + +#: scheduler/ipp.c:4459 +#, c-format +msgid "Job #%d is already aborted - can't cancel." +msgstr "" + +#: scheduler/ipp.c:4453 +#, c-format +msgid "Job #%d is already canceled - can't cancel." +msgstr "" + +#: scheduler/ipp.c:4465 +#, c-format +msgid "Job #%d is already completed - can't cancel." +msgstr "" + +#: scheduler/ipp.c:8591 scheduler/ipp.c:8875 scheduler/ipp.c:10843 +#, c-format +msgid "Job #%d is finished and cannot be altered." +msgstr "" + +#: scheduler/ipp.c:9921 +#, c-format +msgid "Job #%d is not complete." +msgstr "" + +#: scheduler/ipp.c:3979 +#, c-format +msgid "Job #%d is not held for authentication." +msgstr "" + +#: scheduler/ipp.c:9696 +#, c-format +msgid "Job #%d is not held." +msgstr "" + +#: cgi-bin/ipp-var.c:1055 +msgid "Job Completed" +msgstr "" + +#: cgi-bin/ipp-var.c:1053 +msgid "Job Created" +msgstr "" + +#: cgi-bin/ipp-var.c:1059 +msgid "Job Options Changed" +msgstr "" + +#: cgi-bin/ipp-var.c:1057 +msgid "Job Stopped" +msgstr "" + +#: scheduler/ipp.c:10925 +msgid "Job is completed and cannot be changed." +msgstr "" + +#: cgi-bin/jobs.c:198 +msgid "Job operation failed:" +msgstr "" + +#: scheduler/ipp.c:10961 scheduler/ipp.c:10980 scheduler/ipp.c:10991 +msgid "Job state cannot be changed." +msgstr "" + +#: scheduler/ipp.c:9787 +msgid "Job subscriptions cannot be renewed." +msgstr "" + +#: cgi-bin/jobs.c:103 cgi-bin/jobs.c:114 cgi-bin/jobs.c:195 +msgid "Jobs" +msgstr "" + +#: backend/lpd.c:185 +msgid "LPD/LPR Host or Printer" +msgstr "" + +#: ppdc/sample.c:239 +msgid "Label Printer" +msgstr "" + +#: ppdc/sample.c:455 +msgid "Label Top" +msgstr "" + +#: scheduler/ipp.c:2154 scheduler/ipp.c:6530 +#, c-format +msgid "Language \"%s\" not supported." +msgstr "" + +#: ppdc/sample.c:164 +msgid "Large Address" +msgstr "" + +#: ppdc/sample.c:173 +msgid "Large Address - 1 4/10 x 3 1/2\"" +msgstr "" + +#: ppdc/sample.c:300 +msgid "LaserJet Series PCL 4/5" +msgstr "" + +#: ppdc/sample.c:43 +msgid "Letter Oversize" +msgstr "" + +#: ppdc/sample.c:44 +msgid "Letter Oversize Long Edge" +msgstr "" + +#: ppdc/sample.c:245 +msgid "Light" +msgstr "" + +#: cups/ppd.c:352 +msgid "Line longer than the maximum allowed (255 characters)" +msgstr "" + +#: cgi-bin/admin.c:2374 +msgid "List Available Printers" +msgstr "" + +#: ppdc/sample.c:281 +msgid "Long-Edge (Portrait)" +msgstr "" + +#: cups/http-support.c:1518 +msgid "Looking for printer." +msgstr "" + +#: ppdc/sample.c:277 +msgid "Manual Feed" +msgstr "" + +#: cups/ppd.c:789 cups/ppd.c:1357 +msgid "Media Size" +msgstr "" + +#: cups/ppd.c:793 cups/ppd.c:1361 ppdc/sample.c:271 +msgid "Media Source" +msgstr "" + +#: ppdc/sample.c:373 +msgid "Media Tracking" +msgstr "" + +#: cups/ppd.c:791 cups/ppd.c:1359 ppdc/sample.c:294 +msgid "Media Type" +msgstr "" + +#: ppdc/sample.c:246 +msgid "Medium" +msgstr "" + +#: cups/ppd.c:341 +msgid "Memory allocation error" +msgstr "" + +#: cups/ppd.c:361 +msgid "Missing CloseGroup" +msgstr "" + +#: cups/ppd.c:342 +msgid "Missing PPD-Adobe-4.x header" +msgstr "" + +#: cups/ppd.c:351 +msgid "Missing asterisk in column 1" +msgstr "" + +#: scheduler/ipp.c:7008 +msgid "Missing document-number attribute." +msgstr "" + +#: cups/adminutil.c:273 +#, c-format +msgid "Missing double quote on line %d." +msgstr "" + +#: cgi-bin/admin.c:736 cgi-bin/admin.c:2087 cgi-bin/admin.c:2172 +#: cgi-bin/admin.c:2763 cgi-bin/admin.c:3017 cgi-bin/admin.c:3128 +#: cgi-bin/admin.c:3838 +msgid "Missing form variable" +msgstr "" + +#: scheduler/ipp.c:10288 +msgid "Missing last-document attribute in request." +msgstr "" + +#: cups/pwg-media.c:473 +msgid "Missing media or media-col." +msgstr "" + +#: cups/pwg-media.c:392 +msgid "Missing media-size in media-col." +msgstr "" + +#: scheduler/ipp.c:7562 +msgid "Missing notify-subscription-ids attribute." +msgstr "" + +#: cups/ppd.c:359 +msgid "Missing option keyword" +msgstr "" + +#: scheduler/ipp.c:4106 scheduler/ipp.c:4131 +msgid "Missing requesting-user-name attribute." +msgstr "" + +#: scheduler/ipp.c:482 +msgid "Missing required attributes." +msgstr "" + +#: cups/adminutil.c:254 +#, c-format +msgid "Missing value on line %d." +msgstr "" + +#: cups/ppd.c:343 +msgid "Missing value string" +msgstr "" + +#: cups/pwg-media.c:380 +msgid "Missing x-dimension in media-size." +msgstr "" + +#: cups/pwg-media.c:386 +msgid "Missing y-dimension in media-size." +msgstr "" + +#: systemv/lpinfo.c:470 +#, c-format +msgid "" +"Model: name = %s\n" +" natural_language = %s\n" +" make-and-model = %s\n" +" device-id = %s" +msgstr "" + +#: cgi-bin/admin.c:570 +msgid "Modify Class" +msgstr "" + +#: cgi-bin/admin.c:882 +msgid "Modify Printer" +msgstr "" + +#: cgi-bin/ipp-var.c:425 cgi-bin/ipp-var.c:516 +msgid "Move All Jobs" +msgstr "" + +#: cgi-bin/ipp-var.c:364 cgi-bin/ipp-var.c:423 cgi-bin/ipp-var.c:514 +msgid "Move Job" +msgstr "" + +#: cups/http-support.c:1260 +msgid "Moved Permanently" +msgstr "" + +#: cups/ppd.c:340 +msgid "NULL PPD file pointer" +msgstr "" + +#: cups/snmp.c:1053 +msgid "Name OID uses indefinite length" +msgstr "" + +#: scheduler/ipp.c:1163 +msgid "Nested classes are not allowed." +msgstr "" + +#: ppdc/sample.c:439 +msgid "Never" +msgstr "" + +#: ppdc/sample.c:265 +msgid "New Stylus Color Series" +msgstr "" + +#: ppdc/sample.c:267 +msgid "New Stylus Photo Series" +msgstr "" + +#: cups/ppd.c:1949 +msgid "No" +msgstr "" + +#: cups/http-support.c:1257 +msgid "No Content" +msgstr "" + +#: cups/util.c:1286 +msgid "No PPD name" +msgstr "" + +#: cups/snmp.c:1047 +msgid "No VarBind SEQUENCE" +msgstr "" + +#: cups/adminutil.c:788 +msgid "No Windows printer drivers are installed." +msgstr "" + +#: cups/request.c:571 cups/request.c:914 +msgid "No active connection" +msgstr "" + +#: scheduler/ipp.c:4376 +#, c-format +msgid "No active jobs on %s." +msgstr "" + +#: scheduler/ipp.c:323 +msgid "No attributes in request." +msgstr "" + +#: scheduler/ipp.c:4007 +msgid "No authentication information provided." +msgstr "" + +#: cups/snmp.c:1004 +msgid "No community name" +msgstr "" + +#: scheduler/ipp.c:6808 +msgid "No default printer." +msgstr "" + +#: cgi-bin/ipp-var.c:436 scheduler/ipp.c:8137 +msgid "No destinations added." +msgstr "" + +#: backend/usb.c:200 +msgid "No device URI found in argv[0] or in DEVICE_URI environment variable." +msgstr "" + +#: cups/snmp.c:1034 +msgid "No error-index" +msgstr "" + +#: cups/snmp.c:1026 +msgid "No error-status" +msgstr "" + +#: scheduler/ipp.c:9079 scheduler/ipp.c:10302 +msgid "No file in print request." +msgstr "" + +#: cups/util.c:921 +msgid "No modification time" +msgstr "" + +#: cups/snmp.c:1051 +msgid "No name OID" +msgstr "" + +#: filter/rastertoepson.c:1147 filter/rastertohp.c:876 +#: filter/rastertolabel.c:1302 +msgid "No pages were found." +msgstr "" + +#: cups/util.c:915 +msgid "No printer name" +msgstr "" + +#: cups/util.c:1789 +msgid "No printer-uri found" +msgstr "" + +#: cups/util.c:1774 +msgid "No printer-uri found for class" +msgstr "" + +#: scheduler/ipp.c:7211 +msgid "No printer-uri in request." +msgstr "" + +#: cups/snmp.c:1018 +msgid "No request-id" +msgstr "" + +#: scheduler/ipp.c:6415 +msgid "No subscription attributes in request." +msgstr "" + +#: scheduler/ipp.c:8464 +msgid "No subscriptions found." +msgstr "" + +#: cups/snmp.c:1042 +msgid "No variable-bindings SEQUENCE" +msgstr "" + +#: cups/snmp.c:997 +msgid "No version number" +msgstr "" + +#: ppdc/sample.c:376 +msgid "Non-continuous (Mark sensing)" +msgstr "" + +#: ppdc/sample.c:375 +msgid "Non-continuous (Web sensing)" +msgstr "" + +#: ppdc/sample.c:247 +msgid "Normal" +msgstr "" + +#: cups/http-support.c:1279 +msgid "Not Found" +msgstr "" + +#: cups/http-support.c:1291 +msgid "Not Implemented" +msgstr "" + +#: ppdc/sample.c:285 +msgid "Not Installed" +msgstr "" + +#: cups/http-support.c:1266 +msgid "Not Modified" +msgstr "" + +#: cups/http-support.c:1294 +msgid "Not Supported" +msgstr "" + +#: scheduler/ipp.c:1539 scheduler/ipp.c:11523 +msgid "Not allowed to print." +msgstr "" + +#: ppdc/sample.c:146 +msgid "Note" +msgstr "" + +#: systemv/cupstestdsc.c:433 +msgid "Note: this program only validates the DSC comments, not the PostScript itself." +msgstr "" + +#: cups/http-support.c:1248 cups/ppd.c:338 +msgid "OK" +msgstr "" + +#: ppdc/sample.c:280 +msgid "Off (1-Sided)" +msgstr "" + +#: ppdc/sample.c:370 +msgid "Oki" +msgstr "" + +#: cgi-bin/help.c:90 cgi-bin/help.c:131 cgi-bin/help.c:141 cgi-bin/help.c:172 +msgid "Online Help" +msgstr "" + +#: cups/adminutil.c:955 +#, c-format +msgid "Open of %s failed: %s" +msgstr "" + +#: cups/ppd.c:346 +msgid "OpenGroup without a CloseGroup first" +msgstr "" + +#: cups/ppd.c:348 +msgid "OpenUI/JCLOpenUI without a CloseUI/JCLCloseUI first" +msgstr "" + +#: cgi-bin/admin.c:3609 +msgid "Operation Policy" +msgstr "" + +#: filter/pstops.c:2217 +#, c-format +msgid "Option \"%s\" cannot be included via %%%%IncludeFeature." +msgstr "" + +#: cgi-bin/admin.c:3259 cgi-bin/admin.c:3343 +msgid "Options Installed" +msgstr "" + +#: scheduler/cupsfilter.c:1438 scheduler/cupsfilter.c:1465 +#: scheduler/main.c:1973 systemv/cupsaddsmb.c:284 systemv/cupsctl.c:203 +#: systemv/cupstestdsc.c:429 systemv/cupstestppd.c:3777 test/ipptool.c:4381 +#: ppdc/ppdc.cxx:437 ppdc/ppdhtml.cxx:174 ppdc/ppdi.cxx:130 +#: ppdc/ppdmerge.cxx:369 ppdc/ppdpo.cxx:254 +msgid "Options:" +msgstr "" + +#: cups/ppd-cache.c:156 +msgid "Out of date PPD cache file." +msgstr "" + +#: cups/ppd-cache.c:1361 +msgid "Out of memory." +msgstr "" + +#: cups/ppd.c:795 cups/ppd.c:1363 +msgid "Output Mode" +msgstr "" + +#: systemv/lpstat.c:1191 systemv/lpstat.c:1195 +#, c-format +msgid "Output for printer %s is sent to %s" +msgstr "" + +#: systemv/lpstat.c:1185 +#, c-format +msgid "Output for printer %s is sent to remote printer %s on %s" +msgstr "" + +#: systemv/lpstat.c:1209 systemv/lpstat.c:1213 +#, c-format +msgid "Output for printer %s/%s is sent to %s" +msgstr "" + +#: systemv/lpstat.c:1203 +#, c-format +msgid "Output for printer %s/%s is sent to remote printer %s on %s" +msgstr "" + +#: systemv/cupstestdsc.c:399 +msgid "PASS" +msgstr "" + +#: ppdc/sample.c:269 +msgid "PCL Laser Printer" +msgstr "" + +#: ppdc/sample.c:149 +msgid "PRC16K" +msgstr "" + +#: ppdc/sample.c:150 +msgid "PRC16K Long Edge" +msgstr "" + +#: ppdc/sample.c:151 +msgid "PRC32K" +msgstr "" + +#: ppdc/sample.c:154 +msgid "PRC32K Long Edge" +msgstr "" + +#: ppdc/sample.c:152 +msgid "PRC32K Oversize" +msgstr "" + +#: ppdc/sample.c:153 +msgid "PRC32K Oversize Long Edge" +msgstr "" + +#: cups/snmp.c:1014 +msgid "Packet does not contain a Get-Response-PDU" +msgstr "" + +#: cups/snmp.c:993 +msgid "Packet does not start with SEQUENCE" +msgstr "" + +#: ppdc/sample.c:369 +msgid "ParamCustominCutInterval" +msgstr "" + +#: ppdc/sample.c:367 +msgid "ParamCustominTearInterval" +msgstr "" + +#: cups/auth.c:196 cups/auth.c:365 +#, c-format +msgid "Password for %s on %s? " +msgstr "" + +#: systemv/cupsaddsmb.c:252 +#, c-format +msgid "Password for %s required to access %s via SAMBA: " +msgstr "" + +#: cgi-bin/classes.c:167 +msgid "Pause Class" +msgstr "" + +#: cgi-bin/printers.c:170 +msgid "Pause Printer" +msgstr "" + +#: ppdc/sample.c:457 +msgid "Peel-Off" +msgstr "" + +#: ppdc/sample.c:160 +msgid "Photo" +msgstr "" + +#: ppdc/sample.c:161 +msgid "Photo Labels" +msgstr "" + +#: ppdc/sample.c:295 +msgid "Plain Paper" +msgstr "" + +#: cgi-bin/admin.c:3277 cgi-bin/admin.c:3558 +msgid "Policies" +msgstr "" + +#: cgi-bin/admin.c:3284 cgi-bin/admin.c:3627 cgi-bin/admin.c:3640 +msgid "Port Monitor" +msgstr "" + +#: ppdc/sample.c:287 +msgid "PostScript Printer" +msgstr "" + +#: ppdc/sample.c:147 +msgid "Postcard" +msgstr "" + +#: ppdc/sample.c:71 +msgid "Postcard Double " +msgstr "" + +#: ppdc/sample.c:72 +msgid "Postcard Double Long Edge" +msgstr "" + +#: ppdc/sample.c:148 +msgid "Postcard Long Edge" +msgstr "" + +#: ppdc/sample.c:304 +msgid "Print Density" +msgstr "" + +#: cups/notify.c:82 +msgid "Print Job:" +msgstr "" + +#: ppdc/sample.c:349 +msgid "Print Mode" +msgstr "" + +#: ppdc/sample.c:392 +msgid "Print Rate" +msgstr "" + +#: cgi-bin/printers.c:179 +msgid "Print Self-Test Page" +msgstr "" + +#: ppdc/sample.c:336 +msgid "Print Speed" +msgstr "" + +#: cgi-bin/ipp-var.c:792 +msgid "Print Test Page" +msgstr "" + +#: ppdc/sample.c:365 +msgid "Print and Cut" +msgstr "" + +#: ppdc/sample.c:353 +msgid "Print and Tear" +msgstr "" + +#: backend/ipp.c:1466 +#, c-format +msgid "Print file accepted - job ID %d." +msgstr "" + +#: backend/ipp.c:1457 +msgid "Print file accepted - job ID unknown." +msgstr "" + +#: backend/socket.c:424 backend/usb-unix.c:195 +msgid "Print file sent." +msgstr "" + +#: backend/ipp.c:1421 +msgid "Print file was not accepted." +msgstr "" + +#: cgi-bin/ipp-var.c:1047 +msgid "Printer Added" +msgstr "" + +#: ppdc/sample.c:272 +msgid "Printer Default" +msgstr "" + +#: cgi-bin/ipp-var.c:1051 +msgid "Printer Deleted" +msgstr "" + +#: cgi-bin/ipp-var.c:1049 +msgid "Printer Modified" +msgstr "" + +#: cgi-bin/ipp-var.c:1045 +msgid "Printer Paused" +msgstr "" + +#: ppdc/sample.c:303 +msgid "Printer Settings" +msgstr "" + +#: backend/usb-unix.c:132 +msgid "Printer busy, will retry in 10 seconds." +msgstr "" + +#: backend/lpd.c:617 backend/lpd.c:1005 backend/lpd.c:1092 backend/lpd.c:1147 +#, c-format +msgid "Printer did not respond after %d seconds." +msgstr "" + +#: backend/ipp.c:870 backend/ipp.c:877 +#, c-format +msgid "Printer does not support IPP/%d.%d, trying IPP/%s." +msgstr "" + +#: backend/usb-unix.c:429 backend/usb-unix.c:513 +msgid "Printer is busy, will retry in 5 seconds." +msgstr "" + +#: backend/runloop.c:253 backend/runloop.c:371 +msgid "Printer is not currently connected." +msgstr "" + +#: backend/runloop.c:392 +msgid "Printer is now connected." +msgstr "" + +#: backend/usb-darwin.c:1286 +msgid "Printer is now online." +msgstr "" + +#: backend/usb-darwin.c:1307 +msgid "Printer is offline." +msgstr "" + +#: backend/usb-unix.c:139 +msgid "Printer not connected, will retry in 30 seconds." +msgstr "" + +#: cups/notify.c:126 +msgid "Printer:" +msgstr "" + +#: cgi-bin/printers.c:204 cgi-bin/printers.c:332 +msgid "Printers" +msgstr "" + +#: filter/rastertoepson.c:1093 filter/rastertohp.c:817 +#: filter/rastertolabel.c:1249 +#, c-format +msgid "Printing page %d, %d%% complete." +msgstr "" + +#: cgi-bin/classes.c:173 cgi-bin/printers.c:176 +msgid "Purge Jobs" +msgstr "" + +#: ppdc/sample.c:155 +msgid "Quarto" +msgstr "" + +#: scheduler/ipp.c:1534 scheduler/ipp.c:11518 +msgid "Quota limit reached." +msgstr "" + +#: berkeley/lpq.c:515 +msgid "Rank Owner Job File(s) Total Size" +msgstr "" + +#. TRANSLATORS: Pri is job priority. +#: berkeley/lpq.c:511 +msgid "Rank Owner Pri Job Files Total Size" +msgstr "" + +#: backend/ipp.c:1784 backend/socket.c:475 filter/rastertoepson.c:1152 +#: filter/rastertohp.c:881 filter/rastertolabel.c:1307 +msgid "Ready to print." +msgstr "" + +#: cgi-bin/classes.c:171 cgi-bin/printers.c:174 +msgid "Reject Jobs" +msgstr "" + +#: backend/lpd.c:1015 backend/lpd.c:1157 +#, c-format +msgid "Remote host did not accept control file (%d)." +msgstr "" + +#: backend/lpd.c:1105 +#, c-format +msgid "Remote host did not accept data file (%d)." +msgstr "" + +#: ppdc/sample.c:437 +msgid "Reprint After Error" +msgstr "" + +#: cups/http-support.c:1282 +msgid "Request Entity Too Large" +msgstr "" + +#: cups/ppd.c:797 cups/ppd.c:1365 ppdc/sample.c:240 +msgid "Resolution" +msgstr "" + +#: cgi-bin/classes.c:165 +msgid "Resume Class" +msgstr "" + +#: cgi-bin/printers.c:167 +msgid "Resume Printer" +msgstr "" + +#: ppdc/sample.c:165 +msgid "Return Address" +msgstr "" + +#: ppdc/sample.c:174 +msgid "Return Address - 3/4 x 2\"" +msgstr "" + +#: ppdc/sample.c:458 +msgid "Rewind" +msgstr "" + +#: cups/adminutil.c:2052 +#, c-format +msgid "Running command: %s %s -N -A %s -c '%s'" +msgstr "" + +#: cups/snmp.c:995 +msgid "SEQUENCE uses indefinite length" +msgstr "" + +#: cups/http-support.c:1306 +msgid "SSL/TLS Negotiation Error" +msgstr "" + +#: cups/http-support.c:1263 +msgid "See Other" +msgstr "" + +#: backend/usb-darwin.c:543 +msgid "Sending data to printer." +msgstr "" + +#: cgi-bin/ipp-var.c:1061 +msgid "Server Restarted" +msgstr "" + +#: cgi-bin/ipp-var.c:1067 +msgid "Server Security Auditing" +msgstr "" + +#: cgi-bin/ipp-var.c:1063 +msgid "Server Started" +msgstr "" + +#: cgi-bin/ipp-var.c:1065 +msgid "Server Stopped" +msgstr "" + +#: cups/http-support.c:1300 +msgid "Service Unavailable" +msgstr "" + +#: cgi-bin/admin.c:2764 cgi-bin/admin.c:2810 cgi-bin/admin.c:2967 +#: cgi-bin/admin.c:2986 +msgid "Set Allowed Users" +msgstr "" + +#: cgi-bin/admin.c:3013 +msgid "Set As Server Default" +msgstr "" + +#: cgi-bin/admin.c:3113 +msgid "Set Class Options" +msgstr "" + +#: cgi-bin/admin.c:3113 cgi-bin/admin.c:3287 cgi-bin/admin.c:3669 +msgid "Set Printer Options" +msgstr "" + +#: cgi-bin/admin.c:3839 cgi-bin/admin.c:3883 cgi-bin/admin.c:3901 +msgid "Set Publishing" +msgstr "" + +#: ppdc/sample.c:166 +msgid "Shipping Address" +msgstr "" + +#: ppdc/sample.c:175 +msgid "Shipping Address - 2 5/16 x 4\"" +msgstr "" + +#: ppdc/sample.c:282 +msgid "Short-Edge (Landscape)" +msgstr "" + +#: ppdc/sample.c:297 +msgid "Special Paper" +msgstr "" + +#: backend/lpd.c:1056 +#, c-format +msgid "Spooling job, %.0f%% complete." +msgstr "" + +#: ppdc/sample.c:350 +msgid "Standard" +msgstr "" + +#. TRANSLATORS: Banner/cover sheet before the print job. +#: cgi-bin/admin.c:3530 +msgid "Starting Banner" +msgstr "" + +#: filter/rastertoepson.c:1069 filter/rastertohp.c:793 +#: filter/rastertolabel.c:1225 +#, c-format +msgid "Starting page %d." +msgstr "" + +#: ppdc/sample.c:156 +msgid "Statement" +msgstr "" + +#: ppdc/sample.c:260 +msgid "Stylus Color Series" +msgstr "" + +#: ppdc/sample.c:266 +msgid "Stylus Photo Series" +msgstr "" + +#: scheduler/ipp.c:4522 scheduler/ipp.c:7578 scheduler/ipp.c:8277 +#: scheduler/ipp.c:9775 +#, c-format +msgid "Subscription #%d does not exist." +msgstr "" + +#: ppdc/sample.c:157 +msgid "Super A" +msgstr "" + +#: ppdc/sample.c:158 +msgid "Super B" +msgstr "" + +#: ppdc/sample.c:162 +msgid "Super B/A3" +msgstr "" + +#: cups/http-support.c:1245 +msgid "Switching Protocols" +msgstr "" + +#: ppdc/sample.c:159 +msgid "Tabloid" +msgstr "" + +#: ppdc/sample.c:45 +msgid "Tabloid Oversize" +msgstr "" + +#: ppdc/sample.c:46 +msgid "Tabloid Oversize Long Edge" +msgstr "" + +#: ppdc/sample.c:351 +msgid "Tear" +msgstr "" + +#: ppdc/sample.c:456 +msgid "Tear-Off" +msgstr "" + +#: ppdc/sample.c:397 +msgid "Tear-Off Adjust Position" +msgstr "" + +#: scheduler/ipp.c:7282 scheduler/ipp.c:7360 scheduler/ipp.c:7376 +#: scheduler/ipp.c:7394 +#, c-format +msgid "The %s attribute cannot be provided with job-ids." +msgstr "" + +#: scheduler/ipp.c:7808 +#, c-format +msgid "The PPD file \"%s\" could not be found." +msgstr "" + +#: scheduler/ipp.c:7795 +#, c-format +msgid "The PPD file \"%s\" could not be opened: %s" +msgstr "" + +#: filter/rastertoepson.c:1038 filter/rastertohp.c:764 +#: filter/rastertolabel.c:1189 +msgid "The PPD file could not be opened." +msgstr "" + +#: cgi-bin/admin.c:749 +msgid "The class name may only contain up to 127 printable characters and may not contain spaces, slashes (/), or the pound sign (#)." +msgstr "" + +#: cups/localize.c:353 +msgid "The developer unit needs to be replaced." +msgstr "" + +#: cups/localize.c:351 +msgid "The developer unit will need to be replaced soon." +msgstr "" + +#: cups/localize.c:343 +msgid "The fuser's temperature is high." +msgstr "" + +#: cups/localize.c:345 +msgid "The fuser's temperature is low." +msgstr "" + +#: scheduler/ipp.c:2181 +msgid "The notify-lease-duration attribute cannot be used with job subscriptions." +msgstr "" + +#: scheduler/ipp.c:2164 scheduler/ipp.c:6540 +#, c-format +msgid "The notify-user-data value is too large (%d > 63 octets)." +msgstr "" + +#: cups/localize.c:349 +msgid "The optical photoconductor needs to be replaced." +msgstr "" + +#: cups/localize.c:347 +msgid "The optical photoconductor will need to be replaced soon." +msgstr "" + +#: cups/localize.c:331 +msgid "The output bin is almost full." +msgstr "" + +#: cups/localize.c:333 +msgid "The output bin is full." +msgstr "" + +#: cups/localize.c:329 +msgid "The output bin is missing." +msgstr "" + +#: cups/localize.c:325 +msgid "The paper tray is almost empty." +msgstr "" + +#: cups/localize.c:327 +msgid "The paper tray is empty." +msgstr "" + +#: cups/localize.c:323 +msgid "The paper tray is missing." +msgstr "" + +#: cups/localize.c:306 +msgid "The paper tray needs to be filled." +msgstr "" + +#: backend/ipp.c:887 +msgid "The printer URI is incorrect or no longer exists." +msgstr "" + +#: backend/ipp.c:738 backend/ipp.c:852 backend/ipp.c:953 backend/ipp.c:1248 +#: backend/ipp.c:1399 backend/lpd.c:842 backend/socket.c:374 +msgid "The printer is busy." +msgstr "" + +#: cups/localize.c:335 +msgid "The printer is low on ink." +msgstr "" + +#: cups/localize.c:313 +msgid "The printer is low on toner." +msgstr "" + +#: cups/localize.c:311 +msgid "The printer is not connected." +msgstr "" + +#: backend/ipp.c:716 backend/ipp.c:749 backend/ipp.c:848 backend/lpd.c:821 +#: backend/lpd.c:862 backend/socket.c:353 backend/socket.c:386 +msgid "The printer is not responding." +msgstr "" + +#: cups/localize.c:315 +msgid "The printer is out of toner." +msgstr "" + +#: backend/ipp.c:731 backend/lpd.c:835 backend/socket.c:367 +msgid "The printer is unreachable at this time." +msgstr "" + +#: cups/localize.c:337 +msgid "The printer may be out of ink." +msgstr "" + +#: backend/ipp.c:725 backend/lpd.c:829 backend/socket.c:361 +msgid "The printer may not exist or is unavailable at this time." +msgstr "" + +#: cgi-bin/admin.c:931 +msgid "The printer name may only contain up to 127 printable characters and may not contain spaces, slashes (/), or the pound sign (#)." +msgstr "" + +#: scheduler/ipp.c:897 scheduler/ipp.c:1157 scheduler/ipp.c:4171 +#: scheduler/ipp.c:4342 scheduler/ipp.c:6071 scheduler/ipp.c:6374 +#: scheduler/ipp.c:6688 scheduler/ipp.c:7248 scheduler/ipp.c:8013 +#: scheduler/ipp.c:8069 scheduler/ipp.c:8383 scheduler/ipp.c:8649 +#: scheduler/ipp.c:8738 scheduler/ipp.c:8771 scheduler/ipp.c:9094 +#: scheduler/ipp.c:9487 scheduler/ipp.c:9568 scheduler/ipp.c:10697 +#: scheduler/ipp.c:11151 scheduler/ipp.c:11481 scheduler/ipp.c:11563 +#: scheduler/ipp.c:11855 +msgid "The printer or class does not exist." +msgstr "" + +#: scheduler/ipp.c:1325 +msgid "The printer or class is not shared." +msgstr "" + +#: cups/localize.c:317 +msgid "The printer's cover is open." +msgstr "" + +#: cups/localize.c:321 +msgid "The printer's door is open." +msgstr "" + +#: cups/localize.c:319 +msgid "The printer's interlock is open." +msgstr "" + +#: cups/localize.c:339 +msgid "The printer's waste bin is almost full." +msgstr "" + +#: cups/localize.c:341 +msgid "The printer's waste bin is full." +msgstr "" + +#: scheduler/ipp.c:1003 scheduler/ipp.c:2345 +#, c-format +msgid "The printer-uri \"%s\" contains invalid characters." +msgstr "" + +#: scheduler/ipp.c:4148 +msgid "The printer-uri attribute is required." +msgstr "" + +#: scheduler/ipp.c:987 +msgid "The printer-uri must be of the form \"ipp://HOSTNAME/classes/CLASSNAME\"." +msgstr "" + +#: scheduler/ipp.c:2329 +msgid "The printer-uri must be of the form \"ipp://HOSTNAME/printers/PRINTERNAME\"." +msgstr "" + +#: cgi-bin/admin.c:474 +msgid "The subscription name may not contain spaces, slashes (/), question marks (?), or the pound sign (#)." +msgstr "" + +#: scheduler/client.c:2521 +msgid "The web interface is currently disabled. Run \"cupsctl WebInterface=yes\" to enable it." +msgstr "" + +#: scheduler/ipp.c:7343 +#, c-format +msgid "The which-jobs value \"%s\" is not supported." +msgstr "" + +#: scheduler/ipp.c:6618 +msgid "There are too many subscriptions." +msgstr "" + +#: cups/localize.c:308 +msgid "There is a paper jam." +msgstr "" + +#: backend/usb-darwin.c:379 backend/usb-darwin.c:438 backend/usb-darwin.c:505 +#: backend/usb-darwin.c:526 +msgid "There was an unrecoverable USB error." +msgstr "" + +#: ppdc/sample.c:444 +msgid "Thermal Transfer Media" +msgstr "" + +#: scheduler/ipp.c:1528 +msgid "Too many active jobs." +msgstr "" + +#: scheduler/ipp.c:1422 +#, c-format +msgid "Too many job-sheets values (%d > 2)." +msgstr "" + +#: scheduler/ipp.c:2613 +#, c-format +msgid "Too many printer-state-reasons values (%d > %d)." +msgstr "" + +#: ppdc/sample.c:298 +msgid "Transparency" +msgstr "" + +#: ppdc/sample.c:293 +msgid "Tray" +msgstr "" + +#: ppdc/sample.c:273 +msgid "Tray 1" +msgstr "" + +#: ppdc/sample.c:274 +msgid "Tray 2" +msgstr "" + +#: ppdc/sample.c:275 +msgid "Tray 3" +msgstr "" + +#: ppdc/sample.c:276 +msgid "Tray 4" +msgstr "" + +#: cups/http-support.c:1285 +msgid "URI Too Long" +msgstr "" + +#: ppdc/sample.c:138 +msgid "US Ledger" +msgstr "" + +#: ppdc/sample.c:139 +msgid "US Legal" +msgstr "" + +#: ppdc/sample.c:140 +msgid "US Legal Oversize" +msgstr "" + +#: ppdc/sample.c:141 +msgid "US Letter" +msgstr "" + +#: ppdc/sample.c:142 +msgid "US Letter Long Edge" +msgstr "" + +#: ppdc/sample.c:143 +msgid "US Letter Oversize" +msgstr "" + +#: ppdc/sample.c:144 +msgid "US Letter Oversize Long Edge" +msgstr "" + +#: ppdc/sample.c:145 +msgid "US Letter Small" +msgstr "" + +#: cgi-bin/admin.c:1940 cgi-bin/admin.c:1953 cgi-bin/admin.c:1977 +msgid "Unable to access cupsd.conf file:" +msgstr "" + +#: cgi-bin/admin.c:526 +msgid "Unable to add RSS subscription:" +msgstr "" + +#: cgi-bin/admin.c:814 +msgid "Unable to add class:" +msgstr "" + +#: backend/ipp.c:1542 +msgid "Unable to add document to print job." +msgstr "" + +#: scheduler/ipp.c:1569 +#, c-format +msgid "Unable to add job for destination \"%s\"." +msgstr "" + +#: cgi-bin/admin.c:1059 cgi-bin/admin.c:1419 +msgid "Unable to add printer:" +msgstr "" + +#: scheduler/ipp.c:1267 +msgid "Unable to allocate memory for file types." +msgstr "" + +#: filter/pstops.c:453 +msgid "Unable to allocate memory for page info" +msgstr "" + +#: filter/pstops.c:447 +msgid "Unable to allocate memory for pages array" +msgstr "" + +#: cgi-bin/admin.c:1525 +msgid "Unable to cancel RSS subscription:" +msgstr "" + +#: backend/ipp.c:1826 +msgid "Unable to cancel print job." +msgstr "" + +#: cgi-bin/admin.c:3884 +msgid "Unable to change printer-is-shared attribute:" +msgstr "" + +#: cgi-bin/admin.c:2968 +msgid "Unable to change printer:" +msgstr "" + +#: cgi-bin/admin.c:1638 cgi-bin/admin.c:1780 +msgid "Unable to change server settings:" +msgstr "" + +#: cups/adminutil.c:911 cups/request.c:1022 +msgid "Unable to connect to host." +msgstr "" + +#: backend/ipp.c:694 backend/ipp.c:1093 backend/lpd.c:801 backend/socket.c:333 +#: backend/usb-unix.c:117 +msgid "Unable to contact printer, queuing on next printer in class." +msgstr "" + +#: cups/adminutil.c:726 +#, c-format +msgid "Unable to copy 64-bit CUPS printer driver files (%d)." +msgstr "" + +#: cups/adminutil.c:691 +#, c-format +msgid "Unable to copy 64-bit Windows printer driver files (%d)." +msgstr "" + +#: cups/adminutil.c:522 +#, c-format +msgid "Unable to copy CUPS printer driver files (%d)." +msgstr "" + +#: scheduler/ipp.c:2733 +#, c-format +msgid "Unable to copy PPD file - %s" +msgstr "" + +#: scheduler/ipp.c:2788 +msgid "Unable to copy PPD file." +msgstr "" + +#: cups/adminutil.c:487 +#, c-format +msgid "Unable to copy Windows 2000 printer driver files (%d)." +msgstr "" + +#: cups/adminutil.c:610 +#, c-format +msgid "Unable to copy Windows 9x printer driver files (%d)." +msgstr "" + +#: scheduler/ipp.c:2710 +#, c-format +msgid "Unable to copy interface script - %s" +msgstr "" + +#: backend/ipp.c:1915 +msgid "Unable to create compressed print file" +msgstr "" + +#: cups/util.c:601 cups/util.c:1644 +msgid "Unable to create printer-uri" +msgstr "" + +#: scheduler/cupsfilter.c:1242 +msgid "Unable to create temporary file" +msgstr "" + +#: cgi-bin/admin.c:1831 cgi-bin/admin.c:1843 +msgid "Unable to create temporary file:" +msgstr "" + +#: cgi-bin/admin.c:2134 +msgid "Unable to delete class:" +msgstr "" + +#: cgi-bin/admin.c:2219 +msgid "Unable to delete printer:" +msgstr "" + +#: cgi-bin/classes.c:260 cgi-bin/printers.c:269 +msgid "Unable to do maintenance command:" +msgstr "" + +#: cgi-bin/admin.c:1955 +msgid "Unable to edit cupsd.conf files larger than 1MB" +msgstr "" + +#: cups/http.c:4209 +msgid "Unable to establish a secure connection to host (certificate chain invalid)." +msgstr "" + +#: cups/http.c:4199 +msgid "Unable to establish a secure connection to host (certificate not yet valid)." +msgstr "" + +#: cups/http.c:4194 +msgid "Unable to establish a secure connection to host (expired certificate)." +msgstr "" + +#: cups/http.c:4204 +msgid "Unable to establish a secure connection to host (host name mismatch)." +msgstr "" + +#: cups/http.c:4214 +msgid "Unable to establish a secure connection to host (peer dropped connection before responding)." +msgstr "" + +#: cups/http.c:4189 +msgid "Unable to establish a secure connection to host (self-signed certificate)." +msgstr "" + +#: cups/http.c:4184 +msgid "Unable to establish a secure connection to host (untrusted certificate)." +msgstr "" + +#: cups/http.c:3937 cups/http.c:4241 cups/http.c:4274 cups/http.c:4291 +msgid "Unable to establish a secure connection to host." +msgstr "" + +#: cgi-bin/ipp-var.c:365 +msgid "Unable to find destination for job" +msgstr "" + +#: cups/http-support.c:1631 +msgid "Unable to find printer." +msgstr "" + +#: backend/ipp.c:1937 +msgid "Unable to generate compressed print file" +msgstr "" + +#: backend/ipp.c:2739 +msgid "Unable to get backend exit status." +msgstr "" + +#: cgi-bin/classes.c:450 +msgid "Unable to get class list:" +msgstr "" + +#: cgi-bin/classes.c:549 +msgid "Unable to get class status:" +msgstr "" + +#: cgi-bin/admin.c:1320 +msgid "Unable to get list of printer drivers:" +msgstr "" + +#: backend/ipp.c:1633 +msgid "Unable to get print job status." +msgstr "" + +#: cgi-bin/admin.c:2818 +msgid "Unable to get printer attributes:" +msgstr "" + +#: cgi-bin/printers.c:467 +msgid "Unable to get printer list:" +msgstr "" + +#: backend/ipp.c:906 +msgid "Unable to get printer status." +msgstr "" + +#: cgi-bin/printers.c:569 +msgid "Unable to get printer status:" +msgstr "" + +#: cups/adminutil.c:565 cups/adminutil.c:769 +#, c-format +msgid "Unable to install Windows 2000 printer driver files (%d)." +msgstr "" + +#: cups/adminutil.c:639 +#, c-format +msgid "Unable to install Windows 9x printer driver files (%d)." +msgstr "" + +#: backend/ipp.c:626 backend/lpd.c:419 backend/socket.c:275 +#, c-format +msgid "Unable to locate printer \"%s\"." +msgstr "" + +#: backend/dnssd.c:529 backend/ipp.c:311 backend/lpd.c:202 +#: backend/socket.c:171 +msgid "Unable to locate printer." +msgstr "" + +#: cgi-bin/admin.c:813 +msgid "Unable to modify class:" +msgstr "" + +#: cgi-bin/admin.c:1058 cgi-bin/admin.c:1418 +msgid "Unable to modify printer:" +msgstr "" + +#: cgi-bin/ipp-var.c:432 cgi-bin/ipp-var.c:521 +msgid "Unable to move job" +msgstr "" + +#: cgi-bin/ipp-var.c:434 cgi-bin/ipp-var.c:523 +msgid "Unable to move jobs" +msgstr "" + +#: cups/ppd.c:339 +msgid "Unable to open PPD file" +msgstr "" + +#: cgi-bin/admin.c:3164 +msgid "Unable to open PPD file:" +msgstr "" + +#: backend/ipp.c:1921 +msgid "Unable to open compressed print file" +msgstr "" + +#: cgi-bin/admin.c:2589 +msgid "Unable to open cupsd.conf file:" +msgstr "" + +#: backend/usb-unix.c:145 +msgid "Unable to open device file" +msgstr "" + +#: scheduler/ipp.c:7029 +#, c-format +msgid "Unable to open document #%d in job #%d." +msgstr "" + +#: backend/ipp.c:352 backend/ipp.c:1927 backend/lpd.c:486 backend/socket.c:158 +#: backend/usb.c:237 filter/gziptoany.c:71 filter/pstops.c:302 +msgid "Unable to open print file" +msgstr "" + +#: filter/rastertoepson.c:998 filter/rastertohp.c:724 +#: filter/rastertolabel.c:1147 +msgid "Unable to open raster file" +msgstr "" + +#: cgi-bin/ipp-var.c:795 +msgid "Unable to print test page:" +msgstr "" + +#: backend/runloop.c:95 backend/runloop.c:322 +msgid "Unable to read print data" +msgstr "" + +#: backend/usb-darwin.c:613 backend/usb-darwin.c:657 +msgid "Unable to read print data." +msgstr "" + +#: cups/adminutil.c:2088 +#, c-format +msgid "Unable to run \"%s\": %s" +msgstr "" + +#: filter/pstops.c:565 +msgid "Unable to see in file" +msgstr "" + +#: cgi-bin/ipp-var.c:598 cgi-bin/ipp-var.c:618 +msgid "Unable to send command to printer driver" +msgstr "" + +#: backend/usb-darwin.c:735 backend/usb-libusb.c:179 backend/usb-libusb.c:872 +msgid "Unable to send data to printer." +msgstr "" + +#: cups/adminutil.c:821 +#, c-format +msgid "Unable to set Windows printer driver (%d)." +msgstr "" + +#: cgi-bin/admin.c:3785 +msgid "Unable to set options:" +msgstr "" + +#: cgi-bin/admin.c:3055 +msgid "Unable to set server default:" +msgstr "" + +#: backend/ipp.c:2598 backend/ipp.c:2675 backend/ipp.c:2683 +msgid "Unable to start backend process." +msgstr "" + +#: cgi-bin/admin.c:1893 +msgid "Unable to upload cupsd.conf file:" +msgstr "" + +#: backend/usb-darwin.c:1985 backend/usb-darwin.c:2009 +msgid "Unable to use legacy USB class driver." +msgstr "" + +#: backend/runloop.c:124 backend/runloop.c:377 +msgid "Unable to write print data" +msgstr "" + +#: filter/gziptoany.c:90 +#, c-format +msgid "Unable to write uncompressed print data: %s" +msgstr "" + +#: cups/http-support.c:1273 +msgid "Unauthorized" +msgstr "" + +#: cgi-bin/admin.c:3481 +msgid "Units" +msgstr "" + +#: cups/http-support.c:1313 cups/ppd.c:366 +msgid "Unknown" +msgstr "" + +#: filter/pstops.c:2225 +#, c-format +msgid "Unknown choice \"%s\" for option \"%s\"." +msgstr "" + +#: backend/ipp.c:494 +#, c-format +msgid "Unknown encryption option value: \"%s\"." +msgstr "" + +#: backend/lpd.c:348 +#, c-format +msgid "Unknown file order: \"%s\"." +msgstr "" + +#: backend/lpd.c:319 +#, c-format +msgid "Unknown format character: \"%c\"." +msgstr "" + +#: backend/ipp.c:541 +#, c-format +msgid "Unknown option \"%s\" with value \"%s\"." +msgstr "" + +#: filter/pstops.c:2208 +#, c-format +msgid "Unknown option \"%s\"." +msgstr "" + +#: backend/lpd.c:334 +#, c-format +msgid "Unknown print mode: \"%s\"." +msgstr "" + +#: scheduler/ipp.c:11353 +#, c-format +msgid "Unknown printer-error-policy \"%s\"." +msgstr "" + +#: scheduler/ipp.c:11336 +#, c-format +msgid "Unknown printer-op-policy \"%s\"." +msgstr "" + +#: backend/ipp.c:513 +#, c-format +msgid "Unknown version option value: \"%s\"." +msgstr "" + +#: filter/pstops.c:2416 +#, c-format +msgid "Unsupported brightness value %s, using brightness=100." +msgstr "" + +#: scheduler/ipp.c:423 +#, c-format +msgid "Unsupported character set \"%s\"." +msgstr "" + +#: scheduler/ipp.c:9060 scheduler/ipp.c:10267 scheduler/ipp.c:11807 +#, c-format +msgid "Unsupported compression \"%s\"." +msgstr "" + +#: scheduler/ipp.c:9194 scheduler/ipp.c:10417 scheduler/ipp.c:11836 +#, c-format +msgid "Unsupported document-format \"%s\"." +msgstr "" + +#: scheduler/ipp.c:10400 +#, c-format +msgid "Unsupported document-format \"%s/%s\"." +msgstr "" + +#: scheduler/ipp.c:1388 +#, c-format +msgid "Unsupported format \"%s\"." +msgstr "" + +#: filter/pstops.c:2498 +#, c-format +msgid "Unsupported gamma value %s, using gamma=1000." +msgstr "" + +#: scheduler/ipp.c:1486 +msgid "Unsupported margins." +msgstr "" + +#: cups/pwg-media.c:467 +msgid "Unsupported media value." +msgstr "" + +#: filter/pstops.c:2542 +#, c-format +msgid "Unsupported number-up value %d, using number-up=1." +msgstr "" + +#: filter/pstops.c:2576 +#, c-format +msgid "Unsupported number-up-layout value %s, using number-up-layout=lrtb." +msgstr "" + +#: filter/pstops.c:2627 +#, c-format +msgid "Unsupported page-border value %s, using page-border=none." +msgstr "" + +#: filter/rastertopwg.c:147 filter/rastertopwg.c:155 filter/rastertopwg.c:164 +msgid "Unsupported raster data." +msgstr "" + +#: cups/snmp.c:1112 +msgid "Unsupported value type" +msgstr "" + +#: cups/http-support.c:1288 +msgid "Upgrade Required" +msgstr "" + +#: systemv/lpadmin.c:668 +msgid "" +"Usage:\n" +"\n" +" lpadmin [-h server] -d destination\n" +" lpadmin [-h server] -x destination\n" +" lpadmin [-h server] -p printer [-c add-class] [-i interface] [-m model]\n" +" [-r remove-class] [-v device] [-D description]\n" +" [-P ppd-file] [-o name=value]\n" +" [-u allow:user,user] [-u deny:user,user]" +msgstr "" + +#: backend/dnssd.c:174 backend/ipp.c:300 backend/lpd.c:191 +#: backend/socket.c:135 backend/usb.c:183 monitor/bcp.c:62 monitor/tbcp.c:61 +#, c-format +msgid "Usage: %s job-id user title copies options [file]" +msgstr "" + +#: filter/commandtops.c:73 filter/gziptoany.c:50 filter/pstops.c:266 +#, c-format +msgid "Usage: %s job-id user title copies options file" +msgstr "" + +#: scheduler/cupsfilter.c:1464 +msgid "Usage: convert [ options ]" +msgstr "" + +#: systemv/cupsaddsmb.c:281 +msgid "Usage: cupsaddsmb [options] printer1 ... printerN" +msgstr "" + +#: systemv/cupsctl.c:200 +msgid "Usage: cupsctl [options] [param=value ... paramN=valueN]" +msgstr "" + +#: scheduler/main.c:1972 +msgid "Usage: cupsd [options]" +msgstr "" + +#: scheduler/cupsfilter.c:1437 +msgid "Usage: cupsfilter [ options ] filename" +msgstr "" + +#: systemv/cupstestdsc.c:425 +msgid "Usage: cupstestdsc [options] filename.ps [... filename.ps]" +msgstr "" + +#: systemv/cupstestppd.c:3773 +msgid "Usage: cupstestppd [options] filename1.ppd[.gz] [... filenameN.ppd[.gz]]" +msgstr "" + +#: test/ipptool.c:4379 +msgid "Usage: ipptool [options] URI filename [ ... filenameN ]" +msgstr "" + +#: systemv/lpmove.c:125 +msgid "Usage: lpmove job/src dest" +msgstr "" + +#: systemv/lpoptions.c:553 +msgid "" +"Usage: lpoptions [-h server] [-E] -d printer\n" +" lpoptions [-h server] [-E] [-p printer] -l\n" +" lpoptions [-h server] [-E] -p printer -o option[=value] ...\n" +" lpoptions [-h server] [-E] -x printer" +msgstr "" + +#: systemv/lppasswd.c:476 +msgid "Usage: lppasswd [-g groupname]" +msgstr "" + +#: systemv/lppasswd.c:479 +msgid "" +"Usage: lppasswd [-g groupname] [username]\n" +" lppasswd [-g groupname] -a [username]\n" +" lppasswd [-g groupname] -x [username]" +msgstr "" + +#: berkeley/lpq.c:670 +msgid "Usage: lpq [-P dest] [-U username] [-h hostname[:port]] [-l] [+interval]" +msgstr "" + +#: ppdc/ppdc.cxx:435 +msgid "Usage: ppdc [options] filename.drv [ ... filenameN.drv ]" +msgstr "" + +#: ppdc/ppdhtml.cxx:172 +msgid "Usage: ppdhtml [options] filename.drv >filename.html" +msgstr "" + +#: ppdc/ppdi.cxx:128 +msgid "Usage: ppdi [options] filename.ppd [ ... filenameN.ppd ]" +msgstr "" + +#: ppdc/ppdmerge.cxx:367 +msgid "Usage: ppdmerge [options] filename.ppd [ ... filenameN.ppd ]" +msgstr "" + +#: ppdc/ppdpo.cxx:252 +msgid "Usage: ppdpo [options] -o filename.po filename.drv [ ... filenameN.drv ]" +msgstr "" + +#: backend/snmp.c:218 +msgid "Usage: snmp [host-or-ip-address]" +msgstr "" + +#: cups/snmp.c:1064 +msgid "Value uses indefinite length" +msgstr "" + +#: cups/snmp.c:1049 +msgid "VarBind uses indefinite length" +msgstr "" + +#: cups/snmp.c:999 +msgid "Version uses indefinite length" +msgstr "" + +#: backend/ipp.c:1567 +msgid "Waiting for job to complete." +msgstr "" + +#: backend/usb-darwin.c:457 backend/usb-libusb.c:118 +msgid "Waiting for printer to become available." +msgstr "" + +#: backend/socket.c:444 +msgid "Waiting for printer to finish." +msgstr "" + +#: cups/adminutil.c:793 +msgid "Warning, no Windows 2000 printer drivers are installed." +msgstr "" + +#: cups/http-support.c:1309 +msgid "Web Interface is Disabled" +msgstr "" + +#: cups/ppd.c:1947 +msgid "Yes" +msgstr "" + +#: scheduler/client.c:2508 +#, c-format +msgid "You must access this page using the URL https://%s:%d%s." +msgstr "" + +#: systemv/lppasswd.c:254 +msgid "Your password must be at least 6 characters long, cannot contain your username, and must contain at least one letter and number." +msgstr "" + +#: ppdc/sample.c:448 +msgid "ZPL Label Printer" +msgstr "" + +#: ppdc/sample.c:371 +msgid "Zebra" +msgstr "" + +#: cups/notify.c:102 +msgid "aborted" +msgstr "" + +#: cups/notify.c:99 +msgid "canceled" +msgstr "" + +#: cups/notify.c:105 +msgid "completed" +msgstr "" + +#: scheduler/cupsfilter.c:356 +msgid "convert: Use the -f option to specify a file to convert." +msgstr "" + +#: scheduler/ipp.c:6901 +msgid "cups-deviced failed to execute." +msgstr "" + +#: scheduler/ipp.c:7731 scheduler/ipp.c:7980 +msgid "cups-driverd failed to execute." +msgstr "" + +#: systemv/cupsaddsmb.c:233 +#, c-format +msgid "cupsaddsmb: No PPD file for printer \"%s\" - %s" +msgstr "" + +#: systemv/cupsctl.c:141 +msgid "cupsctl: Cannot set Listen or Port directly." +msgstr "" + +#: systemv/cupsctl.c:152 +#, c-format +msgid "cupsctl: Unable to connect to server: %s" +msgstr "" + +#: systemv/cupsctl.c:195 +#, c-format +msgid "cupsctl: Unknown option \"%s\"" +msgstr "" + +#: systemv/cupsctl.c:197 +#, c-format +msgid "cupsctl: Unknown option \"-%c\"" +msgstr "" + +#: scheduler/main.c:189 +msgid "cupsd: Expected config filename after \"-c\" option." +msgstr "" + +#: scheduler/main.c:221 scheduler/main.c:228 +msgid "cupsd: Unable to get current directory." +msgstr "" + +#: scheduler/main.c:295 +#, c-format +msgid "cupsd: Unknown argument \"%s\" - aborting." +msgstr "" + +#: scheduler/main.c:288 +#, c-format +msgid "cupsd: Unknown option \"%c\" - aborting." +msgstr "" + +#: scheduler/main.c:255 +msgid "cupsd: launchd(8) support not compiled in, running in normal mode." +msgstr "" + +#: scheduler/cupsfilter.c:1215 +#, c-format +msgid "cupsfilter: Invalid document number %d." +msgstr "" + +#: scheduler/cupsfilter.c:1209 +#, c-format +msgid "cupsfilter: Invalid job ID %d." +msgstr "" + +#: scheduler/cupsfilter.c:364 +msgid "cupsfilter: Only one filename can be specified." +msgstr "" + +#: scheduler/cupsfilter.c:1257 +#, c-format +msgid "cupsfilter: Unable to get job file - %s" +msgstr "" + +#: systemv/cupstestppd.c:260 +msgid "cupstestppd: The -q option is incompatible with the -v option." +msgstr "" + +#: systemv/cupstestppd.c:276 +msgid "cupstestppd: The -v option is incompatible with the -q option." +msgstr "" + +#: systemv/lpstat.c:1231 systemv/lpstat.c:1234 systemv/lpstat.c:1237 +#, c-format +msgid "device for %s/%s: %s" +msgstr "" + +#: systemv/lpstat.c:1218 systemv/lpstat.c:1221 systemv/lpstat.c:1224 +#, c-format +msgid "device for %s: %s" +msgstr "" + +#: cups/snmp.c:1036 +msgid "error-index uses indefinite length" +msgstr "" + +#: cups/snmp.c:1028 +msgid "error-status uses indefinite length" +msgstr "" + +#: cups/notify.c:90 +msgid "held" +msgstr "" + +#: berkeley/lpc.c:209 +msgid "help\t\tGet help on commands." +msgstr "" + +#: cups/notify.c:131 +msgid "idle" +msgstr "" + +#: test/ipptool.c:363 +msgid "ipptool: \"-i\" and \"-n\" are incompatible with -X\"." +msgstr "" + +#: test/ipptool.c:498 +msgid "ipptool: \"-i\" is incompatible with \"-X\"." +msgstr "" + +#: test/ipptool.c:522 +msgid "ipptool: \"-n\" is incompatible with \"-X\"." +msgstr "" + +#: test/ipptool.c:580 +#, c-format +msgid "ipptool: Bad URI - %s." +msgstr "" + +#: test/ipptool.c:352 +#, c-format +msgid "ipptool: Bad version %s for \"-V\"." +msgstr "" + +#: test/ipptool.c:491 +msgid "ipptool: Invalid seconds for \"-i\"." +msgstr "" + +#: test/ipptool.c:561 +msgid "ipptool: May only specify a single URI." +msgstr "" + +#: test/ipptool.c:514 +msgid "ipptool: Missing count for \"-n\"." +msgstr "" + +#: test/ipptool.c:398 +msgid "ipptool: Missing filename for \"-f\"." +msgstr "" + +#: test/ipptool.c:379 +msgid "ipptool: Missing name=value for \"-d\"." +msgstr "" + +#: test/ipptool.c:481 +msgid "ipptool: Missing seconds for \"-i\"." +msgstr "" + +#: test/ipptool.c:322 +msgid "ipptool: Missing timeout for \"-T\"." +msgstr "" + +#: test/ipptool.c:335 +msgid "ipptool: Missing version for \"-V\"." +msgstr "" + +#: test/ipptool.c:607 +msgid "ipptool: URI required before test file." +msgstr "" + +#: test/ipptool.c:541 +#, c-format +msgid "ipptool: Unknown option \"-%c\"." +msgstr "" + +#: scheduler/ipp.c:8727 +msgid "job-printer-uri attribute missing." +msgstr "" + +#: systemv/lpadmin.c:131 systemv/lpadmin.c:375 +msgid "lpadmin: Class name can only contain printable characters." +msgstr "" + +#: systemv/lpadmin.c:614 +msgid "lpadmin: Expected PPD after \"-P\" option." +msgstr "" + +#: systemv/lpadmin.c:457 +msgid "lpadmin: Expected allow/deny:userlist after \"-u\" option." +msgstr "" + +#: systemv/lpadmin.c:364 +msgid "lpadmin: Expected class after \"-r\" option." +msgstr "" + +#: systemv/lpadmin.c:120 +msgid "lpadmin: Expected class name after \"-c\" option." +msgstr "" + +#: systemv/lpadmin.c:558 +msgid "lpadmin: Expected description after \"-D\" option." +msgstr "" + +#: systemv/lpadmin.c:491 +msgid "lpadmin: Expected device URI after \"-v\" option." +msgstr "" + +#: systemv/lpadmin.c:574 +msgid "lpadmin: Expected file type(s) after \"-I\" option." +msgstr "" + +#: systemv/lpadmin.c:202 +msgid "lpadmin: Expected hostname after \"-h\" option." +msgstr "" + +#: systemv/lpadmin.c:221 +msgid "lpadmin: Expected interface after \"-i\" option." +msgstr "" + +#: systemv/lpadmin.c:594 +msgid "lpadmin: Expected location after \"-L\" option." +msgstr "" + +#: systemv/lpadmin.c:274 +msgid "lpadmin: Expected model after \"-m\" option." +msgstr "" + +#: systemv/lpadmin.c:417 +msgid "lpadmin: Expected name after \"-R\" option." +msgstr "" + +#: systemv/lpadmin.c:294 +msgid "lpadmin: Expected name=value after \"-o\" option." +msgstr "" + +#: systemv/lpadmin.c:313 +msgid "lpadmin: Expected printer after \"-p\" option." +msgstr "" + +#: systemv/lpadmin.c:164 +msgid "lpadmin: Expected printer name after \"-d\" option." +msgstr "" + +#: systemv/lpadmin.c:525 +msgid "lpadmin: Expected printer or class after \"-x\" option." +msgstr "" + +#: systemv/lpadmin.c:975 +msgid "lpadmin: No member names were seen." +msgstr "" + +#: systemv/lpadmin.c:762 +#, c-format +msgid "lpadmin: Printer %s is already a member of class %s." +msgstr "" + +#: systemv/lpadmin.c:989 +#, c-format +msgid "lpadmin: Printer %s is not a member of class %s." +msgstr "" + +#: systemv/lpadmin.c:175 systemv/lpadmin.c:324 systemv/lpadmin.c:536 +msgid "lpadmin: Printer name can only contain printable characters." +msgstr "" + +#: systemv/lpadmin.c:105 +msgid "" +"lpadmin: Unable to add a printer to the class:\n" +" You must specify a printer name first." +msgstr "" + +#: systemv/lpadmin.c:96 systemv/lpadmin.c:149 systemv/lpadmin.c:253 +#: systemv/lpadmin.c:339 systemv/lpadmin.c:393 systemv/lpadmin.c:510 +#: systemv/lpadmin.c:647 +#, c-format +msgid "lpadmin: Unable to connect to server: %s" +msgstr "" + +#: systemv/lpadmin.c:1329 +msgid "lpadmin: Unable to create temporary file" +msgstr "" + +#: systemv/lpadmin.c:402 +msgid "" +"lpadmin: Unable to delete option:\n" +" You must specify a printer name first." +msgstr "" + +#: systemv/lpadmin.c:1339 +#, c-format +msgid "lpadmin: Unable to open PPD file \"%s\" - %s" +msgstr "" + +#: systemv/lpadmin.c:348 +msgid "" +"lpadmin: Unable to remove a printer from the class:\n" +" You must specify a printer name first." +msgstr "" + +#: systemv/lpadmin.c:656 +msgid "" +"lpadmin: Unable to set the printer options:\n" +" You must specify a printer name first." +msgstr "" + +#: systemv/lpadmin.c:474 +#, c-format +msgid "lpadmin: Unknown allow/deny option \"%s\"." +msgstr "" + +#: systemv/lpadmin.c:629 +#, c-format +msgid "lpadmin: Unknown argument \"%s\"." +msgstr "" + +#: systemv/lpadmin.c:624 +#, c-format +msgid "lpadmin: Unknown option \"%c\"." +msgstr "" + +#: systemv/lpadmin.c:580 +msgid "lpadmin: Warning - content type list ignored." +msgstr "" + +#: berkeley/lpc.c:76 berkeley/lpc.c:104 berkeley/lpc.c:140 +msgid "lpc> " +msgstr "" + +#: systemv/lpinfo.c:137 +msgid "lpinfo: Expected 1284 device ID string after \"--device-id\"." +msgstr "" + +#: systemv/lpinfo.c:190 +msgid "lpinfo: Expected language after \"--language\"." +msgstr "" + +#: systemv/lpinfo.c:207 +msgid "lpinfo: Expected make and model after \"--make-and-model\"." +msgstr "" + +#: systemv/lpinfo.c:224 +msgid "lpinfo: Expected product string after \"--product\"." +msgstr "" + +#: systemv/lpinfo.c:155 +msgid "lpinfo: Expected scheme list after \"--exclude-schemes\"." +msgstr "" + +#: systemv/lpinfo.c:173 +msgid "lpinfo: Expected scheme list after \"--include-schemes\"." +msgstr "" + +#: systemv/lpinfo.c:241 +msgid "lpinfo: Expected timeout after \"--timeout\"." +msgstr "" + +#: systemv/lpinfo.c:265 +#, c-format +msgid "lpinfo: Unknown argument \"%s\"." +msgstr "" + +#: systemv/lpinfo.c:259 +#, c-format +msgid "lpinfo: Unknown option \"%c\"." +msgstr "" + +#: systemv/lpinfo.c:252 +#, c-format +msgid "lpinfo: Unknown option \"%s\"." +msgstr "" + +#: systemv/lpmove.c:133 +#, c-format +msgid "lpmove: Unable to connect to server: %s" +msgstr "" + +#: systemv/lpmove.c:119 +#, c-format +msgid "lpmove: Unknown argument \"%s\"." +msgstr "" + +#: systemv/lpmove.c:97 +#, c-format +msgid "lpmove: Unknown option \"%c\"." +msgstr "" + +#: systemv/lpoptions.c:150 systemv/lpoptions.c:168 systemv/lpoptions.c:244 +msgid "lpoptions: No printers." +msgstr "" + +#: systemv/lpoptions.c:219 +#, c-format +msgid "lpoptions: Unable to add printer or instance: %s" +msgstr "" + +#: systemv/lpoptions.c:521 +#, c-format +msgid "lpoptions: Unable to get PPD file for %s: %s" +msgstr "" + +#: systemv/lpoptions.c:529 +#, c-format +msgid "lpoptions: Unable to open PPD file for %s." +msgstr "" + +#: systemv/lpoptions.c:99 +msgid "lpoptions: Unknown printer or class." +msgstr "" + +#: systemv/lppasswd.c:173 +msgid "lppasswd: Only root can add or delete passwords." +msgstr "" + +#: systemv/lppasswd.c:302 +msgid "lppasswd: Password file busy." +msgstr "" + +#: systemv/lppasswd.c:431 +msgid "lppasswd: Password file not updated." +msgstr "" + +#: systemv/lppasswd.c:398 +msgid "lppasswd: Sorry, password doesn't match." +msgstr "" + +#: systemv/lppasswd.c:253 +msgid "lppasswd: Sorry, password rejected." +msgstr "" + +#: systemv/lppasswd.c:230 +msgid "lppasswd: Sorry, passwords don't match." +msgstr "" + +#: systemv/lppasswd.c:199 systemv/lppasswd.c:218 +#, c-format +msgid "lppasswd: Unable to copy password string: %s" +msgstr "" + +#: systemv/lppasswd.c:304 systemv/lppasswd.c:312 systemv/lppasswd.c:329 +#, c-format +msgid "lppasswd: Unable to open password file: %s" +msgstr "" + +#: systemv/lppasswd.c:364 systemv/lppasswd.c:377 systemv/lppasswd.c:408 +#, c-format +msgid "lppasswd: Unable to write to password file: %s" +msgstr "" + +#: systemv/lppasswd.c:446 +#, c-format +msgid "lppasswd: failed to backup old password file: %s" +msgstr "" + +#: systemv/lppasswd.c:458 +#, c-format +msgid "lppasswd: failed to rename password file: %s" +msgstr "" + +#: systemv/lppasswd.c:389 +#, c-format +msgid "lppasswd: user \"%s\" and group \"%s\" do not exist." +msgstr "" + +#: systemv/lpstat.c:1039 +#, c-format +msgid "lpstat: error - %s environment variable names non-existent destination \"%s\"." +msgstr "" + +#: systemv/lpstat.c:970 +#, c-format +msgid "members of class %s:" +msgstr "" + +#: berkeley/lpq.c:582 +msgid "no entries" +msgstr "" + +#: systemv/lpstat.c:1043 +msgid "no system default destination" +msgstr "" + +#: scheduler/ipp.c:6589 +msgid "notify-events not specified." +msgstr "" + +#: scheduler/ipp.c:2118 scheduler/ipp.c:6494 +#, c-format +msgid "notify-recipient-uri URI \"%s\" is already used." +msgstr "" + +#: scheduler/ipp.c:2108 scheduler/ipp.c:6484 +#, c-format +msgid "notify-recipient-uri URI \"%s\" uses unknown scheme." +msgstr "" + +#: cups/notify.c:87 +msgid "pending" +msgstr "" + +#: ppdc/ppdc.cxx:113 ppdc/ppdpo.cxx:93 +#, c-format +msgid "ppdc: Adding include directory \"%s\"." +msgstr "" + +#: ppdc/ppdpo.cxx:134 +#, c-format +msgid "ppdc: Adding/updating UI text from %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:410 +#, c-format +msgid "ppdc: Bad boolean value (%s) on line %d of %s." +msgstr "" + +#: ppdc/ppdc-import.cxx:264 +#, c-format +msgid "ppdc: Bad font attribute: %s" +msgstr "" + +#: ppdc/ppdc-source.cxx:1795 +#, c-format +msgid "ppdc: Bad resolution name \"%s\" on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1113 +#, c-format +msgid "ppdc: Bad status keyword %s on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:2032 +#, c-format +msgid "ppdc: Bad variable substitution ($%c) on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:2718 +#, c-format +msgid "ppdc: Choice found on line %d of %s with no Option." +msgstr "" + +#: ppdc/ppdc-source.cxx:1697 +#, c-format +msgid "ppdc: Duplicate #po for locale %s on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:932 +#, c-format +msgid "ppdc: Expected a filter definition on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:955 +#, c-format +msgid "ppdc: Expected a program name on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:394 +#, c-format +msgid "ppdc: Expected boolean value on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1093 +#, c-format +msgid "ppdc: Expected charset after Font on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:447 +#, c-format +msgid "ppdc: Expected choice code on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:435 +#, c-format +msgid "ppdc: Expected choice name/text on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:503 +#, c-format +msgid "ppdc: Expected color order for ColorModel on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:492 +#, c-format +msgid "ppdc: Expected colorspace for ColorModel on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:514 +#, c-format +msgid "ppdc: Expected compression for ColorModel on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:695 +#, c-format +msgid "ppdc: Expected constraints string for UIConstraints on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:2904 +#, c-format +msgid "ppdc: Expected driver type keyword following DriverType on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:826 +#, c-format +msgid "ppdc: Expected duplex type after Duplex on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1077 +#, c-format +msgid "ppdc: Expected encoding after Font on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1688 +#, c-format +msgid "ppdc: Expected filename after #po %s on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1205 +#, c-format +msgid "ppdc: Expected group name/text on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:2618 +#, c-format +msgid "ppdc: Expected include filename on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1501 +#, c-format +msgid "ppdc: Expected integer on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1680 +#, c-format +msgid "ppdc: Expected locale after #po on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:353 +#, c-format +msgid "ppdc: Expected name after %s on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:3276 +#, c-format +msgid "ppdc: Expected name after FileName on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1058 +#, c-format +msgid "ppdc: Expected name after Font on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:3107 +#, c-format +msgid "ppdc: Expected name after Manufacturer on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:3140 +#, c-format +msgid "ppdc: Expected name after MediaSize on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:3230 +#, c-format +msgid "ppdc: Expected name after ModelName on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:3293 +#, c-format +msgid "ppdc: Expected name after PCFileName on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1156 +#, c-format +msgid "ppdc: Expected name/text after %s on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1245 +#, c-format +msgid "ppdc: Expected name/text after Installable on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1781 +#, c-format +msgid "ppdc: Expected name/text after Resolution on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:479 +#, c-format +msgid "ppdc: Expected name/text combination for ColorModel on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1573 +#, c-format +msgid "ppdc: Expected option name/text on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1607 +#, c-format +msgid "ppdc: Expected option section on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1585 +#, c-format +msgid "ppdc: Expected option type on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1764 +#, c-format +msgid "ppdc: Expected override field after Resolution on line %d of %s." +msgstr "" + +#: ppdc/ppdc-catalog.cxx:341 ppdc/ppdc-catalog.cxx:353 +#, c-format +msgid "ppdc: Expected quoted string on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1004 +#, c-format +msgid "ppdc: Expected real number on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:572 +#, c-format +msgid "ppdc: Expected resolution/mediatype following ColorProfile on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1862 +#, c-format +msgid "ppdc: Expected resolution/mediatype following SimpleColorProfile on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:361 +#, c-format +msgid "ppdc: Expected selector after %s on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1101 +#, c-format +msgid "ppdc: Expected status after Font on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:2793 +#, c-format +msgid "ppdc: Expected string after Copyright on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:3396 +#, c-format +msgid "ppdc: Expected string after Version on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:728 +#, c-format +msgid "ppdc: Expected two option names on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:372 +#, c-format +msgid "ppdc: Expected value after %s on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1085 +#, c-format +msgid "ppdc: Expected version after Font on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:227 +#, c-format +msgid "ppdc: Invalid #include/#po filename \"%s\"." +msgstr "" + +#: ppdc/ppdc-source.cxx:972 +#, c-format +msgid "ppdc: Invalid cost for filter on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:964 +#, c-format +msgid "ppdc: Invalid empty MIME type for filter on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:980 +#, c-format +msgid "ppdc: Invalid empty program name for filter on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1627 +#, c-format +msgid "ppdc: Invalid option section \"%s\" on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1599 +#, c-format +msgid "ppdc: Invalid option type \"%s\" on line %d of %s." +msgstr "" + +#: ppdc/ppdc.cxx:251 ppdc/ppdpo.cxx:123 +#, c-format +msgid "ppdc: Loading driver information file \"%s\"." +msgstr "" + +#: ppdc/ppdc.cxx:187 +#, c-format +msgid "ppdc: Loading messages for locale \"%s\"." +msgstr "" + +#: ppdc/ppdc.cxx:126 +#, c-format +msgid "ppdc: Loading messages from \"%s\"." +msgstr "" + +#: ppdc/ppdc-source.cxx:2411 ppdc/ppdc-source.cxx:2643 +#, c-format +msgid "ppdc: Missing #endif at end of \"%s\"." +msgstr "" + +#: ppdc/ppdc-source.cxx:2512 ppdc/ppdc-source.cxx:2547 +#: ppdc/ppdc-source.cxx:2577 +#, c-format +msgid "ppdc: Missing #if on line %d of %s." +msgstr "" + +#: ppdc/ppdc-catalog.cxx:418 +#, c-format +msgid "ppdc: Need a msgid line before any translation strings on line %d of %s." +msgstr "" + +#: ppdc/ppdc-driver.cxx:730 +#, c-format +msgid "ppdc: No message catalog provided for locale %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1650 ppdc/ppdc-source.cxx:2881 +#: ppdc/ppdc-source.cxx:2967 ppdc/ppdc-source.cxx:3060 +#: ppdc/ppdc-source.cxx:3193 ppdc/ppdc-source.cxx:3326 +#, c-format +msgid "ppdc: Option %s defined in two different groups on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1643 +#, c-format +msgid "ppdc: Option %s redefined with a different type on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:705 +#, c-format +msgid "ppdc: Option constraint must *name on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:2494 +#, c-format +msgid "ppdc: Too many nested #if's on line %d of %s." +msgstr "" + +#: ppdc/ppdc.cxx:374 +#, c-format +msgid "ppdc: Unable to create PPD file \"%s\" - %s." +msgstr "" + +#: ppdc/ppdc.cxx:266 +#, c-format +msgid "ppdc: Unable to create output directory %s: %s" +msgstr "" + +#: ppdc/ppdc.cxx:287 +#, c-format +msgid "ppdc: Unable to create output pipes: %s" +msgstr "" + +#: ppdc/ppdc.cxx:303 ppdc/ppdc.cxx:309 +#, c-format +msgid "ppdc: Unable to execute cupstestppd: %s" +msgstr "" + +#: ppdc/ppdc-source.cxx:1729 +#, c-format +msgid "ppdc: Unable to find #po file %s on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:2650 +#, c-format +msgid "ppdc: Unable to find include file \"%s\" on line %d of %s." +msgstr "" + +#: ppdc/ppdc.cxx:198 +#, c-format +msgid "ppdc: Unable to find localization for \"%s\" - %s" +msgstr "" + +#: ppdc/ppdc.cxx:135 +#, c-format +msgid "ppdc: Unable to load localization file \"%s\" - %s" +msgstr "" + +#: ppdc/ppdc-file.cxx:49 +#, c-format +msgid "ppdc: Unable to open %s: %s" +msgstr "" + +#: ppdc/ppdc-source.cxx:2053 +#, c-format +msgid "ppdc: Undefined variable (%s) on line %d of %s." +msgstr "" + +#: ppdc/ppdc-catalog.cxx:435 +#, c-format +msgid "ppdc: Unexpected text on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:2923 +#, c-format +msgid "ppdc: Unknown driver type %s on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:906 +#, c-format +msgid "ppdc: Unknown duplex type \"%s\" on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:3153 +#, c-format +msgid "ppdc: Unknown media size \"%s\" on line %d of %s." +msgstr "" + +#: ppdc/ppdc-catalog.cxx:463 +#, c-format +msgid "ppdc: Unknown message catalog format for \"%s\"." +msgstr "" + +#: ppdc/ppdc-source.cxx:3407 +#, c-format +msgid "ppdc: Unknown token \"%s\" seen on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:1014 +#, c-format +msgid "ppdc: Unknown trailing characters in real number \"%s\" on line %d of %s." +msgstr "" + +#: ppdc/ppdc-source.cxx:2163 +#, c-format +msgid "ppdc: Unterminated string starting with %c on line %d of %s." +msgstr "" + +#: ppdc/ppdc.cxx:365 +#, c-format +msgid "ppdc: Warning - overlapping filename \"%s\"." +msgstr "" + +#: ppdc/ppdc.cxx:380 +#, c-format +msgid "ppdc: Writing %s." +msgstr "" + +#: ppdc/ppdc.cxx:148 +#, c-format +msgid "ppdc: Writing PPD files to directory \"%s\"." +msgstr "" + +#: ppdc/ppdmerge.cxx:136 +#, c-format +msgid "ppdmerge: Bad LanguageVersion \"%s\" in %s." +msgstr "" + +#: ppdc/ppdmerge.cxx:176 +#, c-format +msgid "ppdmerge: Ignoring PPD file %s." +msgstr "" + +#: ppdc/ppdmerge.cxx:160 +#, c-format +msgid "ppdmerge: Unable to backup %s to %s - %s" +msgstr "" + +#: systemv/lpstat.c:1784 +#, c-format +msgid "printer %s disabled since %s -" +msgstr "" + +#: systemv/lpstat.c:1773 +#, c-format +msgid "printer %s is idle. enabled since %s" +msgstr "" + +#: systemv/lpstat.c:1778 +#, c-format +msgid "printer %s now printing %s-%d. enabled since %s" +msgstr "" + +#: systemv/lpstat.c:1909 +#, c-format +msgid "printer %s/%s disabled since %s -" +msgstr "" + +#: systemv/lpstat.c:1895 +#, c-format +msgid "printer %s/%s is idle. enabled since %s" +msgstr "" + +#: systemv/lpstat.c:1902 +#, c-format +msgid "printer %s/%s now printing %s-%d. enabled since %s" +msgstr "" + +#: cups/notify.c:93 cups/notify.c:134 +msgid "processing" +msgstr "" + +#: systemv/lp.c:639 +#, c-format +msgid "request id is %s-%d (%d file(s))" +msgstr "" + +#: cups/snmp.c:1020 +msgid "request-id uses indefinite length" +msgstr "" + +#: systemv/lpstat.c:2048 +msgid "scheduler is not running" +msgstr "" + +#: systemv/lpstat.c:2044 +msgid "scheduler is running" +msgstr "" + +#: cups/adminutil.c:2159 +#, c-format +msgid "stat of %s failed: %s" +msgstr "" + +#: berkeley/lpc.c:211 +msgid "status\t\tShow status of daemon and queue." +msgstr "" + +#: cups/notify.c:96 cups/notify.c:137 +msgid "stopped" +msgstr "" + +#: systemv/lpstat.c:1017 +#, c-format +msgid "system default destination: %s" +msgstr "" + +#: systemv/lpstat.c:1014 +#, c-format +msgid "system default destination: %s/%s" +msgstr "" + +#: cups/notify.c:108 cups/notify.c:140 +msgid "unknown" +msgstr "" + +#: cups/notify.c:117 +msgid "untitled" +msgstr "" + +#: cups/snmp.c:1045 +msgid "variable-bindings uses indefinite length" +msgstr "" + + +# +# End of "$Id$". +# diff --git a/locale/cups.strings b/locale/cups.strings new file mode 100644 index 0000000000..c8ef400d83 --- /dev/null +++ b/locale/cups.strings @@ -0,0 +1,1517 @@ +"\t\t(all)" = "\t\t(all)"; +"\t\t(none)" = "\t\t(none)"; +"\t%d entries" = "\t%d entries"; +"\t%s" = "\t%s"; +"\tAfter fault: continue" = "\tAfter fault: continue"; +"\tAlerts: %s" = "\tAlerts: %s"; +"\tBanner required" = "\tBanner required"; +"\tCharset sets:" = "\tCharset sets:"; +"\tConnection: direct" = "\tConnection: direct"; +"\tConnection: remote" = "\tConnection: remote"; +"\tContent types: any" = "\tContent types: any"; +"\tDefault page size:" = "\tDefault page size:"; +"\tDefault pitch:" = "\tDefault pitch:"; +"\tDefault port settings:" = "\tDefault port settings:"; +"\tDescription: %s" = "\tDescription: %s"; +"\tForm mounted:" = "\tForm mounted:"; +"\tForms allowed:" = "\tForms allowed:"; +"\tInterface: %s.ppd" = "\tInterface: %s.ppd"; +"\tInterface: %s/interfaces/%s" = "\tInterface: %s/interfaces/%s"; +"\tInterface: %s/ppd/%s.ppd" = "\tInterface: %s/ppd/%s.ppd"; +"\tLocation: %s" = "\tLocation: %s"; +"\tOn fault: no alert" = "\tOn fault: no alert"; +"\tPrinter types: unknown" = "\tPrinter types: unknown"; +"\tStatus: %s" = "\tStatus: %s"; +"\tUsers allowed:" = "\tUsers allowed:"; +"\tUsers denied:" = "\tUsers denied:"; +"\tdaemon present" = "\tdaemon present"; +"\tno entries" = "\tno entries"; +"\tprinter is on device '%s' speed -1" = "\tprinter is on device '%s' speed -1"; +"\tprinting is disabled" = "\tprinting is disabled"; +"\tprinting is enabled" = "\tprinting is enabled"; +"\tqueued for %s" = "\tqueued for %s"; +"\tqueuing is disabled" = "\tqueuing is disabled"; +"\tqueuing is enabled" = "\tqueuing is enabled"; +"\treason unknown" = "\treason unknown"; +"\n DETAILED CONFORMANCE TEST RESULTS" = "\n DETAILED CONFORMANCE TEST RESULTS"; +" Ignore specific warnings." = " Ignore specific warnings."; +" Issue warnings instead of errors." = " Issue warnings instead of errors."; +" REF: Page 15, section 3.1." = " REF: Page 15, section 3.1."; +" REF: Page 15, section 3.2." = " REF: Page 15, section 3.2."; +" REF: Page 19, section 3.3." = " REF: Page 19, section 3.3."; +" REF: Page 20, section 3.4." = " REF: Page 20, section 3.4."; +" REF: Page 27, section 3.5." = " REF: Page 27, section 3.5."; +" REF: Page 42, section 5.2." = " REF: Page 42, section 5.2."; +" REF: Pages 16-17, section 3.2." = " REF: Pages 16-17, section 3.2."; +" REF: Pages 42-45, section 5.2." = " REF: Pages 42-45, section 5.2."; +" REF: Pages 45-46, section 5.2." = " REF: Pages 45-46, section 5.2."; +" REF: Pages 48-49, section 5.2." = " REF: Pages 48-49, section 5.2."; +" REF: Pages 52-54, section 5.2." = " REF: Pages 52-54, section 5.2."; +" %-39.39s %.0f bytes" = " %-39.39s %.0f bytes"; +" PASS Default%s" = " PASS Default%s"; +" PASS DefaultImageableArea" = " PASS DefaultImageableArea"; +" PASS DefaultPaperDimension" = " PASS DefaultPaperDimension"; +" PASS FileVersion" = " PASS FileVersion"; +" PASS FormatVersion" = " PASS FormatVersion"; +" PASS LanguageEncoding" = " PASS LanguageEncoding"; +" PASS LanguageVersion" = " PASS LanguageVersion"; +" PASS Manufacturer" = " PASS Manufacturer"; +" PASS ModelName" = " PASS ModelName"; +" PASS NickName" = " PASS NickName"; +" PASS PCFileName" = " PASS PCFileName"; +" PASS PSVersion" = " PASS PSVersion"; +" PASS PageRegion" = " PASS PageRegion"; +" PASS PageSize" = " PASS PageSize"; +" PASS Product" = " PASS Product"; +" PASS ShortNickName" = " PASS ShortNickName"; +" WARN %s has no corresponding options." = " WARN %s has no corresponding options."; +" WARN %s shares a common prefix with %s\n REF: Page 15, section 3.2." = " WARN %s shares a common prefix with %s\n REF: Page 15, section 3.2."; +" WARN Duplex option keyword %s may not work as expected and should be named Duplex.\n REF: Page 122, section 5.17" = " WARN Duplex option keyword %s may not work as expected and should be named Duplex.\n REF: Page 122, section 5.17"; +" WARN File contains a mix of CR, LF, and CR LF line endings." = " WARN File contains a mix of CR, LF, and CR LF line endings."; +" WARN LanguageEncoding required by PPD 4.3 spec.\n REF: Pages 56-57, section 5.3." = " WARN LanguageEncoding required by PPD 4.3 spec.\n REF: Pages 56-57, section 5.3."; +" WARN Line %d only contains whitespace." = " WARN Line %d only contains whitespace."; +" WARN Manufacturer required by PPD 4.3 spec.\n REF: Pages 58-59, section 5.3." = " WARN Manufacturer required by PPD 4.3 spec.\n REF: Pages 58-59, section 5.3."; +" WARN Non-Windows PPD files should use lines ending with only LF, not CR LF." = " WARN Non-Windows PPD files should use lines ending with only LF, not CR LF."; +" WARN Obsolete PPD version %.1f.\n REF: Page 42, section 5.2." = " WARN Obsolete PPD version %.1f.\n REF: Page 42, section 5.2."; +" WARN PCFileName longer than 8.3 in violation of PPD spec.\n REF: Pages 61-62, section 5.3." = " WARN PCFileName longer than 8.3 in violation of PPD spec.\n REF: Pages 61-62, section 5.3."; +" WARN PCFileName should contain a unique filename.\n REF: Pages 61-62, section 5.3." = " WARN PCFileName should contain a unique filename.\n REF: Pages 61-62, section 5.3."; +" WARN Protocols contains PJL but JCL attributes are not set.\n REF: Pages 78-79, section 5.7." = " WARN Protocols contains PJL but JCL attributes are not set.\n REF: Pages 78-79, section 5.7."; +" WARN Protocols contains both PJL and BCP; expected TBCP.\n REF: Pages 78-79, section 5.7." = " WARN Protocols contains both PJL and BCP; expected TBCP.\n REF: Pages 78-79, section 5.7."; +" WARN ShortNickName required by PPD 4.3 spec.\n REF: Pages 64-65, section 5.3." = " WARN ShortNickName required by PPD 4.3 spec.\n REF: Pages 64-65, section 5.3."; +" cupsaddsmb [options] -a" = " cupsaddsmb [options] -a"; +" cupstestdsc [options] -" = " cupstestdsc [options] -"; +" program | cupstestppd [options] -" = " program | cupstestppd [options] -"; +" %s \"%s %s\" conflicts with \"%s %s\"\n (constraint=\"%s %s %s %s\")." = " %s \"%s %s\" conflicts with \"%s %s\"\n (constraint=\"%s %s %s %s\")."; +" %s %s %s does not exist." = " %s %s %s does not exist."; +" %s %s file \"%s\" has the wrong capitalization." = " %s %s file \"%s\" has the wrong capitalization."; +" %s Bad %s choice %s.\n REF: Page 122, section 5.17" = " %s Bad %s choice %s.\n REF: Page 122, section 5.17"; +" %s Bad UTF-8 \"%s\" translation string for option %s, choice %s." = " %s Bad UTF-8 \"%s\" translation string for option %s, choice %s."; +" %s Bad UTF-8 \"%s\" translation string for option %s." = " %s Bad UTF-8 \"%s\" translation string for option %s."; +" %s Bad cupsFilter value \"%s\"." = " %s Bad cupsFilter value \"%s\"."; +" %s Bad cupsFilter2 value \"%s\"." = " %s Bad cupsFilter2 value \"%s\"."; +" %s Bad cupsICCProfile %s." = " %s Bad cupsICCProfile %s."; +" %s Bad cupsPreFilter value \"%s\"." = " %s Bad cupsPreFilter value \"%s\"."; +" %s Bad cupsUIConstraints %s: \"%s\"" = " %s Bad cupsUIConstraints %s: \"%s\""; +" %s Bad language \"%s\"." = " %s Bad language \"%s\"."; +" %s Bad permissions on %s file \"%s\"." = " %s Bad permissions on %s file \"%s\"."; +" %s Bad spelling of %s - should be %s." = " %s Bad spelling of %s - should be %s."; +" %s Cannot provide both APScanAppPath and APScanAppBundleID." = " %s Cannot provide both APScanAppPath and APScanAppBundleID."; +" %s Default choices conflicting." = " %s Default choices conflicting."; +" %s Empty cupsUIConstraints %s" = " %s Empty cupsUIConstraints %s"; +" %s Missing \"%s\" translation string for option %s, choice %s." = " %s Missing \"%s\" translation string for option %s, choice %s."; +" %s Missing \"%s\" translation string for option %s." = " %s Missing \"%s\" translation string for option %s."; +" %s Missing %s file \"%s\"." = " %s Missing %s file \"%s\"."; +" %s Missing REQUIRED PageRegion option.\n REF: Page 100, section 5.14." = " %s Missing REQUIRED PageRegion option.\n REF: Page 100, section 5.14."; +" %s Missing REQUIRED PageSize option.\n REF: Page 99, section 5.14." = " %s Missing REQUIRED PageSize option.\n REF: Page 99, section 5.14."; +" %s Missing choice *%s %s in UIConstraints \"*%s %s *%s %s\"." = " %s Missing choice *%s %s in UIConstraints \"*%s %s *%s %s\"."; +" %s Missing choice *%s %s in cupsUIConstraints %s: \"%s\"" = " %s Missing choice *%s %s in cupsUIConstraints %s: \"%s\""; +" %s Missing cupsUIResolver %s" = " %s Missing cupsUIResolver %s"; +" %s Missing option %s in UIConstraints \"*%s %s *%s %s\"." = " %s Missing option %s in UIConstraints \"*%s %s *%s %s\"."; +" %s Missing option %s in cupsUIConstraints %s: \"%s\"" = " %s Missing option %s in cupsUIConstraints %s: \"%s\""; +" %s No base translation \"%s\" is included in file." = " %s No base translation \"%s\" is included in file."; +" %s REQUIRED %s does not define choice None.\n REF: Page 122, section 5.17" = " %s REQUIRED %s does not define choice None.\n REF: Page 122, section 5.17"; +" %s Size \"%s\" defined for %s but not for %s." = " %s Size \"%s\" defined for %s but not for %s."; +" %s Size \"%s\" has unexpected dimensions (%gx%g)." = " %s Size \"%s\" has unexpected dimensions (%gx%g)."; +" %s Size \"%s\" should be \"%s\"." = " %s Size \"%s\" should be \"%s\"."; +" %s Size \"%s\" should be the Adobe standard name \"%s\"." = " %s Size \"%s\" should be the Adobe standard name \"%s\"."; +" %s cupsICCProfile %s hash value collides with %s." = " %s cupsICCProfile %s hash value collides with %s."; +" %s cupsUIResolver %s causes a loop." = " %s cupsUIResolver %s causes a loop."; +" %s cupsUIResolver %s does not list at least two different options." = " %s cupsUIResolver %s does not list at least two different options."; +" **FAIL** %s choice names %s and %s differ only by case." = " **FAIL** %s choice names %s and %s differ only by case."; +" **FAIL** %s must be 1284DeviceID\n REF: Page 72, section 5.5" = " **FAIL** %s must be 1284DeviceID\n REF: Page 72, section 5.5"; +" **FAIL** BAD Default%s %s\n REF: Page 40, section 4.5." = " **FAIL** BAD Default%s %s\n REF: Page 40, section 4.5."; +" **FAIL** BAD DefaultImageableArea %s\n REF: Page 102, section 5.15." = " **FAIL** BAD DefaultImageableArea %s\n REF: Page 102, section 5.15."; +" **FAIL** BAD DefaultPaperDimension %s\n REF: Page 103, section 5.15." = " **FAIL** BAD DefaultPaperDimension %s\n REF: Page 103, section 5.15."; +" **FAIL** BAD JobPatchFile attribute in file\n REF: Page 24, section 3.4." = " **FAIL** BAD JobPatchFile attribute in file\n REF: Page 24, section 3.4."; +" **FAIL** BAD Manufacturer (should be \"HP\")\n REF: Page 211, table D.1." = " **FAIL** BAD Manufacturer (should be \"HP\")\n REF: Page 211, table D.1."; +" **FAIL** BAD Manufacturer (should be \"Oki\")\n REF: Page 211, table D.1." = " **FAIL** BAD Manufacturer (should be \"Oki\")\n REF: Page 211, table D.1."; +" **FAIL** BAD ModelName - \"%c\" not allowed in string.\n REF: Pages 59-60, section 5.3." = " **FAIL** BAD ModelName - \"%c\" not allowed in string.\n REF: Pages 59-60, section 5.3."; +" **FAIL** BAD PSVersion - not \"(string) int\".\n REF: Pages 62-64, section 5.3." = " **FAIL** BAD PSVersion - not \"(string) int\".\n REF: Pages 62-64, section 5.3."; +" **FAIL** BAD Product - not \"(string)\".\n REF: Page 62, section 5.3." = " **FAIL** BAD Product - not \"(string)\".\n REF: Page 62, section 5.3."; +" **FAIL** BAD ShortNickName - longer than 31 chars.\n REF: Pages 64-65, section 5.3." = " **FAIL** BAD ShortNickName - longer than 31 chars.\n REF: Pages 64-65, section 5.3."; +" **FAIL** Bad %s choice %s\n REF: Page 84, section 5.9" = " **FAIL** Bad %s choice %s\n REF: Page 84, section 5.9"; +" **FAIL** Bad FileVersion \"%s\"\n REF: Page 56, section 5.3." = " **FAIL** Bad FileVersion \"%s\"\n REF: Page 56, section 5.3."; +" **FAIL** Bad FormatVersion \"%s\"\n REF: Page 56, section 5.3." = " **FAIL** Bad FormatVersion \"%s\"\n REF: Page 56, section 5.3."; +" **FAIL** Bad LanguageEncoding %s - must be ISOLatin1." = " **FAIL** Bad LanguageEncoding %s - must be ISOLatin1."; +" **FAIL** Bad LanguageVersion %s - must be English." = " **FAIL** Bad LanguageVersion %s - must be English."; +" **FAIL** Default option code cannot be interpreted: %s" = " **FAIL** Default option code cannot be interpreted: %s"; +" **FAIL** Default translation string for option %s choice %s contains 8-bit characters." = " **FAIL** Default translation string for option %s choice %s contains 8-bit characters."; +" **FAIL** Default translation string for option %s contains 8-bit characters." = " **FAIL** Default translation string for option %s contains 8-bit characters."; +" **FAIL** Group names %s and %s differ only by case." = " **FAIL** Group names %s and %s differ only by case."; +" **FAIL** Multiple occurrences of %s choice name %s." = " **FAIL** Multiple occurrences of %s choice name %s."; +" **FAIL** Option names %s and %s differ only by case." = " **FAIL** Option names %s and %s differ only by case."; +" **FAIL** REQUIRED Default%s\n REF: Page 40, section 4.5." = " **FAIL** REQUIRED Default%s\n REF: Page 40, section 4.5."; +" **FAIL** REQUIRED DefaultImageableArea\n REF: Page 102, section 5.15." = " **FAIL** REQUIRED DefaultImageableArea\n REF: Page 102, section 5.15."; +" **FAIL** REQUIRED DefaultPaperDimension\n REF: Page 103, section 5.15." = " **FAIL** REQUIRED DefaultPaperDimension\n REF: Page 103, section 5.15."; +" **FAIL** REQUIRED FileVersion\n REF: Page 56, section 5.3." = " **FAIL** REQUIRED FileVersion\n REF: Page 56, section 5.3."; +" **FAIL** REQUIRED FormatVersion\n REF: Page 56, section 5.3." = " **FAIL** REQUIRED FormatVersion\n REF: Page 56, section 5.3."; +" **FAIL** REQUIRED ImageableArea for PageSize %s\n REF: Page 41, section 5.\n REF: Page 102, section 5.15." = " **FAIL** REQUIRED ImageableArea for PageSize %s\n REF: Page 41, section 5.\n REF: Page 102, section 5.15."; +" **FAIL** REQUIRED LanguageEncoding\n REF: Pages 56-57, section 5.3." = " **FAIL** REQUIRED LanguageEncoding\n REF: Pages 56-57, section 5.3."; +" **FAIL** REQUIRED LanguageVersion\n REF: Pages 57-58, section 5.3." = " **FAIL** REQUIRED LanguageVersion\n REF: Pages 57-58, section 5.3."; +" **FAIL** REQUIRED Manufacturer\n REF: Pages 58-59, section 5.3." = " **FAIL** REQUIRED Manufacturer\n REF: Pages 58-59, section 5.3."; +" **FAIL** REQUIRED ModelName\n REF: Pages 59-60, section 5.3." = " **FAIL** REQUIRED ModelName\n REF: Pages 59-60, section 5.3."; +" **FAIL** REQUIRED NickName\n REF: Page 60, section 5.3." = " **FAIL** REQUIRED NickName\n REF: Page 60, section 5.3."; +" **FAIL** REQUIRED PCFileName\n REF: Pages 61-62, section 5.3." = " **FAIL** REQUIRED PCFileName\n REF: Pages 61-62, section 5.3."; +" **FAIL** REQUIRED PSVersion\n REF: Pages 62-64, section 5.3." = " **FAIL** REQUIRED PSVersion\n REF: Pages 62-64, section 5.3."; +" **FAIL** REQUIRED PageRegion\n REF: Page 100, section 5.14." = " **FAIL** REQUIRED PageRegion\n REF: Page 100, section 5.14."; +" **FAIL** REQUIRED PageSize\n REF: Page 41, section 5.\n REF: Page 99, section 5.14." = " **FAIL** REQUIRED PageSize\n REF: Page 41, section 5.\n REF: Page 99, section 5.14."; +" **FAIL** REQUIRED PageSize\n REF: Pages 99-100, section 5.14." = " **FAIL** REQUIRED PageSize\n REF: Pages 99-100, section 5.14."; +" **FAIL** REQUIRED PaperDimension for PageSize %s\n REF: Page 41, section 5.\n REF: Page 103, section 5.15." = " **FAIL** REQUIRED PaperDimension for PageSize %s\n REF: Page 41, section 5.\n REF: Page 103, section 5.15."; +" **FAIL** REQUIRED Product\n REF: Page 62, section 5.3." = " **FAIL** REQUIRED Product\n REF: Page 62, section 5.3."; +" **FAIL** REQUIRED ShortNickName\n REF: Page 64-65, section 5.3." = " **FAIL** REQUIRED ShortNickName\n REF: Page 64-65, section 5.3."; +" **FAIL** Unable to open PPD file - %s" = " **FAIL** Unable to open PPD file - %s"; +" **FAIL** Unable to open PPD file - %s on line %d." = " **FAIL** Unable to open PPD file - %s on line %d."; +" %d ERRORS FOUND" = " %d ERRORS FOUND"; +" -h Show program usage" = " -h Show program usage"; +" Bad %%%%BoundingBox: on line %d.\n REF: Page 39, %%%%BoundingBox:" = " Bad %%%%BoundingBox: on line %d.\n REF: Page 39, %%%%BoundingBox:"; +" Bad %%%%Page: on line %d.\n REF: Page 53, %%%%Page:" = " Bad %%%%Page: on line %d.\n REF: Page 53, %%%%Page:"; +" Bad %%%%Pages: on line %d.\n REF: Page 43, %%%%Pages:" = " Bad %%%%Pages: on line %d.\n REF: Page 43, %%%%Pages:"; +" Line %d is longer than 255 characters (%d).\n REF: Page 25, Line Length" = " Line %d is longer than 255 characters (%d).\n REF: Page 25, Line Length"; +" Missing %!PS-Adobe-3.0 on first line.\n REF: Page 17, 3.1 Conforming Documents" = " Missing %!PS-Adobe-3.0 on first line.\n REF: Page 17, 3.1 Conforming Documents"; +" Missing %%EndComments comment. REF: Page 41, %%EndComments" = " Missing %%EndComments comment. REF: Page 41, %%EndComments"; +" Missing or bad %%BoundingBox: comment.\n REF: Page 39, %%BoundingBox:" = " Missing or bad %%BoundingBox: comment.\n REF: Page 39, %%BoundingBox:"; +" Missing or bad %%Page: comments.\n REF: Page 53, %%Page:" = " Missing or bad %%Page: comments.\n REF: Page 53, %%Page:"; +" Missing or bad %%Pages: comment.\n REF: Page 43, %%Pages:" = " Missing or bad %%Pages: comment.\n REF: Page 43, %%Pages:"; +" NO ERRORS FOUND" = " NO ERRORS FOUND"; +" Saw %d lines that exceeded 255 characters." = " Saw %d lines that exceeded 255 characters."; +" Too many %%BeginDocument comments." = " Too many %%BeginDocument comments."; +" Too many %%EndDocument comments." = " Too many %%EndDocument comments."; +" Warning: file contains binary data." = " Warning: file contains binary data."; +" Warning: no %%EndComments comment in file." = " Warning: no %%EndComments comment in file."; +" Warning: obsolete DSC version %.1f in file." = " Warning: obsolete DSC version %.1f in file."; +" --[no-]debug-logging Turn debug logging on/off." = " --[no-]debug-logging Turn debug logging on/off."; +" --[no-]remote-admin Turn remote administration on/off." = " --[no-]remote-admin Turn remote administration on/off."; +" --[no-]remote-any Allow/prevent access from the Internet." = " --[no-]remote-any Allow/prevent access from the Internet."; +" --[no-]share-printers Turn printer sharing on/off." = " --[no-]share-printers Turn printer sharing on/off."; +" --[no-]user-cancel-any Allow/prevent users to cancel any job." = " --[no-]user-cancel-any Allow/prevent users to cancel any job."; +" --cr End lines with CR (Mac OS 9)." = " --cr End lines with CR (Mac OS 9)."; +" --crlf End lines with CR + LF (Windows)." = " --crlf End lines with CR + LF (Windows)."; +" --lf End lines with LF (UNIX/Linux/Mac OS X)." = " --lf End lines with LF (UNIX/Linux/Mac OS X)."; +" -4 Connect using IPv4." = " -4 Connect using IPv4."; +" -6 Connect using IPv6." = " -6 Connect using IPv6."; +" -C Send requests using chunking (default)." = " -C Send requests using chunking (default)."; +" -D Remove the input file when finished." = " -D Remove the input file when finished."; +" -D name=value Set named variable to value." = " -D name=value Set named variable to value."; +" -E Enable encryption." = " -E Enable encryption."; +" -E Encrypt the connection to the server." = " -E Encrypt the connection to the server."; +" -E Test with TLS encryption." = " -E Test with TLS encryption."; +" -F Run in the foreground but detach from console." = " -F Run in the foreground but detach from console."; +" -H samba-server Use the named SAMBA server." = " -H samba-server Use the named SAMBA server."; +" -I Ignore errors." = " -I Ignore errors."; +" -I include-dir Add include directory to search path." = " -I include-dir Add include directory to search path."; +" -I {filename,filters,none,profiles}" = " -I {filename,filters,none,profiles}"; +" -J title Set title." = " -J title Set title."; +" -L Send requests using content-length." = " -L Send requests using content-length."; +" -P filename.ppd Set PPD file." = " -P filename.ppd Set PPD file."; +" -R root-directory Set alternate root." = " -R root-directory Set alternate root."; +" -S Test with SSL encryption." = " -S Test with SSL encryption."; +" -T Set the receive/send timeout in seconds." = " -T Set the receive/send timeout in seconds."; +" -U samba-user Authenticate using the named SAMBA user." = " -U samba-user Authenticate using the named SAMBA user."; +" -U username Set username for job." = " -U username Set username for job."; +" -U username Specify username." = " -U username Specify username."; +" -V version Set default IPP version." = " -V version Set default IPP version."; +" -W {all,none,constraints,defaults,duplex,filters,profiles,sizes,translations}" = " -W {all,none,constraints,defaults,duplex,filters,profiles,sizes,translations}"; +" -X Produce XML plist instead of plain text." = " -X Produce XML plist instead of plain text."; +" -a Export all printers." = " -a Export all printers."; +" -a 'name=value ...' Set option(s)." = " -a 'name=value ...' Set option(s)."; +" -c catalog.po Load the specified message catalog." = " -c catalog.po Load the specified message catalog."; +" -c config-file Load alternate configuration file." = " -c config-file Load alternate configuration file."; +" -c copies Set number of copies." = " -c copies Set number of copies."; +" -c cupsd.conf Set cupsd.conf file to use." = " -c cupsd.conf Set cupsd.conf file to use."; +" -d name=value Set named variable to value." = " -d name=value Set named variable to value."; +" -d output-dir Specify the output directory." = " -d output-dir Specify the output directory."; +" -d printer Use the named printer." = " -d printer Use the named printer."; +" -e Use every filter from the PPD file." = " -e Use every filter from the PPD file."; +" -f Run in the foreground." = " -f Run in the foreground."; +" -f filename Set default request filename." = " -f filename Set default request filename."; +" -f filename Set file to be converted (otherwise stdin)." = " -f filename Set file to be converted (otherwise stdin)."; +" -h Show this usage message." = " -h Show this usage message."; +" -h cups-server Use the named CUPS server." = " -h cups-server Use the named CUPS server."; +" -h server[:port] Specify server address." = " -h server[:port] Specify server address."; +" -i mime/type Set input MIME type (otherwise auto-typed)." = " -i mime/type Set input MIME type (otherwise auto-typed)."; +" -i seconds Repeat the last file with the given time interval." = " -i seconds Repeat the last file with the given time interval."; +" -j job-id[,N] Filter file N from the specified job (default is file 1)." = " -j job-id[,N] Filter file N from the specified job (default is file 1)."; +" -j mime/type Set output MIME type (otherwise application/pdf)." = " -j mime/type Set output MIME type (otherwise application/pdf)."; +" -l Run cupsd from launchd(8)." = " -l Run cupsd from launchd(8)."; +" -l lang[,lang,...] Specify the output language(s) (locale)." = " -l lang[,lang,...] Specify the output language(s) (locale)."; +" -m Use the ModelName value as the filename." = " -m Use the ModelName value as the filename."; +" -m mime/type Set output MIME type (otherwise application/pdf)." = " -m mime/type Set output MIME type (otherwise application/pdf)."; +" -n copies Set number of copies." = " -n copies Set number of copies."; +" -n count Repeat the last file the given number of times." = " -n count Repeat the last file the given number of times."; +" -o filename Set file to be generated (otherwise stdout)." = " -o filename Set file to be generated (otherwise stdout)."; +" -o filename.drv Set driver information file (otherwise ppdi.drv)." = " -o filename.drv Set driver information file (otherwise ppdi.drv)."; +" -o filename.ppd[.gz] Set output file (otherwise stdout)." = " -o filename.ppd[.gz] Set output file (otherwise stdout)."; +" -o name=value Set option(s)." = " -o name=value Set option(s)."; +" -p filename.ppd Set PPD file." = " -p filename.ppd Set PPD file."; +" -q Be quiet - no output except errors." = " -q Be quiet - no output except errors."; +" -q Run silently." = " -q Run silently."; +" -r Use 'relaxed' open mode." = " -r Use 'relaxed' open mode."; +" -t Produce a test report." = " -t Produce a test report."; +" -t Test PPDs instead of generating them." = " -t Test PPDs instead of generating them."; +" -t Test the configuration file." = " -t Test the configuration file."; +" -t title Set title." = " -t title Set title."; +" -u Remove the PPD file when finished." = " -u Remove the PPD file when finished."; +" -v Be slightly verbose." = " -v Be slightly verbose."; +" -v Be verbose (more v's for more verbosity)." = " -v Be verbose (more v's for more verbosity)."; +" -v Be verbose (show commands)." = " -v Be verbose (show commands)."; +" -v Show all attributes sent and received." = " -v Show all attributes sent and received."; +" -vv Be very verbose." = " -vv Be very verbose."; +" -z Compress PPD files using GNU zip." = " -z Compress PPD files using GNU zip."; +" FAIL" = " FAIL"; +" PASS" = " PASS"; +"%-6s %-10.10s %-4d %-10d %-27.27s %.0f bytes" = "%-6s %-10.10s %-4d %-10d %-27.27s %.0f bytes"; +"%-7s %-7.7s %-7d %-31.31s %.0f bytes" = "%-7s %-7.7s %-7d %-31.31s %.0f bytes"; +"%s accepting requests since %s" = "%s accepting requests since %s"; +"%s cannot be changed." = "%s cannot be changed."; +"%s is not implemented by the CUPS version of lpc." = "%s is not implemented by the CUPS version of lpc."; +"%s is not ready" = "%s is not ready"; +"%s is ready" = "%s is ready"; +"%s is ready and printing" = "%s is ready and printing"; +"%s job-id user title copies options [file]" = "%s job-id user title copies options [file]"; +"%s not accepting requests since %s -" = "%s not accepting requests since %s -"; +"%s not supported." = "%s not supported."; +"%s/%s accepting requests since %s" = "%s/%s accepting requests since %s"; +"%s/%s not accepting requests since %s -" = "%s/%s not accepting requests since %s -"; +"%s: %-33.33s [job %d localhost]" = "%s: %-33.33s [job %d localhost]"; +// TRANSLATORS: Message is "subject: error" +"%s: %s" = "%s: %s"; +"%s: %s failed: %s" = "%s: %s failed: %s"; +"%s: Don't know what to do." = "%s: Don't know what to do."; +"%s: Error - %s environment variable names non-existent destination \"%s\"." = "%s: Error - %s environment variable names non-existent destination \"%s\"."; +"%s: Error - bad job ID." = "%s: Error - bad job ID."; +"%s: Error - cannot print files and alter jobs simultaneously." = "%s: Error - cannot print files and alter jobs simultaneously."; +"%s: Error - cannot print from stdin if files or a job ID are provided." = "%s: Error - cannot print from stdin if files or a job ID are provided."; +"%s: Error - expected character set after \"-S\" option." = "%s: Error - expected character set after \"-S\" option."; +"%s: Error - expected content type after \"-T\" option." = "%s: Error - expected content type after \"-T\" option."; +"%s: Error - expected copies after \"-#\" option." = "%s: Error - expected copies after \"-#\" option."; +"%s: Error - expected copies after \"-n\" option." = "%s: Error - expected copies after \"-n\" option."; +"%s: Error - expected destination after \"-P\" option." = "%s: Error - expected destination after \"-P\" option."; +"%s: Error - expected destination after \"-b\" option." = "%s: Error - expected destination after \"-b\" option."; +"%s: Error - expected destination after \"-d\" option." = "%s: Error - expected destination after \"-d\" option."; +"%s: Error - expected form after \"-f\" option." = "%s: Error - expected form after \"-f\" option."; +"%s: Error - expected hold name after \"-H\" option." = "%s: Error - expected hold name after \"-H\" option."; +"%s: Error - expected hostname after \"-H\" option." = "%s: Error - expected hostname after \"-H\" option."; +"%s: Error - expected hostname after \"-h\" option." = "%s: Error - expected hostname after \"-h\" option."; +"%s: Error - expected mode list after \"-y\" option." = "%s: Error - expected mode list after \"-y\" option."; +"%s: Error - expected name after \"-%c\" option." = "%s: Error - expected name after \"-%c\" option."; +"%s: Error - expected option=value after \"-o\" option." = "%s: Error - expected option=value after \"-o\" option."; +"%s: Error - expected page list after \"-P\" option." = "%s: Error - expected page list after \"-P\" option."; +"%s: Error - expected priority after \"-%c\" option." = "%s: Error - expected priority after \"-%c\" option."; +"%s: Error - expected reason text after \"-r\" option." = "%s: Error - expected reason text after \"-r\" option."; +"%s: Error - expected title after \"-t\" option." = "%s: Error - expected title after \"-t\" option."; +"%s: Error - expected username after \"-U\" option." = "%s: Error - expected username after \"-U\" option."; +"%s: Error - expected username after \"-u\" option." = "%s: Error - expected username after \"-u\" option."; +"%s: Error - expected value after \"-%c\" option." = "%s: Error - expected value after \"-%c\" option."; +"%s: Error - need \"completed\", \"not-completed\", or \"all\" after \"-W\" option." = "%s: Error - need \"completed\", \"not-completed\", or \"all\" after \"-W\" option."; +"%s: Error - no default destination available." = "%s: Error - no default destination available."; +"%s: Error - priority must be between 1 and 100." = "%s: Error - priority must be between 1 and 100."; +"%s: Error - scheduler not responding." = "%s: Error - scheduler not responding."; +"%s: Error - too many files - \"%s\"." = "%s: Error - too many files - \"%s\"."; +"%s: Error - unable to access \"%s\" - %s" = "%s: Error - unable to access \"%s\" - %s"; +"%s: Error - unable to queue from stdin - %s." = "%s: Error - unable to queue from stdin - %s."; +"%s: Error - unknown destination \"%s\"." = "%s: Error - unknown destination \"%s\"."; +"%s: Error - unknown destination \"%s/%s\"." = "%s: Error - unknown destination \"%s/%s\"."; +"%s: Error - unknown option \"%c\"." = "%s: Error - unknown option \"%c\"."; +"%s: Error - unknown option \"%s\"." = "%s: Error - unknown option \"%s\"."; +"%s: Expected job ID after \"-i\" option." = "%s: Expected job ID after \"-i\" option."; +"%s: Invalid destination name in list \"%s\"." = "%s: Invalid destination name in list \"%s\"."; +"%s: Invalid filter string \"%s\"." = "%s: Invalid filter string \"%s\"."; +"%s: Need job ID (\"-i jobid\") before \"-H restart\"." = "%s: Need job ID (\"-i jobid\") before \"-H restart\"."; +"%s: No filter to convert from %s/%s to %s/%s." = "%s: No filter to convert from %s/%s to %s/%s."; +"%s: Operation failed: %s" = "%s: Operation failed: %s"; +"%s: Sorry, no encryption support." = "%s: Sorry, no encryption support."; +"%s: Unable to connect to server." = "%s: Unable to connect to server."; +"%s: Unable to contact server." = "%s: Unable to contact server."; +"%s: Unable to determine MIME type of \"%s\"." = "%s: Unable to determine MIME type of \"%s\"."; +"%s: Unable to open %s: %s" = "%s: Unable to open %s: %s"; +"%s: Unable to open PPD file: %s on line %d." = "%s: Unable to open PPD file: %s on line %d."; +"%s: Unable to read MIME database from \"%s\" or \"%s\"." = "%s: Unable to read MIME database from \"%s\" or \"%s\"."; +"%s: Unknown destination \"%s\"." = "%s: Unknown destination \"%s\"."; +"%s: Unknown destination MIME type %s/%s." = "%s: Unknown destination MIME type %s/%s."; +"%s: Unknown option \"%c\"." = "%s: Unknown option \"%c\"."; +"%s: Unknown source MIME type %s/%s." = "%s: Unknown source MIME type %s/%s."; +"%s: Warning - \"%c\" format modifier not supported - output may not be correct." = "%s: Warning - \"%c\" format modifier not supported - output may not be correct."; +"%s: Warning - character set option ignored." = "%s: Warning - character set option ignored."; +"%s: Warning - content type option ignored." = "%s: Warning - content type option ignored."; +"%s: Warning - form option ignored." = "%s: Warning - form option ignored."; +"%s: Warning - mode option ignored." = "%s: Warning - mode option ignored."; +"-1" = "-1"; +"-10" = "-10"; +"-100" = "-100"; +"-105" = "-105"; +"-11" = "-11"; +"-110" = "-110"; +"-115" = "-115"; +"-12" = "-12"; +"-120" = "-120"; +"-13" = "-13"; +"-14" = "-14"; +"-15" = "-15"; +"-2" = "-2"; +"-20" = "-20"; +"-25" = "-25"; +"-3" = "-3"; +"-30" = "-30"; +"-35" = "-35"; +"-4" = "-4"; +"-40" = "-40"; +"-45" = "-45"; +"-5" = "-5"; +"-50" = "-50"; +"-55" = "-55"; +"-6" = "-6"; +"-60" = "-60"; +"-65" = "-65"; +"-7" = "-7"; +"-70" = "-70"; +"-75" = "-75"; +"-8" = "-8"; +"-80" = "-80"; +"-85" = "-85"; +"-9" = "-9"; +"-90" = "-90"; +"-95" = "-95"; +"0" = "0"; +"1" = "1"; +"1 inch/sec." = "1 inch/sec."; +"1.25x0.25\"" = "1.25x0.25\""; +"1.25x2.25\"" = "1.25x2.25\""; +"1.5 inch/sec." = "1.5 inch/sec."; +"1.50x0.25\"" = "1.50x0.25\""; +"1.50x0.50\"" = "1.50x0.50\""; +"1.50x1.00\"" = "1.50x1.00\""; +"1.50x2.00\"" = "1.50x2.00\""; +"10" = "10"; +"10 inches/sec." = "10 inches/sec."; +"10 x 11" = "10 x 11"; +"10 x 13" = "10 x 13"; +"10 x 14" = "10 x 14"; +"100" = "100"; +"100 mm/sec." = "100 mm/sec."; +"105" = "105"; +"11" = "11"; +"11 inches/sec." = "11 inches/sec."; +"110" = "110"; +"115" = "115"; +"12" = "12"; +"12 inches/sec." = "12 inches/sec."; +"12 x 11" = "12 x 11"; +"120" = "120"; +"120 mm/sec." = "120 mm/sec."; +"120x60dpi" = "120x60dpi"; +"120x72dpi" = "120x72dpi"; +"13" = "13"; +"136dpi" = "136dpi"; +"14" = "14"; +"15" = "15"; +"15 mm/sec." = "15 mm/sec."; +"15 x 11" = "15 x 11"; +"150 mm/sec." = "150 mm/sec."; +"150dpi" = "150dpi"; +"16" = "16"; +"17" = "17"; +"18" = "18"; +"180dpi" = "180dpi"; +"19" = "19"; +"2" = "2"; +"2 inches/sec." = "2 inches/sec."; +"2-Sided Printing" = "2-Sided Printing"; +"2.00x0.37\"" = "2.00x0.37\""; +"2.00x0.50\"" = "2.00x0.50\""; +"2.00x1.00\"" = "2.00x1.00\""; +"2.00x1.25\"" = "2.00x1.25\""; +"2.00x2.00\"" = "2.00x2.00\""; +"2.00x3.00\"" = "2.00x3.00\""; +"2.00x4.00\"" = "2.00x4.00\""; +"2.00x5.50\"" = "2.00x5.50\""; +"2.25x0.50\"" = "2.25x0.50\""; +"2.25x1.25\"" = "2.25x1.25\""; +"2.25x4.00\"" = "2.25x4.00\""; +"2.25x5.50\"" = "2.25x5.50\""; +"2.38x5.50\"" = "2.38x5.50\""; +"2.5 inches/sec." = "2.5 inches/sec."; +"2.50x1.00\"" = "2.50x1.00\""; +"2.50x2.00\"" = "2.50x2.00\""; +"2.75x1.25\"" = "2.75x1.25\""; +"2.9 x 1\"" = "2.9 x 1\""; +"20" = "20"; +"20 mm/sec." = "20 mm/sec."; +"200 mm/sec." = "200 mm/sec."; +"203dpi" = "203dpi"; +"21" = "21"; +"22" = "22"; +"23" = "23"; +"24" = "24"; +"24-Pin Series" = "24-Pin Series"; +"240x72dpi" = "240x72dpi"; +"25" = "25"; +"250 mm/sec." = "250 mm/sec."; +"26" = "26"; +"27" = "27"; +"28" = "28"; +"29" = "29"; +"3" = "3"; +"3 inches/sec." = "3 inches/sec."; +"3 x 5" = "3 x 5"; +"3.00x1.00\"" = "3.00x1.00\""; +"3.00x1.25\"" = "3.00x1.25\""; +"3.00x2.00\"" = "3.00x2.00\""; +"3.00x3.00\"" = "3.00x3.00\""; +"3.00x5.00\"" = "3.00x5.00\""; +"3.25x2.00\"" = "3.25x2.00\""; +"3.25x5.00\"" = "3.25x5.00\""; +"3.25x5.50\"" = "3.25x5.50\""; +"3.25x5.83\"" = "3.25x5.83\""; +"3.25x7.83\"" = "3.25x7.83\""; +"3.5 x 5" = "3.5 x 5"; +"3.5\" Disk" = "3.5\" Disk"; +"3.5\" Disk - 2 1/8 x 2 3/4\"" = "3.5\" Disk - 2 1/8 x 2 3/4\""; +"3.50x1.00\"" = "3.50x1.00\""; +"30" = "30"; +"30 mm/sec." = "30 mm/sec."; +"300 mm/sec." = "300 mm/sec."; +"300dpi" = "300dpi"; +"35" = "35"; +"360dpi" = "360dpi"; +"360x180dpi" = "360x180dpi"; +"4" = "4"; +"4 inches/sec." = "4 inches/sec."; +"4.00x1.00\"" = "4.00x1.00\""; +"4.00x13.00\"" = "4.00x13.00\""; +"4.00x2.00\"" = "4.00x2.00\""; +"4.00x2.50\"" = "4.00x2.50\""; +"4.00x3.00\"" = "4.00x3.00\""; +"4.00x4.00\"" = "4.00x4.00\""; +"4.00x5.00\"" = "4.00x5.00\""; +"4.00x6.00\"" = "4.00x6.00\""; +"4.00x6.50\"" = "4.00x6.50\""; +"40" = "40"; +"40 mm/sec." = "40 mm/sec."; +"45" = "45"; +"5" = "5"; +"5 inches/sec." = "5 inches/sec."; +"5 x 7" = "5 x 7"; +"50" = "50"; +"55" = "55"; +"6" = "6"; +"6 inches/sec." = "6 inches/sec."; +"6.00x1.00\"" = "6.00x1.00\""; +"6.00x2.00\"" = "6.00x2.00\""; +"6.00x3.00\"" = "6.00x3.00\""; +"6.00x4.00\"" = "6.00x4.00\""; +"6.00x5.00\"" = "6.00x5.00\""; +"6.00x6.00\"" = "6.00x6.00\""; +"6.00x6.50\"" = "6.00x6.50\""; +"60" = "60"; +"60 mm/sec." = "60 mm/sec."; +"600dpi" = "600dpi"; +"60dpi" = "60dpi"; +"60x72dpi" = "60x72dpi"; +"65" = "65"; +"7" = "7"; +"7 inches/sec." = "7 inches/sec."; +"7 x 9" = "7 x 9"; +"70" = "70"; +"720dpi" = "720dpi"; +"75" = "75"; +"8" = "8"; +"8 inches/sec." = "8 inches/sec."; +"8 x 10" = "8 x 10"; +"8.00x1.00\"" = "8.00x1.00\""; +"8.00x2.00\"" = "8.00x2.00\""; +"8.00x3.00\"" = "8.00x3.00\""; +"8.00x4.00\"" = "8.00x4.00\""; +"8.00x5.00\"" = "8.00x5.00\""; +"8.00x6.00\"" = "8.00x6.00\""; +"8.00x6.50\"" = "8.00x6.50\""; +"80" = "80"; +"80 mm/sec." = "80 mm/sec."; +"85" = "85"; +"9" = "9"; +"9 inches/sec." = "9 inches/sec."; +"9 x 11" = "9 x 11"; +"9 x 12" = "9 x 12"; +"9-Pin Series" = "9-Pin Series"; +"90" = "90"; +"95" = "95"; +"?Invalid help command unknown." = "?Invalid help command unknown."; +"A Samba password is required to export printer drivers" = "A Samba password is required to export printer drivers"; +"A Samba username is required to export printer drivers" = "A Samba username is required to export printer drivers"; +"A class named \"%s\" already exists." = "A class named \"%s\" already exists."; +"A printer named \"%s\" already exists." = "A printer named \"%s\" already exists."; +"A0" = "A0"; +"A0 Long Edge" = "A0 Long Edge"; +"A1" = "A1"; +"A1 Long Edge" = "A1 Long Edge"; +"A10" = "A10"; +"A2" = "A2"; +"A2 Long Edge" = "A2 Long Edge"; +"A3" = "A3"; +"A3 Long Edge" = "A3 Long Edge"; +"A3 Oversize" = "A3 Oversize"; +"A3 Oversize Long Edge" = "A3 Oversize Long Edge"; +"A4" = "A4"; +"A4 Long Edge" = "A4 Long Edge"; +"A4 Oversize" = "A4 Oversize"; +"A4 Small" = "A4 Small"; +"A5" = "A5"; +"A5 Long Edge" = "A5 Long Edge"; +"A5 Oversize" = "A5 Oversize"; +"A6" = "A6"; +"A6 Long Edge" = "A6 Long Edge"; +"A7" = "A7"; +"A8" = "A8"; +"A9" = "A9"; +"ANSI A" = "ANSI A"; +"ANSI B" = "ANSI B"; +"ANSI C" = "ANSI C"; +"ANSI D" = "ANSI D"; +"ANSI E" = "ANSI E"; +"ARCH C" = "ARCH C"; +"ARCH C Long Edge" = "ARCH C Long Edge"; +"ARCH D" = "ARCH D"; +"ARCH D Long Edge" = "ARCH D Long Edge"; +"ARCH E" = "ARCH E"; +"ARCH E Long Edge" = "ARCH E Long Edge"; +"Accept Jobs" = "Accept Jobs"; +"Accepted" = "Accepted"; +"Add Class" = "Add Class"; +"Add Printer" = "Add Printer"; +"Add RSS Subscription" = "Add RSS Subscription"; +"Address" = "Address"; +"Address - 1 1/8 x 3 1/2\"" = "Address - 1 1/8 x 3 1/2\""; +"Administration" = "Administration"; +"Always" = "Always"; +"AppSocket/HP JetDirect" = "AppSocket/HP JetDirect"; +"Applicator" = "Applicator"; +"Attempt to set %s printer-state to bad value %d." = "Attempt to set %s printer-state to bad value %d."; +"Attribute groups are out of order (%x < %x)." = "Attribute groups are out of order (%x < %x)."; +"B0" = "B0"; +"B1" = "B1"; +"B10" = "B10"; +"B2" = "B2"; +"B3" = "B3"; +"B4" = "B4"; +"B5" = "B5"; +"B5 Oversize" = "B5 Oversize"; +"B6" = "B6"; +"B7" = "B7"; +"B8" = "B8"; +"B9" = "B9"; +"Bad NULL dests pointer" = "Bad NULL dests pointer"; +"Bad OpenGroup" = "Bad OpenGroup"; +"Bad OpenUI/JCLOpenUI" = "Bad OpenUI/JCLOpenUI"; +"Bad OrderDependency" = "Bad OrderDependency"; +"Bad PPD cache file." = "Bad PPD cache file."; +"Bad Request" = "Bad Request"; +"Bad SNMP version number" = "Bad SNMP version number"; +"Bad UIConstraints" = "Bad UIConstraints"; +"Bad copies value %d." = "Bad copies value %d."; +"Bad custom parameter" = "Bad custom parameter"; +"Bad device-uri \"%s\"." = "Bad device-uri \"%s\"."; +"Bad device-uri scheme \"%s\"." = "Bad device-uri scheme \"%s\"."; +"Bad document-format \"%s\"." = "Bad document-format \"%s\"."; +"Bad document-format-default \"%s\"." = "Bad document-format-default \"%s\"."; +"Bad filename buffer" = "Bad filename buffer"; +"Bad job-priority value." = "Bad job-priority value."; +"Bad job-sheets value \"%s\"." = "Bad job-sheets value \"%s\"."; +"Bad job-sheets value type." = "Bad job-sheets value type."; +"Bad job-state value." = "Bad job-state value."; +"Bad job-uri \"%s\"." = "Bad job-uri \"%s\"."; +"Bad notify-pull-method \"%s\"." = "Bad notify-pull-method \"%s\"."; +"Bad notify-recipient-uri \"%s\"." = "Bad notify-recipient-uri \"%s\"."; +"Bad number-up value %d." = "Bad number-up value %d."; +"Bad option + choice on line %d." = "Bad option + choice on line %d."; +"Bad page-ranges values %d-%d." = "Bad page-ranges values %d-%d."; +"Bad port-monitor \"%s\"." = "Bad port-monitor \"%s\"."; +"Bad printer-state value %d." = "Bad printer-state value %d."; +"Bad request ID %d." = "Bad request ID %d."; +"Bad request version number %d.%d." = "Bad request version number %d.%d."; +"Bad subscription ID" = "Bad subscription ID"; +"Bad value string" = "Bad value string"; +"Banners" = "Banners"; +"Bond Paper" = "Bond Paper"; +"Boolean expected for waiteof option \"%s\"." = "Boolean expected for waiteof option \"%s\"."; +"Buffer overflow detected, aborting." = "Buffer overflow detected, aborting."; +"CMYK" = "CMYK"; +"CPCL Label Printer" = "CPCL Label Printer"; +"Cancel RSS Subscription" = "Cancel RSS Subscription"; +"Canceling print job." = "Canceling print job."; +"Cannot share a remote Kerberized printer." = "Cannot share a remote Kerberized printer."; +"Cassette" = "Cassette"; +"Change Settings" = "Change Settings"; +"Character set \"%s\" not supported." = "Character set \"%s\" not supported."; +"Classes" = "Classes"; +"Clean Print Heads" = "Clean Print Heads"; +"Close-Job doesn't support the job-uri attribute." = "Close-Job doesn't support the job-uri attribute."; +"Color" = "Color"; +"Color Mode" = "Color Mode"; +"Commands may be abbreviated. Commands are:\n\nexit help quit status ?" = "Commands may be abbreviated. Commands are:\n\nexit help quit status ?"; +"Community name uses indefinite length" = "Community name uses indefinite length"; +"Connected to printer." = "Connected to printer."; +"Connecting to printer." = "Connecting to printer."; +"Continue" = "Continue"; +"Continuous" = "Continuous"; +"Control file sent successfully." = "Control file sent successfully."; +"Copying print data." = "Copying print data."; +"Created" = "Created"; +"Custom" = "Custom"; +"CustominCutInterval" = "CustominCutInterval"; +"CustominTearInterval" = "CustominTearInterval"; +"Cut" = "Cut"; +"Cutter" = "Cutter"; +"Dark" = "Dark"; +"Darkness" = "Darkness"; +"Data file sent successfully." = "Data file sent successfully."; +"Delete Class" = "Delete Class"; +"Delete Printer" = "Delete Printer"; +"DeskJet Series" = "DeskJet Series"; +"Destination \"%s\" is not accepting jobs." = "Destination \"%s\" is not accepting jobs."; +"Device: uri = %s\n class = %s\n info = %s\n make-and-model = %s\n device-id = %s\n location = %s" = "Device: uri = %s\n class = %s\n info = %s\n make-and-model = %s\n device-id = %s\n location = %s"; +"Direct Thermal Media" = "Direct Thermal Media"; +"Directory \"%s\" contains a relative path." = "Directory \"%s\" contains a relative path."; +"Directory \"%s\" has insecure permissions (0%o/uid=%d/gid=%d)." = "Directory \"%s\" has insecure permissions (0%o/uid=%d/gid=%d)."; +"Directory \"%s\" is a file." = "Directory \"%s\" is a file."; +"Directory \"%s\" not available: %s" = "Directory \"%s\" not available: %s"; +"Directory \"%s\" permissions OK (0%o/uid=%d/gid=%d)." = "Directory \"%s\" permissions OK (0%o/uid=%d/gid=%d)."; +"Disabled" = "Disabled"; +"Document #%d does not exist in job #%d." = "Document #%d does not exist in job #%d."; +"Duplexer" = "Duplexer"; +"Dymo" = "Dymo"; +"EPL1 Label Printer" = "EPL1 Label Printer"; +"EPL2 Label Printer" = "EPL2 Label Printer"; +"Edit Configuration File" = "Edit Configuration File"; +"Empty PPD file." = "Empty PPD file."; +// TRANSLATORS: Banner/cover sheet after the print job. +"Ending Banner" = "Ending Banner"; +"English" = "English"; +"Enter old password:" = "Enter old password:"; +"Enter password again:" = "Enter password again:"; +"Enter password:" = "Enter password:"; +"Enter your username and password or the root username and password to access this page. If you are using Kerberos authentication, make sure you have a valid Kerberos ticket." = "Enter your username and password or the root username and password to access this page. If you are using Kerberos authentication, make sure you have a valid Kerberos ticket."; +"Envelope #10 " = "Envelope #10 "; +"Envelope #11" = "Envelope #11"; +"Envelope #12" = "Envelope #12"; +"Envelope #14" = "Envelope #14"; +"Envelope #9" = "Envelope #9"; +"Envelope B4" = "Envelope B4"; +"Envelope B5" = "Envelope B5"; +"Envelope B6" = "Envelope B6"; +"Envelope C0" = "Envelope C0"; +"Envelope C1" = "Envelope C1"; +"Envelope C2" = "Envelope C2"; +"Envelope C3" = "Envelope C3"; +"Envelope C4" = "Envelope C4"; +"Envelope C5" = "Envelope C5"; +"Envelope C6" = "Envelope C6"; +"Envelope C65" = "Envelope C65"; +"Envelope C7" = "Envelope C7"; +"Envelope Choukei 3" = "Envelope Choukei 3"; +"Envelope Choukei 3 Long Edge" = "Envelope Choukei 3 Long Edge"; +"Envelope Choukei 4" = "Envelope Choukei 4"; +"Envelope Choukei 4 Long Edge" = "Envelope Choukei 4 Long Edge"; +"Envelope DL" = "Envelope DL"; +"Envelope Feed" = "Envelope Feed"; +"Envelope Invite" = "Envelope Invite"; +"Envelope Italian" = "Envelope Italian"; +"Envelope Kaku2" = "Envelope Kaku2"; +"Envelope Kaku2 Long Edge" = "Envelope Kaku2 Long Edge"; +"Envelope Kaku3" = "Envelope Kaku3"; +"Envelope Kaku3 Long Edge" = "Envelope Kaku3 Long Edge"; +"Envelope Monarch" = "Envelope Monarch"; +"Envelope PRC1 " = "Envelope PRC1 "; +"Envelope PRC1 Long Edge" = "Envelope PRC1 Long Edge"; +"Envelope PRC10" = "Envelope PRC10"; +"Envelope PRC10 Long Edge" = "Envelope PRC10 Long Edge"; +"Envelope PRC2" = "Envelope PRC2"; +"Envelope PRC2 Long Edge" = "Envelope PRC2 Long Edge"; +"Envelope PRC3" = "Envelope PRC3"; +"Envelope PRC3 Long Edge" = "Envelope PRC3 Long Edge"; +"Envelope PRC4" = "Envelope PRC4"; +"Envelope PRC4 Long Edge" = "Envelope PRC4 Long Edge"; +"Envelope PRC5 Long Edge" = "Envelope PRC5 Long Edge"; +"Envelope PRC5PRC5" = "Envelope PRC5PRC5"; +"Envelope PRC6" = "Envelope PRC6"; +"Envelope PRC6 Long Edge" = "Envelope PRC6 Long Edge"; +"Envelope PRC7" = "Envelope PRC7"; +"Envelope PRC7 Long Edge" = "Envelope PRC7 Long Edge"; +"Envelope PRC8" = "Envelope PRC8"; +"Envelope PRC8 Long Edge" = "Envelope PRC8 Long Edge"; +"Envelope PRC9" = "Envelope PRC9"; +"Envelope PRC9 Long Edge" = "Envelope PRC9 Long Edge"; +"Envelope Personal" = "Envelope Personal"; +"Envelope You4" = "Envelope You4"; +"Envelope You4 Long Edge" = "Envelope You4 Long Edge"; +"Epson" = "Epson"; +"Error Policy" = "Error Policy"; +"Error sending raster data." = "Error sending raster data."; +"Error: need hostname after \"-h\" option." = "Error: need hostname after \"-h\" option."; +"Every 10 Labels" = "Every 10 Labels"; +"Every 2 Labels" = "Every 2 Labels"; +"Every 3 Labels" = "Every 3 Labels"; +"Every 4 Labels" = "Every 4 Labels"; +"Every 5 Labels" = "Every 5 Labels"; +"Every 6 Labels" = "Every 6 Labels"; +"Every 7 Labels" = "Every 7 Labels"; +"Every 8 Labels" = "Every 8 Labels"; +"Every 9 Labels" = "Every 9 Labels"; +"Every Label" = "Every Label"; +"Executive" = "Executive"; +"Expectation Failed" = "Expectation Failed"; +"Export Printers to Samba" = "Export Printers to Samba"; +"FAIL" = "FAIL"; +"FanFold German" = "FanFold German"; +"FanFold Legal German" = "FanFold Legal German"; +"Fanfold US" = "Fanfold US"; +"File \"%s\" contains a relative path." = "File \"%s\" contains a relative path."; +"File \"%s\" has insecure permissions (0%o/uid=%d/gid=%d)." = "File \"%s\" has insecure permissions (0%o/uid=%d/gid=%d)."; +"File \"%s\" is a directory." = "File \"%s\" is a directory."; +"File \"%s\" not available: %s" = "File \"%s\" not available: %s"; +"File \"%s\" permissions OK (0%o/uid=%d/gid=%d)." = "File \"%s\" permissions OK (0%o/uid=%d/gid=%d)."; +"File Folder" = "File Folder"; +"File Folder - 9/16 x 3 7/16\"" = "File Folder - 9/16 x 3 7/16\""; +"File device URIs have been disabled. To enable, see the FileDevice directive in \"%s/cupsd.conf\"." = "File device URIs have been disabled. To enable, see the FileDevice directive in \"%s/cupsd.conf\"."; +"Finished page %d." = "Finished page %d."; +"Folio" = "Folio"; +"Forbidden" = "Forbidden"; +"General" = "General"; +"Generic" = "Generic"; +"Get-Response-PDU uses indefinite length" = "Get-Response-PDU uses indefinite length"; +"Glossy Paper" = "Glossy Paper"; +"Got a printer-uri attribute but no job-id." = "Got a printer-uri attribute but no job-id."; +"Grayscale" = "Grayscale"; +"HP" = "HP"; +"Hanging Folder" = "Hanging Folder"; +"Hanging Folder - 9/16 x 2\"" = "Hanging Folder - 9/16 x 2\""; +"IPP 1setOf attribute with incompatible value tags." = "IPP 1setOf attribute with incompatible value tags."; +"IPP attribute has no name." = "IPP attribute has no name."; +"IPP attribute is not a member of the message." = "IPP attribute is not a member of the message."; +"IPP begCollection value not 0 bytes." = "IPP begCollection value not 0 bytes."; +"IPP boolean value not 1 byte." = "IPP boolean value not 1 byte."; +"IPP date value not 11 bytes." = "IPP date value not 11 bytes."; +"IPP endCollection value not 0 bytes." = "IPP endCollection value not 0 bytes."; +"IPP enum value not 4 bytes." = "IPP enum value not 4 bytes."; +"IPP extension tag larger than 0x7FFFFFFF." = "IPP extension tag larger than 0x7FFFFFFF."; +"IPP integer value not 4 bytes." = "IPP integer value not 4 bytes."; +"IPP language length overflows value." = "IPP language length overflows value."; +"IPP member name is not empty." = "IPP member name is not empty."; +"IPP name larger than 32767 bytes." = "IPP name larger than 32767 bytes."; +"IPP nameWithLanguage value less than minimum 4 bytes." = "IPP nameWithLanguage value less than minimum 4 bytes."; +"IPP rangeOfInteger value not 8 bytes." = "IPP rangeOfInteger value not 8 bytes."; +"IPP resolution value not 9 bytes." = "IPP resolution value not 9 bytes."; +"IPP string length overflows value." = "IPP string length overflows value."; +"IPP textWithLanguage value less than minimum 4 bytes." = "IPP textWithLanguage value less than minimum 4 bytes."; +"IPP value larger than 32767 bytes." = "IPP value larger than 32767 bytes."; +"ISOLatin1" = "ISOLatin1"; +"Illegal control character" = "Illegal control character"; +"Illegal main keyword string" = "Illegal main keyword string"; +"Illegal option keyword string" = "Illegal option keyword string"; +"Illegal translation string" = "Illegal translation string"; +"Illegal whitespace character" = "Illegal whitespace character"; +"Installable Options" = "Installable Options"; +"Installed" = "Installed"; +"IntelliBar Label Printer" = "IntelliBar Label Printer"; +"Intellitech" = "Intellitech"; +"Internal Server Error" = "Internal Server Error"; +"Internal error" = "Internal error"; +"Internet Postage 2-Part" = "Internet Postage 2-Part"; +"Internet Postage 2-Part - 2 1/4 x 7 1/2\"" = "Internet Postage 2-Part - 2 1/4 x 7 1/2\""; +"Internet Postage 3-Part" = "Internet Postage 3-Part"; +"Internet Postage 3-Part - 2 1/4 x 7\"" = "Internet Postage 3-Part - 2 1/4 x 7\""; +"Internet Printing Protocol" = "Internet Printing Protocol"; +"JCL" = "JCL"; +"JIS B0" = "JIS B0"; +"JIS B1" = "JIS B1"; +"JIS B10" = "JIS B10"; +"JIS B2" = "JIS B2"; +"JIS B3" = "JIS B3"; +"JIS B4" = "JIS B4"; +"JIS B4 Long Edge" = "JIS B4 Long Edge"; +"JIS B5" = "JIS B5"; +"JIS B5 Long Edge" = "JIS B5 Long Edge"; +"JIS B6" = "JIS B6"; +"JIS B6 Long Edge" = "JIS B6 Long Edge"; +"JIS B7" = "JIS B7"; +"JIS B8" = "JIS B8"; +"JIS B9" = "JIS B9"; +"Job #%d cannot be restarted - no files." = "Job #%d cannot be restarted - no files."; +"Job #%d does not exist." = "Job #%d does not exist."; +"Job #%d is already aborted - can't cancel." = "Job #%d is already aborted - can't cancel."; +"Job #%d is already canceled - can't cancel." = "Job #%d is already canceled - can't cancel."; +"Job #%d is already completed - can't cancel." = "Job #%d is already completed - can't cancel."; +"Job #%d is finished and cannot be altered." = "Job #%d is finished and cannot be altered."; +"Job #%d is not complete." = "Job #%d is not complete."; +"Job #%d is not held for authentication." = "Job #%d is not held for authentication."; +"Job #%d is not held." = "Job #%d is not held."; +"Job Completed" = "Job Completed"; +"Job Created" = "Job Created"; +"Job Options Changed" = "Job Options Changed"; +"Job Stopped" = "Job Stopped"; +"Job is completed and cannot be changed." = "Job is completed and cannot be changed."; +"Job operation failed:" = "Job operation failed:"; +"Job state cannot be changed." = "Job state cannot be changed."; +"Job subscriptions cannot be renewed." = "Job subscriptions cannot be renewed."; +"Jobs" = "Jobs"; +"LPD/LPR Host or Printer" = "LPD/LPR Host or Printer"; +"Label Printer" = "Label Printer"; +"Label Top" = "Label Top"; +"Language \"%s\" not supported." = "Language \"%s\" not supported."; +"Large Address" = "Large Address"; +"Large Address - 1 4/10 x 3 1/2\"" = "Large Address - 1 4/10 x 3 1/2\""; +"LaserJet Series PCL 4/5" = "LaserJet Series PCL 4/5"; +"Letter Oversize" = "Letter Oversize"; +"Letter Oversize Long Edge" = "Letter Oversize Long Edge"; +"Light" = "Light"; +"Line longer than the maximum allowed (255 characters)" = "Line longer than the maximum allowed (255 characters)"; +"List Available Printers" = "List Available Printers"; +"Long-Edge (Portrait)" = "Long-Edge (Portrait)"; +"Looking for printer." = "Looking for printer."; +"Manual Feed" = "Manual Feed"; +"Media Size" = "Media Size"; +"Media Source" = "Media Source"; +"Media Tracking" = "Media Tracking"; +"Media Type" = "Media Type"; +"Medium" = "Medium"; +"Memory allocation error" = "Memory allocation error"; +"Missing CloseGroup" = "Missing CloseGroup"; +"Missing PPD-Adobe-4.x header" = "Missing PPD-Adobe-4.x header"; +"Missing asterisk in column 1" = "Missing asterisk in column 1"; +"Missing document-number attribute." = "Missing document-number attribute."; +"Missing double quote on line %d." = "Missing double quote on line %d."; +"Missing form variable" = "Missing form variable"; +"Missing last-document attribute in request." = "Missing last-document attribute in request."; +"Missing media or media-col." = "Missing media or media-col."; +"Missing media-size in media-col." = "Missing media-size in media-col."; +"Missing notify-subscription-ids attribute." = "Missing notify-subscription-ids attribute."; +"Missing option keyword" = "Missing option keyword"; +"Missing requesting-user-name attribute." = "Missing requesting-user-name attribute."; +"Missing required attributes." = "Missing required attributes."; +"Missing value on line %d." = "Missing value on line %d."; +"Missing value string" = "Missing value string"; +"Missing x-dimension in media-size." = "Missing x-dimension in media-size."; +"Missing y-dimension in media-size." = "Missing y-dimension in media-size."; +"Model: name = %s\n natural_language = %s\n make-and-model = %s\n device-id = %s" = "Model: name = %s\n natural_language = %s\n make-and-model = %s\n device-id = %s"; +"Modify Class" = "Modify Class"; +"Modify Printer" = "Modify Printer"; +"Move All Jobs" = "Move All Jobs"; +"Move Job" = "Move Job"; +"Moved Permanently" = "Moved Permanently"; +"NULL PPD file pointer" = "NULL PPD file pointer"; +"Name OID uses indefinite length" = "Name OID uses indefinite length"; +"Nested classes are not allowed." = "Nested classes are not allowed."; +"Never" = "Never"; +"New Stylus Color Series" = "New Stylus Color Series"; +"New Stylus Photo Series" = "New Stylus Photo Series"; +"No" = "No"; +"No Content" = "No Content"; +"No PPD name" = "No PPD name"; +"No VarBind SEQUENCE" = "No VarBind SEQUENCE"; +"No Windows printer drivers are installed." = "No Windows printer drivers are installed."; +"No active connection" = "No active connection"; +"No active jobs on %s." = "No active jobs on %s."; +"No attributes in request." = "No attributes in request."; +"No authentication information provided." = "No authentication information provided."; +"No community name" = "No community name"; +"No default printer." = "No default printer."; +"No destinations added." = "No destinations added."; +"No device URI found in argv[0] or in DEVICE_URI environment variable." = "No device URI found in argv[0] or in DEVICE_URI environment variable."; +"No error-index" = "No error-index"; +"No error-status" = "No error-status"; +"No file in print request." = "No file in print request."; +"No modification time" = "No modification time"; +"No name OID" = "No name OID"; +"No pages were found." = "No pages were found."; +"No printer name" = "No printer name"; +"No printer-uri found" = "No printer-uri found"; +"No printer-uri found for class" = "No printer-uri found for class"; +"No printer-uri in request." = "No printer-uri in request."; +"No request-id" = "No request-id"; +"No subscription attributes in request." = "No subscription attributes in request."; +"No subscriptions found." = "No subscriptions found."; +"No variable-bindings SEQUENCE" = "No variable-bindings SEQUENCE"; +"No version number" = "No version number"; +"Non-continuous (Mark sensing)" = "Non-continuous (Mark sensing)"; +"Non-continuous (Web sensing)" = "Non-continuous (Web sensing)"; +"Normal" = "Normal"; +"Not Found" = "Not Found"; +"Not Implemented" = "Not Implemented"; +"Not Installed" = "Not Installed"; +"Not Modified" = "Not Modified"; +"Not Supported" = "Not Supported"; +"Not allowed to print." = "Not allowed to print."; +"Note" = "Note"; +"Note: this program only validates the DSC comments, not the PostScript itself." = "Note: this program only validates the DSC comments, not the PostScript itself."; +"OK" = "OK"; +"Off (1-Sided)" = "Off (1-Sided)"; +"Oki" = "Oki"; +"Online Help" = "Online Help"; +"Open of %s failed: %s" = "Open of %s failed: %s"; +"OpenGroup without a CloseGroup first" = "OpenGroup without a CloseGroup first"; +"OpenUI/JCLOpenUI without a CloseUI/JCLCloseUI first" = "OpenUI/JCLOpenUI without a CloseUI/JCLCloseUI first"; +"Operation Policy" = "Operation Policy"; +"Option \"%s\" cannot be included via %%%%IncludeFeature." = "Option \"%s\" cannot be included via %%%%IncludeFeature."; +"Options Installed" = "Options Installed"; +"Options:" = "Options:"; +"Out of date PPD cache file." = "Out of date PPD cache file."; +"Out of memory." = "Out of memory."; +"Output Mode" = "Output Mode"; +"Output for printer %s is sent to %s" = "Output for printer %s is sent to %s"; +"Output for printer %s is sent to remote printer %s on %s" = "Output for printer %s is sent to remote printer %s on %s"; +"Output for printer %s/%s is sent to %s" = "Output for printer %s/%s is sent to %s"; +"Output for printer %s/%s is sent to remote printer %s on %s" = "Output for printer %s/%s is sent to remote printer %s on %s"; +"PASS" = "PASS"; +"PCL Laser Printer" = "PCL Laser Printer"; +"PRC16K" = "PRC16K"; +"PRC16K Long Edge" = "PRC16K Long Edge"; +"PRC32K" = "PRC32K"; +"PRC32K Long Edge" = "PRC32K Long Edge"; +"PRC32K Oversize" = "PRC32K Oversize"; +"PRC32K Oversize Long Edge" = "PRC32K Oversize Long Edge"; +"Packet does not contain a Get-Response-PDU" = "Packet does not contain a Get-Response-PDU"; +"Packet does not start with SEQUENCE" = "Packet does not start with SEQUENCE"; +"ParamCustominCutInterval" = "ParamCustominCutInterval"; +"ParamCustominTearInterval" = "ParamCustominTearInterval"; +"Password for %s on %s? " = "Password for %s on %s? "; +"Password for %s required to access %s via SAMBA: " = "Password for %s required to access %s via SAMBA: "; +"Pause Class" = "Pause Class"; +"Pause Printer" = "Pause Printer"; +"Peel-Off" = "Peel-Off"; +"Photo" = "Photo"; +"Photo Labels" = "Photo Labels"; +"Plain Paper" = "Plain Paper"; +"Policies" = "Policies"; +"Port Monitor" = "Port Monitor"; +"PostScript Printer" = "PostScript Printer"; +"Postcard" = "Postcard"; +"Postcard Double " = "Postcard Double "; +"Postcard Double Long Edge" = "Postcard Double Long Edge"; +"Postcard Long Edge" = "Postcard Long Edge"; +"Print Density" = "Print Density"; +"Print Job:" = "Print Job:"; +"Print Mode" = "Print Mode"; +"Print Rate" = "Print Rate"; +"Print Self-Test Page" = "Print Self-Test Page"; +"Print Speed" = "Print Speed"; +"Print Test Page" = "Print Test Page"; +"Print and Cut" = "Print and Cut"; +"Print and Tear" = "Print and Tear"; +"Print file accepted - job ID %d." = "Print file accepted - job ID %d."; +"Print file accepted - job ID unknown." = "Print file accepted - job ID unknown."; +"Print file sent." = "Print file sent."; +"Print file was not accepted." = "Print file was not accepted."; +"Printer Added" = "Printer Added"; +"Printer Default" = "Printer Default"; +"Printer Deleted" = "Printer Deleted"; +"Printer Modified" = "Printer Modified"; +"Printer Paused" = "Printer Paused"; +"Printer Settings" = "Printer Settings"; +"Printer busy, will retry in 10 seconds." = "Printer busy, will retry in 10 seconds."; +"Printer did not respond after %d seconds." = "Printer did not respond after %d seconds."; +"Printer does not support IPP/%d.%d, trying IPP/%s." = "Printer does not support IPP/%d.%d, trying IPP/%s."; +"Printer is busy, will retry in 5 seconds." = "Printer is busy, will retry in 5 seconds."; +"Printer is not currently connected." = "Printer is not currently connected."; +"Printer is now connected." = "Printer is now connected."; +"Printer is now online." = "Printer is now online."; +"Printer is offline." = "Printer is offline."; +"Printer not connected, will retry in 30 seconds." = "Printer not connected, will retry in 30 seconds."; +"Printer:" = "Printer:"; +"Printers" = "Printers"; +"Printing page %d, %d%% complete." = "Printing page %d, %d%% complete."; +"Purge Jobs" = "Purge Jobs"; +"Quarto" = "Quarto"; +"Quota limit reached." = "Quota limit reached."; +"Rank Owner Job File(s) Total Size" = "Rank Owner Job File(s) Total Size"; +// TRANSLATORS: Pri is job priority. +"Rank Owner Pri Job Files Total Size" = "Rank Owner Pri Job Files Total Size"; +"Ready to print." = "Ready to print."; +"Reject Jobs" = "Reject Jobs"; +"Remote host did not accept control file (%d)." = "Remote host did not accept control file (%d)."; +"Remote host did not accept data file (%d)." = "Remote host did not accept data file (%d)."; +"Reprint After Error" = "Reprint After Error"; +"Request Entity Too Large" = "Request Entity Too Large"; +"Resolution" = "Resolution"; +"Resume Class" = "Resume Class"; +"Resume Printer" = "Resume Printer"; +"Return Address" = "Return Address"; +"Return Address - 3/4 x 2\"" = "Return Address - 3/4 x 2\""; +"Rewind" = "Rewind"; +"Running command: %s %s -N -A %s -c '%s'" = "Running command: %s %s -N -A %s -c '%s'"; +"SEQUENCE uses indefinite length" = "SEQUENCE uses indefinite length"; +"SSL/TLS Negotiation Error" = "SSL/TLS Negotiation Error"; +"See Other" = "See Other"; +"Sending data to printer." = "Sending data to printer."; +"Server Restarted" = "Server Restarted"; +"Server Security Auditing" = "Server Security Auditing"; +"Server Started" = "Server Started"; +"Server Stopped" = "Server Stopped"; +"Service Unavailable" = "Service Unavailable"; +"Set Allowed Users" = "Set Allowed Users"; +"Set As Server Default" = "Set As Server Default"; +"Set Class Options" = "Set Class Options"; +"Set Printer Options" = "Set Printer Options"; +"Set Publishing" = "Set Publishing"; +"Shipping Address" = "Shipping Address"; +"Shipping Address - 2 5/16 x 4\"" = "Shipping Address - 2 5/16 x 4\""; +"Short-Edge (Landscape)" = "Short-Edge (Landscape)"; +"Special Paper" = "Special Paper"; +"Spooling job, %.0f%% complete." = "Spooling job, %.0f%% complete."; +"Standard" = "Standard"; +// TRANSLATORS: Banner/cover sheet before the print job. +"Starting Banner" = "Starting Banner"; +"Starting page %d." = "Starting page %d."; +"Statement" = "Statement"; +"Stylus Color Series" = "Stylus Color Series"; +"Stylus Photo Series" = "Stylus Photo Series"; +"Subscription #%d does not exist." = "Subscription #%d does not exist."; +"Super A" = "Super A"; +"Super B" = "Super B"; +"Super B/A3" = "Super B/A3"; +"Switching Protocols" = "Switching Protocols"; +"Tabloid" = "Tabloid"; +"Tabloid Oversize" = "Tabloid Oversize"; +"Tabloid Oversize Long Edge" = "Tabloid Oversize Long Edge"; +"Tear" = "Tear"; +"Tear-Off" = "Tear-Off"; +"Tear-Off Adjust Position" = "Tear-Off Adjust Position"; +"The %s attribute cannot be provided with job-ids." = "The %s attribute cannot be provided with job-ids."; +"The PPD file \"%s\" could not be found." = "The PPD file \"%s\" could not be found."; +"The PPD file \"%s\" could not be opened: %s" = "The PPD file \"%s\" could not be opened: %s"; +"The PPD file could not be opened." = "The PPD file could not be opened."; +"The class name may only contain up to 127 printable characters and may not contain spaces, slashes (/), or the pound sign (#)." = "The class name may only contain up to 127 printable characters and may not contain spaces, slashes (/), or the pound sign (#)."; +"The developer unit needs to be replaced." = "The developer unit needs to be replaced."; +"The developer unit will need to be replaced soon." = "The developer unit will need to be replaced soon."; +"The fuser's temperature is high." = "The fuser's temperature is high."; +"The fuser's temperature is low." = "The fuser's temperature is low."; +"The notify-lease-duration attribute cannot be used with job subscriptions." = "The notify-lease-duration attribute cannot be used with job subscriptions."; +"The notify-user-data value is too large (%d > 63 octets)." = "The notify-user-data value is too large (%d > 63 octets)."; +"The optical photoconductor needs to be replaced." = "The optical photoconductor needs to be replaced."; +"The optical photoconductor will need to be replaced soon." = "The optical photoconductor will need to be replaced soon."; +"The output bin is almost full." = "The output bin is almost full."; +"The output bin is full." = "The output bin is full."; +"The output bin is missing." = "The output bin is missing."; +"The paper tray is almost empty." = "The paper tray is almost empty."; +"The paper tray is empty." = "The paper tray is empty."; +"The paper tray is missing." = "The paper tray is missing."; +"The paper tray needs to be filled." = "The paper tray needs to be filled."; +"The printer URI is incorrect or no longer exists." = "The printer URI is incorrect or no longer exists."; +"The printer is busy." = "The printer is busy."; +"The printer is low on ink." = "The printer is low on ink."; +"The printer is low on toner." = "The printer is low on toner."; +"The printer is not connected." = "The printer is not connected."; +"The printer is not responding." = "The printer is not responding."; +"The printer is out of toner." = "The printer is out of toner."; +"The printer is unreachable at this time." = "The printer is unreachable at this time."; +"The printer may be out of ink." = "The printer may be out of ink."; +"The printer may not exist or is unavailable at this time." = "The printer may not exist or is unavailable at this time."; +"The printer name may only contain up to 127 printable characters and may not contain spaces, slashes (/), or the pound sign (#)." = "The printer name may only contain up to 127 printable characters and may not contain spaces, slashes (/), or the pound sign (#)."; +"The printer or class does not exist." = "The printer or class does not exist."; +"The printer or class is not shared." = "The printer or class is not shared."; +"The printer's cover is open." = "The printer's cover is open."; +"The printer's door is open." = "The printer's door is open."; +"The printer's interlock is open." = "The printer's interlock is open."; +"The printer's waste bin is almost full." = "The printer's waste bin is almost full."; +"The printer's waste bin is full." = "The printer's waste bin is full."; +"The printer-uri \"%s\" contains invalid characters." = "The printer-uri \"%s\" contains invalid characters."; +"The printer-uri attribute is required." = "The printer-uri attribute is required."; +"The printer-uri must be of the form \"ipp://HOSTNAME/classes/CLASSNAME\"." = "The printer-uri must be of the form \"ipp://HOSTNAME/classes/CLASSNAME\"."; +"The printer-uri must be of the form \"ipp://HOSTNAME/printers/PRINTERNAME\"." = "The printer-uri must be of the form \"ipp://HOSTNAME/printers/PRINTERNAME\"."; +"The subscription name may not contain spaces, slashes (/), question marks (?), or the pound sign (#)." = "The subscription name may not contain spaces, slashes (/), question marks (?), or the pound sign (#)."; +"The web interface is currently disabled. Run \"cupsctl WebInterface=yes\" to enable it." = "The web interface is currently disabled. Run \"cupsctl WebInterface=yes\" to enable it."; +"The which-jobs value \"%s\" is not supported." = "The which-jobs value \"%s\" is not supported."; +"There are too many subscriptions." = "There are too many subscriptions."; +"There is a paper jam." = "There is a paper jam."; +"There was an unrecoverable USB error." = "There was an unrecoverable USB error."; +"Thermal Transfer Media" = "Thermal Transfer Media"; +"Too many active jobs." = "Too many active jobs."; +"Too many job-sheets values (%d > 2)." = "Too many job-sheets values (%d > 2)."; +"Too many printer-state-reasons values (%d > %d)." = "Too many printer-state-reasons values (%d > %d)."; +"Transparency" = "Transparency"; +"Tray" = "Tray"; +"Tray 1" = "Tray 1"; +"Tray 2" = "Tray 2"; +"Tray 3" = "Tray 3"; +"Tray 4" = "Tray 4"; +"URI Too Long" = "URI Too Long"; +"US Ledger" = "US Ledger"; +"US Legal" = "US Legal"; +"US Legal Oversize" = "US Legal Oversize"; +"US Letter" = "US Letter"; +"US Letter Long Edge" = "US Letter Long Edge"; +"US Letter Oversize" = "US Letter Oversize"; +"US Letter Oversize Long Edge" = "US Letter Oversize Long Edge"; +"US Letter Small" = "US Letter Small"; +"Unable to access cupsd.conf file:" = "Unable to access cupsd.conf file:"; +"Unable to add RSS subscription:" = "Unable to add RSS subscription:"; +"Unable to add class:" = "Unable to add class:"; +"Unable to add document to print job." = "Unable to add document to print job."; +"Unable to add job for destination \"%s\"." = "Unable to add job for destination \"%s\"."; +"Unable to add printer:" = "Unable to add printer:"; +"Unable to allocate memory for file types." = "Unable to allocate memory for file types."; +"Unable to allocate memory for page info" = "Unable to allocate memory for page info"; +"Unable to allocate memory for pages array" = "Unable to allocate memory for pages array"; +"Unable to cancel RSS subscription:" = "Unable to cancel RSS subscription:"; +"Unable to cancel print job." = "Unable to cancel print job."; +"Unable to change printer-is-shared attribute:" = "Unable to change printer-is-shared attribute:"; +"Unable to change printer:" = "Unable to change printer:"; +"Unable to change server settings:" = "Unable to change server settings:"; +"Unable to connect to host." = "Unable to connect to host."; +"Unable to contact printer, queuing on next printer in class." = "Unable to contact printer, queuing on next printer in class."; +"Unable to copy 64-bit CUPS printer driver files (%d)." = "Unable to copy 64-bit CUPS printer driver files (%d)."; +"Unable to copy 64-bit Windows printer driver files (%d)." = "Unable to copy 64-bit Windows printer driver files (%d)."; +"Unable to copy CUPS printer driver files (%d)." = "Unable to copy CUPS printer driver files (%d)."; +"Unable to copy PPD file - %s" = "Unable to copy PPD file - %s"; +"Unable to copy PPD file." = "Unable to copy PPD file."; +"Unable to copy Windows 2000 printer driver files (%d)." = "Unable to copy Windows 2000 printer driver files (%d)."; +"Unable to copy Windows 9x printer driver files (%d)." = "Unable to copy Windows 9x printer driver files (%d)."; +"Unable to copy interface script - %s" = "Unable to copy interface script - %s"; +"Unable to create compressed print file" = "Unable to create compressed print file"; +"Unable to create printer-uri" = "Unable to create printer-uri"; +"Unable to create temporary file" = "Unable to create temporary file"; +"Unable to create temporary file:" = "Unable to create temporary file:"; +"Unable to delete class:" = "Unable to delete class:"; +"Unable to delete printer:" = "Unable to delete printer:"; +"Unable to do maintenance command:" = "Unable to do maintenance command:"; +"Unable to edit cupsd.conf files larger than 1MB" = "Unable to edit cupsd.conf files larger than 1MB"; +"Unable to establish a secure connection to host (certificate chain invalid)." = "Unable to establish a secure connection to host (certificate chain invalid)."; +"Unable to establish a secure connection to host (certificate not yet valid)." = "Unable to establish a secure connection to host (certificate not yet valid)."; +"Unable to establish a secure connection to host (expired certificate)." = "Unable to establish a secure connection to host (expired certificate)."; +"Unable to establish a secure connection to host (host name mismatch)." = "Unable to establish a secure connection to host (host name mismatch)."; +"Unable to establish a secure connection to host (peer dropped connection before responding)." = "Unable to establish a secure connection to host (peer dropped connection before responding)."; +"Unable to establish a secure connection to host (self-signed certificate)." = "Unable to establish a secure connection to host (self-signed certificate)."; +"Unable to establish a secure connection to host (untrusted certificate)." = "Unable to establish a secure connection to host (untrusted certificate)."; +"Unable to establish a secure connection to host." = "Unable to establish a secure connection to host."; +"Unable to find destination for job" = "Unable to find destination for job"; +"Unable to find printer." = "Unable to find printer."; +"Unable to generate compressed print file" = "Unable to generate compressed print file"; +"Unable to get backend exit status." = "Unable to get backend exit status."; +"Unable to get class list:" = "Unable to get class list:"; +"Unable to get class status:" = "Unable to get class status:"; +"Unable to get list of printer drivers:" = "Unable to get list of printer drivers:"; +"Unable to get print job status." = "Unable to get print job status."; +"Unable to get printer attributes:" = "Unable to get printer attributes:"; +"Unable to get printer list:" = "Unable to get printer list:"; +"Unable to get printer status." = "Unable to get printer status."; +"Unable to get printer status:" = "Unable to get printer status:"; +"Unable to install Windows 2000 printer driver files (%d)." = "Unable to install Windows 2000 printer driver files (%d)."; +"Unable to install Windows 9x printer driver files (%d)." = "Unable to install Windows 9x printer driver files (%d)."; +"Unable to locate printer \"%s\"." = "Unable to locate printer \"%s\"."; +"Unable to locate printer." = "Unable to locate printer."; +"Unable to modify class:" = "Unable to modify class:"; +"Unable to modify printer:" = "Unable to modify printer:"; +"Unable to move job" = "Unable to move job"; +"Unable to move jobs" = "Unable to move jobs"; +"Unable to open PPD file" = "Unable to open PPD file"; +"Unable to open PPD file:" = "Unable to open PPD file:"; +"Unable to open compressed print file" = "Unable to open compressed print file"; +"Unable to open cupsd.conf file:" = "Unable to open cupsd.conf file:"; +"Unable to open device file" = "Unable to open device file"; +"Unable to open document #%d in job #%d." = "Unable to open document #%d in job #%d."; +"Unable to open print file" = "Unable to open print file"; +"Unable to open raster file" = "Unable to open raster file"; +"Unable to print test page:" = "Unable to print test page:"; +"Unable to read print data" = "Unable to read print data"; +"Unable to read print data." = "Unable to read print data."; +"Unable to run \"%s\": %s" = "Unable to run \"%s\": %s"; +"Unable to see in file" = "Unable to see in file"; +"Unable to send command to printer driver" = "Unable to send command to printer driver"; +"Unable to send data to printer." = "Unable to send data to printer."; +"Unable to set Windows printer driver (%d)." = "Unable to set Windows printer driver (%d)."; +"Unable to set options:" = "Unable to set options:"; +"Unable to set server default:" = "Unable to set server default:"; +"Unable to start backend process." = "Unable to start backend process."; +"Unable to upload cupsd.conf file:" = "Unable to upload cupsd.conf file:"; +"Unable to use legacy USB class driver." = "Unable to use legacy USB class driver."; +"Unable to write print data" = "Unable to write print data"; +"Unable to write uncompressed print data: %s" = "Unable to write uncompressed print data: %s"; +"Unauthorized" = "Unauthorized"; +"Units" = "Units"; +"Unknown" = "Unknown"; +"Unknown choice \"%s\" for option \"%s\"." = "Unknown choice \"%s\" for option \"%s\"."; +"Unknown encryption option value: \"%s\"." = "Unknown encryption option value: \"%s\"."; +"Unknown file order: \"%s\"." = "Unknown file order: \"%s\"."; +"Unknown format character: \"%c\"." = "Unknown format character: \"%c\"."; +"Unknown option \"%s\" with value \"%s\"." = "Unknown option \"%s\" with value \"%s\"."; +"Unknown option \"%s\"." = "Unknown option \"%s\"."; +"Unknown print mode: \"%s\"." = "Unknown print mode: \"%s\"."; +"Unknown printer-error-policy \"%s\"." = "Unknown printer-error-policy \"%s\"."; +"Unknown printer-op-policy \"%s\"." = "Unknown printer-op-policy \"%s\"."; +"Unknown version option value: \"%s\"." = "Unknown version option value: \"%s\"."; +"Unsupported brightness value %s, using brightness=100." = "Unsupported brightness value %s, using brightness=100."; +"Unsupported character set \"%s\"." = "Unsupported character set \"%s\"."; +"Unsupported compression \"%s\"." = "Unsupported compression \"%s\"."; +"Unsupported document-format \"%s\"." = "Unsupported document-format \"%s\"."; +"Unsupported document-format \"%s/%s\"." = "Unsupported document-format \"%s/%s\"."; +"Unsupported format \"%s\"." = "Unsupported format \"%s\"."; +"Unsupported gamma value %s, using gamma=1000." = "Unsupported gamma value %s, using gamma=1000."; +"Unsupported margins." = "Unsupported margins."; +"Unsupported media value." = "Unsupported media value."; +"Unsupported number-up value %d, using number-up=1." = "Unsupported number-up value %d, using number-up=1."; +"Unsupported number-up-layout value %s, using number-up-layout=lrtb." = "Unsupported number-up-layout value %s, using number-up-layout=lrtb."; +"Unsupported page-border value %s, using page-border=none." = "Unsupported page-border value %s, using page-border=none."; +"Unsupported raster data." = "Unsupported raster data."; +"Unsupported value type" = "Unsupported value type"; +"Upgrade Required" = "Upgrade Required"; +"Usage:\n\n lpadmin [-h server] -d destination\n lpadmin [-h server] -x destination\n lpadmin [-h server] -p printer [-c add-class] [-i interface] [-m model]\n [-r remove-class] [-v device] [-D description]\n [-P ppd-file] [-o name=value]\n [-u allow:user,user] [-u deny:user,user]" = "Usage:\n\n lpadmin [-h server] -d destination\n lpadmin [-h server] -x destination\n lpadmin [-h server] -p printer [-c add-class] [-i interface] [-m model]\n [-r remove-class] [-v device] [-D description]\n [-P ppd-file] [-o name=value]\n [-u allow:user,user] [-u deny:user,user]"; +"Usage: %s job-id user title copies options [file]" = "Usage: %s job-id user title copies options [file]"; +"Usage: %s job-id user title copies options file" = "Usage: %s job-id user title copies options file"; +"Usage: convert [ options ]" = "Usage: convert [ options ]"; +"Usage: cupsaddsmb [options] printer1 ... printerN" = "Usage: cupsaddsmb [options] printer1 ... printerN"; +"Usage: cupsctl [options] [param=value ... paramN=valueN]" = "Usage: cupsctl [options] [param=value ... paramN=valueN]"; +"Usage: cupsd [options]" = "Usage: cupsd [options]"; +"Usage: cupsfilter [ options ] filename" = "Usage: cupsfilter [ options ] filename"; +"Usage: cupstestdsc [options] filename.ps [... filename.ps]" = "Usage: cupstestdsc [options] filename.ps [... filename.ps]"; +"Usage: cupstestppd [options] filename1.ppd[.gz] [... filenameN.ppd[.gz]]" = "Usage: cupstestppd [options] filename1.ppd[.gz] [... filenameN.ppd[.gz]]"; +"Usage: ipptool [options] URI filename [ ... filenameN ]" = "Usage: ipptool [options] URI filename [ ... filenameN ]"; +"Usage: lpmove job/src dest" = "Usage: lpmove job/src dest"; +"Usage: lpoptions [-h server] [-E] -d printer\n lpoptions [-h server] [-E] [-p printer] -l\n lpoptions [-h server] [-E] -p printer -o option[=value] ...\n lpoptions [-h server] [-E] -x printer" = "Usage: lpoptions [-h server] [-E] -d printer\n lpoptions [-h server] [-E] [-p printer] -l\n lpoptions [-h server] [-E] -p printer -o option[=value] ...\n lpoptions [-h server] [-E] -x printer"; +"Usage: lppasswd [-g groupname]" = "Usage: lppasswd [-g groupname]"; +"Usage: lppasswd [-g groupname] [username]\n lppasswd [-g groupname] -a [username]\n lppasswd [-g groupname] -x [username]" = "Usage: lppasswd [-g groupname] [username]\n lppasswd [-g groupname] -a [username]\n lppasswd [-g groupname] -x [username]"; +"Usage: lpq [-P dest] [-U username] [-h hostname[:port]] [-l] [+interval]" = "Usage: lpq [-P dest] [-U username] [-h hostname[:port]] [-l] [+interval]"; +"Usage: ppdc [options] filename.drv [ ... filenameN.drv ]" = "Usage: ppdc [options] filename.drv [ ... filenameN.drv ]"; +"Usage: ppdhtml [options] filename.drv >filename.html" = "Usage: ppdhtml [options] filename.drv >filename.html"; +"Usage: ppdi [options] filename.ppd [ ... filenameN.ppd ]" = "Usage: ppdi [options] filename.ppd [ ... filenameN.ppd ]"; +"Usage: ppdmerge [options] filename.ppd [ ... filenameN.ppd ]" = "Usage: ppdmerge [options] filename.ppd [ ... filenameN.ppd ]"; +"Usage: ppdpo [options] -o filename.po filename.drv [ ... filenameN.drv ]" = "Usage: ppdpo [options] -o filename.po filename.drv [ ... filenameN.drv ]"; +"Usage: snmp [host-or-ip-address]" = "Usage: snmp [host-or-ip-address]"; +"Value uses indefinite length" = "Value uses indefinite length"; +"VarBind uses indefinite length" = "VarBind uses indefinite length"; +"Version uses indefinite length" = "Version uses indefinite length"; +"Waiting for job to complete." = "Waiting for job to complete."; +"Waiting for printer to become available." = "Waiting for printer to become available."; +"Waiting for printer to finish." = "Waiting for printer to finish."; +"Warning, no Windows 2000 printer drivers are installed." = "Warning, no Windows 2000 printer drivers are installed."; +"Web Interface is Disabled" = "Web Interface is Disabled"; +"Yes" = "Yes"; +"You must access this page using the URL https://%s:%d%s." = "You must access this page using the URL https://%s:%d%s."; +"Your password must be at least 6 characters long, cannot contain your username, and must contain at least one letter and number." = "Your password must be at least 6 characters long, cannot contain your username, and must contain at least one letter and number."; +"ZPL Label Printer" = "ZPL Label Printer"; +"Zebra" = "Zebra"; +"aborted" = "aborted"; +"canceled" = "canceled"; +"completed" = "completed"; +"convert: Use the -f option to specify a file to convert." = "convert: Use the -f option to specify a file to convert."; +"cups-deviced failed to execute." = "cups-deviced failed to execute."; +"cups-driverd failed to execute." = "cups-driverd failed to execute."; +"cupsaddsmb: No PPD file for printer \"%s\" - %s" = "cupsaddsmb: No PPD file for printer \"%s\" - %s"; +"cupsctl: Cannot set Listen or Port directly." = "cupsctl: Cannot set Listen or Port directly."; +"cupsctl: Unable to connect to server: %s" = "cupsctl: Unable to connect to server: %s"; +"cupsctl: Unknown option \"%s\"" = "cupsctl: Unknown option \"%s\""; +"cupsctl: Unknown option \"-%c\"" = "cupsctl: Unknown option \"-%c\""; +"cupsd: Expected config filename after \"-c\" option." = "cupsd: Expected config filename after \"-c\" option."; +"cupsd: Unable to get current directory." = "cupsd: Unable to get current directory."; +"cupsd: Unknown argument \"%s\" - aborting." = "cupsd: Unknown argument \"%s\" - aborting."; +"cupsd: Unknown option \"%c\" - aborting." = "cupsd: Unknown option \"%c\" - aborting."; +"cupsd: launchd(8) support not compiled in, running in normal mode." = "cupsd: launchd(8) support not compiled in, running in normal mode."; +"cupsfilter: Invalid document number %d." = "cupsfilter: Invalid document number %d."; +"cupsfilter: Invalid job ID %d." = "cupsfilter: Invalid job ID %d."; +"cupsfilter: Only one filename can be specified." = "cupsfilter: Only one filename can be specified."; +"cupsfilter: Unable to get job file - %s" = "cupsfilter: Unable to get job file - %s"; +"cupstestppd: The -q option is incompatible with the -v option." = "cupstestppd: The -q option is incompatible with the -v option."; +"cupstestppd: The -v option is incompatible with the -q option." = "cupstestppd: The -v option is incompatible with the -q option."; +"device for %s/%s: %s" = "device for %s/%s: %s"; +"device for %s: %s" = "device for %s: %s"; +"error-index uses indefinite length" = "error-index uses indefinite length"; +"error-status uses indefinite length" = "error-status uses indefinite length"; +"held" = "held"; +"help\t\tGet help on commands." = "help\t\tGet help on commands."; +"idle" = "idle"; +"ipptool: \"-i\" and \"-n\" are incompatible with -X\"." = "ipptool: \"-i\" and \"-n\" are incompatible with -X\"."; +"ipptool: \"-i\" is incompatible with \"-X\"." = "ipptool: \"-i\" is incompatible with \"-X\"."; +"ipptool: \"-n\" is incompatible with \"-X\"." = "ipptool: \"-n\" is incompatible with \"-X\"."; +"ipptool: Bad URI - %s." = "ipptool: Bad URI - %s."; +"ipptool: Bad version %s for \"-V\"." = "ipptool: Bad version %s for \"-V\"."; +"ipptool: Invalid seconds for \"-i\"." = "ipptool: Invalid seconds for \"-i\"."; +"ipptool: May only specify a single URI." = "ipptool: May only specify a single URI."; +"ipptool: Missing count for \"-n\"." = "ipptool: Missing count for \"-n\"."; +"ipptool: Missing filename for \"-f\"." = "ipptool: Missing filename for \"-f\"."; +"ipptool: Missing name=value for \"-d\"." = "ipptool: Missing name=value for \"-d\"."; +"ipptool: Missing seconds for \"-i\"." = "ipptool: Missing seconds for \"-i\"."; +"ipptool: Missing timeout for \"-T\"." = "ipptool: Missing timeout for \"-T\"."; +"ipptool: Missing version for \"-V\"." = "ipptool: Missing version for \"-V\"."; +"ipptool: URI required before test file." = "ipptool: URI required before test file."; +"ipptool: Unknown option \"-%c\"." = "ipptool: Unknown option \"-%c\"."; +"job-printer-uri attribute missing." = "job-printer-uri attribute missing."; +"lpadmin: Class name can only contain printable characters." = "lpadmin: Class name can only contain printable characters."; +"lpadmin: Expected PPD after \"-P\" option." = "lpadmin: Expected PPD after \"-P\" option."; +"lpadmin: Expected allow/deny:userlist after \"-u\" option." = "lpadmin: Expected allow/deny:userlist after \"-u\" option."; +"lpadmin: Expected class after \"-r\" option." = "lpadmin: Expected class after \"-r\" option."; +"lpadmin: Expected class name after \"-c\" option." = "lpadmin: Expected class name after \"-c\" option."; +"lpadmin: Expected description after \"-D\" option." = "lpadmin: Expected description after \"-D\" option."; +"lpadmin: Expected device URI after \"-v\" option." = "lpadmin: Expected device URI after \"-v\" option."; +"lpadmin: Expected file type(s) after \"-I\" option." = "lpadmin: Expected file type(s) after \"-I\" option."; +"lpadmin: Expected hostname after \"-h\" option." = "lpadmin: Expected hostname after \"-h\" option."; +"lpadmin: Expected interface after \"-i\" option." = "lpadmin: Expected interface after \"-i\" option."; +"lpadmin: Expected location after \"-L\" option." = "lpadmin: Expected location after \"-L\" option."; +"lpadmin: Expected model after \"-m\" option." = "lpadmin: Expected model after \"-m\" option."; +"lpadmin: Expected name after \"-R\" option." = "lpadmin: Expected name after \"-R\" option."; +"lpadmin: Expected name=value after \"-o\" option." = "lpadmin: Expected name=value after \"-o\" option."; +"lpadmin: Expected printer after \"-p\" option." = "lpadmin: Expected printer after \"-p\" option."; +"lpadmin: Expected printer name after \"-d\" option." = "lpadmin: Expected printer name after \"-d\" option."; +"lpadmin: Expected printer or class after \"-x\" option." = "lpadmin: Expected printer or class after \"-x\" option."; +"lpadmin: No member names were seen." = "lpadmin: No member names were seen."; +"lpadmin: Printer %s is already a member of class %s." = "lpadmin: Printer %s is already a member of class %s."; +"lpadmin: Printer %s is not a member of class %s." = "lpadmin: Printer %s is not a member of class %s."; +"lpadmin: Printer name can only contain printable characters." = "lpadmin: Printer name can only contain printable characters."; +"lpadmin: Unable to add a printer to the class:\n You must specify a printer name first." = "lpadmin: Unable to add a printer to the class:\n You must specify a printer name first."; +"lpadmin: Unable to connect to server: %s" = "lpadmin: Unable to connect to server: %s"; +"lpadmin: Unable to create temporary file" = "lpadmin: Unable to create temporary file"; +"lpadmin: Unable to delete option:\n You must specify a printer name first." = "lpadmin: Unable to delete option:\n You must specify a printer name first."; +"lpadmin: Unable to open PPD file \"%s\" - %s" = "lpadmin: Unable to open PPD file \"%s\" - %s"; +"lpadmin: Unable to remove a printer from the class:\n You must specify a printer name first." = "lpadmin: Unable to remove a printer from the class:\n You must specify a printer name first."; +"lpadmin: Unable to set the printer options:\n You must specify a printer name first." = "lpadmin: Unable to set the printer options:\n You must specify a printer name first."; +"lpadmin: Unknown allow/deny option \"%s\"." = "lpadmin: Unknown allow/deny option \"%s\"."; +"lpadmin: Unknown argument \"%s\"." = "lpadmin: Unknown argument \"%s\"."; +"lpadmin: Unknown option \"%c\"." = "lpadmin: Unknown option \"%c\"."; +"lpadmin: Warning - content type list ignored." = "lpadmin: Warning - content type list ignored."; +"lpc> " = "lpc> "; +"lpinfo: Expected 1284 device ID string after \"--device-id\"." = "lpinfo: Expected 1284 device ID string after \"--device-id\"."; +"lpinfo: Expected language after \"--language\"." = "lpinfo: Expected language after \"--language\"."; +"lpinfo: Expected make and model after \"--make-and-model\"." = "lpinfo: Expected make and model after \"--make-and-model\"."; +"lpinfo: Expected product string after \"--product\"." = "lpinfo: Expected product string after \"--product\"."; +"lpinfo: Expected scheme list after \"--exclude-schemes\"." = "lpinfo: Expected scheme list after \"--exclude-schemes\"."; +"lpinfo: Expected scheme list after \"--include-schemes\"." = "lpinfo: Expected scheme list after \"--include-schemes\"."; +"lpinfo: Expected timeout after \"--timeout\"." = "lpinfo: Expected timeout after \"--timeout\"."; +"lpinfo: Unknown argument \"%s\"." = "lpinfo: Unknown argument \"%s\"."; +"lpinfo: Unknown option \"%c\"." = "lpinfo: Unknown option \"%c\"."; +"lpinfo: Unknown option \"%s\"." = "lpinfo: Unknown option \"%s\"."; +"lpmove: Unable to connect to server: %s" = "lpmove: Unable to connect to server: %s"; +"lpmove: Unknown argument \"%s\"." = "lpmove: Unknown argument \"%s\"."; +"lpmove: Unknown option \"%c\"." = "lpmove: Unknown option \"%c\"."; +"lpoptions: No printers." = "lpoptions: No printers."; +"lpoptions: Unable to add printer or instance: %s" = "lpoptions: Unable to add printer or instance: %s"; +"lpoptions: Unable to get PPD file for %s: %s" = "lpoptions: Unable to get PPD file for %s: %s"; +"lpoptions: Unable to open PPD file for %s." = "lpoptions: Unable to open PPD file for %s."; +"lpoptions: Unknown printer or class." = "lpoptions: Unknown printer or class."; +"lppasswd: Only root can add or delete passwords." = "lppasswd: Only root can add or delete passwords."; +"lppasswd: Password file busy." = "lppasswd: Password file busy."; +"lppasswd: Password file not updated." = "lppasswd: Password file not updated."; +"lppasswd: Sorry, password doesn't match." = "lppasswd: Sorry, password doesn't match."; +"lppasswd: Sorry, password rejected." = "lppasswd: Sorry, password rejected."; +"lppasswd: Sorry, passwords don't match." = "lppasswd: Sorry, passwords don't match."; +"lppasswd: Unable to copy password string: %s" = "lppasswd: Unable to copy password string: %s"; +"lppasswd: Unable to open password file: %s" = "lppasswd: Unable to open password file: %s"; +"lppasswd: Unable to write to password file: %s" = "lppasswd: Unable to write to password file: %s"; +"lppasswd: failed to backup old password file: %s" = "lppasswd: failed to backup old password file: %s"; +"lppasswd: failed to rename password file: %s" = "lppasswd: failed to rename password file: %s"; +"lppasswd: user \"%s\" and group \"%s\" do not exist." = "lppasswd: user \"%s\" and group \"%s\" do not exist."; +"lpstat: error - %s environment variable names non-existent destination \"%s\"." = "lpstat: error - %s environment variable names non-existent destination \"%s\"."; +"members of class %s:" = "members of class %s:"; +"no entries" = "no entries"; +"no system default destination" = "no system default destination"; +"notify-events not specified." = "notify-events not specified."; +"notify-recipient-uri URI \"%s\" is already used." = "notify-recipient-uri URI \"%s\" is already used."; +"notify-recipient-uri URI \"%s\" uses unknown scheme." = "notify-recipient-uri URI \"%s\" uses unknown scheme."; +"pending" = "pending"; +"ppdc: Adding include directory \"%s\"." = "ppdc: Adding include directory \"%s\"."; +"ppdc: Adding/updating UI text from %s." = "ppdc: Adding/updating UI text from %s."; +"ppdc: Bad boolean value (%s) on line %d of %s." = "ppdc: Bad boolean value (%s) on line %d of %s."; +"ppdc: Bad font attribute: %s" = "ppdc: Bad font attribute: %s"; +"ppdc: Bad resolution name \"%s\" on line %d of %s." = "ppdc: Bad resolution name \"%s\" on line %d of %s."; +"ppdc: Bad status keyword %s on line %d of %s." = "ppdc: Bad status keyword %s on line %d of %s."; +"ppdc: Bad variable substitution ($%c) on line %d of %s." = "ppdc: Bad variable substitution ($%c) on line %d of %s."; +"ppdc: Choice found on line %d of %s with no Option." = "ppdc: Choice found on line %d of %s with no Option."; +"ppdc: Duplicate #po for locale %s on line %d of %s." = "ppdc: Duplicate #po for locale %s on line %d of %s."; +"ppdc: Expected a filter definition on line %d of %s." = "ppdc: Expected a filter definition on line %d of %s."; +"ppdc: Expected a program name on line %d of %s." = "ppdc: Expected a program name on line %d of %s."; +"ppdc: Expected boolean value on line %d of %s." = "ppdc: Expected boolean value on line %d of %s."; +"ppdc: Expected charset after Font on line %d of %s." = "ppdc: Expected charset after Font on line %d of %s."; +"ppdc: Expected choice code on line %d of %s." = "ppdc: Expected choice code on line %d of %s."; +"ppdc: Expected choice name/text on line %d of %s." = "ppdc: Expected choice name/text on line %d of %s."; +"ppdc: Expected color order for ColorModel on line %d of %s." = "ppdc: Expected color order for ColorModel on line %d of %s."; +"ppdc: Expected colorspace for ColorModel on line %d of %s." = "ppdc: Expected colorspace for ColorModel on line %d of %s."; +"ppdc: Expected compression for ColorModel on line %d of %s." = "ppdc: Expected compression for ColorModel on line %d of %s."; +"ppdc: Expected constraints string for UIConstraints on line %d of %s." = "ppdc: Expected constraints string for UIConstraints on line %d of %s."; +"ppdc: Expected driver type keyword following DriverType on line %d of %s." = "ppdc: Expected driver type keyword following DriverType on line %d of %s."; +"ppdc: Expected duplex type after Duplex on line %d of %s." = "ppdc: Expected duplex type after Duplex on line %d of %s."; +"ppdc: Expected encoding after Font on line %d of %s." = "ppdc: Expected encoding after Font on line %d of %s."; +"ppdc: Expected filename after #po %s on line %d of %s." = "ppdc: Expected filename after #po %s on line %d of %s."; +"ppdc: Expected group name/text on line %d of %s." = "ppdc: Expected group name/text on line %d of %s."; +"ppdc: Expected include filename on line %d of %s." = "ppdc: Expected include filename on line %d of %s."; +"ppdc: Expected integer on line %d of %s." = "ppdc: Expected integer on line %d of %s."; +"ppdc: Expected locale after #po on line %d of %s." = "ppdc: Expected locale after #po on line %d of %s."; +"ppdc: Expected name after %s on line %d of %s." = "ppdc: Expected name after %s on line %d of %s."; +"ppdc: Expected name after FileName on line %d of %s." = "ppdc: Expected name after FileName on line %d of %s."; +"ppdc: Expected name after Font on line %d of %s." = "ppdc: Expected name after Font on line %d of %s."; +"ppdc: Expected name after Manufacturer on line %d of %s." = "ppdc: Expected name after Manufacturer on line %d of %s."; +"ppdc: Expected name after MediaSize on line %d of %s." = "ppdc: Expected name after MediaSize on line %d of %s."; +"ppdc: Expected name after ModelName on line %d of %s." = "ppdc: Expected name after ModelName on line %d of %s."; +"ppdc: Expected name after PCFileName on line %d of %s." = "ppdc: Expected name after PCFileName on line %d of %s."; +"ppdc: Expected name/text after %s on line %d of %s." = "ppdc: Expected name/text after %s on line %d of %s."; +"ppdc: Expected name/text after Installable on line %d of %s." = "ppdc: Expected name/text after Installable on line %d of %s."; +"ppdc: Expected name/text after Resolution on line %d of %s." = "ppdc: Expected name/text after Resolution on line %d of %s."; +"ppdc: Expected name/text combination for ColorModel on line %d of %s." = "ppdc: Expected name/text combination for ColorModel on line %d of %s."; +"ppdc: Expected option name/text on line %d of %s." = "ppdc: Expected option name/text on line %d of %s."; +"ppdc: Expected option section on line %d of %s." = "ppdc: Expected option section on line %d of %s."; +"ppdc: Expected option type on line %d of %s." = "ppdc: Expected option type on line %d of %s."; +"ppdc: Expected override field after Resolution on line %d of %s." = "ppdc: Expected override field after Resolution on line %d of %s."; +"ppdc: Expected quoted string on line %d of %s." = "ppdc: Expected quoted string on line %d of %s."; +"ppdc: Expected real number on line %d of %s." = "ppdc: Expected real number on line %d of %s."; +"ppdc: Expected resolution/mediatype following ColorProfile on line %d of %s." = "ppdc: Expected resolution/mediatype following ColorProfile on line %d of %s."; +"ppdc: Expected resolution/mediatype following SimpleColorProfile on line %d of %s." = "ppdc: Expected resolution/mediatype following SimpleColorProfile on line %d of %s."; +"ppdc: Expected selector after %s on line %d of %s." = "ppdc: Expected selector after %s on line %d of %s."; +"ppdc: Expected status after Font on line %d of %s." = "ppdc: Expected status after Font on line %d of %s."; +"ppdc: Expected string after Copyright on line %d of %s." = "ppdc: Expected string after Copyright on line %d of %s."; +"ppdc: Expected string after Version on line %d of %s." = "ppdc: Expected string after Version on line %d of %s."; +"ppdc: Expected two option names on line %d of %s." = "ppdc: Expected two option names on line %d of %s."; +"ppdc: Expected value after %s on line %d of %s." = "ppdc: Expected value after %s on line %d of %s."; +"ppdc: Expected version after Font on line %d of %s." = "ppdc: Expected version after Font on line %d of %s."; +"ppdc: Invalid #include/#po filename \"%s\"." = "ppdc: Invalid #include/#po filename \"%s\"."; +"ppdc: Invalid cost for filter on line %d of %s." = "ppdc: Invalid cost for filter on line %d of %s."; +"ppdc: Invalid empty MIME type for filter on line %d of %s." = "ppdc: Invalid empty MIME type for filter on line %d of %s."; +"ppdc: Invalid empty program name for filter on line %d of %s." = "ppdc: Invalid empty program name for filter on line %d of %s."; +"ppdc: Invalid option section \"%s\" on line %d of %s." = "ppdc: Invalid option section \"%s\" on line %d of %s."; +"ppdc: Invalid option type \"%s\" on line %d of %s." = "ppdc: Invalid option type \"%s\" on line %d of %s."; +"ppdc: Loading driver information file \"%s\"." = "ppdc: Loading driver information file \"%s\"."; +"ppdc: Loading messages for locale \"%s\"." = "ppdc: Loading messages for locale \"%s\"."; +"ppdc: Loading messages from \"%s\"." = "ppdc: Loading messages from \"%s\"."; +"ppdc: Missing #endif at end of \"%s\"." = "ppdc: Missing #endif at end of \"%s\"."; +"ppdc: Missing #if on line %d of %s." = "ppdc: Missing #if on line %d of %s."; +"ppdc: Need a msgid line before any translation strings on line %d of %s." = "ppdc: Need a msgid line before any translation strings on line %d of %s."; +"ppdc: No message catalog provided for locale %s." = "ppdc: No message catalog provided for locale %s."; +"ppdc: Option %s defined in two different groups on line %d of %s." = "ppdc: Option %s defined in two different groups on line %d of %s."; +"ppdc: Option %s redefined with a different type on line %d of %s." = "ppdc: Option %s redefined with a different type on line %d of %s."; +"ppdc: Option constraint must *name on line %d of %s." = "ppdc: Option constraint must *name on line %d of %s."; +"ppdc: Too many nested #if's on line %d of %s." = "ppdc: Too many nested #if's on line %d of %s."; +"ppdc: Unable to create PPD file \"%s\" - %s." = "ppdc: Unable to create PPD file \"%s\" - %s."; +"ppdc: Unable to create output directory %s: %s" = "ppdc: Unable to create output directory %s: %s"; +"ppdc: Unable to create output pipes: %s" = "ppdc: Unable to create output pipes: %s"; +"ppdc: Unable to execute cupstestppd: %s" = "ppdc: Unable to execute cupstestppd: %s"; +"ppdc: Unable to find #po file %s on line %d of %s." = "ppdc: Unable to find #po file %s on line %d of %s."; +"ppdc: Unable to find include file \"%s\" on line %d of %s." = "ppdc: Unable to find include file \"%s\" on line %d of %s."; +"ppdc: Unable to find localization for \"%s\" - %s" = "ppdc: Unable to find localization for \"%s\" - %s"; +"ppdc: Unable to load localization file \"%s\" - %s" = "ppdc: Unable to load localization file \"%s\" - %s"; +"ppdc: Unable to open %s: %s" = "ppdc: Unable to open %s: %s"; +"ppdc: Undefined variable (%s) on line %d of %s." = "ppdc: Undefined variable (%s) on line %d of %s."; +"ppdc: Unexpected text on line %d of %s." = "ppdc: Unexpected text on line %d of %s."; +"ppdc: Unknown driver type %s on line %d of %s." = "ppdc: Unknown driver type %s on line %d of %s."; +"ppdc: Unknown duplex type \"%s\" on line %d of %s." = "ppdc: Unknown duplex type \"%s\" on line %d of %s."; +"ppdc: Unknown media size \"%s\" on line %d of %s." = "ppdc: Unknown media size \"%s\" on line %d of %s."; +"ppdc: Unknown message catalog format for \"%s\"." = "ppdc: Unknown message catalog format for \"%s\"."; +"ppdc: Unknown token \"%s\" seen on line %d of %s." = "ppdc: Unknown token \"%s\" seen on line %d of %s."; +"ppdc: Unknown trailing characters in real number \"%s\" on line %d of %s." = "ppdc: Unknown trailing characters in real number \"%s\" on line %d of %s."; +"ppdc: Unterminated string starting with %c on line %d of %s." = "ppdc: Unterminated string starting with %c on line %d of %s."; +"ppdc: Warning - overlapping filename \"%s\"." = "ppdc: Warning - overlapping filename \"%s\"."; +"ppdc: Writing %s." = "ppdc: Writing %s."; +"ppdc: Writing PPD files to directory \"%s\"." = "ppdc: Writing PPD files to directory \"%s\"."; +"ppdmerge: Bad LanguageVersion \"%s\" in %s." = "ppdmerge: Bad LanguageVersion \"%s\" in %s."; +"ppdmerge: Ignoring PPD file %s." = "ppdmerge: Ignoring PPD file %s."; +"ppdmerge: Unable to backup %s to %s - %s" = "ppdmerge: Unable to backup %s to %s - %s"; +"printer %s disabled since %s -" = "printer %s disabled since %s -"; +"printer %s is idle. enabled since %s" = "printer %s is idle. enabled since %s"; +"printer %s now printing %s-%d. enabled since %s" = "printer %s now printing %s-%d. enabled since %s"; +"printer %s/%s disabled since %s -" = "printer %s/%s disabled since %s -"; +"printer %s/%s is idle. enabled since %s" = "printer %s/%s is idle. enabled since %s"; +"printer %s/%s now printing %s-%d. enabled since %s" = "printer %s/%s now printing %s-%d. enabled since %s"; +"processing" = "processing"; +"request id is %s-%d (%d file(s))" = "request id is %s-%d (%d file(s))"; +"request-id uses indefinite length" = "request-id uses indefinite length"; +"scheduler is not running" = "scheduler is not running"; +"scheduler is running" = "scheduler is running"; +"stat of %s failed: %s" = "stat of %s failed: %s"; +"status\t\tShow status of daemon and queue." = "status\t\tShow status of daemon and queue."; +"stopped" = "stopped"; +"system default destination: %s" = "system default destination: %s"; +"system default destination: %s/%s" = "system default destination: %s/%s"; +"unknown" = "unknown"; +"untitled" = "untitled"; +"variable-bindings uses indefinite length" = "variable-bindings uses indefinite length"; diff --git a/locale/locale.txt b/locale/locale.txt new file mode 100644 index 0000000000..f9abe72d67 --- /dev/null +++ b/locale/locale.txt @@ -0,0 +1,32 @@ +This directory contains the message strings used by CUPS for various +languages. Each subdirectory corresponds to a different locale, and +the cups_xx and cups_xx_YY files contain the messages for the locales +named "xx" or "xx_YY". + +Each message file starts with a character set identifier, which can be +one of the following: + + us-ascii + iso-8859-1 + iso-8859-2 + iso-8859-3 + iso-8859-4 + iso-8859-5 + iso-8859-6 + iso-8859-7 + iso-8859-8 + iso-8859-9 + utf-8 + +After that, all non-blank lines are treated as messages, with any +leading whitespace removed. If a line starts with a number then the +message index is updated to the number. Otherwise, the next message +number is used. + +The message indices are defined in the include file . +The HTTP status messages use the status codes defined in . + +If you would like to contribute a new message file for your locale, or +have corrections to the current ones, please send them to: + + cups-support@cups.org diff --git a/locale/po2strings.c b/locale/po2strings.c new file mode 100644 index 0000000000..430ace499f --- /dev/null +++ b/locale/po2strings.c @@ -0,0 +1,281 @@ +/* + * "$Id$" + * + * Convert a GNU gettext .po file to an Apple .strings file. + * + * Copyright 2007-2010 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Usage: + * + * po2strings filename.strings filename.po + * + * Compile with: + * + * gcc -o po2strings po2strings.c `cups-config --libs` + * + * Contents: + * + * main() - Convert .po file to .strings. + */ + +#include + + +/* + * The .strings file format is simple: + * + * // comment + * "msgid" = "msgstr"; + * + * The GNU gettext .po format is also fairly simple: + * + * #. comment + * msgid "some text" + * msgstr "localized text" + * + * The comment, msgid, and msgstr text can span multiple lines using the form: + * + * #. comment + * #. more comments + * msgid "" + * "some long text" + * msgstr "" + * "localized text spanning " + * "multiple lines" + * + * Both the msgid and msgstr strings use standard C quoting for special + * characters like newline and the double quote character. + */ + +/* + * main() - Convert .po file to .strings. + */ + +int /* O - Exit code */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + const char *pofile, /* .po filename */ + *stringsfile; /* .strings filename */ + cups_file_t *po, /* .po file */ + *strings; /* .strings file */ + char s[4096], /* String buffer */ + *ptr, /* Pointer into buffer */ + *temp, /* New string */ + *msgid, /* msgid string */ + *msgstr; /* msgstr string */ + int length; /* Length of combined strings */ + int use_msgid; /* Use msgid strings for msgstr? */ + + + /* + * Process command-line arguments... + */ + + pofile = NULL; + stringsfile = NULL; + use_msgid = 0; + + for (i = 1; i < argc; i ++) + { + if (!strcmp(argv[i], "-m")) + use_msgid = 1; + else if (argv[i][0] == '-') + { + puts("Usage: po2strings [-m] filename.po filename.strings"); + return (1); + } + else if (!pofile) + pofile = argv[i]; + else if (!stringsfile) + stringsfile = argv[i]; + else + { + puts("Usage: po2strings [-m] filename.po filename.strings"); + return (1); + } + } + + if (!pofile || !stringsfile) + { + puts("Usage: po2strings [-m] filename.po filename.strings"); + return (1); + } + + /* + * Read strings from the .po file and write to the .strings file... + */ + + if ((po = cupsFileOpen(pofile, "r")) == NULL) + { + perror(pofile); + return (1); + } + + if ((strings = cupsFileOpen(stringsfile, "w")) == NULL) + { + perror(stringsfile); + cupsFileClose(po); + return (1); + } + + msgid = msgstr = NULL; + + while (cupsFileGets(po, s, sizeof(s)) != NULL) + { + if (s[0] == '#' && s[1] == '.') + { + /* + * Copy comment string... + */ + + if (msgid && msgstr) + { + /* + * First output the last localization string... + */ + + if (*msgid) + cupsFilePrintf(strings, "\"%s\" = \"%s\";\n", msgid, + (use_msgid || !*msgstr) ? msgid : msgstr); + + free(msgid); + free(msgstr); + msgid = msgstr = NULL; + } + + cupsFilePrintf(strings, "//%s\n", s + 2); + } + else if (s[0] == '#' || !s[0]) + { + /* + * Skip blank and file comment lines... + */ + + continue; + } + else + { + /* + * Strip the trailing quote... + */ + + if ((ptr = strrchr(s, '\"')) == NULL) + continue; + + *ptr = '\0'; + + /* + * Find start of value... + */ + + if ((ptr = strchr(s, '\"')) == NULL) + continue; + + ptr ++; + + /* + * Create or add to a message... + */ + + if (!strncmp(s, "msgid", 5)) + { + /* + * Output previous message as needed... + */ + + if (msgid && msgstr) + { + if (*msgid) + cupsFilePrintf(strings, "\"%s\" = \"%s\";\n", msgid, + (use_msgid || !*msgstr) ? msgid : msgstr); + + free(msgid); + free(msgstr); + } + + msgid = strdup(ptr); + msgstr = NULL; + } + else if (s[0] == '\"' ) + { + /* + * Append to current string... + */ + + length = (int)strlen(msgstr ? msgstr : msgid); + + if ((temp = realloc(msgstr ? msgstr : msgid, + length + strlen(ptr) + 1)) == NULL) + { + perror("Unable to allocate string"); + return (1); + } + + if (msgstr) + { + /* + * Copy the new portion to the end of the msgstr string - safe + * to use strcpy because the buffer is allocated to the correct + * size... + */ + + msgstr = temp; + + strcpy(msgstr + length, ptr); + } + else + { + /* + * Copy the new portion to the end of the msgid string - safe + * to use strcpy because the buffer is allocated to the correct + * size... + */ + + msgid = temp; + + strcpy(msgid + length, ptr); + } + } + else if (!strncmp(s, "msgstr", 6) && msgid) + { + /* + * Set the string... + */ + + if ((msgstr = strdup(ptr)) == NULL) + { + perror("Unable to allocate msgstr"); + return (1); + } + } + } + } + + if (msgid && msgstr) + { + if (*msgid) + cupsFilePrintf(strings, "\"%s\" = \"%s\";\n", msgid, + (use_msgid || !*msgstr) ? msgid : msgstr); + + free(msgid); + free(msgstr); + } + + cupsFileClose(po); + cupsFileClose(strings); + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/locale/strings2po.c b/locale/strings2po.c new file mode 100644 index 0000000000..8df9059be4 --- /dev/null +++ b/locale/strings2po.c @@ -0,0 +1,175 @@ +/* + * "$Id$" + * + * Convert Apple .strings file (UTF-16 BE text file) to GNU gettext .po files. + * + * Usage: + * + * strings2po filename.strings filename.po + * + * Compile with: + * + * gcc -o strings2po strings2po.c + * + * Contents: + * + * main() - Convert .strings file to .po. + * read_strings() - Read a line from a .strings file. + * write_po() - Write a line to the .po file. + */ + +#include +#include + + +/* + * The .strings file format is simple: + * + * // comment + * "id" = "str"; + * + * Both the id and str strings use standard C quoting for special characters + * like newline and the double quote character. + */ + +/* + * Local functions... + */ + +static int read_strings(FILE *strings, char *buffer, size_t bufsize, + char **id, char **str); +static void write_po(FILE *po, const char *what, const char *s); + + +/* + * main() - Convert .strings file to .po. + */ + +int /* O - Exit code */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + FILE *strings, /* .strings file */ + *po; /* .po file */ + char iconv[1024], /* iconv command */ + buffer[8192], /* Line buffer */ + *id, /* ID string */ + *str; /* Translation string */ + int count; /* Number of messages converted */ + + + if (argc != 3) + { + puts("Usage: strings2po filename.strings filename.po"); + return (1); + } + + /* + * Cheat by using iconv to convert the .strings file from UTF-16 to UTF-8 + * which is what we need for the .po file (and it makes things a lot + * simpler...) + */ + + snprintf(iconv, sizeof(iconv), "iconv -f utf-16 -t utf-8 '%s'", argv[1]); + if ((strings = popen(iconv, "r")) == NULL) + { + perror(argv[1]); + return (1); + } + + if ((po = fopen(argv[2], "w")) == NULL) + { + perror(argv[2]); + pclose(strings); + return (1); + } + + count = 0; + + while (read_strings(strings, buffer, sizeof(buffer), &id, &str)) + { + count ++; + write_po(po, "msgid", id); + write_po(po, "msgstr", str); + } + + pclose(strings); + fclose(po); + + printf("%s: %d messages.\n", argv[2], count); + + return (0); +} + + +/* + * 'read_strings()' - Read a line from a .strings file. + */ + +static int /* O - 1 on success, 0 on failure */ +read_strings(FILE *strings, /* I - .strings file */ + char *buffer, /* I - Line buffer */ + size_t bufsize, /* I - Size of line buffer */ + char **id, /* O - Pointer to ID string */ + char **str) /* O - Pointer to translation string */ +{ + char *bufptr; /* Pointer into buffer */ + + + while (fgets(buffer, bufsize, strings)) + { + if (buffer[0] != '\"') + continue; + + *id = buffer + 1; + + for (bufptr = buffer + 1; *bufptr && *bufptr != '\"'; bufptr ++) + if (*bufptr == '\\') + bufptr ++; + + if (*bufptr != '\"') + continue; + + *bufptr++ = '\0'; + + while (*bufptr && *bufptr != '\"') + bufptr ++; + + if (!*bufptr) + continue; + + bufptr ++; + *str = bufptr; + + for (; *bufptr && *bufptr != '\"'; bufptr ++) + if (*bufptr == '\\') + bufptr ++; + + if (*bufptr != '\"') + continue; + + *bufptr = '\0'; + + return (1); + } + + return (0); +} + + +/* + * 'write_po()' - Write a line to the .po file. + */ + +static void +write_po(FILE *po, /* I - .po file */ + const char *what, /* I - Type of string */ + const char *s) /* I - String to write */ +{ + fprintf(po, "%s \"%s\"\n", what, s); +} + + +/* + * End of "$Id$". + */ diff --git a/locale/translate.c b/locale/translate.c new file mode 100644 index 0000000000..13228d27a2 --- /dev/null +++ b/locale/translate.c @@ -0,0 +1,439 @@ +/* + * "$Id$" + * + * HTTP-based translation program for CUPS. + * + * This program uses Google to translate the CUPS template (cups.pot) to + * several different languages. The translation isn't perfect, but it's + * a start (better than working from scratch.) + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Main entry. + * save_messages() - Save messages to a .po file. + * translate_messages() - Translate messages using Google. + * write_string() - Write a quoted string to a file. + */ + +/* + * Include necessary headers... + */ + +#include +#include + + +/* + * Local functions... + */ + +int save_messages(cups_array_t *cat, const char *filename); +int translate_messages(cups_array_t *cat, const char *lang); +int write_string(cups_file_t *fp, const char *s); + + +/* + * 'main()' - Main entry. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + cups_array_t *cat; /* Message catalog */ + + + if (argc != 3) + { + fputs("Usage: translate cups_language.po language\n", stderr); + return (1); + } + + if (access(argv[1], 0)) + cat = _cupsMessageLoad("cups.pot", 1); + else + cat = _cupsMessageLoad(argv[1], 1); + + if (!cat) + { + puts("Unable to load message catalog."); + return (1); + } + + if (!translate_messages(cat, argv[2])) + { + puts("Unable to translate message catalog."); + return (1); + } + + if (!save_messages(cat, argv[1])) + { + puts("Unable to save message catalog."); + return (1); + } + + return (0); +} + + +/* + * 'save_messages()' - Save messages to a .po file. + */ + +int /* O - 1 on success, 0 on error */ +save_messages(cups_array_t *cat, /* I - Message catalog */ + const char *filename) /* I - File to save to */ +{ + _cups_message_t *m; /* Current message */ + cups_file_t *fp; /* File pointer */ + + + /* + * Open the message catalog... + */ + + if ((fp = cupsFileOpen(filename, "w")) == NULL) + return (0); + + /* + * Save the messages to a file... + */ + + for (m = (_cups_message_t *)cupsArrayFirst(cat); + m; + m = (_cups_message_t *)cupsArrayNext(cat)) + { + if (cupsFilePuts(fp, "msgid \"") < 0) + break; + + if (!write_string(fp, m->id)) + break; + + if (cupsFilePuts(fp, "\"\nmsgstr \"") < 0) + break; + + if (m->str) + { + if (!write_string(fp, m->str)) + break; + } + + if (cupsFilePuts(fp, "\"\n") < 0) + break; + } + + cupsFileClose(fp); + + return (!m); +} + + +/* + * 'translate_messages()' - Translate messages using Google. + */ + +int /* O - 1 on success, 0 on error */ +translate_messages(cups_array_t *cat, /* I - Message catalog */ + const char *lang) /* I - Output language... */ +{ + /* + * Google provides a simple translation/language tool for translating + * from one language to another. It is far from perfect, however it + * can be used to get a basic translation done or update an existing + * translation when no other resources are available. + * + * Translation requests are sent as HTTP POSTs to + * "http://translate.google.com/translate_t" with the following form + * variables: + * + * Name Description Value + * -------- ---------------------------------- ---------------- + * hl Help language? "en" + * ie Input encoding "UTF8" + * langpair Language pair "en|" + language + * oe Output encoding "UTF8" + * text Text to translate translation string + */ + + int ret; /* Return value */ + _cups_message_t *m; /* Current message */ + int tries; /* Number of tries... */ + http_t *http; /* HTTP connection */ + http_status_t status; /* Status of POST request */ + char *idptr, /* Pointer into msgid */ + buffer[65536], /* Input/output buffer */ + *bufptr, /* Pointer into buffer */ + *bufend, /* Pointer to end of buffer */ + length[16]; /* Content length */ + int bytes; /* Number of bytes read */ + + + /* + * Connect to translate.google.com... + */ + + puts("Connecting to translate.google.com..."); + + if ((http = httpConnect("translate.google.com", 80)) == NULL) + { + perror("Unable to connect to translate.google.com"); + return (0); + } + + /* + * Scan the current messages, requesting a translation of any untranslated + * messages... + */ + + for (m = (_cups_message_t *)cupsArrayFirst(cat), ret = 1; + m; + m = (_cups_message_t *)cupsArrayNext(cat)) + { + /* + * Skip messages that are already translated... + */ + + if (m->str && m->str[0]) + continue; + + /* + * Encode the form data into the buffer... + */ + + snprintf(buffer, sizeof(buffer), + "hl=en&ie=UTF8&langpair=en|%s&oe=UTF8&text=", lang); + bufptr = buffer + strlen(buffer); + bufend = buffer + sizeof(buffer) - 5; + + for (idptr = m->id; *idptr && bufptr < bufend; idptr ++) + if (*idptr == ' ') + *bufptr++ = '+'; + else if (*idptr < ' ' || *idptr == '%') + { + sprintf(bufptr, "%%%02X", *idptr & 255); + bufptr += 3; + } + else if (*idptr != '&') + *bufptr++ = *idptr; + + *bufptr++ = '&'; + *bufptr = '\0'; + + sprintf(length, "%d", (int)(bufptr - buffer)); + + /* + * Send the request... + */ + + printf("\"%s\" = ", m->id); + fflush(stdout); + + tries = 0; + + do + { + httpClearFields(http); + httpSetField(http, HTTP_FIELD_CONTENT_TYPE, + "application/x-www-form-urlencoded"); + httpSetField(http, HTTP_FIELD_CONTENT_LENGTH, length); + + if (httpPost(http, "/translate_t")) + { + httpReconnect(http); + httpPost(http, "/translate_t"); + } + + httpWrite2(http, buffer, bufptr - buffer); + + while ((status = httpUpdate(http)) == HTTP_CONTINUE); + + if (status != HTTP_OK && status != HTTP_ERROR) + httpFlush(http); + + tries ++; + } + while (status == HTTP_ERROR && tries < 10); + + if (status == HTTP_OK) + { + /* + * OK, read the translation back... + */ + + bufptr = buffer; + bufend = buffer + sizeof(buffer) - 1; + + while ((bytes = httpRead2(http, bufptr, bufend - bufptr)) > 0) + bufptr += bytes; + + if (bytes < 0) + { + /* + * Read error, abort! + */ + + puts("READ ERROR!"); + ret = 0; + break; + } + + *bufptr = '\0'; + + /* + * Find the div containing translation + */ + + if ((bufptr = strstr(buffer, "
')) == NULL) + { + /* + * textarea doesn't end, abort! + */ + + puts("DIV SHORT DATA!"); + ret = 0; + break; + } + + bufptr ++; + + if ((bufend = strstr(bufptr, "
")) == NULL) + { + /* + * textarea doesn't close, abort! + */ + + puts("/DIV SHORT DATA!"); + ret = 0; + break; + } + + *bufend = '\0'; + + /* + * Copy the translation... + */ + + m->str = strdup(bufptr); + + /* + * Convert character entities to regular chars... + */ + + for (bufptr = strchr(m->str, '&'); + bufptr; + bufptr = strchr(bufptr + 1, '&')) + { + if (!strncmp(bufptr, "<", 4)) + { + *bufptr = '<'; + _cups_strcpy(bufptr + 1, bufptr + 4); + } + else if (!strncmp(bufptr, ">", 4)) + { + *bufptr = '>'; + _cups_strcpy(bufptr + 1, bufptr + 4); + } + else if (!strncmp(bufptr, "&", 5)) + _cups_strcpy(bufptr + 1, bufptr + 5); + } + + printf("\"%s\"\n", m->str); + } + else if (status == HTTP_ERROR) + { + printf("NETWORK ERROR (%s)!\n", strerror(httpError(http))); + ret = 0; + break; + } + else + { + printf("HTTP ERROR %d!\n", status); + ret = 0; + break; + } + } + + httpClose(http); + + return (ret); +} + + +/* + * 'write_string()' - Write a quoted string to a file. + */ + +int /* O - 1 on success, 0 on failure */ +write_string(cups_file_t *fp, /* I - File to write to */ + const char *s) /* I - String */ +{ + while (*s) + { + switch (*s) + { + case '\n' : + if (cupsFilePuts(fp, "\\n") < 0) + return (0); + break; + + case '\r' : + if (cupsFilePuts(fp, "\\r") < 0) + return (0); + break; + + case '\t' : + if (cupsFilePuts(fp, "\\t") < 0) + return (0); + break; + + case '\\' : + if (cupsFilePuts(fp, "\\\\") < 0) + return (0); + break; + + case '\"' : + if (cupsFilePuts(fp, "\\\"") < 0) + return (0); + break; + + default : + if ((*s & 255) < ' ') + { + if (cupsFilePrintf(fp, "\\%o", *s) < 0) + return (0); + } + else if (cupsFilePutChar(fp, *s) < 0) + return (0); + break; + } + + s ++; + } + + return (1); +} + + +/* + * End of "$Id$". + */ diff --git a/man/Makefile b/man/Makefile new file mode 100644 index 0000000000..1ec8389f48 --- /dev/null +++ b/man/Makefile @@ -0,0 +1,240 @@ +# +# "$Id$" +# +# Man page makefile for CUPS. +# +# Copyright 2007-2012 by Apple Inc. +# Copyright 1993-2006 by Easy Software Products. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +include ../Makedefs + + +# +# Man pages... +# + +MAN1 = cancel.$(MAN1EXT) \ + cups-config.$(MAN1EXT) \ + cupstestdsc.$(MAN1EXT) \ + cupstestppd.$(MAN1EXT) \ + ipptool.$(MAN1EXT) \ + lp.$(MAN1EXT) \ + lpoptions.$(MAN1EXT) \ + lppasswd.$(MAN1EXT) \ + lpq.$(MAN1EXT) \ + lprm.$(MAN1EXT) \ + lpr.$(MAN1EXT) \ + lpstat.$(MAN1EXT) \ + ppdc.$(MAN1EXT) \ + ppdhtml.$(MAN1EXT) \ + ppdi.$(MAN1EXT) \ + ppdmerge.$(MAN1EXT) \ + ppdpo.$(MAN1EXT) +MAN5 = classes.conf.$(MAN5EXT) \ + client.conf.$(MAN5EXT) \ + cups-snmp.conf.$(MAN5EXT) \ + cupsd.conf.$(MAN5EXT) \ + ipptoolfile.$(MAN5EXT) \ + mailto.conf.$(MAN5EXT) \ + mime.convs.$(MAN5EXT) \ + mime.types.$(MAN5EXT) \ + ppdcfile.$(MAN5EXT) \ + printers.conf.$(MAN5EXT) \ + subscriptions.conf.$(MAN5EXT) +MAN7 = backend.$(MAN7EXT) \ + filter.$(MAN7EXT) \ + notifier.$(MAN7EXT) +MAN8 = cupsaccept.$(MAN8EXT) \ + cupsaddsmb.$(MAN8EXT) \ + cupsctl.$(MAN8EXT) \ + cupsfilter.$(MAN8EXT) \ + cups-deviced.$(MAN8EXT) \ + cups-driverd.$(MAN8EXT) \ + cups-lpd.$(MAN8EXT) \ + cups-polld.$(MAN8EXT) \ + cupsd.$(MAN8EXT) \ + cupsenable.$(MAN8EXT) \ + lpadmin.$(MAN8EXT) \ + lpinfo.$(MAN8EXT) \ + lpmove.$(MAN8EXT) \ + lpc.$(MAN8EXT) + + +# +# Make everything... +# + +all: $(MAN1) $(MAN5) $(MAN7) $(MAN8) html + + +# +# Make library targets... +# + +libs: + + +# +# Make unit tests... +# + +unittests: + + +# +# Clean all config and object files... +# + +clean: + $(RM) mantohtml mantohtml.o + $(RM) $(MAN1) $(MAN5) $(MAN7) $(MAN8) + for file in $(MAN1); do \ + if test $$file != ipptool.$(MAN1EXT); then \ + $(RM) ../doc/help/man-`basename $$file .$(MAN1EXT)`.html; \ + fi \ + done + for file in $(MAN5); do \ + if test $$file != ipptoolfile.$(MAN5EXT); then \ + $(RM) ../doc/help/man-`basename $$file .$(MAN5EXT)`.html; \ + fi \ + done + for file in $(MAN7); do \ + $(RM) ../doc/help/man-`basename $$file .$(MAN7EXT)`.html; \ + done + for file in $(MAN8); do \ + $(RM) ../doc/help/man-`basename $$file .$(MAN8EXT)`.html; \ + done + + +# +# Dummy depend target... +# + +depend: + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: all + $(INSTALL_DIR) -m 755 $(MANDIR)/man1 + for file in $(MAN1); do \ + echo Installing $$file in $(MANDIR)/man1...; \ + $(INSTALL_MAN) $$file $(MANDIR)/man1; \ + done + $(INSTALL_DIR) -m 755 $(MANDIR)/man5 + for file in $(MAN5); do \ + echo Installing $$file in $(MANDIR)/man5...; \ + $(INSTALL_MAN) $$file $(MANDIR)/man5; \ + done + $(INSTALL_DIR) -m 755 $(MANDIR)/man7 + for file in $(MAN7); do \ + echo Installing $$file in $(MANDIR)/man7...; \ + $(INSTALL_MAN) $$file $(MANDIR)/man7; \ + done + $(INSTALL_DIR) -m 755 $(AMANDIR)/man$(MAN8DIR) + for file in $(MAN8); do \ + echo Installing $$file in $(AMANDIR)/man$(MAN8DIR)...; \ + $(INSTALL_MAN) $$file $(AMANDIR)/man$(MAN8DIR); \ + done + for file in accept cupsreject reject; do \ + $(RM) $(AMANDIR)/man$(MAN8DIR)/$$file.$(MAN8EXT); \ + $(LN) cupsaccept.$(MAN8EXT) $(AMANDIR)/man$(MAN8DIR)/$$file.$(MAN8EXT); \ + done + $(RM) $(AMANDIR)/man$(MAN8DIR)/cupsdisable.$(MAN8EXT) + $(LN) cupsenable.$(MAN8EXT) $(AMANDIR)/man$(MAN8DIR)/cupsdisable.$(MAN8EXT) + + +# +# Install programs... +# + +install-exec: + + +# +# Install headers... +# + +install-headers: + + +# +# Install libraries... +# + +install-libs: + + +# +# Uninstall files... +# + +uninstall: + for file in $(MAN1); do \ + $(RM) $(MANDIR)/man1/$$file; \ + done + -$(RMDIR) $(MANDIR)/man1 + for file in $(MAN5); do \ + $(RM) $(MANDIR)/man5/$$file; \ + done + -$(RMDIR) $(MANDIR)/man5 + for file in $(MAN7); do \ + $(RM) $(MANDIR)/man7/$$file; \ + done + -$(RMDIR) $(MANDIR)/man7 + for file in $(MAN8); do \ + $(RM) $(AMANDIR)/man$(MAN8DIR)/$$file; \ + done + $(RM) $(AMANDIR)/man$(MAN8DIR)/accept.$(MAN8EXT) + $(RM) $(AMANDIR)/man$(MAN8DIR)/cupsreject.$(MAN8EXT) + $(RM) $(AMANDIR)/man$(MAN8DIR)/reject.$(MAN8EXT) + $(RM) $(AMANDIR)/man$(MAN8DIR)/cupsdisable.$(MAN8EXT) + -$(RMDIR) $(AMANDIR)/man$(MAN8DIR) + + +# +# Make html versions of man pages... +# + +html: $(MAN1) $(MAN5) $(MAN7) $(MAN8) mantohtml + echo Converting man pages to HTML... + for file in $(MAN1); do \ + echo " $$file..."; \ + ./mantohtml `basename $$file .$(MAN1EXT)`.man >../doc/help/man-`basename $$file .$(MAN1EXT)`.html; \ + done + for file in $(MAN5); do \ + echo " $$file..."; \ + ./mantohtml `basename $$file .$(MAN5EXT)`.man >../doc/help/man-`basename $$file .$(MAN5EXT)`.html; \ + done + for file in $(MAN7); do \ + echo " $$file..."; \ + ./mantohtml `basename $$file .$(MAN7EXT)`.man >../doc/help/man-`basename $$file .$(MAN7EXT)`.html; \ + done + for file in $(MAN8); do \ + echo " $$file..."; \ + ./mantohtml `basename $$file .$(MAN8EXT)`.man >../doc/help/man-`basename $$file .$(MAN8EXT)`.html; \ + done + +mantohtml: mantohtml.o + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o $@ mantohtml.o + + +# +# End of "$Id$". +# diff --git a/man/backend.man b/man/backend.man new file mode 100644 index 0000000000..644d85dd48 --- /dev/null +++ b/man/backend.man @@ -0,0 +1,196 @@ +.\" +.\" "$Id: backend.man 9693 2011-04-16 02:51:22Z mike $" +.\" +.\" Backend man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH backend 7 "CUPS" "15 April 2011" "Apple Inc." + +.SH NAME +backend \- cups backend transmission interfaces + +.SH SYNOPSIS +.B backend +.br +.B backend +job user title num-copies options [ +.I filename +] + +.SH DESCRIPTION +Backends are a special type of \fIfilter(7)\fR which is used to +send print data to and discover different devices on the system. + +.LP +Like filters, backends must be capable of reading from a filename +on the command-line or from the standard input, copying the +standard input to a temporary file as required by the physical +interface. + +.LP +The command name (argv[0]) is set to the device URI of the destination printer. +Starting with CUPS 1.1.22, any authentication information in argv[0] is removed, +so backend developers are urged to use the DEVICE_URI environment variable +whenever authentication information is required. The CUPS API includes a +\fIcupsBackendDeviceURI\fR function for retrieving the correct device URI. + +.LP +Back-channel data from the device should be relayed to the job +filters by writing to file descriptor 3. The CUPS API includes +the \fIcupsBackChannelWrite\fR function for this purpose. + +.SH DEVICE DISCOVERY +When run with no arguments, the backend should list the devices +and schemes it supports or is advertising to stdout. The output +consists of zero or more lines consisting of any of the following +forms: + +.nf + device-class scheme "Unknown" "device-info" + device-class device-uri "device-make-and-model" "device-info" + device-class device-uri "device-make-and-model" "device-info" "device-id" + device-class device-uri "device-make-and-model" "device-info" "device-id" "device-location" +.fi + +.LP +The \fIdevice-class\fR field is one of the following values: + +.TP 5 +direct +.br +The device-uri refers to a specific direct-access device with no +options, such as a parallel, USB, or SCSI device. + +.TP 5 +file +.br +The device-uri refers to a file on disk. + +.TP 5 +network +.br +The device-uri refers to a networked device and conforms to the +general form for network URIs. + +.TP 5 +serial +.br +The device-uri refers to a serial device with configurable baud +rate and other options. If the device-uri contains a baud value, +it represents the maximum baud rate supported by the device. + +.LP +The \fIscheme\fR field provides the URI scheme that is supported +by the backend. Backends should use this form only when the +backend supports any URI using that scheme. The \fIdevice-uri\fR +field specifies the full URI to use when communicating with the +device. + +.LP +The \fIdevice-make-and-model\fR field specifies the make and +model of the device, e.g. "Acme Foojet 2000". If the make and +model is not known, you must report "Unknown". + +.LP +The \fIdevice-info\fR field specifies additional information +about the device. Typically this includes the make and model +along with the port number or network address, e.g. "Acme Foojet +2000 USB #1". + +.LP +The optional \fIdevice-id\fR field specifies the IEEE-1284 device +ID string for the device, which is used to select a matching +driver. + +.LP +The optional \fIdevice-location\fR field specifies the physical location of +the device, which is often used to pre-populate the printer-location attribute +when adding a printer. + +.SH PERMISSIONS +Backends without world execute permissions are run as the root +user. Otherwise, the backend is run using the unprivileged user +account, typically "lp". + +.SH EXIT CODES +The following exit codes are defined for backends; C API +constants defined in the header file are defined +in parenthesis: + +.TP 5 +0 (CUPS_BACKEND_OK) +.br +The print file was successfully transmitted to the device or +remote server. + +.TP 5 +1 (CUPS_BACKEND_FAILED) +.br +The print file was not successfully transmitted to the device or +remote server. The scheduler will respond to this by canceling +the job, retrying the job, or stopping the queue depending on the +state of the error-policy attribute. + +.TP 5 +2 (CUPS_BACKEND_AUTH_REQUIRED) +.br +The print file was not successfully transmitted because valid +authentication information is required. The scheduler will +respond to this by holding the job and adding the +authentication-required job-reasons keyword. + +.TP 5 +3 (CUPS_BACKEND_HOLD) +.br +The print file was not successfully transmitted because it cannot +be printed at this time. The scheduler will respond to this by +holding the job. + +.TP 5 +4 (CUPS_BACKEND_STOP) +.br +The print file was not successfully transmitted because it cannot +be printed at this time. The scheduler will respond to this by +stopping the queue. + +.TP 5 +5 (CUPS_BACKEND_CANCEL) +.br +The print file was not successfully transmitted because one or +more attributes are not supported. The scheduler will respond to +this by canceling the job. + +.TP 5 +6 (CUPS_BACKEND_RETRY) +.br +The print file was not successfully transmitted because of a temporary issue. +The scheduler will retry the job at a future time - other jobs may print before +this one. + +.TP 5 +7 (CUPS_BACKEND_RETRY_CURRENT) +.br +The print file was not successfully transmitted because of a temporary issue. +The scheduler will retry the job immediately without allowing intervening jobs. + +.PP +All other exit code values are reserved. + +.SH SEE ALSO +\fIcupsd(8)\fR, \fIcupsd.conf(5)\fR, \fIfilter(7)\fR, +.br +http://localhost:631/help + +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: backend.man 9693 2011-04-16 02:51:22Z mike $". +.\" diff --git a/man/cancel.man b/man/cancel.man new file mode 100644 index 0000000000..eda61eb1f4 --- /dev/null +++ b/man/cancel.man @@ -0,0 +1,75 @@ +.\" +.\" "$Id: cancel.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" cancel man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH cancel 1 "CUPS" "20 March 2006" "Apple Inc." +.SH NAME +cancel - cancel jobs +.SH SYNOPSIS +.B cancel +[ -E ] [ -U +.I username +] [ -a ] [ -h +.I hostname[:port] +] [ -u +.I username +] [ +.I id +] [ +.I destination +] [ +.I destination-id +] +.SH DESCRIPTION +\fIcancel\fR cancels existing print jobs. The \fI-a\fR option will remove +all jobs from the specified destination. +.SH OPTIONS +The following options are recognized by \fIcancel\fR: +.TP 5 +-E +.br +Forces encryption when connecting to the server. +.TP 5 +-U username +.br +Specifies the username to use when connecting to the server. +.TP 5 +-a +.br +Cancel all jobs on the named destination, or all jobs on all +destinations if none is provided. +.TP 5 +-h hostname[:port] +.br +Chooses an alternate server. +.TP 5 +-u username +.br +Cancels jobs owned by \fIusername\fR. +.SH COMPATIBILITY +Unlike the System V printing system, CUPS allows printer names to +contain any printable character except SPACE, TAB, "/", or "#". +Also, printer and class names are \fInot\fR case-sensitive. +.SH NOTES +Administrators wishing to prevent unauthorized cancellation of +jobs via the \fI-u\fR option should require authentication for +Cancel-Jobs operations in \fIcupsd.conf(5)\fR. +.SH SEE ALSO +\fIlp(1)\fR, \fIlpmove(8)\fR, \fIlpstat(1)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: cancel.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/classes.conf.man b/man/classes.conf.man new file mode 100644 index 0000000000..0664442d4d --- /dev/null +++ b/man/classes.conf.man @@ -0,0 +1,110 @@ +.\" +.\" "$Id: classes.conf.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" classes.conf man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH classes.conf 5 "CUPS" "29 April 2009" "Apple Inc." +.SH NAME +classes.conf \- class configuration file for cups +.SH DESCRIPTION +The \fIclasses.conf\fR file defines the local printer classes that are +available. It is normally located in the \fI/etc/cups\fR directory and +is generated automatically by the \fIcupsd(8)\fR program when printer +classes are added or deleted. +.LP +Each line in the file can be a configuration directive, a blank line, +or a comment. Comment lines start with the # character. +.SH DIRECTIVES +.TP 5 + ... +.br +Defines a specific printer class. +.TP 5 + ... +.br +Defines a default printer class. +.TP 5 +Accepting Yes +.TP 5 +Accepting No +.br +Specifies whether the printer is accepting new jobs. +.TP 5 +AllowUser [ user @group ... ] +.br +Allows specific users and groups to print to the printer. +.TP 5 +DenyUser [ user @group ... ] +.br +Prevents specific users and groups from printing to the printer. +.TP 5 +Info text +.br +Specifies human-readable text describing the printer. +.TP 5 +JobSheets banner banner +.br +Specifies the banner pages to use for the printer. +.TP 5 +KLimit number +.br +Specifies the job-k-limit value for the printer. +.TP 5 +Location text +.br +Specifies human-readable text describing the location of the printer. +.TP 5 +OpPolicy name +.br +Specifies the operation policy for the printer. +.TP 5 +PageLimit number +.br +Specifies the job-page-limit value for the printer. +.TP 5 +Printer +.br +Specifies a printer that is a member of the printer class. +.TP 5 +QuotaPeriod seconds +.br +Specifies the job-quota-period value for the printer. +.TP 5 +Shared Yes +.TP 5 +Shared No +.br +Specifies whether the printer is shared. +.TP 5 +State idle +.TP 5 +State stopped +.br +Specifies the initial state of the printer (Idle or Stopped) +.TP 5 +StateMessage text +.br +Specifies the message associated with the state. +.TP 5 +StateTime seconds +.br +Specifies the date/time associated with the state. +.SH SEE ALSO +\fIcupsd(8)\fR, \fIcupsd.conf(5)\fR, \fImime.convs(5)\fR, +\fImime.types(5)\fR, \fIprinters.conf(5)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: classes.conf.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/client.conf.man.in b/man/client.conf.man.in new file mode 100644 index 0000000000..47195e9f1b --- /dev/null +++ b/man/client.conf.man.in @@ -0,0 +1,55 @@ +.\" +.\" "$Id$" +.\" +.\" client.conf man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH client.conf 5 "CUPS" "2 September 2011" "Apple Inc." +.SH NAME +client.conf \- client configuration file for cups +.SH DESCRIPTION +The \fIclient.conf\fR file configures the CUPS client and is +normally located in the \fI@CUPS_SERVERROOT@\fR or \fI~/.cups\fR +directory. Each line in the file can be a configuration +directive, a blank line, or a comment. Comment lines start with +the # character. +.SH DIRECTIVES +The following directives are understood by the client. Consult the +on-line help for detailed descriptions: +.TP 5 +Encryption IfRequested +.TP 5 +Encryption Never +.TP 5 +Encryption Required +.br +Specifies the level of encryption that is required for a particular +location. +.TP 5 +GSSServiceName name +Specifies the Kerberos service name that is used for authentication, typically +"host", "http", or "ipp". CUPS adds the remote hostname +("name@server.example.com") for you. The default name is +"@CUPS_DEFAULT_GSSSERVICENAME@". +.TP 5 +ServerName hostname-or-ip-address[:port] +.TP 5 +ServerName /domain/socket +.br +Specifies the address and optionally the port to use when connecting to the +server. \fBNote: Not supported on Mac OS X 10.7 or later.\fR +.SH SEE ALSO +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id$". +.\" diff --git a/man/cups-config.man b/man/cups-config.man new file mode 100644 index 0000000000..577dd44ffb --- /dev/null +++ b/man/cups-config.man @@ -0,0 +1,117 @@ +.\" +.\" "$Id: cups-config.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" cups-config man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH cups-config 1 "CUPS" "23 October 2008" "Apple Inc." +.SH NAME +cups-config \- get cups api, compiler, directory, and link information. +.SH SYNOPSIS +.B cups-config +--api-version +.br +.B cups-config +--build +.br +.B cups-config +--cflags +.br +.B cups-config +--datadir +.br +.B cups-config +--help +.br +.B cups-config +--ldflags +.br +.B cups-config +[ +.I --image +] [ +.I --static +] --libs +.br +.B cups-config +--serverbin +.br +.B cups-config +--serverroot +.br +.B cups-config +--version +.br +.SH DESCRIPTION +\fBcups-config\fR is the CUPS program configuration utility. It should be +used by application developers to determine the necessary command-line +options for the compiler and linker, as well as determining installation +directories for filters, configuration files, and drivers. +.SH OPTIONS +.TP 5 +--api-version +.br +Displays the current API version (major.minor). +.TP 5 +--build +.br +Displays a system-specific build number. +.TP 5 +--cflags +.br +Displays the necessary compiler options. +.TP 5 +--datadir +.br +Displays the default CUPS data directory. +.TP 5 +--help +.br +Displays the program usage message. +.TP 5 +--image +.br +When used with \fI--libs\fR, adds the CUPS imaging library to the +list of displayed libraries. +.TP 5 +--ldflags +.br +Displays the necessary linker options. +.TP 5 +--libs +.br +Displays the necessary librarys to link to. +.TP 5 +--serverbin +.br +Displays the default CUPS binary directory, +where filters and backends are stored. +.TP 5 +--serverroot +.br +Displays the default CUPS configuration file directory. +.TP 5 +--static +.br +When used with \fI--libs\fR, shows the static libraries instead +of the default (shared) libraries. +.TP 5 +--version +.br +Displays the full version number of the CUPS installation +(major.minor.patch). +.SH SEE ALSO +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: cups-config.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/cups-deviced.man.in b/man/cups-deviced.man.in new file mode 100644 index 0000000000..7e410bc4c9 --- /dev/null +++ b/man/cups-deviced.man.in @@ -0,0 +1,44 @@ +.\" +.\" "$Id$" +.\" +.\" cups-deviced man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH cups-deviced 8 "CUPS" "16 June 2008" "Apple Inc." +.SH NAME +cups-deviced \- cups device daemon +.SH SYNOPSIS +.B cups-deviced +.I request-id limit user-id options +.SH DESCRIPTION +\fIcups-deviced\fR polls the backends in +\fI@CUPS_SERVERBIN@/backend\fR for a list of available devices. +It is run by \fIcupsd(8)\fR in response to a +\fICUPS-Get-Devices\fR request. The output format is an IPP +response message. The \fIrequest-id\fR argument is the request ID +from the original IPP request, typically 1. The \fIlimit\fR +argument is the limit value from the original IPP request - 0 +means no limit. The \fIuser-id\fR argument is the +requesting-user-name value from the original IPP request. +Finally, the \fIoptions\fR argument is a space-delimited list of +attributes ("name=value name=value \...") that were passed in +with the request. Currently \fIcups-deviced\fR looks for the +\fIrequested-attributes\fR attribute and tailors the output +accordingly. +.SH SEE ALSO +backend(7), cupsd(8), cupsd.conf(5), +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id$". +.\" diff --git a/man/cups-driverd.man.in b/man/cups-driverd.man.in new file mode 100644 index 0000000000..ddef4ae4b3 --- /dev/null +++ b/man/cups-driverd.man.in @@ -0,0 +1,122 @@ +.\" +.\" "$Id$" +.\" +.\" cups-driverd man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH cups-driverd 8 "CUPS" "6 January 2011" "Apple Inc." +.SH NAME +cups-driverd \- cups driver daemon +.SH SYNOPSIS +.B cups-driverd +cat +.I ppd-name +.br +.B cups-driverd +list +.I request_id limit options +.SH DESCRIPTION +\fIcups-driverd\fR shows or lists PPD files. It is run in +response to CUPS-Add-Modify-Printer or CUPS-Get-Devices requests. +The first form ("cups-driverd cat ppd-name") writes the named PPD +file to stdout. The output format is an uncompressed PPD file. +.LP +The second form lists the available manufacturers or PPD files to +stdout as indicated by the \fIoptions\fR argument. The output +format is an IPP response message. The \fIrequest_id\fR argument +is the request ID from the original IPP request, typically 1. The +\fIlimit\fR argument is the limit value from the original IPP +request - 0 means no limit. Finally, the \fIoptions\fR argument +is a space-delimited list of attributes ("name=value name=value +\&...") that were passed in with the request. Currently +\fIcups-driverd\fR looks for the \fIppd-make\fR and +\fIrequested-attributes\fR attributes and tailors the output +accordingly. +.SH DRIVERS +Drivers can be static PPD files under the +\fI@CUPS_DATADIR@/model\fR directory or programs under the +\fI@CUPS_SERVERBIN@/driver\fR directory. Static PPD files must +conform to the Adobe PPD File Format Specification version 4.3 +and may be compressed using the \fIgzip(1)\fR program. Driver +programs must implement the command-line interface shown in the +next section. +.SS DRIVER PROGRAMS +Driver programs provide a interface to dynamically-generated PPD +files. The following arguments are currently defined: +.TP 5 +drivername list +.br +Lists the supported PPD files to stdout. +.TP 5 +drivername cat ppdname +.br +Writes the named PPD file to stdout. +.PP +Driver programs MUST NOT query hardware or make other long-term operations that +would delay the return of a driver list. See the NOTES section below for +specific recommendations. +.SS LISTING FILES (drivername list) +When run with the single argument "list", the program must list +the available PPD files it can generate to stdout using the +following format: +.nf + "drivername:ppdname" language "make" "make and model" + "drivername:ppdname" language "make" "make and model" "1284 device id" + "drivername:ppdname" language "make" "make and model" "1284 device id" "(PPD product)" + "drivername:ppdname" language "make" "make and model" "1284 device id" "(PPD product)" "PostScript version" + "drivername:ppdname" language "make" "make and model" "1284 device id" "(PPD product)" "PostScript version" "type" +.fi +.LP +\fIDrivername\fR is the name of the driver program. \fIPpdname\fR +is the name used to select the given driver. \fILanguage\fR is +the locale associated with the default language of the PPD file, +typically "en". \fIMake\fR is the Manufacturer name from the PPD +file. \fIMake and model\fR is the NickName name from the PPD +file. \fI1284 device id\fR is the 1284DeviceId from the PPD file, +if any. \fI(PPD product)\fR is the Product string as it would appear in the PPD +file or from a PostScript query. \fIPostScript version\fR is the PSVersion +string as it would appear in the PPD file or from a PostScript query. \fIType\fR +is "postscript" for PostScript printers, "pdf" for PDF printers, "raster" for +raster printers, or "fax" for facsimile devices. +.SS WRITING FILES (drivername cat ppdname) +When the driver program is run with the "cat ppdname" arguments, +it must write the named PPD file to stdout, uncompressed. If the +named PPD file does not exist, the driver program must not write +any output to stdout and report the error to stderr instead. +.SS DRIVER ERROR MESSAGES +Error messages can be relayed back to \fIcupsd\fR by writing them +to stderr. The following prefixes are recognized: +.TP 5 +DEBUG: [drivername] +.br +Debugging messages +.TP 5 +ERROR: [drivername] +.br +Error messages +.TP 5 +INFO: [drivername] +.br +Informational messages +.SH NOTES +Due to performance considerations, driver programs have been officially +deprecated and should not be used for new development. Currently only the +CUPS web interface and \fIlpinfo(8)\fR command will request lists from all +driver programs. +.SH SEE ALSO +cupsd(8), cupsd.conf(5), cupstestppd(1), lpinfo(8), +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id$". +.\" diff --git a/man/cups-lpd.man.in b/man/cups-lpd.man.in new file mode 100644 index 0000000000..e245712512 --- /dev/null +++ b/man/cups-lpd.man.in @@ -0,0 +1,124 @@ +.\" +.\" "$Id$" +.\" +.\" cups-lpd man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH cups-lpd 8 "CUPS" "4 August 2008" "Apple Inc." +.SH NAME +cups-lpd \- receive print jobs and report printer status to lpd clients +.SH SYNOPSIS +.B cups-lpd +[ -h +.I hostname[:port] +] [ -n ] [ -o +.I option=value +] +.SH DESCRIPTION +\fIcups-lpd\fR is the CUPS Line Printer Daemon ("LPD") +mini-server that supports legacy client systems that use the LPD +protocol. \fIcups-lpd\fR does not act as a standalone network +daemon but instead operates using the Internet "super-server" +\fIinetd(8)\fR or \fIxinetd(8)\fR. If you are using \fIinetd\fR, +add the following line to the \fIinetd.conf\fR file to enable the +\fIcups-lpd\fR mini-server: +.br +.nf + + printer stream tcp nowait lp @CUPS_SERVERBIN@/daemon/cups-lpd cups-lpd \\ + -o document-format=application/octet-stream +.fi +.LP +.LP +\fBNote:\fR If you are using Solaris 10 or higher, you must run +the \fIinetdconv(1m)\fR program to register the changes to the +inetd.conf file. +.LP +If you are using the newer \fIxinetd(8)\fR daemon, create a file +named \fI/etc/xinetd.d/cups\fR containing the following lines: +.br +.nf + + service printer + { + socket_type = stream + protocol = tcp + wait = no + user = lp + group = sys + passenv = + server = @CUPS_SERVERBIN@/daemon/cups-lpd + server_args = -o document-format=application/octet-stream + } +.fi +.SH OPTIONS +.TP 5 +-h hostname[:port] +.br +Sets the CUPS server (and port) to use. +.TP 5 +-n +.br +Disables reverse address lookups; normally \fIcups-lpd\fR will +try to discover the hostname of the client via a reverse DNS +lookup. +.TP 5 +-o name=value +.br +Inserts options for all print queues. Most often this is used to +disable the "l" filter so that remote print jobs are filtered as +needed for printing; the examples in the previous section set the +"document-format" option to "application/octet-stream" which +forces autodetection of the print file format. +.SH PERFORMANCE +\fIcups-lpd\fR performs well with small numbers of clients and +printers. However, since a new process is created for each +connection and since each process must query the printing system +before each job submission, it does not scale to larger +configurations. We highly recommend that large configurations +use the native IPP support provided by CUPS instead. +.SH SECURITY +\fIcups-lpd\fR currently does not perform any access control +based on the settings in \fIcupsd.conf(5)\fR or in the +\fIhosts.allow(5)\fR or \fIhosts.deny(5)\fR files used by TCP +wrappers. Therefore, running \fIcups-lpd\fR on your server will +allow any computer on your network (and perhaps the entire +Internet) to print to your server. +.LP +While \fIxinetd\fR has built-in access control support, you +should use the TCP wrappers package with \fIinetd\fR to limit +access to only those computers that should be able to print +through your server. +.LP +\fIcups-lpd\fR is not enabled by the standard CUPS distribution. +Please consult with your operating system vendor to determine +whether it is enabled on your system. +.SH COMPATIBILITY +\fIcups-lpd\fR does not enforce the restricted source port +number specified in RFC 1179, as using restricted ports does not +prevent users from submitting print jobs. While this behavior is +different than standard Berkeley LPD implementations, it should +not affect normal client operations. +.LP +The output of the status requests follows RFC 2569, Mapping +between LPD and IPP Protocols. Since many LPD implementations +stray from this definition, remote status reporting to LPD +clients may be unreliable. +.SH SEE ALSO +\fIcups(1)\fR, \fIcupsd(8)\fR, \fIinetconv(1m)\fR, +\fIinetd(8)\fR, \fIxinetd(8)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id$". +.\" diff --git a/man/cups-polld.man b/man/cups-polld.man new file mode 100644 index 0000000000..aee7068731 --- /dev/null +++ b/man/cups-polld.man @@ -0,0 +1,38 @@ +.\" +.\" "$Id: cups-polld.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" cups-polld man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH cups-polld 8 "CUPS" "12 February 2006" "Apple Inc." +.SH NAME +cups-polld \- cups printer polling daemon +.SH SYNOPSIS +.B cups-polld +.I address ipp-port interval browse-port +.SH DESCRIPTION +\fIcups-polld\fR polls remote servers for a list of available +printers and printer classes every \fIinterval\fR seconds. +Printer and class information is then broadcast to the localhost +interface (127.0.0.1) on the specified browse port for reception +by \fIcupsd(8)\fR. +.PP +This program is started automatically by \fIcupsd\fR for every +\fIBrowsePoll\fR directive found in the \fIcupsd.conf(5)\fR file. +.SH SEE ALSO +\fIcupsd(8)\fR, \fIcupsd.conf(5)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: cups-polld.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/cups-snmp.conf.man b/man/cups-snmp.conf.man new file mode 100644 index 0000000000..1c778bc846 --- /dev/null +++ b/man/cups-snmp.conf.man @@ -0,0 +1,73 @@ +.\" +.\" "$Id: cups-snmp.conf.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" snmp.conf man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH snmp.conf 5 "CUPS" "31 July 2006" "Apple Inc." +.SH NAME +snmp.conf \- snmp configuration file for cups +.SH DESCRIPTION +The \fIsnmp.conf\fR file configures the CUPS SNMP printer +discovery backend and is normally located in the \fI/etc/cups\fR +directory. Each line in the file can be a configuration +directive, a blank line, or a comment. Comment lines start with +the # character. +.LP +The SNMP backend uses the SNMPv1 protocol to discover network +printers, collecting information from the Host MIB along with +intelligent port probes to determine the correct device URI and +make and model for each printer. Future versions of CUPS will +likely support the new Port Monitor MIB as well. +.SH DIRECTIVES +The following directives are understood by the SNMP backend. +Consult the on-line help for detailed descriptions: +.TP 5 +Address @IF(\fIname\fR) +.TP 5 +Address @LOCAL +.TP 5 +Address \fIaddress\fR +.br +Sends SNMP broadcast queries to the specified address(es). The +default address is "@LOCAL" which broadcasts to all LAN +interfaces. +.TP 5 +Community \fIname\fR +.br +Specifies a SNMP community to query. The default community is +"public". +.TP 5 +DebugLevel \fIN\fR +.br +Sets the debug logging level to \fIN\fR; 0 disables debug +logging, 1 enables basic logging, 2 displays SNMP values, and 3 +displays raw hex data. +.TP 5 +HostNameLookups on +.TP 5 +HostNameLookups off +.br +Specifies whether the addresses of printers should be converted +to hostnames or left as numeric IP addresses. The default is +"off". +.TP 5 +MaxRunTime \fIseconds\fR +.br +Specifies the maximum number of seconds that the SNMP backend +will scan the network for printers. +.SH SEE ALSO +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: cups-snmp.conf.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/cupsaccept.man b/man/cupsaccept.man new file mode 100644 index 0000000000..7c53d862eb --- /dev/null +++ b/man/cupsaccept.man @@ -0,0 +1,79 @@ +.\" +.\" "$Id: cupsaccept.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" accept/reject man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH cupsaccept 8 "CUPS" "12 February 2006" "Apple Inc." +.SH NAME +cupsaccept/cupsreject \- accept/reject jobs sent to a destination +.SH SYNOPSIS +.B cupsaccept +[ -E ] [ -U +.I username +] [ -h +.I hostname[:port] +] destination(s) +.br +.B cupsreject +[ -E ] [ -U +.I username +] [ -h +.I hostname[:port] +] [ -r +.I reason +] destination(s) +.SH DESCRIPTION +\fIcupsaccept\fR instructs the printing system to accept print jobs to the +specified destinations. +.LP +\fIcupsreject\fR instructs the printing system to reject print jobs to the +specified destinations. The \fI-r\fR option sets the reason for rejecting +print jobs. If not specified the reason defaults to "Reason Unknown". +.SH OPTIONS +The following options are supported by both \fIcupsaccept\fR and +\fIcupsreject\fR: +.TP 5 +-E +.br +Forces encryption when connecting to the server. +.TP 5 +-U username +.br +Sets the username that is sent when connecting to the server. +.TP 5 +-h hostname[:port] +.br +Chooses an alternate server. +.TP 5 +-r "reason" +.br +Sets the reason string that is shown for a printer that is +rejecting jobs. +.SH COMPATIBILITY +Unlike the System V printing system, CUPS allows printer names to +contain any printable character except SPACE, TAB, "/", or "#". +Also, printer and class names are \fInot\fR case-sensitive. +.LP +The CUPS versions of \fIaccept\fR and \fIreject\fR may ask the +user for an access password depending on the printing system +configuration. This differs from the System V versions which +require the root user to execute these commands. +.SH SEE ALSO +\fIcancel(1)\fR, \fIcupsenable(8)\fR, \fIlp(1)\fR, +\fIlpadmin(8)\fR, \fIlpstat(1)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: cupsaccept.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/cupsaddsmb.man.in b/man/cupsaddsmb.man.in new file mode 100644 index 0000000000..dc02cedb8b --- /dev/null +++ b/man/cupsaddsmb.man.in @@ -0,0 +1,214 @@ +.\" +.\" "$Id$" +.\" +.\" cupsaddsmb man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH cupsaddsmb 8 "CUPS" "25 July 2007" "Apple Inc." +.SH NAME +cupsaddsmb \- export printers to samba for windows clients + +.SH SYNOPSIS +.B cupsaddsmb +[ -H +.I samba-server +] [ -U +.I samba-user[%samba-password] +] [ -h +.I cups-server[:port] +] [ -v ] -a +.br +.B cupsaddsmb +[ -H +.I samba-server +] [ -U +.I samba-user[%samba-password] +] [ -h +.I cups-server[:port] +] [ -v ] printer [ ... printer ] + +.SH DESCRIPTION +\fIcupsaddsmb\fR exports printers to the SAMBA software (version +2.2.0 or higher) for use with Windows clients. Depending on the +SAMBA configuration, you may need to provide a password to +export the printers. This program requires the Windows printer +driver files described below. + +.SH OPTIONS +\fIcupsaddsmb\fR supports the following options: +.TP 5 +-H samba-server +.br +Specifies the SAMBA server which defaults to the CUPS server. +.TP 5 +-U samba-user[%samba-password] +.br +Specifies the SAMBA print admin username which defaults to your +current username. If the username contains a percent (%) +character, then the text following the percent is treated as the +SAMBA password to use. +.TP 5 +-a +.br +Exports all known printers. Otherwise only the named printers are +exported. +.TP 5 +-h cups-server[:port] +.br +Specifies a different CUPS server to use. +.TP 5 +-v +.br +Specifies that verbose information should be shown. This is +useful for debugging SAMBA configuration problems. + +.SH SAMBA CONFIGURATION +\fIcupsaddsmb\fR uses the new RPC-based printing support in +SAMBA 2.2.x to provide printer drivers and PPD files to Windows +client machines. In order to use this functionality, you must +first configure the SAMBA \fIsmb.conf(5)\fR file to support +printing through CUPS and provide a printer driver download +share, as follows: +.nf + + [global] + load printers = yes + printing = cups + printcap name = cups + + [printers] + comment = All Printers + path = /var/spool/samba + browseable = no + public = yes + guest ok = yes + writable = no + printable = yes + + [print$] + comment = Printer Drivers + path = /etc/samba/drivers + browseable = yes + guest ok = no + read only = yes + write list = root +.fi +.LP +This configuration assumes a FHS-compliant installation of +SAMBA; adjust the [printers] and [print$] share paths +accordingly on your system as needed. + +.SH MICROSOFT POSTSCRIPT DRIVERS FOR WINDOWS +The base driver for Windows 2000 and higher is the Microsoft +PostScript driver, which is available on any system running +Windows 2000 or higher in the +%WINDIR%\\SYSTEM32\\SPOOL\\DRIVERS\\W32X86\\3 folder for 32-bit +drivers and +%WINDIR%\\SYSTEM32\\SPOOL\\DRIVERS\\X64\\3 folder for 64-bit +drivers. +.LP +The CUPS printer driver is preferred over the Microsoft driver +since it supports the page-label, job-billing, and +job-hold-until options fully on all printers. However, currently +only Windows 2000 and higher is supported by the Microsoft +driver, so you will also need to get the Adobe driver to support +Windows 95, 98, and Me clients. The Adobe and Microsoft drivers +for Windows 2000 are identical. +.LP +Once you have extracted the driver files, copy the 32-bit drivers +to the \fI@CUPS_DATADIR@/drivers\fR directory and the 64-bit +drivers to the \fI@CUPS_DATADIR@/drivers/x64\fR directory exactly +as named below: +.nf + + [Windows 2000 and higher] + ps5ui.dll + pscript.hlp + pscript.ntf + pscript5.dll +.fi +.LP +\fBNote:\fR Unlike Windows, case is significant - make sure that +you use the lowercase filenames shown above, otherwise +\fIcupsaddsmb\fR will fail to export the drivers. + +.SH CUPS POSTSCRIPT DRIVERS FOR WINDOWS +\fIcupsaddsmb\fR can use the CUPS v6 PostScript printer driver +for Windows, which is available for download from the CUPS web +site. +.LP +The CUPS printer driver is preferred over the Adobe and +Microsoft drivers since it supports the page-label, job-billing, +and job-hold-until options fully on all printers. However, +currently only Windows 2000 and higher is supported by the CUPS +driver, so you will also need to get the Adobe driver to support +Windows 95, 98, and Me clients. +.LP +Once you have extracted the driver files, copy the 32-bit drivers +to the \fI@CUPS_DATADIR@/drivers\fR directory and the 64-bit +drivers to the \fI@CUPS_DATADIR@/drivers/x64\fR directory exactly +as named below: +.nf + + [Windows 2000 and higher] + cups6.inf (from www.cups.org) + cups6.ini (from www.cups.org) + cupsps6.dll (from www.cups.org) + cupsui6.dll (from www.cups.org) + ps5ui.dll (from your Windows system) + pscript.hlp (from your Windows system) + pscript.ntf (from your Windows system) + pscript5.dll (from your Windows system) +.fi +.LP +\fBNote:\fR Unlike Windows, case is significant - make sure that +you use the lowercase filenames shown above, otherwise +\fIcupsaddsmb\fR will fail to export the drivers. + +.SH ADOBE POSTSCRIPT DRIVERS FOR WINDOWS 95, 98, AND ME +\fIcupsaddsmb\fR can use the Adobe PostScript printer driver for +Windows 95, 98, and ME, which are available for download from the +Adobe web site (http://www.adobe.com). +.LP +The Adobe driver does not support the page-label, job-billing, or +job-hold-until options. +.LP +Once you have installed the driver on a Windows system, copy the +following files to the \fI@CUPS_DATADIR@/drivers\fR directory +exactly as named below: +.nf + + [Windows 95, 98, and Me] + ADFONTS.MFM + ADOBEPS4.DRV + ADOBEPS4.HLP + ICONLIB.DLL + PSMON.DLL +.fi +.LP +\fBNote:\fR Unlike Windows, case is significant - make sure that +you use the UPPERCASE filenames shown above, otherwise +\fIcupsaddsmb\fR will fail to export the drivers. + +.SH KNOWN ISSUES +Getting the full set of Windows driver files should be easier. + +.SH SEE ALSO +\fIsmbd(8)\fR, \fIsmb.conf(5)\fR, +http://localhost:631/help +.br +http://www.cups.org/windows/ + +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id$". +.\" diff --git a/man/cupsctl.man b/man/cupsctl.man new file mode 100644 index 0000000000..411ada83f0 --- /dev/null +++ b/man/cupsctl.man @@ -0,0 +1,107 @@ +.\" +.\" "$Id: cupsctl.man 9457 2011-01-11 03:04:04Z mike $" +.\" +.\" cupsctl man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 2007 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH cupsctl 8 "CUPS" "10 January 2011" "Apple Inc." +.SH NAME +cupsctl \- configure cupsd.conf options +.SH SYNOPSIS +.B cupsctl +[ -E ] [-U +.I username +] [ -h +.I server[:port] +] [ --[no-]debug-logging ] [ --[no-]remote-admin ] [ --[no-]remote-any ] +[ --[no-]remote-printers ] [ --[no-]share-printers ] [ --[no-]user-cancel-any ] +[ +.I name=value +] +.SH DESCRIPTION +\fIcupsctl\fR updates or queries the \fBcupsd.conf\fR file for a server. When +no changes are requested, the current configuration values are written to the +standard output in the format "name=value", one per line. +.SH OPTIONS +The following options are recognized: +.TP 5 +-E +.br +Enables encryption on the connection to the scheduler. +.TP 5 +-U \fIusername\fR +.br +Specifies an alternate username to use when authenticating with the scheduler. +.TP 5 +-h \fIserver[:port]\fR +.br +Specifies the server address. +.TP 5 +--[no-]debug-logging +.br +Enables or disables debug logging in the \fBerror_log\fR file. +.TP 5 +--[no-]remote-admin +.br +Enables or disables remote administration. +.TP 5 +--[no-]remote-any +.br +Enables or disables printing from any address, e.g. the Internet. +.TP 5 +--[no-]remote-printers +.br +Enables or disables the display of remote printers shared via the CUPS, LDAP, +or SLP protocols. +.TP 5 +--[no-]share-printers +.br +Enables or disables sharing of local printers with other computers. +.TP 5 +--[no-]user-cancel-any +.br +Allows or prevents users from canceling jobs owned by others. +.SH EXAMPLES +Display the current settings: +.nf + cupsctl +.fi +.LP +Enable debug logging: +.nf + cupsctl --debug-logging +.fi +.LP +Get the current debug logging state: +.nf + cupsctl | grep '^_debug_logging' | awk -F= '{print $2}' +.fi +.LP +Disable printer sharing: +.nf + cupsctl --no-shared-printers +.fi +.LP +Enable printing using the file: pseudo-device: +.nf + cupsctl FileDevice=Yes +.fi +.SH KNOWN ISSUES +You cannot set the Listen or Port directives using \fIcupsctl\fR. +.SH SEE ALSO +\fIcupsd.conf(5)\fR, \fIcupsd(8)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: cupsctl.man 9457 2011-01-11 03:04:04Z mike $". +.\" diff --git a/man/cupsd.conf.man.in b/man/cupsd.conf.man.in new file mode 100644 index 0000000000..ba402b3556 --- /dev/null +++ b/man/cupsd.conf.man.in @@ -0,0 +1,640 @@ +.\" +.\" "$Id$" +.\" +.\" cupsd.conf man page for CUPS. +.\" +.\" Copyright 2007-2012 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH cupsd.conf 5 "CUPS" "6 January 2012" "Apple Inc." +.SH NAME +cupsd.conf \- server configuration file for cups +.SH DESCRIPTION +The \fIcupsd.conf\fR file configures the CUPS scheduler, \fIcupsd(8)\fR. It +is normally located in the \fI@CUPS_SERVERROOT@\fR directory. +.LP +Each line in the file can be a configuration directive, a blank line, +or a comment. Comment lines start with the # character. The +configuration directives are intentionally similar to those used by the +popular Apache web server software and are described below. +.SH DIRECTIVES +The following directives are understood by \fIcupsd(8)\fR. Consult the +on-line help for detailed descriptions: +.TP 5 +AccessLog filename +.TP 5 +AccessLog syslog +.br +Defines the access log filename. +.TP 5 +AccessLogLevel config +.TP 5 +AccessLogLevel actions +.TP 5 +AccessLogLevel all +.br +Specifies the logging level for the AccessLog file. +.TP 5 +Allow all +.TP 5 +Allow none +.TP 5 +Allow host.domain.com +.TP 5 +Allow *.domain.com +.TP 5 +Allow ip-address +.TP 5 +Allow ip-address/netmask +.TP 5 +Allow ip-address/mm +.TP 5 +Allow @IF(name) +.TP 5 +Allow @LOCAL +.br +Allows access from the named hosts or addresses. +.TP 5 +AuthClass User +.TP 5 +AuthClass Group +.TP 5 +AuthClass System +.br +Specifies the authentication class (User, Group, System) - +\fBthis directive is deprecated\fR. +.TP 5 +AuthGroupName group-name +.br +Specifies the authentication group - \fBthis directive is +deprecated\fR. +.TP 5 +AuthType None +.TP 5 +AuthType Basic +.TP 5 +AuthType BasicDigest +.TP 5 +AuthType Digest +.TP 5 +AuthType Negotiate +.br +Specifies the authentication type (None, Basic, BasicDigest, Digest, Negotiate) +.TP 5 +AutoPurgeJobs Yes +.TP 5 +AutoPurgeJobs No +.br +Specifies whether to purge job history data automatically when +it is no longer required for quotas. +.TP 5 +BrowseLocalProtocols [All] [DNSSD] +.br +Specifies the protocols to use for local printer sharing. +.TP 5 +BrowseWebIF Yes +.TP 5 +BrowseWebIF No +.br +Specifies whether the CUPS web interface is advertised via DNS-SD. +.TP 5 +Browsing Yes +.TP 5 +Browsing No +.br +Specifies whether or not remote printer browsing should be enabled. +.TP 5 +Classification banner +.br +Specifies the security classification of the server. +.TP 5 +ClassifyOverride Yes +.TP 5 +ClassifyOverride No +.br +Specifies whether to allow users to override the classification +of individual print jobs. +.TP 5 +ConfigFilePerm mode +.br +Specifies the permissions for all configuration files that the scheduler +writes. +.TP 5 +DataDir path +.br +Specified the directory where data files can be found. +.TP 5 +DefaultAuthType Basic +.TP 5 +DefaultAuthType BasicDigest +.TP 5 +DefaultAuthType Digest +.TP 5 +DefaultAuthType Negotiate +.br +Specifies the default type of authentication to use. +.TP 5 +DefaultEncryption Never +.TP 5 +DefaultEncryption IfRequested +.TP 5 +DefaultEncryption Required +.br +Specifies the type of encryption to use for authenticated requests. +.TP 5 +DefaultLanguage locale +.br +Specifies the default language to use for text and web content. +.TP 5 +DefaultPaperSize Auto +.TP 5 +DefaultPaperSize None +.TP 5 +DefaultPaperSize sizename +.br +Specifies the default paper size for new print queues. "Auto" uses a locale- +specific default, while "None" specifies there is no default paper size. +.TP 5 +DefaultPolicy policy-name +.br +Specifies the default access policy to use. +.TP 5 +DefaultShared Yes +.TP 5 +DefaultShared No +.br +Specifies whether local printers are shared by default. +.TP 5 +Deny all +.TP 5 +Deny none +.TP 5 +Deny host.domain.com +.TP 5 +Deny *.domain.com +.TP 5 +Deny ip-address +.TP 5 +Deny ip-address/netmask +.TP 5 +Deny ip-address/mm +.TP 5 +Deny @IF(name) +.TP 5 +Deny @LOCAL +.br +Denies access to the named host or address. +.TP 5 +DirtyCleanInterval seconds +.br +Specifies the delay for updating of configuration and state files. A value of 0 +causes the update to happen as soon as possible, typically within a few +milliseconds. +.TP 5 +DocumentRoot directory +.br +Specifies the root directory for the internal web server documents. +.TP 5 +Encryption IfRequested +.TP 5 +Encryption Never +.TP 5 +Encryption Required +.br +Specifies the level of encryption that is required for a particular +location. +.TP 5 +ErrorLog filename +.TP 5 +ErrorLog syslog +.br +Specifies the error log filename. +.TP 5 +FatalErrors none +.TP 5 +FatalErrors all -kind [... -kind] +.TP 5 +FatalErrors kind [... kind] +.br +Specifies which errors are fatal, causing the scheduler to exit. "Kind" is +"browse", "config", "listen", "log", or "permissions". +.TP 5 +FileDevice Yes +.TP 5 +FileDevice No +.br +Specifies whether the file pseudo-device can be used for new +printer queues. +.TP 5 +FilterLimit limit +.br +Specifies the maximum cost of filters that are run concurrently. +.TP 5 +FilterNice nice-value +.br +Specifies the scheduling priority ("nice" value) of filters that +are run to print a job. +.TP 5 +FontPath directory[:directory:...] +.br +Specifies the search path for fonts. +.TP 5 +Group group-name-or-number +.br +Specifies the group name or ID that will be used when executing +external programs. +.TP 5 +GSSServiceName name +.br +Specifies the service name when using Kerberos authentication. The default +service name is "@CUPS_DEFAULT_GSSSERVICENAME@". +.TP 5 +HostNameLookups On +.TP 5 +HostNameLookups Off +.TP 5 +HostNameLookups Double +.br +Specifies whether or not to do reverse lookups on client addresses. +.TP 5 +Include filename +.br +Includes the named file. +.TP 5 +JobKillDelay seconds +.br +Specifies the number of seconds to wait before killing the filters and backend +associated with a canceled or held job. +.TP 5 +JobPrivateAccess all +.TP 5 +JobPrivateAccess default +.TP 5 +JobPrivateAccess {user|@group|@ACL|@OWNER|@SYSTEM}+ +.br +Specifies an access list for a job's private values. The "default" access list +is "@OWNER @SYSTEM". "@ACL" maps to the printer's requesting-user-name-allowed +or requesting-user-name-denied values. +.TP 5 +JobPrivateValues all +.TP 5 +JobPrivateValues default +.TP 5 +JobPrivateValues none +.TP 5 +JobPrivateValues attribute-name-1 [ ... attribute-name-N ] +Specifies the list of job values to make private. The "default" values are +"job-name", "job-originating-host-name", and "job-originating-user-name". +.TP 5 +JobRetryInterval seconds +.br +Specifies the interval between retries of jobs in seconds. +.TP 5 +JobRetryLimit count +.br +Specifies the number of retries that are done for jobs. +.TP 5 +KeepAlive Yes +.TP 5 +KeepAlive No +.br +Specifies whether to support HTTP keep-alive connections. +.TP 5 +KeepAliveTimeout seconds +.br +Specifies the amount of time that connections are kept alive. +.TP 5 + ... +.br +Specifies the IPP operations that are being limited inside a policy. +.TP 5 + ... +.TP 5 + ... +.br +Specifies the HTTP methods that are being limited inside a location. +.TP 5 +LimitRequestBody +.br +Specifies the maximum size of any print job request. +.TP 5 +Listen ip-address:port +.TP 5 +Listen *:port +.TP 5 +Listen /path/to/domain/socket +.br +Listens to the specified address and port or domain socket path. +.TP 5 + ... +.br +Specifies access control for the named location. +.TP 5 +LogDebugHistory #-messages +.br +Specifies the number of debugging messages that are logged when an error +occurs in a print job. +.TP 5 +LogFilePerm mode +.br +Specifies the permissions for all log files that the scheduler writes. +.TP 5 +LogLevel alert +.TP 5 +LogLevel crit +.TP 5 +LogLevel debug2 +.TP 5 +LogLevel debug +.TP 5 +LogLevel emerg +.TP 5 +LogLevel error +.TP 5 +LogLevel info +.TP 5 +LogLevel none +.TP 5 +LogLevel notice +.TP 5 +LogLevel warn +.br +Specifies the logging level for the ErrorLog file. +.TP 5 +LogTimeFormat standard +.TP 5 +LogTimeFormat usecs +.br +Specifies the format of the date and time in the log files. +.TP 5 +MaxClients number +.br +Specifies the maximum number of simultaneous clients to support. +.TP 5 +MaxClientsPerHost number +.br +Specifies the maximum number of simultaneous clients to support from a +single address. +.TP 5 +MaxCopies number +.br +Specifies the maximum number of copies that a user can print of each job. +.TP 5 +MaxJobs number +.br +Specifies the maximum number of simultaneous jobs to support. +.TP 5 +MaxJobsPerPrinter number +.br +Specifies the maximum number of simultaneous jobs per printer to support. +.TP 5 +MaxJobsPerUser number +.br +Specifies the maximum number of simultaneous jobs per user to support. +.TP 5 +MaxJobTime seconds +.br +Specifies the maximum time a job may take to print before it is canceled. The +default is 10800 seconds (3 hours). Set to 0 to disable cancellation of "stuck" +jobs. +.TP 5 +MaxLogSize number-bytes +.br +Specifies the maximum size of the log files before they are +rotated (0 to disable rotation) +.TP 5 +MaxRequestSize number-bytes +.br +Specifies the maximum request/file size in bytes (0 for no limit) +.TP 5 +MultipleOperationTimeout seconds +.br +Specifies the maximum amount of time to allow between files in a multiple file +print job. +.TP 5 +Order allow,deny +.TP 5 +Order deny,allow +.br +Specifies the order of HTTP access control (allow,deny or deny,allow) +.TP 5 +PageLog filename +.TP 5 +PageLog syslog +.br +Specifies the page log filename. +.TP 5 +PageLogFormat format string +.br +Specifies the format of page log lines. +.TP 5 +PassEnv variable [... variable] +.br +Passes the specified environment variable(s) to child processes. +.TP 5 + ... +.br +Specifies access control for the named policy. +.TP 5 +Port number +.br +Specifies a port number to listen to for HTTP requests. +.TP 5 +PreserveJobFiles Yes +.TP 5 +PreserveJobFiles No +.br +Specifies whether or not to preserve job files after they are printed. +.TP 5 +PreserveJobHistory Yes +.TP 5 +PreserveJobHistory No +.br +Specifies whether or not to preserve the job history after they are +printed. +.TP 5 +Printcap +.TP 5 +Printcap filename +.br +Specifies the filename for a printcap file that is updated +automatically with a list of available printers (needed for +legacy applications); specifying Printcap with no filename +disables printcap generation. +.TP 5 +PrintcapFormat bsd +.TP 5 +PrintcapFormat plist +.TP 5 +PrintcapFormat solaris +.br +Specifies the format of the printcap file. +.TP 5 +PrintcapGUI +.TP 5 +PrintcapGUI gui-program-filename +.br +Specifies whether to generate option panel definition files on +some operating systems. When provided with no program filename, +disables option panel definition files. +.TP 5 +ReloadTimeout seconds +.br +Specifies the amount of time to wait for job completion before +restarting the scheduler. +.TP 5 +RemoteRoot user-name +.br +Specifies the username that is associated with unauthenticated root +accesses. +.TP 5 +RequestRoot directory +.br +Specifies the directory to store print jobs and other HTTP request +data. +.TP 5 +Require group group-name-list +.TP 5 +Require user user-name-list +.TP 5 +Require valid-user +.br +Specifies that user or group authentication is required. +.TP 5 +RIPCache bytes +.br +Specifies the maximum amount of memory to use when converting images +and PostScript files to bitmaps for a printer. +.TP 5 +Satisfy all +.TP 5 +Satisfy any +.br +Specifies whether all or any limits set for a Location must be +satisfied to allow access. +.TP 5 +ServerAdmin user@domain.com +.br +Specifies the email address of the server administrator. +.TP 5 +ServerAlias hostname [... hostname] +.TP 5 +ServerAlias * +.br +Specifies an alternate name that the server is known by. The special name "*" +allows any name to be used. +.TP 5 +ServerBin directory +.br +Specifies the directory where backends, CGIs, daemons, and filters may +be found. +.TP 5 +ServerCertificate filename +.br +Specifies the encryption certificate to use. +.TP 5 +ServerKey filename +.br +Specifies the encryption key to use. +.TP 5 +ServerName hostname-or-ip-address +.br +Specifies the fully-qualified hostname of the server. +.TP 5 +ServerRoot directory +.br +Specifies the directory where the server configuration files can be found. +.TP 5 +ServerTokens Full +.TP 5 +ServerTokens Major +.TP 5 +ServerTokens Minimal +.TP 5 +ServerTokens Minor +.TP 5 +ServerTokens None +.TP 5 +ServerTokens OS +.TP 5 +ServerTokens ProductOnly +.br +Specifies what information is included in the Server header of HTTP +responses. +.TP 5 +SetEnv variable value +.br +Set the specified environment variable to be passed to child processes. +.TP 5 +SSLListen +.br +Listens on the specified address and port for encrypted connections. +.TP 5 +SSLOptions None +.TP 5 +SSLOptions NoEmptyFragments +.br +Sets SSL/TLS protocol options for encrypted connections. +.TP 5 +SSLPort +.br +Listens on the specified port for encrypted connections. +.TP 5 +SubscriptionPrivateAccess all +.TP 5 +SubscriptionPrivateAccess default +.TP 5 +SubscriptionPrivateAccess {user|@group|@ACL|@OWNER|@SYSTEM}+ +.br +Specifies an access list for a subscription's private values. The "default" +access list is "@OWNER @SYSTEM". "@ACL" maps to the printer's +requesting-user-name-allowed or requesting-user-name-denied values. +.TP 5 +SubscriptionPrivateValues all +.TP 5 +SubscriptionPrivateValues default +.TP 5 +SubscriptionPrivateValues none +.TP 5 +SubscriptionPrivateValues attribute-name-1 [ ... attribute-name-N ] +Specifies the list of job values to make private. The "default" values are +"notify-events", "notify-pull-method", "notify-recipient-uri", +"notify-subscriber-user-name", and "notify-user-data". +.TP 5 +SystemGroup group-name [group-name ...] +.br +Specifies the group(s) to use for System class authentication. +.TP 5 +TempDir directory +.br +Specifies the directory where temporary files are stored. +.TP 5 +Timeout seconds +.br +Specifies the HTTP request timeout in seconds. +.TP 5 +User user-name +.br +Specifies the user name or ID that is used when running external programs. +.TP 5 +WebInterface yes +.TP 5 +WebInterface no +Specifies whether the web interface is enabled. +.SH SEE ALSO +\fIclasses.conf(5)\fR, \fIcupsd(8)\fR, \fImime.convs(5)\fR, +\fImime.types(5)\fR, \fIprinters.conf(5)\fR, +\fIsubscriptions.conf(5)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id$". +.\" diff --git a/man/cupsd.man.in b/man/cupsd.man.in new file mode 100644 index 0000000000..fb9ab007d4 --- /dev/null +++ b/man/cupsd.man.in @@ -0,0 +1,72 @@ +.\" +.\" "$Id$" +.\" +.\" cupsd man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH cupsd 8 "CUPS" "9 March 2009" "Apple Inc." +.SH NAME +cupsd \- cups scheduler +.SH SYNOPSIS +.B cupsd +[ -c +.I config-file +] [ -f ] [ -F ] [ -h ] [ -l ] [ -t ] +.SH DESCRIPTION +\fIcupsd\fR is the scheduler for CUPS. It implements a printing system based +upon the Internet Printing Protocol, version 2.1. If no options are specified +on the command-line then the default configuration file +\fI@CUPS_SERVERROOT@/cupsd.conf\fR will be used. +.SH OPTIONS +.TP 5 +-c config-file +.br +Uses the named configuration file. +.TP 5 +-f +.br +Run \fIcupsd\fR in the foreground; the default is to run in the +background as a "daemon". +.TP 5 +-F +.br +Run \fIcupsd\fR in the foreground but detach the process from the +controlling terminal and current directory. This is useful for +running \fIcupsd\fR from \fIinit(8)\fR. +.TP 5 +-h +.br +Shows the program usage. +.TP 5 +-l +.br +This option is passed to \fIcupsd\fR when it is run from +\fIlaunchd(8)\fR. +.TP 5 +-t +.br +Test the configuration file for syntax errors. +.SH COMPATIBILITY +\fIcupsd\fR implements all of the required IPP/2.1 attributes and +operations. It also implements several CUPS-specific administration +operations. +.SH SEE ALSO +\fIbackend(7)\fR, \fIclasses.conf(5)\fR, \fIcups-deviced(8)\fR, +\fIcups-driverd(8)\fR, \fIcups-lpd(8)\fR, \fIcups-polld(8)\fR, +\fIcupsd.conf(5)\fR, \fIfilter(7)\fR, \fIlaunchd(8)\fR, +\fImime.convs(5)\fR, \fImime.types(5)\fR, \fIprinters.conf(5)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id$". +.\" diff --git a/man/cupsenable.man b/man/cupsenable.man new file mode 100644 index 0000000000..0552f42711 --- /dev/null +++ b/man/cupsenable.man @@ -0,0 +1,93 @@ +.\" +.\" "$Id: cupsenable.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" cupsenable/cupsdisable man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH cupsenable 8 "CUPS" "9 October 2008" "Apple Inc." +.SH NAME +cupsdisable, cupsenable \- stop/start printers and classes +.SH SYNOPSIS +.B cupsdisable +[ -E ] [-U +.I username +] [ -c ] [ -h +.I server[:port] +] [ -r +.I reason +] [ --hold ] destination(s) +.br +.B cupsenable +[ -E ] [-U +.I username +] [ -c ] [ -h +.I server[:port] +] [ --release ] destination(s) +.SH DESCRIPTION +\fIcupsenable\fR starts the named printers or classes. +.LP +\fIcupsdisable\fR stops the named printers or classes. The +following options may be used: +.TP 5 +-E +.br +Forces encryption of the connection to the server. +.TP 5 +-U username +.br +Uses the specified username when connecting to the server. +.TP 5 +-c +.br +Cancels all jobs on the named destination. +.TP 5 +-h server[:port] +.br +Uses the specified server and port. +.TP 5 +--hold +.br +Holds remaining jobs on the named printer. Useful for allowing the current +job to complete before performing maintenance. +.TP 5 +-r "reason" +.br +Sets the message associated with the stopped state. If no reason is specified +then the message is set to "Reason Unknown". +.TP 5 +--release +.br +Releases pending jobs for printing. Use after running \fIcupsdisable\fR with +the \fI--hold\fR option to resume printing. +.SH COMPATIBILITY +Unlike the System V printing system, CUPS allows printer names to +contain any printable character except SPACE, TAB, "/", or "#". +Also, printer and class names are \fInot\fR case-sensitive. +.LP +The System V versions of these commands are \fIdisable\fR and +\fIenable\fR. They have been renamed to avoid conflicts with the +\fIbash(1)\fR build-in commands of the same name. +.LP +The CUPS versions of \fIdisable\fR and \fIenable\fR may ask the +user for an access password depending on the printing system +configuration. This differs from the System V versions which +require the root user to execute these commands. +.SH SEE ALSO +\fIcupsaccept(8)\fR, \fIcupsreject(8)\fR, \fIcancel(1)\fR, \fIlp(1)\fR, +\fIlpadmin(8)\fR, \fIlpstat(1)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. + +.\" +.\" End of "$Id: cupsenable.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/cupsfilter.man b/man/cupsfilter.man new file mode 100644 index 0000000000..7aa9c83e41 --- /dev/null +++ b/man/cupsfilter.man @@ -0,0 +1,91 @@ +.\" +.\" "$Id: cupsfilter.man 9030 2010-03-05 03:55:56Z mike $" +.\" +.\" cupsfilter man page for CUPS. +.\" +.\" Copyright 2007-2010 by Apple Inc. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH cupsfilter 8 "CUPS" "4 March 2010" "Apple Inc." +.SH NAME +cupsfilter \- convert a file to another format using cups filters +.SH SYNOPSIS +.B cupsfilter +[ -c +.I config-file +] [ -d +.I printer +] [ -e ] -j +.I job-id[,N] +[ -m +.I mime/type +] [ -n +.I copies +] [ -o +.I name=value +] [ -p +.I filename.ppd +] [ -t +.I title +] +.I filename +.SH DESCRIPTION +\fIcupsfilter\fR is a front-end to the CUPS filter subsystem which allows you +to convert a file to a specific format, just as if you had printed the file +through CUPS. By default, \fIcupsfilter\fR generates a PDF file. +.SH OPTIONS +.TP 5 +-c config-file +.br +Uses the named cupsd.conf configuration file. +.TP 5 +-d printer +Uses information from the named printer. +.TP 5 +-e +.br +Use every filter from the PPD file. +.TP 5 +-j job-id[,N] +.br +Converts document N from the specified job. If N is omitted, document 1 is +converted. +.TP 5 +-m mime/type +.br +Specifies the destination file type. The default file type is application/pdf. +Use printer/foo to convert to the printer format defined by the filters in the +PPD file. +.TP 5 +-n copies +.br +Specifies the number of copies to generate. +.TP 5 +-o name=value +.br +Specifies options to pass to the CUPS filters. +.TP 5 +-p filename.ppd +.br +Specifies the PPD file to use. +.TP 5 +-t title +.br +Specifies the document title. +.SH KNOWN ISSUES +\fIcupsfilter\fR currently does not use the filters defined in the PPD file. +This will be addressed in a future CUPS release. +.SH SEE ALSO +\fIcupsd.conf(5)\fR +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2010 by Apple Inc. +.\" +.\" End of "$Id: cupsfilter.man 9030 2010-03-05 03:55:56Z mike $". +.\" diff --git a/man/cupstestdsc.man b/man/cupstestdsc.man new file mode 100644 index 0000000000..781c5efe0f --- /dev/null +++ b/man/cupstestdsc.man @@ -0,0 +1,50 @@ +.\" +.\" "$Id: cupstestdsc.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" cupstestdsc man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH cupstestdsc 1 "CUPS" "20 March 2006" "Apple Inc." +.SH NAME +cupstestdsc \- test conformance of postscript files +.SH SYNOPSIS +.B cupstestdsc +[ -h ] filename.ps [ ... +.I filenameN.ps +] +.br +.B cupstestdsc +[ -h ] - +.SH DESCRIPTION +\fIcupstestdsc\fR tests the conformance of PostScript files to +the Adobe PostScript Language Document Structuring Conventions +Specification version 3.0. The results of testing and any other +output are sent to the standard output. The second form of the +command reads PostScript from the standard input. +.SH LIMITATIONS +\fIcupstestdsc\fR only validates the DSC comments in a PostScript +file and does not attempt to validate the PostScript code itself. +Developers must ensure that the PostScript they generate follows +the rules defined by Adobe. Specifically, all pages must be +independent of each other, code outside page descriptions may not +affect the graphics state (current font, color, transform matrix, +etc.), and device-specific commands such as setpagedevice should +not be used. +.SH SEE ALSO +http://localhost:631/help +.br +Adobe PostScript Language Document Structuring Conventions +Specification, Version 3.0. +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: cupstestdsc.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/cupstestppd.man b/man/cupstestppd.man new file mode 100644 index 0000000000..042f4c2916 --- /dev/null +++ b/man/cupstestppd.man @@ -0,0 +1,165 @@ +.\" +.\" "$Id: cupstestppd.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" cupstestppd man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH cupstestppd 1 "CUPS" "19 November 2009" "Apple Inc." +.SH NAME +cupstestppd \- test conformance of ppd files +.SH SYNOPSIS +.B cupstestppd +[ -I +.I category +] [ -R +.I rootdir +] [ -W +.I category +] [ -q ] [-r] [ -v[v] ] filename.ppd[.gz] [ ... filenameN.ppd[.gz] ] +.br +.B cupstestppd +[ -R +.I rootdir +] [ -W +.I category +] [ -q ] [-r] [ -v[v] ] - +.SH DESCRIPTION +\fIcupstestppd\fR tests the conformance of PPD files to the +Adobe PostScript Printer Description file format specification +version 4.3. It can also be used to list the supported options +and available fonts in a PPD file. The results of testing and +any other output are sent to the standard output. +.LP +The first form of \fIcupstestppd\fR tests one or more PPD files +on the command-line. The second form tests the PPD file provided +on the standard input. +.SH OPTIONS +\fIcupstestppd\fR supports the following options: +.TP 5 +-I filename +.br +Ignores all PCFileName warnings. +.TP 5 +-I filters +.br +Ignores all filter errors. +.TP 5 +-I profiles +.br +Ignores all profile errors. +.TP 5 +-R rootdir +.br +Specifies an alternate root directory for the filter, pre-filter, +and other support file checks. +.TP 5 +-W constraints +.br +Report all UIConstraint errors as warnings. +.TP 5 +-W defaults +.br +Except for size-related options, report all default option errors as warnings. +.TP 5 +-W filters +.br +Report all filter errors as warnings. +.TP 5 +-W profiles +.br +Report all profile errors as warnings. +.TP 5 +-W sizes +.br +Report all media size errors as warnings. +.TP 5 +-W translations +.br +Report all translation errors as warnings. +.TP 5 +-W all +.br +Report all of the previous errors as warnings. +.TP 5 +-W none +.br +Report all of the previous errors as errors. +.TP 5 +-q +.br +Specifies that no information should be displayed. +.TP 5 +-r +.br +Relaxes the PPD conformance requirements so that common +whitespace, control character, and formatting problems are not +treated as hard errors. +.TP 5 +-v +.br +Specifies that detailed conformance testing results should be +displayed rather than the concise PASS/FAIL/ERROR status. +.TP 5 +-vv +.br +Specifies that all information in the PPD file should be +displayed in addition to the detailed conformance testing +results. +.LP +The \fI-q\fR, \fI-v\fR, and \fI-vv\fR options are mutually exclusive. +.SH EXIT STATUS +\fIcupstestppd\fR returns zero on success and non-zero on error. The +error codes are as follows: +.TP 5 +1 +.br +Bad command-line arguments or missing PPD filename. +.TP 5 +2 +.br +Unable to open or read PPD file. +.TP 5 +3 +.br +The PPD file contains format errors that cannot be skipped. +.TP 5 +4 +.br +The PPD file does not conform to the Adobe PPD specification. +.SH EXAMPLES +The following command will test all PPD files under the current +directory and print the names of each file that does not +conform: +.nf + + find . -name \\*.ppd \\! -exec cupstestppd -q '{}' \\; -print + +.fi +The next command tests all PPD files under the current directory +and print detailed conformance testing results for the files +that do not conform: +.nf + + find . -name \\*.ppd \\! -exec cupstestppd -q '{}' \\; \\ + -exec cupstestppd -v '{}' \\; + +.fi +.SH SEE ALSO +\fIlpadmin(8)\fR, +.br +http://localhost:631/help +.br +Adobe PostScript Printer Description File Format Specification, Version 4.3. +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: cupstestppd.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/filter.man b/man/filter.man new file mode 100644 index 0000000000..eef66bab00 --- /dev/null +++ b/man/filter.man @@ -0,0 +1,256 @@ +.\" +.\" "$Id: filter.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" filter man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2007 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH filter 7 "CUPS" "13 May 2009" "Apple Inc." +.SH NAME +filter \- cups file conversion filter interface +.SH SYNOPSIS +.B filter +job user title num-copies options [ +.I filename +] +.SH DESCRIPTION +The CUPS filter interface provides a standard method for adding support for +new document types to CUPS. Each filter is capable of converting from one +or more input formats to another format that can either be printed directly +or piped into another filter to get it to a printable format. +.LP +Filters \fBmust\fR be capable of reading from a filename on the command-line +or from the standard input, copying the standard input to a temporary +file as required by the file format. All output \fBmust\fR be sent to the +standard output. +.LP +The command name (argv[0]) is set to the name of the destination printer but is +also available in the PRINTER environment variable. + +.SH OPTIONS +Options passed on the command-line typically do not include the default choices +the printer's PPD file. In addition, some options may be specified in multiple +ways - "landscape" is a synonym for "orientation-requested=4", "media" is a +synonym for "PageSize", "PageRegion", "InputSlot", and "MediaType", and "sides" +is a synonym for the various "Duplex" options. Non-raster filters \fBmust\fR +support both explicit and implicit specification of PPD options - use the +ppdMarkDefaults and cupsMarkOptions functions in the CUPS library to use the +correct mapping, and ppdFindMarkedChoice to get the user-selected choice. +.LP +Raster filters should use option choices set through the raster page header, as +those reflect the options in effect for a given page. Options specified on the +command-line determine the default values for the entire job, which can be +overridden on a per-page basis. + +.SH LOG MESSAGES +Messages sent to stderr are generally logged to +printer-state-message attribute and the current \fIErrorLog\fR. +Each line begins with a standard prefix: + +.TP 5 +ALERT: message +.br +Sets the printer-state-message attribute and adds the specified +message to the current \fIErrorLog\fR using the "alert" log level. + +.TP 5 +ATTR: attribute=value [attribute=value] +.br +Sets the named job attribute(s). Typically this will be used to +set the job-remote-id attribute. + +.TP 5 +CRIT: message +.br +Sets the printer-state-message attribute and adds the specified +message to the current \fIErrorLog\fR using the "critical" log level. + +.TP 5 +DEBUG: message +.br +Sets the printer-state-message attribute and adds the specified +message to the current \fIErrorLog\fR using the "debug" log level. + +.TP 5 +DEBUG2: message +.br +Sets the printer-state-message attribute and adds the specified +message to the current \fIErrorLog\fR using the "debug2" log level. + +.TP 5 +EMERG: message +.br +Sets the printer-state-message attribute and adds the specified +message to the current \fIErrorLog\fR using the "emergency" log level. + +.TP 5 +ERROR: message +.br +Sets the printer-state-message attribute and adds the specified +message to the current \fIErrorLog\fR using the "error" log level. + +.TP 5 +INFO: message +.br +Sets the printer-state-message attribute. If the current \fILogLevel\fR +is set to "debug2", also adds the specified message to the +current \fIErrorLog\fR using the "info" log level. + +.TP 5 +NOTICE: message +.br +Sets the printer-state-message attribute and adds the specified +message to the current \fIErrorLog\fR using the "notice" log level. + +.TP 5 +PAGE: page-number #-copies +.TP 5 +PAGE: total #-pages +.br +Adds an entry to the current \fIPageLog\fR. The first form adds +#-copies to the job-media-sheets-completed attribute. The second +form sets the job-media-sheets-completed attribute to #-pages. + +.TP 5 +PPD: Keyword=Value ... KeywordN=Value +.br +Sets the named keywords in the printer's PPD file. This is typically +used to update default option keywords such as DefaultPageSize and +the various installable options in the PPD file. + +.TP 5 +STATE: printer-state-reason [printer-state-reason ...] +.TP 5 +STATE: + printer-state-reason [printer-state-reason ...] +.TP 5 +STATE: - printer-state-reason [printer-state-reason ...] +.br +Sets, adds, or removes printer-state-reason keywords to the +current queue. Typically this is used to indicate media, ink, and +toner conditions on a printer. + +.TP 5 +WARNING: message +.br +Sets the printer-state-message attribute and adds the specified +message to the current \fIErrorLog\fR using the "warning" log level. + +.SH ENVIRONMENT VARIABLES +The following environment variables are defined by the CUPS +server when executing the filter: + +.TP 5 +CHARSET +.br +The default text character set, typically utf-8. + +.TP 5 +CLASS +.br +When a job is submitted to a printer class, contains the name of +the destination printer class. Otherwise this environment +variable will not be set. + +.TP 5 +CONTENT_TYPE +.br +The MIME type associated with the file (e.g. +application/postscript). + +.TP 5 +CUPS_CACHEDIR +.br +The directory for semi-persistent cache files can be found. + +.TP 5 +CUPS_DATADIR +.br +The directory where data files can be found. + +.TP 5 +CUPS_FILETYPE +.br +The type of file being printed: "job-sheet" for a banner page and "document" +for a regular print file. + +.TP 5 +CUPS_SERVERROOT +.br +The root directory of the server. + +.TP 5 +DEVICE_URI +.br +The device-uri associated with the printer. + +.TP 5 +FINAL_CONTENT_TYPE +.br +The MIME type associated with the printer (e.g. +application/vnd.cups-postscript). + +.TP 5 +LANG +.br +The default language locale (typically C or en). + +.TP 5 +PATH +.br +The standard execution path for external programs that may be run by +the filter. + +.TP 5 +PPD +.br +The full pathname of the PostScript Printer Description (PPD) +file for this printer. + +.TP 5 +PRINTER +.br +The name of the printer. + +.TP 5 +RIP_CACHE +.br +The recommended amount of memory to use for Raster Image +Processors (RIPs). + +.TP 5 +SOFTWARE +.br +The name and version number of the server (typically CUPS/1.2). + +.TP 5 +TZ +.br +The timezone of the server. + +.TP 5 +USER +.br +The user executing the filter, typically "lp" or "root"; consult the +\fIcupsd.conf(5)\fR file for the current setting. + +.SH COMPATIBILITY +While the filter interface is compatible with System V interface +scripts, it will only work with the System V interface script as the +only filter. Typically the interface script will be provided via the +\fIlpadmin(8)\fR command using the \fI-i\fR option. +.SH SEE ALSO +\fIbackend(7)\fR, \fIcupsd(8)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: filter.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/ipptool.man b/man/ipptool.man new file mode 100644 index 0000000000..6cf8450907 --- /dev/null +++ b/man/ipptool.man @@ -0,0 +1,130 @@ +.\" +.\" "$Id: ipptool.man 9354 2010-11-10 06:48:19Z mike $" +.\" +.\" ipptool man page for CUPS. +.\" +.\" Copyright 2010 by Apple Inc. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH ipptool 1 "CUPS" "9 November 2010" "Apple Inc." +.SH NAME +ipptool - perform internet printing protocol requests +.SH SYNOPSIS +.B ipptool +[ -4 ] [ -6 ] [ -C ] [ -E ] [ -I ] [ -L ] [ -S ] [ -T +.I seconds +] [ -V +.I version +] [ -X ] [ -c ] [ -d +.I name=value +] [ -f +.I filename +] [ -i +.I seconds +] [ -n +.I repeat-count +] [ -q ] [ -t ] [ -v ] +.I URI +.I filename +[ +.I ... filenameN +] +.SH DESCRIPTION +\fIipptool\fR sends IPP requests to the specified URI and tests and/or displays the results. Each named file defines one or more requests, including the expected response status, attributes, and values. Output is either a plain text, formatted text, CSV, or XML report on the standard output, with a non-zero exit status indicating that one or more tests have failed. The file format is described in \fIipptoolfile(5)\fR. +.SH OPTIONS +The following options are recognized by \fIipptool\fR: +.TP 5 +-4 +Specifies that \fIipptool\fR must connect to the printer or server using IPv4. +.TP 5 +-6 +Specifies that \fIipptool\fR must connect to the printer or server using IPv6. +.TP 5 +-C +Specifies that requests should be sent using the HTTP/1.1 "Transfer-Encoding: chunked" header, which is required for conformance by all versions of IPP. The default is to use "Transfer-Encoding: chunked" for requests with attached files and "Content-Length:" for requests without attached files. +.TP 5 +-E +Forces TLS encryption when connecting to the server using the HTTP "Upgrade" header. +.TP 5 +-I +Specifies that \fIipptool\fR will continue past errors. +.TP 5 +-L +Specifies that requests should be sent using the HTTP/1.0 "Content-Length:" header, which is required for conformance by all versions of IPP. The default is to use "Transfer-Encoding: chunked" for requests with attached files and "Content-Length:" for requests without attached files. +.TP 5 +-S +Forces (dedicated) SSL encryption when connecting to the server. +.TP 5 +-T seconds +Specifies a timeout for IPP requests in seconds. +.TP 5 +-V version +Specifies the default IPP version to use: 1.0, 1.1, 2.0, 2.1, or 2.2. If not specified, version 1.1 is used. +.TP 5 +-X +Specifies that XML (Apple plist) output is desired instead of the plain text report. This option is incompatible with the \fI-i\fR (interval) and \fI-n\fR (repeat-count) options. +.TP 5 +-c +Specifies that CSV (comma-separated values) output is desired instead of the plain text output. +.TP 5 +-d name=value +Defines the named variable. +.TP 5 +-f filename +Defines the default request filename for tests. +.TP 5 +-i seconds +Specifies that the (last) file should be repeated at the specified interval. This option is incompatible with the \fI-X\fR (XML plist output) option. +.TP 5 +-l +Specifies that plain text output is desired. +.TP 5 +-n repeat-count +Specifies that the (last) file should be repeated the specified number of times. This option is incompatible with the \fI-X\fR (XML plist output) option. +.TP 5 +-t +Specifies that CUPS test report output is desired instead of the plain text output. +.TP 5 +-v +Specifies that all request and response attributes should be output in CUPS test mode (\fI-t\fR). This is the default for XML output. +.SH COMPATIBILITY +The \fIipptool\fR program is unique to CUPS. +.SH EXAMPLES +Get a list of completed jobs for "myprinter": +.nf + ipptool ipp://localhost/printers/myprinter get-completed-jobs.test +.fi +.LP +Send email notifications to "user@example.com" when "myprinter" changes: +.nf + ipptool -d recipient=mailto:user@example.com \ + ipp://localhost/printers/myprinter create-printer-subscription.test +.fi +.SH STANDARD FILES +The following standard files are available: +.nf + create-printer-subscription.test + get-completed-jobs.test + get-jobs.test + ipp-1.1.test + ipp-2.0.test + ipp-2.1.test + testfile.jpg + testfile.pdf + testfile.ps + testfile.txt +.fi +.SH SEE ALSO +\fIipptoolfile(5)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2010 by Apple Inc. +.\" +.\" End of "$Id: ipptool.man 9354 2010-11-10 06:48:19Z mike $". +.\" diff --git a/man/ipptoolfile.man b/man/ipptoolfile.man new file mode 100644 index 0000000000..c3fb566622 --- /dev/null +++ b/man/ipptoolfile.man @@ -0,0 +1,523 @@ +.\" +.\" "$Id: ipptoolfile.man 10022 2011-09-28 16:21:00Z mike $" +.\" +.\" ipptoolfile man page for CUPS. +.\" +.\" Copyright 2010-2011 by Apple Inc. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH ipptoolfile 5 "CUPS" "28 September 2011" "Apple Inc." +.SH NAME +ipptoolfile \- ipptool file format + +.SH DESCRIPTION +The \fIipptool(1)\fR program accepts free-form plain text files that describe one or more IPP requests. Comments start with the "#" character and continue to the end of the line. Each request is enclosed by curley braces, for example: +.nf + + # This is a comment + { + # The name of the test + NAME "Print PostScript Job" + + # The request to send + OPERATION Print-Job + GROUP operation-attributes-tag + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $uri + ATTR name requesting-user-name $user + FILE testfile.ps + + # The response to expect + STATUS successful-ok + EXPECT attributes-charset OF-TYPE charset + EXPECT attributes-natural-language OF-TYPE naturalLanguage + EXPECT job-id OF-TYPE integer + EXPECT job-uri OF-TYPE uri + } + { + # The name of the test + NAME "Get Attributes of PostScript Job" + + # The request to send + OPERATION Get-Job-Attributes + GROUP operation-attributes-tag + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $uri + ATTR integer job-id $job-id + ATTR name requesting-user-name $user + + # The response to expect + STATUS successful-ok + EXPECT attributes-charset OF-TYPE charset + EXPECT attributes-natural-language OF-TYPE naturalLanguage + EXPECT job-id OF-TYPE integer + EXPECT job-uri OF-TYPE uri + EXPECT job-state OF-TYPE enum + EXPECT job-originating-user-name OF-TYPE name WITH-VALUE "$user" + } +.fi + +.SH TOP-LEVEL DIRECTIVES +The following directives can be used outside of a test: +.TP 5 +{ test } +Defines a test. +.TP 5 +DEFINE variable-name value +Defines the named variable to the given value. This is equivalent to specifying +"-d variable-name=value" on the \fIipptool\fR command-line. +.TP 5 +DEFINE-DEFAULT variable-name value +Defines the named variable to the given value if it does not already have a +value. +.TP 5 +IGNORE-ERRORS yes +.TP 5 +IGNORE-ERRORS no +Specifies whether, by default, \fIipptool\fR will ignore errors and continue with +subsequent tests. +.TP 5 +INCLUDE "filename" +.TP 5 +INCLUDE +Includes another test file. The first form includes a file relative to the +current test file, while the second form includes a file from the \fIipptool\fR +include directory. +.TP 5 +INCLUDE-IF-DEFINED name "filename" +.TP 5 +INCLUDE-IF-DEFINED name +Includes another test file if the named variable is defined. The first form +includes a file relative to the current test file, while the second form +includes a file from the \fIipptool\fR include directory. +.TP 5 +INCLUDE-IF-NOT-DEFINED name "filename" +.TP 5 +INCLUDE-IF-NOT-DEFINED name +Includes another test file if the named variable is not defined. The first form +includes a file relative to the current test file, while the second form +includes a file from the \fIipptool\fR include directory. +.TP 5 +SKIP-IF-DEFINED variable-name +.TP 5 +SKIP-IF-NOT-DEFINED variable-name +Specifies that the remainder of the test file should be skipped when the +variable is or is not defined. +.TP 5 +TRANSFER auto +Specifies that tests will, by default, use "Transfer-Encoding: chunked" for +requests with attached files and "Content-Length:" for requests without attached +files. +.TP 5 +TRANSFER chunked +Specifies that tests will, by default, use the HTTP/1.1 "Transfer-Encoding: +chunked" header. This is the default and is equivalent to specifying "-c" on the +\fIipptool\fR command-line. Support for chunked requests is required for +conformance with all versions of IPP. +.TP 5 +TRANSFER length +Specifies that tests will, by default, use the HTTP/1.0 "Content-Length:" +header. This is equivalent to specifying "-l" on the \fIipptool\fR command-line. +Support for content length requests is required for conformance with all +versions of IPP. +.TP 5 +VERSION 1.0 +.TP 5 +VERSION 1.1 +.TP 5 +VERSION 2.0 +.TP 5 +VERSION 2.1 +.TP 5 +VERSION 2.2 +Specifies the default IPP version number to use for the tests that follow. + +.SH TEST DIRECTIVES +The following directives are understood in a test: +.TP 5 +ATTR tag attribute-name value(s) +Adds an attribute to the test request. Values are separated by the comma (",") +character - escape commas using the "\" character. +.TP 5 +ATTR collection attribute-name { MEMBER tag member-name value(s) ... } [ ... { ... } ] +Adds a collection attribute to the test request. Member attributes follow the +same syntax as regular attributes and can themselves be nested collections. +Multiple collection values can be supplied as needed. +.TP 5 +DELAY seconds +Specifies a delay before this test will be run. +.TP 5 +DISPLAY attribute-name +Specifies that value of the named attribute should be output as part of the +test report. +.TP 5 +EXPECT attribute-name [ predicate(s) ] +.TP 5 +EXPECT ?attribute-name predicate(s) +.TP 5 +EXPECT !attribute-name +Specifies that the response must/may/must not include the named attribute. +Additional requirements can be added as predicates - see the "EXPECT PREDICATES" +section for more information on predicates. +.TP 5 +FILE filename +Specifies a file to include at the end of the request. This is typically used +when sending a test print file. +.TP 5 +GROUP tag +Specifies the group tag for subsequent attributes in the request. +.TP 5 +IGNORE-ERRORS yes +.TP 5 +IGNORE-ERRORS no +Specifies whether \fIipptool\fR will ignore errors and continue with subsequent +tests. +.TP 5 +NAME "literal string" +Specifies the human-readable name of the test. +.TP 5 +OPERATION operation-code +Specifies the operation to be performed. +.TP 5 +REQUEST-ID number +.TP 5 +REQUEST-ID random +Specifies the request-id value to use in the request, either an integer or the +word "random" to use a randomly generated value (the default). +.TP 5 +RESOURCE path +Specifies an alternate resource path that is used for the HTTP POST request. +The default is the resource from the URI provided to the \fIipptool\fR program. +.TP 5 +SKIP-IF-DEFINED variable-name +.TP 5 +SKIP-IF-NOT-DEFINED variable-name +Specifies that the current test should be skipped when the variable is or is not +defined. +.TP 5 +SKIP-PREVIOUS-ERROR yes +.TP 5 +SKIP-PREVIOUS-ERROR no +Specifies whether \fIipptool\fR will skip the current test if the previous test +resulted in an error/failure. +.TP 5 +STATUS status-code [ predicate ] +Specifies an expected response status-code value. Additional requirements can be +added as predicates - see the "STATUS PREDICATES" section for more information +on predicates. +.TP 5 +TRANSFER auto +Specifies that this test will use "Transfer-Encoding: chunked" if it has an +attached file or "Content-Length:" otherwise. +.TP 5 +TRANSFER chunked +Specifies that this test will use the HTTP/1.1 "Transfer-Encoding: chunked" +header. +.TP 5 +TRANSFER length +Specifies that this test will use the HTTP/1.0 "Content-Length:" header. +.TP 5 +VERSION 1.0 +.TP 5 +VERSION 1.1 +.TP 5 +VERSION 2.0 +.TP 5 +VERSION 2.1 +.TP 5 +VERSION 2.2 +Specifies the IPP version number to use for this test. + +.SH EXPECT PREDICATES +The following predicates are understood following the EXPECT test directive: +.TP 5 +COUNT number +Requires the EXPECT attribute to have the specified number of values. +.TP 5 +DEFINE-MATCH variable-name +Defines the variable to "1" when the EXPECT condition matches. A side-effect of +this predicate is that this EXPECT will never fail a test. +.TP 5 +DEFINE-NO-MATCH variable-name +Defines the variable to "1" when the EXPECT condition does not match. A side- +effect of this predicate is that this EXPECT will never fail a test. +.TP 5 +DEFINE-VALUE variable-name +Defines the variable to the value of the attribute when the EXPECT condition +matches. A side-effect of this predicate is that this EXPECT will never fail a test. +.TP 5 +IF-DEFINED variable-name +Makes the EXPECT conditions apply only if the specified variable is defined. +.TP 5 +IF-NOT-DEFINED variable-name +Makes the EXPECT conditions apply only if the specified variable is not +defined. +.TP 5 +IN-GROUP tag +Requires the EXPECT attribute to be in the specified group tag. +.TP 5 +OF-TYPE tag[,tag,...] +Requires the EXPECT attribute to use the specified value tag(s). +.TP 5 +REPEAT-MATCH +.TP 5 +REPEAT-NO-MATCH +Specifies that the current test should be repeated when the EXPECT condition +matches or does not match. +.TP 5 +SAME-COUNT-AS attribute-name +Requires the EXPECT attribute to have the same number of values as the specified +parallel attribute. +.TP 5 +WITH-VALUE "literal string" +Requires at least one value of the EXPECT attribute to match the literal string. +Comparisons are case-sensitive. +.TP 5 +WITH-VALUE "/regular expression/" +Requires that all values of the EXPECT attribute match the regular expression, +which must conform to the POSIX regular expression syntax. +Comparisons are case-sensitive. + +.SH STATUS PREDICATES +The following predicates are understood following the STATUS test directive: +.TP 5 +IF-DEFINED variable-name +Makes the STATUS apply only if the specified variable is defined. +.TP 5 +IF-NOT-DEFINED variable-name +Makes the STATUS apply only if the specified variable is not defined. +.TP 5 +REPEAT-MATCH +.TP 5 +REPEAT-NO-MATCH +Specifies that the current test should be repeated when the response status-code +matches or does not match the value specified by the STATUS directive. + +.SH OPERATION CODES +Operation codes correspond to the hexadecimal numbers (0xHHHH) and names from +RFC 2911 and other IPP extension specifications. Here is a complete list: +.nf + Activate-Printer + CUPS-Accept-Jobs + CUPS-Add-Modify-Class + CUPS-Add-Modify-Printer + CUPS-Authenticate-Job + CUPS-Delete-Class + CUPS-Delete-Printer + CUPS-Get-Classes + CUPS-Get-Default + CUPS-Get-Devices + CUPS-Get-Document + CUPS-Get-PPD + CUPS-Get-PPDs + CUPS-Get-Printers + CUPS-Move-Job + CUPS-Reject-Jobs + CUPS-Set-Default + Cancel-Current-Job + Cancel-Job + Cancel-Jobs + Cancel-My-Jobs + Cancel-Subscription + Close-Job + Create-Job + Create-Job-Subscription + Create-Printer-Subscription + Deactivate-Printer + Disable-Printer + Enable-Printer + Get-Job-Attributes + Get-Jobs + Get-Notifications + Get-Printer-Attributes + Get-Printer-Support-Files + Get-Printer-Supported-Values + Get-Subscription-Attributes + Get-Subscriptions + Hold-Job + Hold-New-Jobs + Pause-Printer + Pause-Printer-After-Current-Job + Print-Job + Print-URI + Promote-Job + Purge-Jobs + Release-Held-New-Jobs + Release-Job + Renew-Subscription + Reprocess-Job + Restart-Job + Restart-Printer + Resubmit-Job + Resume-Job + Resume-Printer + Schedule-Job-After + Send-Document + Send-Notifications + Send-URI + Set-Job-Attributes + Set-Printer-Attributes + Shutdown-Printer + Startup-Printer + Suspend-Current-Job + Validate-Job +.fi + +.SH STATUS CODES +Status codes correspond to the hexadecimal numbers (0xHHHH) and names from RFC +2911 and other IPP extension specifications. Here is a complete list: +.nf + client-error-attributes-not-settable + client-error-attributes-or-values-not-supported + client-error-bad-request + client-error-charset-not-supported + client-error-compression-error + client-error-compression-not-supported + client-error-conflicting-attributes + client-error-document-access-error + client-error-document-format-error + client-error-document-format-not-supported + client-error-forbidden + client-error-gone + client-error-ignored-all-notifications + client-error-ignored-all-subscriptions + client-error-not-authenticated + client-error-not-authorized + client-error-not-found + client-error-not-possible + client-error-print-support-file-not-found + client-error-request-entity-too-large + client-error-request-value-too-long + client-error-timeout + client-error-too-many-subscriptions + client-error-uri-scheme-not-supported + cups-see-other + redirection-other-site + server-error-busy + server-error-device-error + server-error-internal-error + server-error-job-canceled + server-error-multiple-document-jobs-not-supported + server-error-not-accepting-jobs + server-error-operation-not-supported + server-error-printer-is-deactivated + server-error-service-unavailable + server-error-temporary-error + server-error-version-not-supported + successful-ok + successful-ok-but-cancel-subscription + successful-ok-conflicting-attributes + successful-ok-events-complete + successful-ok-ignored-notifications + successful-ok-ignored-or-substituted-attributes + successful-ok-ignored-subscriptions + successful-ok-too-many-events +.fi + +.SH TAGS +Value and group tags correspond to the names from RFC 2911 and other IPP +extension specifications. Here are the group tags: +.nf + event-notification-attributes-tag + job-attributes-tag + operation-attributes-tag + printer-attributes-tag + subscription-attributes-tag + unsupported-attributes-tag +.fi +.LP +Here are the value tags: +.nf + admin-define + boolean + charset + collection + dateTime + default + delete-attribute + enum + integer + keyword + mimeMediaType + nameWithLanguage + nameWithoutLanguage + naturalLanguage + no-value + not-settable + octetString + rangeOfInteger + resolution + textWithLanguage + textWithoutLanguage + unknown + unsupported + uri + uriScheme +.fi + +.SH VARIABLES +The \fIipptool\fR program maintains a list of variables that can be used in any +literal string or attribute value by specifying "$variable-name". Aside from +variables defined using the "-d" option or "DEFINE" directive, the following +pre-defined variables are available: +.TP 5 +$$ +Inserts a single "$" character. +.TP 5 +$ENV[name] +Inserts the value of the named environment variable, or an empty string if the +environment variable is not defined. +.TP 5 +$filename +Inserts the filename provided to \fIipptool\fR with the "-f" option. +.TP 5 +$hostname +Inserts the hostname from the URI provided to \fIipptool\fR. +.TP 5 +$job-id +Inserts the last job-id value returned in a test response or 0 if no job-id has +been seen. +.TP 5 +$job-uri +Inserts the last job-uri value returned in a test response or an empty string if +no job-uri has been seen. +.TP 5 +$scheme +Inserts the scheme from the URI provided to \fIipptool\fR. +.TP 5 +$notify-subscription-id +Inserts the last notify-subscription-id value returnd in a test response or 0 if +no notify-subscription-id has been seen. +.TP 5 +$port +Inserts the port number from the URI provided to \fIipptool\fR. +.TP 5 +$resource +Inserts the resource path from the URI provided to \fIipptool\fR. +.TP 5 +$uri +Inserts the URI provided to \fIipptool\fR. +.TP 5 +$user +Inserts the current user's login name. +.TP 5 +$username +Inserts the username from the URI provided to \fIipptool\fR, if any. + +.SH SEE ALSO +\fIipptool(1)\fR, +.br +http://localhost:631/help + +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: ipptoolfile.man 10022 2011-09-28 16:21:00Z mike $". +.\" diff --git a/man/lp.man b/man/lp.man new file mode 100644 index 0000000000..ff01f7f12d --- /dev/null +++ b/man/lp.man @@ -0,0 +1,258 @@ +.\" +.\" "$Id: lp.man 10040 2011-10-03 17:25:13Z mike $" +.\" +.\" lp man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH lp 1 "CUPS" "3 October 2011" "Apple Inc." +.SH NAME +lp - print files +.SH SYNOPSIS +.B lp +[ -E ] [ -U +.I username +] [ -c ] [ -d +.I destination[/instance] +] [ -h +.I hostname[:port] +] [ -m ] [ -n +.I num-copies +] [ -o +.I option[=value] +] [ -q +.I priority +] [ -s ] [ -t +.I title +] [ -H +.I handling +] [ -P +.I page-list +] [ -- ] [ +.I file(s) +] +.br +.B lp +[ -E ] [ -U +.I username +] [ -c ] [ -h +.I hostname[:port] +] [ -i +.I job-id +] [ -n +.I num-copies +] [ -o +.I option[=value] +] [ -q +.I priority +] [ -t +.I title +] [ -H +.I handling +] [ -P +.I page-list +] +.SH DESCRIPTION +\fIlp\fR submits files for printing or alters a pending job. Use +a filename of "-" to force printing from the standard input. +.SH THE DEFAULT DESTINATION +CUPS provides many ways to set the default destination. The "LPDEST" and +"PRINTER" environment variables are consulted first. If neither are set, +the current default set using the \fIlpoptions(1)\fR command is used, +followed by the default set using the \fIlpadmin(8)\fR command. +.SH OPTIONS +The following options are recognized by \fIlp\fR: +.TP 5 +-- +.br +Marks the end of options; use this to print a file whose name +begins with a dash (-). +.TP 5 +-E +.br +Forces encryption when connecting to the server. +.TP 5 +-U username +.br +Specifies the username to use when connecting to the server. +.TP 5 +-c +.br +This option is provided for backwards-compatibility only. On +systems that support it, this option forces the print file to be +copied to the spool directory before printing. In CUPS, print +files are always sent to the scheduler via IPP which has the +same effect. +.TP 5 +-d destination +.br +Prints files to the named printer. +.TP 5 +-h hostname[:port] +.br +Chooses an alternate server. +.TP 5 +-i job-id +.br +Specifies an existing job to modify. +.TP 5 +-m +.br +Sends an email when the job is completed. +.TP 5 +-n copies +.br +Sets the number of copies to print from 1 to 100. +.TP 5 +-o "name=value [name=value ...]" +.br +Sets one or more job options. +.TP 5 +-q priority +.br +Sets the job priority from 1 (lowest) to 100 (highest). The +default priority is 50. +.TP 5 +-s +.br +Do not report the resulting job IDs (silent mode.) +.TP 5 +-t "name" +.br +Sets the job name. +.TP 5 +-H hh:mm +.TP 5 +-H hold +.TP 5 +-H immediate +.TP 5 +-H restart +.TP 5 +-H resume +.br +Specifies when the job should be printed. A value of \fIimmediate\fR will print +the file immediately, a value of \fIhold\fR will hold the job indefinitely, and +a UTC time value (HH:MM) will hold the job until the specified UTC (not local) +time. Use a value of \fIresume\fR with the \fI-i\fR option to resume a held job. +Use a value of \fIrestart\fR with the \fI-i\fR option to restart +a completed job. +.TP 5 +-P page-list +.br +Specifies which pages to print in the document. The list can +contain a list of numbers and ranges (#-#) separated by commas +(e.g. 1,3-5,16). The page numbers refer to the output pages and +not the document's original pages - options like "number-up" can +affect the numbering of the pages. +.SH COMMON JOB OPTIONS +Aside from the printer-specific options reported by the +\fIlpoptions(1)\fR command, the following generic options are +available: +.TP 5 +-o media=size +.br +Sets the page size to \fIsize\fR. Most printers support at least +the size names "a4", "letter", and "legal". +.TP 5 +-o landscape +.TP 5 +-o orientation-requested=4 +.br +Prints the job in landscape (rotated 90 degrees). +.TP 5 +-o sides=one-sided +.TP 5 +-o sides=two-sided-long-edge +.TP 5 +-o sides=two-sided-short-edge +.br +Prints on one or two sides of the paper. The value +"two-sided-long-edge" is normally used when printing portrait +(unrotated) pages, while "two-sided-short-edge" is used for +landscape pages. +.TP 5 +-o fitplot +.br +Scales the print file to fit on the page. +.TP 5 +-o number-up=2 +.TP 5 +-o number-up=4 +.TP 5 +-o number-up=6 +.TP 5 +-o number-up=9 +.TP 5 +-o number-up=16 +.br +Prints multiple document pages on each output page. +.TP 5 +-o scaling=number +.br +Scales image files to use up to \fInumber\fR percent of the page. +Values greater than 100 cause the image file to be printed across +multiple pages. +.TP 5 +-o cpi=N +.br +Sets the number of characters per inch to use when printing a +text file. The default is 10. +.TP 5 +-o lpi=N +.br +Sets the number of lines per inch to use when printing a text +file. The default is 6. +.TP 5 +-o page-bottom=N +.TP 5 +-o page-left=N +.TP 5 +-o page-right=N +.TP 5 +-o page-top=N +.br +Sets the page margins when printing text files. The values are in +points - there are 72 points to the inch. +.SH EXAMPLES +Print a double-sided legal document to a printer called "foo": +.nf + lp -d foo -o media=legal -o sides=two-sided-long-edge filename +.fi +.LP +Print an image across 4 pages: +.nf + lp -d bar -o scaling=200 filename +.fi +.LP +Print a text file with 12 characters per inch, 8 lines per inch, and +a 1 inch left margin: +.nf + lp -d bar -o cpi=12 -o lpi=8 -o page-left=72 filename +.fi +.SH COMPATIBILITY +Unlike the System V printing system, CUPS allows printer names to +contain any printable character except SPACE, TAB, "/", or "#". +Also, printer and class names are \fInot\fR case-sensitive. +.LP +The "q" option accepts a different range of values than the +Solaris lp command, matching the IPP job priority values (1-100, +100 is highest priority) instead of the Solaris values (0-39, 0 +is highest priority). +.SH SEE ALSO +\fIcancel(1)\fR, \fIlpadmin(8)\fR, \fIlpmove(8)\fR, \fIlpoptions(1)\fR, +\fIlpstat(1)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: lp.man 10040 2011-10-03 17:25:13Z mike $". +.\" diff --git a/man/lpadmin.man b/man/lpadmin.man new file mode 100644 index 0000000000..c388f03b7e --- /dev/null +++ b/man/lpadmin.man @@ -0,0 +1,228 @@ +.\" +.\" "$Id: lpadmin.man 9762 2011-05-11 05:30:50Z mike $" +.\" +.\" lpadmin man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH lpadmin 8 "CUPS" "10 May 2011" "Apple Inc." +.SH NAME +lpadmin \- configure cups printers and classes +.SH SYNOPSIS +.B lpadmin +[ -E ] [-U +.I username +] [ -h +.I server[:port] +] -d +.I destination +.br +.B lpadmin +[ -E ] [-U +.I username +] [ -h +.I server[:port] +] -p +.I destination +[ -R +.I name-default +] +.I option(s) +.br +.B lpadmin +[ -E ] [-U +.I username +] [ -h +.I server[:port] +] -x +.I destination +.SH DESCRIPTION +\fIlpadmin\fR configures printer and class queues provided by +CUPS. It can also be used to set the server default printer or +class. +.LP +When specified before the \fI-d\fR, \fI-p\fR, or \fI-x\fR +options, the \fI-E\fR option forces encryption when connecting to +the server. +.LP +The first form of the command (\fI-d\fR) sets the default printer +or class to \fIdestination\fR. Subsequent print jobs submitted +via the \fIlp(1)\fR or \fIlpr(1)\fR commands will use this +destination unless the user specifies otherwise with the +\fIlpoptions(1)\fR command. +.LP +The second form of the command (\fI-p\fR) configures the named +printer or class. The additional options are described below. +.LP +The third form of the command (\fI-x\fR) deletes the printer or +class \fIdestination\fR. Any jobs that are pending for the +destination will be removed and any job that is currently printed +will be aborted. +.SH CONFIGURATION OPTIONS +The following options are recognized when configuring a printer +queue: +.TP 5 +-c class +.br +Adds the named \fIprinter\fR to \fIclass\fR. If \fIclass\fR does +not exist it is created automatically. +.TP 5 +-i interface +.br +Sets a System V style interface script for the printer. This +option cannot be specified with the \fI-P\fR option (PPD file) +and is intended for providing support for legacy printer drivers. +.TP 5 +-m model +.br +Sets a standard System V interface script or PPD file for the printer from the +\fImodel\fR directory or using one of the driver interfaces. Use the \fI-m\fR +option with the \fIlpinfo(8)\fR command to get a list of supported models. +.TP 5 +-o cupsIPPSupplies=true +.TP 5 +-o cupsIPPSupplies=false +.br +Specifies whether IPP supply level values should be reported. +.TP 5 +-o cupsSNMPSupplies=true +.TP 5 +-o cupsSNMPSupplies=false +.br +Specifies whether SNMP supply level (RFC 3805) values should be reported. +.TP 5 +-o job-k-limit=value +.br +Sets the kilobyte limit for per-user quotas. The value is an +integer number of kilobytes; one kilobyte is 1024 bytes. +.TP 5 +-o job-page-limit=value +.br +Sets the page limit for per-user quotas. The value is the integer +number of pages that can be printed; double-sided pages are +counted as two pages. +.TP 5 +-o job-quota-period=value +.br +Sets the accounting period for per-user quotas. The value is an +integer number of seconds; 86,400 seconds are in one day. +.TP 5 +-o job-sheets-default=banner +.TP 5 +-o job-sheets-default=banner,banner +.br +Sets the default banner page(s) to use for print jobs. +.TP 5 +-o name=value +.br +Sets a PPD option for the printer. PPD options can be listed using the \fI-l\fR +option with the \fIlpoptions(1)\fR command. +.TP 5 +-o name-default=value +.br +Sets a default server-side option for the destination. Any print-time +option can be defaulted, e.g. "-o cpi-default=17" to set the default +"cpi" option value to 17. +.TP 5 +-o port-monitor=name +.br +Sets the binary communications program to use when printing, +"none", "bcp", or "tbcp". The default program is "none". The +specified port monitor must be listed in the printer's PPD file. +.TP 5 +-o printer-error-policy=name +.br +Sets the error policy to be used when the printer backend is +unable to send the job to the printer. The name must be one of +"abort-job", "retry-job", "retry-current-job", or "stop-printer". The default +error policy is "stop-printer" for printers and "retry-current-job" for +classes. +.TP 5 +-o printer-is-shared=true/false +.br +Sets the destination to shared/published or unshared/unpublished. +Shared/published destinations are publicly announced by the server +on the LAN based on the browsing configuration in +\fBcupsd.conf\fR, while unshared/unpublished destinations are not +announced. The default value is "true". +.TP 5 +-o printer-op-policy=name +.br +Sets the IPP operation policy associated with the destination. The +name must be defined in the \fBcupsd.conf\fR in a Policy section. +The default operation policy is "default". +.TP 5 +-R name-default +.br +Deletes the named option from \fIprinter\fR. +.TP 5 +-r class +.br +Removes the named \fIprinter\fR from \fIclass\fR. If the +resulting class becomes empty it is removed. +.TP 5 +-u allow:user,user,@group +.TP 5 +-u deny:user,user,@group +.TP 5 +-u allow:all +.TP 5 +-u deny:none +.br +Sets user-level access control on a destination. Names starting with +"@" are interpreted as UNIX groups. The latter two forms turn +user-level access control off. +.TP 5 +-v "device-uri" +.br +Sets the \fIdevice-uri\fR attribute of the printer queue. If +\fIdevice-uri\fR is a filename it is automatically converted to +the form \fIfile:///file/name\fR. Use the \fI-v\fR option with the +\fIlpinfo(8)\fR command to get a list of supported device URIs and schemes. +.TP 5 +-D "info" +.br +Provides a textual description of the destination. +.TP 5 +-E +.br +Enables the destination and accepts jobs; this is the same as running the +\fIcupsaccept(8)\fR and \fIcupsenable(8)\fR programs on the destination. +.TP 5 +-L "location" +.br +Provides a textual location of the destination. +.TP 5 +-P ppd-file +.br +Specifies a PostScript Printer Description file to use with the +printer. If specified, this option overrides the \fI-i\fR option +(interface script). +.SH COMPATIBILITY +Unlike the System V printing system, CUPS allows printer names to +contain any printable character except SPACE, TAB, "/", or "#". +Also, printer and class names are \fInot\fR case-sensitive. +Finally, the CUPS version of \fIlpadmin\fR may ask the user for +an access password depending on the printing system +configuration. This differs from the System V version which +requires the root user to execute this command. +.SH LIMITATIONS +The CUPS version of \fIlpadmin\fR does not support all of the +System V or Solaris printing system configuration options. +.SH SEE ALSO +\fIcupsaccept(8)\fR, \fIcupsenable(8)\fR, \fIlpinfo(8)\fR, +\fIlpoptions(1)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: lpadmin.man 9762 2011-05-11 05:30:50Z mike $". +.\" diff --git a/man/lpc.man b/man/lpc.man new file mode 100644 index 0000000000..8e5c7101aa --- /dev/null +++ b/man/lpc.man @@ -0,0 +1,71 @@ +.\" +.\" "$Id: lpc.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" lpc man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH lpc 8 "CUPS" "3 November 2008" "Apple Inc." +.SH NAME +lpc \- line printer control program +.SH SYNOPSIS +.B lpc +[ +.I command +[ +.I parameter(s) +] ] +.SH DESCRIPTION +\fIlpc\fR provides limited control over printer and class queues +provided by CUPS. It can also be used to query the state of +queues. +.LP +If no command is specified on the command-line, \fRlpc\fR will +display a prompt and accept commands from the standard input. +.SH COMMANDS +The \fIlpc\fR program accepts a subset of commands accepted by +the Berkeley \fIlpc\fR program of the same name: +.TP 5 +exit +.br +Exits the command interpreter. +.TP 5 +help [command] +.TP 5 +? [command] +.br +Displays a short help message. +.TP 5 +quit +.br +Exits the command interpreter. +.TP 5 +status [queue] +.br +Displays the status of one or more printer or class queues. +.SH LIMITATIONS +Since \fIlpc\fR is geared towards the Berkeley printing system, +it is impossible to use \fIlpc\fR to configure printer or class +queues provided by CUPS. To configure printer or class queues +you must use the \fIlpadmin(8)\fR command or another +CUPS-compatible client with that functionality. +.SH COMPATIBILITY +The CUPS version of \fIlpc\fR does not implement all of the +standard Berkeley or LPRng commands. +.SH SEE ALSO +\fIcancel(1)\fR, \fIcupsaccept(8)\fR, \fIcupsenable(8)\fR, +\fIlp(1)\fR, \fIlpr(1)\fR, \fIlprm(1)\fR, \fIlpstat(1)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: lpc.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/lpinfo.man b/man/lpinfo.man new file mode 100644 index 0000000000..42bda480fd --- /dev/null +++ b/man/lpinfo.man @@ -0,0 +1,115 @@ +.\" +.\" "$Id: lpinfo.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" lpinfo man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH lpinfo 8 "CUPS" "5 December 2008" "Apple Inc." +.SH NAME +lpinfo \- show available devices or drivers +.SH SYNOPSIS +.B lpinfo +[ -E ] [ -U +.I username +] [ -h +.I server[:port] +] [ -l ] [ --device-id +.I device-id-string +] [ --exclude-schemes +.I scheme-list +] [ --include-schemes +.I scheme-list +] [ --language +.I locale +] [ --make-and-model +.I name +] [ --product +.I name +] -m +.br +.B lpinfo +[ -E ] [ -U +.I username +] [ -h +.I server[:port] +] [ -l ] [ --exclude-schemes +.I scheme-list +] [ --include-schemes +.I scheme-list +] [ --timeout +.I seconds +] -v +.SH DESCRIPTION +\fIlpinfo\fR lists the available devices or drivers known to the +CUPS server. The first form (\fI-m\fR) lists the available +drivers, while the second form (\fI-v\fR) lists the available +devices. +.SH OPTIONS +\fIlpinfo\fR accepts the following options: +.TP 5 +-E +.br +Forces encryption when connecting to the server. +.TP 5 +-U username +.br +Sets the username to use when connecting to the server. +.TP 5 +-h server[:port] +.br +Selects an alternate server. +.TP 5 +-l +.br +Shows a "long" listing of devices or drivers. +.TP 5 +--device-id device-id-string +.br +Specifies the IEEE-1284 device ID to match when listing drivers with the +\fI-m\fR option. +.TP 5 +--exclude-schemes scheme-list +.br +Specifies a comma-separated list of device or PPD schemes that should be +excluded from the results. Static PPD files use the "file" scheme. +.TP 5 +--include-schemes scheme-list +.br +Specifies a comma-separated list of device or PPD schemes that should be +included in the results. Static PPD files use the "file" scheme. +.TP 5 +--language locale +.br +Specifies the language to match when listing drivers with the \fI-m\fR option. +.TP 5 +--make-and-model name +.br +Specifies the make and model to match when listing drivers with the \fI-m\fR +option. +.TP 5 +--product name +.br +Specifies the product to match when listing drivers with the \fI-m\fR option. +.TP 5 +--timeout seconds +.br +Specifies the timeout when listing devices with the \fI-v\fR option. +.SH COMPATIBILITY +The \fIlpinfo\fR command is unique to CUPS. +.SH SEE ALSO +\fIlpadmin(8)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: lpinfo.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/lpmove.man b/man/lpmove.man new file mode 100644 index 0000000000..b5688f4161 --- /dev/null +++ b/man/lpmove.man @@ -0,0 +1,66 @@ +.\" +.\" "$Id: lpmove.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" lpmove man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH lpmove 8 "CUPS" "12 February 2006" "Apple Inc." +.SH NAME +lpmove \- move a job or all jobs to a new destination +.SH SYNOPSIS +.B lpmove +[ -E ] [ -h +.I server[:port] +] [ -U +.I username +] +.I job destination +.br +.B lpmove +[ -E ] [ -h +.I server[:port] +] [ -U +.I username +] +.I source destination +.SH DESCRIPTION +\fBlpmove\fR moves the specified \fIjob\fR or all jobs from +\fIsource\fR to \fIdestination\fR. \fIjob\fR can be the job ID +number or the old destination and job ID: +.br +.nf + + lpmove 123 newprinter + lpmove oldprinter-123 newprinter +.fi +.SH OPTIONS +The \fIlpmove\fR command supports the following options: +.TP 5 +-E +.br +Forces encryption when connecting to the server. +.TP 5 +-U username +.br +Specifies an alternate username. +.TP 5 +-h server[:port] +.br +Specifies an alternate server. +.SH SEE ALSO +\fIcancel(1)\fR, \fIlp(1)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: lpmove.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/lpoptions.man.in b/man/lpoptions.man.in new file mode 100644 index 0000000000..3573ba0ee6 --- /dev/null +++ b/man/lpoptions.man.in @@ -0,0 +1,135 @@ +.\" +.\" "$Id$" +.\" +.\" lpoptions man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH lpoptions 1 "CUPS" "29 August 2008" "Apple Inc." +.SH NAME +lpoptions \- display or set printer options and defaults +.SH SYNOPSIS +.B lpoptions +[ -E ] [ -U +.I username +] [ -h +.I server[:port] +] -d +.I destination[/instance] +[ -o +.I option[=value] +] ... [ -o +.I option[=value] +] +.br +.B lpoptions +[ -E ] [ -U +.I username +] [ -h +.I server[:port] +] [ -p +.I destination[/instance] +] -l +.br +.B lpoptions +[ -E ] [ -U +.I username +] [ -h +.I server[:port] +] [ -o +.I option[=value] +] ... [ -o +.I option[=value] +] [ -p +.I destination[/instance] +] -r +.I option +.br +.B lpoptions +[ -E ] [ -U +.I username +] [ -h +.I server[:port] +] -x +.I destination[/instance] +.SH DESCRIPTION +\fIlpoptions\fR displays or sets printer options and defaults. +\fIlpoptions\fR shows the default printer options when run with no +arguments. Other options include: +.TP 5 +-E +.br +Enables encryption when communicating with the CUPS server. +.TP 5 +-U username +.br +Uses an alternate username. +.TP 5 +-d destination[/instance] +.br +Sets the user default printer to \fIdestination\fR. If \fIinstance\fR +is supplied then that particular instance is used. This option +overrides the system default printer for the current user. +.TP 5 +-h server[:port] +.br +Uses an alternate server. +.TP 5 +-l +.br +Lists the printer specific options and their current settings. +.TP 5 +-o option[=value] +.br +Specifies a new option for the named destination. +.TP 5 +-p destination[/instance] +.br +Sets the destination and instance, if specified, for any options +that follow. If the named instance does not exist then it is +created. +.TP 5 +-r option +.br +Removes the specified option for the named destination. +.TP 5 +-x destination[/instance] +.br +Removes the options for the named destination and instance, if +specified. If the named instance does not exist then this does +nothing. +.LP +If no options are specified using the \fI-o\fR option, then the +current options for the named printer are reported on the +standard output. +.LP +Options set with the \fIlpoptions\fR command are used by the +\fIlp(1)\fR and \fIlpr(1)\fR commands when submitting jobs. +.SH ROOT ACCOUNT OPTIONS +When run by the root user, \fIlpoptions\fR gets and sets default +options and instances for \fIall users\fR in the +@CUPS_SERVERROOT@/lpoptions file. +.SH COMPATIBILITY +The \fIlpoptions\fR command is unique to CUPS. +.SH FILES +~/.cups/lpoptions - user defaults and instances created by non-root +users. +.br +@CUPS_SERVERROOT@/lpoptions - system-wide defaults and instances +created by the root user. +.SH SEE ALSO +\fIcancel(1)\fR, \fIlp(1)\fR, \fIlpadmin(8)\fR, \fIlpr(1)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id$". +.\" diff --git a/man/lppasswd.man b/man/lppasswd.man new file mode 100644 index 0000000000..87ade67b86 --- /dev/null +++ b/man/lppasswd.man @@ -0,0 +1,68 @@ +.\" +.\" "$Id: lppasswd.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" lpadmin man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH lppasswd 1 "CUPS" "22 February 2008" "Apple Inc." +.SH NAME +lppasswd \- add, change, or delete digest passwords. +.SH SYNOPSIS +.B lppasswd +[ +.I username +] +.br +.B lppasswd +-a [ -g +.I groupname +] +.I username +.br +.B lppasswd +-x +.I username +.SH DESCRIPTION +\fIlppasswd\fR adds, changes, or deletes passwords in the CUPS +digest password file, \fIpasswd.md5\fR. When run by a normal +user, \fIlppasswd\fR will prompt for the old and new passwords. +When run by the super-user, \fIlppasswd\fR can add new accounts +(\fI-a username\fR), change existing accounts (\fIusername\fR), +or delete accounts (\fI-x username\fR) in the digest password +file. Digest usernames do not have to match local UNIX usernames. +.SH OPTIONS +\fIlppasswd\fR supports the following options: +.TP 5 +-g groupname +.br +Specifies a group other than the default system group. +.SH SECURITY ISSUES +By default, the \fIlppasswd\fR program is not installed to allow ordinary +users to change their passwords. To enable this, the \fIlppasswd\fR command +must be made setuid to root with the command: +.br +.nf +chmod u+s lppasswd +.fi +.PP +While every attempt has been made to make \fIlppasswd\fR secure against +exploits that could grant super-user privileges to unprivileged users, +paranoid system administrators may wish to use Basic authentication with +accounts managed by PAM instead. +.SH SEE ALSO +\fIlp(1)\fR, \fIlpr(1)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: lppasswd.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/lpq.man b/man/lpq.man new file mode 100644 index 0000000000..502d45a084 --- /dev/null +++ b/man/lpq.man @@ -0,0 +1,72 @@ +.\" +.\" "$Id: lpq.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" lpq man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH lpq 1 "CUPS" "16 June 2008" "Apple Inc." +.SH NAME +lpq \- show printer queue status +.SH SYNOPSIS +.B lpq +[ -E ] [ -U +.I username +] [ -h +.I server[:port] +] [ -P +.I destination[/instance] +] [ -a ] [ -l ] [ +.I +interval +] +.SH DESCRIPTION +\fIlpq\fR shows the current print queue status on the named +printer. Jobs queued on the default destination will be shown if +no printer or class is specified on the command-line. +.LP +The \fI+interval\fR option allows you to continuously report the +jobs in the queue until the queue is empty; the list of jobs is +shown once every \fIinterval\fR seconds. +.SH OPTIONS +\fIlpq\fR supports the following options: +.TP 5 +-E +.br +Forces encryption when connecting to the server. +.TP 5 +-P destination[/instance] +.br +Specifies an alternate printer or class name. +.TP 5 +-U username +.br +Specifies an alternate username. +.TP 5 +-a +.br +Reports jobs on all printers. +.TP 5 +-h server[:port] +.br +Specifies an alternate server. +.TP 5 +-l +.br +Requests a more verbose (long) reporting format. +.SH SEE ALSO +\fIcancel(1)\fR, \fIlp(1)\fR, \fIlpr(1)\fR, \fIlprm(1)\fR, +\fIlpstat(1)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: lpq.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/lpr.man b/man/lpr.man new file mode 100644 index 0000000000..b60a265e43 --- /dev/null +++ b/man/lpr.man @@ -0,0 +1,122 @@ +.\" +.\" "$Id: lpr.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" lpr man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH lpr 1 "CUPS" "29 August 2008" "Apple Inc." +.SH NAME +lpr \- print files +.SH SYNOPSIS +.B lpr +[ -E ] [ -H +.I server[:port] +] [ -U +.I username +] [ -P +.I destination[/instance] +] [ -# +.I num-copies +[ -h ] [ -l ] [ -m ] [ -o +.I option[=value] +] [ -p] [ -q ] [ -r ] [ -C/J/T +.I title +] [ +.I file(s) +] +.SH DESCRIPTION +\fIlpr\fR submits files for printing. Files named on the command +line are sent to the named printer (or the default destination if no +destination is specified). If no files are listed on the command-line, +\fIlpr\fR reads the print file from the standard input. +.SH THE DEFAULT DESTINATION +CUPS provides many ways to set the default destination. The "LPDEST" and +"PRINTER" environment variables are consulted first. If neither are set, +the current default set using the \fIlpoptions(1)\fR command is used, +followed by the default set using the \fIlpadmin(8)\fR command. +.SH OPTIONS +The following options are recognized by \fIlpr\fR: +.TP 5 +-E +.br +Forces encryption when connecting to the server. +.TP 5 +-H server[:port] +.br +Specifies an alternate server. +.TP 5 +-C "name" +.TP 5 +-J "name" +.TP 5 +-T "name" +.br +Sets the job name. +.TP 5 +-P destination[/instance] +.br +Prints files to the named printer. +.TP 5 +-U username +.br +Specifies an alternate username. +.TP 5 +-# \fIcopies\fR +.br +Sets the number of copies to print from 1 to 100. +.TP 5 +-h +.br +Disables banner printing. This option is equivalent to "-o +job-sheets=none". +.TP 5 +-l +.br +Specifies that the print file is already formatted for the +destination and should be sent without filtering. This option is +equivalent to "-o raw". +.TP 5 +-m +.br +Send an email on job completion. +.TP 5 +-o option[=value] +.br +Sets a job option. +.TP 5 +-p +.br +Specifies that the print file should be formatted with a shaded +header with the date, time, job name, and page number. This +option is equivalent to "-o prettyprint" and is only useful when +printing text files. +.TP 5 +-q +.br +Hold job for printing. +.TP 5 +-r +.br +Specifies that the named print files should be deleted after +printing them. +.SH COMPATIBILITY +The "c", "d", "f", "g", "i", "n", "t", "v", and "w" options +are not supported by CUPS and produce a warning message if used. +.SH SEE ALSO +\fIcancel(1)\fR, \fIlp(1)\fR, \fIlpadmin(8)\fR, \fIlpoptions(1)\fR, +\fIlpq(1)\fR, \fIlprm(1)\fR, \fIlpstat(1)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: lpr.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/lprm.man b/man/lprm.man new file mode 100644 index 0000000000..d64ac03da6 --- /dev/null +++ b/man/lprm.man @@ -0,0 +1,65 @@ +.\" +.\" "$Id: lprm.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" lprm man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH lprm 1 "CUPS" "28 August 2009" "Apple Inc." +.SH NAME +lprm \- cancel print jobs +.SH SYNOPSIS +.B lprm +[ -E ] [ -U +.I username +] [ -h +.I server[:port] +] [ -P +.I destination[/instance] +] [ - ] [ +.I job ID(s) +] +.SH DESCRIPTION +\fIlprm\fR cancels print jobs that have been queued for printing. +If no arguments are supplied, the current job on the default +destination is cancelled. You can specify one or more job ID +numbers to cancel those jobs or use the \fI-\fR option to cancel +all jobs. +.SH OPTIONS +The \fIlprm\fR command supports the following options: +.TP 5 +-E +.br +Forces encryption when connecting to the server. +.TP 5 +-P destination[/instance] +.br +Specifies the destination printer or class. +.TP 5 +-U username +.br +Specifies an alternate username. +.TP 5 +-h server[:port] +.br +Specifies an alternate server. +.SH COMPATIBILITY +The CUPS version of \fIlprm\fR is compatible with the standard +Berkeley \fIlprm\fR command. +.SH SEE ALSO +\fIcancel(1)\fR, \fIlp(1)\fR, \fIlpq(1)\fR, \fIlpr(1)\fR, +\fIlpstat(1)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: lprm.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/lpstat.man b/man/lpstat.man new file mode 100644 index 0000000000..c51c44ad9b --- /dev/null +++ b/man/lpstat.man @@ -0,0 +1,143 @@ +.\" +.\" "$Id: lpstat.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" lpstat man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH lpstat 1 "CUPS" "10 September 2008" "Apple Inc." +.SH NAME +lpstat \- print cups status information +.SH SYNOPSIS +.B lpstat +[ -E ] [ -H ] [ -U +.I username +] [ -h +.I hostname[:port] +] [ -l ] [ -W +.I which-jobs +] [ -a [ +.I destination(s) +] ] [ -c [ +.I class(es) +] ] [ -d ] [ -o [ +.I destination(s) +] ] [ -p [ +.I printer(s) +] ] [ -r ] [ -R ] [ -s ] [ -t ] [ -u [ +.I user(s) +] ] [ -v [ +.I printer(s) +] ] +.SH DESCRIPTION +\fIlpstat\fR displays status information about the current +classes, jobs, and printers. When run with no arguments, +\fIlpstat\fR will list jobs queued by the current user. +.SH OPTIONS +The \fIlpstat\fR command supports the following options: +.TP 5 +-E +.br +Forces encryption when connecting to the server. +.TP 5 +-H +.br +Shows the server hostname and port. +.TP 5 +-R +.br +Shows the ranking of print jobs. +.TP 5 +-U username +.br +Specifies an alternate username. +.TP 5 +-W which-jobs +.br +Specifies which jobs to show, \fIcompleted\fR or +\fInot-completed\fR (the default). This option \fImust\fR appear +before the \fI-o\fR option and/or any printer names, otherwise +the default (not-completed) value will be used in the request to +the scheduler. +.TP 5 +-a [printer(s)] +.br +Shows the accepting state of printer queues. If no printers are +specified then all printers are listed. +.TP 5 +-c [class(es)] +.br +Shows the printer classes and the printers that belong to them. +If no classes are specified then all classes are listed. +.TP 5 +-d +.br +Shows the current default destination. +.TP 5 +-h server[:port] +.br +Specifies an alternate server. +.TP 5 +-l +.br +Shows a long listing of printers, classes, or jobs. +.TP 5 +-o [destination(s)] +.br +Shows the jobs queue on the specified destinations. If no destinations are +specified all jobs are shown. +.TP 5 +-p [printer(s)] +.br +Shows the printers and whether or not they are enabled for printing. If +no printers are specified then all printers are listed. +.TP 5 +-r +.br +Shows whether the CUPS server is running. +.TP 5 +-s +.br +Shows a status summary, including the default destination, a +list of classes and their member printers, and a list of printers and +their associated devices. This is equivalent to using the "-d", "-c", +and "-v" options. +.TP 5 +-t +.br +Shows all status information. This is equivalent to using the "-r", +"-d", "-c", "-v", "-a", "-p", and "-o" options. +.TP 5 +-u [user(s)] +.br +Shows a list of print jobs queued by the specified users. If no users +are specified, lists the jobs queued by the current user. +.TP 5 +-v [printer(s)] +.br +Shows the printers and what device they are attached to. If no printers +are specified then all printers are listed. +.SH COMPATIBILITY +Unlike the System V printing system, CUPS allows printer names to +contain any printable character except SPACE, TAB, "/", and "#". +Also, printer and class names are \fInot\fR case-sensitive. +.LP +The "-h", "-E", "-U", and "-W" options are unique to CUPS. +.LP +The Solaris "-f", "-P", and "-S" options are silently ignored. +.SH SEE ALSO +\fIcancel(1)\fR, \fIlp(1)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: lpstat.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/mailto.conf.man b/man/mailto.conf.man new file mode 100644 index 0000000000..7f9e30d5ed --- /dev/null +++ b/man/mailto.conf.man @@ -0,0 +1,60 @@ +.\" +.\" "$Id: mailto.conf.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" mailto.conf man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH mailto.conf 5 "CUPS" "12 July 2006" "Apple Inc." +.SH NAME +mailto.conf \- configuration file for cups email notifier +.SH DESCRIPTION +The \fImailto.conf\fR file defines the local mail server and +email notification preferences for CUPS. +.LP +Each line in the file can be a configuration directive, a blank line, +or a comment. Comment lines start with the # character. +.SH DIRECTIVES +.TP 5 +Cc \fIcc-address@domain.com\fR +.br +Specifies an additional recipient for all email notifications. +.TP 5 +From \fIfrom-address@domain.com\fR +.br +Specifies the sender of email notifications. +.TP 5 +Sendmail \fIsendmail command and options\fR +.br +Specifies the sendmail command to use when sending email +notifications. Only one \fISendmail\fR or \fISMTPServer\fR line +may be present in the \fImailto.conf\fR file. If multiple lines +are present, only the last one is used. +.TP 5 +SMTPServer \fIservername\fR +.br +Specifies a SMTP server to send email notifications to. Only one +\fISendmail\fR or \fISMTPServer\fR line may be present in the +\fImailto.conf\fR file. If multiple lines are present, only the +last one is used. +.TP 5 +Subject \fIsubject-prefix\fR +.br +Specifies a prefix string for the subject line of an email notification. +.SH SEE ALSO +\fIclasses.conf(5)\fR, \fIcupsd(8)\fR, \fIcupsd.conf(5)\fR, +\fImime.convs(5)\fR, \fImime.types(5)\fR, \fIprinters.conf(5)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: mailto.conf.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/mantohtml.c b/man/mantohtml.c new file mode 100644 index 0000000000..13e7336977 --- /dev/null +++ b/man/mantohtml.c @@ -0,0 +1,720 @@ +/* + * "$Id$" + * + * Man page to HTML conversion program. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 2004-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Convert a man page to HTML. + * putc_entity() - Put a single character, using entities as needed. + * strmove() - Move characters within a string. + */ + +/* + * Include necessary headers. + */ + +#include +#include + + +/* + * Local functions... + */ + +static void putc_entity(int ch, FILE *fp); +static void strmove(char *d, const char *s); + + +/* + * 'main()' - Convert a man page to HTML. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + FILE *infile, /* Input file */ + *outfile; /* Output file */ + char line[1024], /* Line from file */ + *lineptr, /* Pointer into line */ + *endptr, /* Pointer to end of current */ + endchar, /* End character */ + *paren, /* Pointer to parenthesis */ + name[1024]; /* Man page name */ + int section, /* Man page section */ + pre, /* Preformatted */ + font, /* Current font */ + blist, /* In a bullet list? */ + list, /* In a list? */ + linenum; /* Current line number */ + const char *post; /* Text to add after the current line */ + static const char /* Start/end tags for fonts */ + * const start_fonts[] = { "", "", "" }, + * const end_fonts[] = { "", "", "" }; + + /* + * Check arguments... + */ + + if (argc > 3) + { + fputs("Usage: mantohtml [filename.man [filename.html]]\n", stderr); + return (1); + } + + /* + * Open files as needed... + */ + + if (argc > 1) + { + if ((infile = fopen(argv[1], "r")) == NULL) + { + perror(argv[1]); + return (1); + } + } + else + infile = stdin; + + if (argc > 2) + { + if ((outfile = fopen(argv[2], "w")) == NULL) + { + perror(argv[2]); + fclose(infile); + return (1); + } + } + else + outfile = stdout; + + /* + * Read from input and write the output... + */ + + fputs("\n" + "\n" + "\n" + "\n" + "\t\n", outfile); + + blist = 0; + font = 0; + list = 0; + linenum = 0; + pre = 0; + post = NULL; + section = -1; + + while (fgets(line, sizeof(line), infile)) + { + linenum ++; + + if (line[0] == '.') + { + /* + * Strip leading whitespace... + */ + + while (line[1] == ' ' || line[1] == '\t') + strmove(line + 1, line + 2); + + /* + * Process man page commands... + */ + + if (!strncmp(line, ".TH ", 4) && section < 0) + { + /* + * Grab man page title... + */ + + sscanf(line + 4, "%s%d", name, §ion); + + fprintf(outfile, + "\t%s(%d)\n" + "\n" + "\n" + "

%s(%d)

\n" + "%s", + name, section, name, section, start_fonts[font]); + } + else if (section < 0) + continue; + else if (!strncmp(line, ".SH ", 4) || !strncmp(line, ".SS ", 4)) + { + /* + * Grab heading... + */ + + int first = 1; + + fputs(end_fonts[font], outfile); + + if (blist) + { + fputs("\n\n", outfile); + blist = 0; + } + + if (list) + { + if (list == 1) + fputs("\n", outfile); + else if (list) + fputs("\n", outfile); + + fputs("\n", outfile); + list = 0; + } + + line[strlen(line) - 1] = '\0'; /* Strip LF */ + + if (line[2] == 'H') + fputs("

", outfile); + + for (lineptr = line + 4; *lineptr; lineptr ++) + if (*lineptr == '\"') + continue; + else if (*lineptr == ' ') + { + putc_entity(' ', outfile); + + first = 1; + } + else + { + if (first) + putc_entity(*lineptr, outfile); + else + putc_entity(tolower(*lineptr), outfile); + + first = 0; + } + + if (line[2] == 'H') + fprintf(outfile, "

\n%s", start_fonts[font]); + else + fprintf(outfile, "\n%s", start_fonts[font]); + } + else if (!strncmp(line, ".LP", 3) || !strncmp(line, ".PP", 3)) + { + /* + * New paragraph... + */ + + fputs(end_fonts[font], outfile); + + if (blist) + { + fputs("\n\n", outfile); + blist = 0; + } + + if (list) + { + if (list == 1) + fputs("\n", outfile); + else if (list) + fputs("\n", outfile); + + fputs("\n", outfile); + list = 0; + } + + fputs("

", outfile); + font = 0; + } + else if (!strncmp(line, ".TP ", 4)) + { + /* + * Grab list... + */ + + fputs(end_fonts[font], outfile); + + if (blist) + { + fputs("\n\n", outfile); + blist = 0; + } + + if (!list) + fputs("

\n", outfile); + else if (list == 1) + fputs("\n", outfile); + else if (list) + fputs("\n", outfile); + + fputs("
", outfile); + list = 1; + font = 0; + } + else if (!strncmp(line, ".br", 3)) + { + /* + * Grab line break... + */ + + if (list == 1) + { + fputs("
\n
", outfile); + list = 2; + } + else if (list) + fputs("
\n
", outfile); + else + fputs("
\n", outfile); + } + else if (!strncmp(line, ".de ", 4)) + { + /* + * Define macro - ignore... + */ + + while (fgets(line, sizeof(line), infile)) + { + linenum ++; + + if (!strncmp(line, "..", 2)) + break; + } + } + else if (!strncmp(line, ".RS", 3)) + { + /* + * Indent... + */ + + fputs("
\n", outfile); + } + else if (!strncmp(line, ".RE", 3)) + { + /* + * Unindent... + */ + + fputs("
\n", outfile); + } + else if (!strncmp(line, ".ds ", 4) || !strncmp(line, ".rm ", 4) || + !strncmp(line, ".tr ", 4) || !strncmp(line, ".hy ", 4) || + !strncmp(line, ".IX ", 4) || !strncmp(line, ".PD", 3) || + !strncmp(line, ".Sp", 3)) + { + /* + * Ignore unused commands... + */ + } + else if (!strncmp(line, ".Vb", 3) || !strncmp(line, ".nf", 3)) + { + /* + * Start preformatted... + */ + + pre = 1; + fputs("
\n", outfile);
+      }
+      else if (!strncmp(line, ".Ve", 3) || !strncmp(line, ".fi", 3))
+      {
+       /*
+        * End preformatted...
+	*/
+
+        if (pre)
+	{
+          pre = 0;
+	  fputs("
\n", outfile); + } + } + else if (!strncmp(line, ".IP \\(bu", 8)) + { + /* + * Bullet list... + */ + + if (blist) + fputs("\n", outfile); + else + { + fputs("
    \n", outfile); + blist = 1; + } + + fputs("
  • ", outfile); + } + else if (!strncmp(line, ".IP ", 4)) + { + /* + * Indented paragraph... + */ + + if (blist) + { + fputs("
  • \n
\n", outfile); + blist = 0; + } + + fputs("

", outfile); + + for (lineptr = line + 4; isspace(*lineptr); lineptr ++); + + if (*lineptr == '\"') + { + strmove(line, lineptr + 1); + + if ((lineptr = strchr(line, '\"')) != NULL) + *lineptr = '\0'; + } + else + { + strmove(line, lineptr); + + if ((lineptr = strchr(line, ' ')) != NULL) + *lineptr = '\0'; + } + + /* + * Process the text as if it was in-line... + */ + + post = "\n
\n
"; + goto process_text; + } + else if (!strncmp(line, ".\\}", 3)) + { + /* + * Ignore close block... + */ + } + else if (!strncmp(line, ".ie", 3) || !strncmp(line, ".if", 3) || + !strncmp(line, ".el", 3)) + { + /* + * If/else - ignore... + */ + + if (strchr(line, '{') != NULL) + { + /* + * Skip whole block... + */ + + while (fgets(line, sizeof(line), infile)) + { + linenum ++; + + if (strchr(line, '}') != NULL) + break; + } + } + } +#if 0 + else if (!strncmp(line, ". ", 4)) + { + /* + * Grab ... + */ + } +#endif /* 0 */ + else if (!strncmp(line, ".B ", 3)) + { + /* + * Grab bold text... + */ + + fprintf(outfile, "%s%s%s", end_fonts[font], line + 3, + start_fonts[font]); + } + else if (!strncmp(line, ".I ", 3)) + { + /* + * Grab italic text... + */ + + fprintf(outfile, "%s%s%s", end_fonts[font], line + 3, + start_fonts[font]); + } + else if (strncmp(line, ".\\\"", 3)) + { + /* + * Unknown... + */ + + if ((lineptr = strchr(line, ' ')) != NULL) + *lineptr = '\0'; + else if ((lineptr = strchr(line, '\n')) != NULL) + *lineptr = '\0'; + + fprintf(stderr, "mantohtml: Unknown man page command \'%s\' on line %d!\n", + line, linenum); + } + + /* + * Skip continuation lines... + */ + + lineptr = line + strlen(line) - 2; + if (lineptr >= line && *lineptr == '\\') + { + while (fgets(line, sizeof(line), infile)) + { + linenum ++; + lineptr = line + strlen(line) - 2; + + if (lineptr < line || *lineptr != '\\') + break; + } + } + } + else + { + /* + * Process man page text... + */ + +process_text: + + for (lineptr = line; *lineptr; lineptr ++) + { + if (!strncmp(lineptr, "http://", 7)) + { + /* + * Embed URL... + */ + + for (endptr = lineptr + 7; + *endptr && !isspace(*endptr & 255); + endptr ++); + + endchar = *endptr; + *endptr = '\0'; + + fprintf(outfile, "%s", lineptr, lineptr); + *endptr = endchar; + lineptr = endptr - 1; + } + else if (!strncmp(lineptr, "\\fI", 3) && + (endptr = strstr(lineptr, "\\fR")) != NULL && + (paren = strchr(lineptr, '(')) != NULL && + paren < endptr) + { + /* + * Link to man page? + */ + + char manfile[1024], /* Man page filename */ + manurl[1024]; /* Man page URL */ + + + /* + * See if the man file is available locally... + */ + + lineptr += 3; + endchar = *paren; + *paren = '\0'; + + snprintf(manfile, sizeof(manfile), "%s.man", lineptr); + snprintf(manurl, sizeof(manurl), "man-%s.html?TOPIC=Man+Pages", + lineptr); + + *paren = endchar; + endchar = *endptr; + *endptr = '\0'; + + if (access(manfile, 0)) + { + /* + * Not a local man page, just do it italic... + */ + + fputs("", outfile); + while (*lineptr) + putc_entity(*lineptr++, outfile); + fputs("", outfile); + } + else + { + /* + * Local man page, do a link... + */ + + fprintf(outfile, "", manurl); + while (*lineptr) + putc_entity(*lineptr++, outfile); + fputs("", outfile); + } + + *endptr = endchar; + lineptr = endptr + 2; + } + else if (*lineptr == '\\') + { + lineptr ++; + if (!*lineptr) + break; + else if (isdigit(lineptr[0]) && isdigit(lineptr[1]) && + isdigit(lineptr[2])) + { + fprintf(outfile, "&#%d;", ((lineptr[0] - '0') * 8 + + lineptr[1] - '0') * 8 + + lineptr[2] - '0'); + lineptr += 2; + } + else if (*lineptr == '&') + continue; + else if (*lineptr == 's') + { + while (lineptr[1] == '-' || isdigit(lineptr[1])) + lineptr ++; + } + else if (*lineptr == '*') + { + lineptr += 2; + } + else if (*lineptr != 'f') + putc_entity(*lineptr, outfile); + else + { + lineptr ++; + if (!*lineptr) + break; + else + { + fputs(end_fonts[font], outfile); + + switch (*lineptr) + { + default : /* Regular */ + font = 0; + break; + case 'B' : /* Bold */ + case 'b' : + font = 1; + break; + case 'I' : /* Italic */ + case 'i' : + font = 2; + break; + } + + fputs(start_fonts[font], outfile); + } + } + } + else + putc_entity(*lineptr, outfile); + } + + if (post) + { + fputs(post, outfile); + post = NULL; + } + + if (list == 1) + { + fputs("\n

", outfile); + list = 2; + } + } + } + + fprintf(outfile, "%s\n", end_fonts[font]); + + if (blist) + { + fputs("\n\n", outfile); + } + + if (list) + { + if (list == 1) + fputs("\n", outfile); + else if (list) + fputs("
\n", outfile); + + fputs("
\n", outfile); + } + + fputs("\n" + "\n", outfile); + + /* + * Close files... + */ + + if (infile != stdin) + fclose(infile); + + if (outfile != stdout) + fclose(outfile); + + /* + * Return with no errors... + */ + + return (0); +} + + +/* + * 'putc_entity()' - Put a single character, using entities as needed. + */ + +static void +putc_entity(int ch, /* I - Character */ + FILE *fp) /* I - File */ +{ + if (ch == '&') + fputs("&", fp); + else if (ch == '<') + fputs("<", fp); + else + putc(ch, fp); +} + + +/* + * 'strmove()' - Move characters within a string. + */ + +static void +strmove(char *d, /* I - Destination */ + const char *s) /* I - Source */ +{ + while (*s) + *d++ = *s++; + + *d = '\0'; +} + + +/* + * End of "$Id$". + */ diff --git a/man/mime.convs.man b/man/mime.convs.man new file mode 100644 index 0000000000..d55cad47b1 --- /dev/null +++ b/man/mime.convs.man @@ -0,0 +1,46 @@ +.\" +.\" "$Id: mime.convs.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" mime.convs man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH mime.convs 5 "CUPS" "20 March 2006" "Apple Inc." +.SH NAME +mime.convs \- mime type conversion file for cups +.SH DESCRIPTION +The \fImime.convs\fR file defines the filters that are available +for converting files from one format to another. The standard +filters support text, PDF, PostScript, HP-GL/2, and many types of +image files. +.LP +Additional filters can be added to the \fImime.convs\fR file or +(preferably) to other files in the CUPS configuration directory. +.LP +Each line in the \fImime.convs\fR file is a comment, blank, or filter +line. Comment lines start with the # character. Filter lines specify +the source and destination MIME types along with a relative cost +associated with the filter and the filter to run: +.br +.nf + +super/type super/type cost filter +application/postscript application/vnd.cups-raster 50 pstoraster +.fi +.SH SEE ALSO +\fIclasses.conf(5)\fR, \fIcupsd(8)\fR, \fIcupsd.conf(5)\fR, +\fImime.types(5)\fR, \fIprinters.conf(5)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: mime.convs.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/mime.types.man b/man/mime.types.man new file mode 100644 index 0000000000..9c695660f9 --- /dev/null +++ b/man/mime.types.man @@ -0,0 +1,115 @@ +.\" +.\" "$Id: mime.types.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" mime.types man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH mime.types 5 "CUPS" "16 May 2009" "Apple Inc." +.SH NAME +mime.types \- mime type description file for cups +.SH DESCRIPTION +The \fImime.types\fR file defines the recognized file types. +.LP +Additional file types can be added to \fImime.types\fR or +(preferably) in additional files in the CUPS configuration +directory with the extension ".types". +.LP +Each line in the \fImime.types\fR file is a comment, blank, or +rule line. Comment lines start with the # character. Rule lines +start with the MIME type name and are optionally followed by a +series of file recognition rules that are used to automatically +identify print and web files: +.br +.nf + + super/type rule [ ... ruleN] +.fi +MIME type names are case-insensitive and are sorted in ascending alphanumeric +order for the purposes of matching. See the "TYPE MATCHING AND PRIORITY" +section for more information. +.LP +The rules may be grouped using parenthesis, joined using "+" for a +logical AND and "," or whitespace for a logical OR, and negated using +"!". +.SH RULES +Rules take two forms - a filename extension by itself and functions with test +values inside parenthesis. The following functions are available: +.TP 5 +match("pattern") +.br +Pattern match on filename +.TP 5 +ascii(offset,length) +.br +True if bytes are valid printable ASCII (CR, NL, TAB, BS, 32-126) +.TP 5 +printable(offset,length) +.br +True if bytes are printable 8-bit chars (CR, NL, TAB, BS, 32-126, 128-254) +.TP 5 +priority(number) +.br +Specifies the relative priority of this MIME type. The default priority is 100. +Larger values have higher priority while smaller values have lower priority. +.TP 5 +string(offset,"string") +.br +True if bytes are identical to string +.TP 5 +istring(offset,"string") +.br +True if a case-insensitive comparison of the bytes is identical +.TP 5 +char(offset,value) +.br +True if byte is identical +.TP 5 +short(offset,value) +.br +True if 16-bit integer is identical +.TP 5 +int(offset,value) +.br +True if 32-bit integer is identical +.TP 5 +locale("string") +.br +True if current locale matches string +.TP 5 +contains(offset,range,"string") +.br +True if the range contains the string +.SH STRING CONSTANTS +String constants can be specified inside quotes ("") for strings +containing whitespace and angle brackets (<>) for hexadecimal +strings. +.SH TYPE MATCHING AND PRIORITY +When CUPS needs to determine the MIME type of a given file, it checks every +MIME type defined in the .types files. When two types have the same matching +rules, the type chosen will depend on the type name and priority, with higher- +priority types being used over lower-priority ones. If the types have the same +priority, the type names are sorted alphanumerically in ascending order and the +first type is chosen. +.LP +For example, if two types "text/bar" and "text/foo" are defined as matching the +extension "doc", normally the type "text/bar" will be chosen since its name is +alphanumerically smaller than "text/foo". However, if "text/foo" also defines a +higher priority than "text/bar", "text/foo" will be chosen instead. +.SH SEE ALSO +\fIclasses.conf(5)\fR, \fIcupsd(8)\fR, \fIcupsd.conf(5)\fR, +\fImime.convs(5)\fR, \fIprinters.conf(5)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: mime.types.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/notifier.man b/man/notifier.man new file mode 100644 index 0000000000..498dfd59a9 --- /dev/null +++ b/man/notifier.man @@ -0,0 +1,157 @@ +.\" +.\" "$Id: notifier.man 8999 2010-02-24 01:01:04Z mike $" +.\" +.\" notifier man page for CUPS. +.\" +.\" Copyright 2007-2009 by Apple Inc. +.\" Copyright 1997-2007 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH notifier 7 "CUPS" "12 May 2009" "Apple Inc." +.SH NAME +notifier \- cups notification interface +.SH SYNOPSIS +.B notifier +.I recipient +[ +.I user-data +] +.SH DESCRIPTION +The CUPS notifier interface provides a standard method for adding support for +new event notification methods to CUPS. Each notifier delivers one or more IPP +events from the standard input to the specified recipient. +.LP +Notifiers \fBmust\fR read IPP messages from the standard input using the +ippNew and ippReadFile functions and exit on error. Notifiers are encouraged to +exit after a suitable period of inactivity, however they may exit after reading +the first message or stay running until an error is seen. +.SH LOG MESSAGES +Messages sent to stderr are generally logged to the current \fIErrorLog\fR. +Each line begins with a standard prefix: + +.TP 5 +ALERT: message +.br +Sets the printer-state-message attribute and adds the specified +message to the current \fIErrorLog\fR using the "alert" log level. + +.TP 5 +CRIT: message +.br +Sets the printer-state-message attribute and adds the specified +message to the current \fIErrorLog\fR using the "critical" log level. + +.TP 5 +DEBUG: message +.br +Sets the printer-state-message attribute and adds the specified +message to the current \fIErrorLog\fR using the "debug" log level. + +.TP 5 +DEBUG2: message +.br +Sets the printer-state-message attribute and adds the specified +message to the current \fIErrorLog\fR using the "debug2" log level. + +.TP 5 +EMERG: message +.br +Sets the printer-state-message attribute and adds the specified +message to the current \fIErrorLog\fR using the "emergency" log level. + +.TP 5 +ERROR: message +.br +Sets the printer-state-message attribute and adds the specified +message to the current \fIErrorLog\fR using the "error" log level. + +.TP 5 +INFO: message +.br +Sets the printer-state-message attribute. If the current \fILogLevel\fR +is set to "debug2", also adds the specified message to the +current \fIErrorLog\fR using the "info" log level. + +.TP 5 +NOTICE: message +.br +Sets the printer-state-message attribute and adds the specified +message to the current \fIErrorLog\fR using the "notice" log level. + +.TP 5 +WARNING: message +.br +Sets the printer-state-message attribute and adds the specified +message to the current \fIErrorLog\fR using the "warning" log level. + +.SH ENVIRONMENT VARIABLES +The following environment variables are defined by the CUPS server when +executing the notifier: + +.TP 5 +CHARSET +.br +The default text character set, typically utf-8. + +.TP 5 +CUPS_CACHEDIR +.br +The directory for semi-persistent cache files can be found. + +.TP 5 +CUPS_DATADIR +.br +The directory where data files can be found. + +.TP 5 +CUPS_FILETYPE +.br +The type of file being printed: "job-sheet" for a banner page and "document" +for a regular print file. + +.TP 5 +CUPS_SERVERROOT +.br +The root directory of the server. + +.TP 5 +LANG +.br +The default language locale (typically C or en). + +.TP 5 +PATH +.br +The standard execution path for external programs that may be run by +the filter. + +.TP 5 +SOFTWARE +.br +The name and version number of the server (typically CUPS/1.2). + +.TP 5 +TZ +.br +The timezone of the server. + +.TP 5 +USER +.br +The user executing the filter, typically "lp"; consult the \fIcupsd.conf(5)\fR +file for the current setting. + +.SH SEE ALSO +\fIbackend(7)\fR, \fIcupsd(8)\fR, \fIfilter(7)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2009 by Apple Inc. +.\" +.\" End of "$Id: notifier.man 8999 2010-02-24 01:01:04Z mike $". +.\" diff --git a/man/ppdc.man b/man/ppdc.man new file mode 100644 index 0000000000..b613d42998 --- /dev/null +++ b/man/ppdc.man @@ -0,0 +1,80 @@ +.\" +.\" "$Id: ppdc.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" ppdc man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2007 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH ppdc 1 "CUPS" "10 October 2008" "Apple Inc." +.SH NAME +ppdc \- cups ppd compiler +.SH SYNOPSIS +.B ppdc +[ -D +.I name[=value] +] [ -I +.I include-directory +] [ -c +.I message-catalog +] [ -d +.I output-directory +] [ -l +.I language(s) +] [-m] [-t] [ -v ] [ -z ] [ --cr ] [ --crlf ] [ --lf ] +.I source-file +.SH DESCRIPTION +\fIppdc\fR compiles PPDC source files into one or more PPD +files. +.PP +The \fI-D\fR option sets the named variable for use in the +source file. It is equivalent to using the #define directive +in the source file. +.PP +The \fI-I\fR option specifies an alternate include directory; +multiple \fI-I\fR options can be supplied to add additional +directories. +.PP +The \fI-c\fR option specifies a single message catalog file in GNU +gettext source format (filename.po) to be used for localization. +.PP +The \fI-d\fR option specifies the output directory for PPD +files. The default output directory is "ppd". +.PP +The \fI-l\fR option specifies one or more languages to use when +localizing the PPD file(s). The default language is "en" +(English). Separate multiple languages with commas, for example +"de_DE,en_UK,es_ES,es_MX,es_US,fr_CA,fr_FR,it_IT" will create PPD +files with German, UK English, Spanish (Spain, Mexico, and US), +French (France and Canada), and Italian languages in each file. +.PP +The \fI-m\fR option specifies that the output filename should be +based on the ModelName value instead of FileName or PCFilenName. +.PP +The \fI-t\fR option specifies that PPD files should be tested instead +of generated. +.PP +The \fI-v\fR option provides more verbose output, basically a +running status of which files are being loaded or written. +.PP +The \fI-z\fR option generates compressed PPD files (filename.ppd.gz). +The default is to generate uncompressed PPD files. +.PP +The \fI--cr\fR, \fI--crlf\fR, and \fI--lf\fR options specify the +line ending to use - carriage return, carriage return and line feed, +or line feed. The default is to use the line feed character alone. +.SH SEE ALSO +ppdhtml(1), ppdi(1), ppdmerge(1), ppdpo(1), ppdcfile(5) +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: ppdc.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/ppdcfile.man b/man/ppdcfile.man new file mode 100644 index 0000000000..fbd635866b --- /dev/null +++ b/man/ppdcfile.man @@ -0,0 +1,171 @@ +.\" +.\" "$Id: ppdcfile.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" ppdcfile man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2007 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH ppdcfile 5 "CUPS" "20 May 2008" "Apple Inc." +.SH NAME +ppdcfile \- cups ppd compiler source file format +.SH DESCRIPTION +The CUPS PPD compiler reads meta files that contain descriptions +of one or more PPD files to be generated by \fIppdc\fR. This man +page provides a quick reference to the supported keywords and +should be used in conjuction with the Common UNIX Printing System +Developers Manual. +.PP +The source file format is plain ASCII text that can be edited +using your favorite text editor. Comments are supported using +the C (/* ... */) and C++ (// ...) comment mechanisms. +.PP +Printer driver information can be grouped and shared using +curley braces ({ ... }); PPD files are written when a close +brace or end-of-file is seen and a PCFileName directive has been +defined. +.PP +Directives may be placed anywhere on a line and are followed by +one or more values. The following is a list of the available +directives and the values they accept: +.TP 5 +\fB#define\fR name value +.TP 5 +\fB#elif\fR {name | value} +.TP 5 +\fB#else\fR +.TP 5 +\fB#endif\fR +.TP 5 +\fB#font\fR name encoding "version" charset status +.TP 5 +\fB#if\fR {name | value} +.TP 5 +\fB#include\fR +.TP 5 +\fB#include\fR "filename" +.TP 5 +\fB#media\fR name width length +.TP 5 +\fB#media\fR "name/text" width length +.TP 5 +\fB#po\fR locale "filename.po" +.TP 5 +\fBAttribute\fR name "" value +.TP 5 +\fBAttribute\fR name keyword value +.TP 5 +\fBAttribute\fR name "keyword/text" value +.TP 5 +\fBChoice\fR name "code" +.TP 5 +\fBChoice\fR "name/text" "code" +.TP 5 +\fBColorDevice\fR boolean-value +.TP 5 +\fBColorModel\fR name colorspace colororder compression +.TP 5 +\fBColorModel\fR "name/text" colorspace colororder compression +.TP 5 +\fBColorProfile\fR resolution/mediatype gamma density matrix +.TP 5 +\fBCopyright\fR "text" +.TP 5 +\fBCustomMedia\fR name width length left bottom right top "size-code" "region-code" +.TP 5 +\fBCustomMedia\fR "name/text" width length left bottom right top "size-code" "region-code" +.TP 5 +\fBCutter\fR boolean-value +.TP 5 +\fBDarkness\fR temperature name +.TP 5 +\fBDarkness\fR temperature "name/text" +.TP 5 +\fBDriverType\fR type +.TP 5 +\fBDuplex\fR type +.TP 5 +\fBFilter\fR mime-type cost program +.TP 5 +\fBFinishing\fR name +.TP 5 +\fBFinishing\fR "name/text" +.TP 5 +\fBFont\fR * +.TP 5 +\fBFont\fR name encoding "version" charset status +.TP 5 +\fBGroup\fR name +.TP 5 +\fBGroup\fR "name/text" +.TP 5 +\fBHWMargins\fR left bottom right top +.TP 5 +\fBInputSlot\fR position name +.TP 5 +\fBInputSlot\fR position "name/text" +.TP 5 +\fBInstallable\fR name +.TP 5 +\fBInstallable\fR "name/text" +.TP 5 +\fBLocAttribute\fR name "keyword/text" value +.TP 5 +\fBManualCopies\fR boolean-value +.TP 5 +\fBManufacturer\fR "name" +.TP 5 +\fBMaxSize\fR width length +.TP 5 +\fBMediaSize\fR name +.TP 5 +\fBMediaType\fR type name +.TP 5 +\fBMediaType\fR type "name/text" +.TP 5 +\fBMinSize\fR width length +.TP 5 +\fBModelName\fR "name" +.TP 5 +\fBModelNumber\fR number +.TP 5 +\fBOption\fR name type section order +.TP 5 +\fBOption\fR "name/text" type section order +.TP 5 +\fBPCFileName\fR "filename.ppd" +.TP 5 +\fBResolution\fR colorspace bits-per-color row-count row-feed row-step name +.TP 5 +\fBResolution\fR colorspace bits-per-color row-count row-feed row-step "name/text" +.TP 5 +\fBSimpleColorProfile\fR resolution/mediatype density yellow-density red-density gamma red-adjust green-adjust blue-adjust +.TP 5 +\fBThroughput\fR pages-per-minute +.TP 5 +\fBUIConstraints\fR "*Option1 *Option2" +.TP 5 +\fBUIConstraints\fR "*Option1 Choice1 *Option2" +.TP 5 +\fBUIConstraints\fR "*Option1 *Option2 Choice2" +.TP 5 +\fBUIConstraints\fR "*Option1 Choice1 *Option2 Choice2" +.TP 5 +\fBVariablePaperSize\fR boolean-value +.TP 5 +\fBVersion\fR number +.SH SEE ALSO +ppdc(1), ppdhtml(1), ppdi(1), ppdmerge(1), ppdpo(1) +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: ppdcfile.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/ppdhtml.man b/man/ppdhtml.man new file mode 100644 index 0000000000..bebf4e1e9a --- /dev/null +++ b/man/ppdhtml.man @@ -0,0 +1,46 @@ +.\" +.\" "$Id: ppdhtml.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" ppdhtml man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2007 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH ppdhtml 1 "CUPS" "10 October 2008" "Apple Inc." +.SH NAME +ppdhtml \- cups html summary generator +.SH SYNOPSIS +.B ppdhtml +[ -D +.I name[=value] +] [ -I +.I include-directory +] +.I source-file +.SH DESCRIPTION +\fIppdhtml\fR reads a driver information file and produces a +HTML summary page that lists all of the drivers in a file and +the supported options. +.PP +The \fI-D\fR option sets the named variable for use in the +source file. It is equivalent to using the #define directive +in the source file. +.PP +The \fI-I\fR option specifies an alternate include directory; +multiple \fI-I\fR options can be supplied to add additional +directories. +.SH SEE ALSO +ppdc(1), ppdcfile(5), ppdi(1), ppdmerge(1), ppdpo(1) +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: ppdhtml.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/ppdi.man b/man/ppdi.man new file mode 100644 index 0000000000..610d91cd68 --- /dev/null +++ b/man/ppdi.man @@ -0,0 +1,46 @@ +.\" +.\" "$Id: ppdi.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" ppdi man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2007 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH ppdi 1 "CUPS" "20 May 2008" "Apple Inc." +.SH NAME +ppdi \- import ppd files +.SH SYNOPSIS +.B ppdi +[ \-I +.I include-directory +] [ \-o +.I source-file +] +.I ppd-file +[ +.I ppd-file2 ... ppd-fileN +] +.SH DESCRIPTION +\fIppdi\fR imports one or more PPD files into a PPD compiler source file. +Multiple languages of the same PPD file are merged into a single printer +definition to facilitate accurate changes for all localizations. +.PP +The \fI-o\fR option specifies the PPD source file to update. If the source +file does not exist, a new source file is created. Otherwise the existing +file is merged with the new PPD file(s) on the command-line. If no source +file is specified, the filename "ppdi.drv" is used. +.SH SEE ALSO +ppdc(1), ppdhtml(1), ppdmerge(1), ppdpo(1), ppdcfile(5) +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: ppdi.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/ppdmerge.man b/man/ppdmerge.man new file mode 100644 index 0000000000..90fee820c3 --- /dev/null +++ b/man/ppdmerge.man @@ -0,0 +1,47 @@ +.\" +.\" "$Id: ppdmerge.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" ppdmerge man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2007 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH ppdmerge 1 "CUPS" "20 May 2008" "Apple Inc." +.SH NAME +ppdmerge \- merge ppd files +.SH SYNOPSIS +.B ppdmerge +[ \-o +.I output-ppd-file +] +.I ppd-file +.I ppd-file2 +[ +.I ... ppd-fileN +] +.SH DESCRIPTION +\fIppdmerge\fR merges two or more PPD files into a single, multi-language +PPD file. +.PP +The \fI-o\fR option specifies the PPD file to create. If not specified, +the merged PPD file is written to the standard output. If the output file +already exists, the new +.SH NOTES +\fIppdmerge\fR does not check whether the merged PPD files are for the +same device. Merging of different device PPDs will yield unpredictable +results. +.SH SEE ALSO +cupsprofile(1), ppdc(1), ppdhtml(1), ppdi(1), ppdpo(1), ppdcfile(5) +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: ppdmerge.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/ppdpo.man b/man/ppdpo.man new file mode 100644 index 0000000000..a3f9ac9b27 --- /dev/null +++ b/man/ppdpo.man @@ -0,0 +1,52 @@ +.\" +.\" "$Id: ppdpo.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" ppdpo man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2007 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH ppdpo 1 "CUPS" "10 October 2008" "Apple Inc." +.SH NAME +ppdpo \- ppd message catalog generator +.SH SYNOPSIS +.B ppdpo +[ -D +.I name[=value] +] [ -I +.I include-directory +] [ -o +.I output-file +] +.I source-file +.SH DESCRIPTION +\fIppdpo\fR extracts UI strings from PPDC source files and updates either +a GNU gettext or Mac OS X strings format message catalog source file for +translation. +.PP +The \fI-D\fR option sets the named variable for use in the +source file. It is equivalent to using the #define directive +in the source file. +.PP +The \fI-I\fR option specifies an alternate include directory; +multiple \fI-I\fR options can be supplied to add additional +directories. +.PP +The \fI-o\fR option specifies the output file. The supported extensions are +".po" or ".po.gz" for GNU gettext format message catalogs and ".strings" for +Mac OS X strings files. +.SH SEE ALSO +ppdc(1), ppdhtml(1), ppdi(1), ppdmerge(1), ppdcfile(5) +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: ppdpo.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/printers.conf.man b/man/printers.conf.man new file mode 100644 index 0000000000..abda49b26f --- /dev/null +++ b/man/printers.conf.man @@ -0,0 +1,124 @@ +.\" +.\" "$Id: printers.conf.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" printers.conf man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 1997-2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH printers.conf 5 "CUPS" "29 April 2009" "Apple Inc." +.SH NAME +printers.conf \- printer configuration file for cups +.SH DESCRIPTION +The \fIprinters.conf\fR file defines the local printers that are +available. It is normally located in the \fI/etc/cups\fR directory and +is generated automatically by the \fIcupsd(8)\fR program when printers +are added or deleted. +.LP +Each line in the file can be a configuration directive, a blank line, +or a comment. Comment lines start with the # character. +.SH DIRECTIVES +.TP 5 + ... +.br +Defines a specific printer. +.TP 5 + ... +.br +Defines a default printer. +.TP 5 +Accepting Yes +.TP 5 +Accepting No +.br +Specifies whether the printer is accepting new jobs. +.TP 5 +AllowUser [ user @group ... ] +.br +Allows specific users and groups to print to the printer. +.TP 5 +DenyUser [ user @group ... ] +.br +Prevents specific users and groups from printing to the printer. +.TP 5 +DeviceURI uri +.br +Specifies the device URI for a printer. +.TP 5 +ErrorPolicy abort-job +.TP 5 +ErrorPolicy retry-current-job +.TP 5 +ErrorPolicy retry-job +.TP 5 +ErrorPolicy stop-printer +.br +Specifies the error policy for the printer. +.TP 5 +Info text +.br +Specifies human-readable text describing the printer. +.TP 5 +JobSheets banner banner +.br +Specifies the banner pages to use for the printer. +.TP 5 +KLimit number +.br +Specifies the job-k-limit value for the printer. +.TP 5 +Location text +.br +Specifies human-readable text describing the location of the printer. +.TP 5 +OpPolicy name +.br +Specifies the operation policy for the printer. +.TP 5 +PageLimit number +.br +Specifies the job-page-limit value for the printer. +.TP 5 +PortMonitor monitor +.br +Specifies the port monitor for a printer. +.TP 5 +QuotaPeriod seconds +.br +Specifies the job-quota-period value for the printer. +.TP 5 +Shared Yes +.TP 5 +Shared No +.br +Specifies whether the printer is shared. +.TP 5 +State idle +.TP 5 +State stopped +.br +Specifies the initial state of the printer (Idle or Stopped) +.TP 5 +StateMessage text +.br +Specifies the message associated with the state. +.TP 5 +StateTime seconds +.br +Specifies the date/time associated with the state. +.SH SEE ALSO +\fIclasses.conf(5)\fR, \fIcupsd(8)\fR, \fIcupsd.conf(5)\fR, +\fImime.convs(5)\fR, \fImime.types(5)\fR, +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: printers.conf.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/man/subscriptions.conf.man b/man/subscriptions.conf.man new file mode 100644 index 0000000000..4ff851cb9f --- /dev/null +++ b/man/subscriptions.conf.man @@ -0,0 +1,89 @@ +.\" +.\" "$Id: subscriptions.conf.man 9771 2011-05-12 05:21:56Z mike $" +.\" +.\" subscriptions.conf man page for CUPS. +.\" +.\" Copyright 2007-2011 by Apple Inc. +.\" Copyright 2006 by Easy Software Products. +.\" +.\" These coded instructions, statements, and computer programs are the +.\" property of Apple Inc. and are protected by Federal copyright +.\" law. Distribution and use rights are outlined in the file "LICENSE.txt" +.\" which should have been included with this file. If this file is +.\" file is missing or damaged, see the license at "http://www.cups.org/". +.\" +.TH subscriptions.conf 5 "CUPS" "30 April 2006" "Apple Inc." +.SH NAME +subscriptions.conf \- subscriptions file for cups +.SH DESCRIPTION +The \fIsubscriptions.conf\fR file defines the local subscriptions +that are active. It is normally located in the \fI/etc/cups\fR +directory and is generated automatically by the \fIcupsd(8)\fR +program when subscriptions are created, renewed, or cancelled. +.LP +Each line in the file can be a configuration directive, a blank line, +or a comment. Comment lines start with the # character. +.SH DIRECTIVES +.TP 5 + ... +.br +Defines a subscription. +.TP 5 +Events name [ ... name ] +.br +Specifies the events that are subscribed. +.TP 5 +ExpirationTime unix-time +.br +Specifies the expiration time of a subscription as a UNIX time +value in seconds since January 1st, 1970. +.TP 5 +Interval seconds +.br +Specifies the preferred time interval for event notifications in +seconds. +.TP 5 +JobId job-id +.br +Specifies the job ID associated with the subscription. +.TP 5 +LeaseDuration seconds +.br +Specifies the number of seconds that the subscription is valid +for. If 0, the subscription does not expire. +.TP 5 +NextEventId number +.br +Specifies the next notify-sequence-number to use for the +subscription. +.TP 5 +NextSubscriptionId number +.br +Specifies the next subscription-id to use. +.TP 5 +Owner username +.br +Specifies the user that owns the subscription. +.TP 5 +PrinterName printername +.br +Specifies the printer or class associated with the subscription. +.TP 5 +Recipient uri +.br +Specifies the notify-recipient-uri value for push-type notifications. +.TP 5 +UserData hex-escaped-data +.br +Specifies user data to be included in event notifications. This +is typically the "from" address in mailto: notifications. +.SH SEE ALSO +\fIclasses.conf(5)\fR, \fIcupsd(8)\fR, \fIcupsd.conf(5)\fR, +\fImime.convs(5)\fR, \fImime.types(5)\fR, \fRprinters.conf(5)\fR +.br +http://localhost:631/help +.SH COPYRIGHT +Copyright 2007-2011 by Apple Inc. +.\" +.\" End of "$Id: subscriptions.conf.man 9771 2011-05-12 05:21:56Z mike $". +.\" diff --git a/monitor/Dependencies b/monitor/Dependencies new file mode 100644 index 0000000000..594a83cf87 --- /dev/null +++ b/monitor/Dependencies @@ -0,0 +1,14 @@ +bcp.o: bcp.c ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +tbcp.o: tbcp.c ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h diff --git a/monitor/Makefile b/monitor/Makefile new file mode 100644 index 0000000000..e62ef33d46 --- /dev/null +++ b/monitor/Makefile @@ -0,0 +1,146 @@ +# +# "$Id$" +# +# Port monitor makefile for CUPS. +# +# Copyright 2007-2012 by Apple Inc. +# Copyright 2006 by Easy Software Products. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# +# This file is subject to the Apple OS-Developed Software exception. +# + +include ../Makedefs + +TARGETS = bcp tbcp + +OBJS = bcp.o tbcp.o + + +# +# Make all targets... +# + +all: $(TARGETS) + + +# +# Make library targets... +# + +libs: + + +# +# Make unit tests... +# + +unittests: + + +# +# Clean all object files... +# + +clean: + $(RM) $(OBJS) $(TARGETS) + + +# +# Update dependencies (without system header dependencies...) +# + +depend: + $(CC) -MM $(ALL_CFLAGS) $(OBJS:.o=.c) >Dependencies + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: + + +# +# Install programs... +# + +install-exec: + $(INSTALL_DIR) -m 755 $(SERVERBIN)/monitor + for file in $(TARGETS); do \ + $(INSTALL_BIN) $$file $(SERVERBIN)/monitor; \ + done + if test "x$(SYMROOT)" != "x"; then \ + $(INSTALL_DIR) $(SYMROOT); \ + for file in $(TARGETS); do \ + cp $$file $(SYMROOT); \ + done \ + fi + + +# +# Install headers... +# + +install-headers: + + +# +# Install libraries... +# + +install-libs: + + +# +# Uninstall all targets... +# + +uninstall: + for file in $(TARGETS); do \ + $(RM) $(SERVERBIN)/monitor/$$file; \ + done + -$(RMDIR) $(SERVERBIN)/monitor + -$(RMDIR) $(SERVERBIN) + + +# +# bcp +# + +bcp: bcp.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ bcp.o $(LIBS) + + +# +# tbcp +# + +tbcp: tbcp.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ tbcp.o $(LIBS) + + +# +# Dependencies... +# + +include Dependencies + + +# +# End of "$Id$". +# diff --git a/monitor/bcp.c b/monitor/bcp.c new file mode 100644 index 0000000000..2354d102de --- /dev/null +++ b/monitor/bcp.c @@ -0,0 +1,292 @@ +/* + * "$Id$" + * + * TBCP port monitor for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1993-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Main entry... + * psgets() - Get a line from a file. + * pswrite() - Write data from a file. + */ + +/* + * Include necessary headers... + */ + +#include +#include + + +/* + * Local functions... + */ + +static char *psgets(char *buf, size_t *bytes, FILE *fp); +static size_t pswrite(const char *buf, size_t bytes, FILE *fp); + + +/* + * 'main()' - Main entry... + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + FILE *fp; /* File to print */ + int copies; /* Number of copies left */ + char line[1024]; /* Line/buffer from stream/file */ + size_t linelen; /* Length of line */ + ppd_file_t *ppd; /* PPD file */ + + + /* + * Check command-line... + */ + + if (argc < 6 || argc > 7) + { + _cupsLangPrintf(stderr, + _("Usage: %s job-id user title copies options [file]"), + argv[0]); + return (1); + } + + if (argc == 6) + { + copies = 1; + fp = stdin; + } + else + { + copies = atoi(argv[4]); + fp = fopen(argv[6], "rb"); + + if (!fp) + { + perror(argv[6]); + return (1); + } + } + + /* + * Open the PPD file as needed... + */ + + ppd = ppdOpenFile(getenv("PPD")); + + /* + * Copy the print file to stdout... + */ + + while (copies > 0) + { + copies --; + + if (ppd && ppd->jcl_begin) + fputs(ppd->jcl_begin, stdout); + if (ppd && ppd->jcl_ps) + fputs(ppd->jcl_ps, stdout); + + if (!ppd || ppd->language_level == 1) + { + /* + * Use setsoftwareiomode for BCP mode... + */ + + puts("%!PS-Adobe-3.0 ExitServer"); + puts("%%Title: (BCP - Level 1)"); + puts("%%EndComments"); + puts("%%BeginExitServer: 0"); + puts("serverdict begin 0 exitserver"); + puts("%%EndExitServer"); + puts("statusdict begin"); + puts("/setsoftwareiomode known {100 setsoftwareiomode}"); + puts("end"); + puts("%EOF"); + } + else + { + /* + * Use setdevparams for BCP mode... + */ + + puts("%!PS-Adobe-3.0"); + puts("%%Title: (BCP - Level 2)"); + puts("%%EndComments"); + puts("currentsysparams"); + puts("/CurInputDevice 2 copy known {"); + puts("get"); + puts("<> setdevparams"); + puts("}{"); + puts("pop pop"); + puts("} ifelse"); + puts("%EOF"); + } + + if (ppd && ppd->jcl_end) + fputs(ppd->jcl_end, stdout); + else if (!ppd || ppd->num_filters == 0) + putchar(0x04); + + /* + * Loop until we see end-of-file... + */ + + do + { + linelen = sizeof(line); + if (psgets(line, &linelen, fp) == NULL) + break; + } + while (pswrite(line, linelen, stdout) > 0); + + fflush(stdout); + } + + return (0); +} + + +/* + * 'psgets()' - Get a line from a file. + * + * Note: + * + * This function differs from the gets() function in that it + * handles any combination of CR, LF, or CR LF to end input + * lines. + */ + +static char * /* O - String or NULL if EOF */ +psgets(char *buf, /* I - Buffer to read into */ + size_t *bytes, /* IO - Length of buffer */ + FILE *fp) /* I - File to read from */ +{ + char *bufptr; /* Pointer into buffer */ + int ch; /* Character from file */ + size_t len; /* Max length of string */ + + + len = *bytes - 1; + bufptr = buf; + ch = EOF; + + while ((bufptr - buf) < len) + { + if ((ch = getc(fp)) == EOF) + break; + + if (ch == '\r') + { + /* + * Got a CR; see if there is a LF as well... + */ + + ch = getc(fp); + + if (ch != EOF && ch != '\n') + { + ungetc(ch, fp); /* Nope, save it for later... */ + ch = '\r'; + } + else + *bufptr++ = '\r'; + break; + } + else if (ch == '\n') + break; + else + *bufptr++ = ch; + } + + /* + * Add a trailing newline if it is there... + */ + + if (ch == '\n' || ch == '\r') + { + if ((bufptr - buf) < len) + *bufptr++ = ch; + else + ungetc(ch, fp); + } + + /* + * Nul-terminate the string and return it (or NULL for EOF). + */ + + *bufptr = '\0'; + *bytes = bufptr - buf; + + if (ch == EOF && bufptr == buf) + return (NULL); + else + return (buf); +} + + +/* + * 'pswrite()' - Write data from a file. + */ + +static size_t /* O - Number of bytes written */ +pswrite(const char *buf, /* I - Buffer to write */ + size_t bytes, /* I - Bytes to write */ + FILE *fp) /* I - File to write to */ +{ + size_t count; /* Remaining bytes */ + + + for (count = bytes; count > 0; count --, buf ++) + switch (*buf) + { + case 0x04 : /* CTRL-D */ + if (bytes == 1) + { + /* + * Don't quote the last CTRL-D... + */ + + putchar(0x04); + break; + } + + case 0x01 : /* CTRL-A */ + case 0x03 : /* CTRL-C */ + case 0x05 : /* CTRL-E */ + case 0x11 : /* CTRL-Q */ + case 0x13 : /* CTRL-S */ + case 0x14 : /* CTRL-T */ + case 0x1c : /* CTRL-\ */ + if (putchar(0x01) < 0) + return (-1); + if (putchar(*buf ^ 0x40) < 0) + return (-1); + break; + + default : + if (putchar(*buf) < 0) + return (-1); + break; + } + + return (bytes); +} + + +/* + * End of "$Id$". + */ diff --git a/monitor/tbcp.c b/monitor/tbcp.c new file mode 100644 index 0000000000..250a00302d --- /dev/null +++ b/monitor/tbcp.c @@ -0,0 +1,285 @@ +/* + * "$Id$" + * + * TBCP port monitor for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1993-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Main entry... + * psgets() - Get a line from a file. + * pswrite() - Write data from a file. + */ + +/* + * Include necessary headers... + */ + +#include +#include + + +/* + * Local functions... + */ + +static char *psgets(char *buf, size_t *bytes, FILE *fp); +static size_t pswrite(const char *buf, size_t bytes, FILE *fp); + + +/* + * 'main()' - Main entry... + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + FILE *fp; /* File to print */ + int copies; /* Number of copies left */ + char line[1024]; /* Line/buffer from stream/file */ + size_t linelen; /* Length of line */ + + + /* + * Check command-line... + */ + + if (argc < 6 || argc > 7) + { + _cupsLangPrintf(stderr, + _("Usage: %s job-id user title copies options [file]"), + argv[0]); + return (1); + } + + if (argc == 6) + { + copies = 1; + fp = stdin; + } + else + { + copies = atoi(argv[4]); + fp = fopen(argv[6], "rb"); + + if (!fp) + { + perror(argv[6]); + return (1); + } + } + + /* + * Copy the print file to stdout... + */ + + while (copies > 0) + { + copies --; + + /* + * Read the first line... + */ + + linelen = sizeof(line); + if (psgets(line, &linelen, fp) == NULL) + { + fputs("ERROR: Empty print file!\n", stderr); + return (1); + } + + /* + * Handle leading PJL fun... + */ + + if (!strncmp(line, "\033%-12345X", 9) || !strncmp(line, "@PJL ", 5)) + { + /* + * Yup, we have leading PJL fun, so copy it until we hit a line + * with "ENTER LANGUAGE"... + */ + + while (strstr(line, "ENTER LANGUAGE") == NULL) + { + fwrite(line, 1, linelen, stdout); + + linelen = sizeof(line); + if (psgets(line, &linelen, fp) == NULL) + break; + } + } + else + { + /* + * No PJL stuff, just add the UEL... + */ + + fputs("\033%-12345X", stdout); + } + + /* + * Switch to TBCP mode... + */ + + fputs("\001M", stdout); + + /* + * Loop until we see end-of-file... + */ + + while (pswrite(line, linelen, stdout) > 0) + { + linelen = sizeof(line); + if (psgets(line, &linelen, fp) == NULL) + break; + } + + fflush(stdout); + } + + return (0); +} + + +/* + * 'psgets()' - Get a line from a file. + * + * Note: + * + * This function differs from the gets() function in that it + * handles any combination of CR, LF, or CR LF to end input + * lines. + */ + +static char * /* O - String or NULL if EOF */ +psgets(char *buf, /* I - Buffer to read into */ + size_t *bytes, /* IO - Length of buffer */ + FILE *fp) /* I - File to read from */ +{ + char *bufptr; /* Pointer into buffer */ + int ch; /* Character from file */ + size_t len; /* Max length of string */ + + + len = *bytes - 1; + bufptr = buf; + ch = EOF; + + while ((bufptr - buf) < len) + { + if ((ch = getc(fp)) == EOF) + break; + + if (ch == '\r') + { + /* + * Got a CR; see if there is a LF as well... + */ + + ch = getc(fp); + + if (ch != EOF && ch != '\n') + { + ungetc(ch, fp); /* Nope, save it for later... */ + ch = '\r'; + } + else + *bufptr++ = '\r'; + break; + } + else if (ch == '\n') + break; + else + *bufptr++ = ch; + } + + /* + * Add a trailing newline if it is there... + */ + + if (ch == '\n' || ch == '\r') + { + if ((bufptr - buf) < len) + *bufptr++ = ch; + else + ungetc(ch, fp); + } + + /* + * Nul-terminate the string and return it (or NULL for EOF). + */ + + *bufptr = '\0'; + *bytes = bufptr - buf; + + if (ch == EOF && bufptr == buf) + return (NULL); + else + return (buf); +} + + +/* + * 'pswrite()' - Write data from a file. + */ + +static size_t /* O - Number of bytes written */ +pswrite(const char *buf, /* I - Buffer to write */ + size_t bytes, /* I - Bytes to write */ + FILE *fp) /* I - File to write to */ +{ + size_t count; /* Remaining bytes */ + + + for (count = bytes; count > 0; count --, buf ++) + switch (*buf) + { + case 0x04 : /* CTRL-D */ + if (bytes == 1) + { + /* + * Don't quote the last CTRL-D... + */ + + putchar(0x04); + break; + } + + case 0x01 : /* CTRL-A */ + case 0x03 : /* CTRL-C */ + case 0x05 : /* CTRL-E */ + case 0x11 : /* CTRL-Q */ + case 0x13 : /* CTRL-S */ + case 0x14 : /* CTRL-T */ + case 0x1b : /* CTRL-[ (aka ESC) */ + case 0x1c : /* CTRL-\ */ + if (putchar(0x01) < 0) + return (-1); + if (putchar(*buf ^ 0x40) < 0) + return (-1); + break; + + default : + if (putchar(*buf) < 0) + return (-1); + break; + } + + return (bytes); +} + + +/* + * End of "$Id$". + */ diff --git a/notifier/Dependencies b/notifier/Dependencies new file mode 100644 index 0000000000..582d6f599c --- /dev/null +++ b/notifier/Dependencies @@ -0,0 +1,20 @@ +dbus.o: dbus.c ../cups/cups.h ../cups/file.h ../cups/versioning.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h +mailto.o: mailto.c ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +rss.o: rss.c ../cups/cups.h ../cups/file.h ../cups/versioning.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/ipp-private.h +testnotify.o: testnotify.c ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h diff --git a/notifier/Makefile b/notifier/Makefile new file mode 100644 index 0000000000..87378e35bb --- /dev/null +++ b/notifier/Makefile @@ -0,0 +1,162 @@ +# +# "$Id$" +# +# Notifier makefile for CUPS. +# +# Copyright 2007-2012 by Apple Inc. +# Copyright 1997-2007 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +include ../Makedefs + + +NOTIFIERS = $(DBUS_NOTIFIER) mailto rss +TARGETS = $(NOTIFIERS) testnotify +OBJS = dbus.o mailto.o rss.o testnotify.o + + +# +# Make all targets... +# + +all: $(TARGETS) + + +# +# Make library targets... +# + +libs: + + +# +# Make unit tests... +# + +unittests: + + +# +# Clean all object files... +# + +clean: + $(RM) $(OBJS) $(TARGETS) dbus.h + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: + $(INSTALL_DIR) -m 775 -g $(CUPS_GROUP) $(CACHEDIR)/rss + + +# +# Install programs... +# + +install-exec: + echo Installing notifiers in $(SERVERBIN)/notifier... + $(INSTALL_DIR) -m 755 $(SERVERBIN)/notifier + for file in $(NOTIFIERS); do \ + $(INSTALL_BIN) $$file $(SERVERBIN)/notifier; \ + done + if test "x$(SYMROOT)" != "x"; then \ + $(INSTALL_DIR) $(SYMROOT); \ + cp $(NOTIFIERS) $(SYMROOT); \ + fi + + +# +# Install headers... +# + +install-headers: + + +# +# Install libraries... +# + +install-libs: + + +# +# Uninstall all targets... +# + +uninstall: + for file in $(NOTIFIERS); do \ + $(RM) $(SERVERBIN)/notifier/$$file; \ + done + -$(RMDIR) $(SERVERBIN)/notifier + -$(RMDIR) $(SERVERBIN) + -$(RMDIR) $(CACHEDIR)/rss + + +# +# Update dependencies (without system header dependencies...) +# + +depend: + $(CC) -MM $(ALL_CFLAGS) $(OBJS:.o=.c) >Dependencies + + +# +# dbus +# + +dbus: dbus.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o dbus dbus.o $(DBUS_NOTIFIERLIBS) $(LIBS) + + +# +# mailto +# + +mailto: mailto.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o mailto mailto.o $(LIBS) + + +# +# rss +# + +rss: rss.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o rss rss.o $(LIBS) + + +# +# testnotify +# + +testnotify: testnotify.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o testnotify testnotify.o $(LIBS) + + +$(OBJS): ../Makedefs + +include Dependencies + + +# +# End of "$Id$". +# diff --git a/notifier/dbus.c b/notifier/dbus.c new file mode 100644 index 0000000000..890b2ae09b --- /dev/null +++ b/notifier/dbus.c @@ -0,0 +1,627 @@ +/* + * "$Id$" + * + * D-Bus notifier for CUPS. + * + * Copyright 2008-2011 by Apple Inc. + * Copyright (C) 2011 Red Hat, Inc. + * Copyright (C) 2007 Tim Waugh + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Read events and send DBUS notifications. + * acquire_lock() - Acquire a lock so we only have a single notifier running. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include +#include +#include +#include + +#ifdef HAVE_DBUS +# include +# ifdef HAVE_DBUS_MESSAGE_ITER_INIT_APPEND +# define dbus_message_append_iter_init dbus_message_iter_init_append +# define dbus_message_iter_append_string(i,v) dbus_message_iter_append_basic(i, DBUS_TYPE_STRING, v) +# define dbus_message_iter_append_uint32(i,v) dbus_message_iter_append_basic(i, DBUS_TYPE_UINT32, v) +# define dbus_message_iter_append_boolean(i,v) dbus_message_iter_append_basic(i, DBUS_TYPE_BOOLEAN, v) +# endif /* HAVE_DBUS_MESSAGE_ITER_INIT_APPEND */ + + +/* + * D-Bus object: org.cups.cupsd.Notifier + * D-Bus object path: /org/cups/cupsd/Notifier + * + * D-Bus interface name: org.cups.cupsd.Notifier + * + * Signals: + * + * ServerRestarted(STRING text) + * Server has restarted. + * + * ServerStarted(STRING text) + * Server has started. + * + * ServerStopped(STRING text) + * Server has stopped. + * + * ServerAudit(STRING text) + * Security-related event. + * + * PrinterRestarted(STRING text, + * STRING printer-uri, + * STRING printer-name, + * UINT32 printer-state, + * STRING printer-state-reasons, + * BOOLEAN printer-is-accepting-jobs) + * Printer has restarted. + * + * PrinterShutdown(STRING text, + * STRING printer-uri, + * STRING printer-name, + * UINT32 printer-state, + * STRING printer-state-reasons, + * BOOLEAN printer-is-accepting-jobs) + * Printer has shutdown. + * + * PrinterStopped(STRING text, + * STRING printer-uri, + * STRING printer-name, + * UINT32 printer-state, + * STRING printer-state-reasons, + * BOOLEAN printer-is-accepting-jobs) + * Printer has stopped. + * + * PrinterStateChanged(STRING text, + * STRING printer-uri, + * STRING printer-name, + * UINT32 printer-state, + * STRING printer-state-reasons, + * BOOLEAN printer-is-accepting-jobs) + * Printer state has changed. + * + * PrinterFinishingsChanged(STRING text, + * STRING printer-uri, + * STRING printer-name, + * UINT32 printer-state, + * STRING printer-state-reasons, + * BOOLEAN printer-is-accepting-jobs) + * Printer's finishings-supported attribute has changed. + * + * PrinterMediaChanged(STRING text, + * STRING printer-uri, + * STRING printer-name, + * UINT32 printer-state, + * STRING printer-state-reasons, + * BOOLEAN printer-is-accepting-jobs) + * Printer's media-supported attribute has changed. + * + * PrinterAdded(STRING text, + * STRING printer-uri, + * STRING printer-name, + * UINT32 printer-state, + * STRING printer-state-reasons, + * BOOLEAN printer-is-accepting-jobs) + * Printer has been added. + * + * PrinterDeleted(STRING text, + * STRING printer-uri, + * STRING printer-name, + * UINT32 printer-state, + * STRING printer-state-reasons, + * BOOLEAN printer-is-accepting-jobs) + * Printer has been deleted. + * + * PrinterModified(STRING text, + * STRING printer-uri, + * STRING printer-name, + * UINT32 printer-state, + * STRING printer-state-reasons, + * BOOLEAN printer-is-accepting-jobs) + * Printer has been modified. + * + * text describes the event. + * printer-state-reasons is a comma-separated list. + * If printer-uri is "" in a Job* signal, the other printer-* parameters + * must be ignored. + * If the job name is not know, job-name will be "". + */ + +/* + * Constants... + */ + +enum +{ + PARAMS_NONE, + PARAMS_PRINTER, + PARAMS_JOB +}; + + +/* + * Local functions... + */ + +static int acquire_lock(int *fd, char *lockfile, size_t locksize); + + +/* + * 'main()' - Read events and send DBUS notifications. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + ipp_t *msg; /* Event message from scheduler */ + ipp_state_t state; /* IPP event state */ + struct sigaction action; /* POSIX sigaction data */ + DBusConnection *con = NULL; /* Connection to DBUS server */ + DBusError error; /* Error, if any */ + DBusMessage *message; /* Message to send */ + DBusMessageIter iter; /* Iterator for message data */ + int lock_fd = -1; /* Lock file descriptor */ + char lock_filename[1024]; + /* Lock filename */ + + + /* + * Don't buffer stderr... + */ + + setbuf(stderr, NULL); + + /* + * Ignore SIGPIPE signals... + */ + + memset(&action, 0, sizeof(action)); + action.sa_handler = SIG_IGN; + sigaction(SIGPIPE, &action, NULL); + + /* + * Validate command-line options... + */ + + if (argc != 3) + { + fputs("Usage: dbus dbus:/// notify-user-data\n", stderr); + return (1); + } + + if (strncmp(argv[1], "dbus:", 5)) + { + fprintf(stderr, "ERROR: Bad URI \"%s\"!\n", argv[1]); + return (1); + } + + /* + * Loop forever until we run out of events... + */ + + for (;;) + { + ipp_attribute_t *attr; /* Current attribute */ + const char *event; /* Event name */ + const char *signame = NULL;/* DBUS signal name */ + char *printer_reasons = NULL; + /* Printer reasons string */ + char *job_reasons = NULL; + /* Job reasons string */ + const char *nul = ""; /* Empty string value */ + int no = 0; /* Boolean "no" value */ + int params = PARAMS_NONE; + /* What parameters to include? */ + + + /* + * Get the next event... + */ + + msg = ippNew(); + while ((state = ippReadFile(0, msg)) != IPP_DATA) + { + if (state <= IPP_IDLE) + break; + } + + fprintf(stderr, "DEBUG: state=%d\n", state); + + if (state == IPP_ERROR) + fputs("DEBUG: ippReadFile() returned IPP_ERROR!\n", stderr); + + if (state <= IPP_IDLE) + { + /* + * Out of messages, free memory and then exit... + */ + + ippDelete(msg); + break; + } + + /* + * Verify connection to DBUS server... + */ + + if (con && !dbus_connection_get_is_connected(con)) + { + dbus_connection_unref(con); + con = NULL; + } + + if (!con) + { + dbus_error_init(&error); + + con = dbus_bus_get(DBUS_BUS_SYSTEM, &error); + if (!con) + dbus_error_free(&error); + else + fputs("DEBUG: Connected to D-BUS\n", stderr); + } + + if (!con) + continue; + + if (lock_fd == -1 && + acquire_lock(&lock_fd, lock_filename, sizeof(lock_filename))) + continue; + + attr = ippFindAttribute(msg, "notify-subscribed-event", + IPP_TAG_KEYWORD); + if (!attr) + continue; + + event = ippGetString(attr, 0, NULL); + if (!strncmp(event, "server-", 7)) + { + const char *word2 = event + 7; /* Second word */ + + if (!strcmp(word2, "restarted")) + signame = "ServerRestarted"; + else if (!strcmp(word2, "started")) + signame = "ServerStarted"; + else if (!strcmp(word2, "stopped")) + signame = "ServerStopped"; + else if (!strcmp(word2, "audit")) + signame = "ServerAudit"; + else + continue; + } + else if (!strncmp(event, "printer-", 8)) + { + const char *word2 = event + 8; /* Second word */ + + params = PARAMS_PRINTER; + if (!strcmp(word2, "restarted")) + signame = "PrinterRestarted"; + else if (!strcmp(word2, "shutdown")) + signame = "PrinterShutdown"; + else if (!strcmp(word2, "stopped")) + signame = "PrinterStopped"; + else if (!strcmp(word2, "state-changed")) + signame = "PrinterStateChanged"; + else if (!strcmp(word2, "finishings-changed")) + signame = "PrinterFinishingsChanged"; + else if (!strcmp(word2, "media-changed")) + signame = "PrinterMediaChanged"; + else if (!strcmp(word2, "added")) + signame = "PrinterAdded"; + else if (!strcmp(word2, "deleted")) + signame = "PrinterDeleted"; + else if (!strcmp(word2, "modified")) + signame = "PrinterModified"; + else + continue; + } + else if (!strncmp(event, "job-", 4)) + { + const char *word2 = event + 4; /* Second word */ + + params = PARAMS_JOB; + if (!strcmp(word2, "state-changed")) + signame = "JobState"; + else if (!strcmp(word2, "created")) + signame = "JobCreated"; + else if (!strcmp(word2, "completed")) + signame = "JobCompleted"; + else if (!strcmp(word2, "stopped")) + signame = "JobStopped"; + else if (!strcmp(word2, "config-changed")) + signame = "JobConfigChanged"; + else if (!strcmp(word2, "progress")) + signame = "JobProgress"; + else + continue; + } + else + continue; + + /* + * Create and send the new message... + */ + + fprintf(stderr, "DEBUG: %s\n", signame); + message = dbus_message_new_signal("/org/cups/cupsd/Notifier", + "org.cups.cupsd.Notifier", + signame); + + dbus_message_append_iter_init(message, &iter); + attr = ippFindAttribute(msg, "notify-text", IPP_TAG_TEXT); + if (attr) + { + const char *val = ippGetString(attr, 0, NULL); + if (!dbus_message_iter_append_string(&iter, &val)) + goto bail; + } + else + goto bail; + + if (params >= PARAMS_PRINTER) + { + char *p; /* Pointer into printer_reasons */ + size_t reasons_length; /* Required size of printer_reasons */ + int i; /* Looping var */ + int have_printer_params = 1;/* Do we have printer URI? */ + + /* STRING printer-uri or "" */ + attr = ippFindAttribute(msg, "notify-printer-uri", IPP_TAG_URI); + if (attr) + { + const char *val = ippGetString(attr, 0, NULL); + if (!dbus_message_iter_append_string(&iter, &val)) + goto bail; + } + else + { + have_printer_params = 0; + dbus_message_iter_append_string(&iter, &nul); + } + + /* STRING printer-name */ + if (have_printer_params) + { + attr = ippFindAttribute(msg, "printer-name", IPP_TAG_NAME); + if (attr) + { + const char *val = ippGetString(attr, 0, NULL); + if (!dbus_message_iter_append_string(&iter, &val)) + goto bail; + } + else + goto bail; + } + else + dbus_message_iter_append_string(&iter, &nul); + + /* UINT32 printer-state */ + if (have_printer_params) + { + attr = ippFindAttribute(msg, "printer-state", IPP_TAG_ENUM); + if (attr) + { + dbus_uint32_t val = ippGetInteger(attr, 0); + dbus_message_iter_append_uint32(&iter, &val); + } + else + goto bail; + } + else + dbus_message_iter_append_uint32(&iter, &no); + + /* STRING printer-state-reasons */ + if (have_printer_params) + { + attr = ippFindAttribute(msg, "printer-state-reasons", + IPP_TAG_KEYWORD); + if (attr) + { + int num_values = ippGetCount(attr); + for (reasons_length = 0, i = 0; i < num_values; i++) + /* All need commas except the last, which needs a nul byte. */ + reasons_length += 1 + strlen(ippGetString(attr, i, NULL)); + printer_reasons = malloc(reasons_length); + if (!printer_reasons) + goto bail; + p = printer_reasons; + for (i = 0; i < num_values; i++) + { + if (i) + *p++ = ','; + + strcpy(p, ippGetString(attr, i, NULL)); + p += strlen(p); + } + if (!dbus_message_iter_append_string(&iter, &printer_reasons)) + goto bail; + } + else + goto bail; + } + else + dbus_message_iter_append_string(&iter, &nul); + + /* BOOL printer-is-accepting-jobs */ + if (have_printer_params) + { + attr = ippFindAttribute(msg, "printer-is-accepting-jobs", + IPP_TAG_BOOLEAN); + if (attr) + { + dbus_bool_t val = ippGetBoolean(attr, 0); + dbus_message_iter_append_boolean(&iter, &val); + } + else + goto bail; + } + else + dbus_message_iter_append_boolean(&iter, &no); + } + + if (params >= PARAMS_JOB) + { + char *p; /* Pointer into job_reasons */ + size_t reasons_length; /* Required size of job_reasons */ + int i; /* Looping var */ + + /* UINT32 job-id */ + attr = ippFindAttribute(msg, "notify-job-id", IPP_TAG_INTEGER); + if (attr) + { + dbus_uint32_t val = ippGetInteger(attr, 0); + dbus_message_iter_append_uint32(&iter, &val); + } + else + goto bail; + + /* UINT32 job-state */ + attr = ippFindAttribute(msg, "job-state", IPP_TAG_ENUM); + if (attr) + { + dbus_uint32_t val = ippGetInteger(attr, 0); + dbus_message_iter_append_uint32(&iter, &val); + } + else + goto bail; + + /* STRING job-state-reasons */ + attr = ippFindAttribute(msg, "job-state-reasons", IPP_TAG_KEYWORD); + if (attr) + { + int num_values = ippGetCount(attr); + for (reasons_length = 0, i = 0; i < num_values; i++) + /* All need commas except the last, which needs a nul byte. */ + reasons_length += 1 + strlen(ippGetString(attr, i, NULL)); + job_reasons = malloc(reasons_length); + if (!job_reasons) + goto bail; + p = job_reasons; + for (i = 0; i < num_values; i++) + { + if (i) + *p++ = ','; + + strcpy(p, ippGetString(attr, i, NULL)); + p += strlen(p); + } + if (!dbus_message_iter_append_string(&iter, &job_reasons)) + goto bail; + } + else + goto bail; + + /* STRING job-name or "" */ + attr = ippFindAttribute(msg, "job-name", IPP_TAG_NAME); + if (attr) + { + const char *val = ippGetString(attr, 0, NULL); + if (!dbus_message_iter_append_string(&iter, &val)) + goto bail; + } + else + dbus_message_iter_append_string(&iter, &nul); + + /* UINT32 job-impressions-completed */ + attr = ippFindAttribute(msg, "job-impressions-completed", + IPP_TAG_INTEGER); + if (attr) + { + dbus_uint32_t val = ippGetInteger(attr, 0); + dbus_message_iter_append_uint32(&iter, &val); + } + else + goto bail; + } + + dbus_connection_send(con, message, NULL); + dbus_connection_flush(con); + + /* + * Cleanup... + */ + + bail: + + dbus_message_unref(message); + + if (printer_reasons) + free(printer_reasons); + + if (job_reasons) + free(job_reasons); + + ippDelete(msg); + } + + /* + * Remove lock file... + */ + + if (lock_fd >= 0) + { + close(lock_fd); + unlink(lock_filename); + } + + return (0); +} + + +/* + * 'acquire_lock()' - Acquire a lock so we only have a single notifier running. + */ + +static int /* O - 0 on success, -1 on failure */ +acquire_lock(int *fd, /* O - Lock file descriptor */ + char *lockfile, /* I - Lock filename buffer */ + size_t locksize) /* I - Size of filename buffer */ +{ + const char *tmpdir; /* Temporary directory */ + + + /* + * Figure out where to put the lock file... + */ + + if ((tmpdir = getenv("TMPDIR")) == NULL) + tmpdir = "/tmp"; + + snprintf(lockfile, locksize, "%s/cups-dbus-notifier-lockfile", tmpdir); + + /* + * Create the lock file and fail if it already exists... + */ + + if ((*fd = open(lockfile, O_RDWR | O_CREAT | O_EXCL, S_IRUSR | S_IWUSR)) < 0) + return (-1); + else + return (0); +} +#else /* !HAVE_DBUS */ +int +main(void) +{ + return (1); +} +#endif /* HAVE_DBUS */ + + +/* + * End of "$Id$". + */ diff --git a/notifier/mailto.c b/notifier/mailto.c new file mode 100644 index 0000000000..7b2f8a1a2b --- /dev/null +++ b/notifier/mailto.c @@ -0,0 +1,646 @@ +/* + * "$Id$" + * + * "mailto" notifier for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Main entry for the mailto notifier. + * email_message() - Email a notification message. + * load_configuration() - Load the mailto.conf file. + * pipe_sendmail() - Open a pipe to sendmail... + * print_attributes() - Print the attributes in a request... + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include + + +/* + * Globals... + */ + +char mailtoCc[1024]; /* Cc email address */ +char mailtoFrom[1024]; /* From email address */ +char mailtoReplyTo[1024]; /* Reply-To email address */ +char mailtoSubject[1024]; /* Subject prefix */ +char mailtoSMTPServer[1024]; /* SMTP server to use */ +char mailtoSendmail[1024]; /* Sendmail program to use */ + + +/* + * Local functions... + */ + +void email_message(const char *to, const char *subject, + const char *text); +int load_configuration(void); +cups_file_t *pipe_sendmail(const char *to); +void print_attributes(ipp_t *ipp, int indent); + + +/* + * 'main()' - Main entry for the mailto notifier. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + ipp_t *msg; /* Event message from scheduler */ + ipp_state_t state; /* IPP event state */ + char *subject, /* Subject for notification message */ + *text; /* Text for notification message */ + cups_lang_t *lang; /* Language info */ + char temp[1024]; /* Temporary string */ + int templen; /* Length of temporary string */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* POSIX sigaction data */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ + + + /* + * Don't buffer stderr... + */ + + setbuf(stderr, NULL); + + /* + * Ignore SIGPIPE signals... + */ + +#ifdef HAVE_SIGSET + sigset(SIGPIPE, SIG_IGN); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + action.sa_handler = SIG_IGN; + sigaction(SIGPIPE, &action, NULL); +#else + signal(SIGPIPE, SIG_IGN); +#endif /* HAVE_SIGSET */ + + /* + * Validate command-line options... + */ + + if (argc != 3) + { + fputs("Usage: mailto mailto:user@domain.com notify-user-data\n", stderr); + return (1); + } + + if (strncmp(argv[1], "mailto:", 7)) + { + fprintf(stderr, "ERROR: Bad recipient \"%s\"!\n", argv[1]); + return (1); + } + + fprintf(stderr, "DEBUG: argc=%d\n", argc); + for (i = 0; i < argc; i ++) + fprintf(stderr, "DEBUG: argv[%d]=\"%s\"\n", i, argv[i]); + + /* + * Load configuration data... + */ + + if ((lang = cupsLangDefault()) == NULL) + return (1); + + if (!load_configuration()) + return (1); + + /* + * Get the reply-to address... + */ + + templen = sizeof(temp); + httpDecode64_2(temp, &templen, argv[2]); + + if (!strncmp(temp, "mailto:", 7)) + strlcpy(mailtoReplyTo, temp + 7, sizeof(mailtoReplyTo)); + else if (temp[0]) + fprintf(stderr, "WARNING: Bad notify-user-data value (%d bytes) ignored!\n", + templen); + + /* + * Loop forever until we run out of events... + */ + + for (;;) + { + /* + * Get the next event... + */ + + msg = ippNew(); + while ((state = ippReadFile(0, msg)) != IPP_DATA) + { + if (state <= IPP_IDLE) + break; + } + + fprintf(stderr, "DEBUG: state=%d\n", state); + + if (state == IPP_ERROR) + fputs("DEBUG: ippReadFile() returned IPP_ERROR!\n", stderr); + + if (state <= IPP_IDLE) + { + /* + * Out of messages, free memory and then exit... + */ + + ippDelete(msg); + return (0); + } + + /* + * Get the subject and text for the message, then email it... + */ + + subject = cupsNotifySubject(lang, msg); + text = cupsNotifyText(lang, msg); + + fprintf(stderr, "DEBUG: subject=\"%s\"\n", subject); + fprintf(stderr, "DEBUG: text=\"%s\"\n", text); + + if (subject && text) + email_message(argv[1] + 7, subject, text); + else + { + fputs("ERROR: Missing attributes in event notification!\n", stderr); + print_attributes(msg, 4); + } + + /* + * Free the memory used for this event... + */ + + if (subject) + free(subject); + + if (text) + free(text); + + ippDelete(msg); + } +} + + +/* + * 'email_message()' - Email a notification message. + */ + +void +email_message(const char *to, /* I - Recipient of message */ + const char *subject, /* I - Subject of message */ + const char *text) /* I - Text of message */ +{ + cups_file_t *fp; /* Pipe/socket to mail server */ + const char *nl; /* Newline to use */ + char response[1024]; /* SMTP response buffer */ + + + /* + * Connect to the mail server... + */ + + if (mailtoSendmail[0]) + { + /* + * Use the sendmail command... + */ + + fp = pipe_sendmail(to); + + if (!fp) + return; + + nl = "\n"; + } + else + { + /* + * Use an SMTP server... + */ + + char hostbuf[1024]; /* Local hostname */ + + + if (strchr(mailtoSMTPServer, ':')) + fp = cupsFileOpen(mailtoSMTPServer, "s"); + else + { + char spec[1024]; /* Host:service spec */ + + + snprintf(spec, sizeof(spec), "%s:smtp", mailtoSMTPServer); + fp = cupsFileOpen(spec, "s"); + } + + if (!fp) + { + fprintf(stderr, "ERROR: Unable to connect to SMTP server \"%s\"!\n", + mailtoSMTPServer); + return; + } + + fprintf(stderr, "DEBUG: Connected to \"%s\"...\n", mailtoSMTPServer); + + cupsFilePrintf(fp, "HELO %s\r\n", + httpGetHostname(NULL, hostbuf, sizeof(hostbuf))); + fprintf(stderr, "DEBUG: >>> HELO %s\n", hostbuf); + + if (!cupsFileGets(fp, response, sizeof(response)) || atoi(response) >= 500) + goto smtp_error; + fprintf(stderr, "DEBUG: <<< %s\n", response); + + cupsFilePrintf(fp, "MAIL FROM:%s\r\n", mailtoFrom); + fprintf(stderr, "DEBUG: >>> MAIL FROM:%s\n", mailtoFrom); + + if (!cupsFileGets(fp, response, sizeof(response)) || atoi(response) >= 500) + goto smtp_error; + fprintf(stderr, "DEBUG: <<< %s\n", response); + + cupsFilePrintf(fp, "RCPT TO:%s\r\n", to); + fprintf(stderr, "DEBUG: >>> RCPT TO:%s\n", to); + + if (!cupsFileGets(fp, response, sizeof(response)) || atoi(response) >= 500) + goto smtp_error; + fprintf(stderr, "DEBUG: <<< %s\n", response); + + cupsFilePuts(fp, "DATA\r\n"); + fputs("DEBUG: DATA\n", stderr); + + if (!cupsFileGets(fp, response, sizeof(response)) || atoi(response) >= 500) + goto smtp_error; + fprintf(stderr, "DEBUG: <<< %s\n", response); + + nl = "\r\n"; + } + + /* + * Send the message... + */ + + cupsFilePrintf(fp, "Date: %s%s", httpGetDateString(time(NULL)), nl); + cupsFilePrintf(fp, "From: %s%s", mailtoFrom, nl); + cupsFilePrintf(fp, "Subject: %s %s%s", mailtoSubject, subject, nl); + if (mailtoReplyTo[0]) + { + cupsFilePrintf(fp, "Sender: %s%s", mailtoReplyTo, nl); + cupsFilePrintf(fp, "Reply-To: %s%s", mailtoReplyTo, nl); + } + cupsFilePrintf(fp, "To: %s%s", to, nl); + if (mailtoCc[0]) + cupsFilePrintf(fp, "Cc: %s%s", mailtoCc, nl); + cupsFilePrintf(fp, "Content-Type: text/plain%s", nl); + cupsFilePuts(fp, nl); + cupsFilePrintf(fp, "%s%s", text, nl); + cupsFilePrintf(fp, ".%s", nl); + + /* + * Close the connection to the mail server... + */ + + if (mailtoSendmail[0]) + { + /* + * Close the pipe and wait for the sendmail command to finish... + */ + + int status; /* Exit status */ + + + cupsFileClose(fp); + + while (wait(&status)) + { + if (errno != EINTR) + { + fprintf(stderr, "DEBUG: Unable to get child status: %s\n", + strerror(errno)); + status = 0; + break; + } + } + + /* + * Report any non-zero status... + */ + + if (status) + { + if (WIFEXITED(status)) + fprintf(stderr, "ERROR: Sendmail command returned status %d!\n", + WEXITSTATUS(status)); + else + fprintf(stderr, "ERROR: Sendmail command crashed on signal %d!\n", + WTERMSIG(status)); + } + } + else + { + /* + * Finish up the SMTP submission and close the connection... + */ + + if (!cupsFileGets(fp, response, sizeof(response)) || atoi(response) >= 500) + goto smtp_error; + fprintf(stderr, "DEBUG: <<< %s\n", response); + + /* + * Process SMTP errors here... + */ + + smtp_error: + + cupsFilePuts(fp, "QUIT\r\n"); + fputs("DEBUG: QUIT\n", stderr); + + if (!cupsFileGets(fp, response, sizeof(response)) || atoi(response) >= 500) + fprintf(stderr, "ERROR: Got \"%s\" trying to QUIT connection.\n", + response); + else + fprintf(stderr, "DEBUG: <<< %s\n", response); + + cupsFileClose(fp); + + fprintf(stderr, "DEBUG: Closed connection to \"%s\"...\n", + mailtoSMTPServer); + } +} + + +/* + * 'load_configuration()' - Load the mailto.conf file. + */ + +int /* I - 1 on success, 0 on failure */ +load_configuration(void) +{ + cups_file_t *fp; /* mailto.conf file */ + const char *server_root, /* CUPS_SERVERROOT environment variable */ + *server_admin; /* SERVER_ADMIN environment variable */ + char line[1024], /* Line from file */ + *value; /* Value for directive */ + int linenum; /* Line number in file */ + + + /* + * Initialize defaults... + */ + + mailtoCc[0] = '\0'; + + if ((server_admin = getenv("SERVER_ADMIN")) != NULL) + strlcpy(mailtoFrom, server_admin, sizeof(mailtoFrom)); + else + snprintf(mailtoFrom, sizeof(mailtoFrom), "root@%s", + httpGetHostname(NULL, line, sizeof(line))); + + strlcpy(mailtoSendmail, "/usr/sbin/sendmail", sizeof(mailtoSendmail)); + + mailtoSMTPServer[0] = '\0'; + + mailtoSubject[0] = '\0'; + + /* + * Try loading the config file... + */ + + if ((server_root = getenv("CUPS_SERVERROOT")) == NULL) + server_root = CUPS_SERVERROOT; + + snprintf(line, sizeof(line), "%s/mailto.conf", server_root); + + if ((fp = cupsFileOpen(line, "r")) == NULL) + { + if (errno != ENOENT) + { + fprintf(stderr, "ERROR: Unable to open \"%s\" - %s\n", line, + strerror(errno)); + return (1); + } + else + return (0); + } + + linenum = 0; + + while (cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + { + if (!value) + { + fprintf(stderr, "ERROR: No value found for %s directive on line %d!\n", + line, linenum); + cupsFileClose(fp); + return (0); + } + + if (!_cups_strcasecmp(line, "Cc")) + strlcpy(mailtoCc, value, sizeof(mailtoCc)); + else if (!_cups_strcasecmp(line, "From")) + strlcpy(mailtoFrom, value, sizeof(mailtoFrom)); + else if (!_cups_strcasecmp(line, "Sendmail")) + { + strlcpy(mailtoSendmail, value, sizeof(mailtoSendmail)); + mailtoSMTPServer[0] = '\0'; + } + else if (!_cups_strcasecmp(line, "SMTPServer")) + { + mailtoSendmail[0] = '\0'; + strlcpy(mailtoSMTPServer, value, sizeof(mailtoSMTPServer)); + } + else if (!_cups_strcasecmp(line, "Subject")) + strlcpy(mailtoSubject, value, sizeof(mailtoSubject)); + else + { + fprintf(stderr, + "ERROR: Unknown configuration directive \"%s\" on line %d!\n", + line, linenum); + } + } + + /* + * Close file and return... + */ + + cupsFileClose(fp); + + return (1); +} + + +/* + * 'pipe_sendmail()' - Open a pipe to sendmail... + */ + +cups_file_t * /* O - CUPS file */ +pipe_sendmail(const char *to) /* I - To: address */ +{ + cups_file_t *fp; /* CUPS file */ + int pid; /* Process ID */ + int pipefds[2]; /* Pipe file descriptors */ + int argc; /* Number of arguments */ + char *argv[100], /* Argument array */ + line[1024], /* Sendmail command + args */ + *lineptr; /* Pointer into line */ + + + /* + * First break the mailtoSendmail string into arguments... + */ + + strlcpy(line, mailtoSendmail, sizeof(line)); + argv[0] = line; + argc = 1; + + for (lineptr = strchr(line, ' '); lineptr; lineptr = strchr(lineptr, ' ')) + { + while (*lineptr == ' ') + *lineptr++ = '\0'; + + if (*lineptr) + { + /* + * Point to the next argument... + */ + + argv[argc ++] = lineptr; + + /* + * Stop if we have too many... + */ + + if (argc >= (int)(sizeof(argv) / sizeof(argv[0]) - 2)) + break; + } + } + + argv[argc ++] = (char *)to; + argv[argc] = NULL; + + /* + * Create the pipe... + */ + + if (pipe(pipefds)) + { + perror("ERROR: Unable to create pipe"); + return (NULL); + } + + /* + * Then run the command... + */ + + if ((pid = fork()) == 0) + { + /* + * Child goes here - redirect stdin to the input side of the pipe, + * redirect stdout to stderr, and exec... + */ + + close(0); + dup(pipefds[0]); + + close(1); + dup(2); + + close(pipefds[0]); + close(pipefds[1]); + + execvp(argv[0], argv); + exit(errno); + } + else if (pid < 0) + { + /* + * Unable to fork - error out... + */ + + perror("ERROR: Unable to fork command"); + + close(pipefds[0]); + close(pipefds[1]); + + return (NULL); + } + + /* + * Create a CUPS file using the output side of the pipe and close the + * input side... + */ + + close(pipefds[0]); + + if ((fp = cupsFileOpenFd(pipefds[1], "w")) == NULL) + { + int status; /* Status of command */ + + + close(pipefds[1]); + wait(&status); + } + + return (fp); +} + + +/* + * 'print_attributes()' - Print the attributes in a request... + */ + +void +print_attributes(ipp_t *ipp, /* I - IPP request */ + int indent) /* I - Indentation */ +{ + ipp_tag_t group; /* Current group */ + ipp_attribute_t *attr; /* Current attribute */ + char buffer[1024]; /* Value buffer */ + + + for (group = IPP_TAG_ZERO, attr = ipp->attrs; attr; attr = attr->next) + { + if ((attr->group_tag == IPP_TAG_ZERO && indent <= 8) || !attr->name) + { + group = IPP_TAG_ZERO; + fputc('\n', stderr); + continue; + } + + if (group != attr->group_tag) + { + group = attr->group_tag; + + fprintf(stderr, "DEBUG: %*s%s:\n\n", indent - 4, "", ippTagString(group)); + } + + ippAttributeString(attr, buffer, sizeof(buffer)); + + fprintf(stderr, "DEBUG: %*s%s (%s%s) %s", indent, "", attr->name, + attr->num_values > 1 ? "1setOf " : "", + ippTagString(attr->value_tag), buffer); + } +} + + +/* + * End of "$Id$". + */ diff --git a/notifier/rss.c b/notifier/rss.c new file mode 100644 index 0000000000..80fad631ed --- /dev/null +++ b/notifier/rss.c @@ -0,0 +1,732 @@ +/* + * "$Id$" + * + * RSS notifier for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Main entry for the test notifier. + * compare_rss() - Compare two messages. + * delete_message() - Free all memory used by a message. + * load_rss() - Load an existing RSS feed file. + * new_message() - Create a new RSS message. + * password_cb() - Return the cached password. + * save_rss() - Save messages to a RSS file. + * xml_escape() - Copy a string, escaping &, <, and > as needed. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include +#include +#include /* TODO: Update so we don't need this */ + + +/* + * Structures... + */ + +typedef struct _cups_rss_s /**** RSS message data ****/ +{ + int sequence_number; /* notify-sequence-number */ + char *subject, /* Message subject/summary */ + *text, /* Message text */ + *link_url; /* Link to printer */ + time_t event_time; /* When the event occurred */ +} _cups_rss_t; + + +/* + * Local globals... + */ + +static char *rss_password; /* Password for remote RSS */ + + +/* + * Local functions... + */ + +static int compare_rss(_cups_rss_t *a, _cups_rss_t *b); +static void delete_message(_cups_rss_t *rss); +static void load_rss(cups_array_t *rss, const char *filename); +static _cups_rss_t *new_message(int sequence_number, char *subject, + char *text, char *link_url, + time_t event_time); +static const char *password_cb(const char *prompt); +static int save_rss(cups_array_t *rss, const char *filename, + const char *baseurl); +static char *xml_escape(const char *s); + + +/* + * 'main()' - Main entry for the test notifier. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + ipp_t *event; /* Event from scheduler */ + ipp_state_t state; /* IPP event state */ + char scheme[32], /* URI scheme ("rss") */ + username[256], /* Username for remote RSS */ + host[1024], /* Hostname for remote RSS */ + resource[1024], /* RSS file */ + *options; /* Options */ + int port, /* Port number for remote RSS */ + max_events; /* Maximum number of events */ + http_t *http; /* Connection to remote server */ + http_status_t status; /* HTTP GET/PUT status code */ + char filename[1024], /* Local filename */ + newname[1024]; /* filename.N */ + cups_lang_t *language; /* Language information */ + ipp_attribute_t *printer_up_time, /* Timestamp on event */ + *notify_sequence_number,/* Sequence number */ + *notify_printer_uri; /* Printer URI */ + char *subject, /* Subject for notification message */ + *text, /* Text for notification message */ + link_url[1024], /* Link to printer */ + link_scheme[32], /* Scheme for link */ + link_username[256], /* Username for link */ + link_host[1024], /* Host for link */ + link_resource[1024]; /* Resource for link */ + int link_port; /* Link port */ + cups_array_t *rss; /* RSS message array */ + _cups_rss_t *msg; /* RSS message */ + char baseurl[1024]; /* Base URL */ + fd_set input; /* Input set for select() */ + struct timeval timeout; /* Timeout for select() */ + int changed; /* Has the RSS data changed? */ + int exit_status; /* Exit status */ + + + fprintf(stderr, "DEBUG: argc=%d\n", argc); + for (i = 0; i < argc; i ++) + fprintf(stderr, "DEBUG: argv[%d]=\"%s\"\n", i, argv[i]); + + /* + * See whether we are publishing this RSS feed locally or remotely... + */ + + if (httpSeparateURI(HTTP_URI_CODING_ALL, argv[1], scheme, sizeof(scheme), + username, sizeof(username), host, sizeof(host), &port, + resource, sizeof(resource)) < HTTP_URI_OK) + { + fprintf(stderr, "ERROR: Bad RSS URI \"%s\"!\n", argv[1]); + return (1); + } + + max_events = 20; + + if ((options = strchr(resource, '?')) != NULL) + { + *options++ = '\0'; + + if (!strncmp(options, "max_events=", 11)) + { + max_events = atoi(options + 11); + + if (max_events <= 0) + max_events = 20; + } + } + + rss = cupsArrayNew((cups_array_func_t)compare_rss, NULL); + + if (host[0]) + { + /* + * Remote feed, see if we can get the current file... + */ + + int fd; /* Temporary file */ + + + if ((rss_password = strchr(username, ':')) != NULL) + *rss_password++ = '\0'; + + cupsSetPasswordCB(password_cb); + cupsSetUser(username); + + if ((fd = cupsTempFd(filename, sizeof(filename))) < 0) + { + fprintf(stderr, "ERROR: Unable to create temporary file: %s\n", + strerror(errno)); + + return (1); + } + + if ((http = httpConnect(host, port)) == NULL) + { + fprintf(stderr, "ERROR: Unable to connect to %s on port %d: %s\n", + host, port, strerror(errno)); + + close(fd); + unlink(filename); + + return (1); + } + + status = cupsGetFd(http, resource, fd); + + close(fd); + + if (status != HTTP_OK && status != HTTP_NOT_FOUND) + { + fprintf(stderr, "ERROR: Unable to GET %s from %s on port %d: %d %s\n", + resource, host, port, status, httpStatus(status)); + + httpClose(http); + unlink(filename); + + return (1); + } + + strlcpy(newname, filename, sizeof(newname)); + + httpAssembleURI(HTTP_URI_CODING_ALL, baseurl, sizeof(baseurl), "http", + NULL, host, port, resource); + } + else + { + const char *cachedir, /* CUPS_CACHEDIR */ + *server_name, /* SERVER_NAME */ + *server_port; /* SERVER_PORT */ + + + http = NULL; + + if ((cachedir = getenv("CUPS_CACHEDIR")) == NULL) + cachedir = CUPS_CACHEDIR; + + if ((server_name = getenv("SERVER_NAME")) == NULL) + server_name = "localhost"; + + if ((server_port = getenv("SERVER_PORT")) == NULL) + server_port = "631"; + + snprintf(filename, sizeof(filename), "%s/rss%s", cachedir, resource); + snprintf(newname, sizeof(newname), "%s.N", filename); + + httpAssembleURIf(HTTP_URI_CODING_ALL, baseurl, sizeof(baseurl), "http", + NULL, server_name, atoi(server_port), "/rss%s", resource); + } + + /* + * Load the previous RSS file, if any... + */ + + load_rss(rss, filename); + + changed = cupsArrayCount(rss) == 0; + + /* + * Localize for the user's chosen language... + */ + + language = cupsLangDefault(); + + /* + * Read events and update the RSS file until we are out of events. + */ + + for (exit_status = 0, event = NULL;;) + { + if (changed) + { + /* + * Save the messages to the file again, uploading as needed... + */ + + if (save_rss(rss, newname, baseurl)) + { + if (http) + { + /* + * Upload the RSS file... + */ + + if ((status = cupsPutFile(http, resource, filename)) != HTTP_CREATED) + fprintf(stderr, "ERROR: Unable to PUT %s from %s on port %d: %d %s\n", + resource, host, port, status, httpStatus(status)); + } + else + { + /* + * Move the new RSS file over top the old one... + */ + + if (rename(newname, filename)) + fprintf(stderr, "ERROR: Unable to rename %s to %s: %s\n", + newname, filename, strerror(errno)); + } + + changed = 0; + } + } + + /* + * Wait up to 30 seconds for an event... + */ + + timeout.tv_sec = 30; + timeout.tv_usec = 0; + + FD_ZERO(&input); + FD_SET(0, &input); + + if (select(1, &input, NULL, NULL, &timeout) < 0) + continue; + else if (!FD_ISSET(0, &input)) + { + fprintf(stderr, "DEBUG: %s is bored, exiting...\n", argv[1]); + break; + } + + /* + * Read the next event... + */ + + event = ippNew(); + while ((state = ippReadFile(0, event)) != IPP_DATA) + { + if (state <= IPP_IDLE) + break; + } + + if (state == IPP_ERROR) + fputs("DEBUG: ippReadFile() returned IPP_ERROR!\n", stderr); + + if (state <= IPP_IDLE) + break; + + /* + * Collect the info from the event... + */ + + printer_up_time = ippFindAttribute(event, "printer-up-time", + IPP_TAG_INTEGER); + notify_sequence_number = ippFindAttribute(event, "notify-sequence-number", + IPP_TAG_INTEGER); + notify_printer_uri = ippFindAttribute(event, "notify-printer-uri", + IPP_TAG_URI); + subject = cupsNotifySubject(language, event); + text = cupsNotifyText(language, event); + + if (printer_up_time && notify_sequence_number && subject && text) + { + /* + * Create a new RSS message... + */ + + if (notify_printer_uri) + { + httpSeparateURI(HTTP_URI_CODING_ALL, + notify_printer_uri->values[0].string.text, + link_scheme, sizeof(link_scheme), + link_username, sizeof(link_username), + link_host, sizeof(link_host), &link_port, + link_resource, sizeof(link_resource)); + httpAssembleURI(HTTP_URI_CODING_ALL, link_url, sizeof(link_url), + "http", link_username, link_host, link_port, + link_resource); + } + + msg = new_message(notify_sequence_number->values[0].integer, + xml_escape(subject), xml_escape(text), + notify_printer_uri ? xml_escape(link_url) : NULL, + printer_up_time->values[0].integer); + + if (!msg) + { + fprintf(stderr, "ERROR: Unable to create message: %s\n", + strerror(errno)); + exit_status = 1; + break; + } + + /* + * Add it to the array... + */ + + cupsArrayAdd(rss, msg); + + changed = 1; + + /* + * Trim the array as needed... + */ + + while (cupsArrayCount(rss) > max_events) + { + msg = cupsArrayFirst(rss); + + cupsArrayRemove(rss, msg); + + delete_message(msg); + } + } + + if (subject) + free(subject); + + if (text) + free(text); + + ippDelete(event); + event = NULL; + } + + /* + * We only get here when idle or error... + */ + + ippDelete(event); + + if (http) + { + unlink(filename); + httpClose(http); + } + + return (exit_status); +} + + +/* + * 'compare_rss()' - Compare two messages. + */ + +static int /* O - Result of comparison */ +compare_rss(_cups_rss_t *a, /* I - First message */ + _cups_rss_t *b) /* I - Second message */ +{ + return (a->sequence_number - b->sequence_number); +} + + +/* + * 'delete_message()' - Free all memory used by a message. + */ + +static void +delete_message(_cups_rss_t *msg) /* I - RSS message */ +{ + if (msg->subject) + free(msg->subject); + + if (msg->text) + free(msg->text); + + if (msg->link_url) + free(msg->link_url); + + free(msg); +} + + +/* + * 'load_rss()' - Load an existing RSS feed file. + */ + +static void +load_rss(cups_array_t *rss, /* I - RSS messages */ + const char *filename) /* I - File to load */ +{ + FILE *fp; /* File pointer */ + char line[4096], /* Line from file */ + *subject, /* Subject */ + *text, /* Text */ + *link_url, /* Link URL */ + *start, /* Start of element */ + *end; /* End of element */ + time_t event_time; /* Event time */ + int sequence_number; /* Sequence number */ + int in_item; /* In an item */ + _cups_rss_t *msg; /* New message */ + + + if ((fp = fopen(filename, "r")) == NULL) + { + if (errno != ENOENT) + fprintf(stderr, "ERROR: Unable to open %s: %s\n", filename, + strerror(errno)); + + return; + } + + subject = NULL; + text = NULL; + link_url = NULL; + event_time = 0; + sequence_number = 0; + in_item = 0; + + while (fgets(line, sizeof(line), fp)) + { + if (strstr(line, "")) + in_item = 1; + else if (strstr(line, "") && in_item) + { + if (subject && text) + { + msg = new_message(sequence_number, subject, text, link_url, + event_time); + + if (msg) + cupsArrayAdd(rss, msg); + + } + else + { + if (subject) + free(subject); + + if (text) + free(text); + + if (link_url) + free(link_url); + } + + subject = NULL; + text = NULL; + link_url = NULL; + event_time = 0; + sequence_number = 0; + in_item = 0; + } + else if (!in_item) + continue; + else if ((start = strstr(line, "")) != NULL) + { + start += 7; + if ((end = strstr(start, "")) != NULL) + { + *end = '\0'; + subject = strdup(start); + } + } + else if ((start = strstr(line, "")) != NULL) + { + start += 13; + if ((end = strstr(start, "")) != NULL) + { + *end = '\0'; + text = strdup(start); + } + } + else if ((start = strstr(line, "")) != NULL) + { + start += 6; + if ((end = strstr(start, "")) != NULL) + { + *end = '\0'; + link_url = strdup(start); + } + } + else if ((start = strstr(line, "")) != NULL) + { + start += 9; + if ((end = strstr(start, "")) != NULL) + { + *end = '\0'; + event_time = httpGetDateTime(start); + } + } + else if ((start = strstr(line, "")) != NULL) + sequence_number = atoi(start + 6); + } + + fclose(fp); +} + + +/* + * 'new_message()' - Create a new RSS message. + */ + +static _cups_rss_t * /* O - New message */ +new_message(int sequence_number, /* I - notify-sequence-number */ + char *subject, /* I - Subject/summary */ + char *text, /* I - Text */ + char *link_url, /* I - Link to printer */ + time_t event_time) /* I - Date/time of event */ +{ + _cups_rss_t *msg; /* New message */ + + + if ((msg = calloc(1, sizeof(_cups_rss_t))) == NULL) + return (NULL); + + msg->sequence_number = sequence_number; + msg->subject = subject; + msg->text = text; + msg->link_url = link_url; + msg->event_time = event_time; + + return (msg); +} + + +/* + * 'password_cb()' - Return the cached password. + */ + +static const char * /* O - Cached password */ +password_cb(const char *prompt) /* I - Prompt string, unused */ +{ + (void)prompt; + + return (rss_password); +} + + +/* + * 'save_rss()' - Save messages to a RSS file. + */ + +static int /* O - 1 on success, 0 on failure */ +save_rss(cups_array_t *rss, /* I - RSS messages */ + const char *filename, /* I - File to save to */ + const char *baseurl) /* I - Base URL */ +{ + FILE *fp; /* File pointer */ + _cups_rss_t *msg; /* Current message */ + char date[1024]; /* Current date */ + char *href; /* Escaped base URL */ + + + if ((fp = fopen(filename, "w")) == NULL) + { + fprintf(stderr, "ERROR: Unable to create %s: %s\n", filename, + strerror(errno)); + return (0); + } + + fputs("\n", fp); + fputs("\n", fp); + fputs(" \n", fp); + fputs(" CUPS RSS Feed\n", fp); + + href = xml_escape(baseurl); + fprintf(fp, " %s\n", href); + free(href); + + fputs(" CUPS RSS Feed\n", fp); + fputs(" " CUPS_SVERSION "\n", fp); + fputs(" 1\n", fp); + + fprintf(fp, " %s\n", + httpGetDateString2(time(NULL), date, sizeof(date))); + + for (msg = (_cups_rss_t *)cupsArrayLast(rss); + msg; + msg = (_cups_rss_t *)cupsArrayPrev(rss)) + { + fputs(" \n", fp); + fprintf(fp, " %s\n", msg->subject); + fprintf(fp, " %s\n", msg->text); + if (msg->link_url) + fprintf(fp, " %s\n", msg->link_url); + fprintf(fp, " %s\n", + httpGetDateString2(msg->event_time, date, sizeof(date))); + fprintf(fp, " %d\n", msg->sequence_number); + fputs(" \n", fp); + } + + fputs(" \n", fp); + fputs("\n", fp); + + return (!fclose(fp)); +} + + +/* + * 'xml_escape()' - Copy a string, escaping &, <, and > as needed. + */ + +static char * /* O - Escaped string */ +xml_escape(const char *s) /* I - String to escape */ +{ + char *e, /* Escaped string */ + *eptr; /* Pointer into escaped string */ + const char *sptr; /* Pointer into string */ + size_t bytes; /* Bytes needed for string */ + + + /* + * First figure out how many extra bytes we need... + */ + + for (bytes = 0, sptr = s; *sptr; sptr ++) + if (*sptr == '&') + bytes += 4; /* & */ + else if (*sptr == '<' || *sptr == '>') + bytes += 3; /* < and > */ + + /* + * If there is nothing to escape, just strdup() it... + */ + + if (bytes == 0) + return (strdup(s)); + + /* + * Otherwise allocate memory and copy... + */ + + if ((e = malloc(bytes + 1 + strlen(s))) == NULL) + return (NULL); + + for (eptr = e, sptr = s; *sptr; sptr ++) + if (*sptr == '&') + { + *eptr++ = '&'; + *eptr++ = 'a'; + *eptr++ = 'm'; + *eptr++ = 'p'; + *eptr++ = ';'; + } + else if (*sptr == '<') + { + *eptr++ = '&'; + *eptr++ = 'l'; + *eptr++ = 't'; + *eptr++ = ';'; + } + else if (*sptr == '>') + { + *eptr++ = '&'; + *eptr++ = 'g'; + *eptr++ = 't'; + *eptr++ = ';'; + } + else + *eptr++ = *sptr; + + *eptr = '\0'; + + return (e); +} + + +/* + * End of "$Id$". + */ diff --git a/notifier/testnotify.c b/notifier/testnotify.c new file mode 100644 index 0000000000..c42bd84fc9 --- /dev/null +++ b/notifier/testnotify.c @@ -0,0 +1,127 @@ +/* + * "$Id$" + * + * Test notifier for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Main entry for the test notifier. + * print_attributes() - Print the attributes in a request... + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * Local functions... + */ + +void print_attributes(ipp_t *ipp, int indent); + + +/* + * 'main()' - Main entry for the test notifier. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + ipp_t *event; /* Event from scheduler */ + ipp_state_t state; /* IPP event state */ + + + setbuf(stderr, NULL); + + fprintf(stderr, "DEBUG: argc=%d\n", argc); + for (i = 0; i < argc; i ++) + fprintf(stderr, "DEBUG: argv[%d]=\"%s\"\n", i, argv[i]); + fprintf(stderr, "DEBUG: TMPDIR=\"%s\"\n", getenv("TMPDIR")); + + for (;;) + { + event = ippNew(); + while ((state = ippReadFile(0, event)) != IPP_DATA) + { + if (state <= IPP_IDLE) + break; + } + + if (state == IPP_ERROR) + fputs("DEBUG: ippReadFile() returned IPP_ERROR!\n", stderr); + + if (state <= IPP_IDLE) + { + ippDelete(event); + return (0); + } + + print_attributes(event, 4); + ippDelete(event); + + /* + * If the recipient URI is "testnotify://nowait", then we exit after each + * event... + */ + + if (!strcmp(argv[1], "testnotify://nowait")) + return (0); + } +} + + +/* + * 'print_attributes()' - Print the attributes in a request... + */ + +void +print_attributes(ipp_t *ipp, /* I - IPP request */ + int indent) /* I - Indentation */ +{ + ipp_tag_t group; /* Current group */ + ipp_attribute_t *attr; /* Current attribute */ + char buffer[1024]; /* Value buffer */ + + + for (group = IPP_TAG_ZERO, attr = ipp->attrs; attr; attr = attr->next) + { + if ((attr->group_tag == IPP_TAG_ZERO && indent <= 8) || !attr->name) + { + group = IPP_TAG_ZERO; + fputc('\n', stderr); + continue; + } + + if (group != attr->group_tag) + { + group = attr->group_tag; + + fprintf(stderr, "DEBUG: %*s%s:\n\n", indent - 4, "", ippTagString(group)); + } + + ippAttributeString(attr, buffer, sizeof(buffer)); + + fprintf(stderr, "DEBUG: %*s%s (%s%s) %s", indent, "", attr->name, + attr->num_values > 1 ? "1setOf " : "", + ippTagString(attr->value_tag), buffer); + } +} + + +/* + * End of "$Id$". + */ diff --git a/packaging/InstallationCheck b/packaging/InstallationCheck new file mode 100755 index 0000000000..4e3405caa1 --- /dev/null +++ b/packaging/InstallationCheck @@ -0,0 +1,11 @@ +#!/bin/sh +case `uname -r` in + # Mac OS X 10.4.x + 8.*) exit 0;; + + # Mac OS X 10.5.x + 9.*) exit 0;; + + # Older Mac OS X... + *) exit 97;; +esac diff --git a/packaging/LICENSE.rtf b/packaging/LICENSE.rtf new file mode 100644 index 0000000000..b348f6b08a --- /dev/null +++ b/packaging/LICENSE.rtf @@ -0,0 +1,434 @@ +{\rtf1\ansi\ansicpg1252\cocoartf1011 +{\fonttbl\f0\froman\fcharset0 Times-Roman;\f1\fnil\fcharset0 LucidaGrande;\f2\fmodern\fcharset0 Courier; +\f3\fmodern\fcharset0 Courier-Oblique;} +{\colortbl;\red255\green255\blue255;\red0\green0\blue238;} +{\*\listtable{\list\listtemplateid1\listhybrid{\listlevel\levelnfc23\levelnfcn23\leveljc0\leveljcn0\levelfollow0\levelstartat1\levelspace360\levelindent0{\*\levelmarker \{disc\}}{\leveltext\leveltemplateid1\'01\uc0\u8226 ;}{\levelnumbers;}\fi-360\li720\lin720 }{\listname ;}\listid1} +{\list\listtemplateid2\listhybrid{\listlevel\levelnfc0\levelnfcn0\leveljc0\leveljcn0\levelfollow0\levelstartat1\levelspace360\levelindent0{\*\levelmarker \{decimal\}.}{\leveltext\leveltemplateid101\'02\'00.;}{\levelnumbers\'01;}\fi-360\li720\lin720 }{\listlevel\levelnfc0\levelnfcn0\leveljc0\leveljcn0\levelfollow0\levelstartat1\levelspace360\levelindent0{\*\levelmarker \{decimal\}.}{\leveltext\leveltemplateid102\'02\'01.;}{\levelnumbers\'01;}\fi-360\li1440\lin1440 }{\listname ;}\listid2} +{\list\listtemplateid3\listhybrid{\listlevel\levelnfc0\levelnfcn0\leveljc0\leveljcn0\levelfollow0\levelstartat1\levelspace360\levelindent0{\*\levelmarker \{decimal\}.}{\leveltext\leveltemplateid201\'02\'00.;}{\levelnumbers\'01;}\fi-360\li720\lin720 }{\listlevel\levelnfc0\levelnfcn0\leveljc0\leveljcn0\levelfollow0\levelstartat1\levelspace360\levelindent0{\*\levelmarker \{decimal\}.}{\leveltext\leveltemplateid202\'02\'01.;}{\levelnumbers\'01;}\fi-360\li1440\lin1440 }{\listname ;}\listid3} +{\list\listtemplateid4\listhybrid{\listlevel\levelnfc0\levelnfcn0\leveljc0\leveljcn0\levelfollow0\levelstartat1\levelspace360\levelindent0{\*\levelmarker \{decimal\}.}{\leveltext\leveltemplateid301\'02\'00.;}{\levelnumbers\'01;}\fi-360\li720\lin720 }{\listname ;}\listid4} +{\list\listtemplateid5\listhybrid{\listlevel\levelnfc0\levelnfcn0\leveljc0\leveljcn0\levelfollow0\levelstartat1\levelspace360\levelindent0{\*\levelmarker \{decimal\}.}{\leveltext\leveltemplateid401\'02\'00.;}{\levelnumbers\'01;}\fi-360\li720\lin720 }{\listname ;}\listid5} +{\list\listtemplateid6\listhybrid{\listlevel\levelnfc0\levelnfcn0\leveljc0\leveljcn0\levelfollow0\levelstartat1\levelspace360\levelindent0{\*\levelmarker \{decimal\}.}{\leveltext\leveltemplateid501\'02\'00.;}{\levelnumbers\'01;}\fi-360\li720\lin720 }{\listname ;}\listid6} +{\list\listtemplateid7\listhybrid{\listlevel\levelnfc0\levelnfcn0\leveljc0\leveljcn0\levelfollow0\levelstartat1\levelspace360\levelindent0{\*\levelmarker \{decimal\}.}{\leveltext\leveltemplateid601\'02\'00.;}{\levelnumbers\'01;}\fi-360\li720\lin720 }{\listname ;}\listid7}} +{\*\listoverridetable{\listoverride\listid1\listoverridecount0\ls1}{\listoverride\listid2\listoverridecount0\ls2}{\listoverride\listid3\listoverridecount0\ls3}{\listoverride\listid4\listoverridecount0\ls4}{\listoverride\listid5\listoverridecount0\ls5}{\listoverride\listid6\listoverridecount0\ls6}{\listoverride\listid7\listoverridecount0\ls7}} +\margl1440\margr1440\margb1800\margt1800\vieww9000\viewh8400\viewkind0 +\deftab720 +\pard\pardeftab720\sa280\qc + +\f0\b\fs36 \cf0 CUPS License Agreement\ +\pard\pardeftab720\sa240\qc + +\b0\fs24 \cf0 Copyright 2007-2009 by Apple Inc. +\f1 \uc0\u8232 +\f0 1 Infinite Loop +\f1 \uc0\u8232 +\f0 Cupertino, CA 95014 USA +\f1 \uc0\u8232 \u8232 +\f0 WWW: {\field{\*\fldinst{HYPERLINK "http://www.cups.org/"}}{\fldrslt \cf2 \ul \ulc2 http://www.cups.org}}\ +\pard\pardeftab720\sa280\ql\qnatural + +\b\fs28 \cf0 Introduction\ +\pard\pardeftab720\sa240\ql\qnatural + +\b0\fs24 \cf0 CUPS +\fs20 \super TM +\fs24 \nosupersub is provided under the GNU General Public License ("GPL") and GNU Library General Public License ("LGPL"), Version 2, with exceptions for Apple operating systems and the OpenSSL toolkit. A copy of the exceptions and licenses follow this introduction.\ +The GNU LGPL applies to the CUPS API library, located in the "cups" subdirectory of the CUPS source distribution and in the "cups" include directory and library files in the binary distributions. The GNU GPL applies to the remainder of the CUPS distribution, including the "pdftops" filter which is based upon Xpdf and the CUPS imaging library.\ +For those not familiar with the GNU GPL, the license basically allows you to:\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls1\ilvl0\cf0 {\listtext \'95 }Use the CUPS software at no charge.\ +{\listtext \'95 }Distribute verbatim copies of the software in source or binary form.\ +{\listtext \'95 }Sell verbatim copies of the software for a media fee, or sell support for the software.\ +{\listtext \'95 }Distribute or sell printer drivers and filters that use CUPS so long as source code is made available under the GPL.\ +\pard\pardeftab720\sa240\ql\qnatural +\cf0 What this license +\b does not +\b0 allow you to do is make changes or add features to CUPS and then sell a binary distribution without source code. You must provide source for any new drivers, changes, or additions to the software, and all code must be provided under the GPL or LGPL as appropriate. The only exceptions to this are the portions of the CUPS software covered by the Apple operating system license exceptions outlined later in this license agreement.\ +The GNU LGPL relaxes the "link-to" restriction, allowing you to develop applications that use the CUPS API library under other licenses and/or conditions as appropriate for your application.\ +\pard\pardeftab720\sa280\ql\qnatural + +\b\fs28 \cf0 License Exceptions\ +\pard\pardeftab720\sa240\ql\qnatural + +\b0\fs24 \cf0 In addition, as the copyright holder of CUPS, Apple Inc. grants the following special exceptions:\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls2\ilvl0 +\b \cf0 {\listtext 1. }Apple Operating System Development License Exception +\b0 ;\ +\pard\tx940\tx1440\pardeftab720\li1440\fi-1440\ql\qnatural +\ls2\ilvl1\cf0 {\listtext 1. }Software that is developed by any person or entity for an Apple Operating System ("Apple OS-Developed Software"), including but not limited to Apple and third party printer drivers, filters, and backends for an Apple Operating System, that is linked to the CUPS imaging library or based on any sample filters or backends provided with CUPS shall not be considered to be a derivative work or collective work based on the CUPS program and is exempt from the mandatory source code release clauses of the GNU GPL. You may therefore distribute linked combinations of the CUPS imaging library with Apple OS-Developed Software without releasing the source code of the Apple OS-Developed Software. You may also use sample filters and backends provided with CUPS to develop Apple OS-Developed Software without releasing the source code of the Apple OS-Developed Software.\ +{\listtext 2. }An Apple Operating System means any operating system software developed and/or marketed by Apple Computer, Inc., including but not limited to all existing releases and versions of Apple's Darwin, Mac OS X, and Mac OS X Server products and all follow-on releases and future versions thereof.\ +{\listtext 3. }This exception is only available for Apple OS-Developed Software and does not apply to software that is distributed for use on other operating systems.\ +{\listtext 4. }All CUPS software that falls under this license exception have the following text at the top of each source file:\ +{\listtext 5. }This file is subject to the Apple OS-Developed Software exception.\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls2\ilvl0 +\b \cf0 {\listtext 2. }OpenSSL Toolkit License Exception +\b0 ;\ +\pard\tx940\tx1440\pardeftab720\li1440\fi-1440\ql\qnatural +\ls2\ilvl1\cf0 {\listtext 1. }Apple Inc. explicitly allows the compilation and distribution of the CUPS software with the OpenSSL Toolkit.\ +\pard\pardeftab720\sa240\ql\qnatural +\cf0 No developer is required to provide these exceptions in a derived work.\ +\pard\pardeftab720\sa280\ql\qnatural + +\b\fs28 \cf0 Trademarks\ +\pard\pardeftab720\sa240\ql\qnatural + +\b0\fs24 \cf0 CUPS and the CUPS logo (the "CUPS Marks") are trademarks of Apple Inc. Apple grants you a non-exclusive and non-transferable right to use the CUPS Marks in any direct port or binary distribution incorporating CUPS software and in any promotional material therefor. You agree that your products will meet the highest levels of quality and integrity for similar goods, not be unlawful, and be developed, manufactured, and distributed in compliance with this license. You will not interfere with Apple's rights in the CUPS Marks, and all use of the CUPS Marks shall inure to the benefit of Apple. This license does not apply to use of the CUPS Marks in a derivative products, which requires prior written permission from Apple Inc.\ +\pard\pardeftab720\sa280\ql\qnatural + +\b\fs28 \cf0 \page \pard\pardeftab720\sa280\ql\qnatural + +\fs36 \cf0 GNU GENERAL PUBLIC LICENSE\ +\pard\pardeftab720\sa240\ql\qnatural + +\b0\fs24 \cf0 Version 2, June 1991\ +\pard\pardeftab720\ql\qnatural + +\f2 \cf0 Copyright 1989, 1991 Free Software Foundation, Inc.\ +59 Temple Place, Suite 330, Boston, MA 02111-1307 USA\ +\ +Everyone is permitted to copy and distribute verbatim\ +copies of this license document, but changing it is not allowed.\ +\pard\pardeftab720\sa300\ql\qnatural + +\f0\b \cf0 Preamble\ +\pard\pardeftab720\sa240\ql\qnatural + +\b0 \cf0 The licenses for most software are designed to take away your freedom to share and change it. By contrast, the GNU General Public License is intended to guarantee your freedom to share and change free software--to make sure the software is free for all its users. This General Public License applies to most of the Free Software Foundation's software and to any other program whose authors commit to using it. (Some other Free Software Foundation software is covered by the GNU Library General Public License instead.) You can apply it to your programs, too.\ +When we speak of free software, we are referring to freedom, not price. Our General Public Licenses are designed to make sure that you have the freedom to distribute copies of free software (and charge for this service if you wish), that you receive source code or can get it if you want it, that you can change the software or use pieces of it in new free programs; and that you know you can do these things.\ +To protect your rights, we need to make restrictions that forbid anyone to deny you these rights or to ask you to surrender the rights. These restrictions translate to certain responsibilities for you if you distribute copies of the software, or if you modify it.\ +For example, if you distribute copies of such a program, whether gratis or for a fee, you must give the recipients all the rights that you have. You must make sure that they, too, receive or can get the source code. And you must show them these terms so they know their rights.\ +We protect your rights with two steps: (1) copyright the software, and (2) offer you this license which gives you legal permission to copy, distribute and/or modify the software.\ +Also, for each author's protection and ours, we want to make certain that everyone understands that there is no warranty for this free software. If the software is modified by someone else and passed on, we want its recipients to know that what they have is not the original, so that any problems introduced by others will not reflect on the original authors' reputations.\ +Finally, any free program is threatened constantly by software patents. We wish to avoid the danger that redistributors of a free program will individually obtain patent licenses, in effect making the program proprietary. To prevent this, we have made it clear that any patent must be licensed for everyone's free use or not licensed at all.\ +The precise terms and conditions for copying, distribution and modification follow.\ +\pard\pardeftab720\sa300\ql\qnatural + +\b \cf0 GNU GENERAL PUBLIC LICENSE +\f1 \uc0\u8232 +\f0 TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls3\ilvl0 +\b0 \cf0 {\listtext 1. }This License applies to any program or other work which contains a notice placed by the copyright holder saying it may be distributed under the terms of this General Public License. The "Program", below, refers to any such program or work, and a "work based on the Program" means either the Program or any derivative work under copyright law: that is to say, a work containing the Program or a portion of it, either verbatim or with modifications and/or translated into another language. (Hereinafter, translation is included without limitation in the term "modification".) Each licensee is addressed as "you".\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\sa240\ql\qnatural +\ls3\ilvl0\cf0 {\listtext 2. }Activities other than copying, distribution and modification are not covered by this License; they are outside its scope. The act of running the Program is not restricted, and the output from the Program is covered only if its contents constitute a work based on the Program (independent of having been made by running the Program). Whether that is true depends on what the Program does.\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls3\ilvl0\cf0 {\listtext 3. }You may copy and distribute verbatim copies of the Program's source code as you receive it, in any medium, provided that you conspicuously and appropriately publish on each copy an appropriate copyright notice and disclaimer of warranty; keep intact all the notices that refer to this License and to the absence of any warranty; and give any other recipients of the Program a copy of this License along with the Program.\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\sa240\ql\qnatural +\ls3\ilvl0\cf0 {\listtext 4. }You may charge a fee for the physical act of transferring a copy, and you may at your option offer warranty protection in exchange for a fee.\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls3\ilvl0\cf0 {\listtext 5. }You may modify your copy or copies of the Program or any portion of it, thus forming a work based on the Program, and copy and distribute such modifications or work under the terms of Section 1 above, provided that you also meet all of these conditions:\ +\pard\tx940\tx1440\pardeftab720\li1440\fi-1440\ql\qnatural +\ls3\ilvl1\cf0 {\listtext 1. }You must cause the modified files to carry prominent notices stating that you changed the files and the date of any change.\ +{\listtext 2. }You must cause any work that you distribute or publish, that in whole or in part contains or is derived from the Program or any part thereof, to be licensed as a whole at no charge to all third parties under the terms of this License.\ +{\listtext 3. }if the modified program normally reads commands interactively when run, you must cause it, when started running for such interactive use in the most ordinary way, to print or display an announcement including an appropriate copyright notice and a notice that there is no warranty (or else, saying that you provide a warranty) and that users may redistribute the program under these conditions, and telling the user how to view a copy of this License. (Exception: if the Program itself is interactive but does not normally print such an announcement, your work based on the Program is not required to print an announcement.)\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\sa240\ql\qnatural +\ls3\ilvl0\cf0 {\listtext 6. }These requirements apply to the modified work as a whole. If identifiable sections of that work are not derived from the Program, and can be reasonably considered independent and separate works in themselves, then this License, and its terms, do not apply to those sections when you distribute them as separate works. But when you distribute the same sections as part of a whole which is a work based on the Program, the distribution of the whole must be on the terms of this License, whose permissions for other licensees extend to the entire whole, and thus to each and every part regardless of who wrote it.\ +{\listtext 7. }Thus, it is not the intent of this section to claim rights or contest your rights to work written entirely by you; rather, the intent is to exercise the right to control the distribution of derivative or collective works based on the Program.\ +{\listtext 8. }In addition, mere aggregation of another work not based on the Program with the Program (or with a work based on the Program) on a volume of a storage or distribution medium does not bring the other work under the scope of this License.\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls3\ilvl0\cf0 {\listtext 9. }You may copy and distribute the Program (or a work based on it, under Section 2) in object code or executable form under the terms of Sections 1 and 2 above provided that you also do one of the following:\ +\pard\tx940\tx1440\pardeftab720\li1440\fi-1440\ql\qnatural +\ls3\ilvl1\cf0 {\listtext 1. }Accompany it with the complete corresponding machine-readable source code, which must be distributed under the terms of Sections 1 and 2 above on a medium customarily used for software interchange; or,\ +{\listtext 2. }Accompany it with a written offer, valid for at least three years, to give any third party, for a charge no more than your cost of physically performing source distribution, a complete machine-readable copy of the corresponding source code, to be distributed under the terms of Sections 1 and 2 above on a medium customarily used for software interchange; or,\ +{\listtext 3. }Accompany it with the information you received as to the offer to distribute corresponding source code. (This alternative is allowed only for noncommercial distribution and only if you received the program in object code or executable form with such an offer, in accord with Subsection b above.)\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\sa240\ql\qnatural +\ls3\ilvl0\cf0 {\listtext 10. }The source code for a work means the preferred form of the work for making modifications to it. For an executable work, complete source code means all the source code for all modules it contains, plus any associated interface definition files, plus the scripts used to control compilation and installation of the executable. However, as a special exception, the source code distributed need not include anything that is normally distributed (in either source or binary form) with the major components (compiler, kernel, and so on) of the operating system on which the executable runs, unless that component itself accompanies the executable.\ +{\listtext 11. }If distribution of executable or object code is made by offering access to copy from a designated place, then offering equivalent access to copy the source code from the same place counts as distribution of the source code, even though third parties are not compelled to copy the source along with the object code.\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls3\ilvl0\cf0 {\listtext 12. }You may not copy, modify, sublicense, or distribute the Program except as expressly provided under this License. Any attempt otherwise to copy, modify, sublicense or distribute the Program is void, and will automatically terminate your rights under this License. However, parties who have received copies, or rights, from you under this License will not have their licenses terminated so long as such parties remain in full compliance.\ +{\listtext 13. }You are not required to accept this License, since you have not signed it. However, nothing else grants you permission to modify or distribute the Program or its derivative works. These actions are prohibited by law if you do not accept this License. Therefore, by modifying or distributing the Program (or any work based on the Program), you indicate your acceptance of this License to do so, and all its terms and conditions for copying, distributing or modifying the Program or works based on it.\ +{\listtext 14. }Each time you redistribute the Program (or any work based on the Program), the recipient automatically receives a license from the original licensor to copy, distribute or modify the Program subject to these terms and conditions. You may not impose any further restrictions on the recipients' exercise of the rights granted herein. You are not responsible for enforcing compliance by third parties to this License.\ +{\listtext 15. }If, as a consequence of a court judgment or allegation of patent infringement or for any other reason (not limited to patent issues), conditions are imposed on you (whether by court order, agreement or otherwise) that contradict the conditions of this License, they do not excuse you from the conditions of this License. If you cannot distribute so as to satisfy simultaneously your obligations under this License and any other pertinent obligations, then as a consequence you may not distribute the Program at all. For example, if a patent license would not permit royalty-free redistribution of the Program by all those who receive copies directly or indirectly through you, then the only way you could satisfy both it and this License would be to refrain entirely from distribution of the Program.\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\sa240\ql\qnatural +\ls3\ilvl0\cf0 {\listtext 16. }If any portion of this section is held invalid or unenforceable under any particular circumstance, the balance of the section is intended to apply and the section as a whole is intended to apply in other circumstances.\ +{\listtext 17. }It is not the purpose of this section to induce you to infringe any patents or other property right claims or to contest validity of any such claims; this section has the sole purpose of protecting the integrity of the free software distribution system, which is implemented by public license practices. Many people have made generous contributions to the wide range of software distributed through that system in reliance on consistent application of that system; it is up to the author/donor to decide if he or she is willing to distribute software through any other system and a licensee cannot impose that choice.\ +{\listtext 18. }This section is intended to make thoroughly clear what is believed to be a consequence of the rest of this License.\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls3\ilvl0\cf0 {\listtext 19. }If the distribution and/or use of the Program is restricted in certain countries either by patents or by copyrighted interfaces, the original copyright holder who places the Program under this License may add an explicit geographical distribution limitation excluding those countries, so that distribution is permitted only in or among countries not thus excluded. In such case, this License incorporates the limitation as if written in the body of this License.\ +{\listtext 20. }The Free Software Foundation may publish revised and/or new versions of the General Public License from time to time. Such new versions will be similar in spirit to the present version, but may differ in detail to address new problems or concerns.\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\sa240\ql\qnatural +\ls3\ilvl0\cf0 {\listtext 21. }Each version is given a distinguishing version number. If the Program specifies a version number of this License which applies to it and "any later version", you have the option of following the terms and conditions either of that version or of any later version published by the Free Software Foundation. If the Program does not specify a version number of this License, you may choose any version ever published by the Free Software Foundation.\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls3\ilvl0\cf0 {\listtext 22. }If you wish to incorporate parts of the Program into other free programs whose distribution conditions are different, write to the author to ask for permission. For software which is copyrighted by the Free Software Foundation, write to the Free Software Foundation; we sometimes make exceptions for this. Our decision will be guided by the two goals of preserving the free status of all derivatives of our free software and of promoting the sharing and reuse of software generally.\ +\pard\pardeftab720\sa300\ql\qnatural + +\b \cf0 NO WARRANTY\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls4\ilvl0 +\b0 \cf0 {\listtext 1. }BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION.\ +{\listtext 2. }IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.\ +\pard\pardeftab720\sa300\ql\qnatural + +\b \cf0 END OF TERMS AND CONDITIONS\ +How to Apply These Terms to Your New Programs\ +\pard\pardeftab720\sa240\ql\qnatural + +\b0 \cf0 If you develop a new program, and you want it to be of the greatest possible use to the public, the best way to achieve this is to make it free software which everyone can redistribute and change under these terms.\ +To do so, attach the following notices to the program. It is safest to attach them to the start of each source file to most effectively convey the exclusion of warranty; and each file should have at least the "copyright" line and a pointer to where the full notice is found.\ +\pard\pardeftab720\ql\qnatural + +\f3\i \cf0 one line to give the program's name and an idea of what it does. +\f2\i0 \ +Copyright (C) +\f3\i yyyy +\f2\i0 +\f3\i name of author +\f2\i0 \ +\ +This program is free software; you can redistribute it and/or\ +modify it under the terms of the GNU General Public License\ +as published by the Free Software Foundation; either version 2\ +of the License, or (at your option) any later version.\ +\ +This program is distributed in the hope that it will be useful,\ +but WITHOUT ANY WARRANTY; without even the implied warranty of\ +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\ +GNU General Public License for more details.\ +\ +You should have received a copy of the GNU General Public License\ +along with this program; if not, write to the Free Software\ +Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.\ +\pard\pardeftab720\sa240\ql\qnatural + +\f0 \cf0 Also add information on how to contact you by electronic and paper mail.\ +If the program is interactive, make it output a short notice like this when it starts in an interactive mode:\ +\pard\pardeftab720\ql\qnatural + +\f2 \cf0 Gnomovision version 69, Copyright (C) +\f3\i year +\f2\i0 +\f3\i name of author +\f2\i0 \ +Gnomovision comes with ABSOLUTELY NO WARRANTY; for details\ +type `show w'. This is free software, and you are welcome\ +to redistribute it under certain conditions; type `show c' \ +for details.\ +\pard\pardeftab720\sa240\ql\qnatural + +\f0 \cf0 The hypothetical commands +\f2 `show w' +\f0 and +\f2 `show c' +\f0 should show the appropriate parts of the General Public License. Of course, the commands you use may be called something other than +\f2 `show w' +\f0 and +\f2 `show c' +\f0 ; they could even be mouse-clicks or menu items--whatever suits your program.\ +You should also get your employer (if you work as a programmer) or your school, if any, to sign a "copyright disclaimer" for the program, if necessary. Here is a sample; alter the names:\ +\pard\pardeftab720\ql\qnatural + +\f2 \cf0 Yoyodyne, Inc., hereby disclaims all copyright\ +interest in the program `Gnomovision'\ +(which makes passes at compilers) written \ +by James Hacker.\ +\ +\pard\pardeftab720\ql\qnatural + +\f3\i \cf0 signature of Ty Coon +\f2\i0 , 1 April 1989\ +Ty Coon, President of Vice\ +\pard\pardeftab720\sa280\ql\qnatural + +\f0\b\fs36 \cf0 GNU LIBRARY GENERAL PUBLIC LICENSE\ +\pard\pardeftab720\sa240\ql\qnatural + +\b0\fs24 \cf0 Version 2, June 1991\ +\pard\pardeftab720\ql\qnatural + +\f2 \cf0 Copyright (C) 1991 Free Software Foundation, Inc.\ +59 Temple Place - Suite 330, Boston, MA 02111-1307, USA\ +Everyone is permitted to copy and distribute verbatim copies\ +of this license document, but changing it is not allowed.\ +\ +[This is the first released version of the library GPL. It is\ + numbered 2 because it goes with version 2 of the ordinary GPL.]\ +\pard\pardeftab720\sa300\ql\qnatural + +\f0\b \cf0 Preamble\ +\pard\pardeftab720\sa240\ql\qnatural + +\b0 \cf0 The licenses for most software are designed to take away your freedom to share and change it. By contrast, the GNU General Public Licenses are intended to guarantee your freedom to share and change free software--to make sure the software is free for all its users.\ +This license, the Library General Public License, applies to some specially designated Free Software Foundation software, and to any other libraries whose authors decide to use it. You can use it for your libraries, too.\ +When we speak of free software, we are referring to freedom, not price. Our General Public Licenses are designed to make sure that you have the freedom to distribute copies of free software (and charge for this service if you wish), that you receive source code or can get it if you want it, that you can change the software or use pieces of it in new free programs; and that you know you can do these things.\ +To protect your rights, we need to make restrictions that forbid anyone to deny you these rights or to ask you to surrender the rights. These restrictions translate to certain responsibilities for you if you distribute copies of the library, or if you modify it.\ +For example, if you distribute copies of the library, whether gratis or for a fee, you must give the recipients all the rights that we gave you. You must make sure that they, too, receive or can get the source code. If you link a program with the library, you must provide complete object files to the recipients so that they can relink them with the library, after making changes to the library and recompiling it. And you must show them these terms so they know their rights.\ +Our method of protecting your rights has two steps: (1) copyright the library, and (2) offer you this license which gives you legal permission to copy, distribute and/or modify the library.\ +Also, for each distributor's protection, we want to make certain that everyone understands that there is no warranty for this free library. If the library is modified by someone else and passed on, we want its recipients to know that what they have is not the original version, so that any problems introduced by others will not reflect on the original authors' reputations.\ +Finally, any free program is threatened constantly by software patents. We wish to avoid the danger that companies distributing free software will individually obtain patent licenses, thus in effect transforming the program into proprietary software. To prevent this, we have made it clear that any patent must be licensed for everyone's free use or not licensed at all.\ +Most GNU software, including some libraries, is covered by the ordinary GNU General Public License, which was designed for utility programs. This license, the GNU Library General Public License, applies to certain designated libraries. This license is quite different from the ordinary one; be sure to read it in full, and don't assume that anything in it is the same as in the ordinary license.\ +The reason we have a separate public license for some libraries is that they blur the distinction we usually make between modifying or adding to a program and simply using it. Linking a program with a library, without changing the library, is in some sense simply using the library, and is analogous to running a utility program or application program. However, in a textual and legal sense, the linked executable is a combined work, a derivative of the original library, and the ordinary General Public License treats it as such.\ +Because of this blurred distinction, using the ordinary General Public License for libraries did not effectively promote software sharing, because most developers did not use the libraries. We concluded that weaker conditions might promote sharing better.\ +However, unrestricted linking of non-free programs would deprive the users of those programs of all benefit from the free status of the libraries themselves. This Library General Public License is intended to permit developers of non-free programs to use free libraries, while preserving your freedom as a user of such programs to change the free libraries that are incorporated in them. (We have not seen how to achieve this as regards changes in header files, but we have achieved it as regards changes in the actual functions of the Library.) The hope is that this will lead to faster development of free libraries.\ +The precise terms and conditions for copying, distribution and modification follow. Pay close attention to the difference between a "work based on the library" and a "work that uses the library". The former contains code derived from the library, while the latter only works together with the library.\ +Note that it is possible for a library to be covered by the ordinary General Public License rather than by this special one.\ +\pard\pardeftab720\sa300\ql\qnatural + +\b \cf0 TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION\ +\pard\pardeftab720\sa240\ql\qnatural +\cf0 0. +\b0 This License Agreement applies to any software library which contains a notice placed by the copyright holder or other authorized party saying it may be distributed under the terms of this Library General Public License (also called "this License"). Each licensee is addressed as "you".\ +A "library" means a collection of software functions and/or data prepared so as to be conveniently linked with application programs (which use some of those functions and data) to form executables.\ +The "Library", below, refers to any such software library or work which has been distributed under these terms. A "work based on the Library" means either the Library or any derivative work under copyright law: that is to say, a work containing the Library or a portion of it, either verbatim or with modifications and/or translated straightforwardly into another language. (Hereinafter, translation is included without limitation in the term "modification".)\ +"Source code" for a work means the preferred form of the work for making modifications to it. For a library, complete source code means all the source code for all modules it contains, plus any associated interface definition files, plus the scripts used to control compilation and installation of the library.\ +Activities other than copying, distribution and modification are not covered by this License; they are outside its scope. The act of running a program using the Library is not restricted, and output from such a program is covered only if its contents constitute a work based on the Library (independent of the use of the Library in a tool for writing it). Whether that is true depends on what the Library does and what the program that uses the Library does.\ + +\b 1. +\b0 You may copy and distribute verbatim copies of the Library's complete source code as you receive it, in any medium, provided that you conspicuously and appropriately publish on each copy an appropriate copyright notice and disclaimer of warranty; keep intact all the notices that refer to this License and to the absence of any warranty; and distribute a copy of this License along with the Library.\ +You may charge a fee for the physical act of transferring a copy, and you may at your option offer warranty protection in exchange for a fee.\ + +\b 2. +\b0 You may modify your copy or copies of the Library or any portion of it, thus forming a work based on the Library, and copy and distribute such modifications or work under the terms of Section 1 above, provided that you also meet all of these conditions:\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls5\ilvl0\cf0 {\listtext 1. }The modified work must itself be a software library.\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\sa240\ql\qnatural +\ls5\ilvl0\cf0 {\listtext 2. }\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls5\ilvl0\cf0 {\listtext 3. }You must cause the files modified to carry prominent notices stating that you changed the files and the date of any change.\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\sa240\ql\qnatural +\ls5\ilvl0\cf0 {\listtext 4. }\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls5\ilvl0\cf0 {\listtext 5. }You must cause the whole of the work to be licensed at no charge to all third parties under the terms of this License.\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\sa240\ql\qnatural +\ls5\ilvl0\cf0 {\listtext 6. }\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls5\ilvl0\cf0 {\listtext 7. }If a facility in the modified Library refers to a function or a table of data to be supplied by an application program that uses the facility, other than as an argument passed when the facility is invoked, then you must make a good faith effort to ensure that, in the event an application does not supply such function or table, the facility still operates, and performs whatever part of its purpose remains meaningful.\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\sa240\ql\qnatural +\ls5\ilvl0\cf0 {\listtext 8. }(For example, a function in a library to compute square roots has a purpose that is entirely well-defined independent of the application. Therefore, Subsection 2d requires that any application-supplied function or table used by this function must be optional: if the application does not supply it, the square root function must still compute square roots.)\ +\pard\pardeftab720\sa240\ql\qnatural +\cf0 These requirements apply to the modified work as a whole. If identifiable sections of that work are not derived from the Library, and can be reasonably considered independent and separate works in themselves, then this License, and its terms, do not apply to those sections when you distribute them as separate works. But when you distribute the same sections as part of a whole which is a work based on the Library, the distribution of the whole must be on the terms of this License, whose permissions for other licensees extend to the entire whole, and thus to each and every part regardless of who wrote it.\ +Thus, it is not the intent of this section to claim rights or contest your rights to work written entirely by you; rather, the intent is to exercise the right to control the distribution of derivative or collective works based on the Library.\ +In addition, mere aggregation of another work not based on the Library with the Library (or with a work based on the Library) on a volume of a storage or distribution medium does not bring the other work under the scope of this License.\ +\pard\pardeftab720\sa240\ql\qnatural + +\b \cf0 3. +\b0 You may opt to apply the terms of the ordinary GNU General Public License instead of this License to a given copy of the Library. To do this, you must alter all the notices that refer to this License, so that they refer to the ordinary GNU General Public License, version 2, instead of to this License. (If a newer version than version 2 of the ordinary GNU General Public License has appeared, then you can specify that version instead if you wish.) Do not make any other change in these notices.\ +Once this change is made in a given copy, it is irreversible for that copy, so the ordinary GNU General Public License applies to all subsequent copies and derivative works made from that copy.\ +This option is useful when you wish to copy part of the code of the Library into a program that is not a library.\ + +\b 4. +\b0 You may copy and distribute the Library (or a portion or derivative of it, under Section 2) in object code or executable form under the terms of Sections 1 and 2 above provided that you accompany it with the complete corresponding machine-readable source code, which must be distributed under the terms of Sections 1 and 2 above on a medium customarily used for software interchange.\ +If distribution of object code is made by offering access to copy from a designated place, then offering equivalent access to copy the source code from the same place satisfies the requirement to distribute the source code, even though third parties are not compelled to copy the source along with the object code.\ + +\b 5. +\b0 A program that contains no derivative of any portion of the Library, but is designed to work with the Library by being compiled or linked with it, is called a "work that uses the Library". Such a work, in isolation, is not a derivative work of the Library, and therefore falls outside the scope of this License.\ +However, linking a "work that uses the Library" with the Library creates an executable that is a derivative of the Library (because it contains portions of the Library), rather than a "work that uses the library". The executable is therefore covered by this License. Section 6 states terms for distribution of such executables.\ +When a "work that uses the Library" uses material from a header file that is part of the Library, the object code for the work may be a derivative work of the Library even though the source code is not. Whether this is true is especially significant if the work can be linked without the Library, or if the work is itself a library. The threshold for this to be true is not precisely defined by law.\ +If such an object file uses only numerical parameters, data structure layouts and accessors, and small macros and small inline functions (ten lines or less in length), then the use of the object file is unrestricted, regardless of whether it is legally a derivative work. (Executables containing this object code plus portions of the Library will still fall under Section 6.)\ +Otherwise, if the work is a derivative of the Library, you may distribute the object code for the work under the terms of Section 6. Any executables containing that work also fall under Section 6, whether or not they are linked directly with the Library itself.\ + +\b 6. +\b0 As an exception to the Sections above, you may also compile or link a "work that uses the Library" with the Library to produce a work containing portions of the Library, and distribute that work under terms of your choice, provided that the terms permit modification of the work for the customer's own use and reverse engineering for debugging such modifications.\ +You must give prominent notice with each copy of the work that the Library is used in it and that the Library and its use are covered by this License. You must supply a copy of this License. If the work during execution displays copyright notices, you must include the copyright notice for the Library among them, as well as a reference directing the user to the copy of this License. Also, you must do one of these things:\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls6\ilvl0\cf0 {\listtext 1. }Accompany the work with the complete corresponding machine-readable source code for the Library including whatever changes were used in the work (which must be distributed under Sections 1 and 2 above); and, if the work is an executable linked with the Library, with the complete machine-readable "work that uses the Library", as object code and/or source code, so that the user can modify the Library and then relink to produce a modified executable containing the modified Library. (It is understood that the user who changes the contents of definitions files in the Library will not necessarily be able to recompile the application to use the modified definitions.)\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\sa240\ql\qnatural +\ls6\ilvl0\cf0 {\listtext 2. }\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls6\ilvl0\cf0 {\listtext 3. }Accompany the work with a written offer, valid for at least three years, to give the same user the materials specified in Subsection 6a, above, for a charge no more than the cost of performing this distribution.\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\sa240\ql\qnatural +\ls6\ilvl0\cf0 {\listtext 4. }\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls6\ilvl0\cf0 {\listtext 5. }If distribution of the work is made by offering access to copy from a designated place, offer equivalent access to copy the above specified materials from the same place.\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\sa240\ql\qnatural +\ls6\ilvl0\cf0 {\listtext 6. }\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls6\ilvl0\cf0 {\listtext 7. }Verify that the user has already received a copy of these materials or that you have already sent this user a copy.\ +\pard\pardeftab720\sa240\ql\qnatural +\cf0 For an executable, the required form of the "work that uses the Library" must include any data and utility programs needed for reproducing the executable from it. However, as a special exception, the source code distributed need not include anything that is normally distributed (in either source or binary form) with the major components (compiler, kernel, and so on) of the operating system on which the executable runs, unless that component itself accompanies the executable.\ +It may happen that this requirement contradicts the license restrictions of other proprietary libraries that do not normally accompany the operating system. Such a contradiction means you cannot use both them and the Library together in an executable that you distribute.\ +\pard\pardeftab720\sa240\ql\qnatural + +\b \cf0 7. +\b0 You may place library facilities that are a work based on the Library side-by-side in a single library together with other library facilities not covered by this License, and distribute such a combined library, provided that the separate distribution of the work based on the Library and of the other library facilities is otherwise permitted, and provided that you do these two things:\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls7\ilvl0\cf0 {\listtext 1. }Accompany the combined library with a copy of the same work based on the Library, uncombined with any other library facilities. This must be distributed under the terms of the Sections above.\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\sa240\ql\qnatural +\ls7\ilvl0\cf0 {\listtext 2. }\ +\pard\tx220\tx720\pardeftab720\li720\fi-720\ql\qnatural +\ls7\ilvl0\cf0 {\listtext 3. }Give prominent notice with the combined library of the fact that part of it is a work based on the Library, and explaining where to find the accompanying uncombined form of the same work.\ +\pard\pardeftab720\sa240\ql\qnatural + +\b \cf0 8. +\b0 You may not copy, modify, sublicense, link with, or distribute the Library except as expressly provided under this License. Any attempt otherwise to copy, modify, sublicense, link with, or distribute the Library is void, and will automatically terminate your rights under this License. However, parties who have received copies, or rights, from you under this License will not have their licenses terminated so long as such parties remain in full compliance.\ + +\b 9. +\b0 You are not required to accept this License, since you have not signed it. However, nothing else grants you permission to modify or distribute the Library or its derivative works. These actions are prohibited by law if you do not accept this License. Therefore, by modifying or distributing the Library (or any work based on the Library), you indicate your acceptance of this License to do so, and all its terms and conditions for copying, distributing or modifying the Library or works based on it.\ + +\b 10. +\b0 Each time you redistribute the Library (or any work based on the Library), the recipient automatically receives a license from the original licensor to copy, distribute, link with or modify the Library subject to these terms and conditions. You may not impose any further restrictions on the recipients' exercise of the rights granted herein. You are not responsible for enforcing compliance by third parties to this License.\ + +\b 11. +\b0 If, as a consequence of a court judgment or allegation of patent infringement or for any other reason (not limited to patent issues), conditions are imposed on you (whether by court order, agreement or otherwise) that contradict the conditions of this License, they do not excuse you from the conditions of this License. If you cannot distribute so as to satisfy simultaneously your obligations under this License and any other pertinent obligations, then as a consequence you may not distribute the Library at all. For example, if a patent license would not permit royalty-free redistribution of the Library by all those who receive copies directly or indirectly through you, then the only way you could satisfy both it and this License would be to refrain entirely from distribution of the Library.\ +If any portion of this section is held invalid or unenforceable under any particular circumstance, the balance of the section is intended to apply, and the section as a whole is intended to apply in other circumstances.\ +It is not the purpose of this section to induce you to infringe any patents or other property right claims or to contest validity of any such claims; this section has the sole purpose of protecting the integrity of the free software distribution system which is implemented by public license practices. Many people have made generous contributions to the wide range of software distributed through that system in reliance on consistent application of that system; it is up to the author/donor to decide if he or she is willing to distribute software through any other system and a licensee cannot impose that choice.\ +This section is intended to make thoroughly clear what is believed to be a consequence of the rest of this License.\ + +\b 12. +\b0 If the distribution and/or use of the Library is restricted in certain countries either by patents or by copyrighted interfaces, the original copyright holder who places the Library under this License may add an explicit geographical distribution limitation excluding those countries, so that distribution is permitted only in or among countries not thus excluded. In such case, this License incorporates the limitation as if written in the body of this License.\ + +\b 13. +\b0 The Free Software Foundation may publish revised and/or new versions of the Library General Public License from time to time. Such new versions will be similar in spirit to the present version, but may differ in detail to address new problems or concerns.\ +Each version is given a distinguishing version number. If the Library specifies a version number of this License which applies to it and "any later version", you have the option of following the terms and conditions either of that version or of any later version published by the Free Software Foundation. If the Library does not specify a license version number, you may choose any version ever published by the Free Software Foundation.\ + +\b 14. +\b0 If you wish to incorporate parts of the Library into other free programs whose distribution conditions are incompatible with these, write to the author to ask for permission. For software which is copyrighted by the Free Software Foundation, write to the Free Software Foundation; we sometimes make exceptions for this. Our decision will be guided by the two goals of preserving the free status of all derivatives of our free software and of promoting the sharing and reuse of software generally.\ + +\b NO WARRANTY +\b0 \ + +\b 15. +\b0 BECAUSE THE LIBRARY IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY FOR THE LIBRARY, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES PROVIDE THE LIBRARY "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE LIBRARY IS WITH YOU. SHOULD THE LIBRARY PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION.\ + +\b 16. +\b0 IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR REDISTRIBUTE THE LIBRARY AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE LIBRARY (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A FAILURE OF THE LIBRARY TO OPERATE WITH ANY OTHER SOFTWARE), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.\ +\pard\pardeftab720\sa300\ql\qnatural + +\b \cf0 END OF TERMS AND CONDITIONS\ +How to Apply These Terms to Your New Libraries\ +\pard\pardeftab720\sa240\ql\qnatural + +\b0 \cf0 If you develop a new library, and you want it to be of the greatest possible use to the public, we recommend making it free software that everyone can redistribute and change. You can do so by permitting redistribution under these terms (or, alternatively, under the terms of the ordinary General Public License).\ +To apply these terms, attach the following notices to the library. It is safest to attach them to the start of each source file to most effectively convey the exclusion of warranty; and each file should have at least the "copyright" line and a pointer to where the full notice is found.\ +\pard\pardeftab720\ql\qnatural + +\f3\i \cf0 one line to give the library's name and an idea of what it does. +\f2\i0 \ +Copyright (C) +\f3\i year +\f2\i0 +\f3\i name of author +\f2\i0 \ +\ +This library is free software; you can redistribute it and/or\ +modify it under the terms of the GNU Lesser General Public\ +License as published by the Free Software Foundation; either\ +version 2.1 of the License, or (at your option) any later version.\ +\ +This library is distributed in the hope that it will be useful,\ +but WITHOUT ANY WARRANTY; without even the implied warranty of\ +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU\ +Lesser General Public License for more details.\ +\ +You should have received a copy of the GNU Lesser General Public\ +License along with this library; if not, write to the Free Software\ +Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA\ +\pard\pardeftab720\sa240\ql\qnatural + +\f0 \cf0 Also add information on how to contact you by electronic and paper mail.\ +You should also get your employer (if you work as a programmer) or your school, if any, to sign a "copyright disclaimer" for the library, if necessary. Here is a sample; alter the names:\ +\pard\pardeftab720\ql\qnatural + +\f2 \cf0 Yoyodyne, Inc., hereby disclaims all copyright interest in\ +the library `Frob' (a library for tweaking knobs) written\ +by James Random Hacker.\ +\ +\pard\pardeftab720\ql\qnatural + +\f3\i \cf0 signature of Ty Coon +\f2\i0 , 1 April 1990\ +Ty Coon, President of Vice\ +\pard\pardeftab720\sa240\ql\qnatural + +\f0 \cf0 That's all there is to it!\ +} \ No newline at end of file diff --git a/packaging/WELCOME.rtf b/packaging/WELCOME.rtf new file mode 100644 index 0000000000..aae1174793 --- /dev/null +++ b/packaging/WELCOME.rtf @@ -0,0 +1,24 @@ +{\rtf1\ansi\ansicpg1252\cocoartf1009\cocoasubrtf300 +{\fonttbl\f0\fswiss\fcharset0 Helvetica;} +{\colortbl;\red255\green255\blue255;} +\margl1440\margr1440\margb1800\margt1800\vieww9000\viewh8400\viewkind0 +\pard\tx720\tx1440\tx2160\tx2880\tx3600\tx4320\tx5040\tx5760\tx6480\tx7200\tx7920\tx8640\ql\qnatural\pardirnatural + +\f0\fs24 \cf0 This program will install CUPS @CUPS_VERSION@, replacing the CUPS software that is included with Mac OS X.\ +\ + +\b WARNING\ + +\b0 \ +This is pre-release software and should not be used in production environments. Because Mac OS X packages cannot be uninstalled, you will need to reinstall Mac OS X to revert to the original CUPS software.\ +\ +Please report all problems using the Bugs & Features page on the CUPS home page:\ +\ + {\field{\*\fldinst{HYPERLINK "http://www.cups.org/str.php"}}{\fldrslt http://www.cups.org/str.php}}\ +\ + +\b NO WARRANTY\ + +\b0 \ +CUPS is provided under the terms of the GNU General Public License and GNU Library General Public License versions 2. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\ +} \ No newline at end of file diff --git a/packaging/cups-desc.plist.in b/packaging/cups-desc.plist.in new file mode 100644 index 0000000000..4d1d296887 --- /dev/null +++ b/packaging/cups-desc.plist.in @@ -0,0 +1,14 @@ + + + + + IFPkgDescriptionDeleteWarning + + IFPkgDescriptionDescription + CUPS provides a portable printing layer for UNIX(r) operating systems. + IFPkgDescriptionTitle + CUPS + IFPkgDescriptionVersion + @CUPS_RELEASE@ + + diff --git a/packaging/cups-info.plist.in b/packaging/cups-info.plist.in new file mode 100644 index 0000000000..d1f61348b6 --- /dev/null +++ b/packaging/cups-info.plist.in @@ -0,0 +1,26 @@ + + + + + IFPkgFlagAuthorizationAction + RootAuthorization + IFPkgFlagBackgroundAlignment + bottomleft + IFPkgFlagBackgroundScaling + none + IFPkgFormatVersion + 0.1 + CFBundleIdentifier + org.cups.cups + CFBundleName + CUPS + CFBundleGetInfoString + CUPS, @CUPS_VERSION@ + CFBundleShortVersionString + @CUPS_RELEASE@ + IFPkgFlagAllowBackRev + + IFPkgFlagRootVolumeOnly + + + diff --git a/packaging/cups.list.in b/packaging/cups.list.in new file mode 100644 index 0000000000..1a6e760664 --- /dev/null +++ b/packaging/cups.list.in @@ -0,0 +1,738 @@ +# +# "$Id$" +# +# ESP Package Manager (EPM) file list for CUPS. +# +# Copyright 2007-2012 by Apple Inc. +# Copyright 1997-2007 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +# Product information +%product CUPS +%copyright 2007-2012 by Apple Inc. +%vendor Apple Inc. +#%license LICENSE.txt +%readme LICENSE.txt +%format rpm +# Red Hat and their epochs... +%version 1:@CUPS_VERSION@ +%format !rpm +%version @CUPS_VERSION@ +%format all +%description CUPS is the standards-based, open source printing system developed by +%description Apple Inc. for Mac OS® X and other UNIX®-like operating systems. + +%format rpm +%provides lpd +%provides lpr +%provides LPRng +%replaces lpd +%replaces lpr +%replaces LPRng +%requires cups-libs 1:@CUPS_VERSION@ + +# Replace all of the old localization subpackages from CUPS 1.2/1.3 +%replaces cups-da +%replaces cups-de +%replaces cups-es +%replaces cups-et +%replaces cups-fi +%replaces cups-fr +%replaces cups-he +%replaces cups-id +%replaces cups-it +%replaces cups-ja +%replaces cups-ko +%replaces cups-nl +%replaces cups-no +%replaces cups-pl +%replaces cups-pt +%replaces cups-ru +%replaces cups-sv +%replaces cups-zh + + +%format deb +%provides cupsys +%provides cupsys-client +%provides cupsys-bsd +%requires cups-libs + +# Replace all of the old localization subpackages from CUPS 1.2/1.3 +%replaces cups-da +%replaces cups-de +%replaces cups-es +%replaces cups-et +%replaces cups-fi +%replaces cups-fr +%replaces cups-he +%replaces cups-id +%replaces cups-it +%replaces cups-ja +%replaces cups-ko +%replaces cups-nl +%replaces cups-no +%replaces cups-pl +%replaces cups-pt +%replaces cups-ru +%replaces cups-sv +%replaces cups-zh + + +%format pkg +%replaces SUNWlpmsg LP Alerts +%replaces SUNWlpr LP Print Service, (Root) +%replaces SUNWlps LP Print Service - Server, (Usr) +%replaces SUNWlpu LP Print Service - Client, (Usr) +%replaces SUNWpsu LP Print Server, (Usr) +%replaces SUNWpsr LP Print Server, (Root) +%replaces SUNWpcu LP Print Client, (Usr) +%replaces SUNWpcr LP Print Client, (Root) +%replaces SUNWppm +%replaces SUNWmp +%replaces SUNWscplp SunOS Print Compatibility + +%format inst +%replaces patch*.print_*.* 0 0 1289999999 1289999999 +%replaces maint*.print_*.* 0 0 1289999999 1289999999 +%replaces print 0 0 1289999999 1289999999 +%replaces fw_cups 0 0 1289999999 1289999999 +%incompat patch*.print_*.* 0 0 1289999999 1289999999 +%incompat maint*.print_*.* 0 0 1289999999 1289999999 +%incompat print 0 0 1289999999 1289999999 +%incompat fw_cups 0 0 1289999999 1289999999 +%requires cups.sw.libs + +# Replace all of the old localization subpackages from CUPS 1.2/1.3 +%replaces cups.sw.da +%replaces cups.sw.de +%replaces cups.sw.es +%replaces cups.sw.et +%replaces cups.sw.fi +%replaces cups.sw.fr +%replaces cups.sw.he +%replaces cups.sw.id +%replaces cups.sw.it +%replaces cups.sw.ja +%replaces cups.sw.ko +%replaces cups.sw.nl +%replaces cups.sw.no +%replaces cups.sw.pl +%replaces cups.sw.pt +%replaces cups.sw.ru +%replaces cups.sw.sv +%replaces cups.sw.zh + +%format portable +%requires cups-libs + +# Replace all of the old localization subpackages from CUPS 1.2/1.3 +%replaces cups-da +%replaces cups-de +%replaces cups-es +%replaces cups-et +%replaces cups-fi +%replaces cups-fr +%replaces cups-he +%replaces cups-id +%replaces cups-it +%replaces cups-ja +%replaces cups-ko +%replaces cups-nl +%replaces cups-no +%replaces cups-pl +%replaces cups-pt +%replaces cups-ru +%replaces cups-sv +%replaces cups-zh + +%format all + +%subpackage libs +%description Shared libraries +%format deb +%provides libcups1 +%provides libcupsys2 +%provides libcupsys2-gnutls10 +%provides libcupsimage2 +%format all + +%subpackage devel +%description Development environment +%format deb +%provides libcupsys2-dev +%provides libcupsimage2-dev +%format all + +%subpackage lpd +%description LPD support + +%subpackage + + +# +# GNU variables... +# + +$prefix=@prefix@ +$exec_prefix=@exec_prefix@ +$bindir=@bindir@ +$datarootdir=@datarootdir@ +$datadir=@datadir@ +$includedir=@includedir@ +$infodir=@infodir@ +$libdir=@libdir@ +$libexecdir=@libexecdir@ +$localstatedir=@localstatedir@ +$mandir=@mandir@ +$oldincludedir=@oldincludedir@ +$sbindir=@sbindir@ +$sharedstatedir=@sharedstatedir@ +$srcdir=@srcdir@ +$sysconfdir=@sysconfdir@ +$top_srcdir=@top_srcdir@ + +# +# ESP variables... +# + +$AMANDIR=@AMANDIR@ +$BINDIR=@bindir@ +$CACHEDIR=@CUPS_CACHEDIR@ +$DATADIR=@CUPS_DATADIR@ +$DOCDIR=@CUPS_DOCROOT@ +$INCLUDEDIR=${includedir} +$INITDIR=@INITDIR@ +$INITDDIR=@INITDDIR@ +$LIBDIR=${libdir} +$LOCALEDIR=@CUPS_LOCALEDIR@ +$LOGDIR=@CUPS_LOGDIR@ +$MANDIR=@mandir@ +$PAMDIR=@PAMDIR@ +$PMANDIR=@PMANDIR@ +$REQUESTS=@CUPS_REQUESTS@ +$SBINDIR=@sbindir@ +$SERVERBIN=@CUPS_SERVERBIN@ +$SERVERROOT=@CUPS_SERVERROOT@ +$STATEDIR=@CUPS_STATEDIR@ +$XINETD=@XINETD@ +$LIB32DIR=@LIB32DIR@ +$LIB64DIR=@LIB64DIR@ + +$MDNS=@MDNS@ + +$CUPS_USER=@CUPS_USER@ +$CUPS_GROUP=@CUPS_GROUP@ +$CUPS_PRIMARY_SYSTEM_GROUP=@CUPS_PRIMARY_SYSTEM_GROUP@ +$CUPS_PERM=0@CUPS_CONFIG_FILE_PERM@ + +$INSTALLSTATIC=@INSTALLSTATIC@ + +$MAN1EXT=@MAN1EXT@ +$MAN3EXT=@MAN3EXT@ +$MAN5EXT=@MAN5EXT@ +$MAN7EXT=@MAN7EXT@ +$MAN8EXT=@MAN8EXT@ +$MAN8DIR=@MAN8DIR@ + +$LIBZ=@LIBZ@ +$DSOLIBS=@DSOLIBS@ + +# Make sure the MD5 password file is now owned by CUPS_USER... +%postinstall if test -f $SERVERROOT/passwd.md5; then +%postinstall chown $CUPS_USER $SERVERROOT/passwd.md5 +%postinstall fi + +# Make sure the shared libraries are refreshed... +%subpackage libs +%system linux +%postinstall ldconfig +%system all +%subpackage + +# Server programs +%system all +# Server files +f 0555 root sys $SBINDIR/cupsd scheduler/cupsd + +d 0755 root sys $SERVERBIN - +%system darwin +d 0755 root sys $SERVERBIN/apple - +f 0555 root sys $SERVERBIN/apple/ipp backend/ipp +l 0755 root sys $SERVERBIN/apple/http ipp +l 0755 root sys $SERVERBIN/apple/https ipp +l 0755 root sys $SERVERBIN/apple/ipps ipp +%system all +d 0755 root sys $SERVERBIN/backend - +f 0500 root sys $SERVERBIN/backend/ipp backend/ipp +l 0700 root sys $SERVERBIN/backend/http ipp +l 0700 root sys $SERVERBIN/backend/https ipp +l 0700 root sys $SERVERBIN/backend/ipps ipp +f 0500 root sys $SERVERBIN/backend/lpd backend/lpd +%if DNSSD_BACKEND +f 0500 root sys $SERVERBIN/backend/dnssd backend/dnssd +l 0700 root sys $SERVERBIN/backend/mdns dnssd +%endif +f 0555 root sys $SERVERBIN/backend/snmp backend/snmp +f 0555 root sys $SERVERBIN/backend/socket backend/socket +f 0555 root sys $SERVERBIN/backend/usb backend/usb +d 0755 root sys $SERVERBIN/cgi-bin - +f 0555 root sys $SERVERBIN/cgi-bin/admin.cgi cgi-bin/admin.cgi +f 0555 root sys $SERVERBIN/cgi-bin/classes.cgi cgi-bin/classes.cgi +f 0555 root sys $SERVERBIN/cgi-bin/help.cgi cgi-bin/help.cgi +f 0555 root sys $SERVERBIN/cgi-bin/jobs.cgi cgi-bin/jobs.cgi +f 0555 root sys $SERVERBIN/cgi-bin/printers.cgi cgi-bin/printers.cgi +d 0755 root sys $SERVERBIN/daemon - +f 0555 root sys $SERVERBIN/daemon/cups-deviced scheduler/cups-deviced +f 0555 root sys $SERVERBIN/daemon/cups-driverd scheduler/cups-driverd +f 0555 root sys $SERVERBIN/daemon/cups-polld scheduler/cups-polld +d 0755 root sys $SERVERBIN/driver - +d 0755 root sys $SERVERBIN/filter - +f 0555 root sys $SERVERBIN/filter/commandtops filter/commandtops +f 0555 root sys $SERVERBIN/filter/gziptoany filter/gziptoany +f 0555 root sys $SERVERBIN/filter/pstops filter/pstops +f 0555 root sys $SERVERBIN/filter/rastertolabel filter/rastertolabel +l 0755 root sys $SERVERBIN/filter/rastertodymo rastertolabel +f 0555 root sys $SERVERBIN/filter/rastertoepson filter/rastertoepson +f 0555 root sys $SERVERBIN/filter/rastertohp filter/rastertohp +f 0555 root sys $SERVERBIN/filter/rastertopwg filter/rastertopwg +d 0755 root sys $SERVERBIN/notifier - +f 0555 root sys $SERVERBIN/notifier/mailto notifier/mailto + +%subpackage lpd +d 0755 root sys $SERVERBIN/daemon - +f 0555 root sys $SERVERBIN/daemon/cups-lpd scheduler/cups-lpd +%subpackage + +# Admin commands +d 0755 root sys $BINDIR - +l 0755 root sys $BINDIR/enable $SBINDIR/cupsaccept +l 0755 root sys $BINDIR/disable $SBINDIR/cupsaccept +d 0755 root sys $LIBDIR - +l 0755 root sys $LIBDIR/accept $SBINDIR/cupsaccept +l 0755 root sys $LIBDIR/lpadmin $SBINDIR/lpadmin +l 0755 root sys $LIBDIR/reject $SBINDIR/cupsaccept +d 0755 root sys $SBINDIR - +l 0755 root sys $SBINDIR/accept cupsaccept +f 0555 root sys $SBINDIR/cupsaccept systemv/cupsaccept +f 0555 root sys $SBINDIR/cupsaddsmb systemv/cupsaddsmb +f 0555 root sys $SBINDIR/cupsctl systemv/cupsctl +l 0755 root sys $SBINDIR/cupsdisable accept +l 0755 root sys $SBINDIR/cupsenable accept +l 0755 root sys $SBINDIR/cupsreject accept +f 0555 root sys $SBINDIR/lpadmin systemv/lpadmin +f 0555 root sys $SBINDIR/lpc berkeley/lpc +f 0555 root sys $SBINDIR/lpinfo systemv/lpinfo +f 0555 root sys $SBINDIR/lpmove systemv/lpmove +l 0755 root sys $SBINDIR/reject cupsaccept + +%system irix +l 0755 root sys /usr/etc/lpc $SBINDIR/lpc +%system all + +# User commands +d 0755 root sys $BINDIR - +f 0555 root sys $BINDIR/cancel systemv/cancel +f 0555 root sys $BINDIR/cupstestdsc systemv/cupstestdsc +f 0555 root sys $BINDIR/cupstestppd systemv/cupstestppd +f 0555 root sys $BINDIR/ipptool test/ipptool +f 0555 root sys $BINDIR/lp systemv/lp +f 0555 root sys $BINDIR/lpoptions systemv/lpoptions +f 0555 root sys $BINDIR/lppasswd systemv/lppasswd +f 0555 root sys $BINDIR/lpq berkeley/lpq +f 0555 root sys $BINDIR/lpr berkeley/lpr +f 0555 root sys $BINDIR/lprm berkeley/lprm +f 0555 root sys $BINDIR/lpstat systemv/lpstat + +%system irix +l 0755 root sys /usr/bsd/lpq $BINDIR/lpq +l 0755 root sys /usr/bsd/lpr $BINDIR/lpr +l 0755 root sys /usr/bsd/lprm $BINDIR/lprm +%system all + +# DSOs +%if DSOLIBS +%subpackage libs +%system hpux +f 0555 root sys $LIBDIR/libcups.sl.2 cups/libcups.sl.2 nostrip() +l 0755 root sys $LIBDIR/libcups.sl libcups.sl.2 +f 0555 root sys $LIBDIR/libcupscgi.sl.1 cgi-bin/libcupscgi.sl.1 nostrip() +l 0755 root sys $LIBDIR/libcupscgi.sl libcupscgi.sl.1 +f 0555 root sys $LIBDIR/libcupsdriver.sl.1 driver/libcupsdriver.sl.1 nostrip() +l 0755 root sys $LIBDIR/libcupsdriver.sl libcupsdriver.sl.1 +f 0555 root sys $LIBDIR/libcupsimage.sl.2 filter/libcupsimage.sl.2 nostrip() +l 0755 root sys $LIBDIR/libcupsimage.sl libcupsimage.sl.2 +f 0555 root sys $LIBDIR/libcupsmime.sl.1 scheduler/libcupsmime.sl.1 nostrip() +l 0755 root sys $LIBDIR/libcupsmime.sl libcupsmime.sl.1 +f 0555 root sys $LIBDIR/libcupsppdc.sl.1 ppdc/libcupsppdc.sl.1 nostrip() +l 0755 root sys $LIBDIR/libcupsppdc.sl libcupsppdc.sl.1 +%system aix +f 0555 root sys $LIBDIR/libcups_s.a cups/libcups_s.a nostrip() +f 0555 root sys $LIBDIR/libcupscgi_s.a cgi-bin/libcupscgi_s.a nostrip() +f 0555 root sys $LIBDIR/libcupsdriver_s.a driver/libcupsdriver_s.a nostrip() +f 0555 root sys $LIBDIR/libcupsimage_s.a filter/libcupsimage_s.a nostrip() +f 0555 root sys $LIBDIR/libcupsmime_s.a scheduler/libcupsmime_s.a nostrip() +f 0555 root sys $LIBDIR/libcupsppdc_s.a ppdc/libcupsppdc_s.a nostrip() +%system darwin +f 0555 root sys $LIBDIR/libcups.2.dylib cups/libcups.2.dylib nostrip() +l 0755 root sys $LIBDIR/libcups.dylib libcups.2.dylib +f 0555 root sys $LIBDIR/libcupscgi.1.dylib cgi-bin/libcupscgi.1.dylib nostrip() +l 0755 root sys $LIBDIR/libcupscgi.dylib libcupscgi.1.dylib +f 0555 root sys $LIBDIR/libcupsdriver.1.dylib driver/libcupsdriver.1.dylib nostrip() +l 0755 root sys $LIBDIR/libcupsdriver.dylib libcupsdriver.1.dylib +f 0555 root sys $LIBDIR/libcupsimage.2.dylib filter/libcupsimage.2.dylib nostrip() +l 0755 root sys $LIBDIR/libcupsimage.dylib libcupsimage.2.dylib +f 0555 root sys $LIBDIR/libcupsmime.1.dylib scheduler/libcupsmime.1.dylib nostrip() +l 0755 root sys $LIBDIR/libcupsmime.dylib libcupsmime.1.dylib +f 0555 root sys $LIBDIR/libcupsppdc.1.dylib ppdc/libcupsppdc.1.dylib nostrip() +l 0755 root sys $LIBDIR/libcupsppdc.dylib libcupsppdc.1.dylib +%system !hpux !aix !darwin +f 0555 root sys $LIBDIR/libcups.so.2 cups/libcups.so.2 nostrip() +l 0755 root sys $LIBDIR/libcups.so libcups.so.2 +f 0555 root sys $LIBDIR/libcupscgi.so.1 cgi-bin/libcupscgi.so.1 nostrip() +l 0755 root sys $LIBDIR/libcupscgi.so libcupscgi.so.1 +f 0555 root sys $LIBDIR/libcupsdriver.so.1 driver/libcupsdriver.so.1 nostrip() +l 0755 root sys $LIBDIR/libcupsdriver.so libcupsdriver.so.1 +f 0555 root sys $LIBDIR/libcupsimage.so.2 filter/libcupsimage.so.2 nostrip() +l 0755 root sys $LIBDIR/libcupsimage.so libcupsimage.so.2 +f 0555 root sys $LIBDIR/libcupsmime.so.1 scheduler/libcupsmime.so.1 nostrip() +l 0755 root sys $LIBDIR/libcupsmime.so libcupsmime.so.1 +f 0555 root sys $LIBDIR/libcupsppdc.so.1 ppdc/libcupsppdc.so.1 nostrip() +l 0755 root sys $LIBDIR/libcupsppdc.so libcupsppdc.so.1 +%system all +%subpackage +%endif + +# Directories +d 0755 root sys $LOGDIR - +d 0710 root $CUPS_GROUP $REQUESTS - +d 1770 root $CUPS_GROUP $REQUESTS/tmp - +d 0775 root $CUPS_GROUP $CACHEDIR - +d 0775 root $CUPS_GROUP $CACHEDIR/rss - +#d 0755 root $CUPS_GROUP $CACHEDIR/ppd - +d 0755 root $CUPS_GROUP $STATEDIR - +d 0511 root $CUPS_PRIMARY_SYSTEM_GROUP $STATEDIR/certs - + +# Data files +f 0444 root sys $LOCALEDIR/da/cups_da.po locale/cups_da.po +f 0444 root sys $LOCALEDIR/de/cups_de.po locale/cups_de.po +f 0444 root sys $LOCALEDIR/es/cups_es.po locale/cups_es.po +#f 0444 root sys $LOCALEDIR/et/cups_et.po locale/cups_et.po +f 0444 root sys $LOCALEDIR/eu/cups_eu.po locale/cups_eu.po +f 0444 root sys $LOCALEDIR/fi/cups_fi.po locale/cups_fi.po +f 0444 root sys $LOCALEDIR/fr/cups_fr.po locale/cups_fr.po +#f 0444 root sys $LOCALEDIR/he/cups_he.po locale/cups_he.po +f 0444 root sys $LOCALEDIR/id/cups_id.po locale/cups_id.po +f 0444 root sys $LOCALEDIR/it/cups_it.po locale/cups_it.po +f 0444 root sys $LOCALEDIR/ja/cups_ja.po locale/cups_ja.po +f 0444 root sys $LOCALEDIR/ko/cups_ko.po locale/cups_ko.po +f 0444 root sys $LOCALEDIR/nl/cups_nl.po locale/cups_nl.po +f 0444 root sys $LOCALEDIR/no/cups_no.po locale/cups_no.po +f 0444 root sys $LOCALEDIR/pl/cups_pl.po locale/cups_pl.po +f 0444 root sys $LOCALEDIR/pt/cups_pt.po locale/cups_pt.po +f 0444 root sys $LOCALEDIR/pt_BR/cups_pt_BR.po locale/cups_pt_BR.po +f 0444 root sys $LOCALEDIR/ru/cups_ru.po locale/cups_ru.po +f 0444 root sys $LOCALEDIR/sv/cups_sv.po locale/cups_sv.po +f 0444 root sys $LOCALEDIR/zh/cups_zh.po locale/cups_zh.po +f 0444 root sys $LOCALEDIR/zh_TW/cups_zh_TW.po locale/cups_zh_TW.po + +d 0755 root sys $DATADIR - + +d 0755 root sys $DATADIR/banners - +f 0444 root sys $DATADIR/banners/classified data/classified +f 0444 root sys $DATADIR/banners/confidential data/confidential +f 0444 root sys $DATADIR/banners/secret data/secret +f 0444 root sys $DATADIR/banners/standard data/standard +f 0444 root sys $DATADIR/banners/topsecret data/topsecret +f 0444 root sys $DATADIR/banners/unclassified data/unclassified + +d 0755 root sys $DATADIR/data - +f 0444 root sys $DATADIR/data/testprint data/testprint + +d 0755 root sys $DATADIR/drv - +f 0444 root sys $DATADIR/drv/sample.drv ppdc/sample.drv + +d 0755 root sys $DATADIR/examples - +f 0444 root sys $DATADIR/examples examples/*.drv + +d 0755 root sys $DATADIR/ipptool - +f 0444 root sys $DATADIR/ipptool/create-printer-subscription.test test/create-printer-subscription.test +f 0444 root sys $DATADIR/ipptool/get-completed-jobs.test test/get-completed-jobs.test +f 0444 root sys $DATADIR/ipptool/get-jobs.test test/get-jobs.test +f 0444 root sys $DATADIR/ipptool test/ipp-*.test +f 0444 root sys $DATADIR/ipptool test/testfile.* + +d 0755 root sys $DATADIR/mime - +f 0444 root sys $DATADIR/mime/mime.convs conf/mime.convs +f 0444 root sys $DATADIR/mime/mime.types conf/mime.types + +d 0755 root sys $DATADIR/model - + +d 0755 root sys $DATADIR/ppdc - +f 0444 root sys $DATADIR/ppdc data/*.defs +f 0444 root sys $DATADIR/ppdc data/*.h + +d 0755 root sys $DATADIR/templates - +f 0444 root sys $DATADIR/templates templates/*.tmpl + +## Template files +d 0755 root sys $DATADIR/templates/de +f 0444 root sys $DATADIR/templates/de templates/de/*.tmpl + +d 0755 root sys $DATADIR/templates/es +f 0444 root sys $DATADIR/templates/es templates/es/*.tmpl + +#d 0755 root sys $DATADIR/templates/et +#f 0444 root sys $DATADIR/templates/et templates/et/*.tmpl + +d 0755 root sys $DATADIR/templates/eu +f 0444 root sys $DATADIR/templates/eu templates/eu/*.tmpl + +#d 0755 root sys $DATADIR/templates/fr +#f 0444 root sys $DATADIR/templates/fr templates/fr/*.tmpl + +#d 0755 root sys $DATADIR/templates/he +#f 0444 root sys $DATADIR/templates/he templates/he/*.tmpl + +d 0755 root sys $DATADIR/templates/id +f 0444 root sys $DATADIR/templates/id templates/id/*.tmpl + +d 0755 root sys $DATADIR/templates/it +f 0444 root sys $DATADIR/templates/it templates/it/*.tmpl + +d 0755 root sys $DATADIR/templates/ja +f 0444 root sys $DATADIR/templates/ja templates/ja/*.tmpl + +d 0755 root sys $DATADIR/templates/pl +f 0444 root sys $DATADIR/templates/pl templates/pl/*.tmpl + +d 0755 root sys $DATADIR/templates/ru +f 0444 root sys $DATADIR/templates/ru templates/ru/*.tmpl + +#d 0755 root sys $DATADIR/templates/sv +#f 0444 root sys $DATADIR/templates/sv templates/sv/*.tmpl + +#d 0755 root sys $DATADIR/templates/zh_TW +#f 0444 root sys $DATADIR/templates/zh_TW templates/zh_TW/*.tmpl + +# Config files +d 0755 root sys $SERVERROOT - +d 0755 root $CUPS_GROUP $SERVERROOT/interfaces - +d 0755 root $CUPS_GROUP $SERVERROOT/ppd - +d 0700 root $CUPS_GROUP $SERVERROOT/ssl - +c $CUPS_PERM root $CUPS_GROUP $SERVERROOT/cupsd.conf conf/cupsd.conf +f $CUPS_PERM root $CUPS_GROUP $SERVERROOT/cupsd.conf.default conf/cupsd.conf +c $CUPS_PERM root $CUPS_GROUP $SERVERROOT/snmp.conf conf/snmp.conf + +%if PAMDIR +d 0755 root sys $PAMDIR - +c 0644 root sys $PAMDIR/cups conf/@PAMFILE@ +%endif + +%subpackage devel +# Developer files +f 0555 root sys $BINDIR/cups-config cups-config +d 0755 root sys $INCLUDEDIR/cups - +f 0444 root sys $INCLUDEDIR/cups/adminutil.h cups/adminutil.h +f 0444 root sys $INCLUDEDIR/cups/array.h cups/array.h +f 0444 root sys $INCLUDEDIR/cups/backend.h cups/backend.h +f 0444 root sys $INCLUDEDIR/cups/cups.h cups/cups.h +f 0444 root sys $INCLUDEDIR/cups/dir.h cups/dir.h +f 0444 root sys $INCLUDEDIR/cups/file.h cups/file.h +f 0444 root sys $INCLUDEDIR/cups/http.h cups/http.h +f 0444 root sys $INCLUDEDIR/cups/ipp.h cups/ipp.h +f 0444 root sys $INCLUDEDIR/cups/mime.h scheduler/mime.h +f 0444 root sys $INCLUDEDIR/cups/language.h cups/language.h +f 0444 root sys $INCLUDEDIR/cups/ppd.h cups/ppd.h +f 0444 root sys $INCLUDEDIR/cups/raster.h cups/raster.h +f 0444 root sys $INCLUDEDIR/cups/transcode.h cups/transcode.h + +%if INSTALLSTATIC +f 0444 root sys $LIBDIR/libcups.a cups/libcups.a +f 0444 root sys $LIBDIR/libcupscgi.a cgi-bin/libcupscgi.a +f 0444 root sys $LIBDIR/libcupsdriver.a driver/libcupsdriver.a +f 0444 root sys $LIBDIR/libcupsimage.a filter/libcupsimage.a +f 0444 root sys $LIBDIR/libcupsmime.a scheduler/libcupsmime.a +f 0444 root sys $LIBDIR/libcupsppdc.a ppdc/libcupsppdc.a +%endif + +d 0755 root sys $DOCDIR/help - +f 0444 root sys $DOCDIR/help doc/help/api*.html +f 0444 root sys $DOCDIR/help/postscript-driver.html doc/help/postscript-driver.html +f 0444 root sys $DOCDIR/help/ppd-compiler.html doc/help/ppd-compiler.html +f 0444 root sys $DOCDIR/help/raster-driver.html doc/help/raster-driver.html +f 0444 root sys $DOCDIR/help doc/help/spec*.html +%subpackage + +# Documentation files +d 0755 root sys $DOCDIR - +f 0444 root sys $DOCDIR doc/*.css +f 0444 root sys $DOCDIR doc/*.html +d 0755 root sys $DOCDIR/help - +f 0444 root sys $DOCDIR/help/accounting.html doc/help/accounting.html +f 0444 root sys $DOCDIR/help/cgi.html doc/help/cgi.html +f 0444 root sys $DOCDIR/help/glossary.html doc/help/glossary.html +f 0444 root sys $DOCDIR/help/kerberos.html doc/help/kerberos.html +f 0444 root sys $DOCDIR/help/license.html doc/help/license.html +f 0444 root sys $DOCDIR/help/network.html doc/help/network.html +f 0444 root sys $DOCDIR/help/options.html doc/help/options.html +f 0444 root sys $DOCDIR/help/overview.html doc/help/overview.html +f 0444 root sys $DOCDIR/help/policies.html doc/help/policies.html +f 0444 root sys $DOCDIR/help/security.html doc/help/security.html +f 0444 root sys $DOCDIR/help/sharing.html doc/help/sharing.html +f 0444 root sys $DOCDIR/help/standard.html doc/help/standard.html +f 0444 root sys $DOCDIR/help/translation.html doc/help/translation.html +f 0444 root sys $DOCDIR/help/whatsnew.html doc/help/whatsnew.html +f 0444 root sys $DOCDIR/help doc/help/man-*.html +f 0444 root sys $DOCDIR/help doc/help/ref-*.html +d 0755 root sys $DOCDIR/images - +f 0444 root sys $DOCDIR/images doc/images/*.gif +f 0444 root sys $DOCDIR/images doc/images/*.jpg +f 0444 root sys $DOCDIR/images doc/images/*.png +f 0444 root sys $DOCDIR/robots.txt doc/robots.txt + +# Localized documentation files +d 0755 root sys $DOCDIR/de +f 0444 root sys $DOCDIR/de doc/de/*.html + +d 0755 root sys $DOCDIR/es +f 0444 root sys $DOCDIR/es doc/es/*.html + +#d 0755 root sys $DOCDIR/et +#f 0444 root sys $DOCDIR/et doc/et/*.html + +d 0755 root sys $DOCDIR/eu +f 0444 root sys $DOCDIR/eu doc/eu/*.html + +#d 0755 root sys $DOCDIR/fr +#f 0444 root sys $DOCDIR/fr doc/fr/*.html + +#d 0755 root sys $DOCDIR/he +#f 0444 root sys $DOCDIR/he doc/he/*.html +#f 0444 root sys $DOCDIR/he/cups.css doc/he/cups.css + +d 0755 root sys $DOCDIR/id +f 0444 root sys $DOCDIR/id doc/id/*.html + +d 0755 root sys $DOCDIR/it +f 0444 root sys $DOCDIR/it doc/it/*.html + +d 0755 root sys $DOCDIR/ja +f 0444 root sys $DOCDIR/ja doc/ja/*.html + +d 0755 root sys $DOCDIR/pl +f 0444 root sys $DOCDIR/pl doc/pl/*.html + +d 0755 root sys $DOCDIR/ru +f 0444 root sys $DOCDIR/ru doc/ru/*.html + +#d 0755 root sys $DOCDIR/sv +#f 0444 root sys $DOCDIR/sv doc/sv/*.html + +#d 0755 root sys $DOCDIR/zh_TW +#f 0444 root sys $DOCDIR/zh_TW doc/zh_TW/*.html + +# Man pages +d 0755 root sys $AMANDIR - +d 0755 root sys $AMANDIR/man$MAN8DIR - +d 0755 root sys $MANDIR - +d 0755 root sys $MANDIR/man1 - +d 0755 root sys $MANDIR/man5 - +d 0755 root sys $MANDIR/man7 - + +f 0444 root sys $MANDIR/man1/cancel.$MAN1EXT man/cancel.$MAN1EXT +f 0444 root sys $MANDIR/man1/cupstestdsc.$MAN1EXT man/cupstestdsc.$MAN1EXT +f 0444 root sys $MANDIR/man1/cupstestppd.$MAN1EXT man/cupstestppd.$MAN1EXT +f 0444 root sys $MANDIR/man1/ipptool.$MAN1EXT man/ipptool.$MAN1EXT +f 0444 root sys $MANDIR/man1/lpoptions.$MAN1EXT man/lpoptions.$MAN1EXT +f 0444 root sys $MANDIR/man1/lppasswd.$MAN1EXT man/lppasswd.$MAN1EXT +f 0444 root sys $MANDIR/man1/lpq.$MAN1EXT man/lpq.$MAN1EXT +f 0444 root sys $MANDIR/man1/lprm.$MAN1EXT man/lprm.$MAN1EXT +f 0444 root sys $MANDIR/man1/lpr.$MAN1EXT man/lpr.$MAN1EXT +f 0444 root sys $MANDIR/man1/lpstat.$MAN1EXT man/lpstat.$MAN1EXT +f 0444 root sys $MANDIR/man1/lp.$MAN1EXT man/lp.$MAN1EXT + +f 0444 root sys $MANDIR/man5/classes.conf.$MAN5EXT man/classes.conf.$MAN5EXT +f 0444 root sys $MANDIR/man5/cupsd.conf.$MAN5EXT man/cupsd.conf.$MAN5EXT +f 0444 root sys $MANDIR/man5/ipptoolfile.$MAN5EXT man/ipptoolfile.$MAN5EXT +f 0444 root sys $MANDIR/man5/mailto.conf.$MAN5EXT man/mailto.conf.$MAN5EXT +f 0444 root sys $MANDIR/man5/mime.convs.$MAN5EXT man/mime.convs.$MAN5EXT +f 0444 root sys $MANDIR/man5/mime.types.$MAN5EXT man/mime.types.$MAN5EXT +f 0444 root sys $MANDIR/man5/printers.conf.$MAN5EXT man/printers.conf.$MAN5EXT + +l 0644 root sys $AMANDIR/man$MAN8DIR/accept.$MAN8EXT cupsaccept.$MAN8EXT +f 0444 root sys $AMANDIR/man$MAN8DIR/cupsaccept.$MAN8EXT man/cupsaccept.$MAN8EXT +l 0644 root sys $AMANDIR/man$MAN8DIR/cupsreject.$MAN8EXT cupsaccept.$MAN8EXT +f 0444 root sys $AMANDIR/man$MAN8DIR/cupsaddsmb.$MAN8EXT man/cupsaddsmb.$MAN8EXT +f 0444 root sys $AMANDIR/man$MAN8DIR/cupsctl.$MAN8EXT man/cupsctl.$MAN8EXT +f 0444 root sys $AMANDIR/man$MAN8DIR/cupsfilter.$MAN8EXT man/cupsfilter.$MAN8EXT +f 0444 root sys $AMANDIR/man$MAN8DIR/cups-polld.$MAN8EXT man/cups-polld.$MAN8EXT +f 0444 root sys $AMANDIR/man$MAN8DIR/cupsd.$MAN8EXT man/cupsd.$MAN8EXT +f 0444 root sys $AMANDIR/man$MAN8DIR/cupsenable.$MAN8EXT man/cupsenable.$MAN8EXT +l 0644 root sys $AMANDIR/man$MAN8DIR/cupsdisable.$MAN8EXT cupsenable.$MAN8EXT +l 0644 root sys $AMANDIR/man$MAN8DIR/disable.$MAN8EXT cupsenable.$MAN8EXT +l 0644 root sys $AMANDIR/man$MAN8DIR/enable.$MAN8EXT cupsenable.$MAN8EXT +f 0444 root sys $AMANDIR/man$MAN8DIR/lpadmin.$MAN8EXT man/lpadmin.$MAN8EXT +f 0444 root sys $AMANDIR/man$MAN8DIR/lpc.$MAN8EXT man/lpc.$MAN8EXT +f 0444 root sys $AMANDIR/man$MAN8DIR/lpinfo.$MAN8EXT man/lpinfo.$MAN8EXT +f 0444 root sys $AMANDIR/man$MAN8DIR/lpmove.$MAN8EXT man/lpmove.$MAN8EXT +l 0644 root sys $AMANDIR/man$MAN8DIR/reject.$MAN8EXT cupsaccept.$MAN8EXT + +%subpackage devel +f 0444 root sys $MANDIR/man1/cups-config.$MAN1EXT man/cups-config.$MAN1EXT +f 0444 root sys $MANDIR/man1/ man/ppd*.$MAN1EXT +f 0444 root sys $MANDIR/man5/ppdcfile.$MAN5EXT man/ppdcfile.$MAN5EXT +f 0444 root sys $MANDIR/man7/backend.$MAN7EXT man/backend.$MAN7EXT +f 0444 root sys $MANDIR/man7/filter.$MAN7EXT man/filter.$MAN7EXT +f 0444 root sys $MANDIR/man7/notifier.$MAN7EXT man/notifier.$MAN7EXT + +%subpackage lpd +d 0755 root sys $AMANDIR/man$MAN8DIR - +f 0444 root sys $AMANDIR/man$MAN8DIR/cups-lpd.$MAN8EXT man/cups-lpd.$MAN8EXT +%subpackage + +# Startup scripts +%system darwin +f 0444 root sys /System/Library/LaunchDaemons/org.cups.cupsd.plist scheduler/org.cups.cupsd.plist +%preremove <. +# +# Copyright 2007-2011 by Apple Inc. +# Copyright 1999-2007 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +# Conditional build options (--with name/--without name): +# +# dbus - Enable/disable DBUS support (default = enable) + +%{!?_with_dbus: %{!?_without_dbus: %define _with_dbus --with-dbus}} +%{?_with_dbus: %define _dbus --enable-dbus} +%{!?_with_dbus: %define _dbus --disable-dbus} + +%{!?_with_static: %{!?_without_static: %define _without_static --without-static}} +%{?_with_static: %define _static --enable-static} +%{!?_with_static: %define _static --disable-static} + +Summary: CUPS +Name: cups +Version: @CUPS_VERSION@ +Release: 0 +Epoch: 1 +License: GPL +Group: System Environment/Daemons +Source: http://ftp.easysw.com/pub/cups/%{version}/cups-%{version}-source.tar.bz2 +Url: http://www.cups.org +Packager: Anonymous +Vendor: Apple Inc. + +# Use buildroot so as not to disturb the version already installed +BuildRoot: /tmp/%{name}-root + +# Dependencies... +Requires: %{name}-libs = %{epoch}:%{version} +Obsoletes: lpd, lpr, LPRng +Provides: lpd, lpr, LPRng +Obsoletes: cups-da, cups-de, cups-es, cups-et, cups-fi, cups-fr, cups-he +Obsoletes: cups-id, cups-it, cups-ja, cups-ko, cups-nl, cups-no, cups-pl +Obsoletes: cups-pt, cups-ru, cups-sv, cups-zh + +%package devel +Summary: CUPS - development environment +Group: Development/Libraries +Requires: %{name}-libs = %{epoch}:%{version} + +%package libs +Summary: CUPS - shared libraries +Group: System Environment/Libraries +Provides: libcups1 + +%package lpd +Summary: CUPS - LPD support +Group: System Environment/Daemons +Requires: %{name} = %{epoch}:%{version} xinetd + +%description +CUPS is the standards-based, open source printing system developed by +Apple Inc. for Mac OS® X and other UNIX®-like operating systems. + +%description devel +This package provides the CUPS headers and development environment. + +%description libs +This package provides the CUPS shared libraries. + +%description lpd +This package provides LPD client support. + +%prep +%setup + +%build +CFLAGS="$RPM_OPT_FLAGS" CXXFLAGS="$RPM_OPT_FLAGS" LDFLAGS="$RPM_OPT_FLAGS" \ + ./configure %{_dbus} %{_static} +# If we got this far, all prerequisite libraries must be here. +make + +%install +# Make sure the RPM_BUILD_ROOT directory exists. +rm -rf $RPM_BUILD_ROOT + +make BUILDROOT=$RPM_BUILD_ROOT install + +%post +/sbin/chkconfig --add cups +/sbin/chkconfig cups on + +# Restart cupsd if we are upgrading... +if test $1 -gt 1; then + /sbin/service cups stop + /sbin/service cups start +fi + +%post libs +/sbin/ldconfig + +%preun +if test $1 = 0; then + /sbin/service cups stop + /sbin/chkconfig --del cups +fi + +%postun +if test $1 -ge 1; then + /sbin/service cups stop + /sbin/service cups start +fi + +%postun libs +/sbin/ldconfig + +%clean +rm -rf $RPM_BUILD_ROOT + +%files +%docdir /usr/share/doc/cups +%defattr(-,root,root) +%dir /etc/cups +%config(noreplace) /etc/cups/*.conf +/etc/cups/cupsd.conf.default +%dir /etc/cups/interfaces +%dir /etc/cups/ppd +%attr(0700,root,root) %dir /etc/cups/ssl + +%if %{?_with_dbus:1}%{!?_with_dbus:0} +# DBUS +/etc/dbus-1/system.d/* +%endif + +# PAM +%dir /etc/pam.d +/etc/pam.d/* + +# RC dirs are a pain under Linux... Uncomment the appropriate ones if you +# don't use Red Hat or Mandrake... + +/etc/init.d/* +/etc/rc0.d/* +/etc/rc2.d/* +/etc/rc3.d/* +/etc/rc5.d/* + +# OLD RedHat/Mandrake +#/etc/rc.d/init.d/* +#/etc/rc.d/rc0.d/* +#/etc/rc.d/rc2.d/* +#/etc/rc.d/rc3.d/* +#/etc/rc.d/rc5.d/* + +#/sbin/rc.d/* +#/sbin/rc.d/rc0.d/* +#/sbin/rc.d/rc2.d/* +#/sbin/rc.d/rc3.d/* +#/sbin/rc.d/rc5.d/* + +/usr/bin/cancel +/usr/bin/cupstestdsc +/usr/bin/cupstestppd +/usr/bin/ipptool +/usr/bin/lp* +%dir /usr/lib/cups +%dir /usr/lib/cups/backend +/usr/lib/cups/backend/http +/usr/lib/cups/backend/https +%attr(0700,root,root) /usr/lib/cups/backend/ipp +/usr/lib/cups/backend/ipps +%attr(0700,root,root) /usr/lib/cups/backend/lpd +/usr/lib/cups/backend/snmp +/usr/lib/cups/backend/socket +/usr/lib/cups/backend/usb +%dir /usr/lib/cups/cgi-bin +/usr/lib/cups/cgi-bin/* +%dir /usr/lib/cups/daemon +/usr/lib/cups/daemon/cups-deviced +/usr/lib/cups/daemon/cups-driverd +/usr/lib/cups/daemon/cups-exec +/usr/lib/cups/daemon/cups-polld +%dir /usr/lib/cups/driver +%dir /usr/lib/cups/filter +/usr/lib/cups/filter/* +%dir /usr/lib/cups/monitor +/usr/lib/cups/monitor/* +%dir /usr/lib/cups/notifier +/usr/lib/cups/notifier/* + +/usr/sbin/* +%dir /usr/share/cups +%dir /usr/share/cups/banners +/usr/share/cups/banners/* +%dir /usr/share/cups/charsets +/usr/share/cups/charsets/* +%dir /usr/share/cups/data +/usr/share/cups/data/* +%dir /usr/share/cups/drv +/usr/share/cups/drv/* +%dir /usr/share/cups/fonts +/usr/share/cups/fonts/* +%dir /usr/share/cups/ipptool +/usr/share/cups/ipptool/* +%dir /usr/share/cups/mime +/usr/share/cups/mime/* +%dir /usr/share/cups/model +%dir /usr/share/cups/ppdc +/usr/share/cups/ppdc/* +%dir /usr/share/cups/templates +/usr/share/cups/templates/* +%dir /usr/share/doc/cups +/usr/share/doc/cups/*.* +%dir /usr/share/doc/cups/de +/usr/share/doc/cups/de/* +%dir /usr/share/doc/cups/es +/usr/share/doc/cups/es/* +%dir /usr/share/doc/cups/eu +/usr/share/doc/cups/eu/* +%dir /usr/share/doc/cups/id +/usr/share/doc/cups/id/* +%dir /usr/share/doc/cups/it +/usr/share/doc/cups/it/* +%dir /usr/share/doc/cups/ja +/usr/share/doc/cups/ja/* +%dir /usr/share/doc/cups/pl +/usr/share/doc/cups/pl/* +%dir /usr/share/doc/cups/ru +/usr/share/doc/cups/ru/* +%dir /usr/share/doc/cups/help +/usr/share/doc/cups/help/accounting.html +/usr/share/doc/cups/help/cgi.html +/usr/share/doc/cups/help/glossary.html +/usr/share/doc/cups/help/kerberos.html +/usr/share/doc/cups/help/license.html +/usr/share/doc/cups/help/man-*.html +/usr/share/doc/cups/help/network.html +/usr/share/doc/cups/help/options.html +/usr/share/doc/cups/help/overview.html +/usr/share/doc/cups/help/policies.html +/usr/share/doc/cups/help/ref-*.html +/usr/share/doc/cups/help/security.html +/usr/share/doc/cups/help/sharing.html +/usr/share/doc/cups/help/standard.html +/usr/share/doc/cups/help/translation.html +/usr/share/doc/cups/help/whatsnew.html +%dir /usr/share/doc/cups/images +/usr/share/doc/cups/images/* +/usr/share/locale/* + +%dir /usr/share/man/man1 +/usr/share/man/man1/cancel.1.gz +/usr/share/man/man1/cupstestdsc.1.gz +/usr/share/man/man1/cupstestppd.1.gz +/usr/share/man/man1/ipptool.1.gz +/usr/share/man/man1/lp.1.gz +/usr/share/man/man1/lpoptions.1.gz +/usr/share/man/man1/lppasswd.1.gz +/usr/share/man/man1/lpq.1.gz +/usr/share/man/man1/lpr.1.gz +/usr/share/man/man1/lprm.1.gz +/usr/share/man/man1/lpstat.1.gz +%dir /usr/share/man/man5 +/usr/share/man/man5/*.conf.5.gz +/usr/share/man/man5/ipptoolfile.5.gz +/usr/share/man/man5/mime.*.5.gz +%dir /usr/share/man/man8 +/usr/share/man/man8/accept.8.gz +/usr/share/man/man8/cupsaddsmb.8.gz +/usr/share/man/man8/cupsaccept.8.gz +/usr/share/man/man8/cupsctl.8.gz +/usr/share/man/man8/cupsfilter.8.gz +/usr/share/man/man8/cupsd.8.gz +/usr/share/man/man8/cupsdisable.8.gz +/usr/share/man/man8/cupsenable.8.gz +/usr/share/man/man8/cupsreject.8.gz +/usr/share/man/man8/cups-deviced.8.gz +/usr/share/man/man8/cups-driverd.8.gz +/usr/share/man/man8/cups-polld.8.gz +/usr/share/man/man8/lpadmin.8.gz +/usr/share/man/man8/lpc.8.gz +/usr/share/man/man8/lpinfo.8.gz +/usr/share/man/man8/lpmove.8.gz +/usr/share/man/man8/reject.8.gz + +%dir /var/cache/cups +%attr(0775,root,sys) %dir /var/cache/cups/rss +%dir /var/log/cups +%dir /var/run/cups +%attr(0711,lp,sys) %dir /var/run/cups/certs +%attr(0710,lp,sys) %dir /var/spool/cups +%attr(1770,lp,sys) %dir /var/spool/cups/tmp + +# Desktop files +/usr/share/applications/* +/usr/share/icons/* + +%files devel +%defattr(-,root,root) +%dir /usr/share/cups/examples +/usr/share/cups/examples/* +%dir /usr/share/man/man1 +/usr/share/man/man1/cups-config.1.gz +/usr/share/man/man1/ppd*.1.gz +%dir /usr/share/man/man5 +/usr/share/man/man5/ppdcfile.5.gz +/usr/share/man/man7/backend.7.gz +/usr/share/man/man7/filter.7.gz +/usr/share/man/man7/notifier.7.gz + +/usr/bin/cups-config +/usr/bin/ppd* +%dir /usr/include/cups +/usr/include/cups/* +/usr/lib*/*.so + +%if %{?_with_static:1}%{!?_with_static:0} +/usr/lib*/*.a +%endif + +%dir /usr/share/doc/cups/help +/usr/share/doc/cups/help/api*.html +/usr/share/doc/cups/help/postscript-driver.html +/usr/share/doc/cups/help/ppd-compiler.html +/usr/share/doc/cups/help/raster-driver.html +/usr/share/doc/cups/help/spec*.html + +%files libs +%defattr(-,root,root) +/usr/lib*/*.so.* + +%files lpd +%defattr(-,root,root) +/etc/xinetd.d/cups-lpd +%dir /usr/lib/cups +%dir /usr/lib/cups/daemon +/usr/lib/cups/daemon/cups-lpd +%dir /usr/share/man/man8 +/usr/share/man/man8/cups-lpd.8.gz + + +# +# End of "$Id$". +# diff --git a/packaging/installer.gif b/packaging/installer.gif new file mode 100644 index 0000000000000000000000000000000000000000..3ae6ea77a8c02cd6dc5e5e2a90d4fe4d495076a9 GIT binary patch literal 3392 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 YpH<-Rh5!FZVTAstvH$M>ar_?t2g=;5{{R30 literal 0 Hc-jL100001 diff --git a/ppdc/Dependencies b/ppdc/Dependencies new file mode 100644 index 0000000000..660b67c83b --- /dev/null +++ b/ppdc/Dependencies @@ -0,0 +1,208 @@ +ppdc-array.o: ppdc-array.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc-attr.o: ppdc-attr.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc-catalog.o: ppdc-catalog.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc-choice.o: ppdc-choice.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc-constraint.o: ppdc-constraint.cxx ppdc-private.h ppdc.h \ + ../cups/file.h ../cups/versioning.h ../cups/cups-private.h \ + ../cups/cups.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc-driver.o: ppdc-driver.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc-file.o: ppdc-file.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc-filter.o: ppdc-filter.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc-font.o: ppdc-font.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc-group.o: ppdc-group.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc-import.o: ppdc-import.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc-mediasize.o: ppdc-mediasize.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc-message.o: ppdc-message.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc-option.o: ppdc-option.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc-profile.o: ppdc-profile.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc-shared.o: ppdc-shared.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc-source.o: ppdc-source.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h ../cups/raster.h ../data/epson.h ../data/hp.h \ + ../data/label.h +ppdc-string.o: ppdc-string.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc-variable.o: ppdc-variable.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +genstrings.o: genstrings.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdc.o: ppdc.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdhtml.o: ppdhtml.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdi.o: ppdi.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +ppdmerge.o: ppdmerge.cxx ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h +ppdpo.o: ppdpo.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +testcatalog.o: testcatalog.cxx ppdc-private.h ppdc.h ../cups/file.h \ + ../cups/versioning.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h diff --git a/ppdc/Makefile b/ppdc/Makefile new file mode 100644 index 0000000000..c1fbb8f671 --- /dev/null +++ b/ppdc/Makefile @@ -0,0 +1,404 @@ +# +# "$Id$" +# +# Makefile for the CUPS PPD Compiler. +# +# Copyright 2007-2012 by Apple Inc. +# Copyright 2002-2006 by Easy Software Products. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +# +# Include standard definitions... +# + +include ../Makedefs + + +# +# Object files... +# + +LIBOBJS = \ + ppdc-array.o \ + ppdc-attr.o \ + ppdc-catalog.o \ + ppdc-choice.o \ + ppdc-constraint.o \ + ppdc-driver.o \ + ppdc-file.o \ + ppdc-filter.o \ + ppdc-font.o \ + ppdc-group.o \ + ppdc-import.o \ + ppdc-mediasize.o \ + ppdc-message.o \ + ppdc-option.o \ + ppdc-profile.o \ + ppdc-shared.o \ + ppdc-source.o \ + ppdc-string.o \ + ppdc-variable.o +OBJS = \ + $(LIBOBJS) \ + genstrings.o \ + ppdc.o \ + ppdhtml.o \ + ppdi.o \ + ppdmerge.o \ + ppdpo.o \ + testcatalog.o +LIBTARGETS = \ + $(LIBCUPSPPDC) \ + libcupsppdc.a +UNITTARGETS = \ + ppdc-static \ + ppdi-static \ + testcatalog +EXECTARGETS = \ + ppdc \ + ppdhtml \ + ppdi \ + ppdmerge \ + ppdpo + +TARGETS = \ + $(LIBTARGETS) \ + $(EXECTARGETS) \ + genstrings + + +# +# Make everything... +# + +all: $(TARGETS) + + +# +# Make library targets... +# + +libs: $(LIBTARGETS) + + +# +# Make unit tests... +# + +unittests: $(UNITTARGETS) + + +# +# Clean everything... +# + +clean: + $(RM) $(OBJS) core + $(RM) *.bak *.bck core.* + $(RM) $(TARGETS) $(UNITTARGETS) + $(RM) -r ppd ppd2 + $(RM) sample-import.drv sample.c test.drv + $(RM) libcupsppdc.so libcupsppdc.sl libcupsppdc.dylib + + +# +# Update dependencies... +# + +depend: + $(CXX) -MM $(ALL_CXXFLAGS) $(OBJS:.o=.cxx) >Dependencies + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: + $(INSTALL_DIR) $(DATADIR)/drv + $(INSTALL_DATA) sample.drv $(DATADIR)/drv + + +# +# Install programs... +# + +install-exec: + echo Installing PPD compiler programs... + $(INSTALL_DIR) $(BINDIR) + for file in $(EXECTARGETS); do \ + $(INSTALL_BIN) $$file $(BINDIR); \ + done + if test "x$(SYMROOT)" != "x"; then \ + $(INSTALL_DIR) $(SYMROOT); \ + for file in $(EXECTARGETS); do \ + cp $$file $(SYMROOT); \ + done \ + fi + + +# +# Install headers... +# + +install-headers: + echo Installing header files in $(INCLUDEDIR)/cups... + $(INSTALL_DIR) -m 755 $(INCLUDEDIR)/cups + $(INSTALL_DATA) ppdc.h $(INCLUDEDIR)/cups + + +# +# Install libraries... +# + +install-libs: $(INSTALLSTATIC) + echo Installing libraries in $(LIBDIR)... + $(INSTALL_DIR) -m 755 $(LIBDIR) + $(INSTALL_LIB) $(LIBCUPSPPDC) $(LIBDIR) + if test $(LIBCUPSPPDC) = "libcupsppdc.so.1" -o $(LIBCUPSPPDC) = "libcupsppdc.sl.1"; then \ + $(RM) $(LIBDIR)/`basename $(LIBCUPSPPDC) .1`; \ + $(LN) $(LIBCUPSPPDC) $(LIBDIR)/`basename $(LIBCUPSPPDC) .1`; \ + fi + if test $(LIBCUPSPPDC) = "libcupsppdc.1.dylib"; then \ + $(RM) $(LIBDIR)/libcupsppdc.dylib; \ + $(LN) $(LIBCUPSPPDC) $(LIBDIR)/libcupsppdc.dylib; \ + fi + if test "x$(SYMROOT)" != "x"; then \ + $(INSTALL_DIR) $(SYMROOT); \ + cp $(LIBCUPSPPDC) $(SYMROOT); \ + fi + +installstatic: + $(INSTALL_DIR) -m 755 $(LIBDIR) + $(INSTALL_LIB) -m 755 libcupsppdc.a $(LIBDIR) + $(RANLIB) $(LIBDIR)/libcupsppdc.a + $(CHMOD) 555 $(LIBDIR)/libcupsppdc.a + + +# +# Uninstall... +# + +uninstall: + for file in $(EXECTARGETS); do \ + $(RM) $(BINDIR)/$$file; \ + done + $(RM) $(DATADIR)/drv/sample.drv + $(RMDIR) $(DATADIR)/drv + $(RM) $(LIBDIR)/libcupsppdc.1.dylib + $(RM) $(LIBDIR)/libcupsppdc.a + $(RM) $(LIBDIR)/libcupsppdc.dylib + $(RM) $(LIBDIR)/libcupsppdc_s.a + $(RM) $(LIBDIR)/libcupsppdc.sl + $(RM) $(LIBDIR)/libcupsppdc.sl.1 + $(RM) $(LIBDIR)/libcupsppdc.so + $(RM) $(LIBDIR)/libcupsppdc.so.1 + -$(RMDIR) $(LIBDIR) + $(RM) $(INCLUDEDIR)/cups/ppdc.h + -$(RMDIR) $(INCLUDEDIR)/cups + + +# +# Automatic API help files... +# + +apihelp: + mxmldoc --section "Programming" \ + --title "PPD Compiler API" \ + --css ../doc/cups-printable.css \ + --header api-ppdc.header --intro api-ppdc.shtml \ + api-ppdc.xml \ + ppdc.h $(LIBOBJS:.o=.cxx) >../doc/help/api-ppdc.html + mxmldoc --tokens help/api-ppdc.html api-ppdc.xml >../doc/help/api-ppdc.tokens + $(RM) api-ppdc.xml + +framedhelp: + mxmldoc --framed api-ppdc \ + --section "Programming" \ + --title "PPD Compiler API" \ + --css ../doc/cups-printable.css \ + --header api-ppdc.header --intro api-ppdc.shtml \ + ppdc.h $(LIBOBJS:.o=.cxx) + + +# +# genstrings - generate GNU gettext strings. +# + +genstrings: genstrings.o libcupsppdc.a ../cups/$(LIBCUPSSTATIC) \ + sample.drv ../data/media.defs + echo Linking $@... + $(CXX) $(ARCHFLAGS) $(LDFLAGS) -o genstrings genstrings.o \ + libcupsppdc.a ../cups/$(LIBCUPSSTATIC) $(LIBGSSAPI) $(SSLLIBS) \ + $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + echo Generating localization strings... + ./genstrings >sample.c + + +# +# ppdc, the PPD compiler. +# + +ppdc: ppdc.o $(LIBCUPSPPDC) ../cups/$(LIBCUPS) + echo Linking $@... + $(CXX) $(LDFLAGS) -o $@ ppdc.o -L. -lcupsppdc $(LIBS) + + +ppdc-static: ppdc.o libcupsppdc.a ../cups/$(LIBCUPSSTATIC) foo.drv foo-fr.po + echo Linking $@... + $(CXX) $(ARCHFLAGS) $(LDFLAGS) -o ppdc-static ppdc.o libcupsppdc.a \ + ../cups/$(LIBCUPSSTATIC) $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) \ + $(COMMONLIBS) $(LIBZ) + echo Testing PPD compiler... + ./ppdc-static -l en,fr -I ../data foo.drv + ./ppdc-static -l en,fr -z -I ../data foo.drv + + +# +# ppdhtml, the PPD to HTML utility. +# + +ppdhtml: ppdhtml.o $(LIBCUPSPPDC) ../cups/$(LIBCUPS) + echo Linking $@... + $(CXX) $(LDFLAGS) -o $@ ppdhtml.o -L. -lcupsppdc $(LIBS) + + +# +# ppdi, import PPD files. +# + +ppdi: ppdi.o $(LIBCUPSPPDC) ../cups/$(LIBCUPS) + echo Linking $@... + $(CXX) $(LDFLAGS) -o $@ ppdi.o -L. -lcupsppdc $(LIBS) + + +ppdi-static: ppdc-static ppdi.o libcupsppdc.a ../cups/$(LIBCUPSSTATIC) + echo Linking $@... + $(CXX) $(ARCHFLAGS) $(LDFLAGS) -o ppdi-static ppdi.o libcupsppdc.a \ + ../cups/$(LIBCUPSSTATIC) $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) \ + $(COMMONLIBS) $(LIBZ) + echo Testing PPD importer... + $(RM) -r ppd ppd2 sample-import.drv + ./ppdc-static -I ../data sample.drv + ./ppdi-static -I ../data -o sample-import.drv ppd/* + ./ppdc-static -I ../data -d ppd2 sample-import.drv + if diff -r ppd ppd2 >/dev/null; then \ + echo PPD import OK; \ + else \ + echo PPD import FAILED; \ + exit 1; \ + fi + + +# +# ppdmerge, merge PPD files. +# + +ppdmerge: ppdmerge.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CXX) $(LDFLAGS) -o $@ ppdmerge.o $(LIBS) + + +# +# ppdpo, create message catalog files. +# + +ppdpo: ppdpo.o $(LIBCUPSPPDC) ../cups/$(LIBCUPS) + echo Linking $@... + $(CXX) $(LDFLAGS) -o $@ ppdpo.o -L. -lcupsppdc $(LIBS) + + +# +# testcatalog, test ppdcCatalog class. +# + +testcatalog: testcatalog.o libcupsppdc.a ../cups/$(LIBCUPSSTATIC) + echo Linking $@... + $(CXX) $(LDFLAGS) -o $@ testcatalog.o libcupsppdc.a \ + ../cups/$(LIBCUPSSTATIC) $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) \ + $(COMMONLIBS) $(LIBZ) + + +# +# libcupsppdc.so.1, libcupsppdc.sl.1 +# + +libcupsppdc.so.1 libcupsppdc.sl.1: $(LIBOBJS) ../cups/$(LIBCUPS) + echo Linking $@... + $(DSOXX) $(ARCHFLAGS) $(DSOFLAGS) -o $@ $(LIBOBJS) $(LINKCUPS) + $(RM) `basename $@ .1` + $(LN) $@ `basename $@ .1` + + +# +# libcupsppdc.1.dylib +# + +libcupsppdc.1.dylib: $(LIBOBJS) ../cups/$(LIBCUPS) + echo Creating export list for $@... + nm $(LIBOBJS) | grep "T __" | awk '{print $$3}' | sort >t.exp + echo Linking $@... + $(DSOXX) $(ARCHFLAGS) $(DSOFLAGS) -o $@ \ + -install_name $(libdir)/$@ \ + -current_version 1.0.0 \ + -compatibility_version 1.0.0 \ + -exported_symbols_list t.exp \ + $(LIBOBJS) $(LINKCUPS) + $(RM) libcupsppdc.dylib t.exp + $(LN) $@ libcupsppdc.dylib + + +# +# libcupsppdc_s.a +# + +libcupsppdc_s.a: $(LIBOBJS) ../cups/$(LIBCUPS) + echo Creating $@... + $(DSOXX) $(DSOFLAGS) -o libcupsppdc_s.o $(LIBOBJS) $(LINKCUPS) + $(RM) $@ + $(AR) $(ARFLAGS) $@ libcupsppdc_s.o + + +# +# libcupsppdc.la +# + +libcupsppdc.la: $(LIBOBJS) ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(ARCHFLAGS) $(DSOFLAGS) -o $@ $(LIBOBJS:.o=.lo) -rpath $(LIBDIR) \ + -version-info 1:0 $(LINKCUPS) + + +# +# libcupsppdc.a +# + +libcupsppdc.a: $(LIBOBJS) + echo Archiving $@... + $(RM) $@ + $(AR) $(ARFLAGS) $@ $(LIBOBJS) + $(RANLIB) $@ + + +# +# Include dependencies... +# + +include Dependencies + + +# +# End of "$Id$". +# diff --git a/ppdc/api-ppdc.header b/ppdc/api-ppdc.header new file mode 100644 index 0000000000..418686560e --- /dev/null +++ b/ppdc/api-ppdc.header @@ -0,0 +1,34 @@ + + +

PPD Compiler API

+ +
+ + + + + + + + + + + + + + + + +
Headercups/ppdc.h
Library-lcupsppdc
See AlsoProgramming: Introduction to CUPS Programming
diff --git a/ppdc/api-ppdc.shtml b/ppdc/api-ppdc.shtml new file mode 100644 index 0000000000..e305dcb9d0 --- /dev/null +++ b/ppdc/api-ppdc.shtml @@ -0,0 +1,18 @@ + + +

Overview

+ +

The PPD Compiler API provides access to CUPS driver information files and +methods for generating and importing PPD files.

diff --git a/ppdc/foo-fr.po b/ppdc/foo-fr.po new file mode 100644 index 0000000000..1b1561b395 --- /dev/null +++ b/ppdc/foo-fr.po @@ -0,0 +1,11 @@ +msgid "A Serious Error" +msgstr "La Error Serious" + +msgid "http://foo.com/serious.html" +msgstr "http://foo.com/fr/serious.html" + +msgid "Foo Letter" +msgstr "La Foo Letter" + +msgid "Foo Photo" +msgstr "La Foo Photo" diff --git a/ppdc/foo.drv b/ppdc/foo.drv new file mode 100644 index 0000000000..24a7e04bb2 --- /dev/null +++ b/ppdc/foo.drv @@ -0,0 +1,547 @@ +// +// "$Id$" +// +// PPD file compiler test data file for CUPS. +// +// Copyright 2007-2011 by Apple Inc. +// Copyright 1997-2003 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// + +/* + * C-style comments are supported. + */ + +// +// C++-style comments are supported. +// + +// +// Include the common media size definitions... +// +// #include directives support both to include a standard file +// and "name" or just name without the quotes for a local file. Local +// files resolve relative to the current file's path. Unlike C/C++, +// #include does not look in multiple directories, and +// #include "name" does not look in the standard directory. +// + +#include + + +// +// Include the CUPS raster definitions... +// + +#include + + +// +// Include the standard CUPS fonts... +// + +#include + + +// +// Define variables using the #define directive. In this case we +// are defining constants for the model number, which is used by +// our imaginary rastertofoo filter to determine which model-specific +// features to use/support. +// + +#define MODEL_BW 0 +#define MODEL_COLOR 1 + +#define MODEL_LASER 0 +#define MODEL_PHOTO 2 + + +// +// Media sizes are defined using the #media directive. The order of +// values is: size name/text, width, length. +// +// "Size name" is an alphanumeric string of up to 40 characters as +// defined by the Adobe PPD specification. +// +// "Size text" is a text string of up to 80 characters as defined by +// the Adobe PPD specification. +// +// "Width" and "length" are the width and length of the media size. +// Numbers by themselves represent points (72 points = 1 inch). The +// suffixes "cm", "ft", "in", "m", "mm", and "pt" are recognized to +// specify centimeters, feet, inches, meters, millimeters, and points, +// respectively. +// + +#media "FooLetter/Foo Letter" 8in 10in +#media "FooPhoto/Foo Photo" 200mm 300mm + + +// +// Message catalogs can be included using #po... +// + +#po fr foo-fr.po + + +// +// Specify that the drivers use all of the standard base fonts... +// + +Font * + + +// +// All copyright lines are put at the top of the PPD file in order +// of their appearance. Copyright text can span multiple lines and +// will be properly included in the PPD file with comment prefixes +// on each line. +// +// First an MIT-style copyright/license notice... +// + +Copyright "Copyright 2007 by Foo Industries." +Copyright " +Permission is granted for redistribution of this file as long as +this copyright notice is intact and the contents of the file are +not altered in any way from their original form. + +Permission is hereby granted, free of charge, to any person +obtaining a copy of this software and associated documentation +files (the \"Software\"), to deal in the Software without +restriction, including without limitation the rights to use, +copy, modify, merge, publish, distribute, sublicense, and/or +sell copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. +" + +// +// Then a GPL notice... +// + +Copyright "Copyright 2007 by Foo Industries." +Copyright " +This software is free software; you can redistribute it and/or +modify it under the terms of the GNU General Public License as +published by the Free Software Foundation; either version 2 of +the License, or (at your option) any later version. + +This software is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +You should have received a copy of the GNU General Public +License along with this software; if not, write to the Free +Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, +MA 02111 USA +" + + +// +// All printer drivers must define the manufacturer, model, PC +// filename, and version strings; since this test file contains +// drivers for an imaginary manufacturer "Foo", all of the drivers +// listed in this file share common manufacturer and version +// strings. +// + +Manufacturer "Foo" +Version 1.0 + + +// +// Printer drivers can access driver-specific attributes in a PPD +// file; these attributes are specified using lines of the form: +// +// Attribute name selector value +// +// "Name" is the name of the attribute and should start with either +// the "cups" prefix or the name of the vendor, e.g. "hpFoo", +// "epsonBar", etc. The name can be any alphanumeric character (a-z, +// A-Z, and 0-9) and cannot be a common prefix of another attribute, +// e.g. "fooLines" and "fooLinesPerInch" cannot be in the same file. +// +// "Selector" is a selector string containing any characters except +// colon (:). Typically this will be one or more keywords separated +// by the forward slash (/), however the empty string ("") can be used +// to omit the selector. +// +// "Value" is a quoted value string that can contain any printable +// characters except the double quote ("). Hexadecimal numbers +// inside angle brackets () can be used to substitute escape +// codes and other special characters. +// + +Attribute fooOutputFormat "" "PCL" +Attribute fooPJL Begin "<1B>%-12345X@PJL<0D0A>" +Attribute fooPJL Enter/PCL "@PJL ENTER LANGUAGE=PCL<0D0A>" +Attribute fooPJL End "<1B>%-12345X@PJL END JOB<0D0A>" + +// +// Most printer drivers use filters; exceptions include PostScript +// printers and PPD files for software RIPs. +// +// The format is: +// +// Filter mime-type cost program +// +// The "mime-type" field defines the MIME type that the filter program +// accepts; for CUPS raster printer drivers, this will be +// "application/vnd.cups-raster". +// +// The "cost" field defines the relative cost of the filter in terms of +// both CPU and memory usage, and is used to limit the number of +// simultaneous jobs in some configurations. Most raster filters should +// have a cost of 100, unless the filter does no dithering - then a cost +// of 33 is more appropriate. +// +// The "program" field defined the filter program to run; use the null +// filter "-" to define a MIME type that the printer accepts directly. +// If no path information is provided, then the program will be run +// from the standard CUPS filter directory, usually +// /usr/lib/cups/filter. +// +// When compiling PPD files for PostScript capable devices that use +// additional filters, add a null filter for the MIME type +// "application/vnd.cups-postscript" so that printer commands, user +// job filters, and page markings can be added to the PostScript +// output that is sent to the printer. +// + +Filter application/vnd.cups-raster 100 rastertofoo + + +// +// Attributes are included thusly... +// + +Attribute cupsIPPReason "com.foo-serious-error/A Serious Error" "http://foo.com/serious.html" + + +// +// Curley braces are used for grouping common data and for isolating +// individual printer models. All data values are inherited *except* +// for the PCFilename and ModelName strings. +// + +{ + // + // Define two printer drivers that support only the FooLetter and + // FooPhoto media size. One is color, the other is black-and-white. + // + // Both printers share two MediaSize definitions; the name listed + // after the MediaSize keyword must be one of the Adobe standard + // names listed in the PPD specification or any named size defined + // using the #media directive. + // + // Default options are indicated by placing an asterisk (*) before + // the keyword. + // + // For custom size and margin specification, see the next group of + // printer drivers. + // + + MediaSize FooLetter + *MediaSize FooPhoto + + + // + // These imaginary printers support printing at 300, 600x300, + // and 600 DPI. We'll use the old-style Resolution convenience + // keyword which accepts the following parameters: colorspace/ + // order, bits-per-color, row count, row feed, row step, and + // name/text. + // + // The name must be of the form NNNsuffix or NNNxMMMsuffix, + // where NNN and MMM represent the X and Y resolution in dots + // per inch. + // + + Resolution - 8 0 0 0 "300dpi/300 DPI" + Resolution - 8 0 0 0 "600x300dpi/600 x 300 DPI" + *Resolution - 8 0 0 0 "600dpi/600 DPI" + + + // + // One printer is grayscale only, and the other does grayscale + // and color. Define the grayscale color model for both printers + // using the old-style ColorModel convenience keyword which + // accepts the name/text, colorspace, color order, and compression + // parameters. + // + + ColorModel Gray/Grayscale w chunked 0 + + + { + // + // The first sub-group contains the grayscale printer, which + // only needs the model name, PC filename, and model number + // values set... + // + // The ModelName keyword defines the string that is shown to + // the user. + // + + ModelName "Mono Photo Printer" + + + // + // The ModelNumber keyword defines the cupsModelNumber + // attribute value. We use the "(name name)" notation + // to perform a bitwise OR of the #define'd constants. + // + + ModelNumber ($MODEL_BW $MODEL_PHOTO) + + + // + // The PCFileName keyword defines the filename of the PPD + // file and should be 8 characters or less + the .ppd + // extension. + // + + PCFileName "foogphot.ppd" + } + + + { + // + // The second sub-group contains the color printer, which + // needs another ColorModel definition along with the model + // name, PC filename, and model number values. For fun, we'll + // add some input slots (paper trays) as well. + // + // The ModelName keyword defines the string that is shown to + // the user. + // + + ModelName "Color Photo Printer" + + + // + // The ModelNumber keyword defines the cupsModelNumber + // attribute value. We use the "(name name)" notation + // to perform a bitwise OR of the #define'd constants. + // + + ModelNumber ($MODEL_COLOR $MODEL_PHOTO) + + + // + // The PCFileName keyword defines the filename of the PPD + // file and should be 8 characters or less + the .ppd + // extension. + // + + PCFileName "foocphot.ppd" + + + // + // This printer does color printing, too, so add it and make + // RGB the default... + // + + ColorDevice Yes + + *ColorModel RGB/Color rgb chunked 0 + + + // + // The old-style InputSlot keyword accepts tray definitions + // of the form: + // + // InputSlot position name/text + // + + InputSlot 0 "Upper/Main Paper Tray" + InputSlot 1 "LargeCapacity/Large Paper Tray" + } +} + + +{ + // + // Define two printer drivers that support two typical laser + // printers with custom page sizes. One is color, the other is + // black-and-white. + // + // Both printers share several MediaSize definitions and support + // custom page sizes from 3x5 to 13x19 inches. + // + // All US media sizes use hardware margins of 0.25 inches on the sides + // and 12 points (1/6th inch) at the top and bottom. European sizes + // and custom sizes use margins of 12 points all around. + // + // The order of the HWMargins numbers are left, bottom, right, and top. + // The current HWMargins values are used when defining each media size. + // The last HWMargins values are used for custom page size margins. + // + + HWMargins 0.25in 12pt 0.25in 12pt + + *MediaSize Letter + MediaSize Legal + MediaSize Tabloid + MediaSize TabloidExtra + + HWMargins 12pt 12pt 12pt 12pt + MediaSize A4 + MediaSize A3 + + // + // Specify that custom/variable paper sizes are supported, and the + // range of sizes that are supported... + // + + VariablePaperSize Yes + MinSize 3in 5in + MaxSize 13in 19in + + + // + // These imaginary printers support printing at 600 and 1200 DPI. + // We'll use the new Option and Choice keywords to define the + // Resolution options... + // + // Option option-name option-text option-type + // Choice choice-name choice-text code + // + // "Option-type" is the type of option: boolean, pickone, or pickmany. + // + + Option Resolution PickOne AnySetup 10 + Choice "600dpi/600 DPI" "<>setpagedevice" + Choice "1200dpi/1200 DPI" "<>setpagedevice" + + + // + // One printer is grayscale only, and the other does grayscale + // and color. Define the grayscale color model for both printers + // using the new Option and Choice keywords. + // + + Option "ColorModel/Color Mode" PickOne AnySetup 10 + Choice Gray/Grayscale "<>setpagedevice" + + + // + // Both printers provide two paper trays, which we'll define using + // the new Option and Choice keywords... + // + + Option "InputSlot/Input Slot" PickOne AnySetup 10 + Choice "Upper/Main Paper Tray" "<>setpagedevice" + Choice "LargeCapacity/Large Paper Tray" "<>setpagedevice" + + + // + // Both printers support duplexing... + // + // The Duplex keyword accepts values of "none" (no duplexing capability), + // "normal" (standard duplexing capability), and "flip" (auto-duplex that + // requires the back side to be flipped by the RIP...) + // + + Duplex normal + + + { + // + // The first sub-group contains the grayscale printer, which + // only needs the model name, PC filename, and model number + // values set... + // + // The ModelName keyword defines the string that is shown to + // the user. + // + + ModelName "Mono Laser Printer" + + + // + // The ModelNumber keyword defines the cupsModelNumber + // attribute value. We use the "(name name)" notation + // to perform a bitwise OR of the #define'd constants. + // + + ModelNumber ($MODEL_BW $MODEL_LASER) + + + // + // The PCFileName keyword defines the filename of the PPD + // file and should be 8 characters or less + the .ppd + // extension. + // + + PCFileName "fooglasr.ppd" + } + + + { + // + // The second sub-group contains the color printer, which + // needs another ColorModel definition along with the model + // name, PC filename, and model number values. + // + // The ModelName keyword defines the string that is shown to + // the user. + // + + ModelName "Color Laser Printer" + + + // + // The ModelNumber keyword defines the cupsModelNumber + // attribute value. We use the "(name name)" notation + // to perform a bitwise OR of the #define'd constants. + // + + ModelNumber ($MODEL_COLOR $MODEL_LASER) + + + // + // The PCFileName keyword defines the filename of the PPD + // file and should be 8 characters or less + the .ppd + // extension. + // + + PCFileName "fooclasr.ppd" + + + // + // This printer does color printing, too, so add it and make + // RGB the default... + // + + ColorDevice Yes + + Option "ColorModel/Color Mode" PickOne AnySetup 10 + *Choice RGB/Color "<>setpagedevice" + } +} + + +// +// End of "$Id$". +// diff --git a/ppdc/genstrings.cxx b/ppdc/genstrings.cxx new file mode 100644 index 0000000000..d08416d325 --- /dev/null +++ b/ppdc/genstrings.cxx @@ -0,0 +1,215 @@ +// +// "$Id$" +// +// GNU gettext message generator for the CUPS PPD Compiler. +// +// This program is used to generate a dummy source file containing all of +// the standard media and sample driver strings. The results are picked up +// by GNU gettext and placed in the CUPS message catalog. +// +// Copyright 2008-2011 by Apple Inc. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Usage: +// +// ./genstrings >sample.c +// +// Contents: +// +// main() - Main entry for the PPD compiler. +// add_ui_strings() - Add all UI strings from the driver. +// write_cstring() - Write a translation string as a valid C string to +// stdout. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" +#include + + +// +// Local functions... +// + +static void add_ui_strings(ppdcDriver *d, ppdcCatalog *catalog); +static void write_cstring(const char *s); + + +// +// 'main()' - Main entry for the PPD compiler. +// + +int // O - Exit status +main(void) +{ + ppdcSource *src; // PPD source file data + ppdcCatalog *catalog; // Catalog to hold all of the UI strings + + + // Make sure we are in the right place... + if (access("../data", 0) || access("sample.drv", 0)) + { + puts("You must run genstrings from the ppdc directory."); + return (1); + } + + // Load the sample drivers... + ppdcSource::add_include("../data"); + + src = new ppdcSource("sample.drv"); + catalog = new ppdcCatalog(NULL); + + catalog->add_message("ISOLatin1"); + catalog->add_message("English"); + + // Add the media size strings... + ppdcMediaSize *size; // Current media size + + for (size = (ppdcMediaSize *)src->sizes->first(); + size; + size = (ppdcMediaSize *)src->sizes->next()) + catalog->add_message(size->text->value); + + // Then collect all of the UI strings from the sample drivers... + ppdcDriver *d; // Current driver + + for (d = (ppdcDriver *)src->drivers->first(); + d; + d = (ppdcDriver *)src->drivers->next()) + add_ui_strings(d, catalog); + + // Finally, write all of the strings... + ppdcMessage *message; + + for (message = (ppdcMessage *)catalog->messages->first(); + message; + message = (ppdcMessage *)catalog->messages->next()) + write_cstring(message->id->value); + + src->release(); + catalog->release(); + + // Return with no errors. + return (0); +} + + +// +// 'add_ui_strings()' - Add all UI strings from the driver. +// + +static void +add_ui_strings(ppdcDriver *d, // I - Driver data + ppdcCatalog *catalog) // I - Message catalog +{ + // Add the make/model/language strings... + catalog->add_message(d->manufacturer->value); + catalog->add_message(d->model_name->value); + + // Add the group/option/choice strings... + ppdcGroup *g; // Current group + ppdcOption *o; // Current option + ppdcChoice *c; // Current choice + + for (g = (ppdcGroup *)d->groups->first(); + g; + g = (ppdcGroup *)d->groups->next()) + { + if (!g->options->count) + continue; + + if (_cups_strcasecmp(g->name->value, "General")) + catalog->add_message(g->text->value); + + for (o = (ppdcOption *)g->options->first(); + o; + o = (ppdcOption *)g->options->next()) + { + if (!o->choices->count) + continue; + + if (o->text->value && strcmp(o->name->value, o->text->value)) + catalog->add_message(o->text->value); + else + catalog->add_message(o->name->value); + + for (c = (ppdcChoice *)o->choices->first(); + c; + c = (ppdcChoice *)o->choices->next()) + if (c->text->value && strcmp(c->name->value, c->text->value)) + catalog->add_message(c->text->value); + else + catalog->add_message(c->name->value); + } + } + + // Add profile and preset strings... + ppdcAttr *a; // Current attribute + for (a = (ppdcAttr *)d->attrs->first(); + a; + a = (ppdcAttr *)d->attrs->next()) + { + if (a->text->value && a->text->value[0] && + (a->localizable || + !strncmp(a->name->value, "Custom", 6) || + !strncmp(a->name->value, "ParamCustom", 11) || + !strcmp(a->name->value, "APCustomColorMatchingName") || + !strcmp(a->name->value, "APPrinterPreset") || + !strcmp(a->name->value, "cupsICCProfile") || + !strcmp(a->name->value, "cupsIPPReason") || + !strcmp(a->name->value, "cupsMarkerName"))) + { + catalog->add_message(a->text->value); + + if ((a->localizable && a->value->value[0]) || + !strcmp(a->name->value, "cupsIPPReason")) + catalog->add_message(a->value->value); + } + else if (!strncmp(a->name->value, "Custom", 6) || + !strncmp(a->name->value, "ParamCustom", 11)) + catalog->add_message(a->name->value); + } +} + + +// +// 'write_cstring()' - Write a translation string as a valid C string to stdout. +// + +static void +write_cstring(const char *s) /* I - String to write */ +{ + fputs("_(\"", stdout); + if (s) + { + while (*s) + { + if (*s == '\\') + fputs("\\\\", stdout); + else if (*s == '\"') + fputs("\\\"", stdout); + else if (*s == '\t') + fputs("\\t", stdout); + else if (*s == '\n') + fputs("\\n", stdout); + else + putchar(*s); + + s ++; + } + } + puts("\");"); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-array.cxx b/ppdc/ppdc-array.cxx new file mode 100644 index 0000000000..33d8bf7736 --- /dev/null +++ b/ppdc/ppdc-array.cxx @@ -0,0 +1,168 @@ +// +// "$Id$" +// +// Array class for the CUPS PPD Compiler. +// +// Copyright 2007-2009 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcArray::ppdcArray() - Create a new array. +// ppdcArray::~ppdcArray() - Destroy an array. +// ppdcArray::add() - Add an element to an array. +// ppdcArray::first() - Return the first element in the array. +// ppdcArray::next() - Return the next element in the array. +// ppdcArray::remove() - Remove an element from the array. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// 'ppdcArray::ppdcArray()' - Create a new array. +// + +ppdcArray::ppdcArray(ppdcArray *a) + : ppdcShared() +{ + PPDC_NEW; + + if (a) + { + count = a->count; + alloc = count; + + if (count) + { + // Make a copy of the array... + data = new ppdcShared *[count]; + + memcpy(data, a->data, count * sizeof(ppdcShared *)); + + for (int i = 0; i < count; i ++) + data[i]->retain(); + } + else + data = 0; + } + else + { + count = 0; + alloc = 0; + data = 0; + } + + current = 0; +} + + +// +// 'ppdcArray::~ppdcArray()' - Destroy an array. +// + +ppdcArray::~ppdcArray() +{ + PPDC_DELETE; + + for (int i = 0; i < count; i ++) + data[i]->release(); + + if (alloc) + delete[] data; +} + + +// +// 'ppdcArray::add()' - Add an element to an array. +// + +void +ppdcArray::add(ppdcShared *d) +{ + ppdcShared **temp; + + + if (count >= alloc) + { + alloc += 10; + temp = new ppdcShared *[alloc]; + + memcpy(temp, data, count * sizeof(ppdcShared *)); + + delete[] data; + data = temp; + } + + data[count++] = d; +} + + +// +// 'ppdcArray::first()' - Return the first element in the array. +// + +ppdcShared * +ppdcArray::first() +{ + current = 0; + + if (current >= count) + return (0); + else + return (data[current ++]); +} + + +// +// 'ppdcArray::next()' - Return the next element in the array. +// + +ppdcShared * +ppdcArray::next() +{ + if (current >= count) + return (0); + else + return (data[current ++]); +} + + +// +// 'ppdcArray::remove()' - Remove an element from the array. +// + +void +ppdcArray::remove(ppdcShared *d) // I - Data element +{ + int i; // Looping var + + + for (i = 0; i < count; i ++) + if (d == data[i]) + break; + + if (i >= count) + return; + + count --; + d->release(); + + if (i < count) + memmove(data + i, data + i + 1, (count - i) * sizeof(ppdcShared *)); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-attr.cxx b/ppdc/ppdc-attr.cxx new file mode 100644 index 0000000000..1ea0337a93 --- /dev/null +++ b/ppdc/ppdc-attr.cxx @@ -0,0 +1,66 @@ +// +// "$Id$" +// +// Attribute class for the CUPS PPD Compiler. +// +// Copyright 2007-2009 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcAttr::ppdcAttr() - Create an attribute. +// ppdcAttr::~ppdcAttr() - Destroy an attribute. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// 'ppdcAttr::ppdcAttr()' - Create an attribute. +// + +ppdcAttr::ppdcAttr(const char *n, // I - Name + const char *s, // I - Spec string + const char *t, // I - Human-readable text + const char *v, // I - Value + bool loc) // I - Localize this attribute? + : ppdcShared() +{ + PPDC_NEW; + + name = new ppdcString(n); + selector = new ppdcString(s); + text = new ppdcString(t); + value = new ppdcString(v); + localizable = loc; +} + + +// +// 'ppdcAttr::~ppdcAttr()' - Destroy an attribute. +// + +ppdcAttr::~ppdcAttr() +{ + PPDC_DELETE; + + name->release(); + selector->release(); + text->release(); + value->release(); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-catalog.cxx b/ppdc/ppdc-catalog.cxx new file mode 100644 index 0000000000..acf178b29a --- /dev/null +++ b/ppdc/ppdc-catalog.cxx @@ -0,0 +1,897 @@ +// +// "$Id$" +// +// Shared message catalog class for the CUPS PPD Compiler. +// +// Copyright 2007-2010 by Apple Inc. +// Copyright 2002-2006 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcCatalog::ppdcCatalog() - Create a shared message catalog. +// ppdcCatalog::~ppdcCatalog() - Destroy a shared message catalog. +// ppdcCatalog::add_message() - Add a new message. +// ppdcCatalog::find_message() - Find a message in a catalog... +// ppdcCatalog::load_messages() - Load messages from a .po file. +// ppdcCatalog::save_messages() - Save the messages to a .po file. +// get_utf8() - Get a UTF-8 character. +// get_utf16() - Get a UTF-16 character... +// put_utf8() - Add a UTF-8 character to a string. +// put_utf16() - Write a UTF-16 character to a file. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// Character encodings... +// + +typedef enum +{ + PPDC_CS_AUTO, + PPDC_CS_UTF8, + PPDC_CS_UTF16BE, + PPDC_CS_UTF16LE +} ppdc_cs_t; + + +// +// Local functions... +// + +static int get_utf8(char *&ptr); +static int get_utf16(cups_file_t *fp, ppdc_cs_t &cs); +static int put_utf8(int ch, char *&ptr, char *end); +static int put_utf16(cups_file_t *fp, int ch); + + +// +// 'ppdcCatalog::ppdcCatalog()' - Create a shared message catalog. +// + +ppdcCatalog::ppdcCatalog(const char *l, // I - Locale + const char *f) // I - Message catalog file + : ppdcShared() +{ + _cups_globals_t *cg = _cupsGlobals(); + // Global information + + + PPDC_NEW; + + locale = new ppdcString(l); + filename = new ppdcString(f); + messages = new ppdcArray(); + + if (l) + { + // Try loading the base messages for this locale... + char pofile[1024]; // Message catalog file + + + snprintf(pofile, sizeof(pofile), "%s/%s/cups_%s.po", cg->localedir, l, l); + + if (load_messages(pofile) && strchr(l, '_')) + { + // Try the base locale... + char baseloc[3]; // Base locale... + + + strlcpy(baseloc, l, sizeof(baseloc)); + snprintf(pofile, sizeof(pofile), "%s/%s/cups_%s.po", cg->localedir, + baseloc, baseloc); + + load_messages(pofile); + } + } + + if (f) + load_messages(f); +} + + +// +// 'ppdcCatalog::~ppdcCatalog()' - Destroy a shared message catalog. +// + +ppdcCatalog::~ppdcCatalog() +{ + PPDC_DELETE; + + locale->release(); + filename->release(); + messages->release(); +} + + +// +// 'ppdcCatalog::add_message()' - Add a new message. +// + +void +ppdcCatalog::add_message( + const char *id, // I - Message ID to add + const char *string) // I - Translation string +{ + ppdcMessage *m; // Current message + char text[1024]; // Text to translate + + + // Range check input... + if (!id) + return; + + // Verify that we don't already have the message ID... + for (m = (ppdcMessage *)messages->first(); + m; + m = (ppdcMessage *)messages->next()) + if (!strcmp(m->id->value, id)) + { + if (string) + { + m->string->release(); + m->string = new ppdcString(string); + } + return; + } + + // Add the message... + if (!string) + { + snprintf(text, sizeof(text), "TRANSLATE %s", id); + string = text; + } + + messages->add(new ppdcMessage(id, string)); +} + + +// +// 'ppdcCatalog::find_message()' - Find a message in a catalog... +// + +const char * // O - Message text +ppdcCatalog::find_message( + const char *id) // I - Message ID +{ + ppdcMessage *m; // Current message + + + if (!*id) + return (id); + + for (m = (ppdcMessage *)messages->first(); + m; + m = (ppdcMessage *)messages->next()) + if (!strcmp(m->id->value, id)) + return (m->string->value); + + return (id); +} + + +// +// 'ppdcCatalog::load_messages()' - Load messages from a .po file. +// + +int // O - 0 on success, -1 on failure +ppdcCatalog::load_messages( + const char *f) // I - Message catalog file +{ + cups_file_t *fp; // Message file + char line[4096], // Line buffer + *ptr, // Pointer into buffer + id[4096], // Translation ID + str[4096]; // Translation string + int linenum; // Line number + + + // Open the message catalog file... + if ((fp = cupsFileOpen(f, "r")) == NULL) + return (-1); + + if ((ptr = (char *)strrchr(f, '.')) == NULL) + goto unknown_load_format; + else if (!strcmp(ptr, ".strings")) + { + /* + * Read messages in Mac OS X ".strings" format, which are UTF-16 text + * files of the format: + * + * "id" = "str"; + * + * Strings files can also contain C-style comments. + */ + + ppdc_cs_t cs = PPDC_CS_AUTO; // Character set for file + int ch; // Current character from file + char *end; // End of buffer + + + id[0] = '\0'; + str[0] = '\0'; + ptr = NULL; + end = NULL; + + while ((ch = get_utf16(fp, cs)) != 0) + { + if (ptr) + { + if (ch == '\\') + { + if ((ch = get_utf16(fp, cs)) == 0) + break; + + if (ch == 'n') + ch = '\n'; + else if (ch == 't') + ch = '\t'; + } + else if (ch == '\"') + { + *ptr = '\0'; + ptr = NULL; + } + + if (ptr) + put_utf8(ch, ptr, end); + } + else if (ch == '/') + { + // Start of a comment? + if ((ch = get_utf16(fp, cs)) == 0) + break; + + if (ch == '*') + { + // Skip C comment... + int lastch = 0; + + while ((ch = get_utf16(fp, cs)) != 0) + { + if (ch == '/' && lastch == '*') + break; + + lastch = ch; + } + } + else if (ch == '/') + { + // Skip C++ comment... + while ((ch = get_utf16(fp, cs)) != 0) + if (ch == '\n') + break; + } + } + else if (ch == '\"') + { + // Start quoted string... + if (id[0]) + { + ptr = str; + end = str + sizeof(str) - 1; + } + else + { + ptr = id; + end = id + sizeof(id) - 1; + } + } + else if (ch == ';') + { + // Add string... + add_message(id, str); + id[0] = '\0'; + } + } + } + else if (!strcmp(ptr, ".po") || !strcmp(ptr, ".gz")) + { + /* + * Read messages from the catalog file until EOF... + * + * The format is the GNU gettext .po format, which is fairly simple: + * + * msgid "some text" + * msgstr "localized text" + * + * The ID and localized text can span multiple lines using the form: + * + * msgid "" + * "some long text" + * msgstr "" + * "localized text spanning " + * "multiple lines" + */ + + int which, // In msgid? + haveid, // Did we get a msgid string? + havestr; // Did we get a msgstr string? + + linenum = 0; + id[0] = '\0'; + str[0] = '\0'; + haveid = 0; + havestr = 0; + which = 0; + + while (cupsFileGets(fp, line, sizeof(line))) + { + linenum ++; + + // Skip blank and comment lines... + if (line[0] == '#' || !line[0]) + continue; + + // Strip the trailing quote... + if ((ptr = (char *)strrchr(line, '\"')) == NULL) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected quoted string on line %d of %s."), + linenum, f); + cupsFileClose(fp); + return (-1); + } + + *ptr = '\0'; + + // Find start of value... + if ((ptr = strchr(line, '\"')) == NULL) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected quoted string on line %d of %s."), + linenum, f); + cupsFileClose(fp); + return (-1); + } + + ptr ++; + + // Unquote the text... + char *sptr, *dptr; // Source/destination pointers + + for (sptr = ptr, dptr = ptr; *sptr;) + { + if (*sptr == '\\') + { + sptr ++; + if (isdigit(*sptr)) + { + *dptr = 0; + + while (isdigit(*sptr)) + { + *dptr = *dptr * 8 + *sptr - '0'; + sptr ++; + } + + dptr ++; + } + else + { + if (*sptr == 'n') + *dptr++ = '\n'; + else if (*sptr == 'r') + *dptr++ = '\r'; + else if (*sptr == 't') + *dptr++ = '\t'; + else + *dptr++ = *sptr; + + sptr ++; + } + } + else + *dptr++ = *sptr++; + } + + *dptr = '\0'; + + // Create or add to a message... + if (!strncmp(line, "msgid", 5)) + { + if (haveid && havestr) + add_message(id, str); + + strlcpy(id, ptr, sizeof(id)); + str[0] = '\0'; + haveid = 1; + havestr = 0; + which = 1; + } + else if (!strncmp(line, "msgstr", 6)) + { + if (!haveid) + { + _cupsLangPrintf(stderr, + _("ppdc: Need a msgid line before any " + "translation strings on line %d of %s."), + linenum, f); + cupsFileClose(fp); + return (-1); + } + + strlcpy(str, ptr, sizeof(str)); + havestr = 1; + which = 2; + } + else if (line[0] == '\"' && which == 2) + strlcat(str, ptr, sizeof(str)); + else if (line[0] == '\"' && which == 1) + strlcat(id, ptr, sizeof(id)); + else + { + _cupsLangPrintf(stderr, _("ppdc: Unexpected text on line %d of %s."), + linenum, f); + cupsFileClose(fp); + return (-1); + } + } + + if (haveid && havestr) + add_message(id, str); + } + else + goto unknown_load_format; + + /* + * Close the file and return... + */ + + cupsFileClose(fp); + + return (0); + + /* + * Unknown format error... + */ + + unknown_load_format: + + _cupsLangPrintf(stderr, + _("ppdc: Unknown message catalog format for \"%s\"."), f); + cupsFileClose(fp); + return (-1); +} + + +// +// 'ppdcCatalog::save_messages()' - Save the messages to a .po file. +// + +int // O - 0 on success, -1 on error +ppdcCatalog::save_messages( + const char *f) // I - File to save to +{ + cups_file_t *fp; // Message file + ppdcMessage *m; // Current message + char *ptr; // Pointer into string + int utf16; // Output UTF-16 .strings file? + int ch; // Current character + + + // Open the file... + if ((ptr = (char *)strrchr(f, '.')) == NULL) + return (-1); + + if (!strcmp(ptr, ".gz")) + fp = cupsFileOpen(f, "w9"); + else + fp = cupsFileOpen(f, "w"); + + if (!fp) + return (-1); + + // For .strings files, write a BOM for big-endian output... + utf16 = !strcmp(ptr, ".strings"); + + if (utf16) + put_utf16(fp, 0xfeff); + + // Loop through all of the messages... + for (m = (ppdcMessage *)messages->first(); + m; + m = (ppdcMessage *)messages->next()) + { + if (utf16) + { + put_utf16(fp, '\"'); + + ptr = m->id->value; + while ((ch = get_utf8(ptr)) != 0) + switch (ch) + { + case '\n' : + put_utf16(fp, '\\'); + put_utf16(fp, 'n'); + break; + case '\\' : + put_utf16(fp, '\\'); + put_utf16(fp, '\\'); + break; + case '\"' : + put_utf16(fp, '\\'); + put_utf16(fp, '\"'); + break; + default : + put_utf16(fp, ch); + break; + } + + put_utf16(fp, '\"'); + put_utf16(fp, ' '); + put_utf16(fp, '='); + put_utf16(fp, ' '); + put_utf16(fp, '\"'); + + ptr = m->string->value; + while ((ch = get_utf8(ptr)) != 0) + switch (ch) + { + case '\n' : + put_utf16(fp, '\\'); + put_utf16(fp, 'n'); + break; + case '\\' : + put_utf16(fp, '\\'); + put_utf16(fp, '\\'); + break; + case '\"' : + put_utf16(fp, '\\'); + put_utf16(fp, '\"'); + break; + default : + put_utf16(fp, ch); + break; + } + + put_utf16(fp, '\"'); + put_utf16(fp, ';'); + put_utf16(fp, '\n'); + } + else + { + cupsFilePuts(fp, "msgid \""); + for (ptr = m->id->value; *ptr; ptr ++) + switch (*ptr) + { + case '\n' : + cupsFilePuts(fp, "\\n"); + break; + case '\\' : + cupsFilePuts(fp, "\\\\"); + break; + case '\"' : + cupsFilePuts(fp, "\\\""); + break; + default : + cupsFilePutChar(fp, *ptr); + break; + } + cupsFilePuts(fp, "\"\n"); + + cupsFilePuts(fp, "msgstr \""); + for (ptr = m->string->value; *ptr; ptr ++) + switch (*ptr) + { + case '\n' : + cupsFilePuts(fp, "\\n"); + break; + case '\\' : + cupsFilePuts(fp, "\\\\"); + break; + case '\"' : + cupsFilePuts(fp, "\\\""); + break; + default : + cupsFilePutChar(fp, *ptr); + break; + } + cupsFilePuts(fp, "\"\n"); + + cupsFilePutChar(fp, '\n'); + } + } + + cupsFileClose(fp); + + return (0); +} + + +// +// 'get_utf8()' - Get a UTF-8 character. +// + +static int // O - Unicode character or 0 on EOF +get_utf8(char *&ptr) // IO - Pointer to character +{ + int ch; // Current character + + + if ((ch = *ptr++ & 255) < 0xc0) + return (ch); + + if ((ch & 0xe0) == 0xc0) + { + // Two-byte UTF-8... + if ((*ptr & 0xc0) != 0x80) + return (0); + + ch = ((ch & 0x1f) << 6) | (*ptr++ & 0x3f); + } + else if ((ch & 0xf0) == 0xe0) + { + // Three-byte UTF-8... + if ((*ptr & 0xc0) != 0x80) + return (0); + + ch = ((ch & 0x0f) << 6) | (*ptr++ & 0x3f); + + if ((*ptr & 0xc0) != 0x80) + return (0); + + ch = (ch << 6) | (*ptr++ & 0x3f); + } + else if ((ch & 0xf8) == 0xf0) + { + // Four-byte UTF-8... + if ((*ptr & 0xc0) != 0x80) + return (0); + + ch = ((ch & 0x07) << 6) | (*ptr++ & 0x3f); + + if ((*ptr & 0xc0) != 0x80) + return (0); + + ch = (ch << 6) | (*ptr++ & 0x3f); + + if ((*ptr & 0xc0) != 0x80) + return (0); + + ch = (ch << 6) | (*ptr++ & 0x3f); + } + + return (ch); +} + + +// +// 'get_utf16()' - Get a UTF-16 character... +// + +static int // O - Unicode character or 0 on EOF +get_utf16(cups_file_t *fp, // I - File to read from + ppdc_cs_t &cs) // IO - Character set of file +{ + int ch; // Current character + unsigned char buffer[3]; // Bytes + + + if (cs == PPDC_CS_AUTO) + { + // Get byte-order-mark, if present... + if (cupsFileRead(fp, (char *)buffer, 2) != 2) + return (0); + + if (buffer[0] == 0xfe && buffer[1] == 0xff) + { + // Big-endian UTF-16... + cs = PPDC_CS_UTF16BE; + + if (cupsFileRead(fp, (char *)buffer, 2) != 2) + return (0); + } + else if (buffer[0] == 0xff && buffer[1] == 0xfe) + { + // Little-endian UTF-16... + cs = PPDC_CS_UTF16LE; + + if (cupsFileRead(fp, (char *)buffer, 2) != 2) + return (0); + } + else if (buffer[0] == 0x00 && buffer[1] != 0x00) + { + // No BOM, assume big-endian UTF-16... + cs = PPDC_CS_UTF16BE; + } + else if (buffer[0] != 0x00 && buffer[1] == 0x00) + { + // No BOM, assume little-endian UTF-16... + cs = PPDC_CS_UTF16LE; + } + else + { + // No BOM, assume UTF-8... + cs = PPDC_CS_UTF8; + + cupsFileRewind(fp); + } + } + else if (cs != PPDC_CS_UTF8) + { + if (cupsFileRead(fp, (char *)buffer, 2) != 2) + return (0); + } + + if (cs == PPDC_CS_UTF8) + { + // UTF-8 character... + if ((ch = cupsFileGetChar(fp)) < 0) + return (0); + + if ((ch & 0xe0) == 0xc0) + { + // Two-byte UTF-8... + if (cupsFileRead(fp, (char *)buffer, 1) != 1) + return (0); + + if ((buffer[0] & 0xc0) != 0x80) + return (0); + + ch = ((ch & 0x1f) << 6) | (buffer[0] & 0x3f); + } + else if ((ch & 0xf0) == 0xe0) + { + // Three-byte UTF-8... + if (cupsFileRead(fp, (char *)buffer, 2) != 2) + return (0); + + if ((buffer[0] & 0xc0) != 0x80 || + (buffer[1] & 0xc0) != 0x80) + return (0); + + ch = ((((ch & 0x0f) << 6) | (buffer[0] & 0x3f)) << 6) | + (buffer[1] & 0x3f); + } + else if ((ch & 0xf8) == 0xf0) + { + // Four-byte UTF-8... + if (cupsFileRead(fp, (char *)buffer, 3) != 3) + return (0); + + if ((buffer[0] & 0xc0) != 0x80 || + (buffer[1] & 0xc0) != 0x80 || + (buffer[2] & 0xc0) != 0x80) + return (0); + + ch = ((((((ch & 0x07) << 6) | (buffer[0] & 0x3f)) << 6) | + (buffer[1] & 0x3f)) << 6) | (buffer[2] & 0x3f); + } + } + else + { + // UTF-16 character... + if (cs == PPDC_CS_UTF16BE) + ch = (buffer[0] << 8) | buffer[1]; + else + ch = (buffer[1] << 8) | buffer[0]; + + if (ch >= 0xd800 && ch <= 0xdbff) + { + // Handle multi-word encoding... + int lch; + + if (cupsFileRead(fp, (char *)buffer, 2) != 2) + return (0); + + if (cs == PPDC_CS_UTF16BE) + lch = (buffer[0] << 8) | buffer[1]; + else + lch = (buffer[1] << 8) | buffer[0]; + + if (lch < 0xdc00 || lch >= 0xdfff) + return (0); + + ch = (((ch & 0x3ff) << 10) | (lch & 0x3ff)) + 0x10000; + } + } + + return (ch); +} + + +// +// 'put_utf8()' - Add a UTF-8 character to a string. +// + +static int // O - 0 on success, -1 on failure +put_utf8(int ch, // I - Unicode character + char *&ptr, // IO - String pointer + char *end) // I - End of buffer +{ + if (ch < 0x80) + { + // One-byte ASCII... + if (ptr >= end) + return (-1); + + *ptr++ = ch; + } + else if (ch < 0x800) + { + // Two-byte UTF-8... + if ((ptr + 1) >= end) + return (-1); + + *ptr++ = 0xc0 | (ch >> 6); + *ptr++ = 0x80 | (ch & 0x3f); + } + else if (ch < 0x10000) + { + // Three-byte UTF-8... + if ((ptr + 2) >= end) + return (-1); + + *ptr++ = 0xe0 | (ch >> 12); + *ptr++ = 0x80 | ((ch >> 6) & 0x3f); + *ptr++ = 0x80 | (ch & 0x3f); + } + else + { + // Four-byte UTF-8... + if ((ptr + 3) >= end) + return (-1); + + *ptr++ = 0xf0 | (ch >> 18); + *ptr++ = 0x80 | ((ch >> 12) & 0x3f); + *ptr++ = 0x80 | ((ch >> 6) & 0x3f); + *ptr++ = 0x80 | (ch & 0x3f); + } + + return (0); +} + + +// +// 'put_utf16()' - Write a UTF-16 character to a file. +// + +static int // O - 0 on success, -1 on failure +put_utf16(cups_file_t *fp, // I - File to write to + int ch) // I - Unicode character +{ + unsigned char buffer[4]; // Output buffer + + + if (ch < 0x10000) + { + // One-word UTF-16 big-endian... + buffer[0] = ch >> 8; + buffer[1] = ch; + + if (cupsFileWrite(fp, (char *)buffer, 2) == 2) + return (0); + } + else + { + // Two-word UTF-16 big-endian... + ch -= 0x10000; + + buffer[0] = 0xd8 | (ch >> 18); + buffer[1] = ch >> 10; + buffer[2] = 0xdc | ((ch >> 8) & 0x03); + buffer[3] = ch; + + if (cupsFileWrite(fp, (char *)buffer, 4) == 4) + return (0); + } + + return (-1); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-choice.cxx b/ppdc/ppdc-choice.cxx new file mode 100644 index 0000000000..2c12de2b9c --- /dev/null +++ b/ppdc/ppdc-choice.cxx @@ -0,0 +1,61 @@ +// +// "$Id$" +// +// Option choice class for the CUPS PPD Compiler. +// +// Copyright 2007-2009 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcChoice::ppdcChoice() - Create a new option choice. +// ppdcChoice::~ppdcChoice() - Destroy an option choice. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// 'ppdcChoice::ppdcChoice()' - Create a new option choice. +// + +ppdcChoice::ppdcChoice(const char *n, // I - Name of choice + const char *t, // I - Text of choice + const char *c) // I - Code of choice + : ppdcShared() +{ + PPDC_NEW; + + name = new ppdcString(n); + text = new ppdcString(t); + code = new ppdcString(c); +} + + +// +// 'ppdcChoice::~ppdcChoice()' - Destroy an option choice. +// + +ppdcChoice::~ppdcChoice() +{ + PPDC_DELETE; + + name->release(); + text->release(); + code->release(); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-constraint.cxx b/ppdc/ppdc-constraint.cxx new file mode 100644 index 0000000000..6e6f0241e7 --- /dev/null +++ b/ppdc/ppdc-constraint.cxx @@ -0,0 +1,64 @@ +// +// "$Id$" +// +// Contraint class for the CUPS PPD Compiler. +// +// Copyright 2007-2009 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcConstraint::ppdcConstraint() - Create a constraint. +// ppdcConstraint::~ppdcConstraint() - Destroy a constraint. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// 'ppdcConstraint::ppdcConstraint()' - Create a constraint. +// + +ppdcConstraint::ppdcConstraint(const char *o1, // I - First option + const char *c1, // I - First choice + const char *o2, // I - Second option + const char *c2) // I - Second choice + : ppdcShared() +{ + PPDC_NEW; + + option1 = new ppdcString(o1); + choice1 = new ppdcString(c1); + option2 = new ppdcString(o2); + choice2 = new ppdcString(c2); +} + + +// +// 'ppdcConstraint::~ppdcConstraint()' - Destroy a constraint. +// + +ppdcConstraint::~ppdcConstraint() +{ + PPDC_DELETE; + + option1->release(); + choice1->release(); + option2->release(); + choice2->release(); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-driver.cxx b/ppdc/ppdc-driver.cxx new file mode 100644 index 0000000000..c9a7a1c969 --- /dev/null +++ b/ppdc/ppdc-driver.cxx @@ -0,0 +1,1339 @@ +// +// "$Id$" +// +// PPD file compiler definitions for the CUPS PPD Compiler. +// +// Copyright 2007-2011 by Apple Inc. +// Copyright 2002-2006 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcDriver::ppdcDriver() - Create a new printer driver. +// ppdcDriver::~ppdcDriver() - Destroy a printer driver. +// ppdcDriver::find_attr() - Find an attribute. +// ppdcDriver::find_group() - Find a group. +// ppdcDriver::find_option() - Find an option. +// ppdcDriver::find_option_group() - Find an option and its group. +// ppdcDriver::set_custom_size_code() - Set the custom page size code. +// ppdcDriver::set_default_font() - Set the default font name. +// ppdcDriver::set_default_size() - Set the default size name. +// ppdcDriver::set_file_name() - Set the full filename. +// ppdcDriver::set_manufacturer() - Set the manufacturer name. +// ppdcDriver::set_model_name() - Set the model name. +// ppdcDriver::set_pc_file_name() - Set the PC filename. +// ppdcDriver::set_version() - Set the version string. +// ppdcDriver::write_ppd_file() - Write a PPD file... +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// 'ppdcDriver::ppdcDriver()' - Create a new printer driver. +// + +ppdcDriver::ppdcDriver(ppdcDriver *d) // I - Printer driver template + : ppdcShared() +{ + ppdcGroup *g; // Current group + + + PPDC_NEW; + + if (d) + { + // Bump the use count of any strings we inherit... + if (d->manufacturer) + d->manufacturer->retain(); + if (d->version) + d->version->retain(); + if (d->default_font) + d->default_font->retain(); + if (d->default_size) + d->default_size->retain(); + if (d->custom_size_code) + d->custom_size_code->retain(); + + // Copy all of the data from the driver template... + copyright = new ppdcArray(d->copyright); + manufacturer = d->manufacturer; + model_name = 0; + file_name = 0; + pc_file_name = 0; + type = d->type; + version = d->version; + model_number = d->model_number; + manual_copies = d->manual_copies; + color_device = d->color_device; + throughput = d->throughput; + attrs = new ppdcArray(d->attrs); + constraints = new ppdcArray(d->constraints); + filters = new ppdcArray(d->filters); + fonts = new ppdcArray(d->fonts); + profiles = new ppdcArray(d->profiles); + sizes = new ppdcArray(d->sizes); + default_font = d->default_font; + default_size = d->default_size; + variable_paper_size = d->variable_paper_size; + custom_size_code = d->custom_size_code; + left_margin = d->left_margin; + bottom_margin = d->bottom_margin; + right_margin = d->right_margin; + top_margin = d->top_margin; + max_width = d->max_width; + max_length = d->max_length; + min_width = d->min_width; + min_length = d->min_length; + + // Then copy the groups manually, since we want separate copies + // of the groups and options... + groups = new ppdcArray(); + + for (g = (ppdcGroup *)d->groups->first(); g; g = (ppdcGroup *)d->groups->next()) + groups->add(new ppdcGroup(g)); + } + else + { + // Zero all of the data in the driver... + copyright = new ppdcArray(); + manufacturer = 0; + model_name = 0; + file_name = 0; + pc_file_name = 0; + version = 0; + type = PPDC_DRIVER_CUSTOM; + model_number = 0; + manual_copies = 0; + color_device = 0; + throughput = 1; + attrs = new ppdcArray(); + constraints = new ppdcArray(); + fonts = new ppdcArray(); + filters = new ppdcArray(); + groups = new ppdcArray(); + profiles = new ppdcArray(); + sizes = new ppdcArray(); + default_font = 0; + default_size = 0; + variable_paper_size = 0; + custom_size_code = 0; + left_margin = 0; + bottom_margin = 0; + right_margin = 0; + top_margin = 0; + max_width = 0; + max_length = 0; + min_width = 0; + min_length = 0; + } +} + + +// +// 'ppdcDriver::~ppdcDriver()' - Destroy a printer driver. +// + +ppdcDriver::~ppdcDriver() +{ + PPDC_DELETE; + + copyright->release(); + + if (manufacturer) + manufacturer->release(); + if (model_name) + model_name->release(); + if (file_name) + file_name->release(); + if (pc_file_name) + pc_file_name->release(); + if (version) + version->release(); + if (default_font) + default_font->release(); + if (default_size) + default_size->release(); + if (custom_size_code) + custom_size_code->release(); + + attrs->release(); + constraints->release(); + filters->release(); + fonts->release(); + groups->release(); + profiles->release(); + sizes->release(); +} + + +// +// 'ppdcDriver::find_attr()' - Find an attribute. +// + +ppdcAttr * // O - Attribute or NULL +ppdcDriver::find_attr(const char *k, // I - Keyword string + const char *s) // I - Spec string +{ + ppdcAttr *a; // Current attribute + + + for (a = (ppdcAttr *)attrs->first(); a; a = (ppdcAttr *)attrs->next()) + if (!strcmp(a->name->value, k) && + ((!s && (!a->selector->value || !a->selector->value[0])) || + (s && a->selector->value && !strcmp(a->selector->value, s)))) + return (a); + + return (NULL); +} + + +// +// 'ppdcDriver::find_group()' - Find a group. +// + +ppdcGroup * // O - Matching group or NULL +ppdcDriver::find_group(const char *n) // I - Group name +{ + ppdcGroup *g; // Current group + + + for (g = (ppdcGroup *)groups->first(); g; g = (ppdcGroup *)groups->next()) + if (!_cups_strcasecmp(n, g->name->value)) + return (g); + + return (0); +} + + +// +// 'ppdcDriver::find_option()' - Find an option. +// + +ppdcOption * // O - Matching option or NULL +ppdcDriver::find_option(const char *n) // I - Option name +{ + return (find_option_group(n, (ppdcGroup **)0)); +} + + +// +// 'ppdcDriver::find_option_group()' - Find an option and its group. +// + +ppdcOption * // O - Matching option or NULL +ppdcDriver::find_option_group( + const char *n, // I - Option name + ppdcGroup **mg) // O - Matching group or NULL +{ + ppdcGroup *g; // Current group + ppdcOption *o; // Current option + + + for (g = (ppdcGroup *)groups->first(); g; g = (ppdcGroup *)groups->next()) + for (o = (ppdcOption *)g->options->first(); o; o = (ppdcOption *)g->options->next()) + if (!_cups_strcasecmp(n, o->name->value)) + { + if (mg) + *mg = g; + + return (o); + } + + if (mg) + *mg = (ppdcGroup *)0; + + return (0); +} + + +// +// 'ppdcDriver::set_custom_size_code()' - Set the custom page size code. +// + +void +ppdcDriver::set_custom_size_code( + const char *c) // I - CustomPageSize code +{ + if (custom_size_code) + custom_size_code->release(); + + custom_size_code = new ppdcString(c); +} + + +// +// 'ppdcDriver::set_default_font()' - Set the default font name. +// + +void +ppdcDriver::set_default_font( + ppdcFont *f) // I - Font +{ + if (default_font) + default_font->release(); + + if (f) + { + f->name->retain(); + default_font = f->name; + } + else + default_font = 0; +} + + +// +// 'ppdcDriver::set_default_size()' - Set the default size name. +// + +void +ppdcDriver::set_default_size( + ppdcMediaSize *m) // I - Media size +{ + if (default_size) + default_size->release(); + + if (m) + { + m->name->retain(); + default_size = m->name; + } + else + default_size = 0; +} + + +// +// 'ppdcDriver::set_file_name()' - Set the full filename. +// + +void +ppdcDriver::set_file_name(const char *f)// I - Filename +{ + if (file_name) + file_name->release(); + + file_name = new ppdcString(f); +} + + +// +// 'ppdcDriver::set_manufacturer()' - Set the manufacturer name. +// + +void +ppdcDriver::set_manufacturer( + const char *m) // I - Model name +{ + if (manufacturer) + manufacturer->release(); + + manufacturer = new ppdcString(m); +} + + +// +// 'ppdcDriver::set_model_name()' - Set the model name. +// + +void +ppdcDriver::set_model_name( + const char *m) // I - Model name +{ + if (model_name) + model_name->release(); + + model_name = new ppdcString(m); +} + + +// +// 'ppdcDriver::set_pc_file_name()' - Set the PC filename. +// + +void +ppdcDriver::set_pc_file_name( + const char *f) // I - Filename +{ + if (pc_file_name) + pc_file_name->release(); + + pc_file_name = new ppdcString(f); +} + + +// +// 'ppdcDriver::set_version()' - Set the version string. +// + +void +ppdcDriver::set_version(const char *v) // I - Version +{ + if (version) + version->release(); + + version = new ppdcString(v); +} + + +// +// 'ppdcDriver::write_ppd_file()' - Write a PPD file... +// + +int // O - 0 on success, -1 on failure +ppdcDriver::write_ppd_file( + cups_file_t *fp, // I - PPD file + ppdcCatalog *catalog, // I - Message catalog + ppdcArray *locales, // I - Additional languages to add + ppdcSource *src, // I - Driver source + ppdcLineEnding le) // I - Line endings to use +{ + bool delete_cat; // Delete the catalog when we are done? + char query[42], // Query attribute + custom[42]; // Custom attribute + ppdcString *s; // Copyright string + ppdcGroup *g; // Current group + ppdcOption *o; // Current option + ppdcChoice *c; // Current choice + ppdcMediaSize *m; // Current media size + ppdcProfile *p; // Current color profile + ppdcFilter *f; // Current filter + ppdcFont *fn, // Current font + *bfn; // Current base font + ppdcConstraint *cn; // Current constraint + ppdcAttr *a; // Current attribute + const char *lf; // Linefeed character to use + + + // If we don't have a message catalog, use an empty (English) one... + if (!catalog) + { + catalog = new ppdcCatalog("en"); + delete_cat = true; + } + else + delete_cat = false; + + // Figure out the end-of-line string... + if (le == PPDC_LFONLY) + lf = "\n"; + else if (le == PPDC_CRONLY) + lf = "\r"; + else + lf = "\r\n"; + + // Write the standard header stuff... + cupsFilePrintf(fp, "*PPD-Adobe: \"4.3\"%s", lf); + cupsFilePrintf(fp, "*%%%%%%%% PPD file for %s with CUPS.%s", + model_name->value, lf); + cupsFilePrintf(fp, + "*%%%%%%%% Created by the CUPS PPD Compiler " CUPS_SVERSION + ".%s", lf); + for (s = (ppdcString *)copyright->first(); + s; + s = (ppdcString *)copyright->next()) + cupsFilePrintf(fp, "*%% %s%s", catalog->find_message(s->value), lf); + cupsFilePrintf(fp, "*FormatVersion: \"4.3\"%s", lf); + cupsFilePrintf(fp, "*FileVersion: \"%s\"%s", version->value, lf); + + a = find_attr("LanguageVersion", NULL); + cupsFilePrintf(fp, "*LanguageVersion: %s%s", + catalog->find_message(a ? a->value->value : "English"), lf); + + a = find_attr("LanguageEncoding", NULL); + cupsFilePrintf(fp, "*LanguageEncoding: %s%s", + catalog->find_message(a ? a->value->value : "ISOLatin1"), lf); + + cupsFilePrintf(fp, "*PCFileName: \"%s\"%s", pc_file_name->value, lf); + + for (a = (ppdcAttr *)attrs->first(); a; a = (ppdcAttr *)attrs->next()) + if (!strcmp(a->name->value, "Product")) + break; + + if (a) + { + for (; a; a = (ppdcAttr *)attrs->next()) + if (!strcmp(a->name->value, "Product")) + cupsFilePrintf(fp, "*Product: \"%s\"%s", a->value->value, lf); + } + else + cupsFilePrintf(fp, "*Product: \"(%s)\"%s", model_name->value, lf); + + cupsFilePrintf(fp, "*Manufacturer: \"%s\"%s", + catalog->find_message(manufacturer->value), lf); + + if ((a = find_attr("ModelName", NULL)) != NULL) + cupsFilePrintf(fp, "*ModelName: \"%s\"%s", + catalog->find_message(a->value->value), lf); + else if (_cups_strncasecmp(model_name->value, manufacturer->value, + strlen(manufacturer->value))) + cupsFilePrintf(fp, "*ModelName: \"%s %s\"%s", + catalog->find_message(manufacturer->value), + catalog->find_message(model_name->value), lf); + else + cupsFilePrintf(fp, "*ModelName: \"%s\"%s", + catalog->find_message(model_name->value), lf); + + if ((a = find_attr("ShortNickName", NULL)) != NULL) + cupsFilePrintf(fp, "*ShortNickName: \"%s\"%s", + catalog->find_message(a->value->value), lf); + else if (_cups_strncasecmp(model_name->value, manufacturer->value, + strlen(manufacturer->value))) + cupsFilePrintf(fp, "*ShortNickName: \"%s %s\"%s", + catalog->find_message(manufacturer->value), + catalog->find_message(model_name->value), lf); + else + cupsFilePrintf(fp, "*ShortNickName: \"%s\"%s", + catalog->find_message(model_name->value), lf); + + if ((a = find_attr("NickName", NULL)) != NULL) + cupsFilePrintf(fp, "*NickName: \"%s\"%s", + catalog->find_message(a->value->value), lf); + else if (_cups_strncasecmp(model_name->value, manufacturer->value, + strlen(manufacturer->value))) + cupsFilePrintf(fp, "*NickName: \"%s %s, %s\"%s", + catalog->find_message(manufacturer->value), + catalog->find_message(model_name->value), version->value, + lf); + else + cupsFilePrintf(fp, "*NickName: \"%s, %s\"%s", + catalog->find_message(model_name->value), version->value, + lf); + + for (a = (ppdcAttr *)attrs->first(); a; a = (ppdcAttr *)attrs->next()) + if (!strcmp(a->name->value, "PSVersion")) + break; + + if (a) + { + for (; a; a = (ppdcAttr *)attrs->next()) + if (!strcmp(a->name->value, "PSVersion")) + cupsFilePrintf(fp, "*PSVersion: \"%s\"%s", a->value->value, lf); + } + else + cupsFilePrintf(fp, "*PSVersion: \"(3010.000) 0\"%s", lf); + + if ((a = find_attr("LanguageLevel", NULL)) != NULL) + cupsFilePrintf(fp, "*LanguageLevel: \"%s\"%s", a->value->value, lf); + else + cupsFilePrintf(fp, "*LanguageLevel: \"3\"%s", lf); + + cupsFilePrintf(fp, "*ColorDevice: %s%s", color_device ? "True" : "False", lf); + + if ((a = find_attr("DefaultColorSpace", NULL)) != NULL) + cupsFilePrintf(fp, "*DefaultColorSpace: %s%s", a->value->value, lf); + else + cupsFilePrintf(fp, "*DefaultColorSpace: %s%s", + color_device ? "RGB" : "Gray", lf); + + if ((a = find_attr("FileSystem", NULL)) != NULL) + cupsFilePrintf(fp, "*FileSystem: %s%s", a->value->value, lf); + else + cupsFilePrintf(fp, "*FileSystem: False%s", lf); + + cupsFilePrintf(fp, "*Throughput: \"%d\"%s", throughput, lf); + + if ((a = find_attr("LandscapeOrientation", NULL)) != NULL) + cupsFilePrintf(fp, "*LandscapeOrientation: %s%s", a->value->value, lf); + else + cupsFilePrintf(fp, "*LandscapeOrientation: Plus90%s", lf); + + if ((a = find_attr("TTRasterizer", NULL)) != NULL) + cupsFilePrintf(fp, "*TTRasterizer: %s%s", a->value->value, lf); + else if (type != PPDC_DRIVER_PS) + cupsFilePrintf(fp, "*TTRasterizer: Type42%s", lf); + + struct lconv *loc = localeconv(); + + if (attrs->count) + { + // Write driver-defined attributes... + cupsFilePrintf(fp, "*%% Driver-defined attributes...%s", lf); + for (a = (ppdcAttr *)attrs->first(); a; a = (ppdcAttr *)attrs->next()) + { + if (!strcmp(a->name->value, "Product") || + !strcmp(a->name->value, "PSVersion") || + !strcmp(a->name->value, "LanguageLevel") || + !strcmp(a->name->value, "DefaultColorSpace") || + !strcmp(a->name->value, "FileSystem") || + !strcmp(a->name->value, "LandscapeOrientation") || + !strcmp(a->name->value, "TTRasterizer") || + !strcmp(a->name->value, "LanguageVersion") || + !strcmp(a->name->value, "LanguageEncoding") || + !strcmp(a->name->value, "ModelName") || + !strcmp(a->name->value, "NickName") || + !strcmp(a->name->value, "ShortNickName") || + !strcmp(a->name->value, "cupsVersion")) + continue; + + if (a->name->value[0] == '?' && + (find_option(a->name->value + 1) || + !strcmp(a->name->value, "?ImageableArea") || + !strcmp(a->name->value, "?PageRegion") || + !strcmp(a->name->value, "?PageSize") || + !strcmp(a->name->value, "?PaperDimension"))) + continue; + + if (!strncmp(a->name->value, "Custom", 6) && + find_option(a->name->value + 6)) + continue; + + if (!strncmp(a->name->value, "ParamCustom", 11) && + find_option(a->name->value + 11)) + continue; + + if (!a->selector->value || !a->selector->value[0]) + cupsFilePrintf(fp, "*%s", a->name->value); + else if (!a->text->value || !a->text->value[0]) + cupsFilePrintf(fp, "*%s %s", a->name->value, a->selector->value); + else + cupsFilePrintf(fp, "*%s %s/%s", a->name->value, a->selector->value, + a->text->value); + + if (strcmp(a->value->value, "False") && + strcmp(a->value->value, "True") && + strcmp(a->name->value, "1284Modes") && + strcmp(a->name->value, "InkName") && + strcmp(a->name->value, "PageStackOrder") && + strncmp(a->name->value, "ParamCustom", 11) && + strcmp(a->name->value, "Protocols") && + strcmp(a->name->value, "ReferencePunch") && + strncmp(a->name->value, "Default", 7)) + { + cupsFilePrintf(fp, ": \"%s\"%s", a->value->value, lf); + + if (strchr(a->value->value, '\n') || strchr(a->value->value, '\r')) + cupsFilePrintf(fp, "*End%s", lf); + } + else + cupsFilePrintf(fp, ": %s%s", a->value->value, lf); + } + } + + if (type != PPDC_DRIVER_PS || filters->count) + { + if ((a = find_attr("cupsVersion", NULL)) != NULL) + cupsFilePrintf(fp, "*cupsVersion: %s%s", a->value->value, lf); + else + cupsFilePrintf(fp, "*cupsVersion: %d.%d%s", CUPS_VERSION_MAJOR, + CUPS_VERSION_MINOR, lf); + cupsFilePrintf(fp, "*cupsModelNumber: %d%s", model_number, lf); + cupsFilePrintf(fp, "*cupsManualCopies: %s%s", + manual_copies ? "True" : "False", lf); + + if (filters->count) + { + for (f = (ppdcFilter *)filters->first(); + f; + f = (ppdcFilter *)filters->next()) + cupsFilePrintf(fp, "*cupsFilter: \"%s %d %s\"%s", f->mime_type->value, + f->cost, f->program->value, lf); + } + else + { + switch (type) + { + case PPDC_DRIVER_LABEL : + cupsFilePrintf(fp, "*cupsFilter: \"application/vnd.cups-raster 50 " + "rastertolabel\"%s", lf); + break; + + case PPDC_DRIVER_EPSON : + cupsFilePrintf(fp, "*cupsFilter: \"application/vnd.cups-raster 50 " + "rastertoepson\"%s", lf); + break; + + case PPDC_DRIVER_ESCP : + cupsFilePrintf(fp, "*cupsFilter: \"application/vnd.cups-command 50 " + "commandtoescpx\"%s", lf); + cupsFilePrintf(fp, "*cupsFilter: \"application/vnd.cups-raster 50 " + "rastertoescpx\"%s", lf); + break; + + case PPDC_DRIVER_HP : + cupsFilePrintf(fp, "*cupsFilter: \"application/vnd.cups-raster 50 " + "rastertohp\"%s", lf); + break; + + case PPDC_DRIVER_PCL : + cupsFilePrintf(fp, "*cupsFilter: \"application/vnd.cups-command 50 " + "commandtopclx\"%s", lf); + cupsFilePrintf(fp, "*cupsFilter: \"application/vnd.cups-raster 50 " + "rastertopclx\"%s", lf); + break; + + default : + break; + } + } + + for (p = (ppdcProfile *)profiles->first(); + p; + p = (ppdcProfile *)profiles->next()) + { + char density[255], gamma[255], profile[9][255]; + + _cupsStrFormatd(density, density + sizeof(density), p->density, loc); + _cupsStrFormatd(gamma, gamma + sizeof(gamma), p->gamma, loc); + + for (int i = 0; i < 9; i ++) + _cupsStrFormatd(profile[i], profile[i] + sizeof(profile[0]), + p->profile[i], loc); + + cupsFilePrintf(fp, + "*cupsColorProfile %s/%s: \"%s %s %s %s %s %s %s %s %s %s " + "%s\"%s", p->resolution->value, p->media_type->value, + density, gamma, profile[0], profile[1], profile[2], + profile[3], profile[4], profile[5], profile[6], profile[7], + profile[8], lf); + } + } + + if (locales) + { + // Add localizations for additional languages... + ppdcString *locale; // Locale name + ppdcCatalog *locatalog; // Message catalog for locale + + + // Write the list of languages... + cupsFilePrintf(fp, "*cupsLanguages: \"en"); + + for (locale = (ppdcString *)locales->first(); + locale; + locale = (ppdcString *)locales->next()) + { + // Skip (US) English... + if (!strcmp(locale->value, "en") || !strcmp(locale->value, "en_US")) + continue; + + // See if we have a po file for this language... + if (!src->find_po(locale->value)) + { + // No, see if we can use the base file? + locatalog = new ppdcCatalog(locale->value); + + if (locatalog->messages->count == 0) + { + // No, skip this one... + _cupsLangPrintf(stderr, + _("ppdc: No message catalog provided for locale " + "%s."), locale->value); + continue; + } + + // Add the base file to the list... + src->po_files->add(locatalog); + } + + cupsFilePrintf(fp, " %s", locale->value); + } + + cupsFilePrintf(fp, "\"%s", lf); + } + + for (cn = (ppdcConstraint *)constraints->first(); + cn; + cn = (ppdcConstraint *)constraints->next()) + { + // First constrain 1 against 2... + if (!strncmp(cn->option1->value, "*Custom", 7) || + !strncmp(cn->option2->value, "*Custom", 7)) + cupsFilePuts(fp, "*NonUIConstraints: "); + else + cupsFilePuts(fp, "*UIConstraints: "); + + if (cn->option1->value[0] != '*') + cupsFilePutChar(fp, '*'); + + cupsFilePuts(fp, cn->option1->value); + + if (cn->choice1->value) + cupsFilePrintf(fp, " %s", cn->choice1->value); + + cupsFilePutChar(fp, ' '); + + if (cn->option2->value[0] != '*') + cupsFilePutChar(fp, '*'); + + cupsFilePuts(fp, cn->option2->value); + + if (cn->choice2->value) + cupsFilePrintf(fp, " %s", cn->choice2->value); + + cupsFilePuts(fp, lf); + + // Then constrain 2 against 1... + if (!strncmp(cn->option1->value, "*Custom", 7) || + !strncmp(cn->option2->value, "*Custom", 7)) + cupsFilePuts(fp, "*NonUIConstraints: "); + else + cupsFilePuts(fp, "*UIConstraints: "); + + if (cn->option2->value[0] != '*') + cupsFilePutChar(fp, '*'); + + cupsFilePuts(fp, cn->option2->value); + + if (cn->choice2->value) + cupsFilePrintf(fp, " %s", cn->choice2->value); + + cupsFilePutChar(fp, ' '); + + if (cn->option1->value[0] != '*') + cupsFilePutChar(fp, '*'); + + cupsFilePuts(fp, cn->option1->value); + + if (cn->choice1->value) + cupsFilePrintf(fp, " %s", cn->choice1->value); + + cupsFilePuts(fp, lf); + } + + // PageSize option... + cupsFilePrintf(fp, "*OpenUI *PageSize/Media Size: PickOne%s", lf); + cupsFilePrintf(fp, "*OrderDependency: 10 AnySetup *PageSize%s", lf); + cupsFilePrintf(fp, "*DefaultPageSize: %s%s", + default_size ? default_size->value : "Letter", lf); + + for (m = (ppdcMediaSize *)sizes->first(); + m; + m = (ppdcMediaSize *)sizes->next()) + if (m->size_code->value) + { + cupsFilePrintf(fp, "*PageSize %s/%s: \"%s\"%s", + m->name->value, catalog->find_message(m->text->value), + m->size_code->value, lf); + + if (strchr(m->size_code->value, '\n') || + strchr(m->size_code->value, '\r')) + cupsFilePrintf(fp, "*End%s", lf); + } + else + cupsFilePrintf(fp, + "*PageSize %s/%s: \"<>setpagedevice\"%s", + m->name->value, catalog->find_message(m->text->value), + m->width, m->length, lf); + + if ((a = find_attr("?PageSize", NULL)) != NULL) + { + cupsFilePrintf(fp, "*?PageSize: \"%s\"%s", a->value->value, lf); + + if (strchr(a->value->value, '\n') || + strchr(a->value->value, '\r')) + cupsFilePrintf(fp, "*End%s", lf); + } + + cupsFilePrintf(fp, "*CloseUI: *PageSize%s", lf); + + // PageRegion option... + cupsFilePrintf(fp, "*OpenUI *PageRegion/Media Size: PickOne%s", lf); + cupsFilePrintf(fp, "*OrderDependency: 10 AnySetup *PageRegion%s", lf); + cupsFilePrintf(fp, "*DefaultPageRegion: %s%s", + default_size ? default_size->value : "Letter", lf); + + for (m = (ppdcMediaSize *)sizes->first(); + m; + m = (ppdcMediaSize *)sizes->next()) + if (m->region_code->value) + { + cupsFilePrintf(fp, "*PageRegion %s/%s: \"%s\"%s", + m->name->value, catalog->find_message(m->text->value), + m->region_code->value, lf); + + if (strchr(m->region_code->value, '\n') || + strchr(m->region_code->value, '\r')) + cupsFilePrintf(fp, "*End%s", lf); + } + else + cupsFilePrintf(fp, + "*PageRegion %s/%s: \"<>setpagedevice\"%s", + m->name->value, catalog->find_message(m->text->value), + m->width, m->length, lf); + + if ((a = find_attr("?PageRegion", NULL)) != NULL) + { + cupsFilePrintf(fp, "*?PageRegion: \"%s\"%s", a->value->value, lf); + + if (strchr(a->value->value, '\n') || + strchr(a->value->value, '\r')) + cupsFilePrintf(fp, "*End%s", lf); + } + + cupsFilePrintf(fp, "*CloseUI: *PageRegion%s", lf); + + // ImageableArea info... + cupsFilePrintf(fp, "*DefaultImageableArea: %s%s", + default_size ? default_size->value : "Letter", lf); + + char left[255], right[255], bottom[255], top[255]; + + for (m = (ppdcMediaSize *)sizes->first(); + m; + m = (ppdcMediaSize *)sizes->next()) + { + _cupsStrFormatd(left, left + sizeof(left), m->left, loc); + _cupsStrFormatd(bottom, bottom + sizeof(bottom), m->bottom, loc); + _cupsStrFormatd(right, right + sizeof(right), m->width - m->right, loc); + _cupsStrFormatd(top, top + sizeof(top), m->length - m->top, loc); + + cupsFilePrintf(fp, "*ImageableArea %s/%s: \"%s %s %s %s\"%s", + m->name->value, catalog->find_message(m->text->value), + left, bottom, right, top, lf); + } + + if ((a = find_attr("?ImageableArea", NULL)) != NULL) + { + cupsFilePrintf(fp, "*?ImageableArea: \"%s\"%s", a->value->value, lf); + + if (strchr(a->value->value, '\n') || + strchr(a->value->value, '\r')) + cupsFilePrintf(fp, "*End%s", lf); + } + + // PaperDimension info... + cupsFilePrintf(fp, "*DefaultPaperDimension: %s%s", + default_size ? default_size->value : "Letter", lf); + + char width[255], length[255]; + + for (m = (ppdcMediaSize *)sizes->first(); + m; + m = (ppdcMediaSize *)sizes->next()) + { + _cupsStrFormatd(width, width + sizeof(width), m->width, loc); + _cupsStrFormatd(length, length + sizeof(length), m->length, loc); + + cupsFilePrintf(fp, "*PaperDimension %s/%s: \"%s %s\"%s", + m->name->value, catalog->find_message(m->text->value), + width, length, lf); + } + + if ((a = find_attr("?PaperDimension", NULL)) != NULL) + { + cupsFilePrintf(fp, "*?PaperDimension: \"%s\"%s", a->value->value, lf); + + if (strchr(a->value->value, '\n') || + strchr(a->value->value, '\r')) + cupsFilePrintf(fp, "*End%s", lf); + } + + // Custom size support... + if (variable_paper_size) + { + _cupsStrFormatd(width, width + sizeof(width), max_width, loc); + _cupsStrFormatd(length, length + sizeof(length), max_length, loc); + + _cupsStrFormatd(left, left + sizeof(left), left_margin, loc); + _cupsStrFormatd(bottom, bottom + sizeof(bottom), bottom_margin, loc); + _cupsStrFormatd(right, right + sizeof(right), right_margin, loc); + _cupsStrFormatd(top, top + sizeof(top), top_margin, loc); + + cupsFilePrintf(fp, "*MaxMediaWidth: \"%s\"%s", width, lf); + cupsFilePrintf(fp, "*MaxMediaHeight: \"%s\"%s", length, lf); + cupsFilePrintf(fp, "*HWMargins: %s %s %s %s%s", left, bottom, right, top, + lf); + + if (custom_size_code && custom_size_code->value) + { + cupsFilePrintf(fp, "*CustomPageSize True: \"%s\"%s", + custom_size_code->value, lf); + + if (strchr(custom_size_code->value, '\n') || + strchr(custom_size_code->value, '\r')) + cupsFilePrintf(fp, "*End%s", lf); + } + else + cupsFilePrintf(fp, + "*CustomPageSize True: \"pop pop pop <>setpagedevice\"%s", lf); + + if ((a = find_attr("ParamCustomPageSize", "Width")) != NULL) + cupsFilePrintf(fp, "*ParamCustomPageSize Width: %s%s", a->value->value, + lf); + else + { + char width0[255]; + + _cupsStrFormatd(width0, width0 + sizeof(width0), min_width, loc); + _cupsStrFormatd(width, width + sizeof(width), max_width, loc); + + cupsFilePrintf(fp, "*ParamCustomPageSize Width: 1 points %s %s%s", + width0, width, lf); + } + + if ((a = find_attr("ParamCustomPageSize", "Height")) != NULL) + cupsFilePrintf(fp, "*ParamCustomPageSize Height: %s%s", a->value->value, + lf); + else + { + char length0[255]; + + _cupsStrFormatd(length0, length0 + sizeof(length0), min_length, loc); + _cupsStrFormatd(length, length + sizeof(length), max_length, loc); + + cupsFilePrintf(fp, "*ParamCustomPageSize Height: 2 points %s %s%s", + length0, length, lf); + } + + if ((a = find_attr("ParamCustomPageSize", "WidthOffset")) != NULL) + cupsFilePrintf(fp, "*ParamCustomPageSize WidthOffset: %s%s", + a->value->value, lf); + else + cupsFilePrintf(fp, "*ParamCustomPageSize WidthOffset: 3 points 0 0%s", lf); + + if ((a = find_attr("ParamCustomPageSize", "HeightOffset")) != NULL) + cupsFilePrintf(fp, "*ParamCustomPageSize HeightOffset: %s%s", + a->value->value, lf); + else + cupsFilePrintf(fp, "*ParamCustomPageSize HeightOffset: 4 points 0 0%s", lf); + + if ((a = find_attr("ParamCustomPageSize", "Orientation")) != NULL) + cupsFilePrintf(fp, "*ParamCustomPageSize Orientation: %s%s", + a->value->value, lf); + else + cupsFilePrintf(fp, "*ParamCustomPageSize Orientation: 5 int 0 0%s", lf); + } + + // All other options... + for (g = (ppdcGroup *)groups->first(); g; g = (ppdcGroup *)groups->next()) + { + if (!g->options->count) + continue; + + if (_cups_strcasecmp(g->name->value, "General")) + cupsFilePrintf(fp, "*OpenGroup: %s/%s%s", g->name->value, + catalog->find_message(g->text->value), lf); + + for (o = (ppdcOption *)g->options->first(); + o; + o = (ppdcOption *)g->options->next()) + { + if (!o->choices->count) + continue; + + if (!o->text->value) + cupsFilePrintf(fp, "*OpenUI *%s/%s: ", o->name->value, + catalog->find_message(o->name->value)); + else + cupsFilePrintf(fp, "*OpenUI *%s/%s: ", o->name->value, + catalog->find_message(o->text->value)); + + switch (o->type) + { + case PPDC_BOOLEAN : + cupsFilePrintf(fp, "Boolean%s", lf); + break; + default : + cupsFilePrintf(fp, "PickOne%s", lf); + break; + case PPDC_PICKMANY : + cupsFilePrintf(fp, "PickMany%s", lf); + break; + } + + char order[255]; + _cupsStrFormatd(order, order + sizeof(order), o->order, loc); + + cupsFilePrintf(fp, "*OrderDependency: %s ", order); + switch (o->section) + { + default : + cupsFilePrintf(fp, "AnySetup"); + break; + case PPDC_SECTION_DOCUMENT : + cupsFilePrintf(fp, "DocumentSetup"); + break; + case PPDC_SECTION_EXIT : + cupsFilePrintf(fp, "ExitServer"); + break; + case PPDC_SECTION_JCL : + cupsFilePrintf(fp, "JCLSetup"); + break; + case PPDC_SECTION_PAGE : + cupsFilePrintf(fp, "PageSetup"); + break; + case PPDC_SECTION_PROLOG : + cupsFilePrintf(fp, "Prolog"); + break; + } + + cupsFilePrintf(fp, " *%s%s", o->name->value, lf); + + if (o->defchoice) + { + // Use the programmer-supplied default... + cupsFilePrintf(fp, "*Default%s: %s%s", o->name->value, + o->defchoice->value, lf); + } + else + { + // Make the first choice the default... + c = (ppdcChoice *)o->choices->first(); + cupsFilePrintf(fp, "*Default%s: %s%s", o->name->value, c->name->value, + lf); + } + + for (c = (ppdcChoice *)o->choices->first(); + c; + c = (ppdcChoice *)o->choices->next()) + { + // Write this choice... + if (!c->text->value) + cupsFilePrintf(fp, "*%s %s/%s: \"%s\"%s", o->name->value, + c->name->value, catalog->find_message(c->name->value), + c->code->value, lf); + else + cupsFilePrintf(fp, "*%s %s/%s: \"%s\"%s", o->name->value, + c->name->value, catalog->find_message(c->text->value), + c->code->value, lf); + + // Multi-line commands need a *End line to terminate them. + if (strchr(c->code->value, '\n') || + strchr(c->code->value, '\r')) + cupsFilePrintf(fp, "*End%s", lf); + } + + snprintf(query, sizeof(query), "?%s", o->name->value); + + if ((a = find_attr(query, NULL)) != NULL) + { + cupsFilePrintf(fp, "*%s: \"%s\"%s", query, a->value->value, lf); + + if (strchr(a->value->value, '\n') || + strchr(a->value->value, '\r')) + cupsFilePrintf(fp, "*End%s", lf); + } + + cupsFilePrintf(fp, "*CloseUI: *%s%s", o->name->value, lf); + + snprintf(custom, sizeof(custom), "Custom%s", o->name->value); + if ((a = find_attr(custom, "True")) != NULL) + { + // Output custom option information... + cupsFilePrintf(fp, "*%s True: \"%s\"%s", custom, a->value->value, lf); + if (strchr(a->value->value, '\n') || strchr(a->value->value, '\r')) + cupsFilePrintf(fp, "*End%s", lf); + + snprintf(custom, sizeof(custom), "ParamCustom%s", o->name->value); + for (a = (ppdcAttr *)attrs->first(); a; a = (ppdcAttr *)attrs->next()) + { + if (strcmp(a->name->value, custom)) + continue; + + if (!a->selector->value || !a->selector->value[0]) + cupsFilePrintf(fp, "*%s", a->name->value); + else if (!a->text->value || !a->text->value[0]) + cupsFilePrintf(fp, "*%s %s/%s", a->name->value, a->selector->value, + catalog->find_message(a->selector->value)); + else + cupsFilePrintf(fp, "*%s %s/%s", a->name->value, a->selector->value, + catalog->find_message(a->text->value)); + + cupsFilePrintf(fp, ": %s%s", a->value->value, lf); + } + } + } + + if (_cups_strcasecmp(g->name->value, "General")) + cupsFilePrintf(fp, "*CloseGroup: %s%s", g->name->value, lf); + } + + if (locales) + { + // Add localizations for additional languages... + ppdcString *locale; // Locale name + ppdcCatalog *locatalog; // Message catalog for locale + + + // Write the translation strings for each language... + for (locale = (ppdcString *)locales->first(); + locale; + locale = (ppdcString *)locales->next()) + { + // Skip (US) English... + if (!strcmp(locale->value, "en") || !strcmp(locale->value, "en_US")) + continue; + + // Skip missing languages... + if ((locatalog = src->find_po(locale->value)) == NULL) + continue; + + // Do the core stuff first... + cupsFilePrintf(fp, "*%s.Translation Manufacturer/%s: \"\"%s", + locale->value, + locatalog->find_message(manufacturer->value), lf); + + if ((a = find_attr("ModelName", NULL)) != NULL) + cupsFilePrintf(fp, "*%s.Translation ModelName/%s: \"\"%s", + locale->value, + locatalog->find_message(a->value->value), lf); + else if (_cups_strncasecmp(model_name->value, manufacturer->value, + strlen(manufacturer->value))) + cupsFilePrintf(fp, "*%s.Translation ModelName/%s %s: \"\"%s", + locale->value, + locatalog->find_message(manufacturer->value), + locatalog->find_message(model_name->value), lf); + else + cupsFilePrintf(fp, "*%s.Translation ModelName/%s: \"\"%s", + locale->value, + locatalog->find_message(model_name->value), lf); + + if ((a = find_attr("ShortNickName", NULL)) != NULL) + cupsFilePrintf(fp, "*%s.Translation ShortNickName/%s: \"\"%s", + locale->value, + locatalog->find_message(a->value->value), lf); + else if (_cups_strncasecmp(model_name->value, manufacturer->value, + strlen(manufacturer->value))) + cupsFilePrintf(fp, "*%s.Translation ShortNickName/%s %s: \"\"%s", + locale->value, + locatalog->find_message(manufacturer->value), + locatalog->find_message(model_name->value), lf); + else + cupsFilePrintf(fp, "*%s.Translation ShortNickName/%s: \"\"%s", + locale->value, + locatalog->find_message(model_name->value), lf); + + if ((a = find_attr("NickName", NULL)) != NULL) + cupsFilePrintf(fp, "*%s.Translation NickName/%s: \"\"%s", + locale->value, + locatalog->find_message(a->value->value), lf); + else if (_cups_strncasecmp(model_name->value, manufacturer->value, + strlen(manufacturer->value))) + cupsFilePrintf(fp, "*%s.Translation NickName/%s %s, %s: \"\"%s", + locale->value, + locatalog->find_message(manufacturer->value), + locatalog->find_message(model_name->value), + version->value, lf); + else + cupsFilePrintf(fp, "*%s.Translation NickName/%s, %s: \"\"%s", + locale->value, + locatalog->find_message(model_name->value), + version->value, lf); + + // Then the page sizes... + cupsFilePrintf(fp, "*%s.Translation PageSize/%s: \"\"%s", locale->value, + locatalog->find_message("Media Size"), lf); + + for (m = (ppdcMediaSize *)sizes->first(); + m; + m = (ppdcMediaSize *)sizes->next()) + { + cupsFilePrintf(fp, "*%s.PageSize %s/%s: \"\"%s", locale->value, + m->name->value, locatalog->find_message(m->text->value), + lf); + } + + // Next the groups and options... + for (g = (ppdcGroup *)groups->first(); g; g = (ppdcGroup *)groups->next()) + { + if (!g->options->count) + continue; + + if (_cups_strcasecmp(g->name->value, "General")) + cupsFilePrintf(fp, "*%s.Translation %s/%s: \"\"%s", locale->value, + g->name->value, + locatalog->find_message(g->text->value), lf); + + for (o = (ppdcOption *)g->options->first(); + o; + o = (ppdcOption *)g->options->next()) + { + if (!o->choices->count) + continue; + + cupsFilePrintf(fp, "*%s.Translation %s/%s: \"\"%s", locale->value, + o->name->value, + locatalog->find_message(o->text->value ? + o->text->value : + o->name->value), lf); + + for (c = (ppdcChoice *)o->choices->first(); + c; + c = (ppdcChoice *)o->choices->next()) + { + // Write this choice... + cupsFilePrintf(fp, "*%s.%s %s/%s: \"\"%s", locale->value, + o->name->value, c->name->value, + locatalog->find_message(c->text->value ? + c->text->value : + c->name->value), lf); + } + } + } + + // Finally the localizable attributes... + for (a = (ppdcAttr *)attrs->first(); a; a = (ppdcAttr *)attrs->next()) + { + if (!a->localizable && + (!a->text || !a->text->value || !a->text->value[0]) && + strcmp(a->name->value, "APCustomColorMatchingName") && + strcmp(a->name->value, "APPrinterPreset") && + strcmp(a->name->value, "cupsICCProfile") && + strcmp(a->name->value, "cupsIPPReason") && + strcmp(a->name->value, "cupsMarkerName") && + strncmp(a->name->value, "Custom", 6) && + strncmp(a->name->value, "ParamCustom", 11)) + continue; + + cupsFilePrintf(fp, "*%s.%s %s/%s: \"%s\"%s", locale->value, + a->name->value, a->selector->value, + locatalog->find_message(a->text && a->text->value ? + a->text->value : a->name->value), + ((a->localizable && a->value->value[0]) || + !strcmp(a->name->value, "cupsIPPReason")) ? + locatalog->find_message(a->value->value) : "", + lf); + } + } + } + + if (default_font && default_font->value) + cupsFilePrintf(fp, "*DefaultFont: %s%s", default_font->value, lf); + else + cupsFilePrintf(fp, "*DefaultFont: Courier%s", lf); + + for (fn = (ppdcFont *)fonts->first(); fn; fn = (ppdcFont *)fonts->next()) + if (!strcmp(fn->name->value, "*")) + { + for (bfn = (ppdcFont *)src->base_fonts->first(); + bfn; + bfn = (ppdcFont *)src->base_fonts->next()) + cupsFilePrintf(fp, "*Font %s: %s \"%s\" %s %s%s", + bfn->name->value, bfn->encoding->value, + bfn->version->value, bfn->charset->value, + bfn->status == PPDC_FONT_ROM ? "ROM" : "Disk", lf); + } + else + cupsFilePrintf(fp, "*Font %s: %s \"%s\" %s %s%s", + fn->name->value, fn->encoding->value, fn->version->value, + fn->charset->value, + fn->status == PPDC_FONT_ROM ? "ROM" : "Disk", lf); + + cupsFilePrintf(fp, "*%% End of %s, %05d bytes.%s", pc_file_name->value, + (int)(cupsFileTell(fp) + 25 + strlen(pc_file_name->value)), + lf); + + if (delete_cat) + catalog->release(); + + return (0); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-file.cxx b/ppdc/ppdc-file.cxx new file mode 100644 index 0000000000..3d4520eb09 --- /dev/null +++ b/ppdc/ppdc-file.cxx @@ -0,0 +1,109 @@ +// +// "$Id$" +// +// File class for the CUPS PPD Compiler. +// +// Copyright 2007-2010 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcFile::ppdcFile() - Create (open) a file. +// ppdcFile::~ppdcFile() - Delete (close) a file. +// ppdcFile::get() - Get a character from a file. +// ppdcFile::peek() - Look at the next character from a file. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// 'ppdcFile::ppdcFile()' - Create (open) a file. +// + +ppdcFile::ppdcFile(const char *f, // I - File to open + cups_file_t *ffp) // I - File pointer to use +{ + if (ffp) + { + fp = ffp; + cupsFileRewind(fp); + } + else + fp = cupsFileOpen(f, "r"); + + filename = f; + line = 1; + + if (!fp) + _cupsLangPrintf(stderr, _("ppdc: Unable to open %s: %s"), f, + strerror(errno)); +} + + +// +// 'ppdcFile::~ppdcFile()' - Delete (close) a file. +// + +ppdcFile::~ppdcFile() +{ + if (fp) + cupsFileClose(fp); +} + + +// +// 'ppdcFile::get()' - Get a character from a file. +// + +int +ppdcFile::get() +{ + int ch; // Character from file + + + // Return EOF if there is no open file... + if (!fp) + return (EOF); + + // Get the character... + ch = cupsFileGetChar(fp); + + // Update the line number as needed... + if (ch == '\n') + line ++; + + // Return the character... + return (ch); +} + + +// +// 'ppdcFile::peek()' - Look at the next character from a file. +// + +int // O - Next character in file +ppdcFile::peek() +{ + // Return immediaely if there is no open file... + if (!fp) + return (EOF); + + // Otherwise return the next character without advancing... + return (cupsFilePeekChar(fp)); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-filter.cxx b/ppdc/ppdc-filter.cxx new file mode 100644 index 0000000000..33994849b8 --- /dev/null +++ b/ppdc/ppdc-filter.cxx @@ -0,0 +1,60 @@ +// +// "$Id$" +// +// Filter class for the CUPS PPD Compiler. +// +// Copyright 2007-2009 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcFilter::ppdcFilter() - Create a filter. +// ppdcFilter::~ppdcFilter() - Destroy a filter. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// 'ppdcFilter::ppdcFilter()' - Create a filter. +// + +ppdcFilter::ppdcFilter(const char *t, // I - MIME type + const char *p, // I - Filter program + int c) // I - Relative cost + : ppdcShared() +{ + PPDC_NEW; + + mime_type = new ppdcString(t); + program = new ppdcString(p); + cost = c; +} + + +// +// 'ppdcFilter::~ppdcFilter()' - Destroy a filter. +// + +ppdcFilter::~ppdcFilter() +{ + PPDC_DELETE; + + mime_type->release(); + program->release(); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-font.cxx b/ppdc/ppdc-font.cxx new file mode 100644 index 0000000000..b6d9bcb2ab --- /dev/null +++ b/ppdc/ppdc-font.cxx @@ -0,0 +1,66 @@ +// +// "$Id$" +// +// Shared font class for the CUPS PPD Compiler. +// +// Copyright 2007-2009 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcFont::ppdcFont() - Create a shared font. +// ppdcFont::~ppdcFont() - Destroy a shared font. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// 'ppdcFont::ppdcFont()' - Create a shared font. +// + +ppdcFont::ppdcFont(const char *n, // I - Name of font + const char *e, // I - Font encoding + const char *v, // I - Font version + const char *c, // I - Font charset + ppdcFontStatus s) // I - Font status + : ppdcShared() +{ + PPDC_NEW; + + name = new ppdcString(n); + encoding = new ppdcString(e); + version = new ppdcString(v); + charset = new ppdcString(c); + status = s; +} + + +// +// 'ppdcFont::~ppdcFont()' - Destroy a shared font. +// + +ppdcFont::~ppdcFont() +{ + PPDC_DELETE; + + name->release(); + encoding->release(); + version->release(); + charset->release(); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-group.cxx b/ppdc/ppdc-group.cxx new file mode 100644 index 0000000000..cc0ab280a4 --- /dev/null +++ b/ppdc/ppdc-group.cxx @@ -0,0 +1,103 @@ +// +// "$Id$" +// +// Group class for the CUPS PPD Compiler. +// +// Copyright 2007-2011 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcGroup::ppdcGroup() - Copy a new group. +// ppdcGroup::~ppdcGroup() - Destroy a group. +// ppdcGroup::find_option() - Find an option in a group. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// 'ppdcGroup::ppdcGroup()' - Create a new group. +// + +ppdcGroup::ppdcGroup(const char *n, // I - Name of group + const char *t) // I - Text of group +{ + PPDC_NEWVAL(n); + + name = new ppdcString(n); + text = new ppdcString(t); + options = new ppdcArray(); +} + + +// +// 'ppdcGroup::ppdcGroup()' - Copy a new group. +// + +ppdcGroup::ppdcGroup(ppdcGroup *g) // I - Group template +{ + PPDC_NEWVAL(g->name->value); + + g->name->retain(); + g->text->retain(); + + name = g->name; + text = g->text; + + options = new ppdcArray(); + for (ppdcOption *o = (ppdcOption *)g->options->first(); + o; + o = (ppdcOption *)g->options->next()) + options->add(new ppdcOption(o)); +} + + +// +// 'ppdcGroup::~ppdcGroup()' - Destroy a group. +// + +ppdcGroup::~ppdcGroup() +{ + PPDC_DELETEVAL(name ? name->value : NULL); + + name->release(); + text->release(); + options->release(); + + name = text = 0; + options = 0; +} + + +// +// 'ppdcGroup::find_option()' - Find an option in a group. +// + +ppdcOption * +ppdcGroup::find_option(const char *n) // I - Name of option +{ + ppdcOption *o; // Current option + + + for (o = (ppdcOption *)options->first(); o; o = (ppdcOption *)options->next()) + if (!_cups_strcasecmp(n, o->name->value)) + return (o); + + return (0); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-import.cxx b/ppdc/ppdc-import.cxx new file mode 100644 index 0000000000..4794e77151 --- /dev/null +++ b/ppdc/ppdc-import.cxx @@ -0,0 +1,343 @@ +// +// "$Id$" +// +// PPD file import methods for the CUPS PPD Compiler. +// +// Copyright 2007-2011 by Apple Inc. +// Copyright 2002-2006 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcSource::import_ppd() - Import a PPD file. +// ppd_gets() - Get a line from a PPD file. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" +#include + + +// +// 'ppdcSource::import_ppd()' - Import a PPD file. +// + +int // O - 1 on success, 0 on failure +ppdcSource::import_ppd(const char *f) // I - Filename +{ + int i, j, k; // Looping vars + cups_file_t *fp; // File + char line[256], // Comment line + *ptr; // Pointer into line + int cost; // Cost for filter + ppd_file_t *ppd; // PPD file data + ppd_group_t *group; // PPD group + ppd_option_t *option; // PPD option + ppd_choice_t *choice; // PPD choice + ppd_attr_t *attr; // PPD attribute + ppd_const_t *constraint; // PPD UI constraint + ppd_const_t *constraint2; // Temp PPD UI constraint + ppd_size_t *size; // PPD page size + ppdcDriver *driver; // Driver + ppdcFilter *filter; // Current filter + ppdcFont *font; // Font + ppdcGroup *cgroup; // UI group + ppdcOption *coption; // UI option + ppdcChoice *cchoice; // UI choice + ppdcConstraint *cconstraint; // UI constraint + ppdcMediaSize *csize; // Media size + + + // Try opening the PPD file... + if ((ppd = ppdOpenFile(f)) == NULL) + return (0); + + // All PPD files need a PCFileName attribute... + if (!ppd->pcfilename) + { + ppdClose(ppd); + return (0); + } + + // See if the driver has already been imported... + if ((driver = find_driver(ppd->pcfilename)) == NULL) + { + // Create a new PPD file... + if ((fp = cupsFileOpen(f, "r")) == NULL) + { + ppdClose(ppd); + return (0); + } + + driver = new ppdcDriver(); + driver->type = PPDC_DRIVER_PS; + + drivers->add(driver); + + // Read the initial comments from the PPD file and use them as the + // copyright/license text... + cupsFileGets(fp, line, sizeof(line)); + // Skip *PPD-Adobe-M.m + + while (cupsFileGets(fp, line, sizeof(line))) + if (strncmp(line, "*%", 2)) + break; + else if (strncmp(line, "*%%%% ", 6)) + { + for (ptr = line + 2; isspace(*ptr); ptr ++); + + driver->add_copyright(ptr); + } + + cupsFileClose(fp); + + // Then add the stuff from the PPD file... + if (ppd->modelname && ppd->manufacturer && + !_cups_strncasecmp(ppd->modelname, ppd->manufacturer, + strlen(ppd->manufacturer))) + { + ptr = ppd->modelname + strlen(ppd->manufacturer); + + while (isspace(*ptr)) + ptr ++; + } + else + ptr = ppd->modelname; + + if (ppd->nickname) + driver->add_attr(new ppdcAttr("NickName", NULL, NULL, ppd->nickname)); + + if (ppd->shortnickname) + driver->add_attr(new ppdcAttr("ShortNickName", NULL, NULL, + ppd->shortnickname)); + + driver->manufacturer = new ppdcString(ppd->manufacturer); + driver->model_name = new ppdcString(ptr); + driver->pc_file_name = new ppdcString(ppd->pcfilename); + attr = ppdFindAttr(ppd, "FileVersion", NULL); + driver->version = new ppdcString(attr ? attr->value : NULL); + driver->model_number = ppd->model_number; + driver->manual_copies = ppd->manual_copies; + driver->color_device = ppd->color_device; + driver->throughput = ppd->throughput; + driver->variable_paper_size = ppd->variable_sizes; + driver->max_width = ppd->custom_max[0]; + driver->max_length = ppd->custom_max[1]; + driver->min_width = ppd->custom_min[0]; + driver->min_length = ppd->custom_min[1]; + driver->left_margin = ppd->custom_margins[0]; + driver->bottom_margin = ppd->custom_margins[1]; + driver->right_margin = ppd->custom_margins[2]; + driver->top_margin = ppd->custom_margins[3]; + + for (i = 0; i < ppd->num_filters; i ++) + { + strlcpy(line, ppd->filters[i], sizeof(line)); + + for (ptr = line; *ptr; ptr ++) + if (isspace(*ptr & 255)) + break; + *ptr++ = '\0'; + + cost = strtol(ptr, &ptr, 10); + + while (isspace(*ptr & 255)) + ptr ++; + + filter = new ppdcFilter(line, ptr, cost); + driver->add_filter(filter); + } + + attr = ppdFindAttr(ppd, "DefaultFont", NULL); + driver->default_font = new ppdcString(attr ? attr->value : NULL); + + // Collect media sizes... + ppd_option_t *region_option, // PageRegion option + *size_option; // PageSize option + ppd_choice_t *region_choice, // PageRegion choice + *size_choice; // PageSize choice + + region_option = ppdFindOption(ppd, "PageRegion"); + size_option = ppdFindOption(ppd, "PageSize"); + + for (i = ppd->num_sizes, size = ppd->sizes; i > 0; i --, size ++) + { + // Don't do custom size here... + if (!_cups_strcasecmp(size->name, "Custom")) + continue; + + // Get the code for the PageSize and PageRegion options... + region_choice = ppdFindChoice(region_option, size->name); + size_choice = ppdFindChoice(size_option, size->name); + + // Create a new media size record and add it to the driver... + csize = new ppdcMediaSize(size->name, size_choice->text, size->width, + size->length, size->left, size->bottom, + size->width - size->right, + size->length - size->top, + size_choice->code, region_choice->code); + + driver->add_size(csize); + + if (!_cups_strcasecmp(size_option->defchoice, size->name)) + driver->set_default_size(csize); + } + + // Now all of the options... + for (i = ppd->num_groups, group = ppd->groups; i > 0; i --, group ++) + { + cgroup = new ppdcGroup(group->name, group->text); + driver->add_group(cgroup); + + for (j = group->num_options, option = group->options; j > 0; j --, option ++) + { + if (!strcmp(option->keyword, "PageSize") || !strcmp(option->keyword, "PageRegion")) + continue; + + coption = new ppdcOption((ppdcOptType)option->ui, option->keyword, + option->text, (ppdcOptSection)option->section, + option->order); + cgroup->add_option(coption); + + for (k = option->num_choices, choice = option->choices; k > 0; k --, choice ++) + { + if (!strcmp(choice->choice, "Custom")) + continue; + + cchoice = new ppdcChoice(choice->choice, choice->text, choice->code); + coption->add_choice(cchoice); + + if (!_cups_strcasecmp(option->defchoice, choice->choice)) + coption->set_defchoice(cchoice); + } + } + } + + // Now the constraints... + for (i = ppd->num_consts, constraint = ppd->consts; + i > 0; + i --, constraint ++) + { + // Look for mirrored constraints... + for (j = i - 1, constraint2 = constraint + 1; + j > 0; + j --, constraint2 ++) + if (!strcmp(constraint->option1, constraint2->option2) && + !strcmp(constraint->choice1, constraint2->choice2) && + !strcmp(constraint->option2, constraint2->option1) && + !strcmp(constraint->choice2, constraint2->choice1)) + break; + + if (j) + continue; + + cconstraint = new ppdcConstraint(constraint->option2, constraint->choice2, + constraint->option1, constraint->choice1); + driver->add_constraint(cconstraint); + } + + for (i = 0; i < ppd->num_attrs; i ++) + { + attr = ppd->attrs[i]; + + if (!strcmp(attr->name, "Font")) + { + // Font... + char encoding[256], // Encoding string + version[256], // Version string + charset[256], // Charset string + status[256]; // Status string + ppdcFontStatus fstatus; // Status enumeration + + + if (sscanf(attr->value, "%s%*[^\"]\"%[^\"]\"%s%s", encoding, version, + charset, status) != 4) + { + _cupsLangPrintf(stderr, _("ppdc: Bad font attribute: %s"), + attr->value); + continue; + } + + if (!strcmp(status, "ROM")) + fstatus = PPDC_FONT_ROM; + else + fstatus = PPDC_FONT_DISK; + + font = new ppdcFont(attr->spec, encoding, version, charset, fstatus); + + driver->add_font(font); + } + else if (!strcmp(attr->name, "CustomPageSize")) + { + driver->set_custom_size_code(attr->value); + } + else if ((strncmp(attr->name, "Default", 7) || + !strcmp(attr->name, "DefaultColorSpace")) && + strcmp(attr->name, "ColorDevice") && + strcmp(attr->name, "Manufacturer") && + strcmp(attr->name, "ModelName") && + strcmp(attr->name, "MaxMediaHeight") && + strcmp(attr->name, "MaxMediaWidth") && + strcmp(attr->name, "NickName") && + strcmp(attr->name, "ParamCustomPageSize") && + strcmp(attr->name, "ShortNickName") && + strcmp(attr->name, "Throughput") && + strcmp(attr->name, "PCFileName") && + strcmp(attr->name, "FileVersion") && + strcmp(attr->name, "FormatVersion") && + strcmp(attr->name, "HWMargins") && + strcmp(attr->name, "VariablePaperSize") && + strcmp(attr->name, "LanguageEncoding") && + strcmp(attr->name, "LanguageVersion") && + strcmp(attr->name, "cupsFilter") && + strcmp(attr->name, "cupsFlipDuplex") && + strcmp(attr->name, "cupsLanguages") && + strcmp(attr->name, "cupsManualCopies") && + strcmp(attr->name, "cupsModelNumber") && + strcmp(attr->name, "cupsVersion")) + { + if ((ptr = strchr(attr->name, '.')) != NULL && + ((ptr - attr->name) == 2 || (ptr - attr->name) == 5)) + { + // Might be a localization attribute; test further... + if (isalpha(attr->name[0] & 255) && + isalpha(attr->name[1] & 255) && + (attr->name[2] == '.' || + (attr->name[2] == '_' && isalpha(attr->name[3] & 255) && + isalpha(attr->name[4] & 255)))) + continue; + } + + // Attribute... + driver->add_attr(new ppdcAttr(attr->name, attr->spec, attr->text, + attr->value)); + } + else if (!strncmp(attr->name, "Default", 7) && + !ppdFindOption(ppd, attr->name + 7) && + strcmp(attr->name, "DefaultFont") && + strcmp(attr->name, "DefaultImageableArea") && + strcmp(attr->name, "DefaultPaperDimension") && + strcmp(attr->name, "DefaultFont")) + { + // Default attribute... + driver->add_attr(new ppdcAttr(attr->name, attr->spec, attr->text, + attr->value)); + } + } + } + + return (1); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-mediasize.cxx b/ppdc/ppdc-mediasize.cxx new file mode 100644 index 0000000000..844028a39c --- /dev/null +++ b/ppdc/ppdc-mediasize.cxx @@ -0,0 +1,85 @@ +// +// "$Id$" +// +// Shared media size class for the CUPS PPD Compiler. +// +// Copyright 2007-2009 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcMediaSize::ppdcMediaSize() - Create a new media size. +// ppdcMediaSize::~ppdcMediaSize() - Destroy a media size. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// 'ppdcMediaSize::ppdcMediaSize()' - Create a new media size. +// + +ppdcMediaSize::ppdcMediaSize(const char *n, // I - Name of media size + const char *t, // I - Text of media size + float w, // I - Width in points + float l, // I - Length in points + float lm, // I - Left margin in points + float bm, // I - Bottom margin in points + float rm, // I - Right margin in points + float tm, // I - Top margin in points + const char *sc, // I - PageSize code, if any + const char *rc) // I - PageRegion code, if any + : ppdcShared() +{ + PPDC_NEW; + + name = new ppdcString(n); + text = new ppdcString(t); + width = w; + length = l; + left = lm; + bottom = bm; + right = rm; + top = tm; + size_code = new ppdcString(sc); + region_code = new ppdcString(rc); + + if (left < 0.0f) + left = 0.0f; + if (bottom < 0.0f) + bottom = 0.0f; + if (right < 0.0f) + right = 0.0f; + if (top < 0.0f) + top = 0.0f; +} + + +// +// 'ppdcMediaSize::~ppdcMediaSize()' - Destroy a media size. +// + +ppdcMediaSize::~ppdcMediaSize() +{ + PPDC_DELETE; + + name->release(); + text->release(); + size_code->release(); + region_code->release(); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-message.cxx b/ppdc/ppdc-message.cxx new file mode 100644 index 0000000000..11e5e3eaa7 --- /dev/null +++ b/ppdc/ppdc-message.cxx @@ -0,0 +1,58 @@ +// +// "$Id$" +// +// Shared message class for the CUPS PPD Compiler. +// +// Copyright 2007-2009 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcMessage::ppdcMessage() - Create a shared message. +// ppdcMessage::~ppdcMessage() - Destroy a shared message. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// 'ppdcMessage::ppdcMessage()' - Create a shared message. +// + +ppdcMessage::ppdcMessage(const char *i, // I - ID + const char *s) // I - Text + : ppdcShared() +{ + PPDC_NEW; + + id = new ppdcString(i); + string = new ppdcString(s); +} + + +// +// 'ppdcMessage::~ppdcMessage()' - Destroy a shared message. +// + +ppdcMessage::~ppdcMessage() +{ + PPDC_DELETE; + + id->release(); + string->release(); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-option.cxx b/ppdc/ppdc-option.cxx new file mode 100644 index 0000000000..e80df3c875 --- /dev/null +++ b/ppdc/ppdc-option.cxx @@ -0,0 +1,129 @@ +// +// "$Id$" +// +// Option class for the CUPS PPD Compiler. +// +// Copyright 2007-2011 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcOption::ppdcOption() - Copy a new option. +// ppdcOption::~ppdcOption() - Destroy an option. +// ppdcOption::find_choice() - Find an option choice. +// ppdcOption::set_defchoice() - Set the default choice. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// 'ppdcOption::ppdcOption()' - Create a new option. +// + +ppdcOption::ppdcOption(ppdcOptType ot, // I - Option type + const char *n, // I - Option name + const char *t, // I - Option text + ppdcOptSection s, // I - Section + float o) // I - Ordering number + : ppdcShared() +{ + PPDC_NEW; + + type = ot; + name = new ppdcString(n); + text = new ppdcString(t); + section = s; + order = o; + choices = new ppdcArray(); + defchoice = 0; +} + + +// +// 'ppdcOption::ppdcOption()' - Copy a new option. +// + +ppdcOption::ppdcOption(ppdcOption *o) // I - Template option +{ + PPDC_NEW; + + o->name->retain(); + o->text->retain(); + if (o->defchoice) + o->defchoice->retain(); + + type = o->type; + name = o->name; + text = o->text; + section = o->section; + order = o->order; + choices = new ppdcArray(o->choices); + defchoice = o->defchoice; +} + + +// +// 'ppdcOption::~ppdcOption()' - Destroy an option. +// + +ppdcOption::~ppdcOption() +{ + PPDC_DELETE; + + name->release(); + text->release(); + if (defchoice) + defchoice->release(); + choices->release(); +} + + +// +// 'ppdcOption::find_choice()' - Find an option choice. +// + +ppdcChoice * // O - Choice or NULL +ppdcOption::find_choice(const char *n) // I - Name of choice +{ + ppdcChoice *c; // Current choice + + + for (c = (ppdcChoice *)choices->first(); c; c = (ppdcChoice *)choices->next()) + if (!_cups_strcasecmp(n, c->name->value)) + return (c); + + return (0); +} + + +// +// 'ppdcOption::set_defchoice()' - Set the default choice. +// + +void +ppdcOption::set_defchoice(ppdcChoice *c) // I - Choice +{ + if (defchoice) + defchoice->release(); + + if (c->name) + c->name->retain(); + + defchoice = c->name; +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-private.h b/ppdc/ppdc-private.h new file mode 100644 index 0000000000..2bf5a24e79 --- /dev/null +++ b/ppdc/ppdc-private.h @@ -0,0 +1,40 @@ +// +// "$Id$" +// +// Private definitions for the CUPS PPD Compiler. +// +// Copyright 2009-2010 by Apple Inc. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// + +#ifndef _PPDC_PRIVATE_H_ +# define _PPDC_PRIVATE_H_ + +// +// Include necessary headers... +// + +# include "ppdc.h" +# include + + +// +// Macros... +// + +# define PPDC_NEW DEBUG_printf(("%s: %p new", class_name(), this)) +# define PPDC_NEWVAL(s) DEBUG_printf(("%s(\"%s\"): %p new", class_name(), s, this)) +# define PPDC_DELETE DEBUG_printf(("%s: %p delete", class_name(), this)) +# define PPDC_DELETEVAL(s) DEBUG_printf(("%s(\"%s\"): %p delete", class_name(), s, this)) + + +#endif // !_PPDC_PRIVATE_H_ + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-profile.cxx b/ppdc/ppdc-profile.cxx new file mode 100644 index 0000000000..90625f423d --- /dev/null +++ b/ppdc/ppdc-profile.cxx @@ -0,0 +1,65 @@ +// +// "$Id$" +// +// Color profile class for the CUPS PPD Compiler. +// +// Copyright 2007-2009 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcProfile::ppdcProfile() - Create a color profile. +// ppdcProfile::~ppdcProfile() - Destroy a color profile. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// 'ppdcProfile::ppdcProfile()' - Create a color profile. +// + +ppdcProfile::ppdcProfile(const char *r, // I - Resolution name + const char *m, // I - Media type name + float d, // I - Density + float g, // I - Gamma + const float *p) // I - 3x3 transform matrix + : ppdcShared() +{ + PPDC_NEW; + + resolution = new ppdcString(r); + media_type = new ppdcString(m); + density = d; + gamma = g; + + memcpy(profile, p, sizeof(profile)); +} + + +// +// 'ppdcProfile::~ppdcProfile()' - Destroy a color profile. +// + +ppdcProfile::~ppdcProfile() +{ + PPDC_DELETE; + + resolution->release(); + media_type->release(); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-shared.cxx b/ppdc/ppdc-shared.cxx new file mode 100644 index 0000000000..0c99c08597 --- /dev/null +++ b/ppdc/ppdc-shared.cxx @@ -0,0 +1,88 @@ +// +// "$Id$" +// +// Shared data class for the CUPS PPD Compiler. +// +// Copyright 2007-2009 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcShared::ppdcShared() - Create shared data. +// ppdcShared::~ppdcShared() - Destroy shared data. +// ppdcShared::release() - Decrement the use count and delete as needed. +// ppdcShared::retain() - Increment the use count for this data. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// 'ppdcShared::ppdcShared()' - Create shared data. +// + +ppdcShared::ppdcShared() +{ + use = 1; +} + + +// +// 'ppdcShared::~ppdcShared()' - Destroy shared data. +// + +ppdcShared::~ppdcShared() +{ +} + + +// +// 'ppdcShared::release()' - Decrement the use count and delete as needed. +// + +void +ppdcShared::release(void) +{ + DEBUG_printf(("%s: %p release use=%d", class_name(), this, use)); + + use --; + +#ifdef DEBUG + if (use < 0) + { + fprintf(stderr, "ERROR: Over-release of %s: %p\n", class_name(), this); + abort(); + } +#endif /* DEBUG */ + + if (use == 0) + delete this; +} + + +// +// 'ppdcShared::retain()' - Increment the use count for this data. +// + +void +ppdcShared::retain() +{ + use ++; + + DEBUG_printf(("%s: %p retain use=%d", class_name(), this, use)); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-source.cxx b/ppdc/ppdc-source.cxx new file mode 100644 index 0000000000..e806d24499 --- /dev/null +++ b/ppdc/ppdc-source.cxx @@ -0,0 +1,3850 @@ +// +// "$Id$" +// +// Source class for the CUPS PPD Compiler. +// +// Copyright 2007-2011 by Apple Inc. +// Copyright 2002-2007 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcSource::ppdcSource() - Load a driver source file. +// ppdcSource::~ppdcSource() - Free a driver source file. +// ppdcSource::add_include() - Add an include directory. +// ppdcSource::find_driver() - Find a driver. +// ppdcSource::find_include() - Find an include file. +// ppdcSource::find_po() - Find a message catalog for the given +// locale. +// ppdcSource::find_size() - Find a media size. +// ppdcSource::find_variable() - Find a variable. +// ppdcSource::get_attr() - Get an attribute. +// ppdcSource::get_boolean() - Get a boolean value. +// ppdcSource::get_choice() - Get a choice. +// ppdcSource::get_color_model() - Get an old-style color model option. +// ppdcSource::get_color_order() - Get an old-style color order value. +// ppdcSource::get_color_profile() - Get a color profile definition. +// ppdcSource::get_color_space() - Get an old-style colorspace value. +// ppdcSource::get_constraint() - Get a constraint. +// ppdcSource::get_custom_size() - Get a custom media size definition from +// a file. +// ppdcSource::get_duplex() - Get a duplex option. +// ppdcSource::get_filter() - Get a filter. +// ppdcSource::get_float() - Get a single floating-point number. +// ppdcSource::get_font() - Get a font definition. +// ppdcSource::get_generic() - Get a generic old-style option. +// ppdcSource::get_group() - Get an option group. +// ppdcSource::get_installable() - Get an installable option. +// ppdcSource::get_integer() - Get an integer value from a file. +// ppdcSource::get_measurement() - Get a measurement value. +// ppdcSource::get_option() - Get an option definition. +// ppdcSource::get_po() - Get a message catalog. +// ppdcSource::get_resolution() - Get an old-style resolution option. +// ppdcSource::get_simple_profile() - Get a simple color profile definition. +// ppdcSource::get_size() - Get a media size definition from a file. +// ppdcSource::get_token() - Get a token from a file. +// ppdcSource::get_variable() - Get a variable definition. +// ppdcSource::quotef() - Write a formatted, quoted string... +// ppdcSource::read_file() - Read a driver source file. +// ppdcSource::scan_file() - Scan a driver source file. +// ppdcSource::set_variable() - Set a variable. +// ppdcSource::write_file() - Write the current source data to a file. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" +#include +#include +#include +#include +#include "data/epson.h" +#include "data/hp.h" +#include "data/label.h" +#ifndef WIN32 +# include +#endif // !WIN32 + + +// +// Class globals... +// + +ppdcArray *ppdcSource::includes = 0; +const char *ppdcSource::driver_types[] = + { + "custom", + "ps", + "escp", + "pcl", + "label", + "epson", + "hp" + }; + + +// +// 'ppdcSource::ppdcSource()' - Load a driver source file. +// + +ppdcSource::ppdcSource(const char *f, // I - File to read + cups_file_t *ffp)// I - File pointer to use + : ppdcShared() +{ + PPDC_NEW; + + filename = new ppdcString(f); + base_fonts = new ppdcArray(); + drivers = new ppdcArray(); + po_files = new ppdcArray(); + sizes = new ppdcArray(); + vars = new ppdcArray(); + cond_state = PPDC_COND_NORMAL; + cond_current = cond_stack; + cond_stack[0] = PPDC_COND_NORMAL; + + // Add standard #define variables... +#define MAKE_STRING(x) #x + + vars->add(new ppdcVariable("CUPS_VERSION", MAKE_STRING(CUPS_VERSION))); + vars->add(new ppdcVariable("CUPS_VERSION_MAJOR", MAKE_STRING(CUPS_VERSION_MAJOR))); + vars->add(new ppdcVariable("CUPS_VERSION_MINOR", MAKE_STRING(CUPS_VERSION_MINOR))); + vars->add(new ppdcVariable("CUPS_VERSION_PATCH", MAKE_STRING(CUPS_VERSION_PATCH))); + +#ifdef WIN32 + vars->add(new ppdcVariable("PLATFORM_NAME", "Windows")); + vars->add(new ppdcVariable("PLATFORM_ARCH", "X86")); + +#else + struct utsname name; // uname information + + if (!uname(&name)) + { + vars->add(new ppdcVariable("PLATFORM_NAME", name.sysname)); + vars->add(new ppdcVariable("PLATFORM_ARCH", name.machine)); + } + else + { + vars->add(new ppdcVariable("PLATFORM_NAME", "unknown")); + vars->add(new ppdcVariable("PLATFORM_ARCH", "unknown")); + } +#endif // WIN32 + + if (f) + read_file(f, ffp); +} + + +// +// 'ppdcSource::~ppdcSource()' - Free a driver source file. +// + +ppdcSource::~ppdcSource() +{ + PPDC_DELETE; + + filename->release(); + base_fonts->release(); + drivers->release(); + po_files->release(); + sizes->release(); + vars->release(); +} + + +// +// 'ppdcSource::add_include()' - Add an include directory. +// + +void +ppdcSource::add_include(const char *d) // I - Include directory +{ + if (!d) + return; + + if (!includes) + includes = new ppdcArray(); + + includes->add(new ppdcString(d)); +} + + +// +// 'ppdcSource::find_driver()' - Find a driver. +// + +ppdcDriver * // O - Driver +ppdcSource::find_driver(const char *f) // I - Driver file name +{ + ppdcDriver *d; // Current driver + + + for (d = (ppdcDriver *)drivers->first(); d; d = (ppdcDriver *)drivers->next()) + if (!_cups_strcasecmp(f, d->pc_file_name->value)) + return (d); + + return (NULL); +} + + +// +// 'ppdcSource::find_include()' - Find an include file. +// + +char * // O - Found path or NULL +ppdcSource::find_include( + const char *f, // I - Include filename + const char *base, // I - Current directory + char *n, // I - Path buffer + int nlen) // I - Path buffer length +{ + ppdcString *dir; // Include directory + char temp[1024], // Temporary path + *ptr; // Pointer to end of path + + + // Range check input... + if (!f || !*f || !n || nlen < 2) + return (0); + + // Check the first character to see if we have or "name"... + if (*f == '<') + { + // Remove the surrounding <> from the name... + strlcpy(temp, f + 1, sizeof(temp)); + ptr = temp + strlen(temp) - 1; + + if (*ptr != '>') + { + _cupsLangPrintf(stderr, + _("ppdc: Invalid #include/#po filename \"%s\"."), n); + return (0); + } + + *ptr = '\0'; + f = temp; + } + else + { + // Check for the local file relative to the current directory... + if (base && *base && f[0] != '/') + snprintf(n, nlen, "%s/%s", base, f); + else + strlcpy(n, f, nlen); + + if (!access(n, 0)) + return (n); + else if (*f == '/') + { + // Absolute path that doesn't exist... + return (0); + } + } + + // Search the include directories, if any... + if (includes) + { + for (dir = (ppdcString *)includes->first(); dir; dir = (ppdcString *)includes->next()) + { + snprintf(n, nlen, "%s/%s", dir->value, f); + if (!access(n, 0)) + return (n); + } + } + + // Search the standard include directories... + _cups_globals_t *cg = _cupsGlobals(); // Global data + + snprintf(n, nlen, "%s/ppdc/%s", cg->cups_datadir, f); + if (!access(n, 0)) + return (n); + + snprintf(n, nlen, "%s/po/%s", cg->cups_datadir, f); + if (!access(n, 0)) + return (n); + else + return (0); +} + + +// +// 'ppdcSource::find_po()' - Find a message catalog for the given locale. +// + +ppdcCatalog * // O - Message catalog or NULL +ppdcSource::find_po(const char *l) // I - Locale name +{ + ppdcCatalog *cat; // Current message catalog + + + for (cat = (ppdcCatalog *)po_files->first(); + cat; + cat = (ppdcCatalog *)po_files->next()) + if (!_cups_strcasecmp(l, cat->locale->value)) + return (cat); + + return (NULL); +} + + +// +// 'ppdcSource::find_size()' - Find a media size. +// + +ppdcMediaSize * // O - Size +ppdcSource::find_size(const char *s) // I - Size name +{ + ppdcMediaSize *m; // Current media size + + + for (m = (ppdcMediaSize *)sizes->first(); m; m = (ppdcMediaSize *)sizes->next()) + if (!_cups_strcasecmp(s, m->name->value)) + return (m); + + return (NULL); +} + + +// +// 'ppdcSource::find_variable()' - Find a variable. +// + +ppdcVariable * // O - Variable +ppdcSource::find_variable(const char *n)// I - Variable name +{ + ppdcVariable *v; // Current variable + + + for (v = (ppdcVariable *)vars->first(); v; v = (ppdcVariable *)vars->next()) + if (!_cups_strcasecmp(n, v->name->value)) + return (v); + + return (NULL); +} + + +// +// 'ppdcSource::get_attr()' - Get an attribute. +// + +ppdcAttr * // O - Attribute +ppdcSource::get_attr(ppdcFile *fp, // I - File to read + bool loc) // I - Localize this attribute? +{ + char name[1024], // Name string + selector[1024], // Selector string + *text, // Text string + value[1024]; // Value string + + + // Get the attribute parameters: + // + // Attribute name selector value + if (!get_token(fp, name, sizeof(name))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected name after %s on line %d of %s."), + loc ? "LocAttribute" : "Attribute", fp->line, fp->filename); + return (0); + } + + if (!get_token(fp, selector, sizeof(selector))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected selector after %s on line %d of %s."), + loc ? "LocAttribute" : "Attribute", fp->line, fp->filename); + return (0); + } + + if ((text = strchr(selector, '/')) != NULL) + *text++ = '\0'; + + if (!get_token(fp, value, sizeof(value))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected value after %s on line %d of %s."), + loc ? "LocAttribute" : "Attribute", fp->line, fp->filename); + return (0); + } + + return (new ppdcAttr(name, selector, text, value, loc)); +} + + +// +// 'ppdcSource::get_boolean()' - Get a boolean value. +// + +int // O - Boolean value +ppdcSource::get_boolean(ppdcFile *fp) // I - File to read +{ + char buffer[256]; // String buffer + + + if (!get_token(fp, buffer, sizeof(buffer))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected boolean value on line %d of %s."), + fp->line, fp->filename); + return (-1); + } + + if (!_cups_strcasecmp(buffer, "on") || + !_cups_strcasecmp(buffer, "yes") || + !_cups_strcasecmp(buffer, "true")) + return (1); + else if (!_cups_strcasecmp(buffer, "off") || + !_cups_strcasecmp(buffer, "no") || + !_cups_strcasecmp(buffer, "false")) + return (0); + else + { + _cupsLangPrintf(stderr, + _("ppdc: Bad boolean value (%s) on line %d of %s."), + buffer, fp->line, fp->filename); + return (-1); + } +} + + +// +// 'ppdcSource::get_choice()' - Get a choice. +// + +ppdcChoice * // O - Choice data +ppdcSource::get_choice(ppdcFile *fp) // I - File to read +{ + char name[1024], // Name + *text, // Text + code[10240]; // Code + + + // Read a choice from the file: + // + // Choice name/text code + if (!get_token(fp, name, sizeof(name))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected choice name/text on line %d of %s."), + fp->line, fp->filename); + return (NULL); + } + + if ((text = strchr(name, '/')) != NULL) + *text++ = '\0'; + else + text = name; + + if (!get_token(fp, code, sizeof(code))) + { + _cupsLangPrintf(stderr, _("ppdc: Expected choice code on line %d of %s."), + fp->line, fp->filename); + return (NULL); + } + + // Return the new choice + return (new ppdcChoice(name, text, code)); +} + + +// +// 'ppdcSource::get_color_model()' - Get an old-style color model option. +// + +ppdcChoice * // O - Choice data +ppdcSource::get_color_model(ppdcFile *fp) + // I - File to read +{ + char name[1024], // Option name + *text, // Text option + temp[256]; // Temporary string + int color_space, // Colorspace + color_order, // Color order + compression; // Compression mode + + + // Get the ColorModel parameters: + // + // ColorModel name/text colorspace colororder compression + if (!get_token(fp, name, sizeof(name))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected name/text combination for ColorModel on " + "line %d of %s."), fp->line, fp->filename); + return (NULL); + } + + if ((text = strchr(name, '/')) != NULL) + *text++ = '\0'; + else + text = name; + + if (!get_token(fp, temp, sizeof(temp))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected colorspace for ColorModel on line %d of " + "%s."), fp->line, fp->filename); + return (NULL); + } + + if ((color_space = get_color_space(temp)) < 0) + color_space = get_integer(temp); + + if (!get_token(fp, temp, sizeof(temp))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected color order for ColorModel on line %d of " + "%s."), fp->line, fp->filename); + return (NULL); + } + + if ((color_order = get_color_order(temp)) < 0) + color_order = get_integer(temp); + + if (!get_token(fp, temp, sizeof(temp))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected compression for ColorModel on line %d of " + "%s."), fp->line, fp->filename); + return (NULL); + } + + compression = get_integer(temp); + + snprintf(temp, sizeof(temp), + "<>" + "setpagedevice", + color_space, color_order, compression); + + return (new ppdcChoice(name, text, temp)); +} + + +// +// 'ppdcSource::get_color_order()' - Get an old-style color order value. +// + +int // O - Color order value +ppdcSource::get_color_order( + const char *co) // I - Color order string +{ + if (!_cups_strcasecmp(co, "chunked") || + !_cups_strcasecmp(co, "chunky")) + return (CUPS_ORDER_CHUNKED); + else if (!_cups_strcasecmp(co, "banded")) + return (CUPS_ORDER_BANDED); + else if (!_cups_strcasecmp(co, "planar")) + return (CUPS_ORDER_PLANAR); + else + return (-1); +} + + +// +// 'ppdcSource::get_color_profile()' - Get a color profile definition. +// + +ppdcProfile * // O - Color profile +ppdcSource::get_color_profile( + ppdcFile *fp) // I - File to read +{ + char resolution[1024], // Resolution/media type + *media_type; // Media type + int i; // Looping var + float g, // Gamma value + d, // Density value + m[9]; // Transform matrix + + + // Get the ColorProfile parameters: + // + // ColorProfile resolution/mediatype gamma density m00 m01 m02 ... m22 + if (!get_token(fp, resolution, sizeof(resolution))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected resolution/mediatype following " + "ColorProfile on line %d of %s."), + fp->line, fp->filename); + return (NULL); + } + + if ((media_type = strchr(resolution, '/')) != NULL) + *media_type++ = '\0'; + else + media_type = resolution; + + g = get_float(fp); + d = get_float(fp); + for (i = 0; i < 9; i ++) + m[i] = get_float(fp); + + return (new ppdcProfile(resolution, media_type, g, d, m)); +} + + +// +// 'ppdcSource::get_color_space()' - Get an old-style colorspace value. +// + +int // O - Colorspace value +ppdcSource::get_color_space( + const char *cs) // I - Colorspace string +{ + if (!_cups_strcasecmp(cs, "w")) + return (CUPS_CSPACE_W); + else if (!_cups_strcasecmp(cs, "rgb")) + return (CUPS_CSPACE_RGB); + else if (!_cups_strcasecmp(cs, "rgba")) + return (CUPS_CSPACE_RGBA); + else if (!_cups_strcasecmp(cs, "k")) + return (CUPS_CSPACE_K); + else if (!_cups_strcasecmp(cs, "cmy")) + return (CUPS_CSPACE_CMY); + else if (!_cups_strcasecmp(cs, "ymc")) + return (CUPS_CSPACE_YMC); + else if (!_cups_strcasecmp(cs, "cmyk")) + return (CUPS_CSPACE_CMYK); + else if (!_cups_strcasecmp(cs, "ymck")) + return (CUPS_CSPACE_YMCK); + else if (!_cups_strcasecmp(cs, "kcmy")) + return (CUPS_CSPACE_KCMY); + else if (!_cups_strcasecmp(cs, "kcmycm")) + return (CUPS_CSPACE_KCMYcm); + else if (!_cups_strcasecmp(cs, "gmck")) + return (CUPS_CSPACE_GMCK); + else if (!_cups_strcasecmp(cs, "gmcs")) + return (CUPS_CSPACE_GMCS); + else if (!_cups_strcasecmp(cs, "white")) + return (CUPS_CSPACE_WHITE); + else if (!_cups_strcasecmp(cs, "gold")) + return (CUPS_CSPACE_GOLD); + else if (!_cups_strcasecmp(cs, "silver")) + return (CUPS_CSPACE_SILVER); + else if (!_cups_strcasecmp(cs, "CIEXYZ")) + return (CUPS_CSPACE_CIEXYZ); + else if (!_cups_strcasecmp(cs, "CIELab")) + return (CUPS_CSPACE_CIELab); + else if (!_cups_strcasecmp(cs, "RGBW")) + return (CUPS_CSPACE_RGBW); + else if (!_cups_strcasecmp(cs, "ICC1")) + return (CUPS_CSPACE_ICC1); + else if (!_cups_strcasecmp(cs, "ICC2")) + return (CUPS_CSPACE_ICC2); + else if (!_cups_strcasecmp(cs, "ICC3")) + return (CUPS_CSPACE_ICC3); + else if (!_cups_strcasecmp(cs, "ICC4")) + return (CUPS_CSPACE_ICC4); + else if (!_cups_strcasecmp(cs, "ICC5")) + return (CUPS_CSPACE_ICC5); + else if (!_cups_strcasecmp(cs, "ICC6")) + return (CUPS_CSPACE_ICC6); + else if (!_cups_strcasecmp(cs, "ICC7")) + return (CUPS_CSPACE_ICC7); + else if (!_cups_strcasecmp(cs, "ICC8")) + return (CUPS_CSPACE_ICC8); + else if (!_cups_strcasecmp(cs, "ICC9")) + return (CUPS_CSPACE_ICC9); + else if (!_cups_strcasecmp(cs, "ICCA")) + return (CUPS_CSPACE_ICCA); + else if (!_cups_strcasecmp(cs, "ICCB")) + return (CUPS_CSPACE_ICCB); + else if (!_cups_strcasecmp(cs, "ICCC")) + return (CUPS_CSPACE_ICCC); + else if (!_cups_strcasecmp(cs, "ICCD")) + return (CUPS_CSPACE_ICCD); + else if (!_cups_strcasecmp(cs, "ICCE")) + return (CUPS_CSPACE_ICCE); + else if (!_cups_strcasecmp(cs, "ICCF")) + return (CUPS_CSPACE_ICCF); + else + return (-1); +} + + +// +// 'ppdcSource::get_constraint()' - Get a constraint. +// + +ppdcConstraint * // O - Constraint +ppdcSource::get_constraint(ppdcFile *fp)// I - File to read +{ + char temp[1024], // One string to rule them all + *ptr, // Pointer into string + *option1, // Constraint option 1 + *choice1, // Constraint choice 1 + *option2, // Constraint option 2 + *choice2; // Constraint choice 2 + + + // Read the UIConstaints parameter in one of the following forms: + // + // UIConstraints "*Option1 *Option2" + // UIConstraints "*Option1 Choice1 *Option2" + // UIConstraints "*Option1 *Option2 Choice2" + // UIConstraints "*Option1 Choice1 *Option2 Choice2" + if (!get_token(fp, temp, sizeof(temp))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected constraints string for UIConstraints on " + "line %d of %s."), fp->line, fp->filename); + return (NULL); + } + + for (ptr = temp; isspace(*ptr); ptr ++); + + if (*ptr != '*') + { + _cupsLangPrintf(stderr, + _("ppdc: Option constraint must *name on line %d of %s."), + fp->line, fp->filename); + return (NULL); + } + + option1 = ptr; + + for (; *ptr && !isspace(*ptr); ptr ++); + for (; isspace(*ptr); *ptr++ = '\0'); + + if (*ptr != '*') + { + choice1 = ptr; + + for (; *ptr && !isspace(*ptr); ptr ++); + for (; isspace(*ptr); *ptr++ = '\0'); + } + else + choice1 = NULL; + + if (*ptr != '*') + { + _cupsLangPrintf(stderr, + _("ppdc: Expected two option names on line %d of %s."), + fp->line, fp->filename); + return (NULL); + } + + option2 = ptr; + + for (; *ptr && !isspace(*ptr); ptr ++); + for (; isspace(*ptr); *ptr++ = '\0'); + + if (*ptr) + choice2 = ptr; + else + choice2 = NULL; + + return (new ppdcConstraint(option1, choice1, option2, choice2)); +} + + +// +// 'ppdcSource::get_custom_size()' - Get a custom media size definition from a file. +// + +ppdcMediaSize * // O - Media size +ppdcSource::get_custom_size(ppdcFile *fp) + // I - File to read +{ + char name[1024], // Name + *text, // Text + size_code[10240], // PageSize code + region_code[10240]; // PageRegion + float width, // Width + length, // Length + left, // Left margin + bottom, // Bottom margin + right, // Right margin + top; // Top margin + + + // Get the name, text, width, length, margins, and code: + // + // CustomMedia name/text width length left bottom right top size-code region-code + if (!get_token(fp, name, sizeof(name))) + return (NULL); + + if ((text = strchr(name, '/')) != NULL) + *text++ = '\0'; + else + text = name; + + if ((width = get_measurement(fp)) < 0.0f) + return (NULL); + + if ((length = get_measurement(fp)) < 0.0f) + return (NULL); + + if ((left = get_measurement(fp)) < 0.0f) + return (NULL); + + if ((bottom = get_measurement(fp)) < 0.0f) + return (NULL); + + if ((right = get_measurement(fp)) < 0.0f) + return (NULL); + + if ((top = get_measurement(fp)) < 0.0f) + return (NULL); + + if (!get_token(fp, size_code, sizeof(size_code))) + return (NULL); + + if (!get_token(fp, region_code, sizeof(region_code))) + return (NULL); + + // Return the new media size... + return (new ppdcMediaSize(name, text, width, length, left, bottom, + right, top, size_code, region_code)); +} + + +// +// 'ppdcSource::get_duplex()' - Get a duplex option. +// + +void +ppdcSource::get_duplex(ppdcFile *fp, // I - File to read from + ppdcDriver *d) // I - Current driver +{ + char temp[256]; // Duplex keyword + ppdcAttr *attr; // cupsFlipDuplex attribute + ppdcGroup *g; // Current group + ppdcOption *o; // Duplex option + + + // Duplex {boolean|none|normal|flip} + if (!get_token(fp, temp, sizeof(temp))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected duplex type after Duplex on line %d of " + "%s."), fp->line, fp->filename); + return; + } + + if (cond_state) + return; + + if (!_cups_strcasecmp(temp, "none") || !_cups_strcasecmp(temp, "false") || + !_cups_strcasecmp(temp, "no") || !_cups_strcasecmp(temp, "off")) + { + g = d->find_group("General"); + if ((o = g->find_option("Duplex")) != NULL) + g->options->remove(o); + + for (attr = (ppdcAttr *)d->attrs->first(); + attr; + attr = (ppdcAttr *)d->attrs->next()) + if (!strcmp(attr->name->value, "cupsFlipDuplex")) + { + d->attrs->remove(attr); + break; + } + } + else if (!_cups_strcasecmp(temp, "normal") || !_cups_strcasecmp(temp, "true") || + !_cups_strcasecmp(temp, "yes") || !_cups_strcasecmp(temp, "on") || + !_cups_strcasecmp(temp, "flip") || !_cups_strcasecmp(temp, "rotated") || + !_cups_strcasecmp(temp, "manualtumble")) + { + g = d->find_group("General"); + o = g->find_option("Duplex"); + + if (!o) + { + o = new ppdcOption(PPDC_PICKONE, "Duplex", "2-Sided Printing", + !_cups_strcasecmp(temp, "flip") ? PPDC_SECTION_PAGE : + PPDC_SECTION_ANY, 10.0f); + o->add_choice(new ppdcChoice("None", "Off (1-Sided)", + "<>setpagedevice")); + o->add_choice(new ppdcChoice("DuplexNoTumble", "Long-Edge (Portrait)", + "<>setpagedevice")); + o->add_choice(new ppdcChoice("DuplexTumble", "Short-Edge (Landscape)", + "<>setpagedevice")); + + g->add_option(o); + } + + for (attr = (ppdcAttr *)d->attrs->first(); + attr; + attr = (ppdcAttr *)d->attrs->next()) + if (!strcmp(attr->name->value, "cupsFlipDuplex")) + { + if (_cups_strcasecmp(temp, "flip")) + d->attrs->remove(attr); + break; + } + + if (!_cups_strcasecmp(temp, "flip") && !attr) + d->add_attr(new ppdcAttr("cupsFlipDuplex", NULL, NULL, "true")); + + for (attr = (ppdcAttr *)d->attrs->first(); + attr; + attr = (ppdcAttr *)d->attrs->next()) + if (!strcmp(attr->name->value, "cupsBackSide")) + { + d->attrs->remove(attr); + break; + } + + if (!_cups_strcasecmp(temp, "flip")) + d->add_attr(new ppdcAttr("cupsBackSide", NULL, NULL, "Flipped")); + else if (!_cups_strcasecmp(temp, "rotated")) + d->add_attr(new ppdcAttr("cupsBackSide", NULL, NULL, "Rotated")); + else if (!_cups_strcasecmp(temp, "manualtumble")) + d->add_attr(new ppdcAttr("cupsBackSide", NULL, NULL, "ManualTumble")); + else + d->add_attr(new ppdcAttr("cupsBackSide", NULL, NULL, "Normal")); + } + else + _cupsLangPrintf(stderr, + _("ppdc: Unknown duplex type \"%s\" on line %d of %s."), + temp, fp->line, fp->filename); +} + + +// +// 'ppdcSource::get_filter()' - Get a filter. +// + +ppdcFilter * // O - Filter +ppdcSource::get_filter(ppdcFile *fp) // I - File to read +{ + char type[1024], // MIME type + program[1024], // Filter program + *ptr; // Pointer into MIME type + int cost; // Relative cost + + + // Read filter parameters in one of the following formats: + // + // Filter "type cost program" + // Filter type cost program + + if (!get_token(fp, type, sizeof(type))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected a filter definition on line %d of %s."), + fp->line, fp->filename); + return (NULL); + } + + if ((ptr = strchr(type, ' ')) != NULL) + { + // Old-style filter definition in one string... + *ptr++ = '\0'; + cost = strtol(ptr, &ptr, 10); + + while (isspace(*ptr)) + ptr ++; + + strcpy(program, ptr); + } + else + { + cost = get_integer(fp); + + if (!get_token(fp, program, sizeof(program))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected a program name on line %d of %s."), + fp->line, fp->filename); + return (NULL); + } + } + + if (!type[0]) + { + _cupsLangPrintf(stderr, + _("ppdc: Invalid empty MIME type for filter on line %d of " + "%s."), fp->line, fp->filename); + return (NULL); + } + + if (cost < 0 || cost > 200) + { + _cupsLangPrintf(stderr, + _("ppdc: Invalid cost for filter on line %d of %s."), + fp->line, fp->filename); + return (NULL); + } + + if (!program[0]) + { + _cupsLangPrintf(stderr, + _("ppdc: Invalid empty program name for filter on line %d " + "of %s."), fp->line, fp->filename); + return (NULL); + } + + return (new ppdcFilter(type, program, cost)); +} + + +// +// 'ppdcSource::get_float()' - Get a single floating-point number. +// + +float // O - Number +ppdcSource::get_float(ppdcFile *fp) // I - File to read +{ + char temp[256], // String buffer + *ptr; // Pointer into buffer + float val; // Floating point value + + + // Get the number from the file and range-check... + if (!get_token(fp, temp, sizeof(temp))) + { + _cupsLangPrintf(stderr, _("ppdc: Expected real number on line %d of %s."), + fp->line, fp->filename); + return (-1.0f); + } + + val = (float)strtod(temp, &ptr); + + if (*ptr) + { + _cupsLangPrintf(stderr, + _("ppdc: Unknown trailing characters in real number \"%s\" " + "on line %d of %s."), temp, fp->line, fp->filename); + return (-1.0f); + } + else + return (val); +} + + +// +// 'ppdcSource::get_font()' - Get a font definition. +// + +ppdcFont * // O - Font data +ppdcSource::get_font(ppdcFile *fp) // I - File to read +{ + char name[256], // Font name + encoding[256], // Font encoding + version[256], // Font version + charset[256], // Font charset + temp[256]; // Font status string + ppdcFontStatus status; // Font status enumeration + + + // Read font parameters as follows: + // + // Font * + // Font name encoding version charset status + // %font name encoding version charset status + // + // "Name" is the PostScript font name. + // + // "Encoding" is the default encoding of the font: Standard, ISOLatin1, + // Special, Expert, ExpertSubset, etc. + // + // "Version" is the version number string. + // + // "Charset" specifies the characters that are included in the font: + // Standard, Special, Expert, Adobe-Identity, etc. + // + // "Status" is the keyword ROM or Disk. + if (!get_token(fp, name, sizeof(name))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected name after Font on line %d of %s."), + fp->line, fp->filename); + return (0); + } + + if (!strcmp(name, "*")) + { + // Include all base fonts... + encoding[0] = '\0'; + version[0] = '\0'; + charset[0] = '\0'; + status = PPDC_FONT_ROM; + } + else + { + // Load a full font definition... + if (!get_token(fp, encoding, sizeof(encoding))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected encoding after Font on line %d of " + "%s."), fp->line, fp->filename); + return (0); + } + + if (!get_token(fp, version, sizeof(version))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected version after Font on line %d of " + "%s."), fp->line, fp->filename); + return (0); + } + + if (!get_token(fp, charset, sizeof(charset))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected charset after Font on line %d of " + "%s."), fp->line, fp->filename); + return (0); + } + + if (!get_token(fp, temp, sizeof(temp))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected status after Font on line %d of %s."), + fp->line, fp->filename); + return (0); + } + + if (!_cups_strcasecmp(temp, "ROM")) + status = PPDC_FONT_ROM; + else if (!_cups_strcasecmp(temp, "Disk")) + status = PPDC_FONT_DISK; + else + { + _cupsLangPrintf(stderr, + _("ppdc: Bad status keyword %s on line %d of %s."), + temp, fp->line, fp->filename); + return (0); + } + } + +// printf("Font %s %s %s %s %s\n", name, encoding, version, charset, temp); + + return (new ppdcFont(name, encoding, version, charset, status)); +} + + +// +// 'ppdcSource::get_generic()' - Get a generic old-style option. +// + +ppdcChoice * // O - Choice data +ppdcSource::get_generic(ppdcFile *fp, // I - File to read + const char *keyword, + // I - Keyword name + const char *tattr, + // I - Text attribute + const char *nattr) + // I - Numeric attribute +{ + char name[1024], // Name + *text, // Text + command[256]; // Command string + int val; // Numeric value + + + // Read one of the following parameters: + // + // Foo name/text + // Foo integer name/text + if (nattr) + val = get_integer(fp); + else + val = 0; + + if (!get_token(fp, name, sizeof(name))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected name/text after %s on line %d of %s."), + keyword, fp->line, fp->filename); + return (NULL); + } + + if ((text = strchr(name, '/')) != NULL) + *text++ = '\0'; + else + text = name; + + if (nattr) + { + if (tattr) + snprintf(command, sizeof(command), + "<>setpagedevice", + tattr, name, nattr, val); + else + snprintf(command, sizeof(command), + "<>setpagedevice", + nattr, val); + } + else + snprintf(command, sizeof(command), + "<>setpagedevice", + tattr, name); + + return (new ppdcChoice(name, text, command)); +} + + +// +// 'ppdcSource::get_group()' - Get an option group. +// + +ppdcGroup * // O - Group +ppdcSource::get_group(ppdcFile *fp, // I - File to read + ppdcDriver *d) // I - Printer driver +{ + char name[1024], // UI name + *text; // UI text + ppdcGroup *g; // Group + + + // Read the Group parameters: + // + // Group name/text + if (!get_token(fp, name, sizeof(name))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected group name/text on line %d of %s."), + fp->line, fp->filename); + return (NULL); + } + + if ((text = strchr(name, '/')) != NULL) + *text++ = '\0'; + else + text = name; + + // See if the group already exists... + if ((g = d->find_group(name)) == NULL) + { + // Nope, add a new one... + g = new ppdcGroup(name, text); + } + + return (g); +} + + +// +// 'ppdcSource::get_installable()' - Get an installable option. +// + +ppdcOption * // O - Option +ppdcSource::get_installable(ppdcFile *fp) + // I - File to read +{ + char name[1024], // Name for installable option + *text; // Text for installable option + ppdcOption *o; // Option + + + // Read the parameter for an installable option: + // + // Installable name/text + if (!get_token(fp, name, sizeof(name))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected name/text after Installable on line %d " + "of %s."), fp->line, fp->filename); + return (NULL); + } + + if ((text = strchr(name, '/')) != NULL) + *text++ = '\0'; + else + text = name; + + // Create the option... + o = new ppdcOption(PPDC_BOOLEAN, name, text, PPDC_SECTION_ANY, 10.0f); + + // Add the false and true choices... + o->add_choice(new ppdcChoice("False", "Not Installed", "")); + o->add_choice(new ppdcChoice("True", "Installed", "")); + + return (o); +} + + +// +// 'ppdcSource::get_integer()' - Get an integer value from a string. +// + +#define PPDC_XX -1 // Bad +#define PPDC_EQ 0 // == +#define PPDC_NE 1 // != +#define PPDC_LT 2 // < +#define PPDC_LE 3 // <= +#define PPDC_GT 4 // > +#define PPDC_GE 5 // >= + +int // O - Integer value +ppdcSource::get_integer(const char *v) // I - Value string +{ + long val; // Value + long temp, // Temporary value + temp2; // Second temporary value + char *newv, // New value string pointer + ch; // Temporary character + ppdcVariable *var; // #define variable + int compop; // Comparison operator + + + // Parse the value string... + if (!v) + return (-1); + + if (isdigit(*v & 255) || *v == '-' || *v == '+') + { + // Return a simple integer value + val = strtol(v, (char **)&v, 0); + if (*v || val == LONG_MIN) + return (-1); + else + return ((int)val); + } + else if (*v == '(') + { + // Evaluate and expression in any of the following formats: + // + // (number number ... number) Bitwise OR of all numbers + // (NAME == value) 1 if equal, 0 otherwise + // (NAME != value) 1 if not equal, 0 otherwise + // (NAME < value) 1 if less than, 0 otherwise + // (NAME <= value) 1 if less than or equal, 0 otherwise + // (NAME > value) 1 if greater than, 0 otherwise + // (NAME >= value) 1 if greater than or equal, 0 otherwise + + v ++; + val = 0; + + while (*v && *v != ')') + { + // Skip leading whitespace... + while (*v && isspace(*v & 255)) + v ++; + + if (!*v || *v == ')') + break; + + if (isdigit(*v & 255) || *v == '-' || *v == '+') + { + // Bitwise OR a number... + temp = strtol(v, &newv, 0); + + if (!*newv || newv == v || !(isspace(*newv) || *newv == ')') || + temp == LONG_MIN) + return (-1); + } + else + { + // NAME logicop value + for (newv = (char *)v + 1; + *newv && (isalnum(*newv & 255) || *newv == '_'); + newv ++); + + ch = *newv; + *newv = '\0'; + + if ((var = find_variable(v)) != NULL) + { + if (!var->value || !var->value->value || !var->value->value[0]) + temp = 0; + else if (isdigit(var->value->value[0] & 255) || + var->value->value[0] == '-' || + var->value->value[0] == '+') + temp = strtol(var->value->value, NULL, 0); + else + temp = 1; + } + else + temp = 0; + + *newv = ch; + while (isspace(*newv & 255)) + newv ++; + + if (!strncmp(newv, "==", 2)) + { + compop = PPDC_EQ; + newv += 2; + } + else if (!strncmp(newv, "!=", 2)) + { + compop = PPDC_NE; + newv += 2; + } + else if (!strncmp(newv, "<=", 2)) + { + compop = PPDC_LE; + newv += 2; + } + else if (*newv == '<') + { + compop = PPDC_LT; + newv ++; + } + else if (!strncmp(newv, ">=", 2)) + { + compop = PPDC_GE; + newv += 2; + } + else if (*newv == '>') + { + compop = PPDC_GT; + newv ++; + } + else + compop = PPDC_XX; + + if (compop != PPDC_XX) + { + while (isspace(*newv & 255)) + newv ++; + + if (*newv == ')' || !*newv) + return (-1); + + if (isdigit(*newv & 255) || *newv == '-' || *newv == '+') + { + // Get the second number... + temp2 = strtol(newv, &newv, 0); + if (!*newv || newv == v || !(isspace(*newv) || *newv == ')') || + temp == LONG_MIN) + return (-1); + } + else + { + // Lookup the second name... + for (v = newv, newv ++; + *newv && (isalnum(*newv & 255) || *newv == '_'); + newv ++); + + ch = *newv; + *newv = '\0'; + + if ((var = find_variable(v)) != NULL) + { + if (!var->value || !var->value->value || !var->value->value[0]) + temp2 = 0; + else if (isdigit(var->value->value[0] & 255) || + var->value->value[0] == '-' || + var->value->value[0] == '+') + temp2 = strtol(var->value->value, NULL, 0); + else + temp2 = 1; + } + else + temp2 = 0; + + *newv = ch; + } + + // Do the comparison... + switch (compop) + { + case PPDC_EQ : + temp = temp == temp2; + break; + case PPDC_NE : + temp = temp != temp2; + break; + case PPDC_LT : + temp = temp < temp2; + break; + case PPDC_LE : + temp = temp <= temp2; + break; + case PPDC_GT : + temp = temp > temp2; + break; + case PPDC_GE : + temp = temp >= temp2; + break; + } + } + } + + val |= temp; + v = newv; + } + + if (*v == ')' && !v[1]) + return ((int)val); + else + return (-1); + } + else if ((var = find_variable(v)) != NULL) + { + // NAME by itself returns 1 if the #define variable is not blank and + // not "0"... + return (var->value->value && var->value->value[0] && + strcmp(var->value->value, "0")); + } + else + { + // Anything else is an error... + return (-1); + } +} + + +// +// 'ppdcSource::get_integer()' - Get an integer value from a file. +// + +int // O - Integer value +ppdcSource::get_integer(ppdcFile *fp) // I - File to read +{ + char temp[1024]; // String buffer + + + if (!get_token(fp, temp, sizeof(temp))) + { + _cupsLangPrintf(stderr, _("ppdc: Expected integer on line %d of %s."), + fp->line, fp->filename); + return (-1); + } + else + return (get_integer(temp)); +} + + +// +// 'ppdcSource::get_measurement()' - Get a measurement value. +// + +float // O - Measurement value in points +ppdcSource::get_measurement(ppdcFile *fp) + // I - File to read +{ + char buffer[256], // Number buffer + *ptr; // Pointer into buffer + float val; // Measurement value + + + // Grab a token from the file... + if (!get_token(fp, buffer, sizeof(buffer))) + return (-1.0f); + + // Get the floating point value of "s" and skip all digits and decimal points. + val = (float)strtod(buffer, &ptr); + + // Check for a trailing unit specifier... + if (!_cups_strcasecmp(ptr, "mm")) + val *= 72.0f / 25.4f; + else if (!_cups_strcasecmp(ptr, "cm")) + val *= 72.0f / 2.54f; + else if (!_cups_strcasecmp(ptr, "m")) + val *= 72.0f / 0.0254f; + else if (!_cups_strcasecmp(ptr, "in")) + val *= 72.0f; + else if (!_cups_strcasecmp(ptr, "ft")) + val *= 72.0f * 12.0f; + else if (_cups_strcasecmp(ptr, "pt") && *ptr) + return (-1.0f); + + return (val); +} + + +// +// 'ppdcSource::get_option()' - Get an option definition. +// + +ppdcOption * // O - Option +ppdcSource::get_option(ppdcFile *fp, // I - File to read + ppdcDriver *d, // I - Printer driver + ppdcGroup *g) // I - Current group +{ + char name[1024], // UI name + *text, // UI text + type[256]; // UI type string + ppdcOptType ot; // Option type value + ppdcOptSection section; // Option section + float order; // Option order + ppdcOption *o; // Option + ppdcGroup *mg; // Matching group, if any + + + // Read the Option parameters: + // + // Option name/text type section order + if (!get_token(fp, name, sizeof(name))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected option name/text on line %d of %s."), + fp->line, fp->filename); + return (NULL); + } + + if ((text = strchr(name, '/')) != NULL) + *text++ = '\0'; + else + text = name; + + if (!get_token(fp, type, sizeof(type))) + { + _cupsLangPrintf(stderr, _("ppdc: Expected option type on line %d of %s."), + fp->line, fp->filename); + return (NULL); + } + + if (!_cups_strcasecmp(type, "boolean")) + ot = PPDC_BOOLEAN; + else if (!_cups_strcasecmp(type, "pickone")) + ot = PPDC_PICKONE; + else if (!_cups_strcasecmp(type, "pickmany")) + ot = PPDC_PICKMANY; + else + { + _cupsLangPrintf(stderr, + _("ppdc: Invalid option type \"%s\" on line %d of %s."), + type, fp->line, fp->filename); + return (NULL); + } + + if (!get_token(fp, type, sizeof(type))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected option section on line %d of %s."), + fp->line, fp->filename); + return (NULL); + } + + if (!_cups_strcasecmp(type, "AnySetup")) + section = PPDC_SECTION_ANY; + else if (!_cups_strcasecmp(type, "DocumentSetup")) + section = PPDC_SECTION_DOCUMENT; + else if (!_cups_strcasecmp(type, "ExitServer")) + section = PPDC_SECTION_EXIT; + else if (!_cups_strcasecmp(type, "JCLSetup")) + section = PPDC_SECTION_JCL; + else if (!_cups_strcasecmp(type, "PageSetup")) + section = PPDC_SECTION_PAGE; + else if (!_cups_strcasecmp(type, "Prolog")) + section = PPDC_SECTION_PROLOG; + else + { + _cupsLangPrintf(stderr, + _("ppdc: Invalid option section \"%s\" on line %d of " + "%s."), type, fp->line, fp->filename); + return (NULL); + } + + order = get_float(fp); + + // See if the option already exists... + if ((o = d->find_option_group(name, &mg)) == NULL) + { + // Nope, add a new one... + o = new ppdcOption(ot, name, text, section, order); + } + else if (o->type != ot) + { + _cupsLangPrintf(stderr, + _("ppdc: Option %s redefined with a different type on line " + "%d of %s."), name, fp->line, fp->filename); + return (NULL); + } + else if (g != mg) + { + _cupsLangPrintf(stderr, + _("ppdc: Option %s defined in two different groups on line " + "%d of %s."), name, fp->line, fp->filename); + return (NULL); + } + + return (o); +} + + +// +// 'ppdcSource::get_po()' - Get a message catalog. +// + +ppdcCatalog * // O - Message catalog +ppdcSource::get_po(ppdcFile *fp) // I - File to read +{ + char locale[32], // Locale name + poname[1024], // Message catalog filename + basedir[1024], // Base directory + *baseptr, // Pointer into directory + pofilename[1024]; // Full filename of message catalog + ppdcCatalog *cat; // Message catalog + + + // Read the #po parameters: + // + // #po locale "filename.po" + if (!get_token(fp, locale, sizeof(locale))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected locale after #po on line %d of %s."), + fp->line, fp->filename); + return (NULL); + } + + if (!get_token(fp, poname, sizeof(poname))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected filename after #po %s on line %d of " + "%s."), locale, fp->line, fp->filename); + return (NULL); + } + + // See if the locale is already loaded... + if (find_po(locale)) + { + _cupsLangPrintf(stderr, + _("ppdc: Duplicate #po for locale %s on line %d of %s."), + locale, fp->line, fp->filename); + return (NULL); + } + + // Figure out the current directory... + strlcpy(basedir, fp->filename, sizeof(basedir)); + + if ((baseptr = strrchr(basedir, '/')) != NULL) + *baseptr = '\0'; + else + strcpy(basedir, "."); + + // Find the po file... + pofilename[0] = '\0'; + + if (!poname[0] || + find_include(poname, basedir, pofilename, sizeof(pofilename))) + { + // Found it, so load it... + cat = new ppdcCatalog(locale, pofilename); + + // Reset the filename to the name supplied by the user... + cat->filename->release(); + cat->filename = new ppdcString(poname); + + // Return the catalog... + return (cat); + } + else + { + _cupsLangPrintf(stderr, + _("ppdc: Unable to find #po file %s on line %d of %s."), + poname, fp->line, fp->filename); + return (NULL); + } +} + + +// +// 'ppdcSource::get_resolution()' - Get an old-style resolution option. +// + +ppdcChoice * // O - Choice data +ppdcSource::get_resolution(ppdcFile *fp)// I - File to read +{ + char name[1024], // Name + *text, // Text + temp[256], // Temporary string + command[256], // Command string + *commptr; // Pointer into command + int xdpi, ydpi, // X + Y resolution + color_order, // Color order + color_space, // Colorspace + compression, // Compression mode + depth, // Bits per color + row_count, // Row count + row_feed, // Row feed + row_step; // Row step/interval + + + // Read the resolution parameters: + // + // Resolution colorspace bits row-count row-feed row-step name/text + if (!get_token(fp, temp, sizeof(temp))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected override field after Resolution on line " + "%d of %s."), fp->line, fp->filename); + return (NULL); + } + + color_order = get_color_order(temp); + color_space = get_color_space(temp); + compression = get_integer(temp); + + depth = get_integer(fp); + row_count = get_integer(fp); + row_feed = get_integer(fp); + row_step = get_integer(fp); + + if (!get_token(fp, name, sizeof(name))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected name/text after Resolution on line %d of " + "%s."), fp->line, fp->filename); + return (NULL); + } + + if ((text = strchr(name, '/')) != NULL) + *text++ = '\0'; + else + text = name; + + switch (sscanf(name, "%dx%d", &xdpi, &ydpi)) + { + case 0 : + _cupsLangPrintf(stderr, + _("ppdc: Bad resolution name \"%s\" on line %d of " + "%s."), name, fp->line, fp->filename); + break; + case 1 : + ydpi = xdpi; + break; + } + + // Create the necessary PS commands... + snprintf(command, sizeof(command), + "<= 0) + { + snprintf(commptr, sizeof(command) - (commptr - command), + "/cupsColorOrder %d", color_order); + commptr += strlen(commptr); + } + + if (color_space >= 0) + { + snprintf(commptr, sizeof(command) - (commptr - command), + "/cupsColorSpace %d", color_space); + commptr += strlen(commptr); + } + + if (compression >= 0) + { + snprintf(commptr, sizeof(command) - (commptr - command), + "/cupsCompression %d", compression); + commptr += strlen(commptr); + } + + snprintf(commptr, sizeof(command) - (commptr - command), ">>setpagedevice"); + + // Return the new choice... + return (new ppdcChoice(name, text, command)); +} + + +// +// 'ppdcSource::get_simple_profile()' - Get a simple color profile definition. +// + +ppdcProfile * // O - Color profile +ppdcSource::get_simple_profile(ppdcFile *fp) + // I - File to read +{ + char resolution[1024], // Resolution/media type + *media_type; // Media type + float m[9]; // Transform matrix + float kd, rd, g; // Densities and gamma + float red, green, blue; // RGB adjustments + float yellow; // Yellow density + float color; // Color density values + + + // Get the SimpleColorProfile parameters: + // + // SimpleColorProfile resolution/mediatype black-density yellow-density + // red-density gamma red-adjust green-adjust blue-adjust + if (!get_token(fp, resolution, sizeof(resolution))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected resolution/mediatype following " + "SimpleColorProfile on line %d of %s."), + fp->line, fp->filename); + return (NULL); + } + + if ((media_type = strchr(resolution, '/')) != NULL) + *media_type++ = '\0'; + else + media_type = resolution; + + // Collect the profile parameters... + kd = get_float(fp); + yellow = get_float(fp); + rd = get_float(fp); + g = get_float(fp); + red = get_float(fp); + green = get_float(fp); + blue = get_float(fp); + + // Build the color profile... + color = 0.5f * rd / kd - kd; + m[0] = 1.0f; // C + m[1] = color + blue; // C + M (blue) + m[2] = color - green; // C + Y (green) + m[3] = color - blue; // M + C (blue) + m[4] = 1.0f; // M + m[5] = color + red; // M + Y (red) + m[6] = yellow * (color + green); // Y + C (green) + m[7] = yellow * (color - red); // Y + M (red) + m[8] = yellow; // Y + + if (m[1] > 0.0f) + { + m[3] -= m[1]; + m[1] = 0.0f; + } + else if (m[3] > 0.0f) + { + m[1] -= m[3]; + m[3] = 0.0f; + } + + if (m[2] > 0.0f) + { + m[6] -= m[2]; + m[2] = 0.0f; + } + else if (m[6] > 0.0f) + { + m[2] -= m[6]; + m[6] = 0.0f; + } + + if (m[5] > 0.0f) + { + m[7] -= m[5]; + m[5] = 0.0f; + } + else if (m[7] > 0.0f) + { + m[5] -= m[7]; + m[7] = 0.0f; + } + + // Return the new profile... + return (new ppdcProfile(resolution, media_type, g, kd, m)); +} + + +// +// 'ppdcSource::get_size()' - Get a media size definition from a file. +// + +ppdcMediaSize * // O - Media size +ppdcSource::get_size(ppdcFile *fp) // I - File to read +{ + char name[1024], // Name + *text; // Text + float width, // Width + length; // Length + + + // Get the name, text, width, and length: + // + // #media name/text width length + if (!get_token(fp, name, sizeof(name))) + return (NULL); + + if ((text = strchr(name, '/')) != NULL) + *text++ = '\0'; + else + text = name; + + if ((width = get_measurement(fp)) < 0.0f) + return (NULL); + + if ((length = get_measurement(fp)) < 0.0f) + return (NULL); + + // Return the new media size... + return (new ppdcMediaSize(name, text, width, length, 0.0f, 0.0f, 0.0f, 0.0f)); +} + + +// +// 'ppdcSource::get_token()' - Get a token from a file. +// + +char * // O - Token string or NULL +ppdcSource::get_token(ppdcFile *fp, // I - File to read + char *buffer, // I - Buffer + int buflen) // I - Length of buffer +{ + char *bufptr, // Pointer into string buffer + *bufend; // End of string buffer + int ch, // Character from file + nextch, // Next char in file + quote, // Quote character used... + empty, // Empty input? + startline; // Start line for quote + char name[256], // Name string + *nameptr; // Name pointer + ppdcVariable *var; // Variable pointer + + + // Mark the beginning and end of the buffer... + bufptr = buffer; + bufend = buffer + buflen - 1; + + // Loop intil we've read a token... + quote = 0; + startline = 0; + empty = 1; + + while ((ch = fp->get()) != EOF) + { + if (isspace(ch) && !quote) + { + if (empty) + continue; + else + break; + } + else if (ch == '$') + { + // Variable substitution + empty = 0; + + for (nameptr = name; (ch = fp->peek()) != EOF;) + { + if (!isalnum(ch) && ch != '_') + break; + else if (nameptr < (name + sizeof(name) - 1)) + *nameptr++ = fp->get(); + } + + if (nameptr == name) + { + // Just substitute this character... + if (ch == '$') + { + // $$ = $ + if (bufptr < bufend) + *bufptr++ = fp->get(); + } + else + { + // $ch = $ch + _cupsLangPrintf(stderr, + _("ppdc: Bad variable substitution ($%c) on line %d " + "of %s."), ch, fp->line, fp->filename); + + if (bufptr < bufend) + *bufptr++ = '$'; + } + } + else + { + // Substitute the variable value... + *nameptr = '\0'; + var = find_variable(name); + if (var) + { + strlcpy(bufptr, var->value->value, bufend - bufptr + 1); + bufptr += strlen(bufptr); + } + else + { + if (!(cond_state & PPDC_COND_SKIP)) + _cupsLangPrintf(stderr, + _("ppdc: Undefined variable (%s) on line %d of " + "%s."), name, fp->line, fp->filename); + + snprintf(bufptr, bufend - bufptr + 1, "$%s", name); + bufptr += strlen(bufptr); + } + } + } + else if (ch == '/' && !quote) + { + // Possibly a comment... + nextch = fp->peek(); + + if (nextch == '*') + { + // C comment... + fp->get(); + ch = fp->get(); + while ((nextch = fp->get()) != EOF) + { + if (ch == '*' && nextch == '/') + break; + + ch = nextch; + } + + if (nextch == EOF) + break; + } + else if (nextch == '/') + { + // C++ comment... + while ((nextch = fp->get()) != EOF) + if (nextch == '\n') + break; + + if (nextch == EOF) + break; + } + else + { + // Not a comment... + empty = 0; + + if (bufptr < bufend) + *bufptr++ = ch; + } + } + else if (ch == '\'' || ch == '\"') + { + empty = 0; + + if (quote == ch) + { + // Ending the current quoted string... + quote = 0; + } + else if (quote) + { + // Insert the opposing quote char... + if (bufptr < bufend) + *bufptr++ = ch; + } + else + { + // Start a new quoted string... + startline = fp->line; + quote = ch; + } + } + else if ((ch == '(' || ch == '<') && !quote) + { + empty = 0; + quote = ch; + startline = fp->line; + + if (bufptr < bufend) + *bufptr++ = ch; + } + else if ((ch == ')' && quote == '(') || (ch == '>' && quote == '<')) + { + quote = 0; + + if (bufptr < bufend) + *bufptr++ = ch; + } + else if (ch == '\\') + { + empty = 0; + + if ((ch = fp->get()) == EOF) + break; + + if (bufptr < bufend) + *bufptr++ = ch; + } + else if (bufptr < bufend) + { + empty = 0; + + *bufptr++ = ch; + + if ((ch == '{' || ch == '}') && !quote) + break; + } + } + + if (quote) + { + _cupsLangPrintf(stderr, + _("ppdc: Unterminated string starting with %c on line %d " + "of %s."), quote, startline, fp->filename); + return (NULL); + } + + if (empty) + return (NULL); + else + { + *bufptr = '\0'; +// puts(buffer); + return (buffer); + } +} + + +// +// 'ppdcSource::get_variable()' - Get a variable definition. +// + +ppdcVariable * // O - Variable +ppdcSource::get_variable(ppdcFile *fp) // I - File to read +{ + char name[1024], // Name + value[1024]; // Value + + + // Get the name and value: + // + // #define name value + if (!get_token(fp, name, sizeof(name))) + return (NULL); + + if (!get_token(fp, value, sizeof(value))) + return (NULL); + + // Set the variable... + return (set_variable(name, value)); +} + + +// +// 'ppdcSource::quotef()' - Write a formatted, quoted string... +// + +int // O - Number bytes on success, -1 on failure +ppdcSource::quotef(cups_file_t *fp, // I - File to write to + const char *format, // I - Printf-style format string + ...) // I - Additional args as needed +{ + va_list ap; // Pointer to additional arguments + int bytes; // Bytes written + char sign, // Sign of format width + size, // Size character (h, l, L) + type; // Format type character + const char *bufformat; // Start of format + int width, // Width of field + prec; // Number of characters of precision + char tformat[100]; // Temporary format string for fprintf() + char *s; // Pointer to string + int slen; // Length of string + int i; // Looping var + + + // Range check input... + if (!fp || !format) + return (-1); + + // Loop through the format string, formatting as needed... + va_start(ap, format); + + bytes = 0; + + while (*format) + { + if (*format == '%') + { + bufformat = format; + format ++; + + if (*format == '%') + { + cupsFilePutChar(fp, *format++); + bytes ++; + continue; + } + else if (strchr(" -+#\'", *format)) + sign = *format++; + else + sign = 0; + + width = 0; + while (isdigit(*format)) + width = width * 10 + *format++ - '0'; + + if (*format == '.') + { + format ++; + prec = 0; + + while (isdigit(*format)) + prec = prec * 10 + *format++ - '0'; + } + else + prec = -1; + + if (*format == 'l' && format[1] == 'l') + { + size = 'L'; + format += 2; + } + else if (*format == 'h' || *format == 'l' || *format == 'L') + size = *format++; + else + size = '\0'; + + if (!*format) + break; + + type = *format++; + + switch (type) + { + case 'E' : // Floating point formats + case 'G' : + case 'e' : + case 'f' : + case 'g' : + if ((format - bufformat + 1) > (int)sizeof(tformat)) + break; + + strncpy(tformat, bufformat, format - bufformat); + tformat[format - bufformat] = '\0'; + + bytes += cupsFilePrintf(fp, tformat, va_arg(ap, double)); + break; + + case 'B' : // Integer formats + case 'X' : + case 'b' : + case 'd' : + case 'i' : + case 'o' : + case 'u' : + case 'x' : + if ((format - bufformat + 1) > (int)sizeof(tformat)) + break; + + strncpy(tformat, bufformat, format - bufformat); + tformat[format - bufformat] = '\0'; + +# ifdef HAVE_LONG_LONG + if (size == 'L') + bytes += cupsFilePrintf(fp, tformat, va_arg(ap, long long)); + else +# endif /* HAVE_LONG_LONG */ + if (size == 'l') + bytes += cupsFilePrintf(fp, tformat, va_arg(ap, long)); + else + bytes += cupsFilePrintf(fp, tformat, va_arg(ap, int)); + break; + + case 'p' : // Pointer value + if ((format - bufformat + 1) > (int)sizeof(tformat)) + break; + + strncpy(tformat, bufformat, format - bufformat); + tformat[format - bufformat] = '\0'; + + bytes += cupsFilePrintf(fp, tformat, va_arg(ap, void *)); + break; + + case 'c' : // Character or character array + if (width <= 1) + { + bytes ++; + cupsFilePutChar(fp, va_arg(ap, int)); + } + else + { + cupsFileWrite(fp, va_arg(ap, char *), width); + bytes += width; + } + break; + + case 's' : // String + if ((s = va_arg(ap, char *)) == NULL) + s = (char *)"(nil)"; + + slen = strlen(s); + if (slen > width && prec != width) + width = slen; + + if (slen > width) + slen = width; + + if (sign != '-') + { + for (i = width - slen; i > 0; i --, bytes ++) + cupsFilePutChar(fp, ' '); + } + + for (i = slen; i > 0; i --, s ++, bytes ++) + { + if (*s == '\\' || *s == '\"') + { + cupsFilePutChar(fp, '\\'); + bytes ++; + } + + cupsFilePutChar(fp, *s); + } + + if (sign == '-') + { + for (i = width - slen; i > 0; i --, bytes ++) + cupsFilePutChar(fp, ' '); + } + break; + } + } + else + { + cupsFilePutChar(fp, *format++); + bytes ++; + } + } + + va_end(ap); + + // Return the number of characters written. + return (bytes); +} + + +// +// 'ppdcSource::read_file()' - Read a driver source file. +// + +void +ppdcSource::read_file(const char *f, // I - File to read + cups_file_t *ffp) // I - File pointer to use +{ + ppdcFile *fp = new ppdcFile(f, ffp); + scan_file(fp); + delete fp; + + if (cond_current != cond_stack) + _cupsLangPrintf(stderr, _("ppdc: Missing #endif at end of \"%s\"."), f); +} + + +// +// 'ppdcSource::scan_file()' - Scan a driver source file. +// + +void +ppdcSource::scan_file(ppdcFile *fp, // I - File to read + ppdcDriver *td, // I - Driver template + bool inc) // I - Including? +{ + ppdcDriver *d; // Current driver + ppdcGroup *g, // Current group + *mg, // Matching group + *general, // General options group + *install; // Installable options group + ppdcOption *o; // Current option + ppdcChoice *c; // Current choice + char temp[256], // Token from file... + *ptr; // Pointer into token + int isdefault; // Default option? + + + // Initialize things as needed... + if (inc && td) + { + d = td; + d->retain(); + } + else + d = new ppdcDriver(td); + + if ((general = d->find_group("General")) == NULL) + { + general = new ppdcGroup("General", NULL); + d->add_group(general); + } + + if ((install = d->find_group("InstallableOptions")) == NULL) + { + install = new ppdcGroup("InstallableOptions", "Installable Options"); + d->add_group(install); + } + + // Loop until EOF or } + o = 0; + g = general; + + while (get_token(fp, temp, sizeof(temp))) + { + if (temp[0] == '*') + { + // Mark the next choice as the default + isdefault = 1; + + for (ptr = temp; ptr[1]; ptr ++) + *ptr = ptr[1]; + + *ptr = '\0'; + } + else + { + // Don't mark the next choice as the default + isdefault = 0; + } + + if (!_cups_strcasecmp(temp, "}")) + { + // Close this one out... + break; + } + else if (!_cups_strcasecmp(temp, "{")) + { + // Open a new child... + scan_file(fp, d); + } + else if (!_cups_strcasecmp(temp, "#if")) + { + if ((cond_current - cond_stack) >= 100) + { + _cupsLangPrintf(stderr, + _("ppdc: Too many nested #if's on line %d of %s."), + fp->line, fp->filename); + break; + } + + cond_current ++; + if (get_integer(fp) > 0) + *cond_current = PPDC_COND_SATISFIED; + else + { + *cond_current = PPDC_COND_SKIP; + cond_state |= PPDC_COND_SKIP; + } + } + else if (!_cups_strcasecmp(temp, "#elif")) + { + if (cond_current == cond_stack) + { + _cupsLangPrintf(stderr, _("ppdc: Missing #if on line %d of %s."), + fp->line, fp->filename); + break; + } + + if (*cond_current & PPDC_COND_SATISFIED) + { + get_integer(fp); + *cond_current |= PPDC_COND_SKIP; + } + else if (get_integer(fp) > 0) + { + *cond_current |= PPDC_COND_SATISFIED; + *cond_current &= ~PPDC_COND_SKIP; + } + else + *cond_current |= PPDC_COND_SKIP; + + // Update the current state + int *cond_temp = cond_current; // Temporary stack pointer + + cond_state = PPDC_COND_NORMAL; + while (cond_temp > cond_stack) + if (*cond_temp & PPDC_COND_SKIP) + { + cond_state = PPDC_COND_SKIP; + break; + } + else + cond_temp --; + } + else if (!_cups_strcasecmp(temp, "#else")) + { + if (cond_current == cond_stack) + { + _cupsLangPrintf(stderr, _("ppdc: Missing #if on line %d of %s."), + fp->line, fp->filename); + break; + } + + if (*cond_current & PPDC_COND_SATISFIED) + *cond_current |= PPDC_COND_SKIP; + else + { + *cond_current |= PPDC_COND_SATISFIED; + *cond_current &= ~PPDC_COND_SKIP; + } + + // Update the current state + int *cond_temp = cond_current; // Temporary stack pointer + + cond_state = PPDC_COND_NORMAL; + while (cond_temp > cond_stack) + if (*cond_temp & PPDC_COND_SKIP) + { + cond_state = PPDC_COND_SKIP; + break; + } + else + cond_temp --; + } + else if (!_cups_strcasecmp(temp, "#endif")) + { + if (cond_current == cond_stack) + { + _cupsLangPrintf(stderr, _("ppdc: Missing #if on line %d of %s."), + fp->line, fp->filename); + break; + } + + cond_current --; + + // Update the current state + int *cond_temp = cond_current; // Temporary stack pointer + + cond_state = PPDC_COND_NORMAL; + while (cond_temp > cond_stack) + if (*cond_temp & PPDC_COND_SKIP) + { + cond_state = PPDC_COND_SKIP; + break; + } + else + cond_temp --; + } + else if (!_cups_strcasecmp(temp, "#define")) + { + // Get the variable... + get_variable(fp); + } + else if (!_cups_strcasecmp(temp, "#include")) + { + // #include filename + char basedir[1024], // Base directory + *baseptr, // Pointer into directory + inctemp[1024], // Initial filename + incname[1024]; // Include filename + ppdcFile *incfile; // Include file + int *old_current = cond_current; + // Previous current stack + + + // Get the include name... + if (!get_token(fp, inctemp, sizeof(inctemp))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected include filename on line %d of " + "%s."), fp->line, fp->filename); + break; + } + + if (cond_state) + continue; + + // Figure out the current directory... + strlcpy(basedir, fp->filename, sizeof(basedir)); + + if ((baseptr = strrchr(basedir, '/')) != NULL) + *baseptr = '\0'; + else + strcpy(basedir, "."); + + // Find the include file... + if (find_include(inctemp, basedir, incname, sizeof(incname))) + { + // Open the include file, scan it, and then close it... + incfile = new ppdcFile(incname); + scan_file(incfile, d, true); + delete incfile; + + if (cond_current != old_current) + _cupsLangPrintf(stderr, _("ppdc: Missing #endif at end of \"%s\"."), + incname); + } + else + { + // Can't find it! + _cupsLangPrintf(stderr, + _("ppdc: Unable to find include file \"%s\" on line %d " + "of %s."), inctemp, fp->line, fp->filename); + break; + } + } + else if (!_cups_strcasecmp(temp, "#media")) + { + ppdcMediaSize *m; // Media size + + + // Get a media size... + m = get_size(fp); + if (m) + { + if (cond_state) + m->release(); + else + sizes->add(m); + } + } + else if (!_cups_strcasecmp(temp, "#po")) + { + ppdcCatalog *cat; // Message catalog + + + // Get a message catalog... + cat = get_po(fp); + if (cat) + { + if (cond_state) + cat->release(); + else + po_files->add(cat); + } + } + else if (!_cups_strcasecmp(temp, "Attribute") || + !_cups_strcasecmp(temp, "LocAttribute")) + { + ppdcAttr *a; // Attribute + + + // Get an attribute... + a = get_attr(fp, !_cups_strcasecmp(temp, "LocAttribute")); + if (a) + { + if (cond_state) + a->release(); + else + d->add_attr(a); + } + } + else if (!_cups_strcasecmp(temp, "Choice")) + { + // Get a choice... + c = get_choice(fp); + if (!c) + break; + + if (cond_state) + { + c->release(); + continue; + } + + // Add it to the current option... + if (!o) + { + _cupsLangPrintf(stderr, + _("ppdc: Choice found on line %d of %s with no " + "Option."), fp->line, fp->filename); + break; + } + + o->add_choice(c); + + if (isdefault) + o->set_defchoice(c); + } + else if (!_cups_strcasecmp(temp, "ColorDevice")) + { + // ColorDevice boolean + if (cond_state) + get_boolean(fp); + else + d->color_device = get_boolean(fp); + } + else if (!_cups_strcasecmp(temp, "ColorModel")) + { + // Get the color model + c = get_color_model(fp); + if (!c) + continue; + + if (cond_state) + { + c->release(); + continue; + } + + // Add the choice to the ColorModel option... + if ((o = d->find_option("ColorModel")) == NULL) + { + // Create the ColorModel option... + o = new ppdcOption(PPDC_PICKONE, "ColorModel", "Color Mode", PPDC_SECTION_ANY, 10.0f); + g = general; + g->add_option(o); + } + + o->add_choice(c); + + if (isdefault) + o->set_defchoice(c); + + o = NULL; + } + else if (!_cups_strcasecmp(temp, "ColorProfile")) + { + ppdcProfile *p; // Color profile + + + // Get the color profile... + p = get_color_profile(fp); + + if (p) + { + if (cond_state) + p->release(); + else + d->profiles->add(p); + } + } + else if (!_cups_strcasecmp(temp, "Copyright")) + { + // Copyright string + char copytemp[8192], // Copyright string + *copyptr, // Pointer into string + *copyend; // Pointer to end of string + + + // Get the copyright string... + if (!get_token(fp, copytemp, sizeof(temp))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected string after Copyright on line %d " + "of %s."), fp->line, fp->filename); + break; + } + + if (cond_state) + continue; + + // Break it up into individual lines... + for (copyptr = copytemp; copyptr; copyptr = copyend) + { + if ((copyend = strchr(copyptr, '\n')) != NULL) + *copyend++ = '\0'; + + d->copyright->add(new ppdcString(copyptr)); + } + } + else if (!_cups_strcasecmp(temp, "CustomMedia")) + { + ppdcMediaSize *m; // Media size + + + // Get a custom media size... + m = get_custom_size(fp); + + if (cond_state) + { + m->release(); + continue; + } + + if (m) + d->sizes->add(m); + + if (isdefault) + d->set_default_size(m); + } + else if (!_cups_strcasecmp(temp, "Cutter")) + { + // Cutter boolean + int have_cutter; // Have a paper cutter? + + + have_cutter = get_boolean(fp); + if (have_cutter <= 0 || cond_state) + continue; + + if ((o = d->find_option("CutMedia")) == NULL) + { + o = new ppdcOption(PPDC_BOOLEAN, "CutMedia", "Cut Media", PPDC_SECTION_ANY, 10.0f); + + g = general; + g->add_option(o); + + c = new ppdcChoice("False", NULL, "<>setpagedevice"); + o->add_choice(c); + o->set_defchoice(c); + + c = new ppdcChoice("True", NULL, "<>setpagedevice"); + o->add_choice(c); + } + + o = NULL; + } + else if (!_cups_strcasecmp(temp, "Darkness")) + { + // Get the darkness choice... + c = get_generic(fp, "Darkness", NULL, "cupsCompression"); + if (!c) + continue; + + if (cond_state) + { + c->release(); + continue; + } + + // Add the choice to the cupsDarkness option... + if ((o = d->find_option_group("cupsDarkness", &mg)) == NULL) + { + // Create the cupsDarkness option... + o = new ppdcOption(PPDC_PICKONE, "cupsDarkness", "Darkness", PPDC_SECTION_ANY, 10.0f); + g = general; + g->add_option(o); + } + else if (mg != general) + { + _cupsLangPrintf(stderr, + _("ppdc: Option %s defined in two different groups on " + "line %d of %s."), "cupsDarkness", fp->line, + fp->filename); + c->release(); + continue; + } + + o->add_choice(c); + + if (isdefault) + o->set_defchoice(c); + + o = NULL; + } + else if (!_cups_strcasecmp(temp, "DriverType")) + { + int i; // Looping var + + + // DriverType keyword + if (!get_token(fp, temp, sizeof(temp))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected driver type keyword following " + "DriverType on line %d of %s."), + fp->line, fp->filename); + continue; + } + + if (cond_state) + continue; + + for (i = 0; i < (int)(sizeof(driver_types) / sizeof(driver_types[0])); i ++) + if (!_cups_strcasecmp(temp, driver_types[i])) + break; + + if (i < (int)(sizeof(driver_types) / sizeof(driver_types[0]))) + d->type = (ppdcDrvType)i; + else if (!_cups_strcasecmp(temp, "dymo")) + d->type = PPDC_DRIVER_LABEL; + else + _cupsLangPrintf(stderr, + _("ppdc: Unknown driver type %s on line %d of %s."), + temp, fp->line, fp->filename); + } + else if (!_cups_strcasecmp(temp, "Duplex")) + get_duplex(fp, d); + else if (!_cups_strcasecmp(temp, "Filter")) + { + ppdcFilter *f; // Filter + + + // Get the filter value... + f = get_filter(fp); + if (f) + { + if (cond_state) + f->release(); + else + d->filters->add(f); + } + } + else if (!_cups_strcasecmp(temp, "Finishing")) + { + // Get the finishing choice... + c = get_generic(fp, "Finishing", "OutputType", NULL); + if (!c) + continue; + + if (cond_state) + { + c->release(); + continue; + } + + // Add the choice to the cupsFinishing option... + if ((o = d->find_option_group("cupsFinishing", &mg)) == NULL) + { + // Create the cupsFinishing option... + o = new ppdcOption(PPDC_PICKONE, "cupsFinishing", "Finishing", PPDC_SECTION_ANY, 10.0f); + g = general; + g->add_option(o); + } + else if (mg != general) + { + _cupsLangPrintf(stderr, + _("ppdc: Option %s defined in two different groups on " + "line %d of %s."), "cupsFinishing", fp->line, + fp->filename); + c->release(); + continue; + } + + o->add_choice(c); + + if (isdefault) + o->set_defchoice(c); + + o = NULL; + } + else if (!_cups_strcasecmp(temp, "Font") || + !_cups_strcasecmp(temp, "#font")) + { + ppdcFont *f; // Font + + + // Get a font... + f = get_font(fp); + if (f) + { + if (cond_state) + f->release(); + else + { + if (!_cups_strcasecmp(temp, "#font")) + base_fonts->add(f); + else + d->add_font(f); + + if (isdefault) + d->set_default_font(f); + } + } + } + else if (!_cups_strcasecmp(temp, "Group")) + { + // Get a group... + ppdcGroup *tempg = get_group(fp, d); + + if (!tempg) + break; + + if (cond_state) + { + if (!d->find_group(tempg->name->value)) + tempg->release(); + } + else + { + if (!d->find_group(tempg->name->value)) + d->add_group(tempg); + + g = tempg; + } + } + else if (!_cups_strcasecmp(temp, "HWMargins")) + { + // HWMargins left bottom right top + d->left_margin = get_measurement(fp); + d->bottom_margin = get_measurement(fp); + d->right_margin = get_measurement(fp); + d->top_margin = get_measurement(fp); + } + else if (!_cups_strcasecmp(temp, "InputSlot")) + { + // Get the input slot choice... + c = get_generic(fp, "InputSlot", NULL, "MediaPosition"); + if (!c) + continue; + + if (cond_state) + { + c->release(); + continue; + } + + // Add the choice to the InputSlot option... + + if ((o = d->find_option_group("InputSlot", &mg)) == NULL) + { + // Create the InputSlot option... + o = new ppdcOption(PPDC_PICKONE, "InputSlot", "Media Source", + PPDC_SECTION_ANY, 10.0f); + g = general; + g->add_option(o); + } + else if (mg != general) + { + _cupsLangPrintf(stderr, + _("ppdc: Option %s defined in two different groups on " + "line %d of %s."), "InputSlot", fp->line, + fp->filename); + c->release(); + continue; + } + + o->add_choice(c); + + if (isdefault) + o->set_defchoice(c); + + o = NULL; + } + else if (!_cups_strcasecmp(temp, "Installable")) + { + // Get the installable option... + o = get_installable(fp); + + // Add it as needed... + if (o) + { + if (cond_state) + o->release(); + else + install->add_option(o); + + o = NULL; + } + } + else if (!_cups_strcasecmp(temp, "ManualCopies")) + { + // ManualCopies boolean + if (cond_state) + get_boolean(fp); + else + d->manual_copies = get_boolean(fp); + } + else if (!_cups_strcasecmp(temp, "Manufacturer")) + { + // Manufacturer name + char name[256]; // Model name string + + + if (!get_token(fp, name, sizeof(name))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected name after Manufacturer on line %d " + "of %s."), fp->line, fp->filename); + break; + } + + if (!cond_state) + d->set_manufacturer(name); + } + else if (!_cups_strcasecmp(temp, "MaxSize")) + { + // MaxSize width length + if (cond_state) + { + get_measurement(fp); + get_measurement(fp); + } + else + { + d->max_width = get_measurement(fp); + d->max_length = get_measurement(fp); + } + } + else if (!_cups_strcasecmp(temp, "MediaSize")) + { + // MediaSize keyword + char name[41]; // Media size name + ppdcMediaSize *m, // Matching media size... + *dm; // Driver media size... + + + if (get_token(fp, name, sizeof(name)) == NULL) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected name after MediaSize on line %d of " + "%s."), fp->line, fp->filename); + break; + } + + if (cond_state) + continue; + + m = find_size(name); + + if (!m) + { + _cupsLangPrintf(stderr, + _("ppdc: Unknown media size \"%s\" on line %d of " + "%s."), name, fp->line, fp->filename); + break; + } + + // Add this size to the driver... + dm = new ppdcMediaSize(m->name->value, m->text->value, + m->width, m->length, d->left_margin, + d->bottom_margin, d->right_margin, + d->top_margin); + d->sizes->add(dm); + + if (isdefault) + d->set_default_size(dm); + } + else if (!_cups_strcasecmp(temp, "MediaType")) + { + // Get the media type choice... + c = get_generic(fp, "MediaType", "MediaType", "cupsMediaType"); + if (!c) + continue; + + if (cond_state) + { + c->release(); + continue; + } + + // Add the choice to the MediaType option... + if ((o = d->find_option_group("MediaType", &mg)) == NULL) + { + // Create the MediaType option... + o = new ppdcOption(PPDC_PICKONE, "MediaType", "Media Type", + PPDC_SECTION_ANY, 10.0f); + g = general; + g->add_option(o); + } + else if (mg != general) + { + _cupsLangPrintf(stderr, + _("ppdc: Option %s defined in two different groups on " + "line %d of %s."), "MediaType", fp->line, + fp->filename); + c->release(); + continue; + } + + o->add_choice(c); + + if (isdefault) + o->set_defchoice(c); + + o = NULL; + } + else if (!_cups_strcasecmp(temp, "MinSize")) + { + // MinSize width length + if (cond_state) + { + get_measurement(fp); + get_measurement(fp); + } + else + { + d->min_width = get_measurement(fp); + d->min_length = get_measurement(fp); + } + } + else if (!_cups_strcasecmp(temp, "ModelName")) + { + // ModelName name + char name[256]; // Model name string + + + if (!get_token(fp, name, sizeof(name))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected name after ModelName on line %d of " + "%s."), fp->line, fp->filename); + break; + } + + if (!cond_state) + d->set_model_name(name); + } + else if (!_cups_strcasecmp(temp, "ModelNumber")) + { + // ModelNumber number + if (cond_state) + get_integer(fp); + else + d->model_number = get_integer(fp); + } + else if (!_cups_strcasecmp(temp, "Option")) + { + // Get an option... + ppdcOption *tempo = get_option(fp, d, g); + + if (!tempo) + break; + + if (cond_state) + { + if (!g->find_option(tempo->name->value)) + tempo->release(); + } + else + { + if (!g->find_option(tempo->name->value)) + g->add_option(tempo); + + o = tempo; + } + } + else if (!_cups_strcasecmp(temp, "FileName")) + { + // FileName name + char name[256]; // Filename string + + + if (!get_token(fp, name, sizeof(name))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected name after FileName on line %d of " + "%s."), fp->line, fp->filename); + break; + } + + if (!cond_state) + d->set_file_name(name); + } + else if (!_cups_strcasecmp(temp, "PCFileName")) + { + // PCFileName name + char name[256]; // PC filename string + + + if (!get_token(fp, name, sizeof(name))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected name after PCFileName on line %d of " + "%s."), fp->line, fp->filename); + break; + } + + if (!cond_state) + d->set_pc_file_name(name); + } + else if (!_cups_strcasecmp(temp, "Resolution")) + { + // Get the resolution choice... + c = get_resolution(fp); + if (!c) + continue; + + if (cond_state) + { + c->release(); + continue; + } + + // Add the choice to the Resolution option... + if ((o = d->find_option_group("Resolution", &mg)) == NULL) + { + // Create the Resolution option... + o = new ppdcOption(PPDC_PICKONE, "Resolution", NULL, PPDC_SECTION_ANY, + 10.0f); + g = general; + g->add_option(o); + } + else if (mg != general) + { + _cupsLangPrintf(stderr, + _("ppdc: Option %s defined in two different groups on " + "line %d of %s."), "Resolution", fp->line, + fp->filename); + c->release(); + continue; + } + + o->add_choice(c); + + if (isdefault) + o->set_defchoice(c); + + o = NULL; + } + else if (!_cups_strcasecmp(temp, "SimpleColorProfile")) + { + ppdcProfile *p; // Color profile + + + // Get the color profile... + p = get_simple_profile(fp); + + if (p) + { + if (cond_state) + p->release(); + else + d->profiles->add(p); + } + } + else if (!_cups_strcasecmp(temp, "Throughput")) + { + // Throughput number + if (cond_state) + get_integer(fp); + else + d->throughput = get_integer(fp); + } + else if (!_cups_strcasecmp(temp, "UIConstraints")) + { + ppdcConstraint *con; // Constraint + + + con = get_constraint(fp); + + if (con) + { + if (cond_state) + con->release(); + else + d->constraints->add(con); + } + } + else if (!_cups_strcasecmp(temp, "VariablePaperSize")) + { + // VariablePaperSize boolean + if (cond_state) + get_boolean(fp); + else + d->variable_paper_size = get_boolean(fp); + } + else if (!_cups_strcasecmp(temp, "Version")) + { + // Version string + char name[256]; // Model name string + + + if (!get_token(fp, name, sizeof(name))) + { + _cupsLangPrintf(stderr, + _("ppdc: Expected string after Version on line %d of " + "%s."), fp->line, fp->filename); + break; + } + + if (!cond_state) + d->set_version(name); + } + else + { + _cupsLangPrintf(stderr, + _("ppdc: Unknown token \"%s\" seen on line %d of %s."), + temp, fp->line, fp->filename); + break; + } + } + + // Done processing this block, is there anything to save? + if (!inc) + { + if (!d->pc_file_name || !d->model_name || !d->manufacturer || !d->version || + !d->sizes->count) + { + // Nothing to save... + d->release(); + } + else + { + // Got a driver, save it... + drivers->add(d); + } + } + else if (inc && td) + td->release(); +} + + +// +// 'ppdcSource::set_variable()' - Set a variable. +// + +ppdcVariable * // O - Variable +ppdcSource::set_variable( + const char *name, // I - Name + const char *value) // I - Value +{ + ppdcVariable *v; // Variable + + + // See if the variable exists already... + v = find_variable(name); + if (v) + { + // Change the variable value... + v->set_value(value); + } + else + { + // Create a new variable and add it... + v = new ppdcVariable(name, value); + vars->add(v); + } + + return (v); +} + + +// +// 'ppdcSource::write_file()' - Write the current source data to a file. +// + +int // O - 0 on success, -1 on error +ppdcSource::write_file(const char *f) // I - File to write +{ + cups_file_t *fp; // Output file + char bckname[1024]; // Backup file + ppdcDriver *d; // Current driver + ppdcString *st; // Current string + ppdcAttr *a; // Current attribute + ppdcConstraint *co; // Current constraint + ppdcFilter *fi; // Current filter + ppdcFont *fo; // Current font + ppdcGroup *g; // Current group + ppdcOption *o; // Current option + ppdcChoice *ch; // Current choice + ppdcProfile *p; // Current color profile + ppdcMediaSize *si; // Current media size + float left, // Current left margin + bottom, // Current bottom margin + right, // Current right margin + top; // Current top margin + int dtused[PPDC_DRIVER_MAX];// Driver type usage... + + + // Rename the current file, if any, to .bck... + snprintf(bckname, sizeof(bckname), "%s.bck", f); + rename(f, bckname); + + // Open the output file... + fp = cupsFileOpen(f, "w"); + + if (!fp) + { + // Can't create file; restore backup and return... + rename(bckname, f); + return (-1); + } + + cupsFilePuts(fp, "// CUPS PPD Compiler " CUPS_SVERSION "\n\n"); + + // Include standard files... + cupsFilePuts(fp, "// Include necessary files...\n"); + cupsFilePuts(fp, "#include \n"); + cupsFilePuts(fp, "#include \n"); + + memset(dtused, 0, sizeof(dtused)); + + for (d = (ppdcDriver *)drivers->first(); d; d = (ppdcDriver *)drivers->next()) + if (d->type > PPDC_DRIVER_PS && !dtused[d->type]) + { + cupsFilePrintf(fp, "#include <%s.h>\n", driver_types[d->type]); + dtused[d->type] = 1; + } + + // Output each driver... + for (d = (ppdcDriver *)drivers->first(); d; d = (ppdcDriver *)drivers->next()) + { + // Start the driver... + cupsFilePrintf(fp, "\n// %s %s\n", d->manufacturer->value, + d->model_name->value); + cupsFilePuts(fp, "{\n"); + + // Write the copyright stings... + for (st = (ppdcString *)d->copyright->first(); + st; + st = (ppdcString *)d->copyright->next()) + quotef(fp, " Copyright \"%s\"\n", st->value); + + // Write other strings and values... + if (d->manufacturer && d->manufacturer->value) + quotef(fp, " Manufacturer \"%s\"\n", d->manufacturer->value); + if (d->model_name->value) + quotef(fp, " ModelName \"%s\"\n", d->model_name->value); + if (d->file_name && d->file_name->value) + quotef(fp, " FileName \"%s\"\n", d->file_name->value); + if (d->pc_file_name && d->pc_file_name->value) + quotef(fp, " PCFileName \"%s\"\n", d->pc_file_name->value); + if (d->version && d->version->value) + quotef(fp, " Version \"%s\"\n", d->version->value); + + cupsFilePrintf(fp, " DriverType %s\n", driver_types[d->type]); + + if (d->model_number) + { + switch (d->type) + { + case PPDC_DRIVER_LABEL : + cupsFilePuts(fp, " ModelNumber "); + + switch (d->model_number) + { + case DYMO_3x0 : + cupsFilePuts(fp, "$DYMO_3x0\n"); + break; + + case ZEBRA_EPL_LINE : + cupsFilePuts(fp, "$ZEBRA_EPL_LINE\n"); + break; + + case ZEBRA_EPL_PAGE : + cupsFilePuts(fp, "$ZEBRA_EPL_PAGE\n"); + break; + + case ZEBRA_ZPL : + cupsFilePuts(fp, "$ZEBRA_ZPL\n"); + break; + + case ZEBRA_CPCL : + cupsFilePuts(fp, "$ZEBRA_CPCL\n"); + break; + + case INTELLITECH_PCL : + cupsFilePuts(fp, "$INTELLITECH_PCL\n"); + break; + + default : + cupsFilePrintf(fp, "%d\n", d->model_number); + break; + } + break; + + case PPDC_DRIVER_EPSON : + cupsFilePuts(fp, " ModelNumber "); + + switch (d->model_number) + { + case EPSON_9PIN : + cupsFilePuts(fp, "$EPSON_9PIN\n"); + break; + + case EPSON_24PIN : + cupsFilePuts(fp, "$EPSON_24PIN\n"); + break; + + case EPSON_COLOR : + cupsFilePuts(fp, "$EPSON_COLOR\n"); + break; + + case EPSON_PHOTO : + cupsFilePuts(fp, "$EPSON_PHOTO\n"); + break; + + case EPSON_ICOLOR : + cupsFilePuts(fp, "$EPSON_ICOLOR\n"); + break; + + case EPSON_IPHOTO : + cupsFilePuts(fp, "$EPSON_IPHOTO\n"); + break; + + default : + cupsFilePrintf(fp, "%d\n", d->model_number); + break; + } + break; + + case PPDC_DRIVER_HP : + cupsFilePuts(fp, " ModelNumber "); + switch (d->model_number) + { + case HP_LASERJET : + cupsFilePuts(fp, "$HP_LASERJET\n"); + break; + + case HP_DESKJET : + cupsFilePuts(fp, "$HP_DESKJET\n"); + break; + + case HP_DESKJET2 : + cupsFilePuts(fp, "$HP_DESKJET2\n"); + break; + + default : + cupsFilePrintf(fp, "%d\n", d->model_number); + break; + } + + cupsFilePuts(fp, ")\n"); + break; + + default : + cupsFilePrintf(fp, " ModelNumber %d\n", d->model_number); + break; + } + } + + if (d->manual_copies) + cupsFilePuts(fp, " ManualCopies Yes\n"); + + if (d->color_device) + cupsFilePuts(fp, " ColorDevice Yes\n"); + + if (d->throughput) + cupsFilePrintf(fp, " Throughput %d\n", d->throughput); + + // Output all of the attributes... + for (a = (ppdcAttr *)d->attrs->first(); + a; + a = (ppdcAttr *)d->attrs->next()) + if (a->text->value && a->text->value[0]) + quotef(fp, " Attribute \"%s\" \"%s/%s\" \"%s\"\n", + a->name->value, a->selector->value ? a->selector->value : "", + a->text->value, a->value->value ? a->value->value : ""); + else + quotef(fp, " Attribute \"%s\" \"%s\" \"%s\"\n", + a->name->value, a->selector->value ? a->selector->value : "", + a->value->value ? a->value->value : ""); + + // Output all of the constraints... + for (co = (ppdcConstraint *)d->constraints->first(); + co; + co = (ppdcConstraint *)d->constraints->next()) + { + if (co->option1->value[0] == '*') + cupsFilePrintf(fp, " UIConstraints \"%s %s", co->option1->value, + co->choice1->value ? co->choice1->value : ""); + else + cupsFilePrintf(fp, " UIConstraints \"*%s %s", co->option1->value, + co->choice1->value ? co->choice1->value : ""); + + if (co->option2->value[0] == '*') + cupsFilePrintf(fp, " %s %s\"\n", co->option2->value, + co->choice2->value ? co->choice2->value : ""); + else + cupsFilePrintf(fp, " *%s %s\"\n", co->option2->value, + co->choice2->value ? co->choice2->value : ""); + } + + // Output all of the filters... + for (fi = (ppdcFilter *)d->filters->first(); + fi; + fi = (ppdcFilter *)d->filters->next()) + cupsFilePrintf(fp, " Filter \"%s %d %s\"\n", + fi->mime_type->value, fi->cost, fi->program->value); + + // Output all of the fonts... + for (fo = (ppdcFont *)d->fonts->first(); + fo; + fo = (ppdcFont *)d->fonts->next()) + if (!strcmp(fo->name->value, "*")) + cupsFilePuts(fp, " Font *\n"); + else + cupsFilePrintf(fp, " Font \"%s\" \"%s\" \"%s\" \"%s\" %s\n", + fo->name->value, fo->encoding->value, + fo->version->value, fo->charset->value, + fo->status == PPDC_FONT_ROM ? "ROM" : "Disk"); + + // Output all options... + for (g = (ppdcGroup *)d->groups->first(); + g; + g = (ppdcGroup *)d->groups->next()) + { + if (g->options->count == 0) + continue; + + if (g->text->value && g->text->value[0]) + quotef(fp, " Group \"%s/%s\"\n", g->name->value, g->text->value); + else + cupsFilePrintf(fp, " Group \"%s\"\n", g->name->value); + + for (o = (ppdcOption *)g->options->first(); + o; + o = (ppdcOption *)g->options->next()) + { + if (o->choices->count == 0) + continue; + + if (o->text->value && o->text->value[0]) + quotef(fp, " Option \"%s/%s\"", o->name->value, o->text->value); + else + cupsFilePrintf(fp, " Option \"%s\"", o->name->value); + + cupsFilePrintf(fp, " %s %s %.1f\n", + o->type == PPDC_BOOLEAN ? "Boolean" : + o->type == PPDC_PICKONE ? "PickOne" : "PickMany", + o->section == PPDC_SECTION_ANY ? "AnySetup" : + o->section == PPDC_SECTION_DOCUMENT ? "DocumentSetup" : + o->section == PPDC_SECTION_EXIT ? "ExitServer" : + o->section == PPDC_SECTION_JCL ? "JCLSetup" : + o->section == PPDC_SECTION_PAGE ? "PageSetup" : + "Prolog", + o->order); + + for (ch = (ppdcChoice *)o->choices->first(); + ch; + ch = (ppdcChoice *)o->choices->next()) + { + if (ch->text->value && ch->text->value[0]) + quotef(fp, " %sChoice \"%s/%s\" \"%s\"\n", + o->defchoice == ch->name ? "*" : "", + ch->name->value, ch->text->value, + ch->code->value ? ch->code->value : ""); + else + quotef(fp, " %sChoice \"%s\" \"%s\"\n", + o->defchoice == ch->name ? "*" : "", + ch->name->value, + ch->code->value ? ch->code->value : ""); + } + } + } + + // Output all of the color profiles... + for (p = (ppdcProfile *)d->profiles->first(); + p; + p = (ppdcProfile *)d->profiles->next()) + cupsFilePrintf(fp, " ColorProfile \"%s/%s\" %.3f %.3f " + "%.3f %.3f %.3f %.3f %.3f %.3f %.3f %.3f %.3f\n", + p->resolution->value, p->media_type->value, + p->density, p->gamma, + p->profile[0], p->profile[1], p->profile[2], + p->profile[3], p->profile[4], p->profile[5], + p->profile[6], p->profile[7], p->profile[8]); + + // Output all of the media sizes... + left = 0.0; + bottom = 0.0; + right = 0.0; + top = 0.0; + + for (si = (ppdcMediaSize *)d->sizes->first(); + si; + si = (ppdcMediaSize *)d->sizes->next()) + if (si->size_code->value && si->region_code->value) + { + // Output a custom media size... + quotef(fp, " %sCustomMedia \"%s/%s\" %.2f %.2f %.2f %.2f %.2f %.2f \"%s\" \"%s\"\n", + si->name == d->default_size ? "*" : "", si->name->value, + si->text->value, si->width, si->length, si->left, si->bottom, + si->right, si->top, si->size_code->value, + si->region_code->value); + } + else + { + // Output a standard media size... + if (fabs(left - si->left) > 0.1 || + fabs(bottom - si->bottom) > 0.1 || + fabs(right - si->right) > 0.1 || + fabs(top - si->top) > 0.1) + { + cupsFilePrintf(fp, " HWMargins %.2f %.2f %.2f %.2f\n", + si->left, si->bottom, si->right, si->top); + + left = si->left; + bottom = si->bottom; + right = si->right; + top = si->top; + } + + cupsFilePrintf(fp, " %sMediaSize %s\n", + si->name == d->default_size ? "*" : "", + si->name->value); + } + + if (d->variable_paper_size) + { + cupsFilePuts(fp, " VariablePaperSize Yes\n"); + + if (fabs(left - d->left_margin) > 0.1 || + fabs(bottom - d->bottom_margin) > 0.1 || + fabs(right - d->right_margin) > 0.1 || + fabs(top - d->top_margin) > 0.1) + { + cupsFilePrintf(fp, " HWMargins %.2f %.2f %.2f %.2f\n", + d->left_margin, d->bottom_margin, d->right_margin, + d->top_margin); + } + + cupsFilePrintf(fp, " MinSize %.2f %.2f\n", d->min_width, d->min_length); + cupsFilePrintf(fp, " MaxSize %.2f %.2f\n", d->max_width, d->max_length); + } + + // End the driver... + cupsFilePuts(fp, "}\n"); + } + + // Close the file and return... + cupsFileClose(fp); + + return (0); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-string.cxx b/ppdc/ppdc-string.cxx new file mode 100644 index 0000000000..caf8f8015d --- /dev/null +++ b/ppdc/ppdc-string.cxx @@ -0,0 +1,62 @@ +// +// "$Id$" +// +// Shared string class for the CUPS PPD Compiler. +// +// Copyright 2007-2009 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcString::ppdcString() - Create a shared string. +// ppdcString::~ppdcString() - Destroy a shared string. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// 'ppdcString::ppdcString()' - Create a shared string. +// + +ppdcString::ppdcString(const char *v) // I - String + : ppdcShared() +{ + PPDC_NEWVAL(v); + + if (v) + { + value = new char[strlen(v) + 1]; + strcpy(value, v); + } + else + value = 0; +} + + +// +// 'ppdcString::~ppdcString()' - Destroy a shared string. +// + +ppdcString::~ppdcString() +{ + PPDC_DELETEVAL(value); + + if (value) + delete[] value; +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc-variable.cxx b/ppdc/ppdc-variable.cxx new file mode 100644 index 0000000000..22a175ad44 --- /dev/null +++ b/ppdc/ppdc-variable.cxx @@ -0,0 +1,71 @@ +// +// "$Id$" +// +// Variable class for the CUPS PPD Compiler. +// +// Copyright 2007-2009 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// ppdcVariable::ppdcVariable() - Create a variable. +// ppdcVariable::~ppdcVariable() - Destroy a variable. +// ppdcVariable::set_value() - Set the value of a variable. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// 'ppdcVariable::ppdcVariable()' - Create a variable. +// + +ppdcVariable::ppdcVariable(const char *n, // I - Name of variable + const char *v) // I - Value of variable + : ppdcShared() +{ + PPDC_NEW; + + name = new ppdcString(n); + value = new ppdcString(v); +} + + +// +// 'ppdcVariable::~ppdcVariable()' - Destroy a variable. +// + +ppdcVariable::~ppdcVariable() +{ + PPDC_DELETE; + + name->release(); + value->release(); +} + + +// +// 'ppdcVariable::set_value()' - Set the value of a variable. +// + +void +ppdcVariable::set_value(const char *v) +{ + value->release(); + value = new ppdcString(v); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc.cxx b/ppdc/ppdc.cxx new file mode 100644 index 0000000000..d8d128db8b --- /dev/null +++ b/ppdc/ppdc.cxx @@ -0,0 +1,469 @@ +// +// "$Id$" +// +// PPD file compiler main entry for the CUPS PPD Compiler. +// +// Copyright 2007-2011 by Apple Inc. +// Copyright 2002-2007 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// main() - Main entry for the PPD compiler. +// usage() - Show usage and exit. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" +#include +#include +#include + + +// +// Local functions... +// + +static void usage(void); + + +// +// 'main()' - Main entry for the PPD compiler. +// + +int // O - Exit status +main(int argc, // I - Number of command-line arguments + char *argv[]) // I - Command-line arguments +{ + int i, j; // Looping vars + ppdcCatalog *catalog; // Message catalog + const char *outdir; // Output directory + ppdcSource *src; // PPD source file data + ppdcDriver *d; // Current driver + cups_file_t *fp; // PPD file + char *opt, // Current option + *value, // Value in option + *outname, // Output filename + make_model[1024], + // Make and model + pcfilename[1024], + // Lowercase pcfilename + filename[1024]; // PPD filename + int comp, // Compress + do_test, // Test PPD files + single_language,// Generate single-language files + use_model_name, // Use ModelName for filename + verbose; // Verbosity + ppdcLineEnding le; // Line ending to use + ppdcArray *locales; // List of locales + cups_array_t *filenames; // List of generated filenames + + + _cupsSetLocale(argv); + + // Scan the command-line... + catalog = NULL; + comp = 0; + do_test = 0; + le = PPDC_LFONLY; + locales = NULL; + outdir = "ppd"; + single_language = 0; + src = new ppdcSource(); + use_model_name = 0; + verbose = 0; + filenames = cupsArrayNew((cups_array_func_t)_cups_strcasecmp, NULL); + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-') + { + for (opt = argv[i] + 1; *opt; opt ++) + switch (*opt) + { + case 'D' : // Define variable + i ++; + if (i >= argc) + usage(); + + if ((value = strchr(argv[i], '=')) != NULL) + { + *value++ = '\0'; + + src->set_variable(argv[i], value); + } + else + src->set_variable(argv[i], "1"); + break; + + case 'I' : // Include directory... + i ++; + if (i >= argc) + usage(); + + if (verbose > 1) + _cupsLangPrintf(stdout, + _("ppdc: Adding include directory \"%s\"."), + argv[i]); + + ppdcSource::add_include(argv[i]); + break; + + case 'c' : // Message catalog... + i ++; + if (i >= argc) + usage(); + + if (verbose > 1) + _cupsLangPrintf(stdout, + _("ppdc: Loading messages from \"%s\"."), + argv[i]); + + if (!catalog) + catalog = new ppdcCatalog("en"); + + if (catalog->load_messages(argv[i])) + { + _cupsLangPrintf(stderr, + _("ppdc: Unable to load localization file " + "\"%s\" - %s"), argv[i], strerror(errno)); + return (1); + } + break; + + case 'd' : // Output directory... + i ++; + if (i >= argc) + usage(); + + if (verbose > 1) + _cupsLangPrintf(stdout, + _("ppdc: Writing PPD files to directory " + "\"%s\"."), argv[i]); + + outdir = argv[i]; + break; + + case 'l' : // Language(s)... + i ++; + if (i >= argc) + usage(); + + if (strchr(argv[i], ',')) + { + // Comma-delimited list of languages... + char temp[1024], // Copy of language list + *start, // Start of current locale name + *end; // End of current locale name + + + locales = new ppdcArray(); + + strlcpy(temp, argv[i], sizeof(temp)); + for (start = temp; *start; start = end) + { + if ((end = strchr(start, ',')) != NULL) + *end++ = '\0'; + else + end = start + strlen(start); + + if (end > start) + locales->add(new ppdcString(start)); + } + } + else + { + single_language = 1; + + if (verbose > 1) + _cupsLangPrintf(stdout, + _("ppdc: Loading messages for locale " + "\"%s\"."), argv[i]); + + if (catalog) + catalog->release(); + + catalog = new ppdcCatalog(argv[i]); + + if (catalog->messages->count == 0) + { + _cupsLangPrintf(stderr, + _("ppdc: Unable to find localization for " + "\"%s\" - %s"), argv[i], strerror(errno)); + return (1); + } + } + break; + + case 'm' : // Use ModelName for filename + use_model_name = 1; + break; + + case 't' : // Test PPDs instead of generating them + do_test = 1; + break; + + case 'v' : // Be verbose... + verbose ++; + break; + + case 'z' : // Compress files... + comp = 1; + break; + + case '-' : // --option + if (!strcmp(opt, "-lf")) + { + le = PPDC_LFONLY; + opt += strlen(opt) - 1; + break; + } + else if (!strcmp(opt, "-cr")) + { + le = PPDC_CRONLY; + opt += strlen(opt) - 1; + break; + } + else if (!strcmp(opt, "-crlf")) + { + le = PPDC_CRLF; + opt += strlen(opt) - 1; + break; + } + + default : // Unknown + usage(); + break; + } + } + else + { + // Open and load the driver info file... + if (verbose > 1) + _cupsLangPrintf(stdout, + _("ppdc: Loading driver information file \"%s\"."), + argv[i]); + + src->read_file(argv[i]); + } + + + if (src->drivers->count > 0) + { + // Create the output directory... + if (mkdir(outdir, 0777)) + { + if (errno != EEXIST) + { + _cupsLangPrintf(stderr, + _("ppdc: Unable to create output directory %s: %s"), + outdir, strerror(errno)); + return (1); + } + } + + // Write PPD files... + for (d = (ppdcDriver *)src->drivers->first(); + d; + d = (ppdcDriver *)src->drivers->next()) + { + if (do_test) + { + // Test the PPD file for this driver... + int pid, // Process ID + fds[2]; // Pipe file descriptors + + + if (pipe(fds)) + { + _cupsLangPrintf(stderr, + _("ppdc: Unable to create output pipes: %s"), + strerror(errno)); + return (1); + } + + if ((pid = fork()) == 0) + { + // Child process comes here... + dup2(fds[0], 0); + + close(fds[0]); + close(fds[1]); + + execlp("cupstestppd", "cupstestppd", "-", (char *)0); + + _cupsLangPrintf(stderr, + _("ppdc: Unable to execute cupstestppd: %s"), + strerror(errno)); + return (errno); + } + else if (pid < 0) + { + _cupsLangPrintf(stderr, _("ppdc: Unable to execute cupstestppd: %s"), + strerror(errno)); + return (errno); + } + + close(fds[0]); + fp = cupsFileOpenFd(fds[1], "w"); + } + else + { + // Write the PPD file for this driver... + if (use_model_name) + { + if (!_cups_strncasecmp(d->model_name->value, d->manufacturer->value, + strlen(d->manufacturer->value))) + { + // Model name already starts with the manufacturer... + outname = d->model_name->value; + } + else + { + // Add manufacturer to the front of the model name... + snprintf(make_model, sizeof(make_model), "%s %s", + d->manufacturer->value, d->model_name->value); + outname = make_model; + } + } + else if (d->file_name) + outname = d->file_name->value; + else + outname = d->pc_file_name->value; + + if (strstr(outname, ".PPD")) + { + // Convert PCFileName to lowercase... + for (j = 0; + outname[j] && j < (int)(sizeof(pcfilename) - 1); + j ++) + pcfilename[j] = tolower(outname[j] & 255); + + pcfilename[j] = '\0'; + } + else + { + // Leave PCFileName as-is... + strlcpy(pcfilename, outname, sizeof(pcfilename)); + } + + // Open the PPD file for writing... + if (comp) + snprintf(filename, sizeof(filename), "%s/%s.gz", outdir, pcfilename); + else + snprintf(filename, sizeof(filename), "%s/%s", outdir, pcfilename); + + if (cupsArrayFind(filenames, filename)) + _cupsLangPrintf(stderr, + _("ppdc: Warning - overlapping filename \"%s\"."), + filename); + else + cupsArrayAdd(filenames, strdup(filename)); + + fp = cupsFileOpen(filename, comp ? "w9" : "w"); + if (!fp) + { + _cupsLangPrintf(stderr, + _("ppdc: Unable to create PPD file \"%s\" - %s."), + filename, strerror(errno)); + return (1); + } + + if (verbose) + _cupsLangPrintf(stdout, _("ppdc: Writing %s."), filename); + } + + /* + * Write the PPD file... + */ + + ppdcArray *templocales = locales; + + if (!templocales && !single_language) + { + templocales = new ppdcArray(); + for (ppdcCatalog *tempcatalog = (ppdcCatalog *)src->po_files->first(); + tempcatalog; + tempcatalog = (ppdcCatalog *)src->po_files->next()) + { + tempcatalog->locale->retain(); + templocales->add(tempcatalog->locale); + } + } + + if (d->write_ppd_file(fp, catalog, templocales, src, le)) + { + cupsFileClose(fp); + return (1); + } + + if (templocales != locales) + templocales->release(); + + cupsFileClose(fp); + } + } + else + usage(); + + // Delete the printer driver information... + src->release(); + + // Message catalog... + if (catalog) + catalog->release(); + + // Return with no errors. + return (0); +} + + +// +// 'usage()' - Show usage and exit. +// + +static void +usage(void) +{ + _cupsLangPuts(stdout, _("Usage: ppdc [options] filename.drv [ ... " + "filenameN.drv ]")); + _cupsLangPuts(stdout, _("Options:")); + _cupsLangPuts(stdout, _(" -D name=value Set named variable to " + "value.")); + _cupsLangPuts(stdout, _(" -I include-dir Add include directory to " + "search path.")); + _cupsLangPuts(stdout, _(" -c catalog.po Load the specified " + "message catalog.")); + _cupsLangPuts(stdout, _(" -d output-dir Specify the output " + "directory.")); + _cupsLangPuts(stdout, _(" -l lang[,lang,...] Specify the output " + "language(s) (locale).")); + _cupsLangPuts(stdout, _(" -m Use the ModelName value " + "as the filename.")); + _cupsLangPuts(stdout, _(" -t Test PPDs instead of " + "generating them.")); + _cupsLangPuts(stdout, _(" -v Be verbose (more v's for " + "more verbosity).")); + _cupsLangPuts(stdout, _(" -z Compress PPD files using " + "GNU zip.")); + _cupsLangPuts(stdout, _(" --cr End lines with CR (Mac " + "OS 9).")); + _cupsLangPuts(stdout, _(" --crlf End lines with CR + LF " + "(Windows).")); + _cupsLangPuts(stdout, _(" --lf End lines with LF " + "(UNIX/Linux/Mac OS X).")); + + exit(1); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdc.h b/ppdc/ppdc.h new file mode 100644 index 0000000000..24c9f2c269 --- /dev/null +++ b/ppdc/ppdc.h @@ -0,0 +1,532 @@ +// +// "$Id$" +// +// Definitions for the CUPS PPD Compiler. +// +// Copyright 2007-2009 by Apple Inc. +// Copyright 2002-2007 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// + +#ifndef _PPDC_H_ +# define _PPDC_H_ + +// +// Include necessary headers... +// + +# include +# include + + +// +// Macros... +// + +# define PPDC_NAME(s) const char *class_name() { return (s); } + + +// +// Enumerations... +// + +enum ppdcDrvType //// Driver type +{ + PPDC_DRIVER_CUSTOM, // Custom driver + PPDC_DRIVER_PS, // PostScript driver + PPDC_DRIVER_ESCP, // rastertoescpx driver + PPDC_DRIVER_PCL, // rastertopclx driver + PPDC_DRIVER_LABEL, // rastertolabel/rastertodymo driver + PPDC_DRIVER_EPSON, // rastertoepson driver + PPDC_DRIVER_HP, // rastertohp driver + PPDC_DRIVER_MAX // Number of driver types defined +}; + +enum ppdcFontStatus //// Load status of font +{ + PPDC_FONT_ROM, // Font is in ROM + PPDC_FONT_DISK // Font is on disk +}; + +enum ppdcOptSection //// Option section +{ + PPDC_SECTION_ANY, // AnySetup + PPDC_SECTION_DOCUMENT, // DocumentSetup + PPDC_SECTION_EXIT, // ExitServer + PPDC_SECTION_JCL, // JCLSetup + PPDC_SECTION_PAGE, // PageSetup + PPDC_SECTION_PROLOG // Prolog +}; + +enum ppdcOptType //// Option type +{ + PPDC_BOOLEAN, // True/false option + PPDC_PICKONE, // Single choice from list + PPDC_PICKMANY // Multiple choices from list +}; + +enum ppdcLineEnding //// Line endings +{ + PPDC_LFONLY, // LF only + PPDC_CRONLY, // CR only + PPDC_CRLF // CR + LF +}; + +enum ppdcCondFlags //// Condition flags +{ + PPDC_COND_NORMAL = 0, // Normal state + PPDC_COND_SKIP = 1, // Skip state + PPDC_COND_SATISFIED = 2 // At least one condition satisfied +}; + + +// +// Printer description data... +// + +class ppdcShared //// Shared Data Value +{ + private: + + int use; // Use count (delete when 0) + + public: + + ppdcShared(); + virtual ~ppdcShared(); + + virtual const char *class_name() = 0; + + void retain(); + void release(); +}; + +class ppdcArray //// Shared Array + : public ppdcShared +{ + public: + + int count, // Number of elements + alloc, // Allocated elements + current; // Current element + ppdcShared **data; // Elements + + ppdcArray(ppdcArray *a = 0); + ~ppdcArray(); + + PPDC_NAME("ppdcArray") + + void add(ppdcShared *d); + ppdcShared *first(); + ppdcShared *next(); + void remove(ppdcShared *d); +}; + +class ppdcString //// Shared String + : public ppdcShared +{ + public: + + char *value; // String value + + ppdcString(const char *v); + ~ppdcString(); + + PPDC_NAME("ppdcString") +}; + +class ppdcInteger //// Shared integer + : public ppdcShared +{ + public: + + int *value; // Integer value + + ppdcInteger(int *v) { value = v; } + + PPDC_NAME("ppdcInteger") +}; + +class ppdcMessage //// Translation message + : public ppdcShared +{ + public: + + ppdcString *id, // Translation ID + *string; // Translation string + + ppdcMessage(const char *i, const char *s); + ~ppdcMessage(); + + PPDC_NAME("ppdcMessage") +}; + +class ppdcCatalog //// Translation catalog + : public ppdcShared +{ + public: + + ppdcString *locale; // Name of locale + ppdcString *filename; // Name of translation file + ppdcArray *messages; // Array of translation messages + + ppdcCatalog(const char *l, const char *f = 0); + ~ppdcCatalog(); + + PPDC_NAME("ppdcCatalog") + + void add_message(const char *id, const char *string = NULL); + const char *find_message(const char *id); + int load_messages(const char *f); + int save_messages(const char *f); +}; + +class ppdcAttr //// Attribute + : public ppdcShared +{ + public: + + ppdcString *name, // Name of attribute + *selector, // Selector string + *text, // Text string + *value; // Value string + bool localizable; // Should this attribute be localized? + + ppdcAttr(const char *n, const char *s, const char *t, const char *v, + bool loc = false); + ~ppdcAttr(); + + PPDC_NAME("ppdcAttr") +}; + +class ppdcFont //// Shared Font + : public ppdcShared +{ + public: + + ppdcString *name, // Font name + *encoding, // Font base encoding + *version, // Font version + *charset; // Font charset + ppdcFontStatus status; // Font status (ROM or Disk) + + ppdcFont(const char *n, const char *e, const char *v, const char *c, + ppdcFontStatus s); + ~ppdcFont(); + + PPDC_NAME("ppdcFont") +}; + +class ppdcChoice //// Option Choice + : public ppdcShared +{ + public: + + ppdcString *name, // Name of choice + *text, // Human-readable text of choice + *code; // PS code of choice + + ppdcChoice(const char *n, const char *t, const char *c); + ~ppdcChoice(); + + PPDC_NAME("ppdcChoice") +}; + +class ppdcOption //// Option + : public ppdcShared +{ + public: + + ppdcOptType type; // Type of option + ppdcString *name, // Name of option + *text; // Human-readable text of option + ppdcOptSection section; // Section for option code + float order; // Order number + ppdcArray *choices; // Choices + ppdcString *defchoice; // Default choice + + ppdcOption(ppdcOptType ot, const char *n, const char *t, ppdcOptSection s, + float o); + ppdcOption(ppdcOption *o); + ~ppdcOption(); + + PPDC_NAME("ppdcOption") + + void add_choice(ppdcChoice *c) { choices->add(c); } + ppdcChoice *find_choice(const char *n); + void set_defchoice(ppdcChoice *c); +}; + +class ppdcGroup //// Group of Options + : public ppdcShared +{ + public: + + ppdcString *name, // Name of option + *text; // Human-readable text of option + ppdcArray *options; // Options + + ppdcGroup(const char *n, const char *t); + ppdcGroup(ppdcGroup *g); + ~ppdcGroup(); + + PPDC_NAME("ppdcGroup") + + void add_option(ppdcOption *o) { options->add(o); } + ppdcOption *find_option(const char *n); +}; + +class ppdcConstraint //// Constraint + : public ppdcShared +{ + public: + + ppdcString *option1, // First option + *choice1, // First choice + *option2, // Second option + *choice2; // Second choice + + ppdcConstraint(const char *o1, const char *c1, const char *o2, + const char *c2); + ~ppdcConstraint(); + + PPDC_NAME("ppdcConstraint") +}; + +class ppdcFilter //// Filter Program + : public ppdcShared +{ + public: + + ppdcString *mime_type, // MIME type + *program; // Filter program + int cost; // Relative cost of filter + + ppdcFilter(const char *t, const char *p, int c); + ~ppdcFilter(); + + PPDC_NAME("ppdcFilter") +}; + +class ppdcMediaSize //// Media Size + : public ppdcShared +{ + public: + + ppdcString *name, // Name of size + *text; // Human-readable text + float width, // Width in points + length, // Length in points + left, // Left limit in points + bottom, // Bottom limit in points + right, // Right limit in points + top; // Top limit in points + ppdcString *size_code, // PageSize code, if any + *region_code; // PageRegion code, if any + + ppdcMediaSize(const char *n, const char *t, float w, float l, + float lm, float bm, float rm, float tm, + const char *sc = 0, const char *rc = 0); + ~ppdcMediaSize(); + + PPDC_NAME("ppdcMediaSize") +}; + +class ppdcProfile //// Color Profile + : public ppdcShared +{ + public: + + ppdcString *resolution, // Resolution name + *media_type; // Media type name + float density, // Color profile density + gamma, // Color profile gamma + profile[9]; // Color profile matrix + + ppdcProfile(const char *r, const char *m, float d, float g, const float *p); + ~ppdcProfile(); + + PPDC_NAME("ppdcProfile") +}; + +class ppdcSource; + +class ppdcDriver //// Printer Driver Data + : public ppdcShared +{ + public: + + ppdcDrvType type; // Driver type + ppdcArray *copyright; // Copyright strings + ppdcString *manufacturer, // Manufacturer + *model_name, // Name of printer model + *file_name, // Output filename for PPD + *pc_file_name, // 8 character PC filename for PPD + *version; // Version number + int model_number, // Model number for driver + manual_copies, // Do manual copies? + color_device, // Support color? + throughput; // Throughput in pages per minute + ppdcArray *attrs, // Attributes + *constraints, // Constraints + *filters, // Filters + *fonts, // Fonts + *groups, // Option groups + *profiles, // Color profiles + *sizes; // Fixed sizes + ppdcString *default_font, // Default font + *default_size; // Default size option + int variable_paper_size; // Support variable sizes? + ppdcString *custom_size_code; // Custom page size code, if any + float left_margin, // Margins for device in points + bottom_margin, + right_margin, + top_margin, + max_width, // Maximum width (points) + max_length, // Maximum length (points) + min_width, // Minimum width (points) + min_length; // Minimum length (points) + + ppdcDriver(ppdcDriver *d = 0); + ~ppdcDriver(); + + PPDC_NAME("ppdcDriver") + + void add_attr(ppdcAttr *a) { attrs->add(a); } + void add_constraint(ppdcConstraint *c) { constraints->add(c); } + void add_copyright(const char *c) { + copyright->add(new ppdcString(c)); + } + void add_filter(ppdcFilter *f) { filters->add(f); } + void add_font(ppdcFont *f) { fonts->add(f); } + void add_group(ppdcGroup *g) { groups->add(g); } + void add_profile(ppdcProfile *p) { profiles->add(p); } + void add_size(ppdcMediaSize *m) { sizes->add(m); } + + ppdcAttr *find_attr(const char *k, const char *s); + ppdcGroup *find_group(const char *n); + ppdcOption *find_option(const char *n); + ppdcOption *find_option_group(const char *n, ppdcGroup **mg); + + void set_custom_size_code(const char *c); + void set_default_font(ppdcFont *f); + void set_default_size(ppdcMediaSize *m); + void set_file_name(const char *f); + void set_manufacturer(const char *m); + void set_model_name(const char *m); + void set_pc_file_name(const char *f); + void set_version(const char *v); + + int write_ppd_file(cups_file_t *fp, ppdcCatalog *catalog, + ppdcArray *locales, ppdcSource *src, + ppdcLineEnding le); +}; + +class ppdcVariable //// Variable Definition + : public ppdcShared +{ + public: + + ppdcString *name, // Name of variable + *value; // Value of variable + + ppdcVariable(const char *n, const char *v); + ~ppdcVariable(); + + PPDC_NAME("ppdcVariable") + + void set_value(const char *v); +}; + +class ppdcFile //// File +{ + public: + + cups_file_t *fp; // File pointer + const char *filename; // Filename + int line; // Line in file + + ppdcFile(const char *f, cups_file_t *ffp = (cups_file_t *)0); + ~ppdcFile(); + + int get(); + int peek(); +}; + +class ppdcSource //// Source File + : public ppdcShared +{ + public: + + static ppdcArray *includes; // Include directories + static const char *driver_types[]; // Driver types + + ppdcString *filename; // Filename + ppdcArray *base_fonts, // Base fonts + *drivers, // Printer drivers + *po_files, // Message catalogs + *sizes, // Predefined media sizes + *vars; // Defined variables + int cond_state, // Cummulative conditional state + *cond_current, // Current #if state + cond_stack[101]; // #if state stack + + + ppdcSource(const char *f = 0, cups_file_t *ffp = (cups_file_t *)0); + ~ppdcSource(); + + PPDC_NAME("ppdcSource") + + static void add_include(const char *d); + ppdcDriver *find_driver(const char *f); + static char *find_include(const char *f, const char *base, char *n, + int nlen); + ppdcCatalog *find_po(const char *l); + ppdcMediaSize *find_size(const char *s); + ppdcVariable *find_variable(const char *n); + ppdcAttr *get_attr(ppdcFile *fp, bool loc = false); + int get_boolean(ppdcFile *fp); + ppdcChoice *get_choice(ppdcFile *fp); + ppdcChoice *get_color_model(ppdcFile *fp); + int get_color_order(const char *co); + ppdcProfile *get_color_profile(ppdcFile *fp); + int get_color_space(const char *cs); + ppdcConstraint *get_constraint(ppdcFile *fp); + ppdcMediaSize *get_custom_size(ppdcFile *fp); + void get_duplex(ppdcFile *fp, ppdcDriver *d); + ppdcFilter *get_filter(ppdcFile *fp); + float get_float(ppdcFile *fp); + ppdcFont *get_font(ppdcFile *fp); + ppdcChoice *get_generic(ppdcFile *fp, const char *keyword, + const char *tattr, const char *nattr); + ppdcGroup *get_group(ppdcFile *fp, ppdcDriver *d); + ppdcOption *get_installable(ppdcFile *fp); + int get_integer(const char *v); + int get_integer(ppdcFile *fp); + float get_measurement(ppdcFile *fp); + ppdcOption *get_option(ppdcFile *fp, ppdcDriver *d, ppdcGroup *g); + ppdcCatalog *get_po(ppdcFile *fp); + ppdcChoice *get_resolution(ppdcFile *fp); + ppdcProfile *get_simple_profile(ppdcFile *fp); + ppdcMediaSize *get_size(ppdcFile *fp); + char *get_token(ppdcFile *fp, char *buffer, int buflen); + ppdcVariable *get_variable(ppdcFile *fp); + int import_ppd(const char *f); + int quotef(cups_file_t *fp, const char *format, ...); + void read_file(const char *f, cups_file_t *ffp = (cups_file_t *)0); + void scan_file(ppdcFile *fp, ppdcDriver *td = 0, bool inc = false); + ppdcVariable *set_variable(const char *name, const char *value); + int write_file(const char *f); +}; + + +#endif // !_PPDC_H_ + +// +// End of "$Id$". +// diff --git a/ppdc/ppdhtml.cxx b/ppdc/ppdhtml.cxx new file mode 100644 index 0000000000..d5678898d3 --- /dev/null +++ b/ppdc/ppdhtml.cxx @@ -0,0 +1,186 @@ +// +// "$Id$" +// +// PPD to HTML utility for the CUPS PPD Compiler. +// +// Copyright 2007-2011 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// main() - Main entry for the PPD to HTML utility. +// usage() - Show usage and exit. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" +#include +#include + + +// +// Local functions... +// + +static void usage(void); + + +// +// 'main()' - Main entry for the PPD compiler. +// + +int // O - Exit status +main(int argc, // I - Number of command-line arguments + char *argv[]) // I - Command-line arguments +{ + int i; // Looping var + ppdcSource *src; // PPD source file data + ppdcDriver *d; // Current driver + ppdcGroup *g, // Current group + *composite; // Composite of all drivers + ppdcOption *o, // Current option + *compo; // Composite option + ppdcChoice *c; // Current choice + char *opt; // Current option char + ppdcMediaSize *size; // Current media size + char *value; // Value in option + + + _cupsSetLocale(argv); + + // Scan the command-line... + src = 0; + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-') + { + for (opt = argv[i] + 1; *opt; opt ++) + switch (*opt) + { + case 'D' : // Define variable + i ++; + if (i >= argc) + usage(); + + if ((value = strchr(argv[i], '=')) != NULL) + { + *value++ = '\0'; + + src->set_variable(argv[i], value); + } + else + src->set_variable(argv[i], "1"); + break; + + case 'I' : // Include directory... + i ++; + if (i >= argc) + usage(); + + ppdcSource::add_include(argv[i]); + break; + + default : // Unknown + usage(); + break; + } + } + else + { + // Open and load the driver info file... + src = new ppdcSource(argv[i]); + + // Create a composite group with all of the features from the + // drivers in the info file... + composite = new ppdcGroup("", ""); + + for (d = (ppdcDriver *)src->drivers->first(); d; d = (ppdcDriver *)src->drivers->next()) + for (g = (ppdcGroup *)d->groups->first(); g; g = (ppdcGroup *)d->groups->next()) + for (o = (ppdcOption *)g->options->first(); o; o = (ppdcOption *)g->options->next()) + { + if ((compo = composite->find_option(o->name->value)) == NULL) + composite->add_option(new ppdcOption(o)); + } + + puts(""); + printf("Driver Summary for %s\n", argv[i]); + printf("

Driver Summary for %s

\n", argv[i]); + printf("

"); + for (compo = (ppdcOption *)composite->options->first(); compo; compo = (ppdcOption *)composite->options->next()) + printf("", compo->text->value); + puts(""); + + // Write HTML summary... + for (d = (ppdcDriver *)src->drivers->first(); d; d = (ppdcDriver *)src->drivers->next()) + { + // Write the summary for this driver... + printf(""); + + for (compo = (ppdcOption *)composite->options->first(); compo; + compo = (ppdcOption *)composite->options->next()) + if ((o = d->find_option(compo->name->value)) != NULL) + { + printf(""); + } + else + printf(""); + + puts(""); + } + + puts("
PrinterMedia Size%s
%s", d->model_name->value); + for (size = (ppdcMediaSize *)d->sizes->first(); size; + size = (ppdcMediaSize *)d->sizes->next()) + printf("%s
", size->text->value); + printf("
"); + for (c = (ppdcChoice *)o->choices->first(); c; + c = (ppdcChoice *)o->choices->next()) + printf("%s
", c->text->value); + printf("
N/A

"); + puts(""); + puts(""); + // Delete the printer driver information... + composite->release(); + src->release(); + } + + // If no drivers have been loaded, display the program usage message. + if (!src) + usage(); + + // Return with no errors. + return (0); +} + + +// +// 'usage()' - Show usage and exit. +// + +static void +usage(void) +{ + _cupsLangPuts(stdout, _("Usage: ppdhtml [options] filename.drv " + ">filename.html")); + _cupsLangPuts(stdout, _("Options:")); + _cupsLangPuts(stdout, _(" -D name=value Set named variable to " + "value.")); + _cupsLangPuts(stdout, _(" -I include-dir Add include directory " + "to search path.")); + + exit(1); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdi.cxx b/ppdc/ppdi.cxx new file mode 100644 index 0000000000..a411bfef54 --- /dev/null +++ b/ppdc/ppdi.cxx @@ -0,0 +1,142 @@ +// +// "$Id$" +// +// PPD file import utility for the CUPS PPD Compiler. +// +// Copyright 2007-2011 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// main() - Main entry for the PPD import utility. +// usage() - Show usage and exit. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" +#include +#include +#include + + +// +// Local functions... +// + +static void usage(void); + + +// +// 'main()' - Main entry for the PPD import utility. +// + +int // O - Exit status +main(int argc, // I - Number of command-line arguments + char *argv[]) // I - Command-line arguments +{ + int i; // Looping var + char *opt; // Current option + const char *srcfile; // Output file + ppdcSource *src; // PPD source file data + + + _cupsSetLocale(argv); + + // Scan the command-line... + srcfile = NULL; + src = NULL; + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-') + { + for (opt = argv[i] + 1; *opt; opt ++) + switch (*opt) + { + case 'o' : // Output file + if (srcfile || src) + usage(); + + i ++; + if (i >= argc) + usage(); + + srcfile = argv[i]; + break; + + case 'I' : // Include dir + i ++; + if (i >= argc) + usage(); + + ppdcSource::add_include(argv[i]); + break; + + default : // Unknown + usage(); + break; + } + } + else + { + // Open and load the driver info file... + if (!srcfile) + srcfile = "ppdi.drv"; + + if (!src) + { + if (access(srcfile, 0)) + src = new ppdcSource(); + else + src = new ppdcSource(srcfile); + } + + // Import the PPD file... + src->import_ppd(argv[i]); + } + + // If no drivers have been loaded, display the program usage message. + if (!src) + usage(); + + // Write the driver info file back to disk... + src->write_file(srcfile); + + // Delete the printer driver information... + src->release(); + + // Return with no errors. + return (0); +} + + +// +// 'usage()' - Show usage and exit. +// + +static void +usage(void) +{ + _cupsLangPuts(stdout, _("Usage: ppdi [options] filename.ppd [ ... " + "filenameN.ppd ]")); + _cupsLangPuts(stdout, _("Options:")); + _cupsLangPuts(stdout, _(" -I include-dir Add include directory to " + "search path.")); + _cupsLangPuts(stdout, _(" -o filename.drv Set driver information " + "file (otherwise ppdi.drv).")); + + exit(1); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdmerge.cxx b/ppdc/ppdmerge.cxx new file mode 100644 index 0000000000..68f2916738 --- /dev/null +++ b/ppdc/ppdmerge.cxx @@ -0,0 +1,379 @@ +// +// "$Id$" +// +// PPD file merge utility for the CUPS PPD Compiler. +// +// Copyright 2007-2011 by Apple Inc. +// Copyright 2002-2007 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// main() - Main entry for the PPD merge utility. +// ppd_locale() - Return the locale associated with a PPD file. +// usage() - Show usage and exit. +// + +// +// Include necessary headers... +// + +#include +#include +#include + + +// +// Local functions... +// + +static const char *ppd_locale(ppd_file_t *ppd); +static void usage(void); + + +// +// 'main()' - Main entry for the PPD merge utility. +// + +int // O - Exit status +main(int argc, // I - Number of command-line arguments + char *argv[]) // I - Command-line arguments +{ + int i; // Looping var + char *opt; // Current option + ppd_file_t *ppd; // PPD file + cups_array_t *ppds; // Array of PPD files + const char *inname, // First input filename + *outname; // Output filename (if any) + cups_file_t *infile, // Input file + *outfile; // Output file + cups_array_t *languages; // Languages in file + const char *locale; // Current locale + char line[1024]; // Line from file + + + _cupsSetLocale(argv); + + // Scan the command-line... + inname = NULL; + outname = NULL; + outfile = NULL; + languages = NULL; + ppds = cupsArrayNew(NULL, NULL); + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-') + { + for (opt = argv[i] + 1; *opt; opt ++) + switch (*opt) + { + case 'o' : // Output file + if (outname) + usage(); + + i ++; + if (i >= argc) + usage(); + + outname = argv[i]; + break; + + default : // Unknown + usage(); + break; + } + } + else + { + // Open and load the PPD file... + if ((infile = cupsFileOpen(argv[i], "r")) == NULL) + { + _cupsLangPrintf(stderr, _("%s: Unable to open %s: %s"), "ppdmerge", + argv[i], strerror(errno)); + return (1); + } + + // Open the PPD file... + if ((ppd = ppdOpen2(infile)) == NULL) + { + ppd_status_t status; // PPD open status + int curline, // Current line + linenum; // Line number + + + status = ppdLastError(&linenum); + + _cupsLangPrintf(stderr, + _("%s: Unable to open PPD file: %s on line %d."), + "ppdmerge", ppdErrorString(status), linenum); + cupsFileRewind(infile); + + line[0] = '\0'; + curline = 0; + + while (cupsFileGets(infile, line, sizeof(line))) + { + curline ++; + if (curline >= linenum) + break; + } + + _cupsLangPrintf(stderr, "%d: %s", linenum, line); + + cupsFileClose(infile); + return (1); + } + + // Figure out the locale... + if ((locale = ppd_locale(ppd)) == NULL) + { + _cupsLangPrintf(stderr, + _("ppdmerge: Bad LanguageVersion \"%s\" in %s."), + ppd->lang_version, argv[i]); + cupsFileClose(infile); + ppdClose(ppd); + return (1); + } + + if (!strcmp(locale, "en") && !inname && !outfile) + { + // Set the English PPD's filename... + inname = argv[i]; + languages = _ppdGetLanguages(ppd); + + if (outname && !strcmp(inname, outname)) + { + // Rename input filename so that we don't overwrite it... + char bckname[1024]; // Backup filename + + + snprintf(bckname, sizeof(bckname), "%s.bck", inname); + + if (rename(inname, bckname)) + { + _cupsLangPrintf(stderr, + _("ppdmerge: Unable to backup %s to %s - %s"), + inname, bckname, strerror(errno)); + return (1); + } + + inname = bckname; + } + } + else if (strcmp(locale, "en")) + { + // Save this PPD for later processing... + cupsArrayAdd(ppds, ppd); + } + else + { + // Don't need this PPD... + _cupsLangPrintf(stderr, _("ppdmerge: Ignoring PPD file %s."), + argv[i]); + ppdClose(ppd); + } + + // Close and move on... + cupsFileClose(infile); + } + + // If no PPDs have been loaded, display the program usage message. + if (!inname) + usage(); + + // Loop through the PPD files we loaded to generate a new language list... + if (!languages) + languages = cupsArrayNew((cups_array_func_t)strcmp, NULL); + + for (ppd = (ppd_file_t *)cupsArrayFirst(ppds); + ppd; + ppd = (ppd_file_t *)cupsArrayNext(ppds)) + { + locale = ppd_locale(ppd); + + if (cupsArrayFind(languages, (void *)locale)) + { + // Already have this language, remove the PPD from the list. + ppdClose(ppd); + cupsArrayRemove(ppds, ppd); + } + else + cupsArrayAdd(languages, (void *)locale); + } + + // Copy the English PPD starting with a cupsLanguages line... + infile = cupsFileOpen(inname, "r"); + + if (outname) + { + const char *ext = strrchr(outname, '.'); + if (ext && !strcmp(ext, ".gz")) + outfile = cupsFileOpen(outname, "w9"); + else + outfile = cupsFileOpen(outname, "w"); + } + else + outfile = cupsFileStdout(); + + cupsFileGets(infile, line, sizeof(line)); + cupsFilePrintf(outfile, "%s\n", line); + if ((locale = (char *)cupsArrayFirst(languages)) != NULL) + { + cupsFilePrintf(outfile, "*cupsLanguages: \"%s", locale); + while ((locale = (char *)cupsArrayNext(languages)) != NULL) + cupsFilePrintf(outfile, " %s", locale); + cupsFilePuts(outfile, "\"\n"); + } + + while (cupsFileGets(infile, line, sizeof(line))) + { + if (strncmp(line, "*cupsLanguages:", 15)) + cupsFilePrintf(outfile, "%s\n", line); + } + + // Loop through the other PPD files we loaded to provide the translations... + for (ppd = (ppd_file_t *)cupsArrayFirst(ppds); + ppd; + ppd = (ppd_file_t *)cupsArrayNext(ppds)) + { + // Output all of the UI text for this language... + int j, k, l; // Looping vars + ppd_group_t *g; // Option group + ppd_option_t *o; // Option + ppd_choice_t *c; // Choice + ppd_coption_t *co; // Custom option + ppd_cparam_t *cp; // Custom parameter + ppd_attr_t *attr; // PPD attribute + + locale = ppd_locale(ppd); + + cupsFilePrintf(outfile, "*%% %s localization\n", ppd->lang_version); + cupsFilePrintf(outfile, "*%s.Translation ModelName/%s: \"\"\n", locale, + ppd->modelname); + + for (j = ppd->num_groups, g = ppd->groups; j > 0; j --, g ++) + { + cupsFilePrintf(outfile, "*%s.Translation %s/%s: \"\"\n", locale, + g->name, g->text); + + for (k = g->num_options, o = g->options; k > 0; k --, o ++) + { + cupsFilePrintf(outfile, "*%s.Translation %s/%s: \"\"\n", locale, + o->keyword, o->text); + + for (l = o->num_choices, c = o->choices; l > 0; l --, c ++) + cupsFilePrintf(outfile, "*%s.%s %s/%s: \"\"\n", locale, + o->keyword, c->choice, c->text); + + if ((co = ppdFindCustomOption(ppd, o->keyword)) != NULL) + { + snprintf(line, sizeof(line), "Custom%s", o->keyword); + attr = ppdFindAttr(ppd, line, "True"); + cupsFilePrintf(outfile, "*%s.Custom%s True/%s: \"\"\n", locale, + o->keyword, attr->text); + for (cp = ppdFirstCustomParam(co); cp; cp = ppdNextCustomParam(co)) + cupsFilePrintf(outfile, "*%s.ParamCustom%s %s/%s: \"\"\n", locale, + o->keyword, cp->name, cp->text); + } + } + } + + ppdClose(ppd); + } + + cupsArrayDelete(ppds); + + cupsFileClose(outfile); + + // Return with no errors. + return (0); +} + + +// +// 'ppd_locale()' - Return the locale associated with a PPD file. +// + +static const char * // O - Locale string +ppd_locale(ppd_file_t *ppd) // I - PPD file +{ + int i, // Looping var + vlen; // Length of LanguageVersion string + static char locale[255]; // Locale string + static struct // LanguageVersion translation table + { + const char *version, // LanguageVersion string */ + *language; // Language code */ + } languages[] = + { + { "chinese", "zh" }, + { "czech", "cs" }, + { "danish", "da" }, + { "dutch", "nl" }, + { "english", "en" }, + { "finnish", "fi" }, + { "french", "fr" }, + { "german", "de" }, + { "greek", "el" }, + { "hungarian", "hu" }, + { "italian", "it" }, + { "japanese", "ja" }, + { "korean", "ko" }, + { "norwegian", "no" }, + { "polish", "pl" }, + { "portuguese", "pt" }, + { "russian", "ru" }, + { "simplified chinese", "zh_CN" }, + { "slovak", "sk" }, + { "spanish", "es" }, + { "swedish", "sv" }, + { "traditional chinese", "zh_TW" }, + { "turkish", "tr" } + }; + + + for (i = 0; i < (int)(sizeof(languages) / sizeof(languages[0])); i ++) + { + vlen = strlen(languages[i].version); + + if (!_cups_strncasecmp(ppd->lang_version, languages[i].version, vlen)) + { + if (ppd->lang_version[vlen] == '-' || + ppd->lang_version[vlen] == '_') + snprintf(locale, sizeof(locale), "%s_%s", languages[i].language, + ppd->lang_version + vlen + 1); + else + strlcpy(locale, languages[i].language, sizeof(locale)); + + return (locale); + } + } + + return (NULL); +} + +// +// 'usage()' - Show usage and exit. +// + +static void +usage(void) +{ + _cupsLangPuts(stdout, _("Usage: ppdmerge [options] filename.ppd [ ... " + "filenameN.ppd ]")); + _cupsLangPuts(stdout, _("Options:")); + _cupsLangPuts(stdout, _(" -o filename.ppd[.gz] Set output file " + "(otherwise stdout).")); + + exit(1); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/ppdpo.cxx b/ppdc/ppdpo.cxx new file mode 100644 index 0000000000..bdf0f7c052 --- /dev/null +++ b/ppdc/ppdpo.cxx @@ -0,0 +1,268 @@ +// +// "$Id$" +// +// PPD file message catalog program for the CUPS PPD Compiler. +// +// Copyright 2007-2011 by Apple Inc. +// Copyright 2002-2005 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// main() - Main entry for the PPD compiler. +// add_ui_strings() - Add all UI strings from the driver. +// usage() - Show usage and exit. +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" +#include +#include + + +// +// Local functions... +// + +static void add_ui_strings(ppdcDriver *d, ppdcCatalog *catalog); +static void usage(void); + + +// +// 'main()' - Main entry for the PPD compiler. +// + +int // O - Exit status +main(int argc, // I - Number of command-line arguments + char *argv[]) // I - Command-line arguments +{ + int i; // Looping var + ppdcCatalog *catalog; // Message catalog + ppdcSource *src; // PPD source file data + ppdcDriver *d; // Current driver + char *opt; // Current option + int verbose; // Verbosity + const char *outfile; // Output file + char *value; // Value in option + + + _cupsSetLocale(argv); + + // Scan the command-line... + catalog = new ppdcCatalog("en"); + src = 0; + verbose = 0; + outfile = 0; + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-') + { + for (opt = argv[i] + 1; *opt; opt ++) + switch (*opt) + { + case 'D' : // Define variable + i ++; + if (i >= argc) + usage(); + + if ((value = strchr(argv[i], '=')) != NULL) + { + *value++ = '\0'; + + src->set_variable(argv[i], value); + } + else + src->set_variable(argv[i], "1"); + break; + + case 'I' : // Include directory... + i ++; + if (i >= argc) + usage(); + + if (verbose > 1) + _cupsLangPrintf(stdout, + _("ppdc: Adding include directory \"%s\"."), + argv[i]); + + ppdcSource::add_include(argv[i]); + break; + + case 'o' : // Output file... + i ++; + if (i >= argc || outfile) + usage(); + + outfile = argv[i]; + + catalog->load_messages(outfile); + break; + + case 'v' : // Be verbose... + verbose ++; + break; + + default : // Unknown + usage(); + break; + } + } + else + { + // Open and load the driver info file... + if (verbose > 1) + _cupsLangPrintf(stdout, + _("ppdc: Loading driver information file \"%s\"."), + argv[i]); + + src = new ppdcSource(argv[i]); + + // Add UI strings... + for (d = (ppdcDriver *)src->drivers->first(); + d; + d = (ppdcDriver *)src->drivers->next()) + { + if (verbose) + _cupsLangPrintf(stderr, _("ppdc: Adding/updating UI text from %s."), + argv[i]); + + add_ui_strings(d, catalog); + } + + // Delete the printer driver information... + src->release(); + } + + // Write the message catalog... + if (!outfile) + usage(); + else + catalog->save_messages(outfile); + + catalog->release(); + + // If no drivers have been loaded, display the program usage message. + if (!src) + usage(); + + // Return with no errors. + return (0); +} + + +// +// 'add_ui_strings()' - Add all UI strings from the driver. +// + +static void +add_ui_strings(ppdcDriver *d, // I - Driver data + ppdcCatalog *catalog) // I - Message catalog +{ + // Add the make/model/language strings... + catalog->add_message(d->manufacturer->value); + catalog->add_message(d->model_name->value); + + // Add the media size strings... + ppdcMediaSize *m; // Current media size + + for (m = (ppdcMediaSize *)d->sizes->first(); + m; + m = (ppdcMediaSize *)d->sizes->next()) + catalog->add_message(m->text->value); + + // Add the group/option/choice strings... + ppdcGroup *g; // Current group + ppdcOption *o; // Current option + ppdcChoice *c; // Current choice + + for (g = (ppdcGroup *)d->groups->first(); + g; + g = (ppdcGroup *)d->groups->next()) + { + if (!g->options->count) + continue; + + if (_cups_strcasecmp(g->name->value, "General")) + catalog->add_message(g->text->value); + + for (o = (ppdcOption *)g->options->first(); + o; + o = (ppdcOption *)g->options->next()) + { + if (!o->choices->count) + continue; + + if (o->text->value) + catalog->add_message(o->text->value); + else + catalog->add_message(o->name->value); + + for (c = (ppdcChoice *)o->choices->first(); + c; + c = (ppdcChoice *)o->choices->next()) + if (c->text->value) + catalog->add_message(c->text->value); + else + catalog->add_message(c->name->value); + } + } + + // Add profile and preset strings... + ppdcAttr *a; // Current attribute + for (a = (ppdcAttr *)d->attrs->first(); + a; + a = (ppdcAttr *)d->attrs->next()) + if (a->text->value && a->text->value[0] && + (a->localizable || + !strncmp(a->name->value, "Custom", 6) || + !strncmp(a->name->value, "ParamCustom", 11) || + !strcmp(a->name->value, "APCustomColorMatchingName") || + !strcmp(a->name->value, "APPrinterPreset") || + !strcmp(a->name->value, "cupsICCProfile") || + !strcmp(a->name->value, "cupsIPPReason") || + !strcmp(a->name->value, "cupsMarkerName"))) + { + catalog->add_message(a->text->value); + + if ((a->localizable && a->value->value[0]) || + !strcmp(a->name->value, "cupsIPPReason")) + catalog->add_message(a->value->value); + } + else if (!strncmp(a->name->value, "Custom", 6) || + !strncmp(a->name->value, "ParamCustom", 11)) + catalog->add_message(a->name->value); +} + + +// +// 'usage()' - Show usage and exit. +// + +static void +usage(void) +{ + _cupsLangPuts(stdout, _("Usage: ppdpo [options] -o filename.po filename.drv " + "[ ... filenameN.drv ]")); + _cupsLangPuts(stdout, _("Options:")); + _cupsLangPuts(stdout, _(" -D name=value Set named variable to " + "value.")); + _cupsLangPuts(stdout, _(" -I include-dir Add include directory to " + "search path.")); + _cupsLangPuts(stdout, _(" -v Be verbose (more v's for " + "more verbosity).")); + + exit(1); +} + + +// +// End of "$Id$". +// diff --git a/ppdc/sample.drv b/ppdc/sample.drv new file mode 100644 index 0000000000..1af2a68474 --- /dev/null +++ b/ppdc/sample.drv @@ -0,0 +1,1254 @@ +// +// "$Id$" +// +// Driver info file for CUPS-supplied PPDs. +// +// Copyright 2007-2011 by Apple Inc. +// Copyright 1993-2006 by Easy Software Products. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// + +// Include necessary files... +#include +#include +#include +#include +#include + +// Localizations are provided for all of the base languages supported by +// CUPS... +//#po da "" +//#po de "" +//#po es "" +//#po et "" +//#po fi "" +//#po fr "" +//#po he "" +//#po id "" +//#po it "" +//#po ja "" +//#po ko "" +//#po nl "" +//#po no "" +//#po pl "" +//#po pt "" +//#po pt_BR "" +//#po ru "" +//#po sv "" +//#po zh "" +//#po zh_TW "" + +// MediaSize sizes used by label drivers... +#media "w81h252/Address - 1 1/8 x 3 1/2\"" 81 252 +#media "w101h252/Large Address - 1 4/10 x 3 1/2\"" 101 252 +#media "w54h144/Return Address - 3/4 x 2\"" 54 144 +#media "w167h288/Shipping Address - 2 5/16 x 4\"" 167 288 +#media "w162h540/Internet Postage 2-Part - 2 1/4 x 7 1/2\"" 162 540 +#media "w162h504/Internet Postage 3-Part - 2 1/4 x 7\"" 162 504 +#media "w41h248/File Folder - 9/16 x 3 7/16\"" 41 248 +#media "w41h144/Hanging Folder - 9/16 x 2\"" 41 144 +#media "w153h198/3.5\" Disk - 2 1/8 x 2 3/4\"" 153 198 +#media "w90h18/1.25x0.25\"" 90 18 +#media "w90h162/1.25x2.25\"" 90 162 +#media "w108h18/1.50x0.25\"" 108 18 +#media "w108h36/1.50x0.50\"" 108 36 +#media "w108h72/1.50x1.00\"" 108 72 +#media "w108h144/1.50x2.00\"" 108 144 +#media "w144h26/2.00x0.37\"" 144 26 +#media "w144h36/2.00x0.50\"" 144 36 +#media "w144h72/2.00x1.00\"" 144 72 +#media "w144h90/2.00x1.25\"" 144 90 +#media "w144h144/2.00x2.00\"" 144 144 +#media "w144h216/2.00x3.00\"" 144 216 +#media "w144h288/2.00x4.00\"" 144 288 +#media "w144h396/2.00x5.50\"" 144 396 +#media "w162h36/2.25x0.50\"" 162 36 +#media "w162h90/2.25x1.25\"" 162 90 +#media "w162h288/2.25x4.00\"" 162 288 +#media "w162h396/2.25x5.50\"" 162 396 +#media "w171h396/2.38x5.50\"" 171 396 +#media "w180h72/2.50x1.00\"" 180 72 +#media "w180h144/2.50x2.00\"" 180 144 +#media "w198h90/2.75x1.25\"" 198 90 +#media "w209h72/2.9 x 1\"" 209 72 +#media "w216h72/3.00x1.00\"" 216 72 +#media "w216h90/3.00x1.25\"" 216 90 +#media "w216h144/3.00x2.00\"" 216 144 +#media "w216h216/3.00x3.00\"" 216 216 +#media "w216h360/3.00x5.00\"" 216 360 +#media "w234h144/3.25x2.00\"" 234 144 +#media "w234h360/3.25x5.00\"" 234 360 +#media "w234h396/3.25x5.50\"" 234 396 +#media "w234h419/3.25x5.83\"" 234 419 +#media "w234h563/3.25x7.83\"" 234 563 +#media "w252h72/3.50x1.00\"" 252 72 +#media "w288h72/4.00x1.00\"" 288 72 +#media "w288h144/4.00x2.00\"" 288 144 +#media "w288h180/4.00x2.50\"" 288 180 +#media "w288h216/4.00x3.00\"" 288 216 +#media "w288h288/4.00x4.00\"" 288 288 +#media "w288h360/4.00x5.00\"" 288 360 +#media "w288h432/4.00x6.00\"" 288 432 +#media "w288h468/4.00x6.50\"" 288 468 +#media "w288h936/4.00x13.00\"" 288 936 +#media "w432h72/6.00x1.00\"" 432 72 +#media "w432h144/6.00x2.00\"" 432 144 +#media "w432h216/6.00x3.00\"" 432 216 +#media "w432h288/6.00x4.00\"" 432 288 +#media "w432h360/6.00x5.00\"" 432 360 +#media "w432h432/6.00x6.00\"" 432 432 +#media "w432h468/6.00x6.50\"" 432 468 +#media "w576h72/8.00x1.00\"" 576 72 +#media "w576h144/8.00x2.00\"" 576 144 +#media "w576h216/8.00x3.00\"" 576 216 +#media "w576h288/8.00x4.00\"" 576 288 +#media "w576h360/8.00x5.00\"" 576 360 +#media "w576h432/8.00x6.00\"" 576 432 +#media "w576h468/8.00x6.50\"" 576 468 + +// Common stuff for all drivers... +Attribute "cupsVersion" "" "1.6" +Attribute "FileSystem" "" "False" +Attribute "LandscapeOrientation" "" "Plus90" +Attribute "TTRasterizer" "" "Type42" + +Copyright "Copyright 2007-2011 by Apple Inc." +Copyright "Copyright 1997-2007 by Easy Software Products." +Copyright "" +Copyright "These coded instructions, statements, and computer programs are the" +Copyright "property of Apple Inc. and are protected by Federal copyright" +Copyright "law. Distribution and use rights are outlined in the file \"LICENSE.txt\"" +Copyright "which should have been included with this file. If this file is" +Copyright "file is missing or damaged, see the license at \"http://www.cups.org/\"." + +Font * + +Version "1.5" + +// Dymo Label Printer +{ + Manufacturer "Dymo" + ModelName "Label Printer" + Attribute NickName "" "Dymo Label Printer" + PCFileName "dymo.ppd" + DriverType label + ModelNumber $DYMO_3x0 + Throughput 8 + ManualCopies Yes + ColorDevice No + + HWMargins 2 14.9 2 14.9 + + *MediaSize w81h252 + MediaSize w101h252 + MediaSize w54h144 + MediaSize w167h288 + MediaSize w162h540 + MediaSize w162h504 + MediaSize w41h248 + MediaSize w41h144 + MediaSize w153h198 + + Resolution k 1 0 0 0 136dpi + Resolution k 1 0 0 0 203dpi + *Resolution k 1 0 0 0 300dpi + + Darkness 0 Light + Darkness 1 Medium + *Darkness 2 Normal + Darkness 3 Dark +} + +// Epson +{ + Manufacturer "Epson" + DriverType epson + ManualCopies Yes + ColorDevice No + Throughput 1 + + HWMargins 0 0 0 0 + VariablePaperSize Yes + MinSize 36 36 + MaxSize 1080 86400 + + // Epson 24-Pin Series + { + ModelName "24-Pin Series" + Attribute NickName "" "Epson 24-Pin Series" + PCFileName "epson24.ppd" + ModelNumber $EPSON_24PIN + + HWMargins 18 18 18 18 + *MediaSize Letter + MediaSize Legal + MediaSize A4 + MediaSize FanFoldUS + + Resolution k 1 8 0 0 60dpi + *Resolution k 1 8 0 0 120x60dpi + Resolution k 1 24 0 0 180dpi + Resolution k 1 24 0 0 360x180dpi + Resolution k 1 48 0 0 360dpi + } + + // Epson 9-Pin Series + { + ModelName "9-Pin Series" + Attribute NickName "" "Epson 9-Pin Series" + PCFileName "epson9.ppd" + ModelNumber $EPSON_9PIN + ColorDevice No + + HWMargins 18 18 18 18 + *MediaSize Letter + MediaSize Legal + MediaSize A4 + MediaSize FanFoldUS + + Resolution k 1 8 0 0 60x72dpi + *Resolution k 1 8 0 0 120x72dpi + Resolution k 1 8 0 0 240x72dpi + } + + // Epson Stylus Color Series + { + ModelName "Stylus Color Series" + Attribute NickName "" "Epson Stylus Color Series" + PCFileName "stcolor.ppd" + ModelNumber $EPSON_COLOR + ColorDevice Yes + + HWMargins 8.6 39.6 8.6 25.51 + *MediaSize Letter + MediaSize Legal + MediaSize A4 + + Resolution - 1 0 0 0 180dpi + *Resolution - 1 0 0 0 360dpi + Resolution - 1 0 0 0 720dpi + + *ColorModel CMYK cmyk banded 1 + ColorModel Gray/Grayscale k chunky 1 + } + + // Epson New Stylus Color Series + { + ModelName "New Stylus Color Series" + Attribute NickName "" "Epson New Stylus Color Series" + PCFileName "stcolor2.ppd" + ModelNumber $EPSON_ICOLOR + ColorDevice Yes + + HWMargins 8.6 39.6 8.6 25.51 + *MediaSize Letter + MediaSize Legal + MediaSize A4 + + Resolution - 1 0 0 0 180dpi + *Resolution - 1 0 0 0 360dpi + Resolution - 1 0 0 0 720dpi + + *ColorModel CMYK cmyk banded 1 + ColorModel Gray/Grayscale k chunky 1 + } + + // Epson Stylus Color Series + { + ModelName "Stylus Photo Series" + Attribute NickName "" "Epson Stylus Photo Series" + PCFileName "stphoto.ppd" + ModelNumber $EPSON_PHOTO + ColorDevice Yes + + HWMargins 8.6 39.6 8.6 25.51 + *MediaSize Letter + MediaSize Legal + MediaSize A4 + + Resolution - 1 0 0 0 180dpi + *Resolution - 1 0 0 0 360dpi + Resolution - 1 0 0 0 720dpi + + *ColorModel CMYK cmykcm banded 1 + ColorModel Gray/Grayscale k chunky 1 + } + + // Epson New Stylus Color Series + { + ModelName "New Stylus Photo Series" + Attribute NickName "" "Epson New Stylus Photo Series" + PCFileName "stphoto2.ppd" + ModelNumber $EPSON_IPHOTO + ColorDevice Yes + + HWMargins 8.6 39.6 8.6 25.51 + *MediaSize Letter + MediaSize Legal + MediaSize A4 + + Resolution - 1 0 0 0 180dpi + *Resolution - 1 0 0 0 360dpi + Resolution - 1 0 0 0 720dpi + + *ColorModel CMYK cmykcm banded 1 + ColorModel Gray/Grayscale k chunky 1 + } +} + +// Generic drivers +{ + Manufacturer "Generic" + + // Generic PCL Laser Printer + { + DriverType hp + + ModelName "PCL Laser Printer" + Attribute NickName "" "Generic PCL Laser Printer" + PCFileName "generpcl.ppd" + Throughput 8 + ModelNumber $HP_LASERJET + ColorDevice No + Attribute 1284DeviceID "" "CMD:PCL;" + + UIConstraints "*Duplex *Option1 False" + UIConstraints "*PageSize A3 *InputSlot Envelope" + UIConstraints "*PageSize A4 *InputSlot Envelope" + UIConstraints "*PageSize A5 *InputSlot Envelope" + UIConstraints "*PageSize B5 *InputSlot Envelope" + UIConstraints "*PageSize Executive *InputSlot Envelope" + UIConstraints "*PageSize Legal *InputSlot Envelope" + UIConstraints "*PageSize Letter *InputSlot Envelope" + UIConstraints "*PageSize Tabloid *InputSlot Envelope" + + HWMargins 18 12 18 12 + *MediaSize Letter + MediaSize Legal + MediaSize Executive + MediaSize Tabloid + MediaSize A3 + MediaSize A4 + MediaSize A5 + MediaSize B5 + MediaSize EnvISOB5 + MediaSize Env10 + MediaSize EnvC5 + MediaSize EnvDL + MediaSize EnvMonarch + + *Resolution k 1 0 0 0 300dpi + Resolution k 1 0 0 0 600dpi + + *InputSlot 0 "Default/Printer Default" + InputSlot 8 "Tray1/Tray 1" + InputSlot 1 "Tray2/Tray 2" + InputSlot 4 "Tray3/Tray 3" + InputSlot 5 "Tray4/Tray 4" + InputSlot 2 "Manual/Manual Feed" + InputSlot 3 "Envelope/Envelope Feed" + + Duplex Yes + Installable "Option1/Duplexer" + } + + // Generic PostScript Printer + { + DriverType ps + + ModelName "PostScript Printer" + Attribute NickName "" "Generic PostScript Printer" + PCFileName "generic.ppd" + Throughput 8 + ColorDevice Yes + Attribute PSVersion "" "(2016.0) 0" + Attribute LanguageLevel "" 2 + Attribute 1284DeviceID "" "CMD:PS;" + + UIConstraints "*Duplex *Option1 False" + + HWMargins 12 12 12 12 + *MediaSize Letter + MediaSize Legal + MediaSize Executive + MediaSize Tabloid + MediaSize A3 + MediaSize A4 + MediaSize A5 + MediaSize B5 + MediaSize EnvISOB5 + MediaSize Env10 + MediaSize EnvC5 + MediaSize EnvDL + MediaSize EnvMonarch + + Option "InputSlot/Media Source" PickOne AnySetup 10 + *Choice "Default/Printer Default" "" + Choice "Upper/Cassette" "<>setpagedevice" + Choice "Manual/Manual Feed" "<>setpagedevice" + + Duplex Yes + Installable "Option1/Duplexer" + Attribute "?Option1" "" "save currentpagedevice/Duplex known{(True)}{(False)}ifelse = flush restore" + } +} + +// HP +{ + Manufacturer "HP" + DriverType hp + + // HP DeskJet Series + { + ModelName "DeskJet Series" + Attribute NickName "" "HP DeskJet Series" + PCFileName "deskjet.ppd" + ModelNumber $HP_DESKJET + ManualCopies Yes + ColorDevice Yes + Throughput 1 + Attribute 1284DeviceID "" "MFG:HP;MDL:HP DeskJet;CMD:PCL;" + + UIConstraints "*PageSize A3 *InputSlot Envelope" + UIConstraints "*PageSize A4 *InputSlot Envelope" + UIConstraints "*PageSize A5 *InputSlot Envelope" + UIConstraints "*PageSize B5 *InputSlot Envelope" + UIConstraints "*PageSize Executive *InputSlot Envelope" + UIConstraints "*PageSize Legal *InputSlot Envelope" + UIConstraints "*PageSize Letter *InputSlot Envelope" + UIConstraints "*PageSize Tabloid *InputSlot Envelope" + + HWMargins 18 36 18 36 + *MediaSize Letter + MediaSize Legal + MediaSize Executive + MediaSize Tabloid + MediaSize A3 + MediaSize A4 + MediaSize A5 + MediaSize B5 + MediaSize EnvISOB5 + MediaSize Env10 + MediaSize EnvC5 + MediaSize EnvDL + MediaSize EnvMonarch + + ColorModel Gray/Grayscale k chunky 2 + ColorModel RGB/Color cmy banded 2 + *ColorModel CMYK kcmy banded 2 + + Resolution - 1 0 0 0 150dpi + *Resolution - 1 0 0 0 300dpi + Resolution - 1 0 0 0 600dpi + + *InputSlot 1 Tray + InputSlot 2 "Manual/Manual Feed" + InputSlot 3 "Envelope/Envelope Feed" + + *MediaType 0 "Plain/Plain Paper" + MediaType 1 "Bond/Bond Paper" + MediaType 2 "Special/Special Paper" + MediaType 3 Transparency + MediaType 4 "Glossy/Glossy Paper" + } + + // HP LaserJet Series PCL 4/5 + { + ModelName "LaserJet Series PCL 4/5" + Attribute NickName "" "HP LaserJet Series PCL 4/5" + PCFileName "laserjet.ppd" + Throughput 8 + ModelNumber $HP_LASERJET + ColorDevice No + Attribute 1284DeviceID "" "MFG:HP;MDL:HP LaserJet;CMD:PCL;" + + UIConstraints "*Duplex *Option1 False" + UIConstraints "*PageSize A3 *InputSlot Envelope" + UIConstraints "*PageSize A4 *InputSlot Envelope" + UIConstraints "*PageSize A5 *InputSlot Envelope" + UIConstraints "*PageSize B5 *InputSlot Envelope" + UIConstraints "*PageSize Executive *InputSlot Envelope" + UIConstraints "*PageSize Legal *InputSlot Envelope" + UIConstraints "*PageSize Letter *InputSlot Envelope" + UIConstraints "*PageSize Tabloid *InputSlot Envelope" + + HWMargins 18 36 18 36 + *MediaSize Letter + MediaSize Legal + MediaSize Executive + MediaSize Tabloid + MediaSize A3 + MediaSize A4 + MediaSize A5 + MediaSize B5 + MediaSize EnvISOB5 + MediaSize Env10 + MediaSize EnvC5 + MediaSize EnvDL + MediaSize EnvMonarch + + Resolution k 1 0 0 0 150dpi + *Resolution k 1 0 0 0 300dpi + Resolution k 1 0 0 0 600dpi + + *InputSlot 0 "Default/Printer Default" + InputSlot 8 "Tray1/Tray 1" + InputSlot 1 "Tray2/Tray 2" + InputSlot 4 "Tray3/Tray 3" + InputSlot 5 "Tray4/Tray 4" + InputSlot 2 "Manual/Manual Feed" + InputSlot 3 "Envelope/Envelope Feed" + + Duplex Yes + Installable "Option1/Duplexer" + } +} + +// Intellitech IntelliBar Series Label Printer +{ + Manufacturer "Intellitech" + ModelName "IntelliBar Label Printer" + Attribute ShortNickName "" "IntelliBar Label Printer" + PCFileName "intelbar.ppd" + DriverType label + ModelNumber $INTELLITECH_PCL + Throughput 8 + ColorDevice No + + HWMargins 0 5.76 0 5.76 + VariablePaperSize Yes + MinSize 36 36 + MaxSize 630 7128 + + UIConstraints "*inPrintMode Standard *inCutInterval" + UIConstraints "*inPrintMode Tear *inCutInterval" + UIConstraints "*inPrintMode Standard *inTearInterval" + UIConstraints "*inPrintMode Cut *inTearInterval" + + *MediaSize w288h432 + + *Resolution k 1 0 0 0 300dpi + + Group "PrinterSettings/Printer Settings" + Option "inPrintDensity/Print Density" PickOne DocumentSetup 20.0 + *Choice "Default/Printer Default" "<>setpagedevice" + Choice "-15/-15" "<>setpagedevice" + Choice "-14/-14" "<>setpagedevice" + Choice "-13/-13" "<>setpagedevice" + Choice "-12/-12" "<>setpagedevice" + Choice "-11/-11" "<>setpagedevice" + Choice "-10/-10" "<>setpagedevice" + Choice "-9/-9" "<>setpagedevice" + Choice "-8/-8" "<>setpagedevice" + Choice "-7/-7" "<>setpagedevice" + Choice "-6/-6" "<>setpagedevice" + Choice "-5/-5" "<>setpagedevice" + Choice "-4/-4" "<>setpagedevice" + Choice "-3/-3" "<>setpagedevice" + Choice "-2/-2" "<>setpagedevice" + Choice "-1/-1" "<>setpagedevice" + Choice "0/0" "<>setpagedevice" + Choice "1/1" "<>setpagedevice" + Choice "2/2" "<>setpagedevice" + Choice "3/3" "<>setpagedevice" + Choice "4/4" "<>setpagedevice" + Choice "5/5" "<>setpagedevice" + Choice "6/6" "<>setpagedevice" + Choice "7/7" "<>setpagedevice" + Choice "8/8" "<>setpagedevice" + Choice "9/9" "<>setpagedevice" + Choice "10/10" "<>setpagedevice" + Choice "11/11" "<>setpagedevice" + Choice "12/12" "<>setpagedevice" + Choice "13/13" "<>setpagedevice" + Choice "14/14" "<>setpagedevice" + Choice "15/15" "<>setpagedevice" + Option "inPrintRate/Print Speed" PickOne DocumentSetup 20.0 + *Choice "Default/Printer Default" "<>setpagedevice" + Choice "15/15 mm/sec." "<>setpagedevice" + Choice "20/20 mm/sec." "<>setpagedevice" + Choice "30/30 mm/sec." "<>setpagedevice" + Choice "40/40 mm/sec." "<>setpagedevice" + Choice "60/60 mm/sec." "<>setpagedevice" + Choice "80/80 mm/sec." "<>setpagedevice" + Choice "100/100 mm/sec." "<>setpagedevice" + Choice "120/120 mm/sec." "<>setpagedevice" + Choice "150/150 mm/sec." "<>setpagedevice" + Choice "200/200 mm/sec." "<>setpagedevice" + Choice "250/250 mm/sec." "<>setpagedevice" + Choice "300/300 mm/sec." "<>setpagedevice" + Option "inPrintMode/Print Mode" PickOne DocumentSetup 20.0 + *Choice "Standard/Standard" "" + Choice "Tear/Tear" "" + Choice "Cut/Cut" "" + Option "inTearInterval/Print and Tear" PickOne DocumentSetup 20.0 + *Choice "None/Disabled" "<>setpagedevice" + Choice "1/Every Label" "<>setpagedevice" + Choice "2/Every 2 Labels" "<>setpagedevice" + Choice "3/Every 3 Labels" "<>setpagedevice" + Choice "4/Every 4 Labels" "<>setpagedevice" + Choice "5/Every 5 Labels" "<>setpagedevice" + Choice "6/Every 6 Labels" "<>setpagedevice" + Choice "7/Every 7 Labels" "<>setpagedevice" + Choice "8/Every 8 Labels" "<>setpagedevice" + Choice "9/Every 9 Labels" "<>setpagedevice" + Choice "10/Every 10 Labels" "<>setpagedevice" + Attribute CustominTearInterval True "<>setpagedevice" + Attribute ParamCustominTearInterval Interval "1 int 1 99" + Option "inCutInterval/Print and Cut" PickOne DocumentSetup 20.0 + *Choice "None/Disabled" "<>setpagedevice" + Choice "1/Every Label" "<>setpagedevice" + Choice "2/Every 2 Labels" "<>setpagedevice" + Choice "3/Every 3 Labels" "<>setpagedevice" + Choice "4/Every 4 Labels" "<>setpagedevice" + Choice "5/Every 5 Labels" "<>setpagedevice" + Choice "6/Every 6 Labels" "<>setpagedevice" + Choice "7/Every 7 Labels" "<>setpagedevice" + Choice "8/Every 8 Labels" "<>setpagedevice" + Choice "9/Every 9 Labels" "<>setpagedevice" + Choice "10/Every 10 Labels" "<>setpagedevice" + Attribute CustominCutInterval True "<>setpagedevice" + Attribute ParamCustominCutInterval Interval "1 int 1 99" +} + +// Oki +{ + Manufacturer "Oki" + DriverType epson + ManualCopies Yes + ColorDevice No + Throughput 1 + + HWMargins 18 18 18 18 + *MediaSize Letter + MediaSize Legal + MediaSize A4 + MediaSize FanFoldUS + + HWMargins 0 0 0 0 + VariablePaperSize Yes + MinSize 36 36 + MaxSize 1080 86400 + + // Oki 24-Pin Series + { + ModelName "24-Pin Series" + Attribute NickName "" "Oki 24-Pin Series" + PCFileName "okidat24.ppd" + ModelNumber $EPSON_24PIN + + Resolution k 1 8 0 0 60dpi + *Resolution k 1 8 0 0 120x60dpi + Resolution k 1 24 0 0 180dpi + Resolution k 1 24 0 0 360x180dpi + Resolution k 1 48 0 0 360dpi + } + + // Oki 9-Pin Series + { + ModelName "9-Pin Series" + Attribute NickName "" "Oki 9-Pin Series" + PCFileName "okidata9.ppd" + ModelNumber $EPSON_9PIN + ColorDevice No + + Resolution k 1 8 0 0 60x72dpi + *Resolution k 1 8 0 0 120x72dpi + Resolution k 1 8 0 0 240x72dpi + } +} + + +// Zebra +{ + Manufacturer "Zebra" + DriverType label + Throughput 8 + ColorDevice False + + // Zebra CPCL Label Printer + { + ModelName "CPCL Label Printer" + Attribute NickName "" "Zebra CPCL Label Printer" + PCFileName "zebracpl.ppd" + ModelNumber $ZEBRA_CPCL + + HWMargins 0 0 0 0 + MediaSize w144h72 + MediaSize w144h90 + MediaSize w144h144 + MediaSize w144h216 + MediaSize w209h72 + + HWMargins 0 0 1 0 + MediaSize w288h144 + MediaSize w288h216 + MediaSize w288h288 + *MediaSize w288h360 + MediaSize w288h432 + + VariablePaperSize Yes + HWMargins 0 0 1 0 + MinSize 36 36 + MaxSize 288 3600 + + *Resolution k 1 0 0 0 203dpi + + Group "General/General" + Option "zeMediaTracking/Media Tracking" PickOne AnySetup 20.0 + Choice "Continuous/Continuous" "" + *Choice "Web/Non-continuous (Web sensing)" "" + Choice "Mark/Non-continuous (Mark sensing)" "" + + Group "PrinterSettings/Printer Settings" + Option "Darkness" PickOne AnySetup 20.0 + *Choice "-1/Printer Default" "<>setpagedevice" + Choice "1/1" "<>setpagedevice" + Choice "2/2" "<>setpagedevice" + Choice "3/3" "<>setpagedevice" + Choice "4/4" "<>setpagedevice" + Choice "5/5" "<>setpagedevice" + Choice "6/6" "<>setpagedevice" + Choice "7/7" "<>setpagedevice" + Choice "8/8" "<>setpagedevice" + Choice "9/9" "<>setpagedevice" + Choice "10/10" "<>setpagedevice" + Choice "11/11" "<>setpagedevice" + Choice "12/12" "<>setpagedevice" + Choice "13/13" "<>setpagedevice" + Choice "14/14" "<>setpagedevice" + Choice "15/15" "<>setpagedevice" + Choice "16/16" "<>setpagedevice" + Choice "17/17" "<>setpagedevice" + Choice "18/18" "<>setpagedevice" + Choice "19/19" "<>setpagedevice" + Choice "20/20" "<>setpagedevice" + Choice "21/21" "<>setpagedevice" + Choice "22/22" "<>setpagedevice" + Choice "23/23" "<>setpagedevice" + Choice "24/24" "<>setpagedevice" + Choice "25/25" "<>setpagedevice" + Choice "26/26" "<>setpagedevice" + Choice "27/27" "<>setpagedevice" + Choice "28/28" "<>setpagedevice" + Choice "29/29" "<>setpagedevice" + Choice "30/30" "<>setpagedevice" + Option "zePrintRate/Print Rate" PickOne AnySetup 20.0 + *Choice "Default/Printer Default" "" + Choice "1/1 inch/sec." "" + Choice "2/2 inches/sec." "" + Choice "3/3 inches/sec." "" + Choice "4/4 inches/sec." "" + Option "zeTearOffPosition/Tear-Off Adjust Position" PickOne AnySetup 20.0 + *Choice "1000/Printer Default" "<>setpagedevice" + Choice "-120/-120" "<>setpagedevice" + Choice "-115/-115" "<>setpagedevice" + Choice "-110/-110" "<>setpagedevice" + Choice "-105/-105" "<>setpagedevice" + Choice "-100/-100" "<>setpagedevice" + Choice "-95/-95" "<>setpagedevice" + Choice "-90/-90" "<>setpagedevice" + Choice "-85/-85" "<>setpagedevice" + Choice "-80/-80" "<>setpagedevice" + Choice "-75/-75" "<>setpagedevice" + Choice "-70/-70" "<>setpagedevice" + Choice "-65/-65" "<>setpagedevice" + Choice "-60/-60" "<>setpagedevice" + Choice "-55/-55" "<>setpagedevice" + Choice "-50/-50" "<>setpagedevice" + Choice "-45/-45" "<>setpagedevice" + Choice "-40/-40" "<>setpagedevice" + Choice "-35/-35" "<>setpagedevice" + Choice "-30/-30" "<>setpagedevice" + Choice "-25/-25" "<>setpagedevice" + Choice "-20/-20" "<>setpagedevice" + Choice "-15/-15" "<>setpagedevice" + Choice "-10/-10" "<>setpagedevice" + Choice "-5/-5" "<>setpagedevice" + Choice "0/0" "<>setpagedevice" + Choice "5/5" "<>setpagedevice" + Choice "10/10" "<>setpagedevice" + Choice "15/15" "<>setpagedevice" + Choice "20/20" "<>setpagedevice" + Choice "25/25" "<>setpagedevice" + Choice "30/30" "<>setpagedevice" + Choice "35/35" "<>setpagedevice" + Choice "40/40" "<>setpagedevice" + Choice "45/45" "<>setpagedevice" + Choice "50/50" "<>setpagedevice" + Choice "55/55" "<>setpagedevice" + Choice "60/60" "<>setpagedevice" + Choice "65/65" "<>setpagedevice" + Choice "70/70" "<>setpagedevice" + Choice "75/75" "<>setpagedevice" + Choice "80/80" "<>setpagedevice" + Choice "85/85" "<>setpagedevice" + Choice "90/90" "<>setpagedevice" + Choice "95/95" "<>setpagedevice" + Choice "100/100" "<>setpagedevice" + Choice "105/105" "<>setpagedevice" + Choice "110/110" "<>setpagedevice" + Choice "115/115" "<>setpagedevice" + Choice "120/120" "<>setpagedevice" + Option "zeErrorReprint/Reprint After Error" PickOne AnySetup 20.0 + *Choice "Saved/Printer Default" "" + Choice "Always/Always" "" + Choice "Never/Never" "" + } + + // Zebra EPL1 Label Printer + { + ModelName "EPL1 Label Printer" + Attribute NickName "" "Zebra EPL1 Label Printer" + PCFileName "zebraep1.ppd" + ModelNumber $ZEBRA_EPL_LINE + + HWMargins 0 0 0 0 + VariablePaperSize Yes + MinSize 36 36 + MaxSize 288 3600 + + MediaSize w90h18 + MediaSize w90h162 + MediaSize w108h18 + MediaSize w108h36 + MediaSize w108h72 + MediaSize w108h144 + MediaSize w144h26 + MediaSize w144h36 + MediaSize w144h72 + MediaSize w144h90 + MediaSize w144h288 + MediaSize w144h396 + MediaSize w162h36 + MediaSize w162h90 + MediaSize w162h288 + MediaSize w162h396 + MediaSize w171h396 + MediaSize w180h72 + MediaSize w180h144 + MediaSize w198h90 + MediaSize w216h72 + MediaSize w216h90 + MediaSize w216h144 + MediaSize w216h216 + MediaSize w216h360 + MediaSize w234h144 + MediaSize w234h360 + MediaSize w234h396 + MediaSize w234h419 + MediaSize w234h563 + MediaSize w252h72 + MediaSize w288h72 + MediaSize w288h144 + MediaSize w288h180 + MediaSize w288h216 + MediaSize w288h288 + *MediaSize w288h360 + MediaSize w288h432 + MediaSize w288h468 + MediaSize w288h936 + + *Resolution k 1 0 0 0 203dpi + Resolution k 1 0 0 0 300dpi + Resolution k 1 0 0 0 600dpi + + Group "PrinterSettings/Printer Settings" + Option "Darkness" PickOne AnySetup 20.0 + *Choice "-1/Printer Default" "<>setpagedevice" + Choice "1/1" "<>setpagedevice" + Choice "2/2" "<>setpagedevice" + Choice "3/3" "<>setpagedevice" + Choice "4/4" "<>setpagedevice" + Choice "5/5" "<>setpagedevice" + Choice "6/6" "<>setpagedevice" + Choice "7/7" "<>setpagedevice" + Choice "8/8" "<>setpagedevice" + Choice "9/9" "<>setpagedevice" + Choice "10/10" "<>setpagedevice" + Choice "11/11" "<>setpagedevice" + Choice "12/12" "<>setpagedevice" + Choice "13/13" "<>setpagedevice" + Choice "14/14" "<>setpagedevice" + Choice "15/15" "<>setpagedevice" + Choice "16/16" "<>setpagedevice" + Choice "17/17" "<>setpagedevice" + Choice "18/18" "<>setpagedevice" + Choice "19/19" "<>setpagedevice" + Choice "20/20" "<>setpagedevice" + Choice "21/21" "<>setpagedevice" + Choice "22/22" "<>setpagedevice" + Choice "23/23" "<>setpagedevice" + Choice "24/24" "<>setpagedevice" + Choice "25/25" "<>setpagedevice" + Choice "26/26" "<>setpagedevice" + Choice "27/27" "<>setpagedevice" + Choice "28/28" "<>setpagedevice" + Choice "29/29" "<>setpagedevice" + Choice "30/30" "<>setpagedevice" + Option "zePrintRate/Print Rate" PickOne AnySetup 20.0 + *Choice "Default/Printer Default" "" + Choice "1/1 inch/sec." "" + Choice "1.5/1.5 inch/sec." "" + Choice "2/2 inches/sec." "" + Choice "2.5/2.5 inches/sec." "" + } + + // Zebra EPL2 Label Printer + { + ModelName "EPL2 Label Printer" + Attribute NickName "" "Zebra EPL2 Label Printer" + PCFileName "zebraep2.ppd" + ModelNumber $ZEBRA_EPL_PAGE + + HWMargins 0 0 0 0 + VariablePaperSize Yes + MinSize 36 36 + MaxSize 288 3600 + + MediaSize w90h18 + MediaSize w90h162 + MediaSize w108h18 + MediaSize w108h36 + MediaSize w108h72 + MediaSize w108h144 + MediaSize w144h26 + MediaSize w144h36 + MediaSize w144h72 + MediaSize w144h90 + MediaSize w144h288 + MediaSize w144h396 + MediaSize w162h36 + MediaSize w162h90 + MediaSize w162h288 + MediaSize w162h396 + MediaSize w171h396 + MediaSize w180h72 + MediaSize w180h144 + MediaSize w198h90 + MediaSize w216h72 + MediaSize w216h90 + MediaSize w216h144 + MediaSize w216h216 + MediaSize w216h360 + MediaSize w234h144 + MediaSize w234h360 + MediaSize w234h396 + MediaSize w234h419 + MediaSize w234h563 + MediaSize w252h72 + MediaSize w288h72 + MediaSize w288h144 + MediaSize w288h180 + MediaSize w288h216 + MediaSize w288h288 + *MediaSize w288h360 + MediaSize w288h432 + MediaSize w288h468 + MediaSize w288h936 + + *Resolution k 1 0 0 0 203dpi + Resolution k 1 0 0 0 300dpi + Resolution k 1 0 0 0 600dpi + + Group "General/General" + Option "MediaType/Media Type" PickOne AnySetup 20.0 + *Choice "Saved/Printer Default" "" + Choice "Thermal/Thermal Transfer Media" "<>setpagedevice" + Choice "Direct/Direct Thermal Media" "<>setpagedevice" + Group "PrinterSettings/Printer Settings" + Option "Darkness" PickOne AnySetup 20.0 + *Choice "-1/Printer Default" "<>setpagedevice" + Choice "1/1" "<>setpagedevice" + Choice "2/2" "<>setpagedevice" + Choice "3/3" "<>setpagedevice" + Choice "4/4" "<>setpagedevice" + Choice "5/5" "<>setpagedevice" + Choice "6/6" "<>setpagedevice" + Choice "7/7" "<>setpagedevice" + Choice "8/8" "<>setpagedevice" + Choice "9/9" "<>setpagedevice" + Choice "10/10" "<>setpagedevice" + Choice "11/11" "<>setpagedevice" + Choice "12/12" "<>setpagedevice" + Choice "13/13" "<>setpagedevice" + Choice "14/14" "<>setpagedevice" + Choice "15/15" "<>setpagedevice" + Choice "16/16" "<>setpagedevice" + Choice "17/17" "<>setpagedevice" + Choice "18/18" "<>setpagedevice" + Choice "19/19" "<>setpagedevice" + Choice "20/20" "<>setpagedevice" + Choice "21/21" "<>setpagedevice" + Choice "22/22" "<>setpagedevice" + Choice "23/23" "<>setpagedevice" + Choice "24/24" "<>setpagedevice" + Choice "25/25" "<>setpagedevice" + Choice "26/26" "<>setpagedevice" + Choice "27/27" "<>setpagedevice" + Choice "28/28" "<>setpagedevice" + Choice "29/29" "<>setpagedevice" + Choice "30/30" "<>setpagedevice" + Option "zePrintRate/Print Rate" PickOne AnySetup 20.0 + *Choice "Default/Printer Default" "" + Choice "1/1 inch/sec." "" + Choice "1.5/1.5 inch/sec." "" + Choice "2/2 inches/sec." "" + Choice "2.5/2.5 inches/sec." "" + Choice "3/3 inches/sec." "" + Choice "4/4 inches/sec." "" + Choice "5/5 inches/sec." "" + Choice "6/6 inches/sec." "" + } + + // Zebra ZPL Label Printer + { + ModelName "ZPL Label Printer" + Attribute NickName "" "Zebra ZPL Label Printer" + PCFileName "zebra.ppd" + ModelNumber $ZEBRA_ZPL + + HWMargins 0 0 0 0 + VariablePaperSize Yes + MinSize 36 36 + MaxSize 576 3600 + + MediaSize w90h18 + MediaSize w90h162 + MediaSize w108h18 + MediaSize w108h36 + MediaSize w108h72 + MediaSize w108h144 + MediaSize w144h26 + MediaSize w144h36 + MediaSize w144h72 + MediaSize w144h90 + MediaSize w144h288 + MediaSize w144h396 + MediaSize w162h36 + MediaSize w162h90 + MediaSize w162h288 + MediaSize w162h396 + MediaSize w171h396 + MediaSize w180h72 + MediaSize w180h144 + MediaSize w198h90 + MediaSize w216h72 + MediaSize w216h90 + MediaSize w216h144 + MediaSize w216h216 + MediaSize w216h360 + MediaSize w234h144 + MediaSize w234h360 + MediaSize w234h396 + MediaSize w234h419 + MediaSize w234h563 + MediaSize w252h72 + MediaSize w288h72 + MediaSize w288h144 + MediaSize w288h180 + MediaSize w288h216 + MediaSize w288h288 + *MediaSize w288h360 + MediaSize w288h432 + MediaSize w288h468 + MediaSize w288h936 + MediaSize w432h72 + MediaSize w432h144 + MediaSize w432h216 + MediaSize w432h288 + MediaSize w432h360 + MediaSize w432h432 + MediaSize w432h468 + MediaSize w576h72 + MediaSize w576h144 + MediaSize w576h216 + MediaSize w576h288 + MediaSize w576h360 + MediaSize w576h432 + MediaSize w576h468 + + *Resolution k 1 0 0 0 203dpi + Resolution k 1 0 0 0 300dpi + Resolution k 1 0 0 0 600dpi + + Group "General/General" + Option "zeMediaTracking/Media Tracking" PickOne AnySetup 20.0 + Choice "Continuous/Continuous" "" + *Choice "Web/Non-continuous (Web sensing)" "" + Choice "Mark/Non-continuous (Mark sensing)" "" + Option "MediaType/Media Type" PickOne AnySetup 20.0 + *Choice "Saved/Printer Default" "" + Choice "Thermal/Thermal Transfer Media" "<>setpagedevice" + Choice "Direct/Direct Thermal Media" "<>setpagedevice" + Group "PrinterSettings/Printer Settings" + Option "Darkness" PickOne AnySetup 20.0 + *Choice "-1/Printer Default" "<>setpagedevice" + Choice "1/1" "<>setpagedevice" + Choice "2/2" "<>setpagedevice" + Choice "3/3" "<>setpagedevice" + Choice "4/4" "<>setpagedevice" + Choice "5/5" "<>setpagedevice" + Choice "6/6" "<>setpagedevice" + Choice "7/7" "<>setpagedevice" + Choice "8/8" "<>setpagedevice" + Choice "9/9" "<>setpagedevice" + Choice "10/10" "<>setpagedevice" + Choice "11/11" "<>setpagedevice" + Choice "12/12" "<>setpagedevice" + Choice "13/13" "<>setpagedevice" + Choice "14/14" "<>setpagedevice" + Choice "15/15" "<>setpagedevice" + Choice "16/16" "<>setpagedevice" + Choice "17/17" "<>setpagedevice" + Choice "18/18" "<>setpagedevice" + Choice "19/19" "<>setpagedevice" + Choice "20/20" "<>setpagedevice" + Choice "21/21" "<>setpagedevice" + Choice "22/22" "<>setpagedevice" + Choice "23/23" "<>setpagedevice" + Choice "24/24" "<>setpagedevice" + Choice "25/25" "<>setpagedevice" + Choice "26/26" "<>setpagedevice" + Choice "27/27" "<>setpagedevice" + Choice "28/28" "<>setpagedevice" + Choice "29/29" "<>setpagedevice" + Choice "30/30" "<>setpagedevice" + Option "zePrintRate/Print Rate" PickOne AnySetup 20.0 + *Choice "Default/Printer Default" "" + Choice "1/1 inch/sec." "" + Choice "2/2 inches/sec." "" + Choice "3/3 inches/sec." "" + Choice "4/4 inches/sec." "" + Choice "5/5 inches/sec." "" + Choice "6/6 inches/sec." "" + Choice "7/7 inches/sec." "" + Choice "8/8 inches/sec." "" + Choice "9/9 inches/sec." "" + Choice "10/10 inches/sec." "" + Choice "11/11 inches/sec." "" + Choice "12/12 inches/sec." "" + Option "zeLabelTop/Label Top" PickOne AnySetup 20.0 + *Choice "200/Printer Default" "<>setpagedevice" + Choice "-120/-120" "<>setpagedevice" + Choice "-115/-115" "<>setpagedevice" + Choice "-110/-110" "<>setpagedevice" + Choice "-105/-105" "<>setpagedevice" + Choice "-100/-100" "<>setpagedevice" + Choice "-95/-95" "<>setpagedevice" + Choice "-90/-90" "<>setpagedevice" + Choice "-85/-85" "<>setpagedevice" + Choice "-80/-80" "<>setpagedevice" + Choice "-75/-75" "<>setpagedevice" + Choice "-70/-70" "<>setpagedevice" + Choice "-65/-65" "<>setpagedevice" + Choice "-60/-60" "<>setpagedevice" + Choice "-55/-55" "<>setpagedevice" + Choice "-50/-50" "<>setpagedevice" + Choice "-45/-45" "<>setpagedevice" + Choice "-40/-40" "<>setpagedevice" + Choice "-35/-35" "<>setpagedevice" + Choice "-30/-30" "<>setpagedevice" + Choice "-25/-25" "<>setpagedevice" + Choice "-20/-20" "<>setpagedevice" + Choice "-15/-15" "<>setpagedevice" + Choice "-10/-10" "<>setpagedevice" + Choice "-5/-5" "<>setpagedevice" + Choice "0/0" "<>setpagedevice" + Choice "5/5" "<>setpagedevice" + Choice "10/10" "<>setpagedevice" + Choice "15/15" "<>setpagedevice" + Choice "20/20" "<>setpagedevice" + Choice "25/25" "<>setpagedevice" + Choice "30/30" "<>setpagedevice" + Choice "35/35" "<>setpagedevice" + Choice "40/40" "<>setpagedevice" + Choice "45/45" "<>setpagedevice" + Choice "50/50" "<>setpagedevice" + Choice "55/55" "<>setpagedevice" + Choice "60/60" "<>setpagedevice" + Choice "65/65" "<>setpagedevice" + Choice "70/70" "<>setpagedevice" + Choice "75/75" "<>setpagedevice" + Choice "80/80" "<>setpagedevice" + Choice "85/85" "<>setpagedevice" + Choice "90/90" "<>setpagedevice" + Choice "95/95" "<>setpagedevice" + Choice "100/100" "<>setpagedevice" + Choice "105/105" "<>setpagedevice" + Choice "110/110" "<>setpagedevice" + Choice "115/115" "<>setpagedevice" + Choice "120/120" "<>setpagedevice" + Option "zePrintMode/Print Mode" PickOne AnySetup 20.0 + *Choice "Saved/Printer Default" "" + Choice "Tear/Tear-Off" "" + Choice "Peel/Peel-Off" "" + Choice "Rewind/Rewind" "" + Choice "Applicator/Applicator" "" + Choice "Cutter/Cutter" "" + Option "zeTearOffPosition/Tear-Off Adjust Position" PickOne AnySetup 20.0 + *Choice "1000/Printer Default" "<>setpagedevice" + Choice "-120/-120" "<>setpagedevice" + Choice "-115/-115" "<>setpagedevice" + Choice "-110/-110" "<>setpagedevice" + Choice "-105/-105" "<>setpagedevice" + Choice "-100/-100" "<>setpagedevice" + Choice "-95/-95" "<>setpagedevice" + Choice "-90/-90" "<>setpagedevice" + Choice "-85/-85" "<>setpagedevice" + Choice "-80/-80" "<>setpagedevice" + Choice "-75/-75" "<>setpagedevice" + Choice "-70/-70" "<>setpagedevice" + Choice "-65/-65" "<>setpagedevice" + Choice "-60/-60" "<>setpagedevice" + Choice "-55/-55" "<>setpagedevice" + Choice "-50/-50" "<>setpagedevice" + Choice "-45/-45" "<>setpagedevice" + Choice "-40/-40" "<>setpagedevice" + Choice "-35/-35" "<>setpagedevice" + Choice "-30/-30" "<>setpagedevice" + Choice "-25/-25" "<>setpagedevice" + Choice "-20/-20" "<>setpagedevice" + Choice "-15/-15" "<>setpagedevice" + Choice "-10/-10" "<>setpagedevice" + Choice "-5/-5" "<>setpagedevice" + Choice "0/0" "<>setpagedevice" + Choice "5/5" "<>setpagedevice" + Choice "10/10" "<>setpagedevice" + Choice "15/15" "<>setpagedevice" + Choice "20/20" "<>setpagedevice" + Choice "25/25" "<>setpagedevice" + Choice "30/30" "<>setpagedevice" + Choice "35/35" "<>setpagedevice" + Choice "40/40" "<>setpagedevice" + Choice "45/45" "<>setpagedevice" + Choice "50/50" "<>setpagedevice" + Choice "55/55" "<>setpagedevice" + Choice "60/60" "<>setpagedevice" + Choice "65/65" "<>setpagedevice" + Choice "70/70" "<>setpagedevice" + Choice "75/75" "<>setpagedevice" + Choice "80/80" "<>setpagedevice" + Choice "85/85" "<>setpagedevice" + Choice "90/90" "<>setpagedevice" + Choice "95/95" "<>setpagedevice" + Choice "100/100" "<>setpagedevice" + Choice "105/105" "<>setpagedevice" + Choice "110/110" "<>setpagedevice" + Choice "115/115" "<>setpagedevice" + Choice "120/120" "<>setpagedevice" + Option "zeErrorReprint/Reprint After Error" PickOne AnySetup 20.0 + *Choice "Saved/Printer Default" "" + Choice "Always/Always" "" + Choice "Never/Never" "" + } +} + +// +// End of "$Id$". +// diff --git a/ppdc/testcatalog.cxx b/ppdc/testcatalog.cxx new file mode 100644 index 0000000000..a9e2e86556 --- /dev/null +++ b/ppdc/testcatalog.cxx @@ -0,0 +1,63 @@ +// +// "$Id$" +// +// Test program for message catalog class. +// +// Copyright 2008 by Apple Inc. +// +// These coded instructions, statements, and computer programs are the +// property of Apple Inc. and are protected by Federal copyright +// law. Distribution and use rights are outlined in the file "LICENSE.txt" +// which should have been included with this file. If this file is +// file is missing or damaged, see the license at "http://www.cups.org/". +// +// Contents: +// +// main() - Open a message catalog +// + +// +// Include necessary headers... +// + +#include "ppdc-private.h" + + +// +// 'main()' - Open a message catalog +// + +int // O - Exit status +main(int argc, // I - Number of command-line arguments + char *argv[]) // I - Command-line arguments +{ + ppdcCatalog *catalog; // Message catalog + ppdcMessage *m; // Current message + + + if (argc != 2) + { + puts("Usage: testcatalog filename"); + return (1); + } + + // Scan the command-line... + catalog = new ppdcCatalog(NULL, argv[1]); + + printf("%s: %d messages\n", argv[1], catalog->messages->count); + + for (m = (ppdcMessage *)catalog->messages->first(); + m; + m = (ppdcMessage *)catalog->messages->next()) + printf("%s: %s\n", m->id->value, m->string->value); + + catalog->release(); + + // Return with no errors. + return (0); +} + + +// +// End of "$Id$". +// diff --git a/scheduler/Dependencies b/scheduler/Dependencies new file mode 100644 index 0000000000..8fc424f99f --- /dev/null +++ b/scheduler/Dependencies @@ -0,0 +1,287 @@ +auth.o: auth.c cupsd.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h +banners.o: banners.c cupsd.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h \ + ../cups/dir.h +cert.o: cert.c cupsd.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h +classes.o: classes.c cupsd.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h +client.o: client.c cupsd.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h +conf.o: conf.c cupsd.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h +dirsvc.o: dirsvc.c cupsd.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h +env.o: env.c cupsd.h ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h ../cups/file-private.h mime.h sysman.h \ + statbuf.h cert.h auth.h client.h policy.h printers.h classes.h job.h \ + conf.h banners.h dirsvc.h network.h subscriptions.h +file.o: file.c cupsd.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h \ + ../cups/dir.h +main.o: main.c cupsd.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h +ipp.o: ipp.c cupsd.h ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h ../cups/file-private.h mime.h sysman.h \ + statbuf.h cert.h auth.h client.h policy.h printers.h classes.h job.h \ + conf.h banners.h dirsvc.h network.h subscriptions.h +listen.o: listen.c cupsd.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h +job.o: job.c cupsd.h ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h ../cups/file-private.h mime.h sysman.h \ + statbuf.h cert.h auth.h client.h policy.h printers.h classes.h job.h \ + conf.h banners.h dirsvc.h network.h subscriptions.h ../cups/backend.h \ + ../cups/dir.h +log.o: log.c cupsd.h ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h ../cups/file-private.h mime.h sysman.h \ + statbuf.h cert.h auth.h client.h policy.h printers.h classes.h job.h \ + conf.h banners.h dirsvc.h network.h subscriptions.h +network.o: network.c ../cups/http-private.h ../config.h ../cups/http.h \ + ../cups/versioning.h ../cups/array.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/ipp.h cupsd.h ../cups/cups-private.h \ + ../cups/cups.h ../cups/file.h ../cups/language.h \ + ../cups/string-private.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h +policy.o: policy.c cupsd.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h +printers.o: printers.c cupsd.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h \ + ../cups/dir.h +process.o: process.c cupsd.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h +quotas.o: quotas.c cupsd.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h +select.o: select.c cupsd.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h +server.o: server.c ../cups/http-private.h ../config.h ../cups/http.h \ + ../cups/versioning.h ../cups/array.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/ipp.h cupsd.h ../cups/cups-private.h \ + ../cups/cups.h ../cups/file.h ../cups/language.h \ + ../cups/string-private.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h +statbuf.o: statbuf.c cupsd.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h +subscriptions.o: subscriptions.c cupsd.h ../cups/cups-private.h \ + ../cups/cups.h ../cups/file.h ../cups/versioning.h ../cups/ipp.h \ + ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h ../cups/debug-private.h \ + ../cups/ppd-private.h ../cups/ppd.h ../cups/pwg-private.h \ + ../cups/http-private.h ../cups/md5-private.h ../cups/ipp-private.h \ + ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h ../cups/file-private.h mime.h sysman.h \ + statbuf.h cert.h auth.h client.h policy.h printers.h classes.h job.h \ + conf.h banners.h dirsvc.h network.h subscriptions.h +sysman.o: sysman.c cupsd.h ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h sysman.h statbuf.h cert.h auth.h client.h policy.h printers.h \ + classes.h job.h conf.h banners.h dirsvc.h network.h subscriptions.h +filter.o: filter.c ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/versioning.h mime.h ../cups/array.h \ + ../cups/ipp.h ../cups/http.h ../cups/file.h +mime.o: mime.c ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/versioning.h ../cups/dir.h \ + mime-private.h mime.h ../cups/array.h ../cups/ipp.h ../cups/http.h \ + ../cups/file.h +type.o: type.c ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/versioning.h mime.h ../cups/array.h \ + ../cups/ipp.h ../cups/http.h ../cups/file.h +cupsfilter.o: cupsfilter.c ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/file-private.h \ + mime.h +cups-deviced.o: cups-deviced.c util.h ../cups/array-private.h \ + ../cups/array.h ../cups/versioning.h ../cups/file-private.h \ + ../cups/cups-private.h ../cups/cups.h ../cups/file.h ../cups/ipp.h \ + ../cups/http.h ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h ../cups/dir.h +cups-exec.o: cups-exec.c ../cups/string-private.h ../config.h +cups-lpd.o: cups-lpd.c ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h +testlpd.o: testlpd.c ../cups/cups.h ../cups/file.h ../cups/versioning.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/string-private.h ../config.h +testmime.o: testmime.c ../cups/string-private.h ../config.h ../cups/dir.h \ + ../cups/versioning.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/cups.h ../cups/file.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/ppd.h ../cups/pwg-private.h \ + mime.h +testspeed.o: testspeed.c ../cups/string-private.h ../config.h \ + ../cups/cups.h ../cups/file.h ../cups/versioning.h ../cups/ipp.h \ + ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/debug-private.h +testsub.o: testsub.c ../cups/cups.h ../cups/file.h ../cups/versioning.h \ + ../cups/ipp.h ../cups/http.h ../cups/array.h ../cups/language.h \ + ../cups/debug-private.h ../cups/string-private.h ../config.h \ + ../cups/ipp-private.h +util.o: util.c util.h ../cups/array-private.h ../cups/array.h \ + ../cups/versioning.h ../cups/file-private.h ../cups/cups-private.h \ + ../cups/cups.h ../cups/file.h ../cups/ipp.h ../cups/http.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +cups-driverd.o: cups-driverd.cxx util.h ../cups/array-private.h \ + ../cups/array.h ../cups/versioning.h ../cups/file-private.h \ + ../cups/cups-private.h ../cups/cups.h ../cups/file.h ../cups/ipp.h \ + ../cups/http.h ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h ../cups/dir.h ../ppdc/ppdc.h diff --git a/scheduler/Makefile b/scheduler/Makefile new file mode 100644 index 0000000000..08a01f55a7 --- /dev/null +++ b/scheduler/Makefile @@ -0,0 +1,551 @@ +# +# "$Id$" +# +# Scheduler Makefile for CUPS. +# +# Copyright 2007-2012 by Apple Inc. +# Copyright 1997-2007 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +include ../Makedefs + +CUPSDOBJS = \ + auth.o \ + banners.o \ + cert.o \ + classes.o \ + client.o \ + conf.o \ + dirsvc.o \ + env.o \ + file.o \ + main.o \ + ipp.o \ + listen.o \ + job.o \ + log.o \ + network.o \ + policy.o \ + printers.o \ + process.o \ + quotas.o \ + select.o \ + server.o \ + statbuf.o \ + subscriptions.o \ + sysman.o +LIBOBJS = \ + filter.o \ + mime.o \ + type.o +COBJS = \ + $(CUPSDOBJS) \ + $(LIBOBJS) \ + cupsfilter.o \ + cups-deviced.o \ + cups-exec.o \ + cups-lpd.o \ + testlpd.o \ + testmime.o \ + testspeed.o \ + testsub.o \ + util.o +CXXOBJS = \ + cups-driverd.o +OBJS = \ + $(COBJS) \ + $(CXXOBJS) +LIBTARGETS = \ + $(LIBCUPSMIME) \ + libcupsmime.a + +UNITTARGETS = \ + testlpd \ + testmime \ + testspeed \ + testsub + +PROGRAMS = \ + cupsd \ + cupsfilter \ + cups-deviced \ + cups-driverd \ + cups-exec \ + cups-lpd + +TARGETS = \ + $(LIBTARGETS) \ + $(PROGRAMS) + + +# +# Make everything... +# + +all: $(TARGETS) + + +# +# Make library targets... +# + +libs: $(LIBTARGETS) + + +# +# Make unit tests... +# + +unittests: $(UNITTARGETS) + + +# +# Clean all object files... +# + +clean: + $(RM) $(OBJS) + $(RM) $(TARGETS) $(UNITTARGETS) convert + $(RM) libcupsmime.so libcupsmime.sl libcupsmime.dylib + + +# +# Update dependencies (without system header dependencies...) +# + +depend: + $(CC) -MM $(ALL_CFLAGS) $(COBJS:.o=.c) >Dependencies + $(CXX) -MM $(ALL_CXXFLAGS) $(CXXOBJS:.o=.cxx) >>Dependencies + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: + echo Creating $(SERVERBIN)/driver... + $(INSTALL_DIR) -m 755 $(SERVERBIN)/driver + echo Creating $(SERVERROOT)... + $(INSTALL_DIR) -m 755 -g $(CUPS_GROUP) $(SERVERROOT) + echo Creating $(SERVERROOT)/interfaces... + $(INSTALL_DIR) -m 755 -g $(CUPS_GROUP) $(SERVERROOT)/interfaces + echo Creating $(SERVERROOT)/ppd... + $(INSTALL_DIR) -m 755 -g $(CUPS_GROUP) $(SERVERROOT)/ppd + if test "x`uname`" != xDarwin; then \ + echo Creating $(SERVERROOT)/ssl...; \ + $(INSTALL_DIR) -m 700 -g $(CUPS_GROUP) $(SERVERROOT)/ssl; \ + fi + if test "$(STATEDIR)" != "$(SERVERROOT)"; then \ + echo Creating $(STATEDIR)...; \ + $(INSTALL_DIR) -m 755 $(STATEDIR); \ + fi + echo Creating $(STATEDIR)/certs... + $(INSTALL_DIR) -m 511 -o $(CUPS_USER) -g $(CUPS_PRIMARY_SYSTEM_GROUP) \ + $(STATEDIR)/certs + echo Creating $(LOGDIR)... + $(INSTALL_DIR) -m 755 $(LOGDIR) + echo Creating $(REQUESTS)... + $(INSTALL_DIR) -m 710 -g $(CUPS_GROUP) $(REQUESTS) + echo Creating $(REQUESTS)/tmp... + $(INSTALL_DIR) -m 1770 -g $(CUPS_GROUP) $(REQUESTS)/tmp + echo Creating $(CACHEDIR)... + $(INSTALL_DIR) -m 775 -g $(CUPS_GROUP) $(CACHEDIR) + if test "x$(INITDIR)" != x; then \ + echo Installing init scripts...; \ + $(INSTALL_DIR) -m 755 $(BUILDROOT)$(INITDIR)/init.d; \ + $(INSTALL_SCRIPT) cups.sh $(BUILDROOT)$(INITDIR)/init.d/cups; \ + for level in $(RCLEVELS); do \ + $(INSTALL_DIR) -m 755 $(BUILDROOT)$(INITDIR)/rc$${level}.d; \ + $(LN) ../init.d/cups $(BUILDROOT)$(INITDIR)/rc$${level}.d/S$(RCSTART)cups; \ + if test `uname` = HP-UX; then \ + level=`expr $$level - 1`; \ + $(INSTALL_DIR) -m 755 $(BUILDROOT)$(INITDIR)/rc$${level}.d; \ + fi; \ + $(LN) ../init.d/cups $(BUILDROOT)$(INITDIR)/rc$${level}.d/K$(RCSTOP)cups; \ + done; \ + $(INSTALL_DIR) -m 755 $(BUILDROOT)$(INITDIR)/rc0.d; \ + $(LN) ../init.d/cups $(BUILDROOT)$(INITDIR)/rc0.d/K$(RCSTOP)cups; \ + fi + if test "x$(INITDIR)" = x -a "x$(INITDDIR)" != x; then \ + $(INSTALL_DIR) $(BUILDROOT)$(INITDDIR); \ + if test "$(INITDDIR)" = "/System/Library/LaunchDaemons"; then \ + echo Installing LaunchDaemons configuration files...; \ + $(INSTALL_DATA) org.cups.cupsd.plist $(BUILDROOT)$(DEFAULT_LAUNCHD_CONF); \ + $(INSTALL_DATA) org.cups.cups-lpd.plist $(BUILDROOT)/System/Library/LaunchDaemons; \ + else \ + echo Installing RC script...; \ + $(INSTALL_SCRIPT) cups.sh $(BUILDROOT)$(INITDDIR)/cups; \ + fi \ + fi + if test "x$(SMFMANIFESTDIR)" != x; then \ + echo Installing SMF manifest in $(SMFMANIFESTDIR)...;\ + $(INSTALL_DIR) $(BUILDROOT)/$(SMFMANIFESTDIR); \ + $(INSTALL_SCRIPT) cups.xml $(BUILDROOT)$(SMFMANIFESTDIR)/cups.xml; \ + fi + if test "x$(XINETD)" != x; then \ + echo Installing xinetd configuration file for cups-lpd...; \ + $(INSTALL_DIR) -m 755 $(BUILDROOT)$(XINETD); \ + $(INSTALL_DATA) cups-lpd.xinetd $(BUILDROOT)$(XINETD)/cups-lpd; \ + fi + + +# +# Install programs... +# + +install-exec: + echo Installing programs in $(SBINDIR)... + $(INSTALL_DIR) -m 755 $(SBINDIR) + $(INSTALL_BIN) -m 500 cupsd $(SBINDIR) + $(INSTALL_BIN) cupsfilter $(SBINDIR) + echo Installing programs in $(SERVERBIN)/daemon... + $(INSTALL_DIR) -m 755 $(SERVERBIN) + $(INSTALL_DIR) -m 755 $(SERVERBIN)/daemon + $(INSTALL_BIN) cups-deviced $(SERVERBIN)/daemon + $(INSTALL_BIN) cups-driverd $(SERVERBIN)/daemon + $(INSTALL_BIN) cups-exec $(SERVERBIN)/daemon + $(INSTALL_BIN) cups-lpd $(SERVERBIN)/daemon + if test "x$(SYMROOT)" != "x"; then \ + $(INSTALL_DIR) $(SYMROOT); \ + for file in $(PROGRAMS); do \ + cp $$file $(SYMROOT); \ + done \ + fi + + +# +# Install headers... +# + +install-headers: + echo Installing header files in $(INCLUDEDIR)/cups... + $(INSTALL_DIR) -m 755 $(INCLUDEDIR)/cups + $(INSTALL_DATA) mime.h $(INCLUDEDIR)/cups + + +# +# Install libraries... +# + +install-libs: $(INSTALLSTATIC) + echo Installing libraries in $(LIBDIR)... + $(INSTALL_DIR) -m 755 $(LIBDIR) + $(INSTALL_LIB) $(LIBCUPSMIME) $(LIBDIR) + if test $(LIBCUPSMIME) = "libcupsmime.so.1" -o $(LIBCUPSMIME) = "libcupsmime.sl.1"; then \ + $(RM) $(LIBDIR)/`basename $(LIBCUPSMIME) .1`; \ + $(LN) $(LIBCUPSMIME) $(LIBDIR)/`basename $(LIBCUPSMIME) .1`; \ + fi + if test $(LIBCUPSMIME) = "libcupsmime.1.dylib"; then \ + $(RM) $(LIBDIR)/libcupsmime.dylib; \ + $(LN) $(LIBCUPSMIME) $(LIBDIR)/libcupsmime.dylib; \ + fi + if test "x$(SYMROOT)" != "x"; then \ + $(INSTALL_DIR) $(SYMROOT); \ + cp $(LIBCUPSMIME) $(SYMROOT); \ + fi + +installstatic: + $(INSTALL_DIR) -m 755 $(LIBDIR) + $(INSTALL_LIB) -m 755 libcupsmime.a $(LIBDIR) + $(RANLIB) $(LIBDIR)/libcupsmime.a + $(CHMOD) 555 $(LIBDIR)/libcupsmime.a + + +# +# Uninstall the scheduler... +# + +uninstall: + $(RM) $(SBINDIR)/cupsd + $(RM) $(SBINDIR)/cupsfilter + $(RM) $(SERVERBIN)/daemon/cups-deviced + $(RM) $(SERVERBIN)/daemon/cups-driverd + $(RM) $(SERVERBIN)/daemon/cups-exec + $(RM) $(SERVERBIN)/daemon/cups-lpd + $(RM) $(BUILDROOT)/System/Library/Printers/Libraries/convert + -$(RMDIR) $(STATEDIR)/certs + -$(RMDIR) $(STATEDIR) + -$(RMDIR) $(SERVERROOT)/ppd + -$(RMDIR) $(SERVERROOT)/interfaces + -$(RMDIR) $(SERVERROOT) + -$(RMDIR) $(SERVERBIN)/driver + -$(RMDIR) $(SERVERBIN)/daemon + -$(RMDIR) $(SERVERBIN) + -$(RMDIR) $(SBINDIR) + -$(RMDIR) $(REQUESTS)/tmp + -$(RMDIR) $(REQUESTS) + -$(RMDIR) $(LOGDIR) + -$(RMDIR) $(CACHEDIR) + $(RM) $(LIBDIR)/libcupsmime.1.dylib + $(RM) $(LIBDIR)/libcupsmime.a + $(RM) $(LIBDIR)/libcupsmime.dylib + $(RM) $(LIBDIR)/libcupsmime_s.a + $(RM) $(LIBDIR)/libcupsmime.sl + $(RM) $(LIBDIR)/libcupsmime.sl.1 + $(RM) $(LIBDIR)/libcupsmime.so + $(RM) $(LIBDIR)/libcupsmime.so.1 + -$(RMDIR) $(LIBDIR) + $(RM) $(INCLUDEDIR)/cups/mime.h + -$(RMDIR) $(INCLUDEDIR)/cups + echo Uninstalling startup script... + if test "x$(INITDIR)" != x; then \ + $(RM) $(BUILDROOT)$(INITDIR)/init.d/cups; \ + $(RMDIR) $(BUILDROOT)$(INITDIR)/init.d; \ + $(RM) $(BUILDROOT)$(INITDIR)/rc0.d/K00cups; \ + $(RMDIR) $(BUILDROOT)$(INITDIR)/rc0.d; \ + $(RM) $(BUILDROOT)$(INITDIR)/rc2.d/S99cups; \ + $(RMDIR) $(BUILDROOT)$(INITDIR)/rc2.d; \ + $(RM) $(BUILDROOT)$(INITDIR)/rc3.d/S99cups; \ + $(RMDIR) $(BUILDROOT)$(INITDIR)/rc3.d; \ + $(RM) $(BUILDROOT)$(INITDIR)/rc5.d/S99cups; \ + $(RMDIR) $(BUILDROOT)$(INITDIR)/rc5.d; \ + fi + if test "x$(INITDIR)" = x -a "x$(INITDDIR)" != x; then \ + if test "$(INITDDIR)" = "/System/Library/StartupItems/PrintingServices"; then \ + $(RM) $(BUILDROOT)$(INITDDIR)/PrintingServices; \ + $(RM) $(BUILDROOT)$(INITDDIR)/StartupParameters.plist; \ + $(RM) $(BUILDROOT)$(INITDDIR)/Resources/English.lproj/Localizable.strings; \ + $(RMDIR) $(BUILDROOT)$(INITDDIR)/Resources/English.lproj; \ + elif test "$(INITDDIR)" = "/System/Library/LaunchDaemons"; then \ + $(RM) $(BUILDROOT)$(INITDDIR)/org.cups.cupsd.plist; \ + $(RM) $(BUILDROOT)$(INITDDIR)/org.cups.cups-lpd.plist; \ + $(RMDIR) $(BUILDROOT)/System/Library/StartupItems/PrintingServices; \ + else \ + $(INSTALL_SCRIPT) init/cups.sh $(BUILDROOT)$(INITDDIR)/cups; \ + fi \ + $(RMDIR) $(BUILDROOT)$(INITDDIR); \ + fi + if test "x$(SMFMANIFESTDIR)" != x; then \ + echo Uninstalling SMF manifest in $(SMFMANIFESTDIR)...;\ + $(RM) $(BUILDROOT)$(SMFMANIFESTDIR)/cups.xml; \ + fi + if test "x$(XINETD)" != x; then \ + echo Uninstalling xinetd configuration file for cups-lpd...; \ + $(RM) $(BUILDROOT)$(XINETD)/cups-lpd; \ + fi + + +# +# Automatic API help files... +# + +apihelp: + mxmldoc --section "Programming" \ + --title "MIME API" \ + --css ../doc/cups-printable.css \ + --header api-mime.header --intro api-mime.shtml \ + mime.h $(LIBOBJS:.o=.c) >../doc/help/api-mime.html + mxmldoc --tokens help/api-mime.html api-mime.xml >../doc/help/api-mime.tokens + $(RM) api-mime.xml + +framedhelp: + mxmldoc --framed api-mime \ + --section "Programming" \ + --title "MIME API" \ + --css ../doc/cups-printable.css \ + --header api-mime.header --intro api-mime.shtml \ + mime.h $(LIBOBJS:.o=.c) + + +# +# Make the scheduler executable, "cupsd". +# + +cupsd: $(CUPSDOBJS) $(LIBCUPSMIME) ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o cupsd $(CUPSDOBJS) -L. -lcupsmime \ + $(LIBZ) $(SSLLIBS) $(LIBSLP) $(LIBLDAP) $(PAMLIBS) \ + $(LIBPAPER) $(LIBMALLOC) $(SERVERLIBS) $(DNSSDLIBS) $(LIBS) \ + $(LIBGSSAPI) $(LIBWRAP) + +cupsd-static: $(CUPSDOBJS) libcupsmime.a ../cups/$(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(LDFLAGS) -o cupsd-static $(CUPSDOBJS) libcupsmime.a \ + $(LIBZ) $(SSLLIBS) $(LIBSLP) $(LIBLDAP) $(PAMLIBS) \ + ../cups/$(LIBCUPSSTATIC) $(COMMONLIBS) $(LIBZ) $(LIBPAPER) \ + $(LIBMALLOC) $(SERVERLIBS) $(DNSSDLIBS) $(LIBGSSAPI) \ + $(LIBWRAP) + + +# +# Make the cupsfilter utility. +# + +cupsfilter: cupsfilter.o $(LIBCUPSMIME) ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o cupsfilter cupsfilter.o -L. -lcupsmime $(LIBS) + $(RM) convert + $(LN) cupsfilter convert + + +# +# Make the device daemon, "cups-deviced". +# + +cups-deviced: cups-deviced.o util.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o cups-deviced cups-deviced.o util.o $(LIBS) + + +# +# Make the driver daemon, "cups-driverd". +# + +cups-driverd: cups-driverd.o util.o ../cups/$(LIBCUPS) ../ppdc/$(LIBCUPSPPDC) + echo Linking $@... + $(CXX) $(LDFLAGS) -o cups-driverd cups-driverd.o util.o \ + -L../ppdc -lcupsppdc $(LIBS) + + +# +# Make the sandbox execution helper, "cups-exec". +# + +cups-exec: cups-exec.o + echo Linking $@... + $(CC) $(LDFLAGS) -o cups-exec cups-exec.o $(LIBS) + + +# +# Make the line printer daemon, "cups-lpd". +# + +cups-lpd: cups-lpd.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o cups-lpd cups-lpd.o $(LIBS) + + +# +# libcupsmime.so.1, libcupsmime.sl.1 +# + +libcupsmime.so.1 libcupsmime.sl.1: $(LIBOBJS) + echo Linking $@... + $(DSO) $(ARCHFLAGS) $(DSOFLAGS) -o $@ $(LIBOBJS) $(LIBS) + $(RM) `basename $@ .1` + $(LN) $@ `basename $@ .1` + + +# +# libcupsmime.1.dylib +# + +libcupsmime.1.dylib: $(LIBOBJS) libcupsmime.exp + echo Linking $@... + $(DSO) $(ARCHFLAGS) $(DSOFLAGS) -o $@ \ + -install_name $(libdir)/$@ \ + -current_version 1.0.0 \ + -compatibility_version 1.0.0 \ + -exported_symbols_list libcupsmime.exp \ + $(LIBOBJS) $(LIBS) + $(RM) libcupsmime.dylib + $(LN) $@ libcupsmime.dylib + + +# +# libcupsmime_s.a +# + +libcupsmime_s.a: $(LIBOBJS) + echo Creating $@... + $(DSO) $(DSOFLAGS) -o libcupsmime_s.o $(LIBOBJS) $(LIBS) + $(RM) $@ + $(AR) $(ARFLAGS) $@ libcupsmime_s.o + + +# +# libcupsmime.la +# + +libcupsmime.la: $(LIBOBJS) + echo Linking $@... + $(CC) $(ARCHFLAGS) $(DSOFLAGS) -o $@ $(LIBOBJS:.o=.lo) -rpath $(LIBDIR) \ + -version-info 1:0 $(LIBS) + + +# +# libcupsmime.a +# + +libcupsmime.a: $(LIBOBJS) + echo Archiving $@... + $(RM) $@ + $(AR) $(ARFLAGS) $@ $(LIBOBJS) + $(RANLIB) $@ + + +# +# Make the test program, "testlpd". +# + +testlpd: testlpd.o ../cups/$(LIBCUPSSTATIC) cups-lpd + echo Linking $@... + $(CC) $(LDFLAGS) -o testlpd testlpd.o ../cups/$(LIBCUPSSTATIC) \ + $(COMMONLIBS) $(LIBZ) $(SSLLIBS) $(DNSSDLIBS) $(LIBGSSAPI) + + +# +# testmime +# + +testmime: testmime.o libcupsmime.a ../cups/$(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(ARCHFLAGS) $(LDFLAGS) -o $@ testmime.o libcupsmime.a \ + ../cups/$(LIBCUPSSTATIC) $(COMMONLIBS) $(LIBZ) $(SSLLIBS) \ + $(DNSSDLIBS) $(LIBGSSAPI) + echo Running MIME tests... + ./testmime + + +# +# Make the test program, "testspeed". +# + +testspeed: testspeed.o ../cups/$(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(LDFLAGS) -o testspeed testspeed.o ../cups/$(LIBCUPSSTATIC) \ + $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) $(LIBGSSAPI) + + +# +# Make the test program, "testsub". +# + +testsub: testsub.o ../cups/$(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(LDFLAGS) -o testsub testsub.o ../cups/$(LIBCUPSSTATIC) \ + $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) $(LIBGSSAPI) + + +# +# Lines of code computation... +# + +sloc: + echo "cupsd: \c" + sloccount $(CUPSDOBJS:.o=.c) $(LIBOBJS:.o=.c) cups-driverd.cxx cups-lpd.c 2>/dev/null | grep "Total Physical" | awk '{print $$9}' + + +# +# Dependencies... +# + +include Dependencies + + +# +# End of "$Id$". +# diff --git a/scheduler/api-mime.header b/scheduler/api-mime.header new file mode 100644 index 0000000000..8d5db2ad6f --- /dev/null +++ b/scheduler/api-mime.header @@ -0,0 +1,34 @@ + + +

MIME API

+ +
+ + + + + + + + + + + + + + + + +
Headercups/mime.h
Library-lcupsmime
See AlsoProgramming: Introduction to CUPS Programming
diff --git a/scheduler/api-mime.shtml b/scheduler/api-mime.shtml new file mode 100644 index 0000000000..7d3125fadf --- /dev/null +++ b/scheduler/api-mime.shtml @@ -0,0 +1,17 @@ + + +

Overview

+ +

The MIME API provides file typing and conversion services for CUPS.

diff --git a/scheduler/auth.c b/scheduler/auth.c new file mode 100644 index 0000000000..1774d11c26 --- /dev/null +++ b/scheduler/auth.c @@ -0,0 +1,2599 @@ +/* + * "$Id$" + * + * Authorization routines for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * This file contains Kerberos support code, copyright 2006 by + * Jelmer Vernooij. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdAddIPMask() - Add an IP address authorization mask. + * cupsdAddLocation() - Add a location for authorization. + * cupsdAddName() - Add a name to a location... + * cupsdAddNameMask() - Add a host or interface name authorization + * mask. + * cupsdAuthorize() - Validate any authorization credentials. + * cupsdCheckAccess() - Check whether the given address is allowed to + * access a location. + * cupsdCheckAuth() - Check authorization masks. + * cupsdCheckGroup() - Check for a user's group membership. + * cupsdCopyLocation() - Make a copy of a location... + * cupsdDeleteAllLocations() - Free all memory used for location + * authorization. + * cupsdFindBest() - Find the location entry that best matches the + * resource. + * cupsdFindLocation() - Find the named location. + * cupsdFreeLocation() - Free all memory used by a location. + * cupsdIsAuthorized() - Check to see if the user is authorized... + * cupsdNewLocation() - Create a new location for authorization. + * check_authref() - Check if an authorization services reference + * has the supplied right. + * compare_locations() - Compare two locations. + * copy_authmask() - Copy function for auth masks. + * cups_crypt() - Encrypt the password using the DES or MD5 + * algorithms, as needed. + * free_authmask() - Free function for auth masks. + * get_md5_password() - Get an MD5 password. + * pam_func() - PAM conversation function. + * to64() - Base64-encode an integer value... + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" +#include +#ifdef HAVE_SHADOW_H +# include +#endif /* HAVE_SHADOW_H */ +#ifdef HAVE_CRYPT_H +# include +#endif /* HAVE_CRYPT_H */ +#if HAVE_LIBPAM +# ifdef HAVE_PAM_PAM_APPL_H +# include +# else +# include +# endif /* HAVE_PAM_PAM_APPL_H */ +#endif /* HAVE_LIBPAM */ +#ifdef HAVE_USERSEC_H +# include +#endif /* HAVE_USERSEC_H */ +#ifdef HAVE_MEMBERSHIP_H +# include +#endif /* HAVE_MEMBERSHIP_H */ +#ifdef HAVE_AUTHORIZATION_H +# include +# ifdef HAVE_SECBASEPRIV_H +# include +# else +extern const char *cssmErrorString(int error); +# endif /* HAVE_SECBASEPRIV_H */ +#endif /* HAVE_AUTHORIZATION_H */ +#ifdef HAVE_SYS_PARAM_H +# include +#endif /* HAVE_SYS_PARAM_H */ +#ifdef HAVE_SYS_UCRED_H +# include +typedef struct xucred cupsd_ucred_t; +# define CUPSD_UCRED_UID(c) (c).cr_uid +#else +typedef struct ucred cupsd_ucred_t; +# define CUPSD_UCRED_UID(c) (c).uid +#endif /* HAVE_SYS_UCRED_H */ +#ifdef HAVE_KRB5_IPC_CLIENT_SET_TARGET_UID +/* Not in public headers... */ +extern void krb5_ipc_client_set_target_uid(uid_t); +extern void krb5_ipc_client_clear_target(void); +#endif /* HAVE_KRB5_IPC_CLIENT_SET_TARGET_UID */ + + +/* + * Local functions... + */ + +#ifdef HAVE_AUTHORIZATION_H +static int check_authref(cupsd_client_t *con, const char *right); +#endif /* HAVE_AUTHORIZATION_H */ +static int compare_locations(cupsd_location_t *a, + cupsd_location_t *b); +static cupsd_authmask_t *copy_authmask(cupsd_authmask_t *am, void *data); +#if !HAVE_LIBPAM && !defined(HAVE_USERSEC_H) +static char *cups_crypt(const char *pw, const char *salt); +#endif /* !HAVE_LIBPAM && !HAVE_USERSEC_H */ +static void free_authmask(cupsd_authmask_t *am, void *data); +static char *get_md5_password(const char *username, + const char *group, char passwd[33]); +#if HAVE_LIBPAM +static int pam_func(int, const struct pam_message **, + struct pam_response **, void *); +#elif !defined(HAVE_USERSEC_H) +static void to64(char *s, unsigned long v, int n); +#endif /* HAVE_LIBPAM */ + + +/* + * Local structures... + */ + +#if HAVE_LIBPAM +typedef struct cupsd_authdata_s /**** Authentication data ****/ +{ + char username[33], /* Username string */ + password[33]; /* Password string */ +} cupsd_authdata_t; +#endif /* HAVE_LIBPAM */ + + +/* + * Local globals... + */ + +#if defined(__hpux) && HAVE_LIBPAM +static cupsd_authdata_t *auth_data; /* Current client being authenticated */ +#endif /* __hpux && HAVE_LIBPAM */ + + +/* + * 'cupsdAddIPMask()' - Add an IP address authorization mask. + */ + +int /* O - 1 on success, 0 on failure */ +cupsdAddIPMask( + cups_array_t **masks, /* IO - Masks array (created as needed) */ + const unsigned address[4], /* I - IP address */ + const unsigned netmask[4]) /* I - IP netmask */ +{ + cupsd_authmask_t temp; /* New host/domain mask */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdAddIPMask(masks=%p(%p), address=%x:%x:%x:%x, " + "netmask=%x:%x:%x:%x)", + masks, *masks, + address[0], address[1], address[2], address[3], + netmask[0], netmask[1], netmask[2], netmask[3]); + + temp.type = CUPSD_AUTH_IP; + memcpy(temp.mask.ip.address, address, sizeof(temp.mask.ip.address)); + memcpy(temp.mask.ip.netmask, netmask, sizeof(temp.mask.ip.netmask)); + + /* + * Create the masks array as needed and add... + */ + + if (!*masks) + *masks = cupsArrayNew3(NULL, NULL, NULL, 0, + (cups_acopy_func_t)copy_authmask, + (cups_afree_func_t)free_authmask); + + return (cupsArrayAdd(*masks, &temp)); +} + + +/* + * 'cupsdAddLocation()' - Add a location for authorization. + */ + +void +cupsdAddLocation(cupsd_location_t *loc) /* I - Location to add */ +{ + /* + * Make sure the locations array is created... + */ + + if (!Locations) + Locations = cupsArrayNew3((cups_array_func_t)compare_locations, NULL, + (cups_ahash_func_t)NULL, 0, + (cups_acopy_func_t)NULL, + (cups_afree_func_t)cupsdFreeLocation); + + if (Locations) + { + cupsArrayAdd(Locations, loc); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdAddLocation: Added location \"%s\"", + loc->location ? loc->location : "(null)"); + } +} + + +/* + * 'cupsdAddName()' - Add a name to a location... + */ + +void +cupsdAddName(cupsd_location_t *loc, /* I - Location to add to */ + char *name) /* I - Name to add */ +{ + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdAddName(loc=%p, name=\"%s\")", + loc, name); + + if (!loc->names) + loc->names = cupsArrayNew3(NULL, NULL, NULL, 0, + (cups_acopy_func_t)_cupsStrAlloc, + (cups_afree_func_t)_cupsStrFree); + + if (!cupsArrayAdd(loc->names, name)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to duplicate name for location %s: %s", + loc->location ? loc->location : "nil", strerror(errno)); + return; + } +} + + +/* + * 'cupsdAddNameMask()' - Add a host or interface name authorization mask. + */ + +int /* O - 1 on success, 0 on failure */ +cupsdAddNameMask(cups_array_t **masks, /* IO - Masks array (created as needed) */ + char *name) /* I - Host or interface name */ +{ + cupsd_authmask_t temp; /* New host/domain mask */ + char ifname[32], /* Interface name */ + *ifptr; /* Pointer to end of name */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdAddNameMask(masks=%p(%p), name=\"%s\")", + masks, *masks, name); + + if (!_cups_strcasecmp(name, "@LOCAL")) + { + /* + * Deny *interface*... + */ + + temp.type = CUPSD_AUTH_INTERFACE; + temp.mask.name.name = (char *)"*"; + } + else if (!_cups_strncasecmp(name, "@IF(", 4)) + { + /* + * Deny *interface*... + */ + + strlcpy(ifname, name + 4, sizeof(ifname)); + + ifptr = ifname + strlen(ifname) - 1; + + if (ifptr >= ifname && *ifptr == ')') + { + ifptr --; + *ifptr = '\0'; + } + + temp.type = CUPSD_AUTH_INTERFACE; + temp.mask.name.name = ifname; + } + else + { + /* + * Deny name... + */ + + if (*name == '*') + name ++; + + temp.type = CUPSD_AUTH_NAME; + temp.mask.name.name = (char *)name; + } + + /* + * Set the name length... + */ + + temp.mask.name.length = strlen(temp.mask.name.name); + + /* + * Create the masks array as needed and add... + */ + + if (!*masks) + *masks = cupsArrayNew3(NULL, NULL, NULL, 0, + (cups_acopy_func_t)copy_authmask, + (cups_afree_func_t)free_authmask); + + return (cupsArrayAdd(*masks, &temp)); +} + + +/* + * 'cupsdAuthorize()' - Validate any authorization credentials. + */ + +void +cupsdAuthorize(cupsd_client_t *con) /* I - Client connection */ +{ + int type; /* Authentication type */ + const char *authorization; /* Pointer into Authorization string */ + char *ptr, /* Pointer into string */ + username[256], /* Username string */ + password[33]; /* Password string */ + cupsd_cert_t *localuser; /* Certificate username */ + char nonce[HTTP_MAX_VALUE], /* Nonce value from client */ + md5[33], /* MD5 password */ + basicmd5[33]; /* MD5 of Basic password */ + static const char * const states[] = /* HTTP client states... */ + { + "WAITING", + "OPTIONS", + "GET", + "GET", + "HEAD", + "POST", + "POST", + "POST", + "PUT", + "PUT", + "DELETE", + "TRACE", + "CLOSE", + "STATUS" + }; + + + /* + * Locate the best matching location so we know what kind of + * authentication to expect... + */ + + con->best = cupsdFindBest(con->uri, con->http.state); + con->type = CUPSD_AUTH_NONE; + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] con->uri=\"%s\", con->best=%p(%s)", + con->http.fd, con->uri, con->best, + con->best ? con->best->location : ""); + + if (con->best && con->best->type != CUPSD_AUTH_NONE) + { + if (con->best->type == CUPSD_AUTH_DEFAULT) + type = cupsdDefaultAuthType(); + else + type = con->best->type; + } + else + type = cupsdDefaultAuthType(); + + /* + * Decode the Authorization string... + */ + + authorization = httpGetField(&con->http, HTTP_FIELD_AUTHORIZATION); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "[Client %d] Authorization=\"%s\"", + con->http.fd, authorization); + + username[0] = '\0'; + password[0] = '\0'; + +#ifdef HAVE_GSSAPI + con->gss_uid = 0; +#endif /* HAVE_GSSAPI */ + +#ifdef HAVE_AUTHORIZATION_H + if (con->authref) + { + AuthorizationFree(con->authref, kAuthorizationFlagDefaults); + con->authref = NULL; + } +#endif /* HAVE_AUTHORIZATION_H */ + + if (!*authorization) + { + /* + * No authorization data provided, return early... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] No authentication data provided.", + con->http.fd); + return; + } +#ifdef HAVE_AUTHORIZATION_H + else if (!strncmp(authorization, "AuthRef ", 8) && + !_cups_strcasecmp(con->http.hostname, "localhost")) + { + OSStatus status; /* Status */ + int authlen; /* Auth string length */ + AuthorizationItemSet *authinfo; /* Authorization item set */ + + /* + * Get the Authorization Services data... + */ + + authorization += 8; + while (isspace(*authorization & 255)) + authorization ++; + + authlen = sizeof(nonce); + httpDecode64_2(nonce, &authlen, authorization); + + if (authlen != kAuthorizationExternalFormLength) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] External Authorization reference size is " + "incorrect.", con->http.fd); + return; + } + + if ((status = AuthorizationCreateFromExternalForm( + (AuthorizationExternalForm *)nonce, &con->authref)) != 0) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] AuthorizationCreateFromExternalForm " + "returned %d (%s)", con->http.fd, (int)status, + cssmErrorString(status)); + return; + } + + username[0] = '\0'; + + if (!AuthorizationCopyInfo(con->authref, kAuthorizationEnvironmentUsername, + &authinfo)) + { + if (authinfo->count == 1 && authinfo->items[0].value && + authinfo->items[0].valueLength >= 2) + { + strlcpy(username, authinfo->items[0].value, sizeof(username)); + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Authorized as \"%s\" using AuthRef", + con->http.fd, username); + } + + AuthorizationFreeItemSet(authinfo); + } + + if (!username[0]) + { + /* + * No username in AuthRef, grab username using peer credentials... + */ + + struct passwd *pwd; /* Password entry for this user */ + cupsd_ucred_t peercred; /* Peer credentials */ + socklen_t peersize; /* Size of peer credentials */ + + peersize = sizeof(peercred); + + if (getsockopt(con->http.fd, 0, LOCAL_PEERCRED, &peercred, &peersize)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Unable to get peer credentials - %s", + con->http.fd, strerror(errno)); + return; + } + + if ((pwd = getpwuid(CUPSD_UCRED_UID(peercred))) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Unable to find UID %d for peer " + "credentials.", con->http.fd, + (int)CUPSD_UCRED_UID(peercred)); + return; + } + + strlcpy(username, pwd->pw_name, sizeof(username)); + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Authorized as \"%s\" using " + "AuthRef + PeerCred", con->http.fd, username); + } + + con->type = CUPSD_AUTH_BASIC; + } +#endif /* HAVE_AUTHORIZATION_H */ +#if defined(SO_PEERCRED) && defined(AF_LOCAL) + else if (!strncmp(authorization, "PeerCred ", 9) && + con->http.hostaddr->addr.sa_family == AF_LOCAL) + { + /* + * Use peer credentials from domain socket connection... + */ + + struct passwd *pwd; /* Password entry for this user */ + cupsd_ucred_t peercred; /* Peer credentials */ + socklen_t peersize; /* Size of peer credentials */ +#ifdef HAVE_AUTHORIZATION_H + const char *name; /* Authorizing name */ + + for (name = (char *)cupsArrayFirst(con->best->names); + name; + name = (char *)cupsArrayNext(con->best->names)) + if (!_cups_strncasecmp(name, "@AUTHKEY(", 9) || + !_cups_strcasecmp(name, "@SYSTEM")) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] PeerCred authentication not allowed for " + "resource.", con->http.fd); + return; + } +#endif /* HAVE_AUTHORIZATION_H */ + + if ((pwd = getpwnam(authorization + 9)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] User \"%s\" does not exist.", con->http.fd, + authorization + 9); + return; + } + + peersize = sizeof(peercred); + +# ifdef __APPLE__ + if (getsockopt(con->http.fd, 0, LOCAL_PEERCRED, &peercred, &peersize)) +# else + if (getsockopt(con->http.fd, SOL_SOCKET, SO_PEERCRED, &peercred, &peersize)) +# endif /* __APPLE__ */ + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Unable to get peer credentials - %s", + con->http.fd, strerror(errno)); + return; + } + + if (pwd->pw_uid != CUPSD_UCRED_UID(peercred)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Invalid peer credentials for \"%s\" - got " + "%d, expected %d!", con->http.fd, authorization + 9, + CUPSD_UCRED_UID(peercred), pwd->pw_uid); +# ifdef HAVE_SYS_UCRED_H + cupsdLogMessage(CUPSD_LOG_DEBUG, "[Client %d] cr_version=%d", + con->http.fd, peercred.cr_version); + cupsdLogMessage(CUPSD_LOG_DEBUG, "[Client %d] cr_uid=%d", + con->http.fd, peercred.cr_uid); + cupsdLogMessage(CUPSD_LOG_DEBUG, "[Client %d] cr_ngroups=%d", + con->http.fd, peercred.cr_ngroups); + cupsdLogMessage(CUPSD_LOG_DEBUG, "[Client %d] cr_groups[0]=%d", + con->http.fd, peercred.cr_groups[0]); +# endif /* HAVE_SYS_UCRED_H */ + return; + } + + strlcpy(username, authorization + 9, sizeof(username)); + +# ifdef HAVE_GSSAPI + con->gss_uid = CUPSD_UCRED_UID(peercred); +# endif /* HAVE_GSSAPI */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Authorized as %s using PeerCred", con->http.fd, + username); + + con->type = CUPSD_AUTH_BASIC; + } +#endif /* SO_PEERCRED && AF_LOCAL */ + else if (!strncmp(authorization, "Local", 5) && + !_cups_strcasecmp(con->http.hostname, "localhost")) + { + /* + * Get Local certificate authentication data... + */ + + authorization += 5; + while (isspace(*authorization & 255)) + authorization ++; + + if ((localuser = cupsdFindCert(authorization)) != NULL) + { + strlcpy(username, localuser->username, sizeof(username)); + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Authorized as %s using Local", con->http.fd, + username); + } + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Local authentication certificate not found.", + con->http.fd); + return; + } + +#ifdef HAVE_GSSAPI + if (localuser->ccache) + con->type = CUPSD_AUTH_NEGOTIATE; + else +#endif /* HAVE_GSSAPI */ + con->type = CUPSD_AUTH_BASIC; + } + else if (!strncmp(authorization, "Basic", 5)) + { + /* + * Get the Basic authentication data... + */ + + int userlen; /* Username:password length */ + + + authorization += 5; + while (isspace(*authorization & 255)) + authorization ++; + + userlen = sizeof(username); + httpDecode64_2(username, &userlen, authorization); + + /* + * Pull the username and password out... + */ + + if ((ptr = strchr(username, ':')) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "[Client %d] Missing Basic password.", + con->http.fd); + return; + } + + *ptr++ = '\0'; + + if (!username[0]) + { + /* + * Username must not be empty... + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, "[Client %d] Empty Basic username.", + con->http.fd); + return; + } + + if (!*ptr) + { + /* + * Password must not be empty... + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, "[Client %d] Empty Basic password.", + con->http.fd); + return; + } + + strlcpy(password, ptr, sizeof(password)); + + /* + * Validate the username and password... + */ + + switch (type) + { + default : + case CUPSD_AUTH_BASIC : + { +#if HAVE_LIBPAM + /* + * Only use PAM to do authentication. This supports MD5 + * passwords, among other things... + */ + + pam_handle_t *pamh; /* PAM authentication handle */ + int pamerr; /* PAM error code */ + struct pam_conv pamdata;/* PAM conversation data */ + cupsd_authdata_t data; /* Authentication data */ + + + strlcpy(data.username, username, sizeof(data.username)); + strlcpy(data.password, password, sizeof(data.password)); + +# if defined(__sun) || defined(__hpux) + pamdata.conv = (int (*)(int, struct pam_message **, + struct pam_response **, + void *))pam_func; +# else + pamdata.conv = pam_func; +# endif /* __sun || __hpux */ + pamdata.appdata_ptr = &data; + +# ifdef __hpux + /* + * Workaround for HP-UX bug in pam_unix; see pam_func() below for + * more info... + */ + + auth_data = &data; +# endif /* __hpux */ + + pamerr = pam_start("cups", username, &pamdata, &pamh); + if (pamerr != PAM_SUCCESS) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] pam_start() returned %d (%s)", + con->http.fd, pamerr, pam_strerror(pamh, pamerr)); + return; + } + +# ifdef HAVE_PAM_SET_ITEM +# ifdef PAM_RHOST + pamerr = pam_set_item(pamh, PAM_RHOST, con->http.hostname); + if (pamerr != PAM_SUCCESS) + cupsdLogMessage(CUPSD_LOG_WARN, + "[Client %d] pam_set_item(PAM_RHOST) " + "returned %d (%s)", con->http.fd, pamerr, + pam_strerror(pamh, pamerr)); +# endif /* PAM_RHOST */ + +# ifdef PAM_TTY + pamerr = pam_set_item(pamh, PAM_TTY, "cups"); + if (pamerr != PAM_SUCCESS) + cupsdLogMessage(CUPSD_LOG_WARN, + "[Client %d] pam_set_item(PAM_TTY) " + "returned %d (%s)!", con->http.fd, pamerr, + pam_strerror(pamh, pamerr)); +# endif /* PAM_TTY */ +# endif /* HAVE_PAM_SET_ITEM */ + + pamerr = pam_authenticate(pamh, PAM_SILENT); + if (pamerr != PAM_SUCCESS) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] pam_authenticate() returned %d (%s)", + con->http.fd, pamerr, pam_strerror(pamh, pamerr)); + pam_end(pamh, 0); + return; + } + +# ifdef HAVE_PAM_SETCRED + pamerr = pam_setcred(pamh, PAM_ESTABLISH_CRED | PAM_SILENT); + if (pamerr != PAM_SUCCESS) + cupsdLogMessage(CUPSD_LOG_WARN, + "[Client %d] pam_setcred() returned %d (%s)", + con->http.fd, pamerr, + pam_strerror(pamh, pamerr)); +# endif /* HAVE_PAM_SETCRED */ + + pamerr = pam_acct_mgmt(pamh, PAM_SILENT); + if (pamerr != PAM_SUCCESS) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] pam_acct_mgmt() returned %d (%s)", + con->http.fd, pamerr, pam_strerror(pamh, pamerr)); + pam_end(pamh, 0); + return; + } + + pam_end(pamh, PAM_SUCCESS); + +#elif defined(HAVE_USERSEC_H) + /* + * Use AIX authentication interface... + */ + + char *authmsg; /* Authentication message */ + int reenter; /* ??? */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] AIX authenticate of username \"%s\"", + con->http.fd, username); + + reenter = 1; + if (authenticate(username, password, &reenter, &authmsg) != 0) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Unable to authenticate username " + "\"%s\": %s", con->http.fd, username, + strerror(errno)); + return; + } + +#else + /* + * Use normal UNIX password file-based authentication... + */ + + char *pass; /* Encrypted password */ + struct passwd *pw; /* User password data */ +# ifdef HAVE_SHADOW_H + struct spwd *spw; /* Shadow password data */ +# endif /* HAVE_SHADOW_H */ + + + pw = getpwnam(username); /* Get the current password */ + endpwent(); /* Close the password file */ + + if (!pw) + { + /* + * No such user... + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Unknown username \"%s\".", + con->http.fd, username); + return; + } + +# ifdef HAVE_SHADOW_H + spw = getspnam(username); + endspent(); + + if (!spw && !strcmp(pw->pw_passwd, "x")) + { + /* + * Don't allow blank passwords! + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Username \"%s\" has no shadow " + "password.", con->http.fd, username); + return; + } + + if (spw && !spw->sp_pwdp[0] && !pw->pw_passwd[0]) +# else + if (!pw->pw_passwd[0]) +# endif /* HAVE_SHADOW_H */ + { + /* + * Don't allow blank passwords! + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, + "Username \"%s\" has no password.", con->http.fd, + username); + return; + } + + /* + * OK, the password isn't blank, so compare with what came from the + * client... + */ + + pass = cups_crypt(password, pw->pw_passwd); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] pw_passwd=\"%s\", crypt=\"%s\"", + con->http.fd, pw->pw_passwd, pass); + + if (!pass || strcmp(pw->pw_passwd, pass)) + { +# ifdef HAVE_SHADOW_H + if (spw) + { + pass = cups_crypt(password, spw->sp_pwdp); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] sp_pwdp=\"%s\", crypt=\"%s\"", + con->http.fd, spw->sp_pwdp, pass); + + if (pass == NULL || strcmp(spw->sp_pwdp, pass)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Authentication failed for user " + "\"%s\".", con->http.fd, username); + return; + } + } + else +# endif /* HAVE_SHADOW_H */ + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Authentication failed for user " + "\"%s\".", con->http.fd, username); + return; + } + } +#endif /* HAVE_LIBPAM */ + } + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Authorized as %s using Basic", + con->http.fd, username); + break; + + case CUPSD_AUTH_BASICDIGEST : + /* + * Do Basic authentication with the Digest password file... + */ + + if (!get_md5_password(username, NULL, md5)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Unknown MD5 username \"%s\".", + con->http.fd, username); + return; + } + + httpMD5(username, "CUPS", password, basicmd5); + + if (strcmp(md5, basicmd5)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Authentication failed for \"%s\".", + con->http.fd, username); + return; + } + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Authorized as %s using BasicDigest", + con->http.fd, username); + break; + } + + con->type = type; + } + else if (!strncmp(authorization, "Digest", 6)) + { + /* + * Get the username, password, and nonce from the Digest attributes... + */ + + if (!httpGetSubField2(&(con->http), HTTP_FIELD_AUTHORIZATION, "username", + username, sizeof(username)) || !username[0]) + { + /* + * Username must not be empty... + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Empty or missing Digest username.", + con->http.fd); + return; + } + + if (!httpGetSubField2(&(con->http), HTTP_FIELD_AUTHORIZATION, "response", + password, sizeof(password)) || !password[0]) + { + /* + * Password must not be empty... + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Empty or missing Digest password.", + con->http.fd); + return; + } + + if (!httpGetSubField(&(con->http), HTTP_FIELD_AUTHORIZATION, "nonce", + nonce)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] No nonce value for Digest authentication.", + con->http.fd); + return; + } + + if (strcmp(con->http.hostname, nonce)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Bad nonce value, expected \"%s\", " + "got \"%s\".", con->http.fd, con->http.hostname, nonce); + return; + } + + /* + * Validate the username and password... + */ + + if (!get_md5_password(username, NULL, md5)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Unknown MD5 username \"%s\".", + con->http.fd, username); + return; + } + + httpMD5Final(nonce, states[con->http.state], con->uri, md5); + + if (strcmp(md5, password)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Authentication failed for \"%s\".", + con->http.fd, username); + return; + } + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Authorized as %s using Digest", con->http.fd, + username); + + con->type = CUPSD_AUTH_DIGEST; + } +#ifdef HAVE_GSSAPI + else if (!strncmp(authorization, "Negotiate", 9)) + { + int len; /* Length of authorization string */ + gss_ctx_id_t context; /* Authorization context */ + OM_uint32 major_status, /* Major status code */ + minor_status; /* Minor status code */ + gss_buffer_desc input_token = GSS_C_EMPTY_BUFFER, + /* Input token from string */ + output_token = GSS_C_EMPTY_BUFFER; + /* Output token for username */ + gss_name_t client_name; /* Client name */ + + +# ifdef __APPLE__ + /* + * If the weak-linked GSSAPI/Kerberos library is not present, don't try + * to use it... + */ + + if (gss_init_sec_context == NULL) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "[Client %d] GSSAPI/Kerberos authentication failed " + "because the Kerberos framework is not present.", + con->http.fd); + return; + } +# endif /* __APPLE__ */ + + /* + * Find the start of the Kerberos input token... + */ + + authorization += 9; + while (isspace(*authorization & 255)) + authorization ++; + + if (!*authorization) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] No authentication data specified.", + con->http.fd); + return; + } + + /* + * Decode the authorization string to get the input token... + */ + + len = strlen(authorization); + input_token.value = malloc(len); + input_token.value = httpDecode64_2(input_token.value, &len, + authorization); + input_token.length = len; + + /* + * Accept the input token to get the authorization info... + */ + + context = GSS_C_NO_CONTEXT; + client_name = GSS_C_NO_NAME; + major_status = gss_accept_sec_context(&minor_status, + &context, + ServerCreds, + &input_token, + GSS_C_NO_CHANNEL_BINDINGS, + &client_name, + NULL, + &output_token, + NULL, + NULL, + NULL); + + if (output_token.length > 0) + gss_release_buffer(&minor_status, &output_token); + + if (GSS_ERROR(major_status)) + { + cupsdLogGSSMessage(CUPSD_LOG_DEBUG, major_status, minor_status, + "[Client %d] Error accepting GSSAPI security context", + con->http.fd); + + if (context != GSS_C_NO_CONTEXT) + gss_delete_sec_context(&minor_status, &context, GSS_C_NO_BUFFER); + return; + } + + con->have_gss = 1; + + /* + * Get the username associated with the client's credentials... + */ + + if (major_status == GSS_S_CONTINUE_NEEDED) + cupsdLogGSSMessage(CUPSD_LOG_DEBUG, major_status, minor_status, + "[Client %d] Credentials not complete", con->http.fd); + else if (major_status == GSS_S_COMPLETE) + { + major_status = gss_display_name(&minor_status, client_name, + &output_token, NULL); + + if (GSS_ERROR(major_status)) + { + cupsdLogGSSMessage(CUPSD_LOG_DEBUG, major_status, minor_status, + "[Client %d] Error getting username", con->http.fd); + gss_release_name(&minor_status, &client_name); + gss_delete_sec_context(&minor_status, &context, GSS_C_NO_BUFFER); + return; + } + + strlcpy(username, output_token.value, sizeof(username)); + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Authorized as %s using Negotiate", + con->http.fd, username); + + gss_release_name(&minor_status, &client_name); + gss_release_buffer(&minor_status, &output_token); + + con->type = CUPSD_AUTH_NEGOTIATE; + } + + gss_delete_sec_context(&minor_status, &context, GSS_C_NO_BUFFER); + +# if defined(SO_PEERCRED) && defined(AF_LOCAL) + /* + * Get the client's UID if we are printing locally - that allows a backend + * to run as the correct user to get Kerberos credentials of its own. + */ + + if (_httpAddrFamily(con->http.hostaddr) == AF_LOCAL) + { + cupsd_ucred_t peercred; /* Peer credentials */ + socklen_t peersize; /* Size of peer credentials */ + + peersize = sizeof(peercred); + +# ifdef __APPLE__ + if (getsockopt(con->http.fd, 0, LOCAL_PEERCRED, &peercred, &peersize)) +# else + if (getsockopt(con->http.fd, SOL_SOCKET, SO_PEERCRED, &peercred, + &peersize)) +# endif /* __APPLE__ */ + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Unable to get peer credentials - %s", + con->http.fd, strerror(errno)); + } + else + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Using credentials for UID %d.", + con->http.fd, CUPSD_UCRED_UID(peercred)); + con->gss_uid = CUPSD_UCRED_UID(peercred); + } + } +# endif /* SO_PEERCRED && AF_LOCAL */ + } +#endif /* HAVE_GSSAPI */ + else + { + char scheme[256]; /* Auth scheme... */ + + + if (sscanf(authorization, "%255s", scheme) != 1) + strcpy(scheme, "UNKNOWN"); + + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Bad authentication data \"%s ...\"", + con->http.fd, scheme); + return; + } + + /* + * If we get here, then we were able to validate the username and + * password - copy the validated username and password to the client + * data and return... + */ + + strlcpy(con->username, username, sizeof(con->username)); + strlcpy(con->password, password, sizeof(con->password)); +} + + +/* + * 'cupsdCheckAccess()' - Check whether the given address is allowed to + * access a location. + */ + +int /* O - 1 if allowed, 0 otherwise */ +cupsdCheckAccess( + unsigned ip[4], /* I - Client address */ + char *name, /* I - Client hostname */ + int namelen, /* I - Length of hostname */ + cupsd_location_t *loc) /* I - Location to check */ +{ + int allow; /* 1 if allowed, 0 otherwise */ + + + if (!_cups_strcasecmp(name, "localhost")) + { + /* + * Access from localhost (127.0.0.1 or ::1) is always allowed... + */ + + return (1); + } + else + { + /* + * Do authorization checks on the domain/address... + */ + + switch (loc->order_type) + { + default : + allow = 0; /* anti-compiler-warning-code */ + break; + + case CUPSD_AUTH_ALLOW : /* Order Deny,Allow */ + allow = 1; + + if (cupsdCheckAuth(ip, name, namelen, loc->deny)) + allow = 0; + + if (cupsdCheckAuth(ip, name, namelen, loc->allow)) + allow = 1; + break; + + case CUPSD_AUTH_DENY : /* Order Allow,Deny */ + allow = 0; + + if (cupsdCheckAuth(ip, name, namelen, loc->allow)) + allow = 1; + + if (cupsdCheckAuth(ip, name, namelen, loc->deny)) + allow = 0; + break; + } + } + + return (allow); +} + + +/* + * 'cupsdCheckAuth()' - Check authorization masks. + */ + +int /* O - 1 if mask matches, 0 otherwise */ +cupsdCheckAuth(unsigned ip[4], /* I - Client address */ + char *name, /* I - Client hostname */ + int name_len, /* I - Length of hostname */ + cups_array_t *masks) /* I - Masks */ +{ + int i; /* Looping var */ + cupsd_authmask_t *mask; /* Current mask */ + cupsd_netif_t *iface; /* Network interface */ + unsigned netip4; /* IPv4 network address */ +#ifdef AF_INET6 + unsigned netip6[4]; /* IPv6 network address */ +#endif /* AF_INET6 */ + + + for (mask = (cupsd_authmask_t *)cupsArrayFirst(masks); + mask; + mask = (cupsd_authmask_t *)cupsArrayNext(masks)) + { + switch (mask->type) + { + case CUPSD_AUTH_INTERFACE : + /* + * Check for a match with a network interface... + */ + + netip4 = htonl(ip[3]); + +#ifdef AF_INET6 + netip6[0] = htonl(ip[0]); + netip6[1] = htonl(ip[1]); + netip6[2] = htonl(ip[2]); + netip6[3] = htonl(ip[3]); +#endif /* AF_INET6 */ + + if (!strcmp(mask->mask.name.name, "*")) + { +#ifdef __APPLE__ + /* + * Allow Back-to-My-Mac addresses... + */ + + if ((ip[0] & 0xff000000) == 0xfd000000) + return (1); +#endif /* __APPLE__ */ + + /* + * Check against all local interfaces... + */ + + cupsdNetIFUpdate(); + + for (iface = (cupsd_netif_t *)cupsArrayFirst(NetIFList); + iface; + iface = (cupsd_netif_t *)cupsArrayNext(NetIFList)) + { + /* + * Only check local interfaces... + */ + + if (!iface->is_local) + continue; + + if (iface->address.addr.sa_family == AF_INET) + { + /* + * Check IPv4 address... + */ + + if ((netip4 & iface->mask.ipv4.sin_addr.s_addr) == + (iface->address.ipv4.sin_addr.s_addr & + iface->mask.ipv4.sin_addr.s_addr)) + return (1); + } +#ifdef AF_INET6 + else + { + /* + * Check IPv6 address... + */ + + for (i = 0; i < 4; i ++) + if ((netip6[i] & iface->mask.ipv6.sin6_addr.s6_addr32[i]) != + (iface->address.ipv6.sin6_addr.s6_addr32[i] & + iface->mask.ipv6.sin6_addr.s6_addr32[i])) + break; + + if (i == 4) + return (1); + } +#endif /* AF_INET6 */ + } + } + else + { + /* + * Check the named interface... + */ + + for (iface = (cupsd_netif_t *)cupsArrayFirst(NetIFList); + iface; + iface = (cupsd_netif_t *)cupsArrayNext(NetIFList)) + { + if (strcmp(mask->mask.name.name, iface->name)) + continue; + + if (iface->address.addr.sa_family == AF_INET) + { + /* + * Check IPv4 address... + */ + + if ((netip4 & iface->mask.ipv4.sin_addr.s_addr) == + (iface->address.ipv4.sin_addr.s_addr & + iface->mask.ipv4.sin_addr.s_addr)) + return (1); + } +#ifdef AF_INET6 + else + { + /* + * Check IPv6 address... + */ + + for (i = 0; i < 4; i ++) + if ((netip6[i] & iface->mask.ipv6.sin6_addr.s6_addr32[i]) != + (iface->address.ipv6.sin6_addr.s6_addr32[i] & + iface->mask.ipv6.sin6_addr.s6_addr32[i])) + break; + + if (i == 4) + return (1); + } +#endif /* AF_INET6 */ + } + } + break; + + case CUPSD_AUTH_NAME : + /* + * Check for exact name match... + */ + + if (!_cups_strcasecmp(name, mask->mask.name.name)) + return (1); + + /* + * Check for domain match... + */ + + if (name_len >= mask->mask.name.length && + mask->mask.name.name[0] == '.' && + !_cups_strcasecmp(name + name_len - mask->mask.name.length, + mask->mask.name.name)) + return (1); + break; + + case CUPSD_AUTH_IP : + /* + * Check for IP/network address match... + */ + + for (i = 0; i < 4; i ++) + if ((ip[i] & mask->mask.ip.netmask[i]) != + mask->mask.ip.address[i]) + break; + + if (i == 4) + return (1); + break; + } + } + + return (0); +} + + +/* + * 'cupsdCheckGroup()' - Check for a user's group membership. + */ + +int /* O - 1 if user is a member, 0 otherwise */ +cupsdCheckGroup( + const char *username, /* I - User name */ + struct passwd *user, /* I - System user info */ + const char *groupname) /* I - Group name */ +{ + int i; /* Looping var */ + struct group *group; /* System group info */ + char junk[33]; /* MD5 password (not used) */ +#ifdef HAVE_MBR_UID_TO_UUID + uuid_t useruuid, /* UUID for username */ + groupuuid; /* UUID for groupname */ + int is_member; /* True if user is a member of group */ +#endif /* HAVE_MBR_UID_TO_UUID */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdCheckGroup(username=\"%s\", user=%p, groupname=\"%s\")", + username, user, groupname); + + /* + * Validate input... + */ + + if (!username || !groupname) + return (0); + + /* + * Check to see if the user is a member of the named group... + */ + + group = getgrnam(groupname); + endgrent(); + + if (group != NULL) + { + /* + * Group exists, check it... + */ + + for (i = 0; group->gr_mem[i]; i ++) + if (!_cups_strcasecmp(username, group->gr_mem[i])) + return (1); + } + + /* + * Group doesn't exist or user not in group list, check the group ID + * against the user's group ID... + */ + + if (user && group && group->gr_gid == user->pw_gid) + return (1); + +#ifdef HAVE_MBR_UID_TO_UUID + /* + * Check group membership through MacOS X membership API... + */ + + if (user && !mbr_uid_to_uuid(user->pw_uid, useruuid)) + { + if (group) + { + /* + * Map group name to UUID and check membership... + */ + + if (!mbr_gid_to_uuid(group->gr_gid, groupuuid)) + if (!mbr_check_membership(useruuid, groupuuid, &is_member)) + if (is_member) + return (1); + } + else if (groupname[0] == '#') + { + /* + * Use UUID directly and check for equality (user UUID) and + * membership (group UUID)... + */ + + if (!uuid_parse((char *)groupname + 1, groupuuid)) + { + if (!uuid_compare(useruuid, groupuuid)) + return (1); + else if (!mbr_check_membership(useruuid, groupuuid, &is_member)) + if (is_member) + return (1); + } + + return (0); + } + } + else if (groupname[0] == '#') + return (0); +#endif /* HAVE_MBR_UID_TO_UUID */ + + /* + * Username not found, group not found, or user is not part of the + * system group... Check for a user and group in the MD5 password + * file... + */ + + if (get_md5_password(username, groupname, junk) != NULL) + return (1); + + /* + * If we get this far, then the user isn't part of the named group... + */ + + return (0); +} + + +/* + * 'cupsdCopyLocation()' - Make a copy of a location... + */ + +cupsd_location_t * /* O - New location */ +cupsdCopyLocation( + cupsd_location_t *loc) /* I - Original location */ +{ + cupsd_location_t *temp; /* New location */ + + + /* + * Make a copy of the original location... + */ + + if ((temp = calloc(1, sizeof(cupsd_location_t))) == NULL) + return (NULL); + + /* + * Copy the information from the original location to the new one. + */ + + if (!loc) + return (temp); + + if (loc->location) + temp->location = _cupsStrAlloc(loc->location); + + temp->limit = loc->limit; + temp->order_type = loc->order_type; + temp->type = loc->type; + temp->level = loc->level; + temp->satisfy = loc->satisfy; + temp->encryption = loc->encryption; + + if (loc->names) + { + if ((temp->names = cupsArrayDup(loc->names)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate memory for %d names: %s", + cupsArrayCount(loc->names), strerror(errno)); + + cupsdFreeLocation(temp); + return (NULL); + } + } + + if (loc->allow) + { + /* + * Copy allow rules... + */ + + if ((temp->allow = cupsArrayDup(loc->allow)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate memory for %d allow rules: %s", + cupsArrayCount(loc->allow), strerror(errno)); + cupsdFreeLocation(temp); + return (NULL); + } + } + + if (loc->deny) + { + /* + * Copy deny rules... + */ + + if ((temp->deny = cupsArrayDup(loc->deny)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate memory for %d deny rules: %s", + cupsArrayCount(loc->deny), strerror(errno)); + cupsdFreeLocation(temp); + return (NULL); + } + } + + return (temp); +} + + +/* + * 'cupsdDeleteAllLocations()' - Free all memory used for location authorization. + */ + +void +cupsdDeleteAllLocations(void) +{ + /* + * Free the location array, which will free all of the locations... + */ + + cupsArrayDelete(Locations); + Locations = NULL; +} + + +/* + * 'cupsdFindBest()' - Find the location entry that best matches the resource. + */ + +cupsd_location_t * /* O - Location that matches */ +cupsdFindBest(const char *path, /* I - Resource path */ + http_state_t state) /* I - HTTP state/request */ +{ + char uri[HTTP_MAX_URI], + /* URI in request... */ + *uriptr; /* Pointer into URI */ + cupsd_location_t *loc, /* Current location */ + *best; /* Best match for location so far */ + int bestlen; /* Length of best match */ + int limit; /* Limit field */ + static const int limits[] = /* Map http_status_t to CUPSD_AUTH_LIMIT_xyz */ + { + CUPSD_AUTH_LIMIT_ALL, + CUPSD_AUTH_LIMIT_OPTIONS, + CUPSD_AUTH_LIMIT_GET, + CUPSD_AUTH_LIMIT_GET, + CUPSD_AUTH_LIMIT_HEAD, + CUPSD_AUTH_LIMIT_POST, + CUPSD_AUTH_LIMIT_POST, + CUPSD_AUTH_LIMIT_POST, + CUPSD_AUTH_LIMIT_PUT, + CUPSD_AUTH_LIMIT_PUT, + CUPSD_AUTH_LIMIT_DELETE, + CUPSD_AUTH_LIMIT_TRACE, + CUPSD_AUTH_LIMIT_ALL, + CUPSD_AUTH_LIMIT_ALL + }; + + + /* + * First copy the connection URI to a local string so we have drop + * any .ppd extension from the pathname in /printers or /classes + * URIs... + */ + + strlcpy(uri, path, sizeof(uri)); + + if (!strncmp(uri, "/printers/", 10) || + !strncmp(uri, "/classes/", 9)) + { + /* + * Check if the URI has .ppd on the end... + */ + + uriptr = uri + strlen(uri) - 4; /* len > 4 if we get here... */ + + if (!strcmp(uriptr, ".ppd")) + *uriptr = '\0'; + } + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdFindBest: uri = \"%s\"...", uri); + + /* + * Loop through the list of locations to find a match... + */ + + limit = limits[state]; + best = NULL; + bestlen = 0; + + for (loc = (cupsd_location_t *)cupsArrayFirst(Locations); + loc; + loc = (cupsd_location_t *)cupsArrayNext(Locations)) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdFindBest: Location %s Limit %x", + loc->location ? loc->location : "nil", loc->limit); + + if (!strncmp(uri, "/printers/", 10) || !strncmp(uri, "/classes/", 9)) + { + /* + * Use case-insensitive comparison for queue names... + */ + + if (loc->length > bestlen && loc->location && + !_cups_strncasecmp(uri, loc->location, loc->length) && + loc->location[0] == '/' && + (limit & loc->limit) != 0) + { + best = loc; + bestlen = loc->length; + } + } + else + { + /* + * Use case-sensitive comparison for other URIs... + */ + + if (loc->length > bestlen && loc->location && + !strncmp(uri, loc->location, loc->length) && + loc->location[0] == '/' && + (limit & loc->limit) != 0) + { + best = loc; + bestlen = loc->length; + } + } + } + + /* + * Return the match, if any... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdFindBest: best = %s", + best ? best->location : "NONE"); + + return (best); +} + + +/* + * 'cupsdFindLocation()' - Find the named location. + */ + +cupsd_location_t * /* O - Location that matches */ +cupsdFindLocation(const char *location) /* I - Connection */ +{ + cupsd_location_t key; /* Search key */ + + + key.location = (char *)location; + + return ((cupsd_location_t *)cupsArrayFind(Locations, &key)); +} + + +/* + * 'cupsdFreeLocation()' - Free all memory used by a location. + */ + +void +cupsdFreeLocation(cupsd_location_t *loc)/* I - Location to free */ +{ + cupsArrayDelete(loc->names); + cupsArrayDelete(loc->allow); + cupsArrayDelete(loc->deny); + + _cupsStrFree(loc->location); + free(loc); +} + + +/* + * 'cupsdIsAuthorized()' - Check to see if the user is authorized... + */ + +http_status_t /* O - HTTP_OK if authorized or error code */ +cupsdIsAuthorized(cupsd_client_t *con, /* I - Connection */ + const char *owner)/* I - Owner of object */ +{ + int i, /* Looping vars */ + auth, /* Authorization status */ + type; /* Type of authentication */ + unsigned address[4]; /* Authorization address */ + cupsd_location_t *best; /* Best match for location so far */ + int hostlen; /* Length of hostname */ + char *name, /* Current username */ + username[256], /* Username to authorize */ + ownername[256], /* Owner name to authorize */ + *ptr; /* Pointer into username */ + struct passwd *pw; /* User password data */ + static const char * const levels[] = /* Auth levels */ + { + "ANON", + "USER", + "GROUP" + }; + static const char * const types[] = /* Auth types */ + { + "None", + "Basic", + "Digest", + "BasicDigest", + "Negotiate" + }; + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdIsAuthorized: con->uri=\"%s\", con->best=%p(%s)", + con->uri, con->best, con->best ? con->best->location ? + con->best->location : "(null)" : ""); + if (owner) + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdIsAuthorized: owner=\"%s\"", owner); + + /* + * If there is no "best" authentication rule for this request, then + * access is allowed from the local system and denied from other + * addresses... + */ + + if (!con->best) + { + if (!strcmp(con->http.hostname, "localhost") || + !strcmp(con->http.hostname, ServerName)) + return (HTTP_OK); + else + return (HTTP_FORBIDDEN); + } + + best = con->best; + + if ((type = best->type) == CUPSD_AUTH_DEFAULT) + type = cupsdDefaultAuthType(); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdIsAuthorized: level=CUPSD_AUTH_%s, type=%s, " + "satisfy=CUPSD_AUTH_SATISFY_%s, num_names=%d", + levels[best->level], types[type], + best->satisfy ? "ANY" : "ALL", cupsArrayCount(best->names)); + + if (best->limit == CUPSD_AUTH_LIMIT_IPP) + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdIsAuthorized: op=%x(%s)", + best->op, ippOpString(best->op)); + + /* + * Check host/ip-based accesses... + */ + +#ifdef AF_INET6 + if (con->http.hostaddr->addr.sa_family == AF_INET6) + { + /* + * Copy IPv6 address... + */ + + address[0] = ntohl(con->http.hostaddr->ipv6.sin6_addr.s6_addr32[0]); + address[1] = ntohl(con->http.hostaddr->ipv6.sin6_addr.s6_addr32[1]); + address[2] = ntohl(con->http.hostaddr->ipv6.sin6_addr.s6_addr32[2]); + address[3] = ntohl(con->http.hostaddr->ipv6.sin6_addr.s6_addr32[3]); + } + else +#endif /* AF_INET6 */ + if (con->http.hostaddr->addr.sa_family == AF_INET) + { + /* + * Copy IPv4 address... + */ + + address[0] = 0; + address[1] = 0; + address[2] = 0; + address[3] = ntohl(con->http.hostaddr->ipv4.sin_addr.s_addr); + } + else + memset(address, 0, sizeof(address)); + + hostlen = strlen(con->http.hostname); + + auth = cupsdCheckAccess(address, con->http.hostname, hostlen, best) + ? CUPSD_AUTH_ALLOW : CUPSD_AUTH_DENY; + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdIsAuthorized: auth=CUPSD_AUTH_%s...", + auth ? "DENY" : "ALLOW"); + + if (auth == CUPSD_AUTH_DENY && best->satisfy == CUPSD_AUTH_SATISFY_ALL) + return (HTTP_FORBIDDEN); + +#ifdef HAVE_SSL + /* + * See if encryption is required... + */ + + if ((best->encryption >= HTTP_ENCRYPT_REQUIRED && !con->http.tls && + _cups_strcasecmp(con->http.hostname, "localhost") && + best->satisfy == CUPSD_AUTH_SATISFY_ALL) && + !(type == CUPSD_AUTH_NEGOTIATE || + (type == CUPSD_AUTH_NONE && + cupsdDefaultAuthType() == CUPSD_AUTH_NEGOTIATE))) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "cupsdIsAuthorized: Need upgrade to TLS..."); + return (HTTP_UPGRADE_REQUIRED); + } +#endif /* HAVE_SSL */ + + /* + * Now see what access level is required... + */ + + if (best->level == CUPSD_AUTH_ANON || /* Anonymous access - allow it */ + (type == CUPSD_AUTH_NONE && cupsArrayCount(best->names) == 0)) + return (HTTP_OK); + + if (!con->username[0] && type == CUPSD_AUTH_NONE && + best->limit == CUPSD_AUTH_LIMIT_IPP) + { + /* + * Check for unauthenticated username... + */ + + ipp_attribute_t *attr; /* requesting-user-name attribute */ + + + attr = ippFindAttribute(con->request, "requesting-user-name", IPP_TAG_NAME); + if (attr) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "cupsdIsAuthorized: requesting-user-name=\"%s\"", + attr->values[0].string.text); + strlcpy(username, attr->values[0].string.text, sizeof(username)); + } + else if (best->satisfy == CUPSD_AUTH_SATISFY_ALL || auth == CUPSD_AUTH_DENY) + return (HTTP_UNAUTHORIZED); /* Non-anonymous needs user/pass */ + else + return (HTTP_OK); /* unless overridden with Satisfy */ + } + else + { + cupsdLogMessage(CUPSD_LOG_DEBUG, "cupsdIsAuthorized: username=\"%s\"", + con->username); + +#ifdef HAVE_AUTHORIZATION_H + if (!con->username[0] && !con->authref) +#else + if (!con->username[0]) +#endif /* HAVE_AUTHORIZATION_H */ + { + if (best->satisfy == CUPSD_AUTH_SATISFY_ALL || auth == CUPSD_AUTH_DENY) + return (HTTP_UNAUTHORIZED); /* Non-anonymous needs user/pass */ + else + return (HTTP_OK); /* unless overridden with Satisfy */ + } + + + if (con->type != type && type != CUPSD_AUTH_NONE && +#ifdef HAVE_GSSAPI + (type != CUPSD_AUTH_NEGOTIATE || con->gss_uid <= 0) && +#endif /* HAVE_GSSAPI */ + (con->type != CUPSD_AUTH_BASIC || type != CUPSD_AUTH_BASICDIGEST)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Authorized using %s, expected %s!", + types[con->type], types[type]); + + return (HTTP_UNAUTHORIZED); + } + + strlcpy(username, con->username, sizeof(username)); + } + + /* + * OK, got a username. See if we need normal user access, or group + * access... (root always matches) + */ + + if (!strcmp(username, "root")) + return (HTTP_OK); + + /* + * Strip any @domain or @KDC from the username and owner... + */ + + if ((ptr = strchr(username, '@')) != NULL) + *ptr = '\0'; + + if (owner) + { + strlcpy(ownername, owner, sizeof(ownername)); + + if ((ptr = strchr(ownername, '@')) != NULL) + *ptr = '\0'; + } + else + ownername[0] = '\0'; + + /* + * Get the user info... + */ + + if (username[0]) + { + pw = getpwnam(username); + endpwent(); + } + else + pw = NULL; + + if (best->level == CUPSD_AUTH_USER) + { + /* + * If there are no names associated with this location, then + * any valid user is OK... + */ + + if (cupsArrayCount(best->names) == 0) + return (HTTP_OK); + + /* + * Otherwise check the user list and return OK if this user is + * allowed... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdIsAuthorized: Checking user membership..."); + +#ifdef HAVE_AUTHORIZATION_H + /* + * If an authorization reference was supplied it must match a right name... + */ + + if (con->authref) + { + for (name = (char *)cupsArrayFirst(best->names); + name; + name = (char *)cupsArrayNext(best->names)) + { + if (!_cups_strncasecmp(name, "@AUTHKEY(", 9) && check_authref(con, name + 9)) + return (HTTP_OK); + else if (!_cups_strcasecmp(name, "@SYSTEM") && SystemGroupAuthKey && + check_authref(con, SystemGroupAuthKey)) + return (HTTP_OK); + } + + return (HTTP_FORBIDDEN); + } +#endif /* HAVE_AUTHORIZATION_H */ + + for (name = (char *)cupsArrayFirst(best->names); + name; + name = (char *)cupsArrayNext(best->names)) + { + if (!_cups_strcasecmp(name, "@OWNER") && owner && + !_cups_strcasecmp(username, ownername)) + return (HTTP_OK); + else if (!_cups_strcasecmp(name, "@SYSTEM")) + { + for (i = 0; i < NumSystemGroups; i ++) + if (cupsdCheckGroup(username, pw, SystemGroups[i])) + return (HTTP_OK); + } + else if (name[0] == '@') + { + if (cupsdCheckGroup(username, pw, name + 1)) + return (HTTP_OK); + } + else if (!_cups_strcasecmp(username, name)) + return (HTTP_OK); + } + + return (con->username[0] ? HTTP_FORBIDDEN : HTTP_UNAUTHORIZED); + } + + /* + * Check to see if this user is in any of the named groups... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdIsAuthorized: Checking group membership..."); + + /* + * Check to see if this user is in any of the named groups... + */ + + for (name = (char *)cupsArrayFirst(best->names); + name; + name = (char *)cupsArrayNext(best->names)) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdIsAuthorized: Checking group \"%s\" membership...", + name); + + if (!_cups_strcasecmp(name, "@SYSTEM")) + { + for (i = 0; i < NumSystemGroups; i ++) + if (cupsdCheckGroup(username, pw, SystemGroups[i])) + return (HTTP_OK); + } + else if (cupsdCheckGroup(username, pw, name)) + return (HTTP_OK); + } + + /* + * The user isn't part of the specified group, so deny access... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "cupsdIsAuthorized: User not in group(s)!"); + + return (con->username[0] ? HTTP_FORBIDDEN : HTTP_UNAUTHORIZED); +} + + +/* + * 'cupsdNewLocation()' - Create a new location for authorization. + * + * Note: Still need to call cupsdAddLocation() to add it to the list of global + * locations. + */ + +cupsd_location_t * /* O - Pointer to new location record */ +cupsdNewLocation(const char *location) /* I - Location path */ +{ + cupsd_location_t *temp; /* New location */ + + + /* + * Try to allocate memory for the new location. + */ + + if ((temp = calloc(1, sizeof(cupsd_location_t))) == NULL) + return (NULL); + + /* + * Initialize the record and copy the name over... + */ + + if ((temp->location = _cupsStrAlloc(location)) == NULL) + { + free(temp); + return (NULL); + } + + temp->length = strlen(temp->location); + + /* + * Return the new record... + */ + + return (temp); +} + + +#ifdef HAVE_AUTHORIZATION_H +/* + * 'check_authref()' - Check if an authorization services reference has the + * supplied right. + */ + +static int /* O - 1 if right is valid, 0 otherwise */ +check_authref(cupsd_client_t *con, /* I - Connection */ + const char *right) /* I - Right name */ +{ + OSStatus status; /* OS Status */ + AuthorizationItem authright; /* Authorization right */ + AuthorizationRights authrights; /* Authorization rights */ + AuthorizationFlags authflags; /* Authorization flags */ + + + /* + * Check to see if the user is allowed to perform the task... + */ + + if (!con->authref) + return (0); + + authright.name = right; + authright.valueLength = 0; + authright.value = NULL; + authright.flags = 0; + + authrights.count = 1; + authrights.items = &authright; + + authflags = kAuthorizationFlagDefaults | + kAuthorizationFlagExtendRights; + + if ((status = AuthorizationCopyRights(con->authref, &authrights, + kAuthorizationEmptyEnvironment, + authflags, NULL)) != 0) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "AuthorizationCopyRights(\"%s\") returned %d (%s)", + authright.name, (int)status, cssmErrorString(status)); + return (0); + } + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "AuthorizationCopyRights(\"%s\") succeeded!", + authright.name); + + return (1); +} +#endif /* HAVE_AUTHORIZATION_H */ + + +/* + * 'compare_locations()' - Compare two locations. + */ + +static int /* O - Result of comparison */ +compare_locations(cupsd_location_t *a, /* I - First location */ + cupsd_location_t *b) /* I - Second location */ +{ + return (strcmp(b->location, a->location)); +} + + +/* + * 'copy_authmask()' - Copy function for auth masks. + */ + +static cupsd_authmask_t * /* O - New auth mask */ +copy_authmask(cupsd_authmask_t *mask, /* I - Existing auth mask */ + void *data) /* I - User data (unused) */ +{ + cupsd_authmask_t *temp; /* New auth mask */ + + + (void)data; + + if ((temp = malloc(sizeof(cupsd_authmask_t))) != NULL) + { + memcpy(temp, mask, sizeof(cupsd_authmask_t)); + + if (temp->type == CUPSD_AUTH_NAME || temp->type == CUPSD_AUTH_INTERFACE) + { + /* + * Make a copy of the name... + */ + + if ((temp->mask.name.name = _cupsStrAlloc(temp->mask.name.name)) == NULL) + { + /* + * Failed to make copy... + */ + + free(temp); + temp = NULL; + } + } + } + + return (temp); +} + + +#if !HAVE_LIBPAM && !defined(HAVE_USERSEC_H) +/* + * 'cups_crypt()' - Encrypt the password using the DES or MD5 algorithms, + * as needed. + */ + +static char * /* O - Encrypted password */ +cups_crypt(const char *pw, /* I - Password string */ + const char *salt) /* I - Salt (key) string */ +{ + if (!strncmp(salt, "$1$", 3)) + { + /* + * Use MD5 passwords without the benefit of PAM; this is for + * Slackware Linux, and the algorithm was taken from the + * old shadow-19990827/lib/md5crypt.c source code... :( + */ + + int i; /* Looping var */ + unsigned long n; /* Output number */ + int pwlen; /* Length of password string */ + const char *salt_end; /* End of "salt" data for MD5 */ + char *ptr; /* Pointer into result string */ + _cups_md5_state_t state; /* Primary MD5 state info */ + _cups_md5_state_t state2; /* Secondary MD5 state info */ + unsigned char digest[16]; /* MD5 digest result */ + static char result[120]; /* Final password string */ + + + /* + * Get the salt data between dollar signs, e.g. $1$saltdata$md5. + * Get a maximum of 8 characters of salt data after $1$... + */ + + for (salt_end = salt + 3; *salt_end && (salt_end - salt) < 11; salt_end ++) + if (*salt_end == '$') + break; + + /* + * Compute the MD5 sum we need... + */ + + pwlen = strlen(pw); + + _cupsMD5Init(&state); + _cupsMD5Append(&state, (unsigned char *)pw, pwlen); + _cupsMD5Append(&state, (unsigned char *)salt, salt_end - salt); + + _cupsMD5Init(&state2); + _cupsMD5Append(&state2, (unsigned char *)pw, pwlen); + _cupsMD5Append(&state2, (unsigned char *)salt + 3, salt_end - salt - 3); + _cupsMD5Append(&state2, (unsigned char *)pw, pwlen); + _cupsMD5Finish(&state2, digest); + + for (i = pwlen; i > 0; i -= 16) + _cupsMD5Append(&state, digest, i > 16 ? 16 : i); + + for (i = pwlen; i > 0; i >>= 1) + _cupsMD5Append(&state, (unsigned char *)((i & 1) ? "" : pw), 1); + + _cupsMD5Finish(&state, digest); + + for (i = 0; i < 1000; i ++) + { + _cupsMD5Init(&state); + + if (i & 1) + _cupsMD5Append(&state, (unsigned char *)pw, pwlen); + else + _cupsMD5Append(&state, digest, 16); + + if (i % 3) + _cupsMD5Append(&state, (unsigned char *)salt + 3, salt_end - salt - 3); + + if (i % 7) + _cupsMD5Append(&state, (unsigned char *)pw, pwlen); + + if (i & 1) + _cupsMD5Append(&state, digest, 16); + else + _cupsMD5Append(&state, (unsigned char *)pw, pwlen); + + _cupsMD5Finish(&state, digest); + } + + /* + * Copy the final sum to the result string and return... + */ + + memcpy(result, salt, salt_end - salt); + ptr = result + (salt_end - salt); + *ptr++ = '$'; + + for (i = 0; i < 5; i ++, ptr += 4) + { + n = (((digest[i] << 8) | digest[i + 6]) << 8); + + if (i < 4) + n |= digest[i + 12]; + else + n |= digest[5]; + + to64(ptr, n, 4); + } + + to64(ptr, digest[11], 2); + ptr += 2; + *ptr = '\0'; + + return (result); + } + else + { + /* + * Use the standard crypt() function... + */ + + return (crypt(pw, salt)); + } +} +#endif /* !HAVE_LIBPAM && !HAVE_USERSEC_H */ + + +/* + * 'free_authmask()' - Free function for auth masks. + */ + +static void +free_authmask(cupsd_authmask_t *mask, /* I - Auth mask to free */ + void *data) /* I - User data (unused) */ +{ + (void)data; + + if (mask->type == CUPSD_AUTH_NAME || mask->type == CUPSD_AUTH_INTERFACE) + _cupsStrFree(mask->mask.name.name); + + free(mask); +} + + +/* + * 'get_md5_password()' - Get an MD5 password. + */ + +static char * /* O - MD5 password string */ +get_md5_password(const char *username, /* I - Username */ + const char *group, /* I - Group */ + char passwd[33]) /* O - MD5 password string */ +{ + cups_file_t *fp; /* passwd.md5 file */ + char filename[1024], /* passwd.md5 filename */ + line[256], /* Line from file */ + tempuser[33], /* User from file */ + tempgroup[33]; /* Group from file */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "get_md5_password(username=\"%s\", group=\"%s\", passwd=%p)", + username, group ? group : "(null)", passwd); + + snprintf(filename, sizeof(filename), "%s/passwd.md5", ServerRoot); + if ((fp = cupsFileOpen(filename, "r")) == NULL) + { + if (errno != ENOENT) + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to open %s - %s", filename, + strerror(errno)); + + return (NULL); + } + + while (cupsFileGets(fp, line, sizeof(line)) != NULL) + { + if (sscanf(line, "%32[^:]:%32[^:]:%32s", tempuser, tempgroup, passwd) != 3) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Bad MD5 password line: %s", line); + continue; + } + + if (!strcmp(username, tempuser) && + (group == NULL || !strcmp(group, tempgroup))) + { + /* + * Found the password entry! + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "Found MD5 user %s, group %s...", + username, tempgroup); + + cupsFileClose(fp); + return (passwd); + } + } + + /* + * Didn't find a password entry - return NULL! + */ + + cupsFileClose(fp); + return (NULL); +} + + +#if HAVE_LIBPAM +/* + * 'pam_func()' - PAM conversation function. + */ + +static int /* O - Success or failure */ +pam_func( + int num_msg, /* I - Number of messages */ + const struct pam_message **msg, /* I - Messages */ + struct pam_response **resp, /* O - Responses */ + void *appdata_ptr) + /* I - Pointer to connection */ +{ + int i; /* Looping var */ + struct pam_response *replies; /* Replies */ + cupsd_authdata_t *data; /* Pointer to auth data */ + + + /* + * Allocate memory for the responses... + */ + + if ((replies = malloc(sizeof(struct pam_response) * num_msg)) == NULL) + return (PAM_CONV_ERR); + + /* + * Answer all of the messages... + */ + + DEBUG_printf(("pam_func: appdata_ptr = %p\n", appdata_ptr)); + +#ifdef __hpux + /* + * Apparently some versions of HP-UX 11 have a broken pam_unix security + * module. This is a workaround... + */ + + data = auth_data; + (void)appdata_ptr; +#else + data = (cupsd_authdata_t *)appdata_ptr; +#endif /* __hpux */ + + for (i = 0; i < num_msg; i ++) + { + DEBUG_printf(("pam_func: Message = \"%s\"\n", msg[i]->msg)); + + switch (msg[i]->msg_style) + { + case PAM_PROMPT_ECHO_ON: + DEBUG_printf(("pam_func: PAM_PROMPT_ECHO_ON, returning \"%s\"...\n", + data->username)); + replies[i].resp_retcode = PAM_SUCCESS; + replies[i].resp = strdup(data->username); + break; + + case PAM_PROMPT_ECHO_OFF: + DEBUG_printf(("pam_func: PAM_PROMPT_ECHO_OFF, returning \"%s\"...\n", + data->password)); + replies[i].resp_retcode = PAM_SUCCESS; + replies[i].resp = strdup(data->password); + break; + + case PAM_TEXT_INFO: + DEBUG_puts("pam_func: PAM_TEXT_INFO..."); + replies[i].resp_retcode = PAM_SUCCESS; + replies[i].resp = NULL; + break; + + case PAM_ERROR_MSG: + DEBUG_puts("pam_func: PAM_ERROR_MSG..."); + replies[i].resp_retcode = PAM_SUCCESS; + replies[i].resp = NULL; + break; + + default: + DEBUG_printf(("pam_func: Unknown PAM message %d...\n", + msg[i]->msg_style)); + free(replies); + return (PAM_CONV_ERR); + } + } + + /* + * Return the responses back to PAM... + */ + + *resp = replies; + + return (PAM_SUCCESS); +} +#elif !defined(HAVE_USERSEC_H) + + +/* + * 'to64()' - Base64-encode an integer value... + */ + +static void +to64(char *s, /* O - Output string */ + unsigned long v, /* I - Value to encode */ + int n) /* I - Number of digits */ +{ + const char *itoa64 = "./0123456789" + "ABCDEFGHIJKLMNOPQRSTUVWXYZ" + "abcdefghijklmnopqrstuvwxyz"; + + + for (; n > 0; n --, v >>= 6) + *s++ = itoa64[v & 0x3f]; +} +#endif /* HAVE_LIBPAM */ + + +/* + * End of "$Id$". + */ diff --git a/scheduler/auth.h b/scheduler/auth.h new file mode 100644 index 0000000000..feb2fcd523 --- /dev/null +++ b/scheduler/auth.h @@ -0,0 +1,151 @@ +/* + * "$Id$" + * + * Authorization definitions for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * HTTP authorization types and levels... + */ + +#define CUPSD_AUTH_DEFAULT -1 /* Use DefaultAuthType */ +#define CUPSD_AUTH_NONE 0 /* No authentication */ +#define CUPSD_AUTH_BASIC 1 /* Basic authentication */ +#define CUPSD_AUTH_DIGEST 2 /* Digest authentication */ +#define CUPSD_AUTH_BASICDIGEST 3 /* Basic authentication w/passwd.md5 */ +#define CUPSD_AUTH_NEGOTIATE 4 /* Kerberos authentication */ +#define CUPSD_AUTH_AUTO 5 /* Kerberos or Basic, depending on configuration of server */ + +#define CUPSD_AUTH_ANON 0 /* Anonymous access */ +#define CUPSD_AUTH_USER 1 /* Must have a valid username/password */ +#define CUPSD_AUTH_GROUP 2 /* Must also be in a named group */ + +#define CUPSD_AUTH_ALLOW 0 /* Allow access */ +#define CUPSD_AUTH_DENY 1 /* Deny access */ + +#define CUPSD_AUTH_NAME 0 /* Authorize host by name */ +#define CUPSD_AUTH_IP 1 /* Authorize host by IP */ +#define CUPSD_AUTH_INTERFACE 2 /* Authorize host by interface */ + +#define CUPSD_AUTH_SATISFY_ALL 0 /* Satisfy both address and auth */ +#define CUPSD_AUTH_SATISFY_ANY 1 /* Satisfy either address or auth */ + +#define CUPSD_AUTH_LIMIT_DELETE 1 /* Limit DELETE requests */ +#define CUPSD_AUTH_LIMIT_GET 2 /* Limit GET requests */ +#define CUPSD_AUTH_LIMIT_HEAD 4 /* Limit HEAD requests */ +#define CUPSD_AUTH_LIMIT_OPTIONS 8 /* Limit OPTIONS requests */ +#define CUPSD_AUTH_LIMIT_POST 16 /* Limit POST requests */ +#define CUPSD_AUTH_LIMIT_PUT 32 /* Limit PUT requests */ +#define CUPSD_AUTH_LIMIT_TRACE 64 /* Limit TRACE requests */ +#define CUPSD_AUTH_LIMIT_ALL 127 /* Limit all requests */ +#define CUPSD_AUTH_LIMIT_IPP 128 /* Limit IPP requests */ + +#define IPP_ANY_OPERATION (ipp_op_t)0 + /* Any IPP operation */ +#define IPP_BAD_OPERATION (ipp_op_t)-1 + /* No IPP operation */ + + +/* + * HTTP access control structures... + */ + +typedef struct +{ + unsigned address[4], /* IP address */ + netmask[4]; /* IP netmask */ +} cupsd_ipmask_t; + +typedef struct +{ + int length; /* Length of name */ + char *name; /* Name string */ +} cupsd_namemask_t; + +typedef struct +{ + int type; /* Mask type */ + union + { + cupsd_namemask_t name; /* Host/Domain name */ + cupsd_ipmask_t ip; /* IP address/network */ + } mask; /* Mask data */ +} cupsd_authmask_t; + +typedef struct +{ + char *location; /* Location of resource */ + ipp_op_t op; /* IPP operation */ + int limit, /* Limit for these types of requests */ + length, /* Length of location string */ + order_type, /* Allow or Deny */ + type, /* Type of authentication */ + level, /* Access level required */ + satisfy; /* Satisfy any or all limits? */ + cups_array_t *names, /* User or group names */ + *allow, /* Allow lines */ + *deny; /* Deny lines */ + http_encryption_t encryption; /* To encrypt or not to encrypt... */ +} cupsd_location_t; + +typedef struct cupsd_client_s cupsd_client_t; + + +/* + * Globals... + */ + +VAR cups_array_t *Locations VALUE(NULL); + /* Authorization locations */ +#ifdef HAVE_SSL +VAR http_encryption_t DefaultEncryption VALUE(HTTP_ENCRYPT_REQUIRED); + /* Default encryption for authentication */ +#endif /* HAVE_SSL */ + + +/* + * Prototypes... + */ + +extern int cupsdAddIPMask(cups_array_t **masks, + const unsigned address[4], + const unsigned netmask[4]); +extern void cupsdAddLocation(cupsd_location_t *loc); +extern void cupsdAddName(cupsd_location_t *loc, char *name); +extern int cupsdAddNameMask(cups_array_t **masks, char *name); +extern void cupsdAuthorize(cupsd_client_t *con); +extern int cupsdCheckAccess(unsigned ip[4], char *name, + int namelen, cupsd_location_t *loc); +extern int cupsdCheckAuth(unsigned ip[4], char *name, int namelen, + cups_array_t *masks); +extern int cupsdCheckGroup(const char *username, + struct passwd *user, + const char *groupname); +extern cupsd_location_t *cupsdCopyLocation(cupsd_location_t *loc); +extern void cupsdDeleteAllLocations(void); +extern cupsd_location_t *cupsdFindBest(const char *path, http_state_t state); +extern cupsd_location_t *cupsdFindLocation(const char *location); +extern void cupsdFreeLocation(cupsd_location_t *loc); +extern http_status_t cupsdIsAuthorized(cupsd_client_t *con, const char *owner); +extern cupsd_location_t *cupsdNewLocation(const char *location); + + +/* + * End of "$Id$". + */ diff --git a/scheduler/banners.c b/scheduler/banners.c new file mode 100644 index 0000000000..77ae797ae1 --- /dev/null +++ b/scheduler/banners.c @@ -0,0 +1,224 @@ +/* + * "$Id$" + * + * Banner routines for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdFindBanner() - Find a named banner. + * cupsdLoadBanners() - Load all available banner files... + * add_banner() - Add a banner to the array. + * compare_banners() - Compare two banners. + * free_banners() - Free all banners. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" +#include + + +/* + * Local functions... + */ + +static void add_banner(const char *name, const char *filename); +static int compare_banners(const cupsd_banner_t *b0, + const cupsd_banner_t *b1); +static void free_banners(void); + + +/* + * 'cupsdFindBanner()' - Find a named banner. + */ + +cupsd_banner_t * /* O - Pointer to banner or NULL */ +cupsdFindBanner(const char *name) /* I - Name of banner */ +{ + cupsd_banner_t key; /* Search key */ + + + key.name = (char *)name; + + return ((cupsd_banner_t *)cupsArrayFind(Banners, &key)); +} + + +/* + * 'cupsdLoadBanners()' - Load all available banner files... + */ + +void +cupsdLoadBanners(const char *d) /* I - Directory to search */ +{ + cups_dir_t *dir; /* Directory pointer */ + cups_dentry_t *dent; /* Directory entry */ + char filename[1024], /* Name of banner */ + *ext; /* Pointer to extension */ + + + /* + * Free old banner info... + */ + + free_banners(); + + /* + * Try opening the banner directory... + */ + + if ((dir = cupsDirOpen(d)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "cupsdLoadBanners: Unable to open banner directory \"%s\": %s", + d, strerror(errno)); + return; + } + + /* + * Read entries, skipping directories and backup files. + */ + + Banners = cupsArrayNew((cups_array_func_t)compare_banners, NULL); + + while ((dent = cupsDirRead(dir)) != NULL) + { + /* + * Check the file to make sure it isn't a directory or a backup + * file of some sort... + */ + + snprintf(filename, sizeof(filename), "%s/%s", d, dent->filename); + + if (S_ISDIR(dent->fileinfo.st_mode)) + continue; + + if (dent->filename[0] == '~' || + dent->filename[strlen(dent->filename) - 1] == '~') + continue; + + if ((ext = strrchr(dent->filename, '.')) != NULL) + if (!strcmp(ext, ".bck") || + !strcmp(ext, ".bak") || + !strcmp(ext, ".sav")) + continue; + + /* + * Must be a valid file; add it! + */ + + add_banner(dent->filename, filename); + } + + /* + * Close the directory... + */ + + cupsDirClose(dir); +} + + +/* + * 'add_banner()' - Add a banner to the array. + */ + +static void +add_banner(const char *name, /* I - Name of banner */ + const char *filename) /* I - Filename for banner */ +{ + mime_type_t *filetype; /* Filetype */ + cupsd_banner_t *temp; /* New banner data */ + + + /* + * See what the filetype is... + */ + + if ((filetype = mimeFileType(MimeDatabase, filename, NULL, NULL)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "add_banner: Banner \"%s\" (\"%s\") is of an unknown file " + "type - skipping!", name, filename); + return; + } + + /* + * Allocate memory... + */ + + if ((temp = calloc(1, sizeof(cupsd_banner_t))) == NULL) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "add_banner: Unable to allocate memory for banner \"%s\" - " + "skipping!", name); + return; + } + + /* + * Copy the new banner data over... + */ + + if ((temp->name = strdup(name)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "add_banner: Unable to allocate memory for banner \"%s\" - " + "skipping!", name); + free(temp); + return; + } + + temp->filetype = filetype; + + cupsArrayAdd(Banners, temp); +} + + +/* + * 'compare_banners()' - Compare two banners. + */ + +static int /* O - -1 if name0 < name1, etc. */ +compare_banners( + const cupsd_banner_t *b0, /* I - First banner */ + const cupsd_banner_t *b1) /* I - Second banner */ +{ + return (_cups_strcasecmp(b0->name, b1->name)); +} + + +/* + * 'free_banners()' - Free all banners. + */ + +static void +free_banners(void) +{ + cupsd_banner_t *temp; /* Current banner */ + + + for (temp = (cupsd_banner_t *)cupsArrayFirst(Banners); + temp; + temp = (cupsd_banner_t *)cupsArrayNext(Banners)) + { + free(temp->name); + free(temp); + } + + cupsArrayDelete(Banners); + Banners = NULL; +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/banners.h b/scheduler/banners.h new file mode 100644 index 0000000000..fb933aaae4 --- /dev/null +++ b/scheduler/banners.h @@ -0,0 +1,45 @@ +/* + * "$Id$" + * + * Banner definitions for the CUPS scheduler. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +/* + * Banner information structure... + */ + +typedef struct /**** Banner file information ****/ +{ + char *name; /* Name of banner */ + mime_type_t *filetype; /* Filetype for banner */ +} cupsd_banner_t; + + +/* + * Globals... + */ + +VAR cups_array_t *Banners VALUE(NULL); + /* Available banner files */ + + +/* + * Prototypes... + */ + +extern cupsd_banner_t *cupsdFindBanner(const char *name); +extern void cupsdLoadBanners(const char *d); + + +/* + * End of "$Id$". + */ diff --git a/scheduler/cert.c b/scheduler/cert.c new file mode 100644 index 0000000000..087e2564d2 --- /dev/null +++ b/scheduler/cert.c @@ -0,0 +1,442 @@ +/* + * "$Id$" + * + * Authentication certificate routines for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdAddCert() - Add a certificate. + * cupsdDeleteCert() - Delete a single certificate. + * cupsdDeleteAllCerts() - Delete all certificates... + * cupsdFindCert() - Find a certificate. + * cupsdInitCerts() - Initialize the certificate "system" and root + * certificate. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" +#ifdef HAVE_ACL_INIT +# include +# ifdef HAVE_MEMBERSHIP_H +# include +# endif /* HAVE_MEMBERSHIP_H */ +#endif /* HAVE_ACL_INIT */ + + +/* + * 'cupsdAddCert()' - Add a certificate. + */ + +void +cupsdAddCert(int pid, /* I - Process ID */ + const char *username, /* I - Username */ + void *ccache) /* I - Kerberos credentials or NULL */ +{ + int i; /* Looping var */ + cupsd_cert_t *cert; /* Current certificate */ + int fd; /* Certificate file */ + char filename[1024]; /* Certificate filename */ + static const char hex[] = "0123456789ABCDEF"; + /* Hex constants... */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdAddCert: Adding certificate for PID %d", pid); + + /* + * Allocate memory for the certificate... + */ + + if ((cert = calloc(sizeof(cupsd_cert_t), 1)) == NULL) + return; + + /* + * Fill in the certificate information... + */ + + cert->pid = pid; + strlcpy(cert->username, username, sizeof(cert->username)); + + for (i = 0; i < 32; i ++) + cert->certificate[i] = hex[CUPS_RAND() & 15]; + + /* + * Save the certificate to a file readable only by the User and Group + * (or root and SystemGroup for PID == 0)... + */ + + snprintf(filename, sizeof(filename), "%s/certs/%d", StateDir, pid); + unlink(filename); + + if ((fd = open(filename, O_WRONLY | O_CREAT | O_EXCL, 0400)) < 0) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to create certificate file %s - %s", + filename, strerror(errno)); + free(cert); + return; + } + + if (pid == 0) + { +#ifdef HAVE_ACL_INIT + acl_t acl; /* ACL information */ + acl_entry_t entry; /* ACL entry */ + acl_permset_t permset; /* Permissions */ +# ifdef HAVE_MBR_UID_TO_UUID + uuid_t group; /* Group ID */ +# endif /* HAVE_MBR_UID_TO_UUID */ + static int acls_not_supported = 0; + /* Only warn once */ +#endif /* HAVE_ACL_INIT */ + + + /* + * Root certificate... + */ + + fchmod(fd, 0440); + fchown(fd, RunUser, SystemGroupIDs[0]); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdAddCert: NumSystemGroups=%d", + NumSystemGroups); + +#ifdef HAVE_ACL_INIT + if (NumSystemGroups > 1) + { + /* + * Set POSIX ACLs for the root certificate so that all system + * groups can access it... + */ + +# ifdef HAVE_MBR_UID_TO_UUID + /* + * On MacOS X, ACLs use UUIDs instead of GIDs... + */ + + acl = acl_init(NumSystemGroups - 1); + + for (i = 1; i < NumSystemGroups; i ++) + { + /* + * Add each group ID to the ACL... + */ + + acl_create_entry(&acl, &entry); + acl_get_permset(entry, &permset); + acl_add_perm(permset, ACL_READ_DATA); + acl_set_tag_type(entry, ACL_EXTENDED_ALLOW); + mbr_gid_to_uuid((gid_t)SystemGroupIDs[i], group); + acl_set_qualifier(entry, &group); + acl_set_permset(entry, permset); + } +# else + /* + * POSIX ACLs need permissions for owner, group, other, and mask + * in addition to the rest of the system groups... + */ + + acl = acl_init(NumSystemGroups + 3); + + /* Owner */ + acl_create_entry(&acl, &entry); + acl_get_permset(entry, &permset); + acl_add_perm(permset, ACL_READ); + acl_set_tag_type(entry, ACL_USER_OBJ); + acl_set_permset(entry, permset); + + /* Group */ + acl_create_entry(&acl, &entry); + acl_get_permset(entry, &permset); + acl_add_perm(permset, ACL_READ); + acl_set_tag_type(entry, ACL_GROUP_OBJ); + acl_set_permset(entry, permset); + + /* Others */ + acl_create_entry(&acl, &entry); + acl_get_permset(entry, &permset); + acl_add_perm(permset, 0); + acl_set_tag_type(entry, ACL_OTHER); + acl_set_permset(entry, permset); + + /* Mask */ + acl_create_entry(&acl, &entry); + acl_get_permset(entry, &permset); + acl_add_perm(permset, ACL_READ); + acl_set_tag_type(entry, ACL_MASK); + acl_set_permset(entry, permset); + + for (i = 1; i < NumSystemGroups; i ++) + { + /* + * Add each group ID to the ACL... + */ + + acl_create_entry(&acl, &entry); + acl_get_permset(entry, &permset); + acl_add_perm(permset, ACL_READ); + acl_set_tag_type(entry, ACL_GROUP); + acl_set_qualifier(entry, SystemGroupIDs + i); + acl_set_permset(entry, permset); + } + + if (acl_valid(acl)) + { + char *text, *textptr; /* Temporary string */ + + + cupsdLogMessage(CUPSD_LOG_ERROR, "ACL did not validate: %s", + strerror(errno)); + text = acl_to_text(acl, NULL); + for (textptr = strchr(text, '\n'); + textptr; + textptr = strchr(textptr + 1, '\n')) + *textptr = ','; + + cupsdLogMessage(CUPSD_LOG_ERROR, "ACL: %s", text); + acl_free(text); + } +# endif /* HAVE_MBR_UID_TO_UUID */ + + if (acl_set_fd(fd, acl)) + { + if (errno != EOPNOTSUPP || !acls_not_supported) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to set ACLs on root certificate \"%s\" - %s", + filename, strerror(errno)); + + if (errno == EOPNOTSUPP) + acls_not_supported = 1; + } + + acl_free(acl); + } +#endif /* HAVE_ACL_INIT */ + + RootCertTime = time(NULL); + } + else + { + /* + * CGI certificate... + */ + + fchmod(fd, 0400); + fchown(fd, User, Group); + } + + DEBUG_printf(("ADD pid=%d, username=%s, cert=%s\n", pid, username, + cert->certificate)); + + write(fd, cert->certificate, strlen(cert->certificate)); + close(fd); + + /* + * Add Kerberos credentials as needed... + */ + +#ifdef HAVE_GSSAPI + cert->ccache = (krb5_ccache)ccache; +#else + (void)ccache; +#endif /* HAVE_GSSAPI */ + + /* + * Insert the certificate at the front of the list... + */ + + cert->next = Certs; + Certs = cert; +} + + +/* + * 'cupsdDeleteCert()' - Delete a single certificate. + */ + +void +cupsdDeleteCert(int pid) /* I - Process ID */ +{ + cupsd_cert_t *cert, /* Current certificate */ + *prev; /* Previous certificate */ + char filename[1024]; /* Certificate file */ + + + for (prev = NULL, cert = Certs; cert != NULL; prev = cert, cert = cert->next) + if (cert->pid == pid) + { + /* + * Remove this certificate from the list... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdDeleteCert: Removing certificate for PID %d", pid); + + DEBUG_printf(("DELETE pid=%d, username=%s, cert=%s\n", cert->pid, + cert->username, cert->certificate)); + + if (prev == NULL) + Certs = cert->next; + else + prev->next = cert->next; + +#ifdef HAVE_GSSAPI + /* + * Release Kerberos credentials as needed... + */ + + if (cert->ccache) + krb5_cc_destroy(KerberosContext, cert->ccache); +#endif /* HAVE_GSSAPI */ + + free(cert); + + /* + * Delete the file and return... + */ + + snprintf(filename, sizeof(filename), "%s/certs/%d", StateDir, pid); + if (unlink(filename)) + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to remove %s!", filename); + + return; + } +} + + +/* + * 'cupsdDeleteAllCerts()' - Delete all certificates... + */ + +void +cupsdDeleteAllCerts(void) +{ + cupsd_cert_t *cert, /* Current certificate */ + *next; /* Next certificate */ + char filename[1024]; /* Certificate file */ + + + /* + * Loop through each certificate, deleting them... + */ + + for (cert = Certs; cert != NULL; cert = next) + { + /* + * Delete the file... + */ + + snprintf(filename, sizeof(filename), "%s/certs/%d", StateDir, cert->pid); + if (unlink(filename)) + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to remove %s!", filename); + + /* + * Free memory... + */ + + next = cert->next; + free(cert); + } + + Certs = NULL; + RootCertTime = 0; +} + + +/* + * 'cupsdFindCert()' - Find a certificate. + */ + +cupsd_cert_t * /* O - Matching certificate or NULL */ +cupsdFindCert(const char *certificate) /* I - Certificate */ +{ + cupsd_cert_t *cert; /* Current certificate */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdFindCert(certificate=%s)", + certificate); + for (cert = Certs; cert != NULL; cert = cert->next) + if (!_cups_strcasecmp(certificate, cert->certificate)) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdFindCert: Returning %s...", + cert->username); + return (cert); + } + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdFindCert: Certificate not found!"); + + return (NULL); +} + + +/* + * 'cupsdInitCerts()' - Initialize the certificate "system" and root + * certificate. + */ + +void +cupsdInitCerts(void) +{ +#ifndef HAVE_ARC4RANDOM + cups_file_t *fp; /* /dev/random file */ + + + /* + * Initialize the random number generator using the random device or + * the current time, as available... + */ + + if ((fp = cupsFileOpen("/dev/urandom", "rb")) == NULL) + { + struct timeval tod; /* Time of day */ + + /* + * Get the time in usecs and use it as the initial seed... + */ + + gettimeofday(&tod, NULL); + + CUPS_SRAND((unsigned)(tod.tv_sec + tod.tv_usec)); + } + else + { + unsigned seed; /* Seed for random number generator */ + + /* + * Read 4 random characters from the random device and use + * them as the seed... + */ + + seed = cupsFileGetChar(fp); + seed = (seed << 8) | cupsFileGetChar(fp); + seed = (seed << 8) | cupsFileGetChar(fp); + CUPS_SRAND((seed << 8) | cupsFileGetChar(fp)); + + cupsFileClose(fp); + } +#endif /* !HAVE_ARC4RANDOM */ + + /* + * Create a root certificate and return... + */ + + if (!RunUser) + cupsdAddCert(0, "root", NULL); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/cert.h b/scheduler/cert.h new file mode 100644 index 0000000000..a8206b1427 --- /dev/null +++ b/scheduler/cert.h @@ -0,0 +1,56 @@ +/* + * "$Id$" + * + * Authentication certificate definitions for the CUPS scheduler. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +/* + * Certificate structure... + */ + +typedef struct cupsd_cert_s +{ + struct cupsd_cert_s *next; /* Next certificate in list */ + int pid; /* Process ID (0 for root certificate) */ + char certificate[33]; /* 32 hex characters, or 128 bits */ + char username[33]; /* Authenticated username */ +#ifdef HAVE_GSSAPI + krb5_ccache ccache; /* Kerberos credential cache */ +#endif /* HAVE_GSSAPI */ +} cupsd_cert_t; + + +/* + * Globals... + */ + +VAR cupsd_cert_t *Certs /* List of certificates */ + VALUE(NULL); +VAR time_t RootCertTime /* Root certificate update time */ + VALUE(0); + + +/* + * Prototypes... + */ + +extern void cupsdAddCert(int pid, const char *username, + void *ccache); +extern void cupsdDeleteCert(int pid); +extern void cupsdDeleteAllCerts(void); +extern cupsd_cert_t *cupsdFindCert(const char *certificate); +extern void cupsdInitCerts(void); + + +/* + * End of "$Id$". + */ diff --git a/scheduler/classes.c b/scheduler/classes.c new file mode 100644 index 0000000000..5ced554f59 --- /dev/null +++ b/scheduler/classes.c @@ -0,0 +1,818 @@ +/* + * "$Id$" + * + * Printer class routines for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdAddClass() - Add a class to the system. + * cupsdAddPrinterToClass() - Add a printer to a class... + * cupsdDeletePrinterFromClass() - Delete a printer from a class. + * cupsdDeletePrinterFromClasses() - Delete a printer from all classes. + * cupsdFindAvailablePrinter() - Find an available printer in a class. + * cupsdFindClass() - Find the named class. + * cupsdLoadAllClasses() - Load classes from the classes.conf file. + * cupsdSaveAllClasses() - Save classes to the classes.conf file. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" + + +/* + * 'cupsdAddClass()' - Add a class to the system. + */ + +cupsd_printer_t * /* O - New class */ +cupsdAddClass(const char *name) /* I - Name of class */ +{ + cupsd_printer_t *c; /* New class */ + char uri[1024]; /* Class URI */ + + + /* + * Add the printer and set the type to "class"... + */ + + if ((c = cupsdAddPrinter(name)) != NULL) + { + /* + * Change from a printer to a class... + */ + + c->type = CUPS_PRINTER_CLASS; + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + ServerName, RemotePort, "/classes/%s", name); + cupsdSetString(&c->uri, uri); + + cupsdSetString(&c->error_policy, "retry-current-job"); + } + + return (c); +} + + +/* + * 'cupsdAddPrinterToClass()' - Add a printer to a class... + */ + +void +cupsdAddPrinterToClass( + cupsd_printer_t *c, /* I - Class to add to */ + cupsd_printer_t *p) /* I - Printer to add */ +{ + int i; /* Looping var */ + cupsd_printer_t **temp; /* Pointer to printer array */ + + + /* + * See if this printer is already a member of the class... + */ + + for (i = 0; i < c->num_printers; i ++) + if (c->printers[i] == p) + return; + + /* + * Allocate memory as needed... + */ + + if (c->num_printers == 0) + temp = malloc(sizeof(cupsd_printer_t *)); + else + temp = realloc(c->printers, sizeof(cupsd_printer_t *) * (c->num_printers + 1)); + + if (temp == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to add printer %s to class %s!", + p->name, c->name); + return; + } + + /* + * Add the printer to the end of the array and update the number of printers. + */ + + c->printers = temp; + temp += c->num_printers; + c->num_printers ++; + + *temp = p; +} + + +/* + * 'cupsdDeletePrinterFromClass()' - Delete a printer from a class. + */ + +int /* O - 1 if class changed, 0 otherwise */ +cupsdDeletePrinterFromClass( + cupsd_printer_t *c, /* I - Class to delete from */ + cupsd_printer_t *p) /* I - Printer to delete */ +{ + int i; /* Looping var */ + + + /* + * See if the printer is in the class... + */ + + for (i = 0; i < c->num_printers; i ++) + if (p == c->printers[i]) + break; + + /* + * If it is, remove it from the list... + */ + + if (i < c->num_printers) + { + /* + * Yes, remove the printer... + */ + + c->num_printers --; + if (i < c->num_printers) + memmove(c->printers + i, c->printers + i + 1, + (c->num_printers - i) * sizeof(cupsd_printer_t *)); + } + else + return (0); + + /* + * Update the IPP attributes (have to do this for member-names)... + */ + + cupsdSetPrinterAttrs(c); + + return (1); +} + + +/* + * 'cupsdDeletePrinterFromClasses()' - Delete a printer from all classes. + */ + +int /* O - 1 if class changed, 0 otherwise */ +cupsdDeletePrinterFromClasses( + cupsd_printer_t *p) /* I - Printer to delete */ +{ + int changed = 0; /* Any class changed? */ + cupsd_printer_t *c; /* Pointer to current class */ + + + /* + * Loop through the printer/class list and remove the printer + * from each class listed... + */ + + for (c = (cupsd_printer_t *)cupsArrayFirst(Printers); + c; + c = (cupsd_printer_t *)cupsArrayNext(Printers)) + if (c->type & CUPS_PRINTER_CLASS) + changed |= cupsdDeletePrinterFromClass(c, p); + + return (changed); +} + + +/* + * 'cupsdFindAvailablePrinter()' - Find an available printer in a class. + */ + +cupsd_printer_t * /* O - Available printer or NULL */ +cupsdFindAvailablePrinter( + const char *name) /* I - Class to check */ +{ + int i; /* Looping var */ + cupsd_printer_t *c; /* Printer class */ + + + /* + * Find the class... + */ + + if ((c = cupsdFindClass(name)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to find class \"%s\"!", name); + return (NULL); + } + + if (c->num_printers == 0) + return (NULL); + + /* + * Make sure that the last printer is also a valid index into the printer + * array. If not, reset the last printer to 0... + */ + + if (c->last_printer >= c->num_printers) + c->last_printer = 0; + + /* + * Loop through the printers in the class and return the first idle + * printer... We keep track of the last printer that we used so that + * a "round robin" type of scheduling is realized (otherwise the first + * server might be saturated with print jobs...) + * + * Thanks to Joel Fredrikson for helping us get this right! + */ + + for (i = c->last_printer + 1; ; i ++) + { + if (i >= c->num_printers) + i = 0; + + if (c->printers[i]->accepting && + (c->printers[i]->state == IPP_PRINTER_IDLE || + ((c->printers[i]->type & CUPS_PRINTER_REMOTE) && !c->printers[i]->job))) + { + c->last_printer = i; + return (c->printers[i]); + } + + if (i == c->last_printer) + break; + } + + return (NULL); +} + + +/* + * 'cupsdFindClass()' - Find the named class. + */ + +cupsd_printer_t * /* O - Matching class or NULL */ +cupsdFindClass(const char *name) /* I - Name of class */ +{ + cupsd_printer_t *c; /* Current class/printer */ + + + if ((c = cupsdFindDest(name)) != NULL && (c->type & CUPS_PRINTER_CLASS)) + return (c); + else + return (NULL); +} + + +/* + * 'cupsdLoadAllClasses()' - Load classes from the classes.conf file. + */ + +void +cupsdLoadAllClasses(void) +{ + int i; /* Looping var */ + cups_file_t *fp; /* classes.conf file */ + int linenum; /* Current line number */ + char line[4096], /* Line from file */ + *value, /* Pointer to value */ + *valueptr; /* Pointer into value */ + cupsd_printer_t *p, /* Current printer class */ + *temp; /* Temporary pointer to printer */ + + + /* + * Open the classes.conf file... + */ + + snprintf(line, sizeof(line), "%s/classes.conf", ServerRoot); + if ((fp = cupsdOpenConfFile(line)) == NULL) + return; + + /* + * Read class configurations until we hit EOF... + */ + + linenum = 0; + p = NULL; + + while (cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + { + /* + * Decode the directive... + */ + + if (!_cups_strcasecmp(line, " or + */ + + if (p == NULL && value) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, "Loading class %s...", value); + + /* + * Since prior classes may have implicitly defined this class, + * see if it already exists... + */ + + if ((p = cupsdFindDest(value)) != NULL) + { + p->type = CUPS_PRINTER_CLASS; + cupsdSetStringf(&p->uri, "ipp://%s:%d/classes/%s", ServerName, + LocalPort, value); + cupsdSetString(&p->error_policy, "retry-job"); + } + else + p = cupsdAddClass(value); + + p->accepting = 1; + p->state = IPP_PRINTER_IDLE; + + if (!_cups_strcasecmp(line, "")) + { + if (p != NULL) + { + cupsdSetPrinterAttrs(p); + p = NULL; + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of classes.conf.", linenum); + } + else if (!p) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of classes.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "UUID")) + { + if (value && !strncmp(value, "urn:uuid:", 9)) + cupsdSetString(&(p->uuid), value); + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Bad UUID on line %d of classes.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "AuthInfoRequired")) + { + if (!cupsdSetAuthInfoRequired(p, value, NULL)) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Bad AuthInfoRequired on line %d of classes.conf.", + linenum); + } + else if (!_cups_strcasecmp(line, "Info")) + { + if (value) + cupsdSetString(&p->info, value); + } + else if (!_cups_strcasecmp(line, "Location")) + { + if (value) + cupsdSetString(&p->location, value); + } + else if (!_cups_strcasecmp(line, "Option") && value) + { + /* + * Option name value + */ + + for (valueptr = value; *valueptr && !isspace(*valueptr & 255); valueptr ++); + + if (!*valueptr) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of classes.conf.", linenum); + else + { + for (; *valueptr && isspace(*valueptr & 255); *valueptr++ = '\0'); + + p->num_options = cupsAddOption(value, valueptr, p->num_options, + &(p->options)); + } + } + else if (!_cups_strcasecmp(line, "Printer")) + { + if (!value) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of classes.conf.", linenum); + continue; + } + else if ((temp = cupsdFindPrinter(value)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "Unknown printer %s on line %d of classes.conf.", + value, linenum); + + /* + * Add the missing remote printer... + */ + + if ((temp = cupsdAddPrinter(value)) != NULL) + { + cupsdSetString(&temp->make_model, "Remote Printer on unknown"); + + temp->state = IPP_PRINTER_STOPPED; + temp->type |= CUPS_PRINTER_REMOTE; + + cupsdSetString(&temp->location, "Location Unknown"); + cupsdSetString(&temp->info, "No Information Available"); + temp->hostname[0] = '\0'; + + cupsdSetPrinterAttrs(temp); + } + } + + if (temp) + cupsdAddPrinterToClass(p, temp); + } + else if (!_cups_strcasecmp(line, "State")) + { + /* + * Set the initial queue state... + */ + + if (!_cups_strcasecmp(value, "idle")) + p->state = IPP_PRINTER_IDLE; + else if (!_cups_strcasecmp(value, "stopped")) + { + p->state = IPP_PRINTER_STOPPED; + + for (i = 0 ; i < p->num_reasons; i ++) + if (!strcmp("paused", p->reasons[i])) + break; + + if (i >= p->num_reasons && + p->num_reasons < (int)(sizeof(p->reasons) / sizeof(p->reasons[0]))) + { + p->reasons[p->num_reasons] = _cupsStrAlloc("paused"); + p->num_reasons ++; + } + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of classes.conf.", + linenum); + } + else if (!_cups_strcasecmp(line, "StateMessage")) + { + /* + * Set the initial queue state message... + */ + + if (value) + strlcpy(p->state_message, value, sizeof(p->state_message)); + } + else if (!_cups_strcasecmp(line, "StateTime")) + { + /* + * Set the state time... + */ + + if (value) + p->state_time = atoi(value); + } + else if (!_cups_strcasecmp(line, "Accepting")) + { + /* + * Set the initial accepting state... + */ + + if (value && + (!_cups_strcasecmp(value, "yes") || + !_cups_strcasecmp(value, "on") || + !_cups_strcasecmp(value, "true"))) + p->accepting = 1; + else if (value && + (!_cups_strcasecmp(value, "no") || + !_cups_strcasecmp(value, "off") || + !_cups_strcasecmp(value, "false"))) + p->accepting = 0; + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of classes.conf.", + linenum); + } + else if (!_cups_strcasecmp(line, "Shared")) + { + /* + * Set the initial shared state... + */ + + if (value && + (!_cups_strcasecmp(value, "yes") || + !_cups_strcasecmp(value, "on") || + !_cups_strcasecmp(value, "true"))) + p->shared = 1; + else if (value && + (!_cups_strcasecmp(value, "no") || + !_cups_strcasecmp(value, "off") || + !_cups_strcasecmp(value, "false"))) + p->shared = 0; + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of classes.conf.", + linenum); + } + else if (!_cups_strcasecmp(line, "JobSheets")) + { + /* + * Set the initial job sheets... + */ + + if (value) + { + for (valueptr = value; + *valueptr && !isspace(*valueptr & 255); + valueptr ++); + + if (*valueptr) + *valueptr++ = '\0'; + + cupsdSetString(&p->job_sheets[0], value); + + while (isspace(*valueptr & 255)) + valueptr ++; + + if (*valueptr) + { + for (value = valueptr; + *valueptr && !isspace(*valueptr & 255); + valueptr ++); + + if (*valueptr) + *valueptr = '\0'; + + cupsdSetString(&p->job_sheets[1], value); + } + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of classes.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "AllowUser")) + { + if (value) + { + p->deny_users = 0; + cupsdAddString(&(p->users), value); + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of classes.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "DenyUser")) + { + if (value) + { + p->deny_users = 1; + cupsdAddString(&(p->users), value); + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of classes.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "QuotaPeriod")) + { + if (value) + p->quota_period = atoi(value); + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of classes.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "PageLimit")) + { + if (value) + p->page_limit = atoi(value); + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of classes.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "KLimit")) + { + if (value) + p->k_limit = atoi(value); + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of classes.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "OpPolicy")) + { + if (value) + { + cupsd_policy_t *pol; /* Policy */ + + + if ((pol = cupsdFindPolicy(value)) != NULL) + { + cupsdSetString(&p->op_policy, value); + p->op_policy_ptr = pol; + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Bad policy \"%s\" on line %d of classes.conf", + value, linenum); + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of classes.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "ErrorPolicy")) + { + if (value) + { + if (strcmp(value, "retry-current-job") && strcmp(value, "retry-job")) + cupsdLogMessage(CUPSD_LOG_WARN, + "ErrorPolicy %s ignored on line %d of classes.conf", + value, linenum); + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of classes.conf.", linenum); + } + else + { + /* + * Something else we don't understand... + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unknown configuration directive %s on line %d of classes.conf.", + line, linenum); + } + } + + cupsFileClose(fp); +} + + +/* + * 'cupsdSaveAllClasses()' - Save classes to the classes.conf file. + */ + +void +cupsdSaveAllClasses(void) +{ + cups_file_t *fp; /* classes.conf file */ + char filename[1024], /* classes.conf filename */ + temp[1024], /* Temporary string */ + value[2048], /* Value string */ + *name; /* Current user name */ + cupsd_printer_t *pclass; /* Current printer class */ + int i; /* Looping var */ + time_t curtime; /* Current time */ + struct tm *curdate; /* Current date */ + cups_option_t *option; /* Current option */ + + + /* + * Create the classes.conf file... + */ + + snprintf(filename, sizeof(filename), "%s/classes.conf", ServerRoot); + + if ((fp = cupsdCreateConfFile(filename, ConfigFilePerm)) == NULL) + return; + + cupsdLogMessage(CUPSD_LOG_INFO, "Saving classes.conf..."); + + /* + * Write a small header to the file... + */ + + curtime = time(NULL); + curdate = localtime(&curtime); + strftime(temp, sizeof(temp) - 1, "%Y-%m-%d %H:%M", curdate); + + cupsFilePuts(fp, "# Class configuration file for " CUPS_SVERSION "\n"); + cupsFilePrintf(fp, "# Written by cupsd on %s\n", temp); + cupsFilePuts(fp, "# DO NOT EDIT THIS FILE WHEN CUPSD IS RUNNING\n"); + + /* + * Write each local class known to the system... + */ + + for (pclass = (cupsd_printer_t *)cupsArrayFirst(Printers); + pclass; + pclass = (cupsd_printer_t *)cupsArrayNext(Printers)) + { + /* + * Skip remote destinations and regular printers... + */ + + if ((pclass->type & CUPS_PRINTER_REMOTE) || + !(pclass->type & CUPS_PRINTER_CLASS)) + continue; + + /* + * Write printers as needed... + */ + + if (pclass == DefaultPrinter) + cupsFilePrintf(fp, "\n", pclass->name); + else + cupsFilePrintf(fp, "\n", pclass->name); + + cupsFilePrintf(fp, "UUID %s\n", pclass->uuid); + + if (pclass->num_auth_info_required > 0) + { + switch (pclass->num_auth_info_required) + { + case 1 : + strlcpy(value, pclass->auth_info_required[0], sizeof(value)); + break; + + case 2 : + snprintf(value, sizeof(value), "%s,%s", + pclass->auth_info_required[0], + pclass->auth_info_required[1]); + break; + + case 3 : + default : + snprintf(value, sizeof(value), "%s,%s,%s", + pclass->auth_info_required[0], + pclass->auth_info_required[1], + pclass->auth_info_required[2]); + break; + } + + cupsFilePutConf(fp, "AuthInfoRequired", value); + } + + if (pclass->info) + cupsFilePutConf(fp, "Info", pclass->info); + + if (pclass->location) + cupsFilePutConf(fp, "Location", pclass->location); + + if (pclass->state == IPP_PRINTER_STOPPED) + cupsFilePuts(fp, "State Stopped\n"); + else + cupsFilePuts(fp, "State Idle\n"); + + cupsFilePrintf(fp, "StateTime %d\n", (int)pclass->state_time); + + if (pclass->accepting) + cupsFilePuts(fp, "Accepting Yes\n"); + else + cupsFilePuts(fp, "Accepting No\n"); + + if (pclass->shared) + cupsFilePuts(fp, "Shared Yes\n"); + else + cupsFilePuts(fp, "Shared No\n"); + + snprintf(value, sizeof(value), "%s %s", pclass->job_sheets[0], + pclass->job_sheets[1]); + cupsFilePutConf(fp, "JobSheets", value); + + for (i = 0; i < pclass->num_printers; i ++) + cupsFilePrintf(fp, "Printer %s\n", pclass->printers[i]->name); + + cupsFilePrintf(fp, "QuotaPeriod %d\n", pclass->quota_period); + cupsFilePrintf(fp, "PageLimit %d\n", pclass->page_limit); + cupsFilePrintf(fp, "KLimit %d\n", pclass->k_limit); + + for (name = (char *)cupsArrayFirst(pclass->users); + name; + name = (char *)cupsArrayNext(pclass->users)) + cupsFilePutConf(fp, pclass->deny_users ? "DenyUser" : "AllowUser", name); + + if (pclass->op_policy) + cupsFilePutConf(fp, "OpPolicy", pclass->op_policy); + if (pclass->error_policy) + cupsFilePutConf(fp, "ErrorPolicy", pclass->error_policy); + + for (i = pclass->num_options, option = pclass->options; + i > 0; + i --, option ++) + { + snprintf(value, sizeof(value), "%s %s", option->name, option->value); + cupsFilePutConf(fp, "Option", value); + } + + cupsFilePuts(fp, "\n"); + } + + cupsdCloseCreatedConfFile(fp, filename); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/classes.h b/scheduler/classes.h new file mode 100644 index 0000000000..b24b7de117 --- /dev/null +++ b/scheduler/classes.h @@ -0,0 +1,35 @@ +/* + * "$Id$" + * + * Printer class definitions for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + + +/* + * Prototypes... + */ + +extern cupsd_printer_t *cupsdAddClass(const char *name); +extern void cupsdAddPrinterToClass(cupsd_printer_t *c, + cupsd_printer_t *p); +extern int cupsdDeletePrinterFromClass(cupsd_printer_t *c, + cupsd_printer_t *p); +extern int cupsdDeletePrinterFromClasses(cupsd_printer_t *p); +extern cupsd_printer_t *cupsdFindAvailablePrinter(const char *name); +extern cupsd_printer_t *cupsdFindClass(const char *name); +extern void cupsdLoadAllClasses(void); +extern void cupsdSaveAllClasses(void); + + +/* + * End of "$Id$". + */ diff --git a/scheduler/client.c b/scheduler/client.c new file mode 100644 index 0000000000..0fad604dea --- /dev/null +++ b/scheduler/client.c @@ -0,0 +1,5255 @@ +/* + * "$Id$" + * + * Client routines for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * This file contains Kerberos support code, copyright 2006 by + * Jelmer Vernooij. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdAcceptClient() - Accept a new client. + * cupsdCloseAllClients() - Close all remote clients immediately. + * cupsdCloseClient() - Close a remote client. + * cupsdFlushHeader() - Flush the header fields to the client. + * cupsdReadClient() - Read data from a client. + * cupsdSendCommand() - Send output from a command via HTTP. + * cupsdSendError() - Send an error message via HTTP. + * cupsdSendHeader() - Send an HTTP request. + * cupsdUpdateCGI() - Read status messages from CGI scripts and + * programs. + * cupsdWriteClient() - Write data to a client as needed. + * check_if_modified() - Decode an "If-Modified-Since" line. + * compare_clients() - Compare two client connections. + * copy_cdsa_certificate() - Copy a SSL/TLS certificate from the System + * keychain. + * data_ready() - Check whether data is available from a client. + * encrypt_client() - Enable encryption for the client... + * get_file() - Get a filename and state info. + * install_conf_file() - Install a configuration file. + * is_cgi() - Is the resource a CGI script/program? + * is_path_absolute() - Is a path absolute and free of relative elements + * (i.e. ".."). + * make_certificate() - Make a self-signed SSL/TLS certificate. + * pipe_command() - Pipe the output of a command to the remote + * client. + * valid_host() - Is the Host: field valid? + * write_file() - Send a file via HTTP. + * write_pipe() - Flag that data is available on the CGI pipe. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" + +#ifdef HAVE_TCPD_H +# include +#endif /* HAVE_TCPD_H */ + + +/* + * Local globals... + */ + +static const char * const http_states[] = + { /* HTTP state strings */ + "HTTP_WAITING", + "HTTP_OPTIONS", + "HTTP_GET", + "HTTP_GET_SEND", + "HTTP_HEAD", + "HTTP_POST", + "HTTP_POST_RECV", + "HTTP_POST_SEND", + "HTTP_PUT", + "HTTP_PUT_RECV", + "HTTP_DELETE", + "HTTP_TRACE", + "HTTP_CLOSE", + "HTTP_STATUS" + }; +static const char * const ipp_states[] = + { /* IPP state strings */ + "IPP_IDLE", + "IPP_HEADER", + "IPP_ATTRIBUTE", + "IPP_DATA" + }; + + +/* + * Local functions... + */ + +static int check_if_modified(cupsd_client_t *con, + struct stat *filestats); +static int compare_clients(cupsd_client_t *a, cupsd_client_t *b, + void *data); +#ifdef HAVE_CDSASSL +static CFArrayRef copy_cdsa_certificate(cupsd_client_t *con); +#endif /* HAVE_CDSASSL */ +static int data_ready(cupsd_client_t *con); +#ifdef HAVE_SSL +static int encrypt_client(cupsd_client_t *con); +#endif /* HAVE_SSL */ +static char *get_file(cupsd_client_t *con, struct stat *filestats, + char *filename, int len); +static http_status_t install_conf_file(cupsd_client_t *con); +static int is_cgi(cupsd_client_t *con, const char *filename, + struct stat *filestats, mime_type_t *type); +static int is_path_absolute(const char *path); +#ifdef HAVE_SSL +static int make_certificate(cupsd_client_t *con); +#endif /* HAVE_SSL */ +static int pipe_command(cupsd_client_t *con, int infile, int *outfile, + char *command, char *options, int root); +static int valid_host(cupsd_client_t *con); +static int write_file(cupsd_client_t *con, http_status_t code, + char *filename, char *type, + struct stat *filestats); +static void write_pipe(cupsd_client_t *con); + + +/* + * 'cupsdAcceptClient()' - Accept a new client. + */ + +void +cupsdAcceptClient(cupsd_listener_t *lis)/* I - Listener socket */ +{ + int count; /* Count of connections on a host */ + int val; /* Parameter value */ + cupsd_client_t *con, /* New client pointer */ + *tempcon; /* Temporary client pointer */ + http_addrlist_t *addrlist, /* List of adddresses for host */ + *addr; /* Current address */ + socklen_t addrlen; /* Length of address */ + char *hostname; /* Hostname for address */ + http_addr_t temp; /* Temporary address variable */ + static time_t last_dos = 0; /* Time of last DoS attack */ +#ifdef HAVE_TCPD_H + struct request_info wrap_req; /* TCP wrappers request information */ +#endif /* HAVE_TCPD_H */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdAcceptClient(lis=%p(%d)) Clients=%d", + lis, lis->fd, cupsArrayCount(Clients)); + + /* + * Make sure we don't have a full set of clients already... + */ + + if (cupsArrayCount(Clients) == MaxClients) + return; + + /* + * Get a pointer to the next available client... + */ + + if (!Clients) + Clients = cupsArrayNew(NULL, NULL); + + if (!Clients) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate memory for clients array!"); + cupsdPauseListening(); + return; + } + + if (!ActiveClients) + ActiveClients = cupsArrayNew((cups_array_func_t)compare_clients, NULL); + + if (!ActiveClients) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate memory for active clients array!"); + cupsdPauseListening(); + return; + } + + if ((con = calloc(1, sizeof(cupsd_client_t))) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to allocate memory for client!"); + cupsdPauseListening(); + return; + } + + con->file = -1; + con->http.activity = time(NULL); + con->http.hostaddr = &(con->clientaddr); + con->http.wait_value = 10000; + + /* + * Accept the client and get the remote address... + */ + + addrlen = sizeof(http_addr_t); + + if ((con->http.fd = accept(lis->fd, (struct sockaddr *)con->http.hostaddr, + &addrlen)) < 0) + { + if (errno == ENFILE || errno == EMFILE) + cupsdPauseListening(); + + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to accept client connection - %s.", + strerror(errno)); + free(con); + + return; + } + + /* + * Save the connected port number... + */ + + _httpAddrSetPort(con->http.hostaddr, _httpAddrPort(&(lis->address))); + +#ifdef AF_INET6 + /* + * Convert IPv4 over IPv6 addresses (::ffff:n.n.n.n) to IPv4 forms we + * can more easily use... + */ + + if (lis->address.addr.sa_family == AF_INET6 && + con->http.hostaddr->ipv6.sin6_addr.s6_addr32[0] == 0 && + con->http.hostaddr->ipv6.sin6_addr.s6_addr32[1] == 0 && + ntohl(con->http.hostaddr->ipv6.sin6_addr.s6_addr32[2]) == 0xffff) + con->http.hostaddr->ipv6.sin6_addr.s6_addr32[2] = 0; +#endif /* AF_INET6 */ + + /* + * Check the number of clients on the same address... + */ + + for (count = 0, tempcon = (cupsd_client_t *)cupsArrayFirst(Clients); + tempcon; + tempcon = (cupsd_client_t *)cupsArrayNext(Clients)) + if (httpAddrEqual(tempcon->http.hostaddr, con->http.hostaddr)) + { + count ++; + if (count >= MaxClientsPerHost) + break; + } + + if (count >= MaxClientsPerHost) + { + if ((time(NULL) - last_dos) >= 60) + { + last_dos = time(NULL); + cupsdLogMessage(CUPSD_LOG_WARN, + "Possible DoS attack - more than %d clients connecting " + "from %s!", + MaxClientsPerHost, + httpAddrString(con->http.hostaddr, con->http.hostname, + sizeof(con->http.hostname))); + } + +#ifdef WIN32 + closesocket(con->http.fd); +#else + close(con->http.fd); +#endif /* WIN32 */ + + free(con); + return; + } + + /* + * Get the hostname or format the IP address as needed... + */ + + if (httpAddrLocalhost(con->http.hostaddr)) + { + /* + * Map accesses from the loopback interface to "localhost"... + */ + + strlcpy(con->http.hostname, "localhost", sizeof(con->http.hostname)); + hostname = con->http.hostname; + } + else + { + /* + * Map accesses from the same host to the server name. + */ + + if (HostNameLookups) + hostname = httpAddrLookup(con->http.hostaddr, con->http.hostname, + sizeof(con->http.hostname)); + else + { + hostname = NULL; + httpAddrString(con->http.hostaddr, con->http.hostname, + sizeof(con->http.hostname)); + } + } + + if (hostname == NULL && HostNameLookups == 2) + { + /* + * Can't have an unresolved IP address with double-lookups enabled... + */ + +#ifdef WIN32 + closesocket(con->http.fd); +#else + close(con->http.fd); +#endif /* WIN32 */ + + cupsdLogMessage(CUPSD_LOG_WARN, + "Name lookup failed - connection from %s closed!", + con->http.hostname); + + free(con); + return; + } + + if (HostNameLookups == 2) + { + /* + * Do double lookups as needed... + */ + + if ((addrlist = httpAddrGetList(con->http.hostname, AF_UNSPEC, NULL)) + != NULL) + { + /* + * See if the hostname maps to the same IP address... + */ + + for (addr = addrlist; addr; addr = addr->next) + if (httpAddrEqual(con->http.hostaddr, &(addr->addr))) + break; + } + else + addr = NULL; + + httpAddrFreeList(addrlist); + + if (!addr) + { + /* + * Can't have a hostname that doesn't resolve to the same IP address + * with double-lookups enabled... + */ + +#ifdef WIN32 + closesocket(con->http.fd); +#else + close(con->http.fd); +#endif /* WIN32 */ + + cupsdLogMessage(CUPSD_LOG_WARN, + "IP lookup failed - connection from %s closed!", + con->http.hostname); + free(con); + return; + } + } + +#ifdef HAVE_TCPD_H + /* + * See if the connection is denied by TCP wrappers... + */ + + request_init(&wrap_req, RQ_DAEMON, "cupsd", RQ_FILE, con->http.fd, NULL); + fromhost(&wrap_req); + + if (!hosts_access(&wrap_req)) + { +#ifdef WIN32 + closesocket(con->http.fd); +#else + close(con->http.fd); +#endif /* WIN32 */ + + cupsdLogMessage(CUPSD_LOG_WARN, + "Connection from %s refused by /etc/hosts.allow and " + "/etc/hosts.deny rules.", con->http.hostname); + free(con); + return; + } +#endif /* HAVE_TCPD_H */ + +#ifdef AF_LOCAL + if (con->http.hostaddr->addr.sa_family == AF_LOCAL) + cupsdLogMessage(CUPSD_LOG_DEBUG, "[Client %d] Accepted from %s (Domain)", + con->http.fd, con->http.hostname); + else +#endif /* AF_LOCAL */ + cupsdLogMessage(CUPSD_LOG_DEBUG, "[Client %d] Accepted from %s:%d (IPv%d)", + con->http.fd, con->http.hostname, + _httpAddrPort(con->http.hostaddr), + _httpAddrFamily(con->http.hostaddr) == AF_INET ? 4 : 6); + + /* + * Get the local address the client connected to... + */ + + addrlen = sizeof(temp); + if (getsockname(con->http.fd, (struct sockaddr *)&temp, &addrlen)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to get local address - %s", + strerror(errno)); + + strcpy(con->servername, "localhost"); + con->serverport = LocalPort; + } +#ifdef AF_LOCAL + else if (_httpAddrFamily(&temp) == AF_LOCAL) + { + strcpy(con->servername, "localhost"); + con->serverport = LocalPort; + } +#endif /* AF_LOCAL */ + else + { + if (httpAddrLocalhost(&temp)) + strlcpy(con->servername, "localhost", sizeof(con->servername)); + else if (HostNameLookups || RemotePort) + httpAddrLookup(&temp, con->servername, sizeof(con->servername)); + else + httpAddrString(&temp, con->servername, sizeof(con->servername)); + + con->serverport = _httpAddrPort(&(lis->address)); + } + + /* + * Add the connection to the array of active clients... + */ + + cupsArrayAdd(Clients, con); + + /* + * Using TCP_NODELAY improves responsiveness, especially on systems with a slow + * loopback interface. Since we write large buffers when sending print files + * and requests there shouldn't be any performance penalty for this... + */ + + val = 1; + setsockopt(con->http.fd, IPPROTO_TCP, TCP_NODELAY, &val, sizeof(val)); + + /* + * Close this file on all execs... + */ + + fcntl(con->http.fd, F_SETFD, fcntl(con->http.fd, F_GETFD) | FD_CLOEXEC); + + /* + * Add the socket to the server select. + */ + + cupsdAddSelect(con->http.fd, (cupsd_selfunc_t)cupsdReadClient, NULL, con); + + /* + * Temporarily suspend accept()'s until we lose a client... + */ + + if (cupsArrayCount(Clients) == MaxClients) + cupsdPauseListening(); + +#ifdef HAVE_SSL + /* + * See if we are connecting on a secure port... + */ + + if (lis->encryption == HTTP_ENCRYPT_ALWAYS) + { + /* + * https connection; go secure... + */ + + con->http.encryption = HTTP_ENCRYPT_ALWAYS; + + if (!encrypt_client(con)) + cupsdCloseClient(con); + } + else + con->auto_ssl = 1; +#endif /* HAVE_SSL */ +} + + +/* + * 'cupsdCloseAllClients()' - Close all remote clients immediately. + */ + +void +cupsdCloseAllClients(void) +{ + cupsd_client_t *con; /* Current client */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdCloseAllClients() Clients=%d", + cupsArrayCount(Clients)); + + for (con = (cupsd_client_t *)cupsArrayFirst(Clients); + con; + con = (cupsd_client_t *)cupsArrayNext(Clients)) + if (cupsdCloseClient(con)) + cupsdCloseClient(con); +} + + +/* + * 'cupsdCloseClient()' - Close a remote client. + */ + +int /* O - 1 if partial close, 0 if fully closed */ +cupsdCloseClient(cupsd_client_t *con) /* I - Client to close */ +{ + int partial; /* Do partial close for SSL? */ +#ifdef HAVE_LIBSSL + SSL_CTX *context; /* Context for encryption */ + unsigned long error; /* Error code */ +#elif defined(HAVE_GNUTLS) + int error; /* Error code */ + gnutls_certificate_server_credentials *credentials; + /* TLS credentials */ +# elif defined(HAVE_CDSASSL) +#endif /* HAVE_LIBSSL */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG, "[Client %d] Closing connection.", + con->http.fd); + + /* + * Flush pending writes before closing... + */ + + httpFlushWrite(HTTP(con)); + + partial = 0; + +#ifdef HAVE_SSL + /* + * Shutdown encryption as needed... + */ + + if (con->http.tls) + { + partial = 1; + +# ifdef HAVE_LIBSSL + context = SSL_get_SSL_CTX(con->http.tls); + + switch (SSL_shutdown(con->http.tls)) + { + case 1 : + cupsdLogMessage(CUPSD_LOG_DEBUG, + "SSL shutdown successful!"); + break; + case -1 : + cupsdLogMessage(CUPSD_LOG_ERROR, + "Fatal error during SSL shutdown!"); + default : + while ((error = ERR_get_error()) != 0) + cupsdLogMessage(CUPSD_LOG_ERROR, "SSL shutdown failed: %s", + ERR_error_string(error, NULL)); + break; + } + + SSL_CTX_free(context); + SSL_free(con->http.tls); + +# elif defined(HAVE_GNUTLS) + credentials = (gnutls_certificate_server_credentials *)(con->http.tls_credentials); + + error = gnutls_bye(con->http.tls, GNUTLS_SHUT_WR); + switch (error) + { + case GNUTLS_E_SUCCESS: + cupsdLogMessage(CUPSD_LOG_DEBUG, + "SSL shutdown successful!"); + break; + default: + cupsdLogMessage(CUPSD_LOG_ERROR, + "SSL shutdown failed: %s", gnutls_strerror(error)); + break; + } + + gnutls_deinit(con->http.tls); + gnutls_certificate_free_credentials(*credentials); + free(credentials); + +# elif defined(HAVE_CDSASSL) + while (SSLClose(con->http.tls) == errSSLWouldBlock) + usleep(1000); + + SSLDisposeContext(con->http.tls); + + if (con->http.tls_credentials) + CFRelease(con->http.tls_credentials); + +# endif /* HAVE_LIBSSL */ + + con->http.tls = NULL; + } +#endif /* HAVE_SSL */ + + if (con->pipe_pid != 0) + { + /* + * Stop any CGI process... + */ + + cupsdEndProcess(con->pipe_pid, 1); + con->pipe_pid = 0; + } + + if (con->file >= 0) + { + cupsdRemoveSelect(con->file); + + close(con->file); + con->file = -1; + } + + /* + * Close the socket and clear the file from the input set for select()... + */ + + if (con->http.fd >= 0) + { + cupsArrayRemove(ActiveClients, con); + cupsdSetBusyState(); + + if (partial) + { + /* + * Only do a partial close so that the encrypted client gets everything. + */ + + shutdown(con->http.fd, 0); + cupsdAddSelect(con->http.fd, (cupsd_selfunc_t)cupsdReadClient, NULL, con); + } + else + { + /* + * Shut the socket down fully... + */ + + cupsdRemoveSelect(con->http.fd); + close(con->http.fd); + con->http.fd = -1; + } + } + + if (!partial) + { + /* + * Free memory... + */ + + if (con->http.input_set) + free(con->http.input_set); + + httpClearCookie(HTTP(con)); + httpClearFields(HTTP(con)); + + cupsdClearString(&con->filename); + cupsdClearString(&con->command); + cupsdClearString(&con->options); + cupsdClearString(&con->query_string); + + if (con->request) + { + ippDelete(con->request); + con->request = NULL; + } + + if (con->response) + { + ippDelete(con->response); + con->response = NULL; + } + + if (con->language) + { + cupsLangFree(con->language); + con->language = NULL; + } + +#ifdef HAVE_AUTHORIZATION_H + if (con->authref) + { + AuthorizationFree(con->authref, kAuthorizationFlagDefaults); + con->authref = NULL; + } +#endif /* HAVE_AUTHORIZATION_H */ + + /* + * Re-enable new client connections if we are going back under the + * limit... + */ + + if (cupsArrayCount(Clients) == MaxClients) + cupsdResumeListening(); + + /* + * Compact the list of clients as necessary... + */ + + cupsArrayRemove(Clients, con); + + free(con); + } + + return (partial); +} + + +/* + * 'cupsdFlushHeader()' - Flush the header fields to the client. + */ + +int /* I - Bytes written or -1 on error */ +cupsdFlushHeader(cupsd_client_t *con) /* I - Client to flush to */ +{ + int bytes = httpFlushWrite(HTTP(con)); + + con->http.data_encoding = HTTP_ENCODE_LENGTH; + + return (bytes); +} + + +/* + * 'cupsdReadClient()' - Read data from a client. + */ + +void +cupsdReadClient(cupsd_client_t *con) /* I - Client to read from */ +{ + char line[32768], /* Line from client... */ + operation[64], /* Operation code from socket */ + version[64], /* HTTP version number string */ + locale[64], /* Locale */ + *ptr; /* Pointer into strings */ + int major, minor; /* HTTP version numbers */ + http_status_t status; /* Transfer status */ + ipp_state_t ipp_state; /* State of IPP transfer */ + int bytes; /* Number of bytes to POST */ + char *filename; /* Name of file for GET/HEAD */ + char buf[1024]; /* Buffer for real filename */ + struct stat filestats; /* File information */ + mime_type_t *type; /* MIME type of file */ + cupsd_printer_t *p; /* Printer */ + static unsigned request_id = 0; /* Request ID for temp files */ + + + status = HTTP_CONTINUE; + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] cupsdReadClient " + "error=%d, " + "used=%d, " + "state=%s, " + "data_encoding=HTTP_ENCODE_%s, " + "data_remaining=" CUPS_LLFMT ", " + "request=%p(%s), " + "file=%d", + con->http.fd, con->http.error, con->http.used, + http_states[con->http.state], + con->http.data_encoding == HTTP_ENCODE_CHUNKED ? + "CHUNKED" : "LENGTH", + CUPS_LLCAST con->http.data_remaining, + con->request, + con->request ? ipp_states[con->request->state] : "", + con->file); + +#ifdef HAVE_SSL + if (con->auto_ssl) + { + /* + * Automatically check for a SSL/TLS handshake... + */ + + con->auto_ssl = 0; + + if (recv(con->http.fd, buf, 1, MSG_PEEK) == 1 && + (!buf[0] || !strchr("DGHOPT", buf[0]))) + { + /* + * Encrypt this connection... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] Saw first byte %02X, auto-negotiating " + "SSL/TLS session.", con->http.fd, buf[0] & 255); + + if (!encrypt_client(con)) + cupsdCloseClient(con); + + return; + } + } +#endif /* HAVE_SSL */ + + switch (con->http.state) + { + case HTTP_WAITING : + /* + * See if we've received a request line... + */ + + if (httpGets(line, sizeof(line) - 1, HTTP(con)) == NULL) + { + if (con->http.error && con->http.error != EPIPE) + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] HTTP_WAITING Closing for error %d " + "(%s)", con->http.fd, con->http.error, + strerror(con->http.error)); + else + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] HTTP_WAITING Closing on EOF", + con->http.fd); + + cupsdCloseClient(con); + return; + } + + /* + * Ignore blank request lines... + */ + + if (line[0] == '\0') + break; + + /* + * Clear other state variables... + */ + + httpClearFields(HTTP(con)); + + con->http.activity = time(NULL); + con->http.version = HTTP_1_0; + con->http.keep_alive = HTTP_KEEPALIVE_OFF; + con->http.data_encoding = HTTP_ENCODE_LENGTH; + con->http.data_remaining = 0; + con->http._data_remaining = 0; + con->operation = HTTP_WAITING; + con->bytes = 0; + con->file = -1; + con->file_ready = 0; + con->pipe_pid = 0; + con->username[0] = '\0'; + con->password[0] = '\0'; + con->uri[0] = '\0'; + + cupsdClearString(&con->command); + cupsdClearString(&con->options); + cupsdClearString(&con->query_string); + + if (con->request) + { + ippDelete(con->request); + con->request = NULL; + } + + if (con->response) + { + ippDelete(con->response); + con->response = NULL; + } + + if (con->language) + { + cupsLangFree(con->language); + con->language = NULL; + } + +#ifdef HAVE_GSSAPI + con->have_gss = 0; + con->gss_uid = 0; +#endif /* HAVE_GSSAPI */ + + /* + * Grab the request line... + */ + + switch (sscanf(line, "%63s%1023s%63s", operation, con->uri, version)) + { + case 1 : + if (line[0]) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Bad request line \"%s\" from %s.", + con->http.fd, + _httpEncodeURI(buf, line, sizeof(buf)), + con->http.hostname); + cupsdSendError(con, HTTP_BAD_REQUEST, CUPSD_AUTH_NONE); + cupsdCloseClient(con); + } + return; + case 2 : + con->http.version = HTTP_0_9; + break; + case 3 : + if (sscanf(version, "HTTP/%d.%d", &major, &minor) != 2) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Bad request line \"%s\" from %s.", + con->http.fd, + _httpEncodeURI(buf, line, sizeof(buf)), + con->http.hostname); + cupsdSendError(con, HTTP_BAD_REQUEST, CUPSD_AUTH_NONE); + cupsdCloseClient(con); + return; + } + + if (major < 2) + { + con->http.version = (http_version_t)(major * 100 + minor); + if (con->http.version == HTTP_1_1 && KeepAlive) + con->http.keep_alive = HTTP_KEEPALIVE_ON; + else + con->http.keep_alive = HTTP_KEEPALIVE_OFF; + } + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Unsupported request line \"%s\" " + "from %s.", con->http.fd, + _httpEncodeURI(buf, line, sizeof(buf)), + con->http.hostname); + cupsdSendError(con, HTTP_NOT_SUPPORTED, CUPSD_AUTH_NONE); + cupsdCloseClient(con); + return; + } + break; + } + + /* + * Handle full URLs in the request line... + */ + + if (strcmp(con->uri, "*")) + { + char scheme[HTTP_MAX_URI], /* Method/scheme */ + userpass[HTTP_MAX_URI], /* Username:password */ + hostname[HTTP_MAX_URI], /* Hostname */ + resource[HTTP_MAX_URI]; /* Resource path */ + int port; /* Port number */ + + + /* + * Separate the URI into its components... + */ + + httpSeparateURI(HTTP_URI_CODING_MOST, con->uri, + scheme, sizeof(scheme), + userpass, sizeof(userpass), + hostname, sizeof(hostname), &port, + resource, sizeof(resource)); + + /* + * Only allow URIs with the servername, localhost, or an IP + * address... + */ + + if (strcmp(scheme, "file") && + _cups_strcasecmp(hostname, ServerName) && + _cups_strcasecmp(hostname, "localhost") && + !isdigit(hostname[0]) && hostname[0] != '[') + { + /* + * Nope, we don't do proxies... + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Bad URI \"%s\" in request.", + con->http.fd, con->uri); + cupsdSendError(con, HTTP_METHOD_NOT_ALLOWED, CUPSD_AUTH_NONE); + cupsdCloseClient(con); + return; + } + + /* + * Copy the resource portion back into the URI; both resource and + * con->uri are HTTP_MAX_URI bytes in size... + */ + + strcpy(con->uri, resource); + } + + /* + * Process the request... + */ + + if (!strcmp(operation, "GET")) + con->http.state = HTTP_GET; + else if (!strcmp(operation, "PUT")) + con->http.state = HTTP_PUT; + else if (!strcmp(operation, "POST")) + con->http.state = HTTP_POST; + else if (!strcmp(operation, "DELETE")) + con->http.state = HTTP_DELETE; + else if (!strcmp(operation, "TRACE")) + con->http.state = HTTP_TRACE; + else if (!strcmp(operation, "OPTIONS")) + con->http.state = HTTP_OPTIONS; + else if (!strcmp(operation, "HEAD")) + con->http.state = HTTP_HEAD; + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Bad operation \"%s\".", con->http.fd, + operation); + cupsdSendError(con, HTTP_BAD_REQUEST, CUPSD_AUTH_NONE); + cupsdCloseClient(con); + return; + } + + gettimeofday(&(con->start), NULL); + con->operation = con->http.state; + + cupsdLogMessage(CUPSD_LOG_DEBUG, "[Client %d] %s %s HTTP/%d.%d", + con->http.fd, operation, con->uri, + con->http.version / 100, con->http.version % 100); + + con->http.status = HTTP_OK; + + if (!cupsArrayFind(ActiveClients, con)) + { + cupsArrayAdd(ActiveClients, con); + cupsdSetBusyState(); + } + + case HTTP_OPTIONS : + case HTTP_DELETE : + case HTTP_GET : + case HTTP_HEAD : + case HTTP_POST : + case HTTP_PUT : + case HTTP_TRACE : + /* + * Parse incoming parameters until the status changes... + */ + + while ((status = httpUpdate(HTTP(con))) == HTTP_CONTINUE) + if (!data_ready(con)) + break; + + if (status != HTTP_OK && status != HTTP_CONTINUE) + { + if (con->http.error && con->http.error != EPIPE) + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Closing for error %d (%s) while " + "reading headers.", + con->http.fd, con->http.error, + strerror(con->http.error)); + else + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Closing on EOF while reading headers.", + con->http.fd); + + cupsdSendError(con, HTTP_BAD_REQUEST, CUPSD_AUTH_NONE); + cupsdCloseClient(con); + return; + } + break; + + default : + if (!data_ready(con) && recv(con->http.fd, buf, 1, MSG_PEEK) < 1) + { + /* + * Connection closed... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Closing on EOF", con->http.fd); + cupsdCloseClient(con); + return; + } + break; /* Anti-compiler-warning-code */ + } + + /* + * Handle new transfers... + */ + + if (status == HTTP_OK) + { + if (con->http.fields[HTTP_FIELD_ACCEPT_LANGUAGE][0]) + { + /* + * Figure out the locale from the Accept-Language and Content-Type + * fields... + */ + + if ((ptr = strchr(con->http.fields[HTTP_FIELD_ACCEPT_LANGUAGE], + ',')) != NULL) + *ptr = '\0'; + + if ((ptr = strchr(con->http.fields[HTTP_FIELD_ACCEPT_LANGUAGE], + ';')) != NULL) + *ptr = '\0'; + + if ((ptr = strstr(con->http.fields[HTTP_FIELD_CONTENT_TYPE], + "charset=")) != NULL) + { + /* + * Combine language and charset, and trim any extra params in the + * content-type. + */ + + snprintf(locale, sizeof(locale), "%s.%s", + con->http.fields[HTTP_FIELD_ACCEPT_LANGUAGE], ptr + 8); + + if ((ptr = strchr(locale, ',')) != NULL) + *ptr = '\0'; + } + else + snprintf(locale, sizeof(locale), "%s.UTF-8", + con->http.fields[HTTP_FIELD_ACCEPT_LANGUAGE]); + + con->language = cupsLangGet(locale); + } + else + con->language = cupsLangGet(DefaultLocale); + + cupsdAuthorize(con); + + if (!_cups_strncasecmp(con->http.fields[HTTP_FIELD_CONNECTION], + "Keep-Alive", 10) && KeepAlive) + con->http.keep_alive = HTTP_KEEPALIVE_ON; + else if (!_cups_strncasecmp(con->http.fields[HTTP_FIELD_CONNECTION], + "close", 5)) + con->http.keep_alive = HTTP_KEEPALIVE_OFF; + + if (!con->http.fields[HTTP_FIELD_HOST][0] && + con->http.version >= HTTP_1_1) + { + /* + * HTTP/1.1 and higher require the "Host:" field... + */ + + if (!cupsdSendError(con, HTTP_BAD_REQUEST, CUPSD_AUTH_NONE)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Missing Host: field in request.", + con->http.fd); + cupsdCloseClient(con); + return; + } + } + else if (!valid_host(con)) + { + /* + * Access to localhost must use "localhost" or the corresponding IPv4 + * or IPv6 values in the Host: field. + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Request from \"%s\" using invalid Host: " + "field \"%s\"", con->http.fd, con->http.hostname, + con->http.fields[HTTP_FIELD_HOST]); + + if (!cupsdSendError(con, HTTP_BAD_REQUEST, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + } + else if (con->operation == HTTP_OPTIONS) + { + /* + * Do OPTIONS command... + */ + + if (con->best && con->best->type != CUPSD_AUTH_NONE) + { + if (!cupsdSendHeader(con, HTTP_UNAUTHORIZED, NULL, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + } + + if (!_cups_strcasecmp(con->http.fields[HTTP_FIELD_CONNECTION], "Upgrade") && + con->http.tls == NULL) + { +#ifdef HAVE_SSL + /* + * Do encryption stuff... + */ + + if (!cupsdSendHeader(con, HTTP_SWITCHING_PROTOCOLS, NULL, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + httpPrintf(HTTP(con), "Connection: Upgrade\r\n"); + httpPrintf(HTTP(con), "Upgrade: TLS/1.0,HTTP/1.1\r\n"); + httpPrintf(HTTP(con), "Content-Length: 0\r\n"); + httpPrintf(HTTP(con), "\r\n"); + + if (cupsdFlushHeader(con) < 0) + { + cupsdCloseClient(con); + return; + } + + if (!encrypt_client(con)) + { + cupsdCloseClient(con); + return; + } +#else + if (!cupsdSendError(con, HTTP_NOT_IMPLEMENTED, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } +#endif /* HAVE_SSL */ + } + + if (!cupsdSendHeader(con, HTTP_OK, NULL, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + httpPrintf(HTTP(con), "Allow: GET, HEAD, OPTIONS, POST, PUT\r\n"); + httpPrintf(HTTP(con), "Content-Length: 0\r\n"); + httpPrintf(HTTP(con), "\r\n"); + + if (cupsdFlushHeader(con) < 0) + { + cupsdCloseClient(con); + return; + } + } + else if (!is_path_absolute(con->uri)) + { + /* + * Protect against malicious users! + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Request for non-absolute resource \"%s\".", + con->http.fd, con->uri); + + if (!cupsdSendError(con, HTTP_FORBIDDEN, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + } + else + { + if (!_cups_strcasecmp(con->http.fields[HTTP_FIELD_CONNECTION], + "Upgrade") && con->http.tls == NULL) + { +#ifdef HAVE_SSL + /* + * Do encryption stuff... + */ + + if (!cupsdSendHeader(con, HTTP_SWITCHING_PROTOCOLS, NULL, + CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + httpPrintf(HTTP(con), "Connection: Upgrade\r\n"); + httpPrintf(HTTP(con), "Upgrade: TLS/1.0,HTTP/1.1\r\n"); + httpPrintf(HTTP(con), "Content-Length: 0\r\n"); + httpPrintf(HTTP(con), "\r\n"); + + if (cupsdFlushHeader(con) < 0) + { + cupsdCloseClient(con); + return; + } + + if (!encrypt_client(con)) + { + cupsdCloseClient(con); + return; + } +#else + if (!cupsdSendError(con, HTTP_NOT_IMPLEMENTED, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } +#endif /* HAVE_SSL */ + } + + if ((status = cupsdIsAuthorized(con, NULL)) != HTTP_OK) + { + cupsdSendError(con, status, CUPSD_AUTH_NONE); + cupsdCloseClient(con); + return; + } + + if (con->http.expect && + (con->operation == HTTP_POST || con->operation == HTTP_PUT)) + { + if (con->http.expect == HTTP_CONTINUE) + { + /* + * Send 100-continue header... + */ + + if (!cupsdSendHeader(con, HTTP_CONTINUE, NULL, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + } + else + { + /* + * Send 417-expectation-failed header... + */ + + if (!cupsdSendHeader(con, HTTP_EXPECTATION_FAILED, NULL, + CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + httpPrintf(HTTP(con), "Content-Length: 0\r\n"); + httpPrintf(HTTP(con), "\r\n"); + + if (cupsdFlushHeader(con) < 0) + { + cupsdCloseClient(con); + return; + } + } + } + + switch (con->http.state) + { + case HTTP_GET_SEND : + if (!strncmp(con->uri, "/printers/", 10) && + !strcmp(con->uri + strlen(con->uri) - 4, ".ppd")) + { + /* + * Send PPD file - get the real printer name since printer + * names are not case sensitive but filenames can be... + */ + + con->uri[strlen(con->uri) - 4] = '\0'; /* Drop ".ppd" */ + + if ((p = cupsdFindPrinter(con->uri + 10)) != NULL) + snprintf(con->uri, sizeof(con->uri), "/ppd/%s.ppd", p->name); + else + { + if (!cupsdSendError(con, HTTP_NOT_FOUND, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + break; + } + } + else if ((!strncmp(con->uri, "/printers/", 10) || + !strncmp(con->uri, "/classes/", 9)) && + !strcmp(con->uri + strlen(con->uri) - 4, ".png")) + { + /* + * Send icon file - get the real queue name since queue names are + * not case sensitive but filenames can be... + */ + + con->uri[strlen(con->uri) - 4] = '\0'; /* Drop ".png" */ + + if (!strncmp(con->uri, "/printers/", 10)) + p = cupsdFindPrinter(con->uri + 10); + else + p = cupsdFindClass(con->uri + 9); + + if (p) + snprintf(con->uri, sizeof(con->uri), "/icons/%s.png", p->name); + else + { + if (!cupsdSendError(con, HTTP_NOT_FOUND, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + break; + } + } + else if (!WebInterface) + { + /* + * Web interface is disabled. Show an appropriate message... + */ + + if (!cupsdSendError(con, HTTP_WEBIF_DISABLED, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + break; + } + + if ((!strncmp(con->uri, "/admin", 6) && + strncmp(con->uri, "/admin/conf/", 12) && + strncmp(con->uri, "/admin/log/", 11)) || + !strncmp(con->uri, "/printers", 9) || + !strncmp(con->uri, "/classes", 8) || + !strncmp(con->uri, "/help", 5) || + !strncmp(con->uri, "/jobs", 5)) + { + /* + * Send CGI output... + */ + + if (!strncmp(con->uri, "/admin", 6)) + { + cupsdSetStringf(&con->command, "%s/cgi-bin/admin.cgi", + ServerBin); + + cupsdSetString(&con->options, strchr(con->uri + 6, '?')); + } + else if (!strncmp(con->uri, "/printers", 9)) + { + cupsdSetStringf(&con->command, "%s/cgi-bin/printers.cgi", + ServerBin); + + if (con->uri[9] && con->uri[10]) + cupsdSetString(&con->options, con->uri + 9); + else + cupsdSetString(&con->options, NULL); + } + else if (!strncmp(con->uri, "/classes", 8)) + { + cupsdSetStringf(&con->command, "%s/cgi-bin/classes.cgi", + ServerBin); + + if (con->uri[8] && con->uri[9]) + cupsdSetString(&con->options, con->uri + 8); + else + cupsdSetString(&con->options, NULL); + } + else if (!strncmp(con->uri, "/jobs", 5)) + { + cupsdSetStringf(&con->command, "%s/cgi-bin/jobs.cgi", + ServerBin); + + if (con->uri[5] && con->uri[6]) + cupsdSetString(&con->options, con->uri + 5); + else + cupsdSetString(&con->options, NULL); + } + else + { + cupsdSetStringf(&con->command, "%s/cgi-bin/help.cgi", + ServerBin); + + if (con->uri[5] && con->uri[6]) + cupsdSetString(&con->options, con->uri + 5); + else + cupsdSetString(&con->options, NULL); + } + + if (!cupsdSendCommand(con, con->command, con->options, 0)) + { + if (!cupsdSendError(con, HTTP_NOT_FOUND, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + } + else + cupsdLogRequest(con, HTTP_OK); + + if (con->http.version <= HTTP_1_0) + con->http.keep_alive = HTTP_KEEPALIVE_OFF; + } + else if ((!strncmp(con->uri, "/admin/conf/", 12) && + (strchr(con->uri + 12, '/') || + strlen(con->uri) == 12)) || + (!strncmp(con->uri, "/admin/log/", 11) && + (strchr(con->uri + 11, '/') || + strlen(con->uri) == 11))) + { + /* + * GET can only be done to configuration files directly under + * /admin/conf... + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, + "Request for subdirectory \"%s\"!", con->uri); + + if (!cupsdSendError(con, HTTP_FORBIDDEN, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + break; + } + else + { + /* + * Serve a file... + */ + + if ((filename = get_file(con, &filestats, buf, + sizeof(buf))) == NULL) + { + if (!cupsdSendError(con, HTTP_NOT_FOUND, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + break; + } + + type = mimeFileType(MimeDatabase, filename, NULL, NULL); + + if (is_cgi(con, filename, &filestats, type)) + { + /* + * Note: con->command and con->options were set by + * is_cgi()... + */ + + if (!cupsdSendCommand(con, con->command, con->options, 0)) + { + if (!cupsdSendError(con, HTTP_NOT_FOUND, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + } + else + cupsdLogRequest(con, HTTP_OK); + + if (con->http.version <= HTTP_1_0) + con->http.keep_alive = HTTP_KEEPALIVE_OFF; + break; + } + + if (!check_if_modified(con, &filestats)) + { + if (!cupsdSendError(con, HTTP_NOT_MODIFIED, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + } + else + { + if (type == NULL) + strcpy(line, "text/plain"); + else + snprintf(line, sizeof(line), "%s/%s", type->super, type->type); + + if (!write_file(con, HTTP_OK, filename, line, &filestats)) + { + cupsdCloseClient(con); + return; + } + } + } + break; + + case HTTP_POST_RECV : + /* + * See if the POST request includes a Content-Length field, and if + * so check the length against any limits that are set... + */ + + if (con->http.fields[HTTP_FIELD_CONTENT_LENGTH][0] && + MaxRequestSize > 0 && + con->http.data_remaining > MaxRequestSize) + { + /* + * Request too large... + */ + + if (!cupsdSendError(con, HTTP_REQUEST_TOO_LARGE, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + break; + } + else if (con->http.data_remaining < 0 || + (!con->http.fields[HTTP_FIELD_CONTENT_LENGTH][0] && + con->http.data_encoding == HTTP_ENCODE_LENGTH)) + { + /* + * Negative content lengths are invalid! + */ + + if (!cupsdSendError(con, HTTP_BAD_REQUEST, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + break; + } + + /* + * See what kind of POST request this is; for IPP requests the + * content-type field will be "application/ipp"... + */ + + if (!strcmp(con->http.fields[HTTP_FIELD_CONTENT_TYPE], + "application/ipp")) + con->request = ippNew(); + else if (!WebInterface) + { + /* + * Web interface is disabled. Show an appropriate message... + */ + + if (!cupsdSendError(con, HTTP_WEBIF_DISABLED, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + break; + } + else if ((!strncmp(con->uri, "/admin", 6) && + strncmp(con->uri, "/admin/conf/", 12) && + strncmp(con->uri, "/admin/log/", 11)) || + !strncmp(con->uri, "/printers", 9) || + !strncmp(con->uri, "/classes", 8) || + !strncmp(con->uri, "/help", 5) || + !strncmp(con->uri, "/jobs", 5)) + { + /* + * CGI request... + */ + + if (!strncmp(con->uri, "/admin", 6)) + { + cupsdSetStringf(&con->command, "%s/cgi-bin/admin.cgi", + ServerBin); + + cupsdSetString(&con->options, strchr(con->uri + 6, '?')); + } + else if (!strncmp(con->uri, "/printers", 9)) + { + cupsdSetStringf(&con->command, "%s/cgi-bin/printers.cgi", + ServerBin); + + if (con->uri[9] && con->uri[10]) + cupsdSetString(&con->options, con->uri + 9); + else + cupsdSetString(&con->options, NULL); + } + else if (!strncmp(con->uri, "/classes", 8)) + { + cupsdSetStringf(&con->command, "%s/cgi-bin/classes.cgi", + ServerBin); + + if (con->uri[8] && con->uri[9]) + cupsdSetString(&con->options, con->uri + 8); + else + cupsdSetString(&con->options, NULL); + } + else if (!strncmp(con->uri, "/jobs", 5)) + { + cupsdSetStringf(&con->command, "%s/cgi-bin/jobs.cgi", + ServerBin); + + if (con->uri[5] && con->uri[6]) + cupsdSetString(&con->options, con->uri + 5); + else + cupsdSetString(&con->options, NULL); + } + else + { + cupsdSetStringf(&con->command, "%s/cgi-bin/help.cgi", + ServerBin); + + if (con->uri[5] && con->uri[6]) + cupsdSetString(&con->options, con->uri + 5); + else + cupsdSetString(&con->options, NULL); + } + + if (con->http.version <= HTTP_1_0) + con->http.keep_alive = HTTP_KEEPALIVE_OFF; + } + else + { + /* + * POST to a file... + */ + + if ((filename = get_file(con, &filestats, buf, + sizeof(buf))) == NULL) + { + if (!cupsdSendError(con, HTTP_NOT_FOUND, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + break; + } + + type = mimeFileType(MimeDatabase, filename, NULL, NULL); + + if (!is_cgi(con, filename, &filestats, type)) + { + /* + * Only POST to CGI's... + */ + + if (!cupsdSendError(con, HTTP_UNAUTHORIZED, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + } + } + break; + + case HTTP_PUT_RECV : + /* + * Validate the resource name... + */ + + if (strncmp(con->uri, "/admin/conf/", 12) || + strchr(con->uri + 12, '/') || + strlen(con->uri) == 12) + { + /* + * PUT can only be done to configuration files under + * /admin/conf... + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Request for subdirectory \"%s\".", + con->http.fd, con->uri); + + if (!cupsdSendError(con, HTTP_FORBIDDEN, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + break; + } + + /* + * See if the PUT request includes a Content-Length field, and if + * so check the length against any limits that are set... + */ + + if (con->http.fields[HTTP_FIELD_CONTENT_LENGTH][0] && + MaxRequestSize > 0 && + con->http.data_remaining > MaxRequestSize) + { + /* + * Request too large... + */ + + if (!cupsdSendError(con, HTTP_REQUEST_TOO_LARGE, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + break; + } + else if (con->http.data_remaining < 0) + { + /* + * Negative content lengths are invalid! + */ + + if (!cupsdSendError(con, HTTP_BAD_REQUEST, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + break; + } + + /* + * Open a temporary file to hold the request... + */ + + cupsdSetStringf(&con->filename, "%s/%08x", RequestRoot, + request_id ++); + con->file = open(con->filename, O_WRONLY | O_CREAT | O_TRUNC, 0640); + + if (con->file < 0) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Unable to create request file " + "\"%s\": %s", con->http.fd, con->filename, + strerror(errno)); + + if (!cupsdSendError(con, HTTP_REQUEST_TOO_LARGE, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + } + + fchmod(con->file, 0640); + fchown(con->file, RunUser, Group); + fcntl(con->file, F_SETFD, fcntl(con->file, F_GETFD) | FD_CLOEXEC); + break; + + case HTTP_DELETE : + case HTTP_TRACE : + cupsdSendError(con, HTTP_NOT_IMPLEMENTED, CUPSD_AUTH_NONE); + cupsdCloseClient(con); + return; + + case HTTP_HEAD : + if (!strncmp(con->uri, "/printers/", 10) && + !strcmp(con->uri + strlen(con->uri) - 4, ".ppd")) + { + /* + * Send PPD file - get the real printer name since printer + * names are not case sensitive but filenames can be... + */ + + con->uri[strlen(con->uri) - 4] = '\0'; /* Drop ".ppd" */ + + if ((p = cupsdFindPrinter(con->uri + 10)) != NULL) + snprintf(con->uri, sizeof(con->uri), "/ppd/%s.ppd", p->name); + else + { + if (!cupsdSendError(con, HTTP_NOT_FOUND, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + break; + } + } + else if (!strncmp(con->uri, "/printers/", 10) && + !strcmp(con->uri + strlen(con->uri) - 4, ".png")) + { + /* + * Send PNG file - get the real printer name since printer + * names are not case sensitive but filenames can be... + */ + + con->uri[strlen(con->uri) - 4] = '\0'; /* Drop ".ppd" */ + + if ((p = cupsdFindPrinter(con->uri + 10)) != NULL) + snprintf(con->uri, sizeof(con->uri), "/icons/%s.png", p->name); + else + { + if (!cupsdSendError(con, HTTP_NOT_FOUND, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + break; + } + } + else if (!WebInterface) + { + if (!cupsdSendHeader(con, HTTP_OK, line, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + if (httpPrintf(HTTP(con), "\r\n") < 0) + { + cupsdCloseClient(con); + return; + } + + if (cupsdFlushHeader(con) < 0) + { + cupsdCloseClient(con); + return; + } + + con->http.state = HTTP_WAITING; + break; + } + + if ((!strncmp(con->uri, "/admin", 6) && + strncmp(con->uri, "/admin/conf/", 12) && + strncmp(con->uri, "/admin/log/", 11)) || + !strncmp(con->uri, "/printers", 9) || + !strncmp(con->uri, "/classes", 8) || + !strncmp(con->uri, "/help", 5) || + !strncmp(con->uri, "/jobs", 5)) + { + /* + * CGI output... + */ + + if (!cupsdSendHeader(con, HTTP_OK, "text/html", CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + if (httpPrintf(HTTP(con), "\r\n") < 0) + { + cupsdCloseClient(con); + return; + } + + if (cupsdFlushHeader(con) < 0) + { + cupsdCloseClient(con); + return; + } + + cupsdLogRequest(con, HTTP_OK); + } + else if ((!strncmp(con->uri, "/admin/conf/", 12) && + (strchr(con->uri + 12, '/') || + strlen(con->uri) == 12)) || + (!strncmp(con->uri, "/admin/log/", 11) && + (strchr(con->uri + 11, '/') || + strlen(con->uri) == 11))) + { + /* + * HEAD can only be done to configuration files under + * /admin/conf... + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Request for subdirectory \"%s\".", + con->http.fd, con->uri); + + if (!cupsdSendError(con, HTTP_FORBIDDEN, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + break; + } + else if ((filename = get_file(con, &filestats, buf, + sizeof(buf))) == NULL) + { + if (!cupsdSendHeader(con, HTTP_NOT_FOUND, "text/html", + CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + cupsdLogRequest(con, HTTP_NOT_FOUND); + } + else if (!check_if_modified(con, &filestats)) + { + if (!cupsdSendError(con, HTTP_NOT_MODIFIED, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + cupsdLogRequest(con, HTTP_NOT_MODIFIED); + } + else + { + /* + * Serve a file... + */ + + type = mimeFileType(MimeDatabase, filename, NULL, NULL); + if (type == NULL) + strcpy(line, "text/plain"); + else + snprintf(line, sizeof(line), "%s/%s", type->super, type->type); + + if (!cupsdSendHeader(con, HTTP_OK, line, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + if (httpPrintf(HTTP(con), "Last-Modified: %s\r\n", + httpGetDateString(filestats.st_mtime)) < 0) + { + cupsdCloseClient(con); + return; + } + + if (httpPrintf(HTTP(con), "Content-Length: %lu\r\n", + (unsigned long)filestats.st_size) < 0) + { + cupsdCloseClient(con); + return; + } + + cupsdLogRequest(con, HTTP_OK); + } + + if (httpPrintf(HTTP(con), "\r\n") < 0) + { + cupsdCloseClient(con); + return; + } + + if (cupsdFlushHeader(con) < 0) + { + cupsdCloseClient(con); + return; + } + + con->http.state = HTTP_WAITING; + break; + + default : + break; /* Anti-compiler-warning-code */ + } + } + } + + /* + * Handle any incoming data... + */ + + switch (con->http.state) + { + case HTTP_PUT_RECV : + do + { + if ((bytes = httpRead2(HTTP(con), line, sizeof(line))) < 0) + { + if (con->http.error && con->http.error != EPIPE) + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] HTTP_PUT_RECV Closing for error " + "%d (%s)", con->http.fd, con->http.error, + strerror(con->http.error)); + else + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] HTTP_PUT_RECV Closing on EOF", + con->http.fd); + + cupsdCloseClient(con); + return; + } + else if (bytes > 0) + { + con->bytes += bytes; + + if (write(con->file, line, bytes) < bytes) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Unable to write %d bytes to " + "\"%s\": %s", con->http.fd, bytes, con->filename, + strerror(errno)); + + close(con->file); + con->file = -1; + unlink(con->filename); + cupsdClearString(&con->filename); + + if (!cupsdSendError(con, HTTP_REQUEST_TOO_LARGE, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + } + } + } + while (con->http.state == HTTP_PUT_RECV && data_ready(con)); + + if (con->http.state == HTTP_WAITING) + { + /* + * End of file, see how big it is... + */ + + fstat(con->file, &filestats); + + close(con->file); + con->file = -1; + + if (filestats.st_size > MaxRequestSize && + MaxRequestSize > 0) + { + /* + * Request is too big; remove it and send an error... + */ + + unlink(con->filename); + cupsdClearString(&con->filename); + + if (!cupsdSendError(con, HTTP_REQUEST_TOO_LARGE, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + } + + /* + * Install the configuration file... + */ + + status = install_conf_file(con); + + /* + * Return the status to the client... + */ + + if (!cupsdSendError(con, status, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + } + break; + + case HTTP_POST_RECV : + do + { + if (con->request && con->file < 0) + { + /* + * Grab any request data from the connection... + */ + + if ((ipp_state = ippRead(&(con->http), con->request)) == IPP_ERROR) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] IPP read error: %s", con->http.fd, + cupsLastErrorString()); + + cupsdSendError(con, HTTP_BAD_REQUEST, CUPSD_AUTH_NONE); + cupsdCloseClient(con); + return; + } + else if (ipp_state != IPP_DATA) + { + if (con->http.state == HTTP_POST_SEND) + { + cupsdSendError(con, HTTP_BAD_REQUEST, CUPSD_AUTH_NONE); + cupsdCloseClient(con); + return; + } + + break; + } + else + { + cupsdLogMessage(CUPSD_LOG_DEBUG, "[Client %d] %d.%d %s %d", + con->http.fd, con->request->request.op.version[0], + con->request->request.op.version[1], + ippOpString(con->request->request.op.operation_id), + con->request->request.op.request_id); + con->bytes += ippLength(con->request); + } + } + + if (con->file < 0 && con->http.state != HTTP_POST_SEND) + { + /* + * Create a file as needed for the request data... + */ + + cupsdSetStringf(&con->filename, "%s/%08x", RequestRoot, + request_id ++); + con->file = open(con->filename, O_WRONLY | O_CREAT | O_TRUNC, 0640); + + if (con->file < 0) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Unable to create request file " + "\"%s\": %s", con->http.fd, con->filename, + strerror(errno)); + + if (!cupsdSendError(con, HTTP_REQUEST_TOO_LARGE, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + } + + fchmod(con->file, 0640); + fchown(con->file, RunUser, Group); + fcntl(con->file, F_SETFD, fcntl(con->file, F_GETFD) | FD_CLOEXEC); + } + + if (con->http.state != HTTP_POST_SEND) + { + if ((bytes = httpRead2(HTTP(con), line, sizeof(line))) < 0) + { + if (con->http.error && con->http.error != EPIPE) + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] HTTP_POST_SEND Closing for " + "error %d (%s)", con->http.fd, con->http.error, + strerror(con->http.error)); + else + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] HTTP_POST_SEND Closing on EOF", + con->http.fd); + + cupsdCloseClient(con); + return; + } + else if (bytes > 0) + { + con->bytes += bytes; + + if (write(con->file, line, bytes) < bytes) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Unable to write %d bytes to " + "\"%s\": %s", con->http.fd, bytes, + con->filename, strerror(errno)); + + close(con->file); + con->file = -1; + unlink(con->filename); + cupsdClearString(&con->filename); + + if (!cupsdSendError(con, HTTP_REQUEST_TOO_LARGE, + CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + } + } + else if (con->http.state == HTTP_POST_RECV) + return; + else if (con->http.state != HTTP_POST_SEND) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Closing on unexpected state %s.", + con->http.fd, http_states[con->http.state]); + cupsdCloseClient(con); + return; + } + } + } + while (con->http.state == HTTP_POST_RECV && data_ready(con)); + + if (con->http.state == HTTP_POST_SEND) + { + if (con->file >= 0) + { + fstat(con->file, &filestats); + + close(con->file); + con->file = -1; + + if (filestats.st_size > MaxRequestSize && + MaxRequestSize > 0) + { + /* + * Request is too big; remove it and send an error... + */ + + unlink(con->filename); + cupsdClearString(&con->filename); + + if (con->request) + { + /* + * Delete any IPP request data... + */ + + ippDelete(con->request); + con->request = NULL; + } + + if (!cupsdSendError(con, HTTP_REQUEST_TOO_LARGE, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + } + else if (filestats.st_size == 0) + { + /* + * Don't allow empty file... + */ + + unlink(con->filename); + cupsdClearString(&con->filename); + } + + if (con->command) + { + if (!cupsdSendCommand(con, con->command, con->options, 0)) + { + if (!cupsdSendError(con, HTTP_NOT_FOUND, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + } + else + cupsdLogRequest(con, HTTP_OK); + } + } + + if (con->request) + { + cupsdProcessIPPRequest(con); + + if (con->filename) + { + unlink(con->filename); + cupsdClearString(&con->filename); + } + + return; + } + } + break; + + default : + break; /* Anti-compiler-warning-code */ + } + + if (con->http.state == HTTP_WAITING) + { + if (!con->http.keep_alive) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Closing because Keep-Alive disabled", + con->http.fd); + cupsdCloseClient(con); + } + else + { + cupsArrayRemove(ActiveClients, con); + cupsdSetBusyState(); + } + } +} + + +/* + * 'cupsdSendCommand()' - Send output from a command via HTTP. + */ + +int /* O - 1 on success, 0 on failure */ +cupsdSendCommand( + cupsd_client_t *con, /* I - Client connection */ + char *command, /* I - Command to run */ + char *options, /* I - Command-line options */ + int root) /* I - Run as root? */ +{ + int fd; /* Standard input file descriptor */ + + + if (con->filename) + { + fd = open(con->filename, O_RDONLY); + + if (fd < 0) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Client %d] Unable to open \"%s\" for reading: %s", + con->http.fd, con->filename ? con->filename : "/dev/null", + strerror(errno)); + return (0); + } + + fcntl(fd, F_SETFD, fcntl(fd, F_GETFD) | FD_CLOEXEC); + } + else + fd = -1; + + con->pipe_pid = pipe_command(con, fd, &(con->file), command, options, root); + + if (fd >= 0) + close(fd); + + cupsdLogMessage(CUPSD_LOG_INFO, "[Client %d] Started \"%s\" (pid=%d)", + con->http.fd, command, con->pipe_pid); + + cupsdLogMessage(CUPSD_LOG_DEBUG, "[Client %d] file=%d", con->http.fd, + con->file); + + if (con->pipe_pid == 0) + return (0); + + fcntl(con->file, F_SETFD, fcntl(con->file, F_GETFD) | FD_CLOEXEC); + + cupsdAddSelect(con->file, (cupsd_selfunc_t)write_pipe, NULL, con); + + con->sent_header = 0; + con->file_ready = 0; + con->got_fields = 0; + con->header_used = 0; + + return (1); +} + + +/* + * 'cupsdSendError()' - Send an error message via HTTP. + */ + +int /* O - 1 if successful, 0 otherwise */ +cupsdSendError(cupsd_client_t *con, /* I - Connection */ + http_status_t code, /* I - Error code */ + int auth_type)/* I - Authentication type */ +{ + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] cupsdSendError code=%d, auth_type=%d", + con->http.fd, code, auth_type); + +#ifdef HAVE_SSL + /* + * Force client to upgrade for authentication if that is how the + * server is configured... + */ + + if (code == HTTP_UNAUTHORIZED && + DefaultEncryption == HTTP_ENCRYPT_REQUIRED && + _cups_strcasecmp(con->http.hostname, "localhost") && + !con->http.tls) + { + code = HTTP_UPGRADE_REQUIRED; + } +#endif /* HAVE_SSL */ + + /* + * Put the request in the access_log file... + */ + + cupsdLogRequest(con, code); + + /* + * To work around bugs in some proxies, don't use Keep-Alive for some + * error messages... + * + * Kerberos authentication doesn't work without Keep-Alive, so + * never disable it in that case. + */ + + if (code >= HTTP_BAD_REQUEST && con->http.auth_type != CUPSD_AUTH_NEGOTIATE) + con->http.keep_alive = HTTP_KEEPALIVE_OFF; + + /* + * Send an error message back to the client. If the error code is a + * 400 or 500 series, make sure the message contains some text, too! + */ + + if (!cupsdSendHeader(con, code, NULL, auth_type)) + return (0); + +#ifdef HAVE_SSL + if (code == HTTP_UPGRADE_REQUIRED) + if (httpPrintf(HTTP(con), "Connection: Upgrade\r\n") < 0) + return (0); + + if (httpPrintf(HTTP(con), "Upgrade: TLS/1.0,HTTP/1.1\r\n") < 0) + return (0); +#endif /* HAVE_SSL */ + + if (con->http.version >= HTTP_1_1 && + con->http.keep_alive == HTTP_KEEPALIVE_OFF) + { + if (httpPrintf(HTTP(con), "Connection: close\r\n") < 0) + return (0); + } + + if (code >= HTTP_BAD_REQUEST) + { + /* + * Send a human-readable error message. + */ + + char message[4096], /* Message for user */ + urltext[1024], /* URL redirection text */ + redirect[1024]; /* Redirection link */ + const char *text; /* Status-specific text */ + + + redirect[0] = '\0'; + + if (code == HTTP_UNAUTHORIZED) + text = _cupsLangString(con->language, + _("Enter your username and password or the " + "root username and password to access this " + "page. If you are using Kerberos authentication, " + "make sure you have a valid Kerberos ticket.")); + else if (code == HTTP_UPGRADE_REQUIRED) + { + text = urltext; + + snprintf(urltext, sizeof(urltext), + _cupsLangString(con->language, + _("You must access this page using the URL " + "" + "https://%s:%d%s.")), + con->servername, con->serverport, con->uri, + con->servername, con->serverport, con->uri); + + snprintf(redirect, sizeof(redirect), + "\n", + con->servername, con->serverport, con->uri); + } + else if (code == HTTP_WEBIF_DISABLED) + text = _cupsLangString(con->language, + _("The web interface is currently disabled. Run " + "\"cupsctl WebInterface=yes\" to enable it.")); + else + text = ""; + + snprintf(message, sizeof(message), + "\n" + "\n" + "\n" + "\t\n" + "\t%s - " CUPS_SVERSION "\n" + "\t\n" + "%s" + "\n" + "\n" + "

%s

\n" + "

%s

\n" + "\n" + "\n", + httpStatus(code), redirect, httpStatus(code), text); + + if (httpPrintf(HTTP(con), "Content-Type: text/html; charset=utf-8\r\n") < 0) + return (0); + if (httpPrintf(HTTP(con), "Content-Length: %d\r\n", + (int)strlen(message)) < 0) + return (0); + if (httpPrintf(HTTP(con), "\r\n") < 0) + return (0); + if (httpPrintf(HTTP(con), "%s", message) < 0) + return (0); + } + else if (httpPrintf(HTTP(con), "\r\n") < 0) + return (0); + + if (cupsdFlushHeader(con) < 0) + return (0); + + con->http.state = HTTP_WAITING; + + return (1); +} + + +/* + * 'cupsdSendHeader()' - Send an HTTP request. + */ + +int /* O - 1 on success, 0 on failure */ +cupsdSendHeader( + cupsd_client_t *con, /* I - Client to send to */ + http_status_t code, /* I - HTTP status code */ + char *type, /* I - MIME type of document */ + int auth_type) /* I - Type of authentication */ +{ + char auth_str[1024]; /* Authorization string */ + + + /* + * Send the HTTP status header... + */ + + if (code == HTTP_CONTINUE) + { + /* + * 100-continue doesn't send any headers... + */ + + return (httpPrintf(HTTP(con), "HTTP/%d.%d 100 Continue\r\n\r\n", + con->http.version / 100, con->http.version % 100) > 0); + } + else if (code == HTTP_WEBIF_DISABLED) + { + /* + * Treat our special "web interface is disabled" status as "200 OK" for web + * browsers. + */ + + code = HTTP_OK; + } + + httpFlushWrite(HTTP(con)); + + con->http.data_encoding = HTTP_ENCODE_FIELDS; + + if (httpPrintf(HTTP(con), "HTTP/%d.%d %d %s\r\n", con->http.version / 100, + con->http.version % 100, code, httpStatus(code)) < 0) + return (0); + if (httpPrintf(HTTP(con), "Date: %s\r\n", httpGetDateString(time(NULL))) < 0) + return (0); + if (ServerHeader) + if (httpPrintf(HTTP(con), "Server: %s\r\n", ServerHeader) < 0) + return (0); + if (con->http.keep_alive && con->http.version >= HTTP_1_0) + { + if (httpPrintf(HTTP(con), "Connection: Keep-Alive\r\n") < 0) + return (0); + if (httpPrintf(HTTP(con), "Keep-Alive: timeout=%d\r\n", + KeepAliveTimeout) < 0) + return (0); + } + if (code == HTTP_METHOD_NOT_ALLOWED) + if (httpPrintf(HTTP(con), "Allow: GET, HEAD, OPTIONS, POST, PUT\r\n") < 0) + return (0); + + if (code == HTTP_UNAUTHORIZED) + { + if (auth_type == CUPSD_AUTH_NONE) + { + if (!con->best || con->best->type <= CUPSD_AUTH_NONE) + auth_type = cupsdDefaultAuthType(); + else + auth_type = con->best->type; + } + + auth_str[0] = '\0'; + + if (auth_type == CUPSD_AUTH_BASIC || auth_type == CUPSD_AUTH_BASICDIGEST) + strlcpy(auth_str, "Basic realm=\"CUPS\"", sizeof(auth_str)); + else if (auth_type == CUPSD_AUTH_DIGEST) + snprintf(auth_str, sizeof(auth_str), "Digest realm=\"CUPS\", nonce=\"%s\"", + con->http.hostname); +#ifdef HAVE_GSSAPI + else if (auth_type == CUPSD_AUTH_NEGOTIATE) + { +# ifdef AF_LOCAL + if (_httpAddrFamily(con->http.hostaddr) == AF_LOCAL) + strlcpy(auth_str, "Basic realm=\"CUPS\"", sizeof(auth_str)); + else +# endif /* AF_LOCAL */ + strlcpy(auth_str, "Negotiate", sizeof(auth_str)); + } +#endif /* HAVE_GSSAPI */ + + if (con->best && auth_type != CUPSD_AUTH_NEGOTIATE && + !_cups_strcasecmp(con->http.hostname, "localhost")) + { + /* + * Add a "trc" (try root certification) parameter for local non-Kerberos + * requests when the request requires system group membership - then the + * client knows the root certificate can/should be used. + * + * Also, for Mac OS X we also look for @AUTHKEY and add an "authkey" + * parameter as needed... + */ + + char *name, /* Current user name */ + *auth_key; /* Auth key buffer */ + size_t auth_size; /* Size of remaining buffer */ + + auth_key = auth_str + strlen(auth_str); + auth_size = sizeof(auth_str) - (auth_key - auth_str); + + for (name = (char *)cupsArrayFirst(con->best->names); + name; + name = (char *)cupsArrayNext(con->best->names)) + { +#ifdef HAVE_AUTHORIZATION_H + if (!_cups_strncasecmp(name, "@AUTHKEY(", 9)) + { + snprintf(auth_key, auth_size, ", authkey=\"%s\"", name + 9); + /* end parenthesis is stripped in conf.c */ + break; + } + else +#endif /* HAVE_AUTHORIZATION_H */ + if (!_cups_strcasecmp(name, "@SYSTEM")) + { +#ifdef HAVE_AUTHORIZATION_H + if (SystemGroupAuthKey) + snprintf(auth_key, auth_size, + ", authkey=\"%s\"", + SystemGroupAuthKey); + else +#else + strlcpy(auth_key, ", trc=\"y\"", auth_size); +#endif /* HAVE_AUTHORIZATION_H */ + break; + } + } + } + + if (auth_str[0]) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] WWW-Authenticate: %s", con->http.fd, + auth_str); + + if (httpPrintf(HTTP(con), "WWW-Authenticate: %s\r\n", auth_str) < 0) + return (0); + } + } + + if (con->language && strcmp(con->language->language, "C")) + { + if (httpPrintf(HTTP(con), "Content-Language: %s\r\n", + con->language->language) < 0) + return (0); + } + + if (type) + { + if (!strcmp(type, "text/html")) + { + if (httpPrintf(HTTP(con), + "Content-Type: text/html; charset=utf-8\r\n") < 0) + return (0); + } + else if (httpPrintf(HTTP(con), "Content-Type: %s\r\n", type) < 0) + return (0); + } + + return (1); +} + + +/* + * 'cupsdUpdateCGI()' - Read status messages from CGI scripts and programs. + */ + +void +cupsdUpdateCGI(void) +{ + char *ptr, /* Pointer to end of line in buffer */ + message[1024]; /* Pointer to message text */ + int loglevel; /* Log level for message */ + + + while ((ptr = cupsdStatBufUpdate(CGIStatusBuffer, &loglevel, + message, sizeof(message))) != NULL) + { + if (loglevel == CUPSD_LOG_INFO) + cupsdLogMessage(CUPSD_LOG_INFO, "%s", message); + + if (!strchr(CGIStatusBuffer->buffer, '\n')) + break; + } + + if (ptr == NULL && !CGIStatusBuffer->bufused) + { + /* + * Fatal error on pipe - should never happen! + */ + + cupsdLogMessage(CUPSD_LOG_CRIT, + "cupsdUpdateCGI: error reading from CGI error pipe - %s", + strerror(errno)); + } +} + + +/* + * 'cupsdWriteClient()' - Write data to a client as needed. + */ + +void +cupsdWriteClient(cupsd_client_t *con) /* I - Client connection */ +{ + int bytes, /* Number of bytes written */ + field_col; /* Current column */ + char *bufptr, /* Pointer into buffer */ + *bufend; /* Pointer to end of buffer */ + ipp_state_t ipp_state; /* IPP state value */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] cupsdWriteClient " + "error=%d, " + "used=%d, " + "state=%s, " + "data_encoding=HTTP_ENCODE_%s, " + "data_remaining=" CUPS_LLFMT ", " + "response=%p(%s), " + "pipe_pid=%d, " + "file=%d", + con->http.fd, con->http.error, con->http.used, + http_states[con->http.state], + con->http.data_encoding == HTTP_ENCODE_CHUNKED ? + "CHUNKED" : "LENGTH", + CUPS_LLCAST con->http.data_remaining, + con->response, + con->response ? ipp_states[con->response->state] : "", + con->pipe_pid, con->file); + + if (con->http.state != HTTP_GET_SEND && + con->http.state != HTTP_POST_SEND) + { + /* + * If we get called in the wrong state, then something went wrong with the + * connection and we need to shut it down... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Closing on unexpected HTTP state %s.", + con->http.fd, http_states[con->http.state]); + cupsdCloseClient(con); + return; + } + + if (con->pipe_pid) + { + /* + * Make sure we select on the CGI output... + */ + + cupsdAddSelect(con->file, (cupsd_selfunc_t)write_pipe, NULL, con); + + if (!con->file_ready) + { + /* + * Try again later when there is CGI output available... + */ + + cupsdRemoveSelect(con->http.fd); + return; + } + + con->file_ready = 0; + } + + if (con->response && con->response->state != IPP_DATA) + { + ipp_state = ippWrite(HTTP(con), con->response); + bytes = ipp_state != IPP_ERROR && + (con->file >= 0 || ipp_state != IPP_DATA); + } + else if ((bytes = read(con->file, con->header + con->header_used, + sizeof(con->header) - con->header_used)) > 0) + { + con->header_used += bytes; + + if (con->pipe_pid && !con->got_fields) + { + /* + * Inspect the data for Content-Type and other fields. + */ + + for (bufptr = con->header, bufend = con->header + con->header_used, + field_col = 0; + !con->got_fields && bufptr < bufend; + bufptr ++) + { + if (*bufptr == '\n') + { + /* + * Send line to client... + */ + + if (bufptr > con->header && bufptr[-1] == '\r') + bufptr[-1] = '\0'; + *bufptr++ = '\0'; + + cupsdLogMessage(CUPSD_LOG_DEBUG, "Script header: %s", con->header); + + if (!con->sent_header) + { + /* + * Handle redirection and CGI status codes... + */ + + if (!_cups_strncasecmp(con->header, "Location:", 9)) + { + if (!cupsdSendHeader(con, HTTP_SEE_OTHER, NULL, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + con->sent_header = 2; + + if (httpPrintf(HTTP(con), "Content-Length: 0\r\n") < 0) + return; + } + else if (!_cups_strncasecmp(con->header, "Status:", 7)) + { + cupsdSendError(con, (http_status_t)atoi(con->header + 7), + CUPSD_AUTH_NONE); + con->sent_header = 2; + } + else + { + if (!cupsdSendHeader(con, HTTP_OK, NULL, CUPSD_AUTH_NONE)) + { + cupsdCloseClient(con); + return; + } + + con->sent_header = 1; + + if (con->http.version == HTTP_1_1) + { + if (httpPrintf(HTTP(con), "Transfer-Encoding: chunked\r\n") < 0) + return; + } + } + } + + if (_cups_strncasecmp(con->header, "Status:", 7)) + httpPrintf(HTTP(con), "%s\r\n", con->header); + + /* + * Update buffer... + */ + + con->header_used -= bufptr - con->header; + + if (con->header_used > 0) + memmove(con->header, bufptr, con->header_used); + + bufptr = con->header - 1; + + /* + * See if the line was empty... + */ + + if (field_col == 0) + { + con->got_fields = 1; + + if (cupsdFlushHeader(con) < 0) + { + cupsdCloseClient(con); + return; + } + + if (con->http.version == HTTP_1_1) + con->http.data_encoding = HTTP_ENCODE_CHUNKED; + } + else + field_col = 0; + } + else if (*bufptr != '\r') + field_col ++; + } + + if (!con->got_fields) + { + con->http.activity = time(NULL); + return; + } + } + + if (con->header_used > 0) + { + if (httpWrite2(HTTP(con), con->header, con->header_used) < 0) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Closing for error %d (%s)", + con->http.fd, con->http.error, + strerror(con->http.error)); + cupsdCloseClient(con); + return; + } + + if (con->http.data_encoding == HTTP_ENCODE_CHUNKED) + httpFlushWrite(HTTP(con)); + + con->bytes += con->header_used; + + if (con->http.state == HTTP_WAITING) + bytes = 0; + else + bytes = con->header_used; + + con->header_used = 0; + } + } + + if (bytes <= 0 || + (con->http.state != HTTP_GET_SEND && con->http.state != HTTP_POST_SEND)) + { + if (!con->sent_header && con->pipe_pid) + cupsdSendError(con, HTTP_SERVER_ERROR, CUPSD_AUTH_NONE); + else + { + cupsdLogRequest(con, HTTP_OK); + + httpFlushWrite(HTTP(con)); + + if (con->http.data_encoding == HTTP_ENCODE_CHUNKED && con->sent_header == 1) + { + if (httpWrite2(HTTP(con), "", 0) < 0) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Closing for error %d (%s)", + con->http.fd, con->http.error, + strerror(con->http.error)); + cupsdCloseClient(con); + return; + } + } + } + + con->http.state = HTTP_WAITING; + + cupsdAddSelect(con->http.fd, (cupsd_selfunc_t)cupsdReadClient, NULL, con); + + if (con->file >= 0) + { + cupsdRemoveSelect(con->file); + + if (con->pipe_pid) + cupsdEndProcess(con->pipe_pid, 0); + + close(con->file); + con->file = -1; + con->pipe_pid = 0; + } + + if (con->filename) + { + unlink(con->filename); + cupsdClearString(&con->filename); + } + + if (con->request) + { + ippDelete(con->request); + con->request = NULL; + } + + if (con->response) + { + ippDelete(con->response); + con->response = NULL; + } + + cupsdClearString(&con->command); + cupsdClearString(&con->options); + cupsdClearString(&con->query_string); + + if (!con->http.keep_alive) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Client %d] Closing because Keep-Alive disabled.", + con->http.fd); + cupsdCloseClient(con); + return; + } + else + { + cupsArrayRemove(ActiveClients, con); + cupsdSetBusyState(); + } + } + + con->http.activity = time(NULL); +} + + +/* + * 'check_if_modified()' - Decode an "If-Modified-Since" line. + */ + +static int /* O - 1 if modified since */ +check_if_modified( + cupsd_client_t *con, /* I - Client connection */ + struct stat *filestats) /* I - File information */ +{ + char *ptr; /* Pointer into field */ + time_t date; /* Time/date value */ + off_t size; /* Size/length value */ + + + size = 0; + date = 0; + ptr = con->http.fields[HTTP_FIELD_IF_MODIFIED_SINCE]; + + if (*ptr == '\0') + return (1); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] check_if_modified " + "filestats=%p(" CUPS_LLFMT ", %d)) If-Modified-Since=\"%s\"", + con->http.fd, filestats, CUPS_LLCAST filestats->st_size, + (int)filestats->st_mtime, ptr); + + while (*ptr != '\0') + { + while (isspace(*ptr) || *ptr == ';') + ptr ++; + + if (_cups_strncasecmp(ptr, "length=", 7) == 0) + { + ptr += 7; + size = strtoll(ptr, NULL, 10); + + while (isdigit(*ptr)) + ptr ++; + } + else if (isalpha(*ptr)) + { + date = httpGetDateTime(ptr); + while (*ptr != '\0' && *ptr != ';') + ptr ++; + } + else + ptr ++; + } + + return ((size != filestats->st_size && size != 0) || + (date < filestats->st_mtime && date != 0) || + (size == 0 && date == 0)); +} + + +/* + * 'compare_clients()' - Compare two client connections. + */ + +static int /* O - Result of comparison */ +compare_clients(cupsd_client_t *a, /* I - First client */ + cupsd_client_t *b, /* I - Second client */ + void *data) /* I - User data (not used) */ +{ + (void)data; + + if (a == b) + return (0); + else if (a < b) + return (-1); + else + return (1); +} + + +#ifdef HAVE_CDSASSL +/* + * 'copy_cdsa_certificate()' - Copy a SSL/TLS certificate from the System + * keychain. + */ + +static CFArrayRef /* O - Array of certificates */ +copy_cdsa_certificate( + cupsd_client_t *con) /* I - Client connection */ +{ + OSStatus err; /* Error info */ + SecKeychainRef keychain = NULL;/* Keychain reference */ + SecIdentitySearchRef search = NULL; /* Search reference */ + SecIdentityRef identity = NULL;/* Identity */ + CFArrayRef certificates = NULL; + /* Certificate array */ +# if HAVE_SECPOLICYCREATESSL + SecPolicyRef policy = NULL; /* Policy ref */ + CFStringRef servername = NULL; + /* Server name */ + CFMutableDictionaryRef query = NULL; /* Query qualifiers */ + char localname[1024];/* Local hostname */ +# elif defined(HAVE_SECIDENTITYSEARCHCREATEWITHPOLICY) + SecPolicyRef policy = NULL; /* Policy ref */ + SecPolicySearchRef policy_search = NULL; + /* Policy search ref */ + CSSM_DATA options; /* Policy options */ + CSSM_APPLE_TP_SSL_OPTIONS + ssl_options; /* SSL Option for hostname */ + char localname[1024];/* Local hostname */ +# endif /* HAVE_SECPOLICYCREATESSL */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "copy_cdsa_certificate: Looking for certs for \"%s\"...", + con->servername); + + if ((err = SecKeychainOpen(ServerCertificate, &keychain))) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Cannot open keychain \"%s\" - %s (%d)", + ServerCertificate, cssmErrorString(err), (int)err); + goto cleanup; + } + +# if HAVE_SECPOLICYCREATESSL + servername = CFStringCreateWithCString(kCFAllocatorDefault, con->servername, + kCFStringEncodingUTF8); + + policy = SecPolicyCreateSSL(1, servername); + + if (servername) + CFRelease(servername); + + if (!policy) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Cannot create ssl policy reference"); + goto cleanup; + } + + if (!(query = CFDictionaryCreateMutable(kCFAllocatorDefault, 0, + &kCFTypeDictionaryKeyCallBacks, + &kCFTypeDictionaryValueCallBacks))) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Cannot create query dictionary"); + goto cleanup; + } + + CFDictionaryAddValue(query, kSecClass, kSecClassIdentity); + CFDictionaryAddValue(query, kSecMatchPolicy, policy); + CFDictionaryAddValue(query, kSecReturnRef, kCFBooleanTrue); + CFDictionaryAddValue(query, kSecMatchLimit, kSecMatchLimitOne); + + err = SecItemCopyMatching(query, (CFTypeRef *)&identity); + + if (err && DNSSDHostName) + { + /* + * Search for the connection server name failed; try the DNS-SD .local + * hostname instead... + */ + + snprintf(localname, sizeof(localname), "%s.local", DNSSDHostName); + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "copy_cdsa_certificate: Looking for certs for \"%s\"...", + localname); + + servername = CFStringCreateWithCString(kCFAllocatorDefault, localname, + kCFStringEncodingUTF8); + + CFRelease(policy); + + policy = SecPolicyCreateSSL(1, servername); + + if (servername) + CFRelease(servername); + + if (!policy) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Cannot create ssl policy reference"); + goto cleanup; + } + + CFDictionarySetValue(query, kSecMatchPolicy, policy); + + err = SecItemCopyMatching(query, (CFTypeRef *)&identity); + } + + if (err) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Cannot find signing key in keychain \"%s\": %s (%d)", + ServerCertificate, cssmErrorString(err), (int)err); + goto cleanup; + } + +# elif defined(HAVE_SECIDENTITYSEARCHCREATEWITHPOLICY) + /* + * Use a policy to search for valid certificates whose common name matches the + * servername... + */ + + if (SecPolicySearchCreate(CSSM_CERT_X_509v3, &CSSMOID_APPLE_TP_SSL, + NULL, &policy_search)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Cannot create a policy search reference"); + goto cleanup; + } + + if (SecPolicySearchCopyNext(policy_search, &policy)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Cannot find a policy to use for searching"); + goto cleanup; + } + + memset(&ssl_options, 0, sizeof(ssl_options)); + ssl_options.Version = CSSM_APPLE_TP_SSL_OPTS_VERSION; + ssl_options.ServerName = con->servername; + ssl_options.ServerNameLen = strlen(con->servername); + + options.Data = (uint8 *)&ssl_options; + options.Length = sizeof(ssl_options); + + if (SecPolicySetValue(policy, &options)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Cannot set policy value to use for searching"); + goto cleanup; + } + + if ((err = SecIdentitySearchCreateWithPolicy(policy, NULL, CSSM_KEYUSE_SIGN, + keychain, FALSE, &search))) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Cannot create identity search reference: %s (%d)", + cssmErrorString(err), (int)err); + goto cleanup; + } + + err = SecIdentitySearchCopyNext(search, &identity); + + if (err && DNSSDHostName) + { + /* + * Search for the connection server name failed; try the DNS-SD .local + * hostname instead... + */ + + snprintf(localname, sizeof(localname), "%s.local", DNSSDHostName); + + ssl_options.ServerName = localname; + ssl_options.ServerNameLen = strlen(localname); + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "copy_cdsa_certificate: Looking for certs for \"%s\"...", + localname); + + if (SecPolicySetValue(policy, &options)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Cannot set policy value to use for searching"); + goto cleanup; + } + + CFRelease(search); + search = NULL; + if ((err = SecIdentitySearchCreateWithPolicy(policy, NULL, CSSM_KEYUSE_SIGN, + keychain, FALSE, &search))) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Cannot create identity search reference: %s (%d)", + cssmErrorString(err), (int)err); + goto cleanup; + } + + err = SecIdentitySearchCopyNext(search, &identity); + + } + + if (err) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Cannot find signing key in keychain \"%s\": %s (%d)", + ServerCertificate, cssmErrorString(err), (int)err); + goto cleanup; + } + +# else + /* + * Assume there is exactly one SecIdentity in the keychain... + */ + + if ((err = SecIdentitySearchCreate(keychain, CSSM_KEYUSE_SIGN, &search))) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Cannot create identity search reference (%d)", (int)err); + goto cleanup; + } + + if ((err = SecIdentitySearchCopyNext(search, &identity))) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Cannot find signing key in keychain \"%s\": %s (%d)", + ServerCertificate, cssmErrorString(err), (int)err); + goto cleanup; + } +# endif /* HAVE_SECPOLICYCREATESSL */ + + if (CFGetTypeID(identity) != SecIdentityGetTypeID()) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "SecIdentity CFTypeID failure!"); + goto cleanup; + } + + if ((certificates = CFArrayCreate(NULL, (const void **)&identity, + 1, &kCFTypeArrayCallBacks)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Cannot create certificate array"); + goto cleanup; + } + + cleanup : + + if (keychain) + CFRelease(keychain); + if (search) + CFRelease(search); + if (identity) + CFRelease(identity); + +# if HAVE_SECPOLICYCREATESSL + if (policy) + CFRelease(policy); + if (query) + CFRelease(query); +# elif defined(HAVE_SECIDENTITYSEARCHCREATEWITHPOLICY) + if (policy) + CFRelease(policy); + if (policy_search) + CFRelease(policy_search); +# endif /* HAVE_SECPOLICYCREATESSL */ + + return (certificates); +} +#endif /* HAVE_CDSASSL */ + + +/* + * 'data_ready()' - Check whether data is available from a client. + */ + +static int /* O - 1 if data is ready, 0 otherwise */ +data_ready(cupsd_client_t *con) /* I - Client */ +{ + if (con->http.used > 0) + return (1); +#ifdef HAVE_SSL + else if (con->http.tls) + { +# ifdef HAVE_LIBSSL + if (SSL_pending((SSL *)(con->http.tls))) + return (1); +# elif defined(HAVE_GNUTLS) + if (gnutls_record_check_pending(con->http.tls)) + return (1); +# elif defined(HAVE_CDSASSL) + size_t bytes; /* Bytes that are available */ + + if (!SSLGetBufferedReadSize(con->http.tls, &bytes) && bytes > 0) + return (1); +# endif /* HAVE_LIBSSL */ + } +#endif /* HAVE_SSL */ + + return (0); +} + + +#ifdef HAVE_SSL +/* + * 'encrypt_client()' - Enable encryption for the client... + */ + +static int /* O - 1 on success, 0 on error */ +encrypt_client(cupsd_client_t *con) /* I - Client to encrypt */ +{ +# ifdef HAVE_LIBSSL + SSL_CTX *context; /* Context for encryption */ + BIO *bio; /* BIO data */ + unsigned long error; /* Error code */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG, "[Client %d] Encrypting connection.", + con->http.fd); + + /* + * Verify that we have a certificate... + */ + + if (access(ServerKey, 0) || access(ServerCertificate, 0)) + { + /* + * Nope, make a self-signed certificate... + */ + + if (!make_certificate(con)) + return (0); + } + + /* + * Create the SSL context and accept the connection... + */ + + context = SSL_CTX_new(SSLv23_server_method()); + + SSL_CTX_set_options(context, SSL_OP_NO_SSLv2); /* Only use SSLv3 or TLS */ + if (SSLOptions & CUPSD_SSL_NOEMPTY) + SSL_CTX_set_options(context, SSL_OP_DONT_INSERT_EMPTY_FRAGMENTS); + SSL_CTX_use_PrivateKey_file(context, ServerKey, SSL_FILETYPE_PEM); + SSL_CTX_use_certificate_chain_file(context, ServerCertificate); + + bio = BIO_new(_httpBIOMethods()); + BIO_ctrl(bio, BIO_C_SET_FILE_PTR, 0, (char *)HTTP(con)); + + con->http.tls = SSL_new(context); + SSL_set_bio(con->http.tls, bio, bio); + + if (SSL_accept(con->http.tls) != 1) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to encrypt connection from %s.", + con->http.hostname); + + while ((error = ERR_get_error()) != 0) + cupsdLogMessage(CUPSD_LOG_ERROR, "%s", ERR_error_string(error, NULL)); + + SSL_CTX_free(context); + SSL_free(con->http.tls); + con->http.tls = NULL; + return (0); + } + + cupsdLogMessage(CUPSD_LOG_DEBUG, "Connection from %s now encrypted.", + con->http.hostname); + + return (1); + +# elif defined(HAVE_GNUTLS) + int status; /* Error code */ + gnutls_certificate_server_credentials *credentials; + /* TLS credentials */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG, "[Client %d] Encrypting connection.", + con->http.fd); + + /* + * Verify that we have a certificate... + */ + + if (access(ServerKey, 0) || access(ServerCertificate, 0)) + { + /* + * Nope, make a self-signed certificate... + */ + + if (!make_certificate(con)) + return (0); + } + + /* + * Create the SSL object and perform the SSL handshake... + */ + + credentials = (gnutls_certificate_server_credentials *) + malloc(sizeof(gnutls_certificate_server_credentials)); + if (credentials == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to encrypt connection from %s - %s", + con->http.hostname, strerror(errno)); + + return (0); + } + + gnutls_certificate_allocate_credentials(credentials); + gnutls_certificate_set_x509_key_file(*credentials, ServerCertificate, + ServerKey, GNUTLS_X509_FMT_PEM); + + gnutls_init(&con->http.tls, GNUTLS_SERVER); + gnutls_set_default_priority(con->http.tls); + + gnutls_credentials_set(con->http.tls, GNUTLS_CRD_CERTIFICATE, *credentials); + gnutls_transport_set_ptr(con->http.tls, (gnutls_transport_ptr)HTTP(con)); + gnutls_transport_set_pull_function(con->http.tls, _httpReadGNUTLS); + gnutls_transport_set_push_function(con->http.tls, _httpWriteGNUTLS); + + while ((status = gnutls_handshake(con->http.tls)) != GNUTLS_E_SUCCESS) + { + if (gnutls_error_is_fatal(status)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to encrypt connection from %s - %s", + con->http.hostname, gnutls_strerror(status)); + + gnutls_deinit(con->http.tls); + gnutls_certificate_free_credentials(*credentials); + con->http.tls = NULL; + free(credentials); + return (0); + } + } + + cupsdLogMessage(CUPSD_LOG_DEBUG, "Connection from %s now encrypted.", + con->http.hostname); + + con->http.tls_credentials = credentials; + return (1); + +# elif defined(HAVE_CDSASSL) + OSStatus error = 0; /* Error code */ + CFArrayRef peerCerts; /* Peer certificates */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG, "[Client %d] Encrypting connection.", + con->http.fd); + + con->http.tls_credentials = copy_cdsa_certificate(con); + + if (!con->http.tls_credentials) + { + /* + * No keychain (yet), make a self-signed certificate... + */ + + if (make_certificate(con)) + con->http.tls_credentials = copy_cdsa_certificate(con); + } + + if (!con->http.tls_credentials) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Could not find signing key in keychain \"%s\"", + ServerCertificate); + error = errSSLBadCert; /* errSSLBadConfiguration is a better choice, but not available on 10.2.x */ + } + + if (!error) + error = SSLNewContext(true, &con->http.tls); + + if (!error) + error = SSLSetIOFuncs(con->http.tls, _httpReadCDSA, _httpWriteCDSA); + + if (!error) + error = SSLSetConnection(con->http.tls, HTTP(con)); + + if (!error) + error = SSLSetAllowsExpiredCerts(con->http.tls, true); + + if (!error) + error = SSLSetAllowsAnyRoot(con->http.tls, true); + + if (!error) + error = SSLSetCertificate(con->http.tls, con->http.tls_credentials); + + if (!error) + { + /* + * Perform SSL/TLS handshake + */ + + while ((error = SSLHandshake(con->http.tls)) == errSSLWouldBlock) + usleep(1000); + } + + if (error) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to encrypt connection from %s - %s (%d)", + con->http.hostname, cssmErrorString(error), (int)error); + + con->http.error = error; + con->http.status = HTTP_ERROR; + + if (con->http.tls) + { + SSLDisposeContext(con->http.tls); + con->http.tls = NULL; + } + + if (con->http.tls_credentials) + { + CFRelease(con->http.tls_credentials); + con->http.tls_credentials = NULL; + } + + return (0); + } + + cupsdLogMessage(CUPSD_LOG_DEBUG, "Connection from %s now encrypted.", + con->http.hostname); + + if (!SSLCopyPeerCertificates(con->http.tls, &peerCerts) && peerCerts) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, "Received %d peer certificates!", + (int)CFArrayGetCount(peerCerts)); + CFRelease(peerCerts); + } + else + cupsdLogMessage(CUPSD_LOG_DEBUG, "Received NO peer certificates!"); + + return (1); + +# endif /* HAVE_LIBSSL */ +} +#endif /* HAVE_SSL */ + + +/* + * 'get_file()' - Get a filename and state info. + */ + +static char * /* O - Real filename */ +get_file(cupsd_client_t *con, /* I - Client connection */ + struct stat *filestats, /* O - File information */ + char *filename, /* IO - Filename buffer */ + int len) /* I - Buffer length */ +{ + int status; /* Status of filesystem calls */ + char *ptr; /* Pointer info filename */ + int plen; /* Remaining length after pointer */ + char language[7]; /* Language subdirectory, if any */ + + + /* + * Figure out the real filename... + */ + + language[0] = '\0'; + + if (!strncmp(con->uri, "/ppd/", 5) && !strchr(con->uri + 5, '/')) + snprintf(filename, len, "%s%s", ServerRoot, con->uri); + else if (!strncmp(con->uri, "/icons/", 7) && !strchr(con->uri + 7, '/')) + { + snprintf(filename, len, "%s/%s", CacheDir, con->uri + 7); + if (access(filename, F_OK) < 0) + snprintf(filename, len, "%s/images/generic.png", DocumentRoot); + } + else if (!strncmp(con->uri, "/rss/", 5) && !strchr(con->uri + 5, '/')) + snprintf(filename, len, "%s/rss/%s", CacheDir, con->uri + 5); + else if (!strncmp(con->uri, "/admin/conf/", 12)) + snprintf(filename, len, "%s%s", ServerRoot, con->uri + 11); + else if (!strncmp(con->uri, "/admin/log/", 11)) + { + if (!strncmp(con->uri + 11, "access_log", 10) && AccessLog[0] == '/') + strlcpy(filename, AccessLog, len); + else if (!strncmp(con->uri + 11, "error_log", 9) && ErrorLog[0] == '/') + strlcpy(filename, ErrorLog, len); + else if (!strncmp(con->uri + 11, "page_log", 8) && PageLog[0] == '/') + strlcpy(filename, PageLog, len); + else + return (NULL); + } + else if (con->language) + { + snprintf(language, sizeof(language), "/%s", con->language->language); + snprintf(filename, len, "%s%s%s", DocumentRoot, language, con->uri); + } + else + snprintf(filename, len, "%s%s", DocumentRoot, con->uri); + + if ((ptr = strchr(filename, '?')) != NULL) + *ptr = '\0'; + + /* + * Grab the status for this language; if there isn't a language-specific file + * then fallback to the default one... + */ + + if ((status = stat(filename, filestats)) != 0 && language[0] && + strncmp(con->uri, "/icons/", 7) && + strncmp(con->uri, "/ppd/", 5) && + strncmp(con->uri, "/rss/", 5) && + strncmp(con->uri, "/admin/conf/", 12) && + strncmp(con->uri, "/admin/log/", 11)) + { + /* + * Drop the country code... + */ + + language[3] = '\0'; + snprintf(filename, len, "%s%s%s", DocumentRoot, language, con->uri); + + if ((ptr = strchr(filename, '?')) != NULL) + *ptr = '\0'; + + if ((status = stat(filename, filestats)) != 0) + { + /* + * Drop the language prefix and try the root directory... + */ + + language[0] = '\0'; + snprintf(filename, len, "%s%s", DocumentRoot, con->uri); + + if ((ptr = strchr(filename, '?')) != NULL) + *ptr = '\0'; + + status = stat(filename, filestats); + } + } + + /* + * If we're found a directory, get the index.html file instead... + */ + + if (!status && S_ISDIR(filestats->st_mode)) + { + /* + * Make sure the URI ends with a slash... + */ + + if (con->uri[strlen(con->uri) - 1] != '/') + strlcat(con->uri, "/", sizeof(con->uri)); + + /* + * Find the directory index file, trying every language... + */ + + do + { + if (status && language[0]) + { + /* + * Try a different language subset... + */ + + if (language[3]) + language[0] = '\0'; /* Strip country code */ + else + language[0] = '\0'; /* Strip language */ + } + + /* + * Look for the index file... + */ + + snprintf(filename, len, "%s%s%s", DocumentRoot, language, con->uri); + + if ((ptr = strchr(filename, '?')) != NULL) + *ptr = '\0'; + + ptr = filename + strlen(filename); + plen = len - (ptr - filename); + + strlcpy(ptr, "index.html", plen); + status = stat(filename, filestats); + +#ifdef HAVE_JAVA + if (status) + { + strlcpy(ptr, "index.class", plen); + status = stat(filename, filestats); + } +#endif /* HAVE_JAVA */ + +#ifdef HAVE_PERL + if (status) + { + strlcpy(ptr, "index.pl", plen); + status = stat(filename, filestats); + } +#endif /* HAVE_PERL */ + +#ifdef HAVE_PHP + if (status) + { + strlcpy(ptr, "index.php", plen); + status = stat(filename, filestats); + } +#endif /* HAVE_PHP */ + +#ifdef HAVE_PYTHON + if (status) + { + strlcpy(ptr, "index.pyc", plen); + status = stat(filename, filestats); + } + + if (status) + { + strlcpy(ptr, "index.py", plen); + status = stat(filename, filestats); + } +#endif /* HAVE_PYTHON */ + + } + while (status && language[0]); + } + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] get_file filestats=%p, filename=%p, len=%d, " + "returning \"%s\".", con->http.fd, filestats, filename, len, + status ? "(null)" : filename); + + if (status) + return (NULL); + else + return (filename); +} + + +/* + * 'install_conf_file()' - Install a configuration file. + */ + +static http_status_t /* O - Status */ +install_conf_file(cupsd_client_t *con) /* I - Connection */ +{ + char filename[1024]; /* Configuration filename */ + mode_t mode; /* Permissions */ + cups_file_t *in, /* Input file */ + *out; /* Output file */ + char buffer[16384]; /* Copy buffer */ + ssize_t bytes; /* Number of bytes */ + + + /* + * Open the request file... + */ + + if ((in = cupsFileOpen(con->filename, "rb")) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to open request file \"%s\": %s", + con->filename, strerror(errno)); + return (HTTP_SERVER_ERROR); + } + + /* + * Open the new config file... + */ + + snprintf(filename, sizeof(filename), "%s%s", ServerRoot, con->uri + 11); + if (!strcmp(con->uri, "/admin/conf/printers.conf")) + mode = ConfigFilePerm & 0600; + else + mode = ConfigFilePerm; + + if ((out = cupsdCreateConfFile(filename, mode)) == NULL) + { + cupsFileClose(in); + return (HTTP_SERVER_ERROR); + } + + cupsdLogMessage(CUPSD_LOG_INFO, "Installing config file \"%s\"...", filename); + + /* + * Copy from the request to the new config file... + */ + + while ((bytes = cupsFileRead(in, buffer, sizeof(buffer))) > 0) + if (cupsFileWrite(out, buffer, bytes) < bytes) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to copy to config file \"%s\": %s", + filename, strerror(errno)); + + cupsFileClose(in); + cupsFileClose(out); + + snprintf(filename, sizeof(filename), "%s%s.N", ServerRoot, con->uri + 11); + cupsdRemoveFile(filename); + + return (HTTP_SERVER_ERROR); + } + + /* + * Close the files... + */ + + cupsFileClose(in); + + if (cupsdCloseCreatedConfFile(out, filename)) + return (HTTP_SERVER_ERROR); + + /* + * Remove the request file... + */ + + cupsdRemoveFile(con->filename); + cupsdClearString(&con->filename); + + /* + * If the cupsd.conf file was updated, set the NeedReload flag... + */ + + if (!strcmp(con->uri, "/admin/conf/cupsd.conf")) + NeedReload = RELOAD_CUPSD; + else + NeedReload = RELOAD_ALL; + + ReloadTime = time(NULL); + + /* + * Return that the file was created successfully... + */ + + return (HTTP_CREATED); +} + + +/* + * 'is_cgi()' - Is the resource a CGI script/program? + */ + +static int /* O - 1 = CGI, 0 = file */ +is_cgi(cupsd_client_t *con, /* I - Client connection */ + const char *filename, /* I - Real filename */ + struct stat *filestats, /* I - File information */ + mime_type_t *type) /* I - MIME type */ +{ + const char *options; /* Options on URL */ + + + /* + * Get the options, if any... + */ + + if ((options = strchr(con->uri, '?')) != NULL) + { + options ++; + cupsdSetStringf(&(con->query_string), "QUERY_STRING=%s", options); + } + + /* + * Check for known types... + */ + + if (!type || _cups_strcasecmp(type->super, "application")) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] is_cgi filename=\"%s\", filestats=%p, " + "type=%s/%s, returning 0", con->http.fd, filename, + filestats, type ? type->super : "unknown", + type ? type->type : "unknown"); + return (0); + } + + if (!_cups_strcasecmp(type->type, "x-httpd-cgi") && + (filestats->st_mode & 0111)) + { + /* + * "application/x-httpd-cgi" is a CGI script. + */ + + cupsdSetString(&con->command, filename); + + if (options) + cupsdSetStringf(&con->options, " %s", options); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] is_cgi filename=\"%s\", filestats=%p, " + "type=%s/%s, returning 1", con->http.fd, filename, + filestats, type->super, type->type); + return (1); + } +#ifdef HAVE_JAVA + else if (!_cups_strcasecmp(type->type, "x-httpd-java")) + { + /* + * "application/x-httpd-java" is a Java servlet. + */ + + cupsdSetString(&con->command, CUPS_JAVA); + + if (options) + cupsdSetStringf(&con->options, " %s %s", filename, options); + else + cupsdSetStringf(&con->options, " %s", filename); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] is_cgi filename=\"%s\", filestats=%p, " + "type=%s/%s, returning 1", con->http.fd, filename, + filestats, type->super, type->type); + return (1); + } +#endif /* HAVE_JAVA */ +#ifdef HAVE_PERL + else if (!_cups_strcasecmp(type->type, "x-httpd-perl")) + { + /* + * "application/x-httpd-perl" is a Perl page. + */ + + cupsdSetString(&con->command, CUPS_PERL); + + if (options) + cupsdSetStringf(&con->options, " %s %s", filename, options); + else + cupsdSetStringf(&con->options, " %s", filename); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] is_cgi filename=\"%s\", filestats=%p, " + "type=%s/%s, returning 1", con->http.fd, filename, + filestats, type->super, type->type); + return (1); + } +#endif /* HAVE_PERL */ +#ifdef HAVE_PHP + else if (!_cups_strcasecmp(type->type, "x-httpd-php")) + { + /* + * "application/x-httpd-php" is a PHP page. + */ + + cupsdSetString(&con->command, CUPS_PHP); + + if (options) + cupsdSetStringf(&con->options, " %s %s", filename, options); + else + cupsdSetStringf(&con->options, " %s", filename); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] is_cgi filename=\"%s\", filestats=%p, " + "type=%s/%s, returning 1", con->http.fd, filename, + filestats, type->super, type->type); + return (1); + } +#endif /* HAVE_PHP */ +#ifdef HAVE_PYTHON + else if (!_cups_strcasecmp(type->type, "x-httpd-python")) + { + /* + * "application/x-httpd-python" is a Python page. + */ + + cupsdSetString(&con->command, CUPS_PYTHON); + + if (options) + cupsdSetStringf(&con->options, " %s %s", filename, options); + else + cupsdSetStringf(&con->options, " %s", filename); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] is_cgi filename=\"%s\", filestats=%p, " + "type=%s/%s, returning 1", con->http.fd, filename, + filestats, type->super, type->type); + return (1); + } +#endif /* HAVE_PYTHON */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] is_cgi filename=\"%s\", filestats=%p, " + "type=%s/%s, returning 0", con->http.fd, filename, + filestats, type->super, type->type); + return (0); +} + + +/* + * 'is_path_absolute()' - Is a path absolute and free of relative elements (i.e. ".."). + */ + +static int /* O - 0 if relative, 1 if absolute */ +is_path_absolute(const char *path) /* I - Input path */ +{ + /* + * Check for a leading slash... + */ + + if (path[0] != '/') + return (0); + + /* + * Check for "/.." in the path... + */ + + while ((path = strstr(path, "/..")) != NULL) + { + if (!path[3] || path[3] == '/') + return (0); + + path ++; + } + + /* + * If we haven't found any relative paths, return 1 indicating an + * absolute path... + */ + + return (1); +} + + +#ifdef HAVE_SSL +/* + * 'make_certificate()' - Make a self-signed SSL/TLS certificate. + */ + +static int /* O - 1 on success, 0 on failure */ +make_certificate(cupsd_client_t *con) /* I - Client connection */ +{ +#if defined(HAVE_LIBSSL) && defined(HAVE_WAITPID) + int pid, /* Process ID of command */ + status; /* Status of command */ + char command[1024], /* Command */ + *argv[12], /* Command-line arguments */ + *envp[MAX_ENV + 1], /* Environment variables */ + infofile[1024], /* Type-in information for cert */ + seedfile[1024]; /* Random number seed file */ + int envc, /* Number of environment variables */ + bytes; /* Bytes written */ + cups_file_t *fp; /* Seed/info file */ + int infofd; /* Info file descriptor */ + + + /* + * Run the "openssl" command to seed the random number generator and + * generate a self-signed certificate that is good for 10 years: + * + * openssl rand -rand seedfile 1 + * + * openssl req -new -x509 -keyout ServerKey \ + * -out ServerCertificate -days 3650 -nodes + * + * The seeding step is crucial in ensuring that the openssl command + * does not block on systems without sufficient entropy... + */ + + if (!cupsFileFind("openssl", getenv("PATH"), 1, command, sizeof(command))) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "No SSL certificate and openssl command not found!"); + return (0); + } + + if (access("/dev/urandom", 0)) + { + /* + * If the system doesn't provide /dev/urandom, then any random source + * will probably be blocking-style, so generate some random data to + * use as a seed for the certificate. Note that we have already + * seeded the random number generator in cupsdInitCerts()... + */ + + cupsdLogMessage(CUPSD_LOG_INFO, + "Seeding the random number generator..."); + + /* + * Write the seed file... + */ + + if ((fp = cupsTempFile2(seedfile, sizeof(seedfile))) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to create seed file %s - %s", + seedfile, strerror(errno)); + return (0); + } + + for (bytes = 0; bytes < 262144; bytes ++) + cupsFilePutChar(fp, random()); + + cupsFileClose(fp); + + /* + * Run the openssl command to seed its random number generator... + */ + + argv[0] = "openssl"; + argv[1] = "rand"; + argv[2] = "-rand"; + argv[3] = seedfile; + argv[4] = "1"; + argv[5] = NULL; + + envc = cupsdLoadEnv(envp, MAX_ENV); + envp[envc] = NULL; + + if (!cupsdStartProcess(command, argv, envp, -1, -1, -1, -1, -1, 1, NULL, + NULL, &pid)) + { + unlink(seedfile); + return (0); + } + + while (waitpid(pid, &status, 0) < 0) + if (errno != EINTR) + { + status = 1; + break; + } + + cupsdFinishProcess(pid, command, sizeof(command), NULL); + + /* + * Remove the seed file, as it is no longer needed... + */ + + unlink(seedfile); + + if (status) + { + if (WIFEXITED(status)) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to seed random number generator - " + "the openssl command stopped with status %d!", + WEXITSTATUS(status)); + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to seed random number generator - " + "the openssl command crashed on signal %d!", + WTERMSIG(status)); + + return (0); + } + } + + /* + * Create a file with the certificate information fields... + * + * Note: This assumes that the default questions are asked by the openssl + * command... + */ + + if ((fp = cupsTempFile2(infofile, sizeof(infofile))) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to create certificate information file %s - %s", + infofile, strerror(errno)); + return (0); + } + + cupsFilePrintf(fp, ".\n.\n.\n%s\n.\n%s\n%s\n", + ServerName, ServerName, ServerAdmin); + cupsFileClose(fp); + + cupsdLogMessage(CUPSD_LOG_INFO, + "Generating SSL server key and certificate..."); + + argv[0] = "openssl"; + argv[1] = "req"; + argv[2] = "-new"; + argv[3] = "-x509"; + argv[4] = "-keyout"; + argv[5] = ServerKey; + argv[6] = "-out"; + argv[7] = ServerCertificate; + argv[8] = "-days"; + argv[9] = "3650"; + argv[10] = "-nodes"; + argv[11] = NULL; + + cupsdLoadEnv(envp, MAX_ENV); + + infofd = open(infofile, O_RDONLY); + + if (!cupsdStartProcess(command, argv, envp, infofd, -1, -1, -1, -1, 1, NULL, + NULL, &pid)) + { + close(infofd); + unlink(infofile); + return (0); + } + + close(infofd); + unlink(infofile); + + while (waitpid(pid, &status, 0) < 0) + if (errno != EINTR) + { + status = 1; + break; + } + + cupsdFinishProcess(pid, command, sizeof(command), NULL); + + if (status) + { + if (WIFEXITED(status)) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to create SSL server key and certificate - " + "the openssl command stopped with status %d!", + WEXITSTATUS(status)); + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to create SSL server key and certificate - " + "the openssl command crashed on signal %d!", + WTERMSIG(status)); + } + else + { + cupsdLogMessage(CUPSD_LOG_INFO, "Created SSL server key file \"%s\"...", + ServerKey); + cupsdLogMessage(CUPSD_LOG_INFO, + "Created SSL server certificate file \"%s\"...", + ServerCertificate); + } + + return (!status); + +#elif defined(HAVE_GNUTLS) + gnutls_x509_crt crt; /* Self-signed certificate */ + gnutls_x509_privkey key; /* Encryption key */ + cups_lang_t *language; /* Default language info */ + cups_file_t *fp; /* Key/cert file */ + unsigned char buffer[8192]; /* Buffer for x509 data */ + size_t bytes; /* Number of bytes of data */ + unsigned char serial[4]; /* Serial number buffer */ + time_t curtime; /* Current time */ + int result; /* Result of GNU TLS calls */ + + + /* + * Create the encryption key... + */ + + cupsdLogMessage(CUPSD_LOG_INFO, "Generating SSL server key..."); + + gnutls_x509_privkey_init(&key); + gnutls_x509_privkey_generate(key, GNUTLS_PK_RSA, 2048, 0); + + /* + * Save it... + */ + + bytes = sizeof(buffer); + + if ((result = gnutls_x509_privkey_export(key, GNUTLS_X509_FMT_PEM, + buffer, &bytes)) < 0) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to export SSL server key - %s", + gnutls_strerror(result)); + gnutls_x509_privkey_deinit(key); + return (0); + } + else if ((fp = cupsFileOpen(ServerKey, "w")) != NULL) + { + cupsFileWrite(fp, (char *)buffer, bytes); + cupsFileClose(fp); + + cupsdLogMessage(CUPSD_LOG_INFO, "Created SSL server key file \"%s\"...", + ServerKey); + } + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to create SSL server key file \"%s\" - %s", + ServerKey, strerror(errno)); + gnutls_x509_privkey_deinit(key); + return (0); + } + + /* + * Create the self-signed certificate... + */ + + cupsdLogMessage(CUPSD_LOG_INFO, "Generating self-signed SSL certificate..."); + + language = cupsLangDefault(); + curtime = time(NULL); + serial[0] = curtime >> 24; + serial[1] = curtime >> 16; + serial[2] = curtime >> 8; + serial[3] = curtime; + + gnutls_x509_crt_init(&crt); + if (strlen(language->language) == 5) + gnutls_x509_crt_set_dn_by_oid(crt, GNUTLS_OID_X520_COUNTRY_NAME, 0, + language->language + 3, 2); + else + gnutls_x509_crt_set_dn_by_oid(crt, GNUTLS_OID_X520_COUNTRY_NAME, 0, + "US", 2); + gnutls_x509_crt_set_dn_by_oid(crt, GNUTLS_OID_X520_COMMON_NAME, 0, + ServerName, strlen(ServerName)); + gnutls_x509_crt_set_dn_by_oid(crt, GNUTLS_OID_X520_ORGANIZATION_NAME, 0, + ServerName, strlen(ServerName)); + gnutls_x509_crt_set_dn_by_oid(crt, GNUTLS_OID_X520_ORGANIZATIONAL_UNIT_NAME, + 0, "Unknown", 7); + gnutls_x509_crt_set_dn_by_oid(crt, GNUTLS_OID_X520_STATE_OR_PROVINCE_NAME, 0, + "Unknown", 7); + gnutls_x509_crt_set_dn_by_oid(crt, GNUTLS_OID_X520_LOCALITY_NAME, 0, + "Unknown", 7); + gnutls_x509_crt_set_dn_by_oid(crt, GNUTLS_OID_PKCS9_EMAIL, 0, + ServerAdmin, strlen(ServerAdmin)); + gnutls_x509_crt_set_key(crt, key); + gnutls_x509_crt_set_serial(crt, serial, sizeof(serial)); + gnutls_x509_crt_set_activation_time(crt, curtime); + gnutls_x509_crt_set_expiration_time(crt, curtime + 10 * 365 * 86400); + gnutls_x509_crt_set_ca_status(crt, 0); + gnutls_x509_crt_set_subject_alternative_name(crt, GNUTLS_SAN_DNSNAME, + ServerName); + gnutls_x509_crt_set_key_purpose_oid(crt, GNUTLS_KP_TLS_WWW_SERVER, 0); + gnutls_x509_crt_set_key_usage(crt, GNUTLS_KEY_KEY_ENCIPHERMENT); + gnutls_x509_crt_set_version(crt, 3); + + bytes = sizeof(buffer); + if (gnutls_x509_crt_get_key_id(crt, 0, buffer, &bytes) >= 0) + gnutls_x509_crt_set_subject_key_id(crt, buffer, bytes); + + gnutls_x509_crt_sign(crt, crt, key); + + /* + * Save it... + */ + + bytes = sizeof(buffer); + if ((result = gnutls_x509_crt_export(crt, GNUTLS_X509_FMT_PEM, + buffer, &bytes)) < 0) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to export SSL server certificate - %s", + gnutls_strerror(result)); + else if ((fp = cupsFileOpen(ServerCertificate, "w")) != NULL) + { + cupsFileWrite(fp, (char *)buffer, bytes); + cupsFileClose(fp); + + cupsdLogMessage(CUPSD_LOG_INFO, + "Created SSL server certificate file \"%s\"...", + ServerCertificate); + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to create SSL server certificate file \"%s\" - %s", + ServerCertificate, strerror(errno)); + + /* + * Cleanup... + */ + + gnutls_x509_crt_deinit(crt); + gnutls_x509_privkey_deinit(key); + + return (1); + +#elif defined(HAVE_CDSASSL) && defined(HAVE_WAITPID) + int pid, /* Process ID of command */ + status; /* Status of command */ + char command[1024], /* Command */ + *argv[4], /* Command-line arguments */ + *envp[MAX_ENV + 1], /* Environment variables */ + keychain[1024], /* Keychain argument */ + infofile[1024], /* Type-in information for cert */ + localname[1024], /* Local hostname */ + *servername; /* Name of server in cert */ + cups_file_t *fp; /* Seed/info file */ + int infofd; /* Info file descriptor */ + + + if (con->servername && isdigit(con->servername[0] & 255) && DNSSDHostName) + { + snprintf(localname, sizeof(localname), "%s.local", DNSSDHostName); + servername = localname; + } + else + servername = con->servername; + + /* + * Run the "certtool" command to generate a self-signed certificate... + */ + + if (!cupsFileFind("certtool", getenv("PATH"), 1, command, sizeof(command))) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "No SSL certificate and certtool command not found!"); + return (0); + } + + /* + * Create a file with the certificate information fields... + * + * Note: This assumes that the default questions are asked by the certtool + * command... + */ + + if ((fp = cupsTempFile2(infofile, sizeof(infofile))) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to create certificate information file %s - %s", + infofile, strerror(errno)); + return (0); + } + + cupsFilePrintf(fp, "%s\nr\n\ny\nb\ns\ny\n%s\n\n\n\n\n%s\ny\n", + servername, servername, ServerAdmin); + cupsFileClose(fp); + + cupsdLogMessage(CUPSD_LOG_INFO, + "Generating SSL server key and certificate..."); + + snprintf(keychain, sizeof(keychain), "k=%s", ServerCertificate); + + argv[0] = "certtool"; + argv[1] = "c"; + argv[2] = keychain; + argv[3] = NULL; + + cupsdLoadEnv(envp, MAX_ENV); + + infofd = open(infofile, O_RDONLY); + + if (!cupsdStartProcess(command, argv, envp, infofd, -1, -1, -1, -1, 1, NULL, + NULL, &pid)) + { + close(infofd); + unlink(infofile); + return (0); + } + + close(infofd); + unlink(infofile); + + while (waitpid(pid, &status, 0) < 0) + if (errno != EINTR) + { + status = 1; + break; + } + + cupsdFinishProcess(pid, command, sizeof(command), NULL); + + if (status) + { + if (WIFEXITED(status)) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to create SSL server key and certificate - " + "the certtool command stopped with status %d!", + WEXITSTATUS(status)); + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to create SSL server key and certificate - " + "the certtool command crashed on signal %d!", + WTERMSIG(status)); + } + else + { + cupsdLogMessage(CUPSD_LOG_INFO, + "Created SSL server certificate file \"%s\"...", + ServerCertificate); + } + + return (!status); + +#else + return (0); +#endif /* HAVE_LIBSSL && HAVE_WAITPID */ +} +#endif /* HAVE_SSL */ + + +/* + * 'pipe_command()' - Pipe the output of a command to the remote client. + */ + +static int /* O - Process ID */ +pipe_command(cupsd_client_t *con, /* I - Client connection */ + int infile, /* I - Standard input for command */ + int *outfile, /* O - Standard output for command */ + char *command, /* I - Command to run */ + char *options, /* I - Options for command */ + int root) /* I - Run as root? */ +{ + int i; /* Looping var */ + int pid; /* Process ID */ + char *commptr, /* Command string pointer */ + commch; /* Command string character */ + char *uriptr; /* URI string pointer */ + int fds[2]; /* Pipe FDs */ + int argc; /* Number of arguments */ + int envc; /* Number of environment variables */ + char argbuf[10240], /* Argument buffer */ + *argv[100], /* Argument strings */ + *envp[MAX_ENV + 20]; /* Environment variables */ + char auth_type[256], /* AUTH_TYPE environment variable */ + content_length[1024], /* CONTENT_LENGTH environment variable */ + content_type[1024], /* CONTENT_TYPE environment variable */ + http_cookie[32768], /* HTTP_COOKIE environment variable */ + http_referer[1024], /* HTTP_REFERER environment variable */ + http_user_agent[1024], /* HTTP_USER_AGENT environment variable */ + lang[1024], /* LANG environment variable */ + path_info[1024], /* PATH_INFO environment variable */ + remote_addr[1024], /* REMOTE_ADDR environment variable */ + remote_host[1024], /* REMOTE_HOST environment variable */ + remote_user[1024], /* REMOTE_USER environment variable */ + script_filename[1024], /* SCRIPT_FILENAME environment variable */ + script_name[1024], /* SCRIPT_NAME environment variable */ + server_name[1024], /* SERVER_NAME environment variable */ + server_port[1024]; /* SERVER_PORT environment variable */ + ipp_attribute_t *attr; /* attributes-natural-language attribute */ + void *ccache = NULL; /* Kerberos credentials */ + + + /* + * Parse a copy of the options string, which is of the form: + * + * argument+argument+argument + * ?argument+argument+argument + * param=value¶m=value + * ?param=value¶m=value + * /name?argument+argument+argument + * /name?param=value¶m=value + * + * If the string contains an "=" character after the initial name, + * then we treat it as a HTTP GET form request and make a copy of + * the remaining string for the environment variable. + * + * The string is always parsed out as command-line arguments, to + * be consistent with Apache... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] pipe_command infile=%d, outfile=%p, " + "command=\"%s\", options=\"%s\", root=%d", + con->http.fd, infile, outfile, command, + options ? options : "(null)", root); + + argv[0] = command; + + if (options) + { + commptr = options; + if (*commptr == ' ') + commptr ++; + strlcpy(argbuf, commptr, sizeof(argbuf)); + } + else + argbuf[0] = '\0'; + + if (argbuf[0] == '/') + { + /* + * Found some trailing path information, set PATH_INFO... + */ + + if ((commptr = strchr(argbuf, '?')) == NULL) + commptr = argbuf + strlen(argbuf); + + commch = *commptr; + *commptr = '\0'; + snprintf(path_info, sizeof(path_info), "PATH_INFO=%s", argbuf); + *commptr = commch; + } + else + { + commptr = argbuf; + path_info[0] = '\0'; + + if (*commptr == ' ') + commptr ++; + } + + if (*commptr == '?' && con->operation == HTTP_GET && !con->query_string) + { + commptr ++; + cupsdSetStringf(&(con->query_string), "QUERY_STRING=%s", commptr); + } + + argc = 1; + + if (*commptr) + { + argv[argc ++] = commptr; + + for (; *commptr && argc < 99; commptr ++) + { + /* + * Break arguments whenever we see a + or space... + */ + + if (*commptr == ' ' || *commptr == '+') + { + while (*commptr == ' ' || *commptr == '+') + *commptr++ = '\0'; + + /* + * If we don't have a blank string, save it as another argument... + */ + + if (*commptr) + { + argv[argc] = commptr; + argc ++; + } + else + break; + } + else if (*commptr == '%' && isxdigit(commptr[1] & 255) && + isxdigit(commptr[2] & 255)) + { + /* + * Convert the %xx notation to the individual character. + */ + + if (commptr[1] >= '0' && commptr[1] <= '9') + *commptr = (commptr[1] - '0') << 4; + else + *commptr = (tolower(commptr[1]) - 'a' + 10) << 4; + + if (commptr[2] >= '0' && commptr[2] <= '9') + *commptr |= commptr[2] - '0'; + else + *commptr |= tolower(commptr[2]) - 'a' + 10; + + _cups_strcpy(commptr + 1, commptr + 3); + + /* + * Check for a %00 and break if that is the case... + */ + + if (!*commptr) + break; + } + } + } + + argv[argc] = NULL; + + /* + * Setup the environment variables as needed... + */ + + if (con->username[0]) + { + snprintf(auth_type, sizeof(auth_type), "AUTH_TYPE=%s", + httpGetField(HTTP(con), HTTP_FIELD_AUTHORIZATION)); + + if ((uriptr = strchr(auth_type + 10, ' ')) != NULL) + *uriptr = '\0'; + } + else + auth_type[0] = '\0'; + + if (con->request && + (attr = ippFindAttribute(con->request, "attributes-natural-language", + IPP_TAG_LANGUAGE)) != NULL) + { + switch (strlen(attr->values[0].string.text)) + { + default : + /* + * This is an unknown or badly formatted language code; use + * the POSIX locale... + */ + + strcpy(lang, "LANG=C"); + break; + + case 2 : + /* + * Just the language code (ll)... + */ + + snprintf(lang, sizeof(lang), "LANG=%s.UTF8", + attr->values[0].string.text); + break; + + case 5 : + /* + * Language and country code (ll-cc)... + */ + + snprintf(lang, sizeof(lang), "LANG=%c%c_%c%c.UTF8", + attr->values[0].string.text[0], + attr->values[0].string.text[1], + toupper(attr->values[0].string.text[3] & 255), + toupper(attr->values[0].string.text[4] & 255)); + break; + } + } + else if (con->language) + snprintf(lang, sizeof(lang), "LANG=%s.UTF8", con->language->language); + else + strcpy(lang, "LANG=C"); + + strcpy(remote_addr, "REMOTE_ADDR="); + httpAddrString(con->http.hostaddr, remote_addr + 12, + sizeof(remote_addr) - 12); + + snprintf(remote_host, sizeof(remote_host), "REMOTE_HOST=%s", + con->http.hostname); + + snprintf(script_name, sizeof(script_name), "SCRIPT_NAME=%s", con->uri); + if ((uriptr = strchr(script_name, '?')) != NULL) + *uriptr = '\0'; + + snprintf(script_filename, sizeof(script_filename), "SCRIPT_FILENAME=%s%s", + DocumentRoot, script_name + 12); + + sprintf(server_port, "SERVER_PORT=%d", con->serverport); + + if (con->http.fields[HTTP_FIELD_HOST][0]) + { + char *nameptr; /* Pointer to ":port" */ + + snprintf(server_name, sizeof(server_name), "SERVER_NAME=%s", + con->http.fields[HTTP_FIELD_HOST]); + if ((nameptr = strrchr(server_name, ':')) != NULL && !strchr(nameptr, ']')) + *nameptr = '\0'; /* Strip trailing ":port" */ + } + else + snprintf(server_name, sizeof(server_name), "SERVER_NAME=%s", + con->servername); + + envc = cupsdLoadEnv(envp, (int)(sizeof(envp) / sizeof(envp[0]))); + + if (auth_type[0]) + envp[envc ++] = auth_type; + + envp[envc ++] = lang; + envp[envc ++] = "REDIRECT_STATUS=1"; + envp[envc ++] = "GATEWAY_INTERFACE=CGI/1.1"; + envp[envc ++] = server_name; + envp[envc ++] = server_port; + envp[envc ++] = remote_addr; + envp[envc ++] = remote_host; + envp[envc ++] = script_name; + envp[envc ++] = script_filename; + + if (path_info[0]) + envp[envc ++] = path_info; + + if (con->username[0]) + { + snprintf(remote_user, sizeof(remote_user), "REMOTE_USER=%s", con->username); + + envp[envc ++] = remote_user; + } + + if (con->http.version == HTTP_1_1) + envp[envc ++] = "SERVER_PROTOCOL=HTTP/1.1"; + else if (con->http.version == HTTP_1_0) + envp[envc ++] = "SERVER_PROTOCOL=HTTP/1.0"; + else + envp[envc ++] = "SERVER_PROTOCOL=HTTP/0.9"; + + if (con->http.cookie) + { + snprintf(http_cookie, sizeof(http_cookie), "HTTP_COOKIE=%s", + con->http.cookie); + envp[envc ++] = http_cookie; + } + + if (con->http.fields[HTTP_FIELD_USER_AGENT][0]) + { + snprintf(http_user_agent, sizeof(http_user_agent), "HTTP_USER_AGENT=%s", + con->http.fields[HTTP_FIELD_USER_AGENT]); + envp[envc ++] = http_user_agent; + } + + if (con->http.fields[HTTP_FIELD_REFERER][0]) + { + snprintf(http_referer, sizeof(http_referer), "HTTP_REFERER=%s", + con->http.fields[HTTP_FIELD_REFERER]); + envp[envc ++] = http_referer; + } + + if (con->operation == HTTP_GET) + { + envp[envc ++] = "REQUEST_METHOD=GET"; + + if (con->query_string) + { + /* + * Add GET form variables after ?... + */ + + envp[envc ++] = con->query_string; + } + else + envp[envc ++] = "QUERY_STRING="; + } + else + { + sprintf(content_length, "CONTENT_LENGTH=" CUPS_LLFMT, + CUPS_LLCAST con->bytes); + snprintf(content_type, sizeof(content_type), "CONTENT_TYPE=%s", + con->http.fields[HTTP_FIELD_CONTENT_TYPE]); + + envp[envc ++] = "REQUEST_METHOD=POST"; + envp[envc ++] = content_length; + envp[envc ++] = content_type; + } + + /* + * Tell the CGI if we are using encryption... + */ + + if (con->http.tls) + envp[envc ++] = "HTTPS=ON"; + + /* + * Terminate the environment array... + */ + + envp[envc] = NULL; + + if (LogLevel >= CUPSD_LOG_DEBUG) + { + for (i = 0; i < argc; i ++) + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[CGI] argv[%d] = \"%s\"", i, argv[i]); + for (i = 0; i < envc; i ++) + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[CGI] envp[%d] = \"%s\"", i, envp[i]); + } + + /* + * Create a pipe for the output... + */ + + if (cupsdOpenPipe(fds)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "[CGI] Unable to create pipe for %s - %s", + argv[0], strerror(errno)); + return (0); + } + + /* + * Then execute the command... + */ + + if (cupsdStartProcess(command, argv, envp, infile, fds[1], CGIPipes[1], + -1, -1, root, DefaultProfile, NULL, &pid) < 0) + { + /* + * Error - can't fork! + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, "[CGI] Unable to start %s - %s", argv[0], + strerror(errno)); + + cupsdClosePipe(fds); + pid = 0; + } + else + { + /* + * Fork successful - return the PID... + */ + + if (con->username[0]) + cupsdAddCert(pid, con->username, ccache); + + cupsdLogMessage(CUPSD_LOG_DEBUG, "[CGI] Started %s (PID %d)", command, pid); + + *outfile = fds[0]; + close(fds[1]); + } + + return (pid); +} + + +/* + * 'valid_host()' - Is the Host: field valid? + */ + +static int /* O - 1 if valid, 0 if not */ +valid_host(cupsd_client_t *con) /* I - Client connection */ +{ + cupsd_alias_t *a; /* Current alias */ + cupsd_netif_t *netif; /* Current network interface */ + const char *host, /* Host field */ + *end; /* End character */ + + + host = con->http.fields[HTTP_FIELD_HOST]; + + if (httpAddrLocalhost(con->http.hostaddr)) + { + /* + * Only allow "localhost" or the equivalent IPv4 or IPv6 numerical + * addresses when accessing CUPS via the loopback interface... + */ + + return (!_cups_strcasecmp(host, "localhost") || + !_cups_strncasecmp(host, "localhost:", 10) || + !_cups_strcasecmp(host, "localhost.") || + !_cups_strncasecmp(host, "localhost.:", 11) || +#ifdef __linux + !_cups_strcasecmp(host, "localhost.localdomain") || + !_cups_strncasecmp(host, "localhost.localdomain:", 22) || +#endif /* __linux */ + !strcmp(host, "127.0.0.1") || + !strncmp(host, "127.0.0.1:", 10) || + !strcmp(host, "[::1]") || + !strncmp(host, "[::1]:", 6)); + } + +#ifdef HAVE_DNSSD + /* + * Check if the hostname is something.local (Bonjour); if so, allow it. + */ + + if ((end = strrchr(host, '.')) != NULL && + (!_cups_strcasecmp(end, ".local") || !_cups_strncasecmp(end, ".local:", 7) || + !_cups_strcasecmp(end, ".local.") || !_cups_strncasecmp(end, ".local.:", 8))) + return (1); +#endif /* HAVE_DNSSD */ + + /* + * Check if the hostname is an IP address... + */ + + if (isdigit(*host & 255) || *host == '[') + { + /* + * Possible IPv4/IPv6 address... + */ + + char temp[1024], /* Temporary string */ + *ptr; /* Pointer into temporary string */ + http_addrlist_t *addrlist; /* List of addresses */ + + + strlcpy(temp, host, sizeof(temp)); + if ((ptr = strrchr(temp, ':')) != NULL && !strchr(ptr, ']')) + *ptr = '\0'; /* Strip :port from host value */ + + if ((addrlist = httpAddrGetList(temp, AF_UNSPEC, NULL)) != NULL) + { + /* + * Good IPv4/IPv6 address... + */ + + httpAddrFreeList(addrlist); + return (1); + } + } + + /* + * Check for (alias) name matches... + */ + + for (a = (cupsd_alias_t *)cupsArrayFirst(ServerAlias); + a; + a = (cupsd_alias_t *)cupsArrayNext(ServerAlias)) + { + /* + * "ServerAlias *" allows all host values through... + */ + + if (!strcmp(a->name, "*")) + return (1); + + if (!_cups_strncasecmp(host, a->name, a->namelen)) + { + /* + * Prefix matches; check the character at the end - it must be ":", ".", + * ".:", or nul... + */ + + end = host + a->namelen; + + if (!*end || *end == ':' || (*end == '.' && (!end[1] || end[1] == ':'))) + return (1); + } + } + +#ifdef HAVE_DNSSD + for (a = (cupsd_alias_t *)cupsArrayFirst(DNSSDAlias); + a; + a = (cupsd_alias_t *)cupsArrayNext(DNSSDAlias)) + { + /* + * "ServerAlias *" allows all host values through... + */ + + if (!strcmp(a->name, "*")) + return (1); + + if (!_cups_strncasecmp(host, a->name, a->namelen)) + { + /* + * Prefix matches; check the character at the end - it must be ":", ".", + * ".:", or nul... + */ + + end = host + a->namelen; + + if (!*end || *end == ':' || (*end == '.' && (!end[1] || end[1] == ':'))) + return (1); + } + } +#endif /* HAVE_DNSSD */ + + /* + * Check for interface hostname matches... + */ + + for (netif = (cupsd_netif_t *)cupsArrayFirst(NetIFList); + netif; + netif = (cupsd_netif_t *)cupsArrayNext(NetIFList)) + { + if (!_cups_strncasecmp(host, netif->hostname, netif->hostlen)) + { + /* + * Prefix matches; check the character at the end - it must be ":", ".", + * ".:", or nul... + */ + + end = host + netif->hostlen; + + if (!*end || *end == ':' || (*end == '.' && (!end[1] || end[1] == ':'))) + return (1); + } + } + + return (0); +} + + +/* + * 'write_file()' - Send a file via HTTP. + */ + +static int /* O - 0 on failure, 1 on success */ +write_file(cupsd_client_t *con, /* I - Client connection */ + http_status_t code, /* I - HTTP status */ + char *filename, /* I - Filename */ + char *type, /* I - File type */ + struct stat *filestats) /* O - File information */ +{ + con->file = open(filename, O_RDONLY); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] write_file code=%d, filename=\"%s\" (%d), " + "type=\"%s\", filestats=%p", con->http.fd, + code, filename, con->file, type ? type : "(null)", filestats); + + if (con->file < 0) + return (0); + + fcntl(con->file, F_SETFD, fcntl(con->file, F_GETFD) | FD_CLOEXEC); + + con->pipe_pid = 0; + + if (!cupsdSendHeader(con, code, type, CUPSD_AUTH_NONE)) + return (0); + + if (httpPrintf(HTTP(con), "Last-Modified: %s\r\n", + httpGetDateString(filestats->st_mtime)) < 0) + return (0); + if (httpPrintf(HTTP(con), "Content-Length: " CUPS_LLFMT "\r\n", + CUPS_LLCAST filestats->st_size) < 0) + return (0); + if (httpPrintf(HTTP(con), "\r\n") < 0) + return (0); + + if (cupsdFlushHeader(con) < 0) + return (0); + + con->http.data_encoding = HTTP_ENCODE_LENGTH; + con->http.data_remaining = filestats->st_size; + + if (con->http.data_remaining <= INT_MAX) + con->http._data_remaining = con->http.data_remaining; + else + con->http._data_remaining = INT_MAX; + + cupsdAddSelect(con->http.fd, (cupsd_selfunc_t)cupsdReadClient, + (cupsd_selfunc_t)cupsdWriteClient, con); + + return (1); +} + + +/* + * 'write_pipe()' - Flag that data is available on the CGI pipe. + */ + +static void +write_pipe(cupsd_client_t *con) /* I - Client connection */ +{ + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "[Client %d] write_pipe CGI output on fd %d", + con->http.fd, con->file); + + con->file_ready = 1; + + cupsdRemoveSelect(con->file); + cupsdAddSelect(con->http.fd, NULL, (cupsd_selfunc_t)cupsdWriteClient, con); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/client.h b/scheduler/client.h new file mode 100644 index 0000000000..08dcdf190d --- /dev/null +++ b/scheduler/client.h @@ -0,0 +1,137 @@ +/* + * "$Id$" + * + * Client definitions for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#ifdef HAVE_AUTHORIZATION_H +# include +#endif /* HAVE_AUTHORIZATION_H */ + + +/* + * HTTP client structure... + */ + +struct cupsd_client_s +{ + http_t http; /* HTTP client connection */ + ipp_t *request, /* IPP request information */ + *response; /* IPP response information */ + cupsd_location_t *best; /* Best match for AAA */ + struct timeval start; /* Request start time */ + http_state_t operation; /* Request operation */ + off_t bytes; /* Bytes transferred for this request */ + int type; /* AuthType for username */ + char username[256], /* Username from Authorization: line */ + password[33], /* Password from Authorization: line */ + uri[HTTP_MAX_URI], + /* Localized URL/URI for GET/PUT */ + *filename, /* Filename of output file */ + *command, /* Command to run */ + *options, /* Options for command */ + *query_string; /* QUERY_STRING environment variable */ + int file; /* Input/output file */ + int file_ready; /* Input ready on file/pipe? */ + int pipe_pid; /* Pipe process ID (or 0 if not a pipe) */ + int sent_header, /* Non-zero if sent HTTP header */ + got_fields, /* Non-zero if all fields seen */ + header_used; /* Number of header bytes used */ + char header[2048]; /* Header from CGI program */ + cups_lang_t *language; /* Language to use */ +#ifdef HAVE_SSL + int auto_ssl; /* Automatic test for SSL/TLS */ +#endif /* HAVE_SSL */ + http_addr_t clientaddr; /* Client address */ + char servername[256];/* Server name for connection */ + int serverport; /* Server port for connection */ +#ifdef HAVE_GSSAPI + int have_gss; /* Have GSS credentials? */ + uid_t gss_uid; /* User ID for local prints */ +#endif /* HAVE_GSSAPI */ +#ifdef HAVE_AUTHORIZATION_H + AuthorizationRef authref; /* Authorization ref */ +#endif /* HAVE_AUTHORIZATION_H */ +}; + +#define HTTP(con) &((con)->http) + + +/* + * HTTP listener structure... + */ + +typedef struct +{ + int fd; /* File descriptor for this server */ + http_addr_t address; /* Bind address of socket */ + http_encryption_t encryption; /* To encrypt or not to encrypt... */ +} cupsd_listener_t; + + +/* + * Globals... + */ + +VAR int ListenBackLog VALUE(SOMAXCONN), + /* Max backlog of pending connections */ + LocalPort VALUE(631), + /* Local port to use */ + RemotePort VALUE(0); + /* Remote port to use */ +VAR http_encryption_t LocalEncryption VALUE(HTTP_ENCRYPT_IF_REQUESTED); + /* Local port encryption to use */ +VAR cups_array_t *Listeners VALUE(NULL); + /* Listening sockets */ +VAR time_t ListeningPaused VALUE(0); + /* Time when listening was paused */ +VAR cups_array_t *Clients VALUE(NULL), + /* HTTP clients */ + *ActiveClients VALUE(NULL); + /* Active HTTP clients */ +VAR char *ServerHeader VALUE(NULL); + /* Server header in requests */ +VAR int CGIPipes[2] VALUE2(-1,-1); + /* Pipes for CGI error/debug output */ +VAR cupsd_statbuf_t *CGIStatusBuffer VALUE(NULL); + /* Status buffer for pipes */ + + +/* + * Prototypes... + */ + +extern void cupsdAcceptClient(cupsd_listener_t *lis); +extern void cupsdCloseAllClients(void); +extern int cupsdCloseClient(cupsd_client_t *con); +extern void cupsdDeleteAllListeners(void); +extern int cupsdFlushHeader(cupsd_client_t *con); +extern void cupsdPauseListening(void); +extern int cupsdProcessIPPRequest(cupsd_client_t *con); +extern void cupsdReadClient(cupsd_client_t *con); +extern void cupsdResumeListening(void); +extern int cupsdSendCommand(cupsd_client_t *con, char *command, + char *options, int root); +extern int cupsdSendError(cupsd_client_t *con, http_status_t code, + int auth_type); +extern int cupsdSendHeader(cupsd_client_t *con, http_status_t code, + char *type, int auth_type); +extern void cupsdShutdownClient(cupsd_client_t *con); +extern void cupsdStartListening(void); +extern void cupsdStopListening(void); +extern void cupsdUpdateCGI(void); +extern void cupsdWriteClient(cupsd_client_t *con); + + +/* + * End of "$Id$". + */ diff --git a/scheduler/conf.c b/scheduler/conf.c new file mode 100644 index 0000000000..764d90251e --- /dev/null +++ b/scheduler/conf.c @@ -0,0 +1,3790 @@ +/* + * "$Id$" + * + * Configuration routines for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdAddAlias() - Add a host alias. + * cupsdCheckPermissions() - Fix the mode and ownership of a file or + * directory. + * cupsdDefaultAuthType() - Get the default AuthType. + * cupsdFreeAliases() - Free all of the alias entries. + * cupsdReadConfiguration() - Read the cupsd.conf file. + * get_address() - Get an address + port number from a line. + * get_addr_and_mask() - Get an IP address and netmask. + * mime_error_cb() - Log a MIME error. + * parse_aaa() - Parse authentication, authorization, and access + * control lines. + * parse_fatal_errors() - Parse FatalErrors values in a string. + * parse_groups() - Parse system group names in a string. + * parse_protocols() - Parse browse protocols in a string. + * read_configuration() - Read a configuration file. + * read_location() - Read a definition. + * read_policy() - Read a definition. + * set_policy_defaults() - Set default policy values as needed. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" +#include +#include +#include +#include + +#ifdef HAVE_LIBPAPER +# include +#endif /* HAVE_LIBPAPER */ + + +/* + * Possibly missing network definitions... + */ + +#ifndef INADDR_NONE +# define INADDR_NONE 0xffffffff +#endif /* !INADDR_NONE */ + + +/* + * Configuration variable structure... + */ + +typedef enum +{ + CUPSD_VARTYPE_INTEGER, /* Integer option */ + CUPSD_VARTYPE_STRING, /* String option */ + CUPSD_VARTYPE_BOOLEAN, /* Boolean option */ + CUPSD_VARTYPE_PATHNAME /* File/directory name option */ +} cupsd_vartype_t; + +typedef struct +{ + const char *name; /* Name of variable */ + void *ptr; /* Pointer to variable */ + cupsd_vartype_t type; /* Type (int, string, address) */ +} cupsd_var_t; + + +/* + * Local globals... + */ + +static int default_auth_type = CUPSD_AUTH_AUTO; + /* Default AuthType, if not specified */ +static const cupsd_var_t variables[] = +{ + { "AccessLog", &AccessLog, CUPSD_VARTYPE_STRING }, + { "AutoPurgeJobs", &JobAutoPurge, CUPSD_VARTYPE_BOOLEAN }, +#ifdef HAVE_DNSSD + { "BrowseDNSSDRegType", &DNSSDRegType, CUPSD_VARTYPE_STRING }, +#endif /* HAVE_DNSSD */ + { "BrowseWebIF", &BrowseWebIF, CUPSD_VARTYPE_BOOLEAN }, + { "Browsing", &Browsing, CUPSD_VARTYPE_BOOLEAN }, + { "CacheDir", &CacheDir, CUPSD_VARTYPE_STRING }, + { "Classification", &Classification, CUPSD_VARTYPE_STRING }, + { "ClassifyOverride", &ClassifyOverride, CUPSD_VARTYPE_BOOLEAN }, + { "ConfigFilePerm", &ConfigFilePerm, CUPSD_VARTYPE_INTEGER }, + { "DataDir", &DataDir, CUPSD_VARTYPE_STRING }, + { "DefaultLanguage", &DefaultLanguage, CUPSD_VARTYPE_STRING }, + { "DefaultLeaseDuration", &DefaultLeaseDuration, CUPSD_VARTYPE_INTEGER }, + { "DefaultPaperSize", &DefaultPaperSize, CUPSD_VARTYPE_STRING }, + { "DefaultPolicy", &DefaultPolicy, CUPSD_VARTYPE_STRING }, + { "DefaultShared", &DefaultShared, CUPSD_VARTYPE_BOOLEAN }, + { "DirtyCleanInterval", &DirtyCleanInterval, CUPSD_VARTYPE_INTEGER }, + { "DocumentRoot", &DocumentRoot, CUPSD_VARTYPE_STRING }, + { "ErrorLog", &ErrorLog, CUPSD_VARTYPE_STRING }, + { "ErrorPolicy", &ErrorPolicy, CUPSD_VARTYPE_STRING }, + { "FileDevice", &FileDevice, CUPSD_VARTYPE_BOOLEAN }, + { "FilterLimit", &FilterLimit, CUPSD_VARTYPE_INTEGER }, + { "FilterNice", &FilterNice, CUPSD_VARTYPE_INTEGER }, + { "FontPath", &FontPath, CUPSD_VARTYPE_STRING }, +#ifdef HAVE_GSSAPI + { "GSSServiceName", &GSSServiceName, CUPSD_VARTYPE_STRING }, +#endif /* HAVE_GSSAPI */ + { "JobKillDelay", &JobKillDelay, CUPSD_VARTYPE_INTEGER }, + { "JobRetryLimit", &JobRetryLimit, CUPSD_VARTYPE_INTEGER }, + { "JobRetryInterval", &JobRetryInterval, CUPSD_VARTYPE_INTEGER }, + { "KeepAliveTimeout", &KeepAliveTimeout, CUPSD_VARTYPE_INTEGER }, + { "KeepAlive", &KeepAlive, CUPSD_VARTYPE_BOOLEAN }, +#ifdef HAVE_LAUNCHD + { "LaunchdTimeout", &LaunchdTimeout, CUPSD_VARTYPE_INTEGER }, +#endif /* HAVE_LAUNCHD */ + { "LimitRequestBody", &MaxRequestSize, CUPSD_VARTYPE_INTEGER }, + { "ListenBackLog", &ListenBackLog, CUPSD_VARTYPE_INTEGER }, + { "LogDebugHistory", &LogDebugHistory, CUPSD_VARTYPE_INTEGER }, + { "LogFilePerm", &LogFilePerm, CUPSD_VARTYPE_INTEGER }, + { "LPDConfigFile", &LPDConfigFile, CUPSD_VARTYPE_STRING }, + { "MaxActiveJobs", &MaxActiveJobs, CUPSD_VARTYPE_INTEGER }, + { "MaxClients", &MaxClients, CUPSD_VARTYPE_INTEGER }, + { "MaxClientsPerHost", &MaxClientsPerHost, CUPSD_VARTYPE_INTEGER }, + { "MaxCopies", &MaxCopies, CUPSD_VARTYPE_INTEGER }, + { "MaxEvents", &MaxEvents, CUPSD_VARTYPE_INTEGER }, + { "MaxJobs", &MaxJobs, CUPSD_VARTYPE_INTEGER }, + { "MaxJobsPerPrinter", &MaxJobsPerPrinter, CUPSD_VARTYPE_INTEGER }, + { "MaxJobsPerUser", &MaxJobsPerUser, CUPSD_VARTYPE_INTEGER }, + { "MaxJobTime", &MaxJobTime, CUPSD_VARTYPE_INTEGER }, + { "MaxLeaseDuration", &MaxLeaseDuration, CUPSD_VARTYPE_INTEGER }, + { "MaxLogSize", &MaxLogSize, CUPSD_VARTYPE_INTEGER }, + { "MaxRequestSize", &MaxRequestSize, CUPSD_VARTYPE_INTEGER }, + { "MaxSubscriptions", &MaxSubscriptions, CUPSD_VARTYPE_INTEGER }, + { "MaxSubscriptionsPerJob", &MaxSubscriptionsPerJob, CUPSD_VARTYPE_INTEGER }, + { "MaxSubscriptionsPerPrinter",&MaxSubscriptionsPerPrinter, CUPSD_VARTYPE_INTEGER }, + { "MaxSubscriptionsPerUser", &MaxSubscriptionsPerUser, CUPSD_VARTYPE_INTEGER }, + { "MultipleOperationTimeout", &MultipleOperationTimeout, CUPSD_VARTYPE_INTEGER }, + { "PageLog", &PageLog, CUPSD_VARTYPE_STRING }, + { "PageLogFormat", &PageLogFormat, CUPSD_VARTYPE_STRING }, + { "PreserveJobFiles", &JobFiles, CUPSD_VARTYPE_BOOLEAN }, + { "PreserveJobHistory", &JobHistory, CUPSD_VARTYPE_BOOLEAN }, + { "Printcap", &Printcap, CUPSD_VARTYPE_STRING }, + { "PrintcapGUI", &PrintcapGUI, CUPSD_VARTYPE_STRING }, + { "ReloadTimeout", &ReloadTimeout, CUPSD_VARTYPE_INTEGER }, + { "RemoteRoot", &RemoteRoot, CUPSD_VARTYPE_STRING }, + { "RequestRoot", &RequestRoot, CUPSD_VARTYPE_STRING }, + { "RIPCache", &RIPCache, CUPSD_VARTYPE_STRING }, + { "RootCertDuration", &RootCertDuration, CUPSD_VARTYPE_INTEGER }, + { "ServerAdmin", &ServerAdmin, CUPSD_VARTYPE_STRING }, + { "ServerBin", &ServerBin, CUPSD_VARTYPE_PATHNAME }, +#ifdef HAVE_SSL + { "ServerCertificate", &ServerCertificate, CUPSD_VARTYPE_PATHNAME }, +# if defined(HAVE_LIBSSL) || defined(HAVE_GNUTLS) + { "ServerKey", &ServerKey, CUPSD_VARTYPE_PATHNAME }, +# endif /* HAVE_LIBSSL || HAVE_GNUTLS */ +#endif /* HAVE_SSL */ + { "ServerName", &ServerName, CUPSD_VARTYPE_STRING }, + { "ServerRoot", &ServerRoot, CUPSD_VARTYPE_PATHNAME }, + { "SMBConfigFile", &SMBConfigFile, CUPSD_VARTYPE_STRING }, + { "StateDir", &StateDir, CUPSD_VARTYPE_STRING }, +#ifdef HAVE_AUTHORIZATION_H + { "SystemGroupAuthKey", &SystemGroupAuthKey, CUPSD_VARTYPE_STRING }, +#endif /* HAVE_AUTHORIZATION_H */ + { "TempDir", &TempDir, CUPSD_VARTYPE_PATHNAME }, + { "Timeout", &Timeout, CUPSD_VARTYPE_INTEGER }, + { "WebInterface", &WebInterface, CUPSD_VARTYPE_BOOLEAN } +}; +#define NUM_VARS (sizeof(variables) / sizeof(variables[0])) + + +static const unsigned ones[4] = + { + 0xffffffff, 0xffffffff, 0xffffffff, 0xffffffff + }; +static const unsigned zeros[4] = + { + 0x00000000, 0x00000000, 0x00000000, 0x00000000 + }; + + +/* + * Local functions... + */ + +static http_addrlist_t *get_address(const char *value, int defport); +static int get_addr_and_mask(const char *value, unsigned *ip, + unsigned *mask); +static void mime_error_cb(void *ctx, const char *message); +static int parse_aaa(cupsd_location_t *loc, char *line, + char *value, int linenum); +static int parse_fatal_errors(const char *s); +static int parse_groups(const char *s); +static int parse_protocols(const char *s); +static int read_configuration(cups_file_t *fp); +static int read_location(cups_file_t *fp, char *name, int linenum); +static int read_policy(cups_file_t *fp, char *name, int linenum); +static void set_policy_defaults(cupsd_policy_t *pol); + + +/* + * 'cupsdAddAlias()' - Add a host alias. + */ + +void +cupsdAddAlias(cups_array_t *aliases, /* I - Array of aliases */ + const char *name) /* I - Name to add */ +{ + cupsd_alias_t *a; /* New alias */ + size_t namelen; /* Length of name */ + + + namelen = strlen(name); + + if ((a = (cupsd_alias_t *)malloc(sizeof(cupsd_alias_t) + namelen)) == NULL) + return; + + a->namelen = namelen; + strcpy(a->name, name); /* OK since a->name is allocated */ + + cupsArrayAdd(aliases, a); +} + + +/* + * 'cupsdCheckPermissions()' - Fix the mode and ownership of a file or directory. + */ + +int /* O - 0 on success, -1 on error, 1 on warning */ +cupsdCheckPermissions( + const char *filename, /* I - File/directory name */ + const char *suffix, /* I - Additional file/directory name */ + int mode, /* I - Permissions */ + int user, /* I - Owner */ + int group, /* I - Group */ + int is_dir, /* I - 1 = directory, 0 = file */ + int create_dir) /* I - 1 = create directory, -1 = create w/o logging, 0 = not */ +{ + int dir_created = 0; /* Did we create a directory? */ + char pathname[1024]; /* File name with prefix */ + struct stat fileinfo; /* Stat buffer */ + int is_symlink; /* Is "filename" a symlink? */ + + + /* + * Prepend the given root to the filename before testing it... + */ + + if (suffix) + { + snprintf(pathname, sizeof(pathname), "%s/%s", filename, suffix); + filename = pathname; + } + + /* + * See if we can stat the file/directory... + */ + + if (lstat(filename, &fileinfo)) + { + if (errno == ENOENT && create_dir) + { + if (create_dir > 0) + cupsdLogMessage(CUPSD_LOG_DEBUG, "Creating missing directory \"%s\"", + filename); + + if (mkdir(filename, mode)) + { + if (create_dir > 0) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to create directory \"%s\" - %s", filename, + strerror(errno)); + else + syslog(LOG_ERR, "Unable to create directory \"%s\" - %s", filename, + strerror(errno)); + + return (-1); + } + + dir_created = 1; + fileinfo.st_mode = mode | S_IFDIR; + } + else + return (create_dir ? -1 : 1); + } + + if ((is_symlink = S_ISLNK(fileinfo.st_mode)) != 0) + { + if (stat(filename, &fileinfo)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "\"%s\" is a bad symlink - %s", + filename, strerror(errno)); + return (-1); + } + } + + /* + * Make sure it's a regular file or a directory as needed... + */ + + if (!dir_created && !is_dir && !S_ISREG(fileinfo.st_mode)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "\"%s\" is not a regular file.", filename); + return (-1); + } + + if (!dir_created && is_dir && !S_ISDIR(fileinfo.st_mode)) + { + if (create_dir >= 0) + cupsdLogMessage(CUPSD_LOG_ERROR, "\"%s\" is not a directory.", filename); + else + syslog(LOG_ERR, "\"%s\" is not a directory.", filename); + + return (-1); + } + + /* + * If the filename is a symlink, do not change permissions (STR #2937)... + */ + + if (is_symlink) + return (0); + + /* + * Fix owner, group, and mode as needed... + */ + + if (dir_created || fileinfo.st_uid != user || fileinfo.st_gid != group) + { + if (create_dir >= 0) + cupsdLogMessage(CUPSD_LOG_DEBUG, "Repairing ownership of \"%s\"", + filename); + + if (chown(filename, user, group) && !getuid()) + { + if (create_dir >= 0) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to change ownership of \"%s\" - %s", filename, + strerror(errno)); + else + syslog(LOG_ERR, "Unable to change ownership of \"%s\" - %s", filename, + strerror(errno)); + + return (1); + } + } + + if (dir_created || (fileinfo.st_mode & 07777) != mode) + { + if (create_dir >= 0) + cupsdLogMessage(CUPSD_LOG_DEBUG, "Repairing access permissions of \"%s\"", + filename); + + if (chmod(filename, mode)) + { + if (create_dir >= 0) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to change permissions of \"%s\" - %s", filename, + strerror(errno)); + else + syslog(LOG_ERR, "Unable to change permissions of \"%s\" - %s", filename, + strerror(errno)); + + return (1); + } + } + + /* + * Everything is OK... + */ + + return (0); +} + + +/* + * 'cupsdDefaultAuthType()' - Get the default AuthType. + * + * When the default_auth_type is "auto", this function tries to get the GSS + * credentials for the server. If that succeeds we use Kerberos authentication, + * otherwise we do a fallback to Basic authentication against the local user + * accounts. + */ + +int /* O - Default AuthType value */ +cupsdDefaultAuthType(void) +{ +#ifdef HAVE_GSSAPI + OM_uint32 major_status, /* Major status code */ + minor_status; /* Minor status code */ + gss_name_t server_name; /* Server name */ + gss_buffer_desc token = GSS_C_EMPTY_BUFFER; + /* Service name token */ + char buf[1024]; /* Service name buffer */ +#endif /* HAVE_GSSAPI */ + + + /* + * If we have already determined the correct default AuthType, use it... + */ + + if (default_auth_type != CUPSD_AUTH_AUTO) + return (default_auth_type); + +#ifdef HAVE_GSSAPI +# ifdef __APPLE__ + /* + * If the weak-linked GSSAPI/Kerberos library is not present, don't try + * to use it... + */ + + if (gss_init_sec_context == NULL) + return (default_auth_type = CUPSD_AUTH_BASIC); +# endif /* __APPLE__ */ + + /* + * Try to obtain the server's GSS credentials (GSSServiceName@servername). If + * that fails we must use Basic... + */ + + snprintf(buf, sizeof(buf), "%s@%s", GSSServiceName, ServerName); + + token.value = buf; + token.length = strlen(buf); + server_name = GSS_C_NO_NAME; + major_status = gss_import_name(&minor_status, &token, + GSS_C_NT_HOSTBASED_SERVICE, + &server_name); + + memset(&token, 0, sizeof(token)); + + if (GSS_ERROR(major_status)) + { + cupsdLogGSSMessage(CUPSD_LOG_DEBUG, major_status, minor_status, + "cupsdDefaultAuthType: gss_import_name(%s) failed", buf); + return (default_auth_type = CUPSD_AUTH_BASIC); + } + + major_status = gss_display_name(&minor_status, server_name, &token, NULL); + + if (GSS_ERROR(major_status)) + { + cupsdLogGSSMessage(CUPSD_LOG_DEBUG, major_status, minor_status, + "cupsdDefaultAuthType: gss_display_name(%s) failed", + buf); + return (default_auth_type = CUPSD_AUTH_BASIC); + } + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "cupsdDefaultAuthType: Attempting to acquire Kerberos " + "credentials for %s...", (char *)token.value); + + ServerCreds = GSS_C_NO_CREDENTIAL; + major_status = gss_acquire_cred(&minor_status, server_name, GSS_C_INDEFINITE, + GSS_C_NO_OID_SET, GSS_C_ACCEPT, + &ServerCreds, NULL, NULL); + if (GSS_ERROR(major_status)) + { + cupsdLogGSSMessage(CUPSD_LOG_DEBUG, major_status, minor_status, + "cupsdDefaultAuthType: gss_acquire_cred(%s) failed", + (char *)token.value); + gss_release_name(&minor_status, &server_name); + gss_release_buffer(&minor_status, &token); + return (default_auth_type = CUPSD_AUTH_BASIC); + } + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "cupsdDefaultAuthType: Kerberos credentials acquired " + "successfully for %s.", (char *)token.value); + + gss_release_name(&minor_status, &server_name); + gss_release_buffer(&minor_status, &token); + + HaveServerCreds = 1; + + return (default_auth_type = CUPSD_AUTH_NEGOTIATE); + +#else + /* + * No Kerberos support compiled in so just use Basic all the time... + */ + + return (default_auth_type = CUPSD_AUTH_BASIC); +#endif /* HAVE_GSSAPI */ +} + + +/* + * 'cupsdFreeAliases()' - Free all of the alias entries. + */ + +void +cupsdFreeAliases(cups_array_t *aliases) /* I - Array of aliases */ +{ + cupsd_alias_t *a; /* Current alias */ + + + for (a = (cupsd_alias_t *)cupsArrayFirst(aliases); + a; + a = (cupsd_alias_t *)cupsArrayNext(aliases)) + free(a); + + cupsArrayDelete(aliases); +} + + +/* + * 'cupsdReadConfiguration()' - Read the cupsd.conf file. + */ + +int /* O - 1 on success, 0 otherwise */ +cupsdReadConfiguration(void) +{ + int i; /* Looping var */ + cups_file_t *fp; /* Configuration file */ + int status; /* Return status */ + char temp[1024], /* Temporary buffer */ + mimedir[1024], /* MIME directory */ + *slash; /* Directory separator */ + cups_lang_t *language; /* Language */ + struct passwd *user; /* Default user */ + struct group *group; /* Default group */ + char *old_serverroot, /* Old ServerRoot */ + *old_requestroot; /* Old RequestRoot */ + int old_remote_port; /* Old RemotePort */ + const char *tmpdir; /* TMPDIR environment variable */ + struct stat tmpinfo; /* Temporary directory info */ + cupsd_policy_t *p; /* Policy */ + + + /* + * Save the old root paths... + */ + + old_serverroot = NULL; + cupsdSetString(&old_serverroot, ServerRoot); + old_requestroot = NULL; + cupsdSetString(&old_requestroot, RequestRoot); + + /* + * Reset the server configuration data... + */ + + cupsdDeleteAllLocations(); + + cupsdDeleteAllListeners(); + + old_remote_port = RemotePort; + RemotePort = 0; + + /* + * String options... + */ + + cupsdFreeAliases(ServerAlias); + ServerAlias = NULL; + + cupsdClearString(&ServerName); + cupsdClearString(&ServerAdmin); + cupsdSetString(&ServerBin, CUPS_SERVERBIN); + cupsdSetString(&RequestRoot, CUPS_REQUESTS); + cupsdSetString(&CacheDir, CUPS_CACHEDIR); + cupsdSetString(&DataDir, CUPS_DATADIR); + cupsdSetString(&DocumentRoot, CUPS_DOCROOT); + cupsdSetString(&AccessLog, CUPS_LOGDIR "/access_log"); + cupsdClearString(&ErrorLog); + cupsdSetString(&PageLog, CUPS_LOGDIR "/page_log"); + cupsdSetString(&PageLogFormat, + "%p %u %j %T %P %C %{job-billing} " + "%{job-originating-host-name} %{job-name} %{media} %{sides}"); + cupsdSetString(&Printcap, CUPS_DEFAULT_PRINTCAP); + cupsdSetString(&PrintcapGUI, "/usr/bin/glpoptions"); + cupsdSetString(&FontPath, CUPS_FONTPATH); + cupsdSetString(&RemoteRoot, "remroot"); + cupsdSetStringf(&ServerHeader, "CUPS/%d.%d", CUPS_VERSION_MAJOR, + CUPS_VERSION_MINOR); + cupsdSetString(&StateDir, CUPS_STATEDIR); + + if (!strcmp(CUPS_DEFAULT_PRINTCAP, "/etc/printers.conf")) + PrintcapFormat = PRINTCAP_SOLARIS; + else if (!strcmp(CUPS_DEFAULT_PRINTCAP, + "/Library/Preferences/org.cups.printers.plist")) + PrintcapFormat = PRINTCAP_PLIST; + else + PrintcapFormat = PRINTCAP_BSD; + + strlcpy(temp, ConfigurationFile, sizeof(temp)); + if ((slash = strrchr(temp, '/')) != NULL) + *slash = '\0'; + + cupsdSetString(&ServerRoot, temp); + + cupsdClearString(&Classification); + ClassifyOverride = 0; + +#ifdef HAVE_SSL +# ifdef HAVE_CDSASSL + cupsdSetString(&ServerCertificate, "/Library/Keychains/System.keychain"); +# else + cupsdSetString(&ServerCertificate, "ssl/server.crt"); + cupsdSetString(&ServerKey, "ssl/server.key"); +# endif /* HAVE_CDSASSL */ +#endif /* HAVE_SSL */ + + language = cupsLangDefault(); + + if (!strcmp(language->language, "C") || !strcmp(language->language, "POSIX")) + cupsdSetString(&DefaultLanguage, "en"); + else + cupsdSetString(&DefaultLanguage, language->language); + + cupsdClearString(&DefaultPaperSize); + + cupsdSetString(&RIPCache, "128m"); + + cupsdSetString(&TempDir, NULL); + +#ifdef HAVE_GSSAPI + cupsdSetString(&GSSServiceName, CUPS_DEFAULT_GSSSERVICENAME); + + if (HaveServerCreds) + { + OM_uint32 minor_status; /* Minor status code */ + + gss_release_cred(&minor_status, &ServerCreds); + + HaveServerCreds = 0; + } + + ServerCreds = GSS_C_NO_CREDENTIAL; +#endif /* HAVE_GSSAPI */ + + /* + * Find the default user... + */ + + if ((user = getpwnam(CUPS_DEFAULT_USER)) != NULL) + User = user->pw_uid; + else + { + /* + * Use the (historical) NFS nobody user ID (-2 as a 16-bit twos- + * complement number...) + */ + + User = 65534; + } + + endpwent(); + + /* + * Find the default group... + */ + + group = getgrnam(CUPS_DEFAULT_GROUP); + endgrent(); + + if (group) + Group = group->gr_gid; + else + { + /* + * Fallback to group "nobody"... + */ + + group = getgrnam("nobody"); + endgrent(); + + if (group) + Group = group->gr_gid; + else + { + /* + * Use the (historical) NFS nobody group ID (-2 as a 16-bit twos- + * complement number...) + */ + + Group = 65534; + } + } + + /* + * Numeric options... + */ + + AccessLogLevel = CUPSD_ACCESSLOG_ACTIONS; + ConfigFilePerm = CUPS_DEFAULT_CONFIG_FILE_PERM; + FatalErrors = parse_fatal_errors(CUPS_DEFAULT_FATAL_ERRORS); + default_auth_type = CUPSD_AUTH_BASIC; +#ifdef HAVE_SSL + DefaultEncryption = HTTP_ENCRYPT_REQUIRED; + SSLOptions = CUPSD_SSL_NONE; +#endif /* HAVE_SSL */ + DirtyCleanInterval = DEFAULT_KEEPALIVE; + JobKillDelay = DEFAULT_TIMEOUT; + JobRetryLimit = 5; + JobRetryInterval = 300; + FileDevice = FALSE; + FilterLevel = 0; + FilterLimit = 0; + FilterNice = 0; + HostNameLookups = FALSE; + KeepAlive = TRUE; + KeepAliveTimeout = DEFAULT_KEEPALIVE; + ListenBackLog = SOMAXCONN; + LogDebugHistory = 200; + LogFilePerm = CUPS_DEFAULT_LOG_FILE_PERM; + LogLevel = CUPSD_LOG_WARN; + LogTimeFormat = CUPSD_TIME_STANDARD; + MaxClients = 100; + MaxClientsPerHost = 0; + MaxLogSize = 1024 * 1024; + MaxRequestSize = 0; + MultipleOperationTimeout = DEFAULT_TIMEOUT; + ReloadTimeout = DEFAULT_KEEPALIVE; + RootCertDuration = 300; + Timeout = DEFAULT_TIMEOUT; + NumSystemGroups = 0; + WebInterface = CUPS_DEFAULT_WEBIF; + + BrowseLocalProtocols = parse_protocols(CUPS_DEFAULT_BROWSE_LOCAL_PROTOCOLS); + BrowseWebIF = FALSE; + Browsing = CUPS_DEFAULT_BROWSING; + DefaultShared = CUPS_DEFAULT_DEFAULT_SHARED; + +#ifdef HAVE_DNSSD + cupsdSetString(&DNSSDRegType, "_ipp._tcp,_cups"); +#endif /* HAVE_DNSSD */ + + cupsdSetString(&LPDConfigFile, CUPS_DEFAULT_LPD_CONFIG_FILE); + cupsdSetString(&SMBConfigFile, CUPS_DEFAULT_SMB_CONFIG_FILE); + + cupsdSetString(&ErrorPolicy, "stop-printer"); + + JobHistory = DEFAULT_HISTORY; + JobFiles = DEFAULT_FILES; + JobAutoPurge = 0; + MaxJobs = 500; + MaxActiveJobs = 0; + MaxJobsPerUser = 0; + MaxJobsPerPrinter = 0; + MaxJobTime = 3 * 60 * 60; /* 3 hours */ + MaxCopies = CUPS_DEFAULT_MAX_COPIES; + + cupsdDeleteAllPolicies(); + cupsdClearString(&DefaultPolicy); + +#ifdef HAVE_AUTHORIZATION_H + cupsdClearString(&SystemGroupAuthKey); +#endif /* HAVE_AUTHORIZATION_H */ + + MaxSubscriptions = 100; + MaxSubscriptionsPerJob = 0; + MaxSubscriptionsPerPrinter = 0; + MaxSubscriptionsPerUser = 0; + DefaultLeaseDuration = 86400; + MaxLeaseDuration = 0; + +#ifdef HAVE_LAUNCHD + LaunchdTimeout = 10; +#endif /* HAVE_LAUNCHD */ + + /* + * Setup environment variables... + */ + + cupsdInitEnv(); + + /* + * Read the configuration file... + */ + + if ((fp = cupsFileOpen(ConfigurationFile, "r")) == NULL) + return (0); + + status = read_configuration(fp); + + cupsFileClose(fp); + + if (!status) + return (0); + + if (!ErrorLog) + cupsdSetString(&ErrorLog, CUPS_LOGDIR "/error_log"); + + RunUser = getuid(); + + cupsdLogMessage(CUPSD_LOG_INFO, "Remote access is %s.", + RemotePort ? "enabled" : "disabled"); + + if (!RemotePort) + BrowseLocalProtocols = 0; /* Disable sharing - no remote access */ + + /* + * See if the ServerName is an IP address... + */ + + if (ServerName) + { + if (!ServerAlias) + ServerAlias = cupsArrayNew(NULL, NULL); + + cupsdLogMessage(CUPSD_LOG_DEBUG, "Added auto ServerAlias %s", ServerName); + } + else + { + if (gethostname(temp, sizeof(temp))) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to get hostname: %s", + strerror(errno)); + strlcpy(temp, "localhost", sizeof(temp)); + } + + cupsdSetString(&ServerName, temp); + + if (!ServerAlias) + ServerAlias = cupsArrayNew(NULL, NULL); + + cupsdAddAlias(ServerAlias, temp); + cupsdLogMessage(CUPSD_LOG_DEBUG, "Added auto ServerAlias %s", temp); + + if (HostNameLookups || RemotePort) + { + struct hostent *host; /* Host entry to get FQDN */ + + if ((host = gethostbyname(temp)) != NULL) + { + if (_cups_strcasecmp(temp, host->h_name)) + { + cupsdSetString(&ServerName, host->h_name); + cupsdAddAlias(ServerAlias, host->h_name); + cupsdLogMessage(CUPSD_LOG_DEBUG, "Added auto ServerAlias %s", + host->h_name); + } + + if (host->h_aliases) + { + for (i = 0; host->h_aliases[i]; i ++) + if (_cups_strcasecmp(temp, host->h_aliases[i])) + { + cupsdAddAlias(ServerAlias, host->h_aliases[i]); + cupsdLogMessage(CUPSD_LOG_DEBUG, "Added auto ServerAlias %s", + host->h_aliases[i]); + } + } + } + } + + /* + * Make sure we have the base hostname added as an alias, too! + */ + + if ((slash = strchr(temp, '.')) != NULL) + { + *slash = '\0'; + cupsdAddAlias(ServerAlias, temp); + cupsdLogMessage(CUPSD_LOG_DEBUG, "Added auto ServerAlias %s", temp); + } + } + + for (slash = ServerName; isdigit(*slash & 255) || *slash == '.'; slash ++); + + ServerNameIsIP = !*slash; + + /* + * Make sure ServerAdmin is initialized... + */ + + if (!ServerAdmin) + cupsdSetStringf(&ServerAdmin, "root@%s", ServerName); + + /* + * Use the default system group if none was supplied in cupsd.conf... + */ + + if (NumSystemGroups == 0) + { + if (!parse_groups(CUPS_DEFAULT_SYSTEM_GROUPS)) + { + /* + * Find the group associated with GID 0... + */ + + group = getgrgid(0); + endgrent(); + + if (group != NULL) + cupsdSetString(&SystemGroups[0], group->gr_name); + else + cupsdSetString(&SystemGroups[0], "unknown"); + + SystemGroupIDs[0] = 0; + NumSystemGroups = 1; + } + } + + /* + * Open the system log for cupsd if necessary... + */ + +#ifdef HAVE_VSYSLOG + if (!strcmp(AccessLog, "syslog") || + !strcmp(ErrorLog, "syslog") || + !strcmp(PageLog, "syslog")) + openlog("cupsd", LOG_PID | LOG_NOWAIT | LOG_NDELAY, LOG_LPR); +#endif /* HAVE_VSYSLOG */ + + /* + * Make sure each of the log files exists and gets rotated as necessary... + */ + + if (strcmp(AccessLog, "syslog")) + cupsdCheckLogFile(&AccessFile, AccessLog); + + if (strcmp(ErrorLog, "syslog")) + cupsdCheckLogFile(&ErrorFile, ErrorLog); + + if (strcmp(PageLog, "syslog")) + cupsdCheckLogFile(&PageFile, PageLog); + + /* + * Log the configuration file that was used... + */ + + cupsdLogMessage(CUPSD_LOG_INFO, "Loaded configuration file \"%s\"", + ConfigurationFile); + + /* + * Validate the Group and SystemGroup settings - they cannot be the same, + * otherwise the CGI programs will be able to authenticate as root without + * a password! + */ + + if (!RunUser) + { + for (i = 0; i < NumSystemGroups; i ++) + if (Group == SystemGroupIDs[i]) + break; + + if (i < NumSystemGroups) + { + /* + * Log the error and reset the group to a safe value... + */ + + cupsdLogMessage(CUPSD_LOG_NOTICE, + "Group and SystemGroup cannot use the same groups."); + cupsdLogMessage(CUPSD_LOG_INFO, "Resetting Group to \"nobody\"..."); + + group = getgrnam("nobody"); + endgrent(); + + if (group != NULL) + Group = group->gr_gid; + else + { + /* + * Use the (historical) NFS nobody group ID (-2 as a 16-bit twos- + * complement number...) + */ + + Group = 65534; + } + } + } + + /* + * Check that we have at least one listen/port line; if not, report this + * as an error and exit! + */ + + if (cupsArrayCount(Listeners) == 0) + { + /* + * No listeners! + */ + + cupsdLogMessage(CUPSD_LOG_EMERG, + "No valid Listen or Port lines were found in the " + "configuration file."); + + /* + * Commit suicide... + */ + + cupsdEndProcess(getpid(), 0); + } + + /* + * Set the default locale using the language and charset... + */ + + cupsdSetStringf(&DefaultLocale, "%s.UTF-8", DefaultLanguage); + + /* + * Update all relative filenames to include the full path from ServerRoot... + */ + + if (DocumentRoot[0] != '/') + cupsdSetStringf(&DocumentRoot, "%s/%s", ServerRoot, DocumentRoot); + + if (RequestRoot[0] != '/') + cupsdSetStringf(&RequestRoot, "%s/%s", ServerRoot, RequestRoot); + + if (ServerBin[0] != '/') + cupsdSetStringf(&ServerBin, "%s/%s", ServerRoot, ServerBin); + + if (StateDir[0] != '/') + cupsdSetStringf(&StateDir, "%s/%s", ServerRoot, StateDir); + + if (CacheDir[0] != '/') + cupsdSetStringf(&CacheDir, "%s/%s", ServerRoot, CacheDir); + +#ifdef HAVE_SSL + if (ServerCertificate[0] != '/') + cupsdSetStringf(&ServerCertificate, "%s/%s", ServerRoot, ServerCertificate); + + if (!strncmp(ServerRoot, ServerCertificate, strlen(ServerRoot)) && + cupsdCheckPermissions(ServerCertificate, NULL, 0600, RunUser, Group, + 0, 0) < 0 && + (FatalErrors & CUPSD_FATAL_PERMISSIONS)) + return (0); + +# if defined(HAVE_LIBSSL) || defined(HAVE_GNUTLS) + if (ServerKey[0] != '/') + cupsdSetStringf(&ServerKey, "%s/%s", ServerRoot, ServerKey); + + if (!strncmp(ServerRoot, ServerKey, strlen(ServerRoot)) && + cupsdCheckPermissions(ServerKey, NULL, 0600, RunUser, Group, 0, 0) < 0 && + (FatalErrors & CUPSD_FATAL_PERMISSIONS)) + return (0); +# endif /* HAVE_LIBSSL || HAVE_GNUTLS */ +#endif /* HAVE_SSL */ + + /* + * Make sure that directories and config files are owned and + * writable by the user and group in the cupsd.conf file... + */ + + snprintf(temp, sizeof(temp), "%s/rss", CacheDir); + + if ((cupsdCheckPermissions(RequestRoot, NULL, 0710, RunUser, + Group, 1, 1) < 0 || + cupsdCheckPermissions(CacheDir, NULL, 0775, RunUser, + Group, 1, 1) < 0 || + cupsdCheckPermissions(temp, NULL, 0775, RunUser, + Group, 1, 1) < 0 || + cupsdCheckPermissions(StateDir, NULL, 0755, RunUser, + Group, 1, 1) < 0 || + cupsdCheckPermissions(StateDir, "certs", RunUser ? 0711 : 0511, User, + SystemGroupIDs[0], 1, 1) < 0 || + cupsdCheckPermissions(ServerRoot, NULL, 0755, RunUser, + Group, 1, 0) < 0 || + cupsdCheckPermissions(ServerRoot, "ppd", 0755, RunUser, + Group, 1, 1) < 0 || + cupsdCheckPermissions(ServerRoot, "ssl", 0700, RunUser, + Group, 1, 0) < 0 || + cupsdCheckPermissions(ServerRoot, "cupsd.conf", ConfigFilePerm, RunUser, + Group, 0, 0) < 0 || + cupsdCheckPermissions(ServerRoot, "classes.conf", 0600, RunUser, + Group, 0, 0) < 0 || + cupsdCheckPermissions(ServerRoot, "printers.conf", 0600, RunUser, + Group, 0, 0) < 0 || + cupsdCheckPermissions(ServerRoot, "passwd.md5", 0600, User, + Group, 0, 0) < 0) && + (FatalErrors & CUPSD_FATAL_PERMISSIONS)) + return (0); + + /* + * Update TempDir to the default if it hasn't been set already... + */ + + if (!TempDir) + { +#ifdef __APPLE__ + if ((tmpdir = getenv("TMPDIR")) != NULL && + strncmp(tmpdir, "/private/tmp", 12)) +#else + if ((tmpdir = getenv("TMPDIR")) != NULL) +#endif /* __APPLE__ */ + { + /* + * TMPDIR is defined, see if it is OK for us to use... + */ + + if (stat(tmpdir, &tmpinfo)) + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to access TMPDIR (%s): %s", + tmpdir, strerror(errno)); + else if (!S_ISDIR(tmpinfo.st_mode)) + cupsdLogMessage(CUPSD_LOG_ERROR, "TMPDIR (%s) is not a directory.", + tmpdir); + else if ((tmpinfo.st_uid != User || !(tmpinfo.st_mode & S_IWUSR)) && + (tmpinfo.st_gid != Group || !(tmpinfo.st_mode & S_IWGRP)) && + !(tmpinfo.st_mode & S_IWOTH)) + cupsdLogMessage(CUPSD_LOG_ERROR, + "TMPDIR (%s) has the wrong permissions.", tmpdir); + else + cupsdSetString(&TempDir, tmpdir); + } + + if (!TempDir) + { + cupsdLogMessage(CUPSD_LOG_INFO, "Using default TempDir of %s/tmp...", + RequestRoot); + cupsdSetStringf(&TempDir, "%s/tmp", RequestRoot); + } + } + + /* + * Make sure the temporary directory has the right permissions... + */ + + if (!strncmp(TempDir, RequestRoot, strlen(RequestRoot)) || + access(TempDir, 0)) + { + /* + * Update ownership and permissions if the CUPS temp directory + * is under the spool directory or does not exist... + */ + + if (cupsdCheckPermissions(TempDir, NULL, 01770, RunUser, Group, 1, 1) < 0 && + (FatalErrors & CUPSD_FATAL_PERMISSIONS)) + return (0); + } + + /* + * Update environment variables... + */ + + cupsdUpdateEnv(); + + /* + * Update default paper size setting as needed... + */ + + if (!DefaultPaperSize) + { +#ifdef HAVE_LIBPAPER + char *paper_result; /* Paper size name from libpaper */ + + if ((paper_result = systempapername()) != NULL) + cupsdSetString(&DefaultPaperSize, paper_result); + else +#endif /* HAVE_LIBPAPER */ + if (!DefaultLanguage || + !_cups_strcasecmp(DefaultLanguage, "C") || + !_cups_strcasecmp(DefaultLanguage, "POSIX") || + !_cups_strcasecmp(DefaultLanguage, "en") || + !_cups_strncasecmp(DefaultLanguage, "en.", 3) || + !_cups_strncasecmp(DefaultLanguage, "en_US", 5) || + !_cups_strncasecmp(DefaultLanguage, "en_CA", 5) || + !_cups_strncasecmp(DefaultLanguage, "fr_CA", 5)) + { + /* + * These are the only locales that will default to "letter" size... + */ + + cupsdSetString(&DefaultPaperSize, "Letter"); + } + else + cupsdSetString(&DefaultPaperSize, "A4"); + } + + /* + * Update classification setting as needed... + */ + + if (Classification && !_cups_strcasecmp(Classification, "none")) + cupsdClearString(&Classification); + + if (Classification) + cupsdLogMessage(CUPSD_LOG_INFO, "Security set to \"%s\"", Classification); + + /* + * Check the MaxClients setting, and then allocate memory for it... + */ + + if (MaxClients > (MaxFDs / 3) || MaxClients <= 0) + { + if (MaxClients > 0) + cupsdLogMessage(CUPSD_LOG_INFO, + "MaxClients limited to 1/3 (%d) of the file descriptor " + "limit (%d)...", + MaxFDs / 3, MaxFDs); + + MaxClients = MaxFDs / 3; + } + + cupsdLogMessage(CUPSD_LOG_INFO, "Configured for up to %d clients.", + MaxClients); + + /* + * Check the MaxActiveJobs setting; limit to 1/3 the available + * file descriptors, since we need a pipe for each job... + */ + + if (MaxActiveJobs > (MaxFDs / 3)) + MaxActiveJobs = MaxFDs / 3; + + /* + * Update the MaxClientsPerHost value, as needed... + */ + + if (MaxClientsPerHost <= 0) + MaxClientsPerHost = MaxClients; + + if (MaxClientsPerHost > MaxClients) + MaxClientsPerHost = MaxClients; + + cupsdLogMessage(CUPSD_LOG_INFO, + "Allowing up to %d client connections per host.", + MaxClientsPerHost); + + /* + * Update the default policy, as needed... + */ + + if (DefaultPolicy) + DefaultPolicyPtr = cupsdFindPolicy(DefaultPolicy); + else + DefaultPolicyPtr = NULL; + + if (!DefaultPolicyPtr) + { + cupsd_location_t *po; /* New policy operation */ + + + if (DefaultPolicy) + cupsdLogMessage(CUPSD_LOG_ERROR, "Default policy \"%s\" not found.", + DefaultPolicy); + + cupsdSetString(&DefaultPolicy, "default"); + + if ((DefaultPolicyPtr = cupsdFindPolicy("default")) != NULL) + cupsdLogMessage(CUPSD_LOG_INFO, + "Using policy \"default\" as the default."); + else + { + cupsdLogMessage(CUPSD_LOG_INFO, + "Creating CUPS default administrative policy:"); + + DefaultPolicyPtr = p = cupsdAddPolicy("default"); + + cupsdLogMessage(CUPSD_LOG_INFO, ""); + + cupsdLogMessage(CUPSD_LOG_INFO, "JobPrivateAccess default"); + cupsdAddString(&(p->job_access), "@OWNER"); + cupsdAddString(&(p->job_access), "@SYSTEM"); + + cupsdLogMessage(CUPSD_LOG_INFO, "JobPrivateValues default"); + cupsdAddString(&(p->job_attrs), "job-name"); + cupsdAddString(&(p->job_attrs), "job-originating-host-name"); + cupsdAddString(&(p->job_attrs), "job-originating-user-name"); + + cupsdLogMessage(CUPSD_LOG_INFO, "SubscriptionPrivateAccess default"); + cupsdAddString(&(p->sub_access), "@OWNER"); + cupsdAddString(&(p->sub_access), "@SYSTEM"); + + cupsdLogMessage(CUPSD_LOG_INFO, "SubscriptionPrivateValues default"); + cupsdAddString(&(p->job_attrs), "notify-events"); + cupsdAddString(&(p->job_attrs), "notify-pull-method"); + cupsdAddString(&(p->job_attrs), "notify-recipient-uri"); + cupsdAddString(&(p->job_attrs), "notify-subscriber-user-name"); + cupsdAddString(&(p->job_attrs), "notify-user-data"); + + cupsdLogMessage(CUPSD_LOG_INFO, + ""); + cupsdLogMessage(CUPSD_LOG_INFO, "Order Deny,Allow"); + + po = cupsdAddPolicyOp(p, NULL, IPP_CREATE_JOB); + po->order_type = CUPSD_AUTH_ALLOW; + + cupsdAddPolicyOp(p, po, IPP_PRINT_JOB); + cupsdAddPolicyOp(p, po, IPP_PRINT_URI); + cupsdAddPolicyOp(p, po, IPP_VALIDATE_JOB); + + cupsdLogMessage(CUPSD_LOG_INFO, ""); + + cupsdLogMessage(CUPSD_LOG_INFO, + ""); + cupsdLogMessage(CUPSD_LOG_INFO, "Order Deny,Allow"); + + po = cupsdAddPolicyOp(p, NULL, IPP_SEND_DOCUMENT); + po->order_type = CUPSD_AUTH_ALLOW; + po->level = CUPSD_AUTH_USER; + + cupsdAddName(po, "@OWNER"); + cupsdAddName(po, "@SYSTEM"); + cupsdLogMessage(CUPSD_LOG_INFO, "Require user @OWNER @SYSTEM"); + + cupsdAddPolicyOp(p, po, IPP_SEND_URI); + cupsdAddPolicyOp(p, po, IPP_CANCEL_JOB); + cupsdAddPolicyOp(p, po, IPP_HOLD_JOB); + cupsdAddPolicyOp(p, po, IPP_RELEASE_JOB); + cupsdAddPolicyOp(p, po, IPP_RESTART_JOB); + cupsdAddPolicyOp(p, po, IPP_PURGE_JOBS); + cupsdAddPolicyOp(p, po, IPP_SET_JOB_ATTRIBUTES); + cupsdAddPolicyOp(p, po, IPP_CREATE_JOB_SUBSCRIPTION); + cupsdAddPolicyOp(p, po, IPP_RENEW_SUBSCRIPTION); + cupsdAddPolicyOp(p, po, IPP_CANCEL_SUBSCRIPTION); + cupsdAddPolicyOp(p, po, IPP_GET_NOTIFICATIONS); + cupsdAddPolicyOp(p, po, IPP_REPROCESS_JOB); + cupsdAddPolicyOp(p, po, IPP_CANCEL_CURRENT_JOB); + cupsdAddPolicyOp(p, po, IPP_SUSPEND_CURRENT_JOB); + cupsdAddPolicyOp(p, po, IPP_RESUME_JOB); + cupsdAddPolicyOp(p, po, IPP_CANCEL_MY_JOBS); + cupsdAddPolicyOp(p, po, IPP_CLOSE_JOB); + cupsdAddPolicyOp(p, po, CUPS_MOVE_JOB); + cupsdAddPolicyOp(p, po, CUPS_AUTHENTICATE_JOB); + cupsdAddPolicyOp(p, po, CUPS_GET_DOCUMENT); + + cupsdLogMessage(CUPSD_LOG_INFO, ""); + + cupsdLogMessage(CUPSD_LOG_INFO, + ""); + cupsdLogMessage(CUPSD_LOG_INFO, "Order Deny,Allow"); + cupsdLogMessage(CUPSD_LOG_INFO, "AuthType Default"); + + po = cupsdAddPolicyOp(p, NULL, IPP_PAUSE_PRINTER); + po->order_type = CUPSD_AUTH_ALLOW; + po->type = CUPSD_AUTH_DEFAULT; + po->level = CUPSD_AUTH_USER; + + cupsdAddName(po, "@SYSTEM"); + cupsdLogMessage(CUPSD_LOG_INFO, "Require user @SYSTEM"); + + cupsdAddPolicyOp(p, po, IPP_RESUME_PRINTER); + cupsdAddPolicyOp(p, po, IPP_SET_PRINTER_ATTRIBUTES); + cupsdAddPolicyOp(p, po, IPP_ENABLE_PRINTER); + cupsdAddPolicyOp(p, po, IPP_DISABLE_PRINTER); + cupsdAddPolicyOp(p, po, IPP_PAUSE_PRINTER_AFTER_CURRENT_JOB); + cupsdAddPolicyOp(p, po, IPP_HOLD_NEW_JOBS); + cupsdAddPolicyOp(p, po, IPP_RELEASE_HELD_NEW_JOBS); + cupsdAddPolicyOp(p, po, IPP_DEACTIVATE_PRINTER); + cupsdAddPolicyOp(p, po, IPP_ACTIVATE_PRINTER); + cupsdAddPolicyOp(p, po, IPP_RESTART_PRINTER); + cupsdAddPolicyOp(p, po, IPP_SHUTDOWN_PRINTER); + cupsdAddPolicyOp(p, po, IPP_STARTUP_PRINTER); + cupsdAddPolicyOp(p, po, IPP_PROMOTE_JOB); + cupsdAddPolicyOp(p, po, IPP_SCHEDULE_JOB_AFTER); + cupsdAddPolicyOp(p, po, IPP_CANCEL_JOBS); + cupsdAddPolicyOp(p, po, CUPS_ADD_PRINTER); + cupsdAddPolicyOp(p, po, CUPS_DELETE_PRINTER); + cupsdAddPolicyOp(p, po, CUPS_ADD_CLASS); + cupsdAddPolicyOp(p, po, CUPS_DELETE_CLASS); + cupsdAddPolicyOp(p, po, CUPS_ACCEPT_JOBS); + cupsdAddPolicyOp(p, po, CUPS_REJECT_JOBS); + cupsdAddPolicyOp(p, po, CUPS_SET_DEFAULT); + + cupsdLogMessage(CUPSD_LOG_INFO, ""); + + cupsdLogMessage(CUPSD_LOG_INFO, ""); + cupsdLogMessage(CUPSD_LOG_INFO, "Order Deny,Allow"); + + po = cupsdAddPolicyOp(p, NULL, IPP_ANY_OPERATION); + po->order_type = CUPSD_AUTH_ALLOW; + + cupsdLogMessage(CUPSD_LOG_INFO, ""); + cupsdLogMessage(CUPSD_LOG_INFO, ""); + } + } + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdReadConfiguration: NumPolicies=%d", + cupsArrayCount(Policies)); + for (i = 0, p = (cupsd_policy_t *)cupsArrayFirst(Policies); + p; + i ++, p = (cupsd_policy_t *)cupsArrayNext(Policies)) + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdReadConfiguration: Policies[%d]=\"%s\"", i, p->name); + + /* + * If we are doing a full reload or the server root has changed, flush + * the jobs, printers, etc. and start from scratch... + */ + + if (NeedReload == RELOAD_ALL || + old_remote_port != RemotePort || + !old_serverroot || !ServerRoot || strcmp(old_serverroot, ServerRoot) || + !old_requestroot || !RequestRoot || strcmp(old_requestroot, RequestRoot)) + { + mime_type_t *type; /* Current type */ + char mimetype[MIME_MAX_SUPER + MIME_MAX_TYPE]; + /* MIME type name */ + + + cupsdLogMessage(CUPSD_LOG_INFO, "Full reload is required."); + + /* + * Free all memory... + */ + + cupsdDeleteAllSubscriptions(); + cupsdFreeAllJobs(); + cupsdDeleteAllPrinters(); + + DefaultPrinter = NULL; + + if (MimeDatabase != NULL) + mimeDelete(MimeDatabase); + + if (NumMimeTypes) + { + for (i = 0; i < NumMimeTypes; i ++) + _cupsStrFree(MimeTypes[i]); + + free(MimeTypes); + } + + /* + * Read the MIME type and conversion database... + */ + + snprintf(temp, sizeof(temp), "%s/filter", ServerBin); + snprintf(mimedir, sizeof(mimedir), "%s/mime", DataDir); + + MimeDatabase = mimeNew(); + mimeSetErrorCallback(MimeDatabase, mime_error_cb, NULL); + + MimeDatabase = mimeLoadTypes(MimeDatabase, mimedir); + MimeDatabase = mimeLoadTypes(MimeDatabase, ServerRoot); + MimeDatabase = mimeLoadFilters(MimeDatabase, mimedir, temp); + MimeDatabase = mimeLoadFilters(MimeDatabase, ServerRoot, temp); + + if (!MimeDatabase) + { + cupsdLogMessage(CUPSD_LOG_EMERG, + "Unable to load MIME database from \"%s\" or \"%s\".", + mimedir, ServerRoot); + if (FatalErrors & CUPSD_FATAL_CONFIG) + return (0); + } + + cupsdLogMessage(CUPSD_LOG_INFO, + "Loaded MIME database from \"%s\" and \"%s\": %d types, " + "%d filters...", mimedir, ServerRoot, + mimeNumTypes(MimeDatabase), mimeNumFilters(MimeDatabase)); + + /* + * Create a list of MIME types for the document-format-supported + * attribute... + */ + + NumMimeTypes = mimeNumTypes(MimeDatabase); + if (!mimeType(MimeDatabase, "application", "octet-stream")) + NumMimeTypes ++; + + if ((MimeTypes = calloc(NumMimeTypes, sizeof(const char *))) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate memory for %d MIME types.", + NumMimeTypes); + NumMimeTypes = 0; + } + else + { + for (i = 0, type = mimeFirstType(MimeDatabase); + type; + i ++, type = mimeNextType(MimeDatabase)) + { + snprintf(mimetype, sizeof(mimetype), "%s/%s", type->super, type->type); + + MimeTypes[i] = _cupsStrAlloc(mimetype); + } + + if (i < NumMimeTypes) + MimeTypes[i] = _cupsStrAlloc("application/octet-stream"); + } + + if (LogLevel == CUPSD_LOG_DEBUG2) + { + mime_filter_t *filter; /* Current filter */ + + + for (type = mimeFirstType(MimeDatabase); + type; + type = mimeNextType(MimeDatabase)) + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdReadConfiguration: type %s/%s", + type->super, type->type); + + for (filter = mimeFirstFilter(MimeDatabase); + filter; + filter = mimeNextFilter(MimeDatabase)) + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdReadConfiguration: filter %s/%s to %s/%s %d %s", + filter->src->super, filter->src->type, + filter->dst->super, filter->dst->type, + filter->cost, filter->filter); + } + + /* + * Load banners... + */ + + snprintf(temp, sizeof(temp), "%s/banners", DataDir); + cupsdLoadBanners(temp); + + /* + * Load printers and classes... + */ + + cupsdLoadAllPrinters(); + cupsdLoadAllClasses(); + + cupsdCreateCommonData(); + + /* + * Update the printcap file as needed... + */ + + if (Printcap && *Printcap && access(Printcap, 0)) + cupsdWritePrintcap(); + + /* + * Load queued jobs... + */ + + cupsdLoadAllJobs(); + + /* + * Load subscriptions... + */ + + cupsdLoadAllSubscriptions(); + + cupsdLogMessage(CUPSD_LOG_INFO, "Full reload complete."); + } + else + { + /* + * Not a full reload, so recreate the common printer attributes... + */ + + cupsdCreateCommonData(); + + /* + * Update all printers as needed... + */ + + cupsdUpdatePrinters(); + cupsdMarkDirty(CUPSD_DIRTY_PRINTCAP); + + cupsdLogMessage(CUPSD_LOG_INFO, "Partial reload complete."); + } + + /* + * Reset the reload state... + */ + + NeedReload = RELOAD_NONE; + + cupsdClearString(&old_serverroot); + cupsdClearString(&old_requestroot); + + return (1); +} + + +/* + * 'get_address()' - Get an address + port number from a line. + */ + +static http_addrlist_t * /* O - Pointer to list if address good, NULL if bad */ +get_address(const char *value, /* I - Value string */ + int defport) /* I - Default port */ +{ + char buffer[1024], /* Hostname + port number buffer */ + defpname[255], /* Default port name */ + *hostname, /* Hostname or IP */ + *portname; /* Port number or name */ + http_addrlist_t *addrlist; /* Address list */ + + + /* + * Check for an empty value... + */ + + if (!*value) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Bad (empty) address."); + return (NULL); + } + + /* + * Grab a hostname and port number; if there is no colon and the port name + * is only digits, then we have a port number by itself... + */ + + strlcpy(buffer, value, sizeof(buffer)); + + if ((portname = strrchr(buffer, ':')) != NULL && !strchr(portname, ']')) + { + *portname++ = '\0'; + hostname = buffer; + } + else + { + for (portname = buffer; isdigit(*portname & 255); portname ++); + + if (*portname) + { + /* + * Use the default port... + */ + + sprintf(defpname, "%d", defport); + portname = defpname; + hostname = buffer; + } + else + { + /* + * The buffer contains just a port number... + */ + + portname = buffer; + hostname = NULL; + } + } + + if (hostname && !strcmp(hostname, "*")) + hostname = NULL; + + /* + * Now lookup the address using httpAddrGetList()... + */ + + if ((addrlist = httpAddrGetList(hostname, AF_UNSPEC, portname)) == NULL) + cupsdLogMessage(CUPSD_LOG_ERROR, "Hostname lookup for \"%s\" failed.", + hostname ? hostname : "(nil)"); + + return (addrlist); +} + + +/* + * 'get_addr_and_mask()' - Get an IP address and netmask. + */ + +static int /* O - 1 on success, 0 on failure */ +get_addr_and_mask(const char *value, /* I - String from config file */ + unsigned *ip, /* O - Address value */ + unsigned *mask) /* O - Mask value */ +{ + int i, j, /* Looping vars */ + family, /* Address family */ + ipcount; /* Count of fields in address */ + unsigned ipval; /* Value */ + const char *maskval, /* Pointer to start of mask value */ + *ptr, /* Pointer into value */ + *ptr2; /* ... */ + + + /* + * Get the address... + */ + + ip[0] = ip[1] = ip[2] = ip[3] = 0x00000000; + mask[0] = mask[1] = mask[2] = mask[3] = 0xffffffff; + + if ((maskval = strchr(value, '/')) != NULL) + maskval ++; + else + maskval = value + strlen(value); + +#ifdef AF_INET6 + /* + * Check for an IPv6 address... + */ + + if (*value == '[') + { + /* + * Parse hexadecimal IPv6/IPv4 address... + */ + + family = AF_INET6; + + for (i = 0, ptr = value + 1; *ptr && i < 8; i ++) + { + if (*ptr == ']') + break; + else if (!strncmp(ptr, "::", 2)) + { + for (ptr2 = strchr(ptr + 2, ':'), j = 0; + ptr2; + ptr2 = strchr(ptr2 + 1, ':'), j ++); + + i = 6 - j; + ptr += 2; + } + else if (isdigit(*ptr & 255) && strchr(ptr + 1, '.') && i >= 6) + { + /* + * Read IPv4 dotted quad... + */ + + unsigned val[4] = { 0, 0, 0, 0 }; + /* IPv4 address values */ + + ipcount = sscanf(ptr, "%u.%u.%u.%u", val + 0, val + 1, val + 2, + val + 3); + + /* + * Range check the IP numbers... + */ + + for (i = 0; i < ipcount; i ++) + if (val[i] > 255) + return (0); + + /* + * Merge everything into a 32-bit IPv4 address in ip[3]... + */ + + ip[3] = (((((val[0] << 8) | val[1]) << 8) | val[2]) << 8) | val[3]; + + if (ipcount < 4) + mask[3] = (0xffffffff << (32 - 8 * ipcount)) & 0xffffffff; + + /* + * If the leading words are all 0's then this is an IPv4 address... + */ + + if (!val[0] && !val[1] && !val[2]) + family = AF_INET; + + while (isdigit(*ptr & 255) || *ptr == '.') + ptr ++; + break; + } + else if (isxdigit(*ptr & 255)) + { + ipval = strtoul(ptr, (char **)&ptr, 16); + + if (*ptr == ':' && ptr[1] != ':') + ptr ++; + + if (ipval > 0xffff) + return (0); + + if (i & 1) + ip[i / 2] |= ipval; + else + ip[i / 2] |= ipval << 16; + } + else + return (0); + } + + if (*ptr != ']') + return (0); + + ptr ++; + + if (*ptr && *ptr != '/') + return (0); + } + else +#endif /* AF_INET6 */ + { + /* + * Parse dotted-decimal IPv4 address... + */ + + unsigned val[4] = { 0, 0, 0, 0 }; /* IPv4 address values */ + + + family = AF_INET; + ipcount = sscanf(value, "%u.%u.%u.%u", val + 0, val + 1, val + 2, val + 3); + + /* + * Range check the IP numbers... + */ + + for (i = 0; i < ipcount; i ++) + if (val[i] > 255) + return (0); + + /* + * Merge everything into a 32-bit IPv4 address in ip[3]... + */ + + ip[3] = (((((val[0] << 8) | val[1]) << 8) | val[2]) << 8) | val[3]; + + if (ipcount < 4) + mask[3] = (0xffffffff << (32 - 8 * ipcount)) & 0xffffffff; + } + + if (*maskval) + { + /* + * Get the netmask value(s)... + */ + + memset(mask, 0, sizeof(unsigned) * 4); + + if (strchr(maskval, '.')) + { + /* + * Get dotted-decimal mask... + */ + + if (family != AF_INET) + return (0); + + if (sscanf(maskval, "%u.%u.%u.%u", mask + 0, mask + 1, mask + 2, + mask + 3) != 4) + return (0); + + mask[3] |= ((((mask[0] << 8) | mask[1]) << 8) | mask[2]) << 8; + mask[0] = mask[1] = mask[2] = 0; + } + else + { + /* + * Get address/bits format... + */ + + i = atoi(maskval); + +#ifdef AF_INET6 + if (family == AF_INET6) + { + if (i > 128) + return (0); + + i = 128 - i; + + if (i <= 96) + mask[0] = 0xffffffff; + else + mask[0] = (0xffffffff << (i - 96)) & 0xffffffff; + + if (i <= 64) + mask[1] = 0xffffffff; + else if (i >= 96) + mask[1] = 0; + else + mask[1] = (0xffffffff << (i - 64)) & 0xffffffff; + + if (i <= 32) + mask[2] = 0xffffffff; + else if (i >= 64) + mask[2] = 0; + else + mask[2] = (0xffffffff << (i - 32)) & 0xffffffff; + + if (i == 0) + mask[3] = 0xffffffff; + else if (i >= 32) + mask[3] = 0; + else + mask[3] = (0xffffffff << i) & 0xffffffff; + } + else +#endif /* AF_INET6 */ + { + if (i > 32) + return (0); + + mask[0] = 0xffffffff; + mask[1] = 0xffffffff; + mask[2] = 0xffffffff; + + if (i < 32) + mask[3] = (0xffffffff << (32 - i)) & 0xffffffff; + else + mask[3] = 0xffffffff; + } + } + } + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "get_addr_and_mask(value=\"%s\", " + "ip=[%08x:%08x:%08x:%08x], mask=[%08x:%08x:%08x:%08x])", + value, ip[0], ip[1], ip[2], ip[3], mask[0], mask[1], mask[2], + mask[3]); + + /* + * Check for a valid netmask; no fallback like in CUPS 1.1.x! + */ + + if ((ip[0] & ~mask[0]) != 0 || + (ip[1] & ~mask[1]) != 0 || + (ip[2] & ~mask[2]) != 0 || + (ip[3] & ~mask[3]) != 0) + return (0); + + return (1); +} + + +/* + * 'mime_error_cb()' - Log a MIME error. + */ + +static void +mime_error_cb(void *ctx, /* I - Context pointer (unused) */ + const char *message) /* I - Message */ +{ + (void)ctx; + + cupsdLogMessage(CUPSD_LOG_ERROR, "%s", message); +} + + +/* + * 'parse_aaa()' - Parse authentication, authorization, and access control lines. + */ + +static int /* O - 1 on success, 0 on failure */ +parse_aaa(cupsd_location_t *loc, /* I - Location */ + char *line, /* I - Line from file */ + char *value, /* I - Start of value data */ + int linenum) /* I - Current line number */ +{ + char *valptr; /* Pointer into value */ + unsigned ip[4], /* IP address components */ + mask[4]; /* IP netmask components */ + + + if (!_cups_strcasecmp(line, "Encryption")) + { + /* + * "Encryption xxx" - set required encryption level... + */ + + if (!_cups_strcasecmp(value, "never")) + loc->encryption = HTTP_ENCRYPT_NEVER; + else if (!_cups_strcasecmp(value, "always")) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Encryption value \"%s\" on line %d is invalid in this " + "context. Using \"required\" instead.", value, linenum); + + loc->encryption = HTTP_ENCRYPT_REQUIRED; + } + else if (!_cups_strcasecmp(value, "required")) + loc->encryption = HTTP_ENCRYPT_REQUIRED; + else if (!_cups_strcasecmp(value, "ifrequested")) + loc->encryption = HTTP_ENCRYPT_IF_REQUESTED; + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unknown Encryption value %s on line %d.", value, linenum); + return (0); + } + } + else if (!_cups_strcasecmp(line, "Order")) + { + /* + * "Order Deny,Allow" or "Order Allow,Deny"... + */ + + if (!_cups_strncasecmp(value, "deny", 4)) + loc->order_type = CUPSD_AUTH_ALLOW; + else if (!_cups_strncasecmp(value, "allow", 5)) + loc->order_type = CUPSD_AUTH_DENY; + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unknown Order value %s on line %d.", + value, linenum); + return (0); + } + } + else if (!_cups_strcasecmp(line, "Allow") || !_cups_strcasecmp(line, "Deny")) + { + /* + * Allow [From] host/ip... + * Deny [From] host/ip... + */ + + while (*value) + { + if (!_cups_strncasecmp(value, "from", 4)) + { + /* + * Strip leading "from"... + */ + + value += 4; + + while (_cups_isspace(*value)) + value ++; + + if (!*value) + break; + } + + /* + * Find the end of the value... + */ + + for (valptr = value; *valptr && !_cups_isspace(*valptr); valptr ++); + + while (_cups_isspace(*valptr)) + *valptr++ = '\0'; + + /* + * Figure out what form the allow/deny address takes: + * + * All + * None + * *.domain.com + * .domain.com + * host.domain.com + * nnn.* + * nnn.nnn.* + * nnn.nnn.nnn.* + * nnn.nnn.nnn.nnn + * nnn.nnn.nnn.nnn/mm + * nnn.nnn.nnn.nnn/mmm.mmm.mmm.mmm + */ + + if (!_cups_strcasecmp(value, "all")) + { + /* + * All hosts... + */ + + if (!_cups_strcasecmp(line, "Allow")) + cupsdAddIPMask(&(loc->allow), zeros, zeros); + else + cupsdAddIPMask(&(loc->deny), zeros, zeros); + } + else if (!_cups_strcasecmp(value, "none")) + { + /* + * No hosts... + */ + + if (!_cups_strcasecmp(line, "Allow")) + cupsdAddIPMask(&(loc->allow), ones, zeros); + else + cupsdAddIPMask(&(loc->deny), ones, zeros); + } +#ifdef AF_INET6 + else if (value[0] == '*' || value[0] == '.' || + (!isdigit(value[0] & 255) && value[0] != '[')) +#else + else if (value[0] == '*' || value[0] == '.' || !isdigit(value[0] & 255)) +#endif /* AF_INET6 */ + { + /* + * Host or domain name... + */ + + if (value[0] == '*') + value ++; + + if (!_cups_strcasecmp(line, "Allow")) + cupsdAddNameMask(&(loc->allow), value); + else + cupsdAddNameMask(&(loc->deny), value); + } + else + { + /* + * One of many IP address forms... + */ + + if (!get_addr_and_mask(value, ip, mask)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Bad netmask value %s on line %d.", + value, linenum); + return (0); + } + + if (!_cups_strcasecmp(line, "Allow")) + cupsdAddIPMask(&(loc->allow), ip, mask); + else + cupsdAddIPMask(&(loc->deny), ip, mask); + } + + /* + * Advance to next value... + */ + + value = valptr; + } + } + else if (!_cups_strcasecmp(line, "AuthType")) + { + /* + * AuthType {none,basic,digest,basicdigest,negotiate,default} + */ + + if (!_cups_strcasecmp(value, "none")) + { + loc->type = CUPSD_AUTH_NONE; + loc->level = CUPSD_AUTH_ANON; + } + else if (!_cups_strcasecmp(value, "basic")) + { + loc->type = CUPSD_AUTH_BASIC; + + if (loc->level == CUPSD_AUTH_ANON) + loc->level = CUPSD_AUTH_USER; + } + else if (!_cups_strcasecmp(value, "digest")) + { + loc->type = CUPSD_AUTH_DIGEST; + + if (loc->level == CUPSD_AUTH_ANON) + loc->level = CUPSD_AUTH_USER; + } + else if (!_cups_strcasecmp(value, "basicdigest")) + { + loc->type = CUPSD_AUTH_BASICDIGEST; + + if (loc->level == CUPSD_AUTH_ANON) + loc->level = CUPSD_AUTH_USER; + } + else if (!_cups_strcasecmp(value, "default")) + { + loc->type = CUPSD_AUTH_DEFAULT; + + if (loc->level == CUPSD_AUTH_ANON) + loc->level = CUPSD_AUTH_USER; + } +#ifdef HAVE_GSSAPI + else if (!_cups_strcasecmp(value, "negotiate")) + { + loc->type = CUPSD_AUTH_NEGOTIATE; + + if (loc->level == CUPSD_AUTH_ANON) + loc->level = CUPSD_AUTH_USER; + } +#endif /* HAVE_GSSAPI */ + else + { + cupsdLogMessage(CUPSD_LOG_WARN, + "Unknown authorization type %s on line %d.", + value, linenum); + return (0); + } + } + else if (!_cups_strcasecmp(line, "AuthClass")) + { + /* + * AuthClass anonymous, user, system, group + */ + + if (!_cups_strcasecmp(value, "anonymous")) + { + loc->type = CUPSD_AUTH_NONE; + loc->level = CUPSD_AUTH_ANON; + + cupsdLogMessage(CUPSD_LOG_WARN, + "\"AuthClass %s\" is deprecated; consider removing " + "it from line %d.", + value, linenum); + } + else if (!_cups_strcasecmp(value, "user")) + { + loc->level = CUPSD_AUTH_USER; + + cupsdLogMessage(CUPSD_LOG_WARN, + "\"AuthClass %s\" is deprecated; consider using " + "\"Require valid-user\" on line %d.", + value, linenum); + } + else if (!_cups_strcasecmp(value, "group")) + { + loc->level = CUPSD_AUTH_GROUP; + + cupsdLogMessage(CUPSD_LOG_WARN, + "\"AuthClass %s\" is deprecated; consider using " + "\"Require user @groupname\" on line %d.", + value, linenum); + } + else if (!_cups_strcasecmp(value, "system")) + { + loc->level = CUPSD_AUTH_GROUP; + + cupsdAddName(loc, "@SYSTEM"); + + cupsdLogMessage(CUPSD_LOG_WARN, + "\"AuthClass %s\" is deprecated; consider using " + "\"Require user @SYSTEM\" on line %d.", + value, linenum); + } + else + { + cupsdLogMessage(CUPSD_LOG_WARN, + "Unknown authorization class %s on line %d.", + value, linenum); + return (0); + } + } + else if (!_cups_strcasecmp(line, "AuthGroupName")) + { + cupsdAddName(loc, value); + + cupsdLogMessage(CUPSD_LOG_WARN, + "\"AuthGroupName %s\" directive is deprecated; consider " + "using \"Require user @%s\" on line %d.", + value, value, linenum); + } + else if (!_cups_strcasecmp(line, "Require")) + { + /* + * Apache synonym for AuthClass and AuthGroupName... + * + * Get initial word: + * + * Require valid-user + * Require group names + * Require user names + */ + + for (valptr = value; !_cups_isspace(*valptr) && *valptr; valptr ++); + + if (*valptr) + *valptr++ = '\0'; + + if (!_cups_strcasecmp(value, "valid-user") || + !_cups_strcasecmp(value, "user")) + loc->level = CUPSD_AUTH_USER; + else if (!_cups_strcasecmp(value, "group")) + loc->level = CUPSD_AUTH_GROUP; + else + { + cupsdLogMessage(CUPSD_LOG_WARN, "Unknown Require type %s on line %d.", + value, linenum); + return (0); + } + + /* + * Get the list of names from the line... + */ + + for (value = valptr; *value;) + { + while (_cups_isspace(*value)) + value ++; + +#ifdef HAVE_AUTHORIZATION_H + if (!strncmp(value, "@AUTHKEY(", 9)) + { + /* + * Grab "@AUTHKEY(name)" value... + */ + + for (valptr = value + 9; *valptr != ')' && *valptr; valptr ++); + + if (*valptr) + *valptr++ = '\0'; + } + else +#endif /* HAVE_AUTHORIZATION_H */ + if (*value == '\"' || *value == '\'') + { + /* + * Grab quoted name... + */ + + for (valptr = value + 1; *valptr != *value && *valptr; valptr ++); + + value ++; + } + else + { + /* + * Grab literal name. + */ + + for (valptr = value; !_cups_isspace(*valptr) && *valptr; valptr ++); + } + + if (*valptr) + *valptr++ = '\0'; + + cupsdAddName(loc, value); + + for (value = valptr; _cups_isspace(*value); value ++); + } + } + else if (!_cups_strcasecmp(line, "Satisfy")) + { + if (!_cups_strcasecmp(value, "all")) + loc->satisfy = CUPSD_AUTH_SATISFY_ALL; + else if (!_cups_strcasecmp(value, "any")) + loc->satisfy = CUPSD_AUTH_SATISFY_ANY; + else + { + cupsdLogMessage(CUPSD_LOG_WARN, "Unknown Satisfy value %s on line %d.", + value, linenum); + return (0); + } + } + else + return (0); + + return (1); +} + + +/* + * 'parse_fatal_errors()' - Parse FatalErrors values in a string. + */ + +static int /* O - FatalErrors bits */ +parse_fatal_errors(const char *s) /* I - FatalErrors string */ +{ + int fatal; /* FatalErrors bits */ + char value[1024], /* Value string */ + *valstart, /* Pointer into value */ + *valend; /* End of value */ + + + /* + * Empty FatalErrors line yields NULL pointer... + */ + + if (!s) + return (CUPSD_FATAL_NONE); + + /* + * Loop through the value string,... + */ + + strlcpy(value, s, sizeof(value)); + + fatal = CUPSD_FATAL_NONE; + + for (valstart = value; *valstart;) + { + /* + * Get the current space/comma-delimited kind name... + */ + + for (valend = valstart; *valend; valend ++) + if (_cups_isspace(*valend) || *valend == ',') + break; + + if (*valend) + *valend++ = '\0'; + + /* + * Add the error to the bitmask... + */ + + if (!_cups_strcasecmp(valstart, "all")) + fatal = CUPSD_FATAL_ALL; + else if (!_cups_strcasecmp(valstart, "browse")) + fatal |= CUPSD_FATAL_BROWSE; + else if (!_cups_strcasecmp(valstart, "-browse")) + fatal &= ~CUPSD_FATAL_BROWSE; + else if (!_cups_strcasecmp(valstart, "config")) + fatal |= CUPSD_FATAL_CONFIG; + else if (!_cups_strcasecmp(valstart, "-config")) + fatal &= ~CUPSD_FATAL_CONFIG; + else if (!_cups_strcasecmp(valstart, "listen")) + fatal |= CUPSD_FATAL_LISTEN; + else if (!_cups_strcasecmp(valstart, "-listen")) + fatal &= ~CUPSD_FATAL_LISTEN; + else if (!_cups_strcasecmp(valstart, "log")) + fatal |= CUPSD_FATAL_LOG; + else if (!_cups_strcasecmp(valstart, "-log")) + fatal &= ~CUPSD_FATAL_LOG; + else if (!_cups_strcasecmp(valstart, "permissions")) + fatal |= CUPSD_FATAL_PERMISSIONS; + else if (!_cups_strcasecmp(valstart, "-permissions")) + fatal &= ~CUPSD_FATAL_PERMISSIONS; + else if (_cups_strcasecmp(valstart, "none")) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unknown FatalErrors kind \"%s\" ignored.", valstart); + + for (valstart = valend; *valstart; valstart ++) + if (!_cups_isspace(*valstart) || *valstart != ',') + break; + } + + return (fatal); +} + + +/* + * 'parse_groups()' - Parse system group names in a string. + */ + +static int /* O - 1 on success, 0 on failure */ +parse_groups(const char *s) /* I - Space-delimited groups */ +{ + int status; /* Return status */ + char value[1024], /* Value string */ + *valstart, /* Pointer into value */ + *valend, /* End of value */ + quote; /* Quote character */ + struct group *group; /* Group */ + + + /* + * Make a copy of the string and parse out the groups... + */ + + strlcpy(value, s, sizeof(value)); + + status = 1; + valstart = value; + + while (*valstart && NumSystemGroups < MAX_SYSTEM_GROUPS) + { + if (*valstart == '\'' || *valstart == '\"') + { + /* + * Scan quoted name... + */ + + quote = *valstart++; + + for (valend = valstart; *valend; valend ++) + if (*valend == quote) + break; + } + else + { + /* + * Scan space or comma-delimited name... + */ + + for (valend = valstart; *valend; valend ++) + if (_cups_isspace(*valend) || *valend == ',') + break; + } + + if (*valend) + *valend++ = '\0'; + + group = getgrnam(valstart); + if (group) + { + cupsdSetString(SystemGroups + NumSystemGroups, valstart); + SystemGroupIDs[NumSystemGroups] = group->gr_gid; + + NumSystemGroups ++; + } + else + status = 0; + + endgrent(); + + valstart = valend; + + while (*valstart == ',' || _cups_isspace(*valstart)) + valstart ++; + } + + return (status); +} + + +/* + * 'parse_protocols()' - Parse browse protocols in a string. + */ + +static int /* O - Browse protocol bits */ +parse_protocols(const char *s) /* I - Space-delimited protocols */ +{ + int protocols; /* Browse protocol bits */ + char value[1024], /* Value string */ + *valstart, /* Pointer into value */ + *valend; /* End of value */ + + + /* + * Empty protocol line yields NULL pointer... + */ + + if (!s) + return (0); + + /* + * Loop through the value string,... + */ + + strlcpy(value, s, sizeof(value)); + + protocols = 0; + + for (valstart = value; *valstart;) + { + /* + * Get the current space/comma-delimited protocol name... + */ + + for (valend = valstart; *valend; valend ++) + if (_cups_isspace(*valend) || *valend == ',') + break; + + if (*valend) + *valend++ = '\0'; + + /* + * Add the protocol to the bitmask... + */ + + if (!_cups_strcasecmp(valstart, "dnssd") || + !_cups_strcasecmp(valstart, "dns-sd") || + !_cups_strcasecmp(valstart, "bonjour")) + protocols |= BROWSE_DNSSD; + else if (!_cups_strcasecmp(valstart, "all")) + protocols |= BROWSE_ALL; + else if (_cups_strcasecmp(valstart, "none")) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unknown browse protocol \"%s\" ignored.", valstart); + + for (valstart = valend; *valstart; valstart ++) + if (!_cups_isspace(*valstart) || *valstart != ',') + break; + } + + return (protocols); +} + + +/* + * 'read_configuration()' - Read a configuration file. + */ + +static int /* O - 1 on success, 0 on failure */ +read_configuration(cups_file_t *fp) /* I - File to read from */ +{ + int i; /* Looping var */ + int linenum; /* Current line number */ + char line[HTTP_MAX_BUFFER], + /* Line from file */ + temp[HTTP_MAX_BUFFER], + /* Temporary buffer for value */ + *value, /* Pointer to value */ + *valueptr; /* Pointer into value */ + int valuelen; /* Length of value */ + cupsd_var_t const *var; /* Current variable */ + http_addrlist_t *addrlist, /* Address list */ + *addr; /* Current address */ + cups_file_t *incfile; /* Include file */ + char incname[1024]; /* Include filename */ + struct group *group; /* Group */ + + + /* + * Loop through each line in the file... + */ + + linenum = 0; + + while (cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + { + /* + * Decode the directive... + */ + + if (!_cups_strcasecmp(line, "Include") && value) + { + /* + * Include filename + */ + + if (value[0] == '/') + strlcpy(incname, value, sizeof(incname)); + else + snprintf(incname, sizeof(incname), "%s/%s", ServerRoot, value); + + if ((incfile = cupsFileOpen(incname, "rb")) == NULL) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to include config file \"%s\" - %s", + incname, strerror(errno)); + else + { + read_configuration(incfile); + cupsFileClose(incfile); + } + } + else if (!_cups_strcasecmp(line, " + */ + + linenum = read_location(fp, value, linenum); + if (linenum == 0) + return (0); + } + else if (!_cups_strcasecmp(line, " + */ + + linenum = read_policy(fp, value, linenum); + if (linenum == 0) + return (0); + } + else if (!_cups_strcasecmp(line, "FatalErrors")) + FatalErrors = parse_fatal_errors(value); + else if (!_cups_strcasecmp(line, "FaxRetryInterval") && value) + { + JobRetryInterval = atoi(value); + cupsdLogMessage(CUPSD_LOG_WARN, + "FaxRetryInterval is deprecated; use " + "JobRetryInterval on line %d.", linenum); + } + else if (!_cups_strcasecmp(line, "FaxRetryLimit") && value) + { + JobRetryLimit = atoi(value); + cupsdLogMessage(CUPSD_LOG_WARN, + "FaxRetryLimit is deprecated; use " + "JobRetryLimit on line %d.", linenum); + } + else if ((!_cups_strcasecmp(line, "Port") || !_cups_strcasecmp(line, "Listen") +#ifdef HAVE_SSL + || !_cups_strcasecmp(line, "SSLPort") || !_cups_strcasecmp(line, "SSLListen") +#endif /* HAVE_SSL */ + ) && value) + { + /* + * Add listening address(es) to the list... + */ + + cupsd_listener_t *lis; /* New listeners array */ + + + /* + * Get the address list... + */ + + addrlist = get_address(value, IPP_PORT); + + if (!addrlist) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Bad %s address %s at line %d.", line, + value, linenum); + continue; + } + + /* + * Add each address... + */ + + for (addr = addrlist; addr; addr = addr->next) + { + /* + * See if this address is already present... + */ + + for (lis = (cupsd_listener_t *)cupsArrayFirst(Listeners); + lis; + lis = (cupsd_listener_t *)cupsArrayNext(Listeners)) + if (httpAddrEqual(&(addr->addr), &(lis->address)) && + _httpAddrPort(&(addr->addr)) == _httpAddrPort(&(lis->address))) + break; + + if (lis) + { + httpAddrString(&lis->address, temp, sizeof(temp)); + cupsdLogMessage(CUPSD_LOG_WARN, + "Duplicate listen address \"%s\" ignored.", temp); + continue; + } + + /* + * Allocate another listener... + */ + + if (!Listeners) + Listeners = cupsArrayNew(NULL, NULL); + + if (!Listeners) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate %s at line %d - %s.", + line, linenum, strerror(errno)); + break; + } + + if ((lis = calloc(1, sizeof(cupsd_listener_t))) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate %s at line %d - %s.", + line, linenum, strerror(errno)); + break; + } + + cupsArrayAdd(Listeners, lis); + + /* + * Copy the current address and log it... + */ + + memcpy(&(lis->address), &(addr->addr), sizeof(lis->address)); + lis->fd = -1; + +#ifdef HAVE_SSL + if (!_cups_strcasecmp(line, "SSLPort") || !_cups_strcasecmp(line, "SSLListen")) + lis->encryption = HTTP_ENCRYPT_ALWAYS; +#endif /* HAVE_SSL */ + + httpAddrString(&lis->address, temp, sizeof(temp)); + +#ifdef AF_LOCAL + if (lis->address.addr.sa_family == AF_LOCAL) + cupsdLogMessage(CUPSD_LOG_INFO, "Listening to %s (Domain)", temp); + else +#endif /* AF_LOCAL */ + cupsdLogMessage(CUPSD_LOG_INFO, "Listening to %s:%d (IPv%d)", temp, + _httpAddrPort(&(lis->address)), + _httpAddrFamily(&(lis->address)) == AF_INET ? 4 : 6); + + if (!httpAddrLocalhost(&(lis->address))) + RemotePort = _httpAddrPort(&(lis->address)); + } + + /* + * Free the list... + */ + + httpAddrFreeList(addrlist); + } + else if (!_cups_strcasecmp(line, "BrowseProtocols") || + !_cups_strcasecmp(line, "BrowseLocalProtocols")) + { + /* + * "BrowseProtocols name [... name]" + * "BrowseLocalProtocols name [... name]" + */ + + int protocols = parse_protocols(value); + + if (protocols < 0) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unknown browse protocol \"%s\" on line %d.", + value, linenum); + break; + } + + BrowseLocalProtocols = protocols; + } + else if (!_cups_strcasecmp(line, "default_auth_type") && value) + { + /* + * default_auth_type {basic,digest,basicdigest,negotiate} + */ + + if (!_cups_strcasecmp(value, "none")) + default_auth_type = CUPSD_AUTH_NONE; + else if (!_cups_strcasecmp(value, "basic")) + default_auth_type = CUPSD_AUTH_BASIC; + else if (!_cups_strcasecmp(value, "digest")) + default_auth_type = CUPSD_AUTH_DIGEST; + else if (!_cups_strcasecmp(value, "basicdigest")) + default_auth_type = CUPSD_AUTH_BASICDIGEST; +#ifdef HAVE_GSSAPI + else if (!_cups_strcasecmp(value, "negotiate")) + default_auth_type = CUPSD_AUTH_NEGOTIATE; +#endif /* HAVE_GSSAPI */ + else if (!_cups_strcasecmp(value, "auto")) + default_auth_type = CUPSD_AUTH_AUTO; + else + { + cupsdLogMessage(CUPSD_LOG_WARN, + "Unknown default authorization type %s on line %d.", + value, linenum); + if (FatalErrors & CUPSD_FATAL_CONFIG) + return (0); + } + } +#ifdef HAVE_SSL + else if (!_cups_strcasecmp(line, "DefaultEncryption")) + { + /* + * DefaultEncryption {Never,IfRequested,Required} + */ + + if (!value || !_cups_strcasecmp(value, "never")) + DefaultEncryption = HTTP_ENCRYPT_NEVER; + else if (!_cups_strcasecmp(value, "required")) + DefaultEncryption = HTTP_ENCRYPT_REQUIRED; + else if (!_cups_strcasecmp(value, "ifrequested")) + DefaultEncryption = HTTP_ENCRYPT_IF_REQUESTED; + else + { + cupsdLogMessage(CUPSD_LOG_WARN, + "Unknown default encryption %s on line %d.", + value, linenum); + if (FatalErrors & CUPSD_FATAL_CONFIG) + return (0); + } + } +#endif /* HAVE_SSL */ + else if (!_cups_strcasecmp(line, "User") && value) + { + /* + * User ID to run as... + */ + + if (isdigit(value[0] & 255)) + { + int uid = atoi(value); + + if (!uid) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Will not use User 0 as specified on line %d " + "for security reasons. You must use a non-" + "privileged account instead.", + linenum); + else + User = atoi(value); + } + else + { + struct passwd *p; /* Password information */ + + endpwent(); + p = getpwnam(value); + + if (p) + { + if (!p->pw_uid) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Will not use User %s (UID=0) as specified on line " + "%d for security reasons. You must use a non-" + "privileged account instead.", + value, linenum); + else + User = p->pw_uid; + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unknown User \"%s\" on line %d, ignoring.", + value, linenum); + } + } + else if (!_cups_strcasecmp(line, "Group") && value) + { + /* + * Group ID to run as... + */ + + if (isdigit(value[0])) + Group = atoi(value); + else + { + endgrent(); + group = getgrnam(value); + + if (group != NULL) + Group = group->gr_gid; + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unknown Group \"%s\" on line %d, ignoring.", + value, linenum); + } + } + else if (!_cups_strcasecmp(line, "SystemGroup") && value) + { + /* + * SystemGroup (admin) group(s)... + */ + + if (!parse_groups(value)) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unknown SystemGroup \"%s\" on line %d, ignoring.", + value, linenum); + } + else if (!_cups_strcasecmp(line, "HostNameLookups") && value) + { + /* + * Do hostname lookups? + */ + + if (!_cups_strcasecmp(value, "off") || !_cups_strcasecmp(value, "no") || + !_cups_strcasecmp(value, "false")) + HostNameLookups = 0; + else if (!_cups_strcasecmp(value, "on") || !_cups_strcasecmp(value, "yes") || + !_cups_strcasecmp(value, "true")) + HostNameLookups = 1; + else if (!_cups_strcasecmp(value, "double")) + HostNameLookups = 2; + else + cupsdLogMessage(CUPSD_LOG_WARN, "Unknown HostNameLookups %s on line %d.", + value, linenum); + } + else if (!_cups_strcasecmp(line, "AccessLogLevel") && value) + { + /* + * Amount of logging to do to access log... + */ + + if (!_cups_strcasecmp(value, "all")) + AccessLogLevel = CUPSD_ACCESSLOG_ALL; + else if (!_cups_strcasecmp(value, "actions")) + AccessLogLevel = CUPSD_ACCESSLOG_ACTIONS; + else if (!_cups_strcasecmp(value, "config")) + AccessLogLevel = CUPSD_ACCESSLOG_CONFIG; + else + cupsdLogMessage(CUPSD_LOG_WARN, "Unknown AccessLogLevel %s on line %d.", + value, linenum); + } + else if (!_cups_strcasecmp(line, "LogLevel") && value) + { + /* + * Amount of logging to do to error log... + */ + + if (!_cups_strcasecmp(value, "debug2")) + LogLevel = CUPSD_LOG_DEBUG2; + else if (!_cups_strcasecmp(value, "debug")) + LogLevel = CUPSD_LOG_DEBUG; + else if (!_cups_strcasecmp(value, "info")) + LogLevel = CUPSD_LOG_INFO; + else if (!_cups_strcasecmp(value, "notice")) + LogLevel = CUPSD_LOG_NOTICE; + else if (!_cups_strcasecmp(value, "warn")) + LogLevel = CUPSD_LOG_WARN; + else if (!_cups_strcasecmp(value, "error")) + LogLevel = CUPSD_LOG_ERROR; + else if (!_cups_strcasecmp(value, "crit")) + LogLevel = CUPSD_LOG_CRIT; + else if (!_cups_strcasecmp(value, "alert")) + LogLevel = CUPSD_LOG_ALERT; + else if (!_cups_strcasecmp(value, "emerg")) + LogLevel = CUPSD_LOG_EMERG; + else if (!_cups_strcasecmp(value, "none")) + LogLevel = CUPSD_LOG_NONE; + else + cupsdLogMessage(CUPSD_LOG_WARN, "Unknown LogLevel %s on line %d.", + value, linenum); + } + else if (!_cups_strcasecmp(line, "LogTimeFormat") && value) + { + /* + * Amount of logging to do to error log... + */ + + if (!_cups_strcasecmp(value, "standard")) + LogTimeFormat = CUPSD_TIME_STANDARD; + else if (!_cups_strcasecmp(value, "usecs")) + LogTimeFormat = CUPSD_TIME_USECS; + else + cupsdLogMessage(CUPSD_LOG_WARN, "Unknown LogTimeFormat %s on line %d.", + value, linenum); + } + else if (!_cups_strcasecmp(line, "PrintcapFormat") && value) + { + /* + * Format of printcap file? + */ + + if (!_cups_strcasecmp(value, "bsd")) + PrintcapFormat = PRINTCAP_BSD; + else if (!_cups_strcasecmp(value, "plist")) + PrintcapFormat = PRINTCAP_PLIST; + else if (!_cups_strcasecmp(value, "solaris")) + PrintcapFormat = PRINTCAP_SOLARIS; + else + cupsdLogMessage(CUPSD_LOG_WARN, "Unknown PrintcapFormat %s on line %d.", + value, linenum); + } + else if (!_cups_strcasecmp(line, "ServerTokens") && value) + { + /* + * Set the string used for the Server header... + */ + + struct utsname plat; /* Platform info */ + + + uname(&plat); + + if (!_cups_strcasecmp(value, "ProductOnly")) + cupsdSetString(&ServerHeader, "CUPS"); + else if (!_cups_strcasecmp(value, "Major")) + cupsdSetStringf(&ServerHeader, "CUPS/%d", CUPS_VERSION_MAJOR); + else if (!_cups_strcasecmp(value, "Minor")) + cupsdSetStringf(&ServerHeader, "CUPS/%d.%d", CUPS_VERSION_MAJOR, + CUPS_VERSION_MINOR); + else if (!_cups_strcasecmp(value, "Minimal")) + cupsdSetString(&ServerHeader, CUPS_MINIMAL); + else if (!_cups_strcasecmp(value, "OS")) + cupsdSetStringf(&ServerHeader, CUPS_MINIMAL " (%s)", plat.sysname); + else if (!_cups_strcasecmp(value, "Full")) + cupsdSetStringf(&ServerHeader, CUPS_MINIMAL " (%s) IPP/2.1", + plat.sysname); + else if (!_cups_strcasecmp(value, "None")) + cupsdClearString(&ServerHeader); + else + cupsdLogMessage(CUPSD_LOG_WARN, "Unknown ServerTokens %s on line %d.", + value, linenum); + } + else if (!_cups_strcasecmp(line, "PassEnv") && value) + { + /* + * PassEnv variable [... variable] + */ + + for (; *value;) + { + for (valuelen = 0; value[valuelen]; valuelen ++) + if (_cups_isspace(value[valuelen]) || value[valuelen] == ',') + break; + + if (value[valuelen]) + { + value[valuelen] = '\0'; + valuelen ++; + } + + cupsdSetEnv(value, NULL); + + for (value += valuelen; *value; value ++) + if (!_cups_isspace(*value) || *value != ',') + break; + } + } + else if (!_cups_strcasecmp(line, "ServerAlias") && value) + { + /* + * ServerAlias name [... name] + */ + + if (!ServerAlias) + ServerAlias = cupsArrayNew(NULL, NULL); + + for (; *value;) + { + for (valuelen = 0; value[valuelen]; valuelen ++) + if (_cups_isspace(value[valuelen]) || value[valuelen] == ',') + break; + + if (value[valuelen]) + { + value[valuelen] = '\0'; + valuelen ++; + } + + cupsdAddAlias(ServerAlias, value); + + for (value += valuelen; *value; value ++) + if (!_cups_isspace(*value) || *value != ',') + break; + } + } + else if (!_cups_strcasecmp(line, "SetEnv") && value) + { + /* + * SetEnv variable value + */ + + for (valueptr = value; *valueptr && !isspace(*valueptr & 255); valueptr ++); + + if (*valueptr) + { + /* + * Found a value... + */ + + while (isspace(*valueptr & 255)) + *valueptr++ = '\0'; + + cupsdSetEnv(value, valueptr); + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Missing value for SetEnv directive on line %d.", + linenum); + } +#ifdef HAVE_SSL + else if (!_cups_strcasecmp(line, "SSLOptions")) + { + /* + * SSLOptions options + */ + + if (!value || !_cups_strcasecmp(value, "none")) + SSLOptions = CUPSD_SSL_NONE; + else if (!_cups_strcasecmp(value, "noemptyfragments")) + SSLOptions = CUPSD_SSL_NOEMPTY; + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unknown value \"%s\" for SSLOptions directive on " + "line %d.", value, linenum); + } +#endif /* HAVE_SSL */ + else + { + /* + * Find a simple variable in the list... + */ + + for (i = NUM_VARS, var = variables; i > 0; i --, var ++) + if (!_cups_strcasecmp(line, var->name)) + break; + + if (i == 0) + { + /* + * Unknown directive! Output an error message and continue... + */ + + if (!value) + cupsdLogMessage(CUPSD_LOG_ERROR, "Missing value for %s on line %d.", + line, linenum); + else + cupsdLogMessage(CUPSD_LOG_ERROR, "Unknown directive %s on line %d.", + line, linenum); + continue; + } + + switch (var->type) + { + case CUPSD_VARTYPE_INTEGER : + if (!value) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Missing integer value for %s on line %d.", + line, linenum); + else + { + int n; /* Number */ + char *units; /* Units */ + + + n = strtol(value, &units, 0); + + if (units && *units) + { + if (tolower(units[0] & 255) == 'g') + n *= 1024 * 1024 * 1024; + else if (tolower(units[0] & 255) == 'm') + n *= 1024 * 1024; + else if (tolower(units[0] & 255) == 'k') + n *= 1024; + else if (tolower(units[0] & 255) == 't') + n *= 262144; + } + + if (n < 0) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Bad negative integer value for %s on line %d.", + line, linenum); + else + *((int *)var->ptr) = n; + } + break; + + case CUPSD_VARTYPE_BOOLEAN : + if (!value) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Missing boolean value for %s on line %d.", + line, linenum); + else if (!_cups_strcasecmp(value, "true") || + !_cups_strcasecmp(value, "on") || + !_cups_strcasecmp(value, "enabled") || + !_cups_strcasecmp(value, "yes") || + atoi(value) != 0) + *((int *)var->ptr) = TRUE; + else if (!_cups_strcasecmp(value, "false") || + !_cups_strcasecmp(value, "off") || + !_cups_strcasecmp(value, "disabled") || + !_cups_strcasecmp(value, "no") || + !_cups_strcasecmp(value, "0")) + *((int *)var->ptr) = FALSE; + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unknown boolean value %s on line %d.", + value, linenum); + break; + + case CUPSD_VARTYPE_PATHNAME : + if (!value) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Missing pathname value for %s on line %d.", + line, linenum); + break; + } + + if (value[0] == '/') + strlcpy(temp, value, sizeof(temp)); + else + snprintf(temp, sizeof(temp), "%s/%s", ServerRoot, value); + + if (access(temp, 0)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "File or directory for \"%s %s\" on line %d " + "does not exist.", line, value, linenum); + break; + } + + case CUPSD_VARTYPE_STRING : + cupsdSetString((char **)var->ptr, value); + break; + } + } + } + + return (1); +} + + +/* + * 'read_location()' - Read a definition. + */ + +static int /* O - New line number or 0 on error */ +read_location(cups_file_t *fp, /* I - Configuration file */ + char *location, /* I - Location name/path */ + int linenum) /* I - Current line number */ +{ + cupsd_location_t *loc, /* New location */ + *parent; /* Parent location */ + char line[HTTP_MAX_BUFFER], + /* Line buffer */ + *value, /* Value for directive */ + *valptr; /* Pointer into value */ + + + if ((parent = cupsdFindLocation(location)) != NULL) + cupsdLogMessage(CUPSD_LOG_WARN, "Duplicate on line %d.", + location, linenum); + else if ((parent = cupsdNewLocation(location)) == NULL) + return (0); + else + { + cupsdAddLocation(parent); + + parent->limit = CUPSD_AUTH_LIMIT_ALL; + } + + loc = parent; + + while (cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + { + /* + * Decode the directive... + */ + + if (!_cups_strcasecmp(line, "")) + return (linenum); + else if (!_cups_strcasecmp(line, "limit = 0; + while (*value) + { + for (valptr = value; !isspace(*valptr & 255) && *valptr; valptr ++); + + if (*valptr) + *valptr++ = '\0'; + + if (!strcmp(value, "ALL")) + loc->limit = CUPSD_AUTH_LIMIT_ALL; + else if (!strcmp(value, "GET")) + loc->limit |= CUPSD_AUTH_LIMIT_GET; + else if (!strcmp(value, "HEAD")) + loc->limit |= CUPSD_AUTH_LIMIT_HEAD; + else if (!strcmp(value, "OPTIONS")) + loc->limit |= CUPSD_AUTH_LIMIT_OPTIONS; + else if (!strcmp(value, "POST")) + loc->limit |= CUPSD_AUTH_LIMIT_POST; + else if (!strcmp(value, "PUT")) + loc->limit |= CUPSD_AUTH_LIMIT_PUT; + else if (!strcmp(value, "TRACE")) + loc->limit |= CUPSD_AUTH_LIMIT_TRACE; + else + cupsdLogMessage(CUPSD_LOG_WARN, "Unknown request type %s on line %d.", + value, linenum); + + for (value = valptr; isspace(*value & 255); value ++); + } + + if (!_cups_strcasecmp(line, "limit = CUPSD_AUTH_LIMIT_ALL ^ loc->limit; + + parent->limit &= ~loc->limit; + } + else if (!_cups_strcasecmp(line, "") || + !_cups_strcasecmp(line, "")) + loc = parent; + else if (!value) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Missing value on line %d.", linenum); + if (FatalErrors & CUPSD_FATAL_CONFIG) + return (0); + } + else if (!parse_aaa(loc, line, value, linenum)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unknown Location directive %s on line %d.", + line, linenum); + if (FatalErrors & CUPSD_FATAL_CONFIG) + return (0); + } + } + + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unexpected end-of-file at line %d while reading location.", + linenum); + + return ((FatalErrors & CUPSD_FATAL_CONFIG) ? 0 : linenum); +} + + +/* + * 'read_policy()' - Read a definition. + */ + +static int /* O - New line number or 0 on error */ +read_policy(cups_file_t *fp, /* I - Configuration file */ + char *policy, /* I - Location name/path */ + int linenum) /* I - Current line number */ +{ + int i; /* Looping var */ + cupsd_policy_t *pol; /* Policy */ + cupsd_location_t *op; /* Policy operation */ + int num_ops; /* Number of IPP operations */ + ipp_op_t ops[100]; /* Operations */ + char line[HTTP_MAX_BUFFER], + /* Line buffer */ + *value, /* Value for directive */ + *valptr; /* Pointer into value */ + + + /* + * Create the policy... + */ + + if ((pol = cupsdFindPolicy(policy)) != NULL) + cupsdLogMessage(CUPSD_LOG_WARN, "Duplicate on line %d.", + policy, linenum); + else if ((pol = cupsdAddPolicy(policy)) == NULL) + return (0); + + /* + * Read from the file... + */ + + op = NULL; + num_ops = 0; + + while (cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + { + /* + * Decode the directive... + */ + + if (!_cups_strcasecmp(line, "")) + { + if (op) + cupsdLogMessage(CUPSD_LOG_WARN, + "Missing before on line %d.", + linenum); + + set_policy_defaults(pol); + + return (linenum); + } + else if (!_cups_strcasecmp(line, "") && op) + { + /* + * Finish the current operation limit... + */ + + if (num_ops > 1) + { + /* + * Copy the policy to the other operations... + */ + + for (i = 1; i < num_ops; i ++) + cupsdAddPolicyOp(pol, op, ops[i]); + } + + op = NULL; + } + else if (!value) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Missing value on line %d.", linenum); + if (FatalErrors & CUPSD_FATAL_CONFIG) + return (0); + } + else if (!_cups_strcasecmp(line, "JobPrivateAccess") || + !_cups_strcasecmp(line, "JobPrivateValues") || + !_cups_strcasecmp(line, "SubscriptionPrivateAccess") || + !_cups_strcasecmp(line, "SubscriptionPrivateValues")) + { + if (op) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "%s directive must appear outside ... " + "on line %d.", line, linenum); + if (FatalErrors & CUPSD_FATAL_CONFIG) + return (0); + } + else + { + /* + * Pull out whitespace-delimited values... + */ + + while (*value) + { + /* + * Find the end of the current value... + */ + + for (valptr = value; !isspace(*valptr & 255) && *valptr; valptr ++); + + if (*valptr) + *valptr++ = '\0'; + + /* + * Save it appropriately... + */ + + if (!_cups_strcasecmp(line, "JobPrivateAccess")) + { + /* + * JobPrivateAccess {all|default|user/group list|@@ACL} + */ + + if (!_cups_strcasecmp(value, "default")) + { + cupsdAddString(&(pol->job_access), "@OWNER"); + cupsdAddString(&(pol->job_access), "@SYSTEM"); + } + else + cupsdAddString(&(pol->job_access), value); + } + else if (!_cups_strcasecmp(line, "JobPrivateValues")) + { + /* + * JobPrivateValues {all|none|default|attribute list} + */ + + if (!_cups_strcasecmp(value, "default")) + { + cupsdAddString(&(pol->job_attrs), "job-name"); + cupsdAddString(&(pol->job_attrs), "job-originating-host-name"); + cupsdAddString(&(pol->job_attrs), "job-originating-user-name"); + } + else + cupsdAddString(&(pol->job_attrs), value); + } + else if (!_cups_strcasecmp(line, "SubscriptionPrivateAccess")) + { + /* + * SubscriptionPrivateAccess {all|default|user/group list|@@ACL} + */ + + if (!_cups_strcasecmp(value, "default")) + { + cupsdAddString(&(pol->sub_access), "@OWNER"); + cupsdAddString(&(pol->sub_access), "@SYSTEM"); + } + else + cupsdAddString(&(pol->sub_access), value); + } + else /* if (!_cups_strcasecmp(line, "SubscriptionPrivateValues")) */ + { + /* + * SubscriptionPrivateValues {all|none|default|attribute list} + */ + + if (!_cups_strcasecmp(value, "default")) + { + cupsdAddString(&(pol->sub_attrs), "notify-events"); + cupsdAddString(&(pol->sub_attrs), "notify-pull-method"); + cupsdAddString(&(pol->sub_attrs), "notify-recipient-uri"); + cupsdAddString(&(pol->sub_attrs), "notify-subscriber-user-name"); + cupsdAddString(&(pol->sub_attrs), "notify-user-data"); + } + else + cupsdAddString(&(pol->sub_attrs), value); + } + + /* + * Find the next string on the line... + */ + + for (value = valptr; isspace(*value & 255); value ++); + } + } + } + else if (!op) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Missing directive before %s on line %d.", + line, linenum); + if (FatalErrors & CUPSD_FATAL_CONFIG) + return (0); + } + else if (!parse_aaa(op, line, value, linenum)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unknown Policy Limit directive %s on line %d.", + line, linenum); + + if (FatalErrors & CUPSD_FATAL_CONFIG) + return (0); + } + } + + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unexpected end-of-file at line %d while reading policy " + "\"%s\".", linenum, policy); + + return ((FatalErrors & CUPSD_FATAL_CONFIG) ? 0 : linenum); +} + + +/* + * 'set_policy_defaults()' - Set default policy values as needed. + */ + +static void +set_policy_defaults(cupsd_policy_t *pol)/* I - Policy */ +{ + cupsd_location_t *op; /* Policy operation */ + + + /* + * Verify that we have an explicit policy for Validate-Job, Cancel-Jobs, + * Cancel-My-Jobs, Close-Job, and CUPS-Get-Document, which ensures that + * upgrades do not introduce new security issues... + */ + + if ((op = cupsdFindPolicyOp(pol, IPP_VALIDATE_JOB)) == NULL || + op->op == IPP_ANY_OPERATION) + { + if ((op = cupsdFindPolicyOp(pol, IPP_PRINT_JOB)) != NULL && + op->op != IPP_ANY_OPERATION) + { + /* + * Add a new limit for Validate-Job using the Print-Job limit as a + * template... + */ + + cupsdLogMessage(CUPSD_LOG_WARN, + "No limit for Validate-Job defined in policy %s " + "- using Print-Job's policy.", pol->name); + + cupsdAddPolicyOp(pol, op, IPP_VALIDATE_JOB); + } + else + cupsdLogMessage(CUPSD_LOG_WARN, + "No limit for Validate-Job defined in policy %s " + "and no suitable template found.", pol->name); + } + + if ((op = cupsdFindPolicyOp(pol, IPP_CANCEL_JOBS)) == NULL || + op->op == IPP_ANY_OPERATION) + { + if ((op = cupsdFindPolicyOp(pol, IPP_PAUSE_PRINTER)) != NULL && + op->op != IPP_ANY_OPERATION) + { + /* + * Add a new limit for Cancel-Jobs using the Pause-Printer limit as a + * template... + */ + + cupsdLogMessage(CUPSD_LOG_WARN, + "No limit for Cancel-Jobs defined in policy %s " + "- using Pause-Printer's policy.", pol->name); + + cupsdAddPolicyOp(pol, op, IPP_CANCEL_JOBS); + } + else + cupsdLogMessage(CUPSD_LOG_WARN, + "No limit for Cancel-Jobs defined in policy %s " + "and no suitable template found.", pol->name); + } + + if ((op = cupsdFindPolicyOp(pol, IPP_CANCEL_MY_JOBS)) == NULL || + op->op == IPP_ANY_OPERATION) + { + if ((op = cupsdFindPolicyOp(pol, IPP_SEND_DOCUMENT)) != NULL && + op->op != IPP_ANY_OPERATION) + { + /* + * Add a new limit for Cancel-My-Jobs using the Send-Document limit as + * a template... + */ + + cupsdLogMessage(CUPSD_LOG_WARN, + "No limit for Cancel-My-Jobs defined in policy %s " + "- using Send-Document's policy.", pol->name); + + cupsdAddPolicyOp(pol, op, IPP_CANCEL_MY_JOBS); + } + else + cupsdLogMessage(CUPSD_LOG_WARN, + "No limit for Cancel-My-Jobs defined in policy %s " + "and no suitable template found.", pol->name); + } + + if ((op = cupsdFindPolicyOp(pol, IPP_CLOSE_JOB)) == NULL || + op->op == IPP_ANY_OPERATION) + { + if ((op = cupsdFindPolicyOp(pol, IPP_SEND_DOCUMENT)) != NULL && + op->op != IPP_ANY_OPERATION) + { + /* + * Add a new limit for Close-Job using the Send-Document limit as a + * template... + */ + + cupsdLogMessage(CUPSD_LOG_WARN, + "No limit for Close-Job defined in policy %s " + "- using Send-Document's policy.", pol->name); + + cupsdAddPolicyOp(pol, op, IPP_CLOSE_JOB); + } + else + cupsdLogMessage(CUPSD_LOG_WARN, + "No limit for Close-Job defined in policy %s " + "and no suitable template found.", pol->name); + } + + if ((op = cupsdFindPolicyOp(pol, CUPS_GET_DOCUMENT)) == NULL || + op->op == IPP_ANY_OPERATION) + { + if ((op = cupsdFindPolicyOp(pol, IPP_SEND_DOCUMENT)) != NULL && + op->op != IPP_ANY_OPERATION) + { + /* + * Add a new limit for CUPS-Get-Document using the Send-Document + * limit as a template... + */ + + cupsdLogMessage(CUPSD_LOG_WARN, + "No limit for CUPS-Get-Document defined in policy %s " + "- using Send-Document's policy.", pol->name); + + cupsdAddPolicyOp(pol, op, CUPS_GET_DOCUMENT); + } + else + cupsdLogMessage(CUPSD_LOG_WARN, + "No limit for CUPS-Get-Document defined in policy %s " + "and no suitable template found.", pol->name); + } + + /* + * Verify we have JobPrivateAccess, JobPrivateValues, + * SubscriptionPrivateAccess, and SubscriptionPrivateValues in the policy. + */ + + if (!pol->job_access) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "No JobPrivateAccess defined in policy %s " + "- using defaults.", pol->name); + cupsdAddString(&(pol->job_access), "@OWNER"); + cupsdAddString(&(pol->job_access), "@SYSTEM"); + } + + if (!pol->job_attrs) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "No JobPrivateValues defined in policy %s " + "- using defaults.", pol->name); + cupsdAddString(&(pol->job_attrs), "job-name"); + cupsdAddString(&(pol->job_attrs), "job-originating-host-name"); + cupsdAddString(&(pol->job_attrs), "job-originating-user-name"); + } + + if (!pol->sub_access) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "No SubscriptionPrivateAccess defined in policy %s " + "- using defaults.", pol->name); + cupsdAddString(&(pol->sub_access), "@OWNER"); + cupsdAddString(&(pol->sub_access), "@SYSTEM"); + } + + if (!pol->sub_attrs) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "No SubscriptionPrivateValues defined in policy %s " + "- using defaults.", pol->name); + cupsdAddString(&(pol->sub_attrs), "notify-events"); + cupsdAddString(&(pol->sub_attrs), "notify-pull-method"); + cupsdAddString(&(pol->sub_attrs), "notify-recipient-uri"); + cupsdAddString(&(pol->sub_attrs), "notify-subscriber-user-name"); + cupsdAddString(&(pol->sub_attrs), "notify-user-data"); + } +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/conf.h b/scheduler/conf.h new file mode 100644 index 0000000000..60649e1bde --- /dev/null +++ b/scheduler/conf.h @@ -0,0 +1,296 @@ +/* + * "$Id$" + * + * Configuration file definitions for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + + +/* + * Log levels... + */ + +typedef enum +{ + CUPSD_LOG_PPD = -4, /* Used internally for PPD keywords */ + CUPSD_LOG_ATTR, /* Used internally for attributes */ + CUPSD_LOG_STATE, /* Used internally for state-reasons */ + CUPSD_LOG_PAGE, /* Used internally for page logging */ + CUPSD_LOG_NONE, + CUPSD_LOG_EMERG, /* Emergency issues */ + CUPSD_LOG_ALERT, /* Something bad happened that needs attention */ + CUPSD_LOG_CRIT, /* Critical error but server continues */ + CUPSD_LOG_ERROR, /* Error condition */ + CUPSD_LOG_WARN, /* Warning */ + CUPSD_LOG_NOTICE, /* Normal condition that needs logging */ + CUPSD_LOG_INFO, /* General information */ + CUPSD_LOG_DEBUG, /* General debugging */ + CUPSD_LOG_DEBUG2 /* Detailed debugging */ +} cupsd_loglevel_t; + +typedef enum +{ + CUPSD_ACCESSLOG_CONFIG, /* Log config requests */ + CUPSD_ACCESSLOG_ACTIONS, /* Log config, print, and job management requests */ + CUPSD_ACCESSLOG_ALL /* Log everything */ +} cupsd_accesslog_t; + +typedef enum +{ + CUPSD_TIME_STANDARD, /* "Standard" Apache/CLF format */ + CUPSD_TIME_USECS /* Standard format with microseconds */ +} cupsd_time_t; + + +/* + * FatalErrors flags... + */ + +#define CUPSD_FATAL_NONE 0 /* No errors are fatal */ +#define CUPSD_FATAL_BROWSE 1 /* Browse bind errors are fatal */ +#define CUPSD_FATAL_CONFIG 2 /* Config file syntax errors are fatal */ +#define CUPSD_FATAL_LISTEN 4 /* Listen/Port bind errors are fatal */ +#define CUPSD_FATAL_LOG 8 /* Log file errors are fatal */ +#define CUPSD_FATAL_PERMISSIONS 16 /* File permission errors are fatal */ +#define CUPSD_FATAL_ALL ~0 /* All errors are fatal */ + + +/* + * Printcap formats... + */ + +#define PRINTCAP_BSD 0 /* Berkeley LPD format */ +#define PRINTCAP_SOLARIS 1 /* Solaris lpsched format */ +#define PRINTCAP_PLIST 2 /* Mac OS X plist format */ + + +/* + * SSL options (bits)... + */ + +#define CUPSD_SSL_NONE 0 /* No special options */ +#define CUPSD_SSL_NOEMPTY 1 /* Do not insert empty fragments */ + + +/* + * ServerAlias data... + */ + +typedef struct +{ + size_t namelen; /* Length of alias name */ + char name[1]; /* Alias name */ +} cupsd_alias_t; + + +/* + * Globals... + */ + +VAR char *ConfigurationFile VALUE(NULL), + /* Configuration file to use */ + *ServerName VALUE(NULL), + /* FQDN for server */ + *ServerAdmin VALUE(NULL), + /* Administrator's email */ + *ServerRoot VALUE(NULL), + /* Root directory for scheduler */ + *ServerBin VALUE(NULL), + /* Root directory for binaries */ + *StateDir VALUE(NULL), + /* Root directory for state data */ + *RequestRoot VALUE(NULL), + /* Directory for request files */ + *DocumentRoot VALUE(NULL); + /* Root directory for documents */ +VAR cups_array_t *ServerAlias VALUE(NULL); + /* Alias names for server */ +VAR int ServerNameIsIP VALUE(0); + /* Is the ServerName an IP address? */ +VAR int NumSystemGroups VALUE(0); + /* Number of system group names */ +VAR char *SystemGroups[MAX_SYSTEM_GROUPS] + VALUE({0}); + /* System group names */ +VAR gid_t SystemGroupIDs[MAX_SYSTEM_GROUPS] + VALUE({0}); + /* System group IDs */ +VAR char *AccessLog VALUE(NULL), + /* Access log filename */ + *ErrorLog VALUE(NULL), + /* Error log filename */ + *PageLog VALUE(NULL), + /* Page log filename */ + *CacheDir VALUE(NULL), + /* Cache file directory */ + *DataDir VALUE(NULL), + /* Data file directory */ + *DefaultLanguage VALUE(NULL), + /* Default language encoding */ + *DefaultLocale VALUE(NULL), + /* Default locale */ + *DefaultPaperSize VALUE(NULL), + /* Default paper size */ + *ErrorPolicy VALUE(NULL), + /* Default printer-error-policy */ + *RIPCache VALUE(NULL), + /* Amount of memory for RIPs */ + *TempDir VALUE(NULL), + /* Temporary directory */ + *Printcap VALUE(NULL), + /* Printcap file */ + *PrintcapGUI VALUE(NULL), + /* GUI program to use for IRIX */ + *FontPath VALUE(NULL), + /* Font search path */ + *RemoteRoot VALUE(NULL), + /* Remote root user */ + *Classification VALUE(NULL); + /* Classification of system */ +VAR uid_t User VALUE(1); + /* User ID for server */ +VAR gid_t Group VALUE(0); + /* Group ID for server */ +VAR cupsd_accesslog_t AccessLogLevel VALUE(CUPSD_ACCESSLOG_ACTIONS); + /* Access log level */ +VAR int ClassifyOverride VALUE(0), + /* Allow overrides? */ + ConfigFilePerm VALUE(0640), + /* Permissions for config files */ + LogDebugHistory VALUE(200), + /* Amount of automatic debug history */ + FatalErrors VALUE(CUPSD_FATAL_CONFIG), + /* Which errors are fatal? */ + LogFilePerm VALUE(0644); + /* Permissions for log files */ +VAR cupsd_loglevel_t LogLevel VALUE(CUPSD_LOG_WARN); + /* Error log level */ +VAR cupsd_time_t LogTimeFormat VALUE(CUPSD_TIME_STANDARD); + /* Log file time format */ +VAR int MaxClients VALUE(100), + /* Maximum number of clients */ + MaxClientsPerHost VALUE(0), + /* Maximum number of clients per host */ + MaxCopies VALUE(CUPS_DEFAULT_MAX_COPIES), + /* Maximum number of copies per job */ + MaxLogSize VALUE(1024 * 1024), + /* Maximum size of log files */ + MaxRequestSize VALUE(0), + /* Maximum size of IPP requests */ + HostNameLookups VALUE(FALSE), + /* Do we do reverse lookups? */ + Timeout VALUE(DEFAULT_TIMEOUT), + /* Timeout during requests */ + KeepAlive VALUE(TRUE), + /* Support the Keep-Alive option? */ + KeepAliveTimeout VALUE(DEFAULT_KEEPALIVE), + /* Timeout between requests */ + FileDevice VALUE(FALSE), + /* Allow file: devices? */ + FilterLimit VALUE(0), + /* Max filter cost at any time */ + FilterLevel VALUE(0), + /* Current filter level */ + FilterNice VALUE(0), + /* Nice value for filters */ + ReloadTimeout VALUE(DEFAULT_KEEPALIVE), + /* Timeout before reload from SIGHUP */ + RootCertDuration VALUE(300), + /* Root certificate update interval */ + RunUser VALUE(0), + /* User to run as, used for files */ + PrintcapFormat VALUE(PRINTCAP_BSD), + /* Format of printcap file? */ + DefaultShared VALUE(TRUE), + /* Share printers by default? */ + MultipleOperationTimeout VALUE(DEFAULT_TIMEOUT), + /* multiple-operation-time-out value */ + WebInterface VALUE(CUPS_DEFAULT_WEBIF); + /* Enable the web interface? */ +VAR cups_file_t *AccessFile VALUE(NULL), + /* Access log file */ + *ErrorFile VALUE(NULL), + /* Error log file */ + *PageFile VALUE(NULL); + /* Page log file */ +VAR char *PageLogFormat VALUE(NULL); + /* Page log format */ +VAR mime_t *MimeDatabase VALUE(NULL); + /* MIME type database */ +VAR int NumMimeTypes VALUE(0); + /* Number of MIME types */ +VAR const char **MimeTypes VALUE(NULL); + /* Array of MIME types */ + +#ifdef HAVE_SSL +VAR char *ServerCertificate VALUE(NULL); + /* Server certificate file */ +# if defined(HAVE_LIBSSL) || defined(HAVE_GNUTLS) +VAR char *ServerKey VALUE(NULL); + /* Server key file */ +# endif /* HAVE_LIBSSL || HAVE_GNUTLS */ +VAR int SSLOptions VALUE(CUPSD_SSL_NONE); + /* SSL/TLS options */ +#endif /* HAVE_SSL */ + +#ifdef HAVE_LAUNCHD +VAR int LaunchdTimeout VALUE(10); + /* Time after which an idle cupsd will exit */ +#endif /* HAVE_LAUNCHD */ + +#ifdef HAVE_AUTHORIZATION_H +VAR char *SystemGroupAuthKey VALUE(NULL); + /* System group auth key */ +#endif /* HAVE_AUTHORIZATION_H */ + +#ifdef HAVE_GSSAPI +VAR char *GSSServiceName VALUE(NULL); + /* GSS service name */ +int HaveServerCreds VALUE(0); + /* Do we have server credentials? */ +gss_cred_id_t ServerCreds; /* Server's GSS credentials */ +#endif /* HAVE_GSSAPI */ + + +/* + * Prototypes... + */ + +extern void cupsdAddAlias(cups_array_t *aliases, const char *name); +extern int cupsdCheckLogFile(cups_file_t **lf, const char *logname); +extern int cupsdCheckPermissions(const char *filename, + const char *suffix, int mode, + int user, int group, int is_dir, + int create_dir); +extern int cupsdCheckProgram(const char *filename, cupsd_printer_t *p); +extern int cupsdDefaultAuthType(void); +extern void cupsdFreeAliases(cups_array_t *aliases); +extern char *cupsdGetDateTime(struct timeval *t, cupsd_time_t format); +extern void cupsdLogFCMessage(void *context, _cups_fc_result_t result, + const char *message); +#ifdef HAVE_GSSAPI +extern int cupsdLogGSSMessage(int level, int major_status, + int minor_status, + const char *message, ...); +#endif /* HAVE_GSSAPI */ +extern int cupsdLogJob(cupsd_job_t *job, int level, const char *message, + ...) __attribute__((__format__(__printf__, 3, 4))); +extern int cupsdLogMessage(int level, const char *message, ...) + __attribute__ ((__format__ (__printf__, 2, 3))); +extern int cupsdLogPage(cupsd_job_t *job, const char *page); +extern int cupsdLogRequest(cupsd_client_t *con, http_status_t code); +extern int cupsdReadConfiguration(void); +extern int cupsdWriteErrorLog(int level, const char *message); + + +/* + * End of "$Id$". + */ diff --git a/scheduler/cups-deviced.c b/scheduler/cups-deviced.c new file mode 100644 index 0000000000..0cd5b8ea8c --- /dev/null +++ b/scheduler/cups-deviced.c @@ -0,0 +1,810 @@ +/* + * "$Id$" + * + * Device scanning mini-daemon for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Scan for devices and return an IPP response. + * add_device() - Add a new device to the list. + * compare_devices() - Compare device names to eliminate duplicates. + * get_current_time() - Get the current time as a double value in seconds. + * get_device() - Get a device from a backend. + * process_children() - Process all dead children... + * sigchld_handler() - Handle 'child' signals from old processes. + * start_backend() - Run a backend to gather the available devices. + */ + +/* + * Include necessary headers... + */ + +#include "util.h" +#include +#include +#include +#include +#include + + +/* + * Constants... + */ + +#define MAX_BACKENDS 200 /* Maximum number of backends we'll run */ + + +/* + * Backend information... + */ + +typedef struct +{ + char *name; /* Name of backend */ + int pid, /* Process ID */ + status; /* Exit status */ + cups_file_t *pipe; /* Pipe from backend stdout */ + int count; /* Number of devices found */ +} cupsd_backend_t; + + +/* + * Device information structure... + */ + +typedef struct +{ + char device_class[128], /* Device class */ + device_info[128], /* Device info/description */ + device_uri[1024]; /* Device URI */ +} cupsd_device_t; + + +/* + * Local globals... + */ + +static int num_backends = 0, + /* Total backends */ + active_backends = 0; + /* Active backends */ +static cupsd_backend_t backends[MAX_BACKENDS]; + /* Array of backends */ +static struct pollfd backend_fds[MAX_BACKENDS]; + /* Array for poll() */ +static cups_array_t *devices; /* Array of devices */ +static int normal_user; /* Normal user ID */ +static int device_limit; /* Maximum number of devices */ +static int send_class, /* Send device-class attribute? */ + send_info, /* Send device-info attribute? */ + send_make_and_model, + /* Send device-make-and-model attribute? */ + send_uri, /* Send device-uri attribute? */ + send_id, /* Send device-id attribute? */ + send_location; /* Send device-location attribute? */ +static int dead_children = 0; + /* Dead children? */ + + +/* + * Local functions... + */ + +static int add_device(const char *device_class, + const char *device_make_and_model, + const char *device_info, + const char *device_uri, + const char *device_id, + const char *device_location); +static int compare_devices(cupsd_device_t *p0, + cupsd_device_t *p1); +static double get_current_time(void); +static int get_device(cupsd_backend_t *backend); +static void process_children(void); +static void sigchld_handler(int sig); +static int start_backend(const char *backend, int root); + + +/* + * 'main()' - Scan for devices and return an IPP response. + * + * Usage: + * + * cups-deviced request_id limit options + */ + +int /* O - Exit code */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + int request_id; /* Request ID */ + int timeout; /* Timeout in seconds */ + const char *server_bin; /* CUPS_SERVERBIN environment variable */ + char filename[1024]; /* Backend directory filename */ + cups_dir_t *dir; /* Directory pointer */ + cups_dentry_t *dent; /* Directory entry */ + double current_time, /* Current time */ + end_time; /* Ending time */ + int num_options; /* Number of options */ + cups_option_t *options; /* Options */ + cups_array_t *requested, /* requested-attributes values */ + *exclude, /* exclude-schemes values */ + *include; /* include-schemes values */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* Actions for POSIX signals */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ + + + setbuf(stderr, NULL); + + /* + * Check the command-line... + */ + + if (argc != 6) + { + fputs("Usage: cups-deviced request-id limit timeout user-id options\n", stderr); + + return (1); + } + + request_id = atoi(argv[1]); + if (request_id < 1) + { + fprintf(stderr, "ERROR: [cups-deviced] Bad request ID %d!\n", request_id); + + return (1); + } + + device_limit = atoi(argv[2]); + if (device_limit < 0) + { + fprintf(stderr, "ERROR: [cups-deviced] Bad limit %d!\n", device_limit); + + return (1); + } + + timeout = atoi(argv[3]); + if (timeout < 1) + { + fprintf(stderr, "ERROR: [cups-deviced] Bad timeout %d!\n", timeout); + + return (1); + } + + normal_user = atoi(argv[4]); + if (normal_user <= 0) + { + fprintf(stderr, "ERROR: [cups-deviced] Bad user %d!\n", normal_user); + + return (1); + } + + num_options = cupsParseOptions(argv[5], 0, &options); + requested = cupsdCreateStringsArray(cupsGetOption("requested-attributes", + num_options, options)); + exclude = cupsdCreateStringsArray(cupsGetOption("exclude-schemes", + num_options, options)); + include = cupsdCreateStringsArray(cupsGetOption("include-schemes", + num_options, options)); + + if (!requested || cupsArrayFind(requested, "all") != NULL) + { + send_class = send_info = send_make_and_model = send_uri = send_id = + send_location = 1; + } + else + { + send_class = cupsArrayFind(requested, "device-class") != NULL; + send_info = cupsArrayFind(requested, "device-info") != NULL; + send_make_and_model = cupsArrayFind(requested, "device-make-and-model") != NULL; + send_uri = cupsArrayFind(requested, "device-uri") != NULL; + send_id = cupsArrayFind(requested, "device-id") != NULL; + send_location = cupsArrayFind(requested, "device-location") != NULL; + } + + /* + * Listen to child signals... + */ + +#ifdef HAVE_SIGSET /* Use System V signals over POSIX to avoid bugs */ + sigset(SIGCHLD, sigchld_handler); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + + sigemptyset(&action.sa_mask); + sigaddset(&action.sa_mask, SIGCHLD); + action.sa_handler = sigchld_handler; + sigaction(SIGCHLD, &action, NULL); +#else + signal(SIGCLD, sigchld_handler); /* No, SIGCLD isn't a typo... */ +#endif /* HAVE_SIGSET */ + + /* + * Try opening the backend directory... + */ + + if ((server_bin = getenv("CUPS_SERVERBIN")) == NULL) + server_bin = CUPS_SERVERBIN; + + snprintf(filename, sizeof(filename), "%s/backend", server_bin); + + if ((dir = cupsDirOpen(filename)) == NULL) + { + fprintf(stderr, "ERROR: [cups-deviced] Unable to open backend directory " + "\"%s\": %s", filename, strerror(errno)); + + return (1); + } + + /* + * Setup the devices array... + */ + + devices = cupsArrayNew((cups_array_func_t)compare_devices, NULL); + + /* + * Loop through all of the device backends... + */ + + while ((dent = cupsDirRead(dir)) != NULL) + { + /* + * Skip entries that are not executable files... + */ + + if (!S_ISREG(dent->fileinfo.st_mode) || + !isalnum(dent->filename[0] & 255) || + (dent->fileinfo.st_mode & (S_IRUSR | S_IXUSR)) != (S_IRUSR | S_IXUSR)) + continue; + + /* + * Skip excluded or not included backends... + */ + + if (cupsArrayFind(exclude, dent->filename) || + (include && !cupsArrayFind(include, dent->filename))) + continue; + + /* + * Backends without permissions for normal users run as root, + * all others run as the unprivileged user... + */ + + start_backend(dent->filename, + !(dent->fileinfo.st_mode & (S_IRWXG | S_IRWXO))); + } + + cupsDirClose(dir); + + /* + * Collect devices... + */ + + if (getenv("SOFTWARE")) + puts("Content-Type: application/ipp\n"); + + cupsdSendIPPHeader(IPP_OK, request_id); + cupsdSendIPPGroup(IPP_TAG_OPERATION); + cupsdSendIPPString(IPP_TAG_CHARSET, "attributes-charset", "utf-8"); + cupsdSendIPPString(IPP_TAG_LANGUAGE, "attributes-natural-language", "en-US"); + + end_time = get_current_time() + timeout; + + while (active_backends > 0 && (current_time = get_current_time()) < end_time) + { + /* + * Collect the output from the backends... + */ + + timeout = (int)(1000 * (end_time - current_time)); + + if (poll(backend_fds, num_backends, timeout) > 0) + { + for (i = 0; i < num_backends; i ++) + if (backend_fds[i].revents && backends[i].pipe) + { + cups_file_t *bpipe = backends[i].pipe; + /* Copy of pipe for backend... */ + + do + { + if (get_device(backends + i)) + { + backend_fds[i].fd = 0; + backend_fds[i].events = 0; + break; + } + } + while (bpipe->ptr && + memchr(bpipe->ptr, '\n', bpipe->end - bpipe->ptr)); + } + } + + /* + * Get exit status from children... + */ + + if (dead_children) + process_children(); + } + + cupsdSendIPPTrailer(); + + /* + * Terminate any remaining backends and exit... + */ + + if (active_backends > 0) + { + for (i = 0; i < num_backends; i ++) + if (backends[i].pid) + kill(backends[i].pid, SIGTERM); + } + + return (0); +} + + +/* + * 'add_device()' - Add a new device to the list. + */ + +static int /* O - 0 on success, -1 on error */ +add_device( + const char *device_class, /* I - Device class */ + const char *device_make_and_model, /* I - Device make and model */ + const char *device_info, /* I - Device information */ + const char *device_uri, /* I - Device URI */ + const char *device_id, /* I - 1284 device ID */ + const char *device_location) /* I - Physical location */ +{ + cupsd_device_t *device; /* New device */ + + + /* + * Allocate memory for the device record... + */ + + if ((device = calloc(1, sizeof(cupsd_device_t))) == NULL) + { + fputs("ERROR: [cups-deviced] Ran out of memory allocating a device!\n", + stderr); + return (-1); + } + + /* + * Copy the strings over... + */ + + strlcpy(device->device_class, device_class, sizeof(device->device_class)); + strlcpy(device->device_info, device_info, sizeof(device->device_info)); + strlcpy(device->device_uri, device_uri, sizeof(device->device_uri)); + + /* + * Add the device to the array and return... + */ + + if (cupsArrayFind(devices, device)) + { + /* + * Avoid duplicates! + */ + + free(device); + } + else + { + cupsArrayAdd(devices, device); + + if (device_limit <= 0 || cupsArrayCount(devices) < device_limit) + { + /* + * Send device info... + */ + + cupsdSendIPPGroup(IPP_TAG_PRINTER); + if (send_class) + cupsdSendIPPString(IPP_TAG_KEYWORD, "device-class", + device_class); + if (send_info) + cupsdSendIPPString(IPP_TAG_TEXT, "device-info", device_info); + if (send_make_and_model) + cupsdSendIPPString(IPP_TAG_TEXT, "device-make-and-model", + device_make_and_model); + if (send_uri) + cupsdSendIPPString(IPP_TAG_URI, "device-uri", device_uri); + if (send_id) + cupsdSendIPPString(IPP_TAG_TEXT, "device-id", + device_id ? device_id : ""); + if (send_location) + cupsdSendIPPString(IPP_TAG_TEXT, "device-location", + device_location ? device_location : ""); + + fflush(stdout); + fputs("DEBUG: Flushed attributes...\n", stderr); + } + } + + return (0); +} + + +/* + * 'compare_devices()' - Compare device names to eliminate duplicates. + */ + +static int /* O - Result of comparison */ +compare_devices(cupsd_device_t *d0, /* I - First device */ + cupsd_device_t *d1) /* I - Second device */ +{ + int diff; /* Difference between strings */ + + + /* + * Sort devices by device-info, device-class, and device-uri... + */ + + if ((diff = cupsdCompareNames(d0->device_info, d1->device_info)) != 0) + return (diff); + else if ((diff = _cups_strcasecmp(d0->device_class, d1->device_class)) != 0) + return (diff); + else + return (_cups_strcasecmp(d0->device_uri, d1->device_uri)); +} + + +/* + * 'get_current_time()' - Get the current time as a double value in seconds. + */ + +static double /* O - Time in seconds */ +get_current_time(void) +{ + struct timeval curtime; /* Current time */ + + + gettimeofday(&curtime, NULL); + + return (curtime.tv_sec + 0.000001 * curtime.tv_usec); +} + + +/* + * 'get_device()' - Get a device from a backend. + */ + +static int /* O - 0 on success, -1 on error */ +get_device(cupsd_backend_t *backend) /* I - Backend to read from */ +{ + char line[2048], /* Line from backend */ + temp[2048], /* Copy of line */ + *ptr, /* Pointer into line */ + *dclass, /* Device class */ + *uri, /* Device URI */ + *make_model, /* Make and model */ + *info, /* Device info */ + *device_id, /* 1284 device ID */ + *location; /* Physical location */ + + + if (cupsFileGets(backend->pipe, line, sizeof(line))) + { + /* + * Each line is of the form: + * + * class URI "make model" "name" ["1284 device ID"] ["location"] + */ + + strlcpy(temp, line, sizeof(temp)); + + /* + * device-class + */ + + dclass = temp; + + for (ptr = temp; *ptr; ptr ++) + if (isspace(*ptr & 255)) + break; + + while (isspace(*ptr & 255)) + *ptr++ = '\0'; + + /* + * device-uri + */ + + if (!*ptr) + goto error; + + for (uri = ptr; *ptr; ptr ++) + if (isspace(*ptr & 255)) + break; + + while (isspace(*ptr & 255)) + *ptr++ = '\0'; + + /* + * device-make-and-model + */ + + if (*ptr != '\"') + goto error; + + for (ptr ++, make_model = ptr; *ptr && *ptr != '\"'; ptr ++) + { + if (*ptr == '\\' && ptr[1]) + _cups_strcpy(ptr, ptr + 1); + } + + if (*ptr != '\"') + goto error; + + for (*ptr++ = '\0'; isspace(*ptr & 255); *ptr++ = '\0'); + + /* + * device-info + */ + + if (*ptr != '\"') + goto error; + + for (ptr ++, info = ptr; *ptr && *ptr != '\"'; ptr ++) + { + if (*ptr == '\\' && ptr[1]) + _cups_strcpy(ptr, ptr + 1); + } + + if (*ptr != '\"') + goto error; + + for (*ptr++ = '\0'; isspace(*ptr & 255); *ptr++ = '\0'); + + /* + * device-id + */ + + if (*ptr == '\"') + { + for (ptr ++, device_id = ptr; *ptr && *ptr != '\"'; ptr ++) + { + if (*ptr == '\\' && ptr[1]) + _cups_strcpy(ptr, ptr + 1); + } + + if (*ptr != '\"') + goto error; + + for (*ptr++ = '\0'; isspace(*ptr & 255); *ptr++ = '\0'); + + /* + * device-location + */ + + if (*ptr == '\"') + { + for (ptr ++, location = ptr; *ptr && *ptr != '\"'; ptr ++) + { + if (*ptr == '\\' && ptr[1]) + _cups_strcpy(ptr, ptr + 1); + } + + if (*ptr != '\"') + goto error; + + *ptr = '\0'; + } + else + location = NULL; + } + else + { + device_id = NULL; + location = NULL; + } + + /* + * Add the device to the array of available devices... + */ + + if (!add_device(dclass, make_model, info, uri, device_id, location)) + fprintf(stderr, "DEBUG: [cups-deviced] Found device \"%s\"...\n", uri); + + return (0); + } + + /* + * End of file... + */ + + cupsFileClose(backend->pipe); + backend->pipe = NULL; + + return (-1); + + /* + * Bad format; strip trailing newline and write an error message. + */ + + error: + + if (line[strlen(line) - 1] == '\n') + line[strlen(line) - 1] = '\0'; + + fprintf(stderr, "ERROR: [cups-deviced] Bad line from \"%s\": %s\n", + backend->name, line); + return (0); +} + + +/* + * 'process_children()' - Process all dead children... + */ + +static void +process_children(void) +{ + int i; /* Looping var */ + int status; /* Exit status of child */ + int pid; /* Process ID of child */ + cupsd_backend_t *backend; /* Current backend */ + const char *name; /* Name of process */ + + + /* + * Reset the dead_children flag... + */ + + dead_children = 0; + + /* + * Collect the exit status of some children... + */ + +#ifdef HAVE_WAITPID + while ((pid = waitpid(-1, &status, WNOHANG)) > 0) +#elif defined(HAVE_WAIT3) + while ((pid = wait3(&status, WNOHANG, NULL)) > 0) +#else + if ((pid = wait(&status)) > 0) +#endif /* HAVE_WAITPID */ + { + if (status == SIGTERM) + status = 0; + + for (i = num_backends, backend = backends; i > 0; i --, backend ++) + if (backend->pid == pid) + break; + + if (i > 0) + { + name = backend->name; + backend->pid = 0; + backend->status = status; + + active_backends --; + } + else + name = "Unknown"; + + if (status) + { + if (WIFEXITED(status)) + fprintf(stderr, + "ERROR: [cups-deviced] PID %d (%s) stopped with status %d!\n", + pid, name, WEXITSTATUS(status)); + else + fprintf(stderr, + "ERROR: [cups-deviced] PID %d (%s) crashed on signal %d!\n", + pid, name, WTERMSIG(status)); + } + else + fprintf(stderr, + "DEBUG: [cups-deviced] PID %d (%s) exited with no errors.\n", + pid, name); + } +} + + +/* + * 'sigchld_handler()' - Handle 'child' signals from old processes. + */ + +static void +sigchld_handler(int sig) /* I - Signal number */ +{ + (void)sig; + + /* + * Flag that we have dead children... + */ + + dead_children = 1; + + /* + * Reset the signal handler as needed... + */ + +#if !defined(HAVE_SIGSET) && !defined(HAVE_SIGACTION) + signal(SIGCLD, sigchld_handler); +#endif /* !HAVE_SIGSET && !HAVE_SIGACTION */ +} + + +/* + * 'start_backend()' - Run a backend to gather the available devices. + */ + +static int /* O - 0 on success, -1 on error */ +start_backend(const char *name, /* I - Backend to run */ + int root) /* I - Run as root? */ +{ + const char *server_bin; /* CUPS_SERVERBIN environment variable */ + char program[1024]; /* Full path to backend */ + cupsd_backend_t *backend; /* Current backend */ + char *argv[2]; /* Command-line arguments */ + + + if (num_backends >= MAX_BACKENDS) + { + fprintf(stderr, "ERROR: Too many backends (%d)!\n", num_backends); + return (-1); + } + + if ((server_bin = getenv("CUPS_SERVERBIN")) == NULL) + server_bin = CUPS_SERVERBIN; + + snprintf(program, sizeof(program), "%s/backend/%s", server_bin, name); + + if (_cupsFileCheck(program, _CUPS_FILE_CHECK_PROGRAM, !geteuid(), + _cupsFileCheckFilter, NULL)) + return (-1); + + backend = backends + num_backends; + + argv[0] = (char *)name; + argv[1] = NULL; + + if ((backend->pipe = cupsdPipeCommand(&(backend->pid), program, argv, + root ? 0 : normal_user)) == NULL) + { + fprintf(stderr, "ERROR: [cups-deviced] Unable to execute \"%s\" - %s\n", + program, strerror(errno)); + return (-1); + } + + /* + * Fill in the rest of the backend information... + */ + + fprintf(stderr, "DEBUG: [cups-deviced] Started backend %s (PID %d)\n", + program, backend->pid); + + backend_fds[num_backends].fd = cupsFileNumber(backend->pipe); + backend_fds[num_backends].events = POLLIN; + + backend->name = strdup(name); + backend->status = 0; + backend->count = 0; + + active_backends ++; + num_backends ++; + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/cups-driverd.cxx b/scheduler/cups-driverd.cxx new file mode 100644 index 0000000000..c26621c3cd --- /dev/null +++ b/scheduler/cups-driverd.cxx @@ -0,0 +1,2594 @@ +/* + * "$Id$" + * + * PPD/driver support for CUPS. + * + * This program handles listing and installing static PPD files, PPD files + * created from driver information files, and dynamically generated PPD files + * using driver helper programs. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Scan for drivers and return an IPP response. + * add_ppd() - Add a PPD file. + * cat_drv() - Generate a PPD from a driver info file. + * cat_ppd() - Copy a PPD file to stdout. + * copy_static() - Copy a static PPD file to stdout. + * compare_inodes() - Compare two inodes. + * compare_matches() - Compare PPD match scores for sorting. + * compare_names() - Compare PPD filenames for sorting. + * compare_ppds() - Compare PPD file make and model names for sorting. + * dump_ppds_dat() - Dump the contents of the ppds.dat file. + * free_array() - Free an array of strings. + * list_ppds() - List PPD files. + * load_drv() - Load the PPDs from a driver information file. + * load_drivers() - Load driver-generated PPD files. + * load_ppds() - Load PPD files recursively. + * load_ppds_dat() - Load the ppds.dat file. + * regex_device_id() - Compile a regular expression based on the 1284 device + * ID. + * regex_string() - Construct a regular expression to compare a simple + * string. + */ + +/* + * Include necessary headers... + */ + +#include "util.h" +#include +#include +#include +#include +#include + + +/* + * Constants... + */ + +#define PPD_SYNC 0x50504437 /* Sync word for ppds.dat (PPD7) */ +#define PPD_MAX_LANG 32 /* Maximum languages */ +#define PPD_MAX_PROD 32 /* Maximum products */ +#define PPD_MAX_VERS 32 /* Maximum versions */ + +#define PPD_TYPE_POSTSCRIPT 0 /* PostScript PPD */ +#define PPD_TYPE_PDF 1 /* PDF PPD */ +#define PPD_TYPE_RASTER 2 /* CUPS raster PPD */ +#define PPD_TYPE_FAX 3 /* Facsimile/MFD PPD */ +#define PPD_TYPE_UNKNOWN 4 /* Other/hybrid PPD */ +#define PPD_TYPE_DRV 5 /* Driver info file */ + +static const char * const ppd_types[] = /* ppd-type values */ +{ + "postscript", + "pdf", + "raster", + "fax", + "unknown", + "drv" +}; + + +/* + * PPD information structures... + */ + +typedef struct /**** PPD record ****/ +{ + time_t mtime; /* Modification time */ + off_t size; /* Size in bytes */ + int model_number; /* cupsModelNumber */ + int type; /* ppd-type */ + char filename[512], /* Filename */ + name[512], /* PPD name */ + languages[PPD_MAX_LANG][6], + /* LanguageVersion/cupsLanguages */ + products[PPD_MAX_PROD][128], + /* Product strings */ + psversions[PPD_MAX_VERS][32], + /* PSVersion strings */ + make[128], /* Manufacturer */ + make_and_model[128], /* NickName/ModelName */ + device_id[256], /* IEEE 1284 Device ID */ + scheme[128]; /* PPD scheme */ +} ppd_rec_t; + +typedef struct /**** In-memory record ****/ +{ + int found; /* 1 if PPD is found */ + int matches; /* Match count */ + ppd_rec_t record; /* PPDs.dat record */ +} ppd_info_t; + + +/* + * Globals... + */ + +cups_array_t *Inodes = NULL, /* Inodes of directories we've visited */ + *PPDsByName = NULL, /* PPD files sorted by filename and name */ + *PPDsByMakeModel = NULL;/* PPD files sorted by make and model */ +int ChangedPPD; /* Did we change the PPD database? */ + + +/* + * Local functions... + */ + +static ppd_info_t *add_ppd(const char *filename, const char *name, + const char *language, const char *make, + const char *make_and_model, + const char *device_id, const char *product, + const char *psversion, time_t mtime, + size_t size, int model_number, int type, + const char *scheme); +static int cat_drv(const char *name, int request_id); +static int cat_ppd(const char *name, int request_id); +static int cat_static(const char *name, int request_id); +static int compare_inodes(struct stat *a, struct stat *b); +static int compare_matches(const ppd_info_t *p0, + const ppd_info_t *p1); +static int compare_names(const ppd_info_t *p0, + const ppd_info_t *p1); +static int compare_ppds(const ppd_info_t *p0, + const ppd_info_t *p1); +static int dump_ppds_dat(const char *filename); +static void free_array(cups_array_t *a); +static int list_ppds(int request_id, int limit, const char *opt); +static int load_drivers(cups_array_t *include, + cups_array_t *exclude); +static int load_drv(const char *filename, const char *name, + cups_file_t *fp, time_t mtime, off_t size); +static int load_ppds(const char *d, const char *p, int descend); +static void load_ppds_dat(char *filename, size_t filesize, + int verbose); +static regex_t *regex_device_id(const char *device_id); +static regex_t *regex_string(const char *s); + + +/* + * 'main()' - Scan for drivers and return an IPP response. + * + * Usage: + * + * cups-driverd request_id limit options + */ + +int /* O - Exit code */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + /* + * Install or list PPDs... + */ + + if (argc == 3 && !strcmp(argv[1], "cat")) + return (cat_ppd(argv[2], 0)); + else if ((argc == 2 || argc == 3) && !strcmp(argv[1], "dump")) + return (dump_ppds_dat(argv[2])); + else if (argc == 4 && !strcmp(argv[1], "get")) + return (cat_ppd(argv[3], atoi(argv[2]))); + else if (argc == 5 && !strcmp(argv[1], "list")) + return (list_ppds(atoi(argv[2]), atoi(argv[3]), argv[4])); + else + { + fputs("Usage: cups-driverd cat ppd-name\n", stderr); + fputs("Usage: cups-driverd dump\n", stderr); + fputs("Usage: cups-driverd get request_id ppd-name\n", stderr); + fputs("Usage: cups-driverd list request_id limit options\n", stderr); + return (1); + } +} + + +/* + * 'add_ppd()' - Add a PPD file. + */ + +static ppd_info_t * /* O - PPD */ +add_ppd(const char *filename, /* I - PPD filename */ + const char *name, /* I - PPD name */ + const char *language, /* I - LanguageVersion */ + const char *make, /* I - Manufacturer */ + const char *make_and_model, /* I - NickName/ModelName */ + const char *device_id, /* I - 1284DeviceID */ + const char *product, /* I - Product */ + const char *psversion, /* I - PSVersion */ + time_t mtime, /* I - Modification time */ + size_t size, /* I - File size */ + int model_number, /* I - Model number */ + int type, /* I - Driver type */ + const char *scheme) /* I - PPD scheme */ +{ + ppd_info_t *ppd; /* PPD */ + char *recommended; /* Foomatic driver string */ + + + /* + * Add a new PPD file... + */ + + if ((ppd = (ppd_info_t *)calloc(1, sizeof(ppd_info_t))) == NULL) + { + fprintf(stderr, + "ERROR: [cups-driverd] Ran out of memory for %d PPD files!\n", + cupsArrayCount(PPDsByName)); + return (NULL); + } + + /* + * Zero-out the PPD data and copy the values over... + */ + + ppd->found = 1; + ppd->record.mtime = mtime; + ppd->record.size = size; + ppd->record.model_number = model_number; + ppd->record.type = type; + + strlcpy(ppd->record.filename, filename, sizeof(ppd->record.filename)); + strlcpy(ppd->record.name, name, sizeof(ppd->record.name)); + strlcpy(ppd->record.languages[0], language, + sizeof(ppd->record.languages[0])); + strlcpy(ppd->record.products[0], product, sizeof(ppd->record.products[0])); + strlcpy(ppd->record.psversions[0], psversion, + sizeof(ppd->record.psversions[0])); + strlcpy(ppd->record.make, make, sizeof(ppd->record.make)); + strlcpy(ppd->record.make_and_model, make_and_model, + sizeof(ppd->record.make_and_model)); + strlcpy(ppd->record.device_id, device_id, sizeof(ppd->record.device_id)); + strlcpy(ppd->record.scheme, scheme, sizeof(ppd->record.scheme)); + + /* + * Strip confusing (and often wrong) "recommended" suffix added by + * Foomatic drivers... + */ + + if ((recommended = strstr(ppd->record.make_and_model, + " (recommended)")) != NULL) + *recommended = '\0'; + + /* + * Add the PPD to the PPD arrays... + */ + + cupsArrayAdd(PPDsByName, ppd); + cupsArrayAdd(PPDsByMakeModel, ppd); + + /* + * Return the new PPD pointer... + */ + + return (ppd); +} + + +/* + * 'cat_drv()' - Generate a PPD from a driver info file. + */ + +static int /* O - Exit code */ +cat_drv(const char *name, /* I - PPD name */ + int request_id) /* I - Request ID for response? */ +{ + const char *datadir; // CUPS_DATADIR env var + ppdcSource *src; // PPD source file data + ppdcDriver *d; // Current driver + cups_file_t *out; // Stdout via CUPS file API + char message[2048], // status-message + filename[1024], // Full path to .drv file(s) + scheme[32], // URI scheme ("drv") + userpass[256], // User/password info (unused) + host[2], // Hostname (unused) + resource[1024], // Resource path (/dir/to/filename.drv) + *pc_file_name; // Filename portion of URI + int port; // Port number (unused) + + + // Determine where CUPS has installed the data files... + if ((datadir = getenv("CUPS_DATADIR")) == NULL) + datadir = CUPS_DATADIR; + + // Pull out the path to the .drv file... + if (httpSeparateURI(HTTP_URI_CODING_ALL, name, scheme, sizeof(scheme), + userpass, sizeof(userpass), host, sizeof(host), &port, + resource, sizeof(resource)) < HTTP_URI_OK || + strstr(resource, "../") || + (pc_file_name = strrchr(resource, '/')) == NULL || + pc_file_name == resource) + { + fprintf(stderr, "ERROR: Bad PPD name \"%s\"!\n", name); + + if (request_id) + { + snprintf(message, sizeof(message), "Bad PPD name \"%s\"!", name); + + cupsdSendIPPHeader(IPP_NOT_FOUND, request_id); + cupsdSendIPPGroup(IPP_TAG_OPERATION); + cupsdSendIPPString(IPP_TAG_CHARSET, "attributes-charset", "utf-8"); + cupsdSendIPPString(IPP_TAG_LANGUAGE, "attributes-natural-language", + "en-US"); + cupsdSendIPPString(IPP_TAG_TEXT, "status-message", message); + cupsdSendIPPTrailer(); + } + + return (1); + } + + *pc_file_name++ = '\0'; + +#ifdef __APPLE__ + if (!strncmp(resource, "/Library/Printers/PPDs/Contents/Resources/", 42) || + !strncmp(resource, "/System/Library/Printers/PPDs/Contents/Resources/", 49)) + strlcpy(filename, resource, sizeof(filename)); + else +#endif // __APPLE__ + { + snprintf(filename, sizeof(filename), "%s/drv%s", datadir, resource); + if (access(filename, 0)) + snprintf(filename, sizeof(filename), "%s/model%s", datadir, resource); + } + + src = new ppdcSource(filename); + + for (d = (ppdcDriver *)src->drivers->first(); + d; + d = (ppdcDriver *)src->drivers->next()) + if (!strcmp(pc_file_name, d->pc_file_name->value) || + (d->file_name && !strcmp(pc_file_name, d->file_name->value))) + break; + + if (d) + { + ppdcArray *locales; // Locale names + ppdcCatalog *catalog; // Message catalog in .drv file + + + fprintf(stderr, "DEBUG2: [cups-driverd] %d locales defined in \"%s\"...\n", + src->po_files->count, filename); + + locales = new ppdcArray(); + for (catalog = (ppdcCatalog *)src->po_files->first(); + catalog; + catalog = (ppdcCatalog *)src->po_files->next()) + { + fprintf(stderr, "DEBUG2: [cups-driverd] Adding locale \"%s\"...\n", + catalog->locale->value); + catalog->locale->retain(); + locales->add(catalog->locale); + } + + if (request_id) + { + cupsdSendIPPHeader(IPP_OK, request_id); + cupsdSendIPPGroup(IPP_TAG_OPERATION); + cupsdSendIPPString(IPP_TAG_CHARSET, "attributes-charset", "utf-8"); + cupsdSendIPPString(IPP_TAG_LANGUAGE, "attributes-natural-language", + "en-US"); + cupsdSendIPPTrailer(); + fflush(stdout); + } + + out = cupsFileStdout(); + d->write_ppd_file(out, NULL, locales, src, PPDC_LFONLY); + cupsFileClose(out); + + locales->release(); + } + else + { + fprintf(stderr, "ERROR: PPD \"%s\" not found!\n", name); + + if (request_id) + { + snprintf(message, sizeof(message), "PPD \"%s\" not found!", name); + + cupsdSendIPPHeader(IPP_NOT_FOUND, request_id); + cupsdSendIPPGroup(IPP_TAG_OPERATION); + cupsdSendIPPString(IPP_TAG_CHARSET, "attributes-charset", "utf-8"); + cupsdSendIPPString(IPP_TAG_LANGUAGE, "attributes-natural-language", + "en-US"); + cupsdSendIPPString(IPP_TAG_TEXT, "status-message", message); + cupsdSendIPPTrailer(); + } + } + + src->release(); + + return (!d); +} + + +/* + * 'cat_ppd()' - Copy a PPD file to stdout. + */ + +static int /* O - Exit code */ +cat_ppd(const char *name, /* I - PPD name */ + int request_id) /* I - Request ID for response? */ +{ + char scheme[256], /* Scheme from PPD name */ + *sptr, /* Pointer into scheme */ + line[1024], /* Line/filename */ + message[2048]; /* status-message */ + + + /* + * Figure out if this is a static or dynamic PPD file... + */ + + strlcpy(scheme, name, sizeof(scheme)); + if ((sptr = strchr(scheme, ':')) != NULL) + { + *sptr = '\0'; + + if (!strcmp(scheme, "file")) + { + /* + * "file:name" == "name"... + */ + + name += 5; + scheme[0] = '\0'; + } + } + else + scheme[0] = '\0'; + + if (request_id > 0) + puts("Content-Type: application/ipp\n"); + + if (!scheme[0]) + return (cat_static(name, request_id)); + else if (!strcmp(scheme, "drv")) + return (cat_drv(name, request_id)); + else + { + /* + * Dynamic PPD, see if we have a driver program to support it... + */ + + const char *serverbin; /* CUPS_SERVERBIN env var */ + char *argv[4]; /* Arguments for program */ + + + if ((serverbin = getenv("CUPS_SERVERBIN")) == NULL) + serverbin = CUPS_SERVERBIN; + + snprintf(line, sizeof(line), "%s/driver/%s", serverbin, scheme); + if (access(line, X_OK)) + { + /* + * File does not exist or is not executable... + */ + + fprintf(stderr, "ERROR: [cups-driverd] Unable to access \"%s\" - %s\n", + line, strerror(errno)); + + if (request_id > 0) + { + snprintf(message, sizeof(message), "Unable to access \"%s\" - %s", + line, strerror(errno)); + + cupsdSendIPPHeader(IPP_NOT_FOUND, request_id); + cupsdSendIPPGroup(IPP_TAG_OPERATION); + cupsdSendIPPString(IPP_TAG_CHARSET, "attributes-charset", "utf-8"); + cupsdSendIPPString(IPP_TAG_LANGUAGE, "attributes-natural-language", + "en-US"); + cupsdSendIPPString(IPP_TAG_TEXT, "status-message", message); + cupsdSendIPPTrailer(); + } + + return (1); + } + + /* + * Yes, let it cat the PPD file... + */ + + if (request_id) + { + cupsdSendIPPHeader(IPP_OK, request_id); + cupsdSendIPPGroup(IPP_TAG_OPERATION); + cupsdSendIPPString(IPP_TAG_CHARSET, "attributes-charset", "utf-8"); + cupsdSendIPPString(IPP_TAG_LANGUAGE, "attributes-natural-language", + "en-US"); + cupsdSendIPPTrailer(); + } + + argv[0] = scheme; + argv[1] = (char *)"cat"; + argv[2] = (char *)name; + argv[3] = NULL; + + if (cupsdExec(line, argv)) + { + /* + * Unable to execute driver... + */ + + fprintf(stderr, "ERROR: [cups-driverd] Unable to execute \"%s\" - %s\n", + line, strerror(errno)); + return (1); + } + } + + /* + * Return with no errors... + */ + + return (0); +} + + +/* + * 'copy_static()' - Copy a static PPD file to stdout. + */ + +static int /* O - Exit code */ +cat_static(const char *name, /* I - PPD name */ + int request_id) /* I - Request ID for response? */ +{ + cups_file_t *fp; /* PPD file */ + const char *datadir; /* CUPS_DATADIR env var */ + char line[1024], /* Line/filename */ + message[2048]; /* status-message */ +#ifdef __APPLE__ + const char *printerDriver, /* Pointer to .printerDriver extension */ + *slash; /* Pointer to next slash */ +#endif /* __APPLE__ */ + + + if (name[0] == '/' || strstr(name, "../") || strstr(name, "/..")) + { + /* + * Bad name... + */ + + fprintf(stderr, "ERROR: [cups-driverd] Bad PPD name \"%s\"!\n", name); + + if (request_id) + { + snprintf(message, sizeof(message), "Bad PPD name \"%s\"!", name); + + cupsdSendIPPHeader(IPP_NOT_FOUND, request_id); + cupsdSendIPPGroup(IPP_TAG_OPERATION); + cupsdSendIPPString(IPP_TAG_CHARSET, "attributes-charset", "utf-8"); + cupsdSendIPPString(IPP_TAG_LANGUAGE, "attributes-natural-language", + "en-US"); + cupsdSendIPPString(IPP_TAG_TEXT, "status-message", message); + cupsdSendIPPTrailer(); + } + + return (1); + } + + /* + * Try opening the file... + */ + +#ifdef __APPLE__ + if (!strncmp(name, "System/Library/Printers/PPDs/Contents/Resources/", 48) || + !strncmp(name, "Library/Printers/PPDs/Contents/Resources/", 41) || + (!strncmp(name, "System/Library/Printers/", 24) && + (printerDriver = + strstr(name + 24, + ".printerDriver/Contents/Resources/PPDs")) != NULL && + (slash = strchr(name + 24, '/')) != NULL && + slash > printerDriver) || + (!strncmp(name, "Library/Printers/", 17) && + (printerDriver = + strstr(name + 17, + ".printerDriver/Contents/Resources/PPDs")) != NULL && + (slash = strchr(name + 17, '/')) != NULL && + slash > printerDriver)) + { + /* + * Map ppd-name to Mac OS X standard locations... + */ + + snprintf(line, sizeof(line), "/%s", name); + } + else + +#elif defined(__linux) + if (!strncmp(name, "lsb/usr/", 8)) + { + /* + * Map ppd-name to LSB standard /usr/share/ppd location... + */ + + snprintf(line, sizeof(line), "/usr/share/ppd/%s", name + 8); + } + else if (!strncmp(name, "lsb/opt/", 8)) + { + /* + * Map ppd-name to LSB standard /opt/share/ppd location... + */ + + snprintf(line, sizeof(line), "/opt/share/ppd/%s", name + 8); + } + else if (!strncmp(name, "lsb/local/", 10)) + { + /* + * Map ppd-name to LSB standard /usr/local/share/ppd location... + */ + + snprintf(line, sizeof(line), "/usr/local/share/ppd/%s", name + 10); + } + else + +#endif /* __APPLE__ */ + { + if ((datadir = getenv("CUPS_DATADIR")) == NULL) + datadir = CUPS_DATADIR; + + snprintf(line, sizeof(line), "%s/model/%s", datadir, name); + } + + if ((fp = cupsFileOpen(line, "r")) == NULL) + { + fprintf(stderr, "ERROR: [cups-driverd] Unable to open \"%s\" - %s\n", + line, strerror(errno)); + + if (request_id) + { + snprintf(message, sizeof(message), "Unable to open \"%s\" - %s", + line, strerror(errno)); + + cupsdSendIPPHeader(IPP_NOT_FOUND, request_id); + cupsdSendIPPGroup(IPP_TAG_OPERATION); + cupsdSendIPPString(IPP_TAG_CHARSET, "attributes-charset", "utf-8"); + cupsdSendIPPString(IPP_TAG_LANGUAGE, "attributes-natural-language", + "en-US"); + cupsdSendIPPString(IPP_TAG_TEXT, "status-message", message); + cupsdSendIPPTrailer(); + } + + return (1); + } + + if (request_id) + { + cupsdSendIPPHeader(IPP_OK, request_id); + cupsdSendIPPGroup(IPP_TAG_OPERATION); + cupsdSendIPPString(IPP_TAG_CHARSET, "attributes-charset", "utf-8"); + cupsdSendIPPString(IPP_TAG_LANGUAGE, "attributes-natural-language", + "en-US"); + cupsdSendIPPTrailer(); + } + + /* + * Now copy the file to stdout... + */ + + while (cupsFileGets(fp, line, sizeof(line))) + puts(line); + + cupsFileClose(fp); + + return (0); +} + + +/* + * 'compare_inodes()' - Compare two inodes. + */ + +static int /* O - Result of comparison */ +compare_inodes(struct stat *a, /* I - First inode */ + struct stat *b) /* I - Second inode */ +{ + if (a->st_dev != b->st_dev) + return (a->st_dev - b->st_dev); + else + return (a->st_ino - b->st_ino); +} + + +/* + * 'compare_matches()' - Compare PPD match scores for sorting. + */ + +static int +compare_matches(const ppd_info_t *p0, /* I - First PPD */ + const ppd_info_t *p1) /* I - Second PPD */ +{ + if (p1->matches != p0->matches) + return (p1->matches - p0->matches); + else + return (cupsdCompareNames(p0->record.make_and_model, + p1->record.make_and_model)); +} + + +/* + * 'compare_names()' - Compare PPD filenames for sorting. + */ + +static int /* O - Result of comparison */ +compare_names(const ppd_info_t *p0, /* I - First PPD file */ + const ppd_info_t *p1) /* I - Second PPD file */ +{ + int diff; /* Difference between strings */ + + + if ((diff = strcmp(p0->record.filename, p1->record.filename)) != 0) + return (diff); + else + return (strcmp(p0->record.name, p1->record.name)); +} + + +/* + * 'compare_ppds()' - Compare PPD file make and model names for sorting. + */ + +static int /* O - Result of comparison */ +compare_ppds(const ppd_info_t *p0, /* I - First PPD file */ + const ppd_info_t *p1) /* I - Second PPD file */ +{ + int diff; /* Difference between strings */ + + + /* + * First compare manufacturers... + */ + + if ((diff = _cups_strcasecmp(p0->record.make, p1->record.make)) != 0) + return (diff); + else if ((diff = cupsdCompareNames(p0->record.make_and_model, + p1->record.make_and_model)) != 0) + return (diff); + else if ((diff = strcmp(p0->record.languages[0], + p1->record.languages[0])) != 0) + return (diff); + else + return (compare_names(p0, p1)); +} + + +/* + * 'dump_ppds_dat()' - Dump the contents of the ppds.dat file. + */ + +static int /* O - Exit status */ +dump_ppds_dat(const char *filename) /* I - Filename */ +{ + char temp[1024]; /* ppds.dat filename */ + ppd_info_t *ppd; /* Current PPD */ + + + /* + * See if we a PPD database file... + */ + + if (filename) + strlcpy(temp, filename, sizeof(temp)); + else + temp[0] = '\0'; + + load_ppds_dat(temp, sizeof(temp), 0); + + puts("mtime,size,model_number,type,filename,name,languages0,products0," + "psversions0,make,make_and_model,device_id,scheme"); + for (ppd = (ppd_info_t *)cupsArrayFirst(PPDsByName); + ppd; + ppd = (ppd_info_t *)cupsArrayNext(PPDsByName)) + printf("%d,%ld,%d,%d,\"%s\",\"%s\",\"%s\",\"%s\",\"%s\",\"%s\",\"%s\"," + "\"%s\",\"%s\"\n", + (int)ppd->record.mtime, (long)ppd->record.size, + ppd->record.model_number, ppd->record.type, ppd->record.filename, + ppd->record.name, ppd->record.languages[0], ppd->record.products[0], + ppd->record.psversions[0], ppd->record.make, + ppd->record.make_and_model, ppd->record.device_id, + ppd->record.scheme); + + return (0); +} + + +/* + * 'free_array()' - Free an array of strings. + */ + +static void +free_array(cups_array_t *a) /* I - Array to free */ +{ + char *ptr; /* Pointer to string */ + + + for (ptr = (char *)cupsArrayFirst(a); + ptr; + ptr = (char *)cupsArrayNext(a)) + free(ptr); + + cupsArrayDelete(a); +} + + +/* + * 'list_ppds()' - List PPD files. + */ + +static int /* O - Exit code */ +list_ppds(int request_id, /* I - Request ID */ + int limit, /* I - Limit */ + const char *opt) /* I - Option argument */ +{ + int i; /* Looping vars */ + int count; /* Number of PPDs to send */ + ppd_info_t *ppd; /* Current PPD file */ + cups_file_t *fp; /* ppds.dat file */ + char filename[1024], /* ppds.dat filename */ + model[1024]; /* Model directory */ + const char *cups_datadir; /* CUPS_DATADIR environment variable */ + int num_options; /* Number of options */ + cups_option_t *options; /* Options */ + cups_array_t *requested, /* requested-attributes values */ + *include, /* PPD schemes to include */ + *exclude; /* PPD schemes to exclude */ + const char *device_id, /* ppd-device-id option */ + *language, /* ppd-natural-language option */ + *make, /* ppd-make option */ + *make_and_model, /* ppd-make-and-model option */ + *model_number_str, /* ppd-model-number option */ + *product, /* ppd-product option */ + *psversion, /* ppd-psversion option */ + *type_str; /* ppd-type option */ + int model_number, /* ppd-model-number value */ + type, /* ppd-type value */ + make_and_model_len, /* Length of ppd-make-and-model */ + product_len, /* Length of ppd-product */ + send_device_id, /* Send ppd-device-id? */ + send_make, /* Send ppd-make? */ + send_make_and_model, /* Send ppd-make-and-model? */ + send_model_number, /* Send ppd-model-number? */ + send_name, /* Send ppd-name? */ + send_natural_language, /* Send ppd-natural-language? */ + send_product, /* Send ppd-product? */ + send_psversion, /* Send ppd-psversion? */ + send_type, /* Send ppd-type? */ + sent_header; /* Sent the IPP header? */ + regex_t *device_id_re, /* Regular expression for matching device ID */ + *make_and_model_re; /* Regular expression for matching make and model */ + regmatch_t re_matches[6]; /* Regular expression matches */ + cups_array_t *matches; /* Matching PPDs */ + + + fprintf(stderr, + "DEBUG2: [cups-driverd] list_ppds(request_id=%d, limit=%d, " + "opt=\"%s\"\n", request_id, limit, opt); + + /* + * See if we a PPD database file... + */ + + filename[0] = '\0'; + load_ppds_dat(filename, sizeof(filename), 1); + + /* + * Load all PPDs in the specified directory and below... + */ + + if ((cups_datadir = getenv("CUPS_DATADIR")) == NULL) + cups_datadir = CUPS_DATADIR; + + Inodes = cupsArrayNew((cups_array_func_t)compare_inodes, NULL); + + snprintf(model, sizeof(model), "%s/model", cups_datadir); + load_ppds(model, "", 1); + + snprintf(model, sizeof(model), "%s/drv", cups_datadir); + load_ppds(model, "", 1); + +#ifdef __APPLE__ + /* + * Load PPDs from standard Mac OS X locations... + */ + + load_ppds("/Library/Printers", + "Library/Printers", 0); + load_ppds("/Library/Printers/PPDs/Contents/Resources", + "Library/Printers/PPDs/Contents/Resources", 0); + load_ppds("/Library/Printers/PPDs/Contents/Resources/en.lproj", + "Library/Printers/PPDs/Contents/Resources/en.lproj", 0); + load_ppds("/System/Library/Printers", + "System/Library/Printers", 0); + load_ppds("/System/Library/Printers/PPDs/Contents/Resources", + "System/Library/Printers/PPDs/Contents/Resources", 0); + load_ppds("/System/Library/Printers/PPDs/Contents/Resources/en.lproj", + "System/Library/Printers/PPDs/Contents/Resources/en.lproj", 0); + +#elif defined(__linux) + /* + * Load PPDs from LSB-defined locations... + */ + + if (!access("/usr/local/share/ppd", 0)) + load_ppds("/usr/local/share/ppd", "lsb/local", 1); + if (!access("/usr/share/ppd", 0)) + load_ppds("/usr/share/ppd", "lsb/usr", 1); + if (!access("/opt/share/ppd", 0)) + load_ppds("/opt/share/ppd", "lsb/opt", 1); +#endif /* __APPLE__ */ + + /* + * Cull PPD files that are no longer present... + */ + + for (ppd = (ppd_info_t *)cupsArrayFirst(PPDsByName); + ppd; + ppd = (ppd_info_t *)cupsArrayNext(PPDsByName)) + if (!ppd->found) + { + /* + * Remove this PPD file from the list... + */ + + cupsArrayRemove(PPDsByName, ppd); + cupsArrayRemove(PPDsByMakeModel, ppd); + free(ppd); + + ChangedPPD = 1; + } + + /* + * Write the new ppds.dat file... + */ + + fprintf(stderr, "DEBUG: [cups-driverd] ChangedPPD=%d\n", ChangedPPD); + + if (ChangedPPD) + { + char newname[1024]; /* New filename */ + + snprintf(newname, sizeof(newname), "%s.%d", filename, (int)getpid()); + + if ((fp = cupsFileOpen(newname, "w")) != NULL) + { + unsigned ppdsync = PPD_SYNC; /* Sync word */ + + cupsFileWrite(fp, (char *)&ppdsync, sizeof(ppdsync)); + + for (ppd = (ppd_info_t *)cupsArrayFirst(PPDsByName); + ppd; + ppd = (ppd_info_t *)cupsArrayNext(PPDsByName)) + cupsFileWrite(fp, (char *)&(ppd->record), sizeof(ppd_rec_t)); + + cupsFileClose(fp); + + if (rename(newname, filename)) + fprintf(stderr, "ERROR: [cups-driverd] Unable to rename \"%s\" - %s\n", + newname, strerror(errno)); + else + fprintf(stderr, "INFO: [cups-driverd] Wrote \"%s\", %d PPDs...\n", + filename, cupsArrayCount(PPDsByName)); + } + else + fprintf(stderr, "ERROR: [cups-driverd] Unable to write \"%s\" - %s\n", + filename, strerror(errno)); + } + else + fputs("INFO: [cups-driverd] No new or changed PPDs...\n", stderr); + + /* + * Scan for dynamic PPD files... + */ + + num_options = cupsParseOptions(opt, 0, &options); + exclude = cupsdCreateStringsArray(cupsGetOption("exclude-schemes", + num_options, options)); + include = cupsdCreateStringsArray(cupsGetOption("include-schemes", + num_options, options)); + + load_drivers(include, exclude); + + /* + * Add the raw driver... + */ + + add_ppd("", "raw", "en", "Raw", "Raw Queue", "", "", "", 0, 0, 0, + PPD_TYPE_UNKNOWN, "raw"); + + /* + * Send IPP attributes... + */ + + requested = cupsdCreateStringsArray( + cupsGetOption("requested-attributes", num_options, + options)); + device_id = cupsGetOption("ppd-device-id", num_options, options); + language = cupsGetOption("ppd-natural-language", num_options, options); + make = cupsGetOption("ppd-make", num_options, options); + make_and_model = cupsGetOption("ppd-make-and-model", num_options, options); + model_number_str = cupsGetOption("ppd-model-number", num_options, options); + product = cupsGetOption("ppd-product", num_options, options); + psversion = cupsGetOption("ppd-psversion", num_options, options); + type_str = cupsGetOption("ppd-type", num_options, options); + + if (make_and_model) + make_and_model_len = strlen(make_and_model); + else + make_and_model_len = 0; + + if (product) + product_len = strlen(product); + else + product_len = 0; + + if (model_number_str) + model_number = atoi(model_number_str); + else + model_number = 0; + + if (type_str) + { + for (type = 0; + type < (int)(sizeof(ppd_types) / sizeof(ppd_types[0])); + type ++) + if (!strcmp(type_str, ppd_types[type])) + break; + + if (type >= (int)(sizeof(ppd_types) / sizeof(ppd_types[0]))) + { + fprintf(stderr, "ERROR: [cups-driverd] Bad ppd-type=\"%s\" ignored!\n", + type_str); + type_str = NULL; + } + } + else + type = 0; + + for (i = 0; i < num_options; i ++) + fprintf(stderr, "DEBUG2: [cups-driverd] %s=\"%s\"\n", options[i].name, + options[i].value); + + if (!requested || cupsArrayFind(requested, (void *)"all") != NULL) + { + send_name = 1; + send_make = 1; + send_make_and_model = 1; + send_model_number = 1; + send_natural_language = 1; + send_device_id = 1; + send_product = 1; + send_psversion = 1; + send_type = 1; + } + else + { + send_name = cupsArrayFind(requested, + (void *)"ppd-name") != NULL; + send_make = cupsArrayFind(requested, + (void *)"ppd-make") != NULL; + send_make_and_model = cupsArrayFind(requested, + (void *)"ppd-make-and-model") != NULL; + send_model_number = cupsArrayFind(requested, + (void *)"ppd-model-number") != NULL; + send_natural_language = cupsArrayFind(requested, + (void *)"ppd-natural-language") != NULL; + send_device_id = cupsArrayFind(requested, + (void *)"ppd-device-id") != NULL; + send_product = cupsArrayFind(requested, + (void *)"ppd-product") != NULL; + send_psversion = cupsArrayFind(requested, + (void *)"ppd-psversion") != NULL; + send_type = cupsArrayFind(requested, + (void *)"ppd-type") != NULL; + } + + /* + * Send the content type header to the scheduler; request_id can only be + * 0 when run manually since the scheduler enforces the IPP requirement for + * a request ID from 1 to 2^31-1... + */ + + if (request_id > 0) + puts("Content-Type: application/ipp\n"); + + sent_header = 0; + + if (limit <= 0 || limit > cupsArrayCount(PPDsByMakeModel)) + count = cupsArrayCount(PPDsByMakeModel); + else + count = limit; + + if (device_id || language || make || make_and_model || model_number_str || + product) + { + matches = cupsArrayNew((cups_array_func_t)compare_matches, NULL); + + if (device_id) + device_id_re = regex_device_id(device_id); + else + device_id_re = NULL; + + if (make_and_model) + make_and_model_re = regex_string(make_and_model); + else + make_and_model_re = NULL; + + for (ppd = (ppd_info_t *)cupsArrayFirst(PPDsByMakeModel); + ppd; + ppd = (ppd_info_t *)cupsArrayNext(PPDsByMakeModel)) + { + /* + * Filter PPDs based on make, model, product, language, model number, + * and/or device ID using the "matches" score value. An exact match + * for product, make-and-model, or device-id adds 3 to the score. + * Partial matches for make-and-model yield 1 or 2 points, and matches + * for the make and language add a single point. Results are then sorted + * by score, highest score first. + */ + + if (ppd->record.type < PPD_TYPE_POSTSCRIPT || + ppd->record.type >= PPD_TYPE_DRV) + continue; + + if (cupsArrayFind(exclude, ppd->record.scheme) || + (include && !cupsArrayFind(include, ppd->record.scheme))) + continue; + + ppd->matches = 0; + + if (device_id_re && + !regexec(device_id_re, ppd->record.device_id, + (int)(sizeof(re_matches) / sizeof(re_matches[0])), + re_matches, 0)) + { + /* + * Add the number of matching values from the device ID - it will be + * at least 2 (manufacturer and model), and as much as 3 (command set). + */ + + for (i = 1; i < (int)(sizeof(re_matches) / sizeof(re_matches[0])); i ++) + if (re_matches[i].rm_so >= 0) + ppd->matches ++; + } + + if (language) + { + for (i = 0; i < PPD_MAX_LANG; i ++) + if (!ppd->record.languages[i][0]) + break; + else if (!strcmp(ppd->record.languages[i], language)) + { + ppd->matches ++; + break; + } + } + + if (make && !_cups_strcasecmp(ppd->record.make, make)) + ppd->matches ++; + + if (make_and_model_re && + !regexec(make_and_model_re, ppd->record.make_and_model, + (int)(sizeof(re_matches) / sizeof(re_matches[0])), + re_matches, 0)) + { + // See how much of the make-and-model string we matched... + if (re_matches[0].rm_so == 0) + { + if (re_matches[0].rm_eo == make_and_model_len) + ppd->matches += 3; // Exact match + else + ppd->matches += 2; // Prefix match + } + else + ppd->matches ++; // Infix match + } + + if (model_number_str && ppd->record.model_number == model_number) + ppd->matches ++; + + if (product) + { + for (i = 0; i < PPD_MAX_PROD; i ++) + if (!ppd->record.products[i][0]) + break; + else if (!_cups_strcasecmp(ppd->record.products[i], product)) + { + ppd->matches += 3; + break; + } + else if (!_cups_strncasecmp(ppd->record.products[i], product, + product_len)) + { + ppd->matches += 2; + break; + } + } + + if (psversion) + { + for (i = 0; i < PPD_MAX_VERS; i ++) + if (!ppd->record.psversions[i][0]) + break; + else if (!_cups_strcasecmp(ppd->record.psversions[i], psversion)) + { + ppd->matches ++; + break; + } + } + + if (type_str && ppd->record.type == type) + ppd->matches ++; + + if (ppd->matches) + { + fprintf(stderr, "DEBUG2: [cups-driverd] %s matches with score %d!\n", + ppd->record.name, ppd->matches); + cupsArrayAdd(matches, ppd); + } + } + } + else if (include || exclude) + { + matches = cupsArrayNew((cups_array_func_t)compare_ppds, NULL); + + for (ppd = (ppd_info_t *)cupsArrayFirst(PPDsByMakeModel); + ppd; + ppd = (ppd_info_t *)cupsArrayNext(PPDsByMakeModel)) + { + /* + * Filter PPDs based on the include/exclude lists. + */ + + if (ppd->record.type < PPD_TYPE_POSTSCRIPT || + ppd->record.type >= PPD_TYPE_DRV) + continue; + + if (cupsArrayFind(exclude, ppd->record.scheme) || + (include && !cupsArrayFind(include, ppd->record.scheme))) + continue; + + cupsArrayAdd(matches, ppd); + } + } + else + matches = PPDsByMakeModel; + + for (ppd = (ppd_info_t *)cupsArrayFirst(matches); + count > 0 && ppd; + ppd = (ppd_info_t *)cupsArrayNext(matches)) + { + /* + * Skip invalid PPDs... + */ + + if (ppd->record.type < PPD_TYPE_POSTSCRIPT || + ppd->record.type >= PPD_TYPE_DRV) + continue; + + /* + * Send this PPD... + */ + + if (!sent_header) + { + sent_header = 1; + + cupsdSendIPPHeader(IPP_OK, request_id); + cupsdSendIPPGroup(IPP_TAG_OPERATION); + cupsdSendIPPString(IPP_TAG_CHARSET, "attributes-charset", "utf-8"); + cupsdSendIPPString(IPP_TAG_LANGUAGE, "attributes-natural-language", + "en-US"); + } + + fprintf(stderr, "DEBUG2: [cups-driverd] Sending %s (%s)...\n", + ppd->record.name, ppd->record.make_and_model); + + count --; + + cupsdSendIPPGroup(IPP_TAG_PRINTER); + + if (send_name) + cupsdSendIPPString(IPP_TAG_NAME, "ppd-name", ppd->record.name); + + if (send_natural_language) + { + cupsdSendIPPString(IPP_TAG_LANGUAGE, "ppd-natural-language", + ppd->record.languages[0]); + + for (i = 1; i < PPD_MAX_LANG && ppd->record.languages[i][0]; i ++) + cupsdSendIPPString(IPP_TAG_LANGUAGE, "", ppd->record.languages[i]); + } + + if (send_make) + cupsdSendIPPString(IPP_TAG_TEXT, "ppd-make", ppd->record.make); + + if (send_make_and_model) + cupsdSendIPPString(IPP_TAG_TEXT, "ppd-make-and-model", + ppd->record.make_and_model); + + if (send_device_id) + cupsdSendIPPString(IPP_TAG_TEXT, "ppd-device-id", + ppd->record.device_id); + + if (send_product) + { + cupsdSendIPPString(IPP_TAG_TEXT, "ppd-product", + ppd->record.products[0]); + + for (i = 1; i < PPD_MAX_PROD && ppd->record.products[i][0]; i ++) + cupsdSendIPPString(IPP_TAG_TEXT, "", ppd->record.products[i]); + } + + if (send_psversion) + { + cupsdSendIPPString(IPP_TAG_TEXT, "ppd-psversion", + ppd->record.psversions[0]); + + for (i = 1; i < PPD_MAX_VERS && ppd->record.psversions[i][0]; i ++) + cupsdSendIPPString(IPP_TAG_TEXT, "", ppd->record.psversions[i]); + } + + if (send_type) + cupsdSendIPPString(IPP_TAG_KEYWORD, "ppd-type", + ppd_types[ppd->record.type]); + + if (send_model_number) + cupsdSendIPPInteger(IPP_TAG_INTEGER, "ppd-model-number", + ppd->record.model_number); + + /* + * If we have only requested the ppd-make attribute, then skip + * the remaining PPDs with this make... + */ + + if (cupsArrayFind(requested, (void *)"ppd-make") && + cupsArrayCount(requested) == 1) + { + const char *this_make; /* This ppd-make */ + + + for (this_make = ppd->record.make, + ppd = (ppd_info_t *)cupsArrayNext(matches); + ppd; + ppd = (ppd_info_t *)cupsArrayNext(matches)) + if (_cups_strcasecmp(this_make, ppd->record.make)) + break; + + cupsArrayPrev(matches); + } + } + + if (!sent_header) + { + cupsdSendIPPHeader(IPP_NOT_FOUND, request_id); + cupsdSendIPPGroup(IPP_TAG_OPERATION); + cupsdSendIPPString(IPP_TAG_CHARSET, "attributes-charset", "utf-8"); + cupsdSendIPPString(IPP_TAG_LANGUAGE, "attributes-natural-language", "en-US"); + } + + cupsdSendIPPTrailer(); + + return (0); +} + + +/* + * 'load_drv()' - Load the PPDs from a driver information file. + */ + +static int /* O - 1 on success, 0 on failure */ +load_drv(const char *filename, /* I - Actual filename */ + const char *name, /* I - Name to the rest of the world */ + cups_file_t *fp, /* I - File to read from */ + time_t mtime, /* I - Mod time of driver info file */ + off_t size) /* I - Size of driver info file */ +{ + ppdcSource *src; // Driver information file + ppdcDriver *d; // Current driver + ppdcAttr *device_id, // 1284DeviceID attribute + *product, // Current product value + *ps_version, // PSVersion attribute + *cups_fax, // cupsFax attribute + *nick_name; // NickName attribute + ppdcFilter *filter; // Current filter + ppd_info_t *ppd; // Current PPD + int products_found; // Number of products found + char uri[1024], // Driver URI + make_model[1024]; // Make and model + int type; // Driver type + + + /* + * Load the driver info file... + */ + + src = new ppdcSource(filename, fp); + + if (src->drivers->count == 0) + { + fprintf(stderr, + "ERROR: [cups-driverd] Bad driver information file \"%s\"!\n", + filename); + src->release(); + return (0); + } + + /* + * Add a dummy entry for the file... + */ + + add_ppd(name, name, "", "", "", "", "", "", mtime, size, 0, + PPD_TYPE_DRV, "drv"); + ChangedPPD = 1; + + /* + * Then the drivers in the file... + */ + + for (d = (ppdcDriver *)src->drivers->first(); + d; + d = (ppdcDriver *)src->drivers->next()) + { + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "drv", "", "", 0, + "/%s/%s", name, + d->file_name ? d->file_name->value : + d->pc_file_name->value); + + device_id = d->find_attr("1284DeviceID", NULL); + ps_version = d->find_attr("PSVersion", NULL); + nick_name = d->find_attr("NickName", NULL); + + if (nick_name) + strlcpy(make_model, nick_name->value->value, sizeof(make_model)); + else if (_cups_strncasecmp(d->model_name->value, d->manufacturer->value, + strlen(d->manufacturer->value))) + snprintf(make_model, sizeof(make_model), "%s %s, %s", + d->manufacturer->value, d->model_name->value, + d->version->value); + else + snprintf(make_model, sizeof(make_model), "%s, %s", d->model_name->value, + d->version->value); + + if ((cups_fax = d->find_attr("cupsFax", NULL)) != NULL && + !_cups_strcasecmp(cups_fax->value->value, "true")) + type = PPD_TYPE_FAX; + else if (d->type == PPDC_DRIVER_PS) + type = PPD_TYPE_POSTSCRIPT; + else if (d->type != PPDC_DRIVER_CUSTOM) + type = PPD_TYPE_RASTER; + else + { + for (filter = (ppdcFilter *)d->filters->first(), + type = PPD_TYPE_POSTSCRIPT; + filter; + filter = (ppdcFilter *)d->filters->next()) + if (_cups_strcasecmp(filter->mime_type->value, "application/vnd.cups-raster")) + type = PPD_TYPE_RASTER; + else if (_cups_strcasecmp(filter->mime_type->value, + "application/vnd.cups-pdf")) + type = PPD_TYPE_PDF; + } + + for (product = (ppdcAttr *)d->attrs->first(), products_found = 0, + ppd = NULL; + product; + product = (ppdcAttr *)d->attrs->next()) + if (!strcmp(product->name->value, "Product")) + { + if (!products_found) + ppd = add_ppd(name, uri, "en", d->manufacturer->value, make_model, + device_id ? device_id->value->value : "", + product->value->value, + ps_version ? ps_version->value->value : "(3010) 0", + mtime, size, d->model_number, type, "drv"); + else if (products_found < PPD_MAX_PROD) + strlcpy(ppd->record.products[products_found], product->value->value, + sizeof(ppd->record.products[0])); + else + break; + + products_found ++; + } + + if (!products_found) + add_ppd(name, uri, "en", d->manufacturer->value, make_model, + device_id ? device_id->value->value : "", + d->model_name->value, + ps_version ? ps_version->value->value : "(3010) 0", + mtime, size, d->model_number, type, "drv"); + } + + src->release(); + + return (1); +} + + +/* + * 'load_drivers()' - Load driver-generated PPD files. + */ + +static int /* O - 1 on success, 0 on failure */ +load_drivers(cups_array_t *include, /* I - Drivers to include */ + cups_array_t *exclude) /* I - Drivers to exclude */ +{ + int i; /* Looping var */ + char *start, /* Start of value */ + *ptr; /* Pointer into string */ + const char *server_bin, /* CUPS_SERVERBIN env variable */ + *scheme, /* Scheme for this driver */ + *scheme_end; /* Pointer to end of scheme */ + char drivers[1024]; /* Location of driver programs */ + int pid; /* Process ID for driver program */ + cups_file_t *fp; /* Pipe to driver program */ + cups_dir_t *dir; /* Directory pointer */ + cups_dentry_t *dent; /* Directory entry */ + char *argv[3], /* Arguments for command */ + filename[1024], /* Name of driver */ + line[2048], /* Line from driver */ + name[512], /* ppd-name */ + make[128], /* ppd-make */ + make_and_model[128], /* ppd-make-and-model */ + device_id[256], /* ppd-device-id */ + languages[128], /* ppd-natural-language */ + product[128], /* ppd-product */ + psversion[128], /* ppd-psversion */ + type_str[128]; /* ppd-type */ + int type; /* PPD type */ + ppd_info_t *ppd; /* Newly added PPD */ + + + /* + * Try opening the driver directory... + */ + + if ((server_bin = getenv("CUPS_SERVERBIN")) == NULL) + server_bin = CUPS_SERVERBIN; + + snprintf(drivers, sizeof(drivers), "%s/driver", server_bin); + + if ((dir = cupsDirOpen(drivers)) == NULL) + { + fprintf(stderr, "ERROR: [cups-driverd] Unable to open driver directory " + "\"%s\": %s\n", + drivers, strerror(errno)); + return (0); + } + + /* + * Loop through all of the device drivers... + */ + + argv[1] = (char *)"list"; + argv[2] = NULL; + + while ((dent = cupsDirRead(dir)) != NULL) + { + /* + * Only look at executable files... + */ + + if (!(dent->fileinfo.st_mode & 0111) || !S_ISREG(dent->fileinfo.st_mode)) + continue; + + /* + * Include/exclude specific drivers... + */ + + if (exclude) + { + /* + * Look for "scheme" or "scheme*" (prefix match), and skip any matches. + */ + + for (scheme = (char *)cupsArrayFirst(exclude); + scheme; + scheme = (char *)cupsArrayNext(exclude)) + { + fprintf(stderr, "DEBUG: [cups-driverd] Exclude \"%s\" with \"%s\"?\n", + dent->filename, scheme); + scheme_end = scheme + strlen(scheme) - 1; + + if ((scheme_end > scheme && *scheme_end == '*' && + !strncmp(scheme, dent->filename, scheme_end - scheme)) || + !strcmp(scheme, dent->filename)) + { + fputs("DEBUG: [cups-driverd] Yes, exclude!\n", stderr); + break; + } + } + + if (scheme) + continue; + } + + if (include) + { + /* + * Look for "scheme" or "scheme*" (prefix match), and skip any non-matches. + */ + + for (scheme = (char *)cupsArrayFirst(include); + scheme; + scheme = (char *)cupsArrayNext(include)) + { + fprintf(stderr, "DEBUG: [cups-driverd] Include \"%s\" with \"%s\"?\n", + dent->filename, scheme); + scheme_end = scheme + strlen(scheme) - 1; + + if ((scheme_end > scheme && *scheme_end == '*' && + !strncmp(scheme, dent->filename, scheme_end - scheme)) || + !strcmp(scheme, dent->filename)) + { + fputs("DEBUG: [cups-driverd] Yes, include!\n", stderr); + break; + } + } + + if (!scheme) + continue; + } + else + scheme = dent->filename; + + /* + * Run the driver with no arguments and collect the output... + */ + + snprintf(filename, sizeof(filename), "%s/%s", drivers, dent->filename); + + if (_cupsFileCheck(filename, _CUPS_FILE_CHECK_PROGRAM, !geteuid(), + _cupsFileCheckFilter, NULL)) + continue; + + argv[0] = dent->filename; + + if ((fp = cupsdPipeCommand(&pid, filename, argv, 0)) != NULL) + { + while (cupsFileGets(fp, line, sizeof(line))) + { + /* + * Each line is of the form: + * + * "ppd-name" ppd-natural-language "ppd-make" "ppd-make-and-model" \ + * "ppd-device-id" "ppd-product" "ppd-psversion" + */ + + device_id[0] = '\0'; + product[0] = '\0'; + psversion[0] = '\0'; + strcpy(type_str, "postscript"); + + if (sscanf(line, "\"%511[^\"]\"%127s%*[ \t]\"%127[^\"]\"" + "%*[ \t]\"%127[^\"]\"%*[ \t]\"%255[^\"]\"" + "%*[ \t]\"%127[^\"]\"%*[ \t]\"%127[^\"]\"" + "%*[ \t]\"%127[^\"]\"", + name, languages, make, make_and_model, + device_id, product, psversion, type_str) < 4) + { + /* + * Bad format; strip trailing newline and write an error message. + */ + + if (line[strlen(line) - 1] == '\n') + line[strlen(line) - 1] = '\0'; + + fprintf(stderr, "ERROR: [cups-driverd] Bad line from \"%s\": %s\n", + dent->filename, line); + break; + } + else + { + /* + * Add the device to the array of available devices... + */ + + if ((start = strchr(languages, ',')) != NULL) + *start++ = '\0'; + + for (type = 0; + type < (int)(sizeof(ppd_types) / sizeof(ppd_types[0])); + type ++) + if (!strcmp(type_str, ppd_types[type])) + break; + + if (type >= (int)(sizeof(ppd_types) / sizeof(ppd_types[0]))) + { + fprintf(stderr, + "ERROR: [cups-driverd] Bad ppd-type \"%s\" ignored!\n", + type_str); + type = PPD_TYPE_UNKNOWN; + } + + ppd = add_ppd(filename, name, languages, make, make_and_model, + device_id, product, psversion, 0, 0, 0, type, scheme); + + if (!ppd) + { + cupsDirClose(dir); + cupsFileClose(fp); + return (0); + } + + if (start && *start) + { + for (i = 1; i < PPD_MAX_LANG && *start; i ++) + { + if ((ptr = strchr(start, ',')) != NULL) + *ptr++ = '\0'; + else + ptr = start + strlen(start); + + strlcpy(ppd->record.languages[i], start, + sizeof(ppd->record.languages[0])); + + start = ptr; + } + } + + fprintf(stderr, "DEBUG2: [cups-driverd] Added dynamic PPD \"%s\"...\n", + name); + } + } + + cupsFileClose(fp); + } + else + fprintf(stderr, "WARNING: [cups-driverd] Unable to execute \"%s\": %s\n", + filename, strerror(errno)); + } + + cupsDirClose(dir); + + return (1); +} + + +/* + * 'load_ppds()' - Load PPD files recursively. + */ + +static int /* O - 1 on success, 0 on failure */ +load_ppds(const char *d, /* I - Actual directory */ + const char *p, /* I - Virtual path in name */ + int descend) /* I - Descend into directories? */ +{ + struct stat dinfo, /* Directory information */ + *dinfoptr; /* Pointer to match */ + int i; /* Looping var */ + cups_file_t *fp; /* Pointer to file */ + cups_dir_t *dir; /* Directory pointer */ + cups_dentry_t *dent; /* Directory entry */ + char filename[1024], /* Name of PPD or directory */ + line[256], /* Line from backend */ + *ptr, /* Pointer into name */ + name[128], /* Name of PPD file */ + lang_version[64], /* PPD LanguageVersion */ + lang_encoding[64], /* PPD LanguageEncoding */ + country[64], /* Country code */ + manufacturer[256], /* Manufacturer */ + make_model[256], /* Make and Model */ + model_name[256], /* ModelName */ + nick_name[256], /* NickName */ + device_id[256], /* 1284DeviceID */ + product[256], /* Product */ + psversion[256], /* PSVersion */ + temp[512]; /* Temporary make and model */ + int install_group, /* In the installable options group? */ + model_number, /* cupsModelNumber */ + type; /* ppd-type */ + cups_array_t *products, /* Product array */ + *psversions, /* PSVersion array */ + *cups_languages; /* cupsLanguages array */ + ppd_info_t *ppd, /* New PPD file */ + key; /* Search key */ + int new_ppd; /* Is this a new PPD? */ + struct /* LanguageVersion translation table */ + { + const char *version, /* LanguageVersion string */ + *language; /* Language code */ + } languages[] = + { + { "chinese", "zh" }, + { "czech", "cs" }, + { "danish", "da" }, + { "dutch", "nl" }, + { "english", "en" }, + { "finnish", "fi" }, + { "french", "fr" }, + { "german", "de" }, + { "greek", "el" }, + { "hungarian", "hu" }, + { "italian", "it" }, + { "japanese", "ja" }, + { "korean", "ko" }, + { "norwegian", "no" }, + { "polish", "pl" }, + { "portuguese", "pt" }, + { "russian", "ru" }, + { "simplified chinese", "zh_CN" }, + { "slovak", "sk" }, + { "spanish", "es" }, + { "swedish", "sv" }, + { "traditional chinese", "zh_TW" }, + { "turkish", "tr" } + }; + + + /* + * See if we've loaded this directory before... + */ + + if (stat(d, &dinfo)) + { + if (errno != ENOENT) + fprintf(stderr, "ERROR: [cups-driverd] Unable to stat \"%s\": %s\n", d, + strerror(errno)); + + return (0); + } + else if (cupsArrayFind(Inodes, &dinfo)) + { + fprintf(stderr, "ERROR: [cups-driverd] Skipping \"%s\": loop detected!\n", + d); + return (0); + } + + /* + * Nope, add it to the Inodes array and continue... + */ + + dinfoptr = (struct stat *)malloc(sizeof(struct stat)); + memcpy(dinfoptr, &dinfo, sizeof(struct stat)); + cupsArrayAdd(Inodes, dinfoptr); + + /* + * Check permissions... + */ + + if (_cupsFileCheck(d, _CUPS_FILE_CHECK_DIRECTORY, !geteuid(), + _cupsFileCheckFilter, NULL)) + return (0); + + if ((dir = cupsDirOpen(d)) == NULL) + { + if (errno != ENOENT) + fprintf(stderr, + "ERROR: [cups-driverd] Unable to open PPD directory \"%s\": %s\n", + d, strerror(errno)); + + return (0); + } + + fprintf(stderr, "DEBUG: [cups-driverd] Loading \"%s\"...\n", d); + + while ((dent = cupsDirRead(dir)) != NULL) + { + /* + * Skip files/directories starting with "."... + */ + + if (dent->filename[0] == '.') + continue; + + /* + * See if this is a file... + */ + + snprintf(filename, sizeof(filename), "%s/%s", d, dent->filename); + + if (p[0]) + snprintf(name, sizeof(name), "%s/%s", p, dent->filename); + else + strlcpy(name, dent->filename, sizeof(name)); + + if (S_ISDIR(dent->fileinfo.st_mode)) + { + /* + * Do subdirectory... + */ + + if (descend) + { + if (!load_ppds(filename, name, 1)) + { + cupsDirClose(dir); + return (1); + } + } + else if ((ptr = filename + strlen(filename) - 14) > filename && + !strcmp(ptr, ".printerDriver")) + { + /* + * Load PPDs in a printer driver bundle. + */ + + if (_cupsFileCheck(filename, _CUPS_FILE_CHECK_DIRECTORY, !geteuid(), + _cupsFileCheckFilter, NULL)) + continue; + + strlcat(filename, "/Contents/Resources/PPDs", sizeof(filename)); + strlcat(name, "/Contents/Resources/PPDs", sizeof(name)); + + load_ppds(filename, name, 0); + } + + continue; + } + else if ((ptr = filename + strlen(filename) - 6) > filename && + !strcmp(ptr, ".plist")) + { + /* + * Skip plist files in the PPDs directory... + */ + + continue; + } + else if (_cupsFileCheck(filename, _CUPS_FILE_CHECK_FILE_ONLY, !geteuid(), + _cupsFileCheckFilter, NULL)) + continue; + + /* + * See if this file has been scanned before... + */ + + strcpy(key.record.filename, name); + strcpy(key.record.name, name); + + ppd = (ppd_info_t *)cupsArrayFind(PPDsByName, &key); + + if (ppd && + ppd->record.size == dent->fileinfo.st_size && + ppd->record.mtime == dent->fileinfo.st_mtime) + { + /* + * Rewind to the first entry for this file... + */ + + while ((ppd = (ppd_info_t *)cupsArrayPrev(PPDsByName)) != NULL && + !strcmp(ppd->record.filename, name)); + + /* + * Then mark all of the matches for this file as found... + */ + + while ((ppd = (ppd_info_t *)cupsArrayNext(PPDsByName)) != NULL && + !strcmp(ppd->record.filename, name)) + ppd->found = 1; + + continue; + } + + /* + * No, file is new/changed, so re-scan it... + */ + + if ((fp = cupsFileOpen(filename, "r")) == NULL) + continue; + + /* + * Now see if this is a PPD file... + */ + + line[0] = '\0'; + cupsFileGets(fp, line, sizeof(line)); + + if (strncmp(line, "*PPD-Adobe:", 11)) + { + /* + * Nope, treat it as a driver information file... + */ + + load_drv(filename, name, fp, dent->fileinfo.st_mtime, + dent->fileinfo.st_size); + continue; + } + + /* + * Now read until we get the NickName field... + */ + + cups_languages = cupsArrayNew(NULL, NULL); + products = cupsArrayNew(NULL, NULL); + psversions = cupsArrayNew(NULL, NULL); + + model_name[0] = '\0'; + nick_name[0] = '\0'; + manufacturer[0] = '\0'; + device_id[0] = '\0'; + lang_encoding[0] = '\0'; + strcpy(lang_version, "en"); + model_number = 0; + install_group = 0; + type = PPD_TYPE_POSTSCRIPT; + + while (cupsFileGets(fp, line, sizeof(line)) != NULL) + { + if (!strncmp(line, "*Manufacturer:", 14)) + sscanf(line, "%*[^\"]\"%255[^\"]", manufacturer); + else if (!strncmp(line, "*ModelName:", 11)) + sscanf(line, "%*[^\"]\"%127[^\"]", model_name); + else if (!strncmp(line, "*LanguageEncoding:", 18)) + sscanf(line, "%*[^:]:%63s", lang_encoding); + else if (!strncmp(line, "*LanguageVersion:", 17)) + sscanf(line, "%*[^:]:%63s", lang_version); + else if (!strncmp(line, "*NickName:", 10)) + sscanf(line, "%*[^\"]\"%255[^\"]", nick_name); + else if (!_cups_strncasecmp(line, "*1284DeviceID:", 14)) + { + sscanf(line, "%*[^\"]\"%255[^\"]", device_id); + + // Make sure device ID ends with a semicolon... + if (device_id[0] && device_id[strlen(device_id) - 1] != ';') + strlcat(device_id, ";", sizeof(device_id)); + } + else if (!strncmp(line, "*Product:", 9)) + { + if (sscanf(line, "%*[^\"]\"(%255[^\"]", product) == 1) + { + /* + * Make sure the value ends with a right parenthesis - can't stop at + * the first right paren since the product name may contain escaped + * parenthesis... + */ + + ptr = product + strlen(product) - 1; + if (ptr > product && *ptr == ')') + { + /* + * Yes, ends with a parenthesis, so remove it from the end and + * add the product to the list... + */ + + *ptr = '\0'; + cupsArrayAdd(products, strdup(product)); + } + } + } + else if (!strncmp(line, "*PSVersion:", 11)) + { + sscanf(line, "%*[^\"]\"%255[^\"]", psversion); + cupsArrayAdd(psversions, strdup(psversion)); + } + else if (!strncmp(line, "*cupsLanguages:", 15)) + { + char *start; /* Start of language */ + + + for (start = line + 15; *start && isspace(*start & 255); start ++); + + if (*start++ == '\"') + { + while (*start) + { + for (ptr = start + 1; + *ptr && *ptr != '\"' && !isspace(*ptr & 255); + ptr ++); + + if (*ptr) + { + *ptr++ = '\0'; + + while (isspace(*ptr & 255)) + *ptr++ = '\0'; + } + + cupsArrayAdd(cups_languages, strdup(start)); + start = ptr; + } + } + } + else if (!strncmp(line, "*cupsFax:", 9)) + { + for (ptr = line + 9; isspace(*ptr & 255); ptr ++); + + if (!_cups_strncasecmp(ptr, "true", 4)) + type = PPD_TYPE_FAX; + } + else if (!strncmp(line, "*cupsFilter:", 12) && type == PPD_TYPE_POSTSCRIPT) + { + if (strstr(line + 12, "application/vnd.cups-raster")) + type = PPD_TYPE_RASTER; + else if (strstr(line + 12, "application/vnd.cups-pdf")) + type = PPD_TYPE_PDF; + } + else if (!strncmp(line, "*cupsModelNumber:", 17)) + sscanf(line, "*cupsModelNumber:%d", &model_number); + else if (!strncmp(line, "*OpenGroup: Installable", 23)) + install_group = 1; + else if (!strncmp(line, "*CloseGroup:", 12)) + install_group = 0; + else if (!strncmp(line, "*OpenUI", 7)) + { + /* + * Stop early if we have a NickName or ModelName attributes + * before the first non-installable OpenUI... + */ + + if (!install_group && (model_name[0] || nick_name[0]) && + cupsArrayCount(products) > 0 && cupsArrayCount(psversions) > 0) + break; + } + } + + /* + * Close the file... + */ + + cupsFileClose(fp); + + /* + * See if we got all of the required info... + */ + + if (nick_name[0]) + cupsCharsetToUTF8((cups_utf8_t *)make_model, nick_name, + sizeof(make_model), _ppdGetEncoding(lang_encoding)); + else + strcpy(make_model, model_name); + + while (isspace(make_model[0] & 255)) + _cups_strcpy(make_model, make_model + 1); + + if (!make_model[0] || cupsArrayCount(products) == 0 || + cupsArrayCount(psversions) == 0) + { + /* + * We don't have all the info needed, so skip this file... + */ + + if (!make_model[0]) + fprintf(stderr, "WARNING: Missing NickName and ModelName in %s!\n", + filename); + + if (cupsArrayCount(products) == 0) + fprintf(stderr, "WARNING: Missing Product in %s!\n", filename); + + if (cupsArrayCount(psversions) == 0) + fprintf(stderr, "WARNING: Missing PSVersion in %s!\n", filename); + + free_array(products); + free_array(psversions); + free_array(cups_languages); + + continue; + } + + if (model_name[0]) + cupsArrayAdd(products, strdup(model_name)); + + /* + * Normalize the make and model string... + */ + + while (isspace(manufacturer[0] & 255)) + _cups_strcpy(manufacturer, manufacturer + 1); + + if (!_cups_strncasecmp(make_model, manufacturer, strlen(manufacturer))) + strlcpy(temp, make_model, sizeof(temp)); + else + snprintf(temp, sizeof(temp), "%s %s", manufacturer, make_model); + + _ppdNormalizeMakeAndModel(temp, make_model, sizeof(make_model)); + + /* + * See if we got a manufacturer... + */ + + if (!manufacturer[0] || !strcmp(manufacturer, "ESP")) + { + /* + * Nope, copy the first part of the make and model then... + */ + + strlcpy(manufacturer, make_model, sizeof(manufacturer)); + + /* + * Truncate at the first space, dash, or slash, or make the + * manufacturer "Other"... + */ + + for (ptr = manufacturer; *ptr; ptr ++) + if (*ptr == ' ' || *ptr == '-' || *ptr == '/') + break; + + if (*ptr && ptr > manufacturer) + *ptr = '\0'; + else + strcpy(manufacturer, "Other"); + } + else if (!_cups_strncasecmp(manufacturer, "LHAG", 4) || + !_cups_strncasecmp(manufacturer, "linotype", 8)) + strcpy(manufacturer, "LHAG"); + else if (!_cups_strncasecmp(manufacturer, "Hewlett", 7)) + strcpy(manufacturer, "HP"); + + /* + * Fix the lang_version as needed... + */ + + if ((ptr = strchr(lang_version, '-')) != NULL) + *ptr++ = '\0'; + else if ((ptr = strchr(lang_version, '_')) != NULL) + *ptr++ = '\0'; + + if (ptr) + { + /* + * Setup the country suffix... + */ + + country[0] = '_'; + _cups_strcpy(country + 1, ptr); + } + else + { + /* + * No country suffix... + */ + + country[0] = '\0'; + } + + for (i = 0; i < (int)(sizeof(languages) / sizeof(languages[0])); i ++) + if (!_cups_strcasecmp(languages[i].version, lang_version)) + break; + + if (i < (int)(sizeof(languages) / sizeof(languages[0]))) + { + /* + * Found a known language... + */ + + snprintf(lang_version, sizeof(lang_version), "%s%s", + languages[i].language, country); + } + else + { + /* + * Unknown language; use "xx"... + */ + + strcpy(lang_version, "xx"); + } + + /* + * Record the PPD file... + */ + + new_ppd = !ppd; + + if (new_ppd) + { + /* + * Add new PPD file... + */ + + fprintf(stderr, "DEBUG2: [cups-driverd] Adding ppd \"%s\"...\n", name); + + ppd = add_ppd(name, name, lang_version, manufacturer, make_model, + device_id, (char *)cupsArrayFirst(products), + (char *)cupsArrayFirst(psversions), + dent->fileinfo.st_mtime, dent->fileinfo.st_size, + model_number, type, "file"); + + if (!ppd) + { + cupsDirClose(dir); + return (0); + } + } + else + { + /* + * Update existing record... + */ + + fprintf(stderr, "DEBUG2: [cups-driverd] Updating ppd \"%s\"...\n", name); + + memset(ppd, 0, sizeof(ppd_info_t)); + + ppd->found = 1; + ppd->record.mtime = dent->fileinfo.st_mtime; + ppd->record.size = dent->fileinfo.st_size; + ppd->record.model_number = model_number; + ppd->record.type = type; + + strlcpy(ppd->record.name, name, sizeof(ppd->record.name)); + strlcpy(ppd->record.make, manufacturer, sizeof(ppd->record.make)); + strlcpy(ppd->record.make_and_model, make_model, + sizeof(ppd->record.make_and_model)); + strlcpy(ppd->record.languages[0], lang_version, + sizeof(ppd->record.languages[0])); + strlcpy(ppd->record.products[0], (char *)cupsArrayFirst(products), + sizeof(ppd->record.products[0])); + strlcpy(ppd->record.psversions[0], (char *)cupsArrayFirst(psversions), + sizeof(ppd->record.psversions[0])); + strlcpy(ppd->record.device_id, device_id, sizeof(ppd->record.device_id)); + } + + /* + * Add remaining products, versions, and languages... + */ + + for (i = 1; + i < PPD_MAX_PROD && (ptr = (char *)cupsArrayNext(products)) != NULL; + i ++) + strlcpy(ppd->record.products[i], ptr, + sizeof(ppd->record.products[0])); + + for (i = 1; + i < PPD_MAX_VERS && (ptr = (char *)cupsArrayNext(psversions)) != NULL; + i ++) + strlcpy(ppd->record.psversions[i], ptr, + sizeof(ppd->record.psversions[0])); + + for (i = 1, ptr = (char *)cupsArrayFirst(cups_languages); + i < PPD_MAX_LANG && ptr; + i ++, ptr = (char *)cupsArrayNext(cups_languages)) + strlcpy(ppd->record.languages[i], ptr, + sizeof(ppd->record.languages[0])); + + /* + * Free products, versions, and languages... + */ + + free_array(cups_languages); + free_array(products); + free_array(psversions); + + ChangedPPD = 1; + } + + cupsDirClose(dir); + + return (1); +} + + +/* + * 'load_ppds_dat()' - Load the ppds.dat file. + */ + +static void +load_ppds_dat(char *filename, /* I - Filename buffer */ + size_t filesize, /* I - Size of filename buffer */ + int verbose) /* I - Be verbose? */ +{ + ppd_info_t *ppd; /* Current PPD file */ + cups_file_t *fp; /* ppds.dat file */ + struct stat fileinfo; /* ppds.dat information */ + const char *cups_cachedir; /* CUPS_CACHEDIR environment variable */ + + + PPDsByName = cupsArrayNew((cups_array_func_t)compare_names, NULL); + PPDsByMakeModel = cupsArrayNew((cups_array_func_t)compare_ppds, NULL); + ChangedPPD = 0; + + if (!filename[0]) + { + if ((cups_cachedir = getenv("CUPS_CACHEDIR")) == NULL) + cups_cachedir = CUPS_CACHEDIR; + + snprintf(filename, filesize, "%s/ppds.dat", cups_cachedir); + } + + if ((fp = cupsFileOpen(filename, "r")) != NULL) + { + /* + * See if we have the right sync word... + */ + + unsigned ppdsync; /* Sync word */ + int num_ppds; /* Number of PPDs */ + + if (cupsFileRead(fp, (char *)&ppdsync, sizeof(ppdsync)) + == sizeof(ppdsync) && + ppdsync == PPD_SYNC && + !stat(filename, &fileinfo) && + ((fileinfo.st_size - sizeof(ppdsync)) % sizeof(ppd_rec_t)) == 0 && + (num_ppds = (fileinfo.st_size - sizeof(ppdsync)) / + sizeof(ppd_rec_t)) > 0) + { + /* + * We have a ppds.dat file, so read it! + */ + + for (; num_ppds > 0; num_ppds --) + { + if ((ppd = (ppd_info_t *)calloc(1, sizeof(ppd_info_t))) == NULL) + { + if (verbose) + fputs("ERROR: [cups-driverd] Unable to allocate memory for PPD!\n", + stderr); + exit(1); + } + + if (cupsFileRead(fp, (char *)&(ppd->record), sizeof(ppd_rec_t)) > 0) + { + cupsArrayAdd(PPDsByName, ppd); + cupsArrayAdd(PPDsByMakeModel, ppd); + } + else + { + free(ppd); + break; + } + } + + if (verbose) + fprintf(stderr, "INFO: [cups-driverd] Read \"%s\", %d PPDs...\n", + filename, cupsArrayCount(PPDsByName)); + } + + cupsFileClose(fp); + } +} + + +/* + * 'regex_device_id()' - Compile a regular expression based on the 1284 device + * ID. + */ + +static regex_t * /* O - Regular expression */ +regex_device_id(const char *device_id) /* I - IEEE-1284 device ID */ +{ + char res[2048], /* Regular expression string */ + *ptr; /* Pointer into string */ + regex_t *re; /* Regular expression */ + int cmd; /* Command set string? */ + + + fprintf(stderr, "DEBUG: [cups-driverd] regex_device_id(\"%s\")\n", device_id); + + /* + * Scan the device ID string and insert class, command set, manufacturer, and + * model attributes to match. We assume that the device ID in the PPD and the + * device ID reported by the device itself use the same attribute names and + * order of attributes. + */ + + ptr = res; + + while (*device_id && ptr < (res + sizeof(res) - 6)) + { + cmd = !_cups_strncasecmp(device_id, "COMMAND SET:", 12) || + !_cups_strncasecmp(device_id, "CMD:", 4); + + if (cmd || !_cups_strncasecmp(device_id, "MANUFACTURER:", 13) || + !_cups_strncasecmp(device_id, "MFG:", 4) || + !_cups_strncasecmp(device_id, "MFR:", 4) || + !_cups_strncasecmp(device_id, "MODEL:", 6) || + !_cups_strncasecmp(device_id, "MDL:", 4)) + { + if (ptr > res) + { + *ptr++ = '.'; + *ptr++ = '*'; + } + + *ptr++ = '('; + + while (*device_id && *device_id != ';' && ptr < (res + sizeof(res) - 8)) + { + if (strchr("[]{}().*\\|", *device_id)) + *ptr++ = '\\'; + if (*device_id == ':') + { + /* + * KEY:.*value + */ + + *ptr++ = *device_id++; + *ptr++ = '.'; + *ptr++ = '*'; + } + else + *ptr++ = *device_id++; + } + + if (*device_id == ';' || !*device_id) + { + /* + * KEY:.*value.*; + */ + + *ptr++ = '.'; + *ptr++ = '*'; + *ptr++ = ';'; + } + *ptr++ = ')'; + if (cmd) + *ptr++ = '?'; + } + else if ((device_id = strchr(device_id, ';')) == NULL) + break; + else + device_id ++; + } + + *ptr = '\0'; + + fprintf(stderr, "DEBUG: [cups-driverd] regex_device_id: \"%s\"\n", res); + + /* + * Compile the regular expression and return... + */ + + if (res[0] && (re = (regex_t *)calloc(1, sizeof(regex_t))) != NULL) + { + if (!regcomp(re, res, REG_EXTENDED | REG_ICASE)) + { + fputs("DEBUG: [cups-driverd] regex_device_id: OK\n", stderr); + return (re); + } + + free(re); + } + + return (NULL); +} + + +/* + * 'regex_string()' - Construct a regular expression to compare a simple string. + */ + +static regex_t * /* O - Regular expression */ +regex_string(const char *s) /* I - String to compare */ +{ + char res[2048], /* Regular expression string */ + *ptr; /* Pointer into string */ + regex_t *re; /* Regular expression */ + + + fprintf(stderr, "DEBUG: [cups-driverd] regex_string(\"%s\")\n", s); + + /* + * Convert the string to a regular expression, escaping special characters + * as needed. + */ + + ptr = res; + + while (*s && ptr < (res + sizeof(res) - 2)) + { + if (strchr("[]{}().*\\", *s)) + *ptr++ = '\\'; + + *ptr++ = *s++; + } + + *ptr = '\0'; + + fprintf(stderr, "DEBUG: [cups-driverd] regex_string: \"%s\"\n", res); + + /* + * Create a case-insensitive regular expression... + */ + + if (res[0] && (re = (regex_t *)calloc(1, sizeof(regex_t))) != NULL) + { + if (!regcomp(re, res, REG_ICASE)) + { + fputs("DEBUG: [cups-driverd] regex_string: OK\n", stderr); + return (re); + } + + free(re); + } + + return (NULL); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/cups-exec.c b/scheduler/cups-exec.c new file mode 100644 index 0000000000..f6c5d66b35 --- /dev/null +++ b/scheduler/cups-exec.c @@ -0,0 +1,108 @@ +/* + * "$Id$" + * + * Sandbox helper for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Usage: + * + * cups-exec /path/to/profile /path/to/program argv0 argv1 ... argvN + * + * Contents: + * + * main() - Apply sandbox profile and execute program. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#ifdef HAVE_SANDBOX_H +# include +# ifndef SANDBOX_NAMED_EXTERNAL +# define SANDBOX_NAMED_EXTERNAL 0x0003 +# endif /* !SANDBOX_NAMED_EXTERNAL */ +#endif /* HAVE_SANDBOX_H */ + + +/* + * 'main()' - Apply sandbox profile and execute program. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ +#ifdef HAVE_SANDBOX_H + char *sandbox_error = NULL; /* Sandbox error, if any */ +#endif /* HAVE_SANDBOX_H */ + + + /* + * Check that we have enough arguments... + */ + + if (argc < 4) + { + puts("Usage: cups-exec /path/to/profile /path/to/program argv0 argv1 ... " + "argvN"); + return (1); + } + +#ifdef HAVE_SANDBOX_H + /* + * Run in a separate security profile... + */ + + if (strcmp(argv[1], "none") && + sandbox_init(argv[1], SANDBOX_NAMED_EXTERNAL, &sandbox_error)) + { + fprintf(stderr, "DEBUG: sandbox_init failed: %s (%s)\n", sandbox_error, + strerror(errno)); + sandbox_free_error(sandbox_error); + return (1); + } +#endif /* HAVE_SANDBOX_H */ + + /* + * Close file descriptors we don't need (insurance): + * + * 0 = stdin + * 1 = stdout + * 2 = stderr + * 3 = back-channel + * 4 = side-channel + * 5-N = unused + */ + + for (i = 5; i < 1024; i ++) + close(i); + + /* + * Execute the program... + */ + + execv(argv[2], argv + 3); + + /* + * If we get here, execv() failed... + */ + + fprintf(stderr, "DEBUG: execv failed: %s\n", strerror(errno)); + return (1); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/cups-lpd.c b/scheduler/cups-lpd.c new file mode 100644 index 0000000000..b8224d42bf --- /dev/null +++ b/scheduler/cups-lpd.c @@ -0,0 +1,1627 @@ +/* + * "$Id$" + * + * Line Printer Daemon interface for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Process an incoming LPD request... + * create_job() - Create a new print job. + * get_printer() - Get the named printer and its options. + * print_file() - Add a file to the current job. + * recv_print_job() - Receive a print job from the client. + * remove_jobs() - Cancel one or more jobs. + * send_state() - Send the queue state. + * smart_gets() - Get a line of text, removing the trailing CR and/or LF. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include +#include +#include +#include +#include + +#ifdef HAVE_INTTYPES_H +# include +#endif /* HAVE_INTTYPES_H */ + + +/* + * LPD "mini-daemon" for CUPS. This program must be used in conjunction + * with inetd or another similar program that monitors ports and starts + * daemons for each client connection. A typical configuration is: + * + * printer stream tcp nowait lp /usr/lib/cups/daemon/cups-lpd cups-lpd + * + * This daemon implements most of RFC 1179 (the unofficial LPD specification) + * except for: + * + * - This daemon does not check to make sure that the source port is + * between 721 and 731, since it isn't necessary for proper + * functioning and port-based security is no security at all! + * + * - The "Print any waiting jobs" command is a no-op. + * + * The LPD-to-IPP mapping is as defined in RFC 2569. The report formats + * currently match the Solaris LPD mini-daemon. + */ + +/* + * Prototypes... + */ + +static int create_job(http_t *http, const char *dest, const char *title, + const char *docname, const char *user, + int num_options, cups_option_t *options); +static int get_printer(http_t *http, const char *name, char *dest, + int destsize, cups_option_t **options, + int *accepting, int *shared, ipp_pstate_t *state); +static int print_file(http_t *http, int id, const char *filename, + const char *docname, const char *user, + const char *format, int last); +static int recv_print_job(const char *name, int num_defaults, + cups_option_t *defaults); +static int remove_jobs(const char *name, const char *agent, + const char *list); +static int send_state(const char *name, const char *list, + int longstatus); +static char *smart_gets(char *s, int len, FILE *fp); + + +/* + * 'main()' - Process an incoming LPD request... + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + int num_defaults; /* Number of default options */ + cups_option_t *defaults; /* Default options */ + char line[256], /* Command string */ + command, /* Command code */ + *dest, /* Pointer to destination */ + *list, /* Pointer to list */ + *agent, /* Pointer to user */ + status; /* Status for client */ + socklen_t hostlen; /* Size of client address */ + http_addr_t hostaddr; /* Address of client */ + char hostname[256], /* Name of client */ + hostip[256], /* IP address */ + *hostfamily; /* Address family */ + int hostlookups; /* Do hostname lookups? */ + + + /* + * Don't buffer the output... + */ + + setbuf(stdout, NULL); + + /* + * Log things using the "cups-lpd" name... + */ + + openlog("cups-lpd", LOG_PID, LOG_LPR); + + /* + * Scan the command-line for options... + */ + + num_defaults = 0; + defaults = NULL; + hostlookups = 1; + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-') + { + switch (argv[i][1]) + { + case 'h' : /* -h hostname[:port] */ + if (argv[i][2]) + cupsSetServer(argv[i] + 2); + else + { + i ++; + if (i < argc) + cupsSetServer(argv[i]); + else + syslog(LOG_WARNING, "Expected hostname string after -h option!"); + } + break; + + case 'o' : /* Option */ + if (argv[i][2]) + num_defaults = cupsParseOptions(argv[i] + 2, num_defaults, + &defaults); + else + { + i ++; + if (i < argc) + num_defaults = cupsParseOptions(argv[i], num_defaults, + &defaults); + else + syslog(LOG_WARNING, "Expected option string after -o option!"); + } + break; + + case 'n' : /* Don't do hostname lookups */ + hostlookups = 0; + break; + + default : + syslog(LOG_WARNING, "Unknown option \"%c\" ignored!", argv[i][1]); + break; + } + } + else + syslog(LOG_WARNING, "Unknown command-line option \"%s\" ignored!", + argv[i]); + + /* + * Get the address of the client... + */ + + hostlen = sizeof(hostaddr); + + if (getpeername(0, (struct sockaddr *)&hostaddr, &hostlen)) + { + syslog(LOG_WARNING, "Unable to get client address - %s", strerror(errno)); + strcpy(hostname, "unknown"); + } + else + { + httpAddrString(&hostaddr, hostip, sizeof(hostip)); + + if (hostlookups) + httpAddrLookup(&hostaddr, hostname, sizeof(hostname)); + else + strlcpy(hostname, hostip, sizeof(hostname)); + +#ifdef AF_INET6 + if (hostaddr.addr.sa_family == AF_INET6) + hostfamily = "IPv6"; + else +#endif /* AF_INET6 */ + hostfamily = "IPv4"; + + syslog(LOG_INFO, "Connection from %s (%s %s)", hostname, hostfamily, + hostip); + } + + num_defaults = cupsAddOption("job-originating-host-name", hostname, + num_defaults, &defaults); + + /* + * RFC1179 specifies that only 1 daemon command can be received for + * every connection. + */ + + if (smart_gets(line, sizeof(line), stdin) == NULL) + { + /* + * Unable to get command from client! Send an error status and return. + */ + + syslog(LOG_ERR, "Unable to get command line from client!"); + putchar(1); + return (1); + } + + /* + * The first byte is the command byte. After that will be the queue name, + * resource list, and/or user name. + */ + + command = line[0]; + dest = line + 1; + + if (command == 0x02) + list = NULL; + else + { + for (list = dest + 1; *list && !isspace(*list & 255); list ++); + + while (isspace(*list & 255)) + *list++ = '\0'; + } + + /* + * Do the command... + */ + + switch (command) + { + default : /* Unknown command */ + syslog(LOG_ERR, "Unknown LPD command 0x%02X!", command); + syslog(LOG_ERR, "Command line = %s", line + 1); + putchar(1); + + status = 1; + break; + + case 0x01 : /* Print any waiting jobs */ + syslog(LOG_INFO, "Print waiting jobs (no-op)"); + putchar(0); + + status = 0; + break; + + case 0x02 : /* Receive a printer job */ + syslog(LOG_INFO, "Receive print job for %s", dest); + /* recv_print_job() sends initial status byte */ + + status = recv_print_job(dest, num_defaults, defaults); + break; + + case 0x03 : /* Send queue state (short) */ + syslog(LOG_INFO, "Send queue state (short) for %s %s", dest, list); + /* no status byte for this command */ + + status = send_state(dest, list, 0); + break; + + case 0x04 : /* Send queue state (long) */ + syslog(LOG_INFO, "Send queue state (long) for %s %s", dest, list); + /* no status byte for this command */ + + status = send_state(dest, list, 1); + break; + + case 0x05 : /* Remove jobs */ + if (list) + { + /* + * Grab the agent and skip to the list of users and/or jobs. + */ + + agent = list; + + for (; *list && !isspace(*list & 255); list ++); + while (isspace(*list & 255)) + *list++ = '\0'; + + syslog(LOG_INFO, "Remove jobs %s on %s by %s", list, dest, agent); + + status = remove_jobs(dest, agent, list); + } + else + status = 1; + + putchar(status); + break; + } + + syslog(LOG_INFO, "Closing connection"); + closelog(); + + return (status); +} + + +/* + * 'create_job()' - Create a new print job. + */ + +static int /* O - Job ID or -1 on error */ +create_job(http_t *http, /* I - HTTP connection */ + const char *dest, /* I - Destination name */ + const char *title, /* I - job-name */ + const char *docname, /* I - Name of job file */ + const char *user, /* I - requesting-user-name */ + int num_options, /* I - Number of options for job */ + cups_option_t *options) /* I - Options for job */ +{ + ipp_t *request; /* IPP request */ + ipp_t *response; /* IPP response */ + ipp_attribute_t *attr; /* IPP attribute */ + char uri[HTTP_MAX_URI]; /* Printer URI */ + int id; /* Job ID */ + + + /* + * Setup the Create-Job request... + */ + + request = ippNewRequest(IPP_CREATE_JOB); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", dest); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, user); + + if (title[0]) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "job-name", + NULL, title); + + if (docname[0]) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "document-name", + NULL, docname); + + cupsEncodeOptions(request, num_options, options); + + /* + * Do the request... + */ + + snprintf(uri, sizeof(uri), "/printers/%s", dest); + + response = cupsDoRequest(http, request, uri); + + if (!response || cupsLastError() > IPP_OK_CONFLICT) + { + syslog(LOG_ERR, "Unable to create job - %s", cupsLastErrorString()); + + ippDelete(response); + + return (-1); + } + + /* + * Get the job-id value from the response and return it... + */ + + if ((attr = ippFindAttribute(response, "job-id", IPP_TAG_INTEGER)) == NULL) + { + id = -1; + + syslog(LOG_ERR, "No job-id attribute found in response from server!"); + } + else + { + id = attr->values[0].integer; + + syslog(LOG_INFO, "Print file - job ID = %d", id); + } + + ippDelete(response); + + return (id); +} + + +/* + * 'get_printer()' - Get the named printer and its options. + */ + +static int /* O - Number of options or -1 on error */ +get_printer(http_t *http, /* I - HTTP connection */ + const char *name, /* I - Printer name from request */ + char *dest, /* I - Destination buffer */ + int destsize, /* I - Size of destination buffer */ + cups_option_t **options, /* O - Printer options */ + int *accepting, /* O - printer-is-accepting-jobs value */ + int *shared, /* O - printer-is-shared value */ + ipp_pstate_t *state) /* O - printer-state value */ +{ + int num_options; /* Number of options */ + cups_file_t *fp; /* lpoptions file */ + char line[1024], /* Line from lpoptions file */ + *value, /* Pointer to value on line */ + *optptr; /* Pointer to options on line */ + int linenum; /* Line number in file */ + const char *cups_serverroot; /* CUPS_SERVERROOT env var */ + ipp_t *request; /* IPP request */ + ipp_t *response; /* IPP response */ + ipp_attribute_t *attr; /* IPP attribute */ + char uri[HTTP_MAX_URI]; /* Printer URI */ + static const char * const requested[] = + { /* Requested attributes */ + "printer-info", + "printer-is-accepting-jobs", + "printer-is-shared", + "printer-name", + "printer-state" + }; + + + /* + * Initialize everything... + */ + + if (accepting) + *accepting = 0; + if (shared) + *shared = 0; + if (state) + *state = IPP_PRINTER_STOPPED; + if (options) + *options = NULL; + + /* + * See if the name is a queue name optionally with an instance name. + */ + + strlcpy(dest, name, destsize); + if ((value = strchr(dest, '/')) != NULL) + *value = '\0'; + + /* + * Setup the Get-Printer-Attributes request... + */ + + request = ippNewRequest(IPP_GET_PRINTER_ATTRIBUTES); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", dest); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", + (int)(sizeof(requested) / sizeof(requested[0])), + NULL, requested); + + /* + * Do the request... + */ + + response = cupsDoRequest(http, request, "/"); + + if (!response || cupsLastError() > IPP_OK_CONFLICT) + { + /* + * If we can't find the printer by name, look up the printer-name + * using the printer-info values... + */ + + ipp_attribute_t *accepting_attr,/* printer-is-accepting-jobs */ + *info_attr, /* printer-info */ + *name_attr, /* printer-name */ + *shared_attr, /* printer-is-shared */ + *state_attr; /* printer-state */ + + + ippDelete(response); + + /* + * Setup the CUPS-Get-Printers request... + */ + + request = ippNewRequest(CUPS_GET_PRINTERS); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", + (int)(sizeof(requested) / sizeof(requested[0])), + NULL, requested); + + /* + * Do the request... + */ + + response = cupsDoRequest(http, request, "/"); + + if (!response || cupsLastError() > IPP_OK_CONFLICT) + { + syslog(LOG_ERR, "Unable to get list of printers - %s", + cupsLastErrorString()); + + ippDelete(response); + + return (-1); + } + + /* + * Scan the response for printers... + */ + + *dest = '\0'; + attr = response->attrs; + + while (attr) + { + /* + * Skip to the next printer... + */ + + while (attr && attr->group_tag != IPP_TAG_PRINTER) + attr = attr->next; + + if (!attr) + break; + + /* + * Get all of the attributes for the current printer... + */ + + accepting_attr = NULL; + info_attr = NULL; + name_attr = NULL; + shared_attr = NULL; + state_attr = NULL; + + while (attr && attr->group_tag == IPP_TAG_PRINTER) + { + if (!strcmp(attr->name, "printer-is-accepting-jobs") && + attr->value_tag == IPP_TAG_BOOLEAN) + accepting_attr = attr; + else if (!strcmp(attr->name, "printer-info") && + attr->value_tag == IPP_TAG_TEXT) + info_attr = attr; + else if (!strcmp(attr->name, "printer-name") && + attr->value_tag == IPP_TAG_NAME) + name_attr = attr; + else if (!strcmp(attr->name, "printer-is-shared") && + attr->value_tag == IPP_TAG_BOOLEAN) + shared_attr = attr; + else if (!strcmp(attr->name, "printer-state") && + attr->value_tag == IPP_TAG_ENUM) + state_attr = attr; + + attr = attr->next; + } + + if (info_attr && name_attr && + !_cups_strcasecmp(name, info_attr->values[0].string.text)) + { + /* + * Found a match, use this one! + */ + + strlcpy(dest, name_attr->values[0].string.text, destsize); + + if (accepting && accepting_attr) + *accepting = accepting_attr->values[0].boolean; + + if (shared && shared_attr) + *shared = shared_attr->values[0].boolean; + + if (state && state_attr) + *state = (ipp_pstate_t)state_attr->values[0].integer; + + break; + } + } + + ippDelete(response); + + if (!*dest) + { + syslog(LOG_ERR, "Unable to find \"%s\" in list of printers!", name); + + return (-1); + } + + name = dest; + } + else + { + /* + * Get values from the response... + */ + + if (accepting) + { + if ((attr = ippFindAttribute(response, "printer-is-accepting-jobs", + IPP_TAG_BOOLEAN)) == NULL) + syslog(LOG_ERR, "No printer-is-accepting-jobs attribute found in " + "response from server!"); + else + *accepting = attr->values[0].boolean; + } + + if (shared) + { + if ((attr = ippFindAttribute(response, "printer-is-shared", + IPP_TAG_BOOLEAN)) == NULL) + { + syslog(LOG_ERR, "No printer-is-shared attribute found in " + "response from server!"); + *shared = 1; + } + else + *shared = attr->values[0].boolean; + } + + if (state) + { + if ((attr = ippFindAttribute(response, "printer-state", + IPP_TAG_ENUM)) == NULL) + syslog(LOG_ERR, "No printer-state attribute found in " + "response from server!"); + else + *state = (ipp_pstate_t)attr->values[0].integer; + } + + ippDelete(response); + } + + /* + * Next look for the printer in the lpoptions file... + */ + + num_options = 0; + + if (options && shared && accepting) + { + if ((cups_serverroot = getenv("CUPS_SERVERROOT")) == NULL) + cups_serverroot = CUPS_SERVERROOT; + + snprintf(line, sizeof(line), "%s/lpoptions", cups_serverroot); + if ((fp = cupsFileOpen(line, "r")) != NULL) + { + linenum = 0; + while (cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + { + /* + * Make sure we have "Dest name options" or "Default name options"... + */ + + if ((_cups_strcasecmp(line, "Dest") && _cups_strcasecmp(line, "Default")) || !value) + continue; + + /* + * Separate destination name from options... + */ + + for (optptr = value; *optptr && !isspace(*optptr & 255); optptr ++); + + while (*optptr == ' ') + *optptr++ = '\0'; + + /* + * If this is our destination, parse the options and break out of + * the loop - we're done! + */ + + if (!_cups_strcasecmp(value, name)) + { + num_options = cupsParseOptions(optptr, num_options, options); + break; + } + } + + cupsFileClose(fp); + } + } + else if (options) + *options = NULL; + + /* + * Return the number of options for this destination... + */ + + return (num_options); +} + + +/* + * 'print_file()' - Add a file to the current job. + */ + +static int /* O - 0 on success, -1 on failure */ +print_file(http_t *http, /* I - HTTP connection */ + int id, /* I - Job ID */ + const char *filename, /* I - File to print */ + const char *docname, /* I - document-name */ + const char *user, /* I - requesting-user-name */ + const char *format, /* I - document-format */ + int last) /* I - 1 = last file in job */ +{ + ipp_t *request; /* IPP request */ + char uri[HTTP_MAX_URI]; /* Printer URI */ + + + /* + * Setup the Send-Document request... + */ + + request = ippNewRequest(IPP_SEND_DOCUMENT); + + snprintf(uri, sizeof(uri), "ipp://localhost/jobs/%d", id); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-uri", NULL, uri); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, user); + + if (docname) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "document-name", NULL, docname); + + if (format) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_MIMETYPE, + "document-format", NULL, format); + + if (last) + ippAddBoolean(request, IPP_TAG_OPERATION, "last-document", 1); + + /* + * Do the request... + */ + + snprintf(uri, sizeof(uri), "/jobs/%d", id); + + ippDelete(cupsDoFileRequest(http, request, uri, filename)); + + if (cupsLastError() > IPP_OK_CONFLICT) + { + syslog(LOG_ERR, "Unable to send document - %s", cupsLastErrorString()); + + return (-1); + } + + return (0); +} + + +/* + * 'recv_print_job()' - Receive a print job from the client. + */ + +static int /* O - Command status */ +recv_print_job( + const char *queue, /* I - Printer name */ + int num_defaults, /* I - Number of default options */ + cups_option_t *defaults) /* I - Default options */ +{ + http_t *http; /* HTTP connection */ + int i; /* Looping var */ + int status; /* Command status */ + int fd; /* Temporary file */ + FILE *fp; /* File pointer */ + char filename[1024]; /* Temporary filename */ + int bytes; /* Bytes received */ + char line[256], /* Line from file/stdin */ + command, /* Command from line */ + *count, /* Number of bytes */ + *name; /* Name of file */ + const char *job_sheets; /* Job sheets */ + int num_data; /* Number of data files */ + char control[1024], /* Control filename */ + data[100][256], /* Data files */ + temp[100][1024]; /* Temporary files */ + char user[1024], /* User name */ + title[1024], /* Job title */ + docname[1024], /* Document name */ + dest[256]; /* Printer/class queue */ + int accepting, /* printer-is-accepting */ + shared, /* printer-is-shared */ + num_options; /* Number of options */ + cups_option_t *options; /* Options */ + int id; /* Job ID */ + int docnumber, /* Current document number */ + doccount; /* Count of documents */ + + + /* + * Connect to the server... + */ + + http = httpConnectEncrypt(cupsServer(), ippPort(), cupsEncryption()); + if (!http) + { + syslog(LOG_ERR, "Unable to connect to server: %s", strerror(errno)); + + putchar(1); + + return (1); + } + + /* + * See if the printer is available... + */ + + num_options = get_printer(http, queue, dest, sizeof(dest), &options, + &accepting, &shared, NULL); + + if (num_options < 0 || !accepting || !shared) + { + if (dest[0]) + syslog(LOG_INFO, "Rejecting job because \"%s\" is not %s", dest, + !accepting ? "accepting jobs" : "shared"); + else + syslog(LOG_ERR, "Unable to get printer information for \"%s\"", queue); + + httpClose(http); + + putchar(1); + + return (1); + } + + putchar(0); /* OK so far... */ + + /* + * Read the request... + */ + + status = 0; + num_data = 0; + fd = -1; + + control[0] = '\0'; + + while (smart_gets(line, sizeof(line), stdin) != NULL) + { + if (strlen(line) < 2) + { + status = 1; + break; + } + + command = line[0]; + count = line + 1; + + for (name = count + 1; *name && !isspace(*name & 255); name ++); + while (isspace(*name & 255)) + *name++ = '\0'; + + switch (command) + { + default : + case 0x01 : /* Abort */ + status = 1; + break; + + case 0x02 : /* Receive control file */ + if (strlen(name) < 2) + { + syslog(LOG_ERR, "Bad control file name \"%s\"", name); + putchar(1); + status = 1; + break; + } + + if (control[0]) + { + /* + * Append to the existing control file - the LPD spec is + * not entirely clear, but at least the OS/2 LPD code sends + * multiple control files per connection... + */ + + if ((fd = open(control, O_WRONLY)) < 0) + { + syslog(LOG_ERR, + "Unable to append to temporary control file \"%s\" - %s", + control, strerror(errno)); + putchar(1); + status = 1; + break; + } + + lseek(fd, 0, SEEK_END); + } + else + { + if ((fd = cupsTempFd(control, sizeof(control))) < 0) + { + syslog(LOG_ERR, "Unable to open temporary control file \"%s\" - %s", + control, strerror(errno)); + putchar(1); + status = 1; + break; + } + + strcpy(filename, control); + } + break; + + case 0x03 : /* Receive data file */ + if (strlen(name) < 2) + { + syslog(LOG_ERR, "Bad data file name \"%s\"", name); + putchar(1); + status = 1; + break; + } + + if (num_data >= (int)(sizeof(data) / sizeof(data[0]))) + { + /* + * Too many data files... + */ + + syslog(LOG_ERR, "Too many data files (%d)", num_data); + putchar(1); + status = 1; + break; + } + + strlcpy(data[num_data], name, sizeof(data[0])); + + if ((fd = cupsTempFd(temp[num_data], sizeof(temp[0]))) < 0) + { + syslog(LOG_ERR, "Unable to open temporary data file \"%s\" - %s", + temp[num_data], strerror(errno)); + putchar(1); + status = 1; + break; + } + + strcpy(filename, temp[num_data]); + + num_data ++; + break; + } + + putchar(status); + + if (status) + break; + + /* + * Copy the data or control file from the client... + */ + + for (i = atoi(count); i > 0; i -= bytes) + { + if (i > sizeof(line)) + bytes = sizeof(line); + else + bytes = i; + + if ((bytes = fread(line, 1, bytes, stdin)) > 0) + bytes = write(fd, line, bytes); + + if (bytes < 1) + { + syslog(LOG_ERR, "Error while reading file - %s", + strerror(errno)); + status = 1; + break; + } + } + + /* + * Read trailing nul... + */ + + if (!status) + { + if (fread(line, 1, 1, stdin) < 1) + { + status = 1; + syslog(LOG_ERR, "Error while reading trailing nul - %s", + strerror(errno)); + } + else if (line[0]) + { + status = 1; + syslog(LOG_ERR, "Trailing character after file is not nul (%02X)!", + line[0]); + } + } + + /* + * Close the file and send an acknowledgement... + */ + + close(fd); + + putchar(status); + + if (status) + break; + } + + if (!status) + { + /* + * Process the control file and print stuff... + */ + + if ((fp = fopen(control, "rb")) == NULL) + status = 1; + else + { + /* + * Copy the default options... + */ + + for (i = 0; i < num_defaults; i ++) + num_options = cupsAddOption(defaults[i].name, + defaults[i].value, + num_options, &options); + + /* + * Grab the job information... + */ + + title[0] = '\0'; + user[0] = '\0'; + docname[0] = '\0'; + doccount = 0; + + while (smart_gets(line, sizeof(line), fp) != NULL) + { + /* + * Process control lines... + */ + + switch (line[0]) + { + case 'J' : /* Job name */ + strlcpy(title, line + 1, sizeof(title)); + break; + + case 'N' : /* Document name */ + strlcpy(docname, line + 1, sizeof(docname)); + break; + + case 'P' : /* User identification */ + strlcpy(user, line + 1, sizeof(user)); + break; + + case 'L' : /* Print banner page */ + /* + * If a banner was requested and it's not overridden by a + * command line option and the destination's default is none + * then add the standard banner... + */ + + if (cupsGetOption("job-sheets", num_defaults, defaults) == NULL && + ((job_sheets = cupsGetOption("job-sheets", num_options, + options)) == NULL || + !strcmp(job_sheets, "none,none"))) + { + num_options = cupsAddOption("job-sheets", "standard", + num_options, &options); + } + break; + + case 'c' : /* Plot CIF file */ + case 'd' : /* Print DVI file */ + case 'f' : /* Print formatted file */ + case 'g' : /* Plot file */ + case 'l' : /* Print file leaving control characters (raw) */ + case 'n' : /* Print ditroff output file */ + case 'o' : /* Print PostScript output file */ + case 'p' : /* Print file with 'pr' format (prettyprint) */ + case 'r' : /* File to print with FORTRAN carriage control */ + case 't' : /* Print troff output file */ + case 'v' : /* Print raster file */ + doccount ++; + + if (line[0] == 'l' && + !cupsGetOption("document-format", num_options, options)) + num_options = cupsAddOption("raw", "", num_options, &options); + + if (line[0] == 'p') + num_options = cupsAddOption("prettyprint", "", num_options, + &options); + break; + } + + if (status) + break; + } + + /* + * Check that we have a username... + */ + + if (!user[0]) + { + syslog(LOG_WARNING, "No username specified by client! " + "Using \"anonymous\"..."); + strcpy(user, "anonymous"); + } + + /* + * Create the job... + */ + + if ((id = create_job(http, dest, title, docname, user, num_options, + options)) < 0) + status = 1; + else + { + /* + * Then print the job files... + */ + + rewind(fp); + + docname[0] = '\0'; + docnumber = 0; + + while (smart_gets(line, sizeof(line), fp) != NULL) + { + /* + * Process control lines... + */ + + switch (line[0]) + { + case 'N' : /* Document name */ + strlcpy(docname, line + 1, sizeof(docname)); + break; + + case 'c' : /* Plot CIF file */ + case 'd' : /* Print DVI file */ + case 'f' : /* Print formatted file */ + case 'g' : /* Plot file */ + case 'l' : /* Print file leaving control characters (raw) */ + case 'n' : /* Print ditroff output file */ + case 'o' : /* Print PostScript output file */ + case 'p' : /* Print file with 'pr' format (prettyprint) */ + case 'r' : /* File to print with FORTRAN carriage control */ + case 't' : /* Print troff output file */ + case 'v' : /* Print raster file */ + /* + * Figure out which file we are printing... + */ + + for (i = 0; i < num_data; i ++) + if (!strcmp(data[i], line + 1)) + break; + + if (i >= num_data) + { + status = 1; + break; + } + + /* + * Send the print file... + */ + + docnumber ++; + + if (print_file(http, id, temp[i], docname, user, + cupsGetOption("document-format", num_options, + options), + docnumber == doccount)) + status = 1; + else + status = 0; + + break; + } + + if (status) + break; + } + } + + fclose(fp); + } + } + + cupsFreeOptions(num_options, options); + + httpClose(http); + + /* + * Clean up all temporary files and return... + */ + + unlink(control); + + for (i = 0; i < num_data; i ++) + unlink(temp[i]); + + return (status); +} + + +/* + * 'remove_jobs()' - Cancel one or more jobs. + */ + +static int /* O - Command status */ +remove_jobs(const char *dest, /* I - Destination */ + const char *agent, /* I - User agent */ + const char *list) /* I - List of jobs or users */ +{ + int id; /* Job ID */ + http_t *http; /* HTTP server connection */ + ipp_t *request; /* IPP Request */ + char uri[HTTP_MAX_URI]; /* Job URI */ + + + (void)dest; /* Suppress compiler warnings... */ + + /* + * Try connecting to the local server... + */ + + if ((http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption())) == NULL) + { + syslog(LOG_ERR, "Unable to connect to server %s: %s", cupsServer(), + strerror(errno)); + return (1); + } + + /* + * Loop for each job... + */ + + while ((id = atoi(list)) > 0) + { + /* + * Skip job ID in list... + */ + + while (isdigit(*list & 255)) + list ++; + while (isspace(*list & 255)) + list ++; + + /* + * Build an IPP_CANCEL_JOB request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * job-uri + * requesting-user-name + */ + + request = ippNewRequest(IPP_CANCEL_JOB); + + sprintf(uri, "ipp://localhost/jobs/%d", id); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-uri", NULL, uri); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, agent); + + /* + * Do the request and get back a response... + */ + + ippDelete(cupsDoRequest(http, request, "/jobs")); + + if (cupsLastError() > IPP_OK_CONFLICT) + { + syslog(LOG_WARNING, "Cancel of job ID %d failed: %s\n", id, + cupsLastErrorString()); + httpClose(http); + return (1); + } + else + syslog(LOG_INFO, "Job ID %d canceled", id); + } + + httpClose(http); + + return (0); +} + + +/* + * 'send_state()' - Send the queue state. + */ + +static int /* O - Command status */ +send_state(const char *queue, /* I - Destination */ + const char *list, /* I - Job or user */ + int longstatus) /* I - List of jobs or users */ +{ + int id; /* Job ID from list */ + http_t *http; /* HTTP server connection */ + ipp_t *request, /* IPP Request */ + *response; /* IPP Response */ + ipp_attribute_t *attr; /* Current attribute */ + ipp_pstate_t state; /* Printer state */ + const char *jobdest, /* Pointer into job-printer-uri */ + *jobuser, /* Pointer to job-originating-user-name */ + *jobname; /* Pointer to job-name */ + ipp_jstate_t jobstate; /* job-state */ + int jobid, /* job-id */ + jobsize, /* job-k-octets */ + jobcount, /* Number of jobs */ + jobcopies, /* Number of copies */ + rank; /* Rank of job */ + char rankstr[255]; /* Rank string */ + char namestr[1024]; /* Job name string */ + char uri[HTTP_MAX_URI]; /* Printer URI */ + char dest[256]; /* Printer/class queue */ + static const char * const ranks[10] = /* Ranking strings */ + { + "th", + "st", + "nd", + "rd", + "th", + "th", + "th", + "th", + "th", + "th" + }; + static const char * const requested[] = + { /* Requested attributes */ + "job-id", + "job-k-octets", + "job-state", + "job-printer-uri", + "job-originating-user-name", + "job-name", + "copies" + }; + + + /* + * Try connecting to the local server... + */ + + if ((http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption())) == NULL) + { + syslog(LOG_ERR, "Unable to connect to server %s: %s", cupsServer(), + strerror(errno)); + printf("Unable to connect to server %s: %s", cupsServer(), strerror(errno)); + return (1); + } + + /* + * Get the actual destination name and printer state... + */ + + if (get_printer(http, queue, dest, sizeof(dest), NULL, NULL, NULL, &state)) + { + syslog(LOG_ERR, "Unable to get printer %s: %s", queue, + cupsLastErrorString()); + printf("Unable to get printer %s: %s", queue, cupsLastErrorString()); + return (1); + } + + /* + * Show the queue state... + */ + + switch (state) + { + case IPP_PRINTER_IDLE : + printf("%s is ready\n", dest); + break; + case IPP_PRINTER_PROCESSING : + printf("%s is ready and printing\n", dest); + break; + case IPP_PRINTER_STOPPED : + printf("%s is not ready\n", dest); + break; + } + + /* + * Build an IPP_GET_JOBS or IPP_GET_JOB_ATTRIBUTES request, which requires + * the following attributes: + * + * attributes-charset + * attributes-natural-language + * job-uri or printer-uri + */ + + id = atoi(list); + + request = ippNewRequest(id ? IPP_GET_JOB_ATTRIBUTES : IPP_GET_JOBS); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", dest); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, uri); + + if (id) + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_INTEGER, "job-id", id); + else + { + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, list); + ippAddBoolean(request, IPP_TAG_OPERATION, "my-jobs", 1); + } + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", + sizeof(requested) / sizeof(requested[0]), + NULL, requested); + + /* + * Do the request and get back a response... + */ + + jobcount = 0; + response = cupsDoRequest(http, request, "/"); + + if (cupsLastError() > IPP_OK_CONFLICT) + { + printf("get-jobs failed: %s\n", cupsLastErrorString()); + ippDelete(response); + return (1); + } + + /* + * Loop through the job list and display them... + */ + + for (attr = response->attrs, rank = 1; attr; attr = attr->next) + { + /* + * Skip leading attributes until we hit a job... + */ + + while (attr && (attr->group_tag != IPP_TAG_JOB || !attr->name)) + attr = attr->next; + + if (!attr) + break; + + /* + * Pull the needed attributes from this job... + */ + + jobid = 0; + jobsize = 0; + jobstate = IPP_JOB_PENDING; + jobname = "untitled"; + jobuser = NULL; + jobdest = NULL; + jobcopies = 1; + + while (attr && attr->group_tag == IPP_TAG_JOB) + { + if (!strcmp(attr->name, "job-id") && + attr->value_tag == IPP_TAG_INTEGER) + jobid = attr->values[0].integer; + + if (!strcmp(attr->name, "job-k-octets") && + attr->value_tag == IPP_TAG_INTEGER) + jobsize = attr->values[0].integer; + + if (!strcmp(attr->name, "job-state") && + attr->value_tag == IPP_TAG_ENUM) + jobstate = (ipp_jstate_t)attr->values[0].integer; + + if (!strcmp(attr->name, "job-printer-uri") && + attr->value_tag == IPP_TAG_URI) + if ((jobdest = strrchr(attr->values[0].string.text, '/')) != NULL) + jobdest ++; + + if (!strcmp(attr->name, "job-originating-user-name") && + attr->value_tag == IPP_TAG_NAME) + jobuser = attr->values[0].string.text; + + if (!strcmp(attr->name, "job-name") && + attr->value_tag == IPP_TAG_NAME) + jobname = attr->values[0].string.text; + + if (!strcmp(attr->name, "copies") && + attr->value_tag == IPP_TAG_INTEGER) + jobcopies = attr->values[0].integer; + + attr = attr->next; + } + + /* + * See if we have everything needed... + */ + + if (!jobdest || !jobid) + { + if (!attr) + break; + else + continue; + } + + if (!longstatus && jobcount == 0) + puts("Rank Owner Job File(s) Total Size"); + + jobcount ++; + + /* + * Display the job... + */ + + if (jobstate == IPP_JOB_PROCESSING) + strcpy(rankstr, "active"); + else + { + snprintf(rankstr, sizeof(rankstr), "%d%s", rank, ranks[rank % 10]); + rank ++; + } + + if (longstatus) + { + puts(""); + + if (jobcopies > 1) + snprintf(namestr, sizeof(namestr), "%d copies of %s", jobcopies, + jobname); + else + strlcpy(namestr, jobname, sizeof(namestr)); + + printf("%s: %-33.33s [job %d localhost]\n", jobuser, rankstr, jobid); + printf(" %-39.39s %.0f bytes\n", namestr, 1024.0 * jobsize); + } + else + printf("%-7s %-7.7s %-7d %-31.31s %.0f bytes\n", rankstr, jobuser, + jobid, jobname, 1024.0 * jobsize); + + if (!attr) + break; + } + + ippDelete(response); + + if (jobcount == 0) + puts("no entries"); + + httpClose(http); + + return (0); +} + + +/* + * 'smart_gets()' - Get a line of text, removing the trailing CR and/or LF. + */ + +static char * /* O - Line read or NULL */ +smart_gets(char *s, /* I - Pointer to line buffer */ + int len, /* I - Size of line buffer */ + FILE *fp) /* I - File to read from */ +{ + char *ptr, /* Pointer into line */ + *end; /* End of line */ + int ch; /* Character from file */ + + + /* + * Read the line; unlike fgets(), we read the entire line but dump + * characters that go past the end of the buffer. Also, we accept + * CR, LF, or CR LF for the line endings to be "safe", although + * RFC 1179 specifically says "just use LF". + */ + + ptr = s; + end = s + len - 1; + + while ((ch = getc(fp)) != EOF) + { + if (ch == '\n') + break; + else if (ch == '\r') + { + /* + * See if a LF follows... + */ + + ch = getc(fp); + + if (ch != '\n') + ungetc(ch, fp); + + break; + } + else if (ptr < end) + *ptr++ = ch; + } + + *ptr = '\0'; + + if (ch == EOF && ptr == s) + return (NULL); + else + return (s); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/cups-lpd.xinetd.in b/scheduler/cups-lpd.xinetd.in new file mode 100644 index 0000000000..05fbba754e --- /dev/null +++ b/scheduler/cups-lpd.xinetd.in @@ -0,0 +1,12 @@ +service printer +{ + disable = yes + socket_type = stream + protocol = tcp + wait = no + user = @CUPS_USER@ + group = @CUPS_GROUP@ + passenv = + server = @CUPS_SERVERBIN@/daemon/cups-lpd + server_args = -o document-format=application/octet-stream +} diff --git a/scheduler/cups.sh.in b/scheduler/cups.sh.in new file mode 100644 index 0000000000..e677620435 --- /dev/null +++ b/scheduler/cups.sh.in @@ -0,0 +1,237 @@ +#!/bin/sh +# +# "$Id$" +# +# Startup/shutdown script for CUPS. +# +# Copyright 2007-2011 by Apple Inc. +# Copyright 1997-2007 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +#### OS-Dependent Information + +# +# Linux chkconfig stuff: +# +# chkconfig: 235 99 00 +# description: Startup/shutdown script for CUPS. +# + +# +# NetBSD 1.5+ rcorder script lines. The format of the following two +# lines is very strict -- please don't add additional spaces! +# +# PROVIDE: cups +# REQUIRE: DAEMON +# + + +#### OS-Dependent Configuration + +case "`uname`" in + IRIX*) + IS_ON=/sbin/chkconfig + + if $IS_ON verbose; then + ECHO=echo + else + ECHO=: + fi + ECHO_OK=: + ECHO_ERROR=: + ;; + + *BSD*) + IS_ON=: + ECHO=echo + ECHO_OK=: + ECHO_ERROR=: + ;; + + Darwin*) + . /etc/rc.common + + if test "${CUPS:=-YES-}" = "-NO-"; then + exit 0 + fi + + IS_ON=: + ECHO=ConsoleMessage + ECHO_OK=: + ECHO_ERROR=: + ;; + + Linux*) + IS_ON=/bin/true + if test -f /etc/init.d/functions; then + . /etc/init.d/functions + ECHO=echo + ECHO_OK="echo_success" + ECHO_ERROR="echo_failure" + else + ECHO=echo + ECHO_OK=: + ECHO_ERROR=: + fi + ;; + + *) + IS_ON=/bin/true + ECHO=echo + ECHO_OK=: + ECHO_ERROR=: + ;; +esac + +#### OS-Independent Stuff + +# +# Set the timezone, if possible... This allows the scheduler and +# all child processes to know the local timezone when reporting +# dates and times to the user. If no timezone information is +# found, then Greenwich Mean Time (GMT) will probably be used. +# + +for file in /etc/TIMEZONE /etc/rc.config /etc/sysconfig/clock; do + if test -f $file; then + . $file + fi +done + +if test "x$ZONE" != x; then + TZ="$ZONE" +fi + +if test "x$TIMEZONE" != x; then + TZ="$TIMEZONE" +fi + +if test "x$TZ" != x; then + export TZ +fi + +# +# Don't use TMPDIR environment variable from init script, as that can +# cause cupsd to set TempDir to a user's temporary directory instead +# of the default... +# + +unset TMPDIR + + +# +# Make sure we have the standard program directories in the path +# since some operating systems (this means YOU HP-UX!) don't +# provide a standard path on boot-up... +# + +if test "x$PATH" = x; then + PATH="/bin:/usr/bin:/sbin:/usr/sbin" +else + PATH="/bin:/usr/bin:/sbin:/usr/sbin:$PATH" +fi + +export PATH + +# +# See if the CUPS server (cupsd) is running... +# + +case "`uname`" in + HP-UX* | AIX* | SINIX*) + pid=`ps -e | awk '{if (match($4, ".*/cupsd$") || $4 == "cupsd") print $1}'` + ;; + IRIX* | SunOS*) + pid=`ps -e | nawk '{if (match($4, ".*/cupsd$") || $4 == "cupsd") print $1}'` + ;; + UnixWare*) + pid=`ps -e | awk '{if (match($6, ".*/cupsd$") || $6 == "cupsd") print $1}'` + . /etc/TIMEZONE + ;; + OSF1*) + pid=`ps -e | awk '{if (match($5, ".*/cupsd$") || $5 == "cupsd") print $1}'` + ;; + Linux* | *BSD* | Darwin*) + pid=`ps ax | awk '{if (match($5, ".*/cupsd$") || $5 == "cupsd") print $1}'` + ;; + *) + pid="" + ;; +esac + +# +# Start or stop the CUPS server based upon the first argument to the script. +# + +case $1 in + start | restart | reload) + if $IS_ON cups; then + if test -x /sbin/portrelease; then + /sbin/portrelease cups + fi + + if test "$pid" != ""; then + kill -HUP $pid + else + prefix=@prefix@ + exec_prefix=@exec_prefix@ + @sbindir@/cupsd + if test $? != 0; then + $ECHO_FAIL + $ECHO "cups: unable to $1 scheduler." + exit 1 + fi + fi + $ECHO_OK + $ECHO "cups: ${1}ed scheduler." + fi + ;; + + stop) + if test "$pid" != ""; then + kill $pid + $ECHO_OK + $ECHO "cups: stopped scheduler." + fi + ;; + + status) + if test "$pid" != ""; then + echo "cups: scheduler is running." + else + echo "cups: scheduler is not running." + fi + ;; + + start_msg) + # HP-UX non-standard... + echo "Starting CUPS Server" + ;; + + stop_msg) + # HP-UX non-standard... + echo "Starting CUPS Server" + ;; + + *) + echo "Usage: cups {reload|restart|start|status|stop}" + exit 1 + ;; +esac + +# +# Exit with no errors. +# + +exit 0 + + +# +# End of "$Id$". +# diff --git a/scheduler/cups.xml.in b/scheduler/cups.xml.in new file mode 100644 index 0000000000..211c086b7f --- /dev/null +++ b/scheduler/cups.xml.in @@ -0,0 +1,214 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/scheduler/cupsd.h b/scheduler/cupsd.h new file mode 100644 index 0000000000..151bef273d --- /dev/null +++ b/scheduler/cupsd.h @@ -0,0 +1,242 @@ +/* + * "$Id$" + * + * Main header file for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + + +/* + * Include necessary headers. + */ + +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#ifdef WIN32 +# include +#else +# include +#endif /* WIN32 */ + +#include "mime.h" + +#if defined(HAVE_CDSASSL) +# include +#endif /* HAVE_CDSASSL */ + + +/* + * Some OS's don't have hstrerror(), most notably Solaris... + */ + +#ifndef HAVE_HSTRERROR +# ifdef hstrerror +# undef hstrerror +# endif /* hstrerror */ +# define hstrerror cups_hstrerror + +extern const char *cups_hstrerror(int); +#endif /* !HAVE_HSTRERROR */ + + +/* + * Common constants. + */ + +#ifndef FALSE +# define FALSE 0 +# define TRUE (!FALSE) +#endif /* !FALSE */ + + +/* + * Implementation limits... + */ + +#define MAX_ENV 100 /* Maximum number of environment strings */ +#define MAX_USERPASS 33 /* Maximum size of username/password */ +#define MAX_FILTERS 20 /* Maximum number of filters */ +#define MAX_SYSTEM_GROUPS 32 /* Maximum number of system groups */ + + +/* + * Defaults... + */ + +#define DEFAULT_HISTORY 1 /* Preserve job history? */ +#define DEFAULT_FILES 0 /* Preserve job files? */ +#define DEFAULT_TIMEOUT 300 /* Timeout during requests/updates */ +#define DEFAULT_KEEPALIVE 30 /* Timeout between requests */ +#define DEFAULT_INTERVAL 30 /* Interval between browse updates */ +#define DEFAULT_CHARSET "utf-8" /* Default charset */ + + +/* + * Global variable macros... + */ + +#ifdef _MAIN_C_ +# define VAR +# define VALUE(x) =x +# define VALUE2(x,y) ={x,y} +#else +# define VAR extern +# define VALUE(x) +# define VALUE2(x,y) +#endif /* _MAIN_C */ + + +/* + * Other stuff for the scheduler... + */ + +#include "sysman.h" +#include "statbuf.h" +#include "cert.h" +#include "auth.h" +#include "client.h" +#include "policy.h" +#include "printers.h" +#include "classes.h" +#include "job.h" +#include "conf.h" +#include "banners.h" +#include "dirsvc.h" +#include "network.h" +#include "subscriptions.h" + + +/* + * Reload types... + */ + +#define RELOAD_NONE 0 /* No reload needed */ +#define RELOAD_ALL 1 /* Reload everything */ +#define RELOAD_CUPSD 2 /* Reload only cupsd.conf */ + + +/* + * Select callback function type... + */ + +typedef void (*cupsd_selfunc_t)(void *data); + + +/* + * Globals... + */ + +VAR int TestConfigFile VALUE(0), + /* Test the cupsd.conf file? */ + UseProfiles VALUE(1); + /* Use security profiles for child procs? */ +VAR int MaxFDs VALUE(0); + /* Maximum number of files */ + +VAR time_t ReloadTime VALUE(0); + /* Time of reload request... */ +VAR int NeedReload VALUE(RELOAD_ALL), + /* Need to load configuration? */ + DoingShutdown VALUE(0); + /* Shutting down the scheduler? */ +VAR void *DefaultProfile VALUE(0); + /* Default security profile */ + +#ifdef HAVE_GSSAPI +VAR int KerberosInitialized VALUE(0); + /* Has Kerberos been initialized? */ +VAR krb5_context KerberosContext VALUE(NULL); + /* Kerberos context for credentials */ +#endif /* HAVE_GSSAPI */ + +#ifdef HAVE_LAUNCH_H +VAR int Launchd VALUE(0); + /* Running from launchd */ +#endif /* HAVE_LAUNCH_H */ + + +/* + * Prototypes... + */ + +/* env.c */ +extern void cupsdInitEnv(void); +extern int cupsdLoadEnv(char *envp[], int envmax); +extern void cupsdSetEnv(const char *name, const char *value); +extern void cupsdSetEnvf(const char *name, const char *value, ...) + __attribute__ ((__format__ (__printf__, 2, 3))); +extern void cupsdUpdateEnv(void); + +/* file.c */ +extern void cupsdCleanFiles(const char *path, const char *pattern); +extern int cupsdCloseCreatedConfFile(cups_file_t *fp, + const char *filename); +extern void cupsdClosePipe(int *fds); +extern cups_file_t *cupsdCreateConfFile(const char *filename, mode_t mode); +extern cups_file_t *cupsdOpenConfFile(const char *filename); +extern int cupsdOpenPipe(int *fds); +extern int cupsdRemoveFile(const char *filename); + +/* main.c */ +extern int cupsdAddString(cups_array_t **a, const char *s); +extern void cupsdCheckProcess(void); +extern void cupsdClearString(char **s); +extern void cupsdFreeStrings(cups_array_t **a); +extern void cupsdHoldSignals(void); +extern char *cupsdMakeUUID(const char *name, int number, + char *buffer, size_t bufsize); +extern void cupsdReleaseSignals(void); +extern void cupsdSetString(char **s, const char *v); +extern void cupsdSetStringf(char **s, const char *f, ...) + __attribute__ ((__format__ (__printf__, 2, 3))); + +/* process.c */ +extern void *cupsdCreateProfile(int job_id); +extern void cupsdDestroyProfile(void *profile); +extern int cupsdEndProcess(int pid, int force); +extern const char *cupsdFinishProcess(int pid, char *name, int namelen, + int *job_id); +extern int cupsdStartProcess(const char *command, char *argv[], + char *envp[], int infd, int outfd, + int errfd, int backfd, int sidefd, + int root, void *profile, + cupsd_job_t *job, int *pid); + +/* select.c */ +extern int cupsdAddSelect(int fd, cupsd_selfunc_t read_cb, + cupsd_selfunc_t write_cb, void *data); +extern int cupsdDoSelect(long timeout); +#ifdef CUPSD_IS_SELECTING +extern int cupsdIsSelecting(int fd); +#endif /* CUPSD_IS_SELECTING */ +extern void cupsdRemoveSelect(int fd); +extern void cupsdStartSelect(void); +extern void cupsdStopSelect(void); + +/* server.c */ +extern void cupsdStartServer(void); +extern void cupsdStopServer(void); + + +/* + * End of "$Id$". + */ diff --git a/scheduler/cupsfilter.c b/scheduler/cupsfilter.c new file mode 100644 index 0000000000..c9692b7ef1 --- /dev/null +++ b/scheduler/cupsfilter.c @@ -0,0 +1,1495 @@ +/* + * "$Id$" + * + * Filtering program for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Main entry for the test program. + * add_printer_filter() - Add a single filters from a PPD file. + * add_printer_filters() - Add filters from a PPD file. + * check_cb() - Callback function for _cupsFileCheck. + * compare_pids() - Compare two filter PIDs... + * escape_options() - Convert an options array to a string. + * exec_filter() - Execute a single filter. + * exec_filters() - Execute filters for the given file and options. + * get_job_file() - Get the specified job file. + * open_pipe() - Create a pipe which is closed on exec. + * read_cupsd_conf() - Read the cupsd.conf file to get the filter + * settings. + * set_string() - Copy and set a string. + * sighandler() - Signal catcher for when we print from stdin... + * usage() - Show program usage... + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include "mime.h" +#include +#include +#include +#include +#include +#if defined(__APPLE__) +# include +#endif /* __APPLE__ */ + + +/* + * Local globals... + */ + +static char *DataDir = NULL;/* CUPS_DATADIR environment variable */ +static char *FontPath = NULL; + /* CUPS_FONTPATH environment variable */ +static mime_filter_t GZIPFilter = /* gziptoany filter */ +{ + NULL, /* Source type */ + NULL, /* Destination type */ + 0, /* Cost */ + "gziptoany" /* Filter program to run */ +}; +static char *Path = NULL; /* PATH environment variable */ +static char *ServerBin = NULL; + /* CUPS_SERVERBIN environment variable */ +static char *ServerRoot = NULL; + /* CUPS_SERVERROOT environment variable */ +static char *RIPCache = NULL; + /* RIP_MAX_CACHE environment variable */ +static char TempFile[1024] = ""; + /* Temporary file */ + + +/* + * Local functions... + */ + +static void add_printer_filter(const char *command, mime_t *mime, + mime_type_t *printer_type, + const char *filter); +static mime_type_t *add_printer_filters(const char *command, + mime_t *mime, const char *printer, + const char *ppdfile, + mime_type_t **prefilter_type); +static void check_cb(void *context, _cups_fc_result_t result, + const char *message); +static int compare_pids(mime_filter_t *a, mime_filter_t *b); +static char *escape_options(int num_options, cups_option_t *options); +static int exec_filter(const char *filter, char **argv, + char **envp, int infd, int outfd); +static int exec_filters(mime_type_t *srctype, + cups_array_t *filters, const char *infile, + const char *outfile, const char *ppdfile, + const char *printer, const char *user, + const char *title, int num_options, + cups_option_t *options); +static void get_job_file(const char *job); +static int open_pipe(int *fds); +static int read_cupsd_conf(const char *filename); +static void set_string(char **s, const char *val); +static void sighandler(int sig); +static void usage(const char *command, const char *opt) + __attribute__((noreturn)); + + +/* + * 'main()' - Main entry for the test program. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping vars */ + const char *command, /* Command name */ + *opt, /* Current option */ + *printer; /* Printer name */ + mime_type_t *printer_type, /* Printer MIME type */ + *prefilter_type; /* Printer prefilter MIME type */ + char *srctype, /* Source type */ + *dsttype, /* Destination type */ + super[MIME_MAX_SUPER], /* Super-type name */ + type[MIME_MAX_TYPE]; /* Type name */ + int compression; /* Compression of file */ + int cost; /* Cost of filters */ + mime_t *mime; /* MIME database */ + char mimedir[1024]; /* MIME directory */ + char *infile, /* File to filter */ + *outfile; /* File to create */ + char cupsdconf[1024]; /* cupsd.conf file */ + const char *server_root; /* CUPS_SERVERROOT environment variable */ + mime_type_t *src, /* Source type */ + *dst; /* Destination type */ + cups_array_t *filters; /* Filters for the file */ + int num_options; /* Number of options */ + cups_option_t *options; /* Options */ + const char *ppdfile; /* PPD file */ + const char *title, /* Title string */ + *user; /* Username */ + int all_filters, /* Use all filters */ + removeppd, /* Remove PPD file */ + removeinfile; /* Remove input file */ + int status; /* Execution status */ + + + /* + * Setup defaults... + */ + + if ((command = strrchr(argv[0], '/')) != NULL) + command ++; + else + command = argv[0]; + + printer = !strcmp(command, "convert") ? "tofile" : "cupsfilter"; + mime = NULL; + srctype = NULL; + compression = 0; + dsttype = "application/pdf"; + infile = NULL; + outfile = NULL; + num_options = 0; + options = NULL; + ppdfile = NULL; + title = NULL; + user = cupsUser(); + all_filters = 0; + removeppd = 0; + removeinfile = 0; + + if ((server_root = getenv("CUPS_SERVERROOT")) == NULL) + server_root = CUPS_SERVERROOT; + + snprintf(cupsdconf, sizeof(cupsdconf), "%s/cupsd.conf", server_root); + + /* + * Process command-line arguments... + */ + + _cupsSetLocale(argv); + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-') + { + for (opt = argv[i] + 1; *opt; opt ++) + switch (*opt) + { + case '-' : /* Next argument is a filename... */ + i ++; + if (i < argc && !infile) + infile = argv[i]; + else + usage(command, opt); + break; + + case 'a' : /* Specify option... */ + i ++; + if (i < argc) + num_options = cupsParseOptions(argv[i], num_options, &options); + else + usage(command, opt); + break; + + case 'c' : /* Specify cupsd.conf file location... */ + i ++; + if (i < argc) + { + if (!strcmp(command, "convert")) + num_options = cupsAddOption("copies", argv[i], num_options, + &options); + else + strlcpy(cupsdconf, argv[i], sizeof(cupsdconf)); + } + else + usage(command, opt); + break; + + case 'd' : /* Specify the real printer name */ + i ++; + if (i < argc) + printer = argv[i]; + else + usage(command, opt); + break; + + case 'D' : /* Delete input file after conversion */ + removeinfile = 1; + break; + + case 'e' : /* Use every filter from the PPD file */ + all_filters = 1; + break; + + case 'f' : /* Specify input file... */ + i ++; + if (i < argc && !infile) + infile = argv[i]; + else + usage(command, opt); + break; + + case 'i' : /* Specify source MIME type... */ + i ++; + if (i < argc) + { + if (sscanf(argv[i], "%15[^/]/%255s", super, type) != 2) + usage(command, opt); + + srctype = argv[i]; + } + else + usage(command, opt); + break; + + case 'j' : /* Get job file or specify destination MIME type... */ + if (strcmp(command, "convert")) + { + i ++; + if (i < argc) + { + get_job_file(argv[i]); + infile = TempFile; + } + else + usage(command, opt); + + break; + } + + case 'm' : /* Specify destination MIME type... */ + i ++; + if (i < argc) + { + if (sscanf(argv[i], "%15[^/]/%255s", super, type) != 2) + usage(command, opt); + + dsttype = argv[i]; + } + else + usage(command, opt); + break; + + case 'n' : /* Specify number of copies... */ + i ++; + if (i < argc) + num_options = cupsAddOption("copies", argv[i], num_options, + &options); + else + usage(command, opt); + break; + + case 'o' : /* Specify option(s) or output filename */ + i ++; + if (i < argc) + { + if (!strcmp(command, "convert")) + { + if (outfile) + usage(command, NULL); + else + outfile = argv[i]; + } + else + num_options = cupsParseOptions(argv[i], num_options, + &options); + } + else + usage(command, opt); + break; + + case 'p' : /* Specify PPD file... */ + case 'P' : /* Specify PPD file... */ + i ++; + if (i < argc) + ppdfile = argv[i]; + else + usage(command, opt); + break; + + case 't' : /* Specify title... */ + case 'J' : /* Specify title... */ + i ++; + if (i < argc) + title = argv[i]; + else + usage(command, opt); + break; + + case 'u' : /* Delete PPD file after conversion */ + removeppd = 1; + break; + + case 'U' : /* Specify username... */ + i ++; + if (i < argc) + user = argv[i]; + else + usage(command, opt); + break; + + default : /* Something we don't understand... */ + usage(command, opt); + break; + } + } + else if (!infile) + { + if (strcmp(command, "convert")) + infile = argv[i]; + else + { + _cupsLangPuts(stderr, + _("convert: Use the -f option to specify a file to " + "convert.")); + usage(command, NULL); + } + } + else + { + _cupsLangPuts(stderr, + _("cupsfilter: Only one filename can be specified.")); + usage(command, NULL); + } + + if (!infile && !srctype) + usage(command, NULL); + + if (!title) + { + if (!infile) + title = "(stdin)"; + else if ((title = strrchr(infile, '/')) != NULL) + title ++; + else + title = infile; + } + + /* + * Load the cupsd.conf file and create the MIME database... + */ + + if (read_cupsd_conf(cupsdconf)) + return (1); + + snprintf(mimedir, sizeof(mimedir), "%s/mime", DataDir); + + mime = mimeLoadTypes(NULL, mimedir); + mime = mimeLoadTypes(mime, ServerRoot); + mime = mimeLoadFilters(mime, mimedir, Path); + mime = mimeLoadFilters(mime, ServerRoot, Path); + + if (!mime) + { + _cupsLangPrintf(stderr, + _("%s: Unable to read MIME database from \"%s\" or " + "\"%s\"."), + command, mimedir, ServerRoot); + return (1); + } + + prefilter_type = NULL; + + if (all_filters) + printer_type = add_printer_filters(command, mime, printer, ppdfile, + &prefilter_type); + else + printer_type = mimeType(mime, "application", "vnd.cups-postscript"); + + /* + * Get the source and destination types... + */ + + if (srctype) + { + sscanf(srctype, "%15[^/]/%255s", super, type); + if ((src = mimeType(mime, super, type)) == NULL) + { + _cupsLangPrintf(stderr, + _("%s: Unknown source MIME type %s/%s."), + command, super, type); + return (1); + } + } + else if ((src = mimeFileType(mime, infile, infile, &compression)) == NULL) + { + _cupsLangPrintf(stderr, + _("%s: Unable to determine MIME type of \"%s\"."), + command, infile); + return (1); + } + + sscanf(dsttype, "%15[^/]/%255s", super, type); + if (!_cups_strcasecmp(super, "printer")) + dst = printer_type; + else if ((dst = mimeType(mime, super, type)) == NULL) + { + _cupsLangPrintf(stderr, + _("%s: Unknown destination MIME type %s/%s."), + command, super, type); + return (1); + } + + /* + * Figure out how to filter the file... + */ + + if (src == dst) + { + /* + * Special case - no filtering needed... + */ + + filters = cupsArrayNew(NULL, NULL); + cupsArrayAdd(filters, &GZIPFilter); + GZIPFilter.src = src; + GZIPFilter.dst = dst; + } + else if ((filters = mimeFilter(mime, src, dst, &cost)) == NULL) + { + _cupsLangPrintf(stderr, + _("%s: No filter to convert from %s/%s to %s/%s."), + command, src->super, src->type, dst->super, dst->type); + return (1); + } + else if (compression) + cupsArrayInsert(filters, &GZIPFilter); + + if (prefilter_type) + { + /* + * Add pre-filters... + */ + + mime_filter_t *filter, /* Current filter */ + *prefilter; /* Current pre-filter */ + cups_array_t *prefilters = cupsArrayNew(NULL, NULL); + /* New filters array */ + + + for (filter = (mime_filter_t *)cupsArrayFirst(filters); + filter; + filter = (mime_filter_t *)cupsArrayNext(filters)) + { + if ((prefilter = mimeFilterLookup(mime, filter->src, + prefilter_type)) != NULL) + cupsArrayAdd(prefilters, prefilter); + + cupsArrayAdd(prefilters, filter); + } + + cupsArrayDelete(filters); + filters = prefilters; + } + + /* + * Do it! + */ + + status = exec_filters(src, filters, infile, outfile, ppdfile, printer, user, + title, num_options, options); + + /* + * Remove files as needed, then exit... + */ + + if (TempFile[0]) + unlink(TempFile); + + if (removeppd && ppdfile) + unlink(ppdfile); + + if (removeinfile && infile) + unlink(infile); + + return (status); +} + + +/* + * 'add_printer_filter()' - Add a single filters from a PPD file. + */ + +static void +add_printer_filter( + const char *command, /* I - Command name */ + mime_t *mime, /* I - MIME database */ + mime_type_t *filtertype, /* I - Printer or prefilter MIME type */ + const char *filter) /* I - Filter to add */ +{ + char super[MIME_MAX_SUPER], /* Super-type for filter */ + type[MIME_MAX_TYPE], /* Type for filter */ + dsuper[MIME_MAX_SUPER], /* Destination super-type for filter */ + dtype[MIME_MAX_TYPE], /* Destination type for filter */ + dest[MIME_MAX_SUPER + MIME_MAX_TYPE + 2], + /* Destination super/type */ + program[1024]; /* Program/filter name */ + int cost; /* Cost of filter */ + size_t maxsize = 0; /* Maximum supported file size */ + mime_type_t *temptype, /* MIME type looping var */ + *desttype; /* Destination MIME type */ + mime_filter_t *filterptr; /* MIME filter */ + + + /* + * Parse the filter string; it should be in one of the following formats: + * + * source/type cost program + * source/type cost maxsize(nnnn) program + * source/type dest/type cost program + * source/type dest/type cost maxsize(nnnn) program + */ + + if (sscanf(filter, "%15[^/]/%255s%*[ \t]%15[^/]/%255s%d%*[ \t]%1023[^\n]", + super, type, dsuper, dtype, &cost, program) == 6) + { + snprintf(dest, sizeof(dest), "%s/%s/%s", filtertype->type, dsuper, dtype); + + if ((desttype = mimeType(mime, "printer", dest)) == NULL) + desttype = mimeAddType(mime, "printer", dest); + } + else + { + if (sscanf(filter, "%15[^/]/%255s%d%*[ \t]%1023[^\n]", super, type, &cost, + program) == 4) + { + desttype = filtertype; + } + else + { + _cupsLangPrintf(stderr, _("%s: Invalid filter string \"%s\"."), command, + filter); + return; + } + } + + if (!strncmp(program, "maxsize(", 8)) + { + char *ptr; /* Pointer into maxsize(nnnn) program */ + + maxsize = strtoll(program + 8, &ptr, 10); + + if (*ptr != ')') + { + printf("testmime: Invalid filter string \"%s\".\n", filter); + return; + } + + ptr ++; + while (_cups_isspace(*ptr)) + ptr ++; + + _cups_strcpy(program, ptr); + } + + /* + * See if the filter program exists; if not, stop the printer and flag + * the error! + */ + + if (strcmp(program, "-")) + { + char filename[1024]; /* Full path to program */ + + if (program[0] == '/') + strlcpy(filename, program, sizeof(filename)); + else + snprintf(filename, sizeof(filename), "%s/filter/%s", ServerBin, program); + + if (_cupsFileCheck(filename, _CUPS_FILE_CHECK_PROGRAM, !geteuid(), check_cb, + (void *)command)) + return; + } + + /* + * Add the filter to the MIME database, supporting wildcards as needed... + */ + + for (temptype = mimeFirstType(mime); + temptype; + temptype = mimeNextType(mime)) + if (((super[0] == '*' && _cups_strcasecmp(temptype->super, "printer")) || + !_cups_strcasecmp(temptype->super, super)) && + (type[0] == '*' || !_cups_strcasecmp(temptype->type, type))) + { + if (desttype != filtertype) + { + filterptr = mimeAddFilter(mime, temptype, desttype, cost, program); + + if (!mimeFilterLookup(mime, desttype, filtertype)) + mimeAddFilter(mime, desttype, filtertype, 0, "-"); + } + else + filterptr = mimeAddFilter(mime, temptype, filtertype, cost, program); + + if (filterptr) + filterptr->maxsize = maxsize; + } +} + + +/* + * 'add_printer_filters()' - Add filters from a PPD file. + */ + +static mime_type_t * /* O - Printer type or NULL on error */ +add_printer_filters( + const char *command, /* I - Command name */ + mime_t *mime, /* I - MIME database */ + const char *printer, /* I - Printer name */ + const char *ppdfile, /* I - PPD file */ + mime_type_t **prefilter_type) /* O - Prefilter type */ +{ + ppd_file_t *ppd; /* PPD file data */ + _ppd_cache_t *pc; /* Cache data for PPD */ + const char *value; /* Filter definition value */ + mime_type_t *printer_type; /* Printer filter type */ + + + if ((ppd = _ppdOpenFile(ppdfile, _PPD_LOCALIZATION_NONE)) == NULL) + { + ppd_status_t status; /* PPD load status */ + int linenum; /* Line number */ + + status = ppdLastError(&linenum); + _cupsLangPrintf(stderr, _("%s: Unable to open PPD file: %s on line %d."), + command, ppdErrorString(status), linenum); + return (NULL); + } + + pc = _ppdCacheCreateWithPPD(ppd); + if (!pc) + return (NULL); + + printer_type = mimeAddType(mime, "printer", printer); + *prefilter_type = NULL; + + if (pc->filters) + { + for (value = (const char *)cupsArrayFirst(pc->filters); + value; + value = (const char *)cupsArrayNext(pc->filters)) + add_printer_filter(command, mime, printer_type, value); + } + else + { + add_printer_filter(command, mime, printer_type, + "application/vnd.cups-raw 0 -"); + add_printer_filter(command, mime, printer_type, + "application/vnd.cups-postscript 0 -"); + } + + if (pc->prefilters) + { + *prefilter_type = mimeAddType(mime, "prefilter", printer); + + for (value = (const char *)cupsArrayFirst(pc->prefilters); + value; + value = (const char *)cupsArrayNext(pc->prefilters)) + add_printer_filter(command, mime, *prefilter_type, value); + } + + return (printer_type); +} + + +/* + * 'check_cb()' - Callback function for _cupsFileCheck. + */ + +static void +check_cb(void *context, /* I - Context (command name) */ + _cups_fc_result_t result, /* I - Result of check */ + const char *message) /* I - Localized message */ +{ + (void)result; + + _cupsLangPrintf(stderr, _("%s: %s"), (char *)context, message); +} + + +/* + * 'compare_pids()' - Compare two filter PIDs... + */ + +static int /* O - Result of comparison */ +compare_pids(mime_filter_t *a, /* I - First filter */ + mime_filter_t *b) /* I - Second filter */ +{ + /* + * Because we're particularly lazy, we store the process ID in the "cost" + * variable... + */ + + return (a->cost - b->cost); +} + + +/* + * 'escape_options()' - Convert an options array to a string. + */ + +static char * /* O - Option string */ +escape_options( + int num_options, /* I - Number of options */ + cups_option_t *options) /* I - Options */ +{ + int i; /* Looping var */ + cups_option_t *option; /* Current option */ + int bytes; /* Number of bytes needed */ + char *s, /* Option string */ + *sptr, /* Pointer into string */ + *vptr; /* Pointer into value */ + + + /* + * Figure out the worst-case number of bytes we need for the option string. + */ + + for (i = num_options, option = options, bytes = 1; i > 0; i --, option ++) + bytes += 2 * (strlen(option->name) + strlen(option->value)) + 2; + + if ((s = malloc(bytes)) == NULL) + return (NULL); + + /* + * Copy the options to the string... + */ + + for (i = num_options, option = options, sptr = s; i > 0; i --, option ++) + { + if (!strcmp(option->name, "copies")) + continue; + + if (sptr > s) + *sptr++ = ' '; + + strcpy(sptr, option->name); + sptr += strlen(sptr); + *sptr++ = '='; + + for (vptr = option->value; *vptr;) + { + if (strchr("\\ \t\n", *vptr)) + *sptr++ = '\\'; + + *sptr++ = *vptr++; + } + } + + *sptr = '\0'; + + return (s); +} + + +/* + * 'exec_filter()' - Execute a single filter. + */ + +static int /* O - Process ID or -1 on error */ +exec_filter(const char *filter, /* I - Filter to execute */ + char **argv, /* I - Argument list */ + char **envp, /* I - Environment list */ + int infd, /* I - Stdin file descriptor */ + int outfd) /* I - Stdout file descriptor */ +{ + int pid, /* Process ID */ + fd; /* Temporary file descriptor */ +#if defined(__APPLE__) + char processPath[1024], /* CFProcessPath environment variable */ + linkpath[1024]; /* Link path for symlinks... */ + int linkbytes; /* Bytes for link path */ + + + /* + * Add special voodoo magic for MacOS X - this allows MacOS X + * programs to access their bundle resources properly... + */ + + if ((linkbytes = readlink(filter, linkpath, sizeof(linkpath) - 1)) > 0) + { + /* + * Yes, this is a symlink to the actual program, nul-terminate and + * use it... + */ + + linkpath[linkbytes] = '\0'; + + if (linkpath[0] == '/') + snprintf(processPath, sizeof(processPath), "CFProcessPath=%s", + linkpath); + else + snprintf(processPath, sizeof(processPath), "CFProcessPath=%s/%s", + dirname((char *)filter), linkpath); + } + else + snprintf(processPath, sizeof(processPath), "CFProcessPath=%s", filter); + + envp[0] = processPath; /* Replace string */ +#endif /* __APPLE__ */ + + if ((pid = fork()) == 0) + { + /* + * Child process goes here... + * + * Update stdin/stdout/stderr as needed... + */ + + if (infd != 0) + { + if (infd < 0) + infd = open("/dev/null", O_RDONLY); + + if (infd > 0) + { + dup2(infd, 0); + close(infd); + } + } + + if (outfd != 1) + { + if (outfd < 0) + outfd = open("/dev/null", O_WRONLY); + + if (outfd > 1) + { + dup2(outfd, 1); + close(outfd); + } + } + + if ((fd = open("/dev/null", O_RDWR)) > 3) + { + dup2(fd, 3); + close(fd); + } + fcntl(3, F_SETFL, O_NDELAY); + + if ((fd = open("/dev/null", O_RDWR)) > 4) + { + dup2(fd, 4); + close(fd); + } + fcntl(4, F_SETFL, O_NDELAY); + + /* + * Execute command... + */ + + execve(filter, argv, envp); + + perror(filter); + + exit(errno); + } + + return (pid); +} + + +/* + * 'exec_filters()' - Execute filters for the given file and options. + */ + +static int /* O - 0 on success, 1 on error */ +exec_filters(mime_type_t *srctype, /* I - Source type */ + cups_array_t *filters, /* I - Array of filters to run */ + const char *infile, /* I - File to filter */ + const char *outfile, /* I - File to create */ + const char *ppdfile, /* I - PPD file, if any */ + const char *printer, /* I - Printer name */ + const char *user, /* I - Username */ + const char *title, /* I - Job title */ + int num_options, /* I - Number of filter options */ + cups_option_t *options) /* I - Filter options */ +{ + int i; /* Looping var */ + const char *argv[8], /* Command-line arguments */ + *envp[15], /* Environment variables */ + *temp; /* Temporary string */ + char *optstr, /* Filter options */ + content_type[1024], /* CONTENT_TYPE */ + cups_datadir[1024], /* CUPS_DATADIR */ + cups_fontpath[1024], /* CUPS_FONTPATH */ + cups_serverbin[1024], /* CUPS_SERVERBIN */ + cups_serverroot[1024], /* CUPS_SERVERROOT */ + lang[1024], /* LANG */ + path[1024], /* PATH */ + ppd[1024], /* PPD */ + printer_info[255], /* PRINTER_INFO env variable */ + printer_location[255], /* PRINTER_LOCATION env variable */ + printer_name[255], /* PRINTER env variable */ + rip_max_cache[1024], /* RIP_MAX_CACHE */ + userenv[1024], /* USER */ + program[1024]; /* Program to run */ + mime_filter_t *filter, /* Current filter */ + *next; /* Next filter */ + int current, /* Current filter */ + filterfds[2][2], /* Pipes for filters */ + pid, /* Process ID of filter */ + status, /* Exit status */ + retval; /* Return value */ + cups_array_t *pids; /* Executed filters array */ + mime_filter_t key; /* Search key for filters */ + cups_lang_t *language; /* Current language */ + cups_dest_t *dest; /* Destination information */ + + + /* + * Setup the filter environment and command-line... + */ + + optstr = escape_options(num_options, options); + + snprintf(content_type, sizeof(content_type), "CONTENT_TYPE=%s/%s", + srctype->super, srctype->type); + snprintf(cups_datadir, sizeof(cups_datadir), "CUPS_DATADIR=%s", DataDir); + snprintf(cups_fontpath, sizeof(cups_fontpath), "CUPS_FONTPATH=%s", FontPath); + snprintf(cups_serverbin, sizeof(cups_serverbin), "CUPS_SERVERBIN=%s", + ServerBin); + snprintf(cups_serverroot, sizeof(cups_serverroot), "CUPS_SERVERROOT=%s", + ServerRoot); + language = cupsLangDefault(); + snprintf(lang, sizeof(lang), "LANG=%s.UTF8", language->language); + snprintf(path, sizeof(path), "PATH=%s", Path); + if (ppdfile) + snprintf(ppd, sizeof(ppd), "PPD=%s", ppdfile); + else if ((temp = getenv("PPD")) != NULL) + snprintf(ppd, sizeof(ppd), "PPD=%s", temp); + else +#ifdef __APPLE__ + if (!access("/System/Library/Frameworks/ApplicationServices.framework/" + "Versions/A/Frameworks/PrintCore.framework/Versions/A/" + "Resources/English.lproj/Generic.ppd", 0)) + strlcpy(ppd, "PPD=/System/Library/Frameworks/ApplicationServices.framework/" + "Versions/A/Frameworks/PrintCore.framework/Versions/A/" + "Resources/English.lproj/Generic.ppd", sizeof(ppd)); + else + strlcpy(ppd, "PPD=/System/Library/Frameworks/ApplicationServices.framework/" + "Versions/A/Frameworks/PrintCore.framework/Versions/A/" + "Resources/Generic.ppd", sizeof(ppd)); +#else + snprintf(ppd, sizeof(ppd), "PPD=%s/model/laserjet.ppd", DataDir); +#endif /* __APPLE__ */ + snprintf(rip_max_cache, sizeof(rip_max_cache), "RIP_MAX_CACHE=%s", RIPCache); + snprintf(userenv, sizeof(userenv), "USER=%s", user); + + if (printer && + (dest = cupsGetNamedDest(CUPS_HTTP_DEFAULT, printer, NULL)) != NULL) + { + if ((temp = cupsGetOption("printer-info", dest->num_options, + dest->options)) != NULL) + snprintf(printer_info, sizeof(printer_info), "PRINTER_INFO=%s", temp); + else + snprintf(printer_info, sizeof(printer_info), "PRINTER_INFO=%s", printer); + + if ((temp = cupsGetOption("printer-location", dest->num_options, + dest->options)) != NULL) + snprintf(printer_location, sizeof(printer_location), + "PRINTER_LOCATION=%s", temp); + else + strlcpy(printer_location, "PRINTER_LOCATION=Unknown", + sizeof(printer_location)); + } + else + { + snprintf(printer_info, sizeof(printer_info), "PRINTER_INFO=%s", + printer ? printer : "Unknown"); + strlcpy(printer_location, "PRINTER_LOCATION=Unknown", + sizeof(printer_location)); + } + + snprintf(printer_name, sizeof(printer_name), "PRINTER=%s", + printer ? printer : "Unknown"); + + argv[0] = (char *)printer; + argv[1] = "1"; + argv[2] = user; + argv[3] = title; + argv[4] = cupsGetOption("copies", num_options, options); + argv[5] = optstr; + argv[6] = infile; + argv[7] = NULL; + + if (!argv[4]) + argv[4] = "1"; + + envp[0] = ""; + envp[1] = content_type; + envp[2] = cups_datadir; + envp[3] = cups_fontpath; + envp[4] = cups_serverbin; + envp[5] = cups_serverroot; + envp[6] = lang; + envp[7] = path; + envp[8] = ppd; + envp[9] = printer_info; + envp[10] = printer_location; + envp[11] = printer_name; + envp[12] = rip_max_cache; + envp[13] = userenv; + envp[14] = NULL; + + for (i = 0; argv[i]; i ++) + fprintf(stderr, "DEBUG: argv[%d]=\"%s\"\n", i, argv[i]); + + for (i = 0; envp[i]; i ++) + fprintf(stderr, "DEBUG: envp[%d]=\"%s\"\n", i, envp[i]); + + /* + * Execute all of the filters... + */ + + pids = cupsArrayNew((cups_array_func_t)compare_pids, NULL); + current = 0; + filterfds[0][0] = -1; + filterfds[0][1] = -1; + filterfds[1][0] = -1; + filterfds[1][1] = -1; + + if (!infile) + filterfds[0][0] = 0; + + for (filter = (mime_filter_t *)cupsArrayFirst(filters); + filter; + filter = next, current = 1 - current) + { + next = (mime_filter_t *)cupsArrayNext(filters); + + if (filter->filter[0] == '/') + strlcpy(program, filter->filter, sizeof(program)); + else + snprintf(program, sizeof(program), "%s/filter/%s", ServerBin, + filter->filter); + + if (filterfds[!current][1] > 1) + { + close(filterfds[1 - current][0]); + close(filterfds[1 - current][1]); + + filterfds[1 - current][0] = -1; + filterfds[1 - current][0] = -1; + } + + if (next) + open_pipe(filterfds[1 - current]); + else if (outfile) + { + filterfds[1 - current][1] = open(outfile, O_CREAT | O_TRUNC | O_WRONLY, + 0666); + + if (filterfds[1 - current][1] < 0) + fprintf(stderr, "ERROR: Unable to create \"%s\" - %s\n", outfile, + strerror(errno)); + } + else + filterfds[1 - current][1] = 1; + + pid = exec_filter(program, (char **)argv, (char **)envp, + filterfds[current][0], filterfds[1 - current][1]); + + if (pid > 0) + { + fprintf(stderr, "INFO: %s (PID %d) started.\n", filter->filter, pid); + + filter->cost = pid; + cupsArrayAdd(pids, filter); + } + else + break; + + argv[6] = NULL; + } + + /* + * Close remaining pipes... + */ + + if (filterfds[0][1] > 1) + { + close(filterfds[0][0]); + close(filterfds[0][1]); + } + + if (filterfds[1][1] > 1) + { + close(filterfds[1][0]); + close(filterfds[1][1]); + } + + /* + * Wait for the children to exit... + */ + + retval = 0; + + while (cupsArrayCount(pids) > 0) + { + if ((pid = wait(&status)) < 0) + continue; + + key.cost = pid; + if ((filter = (mime_filter_t *)cupsArrayFind(pids, &key)) != NULL) + { + cupsArrayRemove(pids, filter); + + if (status) + { + if (WIFEXITED(status)) + fprintf(stderr, "ERROR: %s (PID %d) stopped with status %d\n", + filter->filter, pid, WEXITSTATUS(status)); + else + fprintf(stderr, "ERROR: %s (PID %d) crashed on signal %d\n", + filter->filter, pid, WTERMSIG(status)); + + retval = 1; + } + else + fprintf(stderr, "INFO: %s (PID %d) exited with no errors.\n", + filter->filter, pid); + } + } + + cupsArrayDelete(pids); + + return (retval); +} + + +/* + * 'get_job_file()' - Get the specified job file. + */ + +static void +get_job_file(const char *job) /* I - Job ID */ +{ + long jobid, /* Job ID */ + docnum; /* Document number */ + const char *jobptr; /* Pointer into job ID string */ + char uri[1024]; /* job-uri */ + http_t *http; /* Connection to server */ + ipp_t *request; /* Request data */ + int tempfd; /* Temporary file */ + + + /* + * Get the job ID and document number, if any... + */ + + if ((jobptr = strrchr(job, '-')) != NULL) + jobptr ++; + else + jobptr = job; + + jobid = strtol(jobptr, (char **)&jobptr, 10); + + if (*jobptr == ',') + docnum = strtol(jobptr + 1, NULL, 10); + else + docnum = 1; + + if (jobid < 1 || jobid > INT_MAX) + { + _cupsLangPrintf(stderr, _("cupsfilter: Invalid job ID %d."), (int)jobid); + exit(1); + } + + if (docnum < 1 || docnum > INT_MAX) + { + _cupsLangPrintf(stderr, _("cupsfilter: Invalid document number %d."), + (int)docnum); + exit(1); + } + + /* + * Ask the server for the document file... + */ + + if ((http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption())) == NULL) + { + _cupsLangPrintf(stderr, _("%s: Unable to connect to server."), + "cupsfilter"); + exit(1); + } + + request = ippNewRequest(CUPS_GET_DOCUMENT); + + snprintf(uri, sizeof(uri), "ipp://localhost/jobs/%d", (int)jobid); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-uri", NULL, uri); + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_INTEGER, "document-number", + (int)docnum); + + if ((tempfd = cupsTempFd(TempFile, sizeof(TempFile))) == -1) + { + _cupsLangPrintError("ERROR", _("Unable to create temporary file")); + httpClose(http); + exit(1); + } + + signal(SIGTERM, sighandler); + + ippDelete(cupsDoIORequest(http, request, "/", -1, tempfd)); + + close(tempfd); + + httpClose(http); + + if (cupsLastError() != IPP_OK) + { + _cupsLangPrintf(stderr, _("cupsfilter: Unable to get job file - %s"), + cupsLastErrorString()); + unlink(TempFile); + exit(1); + } +} + + +/* + * 'open_pipe()' - Create a pipe which is closed on exec. + */ + +static int /* O - 0 on success, -1 on error */ +open_pipe(int *fds) /* O - Pipe file descriptors (2) */ +{ + /* + * Create the pipe... + */ + + if (pipe(fds)) + { + fds[0] = -1; + fds[1] = -1; + + return (-1); + } + + /* + * Set the "close on exec" flag on each end of the pipe... + */ + + if (fcntl(fds[0], F_SETFD, fcntl(fds[0], F_GETFD) | FD_CLOEXEC)) + { + close(fds[0]); + close(fds[1]); + + fds[0] = -1; + fds[1] = -1; + + return (-1); + } + + if (fcntl(fds[1], F_SETFD, fcntl(fds[1], F_GETFD) | FD_CLOEXEC)) + { + close(fds[0]); + close(fds[1]); + + fds[0] = -1; + fds[1] = -1; + + return (-1); + } + + /* + * Return 0 indicating success... + */ + + return (0); +} + + +/* + * 'read_cupsd_conf()' - Read the cupsd.conf file to get the filter settings. + */ + +static int /* O - 0 on success, 1 on error */ +read_cupsd_conf(const char *filename) /* I - File to read */ +{ + cups_file_t *fp; /* cupsd.conf file */ + const char *temp; /* Temporary string */ + char line[1024], /* Line from file */ + *ptr; /* Pointer into line */ + int linenum; /* Current line number */ + + + if ((temp = getenv("CUPS_DATADIR")) != NULL) + set_string(&DataDir, temp); + else + set_string(&DataDir, CUPS_DATADIR); + + if ((temp = getenv("CUPS_FONTPATH")) != NULL) + set_string(&FontPath, temp); + else + set_string(&FontPath, CUPS_FONTPATH); + + set_string(&RIPCache, "128m"); + + if ((temp = getenv("CUPS_SERVERBIN")) != NULL) + set_string(&ServerBin, temp); + else + set_string(&ServerBin, CUPS_SERVERBIN); + + strlcpy(line, filename, sizeof(line)); + if ((ptr = strrchr(line, '/')) != NULL) + *ptr = '\0'; + else + getcwd(line, sizeof(line)); + + set_string(&ServerRoot, line); + + if ((fp = cupsFileOpen(filename, "r")) != NULL) + { + linenum = 0; + + while (cupsFileGetConf(fp, line, sizeof(line), &ptr, &linenum)) + { + if (!_cups_strcasecmp(line, "DataDir")) + set_string(&DataDir, ptr); + else if (!_cups_strcasecmp(line, "FontPath")) + set_string(&FontPath, ptr); + else if (!_cups_strcasecmp(line, "RIPCache")) + set_string(&RIPCache, ptr); + else if (!_cups_strcasecmp(line, "ServerBin")) + set_string(&ServerBin, ptr); + else if (!_cups_strcasecmp(line, "ServerRoot")) + set_string(&ServerRoot, ptr); + } + + cupsFileClose(fp); + } + + snprintf(line, sizeof(line), + "%s/filter:" CUPS_BINDIR ":" CUPS_SBINDIR ":/bin:/usr/bin", + ServerBin); + set_string(&Path, line); + + return (0); +} + + +/* + * 'set_string()' - Copy and set a string. + */ + +static void +set_string(char **s, /* O - Copy of string */ + const char *val) /* I - String to copy */ +{ + if (*s) + free(*s); + + *s = strdup(val); +} + + +/* + * 'sighandler()' - Signal catcher for when we print from stdin... + */ + +static void +sighandler(int s) /* I - Signal number */ +{ + /* + * Remove the temporary file we're using to print a job file... + */ + + if (TempFile[0]) + unlink(TempFile); + + /* + * Exit... + */ + + exit(s); +} + + +/* + * 'usage()' - Show program usage... + */ + +static void +usage(const char *command, /* I - Command name */ + const char *opt) /* I - Incorrect option, if any */ +{ + if (opt) + _cupsLangPrintf(stderr, _("%s: Unknown option \"%c\"."), command, *opt); + + if (!strcmp(command, "cupsfilter")) + { + _cupsLangPuts(stdout, _("Usage: cupsfilter [ options ] filename")); + _cupsLangPuts(stdout, _("Options:")); + _cupsLangPuts(stdout, _(" -D Remove the input file " + "when finished.")); + _cupsLangPuts(stdout, _(" -P filename.ppd Set PPD file.")); + _cupsLangPuts(stdout, _(" -U username Set username for job.")); + _cupsLangPuts(stdout, _(" -c cupsd.conf Set cupsd.conf file to " + "use.")); + _cupsLangPuts(stdout, _(" -d printer Use the named " + "printer.")); + _cupsLangPuts(stdout, _(" -e Use every filter from " + "the PPD file.")); + _cupsLangPuts(stdout, _(" -i mime/type Set input MIME type " + "(otherwise auto-typed).")); + _cupsLangPuts(stdout, _(" -j job-id[,N] Filter file N from the " + "specified job (default is file 1).")); + _cupsLangPuts(stdout, _(" -m mime/type Set output MIME type " + "(otherwise application/pdf).")); + _cupsLangPuts(stdout, _(" -n copies Set number of copies.")); + _cupsLangPuts(stdout, _(" -o name=value Set option(s).")); + _cupsLangPuts(stdout, _(" -p filename.ppd Set PPD file.")); + _cupsLangPuts(stdout, _(" -t title Set title.")); + _cupsLangPuts(stdout, _(" -u Remove the PPD file " + "when finished.")); + } + else + { + _cupsLangPuts(stdout, _("Usage: convert [ options ]")); + _cupsLangPuts(stdout, _("Options:")); + _cupsLangPuts(stdout, _(" -D Remove the input file " + "when finished.")); + _cupsLangPuts(stdout, _(" -J title Set title.")); + _cupsLangPuts(stdout, _(" -P filename.ppd Set PPD file.")); + _cupsLangPuts(stdout, _(" -U username Set username for job.")); + _cupsLangPuts(stdout, _(" -a 'name=value ...' Set option(s).")); + _cupsLangPuts(stdout, _(" -c copies Set number of copies.")); + _cupsLangPuts(stdout, _(" -d printer Use the named " + "printer.")); + _cupsLangPuts(stdout, _(" -e Use every filter from " + "the PPD file.")); + _cupsLangPuts(stdout, _(" -f filename Set file to be " + "converted (otherwise stdin).")); + _cupsLangPuts(stdout, _(" -i mime/type Set input MIME type " + "(otherwise auto-typed).")); + _cupsLangPuts(stdout, _(" -j mime/type Set output MIME type " + "(otherwise application/pdf).")); + _cupsLangPuts(stdout, _(" -o filename Set file to be " + "generated (otherwise stdout).")); + _cupsLangPuts(stdout, _(" -u Remove the PPD file " + "when finished.")); + } + + exit(1); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/dirsvc.c b/scheduler/dirsvc.c new file mode 100644 index 0000000000..72950a0c6f --- /dev/null +++ b/scheduler/dirsvc.c @@ -0,0 +1,1392 @@ +/* + * "$Id$" + * + * Directory services routines for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdDeregisterPrinter() - Stop sending broadcast information for a local + * printer and remove any pending references to + * remote printers. + * cupsdRegisterPrinter() - Start sending broadcast information for a + * printer or update the broadcast contents. + * cupsdStartBrowsing() - Start sending and receiving broadcast + * information. + * cupsdStopBrowsing() - Stop sending and receiving broadcast + * information. + * cupsdUpdateDNSSDName() - Update the computer name we use for browsing... + * dnssdAddAlias() - Add a DNS-SD alias name. + * dnssdBuildTxtRecord() - Build a TXT record from printer info. + * dnssdDeregisterPrinter() - Stop sending broadcast information for a + * printer. + * dnssdPackTxtRecord() - Pack an array of key/value pairs into the TXT + * record format. + * dnssdRegisterCallback() - DNSServiceRegister callback. + * dnssdRegisterPrinter() - Start sending broadcast information for a + * printer or update the broadcast contents. + * dnssdStop() - Stop all DNS-SD registrations. + * dnssdUpdate() - Handle DNS-SD queries. + * get_auth_info_required() - Get the auth-info-required value to advertise. + * get_hostconfig() - Get an /etc/hostconfig service setting. + * update_lpd() - Update the LPD configuration as needed. + * update_smb() - Update the SMB configuration as needed. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" +#include + +#ifdef HAVE_DNSSD +# include +# ifdef __APPLE__ +# include +# ifdef HAVE_COREFOUNDATION +# include +# endif /* HAVE_COREFOUNDATION */ +# ifdef HAVE_SYSTEMCONFIGURATION +# include +# endif /* HAVE_SYSTEMCONFIGURATION */ +# endif /* __APPLE__ */ +#endif /* HAVE_DNSSD */ + + +/* + * Local functions... + */ + +#ifdef HAVE_DNSSD +static char *get_auth_info_required(cupsd_printer_t *p, char *buffer, + size_t bufsize); +#endif /* HAVE_DNSSD */ +#ifdef __APPLE__ +static int get_hostconfig(const char *name); +#endif /* __APPLE__ */ +static void update_lpd(int onoff); +static void update_smb(int onoff); + + +#ifdef HAVE_DNSSD +# ifdef HAVE_COREFOUNDATION +static void dnssdAddAlias(const void *key, const void *value, + void *context); +# endif /* HAVE_COREFOUNDATION */ +static char *dnssdBuildTxtRecord(int *txt_len, cupsd_printer_t *p, + int for_lpd); +static void dnssdDeregisterPrinter(cupsd_printer_t *p); +static char *dnssdPackTxtRecord(int *txt_len, char *keyvalue[][2], + int count); +static void dnssdRegisterCallback(DNSServiceRef sdRef, + DNSServiceFlags flags, + DNSServiceErrorType errorCode, + const char *name, const char *regtype, + const char *domain, void *context); +static void dnssdRegisterPrinter(cupsd_printer_t *p); +static void dnssdStop(void); +static void dnssdUpdate(void); +#endif /* HAVE_DNSSD */ + + +/* + * 'cupsdDeregisterPrinter()' - Stop sending broadcast information for a + * local printer and remove any pending + * references to remote printers. + */ + +void +cupsdDeregisterPrinter( + cupsd_printer_t *p, /* I - Printer to register */ + int removeit) /* I - Printer being permanently removed */ +{ + /* + * Only deregister if browsing is enabled and it's a local printer... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "cupsdDeregisterPrinter(p=%p(%s), removeit=%d)", p, p->name, + removeit); + + if (!Browsing || !p->shared || + (p->type & (CUPS_PRINTER_REMOTE | CUPS_PRINTER_SCANNER))) + return; + + /* + * Announce the deletion... + */ + +#ifdef HAVE_DNSSD + if (removeit && (BrowseLocalProtocols & BROWSE_DNSSD) && DNSSDRef) + dnssdDeregisterPrinter(p); +#endif /* HAVE_DNSSD */ +} + + +/* + * 'cupsdRegisterPrinter()' - Start sending broadcast information for a + * printer or update the broadcast contents. + */ + +void +cupsdRegisterPrinter(cupsd_printer_t *p)/* I - Printer */ +{ + cupsdLogMessage(CUPSD_LOG_DEBUG, "cupsdRegisterPrinter(p=%p(%s))", p, + p->name); + + if (!Browsing || !BrowseLocalProtocols || + (p->type & (CUPS_PRINTER_REMOTE | CUPS_PRINTER_SCANNER))) + return; + +#ifdef HAVE_DNSSD + if ((BrowseLocalProtocols & BROWSE_DNSSD) && DNSSDRef) + dnssdRegisterPrinter(p); +#endif /* HAVE_DNSSD */ +} + + +/* + * 'cupsdStartBrowsing()' - Start sending and receiving broadcast information. + */ + +void +cupsdStartBrowsing(void) +{ + cupsd_printer_t *p; /* Current printer */ + + + if (!Browsing || !BrowseLocalProtocols) + return; + +#ifdef HAVE_DNSSD + if (BrowseLocalProtocols & BROWSE_DNSSD) + { + DNSServiceErrorType error; /* Error from service creation */ + cupsd_listener_t *lis; /* Current listening socket */ + + + /* + * First create a "master" connection for all registrations... + */ + + if ((error = DNSServiceCreateConnection(&DNSSDRef)) + != kDNSServiceErr_NoError) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to create master DNS-SD reference: %d", error); + + if (FatalErrors & CUPSD_FATAL_BROWSE) + cupsdEndProcess(getpid(), 0); + } + else + { + /* + * Add the master connection to the select list... + */ + + int fd = DNSServiceRefSockFD(DNSSDRef); + + fcntl(fd, F_SETFD, fcntl(fd, F_GETFD) | FD_CLOEXEC); + + cupsdAddSelect(fd, (cupsd_selfunc_t)dnssdUpdate, NULL, NULL); + + /* + * Then get the port we use for registrations. If we are not listening + * on any non-local ports, there is no sense sharing local printers via + * Bonjour... + */ + + DNSSDPort = 0; + + for (lis = (cupsd_listener_t *)cupsArrayFirst(Listeners); + lis; + lis = (cupsd_listener_t *)cupsArrayNext(Listeners)) + { + if (httpAddrLocalhost(&(lis->address))) + continue; + + DNSSDPort = _httpAddrPort(&(lis->address)); + break; + } + + /* + * Set the computer name and register the web interface... + */ + + cupsdUpdateDNSSDName(); + } + } +#endif /* HAVE_DNSSD */ + + /* + * Enable LPD and SMB printer sharing as needed through external programs... + */ + + if (BrowseLocalProtocols & BROWSE_LPD) + update_lpd(1); + + if (BrowseLocalProtocols & BROWSE_SMB) + update_smb(1); + + /* + * Register the individual printers + */ + + for (p = (cupsd_printer_t *)cupsArrayFirst(Printers); + p; + p = (cupsd_printer_t *)cupsArrayNext(Printers)) + if (!(p->type & (CUPS_PRINTER_REMOTE | CUPS_PRINTER_SCANNER))) + cupsdRegisterPrinter(p); +} + + +/* + * 'cupsdStopBrowsing()' - Stop sending and receiving broadcast information. + */ + +void +cupsdStopBrowsing(void) +{ + cupsd_printer_t *p; /* Current printer */ + + + if (!Browsing || !BrowseLocalProtocols) + return; + + /* + * De-register the individual printers + */ + + for (p = (cupsd_printer_t *)cupsArrayFirst(Printers); + p; + p = (cupsd_printer_t *)cupsArrayNext(Printers)) + if (!(p->type & (CUPS_PRINTER_REMOTE | CUPS_PRINTER_SCANNER))) + cupsdDeregisterPrinter(p, 1); + + /* + * Shut down browsing sockets... + */ + +#ifdef HAVE_DNSSD + if ((BrowseLocalProtocols & BROWSE_DNSSD) && DNSSDRef) + dnssdStop(); +#endif /* HAVE_DNSSD */ + + /* + * Disable LPD and SMB printer sharing as needed through external programs... + */ + + if (BrowseLocalProtocols & BROWSE_LPD) + update_lpd(0); + + if (BrowseLocalProtocols & BROWSE_SMB) + update_smb(0); +} + + +#ifdef HAVE_DNSSD +/* + * 'cupsdUpdateDNSSDName()' - Update the computer name we use for browsing... + */ + +void +cupsdUpdateDNSSDName(void) +{ + DNSServiceErrorType error; /* Error from service creation */ + char webif[1024]; /* Web interface share name */ +# ifdef HAVE_SYSTEMCONFIGURATION + SCDynamicStoreRef sc; /* Context for dynamic store */ + CFDictionaryRef btmm; /* Back-to-My-Mac domains */ + CFStringEncoding nameEncoding; /* Encoding of computer name */ + CFStringRef nameRef; /* Host name CFString */ + char nameBuffer[1024]; /* C-string buffer */ +# endif /* HAVE_SYSTEMCONFIGURATION */ + + + /* + * Only share the web interface and printers when non-local listening is + * enabled... + */ + + if (!DNSSDPort) + return; + + /* + * Get the computer name as a c-string... + */ + +# ifdef HAVE_SYSTEMCONFIGURATION + sc = SCDynamicStoreCreate(kCFAllocatorDefault, CFSTR("cupsd"), NULL, NULL); + + if (sc) + { + /* + * Get the computer name from the dynamic store... + */ + + cupsdClearString(&DNSSDComputerName); + + if ((nameRef = SCDynamicStoreCopyComputerName(sc, &nameEncoding)) != NULL) + { + if (CFStringGetCString(nameRef, nameBuffer, sizeof(nameBuffer), + kCFStringEncodingUTF8)) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Dynamic store computer name is \"%s\".", nameBuffer); + cupsdSetString(&DNSSDComputerName, nameBuffer); + } + + CFRelease(nameRef); + } + + if (!DNSSDComputerName) + { + /* + * Use the ServerName instead... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Using ServerName \"%s\" as computer name.", ServerName); + cupsdSetString(&DNSSDComputerName, ServerName); + } + + /* + * Get the local hostname from the dynamic store... + */ + + cupsdClearString(&DNSSDHostName); + + if ((nameRef = SCDynamicStoreCopyLocalHostName(sc)) != NULL) + { + if (CFStringGetCString(nameRef, nameBuffer, sizeof(nameBuffer), + kCFStringEncodingUTF8)) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Dynamic store host name is \"%s\".", nameBuffer); + cupsdSetString(&DNSSDHostName, nameBuffer); + } + + CFRelease(nameRef); + } + + if (!DNSSDHostName) + { + /* + * Use the ServerName instead... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Using ServerName \"%s\" as host name.", ServerName); + cupsdSetString(&DNSSDHostName, ServerName); + } + + /* + * Get any Back-to-My-Mac domains and add them as aliases... + */ + + cupsdFreeAliases(DNSSDAlias); + DNSSDAlias = NULL; + + btmm = SCDynamicStoreCopyValue(sc, CFSTR("Setup:/Network/BackToMyMac")); + if (btmm && CFGetTypeID(btmm) == CFDictionaryGetTypeID()) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, "%d Back to My Mac aliases to add.", + (int)CFDictionaryGetCount(btmm)); + CFDictionaryApplyFunction(btmm, dnssdAddAlias, NULL); + } + else if (btmm) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Bad Back to My Mac data in dynamic store!"); + else + cupsdLogMessage(CUPSD_LOG_DEBUG, "No Back to My Mac aliases to add."); + + if (btmm) + CFRelease(btmm); + + CFRelease(sc); + } + else +# endif /* HAVE_SYSTEMCONFIGURATION */ + { + cupsdSetString(&DNSSDComputerName, ServerName); + cupsdSetString(&DNSSDHostName, ServerName); + } + + /* + * Then (re)register the web interface if enabled... + */ + + if (BrowseWebIF) + { + if (DNSSDComputerName) + snprintf(webif, sizeof(webif), "CUPS @ %s", DNSSDComputerName); + else + strlcpy(webif, "CUPS Web Interface", sizeof(webif)); + + if (WebIFRef) + DNSServiceRefDeallocate(WebIFRef); + + WebIFRef = DNSSDRef; + if ((error = DNSServiceRegister(&WebIFRef, + kDNSServiceFlagsShareConnection, + 0, webif, "_http._tcp", NULL, + NULL, htons(DNSSDPort), 7, + "\006path=/", dnssdRegisterCallback, + NULL)) != kDNSServiceErr_NoError) + cupsdLogMessage(CUPSD_LOG_ERROR, + "DNS-SD web interface registration failed: %d", error); + } +} +#endif /* HAVE_DNSSD */ + + +#ifdef HAVE_DNSSD +# ifdef HAVE_COREFOUNDATION +/* + * 'dnssdAddAlias()' - Add a DNS-SD alias name. + */ + +static void +dnssdAddAlias(const void *key, /* I - Key */ + const void *value, /* I - Value (domain) */ + void *context) /* I - Unused */ +{ + char valueStr[1024], /* Domain string */ + hostname[1024], /* Complete hostname */ + *hostptr; /* Pointer into hostname */ + + + (void)key; + (void)context; + + if (CFGetTypeID((CFStringRef)value) == CFStringGetTypeID() && + CFStringGetCString((CFStringRef)value, valueStr, sizeof(valueStr), + kCFStringEncodingUTF8)) + { + snprintf(hostname, sizeof(hostname), "%s.%s", DNSSDHostName, valueStr); + hostptr = hostname + strlen(hostname) - 1; + if (*hostptr == '.') + *hostptr = '\0'; /* Strip trailing dot */ + + if (!DNSSDAlias) + DNSSDAlias = cupsArrayNew(NULL, NULL); + + cupsdAddAlias(DNSSDAlias, hostname); + cupsdLogMessage(CUPSD_LOG_DEBUG, "Added Back to My Mac ServerAlias %s", + hostname); + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Bad Back to My Mac domain in dynamic store!"); +} +# endif /* HAVE_COREFOUNDATION */ + + +/* + * 'dnssdBuildTxtRecord()' - Build a TXT record from printer info. + */ + +static char * /* O - TXT record */ +dnssdBuildTxtRecord( + int *txt_len, /* O - TXT record length */ + cupsd_printer_t *p, /* I - Printer information */ + int for_lpd) /* I - 1 = LPD, 0 = IPP */ +{ + int i; /* Looping var */ + char admin_hostname[256], /* .local hostname for admin page */ + adminurl_str[256], /* URL for the admin page */ + type_str[32], /* Type to string buffer */ + state_str[32], /* State to string buffer */ + rp_str[1024], /* Queue name string buffer */ + air_str[1024], /* auth-info-required string buffer */ + *keyvalue[32][2]; /* Table of key/value pairs */ + + + /* + * Load up the key value pairs... + */ + + i = 0; + + keyvalue[i ][0] = "txtvers"; + keyvalue[i++][1] = "1"; + + keyvalue[i ][0] = "qtotal"; + keyvalue[i++][1] = "1"; + + keyvalue[i ][0] = "rp"; + keyvalue[i++][1] = rp_str; + if (for_lpd) + strlcpy(rp_str, p->name, sizeof(rp_str)); + else + snprintf(rp_str, sizeof(rp_str), "%s/%s", + (p->type & CUPS_PRINTER_CLASS) ? "classes" : "printers", p->name); + + keyvalue[i ][0] = "ty"; + keyvalue[i++][1] = p->make_model ? p->make_model : "Unknown"; + + snprintf(admin_hostname, sizeof(admin_hostname), "%s.local.", DNSSDHostName); + httpAssembleURIf(HTTP_URI_CODING_ALL, adminurl_str, sizeof(adminurl_str), + "http", NULL, admin_hostname, DNSSDPort, "/%s/%s", + (p->type & CUPS_PRINTER_CLASS) ? "classes" : "printers", + p->name); + keyvalue[i ][0] = "adminurl"; + keyvalue[i++][1] = adminurl_str; + + keyvalue[i ][0] = "note"; + keyvalue[i++][1] = p->location ? p->location : ""; + + keyvalue[i ][0] = "priority"; + keyvalue[i++][1] = for_lpd ? "100" : "0"; + + keyvalue[i ][0] = "product"; + keyvalue[i++][1] = p->pc && p->pc->product ? p->pc->product : "Unknown"; + + keyvalue[i ][0] = "pdl"; + keyvalue[i++][1] = p->pdl ? p->pdl : "application/postscript"; + + if (get_auth_info_required(p, air_str, sizeof(air_str))) + { + keyvalue[i ][0] = "air"; + keyvalue[i++][1] = air_str; + } + + keyvalue[i ][0] = "UUID"; + keyvalue[i++][1] = p->uuid + 9; + +#ifdef HAVE_SSL + keyvalue[i ][0] = "TLS"; + keyvalue[i++][1] = "1.2"; +#endif /* HAVE_SSL */ + + keyvalue[i ][0] = "Transparent"; + keyvalue[i++][1] = "F"; + + keyvalue[i ][0] = "Binary"; + keyvalue[i++][1] = "F"; + + keyvalue[i ][0] = "Fax"; + keyvalue[i++][1] = (p->type & CUPS_PRINTER_FAX) ? "T" : "F"; + + keyvalue[i ][0] = "Color"; + keyvalue[i++][1] = (p->type & CUPS_PRINTER_COLOR) ? "T" : "F"; + + keyvalue[i ][0] = "Duplex"; + keyvalue[i++][1] = (p->type & CUPS_PRINTER_DUPLEX) ? "T" : "F"; + + keyvalue[i ][0] = "Staple"; + keyvalue[i++][1] = (p->type & CUPS_PRINTER_STAPLE) ? "T" : "F"; + + keyvalue[i ][0] = "Copies"; + keyvalue[i++][1] = (p->type & CUPS_PRINTER_COPIES) ? "T" : "F"; + + keyvalue[i ][0] = "Collate"; + keyvalue[i++][1] = (p->type & CUPS_PRINTER_COLLATE) ? "T" : "F"; + + keyvalue[i ][0] = "Punch"; + keyvalue[i++][1] = (p->type & CUPS_PRINTER_PUNCH) ? "T" : "F"; + + keyvalue[i ][0] = "Bind"; + keyvalue[i++][1] = (p->type & CUPS_PRINTER_BIND) ? "T" : "F"; + + keyvalue[i ][0] = "Sort"; + keyvalue[i++][1] = (p->type & CUPS_PRINTER_SORT) ? "T" : "F"; + + keyvalue[i ][0] = "Scan"; + keyvalue[i++][1] = (p->type & CUPS_PRINTER_MFP) ? "T" : "F"; + + snprintf(type_str, sizeof(type_str), "0x%X", p->type | CUPS_PRINTER_REMOTE); + snprintf(state_str, sizeof(state_str), "%d", p->state); + + keyvalue[i ][0] = "printer-state"; + keyvalue[i++][1] = state_str; + + keyvalue[i ][0] = "printer-type"; + keyvalue[i++][1] = type_str; + + /* + * Then pack them into a proper txt record... + */ + + return (dnssdPackTxtRecord(txt_len, keyvalue, i)); +} + + +/* + * 'dnssdDeregisterPrinter()' - Stop sending broadcast information for a + * printer. + */ + +static void +dnssdDeregisterPrinter( + cupsd_printer_t *p) /* I - Printer */ +{ + cupsdLogMessage(CUPSD_LOG_DEBUG2, "dnssdDeregisterPrinter(%s)", p->name); + + /* + * Closing the socket deregisters the service + */ + + if (p->ipp_ref) + { + DNSServiceRefDeallocate(p->ipp_ref); + p->ipp_ref = NULL; + } + + if (p->ipp_txt) + { + /* + * p->ipp_txt is malloc'd, not _cupsStrAlloc'd... + */ + + free(p->ipp_txt); + p->ipp_txt = NULL; + } + + if (p->printer_ref) + { + DNSServiceRefDeallocate(p->printer_ref); + p->printer_ref = NULL; + } + + if (p->printer_txt) + { + /* + * p->printer_txt is malloc'd, not _cupsStrAlloc'd... + */ + + free(p->printer_txt); + p->printer_txt = NULL; + } + + /* + * Remove the printer from the array of DNS-SD printers, then clear the + * registered name... + */ + + cupsArrayRemove(DNSSDPrinters, p); + cupsdClearString(&p->reg_name); +} + + +/* + * 'dnssdPackTxtRecord()' - Pack an array of key/value pairs into the + * TXT record format. + */ + +static char * /* O - TXT record */ +dnssdPackTxtRecord(int *txt_len, /* O - TXT record length */ + char *keyvalue[][2], /* I - Table of key value pairs */ + int count) /* I - Items in table */ +{ + int i; /* Looping var */ + int length; /* Length of TXT record */ + int length2; /* Length of value */ + char *txtRecord; /* TXT record buffer */ + char *cursor; /* Looping pointer */ + + + /* + * Calculate the buffer size + */ + + if (count <= 0) + return (NULL); + + for (length = i = 0; i < count; i++) + length += 1 + strlen(keyvalue[i][0]) + + (keyvalue[i][1] ? 1 + strlen(keyvalue[i][1]) : 0); + + /* + * Allocate and fill it + */ + + txtRecord = malloc(length); + if (txtRecord) + { + *txt_len = length; + + for (cursor = txtRecord, i = 0; i < count; i++) + { + /* + * Drop in the p-string style length byte followed by the data + */ + + length = strlen(keyvalue[i][0]); + length2 = keyvalue[i][1] ? 1 + strlen(keyvalue[i][1]) : 0; + + *cursor++ = (unsigned char)(length + length2); + + memcpy(cursor, keyvalue[i][0], length); + cursor += length; + + if (length2) + { + length2 --; + *cursor++ = '='; + memcpy(cursor, keyvalue[i][1], length2); + cursor += length2; + } + } + } + + return (txtRecord); +} + + +/* + * 'dnssdRegisterCallback()' - DNSServiceRegister callback. + */ + +static void +dnssdRegisterCallback( + DNSServiceRef sdRef, /* I - DNS Service reference */ + DNSServiceFlags flags, /* I - Reserved for future use */ + DNSServiceErrorType errorCode, /* I - Error code */ + const char *name, /* I - Service name */ + const char *regtype, /* I - Service type */ + const char *domain, /* I - Domain. ".local" for now */ + void *context) /* I - User-defined context */ +{ + cupsd_printer_t *p = (cupsd_printer_t *)context; + /* Current printer */ + + + (void)sdRef; + (void)flags; + (void)domain; + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "dnssdRegisterCallback(%s, %s) for %s (%s)", + name, regtype, p ? p->name : "Web Interface", + p ? (p->reg_name ? p->reg_name : "(null)") : "NA"); + + if (errorCode) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "DNSServiceRegister failed with error %d", (int)errorCode); + return; + } + else if (p && (!p->reg_name || _cups_strcasecmp(name, p->reg_name))) + { + cupsdLogMessage(CUPSD_LOG_INFO, "Using service name \"%s\" for \"%s\"", + name, p->name); + + cupsArrayRemove(DNSSDPrinters, p); + cupsdSetString(&p->reg_name, name); + cupsArrayAdd(DNSSDPrinters, p); + + LastEvent |= CUPSD_EVENT_PRINTER_MODIFIED; + } +} + + +/* + * 'dnssdRegisterPrinter()' - Start sending broadcast information for a printer + * or update the broadcast contents. + */ + +static void +dnssdRegisterPrinter(cupsd_printer_t *p)/* I - Printer */ +{ + DNSServiceErrorType se; /* dnssd errors */ + char *ipp_txt, /* IPP TXT record buffer */ + *printer_txt, /* LPD TXT record buffer */ + name[1024], /* Service name */ + *nameptr; /* Pointer into name */ + int ipp_len, /* IPP TXT record length */ + printer_len, /* LPD TXT record length */ + printer_port; /* LPD port number */ + const char *regtype; /* Registration type */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "dnssdRegisterPrinter(%s) %s", p->name, + !p->ipp_ref ? "new" : "update"); + + /* + * If per-printer sharing was just disabled make sure we're not + * registered before returning. + */ + + if (!p->shared) + { + dnssdDeregisterPrinter(p); + return; + } + + /* + * The registered name takes the form of " @ "... + */ + + if (p->info && strlen(p->info) > 0) + { + if (DNSSDComputerName) + snprintf(name, sizeof(name), "%s @ %s", p->info, DNSSDComputerName); + else + strlcpy(name, p->info, sizeof(name)); + } + else if (DNSSDComputerName) + snprintf(name, sizeof(name), "%s @ %s", p->name, DNSSDComputerName); + else + strlcpy(name, p->name, sizeof(name)); + + /* + * If an existing printer was renamed, unregister it and start over... + */ + + if (p->reg_name && strcmp(p->reg_name, name)) + dnssdDeregisterPrinter(p); + + if (!p->reg_name) + { + cupsdSetString(&p->reg_name, name); + cupsArrayAdd(DNSSDPrinters, p); + } + + /* + * Register IPP and (optionally) LPD... + */ + + ipp_len = 0; /* anti-compiler-warning-code */ + ipp_txt = dnssdBuildTxtRecord(&ipp_len, p, 0); + + if (p->ipp_ref && + (ipp_len != p->ipp_len || memcmp(ipp_txt, p->ipp_txt, ipp_len))) + { + /* + * Update the existing registration... + */ + + /* A TTL of 0 means use record's original value (Radar 3176248) */ + if ((se = DNSServiceUpdateRecord(p->ipp_ref, NULL, 0, ipp_len, ipp_txt, + 0)) == kDNSServiceErr_NoError) + { + if (p->ipp_txt) + free(p->ipp_txt); + + p->ipp_txt = ipp_txt; + p->ipp_len = ipp_len; + ipp_txt = NULL; + } + else + { + /* + * Failed to update record, lets close this reference and move on... + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to update IPP DNS-SD record for %s - %d", p->name, + se); + + DNSServiceRefDeallocate(p->ipp_ref); + p->ipp_ref = NULL; + } + } + + if (!p->ipp_ref) + { + /* + * Initial registration. Use the _fax-ipp regtype for fax queues... + */ + + regtype = (p->type & CUPS_PRINTER_FAX) ? "_fax-ipp._tcp" : DNSSDRegType; + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Registering DNS-SD printer %s with name \"%s\" and " + "type \"%s\"", p->name, name, regtype); + + /* + * Register the queue, dropping characters as needed until we succeed... + */ + + nameptr = name + strlen(name); + + do + { + p->ipp_ref = DNSSDRef; + if ((se = DNSServiceRegister(&p->ipp_ref, kDNSServiceFlagsShareConnection, + 0, name, regtype, NULL, NULL, + htons(DNSSDPort), ipp_len, ipp_txt, + dnssdRegisterCallback, + p)) == kDNSServiceErr_BadParam) + { + /* + * Name is too long, drop trailing characters, taking into account + * UTF-8 encoding... + */ + + nameptr --; + + while (nameptr > name && (*nameptr & 0xc0) == 0x80) + nameptr --; + + if (nameptr > name) + *nameptr = '\0'; + } + } + while (se == kDNSServiceErr_BadParam && nameptr > name); + + if (se == kDNSServiceErr_NoError) + { + p->ipp_txt = ipp_txt; + p->ipp_len = ipp_len; + ipp_txt = NULL; + } + else + cupsdLogMessage(CUPSD_LOG_WARN, + "DNS-SD IPP registration of \"%s\" failed: %d", + p->name, se); + } + + if (ipp_txt) + free(ipp_txt); + + if (BrowseLocalProtocols & BROWSE_LPD) + { + printer_len = 0; /* anti-compiler-warning-code */ + printer_port = 515; + printer_txt = dnssdBuildTxtRecord(&printer_len, p, 1); + } + else + { + printer_len = 0; + printer_port = 0; + printer_txt = NULL; + } + + if (p->printer_ref && + (printer_len != p->printer_len || + memcmp(printer_txt, p->printer_txt, printer_len))) + { + /* + * Update the existing registration... + */ + + /* A TTL of 0 means use record's original value (Radar 3176248) */ + if ((se = DNSServiceUpdateRecord(p->printer_ref, NULL, 0, printer_len, + printer_txt, + 0)) == kDNSServiceErr_NoError) + { + if (p->printer_txt) + free(p->printer_txt); + + p->printer_txt = printer_txt; + p->printer_len = printer_len; + printer_txt = NULL; + } + else + { + /* + * Failed to update record, lets close this reference and move on... + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to update LPD DNS-SD record for %s - %d", + p->name, se); + + DNSServiceRefDeallocate(p->printer_ref); + p->printer_ref = NULL; + } + } + + if (!p->printer_ref) + { + /* + * Initial registration... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Registering DNS-SD printer %s with name \"%s\" and " + "type \"_printer._tcp\"", p->name, name); + + p->printer_ref = DNSSDRef; + if ((se = DNSServiceRegister(&p->printer_ref, + kDNSServiceFlagsShareConnection, + 0, name, "_printer._tcp", NULL, NULL, + htons(printer_port), printer_len, printer_txt, + dnssdRegisterCallback, + p)) == kDNSServiceErr_NoError) + { + p->printer_txt = printer_txt; + p->printer_len = printer_len; + printer_txt = NULL; + } + else + cupsdLogMessage(CUPSD_LOG_WARN, + "DNS-SD LPD registration of \"%s\" failed: %d", + p->name, se); + } + + if (printer_txt) + free(printer_txt); +} + + +/* + * 'dnssdStop()' - Stop all DNS-SD registrations. + */ + +static void +dnssdStop(void) +{ + cupsd_printer_t *p; /* Current printer */ + + + /* + * De-register the individual printers + */ + + for (p = (cupsd_printer_t *)cupsArrayFirst(Printers); + p; + p = (cupsd_printer_t *)cupsArrayNext(Printers)) + dnssdDeregisterPrinter(p); + + /* + * Shutdown the rest of the service refs... + */ + + if (WebIFRef) + { + DNSServiceRefDeallocate(WebIFRef); + WebIFRef = NULL; + } + + cupsdRemoveSelect(DNSServiceRefSockFD(DNSSDRef)); + + DNSServiceRefDeallocate(DNSSDRef); + DNSSDRef = NULL; + + cupsArrayDelete(DNSSDPrinters); + DNSSDPrinters = NULL; + + DNSSDPort = 0; +} + + +/* + * 'dnssdUpdate()' - Handle DNS-SD queries. + */ + +static void +dnssdUpdate(void) +{ + DNSServiceErrorType sdErr; /* Service discovery error */ + + + if ((sdErr = DNSServiceProcessResult(DNSSDRef)) != kDNSServiceErr_NoError) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "DNS Service Discovery registration error %d!", + sdErr); + dnssdStop(); + } +} + + +/* + * 'get_auth_info_required()' - Get the auth-info-required value to advertise. + */ + +static char * /* O - String or NULL if none */ +get_auth_info_required( + cupsd_printer_t *p, /* I - Printer */ + char *buffer, /* I - Value buffer */ + size_t bufsize) /* I - Size of value buffer */ +{ + cupsd_location_t *auth; /* Pointer to authentication element */ + char resource[1024]; /* Printer/class resource path */ + + + /* + * If auth-info-required is set for this printer, return that... + */ + + if (p->num_auth_info_required > 0 && strcmp(p->auth_info_required[0], "none")) + { + int i; /* Looping var */ + char *bufptr; /* Pointer into buffer */ + + for (i = 0, bufptr = buffer; i < p->num_auth_info_required; i ++) + { + if (bufptr >= (buffer + bufsize - 2)) + break; + + if (i) + *bufptr++ = ','; + + strlcpy(bufptr, p->auth_info_required[i], bufsize - (bufptr - buffer)); + bufptr += strlen(bufptr); + } + + return (buffer); + } + + /* + * Figure out the authentication data requirements to advertise... + */ + + if (p->type & CUPS_PRINTER_CLASS) + snprintf(resource, sizeof(resource), "/classes/%s", p->name); + else + snprintf(resource, sizeof(resource), "/printers/%s", p->name); + + if ((auth = cupsdFindBest(resource, HTTP_POST)) == NULL || + auth->type == CUPSD_AUTH_NONE) + auth = cupsdFindPolicyOp(p->op_policy_ptr, IPP_PRINT_JOB); + + if (auth) + { + int auth_type; /* Authentication type */ + + if ((auth_type = auth->type) == CUPSD_AUTH_DEFAULT) + auth_type = cupsdDefaultAuthType(); + + switch (auth_type) + { + case CUPSD_AUTH_NONE : + return (NULL); + + case CUPSD_AUTH_NEGOTIATE : + strlcpy(buffer, "negotiate", bufsize); + break; + + default : + strlcpy(buffer, "username,password", bufsize); + break; + } + + return (buffer); + } + + return ("none"); +} +#endif /* HAVE_DNSSD */ + + +#ifdef __APPLE__ +/* + * 'get_hostconfig()' - Get an /etc/hostconfig service setting. + */ + +static int /* O - 1 for YES or AUTOMATIC, 0 for NO */ +get_hostconfig(const char *name) /* I - Name of service */ +{ + cups_file_t *fp; /* Hostconfig file */ + char line[1024], /* Line from file */ + *ptr; /* Pointer to value */ + int state = 1; /* State of service */ + + + /* + * Try opening the /etc/hostconfig file; if we can't open it, assume that + * the service is enabled/auto. + */ + + if ((fp = cupsFileOpen("/etc/hostconfig", "r")) != NULL) + { + /* + * Read lines from the file until we find the service... + */ + + while (cupsFileGets(fp, line, sizeof(line))) + { + if (line[0] == '#' || (ptr = strchr(line, '=')) == NULL) + continue; + + *ptr++ = '\0'; + + if (!_cups_strcasecmp(line, name)) + { + /* + * Found the service, see if it is set to "-NO-"... + */ + + if (!_cups_strncasecmp(ptr, "-NO-", 4)) + state = 0; + break; + } + } + + cupsFileClose(fp); + } + + return (state); +} +#endif /* __APPLE__ */ + + +/* + * 'update_lpd()' - Update the LPD configuration as needed. + */ + +static void +update_lpd(int onoff) /* - 1 = turn on, 0 = turn off */ +{ + if (!LPDConfigFile) + return; + +#ifdef __APPLE__ + /* + * Allow /etc/hostconfig CUPS_LPD service setting to override cupsd.conf + * setting for backwards-compatibility. + */ + + if (onoff && !get_hostconfig("CUPS_LPD")) + onoff = 0; +#endif /* __APPLE__ */ + + if (!strncmp(LPDConfigFile, "xinetd:///", 10)) + { + /* + * Enable/disable LPD via the xinetd.d config file for cups-lpd... + */ + + char newfile[1024]; /* New cups-lpd.N file */ + cups_file_t *ofp, /* Original file pointer */ + *nfp; /* New file pointer */ + char line[1024]; /* Line from file */ + + + snprintf(newfile, sizeof(newfile), "%s.N", LPDConfigFile + 9); + + if ((ofp = cupsFileOpen(LPDConfigFile + 9, "r")) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to open \"%s\" - %s", + LPDConfigFile + 9, strerror(errno)); + return; + } + + if ((nfp = cupsFileOpen(newfile, "w")) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to create \"%s\" - %s", + newfile, strerror(errno)); + cupsFileClose(ofp); + return; + } + + /* + * Copy all of the lines from the cups-lpd file... + */ + + while (cupsFileGets(ofp, line, sizeof(line))) + { + if (line[0] == '{') + { + cupsFilePrintf(nfp, "%s\n", line); + snprintf(line, sizeof(line), "\tdisable = %s", + onoff ? "no" : "yes"); + } + else if (!strstr(line, "disable =")) + cupsFilePrintf(nfp, "%s\n", line); + } + + cupsFileClose(nfp); + cupsFileClose(ofp); + rename(newfile, LPDConfigFile + 9); + } +#ifdef __APPLE__ + else if (!strncmp(LPDConfigFile, "launchd:///", 11)) + { + /* + * Enable/disable LPD via the launchctl command... + */ + + char *argv[5], /* Arguments for command */ + *envp[MAX_ENV]; /* Environment for command */ + int pid; /* Process ID */ + + + cupsdLoadEnv(envp, (int)(sizeof(envp) / sizeof(envp[0]))); + argv[0] = (char *)"launchctl"; + argv[1] = (char *)(onoff ? "load" : "unload"); + argv[2] = (char *)"-w"; + argv[3] = LPDConfigFile + 10; + argv[4] = NULL; + + cupsdStartProcess("/bin/launchctl", argv, envp, -1, -1, -1, -1, -1, 1, + NULL, NULL, &pid); + } +#endif /* __APPLE__ */ + else + cupsdLogMessage(CUPSD_LOG_INFO, "Unknown LPDConfigFile scheme!"); +} + + +/* + * 'update_smb()' - Update the SMB configuration as needed. + */ + +static void +update_smb(int onoff) /* I - 1 = turn on, 0 = turn off */ +{ + if (!SMBConfigFile) + return; + + if (!strncmp(SMBConfigFile, "samba:///", 9)) + { + /* + * Enable/disable SMB via the specified smb.conf config file... + */ + + char newfile[1024]; /* New smb.conf.N file */ + cups_file_t *ofp, /* Original file pointer */ + *nfp; /* New file pointer */ + char line[1024]; /* Line from file */ + int in_printers; /* In [printers] section? */ + + + snprintf(newfile, sizeof(newfile), "%s.N", SMBConfigFile + 8); + + if ((ofp = cupsFileOpen(SMBConfigFile + 8, "r")) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to open \"%s\" - %s", + SMBConfigFile + 8, strerror(errno)); + return; + } + + if ((nfp = cupsFileOpen(newfile, "w")) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to create \"%s\" - %s", + newfile, strerror(errno)); + cupsFileClose(ofp); + return; + } + + /* + * Copy all of the lines from the smb.conf file... + */ + + in_printers = 0; + + while (cupsFileGets(ofp, line, sizeof(line))) + { + if (in_printers && strstr(line, "printable =")) + snprintf(line, sizeof(line), " printable = %s", + onoff ? "yes" : "no"); + + cupsFilePrintf(nfp, "%s\n", line); + + if (line[0] == '[') + in_printers = !strcmp(line, "[printers]"); + } + + cupsFileClose(nfp); + cupsFileClose(ofp); + rename(newfile, SMBConfigFile + 8); + } + else + cupsdLogMessage(CUPSD_LOG_INFO, "Unknown SMBConfigFile scheme!"); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/dirsvc.h b/scheduler/dirsvc.h new file mode 100644 index 0000000000..9caaf3fc42 --- /dev/null +++ b/scheduler/dirsvc.h @@ -0,0 +1,77 @@ +/* + * "$Id$" + * + * Directory services definitions for the CUPS scheduler. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +/* + * Browse protocols... + */ + +#define BROWSE_DNSSD 1 /* DNS Service Discovery (aka Bonjour) */ +#define BROWSE_SMB 2 /* SMB/Samba */ +#define BROWSE_LPD 4 /* LPD via xinetd or launchd */ +#define BROWSE_ALL 7 /* All protocols */ + + +/* + * Globals... + */ + +VAR int Browsing VALUE(TRUE), + /* Whether or not browsing is enabled */ + BrowseWebIF VALUE(FALSE), + /* Whether the web interface is advertised */ + BrowseLocalProtocols + VALUE(BROWSE_ALL); + /* Protocols to support for local printers */ +#ifdef HAVE_DNSSD +VAR char *DNSSDComputerName VALUE(NULL), + /* Computer/server name */ + *DNSSDHostName VALUE(NULL), + /* Hostname */ + *DNSSDRegType VALUE(NULL); + /* Bonjour registration type */ +VAR cups_array_t *DNSSDAlias VALUE(NULL); + /* List of dynamic ServerAlias's */ +VAR int DNSSDPort VALUE(0); + /* Port number to register */ +VAR cups_array_t *DNSSDPrinters VALUE(NULL); + /* Printers we have registered */ +VAR DNSServiceRef DNSSDRef VALUE(NULL), + /* Master DNS-SD service reference */ + WebIFRef VALUE(NULL); + /* Service reference for the web interface */ +#endif /* HAVE_DNSSD */ + +VAR char *LPDConfigFile VALUE(NULL), + /* LPD configuration file */ + *SMBConfigFile VALUE(NULL); + /* SMB configuration file */ + + +/* + * Prototypes... + */ + +extern void cupsdDeregisterPrinter(cupsd_printer_t *p, int removeit); +extern void cupsdRegisterPrinter(cupsd_printer_t *p); +extern void cupsdStartBrowsing(void); +extern void cupsdStopBrowsing(void); +#ifdef HAVE_DNSSD +extern void cupsdUpdateDNSSDName(void); +#endif /* HAVE_DNSSD */ + + +/* + * End of "$Id$". + */ diff --git a/scheduler/env.c b/scheduler/env.c new file mode 100644 index 0000000000..19a942a070 --- /dev/null +++ b/scheduler/env.c @@ -0,0 +1,271 @@ +/* + * "$Id$" + * + * Environment management routines for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdInitEnv() - Initialize the current environment with standard + * variables. + * cupsdLoadEnv() - Copy common environment variables into an array. + * cupsdSetEnv() - Set a common environment variable. + * cupsdSetEnvf() - Set a formatted common environment variable. + * cupsdUpdateEnv() - Update the environment for the configured directories. + * clear_env() - Clear common environment variables. + * find_env() - Find a common environment variable. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" + + +/* + * Local globals... + */ + +static int num_common_env = 0; /* Number of common env vars */ +static char *common_env[MAX_ENV]; /* Common env vars */ + + +/* + * Local functions... + */ + +static void clear_env(void); +static int find_env(const char *name); + + +/* + * 'cupsdInitEnv()' - Initialize the current environment with standard variables. + */ + +void +cupsdInitEnv(void) +{ + /* + * Clear existing environment variables... + */ + + clear_env(); + +#if defined(__APPLE__) + /* + * Add special voodoo magic for MacOS X - this allows MacOS X + * programs to access their bundle resources properly... + * + * This string is replaced in cupsdStartProcess()... + */ + + cupsdSetString(common_env, ""); + num_common_env = 1; +#endif /* __APPLE__ */ +} + + +/* + * 'cupsdLoadEnv()' - Copy common environment variables into an array. + */ + +int /* O - Number of environment variables */ +cupsdLoadEnv(char *envp[], /* I - Environment array */ + int envmax) /* I - Maximum number of elements */ +{ + int i; /* Looping var */ + + + /* + * Leave room for a NULL pointer at the end... + */ + + envmax --; + + /* + * Copy pointers to the environment... + */ + + for (i = 0; i < num_common_env && i < envmax; i ++) + envp[i] = common_env[i]; + + /* + * NULL terminate the environment array and return the number of + * elements we added... + */ + + envp[i] = NULL; + + return (i); +} + + +/* + * 'cupsdSetEnv()' - Set a common environment variable. + */ + +void +cupsdSetEnv(const char *name, /* I - Name of variable */ + const char *value) /* I - Value of variable */ +{ + int i; /* Index into environent array */ + + + /* + * If "value" is NULL, try getting value from current environment... + */ + + if (!value) + value = getenv(name); + + if (!value) + return; + + /* + * See if this variable has already been defined... + */ + + if ((i = find_env(name)) < 0) + { + /* + * Check for room... + */ + + if (num_common_env >= (int)(sizeof(common_env) / sizeof(common_env[0]))) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "cupsdSetEnv: Too many environment variables set!"); + return; + } + + i = num_common_env; + num_common_env ++; + } + + /* + * Set the new environment variable... + */ + + cupsdSetStringf(common_env + i, "%s=%s", name, value); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdSetEnv: %s", common_env[i]); +} + + +/* + * 'cupsdSetEnvf()' - Set a formatted common environment variable. + */ + +void +cupsdSetEnvf(const char *name, /* I - Name of variable */ + const char *value, /* I - Printf-style value of variable */ + ...) /* I - Additional args as needed */ +{ + char v[4096]; /* Formatting string value */ + va_list ap; /* Argument pointer */ + + + /* + * Format the value string... + */ + + va_start(ap, value); + vsnprintf(v, sizeof(v), value, ap); + va_end(ap); + + /* + * Set the env variable... + */ + + cupsdSetEnv(name, v); +} + + +/* + * 'cupsdUpdateEnv()' - Update the environment for the configured directories. + */ + +void +cupsdUpdateEnv(void) +{ + /* + * Set common variables... + */ + +#define set_if_undefined(name,value) if (find_env(name) < 0) cupsdSetEnv(name,value) + + set_if_undefined("CUPS_CACHEDIR", CacheDir); + set_if_undefined("CUPS_DATADIR", DataDir); + set_if_undefined("CUPS_DOCROOT", DocumentRoot); + set_if_undefined("CUPS_FONTPATH", FontPath); + set_if_undefined("CUPS_REQUESTROOT", RequestRoot); + set_if_undefined("CUPS_SERVERBIN", ServerBin); + set_if_undefined("CUPS_SERVERROOT", ServerRoot); + set_if_undefined("CUPS_STATEDIR", StateDir); + set_if_undefined("DYLD_LIBRARY_PATH", NULL); + set_if_undefined("HOME", TempDir); + set_if_undefined("LD_ASSUME_KERNEL", NULL); + set_if_undefined("LD_LIBRARY_PATH", NULL); + set_if_undefined("LD_PRELOAD", NULL); + set_if_undefined("NLSPATH", NULL); + if (find_env("PATH") < 0) + cupsdSetEnvf("PATH", "%s/filter:" CUPS_BINDIR ":" CUPS_SBINDIR + ":/bin:/usr/bin", ServerBin); + set_if_undefined("SERVER_ADMIN", ServerAdmin); + set_if_undefined("SHLIB_PATH", NULL); + set_if_undefined("SOFTWARE", CUPS_MINIMAL); + set_if_undefined("TMPDIR", TempDir); + set_if_undefined("TZ", NULL); + set_if_undefined("USER", "root"); + set_if_undefined("VG_ARGS", NULL); +} + + +/* + * 'clear_env()' - Clear common environment variables. + */ + +static void +clear_env(void) +{ + int i; /* Looping var */ + + + for (i = 0; i < num_common_env; i ++) + cupsdClearString(common_env + i); + + num_common_env = 0; +} + + +/* + * 'find_env()' - Find a common environment variable. + */ + +static int /* O - Index or -1 if not found */ +find_env(const char *name) /* I - Variable name */ +{ + int i; /* Looping var */ + size_t namelen; /* Length of name */ + + + for (i = 0, namelen = strlen(name); i < num_common_env; i ++) + if (!strncmp(common_env[i], name, namelen) && common_env[i][namelen] == '=') + return (i); + + return (-1); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/file.c b/scheduler/file.c new file mode 100644 index 0000000000..8d33610c19 --- /dev/null +++ b/scheduler/file.c @@ -0,0 +1,450 @@ +/* + * "$Id$" + * + * File functions for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdCleanFiles() - Clean out old files. + * cupsdCloseCreatedConfFile() - Close a created configuration file and move + * into place. + * cupsdClosePipe() - Close a pipe as necessary. + * cupsdCreateConfFile() - Create a configuration file safely. + * cupsdOpenConfFile() - Open a configuration file. + * cupsdOpenPipe() - Create a pipe which is closed on exec. + * cupsdRemoveFile() - Remove a file using the 7-pass US DoD method. + * overwrite_data() - Overwrite the data in a file. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" +#include +#include +#ifdef HAVE_REMOVEFILE +# include +#else +static int overwrite_data(int fd, const char *buffer, int bufsize, + int filesize); +#endif /* HAVE_REMOVEFILE */ + + +/* + * 'cupsdCleanFiles()' - Clean out old files. + */ + +void +cupsdCleanFiles(const char *path, /* I - Directory to clean */ + const char *pattern) /* I - Filename pattern or NULL */ +{ + cups_dir_t *dir; /* Directory */ + cups_dentry_t *dent; /* Directory entry */ + char filename[1024]; /* Filename */ + int status; /* Status from unlink/rmdir */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "cupsdCleanFiles(path=\"%s\", pattern=\"%s\")", path, + pattern ? pattern : "(null)"); + + if ((dir = cupsDirOpen(path)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to open directory \"%s\" - %s", + path, strerror(errno)); + return; + } + + cupsdLogMessage(CUPSD_LOG_INFO, "Cleaning out old files in \"%s\"...", path); + + while ((dent = cupsDirRead(dir)) != NULL) + { + if (pattern && fnmatch(pattern, dent->filename, 0)) + continue; + + snprintf(filename, sizeof(filename), "%s/%s", path, dent->filename); + + if (S_ISDIR(dent->fileinfo.st_mode)) + { + cupsdCleanFiles(filename, pattern); + + status = rmdir(filename); + } + else + status = unlink(filename); + + if (status) + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to remove \"%s\" - %s", filename, + strerror(errno)); + else + cupsdLogMessage(CUPSD_LOG_DEBUG, "Removed \"%s\"...", filename); + } + + cupsDirClose(dir); +} + + +/* + * 'cupsdCloseCreatedConfFile()' - Close a created configuration file and move + * into place. + */ + +int /* O - 0 on success, -1 on error */ +cupsdCloseCreatedConfFile( + cups_file_t *fp, /* I - File to close */ + const char *filename) /* I - Filename */ +{ + char newfile[1024], /* filename.N */ + oldfile[1024]; /* filename.O */ + + + /* + * First close the file... + */ + + if (cupsFileClose(fp)) + return (-1); + + /* + * Then remove "filename.O", rename "filename" to "filename.O", and rename + * "filename.N" to "filename". + */ + + snprintf(newfile, sizeof(newfile), "%s.N", filename); + snprintf(oldfile, sizeof(oldfile), "%s.O", filename); + + if ((cupsdRemoveFile(oldfile) && errno != ENOENT) || + (rename(filename, oldfile) && errno != ENOENT) || + rename(newfile, filename)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to finalize \"%s\": %s", + filename, strerror(errno)); + return (-1); + } + + return (0); +} + + +/* + * 'cupsdClosePipe()' - Close a pipe as necessary. + */ + +void +cupsdClosePipe(int *fds) /* I - Pipe file descriptors (2) */ +{ + /* + * Close file descriptors as needed... + */ + + if (fds[0] >= 0) + { + close(fds[0]); + fds[0] = -1; + } + + if (fds[1] >= 0) + { + close(fds[1]); + fds[1] = -1; + } +} + + +/* + * 'cupsdCreateConfFile()' - Create a configuration file safely. + */ + +cups_file_t * /* O - File pointer */ +cupsdCreateConfFile( + const char *filename, /* I - Filename */ + mode_t mode) /* I - Permissions */ +{ + cups_file_t *fp; /* File pointer */ + char newfile[1024]; /* filename.N */ + + + snprintf(newfile, sizeof(newfile), "%s.N", filename); + if ((fp = cupsFileOpen(newfile, "w")) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to create \"%s\": %s", newfile, + strerror(errno)); + } + else + { + if (!getuid() && fchown(cupsFileNumber(fp), getuid(), Group)) + cupsdLogMessage(CUPSD_LOG_WARN, "Unable to change group for \"%s\": %s", + newfile, strerror(errno)); + + if (fchmod(cupsFileNumber(fp), mode)) + cupsdLogMessage(CUPSD_LOG_WARN, + "Unable to change permissions for \"%s\": %s", + newfile, strerror(errno)); + } + + return (fp); +} + + +/* + * 'cupsdOpenConfFile()' - Open a configuration file. + * + * This function looks for "filename.O" if "filename" does not exist and does + * a rename as needed. + */ + +cups_file_t * /* O - File pointer */ +cupsdOpenConfFile(const char *filename) /* I - Filename */ +{ + cups_file_t *fp; /* File pointer */ + + + if ((fp = cupsFileOpen(filename, "r")) == NULL) + { + if (errno == ENOENT) + { + /* + * Try opening the backup file... + */ + + char oldfile[1024]; /* filename.O */ + + snprintf(oldfile, sizeof(oldfile), "%s.O", filename); + fp = cupsFileOpen(oldfile, "r"); + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to open \"%s\": %s", filename, + strerror(errno)); + } + + return (fp); +} + + +/* + * 'cupsdOpenPipe()' - Create a pipe which is closed on exec. + */ + +int /* O - 0 on success, -1 on error */ +cupsdOpenPipe(int *fds) /* O - Pipe file descriptors (2) */ +{ + /* + * Create the pipe... + */ + + if (pipe(fds)) + { + fds[0] = -1; + fds[1] = -1; + + return (-1); + } + + /* + * Set the "close on exec" flag on each end of the pipe... + */ + + if (fcntl(fds[0], F_SETFD, fcntl(fds[0], F_GETFD) | FD_CLOEXEC)) + { + close(fds[0]); + close(fds[1]); + + fds[0] = -1; + fds[1] = -1; + + return (-1); + } + + if (fcntl(fds[1], F_SETFD, fcntl(fds[1], F_GETFD) | FD_CLOEXEC)) + { + close(fds[0]); + close(fds[1]); + + fds[0] = -1; + fds[1] = -1; + + return (-1); + } + + /* + * Return 0 indicating success... + */ + + return (0); +} + + +/* + * 'cupsdRemoveFile()' - Remove a file using the 7-pass US DoD method. + */ + +int /* O - 0 on success, -1 on error */ +cupsdRemoveFile(const char *filename) /* I - File to remove */ +{ +#ifdef HAVE_REMOVEFILE + return (removefile(filename, NULL, REMOVEFILE_SECURE_7_PASS)); + +#else + int fd; /* File descriptor */ + struct stat info; /* File information */ + char buffer[512]; /* Data buffer */ + int i; /* Looping var */ + + + /* + * First open the file for writing in exclusive mode. + */ + + if ((fd = open(filename, O_WRONLY | O_EXCL)) < 0) + return (-1); + + /* + * Delete the file now - it will still be around as long as the file is + * open... + */ + + if (unlink(filename)) + { + close(fd); + return (-1); + } + + /* + * Then get the file size... + */ + + if (fstat(fd, &info)) + { + close(fd); + return (-1); + } + + /* + * Overwrite the file 7 times with 0xF6, 0x00, 0xFF, random, 0x00, 0xFF, + * and more random data. + */ + + memset(buffer, 0xF6, sizeof(buffer)); + if (overwrite_data(fd, buffer, sizeof(buffer), (int)info.st_size)) + { + close(fd); + return (-1); + } + + memset(buffer, 0x00, sizeof(buffer)); + if (overwrite_data(fd, buffer, sizeof(buffer), (int)info.st_size)) + { + close(fd); + return (-1); + } + + memset(buffer, 0xFF, sizeof(buffer)); + if (overwrite_data(fd, buffer, sizeof(buffer), (int)info.st_size)) + { + close(fd); + return (-1); + } + + CUPS_SRAND(time(NULL)); + + for (i = 0; i < sizeof(buffer); i ++) + buffer[i] = CUPS_RAND(); + if (overwrite_data(fd, buffer, sizeof(buffer), (int)info.st_size)) + { + close(fd); + return (-1); + } + + memset(buffer, 0x00, sizeof(buffer)); + if (overwrite_data(fd, buffer, sizeof(buffer), (int)info.st_size)) + { + close(fd); + return (-1); + } + + memset(buffer, 0xFF, sizeof(buffer)); + if (overwrite_data(fd, buffer, sizeof(buffer), (int)info.st_size)) + { + close(fd); + return (-1); + } + + for (i = 0; i < sizeof(buffer); i ++) + buffer[i] = CUPS_RAND(); + if (overwrite_data(fd, buffer, sizeof(buffer), (int)info.st_size)) + { + close(fd); + return (-1); + } + + /* + * Whew! Close the file (which will lead to the actual deletion) and + * return success... + */ + + close(fd); + return (0); +#endif /* HAVE_REMOVEFILE */ +} + + +#ifndef HAVE_REMOVEFILE +/* + * 'overwrite_data()' - Overwrite the data in a file. + */ + +static int /* O - 0 on success, -1 on error */ +overwrite_data(int fd, /* I - File descriptor */ + const char *buffer, /* I - Buffer to write */ + int bufsize, /* I - Size of buffer */ + int filesize) /* I - Size of file */ +{ + int bytes; /* Bytes to write/written */ + + + /* + * Start at the beginning of the file... + */ + + if (lseek(fd, 0, SEEK_SET) < 0) + return (-1); + + /* + * Fill the file with the provided data... + */ + + while (filesize > 0) + { + if (filesize > bufsize) + bytes = bufsize; + else + bytes = filesize; + + if ((bytes = write(fd, buffer, bytes)) < 0) + return (-1); + + filesize -= bytes; + } + + /* + * Force the changes to disk... + */ + + return (fsync(fd)); +} +#endif /* HAVE_REMOVEFILE */ + + +/* + * End of "$Id$". + */ diff --git a/scheduler/filter.c b/scheduler/filter.c new file mode 100644 index 0000000000..d1446f6543 --- /dev/null +++ b/scheduler/filter.c @@ -0,0 +1,504 @@ +/* + * "$Id$" + * + * File type conversion routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * mimeAddFilter() - Add a filter to the current MIME database. + * mimeFilter() - Find the fastest way to convert from one type to + * another. + * mimeFilter2() - Find the fastest way to convert from one type to + * another, including the file size. + * mimeFilterLookup() - Lookup a filter. + * mime_compare_filters() - Compare two filters. + * mime_compare_srcs() - Compare two filter source types. + * mime_find_filters() - Find the filters to convert from one type to + * another. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include "mime.h" + + +/* + * Local types... + */ + +typedef struct _mime_typelist_s /**** List of source types ****/ +{ + struct _mime_typelist_s *next; /* Next source type */ + mime_type_t *src; /* Source type */ +} _mime_typelist_t; + + +/* + * Local functions... + */ + +static int mime_compare_filters(mime_filter_t *, mime_filter_t *); +static int mime_compare_srcs(mime_filter_t *, mime_filter_t *); +static cups_array_t *mime_find_filters(mime_t *mime, mime_type_t *src, + size_t srcsize, mime_type_t *dst, + int *cost, _mime_typelist_t *visited); + + +/* + * 'mimeAddFilter()' - Add a filter to the current MIME database. + */ + +mime_filter_t * /* O - New filter */ +mimeAddFilter(mime_t *mime, /* I - MIME database */ + mime_type_t *src, /* I - Source type */ + mime_type_t *dst, /* I - Destination type */ + int cost, /* I - Relative time/resource cost */ + const char *filter) /* I - Filter program to run */ +{ + mime_filter_t *temp; /* New filter */ + + + DEBUG_printf(("mimeAddFilter(mime=%p, src=%p(%s/%s), dst=%p(%s/%s), cost=%d, " + "filter=\"%s\")", mime, + src, src ? src->super : "???", src ? src->type : "???", + dst, dst ? dst->super : "???", dst ? dst->type : "???", + cost, filter)); + + /* + * Range-check the input... + */ + + if (!mime || !src || !dst || !filter) + { + DEBUG_puts("1mimeAddFilter: Returning NULL."); + return (NULL); + } + + /* + * See if we already have an existing filter for the given source and + * destination... + */ + + if ((temp = mimeFilterLookup(mime, src, dst)) != NULL) + { + /* + * Yup, does the existing filter have a higher cost? If so, copy the + * filter and cost to the existing filter entry and return it... + */ + + if (temp->cost > cost) + { + DEBUG_printf(("1mimeAddFilter: Replacing filter \"%s\", cost %d.", + temp->filter, temp->cost)); + temp->cost = cost; + strlcpy(temp->filter, filter, sizeof(temp->filter)); + } + } + else + { + /* + * Nope, add a new one... + */ + + if (!mime->filters) + mime->filters = cupsArrayNew((cups_array_func_t)mime_compare_filters, NULL); + + if (!mime->filters) + return (NULL); + + if ((temp = calloc(1, sizeof(mime_filter_t))) == NULL) + return (NULL); + + /* + * Copy the information over and sort if necessary... + */ + + temp->src = src; + temp->dst = dst; + temp->cost = cost; + strlcpy(temp->filter, filter, sizeof(temp->filter)); + + DEBUG_puts("1mimeAddFilter: Adding new filter."); + cupsArrayAdd(mime->filters, temp); + cupsArrayAdd(mime->srcs, temp); + } + + /* + * Return the new/updated filter... + */ + + DEBUG_printf(("1mimeAddFilter: Returning %p.", temp)); + + return (temp); +} + + +/* + * 'mimeFilter()' - Find the fastest way to convert from one type to another. + */ + +cups_array_t * /* O - Array of filters to run */ +mimeFilter(mime_t *mime, /* I - MIME database */ + mime_type_t *src, /* I - Source file type */ + mime_type_t *dst, /* I - Destination file type */ + int *cost) /* O - Cost of filters */ +{ + DEBUG_printf(("mimeFilter(mime=%p, src=%p(%s/%s), dst=%p(%s/%s), " + "cost=%p(%d))", mime, + src, src ? src->super : "???", src ? src->type : "???", + dst, dst ? dst->super : "???", dst ? dst->type : "???", + cost, cost ? *cost : 0)); + + return (mimeFilter2(mime, src, 0, dst, cost)); +} + + +/* + * 'mimeFilter2()' - Find the fastest way to convert from one type to another, + * including file size. + */ + +cups_array_t * /* O - Array of filters to run */ +mimeFilter2(mime_t *mime, /* I - MIME database */ + mime_type_t *src, /* I - Source file type */ + size_t srcsize, /* I - Size of source file */ + mime_type_t *dst, /* I - Destination file type */ + int *cost) /* O - Cost of filters */ +{ + cups_array_t *filters; /* Array of filters to run */ + + + /* + * Range-check the input... + */ + + DEBUG_printf(("mimeFilter2(mime=%p, src=%p(%s/%s), srcsize=" CUPS_LLFMT + ", dst=%p(%s/%s), cost=%p(%d))", mime, + src, src ? src->super : "???", src ? src->type : "???", + CUPS_LLCAST srcsize, + dst, dst ? dst->super : "???", dst ? dst->type : "???", + cost, cost ? *cost : 0)); + + if (cost) + *cost = 0; + + if (!mime || !src || !dst) + return (NULL); + + /* + * (Re)build the source lookup array as needed... + */ + + if (!mime->srcs) + { + mime_filter_t *current; /* Current filter */ + + mime->srcs = cupsArrayNew((cups_array_func_t)mime_compare_srcs, NULL); + + for (current = mimeFirstFilter(mime); + current; + current = mimeNextFilter(mime)) + cupsArrayAdd(mime->srcs, current); + } + + /* + * Find the filters... + */ + + filters = mime_find_filters(mime, src, srcsize, dst, cost, NULL); + + DEBUG_printf(("1mimeFilter2: Returning %d filter(s), cost %d:", + cupsArrayCount(filters), cost ? *cost : -1)); +#ifdef DEBUG + { + mime_filter_t *filter; /* Current filter */ + + for (filter = (mime_filter_t *)cupsArrayFirst(filters); + filter; + filter = (mime_filter_t *)cupsArrayNext(filters)) + DEBUG_printf(("1mimeFilter2: %s/%s %s/%s %d %s", filter->src->super, + filter->src->type, filter->dst->super, filter->dst->type, + filter->cost, filter->filter)); + } +#endif /* DEBUG */ + + return (filters); +} + + +/* + * 'mimeFilterLookup()' - Lookup a filter. + */ + +mime_filter_t * /* O - Filter for src->dst */ +mimeFilterLookup(mime_t *mime, /* I - MIME database */ + mime_type_t *src, /* I - Source type */ + mime_type_t *dst) /* I - Destination type */ +{ + mime_filter_t key, /* Key record for filter search */ + *filter; /* Matching filter */ + + + DEBUG_printf(("2mimeFilterLookup(mime=%p, src=%p(%s/%s), dst=%p(%s/%s))", mime, + src, src ? src->super : "???", src ? src->type : "???", + dst, dst ? dst->super : "???", dst ? dst->type : "???")); + + key.src = src; + key.dst = dst; + + filter = (mime_filter_t *)cupsArrayFind(mime->filters, &key); + DEBUG_printf(("3mimeFilterLookup: Returning %p(%s).", filter, + filter ? filter->filter : "???")); + return (filter); +} + + +/* + * 'mime_compare_filters()' - Compare two filters. + */ + +static int /* O - Comparison result */ +mime_compare_filters(mime_filter_t *f0, /* I - First filter */ + mime_filter_t *f1) /* I - Second filter */ +{ + int i; /* Result of comparison */ + + + if ((i = strcmp(f0->src->super, f1->src->super)) == 0) + if ((i = strcmp(f0->src->type, f1->src->type)) == 0) + if ((i = strcmp(f0->dst->super, f1->dst->super)) == 0) + i = strcmp(f0->dst->type, f1->dst->type); + + return (i); +} + + +/* + * 'mime_compare_srcs()' - Compare two filter source types. + */ + +static int /* O - Comparison result */ +mime_compare_srcs(mime_filter_t *f0, /* I - First filter */ + mime_filter_t *f1) /* I - Second filter */ +{ + int i; /* Result of comparison */ + + + if ((i = strcmp(f0->src->super, f1->src->super)) == 0) + i = strcmp(f0->src->type, f1->src->type); + + return (i); +} + + +/* + * 'mime_find_filters()' - Find the filters to convert from one type to another. + */ + +static cups_array_t * /* O - Array of filters to run */ +mime_find_filters( + mime_t *mime, /* I - MIME database */ + mime_type_t *src, /* I - Source file type */ + size_t srcsize, /* I - Size of source file */ + mime_type_t *dst, /* I - Destination file type */ + int *cost, /* O - Cost of filters */ + _mime_typelist_t *list) /* I - Source types we've used */ +{ + int tempcost, /* Temporary cost */ + mincost; /* Current minimum */ + cups_array_t *temp, /* Temporary filter */ + *mintemp; /* Current minimum */ + mime_filter_t *current, /* Current filter */ + srckey; /* Source type key */ + _mime_typelist_t listnode, /* New list node */ + *listptr; /* Pointer in list */ + + + DEBUG_printf(("2mime_find_filters(mime=%p, src=%p(%s/%s), srcsize=" CUPS_LLFMT + ", dst=%p(%s/%s), cost=%p, list=%p)", mime, src, src->super, + src->type, CUPS_LLCAST srcsize, dst, dst->super, dst->type, + cost, list)); + + /* + * See if there is a filter that can convert the files directly... + */ + + if ((current = mimeFilterLookup(mime, src, dst)) != NULL && + (current->maxsize == 0 || srcsize <= current->maxsize)) + { + /* + * Got a direct filter! + */ + + DEBUG_puts("3mime_find_filters: Direct filter found."); + + if ((mintemp = cupsArrayNew(NULL, NULL)) == NULL) + { + DEBUG_puts("3mime_find_filters: Returning NULL (out of memory)."); + return (NULL); + } + + cupsArrayAdd(mintemp, current); + + mincost = current->cost; + + if (!cost) + { + DEBUG_printf(("3mime_find_filters: Returning 1 filter, cost %d:", + mincost)); + DEBUG_printf(("3mime_find_filters: %s/%s %s/%s %d %s", + current->src->super, current->src->type, + current->dst->super, current->dst->type, + current->cost, current->filter)); + return (mintemp); + } + } + else + { + /* + * No direct filter... + */ + + mintemp = NULL; + mincost = 9999999; + } + + /* + * Initialize this node in the type list... + */ + + listnode.next = list; + + /* + * OK, now look for filters from the source type to any other type... + */ + + srckey.src = src; + + for (current = (mime_filter_t *)cupsArrayFind(mime->srcs, &srckey); + current && current->src == src; + current = (mime_filter_t *)cupsArrayNext(mime->srcs)) + { + /* + * See if we have already tried the destination type as a source + * type (this avoids extra filter looping...) + */ + + mime_type_t *current_dst; /* Current destination type */ + + if (current->maxsize > 0 && srcsize > current->maxsize) + continue; + + for (listptr = list, current_dst = current->dst; + listptr; + listptr = listptr->next) + if (current_dst == listptr->src) + break; + + if (listptr) + continue; + + /* + * See if we have any filters that can convert from the destination type + * of this filter to the final type... + */ + + listnode.src = current->src; + + cupsArraySave(mime->srcs); + temp = mime_find_filters(mime, current->dst, srcsize, dst, &tempcost, + &listnode); + cupsArrayRestore(mime->srcs); + + if (!temp) + continue; + + if (!cost) + { + DEBUG_printf(("3mime_find_filters: Returning %d filter(s), cost %d:", + cupsArrayCount(temp), tempcost)); + +#ifdef DEBUG + for (current = (mime_filter_t *)cupsArrayFirst(temp); + current; + current = (mime_filter_t *)cupsArrayNext(temp)) + DEBUG_printf(("3mime_find_filters: %s/%s %s/%s %d %s", + current->src->super, current->src->type, + current->dst->super, current->dst->type, + current->cost, current->filter)); +#endif /* DEBUG */ + + return (temp); + } + + /* + * Found a match; see if this one is less costly than the last (if + * any...) + */ + + tempcost += current->cost; + + if (tempcost < mincost) + { + cupsArrayDelete(mintemp); + + /* + * Hey, we got a match! Add the current filter to the beginning of the + * filter list... + */ + + mintemp = temp; + mincost = tempcost; + cupsArrayInsert(mintemp, current); + } + else + cupsArrayDelete(temp); + } + + if (mintemp) + { + /* + * Hey, we got a match! + */ + + DEBUG_printf(("3mime_find_filters: Returning %d filter(s), cost %d:", + cupsArrayCount(mintemp), mincost)); + +#ifdef DEBUG + for (current = (mime_filter_t *)cupsArrayFirst(mintemp); + current; + current = (mime_filter_t *)cupsArrayNext(mintemp)) + DEBUG_printf(("3mime_find_filters: %s/%s %s/%s %d %s", + current->src->super, current->src->type, + current->dst->super, current->dst->type, + current->cost, current->filter)); +#endif /* DEBUG */ + + if (cost) + *cost = mincost; + + return (mintemp); + } + + DEBUG_puts("3mime_find_filters: Returning NULL (no matches)."); + + return (NULL); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/ipp.c b/scheduler/ipp.c new file mode 100644 index 0000000000..3663eb05a8 --- /dev/null +++ b/scheduler/ipp.c @@ -0,0 +1,11989 @@ +/* + * "$Id$" + * + * IPP routines for the CUPS scheduler. + * + * Copyright 2007-2012 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * This file contains Kerberos support code, copyright 2006 by + * Jelmer Vernooij. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdProcessIPPRequest() - Process an incoming IPP request. + * cupsdTimeoutJob() - Timeout a job waiting on job files. + * accept_jobs() - Accept print jobs to a printer. + * add_class() - Add a class to the system. + * add_file() - Add a file to a job. + * add_job() - Add a job to a print queue. + * add_job_subscriptions() - Add any subscriptions for a job. + * add_job_uuid() - Add job-uuid attribute to a job. + * add_printer() - Add a printer to the system. + * add_printer_state_reasons() - Add the "printer-state-reasons" attribute + * based upon the printer state... + * add_queued_job_count() - Add the "queued-job-count" attribute for the + * specified printer or class. + * apple_init_profile() - Initialize a color profile. + * apple_register_profiles() - Register color profiles for a printer. + * apple_unregister_profiles() - Remove color profiles for the specified + * printer. + * apply_printer_defaults() - Apply printer default options to a job. + * authenticate_job() - Set job authentication info. + * cancel_all_jobs() - Cancel all or selected print jobs. + * cancel_job() - Cancel a print job. + * cancel_subscription() - Cancel a subscription. + * check_rss_recipient() - Check that we do not have a duplicate RSS + * feed URI. + * check_quotas() - Check quotas for a printer and user. + * close_job() - Close a multi-file job. + * copy_attrs() - Copy attributes from one request to another. + * copy_banner() - Copy a banner file to the requests directory + * for the specified job. + * copy_file() - Copy a PPD file or interface script... + * copy_model() - Copy a PPD model file, substituting default + * values as needed... + * copy_job_attrs() - Copy job attributes. + * copy_printer_attrs() - Copy printer attributes. + * copy_subscription_attrs() - Copy subscription attributes. + * create_job() - Print a file to a printer or class. + * create_requested_array() - Create an array for the requested-attributes. + * create_subscription() - Create a notification subscription. + * delete_printer() - Remove a printer or class from the system. + * get_default() - Get the default destination. + * get_devices() - Get the list of available devices on the + * local system. + * get_document() - Get a copy of a job file. + * get_job_attrs() - Get job attributes. + * get_jobs() - Get a list of jobs for the specified printer. + * get_notifications() - Get events for a subscription. + * get_ppd() - Get a named PPD from the local system. + * get_ppds() - Get the list of PPD files on the local + * system. + * get_printer_attrs() - Get printer attributes. + * get_printer_supported() - Get printer supported values. + * get_printers() - Get a list of printers or classes. + * get_subscription_attrs() - Get subscription attributes. + * get_subscriptions() - Get subscriptions. + * get_username() - Get the username associated with a request. + * hold_job() - Hold a print job. + * hold_new_jobs() - Hold pending/new jobs on a printer or class. + * move_job() - Move a job to a new destination. + * ppd_parse_line() - Parse a PPD default line. + * print_job() - Print a file to a printer or class. + * read_job_ticket() - Read a job ticket embedded in a print file. + * reject_jobs() - Reject print jobs to a printer. + * release_held_new_jobs() - Release pending/new jobs on a printer or + * class. + * release_job() - Release a held print job. + * renew_subscription() - Renew an existing subscription... + * restart_job() - Restart an old print job. + * save_auth_info() - Save authentication information for a job. + * send_document() - Send a file to a printer or class. + * send_http_error() - Send a HTTP error back to the IPP client. + * send_ipp_status() - Send a status back to the IPP client. + * set_default() - Set the default destination... + * set_job_attrs() - Set job attributes. + * set_printer_attrs() - Set printer attributes. + * set_printer_defaults() - Set printer default options from a request. + * start_printer() - Start a printer. + * stop_printer() - Stop a printer. + * url_encode_attr() - URL-encode a string attribute. + * url_encode_string() - URL-encode a string. + * user_allowed() - See if a user is allowed to print to a queue. + * validate_job() - Validate printer options and destination. + * validate_name() - Make sure the printer name only contains + * valid chars. + * validate_user() - Validate the user for the request. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" +#include + +#ifdef __APPLE__ +# include +# ifdef HAVE_COLORSYNCREGISTERDEVICE +extern CFUUIDRef ColorSyncCreateUUIDFromUInt32(unsigned id); +# endif /* HAVE_COLORSYNCREGISTERDEVICE */ +# include +# ifdef HAVE_MEMBERSHIP_H +# include +# endif /* HAVE_MEMBERSHIP_H */ +# ifdef HAVE_MEMBERSHIPPRIV_H +# include +# else +extern int mbr_user_name_to_uuid(const char* name, uuid_t uu); +extern int mbr_group_name_to_uuid(const char* name, uuid_t uu); +extern int mbr_check_membership_by_id(uuid_t user, gid_t group, int* ismember); +# endif /* HAVE_MEMBERSHIPPRIV_H */ +#endif /* __APPLE__ */ + + +/* + * Local functions... + */ + +static void accept_jobs(cupsd_client_t *con, ipp_attribute_t *uri); +static void add_class(cupsd_client_t *con, ipp_attribute_t *uri); +static int add_file(cupsd_client_t *con, cupsd_job_t *job, + mime_type_t *filetype, int compression); +static cupsd_job_t *add_job(cupsd_client_t *con, cupsd_printer_t *printer, + mime_type_t *filetype); +static void add_job_subscriptions(cupsd_client_t *con, cupsd_job_t *job); +static void add_job_uuid(cupsd_job_t *job); +static void add_printer(cupsd_client_t *con, ipp_attribute_t *uri); +static void add_printer_state_reasons(cupsd_client_t *con, + cupsd_printer_t *p); +static void add_queued_job_count(cupsd_client_t *con, cupsd_printer_t *p); +#ifdef __APPLE__ +static void apple_init_profile(ppd_file_t *ppd, cups_array_t *languages, +# ifdef HAVE_COLORSYNCREGISTERDEVICE + CFMutableDictionaryRef profile, +# else + CMDeviceProfileInfo *profile, +# endif /* HAVE_COLORSYNCREGISTERDEVICE */ + unsigned id, const char *name, + const char *text, const char *iccfile); +static void apple_register_profiles(cupsd_printer_t *p); +static void apple_unregister_profiles(cupsd_printer_t *p); +#endif /* __APPLE__ */ +static void apply_printer_defaults(cupsd_printer_t *printer, + cupsd_job_t *job); +static void authenticate_job(cupsd_client_t *con, ipp_attribute_t *uri); +static void cancel_all_jobs(cupsd_client_t *con, ipp_attribute_t *uri); +static void cancel_job(cupsd_client_t *con, ipp_attribute_t *uri); +static void cancel_subscription(cupsd_client_t *con, int id); +static int check_rss_recipient(const char *recipient); +static int check_quotas(cupsd_client_t *con, cupsd_printer_t *p); +static void close_job(cupsd_client_t *con, ipp_attribute_t *uri); +static void copy_attrs(ipp_t *to, ipp_t *from, cups_array_t *ra, + ipp_tag_t group, int quickcopy, + cups_array_t *exclude); +static int copy_banner(cupsd_client_t *con, cupsd_job_t *job, + const char *name); +static int copy_file(const char *from, const char *to); +static int copy_model(cupsd_client_t *con, const char *from, + const char *to); +static void copy_job_attrs(cupsd_client_t *con, + cupsd_job_t *job, + cups_array_t *ra, cups_array_t *exclude); +static void copy_printer_attrs(cupsd_client_t *con, + cupsd_printer_t *printer, + cups_array_t *ra); +static void copy_subscription_attrs(cupsd_client_t *con, + cupsd_subscription_t *sub, + cups_array_t *ra, + cups_array_t *exclude); +static void create_job(cupsd_client_t *con, ipp_attribute_t *uri); +static cups_array_t *create_requested_array(ipp_t *request); +static void create_subscription(cupsd_client_t *con, ipp_attribute_t *uri); +static void delete_printer(cupsd_client_t *con, ipp_attribute_t *uri); +static void get_default(cupsd_client_t *con); +static void get_devices(cupsd_client_t *con); +static void get_document(cupsd_client_t *con, ipp_attribute_t *uri); +static void get_jobs(cupsd_client_t *con, ipp_attribute_t *uri); +static void get_job_attrs(cupsd_client_t *con, ipp_attribute_t *uri); +static void get_notifications(cupsd_client_t *con); +static void get_ppd(cupsd_client_t *con, ipp_attribute_t *uri); +static void get_ppds(cupsd_client_t *con); +static void get_printers(cupsd_client_t *con, int type); +static void get_printer_attrs(cupsd_client_t *con, ipp_attribute_t *uri); +static void get_printer_supported(cupsd_client_t *con, ipp_attribute_t *uri); +static void get_subscription_attrs(cupsd_client_t *con, int sub_id); +static void get_subscriptions(cupsd_client_t *con, ipp_attribute_t *uri); +static const char *get_username(cupsd_client_t *con); +static void hold_job(cupsd_client_t *con, ipp_attribute_t *uri); +static void hold_new_jobs(cupsd_client_t *con, ipp_attribute_t *uri); +static void move_job(cupsd_client_t *con, ipp_attribute_t *uri); +static int ppd_parse_line(const char *line, char *option, int olen, + char *choice, int clen); +static void print_job(cupsd_client_t *con, ipp_attribute_t *uri); +static void read_job_ticket(cupsd_client_t *con); +static void reject_jobs(cupsd_client_t *con, ipp_attribute_t *uri); +static void release_held_new_jobs(cupsd_client_t *con, + ipp_attribute_t *uri); +static void release_job(cupsd_client_t *con, ipp_attribute_t *uri); +static void renew_subscription(cupsd_client_t *con, int sub_id); +static void restart_job(cupsd_client_t *con, ipp_attribute_t *uri); +static void save_auth_info(cupsd_client_t *con, cupsd_job_t *job, + ipp_attribute_t *auth_info); +static void send_document(cupsd_client_t *con, ipp_attribute_t *uri); +static void send_http_error(cupsd_client_t *con, http_status_t status, + cupsd_printer_t *printer); +static void send_ipp_status(cupsd_client_t *con, ipp_status_t status, + const char *message, ...) + __attribute__((__format__(__printf__, 3, 4))); +static void set_default(cupsd_client_t *con, ipp_attribute_t *uri); +static void set_job_attrs(cupsd_client_t *con, ipp_attribute_t *uri); +static void set_printer_attrs(cupsd_client_t *con, ipp_attribute_t *uri); +static void set_printer_defaults(cupsd_client_t *con, + cupsd_printer_t *printer); +static void start_printer(cupsd_client_t *con, ipp_attribute_t *uri); +static void stop_printer(cupsd_client_t *con, ipp_attribute_t *uri); +static void url_encode_attr(ipp_attribute_t *attr, char *buffer, + int bufsize); +static char *url_encode_string(const char *s, char *buffer, int bufsize); +static int user_allowed(cupsd_printer_t *p, const char *username); +static void validate_job(cupsd_client_t *con, ipp_attribute_t *uri); +static int validate_name(const char *name); +static int validate_user(cupsd_job_t *job, cupsd_client_t *con, + const char *owner, char *username, + int userlen); + + +/* + * 'cupsdProcessIPPRequest()' - Process an incoming IPP request. + */ + +int /* O - 1 on success, 0 on failure */ +cupsdProcessIPPRequest( + cupsd_client_t *con) /* I - Client connection */ +{ + ipp_tag_t group; /* Current group tag */ + ipp_attribute_t *attr; /* Current attribute */ + ipp_attribute_t *charset; /* Character set attribute */ + ipp_attribute_t *language; /* Language attribute */ + ipp_attribute_t *uri = NULL; /* Printer or job URI attribute */ + ipp_attribute_t *username; /* requesting-user-name attr */ + int sub_id; /* Subscription ID */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdProcessIPPRequest(%p[%d]): operation_id = %04x", + con, con->http.fd, con->request->request.op.operation_id); + + /* + * First build an empty response message for this request... + */ + + con->response = ippNew(); + + con->response->request.status.version[0] = + con->request->request.op.version[0]; + con->response->request.status.version[1] = + con->request->request.op.version[1]; + con->response->request.status.request_id = + con->request->request.op.request_id; + + /* + * Then validate the request header and required attributes... + */ + + if (con->request->request.any.version[0] != 1 && + con->request->request.any.version[0] != 2) + { + /* + * Return an error, since we only support IPP 1.x and 2.x. + */ + + cupsdAddEvent(CUPSD_EVENT_SERVER_AUDIT, NULL, NULL, + "%04X %s Bad request version number %d.%d", + IPP_VERSION_NOT_SUPPORTED, con->http.hostname, + con->request->request.any.version[0], + con->request->request.any.version[1]); + + send_ipp_status(con, IPP_VERSION_NOT_SUPPORTED, + _("Bad request version number %d.%d."), + con->request->request.any.version[0], + con->request->request.any.version[1]); + } + else if (con->request->request.any.request_id < 1) + { + /* + * Return an error, since request IDs must be between 1 and 2^31-1 + */ + + cupsdAddEvent(CUPSD_EVENT_SERVER_AUDIT, NULL, NULL, + "%04X %s Bad request ID %d", + IPP_BAD_REQUEST, con->http.hostname, + con->request->request.any.request_id); + + send_ipp_status(con, IPP_BAD_REQUEST, _("Bad request ID %d."), + con->request->request.any.request_id); + } + else if (!con->request->attrs) + { + cupsdAddEvent(CUPSD_EVENT_SERVER_AUDIT, NULL, NULL, + "%04X %s No attributes in request", + IPP_BAD_REQUEST, con->http.hostname); + + send_ipp_status(con, IPP_BAD_REQUEST, _("No attributes in request.")); + } + else + { + /* + * Make sure that the attributes are provided in the correct order and + * don't repeat groups... + */ + + for (attr = con->request->attrs, group = attr->group_tag; + attr; + attr = attr->next) + if (attr->group_tag < group && attr->group_tag != IPP_TAG_ZERO) + { + /* + * Out of order; return an error... + */ + + cupsdAddEvent(CUPSD_EVENT_SERVER_AUDIT, NULL, NULL, + "%04X %s Attribute groups are out of order", + IPP_BAD_REQUEST, con->http.hostname); + + send_ipp_status(con, IPP_BAD_REQUEST, + _("Attribute groups are out of order (%x < %x)."), + attr->group_tag, group); + break; + } + else + group = attr->group_tag; + + if (!attr) + { + /* + * Then make sure that the first three attributes are: + * + * attributes-charset + * attributes-natural-language + * printer-uri/job-uri + */ + + attr = con->request->attrs; + if (attr && attr->name && + !strcmp(attr->name, "attributes-charset") && + (attr->value_tag & IPP_TAG_MASK) == IPP_TAG_CHARSET) + charset = attr; + else + charset = NULL; + + if (attr) + attr = attr->next; + + if (attr && attr->name && + !strcmp(attr->name, "attributes-natural-language") && + (attr->value_tag & IPP_TAG_MASK) == IPP_TAG_LANGUAGE) + language = attr; + else + language = NULL; + + if ((attr = ippFindAttribute(con->request, "printer-uri", + IPP_TAG_URI)) != NULL) + uri = attr; + else if ((attr = ippFindAttribute(con->request, "job-uri", + IPP_TAG_URI)) != NULL) + uri = attr; + else if (con->request->request.op.operation_id == CUPS_GET_PPD) + uri = ippFindAttribute(con->request, "ppd-name", IPP_TAG_NAME); + else + uri = NULL; + + if (charset) + ippAddString(con->response, IPP_TAG_OPERATION, IPP_TAG_CHARSET, + "attributes-charset", NULL, + charset->values[0].string.text); + else + ippAddString(con->response, IPP_TAG_OPERATION, IPP_TAG_CHARSET, + "attributes-charset", NULL, "utf-8"); + + if (language) + ippAddString(con->response, IPP_TAG_OPERATION, IPP_TAG_LANGUAGE, + "attributes-natural-language", NULL, + language->values[0].string.text); + else + ippAddString(con->response, IPP_TAG_OPERATION, IPP_TAG_LANGUAGE, + "attributes-natural-language", NULL, DefaultLanguage); + + if (charset && + _cups_strcasecmp(charset->values[0].string.text, "us-ascii") && + _cups_strcasecmp(charset->values[0].string.text, "utf-8")) + { + /* + * Bad character set... + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, "Unsupported character set \"%s\"", + charset->values[0].string.text); + cupsdAddEvent(CUPSD_EVENT_SERVER_AUDIT, NULL, NULL, + "%04X %s Unsupported attributes-charset value \"%s\"", + IPP_CHARSET, con->http.hostname, + charset->values[0].string.text); + send_ipp_status(con, IPP_BAD_REQUEST, + _("Unsupported character set \"%s\"."), + charset->values[0].string.text); + } + else if (!charset || !language || + (!uri && + con->request->request.op.operation_id != CUPS_GET_DEFAULT && + con->request->request.op.operation_id != CUPS_GET_PRINTERS && + con->request->request.op.operation_id != CUPS_GET_CLASSES && + con->request->request.op.operation_id != CUPS_GET_DEVICES && + con->request->request.op.operation_id != CUPS_GET_PPDS)) + { + /* + * Return an error, since attributes-charset, + * attributes-natural-language, and printer-uri/job-uri are required + * for all operations. + */ + + if (!charset) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Missing attributes-charset attribute"); + + cupsdAddEvent(CUPSD_EVENT_SERVER_AUDIT, NULL, NULL, + "%04X %s Missing attributes-charset attribute", + IPP_BAD_REQUEST, con->http.hostname); + } + + if (!language) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Missing attributes-natural-language attribute"); + + cupsdAddEvent(CUPSD_EVENT_SERVER_AUDIT, NULL, NULL, + "%04X %s Missing attributes-natural-language attribute", + IPP_BAD_REQUEST, con->http.hostname); + } + + if (!uri) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Missing printer-uri, job-uri, or ppd-name " + "attribute"); + + cupsdAddEvent(CUPSD_EVENT_SERVER_AUDIT, NULL, NULL, + "%04X %s Missing printer-uri, job-uri, or ppd-name " + "attribute", IPP_BAD_REQUEST, con->http.hostname); + } + + cupsdLogMessage(CUPSD_LOG_DEBUG, "Request attributes follow..."); + + for (attr = con->request->attrs; attr; attr = attr->next) + cupsdLogMessage(CUPSD_LOG_DEBUG, + "attr \"%s\": group_tag = %x, value_tag = %x", + attr->name ? attr->name : "(null)", attr->group_tag, + attr->value_tag); + + cupsdLogMessage(CUPSD_LOG_DEBUG, "End of attributes..."); + + send_ipp_status(con, IPP_BAD_REQUEST, + _("Missing required attributes.")); + } + else + { + /* + * OK, all the checks pass so far; make sure requesting-user-name is + * not "root" from a remote host... + */ + + if ((username = ippFindAttribute(con->request, "requesting-user-name", + IPP_TAG_NAME)) != NULL) + { + /* + * Check for root user... + */ + + if (!strcmp(username->values[0].string.text, "root") && + _cups_strcasecmp(con->http.hostname, "localhost") && + strcmp(con->username, "root")) + { + /* + * Remote unauthenticated user masquerading as local root... + */ + + _cupsStrFree(username->values[0].string.text); + username->values[0].string.text = _cupsStrAlloc(RemoteRoot); + } + } + + if ((attr = ippFindAttribute(con->request, "notify-subscription-id", + IPP_TAG_INTEGER)) != NULL) + sub_id = attr->values[0].integer; + else + sub_id = 0; + + /* + * Then try processing the operation... + */ + + if (uri) + cupsdLogMessage(CUPSD_LOG_DEBUG, "%s %s", + ippOpString(con->request->request.op.operation_id), + uri->values[0].string.text); + else + cupsdLogMessage(CUPSD_LOG_DEBUG, "%s", + ippOpString(con->request->request.op.operation_id)); + + switch (con->request->request.op.operation_id) + { + case IPP_PRINT_JOB : + print_job(con, uri); + break; + + case IPP_VALIDATE_JOB : + validate_job(con, uri); + break; + + case IPP_CREATE_JOB : + create_job(con, uri); + break; + + case IPP_SEND_DOCUMENT : + send_document(con, uri); + break; + + case IPP_CANCEL_JOB : + cancel_job(con, uri); + break; + + case IPP_GET_JOB_ATTRIBUTES : + get_job_attrs(con, uri); + break; + + case IPP_GET_JOBS : + get_jobs(con, uri); + break; + + case IPP_GET_PRINTER_ATTRIBUTES : + get_printer_attrs(con, uri); + break; + + case IPP_GET_PRINTER_SUPPORTED_VALUES : + get_printer_supported(con, uri); + break; + + case IPP_HOLD_JOB : + hold_job(con, uri); + break; + + case IPP_RELEASE_JOB : + release_job(con, uri); + break; + + case IPP_RESTART_JOB : + restart_job(con, uri); + break; + + case IPP_PAUSE_PRINTER : + stop_printer(con, uri); + break; + + case IPP_RESUME_PRINTER : + start_printer(con, uri); + break; + + case IPP_PURGE_JOBS : + case IPP_CANCEL_JOBS : + case IPP_CANCEL_MY_JOBS : + cancel_all_jobs(con, uri); + break; + + case IPP_SET_JOB_ATTRIBUTES : + set_job_attrs(con, uri); + break; + + case IPP_SET_PRINTER_ATTRIBUTES : + set_printer_attrs(con, uri); + break; + + case IPP_HOLD_NEW_JOBS : + hold_new_jobs(con, uri); + break; + + case IPP_RELEASE_HELD_NEW_JOBS : + release_held_new_jobs(con, uri); + break; + + case IPP_CLOSE_JOB : + close_job(con, uri); + break; + + case CUPS_GET_DEFAULT : + get_default(con); + break; + + case CUPS_GET_PRINTERS : + get_printers(con, 0); + break; + + case CUPS_GET_CLASSES : + get_printers(con, CUPS_PRINTER_CLASS); + break; + + case CUPS_ADD_PRINTER : + add_printer(con, uri); + break; + + case CUPS_DELETE_PRINTER : + delete_printer(con, uri); + break; + + case CUPS_ADD_CLASS : + add_class(con, uri); + break; + + case CUPS_DELETE_CLASS : + delete_printer(con, uri); + break; + + case CUPS_ACCEPT_JOBS : + case IPP_ENABLE_PRINTER : + accept_jobs(con, uri); + break; + + case CUPS_REJECT_JOBS : + case IPP_DISABLE_PRINTER : + reject_jobs(con, uri); + break; + + case CUPS_SET_DEFAULT : + set_default(con, uri); + break; + + case CUPS_GET_DEVICES : + get_devices(con); + break; + + case CUPS_GET_DOCUMENT : + get_document(con, uri); + break; + + case CUPS_GET_PPD : + get_ppd(con, uri); + break; + + case CUPS_GET_PPDS : + get_ppds(con); + break; + + case CUPS_MOVE_JOB : + move_job(con, uri); + break; + + case CUPS_AUTHENTICATE_JOB : + authenticate_job(con, uri); + break; + + case IPP_CREATE_PRINTER_SUBSCRIPTION : + case IPP_CREATE_JOB_SUBSCRIPTION : + create_subscription(con, uri); + break; + + case IPP_GET_SUBSCRIPTION_ATTRIBUTES : + get_subscription_attrs(con, sub_id); + break; + + case IPP_GET_SUBSCRIPTIONS : + get_subscriptions(con, uri); + break; + + case IPP_RENEW_SUBSCRIPTION : + renew_subscription(con, sub_id); + break; + + case IPP_CANCEL_SUBSCRIPTION : + cancel_subscription(con, sub_id); + break; + + case IPP_GET_NOTIFICATIONS : + get_notifications(con); + break; + + default : + cupsdAddEvent(CUPSD_EVENT_SERVER_AUDIT, NULL, NULL, + "%04X %s Operation %04X (%s) not supported", + IPP_OPERATION_NOT_SUPPORTED, con->http.hostname, + con->request->request.op.operation_id, + ippOpString(con->request->request.op.operation_id)); + + send_ipp_status(con, IPP_OPERATION_NOT_SUPPORTED, + _("%s not supported."), + ippOpString( + con->request->request.op.operation_id)); + break; + } + } + } + } + + if (con->response) + { + /* + * Sending data from the scheduler... + */ + + cupsdLogMessage(con->response->request.status.status_code + >= IPP_BAD_REQUEST && + con->response->request.status.status_code + != IPP_NOT_FOUND ? CUPSD_LOG_ERROR : CUPSD_LOG_DEBUG, + "Returning IPP %s for %s (%s) from %s", + ippErrorString(con->response->request.status.status_code), + ippOpString(con->request->request.op.operation_id), + uri ? uri->values[0].string.text : "no URI", + con->http.hostname); + + if (LogLevel == CUPSD_LOG_DEBUG2) + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdProcessIPPRequest: ippLength(response)=%ld", + (long)ippLength(con->response)); + + if (cupsdSendHeader(con, HTTP_OK, "application/ipp", CUPSD_AUTH_NONE)) + { +#ifdef CUPSD_USE_CHUNKING + /* + * Because older versions of CUPS (1.1.17 and older) and some IPP + * clients do not implement chunking properly, we cannot use + * chunking by default. This may become the default in future + * CUPS releases, or we might add a configuration directive for + * it. + */ + + if (con->http.version == HTTP_1_1) + { + if (httpPrintf(HTTP(con), "Transfer-Encoding: chunked\r\n\r\n") < 0) + return (0); + + if (cupsdFlushHeader(con) < 0) + return (0); + + con->http.data_encoding = HTTP_ENCODE_CHUNKED; + } + else +#endif /* CUPSD_USE_CHUNKING */ + { + size_t length; /* Length of response */ + + + length = ippLength(con->response); + + if (con->file >= 0 && !con->pipe_pid) + { + struct stat fileinfo; /* File information */ + + + if (!fstat(con->file, &fileinfo)) + length += fileinfo.st_size; + } + + if (httpPrintf(HTTP(con), "Content-Length: " CUPS_LLFMT "\r\n\r\n", + CUPS_LLCAST length) < 0) + return (0); + + if (cupsdFlushHeader(con) < 0) + return (0); + + con->http.data_encoding = HTTP_ENCODE_LENGTH; + con->http.data_remaining = length; + + if (con->http.data_remaining <= INT_MAX) + con->http._data_remaining = con->http.data_remaining; + else + con->http._data_remaining = INT_MAX; + } + + cupsdAddSelect(con->http.fd, (cupsd_selfunc_t)cupsdReadClient, + (cupsd_selfunc_t)cupsdWriteClient, con); + + /* + * Tell the caller the response header was sent successfully... + */ + + return (1); + } + else + { + /* + * Tell the caller the response header could not be sent... + */ + + return (0); + } + } + else + { + /* + * Sending data from a subprocess like cups-deviced; tell the caller + * everything is A-OK so far... + */ + + return (1); + } +} + + +/* + * 'cupsdTimeoutJob()' - Timeout a job waiting on job files. + */ + +int /* O - 0 on success, -1 on error */ +cupsdTimeoutJob(cupsd_job_t *job) /* I - Job to timeout */ +{ + cupsd_printer_t *printer; /* Destination printer or class */ + ipp_attribute_t *attr; /* job-sheets attribute */ + int kbytes; /* Kilobytes in banner */ + + + job->pending_timeout = 0; + + /* + * See if we need to add the ending sheet... + */ + + if (!cupsdLoadJob(job)) + return (-1); + + printer = cupsdFindDest(job->dest); + attr = ippFindAttribute(job->attrs, "job-sheets", IPP_TAG_NAME); + + if (printer && !(printer->type & CUPS_PRINTER_REMOTE) && + attr && attr->num_values > 1) + { + /* + * Yes... + */ + + cupsdLogJob(job, CUPSD_LOG_INFO, "Adding end banner page \"%s\".", + attr->values[1].string.text); + + if ((kbytes = copy_banner(NULL, job, attr->values[1].string.text)) < 0) + return (-1); + + cupsdUpdateQuota(printer, job->username, 0, kbytes); + } + + return (0); +} + + +/* + * 'accept_jobs()' - Accept print jobs to a printer. + */ + +static void +accept_jobs(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Printer or class URI */ +{ + http_status_t status; /* Policy status */ + cups_ptype_t dtype; /* Destination type (printer/class) */ + cupsd_printer_t *printer; /* Printer data */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "accept_jobs(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * Is the destination valid? + */ + + if (!cupsdValidateDest(uri->values[0].string.text, &dtype, &printer)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(printer->op_policy_ptr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, printer); + return; + } + + /* + * Accept jobs sent to the printer... + */ + + printer->accepting = 1; + printer->state_message[0] = '\0'; + + cupsdAddEvent(CUPSD_EVENT_PRINTER_STATE, printer, NULL, + "Now accepting jobs."); + + if (dtype & CUPS_PRINTER_CLASS) + { + cupsdMarkDirty(CUPSD_DIRTY_CLASSES); + + cupsdLogMessage(CUPSD_LOG_INFO, "Class \"%s\" now accepting jobs (\"%s\").", + printer->name, get_username(con)); + } + else + { + cupsdMarkDirty(CUPSD_DIRTY_PRINTERS); + + cupsdLogMessage(CUPSD_LOG_INFO, + "Printer \"%s\" now accepting jobs (\"%s\").", + printer->name, get_username(con)); + } + + /* + * Everything was ok, so return OK status... + */ + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'add_class()' - Add a class to the system. + */ + +static void +add_class(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - URI of class */ +{ + http_status_t status; /* Policy status */ + int i; /* Looping var */ + char scheme[HTTP_MAX_URI], /* Method portion of URI */ + username[HTTP_MAX_URI], /* Username portion of URI */ + host[HTTP_MAX_URI], /* Host portion of URI */ + resource[HTTP_MAX_URI]; /* Resource portion of URI */ + int port; /* Port portion of URI */ + cupsd_printer_t *pclass, /* Class */ + *member; /* Member printer/class */ + cups_ptype_t dtype; /* Destination type */ + ipp_attribute_t *attr; /* Printer attribute */ + int modify; /* Non-zero if we just modified */ + int need_restart_job; /* Need to restart job? */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "add_class(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * Do we have a valid URI? + */ + + httpSeparateURI(HTTP_URI_CODING_ALL, uri->values[0].string.text, scheme, + sizeof(scheme), username, sizeof(username), host, + sizeof(host), &port, resource, sizeof(resource)); + + + if (strncmp(resource, "/classes/", 9) || strlen(resource) == 9) + { + /* + * No, return an error... + */ + + send_ipp_status(con, IPP_BAD_REQUEST, + _("The printer-uri must be of the form " + "\"ipp://HOSTNAME/classes/CLASSNAME\".")); + return; + } + + /* + * Do we have a valid printer name? + */ + + if (!validate_name(resource + 9)) + { + /* + * No, return an error... + */ + + send_ipp_status(con, IPP_BAD_REQUEST, + _("The printer-uri \"%s\" contains invalid characters."), + uri->values[0].string.text); + return; + } + + /* + * See if the class already exists; if not, create a new class... + */ + + if ((pclass = cupsdFindClass(resource + 9)) == NULL) + { + /* + * Class doesn't exist; see if we have a printer of the same name... + */ + + if ((pclass = cupsdFindPrinter(resource + 9)) != NULL) + { + /* + * Yes, return an error... + */ + + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("A printer named \"%s\" already exists."), + resource + 9); + return; + } + + /* + * No, check the default policy and then add the class... + */ + + if ((status = cupsdCheckPolicy(DefaultPolicyPtr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, NULL); + return; + } + + pclass = cupsdAddClass(resource + 9); + modify = 0; + } + else if ((status = cupsdCheckPolicy(pclass->op_policy_ptr, con, + NULL)) != HTTP_OK) + { + send_http_error(con, status, pclass); + return; + } + else + modify = 1; + + /* + * Look for attributes and copy them over as needed... + */ + + need_restart_job = 0; + + if ((attr = ippFindAttribute(con->request, "printer-location", + IPP_TAG_TEXT)) != NULL) + cupsdSetString(&pclass->location, attr->values[0].string.text); + + if ((attr = ippFindAttribute(con->request, "printer-info", + IPP_TAG_TEXT)) != NULL) + cupsdSetString(&pclass->info, attr->values[0].string.text); + + if ((attr = ippFindAttribute(con->request, "printer-is-accepting-jobs", + IPP_TAG_BOOLEAN)) != NULL && + attr->values[0].boolean != pclass->accepting) + { + cupsdLogMessage(CUPSD_LOG_INFO, + "Setting %s printer-is-accepting-jobs to %d (was %d.)", + pclass->name, attr->values[0].boolean, pclass->accepting); + + pclass->accepting = attr->values[0].boolean; + + cupsdAddEvent(CUPSD_EVENT_PRINTER_STATE, pclass, NULL, "%s accepting jobs.", + pclass->accepting ? "Now" : "No longer"); + } + + if ((attr = ippFindAttribute(con->request, "printer-is-shared", + IPP_TAG_BOOLEAN)) != NULL) + { + if (pclass->shared && !attr->values[0].boolean) + cupsdDeregisterPrinter(pclass, 1); + + cupsdLogMessage(CUPSD_LOG_INFO, + "Setting %s printer-is-shared to %d (was %d.)", + pclass->name, attr->values[0].boolean, pclass->shared); + + pclass->shared = attr->values[0].boolean; + } + + if ((attr = ippFindAttribute(con->request, "printer-state", + IPP_TAG_ENUM)) != NULL) + { + if (attr->values[0].integer != IPP_PRINTER_IDLE && + attr->values[0].integer != IPP_PRINTER_STOPPED) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Attempt to set %s printer-state to bad value %d."), + pclass->name, attr->values[0].integer); + return; + } + + cupsdLogMessage(CUPSD_LOG_INFO, "Setting %s printer-state to %d (was %d.)", + pclass->name, attr->values[0].integer, pclass->state); + + if (attr->values[0].integer == IPP_PRINTER_STOPPED) + cupsdStopPrinter(pclass, 0); + else + { + cupsdSetPrinterState(pclass, (ipp_pstate_t)(attr->values[0].integer), 0); + need_restart_job = 1; + } + } + if ((attr = ippFindAttribute(con->request, "printer-state-message", + IPP_TAG_TEXT)) != NULL) + { + strlcpy(pclass->state_message, attr->values[0].string.text, + sizeof(pclass->state_message)); + + cupsdAddEvent(CUPSD_EVENT_PRINTER_STATE, pclass, NULL, "%s", + pclass->state_message); + } + if ((attr = ippFindAttribute(con->request, "member-uris", + IPP_TAG_URI)) != NULL) + { + /* + * Clear the printer array as needed... + */ + + need_restart_job = 1; + + if (pclass->num_printers > 0) + { + free(pclass->printers); + pclass->num_printers = 0; + } + + /* + * Add each printer or class that is listed... + */ + + for (i = 0; i < attr->num_values; i ++) + { + /* + * Search for the printer or class URI... + */ + + if (!cupsdValidateDest(attr->values[i].string.text, &dtype, &member)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + else if (dtype & CUPS_PRINTER_CLASS) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Nested classes are not allowed.")); + return; + } + + /* + * Add it to the class... + */ + + cupsdAddPrinterToClass(pclass, member); + } + } + + set_printer_defaults(con, pclass); + + if ((attr = ippFindAttribute(con->request, "auth-info-required", + IPP_TAG_KEYWORD)) != NULL) + cupsdSetAuthInfoRequired(pclass, NULL, attr); + + /* + * Update the printer class attributes and return... + */ + + cupsdSetPrinterAttrs(pclass); + cupsdMarkDirty(CUPSD_DIRTY_CLASSES); + + if (need_restart_job && pclass->job) + { + /* + * Reset the current job to a "pending" status... + */ + + cupsdSetJobState(pclass->job, IPP_JOB_PENDING, CUPSD_JOB_FORCE, + "Job restarted because the class was modified."); + } + + cupsdMarkDirty(CUPSD_DIRTY_PRINTCAP); + + if (modify) + { + cupsdAddEvent(CUPSD_EVENT_PRINTER_MODIFIED, + pclass, NULL, "Class \"%s\" modified by \"%s\".", + pclass->name, get_username(con)); + + cupsdLogMessage(CUPSD_LOG_INFO, "Class \"%s\" modified by \"%s\".", + pclass->name, get_username(con)); + } + else + { + cupsdAddEvent(CUPSD_EVENT_PRINTER_ADDED, + pclass, NULL, "New class \"%s\" added by \"%s\".", + pclass->name, get_username(con)); + + cupsdLogMessage(CUPSD_LOG_INFO, "New class \"%s\" added by \"%s\".", + pclass->name, get_username(con)); + } + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'add_file()' - Add a file to a job. + */ + +static int /* O - 0 on success, -1 on error */ +add_file(cupsd_client_t *con, /* I - Connection to client */ + cupsd_job_t *job, /* I - Job to add to */ + mime_type_t *filetype, /* I - Type of file */ + int compression) /* I - Compression */ +{ + mime_type_t **filetypes; /* New filetypes array... */ + int *compressions; /* New compressions array... */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "add_file(con=%p[%d], job=%d, filetype=%s/%s, " + "compression=%d)", con, con ? con->http.fd : -1, job->id, + filetype->super, filetype->type, compression); + + /* + * Add the file to the job... + */ + + if (job->num_files == 0) + { + compressions = (int *)malloc(sizeof(int)); + filetypes = (mime_type_t **)malloc(sizeof(mime_type_t *)); + } + else + { + compressions = (int *)realloc(job->compressions, + (job->num_files + 1) * sizeof(int)); + filetypes = (mime_type_t **)realloc(job->filetypes, + (job->num_files + 1) * + sizeof(mime_type_t *)); + } + + if (!compressions || !filetypes) + { + cupsdSetJobState(job, IPP_JOB_ABORTED, CUPSD_JOB_PURGE, + "Job aborted because the scheduler ran out of memory."); + + if (con) + send_ipp_status(con, IPP_INTERNAL_ERROR, + _("Unable to allocate memory for file types.")); + + return (-1); + } + + job->compressions = compressions; + job->compressions[job->num_files] = compression; + job->filetypes = filetypes; + job->filetypes[job->num_files] = filetype; + + job->num_files ++; + + job->dirty = 1; + cupsdMarkDirty(CUPSD_DIRTY_JOBS); + + return (0); +} + + +/* + * 'add_job()' - Add a job to a print queue. + */ + +static cupsd_job_t * /* O - Job object */ +add_job(cupsd_client_t *con, /* I - Client connection */ + cupsd_printer_t *printer, /* I - Destination printer */ + mime_type_t *filetype) /* I - First print file type, if any */ +{ + http_status_t status; /* Policy status */ + ipp_attribute_t *attr, /* Current attribute */ + *auth_info; /* auth-info attribute */ + const char *val; /* Default option value */ + int priority; /* Job priority */ + cupsd_job_t *job; /* Current job */ + char job_uri[HTTP_MAX_URI]; /* Job URI */ + int kbytes; /* Size of print file */ + int i; /* Looping var */ + int lowerpagerange; /* Page range bound */ + int exact; /* Did we have an exact match? */ + ipp_attribute_t *media_col, /* media-col attribute */ + *media_margin; /* media-*-margin attribute */ + ipp_t *unsup_col; /* media-col in unsupported response */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "add_job(%p[%d], %p(%s), %p(%s/%s))", + con, con->http.fd, printer, printer->name, + filetype, filetype ? filetype->super : "none", + filetype ? filetype->type : "none"); + + /* + * Check remote printing to non-shared printer... + */ + + if (!printer->shared && + _cups_strcasecmp(con->http.hostname, "localhost") && + _cups_strcasecmp(con->http.hostname, ServerName)) + { + send_ipp_status(con, IPP_NOT_AUTHORIZED, + _("The printer or class is not shared.")); + return (NULL); + } + + /* + * Check policy... + */ + + auth_info = ippFindAttribute(con->request, "auth-info", IPP_TAG_TEXT); + + if ((status = cupsdCheckPolicy(printer->op_policy_ptr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, printer); + return (NULL); + } + else if (printer->num_auth_info_required == 1 && + !strcmp(printer->auth_info_required[0], "negotiate") && + !con->username[0]) + { + send_http_error(con, HTTP_UNAUTHORIZED, printer); + return (NULL); + } +#ifdef HAVE_SSL + else if (auth_info && !con->http.tls && + !httpAddrLocalhost(con->http.hostaddr)) + { + /* + * Require encryption of auth-info over non-local connections... + */ + + send_http_error(con, HTTP_UPGRADE_REQUIRED, printer); + return (NULL); + } +#endif /* HAVE_SSL */ + + /* + * See if the printer is accepting jobs... + */ + + if (!printer->accepting) + { + send_ipp_status(con, IPP_NOT_ACCEPTING, + _("Destination \"%s\" is not accepting jobs."), + printer->name); + return (NULL); + } + + /* + * Validate job template attributes; for now just document-format, + * copies, number-up, and page-ranges... + */ + + if (filetype && printer->filetypes && + !cupsArrayFind(printer->filetypes, filetype)) + { + char mimetype[MIME_MAX_SUPER + MIME_MAX_TYPE + 2]; + /* MIME media type string */ + + + snprintf(mimetype, sizeof(mimetype), "%s/%s", filetype->super, + filetype->type); + + send_ipp_status(con, IPP_DOCUMENT_FORMAT, + _("Unsupported format \"%s\"."), mimetype); + + ippAddString(con->response, IPP_TAG_UNSUPPORTED_GROUP, IPP_TAG_MIMETYPE, + "document-format", NULL, mimetype); + + return (NULL); + } + + if ((attr = ippFindAttribute(con->request, "copies", + IPP_TAG_INTEGER)) != NULL) + { + if (attr->values[0].integer < 1 || attr->values[0].integer > MaxCopies) + { + send_ipp_status(con, IPP_ATTRIBUTES, _("Bad copies value %d."), + attr->values[0].integer); + ippAddInteger(con->response, IPP_TAG_UNSUPPORTED_GROUP, IPP_TAG_INTEGER, + "copies", attr->values[0].integer); + return (NULL); + } + } + + if ((attr = ippFindAttribute(con->request, "job-sheets", + IPP_TAG_ZERO)) != NULL) + { + if (attr->value_tag != IPP_TAG_KEYWORD && + attr->value_tag != IPP_TAG_NAME) + { + send_ipp_status(con, IPP_BAD_REQUEST, _("Bad job-sheets value type.")); + return (NULL); + } + + if (attr->num_values > 2) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Too many job-sheets values (%d > 2)."), + attr->num_values); + return (NULL); + } + + for (i = 0; i < attr->num_values; i ++) + if (strcmp(attr->values[i].string.text, "none") && + !cupsdFindBanner(attr->values[i].string.text)) + { + send_ipp_status(con, IPP_BAD_REQUEST, _("Bad job-sheets value \"%s\"."), + attr->values[i].string.text); + return (NULL); + } + } + + if ((attr = ippFindAttribute(con->request, "number-up", + IPP_TAG_INTEGER)) != NULL) + { + if (attr->values[0].integer != 1 && + attr->values[0].integer != 2 && + attr->values[0].integer != 4 && + attr->values[0].integer != 6 && + attr->values[0].integer != 9 && + attr->values[0].integer != 16) + { + send_ipp_status(con, IPP_ATTRIBUTES, _("Bad number-up value %d."), + attr->values[0].integer); + ippAddInteger(con->response, IPP_TAG_UNSUPPORTED_GROUP, IPP_TAG_INTEGER, + "number-up", attr->values[0].integer); + return (NULL); + } + } + + if ((attr = ippFindAttribute(con->request, "page-ranges", + IPP_TAG_RANGE)) != NULL) + { + for (i = 0, lowerpagerange = 1; i < attr->num_values; i ++) + { + if (attr->values[i].range.lower < lowerpagerange || + attr->values[i].range.lower > attr->values[i].range.upper) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Bad page-ranges values %d-%d."), + attr->values[i].range.lower, + attr->values[i].range.upper); + return (NULL); + } + + lowerpagerange = attr->values[i].range.upper + 1; + } + } + + /* + * Do media selection as needed... + */ + + if (!ippFindAttribute(con->request, "PageRegion", IPP_TAG_ZERO) && + !ippFindAttribute(con->request, "PageSize", IPP_TAG_ZERO) && + _ppdCacheGetPageSize(printer->pc, con->request, NULL, &exact)) + { + if (!exact && + (media_col = ippFindAttribute(con->request, "media-col", + IPP_TAG_BEGIN_COLLECTION)) != NULL) + { + send_ipp_status(con, IPP_OK_SUBST, _("Unsupported margins.")); + + unsup_col = ippNew(); + if ((media_margin = ippFindAttribute(media_col->values[0].collection, + "media-bottom-margin", + IPP_TAG_INTEGER)) != NULL) + ippAddInteger(unsup_col, IPP_TAG_ZERO, IPP_TAG_INTEGER, + "media-bottom-margin", media_margin->values[0].integer); + + if ((media_margin = ippFindAttribute(media_col->values[0].collection, + "media-left-margin", + IPP_TAG_INTEGER)) != NULL) + ippAddInteger(unsup_col, IPP_TAG_ZERO, IPP_TAG_INTEGER, + "media-left-margin", media_margin->values[0].integer); + + if ((media_margin = ippFindAttribute(media_col->values[0].collection, + "media-right-margin", + IPP_TAG_INTEGER)) != NULL) + ippAddInteger(unsup_col, IPP_TAG_ZERO, IPP_TAG_INTEGER, + "media-right-margin", media_margin->values[0].integer); + + if ((media_margin = ippFindAttribute(media_col->values[0].collection, + "media-top-margin", + IPP_TAG_INTEGER)) != NULL) + ippAddInteger(unsup_col, IPP_TAG_ZERO, IPP_TAG_INTEGER, + "media-top-margin", media_margin->values[0].integer); + + ippAddCollection(con->response, IPP_TAG_UNSUPPORTED_GROUP, "media-col", + unsup_col); + ippDelete(unsup_col); + } + } + + /* + * Make sure we aren't over our limit... + */ + + if (MaxJobs && cupsArrayCount(Jobs) >= MaxJobs) + cupsdCleanJobs(); + + if (MaxJobs && cupsArrayCount(Jobs) >= MaxJobs) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, _("Too many active jobs.")); + return (NULL); + } + + if ((i = check_quotas(con, printer)) < 0) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, _("Quota limit reached.")); + return (NULL); + } + else if (i == 0) + { + send_ipp_status(con, IPP_NOT_AUTHORIZED, _("Not allowed to print.")); + return (NULL); + } + + /* + * Create the job and set things up... + */ + + if ((attr = ippFindAttribute(con->request, "job-priority", + IPP_TAG_INTEGER)) != NULL) + priority = attr->values[0].integer; + else + { + if ((val = cupsGetOption("job-priority", printer->num_options, + printer->options)) != NULL) + priority = atoi(val); + else + priority = 50; + + ippAddInteger(con->request, IPP_TAG_JOB, IPP_TAG_INTEGER, "job-priority", + priority); + } + + if (!ippFindAttribute(con->request, "job-name", IPP_TAG_NAME)) + ippAddString(con->request, IPP_TAG_JOB, IPP_TAG_NAME, "job-name", NULL, + "Untitled"); + + if ((job = cupsdAddJob(priority, printer->name)) == NULL) + { + send_ipp_status(con, IPP_INTERNAL_ERROR, + _("Unable to add job for destination \"%s\"."), + printer->name); + return (NULL); + } + + job->dtype = printer->type & (CUPS_PRINTER_CLASS | CUPS_PRINTER_REMOTE); + job->attrs = con->request; + job->dirty = 1; + con->request = ippNewRequest(job->attrs->request.op.operation_id); + + cupsdMarkDirty(CUPSD_DIRTY_JOBS); + + add_job_uuid(job); + apply_printer_defaults(printer, job); + + attr = ippFindAttribute(job->attrs, "requesting-user-name", IPP_TAG_NAME); + + if (con->username[0]) + { + cupsdSetString(&job->username, con->username); + + if (attr) + cupsdSetString(&attr->values[0].string.text, con->username); + } + else if (attr) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "add_job: requesting-user-name=\"%s\"", + attr->values[0].string.text); + + cupsdSetString(&job->username, attr->values[0].string.text); + } + else + cupsdSetString(&job->username, "anonymous"); + + if (!attr) + ippAddString(job->attrs, IPP_TAG_JOB, IPP_TAG_NAME, + "job-originating-user-name", NULL, job->username); + else + { + attr->group_tag = IPP_TAG_JOB; + _cupsStrFree(attr->name); + attr->name = _cupsStrAlloc("job-originating-user-name"); + } + + if (con->username[0] || auth_info) + { + save_auth_info(con, job, auth_info); + + /* + * Remove the auth-info attribute from the attribute data... + */ + + if (auth_info) + ippDeleteAttribute(job->attrs, auth_info); + } + + if ((attr = ippFindAttribute(job->attrs, "job-originating-host-name", + IPP_TAG_ZERO)) != NULL) + { + /* + * Request contains a job-originating-host-name attribute; validate it... + */ + + if (attr->value_tag != IPP_TAG_NAME || + attr->num_values != 1 || + strcmp(con->http.hostname, "localhost")) + { + /* + * Can't override the value if we aren't connected via localhost. + * Also, we can only have 1 value and it must be a name value. + */ + + switch (attr->value_tag) + { + case IPP_TAG_STRING : + case IPP_TAG_TEXTLANG : + case IPP_TAG_NAMELANG : + case IPP_TAG_TEXT : + case IPP_TAG_NAME : + case IPP_TAG_KEYWORD : + case IPP_TAG_URI : + case IPP_TAG_URISCHEME : + case IPP_TAG_CHARSET : + case IPP_TAG_LANGUAGE : + case IPP_TAG_MIMETYPE : + /* + * Free old strings... + */ + + for (i = 0; i < attr->num_values; i ++) + { + _cupsStrFree(attr->values[i].string.text); + attr->values[i].string.text = NULL; + if (attr->values[i].string.language) + { + _cupsStrFree(attr->values[i].string.language); + attr->values[i].string.language = NULL; + } + } + + default : + break; + } + + /* + * Use the default connection hostname instead... + */ + + attr->value_tag = IPP_TAG_NAME; + attr->num_values = 1; + attr->values[0].string.text = _cupsStrAlloc(con->http.hostname); + } + + attr->group_tag = IPP_TAG_JOB; + } + else + { + /* + * No job-originating-host-name attribute, so use the hostname from + * the connection... + */ + + ippAddString(job->attrs, IPP_TAG_JOB, IPP_TAG_NAME, + "job-originating-host-name", NULL, con->http.hostname); + } + + ippAddInteger(job->attrs, IPP_TAG_JOB, IPP_TAG_INTEGER, "time-at-creation", + time(NULL)); + attr = ippAddInteger(job->attrs, IPP_TAG_JOB, IPP_TAG_INTEGER, + "time-at-processing", 0); + attr->value_tag = IPP_TAG_NOVALUE; + attr = ippAddInteger(job->attrs, IPP_TAG_JOB, IPP_TAG_INTEGER, + "time-at-completed", 0); + attr->value_tag = IPP_TAG_NOVALUE; + + /* + * Add remaining job attributes... + */ + + ippAddInteger(job->attrs, IPP_TAG_JOB, IPP_TAG_INTEGER, "job-id", job->id); + job->state = ippAddInteger(job->attrs, IPP_TAG_JOB, IPP_TAG_ENUM, + "job-state", IPP_JOB_STOPPED); + job->state_value = (ipp_jstate_t)job->state->values[0].integer; + job->reasons = ippAddString(job->attrs, IPP_TAG_JOB, IPP_TAG_KEYWORD, + "job-state-reasons", NULL, "job-incoming"); + job->sheets = ippAddInteger(job->attrs, IPP_TAG_JOB, IPP_TAG_INTEGER, + "job-media-sheets-completed", 0); + ippAddString(job->attrs, IPP_TAG_JOB, IPP_TAG_URI, "job-printer-uri", NULL, + printer->uri); + + if ((attr = ippFindAttribute(job->attrs, "job-k-octets", + IPP_TAG_INTEGER)) != NULL) + attr->values[0].integer = 0; + else + ippAddInteger(job->attrs, IPP_TAG_JOB, IPP_TAG_INTEGER, "job-k-octets", 0); + + if ((attr = ippFindAttribute(job->attrs, "job-hold-until", + IPP_TAG_KEYWORD)) == NULL) + attr = ippFindAttribute(job->attrs, "job-hold-until", IPP_TAG_NAME); + if (!attr) + { + if ((val = cupsGetOption("job-hold-until", printer->num_options, + printer->options)) == NULL) + val = "no-hold"; + + attr = ippAddString(job->attrs, IPP_TAG_JOB, IPP_TAG_KEYWORD, + "job-hold-until", NULL, val); + } + if (attr && strcmp(attr->values[0].string.text, "no-hold")) + { + /* + * Hold job until specified time... + */ + + cupsdSetJobHoldUntil(job, attr->values[0].string.text, 0); + + job->state->values[0].integer = IPP_JOB_HELD; + job->state_value = IPP_JOB_HELD; + + ippSetString(job->attrs, &job->reasons, 0, "job-hold-until-specified"); + } + else if (job->attrs->request.op.operation_id == IPP_CREATE_JOB) + { + job->hold_until = time(NULL) + MultipleOperationTimeout; + job->state->values[0].integer = IPP_JOB_HELD; + job->state_value = IPP_JOB_HELD; + } + else + { + job->state->values[0].integer = IPP_JOB_PENDING; + job->state_value = IPP_JOB_PENDING; + + ippSetString(job->attrs, &job->reasons, 0, "none"); + } + + if (!(printer->type & CUPS_PRINTER_REMOTE) || Classification) + { + /* + * Add job sheets options... + */ + + if ((attr = ippFindAttribute(job->attrs, "job-sheets", + IPP_TAG_ZERO)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Adding default job-sheets values \"%s,%s\"...", + printer->job_sheets[0], printer->job_sheets[1]); + + attr = ippAddStrings(job->attrs, IPP_TAG_JOB, IPP_TAG_NAME, "job-sheets", + 2, NULL, NULL); + attr->values[0].string.text = _cupsStrRetain(printer->job_sheets[0]); + attr->values[1].string.text = _cupsStrRetain(printer->job_sheets[1]); + } + + job->job_sheets = attr; + + /* + * Enforce classification level if set... + */ + + if (Classification) + { + cupsdLogMessage(CUPSD_LOG_INFO, + "Classification=\"%s\", ClassifyOverride=%d", + Classification ? Classification : "(null)", + ClassifyOverride); + + if (ClassifyOverride) + { + if (!strcmp(attr->values[0].string.text, "none") && + (attr->num_values == 1 || + !strcmp(attr->values[1].string.text, "none"))) + { + /* + * Force the leading banner to have the classification on it... + */ + + cupsdSetString(&attr->values[0].string.text, Classification); + + cupsdLogJob(job, CUPSD_LOG_NOTICE, "CLASSIFICATION FORCED " + "job-sheets=\"%s,none\", " + "job-originating-user-name=\"%s\"", + Classification, job->username); + } + else if (attr->num_values == 2 && + strcmp(attr->values[0].string.text, + attr->values[1].string.text) && + strcmp(attr->values[0].string.text, "none") && + strcmp(attr->values[1].string.text, "none")) + { + /* + * Can't put two different security markings on the same document! + */ + + cupsdSetString(&attr->values[1].string.text, attr->values[0].string.text); + + cupsdLogJob(job, CUPSD_LOG_NOTICE, "CLASSIFICATION FORCED " + "job-sheets=\"%s,%s\", " + "job-originating-user-name=\"%s\"", + attr->values[0].string.text, + attr->values[1].string.text, job->username); + } + else if (strcmp(attr->values[0].string.text, Classification) && + strcmp(attr->values[0].string.text, "none") && + (attr->num_values == 1 || + (strcmp(attr->values[1].string.text, Classification) && + strcmp(attr->values[1].string.text, "none")))) + { + if (attr->num_values == 1) + cupsdLogJob(job, CUPSD_LOG_NOTICE, + "CLASSIFICATION OVERRIDDEN " + "job-sheets=\"%s\", " + "job-originating-user-name=\"%s\"", + attr->values[0].string.text, job->username); + else + cupsdLogJob(job, CUPSD_LOG_NOTICE, + "CLASSIFICATION OVERRIDDEN " + "job-sheets=\"%s,%s\",fffff " + "job-originating-user-name=\"%s\"", + attr->values[0].string.text, + attr->values[1].string.text, job->username); + } + } + else if (strcmp(attr->values[0].string.text, Classification) && + (attr->num_values == 1 || + strcmp(attr->values[1].string.text, Classification))) + { + /* + * Force the banner to have the classification on it... + */ + + if (attr->num_values > 1 && + !strcmp(attr->values[0].string.text, attr->values[1].string.text)) + { + cupsdSetString(&(attr->values[0].string.text), Classification); + cupsdSetString(&(attr->values[1].string.text), Classification); + } + else + { + if (attr->num_values == 1 || + strcmp(attr->values[0].string.text, "none")) + cupsdSetString(&(attr->values[0].string.text), Classification); + + if (attr->num_values > 1 && + strcmp(attr->values[1].string.text, "none")) + cupsdSetString(&(attr->values[1].string.text), Classification); + } + + if (attr->num_values > 1) + cupsdLogJob(job, CUPSD_LOG_NOTICE, + "CLASSIFICATION FORCED " + "job-sheets=\"%s,%s\", " + "job-originating-user-name=\"%s\"", + attr->values[0].string.text, + attr->values[1].string.text, job->username); + else + cupsdLogJob(job, CUPSD_LOG_NOTICE, + "CLASSIFICATION FORCED " + "job-sheets=\"%s\", " + "job-originating-user-name=\"%s\"", + Classification, job->username); + } + } + + /* + * See if we need to add the starting sheet... + */ + + if (!(printer->type & CUPS_PRINTER_REMOTE)) + { + cupsdLogJob(job, CUPSD_LOG_INFO, "Adding start banner page \"%s\".", + attr->values[0].string.text); + + if ((kbytes = copy_banner(con, job, attr->values[0].string.text)) < 0) + { + cupsdSetJobState(job, IPP_JOB_ABORTED, CUPSD_JOB_PURGE, + "Aborting job because the start banner could not be " + "copied."); + return (NULL); + } + + cupsdUpdateQuota(printer, job->username, 0, kbytes); + } + } + else if ((attr = ippFindAttribute(job->attrs, "job-sheets", + IPP_TAG_ZERO)) != NULL) + job->job_sheets = attr; + + /* + * Fill in the response info... + */ + + httpAssembleURIf(HTTP_URI_CODING_ALL, job_uri, sizeof(job_uri), "ipp", NULL, + con->servername, con->serverport, "/jobs/%d", job->id); + ippAddString(con->response, IPP_TAG_JOB, IPP_TAG_URI, "job-uri", NULL, + job_uri); + + ippAddInteger(con->response, IPP_TAG_JOB, IPP_TAG_INTEGER, "job-id", job->id); + + ippAddInteger(con->response, IPP_TAG_JOB, IPP_TAG_ENUM, "job-state", + job->state_value); + ippAddString(con->response, IPP_TAG_JOB, IPP_TAG_KEYWORD, "job-state-reasons", + NULL, job->reasons->values[0].string.text); + + con->response->request.status.status_code = IPP_OK; + + /* + * Add any job subscriptions... + */ + + add_job_subscriptions(con, job); + + /* + * Set all but the first two attributes to the job attributes group... + */ + + for (attr = job->attrs->attrs->next->next; attr; attr = attr->next) + attr->group_tag = IPP_TAG_JOB; + + /* + * Fire the "job created" event... + */ + + cupsdAddEvent(CUPSD_EVENT_JOB_CREATED, printer, job, "Job created."); + + /* + * Return the new job... + */ + + return (job); +} + + +/* + * 'add_job_subscriptions()' - Add any subscriptions for a job. + */ + +static void +add_job_subscriptions( + cupsd_client_t *con, /* I - Client connection */ + cupsd_job_t *job) /* I - Newly created job */ +{ + int i; /* Looping var */ + ipp_attribute_t *prev, /* Previous attribute */ + *next, /* Next attribute */ + *attr; /* Current attribute */ + cupsd_subscription_t *sub; /* Subscription object */ + const char *recipient, /* notify-recipient-uri */ + *pullmethod; /* notify-pull-method */ + ipp_attribute_t *user_data; /* notify-user-data */ + int interval; /* notify-time-interval */ + unsigned mask; /* notify-events */ + + + /* + * Find the first subscription group attribute; return if we have + * none... + */ + + for (attr = job->attrs->attrs; attr; attr = attr->next) + if (attr->group_tag == IPP_TAG_SUBSCRIPTION) + break; + + if (!attr) + return; + + /* + * Process the subscription attributes in the request... + */ + + while (attr) + { + recipient = NULL; + pullmethod = NULL; + user_data = NULL; + interval = 0; + mask = CUPSD_EVENT_NONE; + + while (attr && attr->group_tag != IPP_TAG_ZERO) + { + if (!strcmp(attr->name, "notify-recipient-uri") && + attr->value_tag == IPP_TAG_URI) + { + /* + * Validate the recipient scheme against the ServerBin/notifier + * directory... + */ + + char notifier[1024], /* Notifier filename */ + scheme[HTTP_MAX_URI], /* Scheme portion of URI */ + userpass[HTTP_MAX_URI], /* Username portion of URI */ + host[HTTP_MAX_URI], /* Host portion of URI */ + resource[HTTP_MAX_URI]; /* Resource portion of URI */ + int port; /* Port portion of URI */ + + + recipient = attr->values[0].string.text; + + if (httpSeparateURI(HTTP_URI_CODING_ALL, recipient, + scheme, sizeof(scheme), userpass, sizeof(userpass), + host, sizeof(host), &port, + resource, sizeof(resource)) < HTTP_URI_OK) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Bad notify-recipient-uri \"%s\"."), recipient); + ippAddInteger(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_ENUM, + "notify-status-code", IPP_URI_SCHEME); + return; + } + + snprintf(notifier, sizeof(notifier), "%s/notifier/%s", ServerBin, + scheme); + if (access(notifier, X_OK)) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("notify-recipient-uri URI \"%s\" uses unknown " + "scheme."), recipient); + ippAddInteger(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_ENUM, + "notify-status-code", IPP_URI_SCHEME); + return; + } + + if (!strcmp(scheme, "rss") && !check_rss_recipient(recipient)) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("notify-recipient-uri URI \"%s\" is already used."), + recipient); + ippAddInteger(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_ENUM, + "notify-status-code", IPP_ATTRIBUTES); + return; + } + } + else if (!strcmp(attr->name, "notify-pull-method") && + attr->value_tag == IPP_TAG_KEYWORD) + { + pullmethod = attr->values[0].string.text; + + if (strcmp(pullmethod, "ippget")) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Bad notify-pull-method \"%s\"."), pullmethod); + ippAddInteger(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_ENUM, + "notify-status-code", IPP_ATTRIBUTES); + return; + } + } + else if (!strcmp(attr->name, "notify-charset") && + attr->value_tag == IPP_TAG_CHARSET && + strcmp(attr->values[0].string.text, "us-ascii") && + strcmp(attr->values[0].string.text, "utf-8")) + { + send_ipp_status(con, IPP_CHARSET, + _("Character set \"%s\" not supported."), + attr->values[0].string.text); + return; + } + else if (!strcmp(attr->name, "notify-natural-language") && + (attr->value_tag != IPP_TAG_LANGUAGE || + strcmp(attr->values[0].string.text, DefaultLanguage))) + { + send_ipp_status(con, IPP_CHARSET, + _("Language \"%s\" not supported."), + attr->values[0].string.text); + return; + } + else if (!strcmp(attr->name, "notify-user-data") && + attr->value_tag == IPP_TAG_STRING) + { + if (attr->num_values > 1 || attr->values[0].unknown.length > 63) + { + send_ipp_status(con, IPP_REQUEST_VALUE, + _("The notify-user-data value is too large " + "(%d > 63 octets)."), + attr->values[0].unknown.length); + return; + } + + user_data = attr; + } + else if (!strcmp(attr->name, "notify-events") && + attr->value_tag == IPP_TAG_KEYWORD) + { + for (i = 0; i < attr->num_values; i ++) + mask |= cupsdEventValue(attr->values[i].string.text); + } + else if (!strcmp(attr->name, "notify-lease-duration")) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("The notify-lease-duration attribute cannot be " + "used with job subscriptions.")); + return; + } + else if (!strcmp(attr->name, "notify-time-interval") && + attr->value_tag == IPP_TAG_INTEGER) + interval = attr->values[0].integer; + + attr = attr->next; + } + + if (!recipient && !pullmethod) + break; + + if (mask == CUPSD_EVENT_NONE) + mask = CUPSD_EVENT_JOB_COMPLETED; + + if ((sub = cupsdAddSubscription(mask, cupsdFindDest(job->dest), job, + recipient, 0)) != NULL) + { + sub->interval = interval; + + cupsdSetString(&sub->owner, job->username); + + if (user_data) + { + sub->user_data_len = user_data->values[0].unknown.length; + memcpy(sub->user_data, user_data->values[0].unknown.data, + sub->user_data_len); + } + + ippAddSeparator(con->response); + ippAddInteger(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_INTEGER, + "notify-subscription-id", sub->id); + + cupsdLogMessage(CUPSD_LOG_DEBUG, "Added subscription %d for job %d", + sub->id, job->id); + } + + if (attr) + attr = attr->next; + } + + cupsdMarkDirty(CUPSD_DIRTY_SUBSCRIPTIONS); + + /* + * Remove all of the subscription attributes from the job request... + * + * TODO: Optimize this since subscription groups have to come at the + * end of the request... + */ + + for (attr = job->attrs->attrs, prev = NULL; attr; attr = next) + { + next = attr->next; + + if (attr->group_tag == IPP_TAG_SUBSCRIPTION || + attr->group_tag == IPP_TAG_ZERO) + { + /* + * Free and remove this attribute... + */ + + ippDeleteAttribute(NULL, attr); + + if (prev) + prev->next = next; + else + job->attrs->attrs = next; + } + else + prev = attr; + } + + job->attrs->last = prev; + job->attrs->current = prev; +} + + +/* + * 'add_job_uuid()' - Add job-uuid attribute to a job. + * + * See RFC 4122 for the definition of UUIDs and the format. + */ + +static void +add_job_uuid(cupsd_job_t *job) /* I - Job */ +{ + char uuid[64]; /* job-uuid string */ + + + /* + * Add a job-uuid attribute if none exists... + */ + + if (!ippFindAttribute(job->attrs, "job-uuid", IPP_TAG_URI)) + ippAddString(job->attrs, IPP_TAG_JOB, IPP_TAG_URI, "job-uuid", NULL, + _httpAssembleUUID(ServerName, RemotePort, job->dest, job->id, + uuid, sizeof(uuid))); +} + + +/* + * 'add_printer()' - Add a printer to the system. + */ + +static void +add_printer(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - URI of printer */ +{ + http_status_t status; /* Policy status */ + int i; /* Looping var */ + char scheme[HTTP_MAX_URI], /* Method portion of URI */ + username[HTTP_MAX_URI], /* Username portion of URI */ + host[HTTP_MAX_URI], /* Host portion of URI */ + resource[HTTP_MAX_URI]; /* Resource portion of URI */ + int port; /* Port portion of URI */ + cupsd_printer_t *printer; /* Printer/class */ + ipp_attribute_t *attr; /* Printer attribute */ + cups_file_t *fp; /* Script/PPD file */ + char line[1024]; /* Line from file... */ + char srcfile[1024], /* Source Script/PPD file */ + dstfile[1024]; /* Destination Script/PPD file */ + int modify; /* Non-zero if we are modifying */ + int changed_driver, /* Changed the PPD/interface script? */ + need_restart_job, /* Need to restart job? */ + set_device_uri, /* Did we set the device URI? */ + set_port_monitor; /* Did we set the port monitor? */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "add_printer(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * Do we have a valid URI? + */ + + httpSeparateURI(HTTP_URI_CODING_ALL, uri->values[0].string.text, scheme, + sizeof(scheme), username, sizeof(username), host, + sizeof(host), &port, resource, sizeof(resource)); + + if (strncmp(resource, "/printers/", 10) || strlen(resource) == 10) + { + /* + * No, return an error... + */ + + send_ipp_status(con, IPP_BAD_REQUEST, + _("The printer-uri must be of the form " + "\"ipp://HOSTNAME/printers/PRINTERNAME\".")); + return; + } + + /* + * Do we have a valid printer name? + */ + + if (!validate_name(resource + 10)) + { + /* + * No, return an error... + */ + + send_ipp_status(con, IPP_BAD_REQUEST, + _("The printer-uri \"%s\" contains invalid characters."), + uri->values[0].string.text); + return; + } + + /* + * See if the printer already exists; if not, create a new printer... + */ + + if ((printer = cupsdFindPrinter(resource + 10)) == NULL) + { + /* + * Printer doesn't exist; see if we have a class of the same name... + */ + + if ((printer = cupsdFindClass(resource + 10)) != NULL) + { + /* + * Yes, return an error... + */ + + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("A class named \"%s\" already exists."), + resource + 10); + return; + } + + /* + * No, check the default policy then add the printer... + */ + + if ((status = cupsdCheckPolicy(DefaultPolicyPtr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, NULL); + return; + } + + printer = cupsdAddPrinter(resource + 10); + modify = 0; + } + else if ((status = cupsdCheckPolicy(printer->op_policy_ptr, con, + NULL)) != HTTP_OK) + { + send_http_error(con, status, printer); + return; + } + else + modify = 1; + + /* + * Look for attributes and copy them over as needed... + */ + + changed_driver = 0; + need_restart_job = 0; + + if ((attr = ippFindAttribute(con->request, "printer-location", + IPP_TAG_TEXT)) != NULL) + cupsdSetString(&printer->location, attr->values[0].string.text); + + if ((attr = ippFindAttribute(con->request, "printer-info", + IPP_TAG_TEXT)) != NULL) + cupsdSetString(&printer->info, attr->values[0].string.text); + + set_device_uri = 0; + + if ((attr = ippFindAttribute(con->request, "device-uri", + IPP_TAG_URI)) != NULL) + { + /* + * Do we have a valid device URI? + */ + + http_uri_status_t uri_status; /* URI separation status */ + char old_device_uri[1024]; + /* Old device URI */ + + + need_restart_job = 1; + + uri_status = httpSeparateURI(HTTP_URI_CODING_ALL, + attr->values[0].string.text, + scheme, sizeof(scheme), + username, sizeof(username), + host, sizeof(host), &port, + resource, sizeof(resource)); + + if (uri_status < HTTP_URI_OK) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, _("Bad device-uri \"%s\"."), + attr->values[0].string.text); + cupsdLogMessage(CUPSD_LOG_DEBUG, + "add_printer: httpSeparateURI returned %d", uri_status); + return; + } + + if (!strcmp(scheme, "file")) + { + /* + * See if the administrator has enabled file devices... + */ + + if (!FileDevice && strcmp(resource, "/dev/null")) + { + /* + * File devices are disabled and the URL is not file:/dev/null... + */ + + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("File device URIs have been disabled. " + "To enable, see the FileDevice directive in " + "\"%s/cupsd.conf\"."), + ServerRoot); + return; + } + } + else + { + /* + * See if the backend exists and is executable... + */ + + snprintf(srcfile, sizeof(srcfile), "%s/backend/%s", ServerBin, scheme); + if (access(srcfile, X_OK)) + { + /* + * Could not find device in list! + */ + + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Bad device-uri scheme \"%s\"."), scheme); + return; + } + } + + if (printer->sanitized_device_uri) + strlcpy(old_device_uri, printer->sanitized_device_uri, + sizeof(old_device_uri)); + else + old_device_uri[0] = '\0'; + + cupsdSetDeviceURI(printer, attr->values[0].string.text); + + cupsdLogMessage(CUPSD_LOG_INFO, + "Setting %s device-uri to \"%s\" (was \"%s\".)", + printer->name, printer->sanitized_device_uri, + old_device_uri); + + set_device_uri = 1; + } + + set_port_monitor = 0; + + if ((attr = ippFindAttribute(con->request, "port-monitor", + IPP_TAG_NAME)) != NULL) + { + ipp_attribute_t *supported; /* port-monitor-supported attribute */ + + + need_restart_job = 1; + + supported = ippFindAttribute(printer->ppd_attrs, "port-monitor-supported", + IPP_TAG_NAME); + if (supported) + { + for (i = 0; i < supported->num_values; i ++) + if (!strcmp(supported->values[i].string.text, + attr->values[0].string.text)) + break; + } + + if (!supported || i >= supported->num_values) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, _("Bad port-monitor \"%s\"."), + attr->values[0].string.text); + return; + } + + cupsdLogMessage(CUPSD_LOG_INFO, + "Setting %s port-monitor to \"%s\" (was \"%s\".)", + printer->name, attr->values[0].string.text, + printer->port_monitor ? printer->port_monitor : "none"); + + if (strcmp(attr->values[0].string.text, "none")) + cupsdSetString(&printer->port_monitor, attr->values[0].string.text); + else + cupsdClearString(&printer->port_monitor); + + set_port_monitor = 1; + } + + if ((attr = ippFindAttribute(con->request, "printer-is-accepting-jobs", + IPP_TAG_BOOLEAN)) != NULL && + attr->values[0].boolean != printer->accepting) + { + cupsdLogMessage(CUPSD_LOG_INFO, + "Setting %s printer-is-accepting-jobs to %d (was %d.)", + printer->name, attr->values[0].boolean, printer->accepting); + + printer->accepting = attr->values[0].boolean; + + cupsdAddEvent(CUPSD_EVENT_PRINTER_STATE, printer, NULL, + "%s accepting jobs.", + printer->accepting ? "Now" : "No longer"); + } + + if ((attr = ippFindAttribute(con->request, "printer-is-shared", + IPP_TAG_BOOLEAN)) != NULL) + { + if (attr->values[0].boolean && + printer->num_auth_info_required == 1 && + !strcmp(printer->auth_info_required[0], "negotiate")) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Cannot share a remote Kerberized printer.")); + return; + } + + if (printer->shared && !attr->values[0].boolean) + cupsdDeregisterPrinter(printer, 1); + + cupsdLogMessage(CUPSD_LOG_INFO, + "Setting %s printer-is-shared to %d (was %d.)", + printer->name, attr->values[0].boolean, printer->shared); + + printer->shared = attr->values[0].boolean; + } + + if ((attr = ippFindAttribute(con->request, "printer-state", + IPP_TAG_ENUM)) != NULL) + { + if (attr->values[0].integer != IPP_PRINTER_IDLE && + attr->values[0].integer != IPP_PRINTER_STOPPED) + { + send_ipp_status(con, IPP_BAD_REQUEST, _("Bad printer-state value %d."), + attr->values[0].integer); + return; + } + + cupsdLogMessage(CUPSD_LOG_INFO, "Setting %s printer-state to %d (was %d.)", + printer->name, attr->values[0].integer, printer->state); + + if (attr->values[0].integer == IPP_PRINTER_STOPPED) + cupsdStopPrinter(printer, 0); + else + { + need_restart_job = 1; + cupsdSetPrinterState(printer, (ipp_pstate_t)(attr->values[0].integer), 0); + } + } + + if ((attr = ippFindAttribute(con->request, "printer-state-message", + IPP_TAG_TEXT)) != NULL) + { + strlcpy(printer->state_message, attr->values[0].string.text, + sizeof(printer->state_message)); + + cupsdAddEvent(CUPSD_EVENT_PRINTER_STATE, printer, NULL, "%s", + printer->state_message); + } + + if ((attr = ippFindAttribute(con->request, "printer-state-reasons", + IPP_TAG_KEYWORD)) != NULL) + { + if (attr->num_values > + (int)(sizeof(printer->reasons) / sizeof(printer->reasons[0]))) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Too many printer-state-reasons values (%d > %d)."), + attr->num_values, + (int)(sizeof(printer->reasons) / + sizeof(printer->reasons[0]))); + return; + } + + for (i = 0; i < printer->num_reasons; i ++) + _cupsStrFree(printer->reasons[i]); + + printer->num_reasons = 0; + for (i = 0; i < attr->num_values; i ++) + { + if (!strcmp(attr->values[i].string.text, "none")) + continue; + + printer->reasons[printer->num_reasons] = + _cupsStrRetain(attr->values[i].string.text); + printer->num_reasons ++; + + if (!strcmp(attr->values[i].string.text, "paused") && + printer->state != IPP_PRINTER_STOPPED) + { + cupsdLogMessage(CUPSD_LOG_INFO, + "Setting %s printer-state to %d (was %d.)", + printer->name, IPP_PRINTER_STOPPED, printer->state); + cupsdStopPrinter(printer, 0); + } + } + + if (PrintcapFormat == PRINTCAP_PLIST) + cupsdMarkDirty(CUPSD_DIRTY_PRINTCAP); + + cupsdAddEvent(CUPSD_EVENT_PRINTER_STATE, printer, NULL, + "Printer \"%s\" state changed.", printer->name); + } + + set_printer_defaults(con, printer); + + if ((attr = ippFindAttribute(con->request, "auth-info-required", + IPP_TAG_KEYWORD)) != NULL) + cupsdSetAuthInfoRequired(printer, NULL, attr); + + /* + * See if we have all required attributes... + */ + + if (!printer->device_uri) + cupsdSetString(&printer->device_uri, "file:///dev/null"); + + /* + * See if we have an interface script or PPD file attached to the request... + */ + + if (con->filename) + { + need_restart_job = 1; + changed_driver = 1; + + strlcpy(srcfile, con->filename, sizeof(srcfile)); + + if ((fp = cupsFileOpen(srcfile, "rb"))) + { + /* + * Yes; get the first line from it... + */ + + line[0] = '\0'; + cupsFileGets(fp, line, sizeof(line)); + cupsFileClose(fp); + + /* + * Then see what kind of file it is... + */ + + snprintf(dstfile, sizeof(dstfile), "%s/interfaces/%s", ServerRoot, + printer->name); + + if (!strncmp(line, "*PPD-Adobe", 10)) + { + /* + * The new file is a PPD file, so remove any old interface script + * that might be lying around... + */ + + unlink(dstfile); + } + else + { + /* + * This must be an interface script, so move the file over to the + * interfaces directory and make it executable... + */ + + if (copy_file(srcfile, dstfile)) + { + send_ipp_status(con, IPP_INTERNAL_ERROR, + _("Unable to copy interface script - %s"), + strerror(errno)); + return; + } + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Copied interface script successfully"); + chmod(dstfile, 0755); + } + + snprintf(dstfile, sizeof(dstfile), "%s/ppd/%s.ppd", ServerRoot, + printer->name); + + if (!strncmp(line, "*PPD-Adobe", 10)) + { + /* + * The new file is a PPD file, so move the file over to the + * ppd directory and make it readable by all... + */ + + if (copy_file(srcfile, dstfile)) + { + send_ipp_status(con, IPP_INTERNAL_ERROR, + _("Unable to copy PPD file - %s"), + strerror(errno)); + return; + } + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Copied PPD file successfully"); + chmod(dstfile, 0644); + } + else + { + /* + * This must be an interface script, so remove any old PPD file that + * may be lying around... + */ + + unlink(dstfile); + } + } + } + else if ((attr = ippFindAttribute(con->request, "ppd-name", + IPP_TAG_NAME)) != NULL) + { + need_restart_job = 1; + changed_driver = 1; + + if (!strcmp(attr->values[0].string.text, "raw")) + { + /* + * Raw driver, remove any existing PPD or interface script files. + */ + + snprintf(dstfile, sizeof(dstfile), "%s/interfaces/%s", ServerRoot, + printer->name); + unlink(dstfile); + + snprintf(dstfile, sizeof(dstfile), "%s/ppd/%s.ppd", ServerRoot, + printer->name); + unlink(dstfile); + } + else + { + /* + * PPD model file... + */ + + snprintf(dstfile, sizeof(dstfile), "%s/interfaces/%s", ServerRoot, + printer->name); + unlink(dstfile); + + snprintf(dstfile, sizeof(dstfile), "%s/ppd/%s.ppd", ServerRoot, + printer->name); + + if (copy_model(con, attr->values[0].string.text, dstfile)) + { + send_ipp_status(con, IPP_INTERNAL_ERROR, _("Unable to copy PPD file.")); + return; + } + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Copied PPD file successfully"); + chmod(dstfile, 0644); + } + } + + if (changed_driver) + { + /* + * If we changed the PPD/interface script, then remove the printer's cache + * file and clear the printer-state-reasons... + */ + + char cache_name[1024]; /* Cache filename for printer attrs */ + + snprintf(cache_name, sizeof(cache_name), "%s/%s.data", CacheDir, + printer->name); + unlink(cache_name); + + cupsdSetPrinterReasons(printer, "none"); + +#ifdef __APPLE__ + /* + * (Re)register color profiles... + */ + + if (!RunUser) + { + apple_unregister_profiles(printer); + apple_register_profiles(printer); + } +#endif /* __APPLE__ */ + } + + /* + * If we set the device URI but not the port monitor, check which port + * monitor to use by default... + */ + + if (set_device_uri && !set_port_monitor) + { + ppd_file_t *ppd; /* PPD file */ + ppd_attr_t *ppdattr; /* cupsPortMonitor attribute */ + + + httpSeparateURI(HTTP_URI_CODING_ALL, printer->device_uri, scheme, + sizeof(scheme), username, sizeof(username), host, + sizeof(host), &port, resource, sizeof(resource)); + + snprintf(srcfile, sizeof(srcfile), "%s/ppd/%s.ppd", ServerRoot, + printer->name); + if ((ppd = _ppdOpenFile(srcfile, _PPD_LOCALIZATION_NONE)) != NULL) + { + for (ppdattr = ppdFindAttr(ppd, "cupsPortMonitor", NULL); + ppdattr; + ppdattr = ppdFindNextAttr(ppd, "cupsPortMonitor", NULL)) + if (!strcmp(scheme, ppdattr->spec)) + { + cupsdLogMessage(CUPSD_LOG_INFO, + "Setting %s port-monitor to \"%s\" (was \"%s\".)", + printer->name, ppdattr->value, + printer->port_monitor ? printer->port_monitor + : "none"); + + if (strcmp(ppdattr->value, "none")) + cupsdSetString(&printer->port_monitor, ppdattr->value); + else + cupsdClearString(&printer->port_monitor); + + break; + } + + ppdClose(ppd); + } + } + + /* + * Update the printer attributes and return... + */ + + cupsdSetPrinterAttrs(printer); + cupsdMarkDirty(CUPSD_DIRTY_PRINTERS); + + if (need_restart_job && printer->job) + { + /* + * Restart the current job... + */ + + cupsdSetJobState(printer->job, IPP_JOB_PENDING, CUPSD_JOB_FORCE, + "Job restarted because the printer was modified."); + } + + cupsdMarkDirty(CUPSD_DIRTY_PRINTCAP); + + if (modify) + { + cupsdAddEvent(CUPSD_EVENT_PRINTER_MODIFIED, + printer, NULL, "Printer \"%s\" modified by \"%s\".", + printer->name, get_username(con)); + + cupsdLogMessage(CUPSD_LOG_INFO, "Printer \"%s\" modified by \"%s\".", + printer->name, get_username(con)); + } + else + { + cupsdAddEvent(CUPSD_EVENT_PRINTER_ADDED, + printer, NULL, "New printer \"%s\" added by \"%s\".", + printer->name, get_username(con)); + + cupsdLogMessage(CUPSD_LOG_INFO, "New printer \"%s\" added by \"%s\".", + printer->name, get_username(con)); + } + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'add_printer_state_reasons()' - Add the "printer-state-reasons" attribute + * based upon the printer state... + */ + +static void +add_printer_state_reasons( + cupsd_client_t *con, /* I - Client connection */ + cupsd_printer_t *p) /* I - Printer info */ +{ + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "add_printer_state_reasons(%p[%d], %p[%s])", + con, con->http.fd, p, p->name); + + if (p->num_reasons == 0) + ippAddString(con->response, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "printer-state-reasons", NULL, "none"); + else + ippAddStrings(con->response, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "printer-state-reasons", p->num_reasons, NULL, + (const char * const *)p->reasons); +} + + +/* + * 'add_queued_job_count()' - Add the "queued-job-count" attribute for + * the specified printer or class. + */ + +static void +add_queued_job_count( + cupsd_client_t *con, /* I - Client connection */ + cupsd_printer_t *p) /* I - Printer or class */ +{ + int count; /* Number of jobs on destination */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "add_queued_job_count(%p[%d], %p[%s])", + con, con->http.fd, p, p->name); + + count = cupsdGetPrinterJobCount(p->name); + + ippAddInteger(con->response, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "queued-job-count", count); +} + + +#ifdef __APPLE__ +/* + * 'apple_init_profile()' - Initialize a color profile. + */ + +static void +apple_init_profile( + ppd_file_t *ppd, /* I - PPD file */ + cups_array_t *languages, /* I - Languages in the PPD file */ +# ifdef HAVE_COLORSYNCREGISTERDEVICE + CFMutableDictionaryRef profile, /* I - Profile dictionary */ +# else + CMDeviceProfileInfo *profile, /* I - Profile record */ +# endif /* HAVE_COLORSYNCREGISTERDEVICE */ + unsigned id, /* I - Profile ID */ + const char *name, /* I - Profile name */ + const char *text, /* I - Profile UI text */ + const char *iccfile) /* I - ICC filename */ +{ +# ifdef HAVE_COLORSYNCREGISTERDEVICE + CFURLRef url; /* URL for profile filename */ +# endif /* HAVE_COLORSYNCREGISTERDEVICE */ + CFMutableDictionaryRef dict; /* Dictionary for name */ + char *language; /* Current language */ + ppd_attr_t *attr; /* Profile attribute */ + CFStringRef cflang, /* Language string */ + cftext; /* Localized text */ + + + (void)id; + + /* + * Build the profile name dictionary... + */ + + dict = CFDictionaryCreateMutable(kCFAllocatorDefault, 0, + &kCFTypeDictionaryKeyCallBacks, + &kCFTypeDictionaryValueCallBacks); + if (!dict) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to initialize profile \"%s\".", + iccfile); + return; + } + + cftext = CFStringCreateWithCString(kCFAllocatorDefault, text, + kCFStringEncodingUTF8); + + if (cftext) + { + CFDictionarySetValue(dict, CFSTR("en_US"), cftext); + CFRelease(cftext); + } + + if (languages) + { + /* + * Find localized names for the color profiles... + */ + + cupsArraySave(ppd->sorted_attrs); + + for (language = (char *)cupsArrayFirst(languages); + language; + language = (char *)cupsArrayNext(languages)) + { + if (iccfile) + { + if ((attr = _ppdLocalizedAttr(ppd, "cupsICCProfile", name, + language)) == NULL) + attr = _ppdLocalizedAttr(ppd, "APTiogaProfile", name, language); + } + else + attr = _ppdLocalizedAttr(ppd, "ColorModel", name, language); + + if (attr && attr->text[0]) + { + cflang = CFStringCreateWithCString(kCFAllocatorDefault, language, + kCFStringEncodingUTF8); + cftext = CFStringCreateWithCString(kCFAllocatorDefault, attr->text, + kCFStringEncodingUTF8); + + if (cflang && cftext) + CFDictionarySetValue(dict, cflang, cftext); + + if (cflang) + CFRelease(cflang); + + if (cftext) + CFRelease(cftext); + } + } + + cupsArrayRestore(ppd->sorted_attrs); + } + + /* + * Fill in the profile data... + */ + +# ifdef HAVE_COLORSYNCREGISTERDEVICE + if (iccfile) + { + url = CFURLCreateFromFileSystemRepresentation(kCFAllocatorDefault, + (const UInt8 *)iccfile, + strlen(iccfile), false); + + if (url) + { + CFDictionarySetValue(profile, kColorSyncDeviceProfileURL, url); + CFRelease(url); + } + } + + CFDictionarySetValue(profile, kColorSyncDeviceModeDescriptions, dict); + CFRelease(dict); + +# else + profile->dataVersion = cmDeviceProfileInfoVersion1; + profile->profileID = id; + profile->profileLoc.locType = iccfile ? cmPathBasedProfile : cmNoProfileBase; + profile->profileName = dict; + + if (iccfile) + strlcpy(profile->profileLoc.u.pathLoc.path, iccfile, + sizeof(profile->profileLoc.u.pathLoc.path)); +# endif /* HAVE_COLORSYNCREGISTERDEVICE */ +} + + +/* + * 'apple_register_profiles()' - Register color profiles for a printer. + */ + +static void +apple_register_profiles( + cupsd_printer_t *p) /* I - Printer */ +{ + int i; /* Looping var */ + char ppdfile[1024], /* PPD filename */ + iccfile[1024], /* ICC filename */ + selector[PPD_MAX_NAME]; + /* Profile selection string */ + ppd_file_t *ppd; /* PPD file */ + ppd_attr_t *attr, /* Profile attributes */ + *profileid_attr,/* cupsProfileID attribute */ + *q1_attr, /* ColorModel (or other) qualifier */ + *q2_attr, /* MediaType (or other) qualifier */ + *q3_attr; /* Resolution (or other) qualifier */ + char q_keyword[PPD_MAX_NAME]; + /* Qualifier keyword */ + const char *q1_choice, /* ColorModel (or other) choice */ + *q2_choice, /* MediaType (or other) choice */ + *q3_choice; /* Resolution (or other) choice */ + const char *profile_key; /* Profile keyword */ + ppd_option_t *cm_option; /* Color model option */ + ppd_choice_t *cm_choice; /* Color model choice */ + int num_profiles; /* Number of profiles */ + OSStatus error = 0; /* Last error */ + unsigned device_id, /* Printer device ID */ + profile_id = 0, /* Profile ID */ + default_profile_id = 0; + /* Default profile ID */ + CFMutableDictionaryRef device_name; /* Printer device name dictionary */ + CFStringRef printer_name; /* Printer name string */ + cups_array_t *languages; /* Languages array */ +# ifdef HAVE_COLORSYNCREGISTERDEVICE + CFMutableDictionaryRef profiles, /* Dictionary of profiles */ + profile; /* Current profile info dictionary */ + CFStringRef dict_key; /* Key in factory profile dictionary */ +# else + CMDeviceScope scope = /* Scope of the registration */ + { + kCFPreferencesAnyUser, + kCFPreferencesCurrentHost + }; + CMDeviceProfileArrayPtr profiles; /* Profiles */ + CMDeviceProfileInfo *profile; /* Current profile */ +# endif /* HAVE_COLORSYNCREGISTERDEVICE */ + + + /* + * Make sure ColorSync is available... + */ + +# ifdef HAVE_COLORSYNCREGISTERDEVICE + if (ColorSyncRegisterDevice == NULL) + return; + +# else + if (CMRegisterColorDevice == NULL) + return; +# endif /* HAVE_COLORSYNCREGISTERDEVICE */ + + /* + * Try opening the PPD file for this printer... + */ + + snprintf(ppdfile, sizeof(ppdfile), "%s/ppd/%s.ppd", ServerRoot, p->name); + if ((ppd = _ppdOpenFile(ppdfile, _PPD_LOCALIZATION_ICC_PROFILES)) == NULL) + return; + + /* + * See if we have any profiles... + */ + + if ((attr = ppdFindAttr(ppd, "APTiogaProfile", NULL)) != NULL) + profile_key = "APTiogaProfile"; + else + { + attr = ppdFindAttr(ppd, "cupsICCProfile", NULL); + profile_key = "cupsICCProfile"; + } + + for (num_profiles = 0; attr; attr = ppdFindNextAttr(ppd, profile_key, NULL)) + if (attr->spec[0] && attr->value && attr->value[0]) + { + if (attr->value[0] != '/') + snprintf(iccfile, sizeof(iccfile), "%s/profiles/%s", DataDir, + attr->value); + else + strlcpy(iccfile, attr->value, sizeof(iccfile)); + + if (access(iccfile, 0)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "%s: ICC Profile \"%s\" does not exist.", p->name, + iccfile); + continue; + } + + num_profiles ++; + } + +# ifdef HAVE_COLORSYNCREGISTERDEVICE + /* + * Create a dictionary for the factory profiles... + */ + + profiles = CFDictionaryCreateMutable(kCFAllocatorDefault, 0, + &kCFTypeDictionaryKeyCallBacks, + &kCFTypeDictionaryValueCallBacks); + if (!profiles) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate memory for factory profiles."); + ppdClose(ppd); + return; + } +# endif /* HAVE_COLORSYNCREGISTERDEVICE */ + + /* + * If we have profiles, add them... + */ + + if (num_profiles > 0) + { + if (profile_key[0] == 'A') + { + /* + * For Tioga PPDs, get the default profile using the DefaultAPTiogaProfile + * attribute... + */ + + if ((attr = ppdFindAttr(ppd, "DefaultAPTiogaProfile", NULL)) != NULL && + attr->value) + default_profile_id = atoi(attr->value); + + q1_choice = q2_choice = q3_choice = NULL; + } + else + { + /* + * For CUPS PPDs, figure out the default profile selector values... + */ + + if ((attr = ppdFindAttr(ppd, "cupsICCQualifier1", NULL)) != NULL && + attr->value && attr->value[0]) + { + snprintf(q_keyword, sizeof(q_keyword), "Default%s", attr->value); + q1_attr = ppdFindAttr(ppd, q_keyword, NULL); + } + else if ((q1_attr = ppdFindAttr(ppd, "DefaultColorModel", NULL)) == NULL) + q1_attr = ppdFindAttr(ppd, "DefaultColorSpace", NULL); + + if (q1_attr && q1_attr->value && q1_attr->value[0]) + q1_choice = q1_attr->value; + else + q1_choice = ""; + + if ((attr = ppdFindAttr(ppd, "cupsICCQualifier2", NULL)) != NULL && + attr->value && attr->value[0]) + { + snprintf(q_keyword, sizeof(q_keyword), "Default%s", attr->value); + q2_attr = ppdFindAttr(ppd, q_keyword, NULL); + } + else + q2_attr = ppdFindAttr(ppd, "DefaultMediaType", NULL); + + if (q2_attr && q2_attr->value && q2_attr->value[0]) + q2_choice = q2_attr->value; + else + q2_choice = NULL; + + if ((attr = ppdFindAttr(ppd, "cupsICCQualifier3", NULL)) != NULL && + attr->value && attr->value[0]) + { + snprintf(q_keyword, sizeof(q_keyword), "Default%s", attr->value); + q3_attr = ppdFindAttr(ppd, q_keyword, NULL); + } + else + q3_attr = ppdFindAttr(ppd, "DefaultResolution", NULL); + + if (q3_attr && q3_attr->value && q3_attr->value[0]) + q3_choice = q3_attr->value; + else + q3_choice = NULL; + } + +# ifndef HAVE_COLORSYNCREGISTERDEVICE + /* + * Build the array of profiles... + * + * Note: This calloc actually requests slightly more memory than needed. + */ + + if ((profiles = calloc(num_profiles, sizeof(CMDeviceProfileArray))) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate memory for factory profiles."); + ppdClose(ppd); + return; + } + + profiles->profileCount = num_profiles; + profile = profiles->profiles; +# endif /* !HAVE_COLORSYNCREGISTERDEVICE */ + + /* + * Loop through the profiles listed in the PPD... + */ + + languages = _ppdGetLanguages(ppd); + + for (attr = ppdFindAttr(ppd, profile_key, NULL); + attr; + attr = ppdFindNextAttr(ppd, profile_key, NULL)) + if (attr->spec[0] && attr->value && attr->value[0]) + { + /* + * Add this profile... + */ + + if (attr->value[0] != '/') + snprintf(iccfile, sizeof(iccfile), "%s/profiles/%s", DataDir, + attr->value); + else + strlcpy(iccfile, attr->value, sizeof(iccfile)); + + if (_cupsFileCheck(iccfile, _CUPS_FILE_CHECK_FILE, !RunUser, + cupsdLogFCMessage, p)) + continue; + + if (profile_key[0] == 'c') + { + cupsArraySave(ppd->sorted_attrs); + + if ((profileid_attr = ppdFindAttr(ppd, "cupsProfileID", + attr->spec)) != NULL && + profileid_attr->value && isdigit(profileid_attr->value[0] & 255)) + profile_id = (unsigned)strtoul(profileid_attr->value, NULL, 10); + else + profile_id = _ppdHashName(attr->spec); + + cupsArrayRestore(ppd->sorted_attrs); + } + else + profile_id = atoi(attr->spec); + +# ifdef HAVE_COLORSYNCREGISTERDEVICE + profile = CFDictionaryCreateMutable(kCFAllocatorDefault, 0, + &kCFTypeDictionaryKeyCallBacks, + &kCFTypeDictionaryValueCallBacks); + if (!profile) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate memory for color profile."); + CFRelease(profiles); + ppdClose(ppd); + return; + } + + apple_init_profile(ppd, languages, profile, profile_id, attr->spec, + attr->text[0] ? attr->text : attr->spec, iccfile); + + dict_key = CFStringCreateWithFormat(kCFAllocatorDefault, NULL, + CFSTR("%u"), profile_id); + if (dict_key) + { + CFDictionarySetValue(profiles, dict_key, profile); + CFRelease(dict_key); + } + + CFRelease(profile); + +# else + apple_init_profile(ppd, languages, profile, profile_id, attr->spec, + attr->text[0] ? attr->text : attr->spec, iccfile); + + profile ++; +# endif /* HAVE_COLORSYNCREGISTERDEVICE */ + + /* + * See if this is the default profile... + */ + + if (!default_profile_id && q1_choice && q2_choice && q3_choice) + { + snprintf(selector, sizeof(selector), "%s.%s.%s", q1_choice, q2_choice, + q3_choice); + if (!strcmp(selector, attr->spec)) + default_profile_id = profile_id; + } + + if (!default_profile_id && q1_choice && q2_choice) + { + snprintf(selector, sizeof(selector), "%s.%s.", q1_choice, q2_choice); + if (!strcmp(selector, attr->spec)) + default_profile_id = profile_id; + } + + if (!default_profile_id && q1_choice && q3_choice) + { + snprintf(selector, sizeof(selector), "%s..%s", q1_choice, q3_choice); + if (!strcmp(selector, attr->spec)) + default_profile_id = profile_id; + } + + if (!default_profile_id && q1_choice) + { + snprintf(selector, sizeof(selector), "%s..", q1_choice); + if (!strcmp(selector, attr->spec)) + default_profile_id = profile_id; + } + + if (!default_profile_id && q2_choice && q3_choice) + { + snprintf(selector, sizeof(selector), ".%s.%s", q2_choice, q3_choice); + if (!strcmp(selector, attr->spec)) + default_profile_id = profile_id; + } + + if (!default_profile_id && q2_choice) + { + snprintf(selector, sizeof(selector), ".%s.", q2_choice); + if (!strcmp(selector, attr->spec)) + default_profile_id = profile_id; + } + + if (!default_profile_id && q3_choice) + { + snprintf(selector, sizeof(selector), "..%s", q3_choice); + if (!strcmp(selector, attr->spec)) + default_profile_id = profile_id; + } + } + + _ppdFreeLanguages(languages); + } + else if ((cm_option = ppdFindOption(ppd, "ColorModel")) != NULL) + { + /* + * Extract profiles from ColorModel option... + */ + + const char *profile_name; /* Name of generic profile */ + + + num_profiles = cm_option->num_choices; + +# ifndef HAVE_COLORSYNCREGISTERDEVICE + /* + * Create an array for the factory profiles... + */ + + if ((profiles = calloc(num_profiles, sizeof(CMDeviceProfileArray))) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate memory for factory profiles."); + ppdClose(ppd); + return; + } + + profiles->profileCount = num_profiles; + profile = profiles->profiles; +# endif /* HAVE_COLORSYNCREGISTERDEVICE */ + + for (i = cm_option->num_choices, cm_choice = cm_option->choices; + i > 0; + i --, cm_choice ++) + { + if (!strcmp(cm_choice->choice, "Gray") || + !strcmp(cm_choice->choice, "Black")) + profile_name = "Gray"; + else if (!strcmp(cm_choice->choice, "RGB") || + !strcmp(cm_choice->choice, "CMY")) + profile_name = "RGB"; + else if (!strcmp(cm_choice->choice, "CMYK") || + !strcmp(cm_choice->choice, "KCMY")) + profile_name = "CMYK"; + else + profile_name = "DeviceN"; + + snprintf(selector, sizeof(selector), "%s..", profile_name); + profile_id = _ppdHashName(selector); + +# ifdef HAVE_COLORSYNCREGISTERDEVICE + profile = CFDictionaryCreateMutable(kCFAllocatorDefault, 0, + &kCFTypeDictionaryKeyCallBacks, + &kCFTypeDictionaryValueCallBacks); + if (!profile) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate memory for color profile."); + CFRelease(profiles); + ppdClose(ppd); + return; + } + + apple_init_profile(ppd, NULL, profile, profile_id, cm_choice->choice, + cm_choice->text, NULL); + + dict_key = CFStringCreateWithFormat(kCFAllocatorDefault, NULL, + CFSTR("%u"), profile_id); + if (dict_key) + { + CFDictionarySetValue(profiles, dict_key, profile); + CFRelease(dict_key); + } + + CFRelease(profile); + +# else + apple_init_profile(ppd, NULL, profile, profile_id, cm_choice->choice, + cm_choice->text, NULL); + profile ++; +# endif /* HAVE_COLORSYNCREGISTERDEVICE */ + + if (cm_choice->marked) + default_profile_id = profile_id; + } + } + else + { + /* + * Use the default colorspace... + */ + + attr = ppdFindAttr(ppd, "DefaultColorSpace", NULL); + + num_profiles = (attr && ppd->colorspace == PPD_CS_GRAY) ? 1 : 2; + +# ifdef HAVE_COLORSYNCREGISTERDEVICE + /* + * Add the grayscale profile first. We always have a grayscale profile. + */ + + profile = CFDictionaryCreateMutable(kCFAllocatorDefault, 0, + &kCFTypeDictionaryKeyCallBacks, + &kCFTypeDictionaryValueCallBacks); + + if (!profile) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate memory for color profile."); + CFRelease(profiles); + ppdClose(ppd); + return; + } + + profile_id = _ppdHashName("Gray.."); + apple_init_profile(ppd, NULL, profile, profile_id, "Gray", "Gray", NULL); + + dict_key = CFStringCreateWithFormat(kCFAllocatorDefault, NULL, CFSTR("%u"), + profile_id); + if (dict_key) + { + CFDictionarySetValue(profiles, dict_key, profile); + CFRelease(dict_key); + } + + CFRelease(profile); + + /* + * Then add the RGB/CMYK/DeviceN color profile... + */ + + profile = CFDictionaryCreateMutable(kCFAllocatorDefault, 0, + &kCFTypeDictionaryKeyCallBacks, + &kCFTypeDictionaryValueCallBacks); + + if (!profile) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate memory for color profile."); + CFRelease(profiles); + ppdClose(ppd); + return; + } + + switch (ppd->colorspace) + { + default : + case PPD_CS_RGB : + case PPD_CS_CMY : + profile_id = _ppdHashName("RGB.."); + apple_init_profile(ppd, NULL, profile, profile_id, "RGB", "RGB", + NULL); + break; + + case PPD_CS_RGBK : + case PPD_CS_CMYK : + profile_id = _ppdHashName("CMYK.."); + apple_init_profile(ppd, NULL, profile, profile_id, "CMYK", "CMYK", + NULL); + break; + + case PPD_CS_GRAY : + if (attr) + break; + + case PPD_CS_N : + profile_id = _ppdHashName("DeviceN.."); + apple_init_profile(ppd, NULL, profile, profile_id, "DeviceN", + "DeviceN", NULL); + break; + } + + if (CFDictionaryGetCount(profile) > 0) + { + dict_key = CFStringCreateWithFormat(kCFAllocatorDefault, NULL, + CFSTR("%u"), profile_id); + if (dict_key) + { + CFDictionarySetValue(profiles, dict_key, profile); + CFRelease(dict_key); + } + } + + CFRelease(profile); + +# else + /* + * Create an array for the factory profiles... + */ + + if ((profiles = calloc(num_profiles, sizeof(CMDeviceProfileArray))) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate memory for factory profiles."); + ppdClose(ppd); + return; + } + + profiles->profileCount = num_profiles; + + /* + * Add the grayscale profile first. We always have a grayscale profile. + */ + + profile_id = _ppdHashName("Gray.."); + apple_init_profile(ppd, NULL, profiles->profiles, profile_id, "Gray", + "Gray", NULL); + + /* + * Then add the RGB/CMYK/DeviceN color profile... + */ + + switch (ppd->colorspace) + { + default : + case PPD_CS_RGB : + case PPD_CS_CMY : + profile_id = _ppdHashName("RGB.."); + apple_init_profile(ppd, NULL, profiles->profiles + 1, profile_id, + "RGB", "RGB", NULL); + break; + case PPD_CS_RGBK : + case PPD_CS_CMYK : + profile_id = _ppdHashName("CMYK.."); + apple_init_profile(ppd, NULL, profiles->profiles + 1, profile_id, + "CMYK", "CMYK", NULL); + break; + + case PPD_CS_GRAY : + if (attr) + break; + + case PPD_CS_N : + profile_id = _ppdHashName("DeviceN.."); + apple_init_profile(ppd, NULL, profiles->profiles + 1, profile_id, + "DeviceN", "DeviceN", NULL); + break; + } +# endif /* HAVE_COLORSYNCREGISTERDEVICE */ + } + + if (num_profiles > 0) + { + /* + * Make sure we have a default profile ID... + */ + + if (!default_profile_id) + default_profile_id = profile_id; /* Last profile */ + +# ifdef HAVE_COLORSYNCREGISTERDEVICE + dict_key = CFStringCreateWithFormat(kCFAllocatorDefault, NULL, CFSTR("%u"), + default_profile_id); + if (dict_key) + { + CFDictionarySetValue(profiles, kColorSyncDeviceDefaultProfileID, + dict_key); + CFRelease(dict_key); + } +# endif /* HAVE_COLORSYNCREGISTERDEVICE */ + + /* + * Get the device ID hash and pathelogical name dictionary. + */ + + cupsdLogMessage(CUPSD_LOG_INFO, "Registering ICC color profiles for \"%s\"", + p->name); + + device_id = _ppdHashName(p->name); + device_name = CFDictionaryCreateMutable(kCFAllocatorDefault, 0, + &kCFTypeDictionaryKeyCallBacks, + &kCFTypeDictionaryValueCallBacks); + printer_name = CFStringCreateWithCString(kCFAllocatorDefault, + p->name, kCFStringEncodingUTF8); + + if (device_name && printer_name) + { + CFDictionarySetValue(device_name, CFSTR("en_US"), printer_name); + + /* + * Register the device with ColorSync... + */ + +# ifdef HAVE_COLORSYNCREGISTERDEVICE + CFTypeRef deviceDictKeys[] = + { /* Device keys */ + kColorSyncDeviceDescriptions, + kColorSyncFactoryProfiles, + kColorSyncDeviceUserScope, + kColorSyncDeviceHostScope + }; + CFTypeRef deviceDictVals[] = + { /* Device values */ + device_name, + profiles, + kCFPreferencesAnyUser, + kCFPreferencesCurrentHost + }; + CFDictionaryRef deviceDict; /* Device dictionary */ + CFUUIDRef deviceUUID; /* Device UUID */ + + deviceDict = CFDictionaryCreate(kCFAllocatorDefault, + (const void **)deviceDictKeys, + (const void **)deviceDictVals, + sizeof(deviceDictKeys) / + sizeof(deviceDictKeys[0]), + &kCFTypeDictionaryKeyCallBacks, + &kCFTypeDictionaryValueCallBacks); + deviceUUID = ColorSyncCreateUUIDFromUInt32(device_id); + + if (!deviceDict || !deviceUUID || + !ColorSyncRegisterDevice(kColorSyncPrinterDeviceClass, deviceUUID, + deviceDict)) + error = 1001; + + if (deviceUUID) + CFRelease(deviceUUID); + + if (deviceDict) + CFRelease(deviceDict); + +# else + error = CMRegisterColorDevice(cmPrinterDeviceClass, device_id, + device_name, &scope); + + /* + * Register the profiles... + */ + + if (error == noErr) + error = CMSetDeviceFactoryProfiles(cmPrinterDeviceClass, device_id, + default_profile_id, profiles); +# endif /* HAVE_COLORSYNCREGISTERDEVICE */ + } + else + error = 1000; + + /* + * Clean up... + */ + + if (error != noErr) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to register ICC color profiles for \"%s\": %d", + p->name, (int)error); + + if (printer_name) + CFRelease(printer_name); + + if (device_name) + CFRelease(device_name); + } + + /* + * Free any memory we used... + */ + +# ifdef HAVE_COLORSYNCREGISTERDEVICE + CFRelease(profiles); + +# else + if (num_profiles > 0) + { + for (profile = profiles->profiles; + num_profiles > 0; + profile ++, num_profiles --) + CFRelease(profile->profileName); + + free(profiles); + } +# endif /* HAVE_COLORSYNCREGISTERDEVICE */ + + ppdClose(ppd); +} + + +/* + * 'apple_unregister_profiles()' - Remove color profiles for the specified + * printer. + */ + +static void +apple_unregister_profiles( + cupsd_printer_t *p) /* I - Printer */ +{ + /* + * Make sure ColorSync is available... + */ + +# ifdef HAVE_COLORSYNCREGISTERDEVICE + if (ColorSyncUnregisterDevice != NULL) + { + /* + * Because we may have registered the printer profiles using a prior device + * ID-based UUID, remove both the old style UUID and current UUID for the + * printer. + */ + + CFUUIDRef deviceUUID; /* Device UUID */ + + deviceUUID = ColorSyncCreateUUIDFromUInt32(_ppdHashName(p->name)); + if (deviceUUID) + { + ColorSyncUnregisterDevice(kColorSyncPrinterDeviceClass, deviceUUID); + CFRelease(deviceUUID); + } + } + +# else + if (CMUnregisterColorDevice != NULL) + CMUnregisterColorDevice(cmPrinterDeviceClass, _ppdHashName(p->name)); +# endif /* HAVE_COLORSYNCREGISTERDEVICE */ +} +#endif /* __APPLE__ */ + + +/* + * 'apply_printer_defaults()' - Apply printer default options to a job. + */ + +static void +apply_printer_defaults( + cupsd_printer_t *printer, /* I - Printer */ + cupsd_job_t *job) /* I - Job */ +{ + int i, /* Looping var */ + num_options; /* Number of default options */ + cups_option_t *options, /* Default options */ + *option; /* Current option */ + + + /* + * Collect all of the default options and add the missing ones to the + * job object... + */ + + for (i = printer->num_options, num_options = 0, options = NULL, + option = printer->options; + i > 0; + i --, option ++) + if (!ippFindAttribute(job->attrs, option->name, IPP_TAG_ZERO)) + { + num_options = cupsAddOption(option->name, option->value, num_options, + &options); + } + + /* + * Encode these options as attributes in the job object... + */ + + cupsEncodeOptions2(job->attrs, num_options, options, IPP_TAG_JOB); + cupsFreeOptions(num_options, options); +} + + +/* + * 'authenticate_job()' - Set job authentication info. + */ + +static void +authenticate_job(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Job URI */ +{ + ipp_attribute_t *attr, /* job-id attribute */ + *auth_info; /* auth-info attribute */ + int jobid; /* Job ID */ + cupsd_job_t *job; /* Current job */ + char scheme[HTTP_MAX_URI], + /* Method portion of URI */ + username[HTTP_MAX_URI], + /* Username portion of URI */ + host[HTTP_MAX_URI], + /* Host portion of URI */ + resource[HTTP_MAX_URI]; + /* Resource portion of URI */ + int port; /* Port portion of URI */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "authenticate_job(%p[%d], %s)", + con, con->http.fd, uri->values[0].string.text); + + /* + * Start with "everything is OK" status... + */ + + con->response->request.status.status_code = IPP_OK; + + /* + * See if we have a job URI or a printer URI... + */ + + if (!strcmp(uri->name, "printer-uri")) + { + /* + * Got a printer URI; see if we also have a job-id attribute... + */ + + if ((attr = ippFindAttribute(con->request, "job-id", + IPP_TAG_INTEGER)) == NULL) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Got a printer-uri attribute but no job-id.")); + return; + } + + jobid = attr->values[0].integer; + } + else + { + /* + * Got a job URI; parse it to get the job ID... + */ + + httpSeparateURI(HTTP_URI_CODING_ALL, uri->values[0].string.text, scheme, + sizeof(scheme), username, sizeof(username), host, + sizeof(host), &port, resource, sizeof(resource)); + + if (strncmp(resource, "/jobs/", 6)) + { + /* + * Not a valid URI! + */ + + send_ipp_status(con, IPP_BAD_REQUEST, _("Bad job-uri \"%s\"."), + uri->values[0].string.text); + return; + } + + jobid = atoi(resource + 6); + } + + /* + * See if the job exists... + */ + + if ((job = cupsdFindJob(jobid)) == NULL) + { + /* + * Nope - return a "not found" error... + */ + + send_ipp_status(con, IPP_NOT_FOUND, _("Job #%d does not exist."), jobid); + return; + } + + /* + * See if the job has been completed... + */ + + if (job->state_value != IPP_JOB_HELD) + { + /* + * Return a "not-possible" error... + */ + + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Job #%d is not held for authentication."), + jobid); + return; + } + + /* + * See if we have already authenticated... + */ + + auth_info = ippFindAttribute(con->request, "auth-info", IPP_TAG_TEXT); + + if (!con->username[0] && !auth_info) + { + cupsd_printer_t *printer; /* Job destination */ + + /* + * No auth data. If we need to authenticate via Kerberos, send a + * HTTP auth challenge, otherwise just return an IPP error... + */ + + printer = cupsdFindDest(job->dest); + + if (printer && printer->num_auth_info_required > 0 && + !strcmp(printer->auth_info_required[0], "negotiate")) + send_http_error(con, HTTP_UNAUTHORIZED, printer); + else + send_ipp_status(con, IPP_NOT_AUTHORIZED, + _("No authentication information provided.")); + return; + } + + /* + * See if the job is owned by the requesting user... + */ + + if (!validate_user(job, con, job->username, username, sizeof(username))) + { + send_http_error(con, con->username[0] ? HTTP_FORBIDDEN : HTTP_UNAUTHORIZED, + cupsdFindDest(job->dest)); + return; + } + + /* + * Save the authentication information for this job... + */ + + save_auth_info(con, job, auth_info); + + /* + * Reset the job-hold-until value to "no-hold"... + */ + + if ((attr = ippFindAttribute(job->attrs, "job-hold-until", + IPP_TAG_KEYWORD)) == NULL) + attr = ippFindAttribute(job->attrs, "job-hold-until", IPP_TAG_NAME); + + if (attr) + { + attr->value_tag = IPP_TAG_KEYWORD; + cupsdSetString(&(attr->values[0].string.text), "no-hold"); + } + + /* + * Release the job and return... + */ + + cupsdReleaseJob(job); + + cupsdAddEvent(CUPSD_EVENT_JOB_STATE, NULL, job, "Job authenticated by user"); + + cupsdLogJob(job, CUPSD_LOG_INFO, "Authenticated by \"%s\".", con->username); + + cupsdCheckJobs(); +} + + +/* + * 'cancel_all_jobs()' - Cancel all or selected print jobs. + */ + +static void +cancel_all_jobs(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Job or Printer URI */ +{ + int i; /* Looping var */ + http_status_t status; /* Policy status */ + cups_ptype_t dtype; /* Destination type */ + char scheme[HTTP_MAX_URI], /* Scheme portion of URI */ + userpass[HTTP_MAX_URI], /* Username portion of URI */ + hostname[HTTP_MAX_URI], /* Host portion of URI */ + resource[HTTP_MAX_URI]; /* Resource portion of URI */ + int port; /* Port portion of URI */ + ipp_attribute_t *attr; /* Attribute in request */ + const char *username = NULL; /* Username */ + cupsd_jobaction_t purge = CUPSD_JOB_DEFAULT; + /* Purge? */ + cupsd_printer_t *printer; /* Printer */ + ipp_attribute_t *job_ids; /* job-ids attribute */ + cupsd_job_t *job; /* Job */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cancel_all_jobs(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * Get the jobs to cancel/purge... + */ + + switch (con->request->request.op.operation_id) + { + case IPP_PURGE_JOBS : + /* + * Get the username (if any) for the jobs we want to cancel (only if + * "my-jobs" is specified... + */ + + if ((attr = ippFindAttribute(con->request, "my-jobs", + IPP_TAG_BOOLEAN)) != NULL && + attr->values[0].boolean) + { + if ((attr = ippFindAttribute(con->request, "requesting-user-name", + IPP_TAG_NAME)) != NULL) + username = attr->values[0].string.text; + else + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Missing requesting-user-name attribute.")); + return; + } + } + + /* + * Look for the "purge-jobs" attribute... + */ + + if ((attr = ippFindAttribute(con->request, "purge-jobs", + IPP_TAG_BOOLEAN)) != NULL) + purge = attr->values[0].boolean ? CUPSD_JOB_PURGE : CUPSD_JOB_DEFAULT; + else + purge = CUPSD_JOB_PURGE; + break; + + case IPP_CANCEL_MY_JOBS : + if (con->username[0]) + username = con->username; + else if ((attr = ippFindAttribute(con->request, "requesting-user-name", + IPP_TAG_NAME)) != NULL) + username = attr->values[0].string.text; + else + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Missing requesting-user-name attribute.")); + return; + } + + default : + break; + } + + job_ids = ippFindAttribute(con->request, "job-ids", IPP_TAG_INTEGER); + + /* + * See if we have a printer URI... + */ + + if (strcmp(uri->name, "printer-uri")) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("The printer-uri attribute is required.")); + return; + } + + /* + * And if the destination is valid... + */ + + if (!cupsdValidateDest(uri->values[0].string.text, &dtype, &printer)) + { + /* + * Bad URI? + */ + + httpSeparateURI(HTTP_URI_CODING_ALL, uri->values[0].string.text, + scheme, sizeof(scheme), userpass, sizeof(userpass), + hostname, sizeof(hostname), &port, + resource, sizeof(resource)); + + if ((!strncmp(resource, "/printers/", 10) && resource[10]) || + (!strncmp(resource, "/classes/", 9) && resource[9])) + { + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(DefaultPolicyPtr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, NULL); + return; + } + + if (job_ids) + { + for (i = 0; i < job_ids->num_values; i ++) + { + if (!cupsdFindJob(job_ids->values[i].integer)) + break; + } + + if (i < job_ids->num_values) + { + send_ipp_status(con, IPP_NOT_FOUND, _("Job #%d does not exist."), + job_ids->values[i].integer); + return; + } + + for (i = 0; i < job_ids->num_values; i ++) + { + job = cupsdFindJob(job_ids->values[i].integer); + + cupsdSetJobState(job, IPP_JOB_CANCELED, purge, + purge == CUPSD_JOB_PURGE ? "Job purged by user." : + "Job canceled by user."); + } + + cupsdLogMessage(CUPSD_LOG_INFO, "Selected jobs were %s by \"%s\".", + purge == CUPSD_JOB_PURGE ? "purged" : "canceled", + get_username(con)); + } + else + { + /* + * Cancel all jobs on all printers... + */ + + cupsdCancelJobs(NULL, username, purge); + + cupsdLogMessage(CUPSD_LOG_INFO, "All jobs were %s by \"%s\".", + purge == CUPSD_JOB_PURGE ? "purged" : "canceled", + get_username(con)); + } + } + else + { + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(printer->op_policy_ptr, con, + NULL)) != HTTP_OK) + { + send_http_error(con, status, printer); + return; + } + + if (job_ids) + { + for (i = 0; i < job_ids->num_values; i ++) + { + if ((job = cupsdFindJob(job_ids->values[i].integer)) == NULL || + _cups_strcasecmp(job->dest, printer->name)) + break; + } + + if (i < job_ids->num_values) + { + send_ipp_status(con, IPP_NOT_FOUND, _("Job #%d does not exist."), + job_ids->values[i].integer); + return; + } + + for (i = 0; i < job_ids->num_values; i ++) + { + job = cupsdFindJob(job_ids->values[i].integer); + + cupsdSetJobState(job, IPP_JOB_CANCELED, purge, + purge == CUPSD_JOB_PURGE ? "Job purged by user." : + "Job canceled by user."); + } + + cupsdLogMessage(CUPSD_LOG_INFO, "Selected jobs were %s by \"%s\".", + purge == CUPSD_JOB_PURGE ? "purged" : "canceled", + get_username(con)); + } + else + { + /* + * Cancel all of the jobs on the named printer... + */ + + cupsdCancelJobs(printer->name, username, purge); + + cupsdLogMessage(CUPSD_LOG_INFO, "All jobs on \"%s\" were %s by \"%s\".", + printer->name, + purge == CUPSD_JOB_PURGE ? "purged" : "canceled", + get_username(con)); + } + } + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'cancel_job()' - Cancel a print job. + */ + +static void +cancel_job(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Job or Printer URI */ +{ + ipp_attribute_t *attr; /* Current attribute */ + int jobid; /* Job ID */ + char scheme[HTTP_MAX_URI], /* Scheme portion of URI */ + username[HTTP_MAX_URI], /* Username portion of URI */ + host[HTTP_MAX_URI], /* Host portion of URI */ + resource[HTTP_MAX_URI]; /* Resource portion of URI */ + int port; /* Port portion of URI */ + cupsd_job_t *job; /* Job information */ + cups_ptype_t dtype; /* Destination type (printer/class) */ + cupsd_printer_t *printer; /* Printer data */ + cupsd_jobaction_t purge; /* Purge the job? */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cancel_job(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * See if we have a job URI or a printer URI... + */ + + if (!strcmp(uri->name, "printer-uri")) + { + /* + * Got a printer URI; see if we also have a job-id attribute... + */ + + if ((attr = ippFindAttribute(con->request, "job-id", + IPP_TAG_INTEGER)) == NULL) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Got a printer-uri attribute but no job-id.")); + return; + } + + if ((jobid = attr->values[0].integer) == 0) + { + /* + * Find the current job on the specified printer... + */ + + if (!cupsdValidateDest(uri->values[0].string.text, &dtype, &printer)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * See if there are any pending jobs... + */ + + for (job = (cupsd_job_t *)cupsArrayFirst(ActiveJobs); + job; + job = (cupsd_job_t *)cupsArrayNext(ActiveJobs)) + if (job->state_value <= IPP_JOB_PROCESSING && + !_cups_strcasecmp(job->dest, printer->name)) + break; + + if (job) + jobid = job->id; + else + { + /* + * No, try stopped jobs... + */ + + for (job = (cupsd_job_t *)cupsArrayFirst(ActiveJobs); + job; + job = (cupsd_job_t *)cupsArrayNext(ActiveJobs)) + if (job->state_value == IPP_JOB_STOPPED && + !_cups_strcasecmp(job->dest, printer->name)) + break; + + if (job) + jobid = job->id; + else + { + send_ipp_status(con, IPP_NOT_POSSIBLE, _("No active jobs on %s."), + printer->name); + return; + } + } + } + } + else + { + /* + * Got a job URI; parse it to get the job ID... + */ + + httpSeparateURI(HTTP_URI_CODING_ALL, uri->values[0].string.text, scheme, + sizeof(scheme), username, sizeof(username), host, + sizeof(host), &port, resource, sizeof(resource)); + + if (strncmp(resource, "/jobs/", 6)) + { + /* + * Not a valid URI! + */ + + send_ipp_status(con, IPP_BAD_REQUEST, _("Bad job-uri \"%s\"."), + uri->values[0].string.text); + return; + } + + jobid = atoi(resource + 6); + } + + /* + * Look for the "purge-job" attribute... + */ + + if ((attr = ippFindAttribute(con->request, "purge-job", + IPP_TAG_BOOLEAN)) != NULL) + purge = attr->values[0].boolean ? CUPSD_JOB_PURGE : CUPSD_JOB_DEFAULT; + else + purge = CUPSD_JOB_DEFAULT; + + /* + * See if the job exists... + */ + + if ((job = cupsdFindJob(jobid)) == NULL) + { + /* + * Nope - return a "not found" error... + */ + + send_ipp_status(con, IPP_NOT_FOUND, _("Job #%d does not exist."), jobid); + return; + } + + /* + * See if the job is owned by the requesting user... + */ + + if (!validate_user(job, con, job->username, username, sizeof(username))) + { + send_http_error(con, con->username[0] ? HTTP_FORBIDDEN : HTTP_UNAUTHORIZED, + cupsdFindDest(job->dest)); + return; + } + + /* + * See if the job is already completed, canceled, or aborted; if so, + * we can't cancel... + */ + + if (job->state_value >= IPP_JOB_CANCELED && purge != CUPSD_JOB_PURGE) + { + switch (job->state_value) + { + case IPP_JOB_CANCELED : + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Job #%d is already canceled - can\'t cancel."), + jobid); + break; + + case IPP_JOB_ABORTED : + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Job #%d is already aborted - can\'t cancel."), + jobid); + break; + + default : + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Job #%d is already completed - can\'t cancel."), + jobid); + break; + } + + return; + } + + /* + * Cancel the job and return... + */ + + cupsdSetJobState(job, IPP_JOB_CANCELED, purge, + purge == CUPSD_JOB_PURGE ? "Job purged by \"%s\"" : + "Job canceled by \"%s\"", + username); + cupsdCheckJobs(); + + if (purge == CUPSD_JOB_PURGE) + cupsdLogMessage(CUPSD_LOG_INFO, "[Job %d] Purged by \"%s\".", jobid, + username); + else + cupsdLogMessage(CUPSD_LOG_INFO, "[Job %d] Canceled by \"%s\".", jobid, + username); + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'cancel_subscription()' - Cancel a subscription. + */ + +static void +cancel_subscription( + cupsd_client_t *con, /* I - Client connection */ + int sub_id) /* I - Subscription ID */ +{ + http_status_t status; /* Policy status */ + cupsd_subscription_t *sub; /* Subscription */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cancel_subscription(con=%p[%d], sub_id=%d)", + con, con->http.fd, sub_id); + + /* + * Is the subscription ID valid? + */ + + if ((sub = cupsdFindSubscription(sub_id)) == NULL) + { + /* + * Bad subscription ID... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("Subscription #%d does not exist."), sub_id); + return; + } + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(sub->dest ? sub->dest->op_policy_ptr : + DefaultPolicyPtr, + con, sub->owner)) != HTTP_OK) + { + send_http_error(con, status, sub->dest); + return; + } + + /* + * Cancel the subscription... + */ + + cupsdDeleteSubscription(sub, 1); + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'check_rss_recipient()' - Check that we do not have a duplicate RSS feed URI. + */ + +static int /* O - 1 if OK, 0 if not */ +check_rss_recipient( + const char *recipient) /* I - Recipient URI */ +{ + cupsd_subscription_t *sub; /* Current subscription */ + + + for (sub = (cupsd_subscription_t *)cupsArrayFirst(Subscriptions); + sub; + sub = (cupsd_subscription_t *)cupsArrayNext(Subscriptions)) + if (sub->recipient) + { + /* + * Compare the URIs up to the first ?... + */ + + const char *r1, *r2; + + for (r1 = recipient, r2 = sub->recipient; + *r1 == *r2 && *r1 && *r1 != '?' && *r2 && *r2 != '?'; + r1 ++, r2 ++); + + if (*r1 == *r2) + return (0); + } + + return (1); +} + + +/* + * 'check_quotas()' - Check quotas for a printer and user. + */ + +static int /* O - 1 if OK, 0 if forbidden, + -1 if limit reached */ +check_quotas(cupsd_client_t *con, /* I - Client connection */ + cupsd_printer_t *p) /* I - Printer or class */ +{ + char username[33], /* Username */ + *name; /* Current user name */ + cupsd_quota_t *q; /* Quota data */ +#ifdef HAVE_MBR_UID_TO_UUID + /* + * Use Apple membership APIs which require that all names represent + * valid user account or group records accessible by the server. + */ + + uuid_t usr_uuid; /* UUID for job requesting user */ + uuid_t usr2_uuid; /* UUID for ACL user name entry */ + uuid_t grp_uuid; /* UUID for ACL group name entry */ + int mbr_err; /* Error from membership function */ + int is_member; /* Is this user a member? */ +#else + /* + * Use standard POSIX APIs for checking users and groups... + */ + + struct passwd *pw; /* User password data */ +#endif /* HAVE_MBR_UID_TO_UUID */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "check_quotas(%p[%d], %p[%s])", + con, con->http.fd, p, p->name); + + /* + * Figure out who is printing... + */ + + strlcpy(username, get_username(con), sizeof(username)); + + if ((name = strchr(username, '@')) != NULL) + *name = '\0'; /* Strip @REALM */ + + /* + * Check global active job limits for printers and users... + */ + + if (MaxJobsPerPrinter) + { + /* + * Check if there are too many pending jobs on this printer... + */ + + if (cupsdGetPrinterJobCount(p->name) >= MaxJobsPerPrinter) + { + cupsdLogMessage(CUPSD_LOG_INFO, "Too many jobs for printer \"%s\"...", + p->name); + return (-1); + } + } + + if (MaxJobsPerUser) + { + /* + * Check if there are too many pending jobs for this user... + */ + + if (cupsdGetUserJobCount(username) >= MaxJobsPerUser) + { + cupsdLogMessage(CUPSD_LOG_INFO, "Too many jobs for user \"%s\"...", + username); + return (-1); + } + } + + /* + * Check against users... + */ + + if (cupsArrayCount(p->users) == 0 && p->k_limit == 0 && p->page_limit == 0) + return (1); + + if (cupsArrayCount(p->users)) + { +#ifdef HAVE_MBR_UID_TO_UUID + /* + * Get UUID for job requesting user... + */ + + if (mbr_user_name_to_uuid((char *)username, usr_uuid)) + { + /* + * Unknown user... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "check_quotas: UUID lookup failed for user \"%s\"", + username); + cupsdLogMessage(CUPSD_LOG_INFO, + "Denying user \"%s\" access to printer \"%s\" " + "(unknown user)...", + username, p->name); + return (0); + } +#else + /* + * Get UID and GID of requesting user... + */ + + pw = getpwnam(username); + endpwent(); +#endif /* HAVE_MBR_UID_TO_UUID */ + + for (name = (char *)cupsArrayFirst(p->users); + name; + name = (char *)cupsArrayNext(p->users)) + if (name[0] == '@') + { + /* + * Check group membership... + */ + +#ifdef HAVE_MBR_UID_TO_UUID + if (name[1] == '#') + { + if (uuid_parse(name + 2, grp_uuid)) + uuid_clear(grp_uuid); + } + else if ((mbr_err = mbr_group_name_to_uuid(name + 1, grp_uuid)) != 0) + { + /* + * Invalid ACL entries are ignored for matching; just record a + * warning in the log... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "check_quotas: UUID lookup failed for ACL entry " + "\"%s\" (err=%d)", name, mbr_err); + cupsdLogMessage(CUPSD_LOG_WARN, + "Access control entry \"%s\" not a valid group name; " + "entry ignored", name); + } + + if ((mbr_err = mbr_check_membership(usr_uuid, grp_uuid, + &is_member)) != 0) + { + /* + * At this point, there should be no errors, but check anyways... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "check_quotas: group \"%s\" membership check " + "failed (err=%d)", name + 1, mbr_err); + is_member = 0; + } + + /* + * Stop if we found a match... + */ + + if (is_member) + break; + +#else + if (cupsdCheckGroup(username, pw, name + 1)) + break; +#endif /* HAVE_MBR_UID_TO_UUID */ + } +#ifdef HAVE_MBR_UID_TO_UUID + else + { + if (name[0] == '#') + { + if (uuid_parse(name + 1, usr2_uuid)) + uuid_clear(usr2_uuid); + } + else if ((mbr_err = mbr_user_name_to_uuid(name, usr2_uuid)) != 0) + { + /* + * Invalid ACL entries are ignored for matching; just record a + * warning in the log... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "check_quotas: UUID lookup failed for ACL entry " + "\"%s\" (err=%d)", name, mbr_err); + cupsdLogMessage(CUPSD_LOG_WARN, + "Access control entry \"%s\" not a valid user name; " + "entry ignored", name); + } + + if (!uuid_compare(usr_uuid, usr2_uuid)) + break; + } +#else + else if (!_cups_strcasecmp(username, name)) + break; +#endif /* HAVE_MBR_UID_TO_UUID */ + + if ((name != NULL) == p->deny_users) + { + cupsdLogMessage(CUPSD_LOG_INFO, + "Denying user \"%s\" access to printer \"%s\"...", + username, p->name); + return (0); + } + } + + /* + * Check quotas... + */ + + if (p->k_limit || p->page_limit) + { + if ((q = cupsdUpdateQuota(p, username, 0, 0)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate quota data for user \"%s\"", + username); + return (-1); + } + + if ((q->k_count >= p->k_limit && p->k_limit) || + (q->page_count >= p->page_limit && p->page_limit)) + { + cupsdLogMessage(CUPSD_LOG_INFO, "User \"%s\" is over the quota limit...", + username); + return (-1); + } + } + + /* + * If we have gotten this far, we're done! + */ + + return (1); +} + + +/* + * 'close_job()' - Close a multi-file job. + */ + +static void +close_job(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Printer URI */ +{ + cupsd_job_t *job; /* Job */ + ipp_attribute_t *attr; /* Attribute */ + char job_uri[HTTP_MAX_URI], + /* Job URI */ + username[256]; /* User name */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "close_job(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * See if we have a job URI or a printer URI... + */ + + if (strcmp(uri->name, "printer-uri")) + { + /* + * job-uri is not supported by Close-Job! + */ + + send_ipp_status(con, IPP_BAD_REQUEST, + _("Close-Job doesn't support the job-uri attribute.")); + return; + } + + /* + * Got a printer URI; see if we also have a job-id attribute... + */ + + if ((attr = ippFindAttribute(con->request, "job-id", + IPP_TAG_INTEGER)) == NULL) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Got a printer-uri attribute but no job-id.")); + return; + } + + if ((job = cupsdFindJob(attr->values[0].integer)) == NULL) + { + /* + * Nope - return a "not found" error... + */ + + send_ipp_status(con, IPP_NOT_FOUND, _("Job #%d does not exist."), + attr->values[0].integer); + return; + } + + /* + * See if the job is owned by the requesting user... + */ + + if (!validate_user(job, con, job->username, username, sizeof(username))) + { + send_http_error(con, con->username[0] ? HTTP_FORBIDDEN : HTTP_UNAUTHORIZED, + cupsdFindDest(job->dest)); + return; + } + + /* + * Add any ending sheet... + */ + + if (cupsdTimeoutJob(job)) + return; + + if (job->state_value == IPP_JOB_STOPPED) + { + job->state->values[0].integer = IPP_JOB_PENDING; + job->state_value = IPP_JOB_PENDING; + } + else if (job->state_value == IPP_JOB_HELD) + { + if ((attr = ippFindAttribute(job->attrs, "job-hold-until", + IPP_TAG_KEYWORD)) == NULL) + attr = ippFindAttribute(job->attrs, "job-hold-until", IPP_TAG_NAME); + + if (!attr || !strcmp(attr->values[0].string.text, "no-hold")) + { + job->state->values[0].integer = IPP_JOB_PENDING; + job->state_value = IPP_JOB_PENDING; + } + } + + job->dirty = 1; + cupsdMarkDirty(CUPSD_DIRTY_JOBS); + + /* + * Fill in the response info... + */ + + httpAssembleURIf(HTTP_URI_CODING_ALL, job_uri, sizeof(job_uri), "ipp", NULL, + con->servername, con->serverport, "/jobs/%d", job->id); + ippAddString(con->response, IPP_TAG_JOB, IPP_TAG_URI, "job-uri", NULL, + job_uri); + + ippAddInteger(con->response, IPP_TAG_JOB, IPP_TAG_INTEGER, "job-id", job->id); + + ippAddInteger(con->response, IPP_TAG_JOB, IPP_TAG_ENUM, "job-state", + job->state_value); + + con->response->request.status.status_code = IPP_OK; + + /* + * Start the job if necessary... + */ + + cupsdCheckJobs(); +} + + +/* + * 'copy_attrs()' - Copy attributes from one request to another. + */ + +static void +copy_attrs(ipp_t *to, /* I - Destination request */ + ipp_t *from, /* I - Source request */ + cups_array_t *ra, /* I - Requested attributes */ + ipp_tag_t group, /* I - Group to copy */ + int quickcopy, /* I - Do a quick copy? */ + cups_array_t *exclude) /* I - Attributes to exclude? */ +{ + ipp_attribute_t *fromattr; /* Source attribute */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "copy_attrs(to=%p, from=%p, ra=%p, group=%x, quickcopy=%d)", + to, from, ra, group, quickcopy); + + if (!to || !from) + return; + + for (fromattr = from->attrs; fromattr; fromattr = fromattr->next) + { + /* + * Filter attributes as needed... + */ + + if ((group != IPP_TAG_ZERO && fromattr->group_tag != group && + fromattr->group_tag != IPP_TAG_ZERO) || !fromattr->name) + continue; + + if (!strcmp(fromattr->name, "job-printer-uri")) + continue; + + if (exclude && + (cupsArrayFind(exclude, fromattr->name) || + cupsArrayFind(exclude, "all"))) + { + /* + * We need to exclude this attribute for security reasons; we require the + * job-id attribute regardless of the security settings for IPP + * conformance. + * + * The job-printer-uri attribute is handled by copy_job_attrs(). + * + * Subscription attribute security is handled by copy_subscription_attrs(). + */ + + if (strcmp(fromattr->name, "job-id")) + continue; + } + + if (!ra || cupsArrayFind(ra, fromattr->name)) + { + /* + * Don't send collection attributes by default to IPP/1.x clients + * since many do not support collections. Also don't send + * media-col-database unless specifically requested by the client. + */ + + if (fromattr->value_tag == IPP_TAG_BEGIN_COLLECTION && + !ra && + (to->request.status.version[0] == 1 || + !strcmp(fromattr->name, "media-col-database"))) + continue; + + ippCopyAttribute(to, fromattr, quickcopy); + } + } +} + + +/* + * 'copy_banner()' - Copy a banner file to the requests directory for the + * specified job. + */ + +static int /* O - Size of banner file in kbytes */ +copy_banner(cupsd_client_t *con, /* I - Client connection */ + cupsd_job_t *job, /* I - Job information */ + const char *name) /* I - Name of banner */ +{ + int i; /* Looping var */ + int kbytes; /* Size of banner file in kbytes */ + char filename[1024]; /* Job filename */ + cupsd_banner_t *banner; /* Pointer to banner */ + cups_file_t *in; /* Input file */ + cups_file_t *out; /* Output file */ + int ch; /* Character from file */ + char attrname[255], /* Name of attribute */ + *s; /* Pointer into name */ + ipp_attribute_t *attr; /* Attribute */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "copy_banner(con=%p[%d], job=%p[%d], name=\"%s\")", + con, con ? con->http.fd : -1, job, job->id, + name ? name : "(null)"); + + /* + * Find the banner; return if not found or "none"... + */ + + if (!name || !strcmp(name, "none") || + (banner = cupsdFindBanner(name)) == NULL) + return (0); + + /* + * Open the banner and job files... + */ + + if (add_file(con, job, banner->filetype, 0)) + return (-1); + + snprintf(filename, sizeof(filename), "%s/d%05d-%03d", RequestRoot, job->id, + job->num_files); + if ((out = cupsFileOpen(filename, "w")) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to create banner job file %s - %s", + filename, strerror(errno)); + job->num_files --; + return (0); + } + + fchmod(cupsFileNumber(out), 0640); + fchown(cupsFileNumber(out), RunUser, Group); + + /* + * Try the localized banner file under the subdirectory... + */ + + strlcpy(attrname, job->attrs->attrs->next->values[0].string.text, + sizeof(attrname)); + if (strlen(attrname) > 2 && attrname[2] == '-') + { + /* + * Convert ll-cc to ll_CC... + */ + + attrname[2] = '_'; + attrname[3] = toupper(attrname[3] & 255); + attrname[4] = toupper(attrname[4] & 255); + } + + snprintf(filename, sizeof(filename), "%s/banners/%s/%s", DataDir, + attrname, name); + + if (access(filename, 0) && strlen(attrname) > 2) + { + /* + * Wasn't able to find "ll_CC" locale file; try the non-national + * localization banner directory. + */ + + attrname[2] = '\0'; + + snprintf(filename, sizeof(filename), "%s/banners/%s/%s", DataDir, + attrname, name); + } + + if (access(filename, 0)) + { + /* + * Use the non-localized banner file. + */ + + snprintf(filename, sizeof(filename), "%s/banners/%s", DataDir, name); + } + + if ((in = cupsFileOpen(filename, "r")) == NULL) + { + cupsFileClose(out); + unlink(filename); + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to open banner template file %s - %s", + filename, strerror(errno)); + job->num_files --; + return (0); + } + + /* + * Parse the file to the end... + */ + + while ((ch = cupsFileGetChar(in)) != EOF) + if (ch == '{') + { + /* + * Get an attribute name... + */ + + for (s = attrname; (ch = cupsFileGetChar(in)) != EOF;) + if (!isalpha(ch & 255) && ch != '-' && ch != '?') + break; + else if (s < (attrname + sizeof(attrname) - 1)) + *s++ = ch; + else + break; + + *s = '\0'; + + if (ch != '}') + { + /* + * Ignore { followed by stuff that is not an attribute name... + */ + + cupsFilePrintf(out, "{%s%c", attrname, ch); + continue; + } + + /* + * See if it is defined... + */ + + if (attrname[0] == '?') + s = attrname + 1; + else + s = attrname; + + if (!strcmp(s, "printer-name")) + { + cupsFilePuts(out, job->dest); + continue; + } + else if ((attr = ippFindAttribute(job->attrs, s, IPP_TAG_ZERO)) == NULL) + { + /* + * See if we have a leading question mark... + */ + + if (attrname[0] != '?') + { + /* + * Nope, write to file as-is; probably a PostScript procedure... + */ + + cupsFilePrintf(out, "{%s}", attrname); + } + + continue; + } + + /* + * Output value(s)... + */ + + for (i = 0; i < attr->num_values; i ++) + { + if (i) + cupsFilePutChar(out, ','); + + switch (attr->value_tag) + { + case IPP_TAG_INTEGER : + case IPP_TAG_ENUM : + if (!strncmp(s, "time-at-", 8)) + { + struct timeval tv; /* Time value */ + + tv.tv_sec = attr->values[i].integer; + tv.tv_usec = 0; + + cupsFilePuts(out, cupsdGetDateTime(&tv, CUPSD_TIME_STANDARD)); + } + else + cupsFilePrintf(out, "%d", attr->values[i].integer); + break; + + case IPP_TAG_BOOLEAN : + cupsFilePrintf(out, "%d", attr->values[i].boolean); + break; + + case IPP_TAG_NOVALUE : + cupsFilePuts(out, "novalue"); + break; + + case IPP_TAG_RANGE : + cupsFilePrintf(out, "%d-%d", attr->values[i].range.lower, + attr->values[i].range.upper); + break; + + case IPP_TAG_RESOLUTION : + cupsFilePrintf(out, "%dx%d%s", attr->values[i].resolution.xres, + attr->values[i].resolution.yres, + attr->values[i].resolution.units == IPP_RES_PER_INCH ? + "dpi" : "dpc"); + break; + + case IPP_TAG_URI : + case IPP_TAG_STRING : + case IPP_TAG_TEXT : + case IPP_TAG_NAME : + case IPP_TAG_KEYWORD : + case IPP_TAG_CHARSET : + case IPP_TAG_LANGUAGE : + if (!_cups_strcasecmp(banner->filetype->type, "postscript")) + { + /* + * Need to quote strings for PS banners... + */ + + const char *p; + + for (p = attr->values[i].string.text; *p; p ++) + { + if (*p == '(' || *p == ')' || *p == '\\') + { + cupsFilePutChar(out, '\\'); + cupsFilePutChar(out, *p); + } + else if (*p < 32 || *p > 126) + cupsFilePrintf(out, "\\%03o", *p & 255); + else + cupsFilePutChar(out, *p); + } + } + else + cupsFilePuts(out, attr->values[i].string.text); + break; + + default : + break; /* anti-compiler-warning-code */ + } + } + } + else if (ch == '\\') /* Quoted char */ + { + ch = cupsFileGetChar(in); + + if (ch != '{') /* Only do special handling for \{ */ + cupsFilePutChar(out, '\\'); + + cupsFilePutChar(out, ch); + } + else + cupsFilePutChar(out, ch); + + cupsFileClose(in); + + kbytes = (cupsFileTell(out) + 1023) / 1024; + + if ((attr = ippFindAttribute(job->attrs, "job-k-octets", + IPP_TAG_INTEGER)) != NULL) + attr->values[0].integer += kbytes; + + cupsFileClose(out); + + return (kbytes); +} + + +/* + * 'copy_file()' - Copy a PPD file or interface script... + */ + +static int /* O - 0 = success, -1 = error */ +copy_file(const char *from, /* I - Source file */ + const char *to) /* I - Destination file */ +{ + cups_file_t *src, /* Source file */ + *dst; /* Destination file */ + int bytes; /* Bytes to read/write */ + char buffer[2048]; /* Copy buffer */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "copy_file(\"%s\", \"%s\")", from, to); + + /* + * Open the source and destination file for a copy... + */ + + if ((src = cupsFileOpen(from, "rb")) == NULL) + return (-1); + + if ((dst = cupsFileOpen(to, "wb")) == NULL) + { + cupsFileClose(src); + return (-1); + } + + /* + * Copy the source file to the destination... + */ + + while ((bytes = cupsFileRead(src, buffer, sizeof(buffer))) > 0) + if (cupsFileWrite(dst, buffer, bytes) < bytes) + { + cupsFileClose(src); + cupsFileClose(dst); + return (-1); + } + + /* + * Close both files and return... + */ + + cupsFileClose(src); + + return (cupsFileClose(dst)); +} + + +/* + * 'copy_model()' - Copy a PPD model file, substituting default values + * as needed... + */ + +static int /* O - 0 = success, -1 = error */ +copy_model(cupsd_client_t *con, /* I - Client connection */ + const char *from, /* I - Source file */ + const char *to) /* I - Destination file */ +{ + fd_set input; /* select() input set */ + struct timeval timeout; /* select() timeout */ + int maxfd; /* Max file descriptor for select() */ + char tempfile[1024]; /* Temporary PPD file */ + int tempfd; /* Temporary PPD file descriptor */ + int temppid; /* Process ID of cups-driverd */ + int temppipe[2]; /* Temporary pipes */ + char *argv[4], /* Command-line arguments */ + *envp[MAX_ENV]; /* Environment */ + cups_file_t *src, /* Source file */ + *dst; /* Destination file */ + ppd_file_t *ppd; /* PPD file */ + int bytes, /* Bytes from pipe */ + total; /* Total bytes from pipe */ + char buffer[2048]; /* Copy buffer */ + int i; /* Looping var */ + char option[PPD_MAX_NAME], /* Option name */ + choice[PPD_MAX_NAME]; /* Choice name */ + ppd_size_t *size; /* Default size */ + int num_defaults; /* Number of default options */ + cups_option_t *defaults; /* Default options */ + char cups_protocol[PPD_MAX_LINE]; + /* cupsProtocol attribute */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "copy_model(con=%p, from=\"%s\", to=\"%s\")", + con, from, to); + + /* + * Run cups-driverd to get the PPD file... + */ + + argv[0] = "cups-driverd"; + argv[1] = "cat"; + argv[2] = (char *)from; + argv[3] = NULL; + + cupsdLoadEnv(envp, (int)(sizeof(envp) / sizeof(envp[0]))); + + snprintf(buffer, sizeof(buffer), "%s/daemon/cups-driverd", ServerBin); + snprintf(tempfile, sizeof(tempfile), "%s/%d.ppd", TempDir, con->http.fd); + tempfd = open(tempfile, O_WRONLY | O_CREAT | O_TRUNC, 0600); + if (tempfd < 0 || cupsdOpenPipe(temppipe)) + return (-1); + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "copy_model: Running \"cups-driverd cat %s\"...", from); + + if (!cupsdStartProcess(buffer, argv, envp, -1, temppipe[1], CGIPipes[1], + -1, -1, 0, DefaultProfile, NULL, &temppid)) + { + close(tempfd); + unlink(tempfile); + + return (-1); + } + + close(temppipe[1]); + + /* + * Wait up to 30 seconds for the PPD file to be copied... + */ + + total = 0; + + if (temppipe[0] > CGIPipes[0]) + maxfd = temppipe[0] + 1; + else + maxfd = CGIPipes[0] + 1; + + for (;;) + { + /* + * See if we have data ready... + */ + + FD_ZERO(&input); + FD_SET(temppipe[0], &input); + FD_SET(CGIPipes[0], &input); + + timeout.tv_sec = 30; + timeout.tv_usec = 0; + + if ((i = select(maxfd, &input, NULL, NULL, &timeout)) < 0) + { + if (errno == EINTR) + continue; + else + break; + } + else if (i == 0) + { + /* + * We have timed out... + */ + + break; + } + + if (FD_ISSET(temppipe[0], &input)) + { + /* + * Read the PPD file from the pipe, and write it to the PPD file. + */ + + if ((bytes = read(temppipe[0], buffer, sizeof(buffer))) > 0) + { + if (write(tempfd, buffer, bytes) < bytes) + break; + + total += bytes; + } + else + break; + } + + if (FD_ISSET(CGIPipes[0], &input)) + cupsdUpdateCGI(); + } + + close(temppipe[0]); + close(tempfd); + + if (!total) + { + /* + * No data from cups-deviced... + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, "copy_model: empty PPD file"); + unlink(tempfile); + return (-1); + } + + /* + * Open the source file for a copy... + */ + + if ((src = cupsFileOpen(tempfile, "rb")) == NULL) + { + unlink(tempfile); + return (-1); + } + + /* + * Read the source file and see what page sizes are supported... + */ + + if ((ppd = _ppdOpen(src, _PPD_LOCALIZATION_NONE)) == NULL) + { + cupsFileClose(src); + unlink(tempfile); + return (-1); + } + + /* + * Open the destination (if possible) and set the default options... + */ + + num_defaults = 0; + defaults = NULL; + cups_protocol[0] = '\0'; + + if ((dst = cupsFileOpen(to, "rb")) != NULL) + { + /* + * Read all of the default lines from the old PPD... + */ + + while (cupsFileGets(dst, buffer, sizeof(buffer))) + if (!strncmp(buffer, "*Default", 8)) + { + /* + * Add the default option... + */ + + if (!ppd_parse_line(buffer, option, sizeof(option), + choice, sizeof(choice))) + { + ppd_option_t *ppdo; /* PPD option */ + + + /* + * Only add the default if the default hasn't already been + * set and the choice exists in the new PPD... + */ + + if (!cupsGetOption(option, num_defaults, defaults) && + (ppdo = ppdFindOption(ppd, option)) != NULL && + ppdFindChoice(ppdo, choice)) + num_defaults = cupsAddOption(option, choice, num_defaults, + &defaults); + } + } + else if (!strncmp(buffer, "*cupsProtocol:", 14)) + strlcpy(cups_protocol, buffer, sizeof(cups_protocol)); + + cupsFileClose(dst); + } + else if ((size = ppdPageSize(ppd, DefaultPaperSize)) != NULL) + { + /* + * Add the default media sizes... + */ + + num_defaults = cupsAddOption("PageSize", size->name, + num_defaults, &defaults); + num_defaults = cupsAddOption("PageRegion", size->name, + num_defaults, &defaults); + num_defaults = cupsAddOption("PaperDimension", size->name, + num_defaults, &defaults); + num_defaults = cupsAddOption("ImageableArea", size->name, + num_defaults, &defaults); + } + + ppdClose(ppd); + + /* + * Open the destination file for a copy... + */ + + if ((dst = cupsFileOpen(to, "wb")) == NULL) + { + cupsFreeOptions(num_defaults, defaults); + cupsFileClose(src); + unlink(tempfile); + return (-1); + } + + /* + * Copy the source file to the destination... + */ + + cupsFileRewind(src); + + while (cupsFileGets(src, buffer, sizeof(buffer))) + { + if (!strncmp(buffer, "*Default", 8)) + { + /* + * Check for an previous default option choice... + */ + + if (!ppd_parse_line(buffer, option, sizeof(option), + choice, sizeof(choice))) + { + const char *val; /* Default option value */ + + + if ((val = cupsGetOption(option, num_defaults, defaults)) != NULL) + { + /* + * Substitute the previous choice... + */ + + snprintf(buffer, sizeof(buffer), "*Default%s: %s", option, val); + } + } + } + + cupsFilePrintf(dst, "%s\n", buffer); + } + + if (cups_protocol[0]) + cupsFilePrintf(dst, "%s\n", cups_protocol); + + cupsFreeOptions(num_defaults, defaults); + + /* + * Close both files and return... + */ + + cupsFileClose(src); + + unlink(tempfile); + + return (cupsFileClose(dst)); +} + + +/* + * 'copy_job_attrs()' - Copy job attributes. + */ + +static void +copy_job_attrs(cupsd_client_t *con, /* I - Client connection */ + cupsd_job_t *job, /* I - Job */ + cups_array_t *ra, /* I - Requested attributes array */ + cups_array_t *exclude) /* I - Private attributes array */ +{ + char job_uri[HTTP_MAX_URI]; /* Job URI */ + + + /* + * Send the requested attributes for each job... + */ + + if (!cupsArrayFind(exclude, "all")) + { + if ((!exclude || !cupsArrayFind(exclude, "number-of-documents")) && + (!ra || cupsArrayFind(ra, "number-of-documents"))) + ippAddInteger(con->response, IPP_TAG_JOB, IPP_TAG_INTEGER, + "number-of-documents", job->num_files); + + if ((!exclude || !cupsArrayFind(exclude, "job-media-progress")) && + (!ra || cupsArrayFind(ra, "job-media-progress"))) + ippAddInteger(con->response, IPP_TAG_JOB, IPP_TAG_INTEGER, + "job-media-progress", job->progress); + + if ((!exclude || !cupsArrayFind(exclude, "job-more-info")) && + (!ra || cupsArrayFind(ra, "job-more-info"))) + { + httpAssembleURIf(HTTP_URI_CODING_ALL, job_uri, sizeof(job_uri), "http", + NULL, con->servername, con->serverport, "/jobs/%d", + job->id); + ippAddString(con->response, IPP_TAG_JOB, IPP_TAG_URI, + "job-more-info", NULL, job_uri); + } + + if (job->state_value > IPP_JOB_PROCESSING && + (!exclude || !cupsArrayFind(exclude, "job-preserved")) && + (!ra || cupsArrayFind(ra, "job-preserved"))) + ippAddBoolean(con->response, IPP_TAG_JOB, "job-preserved", + job->num_files > 0); + + if ((!exclude || !cupsArrayFind(exclude, "job-printer-up-time")) && + (!ra || cupsArrayFind(ra, "job-printer-up-time"))) + ippAddInteger(con->response, IPP_TAG_JOB, IPP_TAG_INTEGER, + "job-printer-up-time", time(NULL)); + } + + if (!ra || cupsArrayFind(ra, "job-printer-uri")) + { + httpAssembleURIf(HTTP_URI_CODING_ALL, job_uri, sizeof(job_uri), "ipp", NULL, + con->servername, con->serverport, + (job->dtype & CUPS_PRINTER_CLASS) ? "/classes/%s" : + "/printers/%s", + job->dest); + ippAddString(con->response, IPP_TAG_JOB, IPP_TAG_URI, + "job-printer-uri", NULL, job_uri); + } + + if (!ra || cupsArrayFind(ra, "job-uri")) + { + httpAssembleURIf(HTTP_URI_CODING_ALL, job_uri, sizeof(job_uri), "ipp", NULL, + con->servername, con->serverport, "/jobs/%d", + job->id); + ippAddString(con->response, IPP_TAG_JOB, IPP_TAG_URI, + "job-uri", NULL, job_uri); + } + + copy_attrs(con->response, job->attrs, ra, IPP_TAG_JOB, 0, exclude); +} + + +/* + * 'copy_printer_attrs()' - Copy printer attributes. + */ + +static void +copy_printer_attrs( + cupsd_client_t *con, /* I - Client connection */ + cupsd_printer_t *printer, /* I - Printer */ + cups_array_t *ra) /* I - Requested attributes array */ +{ + char printer_uri[HTTP_MAX_URI]; + /* Printer URI */ + char printer_icons[HTTP_MAX_URI]; + /* Printer icons */ + time_t curtime; /* Current time */ + int i; /* Looping var */ + + + /* + * Copy the printer attributes to the response using requested-attributes + * and document-format attributes that may be provided by the client. + */ + + curtime = time(NULL); + + if (!ra || cupsArrayFind(ra, "marker-change-time")) + ippAddInteger(con->response, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "marker-change-time", printer->marker_time); + + if (printer->num_printers > 0 && + (!ra || cupsArrayFind(ra, "member-uris"))) + { + ipp_attribute_t *member_uris; /* member-uris attribute */ + cupsd_printer_t *p2; /* Printer in class */ + ipp_attribute_t *p2_uri; /* printer-uri-supported for class printer */ + + + if ((member_uris = ippAddStrings(con->response, IPP_TAG_PRINTER, + IPP_TAG_URI, "member-uris", + printer->num_printers, NULL, + NULL)) != NULL) + { + for (i = 0; i < printer->num_printers; i ++) + { + p2 = printer->printers[i]; + + if ((p2_uri = ippFindAttribute(p2->attrs, "printer-uri-supported", + IPP_TAG_URI)) != NULL) + member_uris->values[i].string.text = + _cupsStrRetain(p2_uri->values[0].string.text); + else + { + httpAssembleURIf(HTTP_URI_CODING_ALL, printer_uri, + sizeof(printer_uri), "ipp", NULL, con->servername, + con->serverport, + (p2->type & CUPS_PRINTER_CLASS) ? + "/classes/%s" : "/printers/%s", p2->name); + member_uris->values[i].string.text = _cupsStrAlloc(printer_uri); + } + } + } + } + + if (printer->alert && (!ra || cupsArrayFind(ra, "printer-alert"))) + ippAddString(con->response, IPP_TAG_PRINTER, IPP_TAG_STRING, + "printer-alert", NULL, printer->alert); + + if (printer->alert_description && + (!ra || cupsArrayFind(ra, "printer-alert-description"))) + ippAddString(con->response, IPP_TAG_PRINTER, IPP_TAG_TEXT, + "printer-alert-description", NULL, + printer->alert_description); + + if (!ra || cupsArrayFind(ra, "printer-current-time")) + ippAddDate(con->response, IPP_TAG_PRINTER, "printer-current-time", + ippTimeToDate(curtime)); + +#ifdef HAVE_DNSSD + if (!ra || cupsArrayFind(ra, "printer-dns-sd-name")) + { + if (printer->reg_name) + ippAddString(con->response, IPP_TAG_PRINTER, IPP_TAG_NAME, + "printer-dns-sd-name", NULL, printer->reg_name); + else + ippAddInteger(con->response, IPP_TAG_PRINTER, IPP_TAG_NOVALUE, + "printer-dns-sd-name", 0); + } +#endif /* HAVE_DNSSD */ + + if (!ra || cupsArrayFind(ra, "printer-error-policy")) + ippAddString(con->response, IPP_TAG_PRINTER, IPP_TAG_NAME, + "printer-error-policy", NULL, printer->error_policy); + + if (!ra || cupsArrayFind(ra, "printer-error-policy-supported")) + { + static const char * const errors[] =/* printer-error-policy-supported values */ + { + "abort-job", + "retry-current-job", + "retry-job", + "stop-printer" + }; + + if (printer->type & CUPS_PRINTER_CLASS) + ippAddString(con->response, IPP_TAG_PRINTER, IPP_TAG_NAME | IPP_TAG_COPY, + "printer-error-policy-supported", NULL, "retry-current-job"); + else + ippAddStrings(con->response, IPP_TAG_PRINTER, IPP_TAG_NAME | IPP_TAG_COPY, + "printer-error-policy-supported", + sizeof(errors) / sizeof(errors[0]), NULL, errors); + } + + if (!ra || cupsArrayFind(ra, "printer-icons")) + { + httpAssembleURIf(HTTP_URI_CODING_ALL, printer_icons, sizeof(printer_icons), + "http", NULL, con->servername, con->serverport, + "/icons/%s.png", printer->name); + ippAddString(con->response, IPP_TAG_PRINTER, IPP_TAG_URI, "printer-icons", + NULL, printer_icons); + cupsdLogMessage(CUPSD_LOG_DEBUG2, "printer-icons=\"%s\"", printer_icons); + } + + if (!ra || cupsArrayFind(ra, "printer-is-accepting-jobs")) + ippAddBoolean(con->response, IPP_TAG_PRINTER, "printer-is-accepting-jobs", + printer->accepting); + + if (!ra || cupsArrayFind(ra, "printer-is-shared")) + ippAddBoolean(con->response, IPP_TAG_PRINTER, "printer-is-shared", + printer->shared); + + if (!ra || cupsArrayFind(ra, "printer-more-info")) + { + httpAssembleURIf(HTTP_URI_CODING_ALL, printer_uri, sizeof(printer_uri), + "http", NULL, con->servername, con->serverport, + (printer->type & CUPS_PRINTER_CLASS) ? + "/classes/%s" : "/printers/%s", printer->name); + ippAddString(con->response, IPP_TAG_PRINTER, IPP_TAG_URI, + "printer-more-info", NULL, printer_uri); + } + + if (!ra || cupsArrayFind(ra, "printer-op-policy")) + ippAddString(con->response, IPP_TAG_PRINTER, IPP_TAG_NAME, + "printer-op-policy", NULL, printer->op_policy); + + if (!ra || cupsArrayFind(ra, "printer-state")) + ippAddInteger(con->response, IPP_TAG_PRINTER, IPP_TAG_ENUM, "printer-state", + printer->state); + + if (!ra || cupsArrayFind(ra, "printer-state-change-time")) + ippAddInteger(con->response, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "printer-state-change-time", printer->state_time); + + if (!ra || cupsArrayFind(ra, "printer-state-message")) + ippAddString(con->response, IPP_TAG_PRINTER, IPP_TAG_TEXT, + "printer-state-message", NULL, printer->state_message); + + if (!ra || cupsArrayFind(ra, "printer-state-reasons")) + add_printer_state_reasons(con, printer); + + if (!ra || cupsArrayFind(ra, "printer-type")) + { + int type; /* printer-type value */ + + /* + * Add the CUPS-specific printer-type attribute... + */ + + type = printer->type; + + if (printer == DefaultPrinter) + type |= CUPS_PRINTER_DEFAULT; + + if (!printer->accepting) + type |= CUPS_PRINTER_REJECTING; + + if (!printer->shared) + type |= CUPS_PRINTER_NOT_SHARED; + + ippAddInteger(con->response, IPP_TAG_PRINTER, IPP_TAG_ENUM, "printer-type", + type); + } + + if (!ra || cupsArrayFind(ra, "printer-up-time")) + ippAddInteger(con->response, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "printer-up-time", curtime); + + if (!ra || cupsArrayFind(ra, "printer-uri-supported")) + { + httpAssembleURIf(HTTP_URI_CODING_ALL, printer_uri, sizeof(printer_uri), + "ipp", NULL, con->servername, con->serverport, + (printer->type & CUPS_PRINTER_CLASS) ? + "/classes/%s" : "/printers/%s", printer->name); + ippAddString(con->response, IPP_TAG_PRINTER, IPP_TAG_URI, + "printer-uri-supported", NULL, printer_uri); + cupsdLogMessage(CUPSD_LOG_DEBUG2, "printer-uri-supported=\"%s\"", + printer_uri); + } + + if (!ra || cupsArrayFind(ra, "queued-job-count")) + add_queued_job_count(con, printer); + + copy_attrs(con->response, printer->attrs, ra, IPP_TAG_ZERO, 0, NULL); + if (printer->ppd_attrs) + copy_attrs(con->response, printer->ppd_attrs, ra, IPP_TAG_ZERO, 0, NULL); + copy_attrs(con->response, CommonData, ra, IPP_TAG_ZERO, IPP_TAG_COPY, NULL); +} + + +/* + * 'copy_subscription_attrs()' - Copy subscription attributes. + */ + +static void +copy_subscription_attrs( + cupsd_client_t *con, /* I - Client connection */ + cupsd_subscription_t *sub, /* I - Subscription */ + cups_array_t *ra, /* I - Requested attributes array */ + cups_array_t *exclude) /* I - Private attributes array */ +{ + ipp_attribute_t *attr; /* Current attribute */ + char printer_uri[HTTP_MAX_URI]; + /* Printer URI */ + int count; /* Number of events */ + unsigned mask; /* Current event mask */ + const char *name; /* Current event name */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "copy_subscription_attrs(con=%p, sub=%p, ra=%p, exclude=%p)", + con, sub, ra, exclude); + + /* + * Copy the subscription attributes to the response using the + * requested-attributes attribute that may be provided by the client. + */ + + if (!exclude || !cupsArrayFind(exclude, "all")) + { + if ((!exclude || !cupsArrayFind(exclude, "notify-events")) && + (!ra || cupsArrayFind(ra, "notify-events"))) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, "copy_subscription_attrs: notify-events"); + + if ((name = cupsdEventName((cupsd_eventmask_t)sub->mask)) != NULL) + { + /* + * Simple event list... + */ + + ippAddString(con->response, IPP_TAG_SUBSCRIPTION, + (ipp_tag_t)(IPP_TAG_KEYWORD | IPP_TAG_COPY), + "notify-events", NULL, name); + } + else + { + /* + * Complex event list... + */ + + for (mask = 1, count = 0; mask < CUPSD_EVENT_ALL; mask <<= 1) + if (sub->mask & mask) + count ++; + + attr = ippAddStrings(con->response, IPP_TAG_SUBSCRIPTION, + (ipp_tag_t)(IPP_TAG_KEYWORD | IPP_TAG_COPY), + "notify-events", count, NULL, NULL); + + for (mask = 1, count = 0; mask < CUPSD_EVENT_ALL; mask <<= 1) + if (sub->mask & mask) + { + attr->values[count].string.text = + (char *)cupsdEventName((cupsd_eventmask_t)mask); + + count ++; + } + } + } + + if ((!exclude || !cupsArrayFind(exclude, "notify-lease-duration")) && + (!sub->job && (!ra || cupsArrayFind(ra, "notify-lease-duration")))) + ippAddInteger(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_INTEGER, + "notify-lease-duration", sub->lease); + + if ((!exclude || !cupsArrayFind(exclude, "notify-recipient-uri")) && + (sub->recipient && (!ra || cupsArrayFind(ra, "notify-recipient-uri")))) + ippAddString(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_URI, + "notify-recipient-uri", NULL, sub->recipient); + else if ((!exclude || !cupsArrayFind(exclude, "notify-pull-method")) && + (!ra || cupsArrayFind(ra, "notify-pull-method"))) + ippAddString(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_KEYWORD, + "notify-pull-method", NULL, "ippget"); + + if ((!exclude || !cupsArrayFind(exclude, "notify-subscriber-user-name")) && + (!ra || cupsArrayFind(ra, "notify-subscriber-user-name"))) + ippAddString(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_NAME, + "notify-subscriber-user-name", NULL, sub->owner); + + if ((!exclude || !cupsArrayFind(exclude, "notify-time-interval")) && + (!ra || cupsArrayFind(ra, "notify-time-interval"))) + ippAddInteger(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_INTEGER, + "notify-time-interval", sub->interval); + + if (sub->user_data_len > 0 && + (!exclude || !cupsArrayFind(exclude, "notify-user-data")) && + (!ra || cupsArrayFind(ra, "notify-user-data"))) + ippAddOctetString(con->response, IPP_TAG_SUBSCRIPTION, "notify-user-data", + sub->user_data, sub->user_data_len); + } + + if (sub->job && (!ra || cupsArrayFind(ra, "notify-job-id"))) + ippAddInteger(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_INTEGER, + "notify-job-id", sub->job->id); + + if (sub->dest && (!ra || cupsArrayFind(ra, "notify-printer-uri"))) + { + httpAssembleURIf(HTTP_URI_CODING_ALL, printer_uri, sizeof(printer_uri), + "ipp", NULL, con->servername, con->serverport, + "/printers/%s", sub->dest->name); + ippAddString(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_URI, + "notify-printer-uri", NULL, printer_uri); + } + + if (!ra || cupsArrayFind(ra, "notify-subscription-id")) + ippAddInteger(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_INTEGER, + "notify-subscription-id", sub->id); +} + + +/* + * 'create_job()' - Print a file to a printer or class. + */ + +static void +create_job(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Printer URI */ +{ + cupsd_printer_t *printer; /* Printer */ + cupsd_job_t *job; /* New job */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "create_job(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * Is the destination valid? + */ + + if (!cupsdValidateDest(uri->values[0].string.text, NULL, &printer)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * Create the job object... + */ + + if ((job = add_job(con, printer, NULL)) == NULL) + return; + + job->pending_timeout = 1; + + /* + * Save and log the job... + */ + + cupsdLogJob(job, CUPSD_LOG_INFO, "Queued on \"%s\" by \"%s\".", + job->dest, job->username); +} + + +/* + * 'create_requested_array()' - Create an array for the requested-attributes. + */ + +static cups_array_t * /* O - Array of attributes or NULL */ +create_requested_array(ipp_t *request) /* I - IPP request */ +{ + int i; /* Looping var */ + ipp_attribute_t *requested; /* requested-attributes attribute */ + cups_array_t *ra; /* Requested attributes array */ + char *value; /* Current value */ + + + /* + * Get the requested-attributes attribute, and return NULL if we don't + * have one... + */ + + if ((requested = ippFindAttribute(request, "requested-attributes", + IPP_TAG_KEYWORD)) == NULL) + return (NULL); + + /* + * If the attribute contains a single "all" keyword, return NULL... + */ + + if (requested->num_values == 1 && + !strcmp(requested->values[0].string.text, "all")) + return (NULL); + + /* + * Create an array using "strcmp" as the comparison function... + */ + + ra = cupsArrayNew((cups_array_func_t)strcmp, NULL); + + for (i = 0; i < requested->num_values; i ++) + { + value = requested->values[i].string.text; + + if (!strcmp(value, "job-template")) + { + /* Only includes the set of Job Template attributes supported by CUPS */ + cupsArrayAdd(ra, "copies"); + cupsArrayAdd(ra, "copies-default"); + cupsArrayAdd(ra, "copies-supported"); + cupsArrayAdd(ra, "finishings"); + cupsArrayAdd(ra, "finishings-default"); + cupsArrayAdd(ra, "finishings-supported"); + cupsArrayAdd(ra, "job-hold-until"); + cupsArrayAdd(ra, "job-hold-until-default"); + cupsArrayAdd(ra, "job-hold-until-supported"); + cupsArrayAdd(ra, "job-priority"); + cupsArrayAdd(ra, "job-priority-default"); + cupsArrayAdd(ra, "job-priority-supported"); + cupsArrayAdd(ra, "job-sheets"); + cupsArrayAdd(ra, "job-sheets-default"); + cupsArrayAdd(ra, "job-sheets-supported"); + cupsArrayAdd(ra, "media"); + cupsArrayAdd(ra, "media-col"); + cupsArrayAdd(ra, "media-col-default"); + cupsArrayAdd(ra, "media-default"); + cupsArrayAdd(ra, "media-supported"); + cupsArrayAdd(ra, "multiple-document-handling"); + cupsArrayAdd(ra, "multiple-document-handling-default"); + cupsArrayAdd(ra, "multiple-document-handling-supported"); + cupsArrayAdd(ra, "number-up"); + cupsArrayAdd(ra, "number-up-default"); + cupsArrayAdd(ra, "number-up-layout"); + cupsArrayAdd(ra, "number-up-layout-default"); + cupsArrayAdd(ra, "number-up-layout-supported"); + cupsArrayAdd(ra, "number-up-supported"); + cupsArrayAdd(ra, "orientation-requested"); + cupsArrayAdd(ra, "orientation-requested-default"); + cupsArrayAdd(ra, "orientation-requested-supported"); + cupsArrayAdd(ra, "output-bin"); + cupsArrayAdd(ra, "output-bin-default"); + cupsArrayAdd(ra, "output-bin-supported"); + cupsArrayAdd(ra, "page-delivery"); + cupsArrayAdd(ra, "page-delivery-default"); + cupsArrayAdd(ra, "page-delivery-supported"); + cupsArrayAdd(ra, "page-order-received"); + cupsArrayAdd(ra, "page-order-received-default"); + cupsArrayAdd(ra, "page-order-received-supported"); + cupsArrayAdd(ra, "page-ranges"); + cupsArrayAdd(ra, "page-ranges-supported"); + cupsArrayAdd(ra, "presentation-direction-number-up"); + cupsArrayAdd(ra, "presentation-direction-number-up-default"); + cupsArrayAdd(ra, "presentation-direction-number-up-supported"); + cupsArrayAdd(ra, "print-color-mode"); + cupsArrayAdd(ra, "print-color-mode-default"); + cupsArrayAdd(ra, "print-color-mode-supported"); + cupsArrayAdd(ra, "print-content-optimize"); + cupsArrayAdd(ra, "print-content-optimize-default"); + cupsArrayAdd(ra, "print-content-optimize-supported"); + cupsArrayAdd(ra, "print-quality"); + cupsArrayAdd(ra, "print-quality-default"); + cupsArrayAdd(ra, "print-quality-supported"); + cupsArrayAdd(ra, "printer-resolution"); + cupsArrayAdd(ra, "printer-resolution-default"); + cupsArrayAdd(ra, "printer-resolution-supported"); + cupsArrayAdd(ra, "sheet-collate"); + cupsArrayAdd(ra, "sheet-collate-default"); + cupsArrayAdd(ra, "sheet-collate-supported"); + cupsArrayAdd(ra, "sides"); + cupsArrayAdd(ra, "sides-default"); + cupsArrayAdd(ra, "sides-supported"); + } + else if (!strcmp(value, "job-description")) + { + /* Only includes the set of Job Description attributes supported by CUPS */ + cupsArrayAdd(ra, "date-time-at-completed"); + cupsArrayAdd(ra, "date-time-at-creation"); + cupsArrayAdd(ra, "date-time-at-processing"); + cupsArrayAdd(ra, "job-detailed-status-message"); + cupsArrayAdd(ra, "job-document-access-errors"); + cupsArrayAdd(ra, "job-id"); + cupsArrayAdd(ra, "job-impressions"); + cupsArrayAdd(ra, "job-impressions-completed"); + cupsArrayAdd(ra, "job-k-octets"); + cupsArrayAdd(ra, "job-k-octets-processed"); + cupsArrayAdd(ra, "job-mandatory-attributes"); + cupsArrayAdd(ra, "job-media-progress"); + cupsArrayAdd(ra, "job-media-sheets"); + cupsArrayAdd(ra, "job-media-sheets-completed"); + cupsArrayAdd(ra, "job-message-from-operator"); + cupsArrayAdd(ra, "job-more-info"); + cupsArrayAdd(ra, "job-name"); + cupsArrayAdd(ra, "job-originating-user-name"); + cupsArrayAdd(ra, "job-printer-up-time"); + cupsArrayAdd(ra, "job-printer-uri"); + cupsArrayAdd(ra, "job-state"); + cupsArrayAdd(ra, "job-state-message"); + cupsArrayAdd(ra, "job-state-reasons"); + cupsArrayAdd(ra, "job-uri"); + cupsArrayAdd(ra, "number-of-documents"); + cupsArrayAdd(ra, "number-of-intervening-jobs"); + cupsArrayAdd(ra, "output-device-assigned"); + cupsArrayAdd(ra, "time-at-completed"); + cupsArrayAdd(ra, "time-at-creation"); + cupsArrayAdd(ra, "time-at-processing"); + } + else if (!strcmp(value, "printer-description")) + { + /* Only includes the set of Printer Description attributes supported by CUPS */ + cupsArrayAdd(ra, "charset-configured"); + cupsArrayAdd(ra, "charset-supported"); + cupsArrayAdd(ra, "color-supported"); + cupsArrayAdd(ra, "compression-supported"); + cupsArrayAdd(ra, "document-format-default"); + cupsArrayAdd(ra, "document-format-supported"); + cupsArrayAdd(ra, "generated-natural-language-supported"); + cupsArrayAdd(ra, "ipp-versions-supported"); + cupsArrayAdd(ra, "job-creation-attributes-supported"); + cupsArrayAdd(ra, "job-ids-supported"); + cupsArrayAdd(ra, "job-impressions-supported"); + cupsArrayAdd(ra, "job-k-octets-supported"); + cupsArrayAdd(ra, "job-media-sheets-supported"); + cupsArrayAdd(ra, "job-settable-attributes-supported"); + cupsArrayAdd(ra, "jpeg-k-octets-supported"); + cupsArrayAdd(ra, "jpeg-x-dimension-supported"); + cupsArrayAdd(ra, "jpeg-y-dimension-supported"); + cupsArrayAdd(ra, "media-bottom-margin-supported"); + cupsArrayAdd(ra, "media-col-supported"); + cupsArrayAdd(ra, "media-key-supported"); + cupsArrayAdd(ra, "media-left-margin-supported"); + cupsArrayAdd(ra, "media-right-margin-supported"); + cupsArrayAdd(ra, "media-size-supported"); + cupsArrayAdd(ra, "media-source-supported"); + cupsArrayAdd(ra, "media-top-margin-supported"); + cupsArrayAdd(ra, "media-type-supported"); + cupsArrayAdd(ra, "multiple-document-jobs-supported"); + cupsArrayAdd(ra, "multiple-operation-time-out"); + cupsArrayAdd(ra, "natural-language-configured"); + cupsArrayAdd(ra, "notify-max-events-supported"); + cupsArrayAdd(ra, "notify-schemes-supported"); + cupsArrayAdd(ra, "operations-supported"); + cupsArrayAdd(ra, "pages-per-minute"); + cupsArrayAdd(ra, "pages-per-minute-color"); + cupsArrayAdd(ra, "pdf-k-octets-supported"); + cupsArrayAdd(ra, "pdl-override-supported"); + cupsArrayAdd(ra, "printer-alert"); + cupsArrayAdd(ra, "printer-alert-description"); + cupsArrayAdd(ra, "printer-commands"); + cupsArrayAdd(ra, "printer-current-time"); + cupsArrayAdd(ra, "printer-dns-sd-name"); + cupsArrayAdd(ra, "printer-info"); + cupsArrayAdd(ra, "printer-is-accepting-jobs"); + cupsArrayAdd(ra, "printer-is-shared"); + cupsArrayAdd(ra, "printer-location"); + cupsArrayAdd(ra, "printer-make-and-model"); + cupsArrayAdd(ra, "printer-message-from-operator"); + cupsArrayAdd(ra, "printer-more-info"); + cupsArrayAdd(ra, "printer-more-info-manufacturer"); + cupsArrayAdd(ra, "printer-name"); + cupsArrayAdd(ra, "printer-settable-attributes-supported"); + cupsArrayAdd(ra, "printer-state"); + cupsArrayAdd(ra, "printer-state-change-date-time"); + cupsArrayAdd(ra, "printer-state-change-time"); + cupsArrayAdd(ra, "printer-state-message"); + cupsArrayAdd(ra, "printer-state-reasons"); + cupsArrayAdd(ra, "printer-type"); + cupsArrayAdd(ra, "printer-up-time"); + cupsArrayAdd(ra, "printer-uri-supported"); + cupsArrayAdd(ra, "queued-job-count"); + cupsArrayAdd(ra, "reference-uri-schemes-supported"); + cupsArrayAdd(ra, "uri-authentication-supported"); + cupsArrayAdd(ra, "uri-security-supported"); + cupsArrayAdd(ra, "which-jobs-supported"); + } + else if (!strcmp(value, "printer-defaults")) + { + char *name; /* Option name */ + + + for (name = (char *)cupsArrayFirst(CommonDefaults); + name; + name = (char *)cupsArrayNext(CommonDefaults)) + cupsArrayAdd(ra, name); + } + else if (!strcmp(value, "subscription-description")) + { + /* Only includes the set of Subscription Description attributes supported by CUPS */ + cupsArrayAdd(ra, "notify-job-id"); + cupsArrayAdd(ra, "notify-lease-expiration-time"); + cupsArrayAdd(ra, "notify-printer-up-time"); + cupsArrayAdd(ra, "notify-printer-uri"); + cupsArrayAdd(ra, "notify-sequence-number"); + cupsArrayAdd(ra, "notify-subscriber-user-name"); + cupsArrayAdd(ra, "notify-subscription-id"); + } + else if (!strcmp(value, "subscription-template")) + { + /* Only includes the set of Subscription Template attributes supported by CUPS */ + cupsArrayAdd(ra, "notify-attributes"); + cupsArrayAdd(ra, "notify-attributes-supported"); + cupsArrayAdd(ra, "notify-charset"); + cupsArrayAdd(ra, "notify-events"); + cupsArrayAdd(ra, "notify-events-default"); + cupsArrayAdd(ra, "notify-events-supported"); + cupsArrayAdd(ra, "notify-lease-duration"); + cupsArrayAdd(ra, "notify-lease-duration-default"); + cupsArrayAdd(ra, "notify-lease-duration-supported"); + cupsArrayAdd(ra, "notify-natural-language"); + cupsArrayAdd(ra, "notify-pull-method"); + cupsArrayAdd(ra, "notify-pull-method-supported"); + cupsArrayAdd(ra, "notify-recipient-uri"); + cupsArrayAdd(ra, "notify-time-interval"); + cupsArrayAdd(ra, "notify-user-data"); + } + else + cupsArrayAdd(ra, value); + } + + return (ra); +} + + +/* + * 'create_subscription()' - Create a notification subscription. + */ + +static void +create_subscription( + cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Printer URI */ +{ + http_status_t status; /* Policy status */ + int i; /* Looping var */ + ipp_attribute_t *attr; /* Current attribute */ + cups_ptype_t dtype; /* Destination type (printer/class) */ + char scheme[HTTP_MAX_URI], + /* Scheme portion of URI */ + userpass[HTTP_MAX_URI], + /* Username portion of URI */ + host[HTTP_MAX_URI], + /* Host portion of URI */ + resource[HTTP_MAX_URI]; + /* Resource portion of URI */ + int port; /* Port portion of URI */ + cupsd_printer_t *printer; /* Printer/class */ + cupsd_job_t *job; /* Job */ + int jobid; /* Job ID */ + cupsd_subscription_t *sub; /* Subscription object */ + const char *username, /* requesting-user-name or + authenticated username */ + *recipient, /* notify-recipient-uri */ + *pullmethod; /* notify-pull-method */ + ipp_attribute_t *user_data; /* notify-user-data */ + int interval, /* notify-time-interval */ + lease; /* notify-lease-duration */ + unsigned mask; /* notify-events */ + ipp_attribute_t *notify_events,/* notify-events(-default) */ + *notify_lease; /* notify-lease-duration(-default) */ + + +#ifdef DEBUG + for (attr = con->request->attrs; attr; attr = attr->next) + { + if (attr->group_tag != IPP_TAG_ZERO) + cupsdLogMessage(CUPSD_LOG_DEBUG2, "g%04x v%04x %s", attr->group_tag, + attr->value_tag, attr->name); + else + cupsdLogMessage(CUPSD_LOG_DEBUG2, "----SEP----"); + } +#endif /* DEBUG */ + + /* + * Is the destination valid? + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "cupsdCreateSubscription(con=%p(%d), uri=\"%s\")", + con, con->http.fd, uri->values[0].string.text); + + httpSeparateURI(HTTP_URI_CODING_ALL, uri->values[0].string.text, scheme, + sizeof(scheme), userpass, sizeof(userpass), host, + sizeof(host), &port, resource, sizeof(resource)); + + if (!strcmp(resource, "/")) + { + dtype = (cups_ptype_t)0; + printer = NULL; + } + else if (!strncmp(resource, "/printers", 9) && strlen(resource) <= 10) + { + dtype = (cups_ptype_t)0; + printer = NULL; + } + else if (!strncmp(resource, "/classes", 8) && strlen(resource) <= 9) + { + dtype = CUPS_PRINTER_CLASS; + printer = NULL; + } + else if (!cupsdValidateDest(uri->values[0].string.text, &dtype, &printer)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * Check policy... + */ + + if (printer) + { + if ((status = cupsdCheckPolicy(printer->op_policy_ptr, con, + NULL)) != HTTP_OK) + { + send_http_error(con, status, printer); + return; + } + } + else if ((status = cupsdCheckPolicy(DefaultPolicyPtr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, NULL); + return; + } + + /* + * Get the user that is requesting the subscription... + */ + + username = get_username(con); + + /* + * Find the first subscription group attribute; return if we have + * none... + */ + + for (attr = con->request->attrs; attr; attr = attr->next) + if (attr->group_tag == IPP_TAG_SUBSCRIPTION) + break; + + if (!attr) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("No subscription attributes in request.")); + return; + } + + /* + * Process the subscription attributes in the request... + */ + + con->response->request.status.status_code = IPP_BAD_REQUEST; + + while (attr) + { + recipient = NULL; + pullmethod = NULL; + user_data = NULL; + interval = 0; + lease = DefaultLeaseDuration; + jobid = 0; + mask = CUPSD_EVENT_NONE; + + if (printer) + { + notify_events = ippFindAttribute(printer->attrs, "notify-events-default", + IPP_TAG_KEYWORD); + notify_lease = ippFindAttribute(printer->attrs, + "notify-lease-duration-default", + IPP_TAG_INTEGER); + + if (notify_lease) + lease = notify_lease->values[0].integer; + } + else + { + notify_events = NULL; + notify_lease = NULL; + } + + while (attr && attr->group_tag != IPP_TAG_ZERO) + { + if (!strcmp(attr->name, "notify-recipient-uri") && + attr->value_tag == IPP_TAG_URI) + { + /* + * Validate the recipient scheme against the ServerBin/notifier + * directory... + */ + + char notifier[1024]; /* Notifier filename */ + + + recipient = attr->values[0].string.text; + + if (httpSeparateURI(HTTP_URI_CODING_ALL, recipient, + scheme, sizeof(scheme), userpass, sizeof(userpass), + host, sizeof(host), &port, + resource, sizeof(resource)) < HTTP_URI_OK) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Bad notify-recipient-uri \"%s\"."), recipient); + ippAddInteger(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_ENUM, + "notify-status-code", IPP_URI_SCHEME); + return; + } + + snprintf(notifier, sizeof(notifier), "%s/notifier/%s", ServerBin, + scheme); + if (access(notifier, X_OK)) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("notify-recipient-uri URI \"%s\" uses unknown " + "scheme."), recipient); + ippAddInteger(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_ENUM, + "notify-status-code", IPP_URI_SCHEME); + return; + } + + if (!strcmp(scheme, "rss") && !check_rss_recipient(recipient)) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("notify-recipient-uri URI \"%s\" is already used."), + recipient); + ippAddInteger(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_ENUM, + "notify-status-code", IPP_ATTRIBUTES); + return; + } + } + else if (!strcmp(attr->name, "notify-pull-method") && + attr->value_tag == IPP_TAG_KEYWORD) + { + pullmethod = attr->values[0].string.text; + + if (strcmp(pullmethod, "ippget")) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Bad notify-pull-method \"%s\"."), pullmethod); + ippAddInteger(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_ENUM, + "notify-status-code", IPP_ATTRIBUTES); + return; + } + } + else if (!strcmp(attr->name, "notify-charset") && + attr->value_tag == IPP_TAG_CHARSET && + strcmp(attr->values[0].string.text, "us-ascii") && + strcmp(attr->values[0].string.text, "utf-8")) + { + send_ipp_status(con, IPP_CHARSET, + _("Character set \"%s\" not supported."), + attr->values[0].string.text); + return; + } + else if (!strcmp(attr->name, "notify-natural-language") && + (attr->value_tag != IPP_TAG_LANGUAGE || + strcmp(attr->values[0].string.text, DefaultLanguage))) + { + send_ipp_status(con, IPP_CHARSET, + _("Language \"%s\" not supported."), + attr->values[0].string.text); + return; + } + else if (!strcmp(attr->name, "notify-user-data") && + attr->value_tag == IPP_TAG_STRING) + { + if (attr->num_values > 1 || attr->values[0].unknown.length > 63) + { + send_ipp_status(con, IPP_REQUEST_VALUE, + _("The notify-user-data value is too large " + "(%d > 63 octets)."), + attr->values[0].unknown.length); + return; + } + + user_data = attr; + } + else if (!strcmp(attr->name, "notify-events") && + attr->value_tag == IPP_TAG_KEYWORD) + notify_events = attr; + else if (!strcmp(attr->name, "notify-lease-duration") && + attr->value_tag == IPP_TAG_INTEGER) + lease = attr->values[0].integer; + else if (!strcmp(attr->name, "notify-time-interval") && + attr->value_tag == IPP_TAG_INTEGER) + interval = attr->values[0].integer; + else if (!strcmp(attr->name, "notify-job-id") && + attr->value_tag == IPP_TAG_INTEGER) + jobid = attr->values[0].integer; + + attr = attr->next; + } + + if (notify_events) + { + for (i = 0; i < notify_events->num_values; i ++) + mask |= cupsdEventValue(notify_events->values[i].string.text); + } + + if (recipient) + cupsdLogMessage(CUPSD_LOG_DEBUG, "recipient=\"%s\"", recipient); + if (pullmethod) + cupsdLogMessage(CUPSD_LOG_DEBUG, "pullmethod=\"%s\"", pullmethod); + cupsdLogMessage(CUPSD_LOG_DEBUG, "notify-lease-duration=%d", lease); + cupsdLogMessage(CUPSD_LOG_DEBUG, "notify-time-interval=%d", interval); + + if (!recipient && !pullmethod) + break; + + if (mask == CUPSD_EVENT_NONE) + { + if (jobid) + mask = CUPSD_EVENT_JOB_COMPLETED; + else if (printer) + mask = CUPSD_EVENT_PRINTER_STATE_CHANGED; + else + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("notify-events not specified.")); + return; + } + } + + if (MaxLeaseDuration && (lease == 0 || lease > MaxLeaseDuration)) + { + cupsdLogMessage(CUPSD_LOG_INFO, + "create_subscription: Limiting notify-lease-duration to " + "%d seconds.", + MaxLeaseDuration); + lease = MaxLeaseDuration; + } + + if (jobid) + { + if ((job = cupsdFindJob(jobid)) == NULL) + { + send_ipp_status(con, IPP_NOT_FOUND, _("Job #%d does not exist."), + jobid); + return; + } + } + else + job = NULL; + + if ((sub = cupsdAddSubscription(mask, printer, job, recipient, 0)) == NULL) + { + send_ipp_status(con, IPP_TOO_MANY_SUBSCRIPTIONS, + _("There are too many subscriptions.")); + return; + } + + if (job) + cupsdLogMessage(CUPSD_LOG_DEBUG, "Added subscription #%d for job %d.", + sub->id, job->id); + else if (printer) + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Added subscription #%d for printer \"%s\".", + sub->id, printer->name); + else + cupsdLogMessage(CUPSD_LOG_DEBUG, "Added subscription #%d for server.", + sub->id); + + sub->interval = interval; + sub->lease = lease; + sub->expire = lease ? time(NULL) + lease : 0; + + cupsdSetString(&sub->owner, username); + + if (user_data) + { + sub->user_data_len = user_data->values[0].unknown.length; + memcpy(sub->user_data, user_data->values[0].unknown.data, + sub->user_data_len); + } + + ippAddSeparator(con->response); + ippAddInteger(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_INTEGER, + "notify-subscription-id", sub->id); + + con->response->request.status.status_code = IPP_OK; + + if (attr) + attr = attr->next; + } + + cupsdMarkDirty(CUPSD_DIRTY_SUBSCRIPTIONS); +} + + +/* + * 'delete_printer()' - Remove a printer or class from the system. + */ + +static void +delete_printer(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - URI of printer or class */ +{ + http_status_t status; /* Policy status */ + cups_ptype_t dtype; /* Destination type (printer/class) */ + cupsd_printer_t *printer; /* Printer/class */ + char filename[1024]; /* Script/PPD filename */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "delete_printer(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * Do we have a valid URI? + */ + + if (!cupsdValidateDest(uri->values[0].string.text, &dtype, &printer)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(DefaultPolicyPtr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, NULL); + return; + } + + /* + * Remove old jobs... + */ + + cupsdCancelJobs(printer->name, NULL, 1); + + /* + * Remove old subscriptions and send a "deleted printer" event... + */ + + cupsdAddEvent(CUPSD_EVENT_PRINTER_DELETED, printer, NULL, + "%s \"%s\" deleted by \"%s\".", + (dtype & CUPS_PRINTER_CLASS) ? "Class" : "Printer", + printer->name, get_username(con)); + + cupsdExpireSubscriptions(printer, NULL); + + /* + * Remove any old PPD or script files... + */ + + snprintf(filename, sizeof(filename), "%s/interfaces/%s", ServerRoot, + printer->name); + unlink(filename); + + snprintf(filename, sizeof(filename), "%s/ppd/%s.ppd", ServerRoot, + printer->name); + unlink(filename); + + snprintf(filename, sizeof(filename), "%s/%s.png", CacheDir, printer->name); + unlink(filename); + + snprintf(filename, sizeof(filename), "%s/%s.data", CacheDir, printer->name); + unlink(filename); + +#ifdef __APPLE__ + /* + * Unregister color profiles... + */ + + apple_unregister_profiles(printer); +#endif /* __APPLE__ */ + + if (dtype & CUPS_PRINTER_CLASS) + { + cupsdLogMessage(CUPSD_LOG_INFO, "Class \"%s\" deleted by \"%s\".", + printer->name, get_username(con)); + + cupsdDeletePrinter(printer, 0); + cupsdMarkDirty(CUPSD_DIRTY_CLASSES); + } + else + { + cupsdLogMessage(CUPSD_LOG_INFO, "Printer \"%s\" deleted by \"%s\".", + printer->name, get_username(con)); + + if (cupsdDeletePrinter(printer, 0)) + cupsdMarkDirty(CUPSD_DIRTY_CLASSES); + + cupsdMarkDirty(CUPSD_DIRTY_PRINTERS); + } + + cupsdMarkDirty(CUPSD_DIRTY_PRINTCAP); + + /* + * Return with no errors... + */ + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'get_default()' - Get the default destination. + */ + +static void +get_default(cupsd_client_t *con) /* I - Client connection */ +{ + http_status_t status; /* Policy status */ + cups_array_t *ra; /* Requested attributes array */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "get_default(%p[%d])", con, con->http.fd); + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(DefaultPolicyPtr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, NULL); + return; + } + + if (DefaultPrinter) + { + ra = create_requested_array(con->request); + + copy_printer_attrs(con, DefaultPrinter, ra); + + cupsArrayDelete(ra); + + con->response->request.status.status_code = IPP_OK; + } + else + send_ipp_status(con, IPP_NOT_FOUND, _("No default printer.")); +} + + +/* + * 'get_devices()' - Get the list of available devices on the local system. + */ + +static void +get_devices(cupsd_client_t *con) /* I - Client connection */ +{ + http_status_t status; /* Policy status */ + ipp_attribute_t *limit, /* limit attribute */ + *timeout, /* timeout attribute */ + *requested, /* requested-attributes attribute */ + *exclude, /* exclude-schemes attribute */ + *include; /* include-schemes attribute */ + char command[1024], /* cups-deviced command */ + options[2048], /* Options to pass to command */ + requested_str[256], + /* String for requested attributes */ + exclude_str[512], + /* String for excluded schemes */ + include_str[512]; + /* String for included schemes */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "get_devices(%p[%d])", con, con->http.fd); + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(DefaultPolicyPtr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, NULL); + return; + } + + /* + * Run cups-deviced command with the given options... + */ + + limit = ippFindAttribute(con->request, "limit", IPP_TAG_INTEGER); + timeout = ippFindAttribute(con->request, "timeout", IPP_TAG_INTEGER); + requested = ippFindAttribute(con->request, "requested-attributes", + IPP_TAG_KEYWORD); + exclude = ippFindAttribute(con->request, "exclude-schemes", IPP_TAG_NAME); + include = ippFindAttribute(con->request, "include-schemes", IPP_TAG_NAME); + + if (requested) + url_encode_attr(requested, requested_str, sizeof(requested_str)); + else + strlcpy(requested_str, "requested-attributes=all", sizeof(requested_str)); + + if (exclude) + url_encode_attr(exclude, exclude_str, sizeof(exclude_str)); + else + exclude_str[0] = '\0'; + + if (include) + url_encode_attr(include, include_str, sizeof(include_str)); + else + include_str[0] = '\0'; + + snprintf(command, sizeof(command), "%s/daemon/cups-deviced", ServerBin); + snprintf(options, sizeof(options), + "%d+%d+%d+%d+%s%s%s%s%s", + con->request->request.op.request_id, + limit ? limit->values[0].integer : 0, + timeout ? timeout->values[0].integer : 15, + (int)User, + requested_str, + exclude_str[0] ? "%20" : "", exclude_str, + include_str[0] ? "%20" : "", include_str); + + if (cupsdSendCommand(con, command, options, 1)) + { + /* + * Command started successfully, don't send an IPP response here... + */ + + ippDelete(con->response); + con->response = NULL; + } + else + { + /* + * Command failed, return "internal error" so the user knows something + * went wrong... + */ + + send_ipp_status(con, IPP_INTERNAL_ERROR, + _("cups-deviced failed to execute.")); + } +} + + +/* + * 'get_document()' - Get a copy of a job file. + */ + +static void +get_document(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Job URI */ +{ + http_status_t status; /* Policy status */ + ipp_attribute_t *attr; /* Current attribute */ + int jobid; /* Job ID */ + int docnum; /* Document number */ + cupsd_job_t *job; /* Current job */ + char scheme[HTTP_MAX_URI], /* Method portion of URI */ + username[HTTP_MAX_URI], /* Username portion of URI */ + host[HTTP_MAX_URI], /* Host portion of URI */ + resource[HTTP_MAX_URI]; /* Resource portion of URI */ + int port; /* Port portion of URI */ + char filename[1024], /* Filename for document */ + format[1024]; /* Format for document */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "get_document(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * See if we have a job URI or a printer URI... + */ + + if (!strcmp(uri->name, "printer-uri")) + { + /* + * Got a printer URI; see if we also have a job-id attribute... + */ + + if ((attr = ippFindAttribute(con->request, "job-id", + IPP_TAG_INTEGER)) == NULL) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Got a printer-uri attribute but no job-id.")); + return; + } + + jobid = attr->values[0].integer; + } + else + { + /* + * Got a job URI; parse it to get the job ID... + */ + + httpSeparateURI(HTTP_URI_CODING_ALL, uri->values[0].string.text, scheme, + sizeof(scheme), username, sizeof(username), host, + sizeof(host), &port, resource, sizeof(resource)); + + if (strncmp(resource, "/jobs/", 6)) + { + /* + * Not a valid URI! + */ + + send_ipp_status(con, IPP_BAD_REQUEST, _("Bad job-uri \"%s\"."), + uri->values[0].string.text); + return; + } + + jobid = atoi(resource + 6); + } + + /* + * See if the job exists... + */ + + if ((job = cupsdFindJob(jobid)) == NULL) + { + /* + * Nope - return a "not found" error... + */ + + send_ipp_status(con, IPP_NOT_FOUND, _("Job #%d does not exist."), jobid); + return; + } + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(DefaultPolicyPtr, con, + job->username)) != HTTP_OK) + { + send_http_error(con, status, NULL); + return; + } + + /* + * Get the document number... + */ + + if ((attr = ippFindAttribute(con->request, "document-number", + IPP_TAG_INTEGER)) == NULL) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Missing document-number attribute.")); + return; + } + + if ((docnum = attr->values[0].integer) < 1 || docnum > job->num_files || + attr->num_values > 1) + { + send_ipp_status(con, IPP_NOT_FOUND, + _("Document #%d does not exist in job #%d."), docnum, + jobid); + return; + } + + snprintf(filename, sizeof(filename), "%s/d%05d-%03d", RequestRoot, jobid, + docnum); + if ((con->file = open(filename, O_RDONLY)) == -1) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to open document %d in job %d - %s", docnum, jobid, + strerror(errno)); + send_ipp_status(con, IPP_NOT_FOUND, + _("Unable to open document #%d in job #%d."), docnum, + jobid); + return; + } + + fcntl(con->file, F_SETFD, fcntl(con->file, F_GETFD) | FD_CLOEXEC); + + cupsdLoadJob(job); + + snprintf(format, sizeof(format), "%s/%s", job->filetypes[docnum - 1]->super, + job->filetypes[docnum - 1]->type); + + ippAddString(con->response, IPP_TAG_JOB, IPP_TAG_MIMETYPE, "document-format", + NULL, format); + ippAddInteger(con->response, IPP_TAG_JOB, IPP_TAG_INTEGER, "document-number", + docnum); + if ((attr = ippFindAttribute(job->attrs, "document-name", + IPP_TAG_NAME)) != NULL) + ippAddString(con->response, IPP_TAG_JOB, IPP_TAG_NAME, "document-name", + NULL, attr->values[0].string.text); +} + + +/* + * 'get_job_attrs()' - Get job attributes. + */ + +static void +get_job_attrs(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Job URI */ +{ + http_status_t status; /* Policy status */ + ipp_attribute_t *attr; /* Current attribute */ + int jobid; /* Job ID */ + cupsd_job_t *job; /* Current job */ + cupsd_printer_t *printer; /* Current printer */ + cupsd_policy_t *policy; /* Current security policy */ + char scheme[HTTP_MAX_URI], /* Scheme portion of URI */ + username[HTTP_MAX_URI], /* Username portion of URI */ + host[HTTP_MAX_URI], /* Host portion of URI */ + resource[HTTP_MAX_URI]; /* Resource portion of URI */ + int port; /* Port portion of URI */ + cups_array_t *ra, /* Requested attributes array */ + *exclude; /* Private attributes array */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "get_job_attrs(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * See if we have a job URI or a printer URI... + */ + + if (!strcmp(uri->name, "printer-uri")) + { + /* + * Got a printer URI; see if we also have a job-id attribute... + */ + + if ((attr = ippFindAttribute(con->request, "job-id", + IPP_TAG_INTEGER)) == NULL) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Got a printer-uri attribute but no job-id.")); + return; + } + + jobid = attr->values[0].integer; + } + else + { + /* + * Got a job URI; parse it to get the job ID... + */ + + httpSeparateURI(HTTP_URI_CODING_ALL, uri->values[0].string.text, scheme, + sizeof(scheme), username, sizeof(username), host, + sizeof(host), &port, resource, sizeof(resource)); + + if (strncmp(resource, "/jobs/", 6)) + { + /* + * Not a valid URI! + */ + + send_ipp_status(con, IPP_BAD_REQUEST, _("Bad job-uri \"%s\"."), + uri->values[0].string.text); + return; + } + + jobid = atoi(resource + 6); + } + + /* + * See if the job exists... + */ + + if ((job = cupsdFindJob(jobid)) == NULL) + { + /* + * Nope - return a "not found" error... + */ + + send_ipp_status(con, IPP_NOT_FOUND, _("Job #%d does not exist."), jobid); + return; + } + + /* + * Check policy... + */ + + if ((printer = job->printer) == NULL) + printer = cupsdFindDest(job->dest); + + if (printer) + policy = printer->op_policy_ptr; + else + policy = DefaultPolicyPtr; + + if ((status = cupsdCheckPolicy(policy, con, job->username)) != HTTP_OK) + { + send_http_error(con, status, NULL); + return; + } + + exclude = cupsdGetPrivateAttrs(policy, con, printer, job->username); + + /* + * Copy attributes... + */ + + cupsdLoadJob(job); + + ra = create_requested_array(con->request); + copy_job_attrs(con, job, ra, exclude); + cupsArrayDelete(ra); + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'get_jobs()' - Get a list of jobs for the specified printer. + */ + +static void +get_jobs(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Printer URI */ +{ + http_status_t status; /* Policy status */ + ipp_attribute_t *attr; /* Current attribute */ + const char *dest; /* Destination */ + cups_ptype_t dtype; /* Destination type (printer/class) */ + cups_ptype_t dmask; /* Destination type mask */ + char scheme[HTTP_MAX_URI], /* Scheme portion of URI */ + username[HTTP_MAX_URI], /* Username portion of URI */ + host[HTTP_MAX_URI], /* Host portion of URI */ + resource[HTTP_MAX_URI]; /* Resource portion of URI */ + int port; /* Port portion of URI */ + int job_comparison; /* Job comparison */ + ipp_jstate_t job_state; /* job-state value */ + int first_job_id; /* First job ID */ + int limit; /* Maximum number of jobs to return */ + int count; /* Number of jobs that match */ + ipp_attribute_t *job_ids; /* job-ids attribute */ + cupsd_job_t *job; /* Current job pointer */ + cupsd_printer_t *printer; /* Printer */ + cups_array_t *list; /* Which job list... */ + cups_array_t *ra, /* Requested attributes array */ + *exclude; /* Private attributes array */ + cupsd_policy_t *policy; /* Current policy */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "get_jobs(%p[%d], %s)", con, con->http.fd, + uri->values[0].string.text); + + /* + * Is the destination valid? + */ + + if (strcmp(uri->name, "printer-uri")) + { + send_ipp_status(con, IPP_BAD_REQUEST, _("No printer-uri in request.")); + return; + } + + httpSeparateURI(HTTP_URI_CODING_ALL, uri->values[0].string.text, scheme, + sizeof(scheme), username, sizeof(username), host, + sizeof(host), &port, resource, sizeof(resource)); + + if (!strcmp(resource, "/") || !strcmp(resource, "/jobs")) + { + dest = NULL; + dtype = (cups_ptype_t)0; + dmask = (cups_ptype_t)0; + printer = NULL; + } + else if (!strncmp(resource, "/printers", 9) && strlen(resource) <= 10) + { + dest = NULL; + dtype = (cups_ptype_t)0; + dmask = CUPS_PRINTER_CLASS; + printer = NULL; + } + else if (!strncmp(resource, "/classes", 8) && strlen(resource) <= 9) + { + dest = NULL; + dtype = CUPS_PRINTER_CLASS; + dmask = CUPS_PRINTER_CLASS; + printer = NULL; + } + else if ((dest = cupsdValidateDest(uri->values[0].string.text, &dtype, + &printer)) == NULL) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + else + { + dtype &= CUPS_PRINTER_CLASS; + dmask = CUPS_PRINTER_CLASS; + } + + /* + * Check policy... + */ + + if (printer) + policy = printer->op_policy_ptr; + else + policy = DefaultPolicyPtr; + + if ((status = cupsdCheckPolicy(policy, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, NULL); + return; + } + + job_ids = ippFindAttribute(con->request, "job-ids", IPP_TAG_INTEGER); + + /* + * See if the "which-jobs" attribute have been specified... + */ + + if ((attr = ippFindAttribute(con->request, "which-jobs", + IPP_TAG_KEYWORD)) != NULL && job_ids) + { + send_ipp_status(con, IPP_CONFLICT, + _("The %s attribute cannot be provided with job-ids."), + "which-jobs"); + return; + } + else if (!attr || !strcmp(attr->values[0].string.text, "not-completed")) + { + job_comparison = -1; + job_state = IPP_JOB_STOPPED; + list = Jobs; + } + else if (!strcmp(attr->values[0].string.text, "completed")) + { + job_comparison = 1; + job_state = IPP_JOB_CANCELED; + list = Jobs; + } + else if (!strcmp(attr->values[0].string.text, "aborted")) + { + job_comparison = 0; + job_state = IPP_JOB_ABORTED; + list = Jobs; + } + else if (!strcmp(attr->values[0].string.text, "all")) + { + job_comparison = 1; + job_state = IPP_JOB_PENDING; + list = Jobs; + } + else if (!strcmp(attr->values[0].string.text, "canceled")) + { + job_comparison = 0; + job_state = IPP_JOB_CANCELED; + list = Jobs; + } + else if (!strcmp(attr->values[0].string.text, "pending")) + { + job_comparison = 0; + job_state = IPP_JOB_PENDING; + list = ActiveJobs; + } + else if (!strcmp(attr->values[0].string.text, "pending-held")) + { + job_comparison = 0; + job_state = IPP_JOB_HELD; + list = ActiveJobs; + } + else if (!strcmp(attr->values[0].string.text, "processing")) + { + job_comparison = 0; + job_state = IPP_JOB_PROCESSING; + list = PrintingJobs; + } + else if (!strcmp(attr->values[0].string.text, "processing-stopped")) + { + job_comparison = 0; + job_state = IPP_JOB_STOPPED; + list = ActiveJobs; + } + else + { + send_ipp_status(con, IPP_ATTRIBUTES, + _("The which-jobs value \"%s\" is not supported."), + attr->values[0].string.text); + ippAddString(con->response, IPP_TAG_UNSUPPORTED_GROUP, IPP_TAG_KEYWORD, + "which-jobs", NULL, attr->values[0].string.text); + return; + } + + /* + * See if they want to limit the number of jobs reported... + */ + + if ((attr = ippFindAttribute(con->request, "limit", + IPP_TAG_INTEGER)) != NULL) + { + if (job_ids) + { + send_ipp_status(con, IPP_CONFLICT, + _("The %s attribute cannot be provided with job-ids."), + "limit"); + return; + } + + limit = attr->values[0].integer; + } + else + limit = 0; + + if ((attr = ippFindAttribute(con->request, "first-job-id", + IPP_TAG_INTEGER)) != NULL) + { + if (job_ids) + { + send_ipp_status(con, IPP_CONFLICT, + _("The %s attribute cannot be provided with job-ids."), + "first-job-id"); + return; + } + + first_job_id = attr->values[0].integer; + } + else + first_job_id = 1; + + /* + * See if we only want to see jobs for a specific user... + */ + + if ((attr = ippFindAttribute(con->request, "my-jobs", + IPP_TAG_BOOLEAN)) != NULL && job_ids) + { + send_ipp_status(con, IPP_CONFLICT, + _("The %s attribute cannot be provided with job-ids."), + "my-jobs"); + return; + } + else if (attr && attr->values[0].boolean) + strlcpy(username, get_username(con), sizeof(username)); + else + username[0] = '\0'; + + if ((ra = create_requested_array(con->request)) == NULL && + !ippFindAttribute(con->request, "requested-attributes", IPP_TAG_KEYWORD)) + { + /* + * IPP conformance - Get-Jobs has a default requested-attributes value of + * "job-id" and "job-uri". + */ + + ra = cupsArrayNew((cups_array_func_t)strcmp, NULL); + cupsArrayAdd(ra, "job-id"); + cupsArrayAdd(ra, "job-uri"); + } + + /* + * OK, build a list of jobs for this printer... + */ + + if (job_ids) + { + int i; /* Looping var */ + + for (i = 0; i < job_ids->num_values; i ++) + { + if (!cupsdFindJob(job_ids->values[i].integer)) + break; + } + + if (i < job_ids->num_values) + { + send_ipp_status(con, IPP_NOT_FOUND, _("Job #%d does not exist."), + job_ids->values[i].integer); + return; + } + + for (i = 0; i < job_ids->num_values; i ++) + { + job = cupsdFindJob(job_ids->values[i].integer); + + cupsdLoadJob(job); + + if (!job->attrs) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, "get_jobs: No attributes for job %d", + job->id); + continue; + } + + if (i > 0) + ippAddSeparator(con->response); + + exclude = cupsdGetPrivateAttrs(job->printer ? + job->printer->op_policy_ptr : + policy, con, job->printer, + job->username); + + copy_job_attrs(con, job, ra, exclude); + } + } + else + { + for (count = 0, job = (cupsd_job_t *)cupsArrayFirst(list); + (limit <= 0 || count < limit) && job; + job = (cupsd_job_t *)cupsArrayNext(list)) + { + /* + * Filter out jobs that don't match... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "get_jobs: job->id=%d, dest=\"%s\", username=\"%s\", " + "state_value=%d, attrs=%p", job->id, job->dest, + job->username, job->state_value, job->attrs); + + if (!job->dest || !job->username) + cupsdLoadJob(job); + + if (!job->dest || !job->username) + continue; + + if ((dest && strcmp(job->dest, dest)) && + (!job->printer || !dest || strcmp(job->printer->name, dest))) + continue; + if ((job->dtype & dmask) != dtype && + (!job->printer || (job->printer->type & dmask) != dtype)) + continue; + + if ((job_comparison < 0 && job->state_value > job_state) || + (job_comparison == 0 && job->state_value != job_state) || + (job_comparison > 0 && job->state_value < job_state)) + continue; + + if (job->id < first_job_id) + continue; + + cupsdLoadJob(job); + + if (!job->attrs) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, "get_jobs: No attributes for job %d", + job->id); + continue; + } + + if (username[0] && _cups_strcasecmp(username, job->username)) + continue; + + if (count > 0) + ippAddSeparator(con->response); + + count ++; + + exclude = cupsdGetPrivateAttrs(job->printer ? + job->printer->op_policy_ptr : + policy, con, job->printer, + job->username); + + copy_job_attrs(con, job, ra, exclude); + } + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "get_jobs: count=%d", count); + } + + cupsArrayDelete(ra); + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'get_notifications()' - Get events for a subscription. + */ + +static void +get_notifications(cupsd_client_t *con) /* I - Client connection */ +{ + int i, j; /* Looping vars */ + http_status_t status; /* Policy status */ + cupsd_subscription_t *sub; /* Subscription */ + ipp_attribute_t *ids, /* notify-subscription-ids */ + *sequences; /* notify-sequence-numbers */ + int min_seq; /* Minimum sequence number */ + int interval; /* Poll interval */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "get_notifications(con=%p[%d])", + con, con->http.fd); + + /* + * Get subscription attributes... + */ + + ids = ippFindAttribute(con->request, "notify-subscription-ids", + IPP_TAG_INTEGER); + sequences = ippFindAttribute(con->request, "notify-sequence-numbers", + IPP_TAG_INTEGER); + + if (!ids) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Missing notify-subscription-ids attribute.")); + return; + } + + /* + * Are the subscription IDs valid? + */ + + for (i = 0, interval = 60; i < ids->num_values; i ++) + { + if ((sub = cupsdFindSubscription(ids->values[i].integer)) == NULL) + { + /* + * Bad subscription ID... + */ + + send_ipp_status(con, IPP_NOT_FOUND, _("Subscription #%d does not exist."), + ids->values[i].integer); + return; + } + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(sub->dest ? sub->dest->op_policy_ptr : + DefaultPolicyPtr, + con, sub->owner)) != HTTP_OK) + { + send_http_error(con, status, sub->dest); + return; + } + + /* + * Check the subscription type and update the interval accordingly. + */ + + if (sub->job && sub->job->state_value == IPP_JOB_PROCESSING && + interval > 10) + interval = 10; + else if (sub->job && sub->job->state_value >= IPP_JOB_STOPPED) + interval = 0; + else if (sub->dest && sub->dest->state == IPP_PRINTER_PROCESSING && + interval > 30) + interval = 30; + } + + /* + * Tell the client to poll again in N seconds... + */ + + if (interval > 0) + ippAddInteger(con->response, IPP_TAG_OPERATION, IPP_TAG_INTEGER, + "notify-get-interval", interval); + + ippAddInteger(con->response, IPP_TAG_OPERATION, IPP_TAG_INTEGER, + "printer-up-time", time(NULL)); + + /* + * Copy the subscription event attributes to the response. + */ + + con->response->request.status.status_code = + interval ? IPP_OK : IPP_OK_EVENTS_COMPLETE; + + for (i = 0; i < ids->num_values; i ++) + { + /* + * Get the subscription and sequence number... + */ + + sub = cupsdFindSubscription(ids->values[i].integer); + + if (sequences && i < sequences->num_values) + min_seq = sequences->values[i].integer; + else + min_seq = 1; + + /* + * If we don't have any new events, nothing to do here... + */ + + if (min_seq > (sub->first_event_id + cupsArrayCount(sub->events))) + continue; + + /* + * Otherwise copy all of the new events... + */ + + if (sub->first_event_id > min_seq) + j = 0; + else + j = min_seq - sub->first_event_id; + + for (; j < cupsArrayCount(sub->events); j ++) + { + ippAddSeparator(con->response); + + copy_attrs(con->response, + ((cupsd_event_t *)cupsArrayIndex(sub->events, j))->attrs, NULL, + IPP_TAG_EVENT_NOTIFICATION, 0, NULL); + } + } +} + + +/* + * 'get_ppd()' - Get a named PPD from the local system. + */ + +static void +get_ppd(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Printer URI or PPD name */ +{ + http_status_t status; /* Policy status */ + cupsd_printer_t *dest; /* Destination */ + cups_ptype_t dtype; /* Destination type */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "get_ppd(%p[%d], %p[%s=%s])", con, + con->http.fd, uri, uri->name, uri->values[0].string.text); + + if (!strcmp(uri->name, "ppd-name")) + { + /* + * Return a PPD file from cups-driverd... + */ + + char command[1024], /* cups-driverd command */ + options[1024], /* Options to pass to command */ + ppd_name[1024]; /* ppd-name */ + + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(DefaultPolicyPtr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, NULL); + return; + } + + /* + * Run cups-driverd command with the given options... + */ + + snprintf(command, sizeof(command), "%s/daemon/cups-driverd", ServerBin); + url_encode_string(uri->values[0].string.text, ppd_name, sizeof(ppd_name)); + snprintf(options, sizeof(options), "get+%d+%s", + con->request->request.op.request_id, ppd_name); + + if (cupsdSendCommand(con, command, options, 0)) + { + /* + * Command started successfully, don't send an IPP response here... + */ + + ippDelete(con->response); + con->response = NULL; + } + else + { + /* + * Command failed, return "internal error" so the user knows something + * went wrong... + */ + + send_ipp_status(con, IPP_INTERNAL_ERROR, + _("cups-driverd failed to execute.")); + } + } + else if (!strcmp(uri->name, "printer-uri") && + cupsdValidateDest(uri->values[0].string.text, &dtype, &dest)) + { + int i; /* Looping var */ + char filename[1024]; /* PPD filename */ + + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(dest->op_policy_ptr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, dest); + return; + } + + /* + * See if we need the PPD for a class or remote printer... + */ + + snprintf(filename, sizeof(filename), "%s/ppd/%s.ppd", ServerRoot, + dest->name); + + if ((dtype & CUPS_PRINTER_REMOTE) && access(filename, 0)) + { + con->response->request.status.status_code = CUPS_SEE_OTHER; + ippAddString(con->response, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, dest->uri); + return; + } + else if (dtype & CUPS_PRINTER_CLASS) + { + for (i = 0; i < dest->num_printers; i ++) + if (!(dest->printers[i]->type & CUPS_PRINTER_CLASS)) + { + snprintf(filename, sizeof(filename), "%s/ppd/%s.ppd", ServerRoot, + dest->printers[i]->name); + + if (!access(filename, 0)) + break; + } + + if (i < dest->num_printers) + dest = dest->printers[i]; + else + { + con->response->request.status.status_code = CUPS_SEE_OTHER; + ippAddString(con->response, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, dest->printers[0]->uri); + return; + } + } + + /* + * Found the printer with the PPD file, now see if there is one... + */ + + if ((con->file = open(filename, O_RDONLY)) < 0) + { + send_ipp_status(con, IPP_NOT_FOUND, + _("The PPD file \"%s\" could not be opened: %s"), + uri->values[0].string.text, strerror(errno)); + return; + } + + fcntl(con->file, F_SETFD, fcntl(con->file, F_GETFD) | FD_CLOEXEC); + + con->pipe_pid = 0; + + con->response->request.status.status_code = IPP_OK; + } + else + send_ipp_status(con, IPP_NOT_FOUND, + _("The PPD file \"%s\" could not be found."), + uri->values[0].string.text); +} + + +/* + * 'get_ppds()' - Get the list of PPD files on the local system. + */ + +static void +get_ppds(cupsd_client_t *con) /* I - Client connection */ +{ + http_status_t status; /* Policy status */ + ipp_attribute_t *limit, /* Limit attribute */ + *device, /* ppd-device-id attribute */ + *language, /* ppd-natural-language attribute */ + *make, /* ppd-make attribute */ + *model, /* ppd-make-and-model attribute */ + *model_number, /* ppd-model-number attribute */ + *product, /* ppd-product attribute */ + *psversion, /* ppd-psverion attribute */ + *type, /* ppd-type attribute */ + *requested, /* requested-attributes attribute */ + *exclude, /* exclude-schemes attribute */ + *include; /* include-schemes attribute */ + char command[1024], /* cups-driverd command */ + options[4096], /* Options to pass to command */ + device_str[256],/* Escaped ppd-device-id string */ + language_str[256], + /* Escaped ppd-natural-language */ + make_str[256], /* Escaped ppd-make string */ + model_str[256], /* Escaped ppd-make-and-model string */ + model_number_str[256], + /* ppd-model-number string */ + product_str[256], + /* Escaped ppd-product string */ + psversion_str[256], + /* Escaped ppd-psversion string */ + type_str[256], /* Escaped ppd-type string */ + requested_str[256], + /* String for requested attributes */ + exclude_str[512], + /* String for excluded schemes */ + include_str[512]; + /* String for included schemes */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "get_ppds(%p[%d])", con, con->http.fd); + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(DefaultPolicyPtr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, NULL); + return; + } + + /* + * Run cups-driverd command with the given options... + */ + + limit = ippFindAttribute(con->request, "limit", IPP_TAG_INTEGER); + device = ippFindAttribute(con->request, "ppd-device-id", IPP_TAG_TEXT); + language = ippFindAttribute(con->request, "ppd-natural-language", + IPP_TAG_LANGUAGE); + make = ippFindAttribute(con->request, "ppd-make", IPP_TAG_TEXT); + model = ippFindAttribute(con->request, "ppd-make-and-model", + IPP_TAG_TEXT); + model_number = ippFindAttribute(con->request, "ppd-model-number", + IPP_TAG_INTEGER); + product = ippFindAttribute(con->request, "ppd-product", IPP_TAG_TEXT); + psversion = ippFindAttribute(con->request, "ppd-psversion", IPP_TAG_TEXT); + type = ippFindAttribute(con->request, "ppd-type", IPP_TAG_KEYWORD); + requested = ippFindAttribute(con->request, "requested-attributes", + IPP_TAG_KEYWORD); + exclude = ippFindAttribute(con->request, "exclude-schemes", + IPP_TAG_NAME); + include = ippFindAttribute(con->request, "include-schemes", + IPP_TAG_NAME); + + if (requested) + url_encode_attr(requested, requested_str, sizeof(requested_str)); + else + strlcpy(requested_str, "requested-attributes=all", sizeof(requested_str)); + + if (device) + url_encode_attr(device, device_str, sizeof(device_str)); + else + device_str[0] = '\0'; + + if (language) + url_encode_attr(language, language_str, sizeof(language_str)); + else + language_str[0] = '\0'; + + if (make) + url_encode_attr(make, make_str, sizeof(make_str)); + else + make_str[0] = '\0'; + + if (model) + url_encode_attr(model, model_str, sizeof(model_str)); + else + model_str[0] = '\0'; + + if (model_number) + snprintf(model_number_str, sizeof(model_number_str), "ppd-model-number=%d", + model_number->values[0].integer); + else + model_number_str[0] = '\0'; + + if (product) + url_encode_attr(product, product_str, sizeof(product_str)); + else + product_str[0] = '\0'; + + if (psversion) + url_encode_attr(psversion, psversion_str, sizeof(psversion_str)); + else + psversion_str[0] = '\0'; + + if (type) + url_encode_attr(type, type_str, sizeof(type_str)); + else + type_str[0] = '\0'; + + if (exclude) + url_encode_attr(exclude, exclude_str, sizeof(exclude_str)); + else + exclude_str[0] = '\0'; + + if (include) + url_encode_attr(include, include_str, sizeof(include_str)); + else + include_str[0] = '\0'; + + snprintf(command, sizeof(command), "%s/daemon/cups-driverd", ServerBin); + snprintf(options, sizeof(options), + "list+%d+%d+%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s", + con->request->request.op.request_id, + limit ? limit->values[0].integer : 0, + requested_str, + device ? "%20" : "", device_str, + language ? "%20" : "", language_str, + make ? "%20" : "", make_str, + model ? "%20" : "", model_str, + model_number ? "%20" : "", model_number_str, + product ? "%20" : "", product_str, + psversion ? "%20" : "", psversion_str, + type ? "%20" : "", type_str, + exclude_str[0] ? "%20" : "", exclude_str, + include_str[0] ? "%20" : "", include_str); + + if (cupsdSendCommand(con, command, options, 0)) + { + /* + * Command started successfully, don't send an IPP response here... + */ + + ippDelete(con->response); + con->response = NULL; + } + else + { + /* + * Command failed, return "internal error" so the user knows something + * went wrong... + */ + + send_ipp_status(con, IPP_INTERNAL_ERROR, + _("cups-driverd failed to execute.")); + } +} + + +/* + * 'get_printer_attrs()' - Get printer attributes. + */ + +static void +get_printer_attrs(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Printer URI */ +{ + http_status_t status; /* Policy status */ + cups_ptype_t dtype; /* Destination type (printer/class) */ + cupsd_printer_t *printer; /* Printer/class */ + cups_array_t *ra; /* Requested attributes array */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "get_printer_attrs(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * Is the destination valid? + */ + + if (!cupsdValidateDest(uri->values[0].string.text, &dtype, &printer)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(printer->op_policy_ptr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, printer); + return; + } + + /* + * Send the attributes... + */ + + ra = create_requested_array(con->request); + + copy_printer_attrs(con, printer, ra); + + cupsArrayDelete(ra); + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'get_printer_supported()' - Get printer supported values. + */ + +static void +get_printer_supported( + cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Printer URI */ +{ + http_status_t status; /* Policy status */ + cups_ptype_t dtype; /* Destination type (printer/class) */ + cupsd_printer_t *printer; /* Printer/class */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "get_printer_supported(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * Is the destination valid? + */ + + if (!cupsdValidateDest(uri->values[0].string.text, &dtype, &printer)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(printer->op_policy_ptr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, printer); + return; + } + + /* + * Return a list of attributes that can be set via Set-Printer-Attributes. + */ + + ippAddInteger(con->response, IPP_TAG_PRINTER, IPP_TAG_ADMINDEFINE, + "printer-info", 0); + ippAddInteger(con->response, IPP_TAG_PRINTER, IPP_TAG_ADMINDEFINE, + "printer-location", 0); + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'get_printers()' - Get a list of printers or classes. + */ + +static void +get_printers(cupsd_client_t *con, /* I - Client connection */ + int type) /* I - 0 or CUPS_PRINTER_CLASS */ +{ + http_status_t status; /* Policy status */ + ipp_attribute_t *attr; /* Current attribute */ + int limit; /* Max number of printers to return */ + int count; /* Number of printers that match */ + cupsd_printer_t *printer; /* Current printer pointer */ + int printer_type, /* printer-type attribute */ + printer_mask; /* printer-type-mask attribute */ + char *location; /* Location string */ + const char *username; /* Current user */ + char *first_printer_name; /* first-printer-name attribute */ + cups_array_t *ra; /* Requested attributes array */ + int local; /* Local connection? */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "get_printers(%p[%d], %x)", con, + con->http.fd, type); + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(DefaultPolicyPtr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, NULL); + return; + } + + /* + * Check for printers... + */ + + if (!Printers || !cupsArrayCount(Printers)) + { + send_ipp_status(con, IPP_NOT_FOUND, _("No destinations added.")); + return; + } + + /* + * See if they want to limit the number of printers reported... + */ + + if ((attr = ippFindAttribute(con->request, "limit", + IPP_TAG_INTEGER)) != NULL) + limit = attr->values[0].integer; + else + limit = 10000000; + + if ((attr = ippFindAttribute(con->request, "first-printer-name", + IPP_TAG_NAME)) != NULL) + first_printer_name = attr->values[0].string.text; + else + first_printer_name = NULL; + + /* + * Support filtering... + */ + + if ((attr = ippFindAttribute(con->request, "printer-type", + IPP_TAG_ENUM)) != NULL) + printer_type = attr->values[0].integer; + else + printer_type = 0; + + if ((attr = ippFindAttribute(con->request, "printer-type-mask", + IPP_TAG_ENUM)) != NULL) + printer_mask = attr->values[0].integer; + else + printer_mask = 0; + + local = httpAddrLocalhost(&(con->clientaddr)); + + if ((attr = ippFindAttribute(con->request, "printer-location", + IPP_TAG_TEXT)) != NULL) + location = attr->values[0].string.text; + else + location = NULL; + + if (con->username[0]) + username = con->username; + else if ((attr = ippFindAttribute(con->request, "requesting-user-name", + IPP_TAG_NAME)) != NULL) + username = attr->values[0].string.text; + else + username = NULL; + + ra = create_requested_array(con->request); + + /* + * OK, build a list of printers for this printer... + */ + + if (first_printer_name) + { + if ((printer = cupsdFindDest(first_printer_name)) == NULL) + printer = (cupsd_printer_t *)cupsArrayFirst(Printers); + } + else + printer = (cupsd_printer_t *)cupsArrayFirst(Printers); + + for (count = 0; + count < limit && printer; + printer = (cupsd_printer_t *)cupsArrayNext(Printers)) + { + if (!local && !printer->shared) + continue; + + if ((!type || (printer->type & CUPS_PRINTER_CLASS) == type) && + (printer->type & printer_mask) == printer_type && + (!location || + (printer->location && !_cups_strcasecmp(printer->location, location)))) + { + /* + * If a username is specified, see if it is allowed or denied + * access... + */ + + if (cupsArrayCount(printer->users) && username && + !user_allowed(printer, username)) + continue; + + /* + * Add the group separator as needed... + */ + + if (count > 0) + ippAddSeparator(con->response); + + count ++; + + /* + * Send the attributes... + */ + + copy_printer_attrs(con, printer, ra); + } + } + + cupsArrayDelete(ra); + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'get_subscription_attrs()' - Get subscription attributes. + */ + +static void +get_subscription_attrs( + cupsd_client_t *con, /* I - Client connection */ + int sub_id) /* I - Subscription ID */ +{ + http_status_t status; /* Policy status */ + cupsd_subscription_t *sub; /* Subscription */ + cupsd_policy_t *policy; /* Current security policy */ + cups_array_t *ra, /* Requested attributes array */ + *exclude; /* Private attributes array */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "get_subscription_attrs(con=%p[%d], sub_id=%d)", + con, con->http.fd, sub_id); + + /* + * Is the subscription ID valid? + */ + + if ((sub = cupsdFindSubscription(sub_id)) == NULL) + { + /* + * Bad subscription ID... + */ + + send_ipp_status(con, IPP_NOT_FOUND, _("Subscription #%d does not exist."), + sub_id); + return; + } + + /* + * Check policy... + */ + + if (sub->dest) + policy = sub->dest->op_policy_ptr; + else + policy = DefaultPolicyPtr; + + if ((status = cupsdCheckPolicy(policy, con, sub->owner)) != HTTP_OK) + { + send_http_error(con, status, sub->dest); + return; + } + + exclude = cupsdGetPrivateAttrs(policy, con, sub->dest, sub->owner); + + /* + * Copy the subscription attributes to the response using the + * requested-attributes attribute that may be provided by the client. + */ + + ra = create_requested_array(con->request); + + copy_subscription_attrs(con, sub, ra, exclude); + + cupsArrayDelete(ra); + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'get_subscriptions()' - Get subscriptions. + */ + +static void +get_subscriptions(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Printer/job URI */ +{ + http_status_t status; /* Policy status */ + int count; /* Number of subscriptions */ + int limit; /* Limit */ + cupsd_subscription_t *sub; /* Subscription */ + cups_array_t *ra; /* Requested attributes array */ + ipp_attribute_t *attr; /* Attribute */ + cups_ptype_t dtype; /* Destination type (printer/class) */ + char scheme[HTTP_MAX_URI], + /* Scheme portion of URI */ + username[HTTP_MAX_URI], + /* Username portion of URI */ + host[HTTP_MAX_URI], + /* Host portion of URI */ + resource[HTTP_MAX_URI]; + /* Resource portion of URI */ + int port; /* Port portion of URI */ + cupsd_job_t *job; /* Job pointer */ + cupsd_printer_t *printer; /* Printer */ + cupsd_policy_t *policy; /* Policy */ + cups_array_t *exclude; /* Private attributes array */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "get_subscriptions(con=%p[%d], uri=%s)", + con, con->http.fd, uri->values[0].string.text); + + /* + * Is the destination valid? + */ + + httpSeparateURI(HTTP_URI_CODING_ALL, uri->values[0].string.text, scheme, + sizeof(scheme), username, sizeof(username), host, + sizeof(host), &port, resource, sizeof(resource)); + + if (!strcmp(resource, "/") || + (!strncmp(resource, "/jobs", 5) && strlen(resource) <= 6) || + (!strncmp(resource, "/printers", 9) && strlen(resource) <= 10) || + (!strncmp(resource, "/classes", 8) && strlen(resource) <= 9)) + { + printer = NULL; + job = NULL; + } + else if (!strncmp(resource, "/jobs/", 6) && resource[6]) + { + printer = NULL; + job = cupsdFindJob(atoi(resource + 6)); + + if (!job) + { + send_ipp_status(con, IPP_NOT_FOUND, _("Job #%d does not exist."), + atoi(resource + 6)); + return; + } + } + else if (!cupsdValidateDest(uri->values[0].string.text, &dtype, &printer)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + else if ((attr = ippFindAttribute(con->request, "notify-job-id", + IPP_TAG_INTEGER)) != NULL) + { + job = cupsdFindJob(attr->values[0].integer); + + if (!job) + { + send_ipp_status(con, IPP_NOT_FOUND, _("Job #%d does not exist."), + attr->values[0].integer); + return; + } + } + else + job = NULL; + + /* + * Check policy... + */ + + if (printer) + policy = printer->op_policy_ptr; + else + policy = DefaultPolicyPtr; + + if ((status = cupsdCheckPolicy(policy, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, printer); + return; + } + + /* + * Copy the subscription attributes to the response using the + * requested-attributes attribute that may be provided by the client. + */ + + ra = create_requested_array(con->request); + + if ((attr = ippFindAttribute(con->request, "limit", + IPP_TAG_INTEGER)) != NULL) + limit = attr->values[0].integer; + else + limit = 0; + + /* + * See if we only want to see subscriptions for a specific user... + */ + + if ((attr = ippFindAttribute(con->request, "my-subscriptions", + IPP_TAG_BOOLEAN)) != NULL && + attr->values[0].boolean) + strlcpy(username, get_username(con), sizeof(username)); + else + username[0] = '\0'; + + for (sub = (cupsd_subscription_t *)cupsArrayFirst(Subscriptions), count = 0; + sub; + sub = (cupsd_subscription_t *)cupsArrayNext(Subscriptions)) + if ((!printer || sub->dest == printer) && (!job || sub->job == job) && + (!username[0] || !_cups_strcasecmp(username, sub->owner))) + { + ippAddSeparator(con->response); + + exclude = cupsdGetPrivateAttrs(sub->dest ? sub->dest->op_policy_ptr : + policy, con, sub->dest, + sub->owner); + + copy_subscription_attrs(con, sub, ra, exclude); + + count ++; + if (limit && count >= limit) + break; + } + + cupsArrayDelete(ra); + + if (count) + con->response->request.status.status_code = IPP_OK; + else + send_ipp_status(con, IPP_NOT_FOUND, _("No subscriptions found.")); +} + + +/* + * 'get_username()' - Get the username associated with a request. + */ + +static const char * /* O - Username */ +get_username(cupsd_client_t *con) /* I - Connection */ +{ + ipp_attribute_t *attr; /* Attribute */ + + + if (con->username[0]) + return (con->username); + else if ((attr = ippFindAttribute(con->request, "requesting-user-name", + IPP_TAG_NAME)) != NULL) + return (attr->values[0].string.text); + else + return ("anonymous"); +} + + +/* + * 'hold_job()' - Hold a print job. + */ + +static void +hold_job(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Job or Printer URI */ +{ + ipp_attribute_t *attr; /* Current job-hold-until */ + const char *when; /* New value */ + int jobid; /* Job ID */ + char scheme[HTTP_MAX_URI], /* Method portion of URI */ + username[HTTP_MAX_URI], /* Username portion of URI */ + host[HTTP_MAX_URI], /* Host portion of URI */ + resource[HTTP_MAX_URI]; /* Resource portion of URI */ + int port; /* Port portion of URI */ + cupsd_job_t *job; /* Job information */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "hold_job(%p[%d], %s)", con, con->http.fd, + uri->values[0].string.text); + + /* + * See if we have a job URI or a printer URI... + */ + + if (!strcmp(uri->name, "printer-uri")) + { + /* + * Got a printer URI; see if we also have a job-id attribute... + */ + + if ((attr = ippFindAttribute(con->request, "job-id", + IPP_TAG_INTEGER)) == NULL) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Got a printer-uri attribute but no job-id.")); + return; + } + + jobid = attr->values[0].integer; + } + else + { + /* + * Got a job URI; parse it to get the job ID... + */ + + httpSeparateURI(HTTP_URI_CODING_ALL, uri->values[0].string.text, scheme, + sizeof(scheme), username, sizeof(username), host, + sizeof(host), &port, resource, sizeof(resource)); + + if (strncmp(resource, "/jobs/", 6)) + { + /* + * Not a valid URI! + */ + + send_ipp_status(con, IPP_BAD_REQUEST, + _("Bad job-uri \"%s\"."), + uri->values[0].string.text); + return; + } + + jobid = atoi(resource + 6); + } + + /* + * See if the job exists... + */ + + if ((job = cupsdFindJob(jobid)) == NULL) + { + /* + * Nope - return a "not found" error... + */ + + send_ipp_status(con, IPP_NOT_FOUND, _("Job #%d does not exist."), jobid); + return; + } + + /* + * See if the job is owned by the requesting user... + */ + + if (!validate_user(job, con, job->username, username, sizeof(username))) + { + send_http_error(con, con->username[0] ? HTTP_FORBIDDEN : HTTP_UNAUTHORIZED, + cupsdFindDest(job->dest)); + return; + } + + /* + * See if the job is in a state that allows holding... + */ + + if (job->state_value > IPP_JOB_STOPPED) + { + /* + * Return a "not-possible" error... + */ + + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Job #%d is finished and cannot be altered."), + job->id); + return; + } + + /* + * Hold the job and return... + */ + + if ((attr = ippFindAttribute(con->request, "job-hold-until", + IPP_TAG_KEYWORD)) == NULL) + attr = ippFindAttribute(con->request, "job-hold-until", IPP_TAG_NAME); + + if (attr) + { + when = attr->values[0].string.text; + + cupsdAddEvent(CUPSD_EVENT_JOB_CONFIG_CHANGED, cupsdFindDest(job->dest), job, + "Job job-hold-until value changed by user."); + } + else + when = "indefinite"; + + cupsdSetJobHoldUntil(job, when, 1); + cupsdSetJobState(job, IPP_JOB_HELD, CUPSD_JOB_DEFAULT, "Job held by \"%s\".", + username); + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'hold_new_jobs()' - Hold pending/new jobs on a printer or class. + */ + +static void +hold_new_jobs(cupsd_client_t *con, /* I - Connection */ + ipp_attribute_t *uri) /* I - Printer URI */ +{ + http_status_t status; /* Policy status */ + cups_ptype_t dtype; /* Destination type (printer/class) */ + cupsd_printer_t *printer; /* Printer data */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "hold_new_jobs(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * Is the destination valid? + */ + + if (!cupsdValidateDest(uri->values[0].string.text, &dtype, &printer)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(printer->op_policy_ptr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, printer); + return; + } + + /* + * Hold pending/new jobs sent to the printer... + */ + + printer->holding_new_jobs = 1; + + cupsdSetPrinterReasons(printer, "+hold-new-jobs"); + + if (dtype & CUPS_PRINTER_CLASS) + cupsdLogMessage(CUPSD_LOG_INFO, + "Class \"%s\" now holding pending/new jobs (\"%s\").", + printer->name, get_username(con)); + else + cupsdLogMessage(CUPSD_LOG_INFO, + "Printer \"%s\" now holding pending/new jobs (\"%s\").", + printer->name, get_username(con)); + + /* + * Everything was ok, so return OK status... + */ + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'move_job()' - Move a job to a new destination. + */ + +static void +move_job(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Job URI */ +{ + http_status_t status; /* Policy status */ + ipp_attribute_t *attr; /* Current attribute */ + int jobid; /* Job ID */ + cupsd_job_t *job; /* Current job */ + const char *src; /* Source printer/class */ + cups_ptype_t stype, /* Source type (printer or class) */ + dtype; /* Destination type (printer/class) */ + char scheme[HTTP_MAX_URI], /* Scheme portion of URI */ + username[HTTP_MAX_URI], /* Username portion of URI */ + host[HTTP_MAX_URI], /* Host portion of URI */ + resource[HTTP_MAX_URI]; /* Resource portion of URI */ + int port; /* Port portion of URI */ + cupsd_printer_t *sprinter, /* Source printer */ + *dprinter; /* Destination printer */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "move_job(%p[%d], %s)", con, con->http.fd, + uri->values[0].string.text); + + /* + * Get the new printer or class... + */ + + if ((attr = ippFindAttribute(con->request, "job-printer-uri", + IPP_TAG_URI)) == NULL) + { + /* + * Need job-printer-uri... + */ + + send_ipp_status(con, IPP_BAD_REQUEST, + _("job-printer-uri attribute missing.")); + return; + } + + if (!cupsdValidateDest(attr->values[0].string.text, &dtype, &dprinter)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * See if we have a job URI or a printer URI... + */ + + httpSeparateURI(HTTP_URI_CODING_ALL, uri->values[0].string.text, scheme, + sizeof(scheme), username, sizeof(username), host, + sizeof(host), &port, resource, sizeof(resource)); + + if (!strcmp(uri->name, "printer-uri")) + { + /* + * Got a printer URI; see if we also have a job-id attribute... + */ + + if ((attr = ippFindAttribute(con->request, "job-id", + IPP_TAG_INTEGER)) == NULL) + { + /* + * Move all jobs... + */ + + if ((src = cupsdValidateDest(uri->values[0].string.text, &stype, + &sprinter)) == NULL) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + job = NULL; + } + else + { + /* + * Otherwise, just move a single job... + */ + + if ((job = cupsdFindJob(attr->values[0].integer)) == NULL) + { + /* + * Nope - return a "not found" error... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("Job #%d does not exist."), attr->values[0].integer); + return; + } + else + { + /* + * Job found, initialize source pointers... + */ + + src = NULL; + sprinter = NULL; + } + } + } + else + { + /* + * Got a job URI; parse it to get the job ID... + */ + + if (strncmp(resource, "/jobs/", 6)) + { + /* + * Not a valid URI! + */ + + send_ipp_status(con, IPP_BAD_REQUEST, _("Bad job-uri \"%s\"."), + uri->values[0].string.text); + return; + } + + /* + * See if the job exists... + */ + + jobid = atoi(resource + 6); + + if ((job = cupsdFindJob(jobid)) == NULL) + { + /* + * Nope - return a "not found" error... + */ + + send_ipp_status(con, IPP_NOT_FOUND, _("Job #%d does not exist."), jobid); + return; + } + else + { + /* + * Job found, initialize source pointers... + */ + + src = NULL; + sprinter = NULL; + } + } + + /* + * Check the policy of the destination printer... + */ + + if ((status = cupsdCheckPolicy(dprinter->op_policy_ptr, con, + job ? job->username : NULL)) != HTTP_OK) + { + send_http_error(con, status, dprinter); + return; + } + + /* + * Now move the job or jobs... + */ + + if (job) + { + /* + * See if the job has been completed... + */ + + if (job->state_value > IPP_JOB_STOPPED) + { + /* + * Return a "not-possible" error... + */ + + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Job #%d is finished and cannot be altered."), + job->id); + return; + } + + /* + * See if the job is owned by the requesting user... + */ + + if (!validate_user(job, con, job->username, username, sizeof(username))) + { + send_http_error(con, con->username[0] ? HTTP_FORBIDDEN : HTTP_UNAUTHORIZED, + cupsdFindDest(job->dest)); + return; + } + + /* + * Move the job to a different printer or class... + */ + + cupsdMoveJob(job, dprinter); + } + else + { + /* + * Got the source printer, now look through the jobs... + */ + + for (job = (cupsd_job_t *)cupsArrayFirst(Jobs); + job; + job = (cupsd_job_t *)cupsArrayNext(Jobs)) + { + /* + * See if the job is pointing at the source printer or has not been + * completed... + */ + + if (_cups_strcasecmp(job->dest, src) || + job->state_value > IPP_JOB_STOPPED) + continue; + + /* + * See if the job can be moved by the requesting user... + */ + + if (!validate_user(job, con, job->username, username, sizeof(username))) + continue; + + /* + * Move the job to a different printer or class... + */ + + cupsdMoveJob(job, dprinter); + } + } + + /* + * Start jobs if possible... + */ + + cupsdCheckJobs(); + + /* + * Return with "everything is OK" status... + */ + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'ppd_parse_line()' - Parse a PPD default line. + */ + +static int /* O - 0 on success, -1 on failure */ +ppd_parse_line(const char *line, /* I - Line */ + char *option, /* O - Option name */ + int olen, /* I - Size of option name */ + char *choice, /* O - Choice name */ + int clen) /* I - Size of choice name */ +{ + /* + * Verify this is a default option line... + */ + + if (strncmp(line, "*Default", 8)) + return (-1); + + /* + * Read the option name... + */ + + for (line += 8, olen --; + *line > ' ' && *line < 0x7f && *line != ':' && *line != '/'; + line ++) + if (olen > 0) + { + *option++ = *line; + olen --; + } + + *option = '\0'; + + /* + * Skip everything else up to the colon (:)... + */ + + while (*line && *line != ':') + line ++; + + if (!*line) + return (-1); + + line ++; + + /* + * Now grab the option choice, skipping leading whitespace... + */ + + while (isspace(*line & 255)) + line ++; + + for (clen --; + *line > ' ' && *line < 0x7f && *line != ':' && *line != '/'; + line ++) + if (clen > 0) + { + *choice++ = *line; + clen --; + } + + *choice = '\0'; + + /* + * Return with no errors... + */ + + return (0); +} + + +/* + * 'print_job()' - Print a file to a printer or class. + */ + +static void +print_job(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Printer URI */ +{ + ipp_attribute_t *attr; /* Current attribute */ + ipp_attribute_t *format; /* Document-format attribute */ + const char *default_format; /* document-format-default value */ + cupsd_job_t *job; /* New job */ + char filename[1024]; /* Job filename */ + mime_type_t *filetype; /* Type of file */ + char super[MIME_MAX_SUPER], /* Supertype of file */ + type[MIME_MAX_TYPE], /* Subtype of file */ + mimetype[MIME_MAX_SUPER + MIME_MAX_TYPE + 2]; + /* Textual name of mime type */ + cupsd_printer_t *printer; /* Printer data */ + struct stat fileinfo; /* File information */ + int kbytes; /* Size of file */ + int compression; /* Document compression */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "print_job(%p[%d], %s)", con, con->http.fd, + uri->values[0].string.text); + + /* + * Validate print file attributes, for now just document-format and + * compression (CUPS only supports "none" and "gzip")... + */ + + compression = CUPS_FILE_NONE; + + if ((attr = ippFindAttribute(con->request, "compression", + IPP_TAG_KEYWORD)) != NULL) + { + if (strcmp(attr->values[0].string.text, "none") +#ifdef HAVE_LIBZ + && strcmp(attr->values[0].string.text, "gzip") +#endif /* HAVE_LIBZ */ + ) + { + send_ipp_status(con, IPP_ATTRIBUTES, + _("Unsupported compression \"%s\"."), + attr->values[0].string.text); + ippAddString(con->response, IPP_TAG_UNSUPPORTED_GROUP, IPP_TAG_KEYWORD, + "compression", NULL, attr->values[0].string.text); + return; + } + +#ifdef HAVE_LIBZ + if (!strcmp(attr->values[0].string.text, "gzip")) + compression = CUPS_FILE_GZIP; +#endif /* HAVE_LIBZ */ + } + + /* + * Do we have a file to print? + */ + + if (!con->filename) + { + send_ipp_status(con, IPP_BAD_REQUEST, _("No file in print request.")); + return; + } + + /* + * Is the destination valid? + */ + + if (!cupsdValidateDest(uri->values[0].string.text, NULL, &printer)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * Is it a format we support? + */ + + if ((format = ippFindAttribute(con->request, "document-format", + IPP_TAG_MIMETYPE)) != NULL) + { + /* + * Grab format from client... + */ + + if (sscanf(format->values[0].string.text, "%15[^/]/%31[^;]", super, + type) != 2) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Bad document-format \"%s\"."), + format->values[0].string.text); + return; + } + } + else if ((default_format = cupsGetOption("document-format", + printer->num_options, + printer->options)) != NULL) + { + /* + * Use default document format... + */ + + if (sscanf(default_format, "%15[^/]/%31[^;]", super, type) != 2) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Bad document-format \"%s\"."), + default_format); + return; + } + } + else + { + /* + * Auto-type it! + */ + + strcpy(super, "application"); + strcpy(type, "octet-stream"); + } + + if (!strcmp(super, "application") && !strcmp(type, "octet-stream")) + { + /* + * Auto-type the file... + */ + + ipp_attribute_t *doc_name; /* document-name attribute */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG, "[Job ???] Auto-typing file..."); + + doc_name = ippFindAttribute(con->request, "document-name", IPP_TAG_NAME); + filetype = mimeFileType(MimeDatabase, con->filename, + doc_name ? doc_name->values[0].string.text : NULL, + &compression); + + if (!filetype) + filetype = mimeType(MimeDatabase, super, type); + + cupsdLogMessage(CUPSD_LOG_INFO, "[Job ???] Request file type is %s/%s.", + filetype->super, filetype->type); + } + else + filetype = mimeType(MimeDatabase, super, type); + + if (filetype && + (!format || + (!strcmp(super, "application") && !strcmp(type, "octet-stream")))) + { + /* + * Replace the document-format attribute value with the auto-typed or + * default one. + */ + + snprintf(mimetype, sizeof(mimetype), "%s/%s", filetype->super, + filetype->type); + + if (format) + { + _cupsStrFree(format->values[0].string.text); + + format->values[0].string.text = _cupsStrAlloc(mimetype); + } + else + ippAddString(con->request, IPP_TAG_JOB, IPP_TAG_MIMETYPE, + "document-format", NULL, mimetype); + } + else if (!filetype) + { + send_ipp_status(con, IPP_DOCUMENT_FORMAT, + _("Unsupported document-format \"%s\"."), + format ? format->values[0].string.text : + "application/octet-stream"); + cupsdLogMessage(CUPSD_LOG_INFO, + "Hint: Do you have the raw file printing rules enabled?"); + + if (format) + ippAddString(con->response, IPP_TAG_UNSUPPORTED_GROUP, IPP_TAG_MIMETYPE, + "document-format", NULL, format->values[0].string.text); + + return; + } + + /* + * Read any embedded job ticket info from PS files... + */ + + if (!_cups_strcasecmp(filetype->super, "application") && + (!_cups_strcasecmp(filetype->type, "postscript") || + !_cups_strcasecmp(filetype->type, "pdf"))) + read_job_ticket(con); + + /* + * Create the job object... + */ + + if ((job = add_job(con, printer, filetype)) == NULL) + return; + + /* + * Update quota data... + */ + + if (stat(con->filename, &fileinfo)) + kbytes = 0; + else + kbytes = (fileinfo.st_size + 1023) / 1024; + + cupsdUpdateQuota(printer, job->username, 0, kbytes); + + if ((attr = ippFindAttribute(job->attrs, "job-k-octets", + IPP_TAG_INTEGER)) != NULL) + attr->values[0].integer += kbytes; + + /* + * Add the job file... + */ + + if (add_file(con, job, filetype, compression)) + return; + + snprintf(filename, sizeof(filename), "%s/d%05d-%03d", RequestRoot, job->id, + job->num_files); + rename(con->filename, filename); + cupsdClearString(&con->filename); + + /* + * See if we need to add the ending sheet... + */ + + if (cupsdTimeoutJob(job)) + return; + + /* + * Log and save the job... + */ + + cupsdLogJob(job, CUPSD_LOG_INFO, + "File of type %s/%s queued by \"%s\".", + filetype->super, filetype->type, job->username); + cupsdLogJob(job, CUPSD_LOG_DEBUG, "hold_until=%d", (int)job->hold_until); + cupsdLogJob(job, CUPSD_LOG_INFO, "Queued on \"%s\" by \"%s\".", + job->dest, job->username); + + /* + * Start the job if possible... + */ + + cupsdCheckJobs(); +} + + +/* + * 'read_job_ticket()' - Read a job ticket embedded in a print file. + * + * This function only gets called when printing a single PDF or PostScript + * file using the Print-Job operation. It doesn't work for Create-Job + + * Send-File, since the job attributes need to be set at job creation + * time for banners to work. The embedded job ticket stuff is here + * primarily to allow the Windows printer driver for CUPS to pass in JCL + * options and IPP attributes which otherwise would be lost. + * + * The format of a job ticket is simple: + * + * %cupsJobTicket: attr1=value1 attr2=value2 ... attrN=valueN + * + * %cupsJobTicket: attr1=value1 + * %cupsJobTicket: attr2=value2 + * ... + * %cupsJobTicket: attrN=valueN + * + * Job ticket lines must appear immediately after the first line that + * specifies PostScript (%!PS-Adobe-3.0) or PDF (%PDF) format, and CUPS + * stops looking for job ticket info when it finds a line that does not begin + * with "%cupsJobTicket:". + * + * The maximum length of a job ticket line, including the prefix, is + * 255 characters to conform with the Adobe DSC. + * + * Read-only attributes are rejected with a notice to the error log in + * case a malicious user tries anything. Since the job ticket is read + * prior to attribute validation in print_job(), job ticket attributes + * will go through the same validation as IPP attributes... + */ + +static void +read_job_ticket(cupsd_client_t *con) /* I - Client connection */ +{ + cups_file_t *fp; /* File to read from */ + char line[256]; /* Line data */ + int num_options; /* Number of options */ + cups_option_t *options; /* Options */ + ipp_t *ticket; /* New attributes */ + ipp_attribute_t *attr, /* Current attribute */ + *attr2, /* Job attribute */ + *prev2; /* Previous job attribute */ + + + /* + * First open the print file... + */ + + if ((fp = cupsFileOpen(con->filename, "rb")) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to open print file for job ticket - %s", + strerror(errno)); + return; + } + + /* + * Skip the first line... + */ + + if (cupsFileGets(fp, line, sizeof(line)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to read from print file for job ticket - %s", + strerror(errno)); + cupsFileClose(fp); + return; + } + + if (strncmp(line, "%!PS-Adobe-", 11) && strncmp(line, "%PDF-", 5)) + { + /* + * Not a DSC-compliant file, so no job ticket info will be available... + */ + + cupsFileClose(fp); + return; + } + + /* + * Read job ticket info from the file... + */ + + num_options = 0; + options = NULL; + + while (cupsFileGets(fp, line, sizeof(line))) + { + /* + * Stop at the first non-ticket line... + */ + + if (strncmp(line, "%cupsJobTicket:", 15)) + break; + + /* + * Add the options to the option array... + */ + + num_options = cupsParseOptions(line + 15, num_options, &options); + } + + /* + * Done with the file; see if we have any options... + */ + + cupsFileClose(fp); + + if (num_options == 0) + return; + + /* + * OK, convert the options to an attribute list, and apply them to + * the request... + */ + + ticket = ippNew(); + cupsEncodeOptions(ticket, num_options, options); + + /* + * See what the user wants to change. + */ + + for (attr = ticket->attrs; attr; attr = attr->next) + { + if (attr->group_tag != IPP_TAG_JOB || !attr->name) + continue; + + if (!strcmp(attr->name, "job-originating-host-name") || + !strcmp(attr->name, "job-originating-user-name") || + !strcmp(attr->name, "job-media-sheets-completed") || + !strcmp(attr->name, "job-k-octets") || + !strcmp(attr->name, "job-id") || + !strncmp(attr->name, "job-state", 9) || + !strncmp(attr->name, "time-at-", 8)) + continue; /* Read-only attrs */ + + if ((attr2 = ippFindAttribute(con->request, attr->name, + IPP_TAG_ZERO)) != NULL) + { + /* + * Some other value; first free the old value... + */ + + if (con->request->attrs == attr2) + { + con->request->attrs = attr2->next; + prev2 = NULL; + } + else + { + for (prev2 = con->request->attrs; prev2; prev2 = prev2->next) + if (prev2->next == attr2) + { + prev2->next = attr2->next; + break; + } + } + + if (con->request->last == attr2) + con->request->last = prev2; + + ippDeleteAttribute(NULL, attr2); + } + + /* + * Add new option by copying it... + */ + + ippCopyAttribute(con->request, attr, 0); + } + + /* + * Then free the attribute list and option array... + */ + + ippDelete(ticket); + cupsFreeOptions(num_options, options); +} + + +/* + * 'reject_jobs()' - Reject print jobs to a printer. + */ + +static void +reject_jobs(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Printer or class URI */ +{ + http_status_t status; /* Policy status */ + cups_ptype_t dtype; /* Destination type (printer/class) */ + cupsd_printer_t *printer; /* Printer data */ + ipp_attribute_t *attr; /* printer-state-message text */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "reject_jobs(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * Is the destination valid? + */ + + if (!cupsdValidateDest(uri->values[0].string.text, &dtype, &printer)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(printer->op_policy_ptr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, printer); + return; + } + + /* + * Reject jobs sent to the printer... + */ + + printer->accepting = 0; + + if ((attr = ippFindAttribute(con->request, "printer-state-message", + IPP_TAG_TEXT)) == NULL) + strcpy(printer->state_message, "Rejecting Jobs"); + else + strlcpy(printer->state_message, attr->values[0].string.text, + sizeof(printer->state_message)); + + cupsdAddEvent(CUPSD_EVENT_PRINTER_STATE, printer, NULL, + "No longer accepting jobs."); + + if (dtype & CUPS_PRINTER_CLASS) + { + cupsdMarkDirty(CUPSD_DIRTY_CLASSES); + + cupsdLogMessage(CUPSD_LOG_INFO, "Class \"%s\" rejecting jobs (\"%s\").", + printer->name, get_username(con)); + } + else + { + cupsdMarkDirty(CUPSD_DIRTY_PRINTERS); + + cupsdLogMessage(CUPSD_LOG_INFO, "Printer \"%s\" rejecting jobs (\"%s\").", + printer->name, get_username(con)); + } + + /* + * Everything was ok, so return OK status... + */ + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'release_held_new_jobs()' - Release pending/new jobs on a printer or class. + */ + +static void +release_held_new_jobs( + cupsd_client_t *con, /* I - Connection */ + ipp_attribute_t *uri) /* I - Printer URI */ +{ + http_status_t status; /* Policy status */ + cups_ptype_t dtype; /* Destination type (printer/class) */ + cupsd_printer_t *printer; /* Printer data */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "release_held_new_jobs(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * Is the destination valid? + */ + + if (!cupsdValidateDest(uri->values[0].string.text, &dtype, &printer)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(printer->op_policy_ptr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, printer); + return; + } + + /* + * Hold pending/new jobs sent to the printer... + */ + + printer->holding_new_jobs = 0; + + cupsdSetPrinterReasons(printer, "-hold-new-jobs"); + + if (dtype & CUPS_PRINTER_CLASS) + cupsdLogMessage(CUPSD_LOG_INFO, + "Class \"%s\" now printing pending/new jobs (\"%s\").", + printer->name, get_username(con)); + else + cupsdLogMessage(CUPSD_LOG_INFO, + "Printer \"%s\" now printing pending/new jobs (\"%s\").", + printer->name, get_username(con)); + + /* + * Everything was ok, so return OK status... + */ + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'release_job()' - Release a held print job. + */ + +static void +release_job(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Job or Printer URI */ +{ + ipp_attribute_t *attr; /* Current attribute */ + int jobid; /* Job ID */ + char scheme[HTTP_MAX_URI], /* Method portion of URI */ + username[HTTP_MAX_URI], /* Username portion of URI */ + host[HTTP_MAX_URI], /* Host portion of URI */ + resource[HTTP_MAX_URI]; /* Resource portion of URI */ + int port; /* Port portion of URI */ + cupsd_job_t *job; /* Job information */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "release_job(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * See if we have a job URI or a printer URI... + */ + + if (!strcmp(uri->name, "printer-uri")) + { + /* + * Got a printer URI; see if we also have a job-id attribute... + */ + + if ((attr = ippFindAttribute(con->request, "job-id", + IPP_TAG_INTEGER)) == NULL) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Got a printer-uri attribute but no job-id.")); + return; + } + + jobid = attr->values[0].integer; + } + else + { + /* + * Got a job URI; parse it to get the job ID... + */ + + httpSeparateURI(HTTP_URI_CODING_ALL, uri->values[0].string.text, scheme, + sizeof(scheme), username, sizeof(username), host, + sizeof(host), &port, resource, sizeof(resource)); + + if (strncmp(resource, "/jobs/", 6)) + { + /* + * Not a valid URI! + */ + + send_ipp_status(con, IPP_BAD_REQUEST, _("Bad job-uri \"%s\"."), + uri->values[0].string.text); + return; + } + + jobid = atoi(resource + 6); + } + + /* + * See if the job exists... + */ + + if ((job = cupsdFindJob(jobid)) == NULL) + { + /* + * Nope - return a "not found" error... + */ + + send_ipp_status(con, IPP_NOT_FOUND, _("Job #%d does not exist."), jobid); + return; + } + + /* + * See if job is "held"... + */ + + if (job->state_value != IPP_JOB_HELD) + { + /* + * Nope - return a "not possible" error... + */ + + send_ipp_status(con, IPP_NOT_POSSIBLE, _("Job #%d is not held."), jobid); + return; + } + + /* + * See if the job is owned by the requesting user... + */ + + if (!validate_user(job, con, job->username, username, sizeof(username))) + { + send_http_error(con, con->username[0] ? HTTP_FORBIDDEN : HTTP_UNAUTHORIZED, + cupsdFindDest(job->dest)); + return; + } + + /* + * Reset the job-hold-until value to "no-hold"... + */ + + if ((attr = ippFindAttribute(job->attrs, "job-hold-until", + IPP_TAG_KEYWORD)) == NULL) + attr = ippFindAttribute(job->attrs, "job-hold-until", IPP_TAG_NAME); + + if (attr) + { + _cupsStrFree(attr->values[0].string.text); + + attr->value_tag = IPP_TAG_KEYWORD; + attr->values[0].string.text = _cupsStrAlloc("no-hold"); + + cupsdAddEvent(CUPSD_EVENT_JOB_CONFIG_CHANGED, cupsdFindDest(job->dest), job, + "Job job-hold-until value changed by user."); + } + + /* + * Release the job and return... + */ + + cupsdReleaseJob(job); + + cupsdAddEvent(CUPSD_EVENT_JOB_STATE, cupsdFindDest(job->dest), job, + "Job released by user."); + + cupsdLogJob(job, CUPSD_LOG_INFO, "Released by \"%s\".", username); + + con->response->request.status.status_code = IPP_OK; + + cupsdCheckJobs(); +} + + +/* + * 'renew_subscription()' - Renew an existing subscription... + */ + +static void +renew_subscription( + cupsd_client_t *con, /* I - Client connection */ + int sub_id) /* I - Subscription ID */ +{ + http_status_t status; /* Policy status */ + cupsd_subscription_t *sub; /* Subscription */ + ipp_attribute_t *lease; /* notify-lease-duration */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "renew_subscription(con=%p[%d], sub_id=%d)", + con, con->http.fd, sub_id); + + /* + * Is the subscription ID valid? + */ + + if ((sub = cupsdFindSubscription(sub_id)) == NULL) + { + /* + * Bad subscription ID... + */ + + send_ipp_status(con, IPP_NOT_FOUND, _("Subscription #%d does not exist."), + sub_id); + return; + } + + if (sub->job) + { + /* + * Job subscriptions cannot be renewed... + */ + + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Job subscriptions cannot be renewed.")); + return; + } + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(sub->dest ? sub->dest->op_policy_ptr : + DefaultPolicyPtr, + con, sub->owner)) != HTTP_OK) + { + send_http_error(con, status, sub->dest); + return; + } + + /* + * Renew the subscription... + */ + + lease = ippFindAttribute(con->request, "notify-lease-duration", + IPP_TAG_INTEGER); + + sub->lease = lease ? lease->values[0].integer : DefaultLeaseDuration; + + if (MaxLeaseDuration && (sub->lease == 0 || sub->lease > MaxLeaseDuration)) + { + cupsdLogMessage(CUPSD_LOG_INFO, + "renew_subscription: Limiting notify-lease-duration to " + "%d seconds.", + MaxLeaseDuration); + sub->lease = MaxLeaseDuration; + } + + sub->expire = sub->lease ? time(NULL) + sub->lease : 0; + + cupsdMarkDirty(CUPSD_DIRTY_SUBSCRIPTIONS); + + con->response->request.status.status_code = IPP_OK; + + ippAddInteger(con->response, IPP_TAG_SUBSCRIPTION, IPP_TAG_INTEGER, + "notify-lease-duration", sub->lease); +} + + +/* + * 'restart_job()' - Restart an old print job. + */ + +static void +restart_job(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Job or Printer URI */ +{ + ipp_attribute_t *attr; /* Current attribute */ + int jobid; /* Job ID */ + cupsd_job_t *job; /* Job information */ + char scheme[HTTP_MAX_URI], /* Method portion of URI */ + username[HTTP_MAX_URI], /* Username portion of URI */ + host[HTTP_MAX_URI], /* Host portion of URI */ + resource[HTTP_MAX_URI]; /* Resource portion of URI */ + int port; /* Port portion of URI */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "restart_job(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * See if we have a job URI or a printer URI... + */ + + if (!strcmp(uri->name, "printer-uri")) + { + /* + * Got a printer URI; see if we also have a job-id attribute... + */ + + if ((attr = ippFindAttribute(con->request, "job-id", + IPP_TAG_INTEGER)) == NULL) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Got a printer-uri attribute but no job-id.")); + return; + } + + jobid = attr->values[0].integer; + } + else + { + /* + * Got a job URI; parse it to get the job ID... + */ + + httpSeparateURI(HTTP_URI_CODING_ALL, uri->values[0].string.text, scheme, + sizeof(scheme), username, sizeof(username), host, + sizeof(host), &port, resource, sizeof(resource)); + + if (strncmp(resource, "/jobs/", 6)) + { + /* + * Not a valid URI! + */ + + send_ipp_status(con, IPP_BAD_REQUEST, _("Bad job-uri \"%s\"."), + uri->values[0].string.text); + return; + } + + jobid = atoi(resource + 6); + } + + /* + * See if the job exists... + */ + + if ((job = cupsdFindJob(jobid)) == NULL) + { + /* + * Nope - return a "not found" error... + */ + + send_ipp_status(con, IPP_NOT_FOUND, _("Job #%d does not exist."), jobid); + return; + } + + /* + * See if job is in any of the "completed" states... + */ + + if (job->state_value <= IPP_JOB_PROCESSING) + { + /* + * Nope - return a "not possible" error... + */ + + send_ipp_status(con, IPP_NOT_POSSIBLE, _("Job #%d is not complete."), + jobid); + return; + } + + /* + * See if we have retained the job files... + */ + + cupsdLoadJob(job); + + if (!job->attrs || job->num_files == 0) + { + /* + * Nope - return a "not possible" error... + */ + + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Job #%d cannot be restarted - no files."), jobid); + return; + } + + /* + * See if the job is owned by the requesting user... + */ + + if (!validate_user(job, con, job->username, username, sizeof(username))) + { + send_http_error(con, con->username[0] ? HTTP_FORBIDDEN : HTTP_UNAUTHORIZED, + cupsdFindDest(job->dest)); + return; + } + + /* + * See if the job-hold-until attribute is specified... + */ + + if ((attr = ippFindAttribute(con->request, "job-hold-until", + IPP_TAG_KEYWORD)) == NULL) + attr = ippFindAttribute(con->request, "job-hold-until", IPP_TAG_NAME); + + if (attr && strcmp(attr->values[0].string.text, "no-hold")) + { + /* + * Return the job to a held state... + */ + + cupsdLogJob(job, CUPSD_LOG_DEBUG, + "Restarted by \"%s\" with job-hold-until=%s.", + username, attr->values[0].string.text); + cupsdSetJobHoldUntil(job, attr->values[0].string.text, 0); + + cupsdAddEvent(CUPSD_EVENT_JOB_CONFIG_CHANGED | CUPSD_EVENT_JOB_STATE, + NULL, job, "Job restarted by user with job-hold-until=%s", + attr->values[0].string.text); + } + else + { + /* + * Restart the job... + */ + + cupsdRestartJob(job); + cupsdCheckJobs(); + } + + cupsdLogJob(job, CUPSD_LOG_INFO, "Restarted by \"%s\".", username); + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'save_auth_info()' - Save authentication information for a job. + */ + +static void +save_auth_info( + cupsd_client_t *con, /* I - Client connection */ + cupsd_job_t *job, /* I - Job */ + ipp_attribute_t *auth_info) /* I - auth-info attribute, if any */ +{ + int i; /* Looping var */ + char filename[1024]; /* Job authentication filename */ + cups_file_t *fp; /* Job authentication file */ + char line[65536]; /* Line for file */ + cupsd_printer_t *dest; /* Destination printer/class */ + + + /* + * This function saves the in-memory authentication information for + * a job so that it can be used to authenticate with a remote host. + * The information is stored in a file that is readable only by the + * root user. The fields are Base-64 encoded, each on a separate line, + * followed by random number (up to 1024) of newlines to limit the + * amount of information that is exposed. + * + * Because of the potential for exposing of authentication information, + * this functionality is only enabled when running cupsd as root. + * + * This caching only works for the Basic and BasicDigest authentication + * types. Digest authentication cannot be cached this way, and in + * the future Kerberos authentication may make all of this obsolete. + * + * Authentication information is saved whenever an authenticated + * Print-Job, Create-Job, or CUPS-Authenticate-Job operation is + * performed. + * + * This information is deleted after a job is completed or canceled, + * so reprints may require subsequent re-authentication. + */ + + if (RunUser) + return; + + if ((dest = cupsdFindDest(job->dest)) == NULL) + return; + + /* + * Create the authentication file and change permissions... + */ + + snprintf(filename, sizeof(filename), "%s/a%05d", RequestRoot, job->id); + if ((fp = cupsFileOpen(filename, "w")) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to save authentication info to \"%s\" - %s", + filename, strerror(errno)); + return; + } + + fchown(cupsFileNumber(fp), 0, 0); + fchmod(cupsFileNumber(fp), 0400); + + cupsFilePuts(fp, "CUPSD-AUTH-V2\n"); + + for (i = 0; + i < (int)(sizeof(job->auth_env) / sizeof(job->auth_env[0])); + i ++) + cupsdClearString(job->auth_env + i); + + if (auth_info && auth_info->num_values == dest->num_auth_info_required) + { + /* + * Write 1 to 3 auth values... + */ + + for (i = 0; + i < auth_info->num_values && + i < (int)(sizeof(job->auth_env) / sizeof(job->auth_env[0])); + i ++) + { + httpEncode64_2(line, sizeof(line), auth_info->values[i].string.text, + strlen(auth_info->values[i].string.text)); + cupsFilePutConf(fp, dest->auth_info_required[i], line); + + if (!strcmp(dest->auth_info_required[i], "username")) + cupsdSetStringf(job->auth_env + i, "AUTH_USERNAME=%s", + auth_info->values[i].string.text); + else if (!strcmp(dest->auth_info_required[i], "domain")) + cupsdSetStringf(job->auth_env + i, "AUTH_DOMAIN=%s", + auth_info->values[i].string.text); + else if (!strcmp(dest->auth_info_required[i], "password")) + cupsdSetStringf(job->auth_env + i, "AUTH_PASSWORD=%s", + auth_info->values[i].string.text); + else if (!strcmp(dest->auth_info_required[i], "negotiate")) + cupsdSetStringf(job->auth_env + i, "AUTH_NEGOTIATE=%s", + auth_info->values[i].string.text); + else + i --; + } + } + else if (auth_info && auth_info->num_values == 2 && + dest->num_auth_info_required == 1 && + !strcmp(dest->auth_info_required[0], "negotiate")) + { + /* + * Allow fallback to username+password for Kerberized queues... + */ + + httpEncode64_2(line, sizeof(line), auth_info->values[0].string.text, + strlen(auth_info->values[0].string.text)); + cupsFilePutConf(fp, "username", line); + + cupsdSetStringf(job->auth_env + 0, "AUTH_USERNAME=%s", + auth_info->values[0].string.text); + + httpEncode64_2(line, sizeof(line), auth_info->values[1].string.text, + strlen(auth_info->values[1].string.text)); + cupsFilePutConf(fp, "password", line); + + cupsdSetStringf(job->auth_env + 1, "AUTH_PASSWORD=%s", + auth_info->values[1].string.text); + } + else if (con->username[0]) + { + /* + * Write the authenticated username... + */ + + httpEncode64_2(line, sizeof(line), con->username, strlen(con->username)); + cupsFilePutConf(fp, "username", line); + + cupsdSetStringf(job->auth_env + 0, "AUTH_USERNAME=%s", con->username); + + /* + * Write the authenticated password... + */ + + httpEncode64_2(line, sizeof(line), con->password, strlen(con->password)); + cupsFilePutConf(fp, "password", line); + + cupsdSetStringf(job->auth_env + 1, "AUTH_PASSWORD=%s", con->password); + } + +#ifdef HAVE_GSSAPI + if (con->gss_uid > 0) + { + cupsFilePrintf(fp, "uid %d\n", (int)con->gss_uid); + cupsdSetStringf(&job->auth_uid, "AUTH_UID=%d", (int)con->gss_uid); + } +#endif /* HAVE_GSSAPI */ + + /* + * Write a random number of newlines to the end of the file... + */ + + for (i = (CUPS_RAND() % 1024); i >= 0; i --) + cupsFilePutChar(fp, '\n'); + + /* + * Close the file and return... + */ + + cupsFileClose(fp); +} + + +/* + * 'send_document()' - Send a file to a printer or class. + */ + +static void +send_document(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Printer URI */ +{ + ipp_attribute_t *attr; /* Current attribute */ + ipp_attribute_t *format; /* Request's document-format attribute */ + ipp_attribute_t *jformat; /* Job's document-format attribute */ + const char *default_format;/* document-format-default value */ + int jobid; /* Job ID number */ + cupsd_job_t *job; /* Current job */ + char job_uri[HTTP_MAX_URI], + /* Job URI */ + scheme[HTTP_MAX_URI], + /* Method portion of URI */ + username[HTTP_MAX_URI], + /* Username portion of URI */ + host[HTTP_MAX_URI], + /* Host portion of URI */ + resource[HTTP_MAX_URI]; + /* Resource portion of URI */ + int port; /* Port portion of URI */ + mime_type_t *filetype; /* Type of file */ + char super[MIME_MAX_SUPER], + /* Supertype of file */ + type[MIME_MAX_TYPE], + /* Subtype of file */ + mimetype[MIME_MAX_SUPER + MIME_MAX_TYPE + 2]; + /* Textual name of mime type */ + char filename[1024]; /* Job filename */ + cupsd_printer_t *printer; /* Current printer */ + struct stat fileinfo; /* File information */ + int kbytes; /* Size of file */ + int compression; /* Type of compression */ + int start_job; /* Start the job? */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "send_document(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * See if we have a job URI or a printer URI... + */ + + if (!strcmp(uri->name, "printer-uri")) + { + /* + * Got a printer URI; see if we also have a job-id attribute... + */ + + if ((attr = ippFindAttribute(con->request, "job-id", + IPP_TAG_INTEGER)) == NULL) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Got a printer-uri attribute but no job-id.")); + return; + } + + jobid = attr->values[0].integer; + } + else + { + /* + * Got a job URI; parse it to get the job ID... + */ + + httpSeparateURI(HTTP_URI_CODING_ALL, uri->values[0].string.text, scheme, + sizeof(scheme), username, sizeof(username), host, + sizeof(host), &port, resource, sizeof(resource)); + + if (strncmp(resource, "/jobs/", 6)) + { + /* + * Not a valid URI! + */ + + send_ipp_status(con, IPP_BAD_REQUEST, _("Bad job-uri \"%s\"."), + uri->values[0].string.text); + return; + } + + jobid = atoi(resource + 6); + } + + /* + * See if the job exists... + */ + + if ((job = cupsdFindJob(jobid)) == NULL) + { + /* + * Nope - return a "not found" error... + */ + + send_ipp_status(con, IPP_NOT_FOUND, _("Job #%d does not exist."), jobid); + return; + } + + printer = cupsdFindDest(job->dest); + + /* + * See if the job is owned by the requesting user... + */ + + if (!validate_user(job, con, job->username, username, sizeof(username))) + { + send_http_error(con, con->username[0] ? HTTP_FORBIDDEN : HTTP_UNAUTHORIZED, + cupsdFindDest(job->dest)); + return; + } + + /* + * OK, see if the client is sending the document compressed - CUPS + * only supports "none" and "gzip". + */ + + compression = CUPS_FILE_NONE; + + if ((attr = ippFindAttribute(con->request, "compression", + IPP_TAG_KEYWORD)) != NULL) + { + if (strcmp(attr->values[0].string.text, "none") +#ifdef HAVE_LIBZ + && strcmp(attr->values[0].string.text, "gzip") +#endif /* HAVE_LIBZ */ + ) + { + send_ipp_status(con, IPP_ATTRIBUTES, _("Unsupported compression \"%s\"."), + attr->values[0].string.text); + ippAddString(con->response, IPP_TAG_UNSUPPORTED_GROUP, IPP_TAG_KEYWORD, + "compression", NULL, attr->values[0].string.text); + return; + } + +#ifdef HAVE_LIBZ + if (!strcmp(attr->values[0].string.text, "gzip")) + compression = CUPS_FILE_GZIP; +#endif /* HAVE_LIBZ */ + } + + /* + * Do we have a file to print? + */ + + if ((attr = ippFindAttribute(con->request, "last-document", + IPP_TAG_BOOLEAN)) == NULL) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Missing last-document attribute in request.")); + return; + } + + if (!con->filename) + { + /* + * Check for an empty request with "last-document" set to true, which is + * used to close an "open" job by RFC 2911, section 3.3.2. + */ + + if (job->num_files > 0 && attr->values[0].boolean) + goto last_document; + + send_ipp_status(con, IPP_BAD_REQUEST, _("No file in print request.")); + return; + } + + /* + * Is it a format we support? + */ + + if ((format = ippFindAttribute(con->request, "document-format", + IPP_TAG_MIMETYPE)) != NULL) + { + /* + * Grab format from client... + */ + + if (sscanf(format->values[0].string.text, "%15[^/]/%31[^;]", + super, type) != 2) + { + send_ipp_status(con, IPP_BAD_REQUEST, _("Bad document-format \"%s\"."), + format->values[0].string.text); + return; + } + } + else if ((default_format = cupsGetOption("document-format", + printer->num_options, + printer->options)) != NULL) + { + /* + * Use default document format... + */ + + if (sscanf(default_format, "%15[^/]/%31[^;]", super, type) != 2) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Bad document-format-default \"%s\"."), default_format); + return; + } + } + else + { + /* + * No document format attribute? Auto-type it! + */ + + strcpy(super, "application"); + strcpy(type, "octet-stream"); + } + + if (!strcmp(super, "application") && !strcmp(type, "octet-stream")) + { + /* + * Auto-type the file... + */ + + ipp_attribute_t *doc_name; /* document-name attribute */ + + + cupsdLogJob(job, CUPSD_LOG_DEBUG, "Auto-typing file..."); + + doc_name = ippFindAttribute(con->request, "document-name", IPP_TAG_NAME); + filetype = mimeFileType(MimeDatabase, con->filename, + doc_name ? doc_name->values[0].string.text : NULL, + &compression); + + if (!filetype) + filetype = mimeType(MimeDatabase, super, type); + + if (filetype) + cupsdLogJob(job, CUPSD_LOG_DEBUG, "Request file type is %s/%s.", + filetype->super, filetype->type); + } + else + filetype = mimeType(MimeDatabase, super, type); + + if (filetype) + { + /* + * Replace the document-format attribute value with the auto-typed or + * default one. + */ + + snprintf(mimetype, sizeof(mimetype), "%s/%s", filetype->super, + filetype->type); + + if ((jformat = ippFindAttribute(job->attrs, "document-format", + IPP_TAG_MIMETYPE)) != NULL) + { + _cupsStrFree(jformat->values[0].string.text); + + jformat->values[0].string.text = _cupsStrAlloc(mimetype); + } + else + ippAddString(job->attrs, IPP_TAG_JOB, IPP_TAG_MIMETYPE, + "document-format", NULL, mimetype); + } + else if (!filetype) + { + send_ipp_status(con, IPP_DOCUMENT_FORMAT, + _("Unsupported document-format \"%s/%s\"."), super, type); + cupsdLogMessage(CUPSD_LOG_INFO, + "Hint: Do you have the raw file printing rules enabled?"); + + if (format) + ippAddString(con->response, IPP_TAG_UNSUPPORTED_GROUP, IPP_TAG_MIMETYPE, + "document-format", NULL, format->values[0].string.text); + + return; + } + + if (printer->filetypes && !cupsArrayFind(printer->filetypes, filetype)) + { + snprintf(mimetype, sizeof(mimetype), "%s/%s", filetype->super, + filetype->type); + + send_ipp_status(con, IPP_DOCUMENT_FORMAT, + _("Unsupported document-format \"%s\"."), mimetype); + + ippAddString(con->response, IPP_TAG_UNSUPPORTED_GROUP, IPP_TAG_MIMETYPE, + "document-format", NULL, mimetype); + + return; + } + + /* + * Add the file to the job... + */ + + cupsdLoadJob(job); + + if (add_file(con, job, filetype, compression)) + return; + + if (stat(con->filename, &fileinfo)) + kbytes = 0; + else + kbytes = (fileinfo.st_size + 1023) / 1024; + + cupsdUpdateQuota(printer, job->username, 0, kbytes); + + if ((attr = ippFindAttribute(job->attrs, "job-k-octets", + IPP_TAG_INTEGER)) != NULL) + attr->values[0].integer += kbytes; + + snprintf(filename, sizeof(filename), "%s/d%05d-%03d", RequestRoot, job->id, + job->num_files); + rename(con->filename, filename); + + cupsdClearString(&con->filename); + + cupsdLogJob(job, CUPSD_LOG_INFO, "File of type %s/%s queued by \"%s\".", + filetype->super, filetype->type, job->username); + + /* + * Start the job if this is the last document... + */ + + last_document: + + if ((attr = ippFindAttribute(con->request, "last-document", + IPP_TAG_BOOLEAN)) != NULL && + attr->values[0].boolean) + { + /* + * See if we need to add the ending sheet... + */ + + if (cupsdTimeoutJob(job)) + return; + + if (job->state_value == IPP_JOB_STOPPED) + { + job->state->values[0].integer = IPP_JOB_PENDING; + job->state_value = IPP_JOB_PENDING; + + ippSetString(job->attrs, &job->reasons, 0, "none"); + } + else if (job->state_value == IPP_JOB_HELD) + { + if ((attr = ippFindAttribute(job->attrs, "job-hold-until", + IPP_TAG_KEYWORD)) == NULL) + attr = ippFindAttribute(job->attrs, "job-hold-until", IPP_TAG_NAME); + + if (!attr || !strcmp(attr->values[0].string.text, "no-hold")) + { + job->state->values[0].integer = IPP_JOB_PENDING; + job->state_value = IPP_JOB_PENDING; + + ippSetString(job->attrs, &job->reasons, 0, "none"); + } + else + ippSetString(job->attrs, &job->reasons, 0, "job-hold-until-specified"); + } + + job->dirty = 1; + cupsdMarkDirty(CUPSD_DIRTY_JOBS); + + start_job = 1; + } + else + { + if ((attr = ippFindAttribute(job->attrs, "job-hold-until", + IPP_TAG_KEYWORD)) == NULL) + attr = ippFindAttribute(job->attrs, "job-hold-until", IPP_TAG_NAME); + + if (!attr || !strcmp(attr->values[0].string.text, "no-hold")) + { + job->state->values[0].integer = IPP_JOB_HELD; + job->state_value = IPP_JOB_HELD; + job->hold_until = time(NULL) + MultipleOperationTimeout; + + ippSetString(job->attrs, &job->reasons, 0, "job-incoming"); + + job->dirty = 1; + cupsdMarkDirty(CUPSD_DIRTY_JOBS); + } + + start_job = 0; + } + + /* + * Fill in the response info... + */ + + httpAssembleURIf(HTTP_URI_CODING_ALL, job_uri, sizeof(job_uri), "ipp", NULL, + con->servername, con->serverport, "/jobs/%d", jobid); + ippAddString(con->response, IPP_TAG_JOB, IPP_TAG_URI, "job-uri", NULL, + job_uri); + + ippAddInteger(con->response, IPP_TAG_JOB, IPP_TAG_INTEGER, "job-id", jobid); + + ippAddInteger(con->response, IPP_TAG_JOB, IPP_TAG_ENUM, "job-state", + job->state_value); + ippAddString(con->response, IPP_TAG_JOB, IPP_TAG_KEYWORD, "job-state-reasons", + NULL, job->reasons->values[0].string.text); + + con->response->request.status.status_code = IPP_OK; + + /* + * Start the job if necessary... + */ + + if (start_job) + cupsdCheckJobs(); +} + + +/* + * 'send_http_error()' - Send a HTTP error back to the IPP client. + */ + +static void +send_http_error( + cupsd_client_t *con, /* I - Client connection */ + http_status_t status, /* I - HTTP status code */ + cupsd_printer_t *printer) /* I - Printer, if any */ +{ + ipp_attribute_t *uri; /* Request URI, if any */ + + + if ((uri = ippFindAttribute(con->request, "printer-uri", + IPP_TAG_URI)) == NULL) + uri = ippFindAttribute(con->request, "job-uri", IPP_TAG_URI); + + cupsdLogMessage(status == HTTP_FORBIDDEN ? CUPSD_LOG_ERROR : CUPSD_LOG_DEBUG, + "Returning HTTP %s for %s (%s) from %s", + httpStatus(status), + con->request ? + ippOpString(con->request->request.op.operation_id) : + "no operation-id", + uri ? uri->values[0].string.text : "no URI", + con->http.hostname); + + if (printer) + { + int auth_type; /* Type of authentication required */ + + + auth_type = CUPSD_AUTH_NONE; + + if (status == HTTP_UNAUTHORIZED && + printer->num_auth_info_required > 0 && + !strcmp(printer->auth_info_required[0], "negotiate") && + con->request && + (con->request->request.op.operation_id == IPP_PRINT_JOB || + con->request->request.op.operation_id == IPP_CREATE_JOB || + con->request->request.op.operation_id == CUPS_AUTHENTICATE_JOB)) + { + /* + * Creating and authenticating jobs requires Kerberos... + */ + + auth_type = CUPSD_AUTH_NEGOTIATE; + } + else + { + /* + * Use policy/location-defined authentication requirements... + */ + + char resource[HTTP_MAX_URI]; /* Resource portion of URI */ + cupsd_location_t *auth; /* Pointer to authentication element */ + + + if (printer->type & CUPS_PRINTER_CLASS) + snprintf(resource, sizeof(resource), "/classes/%s", printer->name); + else + snprintf(resource, sizeof(resource), "/printers/%s", printer->name); + + if ((auth = cupsdFindBest(resource, HTTP_POST)) == NULL || + auth->type == CUPSD_AUTH_NONE) + auth = cupsdFindPolicyOp(printer->op_policy_ptr, + con->request ? + con->request->request.op.operation_id : + IPP_PRINT_JOB); + + if (auth) + { + if (auth->type == CUPSD_AUTH_DEFAULT) + auth_type = cupsdDefaultAuthType(); + else + auth_type = auth->type; + } + } + + cupsdSendError(con, status, auth_type); + } + else + cupsdSendError(con, status, CUPSD_AUTH_NONE); + + ippDelete(con->response); + con->response = NULL; + + return; +} + + +/* + * 'send_ipp_status()' - Send a status back to the IPP client. + */ + +static void +send_ipp_status(cupsd_client_t *con, /* I - Client connection */ + ipp_status_t status, /* I - IPP status code */ + const char *message,/* I - Status message */ + ...) /* I - Additional args as needed */ +{ + va_list ap; /* Pointer to additional args */ + char formatted[1024]; /* Formatted errror message */ + + + va_start(ap, message); + vsnprintf(formatted, sizeof(formatted), + _cupsLangString(con->language, message), ap); + va_end(ap); + + cupsdLogMessage(CUPSD_LOG_DEBUG, "%s %s: %s", + ippOpString(con->request->request.op.operation_id), + ippErrorString(status), formatted); + + con->response->request.status.status_code = status; + + if (ippFindAttribute(con->response, "attributes-charset", + IPP_TAG_ZERO) == NULL) + ippAddString(con->response, IPP_TAG_OPERATION, IPP_TAG_CHARSET, + "attributes-charset", NULL, "utf-8"); + + if (ippFindAttribute(con->response, "attributes-natural-language", + IPP_TAG_ZERO) == NULL) + ippAddString(con->response, IPP_TAG_OPERATION, IPP_TAG_LANGUAGE, + "attributes-natural-language", NULL, DefaultLanguage); + + ippAddString(con->response, IPP_TAG_OPERATION, IPP_TAG_TEXT, + "status-message", NULL, formatted); +} + + +/* + * 'set_default()' - Set the default destination... + */ + +static void +set_default(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Printer URI */ +{ + http_status_t status; /* Policy status */ + cups_ptype_t dtype; /* Destination type (printer/class) */ + cupsd_printer_t *printer, /* Printer */ + *oldprinter; /* Old default printer */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "set_default(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * Is the destination valid? + */ + + if (!cupsdValidateDest(uri->values[0].string.text, &dtype, &printer)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(DefaultPolicyPtr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, NULL); + return; + } + + /* + * Set it as the default... + */ + + oldprinter = DefaultPrinter; + DefaultPrinter = printer; + + if (oldprinter) + cupsdAddEvent(CUPSD_EVENT_PRINTER_STATE, oldprinter, NULL, + "%s is no longer the default printer.", oldprinter->name); + + cupsdAddEvent(CUPSD_EVENT_PRINTER_STATE, printer, NULL, + "%s is now the default printer.", printer->name); + + cupsdMarkDirty(CUPSD_DIRTY_PRINTERS | CUPSD_DIRTY_CLASSES | + CUPSD_DIRTY_PRINTCAP); + + cupsdLogMessage(CUPSD_LOG_INFO, + "Default destination set to \"%s\" by \"%s\".", + printer->name, get_username(con)); + + /* + * Everything was ok, so return OK status... + */ + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'set_job_attrs()' - Set job attributes. + */ + +static void +set_job_attrs(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Job URI */ +{ + ipp_attribute_t *attr, /* Current attribute */ + *attr2; /* Job attribute */ + int jobid; /* Job ID */ + cupsd_job_t *job; /* Current job */ + char scheme[HTTP_MAX_URI], + /* Method portion of URI */ + username[HTTP_MAX_URI], + /* Username portion of URI */ + host[HTTP_MAX_URI], + /* Host portion of URI */ + resource[HTTP_MAX_URI]; + /* Resource portion of URI */ + int port; /* Port portion of URI */ + int event; /* Events? */ + int check_jobs; /* Check jobs? */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "set_job_attrs(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * Start with "everything is OK" status... + */ + + con->response->request.status.status_code = IPP_OK; + + /* + * See if we have a job URI or a printer URI... + */ + + if (!strcmp(uri->name, "printer-uri")) + { + /* + * Got a printer URI; see if we also have a job-id attribute... + */ + + if ((attr = ippFindAttribute(con->request, "job-id", + IPP_TAG_INTEGER)) == NULL) + { + send_ipp_status(con, IPP_BAD_REQUEST, + _("Got a printer-uri attribute but no job-id.")); + return; + } + + jobid = attr->values[0].integer; + } + else + { + /* + * Got a job URI; parse it to get the job ID... + */ + + httpSeparateURI(HTTP_URI_CODING_ALL, uri->values[0].string.text, scheme, + sizeof(scheme), username, sizeof(username), host, + sizeof(host), &port, resource, sizeof(resource)); + + if (strncmp(resource, "/jobs/", 6)) + { + /* + * Not a valid URI! + */ + + send_ipp_status(con, IPP_BAD_REQUEST, _("Bad job-uri \"%s\"."), + uri->values[0].string.text); + return; + } + + jobid = atoi(resource + 6); + } + + /* + * See if the job exists... + */ + + if ((job = cupsdFindJob(jobid)) == NULL) + { + /* + * Nope - return a "not found" error... + */ + + send_ipp_status(con, IPP_NOT_FOUND, _("Job #%d does not exist."), jobid); + return; + } + + /* + * See if the job has been completed... + */ + + if (job->state_value > IPP_JOB_STOPPED) + { + /* + * Return a "not-possible" error... + */ + + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Job #%d is finished and cannot be altered."), jobid); + return; + } + + /* + * See if the job is owned by the requesting user... + */ + + if (!validate_user(job, con, job->username, username, sizeof(username))) + { + send_http_error(con, con->username[0] ? HTTP_FORBIDDEN : HTTP_UNAUTHORIZED, + cupsdFindDest(job->dest)); + return; + } + + /* + * See what the user wants to change. + */ + + cupsdLoadJob(job); + + check_jobs = 0; + event = 0; + + for (attr = con->request->attrs; attr; attr = attr->next) + { + if (attr->group_tag != IPP_TAG_JOB || !attr->name) + continue; + + if (!strcmp(attr->name, "attributes-charset") || + !strcmp(attr->name, "attributes-natural-language") || + !strcmp(attr->name, "document-compression") || + !strcmp(attr->name, "document-format") || + !strcmp(attr->name, "job-detailed-status-messages") || + !strcmp(attr->name, "job-document-access-errors") || + !strcmp(attr->name, "job-id") || + !strcmp(attr->name, "job-impressions-completed") || + !strcmp(attr->name, "job-k-octets") || + !strcmp(attr->name, "job-originating-host-name") || + !strcmp(attr->name, "job-originating-user-name") || + !strcmp(attr->name, "job-printer-up-time") || + !strcmp(attr->name, "job-printer-uri") || + !strcmp(attr->name, "job-sheets") || + !strcmp(attr->name, "job-state-message") || + !strcmp(attr->name, "job-state-reasons") || + !strcmp(attr->name, "job-uri") || + !strcmp(attr->name, "number-of-documents") || + !strcmp(attr->name, "number-of-intervening-jobs") || + !strcmp(attr->name, "output-device-assigned") || + !strncmp(attr->name, "date-time-at-", 13) || + !strncmp(attr->name, "job-k-octets", 12) || + !strncmp(attr->name, "job-media-sheets", 16) || + !strncmp(attr->name, "time-at-", 8)) + { + /* + * Read-only attrs! + */ + + send_ipp_status(con, IPP_ATTRIBUTES_NOT_SETTABLE, + _("%s cannot be changed."), attr->name); + + attr2 = ippCopyAttribute(con->response, attr, 0); + ippSetGroupTag(con->response, &attr2, IPP_TAG_UNSUPPORTED_GROUP); + continue; + } + + if (!strcmp(attr->name, "job-priority")) + { + /* + * Change the job priority... + */ + + if (attr->value_tag != IPP_TAG_INTEGER) + { + send_ipp_status(con, IPP_REQUEST_VALUE, _("Bad job-priority value.")); + + attr2 = ippCopyAttribute(con->response, attr, 0); + ippSetGroupTag(con->response, &attr2, IPP_TAG_UNSUPPORTED_GROUP); + } + else if (job->state_value >= IPP_JOB_PROCESSING) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Job is completed and cannot be changed.")); + return; + } + else if (con->response->request.status.status_code == IPP_OK) + { + cupsdLogJob(job, CUPSD_LOG_DEBUG, "Setting job-priority to %d", + attr->values[0].integer); + cupsdSetJobPriority(job, attr->values[0].integer); + + check_jobs = 1; + event |= CUPSD_EVENT_JOB_CONFIG_CHANGED | + CUPSD_EVENT_PRINTER_QUEUE_ORDER_CHANGED; + } + } + else if (!strcmp(attr->name, "job-state")) + { + /* + * Change the job state... + */ + + if (attr->value_tag != IPP_TAG_ENUM) + { + send_ipp_status(con, IPP_REQUEST_VALUE, _("Bad job-state value.")); + + attr2 = ippCopyAttribute(con->response, attr, 0); + ippSetGroupTag(con->response, &attr2, IPP_TAG_UNSUPPORTED_GROUP); + } + else + { + switch (attr->values[0].integer) + { + case IPP_JOB_PENDING : + case IPP_JOB_HELD : + if (job->state_value > IPP_JOB_HELD) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Job state cannot be changed.")); + return; + } + else if (con->response->request.status.status_code == IPP_OK) + { + cupsdLogJob(job, CUPSD_LOG_DEBUG, "Setting job-state to %d", + attr->values[0].integer); + cupsdSetJobState(job, attr->values[0].integer, + CUPSD_JOB_DEFAULT, + "Job state changed by \"%s\"", username); + check_jobs = 1; + } + break; + + case IPP_JOB_PROCESSING : + case IPP_JOB_STOPPED : + if (job->state_value != attr->values[0].integer) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Job state cannot be changed.")); + return; + } + break; + + case IPP_JOB_CANCELED : + case IPP_JOB_ABORTED : + case IPP_JOB_COMPLETED : + if (job->state_value > IPP_JOB_PROCESSING) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Job state cannot be changed.")); + return; + } + else if (con->response->request.status.status_code == IPP_OK) + { + cupsdLogJob(job, CUPSD_LOG_DEBUG, "Setting job-state to %d", + attr->values[0].integer); + cupsdSetJobState(job, (ipp_jstate_t)attr->values[0].integer, + CUPSD_JOB_DEFAULT, + "Job state changed by \"%s\"", username); + check_jobs = 1; + } + break; + } + } + } + else if (con->response->request.status.status_code != IPP_OK) + continue; + else if ((attr2 = ippFindAttribute(job->attrs, attr->name, + IPP_TAG_ZERO)) != NULL) + { + /* + * Some other value; first free the old value... + */ + + if (job->attrs->prev) + job->attrs->prev->next = attr2->next; + else + job->attrs->attrs = attr2->next; + + if (job->attrs->last == attr2) + job->attrs->last = job->attrs->prev; + + ippDeleteAttribute(NULL, attr2); + + /* + * Then copy the attribute... + */ + + ippCopyAttribute(job->attrs, attr, 0); + + /* + * See if the job-name or job-hold-until is being changed. + */ + + if (!strcmp(attr->name, "job-hold-until")) + { + cupsdLogJob(job, CUPSD_LOG_DEBUG, "Setting job-hold-until to %s", + attr->values[0].string.text); + cupsdSetJobHoldUntil(job, attr->values[0].string.text, 0); + + if (!strcmp(attr->values[0].string.text, "no-hold")) + { + cupsdReleaseJob(job); + check_jobs = 1; + } + else + cupsdSetJobState(job, IPP_JOB_HELD, CUPSD_JOB_DEFAULT, + "Job held by \"%s\".", username); + + event |= CUPSD_EVENT_JOB_CONFIG_CHANGED | CUPSD_EVENT_JOB_STATE; + } + } + else if (attr->value_tag == IPP_TAG_DELETEATTR) + { + /* + * Delete the attribute... + */ + + if ((attr2 = ippFindAttribute(job->attrs, attr->name, + IPP_TAG_ZERO)) != NULL) + { + if (job->attrs->prev) + job->attrs->prev->next = attr2->next; + else + job->attrs->attrs = attr2->next; + + if (attr2 == job->attrs->last) + job->attrs->last = job->attrs->prev; + + ippDeleteAttribute(NULL, attr2); + + event |= CUPSD_EVENT_JOB_CONFIG_CHANGED; + } + } + else + { + /* + * Add new option by copying it... + */ + + ippCopyAttribute(job->attrs, attr, 0); + + event |= CUPSD_EVENT_JOB_CONFIG_CHANGED; + } + } + + /* + * Save the job... + */ + + job->dirty = 1; + cupsdMarkDirty(CUPSD_DIRTY_JOBS); + + /* + * Send events as needed... + */ + + if (event & CUPSD_EVENT_PRINTER_QUEUE_ORDER_CHANGED) + cupsdAddEvent(CUPSD_EVENT_PRINTER_QUEUE_ORDER_CHANGED, + cupsdFindDest(job->dest), job, + "Job priority changed by user."); + + if (event & CUPSD_EVENT_JOB_STATE) + cupsdAddEvent(CUPSD_EVENT_JOB_STATE, cupsdFindDest(job->dest), job, + job->state_value == IPP_JOB_HELD ? + "Job held by user." : "Job restarted by user."); + + if (event & CUPSD_EVENT_JOB_CONFIG_CHANGED) + cupsdAddEvent(CUPSD_EVENT_JOB_CONFIG_CHANGED, cupsdFindDest(job->dest), job, + "Job options changed by user."); + + /* + * Start jobs if possible... + */ + + if (check_jobs) + cupsdCheckJobs(); +} + + +/* + * 'set_printer_attrs()' - Set printer attributes. + */ + +static void +set_printer_attrs(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Printer */ +{ + http_status_t status; /* Policy status */ + cups_ptype_t dtype; /* Destination type (printer/class) */ + cupsd_printer_t *printer; /* Printer/class */ + ipp_attribute_t *attr; /* Printer attribute */ + int changed = 0; /* Was anything changed? */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "set_printer_attrs(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * Is the destination valid? + */ + + if (!cupsdValidateDest(uri->values[0].string.text, &dtype, &printer)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(printer->op_policy_ptr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, printer); + return; + } + + /* + * Return a list of attributes that can be set via Set-Printer-Attributes. + */ + + if ((attr = ippFindAttribute(con->request, "printer-location", + IPP_TAG_TEXT)) != NULL) + { + cupsdSetString(&printer->location, attr->values[0].string.text); + changed = 1; + } + + if ((attr = ippFindAttribute(con->request, "printer-info", + IPP_TAG_TEXT)) != NULL) + { + cupsdSetString(&printer->info, attr->values[0].string.text); + changed = 1; + } + + /* + * Update the printer attributes and return... + */ + + if (changed) + { + cupsdSetPrinterAttrs(printer); + cupsdMarkDirty(CUPSD_DIRTY_PRINTERS); + + cupsdAddEvent(CUPSD_EVENT_PRINTER_CONFIG, printer, NULL, + "Printer \"%s\" description or location changed by \"%s\".", + printer->name, get_username(con)); + + cupsdLogMessage(CUPSD_LOG_INFO, + "Printer \"%s\" description or location changed by \"%s\".", + printer->name, get_username(con)); + } + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'set_printer_defaults()' - Set printer default options from a request. + */ + +static void +set_printer_defaults( + cupsd_client_t *con, /* I - Client connection */ + cupsd_printer_t *printer) /* I - Printer */ +{ + int i; /* Looping var */ + ipp_attribute_t *attr; /* Current attribute */ + int namelen; /* Length of attribute name */ + char name[256], /* New attribute name */ + value[256]; /* String version of integer attrs */ + + + for (attr = con->request->attrs; attr; attr = attr->next) + { + /* + * Skip non-printer attributes... + */ + + if (attr->group_tag != IPP_TAG_PRINTER || !attr->name) + continue; + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "set_printer_defaults: %s", attr->name); + + if (!strcmp(attr->name, "job-sheets-default")) + { + /* + * Only allow keywords and names... + */ + + if (attr->value_tag != IPP_TAG_NAME && attr->value_tag != IPP_TAG_KEYWORD) + continue; + + /* + * Only allow job-sheets-default to be set when running without a + * system high classification level... + */ + + if (Classification) + continue; + + cupsdSetString(&printer->job_sheets[0], attr->values[0].string.text); + + if (attr->num_values > 1) + cupsdSetString(&printer->job_sheets[1], attr->values[1].string.text); + else + cupsdSetString(&printer->job_sheets[1], "none"); + } + else if (!strcmp(attr->name, "requesting-user-name-allowed")) + { + cupsdFreeStrings(&(printer->users)); + + printer->deny_users = 0; + + if (attr->value_tag == IPP_TAG_NAME && + (attr->num_values > 1 || + strcmp(attr->values[0].string.text, "all"))) + { + for (i = 0; i < attr->num_values; i ++) + cupsdAddString(&(printer->users), attr->values[i].string.text); + } + } + else if (!strcmp(attr->name, "requesting-user-name-denied")) + { + cupsdFreeStrings(&(printer->users)); + + printer->deny_users = 1; + + if (attr->value_tag == IPP_TAG_NAME && + (attr->num_values > 1 || + strcmp(attr->values[0].string.text, "none"))) + { + for (i = 0; i < attr->num_values; i ++) + cupsdAddString(&(printer->users), attr->values[i].string.text); + } + } + else if (!strcmp(attr->name, "job-quota-period")) + { + if (attr->value_tag != IPP_TAG_INTEGER) + continue; + + cupsdLogMessage(CUPSD_LOG_DEBUG, "Setting job-quota-period to %d...", + attr->values[0].integer); + cupsdFreeQuotas(printer); + + printer->quota_period = attr->values[0].integer; + } + else if (!strcmp(attr->name, "job-k-limit")) + { + if (attr->value_tag != IPP_TAG_INTEGER) + continue; + + cupsdLogMessage(CUPSD_LOG_DEBUG, "Setting job-k-limit to %d...", + attr->values[0].integer); + cupsdFreeQuotas(printer); + + printer->k_limit = attr->values[0].integer; + } + else if (!strcmp(attr->name, "job-page-limit")) + { + if (attr->value_tag != IPP_TAG_INTEGER) + continue; + + cupsdLogMessage(CUPSD_LOG_DEBUG, "Setting job-page-limit to %d...", + attr->values[0].integer); + cupsdFreeQuotas(printer); + + printer->page_limit = attr->values[0].integer; + } + else if (!strcmp(attr->name, "printer-op-policy")) + { + cupsd_policy_t *p; /* Policy */ + + + if (attr->value_tag != IPP_TAG_NAME) + continue; + + if ((p = cupsdFindPolicy(attr->values[0].string.text)) != NULL) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Setting printer-op-policy to \"%s\"...", + attr->values[0].string.text); + cupsdSetString(&printer->op_policy, attr->values[0].string.text); + printer->op_policy_ptr = p; + } + else + { + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Unknown printer-op-policy \"%s\"."), + attr->values[0].string.text); + return; + } + } + else if (!strcmp(attr->name, "printer-error-policy")) + { + if (attr->value_tag != IPP_TAG_NAME && attr->value_tag != IPP_TAG_KEYWORD) + continue; + + if (strcmp(attr->values[0].string.text, "retry-current-job") && + ((printer->type & CUPS_PRINTER_CLASS) || + (strcmp(attr->values[0].string.text, "abort-job") && + strcmp(attr->values[0].string.text, "retry-job") && + strcmp(attr->values[0].string.text, "stop-printer")))) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, + _("Unknown printer-error-policy \"%s\"."), + attr->values[0].string.text); + return; + } + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Setting printer-error-policy to \"%s\"...", + attr->values[0].string.text); + cupsdSetString(&printer->error_policy, attr->values[0].string.text); + } + + /* + * Skip any other non-default attributes... + */ + + namelen = strlen(attr->name); + if (namelen < 9 || strcmp(attr->name + namelen - 8, "-default") || + namelen > (sizeof(name) - 1) || attr->num_values != 1) + continue; + + /* + * OK, anything else must be a user-defined default... + */ + + strlcpy(name, attr->name, sizeof(name)); + name[namelen - 8] = '\0'; /* Strip "-default" */ + + switch (attr->value_tag) + { + case IPP_TAG_DELETEATTR : + printer->num_options = cupsRemoveOption(name, + printer->num_options, + &(printer->options)); + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Deleting %s", attr->name); + break; + + case IPP_TAG_NAME : + case IPP_TAG_KEYWORD : + case IPP_TAG_URI : + printer->num_options = cupsAddOption(name, + attr->values[0].string.text, + printer->num_options, + &(printer->options)); + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Setting %s to \"%s\"...", attr->name, + attr->values[0].string.text); + break; + + case IPP_TAG_BOOLEAN : + printer->num_options = cupsAddOption(name, + attr->values[0].boolean ? + "true" : "false", + printer->num_options, + &(printer->options)); + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Setting %s to %s...", attr->name, + attr->values[0].boolean ? "true" : "false"); + break; + + case IPP_TAG_INTEGER : + case IPP_TAG_ENUM : + sprintf(value, "%d", attr->values[0].integer); + printer->num_options = cupsAddOption(name, value, + printer->num_options, + &(printer->options)); + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Setting %s to %s...", attr->name, value); + break; + + case IPP_TAG_RANGE : + sprintf(value, "%d-%d", attr->values[0].range.lower, + attr->values[0].range.upper); + printer->num_options = cupsAddOption(name, value, + printer->num_options, + &(printer->options)); + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Setting %s to %s...", attr->name, value); + break; + + case IPP_TAG_RESOLUTION : + sprintf(value, "%dx%d%s", attr->values[0].resolution.xres, + attr->values[0].resolution.yres, + attr->values[0].resolution.units == IPP_RES_PER_INCH ? + "dpi" : "dpc"); + printer->num_options = cupsAddOption(name, value, + printer->num_options, + &(printer->options)); + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Setting %s to %s...", attr->name, value); + break; + + default : + /* Do nothing for other values */ + break; + } + } +} + + +/* + * 'start_printer()' - Start a printer. + */ + +static void +start_printer(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Printer URI */ +{ + int i; /* Temporary variable */ + http_status_t status; /* Policy status */ + cups_ptype_t dtype; /* Destination type (printer/class) */ + cupsd_printer_t *printer; /* Printer data */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "start_printer(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * Is the destination valid? + */ + + if (!cupsdValidateDest(uri->values[0].string.text, &dtype, &printer)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(printer->op_policy_ptr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, printer); + return; + } + + /* + * Start the printer... + */ + + printer->state_message[0] = '\0'; + + cupsdStartPrinter(printer, 1); + + if (dtype & CUPS_PRINTER_CLASS) + cupsdLogMessage(CUPSD_LOG_INFO, "Class \"%s\" started by \"%s\".", + printer->name, get_username(con)); + else + cupsdLogMessage(CUPSD_LOG_INFO, "Printer \"%s\" started by \"%s\".", + printer->name, get_username(con)); + + cupsdCheckJobs(); + + /* + * Check quotas... + */ + + if ((i = check_quotas(con, printer)) < 0) + { + send_ipp_status(con, IPP_NOT_POSSIBLE, _("Quota limit reached.")); + return; + } + else if (i == 0) + { + send_ipp_status(con, IPP_NOT_AUTHORIZED, _("Not allowed to print.")); + return; + } + + /* + * Everything was ok, so return OK status... + */ + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'stop_printer()' - Stop a printer. + */ + +static void +stop_printer(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Printer URI */ +{ + http_status_t status; /* Policy status */ + cups_ptype_t dtype; /* Destination type (printer/class) */ + cupsd_printer_t *printer; /* Printer data */ + ipp_attribute_t *attr; /* printer-state-message attribute */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "stop_printer(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * Is the destination valid? + */ + + if (!cupsdValidateDest(uri->values[0].string.text, &dtype, &printer)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * Check policy... + */ + + if ((status = cupsdCheckPolicy(printer->op_policy_ptr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, printer); + return; + } + + /* + * Stop the printer... + */ + + if ((attr = ippFindAttribute(con->request, "printer-state-message", + IPP_TAG_TEXT)) == NULL) + strcpy(printer->state_message, "Paused"); + else + { + strlcpy(printer->state_message, attr->values[0].string.text, + sizeof(printer->state_message)); + } + + cupsdStopPrinter(printer, 1); + + if (dtype & CUPS_PRINTER_CLASS) + cupsdLogMessage(CUPSD_LOG_INFO, "Class \"%s\" stopped by \"%s\".", + printer->name, get_username(con)); + else + cupsdLogMessage(CUPSD_LOG_INFO, "Printer \"%s\" stopped by \"%s\".", + printer->name, get_username(con)); + + /* + * Everything was ok, so return OK status... + */ + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'url_encode_attr()' - URL-encode a string attribute. + */ + +static void +url_encode_attr(ipp_attribute_t *attr, /* I - Attribute */ + char *buffer,/* I - String buffer */ + int bufsize)/* I - Size of buffer */ +{ + int i; /* Looping var */ + char *bufptr, /* Pointer into buffer */ + *bufend; /* End of buffer */ + + + strlcpy(buffer, attr->name, bufsize); + bufptr = buffer + strlen(buffer); + bufend = buffer + bufsize - 1; + + for (i = 0; i < attr->num_values; i ++) + { + if (bufptr >= bufend) + break; + + if (i) + *bufptr++ = ','; + else + *bufptr++ = '='; + + if (bufptr >= bufend) + break; + + *bufptr++ = '\''; + + bufptr = url_encode_string(attr->values[i].string.text, + bufptr, bufend - bufptr + 1); + + if (bufptr >= bufend) + break; + + *bufptr++ = '\''; + } + + *bufptr = '\0'; +} + + +/* + * 'url_encode_string()' - URL-encode a string. + */ + +static char * /* O - End of string */ +url_encode_string(const char *s, /* I - String */ + char *buffer, /* I - String buffer */ + int bufsize) /* I - Size of buffer */ +{ + char *bufptr, /* Pointer into buffer */ + *bufend; /* End of buffer */ + static const char *hex = "0123456789ABCDEF"; + /* Hex digits */ + + + bufptr = buffer; + bufend = buffer + bufsize - 1; + + while (*s && bufptr < bufend) + { + if (*s == ' ' || *s == '%' || *s == '+') + { + if (bufptr >= (bufend - 2)) + break; + + *bufptr++ = '%'; + *bufptr++ = hex[(*s >> 4) & 15]; + *bufptr++ = hex[*s & 15]; + + s ++; + } + else if (*s == '\'' || *s == '\\') + { + if (bufptr >= (bufend - 1)) + break; + + *bufptr++ = '\\'; + *bufptr++ = *s++; + } + else + *bufptr++ = *s++; + } + + *bufptr = '\0'; + + return (bufptr); +} + + +/* + * 'user_allowed()' - See if a user is allowed to print to a queue. + */ + +static int /* O - 0 if not allowed, 1 if allowed */ +user_allowed(cupsd_printer_t *p, /* I - Printer or class */ + const char *username) /* I - Username */ +{ + struct passwd *pw; /* User password data */ + char baseuser[256], /* Base username */ + *baseptr, /* Pointer to "@" in base username */ + *name; /* Current user name */ + + + if (cupsArrayCount(p->users) == 0) + return (1); + + if (!strcmp(username, "root")) + return (1); + + if (strchr(username, '@')) + { + /* + * Strip @REALM for username check... + */ + + strlcpy(baseuser, username, sizeof(baseuser)); + + if ((baseptr = strchr(baseuser, '@')) != NULL) + *baseptr = '\0'; + + username = baseuser; + } + + pw = getpwnam(username); + endpwent(); + + for (name = (char *)cupsArrayFirst(p->users); + name; + name = (char *)cupsArrayNext(p->users)) + { + if (name[0] == '@') + { + /* + * Check group membership... + */ + + if (cupsdCheckGroup(username, pw, name + 1)) + break; + } + else if (name[0] == '#') + { + /* + * Check UUID... + */ + + if (cupsdCheckGroup(username, pw, name)) + break; + } + else if (!_cups_strcasecmp(username, name)) + break; + } + + return ((name != NULL) != p->deny_users); +} + + +/* + * 'validate_job()' - Validate printer options and destination. + */ + +static void +validate_job(cupsd_client_t *con, /* I - Client connection */ + ipp_attribute_t *uri) /* I - Printer URI */ +{ + http_status_t status; /* Policy status */ + ipp_attribute_t *attr, /* Current attribute */ + *auth_info; /* auth-info attribute */ + ipp_attribute_t *format; /* Document-format attribute */ + cups_ptype_t dtype; /* Destination type (printer/class) */ + char super[MIME_MAX_SUPER], + /* Supertype of file */ + type[MIME_MAX_TYPE]; + /* Subtype of file */ + cupsd_printer_t *printer; /* Printer */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "validate_job(%p[%d], %s)", con, + con->http.fd, uri->values[0].string.text); + + /* + * OK, see if the client is sending the document compressed - CUPS + * doesn't support compression yet... + */ + + if ((attr = ippFindAttribute(con->request, "compression", + IPP_TAG_KEYWORD)) != NULL) + { + if (strcmp(attr->values[0].string.text, "none") +#ifdef HAVE_LIBZ + && strcmp(attr->values[0].string.text, "gzip") +#endif /* HAVE_LIBZ */ + ) + { + send_ipp_status(con, IPP_ATTRIBUTES, + _("Unsupported compression \"%s\"."), + attr->values[0].string.text); + ippAddString(con->response, IPP_TAG_UNSUPPORTED_GROUP, IPP_TAG_KEYWORD, + "compression", NULL, attr->values[0].string.text); + return; + } + } + + /* + * Is it a format we support? + */ + + if ((format = ippFindAttribute(con->request, "document-format", + IPP_TAG_MIMETYPE)) != NULL) + { + if (sscanf(format->values[0].string.text, "%15[^/]/%31[^;]", + super, type) != 2) + { + send_ipp_status(con, IPP_BAD_REQUEST, _("Bad document-format \"%s\"."), + format->values[0].string.text); + return; + } + + if ((strcmp(super, "application") || strcmp(type, "octet-stream")) && + !mimeType(MimeDatabase, super, type)) + { + cupsdLogMessage(CUPSD_LOG_INFO, + "Hint: Do you have the raw file printing rules enabled?"); + send_ipp_status(con, IPP_DOCUMENT_FORMAT, + _("Unsupported document-format \"%s\"."), + format->values[0].string.text); + ippAddString(con->response, IPP_TAG_UNSUPPORTED_GROUP, IPP_TAG_MIMETYPE, + "document-format", NULL, format->values[0].string.text); + return; + } + } + + /* + * Is the destination valid? + */ + + if (!cupsdValidateDest(uri->values[0].string.text, &dtype, &printer)) + { + /* + * Bad URI... + */ + + send_ipp_status(con, IPP_NOT_FOUND, + _("The printer or class does not exist.")); + return; + } + + /* + * Check policy... + */ + + auth_info = ippFindAttribute(con->request, "auth-info", IPP_TAG_TEXT); + + if ((status = cupsdCheckPolicy(printer->op_policy_ptr, con, NULL)) != HTTP_OK) + { + send_http_error(con, status, printer); + return; + } + else if (printer->num_auth_info_required == 1 && + !strcmp(printer->auth_info_required[0], "negotiate") && + !con->username[0]) + { + send_http_error(con, HTTP_UNAUTHORIZED, printer); + return; + } +#ifdef HAVE_SSL + else if (auth_info && !con->http.tls && + !httpAddrLocalhost(con->http.hostaddr)) + { + /* + * Require encryption of auth-info over non-local connections... + */ + + send_http_error(con, HTTP_UPGRADE_REQUIRED, printer); + return; + } +#endif /* HAVE_SSL */ + + /* + * Everything was ok, so return OK status... + */ + + con->response->request.status.status_code = IPP_OK; +} + + +/* + * 'validate_name()' - Make sure the printer name only contains valid chars. + */ + +static int /* O - 0 if name is no good, 1 if good */ +validate_name(const char *name) /* I - Name to check */ +{ + const char *ptr; /* Pointer into name */ + + + /* + * Scan the whole name... + */ + + for (ptr = name; *ptr; ptr ++) + if ((*ptr > 0 && *ptr <= ' ') || *ptr == 127 || *ptr == '/' || *ptr == '#') + return (0); + + /* + * All the characters are good; validate the length, too... + */ + + return ((ptr - name) < 128); +} + + +/* + * 'validate_user()' - Validate the user for the request. + */ + +static int /* O - 1 if permitted, 0 otherwise */ +validate_user(cupsd_job_t *job, /* I - Job */ + cupsd_client_t *con, /* I - Client connection */ + const char *owner, /* I - Owner of job/resource */ + char *username, /* O - Authenticated username */ + int userlen) /* I - Length of username */ +{ + cupsd_printer_t *printer; /* Printer for job */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "validate_user(job=%d, con=%d, owner=\"%s\", username=%p, " + "userlen=%d)", + job->id, con ? con->http.fd : 0, + owner ? owner : "(null)", username, userlen); + + /* + * Validate input... + */ + + if (!con || !owner || !username || userlen <= 0) + return (0); + + /* + * Get the best authenticated username that is available. + */ + + strlcpy(username, get_username(con), userlen); + + /* + * Check the username against the owner... + */ + + printer = cupsdFindDest(job->dest); + + return (cupsdCheckPolicy(printer ? printer->op_policy_ptr : DefaultPolicyPtr, + con, owner) == HTTP_OK); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/job.c b/scheduler/job.c new file mode 100644 index 0000000000..5d46a39c67 --- /dev/null +++ b/scheduler/job.c @@ -0,0 +1,4930 @@ +/* + * "$Id$" + * + * Job management routines for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdAddJob() - Add a new job to the job queue. + * cupsdCancelJobs() - Cancel all jobs for the given + * destination/user. + * cupsdCheckJobs() - Check the pending jobs and start any if the + * destination is available. + * cupsdCleanJobs() - Clean out old jobs. + * cupsdContinueJob() - Continue printing with the next file in a job. + * cupsdDeleteJob() - Free all memory used by a job. + * cupsdFreeAllJobs() - Free all jobs from memory. + * cupsdFindJob() - Find the specified job. + * cupsdGetPrinterJobCount() - Get the number of pending, processing, or held + * jobs in a printer or class. + * cupsdGetUserJobCount() - Get the number of pending, processing, or held + * jobs for a user. + * cupsdLoadAllJobs() - Load all jobs from disk. + * cupsdLoadJob() - Load a single job. + * cupsdMoveJob() - Move the specified job to a different + * destination. + * cupsdReleaseJob() - Release the specified job. + * cupsdRestartJob() - Restart the specified job. + * cupsdSaveAllJobs() - Save a summary of all jobs to disk. + * cupsdSaveJob() - Save a job to disk. + * cupsdSetJobHoldUntil() - Set the hold time for a job. + * cupsdSetJobPriority() - Set the priority of a job, moving it up/down + * in the list as needed. + * cupsdSetJobState() - Set the state of the specified print job. + * cupsdStopAllJobs() - Stop all print jobs. + * cupsdUnloadCompletedJobs() - Flush completed job history from memory. + * compare_active_jobs() - Compare the job IDs and priorities of two + * jobs. + * compare_jobs() - Compare the job IDs of two jobs. + * dump_job_history() - Dump any debug messages for a job. + * free_job_history() - Free any log history. + * finalize_job() - Cleanup after job filter processes and support + * data. + * get_options() - Get a string containing the job options. + * ipp_length() - Compute the size of the buffer needed to hold + * the textual IPP attributes. + * load_job_cache() - Load jobs from the job.cache file. + * load_next_job_id() - Load the NextJobId value from the job.cache + * file. + * load_request_root() - Load jobs from the RequestRoot directory. + * set_time() - Set one of the "time-at-xyz" attributes. + * start_job() - Start a print job. + * stop_job() - Stop a print job. + * unload_job() - Unload a job from memory. + * update_job() - Read a status update from a job's filters. + * update_job_attrs() - Update the job-printer-* attributes. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" +#include +#include +#include +#ifdef __APPLE__ +# include +# ifdef HAVE_IOKIT_PWR_MGT_IOPMLIBPRIVATE_H +# include +# endif /* HAVE_IOKIT_PWR_MGT_IOPMLIBPRIVATE_H */ +#endif /* __APPLE__ */ + + +/* + * Design Notes for Job Management + * ------------------------------- + * + * STATE CHANGES + * + * pending Do nothing/check jobs + * pending-held Send SIGTERM to filters and backend + * processing Do nothing/start job + * stopped Send SIGKILL to filters and backend + * canceled Send SIGTERM to filters and backend + * aborted Finalize + * completed Finalize + * + * Finalize clears the printer <-> job association, deletes the status + * buffer, closes all of the pipes, etc. and doesn't get run until all of + * the print processes are finished. + * + * UNLOADING OF JOBS (cupsdUnloadCompletedJobs) + * + * We unload the job attributes when they are not needed to reduce overall + * memory consumption. We don't unload jobs where job->state_value < + * IPP_JOB_STOPPED, job->printer != NULL, or job->access_time is recent. + * + * STARTING OF JOBS (start_job) + * + * When a job is started, a status buffer, several pipes, a security + * profile, and a backend process are created for the life of that job. + * These are shared for every file in a job. For remote print jobs, the + * IPP backend is provided with every file in the job and no filters are + * run. + * + * The job->printer member tracks which printer is printing a job, which + * can be different than the destination in job->dest for classes. The + * printer object also has a job pointer to track which job is being + * printed. + * + * PRINTING OF JOB FILES (cupsdContinueJob) + * + * Each file in a job is filtered by 0 or more programs. After getting the + * list of filters needed and the total cost, the job is either passed or + * put back to the processing state until the current FilterLevel comes down + * enough to allow printing. + * + * If we can print, we build a string for the print options and run each of + * the filters, piping the output from one into the next. + * + * JOB STATUS UPDATES (update_job) + * + * The update_job function gets called whenever there are pending messages + * on the status pipe. These generally are updates to the marker-*, + * printer-state-message, or printer-state-reasons attributes. On EOF, + * finalize_job is called to clean up. + * + * FINALIZING JOBS (finalize_job) + * + * When all filters and the backend are done, we set the job state to + * completed (no errors), aborted (filter errors or abort-job policy), + * pending-held (auth required or retry-job policy), or pending + * (retry-current-job or stop-printer policies) as appropriate. + * + * Then we close the pipes and free the status buffers and profiles. + * + * JOB FILE COMPLETION (process_children in main.c) + * + * For multiple-file jobs, process_children (in main.c) sees that all + * filters have exited and calls in to print the next file if there are + * more files in the job, otherwise it waits for the backend to exit and + * update_job to do the cleanup. + */ + + +/* + * Local globals... + */ + +static mime_filter_t gziptoany_filter = + { + NULL, /* Source type */ + NULL, /* Destination type */ + 0, /* Cost */ + "gziptoany" /* Filter program to run */ + }; + + +/* + * Local functions... + */ + +static int compare_active_jobs(void *first, void *second, void *data); +static int compare_jobs(void *first, void *second, void *data); +static void dump_job_history(cupsd_job_t *job); +static void finalize_job(cupsd_job_t *job, int set_job_state); +static void free_job_history(cupsd_job_t *job); +static char *get_options(cupsd_job_t *job, int banner_page, char *copies, + size_t copies_size, char *title, + size_t title_size); +static size_t ipp_length(ipp_t *ipp); +static void load_job_cache(const char *filename); +static void load_next_job_id(const char *filename); +static void load_request_root(void); +static void set_time(cupsd_job_t *job, const char *name); +static void start_job(cupsd_job_t *job, cupsd_printer_t *printer); +static void stop_job(cupsd_job_t *job, cupsd_jobaction_t action); +static void unload_job(cupsd_job_t *job); +static void update_job(cupsd_job_t *job); +static void update_job_attrs(cupsd_job_t *job, int do_message); + + +/* + * 'cupsdAddJob()' - Add a new job to the job queue. + */ + +cupsd_job_t * /* O - New job record */ +cupsdAddJob(int priority, /* I - Job priority */ + const char *dest) /* I - Job destination */ +{ + cupsd_job_t *job; /* New job record */ + + + if ((job = calloc(sizeof(cupsd_job_t), 1)) == NULL) + return (NULL); + + job->id = NextJobId ++; + job->priority = priority; + job->back_pipes[0] = -1; + job->back_pipes[1] = -1; + job->print_pipes[0] = -1; + job->print_pipes[1] = -1; + job->side_pipes[0] = -1; + job->side_pipes[1] = -1; + job->status_pipes[0] = -1; + job->status_pipes[1] = -1; + + cupsdSetString(&job->dest, dest); + + /* + * Add the new job to the "all jobs" and "active jobs" lists... + */ + + cupsArrayAdd(Jobs, job); + cupsArrayAdd(ActiveJobs, job); + + return (job); +} + + +/* + * 'cupsdCancelJobs()' - Cancel all jobs for the given destination/user. + */ + +void +cupsdCancelJobs(const char *dest, /* I - Destination to cancel */ + const char *username, /* I - Username or NULL */ + int purge) /* I - Purge jobs? */ +{ + cupsd_job_t *job; /* Current job */ + + + for (job = (cupsd_job_t *)cupsArrayFirst(Jobs); + job; + job = (cupsd_job_t *)cupsArrayNext(Jobs)) + { + if ((!job->dest || !job->username) && !cupsdLoadJob(job)) + continue; + + if ((!dest || !strcmp(job->dest, dest)) && + (!username || !strcmp(job->username, username))) + { + /* + * Cancel all jobs matching this destination/user... + */ + + if (purge) + cupsdSetJobState(job, IPP_JOB_CANCELED, CUPSD_JOB_PURGE, + "Job purged by user."); + else if (job->state_value < IPP_JOB_CANCELED) + cupsdSetJobState(job, IPP_JOB_CANCELED, CUPSD_JOB_DEFAULT, + "Job canceled by user."); + } + } + + cupsdCheckJobs(); +} + + +/* + * 'cupsdCheckJobs()' - Check the pending jobs and start any if the destination + * is available. + */ + +void +cupsdCheckJobs(void) +{ + cupsd_job_t *job; /* Current job in queue */ + cupsd_printer_t *printer, /* Printer destination */ + *pclass; /* Printer class destination */ + ipp_attribute_t *attr; /* Job attribute */ + time_t curtime; /* Current time */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdCheckJobs: %d active jobs, sleeping=%d, reload=%d", + cupsArrayCount(ActiveJobs), Sleeping, NeedReload); + + curtime = time(NULL); + + for (job = (cupsd_job_t *)cupsArrayFirst(ActiveJobs); + job; + job = (cupsd_job_t *)cupsArrayNext(ActiveJobs)) + { + /* + * Kill jobs if they are unresponsive... + */ + + if (job->kill_time && job->kill_time <= curtime) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "[Job %d] Stopping unresponsive job.", + job->id); + + stop_job(job, CUPSD_JOB_FORCE); + continue; + } + + /* + * Cancel stuck jobs... + */ + + if (job->cancel_time && job->cancel_time <= curtime) + { + cupsdSetJobState(job, IPP_JOB_CANCELED, CUPSD_JOB_DEFAULT, + "Canceling stuck job after %d seconds.", MaxJobTime); + continue; + } + + /* + * Start held jobs if they are ready... + */ + + if (job->state_value == IPP_JOB_HELD && + job->hold_until && + job->hold_until < curtime) + { + if (job->pending_timeout) + { + /* + * This job is pending; check that we don't have an active Send-Document + * operation in progress on any of the client connections, then timeout + * the job so we can start printing... + */ + + cupsd_client_t *con; /* Current client connection */ + + + for (con = (cupsd_client_t *)cupsArrayFirst(Clients); + con; + con = (cupsd_client_t *)cupsArrayNext(Clients)) + if (con->request && + con->request->request.op.operation_id == IPP_SEND_DOCUMENT) + break; + + if (con) + continue; + + if (cupsdTimeoutJob(job)) + continue; + } + + cupsdSetJobState(job, IPP_JOB_PENDING, CUPSD_JOB_DEFAULT, + "Job submission timed out."); + } + + /* + * Continue jobs that are waiting on the FilterLimit... + */ + + if (job->pending_cost > 0 && + ((FilterLevel + job->pending_cost) < FilterLimit || FilterLevel == 0)) + cupsdContinueJob(job); + + /* + * Start pending jobs if the destination is available... + */ + + if (job->state_value == IPP_JOB_PENDING && !NeedReload && +#ifndef kIOPMAssertionTypeDenySystemSleep + !Sleeping && +#endif /* !kIOPMAssertionTypeDenySystemSleep */ + !DoingShutdown && !job->printer) + { + printer = cupsdFindDest(job->dest); + pclass = NULL; + + while (printer && (printer->type & CUPS_PRINTER_CLASS)) + { + /* + * If the class is remote, just pass it to the remote server... + */ + + pclass = printer; + + if (pclass->state == IPP_PRINTER_STOPPED) + printer = NULL; + else if (pclass->type & CUPS_PRINTER_REMOTE) + break; + else + printer = cupsdFindAvailablePrinter(printer->name); + } + + if (!printer && !pclass) + { + /* + * Whoa, the printer and/or class for this destination went away; + * cancel the job... + */ + + cupsdSetJobState(job, IPP_JOB_ABORTED, CUPSD_JOB_PURGE, + "Job aborted because the destination printer/class " + "has gone away."); + } + else if (printer && !printer->holding_new_jobs) + { + /* + * See if the printer is available or remote and not printing a job; + * if so, start the job... + */ + + if (pclass) + { + /* + * Add/update a job-actual-printer-uri attribute for this job + * so that we know which printer actually printed the job... + */ + + if ((attr = ippFindAttribute(job->attrs, "job-actual-printer-uri", + IPP_TAG_URI)) != NULL) + cupsdSetString(&attr->values[0].string.text, printer->uri); + else + ippAddString(job->attrs, IPP_TAG_JOB, IPP_TAG_URI, + "job-actual-printer-uri", NULL, printer->uri); + + job->dirty = 1; + cupsdMarkDirty(CUPSD_DIRTY_JOBS); + } + + if (printer->state == IPP_PRINTER_IDLE) + { + /* + * Start the job... + */ + + start_job(job, printer); + } + } + } + } +} + + +/* + * 'cupsdCleanJobs()' - Clean out old jobs. + */ + +void +cupsdCleanJobs(void) +{ + cupsd_job_t *job; /* Current job */ + + + if (MaxJobs <= 0 && JobHistory) + return; + + for (job = (cupsd_job_t *)cupsArrayFirst(Jobs); + job && (cupsArrayCount(Jobs) >= MaxJobs || !JobHistory); + job = (cupsd_job_t *)cupsArrayNext(Jobs)) + if (job->state_value >= IPP_JOB_CANCELED && !job->printer) + cupsdDeleteJob(job, CUPSD_JOB_PURGE); +} + + +/* + * 'cupsdContinueJob()' - Continue printing with the next file in a job. + */ + +void +cupsdContinueJob(cupsd_job_t *job) /* I - Job */ +{ + int i; /* Looping var */ + int slot; /* Pipe slot */ + cups_array_t *filters = NULL,/* Filters for job */ + *prefilters; /* Filters with prefilters */ + mime_filter_t *filter, /* Current filter */ + *prefilter, /* Prefilter */ + port_monitor; /* Port monitor filter */ + char scheme[255]; /* Device URI scheme */ + ipp_attribute_t *attr; /* Current attribute */ + const char *ptr, /* Pointer into value */ + *abort_message; /* Abort message */ + ipp_jstate_t abort_state = IPP_JOB_STOPPED; + /* New job state on abort */ + struct stat backinfo; /* Backend file information */ + int backroot; /* Run backend as root? */ + int pid; /* Process ID of new filter process */ + int banner_page; /* 1 if banner page, 0 otherwise */ + int filterfds[2][2] = { { -1, -1 }, { -1, -1 } }; + /* Pipes used between filters */ + int envc; /* Number of environment variables */ + struct stat fileinfo; /* Job file information */ + char **argv = NULL, /* Filter command-line arguments */ + filename[1024], /* Job filename */ + command[1024], /* Full path to command */ + jobid[255], /* Job ID string */ + title[IPP_MAX_NAME], + /* Job title string */ + copies[255], /* # copies string */ + *options, /* Options string */ + *envp[MAX_ENV + 21], + /* Environment variables */ + charset[255], /* CHARSET env variable */ + class_name[255],/* CLASS env variable */ + classification[1024], + /* CLASSIFICATION env variable */ + content_type[1024], + /* CONTENT_TYPE env variable */ + device_uri[1024], + /* DEVICE_URI env variable */ + final_content_type[1024], + /* FINAL_CONTENT_TYPE env variable */ + lang[255], /* LANG env variable */ +#ifdef __APPLE__ + apple_language[255], + /* APPLE_LANGUAGE env variable */ +#endif /* __APPLE__ */ + auth_info_required[255], + /* AUTH_INFO_REQUIRED env variable */ + ppd[1024], /* PPD env variable */ + printer_info[255], + /* PRINTER_INFO env variable */ + printer_location[255], + /* PRINTER_LOCATION env variable */ + printer_name[255], + /* PRINTER env variable */ + *printer_state_reasons = NULL, + /* PRINTER_STATE_REASONS env var */ + rip_max_cache[255]; + /* RIP_MAX_CACHE env variable */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdContinueJob(job=%p(%d)): current_file=%d, num_files=%d", + job, job->id, job->current_file, job->num_files); + + /* + * Figure out what filters are required to convert from + * the source to the destination type... + */ + + FilterLevel -= job->cost; + + job->cost = 0; + job->pending_cost = 0; + + memset(job->filters, 0, sizeof(job->filters)); + + + if (job->printer->raw) + { + /* + * Remote jobs and raw queues go directly to the printer without + * filtering... + */ + + cupsdLogJob(job, CUPSD_LOG_DEBUG, "Sending job to queue tagged as raw..."); + } + else + { + /* + * Local jobs get filtered... + */ + + snprintf(filename, sizeof(filename), "%s/d%05d-%03d", RequestRoot, + job->id, job->current_file + 1); + if (stat(filename, &fileinfo)) + fileinfo.st_size = 0; + + filters = mimeFilter2(MimeDatabase, job->filetypes[job->current_file], + fileinfo.st_size, job->printer->filetype, + &(job->cost)); + + if (!filters) + { + cupsdLogJob(job, CUPSD_LOG_ERROR, + "Unable to convert file %d to printable format.", + job->current_file); + + abort_message = "Aborting job because it cannot be printed."; + abort_state = IPP_JOB_ABORTED; + + ippSetString(job->attrs, &job->reasons, 0, "document-unprintable-error"); + goto abort_job; + } + + /* + * Remove NULL ("-") filters... + */ + + for (filter = (mime_filter_t *)cupsArrayFirst(filters); + filter; + filter = (mime_filter_t *)cupsArrayNext(filters)) + if (!strcmp(filter->filter, "-")) + cupsArrayRemove(filters, filter); + + if (cupsArrayCount(filters) == 0) + { + cupsArrayDelete(filters); + filters = NULL; + } + + /* + * If this printer has any pre-filters, insert the required pre-filter + * in the filters array... + */ + + if (job->printer->prefiltertype && filters) + { + prefilters = cupsArrayNew(NULL, NULL); + + for (filter = (mime_filter_t *)cupsArrayFirst(filters); + filter; + filter = (mime_filter_t *)cupsArrayNext(filters)) + { + if ((prefilter = mimeFilterLookup(MimeDatabase, filter->src, + job->printer->prefiltertype))) + { + cupsArrayAdd(prefilters, prefilter); + job->cost += prefilter->cost; + } + + cupsArrayAdd(prefilters, filter); + } + + cupsArrayDelete(filters); + filters = prefilters; + } + } + + /* + * Set a minimum cost of 100 for all jobs so that FilterLimit + * works with raw queues and other low-cost paths. + */ + + if (job->cost < 100) + job->cost = 100; + + /* + * See if the filter cost is too high... + */ + + if ((FilterLevel + job->cost) > FilterLimit && FilterLevel > 0 && + FilterLimit > 0) + { + /* + * Don't print this job quite yet... + */ + + cupsArrayDelete(filters); + + cupsdLogJob(job, CUPSD_LOG_INFO, + "Holding because filter limit has been reached."); + cupsdLogJob(job, CUPSD_LOG_DEBUG2, + "cupsdContinueJob: file=%d, cost=%d, level=%d, limit=%d", + job->current_file, job->cost, FilterLevel, + FilterLimit); + + job->pending_cost = job->cost; + job->cost = 0; + return; + } + + FilterLevel += job->cost; + + /* + * Add decompression/raw filter as needed... + */ + + if ((!job->printer->raw && job->compressions[job->current_file]) || + (!filters && !job->printer->remote && + (job->num_files > 1 || !strncmp(job->printer->device_uri, "file:", 5)))) + { + /* + * Add gziptoany filter to the front of the list... + */ + + if (!filters) + filters = cupsArrayNew(NULL, NULL); + + if (!cupsArrayInsert(filters, &gziptoany_filter)) + { + cupsdLogJob(job, CUPSD_LOG_DEBUG, + "Unable to add decompression filter - %s", strerror(errno)); + + cupsArrayDelete(filters); + + abort_message = "Stopping job because the scheduler ran out of memory."; + + goto abort_job; + } + } + + /* + * Add port monitor, if any... + */ + + if (job->printer->port_monitor) + { + /* + * Add port monitor to the end of the list... + */ + + if (!filters) + filters = cupsArrayNew(NULL, NULL); + + port_monitor.src = NULL; + port_monitor.dst = NULL; + port_monitor.cost = 0; + + snprintf(port_monitor.filter, sizeof(port_monitor.filter), + "%s/monitor/%s", ServerBin, job->printer->port_monitor); + + if (!cupsArrayAdd(filters, &port_monitor)) + { + cupsdLogJob(job, CUPSD_LOG_DEBUG, + "Unable to add port monitor - %s", strerror(errno)); + + abort_message = "Stopping job because the scheduler ran out of memory."; + + goto abort_job; + } + } + + /* + * Make sure we don't go over the "MAX_FILTERS" limit... + */ + + if (cupsArrayCount(filters) > MAX_FILTERS) + { + cupsdLogJob(job, CUPSD_LOG_DEBUG, + "Too many filters (%d > %d), unable to print.", + cupsArrayCount(filters), MAX_FILTERS); + + abort_message = "Aborting job because it needs too many filters to print."; + abort_state = IPP_JOB_ABORTED; + + ippSetString(job->attrs, &job->reasons, 0, "document-unprintable-error"); + + goto abort_job; + } + + /* + * Determine if we are printing a banner page or not... + */ + + if (job->job_sheets == NULL) + { + cupsdLogJob(job, CUPSD_LOG_DEBUG, "No job-sheets attribute."); + if ((job->job_sheets = + ippFindAttribute(job->attrs, "job-sheets", IPP_TAG_ZERO)) != NULL) + cupsdLogJob(job, CUPSD_LOG_DEBUG, + "... but someone added one without setting job_sheets."); + } + else if (job->job_sheets->num_values == 1) + cupsdLogJob(job, CUPSD_LOG_DEBUG, "job-sheets=%s", + job->job_sheets->values[0].string.text); + else + cupsdLogJob(job, CUPSD_LOG_DEBUG, "job-sheets=%s,%s", + job->job_sheets->values[0].string.text, + job->job_sheets->values[1].string.text); + + if (job->printer->type & CUPS_PRINTER_REMOTE) + banner_page = 0; + else if (job->job_sheets == NULL) + banner_page = 0; + else if (_cups_strcasecmp(job->job_sheets->values[0].string.text, "none") != 0 && + job->current_file == 0) + banner_page = 1; + else if (job->job_sheets->num_values > 1 && + _cups_strcasecmp(job->job_sheets->values[1].string.text, "none") != 0 && + job->current_file == (job->num_files - 1)) + banner_page = 1; + else + banner_page = 0; + + if ((options = get_options(job, banner_page, copies, sizeof(copies), title, + sizeof(title))) == NULL) + { + abort_message = "Stopping job because the scheduler ran out of memory."; + + goto abort_job; + } + + /* + * Build the command-line arguments for the filters. Each filter + * has 6 or 7 arguments: + * + * argv[0] = printer + * argv[1] = job ID + * argv[2] = username + * argv[3] = title + * argv[4] = # copies + * argv[5] = options + * argv[6] = filename (optional; normally stdin) + * + * This allows legacy printer drivers that use the old System V + * printing interface to be used by CUPS. + * + * For remote jobs, we send all of the files in the argument list. + */ + + if (job->printer->remote) + argv = calloc(7 + job->num_files, sizeof(char *)); + else + argv = calloc(8, sizeof(char *)); + + if (!argv) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, "Unable to allocate argument array - %s", + strerror(errno)); + + abort_message = "Stopping job because the scheduler ran out of memory."; + + goto abort_job; + } + + sprintf(jobid, "%d", job->id); + + argv[0] = job->printer->name; + argv[1] = jobid; + argv[2] = job->username; + argv[3] = title; + argv[4] = copies; + argv[5] = options; + + if (job->printer->remote && job->num_files > 1) + { + for (i = 0; i < job->num_files; i ++) + { + snprintf(filename, sizeof(filename), "%s/d%05d-%03d", RequestRoot, + job->id, i + 1); + argv[6 + i] = strdup(filename); + } + } + else + { + snprintf(filename, sizeof(filename), "%s/d%05d-%03d", RequestRoot, + job->id, job->current_file + 1); + argv[6] = filename; + } + + for (i = 0; argv[i]; i ++) + cupsdLogJob(job, CUPSD_LOG_DEBUG, "argv[%d]=\"%s\"", i, argv[i]); + + /* + * Create environment variable strings for the filters... + */ + + attr = ippFindAttribute(job->attrs, "attributes-natural-language", + IPP_TAG_LANGUAGE); + +#ifdef __APPLE__ + strcpy(apple_language, "APPLE_LANGUAGE="); + _cupsAppleLanguage(attr->values[0].string.text, + apple_language + 15, sizeof(apple_language) - 15); +#endif /* __APPLE__ */ + + switch (strlen(attr->values[0].string.text)) + { + default : + /* + * This is an unknown or badly formatted language code; use + * the POSIX locale... + */ + + strcpy(lang, "LANG=C"); + break; + + case 2 : + /* + * Just the language code (ll)... + */ + + snprintf(lang, sizeof(lang), "LANG=%s.UTF-8", + attr->values[0].string.text); + break; + + case 5 : + /* + * Language and country code (ll-cc)... + */ + + snprintf(lang, sizeof(lang), "LANG=%c%c_%c%c.UTF-8", + attr->values[0].string.text[0], + attr->values[0].string.text[1], + toupper(attr->values[0].string.text[3] & 255), + toupper(attr->values[0].string.text[4] & 255)); + break; + } + + if ((attr = ippFindAttribute(job->attrs, "document-format", + IPP_TAG_MIMETYPE)) != NULL && + (ptr = strstr(attr->values[0].string.text, "charset=")) != NULL) + snprintf(charset, sizeof(charset), "CHARSET=%s", ptr + 8); + else + strlcpy(charset, "CHARSET=utf-8", sizeof(charset)); + + snprintf(content_type, sizeof(content_type), "CONTENT_TYPE=%s/%s", + job->filetypes[job->current_file]->super, + job->filetypes[job->current_file]->type); + snprintf(device_uri, sizeof(device_uri), "DEVICE_URI=%s", + job->printer->device_uri); + snprintf(ppd, sizeof(ppd), "PPD=%s/ppd/%s.ppd", ServerRoot, + job->printer->name); + snprintf(printer_info, sizeof(printer_name), "PRINTER_INFO=%s", + job->printer->info ? job->printer->info : ""); + snprintf(printer_location, sizeof(printer_name), "PRINTER_LOCATION=%s", + job->printer->location ? job->printer->location : ""); + snprintf(printer_name, sizeof(printer_name), "PRINTER=%s", job->printer->name); + if (job->printer->num_reasons > 0) + { + char *psrptr; /* Pointer into PRINTER_STATE_REASONS */ + size_t psrlen; /* Size of PRINTER_STATE_REASONS */ + + for (psrlen = 22, i = 0; i < job->printer->num_reasons; i ++) + psrlen += strlen(job->printer->reasons[i]) + 1; + + if ((printer_state_reasons = malloc(psrlen)) != NULL) + { + /* + * All of these strcpy's are safe because we allocated the psr string... + */ + + strcpy(printer_state_reasons, "PRINTER_STATE_REASONS="); + for (psrptr = printer_state_reasons + 22, i = 0; + i < job->printer->num_reasons; + i ++) + { + if (i) + *psrptr++ = ','; + strcpy(psrptr, job->printer->reasons[i]); + psrptr += strlen(psrptr); + } + } + } + snprintf(rip_max_cache, sizeof(rip_max_cache), "RIP_MAX_CACHE=%s", RIPCache); + + if (job->printer->num_auth_info_required == 1) + snprintf(auth_info_required, sizeof(auth_info_required), + "AUTH_INFO_REQUIRED=%s", + job->printer->auth_info_required[0]); + else if (job->printer->num_auth_info_required == 2) + snprintf(auth_info_required, sizeof(auth_info_required), + "AUTH_INFO_REQUIRED=%s,%s", + job->printer->auth_info_required[0], + job->printer->auth_info_required[1]); + else if (job->printer->num_auth_info_required == 3) + snprintf(auth_info_required, sizeof(auth_info_required), + "AUTH_INFO_REQUIRED=%s,%s,%s", + job->printer->auth_info_required[0], + job->printer->auth_info_required[1], + job->printer->auth_info_required[2]); + else if (job->printer->num_auth_info_required == 4) + snprintf(auth_info_required, sizeof(auth_info_required), + "AUTH_INFO_REQUIRED=%s,%s,%s,%s", + job->printer->auth_info_required[0], + job->printer->auth_info_required[1], + job->printer->auth_info_required[2], + job->printer->auth_info_required[3]); + else + strlcpy(auth_info_required, "AUTH_INFO_REQUIRED=none", + sizeof(auth_info_required)); + + envc = cupsdLoadEnv(envp, (int)(sizeof(envp) / sizeof(envp[0]))); + + envp[envc ++] = charset; + envp[envc ++] = lang; +#ifdef __APPLE__ + envp[envc ++] = apple_language; +#endif /* __APPLE__ */ + envp[envc ++] = ppd; + envp[envc ++] = rip_max_cache; + envp[envc ++] = content_type; + envp[envc ++] = device_uri; + envp[envc ++] = printer_info; + envp[envc ++] = printer_location; + envp[envc ++] = printer_name; + envp[envc ++] = printer_state_reasons ? printer_state_reasons : + "PRINTER_STATE_REASONS=none"; + envp[envc ++] = banner_page ? "CUPS_FILETYPE=job-sheet" : + "CUPS_FILETYPE=document"; + + if (!job->printer->remote && !job->printer->raw) + { + filter = (mime_filter_t *)cupsArrayLast(filters); + + if (job->printer->port_monitor) + filter = (mime_filter_t *)cupsArrayPrev(filters); + + if (filter && filter->dst) + { + if ((ptr = strchr(filter->dst->type, '/')) != NULL) + snprintf(final_content_type, sizeof(final_content_type), + "FINAL_CONTENT_TYPE=%s", ptr + 1); + else + snprintf(final_content_type, sizeof(final_content_type), + "FINAL_CONTENT_TYPE=%s/%s", filter->dst->super, + filter->dst->type); + envp[envc ++] = final_content_type; + } + } + + if (Classification && !banner_page) + { + if ((attr = ippFindAttribute(job->attrs, "job-sheets", + IPP_TAG_NAME)) == NULL) + snprintf(classification, sizeof(classification), "CLASSIFICATION=%s", + Classification); + else if (attr->num_values > 1 && + strcmp(attr->values[1].string.text, "none") != 0) + snprintf(classification, sizeof(classification), "CLASSIFICATION=%s", + attr->values[1].string.text); + else + snprintf(classification, sizeof(classification), "CLASSIFICATION=%s", + attr->values[0].string.text); + + envp[envc ++] = classification; + } + + if (job->dtype & CUPS_PRINTER_CLASS) + { + snprintf(class_name, sizeof(class_name), "CLASS=%s", job->dest); + envp[envc ++] = class_name; + } + + envp[envc ++] = auth_info_required; + + for (i = 0; + i < (int)(sizeof(job->auth_env) / sizeof(job->auth_env[0])); + i ++) + if (job->auth_env[i]) + envp[envc ++] = job->auth_env[i]; + else + break; + + if (job->auth_uid) + envp[envc ++] = job->auth_uid; + + envp[envc] = NULL; + + for (i = 0; i < envc; i ++) + if (!strncmp(envp[i], "AUTH_", 5)) + cupsdLogJob(job, CUPSD_LOG_DEBUG, "envp[%d]=\"AUTH_%c****\"", i, + envp[i][5]); + else if (strncmp(envp[i], "DEVICE_URI=", 11)) + cupsdLogJob(job, CUPSD_LOG_DEBUG, "envp[%d]=\"%s\"", i, envp[i]); + else + cupsdLogJob(job, CUPSD_LOG_DEBUG, "envp[%d]=\"DEVICE_URI=%s\"", i, + job->printer->sanitized_device_uri); + + if (job->printer->remote) + job->current_file = job->num_files; + else + job->current_file ++; + + /* + * Now create processes for all of the filters... + */ + + cupsdSetPrinterReasons(job->printer, "-cups-missing-filter-warning," + "cups-insecure-filter-warning"); + + for (i = 0, slot = 0, filter = (mime_filter_t *)cupsArrayFirst(filters); + filter; + i ++, filter = (mime_filter_t *)cupsArrayNext(filters)) + { + if (filter->filter[0] != '/') + snprintf(command, sizeof(command), "%s/filter/%s", ServerBin, + filter->filter); + else + strlcpy(command, filter->filter, sizeof(command)); + + if (i < (cupsArrayCount(filters) - 1)) + { + if (cupsdOpenPipe(filterfds[slot])) + { + abort_message = "Stopping job because the scheduler could not create " + "the filter pipes."; + + goto abort_job; + } + } + else + { + if (job->current_file == 1 || + (job->printer->pc && job->printer->pc->single_file)) + { + if (strncmp(job->printer->device_uri, "file:", 5) != 0) + { + if (cupsdOpenPipe(job->print_pipes)) + { + abort_message = "Stopping job because the scheduler could not " + "create the backend pipes."; + + goto abort_job; + } + } + else + { + job->print_pipes[0] = -1; + if (!strcmp(job->printer->device_uri, "file:/dev/null") || + !strcmp(job->printer->device_uri, "file:///dev/null")) + job->print_pipes[1] = -1; + else + { + if (!strncmp(job->printer->device_uri, "file:/dev/", 10)) + job->print_pipes[1] = open(job->printer->device_uri + 5, + O_WRONLY | O_EXCL); + else if (!strncmp(job->printer->device_uri, "file:///dev/", 12)) + job->print_pipes[1] = open(job->printer->device_uri + 7, + O_WRONLY | O_EXCL); + else if (!strncmp(job->printer->device_uri, "file:///", 8)) + job->print_pipes[1] = open(job->printer->device_uri + 7, + O_WRONLY | O_CREAT | O_TRUNC, 0600); + else + job->print_pipes[1] = open(job->printer->device_uri + 5, + O_WRONLY | O_CREAT | O_TRUNC, 0600); + + if (job->print_pipes[1] < 0) + { + abort_message = "Stopping job because the scheduler could not " + "open the output file."; + + goto abort_job; + } + + fcntl(job->print_pipes[1], F_SETFD, + fcntl(job->print_pipes[1], F_GETFD) | FD_CLOEXEC); + } + } + } + + filterfds[slot][0] = job->print_pipes[0]; + filterfds[slot][1] = job->print_pipes[1]; + } + + pid = cupsdStartProcess(command, argv, envp, filterfds[!slot][0], + filterfds[slot][1], job->status_pipes[1], + job->back_pipes[0], job->side_pipes[0], 0, + job->profile, job, job->filters + i); + + cupsdClosePipe(filterfds[!slot]); + + if (pid == 0) + { + cupsdLogJob(job, CUPSD_LOG_ERROR, "Unable to start filter \"%s\" - %s.", + filter->filter, strerror(errno)); + + abort_message = "Stopping job because the scheduler could not execute a " + "filter."; + + goto abort_job; + } + + cupsdLogJob(job, CUPSD_LOG_INFO, "Started filter %s (PID %d)", command, + pid); + + argv[6] = NULL; + slot = !slot; + } + + cupsArrayDelete(filters); + filters = NULL; + + /* + * Finally, pipe the final output into a backend process if needed... + */ + + if (strncmp(job->printer->device_uri, "file:", 5) != 0) + { + if (job->current_file == 1 || job->printer->remote || + (job->printer->pc && job->printer->pc->single_file)) + { + sscanf(job->printer->device_uri, "%254[^:]", scheme); + snprintf(command, sizeof(command), "%s/backend/%s", ServerBin, scheme); + + /* + * See if the backend needs to run as root... + */ + + if (RunUser) + backroot = 0; + else if (stat(command, &backinfo)) + backroot = 0; + else + backroot = !(backinfo.st_mode & (S_IRWXG | S_IRWXO)); + + argv[0] = job->printer->sanitized_device_uri; + + filterfds[slot][0] = -1; + filterfds[slot][1] = -1; + + pid = cupsdStartProcess(command, argv, envp, filterfds[!slot][0], + filterfds[slot][1], job->status_pipes[1], + job->back_pipes[1], job->side_pipes[1], + backroot, job->profile, job, &(job->backend)); + + if (pid == 0) + { + abort_message = "Stopping job because the sheduler could not execute " + "the backend."; + + goto abort_job; + } + else + { + cupsdLogJob(job, CUPSD_LOG_INFO, "Started backend %s (PID %d)", + command, pid); + } + } + + if (job->current_file == job->num_files || + (job->printer->pc && job->printer->pc->single_file)) + cupsdClosePipe(job->print_pipes); + + if (job->current_file == job->num_files) + { + cupsdClosePipe(job->back_pipes); + cupsdClosePipe(job->side_pipes); + + close(job->status_pipes[1]); + job->status_pipes[1] = -1; + } + } + else + { + filterfds[slot][0] = -1; + filterfds[slot][1] = -1; + + if (job->current_file == job->num_files || + (job->printer->pc && job->printer->pc->single_file)) + cupsdClosePipe(job->print_pipes); + + if (job->current_file == job->num_files) + { + close(job->status_pipes[1]); + job->status_pipes[1] = -1; + } + } + + cupsdClosePipe(filterfds[slot]); + + if (job->printer->remote && job->num_files > 1) + { + for (i = 0; i < job->num_files; i ++) + free(argv[i + 6]); + } + + free(argv); + if (printer_state_reasons) + free(printer_state_reasons); + + cupsdAddSelect(job->status_buffer->fd, (cupsd_selfunc_t)update_job, NULL, + job); + + cupsdAddEvent(CUPSD_EVENT_JOB_STATE, job->printer, job, "Job #%d started.", + job->id); + + return; + + + /* + * If we get here, we need to abort the current job and close out all + * files and pipes... + */ + + abort_job: + + FilterLevel -= job->cost; + job->cost = 0; + + for (slot = 0; slot < 2; slot ++) + cupsdClosePipe(filterfds[slot]); + + cupsArrayDelete(filters); + + if (argv) + { + if (job->printer->remote && job->num_files > 1) + { + for (i = 0; i < job->num_files; i ++) + free(argv[i + 6]); + } + + free(argv); + } + + if (printer_state_reasons) + free(printer_state_reasons); + + cupsdClosePipe(job->print_pipes); + cupsdClosePipe(job->back_pipes); + cupsdClosePipe(job->side_pipes); + + cupsdRemoveSelect(job->status_pipes[0]); + cupsdClosePipe(job->status_pipes); + cupsdStatBufDelete(job->status_buffer); + job->status_buffer = NULL; + + /* + * Update the printer and job state. + */ + + cupsdSetJobState(job, abort_state, CUPSD_JOB_DEFAULT, "%s", abort_message); + cupsdSetPrinterState(job->printer, IPP_PRINTER_IDLE, 0); + update_job_attrs(job, 0); + + if (job->history) + free_job_history(job); + + cupsArrayRemove(PrintingJobs, job); + + /* + * Clear the printer <-> job association... + */ + + job->printer->job = NULL; + job->printer = NULL; +} + + +/* + * 'cupsdDeleteJob()' - Free all memory used by a job. + */ + +void +cupsdDeleteJob(cupsd_job_t *job, /* I - Job */ + cupsd_jobaction_t action)/* I - Action */ +{ + int i; /* Looping var */ + char filename[1024]; /* Job filename */ + + + if (job->printer) + finalize_job(job, 1); + + if (action == CUPSD_JOB_PURGE) + { + /* + * Remove the job info file... + */ + + snprintf(filename, sizeof(filename), "%s/c%05d", RequestRoot, + job->id); + if (Classification) + cupsdRemoveFile(filename); + else + unlink(filename); + } + + cupsdClearString(&job->username); + cupsdClearString(&job->dest); + for (i = 0; + i < (int)(sizeof(job->auth_env) / sizeof(job->auth_env[0])); + i ++) + cupsdClearString(job->auth_env + i); + cupsdClearString(&job->auth_uid); + + if (job->num_files > 0) + { + free(job->compressions); + free(job->filetypes); + + if (action == CUPSD_JOB_PURGE) + { + while (job->num_files > 0) + { + snprintf(filename, sizeof(filename), "%s/d%05d-%03d", RequestRoot, + job->id, job->num_files); + if (Classification) + cupsdRemoveFile(filename); + else + unlink(filename); + + job->num_files --; + } + } + else + job->num_files = 0; + } + + if (job->history) + free_job_history(job); + + unload_job(job); + + cupsArrayRemove(Jobs, job); + cupsArrayRemove(ActiveJobs, job); + cupsArrayRemove(PrintingJobs, job); + + free(job); +} + + +/* + * 'cupsdFreeAllJobs()' - Free all jobs from memory. + */ + +void +cupsdFreeAllJobs(void) +{ + cupsd_job_t *job; /* Current job */ + + + if (!Jobs) + return; + + cupsdHoldSignals(); + + cupsdStopAllJobs(CUPSD_JOB_FORCE, 0); + cupsdSaveAllJobs(); + + for (job = (cupsd_job_t *)cupsArrayFirst(Jobs); + job; + job = (cupsd_job_t *)cupsArrayNext(Jobs)) + cupsdDeleteJob(job, CUPSD_JOB_DEFAULT); + + cupsdReleaseSignals(); +} + + +/* + * 'cupsdFindJob()' - Find the specified job. + */ + +cupsd_job_t * /* O - Job data */ +cupsdFindJob(int id) /* I - Job ID */ +{ + cupsd_job_t key; /* Search key */ + + + key.id = id; + + return ((cupsd_job_t *)cupsArrayFind(Jobs, &key)); +} + + +/* + * 'cupsdGetPrinterJobCount()' - Get the number of pending, processing, + * or held jobs in a printer or class. + */ + +int /* O - Job count */ +cupsdGetPrinterJobCount( + const char *dest) /* I - Printer or class name */ +{ + int count; /* Job count */ + cupsd_job_t *job; /* Current job */ + + + for (job = (cupsd_job_t *)cupsArrayFirst(ActiveJobs), count = 0; + job; + job = (cupsd_job_t *)cupsArrayNext(ActiveJobs)) + if (job->dest && !_cups_strcasecmp(job->dest, dest)) + count ++; + + return (count); +} + + +/* + * 'cupsdGetUserJobCount()' - Get the number of pending, processing, + * or held jobs for a user. + */ + +int /* O - Job count */ +cupsdGetUserJobCount( + const char *username) /* I - Username */ +{ + int count; /* Job count */ + cupsd_job_t *job; /* Current job */ + + + for (job = (cupsd_job_t *)cupsArrayFirst(ActiveJobs), count = 0; + job; + job = (cupsd_job_t *)cupsArrayNext(ActiveJobs)) + if (!_cups_strcasecmp(job->username, username)) + count ++; + + return (count); +} + + +/* + * 'cupsdLoadAllJobs()' - Load all jobs from disk. + */ + +void +cupsdLoadAllJobs(void) +{ + char filename[1024]; /* Full filename of job.cache file */ + struct stat fileinfo, /* Information on job.cache file */ + dirinfo; /* Information on RequestRoot dir */ + + + + /* + * Create the job arrays as needed... + */ + + if (!Jobs) + Jobs = cupsArrayNew(compare_jobs, NULL); + + if (!ActiveJobs) + ActiveJobs = cupsArrayNew(compare_active_jobs, NULL); + + if (!PrintingJobs) + PrintingJobs = cupsArrayNew(compare_jobs, NULL); + + /* + * See whether the job.cache file is older than the RequestRoot directory... + */ + + snprintf(filename, sizeof(filename), "%s/job.cache", CacheDir); + + if (stat(filename, &fileinfo)) + { + fileinfo.st_mtime = 0; + + if (errno != ENOENT) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to get file information for \"%s\" - %s", + filename, strerror(errno)); + } + + if (stat(RequestRoot, &dirinfo)) + { + dirinfo.st_mtime = 0; + + if (errno != ENOENT) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to get directory information for \"%s\" - %s", + RequestRoot, strerror(errno)); + } + + /* + * Load the most recent source for job data... + */ + + if (dirinfo.st_mtime > fileinfo.st_mtime) + { + load_request_root(); + + load_next_job_id(filename); + } + else + load_job_cache(filename); + + /* + * Clean out old jobs as needed... + */ + + if (MaxJobs > 0 && cupsArrayCount(Jobs) >= MaxJobs) + cupsdCleanJobs(); +} + + +/* + * 'cupsdLoadJob()' - Load a single job. + */ + +int /* O - 1 on success, 0 on failure */ +cupsdLoadJob(cupsd_job_t *job) /* I - Job */ +{ + int i; /* Looping var */ + char jobfile[1024]; /* Job filename */ + cups_file_t *fp; /* Job file */ + int fileid; /* Current file ID */ + ipp_attribute_t *attr; /* Job attribute */ + const char *dest; /* Destination name */ + cupsd_printer_t *destptr; /* Pointer to destination */ + mime_type_t **filetypes; /* New filetypes array */ + int *compressions; /* New compressions array */ + + + if (job->attrs) + { + if (job->state_value > IPP_JOB_STOPPED) + job->access_time = time(NULL); + + return (1); + } + + if ((job->attrs = ippNew()) == NULL) + { + cupsdLogJob(job, CUPSD_LOG_ERROR, "Ran out of memory for job attributes."); + return (0); + } + + /* + * Load job attributes... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, "[Job %d] Loading attributes...", job->id); + + snprintf(jobfile, sizeof(jobfile), "%s/c%05d", RequestRoot, job->id); + if ((fp = cupsFileOpen(jobfile, "r")) == NULL) + { + char newfile[1024]; /* New job filename */ + + snprintf(newfile, sizeof(newfile), "%s/c%05d.N", RequestRoot, job->id); + if ((fp = cupsFileOpen(newfile, "r")) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Job %d] Unable to open job control file \"%s\": %s", + job->id, jobfile, strerror(errno)); + goto error; + } + + unlink(jobfile); + rename(newfile, jobfile); + } + + if (ippReadIO(fp, (ipp_iocb_t)cupsFileRead, 1, NULL, job->attrs) != IPP_DATA) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Job %d] Unable to read job control file \"%s\".", job->id, + jobfile); + cupsFileClose(fp); + goto error; + } + + cupsFileClose(fp); + + /* + * Copy attribute data to the job object... + */ + + if (!ippFindAttribute(job->attrs, "time-at-creation", IPP_TAG_INTEGER)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Job %d] Missing or bad time-at-creation attribute in " + "control file.", job->id); + goto error; + } + + if ((job->state = ippFindAttribute(job->attrs, "job-state", + IPP_TAG_ENUM)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Job %d] Missing or bad job-state attribute in control " + "file.", job->id); + goto error; + } + + job->state_value = (ipp_jstate_t)job->state->values[0].integer; + + if (!job->dest) + { + if ((attr = ippFindAttribute(job->attrs, "job-printer-uri", + IPP_TAG_URI)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Job %d] No job-printer-uri attribute in control file.", + job->id); + goto error; + } + + if ((dest = cupsdValidateDest(attr->values[0].string.text, &(job->dtype), + &destptr)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Job %d] Unable to queue job for destination \"%s\".", + job->id, attr->values[0].string.text); + goto error; + } + + cupsdSetString(&job->dest, dest); + } + else if ((destptr = cupsdFindDest(job->dest)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Job %d] Unable to queue job for destination \"%s\".", + job->id, job->dest); + goto error; + } + + if ((job->reasons = ippFindAttribute(job->attrs, "job-state-reasons", + IPP_TAG_KEYWORD)) == NULL) + { + const char *reason; /* job-state-reason keyword */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Job %d] Adding missing job-state-reasons attribute to " + " control file.", job->id); + + switch (job->state_value) + { + case IPP_JOB_PENDING : + if (destptr->state == IPP_PRINTER_STOPPED) + reason = "printer-stopped"; + else + reason = "none"; + break; + + case IPP_JOB_HELD : + if ((attr = ippFindAttribute(job->attrs, "job-hold-until", + IPP_TAG_ZERO)) != NULL && + (attr->value_tag == IPP_TAG_NAME || + attr->value_tag == IPP_TAG_NAMELANG || + attr->value_tag == IPP_TAG_KEYWORD) && + strcmp(attr->values[0].string.text, "no-hold")) + reason = "job-hold-until-specified"; + else + reason = "job-incoming"; + break; + + case IPP_JOB_PROCESSING : + reason = "job-printing"; + break; + + case IPP_JOB_STOPPED : + reason = "job-stopped"; + break; + + case IPP_JOB_CANCELED : + reason = "job-canceled-by-user"; + break; + + case IPP_JOB_ABORTED : + reason = "aborted-by-system"; + break; + + case IPP_JOB_COMPLETED : + reason = "job-completed-successfully"; + break; + } + + job->reasons = ippAddString(job->attrs, IPP_TAG_JOB, IPP_TAG_KEYWORD, + "job-state-reasons", NULL, reason); + } + else if (job->state_value == IPP_JOB_PENDING) + { + if (destptr->state == IPP_PRINTER_STOPPED) + ippSetString(job->attrs, &job->reasons, 0, "printer-stopped"); + else + ippSetString(job->attrs, &job->reasons, 0, "none"); + } + + job->sheets = ippFindAttribute(job->attrs, "job-media-sheets-completed", + IPP_TAG_INTEGER); + job->job_sheets = ippFindAttribute(job->attrs, "job-sheets", IPP_TAG_NAME); + + if (!job->priority) + { + if ((attr = ippFindAttribute(job->attrs, "job-priority", + IPP_TAG_INTEGER)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Job %d] Missing or bad job-priority attribute in " + "control file.", job->id); + goto error; + } + + job->priority = attr->values[0].integer; + } + + if (!job->username) + { + if ((attr = ippFindAttribute(job->attrs, "job-originating-user-name", + IPP_TAG_NAME)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Job %d] Missing or bad job-originating-user-name " + "attribute in control file.", job->id); + goto error; + } + + cupsdSetString(&job->username, attr->values[0].string.text); + } + + /* + * Set the job hold-until time and state... + */ + + if (job->state_value == IPP_JOB_HELD) + { + if ((attr = ippFindAttribute(job->attrs, "job-hold-until", + IPP_TAG_KEYWORD)) == NULL) + attr = ippFindAttribute(job->attrs, "job-hold-until", IPP_TAG_NAME); + + if (attr) + cupsdSetJobHoldUntil(job, attr->values[0].string.text, CUPSD_JOB_DEFAULT); + else + { + job->state->values[0].integer = IPP_JOB_PENDING; + job->state_value = IPP_JOB_PENDING; + } + } + else if (job->state_value == IPP_JOB_PROCESSING) + { + job->state->values[0].integer = IPP_JOB_PENDING; + job->state_value = IPP_JOB_PENDING; + } + + if (!job->num_files) + { + /* + * Find all the d##### files... + */ + + for (fileid = 1; fileid < 10000; fileid ++) + { + snprintf(jobfile, sizeof(jobfile), "%s/d%05d-%03d", RequestRoot, + job->id, fileid); + + if (access(jobfile, 0)) + break; + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "[Job %d] Auto-typing document file \"%s\"...", job->id, + jobfile); + + if (fileid > job->num_files) + { + if (job->num_files == 0) + { + compressions = (int *)calloc(fileid, sizeof(int)); + filetypes = (mime_type_t **)calloc(fileid, sizeof(mime_type_t *)); + } + else + { + compressions = (int *)realloc(job->compressions, + sizeof(int) * fileid); + filetypes = (mime_type_t **)realloc(job->filetypes, + sizeof(mime_type_t *) * + fileid); + } + + if (!compressions || !filetypes) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Job %d] Ran out of memory for job file types.", + job->id); + + ippDelete(job->attrs); + job->attrs = NULL; + + if (compressions) + free(compressions); + + if (filetypes) + free(filetypes); + + if (job->compressions) + { + free(job->compressions); + job->compressions = NULL; + } + + if (job->filetypes) + { + free(job->filetypes); + job->filetypes = NULL; + } + + job->num_files = 0; + return (0); + } + + job->compressions = compressions; + job->filetypes = filetypes; + job->num_files = fileid; + } + + job->filetypes[fileid - 1] = mimeFileType(MimeDatabase, jobfile, NULL, + job->compressions + fileid - 1); + + if (!job->filetypes[fileid - 1]) + job->filetypes[fileid - 1] = mimeType(MimeDatabase, "application", + "vnd.cups-raw"); + } + } + + /* + * Load authentication information as needed... + */ + + if (job->state_value < IPP_JOB_STOPPED) + { + snprintf(jobfile, sizeof(jobfile), "%s/a%05d", RequestRoot, job->id); + + for (i = 0; + i < (int)(sizeof(job->auth_env) / sizeof(job->auth_env[0])); + i ++) + cupsdClearString(job->auth_env + i); + cupsdClearString(&job->auth_uid); + + if ((fp = cupsFileOpen(jobfile, "r")) != NULL) + { + int bytes, /* Size of auth data */ + linenum = 1; /* Current line number */ + char line[65536], /* Line from file */ + *value, /* Value from line */ + data[65536]; /* Decoded data */ + + + if (cupsFileGets(fp, line, sizeof(line)) && + !strcmp(line, "CUPSD-AUTH-V2")) + { + i = 0; + while (cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + { + /* + * Decode value... + */ + + bytes = sizeof(data); + httpDecode64_2(data, &bytes, value); + + /* + * Assign environment variables... + */ + + if (!strcmp(line, "uid")) + { + cupsdSetStringf(&job->auth_uid, "AUTH_UID=%s", value); + continue; + } + else if (i >= (int)(sizeof(job->auth_env) / sizeof(job->auth_env[0]))) + break; + + if (!strcmp(line, "username")) + cupsdSetStringf(job->auth_env + i, "AUTH_USERNAME=%s", data); + else if (!strcmp(line, "domain")) + cupsdSetStringf(job->auth_env + i, "AUTH_DOMAIN=%s", data); + else if (!strcmp(line, "password")) + cupsdSetStringf(job->auth_env + i, "AUTH_PASSWORD=%s", data); + else if (!strcmp(line, "negotiate")) + cupsdSetStringf(job->auth_env + i, "AUTH_NEGOTIATE=%s", line); + else + continue; + + i ++; + } + } + + cupsFileClose(fp); + } + } + + job->access_time = time(NULL); + return (1); + + /* + * If we get here then something bad happened... + */ + + error: + + ippDelete(job->attrs); + job->attrs = NULL; + + if (job->compressions) + { + free(job->compressions); + job->compressions = NULL; + } + + if (job->filetypes) + { + free(job->filetypes); + job->filetypes = NULL; + } + + job->num_files = 0; + + if (Classification) + cupsdRemoveFile(jobfile); + else + unlink(jobfile); + + return (0); +} + + +/* + * 'cupsdMoveJob()' - Move the specified job to a different destination. + */ + +void +cupsdMoveJob(cupsd_job_t *job, /* I - Job */ + cupsd_printer_t *p) /* I - Destination printer or class */ +{ + ipp_attribute_t *attr; /* job-printer-uri attribute */ + const char *olddest; /* Old destination */ + cupsd_printer_t *oldp; /* Old pointer */ + + + /* + * Don't move completed jobs... + */ + + if (job->state_value > IPP_JOB_STOPPED) + return; + + /* + * Get the old destination... + */ + + olddest = job->dest; + + if (job->printer) + oldp = job->printer; + else + oldp = cupsdFindDest(olddest); + + /* + * Change the destination information... + */ + + if (job->state_value > IPP_JOB_HELD) + cupsdSetJobState(job, IPP_JOB_PENDING, CUPSD_JOB_DEFAULT, + "Stopping job prior to move."); + + cupsdAddEvent(CUPSD_EVENT_JOB_CONFIG_CHANGED, oldp, job, + "Job #%d moved from %s to %s.", job->id, olddest, + p->name); + + cupsdSetString(&job->dest, p->name); + job->dtype = p->type & (CUPS_PRINTER_CLASS | CUPS_PRINTER_REMOTE); + + if ((attr = ippFindAttribute(job->attrs, "job-printer-uri", + IPP_TAG_URI)) != NULL) + cupsdSetString(&(attr->values[0].string.text), p->uri); + + cupsdAddEvent(CUPSD_EVENT_JOB_STOPPED, p, job, + "Job #%d moved from %s to %s.", job->id, olddest, + p->name); + + job->dirty = 1; + cupsdMarkDirty(CUPSD_DIRTY_JOBS); +} + + +/* + * 'cupsdReleaseJob()' - Release the specified job. + */ + +void +cupsdReleaseJob(cupsd_job_t *job) /* I - Job */ +{ + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdReleaseJob(job=%p(%d))", job, + job->id); + + if (job->state_value == IPP_JOB_HELD) + { + /* + * Add trailing banner as needed... + */ + + if (job->pending_timeout) + cupsdTimeoutJob(job); + + cupsdSetJobState(job, IPP_JOB_PENDING, CUPSD_JOB_DEFAULT, + "Job released by user."); + } +} + + +/* + * 'cupsdRestartJob()' - Restart the specified job. + */ + +void +cupsdRestartJob(cupsd_job_t *job) /* I - Job */ +{ + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdRestartJob(job=%p(%d))", job, + job->id); + + if (job->state_value == IPP_JOB_STOPPED || job->num_files) + cupsdSetJobState(job, IPP_JOB_PENDING, CUPSD_JOB_DEFAULT, + "Job restarted by user."); +} + + +/* + * 'cupsdSaveAllJobs()' - Save a summary of all jobs to disk. + */ + +void +cupsdSaveAllJobs(void) +{ + int i; /* Looping var */ + cups_file_t *fp; /* job.cache file */ + char filename[1024], /* job.cache filename */ + temp[1024]; /* Temporary string */ + cupsd_job_t *job; /* Current job */ + time_t curtime; /* Current time */ + struct tm *curdate; /* Current date */ + + + snprintf(filename, sizeof(filename), "%s/job.cache", CacheDir); + if ((fp = cupsdCreateConfFile(filename, ConfigFilePerm)) == NULL) + return; + + cupsdLogMessage(CUPSD_LOG_INFO, "Saving job.cache..."); + + /* + * Write a small header to the file... + */ + + curtime = time(NULL); + curdate = localtime(&curtime); + strftime(temp, sizeof(temp) - 1, "%Y-%m-%d %H:%M", curdate); + + cupsFilePuts(fp, "# Job cache file for " CUPS_SVERSION "\n"); + cupsFilePrintf(fp, "# Written by cupsd on %s\n", temp); + cupsFilePrintf(fp, "NextJobId %d\n", NextJobId); + + /* + * Write each job known to the system... + */ + + for (job = (cupsd_job_t *)cupsArrayFirst(Jobs); + job; + job = (cupsd_job_t *)cupsArrayNext(Jobs)) + { + cupsFilePrintf(fp, "\n", job->id); + cupsFilePrintf(fp, "State %d\n", job->state_value); + cupsFilePrintf(fp, "Priority %d\n", job->priority); + cupsFilePrintf(fp, "HoldUntil %d\n", (int)job->hold_until); + cupsFilePrintf(fp, "Username %s\n", job->username); + cupsFilePrintf(fp, "Destination %s\n", job->dest); + cupsFilePrintf(fp, "DestType %d\n", job->dtype); + cupsFilePrintf(fp, "NumFiles %d\n", job->num_files); + for (i = 0; i < job->num_files; i ++) + cupsFilePrintf(fp, "File %d %s/%s %d\n", i + 1, job->filetypes[i]->super, + job->filetypes[i]->type, job->compressions[i]); + cupsFilePuts(fp, "\n"); + } + + cupsdCloseCreatedConfFile(fp, filename); +} + + +/* + * 'cupsdSaveJob()' - Save a job to disk. + */ + +void +cupsdSaveJob(cupsd_job_t *job) /* I - Job */ +{ + char filename[1024], /* Job control filename */ + newfile[1024]; /* New job control filename */ + cups_file_t *fp; /* Job file */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdSaveJob(job=%p(%d)): job->attrs=%p", + job, job->id, job->attrs); + + snprintf(filename, sizeof(filename), "%s/c%05d", RequestRoot, job->id); + snprintf(newfile, sizeof(newfile), "%s/c%05d.N", RequestRoot, job->id); + + if ((fp = cupsFileOpen(newfile, "w")) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Job %d] Unable to create job control file \"%s\": %s", + job->id, newfile, strerror(errno)); + return; + } + + fchmod(cupsFileNumber(fp), 0600); + fchown(cupsFileNumber(fp), RunUser, Group); + + job->attrs->state = IPP_IDLE; + + if (ippWriteIO(fp, (ipp_iocb_t)cupsFileWrite, 1, NULL, + job->attrs) != IPP_DATA) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Job %d] Unable to write job control file.", job->id); + cupsFileClose(fp); + unlink(newfile); + return; + } + + if (cupsFileClose(fp)) + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Job %d] Unable to close job control file: %s", + job->id, strerror(errno)); + else + { + unlink(filename); + if (rename(newfile, filename)) + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Job %d] Unable to finalize job control file: %s", + job->id, strerror(errno)); + else + job->dirty = 0; + } +} + + +/* + * 'cupsdSetJobHoldUntil()' - Set the hold time for a job. + */ + +void +cupsdSetJobHoldUntil(cupsd_job_t *job, /* I - Job */ + const char *when, /* I - When to resume */ + int update)/* I - Update job-hold-until attr? */ +{ + time_t curtime; /* Current time */ + struct tm *curdate; /* Current date */ + int hour; /* Hold hour */ + int minute; /* Hold minute */ + int second = 0; /* Hold second */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdSetJobHoldUntil(job=%p(%d), when=\"%s\", update=%d)", + job, job->id, when, update); + + if (update) + { + /* + * Update the job-hold-until attribute... + */ + + ipp_attribute_t *attr; /* job-hold-until attribute */ + + if ((attr = ippFindAttribute(job->attrs, "job-hold-until", + IPP_TAG_KEYWORD)) == NULL) + attr = ippFindAttribute(job->attrs, "job-hold-until", IPP_TAG_NAME); + + if (attr) + cupsdSetString(&(attr->values[0].string.text), when); + else + attr = ippAddString(job->attrs, IPP_TAG_JOB, IPP_TAG_KEYWORD, + "job-hold-until", NULL, when); + + if (attr) + { + if (isdigit(when[0] & 255)) + attr->value_tag = IPP_TAG_NAME; + else + attr->value_tag = IPP_TAG_KEYWORD; + + job->dirty = 1; + cupsdMarkDirty(CUPSD_DIRTY_JOBS); + } + } + + /* + * Update the hold time... + */ + + if (!strcmp(when, "indefinite") || !strcmp(when, "auth-info-required")) + { + /* + * Hold indefinitely... + */ + + job->hold_until = 0; + } + else if (!strcmp(when, "day-time")) + { + /* + * Hold to 6am the next morning unless local time is < 6pm. + */ + + curtime = time(NULL); + curdate = localtime(&curtime); + + if (curdate->tm_hour < 18) + job->hold_until = curtime; + else + job->hold_until = curtime + + ((29 - curdate->tm_hour) * 60 + 59 - + curdate->tm_min) * 60 + 60 - curdate->tm_sec; + } + else if (!strcmp(when, "evening") || !strcmp(when, "night")) + { + /* + * Hold to 6pm unless local time is > 6pm or < 6am. + */ + + curtime = time(NULL); + curdate = localtime(&curtime); + + if (curdate->tm_hour < 6 || curdate->tm_hour >= 18) + job->hold_until = curtime; + else + job->hold_until = curtime + + ((17 - curdate->tm_hour) * 60 + 59 - + curdate->tm_min) * 60 + 60 - curdate->tm_sec; + } + else if (!strcmp(when, "second-shift")) + { + /* + * Hold to 4pm unless local time is > 4pm. + */ + + curtime = time(NULL); + curdate = localtime(&curtime); + + if (curdate->tm_hour >= 16) + job->hold_until = curtime; + else + job->hold_until = curtime + + ((15 - curdate->tm_hour) * 60 + 59 - + curdate->tm_min) * 60 + 60 - curdate->tm_sec; + } + else if (!strcmp(when, "third-shift")) + { + /* + * Hold to 12am unless local time is < 8am. + */ + + curtime = time(NULL); + curdate = localtime(&curtime); + + if (curdate->tm_hour < 8) + job->hold_until = curtime; + else + job->hold_until = curtime + + ((23 - curdate->tm_hour) * 60 + 59 - + curdate->tm_min) * 60 + 60 - curdate->tm_sec; + } + else if (!strcmp(when, "weekend")) + { + /* + * Hold to weekend unless we are in the weekend. + */ + + curtime = time(NULL); + curdate = localtime(&curtime); + + if (curdate->tm_wday == 0 || curdate->tm_wday == 6) + job->hold_until = curtime; + else + job->hold_until = curtime + + (((5 - curdate->tm_wday) * 24 + + (17 - curdate->tm_hour)) * 60 + 59 - + curdate->tm_min) * 60 + 60 - curdate->tm_sec; + } + else if (sscanf(when, "%d:%d:%d", &hour, &minute, &second) >= 2) + { + /* + * Hold to specified GMT time (HH:MM or HH:MM:SS)... + */ + + curtime = time(NULL); + curdate = gmtime(&curtime); + + job->hold_until = curtime + + ((hour - curdate->tm_hour) * 60 + minute - + curdate->tm_min) * 60 + second - curdate->tm_sec; + + /* + * Hold until next day as needed... + */ + + if (job->hold_until < curtime) + job->hold_until += 24 * 60 * 60; + } + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdSetJobHoldUntil: hold_until=%d", + (int)job->hold_until); +} + + +/* + * 'cupsdSetJobPriority()' - Set the priority of a job, moving it up/down in + * the list as needed. + */ + +void +cupsdSetJobPriority( + cupsd_job_t *job, /* I - Job ID */ + int priority) /* I - New priority (0 to 100) */ +{ + ipp_attribute_t *attr; /* Job attribute */ + + + /* + * Don't change completed jobs... + */ + + if (job->state_value >= IPP_JOB_PROCESSING) + return; + + /* + * Set the new priority and re-add the job into the active list... + */ + + cupsArrayRemove(ActiveJobs, job); + + job->priority = priority; + + if ((attr = ippFindAttribute(job->attrs, "job-priority", + IPP_TAG_INTEGER)) != NULL) + attr->values[0].integer = priority; + else + ippAddInteger(job->attrs, IPP_TAG_JOB, IPP_TAG_INTEGER, "job-priority", + priority); + + cupsArrayAdd(ActiveJobs, job); + + job->dirty = 1; + cupsdMarkDirty(CUPSD_DIRTY_JOBS); +} + + +/* + * 'cupsdSetJobState()' - Set the state of the specified print job. + */ + +void +cupsdSetJobState( + cupsd_job_t *job, /* I - Job to cancel */ + ipp_jstate_t newstate, /* I - New job state */ + cupsd_jobaction_t action, /* I - Action to take */ + const char *message, /* I - Message to log */ + ...) /* I - Additional arguments as needed */ +{ + int i; /* Looping var */ + ipp_jstate_t oldstate; /* Old state */ + char filename[1024]; /* Job filename */ + ipp_attribute_t *attr; /* Job attribute */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdSetJobState(job=%p(%d), state=%d, newstate=%d, " + "action=%d, message=\"%s\")", job, job->id, job->state_value, + newstate, action, message ? message : "(null)"); + + + /* + * Make sure we have the job attributes... + */ + + if (!cupsdLoadJob(job)) + return; + + /* + * Don't do anything if the state is unchanged and we aren't purging the + * job... + */ + + oldstate = job->state_value; + if (newstate == oldstate && action != CUPSD_JOB_PURGE) + return; + + /* + * Stop any processes that are working on the current job... + */ + + if (oldstate == IPP_JOB_PROCESSING) + stop_job(job, action); + + /* + * Set the new job state... + */ + + job->state->values[0].integer = newstate; + job->state_value = newstate; + + switch (newstate) + { + case IPP_JOB_PENDING : + /* + * Update job-hold-until as needed... + */ + + if ((attr = ippFindAttribute(job->attrs, "job-hold-until", + IPP_TAG_KEYWORD)) == NULL) + attr = ippFindAttribute(job->attrs, "job-hold-until", IPP_TAG_NAME); + + if (attr) + { + attr->value_tag = IPP_TAG_KEYWORD; + cupsdSetString(&(attr->values[0].string.text), "no-hold"); + } + + default : + break; + + case IPP_JOB_ABORTED : + case IPP_JOB_CANCELED : + case IPP_JOB_COMPLETED : + set_time(job, "time-at-completed"); + ippSetString(job->attrs, &job->reasons, 0, "processing-to-stop-point"); + break; + } + + /* + * Log message as needed... + */ + + if (message) + { + char buffer[2048]; /* Message buffer */ + va_list ap; /* Pointer to additional arguments */ + + va_start(ap, message); + vsnprintf(buffer, sizeof(buffer), message, ap); + va_end(ap); + + if (newstate > IPP_JOB_STOPPED) + cupsdAddEvent(CUPSD_EVENT_JOB_COMPLETED, job->printer, job, "%s", buffer); + else + cupsdAddEvent(CUPSD_EVENT_JOB_STATE, job->printer, job, "%s", buffer); + + if (newstate == IPP_JOB_STOPPED || newstate == IPP_JOB_ABORTED) + cupsdLogJob(job, CUPSD_LOG_ERROR, "%s", buffer); + else + cupsdLogJob(job, CUPSD_LOG_INFO, "%s", buffer); + } + + /* + * Handle post-state-change actions... + */ + + switch (newstate) + { + case IPP_JOB_PROCESSING : + /* + * Add the job to the "printing" list... + */ + + if (!cupsArrayFind(PrintingJobs, job)) + cupsArrayAdd(PrintingJobs, job); + + /* + * Set the processing time... + */ + + set_time(job, "time-at-processing"); + + case IPP_JOB_PENDING : + case IPP_JOB_HELD : + case IPP_JOB_STOPPED : + /* + * Make sure the job is in the active list... + */ + + if (!cupsArrayFind(ActiveJobs, job)) + cupsArrayAdd(ActiveJobs, job); + + /* + * Save the job state to disk... + */ + + job->dirty = 1; + cupsdMarkDirty(CUPSD_DIRTY_JOBS); + break; + + case IPP_JOB_ABORTED : + case IPP_JOB_CANCELED : + case IPP_JOB_COMPLETED : + if (newstate == IPP_JOB_CANCELED) + { + /* + * Remove the job from the active list if there are no processes still + * running for it... + */ + + for (i = 0; job->filters[i] < 0; i++); + + if (!job->filters[i] && job->backend <= 0) + cupsArrayRemove(ActiveJobs, job); + } + else + { + /* + * Otherwise just remove the job from the active list immediately... + */ + + cupsArrayRemove(ActiveJobs, job); + } + + /* + * Expire job subscriptions since the job is now "completed"... + */ + + cupsdExpireSubscriptions(NULL, job); + +#ifdef __APPLE__ + /* + * If we are going to sleep and the PrintingJobs count is now 0, allow the + * sleep to happen immediately... + */ + + if (Sleeping && cupsArrayCount(PrintingJobs) == 0) + cupsdAllowSleep(); +#endif /* __APPLE__ */ + + /* + * Remove any authentication data... + */ + + snprintf(filename, sizeof(filename), "%s/a%05d", RequestRoot, job->id); + if (cupsdRemoveFile(filename) && errno != ENOENT) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to remove authentication cache: %s", + strerror(errno)); + + for (i = 0; + i < (int)(sizeof(job->auth_env) / sizeof(job->auth_env[0])); + i ++) + cupsdClearString(job->auth_env + i); + + cupsdClearString(&job->auth_uid); + + /* + * Remove the print file for good if we aren't preserving jobs or + * files... + */ + + if (!JobHistory || !JobFiles || action == CUPSD_JOB_PURGE) + { + for (i = 1; i <= job->num_files; i ++) + { + snprintf(filename, sizeof(filename), "%s/d%05d-%03d", RequestRoot, + job->id, i); + if (Classification) + cupsdRemoveFile(filename); + else + unlink(filename); + } + + if (job->num_files > 0) + { + free(job->filetypes); + free(job->compressions); + + job->num_files = 0; + job->filetypes = NULL; + job->compressions = NULL; + } + } + + if (JobHistory && action != CUPSD_JOB_PURGE) + { + /* + * Save job state info... + */ + + job->dirty = 1; + cupsdMarkDirty(CUPSD_DIRTY_JOBS); + } + else if (!job->printer) + { + /* + * Delete the job immediately if not actively printing... + */ + + cupsdDeleteJob(job, CUPSD_JOB_PURGE); + job = NULL; + } + break; + } + + /* + * Finalize the job immediately if we forced things... + */ + + if (action >= CUPSD_JOB_FORCE && job && job->printer) + finalize_job(job, 0); + + /* + * Update the server "busy" state... + */ + + cupsdSetBusyState(); +} + + +/* + * 'cupsdStopAllJobs()' - Stop all print jobs. + */ + +void +cupsdStopAllJobs( + cupsd_jobaction_t action, /* I - Action */ + int kill_delay) /* I - Number of seconds before we kill */ +{ + cupsd_job_t *job; /* Current job */ + + + DEBUG_puts("cupsdStopAllJobs()"); + + for (job = (cupsd_job_t *)cupsArrayFirst(PrintingJobs); + job; + job = (cupsd_job_t *)cupsArrayNext(PrintingJobs)) + { + if (kill_delay) + job->kill_time = time(NULL) + kill_delay; + + cupsdSetJobState(job, IPP_JOB_PENDING, action, NULL); + } +} + + +/* + * 'cupsdUnloadCompletedJobs()' - Flush completed job history from memory. + */ + +void +cupsdUnloadCompletedJobs(void) +{ + cupsd_job_t *job; /* Current job */ + time_t expire; /* Expiration time */ + + + expire = time(NULL) - 60; + + for (job = (cupsd_job_t *)cupsArrayFirst(Jobs); + job; + job = (cupsd_job_t *)cupsArrayNext(Jobs)) + if (job->attrs && job->state_value >= IPP_JOB_STOPPED && !job->printer && + job->access_time < expire) + { + if (job->dirty) + cupsdSaveJob(job); + + unload_job(job); + } +} + + +/* + * 'compare_active_jobs()' - Compare the job IDs and priorities of two jobs. + */ + +static int /* O - Difference */ +compare_active_jobs(void *first, /* I - First job */ + void *second, /* I - Second job */ + void *data) /* I - App data (not used) */ +{ + int diff; /* Difference */ + + + (void)data; + + if ((diff = ((cupsd_job_t *)second)->priority - + ((cupsd_job_t *)first)->priority) != 0) + return (diff); + else + return (((cupsd_job_t *)first)->id - ((cupsd_job_t *)second)->id); +} + + +/* + * 'compare_jobs()' - Compare the job IDs of two jobs. + */ + +static int /* O - Difference */ +compare_jobs(void *first, /* I - First job */ + void *second, /* I - Second job */ + void *data) /* I - App data (not used) */ +{ + (void)data; + + return (((cupsd_job_t *)first)->id - ((cupsd_job_t *)second)->id); +} + + +/* + * 'dump_job_history()' - Dump any debug messages for a job. + */ + +static void +dump_job_history(cupsd_job_t *job) /* I - Job */ +{ + int i, /* Looping var */ + oldsize; /* Current MaxLogSize */ + struct tm *date; /* Date/time value */ + cupsd_joblog_t *message; /* Current message */ + char temp[2048], /* Log message */ + *ptr, /* Pointer into log message */ + start[256], /* Start time */ + end[256]; /* End time */ + cupsd_printer_t *printer; /* Printer for job */ + + + /* + * See if we have anything to dump... + */ + + if (!job->history) + return; + + /* + * Disable log rotation temporarily... + */ + + oldsize = MaxLogSize; + MaxLogSize = 0; + + /* + * Copy the debug messages to the log... + */ + + message = (cupsd_joblog_t *)cupsArrayFirst(job->history); + date = localtime(&(message->time)); + strftime(start, sizeof(start), "%X", date); + + message = (cupsd_joblog_t *)cupsArrayLast(job->history); + date = localtime(&(message->time)); + strftime(end, sizeof(end), "%X", date); + + snprintf(temp, sizeof(temp), + "[Job %d] The following messages were recorded from %s to %s", + job->id, start, end); + cupsdWriteErrorLog(CUPSD_LOG_DEBUG, temp); + + for (message = (cupsd_joblog_t *)cupsArrayFirst(job->history); + message; + message = (cupsd_joblog_t *)cupsArrayNext(job->history)) + cupsdWriteErrorLog(CUPSD_LOG_DEBUG, message->message); + + snprintf(temp, sizeof(temp), "[Job %d] End of messages", job->id); + cupsdWriteErrorLog(CUPSD_LOG_DEBUG, temp); + + /* + * Log the printer state values... + */ + + if ((printer = job->printer) == NULL) + printer = cupsdFindDest(job->dest); + + if (printer) + { + snprintf(temp, sizeof(temp), "[Job %d] printer-state=%d(%s)", job->id, + printer->state, + printer->state == IPP_PRINTER_IDLE ? "idle" : + printer->state == IPP_PRINTER_PROCESSING ? "processing" : + "stopped"); + cupsdWriteErrorLog(CUPSD_LOG_DEBUG, temp); + + snprintf(temp, sizeof(temp), "[Job %d] printer-state-message=\"%s\"", + job->id, printer->state_message); + cupsdWriteErrorLog(CUPSD_LOG_DEBUG, temp); + + snprintf(temp, sizeof(temp), "[Job %d] printer-state-reasons=", job->id); + ptr = temp + strlen(temp); + if (printer->num_reasons == 0) + strlcpy(ptr, "none", sizeof(temp) - (ptr - temp)); + else + { + for (i = 0; + i < printer->num_reasons && ptr < (temp + sizeof(temp) - 2); + i ++) + { + if (i) + *ptr++ = ','; + + strlcpy(ptr, printer->reasons[i], sizeof(temp) - (ptr - temp)); + ptr += strlen(ptr); + } + } + cupsdWriteErrorLog(CUPSD_LOG_DEBUG, temp); + } + + /* + * Restore log file rotation... + */ + + MaxLogSize = oldsize; + + /* + * Free all messages... + */ + + free_job_history(job); +} + + +/* + * 'free_job_history()' - Free any log history. + */ + +static void +free_job_history(cupsd_job_t *job) /* I - Job */ +{ + char *message; /* Current message */ + + + if (!job->history) + return; + + for (message = (char *)cupsArrayFirst(job->history); + message; + message = (char *)cupsArrayNext(job->history)) + free(message); + + cupsArrayDelete(job->history); + job->history = NULL; +} + + +/* + * 'finalize_job()' - Cleanup after job filter processes and support data. + */ + +static void +finalize_job(cupsd_job_t *job, /* I - Job */ + int set_job_state) /* I - 1 = set the job state */ +{ + ipp_pstate_t printer_state; /* New printer state value */ + ipp_jstate_t job_state; /* New job state value */ + const char *message; /* Message for job state */ + char buffer[1024]; /* Buffer for formatted messages */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "finalize_job(job=%p(%d))", job, job->id); + + /* + * Clear the "connecting-to-device" reason, which is only valid when a printer + * is processing, along with any remote printing job state... + */ + + cupsdSetPrinterReasons(job->printer, "-connecting-to-device," + "cups-remote-pending," + "cups-remote-pending-held," + "cups-remote-processing," + "cups-remote-stopped," + "cups-remote-canceled," + "cups-remote-aborted," + "cups-remote-completed"); + + /* + * Similarly, clear the "offline-report" reason for non-USB devices since we + * rarely have current information for network devices... + */ + + if (strncmp(job->printer->device_uri, "usb:", 4)) + cupsdSetPrinterReasons(job->printer, "-offline-report"); + + /* + * Free the security profile... + */ + + cupsdDestroyProfile(job->profile); + job->profile = NULL; + + /* + * Clear the unresponsive job watchdog timer... + */ + + job->kill_time = 0; + + /* + * Close pipes and status buffer... + */ + + cupsdClosePipe(job->print_pipes); + cupsdClosePipe(job->back_pipes); + cupsdClosePipe(job->side_pipes); + + cupsdRemoveSelect(job->status_pipes[0]); + cupsdClosePipe(job->status_pipes); + cupsdStatBufDelete(job->status_buffer); + job->status_buffer = NULL; + + /* + * Process the exit status... + */ + + if (job->printer->state == IPP_PRINTER_PROCESSING) + printer_state = IPP_PRINTER_IDLE; + else + printer_state = job->printer->state; + + switch (job_state = job->state_value) + { + case IPP_JOB_PENDING : + message = "Job paused."; + break; + + case IPP_JOB_HELD : + message = "Job held."; + break; + + default : + case IPP_JOB_PROCESSING : + case IPP_JOB_COMPLETED : + job_state = IPP_JOB_COMPLETED; + message = "Job completed."; + + ippSetString(job->attrs, &job->reasons, 0, + "job-completed-successfully"); + break; + + case IPP_JOB_STOPPED : + message = "Job stopped."; + + ippSetString(job->attrs, &job->reasons, 0, "job-stopped"); + break; + + case IPP_JOB_CANCELED : + message = "Job canceled."; + + ippSetString(job->attrs, &job->reasons, 0, "job-canceled-by-user"); + break; + + case IPP_JOB_ABORTED : + message = "Job aborted."; + break; + } + + if (job->status < 0) + { + /* + * Backend had errors... + */ + + int exit_code; /* Exit code from backend */ + + /* + * Convert the status to an exit code. Due to the way the W* macros are + * implemented on MacOS X (bug?), we have to store the exit status in a + * variable first and then convert... + */ + + exit_code = -job->status; + if (WIFEXITED(exit_code)) + exit_code = WEXITSTATUS(exit_code); + else + { + ippSetString(job->attrs, &job->reasons, 0, "cups-backend-crashed"); + exit_code = job->status; + } + + cupsdLogJob(job, CUPSD_LOG_INFO, "Backend returned status %d (%s)", + exit_code, + exit_code == CUPS_BACKEND_FAILED ? "failed" : + exit_code == CUPS_BACKEND_AUTH_REQUIRED ? + "authentication required" : + exit_code == CUPS_BACKEND_HOLD ? "hold job" : + exit_code == CUPS_BACKEND_STOP ? "stop printer" : + exit_code == CUPS_BACKEND_CANCEL ? "cancel job" : + exit_code < 0 ? "crashed" : "unknown"); + + /* + * Do what needs to be done... + */ + + switch (exit_code) + { + default : + case CUPS_BACKEND_FAILED : + /* + * Backend failure, use the error-policy to determine how to + * act... + */ + + if (job->dtype & CUPS_PRINTER_CLASS) + { + /* + * Queued on a class - mark the job as pending and we'll retry on + * another printer... + */ + + if (job_state == IPP_JOB_COMPLETED) + { + job_state = IPP_JOB_PENDING; + message = "Retrying job on another printer."; + + ippSetString(job->attrs, &job->reasons, 0, + "resources-are-not-ready"); + } + } + else if (!strcmp(job->printer->error_policy, "retry-current-job")) + { + /* + * The error policy is "retry-current-job" - mark the job as pending + * and we'll retry on the same printer... + */ + + if (job_state == IPP_JOB_COMPLETED) + { + job_state = IPP_JOB_PENDING; + message = "Retrying job on same printer."; + + ippSetString(job->attrs, &job->reasons, 0, "none"); + } + } + else if ((job->printer->type & CUPS_PRINTER_FAX) || + !strcmp(job->printer->error_policy, "retry-job")) + { + if (job_state == IPP_JOB_COMPLETED) + { + /* + * The job was queued on a fax or the error policy is "retry-job" - + * hold the job if the number of retries is less than the + * JobRetryLimit, otherwise abort the job. + */ + + job->tries ++; + + if (job->tries > JobRetryLimit && JobRetryLimit > 0) + { + /* + * Too many tries... + */ + + snprintf(buffer, sizeof(buffer), + "Job aborted after %d unsuccessful attempts.", + JobRetryLimit); + job_state = IPP_JOB_ABORTED; + message = buffer; + + ippSetString(job->attrs, &job->reasons, 0, "aborted-by-system"); + } + else + { + /* + * Try again in N seconds... + */ + + snprintf(buffer, sizeof(buffer), + "Job held for %d seconds since it could not be sent.", + JobRetryInterval); + + job->hold_until = time(NULL) + JobRetryInterval; + job_state = IPP_JOB_HELD; + message = buffer; + + ippSetString(job->attrs, &job->reasons, 0, + "resources-are-not-ready"); + } + } + } + else if (!strcmp(job->printer->error_policy, "abort-job") && + job_state == IPP_JOB_COMPLETED) + { + job_state = IPP_JOB_ABORTED; + message = "Job aborted due to backend errors; please consult " + "the error_log file for details."; + + ippSetString(job->attrs, &job->reasons, 0, "aborted-by-system"); + } + else if (job->state_value == IPP_JOB_PROCESSING) + { + job_state = IPP_JOB_PENDING; + printer_state = IPP_PRINTER_STOPPED; + message = "Printer stopped due to backend errors; please " + "consult the error_log file for details."; + + ippSetString(job->attrs, &job->reasons, 0, "none"); + } + break; + + case CUPS_BACKEND_CANCEL : + /* + * Abort the job... + */ + + if (job_state == IPP_JOB_COMPLETED) + { + job_state = IPP_JOB_ABORTED; + message = "Job aborted due to backend errors; please consult " + "the error_log file for details."; + + ippSetString(job->attrs, &job->reasons, 0, "aborted-by-system"); + } + break; + + case CUPS_BACKEND_HOLD : + if (job_state == IPP_JOB_COMPLETED) + { + /* + * Hold the job... + */ + + cupsdSetJobHoldUntil(job, "indefinite", 1); + ippSetString(job->attrs, &job->reasons, 0, + "job-hold-until-specified"); + + job_state = IPP_JOB_HELD; + message = "Job held indefinitely due to backend errors; please " + "consult the error_log file for details."; + } + break; + + case CUPS_BACKEND_STOP : + /* + * Stop the printer... + */ + + printer_state = IPP_PRINTER_STOPPED; + message = "Printer stopped due to backend errors; please " + "consult the error_log file for details."; + + if (job_state == IPP_JOB_COMPLETED) + { + job_state = IPP_JOB_PENDING; + + ippSetString(job->attrs, &job->reasons, 0, + "resources-are-not-ready"); + } + break; + + case CUPS_BACKEND_AUTH_REQUIRED : + /* + * Hold the job for authentication... + */ + + if (job_state == IPP_JOB_COMPLETED) + { + cupsdSetJobHoldUntil(job, "auth-info-required", 1); + + job_state = IPP_JOB_HELD; + message = "Job held for authentication."; + + ippSetString(job->attrs, &job->reasons, 0, + "cups-held-for-authentication"); + } + break; + + case CUPS_BACKEND_RETRY : + if (job_state == IPP_JOB_COMPLETED) + { + /* + * Hold the job if the number of retries is less than the + * JobRetryLimit, otherwise abort the job. + */ + + job->tries ++; + + if (job->tries > JobRetryLimit && JobRetryLimit > 0) + { + /* + * Too many tries... + */ + + snprintf(buffer, sizeof(buffer), + "Job aborted after %d unsuccessful attempts.", + JobRetryLimit); + job_state = IPP_JOB_ABORTED; + message = buffer; + + ippSetString(job->attrs, &job->reasons, 0, "aborted-by-system"); + } + else + { + /* + * Try again in N seconds... + */ + + snprintf(buffer, sizeof(buffer), + "Job held for %d seconds since it could not be sent.", + JobRetryInterval); + + job->hold_until = time(NULL) + JobRetryInterval; + job_state = IPP_JOB_HELD; + message = buffer; + + ippSetString(job->attrs, &job->reasons, 0, + "resources-are-not-ready"); + } + } + break; + + case CUPS_BACKEND_RETRY_CURRENT : + /* + * Mark the job as pending and retry on the same printer... + */ + + if (job_state == IPP_JOB_COMPLETED) + { + job_state = IPP_JOB_PENDING; + message = "Retrying job on same printer."; + + ippSetString(job->attrs, &job->reasons, 0, "none"); + } + break; + } + } + else if (job->status > 0) + { + /* + * Filter had errors; stop job... + */ + + if (job_state == IPP_JOB_COMPLETED) + { + job_state = IPP_JOB_STOPPED; + message = "Job stopped due to filter errors; please consult the " + "error_log file for details."; + + if (WIFSIGNALED(job->status)) + ippSetString(job->attrs, &job->reasons, 0, "cups-filter-crashed"); + else + ippSetString(job->attrs, &job->reasons, 0, "job-completed-with-errors"); + } + } + + /* + * Update the printer and job state. + */ + + if (set_job_state && job_state != job->state_value) + cupsdSetJobState(job, job_state, CUPSD_JOB_DEFAULT, "%s", message); + + cupsdSetPrinterState(job->printer, printer_state, + printer_state == IPP_PRINTER_STOPPED); + update_job_attrs(job, 0); + + if (job->history) + { + if (job->status && + (job->state_value == IPP_JOB_ABORTED || + job->state_value == IPP_JOB_STOPPED)) + dump_job_history(job); + else + free_job_history(job); + } + + cupsArrayRemove(PrintingJobs, job); + + /* + * Clear the printer <-> job association... + */ + + job->printer->job = NULL; + job->printer = NULL; + + /* + * Try printing another job... + */ + + if (printer_state != IPP_PRINTER_STOPPED) + cupsdCheckJobs(); +} + + +/* + * 'get_options()' - Get a string containing the job options. + */ + +static char * /* O - Options string */ +get_options(cupsd_job_t *job, /* I - Job */ + int banner_page, /* I - Printing a banner page? */ + char *copies, /* I - Copies buffer */ + size_t copies_size, /* I - Size of copies buffer */ + char *title, /* I - Title buffer */ + size_t title_size) /* I - Size of title buffer */ +{ + int i; /* Looping var */ + size_t newlength; /* New option buffer length */ + char *optptr, /* Pointer to options */ + *valptr; /* Pointer in value string */ + ipp_attribute_t *attr; /* Current attribute */ + _ppd_cache_t *pc; /* PPD cache and mapping data */ + int num_pwgppds; /* Number of PWG->PPD options */ + cups_option_t *pwgppds, /* PWG->PPD options */ + *pwgppd, /* Current PWG->PPD option */ + *preset; /* Current preset option */ + int print_color_mode, + /* Output mode (if any) */ + print_quality; /* Print quality (if any) */ + const char *ppd; /* PPD option choice */ + int exact; /* Did we get an exact match? */ + static char *options = NULL;/* Full list of options */ + static size_t optlength = 0; /* Length of option buffer */ + + + /* + * Building the options string is harder than it needs to be, but for the + * moment we need to pass strings for command-line args and not IPP attribute + * pointers... :) + * + * First build an options array for any PWG->PPD mapped option/choice pairs. + */ + + pc = job->printer->pc; + num_pwgppds = 0; + pwgppds = NULL; + + if (pc && + !ippFindAttribute(job->attrs, + "com.apple.print.DocumentTicket.PMSpoolFormat", + IPP_TAG_ZERO) && + !ippFindAttribute(job->attrs, "APPrinterPreset", IPP_TAG_ZERO) && + (ippFindAttribute(job->attrs, "output-mode", IPP_TAG_ZERO) || + ippFindAttribute(job->attrs, "print-color-mode", IPP_TAG_ZERO) || + ippFindAttribute(job->attrs, "print-quality", IPP_TAG_ZERO))) + { + /* + * Map output-mode and print-quality to a preset... + */ + + if ((attr = ippFindAttribute(job->attrs, "print-color-mode", + IPP_TAG_KEYWORD)) == NULL) + attr = ippFindAttribute(job->attrs, "output-mode", IPP_TAG_KEYWORD); + + if (attr && !strcmp(attr->values[0].string.text, "monochrome")) + print_color_mode = _PWG_PRINT_COLOR_MODE_MONOCHROME; + else + print_color_mode = _PWG_PRINT_COLOR_MODE_COLOR; + + if ((attr = ippFindAttribute(job->attrs, "print-quality", + IPP_TAG_ENUM)) != NULL && + attr->values[0].integer >= IPP_QUALITY_DRAFT && + attr->values[0].integer <= IPP_QUALITY_HIGH) + print_quality = attr->values[0].integer - IPP_QUALITY_DRAFT; + else + print_quality = _PWG_PRINT_QUALITY_NORMAL; + + if (pc->num_presets[print_color_mode][print_quality] == 0) + { + /* + * Try to find a preset that works so that we maximize the chances of us + * getting a good print using IPP attributes. + */ + + if (pc->num_presets[print_color_mode][_PWG_PRINT_QUALITY_NORMAL] > 0) + print_quality = _PWG_PRINT_QUALITY_NORMAL; + else if (pc->num_presets[_PWG_PRINT_COLOR_MODE_COLOR][print_quality] > 0) + print_color_mode = _PWG_PRINT_COLOR_MODE_COLOR; + else + { + print_quality = _PWG_PRINT_QUALITY_NORMAL; + print_color_mode = _PWG_PRINT_COLOR_MODE_COLOR; + } + } + + if (pc->num_presets[print_color_mode][print_quality] > 0) + { + /* + * Copy the preset options as long as the corresponding names are not + * already defined in the IPP request... + */ + + for (i = pc->num_presets[print_color_mode][print_quality], + preset = pc->presets[print_color_mode][print_quality]; + i > 0; + i --, preset ++) + { + if (!ippFindAttribute(job->attrs, preset->name, IPP_TAG_ZERO)) + num_pwgppds = cupsAddOption(preset->name, preset->value, num_pwgppds, + &pwgppds); + } + } + } + + if (pc) + { + if (!ippFindAttribute(job->attrs, "InputSlot", IPP_TAG_ZERO) && + !ippFindAttribute(job->attrs, "HPPaperSource", IPP_TAG_ZERO)) + { + if ((ppd = _ppdCacheGetInputSlot(pc, job->attrs, NULL)) != NULL) + num_pwgppds = cupsAddOption(pc->source_option, ppd, num_pwgppds, + &pwgppds); + else if (!ippFindAttribute(job->attrs, "AP_D_InputSlot", IPP_TAG_ZERO)) + num_pwgppds = cupsAddOption("AP_D_InputSlot", "", num_pwgppds, + &pwgppds); + } + if (!ippFindAttribute(job->attrs, "MediaType", IPP_TAG_ZERO) && + (ppd = _ppdCacheGetMediaType(pc, job->attrs, NULL)) != NULL) + num_pwgppds = cupsAddOption("MediaType", ppd, num_pwgppds, &pwgppds); + + if (!ippFindAttribute(job->attrs, "PageRegion", IPP_TAG_ZERO) && + !ippFindAttribute(job->attrs, "PageSize", IPP_TAG_ZERO) && + (ppd = _ppdCacheGetPageSize(pc, job->attrs, NULL, &exact)) != NULL) + { + num_pwgppds = cupsAddOption("PageSize", ppd, num_pwgppds, &pwgppds); + + if (!ippFindAttribute(job->attrs, "media", IPP_TAG_ZERO)) + num_pwgppds = cupsAddOption("media", ppd, num_pwgppds, &pwgppds); + } + + if (!ippFindAttribute(job->attrs, "OutputBin", IPP_TAG_ZERO) && + (attr = ippFindAttribute(job->attrs, "output-bin", + IPP_TAG_ZERO)) != NULL && + (attr->value_tag == IPP_TAG_KEYWORD || + attr->value_tag == IPP_TAG_NAME) && + (ppd = _ppdCacheGetOutputBin(pc, attr->values[0].string.text)) != NULL) + { + /* + * Map output-bin to OutputBin option... + */ + + num_pwgppds = cupsAddOption("OutputBin", ppd, num_pwgppds, &pwgppds); + } + + if (pc->sides_option && + !ippFindAttribute(job->attrs, pc->sides_option, IPP_TAG_ZERO) && + (attr = ippFindAttribute(job->attrs, "sides", IPP_TAG_KEYWORD)) != NULL) + { + /* + * Map sides to duplex option... + */ + + if (!strcmp(attr->values[0].string.text, "one-sided")) + num_pwgppds = cupsAddOption(pc->sides_option, pc->sides_1sided, + num_pwgppds, &pwgppds); + else if (!strcmp(attr->values[0].string.text, "two-sided-long-edge")) + num_pwgppds = cupsAddOption(pc->sides_option, pc->sides_2sided_long, + num_pwgppds, &pwgppds); + else if (!strcmp(attr->values[0].string.text, "two-sided-short-edge")) + num_pwgppds = cupsAddOption(pc->sides_option, pc->sides_2sided_short, + num_pwgppds, &pwgppds); + } + + /* + * Map finishings values... + */ + + num_pwgppds = _ppdCacheGetFinishingOptions(pc, job->attrs, + IPP_FINISHINGS_NONE, num_pwgppds, + &pwgppds); + } + + /* + * Figure out how much room we need... + */ + + newlength = ipp_length(job->attrs); + + for (i = num_pwgppds, pwgppd = pwgppds; i > 0; i --, pwgppd ++) + newlength += 1 + strlen(pwgppd->name) + 1 + strlen(pwgppd->value); + + /* + * Then allocate/reallocate the option buffer as needed... + */ + + if (newlength == 0) /* This can never happen, but Clang */ + newlength = 1; /* thinks it can... */ + + if (newlength > optlength || !options) + { + if (!options) + optptr = malloc(newlength); + else + optptr = realloc(options, newlength); + + if (!optptr) + { + cupsdLogJob(job, CUPSD_LOG_CRIT, + "Unable to allocate " CUPS_LLFMT " bytes for option buffer.", + CUPS_LLCAST newlength); + return (NULL); + } + + options = optptr; + optlength = newlength; + } + + /* + * Now loop through the attributes and convert them to the textual + * representation used by the filters... + */ + + optptr = options; + *optptr = '\0'; + + snprintf(title, title_size, "%s-%d", job->printer->name, job->id); + strlcpy(copies, "1", copies_size); + + for (attr = job->attrs->attrs; attr != NULL; attr = attr->next) + { + if (!strcmp(attr->name, "copies") && + attr->value_tag == IPP_TAG_INTEGER) + { + /* + * Don't use the # copies attribute if we are printing the job sheets... + */ + + if (!banner_page) + snprintf(copies, copies_size, "%d", attr->values[0].integer); + } + else if (!strcmp(attr->name, "job-name") && + (attr->value_tag == IPP_TAG_NAME || + attr->value_tag == IPP_TAG_NAMELANG)) + strlcpy(title, attr->values[0].string.text, title_size); + else if (attr->group_tag == IPP_TAG_JOB) + { + /* + * Filter out other unwanted attributes... + */ + + if (attr->value_tag == IPP_TAG_NOVALUE || + attr->value_tag == IPP_TAG_MIMETYPE || + attr->value_tag == IPP_TAG_NAMELANG || + attr->value_tag == IPP_TAG_TEXTLANG || + (attr->value_tag == IPP_TAG_URI && strcmp(attr->name, "job-uuid")) || + attr->value_tag == IPP_TAG_URISCHEME || + attr->value_tag == IPP_TAG_BEGIN_COLLECTION) /* Not yet supported */ + continue; + + if (!strcmp(attr->name, "job-hold-until")) + continue; + + if (!strncmp(attr->name, "job-", 4) && + strcmp(attr->name, "job-billing") && + strcmp(attr->name, "job-impressions") && + strcmp(attr->name, "job-originating-host-name") && + strcmp(attr->name, "job-uuid") && + !(job->printer->type & CUPS_PRINTER_REMOTE)) + continue; + + if ((!strcmp(attr->name, "job-impressions") || + !strcmp(attr->name, "page-label") || + !strcmp(attr->name, "page-border") || + !strncmp(attr->name, "number-up", 9) || + !strcmp(attr->name, "page-ranges") || + !strcmp(attr->name, "page-set") || + !_cups_strcasecmp(attr->name, "AP_FIRSTPAGE_InputSlot") || + !_cups_strcasecmp(attr->name, "AP_FIRSTPAGE_ManualFeed") || + !_cups_strcasecmp(attr->name, "com.apple.print.PrintSettings." + "PMTotalSidesImaged..n.") || + !_cups_strcasecmp(attr->name, "com.apple.print.PrintSettings." + "PMTotalBeginPages..n.")) && + banner_page) + continue; + + /* + * Otherwise add them to the list... + */ + + if (optptr > options) + strlcat(optptr, " ", optlength - (optptr - options)); + + if (attr->value_tag != IPP_TAG_BOOLEAN) + { + strlcat(optptr, attr->name, optlength - (optptr - options)); + strlcat(optptr, "=", optlength - (optptr - options)); + } + + for (i = 0; i < attr->num_values; i ++) + { + if (i) + strlcat(optptr, ",", optlength - (optptr - options)); + + optptr += strlen(optptr); + + switch (attr->value_tag) + { + case IPP_TAG_INTEGER : + case IPP_TAG_ENUM : + snprintf(optptr, optlength - (optptr - options), + "%d", attr->values[i].integer); + break; + + case IPP_TAG_BOOLEAN : + if (!attr->values[i].boolean) + strlcat(optptr, "no", optlength - (optptr - options)); + + strlcat(optptr, attr->name, + optlength - (optptr - options)); + break; + + case IPP_TAG_RANGE : + if (attr->values[i].range.lower == attr->values[i].range.upper) + snprintf(optptr, optlength - (optptr - options) - 1, + "%d", attr->values[i].range.lower); + else + snprintf(optptr, optlength - (optptr - options) - 1, + "%d-%d", attr->values[i].range.lower, + attr->values[i].range.upper); + break; + + case IPP_TAG_RESOLUTION : + snprintf(optptr, optlength - (optptr - options) - 1, + "%dx%d%s", attr->values[i].resolution.xres, + attr->values[i].resolution.yres, + attr->values[i].resolution.units == IPP_RES_PER_INCH ? + "dpi" : "dpc"); + break; + + case IPP_TAG_STRING : + case IPP_TAG_TEXT : + case IPP_TAG_NAME : + case IPP_TAG_KEYWORD : + case IPP_TAG_CHARSET : + case IPP_TAG_LANGUAGE : + case IPP_TAG_URI : + for (valptr = attr->values[i].string.text; *valptr;) + { + if (strchr(" \t\n\\\'\"", *valptr)) + *optptr++ = '\\'; + *optptr++ = *valptr++; + } + + *optptr = '\0'; + break; + + default : + break; /* anti-compiler-warning-code */ + } + } + + optptr += strlen(optptr); + } + } + + /* + * Finally loop through the PWG->PPD mapped options and add them... + */ + + for (i = num_pwgppds, pwgppd = pwgppds; i > 0; i --, pwgppd ++) + { + *optptr++ = ' '; + strcpy(optptr, pwgppd->name); + optptr += strlen(optptr); + *optptr++ = '='; + strcpy(optptr, pwgppd->value); + optptr += strlen(optptr); + } + + cupsFreeOptions(num_pwgppds, pwgppds); + + /* + * Return the options string... + */ + + return (options); +} + + +/* + * 'ipp_length()' - Compute the size of the buffer needed to hold + * the textual IPP attributes. + */ + +static size_t /* O - Size of attribute buffer */ +ipp_length(ipp_t *ipp) /* I - IPP request */ +{ + size_t bytes; /* Number of bytes */ + int i; /* Looping var */ + ipp_attribute_t *attr; /* Current attribute */ + + + /* + * Loop through all attributes... + */ + + bytes = 0; + + for (attr = ipp->attrs; attr != NULL; attr = attr->next) + { + /* + * Skip attributes that won't be sent to filters... + */ + + if (attr->value_tag == IPP_TAG_NOVALUE || + attr->value_tag == IPP_TAG_MIMETYPE || + attr->value_tag == IPP_TAG_NAMELANG || + attr->value_tag == IPP_TAG_TEXTLANG || + attr->value_tag == IPP_TAG_URI || + attr->value_tag == IPP_TAG_URISCHEME) + continue; + + /* + * Add space for a leading space and commas between each value. + * For the first attribute, the leading space isn't used, so the + * extra byte can be used as the nul terminator... + */ + + bytes ++; /* " " separator */ + bytes += attr->num_values; /* "," separators */ + + /* + * Boolean attributes appear as "foo,nofoo,foo,nofoo", while + * other attributes appear as "foo=value1,value2,...,valueN". + */ + + if (attr->value_tag != IPP_TAG_BOOLEAN) + bytes += strlen(attr->name); + else + bytes += attr->num_values * strlen(attr->name); + + /* + * Now add the size required for each value in the attribute... + */ + + switch (attr->value_tag) + { + case IPP_TAG_INTEGER : + case IPP_TAG_ENUM : + /* + * Minimum value of a signed integer is -2147483647, or 11 digits. + */ + + bytes += attr->num_values * 11; + break; + + case IPP_TAG_BOOLEAN : + /* + * Add two bytes for each false ("no") value... + */ + + for (i = 0; i < attr->num_values; i ++) + if (!attr->values[i].boolean) + bytes += 2; + break; + + case IPP_TAG_RANGE : + /* + * A range is two signed integers separated by a hyphen, or + * 23 characters max. + */ + + bytes += attr->num_values * 23; + break; + + case IPP_TAG_RESOLUTION : + /* + * A resolution is two signed integers separated by an "x" and + * suffixed by the units, or 26 characters max. + */ + + bytes += attr->num_values * 26; + break; + + case IPP_TAG_STRING : + case IPP_TAG_TEXT : + case IPP_TAG_NAME : + case IPP_TAG_KEYWORD : + case IPP_TAG_CHARSET : + case IPP_TAG_LANGUAGE : + case IPP_TAG_URI : + /* + * Strings can contain characters that need quoting. We need + * at least 2 * len + 2 characters to cover the quotes and + * any backslashes in the string. + */ + + for (i = 0; i < attr->num_values; i ++) + bytes += 2 * strlen(attr->values[i].string.text) + 2; + break; + + default : + break; /* anti-compiler-warning-code */ + } + } + + return (bytes); +} + + +/* + * 'load_job_cache()' - Load jobs from the job.cache file. + */ + +static void +load_job_cache(const char *filename) /* I - job.cache filename */ +{ + cups_file_t *fp; /* job.cache file */ + char line[1024], /* Line buffer */ + *value; /* Value on line */ + int linenum; /* Line number in file */ + cupsd_job_t *job; /* Current job */ + int jobid; /* Job ID */ + char jobfile[1024]; /* Job filename */ + + + /* + * Open the job.cache file... + */ + + if ((fp = cupsdOpenConfFile(filename)) == NULL) + { + load_request_root(); + return; + } + + /* + * Read entries from the job cache file and create jobs as needed. + */ + + cupsdLogMessage(CUPSD_LOG_INFO, "Loading job cache file \"%s\"...", + filename); + + linenum = 0; + job = NULL; + + while (cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + { + if (!_cups_strcasecmp(line, "NextJobId")) + { + if (value) + NextJobId = atoi(value); + } + else if (!_cups_strcasecmp(line, " directive on line %d.", + linenum); + continue; + } + + if (!value) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Missing job ID on line %d.", linenum); + continue; + } + + jobid = atoi(value); + + if (jobid < 1) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Bad job ID %d on line %d.", jobid, + linenum); + continue; + } + + snprintf(jobfile, sizeof(jobfile), "%s/c%05d", RequestRoot, jobid); + if (access(jobfile, 0)) + { + snprintf(jobfile, sizeof(jobfile), "%s/c%05d.N", RequestRoot, jobid); + if (access(jobfile, 0)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "[Job %d] Files have gone away.", + jobid); + continue; + } + } + + job = calloc(1, sizeof(cupsd_job_t)); + if (!job) + { + cupsdLogMessage(CUPSD_LOG_EMERG, + "[Job %d] Unable to allocate memory for job.", jobid); + break; + } + + job->id = jobid; + job->back_pipes[0] = -1; + job->back_pipes[1] = -1; + job->print_pipes[0] = -1; + job->print_pipes[1] = -1; + job->side_pipes[0] = -1; + job->side_pipes[1] = -1; + job->status_pipes[0] = -1; + job->status_pipes[1] = -1; + + cupsdLogMessage(CUPSD_LOG_DEBUG, "[Job %d] Loading from cache...", + job->id); + } + else if (!job) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Missing directive on line %d.", linenum); + continue; + } + else if (!_cups_strcasecmp(line, "")) + { + cupsArrayAdd(Jobs, job); + + if (job->state_value <= IPP_JOB_STOPPED && cupsdLoadJob(job)) + cupsArrayAdd(ActiveJobs, job); + + job = NULL; + } + else if (!value) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Missing value on line %d.", linenum); + continue; + } + else if (!_cups_strcasecmp(line, "State")) + { + job->state_value = (ipp_jstate_t)atoi(value); + + if (job->state_value < IPP_JOB_PENDING) + job->state_value = IPP_JOB_PENDING; + else if (job->state_value > IPP_JOB_COMPLETED) + job->state_value = IPP_JOB_COMPLETED; + } + else if (!_cups_strcasecmp(line, "HoldUntil")) + { + job->hold_until = atoi(value); + } + else if (!_cups_strcasecmp(line, "Priority")) + { + job->priority = atoi(value); + } + else if (!_cups_strcasecmp(line, "Username")) + { + cupsdSetString(&job->username, value); + } + else if (!_cups_strcasecmp(line, "Destination")) + { + cupsdSetString(&job->dest, value); + } + else if (!_cups_strcasecmp(line, "DestType")) + { + job->dtype = (cups_ptype_t)atoi(value); + } + else if (!_cups_strcasecmp(line, "NumFiles")) + { + job->num_files = atoi(value); + + if (job->num_files < 0) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Bad NumFiles value %d on line %d.", + job->num_files, linenum); + job->num_files = 0; + continue; + } + + if (job->num_files > 0) + { + snprintf(jobfile, sizeof(jobfile), "%s/d%05d-001", RequestRoot, + job->id); + if (access(jobfile, 0)) + { + cupsdLogMessage(CUPSD_LOG_INFO, "[Job %d] Data files have gone away.", + job->id); + job->num_files = 0; + continue; + } + + job->filetypes = calloc(job->num_files, sizeof(mime_type_t *)); + job->compressions = calloc(job->num_files, sizeof(int)); + + if (!job->filetypes || !job->compressions) + { + cupsdLogMessage(CUPSD_LOG_EMERG, + "[Job %d] Unable to allocate memory for %d files.", + job->id, job->num_files); + break; + } + } + } + else if (!_cups_strcasecmp(line, "File")) + { + int number, /* File number */ + compression; /* Compression value */ + char super[MIME_MAX_SUPER], /* MIME super type */ + type[MIME_MAX_TYPE]; /* MIME type */ + + + if (sscanf(value, "%d%*[ \t]%15[^/]/%255s%d", &number, super, type, + &compression) != 4) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Bad File on line %d.", linenum); + continue; + } + + if (number < 1 || number > job->num_files) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Bad File number %d on line %d.", + number, linenum); + continue; + } + + number --; + + job->compressions[number] = compression; + job->filetypes[number] = mimeType(MimeDatabase, super, type); + + if (!job->filetypes[number]) + { + /* + * If the original MIME type is unknown, auto-type it! + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, + "[Job %d] Unknown MIME type %s/%s for file %d.", + job->id, super, type, number + 1); + + snprintf(jobfile, sizeof(jobfile), "%s/d%05d-%03d", RequestRoot, + job->id, number + 1); + job->filetypes[number] = mimeFileType(MimeDatabase, jobfile, NULL, + job->compressions + number); + + /* + * If that didn't work, assume it is raw... + */ + + if (!job->filetypes[number]) + job->filetypes[number] = mimeType(MimeDatabase, "application", + "vnd.cups-raw"); + } + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, "Unknown %s directive on line %d.", + line, linenum); + } + + cupsFileClose(fp); +} + + +/* + * 'load_next_job_id()' - Load the NextJobId value from the job.cache file. + */ + +static void +load_next_job_id(const char *filename) /* I - job.cache filename */ +{ + cups_file_t *fp; /* job.cache file */ + char line[1024], /* Line buffer */ + *value; /* Value on line */ + int linenum; /* Line number in file */ + int next_job_id; /* NextJobId value from line */ + + + /* + * Read the NextJobId directive from the job.cache file and use + * the value (if any). + */ + + if ((fp = cupsFileOpen(filename, "r")) == NULL) + { + if (errno != ENOENT) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to open job cache file \"%s\": %s", + filename, strerror(errno)); + + return; + } + + cupsdLogMessage(CUPSD_LOG_INFO, + "Loading NextJobId from job cache file \"%s\"...", filename); + + linenum = 0; + + while (cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + { + if (!_cups_strcasecmp(line, "NextJobId")) + { + if (value) + { + next_job_id = atoi(value); + + if (next_job_id > NextJobId) + NextJobId = next_job_id; + } + break; + } + } + + cupsFileClose(fp); +} + + +/* + * 'load_request_root()' - Load jobs from the RequestRoot directory. + */ + +static void +load_request_root(void) +{ + cups_dir_t *dir; /* Directory */ + cups_dentry_t *dent; /* Directory entry */ + cupsd_job_t *job; /* New job */ + + + /* + * Open the requests directory... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, "Scanning %s for jobs...", RequestRoot); + + if ((dir = cupsDirOpen(RequestRoot)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to open spool directory \"%s\": %s", + RequestRoot, strerror(errno)); + return; + } + + /* + * Read all the c##### files... + */ + + while ((dent = cupsDirRead(dir)) != NULL) + if (strlen(dent->filename) >= 6 && dent->filename[0] == 'c') + { + /* + * Allocate memory for the job... + */ + + if ((job = calloc(sizeof(cupsd_job_t), 1)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Ran out of memory for jobs."); + cupsDirClose(dir); + return; + } + + /* + * Assign the job ID... + */ + + job->id = atoi(dent->filename + 1); + job->back_pipes[0] = -1; + job->back_pipes[1] = -1; + job->print_pipes[0] = -1; + job->print_pipes[1] = -1; + job->side_pipes[0] = -1; + job->side_pipes[1] = -1; + job->status_pipes[0] = -1; + job->status_pipes[1] = -1; + + if (job->id >= NextJobId) + NextJobId = job->id + 1; + + /* + * Load the job... + */ + + if (cupsdLoadJob(job)) + { + /* + * Insert the job into the array, sorting by job priority and ID... + */ + + cupsArrayAdd(Jobs, job); + + if (job->state_value <= IPP_JOB_STOPPED) + cupsArrayAdd(ActiveJobs, job); + else + unload_job(job); + } + } + + cupsDirClose(dir); +} + + +/* + * 'set_time()' - Set one of the "time-at-xyz" attributes. + */ + +static void +set_time(cupsd_job_t *job, /* I - Job to update */ + const char *name) /* I - Name of attribute */ +{ + ipp_attribute_t *attr; /* Time attribute */ + + + if ((attr = ippFindAttribute(job->attrs, name, IPP_TAG_ZERO)) != NULL) + { + attr->value_tag = IPP_TAG_INTEGER; + attr->values[0].integer = time(NULL); + } +} + + +/* + * 'start_job()' - Start a print job. + */ + +static void +start_job(cupsd_job_t *job, /* I - Job ID */ + cupsd_printer_t *printer) /* I - Printer to print job */ +{ + cupsdLogMessage(CUPSD_LOG_DEBUG2, "start_job(job=%p(%d), printer=%p(%s))", + job, job->id, printer, printer->name); + + /* + * Make sure we have some files around before we try to print... + */ + + if (job->num_files == 0) + { + ippSetString(job->attrs, &job->reasons, 0, "aborted-by-system"); + cupsdSetJobState(job, IPP_JOB_ABORTED, CUPSD_JOB_DEFAULT, + "Aborting job because it has no files."); + return; + } + + /* + * Update the printer and job state to "processing"... + */ + + if (!cupsdLoadJob(job)) + return; + + if (job->printer_message) + cupsdSetString(&(job->printer_message->values[0].string.text), ""); + + ippSetString(job->attrs, &job->reasons, 0, "job-printing"); + cupsdSetJobState(job, IPP_JOB_PROCESSING, CUPSD_JOB_DEFAULT, NULL); + cupsdSetPrinterState(printer, IPP_PRINTER_PROCESSING, 0); + cupsdSetPrinterReasons(printer, "-cups-remote-pending," + "cups-remote-pending-held," + "cups-remote-processing," + "cups-remote-stopped," + "cups-remote-canceled," + "cups-remote-aborted," + "cups-remote-completed"); + + job->cost = 0; + job->current_file = 0; + job->progress = 0; + job->printer = printer; + printer->job = job; + + if (MaxJobTime > 0) + job->cancel_time = time(NULL) + MaxJobTime; + else + job->cancel_time = 0; + + /* + * Setup the last exit status and security profiles... + */ + + job->status = 0; + job->profile = cupsdCreateProfile(job->id); + + /* + * Create the status pipes and buffer... + */ + + if (cupsdOpenPipe(job->status_pipes)) + { + cupsdLogJob(job, CUPSD_LOG_DEBUG, + "Unable to create job status pipes - %s.", strerror(errno)); + + cupsdSetJobState(job, IPP_JOB_STOPPED, CUPSD_JOB_DEFAULT, + "Job stopped because the scheduler could not create the " + "job status pipes."); + + cupsdDestroyProfile(job->profile); + job->profile = NULL; + return; + } + + job->status_buffer = cupsdStatBufNew(job->status_pipes[0], NULL); + job->status_level = CUPSD_LOG_INFO; + + /* + * Create the backchannel pipes and make them non-blocking... + */ + + if (cupsdOpenPipe(job->back_pipes)) + { + cupsdLogJob(job, CUPSD_LOG_DEBUG, + "Unable to create back-channel pipes - %s.", strerror(errno)); + + cupsdSetJobState(job, IPP_JOB_STOPPED, CUPSD_JOB_DEFAULT, + "Job stopped because the scheduler could not create the " + "back-channel pipes."); + + cupsdClosePipe(job->status_pipes); + cupsdStatBufDelete(job->status_buffer); + job->status_buffer = NULL; + + cupsdDestroyProfile(job->profile); + job->profile = NULL; + return; + } + + fcntl(job->back_pipes[0], F_SETFL, + fcntl(job->back_pipes[0], F_GETFL) | O_NONBLOCK); + fcntl(job->back_pipes[1], F_SETFL, + fcntl(job->back_pipes[1], F_GETFL) | O_NONBLOCK); + + /* + * Create the side-channel pipes and make them non-blocking... + */ + + if (socketpair(AF_LOCAL, SOCK_STREAM, 0, job->side_pipes)) + { + cupsdLogJob(job, CUPSD_LOG_DEBUG, + "Unable to create side-channel pipes - %s.", strerror(errno)); + + cupsdSetJobState(job, IPP_JOB_STOPPED, CUPSD_JOB_DEFAULT, + "Job stopped because the scheduler could not create the " + "side-channel pipes."); + + cupsdClosePipe(job->back_pipes); + + cupsdClosePipe(job->status_pipes); + cupsdStatBufDelete(job->status_buffer); + job->status_buffer = NULL; + + cupsdDestroyProfile(job->profile); + job->profile = NULL; + return; + } + + fcntl(job->side_pipes[0], F_SETFL, + fcntl(job->side_pipes[0], F_GETFL) | O_NONBLOCK); + fcntl(job->side_pipes[1], F_SETFL, + fcntl(job->side_pipes[1], F_GETFL) | O_NONBLOCK); + + fcntl(job->side_pipes[0], F_SETFD, + fcntl(job->side_pipes[0], F_GETFD) | FD_CLOEXEC); + fcntl(job->side_pipes[1], F_SETFD, + fcntl(job->side_pipes[1], F_GETFD) | FD_CLOEXEC); + + /* + * Now start the first file in the job... + */ + + cupsdContinueJob(job); +} + + +/* + * 'stop_job()' - Stop a print job. + */ + +static void +stop_job(cupsd_job_t *job, /* I - Job */ + cupsd_jobaction_t action) /* I - Action */ +{ + int i; /* Looping var */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "stop_job(job=%p(%d), action=%d)", job, + job->id, action); + + FilterLevel -= job->cost; + job->cost = 0; + + if (action == CUPSD_JOB_DEFAULT && !job->kill_time) + job->kill_time = time(NULL) + JobKillDelay; + else if (action >= CUPSD_JOB_FORCE) + job->kill_time = 0; + + for (i = 0; job->filters[i]; i ++) + if (job->filters[i] > 0) + { + cupsdEndProcess(job->filters[i], action >= CUPSD_JOB_FORCE); + + if (action >= CUPSD_JOB_FORCE) + job->filters[i] = -job->filters[i]; + } + + if (job->backend > 0) + { + cupsdEndProcess(job->backend, action >= CUPSD_JOB_FORCE); + + if (action >= CUPSD_JOB_FORCE) + job->backend = -job->backend; + } + + if (action >= CUPSD_JOB_FORCE) + { + /* + * Clear job status... + */ + + job->status = 0; + } +} + + +/* + * 'unload_job()' - Unload a job from memory. + */ + +static void +unload_job(cupsd_job_t *job) /* I - Job */ +{ + if (!job->attrs) + return; + + cupsdLogMessage(CUPSD_LOG_DEBUG, "[Job %d] Unloading...", job->id); + + ippDelete(job->attrs); + + job->attrs = NULL; + job->state = NULL; + job->reasons = NULL; + job->sheets = NULL; + job->job_sheets = NULL; + job->printer_message = NULL; + job->printer_reasons = NULL; +} + + +/* + * 'update_job()' - Read a status update from a job's filters. + */ + +void +update_job(cupsd_job_t *job) /* I - Job to check */ +{ + int i; /* Looping var */ + int copies; /* Number of copies printed */ + char message[CUPSD_SB_BUFFER_SIZE], + /* Message text */ + *ptr; /* Pointer update... */ + int loglevel, /* Log level for message */ + event = 0; /* Events? */ + static const char * const levels[] = /* Log levels */ + { + "NONE", + "EMERG", + "ALERT", + "CRIT", + "ERROR", + "WARN", + "NOTICE", + "INFO", + "DEBUG", + "DEBUG2" + }; + + + /* + * Get the printer associated with this job; if the printer is stopped for + * any reason then job->printer will be reset to NULL, so make sure we have + * a valid pointer... + */ + + while ((ptr = cupsdStatBufUpdate(job->status_buffer, &loglevel, + message, sizeof(message))) != NULL) + { + /* + * Process page and printer state messages as needed... + */ + + if (loglevel == CUPSD_LOG_PAGE) + { + /* + * Page message; send the message to the page_log file and update the + * job sheet count... + */ + + cupsdLogJob(job, CUPSD_LOG_DEBUG, "PAGE: %s", message); + + if (job->sheets) + { + if (!_cups_strncasecmp(message, "total ", 6)) + { + /* + * Got a total count of pages from a backend or filter... + */ + + copies = atoi(message + 6); + copies -= job->sheets->values[0].integer; /* Just track the delta */ + } + else if (!sscanf(message, "%*d%d", &copies)) + copies = 1; + + job->sheets->values[0].integer += copies; + + if (job->printer->page_limit) + cupsdUpdateQuota(job->printer, job->username, copies, 0); + } + + cupsdLogPage(job, message); + + if (job->sheets) + cupsdAddEvent(CUPSD_EVENT_JOB_PROGRESS, job->printer, job, + "Printed %d page(s).", job->sheets->values[0].integer); + } + else if (loglevel == CUPSD_LOG_STATE) + { + cupsdLogJob(job, CUPSD_LOG_DEBUG, "STATE: %s", message); + + if (!strcmp(message, "paused")) + { + cupsdStopPrinter(job->printer, 1); + return; + } + else if (cupsdSetPrinterReasons(job->printer, message)) + { + event |= CUPSD_EVENT_PRINTER_STATE; + + if (MaxJobTime > 0 && strstr(message, "connecting-to-device") != NULL) + { + /* + * Reset cancel time after connecting to the device... + */ + + for (i = 0; i < job->printer->num_reasons; i ++) + if (!strcmp(job->printer->reasons[i], "connecting-to-device")) + break; + + if (i >= job->printer->num_reasons) + job->cancel_time = time(NULL) + MaxJobTime; + } + } + + update_job_attrs(job, 0); + } + else if (loglevel == CUPSD_LOG_ATTR) + { + /* + * Set attribute(s)... + */ + + int num_attrs; /* Number of attributes */ + cups_option_t *attrs; /* Attributes */ + const char *attr; /* Attribute */ + + + cupsdLogJob(job, CUPSD_LOG_DEBUG, "ATTR: %s", message); + + num_attrs = cupsParseOptions(message, 0, &attrs); + + if ((attr = cupsGetOption("auth-info-required", num_attrs, + attrs)) != NULL) + { + cupsdSetAuthInfoRequired(job->printer, attr, NULL); + cupsdSetPrinterAttrs(job->printer); + + cupsdMarkDirty(CUPSD_DIRTY_PRINTERS); + } + + if ((attr = cupsGetOption("job-media-progress", num_attrs, + attrs)) != NULL) + { + int progress = atoi(attr); + + + if (progress >= 0 && progress <= 100) + { + job->progress = progress; + + if (job->sheets) + cupsdAddEvent(CUPSD_EVENT_JOB_PROGRESS, job->printer, job, + "Printing page %d, %d%%", + job->sheets->values[0].integer, job->progress); + } + } + + if ((attr = cupsGetOption("printer-alert", num_attrs, attrs)) != NULL) + { + cupsdSetString(&job->printer->alert, attr); + event |= CUPSD_EVENT_PRINTER_STATE; + } + + if ((attr = cupsGetOption("printer-alert-description", num_attrs, + attrs)) != NULL) + { + cupsdSetString(&job->printer->alert_description, attr); + event |= CUPSD_EVENT_PRINTER_STATE; + } + + if ((attr = cupsGetOption("marker-colors", num_attrs, attrs)) != NULL) + { + cupsdSetPrinterAttr(job->printer, "marker-colors", (char *)attr); + job->printer->marker_time = time(NULL); + event |= CUPSD_EVENT_PRINTER_STATE; + cupsdMarkDirty(CUPSD_DIRTY_PRINTERS); + } + + if ((attr = cupsGetOption("marker-levels", num_attrs, attrs)) != NULL) + { + cupsdSetPrinterAttr(job->printer, "marker-levels", (char *)attr); + job->printer->marker_time = time(NULL); + event |= CUPSD_EVENT_PRINTER_STATE; + cupsdMarkDirty(CUPSD_DIRTY_PRINTERS); + } + + if ((attr = cupsGetOption("marker-low-levels", num_attrs, attrs)) != NULL) + { + cupsdSetPrinterAttr(job->printer, "marker-low-levels", (char *)attr); + job->printer->marker_time = time(NULL); + event |= CUPSD_EVENT_PRINTER_STATE; + cupsdMarkDirty(CUPSD_DIRTY_PRINTERS); + } + + if ((attr = cupsGetOption("marker-high-levels", num_attrs, attrs)) != NULL) + { + cupsdSetPrinterAttr(job->printer, "marker-high-levels", (char *)attr); + job->printer->marker_time = time(NULL); + event |= CUPSD_EVENT_PRINTER_STATE; + cupsdMarkDirty(CUPSD_DIRTY_PRINTERS); + } + + if ((attr = cupsGetOption("marker-message", num_attrs, attrs)) != NULL) + { + cupsdSetPrinterAttr(job->printer, "marker-message", (char *)attr); + job->printer->marker_time = time(NULL); + event |= CUPSD_EVENT_PRINTER_STATE; + cupsdMarkDirty(CUPSD_DIRTY_PRINTERS); + } + + if ((attr = cupsGetOption("marker-names", num_attrs, attrs)) != NULL) + { + cupsdSetPrinterAttr(job->printer, "marker-names", (char *)attr); + job->printer->marker_time = time(NULL); + event |= CUPSD_EVENT_PRINTER_STATE; + cupsdMarkDirty(CUPSD_DIRTY_PRINTERS); + } + + if ((attr = cupsGetOption("marker-types", num_attrs, attrs)) != NULL) + { + cupsdSetPrinterAttr(job->printer, "marker-types", (char *)attr); + job->printer->marker_time = time(NULL); + event |= CUPSD_EVENT_PRINTER_STATE; + cupsdMarkDirty(CUPSD_DIRTY_PRINTERS); + } + + cupsFreeOptions(num_attrs, attrs); + } + else if (loglevel == CUPSD_LOG_PPD) + { + /* + * Set attribute(s)... + */ + + int num_keywords; /* Number of keywords */ + cups_option_t *keywords; /* Keywords */ + + + cupsdLogJob(job, CUPSD_LOG_DEBUG, "PPD: %s", message); + + num_keywords = cupsParseOptions(message, 0, &keywords); + + if (cupsdUpdatePrinterPPD(job->printer, num_keywords, keywords)) + cupsdSetPrinterAttrs(job->printer); + + cupsFreeOptions(num_keywords, keywords); + } + else + { + /* + * Strip legacy message prefix... + */ + + if (!strncmp(message, "recoverable:", 12)) + { + ptr = message + 12; + while (isspace(*ptr & 255)) + ptr ++; + } + else if (!strncmp(message, "recovered:", 10)) + { + ptr = message + 10; + while (isspace(*ptr & 255)) + ptr ++; + } + else + ptr = message; + + if (*ptr) + cupsdLogJob(job, loglevel, "%s", ptr); + + if (loglevel < CUPSD_LOG_DEBUG && + strcmp(job->printer->state_message, ptr)) + { + strlcpy(job->printer->state_message, ptr, + sizeof(job->printer->state_message)); + + event |= CUPSD_EVENT_PRINTER_STATE | CUPSD_EVENT_JOB_PROGRESS; + + if (loglevel <= job->status_level && job->status_level > CUPSD_LOG_ERROR) + { + /* + * Some messages show in the job-printer-state-message attribute... + */ + + if (loglevel != CUPSD_LOG_NOTICE) + job->status_level = loglevel; + + update_job_attrs(job, 1); + + cupsdLogJob(job, CUPSD_LOG_DEBUG, + "Set job-printer-state-message to \"%s\", " + "current level=%s", + job->printer_message->values[0].string.text, + levels[job->status_level]); + } + } + } + + if (!strchr(job->status_buffer->buffer, '\n')) + break; + } + + if (event & CUPSD_EVENT_JOB_PROGRESS) + cupsdAddEvent(CUPSD_EVENT_JOB_PROGRESS, job->printer, job, + "%s", job->printer->state_message); + if (event & CUPSD_EVENT_PRINTER_STATE) + cupsdAddEvent(CUPSD_EVENT_PRINTER_STATE, job->printer, NULL, + (job->printer->type & CUPS_PRINTER_CLASS) ? + "Class \"%s\" state changed." : + "Printer \"%s\" state changed.", + job->printer->name); + + + if (ptr == NULL && !job->status_buffer->bufused) + { + /* + * See if all of the filters and the backend have returned their + * exit statuses. + */ + + for (i = 0; job->filters[i] < 0; i ++); + + if (job->filters[i]) + { + /* + * EOF but we haven't collected the exit status of all filters... + */ + + cupsdCheckProcess(); + return; + } + + if (job->current_file >= job->num_files && job->backend > 0) + { + /* + * EOF but we haven't collected the exit status of the backend... + */ + + cupsdCheckProcess(); + return; + } + + /* + * Handle the end of job stuff... + */ + + finalize_job(job, 1); + + /* + * Check for new jobs... + */ + + cupsdCheckJobs(); + } +} + + +/* + * 'update_job_attrs()' - Update the job-printer-* attributes. + */ + +void +update_job_attrs(cupsd_job_t *job, /* I - Job to update */ + int do_message)/* I - 1 = copy job-printer-state message */ +{ + int i; /* Looping var */ + int num_reasons; /* Actual number of reasons */ + const char * const *reasons; /* Reasons */ + static const char *none = "none"; /* "none" reason */ + + + /* + * Get/create the job-printer-state-* attributes... + */ + + if (!job->printer_message) + { + if ((job->printer_message = ippFindAttribute(job->attrs, + "job-printer-state-message", + IPP_TAG_TEXT)) == NULL) + job->printer_message = ippAddString(job->attrs, IPP_TAG_JOB, IPP_TAG_TEXT, + "job-printer-state-message", + NULL, ""); + } + + if (!job->printer_reasons) + job->printer_reasons = ippFindAttribute(job->attrs, + "job-printer-state-reasons", + IPP_TAG_KEYWORD); + + /* + * Copy or clear the printer-state-message value as needed... + */ + + if (job->state_value != IPP_JOB_PROCESSING && + job->status_level == CUPSD_LOG_INFO) + { + cupsdSetString(&(job->printer_message->values[0].string.text), ""); + + job->dirty = 1; + cupsdMarkDirty(CUPSD_DIRTY_JOBS); + } + else if (job->printer->state_message[0] && do_message) + { + cupsdSetString(&(job->printer_message->values[0].string.text), + job->printer->state_message); + + job->dirty = 1; + cupsdMarkDirty(CUPSD_DIRTY_JOBS); + } + + /* + * ... and the printer-state-reasons value... + */ + + if (job->printer->num_reasons == 0) + { + num_reasons = 1; + reasons = &none; + } + else + { + num_reasons = job->printer->num_reasons; + reasons = (const char * const *)job->printer->reasons; + } + + if (!job->printer_reasons || job->printer_reasons->num_values != num_reasons) + { + /* + * Replace/create a job-printer-state-reasons attribute... + */ + + ippDeleteAttribute(job->attrs, job->printer_reasons); + + job->printer_reasons = ippAddStrings(job->attrs, + IPP_TAG_JOB, IPP_TAG_KEYWORD, + "job-printer-state-reasons", + num_reasons, NULL, NULL); + } + else + { + /* + * Don't bother clearing the reason strings if they are the same... + */ + + for (i = 0; i < num_reasons; i ++) + if (strcmp(job->printer_reasons->values[i].string.text, reasons[i])) + break; + + if (i >= num_reasons) + return; + + /* + * Not the same, so free the current strings... + */ + + for (i = 0; i < num_reasons; i ++) + _cupsStrFree(job->printer_reasons->values[i].string.text); + } + + /* + * Copy the reasons... + */ + + for (i = 0; i < num_reasons; i ++) + job->printer_reasons->values[i].string.text = _cupsStrAlloc(reasons[i]); + + job->dirty = 1; + cupsdMarkDirty(CUPSD_DIRTY_JOBS); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/job.h b/scheduler/job.h new file mode 100644 index 0000000000..b119664d9a --- /dev/null +++ b/scheduler/job.h @@ -0,0 +1,166 @@ +/* + * "$Id$" + * + * Print job definitions for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +/* + * Constants... + */ + +typedef enum cupsd_jobaction_e /**** Actions for state changes ****/ +{ + CUPSD_JOB_DEFAULT, /* Use default action */ + CUPSD_JOB_FORCE, /* Force the change */ + CUPSD_JOB_PURGE /* Force the change and purge */ +} cupsd_jobaction_t; + + +/* + * Job request structure... + */ + +struct cupsd_job_s /**** Job request ****/ +{ + int id, /* Job ID */ + priority, /* Job priority */ + dirty; /* Do we need to write the "c" file? */ + ipp_jstate_t state_value; /* Cached job-state */ + int pending_timeout;/* Non-zero if the job was created and + * waiting on files */ + char *username; /* Printing user */ + char *dest; /* Destination printer or class */ + cups_ptype_t dtype; /* Destination type */ + cupsd_printer_t *printer; /* Printer this job is assigned to */ + int num_files; /* Number of files in job */ + mime_type_t **filetypes; /* File types */ + int *compressions; /* Compression status of each file */ + ipp_attribute_t *sheets; /* job-media-sheets-completed */ + time_t access_time, /* Last access time */ + cancel_time, /* When to cancel/send SIGTERM */ + kill_time, /* When to send SIGKILL */ + hold_until; /* Hold expiration date/time */ + ipp_attribute_t *state; /* Job state */ + ipp_attribute_t *reasons; /* Job state reasons */ + ipp_attribute_t *job_sheets; /* Job sheets (NULL if none) */ + ipp_attribute_t *printer_message, + /* job-printer-state-message */ + *printer_reasons; + /* job-printer-state-reasons */ + int current_file; /* Current file in job */ + ipp_t *attrs; /* Job attributes */ + int print_pipes[2], /* Print data pipes */ + back_pipes[2], /* Backchannel pipes */ + side_pipes[2], /* Sidechannel pipes */ + status_pipes[2];/* Status pipes */ + cupsd_statbuf_t *status_buffer; /* Status buffer for this job */ + int status_level; /* Highest log level in a status + * message */ + int cost; /* Filtering cost */ + int pending_cost; /* Waiting for FilterLimit */ + int filters[MAX_FILTERS + 1]; + /* Filter process IDs, 0 terminated */ + int backend; /* Backend process ID */ + int status; /* Status code from filters */ + int tries; /* Number of tries for this job */ + char *auth_env[3], /* AUTH_xxx environment variables, + * if any */ + *auth_uid; /* AUTH_UID environment variable */ + void *profile; /* Security profile */ + cups_array_t *history; /* Debug log history */ + int progress; /* Printing progress */ +}; + +typedef struct cupsd_joblog_s /**** Job log message ****/ +{ + time_t time; /* Time of message */ + char message[1]; /* Message string */ +} cupsd_joblog_t; + + +/* + * Globals... + */ + +VAR int JobHistory VALUE(1); + /* Preserve job history? */ +VAR int JobFiles VALUE(0); + /* Preserve job files? */ +VAR int MaxJobs VALUE(0), + /* Max number of jobs */ + MaxActiveJobs VALUE(0), + /* Max number of active jobs */ + MaxJobsPerUser VALUE(0), + /* Max jobs per user */ + MaxJobsPerPrinter VALUE(0), + /* Max jobs per printer */ + MaxJobTime VALUE(3 * 60 * 60); + /* Max time for a job */ +VAR int JobAutoPurge VALUE(0); + /* Automatically purge jobs */ +VAR cups_array_t *Jobs VALUE(NULL), + /* List of current jobs */ + *ActiveJobs VALUE(NULL), + /* List of active jobs */ + *PrintingJobs VALUE(NULL); + /* List of jobs that are printing */ +VAR int NextJobId VALUE(1); + /* Next job ID to use */ +VAR int JobKillDelay VALUE(DEFAULT_TIMEOUT), + /* Delay before killing jobs */ + JobRetryLimit VALUE(5), + /* Max number of tries */ + JobRetryInterval VALUE(300); + /* Seconds between retries */ + + +/* + * Prototypes... + */ + +extern cupsd_job_t *cupsdAddJob(int priority, const char *dest); +extern void cupsdCancelJobs(const char *dest, const char *username, + int purge); +extern void cupsdCheckJobs(void); +extern void cupsdCleanJobs(void); +extern void cupsdContinueJob(cupsd_job_t *job); +extern void cupsdDeleteJob(cupsd_job_t *job, + cupsd_jobaction_t action); +extern cupsd_job_t *cupsdFindJob(int id); +extern void cupsdFreeAllJobs(void); +extern int cupsdGetPrinterJobCount(const char *dest); +extern int cupsdGetUserJobCount(const char *username); +extern void cupsdLoadAllJobs(void); +extern int cupsdLoadJob(cupsd_job_t *job); +extern void cupsdMoveJob(cupsd_job_t *job, cupsd_printer_t *p); +extern void cupsdReleaseJob(cupsd_job_t *job); +extern void cupsdRestartJob(cupsd_job_t *job); +extern void cupsdSaveAllJobs(void); +extern void cupsdSaveJob(cupsd_job_t *job); +extern void cupsdSetJobHoldUntil(cupsd_job_t *job, + const char *when, int update); +extern void cupsdSetJobPriority(cupsd_job_t *job, int priority); +extern void cupsdSetJobState(cupsd_job_t *job, + ipp_jstate_t newstate, + cupsd_jobaction_t action, + const char *message, ...) + __attribute__((__format__(__printf__, + 4, 5))); +extern void cupsdStopAllJobs(cupsd_jobaction_t action, + int kill_delay); +extern int cupsdTimeoutJob(cupsd_job_t *job); +extern void cupsdUnloadCompletedJobs(void); + + +/* + * End of "$Id$". + */ diff --git a/scheduler/libcupsmime.exp b/scheduler/libcupsmime.exp new file mode 100644 index 0000000000..cd02a6fcc3 --- /dev/null +++ b/scheduler/libcupsmime.exp @@ -0,0 +1,22 @@ +_mimeAddFilter +_mimeAddType +_mimeAddTypeRule +_mimeDelete +_mimeDeleteFilter +_mimeDeleteType +_mimeFileType +_mimeFilter +_mimeFilter2 +_mimeFilterLookup +_mimeFirstFilter +_mimeFirstType +_mimeLoad +_mimeLoadFilters +_mimeLoadTypes +_mimeNew +_mimeNextFilter +_mimeNextType +_mimeNumFilters +_mimeNumTypes +_mimeSetErrorCallback +_mimeType diff --git a/scheduler/listen.c b/scheduler/listen.c new file mode 100644 index 0000000000..b8baaf2bdb --- /dev/null +++ b/scheduler/listen.c @@ -0,0 +1,431 @@ +/* + * "$Id$" + * + * Server listening routines for the CUPS scheduler. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdDeleteAllListeners() - Delete all listeners. + * cupsdPauseListening() - Clear input polling on all listening sockets... + * cupsdResumeListening() - Set input polling on all listening sockets... + * cupsdStartListening() - Create all listening sockets... + * cupsdStopListening() - Close all listening sockets... + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" + + +/* + * Make sure the IPV6_V6ONLY is defined on Linux - older versions of + * glibc don't define it even if the kernel supports it... + */ + +#if defined(__linux) && !defined(IPV6_V6ONLY) +# define IPV6_V6ONLY 26 +#endif /* __linux && !IPV6_V6ONLY */ + + +/* + * 'cupsdDeleteAllListeners()' - Delete all listeners. + */ + +void +cupsdDeleteAllListeners(void) +{ + cupsd_listener_t *lis; /* Current listening socket */ + + + for (lis = (cupsd_listener_t *)cupsArrayFirst(Listeners); + lis; + lis = (cupsd_listener_t *)cupsArrayNext(Listeners)) + free(lis); + + cupsArrayDelete(Listeners); + Listeners = NULL; +} + + +/* + * 'cupsdPauseListening()' - Clear input polling on all listening sockets... + */ + +void +cupsdPauseListening(void) +{ + cupsd_listener_t *lis; /* Current listening socket */ + + + if (cupsArrayCount(Listeners) < 1) + return; + + if (cupsArrayCount(Clients) == MaxClients) + cupsdLogMessage(CUPSD_LOG_WARN, + "Max clients reached, holding new connections..."); + else if (errno == ENFILE || errno == EMFILE) + cupsdLogMessage(CUPSD_LOG_WARN, + "Too many open files, holding new connections for " + "30 seconds..."); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdPauseListening: Clearing input bits..."); + + for (lis = (cupsd_listener_t *)cupsArrayFirst(Listeners); + lis; + lis = (cupsd_listener_t *)cupsArrayNext(Listeners)) + cupsdRemoveSelect(lis->fd); + + ListeningPaused = time(NULL) + 30; +} + + +/* + * 'cupsdResumeListening()' - Set input polling on all listening sockets... + */ + +void +cupsdResumeListening(void) +{ + cupsd_listener_t *lis; /* Current listening socket */ + + + if (cupsArrayCount(Listeners) < 1) + return; + + cupsdLogMessage(CUPSD_LOG_INFO, "Resuming new connection processing..."); + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdResumeListening: Setting input bits..."); + + for (lis = (cupsd_listener_t *)cupsArrayFirst(Listeners); + lis; + lis = (cupsd_listener_t *)cupsArrayNext(Listeners)) + cupsdAddSelect(lis->fd, (cupsd_selfunc_t)cupsdAcceptClient, NULL, lis); + + ListeningPaused = 0; +} + + +/* + * 'cupsdStartListening()' - Create all listening sockets... + */ + +void +cupsdStartListening(void) +{ + int status; /* Bind result */ + int p, /* Port number */ + val; /* Parameter value */ + cupsd_listener_t *lis; /* Current listening socket */ + char s[256]; /* String addresss */ + const char *have_domain; /* Have a domain socket? */ + static const char * const encryptions[] = + { /* Encryption values */ + "IfRequested", + "Never", + "Required", + "Always" + }; + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdStartListening: %d Listeners", + cupsArrayCount(Listeners)); + + /* + * Setup socket listeners... + */ + + for (lis = (cupsd_listener_t *)cupsArrayFirst(Listeners), LocalPort = 0, + have_domain = NULL; + lis; + lis = (cupsd_listener_t *)cupsArrayNext(Listeners)) + { + httpAddrString(&(lis->address), s, sizeof(s)); + p = _httpAddrPort(&(lis->address)); + + /* + * If needed, create a socket for listening... + */ + + if (lis->fd == -1) + { + /* + * Create a socket for listening... + */ + + lis->fd = socket(lis->address.addr.sa_family, SOCK_STREAM, 0); + + if (lis->fd == -1) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to open listen socket for address %s:%d - %s.", + s, p, strerror(errno)); + +#ifdef AF_INET6 + /* + * IPv6 is often disabled while DNS returns IPv6 addresses... + */ + + if (lis->address.addr.sa_family != AF_INET6 && + (FatalErrors & CUPSD_FATAL_LISTEN)) + cupsdEndProcess(getpid(), 0); +#else + if (FatalErrors & CUPSD_FATAL_LISTEN) + cupsdEndProcess(getpid(), 0); +#endif /* AF_INET6 */ + + continue; + } + + /* + * Set things up to reuse the local address for this port. + */ + + val = 1; +#ifdef __sun + setsockopt(lis->fd, SOL_SOCKET, SO_REUSEADDR, (char *)&val, sizeof(val)); +#else + setsockopt(lis->fd, SOL_SOCKET, SO_REUSEADDR, &val, sizeof(val)); +#endif /* __sun */ + + /* + * Bind to the port we found... + */ + +#ifdef AF_INET6 + if (lis->address.addr.sa_family == AF_INET6) + { +# ifdef IPV6_V6ONLY + /* + * Accept only IPv6 connections on this socket, to avoid + * potential security issues and to make all platforms behave + * the same. + */ + + val = 1; +# ifdef __sun + setsockopt(lis->fd, IPPROTO_IPV6, IPV6_V6ONLY, (char *)&val, sizeof(val)); +# else + setsockopt(lis->fd, IPPROTO_IPV6, IPV6_V6ONLY, &val, sizeof(val)); +# endif /* __sun */ +# endif /* IPV6_V6ONLY */ + + status = bind(lis->fd, (struct sockaddr *)&(lis->address), + httpAddrLength(&(lis->address))); + } + else +#endif /* AF_INET6 */ +#ifdef AF_LOCAL + if (lis->address.addr.sa_family == AF_LOCAL) + { + mode_t mask; /* Umask setting */ + + + /* + * Remove any existing domain socket file... + */ + + unlink(lis->address.un.sun_path); + + /* + * Save the current umask and set it to 0 so that all users can access + * the domain socket... + */ + + mask = umask(0); + + /* + * Bind the domain socket... + */ + + status = bind(lis->fd, (struct sockaddr *)&(lis->address), + httpAddrLength(&(lis->address))); + + /* + * Restore the umask... + */ + + umask(mask); + } + else +#endif /* AF_LOCAL */ + status = bind(lis->fd, (struct sockaddr *)&(lis->address), + sizeof(lis->address.ipv4)); + + if (status < 0) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to bind socket for address %s:%d - %s.", + s, p, strerror(errno)); + close(lis->fd); + lis->fd = -1; + + if (FatalErrors & CUPSD_FATAL_LISTEN) + cupsdEndProcess(getpid(), 0); + + continue; + } + + /* + * Listen for new clients. + */ + + if (listen(lis->fd, ListenBackLog) < 0) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to listen for clients on address %s:%d - %s.", + s, p, strerror(errno)); + + close(lis->fd); + lis->fd = -1; + + if (FatalErrors & CUPSD_FATAL_LISTEN) + cupsdEndProcess(getpid(), 0); + + continue; + } + } + + fcntl(lis->fd, F_SETFD, fcntl(lis->fd, F_GETFD) | FD_CLOEXEC); + + if (p) + cupsdLogMessage(CUPSD_LOG_INFO, "Listening to %s:%d on fd %d...", + s, p, lis->fd); + else + { + cupsdLogMessage(CUPSD_LOG_INFO, "Listening to %s on fd %d...", + s, lis->fd); + + if (chmod(s, 0140777)) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to change permisssions on domain socket " + "\"%s\" - %s", s, strerror(errno)); + } + + /* + * Save the first port that is bound to the local loopback or + * "any" address... + */ + + if ((!LocalPort || LocalEncryption == HTTP_ENCRYPT_ALWAYS) && p > 0 && + (httpAddrLocalhost(&(lis->address)) || + httpAddrAny(&(lis->address)))) + { + LocalPort = p; + LocalEncryption = lis->encryption; + } + +#ifdef AF_LOCAL + if (lis->address.addr.sa_family == AF_LOCAL && !have_domain) + have_domain = lis->address.un.sun_path; +#endif /* AF_LOCAL */ + } + + /* + * Make sure that we are listening on localhost! + */ + + if (!LocalPort && !have_domain) + { + cupsdLogMessage(CUPSD_LOG_EMERG, + "No Listen or Port lines were found to allow access via " + "localhost!"); + + if (FatalErrors & (CUPSD_FATAL_CONFIG | CUPSD_FATAL_LISTEN)) + cupsdEndProcess(getpid(), 0); + } + + /* + * Set the CUPS_SERVER, IPP_PORT, and CUPS_ENCRYPTION variables based on + * the listeners... + */ + + if (have_domain) + { + /* + * Use domain sockets for the local connection... + */ + + cupsdSetEnv("CUPS_SERVER", have_domain); + + LocalEncryption = HTTP_ENCRYPT_IF_REQUESTED; + } + else + { + /* + * Use the default local loopback address for the server... + */ + + cupsdSetEnv("CUPS_SERVER", "localhost"); + } + + cupsdSetEnv("CUPS_ENCRYPTION", encryptions[LocalEncryption]); + + if (LocalPort) + cupsdSetEnvf("IPP_PORT", "%d", LocalPort); + + /* + * Resume listening for connections... + */ + + cupsdResumeListening(); +} + + +/* + * 'cupsdStopListening()' - Close all listening sockets... + */ + +void +cupsdStopListening(void) +{ + cupsd_listener_t *lis; /* Current listening socket */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdStopListening: closing all listen sockets."); + + cupsdPauseListening(); + + for (lis = (cupsd_listener_t *)cupsArrayFirst(Listeners); + lis; + lis = (cupsd_listener_t *)cupsArrayNext(Listeners)) + { + if (lis->fd != -1) + { +#ifdef WIN32 + closesocket(lis->fd); +#else + close(lis->fd); +#endif /* WIN32 */ + +#ifdef AF_LOCAL + /* + * Remove domain sockets... + */ + +# ifdef HAVE_LAUNCH_H + if (lis->address.addr.sa_family == AF_LOCAL && !Launchd) +# else + if (lis->address.addr.sa_family == AF_LOCAL) +# endif /* HAVE_LAUNCH_H */ + unlink(lis->address.un.sun_path); +#endif /* AF_LOCAL */ + } + } +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/log.c b/scheduler/log.c new file mode 100644 index 0000000000..b0c903c545 --- /dev/null +++ b/scheduler/log.c @@ -0,0 +1,1102 @@ +/* + * "$Id$" + * + * Log file routines for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdCheckLogFile() - Open/rotate a log file if it needs it. + * cupsdGetDateTime() - Returns a pointer to a date/time string. + * cupsdLogGSSMessage() - Log a GSSAPI error... + * cupsdLogJob() - Log a job message. + * cupsdLogMessage() - Log a message to the error log file. + * cupsdLogPage() - Log a page to the page log file. + * cupsdLogRequest() - Log an HTTP request in Common Log Format. + * cupsdWriteErrorLog() - Write a line to the ErrorLog. + * format_log_line() - Format a line for a log file. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" +#include +#include + + +/* + * Local globals... + */ + +static int log_linesize = 0; /* Size of line for output file */ +static char *log_line = NULL; /* Line for output file */ + + +/* + * Local functions... + */ + +static int format_log_line(const char *message, va_list ap); + + +/* + * 'cupsdCheckLogFile()' - Open/rotate a log file if it needs it. + */ + +int /* O - 1 if log file open */ +cupsdCheckLogFile(cups_file_t **lf, /* IO - Log file */ + const char *logname) /* I - Log filename */ +{ + char backname[1024], /* Backup log filename */ + filename[1024], /* Formatted log filename */ + *ptr; /* Pointer into filename */ + const char *logptr; /* Pointer into log filename */ + + + /* + * See if we have a log file to check... + */ + + if (!lf || !logname || !logname[0]) + return (1); + + /* + * Format the filename as needed... + */ + + if (!*lf || + (strncmp(logname, "/dev/", 5) && cupsFileTell(*lf) > MaxLogSize && + MaxLogSize > 0)) + { + /* + * Handle format strings... + */ + + filename[sizeof(filename) - 1] = '\0'; + + if (logname[0] != '/') + { + strlcpy(filename, ServerRoot, sizeof(filename)); + strlcat(filename, "/", sizeof(filename)); + } + else + filename[0] = '\0'; + + for (logptr = logname, ptr = filename + strlen(filename); + *logptr && ptr < (filename + sizeof(filename) - 1); + logptr ++) + if (*logptr == '%') + { + /* + * Format spec... + */ + + logptr ++; + if (*logptr == 's') + { + /* + * Insert the server name... + */ + + strlcpy(ptr, ServerName, sizeof(filename) - (ptr - filename)); + ptr += strlen(ptr); + } + else + { + /* + * Otherwise just insert the character... + */ + + *ptr++ = *logptr; + } + } + else + *ptr++ = *logptr; + + *ptr = '\0'; + } + + /* + * See if the log file is open... + */ + + if (!*lf) + { + /* + * Nope, open the log file... + */ + + if ((*lf = cupsFileOpen(filename, "a")) == NULL) + { + /* + * If the file is in CUPS_LOGDIR then try to create a missing directory... + */ + + if (!strncmp(filename, CUPS_LOGDIR, strlen(CUPS_LOGDIR))) + { + /* + * Try updating the permissions of the containing log directory, using + * the log file permissions as a basis... + */ + + int log_dir_perm = 0300 | LogFilePerm; + /* LogFilePerm + owner write/search */ + if (log_dir_perm & 0040) + log_dir_perm |= 0010; /* Add group search */ + if (log_dir_perm & 0004) + log_dir_perm |= 0001; /* Add other search */ + + cupsdCheckPermissions(CUPS_LOGDIR, NULL, log_dir_perm, RunUser, Group, + 1, -1); + + *lf = cupsFileOpen(filename, "a"); + } + + if (*lf == NULL) + { + syslog(LOG_ERR, "Unable to open log file \"%s\" - %s", filename, + strerror(errno)); + + if (FatalErrors & CUPSD_FATAL_LOG) + cupsdEndProcess(getpid(), 0); + + return (0); + } + } + + if (strncmp(filename, "/dev/", 5)) + { + /* + * Change ownership and permissions of non-device logs... + */ + + fchown(cupsFileNumber(*lf), RunUser, Group); + fchmod(cupsFileNumber(*lf), LogFilePerm); + } + } + + /* + * Do we need to rotate the log? + */ + + if (strncmp(logname, "/dev/", 5) && cupsFileTell(*lf) > MaxLogSize && + MaxLogSize > 0) + { + /* + * Rotate log file... + */ + + cupsFileClose(*lf); + + strcpy(backname, filename); + strlcat(backname, ".O", sizeof(backname)); + + unlink(backname); + rename(filename, backname); + + if ((*lf = cupsFileOpen(filename, "a")) == NULL) + { + syslog(LOG_ERR, "Unable to open log file \"%s\" - %s", filename, + strerror(errno)); + + if (FatalErrors & CUPSD_FATAL_LOG) + cupsdEndProcess(getpid(), 0); + + return (0); + } + + /* + * Change ownership and permissions of non-device logs... + */ + + fchown(cupsFileNumber(*lf), RunUser, Group); + fchmod(cupsFileNumber(*lf), LogFilePerm); + } + + return (1); +} + + +/* + * 'cupsdGetDateTime()' - Returns a pointer to a date/time string. + */ + +char * /* O - Date/time string */ +cupsdGetDateTime(struct timeval *t, /* I - Time value or NULL for current */ + cupsd_time_t format) /* I - Format to use */ +{ + struct timeval curtime; /* Current time value */ + struct tm *date; /* Date/time value */ + static struct timeval last_time = { 0, 0 }; + /* Last time we formatted */ + static char s[1024]; /* Date/time string */ + static const char * const months[12] =/* Months */ + { + "Jan", + "Feb", + "Mar", + "Apr", + "May", + "Jun", + "Jul", + "Aug", + "Sep", + "Oct", + "Nov", + "Dec" + }; + + + /* + * Make sure we have a valid time... + */ + + if (!t) + { + gettimeofday(&curtime, NULL); + t = &curtime; + } + + if (t->tv_sec != last_time.tv_sec || + (LogTimeFormat == CUPSD_TIME_USECS && t->tv_usec != last_time.tv_usec)) + { + last_time = *t; + + /* + * Get the date and time from the UNIX time value, and then format it + * into a string. Note that we *can't* use the strftime() function since + * it is localized and will seriously confuse automatic programs if the + * month names are in the wrong language! + * + * Also, we use the "timezone" variable that contains the current timezone + * offset from GMT in seconds so that we are reporting local time in the + * log files. If you want GMT, set the TZ environment variable accordingly + * before starting the scheduler. + * + * (*BSD and Darwin store the timezone offset in the tm structure) + */ + + date = localtime(&(t->tv_sec)); + + if (format == CUPSD_TIME_STANDARD) + snprintf(s, sizeof(s), "[%02d/%s/%04d:%02d:%02d:%02d %+03ld%02ld]", + date->tm_mday, months[date->tm_mon], 1900 + date->tm_year, + date->tm_hour, date->tm_min, date->tm_sec, +#ifdef HAVE_TM_GMTOFF + date->tm_gmtoff / 3600, (date->tm_gmtoff / 60) % 60); +#else + timezone / 3600, (timezone / 60) % 60); +#endif /* HAVE_TM_GMTOFF */ + else + snprintf(s, sizeof(s), "[%02d/%s/%04d:%02d:%02d:%02d.%06d %+03ld%02ld]", + date->tm_mday, months[date->tm_mon], 1900 + date->tm_year, + date->tm_hour, date->tm_min, date->tm_sec, (int)t->tv_usec, +#ifdef HAVE_TM_GMTOFF + date->tm_gmtoff / 3600, (date->tm_gmtoff / 60) % 60); +#else + timezone / 3600, (timezone / 60) % 60); +#endif /* HAVE_TM_GMTOFF */ + } + + return (s); +} + + +/* + * 'cupsdLogFCMessage()' - Log a file checking message. + */ + +void +cupsdLogFCMessage( + void *context, /* I - Printer (if any) */ + _cups_fc_result_t result, /* I - Check result */ + const char *message) /* I - Message to log */ +{ + cupsd_printer_t *p = (cupsd_printer_t *)context; + /* Printer */ + cupsd_loglevel_t level; /* Log level */ + + + if (result == _CUPS_FILE_CHECK_OK) + level = CUPSD_LOG_DEBUG2; + else + level = CUPSD_LOG_ERROR; + + if (p) + { + cupsdLogMessage(level, "%s: %s", p->name, message); + + if (result == _CUPS_FILE_CHECK_MISSING || + result == _CUPS_FILE_CHECK_WRONG_TYPE) + { + strlcpy(p->state_message, message, sizeof(p->state_message)); + + if (cupsdSetPrinterReasons(p, "+cups-missing-filter-warning")) + cupsdAddEvent(CUPSD_EVENT_PRINTER_STATE, p, NULL, "%s", message); + } + else if (result == _CUPS_FILE_CHECK_PERMISSIONS || + result == _CUPS_FILE_CHECK_RELATIVE_PATH) + { + strlcpy(p->state_message, message, sizeof(p->state_message)); + + if (cupsdSetPrinterReasons(p, "+cups-insecure-filter-warning")) + cupsdAddEvent(CUPSD_EVENT_PRINTER_STATE, p, NULL, "%s", message); + } + } + else + cupsdLogMessage(level, "%s", message); +} + + +#ifdef HAVE_GSSAPI +/* + * 'cupsdLogGSSMessage()' - Log a GSSAPI error... + */ + +int /* O - 1 on success, 0 on error */ +cupsdLogGSSMessage( + int level, /* I - Log level */ + int major_status, /* I - Major GSSAPI status */ + int minor_status, /* I - Minor GSSAPI status */ + const char *message, /* I - printf-style message string */ + ...) /* I - Additional args as needed */ +{ + OM_uint32 err_major_status, /* Major status code for display */ + err_minor_status; /* Minor status code for display */ + OM_uint32 msg_ctx; /* Message context */ + gss_buffer_desc major_status_string = GSS_C_EMPTY_BUFFER, + /* Major status message */ + minor_status_string = GSS_C_EMPTY_BUFFER; + /* Minor status message */ + int ret; /* Return value */ + char buffer[8192]; /* Buffer for vsnprintf */ + + + if (strchr(message, '%')) + { + /* + * Format the message string... + */ + + va_list ap; /* Pointer to arguments */ + + va_start(ap, message); + vsnprintf(buffer, sizeof(buffer), message, ap); + va_end(ap); + + message = buffer; + } + + msg_ctx = 0; + err_major_status = gss_display_status(&err_minor_status, + major_status, + GSS_C_GSS_CODE, + GSS_C_NO_OID, + &msg_ctx, + &major_status_string); + + if (!GSS_ERROR(err_major_status)) + gss_display_status(&err_minor_status, minor_status, GSS_C_MECH_CODE, + GSS_C_NULL_OID, &msg_ctx, &minor_status_string); + + ret = cupsdLogMessage(level, "%s: %s, %s", message, + (char *)major_status_string.value, + (char *)minor_status_string.value); + gss_release_buffer(&err_minor_status, &major_status_string); + gss_release_buffer(&err_minor_status, &minor_status_string); + + return (ret); +} +#endif /* HAVE_GSSAPI */ + + +/* + * 'cupsdLogJob()' - Log a job message. + */ + +int /* O - 1 on success, 0 on error */ +cupsdLogJob(cupsd_job_t *job, /* I - Job */ + int level, /* I - Log level */ + const char *message, /* I - Printf-style message string */ + ...) /* I - Additional arguments as needed */ +{ + va_list ap, ap2; /* Argument pointers */ + char jobmsg[1024]; /* Format string for job message */ + int status; /* Formatting status */ + + + /* + * See if we want to log this message... + */ + + if (TestConfigFile || !ErrorLog) + return (1); + + if ((level > LogLevel || + (level == CUPSD_LOG_INFO && LogLevel < CUPSD_LOG_DEBUG)) && + LogDebugHistory <= 0) + return (1); + + /* + * Format and write the log message... + */ + + if (job) + snprintf(jobmsg, sizeof(jobmsg), "[Job %d] %s", job->id, message); + else + strlcpy(jobmsg, message, sizeof(jobmsg)); + + va_start(ap, message); + + do + { + va_copy(ap2, ap); + status = format_log_line(jobmsg, ap2); + va_end(ap2); + } + while (status == 0); + + va_end(ap); + + if (status > 0) + { + if (job && + (level > LogLevel || + (level == CUPSD_LOG_INFO && LogLevel < CUPSD_LOG_DEBUG)) && + LogDebugHistory > 0) + { + /* + * Add message to the job history... + */ + + cupsd_joblog_t *temp; /* Copy of log message */ + + + if ((temp = malloc(sizeof(cupsd_joblog_t) + strlen(log_line))) != NULL) + { + temp->time = time(NULL); + strcpy(temp->message, log_line); + } + + if (!job->history) + job->history = cupsArrayNew(NULL, NULL); + + if (job->history && temp) + { + cupsArrayAdd(job->history, temp); + + if (cupsArrayCount(job->history) > LogDebugHistory) + { + /* + * Remove excess messages... + */ + + temp = cupsArrayFirst(job->history); + cupsArrayRemove(job->history, temp); + free(temp); + } + } + else if (temp) + free(temp); + + return (1); + } + else if (level <= LogLevel && + (level != CUPSD_LOG_INFO || LogLevel >= CUPSD_LOG_DEBUG)) + return (cupsdWriteErrorLog(level, log_line)); + else + return (1); + } + else + return (cupsdWriteErrorLog(CUPSD_LOG_ERROR, + "Unable to allocate memory for log line!")); +} + + +/* + * 'cupsdLogMessage()' - Log a message to the error log file. + */ + +int /* O - 1 on success, 0 on error */ +cupsdLogMessage(int level, /* I - Log level */ + const char *message, /* I - printf-style message string */ + ...) /* I - Additional args as needed */ +{ + va_list ap, ap2; /* Argument pointers */ + int status; /* Formatting status */ + + + /* + * See if we want to log this message... + */ + + if ((TestConfigFile || !ErrorLog) && level <= CUPSD_LOG_WARN) + { + va_start(ap, message); + vfprintf(stderr, message, ap); + putc('\n', stderr); + va_end(ap); + + return (1); + } + + if (level > LogLevel || !ErrorLog) + return (1); + + /* + * Format and write the log message... + */ + + va_start(ap, message); + + do + { + va_copy(ap2, ap); + status = format_log_line(message, ap2); + va_end(ap2); + } + while (status == 0); + + va_end(ap); + + if (status > 0) + return (cupsdWriteErrorLog(level, log_line)); + else + return (cupsdWriteErrorLog(CUPSD_LOG_ERROR, + "Unable to allocate memory for log line!")); +} + + +/* + * 'cupsdLogPage()' - Log a page to the page log file. + */ + +int /* O - 1 on success, 0 on error */ +cupsdLogPage(cupsd_job_t *job, /* I - Job being printed */ + const char *page) /* I - Page being printed */ +{ + int i; /* Looping var */ + char buffer[2048], /* Buffer for page log */ + *bufptr, /* Pointer into buffer */ + name[256]; /* Attribute name */ + const char *format, /* Pointer into PageLogFormat */ + *nameend; /* End of attribute name */ + ipp_attribute_t *attr; /* Current attribute */ + char number[256]; /* Page number */ + int copies; /* Number of copies */ + + + /* + * Format the line going into the page log... + */ + + if (!PageLogFormat) + return (1); + + strcpy(number, "1"); + copies = 1; + sscanf(page, "%255s%d", number, &copies); + + for (format = PageLogFormat, bufptr = buffer; *format; format ++) + { + if (*format == '%') + { + format ++; + + switch (*format) + { + case '%' : /* Literal % */ + if (bufptr < (buffer + sizeof(buffer) - 1)) + *bufptr++ = '%'; + break; + + case 'p' : /* Printer name */ + strlcpy(bufptr, job->printer->name, + sizeof(buffer) - (bufptr - buffer)); + bufptr += strlen(bufptr); + break; + + case 'j' : /* Job ID */ + snprintf(bufptr, sizeof(buffer) - (bufptr - buffer), "%d", job->id); + bufptr += strlen(bufptr); + break; + + case 'u' : /* Username */ + strlcpy(bufptr, job->username ? job->username : "-", + sizeof(buffer) - (bufptr - buffer)); + bufptr += strlen(bufptr); + break; + + case 'T' : /* Date and time */ + strlcpy(bufptr, cupsdGetDateTime(NULL, LogTimeFormat), + sizeof(buffer) - (bufptr - buffer)); + bufptr += strlen(bufptr); + break; + + case 'P' : /* Page number */ + strlcpy(bufptr, number, sizeof(buffer) - (bufptr - buffer)); + bufptr += strlen(bufptr); + break; + + case 'C' : /* Number of copies */ + snprintf(bufptr, sizeof(buffer) - (bufptr - buffer), "%d", copies); + bufptr += strlen(bufptr); + break; + + case '{' : /* {attribute} */ + if ((nameend = strchr(format, '}')) != NULL && + (nameend - format - 2) < (sizeof(name) - 1)) + { + /* + * Pull the name from inside the brackets... + */ + + memcpy(name, format + 1, nameend - format - 1); + name[nameend - format - 1] = '\0'; + + format = nameend; + + if ((attr = ippFindAttribute(job->attrs, name, + IPP_TAG_ZERO)) != NULL) + { + /* + * Add the attribute value... + */ + + for (i = 0; + i < attr->num_values && + bufptr < (buffer + sizeof(buffer) - 1); + i ++) + { + if (i) + *bufptr++ = ','; + + switch (attr->value_tag) + { + case IPP_TAG_INTEGER : + case IPP_TAG_ENUM : + snprintf(bufptr, sizeof(buffer) - (bufptr - buffer), + "%d", attr->values[i].integer); + bufptr += strlen(bufptr); + break; + + case IPP_TAG_BOOLEAN : + snprintf(bufptr, sizeof(buffer) - (bufptr - buffer), + "%d", attr->values[i].boolean); + bufptr += strlen(bufptr); + break; + + case IPP_TAG_TEXTLANG : + case IPP_TAG_NAMELANG : + case IPP_TAG_TEXT : + case IPP_TAG_NAME : + case IPP_TAG_KEYWORD : + case IPP_TAG_URI : + case IPP_TAG_URISCHEME : + case IPP_TAG_CHARSET : + case IPP_TAG_LANGUAGE : + case IPP_TAG_MIMETYPE : + strlcpy(bufptr, attr->values[i].string.text, + sizeof(buffer) - (bufptr - buffer)); + bufptr += strlen(bufptr); + break; + + default : + strlcpy(bufptr, "???", + sizeof(buffer) - (bufptr - buffer)); + bufptr += strlen(bufptr); + break; + } + } + } + else if (bufptr < (buffer + sizeof(buffer) - 1)) + *bufptr++ = '-'; + break; + } + + default : + if (bufptr < (buffer + sizeof(buffer) - 2)) + { + *bufptr++ = '%'; + *bufptr++ = *format; + } + break; + } + } + else if (bufptr < (buffer + sizeof(buffer) - 1)) + *bufptr++ = *format; + } + + *bufptr = '\0'; + +#ifdef HAVE_VSYSLOG + /* + * See if we are logging pages via syslog... + */ + + if (!strcmp(PageLog, "syslog")) + { + syslog(LOG_INFO, "%s", buffer); + + return (1); + } +#endif /* HAVE_VSYSLOG */ + + /* + * Not using syslog; check the log file... + */ + + if (!cupsdCheckLogFile(&PageFile, PageLog)) + return (0); + + /* + * Print a page log entry of the form: + * + * printer user job-id [DD/MON/YYYY:HH:MM:SS +TTTT] page num-copies \ + * billing hostname + */ + + cupsFilePrintf(PageFile, "%s\n", buffer); + cupsFileFlush(PageFile); + + return (1); +} + + +/* + * 'cupsdLogRequest()' - Log an HTTP request in Common Log Format. + */ + +int /* O - 1 on success, 0 on error */ +cupsdLogRequest(cupsd_client_t *con, /* I - Request to log */ + http_status_t code) /* I - Response code */ +{ + char temp[2048]; /* Temporary string for URI */ + static const char * const states[] = /* HTTP client states... */ + { + "WAITING", + "OPTIONS", + "GET", + "GET", + "HEAD", + "POST", + "POST", + "POST", + "PUT", + "PUT", + "DELETE", + "TRACE", + "CLOSE", + "STATUS" + }; + + + /* + * Filter requests as needed... + */ + + if (AccessLogLevel < CUPSD_ACCESSLOG_ALL) + { + /* + * Eliminate simple GET, POST, and PUT requests... + */ + + if ((con->operation == HTTP_GET && + strncmp(con->uri, "/admin/conf", 11) && + strncmp(con->uri, "/admin/log", 10)) || + (con->operation == HTTP_POST && !con->request && + strncmp(con->uri, "/admin", 6)) || + (con->operation != HTTP_GET && con->operation != HTTP_POST && + con->operation != HTTP_PUT)) + return (1); + + if (con->request && con->response && + (con->response->request.status.status_code < IPP_REDIRECTION_OTHER_SITE || + con->response->request.status.status_code == IPP_NOT_FOUND)) + { + /* + * Check successful requests... + */ + + ipp_op_t op = con->request->request.op.operation_id; + static cupsd_accesslog_t standard_ops[] = + { + CUPSD_ACCESSLOG_ALL, /* reserved */ + CUPSD_ACCESSLOG_ALL, /* reserved */ + CUPSD_ACCESSLOG_ACTIONS,/* Print-Job */ + CUPSD_ACCESSLOG_ACTIONS,/* Print-URI */ + CUPSD_ACCESSLOG_ACTIONS,/* Validate-Job */ + CUPSD_ACCESSLOG_ACTIONS,/* Create-Job */ + CUPSD_ACCESSLOG_ACTIONS,/* Send-Document */ + CUPSD_ACCESSLOG_ACTIONS,/* Send-URI */ + CUPSD_ACCESSLOG_ACTIONS,/* Cancel-Job */ + CUPSD_ACCESSLOG_ALL, /* Get-Job-Attributes */ + CUPSD_ACCESSLOG_ALL, /* Get-Jobs */ + CUPSD_ACCESSLOG_ALL, /* Get-Printer-Attributes */ + CUPSD_ACCESSLOG_ACTIONS,/* Hold-Job */ + CUPSD_ACCESSLOG_ACTIONS,/* Release-Job */ + CUPSD_ACCESSLOG_ACTIONS,/* Restart-Job */ + CUPSD_ACCESSLOG_ALL, /* reserved */ + CUPSD_ACCESSLOG_CONFIG, /* Pause-Printer */ + CUPSD_ACCESSLOG_CONFIG, /* Resume-Printer */ + CUPSD_ACCESSLOG_CONFIG, /* Purge-Jobs */ + CUPSD_ACCESSLOG_CONFIG, /* Set-Printer-Attributes */ + CUPSD_ACCESSLOG_ACTIONS,/* Set-Job-Attributes */ + CUPSD_ACCESSLOG_CONFIG, /* Get-Printer-Supported-Values */ + CUPSD_ACCESSLOG_ACTIONS,/* Create-Printer-Subscription */ + CUPSD_ACCESSLOG_ACTIONS,/* Create-Job-Subscription */ + CUPSD_ACCESSLOG_ALL, /* Get-Subscription-Attributes */ + CUPSD_ACCESSLOG_ALL, /* Get-Subscriptions */ + CUPSD_ACCESSLOG_ACTIONS,/* Renew-Subscription */ + CUPSD_ACCESSLOG_ACTIONS,/* Cancel-Subscription */ + CUPSD_ACCESSLOG_ALL, /* Get-Notifications */ + CUPSD_ACCESSLOG_ACTIONS,/* Send-Notifications */ + CUPSD_ACCESSLOG_ALL, /* reserved */ + CUPSD_ACCESSLOG_ALL, /* reserved */ + CUPSD_ACCESSLOG_ALL, /* reserved */ + CUPSD_ACCESSLOG_ALL, /* Get-Print-Support-Files */ + CUPSD_ACCESSLOG_CONFIG, /* Enable-Printer */ + CUPSD_ACCESSLOG_CONFIG, /* Disable-Printer */ + CUPSD_ACCESSLOG_CONFIG, /* Pause-Printer-After-Current-Job */ + CUPSD_ACCESSLOG_ACTIONS,/* Hold-New-Jobs */ + CUPSD_ACCESSLOG_ACTIONS,/* Release-Held-New-Jobs */ + CUPSD_ACCESSLOG_CONFIG, /* Deactivate-Printer */ + CUPSD_ACCESSLOG_CONFIG, /* Activate-Printer */ + CUPSD_ACCESSLOG_CONFIG, /* Restart-Printer */ + CUPSD_ACCESSLOG_CONFIG, /* Shutdown-Printer */ + CUPSD_ACCESSLOG_CONFIG, /* Startup-Printer */ + CUPSD_ACCESSLOG_ACTIONS,/* Reprocess-Job */ + CUPSD_ACCESSLOG_ACTIONS,/* Cancel-Current-Job */ + CUPSD_ACCESSLOG_ACTIONS,/* Suspend-Current-Job */ + CUPSD_ACCESSLOG_ACTIONS,/* Resume-Job */ + CUPSD_ACCESSLOG_ACTIONS,/* Promote-Job */ + CUPSD_ACCESSLOG_ACTIONS /* Schedule-Job-After */ + }; + static cupsd_accesslog_t cups_ops[] = + { + CUPSD_ACCESSLOG_ALL, /* CUPS-Get-Default */ + CUPSD_ACCESSLOG_ALL, /* CUPS-Get-Printers */ + CUPSD_ACCESSLOG_CONFIG, /* CUPS-Add-Modify-Printer */ + CUPSD_ACCESSLOG_CONFIG, /* CUPS-Delete-Printer */ + CUPSD_ACCESSLOG_ALL, /* CUPS-Get-Classes */ + CUPSD_ACCESSLOG_CONFIG, /* CUPS-Add-Modify-Class */ + CUPSD_ACCESSLOG_CONFIG, /* CUPS-Delete-Class */ + CUPSD_ACCESSLOG_CONFIG, /* CUPS-Accept-Jobs */ + CUPSD_ACCESSLOG_CONFIG, /* CUPS-Reject-Jobs */ + CUPSD_ACCESSLOG_CONFIG, /* CUPS-Set-Default */ + CUPSD_ACCESSLOG_CONFIG, /* CUPS-Get-Devices */ + CUPSD_ACCESSLOG_CONFIG, /* CUPS-Get-PPDs */ + CUPSD_ACCESSLOG_ACTIONS,/* CUPS-Move-Job */ + CUPSD_ACCESSLOG_ACTIONS,/* CUPS-Authenticate-Job */ + CUPSD_ACCESSLOG_ALL /* CUPS-Get-PPD */ + }; + + + if ((op <= IPP_SCHEDULE_JOB_AFTER && standard_ops[op] > AccessLogLevel) || + (op >= CUPS_GET_DEFAULT && op <= CUPS_GET_PPD && + cups_ops[op - CUPS_GET_DEFAULT] > AccessLogLevel)) + return (1); + } + } + +#ifdef HAVE_VSYSLOG + /* + * See if we are logging accesses via syslog... + */ + + if (!strcmp(AccessLog, "syslog")) + { + syslog(LOG_INFO, + "REQUEST %s - %s \"%s %s HTTP/%d.%d\" %d " CUPS_LLFMT " %s %s\n", + con->http.hostname, con->username[0] != '\0' ? con->username : "-", + states[con->operation], _httpEncodeURI(temp, con->uri, sizeof(temp)), + con->http.version / 100, con->http.version % 100, + code, CUPS_LLCAST con->bytes, + con->request ? + ippOpString(con->request->request.op.operation_id) : "-", + con->response ? + ippErrorString(con->response->request.status.status_code) : "-"); + + return (1); + } +#endif /* HAVE_VSYSLOG */ + + /* + * Not using syslog; check the log file... + */ + + if (!cupsdCheckLogFile(&AccessFile, AccessLog)) + return (0); + + /* + * Write a log of the request in "common log format"... + */ + + cupsFilePrintf(AccessFile, + "%s - %s %s \"%s %s HTTP/%d.%d\" %d " CUPS_LLFMT " %s %s\n", + con->http.hostname, + con->username[0] != '\0' ? con->username : "-", + cupsdGetDateTime(&(con->start), LogTimeFormat), + states[con->operation], + _httpEncodeURI(temp, con->uri, sizeof(temp)), + con->http.version / 100, con->http.version % 100, + code, CUPS_LLCAST con->bytes, + con->request ? + ippOpString(con->request->request.op.operation_id) : "-", + con->response ? + ippErrorString(con->response->request.status.status_code) : + "-"); + + cupsFileFlush(AccessFile); + + return (1); +} + + +/* + * 'cupsdWriteErrorLog()' - Write a line to the ErrorLog. + */ + +int /* O - 1 on success, 0 on failure */ +cupsdWriteErrorLog(int level, /* I - Log level */ + const char *message) /* I - Message string */ +{ + static const char levels[] = /* Log levels... */ + { + ' ', + 'X', + 'A', + 'C', + 'E', + 'W', + 'N', + 'I', + 'D', + 'd' + }; +#ifdef HAVE_VSYSLOG + static const int syslevels[] = /* SYSLOG levels... */ + { + 0, + LOG_EMERG, + LOG_ALERT, + LOG_CRIT, + LOG_ERR, + LOG_WARNING, + LOG_NOTICE, + LOG_INFO, + LOG_DEBUG, + LOG_DEBUG + }; +#endif /* HAVE_VSYSLOG */ + + +#ifdef HAVE_VSYSLOG + /* + * See if we are logging errors via syslog... + */ + + if (!strcmp(ErrorLog, "syslog")) + { + syslog(syslevels[level], "%s", message); + return (1); + } +#endif /* HAVE_VSYSLOG */ + + /* + * Not using syslog; check the log file... + */ + + if (!cupsdCheckLogFile(&ErrorFile, ErrorLog)) + return (0); + + /* + * Write the log message... + */ + + cupsFilePrintf(ErrorFile, "%c %s %s\n", levels[level], + cupsdGetDateTime(NULL, LogTimeFormat), message); + cupsFileFlush(ErrorFile); + + return (1); +} + + +/* + * 'format_log_line()' - Format a line for a log file. + * + * This function resizes a global string buffer as needed. Each call returns + * a pointer to this buffer, so the contents are only good until the next call + * to format_log_line()... + */ + +static int /* O - -1 for fatal, 0 for retry, 1 for success */ +format_log_line(const char *message, /* I - Printf-style format string */ + va_list ap) /* I - Argument list */ +{ + int len; /* Length of formatted line */ + + + /* + * Allocate the line buffer as needed... + */ + + if (!log_linesize) + { + log_linesize = 8192; + log_line = malloc(log_linesize); + + if (!log_line) + return (-1); + } + + /* + * Format the log message... + */ + + len = vsnprintf(log_line, log_linesize, message, ap); + + /* + * Resize the buffer as needed... + */ + + if (len >= log_linesize && log_linesize < 65536) + { + char *temp; /* Temporary string pointer */ + + + len ++; + + if (len < 8192) + len = 8192; + else if (len > 65536) + len = 65536; + + temp = realloc(log_line, len); + + if (temp) + { + log_line = temp; + log_linesize = len; + + return (0); + } + } + + return (1); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/main.c b/scheduler/main.c new file mode 100644 index 0000000000..b7eb6ee1d1 --- /dev/null +++ b/scheduler/main.c @@ -0,0 +1,2005 @@ +/* + * "$Id$" + * + * Main loop for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Main entry for the CUPS scheduler. + * cupsdAddString() - Copy and add a string to an array. + * cupsdCheckProcess() - Tell the main loop to check for dead children. + * cupsdClearString() - Clear a string. + * cupsdFreeStrings() - Free an array of strings. + * cupsdHoldSignals() - Hold child and termination signals. + * cupsdReleaseSignals() - Release signals for delivery. + * cupsdSetString() - Set a string value. + * cupsdSetStringf() - Set a formatted string value. + * launchd_checkin() - Check-in with launchd and collect the listening + * fds. + * launchd_checkout() - Update the launchd KeepAlive file as needed. + * parent_handler() - Catch USR1/CHLD signals... + * process_children() - Process all dead children... + * select_timeout() - Calculate the select timeout value. + * sigchld_handler() - Handle 'child' signals from old processes. + * sighup_handler() - Handle 'hangup' signals to reconfigure the + * scheduler. + * sigterm_handler() - Handle 'terminate' signals that stop the scheduler. + * usage() - Show scheduler usage. + */ + +/* + * Include necessary headers... + */ + +#define _MAIN_C_ +#include "cupsd.h" +#include +#include +#include + +#ifdef HAVE_LAUNCH_H +# include +# include +# define CUPS_KEEPALIVE CUPS_CACHEDIR "/org.cups.cupsd" + /* Name of the launchd KeepAlive file */ +# ifndef LAUNCH_JOBKEY_KEEPALIVE +# define LAUNCH_JOBKEY_KEEPALIVE "KeepAlive" +# endif /* !LAUNCH_JOBKEY_KEEPALIVE */ +# ifndef LAUNCH_JOBKEY_PATHSTATE +# define LAUNCH_JOBKEY_PATHSTATE "PathState" +# endif /* !LAUNCH_JOBKEY_PATHSTATE */ +# ifndef LAUNCH_JOBKEY_SERVICEIPC +# define LAUNCH_JOBKEY_SERVICEIPC "ServiceIPC" +# endif /* !LAUNCH_JOBKEY_SERVICEIPC */ +#endif /* HAVE_LAUNCH_H */ + +#if defined(HAVE_MALLOC_H) && defined(HAVE_MALLINFO) +# include +#endif /* HAVE_MALLOC_H && HAVE_MALLINFO */ +#ifdef HAVE_NOTIFY_H +# include +#endif /* HAVE_NOTIFY_H */ + + +/* + * Local functions... + */ + +#ifdef HAVE_LAUNCHD +static void launchd_checkin(void); +static void launchd_checkout(void); +#endif /* HAVE_LAUNCHD */ +static void parent_handler(int sig); +static void process_children(void); +static void sigchld_handler(int sig); +static void sighup_handler(int sig); +static void sigterm_handler(int sig); +static long select_timeout(int fds); +static void usage(int status) __attribute__((noreturn)); + + +/* + * Local globals... + */ + +static int parent_signal = 0; + /* Set to signal number from child */ +static int holdcount = 0; /* Number of times "hold" was called */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) +static sigset_t holdmask; /* Old POSIX signal mask */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ +static int dead_children = 0; + /* Dead children? */ +static int stop_scheduler = 0; + /* Should the scheduler stop? */ + + +/* + * 'main()' - Main entry for the CUPS scheduler. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + char *opt; /* Option character */ + int fg; /* Run in the foreground */ + int fds; /* Number of ready descriptors */ + cupsd_client_t *con; /* Current client */ + cupsd_job_t *job; /* Current job */ + cupsd_listener_t *lis; /* Current listener */ + time_t current_time, /* Current time */ + activity, /* Client activity timer */ + senddoc_time, /* Send-Document time */ + expire_time, /* Subscription expire time */ + report_time, /* Malloc/client/job report time */ + event_time; /* Last event notification time */ + long timeout; /* Timeout for cupsdDoSelect() */ + struct rlimit limit; /* Runtime limit */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* Actions for POSIX signals */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ +#ifdef __sgi + cups_file_t *fp; /* Fake lpsched lock file */ + struct stat statbuf; /* Needed for checking lpsched FIFO */ +#endif /* __sgi */ + int run_as_child = 0; + /* Needed for background fork/exec */ +#ifdef __APPLE__ + int use_sysman = !getuid(); + /* Use system management functions? */ +#else + time_t netif_time = 0; /* Time since last network update */ +#endif /* __APPLE__ */ +#if HAVE_LAUNCHD + int launchd_idle_exit; + /* Idle exit on select timeout? */ +#endif /* HAVE_LAUNCHD */ + + +#ifdef HAVE_GETEUID + /* + * Check for setuid invocation, which we do not support! + */ + + if (getuid() != geteuid()) + { + fputs("cupsd: Cannot run as a setuid program\n", stderr); + return (1); + } +#endif /* HAVE_GETEUID */ + + /* + * Check for command-line arguments... + */ + + fg = 0; + +#ifdef HAVE_LAUNCHD + if (getenv("CUPSD_LAUNCHD")) + { + Launchd = 1; + fg = 1; + } +#endif /* HAVE_LAUNCHD */ + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-') + for (opt = argv[i] + 1; *opt != '\0'; opt ++) + switch (*opt) + { + case 'C' : /* Run as child with config file */ + run_as_child = 1; + fg = -1; + + case 'c' : /* Configuration file */ + i ++; + if (i >= argc) + { + _cupsLangPuts(stderr, _("cupsd: Expected config filename " + "after \"-c\" option.")); + usage(1); + } + + if (argv[i][0] == '/') + { + /* + * Absolute directory... + */ + + cupsdSetString(&ConfigurationFile, argv[i]); + } + else + { + /* + * Relative directory... + */ + + char *current; /* Current directory */ + + + /* + * Allocate a buffer for the current working directory to + * reduce run-time stack usage; this approximates the + * behavior of some implementations of getcwd() when they + * are passed a NULL pointer. + */ + + if ((current = malloc(1024)) == NULL) + { + _cupsLangPuts(stderr, + _("cupsd: Unable to get current directory.")); + return (1); + } + + if (!getcwd(current, 1024)) + { + _cupsLangPuts(stderr, + _("cupsd: Unable to get current directory.")); + free(current); + return (1); + } + + cupsdSetStringf(&ConfigurationFile, "%s/%s", current, argv[i]); + free(current); + } + break; + + case 'f' : /* Run in foreground... */ + fg = 1; + break; + + case 'F' : /* Run in foreground, but disconnect from terminal... */ + fg = -1; + break; + + case 'h' : /* Show usage/help */ + usage(0); + break; + + case 'l' : /* Started by launchd... */ +#ifdef HAVE_LAUNCHD + Launchd = 1; + fg = 1; +#else + _cupsLangPuts(stderr, _("cupsd: launchd(8) support not compiled " + "in, running in normal mode.")); + fg = 0; +#endif /* HAVE_LAUNCHD */ + break; + + case 'p' : /* Stop immediately for profiling */ + fputs("cupsd: -p (startup profiling) is for internal testing " + "use only!\n", stderr); + stop_scheduler = 1; + fg = 1; + break; + + case 'P' : /* Disable security profiles */ + fputs("cupsd: -P (disable security profiles) is for internal " + "testing use only!\n", stderr); + UseProfiles = 0; + break; + +#ifdef __APPLE__ + case 'S' : /* Disable system management functions */ + fputs("cupsd: -S (disable system management) for internal " + "testing use only!\n", stderr); + use_sysman = 0; + break; +#endif /* __APPLE__ */ + + case 't' : /* Test the cupsd.conf file... */ + TestConfigFile = 1; + fg = 1; + break; + + default : /* Unknown option */ + _cupsLangPrintf(stderr, _("cupsd: Unknown option \"%c\" - " + "aborting."), *opt); + usage(1); + break; + } + else + { + _cupsLangPrintf(stderr, _("cupsd: Unknown argument \"%s\" - aborting."), + argv[i]); + usage(1); + } + + if (!ConfigurationFile) + cupsdSetString(&ConfigurationFile, CUPS_SERVERROOT "/cupsd.conf"); + + /* + * If the user hasn't specified "-f", run in the background... + */ + + if (!fg) + { + /* + * Setup signal handlers for the parent... + */ + +#ifdef HAVE_SIGSET /* Use System V signals over POSIX to avoid bugs */ + sigset(SIGUSR1, parent_handler); + sigset(SIGCHLD, parent_handler); + + sigset(SIGHUP, SIG_IGN); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + sigemptyset(&action.sa_mask); + sigaddset(&action.sa_mask, SIGUSR1); + action.sa_handler = parent_handler; + sigaction(SIGUSR1, &action, NULL); + sigaction(SIGCHLD, &action, NULL); + + sigemptyset(&action.sa_mask); + action.sa_handler = SIG_IGN; + sigaction(SIGHUP, &action, NULL); +#else + signal(SIGUSR1, parent_handler); + signal(SIGCLD, parent_handler); + + signal(SIGHUP, SIG_IGN); +#endif /* HAVE_SIGSET */ + + if (fork() > 0) + { + /* + * OK, wait for the child to startup and send us SIGUSR1 or to crash + * and the OS send us SIGCHLD... We also need to ignore SIGHUP which + * might be sent by the init script to restart the scheduler... + */ + + for (; parent_signal == 0;) + sleep(1); + + if (parent_signal == SIGUSR1) + return (0); + + if (wait(&i) < 0) + { + perror("cupsd"); + return (1); + } + else if (WIFEXITED(i)) + { + fprintf(stderr, "cupsd: Child exited with status %d\n", + WEXITSTATUS(i)); + return (2); + } + else + { + fprintf(stderr, "cupsd: Child exited on signal %d\n", WTERMSIG(i)); + return (3); + } + } + +#ifdef __OpenBSD__ + /* + * Call _thread_sys_closefrom() so the child process doesn't reset the + * parent's file descriptors to be blocking. This is a workaround for a + * limitation of userland libpthread on OpenBSD. + */ + + _thread_sys_closefrom(0); +#endif /* __OpenBSD__ */ + + /* + * Since CoreFoundation and DBUS both create fork-unsafe data on execution of + * a program, and since this kind of really unfriendly behavior seems to be + * more common these days in system libraries, we need to re-execute the + * background cupsd with the "-C" option to avoid problems. Unfortunately, + * we also have to assume that argv[0] contains the name of the cupsd + * executable - there is no portable way to get the real pathname... + */ + + execlp(argv[0], argv[0], "-C", ConfigurationFile, (char *)0); + exit(errno); + } + + if (fg < 1) + { + /* + * Make sure we aren't tying up any filesystems... + */ + + chdir("/"); + +#ifndef DEBUG + /* + * Disable core dumps... + */ + + getrlimit(RLIMIT_CORE, &limit); + limit.rlim_cur = 0; + setrlimit(RLIMIT_CORE, &limit); + + /* + * Disconnect from the controlling terminal... + */ + + setsid(); + + /* + * Close all open files... + */ + + getrlimit(RLIMIT_NOFILE, &limit); + + for (i = 0; i < limit.rlim_cur && i < 1024; i ++) + close(i); + + /* + * Redirect stdin/out/err to /dev/null... + */ + + if ((i = open("/dev/null", O_RDONLY)) != 0) + { + dup2(i, 0); + close(i); + } + + if ((i = open("/dev/null", O_WRONLY)) != 1) + { + dup2(i, 1); + close(i); + } + + if ((i = open("/dev/null", O_WRONLY)) != 2) + { + dup2(i, 2); + close(i); + } +#endif /* DEBUG */ + } + + /* + * Set the timezone info... + */ + + tzset(); + +#ifdef LC_TIME + setlocale(LC_TIME, ""); +#endif /* LC_TIME */ + + /* + * Set the maximum number of files... + */ + + getrlimit(RLIMIT_NOFILE, &limit); + +#if !defined(HAVE_POLL) && !defined(HAVE_EPOLL) && !defined(HAVE_KQUEUE) + if (limit.rlim_max > FD_SETSIZE) + MaxFDs = FD_SETSIZE; + else +#endif /* !HAVE_POLL && !HAVE_EPOLL && !HAVE_KQUEUE */ +#ifdef RLIM_INFINITY + if (limit.rlim_max == RLIM_INFINITY) + MaxFDs = 16384; + else +#endif /* RLIM_INFINITY */ + MaxFDs = limit.rlim_max; + + limit.rlim_cur = MaxFDs; + + setrlimit(RLIMIT_NOFILE, &limit); + + cupsdStartSelect(); + + /* + * Read configuration... + */ + + if (!cupsdReadConfiguration()) + { + if (TestConfigFile) + printf("%s contains errors\n", ConfigurationFile); + else + syslog(LOG_LPR, "Unable to read configuration file \'%s\' - exiting!", + ConfigurationFile); + return (1); + } + else if (TestConfigFile) + { + printf("%s is OK\n", ConfigurationFile); + return (0); + } + + /* + * Clean out old temp files and printer cache data. + */ + + if (!strncmp(TempDir, RequestRoot, strlen(RequestRoot))) + cupsdCleanFiles(TempDir, NULL); + + cupsdCleanFiles(CacheDir, "*.ipp"); + +#if HAVE_LAUNCHD + if (Launchd) + { + /* + * If we were started by launchd get the listen sockets file descriptors... + */ + + launchd_checkin(); + launchd_checkout(); + } +#endif /* HAVE_LAUNCHD */ + + /* + * Startup the server... + */ + + httpInitialize(); + + cupsdStartServer(); + + /* + * Catch hangup and child signals and ignore broken pipes... + */ + +#ifdef HAVE_SIGSET /* Use System V signals over POSIX to avoid bugs */ + sigset(SIGCHLD, sigchld_handler); + sigset(SIGHUP, sighup_handler); + sigset(SIGPIPE, SIG_IGN); + sigset(SIGTERM, sigterm_handler); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + + sigemptyset(&action.sa_mask); + sigaddset(&action.sa_mask, SIGTERM); + sigaddset(&action.sa_mask, SIGCHLD); + action.sa_handler = sigchld_handler; + sigaction(SIGCHLD, &action, NULL); + + sigemptyset(&action.sa_mask); + sigaddset(&action.sa_mask, SIGHUP); + action.sa_handler = sighup_handler; + sigaction(SIGHUP, &action, NULL); + + sigemptyset(&action.sa_mask); + action.sa_handler = SIG_IGN; + sigaction(SIGPIPE, &action, NULL); + + sigemptyset(&action.sa_mask); + sigaddset(&action.sa_mask, SIGTERM); + sigaddset(&action.sa_mask, SIGCHLD); + action.sa_handler = sigterm_handler; + sigaction(SIGTERM, &action, NULL); +#else + signal(SIGCLD, sigchld_handler); /* No, SIGCLD isn't a typo... */ + signal(SIGHUP, sighup_handler); + signal(SIGPIPE, SIG_IGN); + signal(SIGTERM, sigterm_handler); +#endif /* HAVE_SIGSET */ + +#ifdef __sgi + /* + * Try to create a fake lpsched lock file if one is not already there. + * Some Adobe applications need it under IRIX in order to enable + * printing... + */ + + if ((fp = cupsFileOpen("/var/spool/lp/SCHEDLOCK", "w")) == NULL) + { + syslog(LOG_LPR, "Unable to create fake lpsched lock file " + "\"/var/spool/lp/SCHEDLOCK\"\' - %s!", + strerror(errno)); + } + else + { + fchmod(cupsFileNumber(fp), 0644); + fchown(cupsFileNumber(fp), User, Group); + + cupsFileClose(fp); + } +#endif /* __sgi */ + + /* + * Initialize authentication certificates... + */ + + cupsdInitCerts(); + + /* + * If we are running in the background, signal the parent process that + * we are up and running... + */ + + if (!fg || run_as_child) + { + /* + * Send a signal to the parent process, but only if the parent is + * not PID 1 (init). This avoids accidentally shutting down the + * system on OpenBSD if you CTRL-C the server before it is up... + */ + + i = getppid(); /* Save parent PID to avoid race condition */ + + if (i != 1) + kill(i, SIGUSR1); + } + +#ifdef __APPLE__ + /* + * Start power management framework... + */ + + if (use_sysman) + cupsdStartSystemMonitor(); +#endif /* __APPLE__ */ + + /* + * Send server-started event... + */ + +#ifdef HAVE_LAUNCHD + if (Launchd) + cupsdAddEvent(CUPSD_EVENT_SERVER_STARTED, NULL, NULL, + "Scheduler started via launchd."); + else +#endif /* HAVE_LAUNCHD */ + if (fg) + cupsdAddEvent(CUPSD_EVENT_SERVER_STARTED, NULL, NULL, + "Scheduler started in foreground."); + else + cupsdAddEvent(CUPSD_EVENT_SERVER_STARTED, NULL, NULL, + "Scheduler started in background."); + + /* + * Start any pending print jobs... + */ + + cupsdCheckJobs(); + + /* + * Loop forever... + */ + + current_time = time(NULL); + event_time = current_time; + expire_time = current_time; + fds = 1; + report_time = 0; + senddoc_time = current_time; + + while (!stop_scheduler) + { + /* + * Check if there are dead children to handle... + */ + + if (dead_children) + process_children(); + + /* + * Check if we need to load the server configuration file... + */ + + if (NeedReload) + { + /* + * Close any idle clients... + */ + + if (cupsArrayCount(Clients) > 0) + { + for (con = (cupsd_client_t *)cupsArrayFirst(Clients); + con; + con = (cupsd_client_t *)cupsArrayNext(Clients)) + if (con->http.state == HTTP_WAITING) + cupsdCloseClient(con); + else + con->http.keep_alive = HTTP_KEEPALIVE_OFF; + + cupsdPauseListening(); + } + + /* + * Restart if all clients are closed and all jobs finished, or + * if the reload timeout has elapsed... + */ + + if ((cupsArrayCount(Clients) == 0 && + (cupsArrayCount(PrintingJobs) == 0 || NeedReload != RELOAD_ALL)) || + (time(NULL) - ReloadTime) >= ReloadTimeout) + { + /* + * Shutdown the server... + */ + + DoingShutdown = 1; + + cupsdStopServer(); + + /* + * Read configuration... + */ + + if (!cupsdReadConfiguration()) + { + syslog(LOG_LPR, "Unable to read configuration file \'%s\' - exiting!", + ConfigurationFile); + break; + } + +#if HAVE_LAUNCHD + if (Launchd) + { + /* + * If we were started by launchd, get the listen socket file + * descriptors... + */ + + launchd_checkin(); + launchd_checkout(); + } +#endif /* HAVE_LAUNCHD */ + + /* + * Startup the server... + */ + + DoingShutdown = 0; + + cupsdStartServer(); + + /* + * Send a server-restarted event... + */ + + cupsdAddEvent(CUPSD_EVENT_SERVER_RESTARTED, NULL, NULL, + "Scheduler restarted."); + } + } + + /* + * Check for available input or ready output. If cupsdDoSelect() + * returns 0 or -1, something bad happened and we should exit + * immediately. + * + * Note that we at least have one listening socket open at all + * times. + */ + + if ((timeout = select_timeout(fds)) > 1 && LastEvent) + timeout = 1; + +#if HAVE_LAUNCHD + /* + * If no other work is scheduled and we're being controlled by + * launchd then timeout after 'LaunchdTimeout' seconds of + * inactivity... + */ + + if (timeout == 86400 && Launchd && LaunchdTimeout && + !cupsArrayCount(ActiveJobs) && + (!Browsing || !BrowseLocalProtocols || !cupsArrayCount(Printers))) + { + timeout = LaunchdTimeout; + launchd_idle_exit = 1; + } + else + launchd_idle_exit = 0; +#endif /* HAVE_LAUNCHD */ + + if ((fds = cupsdDoSelect(timeout)) < 0) + { + /* + * Got an error from select! + */ + +#ifdef HAVE_DNSSD + cupsd_printer_t *p; /* Current printer */ +#endif /* HAVE_DNSSD */ + + + if (errno == EINTR) /* Just interrupted by a signal */ + continue; + + /* + * Log all sorts of debug info to help track down the problem. + */ + + cupsdLogMessage(CUPSD_LOG_EMERG, "cupsdDoSelect() failed - %s!", + strerror(errno)); + + for (i = 0, con = (cupsd_client_t *)cupsArrayFirst(Clients); + con; + i ++, con = (cupsd_client_t *)cupsArrayNext(Clients)) + cupsdLogMessage(CUPSD_LOG_EMERG, + "Clients[%d] = %d, file = %d, state = %d", + i, con->http.fd, con->file, con->http.state); + + for (i = 0, lis = (cupsd_listener_t *)cupsArrayFirst(Listeners); + lis; + i ++, lis = (cupsd_listener_t *)cupsArrayNext(Listeners)) + cupsdLogMessage(CUPSD_LOG_EMERG, "Listeners[%d] = %d", i, lis->fd); + + cupsdLogMessage(CUPSD_LOG_EMERG, "CGIPipes[0] = %d", CGIPipes[0]); + +#ifdef __APPLE__ + cupsdLogMessage(CUPSD_LOG_EMERG, "SysEventPipes[0] = %d", + SysEventPipes[0]); +#endif /* __APPLE__ */ + + for (job = (cupsd_job_t *)cupsArrayFirst(ActiveJobs); + job; + job = (cupsd_job_t *)cupsArrayNext(ActiveJobs)) + cupsdLogMessage(CUPSD_LOG_EMERG, "Jobs[%d] = %d < [%d %d] > [%d %d]", + job->id, + job->status_buffer ? job->status_buffer->fd : -1, + job->print_pipes[0], job->print_pipes[1], + job->back_pipes[0], job->back_pipes[1]); + +#ifdef HAVE_DNSSD + for (p = (cupsd_printer_t *)cupsArrayFirst(Printers); + p; + p = (cupsd_printer_t *)cupsArrayNext(Printers)) + cupsdLogMessage(CUPSD_LOG_EMERG, "printer[%s] reg_name=\"%s\"", p->name, + p->reg_name ? p->reg_name : "(null)"); +#endif /* HAVE_DNSSD */ + + break; + } + + current_time = time(NULL); + + /* + * Write dirty config/state files... + */ + + if (DirtyCleanTime && current_time >= DirtyCleanTime) + cupsdCleanDirty(); + +#ifdef __APPLE__ + /* + * If we are going to sleep and still have pending jobs, stop them after + * a period of time... + */ + + if (SleepJobs > 0 && current_time >= SleepJobs && + cupsArrayCount(PrintingJobs) > 0) + { + SleepJobs = 0; + cupsdStopAllJobs(CUPSD_JOB_DEFAULT, 5); + } +#endif /* __APPLE__ */ + +#ifndef __APPLE__ + /* + * Update the network interfaces once a minute... + */ + + if ((current_time - netif_time) >= 60) + { + netif_time = current_time; + NetIFUpdate = 1; + } +#endif /* !__APPLE__ */ + +#if HAVE_LAUNCHD + /* + * If no other work was scheduled and we're being controlled by launchd + * then timeout after 'LaunchdTimeout' seconds of inactivity... + */ + + if (!fds && launchd_idle_exit) + { + cupsdLogMessage(CUPSD_LOG_INFO, + "Printer sharing is off and there are no jobs pending, " + "will restart on demand."); + stop_scheduler = 1; + break; + } +#endif /* HAVE_LAUNCHD */ + + /* + * Resume listening for new connections as needed... + */ + + if (ListeningPaused && ListeningPaused <= current_time && + cupsArrayCount(Clients) < MaxClients) + cupsdResumeListening(); + + /* + * Expire subscriptions and unload completed jobs as needed... + */ + + if (current_time > expire_time) + { + if (cupsArrayCount(Subscriptions) > 0) + cupsdExpireSubscriptions(NULL, NULL); + + cupsdUnloadCompletedJobs(); + + expire_time = current_time; + } + +#ifndef HAVE_AUTHORIZATION_H + /* + * Update the root certificate once every 5 minutes if we have client + * connections... + */ + + if ((current_time - RootCertTime) >= RootCertDuration && RootCertDuration && + !RunUser && cupsArrayCount(Clients)) + { + /* + * Update the root certificate... + */ + + cupsdDeleteCert(0); + cupsdAddCert(0, "root", NULL); + } +#endif /* !HAVE_AUTHORIZATION_H */ + + /* + * Check for new data on the client sockets... + */ + + for (con = (cupsd_client_t *)cupsArrayFirst(Clients); + con; + con = (cupsd_client_t *)cupsArrayNext(Clients)) + { + /* + * Process pending data in the input buffer... + */ + + if (con->http.used) + { + cupsdReadClient(con); + continue; + } + + /* + * Check the activity and close old clients... + */ + + activity = current_time - Timeout; + if (con->http.activity < activity && !con->pipe_pid) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Closing client %d after %d seconds of inactivity...", + con->http.fd, Timeout); + + cupsdCloseClient(con); + continue; + } + } + + /* + * Update any pending multi-file documents... + */ + + if ((current_time - senddoc_time) >= 10) + { + cupsdCheckJobs(); + cupsdCleanJobs(); + senddoc_time = current_time; + } + + /* + * Log statistics at most once a minute when in debug mode... + */ + + if ((current_time - report_time) >= 60 && LogLevel >= CUPSD_LOG_DEBUG) + { + size_t string_count, /* String count */ + alloc_bytes, /* Allocated string bytes */ + total_bytes; /* Total string bytes */ +#ifdef HAVE_MALLINFO + struct mallinfo mem; /* Malloc information */ + + + mem = mallinfo(); + cupsdLogMessage(CUPSD_LOG_DEBUG, "Report: malloc-arena=%lu", mem.arena); + cupsdLogMessage(CUPSD_LOG_DEBUG, "Report: malloc-used=%lu", + mem.usmblks + mem.uordblks); + cupsdLogMessage(CUPSD_LOG_DEBUG, "Report: malloc-free=%lu", + mem.fsmblks + mem.fordblks); +#endif /* HAVE_MALLINFO */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, "Report: clients=%d", + cupsArrayCount(Clients)); + cupsdLogMessage(CUPSD_LOG_DEBUG, "Report: jobs=%d", + cupsArrayCount(Jobs)); + cupsdLogMessage(CUPSD_LOG_DEBUG, "Report: jobs-active=%d", + cupsArrayCount(ActiveJobs)); + cupsdLogMessage(CUPSD_LOG_DEBUG, "Report: printers=%d", + cupsArrayCount(Printers)); + + string_count = _cupsStrStatistics(&alloc_bytes, &total_bytes); + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Report: stringpool-string-count=" CUPS_LLFMT, + CUPS_LLCAST string_count); + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Report: stringpool-alloc-bytes=" CUPS_LLFMT, + CUPS_LLCAST alloc_bytes); + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Report: stringpool-total-bytes=" CUPS_LLFMT, + CUPS_LLCAST total_bytes); + + report_time = current_time; + } + + /* + * Handle OS-specific event notification for any events that have + * accumulated. Don't send these more than once a second... + */ + + if (LastEvent && (current_time - event_time) >= 1) + { +#ifdef HAVE_NOTIFY_POST + if (LastEvent & (CUPSD_EVENT_PRINTER_ADDED | + CUPSD_EVENT_PRINTER_DELETED | + CUPSD_EVENT_PRINTER_MODIFIED)) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "notify_post(\"com.apple.printerListChange\")"); + notify_post("com.apple.printerListChange"); + } + + if (LastEvent & CUPSD_EVENT_PRINTER_STATE_CHANGED) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "notify_post(\"com.apple.printerHistoryChange\")"); + notify_post("com.apple.printerHistoryChange"); + } + + if (LastEvent & (CUPSD_EVENT_JOB_STATE_CHANGED | + CUPSD_EVENT_JOB_CONFIG_CHANGED | + CUPSD_EVENT_JOB_PROGRESS)) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "notify_post(\"com.apple.jobChange\")"); + notify_post("com.apple.jobChange"); + } +#endif /* HAVE_NOTIFY_POST */ + + /* + * Reset the accumulated events... + */ + + LastEvent = CUPSD_EVENT_NONE; + event_time = current_time; + } + } + + /* + * Log a message based on what happened... + */ + + if (stop_scheduler) + { + cupsdLogMessage(CUPSD_LOG_INFO, "Scheduler shutting down normally."); + cupsdAddEvent(CUPSD_EVENT_SERVER_STOPPED, NULL, NULL, + "Scheduler shutting down normally."); + } + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Scheduler shutting down due to program error."); + cupsdAddEvent(CUPSD_EVENT_SERVER_STOPPED, NULL, NULL, + "Scheduler shutting down due to program error."); + } + + /* + * Close all network clients... + */ + + DoingShutdown = 1; + + cupsdStopServer(); + +#ifdef HAVE_LAUNCHD + /* + * Update the launchd KeepAlive file as needed... + */ + + if (Launchd) + launchd_checkout(); +#endif /* HAVE_LAUNCHD */ + + /* + * Stop all jobs... + */ + + cupsdFreeAllJobs(); + +#ifdef __APPLE__ + /* + * Stop monitoring system event monitoring... + */ + + if (use_sysman) + cupsdStopSystemMonitor(); +#endif /* __APPLE__ */ + +#ifdef HAVE_GSSAPI + /* + * Free the scheduler's Kerberos context... + */ + +# ifdef __APPLE__ + /* + * If the weak-linked GSSAPI/Kerberos library is not present, don't try + * to use it... + */ + + if (krb5_init_context != NULL) +# endif /* __APPLE__ */ + if (KerberosContext) + krb5_free_context(KerberosContext); +#endif /* HAVE_GSSAPI */ + +#ifdef __sgi + /* + * Remove the fake IRIX lpsched lock file, but only if the existing + * file is not a FIFO which indicates that the real IRIX lpsched is + * running... + */ + + if (!stat("/var/spool/lp/FIFO", &statbuf)) + if (!S_ISFIFO(statbuf.st_mode)) + unlink("/var/spool/lp/SCHEDLOCK"); +#endif /* __sgi */ + + cupsdStopSelect(); + + return (!stop_scheduler); +} + + +/* + * 'cupsdAddString()' - Copy and add a string to an array. + */ + +int /* O - 1 on success, 0 on failure */ +cupsdAddString(cups_array_t **a, /* IO - String array */ + const char *s) /* I - String to copy and add */ +{ + if (!*a) + *a = cupsArrayNew3((cups_array_func_t)strcmp, NULL, + (cups_ahash_func_t)NULL, 0, + (cups_acopy_func_t)_cupsStrAlloc, + (cups_afree_func_t)_cupsStrFree); + + return (cupsArrayAdd(*a, (char *)s)); +} + + +/* + * 'cupsdCheckProcess()' - Tell the main loop to check for dead children. + */ + +void +cupsdCheckProcess(void) +{ + /* + * Flag that we have dead children... + */ + + dead_children = 1; +} + + +/* + * 'cupsdClearString()' - Clear a string. + */ + +void +cupsdClearString(char **s) /* O - String value */ +{ + if (s && *s) + { + _cupsStrFree(*s); + *s = NULL; + } +} + + +/* + * 'cupsdFreeStrings()' - Free an array of strings. + */ + +void +cupsdFreeStrings(cups_array_t **a) /* IO - String array */ +{ + if (*a) + { + cupsArrayDelete(*a); + *a = NULL; + } +} + + +/* + * 'cupsdHoldSignals()' - Hold child and termination signals. + */ + +void +cupsdHoldSignals(void) +{ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + sigset_t newmask; /* New POSIX signal mask */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ + + + holdcount ++; + if (holdcount > 1) + return; + +#ifdef HAVE_SIGSET + sighold(SIGTERM); + sighold(SIGCHLD); +#elif defined(HAVE_SIGACTION) + sigemptyset(&newmask); + sigaddset(&newmask, SIGTERM); + sigaddset(&newmask, SIGCHLD); + sigprocmask(SIG_BLOCK, &newmask, &holdmask); +#endif /* HAVE_SIGSET */ +} + + +/* + * 'cupsdReleaseSignals()' - Release signals for delivery. + */ + +void +cupsdReleaseSignals(void) +{ + holdcount --; + if (holdcount > 0) + return; + +#ifdef HAVE_SIGSET + sigrelse(SIGTERM); + sigrelse(SIGCHLD); +#elif defined(HAVE_SIGACTION) + sigprocmask(SIG_SETMASK, &holdmask, NULL); +#endif /* HAVE_SIGSET */ +} + + +/* + * 'cupsdSetString()' - Set a string value. + */ + +void +cupsdSetString(char **s, /* O - New string */ + const char *v) /* I - String value */ +{ + if (!s || *s == v) + return; + + if (*s) + _cupsStrFree(*s); + + if (v) + *s = _cupsStrAlloc(v); + else + *s = NULL; +} + + +/* + * 'cupsdSetStringf()' - Set a formatted string value. + */ + +void +cupsdSetStringf(char **s, /* O - New string */ + const char *f, /* I - Printf-style format string */ + ...) /* I - Additional args as needed */ +{ + char v[4096]; /* Formatting string value */ + va_list ap; /* Argument pointer */ + char *olds; /* Old string */ + + + if (!s) + return; + + olds = *s; + + if (f) + { + va_start(ap, f); + vsnprintf(v, sizeof(v), f, ap); + va_end(ap); + + *s = _cupsStrAlloc(v); + } + else + *s = NULL; + + if (olds) + _cupsStrFree(olds); +} + + +#ifdef HAVE_LAUNCHD +/* + * 'launchd_checkin()' - Check-in with launchd and collect the listening fds. + */ + +static void +launchd_checkin(void) +{ + size_t i, /* Looping var */ + count; /* Number of listeners */ + launch_data_t ld_msg, /* Launch data message */ + ld_resp, /* Launch data response */ + ld_array, /* Launch data array */ + ld_sockets, /* Launch data sockets dictionary */ + tmp; /* Launch data */ + cupsd_listener_t *lis; /* Listeners array */ + http_addr_t addr; /* Address variable */ + socklen_t addrlen; /* Length of address */ + int fd; /* File descriptor */ + char s[256]; /* String addresss */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG, "launchd_checkin: pid=%d", (int)getpid()); + + /* + * Check-in with launchd... + */ + + ld_msg = launch_data_new_string(LAUNCH_KEY_CHECKIN); + if ((ld_resp = launch_msg(ld_msg)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "launchd_checkin: launch_msg(\"" LAUNCH_KEY_CHECKIN + "\") IPC failure"); + exit(EXIT_FAILURE); + return; /* anti-compiler-warning */ + } + + if (launch_data_get_type(ld_resp) == LAUNCH_DATA_ERRNO) + { + errno = launch_data_get_errno(ld_resp); + cupsdLogMessage(CUPSD_LOG_ERROR, "launchd_checkin: Check-in failed: %s", + strerror(errno)); + exit(EXIT_FAILURE); + return; /* anti-compiler-warning */ + } + + /* + * Get the sockets dictionary... + */ + + if ((ld_sockets = launch_data_dict_lookup(ld_resp, LAUNCH_JOBKEY_SOCKETS)) + == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "launchd_checkin: No sockets found to answer requests on!"); + exit(EXIT_FAILURE); + return; /* anti-compiler-warning */ + } + + /* + * Get the array of listener sockets... + */ + + if ((ld_array = launch_data_dict_lookup(ld_sockets, "Listeners")) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "launchd_checkin: No sockets found to answer requests on!"); + exit(EXIT_FAILURE); + return; /* anti-compiler-warning */ + } + + /* + * Add listening fd(s) to the Listener array... + */ + + if (launch_data_get_type(ld_array) == LAUNCH_DATA_ARRAY) + { + count = launch_data_array_get_count(ld_array); + + for (i = 0; i < count; i ++) + { + /* + * Get the launchd file descriptor and address... + */ + + if ((tmp = launch_data_array_get_index(ld_array, i)) != NULL) + { + fd = launch_data_get_fd(tmp); + addrlen = sizeof(addr); + + if (getsockname(fd, (struct sockaddr *)&addr, &addrlen)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "launchd_checkin: Unable to get local address - %s", + strerror(errno)); + continue; + } + + /* + * Try to match the launchd socket address to one of the listeners... + */ + + for (lis = (cupsd_listener_t *)cupsArrayFirst(Listeners); + lis; + lis = (cupsd_listener_t *)cupsArrayNext(Listeners)) + if (httpAddrEqual(&lis->address, &addr)) + break; + + /* + * Add a new listener If there's no match... + */ + + if (lis) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "launchd_checkin: Matched existing listener %s with fd %d...", + httpAddrString(&(lis->address), s, sizeof(s)), fd); + } + else + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "launchd_checkin: Adding new listener %s with fd %d...", + httpAddrString(&addr, s, sizeof(s)), fd); + + if ((lis = calloc(1, sizeof(cupsd_listener_t))) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "launchd_checkin: Unable to allocate listener - " + "%s.", strerror(errno)); + exit(EXIT_FAILURE); + } + + cupsArrayAdd(Listeners, lis); + + memcpy(&lis->address, &addr, sizeof(lis->address)); + } + + lis->fd = fd; + +# ifdef HAVE_SSL + if (_httpAddrPort(&(lis->address)) == 443) + lis->encryption = HTTP_ENCRYPT_ALWAYS; +# endif /* HAVE_SSL */ + } + } + } + + launch_data_free(ld_msg); + launch_data_free(ld_resp); +} + + +/* + * 'launchd_checkout()' - Update the launchd KeepAlive file as needed. + */ + +static void +launchd_checkout(void) +{ + int fd; /* File descriptor */ + + + /* + * Create or remove the launchd KeepAlive file based on whether + * there are active jobs, polling, browsing for remote printers or + * shared printers to advertise... + */ + + if (cupsArrayCount(ActiveJobs) || + (Browsing && BrowseLocalProtocols && cupsArrayCount(Printers))) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Creating launchd keepalive file \"" CUPS_KEEPALIVE + "\"..."); + + if ((fd = open(CUPS_KEEPALIVE, O_RDONLY | O_CREAT | O_EXCL, S_IRUSR)) >= 0) + close(fd); + } + else + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Removing launchd keepalive file \"" CUPS_KEEPALIVE + "\"..."); + + unlink(CUPS_KEEPALIVE); + } +} +#endif /* HAVE_LAUNCHD */ + + +/* + * 'parent_handler()' - Catch USR1/CHLD signals... + */ + +static void +parent_handler(int sig) /* I - Signal */ +{ + /* + * Store the signal we got from the OS and return... + */ + + parent_signal = sig; +} + + +/* + * 'process_children()' - Process all dead children... + */ + +static void +process_children(void) +{ + int status; /* Exit status of child */ + int pid, /* Process ID of child */ + job_id; /* Job ID of child */ + cupsd_job_t *job; /* Current job */ + int i; /* Looping var */ + char name[1024]; /* Process name */ + const char *type; /* Type of program */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "process_children()"); + + /* + * Reset the dead_children flag... + */ + + dead_children = 0; + + /* + * Collect the exit status of some children... + */ + +#ifdef HAVE_WAITPID + while ((pid = waitpid(-1, &status, WNOHANG)) > 0) +#elif defined(HAVE_WAIT3) + while ((pid = wait3(&status, WNOHANG, NULL)) > 0) +#else + if ((pid = wait(&status)) > 0) +#endif /* HAVE_WAITPID */ + { + /* + * Collect the name of the process that finished... + */ + + cupsdFinishProcess(pid, name, sizeof(name), &job_id); + + /* + * Delete certificates for CGI processes... + */ + + if (pid) + cupsdDeleteCert(pid); + + /* + * Handle completed job filters... + */ + + if (job_id > 0) + job = cupsdFindJob(job_id); + else + job = NULL; + + if (job) + { + for (i = 0; job->filters[i]; i ++) + if (job->filters[i] == pid) + break; + + if (job->filters[i] || job->backend == pid) + { + /* + * OK, this process has gone away; what's left? + */ + + if (job->filters[i]) + { + job->filters[i] = -pid; + type = "Filter"; + } + else + { + job->backend = -pid; + type = "Backend"; + } + + if (status && status != SIGTERM && status != SIGKILL && + status != SIGPIPE && job->status >= 0) + { + /* + * An error occurred; save the exit status so we know to stop + * the printer or cancel the job when all of the filters finish... + * + * A negative status indicates that the backend failed and the + * printer needs to be stopped. + */ + + if (job->filters[i]) + job->status = status; /* Filter failed */ + else + job->status = -status; /* Backend failed */ + + if (job->state_value == IPP_JOB_PROCESSING && + job->status_level > CUPSD_LOG_ERROR && + (job->filters[i] || !WIFEXITED(status))) + { + char message[1024]; /* New printer-state-message */ + + + job->status_level = CUPSD_LOG_ERROR; + + snprintf(message, sizeof(message), "%s failed", type); + + if (job->printer) + { + strlcpy(job->printer->state_message, message, + sizeof(job->printer->state_message)); + } + + if (!job->attrs) + cupsdLoadJob(job); + + if (!job->printer_message && job->attrs) + { + if ((job->printer_message = + ippFindAttribute(job->attrs, "job-printer-state-message", + IPP_TAG_TEXT)) == NULL) + job->printer_message = ippAddString(job->attrs, IPP_TAG_JOB, + IPP_TAG_TEXT, + "job-printer-state-message", + NULL, NULL); + } + + if (job->printer_message) + cupsdSetString(&(job->printer_message->values[0].string.text), + message); + } + } + + /* + * If this is not the last file in a job, see if all of the + * filters are done, and if so move to the next file. + */ + + if (job->current_file < job->num_files && job->printer) + { + for (i = 0; job->filters[i] < 0; i ++); + + if (!job->filters[i] && + (!job->printer->pc || !job->printer->pc->single_file || + job->backend <= 0)) + { + /* + * Process the next file... + */ + + cupsdContinueJob(job); + } + } + else if (job->state_value >= IPP_JOB_CANCELED) + { + /* + * Remove the job from the active list if there are no processes still + * running for it... + */ + + for (i = 0; job->filters[i] < 0; i++); + + if (!job->filters[i] && job->backend <= 0) + cupsArrayRemove(ActiveJobs, job); + } + } + } + + /* + * Show the exit status as needed, ignoring SIGTERM and SIGKILL errors + * since they come when we kill/end a process... + */ + + if (status == SIGTERM || status == SIGKILL) + { + cupsdLogJob(job, CUPSD_LOG_DEBUG, + "PID %d (%s) was terminated normally with signal %d.", pid, + name, status); + } + else if (status == SIGPIPE) + { + cupsdLogJob(job, CUPSD_LOG_DEBUG, + "PID %d (%s) did not catch or ignore signal %d.", pid, name, + status); + } + else if (status) + { + if (WIFEXITED(status)) + { + int code = WEXITSTATUS(status); /* Exit code */ + + if (code > 100) + cupsdLogJob(job, CUPSD_LOG_DEBUG, + "PID %d (%s) stopped with status %d (%s)", pid, name, + code, strerror(code - 100)); + else + cupsdLogJob(job, CUPSD_LOG_DEBUG, + "PID %d (%s) stopped with status %d.", pid, name, code); + } + else + cupsdLogJob(job, CUPSD_LOG_DEBUG, "PID %d (%s) crashed on signal %d.", + pid, name, WTERMSIG(status)); + + if (LogLevel < CUPSD_LOG_DEBUG) + cupsdLogJob(job, CUPSD_LOG_INFO, + "Hint: Try setting the LogLevel to \"debug\" to find out " + "more."); + } + else + cupsdLogJob(job, CUPSD_LOG_DEBUG, "PID %d (%s) exited with no errors.", + pid, name); + } + + /* + * If wait*() is interrupted by a signal, tell main() to call us again... + */ + + if (pid < 0 && errno == EINTR) + dead_children = 1; +} + + +/* + * 'select_timeout()' - Calculate the select timeout value. + * + */ + +static long /* O - Number of seconds */ +select_timeout(int fds) /* I - Number of descriptors returned */ +{ + long timeout; /* Timeout for select */ + time_t now; /* Current time */ + cupsd_client_t *con; /* Client information */ + cupsd_job_t *job; /* Job information */ + cupsd_subscription_t *sub; /* Subscription information */ + const char *why; /* Debugging aid */ + + + /* + * Check to see if any of the clients have pending data to be + * processed; if so, the timeout should be 0... + */ + + for (con = (cupsd_client_t *)cupsArrayFirst(Clients); + con; + con = (cupsd_client_t *)cupsArrayNext(Clients)) + if (con->http.used > 0) + return (0); + + /* + * If select has been active in the last second (fds > 0) or we have + * many resources in use then don't bother trying to optimize the + * timeout, just make it 1 second. + */ + + if (fds > 0 || cupsArrayCount(Clients) > 50) + return (1); + + /* + * Otherwise, check all of the possible events that we need to wake for... + */ + + now = time(NULL); + timeout = now + 86400; /* 86400 == 1 day */ + why = "do nothing"; + +#ifdef __APPLE__ + /* + * When going to sleep, wake up to cancel jobs that don't complete in time. + */ + + if (SleepJobs > 0 && SleepJobs < timeout) + { + timeout = SleepJobs; + why = "cancel jobs before sleeping"; + } +#endif /* __APPLE__ */ + + /* + * Check whether we are accepting new connections... + */ + + if (ListeningPaused > 0 && cupsArrayCount(Clients) < MaxClients && + ListeningPaused < timeout) + { + if (ListeningPaused <= now) + timeout = now; + else + timeout = ListeningPaused; + + why = "resume listening"; + } + + /* + * Check the activity and close old clients... + */ + + for (con = (cupsd_client_t *)cupsArrayFirst(Clients); + con; + con = (cupsd_client_t *)cupsArrayNext(Clients)) + if ((con->http.activity + Timeout) < timeout) + { + timeout = con->http.activity + Timeout; + why = "timeout a client connection"; + } + + /* + * Write out changes to configuration and state files... + */ + + if (DirtyCleanTime && timeout > DirtyCleanTime) + { + timeout = DirtyCleanTime; + why = "write dirty config/state files"; + } + + /* + * Check for any active jobs... + */ + + for (job = (cupsd_job_t *)cupsArrayFirst(ActiveJobs); + job; + job = (cupsd_job_t *)cupsArrayNext(ActiveJobs)) + { + if (job->kill_time && job->kill_time < timeout) + { + timeout = job->kill_time; + why = "kill unresponsive jobs"; + } + else if (job->cancel_time && job->cancel_time < timeout) + { + timeout = job->cancel_time; + why = "cancel stuck jobs"; + } + else if (job->state_value == IPP_JOB_HELD && job->hold_until < timeout) + { + timeout = job->hold_until; + why = "release held jobs"; + } + else if (job->state_value == IPP_JOB_PENDING && timeout > (now + 10)) + { + timeout = now + 10; + why = "start pending jobs"; + break; + } + } + +#ifdef HAVE_MALLINFO + /* + * Log memory usage every minute... + */ + + if (LogLevel >= CUPSD_LOG_DEBUG && (mallinfo_time + 60) < timeout) + { + timeout = mallinfo_time + 60; + why = "display memory usage"; + } +#endif /* HAVE_MALLINFO */ + + /* + * Expire subscriptions as needed... + */ + + for (sub = (cupsd_subscription_t *)cupsArrayFirst(Subscriptions); + sub; + sub = (cupsd_subscription_t *)cupsArrayNext(Subscriptions)) + if (!sub->job && sub->expire && sub->expire < timeout) + { + timeout = sub->expire; + why = "expire subscription"; + } + + /* + * Adjust from absolute to relative time. We add 1 second to the timeout since + * events occur after the timeout expires, and limit the timeout to 86400 + * seconds (1 day) to avoid select() timeout limits present on some operating + * systems... + */ + + timeout = timeout - now + 1; + + if (timeout < 1) + timeout = 1; + else if (timeout > 86400) + timeout = 86400; + + /* + * Log and return the timeout value... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "select_timeout(%d): %ld seconds to %s", + fds, timeout, why); + + return (timeout); +} + + +/* + * 'sigchld_handler()' - Handle 'child' signals from old processes. + */ + +static void +sigchld_handler(int sig) /* I - Signal number */ +{ + (void)sig; + + /* + * Flag that we have dead children... + */ + + dead_children = 1; + + /* + * Reset the signal handler as needed... + */ + +#if !defined(HAVE_SIGSET) && !defined(HAVE_SIGACTION) + signal(SIGCLD, sigchld_handler); +#endif /* !HAVE_SIGSET && !HAVE_SIGACTION */ +} + + +/* + * 'sighup_handler()' - Handle 'hangup' signals to reconfigure the scheduler. + */ + +static void +sighup_handler(int sig) /* I - Signal number */ +{ + (void)sig; + + NeedReload = RELOAD_ALL; + ReloadTime = time(NULL); + +#if !defined(HAVE_SIGSET) && !defined(HAVE_SIGACTION) + signal(SIGHUP, sighup_handler); +#endif /* !HAVE_SIGSET && !HAVE_SIGACTION */ +} + + +/* + * 'sigterm_handler()' - Handle 'terminate' signals that stop the scheduler. + */ + +static void +sigterm_handler(int sig) /* I - Signal number */ +{ + (void)sig; /* remove compiler warnings... */ + + /* + * Flag that we should stop and return... + */ + + stop_scheduler = 1; +} + + +/* + * 'usage()' - Show scheduler usage. + */ + +static void +usage(int status) /* O - Exit status */ +{ + FILE *fp = status ? stderr : stdout; /* Output file */ + + + _cupsLangPuts(fp, _("Usage: cupsd [options]")); + _cupsLangPuts(fp, _("Options:")); + _cupsLangPuts(fp, _(" -c config-file Load alternate configuration " + "file.")); + _cupsLangPuts(fp, _(" -f Run in the foreground.")); + _cupsLangPuts(fp, _(" -F Run in the foreground but " + "detach from console.")); + _cupsLangPuts(fp, _(" -h Show this usage message.")); + _cupsLangPuts(fp, _(" -l Run cupsd from launchd(8).")); + _cupsLangPuts(fp, _(" -t Test the configuration " + "file.")); + + exit(status); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/mime-private.h b/scheduler/mime-private.h new file mode 100644 index 0000000000..c448e5ac8a --- /dev/null +++ b/scheduler/mime-private.h @@ -0,0 +1,45 @@ +/* + * "$Id$" + * + * Private MIME type/conversion database definitions for CUPS. + * + * Copyright 2011 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#ifndef _CUPS_MIME_PRIVATE_H_ +# define _CUPS_MIME_PRIVATE_H_ + +# include "mime.h" + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Prototypes... + */ + +extern void _mimeError(mime_t *mime, const char *format, ...) + __attribute__ ((__format__ (__printf__, 2, 3))); + + +# ifdef __cplusplus +} +# endif /* __cplusplus */ +#endif /* !_CUPS_MIME_PRIVATE_H_ */ + +/* + * End of "$Id$". + */ diff --git a/scheduler/mime.c b/scheduler/mime.c new file mode 100644 index 0000000000..abad962eb2 --- /dev/null +++ b/scheduler/mime.c @@ -0,0 +1,960 @@ +/* + * "$Id$" + * + * MIME database file routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * mimeDelete() - Delete (free) a MIME database. + * mimeDeleteFilter() - Delete a filter from the MIME database. + * mimeDeleteType() - Delete a type from the MIME database. + * _mimeError() - Show an error message. + * mimeFirstFilter() - Get the first filter in the MIME database. + * mimeFirstType() - Get the first type in the MIME database. + * mimeLoad() - Create a new MIME database from disk. + * mimeLoadFilters() - Load filter definitions from disk. + * mimeLoadTypes() - Load type definitions from disk. + * mimeNew() - Create a new, empty MIME database. + * mimeNextFilter() - Get the next filter in the MIME database. + * mimeNextType() - Get the next type in the MIME database. + * mimeNumFilters() - Get the number of filters in a MIME database. + * mimeNumTypes() - Get the number of types in a MIME database. + * mimeSetErrorCallback() - Set the callback for error messages. + * mime_add_fcache() - Add a filter to the filter cache. + * mime_compare_fcache() - Compare two filter cache entries. + * mime_delete_fcache() - Free all memory used by the filter cache. + * mime_delete_rules() - Free all memory for the given rule tree. + * mime_load_convs() - Load a xyz.convs file. + * mime_load_types() - Load a xyz.types file. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include "mime-private.h" + + +/* + * Local types... + */ + +typedef struct _mime_fcache_s /**** Filter cache structure ****/ +{ + char *name, /* Filter name */ + *path; /* Full path to filter if available */ +} _mime_fcache_t; + + +/* + * Local functions... + */ + +static const char *mime_add_fcache(cups_array_t *filtercache, const char *name, + const char *filterpath); +static int mime_compare_fcache(_mime_fcache_t *a, _mime_fcache_t *b); +static void mime_delete_fcache(cups_array_t *filtercache); +static void mime_delete_rules(mime_magic_t *rules); +static void mime_load_convs(mime_t *mime, const char *filename, + const char *filterpath, + cups_array_t *filtercache); +static void mime_load_types(mime_t *mime, const char *filename); + + +/* + * 'mimeDelete()' - Delete (free) a MIME database. + */ + +void +mimeDelete(mime_t *mime) /* I - MIME database */ +{ + mime_type_t *type; /* Current type */ + mime_filter_t *filter; /* Current filter */ + + + DEBUG_printf(("mimeDelete(mime=%p)", mime)); + + if (!mime) + return; + + /* + * Loop through filters and free them... + */ + + for (filter = (mime_filter_t *)cupsArrayFirst(mime->filters); + filter; + filter = (mime_filter_t *)cupsArrayNext(mime->filters)) + mimeDeleteFilter(mime, filter); + + /* + * Loop through the file types and delete any rules... + */ + + for (type = (mime_type_t *)cupsArrayFirst(mime->types); + type; + type = (mime_type_t *)cupsArrayNext(mime->types)) + mimeDeleteType(mime, type); + + /* + * Free the types and filters arrays, and then the MIME database structure. + */ + + cupsArrayDelete(mime->types); + cupsArrayDelete(mime->filters); + cupsArrayDelete(mime->srcs); + free(mime); +} + + +/* + * 'mimeDeleteFilter()' - Delete a filter from the MIME database. + */ + +void +mimeDeleteFilter(mime_t *mime, /* I - MIME database */ + mime_filter_t *filter) /* I - Filter */ +{ + DEBUG_printf(("mimeDeleteFilter(mime=%p, filter=%p(%s/%s->%s/%s, cost=%d, " + "maxsize=" CUPS_LLFMT "))", mime, filter, + filter ? filter->src->super : "???", + filter ? filter->src->type : "???", + filter ? filter->dst->super : "???", + filter ? filter->dst->super : "???", + filter ? filter->cost : -1, + filter ? CUPS_LLCAST filter->maxsize : CUPS_LLCAST -1)); + + if (!mime || !filter) + return; + +#ifdef DEBUG + if (!cupsArrayFind(mime->filters, filter)) + DEBUG_puts("1mimeDeleteFilter: Filter not in MIME database."); +#endif /* DEBUG */ + + cupsArrayRemove(mime->filters, filter); + free(filter); + + /* + * Deleting a filter invalidates the source lookup cache used by + * mimeFilter()... + */ + + if (mime->srcs) + { + DEBUG_puts("1mimeDeleteFilter: Deleting source lookup cache."); + cupsArrayDelete(mime->srcs); + mime->srcs = NULL; + } +} + + +/* + * 'mimeDeleteType()' - Delete a type from the MIME database. + */ + +void +mimeDeleteType(mime_t *mime, /* I - MIME database */ + mime_type_t *mt) /* I - Type */ +{ + DEBUG_printf(("mimeDeleteType(mime=%p, mt=%p(%s/%s))", mime, mt, + mt ? mt->super : "???", mt ? mt->type : "???")); + + if (!mime || !mt) + return; + +#ifdef DEBUG + if (!cupsArrayFind(mime->types, mt)) + DEBUG_puts("1mimeDeleteFilter: Type not in MIME database."); +#endif /* DEBUG */ + + cupsArrayRemove(mime->types, mt); + + mime_delete_rules(mt->rules); + free(mt); +} + + +/* + * '_mimeError()' - Show an error message. + */ + +void +_mimeError(mime_t *mime, /* I - MIME database */ + const char *message, /* I - Printf-style message string */ + ...) /* I - Additional arguments as needed */ +{ + va_list ap; /* Argument pointer */ + char buffer[8192]; /* Message buffer */ + + + if (mime->error_cb) + { + va_start(ap, message); + vsnprintf(buffer, sizeof(buffer), message, ap); + va_end(ap); + + (*mime->error_cb)(mime->error_ctx, buffer); + } +} + + +/* + * 'mimeFirstFilter()' - Get the first filter in the MIME database. + */ + +mime_filter_t * /* O - Filter or NULL */ +mimeFirstFilter(mime_t *mime) /* I - MIME database */ +{ + DEBUG_printf(("6mimeFirstFilter(mime=%p)", mime)); + + if (!mime) + { + DEBUG_puts("7mimeFirstFilter: Returning NULL."); + return (NULL); + } + else + { + mime_filter_t *first = (mime_filter_t *)cupsArrayFirst(mime->filters); + /* First filter */ + + DEBUG_printf(("7mimeFirstFilter: Returning %p.", first)); + return (first); + } +} + + +/* + * 'mimeFirstType()' - Get the first type in the MIME database. + */ + +mime_type_t * /* O - Type or NULL */ +mimeFirstType(mime_t *mime) /* I - MIME database */ +{ + DEBUG_printf(("6mimeFirstType(mime=%p)", mime)); + + if (!mime) + { + DEBUG_puts("7mimeFirstType: Returning NULL."); + return (NULL); + } + else + { + mime_type_t *first = (mime_type_t *)cupsArrayFirst(mime->types); + /* First type */ + + DEBUG_printf(("7mimeFirstType: Returning %p.", first)); + return (first); + } +} + + +/* + * 'mimeLoad()' - Create a new MIME database from disk. + * + * This function uses @link mimeLoadFilters@ and @link mimeLoadTypes@ to + * create a MIME database from a single directory. + */ + +mime_t * /* O - New MIME database */ +mimeLoad(const char *pathname, /* I - Directory to load */ + const char *filterpath) /* I - Directory to load */ +{ + mime_t *mime; /* New MIME database */ + + DEBUG_printf(("mimeLoad(pathname=\"%s\", filterpath=\"%s\")", pathname, + filterpath)); + + mime = mimeLoadFilters(mimeLoadTypes(NULL, pathname), pathname, filterpath); + DEBUG_printf(("1mimeLoad: Returning %p.", mime)); + + return (mime); +} + + +/* + * 'mimeLoadFilters()' - Load filter definitions from disk. + * + * This function loads all of the .convs files from the specified directory. + * Use @link mimeLoadTypes@ to load all types before you load the filters. + */ + +mime_t * /* O - MIME database */ +mimeLoadFilters(mime_t *mime, /* I - MIME database */ + const char *pathname, /* I - Directory to load from */ + const char *filterpath) /* I - Default filter program directory */ +{ + cups_dir_t *dir; /* Directory */ + cups_dentry_t *dent; /* Directory entry */ + char filename[1024]; /* Full filename of .convs file */ + cups_array_t *filtercache; /* Filter cache */ + + + DEBUG_printf(("mimeLoadFilters(mime=%p, pathname=\"%s\", filterpath=\"%s\")", + mime, pathname, filterpath)); + + /* + * Range check input... + */ + + if (!mime || !pathname || !filterpath) + { + DEBUG_puts("1mimeLoadFilters: Bad arguments."); + return (mime); + } + + /* + * Then open the directory specified by pathname... + */ + + if ((dir = cupsDirOpen(pathname)) == NULL) + { + DEBUG_printf(("1mimeLoadFilters: Unable to open \"%s\": %s", pathname, + strerror(errno))); + _mimeError(mime, "Unable to open \"%s\": %s", pathname, strerror(errno)); + return (mime); + } + + /* + * Read all the .convs files... + */ + + filtercache = cupsArrayNew((cups_array_func_t)mime_compare_fcache, NULL); + + while ((dent = cupsDirRead(dir)) != NULL) + { + if (strlen(dent->filename) > 6 && + !strcmp(dent->filename + strlen(dent->filename) - 6, ".convs")) + { + /* + * Load a mime.convs file... + */ + + snprintf(filename, sizeof(filename), "%s/%s", pathname, dent->filename); + DEBUG_printf(("1mimeLoadFilters: Loading \"%s\".", filename)); + mime_load_convs(mime, filename, filterpath, filtercache); + } + } + + mime_delete_fcache(filtercache); + + cupsDirClose(dir); + + return (mime); +} + + +/* + * 'mimeLoadTypes()' - Load type definitions from disk. + * + * This function loads all of the .types files from the specified directory. + * Use @link mimeLoadFilters@ to load all filters after you load the types. + */ + +mime_t * /* O - MIME database */ +mimeLoadTypes(mime_t *mime, /* I - MIME database or @code NULL@ to create a new one */ + const char *pathname) /* I - Directory to load from */ +{ + cups_dir_t *dir; /* Directory */ + cups_dentry_t *dent; /* Directory entry */ + char filename[1024]; /* Full filename of .types file */ + + + DEBUG_printf(("mimeLoadTypes(mime=%p, pathname=\"%s\")", mime, pathname)); + + /* + * First open the directory specified by pathname... + */ + + if ((dir = cupsDirOpen(pathname)) == NULL) + { + DEBUG_printf(("1mimeLoadTypes: Unable to open \"%s\": %s", pathname, + strerror(errno))); + DEBUG_printf(("1mimeLoadTypes: Returning %p.", mime)); + _mimeError(mime, "Unable to open \"%s\": %s", pathname, strerror(errno)); + return (mime); + } + + /* + * If "mime" is NULL, make a new, empty database... + */ + + if (!mime) + mime = mimeNew(); + + if (!mime) + { + cupsDirClose(dir); + DEBUG_puts("1mimeLoadTypes: Returning NULL."); + return (NULL); + } + + /* + * Read all the .types files... + */ + + while ((dent = cupsDirRead(dir)) != NULL) + { + if (strlen(dent->filename) > 6 && + !strcmp(dent->filename + strlen(dent->filename) - 6, ".types")) + { + /* + * Load a mime.types file... + */ + + snprintf(filename, sizeof(filename), "%s/%s", pathname, dent->filename); + DEBUG_printf(("1mimeLoadTypes: Loading \"%s\".", filename)); + mime_load_types(mime, filename); + } + } + + cupsDirClose(dir); + + DEBUG_printf(("1mimeLoadTypes: Returning %p.", mime)); + + return (mime); +} + + +/* + * 'mimeNew()' - Create a new, empty MIME database. + */ + +mime_t * /* O - MIME database */ +mimeNew(void) +{ + return ((mime_t *)calloc(1, sizeof(mime_t))); +} + + +/* + * 'mimeNextFilter()' - Get the next filter in the MIME database. + */ + +mime_filter_t * /* O - Filter or NULL */ +mimeNextFilter(mime_t *mime) /* I - MIME database */ +{ + DEBUG_printf(("6mimeNextFilter(mime=%p)", mime)); + + if (!mime) + { + DEBUG_puts("7mimeNextFilter: Returning NULL."); + return (NULL); + } + else + { + mime_filter_t *next = (mime_filter_t *)cupsArrayNext(mime->filters); + /* Next filter */ + + DEBUG_printf(("7mimeNextFilter: Returning %p.", next)); + return (next); + } +} + + +/* + * 'mimeNextType()' - Get the next type in the MIME database. + */ + +mime_type_t * /* O - Type or NULL */ +mimeNextType(mime_t *mime) /* I - MIME database */ +{ + DEBUG_printf(("6mimeNextType(mime=%p)", mime)); + + if (!mime) + { + DEBUG_puts("7mimeNextType: Returning NULL."); + return (NULL); + } + else + { + mime_type_t *next = (mime_type_t *)cupsArrayNext(mime->types); + /* Next type */ + + DEBUG_printf(("7mimeNextType: Returning %p.", next)); + return (next); + } +} + + +/* + * 'mimeNumFilters()' - Get the number of filters in a MIME database. + */ + +int +mimeNumFilters(mime_t *mime) /* I - MIME database */ +{ + DEBUG_printf(("mimeNumFilters(mime=%p)", mime)); + + if (!mime) + { + DEBUG_puts("1mimeNumFilters: Returning 0."); + return (0); + } + else + { + DEBUG_printf(("1mimeNumFilters: Returning %d.", + cupsArrayCount(mime->filters))); + return (cupsArrayCount(mime->filters)); + } +} + + +/* + * 'mimeNumTypes()' - Get the number of types in a MIME database. + */ + +int +mimeNumTypes(mime_t *mime) /* I - MIME database */ +{ + DEBUG_printf(("mimeNumTypes(mime=%p)", mime)); + + if (!mime) + { + DEBUG_puts("1mimeNumTypes: Returning 0."); + return (0); + } + else + { + DEBUG_printf(("1mimeNumTypes: Returning %d.", + cupsArrayCount(mime->types))); + return (cupsArrayCount(mime->types)); + } +} + + +/* + * 'mimeSetErrorCallback()' - Set the callback for error messages. + */ + +void +mimeSetErrorCallback( + mime_t *mime, /* I - MIME database */ + mime_error_cb_t cb, /* I - Callback function */ + void *ctx) /* I - Context pointer for callback */ +{ + if (mime) + { + mime->error_cb = cb; + mime->error_ctx = ctx; + } +} + + +/* + * 'mime_add_fcache()' - Add a filter to the filter cache. + */ + +static const char * /* O - Full path to filter or NULL */ +mime_add_fcache( + cups_array_t *filtercache, /* I - Filter cache */ + const char *name, /* I - Filter name */ + const char *filterpath) /* I - Filter path */ +{ + _mime_fcache_t key, /* Search key */ + *temp; /* New filter cache */ + char path[1024]; /* Full path to filter */ + + + DEBUG_printf(("2mime_add_fcache(filtercache=%p, name=\"%s\", " + "filterpath=\"%s\")", filtercache, name, filterpath)); + + key.name = (char *)name; + if ((temp = (_mime_fcache_t *)cupsArrayFind(filtercache, &key)) != NULL) + { + DEBUG_printf(("3mime_add_fcache: Returning \"%s\".", temp->path)); + return (temp->path); + } + + if ((temp = calloc(1, sizeof(_mime_fcache_t))) == NULL) + { + DEBUG_puts("3mime_add_fcache: Returning NULL."); + return (NULL); + } + + temp->name = strdup(name); + + if (cupsFileFind(name, filterpath, 1, path, sizeof(path))) + temp->path = strdup(path); + + cupsArrayAdd(filtercache, temp); + + DEBUG_printf(("3mime_add_fcache: Returning \"%s\".", temp->path)); + return (temp->path); +} + + +/* + * 'mime_compare_fcache()' - Compare two filter cache entries. + */ + +static int /* O - Result of comparison */ +mime_compare_fcache(_mime_fcache_t *a, /* I - First entry */ + _mime_fcache_t *b) /* I - Second entry */ +{ + return (strcmp(a->name, b->name)); +} + + +/* + * 'mime_delete_fcache()' - Free all memory used by the filter cache. + */ + +static void +mime_delete_fcache( + cups_array_t *filtercache) /* I - Filter cache */ +{ + _mime_fcache_t *current; /* Current cache entry */ + + + DEBUG_printf(("2mime_delete_fcache(filtercache=%p)", filtercache)); + + for (current = (_mime_fcache_t *)cupsArrayFirst(filtercache); + current; + current = (_mime_fcache_t *)cupsArrayNext(filtercache)) + { + free(current->name); + + if (current->path) + free(current->path); + + free(current); + } + + cupsArrayDelete(filtercache); +} + + +/* + * 'mime_delete_rules()' - Free all memory for the given rule tree. + */ + +static void +mime_delete_rules(mime_magic_t *rules) /* I - Rules to free */ +{ + mime_magic_t *next; /* Next rule to free */ + + + DEBUG_printf(("2mime_delete_rules(rules=%p)", rules)); + + /* + * Free the rules list, descending recursively to free any child rules. + */ + + while (rules != NULL) + { + next = rules->next; + + if (rules->child != NULL) + mime_delete_rules(rules->child); + + free(rules); + rules = next; + } +} + + +/* + * 'mime_load_convs()' - Load a xyz.convs file. + */ + +static void +mime_load_convs( + mime_t *mime, /* I - MIME database */ + const char *filename, /* I - Convs file to load */ + const char *filterpath, /* I - Path for filters */ + cups_array_t *filtercache) /* I - Filter program cache */ +{ + cups_file_t *fp; /* Convs file */ + char line[1024], /* Input line from file */ + *lineptr, /* Current position in line */ + super[MIME_MAX_SUPER], /* Super-type name */ + type[MIME_MAX_TYPE], /* Type name */ + *temp, /* Temporary pointer */ + *filter; /* Filter program */ + mime_type_t *temptype, /* MIME type looping var */ + *dsttype; /* Destination MIME type */ + int cost; /* Cost of filter */ + + + DEBUG_printf(("2mime_load_convs(mime=%p, filename=\"%s\", filterpath=\"%s\", " + "filtercache=%p)", mime, filename, filterpath, filtercache)); + + /* + * First try to open the file... + */ + + if ((fp = cupsFileOpen(filename, "r")) == NULL) + { + DEBUG_printf(("3mime_load_convs: Unable to open \"%s\": %s", filename, + strerror(errno))); + _mimeError(mime, "Unable to open \"%s\": %s", filename, strerror(errno)); + return; + } + + /* + * Then read each line from the file, skipping any comments in the file... + */ + + while (cupsFileGets(fp, line, sizeof(line)) != NULL) + { + /* + * Skip blank lines and lines starting with a #... + */ + + if (!line[0] || line[0] == '#') + continue; + + /* + * Strip trailing whitespace... + */ + + for (lineptr = line + strlen(line) - 1; + lineptr >= line && isspace(*lineptr & 255); + lineptr --) + *lineptr = '\0'; + + /* + * Extract the destination super-type and type names from the middle of + * the line. + */ + + lineptr = line; + while (*lineptr != ' ' && *lineptr != '\t' && *lineptr != '\0') + lineptr ++; + + while (*lineptr == ' ' || *lineptr == '\t') + lineptr ++; + + temp = super; + + while (*lineptr != '/' && *lineptr != '\n' && *lineptr != '\0' && + (temp - super + 1) < MIME_MAX_SUPER) + *temp++ = tolower(*lineptr++ & 255); + + *temp = '\0'; + + if (*lineptr != '/') + continue; + + lineptr ++; + temp = type; + + while (*lineptr != ' ' && *lineptr != '\t' && *lineptr != '\n' && + *lineptr != '\0' && (temp - type + 1) < MIME_MAX_TYPE) + *temp++ = tolower(*lineptr++ & 255); + + *temp = '\0'; + + if (*lineptr == '\0' || *lineptr == '\n') + continue; + + if ((dsttype = mimeType(mime, super, type)) == NULL) + { + DEBUG_printf(("3mime_load_convs: Destination type %s/%s not found.", + super, type)); + continue; + } + + /* + * Then get the cost and filter program... + */ + + while (*lineptr == ' ' || *lineptr == '\t') + lineptr ++; + + if (*lineptr < '0' || *lineptr > '9') + continue; + + cost = atoi(lineptr); + + while (*lineptr != ' ' && *lineptr != '\t' && *lineptr != '\0') + lineptr ++; + while (*lineptr == ' ' || *lineptr == '\t') + lineptr ++; + + if (*lineptr == '\0' || *lineptr == '\n') + continue; + + filter = lineptr; + + if (strcmp(filter, "-")) + { + /* + * Verify that the filter exists and is executable... + */ + + if (!mime_add_fcache(filtercache, filter, filterpath)) + { + DEBUG_printf(("mime_load_convs: Filter %s not found in %s.", filter, + filterpath)); + _mimeError(mime, "Filter \"%s\" not found.", filter); + continue; + } + } + + /* + * Finally, get the source super-type and type names from the beginning of + * the line. We do it here so we can support wildcards... + */ + + lineptr = line; + temp = super; + + while (*lineptr != '/' && *lineptr != '\n' && *lineptr != '\0' && + (temp - super + 1) < MIME_MAX_SUPER) + *temp++ = tolower(*lineptr++ & 255); + + *temp = '\0'; + + if (*lineptr != '/') + continue; + + lineptr ++; + temp = type; + + while (*lineptr != ' ' && *lineptr != '\t' && *lineptr != '\n' && + *lineptr != '\0' && (temp - type + 1) < MIME_MAX_TYPE) + *temp++ = tolower(*lineptr++ & 255); + + *temp = '\0'; + + if (!strcmp(super, "*") && !strcmp(type, "*")) + { + /* + * Force * / * to be "application/octet-stream"... + */ + + strcpy(super, "application"); + strcpy(type, "octet-stream"); + } + + /* + * Add the filter to the MIME database, supporting wildcards as needed... + */ + + for (temptype = (mime_type_t *)cupsArrayFirst(mime->types); + temptype; + temptype = (mime_type_t *)cupsArrayNext(mime->types)) + if ((super[0] == '*' || !strcmp(temptype->super, super)) && + (type[0] == '*' || !strcmp(temptype->type, type))) + mimeAddFilter(mime, temptype, dsttype, cost, filter); + } + + cupsFileClose(fp); +} + + +/* + * 'mime_load_types()' - Load a xyz.types file. + */ + +static void +mime_load_types(mime_t *mime, /* I - MIME database */ + const char *filename) /* I - Types file to load */ +{ + cups_file_t *fp; /* Types file */ + int linelen; /* Length of line */ + char line[32768], /* Input line from file */ + *lineptr, /* Current position in line */ + super[MIME_MAX_SUPER], /* Super-type name */ + type[MIME_MAX_TYPE], /* Type name */ + *temp; /* Temporary pointer */ + mime_type_t *typeptr; /* New MIME type */ + + + DEBUG_printf(("2mime_load_types(mime=%p, filename=\"%s\")", mime, filename)); + + /* + * First try to open the file... + */ + + if ((fp = cupsFileOpen(filename, "r")) == NULL) + { + DEBUG_printf(("3mime_load_types: Unable to open \"%s\": %s", filename, + strerror(errno))); + _mimeError(mime, "Unable to open \"%s\": %s", filename, strerror(errno)); + return; + } + + /* + * Then read each line from the file, skipping any comments in the file... + */ + + while (cupsFileGets(fp, line, sizeof(line)) != NULL) + { + /* + * Skip blank lines and lines starting with a #... + */ + + if (!line[0] || line[0] == '#') + continue; + + /* + * While the last character in the line is a backslash, continue on to the + * next line (and the next, etc.) + */ + + linelen = strlen(line); + + while (line[linelen - 1] == '\\') + { + linelen --; + + if (cupsFileGets(fp, line + linelen, sizeof(line) - linelen) == NULL) + line[linelen] = '\0'; + else + linelen += strlen(line + linelen); + } + + /* + * Extract the super-type and type names from the beginning of the line. + */ + + lineptr = line; + temp = super; + + while (*lineptr != '/' && *lineptr != '\n' && *lineptr != '\0' && + (temp - super + 1) < MIME_MAX_SUPER) + *temp++ = tolower(*lineptr++ & 255); + + *temp = '\0'; + + if (*lineptr != '/') + continue; + + lineptr ++; + temp = type; + + while (*lineptr != ' ' && *lineptr != '\t' && *lineptr != '\n' && + *lineptr != '\0' && (temp - type + 1) < MIME_MAX_TYPE) + *temp++ = tolower(*lineptr++ & 255); + + *temp = '\0'; + + /* + * Add the type and rules to the MIME database... + */ + + typeptr = mimeAddType(mime, super, type); + mimeAddTypeRule(typeptr, lineptr); + } + + cupsFileClose(fp); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/mime.h b/scheduler/mime.h new file mode 100644 index 0000000000..6feceecc4a --- /dev/null +++ b/scheduler/mime.h @@ -0,0 +1,162 @@ +/* + * "$Id$" + * + * MIME type/conversion database definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#ifndef _CUPS_MIME_H_ +# define _CUPS_MIME_H_ + +# include +# include +# include + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Constants... + */ + +# define MIME_MAX_SUPER 16 /* Maximum size of supertype name */ +# define MIME_MAX_TYPE IPP_MAX_NAME /* Maximum size of type name */ +# define MIME_MAX_FILTER 256 /* Maximum size of filter pathname */ +# define MIME_MAX_BUFFER 4096 /* Maximum size of file buffer */ + + +/* + * Types/structures... + */ + +typedef enum +{ + MIME_MAGIC_NOP, /* No operation */ + MIME_MAGIC_AND, /* Logical AND of all children */ + MIME_MAGIC_OR, /* Logical OR of all children */ + MIME_MAGIC_MATCH, /* Filename match */ + MIME_MAGIC_ASCII, /* ASCII characters in range */ + MIME_MAGIC_PRINTABLE, /* Printable characters (32-255) in range */ + MIME_MAGIC_STRING, /* String matches */ + MIME_MAGIC_CHAR, /* Character/byte matches */ + MIME_MAGIC_SHORT, /* Short/16-bit word matches */ + MIME_MAGIC_INT, /* Integer/32-bit word matches */ + MIME_MAGIC_LOCALE, /* Current locale matches string */ + MIME_MAGIC_CONTAINS, /* File contains a string */ + MIME_MAGIC_ISTRING /* Case-insensitive string matches */ +} mime_op_t; + +typedef struct _mime_magic_s /**** MIME Magic Data ****/ +{ + struct _mime_magic_s *prev, /* Previous rule */ + *next, /* Next rule */ + *parent, /* Parent rules */ + *child; /* Child rules */ + short op, /* Operation code (see above) */ + invert; /* Invert the result */ + int offset, /* Offset in file */ + region, /* Region length */ + length; /* Length of data */ + union + { + char matchv[64]; /* Match value */ + char localev[64]; /* Locale value */ + char stringv[64]; /* String value */ + unsigned char charv; /* Byte value */ + unsigned short shortv; /* Short value */ + unsigned intv; /* Integer value */ + } value; +} mime_magic_t; + +typedef struct _mime_type_s /**** MIME Type Data ****/ +{ + mime_magic_t *rules; /* Rules used to detect this type */ + int priority; /* Priority of this type */ + char super[MIME_MAX_SUPER], /* Super-type name ("image", "application", etc.) */ + type[MIME_MAX_TYPE]; /* Type name ("png", "postscript", etc.) */ +} mime_type_t; + +typedef struct _mime_filter_s /**** MIME Conversion Filter Data ****/ +{ + mime_type_t *src, /* Source type */ + *dst; /* Destination type */ + int cost; /* Relative cost */ + char filter[MIME_MAX_FILTER];/* Filter program to use */ + size_t maxsize; /* Maximum file size for this filter */ +} mime_filter_t; + +typedef void (*mime_error_cb_t)(void *ctx, const char *message); + +typedef struct _mime_s /**** MIME Database ****/ +{ + cups_array_t *types; /* File types */ + cups_array_t *filters; /* Type conversion filters */ + cups_array_t *srcs; /* Filters sorted by source type */ + mime_error_cb_t error_cb; /* Error message callback */ + void *error_ctx; /* Pointer for callback */ +} mime_t; + + +/* + * Functions... + */ + +extern void mimeDelete(mime_t *mime); +extern mime_t *mimeNew(void) _CUPS_API_1_5; +extern mime_t *mimeLoad(const char *pathname, const char *filterpath); +extern mime_t *mimeLoadFilters(mime_t *mime, const char *pathname, + const char *filterpath); +extern mime_t *mimeLoadTypes(mime_t *mime, const char *pathname); + +extern mime_type_t *mimeAddType(mime_t *mime, const char *super, + const char *type); +extern int mimeAddTypeRule(mime_type_t *mt, const char *rule); +extern void mimeDeleteType(mime_t *mime, mime_type_t *mt); +extern mime_type_t *mimeFileType(mime_t *mime, const char *pathname, + const char *filename, int *compression); +extern mime_type_t *mimeFirstType(mime_t *mime); +extern mime_type_t *mimeNextType(mime_t *mime); +extern int mimeNumTypes(mime_t *mime); +extern mime_type_t *mimeType(mime_t *mime, const char *super, + const char *type); + +extern mime_filter_t *mimeAddFilter(mime_t *mime, mime_type_t *src, + mime_type_t *dst, int cost, + const char *filter); +extern void mimeDeleteFilter(mime_t *mime, mime_filter_t *filter); +extern cups_array_t *mimeFilter(mime_t *mime, mime_type_t *src, + mime_type_t *dst, int *cost); +extern cups_array_t *mimeFilter2(mime_t *mime, mime_type_t *src, + size_t srcsize, mime_type_t *dst, + int *cost); +extern mime_filter_t *mimeFilterLookup(mime_t *mime, mime_type_t *src, + mime_type_t *dst); +extern mime_filter_t *mimeFirstFilter(mime_t *mime); +extern mime_filter_t *mimeNextFilter(mime_t *mime); +extern int mimeNumFilters(mime_t *mime); +extern void mimeSetErrorCallback(mime_t *mime, mime_error_cb_t cb, + void *context) _CUPS_API_1_5; + +# ifdef __cplusplus +} +# endif /* __cplusplus */ +#endif /* !_CUPS_MIME_H_ */ + +/* + * End of "$Id$". + */ diff --git a/scheduler/network.c b/scheduler/network.c new file mode 100644 index 0000000000..29b42ea04d --- /dev/null +++ b/scheduler/network.c @@ -0,0 +1,300 @@ +/* + * "$Id$" + * + * Network interface functions for the CUPS scheduler. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdNetIFFind() - Find a network interface. + * cupsdNetIFFree() - Free the current network interface list. + * cupsdNetIFUpdate() - Update the network interface list as needed... + * compare_netif() - Compare two network interfaces. + */ + +/* + * Include necessary headers. + */ + +#include +#include "cupsd.h" + + +/* + * Local functions... + */ + +static void cupsdNetIFFree(void); +static int compare_netif(cupsd_netif_t *a, cupsd_netif_t *b); + + +/* + * 'cupsdNetIFFind()' - Find a network interface. + */ + +cupsd_netif_t * /* O - Network interface data */ +cupsdNetIFFind(const char *name) /* I - Name of interface */ +{ + cupsd_netif_t key; /* Search key */ + + + /* + * Update the interface list as needed... + */ + + if (NetIFUpdate) + cupsdNetIFUpdate(); + + /* + * Search for the named interface... + */ + + strlcpy(key.name, name, sizeof(key.name)); + + return ((cupsd_netif_t *)cupsArrayFind(NetIFList, &key)); +} + + +/* + * 'cupsdNetIFFree()' - Free the current network interface list. + */ + +static void +cupsdNetIFFree(void) +{ + cupsd_netif_t *current; /* Current interface in array */ + + + /* + * Loop through the interface list and free all the records... + */ + + for (current = (cupsd_netif_t *)cupsArrayFirst(NetIFList); + current; + current = (cupsd_netif_t *)cupsArrayNext(NetIFList)) + { + cupsArrayRemove(NetIFList, current); + free(current); + } +} + + +/* + * 'cupsdNetIFUpdate()' - Update the network interface list as needed... + */ + +void +cupsdNetIFUpdate(void) +{ + int match; /* Matching address? */ + cupsd_listener_t *lis; /* Listen address */ + cupsd_netif_t *temp; /* New interface */ + struct ifaddrs *addrs, /* Interface address list */ + *addr; /* Current interface address */ + char hostname[1024]; /* Hostname for address */ + size_t hostlen; /* Length of hostname */ + + + /* + * Only update the list if we need to... + */ + + if (!NetIFUpdate) + return; + + NetIFUpdate = 0; + + /* + * Free the old interfaces... + */ + + cupsdNetIFFree(); + + /* + * Make sure we have an array... + */ + + if (!NetIFList) + NetIFList = cupsArrayNew((cups_array_func_t)compare_netif, NULL); + + if (!NetIFList) + return; + + /* + * Grab a new list of interfaces... + */ + + if (getifaddrs(&addrs) < 0) + return; + + for (addr = addrs; addr != NULL; addr = addr->ifa_next) + { + /* + * See if this interface address is IPv4 or IPv6... + */ + + if (addr->ifa_addr == NULL || + (addr->ifa_addr->sa_family != AF_INET +#ifdef AF_INET6 + && addr->ifa_addr->sa_family != AF_INET6 +#endif + ) || + addr->ifa_netmask == NULL || addr->ifa_name == NULL) + continue; + + /* + * Try looking up the hostname for the address as needed... + */ + + if (HostNameLookups) + httpAddrLookup((http_addr_t *)(addr->ifa_addr), hostname, + sizeof(hostname)); + else + { + /* + * Map the default server address and localhost to the server name + * and localhost, respectively; for all other addresses, use the + * numeric address... + */ + + if (httpAddrLocalhost((http_addr_t *)(addr->ifa_addr))) + strcpy(hostname, "localhost"); + else + httpAddrString((http_addr_t *)(addr->ifa_addr), hostname, + sizeof(hostname)); + } + + /* + * Create a new address element... + */ + + hostlen = strlen(hostname); + if ((temp = calloc(1, sizeof(cupsd_netif_t) + hostlen)) == NULL) + break; + + /* + * Copy all of the information... + */ + + strlcpy(temp->name, addr->ifa_name, sizeof(temp->name)); + temp->hostlen = hostlen; + strcpy(temp->hostname, hostname); /* Safe because hostname is allocated */ + + if (addr->ifa_addr->sa_family == AF_INET) + { + /* + * Copy IPv4 addresses... + */ + + memcpy(&(temp->address), addr->ifa_addr, sizeof(struct sockaddr_in)); + memcpy(&(temp->mask), addr->ifa_netmask, sizeof(struct sockaddr_in)); + + if (addr->ifa_dstaddr) + memcpy(&(temp->broadcast), addr->ifa_dstaddr, + sizeof(struct sockaddr_in)); + } +#ifdef AF_INET6 + else + { + /* + * Copy IPv6 addresses... + */ + + memcpy(&(temp->address), addr->ifa_addr, sizeof(struct sockaddr_in6)); + memcpy(&(temp->mask), addr->ifa_netmask, sizeof(struct sockaddr_in6)); + + if (addr->ifa_dstaddr) + memcpy(&(temp->broadcast), addr->ifa_dstaddr, + sizeof(struct sockaddr_in6)); + } +#endif /* AF_INET6 */ + + if (!(addr->ifa_flags & IFF_POINTOPOINT) && + !httpAddrLocalhost(&(temp->address))) + temp->is_local = 1; + + /* + * Determine which port to use when advertising printers... + */ + + for (lis = (cupsd_listener_t *)cupsArrayFirst(Listeners); + lis; + lis = (cupsd_listener_t *)cupsArrayNext(Listeners)) + { + match = 0; + + if (httpAddrAny(&(lis->address))) + match = 1; + else if (addr->ifa_addr->sa_family == AF_INET && + lis->address.addr.sa_family == AF_INET && + (lis->address.ipv4.sin_addr.s_addr & + temp->mask.ipv4.sin_addr.s_addr) == + (temp->address.ipv4.sin_addr.s_addr & + temp->mask.ipv4.sin_addr.s_addr)) + match = 1; +#ifdef AF_INET6 + else if (addr->ifa_addr->sa_family == AF_INET6 && + lis->address.addr.sa_family == AF_INET6 && + (lis->address.ipv6.sin6_addr.s6_addr[0] & + temp->mask.ipv6.sin6_addr.s6_addr[0]) == + (temp->address.ipv6.sin6_addr.s6_addr[0] & + temp->mask.ipv6.sin6_addr.s6_addr[0]) && + (lis->address.ipv6.sin6_addr.s6_addr[1] & + temp->mask.ipv6.sin6_addr.s6_addr[1]) == + (temp->address.ipv6.sin6_addr.s6_addr[1] & + temp->mask.ipv6.sin6_addr.s6_addr[1]) && + (lis->address.ipv6.sin6_addr.s6_addr[2] & + temp->mask.ipv6.sin6_addr.s6_addr[2]) == + (temp->address.ipv6.sin6_addr.s6_addr[2] & + temp->mask.ipv6.sin6_addr.s6_addr[2]) && + (lis->address.ipv6.sin6_addr.s6_addr[3] & + temp->mask.ipv6.sin6_addr.s6_addr[3]) == + (temp->address.ipv6.sin6_addr.s6_addr[3] & + temp->mask.ipv6.sin6_addr.s6_addr[3])) + match = 1; +#endif /* AF_INET6 */ + + if (match) + { + temp->port = _httpAddrPort(&(lis->address)); + break; + } + } + + /* + * Add it to the array... + */ + + cupsArrayAdd(NetIFList, temp); + + cupsdLogMessage(CUPSD_LOG_DEBUG, "cupsdNetIFUpdate: \"%s\" = %s:%d", + temp->name, temp->hostname, temp->port); + } + + freeifaddrs(addrs); +} + + +/* + * 'compare_netif()' - Compare two network interfaces. + */ + +static int /* O - Result of comparison */ +compare_netif(cupsd_netif_t *a, /* I - First network interface */ + cupsd_netif_t *b) /* I - Second network interface */ +{ + return (strcmp(a->name, b->name)); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/network.h b/scheduler/network.h new file mode 100644 index 0000000000..c7a6aa0c39 --- /dev/null +++ b/scheduler/network.h @@ -0,0 +1,52 @@ +/* + * "$Id$" + * + * Network interface definitions for the CUPS scheduler. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * "LICENSE" which should have been included with this file. If this + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +/* + * Structures... + */ + +typedef struct cupsd_netif_s /**** Network interface data ****/ +{ + int is_local, /* Local (not point-to-point) interface? */ + port; /* Listen port */ + http_addr_t address, /* Network address */ + mask, /* Network mask */ + broadcast; /* Broadcast address */ + size_t hostlen; /* Length of hostname */ + char name[32], /* Network interface name */ + hostname[1]; /* Hostname associated with interface */ +} cupsd_netif_t; + + +/* + * Globals... + */ + +VAR int NetIFUpdate VALUE(1); + /* Network interface list needs updating */ +VAR cups_array_t *NetIFList VALUE(NULL); + /* Array of network interfaces */ + +/* + * Prototypes... + */ + +extern cupsd_netif_t *cupsdNetIFFind(const char *name); +extern void cupsdNetIFUpdate(void); + + +/* + * End of "$Id$". + */ diff --git a/scheduler/org.cups.cups-lpd.plist.in b/scheduler/org.cups.cups-lpd.plist.in new file mode 100644 index 0000000000..32067ef1da --- /dev/null +++ b/scheduler/org.cups.cups-lpd.plist.in @@ -0,0 +1,33 @@ + + + + + Disabled + + Label + org.cups.cups-lpd + ProgramArguments + + /usr/libexec/cups/daemon/cups-lpd + -o + document-format=application/octet-stream + + Sockets + + Listeners + + SockServiceName + printer + SockType + stream + + + UserName + @CUPS_USER@ + inetdCompatibility + + Wait + + + + diff --git a/scheduler/org.cups.cupsd.plist b/scheduler/org.cups.cupsd.plist new file mode 100644 index 0000000000..ccc861d8f2 --- /dev/null +++ b/scheduler/org.cups.cupsd.plist @@ -0,0 +1,62 @@ + + + + + Label + org.cups.cupsd + EnableTransactions + + ExitTimeOut + 60 + KeepAlive + + PathState + + /private/var/spool/cups/cache/org.cups.cupsd + + + + ProgramArguments + + /usr/sbin/cupsd + -l + + + EnvironmentVariables + + CUPS_DEBUG_LOG + /var/log/cups/debug_log + CUPS_DEBUG_LEVEL + 3 + CUPS_DEBUG_FILTER + ^(http|_http|ipp|_ipp|mime).* + + ServiceIPC + + Sockets + + Listeners + + + SockNodeName + ::1 + SockServiceName + ipp + + + SockNodeName + 127.0.0.1 + SockServiceName + ipp + + + SockPathMode + 49663 + SockPathName + /private/var/run/cupsd + + + + + diff --git a/scheduler/policy.c b/scheduler/policy.c new file mode 100644 index 0000000000..466067036e --- /dev/null +++ b/scheduler/policy.c @@ -0,0 +1,517 @@ +/* + * "$Id$" + * + * Policy routines for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * AddPolicy() - Add a policy to the system. + * cupsdAddPolicyOp() - Add an operation to a policy. + * cupsdCheckPolicy() - Check the IPP operation and username against a + * policy. + * cupsdDeleteAllPolicies() - Delete all policies in memory. + * cupsdFindPolicy() - Find a named policy. + * cupsdFindPolicyOp() - Find a policy operation. + * cupsdGetPrivateAttrs() - Get the private attributes for the current + * request. + * compare_ops() - Compare two operations. + * compare_policies() - Compare two policies. + * free_policy() - Free the memory used by a policy. + * hash_op() - Generate a lookup hash for the operation. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" +#include + + +/* + * Local functions... + */ + +static int compare_ops(cupsd_location_t *a, cupsd_location_t *b); +static int compare_policies(cupsd_policy_t *a, cupsd_policy_t *b); +static void free_policy(cupsd_policy_t *p); +static int hash_op(cupsd_location_t *op); + + +/* + * 'AddPolicy()' - Add a policy to the system. + */ + +cupsd_policy_t * /* O - Policy */ +cupsdAddPolicy(const char *policy) /* I - Name of policy */ +{ + cupsd_policy_t *temp; /* Pointer to policy */ + + + if (!policy) + return (NULL); + + if (!Policies) + Policies = cupsArrayNew3((cups_array_func_t)compare_policies, NULL, + (cups_ahash_func_t)NULL, 0, + (cups_acopy_func_t)NULL, + (cups_afree_func_t)free_policy); + + if (!Policies) + return (NULL); + + if ((temp = calloc(1, sizeof(cupsd_policy_t))) != NULL) + { + cupsdSetString(&temp->name, policy); + cupsArrayAdd(Policies, temp); + } + + return (temp); +} + + +/* + * 'cupsdAddPolicyOp()' - Add an operation to a policy. + */ + +cupsd_location_t * /* O - New policy operation */ +cupsdAddPolicyOp(cupsd_policy_t *p, /* I - Policy */ + cupsd_location_t *po, /* I - Policy operation to copy */ + ipp_op_t op) /* I - IPP operation code */ +{ + cupsd_location_t *temp; /* New policy operation */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdAddPolicyOp(p=%p, po=%p, op=%x(%s))", + p, po, op, ippOpString(op)); + + if (!p) + return (NULL); + + if (!p->ops) + p->ops = cupsArrayNew3((cups_array_func_t)compare_ops, NULL, + (cups_ahash_func_t)hash_op, 128, + (cups_acopy_func_t)NULL, + (cups_afree_func_t)cupsdFreeLocation); + + if (!p->ops) + return (NULL); + + if ((temp = cupsdCopyLocation(po)) != NULL) + { + temp->op = op; + temp->limit = CUPSD_AUTH_LIMIT_IPP; + + cupsArrayAdd(p->ops, temp); + } + + return (temp); +} + + +/* + * 'cupsdCheckPolicy()' - Check the IPP operation and username against a policy. + */ + +http_status_t /* I - 1 if OK, 0 otherwise */ +cupsdCheckPolicy(cupsd_policy_t *p, /* I - Policy */ + cupsd_client_t *con, /* I - Client connection */ + const char *owner) /* I - Owner of object */ +{ + cupsd_location_t *po; /* Current policy operation */ + + + /* + * Range check... + */ + + if (!p || !con) + { + cupsdLogMessage(CUPSD_LOG_CRIT, "cupsdCheckPolicy: p=%p, con=%p!", p, con); + + return ((http_status_t)0); + } + + /* + * Find a match for the operation... + */ + + if ((po = cupsdFindPolicyOp(p, con->request->request.op.operation_id)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdCheckPolicy: No matching operation, returning 0!"); + return ((http_status_t)0); + } + + con->best = po; + + /* + * Return the status of the check... + */ + + return (cupsdIsAuthorized(con, owner)); +} + + +/* + * 'cupsdDeleteAllPolicies()' - Delete all policies in memory. + */ + +void +cupsdDeleteAllPolicies(void) +{ + cupsd_printer_t *printer; /* Current printer */ + + + if (!Policies) + return; + + /* + * First clear the policy pointers for all printers... + */ + + for (printer = (cupsd_printer_t *)cupsArrayFirst(Printers); + printer; + printer = (cupsd_printer_t *)cupsArrayNext(Printers)) + printer->op_policy_ptr = NULL; + + DefaultPolicyPtr = NULL; + + /* + * Then free all of the policies... + */ + + cupsArrayDelete(Policies); + + Policies = NULL; +} + + +/* + * 'cupsdFindPolicy()' - Find a named policy. + */ + +cupsd_policy_t * /* O - Policy */ +cupsdFindPolicy(const char *policy) /* I - Name of policy */ +{ + cupsd_policy_t key; /* Search key */ + + + /* + * Range check... + */ + + if (!policy) + return (NULL); + + /* + * Look it up... + */ + + key.name = (char *)policy; + return ((cupsd_policy_t *)cupsArrayFind(Policies, &key)); +} + + +/* + * 'cupsdFindPolicyOp()' - Find a policy operation. + */ + +cupsd_location_t * /* O - Policy operation */ +cupsdFindPolicyOp(cupsd_policy_t *p, /* I - Policy */ + ipp_op_t op) /* I - IPP operation */ +{ + cupsd_location_t key, /* Search key... */ + *po; /* Current policy operation */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdFindPolicyOp(p=%p, op=%x(%s))", + p, op, ippOpString(op)); + + /* + * Range check... + */ + + if (!p) + return (NULL); + + /* + * Check the operation against the available policies... + */ + + key.op = op; + if ((po = (cupsd_location_t *)cupsArrayFind(p->ops, &key)) != NULL) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdFindPolicyOp: Found exact match..."); + return (po); + } + + key.op = IPP_ANY_OPERATION; + if ((po = (cupsd_location_t *)cupsArrayFind(p->ops, &key)) != NULL) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdFindPolicyOp: Found wildcard match..."); + return (po); + } + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdFindPolicyOp: No match found!"); + + return (NULL); +} + + +/* + * 'cupsdGetPrivateAttrs()' - Get the private attributes for the current + * request. + */ + +cups_array_t * /* O - Array or NULL for no restrictions */ +cupsdGetPrivateAttrs( + cupsd_policy_t *policy, /* I - Policy */ + cupsd_client_t *con, /* I - Client connection */ + cupsd_printer_t *printer, /* I - Printer, if any */ + const char *owner) /* I - Owner of object */ +{ + char *name; /* Current name in access list */ + cups_array_t *access_ptr, /* Access array */ + *attrs_ptr; /* Attributes array */ + const char *username; /* Username associated with request */ + ipp_attribute_t *attr; /* Attribute from request */ + struct passwd *pw; /* User info */ + + +#ifdef DEBUG + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdGetPrivateAttrs(policy=%p(%s), con=%p(%d), " + "printer=%p(%s), owner=\"%s\")", policy, policy->name, con, + con->http.fd, printer, printer ? printer->name : "", owner); +#endif /* DEBUG */ + + /* + * Get the access and attributes lists that correspond to the request... + */ + +#ifdef DEBUG + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdGetPrivateAttrs: %s", + ippOpString(con->request->request.op.operation_id)); +#endif /* DEBUG */ + + switch (con->request->request.op.operation_id) + { + case IPP_GET_SUBSCRIPTIONS : + case IPP_GET_SUBSCRIPTION_ATTRIBUTES : + case IPP_GET_NOTIFICATIONS : + access_ptr = policy->sub_access; + attrs_ptr = policy->sub_attrs; + break; + + default : + access_ptr = policy->job_access; + attrs_ptr = policy->job_attrs; + break; + } + + /* + * If none of the attributes are private, return NULL now... + */ + + if ((name = (char *)cupsArrayFirst(attrs_ptr)) != NULL && + !_cups_strcasecmp(name, "none")) + { +#ifdef DEBUG + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdGetPrivateAttrs: Returning NULL."); +#endif /* DEBUG */ + + return (NULL); + } + + /* + * Otherwise check the user against the access list... + */ + + if (con->username[0]) + username = con->username; + else if ((attr = ippFindAttribute(con->request, "requesting-user-name", + IPP_TAG_NAME)) != NULL) + username = attr->values[0].string.text; + else + username = "anonymous"; + + if (username[0]) + { + pw = getpwnam(username); + endpwent(); + } + else + pw = NULL; + +#ifdef DEBUG + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdGetPrivateAttrs: username=\"%s\"", + username); +#endif /* DEBUG */ + + /* + * Otherwise check the user against the access list... + */ + + for (name = (char *)cupsArrayFirst(access_ptr); + name; + name = (char *)cupsArrayNext(access_ptr)) + { +#ifdef DEBUG + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdGetPrivateAttrs: name=%s", name); +#endif /* DEBUG */ + + if (printer && !_cups_strcasecmp(name, "@ACL")) + { + char *acl; /* Current ACL user/group */ + + for (acl = (char *)cupsArrayFirst(printer->users); + acl; + acl = (char *)cupsArrayNext(printer->users)) + { + if (acl[0] == '@') + { + /* + * Check group membership... + */ + + if (cupsdCheckGroup(username, pw, acl + 1)) + break; + } + else if (acl[0] == '#') + { + /* + * Check UUID... + */ + + if (cupsdCheckGroup(username, pw, acl)) + break; + } + else if (!_cups_strcasecmp(username, acl)) + break; + } + } + else if (owner && !_cups_strcasecmp(name, "@OWNER") && + !_cups_strcasecmp(username, owner)) + { +#ifdef DEBUG + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdGetPrivateAttrs: Returning NULL."); +#endif /* DEBUG */ + + return (NULL); + } + else if (!_cups_strcasecmp(name, "@SYSTEM")) + { + int i; /* Looping var */ + + for (i = 0; i < NumSystemGroups; i ++) + if (cupsdCheckGroup(username, pw, SystemGroups[i])) + { +#ifdef DEBUG + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdGetPrivateAttrs: Returning NULL."); +#endif /* DEBUG */ + + return (NULL); + } + } + else if (name[0] == '@') + { + if (cupsdCheckGroup(username, pw, name + 1)) + { +#ifdef DEBUG + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdGetPrivateAttrs: Returning NULL."); +#endif /* DEBUG */ + + return (NULL); + } + } + else if (!_cups_strcasecmp(username, name)) + { +#ifdef DEBUG + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdGetPrivateAttrs: Returning NULL."); +#endif /* DEBUG */ + + return (NULL); + } + } + + /* + * No direct access, so return private attributes list... + */ + +#ifdef DEBUG + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdGetPrivateAttrs: Returning list."); +#endif /* DEBUG */ + + return (attrs_ptr); +} + + +/* + * 'compare_ops()' - Compare two operations. + */ + +static int /* O - Result of comparison */ +compare_ops(cupsd_location_t *a, /* I - First operation */ + cupsd_location_t *b) /* I - Second operation */ +{ + return (a->op - b->op); +} + + +/* + * 'compare_policies()' - Compare two policies. + */ + +static int /* O - Result of comparison */ +compare_policies(cupsd_policy_t *a, /* I - First policy */ + cupsd_policy_t *b) /* I - Second policy */ +{ + return (_cups_strcasecmp(a->name, b->name)); +} + + +/* + * 'free_policy()' - Free the memory used by a policy. + */ + +static void +free_policy(cupsd_policy_t *p) /* I - Policy to free */ +{ + cupsArrayDelete(p->job_access); + cupsArrayDelete(p->job_attrs); + cupsArrayDelete(p->sub_access); + cupsArrayDelete(p->sub_attrs); + cupsArrayDelete(p->ops); + cupsdClearString(&p->name); + free(p); +} + + +/* + * 'hash_op()' - Generate a lookup hash for the operation. + */ + +static int /* O - Hash value */ +hash_op(cupsd_location_t *op) /* I - Operation */ +{ + return (((op->op >> 6) & 0x40) | (op->op & 0x3f)); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/policy.h b/scheduler/policy.h new file mode 100644 index 0000000000..361271f2b0 --- /dev/null +++ b/scheduler/policy.h @@ -0,0 +1,63 @@ +/* + * "$Id$" + * + * Policy definitions for the CUPS scheduler. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + + +/* + * Policy structure... + */ + +typedef struct +{ + char *name; /* Policy name */ + cups_array_t *job_access, /* Private users/groups for jobs */ + *job_attrs, /* Private attributes for jobs */ + *sub_access, /* Private users/groups for subscriptions */ + *sub_attrs, /* Private attributes for subscriptions */ + *ops; /* Operations */ +} cupsd_policy_t; + +typedef struct cupsd_printer_s cupsd_printer_t; + + +/* + * Globals... + */ + +VAR cups_array_t *Policies VALUE(NULL); + /* Policies */ + + +/* + * Prototypes... + */ + +extern cupsd_policy_t *cupsdAddPolicy(const char *policy); +extern cupsd_location_t *cupsdAddPolicyOp(cupsd_policy_t *p, + cupsd_location_t *po, + ipp_op_t op); +extern http_status_t cupsdCheckPolicy(cupsd_policy_t *p, cupsd_client_t *con, + const char *owner); +extern void cupsdDeleteAllPolicies(void); +extern cupsd_policy_t *cupsdFindPolicy(const char *policy); +extern cupsd_location_t *cupsdFindPolicyOp(cupsd_policy_t *p, ipp_op_t op); +extern cups_array_t *cupsdGetPrivateAttrs(cupsd_policy_t *p, + cupsd_client_t *con, + cupsd_printer_t *printer, + const char *owner); + + +/* + * End of "$Id$". + */ diff --git a/scheduler/printers.c b/scheduler/printers.c new file mode 100644 index 0000000000..81b105ac4d --- /dev/null +++ b/scheduler/printers.c @@ -0,0 +1,5273 @@ +/* + * "$Id$" + * + * Printer routines for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdAddPrinter() - Add a printer to the system. + * cupsdCreateCommonData() - Create the common printer data. + * cupsdDeleteAllPrinters() - Delete all printers from the system. + * cupsdDeletePrinter() - Delete a printer from the system. + * cupsdFindDest() - Find a destination in the list. + * cupsdFindPrinter() - Find a printer in the list. + * cupsdLoadAllPrinters() - Load printers from the printers.conf file. + * cupsdRenamePrinter() - Rename a printer. + * cupsdSaveAllPrinters() - Save all printer definitions to the + * printers.conf file. + * cupsdSetAuthInfoRequired() - Set the required authentication info. + * cupsdSetDeviceURI() - Set the device URI for a printer. + * cupsdSetPrinterAttr() - Set a printer attribute. + * cupsdSetPrinterAttrs() - Set printer attributes based upon the PPD + * file. + * cupsdSetPrinterReasons() - Set/update the reasons strings. + * cupsdSetPrinterState() - Update the current state of a printer. + * cupsdStopPrinter() - Stop a printer from printing any jobs... + * cupsdUpdatePrinterPPD() - Update keywords in a printer's PPD file. + * cupsdUpdatePrinters() - Update printers after a partial reload. + * cupsdValidateDest() - Validate a printer/class destination. + * cupsdWritePrintcap() - Write a pseudo-printcap file for older + * applications that need it... + * add_printer_defaults() - Add name-default attributes to the printer + * attributes. + * add_printer_filter() - Add a MIME filter for a printer. + * add_printer_formats() - Add document-format-supported values for a + * printer. + * compare_printers() - Compare two printers. + * delete_printer_filters() - Delete all MIME filters for a printer. + * dirty_printer() - Mark config and state files dirty for the + * specified printer. + * load_ppd() - Load a cached PPD file, updating the cache as + * needed. + * new_media_col() - Create a media-col collection value. + * write_irix_config() - Update the config files used by the IRIX + * desktop tools. + * write_irix_state() - Update the status files used by IRIX printing + * desktop tools. + * write_xml_string() - Write a string with XML escaping. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" +#include +#ifdef HAVE_APPLICATIONSERVICES_H +# include +#endif /* HAVE_APPLICATIONSERVICES_H */ +#ifdef HAVE_SYS_MOUNT_H +# include +#endif /* HAVE_SYS_MOUNT_H */ +#ifdef HAVE_SYS_STATVFS_H +# include +#elif defined(HAVE_SYS_STATFS_H) +# include +#endif /* HAVE_SYS_STATVFS_H */ +#ifdef HAVE_SYS_VFS_H +# include +#endif /* HAVE_SYS_VFS_H */ +#ifdef __APPLE__ +# include +#endif /* __APPLE__ */ + + +/* + * Local functions... + */ + +static void add_printer_defaults(cupsd_printer_t *p); +static void add_printer_filter(cupsd_printer_t *p, mime_type_t *type, + const char *filter); +static void add_printer_formats(cupsd_printer_t *p); +static int compare_printers(void *first, void *second, void *data); +static void delete_printer_filters(cupsd_printer_t *p); +static void dirty_printer(cupsd_printer_t *p); +static void load_ppd(cupsd_printer_t *p); +static void log_ipp_conformance(cupsd_printer_t *p, const char *reason); +static ipp_t *new_media_col(_pwg_size_t *size, const char *source, + const char *type); +#ifdef __sgi +static void write_irix_config(cupsd_printer_t *p); +static void write_irix_state(cupsd_printer_t *p); +#endif /* __sgi */ +static void write_xml_string(cups_file_t *fp, const char *s); + + +/* + * 'cupsdAddPrinter()' - Add a printer to the system. + */ + +cupsd_printer_t * /* O - New printer */ +cupsdAddPrinter(const char *name) /* I - Name of printer */ +{ + cupsd_printer_t *p; /* New printer */ + char uri[1024], /* Printer URI */ + uuid[64]; /* Printer UUID */ + + + /* + * Range check input... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdAddPrinter(\"%s\")", name); + + /* + * Create a new printer entity... + */ + + if ((p = calloc(1, sizeof(cupsd_printer_t))) == NULL) + { + cupsdLogMessage(CUPSD_LOG_CRIT, "Unable to allocate memory for printer - %s", + strerror(errno)); + return (NULL); + } + + cupsdSetString(&p->name, name); + cupsdSetString(&p->info, name); + cupsdSetString(&p->hostname, ServerName); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + ServerName, RemotePort, "/printers/%s", name); + cupsdSetString(&p->uri, uri); + cupsdSetString(&p->uuid, _httpAssembleUUID(ServerName, RemotePort, name, 0, + uuid, sizeof(uuid))); + cupsdSetDeviceURI(p, "file:///dev/null"); + + p->state = IPP_PRINTER_STOPPED; + p->state_time = time(NULL); + p->accepting = 0; + p->shared = DefaultShared; + p->filetype = mimeAddType(MimeDatabase, "printer", name); + + cupsdSetString(&p->job_sheets[0], "none"); + cupsdSetString(&p->job_sheets[1], "none"); + + cupsdSetString(&p->error_policy, ErrorPolicy); + cupsdSetString(&p->op_policy, DefaultPolicy); + + p->op_policy_ptr = DefaultPolicyPtr; + + /* + * Insert the printer in the printer list alphabetically... + */ + + if (!Printers) + Printers = cupsArrayNew(compare_printers, NULL); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdAddPrinter: Adding %s to Printers", p->name); + cupsArrayAdd(Printers, p); + + /* + * Return the new printer... + */ + + return (p); +} + + +/* + * 'cupsdCreateCommonData()' - Create the common printer data. + */ + +void +cupsdCreateCommonData(void) +{ + int i; /* Looping var */ + ipp_attribute_t *attr; /* Attribute data */ + cups_dir_t *dir; /* Notifier directory */ + cups_dentry_t *dent; /* Notifier directory entry */ + cups_array_t *notifiers; /* Notifier array */ + char filename[1024], /* Filename */ + *notifier; /* Current notifier */ + cupsd_policy_t *p; /* Current policy */ + int k_supported; /* Maximum file size supported */ +#ifdef HAVE_STATVFS + struct statvfs spoolinfo; /* FS info for spool directory */ + double spoolsize; /* FS size */ +#elif defined(HAVE_STATFS) + struct statfs spoolinfo; /* FS info for spool directory */ + double spoolsize; /* FS size */ +#endif /* HAVE_STATVFS */ + static const int nups[] = /* number-up-supported values */ + { 1, 2, 4, 6, 9, 16 }; + static const int orients[4] =/* orientation-requested-supported values */ + { + IPP_PORTRAIT, + IPP_LANDSCAPE, + IPP_REVERSE_LANDSCAPE, + IPP_REVERSE_PORTRAIT + }; + static const char * const holds[] = /* job-hold-until-supported values */ + { + "no-hold", + "indefinite", + "day-time", + "evening", + "night", + "second-shift", + "third-shift", + "weekend" + }; + static const char * const versions[] =/* ipp-versions-supported values */ + { + "1.0", + "1.1", + "2.0", + "2.1" + }; + static const int ops[] = /* operations-supported values */ + { + IPP_PRINT_JOB, + IPP_VALIDATE_JOB, + IPP_CREATE_JOB, + IPP_SEND_DOCUMENT, + IPP_CANCEL_JOB, + IPP_GET_JOB_ATTRIBUTES, + IPP_GET_JOBS, + IPP_GET_PRINTER_ATTRIBUTES, + IPP_HOLD_JOB, + IPP_RELEASE_JOB, + IPP_RESTART_JOB, + IPP_PAUSE_PRINTER, + IPP_RESUME_PRINTER, + IPP_PURGE_JOBS, + IPP_SET_PRINTER_ATTRIBUTES, + IPP_SET_JOB_ATTRIBUTES, + IPP_GET_PRINTER_SUPPORTED_VALUES, + IPP_CREATE_PRINTER_SUBSCRIPTION, + IPP_CREATE_JOB_SUBSCRIPTION, + IPP_GET_SUBSCRIPTION_ATTRIBUTES, + IPP_GET_SUBSCRIPTIONS, + IPP_RENEW_SUBSCRIPTION, + IPP_CANCEL_SUBSCRIPTION, + IPP_GET_NOTIFICATIONS, + IPP_ENABLE_PRINTER, + IPP_DISABLE_PRINTER, + IPP_HOLD_NEW_JOBS, + IPP_RELEASE_HELD_NEW_JOBS, + IPP_CANCEL_JOBS, + IPP_CANCEL_MY_JOBS, + IPP_CLOSE_JOB, + CUPS_GET_DEFAULT, + CUPS_GET_PRINTERS, + CUPS_ADD_PRINTER, + CUPS_DELETE_PRINTER, + CUPS_GET_CLASSES, + CUPS_ADD_CLASS, + CUPS_DELETE_CLASS, + CUPS_ACCEPT_JOBS, + CUPS_REJECT_JOBS, + CUPS_SET_DEFAULT, + CUPS_GET_DEVICES, + CUPS_GET_PPDS, + CUPS_MOVE_JOB, + CUPS_AUTHENTICATE_JOB, + CUPS_GET_PPD, + CUPS_GET_DOCUMENT, + IPP_RESTART_JOB + }; + static const char * const charsets[] =/* charset-supported values */ + { + "us-ascii", + "utf-8" + }; + static const char * const compressions[] = + { /* document-compression-supported values */ + "none" +#ifdef HAVE_LIBZ + ,"gzip" +#endif /* HAVE_LIBZ */ + }; + static const char * const media_col_supported[] = + { /* media-col-supported values */ + "media-bottom-margin", + "media-left-margin", + "media-right-margin", + "media-size", + "media-source", + "media-top-margin", + "media-type" + }; + static const char * const multiple_document_handling[] = + { /* multiple-document-handling-supported values */ + "separate-documents-uncollated-copies", + "separate-documents-collated-copies" + }; + static const char * const notify_attrs[] = + { /* notify-attributes-supported values */ + "printer-state-change-time", + "notify-lease-expiration-time", + "notify-subscriber-user-name" + }; + static const char * const notify_events[] = + { /* notify-events-supported values */ + "job-completed", + "job-config-changed", + "job-created", + "job-progress", + "job-state-changed", + "job-stopped", + "printer-added", + "printer-changed", + "printer-config-changed", + "printer-deleted", + "printer-finishings-changed", + "printer-media-changed", + "printer-modified", + "printer-restarted", + "printer-shutdown", + "printer-state-changed", + "printer-stopped", + "server-audit", + "server-restarted", + "server-started", + "server-stopped" + }; + static const char * const job_creation[] = + { /* job-creation-attributes-supported */ + "copies", + "finishings", + "ipp-attribute-fidelity", + "job-hold-until", + "job-name", + "job-priority", + "job-sheets", + "media", + "media-col", + "multiple-document-handling", + "number-up", + "output-bin", + "output-mode", + "orientation-requested", + "page-ranges", + "print-quality", + "printer-resolution", + "sides" + }; + static const char * const job_settable[] = + { /* job-settable-attributes-supported */ + "copies", + "finishings", + "job-hold-until", + "job-name", + "job-priority", + "media", + "media-col", + "multiple-document-handling", + "number-up", + "output-bin", + "output-mode", + "orientation-requested", + "page-ranges", + "print-quality", + "printer-resolution", + "sides" + }; + static const char * const pdf_versions[] = + { /* pdf-versions-supported */ + "adobe-1.2", + "adobe-1.3", + "adobe-1.4", + "adobe-1.5", + "adobe-1.6", + "adobe-1.7", + "iso-19005-1_2005", + "iso-32000-1_2008", + "pwg-5102.3" + }; + static const char * const printer_settable[] = + { /* printer-settable-attributes-supported */ + "printer-info", + "printer-location" + }; + static const char * const which_jobs[] = + { /* which-jobs-supported values */ + "completed", + "not-completed", + "aborted", + "all", + "canceled", + "pending", + "pending-held", + "processing", + "processing-stopped" + }; + + + if (CommonData) + ippDelete(CommonData); + + CommonData = ippNew(); + + /* + * Get the maximum spool size based on the size of the filesystem used for + * the RequestRoot directory. If the host OS doesn't support the statfs call + * or the filesystem is larger than 2TiB, always report INT_MAX. + */ + +#ifdef HAVE_STATVFS + if (statvfs(RequestRoot, &spoolinfo)) + k_supported = INT_MAX; + else if ((spoolsize = (double)spoolinfo.f_frsize * spoolinfo.f_blocks / 1024) > + INT_MAX) + k_supported = INT_MAX; + else + k_supported = (int)spoolsize; + +#elif defined(HAVE_STATFS) + if (statfs(RequestRoot, &spoolinfo)) + k_supported = INT_MAX; + else if ((spoolsize = (double)spoolinfo.f_bsize * spoolinfo.f_blocks / 1024) > + INT_MAX) + k_supported = INT_MAX; + else + k_supported = (int)spoolsize; + +#else + k_supported = INT_MAX; +#endif /* HAVE_STATVFS */ + + /* + * This list of attributes is sorted to improve performance when the + * client provides a requested-attributes attribute... + */ + + /* charset-configured */ + ippAddString(CommonData, IPP_TAG_PRINTER, IPP_TAG_CHARSET | IPP_TAG_COPY, + "charset-configured", NULL, "utf-8"); + + /* charset-supported */ + ippAddStrings(CommonData, IPP_TAG_PRINTER, IPP_TAG_CHARSET | IPP_TAG_COPY, + "charset-supported", sizeof(charsets) / sizeof(charsets[0]), + NULL, charsets); + + /* compression-supported */ + ippAddStrings(CommonData, IPP_TAG_PRINTER, IPP_TAG_KEYWORD | IPP_TAG_COPY, + "compression-supported", + sizeof(compressions) / sizeof(compressions[0]), + NULL, compressions); + + /* copies-supported */ + ippAddRange(CommonData, IPP_TAG_PRINTER, "copies-supported", 1, MaxCopies); + + /* cups-version */ + ippAddString(CommonData, IPP_TAG_PRINTER, IPP_TAG_TEXT | IPP_TAG_COPY, + "cups-version", NULL, CUPS_SVERSION + 6); + + /* generated-natural-language-supported (no IPP_TAG_COPY) */ + ippAddString(CommonData, IPP_TAG_PRINTER, IPP_TAG_LANGUAGE, + "generated-natural-language-supported", NULL, DefaultLanguage); + + /* ipp-versions-supported */ + ippAddStrings(CommonData, IPP_TAG_PRINTER, IPP_TAG_KEYWORD | IPP_TAG_COPY, + "ipp-versions-supported", sizeof(versions) / sizeof(versions[0]), + NULL, versions); + + /* ippget-event-life */ + ippAddInteger(CommonData, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "ippget-event-life", 15); + + /* job-creation-attributes-supported */ + ippAddStrings(CommonData, IPP_TAG_PRINTER, IPP_TAG_KEYWORD | IPP_TAG_COPY, + "job-creation-attributes-supported", + sizeof(job_creation) / sizeof(job_creation[0]), + NULL, job_creation); + + /* job-hold-until-supported */ + ippAddStrings(CommonData, IPP_TAG_PRINTER, IPP_TAG_KEYWORD | IPP_TAG_COPY, + "job-hold-until-supported", sizeof(holds) / sizeof(holds[0]), + NULL, holds); + + /* job-ids-supported */ + ippAddBoolean(CommonData, IPP_TAG_PRINTER, "job-ids-supported", 1); + + /* job-k-octets-supported */ + ippAddRange(CommonData, IPP_TAG_PRINTER, "job-k-octets-supported", 0, + k_supported); + + /* job-priority-supported */ + ippAddInteger(CommonData, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "job-priority-supported", 100); + + /* job-settable-attributes-supported */ + ippAddStrings(CommonData, IPP_TAG_PRINTER, IPP_TAG_KEYWORD | IPP_TAG_COPY, + "job-settable-attributes-supported", + sizeof(job_settable) / sizeof(job_settable[0]), + NULL, job_settable); + + /* job-sheets-supported */ + if (cupsArrayCount(Banners) > 0) + { + /* + * Setup the job-sheets-supported attribute... + */ + + if (Classification && !ClassifyOverride) + attr = ippAddString(CommonData, IPP_TAG_PRINTER, + IPP_TAG_NAME | IPP_TAG_COPY, + "job-sheets-supported", NULL, Classification); + else + attr = ippAddStrings(CommonData, IPP_TAG_PRINTER, + IPP_TAG_NAME | IPP_TAG_COPY, + "job-sheets-supported", cupsArrayCount(Banners) + 1, + NULL, NULL); + + if (attr == NULL) + cupsdLogMessage(CUPSD_LOG_EMERG, + "Unable to allocate memory for " + "job-sheets-supported attribute: %s!", strerror(errno)); + else if (!Classification || ClassifyOverride) + { + cupsd_banner_t *banner; /* Current banner */ + + + attr->values[0].string.text = _cupsStrAlloc("none"); + + for (i = 1, banner = (cupsd_banner_t *)cupsArrayFirst(Banners); + banner; + i ++, banner = (cupsd_banner_t *)cupsArrayNext(Banners)) + attr->values[i].string.text = banner->name; + } + } + else + ippAddString(CommonData, IPP_TAG_PRINTER, IPP_TAG_NAME | IPP_TAG_COPY, + "job-sheets-supported", NULL, "none"); + + /* jpeg-k-octets-supported */ + ippAddRange(CommonData, IPP_TAG_PRINTER, "jpeg-k-octets-supported", 0, + k_supported); + + /* jpeg-x-dimension-supported */ + ippAddRange(CommonData, IPP_TAG_PRINTER, "jpeg-x-dimension-supported", 0, + 65535); + + /* jpeg-y-dimension-supported */ + ippAddRange(CommonData, IPP_TAG_PRINTER, "jpeg-y-dimension-supported", 1, + 65535); + + /* media-col-supported */ + ippAddStrings(CommonData, IPP_TAG_PRINTER, IPP_TAG_KEYWORD | IPP_TAG_COPY, + "media-col-supported", + sizeof(media_col_supported) / + sizeof(media_col_supported[0]), NULL, + media_col_supported); + + /* multiple-document-handling-supported */ + ippAddStrings(CommonData, IPP_TAG_PRINTER, IPP_TAG_KEYWORD | IPP_TAG_COPY, + "multiple-document-handling-supported", + sizeof(multiple_document_handling) / + sizeof(multiple_document_handling[0]), NULL, + multiple_document_handling); + + /* multiple-document-jobs-supported */ + ippAddBoolean(CommonData, IPP_TAG_PRINTER, + "multiple-document-jobs-supported", 1); + + /* multiple-operation-time-out */ + ippAddInteger(CommonData, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "multiple-operation-time-out", MultipleOperationTimeout); + + /* natural-language-configured (no IPP_TAG_COPY) */ + ippAddString(CommonData, IPP_TAG_PRINTER, IPP_TAG_LANGUAGE, + "natural-language-configured", NULL, DefaultLanguage); + + /* notify-attributes-supported */ + ippAddStrings(CommonData, IPP_TAG_PRINTER, IPP_TAG_KEYWORD | IPP_TAG_COPY, + "notify-attributes-supported", + (int)(sizeof(notify_attrs) / sizeof(notify_attrs[0])), + NULL, notify_attrs); + + /* notify-lease-duration-supported */ + ippAddRange(CommonData, IPP_TAG_PRINTER, + "notify-lease-duration-supported", 0, + MaxLeaseDuration ? MaxLeaseDuration : 2147483647); + + /* notify-max-events-supported */ + ippAddInteger(CommonData, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "notify-max-events-supported", MaxEvents); + + /* notify-events-supported */ + ippAddStrings(CommonData, IPP_TAG_PRINTER, IPP_TAG_KEYWORD | IPP_TAG_COPY, + "notify-events-supported", + (int)(sizeof(notify_events) / sizeof(notify_events[0])), + NULL, notify_events); + + /* notify-pull-method-supported */ + ippAddString(CommonData, IPP_TAG_PRINTER, IPP_TAG_KEYWORD | IPP_TAG_COPY, + "notify-pull-method-supported", NULL, "ippget"); + + /* notify-schemes-supported */ + snprintf(filename, sizeof(filename), "%s/notifier", ServerBin); + if ((dir = cupsDirOpen(filename)) != NULL) + { + notifiers = cupsArrayNew((cups_array_func_t)strcmp, NULL); + + while ((dent = cupsDirRead(dir)) != NULL) + if (S_ISREG(dent->fileinfo.st_mode) && + (dent->fileinfo.st_mode & S_IXOTH) != 0) + cupsArrayAdd(notifiers, _cupsStrAlloc(dent->filename)); + + if (cupsArrayCount(notifiers) > 0) + { + attr = ippAddStrings(CommonData, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "notify-schemes-supported", + cupsArrayCount(notifiers), NULL, NULL); + + for (i = 0, notifier = (char *)cupsArrayFirst(notifiers); + notifier; + i ++, notifier = (char *)cupsArrayNext(notifiers)) + attr->values[i].string.text = notifier; + } + + cupsArrayDelete(notifiers); + cupsDirClose(dir); + } + + /* number-up-supported */ + ippAddIntegers(CommonData, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "number-up-supported", sizeof(nups) / sizeof(nups[0]), nups); + + /* operations-supported */ + ippAddIntegers(CommonData, IPP_TAG_PRINTER, IPP_TAG_ENUM, + "operations-supported", + sizeof(ops) / sizeof(ops[0]) + JobFiles - 1, ops); + + /* orientation-requested-supported */ + ippAddIntegers(CommonData, IPP_TAG_PRINTER, IPP_TAG_ENUM, + "orientation-requested-supported", 4, orients); + + /* page-ranges-supported */ + ippAddBoolean(CommonData, IPP_TAG_PRINTER, "page-ranges-supported", 1); + + /* pdf-k-octets-supported */ + ippAddRange(CommonData, IPP_TAG_PRINTER, "pdf-k-octets-supported", 0, + k_supported); + + /* pdf-versions-supported */ + ippAddStrings(CommonData, IPP_TAG_PRINTER, IPP_TAG_KEYWORD | IPP_TAG_COPY, + "pdf-versions-supported", + sizeof(pdf_versions) / sizeof(pdf_versions[0]), NULL, + pdf_versions); + + /* pdl-override-supported */ + ippAddString(CommonData, IPP_TAG_PRINTER, IPP_TAG_KEYWORD | IPP_TAG_COPY, + "pdl-override-supported", NULL, "attempted"); + + /* printer-op-policy-supported */ + attr = ippAddStrings(CommonData, IPP_TAG_PRINTER, IPP_TAG_NAME | IPP_TAG_COPY, + "printer-op-policy-supported", cupsArrayCount(Policies), + NULL, NULL); + for (i = 0, p = (cupsd_policy_t *)cupsArrayFirst(Policies); + p; + i ++, p = (cupsd_policy_t *)cupsArrayNext(Policies)) + attr->values[i].string.text = p->name; + + /* printer-settable-attributes-supported */ + ippAddStrings(CommonData, IPP_TAG_PRINTER, IPP_TAG_KEYWORD | IPP_TAG_COPY, + "printer-settable-attributes-supported", + sizeof(printer_settable) / sizeof(printer_settable[0]), + NULL, printer_settable); + + /* server-is-sharing-printers */ + ippAddBoolean(CommonData, IPP_TAG_PRINTER, "server-is-sharing-printers", + BrowseLocalProtocols != 0 && Browsing); + + /* which-jobs-supported */ + ippAddStrings(CommonData, IPP_TAG_PRINTER, IPP_TAG_KEYWORD | IPP_TAG_COPY, + "which-jobs-supported", + sizeof(which_jobs) / sizeof(which_jobs[0]), NULL, which_jobs); +} + + +/* + * 'cupsdDeleteAllPrinters()' - Delete all printers from the system. + */ + +void +cupsdDeleteAllPrinters(void) +{ + cupsd_printer_t *p; /* Pointer to current printer/class */ + + + for (p = (cupsd_printer_t *)cupsArrayFirst(Printers); + p; + p = (cupsd_printer_t *)cupsArrayNext(Printers)) + { + p->op_policy_ptr = DefaultPolicyPtr; + cupsdDeletePrinter(p, 0); + } +} + + +/* + * 'cupsdDeletePrinter()' - Delete a printer from the system. + */ + +int /* O - 1 if classes affected, 0 otherwise */ +cupsdDeletePrinter( + cupsd_printer_t *p, /* I - Printer to delete */ + int update) /* I - Update printers.conf? */ +{ + int i, /* Looping var */ + changed = 0; /* Class changed? */ +#ifdef __sgi + char filename[1024]; /* Interface script filename */ +#endif /* __sgi */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdDeletePrinter(p=%p(%s), update=%d)", + p, p->name, update); + + /* + * Save the current position in the Printers array... + */ + + cupsArraySave(Printers); + + /* + * Stop printing on this printer... + */ + + cupsdSetPrinterState(p, IPP_PRINTER_STOPPED, update); + + p->state = IPP_PRINTER_STOPPED; /* Force for browsed printers */ + + if (p->job) + cupsdSetJobState(p->job, IPP_JOB_PENDING, CUPSD_JOB_FORCE, + update ? "Job stopped due to printer being deleted." : + "Job stopped."); + + /* + * Remove the printer from the list... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdDeletePrinter: Removing %s from Printers", p->name); + cupsArrayRemove(Printers, p); + + /* + * Remove the dummy interface/icon/option files under IRIX... + */ + +#ifdef __sgi + snprintf(filename, sizeof(filename), "/var/spool/lp/interface/%s", p->name); + unlink(filename); + + snprintf(filename, sizeof(filename), "/var/spool/lp/gui_interface/ELF/%s.gui", + p->name); + unlink(filename); + + snprintf(filename, sizeof(filename), "/var/spool/lp/activeicons/%s", p->name); + unlink(filename); + + snprintf(filename, sizeof(filename), "/var/spool/lp/pod/%s.config", p->name); + unlink(filename); + + snprintf(filename, sizeof(filename), "/var/spool/lp/pod/%s.status", p->name); + unlink(filename); + + snprintf(filename, sizeof(filename), "/var/spool/lp/member/%s", p->name); + unlink(filename); +#endif /* __sgi */ + + /* + * If p is the default printer, assign a different one... + */ + + if (p == DefaultPrinter) + DefaultPrinter = NULL; + + /* + * Remove this printer from any classes... + */ + + changed = cupsdDeletePrinterFromClasses(p); + + /* + * Deregister from any browse protocols... + */ + + cupsdDeregisterPrinter(p, 1); + + /* + * Free all memory used by the printer... + */ + + if (p->printers != NULL) + free(p->printers); + + delete_printer_filters(p); + + for (i = 0; i < p->num_reasons; i ++) + _cupsStrFree(p->reasons[i]); + + ippDelete(p->attrs); + ippDelete(p->ppd_attrs); + + mimeDeleteType(MimeDatabase, p->filetype); + mimeDeleteType(MimeDatabase, p->prefiltertype); + + cupsdFreeStrings(&(p->users)); + cupsdFreeQuotas(p); + + cupsdClearString(&p->uri); + cupsdClearString(&p->hostname); + cupsdClearString(&p->name); + cupsdClearString(&p->location); + cupsdClearString(&p->make_model); + cupsdClearString(&p->info); + cupsdClearString(&p->job_sheets[0]); + cupsdClearString(&p->job_sheets[1]); + cupsdClearString(&p->device_uri); + cupsdClearString(&p->sanitized_device_uri); + cupsdClearString(&p->port_monitor); + cupsdClearString(&p->op_policy); + cupsdClearString(&p->error_policy); + + cupsdClearString(&p->alert); + cupsdClearString(&p->alert_description); + +#ifdef HAVE_DNSSD + cupsdClearString(&p->pdl); +#endif /* HAVE_DNSSD */ + + cupsArrayDelete(p->filetypes); + + cupsFreeOptions(p->num_options, p->options); + + free(p); + + /* + * Restore the previous position in the Printers array... + */ + + cupsArrayRestore(Printers); + + return (changed); +} + + +/* + * 'cupsdFindDest()' - Find a destination in the list. + */ + +cupsd_printer_t * /* O - Destination in list */ +cupsdFindDest(const char *name) /* I - Name of printer or class to find */ +{ + cupsd_printer_t key; /* Search key */ + + + key.name = (char *)name; + return ((cupsd_printer_t *)cupsArrayFind(Printers, &key)); +} + + +/* + * 'cupsdFindPrinter()' - Find a printer in the list. + */ + +cupsd_printer_t * /* O - Printer in list */ +cupsdFindPrinter(const char *name) /* I - Name of printer to find */ +{ + cupsd_printer_t *p; /* Printer in list */ + + + if ((p = cupsdFindDest(name)) != NULL && (p->type & CUPS_PRINTER_CLASS)) + return (NULL); + else + return (p); +} + + +/* + * 'cupsdLoadAllPrinters()' - Load printers from the printers.conf file. + */ + +void +cupsdLoadAllPrinters(void) +{ + int i; /* Looping var */ + cups_file_t *fp; /* printers.conf file */ + int linenum; /* Current line number */ + char line[4096], /* Line from file */ + *value, /* Pointer to value */ + *valueptr; /* Pointer into value */ + cupsd_printer_t *p; /* Current printer */ + + + /* + * Open the printers.conf file... + */ + + snprintf(line, sizeof(line), "%s/printers.conf", ServerRoot); + if ((fp = cupsdOpenConfFile(line)) == NULL) + return; + + /* + * Read printer configurations until we hit EOF... + */ + + linenum = 0; + p = NULL; + + while (cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + { + /* + * Decode the directive... + */ + + if (!_cups_strcasecmp(line, " or + */ + + if (p == NULL && value) + { + /* + * Add the printer and a base file type... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, "Loading printer %s...", value); + + p = cupsdAddPrinter(value); + p->accepting = 1; + p->state = IPP_PRINTER_IDLE; + + /* + * Set the default printer as needed... + */ + + if (!_cups_strcasecmp(line, "")) + { + if (p != NULL) + { + /* + * Close out the current printer... + */ + + cupsdSetPrinterAttrs(p); + + if (strncmp(p->device_uri, "file:", 5) && + p->state != IPP_PRINTER_STOPPED) + { + /* + * See if the backend exists... + */ + + snprintf(line, sizeof(line), "%s/backend/%s", ServerBin, + p->device_uri); + + if ((valueptr = strchr(line + strlen(ServerBin), ':')) != NULL) + *valueptr = '\0'; /* Chop everything but URI scheme */ + + if (access(line, 0)) + { + /* + * Backend does not exist, stop printer... + */ + + p->state = IPP_PRINTER_STOPPED; + snprintf(p->state_message, sizeof(p->state_message), + "Backend %s does not exist!", line); + } + } + + p = NULL; + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + } + else if (!p) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "UUID")) + { + if (value && !strncmp(value, "urn:uuid:", 9)) + cupsdSetString(&(p->uuid), value); + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Bad UUID on line %d of printers.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "AuthInfoRequired")) + { + if (!cupsdSetAuthInfoRequired(p, value, NULL)) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Bad AuthInfoRequired on line %d of printers.conf.", + linenum); + } + else if (!_cups_strcasecmp(line, "Info")) + { + if (value) + cupsdSetString(&p->info, value); + } + else if (!_cups_strcasecmp(line, "MakeModel")) + { + if (value) + cupsdSetString(&p->make_model, value); + } + else if (!_cups_strcasecmp(line, "Location")) + { + if (value) + cupsdSetString(&p->location, value); + } + else if (!_cups_strcasecmp(line, "DeviceURI")) + { + if (value) + cupsdSetDeviceURI(p, value); + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "Option") && value) + { + /* + * Option name value + */ + + for (valueptr = value; *valueptr && !isspace(*valueptr & 255); valueptr ++); + + if (!*valueptr) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + else + { + for (; *valueptr && isspace(*valueptr & 255); *valueptr++ = '\0'); + + p->num_options = cupsAddOption(value, valueptr, p->num_options, + &(p->options)); + } + } + else if (!_cups_strcasecmp(line, "PortMonitor")) + { + if (value && strcmp(value, "none")) + cupsdSetString(&p->port_monitor, value); + else if (value) + cupsdClearString(&p->port_monitor); + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "Reason")) + { + if (value && + strcmp(value, "connecting-to-device") && + strcmp(value, "cups-insecure-filter-warning") && + strcmp(value, "cups-missing-filter-warning")) + { + for (i = 0 ; i < p->num_reasons; i ++) + if (!strcmp(value, p->reasons[i])) + break; + + if (i >= p->num_reasons && + p->num_reasons < (int)(sizeof(p->reasons) / sizeof(p->reasons[0]))) + { + p->reasons[p->num_reasons] = _cupsStrAlloc(value); + p->num_reasons ++; + } + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "State")) + { + /* + * Set the initial queue state... + */ + + if (value && !_cups_strcasecmp(value, "idle")) + p->state = IPP_PRINTER_IDLE; + else if (value && !_cups_strcasecmp(value, "stopped")) + { + p->state = IPP_PRINTER_STOPPED; + + for (i = 0 ; i < p->num_reasons; i ++) + if (!strcmp("paused", p->reasons[i])) + break; + + if (i >= p->num_reasons && + p->num_reasons < (int)(sizeof(p->reasons) / sizeof(p->reasons[0]))) + { + p->reasons[p->num_reasons] = _cupsStrAlloc("paused"); + p->num_reasons ++; + } + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "StateMessage")) + { + /* + * Set the initial queue state message... + */ + + if (value) + strlcpy(p->state_message, value, sizeof(p->state_message)); + } + else if (!_cups_strcasecmp(line, "StateTime")) + { + /* + * Set the state time... + */ + + if (value) + p->state_time = atoi(value); + } + else if (!_cups_strcasecmp(line, "Accepting")) + { + /* + * Set the initial accepting state... + */ + + if (value && + (!_cups_strcasecmp(value, "yes") || + !_cups_strcasecmp(value, "on") || + !_cups_strcasecmp(value, "true"))) + p->accepting = 1; + else if (value && + (!_cups_strcasecmp(value, "no") || + !_cups_strcasecmp(value, "off") || + !_cups_strcasecmp(value, "false"))) + p->accepting = 0; + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "Type")) + { + if (value) + p->type = atoi(value); + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "Shared")) + { + /* + * Set the initial shared state... + */ + + if (value && + (!_cups_strcasecmp(value, "yes") || + !_cups_strcasecmp(value, "on") || + !_cups_strcasecmp(value, "true"))) + p->shared = 1; + else if (value && + (!_cups_strcasecmp(value, "no") || + !_cups_strcasecmp(value, "off") || + !_cups_strcasecmp(value, "false"))) + p->shared = 0; + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "JobSheets")) + { + /* + * Set the initial job sheets... + */ + + if (value) + { + for (valueptr = value; *valueptr && !isspace(*valueptr & 255); valueptr ++); + + if (*valueptr) + *valueptr++ = '\0'; + + cupsdSetString(&p->job_sheets[0], value); + + while (isspace(*valueptr & 255)) + valueptr ++; + + if (*valueptr) + { + for (value = valueptr; *valueptr && !isspace(*valueptr & 255); valueptr ++); + + if (*valueptr) + *valueptr = '\0'; + + cupsdSetString(&p->job_sheets[1], value); + } + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "AllowUser")) + { + if (value) + { + p->deny_users = 0; + cupsdAddString(&(p->users), value); + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "DenyUser")) + { + if (value) + { + p->deny_users = 1; + cupsdAddString(&(p->users), value); + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "QuotaPeriod")) + { + if (value) + p->quota_period = atoi(value); + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "PageLimit")) + { + if (value) + p->page_limit = atoi(value); + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "KLimit")) + { + if (value) + p->k_limit = atoi(value); + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "OpPolicy")) + { + if (value) + { + cupsd_policy_t *pol; /* Policy */ + + + if ((pol = cupsdFindPolicy(value)) != NULL) + { + cupsdSetString(&p->op_policy, value); + p->op_policy_ptr = pol; + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Bad policy \"%s\" on line %d of printers.conf", + value, linenum); + } + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "ErrorPolicy")) + { + if (value) + cupsdSetString(&p->error_policy, value); + else + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + } + else if (!_cups_strcasecmp(line, "Attribute") && value) + { + for (valueptr = value; *valueptr && !isspace(*valueptr & 255); valueptr ++); + + if (!*valueptr) + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of printers.conf.", linenum); + else + { + for (; *valueptr && isspace(*valueptr & 255); *valueptr++ = '\0'); + + if (!p->attrs) + cupsdSetPrinterAttrs(p); + + if (!strcmp(value, "marker-change-time")) + p->marker_time = atoi(valueptr); + else + cupsdSetPrinterAttr(p, value, valueptr); + } + } + else if (_cups_strcasecmp(line, "Filter") && + _cups_strcasecmp(line, "Prefilter") && + _cups_strcasecmp(line, "Product")) + { + /* + * Something else we don't understand (and that wasn't used in a prior + * release of CUPS... + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unknown configuration directive %s on line %d of " + "printers.conf.", line, linenum); + } + } + + cupsFileClose(fp); +} + + +/* + * 'cupsdRenamePrinter()' - Rename a printer. + */ + +void +cupsdRenamePrinter( + cupsd_printer_t *p, /* I - Printer */ + const char *name) /* I - New name */ +{ + /* + * Remove the printer from the array(s) first... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdRenamePrinter: Removing %s from Printers", p->name); + cupsArrayRemove(Printers, p); + + /* + * Rename the printer type... + */ + + mimeDeleteType(MimeDatabase, p->filetype); + p->filetype = mimeAddType(MimeDatabase, "printer", name); + + if (p->prefiltertype) + { + mimeDeleteType(MimeDatabase, p->prefiltertype); + p->prefiltertype = mimeAddType(MimeDatabase, "prefilter", name); + } + + /* + * Rename the printer... + */ + + cupsdSetString(&p->name, name); + + /* + * Reset printer attributes... + */ + + cupsdSetPrinterAttrs(p); + + /* + * Add the printer back to the printer array(s)... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdRenamePrinter: Adding %s to Printers", p->name); + cupsArrayAdd(Printers, p); +} + + +/* + * 'cupsdSaveAllPrinters()' - Save all printer definitions to the printers.conf + * file. + */ + +void +cupsdSaveAllPrinters(void) +{ + int i; /* Looping var */ + cups_file_t *fp; /* printers.conf file */ + char filename[1024], /* printers.conf filename */ + temp[1024], /* Temporary string */ + value[2048], /* Value string */ + *ptr, /* Pointer into value */ + *name; /* Current user/group name */ + cupsd_printer_t *printer; /* Current printer class */ + time_t curtime; /* Current time */ + struct tm *curdate; /* Current date */ + cups_option_t *option; /* Current option */ + ipp_attribute_t *marker; /* Current marker attribute */ + + + /* + * Create the printers.conf file... + */ + + snprintf(filename, sizeof(filename), "%s/printers.conf", ServerRoot); + + if ((fp = cupsdCreateConfFile(filename, ConfigFilePerm & 0600)) == NULL) + return; + + cupsdLogMessage(CUPSD_LOG_INFO, "Saving printers.conf..."); + + /* + * Write a small header to the file... + */ + + curtime = time(NULL); + curdate = localtime(&curtime); + strftime(temp, sizeof(temp) - 1, "%Y-%m-%d %H:%M", curdate); + + cupsFilePuts(fp, "# Printer configuration file for " CUPS_SVERSION "\n"); + cupsFilePrintf(fp, "# Written by cupsd on %s\n", temp); + cupsFilePuts(fp, "# DO NOT EDIT THIS FILE WHEN CUPSD IS RUNNING\n"); + + /* + * Write each local printer known to the system... + */ + + for (printer = (cupsd_printer_t *)cupsArrayFirst(Printers); + printer; + printer = (cupsd_printer_t *)cupsArrayNext(Printers)) + { + /* + * Skip printer classes... + */ + + if (printer->type & CUPS_PRINTER_CLASS) + continue; + + /* + * Write printers as needed... + */ + + if (printer == DefaultPrinter) + cupsFilePrintf(fp, "\n", printer->name); + else + cupsFilePrintf(fp, "\n", printer->name); + + cupsFilePrintf(fp, "UUID %s\n", printer->uuid); + + if (printer->num_auth_info_required > 0) + { + switch (printer->num_auth_info_required) + { + case 1 : + strlcpy(value, printer->auth_info_required[0], sizeof(value)); + break; + + case 2 : + snprintf(value, sizeof(value), "%s,%s", + printer->auth_info_required[0], + printer->auth_info_required[1]); + break; + + case 3 : + default : + snprintf(value, sizeof(value), "%s,%s,%s", + printer->auth_info_required[0], + printer->auth_info_required[1], + printer->auth_info_required[2]); + break; + } + + cupsFilePutConf(fp, "AuthInfoRequired", value); + } + + if (printer->info) + cupsFilePutConf(fp, "Info", printer->info); + + if (printer->location) + cupsFilePutConf(fp, "Location", printer->location); + + if (printer->make_model) + cupsFilePutConf(fp, "MakeModel", printer->make_model); + + cupsFilePutConf(fp, "DeviceURI", printer->device_uri); + + if (printer->port_monitor) + cupsFilePutConf(fp, "PortMonitor", printer->port_monitor); + + if (printer->state == IPP_PRINTER_STOPPED) + { + cupsFilePuts(fp, "State Stopped\n"); + + if (printer->state_message) + cupsFilePutConf(fp, "StateMessage", printer->state_message); + } + else + cupsFilePuts(fp, "State Idle\n"); + + cupsFilePrintf(fp, "StateTime %d\n", (int)printer->state_time); + + for (i = 0; i < printer->num_reasons; i ++) + if (strcmp(printer->reasons[i], "connecting-to-device") && + strcmp(printer->reasons[i], "cups-insecure-filter-warning") && + strcmp(printer->reasons[i], "cups-missing-filter-warning")) + cupsFilePutConf(fp, "Reason", printer->reasons[i]); + + cupsFilePrintf(fp, "Type %d\n", printer->type); + + if (printer->accepting) + cupsFilePuts(fp, "Accepting Yes\n"); + else + cupsFilePuts(fp, "Accepting No\n"); + + if (printer->shared) + cupsFilePuts(fp, "Shared Yes\n"); + else + cupsFilePuts(fp, "Shared No\n"); + + snprintf(value, sizeof(value), "%s %s", printer->job_sheets[0], + printer->job_sheets[1]); + cupsFilePutConf(fp, "JobSheets", value); + + cupsFilePrintf(fp, "QuotaPeriod %d\n", printer->quota_period); + cupsFilePrintf(fp, "PageLimit %d\n", printer->page_limit); + cupsFilePrintf(fp, "KLimit %d\n", printer->k_limit); + + for (name = (char *)cupsArrayFirst(printer->users); + name; + name = (char *)cupsArrayNext(printer->users)) + cupsFilePutConf(fp, printer->deny_users ? "DenyUser" : "AllowUser", name); + + if (printer->op_policy) + cupsFilePutConf(fp, "OpPolicy", printer->op_policy); + if (printer->error_policy) + cupsFilePutConf(fp, "ErrorPolicy", printer->error_policy); + + for (i = printer->num_options, option = printer->options; + i > 0; + i --, option ++) + { + snprintf(value, sizeof(value), "%s %s", option->name, option->value); + cupsFilePutConf(fp, "Option", value); + } + + if ((marker = ippFindAttribute(printer->attrs, "marker-colors", + IPP_TAG_NAME)) != NULL) + { + snprintf(value, sizeof(value), "%s ", marker->name); + + for (i = 0, ptr = value + strlen(value); + i < marker->num_values && ptr < (value + sizeof(value) - 1); + i ++) + { + if (i) + *ptr++ = ','; + + strlcpy(ptr, marker->values[i].string.text, + value + sizeof(value) - ptr); + ptr += strlen(ptr); + } + + *ptr = '\0'; + cupsFilePutConf(fp, "Attribute", value); + } + + if ((marker = ippFindAttribute(printer->attrs, "marker-levels", + IPP_TAG_INTEGER)) != NULL) + { + cupsFilePrintf(fp, "Attribute %s %d", marker->name, + marker->values[0].integer); + for (i = 1; i < marker->num_values; i ++) + cupsFilePrintf(fp, ",%d", marker->values[i].integer); + cupsFilePuts(fp, "\n"); + } + + if ((marker = ippFindAttribute(printer->attrs, "marker-low-levels", + IPP_TAG_INTEGER)) != NULL) + { + cupsFilePrintf(fp, "Attribute %s %d", marker->name, + marker->values[0].integer); + for (i = 1; i < marker->num_values; i ++) + cupsFilePrintf(fp, ",%d", marker->values[i].integer); + cupsFilePuts(fp, "\n"); + } + + if ((marker = ippFindAttribute(printer->attrs, "marker-high-levels", + IPP_TAG_INTEGER)) != NULL) + { + cupsFilePrintf(fp, "Attribute %s %d", marker->name, + marker->values[0].integer); + for (i = 1; i < marker->num_values; i ++) + cupsFilePrintf(fp, ",%d", marker->values[i].integer); + cupsFilePuts(fp, "\n"); + } + + if ((marker = ippFindAttribute(printer->attrs, "marker-message", + IPP_TAG_TEXT)) != NULL) + { + snprintf(value, sizeof(value), "%s %s", marker->name, + marker->values[0].string.text); + + cupsFilePutConf(fp, "Attribute", value); + } + + if ((marker = ippFindAttribute(printer->attrs, "marker-names", + IPP_TAG_NAME)) != NULL) + { + snprintf(value, sizeof(value), "%s ", marker->name); + + for (i = 0, ptr = value + strlen(value); + i < marker->num_values && ptr < (value + sizeof(value) - 1); + i ++) + { + if (i) + *ptr++ = ','; + + strlcpy(ptr, marker->values[i].string.text, + value + sizeof(value) - ptr); + ptr += strlen(ptr); + } + + *ptr = '\0'; + cupsFilePutConf(fp, "Attribute", value); + } + + if ((marker = ippFindAttribute(printer->attrs, "marker-types", + IPP_TAG_KEYWORD)) != NULL) + { + snprintf(value, sizeof(value), "%s ", marker->name); + + for (i = 0, ptr = value + strlen(value); + i < marker->num_values && ptr < (value + sizeof(value) - 1); + i ++) + { + if (i) + *ptr++ = ','; + + strlcpy(ptr, marker->values[i].string.text, + value + sizeof(value) - ptr); + ptr += strlen(ptr); + } + + *ptr = '\0'; + cupsFilePutConf(fp, "Attribute", value); + } + + if (printer->marker_time) + cupsFilePrintf(fp, "Attribute marker-change-time %ld\n", + (long)printer->marker_time); + + cupsFilePuts(fp, "\n"); + +#ifdef __sgi + /* + * Make IRIX desktop & printer status happy + */ + + write_irix_state(printer); +#endif /* __sgi */ + } + + cupsdCloseCreatedConfFile(fp, filename); +} + + +/* + * 'cupsdSetAuthInfoRequired()' - Set the required authentication info. + */ + +int /* O - 1 if value OK, 0 otherwise */ +cupsdSetAuthInfoRequired( + cupsd_printer_t *p, /* I - Printer */ + const char *values, /* I - Plain text value (or NULL) */ + ipp_attribute_t *attr) /* I - IPP attribute value (or NULL) */ +{ + int i; /* Looping var */ + + + p->num_auth_info_required = 0; + + /* + * Do we have a plain text value? + */ + + if (values) + { + /* + * Yes, grab the keywords... + */ + + const char *end; /* End of current value */ + + + while (*values && p->num_auth_info_required < 4) + { + if ((end = strchr(values, ',')) == NULL) + end = values + strlen(values); + + if ((end - values) == 4 && !strncmp(values, "none", 4)) + { + if (p->num_auth_info_required != 0 || *end) + return (0); + + p->auth_info_required[p->num_auth_info_required] = "none"; + p->num_auth_info_required ++; + + return (1); + } + else if ((end - values) == 9 && !strncmp(values, "negotiate", 9)) + { + if (p->num_auth_info_required != 0 || *end) + return (0); + + p->auth_info_required[p->num_auth_info_required] = "negotiate"; + p->num_auth_info_required ++; + + /* + * Don't allow sharing of queues that require Kerberos authentication. + */ + + if (p->shared) + { + cupsdDeregisterPrinter(p, 1); + p->shared = 0; + } + } + else if ((end - values) == 6 && !strncmp(values, "domain", 6)) + { + p->auth_info_required[p->num_auth_info_required] = "domain"; + p->num_auth_info_required ++; + } + else if ((end - values) == 8 && !strncmp(values, "password", 8)) + { + p->auth_info_required[p->num_auth_info_required] = "password"; + p->num_auth_info_required ++; + } + else if ((end - values) == 8 && !strncmp(values, "username", 8)) + { + p->auth_info_required[p->num_auth_info_required] = "username"; + p->num_auth_info_required ++; + } + else + return (0); + + values = (*end) ? end + 1 : end; + } + + if (p->num_auth_info_required == 0) + { + p->auth_info_required[0] = "none"; + p->num_auth_info_required = 1; + } + + /* + * Update the printer-type value as needed... + */ + + if (p->num_auth_info_required > 1 || + strcmp(p->auth_info_required[0], "none")) + p->type |= CUPS_PRINTER_AUTHENTICATED; + else + p->type &= ~CUPS_PRINTER_AUTHENTICATED; + + return (1); + } + + /* + * Grab values from an attribute instead... + */ + + if (!attr || attr->num_values > 4) + return (0); + + for (i = 0; i < attr->num_values; i ++) + { + if (!strcmp(attr->values[i].string.text, "none")) + { + if (p->num_auth_info_required != 0 || attr->num_values != 1) + return (0); + + p->auth_info_required[p->num_auth_info_required] = "none"; + p->num_auth_info_required ++; + + return (1); + } + else if (!strcmp(attr->values[i].string.text, "negotiate")) + { + if (p->num_auth_info_required != 0 || attr->num_values != 1) + return (0); + + p->auth_info_required[p->num_auth_info_required] = "negotiate"; + p->num_auth_info_required ++; + + /* + * Don't allow sharing of queues that require Kerberos authentication. + */ + + if (p->shared) + { + cupsdDeregisterPrinter(p, 1); + p->shared = 0; + } + + return (1); + } + else if (!strcmp(attr->values[i].string.text, "domain")) + { + p->auth_info_required[p->num_auth_info_required] = "domain"; + p->num_auth_info_required ++; + } + else if (!strcmp(attr->values[i].string.text, "password")) + { + p->auth_info_required[p->num_auth_info_required] = "password"; + p->num_auth_info_required ++; + } + else if (!strcmp(attr->values[i].string.text, "username")) + { + p->auth_info_required[p->num_auth_info_required] = "username"; + p->num_auth_info_required ++; + } + else + return (0); + } + + return (1); +} + + +/* + * 'cupsdSetDeviceURI()' - Set the device URI for a printer. + */ + +void +cupsdSetDeviceURI(cupsd_printer_t *p, /* I - Printer */ + const char *uri) /* I - Device URI */ +{ + char buffer[1024], /* URI buffer */ + *start, /* Start of data after scheme */ + *slash, /* First slash after scheme:// */ + *ptr; /* Pointer into user@host:port part */ + + + /* + * Set the full device URI.. + */ + + cupsdSetString(&(p->device_uri), uri); + + /* + * Copy the device URI to a temporary buffer so we can sanitize any auth + * info in it... + */ + + strlcpy(buffer, uri, sizeof(buffer)); + + /* + * Find the end of the scheme:// part... + */ + + if ((ptr = strchr(buffer, ':')) != NULL) + { + for (start = ptr + 1; *start; start ++) + if (*start != '/') + break; + + /* + * Find the next slash (/) in the URI... + */ + + if ((slash = strchr(start, '/')) == NULL) + slash = start + strlen(start); /* No slash, point to the end */ + + /* + * Check for an @ sign before the slash... + */ + + if ((ptr = strchr(start, '@')) != NULL && ptr < slash) + { + /* + * Found an @ sign and it is before the resource part, so we have + * an authentication string. Copy the remaining URI over the + * authentication string... + */ + + _cups_strcpy(start, ptr + 1); + } + } + + /* + * Save the sanitized URI... + */ + + cupsdSetString(&(p->sanitized_device_uri), buffer); +} + + +/* + * 'cupsdSetPrinterAttr()' - Set a printer attribute. + */ + +void +cupsdSetPrinterAttr( + cupsd_printer_t *p, /* I - Printer */ + const char *name, /* I - Attribute name */ + char *value) /* I - Attribute value string */ +{ + ipp_attribute_t *attr; /* Attribute */ + int i, /* Looping var */ + count; /* Number of values */ + char *ptr; /* Pointer into value */ + ipp_tag_t value_tag; /* Value tag for this attribute */ + + + /* + * Don't allow empty values... + */ + + if (!*value && strcmp(name, "marker-message")) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Ignoring empty \"%s\" attribute", name); + return; + } + + /* + * Count the number of values... + */ + + for (count = 1, ptr = value; + (ptr = strchr(ptr, ',')) != NULL; + ptr ++, count ++); + + /* + * Then add or update the attribute as needed... + */ + + if (!strcmp(name, "marker-levels") || !strcmp(name, "marker-low-levels") || + !strcmp(name, "marker-high-levels")) + { + /* + * Integer values... + */ + + if ((attr = ippFindAttribute(p->attrs, name, IPP_TAG_INTEGER)) != NULL && + attr->num_values < count) + { + ippDeleteAttribute(p->attrs, attr); + attr = NULL; + } + + if (attr) + attr->num_values = count; + else + attr = ippAddIntegers(p->attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, name, + count, NULL); + + if (!attr) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate memory for printer attribute " + "(%d values)", count); + return; + } + + for (i = 0; i < count; i ++) + { + if ((ptr = strchr(value, ',')) != NULL) + *ptr++ = '\0'; + + attr->values[i].integer = strtol(value, NULL, 10); + + if (ptr) + value = ptr; + } + } + else + { + /* + * Name or keyword values... + */ + + if (!strcmp(name, "marker-types")) + value_tag = IPP_TAG_KEYWORD; + else if (!strcmp(name, "marker-message")) + value_tag = IPP_TAG_TEXT; + else + value_tag = IPP_TAG_NAME; + + if ((attr = ippFindAttribute(p->attrs, name, value_tag)) != NULL && + attr->num_values < count) + { + ippDeleteAttribute(p->attrs, attr); + attr = NULL; + } + + if (attr) + { + for (i = 0; i < attr->num_values; i ++) + _cupsStrFree(attr->values[i].string.text); + + attr->num_values = count; + } + else + attr = ippAddStrings(p->attrs, IPP_TAG_PRINTER, value_tag, name, + count, NULL, NULL); + + if (!attr) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to allocate memory for printer attribute " + "(%d values)", count); + return; + } + + for (i = 0; i < count; i ++) + { + if ((ptr = strchr(value, ',')) != NULL) + *ptr++ = '\0'; + + attr->values[i].string.text = _cupsStrAlloc(value); + + if (ptr) + value = ptr; + } + } +} + + +/* + * 'cupsdSetPrinterAttrs()' - Set printer attributes based upon the PPD file. + */ + +void +cupsdSetPrinterAttrs(cupsd_printer_t *p)/* I - Printer to setup */ +{ + int i; /* Looping var */ + char resource[HTTP_MAX_URI]; /* Resource portion of URI */ + int num_air; /* Number of auth-info-required values */ + const char * const *air; /* auth-info-required values */ + cupsd_location_t *auth; /* Pointer to authentication element */ + const char *auth_supported; /* Authentication supported */ + ipp_t *oldattrs; /* Old printer attributes */ + ipp_attribute_t *attr; /* Attribute data */ + char *name, /* Current user/group name */ + *filter; /* Current filter */ + static const char * const air_none[] = + { /* No authentication */ + "none" + }; + static const char * const air_userpass[] = + { /* Basic/Digest authentication */ + "username", + "password" + }; + + + DEBUG_printf(("cupsdSetPrinterAttrs: entering name = %s, type = %x\n", p->name, + p->type)); + + /* + * Make sure that we have the common attributes defined... + */ + + if (!CommonData) + cupsdCreateCommonData(); + + /* + * Clear out old filters, if any... + */ + + delete_printer_filters(p); + + /* + * Figure out the authentication that is required for the printer. + */ + + auth_supported = "requesting-user-name"; + num_air = 1; + air = air_none; + + if (p->num_auth_info_required > 0 && strcmp(p->auth_info_required[0], "none")) + { + num_air = p->num_auth_info_required; + air = p->auth_info_required; + } + else if (p->type & CUPS_PRINTER_AUTHENTICATED) + { + num_air = 2; + air = air_userpass; + } + + if (p->type & CUPS_PRINTER_CLASS) + snprintf(resource, sizeof(resource), "/classes/%s", p->name); + else + snprintf(resource, sizeof(resource), "/printers/%s", p->name); + + if ((auth = cupsdFindBest(resource, HTTP_POST)) == NULL || + auth->type == CUPSD_AUTH_NONE) + auth = cupsdFindPolicyOp(p->op_policy_ptr, IPP_PRINT_JOB); + + if (auth) + { + int auth_type; /* Authentication type */ + + + if ((auth_type = auth->type) == CUPSD_AUTH_DEFAULT) + auth_type = cupsdDefaultAuthType(); + + if (auth_type == CUPSD_AUTH_BASIC || auth_type == CUPSD_AUTH_BASICDIGEST) + auth_supported = "basic"; + else if (auth_type == CUPSD_AUTH_DIGEST) + auth_supported = "digest"; +#ifdef HAVE_GSSAPI + else if (auth_type == CUPSD_AUTH_NEGOTIATE) + auth_supported = "negotiate"; +#endif /* HAVE_GSSAPI */ + + if (auth_type != CUPSD_AUTH_NONE) + p->type |= CUPS_PRINTER_AUTHENTICATED; + else + p->type &= ~CUPS_PRINTER_AUTHENTICATED; + } + else + p->type &= ~CUPS_PRINTER_AUTHENTICATED; + + /* + * Create the required IPP attributes for a printer... + */ + + oldattrs = p->attrs; + p->attrs = ippNew(); + + ippAddString(p->attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "uri-authentication-supported", NULL, auth_supported); + ippAddString(p->attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "uri-security-supported", NULL, "none"); + ippAddString(p->attrs, IPP_TAG_PRINTER, IPP_TAG_NAME, "printer-name", NULL, + p->name); + ippAddString(p->attrs, IPP_TAG_PRINTER, IPP_TAG_TEXT, "printer-location", + NULL, p->location ? p->location : ""); + ippAddString(p->attrs, IPP_TAG_PRINTER, IPP_TAG_TEXT, "printer-info", + NULL, p->info ? p->info : ""); + ippAddString(p->attrs, IPP_TAG_PRINTER, IPP_TAG_URI, "printer-uuid", NULL, + p->uuid); + + if (cupsArrayCount(p->users) > 0) + { + if (p->deny_users) + attr = ippAddStrings(p->attrs, IPP_TAG_PRINTER, IPP_TAG_NAME, + "requesting-user-name-denied", + cupsArrayCount(p->users), NULL, NULL); + else + attr = ippAddStrings(p->attrs, IPP_TAG_PRINTER, IPP_TAG_NAME, + "requesting-user-name-allowed", + cupsArrayCount(p->users), NULL, NULL); + + for (i = 0, name = (char *)cupsArrayFirst(p->users); + name; + i ++, name = (char *)cupsArrayNext(p->users)) + attr->values[i].string.text = _cupsStrRetain(name); + } + + ippAddInteger(p->attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "job-quota-period", p->quota_period); + ippAddInteger(p->attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "job-k-limit", p->k_limit); + ippAddInteger(p->attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "job-page-limit", p->page_limit); + ippAddStrings(p->attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "auth-info-required", num_air, NULL, air); + + if (cupsArrayCount(Banners) > 0) + { + /* + * Setup the job-sheets-default attribute... + */ + + attr = ippAddStrings(p->attrs, IPP_TAG_PRINTER, IPP_TAG_NAME, + "job-sheets-default", 2, NULL, NULL); + + if (attr != NULL) + { + attr->values[0].string.text = _cupsStrAlloc(Classification ? + Classification : p->job_sheets[0]); + attr->values[1].string.text = _cupsStrAlloc(Classification ? + Classification : p->job_sheets[1]); + } + } + + p->raw = 0; + p->remote = 0; + + /* + * Assign additional attributes depending on whether this is a printer + * or class... + */ + + if (p->type & CUPS_PRINTER_CLASS) + { + p->raw = 1; + p->type &= ~CUPS_PRINTER_OPTIONS; + + /* + * Add class-specific attributes... + */ + + ippAddString(p->attrs, IPP_TAG_PRINTER, IPP_TAG_TEXT, + "printer-make-and-model", NULL, "Local Printer Class"); + ippAddString(p->attrs, IPP_TAG_PRINTER, IPP_TAG_URI, "device-uri", NULL, + "file:///dev/null"); + + if (p->num_printers > 0) + { + /* + * Add a list of member names; URIs are added in copy_printer_attrs... + */ + + attr = ippAddStrings(p->attrs, IPP_TAG_PRINTER, IPP_TAG_NAME, + "member-names", p->num_printers, NULL, NULL); + p->type |= CUPS_PRINTER_OPTIONS; + + for (i = 0; i < p->num_printers; i ++) + { + if (attr != NULL) + attr->values[i].string.text = _cupsStrRetain(p->printers[i]->name); + + p->type &= ~CUPS_PRINTER_OPTIONS | p->printers[i]->type; + } + } + } + else + { + /* + * Add printer-specific attributes... + */ + + ippAddString(p->attrs, IPP_TAG_PRINTER, IPP_TAG_URI, "device-uri", NULL, + p->sanitized_device_uri); + + /* + * Assign additional attributes from the PPD file (if any)... + */ + + load_ppd(p); + + /* + * Add filters for printer... + */ + + cupsdSetPrinterReasons(p, "-cups-missing-filter-warning," + "cups-insecure-filter-warning"); + + if (p->pc && p->pc->filters) + { + for (filter = (char *)cupsArrayFirst(p->pc->filters); + filter; + filter = (char *)cupsArrayNext(p->pc->filters)) + add_printer_filter(p, p->filetype, filter); + } + else if (!(p->type & CUPS_PRINTER_REMOTE)) + { + char interface[1024]; /* Interface script */ + + + snprintf(interface, sizeof(interface), "%s/interfaces/%s", ServerRoot, + p->name); + if (!access(interface, X_OK)) + { + /* + * Yes, we have a System V style interface script; use it! + */ + + snprintf(interface, sizeof(interface), "*/* 0 %s/interfaces/%s", + ServerRoot, p->name); + add_printer_filter(p, p->filetype, interface); + } + else + { + /* + * Add a filter from application/vnd.cups-raw to printer/name to + * handle "raw" printing by users. + */ + + add_printer_filter(p, p->filetype, "application/vnd.cups-raw 0 -"); + + /* + * Add a PostScript filter, since this is still possibly PS printer. + */ + + add_printer_filter(p, p->filetype, + "application/vnd.cups-postscript 0 -"); + } + } + + if (p->pc && p->pc->prefilters) + { + if (!p->prefiltertype) + p->prefiltertype = mimeAddType(MimeDatabase, "prefilter", p->name); + + for (filter = (char *)cupsArrayFirst(p->pc->prefilters); + filter; + filter = (char *)cupsArrayNext(p->pc->prefilters)) + add_printer_filter(p, p->prefiltertype, filter); + } + } + + /* + * Copy marker attributes as needed... + */ + + if (oldattrs) + { + ipp_attribute_t *oldattr; /* Old attribute */ + + + if ((oldattr = ippFindAttribute(oldattrs, "marker-colors", + IPP_TAG_NAME)) != NULL) + { + if ((attr = ippAddStrings(p->attrs, IPP_TAG_PRINTER, IPP_TAG_NAME, + "marker-colors", oldattr->num_values, NULL, + NULL)) != NULL) + { + for (i = 0; i < oldattr->num_values; i ++) + attr->values[i].string.text = + _cupsStrRetain(oldattr->values[i].string.text); + } + } + + if ((oldattr = ippFindAttribute(oldattrs, "marker-levels", + IPP_TAG_INTEGER)) != NULL) + { + if ((attr = ippAddIntegers(p->attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "marker-levels", oldattr->num_values, + NULL)) != NULL) + { + for (i = 0; i < oldattr->num_values; i ++) + attr->values[i].integer = oldattr->values[i].integer; + } + } + + if ((oldattr = ippFindAttribute(oldattrs, "marker-message", + IPP_TAG_TEXT)) != NULL) + ippAddString(p->attrs, IPP_TAG_PRINTER, IPP_TAG_TEXT, "marker-message", + NULL, oldattr->values[0].string.text); + + if ((oldattr = ippFindAttribute(oldattrs, "marker-low-levels", + IPP_TAG_INTEGER)) != NULL) + { + if ((attr = ippAddIntegers(p->attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "marker-low-levels", oldattr->num_values, + NULL)) != NULL) + { + for (i = 0; i < oldattr->num_values; i ++) + attr->values[i].integer = oldattr->values[i].integer; + } + } + + if ((oldattr = ippFindAttribute(oldattrs, "marker-high-levels", + IPP_TAG_INTEGER)) != NULL) + { + if ((attr = ippAddIntegers(p->attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "marker-high-levels", oldattr->num_values, + NULL)) != NULL) + { + for (i = 0; i < oldattr->num_values; i ++) + attr->values[i].integer = oldattr->values[i].integer; + } + } + + if ((oldattr = ippFindAttribute(oldattrs, "marker-names", + IPP_TAG_NAME)) != NULL) + { + if ((attr = ippAddStrings(p->attrs, IPP_TAG_PRINTER, IPP_TAG_NAME, + "marker-names", oldattr->num_values, NULL, + NULL)) != NULL) + { + for (i = 0; i < oldattr->num_values; i ++) + attr->values[i].string.text = + _cupsStrRetain(oldattr->values[i].string.text); + } + } + + if ((oldattr = ippFindAttribute(oldattrs, "marker-types", + IPP_TAG_KEYWORD)) != NULL) + { + if ((attr = ippAddStrings(p->attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "marker-types", oldattr->num_values, NULL, + NULL)) != NULL) + { + for (i = 0; i < oldattr->num_values; i ++) + attr->values[i].string.text = + _cupsStrRetain(oldattr->values[i].string.text); + } + } + + ippDelete(oldattrs); + } + + /* + * Force sharing off for remote queues... + */ + + if (p->type & CUPS_PRINTER_REMOTE) + p->shared = 0; + + /* + * Populate the document-format-supported attribute... + */ + + add_printer_formats(p); + + DEBUG_printf(("cupsdSetPrinterAttrs: leaving name = %s, type = %x\n", p->name, + p->type)); + + /* + * Add name-default attributes... + */ + + add_printer_defaults(p); + +#ifdef __sgi + /* + * Write the IRIX printer config and status files... + */ + + write_irix_config(p); + write_irix_state(p); +#endif /* __sgi */ + + /* + * Let the browse protocols reflect the change + */ + + cupsdRegisterPrinter(p); +} + + +/* + * 'cupsdSetPrinterReasons()' - Set/update the reasons strings. + */ + +int /* O - 1 if something changed, 0 otherwise */ +cupsdSetPrinterReasons( + cupsd_printer_t *p, /* I - Printer */ + const char *s) /* I - Reasons strings */ +{ + int i, /* Looping var */ + changed = 0; /* Did something change? */ + const char *sptr; /* Pointer into reasons */ + char reason[255], /* Reason string */ + *rptr; /* Pointer into reason */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdSetPrinterReasons(p=%p(%s),s=\"%s\"", p, p->name, s); + + if (s[0] == '-' || s[0] == '+') + { + /* + * Add/remove reasons... + */ + + sptr = s + 1; + } + else + { + /* + * Replace reasons... + */ + + sptr = s; + + for (i = 0; i < p->num_reasons; i ++) + _cupsStrFree(p->reasons[i]); + + p->num_reasons = 0; + changed = 1; + + dirty_printer(p); + } + + if (!strcmp(s, "none")) + return (changed); + + /* + * Loop through all of the reasons... + */ + + while (*sptr) + { + /* + * Skip leading whitespace and commas... + */ + + while (isspace(*sptr & 255) || *sptr == ',') + sptr ++; + + for (rptr = reason; *sptr && !isspace(*sptr & 255) && *sptr != ','; sptr ++) + if (rptr < (reason + sizeof(reason) - 1)) + *rptr++ = *sptr; + + if (rptr == reason) + break; + + *rptr = '\0'; + + if (s[0] == '-') + { + /* + * Remove reason... + */ + + for (i = 0; i < p->num_reasons; i ++) + if (!strcmp(reason, p->reasons[i])) + { + /* + * Found a match, so remove it... + */ + + p->num_reasons --; + changed = 1; + _cupsStrFree(p->reasons[i]); + + if (i < p->num_reasons) + memmove(p->reasons + i, p->reasons + i + 1, + (p->num_reasons - i) * sizeof(char *)); + + if (!strcmp(reason, "paused") && p->state == IPP_PRINTER_STOPPED) + cupsdSetPrinterState(p, IPP_PRINTER_IDLE, 1); + + if (strcmp(reason, "connecting-to-device")) + dirty_printer(p); + break; + } + } + else if (p->num_reasons < (int)(sizeof(p->reasons) / sizeof(p->reasons[0]))) + { + /* + * Add reason... + */ + + for (i = 0; i < p->num_reasons; i ++) + if (!strcmp(reason, p->reasons[i])) + break; + + if (i >= p->num_reasons) + { + if (!strncmp(reason, "cups-ipp-missing-", 17) || + !strncmp(reason, "cups-ipp-wrong-", 15)) + log_ipp_conformance(p, reason); + + if (i >= (int)(sizeof(p->reasons) / sizeof(p->reasons[0]))) + { + cupsdLogMessage(CUPSD_LOG_ALERT, + "Too many printer-state-reasons values for %s (%d)", + p->name, i + 1); + return (changed); + } + + p->reasons[i] = _cupsStrAlloc(reason); + p->num_reasons ++; + changed = 1; + + if (!strcmp(reason, "paused") && p->state != IPP_PRINTER_STOPPED) + cupsdSetPrinterState(p, IPP_PRINTER_STOPPED, 1); + + if (strcmp(reason, "connecting-to-device")) + dirty_printer(p); + } + } + } + + return (changed); +} + + +/* + * 'cupsdSetPrinterState()' - Update the current state of a printer. + */ + +void +cupsdSetPrinterState( + cupsd_printer_t *p, /* I - Printer to change */ + ipp_pstate_t s, /* I - New state */ + int update) /* I - Update printers.conf? */ +{ + cupsd_job_t *job; /* Current job */ + ipp_pstate_t old_state; /* Old printer state */ + static const char * const printer_states[] = + { /* State strings */ + "idle", + "processing", + "stopped" + }; + + + /* + * Set the new state... + */ + + old_state = p->state; + p->state = s; + + if (old_state != s) + { + cupsdAddEvent(s == IPP_PRINTER_STOPPED ? CUPSD_EVENT_PRINTER_STOPPED : + CUPSD_EVENT_PRINTER_STATE, p, NULL, + "%s \"%s\" state changed to %s.", + (p->type & CUPS_PRINTER_CLASS) ? "Class" : "Printer", + p->name, printer_states[p->state - IPP_PRINTER_IDLE]); + + /* + * Let the browse code know this needs to be updated... + */ + + p->state_time = time(NULL); + +#ifdef __sgi + write_irix_state(p); +#endif /* __sgi */ + } + + /* + * Set/clear the paused reason as needed... + */ + + if (s == IPP_PRINTER_STOPPED) + cupsdSetPrinterReasons(p, "+paused"); + else + cupsdSetPrinterReasons(p, "-paused"); + + if (old_state != s) + { + for (job = (cupsd_job_t *)cupsArrayFirst(ActiveJobs); + job; + job = (cupsd_job_t *)cupsArrayNext(ActiveJobs)) + if (job->reasons && job->state_value == IPP_JOB_PENDING && + !_cups_strcasecmp(job->dest, p->name)) + ippSetString(job->attrs, &job->reasons, 0, + s == IPP_PRINTER_STOPPED ? "printer-stopped" : "none"); + } + + /* + * Clear the message for the queue when going to processing... + */ + + if (s == IPP_PRINTER_PROCESSING) + p->state_message[0] = '\0'; + + /* + * Let the browse protocols reflect the change... + */ + + if (update) + cupsdRegisterPrinter(p); + + /* + * Save the printer configuration if a printer goes from idle or processing + * to stopped (or visa-versa)... + */ + + if (update && + (old_state == IPP_PRINTER_STOPPED) != (s == IPP_PRINTER_STOPPED)) + dirty_printer(p); +} + + +/* + * 'cupsdStopPrinter()' - Stop a printer from printing any jobs... + */ + +void +cupsdStopPrinter(cupsd_printer_t *p, /* I - Printer to stop */ + int update)/* I - Update printers.conf? */ +{ + /* + * Set the printer state... + */ + + cupsdSetPrinterState(p, IPP_PRINTER_STOPPED, update); + + /* + * See if we have a job printing on this printer... + */ + + if (p->job && p->job->state_value == IPP_JOB_PROCESSING) + cupsdSetJobState(p->job, IPP_JOB_PENDING, CUPSD_JOB_DEFAULT, + "Job stopped due to printer being paused."); +} + + +/* + * 'cupsdUpdatePrinterPPD()' - Update keywords in a printer's PPD file. + */ + +int /* O - 1 if successful, 0 otherwise */ +cupsdUpdatePrinterPPD( + cupsd_printer_t *p, /* I - Printer */ + int num_keywords, /* I - Number of keywords */ + cups_option_t *keywords) /* I - Keywords */ +{ + int i; /* Looping var */ + cups_file_t *src, /* Original file */ + *dst; /* New file */ + char srcfile[1024], /* Original filename */ + dstfile[1024], /* New filename */ + line[1024], /* Line from file */ + keystring[41]; /* Keyword from line */ + cups_option_t *keyword; /* Current keyword */ + + + cupsdLogMessage(CUPSD_LOG_INFO, "Updating keywords in PPD file for %s...", + p->name); + + /* + * Get the old and new PPD filenames... + */ + + snprintf(srcfile, sizeof(srcfile), "%s/ppd/%s.ppd.O", ServerRoot, p->name); + snprintf(dstfile, sizeof(srcfile), "%s/ppd/%s.ppd", ServerRoot, p->name); + + /* + * Rename the old file and open the old and new... + */ + + if (rename(dstfile, srcfile)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to backup PPD file for %s: %s", + p->name, strerror(errno)); + return (0); + } + + if ((src = cupsFileOpen(srcfile, "r")) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to open PPD file \"%s\": %s", + srcfile, strerror(errno)); + rename(srcfile, dstfile); + return (0); + } + + if ((dst = cupsFileOpen(dstfile, "w")) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to create PPD file \"%s\": %s", + dstfile, strerror(errno)); + cupsFileClose(src); + rename(srcfile, dstfile); + return (0); + } + + /* + * Copy the first line and then write out all of the keywords... + */ + + if (!cupsFileGets(src, line, sizeof(line))) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to read PPD file \"%s\": %s", + srcfile, strerror(errno)); + cupsFileClose(src); + cupsFileClose(dst); + rename(srcfile, dstfile); + return (0); + } + + cupsFilePrintf(dst, "%s\n", line); + + for (i = num_keywords, keyword = keywords; i > 0; i --, keyword ++) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, "*%s: %s", keyword->name, keyword->value); + cupsFilePrintf(dst, "*%s: %s\n", keyword->name, keyword->value); + } + + /* + * Then copy the rest of the PPD file, dropping any keywords we changed. + */ + + while (cupsFileGets(src, line, sizeof(line))) + { + /* + * Skip keywords we've already set... + */ + + if (sscanf(line, "*%40[^:]:", keystring) == 1 && + cupsGetOption(keystring, num_keywords, keywords)) + continue; + + /* + * Otherwise write the line... + */ + + cupsFilePrintf(dst, "%s\n", line); + } + + /* + * Close files and return... + */ + + cupsFileClose(src); + cupsFileClose(dst); + + return (1); +} + + +/* + * 'cupsdUpdatePrinters()' - Update printers after a partial reload. + */ + +void +cupsdUpdatePrinters(void) +{ + cupsd_printer_t *p; /* Current printer */ + + + /* + * Loop through the printers and recreate the printer attributes + * for any local printers since the policy and/or access control + * stuff may have changed. Also, if browsing is disabled, remove + * any remote printers... + */ + + for (p = (cupsd_printer_t *)cupsArrayFirst(Printers); + p; + p = (cupsd_printer_t *)cupsArrayNext(Printers)) + { + /* + * Update the operation policy pointer... + */ + + if ((p->op_policy_ptr = cupsdFindPolicy(p->op_policy)) == NULL) + p->op_policy_ptr = DefaultPolicyPtr; + + /* + * Update printer attributes... + */ + + cupsdSetPrinterAttrs(p); + } +} + + +/* + * 'cupsdValidateDest()' - Validate a printer/class destination. + */ + +const char * /* O - Printer or class name */ +cupsdValidateDest( + const char *uri, /* I - Printer URI */ + cups_ptype_t *dtype, /* O - Type (printer or class) */ + cupsd_printer_t **printer) /* O - Printer pointer */ +{ + cupsd_printer_t *p; /* Current printer */ + char localname[1024],/* Localized hostname */ + *lptr, /* Pointer into localized hostname */ + *sptr, /* Pointer into server name */ + *rptr, /* Pointer into resource */ + scheme[32], /* Scheme portion of URI */ + username[64], /* Username portion of URI */ + hostname[HTTP_MAX_HOST], + /* Host portion of URI */ + resource[HTTP_MAX_URI]; + /* Resource portion of URI */ + int port; /* Port portion of URI */ + + + DEBUG_printf(("cupsdValidateDest(uri=\"%s\", dtype=%p, printer=%p)\n", uri, + dtype, printer)); + + /* + * Initialize return values... + */ + + if (printer) + *printer = NULL; + + if (dtype) + *dtype = (cups_ptype_t)0; + + /* + * Pull the hostname and resource from the URI... + */ + + httpSeparateURI(HTTP_URI_CODING_ALL, uri, scheme, sizeof(scheme), + username, sizeof(username), hostname, sizeof(hostname), + &port, resource, sizeof(resource)); + + /* + * See if the resource is a class or printer... + */ + + if (!strncmp(resource, "/classes/", 9)) + { + /* + * Class... + */ + + rptr = resource + 9; + } + else if (!strncmp(resource, "/printers/", 10)) + { + /* + * Printer... + */ + + rptr = resource + 10; + } + else + { + /* + * Bad resource name... + */ + + return (NULL); + } + + /* + * See if the printer or class name exists... + */ + + p = cupsdFindDest(rptr); + + if (p == NULL && strchr(rptr, '@') == NULL) + return (NULL); + else if (p != NULL) + { + if (printer) + *printer = p; + + if (dtype) + *dtype = p->type & (CUPS_PRINTER_CLASS | CUPS_PRINTER_REMOTE); + + return (p->name); + } + + /* + * Change localhost to the server name... + */ + + if (!_cups_strcasecmp(hostname, "localhost")) + strlcpy(hostname, ServerName, sizeof(hostname)); + + strlcpy(localname, hostname, sizeof(localname)); + + if (!_cups_strcasecmp(hostname, ServerName)) + { + /* + * Localize the hostname... + */ + + lptr = strchr(localname, '.'); + sptr = strchr(ServerName, '.'); + + if (sptr != NULL && lptr != NULL) + { + /* + * Strip the common domain name components... + */ + + while (lptr != NULL) + { + if (!_cups_strcasecmp(lptr, sptr)) + { + *lptr = '\0'; + break; + } + else + lptr = strchr(lptr + 1, '.'); + } + } + } + + DEBUG_printf(("localized hostname is \"%s\"...\n", localname)); + + /* + * Find a matching printer or class... + */ + + for (p = (cupsd_printer_t *)cupsArrayFirst(Printers); + p; + p = (cupsd_printer_t *)cupsArrayNext(Printers)) + if (!_cups_strcasecmp(p->hostname, localname) && + !_cups_strcasecmp(p->name, rptr)) + { + if (printer) + *printer = p; + + if (dtype) + *dtype = p->type & (CUPS_PRINTER_CLASS | CUPS_PRINTER_REMOTE); + + return (p->name); + } + + return (NULL); +} + + +/* + * 'cupsdWritePrintcap()' - Write a pseudo-printcap file for older applications + * that need it... + */ + +void +cupsdWritePrintcap(void) +{ + int i; /* Looping var */ + cups_file_t *fp; /* Printcap file */ + cupsd_printer_t *p; /* Current printer */ + + +#ifdef __sgi + /* + * Update the IRIX printer state for the default printer; if + * no printers remain, then the default printer file will be + * removed... + */ + + write_irix_state(DefaultPrinter); +#endif /* __sgi */ + + /* + * See if we have a printcap file; if not, don't bother writing it. + */ + + if (!Printcap || !*Printcap) + return; + + cupsdLogMessage(CUPSD_LOG_INFO, "Generating printcap %s...", Printcap); + + /* + * Open the printcap file... + */ + + if ((fp = cupsFileOpen(Printcap, "w")) == NULL) + return; + + /* + * Put a comment header at the top so that users will know where the + * data has come from... + */ + + if (PrintcapFormat != PRINTCAP_PLIST) + cupsFilePrintf(fp, "# This file was automatically generated by cupsd(8) " + "from the\n" + "# %s/printers.conf file. All changes to this file\n" + "# will be lost.\n", ServerRoot); + + /* + * Write a new printcap with the current list of printers. + */ + + switch (PrintcapFormat) + { + case PRINTCAP_BSD : + /* + * Each printer is put in the file as: + * + * Printer1: + * Printer2: + * Printer3: + * ... + * PrinterN: + */ + + if (DefaultPrinter) + cupsFilePrintf(fp, "%s|%s:rm=%s:rp=%s:\n", DefaultPrinter->name, + DefaultPrinter->info, ServerName, + DefaultPrinter->name); + + for (p = (cupsd_printer_t *)cupsArrayFirst(Printers); + p; + p = (cupsd_printer_t *)cupsArrayNext(Printers)) + if (p != DefaultPrinter) + cupsFilePrintf(fp, "%s|%s:rm=%s:rp=%s:\n", p->name, p->info, + ServerName, p->name); + break; + + case PRINTCAP_PLIST : + /* + * Each printer is written as a dictionary in a plist file. + * Currently the printer-name, printer-info, printer-is-accepting-jobs, + * printer-location, printer-make-and-model, printer-state, + * printer-state-reasons, printer-type, and (sanitized) device-uri. + */ + + cupsFilePuts(fp, "\n" + "\n" + "\n" + "\n"); + + for (p = (cupsd_printer_t *)cupsArrayFirst(Printers); + p; + p = (cupsd_printer_t *)cupsArrayNext(Printers)) + { + cupsFilePuts(fp, "\t\n" + "\t\tprinter-name\n" + "\t\t"); + write_xml_string(fp, p->name); + cupsFilePuts(fp, "\n" + "\t\tprinter-info\n" + "\t\t"); + write_xml_string(fp, p->info); + cupsFilePrintf(fp, "\n" + "\t\tprinter-is-accepting-jobs\n" + "\t\t<%s/>\n" + "\t\tprinter-location\n" + "\t\t", p->accepting ? "true" : "false"); + write_xml_string(fp, p->location); + cupsFilePuts(fp, "\n" + "\t\tprinter-make-and-model\n" + "\t\t"); + write_xml_string(fp, p->make_model); + cupsFilePrintf(fp, "\n" + "\t\tprinter-state\n" + "\t\t%d\n" + "\t\tprinter-state-reasons\n" + "\t\t\n", p->state); + for (i = 0; i < p->num_reasons; i ++) + { + cupsFilePuts(fp, "\t\t\t"); + write_xml_string(fp, p->reasons[i]); + cupsFilePuts(fp, "\n"); + } + cupsFilePrintf(fp, "\t\t\n" + "\t\tprinter-type\n" + "\t\t%d\n" + "\t\tdevice-uri\n" + "\t\t", p->type); + write_xml_string(fp, p->sanitized_device_uri); + cupsFilePuts(fp, "\n" + "\t\n"); + } + cupsFilePuts(fp, "\n" + "\n"); + break; + + case PRINTCAP_SOLARIS : + /* + * Each printer is put in the file as: + * + * _all:all=Printer1,Printer2,Printer3,...,PrinterN + * _default:use=DefaultPrinter + * Printer1:\ + * :bsdaddr=ServerName,Printer1:\ + * :description=Description: + * Printer2: + * :bsdaddr=ServerName,Printer2:\ + * :description=Description: + * Printer3: + * :bsdaddr=ServerName,Printer3:\ + * :description=Description: + * ... + * PrinterN: + * :bsdaddr=ServerName,PrinterN:\ + * :description=Description: + */ + + cupsFilePuts(fp, "_all:all="); + for (p = (cupsd_printer_t *)cupsArrayFirst(Printers); + p; + p = (cupsd_printer_t *)cupsArrayCurrent(Printers)) + cupsFilePrintf(fp, "%s%c", p->name, + cupsArrayNext(Printers) ? ',' : '\n'); + + if (DefaultPrinter) + cupsFilePrintf(fp, "_default:use=%s\n", DefaultPrinter->name); + + for (p = (cupsd_printer_t *)cupsArrayFirst(Printers); + p; + p = (cupsd_printer_t *)cupsArrayNext(Printers)) + cupsFilePrintf(fp, "%s:\\\n" + "\t:bsdaddr=%s,%s:\\\n" + "\t:description=%s:\n", + p->name, ServerName, p->name, + p->info ? p->info : ""); + break; + } + + /* + * Close the file... + */ + + cupsFileClose(fp); +} + + +/* + * 'add_printer_defaults()' - Add name-default attributes to the printer attributes. + */ + +static void +add_printer_defaults(cupsd_printer_t *p)/* I - Printer */ +{ + int i; /* Looping var */ + int num_options; /* Number of default options */ + cups_option_t *options, /* Default options */ + *option; /* Current option */ + char name[256]; /* name-default */ + + + /* + * Maintain a common array of default attribute names... + */ + + if (!CommonDefaults) + { + CommonDefaults = cupsArrayNew((cups_array_func_t)strcmp, NULL); + + cupsArrayAdd(CommonDefaults, _cupsStrAlloc("copies-default")); + cupsArrayAdd(CommonDefaults, _cupsStrAlloc("document-format-default")); + cupsArrayAdd(CommonDefaults, _cupsStrAlloc("finishings-default")); + cupsArrayAdd(CommonDefaults, _cupsStrAlloc("job-hold-until-default")); + cupsArrayAdd(CommonDefaults, _cupsStrAlloc("job-priority-default")); + cupsArrayAdd(CommonDefaults, _cupsStrAlloc("job-sheets-default")); + cupsArrayAdd(CommonDefaults, _cupsStrAlloc("media-col-default")); + cupsArrayAdd(CommonDefaults, _cupsStrAlloc("number-up-default")); + cupsArrayAdd(CommonDefaults, + _cupsStrAlloc("orientation-requested-default")); + } + + /* + * Add all of the default options from the .conf files... + */ + + for (num_options = 0, options = NULL, i = p->num_options, option = p->options; + i > 0; + i --, option ++) + { + if (strcmp(option->name, "ipp-options") && + strcmp(option->name, "job-sheets") && + strcmp(option->name, "lease-duration")) + { + snprintf(name, sizeof(name), "%s-default", option->name); + num_options = cupsAddOption(name, option->value, num_options, &options); + + if (!cupsArrayFind(CommonDefaults, name)) + cupsArrayAdd(CommonDefaults, _cupsStrAlloc(name)); + } + } + + /* + * Convert options to IPP attributes... + */ + + cupsEncodeOptions2(p->attrs, num_options, options, IPP_TAG_PRINTER); + cupsFreeOptions(num_options, options); + + /* + * Add standard -default attributes as needed... + */ + + if (!cupsGetOption("copies", p->num_options, p->options)) + ippAddInteger(p->attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, "copies-default", + 1); + + if (!cupsGetOption("document-format", p->num_options, p->options)) + ippAddString(p->attrs, IPP_TAG_PRINTER, IPP_TAG_MIMETYPE, + "document-format-default", NULL, "application/octet-stream"); + + if (!cupsGetOption("job-hold-until", p->num_options, p->options)) + ippAddString(p->attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "job-hold-until-default", NULL, "no-hold"); + + if (!cupsGetOption("job-priority", p->num_options, p->options)) + ippAddInteger(p->attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "job-priority-default", 50); + + if (!cupsGetOption("number-up", p->num_options, p->options)) + ippAddInteger(p->attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "number-up-default", 1); + + if (!cupsGetOption("notify-lease-duration", p->num_options, p->options)) + ippAddInteger(p->attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "notify-lease-duration-default", DefaultLeaseDuration); + + if (!cupsGetOption("notify-events", p->num_options, p->options)) + ippAddString(p->attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "notify-events-default", NULL, "job-completed"); + + if (!cupsGetOption("orientation-requested", p->num_options, p->options)) + ippAddString(p->attrs, IPP_TAG_PRINTER, IPP_TAG_NOVALUE, + "orientation-requested-default", NULL, NULL); + + if (!cupsGetOption("print-quality", p->num_options, p->options)) + ippAddInteger(p->attrs, IPP_TAG_PRINTER, IPP_TAG_ENUM, + "print-quality-default", IPP_QUALITY_NORMAL); +} + + +/* + * 'add_printer_filter()' - Add a MIME filter for a printer. + */ + +static void +add_printer_filter( + cupsd_printer_t *p, /* I - Printer to add to */ + mime_type_t *filtertype, /* I - Filter or prefilter MIME type */ + const char *filter) /* I - Filter to add */ +{ + char super[MIME_MAX_SUPER], /* Super-type for filter */ + type[MIME_MAX_TYPE], /* Type for filter */ + dsuper[MIME_MAX_SUPER], /* Destination super-type for filter */ + dtype[MIME_MAX_TYPE], /* Destination type for filter */ + dest[MIME_MAX_SUPER + MIME_MAX_TYPE + 2], + /* Destination super/type */ + program[1024]; /* Program/filter name */ + int cost; /* Cost of filter */ + size_t maxsize = 0; /* Maximum supported file size */ + mime_type_t *temptype, /* MIME type looping var */ + *desttype; /* Destination MIME type */ + mime_filter_t *filterptr; /* MIME filter */ + char filename[1024]; /* Full filter filename */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "add_printer_filter(p=%p(%s), filtertype=%p(%s/%s), " + "filter=\"%s\")", p, p->name, filtertype, filtertype->super, + filtertype->type, filter); + + /* + * Parse the filter string; it should be in one of the following formats: + * + * source/type cost program + * source/type cost maxsize(nnnn) program + * source/type dest/type cost program + * source/type dest/type cost maxsize(nnnn) program + */ + + if (sscanf(filter, "%15[^/]/%255s%*[ \t]%15[^/]/%255s%d%*[ \t]%1023[^\n]", + super, type, dsuper, dtype, &cost, program) == 6) + { + snprintf(dest, sizeof(dest), "%s/%s/%s", p->name, dsuper, dtype); + + if ((desttype = mimeType(MimeDatabase, "printer", dest)) == NULL) + { + desttype = mimeAddType(MimeDatabase, "printer", dest); + if (!p->dest_types) + p->dest_types = cupsArrayNew(NULL, NULL); + + cupsArrayAdd(p->dest_types, desttype); + } + + } + else + { + if (sscanf(filter, "%15[^/]/%255s%d%*[ \t]%1023[^\n]", super, type, &cost, + program) == 4) + { + desttype = filtertype; + } + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, "%s: invalid filter string \"%s\"!", + p->name, filter); + return; + } + } + + if (!strncmp(program, "maxsize(", 8)) + { + char *ptr; /* Pointer into maxsize(nnnn) program */ + + maxsize = strtoll(program + 8, &ptr, 10); + + if (*ptr != ')') + { + cupsdLogMessage(CUPSD_LOG_ERROR, "%s: invalid filter string \"%s\"!", + p->name, filter); + return; + } + + ptr ++; + while (_cups_isspace(*ptr)) + ptr ++; + + _cups_strcpy(program, ptr); + } + + /* + * Check permissions on the filter and its containing directory... + */ + + if (strcmp(program, "-")) + { + if (program[0] == '/') + strlcpy(filename, program, sizeof(filename)); + else + snprintf(filename, sizeof(filename), "%s/filter/%s", ServerBin, program); + + _cupsFileCheck(filename, _CUPS_FILE_CHECK_PROGRAM, !RunUser, + cupsdLogFCMessage, p); + } + + /* + * Add the filter to the MIME database, supporting wildcards as needed... + */ + + for (temptype = mimeFirstType(MimeDatabase); + temptype; + temptype = mimeNextType(MimeDatabase)) + if (((super[0] == '*' && _cups_strcasecmp(temptype->super, "printer")) || + !_cups_strcasecmp(temptype->super, super)) && + (type[0] == '*' || !_cups_strcasecmp(temptype->type, type))) + { + if (desttype != filtertype) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "add_printer_filter: %s: adding filter %s/%s %s/%s %d " + "%s", p->name, temptype->super, temptype->type, + desttype->super, desttype->type, + cost, program); + filterptr = mimeAddFilter(MimeDatabase, temptype, desttype, cost, + program); + + if (!mimeFilterLookup(MimeDatabase, desttype, filtertype)) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "add_printer_filter: %s: adding filter %s/%s %s/%s " + "0 -", p->name, desttype->super, desttype->type, + filtertype->super, filtertype->type); + mimeAddFilter(MimeDatabase, desttype, filtertype, 0, "-"); + } + } + else + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "add_printer_filter: %s: adding filter %s/%s %s/%s %d " + "%s", p->name, temptype->super, temptype->type, + filtertype->super, filtertype->type, + cost, program); + filterptr = mimeAddFilter(MimeDatabase, temptype, filtertype, cost, + program); + } + + if (filterptr) + filterptr->maxsize = maxsize; + } +} + + +/* + * 'add_printer_formats()' - Add document-format-supported values for a printer. + */ + +static void +add_printer_formats(cupsd_printer_t *p) /* I - Printer */ +{ + int i; /* Looping var */ + mime_type_t *type; /* Current MIME type */ + cups_array_t *filters; /* Filters */ + ipp_attribute_t *attr; /* document-format-supported attribute */ + char mimetype[MIME_MAX_SUPER + MIME_MAX_TYPE + 2]; + /* MIME type name */ + + + /* + * Raw (and remote) queues advertise all of the supported MIME + * types... + */ + + cupsArrayDelete(p->filetypes); + p->filetypes = NULL; + + if (p->raw) + { + ippAddStrings(p->attrs, IPP_TAG_PRINTER, + (ipp_tag_t)(IPP_TAG_MIMETYPE | IPP_TAG_COPY), + "document-format-supported", NumMimeTypes, NULL, MimeTypes); + return; + } + + /* + * Otherwise, loop through the supported MIME types and see if there + * are filters for them... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "add_printer_formats: %d types, %d filters", + mimeNumTypes(MimeDatabase), mimeNumFilters(MimeDatabase)); + + p->filetypes = cupsArrayNew(NULL, NULL); + + for (type = mimeFirstType(MimeDatabase); + type; + type = mimeNextType(MimeDatabase)) + { + if (!_cups_strcasecmp(type->super, "printer")) + continue; + + snprintf(mimetype, sizeof(mimetype), "%s/%s", type->super, type->type); + + if ((filters = mimeFilter(MimeDatabase, type, p->filetype, NULL)) != NULL) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "add_printer_formats: %s: %s needs %d filters", + p->name, mimetype, cupsArrayCount(filters)); + + cupsArrayDelete(filters); + cupsArrayAdd(p->filetypes, type); + } + else + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "add_printer_formats: %s: %s not supported", + p->name, mimetype); + } + + /* + * Add the file formats that can be filtered... + */ + + if ((type = mimeType(MimeDatabase, "application", "octet-stream")) == NULL || + !cupsArrayFind(p->filetypes, type)) + i = 1; + else + i = 0; + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "add_printer_formats: %s: %d supported types", + p->name, cupsArrayCount(p->filetypes) + i); + + attr = ippAddStrings(p->attrs, IPP_TAG_PRINTER, IPP_TAG_MIMETYPE, + "document-format-supported", + cupsArrayCount(p->filetypes) + i, NULL, NULL); + + if (i) + attr->values[0].string.text = _cupsStrAlloc("application/octet-stream"); + + for (type = (mime_type_t *)cupsArrayFirst(p->filetypes); + type; + i ++, type = (mime_type_t *)cupsArrayNext(p->filetypes)) + { + snprintf(mimetype, sizeof(mimetype), "%s/%s", type->super, type->type); + + attr->values[i].string.text = _cupsStrAlloc(mimetype); + } + +#ifdef HAVE_DNSSD + { + char pdl[1024]; /* Buffer to build pdl list */ + mime_filter_t *filter; /* MIME filter looping var */ + + + /* + * We only support raw printing if this is not a Tioga PrintJobMgr based + * queue and if application/octet-stream is a known type... + */ + + for (filter = (mime_filter_t *)cupsArrayFirst(MimeDatabase->filters); + filter; + filter = (mime_filter_t *)cupsArrayNext(MimeDatabase->filters)) + { + if (filter->dst == p->filetype && filter->filter && + strstr(filter->filter, "PrintJobMgr")) + break; + } + + pdl[0] = '\0'; + + if (!filter && mimeType(MimeDatabase, "application", "octet-stream")) + strlcat(pdl, "application/octet-stream,", sizeof(pdl)); + + /* + * Then list a bunch of formats that are supported by the printer... + */ + + for (type = (mime_type_t *)cupsArrayFirst(p->filetypes); + type; + type = (mime_type_t *)cupsArrayNext(p->filetypes)) + { + if (!_cups_strcasecmp(type->super, "application")) + { + if (!_cups_strcasecmp(type->type, "pdf")) + strlcat(pdl, "application/pdf,", sizeof(pdl)); + else if (!_cups_strcasecmp(type->type, "postscript")) + strlcat(pdl, "application/postscript,", sizeof(pdl)); + } + else if (!_cups_strcasecmp(type->super, "image")) + { + if (!_cups_strcasecmp(type->type, "jpeg")) + strlcat(pdl, "image/jpeg,", sizeof(pdl)); + else if (!_cups_strcasecmp(type->type, "png")) + strlcat(pdl, "image/png,", sizeof(pdl)); + else if (!_cups_strcasecmp(type->type, "pwg-raster")) + strlcat(pdl, "image/pwg-raster,", sizeof(pdl)); + } + } + + if (pdl[0]) + pdl[strlen(pdl) - 1] = '\0'; /* Remove trailing comma */ + + cupsdSetString(&p->pdl, pdl); + } +#endif /* HAVE_DNSSD */ +} + + +/* + * 'compare_printers()' - Compare two printers. + */ + +static int /* O - Result of comparison */ +compare_printers(void *first, /* I - First printer */ + void *second, /* I - Second printer */ + void *data) /* I - App data (not used) */ +{ + (void)data; + + return (_cups_strcasecmp(((cupsd_printer_t *)first)->name, + ((cupsd_printer_t *)second)->name)); +} + + +/* + * 'delete_printer_filters()' - Delete all MIME filters for a printer. + */ + +static void +delete_printer_filters( + cupsd_printer_t *p) /* I - Printer to remove from */ +{ + mime_filter_t *filter; /* MIME filter looping var */ + mime_type_t *type; /* Destination types for filters */ + + + /* + * Range check input... + */ + + if (p == NULL) + return; + + /* + * Remove all filters from the MIME database that have a destination + * type == printer... + */ + + for (filter = mimeFirstFilter(MimeDatabase); + filter; + filter = mimeNextFilter(MimeDatabase)) + if (filter->dst == p->filetype || filter->dst == p->prefiltertype || + cupsArrayFind(p->dest_types, filter->dst)) + { + /* + * Delete the current filter... + */ + + mimeDeleteFilter(MimeDatabase, filter); + } + + for (type = (mime_type_t *)cupsArrayFirst(p->dest_types); + type; + type = (mime_type_t *)cupsArrayNext(p->dest_types)) + mimeDeleteType(MimeDatabase, type); + + cupsArrayDelete(p->dest_types); + p->dest_types = NULL; + + cupsdSetPrinterReasons(p, "-cups-insecure-filter-warning" + ",cups-missing-filter-warning"); +} + + +/* + * 'dirty_printer()' - Mark config and state files dirty for the specified + * printer. + */ + +static void +dirty_printer(cupsd_printer_t *p) /* I - Printer */ +{ + if (p->type & CUPS_PRINTER_CLASS) + cupsdMarkDirty(CUPSD_DIRTY_CLASSES); + else + cupsdMarkDirty(CUPSD_DIRTY_PRINTERS); + + if (PrintcapFormat == PRINTCAP_PLIST) + cupsdMarkDirty(CUPSD_DIRTY_PRINTCAP); +} + + +/* + * 'load_ppd()' - Load a cached PPD file, updating the cache as needed. + */ + +static void +load_ppd(cupsd_printer_t *p) /* I - Printer */ +{ + int i, j, k; /* Looping vars */ + char cache_name[1024]; /* Cache filename */ + struct stat cache_info; /* Cache file info */ + ppd_file_t *ppd; /* PPD file */ + char ppd_name[1024]; /* PPD filename */ + struct stat ppd_info; /* PPD file info */ + int num_media; /* Number of media options */ + ppd_size_t *size; /* Current PPD size */ + ppd_option_t *duplex, /* Duplex option */ + *output_bin, /* OutputBin option */ + *output_mode, /* OutputMode option */ + *resolution; /* (Set|JCL|)Resolution option */ + ppd_choice_t *choice, /* Current PPD choice */ + *input_slot, /* Current input slot */ + *media_type; /* Current media type */ + ppd_attr_t *ppd_attr; /* PPD attribute */ + int xdpi, /* Horizontal resolution */ + ydpi; /* Vertical resolution */ + const char *resptr; /* Pointer into resolution keyword */ + _pwg_size_t *pwgsize; /* Current PWG size */ + _pwg_map_t *pwgsource, /* Current PWG source */ + *pwgtype; /* Current PWG type */ + ipp_attribute_t *attr; /* Attribute data */ + _ipp_value_t *val; /* Attribute value */ + int num_finishings, /* Number of finishings */ + finishings[5]; /* finishings-supported values */ + int num_qualities, /* Number of print-quality values */ + qualities[3]; /* print-quality values */ + int num_margins, /* Number of media-*-margin-supported values */ + margins[16]; /* media-*-margin-supported values */ + const char *filter; /* Current filter */ + static const char * const sides[3] = /* sides-supported values */ + { + "one-sided", + "two-sided-long-edge", + "two-sided-short-edge" + }; + static const char * const standard_commands[] = + { /* Standard CUPS commands */ + "AutoConfigure", + "Clean", + "PrintSelfTestPage" + }; + + + /* + * Check to see if the cache is up-to-date... + */ + + snprintf(cache_name, sizeof(cache_name), "%s/%s.data", CacheDir, p->name); + if (stat(cache_name, &cache_info)) + cache_info.st_mtime = 0; + + snprintf(ppd_name, sizeof(ppd_name), "%s/ppd/%s.ppd", ServerRoot, p->name); + if (stat(ppd_name, &ppd_info)) + ppd_info.st_mtime = 1; + + ippDelete(p->ppd_attrs); + p->ppd_attrs = NULL; + + _ppdCacheDestroy(p->pc); + p->pc = NULL; + + if (cache_info.st_mtime >= ppd_info.st_mtime) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, "load_ppd: Loading %s...", cache_name); + + if ((p->pc = _ppdCacheCreateWithFile(cache_name, &p->ppd_attrs)) != NULL && + p->ppd_attrs) + { + /* + * Loaded successfully! + */ + + return; + } + } + + /* + * Reload PPD attributes from disk... + */ + + cupsdMarkDirty(CUPSD_DIRTY_PRINTERS); + + cupsdLogMessage(CUPSD_LOG_DEBUG, "load_ppd: Loading %s...", ppd_name); + + p->type &= ~CUPS_PRINTER_OPTIONS; + p->type |= CUPS_PRINTER_BW; + + finishings[0] = IPP_FINISHINGS_NONE; + num_finishings = 1; + + p->ppd_attrs = ippNew(); + + if ((ppd = _ppdOpenFile(ppd_name, _PPD_LOCALIZATION_NONE)) != NULL) + { + /* + * Add make/model and other various attributes... + */ + + p->pc = _ppdCacheCreateWithPPD(ppd); + + ppdMarkDefaults(ppd); + + if (ppd->color_device) + p->type |= CUPS_PRINTER_COLOR; + if (ppd->variable_sizes) + p->type |= CUPS_PRINTER_VARIABLE; + if (!ppd->manual_copies) + p->type |= CUPS_PRINTER_COPIES; + if ((ppd_attr = ppdFindAttr(ppd, "cupsFax", NULL)) != NULL) + if (ppd_attr->value && !_cups_strcasecmp(ppd_attr->value, "true")) + p->type |= CUPS_PRINTER_FAX; + + ippAddBoolean(p->ppd_attrs, IPP_TAG_PRINTER, "color-supported", + ppd->color_device); + if (ppd->throughput) + { + ippAddInteger(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "pages-per-minute", ppd->throughput); + if (ppd->color_device) + ippAddInteger(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "pages-per-minute-color", ppd->throughput); + } + else + { + /* + * When there is no speed information, just say "1 page per minute". + */ + + ippAddInteger(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "pages-per-minute", 1); + if (ppd->color_device) + ippAddInteger(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "pages-per-minute-color", 1); + } + + num_qualities = 0; + + if ((output_mode = ppdFindOption(ppd, "OutputMode")) != NULL) + { + if (ppdFindChoice(output_mode, "draft") || + ppdFindChoice(output_mode, "fast")) + qualities[num_qualities ++] = IPP_QUALITY_DRAFT; + + qualities[num_qualities ++] = IPP_QUALITY_NORMAL; + + if (ppdFindChoice(output_mode, "best") || + ppdFindChoice(output_mode, "high")) + qualities[num_qualities ++] = IPP_QUALITY_HIGH; + } + else if ((ppd_attr = ppdFindAttr(ppd, "APPrinterPreset", NULL)) != NULL) + { + do + { + if (strstr(ppd_attr->spec, "draft") || + strstr(ppd_attr->spec, "Draft")) + { + qualities[num_qualities ++] = IPP_QUALITY_DRAFT; + break; + } + } + while ((ppd_attr = ppdFindNextAttr(ppd, "APPrinterPreset", + NULL)) != NULL); + + qualities[num_qualities ++] = IPP_QUALITY_NORMAL; + qualities[num_qualities ++] = IPP_QUALITY_HIGH; + } + else + qualities[num_qualities ++] = IPP_QUALITY_NORMAL; + + ippAddIntegers(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_ENUM, + "print-quality-supported", num_qualities, qualities); + + if (ppd->nickname) + { + /* + * The NickName can be localized in the character set specified + * by the LanugageEncoding attribute. However, ppdOpen2() has + * already converted the ppd->nickname member to UTF-8 for us + * (the original attribute value is available separately) + */ + + cupsdSetString(&p->make_model, ppd->nickname); + } + else if (ppd->modelname) + { + /* + * Model name can only contain specific characters... + */ + + cupsdSetString(&p->make_model, ppd->modelname); + } + else + cupsdSetString(&p->make_model, "Bad PPD File"); + + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_TEXT, + "printer-make-and-model", NULL, p->make_model); + + /* + * Add media options from the PPD file... + */ + + if (ppd->num_sizes == 0 || !p->pc) + { + if (!ppdFindAttr(ppd, "APScannerOnly", NULL)) + cupsdLogMessage(CUPSD_LOG_CRIT, + "The PPD file for printer %s contains no media " + "options and is therefore invalid!", p->name); + + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "media-default", NULL, "unknown"); + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "media-supported", NULL, "unknown"); + } + else + { + /* + * media-default + */ + + if ((size = ppdPageSize(ppd, NULL)) != NULL) + pwgsize = _ppdCacheGetSize(p->pc, size->name); + else + pwgsize = NULL; + + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "media-default", NULL, + pwgsize ? pwgsize->map.pwg : "unknown"); + + /* + * media-col-default + */ + + if (pwgsize) + { + ipp_t *col; /* Collection value */ + + input_slot = ppdFindMarkedChoice(ppd, "InputSlot"); + media_type = ppdFindMarkedChoice(ppd, "MediaType"); + col = new_media_col(pwgsize, + input_slot ? + _ppdCacheGetSource(p->pc, + input_slot->choice) : + NULL, + media_type ? + _ppdCacheGetType(p->pc, + media_type->choice) : + NULL); + + ippAddCollection(p->ppd_attrs, IPP_TAG_PRINTER, "media-col-default", + col); + ippDelete(col); + } + + /* + * media-supported + */ + + num_media = p->pc->num_sizes; + if (p->pc->custom_min_keyword) + num_media += 2; + + if ((attr = ippAddStrings(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "media-supported", num_media, NULL, + NULL)) != NULL) + { + val = attr->values; + + for (i = p->pc->num_sizes, pwgsize = p->pc->sizes; + i > 0; + i --, pwgsize ++, val ++) + val->string.text = _cupsStrRetain(pwgsize->map.pwg); + + if (p->pc->custom_min_keyword) + { + val->string.text = _cupsStrRetain(p->pc->custom_min_keyword); + val ++; + val->string.text = _cupsStrRetain(p->pc->custom_max_keyword); + } + } + + /* + * media-source-supported + */ + + if (p->pc->num_sources > 0 && + (attr = ippAddStrings(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "media-source-supported", p->pc->num_sources, + NULL, NULL)) != NULL) + { + for (i = p->pc->num_sources, pwgsource = p->pc->sources, + val = attr->values; + i > 0; + i --, pwgsource ++, val ++) + val->string.text = _cupsStrRetain(pwgsource->pwg); + } + + /* + * media-type-supported + */ + + if (p->pc->num_types > 0 && + (attr = ippAddStrings(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "media-type-supported", p->pc->num_types, + NULL, NULL)) != NULL) + { + for (i = p->pc->num_types, pwgtype = p->pc->types, + val = attr->values; + i > 0; + i --, pwgtype ++, val ++) + val->string.text = _cupsStrRetain(pwgtype->pwg); + } + + /* + * media-*-margin-supported + */ + + for (i = p->pc->num_sizes, pwgsize = p->pc->sizes, num_margins = 0; + i > 0 && num_margins < (int)(sizeof(margins) / sizeof(margins[0])); + i --, pwgsize ++) + { + for (j = 0; j < num_margins; j ++) + if (pwgsize->bottom == margins[j]) + break; + + if (j >= num_margins) + { + margins[num_margins] = pwgsize->bottom; + num_margins ++; + } + } + + if (num_margins > 0) + ippAddIntegers(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "media-bottom-margin-supported", num_margins, margins); + else + ippAddInteger(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "media-bottom-margin-supported", 0); + + for (i = p->pc->num_sizes, pwgsize = p->pc->sizes, num_margins = 0; + i > 0 && num_margins < (int)(sizeof(margins) / sizeof(margins[0])); + i --, pwgsize ++) + { + for (j = 0; j < num_margins; j ++) + if (pwgsize->left == margins[j]) + break; + + if (j >= num_margins) + { + margins[num_margins] = pwgsize->left; + num_margins ++; + } + } + + if (num_margins > 0) + ippAddIntegers(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "media-left-margin-supported", num_margins, margins); + else + ippAddInteger(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "media-left-margin-supported", 0); + + for (i = p->pc->num_sizes, pwgsize = p->pc->sizes, num_margins = 0; + i > 0 && num_margins < (int)(sizeof(margins) / sizeof(margins[0])); + i --, pwgsize ++) + { + for (j = 0; j < num_margins; j ++) + if (pwgsize->right == margins[j]) + break; + + if (j >= num_margins) + { + margins[num_margins] = pwgsize->right; + num_margins ++; + } + } + + if (num_margins > 0) + ippAddIntegers(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "media-right-margin-supported", num_margins, margins); + else + ippAddInteger(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "media-right-margin-supported", 0); + + for (i = p->pc->num_sizes, pwgsize = p->pc->sizes, num_margins = 0; + i > 0 && num_margins < (int)(sizeof(margins) / sizeof(margins[0])); + i --, pwgsize ++) + { + for (j = 0; j < num_margins; j ++) + if (pwgsize->top == margins[j]) + break; + + if (j >= num_margins) + { + margins[num_margins] = pwgsize->top; + num_margins ++; + } + } + + if (num_margins > 0) + ippAddIntegers(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "media-top-margin-supported", num_margins, margins); + else + ippAddInteger(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "media-top-margin-supported", 0); + + /* + * media-col-database + */ + + num_media = p->pc->num_sizes; + if (p->pc->num_sources) + { + if (p->pc->num_types > 0) + num_media += p->pc->num_sizes * p->pc->num_sources * + p->pc->num_types; + else + num_media += p->pc->num_sizes * p->pc->num_sources; + } + else if (p->pc->num_types) + num_media += p->pc->num_sizes * p->pc->num_types; + + if ((attr = ippAddCollections(p->ppd_attrs, IPP_TAG_PRINTER, + "media-col-database", num_media, + NULL)) != NULL) + { + for (i = p->pc->num_sizes, pwgsize = p->pc->sizes, val = attr->values; + i > 0; + i --, pwgsize ++) + { + /* + * Start by adding the page size without source or type... + */ + + ppdMarkOption(ppd, "PageSize", pwgsize->map.ppd); + + val->collection = new_media_col(pwgsize, NULL, NULL); + val ++; + + /* + * Then add the specific, supported combinations of size, source, and + * type... + */ + + if (p->pc->num_sources > 0) + { + for (j = p->pc->num_sources, pwgsource = p->pc->sources; + j > 0; + j --, pwgsource ++) + { + ppdMarkOption(ppd, "InputSlot", pwgsource->ppd); + + if (p->pc->num_types > 0) + { + for (k = p->pc->num_types, pwgtype = p->pc->types; + k > 0; + k --, pwgtype ++) + { + if (!ppdMarkOption(ppd, "MediaType", pwgtype->ppd)) + { + val->collection = new_media_col(pwgsize, pwgsource->pwg, + pwgtype->pwg); + val ++; + } + } + } + else if (!ppdConflicts(ppd)) + { + val->collection = new_media_col(pwgsize, pwgsource->pwg, NULL); + val ++; + } + } + } + else if (p->pc->num_types > 0) + { + for (j = p->pc->num_types, pwgtype = p->pc->types; + j > 0; + j --, pwgtype ++) + { + if (!ppdMarkOption(ppd, "MediaType", pwgtype->ppd)) + { + val->collection = new_media_col(pwgsize, NULL, pwgtype->pwg); + val ++; + } + } + } + } + + /* + * Update the number of media-col-database values... + */ + + attr->num_values = val - attr->values; + } + } + + /* + * Output bin... + */ + + if (p->pc && p->pc->num_bins > 0) + { + attr = ippAddStrings(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "output-bin-supported", p->pc->num_bins, + NULL, NULL); + + if (attr != NULL) + { + for (i = 0, val = attr->values; + i < p->pc->num_bins; + i ++, val ++) + val->string.text = _cupsStrAlloc(p->pc->bins[i].pwg); + } + + if ((output_bin = ppdFindOption(ppd, "OutputBin")) != NULL) + { + for (i = 0; i < p->pc->num_bins; i ++) + if (!strcmp(p->pc->bins[i].ppd, output_bin->defchoice)) + break; + + if (i >= p->pc->num_bins) + i = 0; + + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "output-bin-default", NULL, p->pc->bins[i].pwg); + } + else + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "output-bin-default", NULL, p->pc->bins[0].pwg); + } + else if (((ppd_attr = ppdFindAttr(ppd, "DefaultOutputOrder", + NULL)) != NULL && + !_cups_strcasecmp(ppd_attr->value, "Reverse")) || + (!ppd_attr && ppd->manufacturer && /* "Compatibility heuristic" */ + (!_cups_strcasecmp(ppd->manufacturer, "epson") || + !_cups_strcasecmp(ppd->manufacturer, "lexmark")))) + { + /* + * Report that this printer has a single output bin that leaves pages face + * up. + */ + + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "output-bin-supported", NULL, "face-up"); + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "output-bin-default", NULL, "face-up"); + } + else + { + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "output-bin-supported", NULL, "face-down"); + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "output-bin-default", NULL, "face-down"); + } + + /* + * output-mode and print-color-mode... + */ + + if (ppd->color_device) + { + static const char * const output_modes[] = + { + "monochrome", + "color" + }; + + ippAddStrings(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "output-mode-supported", 2, NULL, output_modes); + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "output-mode-default", NULL, "color"); + + ippAddStrings(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "print-color-mode-supported", 2, NULL, output_modes); + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "print-color-mode-default", NULL, "color"); + } + else + { + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "output-mode-supported", NULL, "monochrome"); + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "output-mode-default", NULL, "monochrome"); + + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "print-color-mode-supported", NULL, "monochrome"); + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "print-color-mode-default", NULL, "monochrome"); + } + + /* + * Printer resolutions... + */ + + if ((resolution = ppdFindOption(ppd, "Resolution")) == NULL) + if ((resolution = ppdFindOption(ppd, "JCLResolution")) == NULL) + if ((resolution = ppdFindOption(ppd, "SetResolution")) == NULL) + resolution = ppdFindOption(ppd, "CNRes_PGP"); + + if (resolution) + { + /* + * Report all supported resolutions... + */ + + attr = ippAddResolutions(p->ppd_attrs, IPP_TAG_PRINTER, + "printer-resolution-supported", + resolution->num_choices, IPP_RES_PER_INCH, + NULL, NULL); + + for (i = 0, choice = resolution->choices; + i < resolution->num_choices; + i ++, choice ++) + { + xdpi = ydpi = (int)strtol(choice->choice, (char **)&resptr, 10); + if (resptr > choice->choice && xdpi > 0 && *resptr == 'x') + ydpi = (int)strtol(resptr + 1, (char **)&resptr, 10); + + if (xdpi <= 0 || ydpi <= 0) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "Bad resolution \"%s\" for printer %s.", + choice->choice, p->name); + xdpi = ydpi = 300; + } + + attr->values[i].resolution.xres = xdpi; + attr->values[i].resolution.yres = ydpi; + attr->values[i].resolution.units = IPP_RES_PER_INCH; + + if (choice->marked) + ippAddResolution(p->ppd_attrs, IPP_TAG_PRINTER, + "printer-resolution-default", IPP_RES_PER_INCH, + xdpi, ydpi); + } + } + else if ((ppd_attr = ppdFindAttr(ppd, "DefaultResolution", NULL)) != NULL && + ppd_attr->value) + { + /* + * Just the DefaultResolution to report... + */ + + xdpi = ydpi = (int)strtol(ppd_attr->value, (char **)&resptr, 10); + if (resptr > ppd_attr->value && xdpi > 0) + { + if (*resptr == 'x') + ydpi = (int)strtol(resptr + 1, (char **)&resptr, 10); + else + ydpi = xdpi; + } + + if (xdpi <= 0 || ydpi <= 0) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "Bad default resolution \"%s\" for printer %s.", + ppd_attr->value, p->name); + xdpi = ydpi = 300; + } + + ippAddResolution(p->ppd_attrs, IPP_TAG_PRINTER, + "printer-resolution-default", IPP_RES_PER_INCH, + xdpi, ydpi); + ippAddResolution(p->ppd_attrs, IPP_TAG_PRINTER, + "printer-resolution-supported", IPP_RES_PER_INCH, + xdpi, ydpi); + } + else + { + /* + * No resolutions in PPD - make one up... + */ + + ippAddResolution(p->ppd_attrs, IPP_TAG_PRINTER, + "printer-resolution-default", IPP_RES_PER_INCH, + 300, 300); + ippAddResolution(p->ppd_attrs, IPP_TAG_PRINTER, + "printer-resolution-supported", IPP_RES_PER_INCH, + 300, 300); + } + + /* + * Duplexing, etc... + */ + + ppdMarkDefaults(ppd); + + if ((duplex = ppdFindOption(ppd, "Duplex")) == NULL) + if ((duplex = ppdFindOption(ppd, "EFDuplex")) == NULL) + if ((duplex = ppdFindOption(ppd, "EFDuplexing")) == NULL) + if ((duplex = ppdFindOption(ppd, "KD03Duplex")) == NULL) + duplex = ppdFindOption(ppd, "JCLDuplex"); + + if (duplex && duplex->num_choices > 1 && + !ppdInstallableConflict(ppd, duplex->keyword, "DuplexTumble")) + { + p->type |= CUPS_PRINTER_DUPLEX; + + ippAddStrings(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "sides-supported", 3, NULL, sides); + + if (!_cups_strcasecmp(duplex->defchoice, "DuplexTumble")) + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "sides-default", NULL, "two-sided-short-edge"); + else if (!_cups_strcasecmp(duplex->defchoice, "DuplexNoTumble")) + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "sides-default", NULL, "two-sided-long-edge"); + else + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "sides-default", NULL, "one-sided"); + } + else + { + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "sides-supported", NULL, "one-sided"); + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "sides-default", NULL, "one-sided"); + } + + if (ppdFindOption(ppd, "Collate") != NULL) + p->type |= CUPS_PRINTER_COLLATE; + + if (ppdFindOption(ppd, "StapleLocation") != NULL) + { + p->type |= CUPS_PRINTER_STAPLE; + finishings[num_finishings++] = IPP_FINISHINGS_STAPLE; + } + + if (ppdFindOption(ppd, "BindEdge") != NULL) + { + p->type |= CUPS_PRINTER_BIND; + finishings[num_finishings++] = IPP_FINISHINGS_BIND; + } + + for (i = 0; i < ppd->num_sizes; i ++) + if (ppd->sizes[i].length > 1728) + p->type |= CUPS_PRINTER_LARGE; + else if (ppd->sizes[i].length > 1008) + p->type |= CUPS_PRINTER_MEDIUM; + else + p->type |= CUPS_PRINTER_SMALL; + + if ((ppd_attr = ppdFindAttr(ppd, "APICADriver", NULL)) != NULL && + ppd_attr->value && !_cups_strcasecmp(ppd_attr->value, "true")) + { + if ((ppd_attr = ppdFindAttr(ppd, "APScannerOnly", NULL)) != NULL && + ppd_attr->value && !_cups_strcasecmp(ppd_attr->value, "true")) + p->type |= CUPS_PRINTER_SCANNER; + else + p->type |= CUPS_PRINTER_MFP; + } + + /* + * Scan the filters in the PPD file... + */ + + if (p->pc) + { + for (filter = (const char *)cupsArrayFirst(p->pc->filters); + filter; + filter = (const char *)cupsArrayNext(p->pc->filters)) + { + if (!_cups_strncasecmp(filter, "application/vnd.cups-command", 28) && + _cups_isspace(filter[28])) + { + p->type |= CUPS_PRINTER_COMMANDS; + break; + } + } + } + + if (p->type & CUPS_PRINTER_COMMANDS) + { + char *commands, /* Copy of commands */ + *start, /* Start of name */ + *end; /* End of name */ + int count; /* Number of commands */ + + if ((ppd_attr = ppdFindAttr(ppd, "cupsCommands", NULL)) != NULL) + { + for (count = 0, start = ppd_attr->value; *start; count ++) + { + while (_cups_isspace(*start)) + start ++; + + if (!*start) + break; + + while (*start && !isspace(*start & 255)) + start ++; + } + } + else + count = 0; + + if (count > 0) + { + /* + * Make a copy of the commands string and count how many commands there + * are... + */ + + attr = ippAddStrings(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "printer-commands", count, NULL, NULL); + + commands = strdup(ppd_attr->value); + + for (count = 0, start = commands; *start; count ++) + { + while (isspace(*start & 255)) + start ++; + + if (!*start) + break; + + end = start; + while (*end && !isspace(*end & 255)) + end ++; + + if (*end) + *end++ = '\0'; + + attr->values[count].string.text = _cupsStrAlloc(start); + + start = end; + } + + free(commands); + } + else + { + /* + * Add the standard list of commands... + */ + + ippAddStrings(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "printer-commands", + (int)(sizeof(standard_commands) / + sizeof(standard_commands[0])), NULL, + standard_commands); + } + } + else + { + /* + * No commands supported... + */ + + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, + "printer-commands", NULL, "none"); + } + + /* + * Show current and available port monitors for this printer... + */ + + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_NAME, "port-monitor", + NULL, p->port_monitor ? p->port_monitor : "none"); + + for (i = 1, ppd_attr = ppdFindAttr(ppd, "cupsPortMonitor", NULL); + ppd_attr; + i ++, ppd_attr = ppdFindNextAttr(ppd, "cupsPortMonitor", NULL)); + + if (ppd->protocols) + { + if (strstr(ppd->protocols, "TBCP")) + i ++; + else if (strstr(ppd->protocols, "BCP")) + i ++; + } + + attr = ippAddStrings(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_NAME, + "port-monitor-supported", i, NULL, NULL); + + attr->values[0].string.text = _cupsStrAlloc("none"); + + for (i = 1, ppd_attr = ppdFindAttr(ppd, "cupsPortMonitor", NULL); + ppd_attr; + i ++, ppd_attr = ppdFindNextAttr(ppd, "cupsPortMonitor", NULL)) + attr->values[i].string.text = _cupsStrAlloc(ppd_attr->value); + + if (ppd->protocols) + { + if (strstr(ppd->protocols, "TBCP")) + attr->values[i].string.text = _cupsStrAlloc("tbcp"); + else if (strstr(ppd->protocols, "BCP")) + attr->values[i].string.text = _cupsStrAlloc("bcp"); + } + + if (ppdFindAttr(ppd, "APRemoteQueueID", NULL)) + p->type |= CUPS_PRINTER_REMOTE; + +#ifdef HAVE_APPLICATIONSERVICES_H + /* + * Convert the file referenced in APPrinterIconPath to a 128x128 PNG + * and save it as cacheDir/printername.png + */ + + if ((ppd_attr = ppdFindAttr(ppd, "APPrinterIconPath", NULL)) != NULL && + ppd_attr->value && + !_cupsFileCheck(ppd_attr->value, _CUPS_FILE_CHECK_FILE, !RunUser, + cupsdLogFCMessage, p)) + { + CGImageRef imageRef = NULL;/* Current icon image */ + CGImageRef biggestIconRef = NULL; + /* Biggest icon image */ + CGImageRef closestTo128IconRef = NULL; + /* Icon image closest to and >= 128 */ + CGImageSourceRef sourceRef; /* The file's image source */ + char outPath[HTTP_MAX_URI]; + /* The path to the PNG file */ + CFURLRef outUrl; /* The URL made from the outPath */ + CFURLRef icnsFileUrl; /* The URL of the original ICNS icon file */ + CGImageDestinationRef destRef; /* The image destination to write */ + size_t bytesPerRow; /* The bytes per row used for resizing */ + CGContextRef context; /* The CG context used for resizing */ + + snprintf(outPath, sizeof(outPath), "%s/%s.png", CacheDir, p->name); + outUrl = CFURLCreateFromFileSystemRepresentation(kCFAllocatorDefault, + (UInt8 *)outPath, + strlen(outPath), + FALSE); + icnsFileUrl = CFURLCreateFromFileSystemRepresentation(kCFAllocatorDefault, + (UInt8 *)ppd_attr->value, + strlen(ppd_attr->value), + FALSE); + if (outUrl && icnsFileUrl) + { + sourceRef = CGImageSourceCreateWithURL(icnsFileUrl, NULL); + if (sourceRef) + { + for (i = 0; i < CGImageSourceGetCount(sourceRef); i ++) + { + imageRef = CGImageSourceCreateImageAtIndex(sourceRef, i, NULL); + if (!imageRef) + continue; + + if (CGImageGetWidth(imageRef) == CGImageGetHeight(imageRef)) + { + /* + * Loop through remembering the icon closest to 128 but >= 128 + * and then remember the largest icon. + */ + + if (CGImageGetWidth(imageRef) >= 128 && + (!closestTo128IconRef || + CGImageGetWidth(imageRef) < + CGImageGetWidth(closestTo128IconRef))) + { + CGImageRelease(closestTo128IconRef); + CGImageRetain(imageRef); + closestTo128IconRef = imageRef; + } + + if (!biggestIconRef || + CGImageGetWidth(imageRef) > CGImageGetWidth(biggestIconRef)) + { + CGImageRelease(biggestIconRef); + CGImageRetain(imageRef); + biggestIconRef = imageRef; + } + } + + CGImageRelease(imageRef); + } + + if (biggestIconRef) + { + /* + * If biggestIconRef is NULL, we found no icons. Otherwise we first + * want the closest to 128, but if none are larger than 128, we want + * the largest icon available. + */ + + imageRef = closestTo128IconRef ? closestTo128IconRef : + biggestIconRef; + CGImageRetain(imageRef); + CGImageRelease(biggestIconRef); + if (closestTo128IconRef) + CGImageRelease(closestTo128IconRef); + destRef = CGImageDestinationCreateWithURL(outUrl, kUTTypePNG, 1, + NULL); + if (destRef) + { + if (CGImageGetWidth(imageRef) != 128) + { + bytesPerRow = CGImageGetBytesPerRow(imageRef) / + CGImageGetWidth(imageRef) * 128; + context = CGBitmapContextCreate(NULL, 128, 128, + CGImageGetBitsPerComponent(imageRef), + bytesPerRow, + CGImageGetColorSpace(imageRef), + kCGImageAlphaPremultipliedFirst); + if (context) + { + CGContextDrawImage(context, CGRectMake(0, 0, 128, 128), + imageRef); + CGImageRelease(imageRef); + imageRef = CGBitmapContextCreateImage(context); + CGContextRelease(context); + } + } + + CGImageDestinationAddImage(destRef, imageRef, NULL); + CGImageDestinationFinalize(destRef); + CFRelease(destRef); + } + + CGImageRelease(imageRef); + } + + CFRelease(sourceRef); + } + } + + if (outUrl) + CFRelease(outUrl); + + if (icnsFileUrl) + CFRelease(icnsFileUrl); + } +#endif /* HAVE_APPLICATIONSERVICES_H */ + + /* + * Close the PPD and set the type... + */ + + ppdClose(ppd); + } + else if (!access(ppd_name, 0)) + { + int pline; /* PPD line number */ + ppd_status_t pstatus; /* PPD load status */ + + + pstatus = ppdLastError(&pline); + + cupsdLogMessage(CUPSD_LOG_ERROR, "PPD file for %s cannot be loaded!", + p->name); + + if (pstatus <= PPD_ALLOC_ERROR) + cupsdLogMessage(CUPSD_LOG_ERROR, "%s", strerror(errno)); + else + cupsdLogMessage(CUPSD_LOG_ERROR, "%s on line %d.", + ppdErrorString(pstatus), pline); + + cupsdLogMessage(CUPSD_LOG_INFO, + "Hint: Run \"cupstestppd %s\" and fix any errors.", + ppd_name); + } + else + { + /* + * If we have an interface script, add a filter entry for it... + */ + + char interface[1024]; /* Interface script */ + + + snprintf(interface, sizeof(interface), "%s/interfaces/%s", ServerRoot, + p->name); + if (!access(interface, X_OK)) + { + /* + * Yes, we have a System V style interface script; use it! + */ + + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_TEXT, + "printer-make-and-model", NULL, + "Local System V Printer"); + } + else if (!strncmp(p->device_uri, "ipp://", 6) && + (strstr(p->device_uri, "/printers/") != NULL || + strstr(p->device_uri, "/classes/") != NULL || + (strstr(p->device_uri, "._ipp.") != NULL && + !strcmp(p->device_uri + strlen(p->device_uri) - 5, + "/cups")))) + { + /* + * Tell the client this is really a hard-wired remote printer. + */ + + p->type |= CUPS_PRINTER_REMOTE; + + /* + * Point the printer-uri-supported attribute to the + * remote printer... + */ + + if (strchr(p->device_uri, '?')) + { + /* + * Strip trailing "?options" from URI... + */ + + char resource[HTTP_MAX_URI], /* New URI */ + *ptr; /* Pointer into URI */ + + strlcpy(resource, p->device_uri, sizeof(resource)); + if ((ptr = strchr(resource, '?')) != NULL) + *ptr = '\0'; + + ippAddString(p->attrs, IPP_TAG_PRINTER, IPP_TAG_URI, + "printer-uri-supported", NULL, resource); + } + else + ippAddString(p->attrs, IPP_TAG_PRINTER, IPP_TAG_URI, + "printer-uri-supported", NULL, p->device_uri); + + /* + * Then set the make-and-model accordingly... + */ + + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_TEXT, + "printer-make-and-model", NULL, "Remote Printer"); + + /* + * Print all files directly... + */ + + p->raw = 1; + p->remote = 1; + } + else + { + /* + * Otherwise we have neither - treat this as a "dumb" printer + * with no PPD file... + */ + + ippAddString(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_TEXT, + "printer-make-and-model", NULL, "Local Raw Printer"); + + p->raw = 1; + } + } + + ippAddIntegers(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_ENUM, + "finishings-supported", num_finishings, finishings); + ippAddInteger(p->ppd_attrs, IPP_TAG_PRINTER, IPP_TAG_ENUM, + "finishings-default", IPP_FINISHINGS_NONE); + + if (ppd && p->pc) + { + /* + * Save cached PPD attributes to disk... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, "load_ppd: Saving %s...", cache_name); + + _ppdCacheWriteFile(p->pc, cache_name, p->ppd_attrs); + } + else + { + /* + * Remove cache files... + */ + + if (cache_info.st_mtime) + unlink(cache_name); + } +} + + +/* + * 'log_ipp_conformance()' - Log an IPP conformance issue with a printer. + */ + +static void +log_ipp_conformance( + cupsd_printer_t *p, /* I - Printer */ + const char *reason) /* I - Printer state reason */ +{ + const char *message; /* Message to log */ +#ifdef __APPLE__ + aslmsg aslm; /* Apple System Log message */ +#endif /* __APPLE__ */ + + + /* + * Strip the leading "cups-ipp-" from the reason and create a log message for + * it... + */ + + reason += 9; + if (!strcmp(reason, "missing-cancel-job")) + message = "Printer does not support REQUIRED Cancel-Job operation."; + else if (!strcmp(reason, "missing-get-job-attributes")) + message = "Printer does not support REQUIRED Get-Job-Attributes operation."; + else if (!strcmp(reason, "missing-print-job")) + message = "Printer does not support REQUIRED Print-Job operation."; + else if (!strcmp(reason, "missing-validate-job")) + message = "Printer does not support REQUIRED Validate-Job operation."; + else if (!strcmp(reason, "missing-get-printer-attributes")) + message = "Printer does not support REQUIRED Get-Printer-Attributes operation."; + else if (!strcmp(reason, "missing-job-history")) + message = "Printer does not provide REQUIRED job history."; + else if (!strcmp(reason, "missing-job-id")) + message = "Printer does not provide REQUIRED job-id attribute."; + else if (!strcmp(reason, "missing-job-state")) + message = "Printer does not provide REQUIRED job-state attribute."; + else if (!strcmp(reason, "missing-operations-supported")) + message = "Printer does not provide REQUIRED operations-supported " + "attribute."; + else if (!strcmp(reason, "missing-printer-is-accepting-jobs")) + message = "Printer does not provide REQUIRED printer-is-accepting-jobs " + "attribute."; + else if (!strcmp(reason, "missing-printer-state-reasons")) + message = "Printer does not provide REQUIRED printer-state-reasons " + "attribute."; + else if (!strcmp(reason, "wrong-http-version")) + message = "Printer does not use REQUIRED HTTP/1.1 transport."; + else + message = "Unknown IPP conformance failure."; + + cupsdLogMessage(CUPSD_LOG_WARN, "%s: %s", p->name, message); + +#ifdef __APPLE__ + /* + * Report the failure information to Apple if the user opts into providing + * feedback to Apple... + */ + + aslm = asl_new(ASL_TYPE_MSG); + if (aslm) + { + asl_set(aslm, "com.apple.message.domain", "com.apple.printing.ipp.conformance"); + asl_set(aslm, "com.apple.message.domain_scope", "com.apple.printing.ipp.conformance"); + asl_set(aslm, "com.apple.message.signature", reason); + asl_set(aslm, "com.apple.message.signature2", + p->make_model ? p->make_model : "Unknown"); + asl_log(NULL, aslm, ASL_LEVEL_NOTICE, "%s: %s", + p->make_model ? p->make_model : "Unknown", message); + asl_free(aslm); + } +#endif /* __APPLE__ */ +} + + +/* + * 'new_media_col()' - Create a media-col collection value. + */ + +static ipp_t * /* O - Collection value */ +new_media_col(_pwg_size_t *size, /* I - media-size/margin values */ + const char *source, /* I - media-source value */ + const char *type) /* I - media-type value */ +{ + ipp_t *media_col, /* Collection value */ + *media_size; /* media-size value */ + + + media_col = ippNew(); + + media_size = ippNew(); + ippAddInteger(media_size, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "x-dimension", size->width); + ippAddInteger(media_size, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "y-dimension", size->length); + ippAddCollection(media_col, IPP_TAG_PRINTER, "media-size", media_size); + ippDelete(media_size); + + ippAddInteger(media_col, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "media-bottom-margin", size->bottom); + ippAddInteger(media_col, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "media-left-margin", size->left); + ippAddInteger(media_col, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "media-right-margin", size->right); + ippAddInteger(media_col, IPP_TAG_PRINTER, IPP_TAG_INTEGER, + "media-top-margin", size->top); + + if (source) + ippAddString(media_col, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, "media-source", + NULL, source); + + if (type) + ippAddString(media_col, IPP_TAG_PRINTER, IPP_TAG_KEYWORD, "media-type", + NULL, type); + + return (media_col); +} + + +#ifdef __sgi +/* + * 'write_irix_config()' - Update the config files used by the IRIX + * desktop tools. + */ + +static void +write_irix_config(cupsd_printer_t *p) /* I - Printer to update */ +{ + char filename[1024]; /* Interface script filename */ + cups_file_t *fp; /* Interface script file */ + ipp_attribute_t *attr; /* Attribute data */ + + + /* + * Add dummy interface and GUI scripts to fool SGI's "challenged" printing + * tools. First the interface script that tells the tools what kind of + * printer we have... + */ + + snprintf(filename, sizeof(filename), "/var/spool/lp/interface/%s", p->name); + + if (p->type & CUPS_PRINTER_CLASS) + unlink(filename); + else if ((fp = cupsFileOpen(filename, "w")) != NULL) + { + cupsFilePuts(fp, "#!/bin/sh\n"); + + if ((attr = ippFindAttribute(p->attrs, "printer-make-and-model", + IPP_TAG_TEXT)) != NULL) + cupsFilePrintf(fp, "NAME=\"%s\"\n", attr->values[0].string.text); + else if (p->type & CUPS_PRINTER_CLASS) + cupsFilePuts(fp, "NAME=\"Printer Class\"\n"); + else + cupsFilePuts(fp, "NAME=\"Remote Destination\"\n"); + + if (p->type & CUPS_PRINTER_COLOR) + cupsFilePuts(fp, "TYPE=ColorPostScript\n"); + else + cupsFilePuts(fp, "TYPE=MonoPostScript\n"); + + cupsFilePrintf(fp, "HOSTNAME=%s\n", ServerName); + cupsFilePrintf(fp, "HOSTPRINTER=%s\n", p->name); + + cupsFileClose(fp); + + chmod(filename, 0755); + chown(filename, User, Group); + } + + /* + * Then the member file that tells which device file the queue is connected + * to... Networked printers use "/dev/null" in this file, so that's what + * we use (the actual device URI can confuse some apps...) + */ + + snprintf(filename, sizeof(filename), "/var/spool/lp/member/%s", p->name); + + if (p->type & CUPS_PRINTER_CLASS) + unlink(filename); + else if ((fp = cupsFileOpen(filename, "w")) != NULL) + { + cupsFilePuts(fp, "/dev/null\n"); + + cupsFileClose(fp); + + chmod(filename, 0644); + chown(filename, User, Group); + } + + /* + * The gui_interface file is a script or program that launches a GUI + * option panel for the printer, using options specified on the + * command-line in the third argument. The option panel must send + * any printing options to stdout on a single line when the user + * accepts them, or nothing if the user cancels the dialog. + * + * The default options panel program is /usr/bin/glpoptions, from + * the ESP Print Pro software. You can select another using the + * PrintcapGUI option. + */ + + snprintf(filename, sizeof(filename), "/var/spool/lp/gui_interface/ELF/%s.gui", p->name); + + if (p->type & CUPS_PRINTER_CLASS) + unlink(filename); + else if ((fp = cupsFileOpen(filename, "w")) != NULL) + { + cupsFilePuts(fp, "#!/bin/sh\n"); + cupsFilePrintf(fp, "%s -d %s -o \"$3\"\n", PrintcapGUI, p->name); + + cupsFileClose(fp); + + chmod(filename, 0755); + chown(filename, User, Group); + } + + /* + * The POD config file is needed by the printstatus command to show + * the printer location and device. + */ + + snprintf(filename, sizeof(filename), "/var/spool/lp/pod/%s.config", p->name); + + if (p->type & CUPS_PRINTER_CLASS) + unlink(filename); + else if ((fp = cupsFileOpen(filename, "w")) != NULL) + { + cupsFilePrintf(fp, "Printer Class | %s\n", + (p->type & CUPS_PRINTER_COLOR) ? "ColorPostScript" : "MonoPostScript"); + cupsFilePrintf(fp, "Printer Model | %s\n", p->make_model ? p->make_model : ""); + cupsFilePrintf(fp, "Location Code | %s\n", p->location ? p->location : ""); + cupsFilePrintf(fp, "Physical Location | %s\n", p->info ? p->info : ""); + cupsFilePrintf(fp, "Port Path | %s\n", p->device_uri); + cupsFilePrintf(fp, "Config Path | /var/spool/lp/pod/%s.config\n", p->name); + cupsFilePrintf(fp, "Active Status Path | /var/spool/lp/pod/%s.status\n", p->name); + cupsFilePuts(fp, "Status Update Wait | 10 seconds\n"); + + cupsFileClose(fp); + + chmod(filename, 0664); + chown(filename, User, Group); + } +} + + +/* + * 'write_irix_state()' - Update the status files used by IRIX printing + * desktop tools. + */ + +static void +write_irix_state(cupsd_printer_t *p) /* I - Printer to update */ +{ + char filename[1024]; /* Interface script filename */ + cups_file_t *fp; /* Interface script file */ + int tag; /* Status tag value */ + + + if (p) + { + /* + * The POD status file is needed for the printstatus window to + * provide the current status of the printer. + */ + + snprintf(filename, sizeof(filename), "/var/spool/lp/pod/%s.status", p->name); + + if (p->type & CUPS_PRINTER_CLASS) + unlink(filename); + else if ((fp = cupsFileOpen(filename, "w")) != NULL) + { + cupsFilePrintf(fp, "Operational Status | %s\n", + (p->state == IPP_PRINTER_IDLE) ? "Idle" : + (p->state == IPP_PRINTER_PROCESSING) ? "Busy" : + "Faulted"); + cupsFilePrintf(fp, "Information | 01 00 00 | %s\n", CUPS_SVERSION); + cupsFilePrintf(fp, "Information | 02 00 00 | Device URI: %s\n", + p->device_uri); + cupsFilePrintf(fp, "Information | 03 00 00 | %s jobs\n", + p->accepting ? "Accepting" : "Not accepting"); + cupsFilePrintf(fp, "Information | 04 00 00 | %s\n", p->state_message); + + cupsFileClose(fp); + + chmod(filename, 0664); + chown(filename, User, Group); + } + + /* + * The activeicons file is needed to provide desktop icons for printers: + * + * [ quoted from /usr/lib/print/tagit ] + * + * --- Type of printer tags (base values) + * + * Dumb=66048 # 0x10200 + * DumbColor=66080 # 0x10220 + * Raster=66112 # 0x10240 + * ColorRaster=66144 # 0x10260 + * Plotter=66176 # 0x10280 + * PostScript=66208 # 0x102A0 + * ColorPostScript=66240 # 0x102C0 + * MonoPostScript=66272 # 0x102E0 + * + * --- Printer state modifiers for local printers + * + * Idle=0 # 0x0 + * Busy=1 # 0x1 + * Faulted=2 # 0x2 + * Unknown=3 # 0x3 (Faulted due to unknown reason) + * + * --- Printer state modifiers for network printers + * + * NetIdle=8 # 0x8 + * NetBusy=9 # 0x9 + * NetFaulted=10 # 0xA + * NetUnknown=11 # 0xB (Faulted due to unknown reason) + */ + + snprintf(filename, sizeof(filename), "/var/spool/lp/activeicons/%s", p->name); + + if (p->type & CUPS_PRINTER_CLASS) + unlink(filename); + else if ((fp = cupsFileOpen(filename, "w")) != NULL) + { + if (p->type & CUPS_PRINTER_COLOR) + tag = 66240; + else + tag = 66272; + + if (p->type & CUPS_PRINTER_REMOTE) + tag |= 8; + + if (p->state == IPP_PRINTER_PROCESSING) + tag |= 1; + + else if (p->state == IPP_PRINTER_STOPPED) + tag |= 2; + + cupsFilePuts(fp, "#!/bin/sh\n"); + cupsFilePrintf(fp, "#Tag %d\n", tag); + + cupsFileClose(fp); + + chmod(filename, 0755); + chown(filename, User, Group); + } + } + + /* + * The default file is needed by the printers window to show + * the default printer. + */ + + snprintf(filename, sizeof(filename), "/var/spool/lp/default"); + + if (DefaultPrinter != NULL) + { + if ((fp = cupsFileOpen(filename, "w")) != NULL) + { + cupsFilePrintf(fp, "%s\n", DefaultPrinter->name); + + cupsFileClose(fp); + + chmod(filename, 0644); + chown(filename, User, Group); + } + } + else + unlink(filename); +} +#endif /* __sgi */ + + +/* + * 'write_xml_string()' - Write a string with XML escaping. + */ + +static void +write_xml_string(cups_file_t *fp, /* I - File to write to */ + const char *s) /* I - String to write */ +{ + const char *start; /* Start of current sequence */ + + + if (!s) + return; + + for (start = s; *s; s ++) + { + if (*s == '&') + { + if (s > start) + cupsFileWrite(fp, start, s - start); + + cupsFilePuts(fp, "&"); + start = s + 1; + } + else if (*s == '<') + { + if (s > start) + cupsFileWrite(fp, start, s - start); + + cupsFilePuts(fp, "<"); + start = s + 1; + } + } + + if (s > start) + cupsFilePuts(fp, start); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/printers.h b/scheduler/printers.h new file mode 100644 index 0000000000..b053a163b2 --- /dev/null +++ b/scheduler/printers.h @@ -0,0 +1,172 @@ +/* + * "$Id$" + * + * Printer definitions for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#ifdef HAVE_DNSSD +# include +#endif /* HAVE_DNSSD */ +#include + + +/* + * Quota data... + */ + +typedef struct +{ + char username[33]; /* User data */ + time_t next_update; /* Next update time */ + int page_count, /* Count of pages */ + k_count; /* Count of kilobytes */ +} cupsd_quota_t; + + +/* + * Printer/class information structure... + */ + +typedef struct cupsd_job_s cupsd_job_t; + +struct cupsd_printer_s +{ + char *uri, /* Printer URI */ + *uuid, /* Printer UUID */ + *hostname, /* Host printer resides on */ + *name, /* Printer name */ + *location, /* Location code */ + *make_model, /* Make and model */ + *info, /* Description */ + *op_policy, /* Operation policy name */ + *error_policy; /* Error policy */ + cupsd_policy_t *op_policy_ptr; /* Pointer to operation policy */ + int shared; /* Shared? */ + int accepting; /* Accepting jobs? */ + int holding_new_jobs; /* Holding new jobs for printing? */ + int in_implicit_class; /* In an implicit class? */ + ipp_pstate_t state; /* Printer state */ + char state_message[1024]; /* Printer state message */ + int num_reasons; /* Number of printer-state-reasons */ + char *reasons[64]; /* printer-state-reasons strings */ + time_t state_time; /* Time at this state */ + char *job_sheets[2]; /* Banners/job sheets */ + cups_ptype_t type; /* Printer type (color, small, etc.) */ + char *device_uri; /* Device URI */ + char *sanitized_device_uri; /* Sanitized device URI */ + char *port_monitor; /* Port monitor */ + int raw; /* Raw queue? */ + int remote; /* Remote queue? */ + mime_type_t *filetype, /* Pseudo-filetype for printer */ + *prefiltertype; /* Pseudo-filetype for pre-filters */ + cups_array_t *filetypes, /* Supported file types */ + *dest_types; /* Destination types for queue */ + cupsd_job_t *job; /* Current job in queue */ + ipp_t *attrs, /* Attributes supported by this printer */ + *ppd_attrs; /* Attributes based on the PPD */ + int num_printers, /* Number of printers in class */ + last_printer; /* Last printer job was sent to */ + struct cupsd_printer_s **printers; /* Printers in class */ + int quota_period, /* Period for quotas */ + page_limit, /* Maximum number of pages */ + k_limit; /* Maximum number of kilobytes */ + cups_array_t *quotas; /* Quota records */ + int deny_users; /* 1 = deny, 0 = allow */ + cups_array_t *users; /* Allowed/denied users */ + int sequence_number; /* Increasing sequence number */ + int num_options; /* Number of default options */ + cups_option_t *options; /* Default options */ + int num_auth_info_required; /* Number of required auth fields */ + const char *auth_info_required[4]; /* Required authentication fields */ + char *alert, /* PSX printer-alert value */ + *alert_description; /* PSX printer-alert-description value */ + time_t marker_time; /* Last time marker attributes were updated */ + _ppd_cache_t *pc; /* PPD cache and mapping data */ + +#ifdef HAVE_DNSSD + char *reg_name, /* Name used for service registration */ + *pdl, /* pdl value for TXT record */ + *ipp_txt, /* IPP TXT record contents */ + *printer_txt; /* LPD TXT record contents */ + int ipp_len, /* IPP TXT record length */ + printer_len; /* LPD TXT record length */ + DNSServiceRef ipp_ref, /* Reference for _ipp._tcp,_cups */ + printer_ref; /* Reference for _printer._tcp */ +#endif /* HAVE_DNSSD */ +}; + + +/* + * Globals... + */ + +VAR ipp_t *CommonData VALUE(NULL); + /* Common printer object attrs */ +VAR cups_array_t *CommonDefaults VALUE(NULL); + /* Common -default option names */ +VAR cups_array_t *Printers VALUE(NULL); + /* Printer list */ +VAR cupsd_printer_t *DefaultPrinter VALUE(NULL); + /* Default printer */ +VAR char *DefaultPolicy VALUE(NULL); + /* Default policy name */ +VAR cupsd_policy_t *DefaultPolicyPtr + VALUE(NULL); + /* Pointer to default policy */ + + +/* + * Prototypes... + */ + +extern cupsd_printer_t *cupsdAddPrinter(const char *name); +extern void cupsdCreateCommonData(void); +extern void cupsdDeleteAllPrinters(void); +extern int cupsdDeletePrinter(cupsd_printer_t *p, int update); +extern cupsd_printer_t *cupsdFindDest(const char *name); +extern cupsd_printer_t *cupsdFindPrinter(const char *name); +extern cupsd_quota_t *cupsdFindQuota(cupsd_printer_t *p, + const char *username); +extern void cupsdFreeQuotas(cupsd_printer_t *p); +extern void cupsdLoadAllPrinters(void); +extern void cupsdRenamePrinter(cupsd_printer_t *p, + const char *name); +extern void cupsdSaveAllPrinters(void); +extern int cupsdSetAuthInfoRequired(cupsd_printer_t *p, + const char *values, + ipp_attribute_t *attr); +extern void cupsdSetDeviceURI(cupsd_printer_t *p, const char *uri); +extern void cupsdSetPrinterAttr(cupsd_printer_t *p, + const char *name, char *value); +extern void cupsdSetPrinterAttrs(cupsd_printer_t *p); +extern int cupsdSetPrinterReasons(cupsd_printer_t *p, + const char *s); +extern void cupsdSetPrinterState(cupsd_printer_t *p, ipp_pstate_t s, + int update); +#define cupsdStartPrinter(p,u) cupsdSetPrinterState((p), IPP_PRINTER_IDLE, (u)) +extern void cupsdStopPrinter(cupsd_printer_t *p, int update); +extern int cupsdUpdatePrinterPPD(cupsd_printer_t *p, + int num_keywords, + cups_option_t *keywords); +extern void cupsdUpdatePrinters(void); +extern cupsd_quota_t *cupsdUpdateQuota(cupsd_printer_t *p, + const char *username, int pages, + int k); +extern const char *cupsdValidateDest(const char *uri, + cups_ptype_t *dtype, + cupsd_printer_t **printer); +extern void cupsdWritePrintcap(void); + + +/* + * End of "$Id$". + */ diff --git a/scheduler/process.c b/scheduler/process.c new file mode 100644 index 0000000000..cceb5f4717 --- /dev/null +++ b/scheduler/process.c @@ -0,0 +1,656 @@ +/* + * "$Id$" + * + * Process management routines for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdCreateProfile() - Create an execution profile for a subprocess. + * cupsdDestroyProfile() - Delete an execution profile. + * cupsdEndProcess() - End a process. + * cupsdFinishProcess() - Finish a process and get its name. + * cupsdStartProcess() - Start a process. + * compare_procs() - Compare two processes. + * cupsd_requote() - Make a regular-expression version of a string. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" +#include +#ifdef __APPLE__ +# include +#endif /* __APPLE__ */ + + +/* + * Process structure... + */ + +typedef struct +{ + int pid, /* Process ID */ + job_id; /* Job associated with process */ + char name[1]; /* Name of process */ +} cupsd_proc_t; + + +/* + * Local globals... + */ + +static cups_array_t *process_array = NULL; + + +/* + * Local functions... + */ + +static int compare_procs(cupsd_proc_t *a, cupsd_proc_t *b); +#ifdef HAVE_SANDBOX_H +static char *cupsd_requote(char *dst, const char *src, size_t dstsize); +#endif /* HAVE_SANDBOX_H */ + + +/* + * 'cupsdCreateProfile()' - Create an execution profile for a subprocess. + */ + +void * /* O - Profile or NULL on error */ +cupsdCreateProfile(int job_id) /* I - Job ID or 0 for none */ +{ +#ifdef HAVE_SANDBOX_H + cups_file_t *fp; /* File pointer */ + char profile[1024], /* File containing the profile */ + cache[1024], /* Quoted CacheDir */ + request[1024], /* Quoted RequestRoot */ + root[1024], /* Quoted ServerRoot */ + temp[1024]; /* Quoted TempDir */ + const char *nodebug; /* " (with no-log)" for no debug */ + + + if (!UseProfiles) + { + /* + * Only use sandbox profiles as root... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdCreateProfile(job_id=%d) = NULL", + job_id); + + return (NULL); + } + + if ((fp = cupsTempFile2(profile, sizeof(profile))) == NULL) + { + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdCreateProfile(job_id=%d) = NULL", + job_id); + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to create security profile: %s", + strerror(errno)); + return (NULL); + } + + fchown(cupsFileNumber(fp), RunUser, Group); + fchmod(cupsFileNumber(fp), 0640); + + cupsd_requote(cache, CacheDir, sizeof(cache)); + cupsd_requote(request, RequestRoot, sizeof(request)); + cupsd_requote(root, ServerRoot, sizeof(root)); + cupsd_requote(temp, TempDir, sizeof(temp)); + + nodebug = LogLevel < CUPSD_LOG_DEBUG ? " (with no-log)" : ""; + + cupsFilePuts(fp, "(version 1)\n"); + cupsFilePuts(fp, "(allow default)\n"); + cupsFilePrintf(fp, + "(deny file-write* file-read-data file-read-metadata\n" + " (regex" + " #\"^%s$\"" /* RequestRoot */ + " #\"^%s/\"" /* RequestRoot/... */ + ")%s)\n", + request, request, nodebug); + if (!RunUser) + cupsFilePrintf(fp, + "(deny file-write* file-read-data file-read-metadata\n" + " (regex" + " #\"^/Users$\"" + " #\"^/Users/\"" + ")%s)\n", nodebug); + cupsFilePrintf(fp, + "(deny file-write*\n" + " (regex" + " #\"^%s$\"" /* ServerRoot */ + " #\"^%s/\"" /* ServerRoot/... */ + " #\"^/private/etc$\"" + " #\"^/private/etc/\"" + " #\"^/usr/local/etc$\"" + " #\"^/usr/local/etc/\"" + " #\"^/Library$\"" + " #\"^/Library/\"" + " #\"^/System$\"" + " #\"^/System/\"" + ")%s)\n", + root, root, nodebug); + /* Specifically allow applications to stat RequestRoot */ + cupsFilePrintf(fp, + "(allow file-read-metadata\n" + " (regex" + " #\"^%s$\"" /* RequestRoot */ + "))\n", + request); + cupsFilePrintf(fp, + "(allow file-write* file-read-data file-read-metadata\n" + " (regex" + " #\"^%s$\"" /* TempDir */ + " #\"^%s/\"" /* TempDir/... */ + " #\"^%s$\"" /* CacheDir */ + " #\"^%s/\"" /* CacheDir/... */ + " #\"^%s/Library$\"" /* RequestRoot/Library */ + " #\"^%s/Library/\"" /* RequestRoot/Library/... */ + " #\"^/Library/Application Support/\"" + " #\"^/Library/Caches/\"" + " #\"^/Library/Preferences/\"" + " #\"^/Library/Printers/.*/\"" + " #\"^/Users/Shared/\"" + "))\n", + temp, temp, cache, cache, request, request); + cupsFilePrintf(fp, + "(deny file-write*\n" + " (regex" + " #\"^/Library/Printers/PPDs$\"" + " #\"^/Library/Printers/PPDs/\"" + " #\"^/Library/Printers/PPD Plugins$\"" + " #\"^/Library/Printers/PPD Plugins/\"" + ")%s)\n", nodebug); + if (job_id) + { + /* + * Allow job filters to read the spool file(s)... + */ + + cupsFilePrintf(fp, + "(allow file-read-data file-read-metadata\n" + " (regex #\"^%s/([ac]%05d|d%05d-[0-9][0-9][0-9])$\"))\n", + request, job_id, job_id); + } + else + { + /* + * Allow email notifications from notifiers... + */ + + cupsFilePuts(fp, + "(allow process-exec\n" + " (literal \"/usr/sbin/sendmail\")\n" + " (with no-sandbox)\n" + ")\n"); + } + + cupsFileClose(fp); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdCreateProfile(job_id=%d) = \"%s\"", + job_id, profile); + return ((void *)strdup(profile)); + +#else + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdCreateProfile(job_id=%d) = NULL", + job_id); + + return (NULL); +#endif /* HAVE_SANDBOX_H */ +} + + +/* + * 'cupsdDestroyProfile()' - Delete an execution profile. + */ + +void +cupsdDestroyProfile(void *profile) /* I - Profile */ +{ + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdDeleteProfile(profile=\"%s\")", + profile ? (char *)profile : "(null)"); + +#ifdef HAVE_SANDBOX_H + if (profile) + { + unlink((char *)profile); + free(profile); + } +#endif /* HAVE_SANDBOX_H */ +} + + +/* + * 'cupsdEndProcess()' - End a process. + */ + +int /* O - 0 on success, -1 on failure */ +cupsdEndProcess(int pid, /* I - Process ID */ + int force) /* I - Force child to die */ +{ + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdEndProcess(pid=%d, force=%d)", pid, + force); + + if (!pid) + return (0); + + if (!RunUser) + { + /* + * When running as root, cupsd puts child processes in their own process + * group. Using "-pid" sends a signal to all processes in the group. + */ + + pid = -pid; + } + + if (force) + return (kill(pid, SIGKILL)); + else + return (kill(pid, SIGTERM)); +} + + +/* + * 'cupsdFinishProcess()' - Finish a process and get its name. + */ + +const char * /* O - Process name */ +cupsdFinishProcess(int pid, /* I - Process ID */ + char *name, /* I - Name buffer */ + int namelen, /* I - Size of name buffer */ + int *job_id) /* O - Job ID pointer or NULL */ +{ + cupsd_proc_t key, /* Search key */ + *proc; /* Matching process */ + + + key.pid = pid; + + if ((proc = (cupsd_proc_t *)cupsArrayFind(process_array, &key)) != NULL) + { + if (job_id) + *job_id = proc->job_id; + + strlcpy(name, proc->name, namelen); + cupsArrayRemove(process_array, proc); + free(proc); + } + else + { + if (job_id) + *job_id = 0; + + strlcpy(name, "unknown", namelen); + } + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdFinishProcess(pid=%d, name=%p, namelen=%d, " + "job_id=%p(%d)) = \"%s\"", pid, name, namelen, job_id, + job_id ? *job_id : 0, name); + + return (name); +} + + +/* + * 'cupsdStartProcess()' - Start a process. + */ + +int /* O - Process ID or 0 */ +cupsdStartProcess( + const char *command, /* I - Full path to command */ + char *argv[], /* I - Command-line arguments */ + char *envp[], /* I - Environment */ + int infd, /* I - Standard input file descriptor */ + int outfd, /* I - Standard output file descriptor */ + int errfd, /* I - Standard error file descriptor */ + int backfd, /* I - Backchannel file descriptor */ + int sidefd, /* I - Sidechannel file descriptor */ + int root, /* I - Run as root? */ + void *profile, /* I - Security profile to use */ + cupsd_job_t *job, /* I - Job associated with process */ + int *pid) /* O - Process ID */ +{ + int i; /* Looping var */ + const char *exec_path = command; /* Command to be exec'd */ + char *real_argv[103], /* Real command-line arguments */ + cups_exec[1024]; /* Path to "cups-exec" program */ + int user; /* Command UID */ + cupsd_proc_t *proc; /* New process record */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* POSIX signal handler */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ +#if defined(__APPLE__) + char processPath[1024], /* CFProcessPath environment variable */ + linkpath[1024]; /* Link path for symlinks... */ + int linkbytes; /* Bytes for link path */ +#endif /* __APPLE__ */ + + + *pid = 0; + + /* + * Figure out the UID for the child process... + */ + + if (RunUser) + user = RunUser; + else if (root) + user = 0; + else + user = User; + + /* + * Check the permissions of the command we are running... + */ + + if (_cupsFileCheck(command, _CUPS_FILE_CHECK_PROGRAM, !RunUser, + cupsdLogFCMessage, job ? job->printer : NULL)) + return (0); + +#if defined(__APPLE__) + if (envp) + { + /* + * Add special voodoo magic for Mac OS X - this allows Mac OS X + * programs to access their bundle resources properly... + */ + + if ((linkbytes = readlink(command, linkpath, sizeof(linkpath) - 1)) > 0) + { + /* + * Yes, this is a symlink to the actual program, nul-terminate and + * use it... + */ + + linkpath[linkbytes] = '\0'; + + if (linkpath[0] == '/') + snprintf(processPath, sizeof(processPath), "CFProcessPath=%s", + linkpath); + else + snprintf(processPath, sizeof(processPath), "CFProcessPath=%s/%s", + dirname((char *)command), linkpath); + } + else + snprintf(processPath, sizeof(processPath), "CFProcessPath=%s", command); + + envp[0] = processPath; /* Replace string */ + } +#endif /* __APPLE__ */ + + /* + * Use helper program when we have a sandbox profile... + */ + + if (profile) + { + snprintf(cups_exec, sizeof(cups_exec), "%s/daemon/cups-exec", ServerBin); + + real_argv[0] = cups_exec; + real_argv[1] = profile; + real_argv[2] = (char *)command; + + for (i = 0; + i < (int)(sizeof(real_argv) / sizeof(real_argv[0]) - 4) && argv[i]; + i ++) + real_argv[i + 3] = argv[i]; + + real_argv[i + 3] = NULL; + + argv = real_argv; + exec_path = cups_exec; + } + + /* + * Block signals before forking... + */ + + cupsdHoldSignals(); + + if ((*pid = fork()) == 0) + { + /* + * Child process goes here; update stderr as needed... + */ + + if (errfd != 2) + { + if (errfd < 0) + errfd = open("/dev/null", O_WRONLY); + + if (errfd != 2) + { + dup2(errfd, 2); + close(errfd); + } + } + + /* + * Put this process in its own process group so that we can kill any child + * processes it creates. + */ + +#ifdef HAVE_SETPGID + if (!RunUser && setpgid(0, 0)) + exit(errno + 100); +#else + if (!RunUser && setpgrp()) + exit(errno + 100); +#endif /* HAVE_SETPGID */ + + /* + * Update the remaining file descriptors as needed... + */ + + if (infd != 0) + { + if (infd < 0) + infd = open("/dev/null", O_RDONLY); + + if (infd != 0) + { + dup2(infd, 0); + close(infd); + } + } + + if (outfd != 1) + { + if (outfd < 0) + outfd = open("/dev/null", O_WRONLY); + + if (outfd != 1) + { + dup2(outfd, 1); + close(outfd); + } + } + + if (backfd != 3 && backfd >= 0) + { + dup2(backfd, 3); + close(backfd); + fcntl(3, F_SETFL, O_NDELAY); + } + + if (sidefd != 4 && sidefd >= 0) + { + dup2(sidefd, 4); + close(sidefd); + fcntl(4, F_SETFL, O_NDELAY); + } + + /* + * Change the priority of the process based on the FilterNice setting. + * (this is not done for root processes...) + */ + + if (!root) + nice(FilterNice); + + /* + * Reset group membership to just the main one we belong to. + */ + + if (!RunUser && setgid(Group)) + exit(errno + 100); + + if (!RunUser && setgroups(1, &Group)) + exit(errno + 100); + + /* + * Change user to something "safe"... + */ + + if (!RunUser && user && setuid(user)) + exit(errno + 100); + + /* + * Change umask to restrict permissions on created files... + */ + + umask(077); + + /* + * Unblock signals before doing the exec... + */ + +#ifdef HAVE_SIGSET + sigset(SIGTERM, SIG_DFL); + sigset(SIGCHLD, SIG_DFL); + sigset(SIGPIPE, SIG_DFL); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + + sigemptyset(&action.sa_mask); + action.sa_handler = SIG_DFL; + + sigaction(SIGTERM, &action, NULL); + sigaction(SIGCHLD, &action, NULL); + sigaction(SIGPIPE, &action, NULL); +#else + signal(SIGTERM, SIG_DFL); + signal(SIGCHLD, SIG_DFL); + signal(SIGPIPE, SIG_DFL); +#endif /* HAVE_SIGSET */ + + cupsdReleaseSignals(); + + /* + * Execute the command; if for some reason this doesn't work, log an error + * exit with a non-zero value... + */ + + if (envp) + execve(exec_path, argv, envp); + else + execv(exec_path, argv); + + exit(errno + 100); + } + else if (*pid < 0) + { + /* + * Error - couldn't fork a new process! + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to fork %s - %s.", command, + strerror(errno)); + + *pid = 0; + } + else + { + if (!process_array) + process_array = cupsArrayNew((cups_array_func_t)compare_procs, NULL); + + if (process_array) + { + if ((proc = calloc(1, sizeof(cupsd_proc_t) + strlen(command))) != NULL) + { + proc->pid = *pid; + proc->job_id = job ? job->id : 0; + _cups_strcpy(proc->name, command); + + cupsArrayAdd(process_array, proc); + } + } + } + + cupsdReleaseSignals(); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdStartProcess(command=\"%s\", argv=%p, envp=%p, " + "infd=%d, outfd=%d, errfd=%d, backfd=%d, sidefd=%d, root=%d, " + "profile=%p, job=%p(%d), pid=%p) = %d", + command, argv, envp, infd, outfd, errfd, backfd, sidefd, + root, profile, job, job ? job->id : 0, pid, *pid); + + return (*pid); +} + + +/* + * 'compare_procs()' - Compare two processes. + */ + +static int /* O - Result of comparison */ +compare_procs(cupsd_proc_t *a, /* I - First process */ + cupsd_proc_t *b) /* I - Second process */ +{ + return (a->pid - b->pid); +} + + +#ifdef HAVE_SANDBOX_H +/* + * 'cupsd_requote()' - Make a regular-expression version of a string. + */ + +static char * /* O - Quoted string */ +cupsd_requote(char *dst, /* I - Destination buffer */ + const char *src, /* I - Source string */ + size_t dstsize) /* I - Size of destination buffer */ +{ + int ch; /* Current character */ + char *dstptr, /* Current position in buffer */ + *dstend; /* End of destination buffer */ + + + dstptr = dst; + dstend = dst + dstsize - 2; + + while (*src && dstptr < dstend) + { + ch = *src++; + + if (strchr(".?*()[]^$\\", ch)) + *dstptr++ = '\\'; + + *dstptr++ = ch; + } + + *dstptr = '\0'; + + return (dst); +} +#endif /* HAVE_SANDBOX_H */ + + +/* + * End of "$Id$". + */ diff --git a/scheduler/quotas.c b/scheduler/quotas.c new file mode 100644 index 0000000000..9f0b6c0a33 --- /dev/null +++ b/scheduler/quotas.c @@ -0,0 +1,244 @@ +/* + * "$Id$" + * + * Quota routines for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdFindQuota() - Find a quota record. + * cupsdFreeQuotas() - Free quotas for a printer. + * cupsdUpdateQuota() - Update quota data for the specified printer and user. + * add_quota() - Add a quota record for this printer and user. + * compare_quotas() - Compare two quota records... + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" + + +/* + * Local functions... + */ + +static cupsd_quota_t *add_quota(cupsd_printer_t *p, const char *username); +static int compare_quotas(const cupsd_quota_t *q1, + const cupsd_quota_t *q2); + + +/* + * 'cupsdFindQuota()' - Find a quota record. + */ + +cupsd_quota_t * /* O - Quota data */ +cupsdFindQuota( + cupsd_printer_t *p, /* I - Printer */ + const char *username) /* I - User */ +{ + cupsd_quota_t *q, /* Quota data pointer */ + match; /* Search data */ + char *ptr; /* Pointer into username */ + + + if (!p || !username) + return (NULL); + + strlcpy(match.username, username, sizeof(match.username)); + if ((ptr = strchr(match.username, '@')) != NULL) + *ptr = '\0'; /* Strip @domain/@KDC */ + + if ((q = (cupsd_quota_t *)cupsArrayFind(p->quotas, &match)) != NULL) + return (q); + else + return (add_quota(p, username)); +} + + +/* + * 'cupsdFreeQuotas()' - Free quotas for a printer. + */ + +void +cupsdFreeQuotas(cupsd_printer_t *p) /* I - Printer */ +{ + cupsd_quota_t *q; /* Current quota record */ + + + if (!p) + return; + + for (q = (cupsd_quota_t *)cupsArrayFirst(p->quotas); + q; + q = (cupsd_quota_t *)cupsArrayNext(p->quotas)) + free(q); + + cupsArrayDelete(p->quotas); + + p->quotas = NULL; +} + + +/* + * 'cupsdUpdateQuota()' - Update quota data for the specified printer and user. + */ + +cupsd_quota_t * /* O - Quota data */ +cupsdUpdateQuota( + cupsd_printer_t *p, /* I - Printer */ + const char *username, /* I - User */ + int pages, /* I - Number of pages */ + int k) /* I - Number of kilobytes */ +{ + cupsd_quota_t *q; /* Quota data */ + cupsd_job_t *job; /* Current job */ + time_t curtime; /* Current time */ + ipp_attribute_t *attr; /* Job attribute */ + + + if (!p || !username) + return (NULL); + + if (!p->k_limit && !p->page_limit) + return (NULL); + + if ((q = cupsdFindQuota(p, username)) == NULL) + return (NULL); + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "cupsdUpdateQuota: p=%s username=%s pages=%d k=%d", + p->name, username, pages, k); + + curtime = time(NULL); + + if (curtime < q->next_update) + { + q->page_count += pages; + q->k_count += k; + + return (q); + } + + if (p->quota_period) + curtime -= p->quota_period; + else + curtime = 0; + + q->next_update = 0; + q->page_count = 0; + q->k_count = 0; + + for (job = (cupsd_job_t *)cupsArrayFirst(Jobs); + job; + job = (cupsd_job_t *)cupsArrayNext(Jobs)) + { + /* + * We only care about the current printer/class and user... + */ + + if (_cups_strcasecmp(job->dest, p->name) != 0 || + _cups_strcasecmp(job->username, q->username) != 0) + continue; + + /* + * Make sure attributes are loaded; we always call cupsdLoadJob() to ensure + * the access_time member is updated so the job isn't unloaded right away... + */ + + if (!cupsdLoadJob(job)) + continue; + + if ((attr = ippFindAttribute(job->attrs, "time-at-completion", + IPP_TAG_INTEGER)) == NULL) + if ((attr = ippFindAttribute(job->attrs, "time-at-processing", + IPP_TAG_INTEGER)) == NULL) + attr = ippFindAttribute(job->attrs, "time-at-creation", + IPP_TAG_INTEGER); + + if (attr->values[0].integer < curtime) + { + /* + * This job is too old to count towards the quota, ignore it... + */ + + if (JobAutoPurge && !job->printer && job->state_value > IPP_JOB_STOPPED) + cupsdDeleteJob(job, CUPSD_JOB_PURGE); + + continue; + } + + if (q->next_update == 0) + q->next_update = attr->values[0].integer + p->quota_period; + + if ((attr = ippFindAttribute(job->attrs, "job-media-sheets-completed", + IPP_TAG_INTEGER)) != NULL) + q->page_count += attr->values[0].integer; + + if ((attr = ippFindAttribute(job->attrs, "job-k-octets", + IPP_TAG_INTEGER)) != NULL) + q->k_count += attr->values[0].integer; + } + + return (q); +} + + +/* + * 'add_quota()' - Add a quota record for this printer and user. + */ + +static cupsd_quota_t * /* O - Quota data */ +add_quota(cupsd_printer_t *p, /* I - Printer */ + const char *username) /* I - User */ +{ + cupsd_quota_t *q; /* New quota data */ + char *ptr; /* Pointer into username */ + + + if (!p || !username) + return (NULL); + + if (!p->quotas) + p->quotas = cupsArrayNew((cups_array_func_t)compare_quotas, NULL); + + if (!p->quotas) + return (NULL); + + if ((q = calloc(1, sizeof(cupsd_quota_t))) == NULL) + return (NULL); + + strlcpy(q->username, username, sizeof(q->username)); + if ((ptr = strchr(q->username, '@')) != NULL) + *ptr = '\0'; /* Strip @domain/@KDC */ + + cupsArrayAdd(p->quotas, q); + + return (q); +} + + +/* + * 'compare_quotas()' - Compare two quota records... + */ + +static int /* O - Result of comparison */ +compare_quotas(const cupsd_quota_t *q1, /* I - First quota record */ + const cupsd_quota_t *q2) /* I - Second quota record */ +{ + return (_cups_strcasecmp(q1->username, q2->username)); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/select.c b/scheduler/select.c new file mode 100644 index 0000000000..a19503d62d --- /dev/null +++ b/scheduler/select.c @@ -0,0 +1,951 @@ +/* + * "$Id$" + * + * Select abstraction functions for the CUPS scheduler. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 2006-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdAddSelect() - Add a file descriptor to the list. + * cupsdDoSelect() - Do a select-like operation. + * cupsdIsSelecting() - Determine whether we are monitoring a file + * descriptor. + * cupsdRemoveSelect() - Remove a file descriptor from the list. + * cupsdStartSelect() - Initialize the file polling engine. + * cupsdStopSelect() - Shutdown the file polling engine. + * compare_fds() - Compare file descriptors. + * find_fd() - Find an existing file descriptor record. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" + +#ifdef HAVE_EPOLL +# include +# include +#elif defined(HAVE_KQUEUE) +# include +# include +#elif defined(HAVE_POLL) +# include +#elif defined(__hpux) +# include +#else +# include +#endif /* HAVE_EPOLL */ + + +/* + * Design Notes for Poll/Select API in CUPSD + * ----------------------------------------- + * + * SUPPORTED APIS + * + * OS select poll epoll kqueue /dev/poll + * -------------- ------ ------ ------ ------ --------- + * AIX YES YES NO NO NO + * FreeBSD YES YES NO YES NO + * HP-UX YES YES NO NO NO + * IRIX YES YES NO NO NO + * Linux YES YES YES NO NO + * MacOS X YES YES NO YES NO + * NetBSD YES YES NO YES NO + * OpenBSD YES YES NO YES NO + * Solaris YES YES NO NO YES + * Tru64 YES YES NO NO NO + * Windows YES NO NO NO NO + * + * + * HIGH-LEVEL API + * + * typedef void (*cupsd_selfunc_t)(void *data); + * + * void cupsdStartSelect(void); + * void cupsdStopSelect(void); + * void cupsdAddSelect(int fd, cupsd_selfunc_t read_cb, + * cupsd_selfunc_t write_cb, void *data); + * void cupsdRemoveSelect(int fd); + * int cupsdDoSelect(int timeout); + * + * + * IMPLEMENTATION STRATEGY + * + * 0. Common Stuff + * a. CUPS array of file descriptor to callback functions + * and data + temporary array of removed fd's. + * b. cupsdStartSelect() creates the arrays + * c. cupsdStopSelect() destroys the arrays and all elements. + * d. cupsdAddSelect() adds to the array and allocates a + * new callback element. + * e. cupsdRemoveSelect() removes from the active array and + * adds to the inactive array. + * f. _cupsd_fd_t provides a reference-counted structure for + * tracking file descriptors that are monitored. + * g. cupsdDoSelect() frees all inactive FDs. + * + * 1. select() O(n) + * a. Input/Output fd_set variables, copied to working + * copies and then used with select(). + * b. Loop through CUPS array, using FD_ISSET and calling + * the read/write callbacks as needed. + * c. cupsdRemoveSelect() clears fd_set bit from main and + * working sets. + * d. cupsdStopSelect() frees all of the memory used by the + * CUPS array and fd_set's. + * + * 2. poll() - O(n log n) + * a. Regular array of pollfd, sorted the same as the CUPS + * array. + * b. Loop through pollfd array, call the corresponding + * read/write callbacks as needed. + * c. cupsdAddSelect() adds first to CUPS array and flags the + * pollfd array as invalid. + * d. cupsdDoSelect() rebuilds pollfd array as needed, calls + * poll(), then loops through the pollfd array looking up + * as needed. + * e. cupsdRemoveSelect() flags the pollfd array as invalid. + * f. cupsdStopSelect() frees all of the memory used by the + * CUPS array and pollfd array. + * + * 3. epoll() - O(n) + * a. cupsdStartSelect() creates epoll file descriptor using + * epoll_create() with the maximum fd count, and + * allocates an events buffer for the maximum fd count. + * b. cupsdAdd/RemoveSelect() uses epoll_ctl() to add + * (EPOLL_CTL_ADD) or remove (EPOLL_CTL_DEL) a single + * event using the level-triggered semantics. The event + * user data field is a pointer to the new callback array + * element. + * c. cupsdDoSelect() uses epoll_wait() with the global event + * buffer allocated in cupsdStartSelect() and then loops + * through the events, using the user data field to find + * the callback record. + * d. cupsdStopSelect() closes the epoll file descriptor and + * frees all of the memory used by the event buffer. + * + * 4. kqueue() - O(n) + * b. cupsdStartSelect() creates kqueue file descriptor + * using kqueue() function and allocates a global event + * buffer. + * c. cupsdAdd/RemoveSelect() uses EV_SET and kevent() to + * register the changes. The event user data field is a + * pointer to the new callback array element. + * d. cupsdDoSelect() uses kevent() to poll for events and + * loops through the events, using the user data field to + * find the callback record. + * e. cupsdStopSelect() closes the kqueue() file descriptor + * and frees all of the memory used by the event buffer. + * + * 5. /dev/poll - O(n log n) - NOT YET IMPLEMENTED + * a. cupsdStartSelect() opens /dev/poll and allocates an + * array of pollfd structs; on failure to open /dev/poll, + * revert to poll() system call. + * b. cupsdAddSelect() writes a single pollfd struct to + * /dev/poll with the new file descriptor and the + * POLLIN/POLLOUT flags. + * c. cupsdRemoveSelect() writes a single pollfd struct to + * /dev/poll with the file descriptor and the POLLREMOVE + * flag. + * d. cupsdDoSelect() uses the DP_POLL ioctl to retrieve + * events from /dev/poll and then loops through the + * returned pollfd array, looking up the file descriptors + * as needed. + * e. cupsdStopSelect() closes /dev/poll and frees the + * pollfd array. + * + * PERFORMANCE + * + * In tests using the "make test" target with option 0 (keep cupsd + * running) and the "testspeed" program with "-c 50 -r 1000", epoll() + * performed 5.5% slower than select(), followed by kqueue() at 16% + * slower than select() and poll() at 18% slower than select(). Similar + * results were seen with twice the number of client connections. + * + * The epoll() and kqueue() performance is likely limited by the + * number of system calls used to add/modify/remove file + * descriptors dynamically. Further optimizations may be possible + * in the area of limiting use of cupsdAddSelect() and + * cupsdRemoveSelect(), however extreme care will be needed to avoid + * excess CPU usage and deadlock conditions. + * + * We may be able to improve the poll() implementation simply by + * keeping the pollfd array sync'd with the _cupsd_fd_t array, as that + * will eliminate the rebuilding of the array whenever there is a + * change and eliminate the fd array lookups in the inner loop of + * cupsdDoSelect(). + * + * Since /dev/poll will never be able to use a shadow array, it may + * not make sense to implement support for it. ioctl() overhead will + * impact performance as well, so my guess would be that, for CUPS, + * /dev/poll will yield a net performance loss. + */ + +/* + * Local structures... + */ + +typedef struct _cupsd_fd_s +{ + int fd, /* File descriptor */ + use; /* Use count */ + cupsd_selfunc_t read_cb, /* Read callback */ + write_cb; /* Write callback */ + void *data; /* Data pointer for callbacks */ +} _cupsd_fd_t; + + +/* + * Local globals... + */ + +static cups_array_t *cupsd_fds = NULL; +#if defined(HAVE_EPOLL) || defined(HAVE_KQUEUE) +static cups_array_t *cupsd_inactive_fds = NULL; +static int cupsd_in_select = 0; +#endif /* HAVE_EPOLL || HAVE_KQUEUE */ + +#ifdef HAVE_KQUEUE +static int cupsd_kqueue_fd = -1, + cupsd_kqueue_changes = 0; +static struct kevent *cupsd_kqueue_events = NULL; +#elif defined(HAVE_POLL) +static int cupsd_alloc_pollfds = 0, + cupsd_update_pollfds = 0; +static struct pollfd *cupsd_pollfds = NULL; +# ifdef HAVE_EPOLL +static int cupsd_epoll_fd = -1; +static struct epoll_event *cupsd_epoll_events = NULL; +# endif /* HAVE_EPOLL */ +#else /* select() */ +static fd_set cupsd_global_input, + cupsd_global_output, + cupsd_current_input, + cupsd_current_output; +#endif /* HAVE_KQUEUE */ + + +/* + * Local functions... + */ + +static int compare_fds(_cupsd_fd_t *a, _cupsd_fd_t *b); +static _cupsd_fd_t *find_fd(int fd); +#define release_fd(f) { \ + (f)->use --; \ + if (!(f)->use) free((f));\ + } +#define retain_fd(f) (f)->use++ + + +/* + * 'cupsdAddSelect()' - Add a file descriptor to the list. + */ + +int /* O - 1 on success, 0 on error */ +cupsdAddSelect(int fd, /* I - File descriptor */ + cupsd_selfunc_t read_cb, /* I - Read callback */ + cupsd_selfunc_t write_cb,/* I - Write callback */ + void *data) /* I - Data to pass to callback */ +{ + _cupsd_fd_t *fdptr; /* File descriptor record */ +#ifdef HAVE_EPOLL + int added; /* 1 if added, 0 if modified */ +#endif /* HAVE_EPOLL */ + + + /* + * Range check input... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdAddSelect(fd=%d, read_cb=%p, write_cb=%p, data=%p)", + fd, read_cb, write_cb, data); + + if (fd < 0) + return (0); + + /* + * See if this FD has already been added... + */ + + if ((fdptr = find_fd(fd)) == NULL) + { + /* + * No, add a new entry... + */ + + if ((fdptr = calloc(1, sizeof(_cupsd_fd_t))) == NULL) + return (0); + + fdptr->fd = fd; + fdptr->use = 1; + + if (!cupsArrayAdd(cupsd_fds, fdptr)) + { + cupsdLogMessage(CUPSD_LOG_EMERG, "Unable to add fd %d to array!", fd); + free(fdptr); + return (0); + } + +#ifdef HAVE_EPOLL + added = 1; + } + else + added = 0; +#else + } +#endif /* HAVE_EPOLL */ + +#ifdef HAVE_KQUEUE + { + struct kevent event; /* Event data */ + struct timespec timeout; /* Timeout value */ + + + timeout.tv_sec = 0; + timeout.tv_nsec = 0; + + if (fdptr->read_cb != read_cb) + { + if (read_cb) + EV_SET(&event, fd, EVFILT_READ, EV_ADD, 0, 0, fdptr); + else + EV_SET(&event, fd, EVFILT_READ, EV_DELETE, 0, 0, fdptr); + + if (kevent(cupsd_kqueue_fd, &event, 1, NULL, 0, &timeout)) + { + cupsdLogMessage(CUPSD_LOG_EMERG, "kevent() returned %s", + strerror(errno)); + return (0); + } + } + + if (fdptr->write_cb != write_cb) + { + if (write_cb) + EV_SET(&event, fd, EVFILT_WRITE, EV_ADD, 0, 0, fdptr); + else + EV_SET(&event, fd, EVFILT_WRITE, EV_DELETE, 0, 0, fdptr); + + if (kevent(cupsd_kqueue_fd, &event, 1, NULL, 0, &timeout)) + { + cupsdLogMessage(CUPSD_LOG_EMERG, "kevent() returned %s", + strerror(errno)); + return (0); + } + } + } + +#elif defined(HAVE_POLL) +# ifdef HAVE_EPOLL + if (cupsd_epoll_fd >= 0) + { + struct epoll_event event; /* Event data */ + + + event.events = 0; + + if (read_cb) + event.events |= EPOLLIN; + + if (write_cb) + event.events |= EPOLLOUT; + + event.data.ptr = fdptr; + + if (epoll_ctl(cupsd_epoll_fd, added ? EPOLL_CTL_ADD : EPOLL_CTL_MOD, fd, + &event)) + { + close(cupsd_epoll_fd); + cupsd_epoll_fd = -1; + cupsd_update_pollfds = 1; + } + } + else +# endif /* HAVE_EPOLL */ + + cupsd_update_pollfds = 1; + +#else /* select() */ + /* + * Add or remove the file descriptor in the input and output sets + * for select()... + */ + + if (read_cb) + FD_SET(fd, &cupsd_global_input); + else + { + FD_CLR(fd, &cupsd_global_input); + FD_CLR(fd, &cupsd_current_input); + } + + if (write_cb) + FD_SET(fd, &cupsd_global_output); + else + { + FD_CLR(fd, &cupsd_global_output); + FD_CLR(fd, &cupsd_current_output); + } +#endif /* HAVE_KQUEUE */ + + /* + * Save the (new) read and write callbacks... + */ + + fdptr->read_cb = read_cb; + fdptr->write_cb = write_cb; + fdptr->data = data; + + return (1); +} + + +/* + * 'cupsdDoSelect()' - Do a select-like operation. + */ + +int /* O - Number of files or -1 on error */ +cupsdDoSelect(long timeout) /* I - Timeout in seconds */ +{ + int nfds; /* Number of file descriptors */ + _cupsd_fd_t *fdptr; /* Current file descriptor */ +#ifdef HAVE_KQUEUE + int i; /* Looping var */ + struct kevent *event; /* Current event */ + struct timespec ktimeout; /* kevent() timeout */ + + + cupsd_in_select = 1; + + if (timeout >= 0 && timeout < 86400) + { + ktimeout.tv_sec = timeout; + ktimeout.tv_nsec = 0; + + nfds = kevent(cupsd_kqueue_fd, NULL, 0, cupsd_kqueue_events, MaxFDs, + &ktimeout); + } + else + nfds = kevent(cupsd_kqueue_fd, NULL, 0, cupsd_kqueue_events, MaxFDs, NULL); + + cupsd_kqueue_changes = 0; + + for (i = nfds, event = cupsd_kqueue_events; i > 0; i --, event ++) + { + fdptr = (_cupsd_fd_t *)event->udata; + + if (cupsArrayFind(cupsd_inactive_fds, fdptr)) + continue; + + retain_fd(fdptr); + + if (fdptr->read_cb && event->filter == EVFILT_READ) + (*(fdptr->read_cb))(fdptr->data); + + if (fdptr->use > 1 && fdptr->write_cb && event->filter == EVFILT_WRITE && + !cupsArrayFind(cupsd_inactive_fds, fdptr)) + (*(fdptr->write_cb))(fdptr->data); + + release_fd(fdptr); + } + +#elif defined(HAVE_POLL) + struct pollfd *pfd; /* Current pollfd structure */ + int count; /* Number of file descriptors */ + + +# ifdef HAVE_EPOLL + cupsd_in_select = 1; + + if (cupsd_epoll_fd >= 0) + { + int i; /* Looping var */ + struct epoll_event *event; /* Current event */ + + + if (timeout >= 0 && timeout < 86400) + nfds = epoll_wait(cupsd_epoll_fd, cupsd_epoll_events, MaxFDs, + timeout * 1000); + else + nfds = epoll_wait(cupsd_epoll_fd, cupsd_epoll_events, MaxFDs, -1); + + if (nfds < 0 && errno != EINTR) + { + close(cupsd_epoll_fd); + cupsd_epoll_fd = -1; + } + else + { + for (i = nfds, event = cupsd_epoll_events; i > 0; i --, event ++) + { + fdptr = (_cupsd_fd_t *)event->data.ptr; + + if (cupsArrayFind(cupsd_inactive_fds, fdptr)) + continue; + + retain_fd(fdptr); + + if (fdptr->read_cb && (event->events & (EPOLLIN | EPOLLERR | EPOLLHUP))) + (*(fdptr->read_cb))(fdptr->data); + + if (fdptr->use > 1 && fdptr->write_cb && + (event->events & (EPOLLOUT | EPOLLERR | EPOLLHUP)) && + !cupsArrayFind(cupsd_inactive_fds, fdptr)) + (*(fdptr->write_cb))(fdptr->data); + + release_fd(fdptr); + } + + goto release_inactive; + } + } +# endif /* HAVE_EPOLL */ + + count = cupsArrayCount(cupsd_fds); + + if (cupsd_update_pollfds) + { + /* + * Update the cupsd_pollfds array to match the current FD array... + */ + + cupsd_update_pollfds = 0; + + /* + * (Re)allocate memory as needed... + */ + + if (count > cupsd_alloc_pollfds) + { + int allocfds = count + 16; + + + if (cupsd_pollfds) + pfd = realloc(cupsd_pollfds, allocfds * sizeof(struct pollfd)); + else + pfd = malloc(allocfds * sizeof(struct pollfd)); + + if (!pfd) + { + cupsdLogMessage(CUPSD_LOG_EMERG, + "Unable to allocate %d bytes for polling!", + (int)(allocfds * sizeof(struct pollfd))); + + return (-1); + } + + cupsd_pollfds = pfd; + cupsd_alloc_pollfds = allocfds; + } + + /* + * Rebuild the array... + */ + + for (fdptr = (_cupsd_fd_t *)cupsArrayFirst(cupsd_fds), pfd = cupsd_pollfds; + fdptr; + fdptr = (_cupsd_fd_t *)cupsArrayNext(cupsd_fds), pfd ++) + { + pfd->fd = fdptr->fd; + pfd->events = 0; + + if (fdptr->read_cb) + pfd->events |= POLLIN; + + if (fdptr->write_cb) + pfd->events |= POLLOUT; + } + } + + if (timeout >= 0 && timeout < 86400) + nfds = poll(cupsd_pollfds, count, timeout * 1000); + else + nfds = poll(cupsd_pollfds, count, -1); + + if (nfds > 0) + { + /* + * Do callbacks for each file descriptor... + */ + + for (pfd = cupsd_pollfds; count > 0; pfd ++, count --) + { + if (!pfd->revents) + continue; + + if ((fdptr = find_fd(pfd->fd)) == NULL) + continue; + + retain_fd(fdptr); + + if (fdptr->read_cb && (pfd->revents & (POLLIN | POLLERR | POLLHUP))) + (*(fdptr->read_cb))(fdptr->data); + + if (fdptr->use > 1 && fdptr->write_cb && + (pfd->revents & (POLLOUT | POLLERR | POLLHUP))) + (*(fdptr->write_cb))(fdptr->data); + + release_fd(fdptr); + } + } + +#else /* select() */ + struct timeval stimeout; /* Timeout for select() */ + int maxfd; /* Maximum file descriptor */ + + + /* + * Figure out the highest file descriptor number... + */ + + if ((fdptr = (_cupsd_fd_t *)cupsArrayLast(cupsd_fds)) == NULL) + maxfd = 1; + else + maxfd = fdptr->fd + 1; + + /* + * Do the select()... + */ + + cupsd_current_input = cupsd_global_input; + cupsd_current_output = cupsd_global_output; + + if (timeout >= 0 && timeout < 86400) + { + stimeout.tv_sec = timeout; + stimeout.tv_usec = 0; + + nfds = select(maxfd, &cupsd_current_input, &cupsd_current_output, NULL, + &stimeout); + } + else + nfds = select(maxfd, &cupsd_current_input, &cupsd_current_output, NULL, + NULL); + + if (nfds > 0) + { + /* + * Do callbacks for each file descriptor... + */ + + for (fdptr = (_cupsd_fd_t *)cupsArrayFirst(cupsd_fds); + fdptr; + fdptr = (_cupsd_fd_t *)cupsArrayNext(cupsd_fds)) + { + retain_fd(fdptr); + + if (fdptr->read_cb && FD_ISSET(fdptr->fd, &cupsd_current_input)) + (*(fdptr->read_cb))(fdptr->data); + + if (fdptr->use > 1 && fdptr->write_cb && + FD_ISSET(fdptr->fd, &cupsd_current_output)) + (*(fdptr->write_cb))(fdptr->data); + + release_fd(fdptr); + } + } + +#endif /* HAVE_KQUEUE */ + +#if defined(HAVE_EPOLL) || defined(HAVE_KQUEUE) + /* + * Release all inactive file descriptors... + */ + +# ifndef HAVE_KQUEUE + release_inactive: +# endif /* !HAVE_KQUEUE */ + + cupsd_in_select = 0; + + for (fdptr = (_cupsd_fd_t *)cupsArrayFirst(cupsd_inactive_fds); + fdptr; + fdptr = (_cupsd_fd_t *)cupsArrayNext(cupsd_inactive_fds)) + { + cupsArrayRemove(cupsd_inactive_fds, fdptr); + release_fd(fdptr); + } +#endif /* HAVE_EPOLL || HAVE_KQUEUE */ + + /* + * Return the number of file descriptors handled... + */ + + return (nfds); +} + + +#ifdef CUPSD_IS_SELECTING +/* + * 'cupsdIsSelecting()' - Determine whether we are monitoring a file + * descriptor. + */ + +int /* O - 1 if selecting, 0 otherwise */ +cupsdIsSelecting(int fd) /* I - File descriptor */ +{ + return (find_fd(fd) != NULL); +} +#endif /* CUPSD_IS_SELECTING */ + + +/* + * 'cupsdRemoveSelect()' - Remove a file descriptor from the list. + */ + +void +cupsdRemoveSelect(int fd) /* I - File descriptor */ +{ + _cupsd_fd_t *fdptr; /* File descriptor record */ +#ifdef HAVE_EPOLL + struct epoll_event event; /* Event data */ +#elif defined(HAVE_KQUEUE) + struct kevent event; /* Event data */ + struct timespec timeout; /* Timeout value */ +#elif defined(HAVE_POLL) + /* No variables for poll() */ +#endif /* HAVE_EPOLL */ + + + /* + * Range check input... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdRemoveSelect(fd=%d)", fd); + + if (fd < 0) + return; + + /* + * Find the file descriptor... + */ + + if ((fdptr = find_fd(fd)) == NULL) + return; + +#ifdef HAVE_EPOLL + if (epoll_ctl(cupsd_epoll_fd, EPOLL_CTL_DEL, fd, &event)) + { + close(cupsd_epoll_fd); + cupsd_epoll_fd = -1; + cupsd_update_pollfds = 1; + } + +#elif defined(HAVE_KQUEUE) + timeout.tv_sec = 0; + timeout.tv_nsec = 0; + + if (fdptr->read_cb) + { + EV_SET(&event, fd, EVFILT_READ, EV_DELETE, 0, 0, fdptr); + + if (kevent(cupsd_kqueue_fd, &event, 1, NULL, 0, &timeout)) + { + cupsdLogMessage(CUPSD_LOG_EMERG, "kevent() returned %s", + strerror(errno)); + goto cleanup; + } + } + + if (fdptr->write_cb) + { + EV_SET(&event, fd, EVFILT_WRITE, EV_DELETE, 0, 0, fdptr); + + if (kevent(cupsd_kqueue_fd, &event, 1, NULL, 0, &timeout)) + { + cupsdLogMessage(CUPSD_LOG_EMERG, "kevent() returned %s", + strerror(errno)); + goto cleanup; + } + } + +#elif defined(HAVE_POLL) + /* + * Update the pollfds array... + */ + + cupsd_update_pollfds = 1; + +#else /* select() */ + FD_CLR(fd, &cupsd_global_input); + FD_CLR(fd, &cupsd_global_output); + FD_CLR(fd, &cupsd_current_input); + FD_CLR(fd, &cupsd_current_output); +#endif /* HAVE_EPOLL */ + +#ifdef HAVE_KQUEUE + cleanup: +#endif /* HAVE_KQUEUE */ + + /* + * Remove the file descriptor from the active array and add to the + * inactive array (or release, if we don't need the inactive array...) + */ + + cupsArrayRemove(cupsd_fds, fdptr); + +#if defined(HAVE_EPOLL) || defined(HAVE_KQUEUE) + if (cupsd_in_select) + cupsArrayAdd(cupsd_inactive_fds, fdptr); + else +#endif /* HAVE_EPOLL || HAVE_KQUEUE */ + + release_fd(fdptr); +} + + +/* + * 'cupsdStartSelect()' - Initialize the file polling engine. + */ + +void +cupsdStartSelect(void) +{ + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdStartSelect()"); + + cupsd_fds = cupsArrayNew((cups_array_func_t)compare_fds, NULL); + +#if defined(HAVE_EPOLL) || defined(HAVE_KQUEUE) + cupsd_inactive_fds = cupsArrayNew((cups_array_func_t)compare_fds, NULL); +#endif /* HAVE_EPOLL || HAVE_KQUEUE */ + +#ifdef HAVE_EPOLL + cupsd_epoll_fd = epoll_create(MaxFDs); + cupsd_epoll_events = calloc(MaxFDs, sizeof(struct epoll_event)); + cupsd_update_pollfds = 0; + +#elif defined(HAVE_KQUEUE) + cupsd_kqueue_fd = kqueue(); + cupsd_kqueue_changes = 0; + cupsd_kqueue_events = calloc(MaxFDs, sizeof(struct kevent)); + +#elif defined(HAVE_POLL) + cupsd_update_pollfds = 0; + +#else /* select() */ + FD_ZERO(&cupsd_global_input); + FD_ZERO(&cupsd_global_output); +#endif /* HAVE_EPOLL */ +} + + +/* + * 'cupsdStopSelect()' - Shutdown the file polling engine. + */ + +void +cupsdStopSelect(void) +{ + _cupsd_fd_t *fdptr; /* Current file descriptor */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "cupsdStopSelect()"); + + for (fdptr = (_cupsd_fd_t *)cupsArrayFirst(cupsd_fds); + fdptr; + fdptr = (_cupsd_fd_t *)cupsArrayNext(cupsd_fds)) + free(fdptr); + + cupsArrayDelete(cupsd_fds); + cupsd_fds = NULL; + +#if defined(HAVE_EPOLL) || defined(HAVE_KQUEUE) + cupsArrayDelete(cupsd_inactive_fds); + cupsd_inactive_fds = NULL; +#endif /* HAVE_EPOLL || HAVE_KQUEUE */ + +#ifdef HAVE_KQUEUE + if (cupsd_kqueue_events) + { + free(cupsd_kqueue_events); + cupsd_kqueue_events = NULL; + } + + if (cupsd_kqueue_fd >= 0) + { + close(cupsd_kqueue_fd); + cupsd_kqueue_fd = -1; + } + + cupsd_kqueue_changes = 0; + +#elif defined(HAVE_POLL) +# ifdef HAVE_EPOLL + if (cupsd_epoll_events) + { + free(cupsd_epoll_events); + cupsd_epoll_events = NULL; + } + + if (cupsd_epoll_fd >= 0) + { + close(cupsd_epoll_fd); + cupsd_epoll_fd = -1; + } +# endif /* HAVE_EPOLL */ + + if (cupsd_pollfds) + { + free(cupsd_pollfds); + cupsd_pollfds = NULL; + cupsd_alloc_pollfds = 0; + } + + cupsd_update_pollfds = 0; + +#else /* select() */ + FD_ZERO(&cupsd_global_input); + FD_ZERO(&cupsd_global_output); +#endif /* HAVE_EPOLL */ +} + + +/* + * 'compare_fds()' - Compare file descriptors. + */ + +static int /* O - Result of comparison */ +compare_fds(_cupsd_fd_t *a, /* I - First file descriptor */ + _cupsd_fd_t *b) /* I - Second file descriptor */ +{ + return (a->fd - b->fd); +} + + +/* + * 'find_fd()' - Find an existing file descriptor record. + */ + +static _cupsd_fd_t * /* O - FD record pointer or NULL */ +find_fd(int fd) /* I - File descriptor */ +{ + _cupsd_fd_t *fdptr, /* Matching record (if any) */ + key; /* Search key */ + + + cupsArraySave(cupsd_fds); + + key.fd = fd; + fdptr = (_cupsd_fd_t *)cupsArrayFind(cupsd_fds, &key); + + cupsArrayRestore(cupsd_fds); + + return (fdptr); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/server.c b/scheduler/server.c new file mode 100644 index 0000000000..4d0eb46026 --- /dev/null +++ b/scheduler/server.c @@ -0,0 +1,172 @@ +/* + * "$Id$" + * + * Server start/stop routines for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdStartServer() - Start the server. + * cupsdStopServer() - Stop the server. + */ + +/* + * Include necessary headers... + */ + +#include +#include "cupsd.h" +#include +#ifdef HAVE_NOTIFY_H +# include +#endif /* HAVE_NOTIFY_H */ + + +/* + * Local globals... + */ + +static int started = 0; /* Did we start the server already? */ + + +/* + * 'cupsdStartServer()' - Start the server. + */ + +void +cupsdStartServer(void) +{ + /* + * Create the default security profile... + */ + + DefaultProfile = cupsdCreateProfile(0); + + /* + * Startup all the networking stuff... + */ + + cupsdStartListening(); + cupsdStartBrowsing(); + + /* + * Create a pipe for CGI processes... + */ + + if (cupsdOpenPipe(CGIPipes)) + cupsdLogMessage(CUPSD_LOG_ERROR, + "cupsdStartServer: Unable to create pipes for CGI status!"); + else + { + CGIStatusBuffer = cupsdStatBufNew(CGIPipes[0], "[CGI]"); + + cupsdAddSelect(CGIPipes[0], (cupsd_selfunc_t)cupsdUpdateCGI, NULL, NULL); + } + + /* + * Mark that the server has started and printers and jobs may be changed... + */ + + LastEvent = CUPSD_EVENT_PRINTER_CHANGED | CUPSD_EVENT_JOB_STATE_CHANGED | + CUPSD_EVENT_SERVER_STARTED; + started = 1; + + cupsdSetBusyState(); +} + + +/* + * 'cupsdStopServer()' - Stop the server. + */ + +void +cupsdStopServer(void) +{ + if (!started) + return; + + /* + * Close all network clients and stop all jobs... + */ + + cupsdCloseAllClients(); + cupsdStopListening(); + cupsdStopBrowsing(); + cupsdStopAllNotifiers(); + cupsdDeleteAllCerts(); + + if (Clients) + { + cupsArrayDelete(Clients); + Clients = NULL; + } + + /* + * Close the pipe for CGI processes... + */ + + if (CGIPipes[0] >= 0) + { + cupsdRemoveSelect(CGIPipes[0]); + + cupsdStatBufDelete(CGIStatusBuffer); + close(CGIPipes[1]); + + CGIPipes[0] = -1; + CGIPipes[1] = -1; + } + + /* + * Close all log files... + */ + + if (AccessFile != NULL) + { + cupsFileClose(AccessFile); + + AccessFile = NULL; + } + + if (ErrorFile != NULL) + { + cupsFileClose(ErrorFile); + + ErrorFile = NULL; + } + + if (PageFile != NULL) + { + cupsFileClose(PageFile); + + PageFile = NULL; + } + + /* + * Delete the default security profile... + */ + + cupsdDestroyProfile(DefaultProfile); + DefaultProfile = NULL; + + /* + * Write out any dirty files... + */ + + if (DirtyFiles) + cupsdCleanDirty(); + + started = 0; +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/statbuf.c b/scheduler/statbuf.c new file mode 100644 index 0000000000..dc88be4495 --- /dev/null +++ b/scheduler/statbuf.c @@ -0,0 +1,330 @@ +/* + * "$Id$" + * + * Status buffer routines for the CUPS scheduler. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdStatBufDelete() - Destroy a status buffer. + * cupsdStatBufNew() - Create a new status buffer. + * cupsdStatBufUpdate() - Update the status buffer. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" +#include + + +/* + * 'cupsdStatBufDelete()' - Destroy a status buffer. + */ + +void +cupsdStatBufDelete(cupsd_statbuf_t *sb) /* I - Status buffer */ +{ + /* + * Range check input... + */ + + if (!sb) + return; + + /* + * Close the status pipe and free memory used... + */ + + close(sb->fd); + + free(sb); +} + + +/* + * 'cupsdStatBufNew()' - Create a new status buffer. + */ + +cupsd_statbuf_t * /* O - New status buffer */ +cupsdStatBufNew(int fd, /* I - File descriptor of pipe */ + const char *prefix, /* I - Printf-style prefix string */ + ...) /* I - Additional args as needed */ +{ + cupsd_statbuf_t *sb; /* New status buffer */ + va_list ap; /* Argument list */ + + + /* + * Range check input... + */ + + if (fd < 0) + return (NULL); + + /* + * Allocate the status buffer... + */ + + if ((sb = calloc(1, sizeof(cupsd_statbuf_t))) != NULL) + { + /* + * Assign the file descriptor... + */ + + sb->fd = fd; + + /* + * Format the prefix string, if any. This is usually "[Job 123]" + * or "[Sub 123]", and so forth. + */ + + if (prefix) + { + /* + * Printf-style prefix string... + */ + + va_start(ap, prefix); + vsnprintf(sb->prefix, sizeof(sb->prefix), prefix, ap); + va_end(ap); + } + else + { + /* + * No prefix string... + */ + + sb->prefix[0] = '\0'; + } + } + + return (sb); +} + + +/* + * 'cupsdStatBufUpdate()' - Update the status buffer. + */ + +char * /* O - Line from buffer, "", or NULL */ +cupsdStatBufUpdate( + cupsd_statbuf_t *sb, /* I - Status buffer */ + int *loglevel, /* O - Log level */ + char *line, /* I - Line buffer */ + int linelen) /* I - Size of line buffer */ +{ + int bytes; /* Number of bytes read */ + char *lineptr, /* Pointer to end of line in buffer */ + *message; /* Pointer to message text */ + + + /* + * Check if the buffer already contains a full line... + */ + + if ((lineptr = strchr(sb->buffer, '\n')) == NULL) + { + /* + * No, read more data... + */ + + if ((bytes = read(sb->fd, sb->buffer + sb->bufused, + CUPSD_SB_BUFFER_SIZE - sb->bufused - 1)) > 0) + { + sb->bufused += bytes; + sb->buffer[sb->bufused] = '\0'; + + /* + * Guard against a line longer than the max buffer size... + */ + + if ((lineptr = strchr(sb->buffer, '\n')) == NULL && + sb->bufused == (CUPSD_SB_BUFFER_SIZE - 1)) + lineptr = sb->buffer + sb->bufused; + } + else if (bytes < 0 && errno == EINTR) + { + /* + * Return an empty line if we are interrupted... + */ + + *loglevel = CUPSD_LOG_NONE; + line[0] = '\0'; + + return (line); + } + else + { + /* + * End-of-file, so use the whole buffer... + */ + + lineptr = sb->buffer + sb->bufused; + *lineptr = '\0'; + } + + /* + * Final check for end-of-file... + */ + + if (sb->bufused == 0 && bytes == 0) + lineptr = NULL; + } + + if (!lineptr) + { + /* + * End of file... + */ + + *loglevel = CUPSD_LOG_NONE; + line[0] = '\0'; + + return (NULL); + } + + /* + * Terminate the line and process it... + */ + + *lineptr++ = '\0'; + + /* + * Figure out the logging level... + */ + + if (!strncmp(sb->buffer, "EMERG:", 6)) + { + *loglevel = CUPSD_LOG_EMERG; + message = sb->buffer + 6; + } + else if (!strncmp(sb->buffer, "ALERT:", 6)) + { + *loglevel = CUPSD_LOG_ALERT; + message = sb->buffer + 6; + } + else if (!strncmp(sb->buffer, "CRIT:", 5)) + { + *loglevel = CUPSD_LOG_CRIT; + message = sb->buffer + 5; + } + else if (!strncmp(sb->buffer, "ERROR:", 6)) + { + *loglevel = CUPSD_LOG_ERROR; + message = sb->buffer + 6; + } + else if (!strncmp(sb->buffer, "WARNING:", 8)) + { + *loglevel = CUPSD_LOG_WARN; + message = sb->buffer + 8; + } + else if (!strncmp(sb->buffer, "NOTICE:", 7)) + { + *loglevel = CUPSD_LOG_NOTICE; + message = sb->buffer + 7; + } + else if (!strncmp(sb->buffer, "INFO:", 5)) + { + *loglevel = CUPSD_LOG_INFO; + message = sb->buffer + 5; + } + else if (!strncmp(sb->buffer, "DEBUG:", 6)) + { + *loglevel = CUPSD_LOG_DEBUG; + message = sb->buffer + 6; + } + else if (!strncmp(sb->buffer, "DEBUG2:", 7)) + { + *loglevel = CUPSD_LOG_DEBUG2; + message = sb->buffer + 7; + } + else if (!strncmp(sb->buffer, "PAGE:", 5)) + { + *loglevel = CUPSD_LOG_PAGE; + message = sb->buffer + 5; + } + else if (!strncmp(sb->buffer, "STATE:", 6)) + { + *loglevel = CUPSD_LOG_STATE; + message = sb->buffer + 6; + } + else if (!strncmp(sb->buffer, "ATTR:", 5)) + { + *loglevel = CUPSD_LOG_ATTR; + message = sb->buffer + 5; + } + else if (!strncmp(sb->buffer, "PPD:", 4)) + { + *loglevel = CUPSD_LOG_PPD; + message = sb->buffer + 4; + } + else + { + *loglevel = CUPSD_LOG_DEBUG; + message = sb->buffer; + } + + /* + * Skip leading whitespace in the message... + */ + + while (isspace(*message & 255)) + message ++; + + /* + * Send it to the log file as needed... + */ + + if (sb->prefix[0]) + { + if (*loglevel > CUPSD_LOG_NONE && + (*loglevel != CUPSD_LOG_INFO || LogLevel >= CUPSD_LOG_DEBUG)) + { + /* + * General status message; send it to the error_log file... + */ + + if (message[0] == '[') + cupsdLogMessage(*loglevel, "%s", message); + else + cupsdLogMessage(*loglevel, "%s %s", sb->prefix, message); + } + else if (*loglevel < CUPSD_LOG_NONE && LogLevel >= CUPSD_LOG_DEBUG) + cupsdLogMessage(CUPSD_LOG_DEBUG2, "%s %s", sb->prefix, sb->buffer); + } + + /* + * Copy the message to the line buffer... + */ + + strlcpy(line, message, linelen); + + /* + * Copy over the buffer data we've used up... + */ + + if (lineptr < sb->buffer + sb->bufused) + _cups_strcpy(sb->buffer, lineptr); + + sb->bufused -= lineptr - sb->buffer; + + if (sb->bufused < 0) + sb->bufused = 0; + + return (line); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/statbuf.h b/scheduler/statbuf.h new file mode 100644 index 0000000000..212acdf18c --- /dev/null +++ b/scheduler/statbuf.h @@ -0,0 +1,49 @@ +/* + * "$Id$" + * + * Status buffer definitions for the CUPS scheduler. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + + +/* + * Constants... + */ + +#define CUPSD_SB_BUFFER_SIZE 2048 /* Bytes for job status buffer */ + + +/* + * Types and structures... + */ + +typedef struct /**** Status buffer */ +{ + int fd; /* File descriptor to read from */ + char prefix[64]; /* Prefix for log messages */ + int bufused; /* How much is used in buffer */ + char buffer[CUPSD_SB_BUFFER_SIZE]; /* Buffer */ +} cupsd_statbuf_t; + + +/* + * Prototypes... + */ + +extern void cupsdStatBufDelete(cupsd_statbuf_t *sb); +extern cupsd_statbuf_t *cupsdStatBufNew(int fd, const char *prefix, ...); +extern char *cupsdStatBufUpdate(cupsd_statbuf_t *sb, int *loglevel, + char *line, int linelen); + + +/* + * End of "$Id$". + */ diff --git a/scheduler/subscriptions.c b/scheduler/subscriptions.c new file mode 100644 index 0000000000..90f88daba8 --- /dev/null +++ b/scheduler/subscriptions.c @@ -0,0 +1,1638 @@ +/* + * "$Id$" + * + * Subscription routines for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdAddEvent() - Add an event to the global event cache. + * cupsdAddSubscription() - Add a new subscription object. + * cupsdDeleteAllSubscriptions() - Delete all subscriptions. + * cupsdDeleteSubscription() - Delete a subscription object. + * cupsdEventName() - Return a single event name. + * cupsdEventValue() - Return the event mask value for a name. + * cupsdExpireSubscriptions() - Expire old subscription objects. + * cupsdFindSubscription() - Find a subscription by ID. + * cupsdLoadAllSubscriptions() - Load all subscriptions from the .conf file. + * cupsdSaveAllSubscriptions() - Save all subscriptions to the .conf file. + * cupsdStopAllNotifiers() - Stop all notifier processes. + * cupsd_compare_subscriptions() - Compare two subscriptions. + * cupsd_delete_event() - Delete a single event... + * cupsd_send_dbus() - Send a DBUS notification... + * cupsd_send_notification() - Send a notification for the specified + * event. + * cupsd_start_notifier() - Start a notifier subprocess... + * cupsd_update_notifier() - Read messages from notifiers. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" +#ifdef HAVE_DBUS +# include +# ifdef HAVE_DBUS_MESSAGE_ITER_INIT_APPEND +# define dbus_message_append_iter_init dbus_message_iter_init_append +# define dbus_message_iter_append_string(i,v) dbus_message_iter_append_basic(i, DBUS_TYPE_STRING, &(v)) +# define dbus_message_iter_append_uint32(i,v) dbus_message_iter_append_basic(i, DBUS_TYPE_UINT32, &(v)) +# endif /* HAVE_DBUS_MESSAGE_ITER_INIT_APPEND */ +#endif /* HAVE_DBUS */ + + +/* + * Local functions... + */ + +static int cupsd_compare_subscriptions(cupsd_subscription_t *first, + cupsd_subscription_t *second, + void *unused); +static void cupsd_delete_event(cupsd_event_t *event); +#ifdef HAVE_DBUS +static void cupsd_send_dbus(cupsd_eventmask_t event, cupsd_printer_t *dest, + cupsd_job_t *job); +#endif /* HAVE_DBUS */ +static void cupsd_send_notification(cupsd_subscription_t *sub, + cupsd_event_t *event); +static void cupsd_start_notifier(cupsd_subscription_t *sub); +static void cupsd_update_notifier(void); + + +/* + * 'cupsdAddEvent()' - Add an event to the global event cache. + */ + +void +cupsdAddEvent( + cupsd_eventmask_t event, /* I - Event */ + cupsd_printer_t *dest, /* I - Printer associated with event */ + cupsd_job_t *job, /* I - Job associated with event */ + const char *text, /* I - Notification text */ + ...) /* I - Additional arguments as needed */ +{ + va_list ap; /* Pointer to additional arguments */ + char ftext[1024]; /* Formatted text buffer */ + ipp_attribute_t *attr; /* Printer/job attribute */ + cupsd_event_t *temp; /* New event pointer */ + cupsd_subscription_t *sub; /* Current subscription */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsdAddEvent(event=%s, dest=%p(%s), job=%p(%d), text=\"%s\", ...)", + cupsdEventName(event), dest, dest ? dest->name : "", + job, job ? job->id : 0, text); + + /* + * Keep track of events with any OS-supplied notification mechanisms... + */ + + LastEvent |= event; + +#ifdef HAVE_DBUS + cupsd_send_dbus(event, dest, job); +#endif /* HAVE_DBUS */ + + /* + * Return if we aren't keeping events... + */ + + if (MaxEvents <= 0) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "cupsdAddEvent: Discarding %s event since MaxEvents is %d!", + cupsdEventName(event), MaxEvents); + return; + } + + /* + * Then loop through the subscriptions and add the event to the corresponding + * caches... + */ + + for (temp = NULL, sub = (cupsd_subscription_t *)cupsArrayFirst(Subscriptions); + sub; + sub = (cupsd_subscription_t *)cupsArrayNext(Subscriptions)) + { + /* + * Check if this subscription requires this event... + */ + + if ((sub->mask & event) != 0 && + (sub->dest == dest || !sub->dest) && + (sub->job == job || !sub->job)) + { + /* + * Need this event, so create a new event record... + */ + + if ((temp = (cupsd_event_t *)calloc(1, sizeof(cupsd_event_t))) == NULL) + { + cupsdLogMessage(CUPSD_LOG_CRIT, + "Unable to allocate memory for event - %s", + strerror(errno)); + return; + } + + temp->event = event; + temp->time = time(NULL); + temp->attrs = ippNew(); + temp->job = job; + temp->dest = dest; + + /* + * Add common event notification attributes... + */ + + ippAddString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, IPP_TAG_CHARSET, + "notify-charset", NULL, "utf-8"); + + ippAddString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, IPP_TAG_LANGUAGE, + "notify-natural-language", NULL, "en-US"); + + ippAddInteger(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, IPP_TAG_INTEGER, + "notify-subscription-id", sub->id); + + ippAddInteger(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, IPP_TAG_INTEGER, + "notify-sequence-number", sub->next_event_id); + + ippAddString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, IPP_TAG_KEYWORD, + "notify-subscribed-event", NULL, cupsdEventName(event)); + + if (sub->user_data_len > 0) + ippAddOctetString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, + "notify-user-data", sub->user_data, + sub->user_data_len); + + ippAddInteger(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, IPP_TAG_INTEGER, + "printer-up-time", time(NULL)); + + va_start(ap, text); + vsnprintf(ftext, sizeof(ftext), text, ap); + va_end(ap); + + ippAddString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, IPP_TAG_TEXT, + "notify-text", NULL, ftext); + + if (dest) + { + /* + * Add printer attributes... + */ + + ippAddString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, IPP_TAG_URI, + "notify-printer-uri", NULL, dest->uri); + + ippAddString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, IPP_TAG_NAME, + "printer-name", NULL, dest->name); + + ippAddInteger(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, IPP_TAG_ENUM, + "printer-state", dest->state); + + if (dest->num_reasons == 0) + ippAddString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, + IPP_TAG_KEYWORD, "printer-state-reasons", NULL, + dest->state == IPP_PRINTER_STOPPED ? "paused" : "none"); + else + ippAddStrings(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, + IPP_TAG_KEYWORD, "printer-state-reasons", + dest->num_reasons, NULL, + (const char * const *)dest->reasons); + + ippAddBoolean(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, + "printer-is-accepting-jobs", dest->accepting); + } + + if (job) + { + /* + * Add job attributes... + */ + + ippAddInteger(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, IPP_TAG_INTEGER, + "notify-job-id", job->id); + ippAddInteger(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, IPP_TAG_ENUM, + "job-state", job->state_value); + + if ((attr = ippFindAttribute(job->attrs, "job-name", + IPP_TAG_NAME)) != NULL) + ippAddString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, IPP_TAG_NAME, + "job-name", NULL, attr->values[0].string.text); + + switch (job->state_value) + { + case IPP_JOB_PENDING : + if (dest && dest->state == IPP_PRINTER_STOPPED) + ippAddString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, + IPP_TAG_KEYWORD, "job-state-reasons", NULL, + "printer-stopped"); + else + ippAddString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, + IPP_TAG_KEYWORD, "job-state-reasons", NULL, + "none"); + break; + + case IPP_JOB_HELD : + if (ippFindAttribute(job->attrs, "job-hold-until", IPP_TAG_KEYWORD) != NULL || + ippFindAttribute(job->attrs, "job-hold-until", IPP_TAG_NAME) != NULL) + ippAddString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, + IPP_TAG_KEYWORD, "job-state-reasons", NULL, + "job-hold-until-specified"); + else + ippAddString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, + IPP_TAG_KEYWORD, "job-state-reasons", NULL, + "job-incoming"); + break; + + case IPP_JOB_PROCESSING : + ippAddString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, + IPP_TAG_KEYWORD, "job-state-reasons", NULL, + "job-printing"); + break; + + case IPP_JOB_STOPPED : + ippAddString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, + IPP_TAG_KEYWORD, "job-state-reasons", NULL, + "job-stopped"); + break; + + case IPP_JOB_CANCELED : + ippAddString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, + IPP_TAG_KEYWORD, "job-state-reasons", NULL, + "job-canceled-by-user"); + break; + + case IPP_JOB_ABORTED : + ippAddString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, + IPP_TAG_KEYWORD, "job-state-reasons", NULL, + "aborted-by-system"); + break; + + case IPP_JOB_COMPLETED : + ippAddString(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, + IPP_TAG_KEYWORD, "job-state-reasons", NULL, + "job-completed-successfully"); + break; + } + + ippAddInteger(temp->attrs, IPP_TAG_EVENT_NOTIFICATION, IPP_TAG_INTEGER, + "job-impressions-completed", + job->sheets ? job->sheets->values[0].integer : 0); + } + + /* + * Send the notification for this subscription... + */ + + cupsd_send_notification(sub, temp); + } + } + + if (temp) + cupsdMarkDirty(CUPSD_DIRTY_SUBSCRIPTIONS); + else + cupsdLogMessage(CUPSD_LOG_DEBUG, "Discarding unused %s event...", + cupsdEventName(event)); +} + + +/* + * 'cupsdAddSubscription()' - Add a new subscription object. + */ + +cupsd_subscription_t * /* O - New subscription object */ +cupsdAddSubscription( + unsigned mask, /* I - Event mask */ + cupsd_printer_t *dest, /* I - Printer, if any */ + cupsd_job_t *job, /* I - Job, if any */ + const char *uri, /* I - notify-recipient-uri, if any */ + int sub_id) /* I - notify-subscription-id or 0 */ +{ + cupsd_subscription_t *temp; /* New subscription object */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "cupsdAddSubscription(mask=%x, dest=%p(%s), job=%p(%d), " + "uri=\"%s\")", + mask, dest, dest ? dest->name : "", job, job ? job->id : 0, + uri ? uri : "(null)"); + + if (!Subscriptions) + Subscriptions = cupsArrayNew((cups_array_func_t)cupsd_compare_subscriptions, + NULL); + + if (!Subscriptions) + { + cupsdLogMessage(CUPSD_LOG_CRIT, + "Unable to allocate memory for subscriptions - %s", + strerror(errno)); + return (NULL); + } + + /* + * Limit the number of subscriptions... + */ + + if (MaxSubscriptions > 0 && cupsArrayCount(Subscriptions) >= MaxSubscriptions) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "cupsdAddSubscription: Reached MaxSubscriptions %d " + "(count=%d)", MaxSubscriptions, + cupsArrayCount(Subscriptions)); + return (NULL); + } + + if (MaxSubscriptionsPerJob > 0 && job) + { + int count; /* Number of job subscriptions */ + + for (temp = (cupsd_subscription_t *)cupsArrayFirst(Subscriptions), + count = 0; + temp; + temp = (cupsd_subscription_t *)cupsArrayNext(Subscriptions)) + if (temp->job == job) + count ++; + + if (count >= MaxSubscriptionsPerJob) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "cupsdAddSubscription: Reached MaxSubscriptionsPerJob %d " + "for job #%d (count=%d)", MaxSubscriptionsPerJob, + job->id, count); + return (NULL); + } + } + + if (MaxSubscriptionsPerPrinter > 0 && dest) + { + int count; /* Number of printer subscriptions */ + + for (temp = (cupsd_subscription_t *)cupsArrayFirst(Subscriptions), + count = 0; + temp; + temp = (cupsd_subscription_t *)cupsArrayNext(Subscriptions)) + if (temp->dest == dest) + count ++; + + if (count >= MaxSubscriptionsPerPrinter) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "cupsdAddSubscription: Reached " + "MaxSubscriptionsPerPrinter %d for %s (count=%d)", + MaxSubscriptionsPerPrinter, dest->name, count); + return (NULL); + } + } + + /* + * Allocate memory for this subscription... + */ + + if ((temp = calloc(1, sizeof(cupsd_subscription_t))) == NULL) + { + cupsdLogMessage(CUPSD_LOG_CRIT, + "Unable to allocate memory for subscription object - %s", + strerror(errno)); + return (NULL); + } + + /* + * Fill in common data... + */ + + if (sub_id) + { + temp->id = sub_id; + + if (sub_id >= NextSubscriptionId) + NextSubscriptionId = sub_id + 1; + } + else + { + temp->id = NextSubscriptionId; + + NextSubscriptionId ++; + } + + temp->mask = mask; + temp->dest = dest; + temp->job = job; + temp->pipe = -1; + temp->first_event_id = 1; + temp->next_event_id = 1; + + cupsdSetString(&(temp->recipient), uri); + + /* + * Add the subscription to the array... + */ + + cupsArrayAdd(Subscriptions, temp); + + /* + * For RSS subscriptions, run the notifier immediately... + */ + + if (uri && !strncmp(uri, "rss:", 4)) + cupsd_start_notifier(temp); + + return (temp); +} + + +/* + * 'cupsdDeleteAllSubscriptions()' - Delete all subscriptions. + */ + +void +cupsdDeleteAllSubscriptions(void) +{ + cupsd_subscription_t *sub; /* Subscription */ + + + if (!Subscriptions) + return; + + for (sub = (cupsd_subscription_t *)cupsArrayFirst(Subscriptions); + sub; + sub = (cupsd_subscription_t *)cupsArrayNext(Subscriptions)) + cupsdDeleteSubscription(sub, 0); + + cupsArrayDelete(Subscriptions); + Subscriptions = NULL; +} + + +/* + * 'cupsdDeleteSubscription()' - Delete a subscription object. + */ + +void +cupsdDeleteSubscription( + cupsd_subscription_t *sub, /* I - Subscription object */ + int update) /* I - 1 = update subscriptions.conf */ +{ + /* + * Close the pipe to the notifier as needed... + */ + + if (sub->pipe >= 0) + close(sub->pipe); + + /* + * Remove subscription from array... + */ + + cupsArrayRemove(Subscriptions, sub); + + /* + * Free memory... + */ + + cupsdClearString(&(sub->owner)); + cupsdClearString(&(sub->recipient)); + + cupsArrayDelete(sub->events); + + free(sub); + + /* + * Update the subscriptions as needed... + */ + + if (update) + cupsdMarkDirty(CUPSD_DIRTY_SUBSCRIPTIONS); +} + + +/* + * 'cupsdEventName()' - Return a single event name. + */ + +const char * /* O - Event name */ +cupsdEventName( + cupsd_eventmask_t event) /* I - Event value */ +{ + switch (event) + { + default : + return (NULL); + + case CUPSD_EVENT_PRINTER_RESTARTED : + return ("printer-restarted"); + + case CUPSD_EVENT_PRINTER_SHUTDOWN : + return ("printer-shutdown"); + + case CUPSD_EVENT_PRINTER_STOPPED : + return ("printer-stopped"); + + case CUPSD_EVENT_PRINTER_FINISHINGS_CHANGED : + return ("printer-finishings-changed"); + + case CUPSD_EVENT_PRINTER_MEDIA_CHANGED : + return ("printer-media-changed"); + + case CUPSD_EVENT_PRINTER_ADDED : + return ("printer-added"); + + case CUPSD_EVENT_PRINTER_DELETED : + return ("printer-deleted"); + + case CUPSD_EVENT_PRINTER_MODIFIED : + return ("printer-modified"); + + case CUPSD_EVENT_PRINTER_QUEUE_ORDER_CHANGED : + return ("printer-queue-order-changed"); + + case CUPSD_EVENT_PRINTER_STATE : + case CUPSD_EVENT_PRINTER_STATE_CHANGED : + return ("printer-state-changed"); + + case CUPSD_EVENT_PRINTER_CONFIG : + case CUPSD_EVENT_PRINTER_CONFIG_CHANGED : + return ("printer-config-changed"); + + case CUPSD_EVENT_PRINTER_CHANGED : + return ("printer-changed"); + + case CUPSD_EVENT_JOB_CREATED : + return ("job-created"); + + case CUPSD_EVENT_JOB_COMPLETED : + return ("job-completed"); + + case CUPSD_EVENT_JOB_STOPPED : + return ("job-stopped"); + + case CUPSD_EVENT_JOB_CONFIG_CHANGED : + return ("job-config-changed"); + + case CUPSD_EVENT_JOB_PROGRESS : + return ("job-progress"); + + case CUPSD_EVENT_JOB_STATE : + case CUPSD_EVENT_JOB_STATE_CHANGED : + return ("job-state-changed"); + + case CUPSD_EVENT_SERVER_RESTARTED : + return ("server-restarted"); + + case CUPSD_EVENT_SERVER_STARTED : + return ("server-started"); + + case CUPSD_EVENT_SERVER_STOPPED : + return ("server-stopped"); + + case CUPSD_EVENT_SERVER_AUDIT : + return ("server-audit"); + + case CUPSD_EVENT_ALL : + return ("all"); + } +} + + +/* + * 'cupsdEventValue()' - Return the event mask value for a name. + */ + +cupsd_eventmask_t /* O - Event mask value */ +cupsdEventValue(const char *name) /* I - Name of event */ +{ + if (!strcmp(name, "all")) + return (CUPSD_EVENT_ALL); + else if (!strcmp(name, "printer-restarted")) + return (CUPSD_EVENT_PRINTER_RESTARTED); + else if (!strcmp(name, "printer-shutdown")) + return (CUPSD_EVENT_PRINTER_SHUTDOWN); + else if (!strcmp(name, "printer-stopped")) + return (CUPSD_EVENT_PRINTER_STOPPED); + else if (!strcmp(name, "printer-finishings-changed")) + return (CUPSD_EVENT_PRINTER_FINISHINGS_CHANGED); + else if (!strcmp(name, "printer-media-changed")) + return (CUPSD_EVENT_PRINTER_MEDIA_CHANGED); + else if (!strcmp(name, "printer-added")) + return (CUPSD_EVENT_PRINTER_ADDED); + else if (!strcmp(name, "printer-deleted")) + return (CUPSD_EVENT_PRINTER_DELETED); + else if (!strcmp(name, "printer-modified")) + return (CUPSD_EVENT_PRINTER_MODIFIED); + else if (!strcmp(name, "printer-queue-order-changed")) + return (CUPSD_EVENT_PRINTER_QUEUE_ORDER_CHANGED); + else if (!strcmp(name, "printer-state-changed")) + return (CUPSD_EVENT_PRINTER_STATE_CHANGED); + else if (!strcmp(name, "printer-config-changed")) + return (CUPSD_EVENT_PRINTER_CONFIG_CHANGED); + else if (!strcmp(name, "printer-changed")) + return (CUPSD_EVENT_PRINTER_CHANGED); + else if (!strcmp(name, "job-created")) + return (CUPSD_EVENT_JOB_CREATED); + else if (!strcmp(name, "job-completed")) + return (CUPSD_EVENT_JOB_COMPLETED); + else if (!strcmp(name, "job-stopped")) + return (CUPSD_EVENT_JOB_STOPPED); + else if (!strcmp(name, "job-config-changed")) + return (CUPSD_EVENT_JOB_CONFIG_CHANGED); + else if (!strcmp(name, "job-progress")) + return (CUPSD_EVENT_JOB_PROGRESS); + else if (!strcmp(name, "job-state-changed")) + return (CUPSD_EVENT_JOB_STATE_CHANGED); + else if (!strcmp(name, "server-restarted")) + return (CUPSD_EVENT_SERVER_RESTARTED); + else if (!strcmp(name, "server-started")) + return (CUPSD_EVENT_SERVER_STARTED); + else if (!strcmp(name, "server-stopped")) + return (CUPSD_EVENT_SERVER_STOPPED); + else if (!strcmp(name, "server-audit")) + return (CUPSD_EVENT_SERVER_AUDIT); + else + return (CUPSD_EVENT_NONE); +} + + +/* + * 'cupsdExpireSubscriptions()' - Expire old subscription objects. + */ + +void +cupsdExpireSubscriptions( + cupsd_printer_t *dest, /* I - Printer, if any */ + cupsd_job_t *job) /* I - Job, if any */ +{ + cupsd_subscription_t *sub; /* Current subscription */ + int update; /* Update subscriptions.conf? */ + time_t curtime; /* Current time */ + + + curtime = time(NULL); + update = 0; + + for (sub = (cupsd_subscription_t *)cupsArrayFirst(Subscriptions); + sub; + sub = (cupsd_subscription_t *)cupsArrayNext(Subscriptions)) + if ((!sub->job && !dest && sub->expire && sub->expire <= curtime) || + (dest && sub->dest == dest) || + (job && sub->job == job)) + { + cupsdLogMessage(CUPSD_LOG_INFO, "Subscription %d has expired...", + sub->id); + + cupsdDeleteSubscription(sub, 0); + + update = 1; + } + + if (update) + cupsdMarkDirty(CUPSD_DIRTY_SUBSCRIPTIONS); +} + + +/* + * 'cupsdFindSubscription()' - Find a subscription by ID. + */ + +cupsd_subscription_t * /* O - Subscription object */ +cupsdFindSubscription(int id) /* I - Subscription ID */ +{ + cupsd_subscription_t sub; /* Subscription template */ + + + sub.id = id; + + return ((cupsd_subscription_t *)cupsArrayFind(Subscriptions, &sub)); +} + + +/* + * 'cupsdLoadAllSubscriptions()' - Load all subscriptions from the .conf file. + */ + +void +cupsdLoadAllSubscriptions(void) +{ + int i; /* Looping var */ + cups_file_t *fp; /* subscriptions.conf file */ + int linenum; /* Current line number */ + char line[1024], /* Line from file */ + *value, /* Pointer to value */ + *valueptr; /* Pointer into value */ + cupsd_subscription_t *sub; /* Current subscription */ + int hex; /* Non-zero if reading hex data */ + int delete_sub; /* Delete subscription? */ + + + /* + * Open the subscriptions.conf file... + */ + + snprintf(line, sizeof(line), "%s/subscriptions.conf", ServerRoot); + if ((fp = cupsdOpenConfFile(line)) == NULL) + return; + + /* + * Read all of the lines from the file... + */ + + linenum = 0; + sub = NULL; + delete_sub = 0; + + while (cupsFileGetConf(fp, line, sizeof(line), &value, &linenum)) + { + if (!_cups_strcasecmp(line, "NextSubscriptionId") && value) + { + /* + * NextSubscriptionId NNN + */ + + i = atoi(value); + if (i >= NextSubscriptionId && i > 0) + NextSubscriptionId = i; + } + else if (!_cups_strcasecmp(line, " + */ + + if (!sub && value && isdigit(value[0] & 255)) + { + sub = cupsdAddSubscription(CUPSD_EVENT_NONE, NULL, NULL, NULL, + atoi(value)); + } + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of subscriptions.conf.", + linenum); + break; + } + } + else if (!_cups_strcasecmp(line, "")) + { + if (!sub) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of subscriptions.conf.", + linenum); + break; + } + + if (delete_sub) + cupsdDeleteSubscription(sub, 0); + + sub = NULL; + delete_sub = 0; + } + else if (!sub) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of subscriptions.conf.", + linenum); + } + else if (!_cups_strcasecmp(line, "Events")) + { + /* + * Events name + * Events name name name ... + */ + + if (!value) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of subscriptions.conf.", + linenum); + break; + } + + while (*value) + { + /* + * Separate event names... + */ + + for (valueptr = value; !isspace(*valueptr) && *valueptr; valueptr ++); + + while (isspace(*valueptr & 255)) + *valueptr++ = '\0'; + + /* + * See if the name exists... + */ + + if ((sub->mask |= cupsdEventValue(value)) == CUPSD_EVENT_NONE) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unknown event name \'%s\' on line %d of subscriptions.conf.", + value, linenum); + break; + } + + value = valueptr; + } + } + else if (!_cups_strcasecmp(line, "Owner")) + { + /* + * Owner + */ + + if (value) + cupsdSetString(&sub->owner, value); + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of subscriptions.conf.", + linenum); + break; + } + } + else if (!_cups_strcasecmp(line, "Recipient")) + { + /* + * Recipient uri + */ + + if (value) + cupsdSetString(&sub->recipient, value); + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of subscriptions.conf.", + linenum); + break; + } + } + else if (!_cups_strcasecmp(line, "JobId")) + { + /* + * JobId # + */ + + if (value && isdigit(*value & 255)) + { + if ((sub->job = cupsdFindJob(atoi(value))) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Job %s not found on line %d of subscriptions.conf.", + value, linenum); + delete_sub = 1; + } + } + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of subscriptions.conf.", + linenum); + break; + } + } + else if (!_cups_strcasecmp(line, "PrinterName")) + { + /* + * PrinterName name + */ + + if (value) + { + if ((sub->dest = cupsdFindDest(value)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Printer \'%s\' not found on line %d of subscriptions.conf.", + value, linenum); + delete_sub = 1; + } + } + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of subscriptions.conf.", + linenum); + break; + } + } + else if (!_cups_strcasecmp(line, "UserData")) + { + /* + * UserData encoded-string + */ + + if (value) + { + for (i = 0, valueptr = value, hex = 0; i < 63 && *valueptr; i ++) + { + if (*valueptr == '<' && !hex) + { + hex = 1; + valueptr ++; + } + + if (hex) + { + if (isxdigit(valueptr[0]) && isxdigit(valueptr[1])) + { + if (isdigit(valueptr[0])) + sub->user_data[i] = (valueptr[0] - '0') << 4; + else + sub->user_data[i] = (tolower(valueptr[0]) - 'a' + 10) << 4; + + if (isdigit(valueptr[1])) + sub->user_data[i] |= valueptr[1] - '0'; + else + sub->user_data[i] |= tolower(valueptr[1]) - 'a' + 10; + + valueptr += 2; + + if (*valueptr == '>') + { + hex = 0; + valueptr ++; + } + } + else + break; + } + else + sub->user_data[i] = *valueptr++; + } + + if (*valueptr) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Bad UserData \'%s\' on line %d of subscriptions.conf.", + value, linenum); + } + else + sub->user_data_len = i; + } + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of subscriptions.conf.", + linenum); + break; + } + } + else if (!_cups_strcasecmp(line, "LeaseDuration")) + { + /* + * LeaseDuration # + */ + + if (value && isdigit(*value & 255)) + { + sub->lease = atoi(value); + sub->expire = sub->lease ? time(NULL) + sub->lease : 0; + } + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of subscriptions.conf.", + linenum); + break; + } + } + else if (!_cups_strcasecmp(line, "Interval")) + { + /* + * Interval # + */ + + if (value && isdigit(*value & 255)) + sub->interval = atoi(value); + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of subscriptions.conf.", + linenum); + break; + } + } + else if (!_cups_strcasecmp(line, "ExpirationTime")) + { + /* + * ExpirationTime # + */ + + if (value && isdigit(*value & 255)) + sub->expire = atoi(value); + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of subscriptions.conf.", + linenum); + break; + } + } + else if (!_cups_strcasecmp(line, "NextEventId")) + { + /* + * NextEventId # + */ + + if (value && isdigit(*value & 255)) + sub->next_event_id = sub->first_event_id = atoi(value); + else + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Syntax error on line %d of subscriptions.conf.", + linenum); + break; + } + } + else + { + /* + * Something else we don't understand... + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unknown configuration directive %s on line %d of subscriptions.conf.", + line, linenum); + } + } + + cupsFileClose(fp); +} + + +/* + * 'cupsdSaveAllSubscriptions()' - Save all subscriptions to the .conf file. + */ + +void +cupsdSaveAllSubscriptions(void) +{ + int i; /* Looping var */ + cups_file_t *fp; /* subscriptions.conf file */ + char filename[1024], /* subscriptions.conf filename */ + temp[1024]; /* Temporary string */ + cupsd_subscription_t *sub; /* Current subscription */ + time_t curtime; /* Current time */ + struct tm *curdate; /* Current date */ + unsigned mask; /* Current event mask */ + const char *name; /* Current event name */ + int hex; /* Non-zero if we are writing hex data */ + + + /* + * Create the subscriptions.conf file... + */ + + snprintf(filename, sizeof(filename), "%s/subscriptions.conf", ServerRoot); + + if ((fp = cupsdCreateConfFile(filename, ConfigFilePerm)) == NULL) + return; + + cupsdLogMessage(CUPSD_LOG_INFO, "Saving subscriptions.conf..."); + + /* + * Write a small header to the file... + */ + + curtime = time(NULL); + curdate = localtime(&curtime); + strftime(temp, sizeof(temp) - 1, "%Y-%m-%d %H:%M", curdate); + + cupsFilePuts(fp, "# Subscription configuration file for " CUPS_SVERSION "\n"); + cupsFilePrintf(fp, "# Written by cupsd on %s\n", temp); + + cupsFilePrintf(fp, "NextSubscriptionId %d\n", NextSubscriptionId); + + /* + * Write every subscription known to the system... + */ + + for (sub = (cupsd_subscription_t *)cupsArrayFirst(Subscriptions); + sub; + sub = (cupsd_subscription_t *)cupsArrayNext(Subscriptions)) + { + cupsFilePrintf(fp, "\n", sub->id); + + if ((name = cupsdEventName((cupsd_eventmask_t)sub->mask)) != NULL) + { + /* + * Simple event list... + */ + + cupsFilePrintf(fp, "Events %s\n", name); + } + else + { + /* + * Complex event list... + */ + + cupsFilePuts(fp, "Events"); + + for (mask = 1; mask < CUPSD_EVENT_ALL; mask <<= 1) + if (sub->mask & mask) + cupsFilePrintf(fp, " %s", cupsdEventName((cupsd_eventmask_t)mask)); + + cupsFilePuts(fp, "\n"); + } + + if (sub->owner) + cupsFilePrintf(fp, "Owner %s\n", sub->owner); + if (sub->recipient) + cupsFilePrintf(fp, "Recipient %s\n", sub->recipient); + if (sub->job) + cupsFilePrintf(fp, "JobId %d\n", sub->job->id); + if (sub->dest) + cupsFilePrintf(fp, "PrinterName %s\n", sub->dest->name); + + if (sub->user_data_len > 0) + { + cupsFilePuts(fp, "UserData "); + + for (i = 0, hex = 0; i < sub->user_data_len; i ++) + { + if (sub->user_data[i] < ' ' || + sub->user_data[i] > 0x7f || + sub->user_data[i] == '<') + { + if (!hex) + { + cupsFilePrintf(fp, "<%02X", sub->user_data[i]); + hex = 1; + } + else + cupsFilePrintf(fp, "%02X", sub->user_data[i]); + } + else + { + if (hex) + { + cupsFilePrintf(fp, ">%c", sub->user_data[i]); + hex = 0; + } + else + cupsFilePutChar(fp, sub->user_data[i]); + } + } + + if (hex) + cupsFilePuts(fp, ">\n"); + else + cupsFilePutChar(fp, '\n'); + } + + cupsFilePrintf(fp, "LeaseDuration %d\n", sub->lease); + cupsFilePrintf(fp, "Interval %d\n", sub->interval); + cupsFilePrintf(fp, "ExpirationTime %ld\n", (long)sub->expire); + cupsFilePrintf(fp, "NextEventId %d\n", sub->next_event_id); + + cupsFilePuts(fp, "\n"); + } + + cupsdCloseCreatedConfFile(fp, filename); +} + + +/* + * 'cupsdStopAllNotifiers()' - Stop all notifier processes. + */ + +void +cupsdStopAllNotifiers(void) +{ + cupsd_subscription_t *sub; /* Current subscription */ + + + /* + * See if we have started any notifiers... + */ + + if (!NotifierStatusBuffer) + return; + + /* + * Yes, kill any processes that are left... + */ + + for (sub = (cupsd_subscription_t *)cupsArrayFirst(Subscriptions); + sub; + sub = (cupsd_subscription_t *)cupsArrayNext(Subscriptions)) + if (sub->pid) + { + cupsdEndProcess(sub->pid, 0); + + close(sub->pipe); + sub->pipe = -1; + } + + /* + * Close the status pipes... + */ + + if (NotifierPipes[0] >= 0) + { + cupsdRemoveSelect(NotifierPipes[0]); + + cupsdStatBufDelete(NotifierStatusBuffer); + + close(NotifierPipes[0]); + close(NotifierPipes[1]); + + NotifierPipes[0] = -1; + NotifierPipes[1] = -1; + NotifierStatusBuffer = NULL; + } +} + + +/* + * 'cupsd_compare_subscriptions()' - Compare two subscriptions. + */ + +static int /* O - Result of comparison */ +cupsd_compare_subscriptions( + cupsd_subscription_t *first, /* I - First subscription object */ + cupsd_subscription_t *second, /* I - Second subscription object */ + void *unused) /* I - Unused user data pointer */ +{ + (void)unused; + + return (first->id - second->id); +} + + +/* + * 'cupsd_delete_event()' - Delete a single event... + * + * Oldest events must be deleted first, otherwise the subscription cache + * flushing code will not work properly. + */ + +static void +cupsd_delete_event(cupsd_event_t *event)/* I - Event to delete */ +{ + /* + * Free memory... + */ + + ippDelete(event->attrs); + free(event); +} + + +#ifdef HAVE_DBUS +/* + * 'cupsd_send_dbus()' - Send a DBUS notification... + */ + +static void +cupsd_send_dbus(cupsd_eventmask_t event,/* I - Event to send */ + cupsd_printer_t *dest,/* I - Destination, if any */ + cupsd_job_t *job) /* I - Job, if any */ +{ + DBusError error; /* Error, if any */ + DBusMessage *message; /* Message to send */ + DBusMessageIter iter; /* Iterator for message data */ + const char *what; /* What to send */ + static DBusConnection *con = NULL; /* Connection to DBUS server */ + + + /* + * Figure out what to send, if anything... + */ + + if (event & CUPSD_EVENT_PRINTER_ADDED) + what = "PrinterAdded"; + else if (event & CUPSD_EVENT_PRINTER_DELETED) + what = "PrinterRemoved"; + else if (event & CUPSD_EVENT_PRINTER_CHANGED) + what = "QueueChanged"; + else if (event & CUPSD_EVENT_JOB_CREATED) + what = "JobQueuedLocal"; + else if ((event & CUPSD_EVENT_JOB_STATE) && job && + job->state_value == IPP_JOB_PROCESSING) + what = "JobStartedLocal"; + else + return; + + /* + * Verify connection to DBUS server... + */ + + if (con && !dbus_connection_get_is_connected(con)) + { + dbus_connection_unref(con); + con = NULL; + } + + if (!con) + { + dbus_error_init(&error); + + con = dbus_bus_get(getuid() ? DBUS_BUS_SESSION : DBUS_BUS_SYSTEM, &error); + if (!con) + { + dbus_error_free(&error); + return; + } + } + + /* + * Create and send the new message... + */ + + message = dbus_message_new_signal("/com/redhat/PrinterSpooler", + "com.redhat.PrinterSpooler", what); + + dbus_message_append_iter_init(message, &iter); + if (dest) + dbus_message_iter_append_string(&iter, dest->name); + if (job) + { + dbus_message_iter_append_uint32(&iter, job->id); + dbus_message_iter_append_string(&iter, job->username); + } + + dbus_connection_send(con, message, NULL); + dbus_connection_flush(con); + dbus_message_unref(message); +} +#endif /* HAVE_DBUS */ + + +/* + * 'cupsd_send_notification()' - Send a notification for the specified event. + */ + +static void +cupsd_send_notification( + cupsd_subscription_t *sub, /* I - Subscription object */ + cupsd_event_t *event) /* I - Event to send */ +{ + ipp_state_t state; /* IPP event state */ + + + cupsdLogMessage(CUPSD_LOG_DEBUG2, + "cupsd_send_notification(sub=%p(%d), event=%p(%s))", + sub, sub->id, event, cupsdEventName(event->event)); + + /* + * Allocate the events array as needed... + */ + + if (!sub->events) + { + sub->events = cupsArrayNew3((cups_array_func_t)NULL, NULL, + (cups_ahash_func_t)NULL, 0, + (cups_acopy_func_t)NULL, + (cups_afree_func_t)cupsd_delete_event); + + if (!sub->events) + { + cupsdLogMessage(CUPSD_LOG_CRIT, + "Unable to allocate memory for subscription #%d!", + sub->id); + return; + } + } + + /* + * Purge an old event as needed... + */ + + if (cupsArrayCount(sub->events) >= MaxEvents) + { + /* + * Purge the oldest event in the cache... + */ + + cupsArrayRemove(sub->events, cupsArrayFirst(sub->events)); + + sub->first_event_id ++; + } + + /* + * Add the event to the subscription. Since the events array is + * always MaxEvents in length, and since we will have already + * removed an event from the subscription cache if we hit the + * event cache limit, we don't need to check for overflow here... + */ + + cupsArrayAdd(sub->events, event); + + /* + * Deliver the event... + */ + + if (sub->recipient) + { + for (;;) + { + if (sub->pipe < 0) + cupsd_start_notifier(sub); + + cupsdLogMessage(CUPSD_LOG_DEBUG2, "sub->pipe=%d", sub->pipe); + + if (sub->pipe < 0) + break; + + event->attrs->state = IPP_IDLE; + + while ((state = ippWriteFile(sub->pipe, event->attrs)) != IPP_DATA) + if (state == IPP_ERROR) + break; + + if (state == IPP_ERROR) + { + if (errno == EPIPE) + { + /* + * Notifier died, try restarting it... + */ + + cupsdLogMessage(CUPSD_LOG_WARN, + "Notifier for subscription %d (%s) went away, " + "retrying!", + sub->id, sub->recipient); + cupsdEndProcess(sub->pid, 0); + + close(sub->pipe); + sub->pipe = -1; + continue; + } + + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to send event for subscription %d (%s)!", + sub->id, sub->recipient); + } + + /* + * If we get this far, break out of the loop... + */ + + break; + } + } + + /* + * Bump the event sequence number... + */ + + sub->next_event_id ++; +} + + +/* + * 'cupsd_start_notifier()' - Start a notifier subprocess... + */ + +static void +cupsd_start_notifier( + cupsd_subscription_t *sub) /* I - Subscription object */ +{ + int pid; /* Notifier process ID */ + int fds[2]; /* Pipe file descriptors */ + char *argv[4], /* Command-line arguments */ + *envp[MAX_ENV], /* Environment variables */ + user_data[128], /* Base-64 encoded user data */ + scheme[256], /* notify-recipient-uri scheme */ + *ptr, /* Pointer into scheme */ + command[1024]; /* Notifier command */ + + + /* + * Extract the scheme name from the recipient URI and point to the + * notifier program... + */ + + strlcpy(scheme, sub->recipient, sizeof(scheme)); + if ((ptr = strchr(scheme, ':')) != NULL) + *ptr = '\0'; + + snprintf(command, sizeof(command), "%s/notifier/%s", ServerBin, scheme); + + /* + * Base-64 encode the user data... + */ + + httpEncode64_2(user_data, sizeof(user_data), (char *)sub->user_data, + sub->user_data_len); + + /* + * Setup the argument array... + */ + + argv[0] = command; + argv[1] = sub->recipient; + argv[2] = user_data; + argv[3] = NULL; + + /* + * Setup the environment... + */ + + cupsdLoadEnv(envp, (int)(sizeof(envp) / sizeof(envp[0]))); + + /* + * Create pipes as needed... + */ + + if (!NotifierStatusBuffer) + { + /* + * Create the status pipe... + */ + + if (cupsdOpenPipe(NotifierPipes)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to create pipes for notifier status - %s", + strerror(errno)); + return; + } + + NotifierStatusBuffer = cupsdStatBufNew(NotifierPipes[0], "[Notifier]"); + + cupsdAddSelect(NotifierPipes[0], (cupsd_selfunc_t)cupsd_update_notifier, + NULL, NULL); + } + + if (cupsdOpenPipe(fds)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, + "Unable to create pipes for notifier %s - %s", + scheme, strerror(errno)); + return; + } + + /* + * Make sure the delivery pipe is non-blocking... + */ + + fcntl(fds[1], F_SETFL, fcntl(fds[1], F_GETFL) | O_NONBLOCK); + + /* + * Create the notifier process... + */ + + if (cupsdStartProcess(command, argv, envp, fds[0], -1, NotifierPipes[1], + -1, -1, 0, DefaultProfile, NULL, &pid) < 0) + { + /* + * Error - can't fork! + */ + + cupsdLogMessage(CUPSD_LOG_ERROR, "Unable to fork for notifier %s - %s", + scheme, strerror(errno)); + + cupsdClosePipe(fds); + } + else + { + /* + * Fork successful - return the PID... + */ + + cupsdLogMessage(CUPSD_LOG_DEBUG, "Notifier %s started - PID = %d", + scheme, pid); + + sub->pid = pid; + sub->pipe = fds[1]; + sub->status = 0; + + close(fds[0]); + } +} + + +/* + * 'cupsd_update_notifier()' - Read messages from notifiers. + */ + +void +cupsd_update_notifier(void) +{ + char message[1024]; /* Pointer to message text */ + int loglevel; /* Log level for message */ + + + while (cupsdStatBufUpdate(NotifierStatusBuffer, &loglevel, + message, sizeof(message))) + { + if (loglevel == CUPSD_LOG_INFO) + cupsdLogMessage(CUPSD_LOG_INFO, "%s", message); + + if (!strchr(NotifierStatusBuffer->buffer, '\n')) + break; + } +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/subscriptions.h b/scheduler/subscriptions.h new file mode 100644 index 0000000000..eac866956c --- /dev/null +++ b/scheduler/subscriptions.h @@ -0,0 +1,166 @@ +/* + * "$Id$" + * + * Subscription definitions for the CUPS scheduler. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +/* + * Event mask enumeration... + */ + +typedef enum +{ + /* Individual printer events... */ + CUPSD_EVENT_PRINTER_STATE = 0x0001, /* Sent after generic printer state change */ + CUPSD_EVENT_PRINTER_RESTARTED = 0x0002, + /* Sent after printer restarted */ + CUPSD_EVENT_PRINTER_SHUTDOWN = 0x0004,/* Sent after printer shutdown */ + CUPSD_EVENT_PRINTER_STOPPED = 0x0008, /* Sent after printer stopped */ + + CUPSD_EVENT_PRINTER_CONFIG = 0x0010, /* Send after add/modify changes attrs */ + CUPSD_EVENT_PRINTER_FINISHINGS_CHANGED = 0x0020, + /* Sent after finishings-supported changed */ + CUPSD_EVENT_PRINTER_MEDIA_CHANGED = 0x0040, + /* Sent after media-supported changed */ + CUPSD_EVENT_PRINTER_ADDED = 0x0080, /* Sent after printer added */ + CUPSD_EVENT_PRINTER_DELETED = 0x0100, /* Sent after printer deleted */ + CUPSD_EVENT_PRINTER_MODIFIED = 0x0200,/* Sent after printer modified */ + CUPSD_EVENT_PRINTER_QUEUE_ORDER_CHANGED = 0x0400, + /* Sent when the order of jobs is changed */ + + /* Convenience printer event groupings... */ + CUPSD_EVENT_PRINTER_STATE_CHANGED = 0x000f, + /* STATE + RESTARTED + SHUTDOWN + STOPPED */ + CUPSD_EVENT_PRINTER_CONFIG_CHANGED = 0x0070, + /* CONFIG + FINISHINGS_CHANGED + MEDIA_CHANGED */ + CUPSD_EVENT_PRINTER_CHANGED = 0x07ff, /* All of the above */ + + /* Individual job events... */ + CUPSD_EVENT_JOB_STATE = 0x0800, /* Any state change */ + CUPSD_EVENT_JOB_CREATED = 0x1000, /* Send after job is created */ + CUPSD_EVENT_JOB_COMPLETED = 0x2000, /* Sent after job is completed */ + CUPSD_EVENT_JOB_STOPPED = 0x4000, /* Sent after job is stopped */ + CUPSD_EVENT_JOB_CONFIG_CHANGED = 0x8000, + /* Sent after set-job-attributes */ + CUPSD_EVENT_JOB_PROGRESS = 0x10000, /* Sent for each page */ + + /* Convenience job event grouping... */ + CUPSD_EVENT_JOB_STATE_CHANGED = 0x7800, + /* STATE + CREATED + COMPLETED + STOPPED */ + + /* Server events... */ + CUPSD_EVENT_SERVER_RESTARTED = 0x20000,/* Sent after server restarts */ + CUPSD_EVENT_SERVER_STARTED = 0x40000, /* Sent when server first starts */ + CUPSD_EVENT_SERVER_STOPPED = 0x80000, /* Sent when server is stopped */ + CUPSD_EVENT_SERVER_AUDIT = 0x100000, /* Security-related stuff */ + + /* Everything and nothing... */ + CUPSD_EVENT_NONE = 0, /* Nothing */ + CUPSD_EVENT_ALL = 0x1fffff /* Everything */ +} cupsd_eventmask_t; + + +/* + * Notiification support structures... + */ + +typedef struct cupsd_event_s /**** Event structure ****/ +{ + cupsd_eventmask_t event; /* Event */ + time_t time; /* Time of event */ + ipp_t *attrs; /* Notification message */ + cupsd_printer_t *dest; /* Associated printer, if any */ + cupsd_job_t *job; /* Associated job, if any */ +} cupsd_event_t; + +typedef struct cupsd_subscription_s /**** Subscription structure ****/ +{ + int id; /* subscription-id */ + unsigned mask; /* Event mask */ + char *owner; /* notify-subscriber-user-name */ + char *recipient; /* notify-recipient-uri, if applicable */ + unsigned char user_data[64]; /* notify-user-data */ + int user_data_len; /* Length of notify-user-data */ + int lease; /* notify-lease-duration */ + int interval; /* notify-time-interval */ + cupsd_printer_t *dest; /* notify-printer-uri, if any */ + cupsd_job_t *job; /* notify-job-id, if any */ + int pid; /* Process ID of notifier */ + int pipe; /* Pipe to notifier */ + int status; /* Exit status of notifier */ + time_t last; /* Time of last notification */ + time_t expire; /* Lease expiration time */ + int first_event_id, /* First event-id in cache */ + next_event_id; /* Next event-id to use */ + cups_array_t *events; /* Cached events */ +} cupsd_subscription_t; + + +/* + * Globals... + */ + +VAR int MaxSubscriptions VALUE(100), + /* Overall subscription limit */ + MaxSubscriptionsPerJob VALUE(0), + /* Per-job subscription limit */ + MaxSubscriptionsPerPrinter VALUE(0), + /* Per-printer subscription limit */ + MaxSubscriptionsPerUser VALUE(0), + /* Per-user subscription limit */ + NextSubscriptionId VALUE(1), + /* Next subscription ID */ + DefaultLeaseDuration VALUE(86400), + /* Default notify-lease-duration */ + MaxLeaseDuration VALUE(0); + /* Maximum notify-lease-duration */ +VAR cups_array_t *Subscriptions VALUE(NULL); + /* Active subscriptions */ + +VAR int MaxEvents VALUE(100); /* Maximum number of events */ + +VAR unsigned LastEvent VALUE(0); /* Last event(s) processed */ +VAR int NotifierPipes[2] VALUE2(-1, -1); + /* Pipes for notifier error/debug output */ +VAR cupsd_statbuf_t *NotifierStatusBuffer VALUE(NULL); + /* Status buffer for pipes */ + + +/* + * Prototypes... + */ + +extern void cupsdAddEvent(cupsd_eventmask_t event, cupsd_printer_t *dest, + cupsd_job_t *job, const char *text, ...); +extern cupsd_subscription_t * + cupsdAddSubscription(unsigned mask, cupsd_printer_t *dest, + cupsd_job_t *job, const char *uri, + int sub_id); +extern void cupsdDeleteAllSubscriptions(void); +extern void cupsdDeleteSubscription(cupsd_subscription_t *sub, int update); +extern const char * + cupsdEventName(cupsd_eventmask_t event); +extern cupsd_eventmask_t + cupsdEventValue(const char *name); + +extern cupsd_subscription_t * + cupsdFindSubscription(int id); +extern void cupsdExpireSubscriptions(cupsd_printer_t *dest, + cupsd_job_t *job); +extern void cupsdLoadAllSubscriptions(void); +extern void cupsdSaveAllSubscriptions(void); +extern void cupsdStopAllNotifiers(void); + + +/* + * End of "$Id$". + */ diff --git a/scheduler/sysman.c b/scheduler/sysman.c new file mode 100644 index 0000000000..6f49e83c73 --- /dev/null +++ b/scheduler/sysman.c @@ -0,0 +1,1004 @@ +/* + * "$Id$" + * + * System management functions for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdCleanDirty() - Write dirty config and state files. + * cupsdMarkDirty() - Mark config or state files as needing a + * write. + * cupsdSetBusyState() - Let the system know when we are busy + * doing something. + * cupsdAllowSleep() - Tell the OS it is now OK to sleep. + * cupsdStartSystemMonitor() - Start monitoring for system change. + * cupsdStopSystemMonitor() - Stop monitoring for system change. + * sysEventThreadEntry() - A thread to receive power and computer + * name change notifications. + * sysEventPowerNotifier() - Handle power notification events. + * sysEventConfigurationNotifier() - Computer name changed notification + * callback. + * sysEventTimerNotifier() - Handle delayed event notifications. + * sysUpdate() - Update the current system state. + */ + + +/* + * Include necessary headers... + */ + +#include "cupsd.h" +#ifdef HAVE_VPROC_TRANSACTION_BEGIN +# include +#endif /* HAVE_VPROC_TRANSACTION_BEGIN */ +#ifdef __APPLE__ +# include +# ifdef HAVE_IOKIT_PWR_MGT_IOPMLIBPRIVATE_H +# include +# endif /* HAVE_IOKIT_PWR_MGT_IOPMLIBPRIVATE_H */ +#endif /* __APPLE__ */ + + +/* + * The system management functions cover disk and power management which + * are primarily used on portable computers. + * + * Disk management involves delaying the write of certain configuration + * and state files to minimize the number of times the disk has to spin + * up. + * + * Power management support is currently only implemented on MacOS X, but + * essentially we use four functions to let the OS know when it is OK to + * put the system to sleep, typically when we are not in the middle of + * printing a job. + * + * Once put to sleep, we invalidate all remote printers since it is common + * to wake up in a new location/on a new wireless network. + */ + +/* + * Local globals... + */ + +#ifdef kIOPMAssertionTypeDenySystemSleep +static IOPMAssertionID dark_wake = 0; /* "Dark wake" assertion for sharing */ +#endif /* kIOPMAssertionTypeDenySystemSleep */ + + +/* + * 'cupsdCleanDirty()' - Write dirty config and state files. + */ + +void +cupsdCleanDirty(void) +{ + if (DirtyFiles & CUPSD_DIRTY_PRINTERS) + cupsdSaveAllPrinters(); + + if (DirtyFiles & CUPSD_DIRTY_CLASSES) + cupsdSaveAllClasses(); + + if (DirtyFiles & CUPSD_DIRTY_PRINTCAP) + cupsdWritePrintcap(); + + if (DirtyFiles & CUPSD_DIRTY_JOBS) + { + cupsd_job_t *job; /* Current job */ + + cupsdSaveAllJobs(); + + for (job = (cupsd_job_t *)cupsArrayFirst(Jobs); + job; + job = (cupsd_job_t *)cupsArrayNext(Jobs)) + if (job->dirty) + cupsdSaveJob(job); + } + + if (DirtyFiles & CUPSD_DIRTY_SUBSCRIPTIONS) + cupsdSaveAllSubscriptions(); + + DirtyFiles = CUPSD_DIRTY_NONE; + DirtyCleanTime = 0; + + cupsdSetBusyState(); +} + + +/* + * 'cupsdMarkDirty()' - Mark config or state files as needing a write. + */ + +void +cupsdMarkDirty(int what) /* I - What file(s) are dirty? */ +{ + cupsdLogMessage(CUPSD_LOG_DEBUG, "cupsdMarkDirty(%c%c%c%c%c)", + (what & CUPSD_DIRTY_PRINTERS) ? 'P' : '-', + (what & CUPSD_DIRTY_CLASSES) ? 'C' : '-', + (what & CUPSD_DIRTY_PRINTCAP) ? 'p' : '-', + (what & CUPSD_DIRTY_JOBS) ? 'J' : '-', + (what & CUPSD_DIRTY_SUBSCRIPTIONS) ? 'S' : '-'); + + if (what == CUPSD_DIRTY_PRINTCAP && !Printcap) + return; + + DirtyFiles |= what; + + if (!DirtyCleanTime) + DirtyCleanTime = time(NULL) + DirtyCleanInterval; + + cupsdSetBusyState(); +} + + +/* + * 'cupsdSetBusyState()' - Let the system know when we are busy doing something. + */ + +void +cupsdSetBusyState(void) +{ + int i; /* Looping var */ + cupsd_job_t *job; /* Current job */ + cupsd_printer_t *p; /* Current printer */ + int newbusy; /* New busy state */ + static int busy = 0; /* Current busy state */ + static const char * const busy_text[] = + { /* Text for busy states */ + "Not busy", + "Dirty files", + "Printing jobs", + "Printing jobs and dirty files", + "Active clients", + "Active clients and dirty files", + "Active clients and printing jobs", + "Active clients, printing jobs, and dirty files" + }; +#ifdef HAVE_VPROC_TRANSACTION_BEGIN + static vproc_transaction_t vtran = 0; /* Current busy transaction */ +#endif /* HAVE_VPROC_TRANSACTION_BEGIN */ + + + /* + * Figure out how busy we are... + */ + + newbusy = (DirtyCleanTime ? 1 : 0) | + (cupsArrayCount(ActiveClients) ? 4 : 0); + + for (job = (cupsd_job_t *)cupsArrayFirst(PrintingJobs); + job; + job = (cupsd_job_t *)cupsArrayNext(PrintingJobs)) + { + if ((p = job->printer) != NULL) + { + for (i = 0; i < p->num_reasons; i ++) + if (!strcmp(p->reasons[i], "connecting-to-device")) + break; + + if (!p->num_reasons || i >= p->num_reasons) + break; + } + } + + if (job) + newbusy |= 2; + + cupsdLogMessage(CUPSD_LOG_DEBUG, + "cupsdSetBusyState: newbusy=\"%s\", busy=\"%s\"", + busy_text[newbusy], busy_text[busy]); + + /* + * Manage state changes... + */ + + if (newbusy != busy) + { + busy = newbusy; + +#ifdef HAVE_VPROC_TRANSACTION_BEGIN + if (busy && !vtran) + vtran = vproc_transaction_begin(NULL); + else if (!busy && vtran) + { + vproc_transaction_end(NULL, vtran); + vtran = 0; + } +#endif /* HAVE_VPROC_TRANSACTION_BEGIN */ + } + +#ifdef kIOPMAssertionTypeDenySystemSleep + if (cupsArrayCount(PrintingJobs) > 0 && !dark_wake) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, "Asserting dark wake."); + IOPMAssertionCreateWithName(kIOPMAssertionTypeDenySystemSleep, + kIOPMAssertionLevelOn, + CFSTR("org.cups.cupsd"), &dark_wake); + } + else if (cupsArrayCount(PrintingJobs) == 0 && dark_wake) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, "Releasing dark wake assertion."); + IOPMAssertionRelease(dark_wake); + dark_wake = 0; + } +#endif /* kIOPMAssertionTypeDenySystemSleep */ +} + + +#ifdef __APPLE__ +/* + * This is the Apple-specific system event code. It works by creating + * a worker thread that waits for events from the OS and relays them + * to the main thread via a traditional pipe. + */ + +/* + * Include MacOS-specific headers... + */ + +# include +# include +# include +# include +# include + + +/* + * Constants... + */ + +# define SYSEVENT_CANSLEEP 0x1 /* Decide whether to allow sleep or not */ +# define SYSEVENT_WILLSLEEP 0x2 /* Computer will go to sleep */ +# define SYSEVENT_WOKE 0x4 /* Computer woke from sleep */ +# define SYSEVENT_NETCHANGED 0x8 /* Network changed */ +# define SYSEVENT_NAMECHANGED 0x10 /* Computer name changed */ + + +/* + * Structures... + */ + +typedef struct cupsd_sysevent_s /*** System event data ****/ +{ + unsigned char event; /* Event bit field */ + io_connect_t powerKernelPort; /* Power context data */ + long powerNotificationID; /* Power event data */ +} cupsd_sysevent_t; + + +typedef struct cupsd_thread_data_s /*** Thread context data ****/ +{ + cupsd_sysevent_t sysevent; /* System event */ + CFRunLoopTimerRef timerRef; /* Timer to delay some change * + * notifications */ +} cupsd_thread_data_t; + + +/* + * Local globals... + */ + +static pthread_t SysEventThread = NULL; + /* Thread to host a runloop */ +static pthread_mutex_t SysEventThreadMutex = { 0 }; + /* Coordinates access to shared gloabals */ +static pthread_cond_t SysEventThreadCond = { 0 }; + /* Thread initialization complete condition */ +static CFRunLoopRef SysEventRunloop = NULL; + /* The runloop. Access must be protected! */ +static CFStringRef ComputerNameKey = NULL, + /* Computer name key */ + BTMMKey = NULL, /* Back to My Mac key */ + NetworkGlobalKeyIPv4 = NULL, + /* Network global IPv4 key */ + NetworkGlobalKeyIPv6 = NULL, + /* Network global IPv6 key */ + NetworkGlobalKeyDNS = NULL, + /* Network global DNS key */ + HostNamesKey = NULL, + /* Host name key */ + NetworkInterfaceKeyIPv4 = NULL, + /* Netowrk interface key */ + NetworkInterfaceKeyIPv6 = NULL; + /* Netowrk interface key */ +static cupsd_sysevent_t LastSysEvent; /* Last system event (for delayed sleep) */ + + +/* + * Local functions... + */ + +static void *sysEventThreadEntry(void); +static void sysEventPowerNotifier(void *context, io_service_t service, + natural_t messageType, + void *messageArgument); +static void sysEventConfigurationNotifier(SCDynamicStoreRef store, + CFArrayRef changedKeys, + void *context); +static void sysEventTimerNotifier(CFRunLoopTimerRef timer, void *context); +static void sysUpdate(void); + + +/* + * 'cupsdAllowSleep()' - Tell the OS it is now OK to sleep. + */ + +void +cupsdAllowSleep(void) +{ + cupsdCleanDirty(); + + IOAllowPowerChange(LastSysEvent.powerKernelPort, + LastSysEvent.powerNotificationID); +} + + +/* + * 'cupsdStartSystemMonitor()' - Start monitoring for system change. + */ + +void +cupsdStartSystemMonitor(void) +{ + int flags; /* fcntl flags on pipe */ + + + if (cupsdOpenPipe(SysEventPipes)) + { + cupsdLogMessage(CUPSD_LOG_ERROR, "System event monitor pipe() failed - %s!", + strerror(errno)); + return; + } + + cupsdAddSelect(SysEventPipes[0], (cupsd_selfunc_t)sysUpdate, NULL, NULL); + + /* + * Set non-blocking mode on the descriptor we will be receiving notification + * events on. + */ + + flags = fcntl(SysEventPipes[0], F_GETFL, 0); + fcntl(SysEventPipes[0], F_SETFL, flags | O_NONBLOCK); + + /* + * Start the thread that runs the runloop... + */ + + pthread_mutex_init(&SysEventThreadMutex, NULL); + pthread_cond_init(&SysEventThreadCond, NULL); + pthread_create(&SysEventThread, NULL, (void *(*)())sysEventThreadEntry, NULL); +} + + +/* + * 'cupsdStopSystemMonitor()' - Stop monitoring for system change. + */ + +void +cupsdStopSystemMonitor(void) +{ + CFRunLoopRef rl; /* The event handler runloop */ + + + if (SysEventThread) + { + /* + * Make sure the thread has completed it's initialization and + * stored it's runloop reference in the shared global. + */ + + pthread_mutex_lock(&SysEventThreadMutex); + + if (!SysEventRunloop) + pthread_cond_wait(&SysEventThreadCond, &SysEventThreadMutex); + + rl = SysEventRunloop; + SysEventRunloop = NULL; + + pthread_mutex_unlock(&SysEventThreadMutex); + + if (rl) + CFRunLoopStop(rl); + + pthread_join(SysEventThread, NULL); + pthread_mutex_destroy(&SysEventThreadMutex); + pthread_cond_destroy(&SysEventThreadCond); + } + + if (SysEventPipes[0] >= 0) + { + cupsdRemoveSelect(SysEventPipes[0]); + cupsdClosePipe(SysEventPipes); + } +} + + +/* + * 'sysEventThreadEntry()' - A thread to receive power and computer name + * change notifications. + */ + +static void * /* O - Return status/value */ +sysEventThreadEntry(void) +{ + io_object_t powerNotifierObj; + /* Power notifier object */ + IONotificationPortRef powerNotifierPort; + /* Power notifier port */ + SCDynamicStoreRef store = NULL;/* System Config dynamic store */ + CFRunLoopSourceRef powerRLS = NULL,/* Power runloop source */ + storeRLS = NULL;/* System Config runloop source */ + CFStringRef key[6], /* System Config keys */ + pattern[2]; /* System Config patterns */ + CFArrayRef keys = NULL, /* System Config key array*/ + patterns = NULL;/* System Config pattern array */ + SCDynamicStoreContext storeContext; /* Dynamic store context */ + CFRunLoopTimerContext timerContext; /* Timer context */ + cupsd_thread_data_t threadData; /* Thread context data for the * + * runloop notifiers */ + + + /* + * Register for power state change notifications + */ + + bzero(&threadData, sizeof(threadData)); + + threadData.sysevent.powerKernelPort = + IORegisterForSystemPower(&threadData, &powerNotifierPort, + sysEventPowerNotifier, &powerNotifierObj); + + if (threadData.sysevent.powerKernelPort) + { + powerRLS = IONotificationPortGetRunLoopSource(powerNotifierPort); + CFRunLoopAddSource(CFRunLoopGetCurrent(), powerRLS, kCFRunLoopDefaultMode); + } + else + DEBUG_puts("sysEventThreadEntry: error registering for system power " + "notifications"); + + /* + * Register for system configuration change notifications + */ + + bzero(&storeContext, sizeof(storeContext)); + storeContext.info = &threadData; + + store = SCDynamicStoreCreate(kCFAllocatorDefault, CFSTR("cupsd"), + sysEventConfigurationNotifier, &storeContext); + + if (!ComputerNameKey) + ComputerNameKey = SCDynamicStoreKeyCreateComputerName(kCFAllocatorDefault); + + if (!BTMMKey) + BTMMKey = SCDynamicStoreKeyCreate(kCFAllocatorDefault, + CFSTR("Setup:/Network/BackToMyMac")); + + if (!NetworkGlobalKeyIPv4) + NetworkGlobalKeyIPv4 = + SCDynamicStoreKeyCreateNetworkGlobalEntity(kCFAllocatorDefault, + kSCDynamicStoreDomainState, + kSCEntNetIPv4); + + if (!NetworkGlobalKeyIPv6) + NetworkGlobalKeyIPv6 = + SCDynamicStoreKeyCreateNetworkGlobalEntity(kCFAllocatorDefault, + kSCDynamicStoreDomainState, + kSCEntNetIPv6); + + if (!NetworkGlobalKeyDNS) + NetworkGlobalKeyDNS = + SCDynamicStoreKeyCreateNetworkGlobalEntity(kCFAllocatorDefault, + kSCDynamicStoreDomainState, + kSCEntNetDNS); + + if (!HostNamesKey) + HostNamesKey = SCDynamicStoreKeyCreateHostNames(kCFAllocatorDefault); + + if (!NetworkInterfaceKeyIPv4) + NetworkInterfaceKeyIPv4 = + SCDynamicStoreKeyCreateNetworkInterfaceEntity(kCFAllocatorDefault, + kSCDynamicStoreDomainState, + kSCCompAnyRegex, + kSCEntNetIPv4); + + if (!NetworkInterfaceKeyIPv6) + NetworkInterfaceKeyIPv6 = + SCDynamicStoreKeyCreateNetworkInterfaceEntity(kCFAllocatorDefault, + kSCDynamicStoreDomainState, + kSCCompAnyRegex, + kSCEntNetIPv6); + + if (store && ComputerNameKey && HostNamesKey && + NetworkGlobalKeyIPv4 && NetworkGlobalKeyIPv6 && NetworkGlobalKeyDNS && + NetworkInterfaceKeyIPv4 && NetworkInterfaceKeyIPv6) + { + key[0] = ComputerNameKey; + key[1] = BTMMKey; + key[2] = NetworkGlobalKeyIPv4; + key[3] = NetworkGlobalKeyIPv6; + key[4] = NetworkGlobalKeyDNS; + key[5] = HostNamesKey; + + pattern[0] = NetworkInterfaceKeyIPv4; + pattern[1] = NetworkInterfaceKeyIPv6; + + keys = CFArrayCreate(kCFAllocatorDefault, (const void **)key, + sizeof(key) / sizeof(key[0]), + &kCFTypeArrayCallBacks); + + patterns = CFArrayCreate(kCFAllocatorDefault, (const void **)pattern, + sizeof(pattern) / sizeof(pattern[0]), + &kCFTypeArrayCallBacks); + + if (keys && patterns && + SCDynamicStoreSetNotificationKeys(store, keys, patterns)) + { + if ((storeRLS = SCDynamicStoreCreateRunLoopSource(kCFAllocatorDefault, + store, 0)) != NULL) + { + CFRunLoopAddSource(CFRunLoopGetCurrent(), storeRLS, + kCFRunLoopDefaultMode); + } + else + DEBUG_printf(("sysEventThreadEntry: SCDynamicStoreCreateRunLoopSource " + "failed: %s\n", SCErrorString(SCError()))); + } + else + DEBUG_printf(("sysEventThreadEntry: SCDynamicStoreSetNotificationKeys " + "failed: %s\n", SCErrorString(SCError()))); + } + else + DEBUG_printf(("sysEventThreadEntry: SCDynamicStoreCreate failed: %s\n", + SCErrorString(SCError()))); + + if (keys) + CFRelease(keys); + + if (patterns) + CFRelease(patterns); + + /* + * Set up a timer to delay the wake change notifications. + * + * The initial time is set a decade or so into the future, we'll adjust + * this later. + */ + + bzero(&timerContext, sizeof(timerContext)); + timerContext.info = &threadData; + + threadData.timerRef = + CFRunLoopTimerCreate(kCFAllocatorDefault, + CFAbsoluteTimeGetCurrent() + (86400L * 365L * 10L), + 86400L * 365L * 10L, 0, 0, sysEventTimerNotifier, + &timerContext); + CFRunLoopAddTimer(CFRunLoopGetCurrent(), threadData.timerRef, + kCFRunLoopDefaultMode); + + /* + * Store our runloop in a global so the main thread can use it to stop us. + */ + + pthread_mutex_lock(&SysEventThreadMutex); + + SysEventRunloop = CFRunLoopGetCurrent(); + + pthread_cond_signal(&SysEventThreadCond); + pthread_mutex_unlock(&SysEventThreadMutex); + + /* + * Disappear into the runloop until it's stopped by the main thread. + */ + + CFRunLoopRun(); + + /* + * Clean up before exiting. + */ + + if (threadData.timerRef) + { + CFRunLoopRemoveTimer(CFRunLoopGetCurrent(), threadData.timerRef, + kCFRunLoopDefaultMode); + CFRelease(threadData.timerRef); + } + + if (threadData.sysevent.powerKernelPort) + { + CFRunLoopRemoveSource(CFRunLoopGetCurrent(), powerRLS, + kCFRunLoopDefaultMode); + IODeregisterForSystemPower(&powerNotifierObj); + IOServiceClose(threadData.sysevent.powerKernelPort); + IONotificationPortDestroy(powerNotifierPort); + } + + if (storeRLS) + { + CFRunLoopRemoveSource(CFRunLoopGetCurrent(), storeRLS, + kCFRunLoopDefaultMode); + CFRunLoopSourceInvalidate(storeRLS); + CFRelease(storeRLS); + } + + if (store) + CFRelease(store); + + pthread_exit(NULL); +} + + +/* + * 'sysEventPowerNotifier()' - Handle power notification events. + */ + +static void +sysEventPowerNotifier( + void *context, /* I - Thread context data */ + io_service_t service, /* I - Unused service info */ + natural_t messageType, /* I - Type of message */ + void *messageArgument) /* I - Message data */ +{ + int sendit = 1; /* Send event to main thread? * + * (0 = no, 1 = yes, 2 = delayed */ + cupsd_thread_data_t *threadData; /* Thread context data */ + + + threadData = (cupsd_thread_data_t *)context; + + (void)service; /* anti-compiler-warning-code */ + + switch (messageType) + { + case kIOMessageCanSystemPowerOff: + case kIOMessageCanSystemSleep: + threadData->sysevent.event |= SYSEVENT_CANSLEEP; + break; + + case kIOMessageSystemWillRestart: + case kIOMessageSystemWillPowerOff: + case kIOMessageSystemWillSleep: + threadData->sysevent.event |= SYSEVENT_WILLSLEEP; + break; + + case kIOMessageSystemHasPoweredOn: + /* + * Because powered on is followed by a net-changed event, delay + * before sending it. + */ + + sendit = 2; + threadData->sysevent.event |= SYSEVENT_WOKE; + break; + + case kIOMessageSystemWillNotPowerOff: + case kIOMessageSystemWillNotSleep: +# ifdef kIOMessageSystemWillPowerOn + case kIOMessageSystemWillPowerOn: +# endif /* kIOMessageSystemWillPowerOn */ + default: + sendit = 0; + break; + } + + if (sendit == 0) + IOAllowPowerChange(threadData->sysevent.powerKernelPort, + (long)messageArgument); + else + { + threadData->sysevent.powerNotificationID = (long)messageArgument; + + if (sendit == 1) + { + /* + * Send the event to the main thread now. + */ + + write(SysEventPipes[1], &threadData->sysevent, + sizeof(threadData->sysevent)); + threadData->sysevent.event = 0; + } + else + { + /* + * Send the event to the main thread after 1 to 2 seconds. + */ + + CFRunLoopTimerSetNextFireDate(threadData->timerRef, + CFAbsoluteTimeGetCurrent() + 2); + } + } +} + + +/* + * 'sysEventConfigurationNotifier()' - Network configuration change notification + * callback. + */ + +static void +sysEventConfigurationNotifier( + SCDynamicStoreRef store, /* I - System data (unused) */ + CFArrayRef changedKeys, /* I - Changed data */ + void *context) /* I - Thread context data */ +{ + cupsd_thread_data_t *threadData; /* Thread context data */ + + + threadData = (cupsd_thread_data_t *)context; + + (void)store; /* anti-compiler-warning-code */ + + CFRange range = CFRangeMake(0, CFArrayGetCount(changedKeys)); + + if (CFArrayContainsValue(changedKeys, range, ComputerNameKey) || + CFArrayContainsValue(changedKeys, range, BTMMKey)) + threadData->sysevent.event |= SYSEVENT_NAMECHANGED; + else + { + threadData->sysevent.event |= SYSEVENT_NETCHANGED; + + /* + * Indicate the network interface list needs updating... + */ + + NetIFUpdate = 1; + } + + /* + * Because we registered for several different kinds of change notifications + * this callback usually gets called several times in a row. We use a timer to + * de-bounce these so we only end up generating one event for the main thread. + */ + + CFRunLoopTimerSetNextFireDate(threadData->timerRef, + CFAbsoluteTimeGetCurrent() + 5); +} + + +/* + * 'sysEventTimerNotifier()' - Handle delayed event notifications. + */ + +static void +sysEventTimerNotifier( + CFRunLoopTimerRef timer, /* I - Timer information */ + void *context) /* I - Thread context data */ +{ + cupsd_thread_data_t *threadData; /* Thread context data */ + + + (void)timer; + + threadData = (cupsd_thread_data_t *)context; + + /* + * If an event is still pending send it to the main thread. + */ + + if (threadData->sysevent.event) + { + write(SysEventPipes[1], &threadData->sysevent, + sizeof(threadData->sysevent)); + threadData->sysevent.event = 0; + } +} + + +/* + * 'sysUpdate()' - Update the current system state. + */ + +static void +sysUpdate(void) +{ + int i; /* Looping var */ + cupsd_sysevent_t sysevent; /* The system event */ + cupsd_printer_t *p; /* Printer information */ + + + /* + * Drain the event pipe... + */ + + while (read((int)SysEventPipes[0], &sysevent, sizeof(sysevent)) + == sizeof(sysevent)) + { + if (sysevent.event & SYSEVENT_CANSLEEP) + { + /* + * If there are active printers that don't have the connecting-to-device + * printer-state-reason then cancel the sleep request (i.e. this reason + * indicates a job that is not yet connected to the printer)... + */ + + for (p = (cupsd_printer_t *)cupsArrayFirst(Printers); + p; + p = (cupsd_printer_t *)cupsArrayNext(Printers)) + { + if (p->job) + { + for (i = 0; i < p->num_reasons; i ++) + if (!strcmp(p->reasons[i], "connecting-to-device")) + break; + + if (!p->num_reasons || i >= p->num_reasons) + break; + } + } + + if (p) + { + cupsdLogMessage(CUPSD_LOG_INFO, + "System sleep canceled because printer %s is active", + p->name); + IOCancelPowerChange(sysevent.powerKernelPort, + sysevent.powerNotificationID); + } + else + { + cupsdLogMessage(CUPSD_LOG_DEBUG, "System wants to sleep"); + IOAllowPowerChange(sysevent.powerKernelPort, + sysevent.powerNotificationID); + } + } + + if (sysevent.event & SYSEVENT_WILLSLEEP) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, "System going to sleep"); + + Sleeping = 1; + + for (p = (cupsd_printer_t *)cupsArrayFirst(Printers); + p; + p = (cupsd_printer_t *)cupsArrayNext(Printers)) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Deregistering local printer \"%s\"", p->name); + cupsdDeregisterPrinter(p, 0); + } + + cupsdCleanDirty(); + +#ifdef kIOPMAssertionTypeDenySystemSleep + /* + * Remove our assertion as needed since the user wants the system to + * sleep (different than idle sleep)... + */ + + if (dark_wake) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, "Releasing dark wake assertion."); + IOPMAssertionRelease(dark_wake); + dark_wake = 0; + } +#endif /* kIOPMAssertionTypeDenySystemSleep */ + + /* + * If we have no printing jobs, allow the power change immediately. + * Otherwise set the SleepJobs time to 15 seconds in the future when + * we'll take more drastic measures... + */ + + if (cupsArrayCount(PrintingJobs) == 0) + IOAllowPowerChange(sysevent.powerKernelPort, + sysevent.powerNotificationID); + else + { + /* + * If there are active printers that don't have the connecting-to-device + * printer-state-reason then delay the sleep request (i.e. this reason + * indicates a job that is not yet connected to the printer)... + */ + + for (p = (cupsd_printer_t *)cupsArrayFirst(Printers); + p; + p = (cupsd_printer_t *)cupsArrayNext(Printers)) + { + if (p->job) + { + for (i = 0; i < p->num_reasons; i ++) + if (!strcmp(p->reasons[i], "connecting-to-device")) + break; + + if (!p->num_reasons || i >= p->num_reasons) + break; + } + } + + if (p) + { + LastSysEvent = sysevent; + SleepJobs = time(NULL) + 10; + } + else + { + IOAllowPowerChange(sysevent.powerKernelPort, + sysevent.powerNotificationID); + } + } + } + + if (sysevent.event & SYSEVENT_WOKE) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, "System woke from sleep"); + IOAllowPowerChange(sysevent.powerKernelPort, + sysevent.powerNotificationID); + Sleeping = 0; + +#ifdef kIOPMAssertionTypeDenySystemSleep + if (cupsArrayCount(PrintingJobs) > 0 && !dark_wake) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, "Asserting dark wake."); + IOPMAssertionCreateWithName(kIOPMAssertionTypeDenySystemSleep, + kIOPMAssertionLevelOn, + CFSTR("org.cups.cupsd"), &dark_wake); + } +#endif /* kIOPMAssertionTypeDenySystemSleep */ + + cupsdCheckJobs(); + } + + if (sysevent.event & SYSEVENT_NETCHANGED) + { + if (!Sleeping) + cupsdLogMessage(CUPSD_LOG_DEBUG, + "System network configuration changed"); + else + cupsdLogMessage(CUPSD_LOG_DEBUG, + "System network configuration changed; " + "ignored while sleeping"); + } + + if (sysevent.event & SYSEVENT_NAMECHANGED) + { + if (!Sleeping) + { + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Computer name or BTMM domains changed"); + + /* + * De-register the individual printers... + */ + + for (p = (cupsd_printer_t *)cupsArrayFirst(Printers); + p; + p = (cupsd_printer_t *)cupsArrayNext(Printers)) + cupsdDeregisterPrinter(p, 1); + + /* + * Update the computer name and BTMM domain list... + */ + + cupsdUpdateDNSSDName(); + + /* + * Now re-register them... + */ + + for (p = (cupsd_printer_t *)cupsArrayFirst(Printers); + p; + p = (cupsd_printer_t *)cupsArrayNext(Printers)) + cupsdRegisterPrinter(p); + } + else + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Computer name or BTMM domains changed; ignored while " + "sleeping"); + } + } +} +#endif /* __APPLE__ */ + + +/* + * End of "$Id$". + */ diff --git a/scheduler/sysman.h b/scheduler/sysman.h new file mode 100644 index 0000000000..810b185f0d --- /dev/null +++ b/scheduler/sysman.h @@ -0,0 +1,64 @@ +/* + * "$Id$" + * + * System management definitions for the CUPS scheduler. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +/* + * Constants... + */ + +#define CUPSD_DIRTY_NONE 0 /* Nothing is dirty */ +#define CUPSD_DIRTY_PRINTERS 1 /* printers.conf is dirty */ +#define CUPSD_DIRTY_CLASSES 2 /* classes.conf is dirty */ +#define CUPSD_DIRTY_PRINTCAP 4 /* printcap is dirty */ +#define CUPSD_DIRTY_JOBS 8 /* jobs.cache or "c" file(s) are dirty */ +#define CUPSD_DIRTY_SUBSCRIPTIONS 16 /* subscriptions.conf is dirty */ + + +/* + * Globals... + */ + +VAR int DirtyFiles VALUE(CUPSD_DIRTY_NONE), + /* What files are dirty? */ + DirtyCleanInterval VALUE(DEFAULT_KEEPALIVE); + /* How often do we write dirty files? */ +VAR time_t DirtyCleanTime VALUE(0); + /* When to clean dirty files next */ +VAR int Sleeping VALUE(0); + /* Non-zero if machine is entering or * + * in a sleep state... */ +VAR time_t SleepJobs VALUE(0); + /* Time when all jobs must be * + * canceled for system sleep. */ +#ifdef __APPLE__ +VAR int SysEventPipes[2] VALUE2(-1,-1); + /* System event notification pipes */ +#endif /* __APPLE__ */ + + +/* + * Prototypes... + */ + +extern void cupsdAllowSleep(void); +extern void cupsdCleanDirty(void); +extern void cupsdMarkDirty(int what); +extern void cupsdSetBusyState(void); +extern void cupsdStartSystemMonitor(void); +extern void cupsdStopSystemMonitor(void); + + +/* + * End of "$Id$". + */ diff --git a/scheduler/testlpd.c b/scheduler/testlpd.c new file mode 100644 index 0000000000..bc90757a98 --- /dev/null +++ b/scheduler/testlpd.c @@ -0,0 +1,550 @@ +/* + * "$Id$" + * + * cups-lpd test program for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Simulate an LPD client. + * do_command() - Send the LPD command and wait for a response. + * print_job() - Submit a file for printing. + * print_waiting() - Print waiting jobs. + * remove_job() - Cancel a print job. + * status_long() - Show the long printer status. + * status_short() - Show the short printer status. + * usage() - Show program usage... + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include +#include +#include +#include + + +/* + * Local functions... + */ + +static int do_command(int outfd, int infd, const char *command); +static int print_job(int outfd, int infd, char *dest, char **args); +static int print_waiting(int outfd, int infd, char *dest); +static int remove_job(int outfd, int infd, char *dest, char **args); +static int status_long(int outfd, int infd, char *dest, char **args); +static int status_short(int outfd, int infd, char *dest, char **args); +static void usage(void) __attribute__((noreturn)); + + +/* + * 'main()' - Simulate an LPD client. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + int status; /* Test status */ + char *op, /* Operation to test */ + **opargs, /* Remaining arguments */ + *dest; /* Destination */ + int cupslpd_argc; /* Argument count for cups-lpd */ + char *cupslpd_argv[1000]; /* Arguments for cups-lpd */ + int cupslpd_stdin[2], /* Standard input for cups-lpd */ + cupslpd_stdout[2], /* Standard output for cups-lpd */ + cupslpd_pid, /* Process ID for cups-lpd */ + cupslpd_status; /* Status of cups-lpd process */ + + + /* + * Collect command-line arguments... + */ + + op = NULL; + opargs = NULL; + dest = NULL; + cupslpd_argc = 1; + cupslpd_argv[0] = (char *)"cups-lpd"; + + for (i = 1; i < argc; i ++) + if (!strncmp(argv[i], "-o", 2)) + { + cupslpd_argv[cupslpd_argc++] = argv[i]; + + if (!argv[i][2]) + { + i ++; + + if (i >= argc) + usage(); + + cupslpd_argv[cupslpd_argc++] = argv[i]; + } + } + else if (argv[i][0] == '-') + usage(); + else if (!op) + op = argv[i]; + else if (!dest) + dest = argv[i]; + else + { + opargs = argv + i; + break; + } + + if (!op || + (!strcmp(op, "print-job") && (!dest || !opargs)) || + (!strcmp(op, "remove-job") && (!dest || !opargs)) || + (strcmp(op, "print-job") && strcmp(op, "print-waiting") && + strcmp(op, "remove-job") && strcmp(op, "status-long") && + strcmp(op, "status-short"))) + { + printf("op=\"%s\", dest=\"%s\", opargs=%p\n", op, dest, opargs); + usage(); + } + + /* + * Run the cups-lpd program using pipes... + */ + + cupslpd_argv[cupslpd_argc] = NULL; + + pipe(cupslpd_stdin); + pipe(cupslpd_stdout); + + if ((cupslpd_pid = fork()) < 0) + { + /* + * Error! + */ + + perror("testlpd: Unable to fork"); + return (1); + } + else if (cupslpd_pid == 0) + { + /* + * Child goes here... + */ + + dup2(cupslpd_stdin[0], 0); + close(cupslpd_stdin[0]); + close(cupslpd_stdin[1]); + + dup2(cupslpd_stdout[1], 1); + close(cupslpd_stdout[0]); + close(cupslpd_stdout[1]); + + execv("./cups-lpd", cupslpd_argv); + + perror("testlpd: Unable to exec ./cups-lpd"); + exit(errno); + } + else + { + close(cupslpd_stdin[0]); + close(cupslpd_stdout[1]); + } + + /* + * Do the operation test... + */ + + if (!strcmp(op, "print-job")) + status = print_job(cupslpd_stdin[1], cupslpd_stdout[0], dest, opargs); + else if (!strcmp(op, "print-waiting")) + status = print_waiting(cupslpd_stdin[1], cupslpd_stdout[0], dest); + else if (!strcmp(op, "remove-job")) + status = remove_job(cupslpd_stdin[1], cupslpd_stdout[0], dest, opargs); + else if (!strcmp(op, "status-long")) + status = status_long(cupslpd_stdin[1], cupslpd_stdout[0], dest, opargs); + else if (!strcmp(op, "status-short")) + status = status_short(cupslpd_stdin[1], cupslpd_stdout[0], dest, opargs); + else + { + printf("Unknown operation \"%s\"!\n", op); + status = 1; + } + + /* + * Kill the test program... + */ + + close(cupslpd_stdin[1]); + close(cupslpd_stdout[0]); + + while (wait(&cupslpd_status) != cupslpd_pid); + + printf("cups-lpd exit status was %d...\n", cupslpd_status); + + /* + * Return the test status... + */ + + return (status); +} + + +/* + * 'do_command()' - Send the LPD command and wait for a response. + */ + +static int /* O - Status from cups-lpd */ +do_command(int outfd, /* I - Command file descriptor */ + int infd, /* I - Response file descriptor */ + const char *command) /* I - Command line to send */ +{ + int len; /* Length of command line */ + char status; /* Status byte */ + + + printf("COMMAND: %02X %s", command[0], command + 1); + + len = strlen(command); + + if (write(outfd, command, len) < len) + { + puts(" Write failed!"); + return (-1); + } + + if (read(infd, &status, 1) < 1) + puts("STATUS: ERROR"); + else + printf("STATUS: %d\n", status); + + return (status); +} + + +/* + * 'print_job()' - Submit a file for printing. + */ + +static int /* O - Status from cups-lpd */ +print_job(int outfd, /* I - Command file descriptor */ + int infd, /* I - Response file descriptor */ + char *dest, /* I - Destination */ + char **args) /* I - Arguments */ +{ + int fd; /* Print file descriptor */ + char command[1024], /* Command buffer */ + control[1024], /* Control file */ + buffer[8192]; /* Print buffer */ + int status; /* Status of command */ + struct stat fileinfo; /* File information */ + char *jobname; /* Job name */ + int sequence; /* Sequence number */ + int bytes; /* Bytes read/written */ + + + /* + * Check the print file... + */ + + if (stat(args[0], &fileinfo)) + { + perror(args[0]); + return (-1); + } + + if ((fd = open(args[0], O_RDONLY)) < 0) + { + perror(args[0]); + return (-1); + } + + /* + * Send the "receive print job" command... + */ + + snprintf(command, sizeof(command), "\002%s\n", dest); + if ((status = do_command(outfd, infd, command)) != 0) + { + close(fd); + return (status); + } + + /* + * Format a control file string that will be used to submit the job... + */ + + if ((jobname = strrchr(args[0], '/')) != NULL) + jobname ++; + else + jobname = args[0]; + + sequence = (int)getpid() % 1000; + + snprintf(control, sizeof(control), + "Hlocalhost\n" + "P%s\n" + "J%s\n" + "ldfA%03dlocalhost\n" + "UdfA%03dlocalhost\n" + "N%s\n", + cupsUser(), jobname, sequence, sequence, jobname); + + /* + * Send the control file... + */ + + bytes = strlen(control); + + snprintf(command, sizeof(command), "\002%d cfA%03dlocalhost\n", + bytes, sequence); + + if ((status = do_command(outfd, infd, command)) != 0) + { + close(fd); + return (status); + } + + bytes ++; + + if (write(outfd, control, bytes) < bytes) + { + printf("CONTROL: Unable to write %d bytes!\n", bytes); + close(fd); + return (-1); + } + + printf("CONTROL: Wrote %d bytes.\n", bytes); + + if (read(infd, command, 1) < 1) + { + puts("STATUS: ERROR"); + close(fd); + return (-1); + } + else + { + status = command[0]; + + printf("STATUS: %d\n", status); + } + + /* + * Send the data file... + */ + + snprintf(command, sizeof(command), "\003%d dfA%03dlocalhost\n", + (int)fileinfo.st_size, sequence); + + if ((status = do_command(outfd, infd, command)) != 0) + { + close(fd); + return (status); + } + + while ((bytes = read(fd, buffer, sizeof(buffer))) > 0) + { + if (write(outfd, buffer, bytes) < bytes) + { + printf("DATA: Unable to write %d bytes!\n", bytes); + close(fd); + return (-1); + } + } + + write(outfd, "", 1); + + close(fd); + + printf("DATA: Wrote %d bytes.\n", (int)fileinfo.st_size); + + if (read(infd, command, 1) < 1) + { + puts("STATUS: ERROR"); + close(fd); + return (-1); + } + else + { + status = command[0]; + + printf("STATUS: %d\n", status); + } + + return (status); +} + + +/* + * 'print_waiting()' - Print waiting jobs. + */ + +static int /* O - Status from cups-lpd */ +print_waiting(int outfd, /* I - Command file descriptor */ + int infd, /* I - Response file descriptor */ + char *dest) /* I - Destination */ +{ + char command[1024]; /* Command buffer */ + + + /* + * Send the "print waiting jobs" command... + */ + + snprintf(command, sizeof(command), "\001%s\n", dest); + + return (do_command(outfd, infd, command)); +} + + +/* + * 'remove_job()' - Cancel a print job. + */ + +static int /* O - Status from cups-lpd */ +remove_job(int outfd, /* I - Command file descriptor */ + int infd, /* I - Response file descriptor */ + char *dest, /* I - Destination */ + char **args) /* I - Arguments */ +{ + int i; /* Looping var */ + char command[1024]; /* Command buffer */ + + /* + * Send the "remove jobs" command... + */ + + snprintf(command, sizeof(command), "\005%s", dest); + + for (i = 0; args[i]; i ++) + { + strlcat(command, " ", sizeof(command)); + strlcat(command, args[i], sizeof(command)); + } + + strlcat(command, "\n", sizeof(command)); + + return (do_command(outfd, infd, command)); +} + + +/* + * 'status_long()' - Show the long printer status. + */ + +static int /* O - Status from cups-lpd */ +status_long(int outfd, /* I - Command file descriptor */ + int infd, /* I - Response file descriptor */ + char *dest, /* I - Destination */ + char **args) /* I - Arguments */ +{ + char command[1024], /* Command buffer */ + buffer[8192]; /* Status buffer */ + int bytes; /* Bytes read/written */ + + + /* + * Send the "send short status" command... + */ + + if (args) + snprintf(command, sizeof(command), "\004%s %s\n", dest, args[0]); + else + snprintf(command, sizeof(command), "\004%s\n", dest); + + bytes = strlen(command); + + if (write(outfd, command, bytes) < bytes) + return (-1); + + /* + * Read the status back... + */ + + while ((bytes = read(infd, buffer, sizeof(buffer))) > 0) + { + fwrite(buffer, 1, bytes, stdout); + fflush(stdout); + } + + return (0); +} + + +/* + * 'status_short()' - Show the short printer status. + */ + +static int /* O - Status from cups-lpd */ +status_short(int outfd, /* I - Command file descriptor */ + int infd, /* I - Response file descriptor */ + char *dest, /* I - Destination */ + char **args) /* I - Arguments */ +{ + char command[1024], /* Command buffer */ + buffer[8192]; /* Status buffer */ + int bytes; /* Bytes read/written */ + + + /* + * Send the "send short status" command... + */ + + if (args) + snprintf(command, sizeof(command), "\003%s %s\n", dest, args[0]); + else + snprintf(command, sizeof(command), "\003%s\n", dest); + + bytes = strlen(command); + + if (write(outfd, command, bytes) < bytes) + return (-1); + + /* + * Read the status back... + */ + + while ((bytes = read(infd, buffer, sizeof(buffer))) > 0) + { + fwrite(buffer, 1, bytes, stdout); + fflush(stdout); + } + + return (0); +} + + +/* + * 'usage()' - Show program usage... + */ + +static void +usage(void) +{ + puts("Usage: testlpd [options] print-job printer filename [... filename]"); + puts(" testlpd [options] print-waiting [printer or user]"); + puts(" testlpd [options] remove-job printer [user [job-id]]"); + puts(" testlpd [options] status-long [printer or user]"); + puts(" testlpd [options] status-short [printer or user]"); + puts(""); + puts("Options:"); + puts(" -o name=value"); + + exit(0); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/testmime.c b/scheduler/testmime.c new file mode 100644 index 0000000000..a754e748db --- /dev/null +++ b/scheduler/testmime.c @@ -0,0 +1,531 @@ +/* + * "$Id$" + * + * MIME test program for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Main entry for the test program. + * add_ppd_filter() - Add a printer filter from a PPD. + * add_ppd_filters() - Add all filters from a PPD. + * print_rules() - Print the rules for a file type... + * type_dir() - Show the MIME types for a given directory. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include +#include "mime.h" + + +/* + * Local functions... + */ + +static void add_ppd_filter(mime_t *mime, mime_type_t *filtertype, + const char *filter); +static void add_ppd_filters(mime_t *mime, ppd_file_t *ppd); +static void print_rules(mime_magic_t *rules); +static void type_dir(mime_t *mime, const char *dirname); + + +/* + * 'main()' - Main entry for the test program. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping vars */ + const char *filter_path; /* Filter path */ + char super[MIME_MAX_SUPER], /* Super-type name */ + type[MIME_MAX_TYPE]; /* Type name */ + int compression; /* Compression of file */ + int cost; /* Cost of filters */ + mime_t *mime; /* MIME database */ + mime_type_t *src, /* Source type */ + *dst; /* Destination type */ + struct stat srcinfo; /* Source information */ + ppd_file_t *ppd; /* PPD file */ + cups_array_t *filters; /* Filters for the file */ + mime_filter_t *filter; /* Current filter */ + + + mime = NULL; + src = NULL; + dst = NULL; + ppd = NULL; + filter_path = "../filter:" CUPS_SERVERBIN "/filter"; + + srcinfo.st_size = 0; + + for (i = 1; i < argc; i ++) + if (!strcmp(argv[i], "-d")) + { + i ++; + + if (i < argc) + { + mime = mimeLoad(argv[i], filter_path); + + if (ppd) + add_ppd_filters(mime, ppd); + } + } + else if (!strcmp(argv[i], "-f")) + { + i ++; + + if (i < argc) + filter_path = argv[i]; + } + else if (!strcmp(argv[i], "-p")) + { + i ++; + + if (i < argc) + { + ppd = ppdOpenFile(argv[i]); + + if (mime) + add_ppd_filters(mime, ppd); + } + } + else if (!src) + { + if (!mime) + mime = mimeLoad("../conf", filter_path); + + if (ppd) + add_ppd_filters(mime, ppd); + + src = mimeFileType(mime, argv[i], NULL, &compression); + stat(argv[i], &srcinfo); + + if (src) + printf("%s: %s/%s%s\n", argv[i], src->super, src->type, + compression ? " (gzipped)" : ""); + else if ((src = mimeType(mime, "application", "octet-stream")) != NULL) + printf("%s: application/octet-stream\n", argv[i]); + else + { + printf("%s: unknown\n", argv[i]); + if (mime) + mimeDelete(mime); + return (1); + } + } + else + { + sscanf(argv[i], "%15[^/]/%31s", super, type); + dst = mimeType(mime, super, type); + + filters = mimeFilter2(mime, src, srcinfo.st_size, dst, &cost); + + if (!filters) + { + printf("No filters to convert from %s/%s to %s.\n", src->super, + src->type, argv[i]); + } + else + { + int first = 1; /* First filter shown? */ + + printf("Filter cost = %d\n", cost); + + for (filter = (mime_filter_t *)cupsArrayFirst(filters); + filter; + filter = (mime_filter_t *)cupsArrayNext(filters)) + { + if (!strcmp(filter->filter, "-")) + continue; + + if (first) + { + first = 0; + fputs(filter->filter, stdout); + } + else + printf(" | %s", filter->filter); + } + + putchar('\n'); + + cupsArrayDelete(filters); + } + } + + if (!mime) + { + mime = mimeLoad("../conf", filter_path); + if (ppd) + add_ppd_filters(mime, ppd); + } + + if (!src) + { + puts("MIME database types:"); + for (src = mimeFirstType(mime); src; src = mimeNextType(mime)) + { + printf("\t%s/%s (%d):\n", src->super, src->type, src->priority); + print_rules(src->rules); + puts(""); + } + + puts(""); + + puts("MIME database filters:"); + for (filter = mimeFirstFilter(mime); filter; filter = mimeNextFilter(mime)) + printf("\t%s/%s to %s/%s: %s (%d)\n", + filter->src->super, filter->src->type, + filter->dst->super, filter->dst->type, + filter->filter, filter->cost); + + type_dir(mime, "../doc"); + } + + return (0); +} + + +/* + * 'add_printer_filter()' - Add a printer filter from a PPD. + */ + +static void +add_ppd_filter(mime_t *mime, /* I - MIME database */ + mime_type_t *filtertype, /* I - Filter or prefilter MIME type */ + const char *filter) /* I - Filter to add */ +{ + char super[MIME_MAX_SUPER], /* Super-type for filter */ + type[MIME_MAX_TYPE], /* Type for filter */ + dsuper[MIME_MAX_SUPER], /* Destination super-type for filter */ + dtype[MIME_MAX_TYPE], /* Destination type for filter */ + dest[MIME_MAX_SUPER + MIME_MAX_TYPE + 2], + /* Destination super/type */ + program[1024]; /* Program/filter name */ + int cost; /* Cost of filter */ + size_t maxsize = 0; /* Maximum supported file size */ + mime_type_t *temptype, /* MIME type looping var */ + *desttype; /* Destination MIME type */ + mime_filter_t *filterptr; /* MIME filter */ + + + DEBUG_printf(("add_ppd_filter(mime=%p, filtertype=%p(%s/%s), filter=\"%s\")", + mime, filtertype, filtertype->super, filtertype->type, filter)); + + /* + * Parse the filter string; it should be in one of the following formats: + * + * source/type cost program + * source/type cost maxsize(nnnn) program + * source/type dest/type cost program + * source/type dest/type cost maxsize(nnnn) program + */ + + if (sscanf(filter, "%15[^/]/%255s%*[ \t]%15[^/]/%255s%d%*[ \t]%1023[^\n]", + super, type, dsuper, dtype, &cost, program) == 6) + { + snprintf(dest, sizeof(dest), "test/%s/%s", dsuper, dtype); + + if ((desttype = mimeType(mime, "printer", dest)) == NULL) + desttype = mimeAddType(mime, "printer", dest); + } + else + { + if (sscanf(filter, "%15[^/]/%255s%d%*[ \t]%1023[^\n]", super, type, &cost, + program) == 4) + { + desttype = filtertype; + } + else + { + printf("testmime: Invalid filter string \"%s\".\n", filter); + return; + } + } + + if (!strncmp(program, "maxsize(", 8)) + { + char *ptr; /* Pointer into maxsize(nnnn) program */ + + maxsize = strtoll(program + 8, &ptr, 10); + + if (*ptr != ')') + { + printf("testmime: Invalid filter string \"%s\".\n", filter); + return; + } + + ptr ++; + while (_cups_isspace(*ptr)) + ptr ++; + + _cups_strcpy(program, ptr); + } + + /* + * Add the filter to the MIME database, supporting wildcards as needed... + */ + + for (temptype = mimeFirstType(mime); + temptype; + temptype = mimeNextType(mime)) + if (((super[0] == '*' && _cups_strcasecmp(temptype->super, "printer")) || + !_cups_strcasecmp(temptype->super, super)) && + (type[0] == '*' || !_cups_strcasecmp(temptype->type, type))) + { + if (desttype != filtertype) + { + DEBUG_printf(("add_ppd_filter: Adding filter %s/%s %s/%s %d %s", + temptype->super, temptype->type, desttype->super, + desttype->type, cost, program)); + filterptr = mimeAddFilter(mime, temptype, desttype, cost, program); + + if (!mimeFilterLookup(mime, desttype, filtertype)) + { + DEBUG_printf(("add_printer_filter: Adding filter %s/%s %s/%s 0 -", + desttype->super, desttype->type, filtertype->super, + filtertype->type)); + mimeAddFilter(mime, desttype, filtertype, 0, "-"); + } + } + else + { + DEBUG_printf(("add_printer_filter: Adding filter %s/%s %s/%s %d %s", + temptype->super, temptype->type, filtertype->super, + filtertype->type, cost, program)); + filterptr = mimeAddFilter(mime, temptype, filtertype, cost, program); + } + + if (filterptr) + filterptr->maxsize = maxsize; + } +} + + +/* + * 'add_ppd_filters()' - Add all filters from a PPD. + */ + +static void +add_ppd_filters(mime_t *mime, /* I - MIME database */ + ppd_file_t *ppd) /* I - PPD file */ +{ + _ppd_cache_t *pc; /* Cache data for PPD */ + const char *value; /* Filter definition value */ + mime_type_t *filter, /* Filter type */ + *prefilter; /* Pre-filter type */ + + + pc = _ppdCacheCreateWithPPD(ppd); + if (!pc) + return; + + filter = mimeAddType(mime, "printer", "test"); + + if (pc->filters) + { + for (value = (const char *)cupsArrayFirst(pc->filters); + value; + value = (const char *)cupsArrayNext(pc->filters)) + add_ppd_filter(mime, filter, value); + } + else + { + add_ppd_filter(mime, filter, "application/vnd.cups-raw 0 -"); + add_ppd_filter(mime, filter, "application/vnd.cups-postscript 0 -"); + } + + if (pc->prefilters) + { + prefilter = mimeAddType(mime, "prefilter", "test"); + + for (value = (const char *)cupsArrayFirst(pc->prefilters); + value; + value = (const char *)cupsArrayNext(pc->prefilters)) + add_ppd_filter(mime, prefilter, value); + } +} + + +/* + * 'print_rules()' - Print the rules for a file type... + */ + +static void +print_rules(mime_magic_t *rules) /* I - Rules to print */ +{ + int i; /* Looping var */ + static char indent[255] = "\t"; /* Indentation for rules */ + + + if (rules == NULL) + return; + + while (rules != NULL) + { + printf("%s[%p] ", indent, rules); + + if (rules->invert) + printf("NOT "); + + switch (rules->op) + { + case MIME_MAGIC_MATCH : + printf("match(%s)", rules->value.matchv); + break; + case MIME_MAGIC_LOCALE : + printf("locale(%s)", rules->value.localev); + break; + case MIME_MAGIC_ASCII : + printf("ascii(%d,%d)", rules->offset, rules->length); + break; + case MIME_MAGIC_PRINTABLE : + printf("printable(%d,%d)", rules->offset, rules->length); + break; + case MIME_MAGIC_STRING : + printf("string(%d,", rules->offset); + for (i = 0; i < rules->length; i ++) + if (rules->value.stringv[i] < ' ' || + rules->value.stringv[i] > 126) + printf("<%02X>", rules->value.stringv[i]); + else + putchar(rules->value.stringv[i]); + putchar(')'); + break; + case MIME_MAGIC_CHAR : + printf("char(%d,%d)", rules->offset, rules->value.charv); + break; + case MIME_MAGIC_SHORT : + printf("short(%d,%d)", rules->offset, rules->value.shortv); + break; + case MIME_MAGIC_INT : + printf("int(%d,%d)", rules->offset, rules->value.intv); + break; + case MIME_MAGIC_CONTAINS : + printf("contains(%d,%d,", rules->offset, rules->region); + for (i = 0; i < rules->length; i ++) + if (rules->value.stringv[i] < ' ' || + rules->value.stringv[i] > 126) + printf("<%02X>", rules->value.stringv[i]); + else + putchar(rules->value.stringv[i]); + putchar(')'); + break; + default : + break; + } + + if (rules->child != NULL) + { + if (rules->op == MIME_MAGIC_OR) + puts("OR ("); + else + puts("AND ("); + + strcat(indent, "\t"); + print_rules(rules->child); + indent[strlen(indent) - 1] = '\0'; + printf("%s)\n", indent); + } + else + putchar('\n'); + + rules = rules->next; + } +} + + +/* + * 'type_dir()' - Show the MIME types for a given directory. + */ + +static void +type_dir(mime_t *mime, /* I - MIME database */ + const char *dirname) /* I - Directory */ +{ + cups_dir_t *dir; /* Directory */ + cups_dentry_t *dent; /* Directory entry */ + char filename[1024]; /* File to type */ + mime_type_t *filetype; /* File type */ + int compression; /* Compressed file? */ + mime_type_t *pstype; /* application/vnd.cups-postscript */ + cups_array_t *filters; /* Filters to pstype */ + mime_filter_t *filter; /* Current filter */ + int cost; /* Filter cost */ + + + dir = cupsDirOpen(dirname); + if (!dir) + return; + + pstype = mimeType(mime, "application", "vnd.cups-postscript"); + + while ((dent = cupsDirRead(dir)) != NULL) + { + if (dent->filename[0] == '.') + continue; + + snprintf(filename, sizeof(filename), "%s/%s", dirname, dent->filename); + + if (S_ISDIR(dent->fileinfo.st_mode)) + type_dir(mime, filename); + + if (!S_ISREG(dent->fileinfo.st_mode)) + continue; + + filetype = mimeFileType(mime, filename, NULL, &compression); + + if (filetype) + { + printf("%s: %s/%s%s\n", filename, filetype->super, filetype->type, + compression ? " (compressed)" : ""); + + filters = mimeFilter(mime, filetype, pstype, &cost); + + if (!filters) + puts(" No filters to convert application/vnd.cups-postscript."); + else + { + printf(" Filter cost = %d\n", cost); + + filter = (mime_filter_t *)cupsArrayFirst(filters); + printf(" %s", filter->filter); + + for (filter = (mime_filter_t *)cupsArrayNext(filters); + filter; + filter = (mime_filter_t *)cupsArrayNext(filters)) + printf(" | %s", filter->filter); + + putchar('\n'); + + cupsArrayDelete(filters); + } + } + else + printf("%s: unknown%s\n", filename, compression ? " (compressed)" : ""); + } + + cupsDirClose(dir); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/testspeed.c b/scheduler/testspeed.c new file mode 100644 index 0000000000..2b59583a40 --- /dev/null +++ b/scheduler/testspeed.c @@ -0,0 +1,365 @@ +/* + * "$Id$" + * + * Scheduler speed test for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Send multiple IPP requests and report on the average response + * time. + * do_test() - Run a test on a specific host... + * usage() - Show program usage... + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include +#include +#include +#include + + +/* + * Local functions... + */ + +static int do_test(const char *server, int port, + http_encryption_t encryption, int requests, + int verbose); +static void usage(void) __attribute__((noreturn)); + + +/* + * 'main()' - Send multiple IPP requests and report on the average response + * time. + */ + +int +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + char *server, /* Server to use */ + *ptr; /* Pointer to port in server */ + int port; /* Port to use */ + http_encryption_t encryption; /* Encryption to use */ + int requests; /* Number of requests to send */ + int children; /* Number of children to fork */ + int good_children; /* Number of children that exited normally */ + int pid; /* Child PID */ + int status; /* Child status */ + time_t start, /* Start time */ + end; /* End time */ + double elapsed; /* Elapsed time */ + int verbose; /* Verbosity */ + + + /* + * Parse command-line options... + */ + + requests = 100; + children = 5; + server = (char *)cupsServer(); + port = ippPort(); + encryption = HTTP_ENCRYPT_IF_REQUESTED; + verbose = 0; + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-') + { + for (ptr = argv[i] + 1; *ptr; ptr ++) + switch (*ptr) + { + case 'c' : /* Number of children */ + i ++; + if (i >= argc) + usage(); + + children = atoi(argv[i]); + break; + + case 'r' : /* Number of requests */ + i ++; + if (i >= argc) + usage(); + + requests = atoi(argv[i]); + break; + + case 'E' : /* Enable encryption */ + encryption = HTTP_ENCRYPT_REQUIRED; + break; + + case 'v' : /* Verbose logging */ + verbose ++; + break; + + default : + usage(); + break; + } + } + else + { + server = argv[i]; + + if (server[0] != '/' && (ptr = strrchr(server, ':')) != NULL) + { + *ptr++ = '\0'; + port = atoi(ptr); + } + } + + /* + * Then create child processes to act as clients... + */ + + if (children > 0) + { + printf("testspeed: Simulating %d clients with %d requests to %s with " + "%sencryption...\n", children, requests, server, + encryption == HTTP_ENCRYPT_IF_REQUESTED ? "no " : ""); + } + + start = time(NULL); + + if (children < 1) + return (do_test(server, port, encryption, requests, verbose)); + else if (children == 1) + good_children = do_test(server, port, encryption, requests, verbose) ? 0 : 1; + else + { + char options[255], /* Command-line options for child */ + reqstr[255], /* Requests string for child */ + serverstr[255]; /* Server:port string for child */ + + + snprintf(reqstr, sizeof(reqstr), "%d", requests); + + if (port == 631 || server[0] == '/') + strlcpy(serverstr, server, sizeof(serverstr)); + else + snprintf(serverstr, sizeof(serverstr), "%s:%d", server, port); + + strlcpy(options, "-cr", sizeof(options)); + + if (encryption == HTTP_ENCRYPT_REQUIRED) + strlcat(options, "E", sizeof(options)); + + if (verbose) + strlcat(options, "v", sizeof(options)); + + for (i = 0; i < children; i ++) + { + fflush(stdout); + + if ((pid = fork()) == 0) + { + /* + * Child goes here... + */ + + execlp(argv[0], argv[0], options, "0", reqstr, serverstr, (char *)NULL); + exit(errno); + } + else if (pid < 0) + { + printf("testspeed: Fork failed: %s\n", strerror(errno)); + break; + } + else + printf("testspeed: Started child %d...\n", pid); + } + + /* + * Wait for children to finish... + */ + + puts("testspeed: Waiting for children to finish..."); + + for (good_children = 0;;) + { + pid = wait(&status); + + if (pid < 0 && errno != EINTR) + break; + + printf("testspeed: Ended child %d (%d)...\n", pid, status / 256); + + if (!status) + good_children ++; + } + } + + /* + * Compute the total run time... + */ + + if (good_children > 0) + { + end = time(NULL); + elapsed = end - start; + i = good_children * requests; + + printf("testspeed: %dx%d=%d requests in %.1fs (%.3fs/r, %.1fr/s)\n", + good_children, requests, i, elapsed, elapsed / i, i / elapsed); + } + + /* + * Exit with no errors... + */ + + return (0); +} + + +/* + * 'do_test()' - Run a test on a specific host... + */ + +static int /* O - Exit status */ +do_test(const char *server, /* I - Server to use */ + int port, /* I - Port number to use */ + http_encryption_t encryption, /* I - Encryption to use */ + int requests, /* I - Number of requests to send */ + int verbose) /* I - Verbose output? */ +{ + int i; /* Looping var */ + http_t *http; /* Connection to server */ + ipp_t *request; /* IPP Request */ + struct timeval start, /* Start time */ + end; /* End time */ + double reqtime, /* Time for this request */ + elapsed; /* Elapsed time */ + int op; /* Current operation */ + static ipp_op_t ops[4] = /* Operations to test... */ + { + IPP_PRINT_JOB, + CUPS_GET_PRINTERS, + CUPS_GET_CLASSES, + IPP_GET_JOBS + }; + + + /* + * Connect to the server... + */ + + if ((http = httpConnectEncrypt(server, port, encryption)) == NULL) + { + printf("testspeed(%d): unable to connect to server - %s\n", (int)getpid(), + strerror(errno)); + return (1); + } + + /* + * Do multiple requests... + */ + + for (elapsed = 0.0, i = 0; i < requests; i ++) + { + /* + * Build a request which requires the following attributes: + * + * attributes-charset + * attributes-natural-language + * + * In addition, IPP_GET_JOBS needs a printer-uri attribute. + */ + + op = ops[i & 3]; + request = ippNewRequest(op); + + gettimeofday(&start, NULL); + + if (verbose) + printf("testspeed(%d): %.6f %s ", (int)getpid(), elapsed, + ippOpString(op)); + + switch (op) + { + case IPP_GET_JOBS : + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, "ipp://localhost/printers/"); + + default : + ippDelete(cupsDoRequest(http, request, "/")); + break; + + case IPP_PRINT_JOB : + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, "ipp://localhost/printers/test"); + ippDelete(cupsDoFileRequest(http, request, "/printers/test", + "../data/testprint.ps")); + break; + } + + gettimeofday(&end, NULL); + + reqtime = (end.tv_sec - start.tv_sec) + + 0.000001 * (end.tv_usec - start.tv_usec); + elapsed += reqtime; + + switch (cupsLastError()) + { + case IPP_OK : + case IPP_NOT_FOUND : + if (verbose) + { + printf("succeeded: %s (%.6f)\n", cupsLastErrorString(), reqtime); + fflush(stdout); + } + break; + + default : + if (!verbose) + printf("testspeed(%d): %s ", (int)getpid(), + ippOpString(ops[i & 3])); + + printf("failed: %s\n", cupsLastErrorString()); + httpClose(http); + return (1); + } + } + + httpClose(http); + + printf("testspeed(%d): %d requests in %.1fs (%.3fs/r, %.1fr/s)\n", + (int)getpid(), i, elapsed, elapsed / i, i / elapsed); + + return (0); +} + + +/* + * 'usage()' - Show program usage... + */ + +static void +usage(void) +{ + puts("Usage: testspeed [-c children] [-h] [-r requests] [-v] [-E] " + "hostname[:port]"); + exit(0); +} + + + +/* + * End of "$Id$". + */ diff --git a/scheduler/testsub.c b/scheduler/testsub.c new file mode 100644 index 0000000000..69c7791f79 --- /dev/null +++ b/scheduler/testsub.c @@ -0,0 +1,523 @@ +/* + * "$Id$" + * + * Scheduler notification tester for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 2006-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Subscribe to the . + * print_attributes() - Print the attributes in a request... + * sigterm_handler() - Flag when the user hits CTRL-C... + * usage() - Show program usage... + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include +#include /* TODO: Update so we don't need this */ + + +/* + * Local globals... + */ + +static int terminate = 0; + + +/* + * Local functions... + */ + +static void print_attributes(ipp_t *ipp, int indent); +static void sigterm_handler(int sig); +static void usage(void) __attribute__((noreturn)); + + +/* + * 'main()' - Subscribe to the . + */ + +int +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + const char *uri; /* URI to use */ + int num_events; /* Number of events */ + const char *events[100]; /* Events */ + int subscription_id, /* notify-subscription-id */ + sequence_number, /* notify-sequence-number */ + interval; /* Interval between polls */ + http_t *http; /* HTTP connection */ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + ipp_attribute_t *attr; /* Current attribute */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* Actions for POSIX signals */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET */ + + + /* + * Parse command-line... + */ + + num_events = 0; + uri = NULL; + + for (i = 1; i < argc; i ++) + if (!strcmp(argv[i], "-E")) + cupsSetEncryption(HTTP_ENCRYPT_REQUIRED); + else if (!strcmp(argv[i], "-e")) + { + i ++; + if (i >= argc || num_events >= 100) + usage(); + + events[num_events] = argv[i]; + num_events ++; + } + else if (!strcmp(argv[i], "-h")) + { + i ++; + if (i >= argc) + usage(); + + cupsSetServer(argv[i]); + } + else if (uri || strncmp(argv[i], "ipp://", 6)) + usage(); + else + uri = argv[i]; + + if (!uri) + usage(); + + if (num_events == 0) + { + events[0] = "all"; + num_events = 1; + } + + /* + * Connect to the server... + */ + + if ((http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption())) == NULL) + { + perror(cupsServer()); + return (1); + } + + /* + * Catch CTRL-C and SIGTERM... + */ + +#ifdef HAVE_SIGSET /* Use System V signals over POSIX to avoid bugs */ + sigset(SIGINT, sigterm_handler); + sigset(SIGTERM, sigterm_handler); +#elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + + sigemptyset(&action.sa_mask); + action.sa_handler = sigterm_handler; + sigaction(SIGINT, &action, NULL); + sigaction(SIGTERM, &action, NULL); +#else + signal(SIGINT, sigterm_handler); + signal(SIGTERM, sigterm_handler); +#endif /* HAVE_SIGSET */ + + /* + * Create the subscription... + */ + + if (strstr(uri, "/jobs/")) + { + request = ippNewRequest(IPP_CREATE_JOB_SUBSCRIPTION); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-uri", NULL, uri); + } + else + { + request = ippNewRequest(IPP_CREATE_PRINTER_SUBSCRIPTION); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", NULL, + uri); + } + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, cupsUser()); + + ippAddStrings(request, IPP_TAG_SUBSCRIPTION, IPP_TAG_KEYWORD, "notify-events", + num_events, NULL, events); + ippAddString(request, IPP_TAG_SUBSCRIPTION, IPP_TAG_KEYWORD, + "notify-pull-method", NULL, "ippget"); + + response = cupsDoRequest(http, request, uri); + if (cupsLastError() >= IPP_BAD_REQUEST) + { + fprintf(stderr, "Create-%s-Subscription: %s\n", + strstr(uri, "/jobs") ? "Job" : "Printer", cupsLastErrorString()); + ippDelete(response); + httpClose(http); + return (1); + } + + if ((attr = ippFindAttribute(response, "notify-subscription-id", + IPP_TAG_INTEGER)) == NULL) + { + fputs("ERROR: No notify-subscription-id in response!\n", stderr); + ippDelete(response); + httpClose(http); + return (1); + } + + subscription_id = attr->values[0].integer; + + printf("Create-%s-Subscription: notify-subscription-id=%d\n", + strstr(uri, "/jobs/") ? "Job" : "Printer", subscription_id); + + ippDelete(response); + + /* + * Monitor for events... + */ + + sequence_number = 0; + + while (!terminate) + { + /* + * Get the current events... + */ + + printf("\nGet-Notifications(%d,%d):", subscription_id, sequence_number); + fflush(stdout); + + request = ippNewRequest(IPP_GET_NOTIFICATIONS); + + if (strstr(uri, "/jobs/")) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-uri", NULL, uri); + else + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", NULL, + uri); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, cupsUser()); + + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_INTEGER, + "notify-subscription-ids", subscription_id); + if (sequence_number) + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_INTEGER, + "notify-sequence-numbers", sequence_number + 1); + + response = cupsDoRequest(http, request, uri); + + printf(" %s\n", ippErrorString(cupsLastError())); + + if (cupsLastError() >= IPP_BAD_REQUEST) + fprintf(stderr, "Get-Notifications: %s\n", cupsLastErrorString()); + else if (response) + { + print_attributes(response, 0); + + for (attr = ippFindAttribute(response, "notify-sequence-number", + IPP_TAG_INTEGER); + attr; + attr = ippFindNextAttribute(response, "notify-sequence-number", + IPP_TAG_INTEGER)) + if (attr->values[0].integer > sequence_number) + sequence_number = attr->values[0].integer; + } + + if ((attr = ippFindAttribute(response, "notify-get-interval", + IPP_TAG_INTEGER)) != NULL && + attr->values[0].integer > 0) + interval = attr->values[0].integer; + else + interval = 5; + + ippDelete(response); + sleep(interval); + } + + /* + * Cancel the subscription... + */ + + printf("\nCancel-Subscription:"); + fflush(stdout); + + request = ippNewRequest(IPP_CANCEL_SUBSCRIPTION); + + if (strstr(uri, "/jobs/")) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-uri", NULL, uri); + else + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", NULL, + uri); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, cupsUser()); + + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_INTEGER, + "notify-subscription-id", subscription_id); + + ippDelete(cupsDoRequest(http, request, uri)); + + printf(" %s\n", ippErrorString(cupsLastError())); + + if (cupsLastError() >= IPP_BAD_REQUEST) + fprintf(stderr, "Cancel-Subscription: %s\n", cupsLastErrorString()); + + /* + * Close the connection and return... + */ + + httpClose(http); + + return (0); +} + + +/* + * 'print_attributes()' - Print the attributes in a request... + */ + +static void +print_attributes(ipp_t *ipp, /* I - IPP request */ + int indent) /* I - Indentation */ +{ + int i; /* Looping var */ + ipp_tag_t group; /* Current group */ + ipp_attribute_t *attr; /* Current attribute */ + _ipp_value_t *val; /* Current value */ + static const char * const tags[] = /* Value/group tag strings */ + { + "reserved-00", + "operation-attributes-tag", + "job-attributes-tag", + "end-of-attributes-tag", + "printer-attributes-tag", + "unsupported-attributes-tag", + "subscription-attributes-tag", + "event-attributes-tag", + "reserved-08", + "reserved-09", + "reserved-0A", + "reserved-0B", + "reserved-0C", + "reserved-0D", + "reserved-0E", + "reserved-0F", + "unsupported", + "default", + "unknown", + "no-value", + "reserved-14", + "not-settable", + "delete-attr", + "admin-define", + "reserved-18", + "reserved-19", + "reserved-1A", + "reserved-1B", + "reserved-1C", + "reserved-1D", + "reserved-1E", + "reserved-1F", + "reserved-20", + "integer", + "boolean", + "enum", + "reserved-24", + "reserved-25", + "reserved-26", + "reserved-27", + "reserved-28", + "reserved-29", + "reserved-2a", + "reserved-2b", + "reserved-2c", + "reserved-2d", + "reserved-2e", + "reserved-2f", + "octetString", + "dateTime", + "resolution", + "rangeOfInteger", + "begCollection", + "textWithLanguage", + "nameWithLanguage", + "endCollection", + "reserved-38", + "reserved-39", + "reserved-3a", + "reserved-3b", + "reserved-3c", + "reserved-3d", + "reserved-3e", + "reserved-3f", + "reserved-40", + "textWithoutLanguage", + "nameWithoutLanguage", + "reserved-43", + "keyword", + "uri", + "uriScheme", + "charset", + "naturalLanguage", + "mimeMediaType", + "memberName" + }; + + + for (group = IPP_TAG_ZERO, attr = ipp->attrs; attr; attr = attr->next) + { + if ((attr->group_tag == IPP_TAG_ZERO && indent <= 8) || !attr->name) + { + group = IPP_TAG_ZERO; + putchar('\n'); + continue; + } + + if (group != attr->group_tag) + { + group = attr->group_tag; + + putchar('\n'); + for (i = 4; i < indent; i ++) + putchar(' '); + + printf("%s:\n\n", tags[group]); + } + + for (i = 0; i < indent; i ++) + putchar(' '); + + printf("%s (", attr->name); + if (attr->num_values > 1) + printf("1setOf "); + printf("%s):", tags[attr->value_tag]); + + switch (attr->value_tag) + { + case IPP_TAG_ENUM : + case IPP_TAG_INTEGER : + for (i = 0, val = attr->values; i < attr->num_values; i ++, val ++) + printf(" %d", val->integer); + putchar('\n'); + break; + + case IPP_TAG_BOOLEAN : + for (i = 0, val = attr->values; i < attr->num_values; i ++, val ++) + printf(" %s", val->boolean ? "true" : "false"); + putchar('\n'); + break; + + case IPP_TAG_RANGE : + for (i = 0, val = attr->values; i < attr->num_values; i ++, val ++) + printf(" %d-%d", val->range.lower, val->range.upper); + putchar('\n'); + break; + + case IPP_TAG_DATE : + { + time_t vtime; /* Date/Time value */ + struct tm *vdate; /* Date info */ + char vstring[256]; /* Formatted time */ + + for (i = 0, val = attr->values; i < attr->num_values; i ++, val ++) + { + vtime = ippDateToTime(val->date); + vdate = localtime(&vtime); + strftime(vstring, sizeof(vstring), "%c", vdate); + printf(" (%s)", vstring); + } + } + putchar('\n'); + break; + + case IPP_TAG_RESOLUTION : + for (i = 0, val = attr->values; i < attr->num_values; i ++, val ++) + printf(" %dx%d%s", val->resolution.xres, val->resolution.yres, + val->resolution.units == IPP_RES_PER_INCH ? "dpi" : "dpc"); + putchar('\n'); + break; + + case IPP_TAG_STRING : + case IPP_TAG_TEXTLANG : + case IPP_TAG_NAMELANG : + case IPP_TAG_TEXT : + case IPP_TAG_NAME : + case IPP_TAG_KEYWORD : + case IPP_TAG_URI : + case IPP_TAG_URISCHEME : + case IPP_TAG_CHARSET : + case IPP_TAG_LANGUAGE : + case IPP_TAG_MIMETYPE : + for (i = 0, val = attr->values; i < attr->num_values; i ++, val ++) + printf(" \"%s\"", val->string.text); + putchar('\n'); + break; + + case IPP_TAG_BEGIN_COLLECTION : + putchar('\n'); + + for (i = 0, val = attr->values; i < attr->num_values; i ++, val ++) + { + if (i) + putchar('\n'); + print_attributes(val->collection, indent + 4); + } + break; + + default : + printf("UNKNOWN (%d values)\n", attr->num_values); + break; + } + } +} + + +/* + * 'sigterm_handler()' - Flag when the user hits CTRL-C... + */ + +static void +sigterm_handler(int sig) /* I - Signal number (unused) */ +{ + (void)sig; + + terminate = 1; +} + + +/* + * 'usage()' - Show program usage... + */ + +static void +usage(void) +{ + puts("Usage: testsub [-E] [-e event ... -e eventN] [-h hostname] URI"); + exit(0); +} + + + +/* + * End of "$Id$". + */ diff --git a/scheduler/type.c b/scheduler/type.c new file mode 100644 index 0000000000..2895116920 --- /dev/null +++ b/scheduler/type.c @@ -0,0 +1,1216 @@ +/* + * "$Id$" + * + * MIME typing routines for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * mimeAddType() - Add a MIME type to a database. + * mimeAddTypeRule() - Add a detection rule for a file type. + * mimeFileType() - Determine the type of a file. + * mimeType() - Lookup a file type. + * mime_compare_types() - Compare two MIME super/type names. + * mime_check_rules() - Check each rule in a list. + * mime_patmatch() - Pattern matching. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include "mime.h" + + +/* + * Local types... + */ + +typedef struct _mime_filebuf_s /**** File buffer for MIME typing ****/ +{ + cups_file_t *fp; /* File pointer */ + int offset, /* Offset in file */ + length; /* Length of buffered data */ + unsigned char buffer[MIME_MAX_BUFFER];/* Buffered data */ +} _mime_filebuf_t; + + +/* + * Local functions... + */ + +static int mime_compare_types(mime_type_t *t0, mime_type_t *t1); +static int mime_check_rules(const char *filename, _mime_filebuf_t *fb, + mime_magic_t *rules); +static int mime_patmatch(const char *s, const char *pat); + + +/* + * Local globals... + */ + +#ifdef DEBUG +static const char * const debug_ops[] = + { /* Test names... */ + "NOP", /* No operation */ + "AND", /* Logical AND of all children */ + "OR", /* Logical OR of all children */ + "MATCH", /* Filename match */ + "ASCII", /* ASCII characters in range */ + "PRINTABLE", /* Printable characters (32-255) */ + "STRING", /* String matches */ + "CHAR", /* Character/byte matches */ + "SHORT", /* Short/16-bit word matches */ + "INT", /* Integer/32-bit word matches */ + "LOCALE", /* Current locale matches string */ + "CONTAINS", /* File contains a string */ + "ISTRING" /* Case-insensitive string matches */ + }; +#endif /* DEBUG */ + + +/* + * 'mimeAddType()' - Add a MIME type to a database. + */ + +mime_type_t * /* O - New (or existing) MIME type */ +mimeAddType(mime_t *mime, /* I - MIME database */ + const char *super, /* I - Super-type name */ + const char *type) /* I - Type name */ +{ + mime_type_t *temp; /* New MIME type */ + size_t typelen; /* Length of type name */ + + + DEBUG_printf(("mimeAddType(mime=%p, super=\"%s\", type=\"%s\")", mime, super, + type)); + + /* + * Range check input... + */ + + if (!mime || !super || !type) + { + DEBUG_puts("1mimeAddType: Returning NULL (bad arguments)."); + return (NULL); + } + + /* + * See if the type already exists; if so, return the existing type... + */ + + if ((temp = mimeType(mime, super, type)) != NULL) + { + DEBUG_printf(("1mimeAddType: Returning %p (existing).", temp)); + return (temp); + } + + /* + * The type doesn't exist; add it... + */ + + if (!mime->types) + mime->types = cupsArrayNew((cups_array_func_t)mime_compare_types, NULL); + + if (!mime->types) + { + DEBUG_puts("1mimeAddType: Returning NULL (no types)."); + return (NULL); + } + + typelen = strlen(type) + 1; + + if ((temp = calloc(1, sizeof(mime_type_t) - MIME_MAX_TYPE + typelen)) == NULL) + { + DEBUG_puts("1mimeAddType: Returning NULL (out of memory)."); + return (NULL); + } + + strlcpy(temp->super, super, sizeof(temp->super)); + memcpy(temp->type, type, typelen); + temp->priority = 100; + + cupsArrayAdd(mime->types, temp); + + DEBUG_printf(("1mimeAddType: Returning %p (new).", temp)); + return (temp); +} + + +/* + * 'mimeAddTypeRule()' - Add a detection rule for a file type. + */ + +int /* O - 0 on success, -1 on failure */ +mimeAddTypeRule(mime_type_t *mt, /* I - Type to add to */ + const char *rule) /* I - Rule to add */ +{ + int num_values, /* Number of values seen */ + op, /* Operation code */ + logic, /* Logic for next rule */ + invert; /* Invert following rule? */ + char name[255], /* Name in rule string */ + value[3][255], /* Value in rule string */ + *ptr, /* Position in name or value */ + quote; /* Quote character */ + int length[3]; /* Length of each parameter */ + mime_magic_t *temp, /* New rule */ + *current; /* Current rule */ + + + DEBUG_printf(("mimeAddTypeRule(mt=%p(%s/%s), rule=\"%s\")", mt, + mt ? mt->super : "???", mt ? mt->type : "???", rule)); + + /* + * Range check input... + */ + + if (!mt || !rule) + return (-1); + + /* + * Find the last rule in the top-level of the rules tree. + */ + + for (current = mt->rules; current; current = current->next) + if (!current->next) + break; + + /* + * Parse the rules string. Most rules are either a file extension or a + * comparison function: + * + * extension + * function(parameters) + */ + + logic = MIME_MAGIC_NOP; + invert = 0; + + while (*rule != '\0') + { + while (isspace(*rule & 255)) + rule ++; + + if (*rule == '(') + { + DEBUG_puts("1mimeAddTypeRule: New parenthesis group"); + logic = MIME_MAGIC_NOP; + rule ++; + } + else if (*rule == ')') + { + DEBUG_puts("1mimeAddTypeRule: Close paren..."); + if (current == NULL || current->parent == NULL) + return (-1); + + current = current->parent; + + if (current->parent == NULL) + logic = MIME_MAGIC_OR; + else + logic = current->parent->op; + + rule ++; + } + else if (*rule == '+' && current != NULL) + { + if (logic != MIME_MAGIC_AND && + current != NULL && current->prev != NULL) + { + /* + * OK, we have more than 1 rule in the current tree level... Make a + * new group tree and move the previous rule to it... + */ + + if ((temp = calloc(1, sizeof(mime_magic_t))) == NULL) + return (-1); + + temp->op = MIME_MAGIC_AND; + temp->child = current; + temp->parent = current->parent; + current->prev->next = temp; + temp->prev = current->prev; + + current->prev = NULL; + current->parent = temp; + + DEBUG_printf(("1mimeAddTypeRule: Creating new AND group %p.", temp)); + } + else if (current->parent) + { + DEBUG_printf(("1mimeAddTypeRule: Setting group %p op to AND.", + current->parent)); + current->parent->op = MIME_MAGIC_AND; + } + + logic = MIME_MAGIC_AND; + rule ++; + } + else if (*rule == ',') + { + if (logic != MIME_MAGIC_OR && current != NULL) + { + /* + * OK, we have two possibilities; either this is the top-level rule or + * we have a bunch of AND rules at this level. + */ + + if (current->parent == NULL) + { + /* + * This is the top-level rule; we have to move *all* of the AND rules + * down a level, as AND has precedence over OR. + */ + + if ((temp = calloc(1, sizeof(mime_magic_t))) == NULL) + return (-1); + + DEBUG_printf(("1mimeAddTypeRule: Creating new AND group %p inside OR " + "group.", temp)); + + while (current->prev != NULL) + { + current->parent = temp; + current = current->prev; + } + + current->parent = temp; + temp->op = MIME_MAGIC_AND; + temp->child = current; + + mt->rules = current = temp; + } + else + { + /* + * This isn't the top rule, so go up one level... + */ + + DEBUG_puts("1mimeAddTypeRule: Going up one level."); + current = current->parent; + } + } + + logic = MIME_MAGIC_OR; + rule ++; + } + else if (*rule == '!') + { + DEBUG_puts("1mimeAddTypeRule: NOT"); + invert = 1; + rule ++; + } + else if (isalnum(*rule & 255)) + { + /* + * Read an extension name or a function... + */ + + ptr = name; + while (isalnum(*rule & 255) && (ptr - name) < (sizeof(name) - 1)) + *ptr++ = *rule++; + + *ptr = '\0'; + + if (*rule == '(') + { + /* + * Read function parameters... + */ + + rule ++; + for (num_values = 0; + num_values < (sizeof(value) / sizeof(value[0])); + num_values ++) + { + ptr = value[num_values]; + + while ((ptr - value[num_values]) < (sizeof(value[0]) - 1) && + *rule != '\0' && *rule != ',' && *rule != ')') + { + if (isspace(*rule & 255)) + { + /* + * Ignore whitespace... + */ + + rule ++; + continue; + } + else if (*rule == '\"' || *rule == '\'') + { + /* + * Copy quoted strings literally... + */ + + quote = *rule++; + + while (*rule != '\0' && *rule != quote && + (ptr - value[num_values]) < (sizeof(value[0]) - 1)) + *ptr++ = *rule++; + + if (*rule == quote) + rule ++; + else + return (-1); + } + else if (*rule == '<') + { + rule ++; + + while (*rule != '>' && *rule != '\0' && + (ptr - value[num_values]) < (sizeof(value[0]) - 1)) + { + if (isxdigit(rule[0] & 255) && isxdigit(rule[1] & 255)) + { + if (isdigit(*rule)) + *ptr = (*rule++ - '0') << 4; + else + *ptr = (tolower(*rule++) - 'a' + 10) << 4; + + if (isdigit(*rule)) + *ptr++ |= *rule++ - '0'; + else + *ptr++ |= tolower(*rule++) - 'a' + 10; + } + else + return (-1); + } + + if (*rule == '>') + rule ++; + else + return (-1); + } + else + *ptr++ = *rule++; + } + + *ptr = '\0'; + length[num_values] = ptr - value[num_values]; + + if (*rule != ',') + { + num_values ++; + break; + } + + rule ++; + } + + if (*rule != ')') + return (-1); + + rule ++; + + /* + * Figure out the function... + */ + + if (!strcmp(name, "match")) + op = MIME_MAGIC_MATCH; + else if (!strcmp(name, "ascii")) + op = MIME_MAGIC_ASCII; + else if (!strcmp(name, "printable")) + op = MIME_MAGIC_PRINTABLE; + else if (!strcmp(name, "string")) + op = MIME_MAGIC_STRING; + else if (!strcmp(name, "istring")) + op = MIME_MAGIC_ISTRING; + else if (!strcmp(name, "char")) + op = MIME_MAGIC_CHAR; + else if (!strcmp(name, "short")) + op = MIME_MAGIC_SHORT; + else if (!strcmp(name, "int")) + op = MIME_MAGIC_INT; + else if (!strcmp(name, "locale")) + op = MIME_MAGIC_LOCALE; + else if (!strcmp(name, "contains")) + op = MIME_MAGIC_CONTAINS; + else if (!strcmp(name, "priority") && num_values == 1) + { + mt->priority = atoi(value[0]); + continue; + } + else + return (-1); + } + else + { + /* + * This is just a filename match on the extension... + */ + + snprintf(value[0], sizeof(value[0]), "*.%s", name); + length[0] = strlen(value[0]); + op = MIME_MAGIC_MATCH; + } + + /* + * Add a rule for this operation. + */ + + if ((temp = calloc(1, sizeof(mime_magic_t))) == NULL) + return (-1); + + temp->invert = invert; + if (current != NULL) + { + temp->parent = current->parent; + current->next = temp; + } + else + mt->rules = temp; + + temp->prev = current; + + if (logic == MIME_MAGIC_NOP) + { + /* + * Add parenthetical grouping... + */ + + DEBUG_printf(("1mimeAddTypeRule: Making new OR group %p for " + "parenthesis.", temp)); + + temp->op = MIME_MAGIC_OR; + + if ((temp->child = calloc(1, sizeof(mime_magic_t))) == NULL) + return (-1); + + temp->child->parent = temp; + temp->child->invert = temp->invert; + temp->invert = 0; + + temp = temp->child; + logic = MIME_MAGIC_OR; + } + + DEBUG_printf(("1mimeAddTypeRule: Adding %p: %s, op=MIME_MAGIC_%s(%d), " + "logic=MIME_MAGIC_%s, invert=%d.", temp, name, + debug_ops[op], op, debug_ops[logic], invert)); + + /* + * Fill in data for the rule... + */ + + current = temp; + temp->op = op; + invert = 0; + + switch (op) + { + case MIME_MAGIC_MATCH : + if (length[0] > (sizeof(temp->value.matchv) - 1)) + return (-1); + strcpy(temp->value.matchv, value[0]); + break; + case MIME_MAGIC_ASCII : + case MIME_MAGIC_PRINTABLE : + temp->offset = strtol(value[0], NULL, 0); + temp->length = strtol(value[1], NULL, 0); + if (temp->length > MIME_MAX_BUFFER) + temp->length = MIME_MAX_BUFFER; + break; + case MIME_MAGIC_STRING : + case MIME_MAGIC_ISTRING : + temp->offset = strtol(value[0], NULL, 0); + if (length[1] > sizeof(temp->value.stringv)) + return (-1); + temp->length = length[1]; + memcpy(temp->value.stringv, value[1], length[1]); + break; + case MIME_MAGIC_CHAR : + temp->offset = strtol(value[0], NULL, 0); + if (length[1] == 1) + temp->value.charv = value[1][0]; + else + temp->value.charv = (unsigned char)strtol(value[1], NULL, 0); + + DEBUG_printf(("1mimeAddTypeRule: CHAR(%d,0x%02x)", temp->offset, + temp->value.charv)); + break; + case MIME_MAGIC_SHORT : + temp->offset = strtol(value[0], NULL, 0); + temp->value.shortv = (unsigned short)strtol(value[1], NULL, 0); + break; + case MIME_MAGIC_INT : + temp->offset = strtol(value[0], NULL, 0); + temp->value.intv = (unsigned)strtol(value[1], NULL, 0); + break; + case MIME_MAGIC_LOCALE : + if (length[0] > (sizeof(temp->value.localev) - 1)) + return (-1); + + strcpy(temp->value.localev, value[0]); + break; + case MIME_MAGIC_CONTAINS : + temp->offset = strtol(value[0], NULL, 0); + temp->region = strtol(value[1], NULL, 0); + if (length[2] > sizeof(temp->value.stringv)) + return (-1); + temp->length = length[2]; + memcpy(temp->value.stringv, value[2], length[2]); + break; + } + } + else + break; + } + + return (0); +} + + +/* + * 'mimeFileType()' - Determine the type of a file. + */ + +mime_type_t * /* O - Type of file */ +mimeFileType(mime_t *mime, /* I - MIME database */ + const char *pathname, /* I - Name of file to check on disk */ + const char *filename, /* I - Original filename or NULL */ + int *compression) /* O - Is the file compressed? */ +{ + _mime_filebuf_t fb; /* File buffer */ + const char *base; /* Base filename of file */ + mime_type_t *type, /* File type */ + *best; /* Best match */ + + + DEBUG_printf(("mimeFileType(mime=%p, pathname=\"%s\", filename=\"%s\", " + "compression=%p)", mime, pathname, filename, compression)); + + /* + * Range check input parameters... + */ + + if (!mime || !pathname) + { + DEBUG_puts("1mimeFileType: Returning NULL."); + return (NULL); + } + + /* + * Try to open the file... + */ + + if ((fb.fp = cupsFileOpen(pathname, "r")) == NULL) + { + DEBUG_printf(("1mimeFileType: Unable to open \"%s\": %s", pathname, + strerror(errno))); + DEBUG_puts("1mimeFileType: Returning NULL."); + return (NULL); + } + + fb.offset = -1; + fb.length = 0; + + /* + * Figure out the base filename (without directory portion)... + */ + + if (filename) + { + if ((base = strrchr(filename, '/')) != NULL) + base ++; + else + base = filename; + } + else if ((base = strrchr(pathname, '/')) != NULL) + base ++; + else + base = pathname; + + /* + * Then check it against all known types... + */ + + for (type = (mime_type_t *)cupsArrayFirst(mime->types), best = NULL; + type; + type = (mime_type_t *)cupsArrayNext(mime->types)) + if (mime_check_rules(base, &fb, type->rules)) + { + if (!best || type->priority > best->priority) + best = type; + } + + /* + * Finally, close the file and return a match (if any)... + */ + + if (compression) + { + *compression = cupsFileCompression(fb.fp); + DEBUG_printf(("1mimeFileType: *compression=%d", *compression)); + } + + cupsFileClose(fb.fp); + + DEBUG_printf(("1mimeFileType: Returning %p(%s/%s).", best, + best ? best->super : "???", best ? best->type : "???")); + return (best); +} + + +/* + * 'mimeType()' - Lookup a file type. + */ + +mime_type_t * /* O - Matching file type definition */ +mimeType(mime_t *mime, /* I - MIME database */ + const char *super, /* I - Super-type name */ + const char *type) /* I - Type name */ +{ + mime_type_t key, /* MIME type search key */ + *mt; /* Matching type */ + + + DEBUG_printf(("mimeType(mime=%p, super=\"%s\", type=\"%s\")", mime, super, + type)); + + /* + * Range check input... + */ + + if (!mime || !super || !type) + { + DEBUG_puts("1mimeType: Returning NULL."); + return (NULL); + } + + /* + * Lookup the type in the array... + */ + + strlcpy(key.super, super, sizeof(key.super)); + strlcpy(key.type, type, sizeof(key.type)); + + mt = (mime_type_t *)cupsArrayFind(mime->types, &key); + DEBUG_printf(("1mimeType: Returning %p.", mt)); + return (mt); +} + + +/* + * 'mime_compare_types()' - Compare two MIME super/type names. + */ + +static int /* O - Result of comparison */ +mime_compare_types(mime_type_t *t0, /* I - First type */ + mime_type_t *t1) /* I - Second type */ +{ + int i; /* Result of comparison */ + + + if ((i = _cups_strcasecmp(t0->super, t1->super)) == 0) + i = _cups_strcasecmp(t0->type, t1->type); + + return (i); +} + + +/* + * 'mime_check_rules()' - Check each rule in a list. + */ + +static int /* O - 1 if match, 0 if no match */ +mime_check_rules( + const char *filename, /* I - Filename */ + _mime_filebuf_t *fb, /* I - File to check */ + mime_magic_t *rules) /* I - Rules to check */ +{ + int n; /* Looping var */ + int region; /* Region to look at */ + int logic, /* Logic to apply */ + result, /* Result of test */ + intv; /* Integer value */ + short shortv; /* Short value */ + unsigned char *bufptr; /* Pointer into buffer */ + + + DEBUG_printf(("4mime_check_rules(filename=\"%s\", fb=%p, rules=%p)", filename, + fb, rules)); + + if (rules == NULL) + return (0); + + if (rules->parent == NULL) + logic = MIME_MAGIC_OR; + else + logic = rules->parent->op; + + result = 0; + + while (rules != NULL) + { + /* + * Compute the result of this rule... + */ + + switch (rules->op) + { + case MIME_MAGIC_MATCH : + result = mime_patmatch(filename, rules->value.matchv); + break; + + case MIME_MAGIC_ASCII : + /* + * Load the buffer if necessary... + */ + + if (fb->offset < 0 || rules->offset < fb->offset || + (rules->offset + rules->length) > (fb->offset + fb->length)) + { + /* + * Reload file buffer... + */ + + cupsFileSeek(fb->fp, rules->offset); + fb->length = cupsFileRead(fb->fp, (char *)fb->buffer, + sizeof(fb->buffer)); + fb->offset = rules->offset; + } + + /* + * Test for ASCII printable characters plus standard control chars. + */ + + if ((rules->offset + rules->length) > (fb->offset + fb->length)) + n = fb->offset + fb->length - rules->offset; + else + n = rules->length; + + bufptr = fb->buffer + rules->offset - fb->offset; + while (n > 0) + if ((*bufptr >= 32 && *bufptr <= 126) || + (*bufptr >= 8 && *bufptr <= 13) || + *bufptr == 26 || *bufptr == 27) + { + n --; + bufptr ++; + } + else + break; + + result = (n == 0); + break; + + case MIME_MAGIC_PRINTABLE : + /* + * Load the buffer if necessary... + */ + + if (fb->offset < 0 || rules->offset < fb->offset || + (rules->offset + rules->length) > (fb->offset + fb->length)) + { + /* + * Reload file buffer... + */ + + cupsFileSeek(fb->fp, rules->offset); + fb->length = cupsFileRead(fb->fp, (char *)fb->buffer, + sizeof(fb->buffer)); + fb->offset = rules->offset; + } + + /* + * Test for 8-bit printable characters plus standard control chars. + */ + + if ((rules->offset + rules->length) > (fb->offset + fb->length)) + n = fb->offset + fb->length - rules->offset; + else + n = rules->length; + + bufptr = fb->buffer + rules->offset - fb->offset; + + while (n > 0) + if (*bufptr >= 128 || + (*bufptr >= 32 && *bufptr <= 126) || + (*bufptr >= 8 && *bufptr <= 13) || + *bufptr == 26 || *bufptr == 27) + { + n --; + bufptr ++; + } + else + break; + + result = (n == 0); + break; + + case MIME_MAGIC_STRING : + DEBUG_printf(("5mime_check_rules: string(%d, \"%s\")", rules->offset, + rules->value.stringv)); + + /* + * Load the buffer if necessary... + */ + + if (fb->offset < 0 || rules->offset < fb->offset || + (rules->offset + rules->length) > (fb->offset + fb->length)) + { + /* + * Reload file buffer... + */ + + cupsFileSeek(fb->fp, rules->offset); + fb->length = cupsFileRead(fb->fp, (char *)fb->buffer, + sizeof(fb->buffer)); + fb->offset = rules->offset; + + DEBUG_printf(("5mime_check_rules: loaded %d byte fb->buffer at %d, starts " + "with \"%c%c%c%c\".", + fb->length, fb->offset, fb->buffer[0], fb->buffer[1], + fb->buffer[2], fb->buffer[3])); + } + + /* + * Compare the buffer against the string. If the file is too + * short then don't compare - it can't match... + */ + + if ((rules->offset + rules->length) > (fb->offset + fb->length)) + result = 0; + else + result = (memcmp(fb->buffer + rules->offset - fb->offset, + rules->value.stringv, rules->length) == 0); + DEBUG_printf(("5mime_check_rules: result=%d", result)); + break; + + case MIME_MAGIC_ISTRING : + /* + * Load the buffer if necessary... + */ + + if (fb->offset < 0 || rules->offset < fb->offset || + (rules->offset + rules->length) > (fb->offset + fb->length)) + { + /* + * Reload file buffer... + */ + + cupsFileSeek(fb->fp, rules->offset); + fb->length = cupsFileRead(fb->fp, (char *)fb->buffer, + sizeof(fb->buffer)); + fb->offset = rules->offset; + } + + /* + * Compare the buffer against the string. If the file is too + * short then don't compare - it can't match... + */ + + if ((rules->offset + rules->length) > (fb->offset + fb->length)) + result = 0; + else + result = (_cups_strncasecmp((char *)fb->buffer + rules->offset - + fb->offset, + rules->value.stringv, rules->length) == 0); + break; + + case MIME_MAGIC_CHAR : + /* + * Load the buffer if necessary... + */ + + if (fb->offset < 0 || rules->offset < fb->offset) + { + /* + * Reload file buffer... + */ + + cupsFileSeek(fb->fp, rules->offset); + fb->length = cupsFileRead(fb->fp, (char *)fb->buffer, + sizeof(fb->buffer)); + fb->offset = rules->offset; + } + + /* + * Compare the character values; if the file is too short, it + * can't match... + */ + + if (fb->length < 1) + result = 0; + else + result = (fb->buffer[rules->offset - fb->offset] == + rules->value.charv); + break; + + case MIME_MAGIC_SHORT : + /* + * Load the buffer if necessary... + */ + + if (fb->offset < 0 || rules->offset < fb->offset || + (rules->offset + 2) > (fb->offset + fb->length)) + { + /* + * Reload file buffer... + */ + + cupsFileSeek(fb->fp, rules->offset); + fb->length = cupsFileRead(fb->fp, (char *)fb->buffer, + sizeof(fb->buffer)); + fb->offset = rules->offset; + } + + /* + * Compare the short values; if the file is too short, it + * can't match... + */ + + if (fb->length < 2) + result = 0; + else + { + bufptr = fb->buffer + rules->offset - fb->offset; + shortv = (bufptr[0] << 8) | bufptr[1]; + result = (shortv == rules->value.shortv); + } + break; + + case MIME_MAGIC_INT : + /* + * Load the buffer if necessary... + */ + + if (fb->offset < 0 || rules->offset < fb->offset || + (rules->offset + 4) > (fb->offset + fb->length)) + { + /* + * Reload file buffer... + */ + + cupsFileSeek(fb->fp, rules->offset); + fb->length = cupsFileRead(fb->fp, (char *)fb->buffer, + sizeof(fb->buffer)); + fb->offset = rules->offset; + } + + /* + * Compare the int values; if the file is too short, it + * can't match... + */ + + if (fb->length < 4) + result = 0; + else + { + bufptr = fb->buffer + rules->offset - fb->offset; + intv = (((((bufptr[0] << 8) | bufptr[1]) << 8) | + bufptr[2]) << 8) | bufptr[3]; + result = (intv == rules->value.intv); + } + break; + + case MIME_MAGIC_LOCALE : +#if defined(WIN32) || defined(__EMX__) || defined(__APPLE__) + result = (strcmp(rules->value.localev, + setlocale(LC_ALL, "")) == 0); +#else + result = (strcmp(rules->value.localev, + setlocale(LC_MESSAGES, "")) == 0); +#endif /* __APPLE__ */ + break; + + case MIME_MAGIC_CONTAINS : + /* + * Load the buffer if necessary... + */ + + if (fb->offset < 0 || rules->offset < fb->offset || + (rules->offset + rules->region) > (fb->offset + fb->length)) + { + /* + * Reload file buffer... + */ + + cupsFileSeek(fb->fp, rules->offset); + fb->length = cupsFileRead(fb->fp, (char *)fb->buffer, + sizeof(fb->buffer)); + fb->offset = rules->offset; + } + + /* + * Compare the buffer against the string. If the file is too + * short then don't compare - it can't match... + */ + + if ((rules->offset + rules->length) > (fb->offset + fb->length)) + result = 0; + else + { + if (fb->length > rules->region) + region = rules->region - rules->length; + else + region = fb->length - rules->length; + + for (n = 0; n < region; n ++) + if ((result = (memcmp(fb->buffer + rules->offset - fb->offset + n, + rules->value.stringv, + rules->length) == 0)) != 0) + break; + } + break; + + default : + if (rules->child != NULL) + result = mime_check_rules(filename, fb, rules->child); + else + result = 0; + break; + } + + /* + * If the logic is inverted, invert the result... + */ + + if (rules->invert) + result = !result; + + /* + * OK, now if the current logic is OR and this result is true, the this + * rule set is true. If the current logic is AND and this result is false, + * the the rule set is false... + */ + + DEBUG_printf(("5mime_check_rules: result of test %p (MIME_MAGIC_%s) is %d", + rules, debug_ops[rules->op], result)); + + if ((result && logic == MIME_MAGIC_OR) || + (!result && logic == MIME_MAGIC_AND)) + return (result); + + /* + * Otherwise the jury is still out on this one, so move to the next rule. + */ + + rules = rules->next; + } + + return (result); +} + + +/* + * 'mime_patmatch()' - Pattern matching. + */ + +static int /* O - 1 if match, 0 if no match */ +mime_patmatch(const char *s, /* I - String to match against */ + const char *pat) /* I - Pattern to match against */ +{ + /* + * Range check the input... + */ + + if (s == NULL || pat == NULL) + return (0); + + /* + * Loop through the pattern and match strings, and stop if we come to a + * point where the strings don't match or we find a complete match. + */ + + while (*s != '\0' && *pat != '\0') + { + if (*pat == '*') + { + /* + * Wildcard - 0 or more characters... + */ + + pat ++; + if (*pat == '\0') + return (1); /* Last pattern char is *, so everything matches... */ + + /* + * Test all remaining combinations until we get to the end of the string. + */ + + while (*s != '\0') + { + if (mime_patmatch(s, pat)) + return (1); + + s ++; + } + } + else if (*pat == '?') + { + /* + * Wildcard - 1 character... + */ + + pat ++; + s ++; + continue; + } + else if (*pat == '[') + { + /* + * Match a character from the input set [chars]... + */ + + pat ++; + while (*pat != ']' && *pat != '\0') + if (*s == *pat) + break; + else + pat ++; + + if (*pat == ']' || *pat == '\0') + return (0); + + while (*pat != ']' && *pat != '\0') + pat ++; + + if (*pat == ']') + pat ++; + + continue; + } + else if (*pat == '\\') + { + /* + * Handle quoted characters... + */ + + pat ++; + } + + /* + * Stop if the pattern and string don't match... + */ + + if (*pat++ != *s++) + return (0); + } + + /* + * Done parsing the pattern and string; return 1 if the last character + * matches and 0 otherwise... + */ + + return (*s == *pat); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/util.c b/scheduler/util.c new file mode 100644 index 0000000000..c8ea8c9e94 --- /dev/null +++ b/scheduler/util.c @@ -0,0 +1,471 @@ +/* + * "$Id$" + * + * Mini-daemon utility functions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * cupsdCompareNames() - Compare two names. + * cupsdCreateStringsArray() - Create a CUPS array of strings. + * cupsdExec() - Run a program with the correct environment. + * cupsdPipeCommand() - Read output from a command. + * cupsdSendIPPGroup() - Send a group tag. + * cupsdSendIPPHeader() - Send the IPP response header. + * cupsdSendIPPInteger() - Send an integer attribute. + * cupsdSendIPPString() - Send a string attribute. + * cupsdSendIPPTrailer() - Send the end-of-message tag. + */ + +/* + * Include necessary headers... + */ + +#include "util.h" +#include +#include +#include +#ifdef __APPLE__ +# include +extern char **environ; +#endif /* __APPLE__ */ + + +/* + * 'cupsdCompareNames()' - Compare two names. + * + * This function basically does a _cups_strcasecmp() of the two strings, + * but is also aware of numbers so that "a2" < "a100". + */ + +int /* O - Result of comparison */ +cupsdCompareNames(const char *s, /* I - First string */ + const char *t) /* I - Second string */ +{ + int diff, /* Difference between digits */ + digits; /* Number of digits */ + + + /* + * Loop through both names, returning only when a difference is + * seen. Also, compare whole numbers rather than just characters, too! + */ + + while (*s && *t) + { + if (isdigit(*s & 255) && isdigit(*t & 255)) + { + /* + * Got a number; start by skipping leading 0's... + */ + + while (*s == '0') + s ++; + while (*t == '0') + t ++; + + /* + * Skip equal digits... + */ + + while (isdigit(*s & 255) && *s == *t) + { + s ++; + t ++; + } + + /* + * Bounce out if *s and *t aren't both digits... + */ + + if (isdigit(*s & 255) && !isdigit(*t & 255)) + return (1); + else if (!isdigit(*s & 255) && isdigit(*t & 255)) + return (-1); + else if (!isdigit(*s & 255) || !isdigit(*t & 255)) + continue; + + if (*s < *t) + diff = -1; + else + diff = 1; + + /* + * Figure out how many more digits there are... + */ + + digits = 0; + s ++; + t ++; + + while (isdigit(*s & 255)) + { + digits ++; + s ++; + } + + while (isdigit(*t & 255)) + { + digits --; + t ++; + } + + /* + * Return if the number or value of the digits is different... + */ + + if (digits < 0) + return (-1); + else if (digits > 0) + return (1); + else if (diff) + return (diff); + } + else if (tolower(*s) < tolower(*t)) + return (-1); + else if (tolower(*s) > tolower(*t)) + return (1); + else + { + s ++; + t ++; + } + } + + /* + * Return the results of the final comparison... + */ + + if (*s) + return (1); + else if (*t) + return (-1); + else + return (0); +} + + +/* + * 'cupsdCreateStringsArray()' - Create a CUPS array of strings. + */ + +cups_array_t * /* O - CUPS array */ +cupsdCreateStringsArray(const char *s) /* I - Comma-delimited strings */ +{ + if (!s || !*s) + return (NULL); + else + return (_cupsArrayNewStrings(s)); +} + + +/* + * 'cupsdExec()' - Run a program with the correct environment. + * + * On Mac OS X, we need to update the CFProcessPath environment variable that + * is passed in the environment so the child can access its bundled resources. + */ + +int /* O - exec() status */ +cupsdExec(const char *command, /* I - Full path to program */ + char **argv) /* I - Command-line arguments */ +{ +#ifdef __APPLE__ + int i, j; /* Looping vars */ + char *envp[500], /* Array of environment variables */ + cfprocesspath[1024], /* CFProcessPath environment variable */ + linkpath[1024]; /* Link path for symlinks... */ + int linkbytes; /* Bytes for link path */ + + + /* + * Some Mac OS X programs are bundled and need the CFProcessPath environment + * variable defined. If the command is a symlink, resolve the link and point + * to the resolved location, otherwise, use the command path itself. + */ + + if ((linkbytes = readlink(command, linkpath, sizeof(linkpath) - 1)) > 0) + { + /* + * Yes, this is a symlink to the actual program, nul-terminate and + * use it... + */ + + linkpath[linkbytes] = '\0'; + + if (linkpath[0] == '/') + snprintf(cfprocesspath, sizeof(cfprocesspath), "CFProcessPath=%s", + linkpath); + else + snprintf(cfprocesspath, sizeof(cfprocesspath), "CFProcessPath=%s/%s", + dirname((char *)command), linkpath); + } + else + snprintf(cfprocesspath, sizeof(cfprocesspath), "CFProcessPath=%s", command); + + envp[0] = cfprocesspath; + + /* + * Copy the rest of the environment except for any CFProcessPath that may + * already be there... + */ + + for (i = 1, j = 0; + environ[j] && i < (int)(sizeof(envp) / sizeof(envp[0]) - 1); + j ++) + if (strncmp(environ[j], "CFProcessPath=", 14)) + envp[i ++] = environ[j]; + + envp[i] = NULL; + + /* + * Use execve() to run the program... + */ + + return (execve(command, argv, envp)); + +#else + /* + * On other operating systems, just call execv() to use the same environment + * variables as the parent... + */ + + return (execv(command, argv)); +#endif /* __APPLE__ */ +} + + +/* + * 'cupsdPipeCommand()' - Read output from a command. + */ + +cups_file_t * /* O - CUPS file or NULL on error */ +cupsdPipeCommand(int *pid, /* O - Process ID or 0 on error */ + const char *command, /* I - Command to run */ + char **argv, /* I - Arguments to pass to command */ + int user) /* I - User to run as or 0 for current */ +{ + int fd, /* Temporary file descriptor */ + fds[2]; /* Pipe file descriptors */ + + + /* + * First create the pipe... + */ + + if (pipe(fds)) + { + *pid = 0; + return (NULL); + } + + /* + * Set the "close on exec" flag on each end of the pipe... + */ + + if (fcntl(fds[0], F_SETFD, fcntl(fds[0], F_GETFD) | FD_CLOEXEC)) + { + close(fds[0]); + close(fds[1]); + + *pid = 0; + + return (NULL); + } + + if (fcntl(fds[1], F_SETFD, fcntl(fds[1], F_GETFD) | FD_CLOEXEC)) + { + close(fds[0]); + close(fds[1]); + + *pid = 0; + + return (NULL); + } + + /* + * Then run the command... + */ + + if ((*pid = fork()) < 0) + { + /* + * Unable to fork! + */ + + *pid = 0; + close(fds[0]); + close(fds[1]); + + return (NULL); + } + else if (!*pid) + { + /* + * Child comes here... + */ + + if (!getuid() && user) + setuid(user); /* Run as restricted user */ + + if ((fd = open("/dev/null", O_RDONLY)) > 0) + { + dup2(fd, 0); /* pipe */ + close(fds[1]); + + cupsdExec(command, argv); + exit(errno); + } + + /* + * Parent comes here, open the input side of the pipe... + */ + + close(fds[1]); + + return (cupsFileOpenFd(fds[0], "r")); +} + + +/* + * 'cupsdSendIPPGroup()' - Send a group tag. + */ + +void +cupsdSendIPPGroup(ipp_tag_t group_tag) /* I - Group tag */ +{ + /* + * Send IPP group tag (1 byte)... + */ + + putchar(group_tag); +} + + +/* + * 'cupsdSendIPPHeader()' - Send the IPP response header. + */ + +void +cupsdSendIPPHeader( + ipp_status_t status_code, /* I - Status code */ + int request_id) /* I - Request ID */ +{ + /* + * Send IPP/1.1 response header: version number (2 bytes), status code + * (2 bytes), and request ID (4 bytes)... + * + * TODO: Add version number (IPP/2.x and IPP/1.0) support. + */ + + putchar(1); + putchar(1); + + putchar(status_code >> 8); + putchar(status_code); + + putchar(request_id >> 24); + putchar(request_id >> 16); + putchar(request_id >> 8); + putchar(request_id); +} + + +/* + * 'cupsdSendIPPInteger()' - Send an integer attribute. + */ + +void +cupsdSendIPPInteger( + ipp_tag_t value_tag, /* I - Value tag */ + const char *name, /* I - Attribute name */ + int value) /* I - Attribute value */ +{ + size_t len; /* Length of attribute name */ + + + /* + * Send IPP integer value: value tag (1 byte), name length (2 bytes), + * name string (without nul), value length (2 bytes), and value (4 bytes)... + */ + + putchar(value_tag); + + len = strlen(name); + putchar(len >> 8); + putchar(len); + + fputs(name, stdout); + + putchar(0); + putchar(4); + + putchar(value >> 24); + putchar(value >> 16); + putchar(value >> 8); + putchar(value); +} + + +/* + * 'cupsdSendIPPString()' - Send a string attribute. + */ + +void +cupsdSendIPPString( + ipp_tag_t value_tag, /* I - Value tag */ + const char *name, /* I - Attribute name */ + const char *value) /* I - Attribute value */ +{ + size_t len; /* Length of attribute name */ + + + /* + * Send IPP string value: value tag (1 byte), name length (2 bytes), + * name string (without nul), value length (2 bytes), and value string + * (without nul)... + */ + + putchar(value_tag); + + len = strlen(name); + putchar(len >> 8); + putchar(len); + + fputs(name, stdout); + + len = strlen(value); + putchar(len >> 8); + putchar(len); + + fputs(value, stdout); +} + + +/* + * 'cupsdSendIPPTrailer()' - Send the end-of-message tag. + */ + +void +cupsdSendIPPTrailer(void) +{ + putchar(IPP_TAG_END); + fflush(stdout); +} + + +/* + * End of "$Id$". + */ diff --git a/scheduler/util.h b/scheduler/util.h new file mode 100644 index 0000000000..6c2a05acea --- /dev/null +++ b/scheduler/util.h @@ -0,0 +1,71 @@ +/* + * "$Id$" + * + * Mini-daemon utility definitions for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2005 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#ifndef _CUPSD_UTIL_H_ +# define _CUPSD_UTIL_H_ + +/* + * Include necessary headers... + */ + +# include +# include +# include + + +/* + * C++ magic... + */ + +# ifdef __cplusplus +extern "C" { +# endif /* __cplusplus */ + + +/* + * Types... + */ + +typedef int (*cupsd_compare_func_t)(const void *, const void *); + + +/* + * Prototypes... + */ + +extern int cupsdCompareNames(const char *s, const char *t); +extern cups_array_t *cupsdCreateStringsArray(const char *s); +extern int cupsdExec(const char *command, char **argv); +extern cups_file_t *cupsdPipeCommand(int *pid, const char *command, + char **argv, int user); +extern void cupsdSendIPPGroup(ipp_tag_t group_tag); +extern void cupsdSendIPPHeader(ipp_status_t status_code, + int request_id); +extern void cupsdSendIPPInteger(ipp_tag_t value_tag, + const char *name, int value); +extern void cupsdSendIPPString(ipp_tag_t value_tag, + const char *name, const char *value); +extern void cupsdSendIPPTrailer(void); + + +# ifdef __cplusplus +} +# endif /* __cplusplus */ + +#endif /* !_CUPSD_UTIL_H_ */ + +/* + * End of "$Id$". + */ diff --git a/systemv/Dependencies b/systemv/Dependencies new file mode 100644 index 0000000000..35ba252338 --- /dev/null +++ b/systemv/Dependencies @@ -0,0 +1,92 @@ +cancel.o: cancel.c ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +cupsaccept.o: cupsaccept.c ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h +cupsaddsmb.o: cupsaddsmb.c ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/adminutil.h +cupsctl.o: cupsctl.c ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h ../cups/adminutil.h +cupstestdsc.o: cupstestdsc.c ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h +cupstestppd.o: cupstestppd.c ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h ../cups/dir.h \ + ../cups/raster.h +lp.o: lp.c ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +lpadmin.o: lpadmin.c ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +lpinfo.o: lpinfo.c ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +lpmove.o: lpmove.c ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h +lpoptions.o: lpoptions.c ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h +lppasswd.o: lppasswd.c ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h +lpstat.o: lpstat.c ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h diff --git a/systemv/Makefile b/systemv/Makefile new file mode 100644 index 0000000000..f1d0f23c8b --- /dev/null +++ b/systemv/Makefile @@ -0,0 +1,293 @@ +# +# "$Id$" +# +# System V commands makefile for CUPS. +# +# Copyright 2007-2012 by Apple Inc. +# Copyright 1997-2006 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +include ../Makedefs + +TARGETS = cancel cupsaccept cupsaddsmb cupsctl cupstestdsc cupstestppd \ + lp lpadmin lpinfo lpmove lpoptions lppasswd lpstat +OBJS = cancel.o cupsaccept.o cupsaddsmb.o cupsctl.o cupstestdsc.o \ + cupstestppd.o lp.o lpadmin.o lpinfo.o lpmove.o lpoptions.o \ + lppasswd.o lpstat.o + + +# +# Make all targets... +# + +all: $(TARGETS) + + +# +# Make library targets... +# + +libs: + + +# +# Make unit tests... +# + +unittests: + + +# +# Clean all object files... +# + +clean: + $(RM) $(OBJS) $(TARGETS) + $(RM) accept cupsdisable cupsenable cupsreject reject + + +# +# Update dependencies (without system header dependencies...) +# + +depend: + $(CC) -MM $(ALL_CFLAGS) $(OBJS:.o=.c) >Dependencies + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: + + +# +# Install programs... +# + +install-exec: + echo Installing System V admin printing commands in $(SBINDIR) + $(INSTALL_DIR) -m 755 $(SBINDIR) + $(INSTALL_BIN) cupsaccept $(SBINDIR) + $(INSTALL_BIN) cupsaddsmb $(SBINDIR) + $(INSTALL_BIN) cupsctl $(SBINDIR) + $(INSTALL_BIN) lpadmin $(SBINDIR) + $(INSTALL_BIN) lpinfo $(SBINDIR) + $(INSTALL_BIN) lpmove $(SBINDIR) + $(RM) $(SBINDIR)/accept + $(LN) cupsaccept $(SBINDIR)/accept + $(RM) $(SBINDIR)/cupsdisable + $(LN) cupsaccept $(SBINDIR)/cupsdisable + $(RM) $(SBINDIR)/cupsenable + $(LN) cupsaccept $(SBINDIR)/cupsenable + $(RM) $(SBINDIR)/cupsreject + $(LN) cupsaccept $(SBINDIR)/cupsreject + $(RM) $(SBINDIR)/reject + $(LN) cupsaccept $(SBINDIR)/reject + echo Installing System V user printing commands in $(BINDIR) + $(INSTALL_DIR) -m 755 $(BINDIR) + $(INSTALL_BIN) cancel $(BINDIR) + $(INSTALL_BIN) cupstestdsc $(BINDIR) + $(INSTALL_BIN) cupstestppd $(BINDIR) + $(INSTALL_BIN) lp $(BINDIR) + $(INSTALL_BIN) lpoptions $(BINDIR) + $(INSTALL_BIN) lpstat $(BINDIR) + $(INSTALL_BIN) lppasswd $(BINDIR) + if test "x$(SYMROOT)" != "x"; then \ + $(INSTALL_DIR) $(SYMROOT); \ + for file in $(TARGETS); do \ + cp $$file $(SYMROOT); \ + done \ + fi + + +# +# Install headers... +# + +install-headers: + + +# +# Install libraries... +# + +install-libs: + + +# +# Uninstall all targets... +# + +uninstall: + $(RM) $(BINDIR)/cancel + $(RM) $(BINDIR)/cupstestdsc + $(RM) $(BINDIR)/cupstestppd + $(RM) $(BINDIR)/lp + $(RM) $(BINDIR)/lpoptions + $(RM) $(BINDIR)/lppasswd + $(RM) $(BINDIR)/lpstat + -$(RMDIR) $(BINDIR) + $(RM) $(SBINDIR)/accept + $(RM) $(SBINDIR)/cupsaccept + $(RM) $(SBINDIR)/cupsaddsmb + $(RM) $(SBINDIR)/cupsaccept + $(RM) $(SBINDIR)/cupsdisable + $(RM) $(SBINDIR)/cupsenable + $(RM) $(SBINDIR)/cupsreject + $(RM) $(SBINDIR)/lpadmin + $(RM) $(SBINDIR)/lpinfo + $(RM) $(SBINDIR)/lpmove + $(RM) $(SBINDIR)/reject + -$(RMDIR) $(SBINDIR) + + +# +# cancel +# + +cancel: cancel.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o cancel cancel.o $(LIBS) + + +# +# cupsaccept +# + +cupsaccept: cupsaccept.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o cupsaccept cupsaccept.o $(LIBS) + for file in accept cupsenable cupsdisable cupsreject reject; do \ + $(RM) $$file; \ + $(LN) cupsaccept $$file; \ + done + + +# +# cupsaddsmb +# + +cupsaddsmb: cupsaddsmb.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o cupsaddsmb cupsaddsmb.o $(LIBS) + + +# +# cupsctl +# + +cupsctl: cupsctl.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o cupsctl cupsctl.o $(LIBS) + + +# +# cupstestdsc +# + +cupstestdsc: cupstestdsc.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ cupstestdsc.o $(LIBS) + + +# +# cupstestppd +# + +cupstestppd: cupstestppd.o ../cups/$(LIBCUPS) ../filter/$(LIBCUPSIMAGE) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ cupstestppd.o $(LINKCUPSIMAGE) $(IMGLIBS) $(LIBS) + +cupstestppd-static: cupstestppd.o ../cups/$(LIBCUPSSTATIC) ../filter/libcupsimage.a + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ cupstestppd.o ../filter/libcupsimage.a \ + ../cups/$(LIBCUPSSTATIC) $(IMGLIBS) $(LIBGSSAPI) $(LIBS) $(LIBZ) + + +# +# lp +# + +lp: lp.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o lp lp.o $(LIBS) + + +# +# lpadmin +# + +lpadmin: lpadmin.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o lpadmin lpadmin.o $(LIBZ) $(LIBS) + + +# +# lpinfo +# + +lpinfo: lpinfo.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o lpinfo lpinfo.o $(LIBS) + + +# +# lpmove +# + +lpmove: lpmove.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o lpmove lpmove.o $(LIBS) + + +# +# lpoptions +# + +lpoptions: lpoptions.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o lpoptions lpoptions.o $(LIBZ) $(LIBS) + + +# +# lppasswd +# + +lppasswd: lppasswd.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o lppasswd lppasswd.o $(LIBZ) $(LIBS) + + +# +# lpstat +# + +lpstat: lpstat.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o lpstat lpstat.o $(LIBS) + + +# +# Dependencies... +# + +include Dependencies + + +# +# End of "$Id$". +# diff --git a/systemv/cancel.c b/systemv/cancel.c new file mode 100644 index 0000000000..9307e19433 --- /dev/null +++ b/systemv/cancel.c @@ -0,0 +1,376 @@ +/* + * "$Id$" + * + * "cancel" command for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Parse options and cancel jobs. + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * 'main()' - Parse options and cancel jobs. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + http_t *http; /* HTTP connection to server */ + int i; /* Looping var */ + int job_id; /* Job ID */ + int num_dests; /* Number of destinations */ + cups_dest_t *dests; /* Destinations */ + char *dest, /* Destination printer */ + *job, /* Job ID pointer */ + *user; /* Cancel jobs for a user */ + int purge; /* Purge or cancel jobs? */ + char uri[1024]; /* Printer or job URI */ + ipp_t *request; /* IPP request */ + ipp_t *response; /* IPP response */ + ipp_op_t op; /* Operation */ + + + _cupsSetLocale(argv); + + /* + * Setup to cancel individual print jobs... + */ + + op = IPP_CANCEL_JOB; + purge = 0; + dest = NULL; + user = NULL; + http = NULL; + num_dests = 0; + dests = NULL; + + + /* + * Process command-line arguments... + */ + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-' && argv[i][1]) + switch (argv[i][1]) + { + case 'E' : /* Encrypt */ +#ifdef HAVE_SSL + cupsSetEncryption(HTTP_ENCRYPT_REQUIRED); + + if (http) + httpEncryption(http, HTTP_ENCRYPT_REQUIRED); +#else + _cupsLangPrintf(stderr, + _("%s: Sorry, no encryption support."), argv[0]); +#endif /* HAVE_SSL */ + break; + + case 'U' : /* Username */ + if (argv[i][2] != '\0') + cupsSetUser(argv[i] + 2); + else + { + i ++; + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected username after " + "\"-U\" option."), argv[0]); + return (1); + } + + cupsSetUser(argv[i]); + } + break; + + case 'a' : /* Cancel all jobs */ + purge = 1; + op = IPP_PURGE_JOBS; + break; + + case 'h' : /* Connect to host */ + if (http != NULL) + { + httpClose(http); + http = NULL; + } + + if (argv[i][2] != '\0') + cupsSetServer(argv[i] + 2); + else + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected hostname after " + "\"-h\" option."), argv[0]); + return (1); + } + else + cupsSetServer(argv[i]); + } + break; + + case 'u' : /* Username */ + op = IPP_PURGE_JOBS; + + if (argv[i][2] != '\0') + user = argv[i] + 2; + else + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected username after " + "\"-u\" option."), argv[0]); + return (1); + } + else + user = argv[i]; + } + break; + + default : + _cupsLangPrintf(stderr, + _("%s: Error - unknown option \"%c\"."), + argv[0], argv[i][1]); + return (1); + } + else + { + /* + * Cancel a job or printer... + */ + + if (num_dests == 0) + num_dests = cupsGetDests(&dests); + + if (!strcmp(argv[i], "-")) + { + /* + * Delete the current job... + */ + + dest = ""; + job_id = 0; + } + else if (cupsGetDest(argv[i], NULL, num_dests, dests) != NULL) + { + /* + * Delete the current job on the named destination... + */ + + dest = argv[i]; + job_id = 0; + } + else if ((job = strrchr(argv[i], '-')) != NULL && isdigit(job[1] & 255)) + { + /* + * Delete the specified job ID. + */ + + dest = NULL; + op = IPP_CANCEL_JOB; + job_id = atoi(job + 1); + } + else if (isdigit(argv[i][0] & 255)) + { + /* + * Delete the specified job ID. + */ + + dest = NULL; + op = IPP_CANCEL_JOB; + job_id = atoi(argv[i]); + } + else + { + /* + * Bad printer name! + */ + + _cupsLangPrintf(stderr, + _("%s: Error - unknown destination \"%s\"."), + argv[0], argv[i]); + return (1); + } + + /* + * For Solaris LP compatibility, ignore a destination name after + * cancelling a specific job ID... + */ + + if (job_id && (i + 1) < argc && + cupsGetDest(argv[i + 1], NULL, num_dests, dests) != NULL) + i ++; + + /* + * Open a connection to the server... + */ + + if (http == NULL) + if ((http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption())) == NULL) + { + _cupsLangPrintf(stderr, + _("%s: Unable to connect to server."), argv[0]); + return (1); + } + + /* + * Build an IPP request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + job-id *or* job-uri + * [requesting-user-name] + */ + + request = ippNewRequest(op); + + if (dest) + { + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", dest); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, uri); + ippAddInteger(request, IPP_TAG_OPERATION, IPP_TAG_INTEGER, "job-id", + job_id); + } + else + { + sprintf(uri, "ipp://localhost/jobs/%d", job_id); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-uri", NULL, + uri); + } + + if (user) + { + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, user); + ippAddBoolean(request, IPP_TAG_OPERATION, "my-jobs", 1); + } + else + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, cupsUser()); + + if (op == IPP_PURGE_JOBS) + ippAddBoolean(request, IPP_TAG_OPERATION, "purge-jobs", purge); + + /* + * Do the request and get back a response... + */ + + if (op == IPP_PURGE_JOBS && (!user || _cups_strcasecmp(user, cupsUser()))) + response = cupsDoRequest(http, request, "/admin/"); + else + response = cupsDoRequest(http, request, "/jobs/"); + + if (response == NULL || + response->request.status.status_code > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, _("%s: %s failed: %s"), argv[0], + op == IPP_PURGE_JOBS ? "purge-jobs" : "cancel-job", + cupsLastErrorString()); + + if (response) + ippDelete(response); + + return (1); + } + + ippDelete(response); + } + + if (num_dests == 0 && op == IPP_PURGE_JOBS) + { + /* + * Open a connection to the server... + */ + + if (http == NULL) + if ((http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption())) == NULL) + { + _cupsLangPrintf(stderr, _("%s: Unable to contact server."), argv[0]); + return (1); + } + + /* + * Build an IPP request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + job-id *or* job-uri + * [requesting-user-name] + */ + + request = ippNewRequest(op); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, "ipp://localhost/printers/"); + + if (user) + { + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, user); + ippAddBoolean(request, IPP_TAG_OPERATION, "my-jobs", 1); + } + else + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, cupsUser()); + + ippAddBoolean(request, IPP_TAG_OPERATION, "purge-jobs", purge); + + /* + * Do the request and get back a response... + */ + + response = cupsDoRequest(http, request, "/admin/"); + + if (response == NULL || + response->request.status.status_code > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, _("%s: %s failed: %s"), argv[0], + op == IPP_PURGE_JOBS ? "purge-jobs" : "cancel-job", + cupsLastErrorString()); + + if (response) + ippDelete(response); + + return (1); + } + + ippDelete(response); + } + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/systemv/cupsaccept.c b/systemv/cupsaccept.c new file mode 100644 index 0000000000..784138ee2f --- /dev/null +++ b/systemv/cupsaccept.c @@ -0,0 +1,239 @@ +/* + * "$Id$" + * + * "cupsaccept", "cupsdisable", "cupsenable", and "cupsreject" commands for + * CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Parse options and accept/reject jobs or disable/enable printers. + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * 'main()' - Parse options and accept/reject jobs or disable/enable printers. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + char *command, /* Command to do */ + uri[1024], /* Printer URI */ + *reason; /* Reason for reject/disable */ + ipp_t *request; /* IPP request */ + ipp_op_t op; /* Operation */ + int cancel; /* Cancel jobs? */ + + + _cupsSetLocale(argv); + + /* + * See what operation we're supposed to do... + */ + + if ((command = strrchr(argv[0], '/')) != NULL) + command ++; + else + command = argv[0]; + + cancel = 0; + + if (!strcmp(command, "cupsaccept") || !strcmp(command, "accept")) + op = CUPS_ACCEPT_JOBS; + else if (!strcmp(command, "cupsreject") || !strcmp(command, "reject")) + op = CUPS_REJECT_JOBS; + else if (!strcmp(command, "cupsdisable") || !strcmp(command, "disable")) + op = IPP_PAUSE_PRINTER; + else if (!strcmp(command, "cupsenable") || !strcmp(command, "enable")) + op = IPP_RESUME_PRINTER; + else + { + _cupsLangPrintf(stderr, _("%s: Don't know what to do."), command); + return (1); + } + + reason = NULL; + + /* + * Process command-line arguments... + */ + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-') + { + switch (argv[i][1]) + { + case 'E' : /* Encrypt */ +#ifdef HAVE_SSL + cupsSetEncryption(HTTP_ENCRYPT_REQUIRED); +#else + _cupsLangPrintf(stderr, + _("%s: Sorry, no encryption support."), command); +#endif /* HAVE_SSL */ + break; + + case 'U' : /* Username */ + if (argv[i][2] != '\0') + cupsSetUser(argv[i] + 2); + else + { + i ++; + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected username after " + "\"-U\" option."), command); + return (1); + } + + cupsSetUser(argv[i]); + } + break; + + case 'c' : /* Cancel jobs */ + cancel = 1; + break; + + case 'h' : /* Connect to host */ + if (argv[i][2] != '\0') + cupsSetServer(argv[i] + 2); + else + { + i ++; + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected hostname after " + "\"-h\" option."), command); + return (1); + } + + cupsSetServer(argv[i]); + } + break; + + case 'r' : /* Reason for cancellation */ + if (argv[i][2] != '\0') + reason = argv[i] + 2; + else + { + i ++; + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected reason text after " + "\"-r\" option."), command); + return (1); + } + + reason = argv[i]; + } + break; + + case '-' : + if (!strcmp(argv[i], "--hold")) + op = IPP_HOLD_NEW_JOBS; + else if (!strcmp(argv[i], "--release")) + op = IPP_RELEASE_HELD_NEW_JOBS; + else + { + _cupsLangPrintf(stderr, _("%s: Error - unknown option \"%s\"."), + command, argv[i]); + return (1); + } + break; + + default : + _cupsLangPrintf(stderr, _("%s: Error - unknown option \"%c\"."), + command, argv[i][1]); + return (1); + } + } + else + { + /* + * Accept/disable/enable/reject a destination... + */ + + request = ippNewRequest(op); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", argv[i]); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, uri); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, cupsUser()); + + if (reason != NULL) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_TEXT, + "printer-state-message", NULL, reason); + + /* + * Do the request and get back a response... + */ + + ippDelete(cupsDoRequest(CUPS_HTTP_DEFAULT, request, "/admin/")); + + if (cupsLastError() > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, + _("%s: Operation failed: %s"), + command, ippErrorString(cupsLastError())); + return (1); + } + + /* + * Cancel all jobs if requested... + */ + + if (cancel) + { + /* + * Build an IPP_PURGE_JOBS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + */ + + request = ippNewRequest(IPP_PURGE_JOBS); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, uri); + + ippDelete(cupsDoRequest(CUPS_HTTP_DEFAULT, request, "/admin/")); + + if (cupsLastError() > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, "%s: %s", command, cupsLastErrorString()); + return (1); + } + } + } + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/systemv/cupsaddsmb.c b/systemv/cupsaddsmb.c new file mode 100644 index 0000000000..0f0431d39b --- /dev/null +++ b/systemv/cupsaddsmb.c @@ -0,0 +1,303 @@ +/* + * "$Id$" + * + * "cupsaddsmb" command for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 2001-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Export printers on the command-line. + * export_dest() - Export a destination to SAMBA. + * usage() - Show program usage and exit... + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include +#include + + +/* + * Local globals... + */ + +int Verbosity = 0; +const char *SAMBAUser, + *SAMBAPassword, + *SAMBAServer; + + +/* + * Local functions... + */ + +int export_dest(http_t *http, const char *dest); +void usage(void) __attribute__((noreturn)); + + +/* + * 'main()' - Export printers on the command-line. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i, j; /* Looping vars */ + int status; /* Status from export_dest() */ + int export_all; /* Export all printers? */ + http_t *http; /* Connection to server */ + int num_dests; /* Number of printers */ + cups_dest_t *dests; /* Printers */ + + + _cupsSetLocale(argv); + + /* + * Parse command-line arguments... + */ + + export_all = 0; + http = NULL; + SAMBAUser = cupsUser(); + SAMBAPassword = NULL; + SAMBAServer = NULL; + + for (i = 1; i < argc; i ++) + if (!strcmp(argv[i], "-E")) + { +#ifdef HAVE_SSL + cupsSetEncryption(HTTP_ENCRYPT_REQUIRED); +#else + _cupsLangPrintf(stderr, + _("%s: Sorry, no encryption support."), + argv[0]); +#endif /* HAVE_SSL */ + } + else if (!strcmp(argv[i], "-H")) + { + i ++; + if (i >= argc) + usage(); + + SAMBAServer = argv[i]; + } + else if (!strcmp(argv[i], "-U")) + { + char *sep; /* Separator for password */ + + + i ++; + if (i >= argc) + usage(); + + SAMBAUser = argv[i]; + + if ((sep = strchr(argv[i], '%')) != NULL) + { + /* + * Nul-terminate the username at the first % and point the + * password at the rest... + */ + + *sep++ = '\0'; + + SAMBAPassword = sep; + } + } + else if (!strcmp(argv[i], "-a")) + export_all = 1; + else if (!strcmp(argv[i], "-h")) + { + i ++; + if (i >= argc) + usage(); + + cupsSetServer(argv[i]); + } + else if (!strcmp(argv[i], "-v")) + Verbosity = 1; + else if (argv[i][0] != '-') + { + if (!http) + { + /* + * Connect to the server... + */ + + if ((http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption())) == NULL) + { + _cupsLangPrintf(stderr, _("%s: Unable to connect to server."), + argv[0]); + exit(1); + } + } + + if (SAMBAServer == NULL) + { + SAMBAServer = cupsServer(); + + if (SAMBAServer[0] == '/') /* Use localhost instead of domain socket */ + SAMBAServer = "localhost"; + } + + if ((status = export_dest(http, argv[i])) != 0) + return (status); + } + else + usage(); + + /* + * Connect to the server... + */ + + if ((http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption())) == NULL) + { + _cupsLangPrintf(stderr, _("%s: Unable to connect to server."), argv[0]); + exit(1); + } + + /* + * See if the user specified "-a"... + */ + + if (export_all) + { + /* + * Export all printers... + */ + + if (SAMBAServer == NULL) + { + SAMBAServer = cupsServer(); + + if (SAMBAServer[0] == '/') /* Use localhost instead of domain socket */ + SAMBAServer = "localhost"; + } + + num_dests = cupsGetDests2(http, &dests); + + for (j = 0, status = 0; j < num_dests; j ++) + if (!dests[j].instance) + { + if ((status = export_dest(http, dests[j].name)) != 0) + break; + } + + cupsFreeDests(num_dests, dests); + + if (status) + return (status); + } + + return (0); +} + + +/* + * 'export_dest()' - Export a destination to SAMBA. + */ + +int /* O - 0 on success, non-zero on error */ +export_dest(http_t *http, /* I - Connection to server */ + const char *dest) /* I - Destination to export */ +{ + int status; /* Status of export */ + char ppdfile[1024], /* PPD file for printer drivers */ + prompt[1024]; /* Password prompt */ + int tries; /* Number of tries */ + + + /* + * Get the Windows PPD file for the printer... + */ + + if (!cupsAdminCreateWindowsPPD(http, dest, ppdfile, sizeof(ppdfile))) + { + _cupsLangPrintf(stderr, + _("cupsaddsmb: No PPD file for printer \"%s\" - %s"), + dest, cupsLastErrorString()); + return (1); + } + + /* + * Try to export it... + */ + + for (status = 0, tries = 0; !status && tries < 3; tries ++) + { + /* + * Get the password, as needed... + */ + + if (!SAMBAPassword) + { + snprintf(prompt, sizeof(prompt), + _cupsLangString(cupsLangDefault(), + _("Password for %s required to access %s via " + "SAMBA: ")), + SAMBAUser, SAMBAServer); + + if ((SAMBAPassword = cupsGetPassword(prompt)) == NULL) + break; + } + + status = cupsAdminExportSamba(dest, ppdfile, SAMBAServer, + SAMBAUser, SAMBAPassword, + Verbosity ? stderr : NULL); + + if (!status && cupsLastError() == IPP_NOT_FOUND) + break; + } + + unlink(ppdfile); + + return (!status); +} + + +/* + * 'usage()' - Show program usage and exit... + */ + +void +usage(void) +{ + _cupsLangPuts(stdout, _("Usage: cupsaddsmb [options] printer1 ... printerN")); + _cupsLangPuts(stdout, _(" cupsaddsmb [options] -a")); + _cupsLangPuts(stdout, ""); + _cupsLangPuts(stdout, _("Options:")); + _cupsLangPuts(stdout, _(" -E Encrypt the connection to " + "the server.")); + _cupsLangPuts(stdout, _(" -H samba-server Use the named SAMBA " + "server.")); + _cupsLangPuts(stdout, _(" -U samba-user Authenticate using the " + "named SAMBA user.")); + _cupsLangPuts(stdout, _(" -a Export all printers.")); + _cupsLangPuts(stdout, _(" -h cups-server Use the named CUPS " + "server.")); + _cupsLangPuts(stdout, _(" -v Be verbose (show " + "commands).")); + + exit(1); +} + + +/* + * End of "$Id$". + */ diff --git a/systemv/cupsctl.c b/systemv/cupsctl.c new file mode 100644 index 0000000000..e65a4f77b0 --- /dev/null +++ b/systemv/cupsctl.c @@ -0,0 +1,227 @@ +/* + * "$Id$" + * + * Scheduler control program for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 2006-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Get/set server settings. + * usage() - Show program usage. + */ + +/* + * Include necessary headers... + */ + +#include +#include + + +/* + * Local functions... + */ + +static void usage(const char *opt) __attribute__((noreturn)); + + +/* + * 'main()' - Get/set server settings. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int i, /* Looping var */ + num_settings; /* Number of settings */ + cups_option_t *settings; /* Settings */ + const char *opt; /* Current option character */ + http_t *http; /* Connection to server */ + + + /* + * Process the command-line... + */ + + _cupsSetLocale(argv); + + num_settings = 0; + settings = NULL; + + for (i = 1; i < argc; i ++) + { + if (argv[i][0] == '-') + { + if (argv[i][1] == '-') + { + if (!strcmp(argv[i], "--debug-logging")) + num_settings = cupsAddOption(CUPS_SERVER_DEBUG_LOGGING, "1", + num_settings, &settings); + else if (!strcmp(argv[i], "--no-debug-logging")) + num_settings = cupsAddOption(CUPS_SERVER_DEBUG_LOGGING, "0", + num_settings, &settings); + else if (!strcmp(argv[i], "--remote-admin")) + num_settings = cupsAddOption(CUPS_SERVER_REMOTE_ADMIN, "1", + num_settings, &settings); + else if (!strcmp(argv[i], "--no-remote-admin")) + num_settings = cupsAddOption(CUPS_SERVER_REMOTE_ADMIN, "0", + num_settings, &settings); + else if (!strcmp(argv[i], "--remote-any")) + num_settings = cupsAddOption(CUPS_SERVER_REMOTE_ANY, "1", + num_settings, &settings); + else if (!strcmp(argv[i], "--no-remote-any")) + num_settings = cupsAddOption(CUPS_SERVER_REMOTE_ANY, "0", + num_settings, &settings); + else if (!strcmp(argv[i], "--share-printers")) + num_settings = cupsAddOption(CUPS_SERVER_SHARE_PRINTERS, "1", + num_settings, &settings); + else if (!strcmp(argv[i], "--no-share-printers")) + num_settings = cupsAddOption(CUPS_SERVER_SHARE_PRINTERS, "0", + num_settings, &settings); + else if (!strcmp(argv[i], "--user-cancel-any")) + num_settings = cupsAddOption(CUPS_SERVER_USER_CANCEL_ANY, "1", + num_settings, &settings); + else if (!strcmp(argv[i], "--no-user-cancel-any")) + num_settings = cupsAddOption(CUPS_SERVER_USER_CANCEL_ANY, "0", + num_settings, &settings); + else + usage(argv[i]); + } + else + { + for (opt = argv[i] + 1; *opt; opt ++) + switch (*opt) + { + case 'E' : + cupsSetEncryption(HTTP_ENCRYPT_REQUIRED); + break; + + case 'U' : + i ++; + if (i >= argc) + usage(NULL); + + cupsSetUser(argv[i]); + break; + + case 'h' : + i ++; + if (i >= argc) + usage(NULL); + + cupsSetServer(argv[i]); + break; + + default : + usage(opt); + break; + } + } + } + else if (strchr(argv[i], '=')) + num_settings = cupsParseOptions(argv[i], num_settings, &settings); + else + usage(argv[i]); + } + + if (cupsGetOption("Listen", num_settings, settings) || + cupsGetOption("Port", num_settings, settings)) + { + _cupsLangPuts(stderr, _("cupsctl: Cannot set Listen or Port directly.")); + return (1); + } + + /* + * Connect to the server using the defaults... + */ + + if ((http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption())) == NULL) + { + _cupsLangPrintf(stderr, _("cupsctl: Unable to connect to server: %s"), + strerror(errno)); + return (1); + } + + /* + * Set the current configuration if we have anything on the command-line... + */ + + if (num_settings > 0) + { + if (!cupsAdminSetServerSettings(http, num_settings, settings)) + { + _cupsLangPrintf(stderr, "cupsctl: %s", cupsLastErrorString()); + return (1); + } + } + else if (!cupsAdminGetServerSettings(http, &num_settings, &settings)) + { + _cupsLangPrintf(stderr, "cupsctl: %s", cupsLastErrorString()); + return (1); + } + else + { + for (i = 0; i < num_settings; i ++) + _cupsLangPrintf(stdout, "%s=%s", settings[i].name, settings[i].value); + } + + cupsFreeOptions(num_settings, settings); + return (0); +} + + +/* + * 'usage()' - Show program usage. + */ + +static void +usage(const char *opt) /* I - Option character/string */ +{ + if (opt) + { + if (*opt == '-') + _cupsLangPrintf(stderr, _("cupsctl: Unknown option \"%s\""), opt); + else + _cupsLangPrintf(stderr, _("cupsctl: Unknown option \"-%c\""), *opt); + } + + _cupsLangPuts(stdout, _("Usage: cupsctl [options] [param=value ... " + "paramN=valueN]")); + _cupsLangPuts(stdout, ""); + _cupsLangPuts(stdout, _("Options:")); + _cupsLangPuts(stdout, ""); + _cupsLangPuts(stdout, _(" -E Enable encryption.")); + _cupsLangPuts(stdout, _(" -U username Specify username.")); + _cupsLangPuts(stdout, _(" -h server[:port] Specify server " + "address.")); + _cupsLangPuts(stdout, ""); + _cupsLangPuts(stdout, _(" --[no-]debug-logging Turn debug logging " + "on/off.")); + _cupsLangPuts(stdout, _(" --[no-]remote-admin Turn remote " + "administration on/off.")); + _cupsLangPuts(stdout, _(" --[no-]remote-any Allow/prevent access " + "from the Internet.")); + _cupsLangPuts(stdout, _(" --[no-]share-printers Turn printer sharing " + "on/off.")); + _cupsLangPuts(stdout, _(" --[no-]user-cancel-any Allow/prevent users to " + "cancel any job.")); + + exit(1); +} + + +/* + * End of "$Id$". + */ diff --git a/systemv/cupstestdsc.c b/systemv/cupstestdsc.c new file mode 100644 index 0000000000..de024c7ed6 --- /dev/null +++ b/systemv/cupstestdsc.c @@ -0,0 +1,442 @@ +/* + * "$Id$" + * + * DSC test program for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 2006 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * PostScript is a trademark of Adobe Systems, Inc. + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Main entry for test program. + * check() - Check a file for conformance. + * usage() - Show program usage. + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * Local functions... + */ + +static int check_file(const char *filename); +static void usage(void) __attribute__((noreturn)); + + +/* + * 'main()' - Main entry for test program. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + int status; /* Status of tests */ + int num_files; /* Number of files tested */ + + + _cupsSetLocale(argv); + + /* + * Collect command-line arguments... + */ + + for (i = 1, num_files = 0, status = 0; i < argc; i ++) + if (argv[i][0] == '-') + { + if (argv[i][1]) + { + /* + * Currently the only supported option is "-h" (help)... + */ + + usage(); + } + else + { + num_files ++; + status += check_file("(stdin)"); + } + } + else + { + num_files ++; + status += check_file(argv[i]); + } + + if (!num_files) + usage(); + + return (status); +} + + +/* + * 'check()' - Check a file for conformance. + */ + +static int /* O - 0 on success, 1 on failure */ +check_file(const char *filename) /* I - File to read from */ +{ + int i; /* Looping var */ + cups_file_t *fp; /* File */ + char line[1024]; /* Line from file */ + int ch; /* Current character */ + size_t bytes; /* Length of line */ + int status; /* Status of test */ + int linenum; /* Line number */ + int binary; /* File contains binary data? */ + float version; /* DSC version */ + int lbrt[4]; /* Bounding box */ + char page_label[256]; /* Page label string */ + int page_number; /* Page number */ + int last_page_number; /* Last page number seen */ + int level; /* Embedded document level */ + int saw_bounding_box, /* %%BoundingBox seen? */ + saw_pages, /* %%Pages seen? */ + saw_end_comments, /* %%EndComments seen? */ + saw_begin_prolog, /* %%BeginProlog seen? */ + saw_end_prolog, /* %%EndProlog seen? */ + saw_begin_setup, /* %%BeginSetup seen? */ + saw_end_setup, /* %%EndSetup seen? */ + saw_page, /* %%Page seen? */ + saw_trailer, /* %%Trailer seen? */ + saw_long_line; /* Saw long lines? */ + + + /* + * Open the file... + */ + + if (!strcmp(filename, "(stdin)")) + fp = cupsFileStdin(); + else + fp = cupsFileOpen(filename, "r"); + + if (!fp) + { + perror(filename); + return (1); + } + + /* + * Scan the file... + */ + + binary = 0; + last_page_number = 0; + level = 0; + linenum = 0; + saw_begin_prolog = 0; + saw_begin_setup = 0; + saw_bounding_box = 0; + saw_end_comments = 0; + saw_end_prolog = 0; + saw_end_setup = 0; + saw_long_line = 0; + saw_page = 0; + saw_pages = 0; + saw_trailer = 0; + status = 0; + version = 0.0f; + + /* TODO: Fixme */ + printf("%s: ", filename); + fflush(stdout); + + while ((bytes = cupsFileGetLine(fp, line, sizeof(line))) > 0) + { + linenum ++; + + if (bytes > 255) + { + if (!saw_long_line) + { + if (!status) + _cupsLangPuts(stdout, _("FAIL")); + + status ++; + _cupsLangPrintf(stdout, + _(" Line %d is longer than 255 characters (%d).\n" + " REF: Page 25, Line Length"), + linenum, (int)bytes); + } + + saw_long_line ++; + } + + if (linenum == 1) + { + if (strncmp(line, "%!PS-Adobe-", 11)) + { + if (!status) + _cupsLangPuts(stdout, _("FAIL")); + + _cupsLangPuts(stdout, + _(" Missing %!PS-Adobe-3.0 on first line.\n" + " REF: Page 17, 3.1 Conforming Documents")); + cupsFileClose(fp); + return (1); + } + else + version = atof(line + 11); + } + else if (level > 0) + { + if (!strncmp(line, "%%BeginDocument:", 16)) + level ++; + else if (!strncmp(line, "%%EndDocument", 13)) + level --; + } + else if (saw_trailer) + { + if (!strncmp(line, "%%Pages:", 8)) + { + if (atoi(line + 8) <= 0) + { + if (!status) + _cupsLangPuts(stdout, _("FAIL")); + + status ++; + _cupsLangPrintf(stdout, + _(" Bad %%%%Pages: on line %d.\n" + " REF: Page 43, %%%%Pages:"), + linenum); + } + else + saw_pages = 1; + } + else if (!strncmp(line, "%%BoundingBox:", 14)) + { + if (sscanf(line + 14, "%d%d%d%d", lbrt + 0, lbrt + 1, lbrt + 2, + lbrt + 3) != 4) + { + if (!status) + _cupsLangPuts(stdout, _("FAIL")); + + status ++; + _cupsLangPrintf(stdout, _(" Bad %%%%BoundingBox: on line %d.\n" + " REF: Page 39, %%%%BoundingBox:"), + linenum); + } + else + saw_bounding_box = 1; + } + } + else if (!saw_end_comments) + { + if (!strncmp(line, "%%EndComments", 13)) + saw_end_comments = 1; + else if (line[0] != '%') + saw_end_comments = -1; + else if (!strncmp(line, "%%Pages:", 8)) + { + if (strstr(line + 8, "(atend)")) + saw_pages = -1; + else if (atoi(line + 8) <= 0) + { + if (!status) + _cupsLangPuts(stdout, _("FAIL")); + + status ++; + _cupsLangPrintf(stdout, _(" Bad %%%%Pages: on line %d.\n" + " REF: Page 43, %%%%Pages:"), + linenum); + } + else + saw_pages = 1; + } + else if (!strncmp(line, "%%BoundingBox:", 14)) + { + if (strstr(line, "(atend)")) + saw_bounding_box = -1; + else if (sscanf(line + 14, "%d%d%d%d", lbrt + 0, lbrt + 1, lbrt + 2, + lbrt + 3) != 4) + { + if (!status) + _cupsLangPuts(stdout, _("FAIL")); + + status ++; + _cupsLangPrintf(stdout, _(" Bad %%%%BoundingBox: on line %d.\n" + " REF: Page 39, %%%%BoundingBox:"), + linenum); + } + else + saw_bounding_box = 1; + } + } + else if (saw_begin_prolog && !saw_end_prolog) + { + if (!strncmp(line, "%%EndProlog", 11)) + saw_end_prolog = 1; + } + else if (saw_begin_setup && !saw_end_setup) + { + if (!strncmp(line, "%%EndSetup", 10)) + saw_end_setup = 1; + } + else if (saw_end_comments) + { + if (!strncmp(line, "%%Page:", 7)) + { + if (sscanf(line + 7, "%255s%d", page_label, &page_number) != 2 || + page_number != (last_page_number + 1) || page_number < 1) + { + if (!status) + _cupsLangPuts(stdout, _("FAIL")); + + status ++; + _cupsLangPrintf(stdout, _(" Bad %%%%Page: on line %d.\n" + " REF: Page 53, %%%%Page:"), + linenum); + } + else + { + last_page_number = page_number; + saw_page = 1; + } + } + else if (!strncmp(line, "%%BeginProlog", 13)) + saw_begin_prolog = 1; + else if (!strncmp(line, "%%BeginSetup", 12)) + saw_begin_setup = 1; + else if (!strncmp(line, "%%BeginDocument:", 16)) + level ++; + else if (!strncmp(line, "%%EndDocument", 13)) + level --; + else if (!strncmp(line, "%%Trailer", 9)) + saw_trailer = 1; + } + + for (i = 0; !binary && i < bytes; i ++) + { + ch = line[i]; + + if ((ch < ' ' || (ch & 0x80)) && ch != '\n' && ch != '\r' && ch != '\t') + binary = 1; + } + } + + if (saw_bounding_box <= 0) + { + if (!status) + _cupsLangPuts(stdout, _("FAIL")); + + status ++; + _cupsLangPuts(stdout, _(" Missing or bad %%BoundingBox: comment.\n" + " REF: Page 39, %%BoundingBox:")); + } + + if (saw_pages <= 0) + { + if (!status) + _cupsLangPuts(stdout, _("FAIL")); + + status ++; + _cupsLangPuts(stdout, _(" Missing or bad %%Pages: comment.\n" + " REF: Page 43, %%Pages:")); + } + + if (!saw_end_comments) + { + if (!status) + _cupsLangPuts(stdout, _("FAIL")); + + status ++; + _cupsLangPuts(stdout, _(" Missing %%EndComments comment." + " REF: Page 41, %%EndComments")); + } + + if (!saw_page) + { + if (!status) + _cupsLangPuts(stdout, _("FAIL")); + + status ++; + _cupsLangPuts(stdout, _(" Missing or bad %%Page: comments.\n" + " REF: Page 53, %%Page:")); + } + + if (level < 0) + { + if (!status) + _cupsLangPuts(stdout, _("FAIL")); + + status ++; + _cupsLangPuts(stdout, _(" Too many %%EndDocument comments.")); + } + else if (level > 0) + { + if (!status) + _cupsLangPuts(stdout, _("FAIL")); + + status ++; + _cupsLangPuts(stdout, _(" Too many %%BeginDocument comments.")); + } + + if (saw_long_line > 1) + _cupsLangPrintf(stderr, + _(" Saw %d lines that exceeded 255 characters."), + saw_long_line); + + if (!status) + _cupsLangPuts(stdout, _("PASS")); + + if (binary) + _cupsLangPuts(stdout, _(" Warning: file contains binary data.")); + + if (version < 3.0f) + _cupsLangPrintf(stdout, + _(" Warning: obsolete DSC version %.1f in file."), + version); + + if (saw_end_comments < 0) + _cupsLangPuts(stdout, _(" Warning: no %%EndComments comment in file.")); + + cupsFileClose(fp); + + return (status); +} + + +/* + * 'usage()' - Show program usage. + */ + +static void +usage(void) +{ + _cupsLangPuts(stdout, _("Usage: cupstestdsc [options] filename.ps [... " + "filename.ps]")); + _cupsLangPuts(stdout, _(" cupstestdsc [options] -")); + _cupsLangPuts(stdout, ""); + _cupsLangPuts(stdout, _("Options:")); + _cupsLangPuts(stdout, ""); + _cupsLangPuts(stdout, _(" -h Show program usage")); + _cupsLangPuts(stdout, ""); + _cupsLangPuts(stdout, _("Note: this program only validates the DSC comments, " + "not the PostScript itself.")); + + exit(1); +} + + +/* + * End of "$Id$". + */ diff --git a/systemv/cupstestppd.c b/systemv/cupstestppd.c new file mode 100644 index 0000000000..8d361524b9 --- /dev/null +++ b/systemv/cupstestppd.c @@ -0,0 +1,3956 @@ +/* + * "$Id$" + * + * PPD test program for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products, all rights reserved. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * PostScript is a trademark of Adobe Systems, Inc. + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Contents: + * + * main() - Main entry for test program. + * check_basics() - Check for CR LF, mixed line endings, and blank + * lines. + * check_constraints() - Check UIConstraints in the PPD file. + * check_case() - Check that there are no duplicate groups, options, + * or choices that differ only by case. + * check_defaults() - Check default option keywords in the PPD file. + * check_duplex() - Check duplex keywords in the PPD file. + * check_filters() - Check filters in the PPD file. + * check_profiles() - Check ICC color profiles in the PPD file. + * check_sizes() - Check media sizes in the PPD file. + * check_translations() - Check translations in the PPD file. + * show_conflicts() - Show option conflicts in a PPD file. + * test_raster() - Test PostScript commands for raster printers. + * usage() - Show program usage. + * valid_path() - Check whether a path has the correct capitalization. + * valid_utf8() - Check whether a string contains valid UTF-8 text. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include +#include +#ifdef WIN32 +# define X_OK 0 +#endif /* WIN32 */ + + +/* + * Error warning overrides... + */ + +enum +{ + WARN_NONE = 0, + WARN_CONSTRAINTS = 1, + WARN_DEFAULTS = 2, + WARN_FILTERS = 4, + WARN_PROFILES = 8, + WARN_TRANSLATIONS = 16, + WARN_DUPLEX = 32, + WARN_SIZES = 64, + WARN_FILENAME = 128, + WARN_ALL = 255 +}; + + +/* + * Error codes... + */ + +enum +{ + ERROR_NONE = 0, + ERROR_USAGE, + ERROR_FILE_OPEN, + ERROR_PPD_FORMAT, + ERROR_CONFORMANCE +}; + + +/* + * Line endings... + */ + +enum +{ + EOL_NONE = 0, + EOL_CR, + EOL_LF, + EOL_CRLF +}; + + +/* + * File permissions... + */ + +#define MODE_WRITE 0022 /* Group/other write */ +#define MODE_MASK 0555 /* Owner/group/other read+exec/search */ +#define MODE_DATAFILE 0444 /* Owner/group/other read */ +#define MODE_DIRECTORY 0555 /* Owner/group/other read+search */ +#define MODE_PROGRAM 0555 /* Owner/group/other read+exec */ + + +/* + * Local functions... + */ + +static void check_basics(const char *filename); +static int check_constraints(ppd_file_t *ppd, int errors, int verbose, + int warn); +static int check_case(ppd_file_t *ppd, int errors, int verbose); +static int check_defaults(ppd_file_t *ppd, int errors, int verbose, + int warn); +static int check_duplex(ppd_file_t *ppd, int errors, int verbose, + int warn); +static int check_filters(ppd_file_t *ppd, const char *root, int errors, + int verbose, int warn); +static int check_profiles(ppd_file_t *ppd, const char *root, int errors, + int verbose, int warn); +static int check_sizes(ppd_file_t *ppd, int errors, int verbose, int warn); +static int check_translations(ppd_file_t *ppd, int errors, int verbose, + int warn); +static void show_conflicts(ppd_file_t *ppd, const char *prefix); +static int test_raster(ppd_file_t *ppd, int verbose); +static void usage(void) __attribute__((noreturn)); +static int valid_path(const char *keyword, const char *path, int errors, + int verbose, int warn); +static int valid_utf8(const char *s); + + +/* + * 'main()' - Main entry for test program. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int i, j, k, m, n; /* Looping vars */ + int len; /* Length of option name */ + char *opt; /* Option character */ + const char *ptr; /* Pointer into string */ + int files; /* Number of files */ + int verbose; /* Want verbose output? */ + int warn; /* Which errors to just warn about */ + int ignore; /* Which errors to ignore */ + int status; /* Exit status */ + int errors; /* Number of conformance errors */ + int ppdversion; /* PPD spec version in PPD file */ + ppd_status_t error; /* Status of ppdOpen*() */ + int line; /* Line number for error */ + char *root; /* Root directory */ + int xdpi, /* X resolution */ + ydpi; /* Y resolution */ + ppd_file_t *ppd; /* PPD file record */ + ppd_attr_t *attr; /* PPD attribute */ + ppd_size_t *size; /* Size record */ + ppd_group_t *group; /* UI group */ + ppd_option_t *option; /* Standard UI option */ + ppd_group_t *group2; /* UI group */ + ppd_option_t *option2; /* Standard UI option */ + ppd_choice_t *choice; /* Standard UI option choice */ + struct lconv *loc; /* Locale data */ + static char *uis[] = { "BOOLEAN", "PICKONE", "PICKMANY" }; + static char *sections[] = { "ANY", "DOCUMENT", "EXIT", + "JCL", "PAGE", "PROLOG" }; + + + _cupsSetLocale(argv); + loc = localeconv(); + + /* + * Display PPD files for each file listed on the command-line... + */ + + ppdSetConformance(PPD_CONFORM_STRICT); + + verbose = 0; + ppd = NULL; + files = 0; + status = ERROR_NONE; + root = ""; + warn = WARN_NONE; + ignore = WARN_NONE; + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-' && argv[i][1]) + { + for (opt = argv[i] + 1; *opt; opt ++) + switch (*opt) + { + case 'I' : /* Ignore errors */ + i ++; + + if (i >= argc) + usage(); + + if (!strcmp(argv[i], "none")) + ignore = WARN_NONE; + else if (!strcmp(argv[i], "filename")) + ignore |= WARN_FILENAME; + else if (!strcmp(argv[i], "filters")) + ignore |= WARN_FILTERS; + else if (!strcmp(argv[i], "profiles")) + ignore |= WARN_PROFILES; + else if (!strcmp(argv[i], "all")) + ignore = WARN_FILTERS | WARN_PROFILES; + else + usage(); + break; + + case 'R' : /* Alternate root directory */ + i ++; + + if (i >= argc) + usage(); + + root = argv[i]; + break; + + case 'W' : /* Turn errors into warnings */ + i ++; + + if (i >= argc) + usage(); + + if (!strcmp(argv[i], "none")) + warn = WARN_NONE; + else if (!strcmp(argv[i], "constraints")) + warn |= WARN_CONSTRAINTS; + else if (!strcmp(argv[i], "defaults")) + warn |= WARN_DEFAULTS; + else if (!strcmp(argv[i], "duplex")) + warn |= WARN_DUPLEX; + else if (!strcmp(argv[i], "filters")) + warn |= WARN_FILTERS; + else if (!strcmp(argv[i], "profiles")) + warn |= WARN_PROFILES; + else if (!strcmp(argv[i], "sizes")) + warn |= WARN_SIZES; + else if (!strcmp(argv[i], "translations")) + warn |= WARN_TRANSLATIONS; + else if (!strcmp(argv[i], "all")) + warn = WARN_ALL; + else + usage(); + break; + + case 'q' : /* Quiet mode */ + if (verbose > 0) + { + _cupsLangPuts(stderr, + _("cupstestppd: The -q option is incompatible " + "with the -v option.")); + return (1); + } + + verbose --; + break; + + case 'r' : /* Relaxed mode */ + ppdSetConformance(PPD_CONFORM_RELAXED); + break; + + case 'v' : /* Verbose mode */ + if (verbose < 0) + { + _cupsLangPuts(stderr, + _("cupstestppd: The -v option is incompatible " + "with the -q option.")); + return (1); + } + + verbose ++; + break; + + default : + usage(); + break; + } + } + else + { + /* + * Open the PPD file... + */ + + if (files && verbose >= 0) + puts(""); + + files ++; + + if (argv[i][0] == '-') + { + /* + * Read from stdin... + */ + + ppd = ppdOpen(stdin); + + if (verbose >= 0) + printf("%s:", (ppd && ppd->pcfilename) ? ppd->pcfilename : "(stdin)"); + } + else + { + /* + * Read from a file... + */ + + if (verbose >= 0) + printf("%s:", argv[i]); + + ppd = ppdOpenFile(argv[i]); + } + + if (ppd == NULL) + { + error = ppdLastError(&line); + + if (error <= PPD_ALLOC_ERROR) + { + status = ERROR_FILE_OPEN; + + if (verbose >= 0) + { + _cupsLangPuts(stdout, _(" FAIL")); + _cupsLangPrintf(stdout, + _(" **FAIL** Unable to open PPD file - %s"), + strerror(errno)); + } + } + else + { + status = ERROR_PPD_FORMAT; + + if (verbose >= 0) + { + _cupsLangPuts(stdout, _(" FAIL")); + _cupsLangPrintf(stdout, + _(" **FAIL** Unable to open PPD file - " + "%s on line %d."), + ppdErrorString(error), line); + + switch (error) + { + case PPD_MISSING_PPDADOBE4 : + _cupsLangPuts(stdout, + _(" REF: Page 42, section " + "5.2.")); + break; + case PPD_MISSING_VALUE : + _cupsLangPuts(stdout, + _(" REF: Page 20, section " + "3.4.")); + break; + case PPD_BAD_OPEN_GROUP : + case PPD_NESTED_OPEN_GROUP : + _cupsLangPuts(stdout, + _(" REF: Pages 45-46, section " + "5.2.")); + break; + case PPD_BAD_OPEN_UI : + case PPD_NESTED_OPEN_UI : + _cupsLangPuts(stdout, + _(" REF: Pages 42-45, section " + "5.2.")); + break; + case PPD_BAD_ORDER_DEPENDENCY : + _cupsLangPuts(stdout, + _(" REF: Pages 48-49, section " + "5.2.")); + break; + case PPD_BAD_UI_CONSTRAINTS : + _cupsLangPuts(stdout, + _(" REF: Pages 52-54, section " + "5.2.")); + break; + case PPD_MISSING_ASTERISK : + _cupsLangPuts(stdout, + _(" REF: Page 15, section " + "3.2.")); + break; + case PPD_LINE_TOO_LONG : + _cupsLangPuts(stdout, + _(" REF: Page 15, section " + "3.1.")); + break; + case PPD_ILLEGAL_CHARACTER : + _cupsLangPuts(stdout, + _(" REF: Page 15, section " + "3.1.")); + break; + case PPD_ILLEGAL_MAIN_KEYWORD : + _cupsLangPuts(stdout, + _(" REF: Pages 16-17, section " + "3.2.")); + break; + case PPD_ILLEGAL_OPTION_KEYWORD : + _cupsLangPuts(stdout, + _(" REF: Page 19, section " + "3.3.")); + break; + case PPD_ILLEGAL_TRANSLATION : + _cupsLangPuts(stdout, + _(" REF: Page 27, section " + "3.5.")); + break; + default : + break; + } + + check_basics(argv[i]); + } + } + + continue; + } + + /* + * Show the header and then perform basic conformance tests (limited + * only by what the CUPS PPD functions actually load...) + */ + + errors = 0; + ppdversion = 43; + + if (verbose > 0) + _cupsLangPuts(stdout, + _("\n DETAILED CONFORMANCE TEST RESULTS")); + + if ((attr = ppdFindAttr(ppd, "FormatVersion", NULL)) != NULL && + attr->value) + ppdversion = (int)(10 * _cupsStrScand(attr->value, NULL, loc) + 0.5); + + if ((attr = ppdFindAttr(ppd, "cupsFilter2", NULL)) != NULL) + { + do + { + if (strstr(attr->value, "application/vnd.cups-raster")) + { + if (!test_raster(ppd, verbose)) + errors ++; + break; + } + } + while ((attr = ppdFindNextAttr(ppd, "cupsFilter2", NULL)) != NULL); + } + else + { + for (j = 0; j < ppd->num_filters; j ++) + if (strstr(ppd->filters[j], "application/vnd.cups-raster")) + { + if (!test_raster(ppd, verbose)) + errors ++; + break; + } + } + + /* + * Look for default keywords with no matching option... + */ + + if (!(warn & WARN_DEFAULTS)) + errors = check_defaults(ppd, errors, verbose, 0); + + if ((attr = ppdFindAttr(ppd, "DefaultImageableArea", NULL)) == NULL) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** REQUIRED DefaultImageableArea\n" + " REF: Page 102, section 5.15.")); + } + + errors ++; + } + else if (ppdPageSize(ppd, attr->value) == NULL && + strcmp(attr->value, "Unknown")) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" **FAIL** BAD DefaultImageableArea %s\n" + " REF: Page 102, section 5.15."), + attr->value); + } + + errors ++; + } + else + { + if (verbose > 0) + _cupsLangPuts(stdout, _(" PASS DefaultImageableArea")); + } + + if ((attr = ppdFindAttr(ppd, "DefaultPaperDimension", NULL)) == NULL) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** REQUIRED DefaultPaperDimension\n" + " REF: Page 103, section 5.15.")); + } + + errors ++; + } + else if (ppdPageSize(ppd, attr->value) == NULL && + strcmp(attr->value, "Unknown")) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" **FAIL** BAD DefaultPaperDimension %s\n" + " REF: Page 103, section 5.15."), + attr->value); + } + + errors ++; + } + else if (verbose > 0) + _cupsLangPuts(stdout, _(" PASS DefaultPaperDimension")); + + for (j = 0, group = ppd->groups; j < ppd->num_groups; j ++, group ++) + for (k = 0, option = group->options; + k < group->num_options; + k ++, option ++) + { + /* + * Verify that we have a default choice... + */ + + if (option->defchoice[0]) + { + if (ppdFindChoice(option, option->defchoice) == NULL && + strcmp(option->defchoice, "Unknown")) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" **FAIL** BAD Default%s %s\n" + " REF: Page 40, section 4.5."), + option->keyword, option->defchoice); + } + + errors ++; + } + else if (verbose > 0) + _cupsLangPrintf(stdout, + _(" PASS Default%s"), + option->keyword); + } + else + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" **FAIL** REQUIRED Default%s\n" + " REF: Page 40, section 4.5."), + option->keyword); + } + + errors ++; + } + } + + if ((attr = ppdFindAttr(ppd, "FileVersion", NULL)) != NULL) + { + for (ptr = attr->value; *ptr; ptr ++) + if (!isdigit(*ptr & 255) && *ptr != '.') + break; + + if (*ptr) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" **FAIL** Bad FileVersion \"%s\"\n" + " REF: Page 56, section 5.3."), + attr->value); + } + + errors ++; + } + else if (verbose > 0) + _cupsLangPuts(stdout, _(" PASS FileVersion")); + } + else + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** REQUIRED FileVersion\n" + " REF: Page 56, section 5.3.")); + } + + errors ++; + } + + if ((attr = ppdFindAttr(ppd, "FormatVersion", NULL)) != NULL) + { + ptr = attr->value; + if (*ptr == '4' && ptr[1] == '.') + { + + for (ptr += 2; *ptr; ptr ++) + if (!isdigit(*ptr & 255)) + break; + } + + if (*ptr) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" **FAIL** Bad FormatVersion \"%s\"\n" + " REF: Page 56, section 5.3."), + attr->value); + } + + errors ++; + } + else if (verbose > 0) + _cupsLangPuts(stdout, _(" PASS FormatVersion")); + } + else + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** REQUIRED FormatVersion\n" + " REF: Page 56, section 5.3.")); + } + + errors ++; + } + + if (ppd->lang_encoding != NULL) + { + if (verbose > 0) + _cupsLangPuts(stdout, _(" PASS LanguageEncoding")); + } + else if (ppdversion > 40) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** REQUIRED LanguageEncoding\n" + " REF: Pages 56-57, section 5.3.")); + } + + errors ++; + } + + if (ppd->lang_version != NULL) + { + if (verbose > 0) + _cupsLangPuts(stdout, _(" PASS LanguageVersion")); + } + else + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** REQUIRED LanguageVersion\n" + " REF: Pages 57-58, section 5.3.")); + } + + errors ++; + } + + if (ppd->manufacturer != NULL) + { + if (!_cups_strncasecmp(ppd->manufacturer, "Hewlett-Packard", 15) || + !_cups_strncasecmp(ppd->manufacturer, "Hewlett Packard", 15)) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** BAD Manufacturer (should be " + "\"HP\")\n" + " REF: Page 211, table D.1.")); + } + + errors ++; + } + else if (!_cups_strncasecmp(ppd->manufacturer, "OkiData", 7) || + !_cups_strncasecmp(ppd->manufacturer, "Oki Data", 8)) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** BAD Manufacturer (should be " + "\"Oki\")\n" + " REF: Page 211, table D.1.")); + } + + errors ++; + } + else if (verbose > 0) + _cupsLangPuts(stdout, _(" PASS Manufacturer")); + } + else if (ppdversion >= 43) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** REQUIRED Manufacturer\n" + " REF: Pages 58-59, section 5.3.")); + } + + errors ++; + } + + if (ppd->modelname != NULL) + { + for (ptr = ppd->modelname; *ptr; ptr ++) + if (!isalnum(*ptr & 255) && !strchr(" ./-+", *ptr)) + break; + + if (*ptr) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" **FAIL** BAD ModelName - \"%c\" not " + "allowed in string.\n" + " REF: Pages 59-60, section 5.3."), + *ptr); + } + + errors ++; + } + else if (verbose > 0) + _cupsLangPuts(stdout, _(" PASS ModelName")); + } + else + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** REQUIRED ModelName\n" + " REF: Pages 59-60, section 5.3.")); + } + + errors ++; + } + + if (ppd->nickname != NULL) + { + if (verbose > 0) + _cupsLangPuts(stdout, _(" PASS NickName")); + } + else + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** REQUIRED NickName\n" + " REF: Page 60, section 5.3.")); + } + + errors ++; + } + + if (ppdFindOption(ppd, "PageSize") != NULL) + { + if (verbose > 0) + _cupsLangPuts(stdout, _(" PASS PageSize")); + } + else + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** REQUIRED PageSize\n" + " REF: Pages 99-100, section 5.14.")); + } + + errors ++; + } + + if (ppdFindOption(ppd, "PageRegion") != NULL) + { + if (verbose > 0) + _cupsLangPuts(stdout, _(" PASS PageRegion")); + } + else + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** REQUIRED PageRegion\n" + " REF: Page 100, section 5.14.")); + } + + errors ++; + } + + if (ppd->pcfilename != NULL) + { + if (verbose > 0) + _cupsLangPuts(stdout, _(" PASS PCFileName")); + } + else if (!(ignore & WARN_FILENAME)) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** REQUIRED PCFileName\n" + " REF: Pages 61-62, section 5.3.")); + } + + errors ++; + } + + if (ppd->product != NULL) + { + if (ppd->product[0] != '(' || + ppd->product[strlen(ppd->product) - 1] != ')') + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** BAD Product - not \"(string)\".\n" + " REF: Page 62, section 5.3.")); + } + + errors ++; + } + else if (verbose > 0) + _cupsLangPuts(stdout, _(" PASS Product")); + } + else + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** REQUIRED Product\n" + " REF: Page 62, section 5.3.")); + } + + errors ++; + } + + if ((attr = ppdFindAttr(ppd, "PSVersion", NULL)) != NULL && + attr->value != NULL) + { + char junkstr[255]; /* Temp string */ + int junkint; /* Temp integer */ + + + if (sscanf(attr->value, "(%[^)])%d", junkstr, &junkint) != 2) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** BAD PSVersion - not \"(string) " + "int\".\n" + " REF: Pages 62-64, section 5.3.")); + } + + errors ++; + } + else if (verbose > 0) + _cupsLangPuts(stdout, _(" PASS PSVersion")); + } + else + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** REQUIRED PSVersion\n" + " REF: Pages 62-64, section 5.3.")); + } + + errors ++; + } + + if (ppd->shortnickname != NULL) + { + if (strlen(ppd->shortnickname) > 31) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** BAD ShortNickName - longer " + "than 31 chars.\n" + " REF: Pages 64-65, section 5.3.")); + } + + errors ++; + } + else if (verbose > 0) + _cupsLangPuts(stdout, _(" PASS ShortNickName")); + } + else if (ppdversion >= 43) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** REQUIRED ShortNickName\n" + " REF: Page 64-65, section 5.3.")); + } + + errors ++; + } + + if (ppd->patches != NULL && strchr(ppd->patches, '\"') && + strstr(ppd->patches, "*End")) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** BAD JobPatchFile attribute in file\n" + " REF: Page 24, section 3.4.")); + } + + errors ++; + } + + /* + * Check for page sizes without the corresponding ImageableArea or + * PaperDimension values... + */ + + if (ppd->num_sizes == 0) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPuts(stdout, + _(" **FAIL** REQUIRED PageSize\n" + " REF: Page 41, section 5.\n" + " REF: Page 99, section 5.14.")); + } + + errors ++; + } + else + { + for (j = 0, size = ppd->sizes; j < ppd->num_sizes; j ++, size ++) + { + /* + * Don't check custom size... + */ + + if (!strcmp(size->name, "Custom")) + continue; + + /* + * Check for ImageableArea... + */ + + if (size->left == 0.0 && size->bottom == 0.0 && + size->right == 0.0 && size->top == 0.0) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" **FAIL** REQUIRED ImageableArea for " + "PageSize %s\n" + " REF: Page 41, section 5.\n" + " REF: Page 102, section 5.15."), + size->name); + } + + errors ++; + } + + /* + * Check for PaperDimension... + */ + + if (size->width == 0.0 && size->length == 0.0) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" **FAIL** REQUIRED PaperDimension " + "for PageSize %s\n" + " REF: Page 41, section 5.\n" + " REF: Page 103, section 5.15."), + size->name); + } + + errors ++; + } + } + } + + /* + * Check for valid Resolution, JCLResolution, or SetResolution values... + */ + + if ((option = ppdFindOption(ppd, "Resolution")) == NULL) + if ((option = ppdFindOption(ppd, "JCLResolution")) == NULL) + option = ppdFindOption(ppd, "SetResolution"); + + if (option != NULL) + { + for (j = option->num_choices, choice = option->choices; + j > 0; + j --, choice ++) + { + /* + * Verify that all resolution options are of the form NNNdpi + * or NNNxNNNdpi... + */ + + xdpi = strtol(choice->choice, (char **)&ptr, 10); + if (ptr > choice->choice && xdpi > 0) + { + if (*ptr == 'x') + ydpi = strtol(ptr + 1, (char **)&ptr, 10); + else + ydpi = xdpi; + } + else + ydpi = xdpi; + + if (xdpi <= 0 || xdpi > 99999 || ydpi <= 0 || ydpi > 99999 || + strcmp(ptr, "dpi")) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" **FAIL** Bad %s choice %s\n" + " REF: Page 84, section 5.9"), + option->keyword, choice->choice); + } + + errors ++; + } + } + } + + if ((attr = ppdFindAttr(ppd, "1284DeviceID", NULL)) && + strcmp(attr->name, "1284DeviceID")) + { + if (verbose >= 0) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" **FAIL** %s must be 1284DeviceID\n" + " REF: Page 72, section 5.5"), + attr->name); + } + + errors ++; + } + + errors = check_case(ppd, errors, verbose); + + if (!(warn & WARN_CONSTRAINTS)) + errors = check_constraints(ppd, errors, verbose, 0); + + if (!(warn & WARN_FILTERS) && !(ignore & WARN_FILTERS)) + errors = check_filters(ppd, root, errors, verbose, 0); + + if (!(warn & WARN_PROFILES) && !(ignore & WARN_PROFILES)) + errors = check_profiles(ppd, root, errors, verbose, 0); + + if (!(warn & WARN_SIZES)) + errors = check_sizes(ppd, errors, verbose, 0); + + if (!(warn & WARN_TRANSLATIONS)) + errors = check_translations(ppd, errors, verbose, 0); + + if (!(warn & WARN_DUPLEX)) + errors = check_duplex(ppd, errors, verbose, 0); + + if ((attr = ppdFindAttr(ppd, "cupsLanguages", NULL)) != NULL && + attr->value) + { + /* + * This file contains localizations, check for conformance of the + * base translation... + */ + + if ((attr = ppdFindAttr(ppd, "LanguageEncoding", NULL)) != NULL) + { + if (!attr->value || strcmp(attr->value, "ISOLatin1")) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" **FAIL** Bad LanguageEncoding %s - " + "must be ISOLatin1."), + attr->value ? attr->value : "(null)"); + + errors ++; + } + + if (!ppd->lang_version || strcmp(ppd->lang_version, "English")) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" **FAIL** Bad LanguageVersion %s - " + "must be English."), + ppd->lang_version ? ppd->lang_version : "(null)"); + + errors ++; + } + + /* + * Loop through all options and choices... + */ + + for (option = ppdFirstOption(ppd); + option; + option = ppdNextOption(ppd)) + { + /* + * Check for special characters outside A0 to BF, F7, or F8 + * that are used for languages other than English. + */ + + for (ptr = option->text; *ptr; ptr ++) + if ((*ptr & 0x80) && (*ptr & 0xe0) != 0xa0 && + (*ptr & 0xff) != 0xf7 && (*ptr & 0xff) != 0xf8) + break; + + if (*ptr) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" **FAIL** Default translation " + "string for option %s contains 8-bit " + "characters."), + option->keyword); + + errors ++; + } + + for (j = 0; j < option->num_choices; j ++) + { + /* + * Check for special characters outside A0 to BF, F7, or F8 + * that are used for languages other than English. + */ + + for (ptr = option->choices[j].text; *ptr; ptr ++) + if ((*ptr & 0x80) && (*ptr & 0xe0) != 0xa0 && + (*ptr & 0xff) != 0xf7 && (*ptr & 0xff) != 0xf8) + break; + + if (*ptr) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" **FAIL** Default translation " + "string for option %s choice %s contains " + "8-bit characters."), + option->keyword, + option->choices[j].choice); + + errors ++; + } + } + } + } + } + + /* + * Final pass/fail notification... + */ + + if (errors) + status = ERROR_CONFORMANCE; + else if (!verbose) + _cupsLangPuts(stdout, _(" PASS")); + + if (verbose >= 0) + { + check_basics(argv[i]); + + if (warn & WARN_DEFAULTS) + errors = check_defaults(ppd, errors, verbose, 1); + + if (warn & WARN_CONSTRAINTS) + errors = check_constraints(ppd, errors, verbose, 1); + + if ((warn & WARN_FILTERS) && !(ignore & WARN_FILTERS)) + errors = check_filters(ppd, root, errors, verbose, 1); + + if ((warn & WARN_PROFILES) && !(ignore & WARN_PROFILES)) + errors = check_profiles(ppd, root, errors, verbose, 1); + + if (warn & WARN_SIZES) + errors = check_sizes(ppd, errors, verbose, 1); + else + errors = check_sizes(ppd, errors, verbose, 2); + + if (warn & WARN_TRANSLATIONS) + errors = check_translations(ppd, errors, verbose, 1); + + if (warn & WARN_DUPLEX) + errors = check_duplex(ppd, errors, verbose, 1); + + /* + * Look for legacy duplex keywords... + */ + + if ((option = ppdFindOption(ppd, "JCLDuplex")) == NULL) + if ((option = ppdFindOption(ppd, "EFDuplex")) == NULL) + option = ppdFindOption(ppd, "KD03Duplex"); + + if (option) + _cupsLangPrintf(stdout, + _(" WARN Duplex option keyword %s may not " + "work as expected and should be named Duplex.\n" + " REF: Page 122, section 5.17"), + option->keyword); + + /* + * Look for default keywords with no corresponding option... + */ + + for (j = 0; j < ppd->num_attrs; j ++) + { + attr = ppd->attrs[j]; + + if (!strcmp(attr->name, "DefaultColorSpace") || + !strcmp(attr->name, "DefaultColorSep") || + !strcmp(attr->name, "DefaultFont") || + !strcmp(attr->name, "DefaultHalftoneType") || + !strcmp(attr->name, "DefaultImageableArea") || + !strcmp(attr->name, "DefaultLeadingEdge") || + !strcmp(attr->name, "DefaultOutputOrder") || + !strcmp(attr->name, "DefaultPaperDimension") || + !strcmp(attr->name, "DefaultResolution") || + !strcmp(attr->name, "DefaultScreenProc") || + !strcmp(attr->name, "DefaultTransfer")) + continue; + + if (!strncmp(attr->name, "Default", 7) && + !ppdFindOption(ppd, attr->name + 7)) + _cupsLangPrintf(stdout, + _(" WARN %s has no corresponding " + "options."), + attr->name); + } + + if (ppdversion < 43) + { + _cupsLangPrintf(stdout, + _(" WARN Obsolete PPD version %.1f.\n" + " REF: Page 42, section 5.2."), + 0.1f * ppdversion); + } + + if (!ppd->lang_encoding && ppdversion < 41) + { + _cupsLangPuts(stdout, + _(" WARN LanguageEncoding required by PPD " + "4.3 spec.\n" + " REF: Pages 56-57, section 5.3.")); + } + + if (!ppd->manufacturer && ppdversion < 43) + { + _cupsLangPuts(stdout, + _(" WARN Manufacturer required by PPD " + "4.3 spec.\n" + " REF: Pages 58-59, section 5.3.")); + } + + /* + * Treat a PCFileName attribute longer than 12 characters as + * a warning and not a hard error... + */ + + if (!(ignore & WARN_FILENAME) && ppd->pcfilename) + { + if (strlen(ppd->pcfilename) > 12) + { + _cupsLangPuts(stdout, + _(" WARN PCFileName longer than 8.3 in " + "violation of PPD spec.\n" + " REF: Pages 61-62, section " + "5.3.")); + } + + if (!_cups_strcasecmp(ppd->pcfilename, "unused.ppd")) + _cupsLangPuts(stdout, + _(" WARN PCFileName should contain a " + "unique filename.\n" + " REF: Pages 61-62, section " + "5.3.")); + } + + if (!ppd->shortnickname && ppdversion < 43) + { + _cupsLangPuts(stdout, + _(" WARN ShortNickName required by PPD " + "4.3 spec.\n" + " REF: Pages 64-65, section 5.3.")); + } + + /* + * Check the Protocols line and flag PJL + BCP since TBCP is + * usually used with PJL... + */ + + if (ppd->protocols) + { + if (strstr(ppd->protocols, "PJL") && + strstr(ppd->protocols, "BCP") && + !strstr(ppd->protocols, "TBCP")) + { + _cupsLangPuts(stdout, + _(" WARN Protocols contains both PJL " + "and BCP; expected TBCP.\n" + " REF: Pages 78-79, section 5.7.")); + } + + if (strstr(ppd->protocols, "PJL") && + (!ppd->jcl_begin || !ppd->jcl_end || !ppd->jcl_ps)) + { + _cupsLangPuts(stdout, + _(" WARN Protocols contains PJL but JCL " + "attributes are not set.\n" + " REF: Pages 78-79, section 5.7.")); + } + } + + /* + * Check for options with a common prefix, e.g. Duplex and Duplexer, + * which are errors according to the spec but won't cause problems + * with CUPS specifically... + */ + + for (j = 0, group = ppd->groups; j < ppd->num_groups; j ++, group ++) + for (k = 0, option = group->options; + k < group->num_options; + k ++, option ++) + { + len = (int)strlen(option->keyword); + + for (m = 0, group2 = ppd->groups; + m < ppd->num_groups; + m ++, group2 ++) + for (n = 0, option2 = group2->options; + n < group2->num_options; + n ++, option2 ++) + if (option != option2 && + len < (int)strlen(option2->keyword) && + !strncmp(option->keyword, option2->keyword, len)) + { + _cupsLangPrintf(stdout, + _(" WARN %s shares a common " + "prefix with %s\n" + " REF: Page 15, section " + "3.2."), + option->keyword, option2->keyword); + } + } + } + + if (verbose > 0) + { + if (errors) + _cupsLangPrintf(stdout, _(" %d ERRORS FOUND"), errors); + else + _cupsLangPuts(stdout, _(" NO ERRORS FOUND")); + } + + /* + * Then list the options, if "-v" was provided... + */ + + if (verbose > 1) + { + _cupsLangPrintf(stdout, + "\n" + " language_level = %d\n" + " color_device = %s\n" + " variable_sizes = %s\n" + " landscape = %d", + ppd->language_level, + ppd->color_device ? "TRUE" : "FALSE", + ppd->variable_sizes ? "TRUE" : "FALSE", + ppd->landscape); + + switch (ppd->colorspace) + { + case PPD_CS_CMYK : + _cupsLangPuts(stdout, " colorspace = PPD_CS_CMYK"); + break; + case PPD_CS_CMY : + _cupsLangPuts(stdout, " colorspace = PPD_CS_CMY"); + break; + case PPD_CS_GRAY : + _cupsLangPuts(stdout, " colorspace = PPD_CS_GRAY"); + break; + case PPD_CS_RGB : + _cupsLangPuts(stdout, " colorspace = PPD_CS_RGB"); + break; + default : + _cupsLangPuts(stdout, " colorspace = "); + break; + } + + _cupsLangPrintf(stdout, " num_emulations = %d", + ppd->num_emulations); + for (j = 0; j < ppd->num_emulations; j ++) + _cupsLangPrintf(stdout, " emulations[%d] = %s", + j, ppd->emulations[j].name); + + _cupsLangPrintf(stdout, " lang_encoding = %s", + ppd->lang_encoding); + _cupsLangPrintf(stdout, " lang_version = %s", + ppd->lang_version); + _cupsLangPrintf(stdout, " modelname = %s", ppd->modelname); + _cupsLangPrintf(stdout, " ttrasterizer = %s", + ppd->ttrasterizer == NULL ? "None" : ppd->ttrasterizer); + _cupsLangPrintf(stdout, " manufacturer = %s", + ppd->manufacturer); + _cupsLangPrintf(stdout, " product = %s", ppd->product); + _cupsLangPrintf(stdout, " nickname = %s", ppd->nickname); + _cupsLangPrintf(stdout, " shortnickname = %s", + ppd->shortnickname); + _cupsLangPrintf(stdout, " patches = %d bytes", + ppd->patches == NULL ? 0 : (int)strlen(ppd->patches)); + + _cupsLangPrintf(stdout, " num_groups = %d", ppd->num_groups); + for (j = 0, group = ppd->groups; j < ppd->num_groups; j ++, group ++) + { + _cupsLangPrintf(stdout, " group[%d] = %s", + j, group->text); + + for (k = 0, option = group->options; k < group->num_options; k ++, option ++) + { + _cupsLangPrintf(stdout, + " options[%d] = %s (%s) %s %s %.0f " + "(%d choices)", + k, option->keyword, option->text, uis[option->ui], + sections[option->section], option->order, + option->num_choices); + + if (!strcmp(option->keyword, "PageSize") || + !strcmp(option->keyword, "PageRegion")) + { + for (m = option->num_choices, choice = option->choices; + m > 0; + m --, choice ++) + { + size = ppdPageSize(ppd, choice->choice); + + if (size == NULL) + _cupsLangPrintf(stdout, + " %s (%s) = ERROR%s", + choice->choice, choice->text, + !strcmp(option->defchoice, choice->choice) + ? " *" : ""); + else + _cupsLangPrintf(stdout, + " %s (%s) = %.2fx%.2fin " + "(%.1f,%.1f,%.1f,%.1f)%s", + choice->choice, choice->text, + size->width / 72.0, size->length / 72.0, + size->left / 72.0, size->bottom / 72.0, + size->right / 72.0, size->top / 72.0, + !strcmp(option->defchoice, choice->choice) + ? " *" : ""); + } + } + else + { + for (m = option->num_choices, choice = option->choices; + m > 0; + m --, choice ++) + { + _cupsLangPrintf(stdout, " %s (%s)%s", + choice->choice, choice->text, + !strcmp(option->defchoice, choice->choice) + ? " *" : ""); + } + } + } + } + + _cupsLangPrintf(stdout, " num_consts = %d", + ppd->num_consts); + for (j = 0; j < ppd->num_consts; j ++) + _cupsLangPrintf(stdout, + " consts[%d] = *%s %s *%s %s", + j, ppd->consts[j].option1, ppd->consts[j].choice1, + ppd->consts[j].option2, ppd->consts[j].choice2); + + _cupsLangPrintf(stdout, " num_profiles = %d", + ppd->num_profiles); + for (j = 0; j < ppd->num_profiles; j ++) + _cupsLangPrintf(stdout, + " profiles[%d] = %s/%s %.3f %.3f " + "[ %.3f %.3f %.3f %.3f %.3f %.3f %.3f %.3f %.3f ]", + j, ppd->profiles[j].resolution, + ppd->profiles[j].media_type, + ppd->profiles[j].gamma, ppd->profiles[j].density, + ppd->profiles[j].matrix[0][0], + ppd->profiles[j].matrix[0][1], + ppd->profiles[j].matrix[0][2], + ppd->profiles[j].matrix[1][0], + ppd->profiles[j].matrix[1][1], + ppd->profiles[j].matrix[1][2], + ppd->profiles[j].matrix[2][0], + ppd->profiles[j].matrix[2][1], + ppd->profiles[j].matrix[2][2]); + + _cupsLangPrintf(stdout, " num_fonts = %d", ppd->num_fonts); + for (j = 0; j < ppd->num_fonts; j ++) + _cupsLangPrintf(stdout, " fonts[%d] = %s", + j, ppd->fonts[j]); + + _cupsLangPrintf(stdout, " num_attrs = %d", ppd->num_attrs); + for (j = 0; j < ppd->num_attrs; j ++) + _cupsLangPrintf(stdout, + " attrs[%d] = %s %s%s%s: \"%s\"", j, + ppd->attrs[j]->name, ppd->attrs[j]->spec, + ppd->attrs[j]->text[0] ? "/" : "", + ppd->attrs[j]->text, + ppd->attrs[j]->value ? + ppd->attrs[j]->value : "(null)"); + } + + ppdClose(ppd); + } + + if (!files) + usage(); + + return (status); +} + + +/* + * 'check_basics()' - Check for CR LF, mixed line endings, and blank lines. + */ + +static void +check_basics(const char *filename) /* I - PPD file to check */ +{ + cups_file_t *fp; /* File pointer */ + int ch; /* Current character */ + int col, /* Current column */ + whitespace; /* Only seen whitespace? */ + int eol; /* Line endings */ + int linenum; /* Line number */ + int mixed; /* Mixed line endings? */ + + + if ((fp = cupsFileOpen(filename, "r")) == NULL) + return; + + linenum = 1; + col = 0; + eol = EOL_NONE; + mixed = 0; + whitespace = 1; + + while ((ch = cupsFileGetChar(fp)) != EOF) + { + if (ch == '\r' || ch == '\n') + { + if (ch == '\n') + { + if (eol == EOL_NONE) + eol = EOL_LF; + else if (eol != EOL_LF) + mixed = 1; + } + else if (ch == '\r') + { + if (cupsFilePeekChar(fp) == '\n') + { + cupsFileGetChar(fp); + + if (eol == EOL_NONE) + eol = EOL_CRLF; + else if (eol != EOL_CRLF) + mixed = 1; + } + else if (eol == EOL_NONE) + eol = EOL_CR; + else if (eol != EOL_CR) + mixed = 1; + } + + if (col > 0 && whitespace) + _cupsLangPrintf(stdout, + _(" WARN Line %d only contains whitespace."), + linenum); + + linenum ++; + col = 0; + whitespace = 1; + } + else + { + if (ch != ' ' && ch != '\t') + whitespace = 0; + + col ++; + } + } + + if (mixed) + _cupsLangPuts(stdout, + _(" WARN File contains a mix of CR, LF, and " + "CR LF line endings.")); + + if (eol == EOL_CRLF) + _cupsLangPuts(stdout, + _(" WARN Non-Windows PPD files should use lines " + "ending with only LF, not CR LF.")); + + cupsFileClose(fp); +} + + +/* + * 'check_constraints()' - Check UIConstraints in the PPD file. + */ + +static int /* O - Errors found */ +check_constraints(ppd_file_t *ppd, /* I - PPD file */ + int errors, /* I - Errors found */ + int verbose, /* I - Verbosity level */ + int warn) /* I - Warnings only? */ +{ + int i; /* Looping var */ + const char *prefix; /* WARN/FAIL prefix */ + ppd_const_t *c; /* Current UIConstraints data */ + ppd_attr_t *constattr; /* Current cupsUIConstraints attribute */ + const char *vptr; /* Pointer into constraint value */ + char option[PPD_MAX_NAME], + /* Option name/MainKeyword */ + choice[PPD_MAX_NAME], + /* Choice/OptionKeyword */ + *ptr; /* Pointer into option or choice */ + int num_options; /* Number of options */ + cups_option_t *options; /* Options */ + ppd_option_t *o; /* PPD option */ + + + prefix = warn ? " WARN " : "**FAIL**"; + + + /* + * See what kind of constraint data we have in the PPD... + */ + + if ((constattr = ppdFindAttr(ppd, "cupsUIConstraints", NULL)) != NULL) + { + /* + * Check new-style cupsUIConstraints data... + */ + + for (; constattr; + constattr = ppdFindNextAttr(ppd, "cupsUIConstraints", NULL)) + { + if (!constattr->value) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" %s Empty cupsUIConstraints %s"), + prefix, constattr->spec); + + if (!warn) + errors ++; + + continue; + } + + for (i = 0, vptr = strchr(constattr->value, '*'); + vptr; + i ++, vptr = strchr(vptr + 1, '*')); + + if (i == 0) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" %s Bad cupsUIConstraints %s: \"%s\""), + prefix, constattr->spec, constattr->value); + + if (!warn) + errors ++; + + continue; + } + + cupsArraySave(ppd->sorted_attrs); + + if (constattr->spec[0] && + !ppdFindAttr(ppd, "cupsUIResolver", constattr->spec)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" %s Missing cupsUIResolver %s"), + prefix, constattr->spec); + + if (!warn) + errors ++; + } + + cupsArrayRestore(ppd->sorted_attrs); + + num_options = 0; + options = NULL; + + for (vptr = strchr(constattr->value, '*'); + vptr; + vptr = strchr(vptr, '*')) + { + /* + * Extract "*Option Choice" or just "*Option"... + */ + + for (vptr ++, ptr = option; *vptr && !isspace(*vptr & 255); vptr ++) + if (ptr < (option + sizeof(option) - 1)) + *ptr++ = *vptr; + + *ptr = '\0'; + + while (isspace(*vptr & 255)) + vptr ++; + + if (*vptr == '*') + choice[0] = '\0'; + else + { + for (ptr = choice; *vptr && !isspace(*vptr & 255); vptr ++) + if (ptr < (choice + sizeof(choice) - 1)) + *ptr++ = *vptr; + + *ptr = '\0'; + } + + if (!_cups_strncasecmp(option, "Custom", 6) && !_cups_strcasecmp(choice, "True")) + { + _cups_strcpy(option, option + 6); + strcpy(choice, "Custom"); + } + + if ((o = ppdFindOption(ppd, option)) == NULL) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" %s Missing option %s in " + "cupsUIConstraints %s: \"%s\""), + prefix, option, constattr->spec, constattr->value); + + if (!warn) + errors ++; + + continue; + } + + if (choice[0] && !ppdFindChoice(o, choice)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" %s Missing choice *%s %s in " + "cupsUIConstraints %s: \"%s\""), + prefix, option, choice, constattr->spec, + constattr->value); + + if (!warn) + errors ++; + + continue; + } + + if (choice[0]) + num_options = cupsAddOption(option, choice, num_options, &options); + else + { + for (i = 0; i < o->num_choices; i ++) + if (_cups_strcasecmp(o->choices[i].choice, "None") && + _cups_strcasecmp(o->choices[i].choice, "Off") && + _cups_strcasecmp(o->choices[i].choice, "False")) + { + num_options = cupsAddOption(option, o->choices[i].choice, + num_options, &options); + break; + } + } + } + + /* + * Resolvers must list at least two options... + */ + + if (num_options < 2) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" %s cupsUIResolver %s does not list at least " + "two different options."), + prefix, constattr->spec); + + if (!warn) + errors ++; + } + + /* + * Test the resolver... + */ + + if (!cupsResolveConflicts(ppd, NULL, NULL, &num_options, &options)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" %s cupsUIResolver %s causes a loop."), + prefix, constattr->spec); + + if (!warn) + errors ++; + } + + cupsFreeOptions(num_options, options); + } + } + else + { + /* + * Check old-style [Non]UIConstraints data... + */ + + for (i = ppd->num_consts, c = ppd->consts; i > 0; i --, c ++) + { + if (!_cups_strncasecmp(c->option1, "Custom", 6) && + !_cups_strcasecmp(c->choice1, "True")) + { + strcpy(option, c->option1 + 6); + strcpy(choice, "Custom"); + } + else + { + strcpy(option, c->option1); + strcpy(choice, c->choice1); + } + + if ((o = ppdFindOption(ppd, option)) == NULL) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" %s Missing option %s in " + "UIConstraints \"*%s %s *%s %s\"."), + prefix, c->option1, + c->option1, c->choice1, c->option2, c->choice2); + + if (!warn) + errors ++; + } + else if (choice[0] && !ppdFindChoice(o, choice)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" %s Missing choice *%s %s in " + "UIConstraints \"*%s %s *%s %s\"."), + prefix, c->option1, c->choice1, + c->option1, c->choice1, c->option2, c->choice2); + + if (!warn) + errors ++; + } + + if (!_cups_strncasecmp(c->option2, "Custom", 6) && + !_cups_strcasecmp(c->choice2, "True")) + { + strcpy(option, c->option2 + 6); + strcpy(choice, "Custom"); + } + else + { + strcpy(option, c->option2); + strcpy(choice, c->choice2); + } + + if ((o = ppdFindOption(ppd, option)) == NULL) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" %s Missing option %s in " + "UIConstraints \"*%s %s *%s %s\"."), + prefix, c->option2, + c->option1, c->choice1, c->option2, c->choice2); + + if (!warn) + errors ++; + } + else if (choice[0] && !ppdFindChoice(o, choice)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" %s Missing choice *%s %s in " + "UIConstraints \"*%s %s *%s %s\"."), + prefix, c->option2, c->choice2, + c->option1, c->choice1, c->option2, c->choice2); + + if (!warn) + errors ++; + } + } + } + + return (errors); +} + + +/* + * 'check_case()' - Check that there are no duplicate groups, options, + * or choices that differ only by case. + */ + +static int /* O - Errors found */ +check_case(ppd_file_t *ppd, /* I - PPD file */ + int errors, /* I - Errors found */ + int verbose) /* I - Verbosity level */ +{ + int i, j; /* Looping vars */ + ppd_group_t *groupa, /* First group */ + *groupb; /* Second group */ + ppd_option_t *optiona, /* First option */ + *optionb; /* Second option */ + ppd_choice_t *choicea, /* First choice */ + *choiceb; /* Second choice */ + + + /* + * Check that the groups do not have any duplicate names... + */ + + for (i = ppd->num_groups, groupa = ppd->groups; i > 1; i --, groupa ++) + for (j = i - 1, groupb = groupa + 1; j > 0; j --, groupb ++) + if (!_cups_strcasecmp(groupa->name, groupb->name)) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" **FAIL** Group names %s and %s differ only " + "by case."), + groupa->name, groupb->name); + + errors ++; + } + + /* + * Check that the options do not have any duplicate names... + */ + + for (optiona = ppdFirstOption(ppd); optiona; optiona = ppdNextOption(ppd)) + { + cupsArraySave(ppd->options); + for (optionb = ppdNextOption(ppd); optionb; optionb = ppdNextOption(ppd)) + if (!_cups_strcasecmp(optiona->keyword, optionb->keyword)) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" **FAIL** Option names %s and %s differ only " + "by case."), + optiona->keyword, optionb->keyword); + + errors ++; + } + cupsArrayRestore(ppd->options); + + /* + * Then the choices... + */ + + for (i = optiona->num_choices, choicea = optiona->choices; + i > 1; + i --, choicea ++) + for (j = i - 1, choiceb = choicea + 1; j > 0; j --, choiceb ++) + if (!strcmp(choicea->choice, choiceb->choice)) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" **FAIL** Multiple occurrences of %s " + "choice name %s."), + optiona->keyword, choicea->choice); + + errors ++; + + choicea ++; + i --; + break; + } + else if (!_cups_strcasecmp(choicea->choice, choiceb->choice)) + { + if (!errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" **FAIL** %s choice names %s and %s " + "differ only by case."), + optiona->keyword, choicea->choice, choiceb->choice); + + errors ++; + } + } + + /* + * Return the number of errors found... + */ + + return (errors); +} + + +/* + * 'check_defaults()' - Check default option keywords in the PPD file. + */ + +static int /* O - Errors found */ +check_defaults(ppd_file_t *ppd, /* I - PPD file */ + int errors, /* I - Errors found */ + int verbose, /* I - Verbosity level */ + int warn) /* I - Warnings only? */ +{ + int j, k; /* Looping vars */ + ppd_attr_t *attr; /* PPD attribute */ + ppd_option_t *option; /* Standard UI option */ + const char *prefix; /* WARN/FAIL prefix */ + + + prefix = warn ? " WARN " : "**FAIL**"; + + ppdMarkDefaults(ppd); + if (ppdConflicts(ppd)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Default choices conflicting."), prefix); + + show_conflicts(ppd, prefix); + + if (!warn) + errors ++; + } + + for (j = 0; j < ppd->num_attrs; j ++) + { + attr = ppd->attrs[j]; + + if (!strcmp(attr->name, "DefaultColorSpace") || + !strcmp(attr->name, "DefaultFont") || + !strcmp(attr->name, "DefaultHalftoneType") || + !strcmp(attr->name, "DefaultImageableArea") || + !strcmp(attr->name, "DefaultLeadingEdge") || + !strcmp(attr->name, "DefaultOutputOrder") || + !strcmp(attr->name, "DefaultPaperDimension") || + !strcmp(attr->name, "DefaultResolution") || + !strcmp(attr->name, "DefaultTransfer")) + continue; + + if (!strncmp(attr->name, "Default", 7)) + { + if ((option = ppdFindOption(ppd, attr->name + 7)) != NULL && + strcmp(attr->value, "Unknown")) + { + /* + * Check that the default option value matches a choice... + */ + + for (k = 0; k < option->num_choices; k ++) + if (!strcmp(option->choices[k].choice, attr->value)) + break; + + if (k >= option->num_choices) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s %s %s does not exist."), + prefix, attr->name, attr->value); + + if (!warn) + errors ++; + } + } + } + } + + return (errors); +} + + +/* + * 'check_duplex()' - Check duplex keywords in the PPD file. + */ + +static int /* O - Errors found */ +check_duplex(ppd_file_t *ppd, /* I - PPD file */ + int errors, /* I - Error found */ + int verbose, /* I - Verbosity level */ + int warn) /* I - Warnings only? */ +{ + int i; /* Looping var */ + ppd_option_t *option; /* PPD option */ + ppd_choice_t *choice; /* Current choice */ + const char *prefix; /* Message prefix */ + + + prefix = warn ? " WARN " : "**FAIL**"; + + /* + * Check for a duplex option, and for standard values... + */ + + if ((option = ppdFindOption(ppd, "Duplex")) != NULL) + { + if (!ppdFindChoice(option, "None")) + { + if (verbose >= 0) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" %s REQUIRED %s does not define " + "choice None.\n" + " REF: Page 122, section 5.17"), + prefix, option->keyword); + } + + if (!warn) + errors ++; + } + + for (i = option->num_choices, choice = option->choices; + i > 0; + i --, choice ++) + if (strcmp(choice->choice, "None") && + strcmp(choice->choice, "DuplexNoTumble") && + strcmp(choice->choice, "DuplexTumble") && + strcmp(choice->choice, "SimplexTumble")) + { + if (verbose >= 0) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + _cupsLangPrintf(stdout, + _(" %s Bad %s choice %s.\n" + " REF: Page 122, section 5.17"), + prefix, option->keyword, choice->choice); + } + + if (!warn) + errors ++; + } + } + + return (errors); +} + + +/* + * 'check_filters()' - Check filters in the PPD file. + */ + +static int /* O - Errors found */ +check_filters(ppd_file_t *ppd, /* I - PPD file */ + const char *root, /* I - Root directory */ + int errors, /* I - Errors found */ + int verbose, /* I - Verbosity level */ + int warn) /* I - Warnings only? */ +{ + ppd_attr_t *attr; /* PPD attribute */ + const char *ptr; /* Pointer into string */ + char super[16], /* Super-type for filter */ + type[256], /* Type for filter */ + dstsuper[16], /* Destination super-type for filter */ + dsttype[256], /* Destination type for filter */ + program[1024], /* Program/filter name */ + pathprog[1024]; /* Complete path to program/filter */ + int cost; /* Cost of filter */ + const char *prefix; /* WARN/FAIL prefix */ + struct stat fileinfo; /* File information */ + + + prefix = warn ? " WARN " : "**FAIL**"; + + /* + * cupsFilter + */ + + for (attr = ppdFindAttr(ppd, "cupsFilter", NULL); + attr; + attr = ppdFindNextAttr(ppd, "cupsFilter", NULL)) + { + if (strcmp(attr->name, "cupsFilter")) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad spelling of %s - should be %s."), + prefix, attr->name, "cupsFilter"); + + if (!warn) + errors ++; + } + + if (!attr->value || + sscanf(attr->value, "%15[^/]/%255s%d%*[ \t]%1023[^\n]", super, type, + &cost, program) != 4) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad cupsFilter value \"%s\"."), + prefix, attr->value); + + if (!warn) + errors ++; + } + else if (strcmp(program, "-")) + { + if (program[0] == '/') + snprintf(pathprog, sizeof(pathprog), "%s%s", root, program); + else + { + if ((ptr = getenv("CUPS_SERVERBIN")) == NULL) + ptr = CUPS_SERVERBIN; + + if (*ptr == '/' || !*root) + snprintf(pathprog, sizeof(pathprog), "%s%s/filter/%s", root, ptr, + program); + else + snprintf(pathprog, sizeof(pathprog), "%s/%s/filter/%s", root, ptr, + program); + } + + if (stat(pathprog, &fileinfo)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, _(" %s Missing %s file \"%s\"."), + prefix, "cupsFilter", pathprog); + + if (!warn) + errors ++; + } + else if (fileinfo.st_uid != 0 || + (fileinfo.st_mode & MODE_WRITE) || + (fileinfo.st_mode & MODE_MASK) != MODE_PROGRAM) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad permissions on %s file \"%s\"."), + prefix, "cupsFilter", pathprog); + + if (!warn) + errors ++; + } + else + errors = valid_path("cupsFilter", pathprog, errors, verbose, warn); + } + } + + /* + * cupsFilter2 + */ + + for (attr = ppdFindAttr(ppd, "cupsFilter2", NULL); + attr; + attr = ppdFindNextAttr(ppd, "cupsFilter2", NULL)) + { + if (strcmp(attr->name, "cupsFilter2")) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad spelling of %s - should be %s."), + prefix, attr->name, "cupsFilter2"); + + if (!warn) + errors ++; + } + + if (!attr->value || + sscanf(attr->value, "%15[^/]/%255s%*[ \t]%15[^/]/%255s%d%*[ \t]%1023[^\n]", + super, type, dstsuper, dsttype, &cost, program) != 6) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad cupsFilter2 value \"%s\"."), + prefix, attr->value); + + if (!warn) + errors ++; + } + else if (strcmp(program, "-")) + { + if (strncmp(program, "maxsize(", 8) && + (ptr = strchr(program + 8, ')')) != NULL) + { + ptr ++; + while (_cups_isspace(*ptr)) + ptr ++; + + _cups_strcpy(program, ptr); + } + + if (program[0] == '/') + snprintf(pathprog, sizeof(pathprog), "%s%s", root, program); + else + { + if ((ptr = getenv("CUPS_SERVERBIN")) == NULL) + ptr = CUPS_SERVERBIN; + + if (*ptr == '/' || !*root) + snprintf(pathprog, sizeof(pathprog), "%s%s/filter/%s", root, ptr, + program); + else + snprintf(pathprog, sizeof(pathprog), "%s/%s/filter/%s", root, ptr, + program); + } + + if (stat(pathprog, &fileinfo)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, _(" %s Missing %s file \"%s\"."), + prefix, "cupsFilter2", pathprog); + + if (!warn) + errors ++; + } + else if (fileinfo.st_uid != 0 || + (fileinfo.st_mode & MODE_WRITE) || + (fileinfo.st_mode & MODE_MASK) != MODE_PROGRAM) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad permissions on %s file \"%s\"."), + prefix, "cupsFilter2", pathprog); + + if (!warn) + errors ++; + } + else + errors = valid_path("cupsFilter2", pathprog, errors, verbose, warn); + } + } + + /* + * cupsPreFilter + */ + + for (attr = ppdFindAttr(ppd, "cupsPreFilter", NULL); + attr; + attr = ppdFindNextAttr(ppd, "cupsPreFilter", NULL)) + { + if (strcmp(attr->name, "cupsPreFilter")) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad spelling of %s - should be %s."), + prefix, attr->name, "cupsPreFilter"); + + if (!warn) + errors ++; + } + + if (!attr->value || + sscanf(attr->value, "%15[^/]/%255s%d%*[ \t]%1023[^\n]", super, type, + &cost, program) != 4) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad cupsPreFilter value \"%s\"."), + prefix, attr->value ? attr->value : ""); + + if (!warn) + errors ++; + } + else if (strcmp(program, "-")) + { + if (program[0] == '/') + snprintf(pathprog, sizeof(pathprog), "%s%s", root, program); + else + { + if ((ptr = getenv("CUPS_SERVERBIN")) == NULL) + ptr = CUPS_SERVERBIN; + + if (*ptr == '/' || !*root) + snprintf(pathprog, sizeof(pathprog), "%s%s/filter/%s", root, ptr, + program); + else + snprintf(pathprog, sizeof(pathprog), "%s/%s/filter/%s", root, ptr, + program); + } + + if (stat(pathprog, &fileinfo)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, _(" %s Missing %s file \"%s\"."), + prefix, "cupsPreFilter", pathprog); + + if (!warn) + errors ++; + } + else if (fileinfo.st_uid != 0 || + (fileinfo.st_mode & MODE_WRITE) || + (fileinfo.st_mode & MODE_MASK) != MODE_PROGRAM) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad permissions on %s file \"%s\"."), + prefix, "cupsPreFilter", pathprog); + + if (!warn) + errors ++; + } + else + errors = valid_path("cupsPreFilter", pathprog, errors, verbose, warn); + } + } + +#ifdef __APPLE__ + /* + * APDialogExtension + */ + + for (attr = ppdFindAttr(ppd, "APDialogExtension", NULL); + attr != NULL; + attr = ppdFindNextAttr(ppd, "APDialogExtension", NULL)) + { + if (strcmp(attr->name, "APDialogExtension")) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad spelling of %s - should be %s."), + prefix, attr->name, "APDialogExtension"); + + if (!warn) + errors ++; + } + + snprintf(pathprog, sizeof(pathprog), "%s%s", root, + attr->value ? attr->value : "(null)"); + + if (!attr->value || stat(pathprog, &fileinfo)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, _(" %s Missing %s file \"%s\"."), + prefix, "APDialogExtension", pathprog); + + if (!warn) + errors ++; + } + else if (fileinfo.st_uid != 0 || + (fileinfo.st_mode & MODE_WRITE) || + (fileinfo.st_mode & MODE_MASK) != MODE_DIRECTORY) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad permissions on %s file \"%s\"."), + prefix, "APDialogExtension", pathprog); + + if (!warn) + errors ++; + } + else + errors = valid_path("APDialogExtension", pathprog, errors, verbose, + warn); + } + + /* + * APPrinterIconPath + */ + + if ((attr = ppdFindAttr(ppd, "APPrinterIconPath", NULL)) != NULL) + { + if (strcmp(attr->name, "APPrinterIconPath")) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad spelling of %s - should be %s."), + prefix, attr->name, "APPrinterIconPath"); + + if (!warn) + errors ++; + } + + snprintf(pathprog, sizeof(pathprog), "%s%s", root, + attr->value ? attr->value : "(null)"); + + if (!attr->value || stat(pathprog, &fileinfo)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, _(" %s Missing %s file \"%s\"."), + prefix, "APPrinterIconPath", pathprog); + + if (!warn) + errors ++; + } + else if (fileinfo.st_uid != 0 || + (fileinfo.st_mode & MODE_WRITE) || + (fileinfo.st_mode & MODE_MASK) != MODE_DATAFILE) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad permissions on %s file \"%s\"."), + prefix, "APPrinterIconPath", pathprog); + + if (!warn) + errors ++; + } + else + errors = valid_path("APPrinterIconPath", pathprog, errors, verbose, + warn); + } + + /* + * APPrinterLowInkTool + */ + + if ((attr = ppdFindAttr(ppd, "APPrinterLowInkTool", NULL)) != NULL) + { + if (strcmp(attr->name, "APPrinterLowInkTool")) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad spelling of %s - should be %s."), + prefix, attr->name, "APPrinterLowInkTool"); + + if (!warn) + errors ++; + } + + snprintf(pathprog, sizeof(pathprog), "%s%s", root, + attr->value ? attr->value : "(null)"); + + if (!attr->value || stat(pathprog, &fileinfo)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, _(" %s Missing %s file \"%s\"."), + prefix, "APPrinterLowInkTool", pathprog); + + if (!warn) + errors ++; + } + else if (fileinfo.st_uid != 0 || + (fileinfo.st_mode & MODE_WRITE) || + (fileinfo.st_mode & MODE_MASK) != MODE_DIRECTORY) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad permissions on %s file \"%s\"."), + prefix, "APPrinterLowInkTool", pathprog); + + if (!warn) + errors ++; + } + else + errors = valid_path("APPrinterLowInkTool", pathprog, errors, verbose, + warn); + } + + /* + * APPrinterUtilityPath + */ + + if ((attr = ppdFindAttr(ppd, "APPrinterUtilityPath", NULL)) != NULL) + { + if (strcmp(attr->name, "APPrinterUtilityPath")) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad spelling of %s - should be %s."), + prefix, attr->name, "APPrinterUtilityPath"); + + if (!warn) + errors ++; + } + + snprintf(pathprog, sizeof(pathprog), "%s%s", root, + attr->value ? attr->value : "(null)"); + + if (!attr->value || stat(pathprog, &fileinfo)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, _(" %s Missing %s file \"%s\"."), + prefix, "APPrinterUtilityPath", pathprog); + + if (!warn) + errors ++; + } + else if (fileinfo.st_uid != 0 || + (fileinfo.st_mode & MODE_WRITE) || + (fileinfo.st_mode & MODE_MASK) != MODE_DIRECTORY) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad permissions on %s file \"%s\"."), + prefix, "APPrinterUtilityPath", pathprog); + + if (!warn) + errors ++; + } + else + errors = valid_path("APPrinterUtilityPath", pathprog, errors, verbose, + warn); + } + + /* + * APScanAppBundleID and APScanAppPath + */ + + if ((attr = ppdFindAttr(ppd, "APScanAppPath", NULL)) != NULL) + { + if (strcmp(attr->name, "APScanAppPath")) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad spelling of %s - should be %s."), + prefix, attr->name, "APScanAppPath"); + + if (!warn) + errors ++; + } + + if (!attr->value || stat(attr->value, &fileinfo)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, _(" %s Missing %s file \"%s\"."), + prefix, "APScanAppPath", + attr->value ? attr->value : ""); + + if (!warn) + errors ++; + } + else if (fileinfo.st_uid != 0 || + (fileinfo.st_mode & MODE_WRITE) || + (fileinfo.st_mode & MODE_MASK) != MODE_DIRECTORY) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad permissions on %s file \"%s\"."), + prefix, "APScanAppPath", attr->value); + + if (!warn) + errors ++; + } + else + errors = valid_path("APScanAppPath", attr->value, errors, verbose, + warn); + + if (ppdFindAttr(ppd, "APScanAppBundleID", NULL)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, _(" %s Cannot provide both " + "APScanAppPath and APScanAppBundleID."), + prefix); + + if (!warn) + errors ++; + } + } +#endif /* __APPLE__ */ + + return (errors); +} + + +/* + * 'check_profiles()' - Check ICC color profiles in the PPD file. + */ + +static int /* O - Errors found */ +check_profiles(ppd_file_t *ppd, /* I - PPD file */ + const char *root, /* I - Root directory */ + int errors, /* I - Errors found */ + int verbose, /* I - Verbosity level */ + int warn) /* I - Warnings only? */ +{ + int i; /* Looping var */ + ppd_attr_t *attr; /* PPD attribute */ + const char *ptr; /* Pointer into string */ + const char *prefix; /* WARN/FAIL prefix */ + char filename[1024]; /* Profile filename */ + struct stat fileinfo; /* File information */ + int num_profiles = 0; /* Number of profiles */ + unsigned hash, /* Current hash value */ + hashes[1000]; /* Hash values of profile names */ + const char *specs[1000]; /* Specifiers for profiles */ + + + prefix = warn ? " WARN " : "**FAIL**"; + + for (attr = ppdFindAttr(ppd, "cupsICCProfile", NULL); + attr; + attr = ppdFindNextAttr(ppd, "cupsICCProfile", NULL)) + { + /* + * Check for valid selector... + */ + + for (i = 0, ptr = strchr(attr->spec, '.'); ptr; ptr = strchr(ptr + 1, '.')) + i ++; + + if (!attr->value || i < 2) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad cupsICCProfile %s."), + prefix, attr->spec); + + if (!warn) + errors ++; + + continue; + } + + /* + * Check for valid profile filename... + */ + + if (attr->value[0] == '/') + snprintf(filename, sizeof(filename), "%s%s", root, attr->value); + else + { + if ((ptr = getenv("CUPS_DATADIR")) == NULL) + ptr = CUPS_DATADIR; + + if (*ptr == '/' || !*root) + snprintf(filename, sizeof(filename), "%s%s/profiles/%s", root, ptr, + attr->value); + else + snprintf(filename, sizeof(filename), "%s/%s/profiles/%s", root, ptr, + attr->value); + } + + if (stat(filename, &fileinfo)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, _(" %s Missing %s file \"%s\"."), + prefix, "cupsICCProfile", filename); + + if (!warn) + errors ++; + } + else if (fileinfo.st_uid != 0 || + (fileinfo.st_mode & MODE_WRITE) || + (fileinfo.st_mode & MODE_MASK) != MODE_DATAFILE) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad permissions on %s file \"%s\"."), + prefix, "cupsICCProfile", filename); + + if (!warn) + errors ++; + } + else + errors = valid_path("cupsICCProfile", filename, errors, verbose, warn); + + /* + * Check for hash collisions... + */ + + hash = _ppdHashName(attr->spec); + + if (num_profiles > 0) + { + for (i = 0; i < num_profiles; i ++) + if (hashes[i] == hash) + break; + + if (i < num_profiles) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s cupsICCProfile %s hash value " + "collides with %s."), prefix, attr->spec, + specs[i]); + + if (!warn) + errors ++; + } + } + + /* + * Remember up to 1000 profiles... + */ + + if (num_profiles < 1000) + { + hashes[num_profiles] = hash; + specs[num_profiles] = attr->spec; + num_profiles ++; + } + } + + return (errors); +} + + +/* + * 'check_sizes()' - Check media sizes in the PPD file. + */ + +static int /* O - Errors found */ +check_sizes(ppd_file_t *ppd, /* I - PPD file */ + int errors, /* I - Errors found */ + int verbose, /* I - Verbosity level */ + int warn) /* I - Warnings only? */ +{ + int i; /* Looping var */ + ppd_size_t *size; /* Current size */ + int width, /* Custom width */ + length; /* Custom length */ + const char *prefix; /* WARN/FAIL prefix */ + ppd_option_t *page_size, /* PageSize option */ + *page_region; /* PageRegion option */ + _pwg_media_t *pwg_media; /* PWG media */ + char buf[1024]; /* PapeSize name that is supposed to be */ + const char *ptr; /* Pointer into string */ + int width_2540ths, /* PageSize width in 2540ths */ + length_2540ths; /* PageSize length in 2540ths */ + int is_ok; /* Flag for PageSize name verification */ + double width_tmp, /* Width after rounded up */ + length_tmp, /* Length after rounded up */ + width_inch, /* Width in inches */ + length_inch, /* Length in inches */ + width_mm, /* Width in millimeters */ + length_mm; /* Length in millimeters */ + + + prefix = warn ? " WARN " : "**FAIL**"; + + if ((page_size = ppdFindOption(ppd, "PageSize")) == NULL && warn != 2) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Missing REQUIRED PageSize option.\n" + " REF: Page 99, section 5.14."), + prefix); + + if (!warn) + errors ++; + } + + if ((page_region = ppdFindOption(ppd, "PageRegion")) == NULL && warn != 2) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Missing REQUIRED PageRegion option.\n" + " REF: Page 100, section 5.14."), + prefix); + + if (!warn) + errors ++; + } + + for (i = ppd->num_sizes, size = ppd->sizes; i > 0; i --, size ++) + { + /* + * Check that the size name is standard... + */ + + if (!strcmp(size->name, "Custom")) + { + /* + * Skip custom page size... + */ + + continue; + } + else if (warn != 2 && size->name[0] == 'w' && + sscanf(size->name, "w%dh%d", &width, &length) == 2) + { + /* + * Validate device-specific size wNNNhNNN should have proper width and + * length... + */ + + if (fabs(width - size->width) >= 1.0 || + fabs(length - size->length) >= 1.0) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Size \"%s\" has unexpected dimensions " + "(%gx%g)."), + prefix, size->name, size->width, size->length); + + if (!warn) + errors ++; + } + } + + /* + * Verify that the size is defined for both PageSize and PageRegion... + */ + + if (warn != 2 && !ppdFindChoice(page_size, size->name)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Size \"%s\" defined for %s but not for " + "%s."), + prefix, size->name, "PageRegion", "PageSize"); + + if (!warn) + errors ++; + } + else if (warn != 2 && !ppdFindChoice(page_region, size->name)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Size \"%s\" defined for %s but not for " + "%s."), + prefix, size->name, "PageSize", "PageRegion"); + + if (!warn) + errors ++; + } + + /* + * Verify that the size name is Adobe standard name if it's a standard size + * and the dimentional name if it's not a standard size. Suffix should be + * .Fullbleed, etc., or numeric, e.g., Letter, Letter.Fullbleed, + * Letter.Transverse, Letter1, Letter2, 4x8, 55x91mm, 55x91mm.Fullbleed, etc. + */ + + if (warn != 0) + { + is_ok = 1; + width_2540ths = (size->length > size->width) ? + _PWG_FROMPTS(size->width) : + _PWG_FROMPTS(size->length); + length_2540ths = (size->length > size->width) ? + _PWG_FROMPTS(size->length) : + _PWG_FROMPTS(size->width); + pwg_media = _pwgMediaForSize(width_2540ths, length_2540ths); + + if (pwg_media && pwg_media->ppd) + { + size_t ppdlen = strlen(pwg_media->ppd); + /* Length of standard PPD name */ + + strlcpy(buf, pwg_media->ppd, sizeof(buf)); + + if (size->left == 0 && size->bottom == 0 && + size->right == size->width && size->top == size->length) + { + snprintf(buf, sizeof(buf), "%s.Fullbleed", pwg_media->ppd); + if (_cups_strcasecmp(size->name, buf)) + { + /* + * Allow an additional qualifier such as ".WithTab"... + */ + + size_t buflen = strlen(buf);/* Length of full bleed name */ + + if (_cups_strncasecmp(size->name, buf, buflen) || + size->name[buflen] != '.') + is_ok = 0; + } + } + else if (strcmp(size->name, buf) && size->width > size->length) + { + if (!strcmp(pwg_media->ppd, "DoublePostcardRotated")) + strlcpy(buf, "DoublePostcard", sizeof(buf)); + else + snprintf(buf, sizeof(buf), "%sRotated", pwg_media->ppd); + + if (strcmp(size->name, buf)) + { + snprintf(buf, sizeof(buf), "%s.Transverse", pwg_media->ppd); + if (strcmp(size->name, buf)) + is_ok = 0; + } + } + else if (!strncmp(size->name, pwg_media->ppd, ppdlen)) + { + /* + * Check for a proper qualifier (number, "Small", or .something)... + */ + + ptr = size->name + ppdlen; + + if (isdigit(*ptr & 255)) + { + for (ptr ++; *ptr; ptr ++) + { + if (!isdigit(*ptr & 255)) + { + is_ok = 0; + break; + } + } + } + else if (*ptr != '.' && *ptr && strcmp(ptr, "Small")) + is_ok = 0; + } + else + { + /* + * Check for EnvSizeName as well... + */ + + snprintf(buf, sizeof(buf), "Env%s", pwg_media->ppd); + + if (strcmp(size->name, buf)) + is_ok = 0; + } + + if (!is_ok) + _cupsLangPrintf(stdout, + _(" %s Size \"%s\" should be the Adobe " + "standard name \"%s\"."), + prefix, size->name, buf); + } + else + { + width_tmp = (fabs(size->width - ceil(size->width)) < 0.1) ? + ceil(size->width) : size->width; + length_tmp = (fabs(size->length - ceil(size->length)) < 0.1) ? + ceil(size->length) : size->length; + + if (fmod(width_tmp, 18.0) == 0.0 || fmod(length_tmp, 18.0) == 0.0) + { + width_inch = width_tmp / 72.0; + length_inch = length_tmp / 72.0; + + snprintf(buf, sizeof(buf), "%gx%g", width_inch, length_inch); + } + else + { + width_mm = size->width / 72.0 * 25.4; + length_mm = size->length / 72.0 * 25.4; + + snprintf(buf, sizeof(buf), "%.0fx%.0fmm", width_mm, length_mm); + } + + if (size->left == 0 && size->bottom == 0 && + size->right == size->width && size->top == size->length) + strlcat(buf, ".Fullbleed", sizeof(buf)); + else if (size->width > size->length) + strlcat(buf, ".Transverse", sizeof(buf)); + + if (_cups_strcasecmp(size->name, buf)) + { + size_t buflen = strlen(buf); /* Length of proposed name */ + + if (_cups_strncasecmp(size->name, buf, buflen) || + strcmp(size->name + buflen, "in")) + _cupsLangPrintf(stdout, + _(" %s Size \"%s\" should be \"%s\"."), + prefix, size->name, buf); + } + } + } + } + + return (errors); +} + + +/* + * 'check_translations()' - Check translations in the PPD file. + */ + +static int /* O - Errors found */ +check_translations(ppd_file_t *ppd, /* I - PPD file */ + int errors, /* I - Errors found */ + int verbose, /* I - Verbosity level */ + int warn) /* I - Warnings only? */ +{ + int j; /* Looping var */ + ppd_attr_t *attr; /* PPD attribute */ + cups_array_t *languages; /* Array of languages */ + int langlen; /* Length of language */ + char *language, /* Current language */ + keyword[PPD_MAX_NAME], /* Localization keyword (full) */ + llkeyword[PPD_MAX_NAME],/* Localization keyword (base) */ + ckeyword[PPD_MAX_NAME], /* Custom option keyword (full) */ + cllkeyword[PPD_MAX_NAME]; + /* Custom option keyword (base) */ + ppd_option_t *option; /* Standard UI option */ + ppd_coption_t *coption; /* Custom option */ + ppd_cparam_t *cparam; /* Custom parameter */ + char ll[3]; /* Base language */ + const char *prefix; /* WARN/FAIL prefix */ + const char *text; /* Pointer into UI text */ + + + prefix = warn ? " WARN " : "**FAIL**"; + + if ((languages = _ppdGetLanguages(ppd)) != NULL) + { + /* + * This file contains localizations, check them... + */ + + for (language = (char *)cupsArrayFirst(languages); + language; + language = (char *)cupsArrayNext(languages)) + { + langlen = (int)strlen(language); + if (langlen != 2 && langlen != 5) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad language \"%s\"."), + prefix, language); + + if (!warn) + errors ++; + + continue; + } + + if (!strcmp(language, "en")) + continue; + + strlcpy(ll, language, sizeof(ll)); + + /* + * Loop through all options and choices... + */ + + for (option = ppdFirstOption(ppd); + option; + option = ppdNextOption(ppd)) + { + if (!strcmp(option->keyword, "PageRegion")) + continue; + + snprintf(keyword, sizeof(keyword), "%s.Translation", language); + snprintf(llkeyword, sizeof(llkeyword), "%s.Translation", ll); + + if ((attr = ppdFindAttr(ppd, keyword, option->keyword)) == NULL && + (attr = ppdFindAttr(ppd, llkeyword, option->keyword)) == NULL) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Missing \"%s\" translation " + "string for option %s."), + prefix, language, option->keyword); + + if (!warn) + errors ++; + } + else if (!valid_utf8(attr->text)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad UTF-8 \"%s\" translation " + "string for option %s."), + prefix, language, option->keyword); + + if (!warn) + errors ++; + } + + snprintf(keyword, sizeof(keyword), "%s.%s", language, + option->keyword); + snprintf(llkeyword, sizeof(llkeyword), "%s.%s", ll, + option->keyword); + + for (j = 0; j < option->num_choices; j ++) + { + /* + * First see if this choice is a number; if so, don't require + * translation... + */ + + for (text = option->choices[j].text; *text; text ++) + if (!strchr("0123456789-+.", *text)) + break; + + if (!*text) + continue; + + /* + * Check custom choices differently... + */ + + if (!_cups_strcasecmp(option->choices[j].choice, "Custom") && + (coption = ppdFindCustomOption(ppd, + option->keyword)) != NULL) + { + snprintf(ckeyword, sizeof(ckeyword), "%s.Custom%s", + language, option->keyword); + + if ((attr = ppdFindAttr(ppd, ckeyword, "True")) != NULL && + !valid_utf8(attr->text)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad UTF-8 \"%s\" " + "translation string for option %s, " + "choice %s."), + prefix, language, + ckeyword + 1 + strlen(language), + "True"); + + if (!warn) + errors ++; + } + + if (_cups_strcasecmp(option->keyword, "PageSize")) + { + for (cparam = (ppd_cparam_t *)cupsArrayFirst(coption->params); + cparam; + cparam = (ppd_cparam_t *)cupsArrayNext(coption->params)) + { + snprintf(ckeyword, sizeof(ckeyword), "%s.ParamCustom%s", + language, option->keyword); + snprintf(cllkeyword, sizeof(cllkeyword), "%s.ParamCustom%s", + ll, option->keyword); + + if ((attr = ppdFindAttr(ppd, ckeyword, + cparam->name)) == NULL && + (attr = ppdFindAttr(ppd, cllkeyword, + cparam->name)) == NULL) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Missing \"%s\" " + "translation string for option %s, " + "choice %s."), + prefix, language, + ckeyword + 1 + strlen(language), + cparam->name); + + if (!warn) + errors ++; + } + else if (!valid_utf8(attr->text)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad UTF-8 \"%s\" " + "translation string for option %s, " + "choice %s."), + prefix, language, + ckeyword + 1 + strlen(language), + cparam->name); + + if (!warn) + errors ++; + } + } + } + } + else if ((attr = ppdFindAttr(ppd, keyword, + option->choices[j].choice)) == NULL && + (attr = ppdFindAttr(ppd, llkeyword, + option->choices[j].choice)) == NULL) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Missing \"%s\" " + "translation string for option %s, " + "choice %s."), + prefix, language, option->keyword, + option->choices[j].choice); + + if (!warn) + errors ++; + } + else if (!valid_utf8(attr->text)) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s Bad UTF-8 \"%s\" " + "translation string for option %s, " + "choice %s."), + prefix, language, option->keyword, + option->choices[j].choice); + + if (!warn) + errors ++; + } + } + } + } + + /* + * Verify that we have the base language for each localized one... + */ + + for (language = (char *)cupsArrayFirst(languages); + language; + language = (char *)cupsArrayNext(languages)) + if (language[2]) + { + /* + * Lookup the base language... + */ + + cupsArraySave(languages); + + strlcpy(ll, language, sizeof(ll)); + + if (!cupsArrayFind(languages, ll) && + strcmp(ll, "zh") && strcmp(ll, "en")) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s No base translation \"%s\" " + "is included in file."), prefix, ll); + + if (!warn) + errors ++; + } + + cupsArrayRestore(languages); + } + + /* + * Free memory used for the languages... + */ + + _ppdFreeLanguages(languages); + } + + return (errors); +} + + +/* + * 'show_conflicts()' - Show option conflicts in a PPD file. + */ + +static void +show_conflicts(ppd_file_t *ppd, /* I - PPD to check */ + const char *prefix) /* I - Prefix string */ +{ + int i, j; /* Looping variables */ + ppd_const_t *c; /* Current constraint */ + ppd_option_t *o1, *o2; /* Options */ + ppd_choice_t *c1, *c2; /* Choices */ + + + /* + * Loop through all of the UI constraints and report any options + * that conflict... + */ + + for (i = ppd->num_consts, c = ppd->consts; i > 0; i --, c ++) + { + /* + * Grab pointers to the first option... + */ + + o1 = ppdFindOption(ppd, c->option1); + + if (o1 == NULL) + continue; + else if (c->choice1[0] != '\0') + { + /* + * This constraint maps to a specific choice. + */ + + c1 = ppdFindChoice(o1, c->choice1); + } + else + { + /* + * This constraint applies to any choice for this option. + */ + + for (j = o1->num_choices, c1 = o1->choices; j > 0; j --, c1 ++) + if (c1->marked) + break; + + if (j == 0 || + !_cups_strcasecmp(c1->choice, "None") || + !_cups_strcasecmp(c1->choice, "Off") || + !_cups_strcasecmp(c1->choice, "False")) + c1 = NULL; + } + + /* + * Grab pointers to the second option... + */ + + o2 = ppdFindOption(ppd, c->option2); + + if (o2 == NULL) + continue; + else if (c->choice2[0] != '\0') + { + /* + * This constraint maps to a specific choice. + */ + + c2 = ppdFindChoice(o2, c->choice2); + } + else + { + /* + * This constraint applies to any choice for this option. + */ + + for (j = o2->num_choices, c2 = o2->choices; j > 0; j --, c2 ++) + if (c2->marked) + break; + + if (j == 0 || + !_cups_strcasecmp(c2->choice, "None") || + !_cups_strcasecmp(c2->choice, "Off") || + !_cups_strcasecmp(c2->choice, "False")) + c2 = NULL; + } + + /* + * If both options are marked then there is a conflict... + */ + + if (c1 != NULL && c1->marked && c2 != NULL && c2->marked) + _cupsLangPrintf(stdout, + _(" %s \"%s %s\" conflicts with \"%s %s\"\n" + " (constraint=\"%s %s %s %s\")."), + prefix, o1->keyword, c1->choice, o2->keyword, c2->choice, + c->option1, c->choice1, c->option2, c->choice2); + } +} + + +/* + * 'test_raster()' - Test PostScript commands for raster printers. + */ + +static int /* O - 1 on success, 0 on failure */ +test_raster(ppd_file_t *ppd, /* I - PPD file */ + int verbose) /* I - Verbosity */ +{ + cups_page_header2_t header; /* Page header */ + + + ppdMarkDefaults(ppd); + if (cupsRasterInterpretPPD(&header, ppd, 0, NULL, 0)) + { + if (!verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" **FAIL** Default option code cannot be " + "interpreted: %s"), cupsRasterErrorString()); + + return (0); + } + + /* + * Try a test of custom page size code, if available... + */ + + if (!ppdPageSize(ppd, "Custom.612x792")) + return (1); + + ppdMarkOption(ppd, "PageSize", "Custom.612x792"); + + if (cupsRasterInterpretPPD(&header, ppd, 0, NULL, 0)) + { + if (!verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" **FAIL** Default option code cannot be " + "interpreted: %s"), cupsRasterErrorString()); + + return (0); + } + + return (1); +} + + +/* + * 'usage()' - Show program usage. + */ + +static void +usage(void) +{ + _cupsLangPuts(stdout, _("Usage: cupstestppd [options] filename1.ppd[.gz] " + "[... filenameN.ppd[.gz]]")); + _cupsLangPuts(stdout, _(" program | cupstestppd [options] -")); + _cupsLangPuts(stdout, ""); + _cupsLangPuts(stdout, _("Options:")); + _cupsLangPuts(stdout, ""); + _cupsLangPuts(stdout, _(" -I {filename,filters,none,profiles}")); + _cupsLangPuts(stdout, _(" Ignore specific warnings.")); + _cupsLangPuts(stdout, _(" -R root-directory Set alternate root.")); + _cupsLangPuts(stdout, _(" -W {all,none,constraints,defaults,duplex," + "filters,profiles,sizes,translations}")); + _cupsLangPuts(stdout, _(" Issue warnings instead of " + "errors.")); + _cupsLangPuts(stdout, _(" -q Run silently.")); + _cupsLangPuts(stdout, _(" -r Use 'relaxed' open mode.")); + _cupsLangPuts(stdout, _(" -v Be slightly verbose.")); + _cupsLangPuts(stdout, _(" -vv Be very verbose.")); + + exit(ERROR_USAGE); +} + + +/* + * 'valid_path()' - Check whether a path has the correct capitalization. + */ + +static int /* O - Errors found */ +valid_path(const char *keyword, /* I - Keyword using path */ + const char *path, /* I - Path to check */ + int errors, /* I - Errors found */ + int verbose, /* I - Verbosity level */ + int warn) /* I - Warnings only? */ +{ + cups_dir_t *dir; /* Current directory */ + cups_dentry_t *dentry; /* Current directory entry */ + char temp[1024], /* Temporary path */ + *ptr; /* Pointer into temporary path */ + const char *prefix; /* WARN/FAIL prefix */ + + + prefix = warn ? " WARN " : "**FAIL**"; + + /* + * Loop over the components of the path, checking that the entry exists with + * the same capitalization... + */ + + strlcpy(temp, path, sizeof(temp)); + + while ((ptr = strrchr(temp, '/')) != NULL) + { + /* + * Chop off the trailing component so temp == dirname and ptr == basename. + */ + + *ptr++ = '\0'; + + /* + * Try opening the directory containing the base name... + */ + + if (temp[0]) + dir = cupsDirOpen(temp); + else + dir = cupsDirOpen("/"); + + if (!dir) + dentry = NULL; + else + { + while ((dentry = cupsDirRead(dir)) != NULL) + { + if (!strcmp(dentry->filename, ptr)) + break; + } + + cupsDirClose(dir); + } + + /* + * Display an error if the filename doesn't exist with the same + * capitalization... + */ + + if (!dentry) + { + if (!warn && !errors && !verbose) + _cupsLangPuts(stdout, _(" FAIL")); + + if (verbose >= 0) + _cupsLangPrintf(stdout, + _(" %s %s file \"%s\" has the wrong " + "capitalization."), prefix, keyword, path); + + if (!warn) + errors ++; + + break; + } + } + + return (errors); +} + + +/* + * 'valid_utf8()' - Check whether a string contains valid UTF-8 text. + */ + +static int /* O - 1 if valid, 0 if not */ +valid_utf8(const char *s) /* I - String to check */ +{ + while (*s) + { + if (*s & 0x80) + { + /* + * Check for valid UTF-8 sequence... + */ + + if ((*s & 0xc0) == 0x80) + return (0); /* Illegal suffix byte */ + else if ((*s & 0xe0) == 0xc0) + { + /* + * 2-byte sequence... + */ + + s ++; + + if ((*s & 0xc0) != 0x80) + return (0); /* Missing suffix byte */ + } + else if ((*s & 0xf0) == 0xe0) + { + /* + * 3-byte sequence... + */ + + s ++; + + if ((*s & 0xc0) != 0x80) + return (0); /* Missing suffix byte */ + + s ++; + + if ((*s & 0xc0) != 0x80) + return (0); /* Missing suffix byte */ + } + else if ((*s & 0xf8) == 0xf0) + { + /* + * 4-byte sequence... + */ + + s ++; + + if ((*s & 0xc0) != 0x80) + return (0); /* Missing suffix byte */ + + s ++; + + if ((*s & 0xc0) != 0x80) + return (0); /* Missing suffix byte */ + + s ++; + + if ((*s & 0xc0) != 0x80) + return (0); /* Missing suffix byte */ + } + else + return (0); /* Bad sequence */ + } + + s ++; + } + + return (1); +} + + +/* + * End of "$Id$". + */ diff --git a/systemv/lp.c b/systemv/lp.c new file mode 100644 index 0000000000..98a7d3980b --- /dev/null +++ b/systemv/lp.c @@ -0,0 +1,723 @@ +/* + * "$Id$" + * + * "lp" command for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Parse options and send files for printing. + * restart_job() - Restart a job. + * set_job_attrs() - Set job attributes. + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * Local functions. + */ + +int restart_job(const char *command, int job_id); +int set_job_attrs(const char *command, int job_id, int num_options, + cups_option_t *options); + + +/* + * 'main()' - Parse options and send files for printing. + */ + +int +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i, j; /* Looping vars */ + int job_id; /* Job ID */ + char *printer, /* Printer name */ + *instance, /* Instance name */ + *val, /* Option value */ + *title; /* Job title */ + int priority; /* Job priority (1-100) */ + int num_copies; /* Number of copies per file */ + int num_files; /* Number of files to print */ + const char *files[1000]; /* Files to print */ + cups_dest_t *dest; /* Selected destination */ + int num_options; /* Number of options */ + cups_option_t *options; /* Options */ + int end_options; /* No more options? */ + int silent; /* Silent or verbose output? */ + char buffer[8192]; /* Copy buffer */ + + +#ifdef __sun + /* + * Solaris does some rather strange things to re-queue remote print + * jobs. On bootup, the "lp" command is run as "printd" to re-spool + * any remote jobs in /var/spool/print. Since CUPS doesn't need this + * nonsense, we just need to add the necessary check here to prevent + * lp from causing boot problems... + */ + + if ((val = strrchr(argv[0], '/')) != NULL) + val ++; + else + val = argv[0]; + + if (!strcmp(val, "printd")) + return (0); +#endif /* __sun */ + + _cupsSetLocale(argv); + + silent = 0; + printer = NULL; + dest = NULL; + num_options = 0; + options = NULL; + num_files = 0; + title = NULL; + job_id = 0; + end_options = 0; + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-' && argv[i][1] && !end_options) + switch (argv[i][1]) + { + case 'E' : /* Encrypt */ +#ifdef HAVE_SSL + cupsSetEncryption(HTTP_ENCRYPT_REQUIRED); +#else + _cupsLangPrintf(stderr, _("%s: Sorry, no encryption support."), + argv[0]); +#endif /* HAVE_SSL */ + break; + + case 'U' : /* Username */ + if (argv[i][2] != '\0') + cupsSetUser(argv[i] + 2); + else + { + i ++; + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected username after \"-U\" " + "option."), argv[0]); + return (1); + } + + cupsSetUser(argv[i]); + } + break; + + case 'c' : /* Copy to spool dir (always enabled) */ + break; + + case 'd' : /* Destination printer or class */ + if (argv[i][2] != '\0') + printer = argv[i] + 2; + else + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected destination after " + "\"-d\" option."), argv[0]); + return (1); + } + + printer = argv[i]; + } + + if ((instance = strrchr(printer, '/')) != NULL) + *instance++ = '\0'; + + if ((dest = cupsGetNamedDest(NULL, printer, instance)) != NULL) + { + for (j = 0; j < dest->num_options; j ++) + if (cupsGetOption(dest->options[j].name, num_options, + options) == NULL) + num_options = cupsAddOption(dest->options[j].name, + dest->options[j].value, + num_options, &options); + } + break; + + case 'f' : /* Form */ + if (!argv[i][2]) + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected form after \"-f\" " + "option."), + argv[0]); + return (1); + } + } + + _cupsLangPrintf(stderr, _("%s: Warning - form option ignored."), + argv[0]); + break; + + case 'h' : /* Destination host */ + if (argv[i][2] != '\0') + cupsSetServer(argv[i] + 2); + else + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected hostname after " + "\"-h\" option."), argv[0]); + return (1); + } + + cupsSetServer(argv[i]); + } + break; + + case 'i' : /* Change job */ + if (argv[i][2]) + val = argv[i] + 2; + else + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Expected job ID after \"-i\" option."), + argv[0]); + return (1); + } + + val = argv[i]; + } + + if (num_files > 0) + { + _cupsLangPrintf(stderr, + _("%s: Error - cannot print files and alter " + "jobs simultaneously."), argv[0]); + return (1); + } + + if (strrchr(val, '-') != NULL) + job_id = atoi(strrchr(val, '-') + 1); + else + job_id = atoi(val); + + if (job_id < 0) + { + _cupsLangPrintf(stderr, _("%s: Error - bad job ID."), argv[0]); + break; + } + break; + + case 'm' : /* Send email when job is done */ +#ifdef __sun + case 'p' : /* Notify on completion */ +#endif /* __sun */ + case 'w' : /* Write to console or email */ + { + char email[1024]; /* EMail address */ + + + snprintf(email, sizeof(email), "mailto:%s@%s", cupsUser(), + httpGetHostname(NULL, buffer, sizeof(buffer))); + num_options = cupsAddOption("notify-recipient-uri", email, + num_options, &options); + } + + silent = 1; + break; + + case 'n' : /* Number of copies */ + if (argv[i][2] != '\0') + num_copies = atoi(argv[i] + 2); + else + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected copies after " + "\"-n\" option."), argv[0]); + return (1); + } + + num_copies = atoi(argv[i]); + } + + sprintf(buffer, "%d", num_copies); + num_options = cupsAddOption("copies", buffer, num_options, + &options); + break; + + case 'o' : /* Option */ + if (argv[i][2] != '\0') + num_options = cupsParseOptions(argv[i] + 2, num_options, + &options); + else + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected option=value after " + "\"-o\" option."), argv[0]); + return (1); + } + + num_options = cupsParseOptions(argv[i], num_options, &options); + } + break; + +#ifndef __sun + case 'p' : /* Queue priority */ +#endif /* !__sun */ + case 'q' : /* Queue priority */ + if (argv[i][2] != '\0') + priority = atoi(argv[i] + 2); + else + { + if ((i + 1) >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected priority after " + "\"-%c\" option."), argv[0], argv[i][1]); + return (1); + } + + i ++; + + priority = atoi(argv[i]); + } + + /* + * For 100% Solaris compatibility, need to add: + * + * priority = 99 * (39 - priority) / 39 + 1; + * + * However, to keep CUPS lp the same across all platforms + * we will break compatibility this far... + */ + + if (priority < 1 || priority > 100) + { + _cupsLangPrintf(stderr, + _("%s: Error - priority must be between 1 and " + "100."), argv[0]); + return (1); + } + + sprintf(buffer, "%d", priority); + num_options = cupsAddOption("job-priority", buffer, num_options, + &options); + break; + + case 's' : /* Silent */ + silent = 1; + break; + + case 't' : /* Title */ + if (argv[i][2] != '\0') + title = argv[i] + 2; + else + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected title after " + "\"-t\" option."), argv[0]); + return (1); + } + + title = argv[i]; + } + break; + + case 'y' : /* mode-list */ + if (!argv[i][2]) + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected mode list after " + "\"-y\" option."), argv[0]); + return (1); + } + } + + _cupsLangPrintf(stderr, + _("%s: Warning - mode option ignored."), argv[0]); + break; + + case 'H' : /* Hold job */ + if (argv[i][2]) + val = argv[i] + 2; + else + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected hold name after " + "\"-H\" option."), argv[0]); + return (1); + } + + val = argv[i]; + } + + if (!strcmp(val, "hold")) + num_options = cupsAddOption("job-hold-until", "indefinite", + num_options, &options); + else if (!strcmp(val, "resume") || + !strcmp(val, "release")) + num_options = cupsAddOption("job-hold-until", "no-hold", + num_options, &options); + else if (!strcmp(val, "immediate")) + { + num_options = cupsAddOption("job-hold-until", "no-hold", + num_options, &options); + num_options = cupsAddOption("job-priority", "100", + num_options, &options); + } + else if (!strcmp(val, "restart")) + { + if (job_id < 1) + { + _cupsLangPrintf(stderr, + _("%s: Need job ID (\"-i jobid\") before " + "\"-H restart\"."), argv[0]); + return (1); + } + + if (restart_job(argv[0], job_id)) + return (1); + } + else + num_options = cupsAddOption("job-hold-until", val, + num_options, &options); + break; + + case 'P' : /* Page list */ + if (argv[i][2]) + val = argv[i] + 2; + else + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected page list after " + "\"-P\" option."), argv[0]); + return (1); + } + + val = argv[i]; + } + + num_options = cupsAddOption("page-ranges", val, num_options, + &options); + break; + + case 'S' : /* character set */ + if (!argv[i][2]) + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected character set after " + "\"-S\" option."), argv[0]); + return (1); + } + } + + _cupsLangPrintf(stderr, + _("%s: Warning - character set option ignored."), + argv[0]); + break; + + case 'T' : /* Content-Type */ + if (!argv[i][2]) + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected content type after " + "\"-T\" option."), argv[0]); + return (1); + } + } + + _cupsLangPrintf(stderr, + _("%s: Warning - content type option ignored."), + argv[0]); + break; + + case '-' : /* Stop processing options */ + end_options = 1; + break; + + default : + _cupsLangPrintf(stderr, _("%s: Error - unknown option \"%c\"."), + argv[0], argv[i][1]); + return (1); + } + else if (!strcmp(argv[i], "-")) + { + if (num_files || job_id) + { + _cupsLangPrintf(stderr, + _("%s: Error - cannot print from stdin if files or a " + "job ID are provided."), argv[0]); + return (1); + } + + break; + } + else if (num_files < 1000 && job_id == 0) + { + /* + * Print a file... + */ + + if (access(argv[i], R_OK) != 0) + { + _cupsLangPrintf(stderr, _("%s: Error - unable to access \"%s\" - %s"), + argv[0], argv[i], strerror(errno)); + return (1); + } + + files[num_files] = argv[i]; + num_files ++; + + if (title == NULL) + { + if ((title = strrchr(argv[i], '/')) != NULL) + title ++; + else + title = argv[i]; + } + } + else + _cupsLangPrintf(stderr, _("%s: Error - too many files - \"%s\"."), + argv[0], argv[i]); + + /* + * See if we are altering an existing job... + */ + + if (job_id) + return (set_job_attrs(argv[0], job_id, num_options, options)); + + /* + * See if we have any files to print; if not, print from stdin... + */ + + if (printer == NULL) + { + if ((dest = cupsGetNamedDest(NULL, NULL, NULL)) != NULL) + { + printer = dest->name; + + for (j = 0; j < dest->num_options; j ++) + if (cupsGetOption(dest->options[j].name, num_options, options) == NULL) + num_options = cupsAddOption(dest->options[j].name, + dest->options[j].value, + num_options, &options); + } + } + + if (printer == NULL) + { + val = NULL; + + if ((printer = getenv("LPDEST")) == NULL) + { + if ((printer = getenv("PRINTER")) != NULL) + { + if (!strcmp(printer, "lp")) + printer = NULL; + else + val = "PRINTER"; + } + } + else + val = "LPDEST"; + + if (printer && !cupsGetNamedDest(NULL, printer, NULL)) + _cupsLangPrintf(stderr, + _("%s: Error - %s environment variable names " + "non-existent destination \"%s\"."), argv[0], val, + printer); + else if (cupsLastError() == IPP_NOT_FOUND) + _cupsLangPrintf(stderr, + _("%s: Error - no default destination available."), + argv[0]); + else + _cupsLangPrintf(stderr, _("%s: Error - scheduler not responding."), + argv[0]); + + return (1); + } + + if (num_files > 0) + job_id = cupsPrintFiles(printer, num_files, files, title, num_options, options); + else if ((job_id = cupsCreateJob(CUPS_HTTP_DEFAULT, printer, + title ? title : "(stdin)", + num_options, options)) > 0) + { + http_status_t status; /* Write status */ + const char *format; /* Document format */ + ssize_t bytes; /* Bytes read */ + + + if (cupsGetOption("raw", num_options, options)) + format = CUPS_FORMAT_RAW; + else if ((format = cupsGetOption("document-format", num_options, + options)) == NULL) + format = CUPS_FORMAT_AUTO; + + status = cupsStartDocument(CUPS_HTTP_DEFAULT, printer, job_id, NULL, + format, 1); + + while (status == HTTP_CONTINUE && + (bytes = read(0, buffer, sizeof(buffer))) > 0) + status = cupsWriteRequestData(CUPS_HTTP_DEFAULT, buffer, bytes); + + if (status != HTTP_CONTINUE) + { + _cupsLangPrintf(stderr, _("%s: Error - unable to queue from stdin - %s."), + argv[0], httpStatus(status)); + return (1); + } + + if (cupsFinishDocument(CUPS_HTTP_DEFAULT, printer) != IPP_OK) + job_id = 0; + } + + if (job_id < 1) + { + _cupsLangPrintf(stderr, "%s: %s", argv[0], cupsLastErrorString()); + return (1); + } + else if (!silent) + _cupsLangPrintf(stdout, _("request id is %s-%d (%d file(s))"), + printer, job_id, num_files); + + return (0); +} + + +/* + * 'restart_job()' - Restart a job. + */ + +int /* O - Exit status */ +restart_job(const char *command, /* I - Command name */ + int job_id) /* I - Job ID */ +{ + ipp_t *request; /* IPP request */ + char uri[HTTP_MAX_URI]; /* URI for job */ + + + request = ippNewRequest(IPP_RESTART_JOB); + + sprintf(uri, "ipp://localhost/jobs/%d", job_id); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "job-uri", NULL, uri); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, cupsUser()); + + ippDelete(cupsDoRequest(CUPS_HTTP_DEFAULT, request, "/jobs")); + + if (cupsLastError() > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, "%s: %s", command, cupsLastErrorString()); + return (1); + } + + return (0); +} + + +/* + * 'set_job_attrs()' - Set job attributes. + */ + +int /* O - Exit status */ +set_job_attrs(const char *command, /* I - Command name */ + int job_id, /* I - Job ID */ + int num_options,/* I - Number of options */ + cups_option_t *options) /* I - Options */ +{ + ipp_t *request; /* IPP request */ + char uri[HTTP_MAX_URI]; /* URI for job */ + + + if (num_options == 0) + return (0); + + request = ippNewRequest(IPP_SET_JOB_ATTRIBUTES); + + sprintf(uri, "ipp://localhost/jobs/%d", job_id); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "job-uri", NULL, uri); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, cupsUser()); + + cupsEncodeOptions(request, num_options, options); + + ippDelete(cupsDoRequest(CUPS_HTTP_DEFAULT, request, "/jobs")); + + if (cupsLastError() > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, "%s: %s", command, cupsLastErrorString()); + return (1); + } + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/systemv/lpadmin.c b/systemv/lpadmin.c new file mode 100644 index 0000000000..d954e75fb8 --- /dev/null +++ b/systemv/lpadmin.c @@ -0,0 +1,1518 @@ +/* + * "$Id$" + * + * "lpadmin" command for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Parse options and configure the scheduler. + * add_printer_to_class() - Add a printer to a class. + * default_printer() - Set the default printing destination. + * delete_printer() - Delete a printer from the system. + * delete_printer_from_class() - Delete a printer from a class. + * delete_printer_option() - Delete a printer option. + * enable_printer() - Enable a printer. + * get_printer_type() - Determine the printer type and URI. + * set_printer_options() - Set the printer options and/or file. + * validate_name() - Make sure the printer name only contains + * valid chars. + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * Local functions... + */ + +static int add_printer_to_class(http_t *http, char *printer, char *pclass); +static int default_printer(http_t *http, char *printer); +static int delete_printer(http_t *http, char *printer); +static int delete_printer_from_class(http_t *http, char *printer, + char *pclass); +static int delete_printer_option(http_t *http, char *printer, + char *option); +static int enable_printer(http_t *http, char *printer); +static cups_ptype_t get_printer_type(http_t *http, char *printer, char *uri, + size_t urisize); +static int set_printer_options(http_t *http, char *printer, + int num_options, cups_option_t *options, + char *file); +static int validate_name(const char *name); + + +/* + * 'main()' - Parse options and configure the scheduler. + */ + +int +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + http_t *http; /* Connection to server */ + char *printer, /* Destination printer */ + *pclass, /* Printer class name */ + *val; /* Pointer to allow/deny value */ + int num_options; /* Number of options */ + cups_option_t *options; /* Options */ + char *file; /* New PPD file/interface script */ + + + _cupsSetLocale(argv); + + http = NULL; + printer = NULL; + num_options = 0; + options = NULL; + file = NULL; + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-') + switch (argv[i][1]) + { + case 'c' : /* Add printer to class */ + if (!http) + { + http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption()); + + if (http == NULL) + { + _cupsLangPrintf(stderr, + _("lpadmin: Unable to connect to server: %s"), + strerror(errno)); + return (1); + } + } + + if (printer == NULL) + { + _cupsLangPuts(stderr, + _("lpadmin: Unable to add a printer to the class:\n" + " You must specify a printer name " + "first.")); + return (1); + } + + if (argv[i][2]) + pclass = argv[i] + 2; + else + { + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("lpadmin: Expected class name after \"-c\" " + "option.")); + return (1); + } + + pclass = argv[i]; + } + + if (!validate_name(pclass)) + { + _cupsLangPuts(stderr, + _("lpadmin: Class name can only contain printable " + "characters.")); + return (1); + } + + if (add_printer_to_class(http, printer, pclass)) + return (1); + break; + + case 'd' : /* Set as default destination */ + if (!http) + { + http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption()); + + if (http == NULL) + { + _cupsLangPrintf(stderr, + _("lpadmin: Unable to connect to server: %s"), + strerror(errno)); + return (1); + } + } + + if (argv[i][2]) + printer = argv[i] + 2; + else + { + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("lpadmin: Expected printer name after \"-d\" " + "option.")); + return (1); + } + + printer = argv[i]; + } + + if (!validate_name(printer)) + { + _cupsLangPuts(stderr, + _("lpadmin: Printer name can only contain " + "printable characters.")); + return (1); + } + + if (default_printer(http, printer)) + return (1); + + i = argc; + break; + + case 'h' : /* Connect to host */ + if (http) + { + httpClose(http); + http = NULL; + } + + if (argv[i][2] != '\0') + cupsSetServer(argv[i] + 2); + else + { + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("lpadmin: Expected hostname after \"-h\" " + "option.")); + return (1); + } + + cupsSetServer(argv[i]); + } + break; + + case 'i' : /* Use the specified interface script */ + if (argv[i][2]) + file = argv[i] + 2; + else + { + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("lpadmin: Expected interface after \"-i\" " + "option.")); + return (1); + } + + file = argv[i]; + } + break; + + case 'E' : /* Enable the printer */ + if (printer == NULL) + { +#ifdef HAVE_SSL + cupsSetEncryption(HTTP_ENCRYPT_REQUIRED); + + if (http) + httpEncryption(http, HTTP_ENCRYPT_REQUIRED); +#else + _cupsLangPrintf(stderr, _("%s: Sorry, no encryption support."), + argv[0]); +#endif /* HAVE_SSL */ + break; + } + + if (!http) + { + http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption()); + + if (http == NULL) + { + _cupsLangPrintf(stderr, + _("lpadmin: Unable to connect to server: %s"), + strerror(errno)); + return (1); + } + } + + if (enable_printer(http, printer)) + return (1); + break; + + case 'm' : /* Use the specified standard script/PPD file */ + if (argv[i][2]) + num_options = cupsAddOption("ppd-name", argv[i] + 2, num_options, + &options); + else + { + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("lpadmin: Expected model after \"-m\" " + "option.")); + return (1); + } + + num_options = cupsAddOption("ppd-name", argv[i], num_options, + &options); + } + break; + + case 'o' : /* Set option */ + if (argv[i][2]) + num_options = cupsParseOptions(argv[i] + 2, num_options, &options); + else + { + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("lpadmin: Expected name=value after \"-o\" " + "option.")); + return (1); + } + + num_options = cupsParseOptions(argv[i], num_options, &options); + } + break; + + case 'p' : /* Add/modify a printer */ + if (argv[i][2]) + printer = argv[i] + 2; + else + { + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("lpadmin: Expected printer after \"-p\" " + "option.")); + return (1); + } + + printer = argv[i]; + } + + if (!validate_name(printer)) + { + _cupsLangPuts(stderr, + _("lpadmin: Printer name can only contain " + "printable characters.")); + return (1); + } + break; + + case 'r' : /* Remove printer from class */ + if (!http) + { + http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption()); + + if (http == NULL) + { + _cupsLangPrintf(stderr, + _("lpadmin: Unable to connect to server: %s"), + strerror(errno)); + return (1); + } + } + + if (printer == NULL) + { + _cupsLangPuts(stderr, + _("lpadmin: Unable to remove a printer from the " + "class:\n" + " You must specify a printer name " + "first.")); + return (1); + } + + if (argv[i][2]) + pclass = argv[i] + 2; + else + { + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("lpadmin: Expected class after \"-r\" " + "option.")); + return (1); + } + + pclass = argv[i]; + } + + if (!validate_name(pclass)) + { + _cupsLangPuts(stderr, + _("lpadmin: Class name can only contain printable " + "characters.")); + return (1); + } + + if (delete_printer_from_class(http, printer, pclass)) + return (1); + break; + + case 'R' : /* Remove option */ + if (!http) + { + http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption()); + + if (http == NULL) + { + _cupsLangPrintf(stderr, + _("lpadmin: Unable to connect to server: %s"), + strerror(errno)); + return (1); + } + } + + if (printer == NULL) + { + _cupsLangPuts(stderr, + _("lpadmin: Unable to delete option:\n" + " You must specify a printer name " + "first.")); + return (1); + } + + if (argv[i][2]) + val = argv[i] + 2; + else + { + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("lpadmin: Expected name after \"-R\" " + "option.")); + return (1); + } + + val = argv[i]; + } + + if (delete_printer_option(http, printer, val)) + return (1); + break; + + case 'U' : /* Username */ + if (argv[i][2] != '\0') + cupsSetUser(argv[i] + 2); + else + { + i ++; + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected username after " + "\"-U\" option."), argv[0]); + return (1); + } + + cupsSetUser(argv[i]); + } + break; + + case 'u' : /* Allow/deny users */ + if (argv[i][2]) + val = argv[i] + 2; + else + { + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("lpadmin: Expected allow/deny:userlist after " + "\"-u\" option.")); + return (1); + } + + val = argv[i]; + } + + if (!_cups_strncasecmp(val, "allow:", 6)) + num_options = cupsAddOption("requesting-user-name-allowed", + val + 6, num_options, &options); + else if (!_cups_strncasecmp(val, "deny:", 5)) + num_options = cupsAddOption("requesting-user-name-denied", + val + 5, num_options, &options); + else + { + _cupsLangPrintf(stderr, + _("lpadmin: Unknown allow/deny option \"%s\"."), + val); + return (1); + } + break; + + case 'v' : /* Set the device-uri attribute */ + if (argv[i][2]) + num_options = cupsAddOption("device-uri", argv[i] + 2, + num_options, &options); + else + { + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("lpadmin: Expected device URI after \"-v\" " + "option.")); + return (1); + } + + num_options = cupsAddOption("device-uri", argv[i], + num_options, &options); + } + break; + + case 'x' : /* Delete a printer */ + if (!http) + { + http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption()); + + if (http == NULL) + { + _cupsLangPrintf(stderr, + _("lpadmin: Unable to connect to server: %s"), + strerror(errno)); + return (1); + } + } + + if (argv[i][2]) + printer = argv[i] + 2; + else + { + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("lpadmin: Expected printer or class after " + "\"-x\" option.")); + return (1); + } + + printer = argv[i]; + } + + if (!validate_name(printer)) + { + _cupsLangPuts(stderr, + _("lpadmin: Printer name can only contain " + "printable characters.")); + return (1); + } + + if (delete_printer(http, printer)) + return (1); + + i = argc; + break; + + case 'D' : /* Set the printer-info attribute */ + if (argv[i][2]) + num_options = cupsAddOption("printer-info", argv[i] + 2, + num_options, &options); + else + { + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("lpadmin: Expected description after " + "\"-D\" option.")); + return (1); + } + + num_options = cupsAddOption("printer-info", argv[i], + num_options, &options); + } + break; + + case 'I' : /* Set the supported file types (ignored) */ + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("lpadmin: Expected file type(s) after \"-I\" " + "option.")); + return (1); + } + + _cupsLangPuts(stderr, + _("lpadmin: Warning - content type list ignored.")); + break; + + case 'L' : /* Set the printer-location attribute */ + if (argv[i][2]) + num_options = cupsAddOption("printer-location", argv[i] + 2, + num_options, &options); + else + { + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("lpadmin: Expected location after \"-L\" " + "option.")); + return (1); + } + + num_options = cupsAddOption("printer-location", argv[i], + num_options, &options); + } + break; + + case 'P' : /* Use the specified PPD file */ + if (argv[i][2]) + file = argv[i] + 2; + else + { + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("lpadmin: Expected PPD after \"-P\" option.")); + return (1); + } + + file = argv[i]; + } + break; + + default : + _cupsLangPrintf(stderr, + _("lpadmin: Unknown option \"%c\"."), argv[i][1]); + return (1); + } + else + { + _cupsLangPrintf(stderr, _("lpadmin: Unknown argument \"%s\"."), + argv[i]); + return (1); + } + + /* + * Set options as needed... + */ + + if (num_options || file) + { + if (!http) + { + http = httpConnectEncrypt(cupsServer(), ippPort(), cupsEncryption()); + + if (http == NULL) + { + _cupsLangPrintf(stderr, + _("lpadmin: Unable to connect to server: %s"), + strerror(errno)); + return (1); + } + } + + if (printer == NULL) + { + _cupsLangPuts(stderr, + _("lpadmin: Unable to set the printer options:\n" + " You must specify a printer name first.")); + return (1); + } + + if (set_printer_options(http, printer, num_options, options, file)) + return (1); + } + + if (printer == NULL) + { + _cupsLangPuts(stdout, + _("Usage:\n" + "\n" + " lpadmin [-h server] -d destination\n" + " lpadmin [-h server] -x destination\n" + " lpadmin [-h server] -p printer [-c add-class] " + "[-i interface] [-m model]\n" + " [-r remove-class] [-v device] " + "[-D description]\n" + " [-P ppd-file] [-o name=value]\n" + " [-u allow:user,user] " + "[-u deny:user,user]")); + } + + if (http) + httpClose(http); + + return (0); +} + + +/* + * 'add_printer_to_class()' - Add a printer to a class. + */ + +static int /* O - 0 on success, 1 on fail */ +add_printer_to_class(http_t *http, /* I - Server connection */ + char *printer, /* I - Printer to add */ + char *pclass) /* I - Class to add to */ +{ + int i; /* Looping var */ + ipp_t *request, /* IPP Request */ + *response; /* IPP Response */ + ipp_attribute_t *attr, /* Current attribute */ + *members; /* Members in class */ + char uri[HTTP_MAX_URI]; /* URI for printer/class */ + + + DEBUG_printf(("add_printer_to_class(%p, \"%s\", \"%s\")\n", http, + printer, pclass)); + + /* + * Build an IPP_GET_PRINTER_ATTRIBUTES request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requesting-user-name + */ + + request = ippNewRequest(IPP_GET_PRINTER_ATTRIBUTES); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/classes/%s", pclass); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, uri); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, cupsUser()); + + /* + * Do the request and get back a response... + */ + + response = cupsDoRequest(http, request, "/"); + + /* + * Build a CUPS_ADD_MODIFY_CLASS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requesting-user-name + * member-uris + */ + + request = ippNewRequest(CUPS_ADD_MODIFY_CLASS); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, uri); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, cupsUser()); + + /* + * See if the printer is already in the class... + */ + + if (response != NULL && + (members = ippFindAttribute(response, "member-names", + IPP_TAG_NAME)) != NULL) + for (i = 0; i < members->num_values; i ++) + if (_cups_strcasecmp(printer, members->values[i].string.text) == 0) + { + _cupsLangPrintf(stderr, + _("lpadmin: Printer %s is already a member of class " + "%s."), printer, pclass); + ippDelete(request); + ippDelete(response); + return (0); + } + + /* + * OK, the printer isn't part of the class, so add it... + */ + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", printer); + + if (response != NULL && + (members = ippFindAttribute(response, "member-uris", + IPP_TAG_URI)) != NULL) + { + /* + * Add the printer to the existing list... + */ + + attr = ippAddStrings(request, IPP_TAG_PRINTER, IPP_TAG_URI, + "member-uris", members->num_values + 1, NULL, NULL); + for (i = 0; i < members->num_values; i ++) + attr->values[i].string.text = + _cupsStrAlloc(members->values[i].string.text); + + attr->values[i].string.text = _cupsStrAlloc(uri); + } + else + ippAddString(request, IPP_TAG_PRINTER, IPP_TAG_URI, "member-uris", NULL, + uri); + + /* + * Then send the request... + */ + + ippDelete(response); + + ippDelete(cupsDoRequest(http, request, "/admin/")); + if (cupsLastError() > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, _("%s: %s"), "lpadmin", cupsLastErrorString()); + + return (1); + } + else + return (0); +} + + +/* + * 'default_printer()' - Set the default printing destination. + */ + +static int /* O - 0 on success, 1 on fail */ +default_printer(http_t *http, /* I - Server connection */ + char *printer) /* I - Printer name */ +{ + ipp_t *request; /* IPP Request */ + char uri[HTTP_MAX_URI]; /* URI for printer/class */ + + + DEBUG_printf(("default_printer(%p, \"%s\")\n", http, printer)); + + /* + * Build a CUPS_SET_DEFAULT request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requesting-user-name + */ + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", printer); + + request = ippNewRequest(CUPS_SET_DEFAULT); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, uri); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, cupsUser()); + + /* + * Do the request and get back a response... + */ + + ippDelete(cupsDoRequest(http, request, "/admin/")); + + if (cupsLastError() > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, _("%s: %s"), "lpadmin", cupsLastErrorString()); + + return (1); + } + else + return (0); +} + + +/* + * 'delete_printer()' - Delete a printer from the system... + */ + +static int /* O - 0 on success, 1 on fail */ +delete_printer(http_t *http, /* I - Server connection */ + char *printer) /* I - Printer to delete */ +{ + ipp_t *request; /* IPP Request */ + char uri[HTTP_MAX_URI]; /* URI for printer/class */ + + + DEBUG_printf(("delete_printer(%p, \"%s\")\n", http, printer)); + + /* + * Build a CUPS_DELETE_PRINTER request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requesting-user-name + */ + + request = ippNewRequest(CUPS_DELETE_PRINTER); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/printers/%s", printer); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, uri); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, cupsUser()); + + /* + * Do the request and get back a response... + */ + + ippDelete(cupsDoRequest(http, request, "/admin/")); + + if (cupsLastError() > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, _("%s: %s"), "lpadmin", cupsLastErrorString()); + + return (1); + } + else + return (0); +} + + +/* + * 'delete_printer_from_class()' - Delete a printer from a class. + */ + +static int /* O - 0 on success, 1 on fail */ +delete_printer_from_class( + http_t *http, /* I - Server connection */ + char *printer, /* I - Printer to remove */ + char *pclass) /* I - Class to remove from */ +{ + int i, j, k; /* Looping vars */ + ipp_t *request, /* IPP Request */ + *response; /* IPP Response */ + ipp_attribute_t *attr, /* Current attribute */ + *members; /* Members in class */ + char uri[HTTP_MAX_URI]; /* URI for printer/class */ + + + DEBUG_printf(("delete_printer_from_class(%p, \"%s\", \"%s\")\n", http, + printer, pclass)); + + /* + * Build an IPP_GET_PRINTER_ATTRIBUTES request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requesting-user-name + */ + + request = ippNewRequest(IPP_GET_PRINTER_ATTRIBUTES); + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, + "localhost", 0, "/classes/%s", pclass); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, uri); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, cupsUser()); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(http, request, "/classes/")) == NULL || + response->request.status.status_code == IPP_NOT_FOUND) + { + _cupsLangPrintf(stderr, _("%s: %s"), "lpadmin", cupsLastErrorString()); + + ippDelete(response); + + return (1); + } + + /* + * See if the printer is already in the class... + */ + + if ((members = ippFindAttribute(response, "member-names", IPP_TAG_NAME)) == NULL) + { + _cupsLangPuts(stderr, _("lpadmin: No member names were seen.")); + + ippDelete(response); + + return (1); + } + + for (i = 0; i < members->num_values; i ++) + if (!_cups_strcasecmp(printer, members->values[i].string.text)) + break; + + if (i >= members->num_values) + { + _cupsLangPrintf(stderr, + _("lpadmin: Printer %s is not a member of class %s."), + printer, pclass); + + ippDelete(response); + + return (1); + } + + if (members->num_values == 1) + { + /* + * Build a CUPS_DELETE_CLASS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requesting-user-name + */ + + request = ippNewRequest(CUPS_DELETE_CLASS); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, uri); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, cupsUser()); + } + else + { + /* + * Build a CUPS_ADD_MODIFY_CLASS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requesting-user-name + * member-uris + */ + + request = ippNewRequest(CUPS_ADD_MODIFY_CLASS); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, uri); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, cupsUser()); + + /* + * Delete the printer from the class... + */ + + members = ippFindAttribute(response, "member-uris", IPP_TAG_URI); + attr = ippAddStrings(request, IPP_TAG_PRINTER, IPP_TAG_URI, + "member-uris", members->num_values - 1, NULL, NULL); + + for (j = 0, k = 0; j < members->num_values; j ++) + if (j != i) + attr->values[k ++].string.text = + _cupsStrAlloc(members->values[j].string.text); + } + + /* + * Then send the request... + */ + + ippDelete(response); + + ippDelete(cupsDoRequest(http, request, "/admin/")); + + if (cupsLastError() > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, _("%s: %s"), "lpadmin", cupsLastErrorString()); + + return (1); + } + else + return (0); +} + + +/* + * 'delete_printer_option()' - Delete a printer option. + */ + +static int /* O - 0 on success, 1 on fail */ +delete_printer_option(http_t *http, /* I - Server connection */ + char *printer, /* I - Printer */ + char *option) /* I - Option to delete */ +{ + ipp_t *request; /* IPP request */ + char uri[HTTP_MAX_URI]; /* URI for printer/class */ + + + /* + * Build a CUPS_ADD_MODIFY_PRINTER or CUPS_ADD_MODIFY_CLASS request, which + * requires the following attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requesting-user-name + * option with deleteAttr tag + */ + + if (get_printer_type(http, printer, uri, sizeof(uri)) & CUPS_PRINTER_CLASS) + request = ippNewRequest(CUPS_ADD_MODIFY_CLASS); + else + request = ippNewRequest(CUPS_ADD_MODIFY_PRINTER); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, uri); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, cupsUser()); + ippAddInteger(request, IPP_TAG_PRINTER, IPP_TAG_DELETEATTR, option, 0); + + /* + * Do the request and get back a response... + */ + + ippDelete(cupsDoRequest(http, request, "/admin/")); + + if (cupsLastError() > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, _("%s: %s"), "lpadmin", cupsLastErrorString()); + + return (1); + } + else + return (0); +} + + +/* + * 'enable_printer()' - Enable a printer... + */ + +static int /* O - 0 on success, 1 on fail */ +enable_printer(http_t *http, /* I - Server connection */ + char *printer) /* I - Printer to enable */ +{ + ipp_t *request; /* IPP Request */ + char uri[HTTP_MAX_URI]; /* URI for printer/class */ + + + DEBUG_printf(("enable_printer(%p, \"%s\")\n", http, printer)); + + /* + * Build a CUPS_ADD_MODIFY_PRINTER or CUPS_ADD_MODIFY_CLASS request, which + * require the following attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requesting-user-name + * printer-state + * printer-is-accepting-jobs + */ + + if (get_printer_type(http, printer, uri, sizeof(uri)) & CUPS_PRINTER_CLASS) + request = ippNewRequest(CUPS_ADD_MODIFY_CLASS); + else + request = ippNewRequest(CUPS_ADD_MODIFY_PRINTER); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, uri); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, cupsUser()); + ippAddInteger(request, IPP_TAG_PRINTER, IPP_TAG_ENUM, "printer-state", + IPP_PRINTER_IDLE); + ippAddBoolean(request, IPP_TAG_PRINTER, "printer-is-accepting-jobs", 1); + + /* + * Do the request and get back a response... + */ + + ippDelete(cupsDoRequest(http, request, "/admin/")); + + if (cupsLastError() > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, _("%s: %s"), "lpadmin", cupsLastErrorString()); + + return (1); + } + else + return (0); +} + + +/* + * 'get_printer_type()' - Determine the printer type and URI. + */ + +static cups_ptype_t /* O - printer-type value */ +get_printer_type(http_t *http, /* I - Server connection */ + char *printer, /* I - Printer name */ + char *uri, /* I - URI buffer */ + size_t urisize) /* I - Size of URI buffer */ +{ + ipp_t *request, /* IPP request */ + *response; /* IPP response */ + ipp_attribute_t *attr; /* printer-type attribute */ + cups_ptype_t type; /* printer-type value */ + + + /* + * Build a GET_PRINTER_ATTRIBUTES request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requested-attributes + * requesting-user-name + */ + + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, urisize, "ipp", NULL, "localhost", + ippPort(), "/printers/%s", printer); + + request = ippNewRequest(IPP_GET_PRINTER_ATTRIBUTES); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, uri); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", NULL, "printer-type"); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, cupsUser()); + + /* + * Do the request... + */ + + response = cupsDoRequest(http, request, "/"); + if ((attr = ippFindAttribute(response, "printer-type", + IPP_TAG_ENUM)) != NULL) + { + type = (cups_ptype_t)attr->values[0].integer; + + if (type & CUPS_PRINTER_CLASS) + httpAssembleURIf(HTTP_URI_CODING_ALL, uri, urisize, "ipp", NULL, + "localhost", ippPort(), "/classes/%s", printer); + } + else + type = CUPS_PRINTER_LOCAL; + + ippDelete(response); + + return (type); +} + + +/* + * 'set_printer_options()' - Set the printer options. + */ + +static int /* O - 0 on success, 1 on fail */ +set_printer_options( + http_t *http, /* I - Server connection */ + char *printer, /* I - Printer */ + int num_options, /* I - Number of options */ + cups_option_t *options, /* I - Options */ + char *file) /* I - PPD file/interface script */ +{ + ipp_t *request; /* IPP Request */ + const char *ppdfile; /* PPD filename */ + int ppdchanged; /* PPD changed? */ + ppd_file_t *ppd; /* PPD file */ + ppd_choice_t *choice; /* Marked choice */ + char uri[HTTP_MAX_URI], /* URI for printer/class */ + line[1024], /* Line from PPD file */ + keyword[1024], /* Keyword from Default line */ + *keyptr, /* Pointer into keyword... */ + tempfile[1024]; /* Temporary filename */ + cups_file_t *in, /* PPD file */ + *out; /* Temporary file */ + const char *protocol, /* Old protocol option */ + *customval, /* Custom option value */ + *boolval; /* Boolean value */ + int wrote_ipp_supplies = 0, /* Wrote cupsIPPSupplies keyword? */ + wrote_snmp_supplies = 0;/* Wrote cupsSNMPSupplies keyword? */ + + + DEBUG_printf(("set_printer_options(http=%p, printer=\"%s\", num_options=%d, " + "options=%p, file=\"%s\")\n", http, printer, num_options, + options, file)); + + /* + * Build a CUPS_ADD_MODIFY_PRINTER or CUPS_ADD_MODIFY_CLASS request, which + * requires the following attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requesting-user-name + * other options + */ + + if (get_printer_type(http, printer, uri, sizeof(uri)) & CUPS_PRINTER_CLASS) + request = ippNewRequest(CUPS_ADD_MODIFY_CLASS); + else + request = ippNewRequest(CUPS_ADD_MODIFY_PRINTER); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, uri); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, + "requesting-user-name", NULL, cupsUser()); + + /* + * Add the options... + */ + + cupsEncodeOptions2(request, num_options, options, IPP_TAG_PRINTER); + + if ((protocol = cupsGetOption("protocol", num_options, options)) != NULL) + { + if (!_cups_strcasecmp(protocol, "bcp")) + ippAddString(request, IPP_TAG_PRINTER, IPP_TAG_NAME, "port-monitor", + NULL, "bcp"); + else if (!_cups_strcasecmp(protocol, "tbcp")) + ippAddString(request, IPP_TAG_PRINTER, IPP_TAG_NAME, "port-monitor", + NULL, "tbcp"); + } + + if (file) + ppdfile = file; + else if (request->request.op.operation_id == CUPS_ADD_MODIFY_PRINTER) + ppdfile = cupsGetPPD(printer); + else + ppdfile = NULL; + + if (ppdfile != NULL) + { + /* + * Set default options in the PPD file... + */ + + ppd = ppdOpenFile(ppdfile); + ppdMarkDefaults(ppd); + cupsMarkOptions(ppd, num_options, options); + + if ((out = cupsTempFile2(tempfile, sizeof(tempfile))) == NULL) + { + _cupsLangPrintError(NULL, _("lpadmin: Unable to create temporary file")); + ippDelete(request); + if (ppdfile != file) + unlink(ppdfile); + return (1); + } + + if ((in = cupsFileOpen(ppdfile, "r")) == NULL) + { + _cupsLangPrintf(stderr, + _("lpadmin: Unable to open PPD file \"%s\" - %s"), + ppdfile, strerror(errno)); + ippDelete(request); + if (ppdfile != file) + unlink(ppdfile); + cupsFileClose(out); + unlink(tempfile); + return (1); + } + + ppdchanged = 0; + + while (cupsFileGets(in, line, sizeof(line))) + { + if (!strncmp(line, "*cupsIPPSupplies:", 17) && + (boolval = cupsGetOption("cupsIPPSupplies", num_options, + options)) != NULL) + { + wrote_ipp_supplies = 1; + cupsFilePrintf(out, "*cupsIPPSupplies: %s\n", + (!_cups_strcasecmp(boolval, "true") || + !_cups_strcasecmp(boolval, "yes") || + !_cups_strcasecmp(boolval, "on")) ? "True" : "False"); + } + else if (!strncmp(line, "*cupsSNMPSupplies:", 18) && + (boolval = cupsGetOption("cupsSNMPSupplies", num_options, + options)) != NULL) + { + wrote_snmp_supplies = 1; + cupsFilePrintf(out, "*cupsSNMPSupplies: %s\n", + (!_cups_strcasecmp(boolval, "true") || + !_cups_strcasecmp(boolval, "yes") || + !_cups_strcasecmp(boolval, "on")) ? "True" : "False"); + } + else if (strncmp(line, "*Default", 8)) + cupsFilePrintf(out, "%s\n", line); + else + { + /* + * Get default option name... + */ + + strlcpy(keyword, line + 8, sizeof(keyword)); + + for (keyptr = keyword; *keyptr; keyptr ++) + if (*keyptr == ':' || isspace(*keyptr & 255)) + break; + + *keyptr++ = '\0'; + while (isspace(*keyptr & 255)) + keyptr ++; + + if (!strcmp(keyword, "PageRegion") || + !strcmp(keyword, "PageSize") || + !strcmp(keyword, "PaperDimension") || + !strcmp(keyword, "ImageableArea")) + { + if ((choice = ppdFindMarkedChoice(ppd, "PageSize")) == NULL) + choice = ppdFindMarkedChoice(ppd, "PageRegion"); + } + else + choice = ppdFindMarkedChoice(ppd, keyword); + + if (choice && strcmp(choice->choice, keyptr)) + { + if (strcmp(choice->choice, "Custom")) + { + cupsFilePrintf(out, "*Default%s: %s\n", keyword, choice->choice); + ppdchanged = 1; + } + else if ((customval = cupsGetOption(keyword, num_options, + options)) != NULL) + { + cupsFilePrintf(out, "*Default%s: %s\n", keyword, customval); + ppdchanged = 1; + } + else + cupsFilePrintf(out, "%s\n", line); + } + else + cupsFilePrintf(out, "%s\n", line); + } + } + + if (!wrote_ipp_supplies && + (boolval = cupsGetOption("cupsIPPSupplies", num_options, + options)) != NULL) + { + cupsFilePrintf(out, "*cupsIPPSupplies: %s\n", + (!_cups_strcasecmp(boolval, "true") || + !_cups_strcasecmp(boolval, "yes") || + !_cups_strcasecmp(boolval, "on")) ? "True" : "False"); + } + + if (!wrote_snmp_supplies && + (boolval = cupsGetOption("cupsSNMPSupplies", num_options, + options)) != NULL) + { + cupsFilePrintf(out, "*cupsSNMPSupplies: %s\n", + (!_cups_strcasecmp(boolval, "true") || + !_cups_strcasecmp(boolval, "yes") || + !_cups_strcasecmp(boolval, "on")) ? "True" : "False"); + } + + cupsFileClose(in); + cupsFileClose(out); + ppdClose(ppd); + + /* + * Do the request... + */ + + ippDelete(cupsDoFileRequest(http, request, "/admin/", + ppdchanged ? tempfile : file)); + + /* + * Clean up temp files... (TODO: catch signals in case we CTRL-C during + * lpadmin) + */ + + if (ppdfile != file) + unlink(ppdfile); + unlink(tempfile); + } + else + { + /* + * No PPD file - just set the options... + */ + + ippDelete(cupsDoRequest(http, request, "/admin/")); + } + + /* + * Check the response... + */ + + if (cupsLastError() > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, _("%s: %s"), "lpadmin", cupsLastErrorString()); + + return (1); + } + else + return (0); +} + + +/* + * 'validate_name()' - Make sure the printer name only contains valid chars. + */ + +static int /* O - 0 if name is no good, 1 if name is good */ +validate_name(const char *name) /* I - Name to check */ +{ + const char *ptr; /* Pointer into name */ + + + /* + * Scan the whole name... + */ + + for (ptr = name; *ptr; ptr ++) + if (*ptr == '@') + break; + else if ((*ptr >= 0 && *ptr <= ' ') || *ptr == 127 || *ptr == '/' || + *ptr == '#') + return (0); + + /* + * All the characters are good; validate the length, too... + */ + + return ((ptr - name) < 128); +} + + +/* + * End of "$Id$". + */ diff --git a/systemv/lpinfo.c b/systemv/lpinfo.c new file mode 100644 index 0000000000..acfe50108d --- /dev/null +++ b/systemv/lpinfo.c @@ -0,0 +1,498 @@ +/* + * "$Id$" + * + * "lpinfo" command for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Parse options and show information. + * device_cb - Device callback. + * show_devices() - Show available devices. + * show_models() - Show available PPDs. + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * Local functions... + */ + +static void device_cb(const char *device_clas, const char *device_id, + const char *device_info, + const char *device_make_and_model, + const char *device_uri, const char *device_location, + void *user_data); +static int show_devices(int long_status, int timeout, + const char *include_schemes, + const char *exclude_schemes); +static int show_models(int long_status, + const char *device_id, const char *language, + const char *make_model, const char *product, + const char *include_schemes, + const char *exclude_schemes); + + +/* + * 'main()' - Parse options and show status information. + */ + +int +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + int long_status; /* Long listing? */ + const char *device_id, /* 1284 device ID */ + *language, /* Language */ + *make_model, /* Make and model */ + *product, /* Product */ + *include_schemes, /* Schemes to include */ + *exclude_schemes; /* Schemes to exclude */ + int timeout; /* Device timeout */ + + + _cupsSetLocale(argv); + + long_status = 0; + device_id = NULL; + language = NULL; + make_model = NULL; + product = NULL; + include_schemes = CUPS_INCLUDE_ALL; + exclude_schemes = CUPS_EXCLUDE_NONE; + timeout = CUPS_TIMEOUT_DEFAULT; + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-') + switch (argv[i][1]) + { + case 'E' : /* Encrypt */ +#ifdef HAVE_SSL + cupsSetEncryption(HTTP_ENCRYPT_REQUIRED); +#else + _cupsLangPrintf(stderr, + _("%s: Sorry, no encryption support."), + argv[0]); +#endif /* HAVE_SSL */ + break; + + case 'h' : /* Connect to host */ + if (argv[i][2] != '\0') + cupsSetServer(argv[i] + 2); + else + { + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("Error: need hostname after \"-h\" option.")); + return (1); + } + + cupsSetServer(argv[i]); + } + break; + + case 'l' : /* Show long listing */ + long_status = 1; + break; + + case 'm' : /* Show models */ + if (show_models(long_status, device_id, language, make_model, + product, include_schemes, exclude_schemes)) + return (1); + break; + + case 'v' : /* Show available devices */ + if (show_devices(long_status, timeout, include_schemes, + exclude_schemes)) + return (1); + break; + + case '-' : /* --something */ + if (!strcmp(argv[i], "--device-id")) + { + i ++; + + if (i < argc) + device_id = argv[i]; + else + { + _cupsLangPuts(stderr, + _("lpinfo: Expected 1284 device ID string " + "after \"--device-id\".")); + return (1); + } + } + else if (!strncmp(argv[i], "--device-id=", 12) && argv[i][12]) + { + device_id = argv[i] + 12; + } + else if (!strcmp(argv[i], "--exclude-schemes")) + { + i ++; + + if (i < argc) + exclude_schemes = argv[i]; + else + { + _cupsLangPuts(stderr, + _("lpinfo: Expected scheme list after " + "\"--exclude-schemes\".")); + return (1); + } + } + else if (!strncmp(argv[i], "--exclude-schemes=", 18) && argv[i][18]) + { + exclude_schemes = argv[i] + 18; + } + else if (!strcmp(argv[i], "--include-schemes")) + { + i ++; + + if (i < argc) + include_schemes = argv[i]; + else + { + _cupsLangPuts(stderr, + _("lpinfo: Expected scheme list after " + "\"--include-schemes\".")); + return (1); + } + } + else if (!strncmp(argv[i], "--include-schemes=", 18) && argv[i][18]) + { + include_schemes = argv[i] + 18; + } + else if (!strcmp(argv[i], "--language")) + { + i ++; + if (i < argc) + language = argv[i]; + else + { + _cupsLangPuts(stderr, + _("lpinfo: Expected language after " + "\"--language\".")); + return (1); + } + } + else if (!strncmp(argv[i], "--language=", 11) && argv[i][11]) + { + language = argv[i] + 11; + } + else if (!strcmp(argv[i], "--make-and-model")) + { + i ++; + if (i < argc) + make_model= argv[i]; + else + { + _cupsLangPuts(stderr, + _("lpinfo: Expected make and model after " + "\"--make-and-model\".")); + return (1); + } + } + else if (!strncmp(argv[i], "--make-and-model=", 17) && argv[i][17]) + { + make_model = argv[i] + 17; + } + else if (!strcmp(argv[i], "--product")) + { + i ++; + if (i < argc) + product = argv[i]; + else + { + _cupsLangPuts(stderr, + _("lpinfo: Expected product string after " + "\"--product\".")); + return (1); + } + } + else if (!strncmp(argv[i], "--product=", 10) && argv[i][10]) + { + product = argv[i] + 10; + } + else if (!strcmp(argv[i], "--timeout")) + { + i ++; + if (i < argc) + timeout = atoi(argv[i]); + else + { + _cupsLangPuts(stderr, + _("lpinfo: Expected timeout after " + "\"--timeout\".")); + return (1); + } + } + else if (!strncmp(argv[i], "--timeout=", 10) && argv[i][10]) + { + timeout = atoi(argv[i] + 10); + } + else + { + _cupsLangPrintf(stderr, _("lpinfo: Unknown option \"%s\"."), + argv[i]); + return (1); + } + break; + + default : + _cupsLangPrintf(stderr, _("lpinfo: Unknown option \"%c\"."), + argv[i][1]); + return (1); + } + else + { + _cupsLangPrintf(stderr, _("lpinfo: Unknown argument \"%s\"."), + argv[i]); + return (1); + } + + return (0); +} + + +/* + * 'device_cb()' - Device callback. + */ + +static void +device_cb( + const char *device_class, /* I - device-class string */ + const char *device_id, /* I - device-id string */ + const char *device_info, /* I - device-info string */ + const char *device_make_and_model, /* I - device-make-and-model string */ + const char *device_uri, /* I - device-uri string */ + const char *device_location, /* I - device-location string */ + void *user_data) /* I - User data */ +{ + int *long_status; /* Show verbose info? */ + + + /* + * Display the device... + */ + + long_status = (int *)user_data; + + if (*long_status) + { + _cupsLangPrintf(stdout, + _("Device: uri = %s\n" + " class = %s\n" + " info = %s\n" + " make-and-model = %s\n" + " device-id = %s\n" + " location = %s"), + device_uri, device_class, device_info, + device_make_and_model, device_id, device_location); + } + else + _cupsLangPrintf(stdout, "%s %s", device_class, device_uri); +} + + +/* + * 'show_devices()' - Show available devices. + */ + +static int /* O - 0 on success, 1 on failure */ +show_devices( + int long_status, /* I - Long status report? */ + int timeout, /* I - Timeout */ + const char *include_schemes, /* I - List of schemes to include */ + const char *exclude_schemes) /* I - List of schemes to exclude */ +{ + if (cupsGetDevices(CUPS_HTTP_DEFAULT, timeout, include_schemes, + exclude_schemes, device_cb, &long_status) != IPP_OK) + { + _cupsLangPrintf(stderr, "lpinfo: %s", cupsLastErrorString()); + return (1); + } + + return (0); +} + + +/* + * 'show_models()' - Show available PPDs. + */ + +static int /* O - 0 on success, 1 on failure */ +show_models( + int long_status, /* I - Long status report? */ + const char *device_id, /* I - 1284 device ID */ + const char *language, /* I - Language */ + const char *make_model, /* I - Make and model */ + const char *product, /* I - Product */ + const char *include_schemes, /* I - List of schemes to include */ + const char *exclude_schemes) /* I - List of schemes to exclude */ +{ + ipp_t *request, /* IPP Request */ + *response; /* IPP Response */ + ipp_attribute_t *attr; /* Current attribute */ + const char *ppd_device_id, /* Pointer to ppd-device-id */ + *ppd_language, /* Pointer to ppd-natural-language */ + *ppd_make_model, /* Pointer to ppd-make-and-model */ + *ppd_name; /* Pointer to ppd-name */ + cups_option_t option; /* in/exclude-schemes option */ + + + /* + * Build a CUPS_GET_PPDS request... + */ + + request = ippNewRequest(CUPS_GET_PPDS); + + if (device_id) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_TEXT, "ppd-device-id", + NULL, device_id); + if (language) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_LANGUAGE, "ppd-language", + NULL, language); + if (make_model) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_TEXT, "ppd-make-and-model", + NULL, make_model); + if (product) + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_TEXT, "ppd-product", + NULL, product); + + if (include_schemes) + { + option.name = "include-schemes"; + option.value = (char *)include_schemes; + + cupsEncodeOptions2(request, 1, &option, IPP_TAG_OPERATION); + } + + if (exclude_schemes) + { + option.name = "exclude-schemes"; + option.value = (char *)exclude_schemes; + + cupsEncodeOptions2(request, 1, &option, IPP_TAG_OPERATION); + } + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(CUPS_HTTP_DEFAULT, request, "/")) != NULL) + { + /* + * Loop through the device list and display them... + */ + + if (response->request.status.status_code > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, "lpinfo: %s", cupsLastErrorString()); + ippDelete(response); + return (1); + } + + for (attr = response->attrs; attr != NULL; attr = attr->next) + { + /* + * Skip leading attributes until we hit a PPD... + */ + + while (attr != NULL && attr->group_tag != IPP_TAG_PRINTER) + attr = attr->next; + + if (attr == NULL) + break; + + /* + * Pull the needed attributes from this PPD... + */ + + ppd_device_id = "NONE"; + ppd_language = NULL; + ppd_make_model = NULL; + ppd_name = NULL; + + while (attr != NULL && attr->group_tag == IPP_TAG_PRINTER) + { + if (!strcmp(attr->name, "ppd-device-id") && + attr->value_tag == IPP_TAG_TEXT) + ppd_device_id = attr->values[0].string.text; + else if (!strcmp(attr->name, "ppd-natural-language") && + attr->value_tag == IPP_TAG_LANGUAGE) + ppd_language = attr->values[0].string.text; + else if (!strcmp(attr->name, "ppd-make-and-model") && + attr->value_tag == IPP_TAG_TEXT) + ppd_make_model = attr->values[0].string.text; + else if (!strcmp(attr->name, "ppd-name") && + attr->value_tag == IPP_TAG_NAME) + ppd_name = attr->values[0].string.text; + + attr = attr->next; + } + + /* + * See if we have everything needed... + */ + + if (ppd_language == NULL || ppd_make_model == NULL || ppd_name == NULL) + { + if (attr == NULL) + break; + else + continue; + } + + /* + * Display the device... + */ + + if (long_status) + { + _cupsLangPrintf(stdout, + _("Model: name = %s\n" + " natural_language = %s\n" + " make-and-model = %s\n" + " device-id = %s"), + ppd_name, ppd_language, ppd_make_model, ppd_device_id); + } + else + _cupsLangPrintf(stdout, "%s %s", ppd_name, ppd_make_model); + + if (attr == NULL) + break; + } + + ippDelete(response); + } + else + { + _cupsLangPrintf(stderr, "lpinfo: %s", cupsLastErrorString()); + + return (1); + } + + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/systemv/lpmove.c b/systemv/lpmove.c new file mode 100644 index 0000000000..54424894f1 --- /dev/null +++ b/systemv/lpmove.c @@ -0,0 +1,213 @@ +/* + * "$Id$" + * + * "lpmove" command for CUPS. + * + * Copyright 2007-2010 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Parse options and move jobs. + * move_job() - Move a job. + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * Local functions... + */ + +static int move_job(http_t *http, const char *src, int jobid, + const char *dest); + + +/* + * 'main()' - Parse options and show status information. + */ + +int +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + http_t *http; /* Connection to server */ + const char *job; /* Job name */ + int jobid; /* Job ID */ + int num_dests; /* Number of destinations */ + cups_dest_t *dests; /* Destinations */ + const char *src, /* Original queue */ + *dest; /* New destination */ + + + _cupsSetLocale(argv); + + dest = NULL; + dests = NULL; + job = NULL; + jobid = 0; + num_dests = 0; + src = NULL; + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-') + switch (argv[i][1]) + { + case 'E' : /* Encrypt */ +#ifdef HAVE_SSL + cupsSetEncryption(HTTP_ENCRYPT_REQUIRED); + +#else + _cupsLangPrintf(stderr, + _("%s: Sorry, no encryption support."), + argv[0]); +#endif /* HAVE_SSL */ + break; + + case 'h' : /* Connect to host */ + if (argv[i][2] != '\0') + cupsSetServer(argv[i] + 2); + else + { + i ++; + + if (i >= argc) + { + _cupsLangPuts(stderr, + _("Error: need hostname after \"-h\" option.")); + return (1); + } + + cupsSetServer(argv[i]); + } + break; + + default : + _cupsLangPrintf(stderr, _("lpmove: Unknown option \"%c\"."), + argv[i][1]); + return (1); + } + else if (!jobid && !src) + { + if (num_dests == 0) + num_dests = cupsGetDests(&dests); + + if ((job = strrchr(argv[i], '-')) != NULL && + cupsGetDest(argv[i], NULL, num_dests, dests) == NULL) + jobid = atoi(job + 1); + else if (isdigit(argv[i][0] & 255) && + !cupsGetDest(argv[i], NULL, num_dests, dests)) + jobid = atoi(argv[i]); + else + src = argv[i]; + } + else if (dest == NULL) + dest = argv[i]; + else + { + _cupsLangPrintf(stderr, _("lpmove: Unknown argument \"%s\"."), argv[i]); + return (1); + } + + if ((!jobid && !src) || !dest) + { + _cupsLangPuts(stdout, _("Usage: lpmove job/src dest")); + return (1); + } + + http = httpConnectEncrypt(cupsServer(), ippPort(), cupsEncryption()); + + if (http == NULL) + { + _cupsLangPrintf(stderr, _("lpmove: Unable to connect to server: %s"), + strerror(errno)); + return (1); + } + + return (move_job(http, src, jobid, dest)); +} + + +/* + * 'move_job()' - Move a job. + */ + +static int /* O - 0 on success, 1 on error */ +move_job(http_t *http, /* I - HTTP connection to server */ + const char *src, /* I - Source queue */ + int jobid, /* I - Job ID */ + const char *dest) /* I - Destination queue */ +{ + ipp_t *request; /* IPP Request */ + char job_uri[HTTP_MAX_URI], /* job-uri */ + printer_uri[HTTP_MAX_URI]; /* job-printer-uri */ + + + if (!http) + return (1); + + /* + * Build a CUPS_MOVE_JOB request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * job-uri/printer-uri + * job-printer-uri + * requesting-user-name + */ + + request = ippNewRequest(CUPS_MOVE_JOB); + + if (jobid) + { + snprintf(job_uri, sizeof(job_uri), "ipp://localhost/jobs/%d", jobid); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-uri", NULL, + job_uri); + } + else + { + httpAssembleURIf(HTTP_URI_CODING_ALL, job_uri, sizeof(job_uri), "ipp", NULL, + "localhost", 0, "/printers/%s", src); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", NULL, + job_uri); + } + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, cupsUser()); + + httpAssembleURIf(HTTP_URI_CODING_ALL, printer_uri, sizeof(printer_uri), + "ipp", NULL, "localhost", 0, "/printers/%s", dest); + ippAddString(request, IPP_TAG_JOB, IPP_TAG_URI, "job-printer-uri", + NULL, printer_uri); + + /* + * Do the request and get back a response... + */ + + ippDelete(cupsDoRequest(http, request, "/jobs")); + + if (cupsLastError() > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, "lpmove: %s", cupsLastErrorString()); + return (1); + } + else + return (0); +} + + +/* + * End of "$Id$". + */ diff --git a/systemv/lpoptions.c b/systemv/lpoptions.c new file mode 100644 index 0000000000..12dbe5f854 --- /dev/null +++ b/systemv/lpoptions.c @@ -0,0 +1,565 @@ +/* + * "$Id$" + * + * Printer option program for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Main entry. + * list_group() - List printer-specific options from the PPD group. + * list_options() - List printer-specific options from the PPD file. + * usage() - Show program usage and exit. + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * Local functions... + */ + +static void list_group(ppd_file_t *ppd, ppd_group_t *group); +static void list_options(cups_dest_t *dest); +static void usage(void) __attribute__((noreturn)); + + +/* + * 'main()' - Main entry. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i, j; /* Looping vars */ + int changes; /* Did we make changes? */ + int num_options; /* Number of options */ + cups_option_t *options; /* Options */ + int num_dests; /* Number of destinations */ + cups_dest_t *dests; /* Destinations */ + cups_dest_t *dest; /* Current destination */ + char *printer, /* Printer name */ + *instance, /* Instance name */ + *option; /* Current option */ + + + _cupsSetLocale(argv); + + /* + * Loop through the command-line arguments... + */ + + dest = NULL; + num_dests = 0; + dests = NULL; + num_options = 0; + options = NULL; + changes = 0; + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-') + { + switch (argv[i][1]) + { + case 'd' : /* -d printer */ + if (argv[i][2]) + printer = argv[i] + 2; + else + { + i ++; + if (i >= argc) + usage(); + + printer = argv[i]; + } + + if ((instance = strrchr(printer, '/')) != NULL) + *instance++ = '\0'; + + if (num_dests == 0) + num_dests = cupsGetDests(&dests); + + if (num_dests == 0 || !dests || + (dest = cupsGetDest(printer, instance, num_dests, + dests)) == NULL) + { + _cupsLangPuts(stderr, _("lpoptions: Unknown printer or class.")); + return (1); + } + + /* + * Set the default destination... + */ + + for (j = 0; j < num_dests; j ++) + dests[j].is_default = 0; + + dest->is_default = 1; + + cupsSetDests(num_dests, dests); + + for (j = 0; j < dest->num_options; j ++) + if (cupsGetOption(dest->options[j].name, num_options, + options) == NULL) + num_options = cupsAddOption(dest->options[j].name, + dest->options[j].value, + num_options, &options); + break; + + case 'h' : /* -h server */ + if (argv[i][2]) + cupsSetServer(argv[i] + 2); + else + { + i ++; + if (i >= argc) + usage(); + + cupsSetServer(argv[i]); + } + break; + + case 'E' : /* Encrypt connection */ + cupsSetEncryption(HTTP_ENCRYPT_REQUIRED); + break; + + case 'l' : /* -l (list options) */ + if (dest == NULL) + { + if (num_dests == 0) + num_dests = cupsGetDests(&dests); + + if ((dest = cupsGetDest(NULL, NULL, num_dests, dests)) == NULL) + dest = dests; + } + + if (dest == NULL) + _cupsLangPuts(stderr, _("lpoptions: No printers.")); + else + list_options(dest); + + changes = -1; + break; + + case 'o' : /* -o option[=value] */ + if (dest == NULL) + { + if (num_dests == 0) + num_dests = cupsGetDests(&dests); + + if ((dest = cupsGetDest(NULL, NULL, num_dests, dests)) == NULL) + dest = dests; + + if (dest == NULL) + { + _cupsLangPuts(stderr, _("lpoptions: No printers.")); + return (1); + } + + for (j = 0; j < dest->num_options; j ++) + if (cupsGetOption(dest->options[j].name, num_options, options) == NULL) + num_options = cupsAddOption(dest->options[j].name, + dest->options[j].value, + num_options, &options); + } + + if (argv[i][2]) + num_options = cupsParseOptions(argv[i] + 2, num_options, &options); + else + { + i ++; + if (i >= argc) + usage(); + + num_options = cupsParseOptions(argv[i], num_options, &options); + } + + changes = 1; + break; + + case 'p' : /* -p printer */ + if (argv[i][2]) + printer = argv[i] + 2; + else + { + i ++; + if (i >= argc) + usage(); + + printer = argv[i]; + } + + if ((instance = strrchr(printer, '/')) != NULL) + *instance++ = '\0'; + + if (num_dests == 0) + num_dests = cupsGetDests(&dests); + + if ((dest = cupsGetDest(printer, instance, num_dests, dests)) == NULL) + { + num_dests = cupsAddDest(printer, instance, num_dests, &dests); + dest = cupsGetDest(printer, instance, num_dests, dests); + + if (dest == NULL) + { + _cupsLangPrintf(stderr, + _("lpoptions: Unable to add printer or " + "instance: %s"), + strerror(errno)); + return (1); + } + } + + for (j = 0; j < dest->num_options; j ++) + if (cupsGetOption(dest->options[j].name, num_options, options) == NULL) + num_options = cupsAddOption(dest->options[j].name, + dest->options[j].value, + num_options, &options); + break; + + case 'r' : /* -r option (remove) */ + if (dest == NULL) + { + if (num_dests == 0) + num_dests = cupsGetDests(&dests); + + if ((dest = cupsGetDest(NULL, NULL, num_dests, dests)) == NULL) + dest = dests; + + if (dest == NULL) + { + _cupsLangPuts(stderr, _("lpoptions: No printers.")); + return (1); + } + + for (j = 0; j < dest->num_options; j ++) + if (cupsGetOption(dest->options[j].name, num_options, + options) == NULL) + num_options = cupsAddOption(dest->options[j].name, + dest->options[j].value, + num_options, &options); + } + + if (argv[i][2]) + option = argv[i] + 2; + else + { + i ++; + if (i >= argc) + usage(); + + option = argv[i]; + } + + for (j = 0; j < num_options; j ++) + if (!_cups_strcasecmp(options[j].name, option)) + { + /* + * Remove this option... + */ + + num_options --; + + if (j < num_options) + memcpy(options + j, options + j + 1, + sizeof(cups_option_t) * (num_options - j)); + break; + } + + changes = 1; + break; + + case 'x' : /* -x printer */ + if (argv[i][2]) + printer = argv[i] + 2; + else + { + i ++; + if (i >= argc) + usage(); + + printer = argv[i]; + } + + if ((instance = strrchr(printer, '/')) != NULL) + *instance++ = '\0'; + + if (num_dests == 0) + num_dests = cupsGetDests(&dests); + + if ((dest = cupsGetDest(printer, instance, num_dests, + dests)) != NULL) + { + cupsFreeOptions(dest->num_options, dest->options); + + /* + * If we are "deleting" the default printer, then just set the + * number of options to 0; if it is also the system default + * then cupsSetDests() will remove it for us... + */ + + if (dest->is_default) + { + dest->num_options = 0; + dest->options = NULL; + } + else + { + num_dests --; + + j = dest - dests; + if (j < num_dests) + memcpy(dest, dest + 1, (num_dests - j) * sizeof(cups_dest_t)); + } + } + + cupsSetDests(num_dests, dests); + dest = NULL; + changes = -1; + break; + + default : + usage(); + } + } + else + usage(); + + if (num_dests == 0) + num_dests = cupsGetDests(&dests); + + if (dest == NULL) + { + if ((dest = cupsGetDest(NULL, NULL, num_dests, dests)) != NULL) + { + for (j = 0; j < dest->num_options; j ++) + if (cupsGetOption(dest->options[j].name, num_options, options) == NULL) + num_options = cupsAddOption(dest->options[j].name, + dest->options[j].value, + num_options, &options); + } + } + + if (dest == NULL) + return (0); + + if (changes > 0) + { + /* + * Set printer options... + */ + + cupsFreeOptions(dest->num_options, dest->options); + + dest->num_options = num_options; + dest->options = options; + + cupsSetDests(num_dests, dests); + } + else if (changes == 0) + { + char buffer[10240], /* String for options */ + *ptr; /* Pointer into string */ + + num_options = dest->num_options; + options = dest->options; + + for (i = 0, ptr = buffer; + ptr < (buffer + sizeof(buffer) - 1) && i < num_options; + i ++) + { + if (i) + *ptr++ = ' '; + + if (!options[i].value[0]) + strlcpy(ptr, options[i].name, sizeof(buffer) - (ptr - buffer)); + else if (strchr(options[i].value, ' ') != NULL || + strchr(options[i].value, '\t') != NULL) + snprintf(ptr, sizeof(buffer) - (ptr - buffer), "%s=\'%s\'", + options[i].name, options[i].value); + else + snprintf(ptr, sizeof(buffer) - (ptr - buffer), "%s=%s", + options[i].name, options[i].value); + + ptr += strlen(ptr); + } + + _cupsLangPuts(stdout, buffer); + } + + return (0); +} + +/* + * 'list_group()' - List printer-specific options from the PPD group. + */ + +static void +list_group(ppd_file_t *ppd, /* I - PPD file */ + ppd_group_t *group) /* I - Group to show */ +{ + int i, j; /* Looping vars */ + ppd_option_t *option; /* Current option */ + ppd_choice_t *choice; /* Current choice */ + ppd_group_t *subgroup; /* Current subgroup */ + char buffer[10240], /* Option string buffer */ + *ptr; /* Pointer into option string */ + + + for (i = group->num_options, option = group->options; i > 0; i --, option ++) + { + if (!_cups_strcasecmp(option->keyword, "PageRegion")) + continue; + + snprintf(buffer, sizeof(buffer), "%s/%s:", option->keyword, option->text); + + for (j = option->num_choices, choice = option->choices, + ptr = buffer + strlen(buffer); + j > 0 && ptr < (buffer + sizeof(buffer) - 1); + j --, choice ++) + { + if (!_cups_strcasecmp(choice->choice, "Custom")) + { + ppd_coption_t *coption; /* Custom option */ + ppd_cparam_t *cparam; /* Custom parameter */ + static const char * const types[] = + { /* Parameter types */ + "CURVE", + "INTEGER", + "INVCURVE", + "PASSCODE", + "PASSWORD", + "POINTS", + "REAL", + "STRING" + }; + + + if ((coption = ppdFindCustomOption(ppd, option->keyword)) == NULL || + cupsArrayCount(coption->params) == 0) + snprintf(ptr, sizeof(buffer) - (ptr - buffer), " %sCustom", + choice->marked ? "*" : ""); + else if (!_cups_strcasecmp(option->keyword, "PageSize") || + !_cups_strcasecmp(option->keyword, "PageRegion")) + snprintf(ptr, sizeof(buffer) - (ptr - buffer), + " %sCustom.WIDTHxHEIGHT", choice->marked ? "*" : ""); + else + { + cparam = (ppd_cparam_t *)cupsArrayFirst(coption->params); + + if (cupsArrayCount(coption->params) == 1) + snprintf(ptr, sizeof(buffer) - (ptr - buffer), " %sCustom.%s", + choice->marked ? "*" : "", types[cparam->type]); + else + { + const char *prefix; /* Prefix string */ + + + if (choice->marked) + prefix = " *{"; + else + prefix = " {"; + + while (cparam) + { + snprintf(ptr, sizeof(buffer) - (ptr - buffer), "%s%s=%s", prefix, + cparam->name, types[cparam->type]); + cparam = (ppd_cparam_t *)cupsArrayNext(coption->params); + prefix = " "; + ptr += strlen(ptr); + } + + if (ptr < (buffer + sizeof(buffer) - 1)) + strlcpy(ptr, "}", sizeof(buffer) - (ptr - buffer)); + } + } + } + else if (choice->marked) + snprintf(ptr, sizeof(buffer) - (ptr - buffer), " *%s", choice->choice); + else + snprintf(ptr, sizeof(buffer) - (ptr - buffer), " %s", choice->choice); + + ptr += strlen(ptr); + } + + _cupsLangPuts(stdout, buffer); + } + + for (i = group->num_subgroups, subgroup = group->subgroups; i > 0; i --, subgroup ++) + list_group(ppd, subgroup); +} + + +/* + * 'list_options()' - List printer-specific options from the PPD file. + */ + +static void +list_options(cups_dest_t *dest) /* I - Destination to list */ +{ + int i; /* Looping var */ + const char *filename; /* PPD filename */ + ppd_file_t *ppd; /* PPD data */ + ppd_group_t *group; /* Current group */ + + + if ((filename = cupsGetPPD(dest->name)) == NULL) + { + _cupsLangPrintf(stderr, _("lpoptions: Unable to get PPD file for %s: %s"), + dest->name, cupsLastErrorString()); + return; + } + + if ((ppd = ppdOpenFile(filename)) == NULL) + { + unlink(filename); + _cupsLangPrintf(stderr, _("lpoptions: Unable to open PPD file for %s."), + dest->name); + return; + } + + ppdMarkDefaults(ppd); + cupsMarkOptions(ppd, dest->num_options, dest->options); + + for (i = ppd->num_groups, group = ppd->groups; i > 0; i --, group ++) + list_group(ppd, group); + + ppdClose(ppd); + unlink(filename); +} + + +/* + * 'usage()' - Show program usage and exit. + */ + +static void +usage(void) +{ + _cupsLangPuts(stdout, + _("Usage: lpoptions [-h server] [-E] -d printer\n" + " lpoptions [-h server] [-E] [-p printer] -l\n" + " lpoptions [-h server] [-E] -p printer -o " + "option[=value] ...\n" + " lpoptions [-h server] [-E] -x printer")); + + exit(1); +} + + +/* + * End of "$Id$". + */ diff --git a/systemv/lppasswd.c b/systemv/lppasswd.c new file mode 100644 index 0000000000..597408a67f --- /dev/null +++ b/systemv/lppasswd.c @@ -0,0 +1,489 @@ +/* + * "$Id$" + * + * MD5 password program for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Add, change, or delete passwords from the MD5 password file. + * usage() - Show program usage. + */ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include +#include +#include + +#ifndef WIN32 +# include +# include +#endif /* !WIN32 */ + + +/* + * Operations... + */ + +#define ADD 0 +#define CHANGE 1 +#define DELETE 2 + + +/* + * Local functions... + */ + +static void usage(FILE *fp) __attribute__((noreturn)); + + +/* + * 'main()' - Add, change, or delete passwords from the MD5 password file. + */ + +int /* O - Exit status */ +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + char *opt; /* Option pointer */ + const char *username; /* Pointer to username */ + const char *groupname; /* Pointer to group name */ + int op; /* Operation (add, change, delete) */ + const char *passwd; /* Password string */ + FILE *infile, /* Input file */ + *outfile; /* Output file */ + char line[256], /* Line from file */ + userline[17], /* User from line */ + groupline[17], /* Group from line */ + md5line[33], /* MD5-sum from line */ + md5new[33]; /* New MD5 sum */ + char passwdmd5[1024], /* passwd.md5 file */ + passwdold[1024], /* passwd.old file */ + passwdnew[1024]; /* passwd.tmp file */ + char *newpass, /* new password */ + *oldpass; /* old password */ + int flag; /* Password check flags... */ + int fd; /* Password file descriptor */ + int error; /* Write error */ + _cups_globals_t *cg = _cupsGlobals(); /* Global data */ + cups_lang_t *lang; /* Language info */ +#if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET) + struct sigaction action; /* Signal action */ +#endif /* HAVE_SIGACTION && !HAVE_SIGSET*/ + + + _cupsSetLocale(argv); + lang = cupsLangDefault(); + + /* + * Check to see if stdin, stdout, and stderr are still open... + */ + + if (fcntl(0, F_GETFD, &i) || + fcntl(1, F_GETFD, &i) || + fcntl(2, F_GETFD, &i)) + { + /* + * No, return exit status 2 and don't try to send any output since + * someone is trying to bypass the security on the server. + */ + + return (2); + } + + /* + * Find the server directory... + */ + + snprintf(passwdmd5, sizeof(passwdmd5), "%s/passwd.md5", cg->cups_serverroot); + snprintf(passwdold, sizeof(passwdold), "%s/passwd.old", cg->cups_serverroot); + snprintf(passwdnew, sizeof(passwdnew), "%s/passwd.new", cg->cups_serverroot); + + /* + * Find the default system group... + */ + + if (getgrnam(CUPS_DEFAULT_GROUP)) + groupname = CUPS_DEFAULT_GROUP; + else + groupname = "unknown"; + + endgrent(); + + username = NULL; + op = CHANGE; + + /* + * Parse command-line options... + */ + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-') + for (opt = argv[i] + 1; *opt; opt ++) + switch (*opt) + { + case 'a' : /* Add */ + op = ADD; + break; + case 'x' : /* Delete */ + op = DELETE; + break; + case 'g' : /* Group */ + i ++; + if (i >= argc) + usage(stderr); + + groupname = argv[i]; + break; + case 'h' : /* Help */ + usage(stdout); + break; + default : /* Bad option */ + usage(stderr); + break; + } + else if (!username) + username = argv[i]; + else + usage(stderr); + + /* + * See if we are trying to add or delete a password when we aren't logged in + * as root... + */ + + if (getuid() && getuid() != geteuid() && (op != CHANGE || username)) + { + _cupsLangPuts(stderr, + _("lppasswd: Only root can add or delete passwords.")); + return (1); + } + + /* + * Fill in missing info... + */ + + if (!username) + username = cupsUser(); + + oldpass = newpass = NULL; + + /* + * Obtain old and new password _before_ locking the database + * to keep users from locking the file indefinitely. + */ + + if (op == CHANGE && getuid()) + { + if ((passwd = cupsGetPassword(_("Enter old password:"))) == NULL) + return (1); + + if ((oldpass = strdup(passwd)) == NULL) + { + _cupsLangPrintf(stderr, + _("lppasswd: Unable to copy password string: %s"), + strerror(errno)); + return (1); + } + } + + /* + * Now get the new password, if necessary... + */ + + if (op != DELETE) + { + if ((passwd = cupsGetPassword( + _cupsLangString(lang, _("Enter password:")))) == NULL) + return (1); + + if ((newpass = strdup(passwd)) == NULL) + { + _cupsLangPrintf(stderr, + _("lppasswd: Unable to copy password string: %s"), + strerror(errno)); + return (1); + } + + if ((passwd = cupsGetPassword( + _cupsLangString(lang, _("Enter password again:")))) == NULL) + return (1); + + if (strcmp(passwd, newpass) != 0) + { + _cupsLangPuts(stderr, + _("lppasswd: Sorry, passwords don't match.")); + return (1); + } + + /* + * Check that the password contains at least one letter and number. + */ + + flag = 0; + + for (passwd = newpass; *passwd; passwd ++) + if (isdigit(*passwd & 255)) + flag |= 1; + else if (isalpha(*passwd & 255)) + flag |= 2; + + /* + * Only allow passwords that are at least 6 chars, have a letter and + * a number, and don't contain the username. + */ + + if (strlen(newpass) < 6 || strstr(newpass, username) != NULL || flag != 3) + { + _cupsLangPuts(stderr, _("lppasswd: Sorry, password rejected.")); + _cupsLangPuts(stderr, _("Your password must be at least 6 characters " + "long, cannot contain your username, and must " + "contain at least one letter and number.")); + return (1); + } + } + + /* + * Ignore SIGHUP, SIGINT, SIGTERM, and SIGXFSZ (if defined) for the + * remainder of the time so that we won't end up with bogus password + * files... + */ + +#ifndef WIN32 +# if defined(HAVE_SIGSET) + sigset(SIGHUP, SIG_IGN); + sigset(SIGINT, SIG_IGN); + sigset(SIGTERM, SIG_IGN); +# ifdef SIGXFSZ + sigset(SIGXFSZ, SIG_IGN); +# endif /* SIGXFSZ */ +# elif defined(HAVE_SIGACTION) + memset(&action, 0, sizeof(action)); + action.sa_handler = SIG_IGN; + + sigaction(SIGHUP, &action, NULL); + sigaction(SIGINT, &action, NULL); + sigaction(SIGTERM, &action, NULL); +# ifdef SIGXFSZ + sigaction(SIGXFSZ, &action, NULL); +# endif /* SIGXFSZ */ +# else + signal(SIGHUP, SIG_IGN); + signal(SIGINT, SIG_IGN); + signal(SIGTERM, SIG_IGN); +# ifdef SIGXFSZ + signal(SIGXFSZ, SIG_IGN); +# endif /* SIGXFSZ */ +# endif +#endif /* !WIN32 */ + + /* + * Open the output file. + */ + + if ((fd = open(passwdnew, O_WRONLY | O_CREAT | O_EXCL, 0400)) < 0) + { + if (errno == EEXIST) + _cupsLangPuts(stderr, _("lppasswd: Password file busy.")); + else + _cupsLangPrintf(stderr, _("lppasswd: Unable to open password file: %s"), + strerror(errno)); + + return (1); + } + + if ((outfile = fdopen(fd, "w")) == NULL) + { + _cupsLangPrintf(stderr, _("lppasswd: Unable to open password file: %s"), + strerror(errno)); + + unlink(passwdnew); + + return (1); + } + + setbuf(outfile, NULL); + + /* + * Open the existing password file and create a new one... + */ + + infile = fopen(passwdmd5, "r"); + if (infile == NULL && errno != ENOENT && op != ADD) + { + _cupsLangPrintf(stderr, _("lppasswd: Unable to open password file: %s"), + strerror(errno)); + + fclose(outfile); + + unlink(passwdnew); + + return (1); + } + + /* + * Read lines from the password file; the format is: + * + * username:group:MD5-sum + */ + + error = 0; + userline[0] = '\0'; + groupline[0] = '\0'; + md5line[0] = '\0'; + + if (infile) + { + while (fgets(line, sizeof(line), infile) != NULL) + { + if (sscanf(line, "%16[^:]:%16[^:]:%32s", userline, groupline, md5line) != 3) + continue; + + if (strcmp(username, userline) == 0 && + strcmp(groupname, groupline) == 0) + break; + + if (fputs(line, outfile) == EOF) + { + _cupsLangPrintf(stderr, + _("lppasswd: Unable to write to password file: %s"), + strerror(errno)); + error = 1; + break; + } + } + + if (!error) + { + while (fgets(line, sizeof(line), infile) != NULL) + if (fputs(line, outfile) == EOF) + { + _cupsLangPrintf(stderr, + _("lppasswd: Unable to write to password file: %s"), + strerror(errno)); + error = 1; + break; + } + } + } + + if (op == CHANGE && + (strcmp(username, userline) || strcmp(groupname, groupline))) + { + _cupsLangPrintf(stderr, + _("lppasswd: user \"%s\" and group \"%s\" do not exist."), + username, groupname); + error = 1; + } + else if (op != DELETE) + { + if (oldpass && + strcmp(httpMD5(username, "CUPS", oldpass, md5new), md5line) != 0) + { + _cupsLangPuts(stderr, _("lppasswd: Sorry, password doesn't match.")); + error = 1; + } + else + { + snprintf(line, sizeof(line), "%s:%s:%s\n", username, groupname, + httpMD5(username, "CUPS", newpass, md5new)); + if (fputs(line, outfile) == EOF) + { + _cupsLangPrintf(stderr, + _("lppasswd: Unable to write to password file: %s"), + strerror(errno)); + error = 1; + } + } + } + + /* + * Close the files... + */ + + if (infile) + fclose(infile); + + if (fclose(outfile) == EOF) + error = 1; + + /* + * Error out gracefully as needed... + */ + + if (error) + { + _cupsLangPuts(stderr, _("lppasswd: Password file not updated.")); + + unlink(passwdnew); + + return (1); + } + + /* + * Save old passwd file + */ + + unlink(passwdold); + if (link(passwdmd5, passwdold) && errno != ENOENT) + { + _cupsLangPrintf(stderr, + _("lppasswd: failed to backup old password file: %s"), + strerror(errno)); + unlink(passwdnew); + return (1); + } + + /* + * Install new password file + */ + + if (rename(passwdnew, passwdmd5) < 0) + { + _cupsLangPrintf(stderr, _("lppasswd: failed to rename password file: %s"), + strerror(errno)); + unlink(passwdnew); + return (1); + } + + return (0); +} + + +/* + * 'usage()' - Show program usage. + */ + +static void +usage(FILE *fp) /* I - File to send usage to */ +{ + if (getuid()) + _cupsLangPuts(fp, _("Usage: lppasswd [-g groupname]")); + else + _cupsLangPuts(fp, + _("Usage: lppasswd [-g groupname] [username]\n" + " lppasswd [-g groupname] -a [username]\n" + " lppasswd [-g groupname] -x [username]")); + + exit(1); +} + + +/* + * End of "$Id$". + */ diff --git a/systemv/lpstat.c b/systemv/lpstat.c new file mode 100644 index 0000000000..124ce7c00e --- /dev/null +++ b/systemv/lpstat.c @@ -0,0 +1,2054 @@ +/* + * "$Id$" + * + * "lpstat" command for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2006 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * main() - Parse options and show status information. + * check_dest() - Verify that the named destination(s) exists. + * match_list() - Match a name from a list of comma or space-separated + * names. + * show_accepting() - Show acceptance status. + * show_classes() - Show printer classes. + * show_default() - Show default destination. + * show_devices() - Show printer devices. + * show_jobs() - Show active print jobs. + * show_printers() - Show printers. + * show_scheduler() - Show scheduler status. + */ + +/* + * Include necessary headers... + */ + +#include + + +/* + * Local functions... + */ + +static void check_dest(const char *command, const char *name, + int *num_dests, cups_dest_t **dests); +static int match_list(const char *list, const char *name); +static int show_accepting(const char *printers, int num_dests, + cups_dest_t *dests); +static int show_classes(const char *dests); +static void show_default(cups_dest_t *dest); +static int show_devices(const char *printers, int num_dests, + cups_dest_t *dests); +static int show_jobs(const char *dests, const char *users, int long_status, + int ranking, const char *which); +static int show_printers(const char *printers, int num_dests, + cups_dest_t *dests, int long_status); +static void show_scheduler(void); + + +/* + * 'main()' - Parse options and show status information. + */ + +int +main(int argc, /* I - Number of command-line arguments */ + char *argv[]) /* I - Command-line arguments */ +{ + int i, /* Looping var */ + status; /* Exit status */ + int num_dests; /* Number of user destinations */ + cups_dest_t *dests; /* User destinations */ + int long_status; /* Long status report? */ + int ranking; /* Show job ranking? */ + const char *which; /* Which jobs to show? */ + char op; /* Last operation on command-line */ + + + _cupsSetLocale(argv); + + /* + * Parse command-line options... + */ + + num_dests = 0; + dests = NULL; + long_status = 0; + ranking = 0; + status = 0; + which = "not-completed"; + op = 0; + + for (i = 1; i < argc; i ++) + if (argv[i][0] == '-') + switch (argv[i][1]) + { + case 'D' : /* Show description */ + long_status = 1; + break; + + case 'E' : /* Encrypt */ +#ifdef HAVE_SSL + cupsSetEncryption(HTTP_ENCRYPT_REQUIRED); +#else + _cupsLangPrintf(stderr, + _("%s: Sorry, no encryption support."), + argv[0]); +#endif /* HAVE_SSL */ + break; + + case 'H' : /* Show server and port */ + if (cupsServer()[0] == '/') + _cupsLangPuts(stdout, cupsServer()); + else + _cupsLangPrintf(stdout, "%s:%d", cupsServer(), ippPort()); + break; + + case 'P' : /* Show paper types */ + op = 'P'; + break; + + case 'R' : /* Show ranking */ + ranking = 1; + break; + + case 'S' : /* Show charsets */ + op = 'S'; + if (!argv[i][2]) + i ++; + break; + + case 'U' : /* Username */ + if (argv[i][2]) + cupsSetUser(argv[i] + 2); + else + { + i ++; + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected username after " + "\"-U\" option."), + argv[0]); + return (1); + } + + cupsSetUser(argv[i]); + } + break; + + case 'W' : /* Show which jobs? */ + if (argv[i][2]) + which = argv[i] + 2; + else + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - need \"completed\", " + "\"not-completed\", or \"all\" after " + "\"-W\" option."), + argv[0]); + return (1); + } + + which = argv[i]; + } + + if (strcmp(which, "completed") && strcmp(which, "not-completed") && + strcmp(which, "all")) + { + _cupsLangPrintf(stderr, + _("%s: Error - need \"completed\", " + "\"not-completed\", or \"all\" after " + "\"-W\" option."), + argv[0]); + return (1); + } + break; + + case 'a' : /* Show acceptance status */ + op = 'a'; + + if (argv[i][2]) + { + check_dest(argv[0], argv[i] + 2, &num_dests, &dests); + + status |= show_accepting(argv[i] + 2, num_dests, dests); + } + else if ((i + 1) < argc && argv[i + 1][0] != '-') + { + i ++; + + check_dest(argv[0], argv[i], &num_dests, &dests); + + status |= show_accepting(argv[i], num_dests, dests); + } + else + { + if (num_dests <= 1) + { + cupsFreeDests(num_dests, dests); + num_dests = cupsGetDests(&dests); + } + + status |= show_accepting(NULL, num_dests, dests); + } + break; + +#ifdef __sgi + case 'b' : /* Show both the local and remote status */ + op = 'b'; + + if (argv[i][2]) + { + /* + * The local and remote status are separated by a blank line; + * since all CUPS jobs are networked, we only output the + * second list for now... In the future, we might further + * emulate this by listing the remote server's queue, but + * for now this is enough to make the SGI printstatus program + * happy... + */ + + check_dest(argv[0], argv[i] + 2, &num_dests, &dests); + + puts(""); + status |= show_jobs(argv[i] + 2, NULL, 3, ranking, which); + } + else + { + _cupsLangPrintf(stderr, + _("%s: Error - expected destination after " + "\"-b\" option."), + argv[0]); + + return (1); + } + break; +#endif /* __sgi */ + + case 'c' : /* Show classes and members */ + op = 'c'; + + if (argv[i][2]) + { + check_dest(argv[0], argv[i] + 2, &num_dests, &dests); + + status |= show_classes(argv[i] + 2); + } + else if ((i + 1) < argc && argv[i + 1][0] != '-') + { + i ++; + + check_dest(argv[0], argv[i], &num_dests, &dests); + + status |= show_classes(argv[i]); + } + else + status |= show_classes(NULL); + break; + + case 'd' : /* Show default destination */ + op = 'd'; + + if (num_dests != 1 || !dests[0].is_default) + { + cupsFreeDests(num_dests, dests); + + dests = cupsGetNamedDest(CUPS_HTTP_DEFAULT, NULL, NULL); + num_dests = dests ? 1 : 0; + } + + show_default(dests); + break; + + case 'f' : /* Show forms */ + op = 'f'; + if (!argv[i][2]) + i ++; + break; + + case 'h' : /* Connect to host */ + if (argv[i][2]) + cupsSetServer(argv[i] + 2); + else + { + i ++; + + if (i >= argc) + { + _cupsLangPrintf(stderr, + _("%s: Error - expected hostname after " + "\"-h\" option."), + argv[0]); + return (1); + } + + cupsSetServer(argv[i]); + } + break; + + case 'l' : /* Long status or long job status */ +#ifdef __sgi + op = 'l'; + + if (argv[i][2]) + { + check_dest(argv[0], argv[i] + 2, &num_dests, &dests); + + status |= show_jobs(argv[i] + 2, NULL, 3, ranking, which); + } + else +#endif /* __sgi */ + long_status = 2; + break; + + case 'o' : /* Show jobs by destination */ + op = 'o'; + + if (argv[i][2]) + { + check_dest(argv[0], argv[i] + 2, &num_dests, &dests); + + status |= show_jobs(argv[i] + 2, NULL, long_status, ranking, + which); + } + else if ((i + 1) < argc && argv[i + 1][0] != '-') + { + i ++; + + check_dest(argv[0], argv[i], &num_dests, &dests); + + status |= show_jobs(argv[i], NULL, long_status, ranking, which); + } + else + status |= show_jobs(NULL, NULL, long_status, ranking, which); + break; + + case 'p' : /* Show printers */ + op = 'p'; + + if (argv[i][2]) + { + check_dest(argv[0], argv[i] + 2, &num_dests, &dests); + + status |= show_printers(argv[i] + 2, num_dests, dests, + long_status); + } + else if ((i + 1) < argc && argv[i + 1][0] != '-') + { + i ++; + + check_dest(argv[0], argv[i], &num_dests, &dests); + + status |= show_printers(argv[i], num_dests, dests, long_status); + } + else + { + if (num_dests <= 1) + { + cupsFreeDests(num_dests, dests); + num_dests = cupsGetDests(&dests); + } + + status |= show_printers(NULL, num_dests, dests, long_status); + } + break; + + case 'r' : /* Show scheduler status */ + op = 'r'; + + show_scheduler(); + break; + + case 's' : /* Show summary */ + op = 's'; + + if (num_dests <= 1) + { + cupsFreeDests(num_dests, dests); + num_dests = cupsGetDests(&dests); + } + + show_default(cupsGetDest(NULL, NULL, num_dests, dests)); + status |= show_classes(NULL); + status |= show_devices(NULL, num_dests, dests); + break; + + case 't' : /* Show all info */ + op = 't'; + + if (num_dests <= 1) + { + cupsFreeDests(num_dests, dests); + num_dests = cupsGetDests(&dests); + } + + show_scheduler(); + show_default(cupsGetDest(NULL, NULL, num_dests, dests)); + status |= show_classes(NULL); + status |= show_devices(NULL, num_dests, dests); + status |= show_accepting(NULL, num_dests, dests); + status |= show_printers(NULL, num_dests, dests, long_status); + status |= show_jobs(NULL, NULL, long_status, ranking, which); + break; + + case 'u' : /* Show jobs by user */ + op = 'u'; + + if (argv[i][2]) + status |= show_jobs(NULL, argv[i] + 2, long_status, ranking, + which); + else if ((i + 1) < argc && argv[i + 1][0] != '-') + { + i ++; + status |= show_jobs(NULL, argv[i], long_status, ranking, which); + } + else + status |= show_jobs(NULL, NULL, long_status, ranking, which); + break; + + case 'v' : /* Show printer devices */ + op = 'v'; + + if (argv[i][2]) + { + check_dest(argv[0], argv[i] + 2, &num_dests, &dests); + + status |= show_devices(argv[i] + 2, num_dests, dests); + } + else if ((i + 1) < argc && argv[i + 1][0] != '-') + { + i ++; + + check_dest(argv[0], argv[i], &num_dests, &dests); + + status |= show_devices(argv[i], num_dests, dests); + } + else + { + if (num_dests <= 1) + { + cupsFreeDests(num_dests, dests); + num_dests = cupsGetDests(&dests); + } + + status |= show_devices(NULL, num_dests, dests); + } + break; + + default : + _cupsLangPrintf(stderr, + _("%s: Error - unknown option \"%c\"."), + argv[0], argv[i][1]); + return (1); + } + else + { + status |= show_jobs(argv[i], NULL, long_status, ranking, which); + op = 'o'; + } + + if (!op) + status |= show_jobs(NULL, cupsUser(), long_status, ranking, which); + + return (status); +} + + +/* + * 'check_dest()' - Verify that the named destination(s) exists. + */ + +static void +check_dest(const char *command, /* I - Command name */ + const char *name, /* I - List of printer/class names */ + int *num_dests, /* IO - Number of destinations */ + cups_dest_t **dests) /* IO - Destinations */ +{ + const char *dptr; /* Pointer into name */ + char *pptr, /* Pointer into printer */ + printer[1024]; /* Current printer/class name */ + + + /* + * Load the destination list as necessary... + */ + + if (*num_dests <= 1) + { + if (*num_dests) + cupsFreeDests(*num_dests, *dests); + + if (strchr(name, ',')) + *num_dests = cupsGetDests(dests); + else + { + strlcpy(printer, name, sizeof(printer)); + if ((pptr = strchr(printer, '/')) != NULL) + *pptr++ = '\0'; + + if ((*dests = cupsGetNamedDest(CUPS_HTTP_DEFAULT, printer, pptr)) == NULL) + { + _cupsLangPrintf(stderr, + _("%s: Invalid destination name in list \"%s\"."), + command, name); + exit(1); + } + else + { + *num_dests = 1; + return; + } + } + } + + /* + * Scan the name string for printer/class name(s)... + */ + + for (dptr = name; *dptr;) + { + /* + * Skip leading whitespace and commas... + */ + + while (isspace(*dptr & 255) || *dptr == ',') + dptr ++; + + if (!*dptr) + break; + + /* + * Extract a single destination name from the name string... + */ + + for (pptr = printer; !isspace(*dptr & 255) && *dptr != ',' && *dptr;) + { + if ((pptr - printer) < (sizeof(printer) - 1)) + *pptr++ = *dptr++; + else + { + _cupsLangPrintf(stderr, + _("%s: Invalid destination name in list \"%s\"."), + command, name); + exit(1); + } + } + + *pptr = '\0'; + + /* + * Check the destination... + */ + + if (!cupsGetDest(printer, NULL, *num_dests, *dests)) + { + _cupsLangPrintf(stderr, + _("%s: Unknown destination \"%s\"."), command, printer); + exit(1); + } + } +} + + +/* + * 'match_list()' - Match a name from a list of comma or space-separated names. + */ + +static int /* O - 1 on match, 0 on no match */ +match_list(const char *list, /* I - List of names */ + const char *name) /* I - Name to find */ +{ + const char *nameptr; /* Pointer into name */ + + + /* + * An empty list always matches... + */ + + if (!list || !*list) + return (1); + + if (!name) + return (0); + + while (*list) + { + /* + * Skip leading whitespace and commas... + */ + + while (isspace(*list & 255) || *list == ',') + list ++; + + if (!*list) + break; + + /* + * Compare names... + */ + + for (nameptr = name; + *nameptr && *list && tolower(*nameptr & 255) == tolower(*list & 255); + nameptr ++, list ++); + + if (!*nameptr && (!*list || *list == ',' || isspace(*list & 255))) + return (1); + + while (*list && !isspace(*list & 255) && *list != ',') + list ++; + } + + return (0); +} + + +/* + * 'show_accepting()' - Show acceptance status. + */ + +static int /* O - 0 on success, 1 on fail */ +show_accepting(const char *printers, /* I - Destinations */ + int num_dests, /* I - Number of user-defined dests */ + cups_dest_t *dests) /* I - User-defined destinations */ +{ + int i; /* Looping var */ + ipp_t *request, /* IPP Request */ + *response; /* IPP Response */ + ipp_attribute_t *attr; /* Current attribute */ + const char *printer, /* Printer name */ + *message; /* Printer device URI */ + int accepting; /* Accepting requests? */ + time_t ptime; /* Printer state time */ + struct tm *pdate; /* Printer state date & time */ + char printer_state_time[255];/* Printer state time */ + static const char *pattrs[] = /* Attributes we need for printers... */ + { + "printer-name", + "printer-state-change-time", + "printer-state-message", + "printer-is-accepting-jobs" + }; + + + DEBUG_printf(("show_accepting(printers=\"%s\")\n", printers)); + + if (printers != NULL && !strcmp(printers, "all")) + printers = NULL; + + /* + * Build a CUPS_GET_PRINTERS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * requested-attributes + * requesting-user-name + */ + + request = ippNewRequest(CUPS_GET_PRINTERS); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", sizeof(pattrs) / sizeof(pattrs[0]), + NULL, pattrs); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, cupsUser()); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(CUPS_HTTP_DEFAULT, request, "/")) != NULL) + { + DEBUG_puts("show_accepting: request succeeded..."); + + if (response->request.status.status_code > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, "lpstat: %s", cupsLastErrorString()); + ippDelete(response); + return (1); + } + + /* + * Loop through the printers returned in the list and display + * their devices... + */ + + for (attr = response->attrs; attr != NULL; attr = attr->next) + { + /* + * Skip leading attributes until we hit a printer... + */ + + while (attr != NULL && attr->group_tag != IPP_TAG_PRINTER) + attr = attr->next; + + if (attr == NULL) + break; + + /* + * Pull the needed attributes from this printer... + */ + + printer = NULL; + message = NULL; + accepting = 1; + ptime = 0; + + while (attr != NULL && attr->group_tag == IPP_TAG_PRINTER) + { + if (!strcmp(attr->name, "printer-name") && + attr->value_tag == IPP_TAG_NAME) + printer = attr->values[0].string.text; + else if (!strcmp(attr->name, "printer-state-change-time") && + attr->value_tag == IPP_TAG_INTEGER) + ptime = (time_t)attr->values[0].integer; + else if (!strcmp(attr->name, "printer-state-message") && + attr->value_tag == IPP_TAG_TEXT) + message = attr->values[0].string.text; + else if (!strcmp(attr->name, "printer-is-accepting-jobs") && + attr->value_tag == IPP_TAG_BOOLEAN) + accepting = attr->values[0].boolean; + + attr = attr->next; + } + + /* + * See if we have everything needed... + */ + + if (printer == NULL) + { + if (attr == NULL) + break; + else + continue; + } + + /* + * Display the printer entry if needed... + */ + + if (match_list(printers, printer)) + { + pdate = localtime(&ptime); + strftime(printer_state_time, sizeof(printer_state_time), "%c", pdate); + + if (accepting) + _cupsLangPrintf(stdout, _("%s accepting requests since %s"), + printer, printer_state_time); + else + { + _cupsLangPrintf(stdout, _("%s not accepting requests since %s -"), + printer, printer_state_time); + _cupsLangPrintf(stdout, _("\t%s"), + (message == NULL || !*message) ? + "reason unknown" : message); + } + + for (i = 0; i < num_dests; i ++) + if (!_cups_strcasecmp(dests[i].name, printer) && dests[i].instance) + { + if (accepting) + _cupsLangPrintf(stdout, _("%s/%s accepting requests since %s"), + printer, dests[i].instance, printer_state_time); + else + { + _cupsLangPrintf(stdout, + _("%s/%s not accepting requests since %s -"), + printer, dests[i].instance, printer_state_time); + _cupsLangPrintf(stdout, _("\t%s"), + (message == NULL || !*message) ? + "reason unknown" : message); + } + } + } + + if (attr == NULL) + break; + } + + ippDelete(response); + } + else + { + _cupsLangPrintf(stderr, "lpstat: %s", cupsLastErrorString()); + return (1); + } + + return (0); +} + + +/* + * 'show_classes()' - Show printer classes. + */ + +static int /* O - 0 on success, 1 on fail */ +show_classes(const char *dests) /* I - Destinations */ +{ + int i; /* Looping var */ + ipp_t *request, /* IPP Request */ + *response, /* IPP Response */ + *response2; /* IPP response from remote server */ + http_t *http2; /* Remote server */ + ipp_attribute_t *attr; /* Current attribute */ + const char *printer, /* Printer class name */ + *printer_uri; /* Printer class URI */ + ipp_attribute_t *members; /* Printer members */ + char method[HTTP_MAX_URI], /* Request method */ + username[HTTP_MAX_URI], /* Username:password */ + server[HTTP_MAX_URI], /* Server name */ + resource[HTTP_MAX_URI]; /* Resource name */ + int port; /* Port number */ + static const char *cattrs[] = /* Attributes we need for classes... */ + { + "printer-name", + "printer-uri-supported", + "member-names" + }; + + + DEBUG_printf(("show_classes(dests=\"%s\")\n", dests)); + + if (dests != NULL && !strcmp(dests, "all")) + dests = NULL; + + /* + * Build a CUPS_GET_CLASSES request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * requested-attributes + * requesting-user-name + */ + + request = ippNewRequest(CUPS_GET_CLASSES); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", sizeof(cattrs) / sizeof(cattrs[0]), + NULL, cattrs); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, cupsUser()); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(CUPS_HTTP_DEFAULT, request, "/")) != NULL) + { + DEBUG_puts("show_classes: request succeeded..."); + + if (response->request.status.status_code > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, "lpstat: %s", cupsLastErrorString()); + ippDelete(response); + return (1); + } + + /* + * Loop through the printers returned in the list and display + * their devices... + */ + + for (attr = response->attrs; attr != NULL; attr = attr->next) + { + /* + * Skip leading attributes until we hit a job... + */ + + while (attr != NULL && attr->group_tag != IPP_TAG_PRINTER) + attr = attr->next; + + if (attr == NULL) + break; + + /* + * Pull the needed attributes from this job... + */ + + printer = NULL; + printer_uri = NULL; + members = NULL; + + while (attr != NULL && attr->group_tag == IPP_TAG_PRINTER) + { + if (!strcmp(attr->name, "printer-name") && + attr->value_tag == IPP_TAG_NAME) + printer = attr->values[0].string.text; + + if (!strcmp(attr->name, "printer-uri-supported") && + attr->value_tag == IPP_TAG_URI) + printer_uri = attr->values[0].string.text; + + if (!strcmp(attr->name, "member-names") && + attr->value_tag == IPP_TAG_NAME) + members = attr; + + attr = attr->next; + } + + /* + * If this is a remote class, grab the class info from the + * remote server... + */ + + response2 = NULL; + if (members == NULL && printer_uri != NULL) + { + httpSeparateURI(HTTP_URI_CODING_ALL, printer_uri, method, sizeof(method), + username, sizeof(username), server, sizeof(server), + &port, resource, sizeof(resource)); + + if (!_cups_strcasecmp(server, cupsServer())) + http2 = CUPS_HTTP_DEFAULT; + else + http2 = httpConnectEncrypt(server, port, cupsEncryption()); + + /* + * Build an IPP_GET_PRINTER_ATTRIBUTES request, which requires the + * following attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requested-attributes + */ + + request = ippNewRequest(IPP_GET_PRINTER_ATTRIBUTES); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, printer_uri); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", + sizeof(cattrs) / sizeof(cattrs[0]), + NULL, cattrs); + + if ((response2 = cupsDoRequest(http2, request, "/")) != NULL) + members = ippFindAttribute(response2, "member-names", IPP_TAG_NAME); + + if (http2) + httpClose(http2); + } + + /* + * See if we have everything needed... + */ + + if (printer == NULL) + { + if (response2) + ippDelete(response2); + + if (attr == NULL) + break; + else + continue; + } + + /* + * Display the printer entry if needed... + */ + + if (match_list(dests, printer)) + { + _cupsLangPrintf(stdout, _("members of class %s:"), printer); + + if (members) + { + for (i = 0; i < members->num_values; i ++) + _cupsLangPrintf(stdout, "\t%s", members->values[i].string.text); + } + else + _cupsLangPuts(stdout, "\tunknown"); + } + + if (response2) + ippDelete(response2); + + if (attr == NULL) + break; + } + + ippDelete(response); + } + else + { + _cupsLangPrintf(stderr, "lpstat: %s", cupsLastErrorString()); + return (1); + } + + return (0); +} + + +/* + * 'show_default()' - Show default destination. + */ + +static void +show_default(cups_dest_t *dest) /* I - Default destination */ +{ + const char *printer, /* Printer name */ + *val; /* Environment variable name */ + + + if (dest) + { + if (dest->instance) + _cupsLangPrintf(stdout, _("system default destination: %s/%s"), + dest->name, dest->instance); + else + _cupsLangPrintf(stdout, _("system default destination: %s"), + dest->name); + } + else + { + val = NULL; + + if ((printer = getenv("LPDEST")) == NULL) + { + if ((printer = getenv("PRINTER")) != NULL) + { + if (!strcmp(printer, "lp")) + printer = NULL; + else + val = "PRINTER"; + } + } + else + val = "LPDEST"; + + if (printer) + _cupsLangPrintf(stdout, + _("lpstat: error - %s environment variable names " + "non-existent destination \"%s\"."), + val, printer); + else + _cupsLangPuts(stdout, _("no system default destination")); + } +} + + +/* + * 'show_devices()' - Show printer devices. + */ + +static int /* O - 0 on success, 1 on fail */ +show_devices(const char *printers, /* I - Destinations */ + int num_dests, /* I - Number of user-defined dests */ + cups_dest_t *dests) /* I - User-defined destinations */ +{ + int i; /* Looping var */ + ipp_t *request, /* IPP Request */ + *response; /* IPP Response */ + ipp_attribute_t *attr; /* Current attribute */ + const char *printer, /* Printer name */ + *uri, /* Printer URI */ + *device; /* Printer device URI */ + static const char *pattrs[] = /* Attributes we need for printers... */ + { + "printer-name", + "printer-uri-supported", + "device-uri" + }; + + + DEBUG_printf(("show_devices(printers=\"%s\")\n", printers)); + + if (printers != NULL && !strcmp(printers, "all")) + printers = NULL; + + /* + * Build a CUPS_GET_PRINTERS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * requested-attributes + * requesting-user-name + */ + + request = ippNewRequest(CUPS_GET_PRINTERS); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", sizeof(pattrs) / sizeof(pattrs[0]), + NULL, pattrs); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, cupsUser()); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(CUPS_HTTP_DEFAULT, request, "/")) != NULL) + { + DEBUG_puts("show_devices: request succeeded..."); + + if (response->request.status.status_code > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, "lpstat: %s", cupsLastErrorString()); + ippDelete(response); + return (1); + } + + /* + * Loop through the printers returned in the list and display + * their devices... + */ + + for (attr = response->attrs; attr != NULL; attr = attr->next) + { + /* + * Skip leading attributes until we hit a job... + */ + + while (attr != NULL && attr->group_tag != IPP_TAG_PRINTER) + attr = attr->next; + + if (attr == NULL) + break; + + /* + * Pull the needed attributes from this job... + */ + + printer = NULL; + device = NULL; + uri = NULL; + + while (attr != NULL && attr->group_tag == IPP_TAG_PRINTER) + { + if (!strcmp(attr->name, "printer-name") && + attr->value_tag == IPP_TAG_NAME) + printer = attr->values[0].string.text; + + if (!strcmp(attr->name, "printer-uri-supported") && + attr->value_tag == IPP_TAG_URI) + uri = attr->values[0].string.text; + + if (!strcmp(attr->name, "device-uri") && + attr->value_tag == IPP_TAG_URI) + device = attr->values[0].string.text; + + attr = attr->next; + } + + /* + * See if we have everything needed... + */ + + if (printer == NULL) + { + if (attr == NULL) + break; + else + continue; + } + + /* + * Display the printer entry if needed... + */ + + if (match_list(printers, printer)) + { +#ifdef __osf__ /* Compaq/Digital like to do it their own way... */ + char scheme[HTTP_MAX_URI], /* Components of printer URI */ + username[HTTP_MAX_URI], + hostname[HTTP_MAX_URI], + resource[HTTP_MAX_URI]; + int port; + + + if (device == NULL) + { + httpSeparateURI(HTTP_URI_CODING_ALL, uri, scheme, sizeof(scheme), + username, sizeof(username), hostname, + sizeof(hostname), &port, resource, sizeof(resource)); + _cupsLangPrintf(stdout, + _("Output for printer %s is sent to remote " + "printer %s on %s"), + printer, strrchr(resource, '/') + 1, hostname); + } + else if (!strncmp(device, "file:", 5)) + _cupsLangPrintf(stdout, + _("Output for printer %s is sent to %s"), + printer, device + 5); + else + _cupsLangPrintf(stdout, + _("Output for printer %s is sent to %s"), + printer, device); + + for (i = 0; i < num_dests; i ++) + if (!_cups_strcasecmp(printer, dests[i].name) && dests[i].instance) + { + if (device == NULL) + _cupsLangPrintf(stdout, + _("Output for printer %s/%s is sent to " + "remote printer %s on %s"), + printer, dests[i].instance, + strrchr(resource, '/') + 1, hostname); + else if (!strncmp(device, "file:", 5)) + _cupsLangPrintf(stdout, + _("Output for printer %s/%s is sent to %s"), + printer, dests[i].instance, device + 5); + else + _cupsLangPrintf(stdout, + _("Output for printer %s/%s is sent to %s"), + printer, dests[i].instance, device); + } +#else + if (device == NULL) + _cupsLangPrintf(stdout, _("device for %s: %s"), + printer, uri); + else if (!strncmp(device, "file:", 5)) + _cupsLangPrintf(stdout, _("device for %s: %s"), + printer, device + 5); + else + _cupsLangPrintf(stdout, _("device for %s: %s"), + printer, device); + + for (i = 0; i < num_dests; i ++) + if (!_cups_strcasecmp(printer, dests[i].name) && dests[i].instance) + { + if (device == NULL) + _cupsLangPrintf(stdout, _("device for %s/%s: %s"), + printer, dests[i].instance, uri); + else if (!strncmp(device, "file:", 5)) + _cupsLangPrintf(stdout, _("device for %s/%s: %s"), + printer, dests[i].instance, device + 5); + else + _cupsLangPrintf(stdout, _("device for %s/%s: %s"), + printer, dests[i].instance, device); + } +#endif /* __osf__ */ + } + + if (attr == NULL) + break; + } + + ippDelete(response); + } + else + { + _cupsLangPrintf(stderr, "lpstat: %s", cupsLastErrorString()); + return (1); + } + + return (0); +} + + +/* + * 'show_jobs()' - Show active print jobs. + */ + +static int /* O - 0 on success, 1 on fail */ +show_jobs(const char *dests, /* I - Destinations */ + const char *users, /* I - Users */ + int long_status, /* I - Show long status? */ + int ranking, /* I - Show job ranking? */ + const char *which) /* I - Show which jobs? */ +{ + int i; /* Looping var */ + ipp_t *request, /* IPP Request */ + *response; /* IPP Response */ + ipp_attribute_t *attr, /* Current attribute */ + *reasons; /* Job state reasons attribute */ + const char *dest, /* Pointer into job-printer-uri */ + *username, /* Pointer to job-originating-user-name */ + *title, /* Pointer to job-name */ + *message; /* Pointer to job-printer-state-message */ + int rank, /* Rank in queue */ + jobid, /* job-id */ + size; /* job-k-octets */ + time_t jobtime; /* time-at-creation */ + struct tm *jobdate; /* Date & time */ + char temp[255], /* Temporary buffer */ + date[255]; /* Date buffer */ + static const char *jattrs[] = /* Attributes we need for jobs... */ + { + "job-id", + "job-k-octets", + "job-name", + "job-originating-user-name", + "job-printer-state-message", + "job-printer-uri", + "job-state-reasons", + "time-at-creation" + }; + + + DEBUG_printf(("show_jobs(dests=\"%s\", users=\"%s\", long_status=%d, " + "ranking=%d, which=\"%s\")\n", dests, users, long_status, + ranking, which)); + + if (dests != NULL && !strcmp(dests, "all")) + dests = NULL; + + /* + * Build a IPP_GET_JOBS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * requested-attributes + * requesting-user-name + * which-jobs + */ + + request = ippNewRequest(IPP_GET_JOBS); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", + NULL, "ipp://localhost/"); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", sizeof(jattrs) / sizeof(jattrs[0]), + NULL, jattrs); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, cupsUser()); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, "which-jobs", + NULL, which); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(CUPS_HTTP_DEFAULT, request, "/")) != NULL) + { + /* + * Loop through the job list and display them... + */ + + if (response->request.status.status_code > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, "lpstat: %s", cupsLastErrorString()); + ippDelete(response); + return (1); + } + + rank = -1; + + for (attr = response->attrs; attr != NULL; attr = attr->next) + { + /* + * Skip leading attributes until we hit a job... + */ + + while (attr != NULL && attr->group_tag != IPP_TAG_JOB) + attr = attr->next; + + if (attr == NULL) + break; + + /* + * Pull the needed attributes from this job... + */ + + jobid = 0; + size = 0; + username = NULL; + dest = NULL; + jobtime = 0; + title = "no title"; + message = NULL; + reasons = NULL; + + while (attr != NULL && attr->group_tag == IPP_TAG_JOB) + { + if (!strcmp(attr->name, "job-id") && + attr->value_tag == IPP_TAG_INTEGER) + jobid = attr->values[0].integer; + else if (!strcmp(attr->name, "job-k-octets") && + attr->value_tag == IPP_TAG_INTEGER) + size = attr->values[0].integer; + else if (!strcmp(attr->name, "time-at-creation") && + attr->value_tag == IPP_TAG_INTEGER) + jobtime = attr->values[0].integer; + else if (!strcmp(attr->name, "job-printer-state-message") && + attr->value_tag == IPP_TAG_TEXT) + message = attr->values[0].string.text; + else if (!strcmp(attr->name, "job-printer-uri") && + attr->value_tag == IPP_TAG_URI) + { + if ((dest = strrchr(attr->values[0].string.text, '/')) != NULL) + dest ++; + } + else if (!strcmp(attr->name, "job-originating-user-name") && + attr->value_tag == IPP_TAG_NAME) + username = attr->values[0].string.text; + else if (!strcmp(attr->name, "job-name") && + attr->value_tag == IPP_TAG_NAME) + title = attr->values[0].string.text; + else if (!strcmp(attr->name, "job-state-reasons") && + attr->value_tag == IPP_TAG_KEYWORD) + reasons = attr; + + attr = attr->next; + } + + /* + * See if we have everything needed... + */ + + if (dest == NULL || jobid == 0) + { + if (attr == NULL) + break; + else + continue; + } + + /* + * Display the job... + */ + + rank ++; + + if (match_list(dests, dest) && match_list(users, username)) + { + jobdate = localtime(&jobtime); + snprintf(temp, sizeof(temp), "%s-%d", dest, jobid); + + if (long_status == 3) + { + /* + * Show the consolidated output format for the SGI tools... + */ + + if (!strftime(date, sizeof(date), "%b %d %H:%M", jobdate)) + strcpy(date, "Unknown"); + + _cupsLangPrintf(stdout, "%s;%s;%d;%s;%s", + temp, username ? username : "unknown", + size, title ? title : "unknown", date); + } + else + { + if (!strftime(date, sizeof(date), "%c", jobdate)) + strcpy(date, "Unknown"); + + if (ranking) + _cupsLangPrintf(stdout, "%3d %-21s %-13s %8.0f %s", + rank, temp, username ? username : "unknown", + 1024.0 * size, date); + else + _cupsLangPrintf(stdout, "%-23s %-13s %8.0f %s", + temp, username ? username : "unknown", + 1024.0 * size, date); + if (long_status) + { + if (message) + _cupsLangPrintf(stdout, _("\tStatus: %s"), message); + + if (reasons) + { + char alerts[1024], /* Alerts string */ + *aptr; /* Pointer into alerts string */ + + for (i = 0, aptr = alerts; i < reasons->num_values; i ++) + { + if (i) + snprintf(aptr, sizeof(alerts) - (aptr - alerts), " %s", + reasons->values[i].string.text); + else + strlcpy(alerts, reasons->values[i].string.text, + sizeof(alerts)); + + aptr += strlen(aptr); + } + + _cupsLangPrintf(stdout, _("\tAlerts: %s"), alerts); + } + + _cupsLangPrintf(stdout, _("\tqueued for %s"), dest); + } + } + } + + if (attr == NULL) + break; + } + + ippDelete(response); + } + else + { + _cupsLangPrintf(stderr, "lpstat: %s", cupsLastErrorString()); + return (1); + } + + return (0); +} + + +/* + * 'show_printers()' - Show printers. + */ + +static int /* O - 0 on success, 1 on fail */ +show_printers(const char *printers, /* I - Destinations */ + int num_dests, /* I - Number of user-defined dests */ + cups_dest_t *dests, /* I - User-defined destinations */ + int long_status) /* I - Show long status? */ +{ + int i, j; /* Looping vars */ + ipp_t *request, /* IPP Request */ + *response, /* IPP Response */ + *jobs; /* IPP Get Jobs response */ + ipp_attribute_t *attr, /* Current attribute */ + *jobattr, /* Job ID attribute */ + *reasons; /* Job state reasons attribute */ + const char *printer, /* Printer name */ + *message, /* Printer state message */ + *description, /* Description of printer */ + *location, /* Location of printer */ + *make_model, /* Make and model of printer */ + *uri; /* URI of printer */ + ipp_attribute_t *allowed, /* requesting-user-name-allowed */ + *denied; /* requestint-user-name-denied */ + ipp_pstate_t pstate; /* Printer state */ + cups_ptype_t ptype; /* Printer type */ + time_t ptime; /* Printer state time */ + struct tm *pdate; /* Printer state date & time */ + int jobid; /* Job ID of current job */ + char printer_uri[HTTP_MAX_URI], + /* Printer URI */ + printer_state_time[255];/* Printer state time */ + _cups_globals_t *cg = _cupsGlobals(); /* Global data */ + static const char *pattrs[] = /* Attributes we need for printers... */ + { + "printer-name", + "printer-state", + "printer-state-message", + "printer-state-reasons", + "printer-state-change-time", + "printer-type", + "printer-info", + "printer-location", + "printer-make-and-model", + "printer-uri-supported", + "requesting-user-name-allowed", + "requesting-user-name-denied" + }; + static const char *jattrs[] = /* Attributes we need for jobs... */ + { + "job-id", + "job-state" + }; + + + DEBUG_printf(("show_printers(printers=\"%s\", num_dests=%d, dests=%p, " + "long_status=%d)\n", printers, num_dests, dests, long_status)); + + if (printers != NULL && !strcmp(printers, "all")) + printers = NULL; + + /* + * Build a CUPS_GET_PRINTERS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * requested-attributes + * requesting-user-name + */ + + request = ippNewRequest(CUPS_GET_PRINTERS); + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", sizeof(pattrs) / sizeof(pattrs[0]), + NULL, pattrs); + + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "requesting-user-name", + NULL, cupsUser()); + + /* + * Do the request and get back a response... + */ + + if ((response = cupsDoRequest(CUPS_HTTP_DEFAULT, request, "/")) != NULL) + { + DEBUG_puts("show_printers: request succeeded..."); + + if (response->request.status.status_code > IPP_OK_CONFLICT) + { + _cupsLangPrintf(stderr, "lpstat: %s", cupsLastErrorString()); + ippDelete(response); + return (1); + } + + /* + * Loop through the printers returned in the list and display + * their status... + */ + + for (attr = response->attrs; attr != NULL; attr = attr->next) + { + /* + * Skip leading attributes until we hit a job... + */ + + while (attr != NULL && attr->group_tag != IPP_TAG_PRINTER) + attr = attr->next; + + if (attr == NULL) + break; + + /* + * Pull the needed attributes from this job... + */ + + printer = NULL; + ptime = 0; + ptype = CUPS_PRINTER_LOCAL; + pstate = IPP_PRINTER_IDLE; + message = NULL; + description = NULL; + location = NULL; + make_model = NULL; + reasons = NULL; + uri = NULL; + jobid = 0; + allowed = NULL; + denied = NULL; + + while (attr != NULL && attr->group_tag == IPP_TAG_PRINTER) + { + if (!strcmp(attr->name, "printer-name") && + attr->value_tag == IPP_TAG_NAME) + printer = attr->values[0].string.text; + else if (!strcmp(attr->name, "printer-state") && + attr->value_tag == IPP_TAG_ENUM) + pstate = (ipp_pstate_t)attr->values[0].integer; + else if (!strcmp(attr->name, "printer-type") && + attr->value_tag == IPP_TAG_ENUM) + ptype = (cups_ptype_t)attr->values[0].integer; + else if (!strcmp(attr->name, "printer-state-message") && + attr->value_tag == IPP_TAG_TEXT) + message = attr->values[0].string.text; + else if (!strcmp(attr->name, "printer-state-change-time") && + attr->value_tag == IPP_TAG_INTEGER) + ptime = (time_t)attr->values[0].integer; + else if (!strcmp(attr->name, "printer-info") && + attr->value_tag == IPP_TAG_TEXT) + description = attr->values[0].string.text; + else if (!strcmp(attr->name, "printer-location") && + attr->value_tag == IPP_TAG_TEXT) + location = attr->values[0].string.text; + else if (!strcmp(attr->name, "printer-make-and-model") && + attr->value_tag == IPP_TAG_TEXT) + make_model = attr->values[0].string.text; + else if (!strcmp(attr->name, "printer-uri-supported") && + attr->value_tag == IPP_TAG_URI) + uri = attr->values[0].string.text; + else if (!strcmp(attr->name, "printer-state-reasons") && + attr->value_tag == IPP_TAG_KEYWORD) + reasons = attr; + else if (!strcmp(attr->name, "requesting-user-name-allowed") && + attr->value_tag == IPP_TAG_NAME) + allowed = attr; + else if (!strcmp(attr->name, "requesting-user-name-denied") && + attr->value_tag == IPP_TAG_NAME) + denied = attr; + + attr = attr->next; + } + + /* + * See if we have everything needed... + */ + + if (printer == NULL) + { + if (attr == NULL) + break; + else + continue; + } + + /* + * Display the printer entry if needed... + */ + + if (match_list(printers, printer)) + { + /* + * If the printer state is "IPP_PRINTER_PROCESSING", then grab the + * current job for the printer. + */ + + if (pstate == IPP_PRINTER_PROCESSING) + { + /* + * Build an IPP_GET_JOBS request, which requires the following + * attributes: + * + * attributes-charset + * attributes-natural-language + * printer-uri + * limit + * requested-attributes + */ + + request = ippNewRequest(IPP_GET_JOBS); + + request->request.op.operation_id = IPP_GET_JOBS; + request->request.op.request_id = 1; + + ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, + "requested-attributes", + sizeof(jattrs) / sizeof(jattrs[0]), NULL, jattrs); + + httpAssembleURIf(HTTP_URI_CODING_ALL, printer_uri, sizeof(printer_uri), + "ipp", NULL, "localhost", 0, "/printers/%s", printer); + ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, + "printer-uri", NULL, printer_uri); + + if ((jobs = cupsDoRequest(CUPS_HTTP_DEFAULT, request, "/")) != NULL) + { + /* + * Get the current active job on this queue... + */ + + ipp_jstate_t jobstate = IPP_JOB_PENDING; + jobid = 0; + + for (jobattr = jobs->attrs; jobattr; jobattr = jobattr->next) + { + if (!jobattr->name) + { + if (jobstate == IPP_JOB_PROCESSING) + break; + else + continue; + } + + if (!strcmp(jobattr->name, "job-id") && + jobattr->value_tag == IPP_TAG_INTEGER) + jobid = jobattr->values[0].integer; + else if (!strcmp(jobattr->name, "job-state") && + jobattr->value_tag == IPP_TAG_ENUM) + jobstate = jobattr->values[0].integer; + } + + if (jobstate != IPP_JOB_PROCESSING) + jobid = 0; + + ippDelete(jobs); + } + } + + /* + * Display it... + */ + + pdate = localtime(&ptime); + strftime(printer_state_time, sizeof(printer_state_time), "%c", pdate); + + switch (pstate) + { + case IPP_PRINTER_IDLE : + _cupsLangPrintf(stdout, + _("printer %s is idle. enabled since %s"), + printer, printer_state_time); + break; + case IPP_PRINTER_PROCESSING : + _cupsLangPrintf(stdout, + _("printer %s now printing %s-%d. " + "enabled since %s"), + printer, printer, jobid, printer_state_time); + break; + case IPP_PRINTER_STOPPED : + _cupsLangPrintf(stdout, + _("printer %s disabled since %s -"), + printer, printer_state_time); + break; + } + + if ((message && *message) || pstate == IPP_PRINTER_STOPPED) + { + if (!message || !*message) + _cupsLangPuts(stdout, _("\treason unknown")); + else + _cupsLangPrintf(stdout, "\t%s", message); + } + + if (long_status > 1) + { + _cupsLangPuts(stdout, _("\tForm mounted:")); + _cupsLangPuts(stdout, _("\tContent types: any")); + _cupsLangPuts(stdout, _("\tPrinter types: unknown")); + } + + if (long_status) + { + _cupsLangPrintf(stdout, _("\tDescription: %s"), + description ? description : ""); + + if (reasons) + { + char alerts[1024], /* Alerts string */ + *aptr; /* Pointer into alerts string */ + + for (i = 0, aptr = alerts; i < reasons->num_values; i ++) + { + if (i) + snprintf(aptr, sizeof(alerts) - (aptr - alerts), " %s", + reasons->values[i].string.text); + else + strlcpy(alerts, reasons->values[i].string.text, + sizeof(alerts)); + + aptr += strlen(aptr); + } + + _cupsLangPrintf(stdout, _("\tAlerts: %s"), alerts); + } + } + if (long_status > 1) + { + _cupsLangPrintf(stdout, _("\tLocation: %s"), + location ? location : ""); + + if (ptype & CUPS_PRINTER_REMOTE) + { + _cupsLangPuts(stdout, _("\tConnection: remote")); + + if (make_model && !strstr(make_model, "System V Printer") && + !strstr(make_model, "Raw Printer") && uri) + _cupsLangPrintf(stdout, _("\tInterface: %s.ppd"), + uri); + } + else + { + _cupsLangPuts(stdout, _("\tConnection: direct")); + + if (make_model && strstr(make_model, "System V Printer")) + _cupsLangPrintf(stdout, + _("\tInterface: %s/interfaces/%s"), + cg->cups_serverroot, printer); + else if (make_model && !strstr(make_model, "Raw Printer")) + _cupsLangPrintf(stdout, + _("\tInterface: %s/ppd/%s.ppd"), + cg->cups_serverroot, printer); + } + _cupsLangPuts(stdout, _("\tOn fault: no alert")); + _cupsLangPuts(stdout, _("\tAfter fault: continue")); + /* TODO update to use printer-error-policy */ + if (allowed) + { + _cupsLangPuts(stdout, _("\tUsers allowed:")); + for (j = 0; j < allowed->num_values; j ++) + _cupsLangPrintf(stdout, "\t\t%s", + allowed->values[j].string.text); + } + else if (denied) + { + _cupsLangPuts(stdout, _("\tUsers denied:")); + for (j = 0; j < denied->num_values; j ++) + _cupsLangPrintf(stdout, "\t\t%s", + denied->values[j].string.text); + } + else + { + _cupsLangPuts(stdout, _("\tUsers allowed:")); + _cupsLangPuts(stdout, _("\t\t(all)")); + } + _cupsLangPuts(stdout, _("\tForms allowed:")); + _cupsLangPuts(stdout, _("\t\t(none)")); + _cupsLangPuts(stdout, _("\tBanner required")); + _cupsLangPuts(stdout, _("\tCharset sets:")); + _cupsLangPuts(stdout, _("\t\t(none)")); + _cupsLangPuts(stdout, _("\tDefault pitch:")); + _cupsLangPuts(stdout, _("\tDefault page size:")); + _cupsLangPuts(stdout, _("\tDefault port settings:")); + } + + for (i = 0; i < num_dests; i ++) + if (!_cups_strcasecmp(printer, dests[i].name) && dests[i].instance) + { + switch (pstate) + { + case IPP_PRINTER_IDLE : + _cupsLangPrintf(stdout, + _("printer %s/%s is idle. " + "enabled since %s"), + printer, dests[i].instance, + printer_state_time); + break; + case IPP_PRINTER_PROCESSING : + _cupsLangPrintf(stdout, + _("printer %s/%s now printing %s-%d. " + "enabled since %s"), + printer, dests[i].instance, printer, jobid, + printer_state_time); + break; + case IPP_PRINTER_STOPPED : + _cupsLangPrintf(stdout, + _("printer %s/%s disabled since %s -"), + printer, dests[i].instance, + printer_state_time); + break; + } + + if ((message && *message) || pstate == IPP_PRINTER_STOPPED) + { + if (!message || !*message) + _cupsLangPuts(stdout, _("\treason unknown")); + else + _cupsLangPrintf(stdout, "\t%s", message); + } + + if (long_status > 1) + { + _cupsLangPuts(stdout, _("\tForm mounted:")); + _cupsLangPuts(stdout, _("\tContent types: any")); + _cupsLangPuts(stdout, _("\tPrinter types: unknown")); + } + + if (long_status) + { + _cupsLangPrintf(stdout, _("\tDescription: %s"), + description ? description : ""); + + if (reasons) + { + char alerts[1024], /* Alerts string */ + *aptr; /* Pointer into alerts string */ + + for (i = 0, aptr = alerts; i < reasons->num_values; i ++) + { + if (i) + snprintf(aptr, sizeof(alerts) - (aptr - alerts), " %s", + reasons->values[i].string.text); + else + strlcpy(alerts, reasons->values[i].string.text, + sizeof(alerts)); + + aptr += strlen(aptr); + } + + _cupsLangPrintf(stdout, _("\tAlerts: %s"), alerts); + } + } + if (long_status > 1) + { + _cupsLangPrintf(stdout, _("\tLocation: %s"), + location ? location : ""); + + if (ptype & CUPS_PRINTER_REMOTE) + { + _cupsLangPuts(stdout, _("\tConnection: remote")); + + if (make_model && !strstr(make_model, "System V Printer") && + !strstr(make_model, "Raw Printer") && uri) + _cupsLangPrintf(stdout, _("\tInterface: %s.ppd"), uri); + } + else + { + _cupsLangPuts(stdout, _("\tConnection: direct")); + + if (make_model && strstr(make_model, "System V Printer")) + _cupsLangPrintf(stdout, + _("\tInterface: %s/interfaces/%s"), + cg->cups_serverroot, printer); + else if (make_model && !strstr(make_model, "Raw Printer")) + _cupsLangPrintf(stdout, + _("\tInterface: %s/ppd/%s.ppd"), + cg->cups_serverroot, printer); + } + _cupsLangPuts(stdout, _("\tOn fault: no alert")); + _cupsLangPuts(stdout, _("\tAfter fault: continue")); + /* TODO update to use printer-error-policy */ + if (allowed) + { + _cupsLangPuts(stdout, _("\tUsers allowed:")); + for (j = 0; j < allowed->num_values; j ++) + _cupsLangPrintf(stdout, "\t\t%s", + allowed->values[j].string.text); + } + else if (denied) + { + _cupsLangPuts(stdout, _("\tUsers denied:")); + for (j = 0; j < denied->num_values; j ++) + _cupsLangPrintf(stdout, "\t\t%s", + denied->values[j].string.text); + } + else + { + _cupsLangPuts(stdout, _("\tUsers allowed:")); + _cupsLangPuts(stdout, _("\t\t(all)")); + } + _cupsLangPuts(stdout, _("\tForms allowed:")); + _cupsLangPuts(stdout, _("\t\t(none)")); + _cupsLangPuts(stdout, _("\tBanner required")); + _cupsLangPuts(stdout, _("\tCharset sets:")); + _cupsLangPuts(stdout, _("\t\t(none)")); + _cupsLangPuts(stdout, _("\tDefault pitch:")); + _cupsLangPuts(stdout, _("\tDefault page size:")); + _cupsLangPuts(stdout, _("\tDefault port settings:")); + } + } + } + + if (attr == NULL) + break; + } + + ippDelete(response); + } + else + { + _cupsLangPrintf(stderr, "lpstat: %s", cupsLastErrorString()); + return (1); + } + + return (0); +} + + +/* + * 'show_scheduler()' - Show scheduler status. + */ + +static void +show_scheduler(void) +{ + http_t *http; /* Connection to server */ + + + if ((http = httpConnectEncrypt(cupsServer(), ippPort(), + cupsEncryption())) != NULL) + { + _cupsLangPuts(stdout, _("scheduler is running")); + httpClose(http); + } + else + _cupsLangPuts(stdout, _("scheduler is not running")); +} + + +/* + * End of "$Id$". + */ diff --git a/templates/Makefile b/templates/Makefile new file mode 100644 index 0000000000..0abd47d9d1 --- /dev/null +++ b/templates/Makefile @@ -0,0 +1,204 @@ +# +# "$Id$" +# +# Template makefile for CUPS. +# +# Copyright 2007-2011 by Apple Inc. +# Copyright 1993-2007 by Easy Software Products. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +include ../Makedefs + +# +# Template files... +# + +FILES = \ + add-class.tmpl \ + add-printer.tmpl \ + add-rss-subscription.tmpl \ + admin.tmpl \ + choose-device.tmpl \ + choose-make.tmpl \ + choose-model.tmpl \ + choose-serial.tmpl \ + choose-uri.tmpl \ + class.tmpl \ + class-added.tmpl \ + class-confirm.tmpl \ + class-deleted.tmpl \ + class-jobs-header.tmpl \ + class-modified.tmpl \ + classes.tmpl \ + classes-header.tmpl \ + command.tmpl \ + edit-config.tmpl \ + error.tmpl \ + error-op.tmpl \ + header.tmpl \ + help-header.tmpl \ + help-trailer.tmpl \ + help-printable.tmpl \ + job-cancel.tmpl \ + job-hold.tmpl \ + job-move.tmpl \ + job-moved.tmpl \ + job-release.tmpl \ + job-restart.tmpl \ + jobs.tmpl \ + jobs-header.tmpl \ + list-available-printers.tmpl \ + modify-class.tmpl \ + modify-printer.tmpl \ + norestart.tmpl \ + option-boolean.tmpl \ + option-conflict.tmpl \ + option-header.tmpl \ + option-pickmany.tmpl \ + option-pickone.tmpl \ + option-trailer.tmpl \ + pager.tmpl \ + printer.tmpl \ + printer-accept.tmpl \ + printer-added.tmpl \ + printer-configured.tmpl \ + printer-confirm.tmpl \ + printer-default.tmpl \ + printer-deleted.tmpl \ + printer-jobs-header.tmpl \ + printer-modified.tmpl \ + printer-purge.tmpl \ + printer-reject.tmpl \ + printer-start.tmpl \ + printer-stop.tmpl \ + printers.tmpl \ + printers-header.tmpl \ + restart.tmpl \ + samba-export.tmpl \ + samba-exported.tmpl \ + search.tmpl \ + set-printer-options-header.tmpl \ + set-printer-options-trailer.tmpl \ + subscription-added.tmpl \ + subscription-canceled.tmpl \ + test-page.tmpl \ + trailer.tmpl \ + users.tmpl + + +# +# Make everything... +# + +all: + + +# +# Make library targets... +# + +libs: + + +# +# Make unit tests... +# + +unittests: + + +# +# Clean all config and object files... +# + +clean: + + +# +# Dummy depend... +# + +depend: + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: $(INSTALL_LANGUAGES) + $(INSTALL_DIR) -m 755 $(DATADIR)/templates + for file in $(FILES); do \ + $(INSTALL_DATA) $$file $(DATADIR)/templates; \ + done + +install-languages: + for lang in $(LANGUAGES); do \ + if test -d $$lang; then \ + $(INSTALL_DIR) -m 755 $(DATADIR)/templates/$$lang; \ + for file in $(FILES); do \ + $(INSTALL_DATA) $$lang/$$file $(DATADIR)/templates/$$lang >/dev/null 2>&1 || true; \ + done \ + fi \ + done + +install-langbundle: + + +# +# Install programs... +# + +install-exec: + + +# +# Install headers... +# + +install-headers: + + +# +# Install libraries... +# + +install-libs: + + +# +# Uninstall files... +# + +uninstall: $(UNINSTALL_LANGUAGES) + for file in $(FILES); do \ + $(RM) $(DATADIR)/templates/$$file; \ + done + -$(RMDIR) $(DATADIR)/templates + +uninstall-languages: + for lang in $(LANGUAGES); do \ + for file in $(FILES); do \ + $(RM) $(DATADIR)/templates/$$lang/$$file; \ + done \ + $(RMDIR) $(DATADIR)/templates/$$lang; \ + done + +uninstall-langbundle: + + +# +# End of "$Id$". +# diff --git a/templates/add-class.tmpl b/templates/add-class.tmpl new file mode 100644 index 0000000000..eaf52d73dc --- /dev/null +++ b/templates/add-class.tmpl @@ -0,0 +1,40 @@ +
+ +

Add Class

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + +
Name:
+(May contain any printable characters except "/", "#", and space)
Description:
+(Human-readable description such as "HP LaserJet with Duplexer")
Location:
+(Human-readable location such as "Lab 1")
Members: + +
+ +
+
\ No newline at end of file diff --git a/templates/add-printer.tmpl b/templates/add-printer.tmpl new file mode 100644 index 0000000000..361958b674 --- /dev/null +++ b/templates/add-printer.tmpl @@ -0,0 +1,47 @@ +
+ +

Add Printer

+ +
+ + + + + + +{?current_make!?:} +{?current_make_and_model!?:} + + + + + + + + + + + + + + + + + + + + + + + + + + +
Name:
+(May contain any printable characters except "/", "#", and space)
Description:
+(Human-readable description such as "HP LaserJet with Duplexer")
Location:
+(Human-readable location such as "Lab 1")
Connection:{device_uri}
Sharing: +Share This Printer
+ +
+
\ No newline at end of file diff --git a/templates/add-rss-subscription.tmpl b/templates/add-rss-subscription.tmpl new file mode 100644 index 0000000000..17e3fa0d37 --- /dev/null +++ b/templates/add-rss-subscription.tmpl @@ -0,0 +1,44 @@ +
+ + + +

Add RSS Subscription

+ + + + + + + + + + + + + + + + + + + + + + + + + + +
Name:
+(May contain any printable characters except space, "/", "?", and "#")
Queue:
Events:Job Created
+Job Completed
+Job Stopped
+Job Options Changed
    Queue Stopped
+Queue Added
+Queue Modified
+Queue Deleted
    Server Started
+Server Stopped
+Server Restarted
+Server Security Auditing
Maximum Events in Feed:
+ +
diff --git a/templates/admin.tmpl b/templates/admin.tmpl new file mode 100644 index 0000000000..c76eb9c5eb --- /dev/null +++ b/templates/admin.tmpl @@ -0,0 +1,99 @@ + + +
+ +

Printers

+ +

+

+
+
+{have_samba?
:} +

+ +

Classes

+ +

+

+
+

+ +

Jobs

+ +

+

+

+ +
          + +

Server

+ +

+

+
+
+
+

+ +{SETTINGS_ERROR?

{SETTINGS_MESSAGE}

+
{SETTINGS_ERROR}
: + +
+ + +{ADVANCEDSETTINGS?

Server Settings\:

+ +

Advanced
+ + + Share printers connected to this system
+        Max clients\: +
+         Allow printing from the Internet
+         Advertise web interface
+ Allow remote administration
+{have_gssapi? Use Kerberos authentication (FAQ)
:} + Allow users to cancel any job (not just their own)
+ Preserve job history
+        Number of jobs\: +
+         Preserve job print files
+ Save debugging information for troubleshooting
+        Max log file size\: +

+ +:

Server Settings:

+ +

Advanced
+ + Share printers connected to this system
+         Allow printing from the Internet
+ Allow remote administration
+{have_gssapi? Use Kerberos authentication (FAQ)
:} + Allow users to cancel any job (not just their own)
+ Save debugging information for troubleshooting

+ +} +

+ +
} + +
+ +
+ +

RSS Subscriptions

+ +

+

+

+ +
+ +{notify_subscription_id? + +{[notify_subscription_id] +} + +
NameEventsQueue Name
{notify_recipient_name}
+
 
{notify_events} {notify_printer_name?{notify_printer_name}:All Queues}
:} diff --git a/templates/choose-device.tmpl b/templates/choose-device.tmpl new file mode 100644 index 0000000000..c5ab251b54 --- /dev/null +++ b/templates/choose-device.tmpl @@ -0,0 +1,53 @@ +
+ +

{op=modify-printer?Modify {printer_name}:Add Printer}

+ +{CUPS_GET_DEVICES_DONE?
+ + +{printer_name?:} + + +{op=add-printer?: + + +} + + + + + + + + + + + + + + + + + +
Current Connection\: +{current_device_uri}
Local Printers\: +{[device_uri]{device_class!network? +{device_info} {?device_make_and_model!Unknown?({device_make_and_model}):}
+:}} +
Discovered Network Printers\: +{[device_uri]{device_class=network?{device_uri~[a-z]+://? +{device_info} {?device_make_and_model!Unknown?({device_make_and_model}):}
+:}:}} +
Other Network Printers\: +{[device_uri]{device_class=network?{device_uri~[a-z]+://?: +{device_info} {?device_make_and_model!Unknown?({device_make_and_model}):}
+}:}} +
+ +
:

Looking for printers...

} + +
diff --git a/templates/choose-make.tmpl b/templates/choose-make.tmpl new file mode 100644 index 0000000000..65096ad9a6 --- /dev/null +++ b/templates/choose-make.tmpl @@ -0,0 +1,64 @@ +
+ +

{op=modify-printer?Modify {printer_name}:Add Printer}

+ +
+ + +{printer_name?:} + + + + + + +{op=modify-printer?: + + +} + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Name:{printer_name}
Description:{printer_info}
Location:{printer_location}
Connection:{device_uri}
Sharing: +{?printer_is_shared=?Do Not:{?printer_is_shared=0?Do Not:}} Share This Printer
Make: + +
 
Or Provide a PPD File:
+ +
+
\ No newline at end of file diff --git a/templates/choose-model.tmpl b/templates/choose-model.tmpl new file mode 100644 index 0000000000..992cd01fd6 --- /dev/null +++ b/templates/choose-model.tmpl @@ -0,0 +1,60 @@ +
+ +

{op=modify-printer?Modify {printer_name}:Add Printer}

+ +
+ + +{printer_name?:} + + + + + +{op=modify-printer?: + + +} + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Name:{printer_name}
Description:{printer_info}
Location:{printer_location}
Connection:{device_uri}
Sharing: +{?printer_is_shared=?Do Not:{?printer_is_shared=0?Do Not:}} Share This Printer
Make:{PPD_MAKE}
Model: + +
Or Provide a PPD File:
+ +
+
\ No newline at end of file diff --git a/templates/choose-serial.tmpl b/templates/choose-serial.tmpl new file mode 100644 index 0000000000..56a14a4cbf --- /dev/null +++ b/templates/choose-serial.tmpl @@ -0,0 +1,52 @@ +
+ +

{op=modify-printer?Modify {printer_name}:Add Printer}

+ +
+ + +{printer_name?:} + + + + + + + + + + + + + + + + + + + + + + + + + + +
Connection:{device_uri}
Baud Rate:
Parity:
Data Bits:
Flow Control:
+ +
+
\ No newline at end of file diff --git a/templates/choose-uri.tmpl b/templates/choose-uri.tmpl new file mode 100644 index 0000000000..45d84de194 --- /dev/null +++ b/templates/choose-uri.tmpl @@ -0,0 +1,44 @@ +
+ +

{op=modify-printer?Modify {printer_name}:Add Printer}

+ +
+ + +{printer_name?:} + + + + + + + + + + + + + + + +
Connection:
Examples: +
+    http://hostname:631/ipp/
+    http://hostname:631/ipp/port1
+
+    ipp://hostname/ipp/
+    ipp://hostname/ipp/port1
+
+    lpd://hostname/queue
+
+    socket://hostname
+    socket://hostname:9100
+
+ +

See "Network +Printers" for the correct URI to use with your printer.

+ +
+ +
+
\ No newline at end of file diff --git a/templates/class-added.tmpl b/templates/class-added.tmpl new file mode 100644 index 0000000000..c062a16d5a --- /dev/null +++ b/templates/class-added.tmpl @@ -0,0 +1,8 @@ +
+ +

Add Class

+ +

Class {printer_name} has been added +successfully. + +

diff --git a/templates/class-confirm.tmpl b/templates/class-confirm.tmpl new file mode 100644 index 0000000000..5f0277cb51 --- /dev/null +++ b/templates/class-confirm.tmpl @@ -0,0 +1,10 @@ +
+ +

Delete Class {printer_name}

+ +

Warning: Are you sure you want to delete class +{printer_name}?

+ +

+ +
diff --git a/templates/class-deleted.tmpl b/templates/class-deleted.tmpl new file mode 100644 index 0000000000..a4ad46a0ad --- /dev/null +++ b/templates/class-deleted.tmpl @@ -0,0 +1,7 @@ +
+ +

Delete Class {printer_name}

+ +

Class {printer_name} has been deleted successfully. + +

\ No newline at end of file diff --git a/templates/class-jobs-header.tmpl b/templates/class-jobs-header.tmpl new file mode 100644 index 0000000000..ba46f10260 --- /dev/null +++ b/templates/class-jobs-header.tmpl @@ -0,0 +1,3 @@ +
+

Jobs

+
diff --git a/templates/class-modified.tmpl b/templates/class-modified.tmpl new file mode 100644 index 0000000000..fe42c907bb --- /dev/null +++ b/templates/class-modified.tmpl @@ -0,0 +1,8 @@ +
+ +

Modify Class {printer_name}

+ +

Class {printer_name} has been +modified successfully. + +

\ No newline at end of file diff --git a/templates/class.tmpl b/templates/class.tmpl new file mode 100644 index 0000000000..477e0e8edb --- /dev/null +++ b/templates/class.tmpl @@ -0,0 +1,44 @@ +
+

{printer_name} +({printer_state=3?Idle:{printer_state=4?Processing:Paused}}, +{printer_is_accepting_jobs=0?Rejecting Jobs:Accepting Jobs}, +{server_is_sharing_printers=0?Not:{printer_is_shared=0?Not:}} Shared{default_name={printer_name}?, Server Default:})

+ +
+ + + +
+ +
+ + + + + +
+ + + + + + +
Description:{printer_info}
Location:{printer_location}
Members:{?member_uris=?None:{member_uris}}
Defaults:job-sheets={job_sheets_default} +media={media_default?{media_default}:unknown} +{sides_default?sides={sides_default}:}
+ +
diff --git a/templates/classes-header.tmpl b/templates/classes-header.tmpl new file mode 100644 index 0000000000..7ac98fbddc --- /dev/null +++ b/templates/classes-header.tmpl @@ -0,0 +1 @@ +

{total=0?No classes:Showing {#printer_name} of {total} class{total=1?:es}}.

diff --git a/templates/classes.tmpl b/templates/classes.tmpl new file mode 100644 index 0000000000..51e454fb1c --- /dev/null +++ b/templates/classes.tmpl @@ -0,0 +1,11 @@ +{#printer_name=0?: + + + + + +{[printer_name] + +} + +
{ORDER=dec? Queue Name : Queue Name }DescriptionLocationMembersStatus
{printer_name}{printer_info}{printer_location}{?member_uris=?None:{member_uris}}{printer_state=3?Idle:{printer_state=4?Processing:Paused}}{printer_state_message? - "{printer_state_message}":}
} diff --git a/templates/command.tmpl b/templates/command.tmpl new file mode 100644 index 0000000000..9a87439ca5 --- /dev/null +++ b/templates/command.tmpl @@ -0,0 +1,12 @@ +
+ +

{title} On {printer_name}

+ +

{job_state>5?:Busy Indicator }Printer command job +{job_state=3?pending:{job_state=4?held: +{job_state=5?processing:{job_state=6?stopped: +{job_state=7?canceled:{job_state=8?aborted:completed}}}}}}{job_state=9?:{job_printer_state_message?, +"{job_printer_state_message}":}}

+ +
diff --git a/templates/edit-config.tmpl b/templates/edit-config.tmpl new file mode 100644 index 0000000000..8947382b19 --- /dev/null +++ b/templates/edit-config.tmpl @@ -0,0 +1,24 @@ + + +
+ +

Edit Configuration File

+ +
+ + + + + +

+

+ +
+ +
diff --git a/templates/error-op.tmpl b/templates/error-op.tmpl new file mode 100644 index 0000000000..feaed0f904 --- /dev/null +++ b/templates/error-op.tmpl @@ -0,0 +1,9 @@ +
+ +

{?title} {?printer_name} Error

+ +

Error:

+ +
Unknown operation "{op}"!
+ +
diff --git a/templates/error.tmpl b/templates/error.tmpl new file mode 100644 index 0000000000..ff30a4c6c3 --- /dev/null +++ b/templates/error.tmpl @@ -0,0 +1,9 @@ +
+ +

{?title} {?printer_name} Error

+ +

{?message?{message}:Error:}

+ +
{error}
+ +
diff --git a/templates/header.tmpl.in b/templates/header.tmpl.in new file mode 100644 index 0000000000..a383725c51 --- /dev/null +++ b/templates/header.tmpl.in @@ -0,0 +1,29 @@ + + + + + {title} - CUPS @CUPS_VERSION@@CUPS_REVISION@ + + + {refresh_page?:} + + + + + + + diff --git a/templates/option-conflict.tmpl b/templates/option-conflict.tmpl new file mode 100644 index 0000000000..3772295a83 --- /dev/null +++ b/templates/option-conflict.tmpl @@ -0,0 +1,7 @@ +

Error: The following options are conflicting:

+ + + +

Please change one or more of the options to resolve the conflicts.

diff --git a/templates/option-header.tmpl b/templates/option-header.tmpl new file mode 100644 index 0000000000..464726a50b --- /dev/null +++ b/templates/option-header.tmpl @@ -0,0 +1,5 @@ +
+ +

{group}

+ +
+ + + + + + + + + + + + + +
  Home    Administration    Classes    Online Help    Jobs    Printers  
 
diff --git a/templates/help-header.tmpl b/templates/help-header.tmpl new file mode 100644 index 0000000000..17afce7388 --- /dev/null +++ b/templates/help-header.tmpl @@ -0,0 +1,51 @@ +
+
+{TOPIC?:} + +

Search in +{HELPTITLE?{HELPTITLE}:{TOPIC?{TOPIC}:All Documents}}: + +

+ +
+ + + + +{QUERY?

Search Results in {HELPFILE?{HELPTITLE}:{TOPIC?{TOPIC}:All Documents}}\:

+{QTEXT?:} +:

No matches found.

} +
:} +{HELPTITLE?

{HELPTITLE}

+
: + +

Online Help

+ +

This is the CUPS online help interface. Enter search words +above or click on any of the documentation links to display +online help information.

+ +

If you are new to CUPS, read the "Overview of CUPS" page. Veteran users +should read the "What's New in CUPS +1.4" page.

+ +

The CUPS home page also +provides many resources including user discussion forums, answers +to frequently-asked questions, and a form for submitting bug +reports and feature requests.

} diff --git a/templates/help-printable.tmpl b/templates/help-printable.tmpl new file mode 100644 index 0000000000..2463c1630c --- /dev/null +++ b/templates/help-printable.tmpl @@ -0,0 +1,9 @@ + + + + + {HELPTITLE} + + + + diff --git a/templates/help-trailer.tmpl b/templates/help-trailer.tmpl new file mode 100644 index 0000000000..4c1ebed851 --- /dev/null +++ b/templates/help-trailer.tmpl @@ -0,0 +1 @@ +
diff --git a/templates/job-cancel.tmpl b/templates/job-cancel.tmpl new file mode 100644 index 0000000000..3e8fd4dadf --- /dev/null +++ b/templates/job-cancel.tmpl @@ -0,0 +1,7 @@ +
+ +

Cancel Job {job_id}

+ +

Job {job_id} has been canceled. + +

diff --git a/templates/job-hold.tmpl b/templates/job-hold.tmpl new file mode 100644 index 0000000000..58ce97fc91 --- /dev/null +++ b/templates/job-hold.tmpl @@ -0,0 +1,7 @@ +
+ +

Hold Job {job_id}

+ +

Job {job_id} has been held from printing. + +

diff --git a/templates/job-move.tmpl b/templates/job-move.tmpl new file mode 100644 index 0000000000..6ef0ff12e1 --- /dev/null +++ b/templates/job-move.tmpl @@ -0,0 +1,27 @@ +
+ +
+ + +{job_id?:} + +

{job_id?Move Job {job_id}:Move All Jobs}

+ + + + + + + + + + +
New Destination: + +
+ +
+ +
diff --git a/templates/job-moved.tmpl b/templates/job-moved.tmpl new file mode 100644 index 0000000000..c4db096968 --- /dev/null +++ b/templates/job-moved.tmpl @@ -0,0 +1,8 @@ +
+ +

{job_id?Move Job {job_id}:Move All Jobs}

+ +

{job_id?Job {job_id}:All jobs} moved to +{job_printer_name}.

+ +
diff --git a/templates/job-release.tmpl b/templates/job-release.tmpl new file mode 100644 index 0000000000..8eb9a96675 --- /dev/null +++ b/templates/job-release.tmpl @@ -0,0 +1,7 @@ +
+ +

Release Job {job_id}

+ +

Job {job_id} has been released for printing. + +

diff --git a/templates/job-restart.tmpl b/templates/job-restart.tmpl new file mode 100644 index 0000000000..b91ced9263 --- /dev/null +++ b/templates/job-restart.tmpl @@ -0,0 +1,7 @@ +
+ +

Reprint Job {job_id}

+ +

Job {job_id} has been restarted. + +

diff --git a/templates/jobs-header.tmpl b/templates/jobs-header.tmpl new file mode 100644 index 0000000000..e7547ba7bb --- /dev/null +++ b/templates/jobs-header.tmpl @@ -0,0 +1,5 @@ +
{?which_jobs=?:
} +{?which_jobs=completed?:
} +{?which_jobs=all?:
}
+ +

{total=0?No jobs:Showing {#job_id} of {total} {?which_jobs=?active:{which_jobs=all?:completed}} job{total=1?:s}}.

diff --git a/templates/jobs.tmpl b/templates/jobs.tmpl new file mode 100644 index 0000000000..625059f621 --- /dev/null +++ b/templates/jobs.tmpl @@ -0,0 +1,36 @@ +{#job_id=0?: + + + + + +{[job_id] + + + + + + + + + +} + +
{ORDER=dec? ID : ID }NameUserSizePagesStateControl
{job_printer_name}-{job_id} {?job_name=?Unknown:{job_name}} {?job_originating_user_name=?Withheld:{job_originating_user_name}} {job_k_octets}k {job_media_sheets_completed=0?Unknown:{?job_media_sheets_completed}} {job_state=3?pending since
{time_at_creation}:{job_state=4?held since
{time_at_creation}: +{job_state=5?processing since
{time_at_processing}:{job_state=6?stopped: +{job_state=7?canceled at
{time_at_completed}:{job_state=8?aborted:completed at
{time_at_completed}}}}}}} {job_printer_state_message?
+"{job_printer_state_message}":}
+{job_preserved>0?{job_state>5? +
:}:} +{job_state=4? +
+
:} +{job_state=3? +
+
:} +{job_state<7? +
+
+
:} + 
+} diff --git a/templates/list-available-printers.tmpl b/templates/list-available-printers.tmpl new file mode 100644 index 0000000000..a594439ee3 --- /dev/null +++ b/templates/list-available-printers.tmpl @@ -0,0 +1,11 @@ +
+ +

Available Printers

+ +{#device_uri=0?

No printers found.

+:
    {[device_uri] +
  • +{device_make_and_model} ({device_info})
  • +}
} + +
diff --git a/templates/modify-class.tmpl b/templates/modify-class.tmpl new file mode 100644 index 0000000000..31f7b9ce3c --- /dev/null +++ b/templates/modify-class.tmpl @@ -0,0 +1,34 @@ +
+ +

Modify Class {printer_name}

+ +
+ + + + + + + + + + + + + + + + + + + + + +
Description:
Location:
Members: + +
+ +
+
\ No newline at end of file diff --git a/templates/modify-printer.tmpl b/templates/modify-printer.tmpl new file mode 100644 index 0000000000..bdf3df6042 --- /dev/null +++ b/templates/modify-printer.tmpl @@ -0,0 +1,42 @@ +
+ +

Modify {printer_name}

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Description:
+(Human-readable description such as "HP LaserJet with Duplexer")
Location:
+(Human-readable location such as "Lab 1")
Connection:{device_uri}
Sharing: +Share This Printer
+ +
+
diff --git a/templates/norestart.tmpl b/templates/norestart.tmpl new file mode 100644 index 0000000000..6b3120337c --- /dev/null +++ b/templates/norestart.tmpl @@ -0,0 +1,8 @@ +
+ +

Change Settings

+ +

The server was not restarted because no changes were made to +the configuration...

+ +
diff --git a/templates/option-boolean.tmpl b/templates/option-boolean.tmpl new file mode 100644 index 0000000000..e832ee1022 --- /dev/null +++ b/templates/option-boolean.tmpl @@ -0,0 +1,6 @@ +
{keytext}: +{[choices]{text}} +
diff --git a/templates/option-pickmany.tmpl b/templates/option-pickmany.tmpl new file mode 100644 index 0000000000..0da75e5d2c --- /dev/null +++ b/templates/option-pickmany.tmpl @@ -0,0 +1,6 @@ + + + + diff --git a/templates/option-pickone.tmpl b/templates/option-pickone.tmpl new file mode 100644 index 0000000000..890ef4e7d3 --- /dev/null +++ b/templates/option-pickone.tmpl @@ -0,0 +1,18 @@ + + +:} + diff --git a/templates/option-trailer.tmpl b/templates/option-trailer.tmpl new file mode 100644 index 0000000000..17c928109e --- /dev/null +++ b/templates/option-trailer.tmpl @@ -0,0 +1,5 @@ +
{keytext}:
{keytext}: +{iscustom=1?{[params] + + +}
{paramtext}:{params=Units?:}
+
+ +

+ + \ No newline at end of file diff --git a/templates/pager.tmpl b/templates/pager.tmpl new file mode 100644 index 0000000000..ebd688d51b --- /dev/null +++ b/templates/pager.tmpl @@ -0,0 +1,6 @@ + + + + + +
{PREV?
: }
{NEXT?
: }
diff --git a/templates/printer-accept.tmpl b/templates/printer-accept.tmpl new file mode 100644 index 0000000000..3e987ce64f --- /dev/null +++ b/templates/printer-accept.tmpl @@ -0,0 +1,9 @@ +
+ +

Accept Jobs On {is_class?Class:Printer} {printer_name}

+ +

{is_class?Class:Printer} {printer_name} +is now accepting jobs.

+ +
diff --git a/templates/printer-added.tmpl b/templates/printer-added.tmpl new file mode 100644 index 0000000000..9a6e798e9a --- /dev/null +++ b/templates/printer-added.tmpl @@ -0,0 +1,8 @@ +
+ +

Add Printer

+ +

Printer {printer_name} has been added +successfully. + +

diff --git a/templates/printer-configured.tmpl b/templates/printer-configured.tmpl new file mode 100644 index 0000000000..6f8104e231 --- /dev/null +++ b/templates/printer-configured.tmpl @@ -0,0 +1,8 @@ +
+ +

Set Default Options for {printer_name}

+ +

{OP=set-class-options?Class :Printer }{printer_name} +default options have been set successfully. + +

diff --git a/templates/printer-confirm.tmpl b/templates/printer-confirm.tmpl new file mode 100644 index 0000000000..fce9e3494b --- /dev/null +++ b/templates/printer-confirm.tmpl @@ -0,0 +1,10 @@ +
+ +

Delete Printer {printer_name}

+ +

Warning: Are you sure you want to delete printer +{printer_name}?

+ +

+ +
diff --git a/templates/printer-default.tmpl b/templates/printer-default.tmpl new file mode 100644 index 0000000000..72a7b5ea8c --- /dev/null +++ b/templates/printer-default.tmpl @@ -0,0 +1,13 @@ +
+ +

Set {is_class?Class:Printer} {printer_name} As Default

+ +

{is_class?Class:Printer} {printer_name} +has been made the default printer on the server.

+ +
Note: Any user default that has been set via +the lpoptions command will override this default +setting.
+ +
diff --git a/templates/printer-deleted.tmpl b/templates/printer-deleted.tmpl new file mode 100644 index 0000000000..d8a31ae615 --- /dev/null +++ b/templates/printer-deleted.tmpl @@ -0,0 +1,7 @@ +
+ +

Delete Printer {printer_name}

+ +

Printer {printer_name} has been deleted successfully. + +

diff --git a/templates/printer-jobs-header.tmpl b/templates/printer-jobs-header.tmpl new file mode 100644 index 0000000000..ba46f10260 --- /dev/null +++ b/templates/printer-jobs-header.tmpl @@ -0,0 +1,3 @@ +
+

Jobs

+
diff --git a/templates/printer-modified.tmpl b/templates/printer-modified.tmpl new file mode 100644 index 0000000000..b61e8e34a2 --- /dev/null +++ b/templates/printer-modified.tmpl @@ -0,0 +1,8 @@ +
+ +

Modify Printer {printer_name}

+ +

Printer {printer_name} has been +modified successfully. + +

\ No newline at end of file diff --git a/templates/printer-purge.tmpl b/templates/printer-purge.tmpl new file mode 100644 index 0000000000..8cf7c14617 --- /dev/null +++ b/templates/printer-purge.tmpl @@ -0,0 +1,9 @@ +
+ +

Purge Jobs On {is_class?Class:Printer} {printer_name}

+ +

{is_class?Class:Printer} {printer_name} +has been purged of all jobs.

+ +
\ No newline at end of file diff --git a/templates/printer-reject.tmpl b/templates/printer-reject.tmpl new file mode 100644 index 0000000000..5f1c79195b --- /dev/null +++ b/templates/printer-reject.tmpl @@ -0,0 +1,9 @@ +
+ +

Reject Jobs On {is_class?Class:Printer} {printer_name}

+ +

{is_class?Class:Printer} {printer_name} +is no longer accepting jobs.

+ +
diff --git a/templates/printer-start.tmpl b/templates/printer-start.tmpl new file mode 100644 index 0000000000..3eaed13d17 --- /dev/null +++ b/templates/printer-start.tmpl @@ -0,0 +1,9 @@ +
+ +

Resume {is_class?Class:Printer} {printer_name}

+ +

{is_class?Class:Printer} {printer_name} +has been resumed.

+ +
\ No newline at end of file diff --git a/templates/printer-stop.tmpl b/templates/printer-stop.tmpl new file mode 100644 index 0000000000..b74e87135d --- /dev/null +++ b/templates/printer-stop.tmpl @@ -0,0 +1,9 @@ +
+ +

Pause {is_class?Class:Printer} {printer_name}

+ +

{is_class?Class:Printer} {printer_name} +has been paused.

+ +
\ No newline at end of file diff --git a/templates/printer.tmpl b/templates/printer.tmpl new file mode 100644 index 0000000000..78fc08beae --- /dev/null +++ b/templates/printer.tmpl @@ -0,0 +1,47 @@ +
+ +

{printer_name} +({printer_state=3?Idle:{printer_state=4?Processing:Paused}}, +{printer_is_accepting_jobs=0?Rejecting Jobs:Accepting Jobs}, +{server_is_sharing_printers=0?Not:{printer_is_shared=0?Not:}} Shared{default_name={printer_name}?, Server Default:})

+ +
+ + + +
+ +
+ + + + +
+ + + + + + +
Description:{printer_info}
Location:{printer_location}
Driver:{printer_make_and_model} ({color_supported=1?color:grayscale}{sides_supported?, 2-sided printing:})
+
Connection:{device_uri}
Defaults:job-sheets={job_sheets_default} +media={media_default?{media_default}:unknown} +{sides_default?sides={sides_default}:}
+ +
\ No newline at end of file diff --git a/templates/printers-header.tmpl b/templates/printers-header.tmpl new file mode 100644 index 0000000000..5196be1d87 --- /dev/null +++ b/templates/printers-header.tmpl @@ -0,0 +1 @@ +

{total=0?No printers:Showing {#printer_name} of {total} printer{total=1?:s}}.

diff --git a/templates/printers.tmpl b/templates/printers.tmpl new file mode 100644 index 0000000000..e0a3228d04 --- /dev/null +++ b/templates/printers.tmpl @@ -0,0 +1,11 @@ +{#printer_name=0?: + + + + + +{[printer_name] + +} + +
{ORDER=dec? Queue Name : Queue Name }DescriptionLocationMake and ModelStatus
{printer_name}{printer_info}{printer_location}{printer_make_and_model}{printer_state=3?Idle:{printer_state=4?Processing:Paused}}{printer_state_message? - "{printer_state_message}":}
} diff --git a/templates/restart.tmpl b/templates/restart.tmpl new file mode 100644 index 0000000000..bdfca876fb --- /dev/null +++ b/templates/restart.tmpl @@ -0,0 +1,8 @@ +
+ +

Change Settings

+ +

Please stand by while the server restarts...

+ +
diff --git a/templates/samba-export.tmpl b/templates/samba-export.tmpl new file mode 100644 index 0000000000..6515a4d3ae --- /dev/null +++ b/templates/samba-export.tmpl @@ -0,0 +1,55 @@ + + +
+ + + +

Export Printers to Samba

+ +{error?

Unable to export printers to Samba\:

+
{error}
+

Consult the error_log file for more information.

: +

This page allows you to export printers to Samba so that +Windows clients can access them through the Network +Neighborhood or Network Places icons on their +desktop. You must previously install the Windows PostScript +printer drivers as described in the cupsaddsmb(8) man page.

} + + + + + + + + + + + + + + + + + + +
Printers: +
+ Export All Printers +
Samba Username: (required)
Samba Password: (required)
+ +
diff --git a/templates/samba-exported.tmpl b/templates/samba-exported.tmpl new file mode 100644 index 0000000000..6fa8eb4a69 --- /dev/null +++ b/templates/samba-exported.tmpl @@ -0,0 +1 @@ +

Printers exported to samba successfully.

diff --git a/templates/search.tmpl b/templates/search.tmpl new file mode 100644 index 0000000000..755269ea37 --- /dev/null +++ b/templates/search.tmpl @@ -0,0 +1,10 @@ +
+{WHICH_JOBS?:} +{ORDER?:} + +

Search in +{SEARCH_DEST?{SEARCH_DEST}:{SECTION=classes?Classes:{SECTION=jobs?Jobs:Printers}}}: +

+ +
diff --git a/templates/set-printer-options-header.tmpl b/templates/set-printer-options-header.tmpl new file mode 100644 index 0000000000..3e7db4669c --- /dev/null +++ b/templates/set-printer-options-header.tmpl @@ -0,0 +1,26 @@ +
+ +

Set Default Options for {printer_name}

+ +
+ + + +{HAVE_AUTOCONFIGURE?:} + + + +

{[group_id] +{group}     }

+ +
diff --git a/templates/set-printer-options-trailer.tmpl b/templates/set-printer-options-trailer.tmpl new file mode 100644 index 0000000000..11adc70125 --- /dev/null +++ b/templates/set-printer-options-trailer.tmpl @@ -0,0 +1,16 @@ +
+ + +
+ +
diff --git a/templates/subscription-added.tmpl b/templates/subscription-added.tmpl new file mode 100644 index 0000000000..c288b3649b --- /dev/null +++ b/templates/subscription-added.tmpl @@ -0,0 +1,5 @@ +
+ +

Subscription {subscription_name} has been added successfully.

+ +
diff --git a/templates/subscription-canceled.tmpl b/templates/subscription-canceled.tmpl new file mode 100644 index 0000000000..86f2c10691 --- /dev/null +++ b/templates/subscription-canceled.tmpl @@ -0,0 +1,5 @@ +
+ +

Subscription #{notify_subscription_id} has been canceled.

+ +
diff --git a/templates/test-page.tmpl b/templates/test-page.tmpl new file mode 100644 index 0000000000..59e314a8d4 --- /dev/null +++ b/templates/test-page.tmpl @@ -0,0 +1,8 @@ +
+ +

Print Test Page On {printer_name}

+ +

Test page sent; job ID is +{printer_name}-{job_id}.

+ +
diff --git a/templates/trailer.tmpl b/templates/trailer.tmpl new file mode 100644 index 0000000000..100d9f7adc --- /dev/null +++ b/templates/trailer.tmpl @@ -0,0 +1,8 @@ + +  +CUPS and the CUPS logo are trademarks of +Apple Inc. CUPS is copyright 2007-2011 Apple +Inc. All rights reserved. + + + diff --git a/templates/users.tmpl b/templates/users.tmpl new file mode 100644 index 0000000000..0889f3f31e --- /dev/null +++ b/templates/users.tmpl @@ -0,0 +1,30 @@ +
+ +
+ + + +{IS_CLASS?:} + +

Allowed Users For {printer_name}

+ + + + + + + + + + +
Users: + +
+Allow these users to print +Prevent these users from printing +
+ +
+ +
+
diff --git a/test/4.1-requests.test b/test/4.1-requests.test new file mode 100644 index 0000000000..e2eb79ca65 --- /dev/null +++ b/test/4.1-requests.test @@ -0,0 +1,159 @@ +# +# "$Id: 4.1-requests.test 9084 2010-04-07 06:54:31Z mike $" +# +# Verify that the server requires the following attributes: +# +# attributes-charset +# attributes-natural-language +# printer-uri/job-uri +# +{ + # The name of the test... + NAME "No Attributes" + + # The operation to use + OPERATION get-jobs + + # What statuses are OK? + STATUS client-error-bad-request + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Charset Attribute" + + # The operation to use + OPERATION get-jobs + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + + # What statuses are OK? + STATUS client-error-bad-request + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Language Attribute" + + # The operation to use + OPERATION get-jobs + + # The attributes to send + GROUP operation + ATTR language attributes-natural-language en + + # What statuses are OK? + STATUS client-error-bad-request + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Language + Charset Attributes" + + # The operation to use + OPERATION get-jobs + + # The attributes to send + GROUP operation + ATTR language attributes-natural-language en + ATTR charset attributes-charset utf-8 + + # What statuses are OK? + STATUS client-error-bad-request + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Charset + Language Attributes" + + # The operation to use + OPERATION get-jobs + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + + # What statuses are OK? + STATUS client-error-bad-request + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Charset + Language + Printer URI Attributes" + + # The operation to use + OPERATION get-jobs + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $uri + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Charset + Language + Job URI Attributes" + + # The operation to use + OPERATION get-jobs + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri job-uri $scheme://$hostname:$port/jobs + + # What statuses are OK? + STATUS client-error-bad-request + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Bad IPP Version" + + # The operation to use + OPERATION get-jobs + + # The version number to use + VERSION 0.0 + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri ipp://localhost/printers + + # What statuses are OK? + STATUS server-error-version-not-supported +} +# +# End of "$Id: 4.1-requests.test 9084 2010-04-07 06:54:31Z mike $" +# diff --git a/test/4.2-cups-printer-ops.test b/test/4.2-cups-printer-ops.test new file mode 100644 index 0000000000..5fa828eb66 --- /dev/null +++ b/test/4.2-cups-printer-ops.test @@ -0,0 +1,327 @@ +# +# "$Id: 4.2-cups-printer-ops.test 9084 2010-04-07 06:54:31Z mike $" +# +# Verify that the CUPS printer operations work. +# +{ + # The name of the test... + NAME "Add Printer Test1" + + # The operation to use + OPERATION cups-add-printer + RESOURCE /admin/ + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test1 + + FILE testhp.ppd + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Verify Printer Test1 Added" + + # The operation to use + OPERATION get-printer-attributes + RESOURCE / + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test1 + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Delete Printer Test1" + + # The operation to use + OPERATION cups-delete-printer + RESOURCE /admin/ + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test1 + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Verify Printer Test1 Deleted" + + # The operation to use + OPERATION get-printer-attributes + RESOURCE / + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test1 + + # What statuses are OK? + STATUS client-error-not-found + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Subscribe to Printer Events" + + # The operation to use + OPERATION Create-Printer-Subscription + RESOURCE / + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/ + ATTR name requesting-user-name $user + + GROUP subscription + ATTR uri notify-recipient-uri testnotify:// + ATTR keyword notify-events printer-added,printer-modified,printer-deleted + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language + EXPECT notify-subscription-id +} +{ + # The name of the test... + NAME "Add Printer Test2" + + # The operation to use + OPERATION cups-add-printer + RESOURCE /admin/ + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test2 + + FILE testhp.ppd + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Verify Printer Test2 Added" + + # The operation to use + OPERATION get-printer-attributes + RESOURCE / + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test2 + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Modify Printer Test2" + + # The operation to use + OPERATION cups-add-printer + RESOURCE /admin/ + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test2 + + GROUP printer + ATTR uri device-uri file:/tmp/Test2 + ATTR enum printer-state 3 + ATTR boolean printer-is-accepting-jobs true + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Re-Add Printer Test1" + + # The operation to use + OPERATION cups-add-printer + RESOURCE /admin/ + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test1 + + GROUP printer + ATTR uri device-uri file:/tmp/Test1 + ATTR enum printer-state 3 + ATTR boolean printer-is-accepting-jobs true + + FILE testps.ppd + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Verify Printer Test1 Re-Added" + + # The operation to use + OPERATION get-printer-attributes + RESOURCE / + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test1 + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Get Default Printer with no default set" + + # The operation to use + OPERATION cups-get-default + RESOURCE / + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + + # What statuses are OK? + STATUS client-error-not-found + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Set Default Printer to Test1" + + # The operation to use + OPERATION cups-set-default + RESOURCE /admin/ + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test1 + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Get Default Printer" + + # The operation to use + OPERATION cups-get-default + RESOURCE / + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language + EXPECT printer-name + EXPECT printer-uri-supported +} +{ + # The name of the test... + NAME "Get IPP/2.x Attributes for Printer Test1" + + # The operation to use + OPERATION get-printer-attributes + RESOURCE / + + # The IPP version to use + VERSION 2.0 + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test1 + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT media-col-default +} + +# +# End of "$Id: 4.2-cups-printer-ops.test 9084 2010-04-07 06:54:31Z mike $" +# diff --git a/test/4.3-job-ops.test b/test/4.3-job-ops.test new file mode 100644 index 0000000000..e82de18dcc --- /dev/null +++ b/test/4.3-job-ops.test @@ -0,0 +1,330 @@ +# +# "$Id: 4.3-job-ops.test 9084 2010-04-07 06:54:31Z mike $" +# +# Verify that the IPP job operations work. +# +{ + # The name of the test... + NAME "Print PostScript Job with bad job-sheets value to Test1" + + # The operation to use + OPERATION print-job + RESOURCE /printers/Test1 + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test1 + ATTR name requesting-user-name $user + ATTR name job-sheets "none\,none" + + FILE testfile.ps + + # What statuses are OK? + STATUS client-error-bad-request +} +{ + # The name of the test... + NAME "Print PostScript Job to Test1" + + # The operation to use + OPERATION print-job + RESOURCE /printers/Test1 + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test1 + ATTR name requesting-user-name $user + + FILE testfile.ps + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language + EXPECT job-id +} +{ + # The name of the test... + NAME "Get Job Attributes" + + # The operation to use + OPERATION get-job-attributes + RESOURCE /jobs + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test1 + ATTR integer job-id $job-id + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language + EXPECT job-id + EXPECT job-uri + EXPECT job-state +} +{ + # The name of the test... + NAME "Print JPEG Job to Test2" + + # The operation to use + OPERATION print-job + RESOURCE /printers/Test2 + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test2 + ATTR name requesting-user-name $user + + GROUP subscription + ATTR uri notify-recipient-uri testnotify:/// + + FILE testfile.jpg + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language + EXPECT job-id + EXPECT notify-subscription-id +} +{ + # The name of the test... + NAME "Get Job Attributes" + + # The operation to use + OPERATION get-job-attributes + RESOURCE /jobs + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test2 + ATTR integer job-id $job-id + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language + EXPECT job-id + EXPECT job-uri + EXPECT job-state +} +{ + # The name of the test... + NAME "Print Text Job to Test1" + + # The operation to use + OPERATION print-job + RESOURCE /printers/Test1 + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test1 + ATTR name requesting-user-name $user + + FILE testfile.txt + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language + EXPECT job-id +} +{ + # The name of the test... + NAME "Print PDF Job to Test1" + + # The operation to use + OPERATION print-job + RESOURCE /printers/Test1 + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test1 + ATTR name requesting-user-name $user + + GROUP job + ATTR keyword job-hold-until weekend + + FILE testfile.pdf + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language + EXPECT job-id +} +{ + # The name of the test... + NAME "Hold Job on Test1" + + # The operation to use + OPERATION hold-job + RESOURCE /printers/Test1 + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test1 + ATTR integer job-id $job-id + ATTR name requesting-user-name $user + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Release Job on Test1" + + # The operation to use + OPERATION release-job + RESOURCE /printers/Test1 + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test1 + ATTR integer job-id $job-id + ATTR name requesting-user-name $user + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Print Held Image Job to Test1" + + # The operation to use + OPERATION print-job + RESOURCE /printers/Test1 + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test1 + ATTR name requesting-user-name $user + GROUP job + ATTR keyword job-hold-until indefinite + + FILE testfile.jpg + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language + EXPECT job-id +} +{ + # The name of the test... + NAME "Cancel Job" + + # The operation to use + OPERATION cancel-job + RESOURCE /jobs + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri job-uri $method://$hostname:$port/jobs/$job-id + ATTR name requesting-user-name $user + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Get Job List on Test1" + + # The operation to use + OPERATION get-jobs + RESOURCE /printers/Test1 + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $method://$hostname:$port/printers/Test1 + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language + EXPECT !job-printer-uri +} +{ + # The name of the test... + NAME "Get All Jobs" + + # The operation to use + OPERATION get-jobs + RESOURCE /jobs + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $scheme://$hostname:$port/ + ATTR keyword requested-attributes all + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language + EXPECT job-uri + EXPECT job-id + EXPECT job-state + EXPECT job-printer-uri +} + +# +# End of "$Id: 4.3-job-ops.test 9084 2010-04-07 06:54:31Z mike $" +# diff --git a/test/4.4-subscription-ops.test b/test/4.4-subscription-ops.test new file mode 100644 index 0000000000..720add80dc --- /dev/null +++ b/test/4.4-subscription-ops.test @@ -0,0 +1,153 @@ +# +# "$Id: 4.4-subscription-ops.test 9352 2010-11-06 04:55:26Z mike $" +# +# Verify that the CUPS subscription operations work. +# +{ + # The name of the test... + NAME "Add Printer Subscription w/Lease" + + # The operation to use + OPERATION Create-Printer-Subscription + RESOURCE / + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $scheme://$hostname:$port/printers/Test1 + ATTR name requesting-user-name $user + + GROUP subscription + ATTR uri notify-recipient-uri testnotify:// + ATTR keyword notify-events printer-state-changed + ATTR integer notify-lease-duration 5 + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language + EXPECT notify-subscription-id + DISPLAY notify-subscription-id +} +{ + # The name of the test... + NAME "Verify Subscription Expiration" + + # Delay test for 7 seconds to allow lease to expire... + DELAY 7 + + # The operation to use + OPERATION Get-Subscription-Attributes + RESOURCE / + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $scheme://$hostname:$port/printers/Test1 + ATTR integer notify-subscription-id $notify-subscription-id + ATTR name requesting-user-name $user + + # What statuses are OK? + STATUS client-error-not-found + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} +{ + # The name of the test... + NAME "Add 2 Printer Subscriptions w/Lease" + + # The operation to use + OPERATION Create-Printer-Subscription + RESOURCE / + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $scheme://$hostname:$port/printers/Test1 + ATTR name requesting-user-name $user + + GROUP subscription + ATTR uri notify-recipient-uri testnotify:// + ATTR keyword notify-events printer-state-changed + ATTR integer notify-lease-duration 5 + + GROUP subscription + ATTR uri notify-recipient-uri testnotify:// + ATTR keyword notify-events printer-config-changed + ATTR integer notify-lease-duration 5 + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language + EXPECT notify-subscription-id + DISPLAY notify-subscription-id +} +{ + # The name of the test... + NAME "List Printer Subscriptions" + + # The operation to use + OPERATION Get-Subscriptions + RESOURCE / + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $scheme://$hostname:$port/printers/Test1 + ATTR name requesting-user-name $user + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language + EXPECT notify-subscription-id + DISPLAY notify-subscription-id + EXPECT notify-printer-uri + DISPLAY notify-printer-uri + EXPECT notify-events + DISPLAY notify-events +} +{ + # The name of the test... + NAME "Check MaxSubscriptions limits" + + # The operation to use + OPERATION Create-Printer-Subscription + RESOURCE / + + # The attributes to send + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $scheme://$hostname:$port/printers/Test1 + ATTR name requesting-user-name $user + + GROUP subscription + ATTR uri notify-recipient-uri testnotify:// + ATTR keyword notify-events printer-state-changed + ATTR integer notify-lease-duration 5 + + # What statuses are OK? + STATUS client-error-too-many-subscriptions + + # What attributes do we expect? + EXPECT attributes-charset + EXPECT attributes-natural-language +} + +# +# End of "$Id: 4.4-subscription-ops.test 9352 2010-11-06 04:55:26Z mike $" +# diff --git a/test/5.1-lpadmin.sh b/test/5.1-lpadmin.sh new file mode 100755 index 0000000000..cf116c3230 --- /dev/null +++ b/test/5.1-lpadmin.sh @@ -0,0 +1,55 @@ +#!/bin/sh +# +# "$Id$" +# +# Test the lpadmin command. +# +# Copyright 2007-2009 by Apple Inc. +# Copyright 1997-2005 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +echo "Add Printer Test" +echo "" +echo " lpadmin -p Test3 -v file:/dev/null -E -m drv:///sample.drv/deskjet.ppd" +$VALGRIND ../systemv/lpadmin -p Test3 -v file:/dev/null -E -m drv:///sample.drv/deskjet.ppd 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +echo "Modify Printer Test" +echo "" +echo " lpadmin -p Test3 -v file:/tmp/Test3 -o PageSize=A4" +$VALGRIND ../systemv/lpadmin -p Test3 -v file:/tmp/Test3 -o PageSize=A4 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +echo "Delete Printer Test" +echo "" +echo " lpadmin -x Test3" +$VALGRIND ../systemv/lpadmin -x Test3 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +# +# End of "$Id$". +# diff --git a/test/5.2-lpc.sh b/test/5.2-lpc.sh new file mode 100755 index 0000000000..f1957a8b2a --- /dev/null +++ b/test/5.2-lpc.sh @@ -0,0 +1,31 @@ +#!/bin/sh +# +# "$Id$" +# +# Test the lpc command. +# +# Copyright 2007 by Apple Inc. +# Copyright 1997-2005 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +echo "LPC Test" +echo "" +echo " lpc status" +$VALGRIND ../berkeley/lpc status 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +# +# End of "$Id$". +# diff --git a/test/5.3-lpq.sh b/test/5.3-lpq.sh new file mode 100755 index 0000000000..03087d2ccf --- /dev/null +++ b/test/5.3-lpq.sh @@ -0,0 +1,31 @@ +#!/bin/sh +# +# "$Id$" +# +# Test the lpq command. +# +# Copyright 2007 by Apple Inc. +# Copyright 1997-2005 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +echo "LPQ Test" +echo "" +echo " lpq -P Test1" +$VALGRIND ../berkeley/lpq -P Test1 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +# +# End of "$Id$". +# diff --git a/test/5.4-lpstat.sh b/test/5.4-lpstat.sh new file mode 100755 index 0000000000..c69003e42c --- /dev/null +++ b/test/5.4-lpstat.sh @@ -0,0 +1,43 @@ +#!/bin/sh +# +# "$Id$" +# +# Test the lpstat command. +# +# Copyright 2007-2011 by Apple Inc. +# Copyright 1997-2005 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +echo "LPSTAT Test" +echo "" +echo " lpstat -t" +$VALGRIND ../systemv/lpstat -t 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +echo "LPSTAT Test" +echo "" +echo " lpstat -H" +server="`$VALGRIND ../systemv/lpstat -H 2>&1`" +if test $? != 0 -o "x$server" != xlocalhost:8631; then + echo " FAILED ($server)" + exit 1 +else + echo " PASSED ($server)" +fi +echo "" + +# +# End of "$Id$". +# diff --git a/test/5.5-lp.sh b/test/5.5-lp.sh new file mode 100755 index 0000000000..1acd881c4a --- /dev/null +++ b/test/5.5-lp.sh @@ -0,0 +1,84 @@ +#!/bin/sh +# +# "$Id$" +# +# Test the lp command. +# +# Copyright 2007-2011 by Apple Inc. +# Copyright 1997-2005 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +echo "LP Default Test" +echo "" +echo " lp testfile.pdf" +$VALGRIND ../systemv/lp testfile.pdf 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +echo "LP Destination Test" +echo "" +echo " lp -d Test2 testfile.jpg" +$VALGRIND ../systemv/lp -d Test2 testfile.jpg 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +echo "LP Options Test" +echo "" +echo " lp -d Test1 -P 1-4 -o job-sheets=classified,classified testfile.pdf" +$VALGRIND ../systemv/lp -d Test1 -P 1-4 -o job-sheets=classified,classified testfile.pdf 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +echo "LP Flood Test ($1 times in parallel)" +echo "" +echo " lp -d Test1 testfile.jpg" +echo " lp -d Test2 testfile.jpg" +i=0 +while test $i -lt $1; do + j=1 + while test $j -le $2; do + $VALGRIND ../systemv/lp -d test-$j testfile.jpg 2>&1 + j=`expr $j + 1` + done + + $VALGRIND ../systemv/lp -d Test1 testfile.jpg 2>&1 & + $VALGRIND ../systemv/lp -d Test2 testfile.jpg 2>&1 & + lppid=$! + + i=`expr $i + 1` +done +wait $lppid +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +./waitjobs.sh + +# +# End of "$Id$". +# diff --git a/test/5.6-lpr.sh b/test/5.6-lpr.sh new file mode 100755 index 0000000000..8e9b982c0b --- /dev/null +++ b/test/5.6-lpr.sh @@ -0,0 +1,84 @@ +#!/bin/sh +# +# "$Id$" +# +# Test the lpr command. +# +# Copyright 2007-2011 by Apple Inc. +# Copyright 1997-2005 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +echo "LPR Default Test" +echo "" +echo " lpr testfile.pdf" +$VALGRIND ../berkeley/lpr testfile.pdf 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +echo "LPR Destination Test" +echo "" +echo " lpr -P Test2 testfile.jpg" +$VALGRIND ../berkeley/lpr -P Test2 testfile.jpg 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +echo "LPR Options Test" +echo "" +echo " lpr -P Test1 -o number-up=4 -o job-sheets=standard,none testfile.pdf" +$VALGRIND ../berkeley/lpr -P Test1 -o number-up=4 -o job-sheets=standard,none testfile.pdf 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +echo "LPR Flood Test ($1 times in parallel)" +echo "" +echo " lpr -P Test1 testfile.jpg" +echo " lpr -P Test2 testfile.jpg" +i=0 +while test $i -lt $1; do + j=1 + while test $j -le $2; do + $VALGRIND ../berkeley/lpr -P test-$j testfile.jpg 2>&1 + j=`expr $j + 1` + done + + $VALGRIND ../berkeley/lpr -P Test1 testfile.jpg 2>&1 & + $VALGRIND ../berkeley/lpr -P Test2 testfile.jpg 2>&1 & + lprpid=$! + + i=`expr $i + 1` +done +wait $lppid +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +./waitjobs.sh + +# +# End of "$Id$". +# diff --git a/test/5.7-lprm.sh b/test/5.7-lprm.sh new file mode 100755 index 0000000000..482052d863 --- /dev/null +++ b/test/5.7-lprm.sh @@ -0,0 +1,47 @@ +#!/bin/sh +# +# "$Id$" +# +# Test the lprm command. +# +# Copyright 2007-2011 by Apple Inc. +# Copyright 1997-2005 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +echo "LPRM Current Test" +echo "" +echo " lpr -o job-hold-until=indefinite testfile.jpg" +$VALGRIND ../berkeley/lpr -o job-hold-until=indefinite testfile.jpg 2>&1 +echo " lprm" +$VALGRIND ../berkeley/lprm 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +echo "LPRM Destination Test" +echo "" +echo " lpr -P Test1 -o job-hold-until=indefinite testfile.jpg" +$VALGRIND ../berkeley/lpr -P Test1 -o job-hold-until=indefinite testfile.jpg 2>&1 +echo " lprm Test1" +$VALGRIND ../berkeley/lprm Test1 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +# +# End of "$Id$". +# diff --git a/test/5.8-cancel.sh b/test/5.8-cancel.sh new file mode 100755 index 0000000000..3b2379a416 --- /dev/null +++ b/test/5.8-cancel.sh @@ -0,0 +1,45 @@ +#!/bin/sh +# +# "$Id$" +# +# Test the cancel command. +# +# Copyright 2007-2011 by Apple Inc. +# Copyright 1997-2006 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +echo "Cancel Destination Test" +echo "" +echo " lp -d Test1 -o job-hold-until=indefinite testfile.jpg" +$VALGRIND ../systemv/lp -d Test1 -o job-hold-until=indefinite testfile.jpg 2>&1 +echo " cancel Test1" +$VALGRIND ../systemv/cancel Test1 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +echo "Cancel All Test" +echo "" +echo " cancel -a" +$VALGRIND ../systemv/cancel -a 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +# +# End of "$Id$". +# diff --git a/test/5.9-lpinfo.sh b/test/5.9-lpinfo.sh new file mode 100755 index 0000000000..adb041b0ad --- /dev/null +++ b/test/5.9-lpinfo.sh @@ -0,0 +1,55 @@ +#!/bin/sh +# +# "$Id$" +# +# Test the lpinfo command. +# +# Copyright 2007-2011 by Apple Inc. +# Copyright 1997-2005 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +echo "LPINFO Devices Test" +echo "" +echo " lpinfo -v" +$VALGRIND ../systemv/lpinfo -v 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +echo "LPINFO Drivers Test" +echo "" +echo " lpinfo -m" +$VALGRIND ../systemv/lpinfo -m 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +echo "LPINFO Drivers Test" +echo "" +echo " lpinfo -m | grep -q sample.drv" +$VALGRIND ../systemv/lpinfo -m | grep -q sample.drv 2>&1 +if test $? != 0; then + echo " FAILED" + exit 1 +else + echo " PASSED" +fi +echo "" + +# +# End of "$Id$". +# diff --git a/test/Dependencies b/test/Dependencies new file mode 100644 index 0000000000..cf06db33ad --- /dev/null +++ b/test/Dependencies @@ -0,0 +1,17 @@ +ippserver.o: ippserver.c ../cups/cups-private.h ../cups/cups.h \ + ../cups/file.h ../cups/versioning.h ../cups/ipp.h ../cups/http.h \ + ../cups/array.h ../cups/language.h ../cups/string-private.h \ + ../config.h ../cups/debug-private.h ../cups/ppd-private.h \ + ../cups/ppd.h ../cups/pwg-private.h ../cups/http-private.h \ + ../cups/md5-private.h ../cups/ipp-private.h ../cups/language-private.h \ + ../cups/transcode.h ../cups/thread-private.h +ipptool.o: ipptool.c ../cups/cups-private.h ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h ../cups/string-private.h ../config.h \ + ../cups/debug-private.h ../cups/ppd-private.h ../cups/ppd.h \ + ../cups/pwg-private.h ../cups/http-private.h ../cups/md5-private.h \ + ../cups/ipp-private.h ../cups/language-private.h ../cups/transcode.h \ + ../cups/thread-private.h ../cups/file-private.h +xmltotest.o: xmltotest.c ../cups/cups.h ../cups/file.h \ + ../cups/versioning.h ../cups/ipp.h ../cups/http.h ../cups/array.h \ + ../cups/language.h diff --git a/test/Makefile b/test/Makefile new file mode 100644 index 0000000000..42b53be94b --- /dev/null +++ b/test/Makefile @@ -0,0 +1,204 @@ +# +# "$Id$" +# +# IPP test makefile for CUPS. +# +# Copyright 2007-2012 by Apple Inc. +# Copyright 1997-2006 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +include ../Makedefs + + +# +# Sample test files. +# + +TESTFILES = \ + color.jpg \ + create-printer-subscription.test \ + document-a4.pdf \ + document-a4.ps \ + document-letter.pdf \ + document-letter.ps \ + get-completed-jobs.test \ + get-jobs.test \ + gray.jpg \ + ipp-1.1.test \ + ipp-2.0.test \ + ipp-2.1.test \ + ipp-2.2.test \ + onepage-a4.pdf \ + onepage-a4.ps \ + onepage-letter.pdf \ + onepage-letter.ps \ + testfile.jpg \ + testfile.pdf \ + testfile.ps \ + testfile.txt +OBJS = \ + ippserver.o \ + ipptool.o \ + xmltotest.o +TARGETS = \ + ippserver \ + ipptool \ + ipptool-static + + +# +# Make all targets... +# + +all: $(TARGETS) + + +# +# Make library targets... +# + +libs: + + +# +# Make unit tests... +# + +unittests: + + +# +# Clean all object files... +# + +clean: + $(RM) $(TARGETS) $(OBJS) + + +# +# Update dependencies (without system header dependencies...) +# + +depend: + $(CC) -MM $(ALL_CFLAGS) $(OBJS:.o=.c) >Dependencies + + +# +# Install all targets... +# + +install: all install-data install-headers install-libs install-exec + + +# +# Install data files... +# + +install-data: + echo Installing sample ipptool files in $(DATADIR)/ipptool... + $(INSTALL_DIR) -m 755 $(DATADIR)/ipptool + for file in $(TESTFILES); do \ + $(INSTALL_DATA) $$file $(DATADIR)/ipptool; \ + done + + +# +# Install programs... +# + +install-exec: + echo Installing ipptool in $(BINDIR)... + $(INSTALL_DIR) -m 755 $(BINDIR) + $(INSTALL_BIN) ipptool $(BINDIR) + if test "x$(SYMROOT)" != "x"; then \ + $(INSTALL_DIR) $(SYMROOT); \ + cp ipptool $(SYMROOT); \ + fi + + +# +# Install headers... +# + +install-headers: + + +# +# Install libraries... +# + +install-libs: + + +# +# Unnstall all targets... +# + +uninstall: + + +# +# ippserver +# + +ippserver: ippserver.o ../cups/$(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ ippserver.o ../cups/$(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + + +# +# ippserver-shared +# + +ippserver-shared: ippserver.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ ippserver.o $(LIBS) + + +# +# ipptool +# + +ipptool: ipptool.o ../cups/$(LIBCUPS) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ ipptool.o $(LIBS) + + +# +# ipptool-static +# + +ipptool-static: ipptool.o ../cups/$(LIBCUPSSTATIC) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ ipptool.o ../cups/$(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) + + +# +# xmltotest +# + +xmltotest: xmltotest.o ../cups/$(LIBUPSSTATIC) + echo Linking $@... + $(CC) $(LDFLAGS) -o $@ xmltotest.o ../cups/$(LIBCUPSSTATIC) \ + $(LIBGSSAPI) $(SSLLIBS) $(DNSSDLIBS) $(COMMONLIBS) $(LIBZ) \ + $(LIBMXML) + + +# +# Dependencies... +# + +include Dependencies + + +# +# End of "$Id$". +# diff --git a/test/color.jpg b/test/color.jpg new file mode 100644 index 0000000000000000000000000000000000000000..46cbfd78f93cf1e5ae620d318c65c605c16be3f1 GIT binary patch literal 90720 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

_v>}%d*hX;p;^$=9VBC%E}v^0MEBPsCxg_N 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 literal 0 Hc-jL100001 diff --git a/test/create-job-format.test b/test/create-job-format.test new file mode 100644 index 0000000000..87d94a5b65 --- /dev/null +++ b/test/create-job-format.test @@ -0,0 +1,56 @@ +# Print a test page using create-job + send-document, specifying the +# document format. +{ + # The name of the test... + NAME "Print test page using create-job" + + # The resource to use for the POST + # RESOURCE /admin + + # The operation to use + OPERATION create-job + + # Attributes, starting in the operation group... + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $uri + ATTR name requesting-user-name $user + + GROUP job + ATTR integer copies 1 + + # What statuses are OK? + STATUS ok + STATUS ok-subst + + # What attributes do we expect? + EXPECT job-id + EXPECT job-uri +} +{ + # The name of the test... + NAME "... and send-document" + + # The resource to use for the POST + # RESOURCE /admin + + # The operation to use + OPERATION send-document + + # Attributes, starting in the operation group... + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $uri + ATTR integer job-id $job-id + ATTR name requesting-user-name $user + ATTR mimetype document-format application/postscript + ATTR boolean last-document true + + FILE ../data/testprint.ps + + # What statuses are OK? + STATUS ok + STATUS ok-subst +} diff --git a/test/create-job-sheets.test b/test/create-job-sheets.test new file mode 100644 index 0000000000..d15bb7d4b6 --- /dev/null +++ b/test/create-job-sheets.test @@ -0,0 +1,55 @@ +# Test create-job + send-document with job-sheets attribute +{ + # The name of the test... + NAME "Print test page using create-job" + + # The resource to use for the POST + # RESOURCE /admin + + # The operation to use + OPERATION create-job + + # Attributes, starting in the operation group... + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $uri + ATTR name requesting-user-name $user + + GROUP job + ATTR integer copies 1 + ATTR name job-sheets standard + + # What statuses are OK? + STATUS ok + STATUS ok-subst + + # What attributes do we expect? + EXPECT job-id + EXPECT job-uri +} +{ + # The name of the test... + NAME "... and send-document" + + # The resource to use for the POST + # RESOURCE /admin + + # The operation to use + OPERATION send-document + + # Attributes, starting in the operation group... + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $uri + ATTR integer job-id $job-id + ATTR name requesting-user-name $user + ATTR boolean last-document true + + FILE ../data/testprint.ps + + # What statuses are OK? + STATUS ok + STATUS ok-subst +} diff --git a/test/create-job-timeout.test b/test/create-job-timeout.test new file mode 100644 index 0000000000..514924d423 --- /dev/null +++ b/test/create-job-timeout.test @@ -0,0 +1,54 @@ +# Print a test page using create-job + send-document, but don't send +# last-document = true +{ + # The name of the test... + NAME "Print test page using create-job" + + # The resource to use for the POST + # RESOURCE /admin + + # The operation to use + OPERATION create-job + + # Attributes, starting in the operation group... + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $uri + ATTR name requesting-user-name $user + + GROUP job + ATTR integer copies 1 + ATTR name job-sheets unclassified,unclassified + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT job-id + EXPECT job-uri +} +{ + # The name of the test... + NAME "... and send-document" + + # The resource to use for the POST + # RESOURCE /admin + + # The operation to use + OPERATION send-document + + # Attributes, starting in the operation group... + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $uri + ATTR integer job-id $job-id + ATTR name requesting-user-name $user + ATTR mimetype document-format application/octet-stream + + FILE ../data/testprint.ps + + # What statuses are OK? + STATUS successful-ok +} diff --git a/test/create-job.test b/test/create-job.test new file mode 100644 index 0000000000..61336c63bf --- /dev/null +++ b/test/create-job.test @@ -0,0 +1,52 @@ +# Print a test page using create-job + send-document +{ + # The name of the test... + NAME "Print test page using create-job" + + # The resource to use for the POST + # RESOURCE /admin + + # The operation to use + OPERATION create-job + + # Attributes, starting in the operation group... + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $uri + ATTR name requesting-user-name $user + + GROUP job + ATTR integer copies 1 + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT job-id + EXPECT job-uri +} +{ + # The name of the test... + NAME "... and send-document" + + # The resource to use for the POST + # RESOURCE /admin + + # The operation to use + OPERATION send-document + + # Attributes, starting in the operation group... + GROUP operation + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $uri + ATTR integer job-id $job-id + ATTR name requesting-user-name $user + ATTR boolean last-document true + + FILE ../data/testprint.ps + + # What statuses are OK? + STATUS successful-ok +} diff --git a/test/create-printer-subscription.test b/test/create-printer-subscription.test new file mode 100644 index 0000000000..19f5c85f5d --- /dev/null +++ b/test/create-printer-subscription.test @@ -0,0 +1,49 @@ +# +# "$Id: create-printer-subscription.test 9084 2010-04-07 06:54:31Z mike $" +# +# Create a printer subscription. +# +# Copyright 2007-2010 by Apple Inc. +# Copyright 2001-2006 by Easy Software Products. All rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# +# Usage: +# +# ./ipptool -d recipient=uri printer-uri create-printer-subscription.test +# + + +{ + # The name of the test... + NAME "Create a printer subscription" + + # The operation to use + OPERATION Create-Printer-Subscription + + # The attributes to send + GROUP operation-attributes-tag + ATTR charset attributes-charset utf-8 + ATTR language attributes-natural-language en + ATTR uri printer-uri $uri + + GROUP subscription-attributes-tag + ATTR uri notify-recipient $recipient + ATTR keyword notify-events printer-state-changed + + # What statuses are OK? + STATUS successful-ok + + # What attributes do we expect? + EXPECT notify-subscription-id OF-TYPE integer WITH-VALUE >0 + DISPLAY notify-subscription-id +} + + +# +# End of "$Id: create-printer-subscription.test 9084 2010-04-07 06:54:31Z mike $" +# diff --git a/test/document-a4.pdf b/test/document-a4.pdf new file mode 100644 index 0000000000000000000000000000000000000000..bd91ca91e4a30f3fc7034acffb40acd84d80e36a GIT binary patch literal 1062429 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

9mms|OD?;fTO;1h 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%0AX@Bv$Vt zYf`X!uD9(^^Dmii^*St8<#5ZtpJ?~h3MSOZ=Mq+of;5?_Au*smU$LjI@4hi_Ca$4V z5RYW@`9KKp6t(T{;57TtLLsGY@~eF6aW>w}V+!4S?i3E>0}~K@bU#W-dxwh)Wc{tM zqYvxV4($KmFgI01dY?ZrN5ug1pD;uAY|TDf|E(;roJp}1RmXfyd}-9mVI|%4`wQ~` z@1H@w8;(&8+mQ-Nm1F(ji(jWg>y1Gui6G|3(Y`-`Q4(@0W=BY(pmQ;0K!jufAcw7_ z;gF$$kC!n(UThVlHjo&H8UqFN7R~0J7eS}<5-8UhM`1Ds@{Bxa2aCIf6<)BJ?%gEE z?wJNJHelgXl7xi(TCI`Tz(88$IBS#HYm+XQ`Bm$|nz*I0l(lxt zCyS<|b&rE3!t)*7vuKLEr5X7j=U;7tOUH$LS7JNe2dF!?+~Aaq6Cwy^d@VT(EHo_I zC<=of;~||VYD4o?xG6lP4a?1Um>dhx`nO0)fWl+@FaLT<2RszhF(3w(Z5jV#nRmPb zPi-IrG$HHmwG3}?6|ez?{V%4(_5YSG-^*0K&-x}rlft^2$-7NVoy5A2htf*BP)~AK zX+BjphO)6^8DE@AeKx{!>_|nkdY{hhoHVIK<@Ga~zH$B$QE3`k*-riS6;(y&9z0hB z4j2!ua@d>;xIKn;;vl%NnY*xtMzGy68G;x@2sZ?7R9lFX$^ZTu##Y7@KVC+BPP1pe zx!G2&_QpqhOk}e)?kYM1iyiheo11lZNZmz?Kg$_%tFmAG5cpCUijTr&a zuJ#_i+O&x}WgE;@Om~&l#ee48!?0ve6t`$#U*=oLyAl~x@f@ifz?pv-E^$doQALH6 zl0B#G&mnm(DtSMNcL#)8zxqn_djNDbVP>-Uc0P8d?75=Hsgcho5IVLZD}0Z1FV3lW z{50{I-t}P$7HUx4Y0sJ2GAAkvc!J`hX8G)F))0WN67qL*D(*VWYUU7gd&H_C7<$Y96Hx+=UAbhI! z1OGqs$z_4%0m%o6SZFTXOsu;nfG`OE<#;~*-it3W&C69OHTstr#cOd~f_65$j(VMXrdb|nuY9N9cCi^o z9}dX*-FwlB>#OV)dhGQJ&%YJ+-n!S)7Q2A4lSjw_X97?JIQ=Mh)MM1XjjHC86#pcD zqYv+jK+bj+#nx0;JVl}R4_g8AozwP_HSp-V%;d6++DdS4%?-czu9)kih{prJ>-G~j zOdY7rDT>SE#9^JLjFAA9RLheSxKnI_3X~`(sUM3&O7yI(wtff$JZ`%#i@Zh{;=kU{>^%Bs5@_Xw8BMWpT*K#frcbI`!}Go*gnj7+rfM_ z7^_Z-YlO{~_?tOe?${TL(Sy$r%ni=-VovU)+nx*+Q>3VkTP7B>^K94VCPtRT;(=4S zHo?UClSAf!hBL_~XBqF;&7DWHo;$+L+p``A1glKa=NZs&EJQGTm17Uw9zuj#YCOY0 zR~xecn;r|D6v0%c7)Y%@PrTd~8=0r?^L&XIrEgu0#Dos!P0u%4=*eT`J}hi^g>=Bz zIjV~Wvj0VY7XKG>vzpj&6o03*8D?)aD&$_iFkK4F*4_?zuI|4gE?tTSkkPmJJ0xul zuj+D%hO}G@PnxaVcoEETZoRd{u|Qhky$nZTmN>m=C0--gq9 z5-Xt0b2P?AbSbfY3rcuo&wuoA%@-M8u7rW+Y-+oTUH(G2e0_P|SRL$|jhXhiG3lon zke|JtpN{$81qiM_1T1kRSj*>F5$Er0D-E2(fYNiZGfzvla;&WAco$Z+H~CXS=&qeN}*whzsg;?O3tOI7Q=-5}xs#6U3q zeyBPKATp+GmJi`Ae-rVaTIp?4I)z(_|D-VZMQ>h3V^&2IU?SA1!_}$IR`r_i2@8@g>6HU4%;T=J+hb^;g;P41$@40H%MpMyn$)qUyNe;41$?>h}cb zPdJo^r2<_=DIn9iKms;__hg1Byo)db9PvA=dA}iu>1ghF?_>L$rzhAb4-sq*EJ%Ap zEUJkOMZ*k_c>t?elIj8O0z!fK+R`MB_zm42av|a>nBwveq}Qn$Up>Q<_=4DIpro7q zZV2#Ho~ry`tg(_eT@1)PCdZWizvKI#Qo|j<%FXlN1m&`X+VprkOuQcdbtGE647xg* zNJP6LexGRsp?V9Qz|<;%n&-@L()%@%e!y%DtA?RLFeF~?*AdLL^QOE|EBMcqun|T9 z2FxP){<877EvO%id83uGA*cI}P}1OT`y1~CME3-5ly@UXJ9rZEXSZ#6d)MF0Pr_-Y z(v>SKwJ(bdO*0@()31FF*2m(F|E2$?&4L)!mpN$!;(h9?!vl4CHOP_oFKrYtV(HK6 zx%$0jV#&FQ7dJq5Z^C>8X}q_0bSwu4-kxH9kRO@9T>N7{^((#hX!7?PuFX*`oE|oG z4<=qU_K$Zl6Tjy*dku5X4H6Fs+k%s`PkWCV&;pY`gJ4tGz?9v>jmpu(sB5KCw~bQB zZ!Oe-IZ`7RZ#*|@%iOJQC&JmGWv4Q1$VKSuNi<>Cwp|AP7G&rZ+y@r75}4g^D;|97 zMfs6npP~JYylXh-;V}Qp&j8H@)@Pm!qOY8zpAv$UEIet1T4Jkx)zVGdb&1*woj;w2 zN|SX(>P+-N-|Bbb$Y>?xKHp9jjD3JXfzsKz8e0d_8Vr6M6=|F2$^%js!+e)nCQbS` zqsW#?9q(6avc44cx6+@FQsv(uwfu-f0M*psB`9ld1>-JX{*O|oEj+(fB}(3 z{BistQJyW)f``bPEkB&)W)LaK_*2G46`VBmhYCr$W6|JZ)KEsVEGDq@%CyEcTs@?f z#z4+7(0-_-F8c%K(oYllMvO0X)X-(3&mXSbCngocDxxBnG;octTUERyC`u&rw=8l9dJMEkP%7@0AR$L#jkF?=>qQ z5i8;i(%@aHLvV|gf2R5Ch+x&sVE;ij8+eE-Rh)bUGl1# zq$_Q$gjv1PxP!(ktIEF7NGG?ISLmbMM6}11P$xLx1QxykY4m@L)%dsJX4SNx=YFw& zE#$+|xtfZ4yt&?RH3zPjuPrJ~A0iOj7W{!%mwvNCGyoPqyoog$?5YY|Nd+!yxAL@7 z*5&m5Zr|KZ4vtv7GPnygye_6M1D@-li+TFpNV&}$_!wu)3C=#nr~dQaJDe>rHsH)4 zN7D>0(+svz4+h8+yrhx zkEOWJU{4q>>BW>HrQ}hNZnJ>l4->_r|T&l!_?@@qh^v0POCvVB|%O>TsGr=_~NJN0_Fl`eN<+*ZVDzhQUmzQWLs z1X*@l_cpOa{pUhjT9IG}5nIM~36<%7V|mt8P=Tt<&QRM&wboSwJ6GY>R~onh%m1GR zTv?Qptsl52EuybpR1*s8zMwuokh0w*x2LF-CH(=RZkNAVYsc?LMT~umJY%5W4k}KN zdq--9+jJ}q1qhoChu;kee7OAv3?<-s;T#J?MFUIp36*l!vPt4En_RUH)A_UJELAp& zTt+)7Bw_orEmkxB@Y#nx7w5sMfWe}ZT3wp(bA%d(H~zl`=MPZeVL~pu(dBhAUhF$B zs1F?Iw2{c@Nn;!Ul2oA}Xmr<#c*O^5Vn^yPXBFAv49h;Up3J_7a_$Ff_1h~gJ4=fN z*=a%wTImw;cSBb{Jw|2_rTeS2HE5dd8LrOc$w1Ekh$+u^tEh%6*Jd$M^@uC^zobt?pCIrmMW6M z)iO0zOP=fupy`QPf>V`|&SBL~2ZBzi5e1Or&j?3#IHXg$T92P3b1>?XqjGv-C5dgD;GqCUlkZ3m) z9&#|B_bD$Xhu^0c?fRZrsf0axbmaIQnLQP+JEYkwOE1# zKw9|MC{vm+U#6>c4a_*;br%4x!bbg8a*D_~F><4>vppobh6R~!XXX2t z=gr}!3+=^&Y(NnEIe0KL2$FEUm)V0U(0&W86m(m?%-)olC|UTqw}zeRtA zq8JHJm(|~6;{#cvF1x;Im8%hzXSa)x&7sr|r^ z>p!He)oa8~FkWf~Vn{ne;UZxRE)%l*=N!;%__dlv1I>X;AE!}Qw9LaYJUo$Lh>Ccpjcld!rZ2Fr~sjSzpd33a1N*;Z;9FEPKi7Dvy`9O@%&&5`J+ ziQioS$M($%Ztg(I=6bFd4FZy;CsL6ltj3F1gAFVQqt7zD6vtpLtMzW;?V6`X#SO0X zja%p750@_?xQM|!n7RWda+16}1ZW#rYb!b^q1hox0zp%uAZI$CUgj3*c<0nQzds`{ z*WP^vv?}p5kMX>n+YDVymeO8g^b(P2En~YNos8XGUp;{zw`iC?)K9JsOqdxR$Tq)p zWd3sVg(!GXqrTfqRIK2c1)_iO>2?5ms#a)w!mjuKEitrKSvn~n70J;4_3)nhC&Mw? z*-QSF^h#;!@eg}(hw_A+#GXh}n0FUvqCzX!Pw0b;P=-YCttI~ba zPt+uiVO^k$05MPLxtW#dk`hzz7FirvHne$MMA2nEx>Fdz)Z}W{A3bB8SX~v(rP@ky3zBy z?s9$H6>tt-L1~PQDJ}MCjj>;K+J&cht=a^@!M2!)Jch?kn|rneXNPLNOsI3jU+cmW zox20oOe9^)4sKz?ySX^J)CN_3>F`H+o@!0W$(f4Z=&Eek($H!?9=ROG&Yt;)TS zP&pe@I;(DZIPIiPo~fxo6)YanE$#R7L(B7&IE-!5p5S&QP)H`flXwYcQ@+)dyzwXm z>7M^=ydZd!nmJLdsaYuR1i%PDs{vz`LCe)`?TEL=ODY$ zQ0BQea!kr_`@&d$DuOcuf?Nr5b06}~B_-r3{^lTmCy+~py#Dp~%4 zsj7wXdigPWxly{g(UiOxqklE$;ZU_JY|b5!~Y_-t(9h2+l7;mQH== z4xFd9Y(@e8AT-L#W_ZoAmt)w%HfWePX;?IC;+WJfn%1w`bnH71ow`q+d*VeKVBo?F z|FxF^>yQFG4_bW{zHt@IMU#Sk*_WZbmr?64BG;aWqrIRv5oAY8}Kj^Sk= z6jVFST)1)e?Ir-QK-#x~tKb4~1A+IxJz5LiwjMZgnRwy9@FH;WRS55O=<=m6Y8-Ki z?F)V3#(n9v0`D+Msm~PjPZn{F6SIwyFi&`7o%{d||ILteVhBA&lY+7Cpz(qR3=#88 zF}u9`u6sd!H%Z*rNn5Yuw_inWzY5uS8@T@3kN4J>_sVPj+>!OtWB!#d_k}lFp@x(P zKk^J$bmNAWLhya6;8yT$(HFKF-Z1Q0BsEMamBSTijAn0PGaON`ezP8U1VVHGH9Wp< z*fy(CH7b|S(X8GwXj>zol-sOL!_Z8>FTKqs}9% z!7Yn{UHdVfb$35GJ6p^)TgoVjU^{A1bQU;q=7aX%qIDMcIecDOo z+un&RY{3wnR)AcFM`?u*0_R$$v_jglYS=!n*T~kbVd*eOh^0MpInU&C;ReCqybZFU zWfQbCfD64+rw%nuHE3q**H4ja#&xPjNfpDIWjzY{eaZy`#NsjS@(Bv+`QJm#?a|2Z z(JJWIEqSJ$-=&$;p_L76XUfF`)$8Pvo0S=z8ri*&BcNv?N65t0q?A%8mjLjGd;yIH z#%2UV*k7w0%T$c6Qi!Tjj3|)}$&*IY0rF)0QFqYCUW~`6HE4pQbDTI*w3I`Hgk6ZZ zRe-o9Sb(R9iHE4Ew}h3Slx@&`yAUb+NGYep$L?t|K4=rs6aPG!V0>k{NG1#fSpZWl zsa8FuUX$LWoz_TVzp_s85o~}J zdB}_)-Lw$x^iZ9QK%+z0yB(V=^gvV4d|0rJ)1(haeXMrrqEb3|85ME042mo0?1-GomD3Ev+l{EW^2 zIY;0(T#-MlLCi1yw;j>H?~49oR~Y#JkL~+^+eG|_`2Tl{sPgkG7XQ!3`Tu*E?`K1V z&qw%wF-rK=1mSmV!k^~_{>&3X8$oxZ{=W0z&s$P|+I;l8QE|L^NrFXrf=zjxRb`xQRg4W-Kt+^I zd4z3QxNT9Gb#ah&VW3^EzkRkpRs}-;OvV{L*eZaBZIX{gf{#hEn`Mc!KRc{pFJkH{ zmIwTQOW(duTz`?c^&)ZmI$;Ma;3{VGJbd*$V)-I+`7DwLI*(pKir@kNfzQ_=XaV=Y zkp%=ku%Fzu8Cx~)Lyd@^4$T`iPwUq#m_oMR!lp8pOd7ZCdr#aZF1*-SSAW#Q|H6xV z6Sx87yb0R`{y&Fz^Ph(p@-%P(-W)6di~t@43xH>D!q(B$MXWCn-ud&yiFNEa4djOq z8Xp7$Vmq!?b!{Z#yjpT;L_`0-kY!$)F93sb&bAspTBUKKX>Ll_gvh1 zI@F<_MZ4!3FJc!fZWSkKnIvVCD)p2h?vN_tkR)LhFN)TG;{^@rLZ)fL*6G6bV`hcV z-K+y)k~ZrTf2-ZSsnqBaBn=j6+{%FI=b`Yruda35g!p5xepMQior@WKa9 z3=wLfsQ;Q%`$gc?sprIr+th_C`vNi@t6}sTnfIY30LO9kiW-CUW zXraNVi$_Ms5Ek_s7jzqDv}&fbDo54J29yi?6!YO`mQv3(YX$!EOxqU?niePxi~5al zoz<=9kQ-S#wUcNNOdX3-3*_%rDR`!w->*_UpECBdlAQy^?|FYqw@{v_4aW%wb5DtI1 zsfI5=Os*rP0{_d=l5}F7Mrw^#22(4eT$5Htj4M}*t0bmYY2~!rls7u3)p!Rp1D(qJ zt;+qMR{FXY_?Z?3S`-ADW%(Io_@UZPRtPygP$xYYqJ6#FI3MFbBUlGnKoIr43~?a~ z2rxO->tcW2 z5c|`HTY3KQESC5NA-7U{acBQI=#?nQT*; zU{e)u!;G`Jvw)Hao1!qQl3<&nAo~LUr#S((nE{p=L8ci2CYipL>0WkpFU0>uUsJlL zMW&78uy5gd_`qQZ`+3~*%fz)8No%iY+czm&*9n_16SiN(?_5NzT}7XbWtbN7gP*@Z;9qVO4y?wLE@H)Viw6_mWhHUDT1a9AuAfc zW%-l9{h*{|@&L6olFF{@I=R<2yqZn)i+Z_$gIHW;=;gbdm;>H_}n zSv*_PshdzNnpP?S{;yM7AqChpM3F!6AB+I_507w3_1n}o;QzMivlaa|cpS)&_I#2X zS9BYev}!Q^Gq(*p_f4O{N3T-a;7vBL4iAX``mO8QO&o=C;Qxwx-3GaN6Ppyg>o_z{ zE-MiAE`H>eE^IQSn|~QRb>TOK`aHW#0sPPGCl9SgAl=xv95}HXffQiV6g8CNLY_iJ zGvk)vcft4%-)d2(0r)?!QxEW;)T|g)E9qA%0Qe89mCWeXE*ZD3Sa$HtTIUU#@H#(7 zuMXu1T1*K3$4S*=S{0Le%rWgsfPa@l4n+Pey&8c3D5gf8vuP=a20p3IhTB(Gq}iEC8RWl31&T z$X}@(4*@?IL5prigHBqFCaqREx=uBzPCdO=GlQu?uU3m=s>ao*CNWj%jg;(0Ge)m> zMx$$Jf$7r*myj;6#7ZyE5_i`k59bme*P;N29DnodK!fa1N=C3YC_PXY;E(a24E$#V zlhOk9K_BuTSpdd=+yZd;e`x`}#+U^dgFb-20r3BS4gPV$f5`j~y#Rj}g!3PD2XdrF zJLrF9As`l;7jSCiFz+!g;PSirvM zKX)ZT|JacL{{PRi(C;|>zhn`9Hc9~UfA)<3hdlzHb@P4JL-=toN(8=`6#5+-tp)rK zPvnnS2Z`UWh<~wkAMt#7tRX0lC9l5K6G4Kv=VI@TIopd!kuINY){#JVKdt}xI(FVGg92n#XK3NnLS zAk)W=;bWiZZJOw7%Dx3zC|N#+DK?fOq1h0dNv%%4XsK2KP=PF%f?U%QS6;BUN$ zUVjm_`66-y56A(?hyLh97=-8-fs5C{E3d=WZ-Uk!n1?I?T;RlYX47_f%?i!(ShMbh z|7RDCT6XRFcWnE1YzFr2hf!yKY^oy~_~^^K_FsMxw2a!mg{%VqK>+;M@!OZ-%SVt1 z1TR7ga30LN2;_oCTws0S@Du=F2tYW+_>WJVfv2u~7q9)fF9T2)h!Z!|ZyETHW)s_u z0Gh5nXJ7j*z7AV`5wL_7VUY#Qqww2}b>ue6GO8-PA3zgw0OrRDStg3x(jPdY9YNT- zFkQ?h`JP3JfO+aYOPYW=UBG(Qq52h#`;NZ-E@=-e;7#P#>#&u#k%<4Ve7R`!usiAn z4fi*`%lOt)$X4JBU$~)_Xb^rX;q>VwS{^kU*fs6nqIS-!F{c&FIT|%Qax>n94K?iq z3!t_wQ(D&a+qO-64z2rv;VhL>mSXWHwG{$*p#7SD`#QB_o!SMQKQQfsq=2VYziZfW zWHY>L-UrWa8}~r)zo1nI@L$%c=aOn6eOS<_2L2zK_w5*UZ5VX!n)X7bfQm(Cy<7V2 z5cYE*{#RwLks7z*%?;bOth>49jji&DrT6`EMI9?-z4u&uuLD^p?jvVz6IZURQ~R+a ztC3^t(L>9@W9y+~TX3JA4WrI=!%m?68WmN3Rt-A94j`RDk-ts@ME+=^rA9TtAIJ|H zQZ1g+u3n%vE}OJ*O(G)nJ%7>RjC6lUnB-SM3zd^optW4=wR`&Gj?M z2_|O*YorIMGXk|30eWbYPY^jZP$xAA{z$=vSAK5wKK{R}|J=p@AMzi!0G$7K#h-sE z@4UzRhc19`{{IU9W1Md-0Qmpj0`xyD1mV~R#kx>I@Oqs8X`aRm50g}PW15RmvWsDY z3-xXgWW1XW&4ZNduUQtNTp#_UJ4tFVO@fmvuwE{7TrcvXUFhrm0-uiYLGb@`4&m2} z0>51r{L`AqUpB=d+W*In7#9DFqRyY&V*l6_`RkhSpLq9vJIDWvalW4n5PsOh_x%q3 z@3kU*zn$*~oqRv(6Zm{s=+~2k-^}p;eqQi*i-N!B3H@$a4EX=~{QVzIOMQDp;`K9$ zqm~DV|3&h>S*k5`)ru5qX)>iWnOdG=TuC>pO0{IBTGgf6*3oPlQf%szt?LpkYrz8I zEGuKo%OlOp!fq{~D99G|f(|jy4KmLOu*vYYP4j{H-!#$3AlcJ6%hA5lHe}qdbjy2U z$8TcWcYMQVc++=eCt&rJbdjk2tgQ_f9l6M@R}P{44+(4HL)% zwqFLJ=_)tT8&}b57m+JR{tG*vGl#w$;Qv+d@+Bl1SVvHFcGzzTbz1O7$HCCBc-Lus z;yiwi0Br~LMUM=z@a_lv2{T*%p zqol*@h}9S2%dew&?*dof_%FjdfrRVMo$ZMR~=gyFEpf*8A4wI0-Ko$T2 z^O5x^z<<{WklV39tYOKQa){N7qI;9y> z0G@WknqDhz0ee`N&JA+ws!r39N%yh!@VO&rfqj!+_yXWTr`ALJ;4El0xoedQDNGwz7i~aYH$>$uurm<`JQZ#_UA0mH%KcJp#)U;ySI;Y=&!hT9U*a6OemJSnA1H9}5{6~`?mGcLP z#S_{U(_|(Z1fW%lkANA~Eg9A>!q$HaNA$7dU|M86z`Cp2hhwb|si*%j2=<&>Dm<=Y0A`T7+GSZDZ= zvVzsqgNZo*5%7cbG5%}Q1HcFJ#-K4`DYV;InVbS zp1>bg1pjAU_^+Fyf87-R+m`S@w?+QBBMjsR{e46D&nx%-2p+*kYra3~<^Ntg-*=k$ zKdIvb{bwWJcbmZi1b)&h^z&iDFUJXAO!I+LpfdtXB3~>@{*v?HGuHi2h9zHhOXB>W zDpu&vQSD?9YZ(Six_(tEwK~m2qN&6>(KQxf9|2z?5XeERnX#*-~53md&hNZ z4}v|fIn=i~a2apW2|Nc5Lkh5NIk06p2--C7U!=5bn)e<(9mDs5gN_}>@lMF7ftwHK z%6|bJ@(Sj@2wQ;^;38t}GIISih`Z-KyYD><9|v9o@IUsNJNLOAC4<8sFQ?;U1yEhc zXCAYFS^OLsSinsn7auixJm_!h7yb$O>`?&)HYLi+GowV~4R5m$7Bb*4jsL zseI0KVaG&K+jt4o^nDn-yV^`0qBz?jBwx!_WFlgxXav~`oPjdG>P z)r(Nhn}C&9e#ix$dvMNOW?p%5L9e}e*KTN6JXioS0+*S?rxU<`NQJIlW>0L#4@?Jk zs67a|#9HA0tZF4H)Kl9wOu9hmSe8Ni3b|zoIIrIZ{0HU(x~Jp{SG4QbbQ|Z@nJYTY z87uNh^*VR zhC=ifu!CdZzi!8(VinGRc(+yk<^@X4m{v}+OhmbaXRe?F^GW!Id*_wc^pV5x4g8nj z!n&{>J2dS(FzH2|qD^}ttpL~9Fz8sNwDNSDS1BD!+Rb>=Auuw$`y5BBc1o>mN~5Ar zK6gO5a7?3oTDNA7TDL&0=NdFF8#T}Bp&dSG>a>0X@E^#}CfCgwG*0W+O^|B_G)lXb z^9R*SrpVPSz3NfT(os^`D5(U5rogLa^{M9$kP3P=a-XT^z|D}(tsM+%!-F|pYH982 zscl3$z#jxLKfoWteh}~5wGlOn>N+@xT|72z(VFUnaCbIs_5{9QY-2QN^;55yVufphN&)7jDo>tJ6rWQm0jF z(8@Jw<;3I?mAE41s9ZU8WEeg=LpHd{ICt=AeY;Uvqi%kiab>q{U58zDo_<2Qfp4*g zcTs>{cA#!Xh_utoZ z;;{3-xc+4!00TQ9JF#F9Zl;ND=x8uD z`H`INLn;hXuZd9The(yDNVmI!>~Eav^5i{EyJ`e&7?Ekv?I-=Bh{pxZrn;UX-+n2Of+hUH?E5{ zsf{$Njxa3`GcFD_E(|u#^Eb`&v(E9iru&#Ac~KMHsp&2@dA4pHUbHO_H1G>t0JQ5f zeiS%$5BN7R0>>UIqRidC#5sae)8NLskIDyDn2(j^hWeENs*M>g%v| zyn8=n0eg;9_@eL*Y5_OCX56=JIe20}al6scYG}`92tsZ&577&ag9KtDT!*1j|8?l{ z^N4lOMd%8A^s)Z}c+I}s^pWQ*IwOV+g~8W<(d0J|KqCw!2dJ5i9@qN;QtP# zV@0!pEnhaPT(PRx49wrcT5-Y+cmd9T$N~<`dLU@$s8&qM7Az2}R&^RUs#OabHIM>q zQ9JjHdO#aG&1q(`Z^OTM7vltIaPz~q%H>%w{R zg&X_YVd}(U5a54k^z6{6=fLFI1~xbhzBX5BsHS=1`InC;6^$H07 z$FVivam|WZO8vZH1IGZO{rY)w?VK(X;6J0ogs>mC0C)jgzkY^__WX=!R|5b0)k`4e zhsb|Yr*fEB)UTE|sE+3R0{IdDHFEkjbGubC@r8gG!Wc? z9kdYXO{%E!kFWUx`I&O@0DrVuNHM8hC!<+272sbf9|Qb{8@xCVs#1zCmyZEf1D`P^ zf$|l?vgJe4Wc-pJdBs0K77+Eo83c(yG*%RfdEkh8L_c()qg4R+toIcp#D7_A3V>WV zBox5^Qu%1KDHK}*K(!r}1o$i_F{4V2j?RUt#TKf?#EtW+i?yVy(Eu;3&#$0+wyxQyRG7GGNirn3Ee9W_h^wNUW86oP7U{Y!z;y+3O z0(H`YbZCJTe5)s(0=yTRy8dDC|7HE>1ODR{fbk#O2yHdWJd;;iAG2x>!!Z)h<{-c)v`|Si|0fU5J zO!EJVE%>Xsd!MsKf679`;=j=+{<2f@xar|al{C9pemq}oD2v#gP43LpZO_#0%+~M9 zF#z>s8}w!y_GcRoWFhru8uVotf}Sx9yXn*p2DLrepf%p0Db}bi+K?G#SQchj9AsSR zZpb^PR-YG+>es?-X-lp{Qoivji*6# zY64b13SI~PAK6Y^yKn&h=TE24>{%BMGw}Ctgter`mjIqS&79hfADIvBQ=jeYcWscG z=M*b||6Ec%-kfLM5H;$Duo>V#uUU&maT#}RQd(w}%f@AL*viGrq`Enka<)=AqySs` z9oy7S%tz`WZr{;u-Zkt7@-LGbHps19%^I$H^_q6=iWZZtPzXQE)vDgoZ`}rSL5l*7 z>y*|_{ic0`u3d81ntBt&|4YhM%bGRII<<>r=D0>~!^4nbQHKH{y9!CKZKv+5fVm3? z)|J!DvGwqYrI>{7-0CjA@mPX_mr2-vBXxP$RFGnA4%e=u}P{(8?RuE*dA7O;9R_NW}x1 z`J=iegIWdMsu?|MnVss4E{!x0@V{9lxd}V!h4%SiyM6FAU;GF_qY4dt0Ua6pKupEN zY6Y+WfPZ4Oa#D?IN`nRiRI5gV2SLS3kp+t3S#rS)X@9VQ#7Ca-kKE!Oy2d_sNsw_* zmi0({;+Fix9UlRiF6#~O&y_)4KtbRFMX0$@REcslSU{0{1gKac8uANVa9<>wTqG9< zuPIWCEF#9D^S{KDd}4BjN*GPaD@h)WrwkXjO?d2Gpb=MRoYimFKIqyu8qnC{kzVE+ zSmf)R>1&uCs0-w$hmh#OI#}f&EI{u)|EYHZ@?#tR4M1Q4seT5xHJ=ame=>YbLH{rQ zzb^q%Kj6OsUd)MlN{M+2c&9>y4~lm(1jRXjsaO}o7-xeI`H!yx;EBT>{NJw>-||1h z$1wda1@JUuc$%WK310d%Zymhte|fM{W5ko51c?cT*j%>gPATD}n((qk@asK-pAO&q z>D0Yn&I_zjx?zz6*X)2~^C&qq*T4{ZNlGvN~^-`7g` zZgL3M8HB4e!gVI$Wgg*m5#i$s{%$#N_zsiDUMtlO{oeqs0cF7_chG$L7h61-3=Hn=9%^m zh4$WT%Oco$AG(9OP{4<7xv-P=>yxT1FMN0vk`ECL!0qqyGe-u53EN} zgG##*d`uXcndmip<%=Tri+}~-Kg9f(p^yu#ocS-F`pkp5Ku)piFplr@Ja(Ld$obNJ z?z!h}XE{hEuH4yRC70fFs7IU^2cIc`Hx9k>UwRe1`YLb*YtA{hX)&;G-oI}-@Z4?s zCF&qO`!ZnhWx&#F|K)3s#Va>7O8(e=qKBB3aSsg$NI|XPEaHUBAq7a4dYUYO+CisD znI+!SV~Cp31+40nVy~j+zLkE28kWXxU4<>*1g*RdS$iF{3KnqVvjj2!vCRbV|I(3t zVmk=}7l6Nt1KWucN3>h=+L?XjIDKH;zo*}`N9hFq^VDkD@}&SzyhwtM}k3DC1M2@*w?OTH}bR^0RG$Lwq;^1pnF>f!F-3(zNE(7&}{_x z^E9fs>XnOXy#nY)X>#-xVet`e3eh2U$J`}#pCKVN(fd7l6 zh6Sy9tOSINW`)wUs9igwR?b$h99J&xmB}7fDVo-;!S;+bpd0-qMrBw-n z6aZiU#hN^n4XT$w;sf1kJgk6 znt5&vm^RJt$qHs;57eq#Iw=HnKC7k}QDfJ{ED|KJt=+YWC|1|XdO(e_BV3)~8O z78e4(6BS$b{qXVktH3VQyFEg<3q(3oLFvAzD>cK*FwN5#$qO<9O1h6uwx3pR0I@V! zwKhVbGhTW)<^F7@I5%H(zl?CkB)n?o|3;7C_l5+2JaO;yX~AEyAy()Az0 zr!D-SbqoAtfPl^jObGmBOz^XD(eDh2f4x@%_1%ZStSGuxx6%&Ry$ni2q04a|fQ&M?N!<1wj0N7CL_t zy0q&%x8pN=6ufW|$-Rl=y^h+t2t`HU~I-98ZZqqd*e zTu6LmFu?!t>BJ${OBwH_e-4qqGqQsVr`sb`U4eZ!V5-KI^g=2eyY z1-bGC#j;gW^@@JYq-I{j!>|G&hx~iCWung8PR-A~(VXQM&YUaz=~IglRQ5OMKA?8( z>vzI;fG-35-_-8_>;wOobenkO7Oq|sI(mi`e*pfI$|a+Ug%JOvB0pBx0shZoi@bBX zwV)Y|$_b5fi2w0zKX5-nEJb@hb*g8m^(<=r6t!kTw{l#kVp6w!N~e5+gjRkb{_j=E z=~K%d(!7QLm`?Gib`d-U*3hq(H=v#ib}&fF2mbeJX0$7(G$|&7&}lFEM0}kWqWw0F z^e*jOPzx~~;13o6_nq3g&BXM2)zn5}8VGLcROoO8p+*gg$#~cgXY zO5yp+5x5FeB0+_Uv4u*Jg~}m?%0UGx;rS|2d8)B_>d~o+Ug^r7AV@11vVj>2AsNbn z=}JD)iVo4LR!IitIj$Ck!KPUutp=a#rY4S2O0cZ{m1Y&1@QZ~ z`VY4FdzS*d=f5d_7WjSr=a&4pSpc@kix>`JJSP0i?_G81L!lDT2W6ixn*w1|!|saz zkD~sJUep{PN}j)NL7-+?s76JYYF(64d%WB*O?oF2-*Ncw*^chp zjvm^NqdoIZEbIu_)VU+;)Cmc-_rZDrp((?tq3Hbk=0ZsD&Tm3j;d)^^0sP;!=s&a{ zIdx|}_n(K)y$M);8@T?)Z}p`&>KA?K!&)@2uX_|#Aml|8u}c=UjTbVH7conbw4h0u zCrKF5B+XKUP141z6A9*Z^3m6j0?>B9may}2!rn(wn@ABGZ-Uoue7WciU1tIKU;(E# zlZO_g2bQD9He;YetC1t?(F+IGsqOf&#qhcL$gxrHntH>6Qq{b2<*ahWJQnaTL$ptA zhg<;hU$t^Zu?#>B!90ZhQ!=>_|3f^#u2Z+BUB9W@vaa2I;RoL2DOqed*t%f=E8VLR&3)nE|fM0~8#RTjF{6PT!N!8L>9n`~ZPQPKATnl28(GGu(ULEj% zTD@YDR0SCUJOWw3xOy4<5e$N3*tBTYHfz|3!+(}qw`9^dL#`TB%IQRn6?v%I+qn_o!#Is?h6XlX6G$4P6R(2057t}$@X@$qtGeAvhX|>8Uz&=<&wE`pp$+c>!kODv=0Kq?~R*hDn z8dstil`j{TEfbveBrr|dH|3#cyp#(l4(tLX@_`w00qJm649LQU4uEB(JoZg`?3EjH5lO(T9^Sn#1JeX zHAIgQs+S(Bmlh17qy-r=0u8=W{KtzwUs?QR_?dt{OcQG|6~7ExBUOt z(_XRv-VVr_9OFX4TRniHUG=e(zli_QuILn4dVm2~K!%Tgwl6i$ms}j6TOOjpj8JWg zQ)`P?=t+_tN_{e&`Cu_ubiL@_ZmHl^E#XxQ;p0xhZ$A_Gw4eV6gM2?4=KJvoKj^bT z{vY=8f3K7OJFWcx*~I@zgWxB10^h6={039t<0^r7rF?G-P}qM(7d%Z6JdD1#9za<2 zBWwf`wnGWv1;_D(>-2kXN(kSoz4vLG0N4j)4L=(}7W2aq(eDn5f2&vWb%*#xlh{F> z_)fLtdWF6Ma4ypo5@rH@w19

5Z&b)^RzBISezI5fWUu=1LDl1fN_4+l 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

  • {>wx=AFA{?OLa-8UZu5EmZ49l3C4&(GB4LVUwah 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

    dmq3T)DTKm49Bc z{B7G~vr13`?_vNJdYLUQHHnvlaJ#6Fncuf4&gw&Ul;y=tuz8vTN`-0-xvGJ20 z>{FWH*)2r)XSeizTvh&UTjTp>`CqpPFQRZ$BbYa70s#Lr3~T8IHS-pA zbmN*?qv{!BWSBh7ZkweTSCb7ZhjoklG_JNO=hsQ6fdrI`#uf`iR7%Cy%B8ld<@V}c z9x*K?SyoVNYp3lSrtO;O4lN9O1g2dJ!@7}a-L&G=i3&d5M%d*&qrmFG`9}d$TOQ~z z2Y4Gocfg?^%Hnri#tz*myO0ZblMlQgsV8kaj{x~^I}YwcoazayF_8Zo{?7<7%Lp*R`9CAr1V<*4 z0EE17bKrk(2W1@pP5ZFATFC=kEajXnWZ96$;c71HYA)?+A?aZ$>1iqEV=Wl!C=l<) zm*dZKJ)Em9fwLzaBjsbWm$8*%Y^Q?rq!zo^j6Lqe9`_(T>4JZf*u6&Vb~Scz16wcQ zSh&VZxgt1lPP8dgtUOUPFJ2wUKQ&4j*dAy+WNl9>cpn#N(iegTQh0yTNw**>oL) zb;D&8HV{9+9>{;)fe8G+>ox(AKkz?b%Z&`#BCI`@o zfCD)^hb`O*XWfoqokT9;C;52U&zlWCBOX7xe-yv;EE!oeJWts`{>)K}c$+ZZw~HG! z;L?Bx{E#EywEtel?uXe2ALkyuld*$afCBBajh&I<&s(XRu!j|Y3z(msT&#mwypy{1 z7HksMfbC&@D|Y31%<{A7rKeGgPa;{TKD67e=n5EU|Hz7X%a(NNNCDh;pgeM>K60X* z+K%0^7`$oRv#rqtw7;fU4cJtv+fi@aR;vf>5*l|1b(^ZykODyb4_HyWzOGV?+}HX`% z^Lbn^hBO}!iSIOEd-dXnH5wImSfeAhca`w0Me|M#KP_68Pyb~};JXRsO<;V&RtB^=+MNdxk}OMmG*kYEI1C?%Q;q+H@Y8H5^*hZ<>@Z>s?>a zy3ABN&rmu?m(QJ*$(fbPrAy?^h~`ZT=S&HwBZz@UWI~(|m=wt(3uPi}Ac_1biF~S9 z-h@aFtkW_V=?VojxdOW4HKt1OoN5^ZBtWg4NvNFDsGLzPqp6hsH|VN2;4Ow`)v{sJ zl2Oxwem$fBK>j%YvrL*%N0?a)!>}G`pJ7rzZ{7r8nA8IQPwQ7q>Q_z~R8f(CP91yv zO7*BgIT11fgEFFi>5$&_e(kH>gp1WOiRBWpMIsT`g~LlkW2$75n^p6=v@Zh&46l!x z-ar=pmemxCswwLlhGP?w1IIQr2pD$_@*LjqCa!spu$@2I^96jy(F_i+agYFH*y2BV z>`O)PrU314+V^kR^=qUmN%gTVhw_PvXCJ*!RwTQ0+^_I*I?Yc9jVk5E2Ja4ZL26eZ+){MRBJ)yeSynbh~u3TzeMXdO-q!?I~7G z<2H3uR?T=&3hBVOO#{)ge$2YL-?+NpqN3NlWXQ67!l999S2yBT+vQP~?&%2RpX_Rt z;9?f}CUYy9t9l`gQ|$T!S4|U^}H8YuB)a zD_m0-c}LC*b>@iGq)C;;if6{Cr9~Ft?feQRD2}q4c5|AG& zQ<7lQ5t}s{R5$6;&UETrb?t^Abme*>cF3T;5~NiJ#otmjqBQUBCgmD ztlAC22C(HkdfL{9QxAsy{C``Tp*oYRLQ_M$i^R) zS<>HFI>TFU%-rrpE#Hq@1CqZVw}L`_wxt_vfc&2&12g06%G0FR)4q^Lv6`227#L~lH=3O7!O?S$XEBVl6{GRL7ZM*TCHe)AtvgQDH?FAGudCN@5gNB%AN#i)YPTKfv_meyz6h@0RH?=Je?_ijU8!_S zwR}^hbVdH!s>1bU#fwV{R~XWlrX{i`C6Xp3W2dF#C&dzH#M7rmQz6PlV@)K_G2|}O z6!J*oNtME8moV*f0(!B$3f|np?p(ay7!5Zd_lM`96N1y zdT&JH-C@aJ)=XZ_s(wgS{)}$;K3V@GssZdjpoqOk68L&f`I{M~Psc^RpB4UzDe&W} zg-hO=M~xaM!4Q3@_jxd|IxU_J0pT$(&fLLkvzL&@ionI zo-L|Y=H@>l~Te75C%u*|!lP{c=xiTYFz>vKJu{{9(($L?V?46BX8lt%fqX2pz7HJSxu)v;#Nv1HK#jlQluhZC z&zUwb%^GKnYo`rsC-kbu^(v`GwNqvdWW(w)gDR3y<+yPb*`$(URt=c2E+v{4_UK$_ zR?Di9Pre}$Un~+`C>(iRG`dJErbHsHLN>WxC96&Ae6QivVe^s+tEwrxx@r4*s!c7$ zx`t|9PjhUU_vl{o>0j|1TnixX1d>q8Kma=VfzWQ>eQeKdbk~&#*mD`#bsB*z01fzd z8QF0g-SZ&9X5D^Z({5nhrfbEiU4H3V2?bmk;SNO`=aedcAuW>v$+RU_7wlcB90e%BJ6EfZWUQar4Z+|5w8 zm!}!T{D~e$!2HMp$kRC4%Ou6yG{wgx)z=gNVL#A5oB2`j56qt#VxARhmK|pPhW7KL zEr9ti#93cTuywan0Jz!8yV@wYSj#(G$pM@!WgX35!@*qI&QubIy_uA~skEJml%0tT zz}{HK!AR2CM8d;VG}uNs!9_UVm;YJ_PhAXGR|*F)hl6^NXQ2RFDdO0;!Ld<}A(Uck z#n@UQ=TZR|<2)xNmuonOuPalyAx*qAQR+&xWNM^ZQkZf=m`YrjT1=RFbf{WX2$BH6 ze@j409O8e#`7lClYzR5Ab~Of#jobAb0c;0M97N0=M=u;lF>Xc790ySkJjRb)DF=>} z9Y^B2&A_rn&$4yjhQsi>{Sd4>uB1b+DG2*P0`}b~0Ce*2I&t6*i6rHw@61U619t&M z6_vrvH{^d3I?s0RoX71z*{#HkyP*sB!WT~?Ad+8x6t~Xy_CaQDs52{m^&V^xx1*zb z2+JQQuRcn8J^B4{^4jxs)VKRQZRhQbJ-~Ze2k&Pe0(%31#(~TMAQt!`|LfR3c^#Jq zTtx67A20hsCquO%(QX#WUAuVrA*o%uVLtdto zAk}3fI_i&2C1E>7qPK5owK-#vd{~2%#9}_j(s}H=Il~`UOr8(RKN(PYK{EJc#`4{9 z!oz-n7bM<~C|p0xOZ`Zf`D#M^(;>kR#*tId%}VT6HTM0S=zsR4ep?ki+t>JJR_b{# z_NX0u+9Ptip8H|D;NK_;?-50gYXnvcFh&MOOXi@)V;_?Af4}AS?zGPP)7H%EqJ6Pg zUnqC3GL|ZdrOIJ*`6?d|>b}!tGU&i}i4!Zwb=$P@#G?6*ZTmylzI*O{CobLhJO=N34xPC49XoU#*mUk%wygvG z8q_cA*Rj;gAee`ceg+g%_R^eu!8~xg!c_o6?kZir5Z+x-y8+SutWxQ;Vlgm3MfNga zO5rL^>Dsj7b(&HUU9}VlpP^PphfmY2SkSIw=~TmquNgHi=-1BcvFk#0YG!q+f&8iJ zWs@4^6!o$x?ehOa)_Xv;m1X(f*MNyOIp-vDl7s{Z2}KY&h@6v2HaVFb48|rHOwKvy zoO90LR8?J_Yo@zrx~99Qd*)4iGq2vhx-L(@?_2A?&br9LrB#OCfA4+H*?ZimbkrDB z4DcT{g@C`fTPthKq-@f%5`uq#e=n}68&}wCSlVY?-fd9QWl-2{SOn@ZEgrBeAHr7* z5b{BT#G+wR$pERi%kp`XVOpK;(^ndCB}x(C0;ytN$s!)95}uh-ylff2a@ELc-GoMy zH1L8hQX#0vwq(ema+LXMoLxKZ*f7rpBj`YIR=pF0!%^TDfRCXMx^k5C zst;e;i?156uWhp~ePfi}WSHM(P}D{!=yESAVuS_L@S${kIFktc2lzw$AIdO=fFH0Q z&avXJeEvIIMY>u>xnTu=JlfL16Ay~>CdTmy34Ydz{-}}yiOC`4lu(=0Fx!l9+pI_m z%LXm)JCGFrhx`NuU_Nl3Vx<7WfNx_i`&kw;WHT9}nJm#%o@lCQZLB~tma#LH1`F`C zc^pC)k8={v@e(Qz6s(O9Y=tP5Yj0S|0{{toie?;~1=1t4}b66KC|>G5BA4qhoUc{`@`j??khAcckz##HT_=G5w{Fw#J?B1p%-y?8-E${DxXu9o?-*m}cKt{Aj%}014a2va zM(Db0*}84syk*t|5BvDmJ@ck*lZI{M`W^l1U0e;g00^FU&~|V&yJ$z~7c0ueON#mP za=8ogIV;Nf8>;y`n#F4x&leQYP0v}?#0A}uJz~g_UF?=c%Dh(Yf_lM%X4ay9;+pA` zW#jO1O}~Cc&rWfAwGciX(GEovxri87SSVgwIa5!z)Li5R2{|lOxq7A5?15C%9y~Wj zO7IV-BOXmeB8Menf0#A-*rqfPgAB(B+%+10+pmA!By!z|{Ipm2@24dGWl7~f7LDIm zNuAdqzvvPAb+63N+a-R{f&5`y;$Ih)f7mwq#~Fn`OiTT4T>Q^VI{!Flc2gsLRw;h+ zLS#7yIWCtuC{*0ZmcD%T=r0@EKirxB?=#zf+jss+m-$Q*SdvH+M=nVNsdbbYh?V}P zQ}{Q%hS$&Wr7D7XLNYIf^j;&nZxF*a0dgn8=oD}qdF(x<=(mo0va0uVUNdDwCwn47k11t9v_GzzJE@*GsR?*5n${|v(#nJP%N7`VWK#p`+VS2HG+(P^3Uou24?;#YmndR2uQcG2%vZVu3#Jmy& zUE&@v6NDVo9(fcf1yyKAR~sia5;EIu^SbFJeXNQR$LbMw&4|P62}Z*lyLFDywnAxG zr=hLtR&3Fs#^@FRwPVGm6{0@S4z+uq*1JpT+PCXEqV$|ndUt7E8}<+rwyoGSFOc4X z3#>V`qxZ=8I|b-A0l(~nH8p~VMQST#2WWFSdOH}wEPu2C3V%w&q}?0Pgl!#=9|Z9~ z@PCB@JrXasp6|7ZAH$?=2y&jRcN^GT-CAZv4Jf~l01 ziL{l8tfjG>g`vEqp)A2j%GTsD8!zE$D;7)_PH+~<_7*M*cvul4+7v0&7Aw+~DAJQC zh$>#FD^|E8TBI#Pq&`IWRgiE=plEKu<1}B{cyFahH$`7JRi3-LkDHp8o4Ti~hP#V8 z2%ddCG<-eO{5_Qey;V>P@K*~9P!A8($ndi6PRQC$9zBbmzlxl_ikUx+Ts@B1I*Z)6 ziCw)3pE>uRI`f%4_nw0Ie4jnMYu^nP06K8!J7NtULWEBrfPf#!f9N<0p*{$Jf9f)c z357pD=EwMt1@@S(d>1i6M13Ey4E%?OTc1UY|9oFq0t-NULHNwx1TH}qfX3$$>tBNN zuhDB?=q0aj6A!R^UhpN>w+)@}2__jS_$B|7B*8B%_YC8ZrB=_zTI=3uaFzs5k?c!U&2o6mfKu2b8Vcj*X1^#d8 zzFO0)SXC`sRxSn$SXL@nRVjj8U{xt^Ss`~y=Gl}?@}_?7fmO<(ZNQ}?@0=5`OO9MK zO`g|#x?m8!ZWX*id4j8Bq&DBzXc7Tm<0?BA%iMJhZy2{Gy$8 zUvKd4t=Vp&eqZ3j!C;}Cbk$$A*nE7WdHGuOsz!1;LExt*&j{~ zez;@YH^^U9Dg0%g{!ORmw;kF)X;VKaM!sqh_>X<<|2j4Q&wct&yDa`P%lfX}WHJPq z3s{up z^BlhOnz(i!y>uQr;Q-2-ckLVZDD_+9nnlyHIiunky+Rb}YB?h+xx>mi!%ArUKdqHF zuUE97R|;CxFI&QuFB+6XHUXM9hB&`;OuuMYuV7FouTLv?NCVY~ZqYcdbkeYV%D7_2 ztODRaW%^>uqH@}@VutW?p7eU2Ts46&A2KZ*H7go0F6=id0QDLccH{Cp_435Y_fV-WdlxC{fx>%IuNyHfKoM1sF)_d znzN~$C)F*J8-eZH_8l9x?Th%PRZ`ooeeaRO;4y9Bh}sMMKOli4bRIBzPaTI~T_!gz zTEAVhZ`+`Cte~w;TQCd2ivKGp{9E`3M7vg4XwT|34%$;_!J&Dc)-*$Xi}8QV?lb?< z2Zy&K^o9|8^e(_OrDoO+?KL_KK5;rf^D9K+9iik zQX^2MMcHM<*k{I4KsmAYxlgFrsUCno6)$IRC2Mbm!XNZM0Y3_VIg+WYwTTST7=^#3 z$zyZF$7V)SrUufc2C^nNDO22IypaUOT!KRs^|BKQV~Z!cil=*t=lDvz@DqL&B2pV6 zRu>^$8~(61RJbNY^mU+kRe*S@uV?{JJc}ps%u6!HLq5z^+0RMI(^=WeS=G~34Y2RZ zRd;bxbLFafxT$%&tMlAdeLYkHyp)4{R73n!L;Tf31JqNzjrx+)4ztFupRT`uy8Zsy z-euCk&9jTkxc%Gc^{Zg?wEUUh%%S(tp$B?|7=nL*KbZa@9c}e|!WqQ^{xjDp2=W2= z`%a^vW7o+a$GFU1^A;i0#|ryEdsshTv*La6g3pcp!D8U=!JGoV3*-N7z!FyKzYjt8 zXTN>23qm7yfIoWkYXgHk*2eQ)%;twD+wWty?xNOjB37|xo|q9}@BErS?d2oi5wOWG zXuoLm-r2&n-`th&?C1Lk{8OEmKGWB{`Oil>eHVTbw)OL)EsI-XV8x5L)5+(7x}CF?`7!Kc|nLP=}5v1AFA2EkehhP46*v=#(~y^#}!z zIJE9MwrDvtYXTjZHtra{+0v_7)2dnk@~fBgH-%qrsF$y*lz;^+D&#H6Jzr5Rnpem= zFw483BwsT9?>v0YoxOIcyluPCIdiuqB4^uyvq@oY+Bt981T2{Ojw^>$J|I3r)T0oE z5JcJ=dFUZ5#1TXr!FBlISF?=2>9YF6GX1L_qx~|;y?mj&3eBI^oBg(%u#tgWypaC$ zB=g%Avv+S5cFT~jJ0Jh7TYE7bxvCXCu0qzIBLiW`%`4fz%$fc7mg#@4YyYYX`F>d9 zZ@a~Q-t_pxYl&ZXtNeLh=f4k3{ z5&fD6=JgLqbqKLZ(6;paY(zR_NIG;%C4N#hXs#%JitYv9zi`^2f^nBBHZ zd%HodTf^6&2RIEYrgclk)N+TEG6xjWhn2HNRlyg~5)fnu6Oah#mChPfK!iVI3bFmm zanp)n!_q0E7ZV0$pGN@e7LDObM{q^`+Ryv-@(1(_Aliq>f7GI6mRtpzB)k~Mmkyc~ z3;-)Y)W2I4Pm^Cx5Gx1GOZrSu$afg#x8a^Q>0~u%roYw7XgA1%SDJORyYvhA1tK&} z0H^5C&4X9KI0np0yNvU@&Cvbdeo`5@z%cb?FS)eYBE8P|X^mmx3$3U$DbF|&$0!kI zs30v;ocUDAqu3;-hLqk!dEU(|A9Q@(%c$zID1d}wnDAnPP&H#+yJFL{Y1g(z>D;vG z+_LQgZCJO14k*1R4ufC;Kn652WDM-n`@kk35!hw)>@a(FS-tD@&Sm@71-oX@3ax#e z(S=p_n7BFFfUVtlP*{fKb^Bw&7Wuw81XeM&g0)Cg*N6g54DmL6jd%7~?AJ+a4> z9Y@KDw`Y(P=+^RB{Et}x#Zu1B0$smE{=k2LKLq)BLrDuG5MW;#uy2Ty#$>E7X=d<< zU@T6t5OpGndsD^3*pFkJr4n2vGd)Cdyv6bX_5LD-ej-IY(L!&D=U(E^Js)LzK2G(L zOz@P9a+e8mmghMtd$HAA92H%iRDk(FekVs2M~*U=qvGPE;_j^C<)Y-{rsV6XUaZdg?yC?>vT;_p#wDH~x!QV+ZVh$Aw*f16fy?~Jf9-wP=Mp`3P!l_`kGA)T-oV;;0`>v^ zpW=5v#_fEaxDOBCBptv5U?22N{2mChgO5K>`??B3tqr?_3MZY zc)0eR{VHhfXAwK!hHZf$?7#M$`53Tr<2etOw@)9~p~6r5zzJ*QlsR(78a-!^!T%F@ zWz(j6i`>0$+jmGAz;t9kc+MERah$&6&R#M{k8OMRtve5hod=dJNBFiQtJYo9x0{A_ zoBB0tIcIeSBafdeKMmOJFXHvt^0IFKY3Uqepo$mLMvs#Fn<^S>ddC$(zf-E()FI!dvD)+ z>ePSfF?#Jadczxi7d-tgbm}%}4AZ-iiFbkH*IvVC&OOKMu0wj~A+>$SrfJoxc9DOj zYX(<7367wVKcJG+qma?3l+~w_-K&<Oe@EY$_H_U z0RK_bvI&dwL6f3B!+c;qxWJfY>7ZFbze(PJY5pjobb?edN+|2MDC)N;86j2-L4trU zhrk}d-m0J7sGVM~ky5Xo(WITz!WX0o?Rt5wI=RhSnQgi`@CJBir^)kH;~f6E(DFW; zvO#KDKc%G4uDAuC{mL+*L^~o=i5DZm3Kp{W7a#`++l4=7B`A94=||R*GJCm|-8Q8i zgu+hCf?kW_A&c^H%c?ni-72YRozx1U;I>`&nspm!gWR!2?%c8K#3}=0#nE;(AJQCfg-| zWgO~g4hnO!2zNG*cCmcoPKfg$#(R?Ey=@bG>=OO#69OoS!IY#ByQDC?r{NS}ep)0g zBbuHOP0NgN$b3T2igN&E$J4VDXqa*msksTC1^oZf0_1+=zYJypwq~+6rZQiGe}bWm z6;8_BK+X&YF~5wFo-|HZ>a+ADjr1kVaAHIwQL2Rmm-N_&Dig?*4P`6DaU~L6A3gJY zl;SCo<|&ct{wT#wGTB8c(OD|qMJn1^D%4prz)9ANqv*<1c4nwKG87$IN+32GkDNEN$dw0plx2Z>G z2^;4zOIP7*7yetvUMojlGbg<9L$5LP_7-~xbix@vavX+#S#hamsN zV;ZoJZS0~&9{+{wfJJ`a=DB13!TvMfIczc`+TkH&71(|oxC*)pT6-6~j*fjnYyN2D zkMchdo$Uf|d&h6fiH+`s72EK=3tIu^f0eKY*azlg*0^UDn{Sbq|<4#LQP$QjyY^kL&{UppKhtjic-?K~W+ouigQ3sA3hOaqOw@x##Ua}@GnB$NXTrkGY><1w7KOl5~ zj!B(I)*XjdtvjYon+Ek8xHpgm>>1YY8#Qd})@*8`7JxN~URNvIP$^wjdO0tjKc$c` zuNA&ee6mf5*uaNuQ-b$d-am^Ddmrq4#p8Sm4!!k^JmZAjxySE0C(haU_Te~<@{G5N z+R}ZyVKq zH>CIdq~Twd@LzQ*d_S-EzqT#@v|{-m+nm1{CVXl!`p{tbT|e&o3C+uw$jADJ-*rFy z`;iC#zN-1Ve&a45q}v~vPe;D#()i;N;oAYNgI5CYJ7qt#sQs)1_lGf)-wzwkJ`qmV zM2k_L!bqF1)UW!;*HtpR>B4{4NO_-Seh{vE6l8Q7KpDkbmk8*$$~w2oc)Sr|RSS|| zBIZ?yQ8i-nR*2LlLhlme^gePNl5!e;95^BqG^r9fr4_qumby>KLJI|P1=Z?LXE(157LqPlYVbkxzr*6Y1?;@u^#LRt)UHm$F z;Zykh$Kbhlep9!eqi5WqBX-{*tNW1AwMT2;p|ot;G;R>@( z)bhGiaypf>yHv73-Re2Lny4uZ=@mgz02(kT95;V4LwGfTuNXEf;n#ffdJJ<1%=3DT zvpWs4x(u_sO`rFe<#ih8wBfQkjC0$Ka(c`Q1_`CZ*5xC_@?k>RpjAnmaRI1VFQ-mD zwMI1=)S#K#qLS{JckkouyNod3o= zrN%P3hLD`A8vyX<30eCH61^S}gC*!uvQ9~=K6Ql524ZeA;duwXU;tk-POO|I*DP4q zEfDLMiElycHf`8O@H(YqiPQ|H!1smDUHfjJ1n~dDdHjGq2>b^D|3RCqzI8^=GPPr# z(h6FzZ(E|Zq4!Cdos0CgB}V%at7C!Dw!-NG7l0MC$nFFq02dguYnrruJ40!lwX1^< z1C}(5uNtM*Vf=@W+G<(YLoR(~8<%L$2yiqFcd>}zZ|q_xdE&f@2|QAwFZo%3-LpVy zN(eP2%poO$mJ&rvjiIK+qRh{J!pMzh_bxyVm=P#NJeo{)I&xP!gk>#Mwa8RPt6dkAv@W5s$aM_B^9C>#qd9VNnEObCm!Pf`Caw|U>`7u>Bwn> zKe2K8$Yl)clXmVs0}(#9U3=~`f9Z>I9&5R9?uTC2!ul`($=`)Sbks{2+B*(w`v4yRoem$kggMK75W33@>!;8S5O#O?#(VDCYxc%{8v7YO za~j=e4Zu(Gx!d%O_X0WvfIkA}%6%4G0NwIt4IeNDPaVf@+-E;{FTM9zx_6!f_+#V1 z0Q=|E;WPW83;W?q$OXu~NBE8-%l0Gdu0zNI2p#+Q_C52KU6ZCmvz8OfRLtr6&zIHG*Y#2kEYo)_lh)0nmo0)fY<%DOMSK+!cJAhX z%nsYO3*RC~98=>~nL#^@kVE^3X&t`-b^l^XJ8wkJ77;WTkg%3f&_xigBC?*vS&g)+ zGZttzdDLMe)M5Luje@Mjh+Y&)-c>0Lh9X1Z$oJEf|8r{l$2E;VuIk@4D}3FKyKhu| z-=gr{px)2>6n;CZ@%^a9d_1z33n_s5e{UQ9WybX9okoA0A^+Pp`G+&RKQ5bJ)Jt!d z3!T=;u4QXh*dbeKn%{L9{AK`mTWhkCh2JUTv_&aiHd6OXOnN+pzN_#WA#1#l5b0MT z)QjM%5xu$xrcDp+IwiOR(*C_tfh{6#4MOx9#JUzCz7?Rf2{XII7~K+#L2;ixQLiBx zzcHoYS>4!Oo7_Wc;RZQ($ueWvJbMP0F|M69sg*sYox5mIzHi-dX4iU3X+3x7xZreM zxc1%n4&Mb&+=NX4|L>z_?;>aJ!e`%y&wUlS{B_jI*OANj!HYM3vzI;9cJDEx_mt6l&gws7_8r)FY>}GQ@b%MXFDHy&j2e{o>g2bpWnpSoNe6Xk z=Jeq5`;Ce~kOK4?q3SaQ{^tV!J8+pm{yvL>e#^pcv%CiFlHgoEEd^&8E3;jk4-AG#YsOb=B~)oWsn$(sGD~eTPkUqZ zv`Q=SrDj64?lUlg8r-wjxWroXXZ57CmsSZ$O3r~|c7CE{pNH0-4~gC)ByVAxAQ4uo zvVWyX;v0+f7R#I-t3vRdG5m{R{$Te}^U5hg?Hsvb-llQUu6dn~Vt>oFV~yMj+P3dH zVDulc1|j~3-yO6w3=16;b-*3N+5)UP^kDp7c0k+5K`sCmfbyTwK2LAOvVc{`?hWTY zY&P)>72w}8L3%r5!*3I3R|9F$0I_n=?)9kSn?a}QR=c8hV$m@DWdkGa8PzGu#U|Q= z5bI5d<5~01@O&=sq=Zn?!)a-e4(Uc|-DN}1!N!6|5#3LI-y7h82Vin^DBv@cU8fFm2kk>N`=h%N8WQu1Rf z`LGo|S+XunStq(2i>5@kSEf_c>Glc?sv@1HM6*}0w^gvWlcVupVLQk>GG$#k@*Z3{ zFIPD)cLg6W)B=J7WTL#)pZnSmKFK=Cow&)|xk%c+P20UoUcXD)yp7tt3R*w+SwtJ) zbEi(6rjDIP_Su66eE9D<4uOtc(WcE9|4+Q8fd5$Eept~q&iwBEScivGzd5Xj^JUNy z+O8*fF+Odd5Wiv?yowJv<;2|k zB%QEh4jmHL@lTdbVs;77HfUkT)F&raF$?MeEmE9JQ423b${GZJTefIG)Iu9 zy5gM_^(F)HYBhl(MWk328FrMK2|~^a#P@O}$0MX(v5?u+hrgXw|L1+pKP($v)@UD< zX&;nJ?^htFwa8EV#6Pu)-@R5`eJ1{vi+tCu^M?t&-;Wx6Y*hNDL+SsVkpAKl$U$j41E?N#LFhRC5rN3>H`}K(S&xY(KvyCE6kqQr;N=Ma1HDtsQ*JY?* zC$2vv>NzRxGb-uWFJ{*vKhPH!j&UE|40g zEUU-NDuzv83>ug985DQvJ#W*>YEn;cR7-2pNN(0lY1B-9tMRN!E2Z5ayVtA$;(Q4B z`z(w4EDHzl#l7bFK>kLZlzPplm6{1JwGvy zl6Ycd4`G6rsI|AC9Z!%F^pKVyfhlGY&i9yW6K!r z3l8YmGT{Fv7hdUJ;SVF8vT2znHBVck*92y0bu;w(Nr(Co`!|F1x)DxY54EDjD!-5R zqRXwYz%e+%*^$4QYx69~HYLO^HI$MTZkHBmkJWavV(D3NjO=)3b|NDuiJ1%G{WE6X zGv;UP1M^``WBs)X)7gdTtU}ZU7?=g{ySgbk+A4wz@GU?FvjCc-Yf+# zro0zJ!Hc2fL09%*s<<*0(XgK`%cRQF?G+vD)M&P9G&^~kog$Fmj;vr~EeEnA%TaCR z81~>1a$E*#0dCF;9YbTaGkm2PM)*JF4wrP2ad^oCnU5B=qaJ=1mAg1=sF^GACr5IN!?%pSc7Qb|DNGnfd7eciK@tR3*@j^_V34RhoZziB@tI~g!NE}Er2}JMdBVKuT;dl)bag7&J&M)CL}y3 zAG=RHa_bjx859VZ6px>J96uowH6jz(C*jp9!>3eZMwO!Hap@b@rRz2?c35wA9UEqC%Gb0?)^tjDjb9yFz1hRR*(bj} zrnMb0I?kQ@QL1~71K!U(M?qJ-$s51v+n@#D|NF3&j}dF{Ll^IZ=kEe$Z+MgNwrj8P z_q^#(zO(PWXTSx{IfECDL+9>epfk786Z9hOAgquz>~MNF7@ey$^hnIIebW-9agO|E zicmd)f7xeL)Q!t;*Uf6uNUc|W_D1FDTlJJCos2eIR;zwym+^D(0#Lt2QIBapd|7Xt z24Vb5?Ra25Jb+&5CB4>v`cgNpQs)Way;3LUg;rFVMtG@uSg~qIsakZgas&wA4=Pd$ zEm8?7(T*t84Jk2+$j14HNzr^ni6A#23vV%^wyL(`Cb-59fBgi+m!FK8nb^|@8ldZy*N20sg;q=Zl*+E0&iK(QZ7 z&5EaHC(v?}=$~sl!2D#^7t|M~vWn8!#pxVaf%Zii91uJdq%l$Q^Hr4gKNjF(uk1`w z0y)`yu>dp+0RB@gzyj2zf&W%GIh6mF ziVj32wymm@y#|*CoL2?fdoY#U8H(-VSYrHJcN`^#`v9*?=*~rl;vK*QWSb#HT0dC5k?kc`M^1eKoFh9k3 zU*mj#Zbxj&V*2Zi)aKo%16z?Jn|?$4Ug+p9_=nqObZ;@*!NEa*{awx=);0<~74P;r z1%Q7_{tU+%_!i6*_um zko;Ky7rt{hU%Gb&%wZE=(BZt%o8SUiVISiB`)Ei6(8wQB0CbH6_^)8^{20CYA#w|y z=g7Y}d>?{tE1)+EAZPJFr#!>&VKY*_{wAXo;!O^A3d`l!X^Uq4~3$@ zKeg^YC-+`cM{j6jmz3diGJ0R^jMxqQ-@(-%;NBeSzd6uB$AN9BlACLnPBcs>nz>TT>)P9Ep`enmGa;Zpe1J z!rydS{mVSf&A)SRZWF3_9sLA<3ggnygFSeVj`E{*r zg*6f+j3lTS(h!6vjhw%<`ArAss~pD;BmH3+)`*nL*dyM&Y~+$$?1XUegmCEi!_aZj zkjck^Q*t3wa#15v(LG`zt%9zN59mz@y%k}#Jz#eTI*!Ts&Z&e?$_Gu#g^bGt4@(74 zD#y-frOfDMY+6?wGV8bOYu0sMtZA2T>A&1FsW~vKKd^XvL})%DwH}gMPbi&dbhN5- z$mls?_g}h>-g{5q2QJ(Npv(u|2F?Qi--XQG22Q^ZnR^$EzIf+14X%NDhv(Rx&&(}v z=FWHa)_?AbH+||c20HheIP;u1aUVT$9p2{-VA^%;K}V}F+PB$V>vZsira9aC3Bv0E zladbo=WV(<4Qk2N%88&_)u;6u&zg19dyJp=n-z3f6?9tWx0&U(nCG@x<~5t=zBS2e zFwSf+&8Rm?uf{#AHb{DnOMImZ1_2pCRH;_v3%)OuYetu=N0q3A7b*wms{~~#@zRt% z5)_>BOoN|lxw;9M`#!Sv7Q?%X;yp$1-Xa7aVZ65>-bVl*ButK!pyg=#=4%F)>qNaV zdfJ809Uv4A5lSXVFJ_4~^Q5{Z+lF-tI@I__bzCNNO!KXrHxp*vR0`K!Rb_(zpwa6rYGs7kUNVJ*r) z!+uylKm5plkSkRMgt-7}0i@4`AO|Zss)Y>j-|qkAznP)9DNe#v?~$?2BV%nTV7{@I z9LP*t-a=QNV5mSgRj|h^GHjGN_NrXEnhQhAjjiRu)^ca7gAuqg)LobwP7F;pU7bx= zWzv=DG(~>!uLAt1ku>ee$`q24owbs+l^ot&8bq*^CRs__5oM@0GIWYGlP1k!$er}=iV=! ze=wnAAe@G_Iel0oDR|i94#ER`dEyQ~?c+fHBd;k;m=BsN5?z*&RzK}UGNsZ z+y%x2WX6U9f1U#AJBMu_V14_qLso7=SJ8&gVJjH_VFln{hAsm0?_)RKp~ZhR9l&OR z!P~$)(0ynB<-5SoTg>pE`+P{rYwpf}Kz6 zk~MKc8$Plh+_&vJu-0VI%nR$1BeSeyLh+tW(U4K-OJ!1wumTqmr3;AJKNPeTL?|Exp$Q`X zYSLkvGF%ZPLKF{GS@ zEEYZb^NQwghOArNMV@IRP0r|%(Thrvl?-Gk^ub85R;QbOh5>R>sq@1H`&WbNr!O8K zri&~@KO6`WuAn1FbZqoXPLzjOY4H$R9)d`x< z=>$wFh7F1a&B??p%f>7`j+m2(n0*v7^(bIkDtJ~tepWtpLN<9wDr!K|zgyI+Rlubg zacUKC8kF*$RtcI=^dFP)8I|;zRScg~jE1aXT=vP7V)BAo&bof_I_}jb?$x$o^}cc4 zfl2+oVcn5Q!?{)awRQJ3x#!ZR=Y-sKMD9Gc>pW-lUptQ6c#MItBctyErtgDhZUUyx zeMYYXr*49#??PtpL+5UT=dJ^1uKcDheJ8JdXKwsvG1mYKxb}tTMXY%s)<5>bdlDQ0 z1l9niaO#Tc*bQ}u1LwhQ$KF+X>!N+*jBV|RRYjLUexpWetx{s0auTElur_L^HtT1! z;4)i{b2=>xy75K)sp!u^t(JN9Mj3SmX?2DvwFb#`xa3#b3FQW{uZ$C3=*1SPhm~nW zff1BxL=>upWXgLdN;!s#*yWo9r|5cc5o5nc)^0*(p5kBumL9?uUSbxWqUIhC&D{h| zJp{~M5tG1&g5dK)tq(h0{P7frE$Zdb(_((!|X;o z$g%r&S-lW60=)tI0DK7k_Z&yi0i4SY^smZ`yS(-{) zm`GWfNf9liZSYbQk`&EWnr<)2a*%Rl%einBU7eM@+?9Adbq{}aFJJA@Af3b@)eIlS zA~(}|_n;xKidEm99rr%8=O=S$+i`fybzqCxw?=~yrE{CzyUFZcqoecS0Q;D*)}T8a zv`ySTcksw{H z`Mmu3k2h%l2lPY`WC7?7?~nYyi&+Qq!pmqR||2AgtW7r0Y{g_=yA7BOjBzhPA2ptOMGY`7+ znFsiP@>>P|-+L`!RiHE0=!wH1!2gUsdcqhzpbj3`^}{RI>~VBD0Bih)Irfe@1u4L# z?I6Ver-Ytki?&0P=0l^VJ-yluEKy>)5g1aZE#d7KN5!2Iv{hA0w1f? z|7qOn-xdf*MdBM7dh>}6Jwf=jVvD6xv432X`t1-UQy1YONT;*PQliv)#)F@C8q6h% z72=V2Wh6}>`DK^Q5BqlivTFJLIAPWg8S;9tk*v8_qH)(O_iyXE|Fvwi8z6?Oq#|b} zBj)827L+n)6|-iPQ)g8ZhovKW#Qpmo`FDwXH3_nt1sNTp?0!j?30a>BDc-n*-?y(YSfZ)FtVhInB~F{i+?4x?}T3bR4IC9i&}X<{g)2?N=6^*My#Hav!+BnQhM* zrT2o`f9}|S#_2nE8n|#7xNsl3@*ch5jU9Up9eNKP^F~hn$1j4WZo=kn!xyhZ=g$KH z_%oM2Q$T(YI`hJ7^3rD-orUB*^`&_@x|{4h2|Dvc)6Xl<8OT77xI<{;P42*f>(IVS zKRSVt)xN@LUZ6IhM<4N(y~YJinrTg1DGlmRs}$lYvsh5Z)?;+6_!3QDar^1|K*??-5 z=mw3rF1_ai#>EpB71Q`C$ORT{>R0WX)~Rh<^iHhr^q$kezSA&PI^JUUZ!-G90*+k9 zP`QmCyN^NqkF^ioa_j^CucI@FdoY0!%u||Ish#j$o7}!N@1C`Q-gW<>O|S6-pZOin z@h#8c9lzm4*Uk}6{it{2fPa1QQ}^Phu7%HBic*}4(j58q9Ckq_voMPVD#~`mzz)nW z%W^8u=9Xl#O0(D{*{A^iWjS2T0{%J?`0M(=2VL3SLB)-x>|(Eor2s6F0`Q+fP@q}L z+ndY&)fOP7ZG%Yh@&3Z7xS4D$(s!(BR(zrM(A7 z&C^jGJbsj(GhAH z{ub^2(OZEXyPmV#Zu7ftkaKk(x(#l*3Qd}57>kI^u8@h=MJ@-e-{|N^s(*G zv*pl(?)CFs{@i;83+4gl{ObVo=RWfg?E~r$T;QL33cLJ;&De#!7~qd>C7@Y=?;Lt| z6r1YHZ`$xh=qvm~``BhNdPg7ty(xUO0=@NB=btJbwyUhxKdT>U)oc3#W-Aek+Vqm&qfiF|0Gv31<{c z0WyLQ+?h{KGar}}_YR})Z2E7ld#(vxXXdSkMsFeh+|;Ss(t5e1RJkl`Xe&Fh=L0u2N5M(aX z`VTX_-%oIk%Q@+0DuH51J01D%waHyMZYWZ8C<^&mlg@w5*#E~Qx`iceFtF#Li*JtvEOy;|MwD*X{D(^~wJ0g2jhYr7jz!+f=wsU+n7)+W#JQ|EE6A*QGA`Dq=+sG#X^B+hmy| z%Kmez@e`8qvy#u&<+68G@|L8MXC6J7k&IoGf3~ccHZPyHrj|Rak}@XuWLP?CL?#l9 zphMWJOVp=Dz@=Tttxw!<=otwZLJpvxSB)bdNAL8 zA9r#ykievJC9vDjhs0So;nSlIuD<^4Pm_a!zwjQv;Z5HAPDAbh>mAB}^uFP_|LlqH?7q*` zuJ`1D=h&BV*zkLY&2T*S7)7UaIrV}LVC8ggIkc_VzMUu5PUBw>o4)AKe%`E+QK$S2 zT;Qc*Y?*x2Yt6(rhUu;N{CEbq9}$YXkhPa6!CQ>zCSv9M(867e;3|f95;k=fHFp+9 zjR2AYo~XHxph@)Su{nenj^niwr$>`b@Lla_aAZw z4%q01_zt6Em)X7nF2EhxcNyPv9>b=`?mD5yhgK4t2TtLC1v<-qc$3?|!t7b)^lh-- zE>r8!F=ULMHO9cY;|Rci-EVv~V0=Uo8)Ss(=_QPIIlfodUVXD;{U6etL9DhC!| zX(&xFlp`9+TN}y~4dgyQ8_SU`6{sX-roEaI4XyLIv9)|0HN2cqc{*yk^DThOQU?p* z^Izo)|EV@A6q1rHQGrNQArO=;Efq{mqzw%p8yGw?Hk33qk+igs##_mdNHX?zvNWnR z-9eholI3z0JY3~Gy%can!HBfIkbym~xjpxpBaew==b=NF zp>@ZeZF(<;cl1Cne_MaszUxabATR=WagRR&Y>zbn9)XSk=6*gY^riVHXy17Zt^T>9 zv-qEr@-w5EB1^e>S5AuRp{$bGt@CyhYAb0rE z@E@Jn#V^`?2wlGqLc@5pBed@lda9Fur59fL6uI#!W(&O$7Po`x>%>D4z#lB&>)5?d z(R=ShH$Q}J-3PB@H<#gWfMfwH4Sfhei$C|ii+8*Q5Im!~fZHrU9#9WDLpfrhTJ^9b6qMz1kJ^@&)DM1%<*DmC`MZ z$|Jp+6I|V?KB{y5H+j2n)1tM@fuf(}qs9BEe^XumkdRo>w1u1R&r_^K1q zs|!lqIsWyIN!EgH$drM{TRCFBpk6Yfl^~!UhNwh7()JNiW+BqHh&WZ~F;hy~`XMr% z!~NA*^!s*?u1FH|0TLvE402^}(u{{aq|0o zYWP(@eej9Qc+A5OHERF0Zu`TL`U zdc>pOHd+6!6aVhD>Mxp2-WM8-vmW)EDdjyt3dE5OAN>`cPA5^LL|P>uQ7c09UI`Mq zrQ8?Pq zKsqoc4@ThICg9%sz^zx<7c5{!?b)tQ_JVvOSiqcI#++={yu$NU)zWRf*9XRLkIY+- zEZYw(J3u=YXpfE)a@PsDKnC%4PJ@b@UwMHhkue>cn*rgjvA3*Vw7o z=$ZG#srSS=z&~IbmEYtAZ-QSDg2;agi~wEz(5(S%G7(zgL`$8YbD|0g)0;eh;a@ ze_t^Q@ZU$=+Cz*076ANr5w+q9Teyf>0RLScVu^scuZV@8usK*jkg!G4L;H+JoI*Lz zmujJn`ib49&quAwXGqm+_N{0m2>xF1wnNu01MMuj&+6Ufj_x>(e4)Xq=M1cS++pDV ziO2X|Fl7Dn;EBNhRaWnYZmn4y<@}uX%SiBtEN33VxBsEz4x0L4Kx3d5%}{b8cZiqoBZ{IFAVt{*U~3F39DU z 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

    +3&FEH;4k<=XM35>VEq36~24 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 z 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�vqrCkrI!k>Y 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�P_KWW|5DYIo+Ma%lR6 zeXWiTJJw&ymARiJ)s4$ly<1)V_Kt0?iZCZ~dq9inKUSd{srLFuoN{cR=V%3G>4jx) zCWOFuD1}Z{3sLG&f;kjRUzm9gcxN*TLneS$Uxr;&HkYmT3k{q$0y@H6D#8urpd@?J z1pDc)R2nG3;@sLqKK*DzCNO3Fw@V|;`0dA*S4z5&!jFzaZoRC27;Qe{H8Y-G0G-c> zLsS_+@S88S7eJLfyZgi9bavlks~{j>OJC@9o@lqT@?}ynEMDZBk3B_I@pMiz3r^ko z=gnEZu`@Ybm&+{wS71Ftw%lwAeeiXAc>dvc3TEt?q#;VhM!SCi+`;4H_j}J5 zp0-C+TVs4FYaK4vw{JP$`Dr@c;_WH4o%6Vz^7(iLTYJ*~C^`$SD4#wI@6stC2m(rn zG)PGBwfy9grP4N6O=bcb|HcX#&!%RcY_{RVU9oZod{_nA2}R!8x)N552$ zK5}f!@p$BfEwv@R8p&P3vjCGue36&s^{>$M90dmv5psnwqp0Y)23a79!4H@eJ}yKJs6fqSJXbgjVM-`7f3ZGiXmiZv zTCDU28=g1_i3uo=ll^9qXcj)CJV{3{-^B;qQdH`G-#fj(Bk^k37Zl77p4n@;H3d6H zKNxYyu2vM&7dPgk$Gs(ewOOmt{Gr-~oRoL+me<1Le zF5_g`gIon+_xiiuWNM#5BlItjIO@LWuISg;Z*I84##k#oh3LK{vO3-)h*EB!If z)f?dtwtN=b`sMURUZk1q#?6!k z2Xc3so+416ALxVat$*m9{;6o0YCXA|%)736Ao$1$?>_lfw0HGuwHe+VFamWo-Bo+R z>37x@VSQKO>%YH**;v`x!~#@?=lwBXS(XcFlwlN$2F1h@^TNFNn$#-BMe6u)aA_OXcN-A^#IckmoF77?>0p^~t2Z_MH z0G0(N{&sH0km-ZFM2LU^hJf+Dp}U&}wgs4sxfWW@cs(HU&qO@#F_XRoDnT0ea6c_{ zz8ZhM`I8XX4|JVxi#;2BJh;occUFO(%Pg%sSv~J!>`_;lcLc(2j~&ZRM!8Mid_nw` z7c;j+O7ynPBzwx^A2acIe3hO{VT@7`-u?ZKatk6IYg^S%p|tV>Bu_({WBxu=|D&{B z1P}gSKFb`Po;~*AEJ^)8>UO3hd2QDv3rg!Jyi>*$CZ*9|~WjUBkV+5$xXf0KnQ&|Un#LQBYbCFy!N_h!GL zNbG&OD`%2tFo*dM0m0v&cWm@4OAIT@KLrnU=9Zb|*S@Fj?yN2DEPeeAy3yIdRB{#j z;A|^*A=rpzixd%-t#ra3gTf%D*;pHpysEJD5y^a}0-ir_`0P+bU?dWx{K;mx5|owZ zax4i{8(ic(3i8o$ObUwa|CRBnW>}-z)*<=;QP5{GDZL()gKdyBZ{~kV4TX>z-E_E~ zEe`(I&6xW;BrhhyW1r<&KzgghRsV0^No_kX2?MxHn@qAkBj#`WME*Z(-E}`US03&hL(QT{6rt$CdW95ACC2qxCs#@Nqw-%i4lK) zD~MnX5@g^se8k8I0?gN#u#of(Eji6gWo(UQgovrIr@5DoZb6ki(54cJ^}4AAO;bA# zzDBZO+$LX~=k{ZBRJO(g-8a=7twVp{AGz<%*pG7$%Lf{FME{8G5BTi?HE*1Z1YE*6Qq_DMliz} z{sIkGIomh@tyZu_D* z_50pTl&^@)d^@-_y99^8!16JTBjk5;&DWTc%I?7z7y)4Cl-F+5SH_9vhLZI_UV_6M9l?3|na>K;bmh>NjdD?sOO(m1VVPQh-#7!d;} z_-nuQg3m1oXdhF6?{QJW*Vn(3dQ7R)Jwt{NYOG-F!D#Mk5IalrwXzIcEhYfS1Ohnd zszHQ70Q8y~MzA2yQBol~$$toXoRg#D+=O&NbHX7D>|tf`Y6Z3=#guUA+XdVU<_oEP ztUCQ1FL>LpV=7h`eA3oXr4_pd>$D#@;<}1|nIe~zH^Vv05>9tM-EdTg<&&_(+`(Yr z5DvA;Dd;4Np0srwb9Pwzv|e3PpPQ54+|+tnioI+xc$cj8Dj9a7B@y{*rnn{>t4dF9 zNAN|sq{UE7Wwsh7aFIW1(BbjPMlcK;cz%J727sV65Zm#^_iNH1Kwv7rCbbLadk=i0upKYN_ldtso*d#;z4Ty3Hd zg#hZpeLCA)zHXF#x2Iwoht=lh*Z)++EB+zzVd`)tcESqk&UD?&YkQup1Sa!7X4*5U zG)HdI9DB9CtA=IAa%X$h{Q6fYzVOpG_m?o^xOl=uR<9mv+bA6U%LbtC&yR>>qQ>WZZuGXv%d7B?t|4I|o zFCzHc-{0>MGF}KgAsw<%*t>LcEZ4=GPg)03b6!rO`eYk_zPJ~01-$L=oGX=~w$Oep zL(OyQk*2XskHIVv-n9K=13M?KiYL>oCxXwmEUk`>$zeBN7ykK|=)S1<=_JEj z!I7{F_HxY~Tnv4SPcQ!|eGM+g3i+I)cW}0uCcbC%5E_p^UyBtb;A&_l zFQ=#^2h}=>#_$IMWGG-=XaTqU^^n`jZD7JzcUGieXPa2vOT`s2s! zn^CBzo*vO2R8rbJ&`(Rli?S#+YF>J~{`-x$SWC9&hZ;>{MP`9R2AX_bhq=G=Y#aEv zdT;^?D#9GD*)0pWU%92E-fnScAnQMrops?UF~P-F$I@7v;}BQRI(w@EN_JDx8B+W@ z46$f&z2FgDYaZR>`d8AymI3VvXPkkC_KM2NU^E~gVC73i$i@0`Id0Rj)EdSH5Z37< zx`2ns#*tcB7a+Gm@1uu$N&nTLOB22^MeqDFUXmjYZ8XA=0QMz<{)N(6b zf82%QZVFcUzL(-}cr2_oP+IJ|_&fLeQ#7uV>wXsY?^ThEs{^Jzt>sT#CllTepT@aj zx!_|JKF~IgDQZ>DfbE^yc_;n5qsv_gq~V15M+)SR^Xz|xpVM*i>;3O7k>B3Ta*Xrx zO7ZAutlSS(FI_^K&}CS$0D_Pai6`m89nOVhI7H(?@smUrV91TIMxeX_+!7vQOCB@-GE`_0K)qeS|k_<2paYsg6n`V>OmO(5^xa?je*hePKytQjyM z|BXLhLS@0^?Ts$R>MpqGovHOx1$w4~a^n)&`Ss2!VHQ+4!|mRCuQ&4_D0Jdn73yMg z#6?il?#ccfy?yxKYzG+b`88SxhP@0bk3AoR!()kE|v8W6N1mby)5jdCroo*Sf6E3=~!NG?CiFsv1lf(i2gsu$PxmQFL`9MkZbV z;Mc?GlLq92Z$HVAT2ACh`9!7Prpp)13_aHQJ@P_L6@y4N_IYQ{J$aG7Nw?kMH)<2V zpO4M@^$0AWPY0sf=)Pt{|M~p$Sxw&#*2W({(meW&#Kq2b@oh59_knfzW6%%8gKgs} zt+I(^nw<6Zobz)WT5}{f+Ng-@Dc@8(b?h$cEocioaRd4W>As@|SKER%zSIP{Na)~P z3|MBn(g010JXY+~HMH>9Wj3sbqBkzc*xd9+1@F!8?xUQU)n6DrDdQQ*h#TH>e{aG) zC_IRIU5~98ES=B=8=U_U8rCwE3$g%_VrUTio0HlX0f?etzyb~k0>tqUTO`T!kIu9U(SIn9kO%6{*c2>e zRSifOUi1K&WO(E75}!XZoR5Rf47|idi?tSRgAA4@JTbXM$)bVJzNre`o^}lo{&?uDo3E-r!xG*+sXQ12Kd<@w;5Ep^`p>M2X^(P9r4zYM}d zl?vzt^_rM!MzR}(2YlL@j5Lxf8HlpOA{0{%$qP!`j6yDsIWyR+n6#az=|uY?eO+tH?grQ~H+8 z-|AMLl^LLZ`{-Y7#qqlP>#Cequblm~f!^_z`utW2(x>&upUTFf>=)H%l;+|M>6iIN zqp2|kz?Ju&v)W#Vi|~1cmXGsK{hPUTMyPN-d~7+T!*k@H$y(D+pSd#B|7#y{tDYEN z{OY39)Z3RXVT?OO9R-&`rh-h z7_z=flUJ$yxIB;>!}iI`qd+{zTah1U==yY(30wa$aYu0c#4tOY3FL zfM>Iifee%lh9wMREm3Wwj~fK>5aEZ4{XP+B2Kt99vgu@KZQl1ih{T6(sL%`Ks}R6E zCc?#fHzeN-$9_p4JHs_e?L+*Q2-llJ;_+G*%$EkXKjfEsi1|*F)RvUH$tKm^wGP81 zDZGMcUDO%TaC5AcMEtz#&?Rtk99gMK<#9(g1j9%-bcu5p)5PH8>?PT#fM13(3~(2s zIGCC5LnXR`QUi$tV8%f%3?jIR5JE)>d@y3d2S~0DK&6nt9z?^Ln5e3XyxU}!9R++h z%kdHqicJK9Hp!Md%A&|i{DyU+R{sbvFgiHADMdxEf!&r;^u5|A$5 z7dWG@k*35TXptn!1ttvb?JFK<4s0YCDS=$g{2|e5g(+NTCj&JHIKi=A(L@Pg^M)oHOiSQlH>uXJ*X>qFLZ2%9?qk1ug9iGaMnMJjL>pY@! zogZj-iud@fy8bWr+pTZumjzdN4ZjZXJ?(rr_fGQT zQ^yc~$u2^n`R0Yr8*E~-V&T_Pg6MZYb`&qjU>7qsL*qF%esouuaOcbE`%W~OKd+Hm z##Ea?(~lpXXszhQZ)aG#M0&izB4q4c=DUw4-(XrV-BUs@I>|R6i-ed6ugt28b?Yju z4`p%aO-F;-ZVU5Rl1POdDr8;$#lTr#zMm4xTJk~*&m zD}Q&+?LWA-K$FPnpX-u1^foV%?Vp(%I2ZlkBf;eoJ*MJp^iLB8pHeYPl*bOOl}hrz z&&0f%fr}2ON-e(jR^eVFid7blV)%mOt-Q!i3gz?8<&HqGgnsXK|IICWc;MQ7vIF?aq6>20p5d_T~K8-iBQS8qi!5>qh+lg%M*J;@nOjhWq7_NJ4b}r9W5DnRf$|5 zbr?cp0*Zoc+KYXNtqcgC{7A*_m-@~H{o{$&&^J5x3AWPI*o@%;j#tG?=jtDuSc^XE zgdli1t3l{f1hMshEK5Q0#F5Gi&YXz@lj<2Ao-U6QfR2yjp|+#1!y&l#0HYzzOn2_JS}(3qdaCbV_!ZQqOoFa82+U(6(Ncznx{Dy-s+{)3FI?{{gUa59Q?T# z##XaQC`U%Z&3;WQ`Fh^?uP#Eo;C~A_+=?ylXjukJUm9Y_>;-n%!ko!v=;bO5;#Gdg zsezr`&_rePslXkkC>!8wL-MFJgrK2?8LFb3Rto6h2 z+10B-l+rmo9fuj6iuw8ShB>a=I`%M6D+ltAB6f;GlvZLN#{4~*uM4H4)>r;M_Bz@^ z#WIm+-+ic!vVco5T=5R)P5FCvf1u}EXST`3+~NaNH2MB9T((Mz?yL6$!F^RnoB8Uq z8#xU1T|po3gTw7aq#6ve34~>)jzSf{)#w|R@v0hw9Jqc;y-q?w06&8H?zEl- z5u;=Q9$$h{zO9#Ncy*(TDRwX+x>HObWG_8_1pD)bBlJ4Sr`?h&cI7Z>k)7^O?qAU? z=YN%M2f8gDY*LSw+k1U(4d4BK;>JAeWn=ASMenYN{kq-w*iUT83#}v88HBp=UEd2@ z0Ecr)jjBcMP%sGVJ&5sBTF_g=_SDP+A3nxLNgEs zCFSpkTWa~Xqz5~rSTR;xT~!vdlcZ+}6p($kW4+@A9hv)J%8>eOmZ$ z9|6(YoyaFLsJqu%s~0C?I^3tqDD`%0Evl%&Vh!@k&Chfbe4Lk_%ZMCjOp)7wyg-%_ z0kiy!G>-U&dLy};YS+{oYJ;&Du3Kb&WG#uiCz9KbReMmDJ?KM-4$oOR4I*2zG^c*3 zBJB>glSA$-G*wRa1sbVWGpiSb&K03w^~N6@1!F+h5_kxa6}O7(b597vFe1MzLt6_3 z*MR)U23BI=kd<$mu3_2zfiULtATl57#84l{F`069UK@ zpxE1~7N=c>1U8|rufm!@?N~G#JNu0H{!GG97~-4#|Br~7yCLI1H)&3Ucd^bDz;j16 z>7t_t-2Eh1Oqum-^d(o(c07fp(mU^MeP=uShak+YLvglA-bfHrb|wS@Lao8|AAHVX8q)oQQvH@A>{| z7m@^oFp~bpez1L`YG7|@plpvpvr!QIAlvqlhdSMaa}-R1LMkz6!Zv7}-1(tVI$@q= z408iL(3WKHHQJ&umGa2g%-8obr-a~x15&s#jxS)fH5;mOTp zfHg-`-lAPb+q>3X>tzuPI2xdBLK73vk%EKzSe7Xo$+$!yfLS`*@}D$X68LB6H&r8A z9Ftd+Fskn2F@b~k_@Q_|;|pi4S!NGig^ZX~ld#RZ0M|{XLhjF8cr{_ckA=u>oNV;y z34!2!@Spi-&k6ZQvGLd@X|%Qv`;U91jb2{99ni_$3HeoV@iXM2;*G{dDzjLJ@Y!V_ zDit;L;>DEsFs4ZLI)|Bav{{u`qVZ&WHmSMVYyuT)*9-2NQx{d@pd=@~QH7qF1v@mt z`@U_xequ@3-2>rqn}5`(ya|rF44j}#O=>%bF<&;s$;)AoG86AFDSgeU;sWyHN%#Ac z8r_p`D6LjK7nI%A_l@z$hS-}$o+~|O7Qn}$mtbEsuw}#VNps`p2-u4glL!eQcmOr3 ztEUm7^Yzn9`0QTvglfxA$_xLly;kYBA87Gv`Or(*0&=B7u|N-Qk%@1v$WkZc5%V1n zbP>R9GUPJPQD*=817dyZ7dEmna>@y6))?Ch^kN85fpeZwvf6me#j&7uyaca#cVjoL z@4;HXzuCHNmevI(<10gU^tC3>!{ow*)FThBI}!aZbRnLsYu;1 zH3|4S!_Ya&J}sJ%oN-Ff$TyP7um2{EXje!oxL0W;8>7GnrAo5DWq+;~PMjaVBd^g> zx#b@vXyj7EYETVt7+0E$b4-utR8s~WpzIBx6N1fyi_s5iXFQMCZZb6M^FFbU2LYG< zfh`hgoGCN_8zcKgr*Sd?(PSp)|=JT;9X*T+@*%V!lQZJoQ{wRLexs0_D~qY2SWW z5RZ*TWg<`tOr4sxkeJcFHUMRcSG3X-|Nm#FBb2jY>P}|u_}oOR9NJ%^xNj#}C-_;f zFLle(x^l+mORhbJQ7Y)xbKXs|xVclL=w?>V6_(zV;5e3kjHgSSDtNgAP3Kl(4~xGL zWO?`^92lqndXo9-ZKChkJJE)twmT2s6=$y>oEwPhW8OqcW8<%W=8;c+en#t%jk{F( zGeC6jp$b!U$}{Il3B+kKteqk;oF*6qNh8jLsWks7hiF?N=0P23vL1q6pmT>#ZY>Cn+nAJ#M^ zKugquWFX=>9VW*?#PwZ*j~`FBl-`g(ncw(}&WTVO*l*L7v{*V&3&`g`tdJ9eNmWZr zq7QArNgjOIYbX~ZC60wQXfkl>=fhsLnQ{*0ayEHFWL1@tLl=Ec2UAII3J%}g6y*_l zH2y~$G<7t-scviCng35V0y;-nU%deKK zC)iI-ZAW=#e^@h{8T|^wG2Nh>3Bg5Z5%xFzrR{I+l-vZV=WO&P$yt_7DE@K*XPr&t z=#g1sX|Z%SM$Q%HLm6BsqTx+kXAWmY#W&rFn6mM3_KB!>BUoIj_^gWEiumk`w_M5u zGAwcA3vpCaNkmh_I9rxag9}b?UyYlej<;%$Gdm_l<}WBKFSNGS#Ky!y#1tR5c!s^y zn*61d{HDLKqkcl`2yh^JZ5GB7Wdg@&hBYE5ly3o6q>!_wj z9GPtuHis}5y`-X67UlmaA|B^kAb73rJWge}CV(r1bD}`y-Pb^I0Dp_Q8DW3#NSq!sGaltYsIhQd<$-&cnty!`SC(xYy}I5 zk(dve9^@|`yauQ7;d?xYh9>QL2GmJ~S{g@>VbS)|IEnnxxp!{-SE3bdAhjTeXjbV$ zB@MeYSA6Nbg2-2VC4A%vu2%!!fbZWvfdw^(vGrWO;eUqWP88{F@USj@VDnloUG>g! z+YYacE_%x1ivD1f?dszbxV^k=w6=O0F^BS^1$nw5H;F8|_wjd~ygtWV0os;#2@Hoi z(|gxmcN-tl8O_IYM%F~$F3|TaJf|;!TbBR*jt-_8*Ts1x{TL|^O1fcvFT1-_?&+!F z*tMvj;gF-iLgF6IodA(lr3*%EGK*w!w^G{vo$%r5@$kzUF=pLWh!~k-A09H>pT4E! z49lt_&oVin1MH#@XL;9{%Rm1U4A*&}Xf;rD|9VxI~9FMr(BVNlvVqwm_3i4vqjQ1F4ozCJz{aey}BKb^C;MiQC>cqnE_ zbDJ@mm}kN3No;3No-+9qbR_lMD?%hS4{5YJ$<~#1@B_0jF4!Vanf3#9TJ0Zf2_&vj zVdSd)4GAjls%%$Q1(pG77z}&xg^@TBSZr&QleU9!vY){Kdot>y2 z)K`YAVvQ-|46kBQDdte7L714Nt9O!uH=tQNEd~^uP#d&KR|$H2I@5L_JQ8+(*;y=f z4O{^e^dP)gW%0glfEozCTP{n_${_J>&?|b`*O`+d3#X3o-^@T1-U_`#nG-V6%-;9L ze5ksHi2e!L)!zV|govwqwbGaJ6{)!`Do6qQ9_sp6F-tgaq84yN%YB}i>IfxBJec0% z8qJhh+3E;g$m=ko&h9IOEPiOFffhgLB=}bpNN|}`TA&X;B%mbLO|qj8{b_y0C^nqn zB+Db}r&B0)@ok!e>nQn^hw{~F-seni>sxjg$qfCZFUgh$uk>N)kj$CVo=9LkPttJL z+vNE~;dl!Tx?J}f(dj1FVz1lar`lm{T;nHMUS-)u<9nG(?#r;+1Ly!2Hk=PeLr ztb5PVCLmVFGZve<`Fq&yjJ0=fsajg;#D(qd2J7y~>ZDz4!6YVLTwE4;6g6^B_IXO^ zLhpI4cc~eJI#20KJFKI9l{*=qJX)BZsWT3scmiX{IdTSOt0ZVtdsDo z2P?eC8@?ESpoQu}aL4ra$m*=Y(t&@UJ^hO8ml6PL1fmBNB>4FFTUES*=Xybq~3xUNK%)^D0cMdC<#~wWnX>QCtC8*Mj?S1%v zyv?k*LRR1YY55{Y_VF}?g~g_uECTvgE_Co<8K&}JmPubNrhTGLk@UO}Zgw(7aq^MZ zQzOJ^+Kg*ya{Wgd=W9l5KhP~=otgIhvBItE|_b;MiReDSEW%xpTiu>Qk6 z&pLjZuzSmji^J$2Op;k`JG+MyV7tqO;BRCPkKQ`mHi5!!d|* zid9nP7;wvq+mzhLrR{As-E1|@T~N}fQ=w*#vYJMW^&JrgdhIwio!~ks*$mbP-H@sA zY>z&O3q2mxgIGB5C*z4h_uM6kn3w4qpcUu;-Ca(ulQFAXzU_s;fd}e`aNs zB%}kk*h;uE+@ZXsm984&h$0LN)=78|&8QYYC9JTej^p2XsX*b@hcC1hDrze&PE*+$ zS0;N>u`Mdb?(W3?oil-*ex~?7+NItiPzC^ z&>~q*W(+v04O9`jd^4GJ-y7LduBIcL!+x4bcCUatnM>F6W{X{)0g63RIe4ULwVbr5 z%ZRJl^?UqqitHwHay&a68QMJ6u!{{I%;tvB0`0iNVv)l}*H^2!rEmXSaZFzoVh`PL zL+d&Na*7t6Do*)Q?D=F}eST@Da3ojYU<%16lqTdJhDgT$kRu+J%5MWrDr*2JMp1HU zMWj1yJ`V~=j>Wdeti6ajVS$SGg1HR5E!-UCapVc|xa>LTyUds<-|KRu3_xAKF{MZ) zaI_22_zNCg{|SBr{Du{kPYa&33wS+GmR{dITAg@3)x@4LL$A1ZkN=EAKCgH`;RIqW zT!Y!>QS~aHszmp3iM}3_XH!?f{h?F$nQqAJ?7OF%sjkj!9K?tVKoIlL;b(G@`Si_r zmDJyy2eH^(tgKaIEw`7xD=ZT-3i=nyB@Fm+idwI( zw7=8I=9<1E;ghITphIw_{H=I2c|Er8__FRt)iSXScq_PKVf`*1>GJw%9yN)JD$oBK9!fpl{OsL*4O{?Kf_|T~mn7iOA4|j! zonqzr;!+0W%ghq%K`upP!;7JOnHhQ8P z^@Mkt81WUeeL9DJMmk(T2B}M*)6cdRf8{+m{Y6Q7w7*(^uO6eN1Y!ZP-=^fqbs39V zTdH|U;l_!TkH1l`_ANX#$|GTmwcXsyP5-+25}w+^ZWLn^SmVjAx)^k zWziCjM_kXHcc^8^{T`^Iitr%yF9oDW`U^m0M-5KRoPE(*&;c4HP}6W+^94lca>(jp zQK2bM?&-n(4m|*U6GA>7vAAtiCV_^xeANvElhzTgI1bl)Sr|qO>Cy{3|B*H{og_WV zvDPDbS2&1JQlw2Yp=X;Jjz@?j%QETI=;)YfrQFyI2L-D1#dX*%sEgJ17gyrEp3;5M z3oT$jPW|I|iFNSOlAb~T-~)@LYAi*7?Y{~smdV1gW8PX!jzV)Ieyv~?-0iFiExLEN z*r&uNhT5k_nv!TI`r6XC)gE#+KAOV3bgwBl@a`Ji7mwI~vj$hnRIzE(~N&nnv6z_tKm8l#MV3|l1&g*&k3WvZ6kKA>S?u^^y~2t|Ek)4 z3JGnc`Z7D^=!;>>|DFRQRDXicR!evWFF#@`SAl;p&45UnBc}w;$C&!!-3L7KRN7r` z#y7viMv^YH3J;{H+i;2V)ljB($IX6g_rwjob`{d%EE?GzdhsBAN$_9C+D68L0bC)X zo`$b3%4tn3+Dk7>?FgnXdp`DTi0e<*3ELR{SPvl0))MQm#_;FDZx!b~^EF?(Efkuc z-^*J6^wr00=#{s%ZLeWulAu+!p8OOOBGYPTr|jj4X7KbPP92&*JL+;GYz)IJn6jD^ z?~nNoC$myj<}{4J@&kjp`Hbx$j=@D<{ZmK1)Qt(qTFR$=+JDRg3fb9TA0LTQ zohKX+mx^(n^A*Z#Y^jJifD7&)Fuwmsz8!Gsx$o^2KR>;KJ+5y8r>wEqK_*_x3i3ux zb*zf&6eHFH!U!&wT-37bY}em;|7CDhsiNMzmCe!n@Sgx04OAOZ^^pAmCF7#gC}TPQ z-GNq3HvrlMG_;5Yp(_ShAD3`);g}B-x1{Ul$co@s%b~;A4Fr0ii&~Fr2YcwDtMlRB zuY|tJmtinPb1IpxI95M?ln`58&>#Zs z_LTw>Ee7k;HN?a56%9En)C?(&2Mxp8XcVCXr2-j1kjdjZ5%?aS4ZDCyd?r+sSRqk^ zuhRnjX)>4p5rc38NjCrI@9_?o^C*3sz%CIom7kOliCKXEz4Hb_1P&wB$dfd*oej~qJ6L2*pPdyIVDkZ~05PxP2#8FtZ|3!?;k9m{x9`W=n@FFujJD_P zLJ%({xu_ha30ukxz1G!~*H-L3`e3S_$X%>@rpNO{QlhJpr}NWdTHSHOpM%&g5QHE| zyDdv|k9vVWrr=M+Zf?im9v!E;@}g=ZPx;Hvsk^OxZE-S_gCc29VlUzKWX;+K78>ZA zP??XZN{8>sRp^RPXn%L{r1^B^JiyO2X9-uZMJDjlb;?w9vN3xA3WW# z{BkKl>@z>&!6b) z0@WUoPdpcp8NhfD$P_frc9m=deG)v;ZdJqbZ^=Xzv6K3%Gt~Ca+gL-?0D%q=_X2bo zZ6TcUWnFj|5Ib-wuXcSpOd2V3S>-(XX&*`&i00lPn`EB2<34gdg{6=gYXFiK&j9Ej z$W#^9w2!@yDk8_~6{AZdH|&L{<3P>+18d=*9tI1@s4J=& zL}w}!OqZjU5_E@xt;zA(N}dw60)>x<1m9iJ@jX1-FnPsT(UGG_AUU6&&jr{{1X%5) zJ>J7GkY2}t&UMu(N*-#=CIk$pmOL?co?zI60JA{MmdOenxPTm?%!v~+y$FLpxOSBS zmo;iQn@~x|P*^eI_&p=OerFPd#(q3i;1%K301E?*6C~ROy#`n_@Gby#ObR3bd4^c9 zy{_K5O}kb{OS`&^<-z@EQ)V0whG=6NfWkl^j@#k?%;YUjgRH*8eA2BA+oOAOZ)gUQ zIX*3*=+;|jn@n#0T$k&lWFnGVK{c-%#iA^~(r)h31oX#U5NalW7t(TUjd83}h<_7U)ClSY1e8^NfFDdM zMs-+RcWUu^040qGrK}i2%TgKrvOO}3At82szuyHv8ExbQ#d{$ozg?{yS@O$igdn`3 zeRyg`-wT*9UKaFW7W6;ymAHOvy4jXe1g~9LE>vGuvJrYfQgxSw@kjQVyi4S z*T!Q%4y#O~p5Bc;bRDQK$97z5O5w@$#@#aQT&G$tg7Sr23!V(lo z09}t-o!)9&ttlrpBi5zsD*aWU*^?CQXA_y}sWCS~#-;~w)Yez7Enrcck=Yew#Lg(` z(M5b^Hl~1a!XyMGC;RA@d;#TG_j5o9fR1<#7BpC(Vzl`|O7oJ?EpK<_6FQgNSmY6j z!%=)_G>a4|KOs2i$7naRqj9XeA7twMdM9VTPcn8S_-q2NRZQ^cuBYi3*IZRluA1_B zA)51~h1^e9zT-CoWPl#B8X=|=`?I7gW?q=wD=w8sG*yL`H;kJVO`=8V-`4YQOm55coxIoMNHr}vj z&#as7H@j*hJF%V#fD7#Ekm#!DK2Z{B^2_yc!n|@l+a-7lxI?FogFK?)1pvqQ(~O19p&F9zL{(3w?-`#(eBYWDoWpV%?pK!ygMxynamlA&TFo51arJkpt+2400E)7k$9%*VnW zb5EEjtzNvYoqOmstt%t!syC70wz7_0xMT!TE@92s56Kgmavli$X%${fdC{OaaW<3u znqStJ(U47^@JetPSEbY}vxQ`bAQ^z=`qt%-e?gMFZI>5@n^yjDX~+lDnaY`n6jo%Z zr%Izmu`0>!oY12CebWC>{y&P&GAxR(4dc6X2#A!xpH?XWK|*rrln|t21*A*5S-Mj| zLb|)VcIlKxy1S&Lmz~%5>wK8&T+ck`zJK?bnRCqJEb+q{Ls-xn?hm}3(Y+Twi>+zC zUtMo`*8>L5Qj=exAT z$mjp2z+?3&c$|4FaZ0y8r!G}G~Zb88^kqVsI!+`zNjX7_b=vbK*yb)fo}-9m9* zVm+k*;rYOG$M|&tRWDeC3f8$)yA;tQFku8}H(?s+KNAxs#^OM9b1ucI?By|-8cn3n zStuXsn}dj+9Fl`0uCuijd2le^X`g&xS5?qoR5RB(F@yoBVcQfX>D&JM3y zml-D8YI1Xwkw&ros!(}fs;$_q9Uu0#OseUqm(U&SKR)b?25hmFQ@}EA;?RDA z`B8-PrX5P-;%n~jWC!xDqG(?WS*?lBI?E9$F7BZ3uZpU*xL?>cA4zk%oN}@C=H;gg z=D;wHGPacQ@D^!`iagCVetfyp`Ro$)+14GeiI4JUTyIg{P;a+RHr>Ol4coJ21!joO zhD%YbkEVQAX}_P@;2LeuT6VXX0i$n2Ci1p3NtG}G@P+JNksN%EcZ>?5THKxgg%Gp%s zHfn9kiGI#|Pe+9l4=uA1W(9dk0~F8F^DRb0mj@Jg_I-Kd!tD z2U%ixRHh;McVG%OzeSZ0g!d#X8h|i)AZ|H&Ffb3h-;IZC!u~8l*%5bB?p!HQZ&{MVoeV~g3z%*2-XZe}go=m@eBvGcTwi`ykG9E5K zq=)}lf$2-`R^8Y#Xkh!EOh5WyRGAcfWYYNpycaeC*U-hdJo}w=d&DMzQbVe`@PE{e zzuB+&b=CAkMxVXri`}BFl`b#eFL&OtDj7sU5t{lJNkgGBPD)LR80#oGS6lyZf3V?{ zf9=;k1MVq9;rEl0CrAw7%uwx2fA&9}Sz3c>8ol4NOL`$iX-%XGKJ38w%0J~CN1WBaDqQBc zhoZAf`0SdDC+G}HcYePraVFsQ&{1;&GZxyJN;v2h#T6E!RC@TU%VDFJ$X93{V|gXp zsJq>k#AZ=4?8Py~+#zH#o=J zG6XHiV-k;;2&6e19puG-hPYser#8y6ocRu#TOfoy3kQ|HqK(HVDL{VU7QkWzfv%K1 zE*OpS3{v0jGnaYPV);dDzMSKVeVG!X%Ffti6XbRrMCcN{Vu2@SPWZw8#N@);eF>qjQvRn=Vg@E{!j9K?dt`0p%WRh`?e&f)?fE7l=nJVO%Tdu)qcWP zi8t?Sf>;-P4;K9r6Pd>kVAQ4?4KDbv11GxMnyNtzVZcn*Qvu$y+UM|E-};#>5h_Q_F-Y zny#o*5fp)mIxGc*r#@0rzdi;Vr)R_aMTL>2p5e7`ckDt%Nq+eF9yjRC0uIa4y zQF6zXMpRI8U_@IxVak>eIreWNxiJmP|1pC38p&S!fLAt=$0WCH*DYMj=0-% ze`V{r_u3dRU-$B=B$6GaIU< zys3~u`fxqJguL|YJfA|q`TN1hR3_nI8`R)#wlWTMDhZx-jT&5!^7{jbtN?-_IGQ@W zH)yPSSiBz|%Qu@K`gI)ho&*I=hN(_TyI&Xr+s+K<5JGRn_@CyyUJq}MgLD~hY@ycv zJa#@SMWMNVnvoZhC@oD~?)yWz@=iXcfhE@Wsc_6OqH*jihd1A<71%DU&IcU{RW^1J!~199+nh3)fL;@=eSx$hlK+5YDk+KE-PGMz1g><_Hx+b zxccfB?O8Le1oSXjU(Cv)V59CLvfWoZW_?9G`~gUk{HGUrKHE&bAdiTc^T_2L8oF9V zgBESuRAvzp{`*%lErrspyM03QlNnzZlh*Jq$vN*O_;8T`;~xM*XK)R~QSeuZY&TwZ zERW#RvU7P=b#$=(lZ*#+7BnMV9Zx|^KFssNP2q{h6;G;D!lxUB2`<`e)Lp@CcC1Hd z{or=8U6-~w`b4x0&8fnIS)Rl1!ilo?#k0DJY~;vcs+oc~8>{zAcfSTwGxtiKewRqy zD9g3=#JPp7&eL?P(>9D!by%a5rpj7X_Vc9yV^Qh;gbQk;lxPkoI`2lw_ zqD0SyNhIi?KW*6O>yq=TXA{(c-n%#Q&;Z}P_<5JtrdxF7BML7 zWH5SsqHNP7Y0{u;)28#8+*zVE6hLCQ2Q<@`6g+5u~y%FfUnAU z&4hN^4;b;O*$)k+EVC4|D-=f#5s~GYtt5slDDdB?5xbSW^bU{t&^-`K5g+?C@upb# z)I`W3w)kV1X`=Chx`367Gz4dWkhT1 zJJgb}W)_f{ANbt(?yK@Mh4)gE())a`n&$!m;$YzCZ$cl-3MK0@xp8LtvGwU5OFOfs z=3=K0Rw71N+1bO}g^sP~qon8X_!Na#fjOciI(A?^1k%bshdcS@V+9=wbdDB&{9fAi z@E=nI`iFx)Ps|ry98HJghzldFQR5~P=!3xoG55R(k06Z}K0l8D#zgGU0Uls^i5r3M= z2p1x8fm^-FfKEf2U4E%n-6%#-4NDZ*WU*OWU*eu+HIy_0rYD8pFDHR-qp= zWTg@5$f<~#ye3i!XS{0e5RD1Ta+yr3A$VJ=-ndHO2;I(viM^r*}J}AZ?q|6TJeu8|3sFdCzI+=I_2)?AM!80#n2Ymlk34a6j*a+GI=F$FiME+xtKm;tA8uc8!4%x-*--Od%JS+#T!67@> zW^XCg>#)ot zBEc!TJ8|g7=30)gp|S8UV*O{v8lgkP%F@*)iOS@w46$<&Bn?~A;!KKa_}kz@mQM_M zBhoIl{)Cru^X^SE9+krxdtX(g)V=M{5T&u$OV(S<$6HIs+k(Foko;_Y7)TKYSXyvZ z{3eTK@Rg=D6-XjSYYLRiL-|k1iDbHxDA+d#!sn4(xE& z)V*ueui*mHUhQGX^avk~T^HpYpR{eKQnTKE^OWLr`F(2c4FgfPB69BzD({LH-US&h zbqj|&-J$4|)J-!PGbZeTT0}c4NRNomMBk9IxU5cwBs!*d}r{ zzp7*j=CrN4zv{2A+PbYeAxv?>{$=uhLK*{ugeixP>jAl+F_;qPjl3#YhMwKtZR9-u z6gIFQ)8LE~7M-r&vyQ_Bw$A8+Vam1*x3Tv(VIo;e!7WHxoMY zJbC){?nf7t#cK^sng37`J7+8?VZHal%YQggOTmI0u%9f>*Go{dHw_^4Hd|8g5O8ma znlbz^W9WYbfN!SNxbOajfXr>Oc4i6GoZ5NN&Dl?z7Kf@oLk6pxi+USfY3;-{4i(n7 z-*(4Qb*G)LubT37l^AL%EMKBz@yOTE3exXcu{brB$3Hjm6moq0gda37z0Iq(eEePC z27!#gnG~}}{g9>lKo_>OlCd%^X=7K|?3jAGLi$`RJDZc#91r6;`0zLYhhF{5FW^8Q zIkPHF zlwf3$K^Rh#BLviuv#y&6L8(8>!Lj@@dc=T+H!la;guR1&LK0EcV=3RA4;qP+0{Nol zC_413?dTV%!Ki;|qy`WzDPh?b-rmvaf$FI;V?cZEC-)3LdeaHKMm3dfS zB7CdAL_WsRrw0OYy)P2_(pK-4m+!Qq!(S%Hfr!sKhux<(Vj!?%MsY@*KOINd6&(k#ecfz2PH&j^IQ1!!;NkUPZ{D8Vdg|zAXt38nlo! zjvIyG?yPI2Oll=g>d_GKY7g*h5AbRaHc4Rc_HUz9Lx<>hxKgH*oI=&XS;av|2{(K$ zJxD4(8pKof+r&4l9=IE@SAZrksCbX7?)Rb$Xyp}TR#?m`v5uLMz-Smuy{i=sob+?V z;6-DaR!*!7_w+n>Yn+hCA5B=iF;L~JJpS0GJ@Lt z?Rafj!OP=_SW=U?X0zrlFP+em?+@cb37q8HAP;{sgZ-};k0nb@EF?L7SmoZZYq+! z_QqW)=4b7+wk8!lD=r18ob#e%M&2uXZUQ9HHM}CQwB(CFl25Zd3?R9p>j4JL$))|% z5}%v_i^cx8VZdXfu2ksC=D5K*WC>Zn2V+4A5T$JNW~+w*xpxp=eV7NWBWajCI2?2i z1mNxfoxPgvSR>s))N0{#@Jx|U2j&+E{7H7@=BQ}|K!f~D!z{Bn9^c4~0FdTX)wrui zbLw(-RWf@HRjRhu^0xLL85>kn;jPKPr)6#Iuz~jWRCA4zG7pYI@`v!LHd=mUIRZ^=P|;)Yo6ddp&)e$pQ1bE*0aCL7&oc^O9p z!vVOKfhZtAM(Mbt)U&GIYHgO+!3t~dEy4S}2Ue2*FDZxq?GI|plkjET1P<4VLKrwV z0mCcBb4WU1`x~rwAA}^b=lgFVZ@sZ(l{@YRc&nm_05qgdO`l?(4EZ)-QMC&>ve$5f z?|;tW;rPiAsY%H&r@&=#hjN(|Y(7o(#hRPssRcjR49#X?5I(BB;#FUfa9>z|M$ zsJLerAb!kt^giJlF&q;~{&$QSS9uD`&&JgJm47D^%Cp4%WSt;1g&Q$;!YLm2(RZq8 z3z6pqqa$Q1z=UU2$3_&Q8AME6a|A#!t9t2lJXyc&wyjqdKXtWlT=iR>snT9PoR{o7 z+<0}hoOnC3?c@d=NNF9pp*zcVzf9p7{95X)xL5b1X*tQIh5b)%zz}n2Z#ETvCl9TU zD-TW!FYWn`4q}^0ViNzqV?seo!nEV))X9+B(w=TW&#^)BB|AwUQ*f=5L`5`RP|-yo z1_LBwG{4c=vSaUqy@rpsl&y`OtrFP>NK+|eQLPA-Wl6YwS)^5Igq2P7cH`@C4}}*J zbp{``)7(|qRt93@&p<`R+=sS>dLTKQ=2Uu#ZG?8C zv=l|o_JE61FGt`Ir?E>Mi?v#Zl>t>l&gE+5tp{QY4+ZCuZxxgDvqSUt#_|oj9;{4) zIP{|GXfF1DvEcYY6ujK~4l}_i?p_9KM^2u-iVw+@-i;*QpTZ2aO(sTa%F#w%bKd(z z%^W@u6tfw`?Fpa$Lb$!F&7@qYj$eGRW$o%^Xf4KAb%47Vwv=fKGOZgy2EjJwsgou^*UYKI9g-;30JIPIk*1^7P~ zUuWjsy%+a~3L~u&`(bwDL@zVkN;@fnsRx(%Z}{4N+W0gkOI%KWGycR|&GJK=bb?)x zG)e8XnTw|&jf^;E%m-k-_MU_usk#E#Y+B?#LC{>RxEesu!G@8~d@zzi#y$TGQS$8* zqq0v~rG;TyPtZEw2HSb*-K^5NE#TZ|J0T=AKPu*yi=S5C5)_W31HoJp)u|=%Win`+ zjM4Ki9iJo5fo837DAY%)3uCg%iV4wu>^7EPqU_P?n6}Jo@pKjsN0)pI1|(2~ z0MpfcK@}HD0=06~zWgZh*z@3mUCu2r1Q*i$ocN~HlsJ(t9$q8}UG8}^R&B-ZB<)|; zD!mw-arUM1P7cK!KU*WCiMatY(pAakBmn`pl)4)LgU#AEiY;1zBzh@I;<1ldy!VDN z44Cq*XsY7_QgXMsuXmq=zxe%Z>rgVBsw_R3(IPsqk&C`2$yA=n{8hjA8&m1=drZ6S zK3Ko)uDs%&?*DIn573_fo}1g$RqAahCA@{*huhBqQki7iO-pHAze9-EekfeT>I!bc z8c^+fuz$H6I5U{E&(`3W7@~O7SD)Wex~X*ze9_iSDV&||=C^d3y?hq(iVPXIGhu-v z_{@(wXgLtj|14)$Cee;_G(uvdNWvDiHF~+dAjs`W`6(?L$6Qk0>7%$$u!BL1s%@%( zgiE-^=Nh@M@xtJabt#MoaP!W){R;VtACaX_H3`sf?Ful6Eb5ZD z_-sOOy>+MQGJ0^td5vAb2LV`89E_sWjz}(n5CGi)PTO+Hqes~F^mYmV@0{+LD%}lo zl>7Gf#=EO-ZvWC{G9UeC?7|DTmnBz`JGi*62D0mAxu@LIu}dTjR9P=^J9%kc?ASK$ z5B(|3(e1~uLk)UY{l(xE6cGWfiry9Z!Ych(r}(DDvCS^tz*gQ62be&S(F5*Fqh{IZ{%A_ zEK`)eC-Y1SYvg~NDodr}6Q8$M!>jlWC{hhUS%b1aevgbl>Rpd0KrQ) z)XBW(|w8PvPuUhNr@_;e86<4`<)m1@YHXVccjB%qUXC}>l#4&1+A@kU6G1N0)S-#xDJ;=er zwWjRtXY~#RMC?{=K8B3rmU>x5G!`-F`7La<9#~(HGyqTVF~e1#ARtanLSl&yCVK(HQalqQ)jDF z5h29p`xdk}&?L6!#R}5$D0ST?>GanL=35hQ83}{){*6RhWCu1F@%(m(?zuow1oiv> zP!w~8mAAmd2pt)q`OHJ;2RcqT_ms8K=R(j1%+XG06Q*g(Ih1`4XF}83h3^5^#Nu~Q z^`EMw0L}`ahICeP?na7NGt|SGY!J^-6O?uPc{m3gpwKq_us;v*m+pRGM+}xB*reV; z3D6CQXHyz3udaStt7Jc${;|kcjyjvtdU8OmM@vZ~w5W*|Om;c-KY%)Hn;N-vmq9&% za|^&MT#f+pAL@t=|4|37H_aW)F-i3fw*N2Gb(pU+4y|vn{L?}26yD{WyKD`+tRMfR zo7Rd6=^&7j`V3P@lU!&>uP^C5_p%tjh1ZXoU}t?THF^|Mo6F{?jH)%m)Shl~k0?FI4x+`LZr4QBUq|NVPZ98vJ9L)4<(M;*03S;#0gfv#*n?D(G8+9)ZIgk!r;KD}2uh5@drpZ4ZaAi$ zX;nUciaI`|O(~b!NnvI^d;1--5b)&6b#vjnU?Hafh{>Xme$)GdHUEe%o2E}}Q3*Au zGP2(ET}*F4EST1q@UfTo<`2$crPzxCgwmU70(3X!5X0mIPP}1@<)OIWrp(qgiPsAx zJ|~e_?N)T$@H^Y@wNL-{3)a8JrcJoRRamtQ@7+yYI zL{6w``%aO)^LQdM26G@(RlR`y=-ZSyFPPW#Q$Dr<G70YsVKb5cynFA+>--(twnMQz8Dxn^CsZJKdZy*A$@Y zCt5Df(1GnJuOrX5X?nCt-EV3!`NV2_z#g!hpxy(tGne0}%zr?}jlo;*vv59$XvE2h zuLeY2cuHbH`$!elWu|#cKS*(!4e#g@7VwZ`fQ6pF$dgEA zApZJ)=7?(wMD_rAp`W+5tASrY`6HzKCC2UD`u~3g#}!BYA4Wz(|2v1bu}*9P*<=7m z$kC7KYYxyqWCyAVVP*JWV>2q(stwRZ1w`pQDXq3xQ%}(STryyw% z17N4MriD)dK64WTO@Ubby};-2R;z)}0&}0<+gH#tei>5Zw=K)4;TyYYCPuA`RW5Mn zukaPG)#t3j3* zNr_w|$}g+Kt2rgY@P1IprtHX&)$#8Zgjw)NrbS4oA%lQYVzCHgBMe*{VsR1NKLZjQ zvh<9zTx@;p+%BfpETtc?wu?=gUf8}LJOpQsw%T9*V*QJYfD$H%2}?ZOG|Ze98#ov` zLp|5<=YXMqc)`NG58bxM%W>VD57aUukcNh>;&ixjoJ00~yF8w0hMv)V^EPdNbgu8R zz(S8(X1b-!OG)|^HPT{6b8TrcJ6u)5jK#zmgDnEtzZ$8#)nD#JSrp$oGQY=GiN%3$S?&Ht?^k_ttX--Y#e~(Qd^R*74;Ckf7;dK$|`w7;|=HePaB-qzE5P8jgNfD-y{ij!KE$GRa`KH;Nk|$bYjgk?* z7B?R2-5o7~CsHoUdY**it?thTiw9isg$xc>bx%~684QM?G;;c)un+Bs%r$8`Wvif;g2_1$Wq+s2`I2iLi7>xuVc-~^FHC){u4v0j4~ z_^|mfh0X~R+Cwfs?YO&rhpvD0vchT?I{G^c!lVia`^br%OzjqKV6iMJOYPHcgA`^9 zoO^8?xg~GKZ4C~O5x-Q#A_l`d(ERw&mS+Anc}R*eNlnG{=Wvh-?#o*k3OI$C>{2NE z@e4p|RwShntK?srz;XUwmf)M1IdOatN)Efvm26xWzewn6fMBr%Q!T%J+ip5q+y8FNt^^QS=TF>G+8x)%tZ$o&poy^qjFohG}40TEV2wc;D^ z;Jv`zIkVKK?9+32Qz2-l+J*xS5M~4gJgyUig8`aj_%RYX#6G&6w6k$J5x50kZ#aU* zL*C6M#_3!EY69Lh_;hzUy};Qu4Bm(_yskbyqdRT$_LMux^-%GA3*J}7hOLHC!sf-N z8Pt|e?7JGAYdkQ^V_|LhuKCwbz5R<6GkhYQWOxX^M9pd_l1KtX?5_t>V!wm^p8c?@ zD{NJf_3m@@@o3mHGNz*Rh^m3s0Jk1Mzvy4;*?{F?E;Q>Z)z~O>no6`{o+1rsP-D^G zn8WmBy=)%s$ZldK>p8W{`@$fSq2VddF87;r$7@@~a%=g!YK3yUcyfz)yXsi5cA)8W zx=Hn{T%vE;`FJ@5!)edtVs5YjW{1lVoNwY766!7PJ#DdHeDVANrXbO6sg;)uIufsb zD-H_b)ux$G`znr5vJ0UdRGOWrKGv$zBDy@IeU6c#blHyWF?)cRyV2jStIO4Dke1Ok zlMV$~q8auFIODwYab3FF-lm3pXxRL9YOgk&O@k-0Nal!Wu__mGihAQ_!F*c4apD3Q zf?!Np?5kFxT1NP1gE$m9Aj;gubaU(Tf;?c!Z7dNWuiWhDr;YwP;%mlJTQf_s50;hO<19fVH&l>R~1Z&lN9ZQlA1z~@$MPAe{uWt-pXrH8+21jPp6B3^<_^?U}{;dOy;N`6VENRP~}Hc614_eh?*T=^YWRF>LHK22h? zpCp^~5l*8KH|l+DCin=(B{1y}+H~hR$GMs0#N|g9X4Aq2YQtI1s}e1bLp_ig7|2F4 z3&Be+(a#6Hm@r?;qvOAv075D)l~-20lFB)DZT@LoF$Mj(ar3Q+rdc*%d{n52$+En%t9%K*GRqqOqP)A` zTYcj%!Z&9gM?%nd(|nJfJ?|JR9ZJ_{HPT5S&{KO2M&%pi=rRBQfaPv|qyuXd^#mZe zTIbmr8Pciv|GoUQ#$?Ut{zG!57Y1&?RH{w1v^l%bD%xcxwXzpVVy4QZcxAvLiiYQ0 zznsR%h0JkJ-mBBR12kTo(BEOB219>^@2SS&q>tSG{BMK1dwH&BHDKRXD&xG4$2n~T z+^XleT6AGhe1#yaL7@Gm3|Pb5FdLCxYrri+z_gWuHcDRwyf5WVolR8Kt9-Yjui#Z2 znopPVrwEHYRGPs2b{ga5XOdg;zP6)fORDnK*uqE>_^ysmDEq3+{{K!kSy<;DQlQ5 zm-ofq>&6zEEVCVc9)pDgMEp~nZttrmieSybUJe|)x0Ru)?uk&^FVE;qq6w$u;-+Qe z)Meu)Wr;q8-G0hGdj)i6BELp;Zxe*VtKI+&!A=&0*Rph{<=|sYUU*iwQ$}E&4`{@krNJ>zFz7he$9jqY+d294=#nhlZn8sugEw^qQmnA^ciJaTe`WWRhsE$W%i#(494!LKT2Nz&8W)O(|!x8BelUU4eFah1RDD{?@VBfu6_HdAB zbhbM;0?po`U|mP%$qDy9-!uC5v2A+^EBE)37%f(Hpu9e>}PsrZO zuff;8fax6i;*>Q`Vq1Iz&5MbC34B2A(SwF#mb3h_7W4C7s}!DE%30L?x)Je8;qLA1 zB|wC7cqyHnU3nFf)2sms1dbooBgn<4BV&X6PaTKQsF*C~Lywij?fSvWn5ag!Y_?YS zv@!fEI&H`PA+(Hyfg!G8nS~U#4sk)n34>$`oT2KWTd?t6e*d;98a^U~zJVlq11@=l zO-@hY9fUbF5Cb~A=Tg2wLXi|P-E`2ys(r`^{?@8Q%M$u}8ueL%oHPmq!^8ke{SdIp zy9WTW4KN!2)kh98>VYgT68+g)pR}EYXi=q~!^)%3@ae&0GRd%TWIrWgI512Y_{tjxmAGw3VrsjcUt!8v zncri6j$Vv)uY7r%6@GU%pyJnGr=P}qLYdnKHP;m@QsZq*XDZDetl3LR?KUaxjum)H2b~K0leKeZB%E!vLc-clg zsq`P|0<`vJr?sF@@&OuDh8b;g4A8uqvLTVJR>p|&u?)4{dO@*D`fX)rSmJjtO|-rZ zB?h1z>;BDs{(|4;!||vGHSfx4V`40S{kR(?G`&9%WAo?SnEvsB`~1wAojI~40>TReQRcGp|eFr;ka|lp3=sl*CgzXtT;|WL$_xy|cSY=(&|c=4O}ICFfm1 zXj^4uosqJhx^x;7v#Sz8y7X+_=TOE3j)?l9@bpx2t7KX;j_^j3*MjnxDm)2^>Vdy8 z$c9&^7x zQKz9%7mQ*5drCq<$>o)g?JFObSKj)=IT&n0W+hl114+d>YAb)JoPO3B<1%-sGN-Z5 zGdw6V$X05t4fJv7+;k8oK#56v`$b<|V%XdzaZCQyO-y)Sk&sxSJxDo0Bz2(T+ZJ{+ zNO*bnbC~FNyq|$=TYdQ7Rolz?dU@#I?_#H~44MBr3LJV79nOeNr}UZZ3o9u=s$V}f zMf{!T#7o6TezF`nu3#lbI?HkC1%^)igo_ONI;KA&UkxGe6-ME4x+vGa%ZXlc*nh?l zL6e6J9qiyX^x_ZsgUV1^GeGp~$JjMNQau4m24#i}nemt(^9wzNQwqc2}`LL(&2J`==c~|uiywdh!`J6T4 z!=)ME?w3nck~2h6z=jJVBgNjKxTp9+hGp{q<68v+cRf+kWJ6p6qjGO98ML}RBpB(@ z3%y6DJDfB`oDx^AzIR@#UC7N3_h$^6(YuxLc%csvd?|SI&;@;-)Y``z+di-kqV})LA!1Oubg+eW#9@BxCxy_fTRG=}vhRRjP z)#|GzaT{5!B&AkvbjfG`ys{l`Jx_y5#;#Avv+3E>pHi1JGuAXRLg!R<5MMm^Ruma5 zyR?w-JBXU#0E`~y`C<>|&{YEz>uRLtxy0Y|$dI*0P>fEES=Z!d0P1a1nCOZ^^q z#kFs8ix}G^e{pd@m9!g(+}{}-reGeNdXpH9 zK=TK(v}P@_(Ocoh8Wo%B5#3oYg6fq+wI3Cr9PjUpNe<^%3oN8U^NUOK_jU3qB8Vq% zUD-D#S>O3G;NWh-lnTm8CW%3JS)(HU!=of8aKbcX+swg@(4K{oPB@U6A$q-Hr?>W7 zajl8%N|Yc`rJ1@@c;;`u=Ly%3pDTI}F7Am_s`{x!O|{~;&F^7L<9EjFPkZLSwU2e? zQ~r4{=NPw`FhXcenfj=J=1)#sgjYeP(5$^jL#9tf5dW;K@Cq8O#QG;?BkgWIy^nOC zr-hwU`4|(wyHtr-1+utB(p#s~FJWClzyArpul_DXvDa#h54xiuIUhv3>xFHDU?I=C z>8_w}TnCUT=8FpTC(fX2$*_!P-lYgm)h|S7C=&8>APx}FR`XjD_iOlxJb7n`4LHA% z{$|hbwf6OI#nibK6V4L!8F$vHsmQ9#qbc?2Ji$)3HY0vZr~l2WK@Zx)q%tS;CK2`C zzbK+BUFpG|n_I0R^zZSTUPyCd_Gw}4fBce6Mi$Dl6KOGBrlE|0JI6Z!nkj>19uZq9 z9`u0GaKm`MUJM#X4N4Wk{^Nc;E}_&xl#`U8-a7?e9i<-DGo3()s33u)Uz#sem!tiG z=K1&jKsILq^ruYgtg_g$q@cR;s)ow4j=15Q{E1=yLoz-7!jF}T4IDAl33D_*=%R#_A5LjK>xIktv-=qWO58wZdZXUK*?2r|9 z2RBnW(hm~vlkjh8zFXH1;UvaF36MU{l0JOi^Dft1d(QR+rR1FHh)@mphp$K#UB>Tw zv*ANc;WZ1@C4F@4&{L=yxth)k9C9^zoVTDlP~k{i4TzVbARd#&>1p=`tOnFUfOVIh zviW-Ayazd4P>F~85xsei=uN)>lP^S3w>2Qa0 zxctkAAfT2UL5)md4nYT;U-YbP!JJ=D1ONJ#qYk9zZTAvEH!FRf{TwbqjU~BL{cNTi zE@DtlSj|RQ&4RjzJZ^sNQD+!EpPehbu&eQ3wfanMIjo% zfO)`t2hnHB&;YZF|HuMbP*+5#ViQ#J4`x@Z8FT)xgbiOO>?gn|^v&*n6K(;mW`kw} zk_9}OA|1+NtE<8%lR`=NA6c(N)b_PcUXd-u#gs1NzObFys?dnHO*FCHW-O8mQNbt5 zqM~}ow_uS6Z!e6)pBRnUKODMt6Tht64-h$RF=~>};+Lo0z?^Aa!2BlUAA@k@88zk^ z^?Nbn$TQbtwF6Kx2T?9s;0X9?PONhpbq+Sco2970RCd;t`rM zZmaIL)#eyMyP5q-XPfq|Yx(ZT_QpbKLEv}InLR()MP$MF@1uR9oc99!j1OfOH#O0& z1F^MOU9tU}%_o~({l%wc_AUr>EXw(CAN%lZ%-p2=@qTd@zn+BxyOd(HGuzMn&9*c} z3up)D%i{GvB$uy?VykQ!Qfzq0YE%$oV1zlLaKkXK}oG5?R z1(k$%hI^T2b)h227*YLElqcTT{(}pbq4MDi{2o z3P^{+0)XwXD5S_g6i`g7oz*IkV=U7DC;X7^$;Y}G!6 z7FjDC%oCVHstxYM_n#7=ySXgq>SXt%Nq2#GN>7oqsD3lB7ugT&z{pXws9CVg>j3&1 zkO8LpF7}Jl4e>Ep$gC;@fm(P1ra0a?Iq+$R=$x7Gn=$y5VS}Q-HJS)D@Gs~@&t7 zBx=9tll6DYut8<2+C`RMbB(ex&oJnJ`I&4^@L@#0f$Y^n!5~v%|I!<2-N8`RD7@N8 zoOG%^VZntN9>iSFdjHBq+Wzs^Kl^>xt~+-N&exYs+xJVCHN-7aobOgicZ4$0IX8Hd zPkBRjKlJ0g{ihK$TfjHrAX_eVr3VI1 zGw3-`UFUiZSG2Sx$C(e26?-SAPrt?ua}2!t8^73_InXRXkAkM#xzvr`iNB|Q`vt~? zZ4~R~#5M8NhiCoZIQfk`KjU4HuE>%pa^EkJ8G9jT@KKlOmZuhTWohWaoaSoXMkl)> z2e4t??R`BH zxLqZZXiZ#9pUfPS%@UpUDmslJx^ett!^{-~GT11-7m7(PE{)2%cOpWFZNb*46JYjD z?_UbULxg9v-Zl)A>-0K2--x@M|W{>eB5y zRg^Q)^)9i{Io%ZP25iyiFFNhC8DU+J^T|}=$tFWS#iNd@do5~V#G@tQVN~z(cJ#10 zMcv^o@140=+0BZP?fEv|(S(g9yrQs20NF&jy8rl<>vWM}DT-lf$nnirawY^2)kg+7 zh!luQz$Y(3Ifv%&L;8?#x*eojJieF55ON+pBkb|lMk2G+0nY)&U`tMMJjH+S?p>=d zo(u;d_Qq?Ah>qIGhklc3W@i@d$7fuw>f?NgVPypF$&~hQUvZ_(>?uZj2a9Y#$^dC$ zmi+(B5>#3fVm%Y8EV&GlqSnypQOsrH_6Q)4^!7V%oO-~c2TFV2Yo67I`fAD~jl2;m zCnK)y`ptnx5&RbkU7@V+5HS7)61w_F%hKmW=h$``iFL+qmq0hrjU{$Y_Z`0v-~o!% z;cX@myIcp)00+k|_kaPz#=BU6Q}?3k zj-)_Gpz|9p4iOSqM-E=oJ+A^t&f`EQ#~PqKRP{ETQ84i3;}jX}FY+J3G-CZi^4bJZ z;h+S5V;m1xr>vo++uWTK5X{M6YcgDG=7IoglY5w%u6#&3RQHAYr!g)r9Cppis7pya zr$D?{-S>+OAtquaEbJAZ|H~bYElZNNan3t7tNj%jtfh&j15yDa(9BcJ##t$Tl|P@T zmNwk*)Le9C?~fOreP*&sgPtUQgKN^L2GvbyN zK`MUbeYy8n9=p9R0C=w-0I;S4>3KOX3ai<$FQ9MS29OlR%i$WQGQ&-23j947`pAA*Mn4#6e@NKx1ypr_D6Z%&>~lz%5edg<2uhNPiL7Ir!He2W zuV3$&ZUAQljzMv)Va3OcVxG)mDnS*q?x=ZQs&nXx!gOeYd2u2?J3*|?aEcHCJXN); zKJ@ZfYvk%_jDq=PuSLu?Qw{#^?v+#7U)f4#h*gCebIH z#xZ;dwpW0pBW6%#r{<1{@L;}3Yz-#0kTBr&9lQl^f7V18=Vsi$LG=zPhaz|!x$?~^a|+lWq4g-P6mNt!Km z5g*9=yilD9+_b(02OLL<#b8n%-weLSzUAmJrdvY@di_LA4A-h7+kIF$pkE~DwYX{t z{_DtuXa}sI_X887LELY}FJEh*R=wU5_rSQYfIr1e`?lj60BPrk@2yX(KT<-f9h$bO zS1zJKG&tgpBxSl9N!cB8c9=Iwr;w#AN~zzWHU_Qj*&keZws*uLP8ibO3iJH6*6Kng zULEi*7FS(Be@p|*T0MA(nnbaW29Gr^ukShrr>o8eFwnuE2q0P($+n>&Gt>%$2f>ir zqIW{MPKZ!6K-t`jFIKBQ0451<}zO>yOav_ks58AODQwcs%O57@Z9 z49|Hl)P(I~Go%J9Opm7v&O9DnycN0hm`D4)jtG4HA7O1L8KQ2Gfu=NzH_EvfRbe;` z1rd#cKX|2QyD^;KvTgl7e#wF1hx(XjV&-2a61QgtzC>G$RoggnSjM~0jQ)7Q`MKh@ zHsy=`j8;`HkPVy!3(g0@tpC{VSXv#M6`hJ!5j{5}*f+7l!OSnHxLHuSc7WI3KmW}2 zoaes}vnP=J1Y5j8`5g$_ze)O4V~m^nu*aH@>+rwa>^0VYL}&!+1!FA$!>{>~MfBm`=zaR+a`38Sv7BsBkiS_;D3)dJ9HS z)0oA`W>i#=MEKV(<8TBHcKA>^N=B2DB%KuRGjq>PZ*uP_Dz$Q`zA`56FoxyZIPT&Z zZF`*Ydv5>d!r9heB+IN6%+kQw#wn7P+$=&9hoX#RuFp}Fy_l2P5}z-8K5o;2`(n2_ z^|zL4As!YDay;HaoZumA8F9}JhKrrv=slgkQ?1si3vBk`KzOw}8!$q8uamw4XNmer zkEVQ>LUI%fnzG4#o%1=|G)=`d`#(<^P1uV0M_g#UsamQ|s2_{UVOm$@9Gwn7ai-pQ z&3@R6M@rf(#0IWz6m7bsf%=eTTm+ zY)ayuPbsswMJ(HulN>?N!6M8|XYWWa7nF9+9TL{Xd0mCJ6h)qZiK_BHLlLgSZ6W6lhYn{4jM{xQWLB0!jU&SxG|e;21(kFU4bOc1m`3kpYGegg6iI%3^i+k;<&ZdvZw?zso>OuAK*lU~H>H9}# zuz-XoUa6A7*|LG-{%Gr&N)438b&x(&2<0U#c0{q&^_Tbtl<^ecxX!rGMO?jyEy}O2 zu%9EYGFMj}u-gTH)o(a$vi_s~aarzT4St2eE7f8(jwuUI95u#n-(V7D z)M`7|QiJxs1^0Qi+3gG?T7n^{IsNX38371znL9rvn#9a$jshllu{AeP)0=<@AVSOY z4C<`qGFtS*6-~P%5p_9x8sv^nwrX^nd=O;)4dxqLQ#x5z(p^^2ty8G_HI(hfg#9~i zO*j`qwJ9kd6Nt_*0e?>XRwBk!QqA##9e>JFIkXURjJqX>ceWk&T><7)jnLjNqbXi0 z-~c}->KvQ}d@+NO6LD+@S4DBj`kObDsfiXKB1&M5dJ=X9vF!H8W)N|!^y9{0xJmHW z9r}Rvxb(+GXApHE%(?@5a|5~qlUGAaa(==G4w}L;0jhK9*=Wg6o6y@xRFoV(P>}t7 zEn6UT%p8>@0tl-J2$C2wp2dI0;9A6}vZV7}P0$i-Iiq)paJ{B5PHrm07s;xew}c{m zLIqONZ7S^ZMF}TD+W+{+_Do0#wIds}#e2ZD)T08l;RG|xAx{77a6+;hv2hn&!xG_u zR%banZP8M6&vO_bd)*xG;v47Ur^C^USWUyjJ#^@i?}jK5qG>VFo(#BvJZoxBqv5Pm z@F*-s+(fqe>qhUoytis@i@8doLN9o5SjLS;v9H{rjFL)4SGF>N`Q?-*&HpTA2YW=M`k)6jO^zAK}yh6Z=0 zVivfjnnH)=j|d14$fVrLNl7xqmX zfYxAI6et)03Z@^s&?c~XKbGLv_6^9^YpqK-J69(`0o#R7u(rGe)gUayvg?-UD<}r| zuO=iiZQf3JF6}`*N>~0WcXfXJ;4XPHC(vhEp$V;&L^~C)7gxS zPHE_~e9Gkv9A+G~=^SJx@D5nDzrY*$ea)9>Q?F(XKmYj=5)K$7zK4*Xa-e|#=t*=) zF64s+wsz#f#iq!yfxIm8@JUE5;m`!H0JPi|=0|(q-XuYUg<812a{^fl_?1TaG$uA8 zQ#(2%+w}X*QwJXmiM=+EuF&~0^YbOW`r@O>PP-V)6p>3Oc#j6hntmv12-oVI*7_aE z^X%hq@ zpc##D6xbY@8Sj_zUkMAJi3nYos*)t1aGJIC>4X;gsnnSn z@`uCRq7ZFUkQb)(AR{xvE>4ffdi~*!84%s0@1k+?HmOB%BUp*=>m#WB_p) z!R_p+B+~c5YII*U!q2ArUbq$BhQS)o(_~OQvBarABkj-!5wI7)jVOlFt&+6AR|LLh zOFL)IM)Lg|#ee799D$kHX+=eEC3_FejUFw`YFlrR6Rxrr4%=JUF|Jw|I8^0?IJ29ss(C;nR z00$ZtBRi|Uv5^yhGD(@;r&@!38!pp4CjMQjz0bucKRi%MSVsy6gR;fa-r%23r-0SW zd7|~SwhVAcV@Pu!Y(k$5Gt%IUypukEY{vd*CQ4^0_avb@aFE2S5yXgU-rNuZwRZc^ zPW&r9j$}mtVTxKbj>`Y&N@_*p<|N_#XiKH#|*PQ0(vv&Pmugx(29u?i{XlD}}L31bGdmC;wV~S_)$m3#(CDaS#7M{YLe27beOXqM#YjoaWP#-}vmQ#fzc6 zVaYiX*_&TG3w1l)FvD$!!4ga#$oLqdz|BcRu+Gk9Ig3s^EC>hH`U)Eg{;{4f?iddBx7( zTbSG@h0ZvXIWy+Z&qUrq&X_AT1X;@*l{4prgTi-+!Y?^eF8o4-Bi^fz?G+dJ9kU=#p~yRnkJ2cl!U(Mq$Rp9LX}QdF z+r(^d)_i07`zfCGQ^jO}kmN{p?%;QPOJQw0Bo#~j|33twL}2U@^&OnK^i zS0sJs`;j(>HBYNa2FV&8dsE{_Oxk{W&)cQSEodjpPdy)<&wsxc&<@s)>xhr*;8-)G z^B*>;{;o>NhSBMa8wr79wimm-s*r6jYvwVc6%_n6>+Wmc$md>wGt3C6F_e#luAcI6 zaDkumV@$lDYfvIvBGUg-1%$FqStrG0l-=ymUxV;2T4QqWi=ACrn3 zZ#Yb=O$j@QIpH@roO<}d?4j>06~=RX&y)J9$tfI6=HrM9sB{MublR*D{h_16EP?N< z+!TfR@d57dvpw1Nz*O)FSCq!9e9Img(jnS~hykH)21}45PWawGx0Gq4-ZI{3i&u;s zLfv9>?)C}0k&pc1HKosq%xog#!H<>~#Af}**b*n%lE1H~vec)Vl%_uMl-wPV=N}d*K$Sy$(gMHsq zHwaEN3Zxb5?!<0V0a39Im zl<<3XIe&>qJtPu4xTbbE8;$=KaNUW`mA|lcLu!-4VOb3V^E}YOuMg+H2~4zENCtd} zy~!4sR{S(7Zhhh2P;M()2x{=nd&yxFun?}x?3I&ny z%BltZ1c=*66=TuBzbE66r{za?BRJl0;JUrl&ri3dDay}VoYBXD_HKtaO@8T*UvoC5 zWek1yDPie=d9|_#gn{Pjgw`N@O7naQQX1Q{8^4Xd)8q868-4TKeuby;jb@%XyLcUk zm$uEGk5+P(%un_;QHln}s0E$gJmqAfEw#1+AGj;Yo{n-R&_?sTt?VWI=A>%u^!IQK z9efR@)9R^V!cp6Xr0xgK7r8Th`c*N=dZ-FzfdB6_7}*_h!R-BM#T~8Lc!CD%L^=Lx z7=VfFNeBTm~j}W%b=`HY7R<@v|#D zEVjO1JuN;%x{*b^wH;ST0j82~6sqio%r8u-&0ii$f;D#y!2d~2i&Jk4M zFl$K&=_%wHwrvfvW`Keg6$x1tCxWmQ@xAfqeWY8jyxfRTP|H2Oz%-zq&c}mU#O9L$ z&Px+5JDk5a-xR@aol*ZdYT;NK&Tfcr^sHNm(>L)E6(TG?u-Pt1)Al+w=wKQS6l7J| z;V|DHsZzCo-i3C6?XVO7h94XbF}doTxN9A`X&t-95Mt-kqd>qcSZOaZWtmL~RHX9k z2d$>QChrHe?QOEVLkamj@XwbnKS#S6lIi3mIUT*MCe>=;{s^u2D?_+Nib5%*Eidh# zwdjHviQYHjlM>YvlGb)JthtO`X7yZV_ngChx(6hF=BMoF^AkzA#&HW%rTxe#i07!e zNX0fJ`*>FMY*uZ#+>zO^m40G(Nz8Xl$S?VnTSSg1>5}_+aYl61Pj#xG9T1R<*{Ack zco2fd_1MDozUcUc;+I-u!MzMx+LNC9tz@a+he66H2l>IhoRYmllx_~_M-fub;f6`q zPY@JOIMIARsNPK~a%(;yGNuM5dJxWJwron^N$$DCPm)A7d28La_B|QhMEMWwfUfBK zX(?Kcb=a3Q0Gv4O^mxXusZoI@XtHCc0z7Y;63 zIaG)Y)|hRu700L$C!VW}*2j7Eovje4AD z8g2Vz0Ekmqa^9Y?;9;l`(DRdK$=i1feyn_FPrZFCH|!}tdD+fAXtDQhj%UV1g|8Bt zjoQ!dueg?}8LzWt6A$>2bm~Y*&ImhxgO4?STkX(;lu=T@K6k!7kx$4!d~zFvJRRXo zOL1^6<{MMjr@g~WP^(}A;@5Q)DpN+jlw*@_VAR%^Q8!8P0Q6+{@87=mrnI1-iks#f zSzAvDypYy${EM-`tLB4Go|R6a4RKeqUW>c&-{yCi!jXuNkRWb1ou-~Pv8npt#o#PA z;06#AW&LC%(18=&ojiz{cl*%0 z*yFE=F}v)q%4Lp^0C@8D4pEpn%{3T|8(D^PimZ|=H;Er$WR<{~b?Rbis%e|CoPe3s z1%<|H6tX@DPH!9-kTkd|D!VOnGxs*2=g}=ue?7HM)%>J;Z$1}ChyfkX&%8mhAju!77-Cj8s3aF+qX7drJ zINrJt2djNkDxyOk(!uyqP+_$chFnpi0Za{z#>|$Q{LV`5Vaw=#E-wr3_=pSAlb9cbxq5k^v z+yA#1(r?@mL|9`NDLy}GzNY!X|9aT>N2h%lhz^cb4{oxTl?q9$cs-~v;)PxGQjbNf zRMvYbJDh)p)3Gm;>CKgcQ6*|fe&VjQO9rYVXX+!Vuv&O*Bv_ZsJ6pM>J5pJm^dV~^ zl6+!mad~=id2mkXXa46yeG`IIR(mb)XC=bV2sC2iySp+q+S*Cj=mOi~FM^Tls8{M3Y#;B|BvFKKj^gHU2`qziN0ofIN}9 zeMNe-)HGx8#b^bgfrCZL$A`KlfAFB#K7W>eV%iY8pcd4ac6k9hSTLl+aL?tj4Z?7- znq_)1vlmv5R4#wG%YN6}57BXGa45MH1O3KthHyDHL0F(8FT}m`H_UWWiN!Jhr0Bt$ z0?$f`e6u09u_vM_zGf51fO_RS6unQHTz_1YtC zKdP_t#Kvq>X>ZatOZocsbrR~hBzR;4Nz6e#z`sE3u4B;{B?*6TS3ZEU6dCu%!AgIt zywwphd*StCKIehEi_Ly<$Paj+yX57%>0(+)!n%X*MGZ!T^3tWoJi)5rK`25xxaV-z z?JB;n68T}8-LqJ6Sr^%y5*Vboy%Q*af?)ekDqZedQf@{V<~-XIfJK*E49DFp_4Mz} z+~39uR+dj$S5{O4FWDI-&P4n!6iHuT0w#naEOdFIvpK`hh4EzPL&M3BZ0@rf z>AUNf4F+b}Hnwli{*hd{j3g*o4x@cIFK>h4(4$`6;eYzWJ?edVvn5Wi_k-zMZF>(# zwxg0#>pOZz{h{b_S>-#IVEKnXKx?qK zjie}B_TBHg@634G=pP0pSioH@N*T?STYoli4K^6<7^a=fK0lnUymGHVoqVT=j0q{= z_tAGphmY@lRV)XC-VpXWkSgPzO~mzuPncb(WP?^ej@oPaU2IgBgFRfsiFEPLkqzvp z5QExz6SVtC7_ia}gxgbwV{AY`xY0zB*aXd|W(uq%7JeUh%TYP}(GC-CT~L%y$?)7T z;(yDcHq0mn;GFb+;2g;@4~1%&0XW!QC-mAj!XGhlz&TNOV<3UcvWfeLQC&7nthk5B z>e&-iA?rtEJ}|{Z>4*%o`ro@);Tn`W4QdYi#n_GDK-xOD)I1&~%oHU$ zC-@p27&e8@uOff-2lrxYUW*W*8ikR;!hu&r*x8hVmmWTPCJ8^d;%ec0|LIknz;M!w z;a)f(Yw-w7MT%*d#{23P*6PN2=F>%EZ7G!qOznMKZsE01jpzDDj-}njP1ZJU*D-V1%So=fz$!HH%~hC#mwVx5hc25#tc|UMqRsBzaw5hicjeFR}8q z{XWMfJFq7sh$3JmA)_>5?OM{p`uymr`bvPT|(GzA)o!R`t>L-K)o{NW4V{2>(h(iepi<=UPK}#V2=>*N7!MX z4xH|)IBcwFNvW0C3Ke%kUA;FIBV0psTu=c$vT5EG5k2Q)Fh1+46t+#mA^0YW~pCtJP1)|3anco6E~LXz`WT zys%VxxN+3Sy>3#SOEm*`q(uL0{uJtUld9nNX|&5mwyN2`w_f{rPA-kxqms1q!eDp$ zZFLd0BkIakbaR`US70%BcTZhHG%Lw!3x3+E&=VRuwD(Kow#JfQ?KC>josIq+xFVRl zW}PHnzFX>n>E$152+G;AMa{&n0dC&b0pzEvH1}?i>MOHju>Bk0jEoWu>`^hE!lbXq zKa778c3yoe26!o6ue!G&syD1Hov(VhrUgkC%yx3DvIg74a!#=zJnW4@<1~fTlOOKC z4z#r*1d8>Eiw*6pK0Dgbh(Du$qQ@d8%-t<8=K1n9-1l|C)0fE(fwgHIKW&(t%XX16 zyRumNatCOR@>hWwRkrAA%#wf)s>yHliB?xO)p>PoYoV3Mc zbN(B-av6x_ropmz$4^aYKhb86wX;Z?KnS!EOozfdC$z6Ky&A0Ci)W93W#^w(=Mik7= zSWX-q0c>M8Z)}rKYzxR9CEq2TLiBL>`;{_*JqC38d+CmZXIPXfUHCE%nmJ{h<5fE8 z4sr!tU*y}1(a@e83*Wy<5(YsBjo?&HUXQyGYslk|8d`6ZKO{iauz_e=%=N(kcAj?z zYD+&(er#*LF-V4%{NQC`#Bd(M*lzG{+>7>KI~;u9eBD$x`?+p0j!6+1(J7kg|J@HZ z6bNV$;(N#^|6G71nF#M|Z?~1oyL=9F&Te5lrMEbEKn0KCrEFa$kO%%{9mQ{swqojr zoFXOM5`fdA{g%Y~KjU%eYV8-mziNYH4_&rR*!O$@&jWtzlZy)nkM7gZKuW`IUmCUL z(8{rHF76kxR`j_Gst%k&*d%)y;Q0<1Gws}Mi}7-{+~edlXHl*C5&cF&r~U<}fj=c0 zqngA`+9X-2G%}L1cG^X&5C2n>n%MXGw6&ymq%mG93{>l4#6lc1zf(`_dYU&s{>iyp zvb_DHhw`l@YLt%kcGA3mWRmhxe45-$@KKQ9q~X1L$Vb~T@s9QcjGb_}LJMfGS+B%a zI+eYW^S&eX;#{op1I@*Vmyi6fh|7{y=`?mvX|y-x5e7 z{!H$I5_ma9p%&WvjTJ}Whsb+>=v|MW3_WPyJ zjX^nX$Rm42`5}MbQBAWtLPr0=1UT}vf_q?s!X7&vx}td>%62p-cRS{LVR?|{Cxwgo zbmAcy+LCtfU-QqF@b|jaPhfMInhOgH$r2?J4&F?#S_^xp0U}*inE(jyWT3|zD=P!{ zZ)ta-tp13qb&NaD?xI)x2x$Q0uGcuF*Z6gv=mj(4KuT`f&8OpYQ%z!BQ(RqJ`VURF zKU_~Lv|KX!-=A^wX-4q55a{Y$Bk~WW6@N|0@lQ(0PI5jQgbgMNy=o1o5hI+ep zEFF9eNw@*4CqzMBaG`{LUHrt<44uyiDDIPV?(2kPONO$UVAV)v9BXBLY=Y3e)sG5p z&+=VIf!PL{i!De5jP4`URm*RyH?j7{>C!+_Sq(xR?S zw1epveu{8}1F0{3f@(s4A!XLdv3OTtYJJ>V_#?*7WJa~t2sOiFi{I=VT8(8A13EM( z1PsbTQ7DybDDSNU>EwKPjMwv8E`H5?Dl!(xP15HOtUNINR8d)2Z@n8B2*l=~C<)GJ zQIu0$XhJiSC_C)Ix~*--b6Z+lv%KfV^#5gNsRYS6WCJz|SZPs}fC{V(YjuY*NNEp$J8ko7ajj-fJ%)dur_`sQ%Khq&cz9Ood? zZnjrWKPrs2gc#*Urr5szx|(v-&`N#!Aso+KSv_6vCEBMJKC{N(4Lk&X5qHtj2J^!I zs|V(ZJh-%V|1|@u<2TGZjyMcdZztB}QMhbE_Ye5cvM&i_YaOgJ7go~G9`}tqMtqf> zVSBy*jbQQ4dTWA^%u}dj#Zvv&iRWhh<#2@L7Q&2a#nKsluNtxBP4;$NtYaZbq^M0& zvr7>X?3c1`AV?0W{EF#{I&&32y8e_9&1z0L$NblFq^&L&yTCQv6rgqrV>sUcZF)i3 zHehSep8)cWqr%E7j1S^>e4o&Z1Ep7y_zdr8s$`V)cE!~r!L@_u5JDg@%vc!(oV6@W zb53gnjfF1sv^K=vU20DskY>5u2~jk?mzvEM6+p}sW*TKFK1~VRcLM>958E$Z1f_Ia z2>FaLa19;YB)Ka^g8gayo*bYNJn!aALQ{{@HmhhI%J6PBxk~N&HS}VW^0VnYKc~i| ze|hs62Q$%_5_7tps$X!55lsz9|CRfie$F~7jUhe9v)mctDy1~vhj}M*U$NaN&su)n zERHxXD@2UOJ~4*NTndI@+y}?J+;CuG=@V>e z4C?@u3n#?D-jiKw!vU1I+LQ(SxRs{LE%FE(0mg3Jml{ebXh@Z7`4ytb9tzx}K zEDDl%_Ms4W4p_%(LL?|hZ4FlI@VeGmpa;eY@x_TOOe}i0TS>%t@h|~|PWnu{nax-u zh|~E46sA!(`803)TT0+7tQlkA97B-g=er*?!tkuio`!6+_fju|kGjZVBIpO+mf6L> z+N58UFXvc^UoO((u&dw$)TE0~cbm9Q{3C!*!D}#Y;SHF_SJ<;wzpTahH6$xFCI|-; zFVLa$Js1b;q=ri3hIM28Ux{*9!Jm6j3E+G>MV5@#;8#WX=K$XoY)4^W8^XMQd1>py zdgtT%r=!|x6-t*G9UT3tktet+O3Bc?H7KXqbMYK9jCT%N+5EOevfSkOra&~8|O2-JXuj3GIL zO^EiKzUcASM?z5XN~)tuqB}mo^0ORz^Xoe%Y)D%jLc;!#ueCt|$D$jyi4;Nep`nw+ zTa2T}YN)|aAUO0a>W3p03R%J5)u~k4DplVlP@+z2Zm}QF8-1?G3GQY9p+v{7MNIv1 z7)XdoAEmeK##?_@Wo+%{5$%5>+Ak};pq0Is$>u2<_hTfcmLYONbJy}~l&&6eb}rpx z>Uz){oZJ6BLAS$KaJkrIQDk1ZYbO%FSKU;TBP<%l1*hXHcZ!7zNPngaM>TTKUXYG` zgnVGdw6CJtWlQ8qANesFcG+rnNlu`aEC_6`l9fEULl$*!8&fbAwA<&VeWLQ@$bIg= znGXIg)Hl&iC6J7?W8pGvOG2O@|Iv=oA|ojoH2Z0w=a+SPaq{u8haaIqLuQ0zN?AI$ zADE8@CAXrJaSzoPQxxRY0(>(GRD;|S?d0t~|R9*>f- z0XQdWB=GeUoUOrdKlFnx0cxrq*!MJ!eoo2X3_Z|A0g+w-so>W>=#tEsU+?h&+Raj4 zK3zS%>?;D`tA2p2rYn|j;!+sv$Do^Cwo_dp^>^u~+~58VIn6toP#<^xWL$+sH}t%c zfH^38Mb=p%aK_bV$RAC0uwWyH40McBrP=@%CwY>U^Krj+A zG|fq2Gb0{Mb6*e>LI372IdV6Ki*+5hMi~B|V6Lu?+`r2C_lcYruyF7R0q^b`P3y$B zGBWBaD9Wsx*Yp=s9Alju?mv~@yS`xQ67lcLq6kSM>Qz1!aJ>b=OXZ(;8 zyD9Kor7g(tqszCUVxgsCg~3jv<5FQ%Ise(4=RwU>!tj;00_?}N1>s;#90s4Lr*@S7 zd{!LVqPkd~zw7bsy>LvyP2MZ-rT>&7A`AclSaqcxn9Ea5OAk9^J-diGWN9xYx<=$u zNaQ^q`h)^Qh<=K){eH@}wJ~EA#r8x?(j zNQ$^&n*gd8)kcBi(idyca9A`53|K2;6J)KaZd}&#Jt5WR@c&-~CZfqDB>#12g zJw_q5kIMM3Od0)OVQEQs6=%RoJR~*tt$fLaGJ_48`-YiErNcfZCGA7gxlZ5KaK~k} z^t*vdrn}KcmL){)uh3*FdlHAZzp!bvoYbiN$Ocd^I2o))O{?O7*#*m65Xcw=j=>CG z8$SNZKlkn}Z$NQspz>2CYnON5qAqImrx+C<+TBUxmDJp6*9&w8ITL`M169_Qbjn1t(vo%p|Nwi4u30dMc>9a%o4pi3sEV{O}zWN-Gp1cAXt zuxzF^HyREqT%Uea2}VO%H&k1KV0_odyerKW{_~8rt0xW=z81%>Th--_OhVaK2zL{q z`f)u!s#2L!9{JL4@iD`PK45ITxYYpqBn-Tj5#Kp*4we6Fk-kF)6Q#h?+L!jLsr>Pz zXH()4c*C^=Q%#`n0o=NQi*&bMANRT;)9fSojp&FkmJe{>5|<<48x!8Hn$13P<4}ha z{6mPAhPKBAyKvuNou@pPv|Qc#aBeFiMl%SRM6Z9_9+pA>>V2!T| zgBXgHJ{#rAJY`U_cp&qP$u4JOH+k!TYS)Ev(}nE-iYKIuz$JQWC3Z7z-!r2kdjhCF>Rl|OS*!he=Nm^le{$h6)xk|(s zXEH8m_Q{G0D7d?e9jlCemRCqBk1ju)W|peqrb9!^gz>$iPcbLuly9ra;cJlphH2-L z7f_5DrX<0>U$CVFvQ3qyaxi>Cpi^5*N`B>-eBhRP66lAa|C3Ey0o0KG`H=Q06;A*W z4jO)6rxWUnO8Qv4O200>+N$>zy-6)`Q5;>4CDYWwVz?WZSL8i)8AOl`3eGZ2TIvIzzYohiPV#>dz!q!i$NT3d(O)q~EH@c_N2w zZ6V~ZzZ-xg=lW^;5i-{I8nk96?3RY?#?^@;f}aG06CVi>nec~i?Zx;IX?|ukN~Se> zM9%*uGF|(IMXQ`QhjL20_xsoNrq=b#R_d|1%*0+bIP}SW7R4(VlECAR@bX2;rRi5+ zv(gVHB|av+zNQa<(C(M(zOK;QFE{x2TKB+>f$AKQg~yf^ag^oVRvTH+Gu~*y_6zSw zs{s4sXV+Zh%1+DiK;~RqOzHk)TU^2Z8OS3G`UH3DS@LQ{6MW`3w+WV_2r3^Ap! z+H18M99v^_C_HlU+_8;6GI7{v3G<>NJYDOV)(%hXaw>j9Z{{!X6WSMs3eO@f^fO~+ zhGeEEbk`j6H=I(o+?Y>P8YBMlski9GzRO|Qky9L=Rp{;KnhCIvIWOTTi|x_hp{{l& zW^c4oY7$G7Fm|lf)?YDHQz|M%OxKIdFxh%aU>X-t4GefO=TJ9}kaEcmBW-zvh9Zw+ zuC}-=_hZEjrw@h$p~#qkXHRchzT9eYZ#Fm&xOiMlw3S2rjH`WEtX!pKmQ-aIdS!k) z`c#GmbmV_%TS2Lc|9hRW&UgQ!mUxII^rZ@{CGlp7;Pxb^JgGSq_@fe9AAjaUDc3}@ z925{EtKcX>f`cL!fB2V5KBqid*?c6(cloUUcdP+LHqaLB=P7@(bh(45uEp5pZ?`I~ z9M8EO98~VSc5b9ws5pN>Z_h|{`J@lbug~lzl7%b%U#I5s(cmsnWpw*O_=s z(!6wWS)}QQxO7NXD{rJC@KAn&>4jae*wH$pS4)}Dsv4iE;n(qbW6pze7AIHJ)e~i* zt?6e++9X>V$T$C82=cF*YMxPNgEsq{e`hq>nUKd4ZAkj6k*?;f(`klVgC%C@Nda{E zEj|M1TEaKO0!pJnq4-sJI^F0r9XfmK{4` z@A5=RtBD70qwF5e#FpMaj{??U0cI3vTgA*Ows9OFr+*T6OZEO_>OF>FrKHtv&aiRL znoJn}Vn{&IHCCv8KDT#lt%cK%B zKIk88Ls&U$+_b1EJvCaj{E|moSc&unuP81sQW9lk&zSkVo0VH63Ucg1c~yTWcOJOp z-Dde%Q7^N9$);^TGp{M3{}x#+M!HvNFBDup{xAdYa@O(Mv|&y5c^1 zFLIzSMi^$i)LCU5KHY@Ez-sV8V{LW#hzSYSPo@qyO2)k+>0U1vGe8MP2#j$8_6)DESnBDNnf-yM(B#Shf}=9G^0M0^ggzF#Uvn25rG-0 z3MaX9{p{ViC|GTK0bSYmod?_#^40*i$55P-?viixpqn%-q`6sNt`K{Yd-;LZvODnj z5*YZVj{-W+5a&|iKv`~63Bcxe06U|veK|~UbU+9G0~qD249yb1$9Ekdp(_T9;h2g2 zOu5)qVdVJtEB(6C)VFNEop!8~_{Ylj_zJx#dbzdt4|Rq6r{~g9I_`Y~!`vU^i*HFB z+FI(W4l_$T_C7AlaDi!F*QdGHZ+=HPuRCoo#5yg+ZZC|udpYFDsPugeyA2$EA?;fR zVxu({;di5S+e_LD&zC5I=J3Ow6qb#)EW)hf*jd%*7bja^`+Se|Y7<(s@?C41gGUqM zv9mo*8sRs1jcSU3)jbC`lnvS@Paf>efAHyxt6svbu?Vlp#I0fZN0~WUEOntF8u&W( zY~w+ouVP>j>Ct^&W75^++v8^th%(}VI)eQvBCjA=*5Lk@?0(4`XX_LUT&tsj5Gqd` z^BA~BQDG^#>i(j;{-j3aZhtD(V0@qx#&z`>?>*oROz}wz2RX`JCGa^RCEbdp#)QY7 zKLsmw;{#8bSO4bE{vCgKzdD~siXDsxf*QbhZ|r;?C#3_OSjfKpWwUrUz^2D6>??Rs z@gM1}yB?H(Gq*9LUwzJMpr)*VMm)eKV+YOp$et);L0^!dnzLRF-6SV>HQk1XeS8Mi zH5PRfZHKvh1Q-CF4M%#GDkAf)hD8?JgTA4eq8)G$`9+~0vTyrkgGIYt%Kjk`#J3kd zdq>9W5Cl=Duh_+=F(+a)IyjgY-O#XJfPGv4A;LPLB|vK9Xfs9hD9PX13=L`isnDjhAGXjkk|wBs zmzLOuYWB-`b)!F)di0$qr{sxyWSw!Jj&-M?E*tUUm&o63m0@w8|Mt5@0%p=@roy+# zR#Y{hBh~}le;oz-f&1BnFct!|?PNFCJ#;RWoSNfkD8<)VFv#l#o=`b(SuB+Hh!x{$ zR+snkhrvYNq2p-?S4PQ3b+Ko;A1WXx^+Iz@{)# zw%eUpz7P26u0qe&U|(?{gk_T$D}Yu=KOv0#k4_0n@yH?iB-CQQzUMAVx%v09P zyn6LaU3p&hJ}DZ28t1hVwQ|wbkbbvMrH0$K&otbYRx>LU*> z-?}>>4RQ~-?%I!{gNH+&4#5h(NKZY7FGUmN*TxFIT_3h2<>mc8FerGUuFQ;vfB@Qb z`3IjlU}X&h+usS7k3R<8dy3DeVF_0-YSuk(aHvdL# z`fuR-1^4HH6t6T8-e)L3thsiMuPWz-l<+MiDTGY11Sq>aqHUYg3K2xZre>6!-r^0> zp)7J3EkRtka`Wi+(;d-`JW+P++6FSj5bNIg2M|BLwMDO%?}P!qBqFYmtr|0i{jI1! zfPKEr&)ImA#$Hud91dK%KC2rlOWDjS_Ms`tN-4_HmXB!z8Z@O*vED_IzC}(b(YGHY zf{my)AY>nW39~H2B`*u7;%a}_qi{l98KoOP)6R`1z8(G)^`U@Mx|ElXz>{J8Q_ZkW z`OeSzcV8s7qbbrOAGy2ce^@M48eZY_u_v>pDt0%zD9^pk48P4xIJ4y6PBFZ&6Kz{g z&`&S=yWnamx0s!H>{1vJ>{FWm;TyU2NwI=LI6Hme>Z3_di9JKMK;`Ia*GVwBTv;K) zkTBnnAlGD4L|~F!IB+zM-LIfbac_ZGK_;Be-+@-C9({DQmzTHKX}#AMf-e2Or16d_ zSz~VVT(jmG=0JT52zZnl1G2(IEQhvq9p(s*6rho^kn6eMe|E126q`JLoy-Y`yi?eT z6!gvD575tg%~&S*k-7N3vT!9AdwjT}@9gRF!N5i8AE07?mZS(B;@p}b0OF(18SX{* ze)XIqepFU}Nw^xgC~c!F$5ae|NlC`m)L7t9gRq_++UNlEYDU){jQV{)lg+mLmxxt# zjJe$VFM3~;3U&XMSihhAao8b!RTw+DIswKy;az6kO_DC1sysM3RDrM;Ez5lrb;4fI zwTTA?TmZU0J!Gm+N*>h7`>_o>r(CMT*3iJVL_>L;1|IMoy?$4uzRQ)dY0JmBM-;w_|G3?DRp zL^{CY`8zinx-0bY>eZBu%&Xl9jG|OmpOxzq`f$MaHuw?tM}IC6>pgM(=JFB4@doR{u(i=$@Dr@tq)k9q~|qB@@QZifN=4mb9} zj7*fF+OfA#zXHK#B&~)r%u*jhjT15^+o`vp_^EOw(Aj{kxPq6|2t=r&Hz&%mg#ZiP zLauRt7IDCX*~om`3%U7d$JYUx`u@D3En;FP6?J2&C!U`UwGdC$KtkA}BAjg6Ab@mdK39 zt|HpoH~dWl$Hmd~b|}k+!rjW!6w-);0r3i-ykXw<3>}$vNV^G`z6! zL=GLD+BOl*7t9=wh##yGUeiVTZEG}o73Z8gu*kf>%-#0AIrUwp_g&`E>lmsoqI%^7 zZ5Dn5sMNq`%Nvf59&&*G+*& zKE#KT1UXv`veRaolySh7OnCb3zx$ft;@9*Dp9*^_KIp_U)zn&+tU=3mYJ{BncZ)PMA ztS+_=RVv@Np!FVXBXmc=${1-h!o^wPvxDlHqtfB2fza>xT@Zgd`oDY>9j-q&>v;G^ zOLzzc<}iSp^meUUPpD9JrNT6Dp!+6~XHPfD916jEy!KsL7byTc7I(L!cC}%4HPCf| zDb`|+7BUdjj(51j-C2P0~EJ_PtN$0dUd~=Eu4C=jBnRHX=ZieE2*BZ&emj78%bnkV(YGBN~2%qPbfJxVsp@EbJSbbP5Eow z-Cl@Rd&d+Gx`kaRyReTS0qe#<#Fm{ zrzjjpjylj&YE@mEo(#lPBIzRV5#PF8|L7bvTp6dLbd>iva`IAo8_?|k-HbPy)}g!> z(uw<(TI+QYQ}g};x~ye&n-T;+e}u+3B9FucWutR(V={}NZicDan-Hw!NE0lrF6yx7-c9Sxg{baPwqxq*nN?%O{DqA9#=SL2zr%~upkoa-h z9i*Q!m@Zz?Gv<}DJIn%W+UdDR)K{Lob#FX+fe>F>Hn6Loi%OuHcA&^>;Dc}F)ssW( z1^M?Q@*N(h?Q!vkqZ$Sk5!3a4kvr0OH67S+Wmosd_?Es?Hm_~0GWs%)wu%?_qhoYC z0B{8zopG%NQs%xOF(9ccvZrierx32Amq3rf^Pe+3tNK`~&|n<{+O%#>Y6jx@NPq}} z2NC`V{xkDYuE&BL>5BCkwD~#1qgl0M)eVRDZ`-eM5g+pibP&$Jn)ALv;9J*jud{@{yF$E%2Q^YR3oQzj)av>3DJfade~gZlbSKl4O-a#6J|i_84Lxdw{>K75m%%(0!tD(Fg?usSFB&)zFSPnn+`Lat7jo_?4s>ydF^^NF2qlB5`|yW@O2 zLwqBY)&}!x)QVdBfAAq49*$6a%bxu#Ou9B)bhd|1%PqrNFe+#e2j9b9RR!WA-3no*FoLz z#n5p?x8>nBRVwIh8$3k7IbPH$lzosxQ6P?Q=iwb@oLP|gpYdRlAZ+`xc!8O5+LLLj zJ`T|J*)1sDl);Rr5ee021IXtlGl9>Cm?9DpwQmeR;)2Qc0Hj0UE;YI4ZlU{ZvHw{= z8ZqoA_7lhm@{h?UvpyVrrL#QdW_^UAvh`eI%}T9Q^H4ySi~GE$ypiNk23`xw?TWt} zQ@wd~9$S5y@e7~9l%JX0PF`jG$04D;zzmB2S8O;PVzBK^!AU&g%>LwlX)EtFN_R)| z$!euizQE(rQKC!D_Wn|uC6r~#o`rFv)N~Mjy~L+}0bI{goPfP0Kav6CJ;3^U#Pb%v z7O?cXF>Uz{>f}DVvbs@JJCXG3BhwnIa!ygpI7=>Bm94C{tmG;@9LmR4 zF)e8naBpOD|H)n0!Cu(#uB+NgXcNHY1*4jvBqKu%AM}d=@v-`!a1Ol-v{gV8asL3$ z`_8AACG=^SP>^2%h;M56Mzfqqn_6`aeFQDpngM+PRVO-CKv=Zn1$p)T<>rg3+iSoW z(b;jubFqy5=hyr2Fk9g-c)m4%d$bP&X%Qn4TXeNBU>SIq)X}rGYZ&lB7OlWr;f-XU zu-p~a2s1W1O#S3nBxIt0+FI~cQyWz5OivC>*omLhAk<#1u--Zq!TI*Sn$682$E;>4 z_5eM}m3Ta8SIx)`XJj#@c2leG_MMD+)_Av#{@K1uaHL{CD?<*r^Y!6zC3gTioS}|YoCRm(};>1%hq7u`2^-hoSk0Hs5KE+=hie*iTHBFNF zK7w;@Jffv<8I&^c$>8J6nO|~^l|-jC1iy8uezs&c*G9C9$JcYwI4Kl7GO(2(MNw+( zkW^%c2uhDd!f%E)ar8+m}@>uZy{TN2wFFr2DosE%fgxH8N0Cw|Q>Enqr3=D9fJ zpE{IMpO)kt6`D}X>B`M`zP^=yajN@ejFElC(qQHMHW@@wo`A0A$2P5!pSD!c(!60^ z452Ox^bPq=(AY_F*S|OdsToGx^wiVU41)cbFRyVFTlTisMVq&ZQ8^IJ0fXniSpTF) zfE-MHp?m?c?*`e7%Be;Zad%^HECHOyt6_|`LR58Gx9^yuGTwXBuU?sj0v|$Q1=7vl z5!a^A>r1^%&+F<*pN!iI>iv6x)`wb_U*KQNje;y+8%wq})zlsffUwZtz)R`o^fG|7 zkF)yCJg4!lMp4adk)sF;Z5JN4aU6K_Ni7(uyy?Oh9ieZAR6fMLuuY3Rk^ZhHK*TsN5is;OUN z`VZahLI4po)6kbF3T*pFRSBLgjGjpzot#h1FKD^(*RhFSUNoKSbduI#n)dTR>7<7a z-wFr+>PLaBXnsl@0ZQ*zML-r#Lk~a?+`YtjfL4Yq0B6)bVBte>=-XAMNn}7gS+odXRmpy3D7guQ+|vHVU_ep$ z^@RP)$LF{+tcTKcGrI*T-IH9k1E<8O^M$-58@M**xZX9#CU-K~g@5PR6DAfQ`2f~H zsKbE{;JO#EQG3k>e)4{C3FrBjZY_b1Xz*Vm$HDNht^NYMw9?*ZMt7dpe4aX1vTulfEgur_N#d6J5TCp$qSo#O)!{EV&d; zg0x&u-yz`uNm)ncvi4vQSzekE(Wic7Aat=8UIdrE)kBc_AiW0ublVHVn_e@p$WdO@ zn7_N|ou}>2hyH%2=Saow7L(i>wrk=B8?fSfSl_Qj0f>pz71hJsmRi?5@09~7kvbZP zeu}zY5+FpVj(Y(N;~eRh4e%EPlmeL^EX7eir^KN$$KFsh##!?-irsPRV`<|YWA)Wo z#dFlkiv(y1QTl8nihKK4UXUmZGS?Igu*RGg!4yw<>JNZHk|1ugu{Hif`&X% zw%&*S>|vVlin<=Mo)8+lIc;AEEqTzoV@{L!K@cl1DMos^U^KG+3lr4EM>sYJs+9*D zgd=$60~{0Y9Kai%F7#-{u+VeUae&QX5OX`wMik@3r!nz$L~h|lpO<%DS9=0aQeTa-MU z9wOKDtL5^%+Uc-BzZVSjl;rJ8FQIMO56iHDtn0{gY9((waG#d=%fOD;hry#o*HaHP z-(z0sS2#JQ81xVl)b|36p`EkHxu095-VRXFY+--DdW}LhjQ9;l`;SU@;I`)1E|?7s zLWRYzt7};5O6)&)GN*zL|M3ZM%4KJK%Hh{Bhv_Ro_RK?R@OJqHoQKL%`qCw#NPL&0 zbYav`SrOCm#r4^)?z0;-Hwr$fJVLjRf@q9;ykR+VUp=fnDd_BuzxyTq*b)BboTkLz z6&J47;fXR0?g1bW{HH6&fe?7@KR1G5a*B!o5G$Hx@E5S!R(JE|Q+K7{`00VaV#JF@c!okr6-F&fJG@A-i@&PAx+;?VF9W8vd9g?+OnX@{B zj$D?4q}D=YRv`e@i-;{q@Uv;A;v?PI)D!?9=y(d~)-qStqG4P+M%VlqRaB0{!uUnN| z+-Q$5#anuCQ=s_+W7jdbn2!!laJ97DhGE;=H3-@S<~n##$o4=x7q`|M@|0K?NZe{6}#x{{@YCCE+jAa`df@yMw0Lw@|%C)?|j7k zorrknW?J);)vqc`Xl_&wHzasZIS{`p&MJnppCni&T2-6{&=CW2B&!AeUyra;HTh%eT z8VGqqYf5#QhDS8LPX652JVNgEO%j@efPd}$yt7<=19dI{>?EEqg^=S_wdg2Ijc(Pa z^k>~9#EN9^M`eD+VZ_{PP}6*}6foAIqA$Ze;v#V38SKH-uug9lThMv6!6id%{kfiS zk|Zy~NH8n1Br(C|#TPkxE<8mdYqVLkH?9=3a)d8ufAC{vF#O=cZzDcL+?atFXOgHk z_GYSKf{fuMQy(aW7ki3Vu~9ldAkQ;5dRi~w?JjUxCp(cQKUKczaCqZ)T#iX)CvR{# zY-nfei^7A4A<0w#=*alYMdiL?O?~0?p#@)aMBo(KO?bCfeDL3v5yBqj0l2IT4vjOlAX z>WV<|iU#VC{jc`Trl%1uHEPwBibw!zQwqn(Ca{)c%9i6)=_{ste{Y3=_-ITjg=?)GCJDQ5I9KkmFtsiPmY$fY4(ff$9IyOm<5D8KrN<4FR*SJ{j~S- z&r(|2`()n4K^I84Ua_KrHw=^ z_16t#tuw2``^kd?H~Y0VU1+fb%yl3BgQGg`ll+Wb@_=yyah*T5%6FslF$J81dNm3O zJS06L4Q53Uj)N(>lZ~4u*LJGk#<$sVnJ3L-Cwu&I?tF#igz{31vnUDube*v13-CU$ z2Yx60s0I)S@zbNotz>L~HAJpWfGq1}omcZUBVQ}(yH@^u0BYU7EVa|4{}b{a*!>7Z z{CE`7ySJ~}b``A1x^%wraN?LI1k}K2IEk5nY`T;u=D{%i_ zu4fv-DvXlANS}>z#$VYnvBfSVt~7CH(NH~L!6j@MrR-g;@ng0}M%uML=c7&aa@eia)vb_-x?LL@4$X-3~zzJpAHHRd~eow?g;+ZlzdMqXhF% zx8}H6MnLO?gwBgSNV4_>d&N?PgYdx-`7xLQr2TcDm-;K z)os%^xWoA^h!4)gJgYF~S^qJ#V1>St|0BEd+{cE-Kb+hR3Q9Rw*e&&Ar4^Tz4y=v0 z5AQ@SKy{;DALBC?cqZV;(7KdlfT_2!%Wv!#Is)>gNxB#8@(jILo;_$0d2sX$S~n3t z>k*(cLy=sy_fKW7ZF@OEK-fYN3Me5(*DO?Uv>b9cKC(wQqrhEswDlHJbgWbKP0UHs zLI9^Nl=x(c(y`?NwRIQOTrW;2AJP3&+d-+Y+`9&$t74kE+OwztckqoNX8*bv34AEH zXXy^TyPb6hMuQCeCc9{vY-C;9?$V7{)ug>CsQl_NM(M4f7XH=jnj-VCw6z2Erdqn- zF|YS1@2tM(Z+*gS6tA7n&t}0HBjqMC26BS7<&jCLGwqQkl;wg1ZVI+Ctg*lZ015k` z5bcIhFlOcrX(VdGJ6t=91j$@yWgNk(l5&AEb_|uNqDcKuioz+i>W7#%%I2T}Atmz^6APt5(B0)NGljtW}}@5d!&m3m%)n*AKptiav{4|Q9#wHXl#;h0iI{IY?N=$>tVWhn~zXCYZ>F0 zvQkc-@6y3;5A$>Be`ee3J1n(Ly)YjBwcxM++w9kGsnpNuuLn~=l3R+RV>6L+Ygad- zPDjG-FI#UqU6y0Ze-G~!AbxV+jCvx|1GdfTx$BAno~9dAn0797FZ|EFXE0e&)E4VN8Dk5V&(eR zThX^D@BVzix^~|XJ>66m6D7uPUjKo4b-!};?w8R8w0{BvW{BHH56Pp8T0e-Z^{T%Y5bymN37w3 zqIkA5>kUh*_E&|*`XO;u`OUakD4((Eu^x}y&=S~Ybcau>Z3s6|Ln6F-H-L8uV6Vl# zY`R7QT%BElr3%DI4ORX@*q64v)Vl$}uE^wZi~)2wafy}|LjRLUNQY7GCp07JZybpj~U&bi>$ zCc((M&6~l!rLGzKv`IJFeKabd*Rt$P7%InzeUB4)Pq}J@#9nvqHTC*Q{`&mU| z+W;{md$R3NfM{le!%lzHo<;g>OWlae7jCToo5mr*_i-3dKc<~(RO#At0&i=A+pwI2 zWvAp)k$g~XK0FY6?P8llbjo0sk7k9j*pae0_I@_TN$`W^hJt60*(+mpGA4GsRC08C zm~>_c5+l{CF9~eK8CpNw*yiLIeCR0)>5&UUV193IoUyyYLwwmjCxpm-Rv?U!>rZ%_ z${t&)m}s{pIHo7&#F-?SsYq@$J4_+yMy5R_MajZnpCC%5FYrSGb(`MRmcf)P(<&d& zXl&}s%Isbc&ht&UbuuV(7I>H$cPKmm8!ss-+vU!GF)on5ZWIB(yGlH-8320;!N!%? zTBcdL!Tgtt{gKlrA0M|<8^iZCdHjg)m+OV?bbVwwZbLX<vwIwXC z_ORuzK7@q92p)iSGT;X21MFgNM{k=VJ%vEfJEsaYrfA@AjYk4V6pjqregtDZnevC| z2C|2VX7YX4&z7t)YLkX^4pV~8O#+?x&t`>Y)f^-I?X^bMg)yaefxhQP-MlLh_07~Q z5+{)Knl1+$abdMFnRH`ViXtic0XX4Q?Q!7=3ji~Rc$3y|fQ{g$anD=tPfi>7T2zMv zz3oTEhtoPeJw~a*EltHKU@Q4QTjIY#{odhrZ&VAILrh}eclN*IAGna5kuT^Ebosn9 z4sNXz=pm5!HXrt_9L9W}a~wY#zk#GL4Hiax2jM)d11wodLu!_7ml>_;Mrp7o=0CtSa z*9{H)g@p{yUef`%kKwOQpQg(DpDfDz1X_?K|Hj|0YqVci3cG!OlNf@=T-f9j`(q#g{I& zE&B3aQ=F?hN9&i9`jX2p5l0~*r{B?nXLeLFUbc z_YH7(0W5_85~=L}BIVu~CXQ=e1b6!WrN_C-m7)^MJ_4((>CoXPAatwane)NKDLFM` zVDPJAe)t6-Y`G_GxT1d_g;~1w6A+Q_Y880gMg37r7w5lkHGp~DOPBise&De&3EPNQ zMGx@iKiG@RBZ!+@mz4FpMmlYhXe4i$2i)j(r;lg1-5=1tQ#pJ5`iAdNwmR6D@jA0o zM1576Shuk5%gSdWl>q|6YZPXfbcInx;s@;-xg)D0gQy)M1&WC8WqJ>55x=Bg=X+?V z6Oi5_2bb}n;mH;Bz7_1u=FgU;`B;cwRd8Edf?qWMo&O-j<7kAP$so+CGQmPmE7q1f753{`ua&>r_$D2nu0BPshIyZcuyG(nr#r;_MbZL$- z3%s)TLqfQqOXd>5cRXfFOHrY3=^hu0XU@69lMukA0SOXach}4N#jtd7+d}6_d9e3O zNGRqW_}1)X{Hf4%h3;|=@l%)Jd$%XkPR{m)b-$!2`Q_r^g-flRKOnV3K!Bn_51iL< zJ@1VrPV;bg1qh!#ZxdNv0t*2e@wJgY9_)JsJeNg`=J9-PU0;jU1_c#=W4~4t)MTRO zrlH|{{mgT)f!F-yG4r^NIVSs%5jyU4HuWd`)0 z0KkZ%V7_i+x@7I*d&-Qw`yBl-T*^i`^{wwI86pW z(gk6qySwo^-%-B?*lf>=7%|7!f5?o0(vqS5BaIYal*LkWrAo(-P0>k;@W|sC_Y<+^(4LyXCmL$|I2Am%1cyXhs-k7z7fL)6!MPC=D2rG)22A|; zOsKe@v4_MOaaRvOTpaQ&OD+=tW|Q7IhF@P!kWWVc#+9y}r{-kJT3k}&G0t=^2F zykb^)r>~S>(WO4GUv$nUe48e{d`|Xhqu$g=oTJvX*8X>q(;s_#jO%q5hvjX@_2Z)l zFTfHe{*H759NzH~?&7tWkOg+zbt!L7ube2cf0u;3OaTdl6b}dgXjJtWe`(2FbwA>sigGA6Hfa6#fq-CwcdSa#^ukF) zGfJT0YHL?O-z?wmQRdfMl%zQ4KA0YY5RFNeP|$DU7D*NfSrvIka4gO~2l!CL8NmMUCokIXR;O@+&TjT&-;J>ao8TBP%Xy;-`wOd~G{k4h29x z+Yu~B$>b0D)zM~xh#DHyoqglk=SbeROhj!Slvx_g92CEQ*rGb)LQC+l@p(fW6JrIniK7AL*?VDWXtr z6H1!o`h@IGFF{v)8Hsyu9WiH+48%0|3@9#C7>2Mw`Bu`eO`9LOv$Kb&BTx_l$a&lsj zXxhWsXi*wI3n&15v`v*rm-+GSu1V8yV!KY)9P=m|Na3G)A88`%N*d6=LU$v@lE+a1 zqS1Z>7xL*uQ(JIb4_*R>pg%%#k92fTe-h^OAXr4M_Y>ho&3g+|&4OD^F282hMhQ&r z)JEuPVbs>hV&)|u63?_Yx4^Yrfq#kbdpu5DmOeaNyYB#wIN+~-7dZEr30L5IJ^B&4 z3Xcs-<%8;Z@1*lmTN&3&FdzJQ+!%Cazi9;DWY9W&-ad2@Z?;UyZgN?YGj_LAUoJMinOJP);pT#!^wqE0rfyi? z;Qz}u;#-GKvuaiG0ajvn(xxY`Z0Cp^=2R_@#GJmH=2)igb9_J_7*Dp?G>iOM>s^%v z`lV0*d7L$#@vORkf5JnDSA6W1Q}II>&j_^IN@#>G3v6DIv}`!3_b0BA0vYWP=?=We z-Un3r9bdrqkv>(2==ckAdn@zxXyOSpS>h%|e0W?BexCNI%#bw3UjT}Le1z!*G*sev zjPdm@ff;PIK^kFV7`K{;deuTjAt%!;tu_WJUbG1XNHZPFfYCzQz&mJUHO%LsAKHRo zZBh$#ec=6lhv1)spN;ZQ%Vzj>l3YA0Np!!gCEsSe@Qp%-L=$-Dgwe_e4^J^(8nn?W zbq^09ce2E-CdhBi?TUJ|39B@J7!>4qMJAg2J)?*-G`*Nuposya5kiPrLG;}s{WX=D z!ehPc43UR*%zMJdVjfz4(LV#!&SPYP8{#3D4qqZdTixaCzYHJ!F2HE6imY_xK!Q1R zVB<>oyMrSGS|7F%zy*0{YRQL< zG-oXusej!MuQV=UFRe?Yen_wenIjTE0IPT@gvj*Z%9^Xk09@R7is-n?r2R&C6%1V7dF&QLnA+SoOo%!Wj zkhBr}q`Z;SfZS7RTL#I^2Y>AClR9#ZSr%*zqNpw^se zj!`cW`%xWSz*!i9eZZK#(c3vgWb>0=K;b99dIE`HC#NfY)GzXn$M^*vbG@*#jF8MX zWf@Th^e8BYIlaNv`N_?^AN`n$8Oc1k3cs?dcO5+*h2F_I$1AtE#7_chv|Q)BNI@~%-A!c!JlPY#w}tT<%Nc68W5sPPaM zlJM2&x(xtQGX!``96mXWnworkxH0N`D%~0-njy|gq|MNG`=eYVrOv20MHz2IrvG5WHQ+P9IMcroJjy?_bTxRd#h!GC?r z3Fgv@9%cCs@uqnG|E5V13Ap%a5~AsQ*zq)&GssN*deYB;kEgbUQ`gRMl)Vr#Xk#?61 z(j1bfc-g9T6*^2j#*^DUba-W_p(RFKj0Fpp#3qG-`~a>)*TY4@!Vhvp?g5VM?6s7DzkstN_MUlS|kKLuA05#kx&H1A`*Z_D9NLQ8A@kox!If(yN>Iary&S zBm7JDI;XtMboD%-{a8OU_9L%j3&=|SP21Y*8I;biZx5;qZ@g4zB;S4j4nFk^UkA7B z*TF#`8Ai1j*cRvWwE(i$9;!2>D8PfFs{wMgqI%?>bpX!6@%|$Z9#QSeE2K>=aFY34 z5^7(aKN@vV2z$K&NZTAp*?a~n5)eBZ-K4!;N+G@ssCj+oxN=U&Wf^jWL76BTO1w=H zoMn6@LSgxP3JYY>VyPNpTWmCuWn3?$(L9PY=+WD?*k*`ui!{D#mCLXt_gXP2lwfJn zVMn_xWQ?3XR|&LNx^h_>Fui2X{yFTj#Hz}nyFBn}27P-l zD&!rEf*j66yB!SY+qu|xepZ_TfAe|<@%PX&zzJ@_^5aLT-#^X$%LRs%#F2QW9OTj9 z3(zM-5pcO<>nM$2cWX9IJ<%$5<}EpC9)?^E8D~WyPTM1(j`+ZETtu2zcvm~mO)P5f zIW6Avh3v>AjPS(&J>na1`I-Qd`{D_g?wt4^1S!nH2+?}d(k}Mar+L1G!a)+G?$K`$ z>tZC&M~HOtyWt=>WQ`b43a#@e>(*dEel0bY-?8g}-*BOGZcMijUBI;mla1tK;XI!i z;y#YcZ+oUb?S3d24s^Z>oZAYNXgaC49TZ)SdQMI0P_7iAYS_D3Ho;rs8yjaA~AlEzm^yU9KNR0jTtFY~rI+r@WR%c<+{G6dF??c&dQ9)(0K6{K;|F zooTO!=O%<}bi3k+F5gsmLlbH=5XJt7M?JRpO|~n84vQ_M>H|eC4V! zhdMxne8+{3(-r?Zps$|IZZW&wrm)#Ly3Q`V%{r~iGrl1J(h)EVA1-}Tox=ZP2v0(^ zbc`zkhBW^es-e2iv6^3>J^uj{cI;U!=9HnMRiTse@;fUI;!9hjBhh;VX%Wy1f2jJz zXUfg?Ndd;LpKma=#2$?|_A;q`qN1FhA&;ygAOZw?V*M;niyDkZ7l*KrRrH>!6fF#47-xr~$301io)tzdPpG);OMI3ukpsUU0 z5s-7l^HXX9_yo(x&E=x<>ClC?pxY}3G3(~-_HTW*?N=mydw zLvH(-5LqOCW__LfW+g6(h*mI{7Q&n#rCb}I{D zvo{T1A^r)FGti@ZvJ{?KEkNGWD3stcX<%s1s8f2rO+o1jdW$!zqnT_csDiBm2|1~p z^`KqRP{}wi{+m#+o;5SS-AFEy*WxwjLWBJb|1{7-o+@<-Ogkt7`*LF-m!beK(kI|o z#zRXyOLwctq04F_7|Ojl^l&9m^I%*upl@9vlbBCqnZOm~!9;)^_~wax55Pf88)pO2 zXOECptSeifMWoi?<{!UqRgl*|sL`E_qrq@6GIs1-a0a2gS;i36>c6-|7f zms2X7KA*WCBUOmN>y#LS|A2z=?>oZFJ9B@(rAaOG7S-^OQ}SovFav!=CXgP*wFxYF z({TMdFTfGuPUJQ9eLW+-D}aCR*Jb*B{9tFm+r7omf4&WTf(~r&gYM)cT12+Oy3GGu z`Q0`{lkItuFI)%F-Xbm|yyC;50Zj1fevMkXC?8MoADj+L$THb=smc$iDb?-8M4|&? zcmL#)bu#jvO4mui#!-;B#M8%FXqt)VQ)Cp0<>1v1ewb^yKN5VJ{GJ4>qm|LPb2AQJ zXVaYb+N+G)?`P_w>EuBPM~r)(WRl)!sG285g(cns??^!TeeS0j{VvL$tdt|2 z`-Rebp^|$MYusil?3D~lW6=lo0p#yCSa1kBpgd8aTq!uuGVU=}|7VP|HuG?a{~2TN z8=K$U1PZ{T8f!45RRAT@s<=b&kHhM`Kyps6O zNqs`F$OkXC(S!SdFnVy0Xq=l03G&&f#q_60uXd7&^Bi_c2)hYV7L+iZk|#TNLrvwP zWSCEY7`^A_ZCa4X@&4pPYlGxjOq#%PZn^xu`yo;^-XAj`>`#X)pr{d3I>)%Xio+UEskd+j<+4P|BTpBL7NbAHbR zP{l^1V(j;b9qDpSA~hCnbp>xio%o%&5o1}&l7f;rOvzXv8mXTGGX*JN$uS*NINHFxpf(MCK2$wTm$<u8^jFCKIb~9;W1)189o~!I_&q~~1Jn5K(j~4!SWkK*FNXIDa#si+ zZ}t-Qz0kTkIX0RC8e0fvCEt;GTvI%N9%_JDUGmu_@^Vwt0eMWYC-+8!mY~1gNPdFX z*Rbwp=AGrHVDR01pk+U16Af{G4(d($FEdN)vQ~N$_<9cMT*|AJCb32Vt|D>KMpnsI zKRYQk&T7`EIhr_nloc5bB`kaVLB#M?e1Wk1yn#j829R1@<|Jw#J$&V&GJqH+O1NnN z{D=v2(W{H_Ym2aTwnR|o1ew$d%QfvD1)~^~#QO73s`7J>@{LIIhj9RsQ49ke0tXG{bvl2FY`Pc{)0sxA1;5=lX52Q(R)j65GxFW{u0|0O-`J(Q@Gei@t5X?5 zG(ybR|IALXk++-Iz;#CKKH}&8Ww~GT6#IBng1wbSNa^ghG zEr}8Pm4kyD;K9$Ln}Edyw5WxUA*eG`{ZAr~V;@n_@W?>LON*6IoWL)5AbKQ9JsHo3 zZ_~iratw%W(6ebv<^{lh0Z3g-AkHUy5Oh<02y&Z5gco5eNd|2hu&7V#RH8ObA;Ld` zg#=ihJ%Y!k&;Y+leG?8;_&MVqXak%8Re{_>5K9!l#fa4$Or%t#@WDL*J@^Fd)!T*e zdcaWx#96V8lEQKoB|oo=7~=-rvD!qj*haIKM)M~5o7q$lhBgsg?N6$bCG_mKmG?C< z5u-Z{`8YgxHf_VGhT723kq|WC-fZ$YWD%@}ns*0o7l5W8w^i`^*p3D^e0SPBHn#FgUZ{=jX^yr^Zm9k*dNhpg}+pJ#6Qrc&aL3WiN@3on9Nyc}WU)ueUHDLIW61B<3 zVm{iZ=?RqsHhtH$lGaRT%=8ApKP?{ z4M9;r_TP+VfdtvkkP)gNBY1kR=`40ZR>(}rCfZH|wMZXh>faYqs0BLqDW~xU7}bA~ z%k_s~K$`r1zw}yprx8t9*R zclgTa8tfT|SVIqbTtmO=RlMSB+cLIq7yUZC7?1?-H1i0_$3FOH{8SLud_%>ZfPZO)YUIWv|7SAw>t5ul z`?qRNX&lQ|QTLyGG(iL*sho-Vsmd0+qxOl2@5*_#w7O5mO?GRZ9LQTT25Qe^-6bg( zUf&ILUKr@cdlp4^Gz_5H)%R+T{qVS@;I_z57E7RogS~3onOaK}11>H3Dq!~5{T2h?OwjGt zCZ@M<-p~hwYU%U-=GOXPlP~s2CPWaF3Wd%{TncoLq5z*Z9|Y;r)7jIrxzS2&a7t~F z{+B3k`1!;Wk=@jNB4qL_ zJ6!bGV)AW<)?Jk%m1W-;wTM8#5NEvPc(rw_pmGQQe94HbLWAQL?)-s$|AqatdVAoT ze%=9I{Q8(s(IEu&yi`p<#5Tp07oK~7C0tSQu(8vzy8h)3baA0EDFKl55+9)(Ls%xQ zvD!ZFE!{y@-}+y6&yE|)*E&)?0a*XNV^mWGe!828V2);v&hW939r|fW((pY)U(mv) zk;yO-JvGB(_$#g(b$EvGPjj2SwMmR__m1Mo!~oz{TVipMQYkJe3bHO*>auRQ)ERaK zZ8Hxomt6V;oz+XiL120e8O72<;FYsG6^m5KW7J^`t6%X;)RVag_bwjvH^9gbrs-W| z+@SFO@tQpwZQbHeI!xNB%X0IrA;zrgBYb=kA`5)D7ZTknQ6M@XG6)bP%gk+ zhv2;4ZxHk<<7*i_e|y0e{e>13mRl|+kynBuhl^QFW3l!=_^HrzF4BgLiiE9}yJUVA z^h1Di1nWWd00no93oPBO&9hpRwnHK7Sjaocpj<^dAaqe^@NLt$AWu32C|a>KId_}r zJoD%}Zj)Pj5}OI_YLv%}V3nJ}Ng5b`ideFpUA>O**GRrwG=CJ4KimKysxx3QX(SMx zwLduG?v~+nUVp3W#my0tZ$6F0auDrIzS7GrZ*(VT0?Pv(Ii;E$$(+}*GP|4j%&+gz zrElO!a?*JL2w%)Bz{{6tRboci%cBGH`wggl4QFjLwg(W0+ZDm>I1r|4FN5lek<e8`stoE zM>4}5X0g1TWofcbd;N@%-B=0_l{Ccl0i+Hb=3h_8V@gQ=L?G~|8+6wDZPpcIT}nC2 z=3xN6&w0-?{ z@l~4cT}S8JUHBi9rP3jz{8w$W$AQ)3+ZNGFZ8qz~c!A?W$PFcViZUZ*Vjk+GEBOje zDaFELa}G?eeEGDPEjYgEYf3kNgUH~A@G~!F%zbQbjp1T9$Bvo{^C^d~jdmIv_Cy!U0!4|?E=Mk7U> zkqx+3VwAPkC;|$X9@pt3_#+EB=gJIxlyev;;V#<4!9cHHGuCJQ6cEx?3v9hAz7Srk zRr^3!3w&Y@hA$p{ie-_h9z_B|VaTz;czc*-b@AP))>*Iw0icQ!ca;TrWFdQ4Cqw=X zchXOWudXzRt^Af^{jNAWKZN4Dx{UPariE{5ZYfiG4fpHMaPi+Vr<`U#V?~i5Ek#L1 zw?d_Nf-Jx*v?j$C5p@bN z%LyqySp|ofaKljeRFeCY=A{gBKEN;)+Q=IdaFrCWmvjx{0BEZaGRSLkcrzxGuth4) zZx9TU80vgnG+^fr`zkGzVm4x#ZTOaJlyq#2B=k7`^C$pocnvaRwF2BQLPNZ%Y4SN3 ziMhX-IF0{Y$UCnRSl7Uh<%dxTn7R#38m>F*t(F@)x0dL;F0u>DfKj2)59v7jIkCB+ z9|nIZ=aP90qxpYRO+!a6RSO^KnIv*{V+^BB#M(L~-G8J0{f*k+HA%49Aa9*f#WkC? zO{H{2!H;xI#;7mym#^ZEyViwI>7OI#!U6b;7^k{JMYe!)EJ=abA_g||BKZc5vY)my zSh#)-%0c$SmfI_3Eey%CU{S>&Fxj&-1zTXp)lQCZq}v_p7E%X^SGPB!4mAif7)Mf3Y~-nY-DjQn4_TWj2T1u|AN+ ztw!TpQ{1yGshB0D!u5X#k6WleHFGZ8cM{}IKt`8fFz;p3QQxbftILBDdVs6$-0oP8 zAV2(}CS07wlENIXY9ESK*q zMH^cf>pyFHZQV~JB+zf%9;Uk1*Z_hJ#91$V^|COQ*RL&oBWAwZ#G+AQvFN+W7VAmxZJwQ)=+bZ`CaIz=GS`*!Q{F>}D=iEaZj zgar!+IJP3DKxXEskjtn>!0R7s0r1Z<5N|aV32evS!~-eYSXgb?F#C~{XG}unu%EW8 z>#G-!%Th|4ch4FuF!vjfovFpYi~C>mop!8?l2WJcWlLv{F%q{~cE^$_3ou@)rK!UR zww98C<3n>Bub)ic6!K=-)vGhgKeAB0z)@o^x#yWe+YSGK2{IX((E0L~T;JKjKe}B| zT_=-?ckt=m4K9g49B2NTr2Us9CL4%?PimDEU++A*RHyiJAX(-kV-O?Q#BZI>^3vw2 zaH~RsiOa{b#z(>ZUj7Q#3iAzd>Gd!<;9152IKj0y!DZ~T+;Nc<#Rl}4JSZ>*<50!m zZXUEyRvje%BTPKFU;n;qE?d&Cb4`69|X4 zZuFm?47UhO_tTz!{`VtTY_(8qx%HPa+hr$?(Qbx@Yrdu8Pd2rezZplTn0LNm)xYk~W_T(CAL4qzq?J3gM<1FJ} zp4=#ihK?DR%kbs?bAN{{J%_u%`A5+RBOBxF}2gY1Y-b?>@}hg!&qn-Op0i;)p@#G&ZoXLsq3s$KQ6Gfv2AARQBRw#h`rDsB;aILA}jgy zBbXD4Zrorq(79uK(B*i>C(O?)^0xkGF_wff@KE+jaYik$5!fh@xuBC~;3=MxxxcMALNb_{5IlY0c?%J(0%QU9-tSa=`3aum%LZup8 zx-|YS(aItq?A7RvY*&wcaWIWU&m|K>Z0K1k+B(LU7!j9z!feoQpZAE$FvLyxp&yJi zb^`J$6!qX0260qBj2PvI4xlvMuI(k$QYac6*@O&J?VDp863Fyyk?a0u;0CrcOUHX3;5j_%7%qTe{KeLi`iI9q95W( z!+jjq_V>%@pVgz|;!`8a{&rnOC#5zg`dg)78$-k4Jq-F%NxEqQ@VjCQ`k4JupVq{T z)+CETzeA$i%I?cva%lWnNY+YEt#5#*4_xn*pk@@eW|DB9BHANipnY##@5#?gR( zj%M~^Z~)PlXRnc?xi>K>uL?(Y=nB~ZKml31`3S_m`~6CZu9_Q$CW1x};SNmY*t|l% z4{1Uyen-E&FW6b}oOGKI?-AI69VZ@u9(EQ=rV<1H(;ok_p(mZLr8V{5A*W!1JT%)@ znVp(xRE5te?|bF){=U4KA5C4WhabSzeixtz9L<0|ou_xca|z`fcw@KL>ui zDlEbDjn*PN!{%BeHt*$X^6-^kC>lPNBwj_Zhgc>S3(_L^>f*KahN_Ak-z0zw8Fi*o z1hJVvY-2ZJ#x}L0Y=Sw_oI?>D|68x4Hlw|VvKnZXYG>>lvN;KHh~p1~4w8)*^U zncMe8#EFMl^7N|c1C8~X^QV>s%tWsWLf>!r-sJ>sl?WkholY8^YBKu<6&Im~_pGi9 z-`ebT^7J?i{8f7dslt2B3k=vkU>##p84%a%r_b@OOuyNfFI$~$=i+QY4t{kU$m(x+k8@E-908~&b^HbHjLH#>0BX`} zwYss=WKcHe-`##JXH}%EJ1Y#53B&_iFP<#=cCX)wcHcU>8ryiQIvDA=tH`=*e^7Df zhCypcmOf8MBd8Sq*qH4shwIkHH zN&C&Zbgw}w*Q-6wK^peY+osj*YE0u~xPQ??#8dJ|_sYLCV(DzDrLm)5Z1UKs?frY{ zpR9@_(~J}1nF_8u|EBn?uHMPR)2TkOjK5}V+aO$Kk^igUW-kO;n+gq)& zXKT7-%?I%w{?qI0_7yJc^M*wKZqZ#hB6KMuxR=o9aeZaqw|C~@bJ9>ZtCbG%IvbwT zuHrG5)#r-1@N}|MU{=sovyc^g}?r&Zr%V#tb;4l;5?jBkts;osl+7P?#NdoT{>?H!gxX7G@bnoDKezUjX3l!~3Js6}IAM%u3Az zIw|8XzqrMljzJ&O$9~Gh}H!xgAV*9x-pLFy{X(m5PO<$W2%RlkI&=eQk!??<$tUG zQ)$RPN@`Q#XNeOE;mKyx`u<&*BY`w?@G_ySBPwkHQ(GUNMjrTU%qNQC43Bp;jANNf zr%3`&Dr~Cuh6z=}TvP-UZb8|43{~3#Q_k#bW3g7(YgtdaVWRI7kU?p*3C@UesV~`umOfKEX}DPPirx4B2FvLh1Y7SP zkH;+h9d~rLL{qQ(zAx?Y2?oYR=y7_u+F zu|ah)@k3y_@Wp>>Q~z1U#2AeGtwEsm=j*yI-nv@h)`9S)L7zMw2|NdvPO`Vyv&MQy zS_#5jxjY`i)4WsXdi(b^B6m(d@o3zrgf@hYnq4f`?u|}(`2j1Zbq|yNwEAB=J#W~a z)F$z=HY5$aEyF#h$PPwBkgZ}f&$v3Cp7qrnLw|GC?^VCHVQbMeU)C=jfB5=Vm&q>e z+OF;nq<@*A%jg_}()|)V)7QA5uaZTMV^eYU37Niq*fje${@M?gs#`ez=2=q#1!fVh zB=Gg+zj#1EwdzlErU#5)jdud0Sc;7%Or3f`v6aMa3~%Fuq*U~} zSpiX&hD?k@#ul{yk$E61$yk}`ud`22dL-3$!dN4bp7SelBn*Iy2Q+ex`o6hdn(^tf z1}?74FRlj%7*YXVx0V&Gv+FnPi;2$yU5gDLo@9_@#n0iPpl1uR7A7G5iypO%p8RZW z*-Oq*?&-mriLiYDsz8ii+SS}ahmOP}U>F)aa zleY)0QfWjsr}cUJ6d*0y}c`!*HwnL1pWy14hoCxLA&a$< z(oi`ROizE?v%DABy8x66efmezWWgX+6MwG%#*1^#IP*IzW_~7Eu`u4OY*diz9lmOt zb{0-wlFGz6ux(JYG~6nY_oET!Mx32 zB&qs;R5`8WZM4JFvCUOjD%rbB`CGU_?VmM2UDY!dfA;-iWNxouZf_GpS;>gTxX19C zJ)SIEp%3ioPfm}t_vab_)A7mjVPlQ|`jWt2YCx^cm(s*(&4@f3ztDIbNvOXy?D6^!h;aAC;xKylwmeIQ`4+9M60Fz8w95A8h8SAxHYH zmw0Srxgcp`W3cTjwDAp{y{mQlwVlrX#rf#znH9S58R6ZJ=)kXJ%)#98GrcX1NUu*p zj1g8=sbXT6zW4Wb-)&0yg1bdrzO!YihRsagZr#t9w6*kS--hkqE|4jQF&?&-KjhAJ zsXLhcx#dM~)}1<@ZVcl!GsqFHRLY;L+~BX;cx%<^-@V^<^yu<*a#7HFp6j;WGoKAz zO(VPp%@_tb7T?xyU3->gy<)57yVx-+BwqEE_OsT$-jw-TVS~O0ju0B7KowULhFMfh zr8|9m`$ml$-6Z_}>+Wt$hYk>N!vWwLp-E*!rBh#kXWCbHYiclvr0k(BEZ@4h3C>Gk zKjDyH!kB2lSVi$8H<1*M3GeS4|Ej4H)uE`)T%MdQPC6^AJ&8Az_OzF<*jtbu5-b1$ z>9P-#^)aLjp~21bz}YG+|A7MQgTt{{X)bg zaahD)!oY2g;$cG~X@fekn>uMG!sRz87ibn+u6&A3VV%Jug)wF=1x%~C9Kv60t4N+% za_j^MJzKqYccc+-!924oOOu}`PCiD*E_;kKlS7_Gv%8~ByRCJoIYAscrj)#V zFBd4VsBnXMqRbbq9u{Z4>1^tIvRt1RPQ>BnEZibdn|xl*mVoacMGe6oo3I^IG%hT z#AIJ~|J>M$c)n*iwF?rpe&{cFj=`c`d;}6Ad@OiZfa>QmKW44?3+X2|%8jCyr}NB# zhsr{z#F1<->!!xlM4Yz;q*%5;E2t?E@+#;HksCeiPnb6mD&L1dNum84Y3M^1FSz@< zlEbS+IB1zx$V&NeuF}W;t?@jSfwy8$w0@<{g5x`<)_>Xhj)JGgLR|&}-}P-ik~!G= z*J!6Z5>-+vj0DJ(X1{9K`h&{ZeofjEF5h0`PlN#@eZxjk<6E`Wzxyj*+11F;TgBc+ z`-{JfmV>#pQOGBM8Eh6=ebi_ZN&>CxS6T82qXQM>X_q_YK$!|$$)RIQJcJUq!Gaxu zU6mH8%Yz~xs!xim#qkM!mylRX&1&XPIB@l-N%sPwIkHi%CUWv_JRo}}I?X}2nMdd} zZ~227UG29bd?Z^TlnHElFJdDkY?uDSz!909_-+4oKf5J+@piE-F_|2AS8{rBX8Z3Y zQvb{@@%|mhpBEqdU?{nuU@Ly>3!)6GfLw(}w@7uFU@@V=^4v%_dSxs|OxA$8KY=VN!@uW!{JvEGxgRz+A14;$ zI!!81+%}fJ44LlS!}kl-TmlJvDQM4lQQ#1IZge<=q2UGGqHuprWB}?eeLG!xn-+0B zL@>s~BeuWqJPPPxgaWO$ZOg+3$Nx6Lap@fCw%GEN)2v6}#dyH5&jvJX!Ty09TgR`P zMMFQx#@|%I$XUz?F2^Sg2WOFS;ukFISB{R<+WhV=)Lnn0_gf0I- z^OJtv3CPccXa7+7L61ZH0WJmie^}1$dJZQzZqK>TBG56i0kr)ag~$^C-Af1A@}L;! z`(mcH@bI_ottozhtQuHryuD%eyeQ~t&Uc#7t#qO>u2FuTE{=MMIpdZ3nQX>#i6fSA zlq9ERu@x56RNUVk*$8aLE|iur(zAQTz#y$6(C9;@?kR!Q>C5sRArO-C5-da1!=I}- z5mdZsF(Z}vEQ1XN6f}_ifI>+XKL(UQi%Naxw*U){Pdr8^EAdxdZEM!#(6D@mot3lu zy)twg`>)o%9}mBLIQCOL^H4kqFgWq~xxVv#@+NWdAoh=)%z2k)TZri?X!xP^I>J#j ztaFsn#whj&OMwY3CMHOK;(ImnZEK>p_?Zs~5+)^RDL zZkmNziJfPolVRN4r>GnI);Y2bf9SA_?6DZ@4ShP?TMa#rrXH%N^Xr7p9gBX7Tw6h8-)9&y>gU_4(`H;un?L?{;; zZ`bo}yP+ur94Kael{W+szA!i)RP^w2;WgH@)L;5>!XV#OR2Y(Se~46LqtG@8yu20j zP6M>QhuATz2NfA9$@Fk{KaO5@x@oqD1dqdO%XlHQ9#5CWD_yv*Zj)V5@f{rE%iXGn z2y`h?v}N=Z84N*rjiukoA4vl7Rp>egZ6(KQ?1b&V-iiD4NHtI9>N^b&iUhzS<9?RS zlrPli@{E3Ne*ceM^Q{_5QmJ1AzXo3{=H-Bq<#!D-rCxtBTzd?G(D+vsoASdozakg5 z5TjbI{`TGuUu*<&8FY=vboBA!h;+;-fAQe$kw@Xg4@A8o-fm)75ku%jhLcuo|NgJc z5bW$op#uI(XflJ8;f`YxciiOwUs_b=FIZ0w>2t-izF`h<|tZr!SnowWeKz#LnC-TU*|uaszGii}x*U z$+yY(xApgl=LEKzubQE_h$-6B3C2B$#t7J%Kf)>~R!m^L=_}YHSE%C{@m%(`Z~P+# zW-r6YDro$Lfvjb$kmNANw_=IVS(kHowx>yc zw!vR)#yWWyR?>Oezg)1@scwCjKXlxtF~1?`&|_mU;PNiX7X_E6D|^|O?tENgQCE2s zdV8vjNe|O>um9lQaCWc;Vhj2rb3NQdwhO?BF<~3?nKsVj^CFu$%(t!1X87ik6>m3l zILd4A8kywS2d)orW>GFdnD&q{&7Dh~4uqa6Qr5CbI(6{iO(;;yY4sWnMN4v+f$0;z z3i)1~ewH7j9R6-KtQh4Tda*q$XfFr$0kNx*iK9Gd^eLp8U5)#bTBH`R6F3UAk)Z$) z2>;!8hc(By*YTN6*Pl;;<4jcz&XgQ9C`cYpFM?Ac+n~^C(Y5NGv-9jh8&y+4;&)5k zV{^}6&g!Q@X5%`+yH^4BJyxYZ4IfnERR+vf+mE?n7~%zUJ`ePEycv?;AefHBhJ}aNiI6; zBSF$tFs&hi;Qrr_A5#xhXV5Xmp$@vU#R_Qn=aA)_Pma<@Nng7w0ID!6OO@oyQ9NO$b4W-YM z37il9NeAEm_{x(0ZHTki{=GPyF+cmcg4^uP)HIjw$-)&~UM#U(pK~41B(zz?y5v-M zLVx%rxP9O}PGk9|=kRyPI4I_bEETgMiMS!9%$KpL(qS4c=)D~l5^$bbX9e7n#XsFJ zSZf7=SI=We??Htg2zbj}iXB~u_yO!jEC;-HcHhvo2A2AnY<2dm>DwX4^u4_A${WI9 z_Hfm7f~DqqF&-$5fv_Gu`PUOdNC-DOF$4F$yYw^ptK+%b8-3WAmuso^A?E9XtRWj<%-wFE!9IPQ}N0E!r9Tsa32k?QAMj!4QIqGepSUuu!W-V5t#`nRHSW!9c*AV zK-n-G)DKp+kD-gix((Ng;2&fms7+JYj*{l2MV0Ph{iZ>OzhvBwdf;U3cqz1K0O;m= zJA^7gTm?SCfv;$`K1oq#CK*O1&5B5tOgIPJB!75BlexL#4Jm>!3#{wu=%l7<;^e3l zXl$nxV5k(}YxiP$F3U9oQ2ht`@%G>=r76eVL-LPJaZGY44n*rDB!3vycBuK_k3lZo z1e&el$6X;(rcdO;<$Zdfz)lg*ni$D{g<>;lIK=$xMPiG8=-nn+M(CYJsa?kD9fs*! z_Y;jpJ_x%MI(Jg#I(GYa9<}}3mmp|a7u#ri)*32eHiKs(H1RXFvOq$YQ1O8-DK5WH z=7>ufoM9lYOce!TF&Am5_?7xnmP3y29yAYZf;+M~w7(_n9x*xONf)#o16k&ww_whg zoL}Uj_cApjvf8E6xqUltzKFRmWV!2;r=nev>&NY7;nkV<2e2&Lfm5>KaV` zN-0q}^L18J*7b*^w<}eX)pL7-#~$MNM$d}-RU9^>B;aD<$1lOK#y_bU>aP#-gLzv5 z%v`;p4QqE{1Vb@hl$u_1knnIQ#4iM;NAyFWf8za1Z`RWg(@XIC6e+VoHA#>}q+wu) zcP{LaJ+xOWJm)zGQ-)s&@QX(xv7hUt9=XmA8CnzB1P^6U&m3`{Pldb6@=p)momOt2 zMKd|9^$XVWNo^_YMY7xrq89cm+;WDqD^*2GE8yvU(mZWb+62~IF_KXTGcS^^gWNI5 z>ahmi#Ydc7M^MctnE8K1`c9<}gG~7rw?ow=+WD5kVLOe3 z3L-PtC!geOJf}F{8MWwNtGgt;RDb3w*|l_c5J->>!^?r=XFcH?z#jB5y!0`CX2#6t zyvR?8$b7TaRQiRuh*D{*B3eP~a>sKWRYRcuzS<7;FQ71z6$Kt;XyBPjAi2QN-p8cA z>8VpVSx%sPF-jR&YmXcBNG!0Exq8l&he>n^5SF0#h&7U6Hh&1i@4OPD)>9O=+UXSrudM-`}-&ytuRj=*uYLJFRQCtY?4m8QQh@uGQ>CoVfcz5m^yO0c8 z6mr%p5?iM!p$mB++K4+|sjnL2&Dz(o%l;EQw1>h1J}pOhjf{%%BBe1P3n@ZXs!S68@r_`ZbnI~(xRdaLrSv~ zQ4&}m!Ow(dp+-5fa0UhpTNJn)5x-L+cfwbNM~J@RSbeRv6(I5%g%NDBr#XpuH^M{r zkrAApxR@=N3J#>X2Tiw9zCx)M64ZY!O_dm=Oe*3^K$x6di{iLSW~)&s!x4k zJsr7M1+A$vsVeE}6)64_d32)%RKv>~kd6jEr%sW;8a5|4win@_+?<4e%c~KT_xX9A z`^`ni?N7|{m{gj$u5WmZA+uNZja?ybD{X%pJ*z!maRA|@!3vDPJ2{)SuRu@puO|ZR z3kxDpH%s7cDjp}mmQr0LUSIDtf(e*VH#;L^C{Pd$^Oa+Q4pufb%6Uwg<<|<;?=4@V zh}1S0d%;L{r^s1k|2J#fyxLiOgnnltVCPBU;z;1(P2%89LRTH#-ZEDFCTHWVV`GECgAGe=T_d7leKfg z+jgnd9hf|1?H&5-1>NN>o!Q$M6a-?0?vG>)HKSD~Ghc_NKX)0*E)Fz(i> zqnQ^~sVLDSu+AXozmo@gGa)!^nHVbW2&5^Sbw?Pn%@?)y-Je zhs=ao>a0wruB27yU{!DSWG2NKn3+ZhQ@mQK$u+hSG!3H`Vi(C~>D9HRT$d@!7 zE1MOrMFsl!3x`4;j>znDpl>=raJVm>j*LoVNa-&vk&>)IZ;& z)A@4yIGY@=Hm>%ZXic5(^hB{eUH!!ZUD9zufBYQ1?|ZciY|3zh9k0=m;K#^)@N>!u z7+2S3IJNew+OZa?`E6Gd^$f)yoq_eL5B?W%gQ)0x7;5ZW8Y+b=G^lG7|J-CH8TE{PMr$#D~&s3WDmkS zbFCLIyYkR393FZa#tsxy9?H=DM20J06c*970Ylh@^HTNqGEZ(7Woi4BV17-&FYSd8 zP>JQh2=Eayv%^M$*X=hTEFq#vA@8H$Ta=II{AVYae=2no38VXf#2o4wUCuxFyq!O= z3jc=hr%!(&a`aWYBN%p$Y4IvAek~A&Avnk!>-gC)>XeJPj+S^69JGmxe-?)Qh2`4x z2!5i~%Qg&Qzcsr8MgxlUft7K>zJ_#3G)|npRHRZSz;UyJNU=aoF2}JtEM;ejlfAYt zPgKmam!*-L`%T>WvGe_mM$sv|rh8gNUBFT}m3m?eqD#^yCU&OO+i+fEbD zO_QuVI>@)mHU;`-s;q9BDjqm;Y+;dRt2hjg_fb&la{bX|>)vU6640@JVj8j0_?Qas ztP=2aZV*8245jc6iV^pbW)QN$Gi!k{G~9{lUiiE9%mw3Xh;1G4J_|QLYuEeu47&l(e^{_RdH^uJeOg)z0TtdopC&I7%N<86`|12)TXb`iLPun z(=jJ1M`5fd2xBDD!>}Bci)1NnB69byDrdrbf+S36lGJZSIl(Qd&8xCs;eBsI>E8rr z2(EG`RHZ4+vooWnLZf8m;DF};#@)4=ZJ*eIN~6U8?xQhj?Rt8g?+)AE2Fij|v`XV5 zkdy?z0u2MeE$9ua7$CJ2+ za#JQogo+jINr<;Mj=wv3@0NQswzkoQs`IJMX@E1IztCjAvu)CQ?fsJXSkO`(?wT+6 zy|c;rx_gz~!PdH&(vuH_mi@h?UeJ;<$1s49-}`-3Gxl!P$us=qrj#b3G$y+uB{c)W zmcsF$4+(mVe2K7FDL_ZEDM`b;#d@MD-n|o2{3e^htN$%-kAnymO<)-G??sSGuk2cEG_YL?@xk*_(gnHH)lGl)5MVWY1J_=9^~cn_=*zFC4)vLh zyg!~rzP3bUh(4^(6Tk?`bFE*3IH1d@44 zs|2Dg32v6h0>Glq4~lG8>Ex2tpH~gz&O1qY>`Hpls+LHFmYGRq10v_KN3j2Mtw$L1 zz(TBf6eW4AL42%Ly^r{N(O;`GumijzwnN~w-1B~-pgh)!yg9sEl1 z+#Bq;Bw2-VoUl;Uk2)K!DZZLh=N0V>Ywru8K5+k+VDn}JsUth9VflWG8R09TXu{I88P?<^n(+iT-mKOfW&Hs&AdTPMcV|Zb;Y&p zEQRNr>a^}75GT#w7eA0tDD(*Rln9_B2t$Dp>`&S^kVRXHHprk-d1)HSQq)ImDtZEo3JAqC9r@wvk9(3baXLeN-!%(43tZtwMz6%NE9yP zo+)uT;0AfgM)Ow9cr%}+i9fL&(=oVt1%LndoMzi_!V~h)f>G#50QyTLWJdbVIhJbv z51s=PTT2PntxODFZ)M{7!H1Lc8GhcQLqnlkC*$R3<41h_=MbT@&iD{WUWl1op!AgA zplo0hs@v#Q0N>Rl(esYOEu8x?AdUbDY*jr^vC&E+0kMakyxVgC>4Tz@;r35 z9`w>ZmDfesdtQH0JC^n|2J0T@1;0oazl?t8PUqd|A~aJws3T}pn$OcH{QXaru}iKI zxnpY75zA|8?Nn7V`HIj)20RYNz39|4>8f&0j8ucD;TpLxAta$8F4H{Pe~uf$;f!2L zknIEXa8Za+kRn{nGDusYd#DkOi;gw&7<^5}AB?sKP7uJ{wQ!pJ0cRM4zV2!iNQhLI z83jlEQDlvp0C?1ocqXcOCPwObMt~jQKiOboctV;XpID`lMmHH^xVjx=33D{r_m(5N zv<2(fe|rZ%9B^ycpsNKoqpYH(n)Pk7r>ZIRA*TPp`|)$b)Ngtyl~J#4Z)7*JeON3fSNND1ni-~3W+O}_=+Rv=*yg6VST%bC z^)(%C4Wthxj}*PH4!Y0G+t!3HHA^F-sa26jj&jkm`7e<%6w*wZ5!j);C?M=?mSq%p z1~R1;&d`ZM0IJV>Wib-RhV|svW`Rl`LQa2|N*GS6HcIM@#OjeNBy1ZJ0y!YxnmbdO z3dB{=!rsJ2?>d@`-yvCBkxIN1O0`?+{M;^wvUwc^hbH%*dge^N>Uw_a&thm1u3FwBQ13bY@9dBfHz^JIlceg!KdQ8z7c(3{!Z0xr1Mz$LOyU-jaCUjYQK0LH zeJaghQL8zfSwAuA$A~@pZ4EnJ8BTJ9@srd?s zqM11yLi{PyqnI4{(=lGV^P7?UN{wW4r&K4h9l~ush?c;%GW|SKf9#dBjL*v>BKEL; zBwBDFs=fW*Y)gE=J%6i1^JNcPuX`ELa8`o%w$_3sODvGuJ@p z9iFz~=;UOsSBa~&+K$IIBO%VIXmR3PjtE%ZX&uB>KfH5Y?uzhXP%oLXonA_4xgZoM zh5-*8Ll#q=$OqRP3z?@^X3|ek3m^DdJePFd##u_>(sPK*8pA^e;J$W}^7I~+9~3$P zgdQ*u20UM-J}?l%5Lc-H1K|r4sO0I>)4&NBJbkhB^c>PNMJOH)wzz(VyIwF%`UU&W zJ|vV$G4MY1wr)O*766{Qy(uaFYm|6Bi?4!8xOP9A411Wsf=tGdZIRXYyH{;3Gi`bW zXgnO~ei>oA>)%*tG1RZtlHAP}|4v5dCO_H_qtUqBT}eBcnX2(5M+C_EB?%9fV!s=c zyKDMXC;E|xKqnzW{})jd|3_Zz{!3X@DiQg~e)jf{$d~3A#bl8g#6;h=fi=KbhKL;X zg}W#@>NHyJ2sVE;(`&R3dF*AJj&=VT4CP;yptex+>xKr)Fa!<21bko6 z*8J8Q=U`n2T!2Y;R5ZpX~-cSlpmtqmX`L+zqk9oMIn=NdnXc}O|oCO zii`T)^!J+(60U{#*{`H+Nu&^7yccLXnZGQWWA?XGyqQWOZy1H1UVqqI=x|I0f^oaC z2yht0(SAd3uXa2xfp={BQ-#`+;2i2*w?^CgG;s8v+;76=)wf5&zzP!zEFYn=U4_jU zckpss3=!#=Dl8Jk(h)Bwxuwpug!bkif>5L!L0TP_G6C!%+C9wCP>8kqRcf{?LV@aI zGtQB3>#tZv68g0Mx2kx&kY8Q*x)0!lD@vmq6))H5xP~znn=_=2dF<8XlO5L?PZ*TB z>PmL(H&2aIH>iBAOOW`S)LHvxS2YHZ>oM0tKt;EZ%Pmh>SOuU0-*3ABoMh)F;!|LE z+rgX-hG8mj5Yvcg1WK+!$?S08%dc}cBye>b4?rbidzUou$dw0kfT|NXLdo>6 zx(YncQD_4e~A66QX5%W1W;f-bAY?&J@t`J&(Joc#R)LVkA_ImM8zG#lF=DLRqIED zDU*dphDGwHvJ%M3GUomvw83UB+`V$`^z!xm<}_#fvuMTa&%NkWW+%6YDPh8D%Gi5N z&h8w&B3#D{s)~#9Ro{b7Sv3Z3F7UEf zw;ERWU02VijXpzjPE`gbJ<-(l)W4=*SgG)gt?rVVj1U<~nq#iS_k;u>`ENW%eu>f4br;v0Wz z5R%CDjhmFGo~yfPHV$wKi2fF_^LiE_gj)e1GHe7g_ktni=@}Wj(Le}C+=pJcJv==> z)J%bS17t7|emPT1Y{60z$4~vQ0g)UuMzkWxy?>D4X@5y7SMY1QE?EsPMIPTj6TWXI zd@9tvzg?USQ#xtVxN2fWFV99-6%HAbJ{oTa=I;dLSyeL@Hx;ksORfC-mT(af zyRC=rJR{%)`137sUy;$J4nQ3RBXG;-+`{DEP&;R<>i zN7_`WISP@i!vJ)zmzK&;xE+i=IfpHrlri;|St`U;N|YH>jQfT^yX!c7m#5%Lm&g4r zfJWT=Qo)_cy<01jk?f7tup!E&2@2vfJs%Dc_N?Fg?(_^tEY6zrG0qHCE|M+PJWyO7 zLeHJF*a<}x!Sb@bVSC#koJ)B(A-zI}HOe#|vR*%q*y|8iXfyBVuUh|X;3izHoN1C& zQy1{twmnZsgho(g>zi~haUL2rW}Q{7dCx+5SA`;2HzLF{U&^j8Q_Mlb=zEfw4Svtx z?LctX%1+OuAL(#UvfE^5u;3Kz^9-`tCWbHP4ElZ$c1?ueCj=0i!gUkgL#5<_bN-!} z0Ur>(CE^Q7!+n22G2oYeHRtd+WoHkFDsq=Rwp7fS55w!Bw3)*)(Wk$MWkOFyvGr4L7=kvtBMikr!opwinY&&`s6zWmhAL8 z{!yViGpD<<=ehcx<=6YwuQBOLw(F$X#Z|M-|NZiLV42JF@BKdAyVqkv#La((alHTL za@#lfJ!kd>W}mbTdz*jv5!D?L)z^?PQU&STf;;E8J~s|rp~aN>`&7Q9Iv73U^Eg}@ zY*F%Ru{%EG`zf$~>KOpW3l>l6eJ}mw`W4xbK>$deT&}DUK+U`!chbigavBj}Xy@B4 z_+95igGf+F*9I=^`GwJL1_H?#?bUn1Kxn-HTq*ZK-c!SJ_!w@1GpP-W`haJTk^iIU zEWDzAx-h;=cXta&hjdCeN(<6RcXusoN3TqyrDVMpgGcDIMtwW)uwVxxNwO(b4z{^S6-q}+AV;w4SJ0H zwmM3o8JNM@Wu0^d#Sb?8VjDtCGTrG;3t~?&ZgkD1dV(v%VXJ-fO{WOw9 z8x3?N`DcRzuhXU|z2E1tL-JAp5tDzxuwOKJQ4hhKPZD zGH5?WkG=f{mxW_ly~HZn>hN+DJlInDH6OVDBd#!NhczW(1~U!LvyEAy@=TFH$F z4a_aKHr{W`dg@$Wez;;$W!2vL3q{oK;M4%cuPejP+lJqdpuuf!3iK9-ZL?meS8D5? zq-W(TMN7(8!4eSQufi&bf z17Yh{jMpG33pUVh6dc0+2*b9eW4-*Oj^Z^AOaPBd3$$W&;#ej#OSm!)?po@#9D`#lyaAkW^mlFpfg0eS<<{EQ=P*pb5u@K@?zyml2iVx zq)t^Nx%QO@Yl4xLu&#-KtuBgKCwhud42}KMFm{vK+KqylsgZ!L z*_#nz0wj#A_bMo|-)D6<4AC`00TLQnqgz<$EKy%i6&P_t<1y zUS1w)Oy1Pwr0ZC=oxOdsK(fT*E8#`Y7kim&ndyQZ3wc$5%)@B>Lf(btiLpsM`%(UW z!9~!6Of#VnKJfCvrs`oK=3XG9`|yf>@wdOPBYXZsWin=YtH7Z%&3DsS3(C2RU)M6u z%9gifO`or1kWD^45c(g)m%9>2?A3Pvcy=EeOBwW~d3vNSC$|{O@+a=GHGFe1Pz7e~ z``#A_mp{0u&nE&yiY)IAj+UUsI>6QFu?MAapL7F!tYSE({MZAJy62CBXRH2&clX~F z(Vd;CecY*Rp|{^MlD=Wmo?_6tJ{`M-oSbzZpGh9OtsS2+p|3i-n$$O{6Y<#a9^o>f zl+f3Zsf5Ez-qgOMvt2d+S}^Mb+=X-N@3-#nG^NXaLo-a|f?4+8Iz+z3nKeeiV- ztXQcZ*aNQhXtyDB;08b)t=Y7nAf4OlH^BQJa4Gw0@eKW+Y@FU_7XlxySgXTcO(0`h z82ORDL7**2%jv*>2&)@@VE%*=i3vxIF%g??vj%3`J$$UlwBL8kI|vnXw1|TU9289%E<4-3Nz=-BgB0}4LU#< zPEx|sct-*GMF3~dQi`SJvv+)yrAP7KkC)J*N@$T{f4~iyT;BlUP}AeIwc#Tp6B6)K zKLsou=2Bv2AIb}xFuN@My=0uK7&8T$$Vv)o4>g3*iC=Ymkvvp-$S-AkYtWeZsih`0 zro|~!s})xsYv>h6e8SI4-B@O^_eqouW4KE;Kk;>QIdKH4BY7k~8uoxR|AV5l_mjb? zSzxwN5jcV+3cn+Zv1B!hlYdp0-`1#^qYWMkP?@`D%8)jWD4om;tQs^>pj)Syz6D?L zU?2N#b8+dCgVo1#17R&Rm5$q(+qhF{qhOKXg`$Gm($|Z&M7>1H#cIRX(4PnEWY_1} znwyOosP4;SP@P84O1&e6u6gy+YZ6O0E^Rk)(^et#ZcgAsy~1BX+N1 z_es8@QF=)xAeQ&|%Jf}}e*_@WOKc5T@|RuUNbu))SvF>3z@Y}U-W>{ODwU5ncB|DI zz6bHQ!#?PeM|AZIGE3+t8asH}tnB@?dF#9=y?TEdc@e*dKt1?%5wX6_XxsjMW>6-g@mnpmd? ziFzHg&4{o}KR`qBqXReeall0Mf8^SG+JVOqfBt?yYX28%y9Qj|9Hc8BfeGbb@0dkV zxC90P{3U`dh$9$ubs1w%*@DIz2fF4Qi~Oqy&^ zLxXvw0Q<|em({=7tHop0{WY|_Sy@@dQN_`>FIS#^2Oi{#)1h5bqc(z-6w6pZ8v5D` zldsiwD-rb0T zFmFaY?l94@P=TpXzg)Pz77yeX0J|v&64+(moUy*7=tt-SK#N+_E#Pr;SnYHi<8{17 zd|oayJ>Y-bi*Xw)5nM~OJ}EadQnG-&ofqA|q_Lk&I~P*r4X#n4&*CUmkO-yZ$%-Pe z)hXnOH>_edx?%RKEKjUtnW|)N6i@qB=o#W3`jSNbhRvKZwo3So15=xB};yHvXEn-@s`rj3D8kI9gh|47b_u{%ClzkKH%l|m#= zM#Us6w{GpY?{|`j^eo-alH*Itq4$0;7413~7MfM%w-{R^+}f_tIMUycQr2b zRRgr`U8RObi~EtmujhW?JK9r9Qbh0#mC!mJ?uFi}U_V>%K2{_lTXLT6UR{i+*IL?l zLHWJMxQN`r)7*$KU2OjAj3jfpvM_8hfYxo2HF!Uo;}{9rKMzsgvJVudM!>p8igr zO*G~b@ewr=F#KrPwbVT}$2~o<PYkDU%Yfuzhb z$)Et${4@pKYv)A#Htm;l=_REGFRtwNNTWoq<65;w41!<%o^FxGiku7uv5G-drOZ{X zs@EJb`{2qH2;)IyvEzgFhrP{j*#^i880P;e0Sxf_yE^Pa9|C7S9y}tx&|b=lkc9?| z?@q?PM^dV=@Fl4d1D1KgCP2pXrqqr0b8)WQmsC57oDU=g?sUcWID?h>gH^f1tObLt zG^*^rGqZCj>xQ;+|nmjv8^ zeuod|**&LkyNqcCR@Sb)zftabIBb$_FmtUMnC>q847lL`TFle_o@YT;zcIW~{mi2K z=gB$_AqKf@o|R0Ib)WPkJh%%6giD|!iNR=bz*Y371Px+F1#}NEs@81&W+{=?(w#*K z6;Xq1B*n-=7a}a`<@m4W&Uk!9zI>+|E|ln}TOZE5`k+!&8*ci2iR#@6(_d$UlEnFU zI>*jEWTYOW2|_tZt!XKKsY(9X$^Kb!o;le5Ik=u#H-71Newo+p2k;dEuqa1A4!UWX zju4rBVu^mNpxoS}&LxXAGLW7|ZMKZG{SfUfmtg^A6EAuFnBCq?+&jDN%p-AS3^g4q z67pQuAU5SM8X2m_@xIB-8TIjKt2KY9mR3p(0~Wi%^ncXpud6A4O0aKBIOs`Gw$`{r zFS-#2cYO}-O}kN%IymFY3kcq2xaH_ssk}7lM)XohsC}##(Kz7FdVx*>=Ze8y9BzGw zsh9PAdyk7{F(Jyp=bp3bR4;Lji?Umt&9C0z2ZMzhD1a#+6DgZD6eL4fyaY8K{KL7w zPRzAI!L`ouu3N7xNUJ17*z!WBYC|qv-sO{jBQ$u3VX5v6=N)ZjU098zhrvPI@j4i1 zM`KxPuHC=p7tPXIiKy!0MKoJS3Cq)5*DGIM@cM*Y6ZtETta8GWBW^r_^K&4!)*gVn{imu5WG{>h?|&I?XE^u1h9;M2K~?FbywZpZp%J| zuNLsrz1ciyD}IFNUMv`|r&RlG_;wSFYN z@mGBHT>SU>@6G+$h5g^_$Fu9Uiigf^=gw^}jhV1fVk4?>5|@ z^H|WOY-*8pCmq;69gC;~v?7Y2GrfeU_bED+DpCW~fYIv{$e6s&Jx>kpM>8CvY)Kr= zGi8Mk-CDOi9XjwJ;EE=mNৗv$B8xu9&5!kNE7NGE@qOfAI8iQ^-WYA|KZASX= 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

  • Rc5lqWSHaiuSkJL_PBEOUSI8 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⪻4UB>ELJ_2Ae09j-AUC()k!A;KL6c8y2z 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 ip=jm*dyL;sm#-he$M3f*_1i;%Lz&K;(J;8c^nU + endcodespacerange + 0 usefont + 1 begincidrange + <0000> 0 + endcidrange + endcmap + currentdict CMapName exch /CMap defineresource pop +end +10 dict begin + begincmap + /CMapType 1 def + /CMapName /Identity-V def + /CIDSystemInfo 3 dict dup begin + /Registry (Adobe) def + /Ordering (Identity) def + /Supplement 0 def + end def + /WMode 1 def + 1 begincodespacerange + <0000> + endcodespacerange + 0 usefont + 1 begincidrange + <0000> 0 + endcidrange + endcmap + currentdict CMapName exch /CMap defineresource pop +end +end +%%EndProlog +%%BeginSetup +xpdf begin +%%BeginResource: font T3_35_0 +8 dict begin +/FontType 3 def +/FontMatrix [0.001 0 0 0.001 0 0] def +/FontBBox [-1 -210 777 728] def +/Encoding 256 array def + 0 1 255 { Encoding exch /.notdef put } for +/BuildGlyph { + exch /CharProcs get exch + 2 copy known not { pop /.notdef } if + get exec +} bind def +/BuildChar { + 1 index /Encoding get exch get + 1 index /BuildGlyph get exec +} bind def +/CharProcs 23 dict def +CharProcs begin +/B { +673 0 73 0 673 716 setcachedevice +q +73.25 715.82812 m +73.25 715.82812 359.375 715.82812 359.375 715.82812 c +359.375 715.82812 444.34375 715.82812 486.07813 708.73438 c +486.07813 708.73438 527.82812 701.65625 560.78125 679.1875 c +560.78125 679.1875 593.75 656.73438 615.71875 619.375 c +615.71875 619.375 637.70312 582.03125 637.70312 535.64062 c +637.70312 535.64062 637.70312 485.35938 610.59375 443.35938 c +610.59375 443.35938 583.5 401.375 537.10938 380.375 c +537.10938 380.375 602.54688 361.32812 637.70312 315.42188 c +637.70312 315.42188 672.85938 269.53125 672.85938 207.51562 c +672.85938 207.51562 672.85938 158.6875 650.14062 112.54688 c +650.14062 112.54688 627.4375 66.40625 588.125 38.8125 c +588.125 38.8125 548.82812 11.23438 491.21875 4.89062 c +491.21875 4.89062 455.07812 0.98438 316.89062 0 c +316.89062 0 73.25 0 73.25 0 c +73.25 0 73.25 715.82812 73.25 715.82812 c +73.25 715.82812 73.25 715.82812 73.25 715.82812 c +h +217.78125 596.6875 m +217.78125 596.6875 217.78125 431.15625 217.78125 431.15625 c +217.78125 431.15625 312.5 431.15625 312.5 431.15625 c +312.5 431.15625 396.96875 431.15625 417.48438 433.59375 c +417.48438 433.59375 454.59375 437.98438 475.82813 459.21875 c +475.82813 459.21875 497.07812 480.46875 497.07812 515.14062 c +497.07812 515.14062 497.07812 548.34375 478.76563 569.09375 c +478.76563 569.09375 460.45312 589.84375 424.3125 594.23438 c +424.3125 594.23438 402.82812 596.6875 300.78125 596.6875 c +300.78125 596.6875 217.78125 596.6875 217.78125 596.6875 c +217.78125 596.6875 217.78125 596.6875 217.78125 596.6875 c +h +217.78125 312.01562 m +217.78125 312.01562 217.78125 120.60938 217.78125 120.60938 c +217.78125 120.60938 351.5625 120.60938 351.5625 120.60938 c +351.5625 120.60938 429.6875 120.60938 450.6875 125 c +450.6875 125 482.90625 130.85938 503.17188 153.5625 c +503.17188 153.5625 523.4375 176.26562 523.4375 214.35938 c +523.4375 214.35938 523.4375 246.57812 507.8125 269.03125 c +507.8125 269.03125 492.1875 291.5 462.64063 301.75 c +462.64063 301.75 433.10938 312.01562 334.46875 312.01562 c +334.46875 312.01562 217.78125 312.01562 217.78125 312.01562 c +h +f* +Q +} def +/C { +671 0 47 -12 671 728 setcachedevice +q +530.76562 263.1875 m +530.76562 263.1875 670.90625 218.75 670.90625 218.75 c +670.90625 218.75 638.67188 101.5625 563.71875 44.67188 c +563.71875 44.67188 488.76562 -12.20312 373.53125 -12.20312 c +373.53125 -12.20312 230.95312 -12.20312 139.15625 85.20312 c +139.15625 85.20312 47.35938 182.625 47.35938 351.5625 c +47.35938 351.5625 47.35938 530.28125 139.64062 629.15625 c +139.64062 629.15625 231.9375 728.03125 382.32812 728.03125 c +382.32812 728.03125 513.67188 728.03125 595.70312 650.39062 c +595.70312 650.39062 644.53125 604.5 668.95312 518.5625 c +668.95312 518.5625 525.875 484.375 525.875 484.375 c +525.875 484.375 513.1875 540.04688 472.90625 572.26562 c +472.90625 572.26562 432.625 604.5 375 604.5 c +375 604.5 295.40625 604.5 245.84375 547.35938 c +245.84375 547.35938 196.29688 490.23438 196.29688 362.3125 c +196.29688 362.3125 196.29688 226.5625 245.125 168.9375 c +245.125 168.9375 293.95312 111.32812 372.07812 111.32812 c +372.07812 111.32812 429.6875 111.32812 471.1875 147.95313 c +471.1875 147.95313 512.70312 184.57812 530.76562 263.1875 c +h +f* +Q +} def +/G { +717 0 48 -12 717 728 setcachedevice +q +405.76562 263.1875 m +405.76562 263.1875 405.76562 383.79688 405.76562 383.79688 c +405.76562 383.79688 717.28125 383.79688 717.28125 383.79688 c +717.28125 383.79688 717.28125 98.64062 717.28125 98.64062 c +717.28125 98.64062 671.875 54.6875 585.6875 21.23438 c +585.6875 21.23438 499.51562 -12.20312 411.14062 -12.20312 c +411.14062 -12.20312 298.82812 -12.20312 215.32812 34.90625 c +215.32812 34.90625 131.84375 82.03125 89.84375 169.67188 c +89.84375 169.67188 47.85938 257.32812 47.85938 360.35938 c +47.85938 360.35938 47.85938 472.17188 94.73438 559.07812 c +94.73438 559.07812 141.60938 646 231.9375 692.39062 c +231.9375 692.39062 300.78125 728.03125 403.32812 728.03125 c +403.32812 728.03125 536.625 728.03125 611.57812 672.125 c +611.57812 672.125 686.53125 616.21875 708.01562 517.57812 c +708.01562 517.57812 564.45312 490.71875 564.45312 490.71875 c +564.45312 490.71875 549.3125 543.45312 507.5625 573.96875 c +507.5625 573.96875 465.82812 604.5 403.32812 604.5 c +403.32812 604.5 308.59375 604.5 252.6875 544.4375 c +252.6875 544.4375 196.78125 484.375 196.78125 366.21875 c +196.78125 366.21875 196.78125 238.76562 253.42188 175.04688 c +253.42188 175.04688 310.0625 111.32812 401.85938 111.32812 c +401.85938 111.32812 447.26562 111.32812 492.92188 129.14062 c +492.92188 129.14062 538.57812 146.96875 571.29688 172.35937 c +571.29688 172.35937 571.29688 263.1875 571.29688 263.1875 c +571.29688 263.1875 405.76562 263.1875 405.76562 263.1875 c +h +f* +Q +} def +/M { +762 0 71 0 762 716 setcachedevice +q +70.79688 0 m +70.79688 0 70.79688 715.82812 70.79688 715.82812 c +70.79688 715.82812 287.10938 715.82812 287.10938 715.82812 c +287.10938 715.82812 417 227.54688 417 227.54688 c +417 227.54688 545.40625 715.82812 545.40625 715.82812 c +545.40625 715.82812 762.20312 715.82812 762.20312 715.82812 c +762.20312 715.82812 762.20312 0 762.20312 0 c +762.20312 0 627.9375 0 627.9375 0 c +627.9375 0 627.9375 563.48438 627.9375 563.48438 c +627.9375 563.48438 485.84375 0 485.84375 0 c +485.84375 0 346.6875 0 346.6875 0 c +346.6875 0 205.07812 563.48438 205.07812 563.48438 c +205.07812 563.48438 205.07812 0 205.07812 0 c +205.07812 0 70.79688 0 70.79688 0 c +h +f* +Q +} def +/R { +717 0 73 0 717 716 setcachedevice +q +73.25 0 m +73.25 0 73.25 715.82812 73.25 715.82812 c +73.25 715.82812 377.4375 715.82812 377.4375 715.82812 c +377.4375 715.82812 492.1875 715.82812 544.1875 696.53125 c +544.1875 696.53125 596.1875 677.25 627.4375 627.92188 c +627.4375 627.92188 658.6875 578.60938 658.6875 515.14062 c +658.6875 515.14062 658.6875 434.57812 611.32812 382.07812 c +611.32812 382.07812 563.96875 329.59375 469.73438 315.92188 c +469.73438 315.92188 516.60938 288.57812 547.125 255.85938 c +547.125 255.85938 577.64062 223.14062 629.39062 139.65625 c +629.39062 139.65625 716.79688 0 716.79688 0 c +716.79688 0 543.95312 0 543.95312 0 c +543.95312 0 439.45312 155.76563 439.45312 155.76563 c +439.45312 155.76563 383.79688 239.26562 363.28125 260.98438 c +363.28125 260.98438 342.78125 282.71875 319.82812 290.76562 c +319.82812 290.76562 296.875 298.82812 247.07812 298.82812 c +247.07812 298.82812 217.78125 298.82812 217.78125 298.82812 c +217.78125 298.82812 217.78125 0 217.78125 0 c +217.78125 0 73.25 0 73.25 0 c +73.25 0 73.25 0 73.25 0 c +h +217.78125 413.09375 m +217.78125 413.09375 324.70312 413.09375 324.70312 413.09375 c +324.70312 413.09375 428.71875 413.09375 454.59375 421.875 c +454.59375 421.875 480.46875 430.67188 495.10938 452.15625 c +495.10938 452.15625 509.76562 473.64062 509.76562 505.85938 c +509.76562 505.85938 509.76562 542 490.46875 564.20312 c +490.46875 564.20312 471.1875 586.42188 436.03125 592.28125 c +436.03125 592.28125 418.45312 594.73438 330.5625 594.73438 c +330.5625 594.73438 217.78125 594.73438 217.78125 594.73438 c +217.78125 594.73438 217.78125 413.09375 217.78125 413.09375 c +h +f* +Q +} def +/T { +590 0 21 0 590 716 setcachedevice +q +233.89062 0 m +233.89062 0 233.89062 594.73438 233.89062 594.73438 c +233.89062 594.73438 21.48438 594.73438 21.48438 594.73438 c +21.48438 594.73438 21.48438 715.82812 21.48438 715.82812 c +21.48438 715.82812 590.32812 715.82812 590.32812 715.82812 c +590.32812 715.82812 590.32812 594.73438 590.32812 594.73438 c +590.32812 594.73438 378.42188 594.73438 378.42188 594.73438 c +378.42188 594.73438 378.42188 0 378.42188 0 c +378.42188 0 233.89062 0 233.89062 0 c +h +f* +Q +} def +/Y { +668 0 -1 0 668 716 setcachedevice +q +260.75 0 m +260.75 0 260.75 301.26562 260.75 301.26562 c +260.75 301.26562 -1.46875 715.82812 -1.46875 715.82812 c +-1.46875 715.82812 167.96875 715.82812 167.96875 715.82812 c +167.96875 715.82812 336.42188 432.625 336.42188 432.625 c +336.42188 432.625 501.46875 715.82812 501.46875 715.82812 c +501.46875 715.82812 667.96875 715.82812 667.96875 715.82812 c +667.96875 715.82812 404.78125 300.29688 404.78125 300.29688 c +404.78125 300.29688 404.78125 0 404.78125 0 c +404.78125 0 260.75 0 260.75 0 c +h +f* +Q +} def +/a { +522 0 36 -12 522 530 setcachedevice +q +174.3125 360.35938 m +174.3125 360.35938 49.8125 382.8125 49.8125 382.8125 c +49.8125 382.8125 70.79688 458.01562 122.0625 494.14062 c +122.0625 494.14062 173.34375 530.28125 274.42188 530.28125 c +274.42188 530.28125 366.21875 530.28125 411.14062 508.54688 c +411.14062 508.54688 456.0625 486.8125 474.35938 453.35938 c +474.35938 453.35938 492.67188 419.92188 492.67188 330.5625 c +492.67188 330.5625 491.21875 170.40625 491.21875 170.40625 c +491.21875 170.40625 491.21875 102.04687 497.79688 69.57812 c +497.79688 69.57812 504.39062 37.10938 522.46875 0 c +522.46875 0 386.71875 0 386.71875 0 c +386.71875 0 381.34375 13.67188 373.53125 40.53125 c +373.53125 40.53125 370.125 52.73438 368.65625 56.64062 c +368.65625 56.64062 333.5 22.46875 293.45312 5.375 c +293.45312 5.375 253.42188 -11.71875 208.01562 -11.71875 c +208.01562 -11.71875 127.9375 -11.71875 81.78125 31.73437 c +81.78125 31.73437 35.64062 75.20312 35.64062 141.60938 c +35.64062 141.60938 35.64062 185.54688 56.64062 219.96875 c +56.64062 219.96875 77.64062 254.39062 115.48438 272.70312 c +115.48438 272.70312 153.32812 291.01562 224.60938 304.6875 c +224.60938 304.6875 320.79688 322.75 357.90625 338.375 c +357.90625 338.375 357.90625 352.04688 357.90625 352.04688 c +357.90625 352.04688 357.90625 391.60938 338.375 408.45312 c +338.375 408.45312 318.84375 425.29688 264.65625 425.29688 c +264.65625 425.29688 228.03125 425.29688 207.51562 410.89062 c +207.51562 410.89062 187.01562 396.48438 174.3125 360.35938 c +174.3125 360.35938 174.3125 360.35938 174.3125 360.35938 c +h +357.90625 249.03125 m +357.90625 249.03125 331.54688 240.23438 274.40625 228.03125 c +274.40625 228.03125 217.28125 215.82812 199.70312 204.10938 c +199.70312 204.10938 172.85938 185.0625 172.85938 155.76563 c +172.85938 155.76563 172.85938 126.95312 194.34375 105.95312 c +194.34375 105.95312 215.82812 84.96875 249.03125 84.96875 c +249.03125 84.96875 286.14062 84.96875 319.82812 109.375 c +319.82812 109.375 344.73438 127.9375 352.54688 154.78125 c +352.54688 154.78125 357.90625 172.35937 357.90625 221.6875 c +357.90625 221.6875 357.90625 249.03125 357.90625 249.03125 c +h +f* +Q +} def +/d { +547 0 41 -12 547 716 setcachedevice +q +547.35938 0 m +547.35938 0 419.92188 0 419.92188 0 c +419.92188 0 419.92188 76.17188 419.92188 76.17188 c +419.92188 76.17188 388.1875 31.73437 344.96875 10 c +344.96875 10 301.76562 -11.71875 257.8125 -11.71875 c +257.8125 -11.71875 168.45312 -11.71875 104.73438 60.29688 c +104.73438 60.29688 41.01562 132.32813 41.01562 261.23438 c +41.01562 261.23438 41.01562 393.0625 103.03125 461.67188 c +103.03125 461.67188 165.04688 530.28125 259.76562 530.28125 c +259.76562 530.28125 346.6875 530.28125 410.15625 458.01562 c +410.15625 458.01562 410.15625 715.82812 410.15625 715.82812 c +410.15625 715.82812 547.35938 715.82812 547.35938 715.82812 c +547.35938 715.82812 547.35938 0 547.35938 0 c +547.35938 0 547.35938 0 547.35938 0 c +h +181.15625 270.51562 m +181.15625 270.51562 181.15625 187.5 204.10938 150.39062 c +204.10938 150.39062 237.3125 96.6875 296.875 96.6875 c +296.875 96.6875 344.23438 96.6875 377.4375 136.96875 c +377.4375 136.96875 410.64062 177.25 410.64062 257.32812 c +410.64062 257.32812 410.64062 346.6875 378.40625 385.98438 c +378.40625 385.98438 346.1875 425.29688 295.90625 425.29688 c +295.90625 425.29688 247.07812 425.29688 214.10938 386.46875 c +214.10938 386.46875 181.15625 347.65625 181.15625 270.51562 c +h +f* +Q +} def +/e { +519 0 32 -12 519 530 setcachedevice +q +372.07812 165.04688 m +372.07812 165.04688 508.79688 142.09375 508.79688 142.09375 c +508.79688 142.09375 482.42188 66.89062 425.53125 27.57812 c +425.53125 27.57812 368.65625 -11.71875 283.20312 -11.71875 c +283.20312 -11.71875 147.95312 -11.71875 83.01562 76.65625 c +83.01562 76.65625 31.73437 147.46875 31.73437 255.375 c +31.73437 255.375 31.73437 384.28125 99.10938 457.28125 c +99.10938 457.28125 166.5 530.28125 269.53125 530.28125 c +269.53125 530.28125 385.25 530.28125 452.14062 453.85938 c +452.14062 453.85938 519.04688 377.4375 516.10938 219.73438 c +516.10938 219.73438 172.35938 219.73438 172.35938 219.73438 c +172.35938 219.73438 173.82812 158.6875 205.5625 124.75 c +205.5625 124.75 237.3125 90.82812 284.67188 90.82812 c +284.67188 90.82812 316.89062 90.82812 338.85938 108.40625 c +338.85938 108.40625 360.84375 125.98438 372.07812 165.04688 c +372.07812 165.04688 372.07812 165.04688 372.07812 165.04688 c +h +379.89062 303.71875 m +379.89062 303.71875 378.42188 363.28125 349.125 394.28125 c +349.125 394.28125 319.82812 425.29688 277.82812 425.29688 c +277.82812 425.29688 232.90625 425.29688 203.60938 392.57812 c +203.60938 392.57812 174.3125 359.85938 174.8125 303.71875 c +174.8125 303.71875 379.89062 303.71875 379.89062 303.71875 c +h +f* +Q +} def +/f { +362 0 12 0 362 728 setcachedevice +q +11.71875 518.5625 m +11.71875 518.5625 87.89062 518.5625 87.89062 518.5625 c +87.89062 518.5625 87.89062 557.625 87.89062 557.625 c +87.89062 557.625 87.89062 623.04688 101.79688 655.26562 c +101.79688 655.26562 115.71875 687.5 153.07812 707.76562 c +153.07812 707.76562 190.4375 728.03125 247.5625 728.03125 c +247.5625 728.03125 306.15625 728.03125 362.3125 710.45312 c +362.3125 710.45312 343.75 614.75 343.75 614.75 c +343.75 614.75 311.03125 622.5625 280.76562 622.5625 c +280.76562 622.5625 250.98438 622.5625 238.03125 608.64062 c +238.03125 608.64062 225.09375 594.73438 225.09375 555.17188 c +225.09375 555.17188 225.09375 518.5625 225.09375 518.5625 c +225.09375 518.5625 327.64062 518.5625 327.64062 518.5625 c +327.64062 518.5625 327.64062 410.64062 327.64062 410.64062 c +327.64062 410.64062 225.09375 410.64062 225.09375 410.64062 c +225.09375 410.64062 225.09375 0 225.09375 0 c +225.09375 0 87.89062 0 87.89062 0 c +87.89062 0 87.89062 410.64062 87.89062 410.64062 c +87.89062 410.64062 11.71875 410.64062 11.71875 410.64062 c +11.71875 410.64062 11.71875 518.5625 11.71875 518.5625 c +h +f* +Q +} def +/g { +547 0 41 -210 547 530 setcachedevice +q +59.07812 -34.1875 m +59.07812 -34.1875 215.82812 -53.21875 215.82812 -53.21875 c +215.82812 -53.21875 219.73438 -80.5625 233.89062 -90.82812 c +233.89062 -90.82812 253.42188 -105.46875 295.40625 -105.46875 c +295.40625 -105.46875 349.125 -105.46875 375.98438 -89.35938 c +375.98438 -89.35938 394.04688 -78.60938 403.32812 -54.6875 c +403.32812 -54.6875 409.67188 -37.59375 409.67188 8.29687 c +409.67188 8.29687 409.67188 83.98438 409.67188 83.98438 c +409.67188 83.98438 348.14062 0 254.39062 0 c +254.39062 0 149.90625 0 88.875 88.375 c +88.875 88.375 41.01562 158.20312 41.01562 262.20312 c +41.01562 262.20312 41.01562 392.57812 103.75 461.42188 c +103.75 461.42188 166.5 530.28125 259.76562 530.28125 c +259.76562 530.28125 355.95312 530.28125 418.45312 445.79688 c +418.45312 445.79688 418.45312 518.5625 418.45312 518.5625 c +418.45312 518.5625 546.875 518.5625 546.875 518.5625 c +546.875 518.5625 546.875 53.21875 546.875 53.21875 c +546.875 53.21875 546.875 -38.57812 531.73438 -83.98438 c +531.73438 -83.98438 516.60938 -129.39062 489.26562 -155.26562 c +489.26562 -155.26562 461.92188 -181.15625 416.26563 -195.79688 c +416.26563 -195.79688 370.60938 -210.45312 300.78125 -210.45312 c +300.78125 -210.45312 168.95312 -210.45312 113.76562 -165.28125 c +113.76562 -165.28125 58.59375 -120.125 58.59375 -50.78125 c +58.59375 -50.78125 58.59375 -43.95312 59.07812 -34.1875 c +59.07812 -34.1875 59.07812 -34.1875 59.07812 -34.1875 c +h +181.64062 270.01562 m +181.64062 270.01562 181.64062 187.5 213.625 149.17188 c +213.625 149.17188 245.60938 110.84375 292.48438 110.84375 c +292.48438 110.84375 342.78125 110.84375 377.4375 150.14062 c +377.4375 150.14062 412.10938 189.45312 412.10938 266.60938 c +412.10938 266.60938 412.10938 347.17188 378.90625 386.23438 c +378.90625 386.23438 345.70312 425.29688 294.92188 425.29688 c +294.92188 425.29688 245.60938 425.29688 213.625 386.96875 c +213.625 386.96875 181.64062 348.64062 181.64062 270.01562 c +h +f* +Q +} def +/h { +543 0 71 0 543 716 setcachedevice +q +208.5 715.82812 m +208.5 715.82812 208.5 452.64062 208.5 452.64062 c +208.5 452.64062 274.90625 530.28125 367.1875 530.28125 c +367.1875 530.28125 414.54688 530.28125 452.625 512.70312 c +452.625 512.70312 490.71875 495.125 510 467.78125 c +510 467.78125 529.29688 440.4375 536.375 407.23438 c +536.375 407.23438 543.45312 374.03125 543.45312 304.20312 c +543.45312 304.20312 543.45312 0 543.45312 0 c +543.45312 0 406.25 0 406.25 0 c +406.25 0 406.25 273.92188 406.25 273.92188 c +406.25 273.92188 406.25 355.46875 398.4375 377.4375 c +398.4375 377.4375 390.625 399.42188 370.84375 412.35938 c +370.84375 412.35938 351.07812 425.29688 321.29688 425.29688 c +321.29688 425.29688 287.10938 425.29688 260.25 408.6875 c +260.25 408.6875 233.40625 392.09375 220.95312 358.64062 c +220.95312 358.64062 208.5 325.20312 208.5 259.76562 c +208.5 259.76562 208.5 0 208.5 0 c +208.5 0 71.29688 0 71.29688 0 c +71.29688 0 71.29688 715.82812 71.29688 715.82812 c +71.29688 715.82812 208.5 715.82812 208.5 715.82812 c +h +f* +Q +} def +/i { +209 0 72 0 209 716 setcachedevice +q +71.78125 588.875 m +71.78125 588.875 71.78125 715.82812 71.78125 715.82812 c +71.78125 715.82812 208.98438 715.82812 208.98438 715.82812 c +208.98438 715.82812 208.98438 588.875 208.98438 588.875 c +208.98438 588.875 71.78125 588.875 71.78125 588.875 c +71.78125 588.875 71.78125 588.875 71.78125 588.875 c +h +71.78125 0 m +71.78125 0 71.78125 518.5625 71.78125 518.5625 c +71.78125 518.5625 208.98438 518.5625 208.98438 518.5625 c +208.98438 518.5625 208.98438 0 208.98438 0 c +208.98438 0 71.78125 0 71.78125 0 c +h +f* +Q +} def +/l { +209 0 72 0 209 716 setcachedevice +q +71.78125 0 m +71.78125 0 71.78125 715.82812 71.78125 715.82812 c +71.78125 715.82812 208.98438 715.82812 208.98438 715.82812 c +208.98438 715.82812 208.98438 0 208.98438 0 c +208.98438 0 71.78125 0 71.78125 0 c +h +f* +Q +} def +/n { +543 0 71 0 543 530 setcachedevice +q +543.45312 0 m +543.45312 0 406.25 0 406.25 0 c +406.25 0 406.25 264.65625 406.25 264.65625 c +406.25 264.65625 406.25 348.64062 397.45312 373.29688 c +397.45312 373.29688 388.67188 397.95312 368.89062 411.625 c +368.89062 411.625 349.125 425.29688 321.29688 425.29688 c +321.29688 425.29688 285.64062 425.29688 257.3125 405.76562 c +257.3125 405.76562 229 386.23438 218.5 354 c +218.5 354 208.01562 321.78125 208.01562 234.85938 c +208.01562 234.85938 208.01562 0 208.01562 0 c +208.01562 0 70.79688 0 70.79688 0 c +70.79688 0 70.79688 518.5625 70.79688 518.5625 c +70.79688 518.5625 198.25 518.5625 198.25 518.5625 c +198.25 518.5625 198.25 442.39062 198.25 442.39062 c +198.25 442.39062 266.10938 530.28125 369.14062 530.28125 c +369.14062 530.28125 414.54688 530.28125 452.14062 513.92188 c +452.14062 513.92188 489.75 497.5625 509.03125 472.17188 c +509.03125 472.17188 528.32812 446.78125 535.89062 414.54688 c +535.89062 414.54688 543.45312 382.32812 543.45312 322.26562 c +543.45312 322.26562 543.45312 0 543.45312 0 c +h +f* +Q +} def +/o { +575 0 40 -12 575 530 setcachedevice +q +40.04688 266.60938 m +40.04688 266.60938 40.04688 334.96875 73.73438 398.92188 c +73.73438 398.92188 107.42188 462.89062 169.1875 496.57812 c +169.1875 496.57812 230.95312 530.28125 307.125 530.28125 c +307.125 530.28125 424.8125 530.28125 500 453.85938 c +500 453.85938 575.20312 377.4375 575.20312 260.75 c +575.20312 260.75 575.20312 143.0625 499.26563 65.67188 c +499.26563 65.67188 423.34375 -11.71875 308.10938 -11.71875 c +308.10938 -11.71875 236.8125 -11.71875 172.10938 20.5 c +172.10938 20.5 107.42188 52.73438 73.73438 114.98438 c +73.73438 114.98438 40.04688 177.25 40.04688 266.60938 c +40.04688 266.60938 40.04688 266.60938 40.04688 266.60938 c +h +180.67188 259.28125 m +180.67188 259.28125 180.67188 182.125 217.28125 141.10937 c +217.28125 141.10937 253.90625 100.09375 307.625 100.09375 c +307.625 100.09375 361.32812 100.09375 397.70312 141.10937 c +397.70312 141.10937 434.07812 182.125 434.07812 260.25 c +434.07812 260.25 434.07812 336.42188 397.70312 377.4375 c +397.70312 377.4375 361.32812 418.45312 307.625 418.45312 c +307.625 418.45312 253.90625 418.45312 217.28125 377.4375 c +217.28125 377.4375 180.67188 336.42188 180.67188 259.28125 c +h +f* +Q +} def +/p { +574 0 68 -197 574 530 setcachedevice +q +67.875 518.5625 m +67.875 518.5625 195.79688 518.5625 195.79688 518.5625 c +195.79688 518.5625 195.79688 442.39062 195.79688 442.39062 c +195.79688 442.39062 220.70312 481.45312 263.1875 505.85938 c +263.1875 505.85938 305.67188 530.28125 357.42188 530.28125 c +357.42188 530.28125 447.75 530.28125 510.73438 459.46875 c +510.73438 459.46875 573.73438 388.67188 573.73438 262.20312 c +573.73438 262.20312 573.73438 132.32813 510.25 60.29688 c +510.25 60.29688 446.78125 -11.71875 356.45312 -11.71875 c +356.45312 -11.71875 313.48438 -11.71875 278.5625 5.375 c +278.5625 5.375 243.65625 22.46875 205.07812 63.96875 c +205.07812 63.96875 205.07812 -197.26562 205.07812 -197.26562 c +205.07812 -197.26562 67.875 -197.26562 67.875 -197.26562 c +67.875 -197.26562 67.875 518.5625 67.875 518.5625 c +67.875 518.5625 67.875 518.5625 67.875 518.5625 c +h +203.60938 268.0625 m +203.60938 268.0625 203.60938 180.67188 238.28125 138.92188 c +238.28125 138.92188 272.95312 97.17188 322.75 97.17188 c +322.75 97.17188 370.60938 97.17188 402.34375 135.5 c +402.34375 135.5 434.07812 173.82812 434.07812 261.23438 c +434.07812 261.23438 434.07812 342.78125 401.35938 382.32812 c +401.35938 382.32812 368.65625 421.875 320.3125 421.875 c +320.3125 421.875 270.01562 421.875 236.8125 383.04688 c +236.8125 383.04688 203.60938 344.23438 203.60938 268.0625 c +h +f* +Q +} def +/r { +402 0 66 0 402 530 setcachedevice +q +203.125 0 m +203.125 0 65.92188 0 65.92188 0 c +65.92188 0 65.92188 518.5625 65.92188 518.5625 c +65.92188 518.5625 193.35938 518.5625 193.35938 518.5625 c +193.35938 518.5625 193.35938 444.82812 193.35938 444.82812 c +193.35938 444.82812 226.07812 497.07812 252.20312 513.67188 c +252.20312 513.67188 278.32812 530.28125 311.53125 530.28125 c +311.53125 530.28125 358.40625 530.28125 401.85938 504.39062 c +401.85938 504.39062 359.375 384.76562 359.375 384.76562 c +359.375 384.76562 324.70312 407.23438 294.92188 407.23438 c +294.92188 407.23438 266.10938 407.23438 246.09375 391.35938 c +246.09375 391.35938 226.07812 375.48438 214.59375 333.98438 c +214.59375 333.98438 203.125 292.48438 203.125 160.15625 c +203.125 160.15625 203.125 0 203.125 0 c +h +f* +Q +} def +/t { +321 0 15 -12 321 702 setcachedevice +q +309.57812 518.5625 m +309.57812 518.5625 309.57812 409.1875 309.57812 409.1875 c +309.57812 409.1875 215.82812 409.1875 215.82812 409.1875 c +215.82812 409.1875 215.82812 200.20312 215.82812 200.20312 c +215.82812 200.20312 215.82812 136.71875 218.5 126.21875 c +218.5 126.21875 221.1875 115.71875 230.70312 108.875 c +230.70312 108.875 240.23438 102.04687 253.90625 102.04687 c +253.90625 102.04687 272.95312 102.04687 309.07812 115.23438 c +309.07812 115.23438 320.79688 8.79688 320.79688 8.79688 c +320.79688 8.79688 272.95312 -11.71875 212.40625 -11.71875 c +212.40625 -11.71875 175.29688 -11.71875 145.5 0.73438 c +145.5 0.73438 115.71875 13.1875 101.79688 32.95312 c +101.79688 32.95312 87.89062 52.73438 82.51562 86.42187 c +82.51562 86.42187 78.125 110.35938 78.125 183.10938 c +78.125 183.10938 78.125 409.1875 78.125 409.1875 c +78.125 409.1875 15.14062 409.1875 15.14062 409.1875 c +15.14062 409.1875 15.14062 518.5625 15.14062 518.5625 c +15.14062 518.5625 78.125 518.5625 78.125 518.5625 c +78.125 518.5625 78.125 621.57812 78.125 621.57812 c +78.125 621.57812 215.82812 701.65625 215.82812 701.65625 c +215.82812 701.65625 215.82812 518.5625 215.82812 518.5625 c +215.82812 518.5625 309.57812 518.5625 309.57812 518.5625 c +h +f* +Q +} def +/u { +541 0 69 -12 541 519 setcachedevice +q +413.09375 0 m +413.09375 0 413.09375 77.64063 413.09375 77.64063 c +413.09375 77.64063 384.76562 36.14062 338.625 12.20312 c +338.625 12.20312 292.48438 -11.71875 241.21875 -11.71875 c +241.21875 -11.71875 188.96875 -11.71875 147.45312 11.23438 c +147.45312 11.23438 105.95312 34.1875 87.39062 75.6875 c +87.39062 75.6875 68.84375 117.1875 68.84375 190.4375 c +68.84375 190.4375 68.84375 518.5625 68.84375 518.5625 c +68.84375 518.5625 206.0625 518.5625 206.0625 518.5625 c +206.0625 518.5625 206.0625 280.28125 206.0625 280.28125 c +206.0625 280.28125 206.0625 170.90625 213.625 146.23438 c +213.625 146.23438 221.1875 121.57812 241.20312 107.17188 c +241.20312 107.17188 261.23438 92.78125 292 92.78125 c +292 92.78125 327.15625 92.78125 354.98438 112.0625 c +354.98438 112.0625 382.8125 131.34375 393.0625 159.90625 c +393.0625 159.90625 403.32812 188.48438 403.32812 299.8125 c +403.32812 299.8125 403.32812 518.5625 403.32812 518.5625 c +403.32812 518.5625 540.53125 518.5625 540.53125 518.5625 c +540.53125 518.5625 540.53125 0 540.53125 0 c +540.53125 0 413.09375 0 413.09375 0 c +h +f* +Q +} def +/w { +777 0 4 0 777 519 setcachedevice +q +168.45312 0 m +168.45312 0 4.39062 518.5625 4.39062 518.5625 c +4.39062 518.5625 137.70312 518.5625 137.70312 518.5625 c +137.70312 518.5625 234.85938 178.71875 234.85938 178.71875 c +234.85938 178.71875 324.21875 518.5625 324.21875 518.5625 c +324.21875 518.5625 456.54688 518.5625 456.54688 518.5625 c +456.54688 518.5625 542.96875 178.71875 542.96875 178.71875 c +542.96875 178.71875 642.09375 518.5625 642.09375 518.5625 c +642.09375 518.5625 777.34375 518.5625 777.34375 518.5625 c +777.34375 518.5625 610.84375 0 610.84375 0 c +610.84375 0 479 0 479 0 c +479 0 389.65625 333.5 389.65625 333.5 c +389.65625 333.5 301.76562 0 301.76562 0 c +301.76562 0 168.45312 0 168.45312 0 c +h +f* +Q +} def +/y { +540 0 7 -210 540 519 setcachedevice +q +6.84375 518.5625 m +6.84375 518.5625 152.82812 518.5625 152.82812 518.5625 c +152.82812 518.5625 276.85938 150.39062 276.85938 150.39062 c +276.85938 150.39062 397.95312 518.5625 397.95312 518.5625 c +397.95312 518.5625 540.04688 518.5625 540.04688 518.5625 c +540.04688 518.5625 356.9375 19.53125 356.9375 19.53125 c +356.9375 19.53125 324.21875 -70.79688 324.21875 -70.79688 c +324.21875 -70.79688 306.15625 -116.21875 289.79688 -140.14062 c +289.79688 -140.14062 273.4375 -164.0625 252.1875 -178.95312 c +252.1875 -178.95312 230.95312 -193.84375 199.95312 -202.14062 c +199.95312 -202.14062 168.95312 -210.45312 129.89062 -210.45312 c +129.89062 -210.45312 90.32812 -210.45312 52.25 -202.15625 c +52.25 -202.15625 40.04688 -94.73438 40.04688 -94.73438 c +40.04688 -94.73438 72.26562 -101.07812 98.14062 -101.07812 c +98.14062 -101.07812 146 -101.07812 168.9375 -73 c +168.9375 -73 191.89062 -44.92188 204.10938 -1.46875 c +204.10938 -1.46875 6.84375 518.5625 6.84375 518.5625 c +h +f* +Q +} def +end +currentdict end +/T3_35_0 exch definefont pop +%%EndResource +/F35_0 /T3_35_0 1 1 +[ /B/C/G/M/R/T/Y/a + /d/e/f/g/h/i/l/n + /o/p/r/t/u/w/y/.notdef + /.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef + /space/exclam/quotedbl/numbersign/dollar/percent/ampersand/quoteright + /parenleft/parenright/asterisk/plus/comma/hyphen/period/slash + /zero/one/two/three/four/five/six/seven + /eight/nine/colon/semicolon/less/equal/greater/question + /at/A/B/C/D/E/F/G + /H/I/J/K/L/M/N/O + /P/Q/R/S/T/U/V/W + /X/Y/Z/bracketleft/backslash/bracketright/asciicircum/underscore + /quoteleft/a/b/c/d/e/f/g + /h/i/j/k/l/m/n/o + /p/q/r/s/t/u/v/w + /x/y/z/braceleft/bar/braceright/asciitilde/.notdef + /.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef + /.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef + /.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef + /.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef + /.notdef/exclamdown/cent/sterling/fraction/yen/florin/section + /currency/quotesingle/quotedblleft/guillemotleft/guilsinglleft/guilsinglright/fi/fl + /.notdef/endash/dagger/daggerdbl/periodcentered/.notdef/paragraph/bullet + /quotesinglbase/quotedblbase/quotedblright/guillemotright/ellipsis/perthousand/.notdef/questiondown + /.notdef/grave/acute/circumflex/tilde/macron/breve/dotaccent + /dieresis/.notdef/ring/cedilla/.notdef/hungarumlaut/ogonek/caron + /emdash/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef + /.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef + /.notdef/AE/.notdef/ordfeminine/.notdef/.notdef/.notdef/.notdef + /Lslash/Oslash/OE/ordmasculine/.notdef/.notdef/.notdef/.notdef + /.notdef/ae/.notdef/.notdef/.notdef/dotlessi/.notdef/.notdef + /lslash/oslash/oe/germandbls/.notdef/.notdef/.notdef/.notdef] +pdfMakeFont +%%BeginResource: font T3_82_0 +8 dict begin +/FontType 3 def +/FontMatrix [0.001 0 0 0.001 0 0] def +/FontBBox [-46 -210 769 729] def +/Encoding 256 array def + 0 1 255 { Encoding exch /.notdef put } for +/BuildGlyph { + exch /CharProcs get exch + 2 copy known not { pop /.notdef } if + get exec +} bind def +/BuildChar { + 1 index /Encoding get exch get + 1 index /BuildGlyph get exec +} bind def +/CharProcs 43 dict def +CharProcs begin +/comma { +189 0 83 -142 189 100 setcachedevice +q +88.875 0 m +88.875 0 88.875 100.09375 88.875 100.09375 c +88.875 100.09375 188.96875 100.09375 188.96875 100.09375 c +188.96875 100.09375 188.96875 0 188.96875 0 c +188.96875 0 188.96875 -55.17188 169.4375 -89.10938 c +169.4375 -89.10938 149.90625 -123.04688 107.42188 -141.60938 c +107.42188 -141.60938 83.01562 -104 83.01562 -104 c +83.01562 -104 110.84375 -91.79688 124.01562 -68.10938 c +124.01562 -68.10938 137.20312 -44.4375 138.67188 0 c +138.67188 0 88.875 0 88.875 0 c +h +f* +Q +} def +/hyphen { +302 0 32 215 302 303 setcachedevice +q +31.73437 214.84375 m +31.73437 214.84375 31.73437 303.21875 31.73437 303.21875 c +31.73437 303.21875 301.76562 303.21875 301.76562 303.21875 c +301.76562 303.21875 301.76562 214.84375 301.76562 214.84375 c +301.76562 214.84375 31.73437 214.84375 31.73437 214.84375 c +h +f* +Q +} def +/period { +191 0 91 0 191 100 setcachedevice +q +90.82812 0 m +90.82812 0 90.82812 100.09375 90.82812 100.09375 c +90.82812 100.09375 190.92188 100.09375 190.92188 100.09375 c +190.92188 100.09375 190.92188 0 190.92188 0 c +190.92188 0 90.82812 0 90.82812 0 c +h +f* +Q +} def +/one { +373 0 109 0 373 719 setcachedevice +q +372.5625 0 m +372.5625 0 284.67188 0 284.67188 0 c +284.67188 0 284.67188 560.0625 284.67188 560.0625 c +284.67188 560.0625 252.9375 529.78125 201.42188 499.5 c +201.42188 499.5 149.90625 469.23438 108.89062 454.10938 c +108.89062 454.10938 108.89062 539.0625 108.89062 539.0625 c +108.89062 539.0625 182.625 573.73438 237.79688 623.04688 c +237.79688 623.04688 292.96875 672.35938 315.92188 718.75 c +315.92188 718.75 372.5625 718.75 372.5625 718.75 c +372.5625 718.75 372.5625 0 372.5625 0 c +h +f* +Q +} def +/two { +503 0 29 0 503 719 setcachedevice +q +503.42188 84.46875 m +503.42188 84.46875 503.42188 0 503.42188 0 c +503.42188 0 30.28125 0 30.28125 0 c +30.28125 0 29.29688 31.73437 40.53125 61.03125 c +40.53125 61.03125 58.59375 109.375 98.39062 156.25 c +98.39062 156.25 138.1875 203.125 213.375 264.65625 c +213.375 264.65625 330.07812 360.35938 371.09375 416.26562 c +371.09375 416.26562 412.10938 472.17188 412.10938 521.96875 c +412.10938 521.96875 412.10938 574.21875 374.75 610.10938 c +374.75 610.10938 337.40625 646 277.34375 646 c +277.34375 646 213.875 646 175.78125 607.90625 c +175.78125 607.90625 137.70312 569.82812 137.20312 502.4375 c +137.20312 502.4375 46.875 511.71875 46.875 511.71875 c +46.875 511.71875 56.15625 612.79688 116.70312 665.76562 c +116.70312 665.76562 177.25 718.75 279.29688 718.75 c +279.29688 718.75 382.32812 718.75 442.375 661.625 c +442.375 661.625 502.4375 604.5 502.4375 520.01562 c +502.4375 520.01562 502.4375 477.04688 484.85938 435.54688 c +484.85938 435.54688 467.28125 394.04688 426.51563 348.14062 c +426.51563 348.14062 385.75 302.25 291.01562 222.17188 c +291.01562 222.17188 211.92188 155.76563 189.45312 132.07812 c +189.45312 132.07812 167 108.40625 152.34375 84.46875 c +152.34375 84.46875 503.42188 84.46875 503.42188 84.46875 c +h +f* +Q +} def +/three { +511 0 42 -13 511 719 setcachedevice +q +42 188.96875 m +42 188.96875 129.89062 200.6875 129.89062 200.6875 c +129.89062 200.6875 145.01562 125.98438 181.39062 93.01562 c +181.39062 93.01562 217.78125 60.0625 270.01562 60.0625 c +270.01562 60.0625 332.03125 60.0625 374.75 103.03125 c +374.75 103.03125 417.48438 146 417.48438 209.46875 c +417.48438 209.46875 417.48438 270.01562 377.92188 309.32812 c +377.92188 309.32812 338.375 348.64062 277.34375 348.64062 c +277.34375 348.64062 252.4375 348.64062 215.32812 338.875 c +215.32812 338.875 225.09375 416.01562 225.09375 416.01562 c +225.09375 416.01562 233.89062 415.04688 239.26562 415.04688 c +239.26562 415.04688 295.40625 415.04688 340.32812 444.34375 c +340.32812 444.34375 385.25 473.64062 385.25 534.67188 c +385.25 534.67188 385.25 583.01562 352.53125 614.75 c +352.53125 614.75 319.82812 646.48438 268.0625 646.48438 c +268.0625 646.48438 216.79688 646.48438 182.60938 614.25 c +182.60938 614.25 148.4375 582.03125 138.67188 517.57812 c +138.67188 517.57812 50.78125 533.20312 50.78125 533.20312 c +50.78125 533.20312 66.89062 621.57812 124.01562 670.15625 c +124.01562 670.15625 181.15625 718.75 266.10938 718.75 c +266.10938 718.75 324.70312 718.75 374.01562 693.59375 c +374.01562 693.59375 423.34375 668.45312 449.46875 625 c +449.46875 625 475.59375 581.54688 475.59375 532.71875 c +475.59375 532.71875 475.59375 486.32812 450.6875 448.23438 c +450.6875 448.23438 425.78125 410.15625 376.95312 387.70312 c +376.95312 387.70312 440.4375 373.04688 475.59375 326.90625 c +475.59375 326.90625 510.75 280.76562 510.75 211.42188 c +510.75 211.42188 510.75 117.67187 442.39062 52.48438 c +442.39062 52.48438 374.03125 -12.70312 269.53125 -12.70312 c +269.53125 -12.70312 175.29688 -12.70312 113.03125 43.45312 c +113.03125 43.45312 50.78125 99.60938 42 188.96875 c +h +f* +Q +} def +/four { +508 0 13 0 508 716 setcachedevice +q +323.25 0 m +323.25 0 323.25 171.39063 323.25 171.39063 c +323.25 171.39063 12.70313 171.39063 12.70313 171.39063 c +12.70313 171.39063 12.70313 251.95312 12.70313 251.95312 c +12.70313 251.95312 339.35938 715.82812 339.35938 715.82812 c +339.35938 715.82812 411.14062 715.82812 411.14062 715.82812 c +411.14062 715.82812 411.14062 251.95312 411.14062 251.95312 c +411.14062 251.95312 507.8125 251.95312 507.8125 251.95312 c +507.8125 251.95312 507.8125 171.39063 507.8125 171.39063 c +507.8125 171.39063 411.14062 171.39063 411.14062 171.39063 c +411.14062 171.39063 411.14062 0 411.14062 0 c +411.14062 0 323.25 0 323.25 0 c +323.25 0 323.25 0 323.25 0 c +h +323.25 251.95312 m +323.25 251.95312 323.25 574.70312 323.25 574.70312 c +323.25 574.70312 99.125 251.95312 99.125 251.95312 c +99.125 251.95312 323.25 251.95312 323.25 251.95312 c +h +f* +Q +} def +/semicolon { +189 0 83 -142 189 519 setcachedevice +q +88.875 418.45312 m +88.875 418.45312 88.875 518.5625 88.875 518.5625 c +88.875 518.5625 188.96875 518.5625 188.96875 518.5625 c +188.96875 518.5625 188.96875 418.45312 188.96875 418.45312 c +188.96875 418.45312 88.875 418.45312 88.875 418.45312 c +88.875 418.45312 88.875 418.45312 88.875 418.45312 c +h +88.875 0 m +88.875 0 88.875 100.09375 88.875 100.09375 c +88.875 100.09375 188.96875 100.09375 188.96875 100.09375 c +188.96875 100.09375 188.96875 0 188.96875 0 c +188.96875 0 188.96875 -55.17188 169.4375 -89.10938 c +169.4375 -89.10938 149.90625 -123.04688 107.42188 -141.60938 c +107.42188 -141.60938 83.01562 -104 83.01562 -104 c +83.01562 -104 110.84375 -91.79688 124.01562 -68.10938 c +124.01562 -68.10938 137.20312 -44.4375 138.67188 0 c +138.67188 0 88.875 0 88.875 0 c +h +f* +Q +} def +/A { +668 0 -1 0 668 716 setcachedevice +q +-1.46875 0 m +-1.46875 0 273.4375 715.82812 273.4375 715.82812 c +273.4375 715.82812 375.48438 715.82812 375.48438 715.82812 c +375.48438 715.82812 668.45312 0 668.45312 0 c +668.45312 0 560.54688 0 560.54688 0 c +560.54688 0 477.04688 216.79688 477.04688 216.79688 c +477.04688 216.79688 177.73438 216.79688 177.73438 216.79688 c +177.73438 216.79688 99.125 0 99.125 0 c +99.125 0 -1.46875 0 -1.46875 0 c +-1.46875 0 -1.46875 0 -1.46875 0 c +h +205.07812 293.95312 m +205.07812 293.95312 447.75 293.95312 447.75 293.95312 c +447.75 293.95312 373.04688 492.1875 373.04688 492.1875 c +373.04688 492.1875 338.875 582.51562 322.26562 640.625 c +322.26562 640.625 308.59375 571.78125 283.6875 503.90625 c +283.6875 503.90625 205.07812 293.95312 205.07812 293.95312 c +h +f* +Q +} def +/C { +683 0 50 -12 683 728 setcachedevice +q +587.89062 250.98438 m +587.89062 250.98438 682.625 227.04688 682.625 227.04688 c +682.625 227.04688 652.82812 110.35938 575.4375 49.07812 c +575.4375 49.07812 498.04688 -12.20312 386.23438 -12.20312 c +386.23438 -12.20312 270.51562 -12.20312 198 34.90625 c +198 34.90625 125.48438 82.03125 87.64062 171.39063 c +87.64062 171.39063 49.8125 260.75 49.8125 363.28125 c +49.8125 363.28125 49.8125 475.09375 92.53125 558.34375 c +92.53125 558.34375 135.25 641.60938 214.10938 684.8125 c +214.10938 684.8125 292.96875 728.03125 387.70312 728.03125 c +387.70312 728.03125 495.125 728.03125 568.35938 673.34375 c +568.35938 673.34375 641.60938 618.65625 670.40625 519.53125 c +670.40625 519.53125 577.15625 497.5625 577.15625 497.5625 c +577.15625 497.5625 552.25 575.6875 504.875 611.32812 c +504.875 611.32812 457.51562 646.96875 385.75 646.96875 c +385.75 646.96875 303.21875 646.96875 247.79688 607.42188 c +247.79688 607.42188 192.39063 567.875 169.92188 501.21875 c +169.92188 501.21875 147.46875 434.57812 147.46875 363.76562 c +147.46875 363.76562 147.46875 272.46875 174.07812 204.34375 c +174.07812 204.34375 200.6875 136.23438 256.82812 102.53125 c +256.82812 102.53125 312.98438 68.84375 378.42188 68.84375 c +378.42188 68.84375 458.01562 68.84375 513.1875 114.73438 c +513.1875 114.73438 568.35938 160.64062 587.89062 250.98438 c +h +f* +Q +} def +/D { +669 0 77 0 669 716 setcachedevice +q +77.15625 0 m +77.15625 0 77.15625 715.82812 77.15625 715.82812 c +77.15625 715.82812 323.73438 715.82812 323.73438 715.82812 c +323.73438 715.82812 407.23438 715.82812 451.17188 705.5625 c +451.17188 705.5625 512.70312 691.40625 556.15625 654.29688 c +556.15625 654.29688 612.79688 606.45312 640.875 531.98438 c +640.875 531.98438 668.95312 457.51562 668.95312 361.8125 c +668.95312 361.8125 668.95312 280.28125 649.90625 217.28125 c +649.90625 217.28125 630.85938 154.29688 601.07812 113.03125 c +601.07812 113.03125 571.29688 71.78125 535.89062 48.09375 c +535.89062 48.09375 500.48438 24.42188 450.4375 12.20312 c +450.4375 12.20312 400.39062 0 335.45312 0 c +335.45312 0 77.15625 0 77.15625 0 c +77.15625 0 77.15625 0 77.15625 0 c +h +171.875 84.46875 m +171.875 84.46875 324.70312 84.46875 324.70312 84.46875 c +324.70312 84.46875 395.51562 84.46875 435.79688 97.65625 c +435.79688 97.65625 476.07812 110.84375 500 134.76562 c +500 134.76562 533.6875 168.45312 552.48438 225.34375 c +552.48438 225.34375 571.29688 282.23438 571.29688 363.28125 c +571.29688 363.28125 571.29688 475.59375 534.42188 535.89062 c +534.42188 535.89062 497.5625 596.1875 444.82812 616.70312 c +444.82812 616.70312 406.73438 631.34375 322.26562 631.34375 c +322.26562 631.34375 171.875 631.34375 171.875 631.34375 c +171.875 631.34375 171.875 84.46875 171.875 84.46875 c +h +f* +Q +} def +/E { +613 0 79 0 613 716 setcachedevice +q +79.10938 0 m +79.10938 0 79.10938 715.82812 79.10938 715.82812 c +79.10938 715.82812 596.6875 715.82812 596.6875 715.82812 c +596.6875 715.82812 596.6875 631.34375 596.6875 631.34375 c +596.6875 631.34375 173.82812 631.34375 173.82812 631.34375 c +173.82812 631.34375 173.82812 412.10938 173.82812 412.10938 c +173.82812 412.10938 569.82812 412.10938 569.82812 412.10938 c +569.82812 412.10938 569.82812 328.125 569.82812 328.125 c +569.82812 328.125 173.82812 328.125 173.82812 328.125 c +173.82812 328.125 173.82812 84.46875 173.82812 84.46875 c +173.82812 84.46875 613.28125 84.46875 613.28125 84.46875 c +613.28125 84.46875 613.28125 0 613.28125 0 c +613.28125 0 79.10938 0 79.10938 0 c +h +f* +Q +} def +/F { +565 0 82 0 565 716 setcachedevice +q +82.03125 0 m +82.03125 0 82.03125 715.82812 82.03125 715.82812 c +82.03125 715.82812 564.9375 715.82812 564.9375 715.82812 c +564.9375 715.82812 564.9375 631.34375 564.9375 631.34375 c +564.9375 631.34375 176.76562 631.34375 176.76562 631.34375 c +176.76562 631.34375 176.76562 409.67188 176.76562 409.67188 c +176.76562 409.67188 512.70312 409.67188 512.70312 409.67188 c +512.70312 409.67188 512.70312 325.20312 512.70312 325.20312 c +512.70312 325.20312 176.76562 325.20312 176.76562 325.20312 c +176.76562 325.20312 176.76562 0 176.76562 0 c +176.76562 0 82.03125 0 82.03125 0 c +h +f* +Q +} def +/I { +188 0 93 0 188 716 setcachedevice +q +93.26562 0 m +93.26562 0 93.26562 715.82812 93.26562 715.82812 c +93.26562 715.82812 187.98438 715.82812 187.98438 715.82812 c +187.98438 715.82812 187.98438 0 187.98438 0 c +187.98438 0 93.26562 0 93.26562 0 c +h +f* +Q +} def +/L { +521 0 73 0 521 716 setcachedevice +q +73.25 0 m +73.25 0 73.25 715.82812 73.25 715.82812 c +73.25 715.82812 167.96875 715.82812 167.96875 715.82812 c +167.96875 715.82812 167.96875 84.46875 167.96875 84.46875 c +167.96875 84.46875 520.51562 84.46875 520.51562 84.46875 c +520.51562 84.46875 520.51562 0 520.51562 0 c +520.51562 0 73.25 0 73.25 0 c +h +f* +Q +} def +/M { +757 0 74 0 757 716 setcachedevice +q +74.21875 0 m +74.21875 0 74.21875 715.82812 74.21875 715.82812 c +74.21875 715.82812 216.79688 715.82812 216.79688 715.82812 c +216.79688 715.82812 386.23438 208.98438 386.23438 208.98438 c +386.23438 208.98438 409.67188 138.1875 420.40625 103.03125 c +420.40625 103.03125 432.625 142.09375 458.5 217.78125 c +458.5 217.78125 629.89062 715.82812 629.89062 715.82812 c +629.89062 715.82812 757.32812 715.82812 757.32812 715.82812 c +757.32812 715.82812 757.32812 0 757.32812 0 c +757.32812 0 666.01562 0 666.01562 0 c +666.01562 0 666.01562 599.125 666.01562 599.125 c +666.01562 599.125 458.01562 0 458.01562 0 c +458.01562 0 372.5625 0 372.5625 0 c +372.5625 0 165.53125 609.375 165.53125 609.375 c +165.53125 609.375 165.53125 0 165.53125 0 c +165.53125 0 74.21875 0 74.21875 0 c +h +f* +Q +} def +/N { +640 0 76 0 640 716 setcachedevice +q +76.17188 0 m +76.17188 0 76.17188 715.82812 76.17188 715.82812 c +76.17188 715.82812 173.34375 715.82812 173.34375 715.82812 c +173.34375 715.82812 549.3125 153.8125 549.3125 153.8125 c +549.3125 153.8125 549.3125 715.82812 549.3125 715.82812 c +549.3125 715.82812 640.14062 715.82812 640.14062 715.82812 c +640.14062 715.82812 640.14062 0 640.14062 0 c +640.14062 0 542.96875 0 542.96875 0 c +542.96875 0 167 562.5 167 562.5 c +167 562.5 167 0 167 0 c +167 0 76.17188 0 76.17188 0 c +h +f* +Q +} def +/P { +624 0 77 0 624 716 setcachedevice +q +77.15625 0 m +77.15625 0 77.15625 715.82812 77.15625 715.82812 c +77.15625 715.82812 347.17188 715.82812 347.17188 715.82812 c +347.17188 715.82812 418.45312 715.82812 456.0625 708.98438 c +456.0625 708.98438 508.79688 700.20312 544.4375 675.53125 c +544.4375 675.53125 580.07812 650.875 601.79688 606.4375 c +601.79688 606.4375 623.53125 562.01562 623.53125 508.79688 c +623.53125 508.79688 623.53125 417.48438 565.42188 354.25 c +565.42188 354.25 507.32812 291.01562 355.46875 291.01562 c +355.46875 291.01562 171.875 291.01562 171.875 291.01562 c +171.875 291.01562 171.875 0 171.875 0 c +171.875 0 77.15625 0 77.15625 0 c +77.15625 0 77.15625 0 77.15625 0 c +h +171.875 375.48438 m +171.875 375.48438 356.9375 375.48438 356.9375 375.48438 c +356.9375 375.48438 448.73438 375.48438 487.29688 409.65625 c +487.29688 409.65625 525.875 443.84375 525.875 505.85938 c +525.875 505.85938 525.875 550.78125 503.17188 582.76562 c +503.17188 582.76562 480.46875 614.75 443.35938 625 c +443.35938 625 419.4375 631.34375 354.98438 631.34375 c +354.98438 631.34375 171.875 631.34375 171.875 631.34375 c +171.875 631.34375 171.875 375.48438 171.875 375.48438 c +h +f* +Q +} def +/Q { +741 0 43 -56 741 729 setcachedevice +q +619.625 76.65625 m +619.625 76.65625 685.54688 31.25 741.21875 10.25 c +741.21875 10.25 713.375 -55.67188 713.375 -55.67188 c +713.375 -55.67188 636.23438 -27.82812 559.57812 32.23438 c +559.57812 32.23438 479.98438 -12.20312 383.79688 -12.20312 c +383.79688 -12.20312 286.625 -12.20312 207.51562 34.67188 c +207.51562 34.67188 128.42188 81.54688 85.6875 166.5 c +85.6875 166.5 42.96875 251.46875 42.96875 357.90625 c +42.96875 357.90625 42.96875 463.875 85.9375 550.78125 c +85.9375 550.78125 128.90625 637.70312 208.25 683.10938 c +208.25 683.10938 287.59375 728.51562 385.75 728.51562 c +385.75 728.51562 484.85938 728.51562 564.45312 681.39062 c +564.45312 681.39062 644.04688 634.28125 685.79688 549.5625 c +685.79688 549.5625 727.54688 464.84375 727.54688 358.40625 c +727.54688 358.40625 727.54688 270.01562 700.6875 199.45312 c +700.6875 199.45312 673.82812 128.90625 619.625 76.65625 c +619.625 76.65625 619.625 76.65625 619.625 76.65625 c +h +411.14062 197.75 m +411.14062 197.75 493.17188 174.8125 546.39062 129.39062 c +546.39062 129.39062 629.89062 205.5625 629.89062 358.40625 c +629.89062 358.40625 629.89062 445.3125 600.34375 510.25 c +600.34375 510.25 570.79688 575.20312 513.90625 611.07812 c +513.90625 611.07812 457.03125 646.96875 386.23438 646.96875 c +386.23438 646.96875 280.28125 646.96875 210.45312 574.45312 c +210.45312 574.45312 140.625 501.95312 140.625 357.90625 c +140.625 357.90625 140.625 218.26562 209.71875 143.54688 c +209.71875 143.54688 278.8125 68.84375 386.23438 68.84375 c +386.23438 68.84375 437.01562 68.84375 481.9375 87.89062 c +481.9375 87.89062 437.5 116.70313 388.1875 128.90625 c +388.1875 128.90625 411.14062 197.75 411.14062 197.75 c +h +f* +Q +} def +/S { +615 0 45 -12 615 728 setcachedevice +q +44.92188 229.98438 m +44.92188 229.98438 134.28125 237.79688 134.28125 237.79688 c +134.28125 237.79688 140.625 184.07812 163.8125 149.65625 c +163.8125 149.65625 187.01562 115.23438 235.84375 93.98438 c +235.84375 93.98438 284.67188 72.75 345.70312 72.75 c +345.70312 72.75 399.90625 72.75 441.40625 88.85938 c +441.40625 88.85938 482.90625 104.98438 503.17188 133.0625 c +503.17188 133.0625 523.4375 161.14062 523.4375 194.34375 c +523.4375 194.34375 523.4375 228.03125 503.90625 253.17188 c +503.90625 253.17188 484.375 278.32812 439.45312 295.40625 c +439.45312 295.40625 410.64062 306.64062 312 330.3125 c +312 330.3125 213.375 354 173.82812 375 c +173.82812 375 122.5625 401.85938 97.40625 441.65625 c +97.40625 441.65625 72.26562 481.45312 72.26562 530.76562 c +72.26562 530.76562 72.26562 584.96875 103.03125 632.07812 c +103.03125 632.07812 133.79688 679.20312 192.875 703.60938 c +192.875 703.60938 251.95312 728.03125 324.21875 728.03125 c +324.21875 728.03125 403.8125 728.03125 464.59375 702.39062 c +464.59375 702.39062 525.39062 676.76562 558.10938 626.95312 c +558.10938 626.95312 590.82812 577.15625 593.26562 514.15625 c +593.26562 514.15625 502.4375 507.32812 502.4375 507.32812 c +502.4375 507.32812 495.125 575.20312 452.875 609.85938 c +452.875 609.85938 410.64062 644.53125 328.125 644.53125 c +328.125 644.53125 242.1875 644.53125 202.875 613.03125 c +202.875 613.03125 163.57812 581.54688 163.57812 537.10938 c +163.57812 537.10938 163.57812 498.53125 191.40625 473.64062 c +191.40625 473.64062 218.75 448.73438 334.21875 422.60938 c +334.21875 422.60938 449.70312 396.48438 492.67188 376.95312 c +492.67188 376.95312 555.17188 348.14062 584.95312 303.95312 c +584.95312 303.95312 614.75 259.76562 614.75 202.15625 c +614.75 202.15625 614.75 145.01562 582.03125 94.48438 c +582.03125 94.48438 549.3125 43.95312 488.03125 15.875 c +488.03125 15.875 426.76562 -12.20312 350.09375 -12.20312 c +350.09375 -12.20312 252.9375 -12.20312 187.25 16.10937 c +187.25 16.10937 121.57812 44.4375 84.21875 101.3125 c +84.21875 101.3125 46.875 158.20312 44.92188 229.98438 c +h +f* +Q +} def +/U { +642 0 79 -12 642 716 setcachedevice +q +546.875 715.82812 m +546.875 715.82812 641.60938 715.82812 641.60938 715.82812 c +641.60938 715.82812 641.60938 302.25 641.60938 302.25 c +641.60938 302.25 641.60938 194.34375 617.1875 130.85938 c +617.1875 130.85938 592.78125 67.39062 529.04688 27.59375 c +529.04688 27.59375 465.32812 -12.20312 361.8125 -12.20312 c +361.8125 -12.20312 261.23438 -12.20312 197.26562 22.45312 c +197.26562 22.45312 133.29688 57.125 105.95312 122.79688 c +105.95312 122.79688 78.60938 188.48438 78.60938 302.25 c +78.60938 302.25 78.60938 715.82812 78.60938 715.82812 c +78.60938 715.82812 173.34375 715.82812 173.34375 715.82812 c +173.34375 715.82812 173.34375 302.73438 173.34375 302.73438 c +173.34375 302.73438 173.34375 209.46875 190.67188 165.28125 c +190.67188 165.28125 208.01562 121.09375 250.25 97.17188 c +250.25 97.17188 292.48438 73.25 353.51562 73.25 c +353.51562 73.25 458.01562 73.25 502.4375 120.60938 c +502.4375 120.60938 546.875 167.96875 546.875 302.73438 c +546.875 302.73438 546.875 715.82812 546.875 715.82812 c +h +f* +Q +} def +/V { +659 0 4 0 659 716 setcachedevice +q +281.73438 0 m +281.73438 0 4.39062 715.82812 4.39062 715.82812 c +4.39062 715.82812 106.9375 715.82812 106.9375 715.82812 c +106.9375 715.82812 292.96875 195.79687 292.96875 195.79687 c +292.96875 195.79687 315.4375 133.29687 330.5625 78.60937 c +330.5625 78.60937 347.17188 137.20312 369.14062 195.79687 c +369.14062 195.79687 562.5 715.82812 562.5 715.82812 c +562.5 715.82812 659.1875 715.82812 659.1875 715.82812 c +659.1875 715.82812 378.90625 0 378.90625 0 c +378.90625 0 281.73438 0 281.73438 0 c +h +f* +Q +} def +/a { +514 0 36 -12 514 530 setcachedevice +q +404.29688 63.96875 m +404.29688 63.96875 355.46875 22.46875 310.29688 5.375 c +310.29688 5.375 265.14062 -11.71875 213.375 -11.71875 c +213.375 -11.71875 127.9375 -11.71875 82.03125 30.03125 c +82.03125 30.03125 36.14062 71.78125 36.14062 136.71875 c +36.14062 136.71875 36.14062 174.8125 53.46875 206.29688 c +53.46875 206.29688 70.79688 237.79688 98.875 256.82812 c +98.875 256.82812 126.95312 275.875 162.10938 285.64062 c +162.10938 285.64062 187.98438 292.48438 240.23438 298.82812 c +240.23438 298.82812 346.6875 311.53125 396.96875 329.10938 c +396.96875 329.10938 397.46875 347.17188 397.46875 352.04688 c +397.46875 352.04688 397.46875 405.76562 372.5625 427.73438 c +372.5625 427.73438 338.875 457.51562 272.46875 457.51562 c +272.46875 457.51562 210.45312 457.51562 180.90625 435.78125 c +180.90625 435.78125 151.375 414.0625 137.20312 358.89062 c +137.20312 358.89062 51.26562 370.60938 51.26562 370.60938 c +51.26562 370.60938 62.98438 425.78125 89.84375 459.71875 c +89.84375 459.71875 116.70312 493.65625 167.48438 511.96875 c +167.48438 511.96875 218.26562 530.28125 285.15625 530.28125 c +285.15625 530.28125 351.5625 530.28125 393.0625 514.65625 c +393.0625 514.65625 434.57812 499.03125 454.10938 475.34375 c +454.10938 475.34375 473.64062 451.65625 481.45312 415.53125 c +481.45312 415.53125 485.84375 393.0625 485.84375 334.46875 c +485.84375 334.46875 485.84375 217.28125 485.84375 217.28125 c +485.84375 217.28125 485.84375 94.73438 491.45313 62.25 c +491.45313 62.25 497.07812 29.78125 513.67188 0 c +513.67188 0 421.875 0 421.875 0 c +421.875 0 408.20312 27.34375 404.29688 63.96875 c +404.29688 63.96875 404.29688 63.96875 404.29688 63.96875 c +h +396.96875 260.25 m +396.96875 260.25 349.125 240.71875 253.42188 227.04688 c +253.42188 227.04688 199.21875 219.23438 176.75 209.46875 c +176.75 209.46875 154.29688 199.70312 142.09375 180.90625 c +142.09375 180.90625 129.89062 162.10938 129.89062 139.15625 c +129.89062 139.15625 129.89062 104 156.5 80.5625 c +156.5 80.5625 183.10938 57.125 234.375 57.125 c +234.375 57.125 285.15625 57.125 324.70312 79.34375 c +324.70312 79.34375 364.26562 101.5625 382.8125 140.14063 c +382.8125 140.14063 396.96875 169.92188 396.96875 228.03125 c +396.96875 228.03125 396.96875 260.25 396.96875 260.25 c +h +f* +Q +} def +/b { +515 0 65 -12 515 716 setcachedevice +q +146.96875 0 m +146.96875 0 65.4375 0 65.4375 0 c +65.4375 0 65.4375 715.82812 65.4375 715.82812 c +65.4375 715.82812 153.32812 715.82812 153.32812 715.82812 c +153.32812 715.82812 153.32812 460.45312 153.32812 460.45312 c +153.32812 460.45312 208.98438 530.28125 295.40625 530.28125 c +295.40625 530.28125 343.26562 530.28125 385.98438 510.98438 c +385.98438 510.98438 428.71875 491.70312 456.29688 456.78125 c +456.29688 456.78125 483.89062 421.875 499.51562 372.5625 c +499.51562 372.5625 515.14062 323.25 515.14062 267.09375 c +515.14062 267.09375 515.14062 133.79688 449.21875 61.03125 c +449.21875 61.03125 383.29688 -11.71875 291.01562 -11.71875 c +291.01562 -11.71875 199.21875 -11.71875 146.96875 64.9375 c +146.96875 64.9375 146.96875 0 146.96875 0 c +146.96875 0 146.96875 0 146.96875 0 c +h +146 263.1875 m +146 263.1875 146 169.92188 171.39062 128.42188 c +171.39062 128.42188 212.89062 60.54688 283.6875 60.54688 c +283.6875 60.54688 341.3125 60.54688 383.29688 110.59375 c +383.29688 110.59375 425.29688 160.64062 425.29688 259.76562 c +425.29688 259.76562 425.29688 361.32812 385.01562 409.67188 c +385.01562 409.67188 344.73438 458.01562 287.59375 458.01562 c +287.59375 458.01562 229.98438 458.01562 187.98438 407.95312 c +187.98438 407.95312 146 357.90625 146 263.1875 c +h +f* +Q +} def +/c { +491 0 39 -12 491 530 setcachedevice +q +404.29688 189.9375 m +404.29688 189.9375 490.71875 178.71875 490.71875 178.71875 c +490.71875 178.71875 476.5625 89.35938 418.20312 38.8125 c +418.20312 38.8125 359.85938 -11.71875 274.90625 -11.71875 c +274.90625 -11.71875 168.45312 -11.71875 103.75 57.85938 c +103.75 57.85938 39.0625 127.4375 39.0625 257.32812 c +39.0625 257.32812 39.0625 341.3125 66.89062 404.29688 c +66.89062 404.29688 94.73438 467.28125 151.60938 498.78125 c +151.60938 498.78125 208.5 530.28125 275.39062 530.28125 c +275.39062 530.28125 359.85938 530.28125 413.5625 487.54688 c +413.5625 487.54688 467.28125 444.82812 482.42188 366.21875 c +482.42188 366.21875 396.96875 353.03125 396.96875 353.03125 c +396.96875 353.03125 384.76562 405.28125 353.75 431.64062 c +353.75 431.64062 322.75 458.01562 278.8125 458.01562 c +278.8125 458.01562 212.40625 458.01562 170.89062 410.40625 c +170.89062 410.40625 129.39062 362.79688 129.39062 259.76562 c +129.39062 259.76562 129.39062 155.28125 169.42188 107.90625 c +169.42188 107.90625 209.46875 60.54688 273.92188 60.54688 c +273.92188 60.54688 325.6875 60.54688 360.34375 92.28125 c +360.34375 92.28125 395.01562 124.03125 404.29688 189.9375 c +h +f* +Q +} def +/d { +484 0 34 -12 484 716 setcachedevice +q +402.34375 0 m +402.34375 0 402.34375 65.4375 402.34375 65.4375 c +402.34375 65.4375 353.03125 -11.71875 257.32812 -11.71875 c +257.32812 -11.71875 195.3125 -11.71875 143.3125 22.45312 c +143.3125 22.45312 91.3125 56.64062 62.75 117.92188 c +62.75 117.92188 34.1875 179.20313 34.1875 258.79688 c +34.1875 258.79688 34.1875 336.42188 60.0625 399.65625 c +60.0625 399.65625 85.9375 462.89062 137.6875 496.57812 c +137.6875 496.57812 189.45312 530.28125 253.42188 530.28125 c +253.42188 530.28125 300.29688 530.28125 336.90625 510.5 c +336.90625 510.5 373.53125 490.71875 396.48438 458.98438 c +396.48438 458.98438 396.48438 715.82812 396.48438 715.82812 c +396.48438 715.82812 483.89062 715.82812 483.89062 715.82812 c +483.89062 715.82812 483.89062 0 483.89062 0 c +483.89062 0 402.34375 0 402.34375 0 c +402.34375 0 402.34375 0 402.34375 0 c +h +124.51562 258.79688 m +124.51562 258.79688 124.51562 159.1875 166.5 109.85937 c +166.5 109.85937 208.5 60.54688 265.625 60.54688 c +265.625 60.54688 323.25 60.54688 363.53125 107.65625 c +363.53125 107.65625 403.8125 154.78125 403.8125 251.46875 c +403.8125 251.46875 403.8125 357.90625 362.79688 407.70312 c +362.79688 407.70312 321.78125 457.51562 261.71875 457.51562 c +261.71875 457.51562 203.125 457.51562 163.8125 409.65625 c +163.8125 409.65625 124.51562 361.8125 124.51562 258.79688 c +h +f* +Q +} def +/e { +515 0 37 -12 515 530 setcachedevice +q +420.90625 167 m +420.90625 167 511.71875 155.76563 511.71875 155.76563 c +511.71875 155.76563 490.23438 76.17188 432.125 32.21875 c +432.125 32.21875 374.03125 -11.71875 283.6875 -11.71875 c +283.6875 -11.71875 169.92188 -11.71875 103.26562 58.34375 c +103.26562 58.34375 36.625 128.42188 36.625 254.89062 c +36.625 254.89062 36.625 385.75 104 458.01562 c +104 458.01562 171.39062 530.28125 278.8125 530.28125 c +278.8125 530.28125 382.8125 530.28125 448.73438 459.46875 c +448.73438 459.46875 514.65625 388.67188 514.65625 260.25 c +514.65625 260.25 514.65625 252.4375 514.15625 236.8125 c +514.15625 236.8125 127.4375 236.8125 127.4375 236.8125 c +127.4375 236.8125 132.32812 151.375 175.78125 105.95312 c +175.78125 105.95312 219.23438 60.54688 284.1875 60.54688 c +284.1875 60.54688 332.51562 60.54688 366.6875 85.9375 c +366.6875 85.9375 400.875 111.32812 420.90625 167 c +420.90625 167 420.90625 167 420.90625 167 c +h +132.32812 309.07812 m +132.32812 309.07812 421.875 309.07812 421.875 309.07812 c +421.875 309.07812 416.01562 374.51562 388.67188 407.23438 c +388.67188 407.23438 346.6875 458.01562 279.78125 458.01562 c +279.78125 458.01562 219.23438 458.01562 177.96875 417.48438 c +177.96875 417.48438 136.71875 376.95312 132.32812 309.07812 c +h +f* +Q +} def +/f { +313 0 9 0 313 728 setcachedevice +q +86.92188 0 m +86.92188 0 86.92188 450.20312 86.92188 450.20312 c +86.92188 450.20312 9.28125 450.20312 9.28125 450.20312 c +9.28125 450.20312 9.28125 518.5625 9.28125 518.5625 c +9.28125 518.5625 86.92188 518.5625 86.92188 518.5625 c +86.92188 518.5625 86.92188 573.73438 86.92188 573.73438 c +86.92188 573.73438 86.92188 625.98438 96.1875 651.375 c +96.1875 651.375 108.89062 685.54688 140.875 706.78125 c +140.875 706.78125 172.85938 728.03125 230.46875 728.03125 c +230.46875 728.03125 267.57812 728.03125 312.5 719.23438 c +312.5 719.23438 299.3125 642.57812 299.3125 642.57812 c +299.3125 642.57812 271.96875 647.46875 247.5625 647.46875 c +247.5625 647.46875 207.51562 647.46875 190.90625 630.375 c +190.90625 630.375 174.3125 613.28125 174.3125 566.40625 c +174.3125 566.40625 174.3125 518.5625 174.3125 518.5625 c +174.3125 518.5625 275.39062 518.5625 275.39062 518.5625 c +275.39062 518.5625 275.39062 450.20312 275.39062 450.20312 c +275.39062 450.20312 174.3125 450.20312 174.3125 450.20312 c +174.3125 450.20312 174.3125 0 174.3125 0 c +174.3125 0 86.92188 0 86.92188 0 c +h +f* +Q +} def +/g { +489 0 32 -210 489 530 setcachedevice +q +49.8125 -42.96875 m +49.8125 -42.96875 135.25 -55.67188 135.25 -55.67188 c +135.25 -55.67188 140.625 -95.21875 165.04688 -113.28125 c +165.04688 -113.28125 197.75 -137.70312 254.39062 -137.70312 c +254.39062 -137.70312 315.4375 -137.70312 348.64062 -113.28125 c +348.64062 -113.28125 381.84375 -88.875 393.5625 -44.92188 c +393.5625 -44.92188 400.39062 -18.0625 399.90625 67.875 c +399.90625 67.875 342.28125 0 256.34375 0 c +256.34375 0 149.42188 0 90.82812 77.14062 c +90.82812 77.14062 32.23438 154.29688 32.23438 262.20312 c +32.23438 262.20312 32.23438 336.42188 59.07812 399.17188 c +59.07812 399.17188 85.9375 461.92188 136.95312 496.09375 c +136.95312 496.09375 187.98438 530.28125 256.84375 530.28125 c +256.84375 530.28125 348.64062 530.28125 408.20312 456.0625 c +408.20312 456.0625 408.20312 518.5625 408.20312 518.5625 c +408.20312 518.5625 489.26562 518.5625 489.26562 518.5625 c +489.26562 518.5625 489.26562 70.3125 489.26562 70.3125 c +489.26562 70.3125 489.26562 -50.78125 464.59375 -101.3125 c +464.59375 -101.3125 439.9375 -151.85938 386.46875 -181.15625 c +386.46875 -181.15625 333.01562 -210.45312 254.89062 -210.45312 c +254.89062 -210.45312 162.10938 -210.45312 104.98438 -168.70312 c +104.98438 -168.70312 47.85938 -126.95312 49.8125 -42.96875 c +49.8125 -42.96875 49.8125 -42.96875 49.8125 -42.96875 c +h +122.5625 268.5625 m +122.5625 268.5625 122.5625 166.5 163.07812 119.625 c +163.07812 119.625 203.60938 72.75 264.65625 72.75 c +264.65625 72.75 325.20312 72.75 366.21875 119.375 c +366.21875 119.375 407.23438 166.01562 407.23438 265.625 c +407.23438 265.625 407.23438 360.84375 364.98438 409.17188 c +364.98438 409.17188 322.75 457.51562 263.1875 457.51562 c +263.1875 457.51562 204.59375 457.51562 163.57812 409.90625 c +163.57812 409.90625 122.5625 362.3125 122.5625 268.5625 c +h +f* +Q +} def +/h { +488 0 66 0 488 716 setcachedevice +q +65.92188 0 m +65.92188 0 65.92188 715.82812 65.92188 715.82812 c +65.92188 715.82812 153.8125 715.82812 153.8125 715.82812 c +153.8125 715.82812 153.8125 458.98438 153.8125 458.98438 c +153.8125 458.98438 215.32812 530.28125 309.07812 530.28125 c +309.07812 530.28125 366.70312 530.28125 409.17188 507.5625 c +409.17188 507.5625 451.65625 484.85938 469.96875 444.8125 c +469.96875 444.8125 488.28125 404.78125 488.28125 328.60938 c +488.28125 328.60938 488.28125 0 488.28125 0 c +488.28125 0 400.39062 0 400.39062 0 c +400.39062 0 400.39062 328.60938 400.39062 328.60938 c +400.39062 328.60938 400.39062 394.53125 371.82812 424.5625 c +371.82812 424.5625 343.26562 454.59375 291.01562 454.59375 c +291.01562 454.59375 251.95312 454.59375 217.53125 434.32812 c +217.53125 434.32812 183.10938 414.0625 168.45312 379.39062 c +168.45312 379.39062 153.8125 344.73438 153.8125 283.6875 c +153.8125 283.6875 153.8125 0 153.8125 0 c +153.8125 0 65.92188 0 65.92188 0 c +h +f* +Q +} def +/i { +154 0 66 0 154 716 setcachedevice +q +66.40625 614.75 m +66.40625 614.75 66.40625 715.82812 66.40625 715.82812 c +66.40625 715.82812 154.29688 715.82812 154.29688 715.82812 c +154.29688 715.82812 154.29688 614.75 154.29688 614.75 c +154.29688 614.75 66.40625 614.75 66.40625 614.75 c +66.40625 614.75 66.40625 614.75 66.40625 614.75 c +h +66.40625 0 m +66.40625 0 66.40625 518.5625 66.40625 518.5625 c +66.40625 518.5625 154.29688 518.5625 154.29688 518.5625 c +154.29688 518.5625 154.29688 0 154.29688 0 c +154.29688 0 66.40625 0 66.40625 0 c +h +f* +Q +} def +/j { +153 0 -46 -210 153 716 setcachedevice +q +65.4375 613.76562 m +65.4375 613.76562 65.4375 715.82812 65.4375 715.82812 c +65.4375 715.82812 153.32812 715.82812 153.32812 715.82812 c +153.32812 715.82812 153.32812 613.76562 153.32812 613.76562 c +153.32812 613.76562 65.4375 613.76562 65.4375 613.76562 c +65.4375 613.76562 65.4375 613.76562 65.4375 613.76562 c +h +-45.90625 -201.17188 m +-45.90625 -201.17188 -29.29688 -126.46875 -29.29688 -126.46875 c +-29.29688 -126.46875 -2.9375 -133.29688 12.20312 -133.29688 c +12.20312 -133.29688 39.0625 -133.29688 52.25 -115.46875 c +52.25 -115.46875 65.4375 -97.65625 65.4375 -26.375 c +65.4375 -26.375 65.4375 518.5625 65.4375 518.5625 c +65.4375 518.5625 153.32812 518.5625 153.32812 518.5625 c +153.32812 518.5625 153.32812 -28.32812 153.32812 -28.32812 c +153.32812 -28.32812 153.32812 -124.03125 128.42188 -161.625 c +128.42188 -161.625 96.6875 -210.45312 22.95312 -210.45312 c +22.95312 -210.45312 -12.70313 -210.45312 -45.90625 -201.17188 c +h +f* +Q +} def +/l { +152 0 64 0 152 716 setcachedevice +q +63.96875 0 m +63.96875 0 63.96875 715.82812 63.96875 715.82812 c +63.96875 715.82812 151.85938 715.82812 151.85938 715.82812 c +151.85938 715.82812 151.85938 0 151.85938 0 c +151.85938 0 63.96875 0 63.96875 0 c +h +f* +Q +} def +/m { +769 0 66 0 769 530 setcachedevice +q +65.92188 0 m +65.92188 0 65.92188 518.5625 65.92188 518.5625 c +65.92188 518.5625 144.53125 518.5625 144.53125 518.5625 c +144.53125 518.5625 144.53125 445.79688 144.53125 445.79688 c +144.53125 445.79688 168.95312 483.89062 209.46875 507.07813 c +209.46875 507.07813 250 530.28125 301.76562 530.28125 c +301.76562 530.28125 359.375 530.28125 396.23438 506.34375 c +396.23438 506.34375 433.10938 482.42188 448.25 439.45312 c +448.25 439.45312 509.76562 530.28125 608.40625 530.28125 c +608.40625 530.28125 685.54688 530.28125 727.04688 487.54688 c +727.04688 487.54688 768.5625 444.82812 768.5625 355.95312 c +768.5625 355.95312 768.5625 0 768.5625 0 c +768.5625 0 681.15625 0 681.15625 0 c +681.15625 0 681.15625 326.65625 681.15625 326.65625 c +681.15625 326.65625 681.15625 379.39062 672.60938 402.57812 c +672.60938 402.57812 664.0625 425.78125 641.59375 439.9375 c +641.59375 439.9375 619.14062 454.10938 588.875 454.10938 c +588.875 454.10938 534.1875 454.10938 498.04688 417.71875 c +498.04688 417.71875 461.92188 381.34375 461.92188 301.26562 c +461.92188 301.26562 461.92188 0 461.92188 0 c +461.92188 0 374.03125 0 374.03125 0 c +374.03125 0 374.03125 336.92188 374.03125 336.92188 c +374.03125 336.92188 374.03125 395.51562 352.54688 424.8125 c +352.54688 424.8125 331.0625 454.10938 282.23438 454.10938 c +282.23438 454.10938 245.125 454.10938 213.625 434.57812 c +213.625 434.57812 182.125 415.04688 167.96875 377.4375 c +167.96875 377.4375 153.8125 339.84375 153.8125 269.04688 c +153.8125 269.04688 153.8125 0 153.8125 0 c +153.8125 0 65.92188 0 65.92188 0 c +h +f* +Q +} def +/n { +487 0 66 0 487 530 setcachedevice +q +65.92188 0 m +65.92188 0 65.92188 518.5625 65.92188 518.5625 c +65.92188 518.5625 145.01562 518.5625 145.01562 518.5625 c +145.01562 518.5625 145.01562 444.82812 145.01562 444.82812 c +145.01562 444.82812 202.15625 530.28125 310.0625 530.28125 c +310.0625 530.28125 356.9375 530.28125 396.23438 513.42188 c +396.23438 513.42188 435.54688 496.57812 455.07812 469.23438 c +455.07812 469.23438 474.60938 441.89062 482.42188 404.29688 c +482.42188 404.29688 487.3125 379.89062 487.3125 318.84375 c +487.3125 318.84375 487.3125 0 487.3125 0 c +487.3125 0 399.42188 0 399.42188 0 c +399.42188 0 399.42188 315.4375 399.42188 315.4375 c +399.42188 315.4375 399.42188 369.14062 389.15625 395.75 c +389.15625 395.75 378.90625 422.35938 352.78125 438.23438 c +352.78125 438.23438 326.65625 454.10938 291.5 454.10938 c +291.5 454.10938 235.35938 454.10938 194.57812 418.45312 c +194.57812 418.45312 153.8125 382.8125 153.8125 283.20312 c +153.8125 283.20312 153.8125 0 153.8125 0 c +153.8125 0 65.92188 0 65.92188 0 c +h +f* +Q +} def +/o { +519 0 33 -12 519 530 setcachedevice +q +33.20312 259.28125 m +33.20312 259.28125 33.20312 403.32812 113.28125 472.65625 c +113.28125 472.65625 180.17188 530.28125 276.375 530.28125 c +276.375 530.28125 383.29688 530.28125 451.17188 460.20312 c +451.17188 460.20312 519.04688 390.14062 519.04688 266.60938 c +519.04688 266.60938 519.04688 166.5 489.01563 109.125 c +489.01563 109.125 458.98438 51.76562 401.60938 20.01562 c +401.60938 20.01562 344.23438 -11.71875 276.375 -11.71875 c +276.375 -11.71875 167.48438 -11.71875 100.34375 58.10938 c +100.34375 58.10938 33.20312 127.9375 33.20312 259.28125 c +33.20312 259.28125 33.20312 259.28125 33.20312 259.28125 c +h +123.53125 259.28125 m +123.53125 259.28125 123.53125 159.67188 166.98438 110.10938 c +166.98438 110.10938 210.45312 60.54688 276.375 60.54688 c +276.375 60.54688 341.79688 60.54688 385.25 110.34375 c +385.25 110.34375 428.71875 160.15625 428.71875 262.20312 c +428.71875 262.20312 428.71875 358.40625 385.01562 407.95312 c +385.01562 407.95312 341.3125 457.51562 276.375 457.51562 c +276.375 457.51562 210.45312 457.51562 166.98438 408.20312 c +166.98438 408.20312 123.53125 358.89062 123.53125 259.28125 c +h +f* +Q +} def +/p { +516 0 66 -199 516 530 setcachedevice +q +65.92188 -198.73438 m +65.92188 -198.73438 65.92188 518.5625 65.92188 518.5625 c +65.92188 518.5625 146 518.5625 146 518.5625 c +146 518.5625 146 451.17188 146 451.17188 c +146 451.17188 174.3125 490.71875 209.95312 510.5 c +209.95312 510.5 245.60938 530.28125 296.39062 530.28125 c +296.39062 530.28125 362.79688 530.28125 413.57812 496.09375 c +413.57812 496.09375 464.35938 461.92188 490.23438 399.65625 c +490.23438 399.65625 516.10938 337.40625 516.10938 263.1875 c +516.10938 263.1875 516.10938 183.59375 487.54688 119.875 c +487.54688 119.875 458.98438 56.15625 404.53125 22.21875 c +404.53125 22.21875 350.09375 -11.71875 290.04688 -11.71875 c +290.04688 -11.71875 246.09375 -11.71875 211.17188 6.82812 c +211.17188 6.82812 176.26562 25.39062 153.8125 53.71875 c +153.8125 53.71875 153.8125 -198.73438 153.8125 -198.73438 c +153.8125 -198.73438 65.92188 -198.73438 65.92188 -198.73438 c +65.92188 -198.73438 65.92188 -198.73438 65.92188 -198.73438 c +h +145.51562 256.34375 m +145.51562 256.34375 145.51562 156.25 186.03125 108.39062 c +186.03125 108.39062 226.5625 60.54688 284.1875 60.54688 c +284.1875 60.54688 342.78125 60.54688 384.51562 110.10938 c +384.51562 110.10938 426.26562 159.67188 426.26562 263.67188 c +426.26562 263.67188 426.26562 362.79688 385.5 412.10938 c +385.5 412.10938 344.73438 461.42188 288.09375 461.42188 c +288.09375 461.42188 231.9375 461.42188 188.71875 408.9375 c +188.71875 408.9375 145.51562 356.45312 145.51562 256.34375 c +h +f* +Q +} def +/q { +484 0 35 -199 484 530 setcachedevice +q +396.48438 -198.73438 m +396.48438 -198.73438 396.48438 55.17187 396.48438 55.17187 c +396.48438 55.17187 375.98438 26.375 339.10938 7.32813 c +339.10938 7.32813 302.25 -11.71875 260.75 -11.71875 c +260.75 -11.71875 168.45312 -11.71875 101.79688 62.01563 c +101.79688 62.01563 35.15625 135.75 35.15625 264.15625 c +35.15625 264.15625 35.15625 342.28125 62.25 404.29688 c +62.25 404.29688 89.35938 466.3125 140.875 498.29688 c +140.875 498.29688 192.39063 530.28125 253.90625 530.28125 c +253.90625 530.28125 350.09375 530.28125 405.28125 449.21875 c +405.28125 449.21875 405.28125 518.5625 405.28125 518.5625 c +405.28125 518.5625 484.375 518.5625 484.375 518.5625 c +484.375 518.5625 484.375 -198.73438 484.375 -198.73438 c +484.375 -198.73438 396.48438 -198.73438 396.48438 -198.73438 c +396.48438 -198.73438 396.48438 -198.73438 396.48438 -198.73438 c +h +125.48438 260.75 m +125.48438 260.75 125.48438 160.64062 167.46875 110.59375 c +167.46875 110.59375 209.46875 60.54688 268.0625 60.54688 c +268.0625 60.54688 324.21875 60.54688 364.75 108.15625 c +364.75 108.15625 405.28125 155.76563 405.28125 252.9375 c +405.28125 252.9375 405.28125 356.45312 362.54688 408.6875 c +362.54688 408.6875 319.82812 460.9375 262.20312 460.9375 c +262.20312 460.9375 205.07812 460.9375 165.28125 412.34375 c +165.28125 412.34375 125.48438 363.76562 125.48438 260.75 c +h +f* +Q +} def +/r { +347 0 65 0 347 530 setcachedevice +q +64.9375 0 m +64.9375 0 64.9375 518.5625 64.9375 518.5625 c +64.9375 518.5625 144.04688 518.5625 144.04688 518.5625 c +144.04688 518.5625 144.04688 439.9375 144.04688 439.9375 c +144.04688 439.9375 174.3125 495.125 199.95312 512.70312 c +199.95312 512.70312 225.59375 530.28125 256.34375 530.28125 c +256.34375 530.28125 300.78125 530.28125 346.6875 501.95312 c +346.6875 501.95312 316.40625 420.40625 316.40625 420.40625 c +316.40625 420.40625 284.1875 439.45312 251.95312 439.45312 c +251.95312 439.45312 223.14062 439.45312 200.1875 422.10938 c +200.1875 422.10938 177.25 404.78125 167.48438 374.03125 c +167.48438 374.03125 152.82812 327.15625 152.82812 271.48438 c +152.82812 271.48438 152.82812 0 152.82812 0 c +152.82812 0 64.9375 0 64.9375 0 c +h +f* +Q +} def +/s { +461 0 31 -12 461 530 setcachedevice +q +30.76563 154.78125 m +30.76563 154.78125 117.67188 168.45312 117.67188 168.45312 c +117.67188 168.45312 125 116.21875 158.4375 88.375 c +158.4375 88.375 191.89062 60.54688 251.95312 60.54688 c +251.95312 60.54688 312.5 60.54688 341.79688 85.20312 c +341.79688 85.20312 371.09375 109.85937 371.09375 143.0625 c +371.09375 143.0625 371.09375 172.85938 345.21875 189.9375 c +345.21875 189.9375 327.15625 201.65625 255.375 219.73438 c +255.375 219.73438 158.6875 244.14062 121.32813 261.95312 c +121.32813 261.95312 83.98438 279.78125 64.6875 311.28125 c +64.6875 311.28125 45.40625 342.78125 45.40625 380.85938 c +45.40625 380.85938 45.40625 415.53125 61.28125 445.0625 c +61.28125 445.0625 77.15625 474.60938 104.5 494.14062 c +104.5 494.14062 125 509.28125 160.39062 519.78125 c +160.39062 519.78125 195.79688 530.28125 236.32812 530.28125 c +236.32812 530.28125 297.35938 530.28125 343.5 512.70312 c +343.5 512.70312 389.65625 495.125 411.625 465.09375 c +411.625 465.09375 433.59375 435.0625 441.89062 384.76562 c +441.89062 384.76562 355.95312 373.04688 355.95312 373.04688 c +355.95312 373.04688 350.09375 413.09375 322.01562 435.54688 c +322.01562 435.54688 293.95312 458.01562 242.67188 458.01562 c +242.67188 458.01562 182.125 458.01562 156.25 437.98438 c +156.25 437.98438 130.375 417.96875 130.375 391.10938 c +130.375 391.10938 130.375 374.03125 141.10938 360.35938 c +141.10938 360.35938 151.85938 346.1875 174.8125 336.92188 c +174.8125 336.92188 187.98438 332.03125 252.4375 314.45312 c +252.4375 314.45312 345.70312 289.54688 382.5625 273.67188 c +382.5625 273.67188 419.4375 257.8125 440.42188 227.53125 c +440.42188 227.53125 461.42188 197.26562 461.42188 152.34375 c +461.42188 152.34375 461.42188 108.40625 435.78125 69.57812 c +435.78125 69.57812 410.15625 30.76563 361.8125 9.51562 c +361.8125 9.51562 313.48438 -11.71875 252.4375 -11.71875 c +252.4375 -11.71875 151.375 -11.71875 98.39062 30.26562 c +98.39062 30.26562 45.40625 72.26562 30.76563 154.78125 c +h +f* +Q +} def +/t { +271 0 18 -7 271 700 setcachedevice +q +257.8125 78.60937 m +257.8125 78.60937 270.51562 0.98438 270.51562 0.98438 c +270.51562 0.98438 233.40625 -6.84375 204.10938 -6.84375 c +204.10938 -6.84375 156.25 -6.84375 129.875 8.29687 c +129.875 8.29687 103.51562 23.4375 92.76562 48.09375 c +92.76562 48.09375 82.03125 72.75 82.03125 151.85938 c +82.03125 151.85938 82.03125 450.20312 82.03125 450.20312 c +82.03125 450.20312 17.57812 450.20312 17.57812 450.20312 c +17.57812 450.20312 17.57812 518.5625 17.57812 518.5625 c +17.57812 518.5625 82.03125 518.5625 82.03125 518.5625 c +82.03125 518.5625 82.03125 646.96875 82.03125 646.96875 c +82.03125 646.96875 169.4375 699.70312 169.4375 699.70312 c +169.4375 699.70312 169.4375 518.5625 169.4375 518.5625 c +169.4375 518.5625 257.8125 518.5625 257.8125 518.5625 c +257.8125 518.5625 257.8125 450.20312 257.8125 450.20312 c +257.8125 450.20312 169.4375 450.20312 169.4375 450.20312 c +169.4375 450.20312 169.4375 146.96875 169.4375 146.96875 c +169.4375 146.96875 169.4375 109.375 174.07812 98.625 c +174.07812 98.625 178.71875 87.89062 189.20312 81.54688 c +189.20312 81.54688 199.70312 75.20312 219.23438 75.20312 c +219.23438 75.20312 233.89062 75.20312 257.8125 78.60937 c +h +f* +Q +} def +/u { +484 0 64 -12 484 519 setcachedevice +q +405.76562 0 m +405.76562 0 405.76562 76.17188 405.76562 76.17188 c +405.76562 76.17188 345.21875 -11.71875 241.21875 -11.71875 c +241.21875 -11.71875 195.3125 -11.71875 155.51562 5.85938 c +155.51562 5.85938 115.71875 23.4375 96.4375 50.04688 c +96.4375 50.04688 77.15625 76.65625 69.34375 115.23438 c +69.34375 115.23438 63.96875 141.10937 63.96875 197.26562 c +63.96875 197.26562 63.96875 518.5625 63.96875 518.5625 c +63.96875 518.5625 151.85938 518.5625 151.85938 518.5625 c +151.85938 518.5625 151.85938 230.95312 151.85938 230.95312 c +151.85938 230.95312 151.85938 162.10938 157.23438 138.1875 c +157.23438 138.1875 165.53125 103.51562 192.375 83.73438 c +192.375 83.73438 219.23438 63.96875 258.79688 63.96875 c +258.79688 63.96875 298.34375 63.96875 333 84.23438 c +333 84.23438 367.67188 104.5 382.07812 139.40625 c +382.07812 139.40625 396.48438 174.3125 396.48438 240.71875 c +396.48438 240.71875 396.48438 518.5625 396.48438 518.5625 c +396.48438 518.5625 484.375 518.5625 484.375 518.5625 c +484.375 518.5625 484.375 0 484.375 0 c +484.375 0 405.76562 0 405.76562 0 c +h +f* +Q +} def +/v { +488 0 13 0 488 519 setcachedevice +q +209.96875 0 m +209.96875 0 12.70313 518.5625 12.70313 518.5625 c +12.70313 518.5625 105.46875 518.5625 105.46875 518.5625 c +105.46875 518.5625 216.79688 208.01562 216.79688 208.01562 c +216.79688 208.01562 234.85938 157.71875 250 103.51562 c +250 103.51562 261.71875 144.53125 282.71875 202.15625 c +282.71875 202.15625 397.95312 518.5625 397.95312 518.5625 c +397.95312 518.5625 488.28125 518.5625 488.28125 518.5625 c +488.28125 518.5625 292 0 292 0 c +292 0 209.96875 0 209.96875 0 c +h +f* +Q +} def +end +currentdict end +/T3_82_0 exch definefont pop +%%EndResource +/F82_0 /T3_82_0 1 1 +[ /comma/hyphen/period/one/two/three/four/semicolon + /A/C/D/E/F/I/L/M + /N/P/Q/S/U/V/a/b + /c/d/e/f/g/h/i/j + /l/m/n/o/p/q/r/s + /t/u/v/plus/comma/hyphen/period/slash + /zero/one/two/three/four/five/six/seven + /eight/nine/colon/semicolon/less/equal/greater/question + /at/A/B/C/D/E/F/G + /H/I/J/K/L/M/N/O + /P/Q/R/S/T/U/V/W + /X/Y/Z/bracketleft/backslash/bracketright/asciicircum/underscore + /quoteleft/a/b/c/d/e/f/g + /h/i/j/k/l/m/n/o + /p/q/r/s/t/u/v/w + /x/y/z/braceleft/bar/braceright/asciitilde/.notdef + /.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef + /.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef + /.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef + /.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef + /.notdef/exclamdown/cent/sterling/fraction/yen/florin/section + /currency/quotesingle/quotedblleft/guillemotleft/guilsinglleft/guilsinglright/fi/fl + /.notdef/endash/dagger/daggerdbl/periodcentered/.notdef/paragraph/bullet + /quotesinglbase/quotedblbase/quotedblright/guillemotright/ellipsis/perthousand/.notdef/questiondown + /.notdef/grave/acute/circumflex/tilde/macron/breve/dotaccent + /dieresis/.notdef/ring/cedilla/.notdef/hungarumlaut/ogonek/caron + /emdash/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef + /.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef + /.notdef/AE/.notdef/ordfeminine/.notdef/.notdef/.notdef/.notdef + /Lslash/Oslash/OE/ordmasculine/.notdef/.notdef/.notdef/.notdef + /.notdef/ae/.notdef/.notdef/.notdef/dotlessi/.notdef/.notdef + /lslash/oslash/oe/germandbls/.notdef/.notdef/.notdef/.notdef] +pdfMakeFont +false pdfSetup +595 842 pdfSetupPaper +%%EndSetup +%%Page: 1 1 +%%BeginPageSetup +%%PageOrientation: Portrait +pdfStartPage +0 0.706376 translate +0.9983 0.9983 scale +0 0 596 842 re W +%%EndPageSetup +[] 0 d +1 i +0 j +0 J +10 M +1 w +/DeviceGray {} cs +[0] sc +/DeviceGray {} CS +[0] SC +false op +false OP +{} settransfer +0 0 595.28 841.89 re +W +q +q +[1 0 0 1 0 0] cm +q +1 w +[] 0 d +0 J +0 j +[1 0 0 1 72 805.89] cm +q +[1 0 0 1 0 0] Tm +0 0 Td +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\016) +[6.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 17.34375 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.01758 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 37.34766 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.01367 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.6875 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.6875 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 59.36133 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.69141 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 79.36523 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.03906 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.70508 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 95.37891 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.70898 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.70898 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 111.375 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.04297 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.7168 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.71289 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.38672 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.7207 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 151.38867 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 157.38867 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.0625 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.73633 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.73633 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.41016 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.41016 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.74414 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.41797 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.75195 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.42578 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.75586 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.42969 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.10352 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.76953 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 239.44336 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.10938 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.10938 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.10938 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.77539 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.44922 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.00586 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.00781 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.01172 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.68555 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.69336 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.69336 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 59.36719 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.04102 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.71484 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.38086 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.37695 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.05078 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.72461 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.05859 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.73242 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.72852 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 125.39648 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.0625 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.73633 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.73633 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 147.41016 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.74023 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.74805 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.74805 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.42188 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.0957 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.76953 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.43555 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.43164 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.10547 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.7793 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.11328 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.78711 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67969 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.01367 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.6875 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 35.36133 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.03516 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.70312 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 55.37695 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.05078 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.72461 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.39844 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.40625 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.08008 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.75391 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 101.41992 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.09375 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 115.42383 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.41992 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.09375 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.76758 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 139.44141 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 145.44141 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.11523 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 161.44922 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.11523 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.78906 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.78906 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.12305 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.79688 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.79688 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.46484 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 205.46484 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.13086 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.79883 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 221.47266 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.46875 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.14258 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.33398 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 7.33008 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.99609 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.66992 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 23.34375 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.00977 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.67578 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.3418 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.34961 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.02344 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.69727 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.37109 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.04492 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.05273 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.04883 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.72266 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.05664 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.73047 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.06445 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.73828 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.40625 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.08008 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.75391 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.41992 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.08594 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.75977 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.42773 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.42383 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.09766 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.77148 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.76758 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.43359 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.76758 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 205.44141 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.10742 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.78125 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 221.45508 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.12891 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.80273 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 241.47656 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.02148 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.01758 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.69141 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.02539 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.02148 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.0293 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\037) +[1.836 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.69531 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.36914 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.36914 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.70312 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.37695 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.04492 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.71875 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 93.39258 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.72656 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 109.40039 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.06641 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.73242 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 121.40625 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.08008 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.75391 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.42188 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.08789 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.76172 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.76953 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.10352 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.77734 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.77344 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.76953 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 191.44336 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.11719 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 201.45117 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.125 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 221.45508 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.12891 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.46289 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.13086 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.79688 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 253.4707 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 21.33984 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.00586 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.33984 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.01367 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.6875 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.02148 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.69531 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.36914 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.04297 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.05078 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.72461 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 93.39844 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.07227 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.74609 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.08008 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.74609 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.41992 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.09375 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 135.42773 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.10156 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.76953 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 158.76562 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 165.43945 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.11328 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.78711 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 181.45312 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.78711 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 197.46094 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.12695 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.46094 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.13477 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.80859 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.80469 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.13867 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.8125 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.80859 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.47461 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.14844 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.82227 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.99609 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.66992 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 23.34375 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.67773 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.35156 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.35156 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.01953 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.69336 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 59.36719 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.36719 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.36719 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.04102 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.375 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.04883 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 95.37891 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.375 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.04102 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.71484 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 111.38086 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.38086 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.05469 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.7207 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.39453 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.72852 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.72461 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.39844 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 165.39844 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.06641 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.73242 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 187.40625 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.07227 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.73828 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.41211 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.74219 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.73828 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.41211 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.07812 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.75195 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.75195 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 251.42578 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.09961 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.77344 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 2.66602 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.00977 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.68359 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.35742 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.02539 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.69141 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.36523 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 55.36523 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.03906 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.03516 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.70312 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.71094 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.04492 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.71875 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 105.39258 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.06641 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.73242 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 121.40625 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.07422 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 131.4082 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.08203 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.74805 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.41406 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.08789 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.42188 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.0957 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.76172 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 175.43555 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.10156 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 181.43555 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.10938 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.7832 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.79102 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.79102 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 213.45703 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.79102 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.46484 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.13867 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.00391 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 14.67773 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 21.35156 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 27.35156 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.02539 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.69141 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.35742 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.03125 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 55.36523 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.36523 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.03906 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.03516 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.03516 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.70898 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.04297 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.7168 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 107.39062 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.06445 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.39453 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.06836 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.07617 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.75 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 157.42383 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.09766 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 177.42773 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.42773 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.10156 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.77539 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.44141 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 205.44141 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.11523 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.78125 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.78906 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.78906 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 237.46289 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.33398 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 10.00781 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.00391 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.67773 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.35156 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.68555 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.35938 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 53.35547 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.0293 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 69.36328 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.69727 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.36328 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.03711 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.03711 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.70312 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.37695 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.05078 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.72461 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.05859 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.72656 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 129.39258 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.06641 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.73242 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.39844 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 151.40625 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 161.40234 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.07617 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 171.41016 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.74414 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 177.41016 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.74414 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 193.41797 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.0918 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.75781 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.76562 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 219.43945 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.11328 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.78711 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 239.45508 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.12891 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.12891 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.67383 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 27.33984 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.01367 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.67969 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 43.34766 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.02148 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.6875 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 59.36133 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.02734 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.36133 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.03516 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.70898 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.38281 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.05078 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.05469 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.05078 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.72461 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.39844 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 123.39844 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.07227 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.74609 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.41406 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.08008 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.75391 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.42773 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.10156 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.09766 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.43164 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.09766 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 185.43164 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.42773 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.09375 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.09375 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.76758 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.10156 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.77539 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.44141 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.11523 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.11523 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.78906 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.78516 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 253.45898 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 2.66602 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.01367 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 35.34375 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.33984 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.01367 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.6875 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 59.36133 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.36133 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.03516 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.36914 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.04297 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.70898 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.04297 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.03906 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.70508 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.70508 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 109.37109 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.04492 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.04492 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.71289 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.37891 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.05273 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.71875 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.38477 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.39258 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.72656 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.72266 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.38867 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 175.38867 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.72266 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 181.38867 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.0625 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.73633 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.74414 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.07812 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.75195 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.74805 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.42188 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.0957 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.42578 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.09961 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.09961 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 261.43359 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.99609 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.66992 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.66602 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 27.33984 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.33984 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.00586 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.01367 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.6875 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 55.35352 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.02734 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.70117 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.375 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.04102 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.70898 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 91.38281 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 95.37891 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.05273 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.72656 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.73242 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\010) +[8.016 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.73633 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 125.40234 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.06836 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.74219 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.41602 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.08984 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 161.41992 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.09375 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.08984 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.76367 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 185.43164 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 191.43164 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.10547 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.77148 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207.44531 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.7793 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.45312 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.12695 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 227.46094 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.00391 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 14.67773 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.68555 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.68555 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 37.35938 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.35547 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 47.35547 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.0293 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.36328 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.03711 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.70312 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.70312 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.70312 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 91.37695 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.05078 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 107.38477 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.05859 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.73242 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.40625 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.08008 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.08789 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.08398 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.75781 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.0918 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 167.42578 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.0918 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.42578 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.09961 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.77344 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.78125 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.45508 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.12891 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.125 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.45898 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 233.4668 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.46289 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.13672 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 253.4707 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.14453 -219.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.6875 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 29.36133 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.03516 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.70898 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.70508 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 59.37305 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 67.37695 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.37305 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.04688 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.7207 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.7207 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.39453 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.06836 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.73633 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.74414 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.41797 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 131.41406 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.08789 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.75586 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 151.42969 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.76367 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.42969 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.10352 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.10352 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.4375 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.11133 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.44531 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.11914 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.79297 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.45898 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.13281 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.79883 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.79883 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.79883 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.46484 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.13867 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.33398 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 21.33984 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.01367 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.6875 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.36133 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.0293 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.70312 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.37695 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 67.37695 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.71094 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 77.38477 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.71484 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 87.38086 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.38867 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.05469 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.7207 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 109.39453 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.06836 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.73438 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 125.4082 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.07617 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.74219 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 147.41602 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.74609 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.74609 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.41992 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.41992 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.08594 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.75195 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.08594 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.75977 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207.43359 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.76758 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.44141 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.11523 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.78906 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.02148 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.69531 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.0293 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.69531 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.36914 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.04297 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 55.37695 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.05078 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.71875 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.71484 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.38867 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.0625 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.73633 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 101.40234 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.73633 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.41016 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.07617 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 129.41016 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.08398 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.75781 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.75391 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.08789 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.76172 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.75781 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.42383 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.09766 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.77148 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.43945 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 193.43555 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.10938 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.7832 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.11719 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.79102 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.79102 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.45898 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.13281 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.80664 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.80664 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.80664 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 261.48047 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.81445 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.48828 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.99609 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.66211 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.33594 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.00195 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.00195 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.67578 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.3418 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.01562 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 47.34961 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.3457 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.01953 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.01953 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.6875 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\012) +[8.028 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.35352 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.02734 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.70117 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 105.375 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.70898 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.375 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.38281 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.05664 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.72266 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.39648 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.39648 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.07031 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.06641 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.74023 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.4082 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.4082 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.08203 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.74805 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 191.41406 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.74805 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 201.41602 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.41992 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.09375 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.76758 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.76758 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.44141 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.10742 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.77344 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.44727 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.78125 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.78125 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.45508 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 10.66992 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 17.34375 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 27.35156 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.02539 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.69922 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.69922 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 53.37305 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.04688 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.04297 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.05078 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.72461 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 87.39844 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.06445 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.07227 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.07227 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.74609 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 115.41211 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.08594 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.75977 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.09375 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.76758 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.10156 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.10938 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 158.10938 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.7832 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 171.45703 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.13086 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.79688 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.79297 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 197.4668 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.14062 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207.47461 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.14844 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.14453 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.8125 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 239.47852 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.15234 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.82617 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.33398 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 10.00781 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 14.00391 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.67773 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 23.34375 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.67773 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.67383 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.33984 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.00781 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 55.3418 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.00781 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.68164 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.68164 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.34766 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.02148 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.69531 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 93.36914 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.03711 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.03711 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.70312 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 115.37109 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.04492 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.04102 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.71484 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 145.38281 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.7168 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 155.39062 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.38672 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.38281 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.05664 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.73047 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.06445 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.73828 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.06836 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.74219 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 219.41602 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.08398 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.75781 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.75391 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.42773 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 249.42773 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.09375 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.76758 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 23.35547 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.0293 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.70312 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 43.37695 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.05078 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.72461 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.39258 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.05859 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.73242 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.0625 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.0625 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.73633 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.73633 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 113.40234 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.06836 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 125.40234 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.07617 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.75 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.08398 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.75781 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 155.43164 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.10547 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.11328 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.78711 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 185.46094 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.13477 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.80859 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.14258 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.80859 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.48242 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.15625 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 227.49023 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.16406 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.83203 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.82812 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.50195 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.17578 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.84961 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 273.51562 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.84961 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.52344 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.18945 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.52344 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.19727 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.87109 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.86719 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.20117 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.875 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.87109 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 335.53711 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.21094 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.88477 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 355.55273 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 365.54883 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.22266 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.89648 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.23047 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.9043 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.9043 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 401.57227 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.24609 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.91992 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.91992 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.91992 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 433.59375 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.92773 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 443.60156 -324.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.99609 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.66211 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.33594 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.00195 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.00195 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.67578 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.3418 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.01562 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 47.34961 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.3457 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.01953 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.01953 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.6875 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\015) +[2.256 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.02148 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.0293 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.70312 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.36914 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.04297 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.03906 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.71289 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 129.38672 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.7207 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 139.39453 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.72461 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.39844 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.07227 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.73828 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.73828 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 177.4043 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.07812 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.75195 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.08203 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.75586 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.42969 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.0957 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.76172 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.76953 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 233.44336 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.77734 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.77734 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 251.44336 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.44336 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.11719 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.11328 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.10938 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.7832 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 285.45117 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 293.45508 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.12891 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.79492 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 305.46094 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.13477 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.80859 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.14258 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.81641 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.81641 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 341.49023 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.16406 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.17188 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.8457 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 371.51953 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.19336 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.85938 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.19336 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.86719 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 397.54102 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.20898 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.20508 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.87891 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.875 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.54883 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 437.54883 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.88281 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.87891 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 447.54492 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 453.54492 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 456.87891 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 459.54492 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 466.21875 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 472.89258 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.34766 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.02148 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.02148 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 35.35547 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.0293 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 51.36328 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.03711 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.70508 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.37891 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.05273 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.38672 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.06055 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.39453 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.06836 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.73633 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.73242 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.40625 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.07227 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.74609 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.74609 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.41992 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.09375 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.76758 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.77539 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.10938 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.7832 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.7793 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.45312 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.78711 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 215.46094 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.79492 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.12891 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.80273 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.79883 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.47266 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.13867 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.47266 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.14648 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.82031 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 277.49414 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 283.49414 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.82812 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 293.50195 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 299.50195 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.16992 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\024) +[7.704 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.83594 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.50391 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.17773 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.85156 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 341.52539 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.19336 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.18945 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.86328 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 371.53711 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.21094 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.21875 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 391.55273 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.22656 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.89258 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.55859 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.23242 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.23242 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.90039 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 429.57422 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.24219 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 439.57617 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 443.57227 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.23828 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.23828 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 455.57227 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.23828 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 464.91211 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 471.58594 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.99609 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.66211 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.33008 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 27.33398 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.00781 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.67383 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.33984 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.01367 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.6875 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.02148 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.69531 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.69531 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.36914 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.04297 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.05078 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.7168 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 101.39062 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 107.39062 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.06445 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.73047 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.39648 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.73047 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.73047 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.4043 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.07031 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.73633 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 153.4043 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.07812 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.75195 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.42578 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.09375 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.42773 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.10156 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.76758 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.43359 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.10742 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.44141 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.10742 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.10352 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.77734 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 237.45117 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 241.44727 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.12109 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.78711 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.46094 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.12891 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.80273 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 277.47656 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.15039 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.81836 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.81836 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 303.49219 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 309.49219 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.82617 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 315.49219 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.16602 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.83984 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.50586 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.17969 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 351.50977 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.17578 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.84961 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.84961 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 373.52344 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.85742 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.85742 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 395.53125 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 401.53125 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.86523 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 407.53125 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.20508 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.87891 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.54492 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.21875 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.21484 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 7.78711 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 10.45312 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.45312 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 23.12695 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.12305 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.79688 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.13086 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 55.80469 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 59.80078 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.80078 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.80859 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.48242 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.47852 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 93.15234 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.82617 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.49414 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.49414 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.16797 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 123.16406 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 125.83008 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.50391 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.83789 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.51172 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 155.17969 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.17578 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 165.84961 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.18359 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.17969 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.85352 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 193.18359 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.85742 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.53125 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 213.20508 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.53906 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.20703 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.88086 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.54688 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.88086 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 239.87695 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.54297 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.54297 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.2168 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.55078 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.22461 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.55859 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.55859 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 293.23242 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 295.89844 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.56445 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.89844 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.56641 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 317.23242 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 323.90625 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.58008 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 339.91406 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.58789 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 353.26172 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 355.92773 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.59375 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 365.26758 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 371.94141 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 375.27539 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 381.94922 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 387.94922 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.62305 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 401.29688 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 411.30469 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 413.9707 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.64453 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.97852 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.65234 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.64844 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 441.32227 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 447.99609 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 17.34375 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.67773 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 27.35156 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.01953 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.68555 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.35938 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 53.35547 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.0293 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.70312 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 69.36914 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.70312 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 79.37695 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.70703 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 89.37305 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.38086 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.04688 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.7207 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.7168 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.39062 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.7207 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 139.39453 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.0625 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.73633 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.73242 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.40625 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.07422 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.74023 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.41406 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.08789 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.08398 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.75781 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 203.43164 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.09961 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.77344 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 219.43945 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.11328 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.78711 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 239.46094 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.12695 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.46094 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.12891 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.79492 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 267.46875 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.13477 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.80078 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.47461 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.80469 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 299.47852 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.14453 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 305.47852 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 309.47461 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.14062 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.14062 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.81445 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.14844 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.82227 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 343.48828 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.16211 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.1582 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.83203 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 373.50586 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.83984 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.51367 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.84375 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.84375 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.51758 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.19141 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.85742 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.19141 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 425.52539 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.19141 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.19141 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.85938 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.86328 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 455.53711 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.34766 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.34766 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.02148 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.6875 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 37.35352 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.01953 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.35352 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.02734 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.70117 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 69.375 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.04883 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.05664 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.73047 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.4043 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.07031 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 111.4043 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.73828 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 121.41211 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 125.4082 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.08203 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.74805 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.08203 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.08203 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.75586 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.75195 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.74805 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.42188 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.0957 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.10352 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.77734 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 213.45117 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.44727 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.78125 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.11523 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.78125 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.11523 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.78906 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.12305 -414.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.99609 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.66992 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 23.34375 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 27.33984 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.00586 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.33984 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.01367 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.6875 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.68359 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.35742 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.69141 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.35742 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.02344 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.69727 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 91.37109 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 95.36719 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.04102 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.70898 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 115.38281 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.05664 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.73047 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.72656 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 139.40039 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.73438 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.73047 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.73828 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.41211 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.74609 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.74219 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.41602 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.08203 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.74805 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 197.41406 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.74805 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.74805 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 219.42188 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.08789 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.76172 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.43555 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.76953 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.44336 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.77734 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.45117 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.11914 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 265.45312 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.12695 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.80078 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.80078 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 287.4668 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.14062 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.81445 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.14844 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.82227 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.83008 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 329.49609 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.16211 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.83594 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 345.50977 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.17578 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.84961 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 361.51758 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 371.51367 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.1875 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.86133 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.85742 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 391.52344 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.85742 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 407.53125 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.20508 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.87109 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.20508 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 429.53906 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.21289 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.87891 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 441.54492 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.87891 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 457.55273 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.33398 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 10.00781 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 14.00391 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 17.33789 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.01172 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.3418 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 37.3418 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.01562 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.01562 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.01562 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.68164 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.35547 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.02148 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.68945 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.36328 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.03711 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.70312 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.70312 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.36914 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.04297 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.7168 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.05078 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.71875 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 131.38477 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.05859 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.73242 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.06641 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.74023 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 167.41406 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.08789 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.08789 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.42188 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.0957 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.42969 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 205.42969 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.0957 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.0957 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.76953 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.76562 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.76172 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.76953 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.76953 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 251.44336 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.11719 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.79102 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 267.45703 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.13086 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.13086 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.79883 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.58594 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 297.25195 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 303.25195 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 309.92578 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 319.92188 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.5957 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 335.92969 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.60352 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 351.9375 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.61133 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.61133 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 371.2793 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 375.27539 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.94141 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.94141 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.61523 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.61523 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.2832 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.2832 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 411.94922 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.61719 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 425.29102 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 435.28711 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 441.96094 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.62891 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 455.30273 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 457.96875 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.63477 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 467.30859 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 473.98242 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 480.65625 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.68164 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.68555 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 37.35938 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 47.36719 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.04102 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.04883 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.72266 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.38867 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 79.38867 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.38867 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 91.38867 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.05469 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.72266 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 107.39648 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.39258 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.06641 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.73438 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.40039 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.07422 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.08203 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.08203 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.75586 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 165.42188 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.08789 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.75391 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.75391 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.41992 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.75391 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.42773 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.10156 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.76758 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.77539 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 215.44922 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.12305 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.79688 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.46289 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.13672 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.81055 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.14453 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.81836 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 267.49219 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.1582 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.49219 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 285.49219 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.1582 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.49219 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.16602 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.17383 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.84766 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.52148 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.1875 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.85547 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 339.52148 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.19531 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.86133 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 351.52734 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 361.53516 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.20898 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 374.88281 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.54883 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.88281 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.2168 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.89062 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 399.55664 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.22266 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 411.55664 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.23047 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.9043 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 427.57031 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.24414 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 437.57812 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.24609 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\014) +[6.78 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 451.57617 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.25 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 464.25 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 470.25 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 29.34375 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.01758 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.69141 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.35742 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 55.35352 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.02148 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.69531 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.36914 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.04297 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.04297 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 91.37695 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.05078 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 107.38477 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.05859 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.06641 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.06641 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.74023 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.41406 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.08789 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.76172 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.76953 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.44336 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.11719 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.11719 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.78516 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.45898 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.13281 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.79883 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 215.47266 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.13867 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.13867 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.13867 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.80469 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 239.47852 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 249.48633 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.15234 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.16016 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.82617 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.5 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 277.5 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.17383 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.16992 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.83789 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.8418 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.83789 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 319.51172 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.17773 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.18555 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 335.51953 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.19336 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.18945 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.18555 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.85938 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.5332 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.86719 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 379.54102 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 392.87109 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.86719 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.54102 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.875 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.87109 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 417.54492 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.875 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 437.54883 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.22266 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.21875 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.89258 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.89258 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.00391 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 14.67773 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.68555 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.35938 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 35.35547 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.0293 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.0293 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.69531 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.36914 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 67.37695 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.05078 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.7168 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 79.38281 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.05664 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.73047 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.4043 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.73438 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.4082 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.08203 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.74805 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.07812 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.74414 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.75195 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.75195 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 167.42578 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.09961 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.76562 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.76562 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.43945 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.10547 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.7793 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 205.44727 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 215.44336 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.11719 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.79102 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.78711 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.45312 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.78711 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.45312 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.46094 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.12695 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.80078 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.80859 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 283.48242 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.81641 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 299.49023 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 303.48633 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 309.48633 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 315.48633 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 325.48242 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.15625 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.82227 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 341.49609 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.49609 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.16992 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.84375 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 367.51758 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.52539 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.52539 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.19922 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.19531 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.19531 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.86914 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.20312 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.20312 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.86914 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.20312 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.87695 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.88477 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 447.55078 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 2.66602 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.33594 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.00977 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.00586 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.67383 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.33984 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.01367 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.67969 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.3457 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 67.35352 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.34961 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.01562 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.01562 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.68945 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.02344 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.35742 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.03125 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.02734 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 113.36133 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.03516 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 123.36914 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.03711 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.03711 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.71094 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.38477 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 155.38477 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.05859 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.73242 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 175.40625 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.08008 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.74805 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.42188 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.0957 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.76953 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 215.4375 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.77148 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 225.44531 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.11914 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.11914 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.78516 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.45898 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.13281 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.4668 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.14062 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.80664 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.48047 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.1543 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.82812 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 295.49414 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.82812 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 305.49609 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.16992 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.84375 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.83984 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.17383 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 329.50781 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.17383 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 335.50781 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.18164 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.51172 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 359.51953 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.51562 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.18945 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 379.52344 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.19727 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 392.19727 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\024) +[7.704 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.66602 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33398 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.00781 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.68164 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.67773 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.01172 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.3457 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.01172 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.3457 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.01953 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 59.34961 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 69.3457 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.01953 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 79.35352 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.02734 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 95.36133 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.03516 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.04297 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.7168 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 125.39062 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.06445 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.73828 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.74609 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 155.41992 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.09375 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.76758 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 171.43359 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.10742 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.78125 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.11523 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.78906 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207.46289 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.13672 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.80469 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.80469 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 233.47852 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.14453 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.81055 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 241.47656 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.47656 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.14258 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 253.47656 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.15039 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.82422 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.49023 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.49805 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.83203 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.50586 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.17188 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.83789 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.17188 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.8457 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 313.51172 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.17773 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.85156 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.84766 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.84766 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 345.52148 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.51758 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.19141 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.86523 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.19922 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.86719 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.87109 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.86719 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 391.54102 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.20703 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.21484 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 407.54883 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.21484 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.88867 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.88867 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 425.55469 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.22852 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.90234 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 445.57617 -549.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.6875 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 29.36133 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.02734 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.69531 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.0293 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.70312 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.69922 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 59.37305 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.03906 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.37305 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.04688 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.7207 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 87.38672 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.7207 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 103.39453 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.06055 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.73438 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.4082 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.08203 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.74805 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.08203 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.08789 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\010) +[8.016 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.0918 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.75781 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 151.42383 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 158.09766 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.77148 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 171.44531 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.77539 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 191.44922 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.44531 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.11914 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.78711 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.78711 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 221.46094 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.12695 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.80078 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.13477 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.80859 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.48242 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.81641 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.48438 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 265.48828 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.16211 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.16992 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.84375 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.83984 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 299.51367 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 305.51367 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.17969 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.85352 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.52734 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.19531 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.5293 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.20312 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.19922 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 345.5332 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.20703 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 359.53711 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.21094 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.88477 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 375.55078 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.88477 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 391.55859 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.23242 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.22852 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 405.5625 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.89648 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 411.5625 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.89648 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 421.57031 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.90039 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.90039 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 441.57422 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.24805 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.92188 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 461.5957 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 468.26953 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 471.60352 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 474.26953 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 480.26953 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.02148 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.69531 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.70312 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 43.37695 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.05078 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.72461 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.39844 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.40625 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 79.40625 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.08008 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.75391 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.42773 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.10156 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.77539 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.10938 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.77539 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.10938 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.7832 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.7793 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 145.45312 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.78711 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 155.45508 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.12891 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.125 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.79883 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.47266 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.46875 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 193.47656 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.15039 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.82422 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.82031 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.1543 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.48828 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.1543 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.48828 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.16211 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 237.49219 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.16602 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.83984 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.17383 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.18164 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 267.51562 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.18945 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.85547 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.52148 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.85547 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.52148 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.5293 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.20312 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.19922 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.19922 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.87305 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.54102 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 339.54492 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.21875 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.88477 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 351.55078 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.22461 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.89844 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.23242 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 374.90625 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.90625 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 387.58008 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.25391 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.26172 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.93555 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 413.60156 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.26758 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.94141 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.9375 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.9375 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 445.61133 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 449.60742 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 456.28125 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 462.95508 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.34766 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.34766 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.02148 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.02148 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.35547 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.0293 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 51.36328 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.03711 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.36719 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.70117 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.375 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.04883 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.72266 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 91.38867 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.0625 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 101.39648 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.06445 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.06836 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.74219 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 125.4082 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.07422 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.74805 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.42188 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.75586 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 151.42969 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 157.42969 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.10352 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.77734 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.78516 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.78516 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.45117 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.78516 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.45898 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.4668 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.14062 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.80664 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 221.47266 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.14062 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.14844 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.82227 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 251.49609 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.83008 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.83789 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.83398 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 281.50781 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.17383 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.84766 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.84766 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.18164 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.84766 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.85547 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 319.5293 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.20312 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.20312 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 335.53711 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.21094 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.54102 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.21484 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.22266 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.22266 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 374.88867 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.22266 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.89648 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.9043 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 401.57812 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.24414 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.91016 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.24414 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.91211 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.9082 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 433.58203 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.25586 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.25195 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.91797 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.6875 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 29.36133 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.02734 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 35.36133 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.0293 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.02539 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.69922 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.69922 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.69922 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.70703 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 87.38086 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.71484 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 103.38867 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.05469 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.72852 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.40234 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.07617 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.75 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 139.42383 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.75391 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.08789 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.08398 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.75 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.75 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.08398 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.75 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 181.42383 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.09766 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.77148 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 201.43945 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.11328 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.7793 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.45312 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.7832 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.11719 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.79102 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.45703 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.12305 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.79688 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.13086 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.13086 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.80469 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 277.4707 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.13672 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.80273 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.80273 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.46875 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.80273 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.47656 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.15039 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.81641 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.82422 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 327.49805 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.16406 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.83789 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 343.51172 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.17969 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.17969 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.85352 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.85352 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.1875 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.86133 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.85742 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 395.53125 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.20508 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 411.53906 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.21289 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.87891 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.87891 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.87891 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 439.55273 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.22656 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.90039 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 466.23047 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 472.9043 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.99609 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.66211 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.33008 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 27.33398 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.33008 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.00391 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.67773 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.67773 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.35156 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.02539 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.69336 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.69336 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 83.36719 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.04102 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.71484 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 103.38867 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 113.39648 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.73047 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.39648 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.07031 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.07031 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.73633 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.41016 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.08398 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.75781 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 161.42578 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.0918 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.76562 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.76172 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 181.43555 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 191.43164 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.09961 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.76562 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.77344 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.77344 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.44727 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 233.44336 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.11719 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.79102 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.12109 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.78711 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.46094 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.46094 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.79492 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.46875 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.80273 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.80273 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.47656 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.15039 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.15039 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.82422 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 327.49805 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.17188 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.8457 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.51367 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.1875 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.86133 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 367.5293 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.19531 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.86914 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 385.53516 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.20117 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.20898 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.88281 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 407.54883 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.88281 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.87891 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 417.54492 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.54492 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.21875 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 439.55273 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\037) +[1.836 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.21875 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.89258 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.89258 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.22656 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 468.23438 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.99609 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.66211 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.66211 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.33594 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.66992 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.00391 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.66602 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.33984 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.01367 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.67969 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.3457 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.01172 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.01953 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.68555 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 79.35938 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.35938 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.0332 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.69922 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.36523 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 103.36523 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.0332 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.82031 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.48633 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.48633 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.16016 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.15625 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.83008 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.16406 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 165.83789 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.51172 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.18555 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 185.18555 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.51953 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.19336 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.52734 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.52734 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.20117 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 219.86719 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.54102 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.875 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.54883 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.22266 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 249.89062 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.89062 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.56445 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 275.89453 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.56836 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 285.23438 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.9082 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.57422 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 297.9082 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.58203 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 311.25586 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.26367 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 327.9375 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.61133 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 337.27734 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 343.27734 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 345.94336 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.61719 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 359.29102 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.625 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.29297 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.29688 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.9707 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 389.9707 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.64453 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.31836 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.99219 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.66602 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 419.33203 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 425.33203 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.33203 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.33398 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 10.00781 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 14.00391 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.67383 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 37.34766 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.68164 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 47.35547 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.68555 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.01953 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.69336 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.69336 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 79.35938 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.02539 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.69141 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.69141 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 93.35742 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.69141 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 105.35742 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.03125 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.70508 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.70117 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 129.375 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.04883 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 139.38281 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.05078 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\015) +[2.256 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.38477 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.39258 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.72656 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.40039 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.39648 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.39258 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.06641 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.74023 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.07422 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.74805 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.74414 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.41211 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.08594 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.75977 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 239.42578 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.75586 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.42969 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.4375 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.11133 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.10742 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.78125 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.78125 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 295.44727 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.12109 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.12891 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.80273 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 325.47656 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 329.47266 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.80664 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.14062 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.80664 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.14062 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.81445 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.14844 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.81641 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 361.48242 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.15625 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 374.15625 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.49023 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.16406 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 393.49805 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.83203 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.50586 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.17188 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.83789 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.17188 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.8457 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.51953 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 437.51953 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.19336 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.86719 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.86328 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.99609 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.66211 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.66211 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.33594 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.33594 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.00391 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.00391 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.67773 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 51.35156 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.02539 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.69922 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.70703 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.38086 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.05469 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.7207 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.39453 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.06055 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.06055 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.06055 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.72656 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 121.40039 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 131.4082 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.74219 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.41602 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 145.41211 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.08594 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.75195 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.08594 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.75977 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.42578 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.42578 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 185.42578 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 191.42578 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.0918 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 197.42578 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.09961 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.10742 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.10352 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.77734 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.77734 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.77734 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 249.45117 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.11914 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\014) +[6.78 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.44922 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.12305 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.12305 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.12305 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.13086 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.80469 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.4707 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.14453 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.81836 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.49219 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.1582 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.82617 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 333.49219 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 343.48828 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.16211 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.83594 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.83203 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 367.50586 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.17188 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.8457 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.51367 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.84766 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 393.52148 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 397.51758 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 407.51367 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.1875 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.86133 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.19531 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.86914 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.86523 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 447.5332 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 456.19922 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.86523 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 465.53906 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 471.53906 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.68164 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 23.35547 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.0293 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.69727 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.03125 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.70508 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.70508 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 55.37109 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.70508 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.70508 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.70508 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 77.37891 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 83.37891 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.04492 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.71875 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.71875 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 105.39258 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 115.40039 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.07422 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.08203 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.74805 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.41406 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.74805 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 147.42188 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 151.41797 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 161.42578 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.75977 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 171.43359 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 175.42969 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.10352 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.77734 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.45117 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.125 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.79297 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 215.4668 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.14062 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.4707 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.4707 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.14453 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.81836 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.49219 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.16602 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.83203 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.83984 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 281.51367 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.1875 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.1875 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 297.52148 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.51758 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.19141 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.85938 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.5332 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.20703 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 335.53711 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.20312 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.87695 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.87695 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 357.55078 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.22461 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 367.55859 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 374.23242 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.56641 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.24023 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 392.90625 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.90234 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.57617 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.25 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.92383 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 429.59766 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.27148 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.27148 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.00391 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 14.67773 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.67773 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 27.35156 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.02539 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.69922 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 47.37305 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.03906 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.03906 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.03906 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.04688 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.7207 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.38672 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.06055 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.73438 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.40039 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 103.40039 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 109.40039 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.06641 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 125.39648 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 131.39648 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.07031 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.74414 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.74414 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 157.41797 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.41797 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.75195 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.42578 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.75977 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.43359 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.76367 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 193.42969 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.10352 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.77734 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 213.44531 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.11914 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.45312 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.45312 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.11914 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.11914 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.79297 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.78906 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.78516 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.79297 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 275.4668 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.46289 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 285.46289 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.46289 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 297.46289 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.13672 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.81055 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 317.48438 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.15039 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.14648 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.82031 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 343.49414 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.82812 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 353.50195 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.83203 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 373.50586 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.83984 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.50781 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.84961 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 397.52344 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.52344 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.85742 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.52344 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.19727 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.87109 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 425.53711 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.21094 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.00586 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67969 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.35352 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.02734 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.35742 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.03125 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.70508 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.37109 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.04492 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.375 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.04883 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.05664 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.05273 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.71875 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.71875 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 111.39258 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.72656 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.40039 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.40039 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 139.40039 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.07422 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.07031 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.07031 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.74414 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.75195 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.75195 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 191.42578 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 201.43359 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.76758 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.44141 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.11523 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.78906 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 227.45508 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.12891 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.79688 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.46289 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.12891 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.80273 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.47656 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.47266 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 273.48047 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.1543 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.82031 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.49414 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 299.49023 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.16406 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.83789 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.17188 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.8457 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.8418 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +Q +Q +q +1 w +[] 0 d +0 J +0 j +[1 0 0 1 360 805.89] cm +q +0 -300 200 300 re +W* +[200 0 0 300 0 -300] cm +/DeviceRGB setcolorspace +<< + /ImageType 1 + /Width 512 + /Height 768 + /ImageMatrix [512 0 0 -768 0 768] + /BitsPerComponent 8 + /Decode [0 1 0 1 0 1] + /DataSource currentfile + /ASCII85Decode filter + << >> /FlateDecode filter +>> +pdfIm +Gas`+LN4>:en+d[1Ms-tPpADPK_M(p4*V=nHgeYCQWI2[f/*F?s81qM,XG&Q0bGok +/jYSV:mKTqkM2P`]Pf+Z?-.C\C%k*>.>^-^0rI.rPf?rI&_C4=\Irp6K7Ig\GHl7< +^?iKF]5/F(kE=HgF!18P7-^D`b+!!5e/q7P'D6UM)[s-F[]32kQ&=mqoMRHT$%\P[:')]]6fXc=)PgCf$<5dMtXG8/;fN'l!-.[gfX +UTUQI'TJ!_?C!ILnlB&IBLnRdEN:C"\&G#hP]Z3=2SPZI[8Mm*;V=cX4)mtNXos`O +U)m$+c6.4[[f,HRu"je\%uID$D;]afhH\ak?2K*@#Mt3PbH^8719ls8Co62W&s%jD5tT]^$$`8t +:5S:W_f61)p<+2keZ.*,k-D2%*G]!5]$I_NpML-3)kioa)O2W^MI]VV:+JUWUWOA% +\Y:EXdMBuo@RXM$C!1W>W&1kD/KLduoq:*f=1UFhmV#719sLP?[o=L%<3eq[RB%jD +20l&K8QWi5N_F^s!+4lg&pc]JOVFfn:oqc2Fa:NkW&0diQE&ZieiLBg=5W3ZRXJ[] +?tp4sS^CJVZI>B3p2B[C2i!%?UZ(&!kl,,jE&r8,"4O1>l-mWcPR2U@![;\\N']>g$s<-3$'(K#CBk*hsGX]6*IuRX`(\L +ekX3+PNI&DdYaH^;,\iOBn]7*TX!H7[JXX`%TnRjJ",X* +G5B$\JAiY<#DV2c!c.WJ0aR.>rB"CHB/G*l^i."Am.e$CfB06ng'_]t/-r?3Trh(7 +)dq&#=rDm.d-t3(f9VlW>kOJa*_XIh0"XfMkVs^V?a(N?1LmOCZ\6U +`o9[_5#nLU<$6H7:*#-H#NG*a\9&jrm7UUfddEfFGUORS8o>-X[muq^a.#lFiqK@? ++3qf3/nAmn0?ekM(*rT`\nY=k2 +j>K339QtC'XkRsX1!SM9M;(.9R79s3ct=5$S&*Ha)@%hBd+0IU`<+X_co_cr(!A.E +'ps_M`%5B[M-:%Z&b@8a!ooA^'Jt>+aNtI_5qens#'HJe(:=#YU?,=_Y-g[PVsF," +Xan,+1j$X4?GtW[=p?]3(rj$3nSKt?nV_$X]Aqtt34U=h8R&*\/Rk6TJ76G+#&2O/ +D&B,YrIh=LqFOiou#3D6oL"HM,?r$N[lI`'dKQFVH(Q/ +<\qptDpm?B8i'2AM+L#Cq<,D+)>@=W[@VJr@s/073]>ssY]CQA&)*?O[WfI/\(Bg3 +]Osn2$PE&@iQ.Oc,kZC0fbiXOt@8?BW, +l1A^;]DB\MQ,YjeoRFJ'",Hr?5>PM7BM(;Gc%_aP^V!U4_uC^^Ak`X42]A0A!%.`+ +"TIjJADO8[fP[+534s5->^J+nGhV>3sk=M*U1pqM$Za-[ku7KB7b(Bio/0a.iMT:!8UL\XO@ +PPZ"I`D>I5muGFI8#,;\j?FP)N/%5Br-LN=2P]$\mF8Scs)$']m@=!5r8/!R=TelO +!8?lE[m2mZrG6W%q6(q:$iYq6)89rs0W4XXP!AcWj0K;KrrIc`o99*A!=N2&riJEY +^Tl]Rj;n)e?'fQp8H'`OTg8fHrlK5(nrts1YqgfK:#iSJ:]FW#+&7`_5" +:D!U,'-k8;Y'kjCq#;[Uq&Yq'->$PB!;N1:s+L^YL]HEp"aiA*ldpb)?H77WqJOq. +6id,qo.NF5!(Mf^0T94l;8UIR+CtSK1b]$Wr2+qc#iL(qraufS;g"j-!;R:i5OmJ- +.J9G8+b)sR6iP+FAoT]LH.TUdIm*$1-1'qpqZ_j96kRISaL:HMH4-l/hKVfQ3)AGC +i3)[\NiRBUSu$+B)()OG+b/0iL<HIZ)R'?k&PgF44cMl$lXND57G>oRGSt"Y#oF,d5]? +FeMEtWSi8hJ,E+Qs/-ARn:,%g1Uk+rhA=*0SHESU_>j8"s.@T4n'<;P0,.bGQ!^^o +he_Rbm@)oPg>AI4J%P?f\TrYu(]Y>d\rZu>aT%H&o>uiip7kRdhPkj@lG9%/E8re! +S;+#d"N^UA&%$7$`do`:Ik_"2ao<]n"`@gNJJTbc04.0K0ao!%V#Ul)=?WGbe-L-u +#([hN+Fh]:`dZDoMODM)$c$3c[/qKTYbhV5&49?_!#At00TD7X@1&A+8P].S6T:]O +bfpYZ;0=7\+B[_9s$rLQd/qP(!%Htn8gq_O`M3:nB]`m@C)Z-SHGpcJ-AqhV@kDS* +NS.>V*0>[.nt#f<i[qQ,=pE=ql7]Il:&PX>n-Y^,,(!@C_Prdp9Q2^2C4.S3Vgir&P4/lM$G;f^X"'D +?``7R_#I[<061B8$M*pP2QiV:51]uhr]1"6;?)EJJ)pW-=]=J\?`iQP1dV>,s!\!l +r)`kN1n"BM%0bfgrm0/ke,h?(Q_g+5,[D,%s$?ib-jUEhkl5Bf?0c7j4mgGgO/4^_ +*aTo'[qKO^Gjje%!VT$SUPLHahM[) +r84ZGiVus3fS'0\VH1tBs0pE'np_>PNu?%='EOco('i'u"8\..5a!3X%Z&(]I4;N' +HsmQ-HRpbX)0@iOh/o&%*qKG&+@L_[Cerb$^akC[;rcU&Sc=[\Q_EJfp4(Z5qd5G7 +nm%SB;^>.!Nofdu.JusdT_(G:s4W%6`VG"..%VDMB?(YIs)[GRL$\cKm*iKb&HMhu +rnj5LP5Hj+s6@u>@ipm67u-5Z;tIi!HZEfg*P+=fDW];uWC&mi)Sqf!!9X;sbLlRC +s3o=1a0Ye;rdt1=i\,n6?3.6-J:_/CiDY8Y!"f3Uk6Hu7?QdS-@1Y.'?cdcui>"0T +"N#c]Je9dIil6G!i6;c]0r!rGs2S7'chUoPs0Due1AUZ2'#FUCKH0R_Z2YX$5D^kH +X@Jk%b>QV[p`K'ffp8C794?RQphDg6[H,)-#/'u^IW8Z+om2&sLR'`aokI4PmAU:' +A+9%Zr@_qcPguM(>S=sSa$Nf5&nXsM1d-/r.^S$Q%8nb +n:*'galh3I=Ob4fQ)\^SJ*)60s/H"\rI9;VH8o)e!&D1eS3Q0>#la"6#HmiWs*Kqm +niu3mlYlC65P)Xdh(Z>'VIn'Ys#3RTcmfmks(9:'9H0EL@k28,5lMa?aIe@jaM/\I +mRCU`>qh'UJ60Smj6.ZV6iXL`Id[6Eb6/_Y,H/5!)K$4e_;T9[c!qmrIiegC#gA3V +nir$4pn&a\r!uCpIF4b341%#[->QPXCoZQ]13'9a>cPk,R+XDF-$S#lC'Le7RSBYPns/WRda@,GfmCJ>,1/d;%!@=D%T8F'hU[4!Pl +,Zqtac*da,a`+>M9Jke7Fu%QA#DS_^H.k09WeOBm0J$lAH$)?L@:Zd`XEK]V!QkIc +">h69V\HP;\/8=@\:9CgrtN'@?do-fS3LYVD_d?LKRG:`?U`k:p.Edt=je[Xq\deS +,SlP`YVAls.R@(,:.R04hPiM\3a2Gq&bb=>^R;"olX1X3c=Vnf>bKn7('p[d6.]Eh>!Fc[U+Crd=*^ +'+V;'gM1R=o'[)h!:kobo-I6h]b6=,KYkp+i,Ud[s0!ONi\;&CT1lGni]efBrt,p7 +r5C)`.kO?#/ZlJ#20^1fn&srf$R>eh\*tGFr_6P'_>>=_-RpERV'N7/V<%YW8(o%7 +7MGH*VF:V]8(sGpl#pIKm9+(=lV5P`WuIonr<-8"efbMm7K"pd)!#d\7498scn)O8 +!V+E>peBQ@\"N\L,6\_*T2MhAAmN14h!`[oG8^-IhR?VrbDJ"h*6q))ZRKHZdR(b*(B)\47HXD)1p.6R;->`qAGjcS"b@ +`ILk%Dr1gcroAPF3X'NaR^ER/O?=Rb,U2eioFLk,dSq_X:ocA:2*Ld8<1u@X/XR3`udl#&CXgYLj2R]'O7buFiklL]h059_U%3hI5 +f@A@(>"Z[V+JEHIO6]sU'?S!.gZladISTn4rF'Ml'L)ZGrZ]hTd3QTPTDU@L!4;bV +6id%T.Xs4'eY1$kio1RJs(W+pL%aL*GqVFd0DZbT;Y:qX*^*ADlUidHdAb5LeN&C7 +@oK2,jJ`OLIj0=mM&hFm"c58'!]%gUr3<&>/?Ku:R`D_ZM5#2D+.gWpBC_O5q<`mq +r_a4/s*qIm5.fIH*iT(-fD%LR*H1`2@P`L"rc`PQND7ISJ#+7AoT6iGs,l%=?MT'h +Y`f$CgK3s,DVrJc%rj+E9Pi)Xc@8N)$S8.T\;bF_Ec[pWFDDl(NZo5?.I]L\OI:9O +8cI'8O-h7K$[Obt2o@VoE?,*&4cfI'c +oeeX8+"@9O%kX"C,I)HSIiu8^&%i8Rd/pDhU7umKSb)jgq*.qc036#QJ$CUY20f/= +n5T;nk$Nf>:[bBonAiYls-5)&s)Ls;s&U`;Q$PgGl,a/TjeDcf6Tt&@^iJ&aH5Q!3 +%(EINkuq[Ne"gk"f?@W#0d-qEAL"AVOrb%S\kZS^8BXW)eTCVlAgd[=YPd:fg:n:ap/%79-IgY*UteXV]Kp]W.E! +$ahB$Sc?=XNUk_$#&,))7"/bD&\&L9l;J,"9/%#M$%p8^b503`4i9H>d9%] +K2-ClnomYL*XV"Z=YccQn:)pb9ZL/dUcP.!(,EHf?rNkX2t2&f$YpOGLs/E3je +pV3$c^RG(-:[74"/b=J?0UMf_5I&[55lR;R'*mFl+7a(DO7%X?o^uIF5QAVi!)8"I +s&0'9s+:.Lj:HD-j7.depS!%l"8Q1BQ5rR&Fc +qs7^O8,pPDpj-fTjtd#hBT3fm\TNuuJGEQnhkhE"knnUBi2-GsJ&206^pYBcg[G[T +k)X?77CNHAr/3Z`\)4*,!#C1dKbV%VmgqF_D2J>Q<=6P>PH]bY,d96i,6(Rp+NU2Q +h[Q%\14MBaisBF'@1>Xj1T]OA.>.t?L1RKK%0#]bQ/qs?:soZ,&HO/,nioRY9E2'7 +s"sNn7sOij6gCaSS9AfrSSJPPCZBd6t. +q,?Acq4l5jo+d79]K&ru7`s:IYuq%d]6I,RUStW_<<$%bL]BH./tc6B=3HIn7>,^:smnIgXp/'XdreJI(cC$fhE5Q2"rIVeVR* +SZ8r#1Ua<<3=)b!IGVljU\Q'ASE<0^pMHq9h<82Jl'Ug"g]Ckcer +TrX1"Q'gqcgSDY9$,@a$Q\Cs8TS''@Zla;/_KKYG+ca.0&W\L.Q,g%=$,7Z!)Hpn>MKLMkZi>$i2q6NL +;Xco@:ZePYL_ht&qOb.qjSu5"'P[uTs5AIK^XCY_\bloo[/`9A[X]1Zs6&e-"ogCG +Et!_GE_u_lSSe*J>O70ncj]r`qSl!kRD7.tesdRA#?9J-'1`.k^Z%QXY:2N_"c!HGm7P(r +\07,C(,%aL?W$e=<)a[LaF +k">GoV1*@7IW4L73q_EpbOF2%r8KGZopWX$>JBp8i2b2pjkBXcC\m*H5.MQdJPUIO +LQ29"3`Q)Xr"!YU^.ek4%pdN\(Zta&rP/<*rX\JQZ'p/4J80Pu!Z'nt+h^M\n0/>R +&BXm+`84d/QTu*)"b\K.E&Bj5s(TD4KC_,!0:jW36L[3(7uGTKDmB/2Fq[2g3a#(1 +8;.=_1$u"\*)-5Q"ePi.Gr$iP8:af4dLM'6MZ>@5patEXqO`ZG2:$@6cf@:tWeda# +1[SLA3`d*q&,E!qpKalrCb$[t=tHtJc&/MYmV!'j5EoSRY+qRqKD]59j\Y^mr9s6l +-3&bYrs3!3$i\Hd2mNhjkDTA3]J35@LT1M[(=`4:JFA))7oA; +dYLVP/&Nut397aD8c'!#KNl@o0t/^ms0]-5&Pt"D=:fH:,PqfH0@e]b+V5Pq"As'&b,&-(889`DGS +\mF">C78m;c2L3_62p.a5m@;?ZRYN1=o]<=SH%=mj4sZ='EOa(pYgU>$XfV):SSsX +s6p'>rtAH5;Zs`M":6"JBE;G2p+b/]VYT9Xni+7D5?I69m6Nt@78O>ANo/V$=E&-S +ciP&@\ei+R#:aQhUS\ +1M-/Wd5b/!cdhObpe!?S;W$+`[s%.k:&X/"U87hIdY!Tu"!eE-6hgZD-_C;*0g2hS +^"@mcG2SSXHl(ispn&a;YaU^edS +mc/Zt:a84jpjb1#2K9\l(+)enBd-_3qYut^FV$I[L2o4\i/a/K;h,HZ6iI-?r!u,A +$c&nGl*LDB'!qc-G(>s-(fp_*'WC;b3;Hnjs,Y%@s20@"jKD#=r4d#)j^9Z*oP-C2 +Yo:,S]fod5>cgE,U/Z6JO;$4bZ;+3L+%[W*o*A0C=r6:pJjhq/QiOp +b$%L?W`;(T+,a`(eK&=Y;_@ee_#LlBdqi\;FnOd\ERqS(;Lc@V0\q5K*B`-tI_7]: +FF3l[mYp1X(16LU5<\L`]K`rrDBf5J7;4[V0&lh6d.-:\rQe?m:W_pPAqP"oM:G'j +g+KB%2.Nh-jtLpn`bn6&0'ZjD<_h8fB]+te>Z6Z]M]lXqTZm43)6YOSP/eTk\4mAE +G"KOtMqXt_J1)1VH-DttPqCYUfQ(XI]Ne)CDE7Wep8bMcaBt-jW,kg15,QRPFN'K) +r*RSab'E%&s$lc*.8VI.i.%J%IAU2q'eWiL8BFY<]7=]m1&b=*0A^RA)p=VL_j8@!k5S[=77hb.hH3_8[Z +_#Jc[fH.L0E%m4tX:^]NRrol3c +Io&+k_uJ8.AaNaU4BJr9Pe@Iti'IPGUM.EI1%#SR:-Ilf.dXtVY8M`#[X@XK_.Wc` +Y5"/dJ+#6Z]$9SB?Pg?5CZ[HJ*C_s\[i)uscJ^UCs1p5chtX[!K3NPnmCUFl\<$'? +p'!W*GWmJK5,Is;V[pS`be]),5/:Wi:YEVrRA9N@6](l0!m*Uj!b`d"'AK'HGDM^o +6h!;^!>.q]kPGWnq0s6d_#Jr_:\U([2]D]E!YgR"36;,5":tSZIgQ7g.:<]eJ72+X +7WB.ReoUWtkOK_keO?Yimp=8Kf@!Dk_Qal]XTYh.!gh(R#NWWlno*"mh7d>P#-qj1 +aZj-7=d6Zcn.'"o#_p\6X?cd4fG`hC88]]YU/[_AIjY1^<$*+-K+s +e;('D8DRZ&5lOiJr5nu(2Qk:(Q)52bYL\!p5Pr3uBQho(&`g,7:Bgb;DC3Bf^bBZl +LCO3XBq;Z@mfo`UpJB_Ag4)=Mq(Oo/G]WTCp)O)3Ee!oXiTf0mmFN`_[gtCVrVb^B +A@paKQF#R]i(_ks2fo)iJ)^=nRGcDffLda%dE3uO-!durka"rVP+c+0\+=p5^R8]/ +".6h=eWb^OU0jJ+AfL/@O##`nV&Th%57-[s!ait64@tkpiDd`9Ea3.MGY?6CN:($i(Uu\me!5:)P5T9%F:Jg*f!Vb7i09?E"Qf%TkbKd!59]eD5lThoDU,R=9R^<-J4TZlkWqG= +h[And3"*Z"K^qDo;]l)9F,=:.PlJO3L&&cZp.&B=$q!+/i0F@9JY88PJ8"fg9s5'+ +#8O_Zlr"u.KGItCs(DFH&-Q>8^O327s6CK$eqL\`Y3qm^qNmYu'q`h?n&bJ`r)s%NECS\*f4 +r;RVi"&H3#LY2T9@LkRIpVHUp(\j.Rr1#hMcp!$Ra35,>!*oR^!e;JFA,eI$LBmmh +;$Qs5Ge9BJl-(Hm2YhjmjtcB(1,Bq!lGNXNI>9sR6IYT\4C[;]$8d?,JrbGTS0@t( +DZh=j$c(3tZ9t>aY\ZT!"`l-.#6Ct>'c<&1ao?h_^o&Y:s,1p#BE*a'9-NK1W,Ess +'p\BR7U=U2WD\hXUIJkJ'6m^3hp;G^jfsg?I*D`IoZ-q\]tW +EW6aIr[f/c=!@FCRM<08MJb`7bo8](@k4=0p0in`64<0^6CLe;;t?pZ*`^LfbdpbA +`i0)TLOu\.Gm(?Ji.R`rW1eM<]FN@bA]h-L2fMp8q`jCZr\$Z]`MajK0-q(sIjJs3 +B$AeqWKoKp$DqRpLb&D\.,h0RcjLDM1/FmF/!LqIOu?BRA4t:`$5%e=a^=Oe/16k_ +T+T7b$G?F^!H_6'Q2bP$<:4EM?!XRN=Rc@Ap\ksM">RFL)7%NE1L)pUrQfn9U%mf* +s1:MfJ$#j7,G:b\EUkauj^b=hQi&T9:D#jO!2TZ>i')EOrkF@G"TQf[ +!.;>*(Y]P;Qh,LMMs!_=o'Z_joE5Kq^O<6HRAW;k:-G6ms'rmIjbSFV5Z[VP!@"2u +B.6*tDFiUlb53=cJEhW;Rf#AU0_p?qHbJ>MAY#Q[KD@as=G_?)U;!IGI&2p3D`4Hj +^2c&>rpOqSot%(ugdn>EUJ[9SDhN[+jA5ac7TVp]eS&bFSleX]@ir=B* +=$"b:kC8Z0^ter^;!t>Cs)U#^jT^ouV']h?rga#Y9"h(f%^^knDD9HHa"$NT^MYs\ +dI87W+&n5PMJ60MO/lL6qT_fVn#h*.e33ZT5P[8urA6Y63'.'r^A'h$r&V04]sT'g +IXcJ=qoSNsa/^jhk>+$lZgLl#TZi-k,4<2N*Ij*Zrdkn.+8hrT=`,K2mBimWd<8FS +P[HYOX[R?^&CZ*EX"Lt@"W'79#I8Maf$9dM%g#2NWQmJ&Lk%=[pP`=mj&MphJZs*m +n1F?>Q^U?nKB1@a+H9F=Vp5IQq&o/+Eh)rtjBp)6R/Au&ImF#5@0BC6PY),[q31ib +,n3D325q3paL_o.q[I=$#.*;o$)1_b13CT>tUW**p +V/O^d?toU`i,6OCs7\g*#;H0*G2WBD]+JEeLArLu;`)kh0_q=bTGuTR"S;bBJ?/^B +,Cnk$[0@D8q7uo9'\J$8B+#r]EF`.$,(KbE@-Ip?T^+CWXpkit63Og*0-``=#^B*q +.gG&F+TFn1UBP=9Wd4VOBSSLks)nL33OI_aWOm<;ao<%U,^^a:"TL\k>OT70i6GhD +rc_T*IkYo3LmS*j;>bq4Iu46s5PUU.JGBBE$ect9ra57E@.BLX=e:Zo>Qc$0-4_1r +LPUKXpq?a)!6'MN#\8B&#Or=t$X"ftq/aju@Wr9Cr;/ZImoftf\%F2J!;;4/NWoaa +TB+N$t6&T25I*gjPiW^PjiF;:p)(Ps;AM' +InmM'8mhTCn,K4(>,t/iXf/67ePq148jHVSCPp^^cMZH`CV,==[tOdl+FsJ,BEoA, +%3g=`"-4)U]W.K'!U*12RsP6dO%U^.!@N@S@Np0Dq`c1RP5g6q+9(Hhs#b'"W!-9? +";tRR!(@nV(A/eWdcZM!>`m,m;cKYel=AJg^cVcg +hmWUT5ICKf[YEjMmHNg**t-2_p]#=WOSMcOs&]$2Nds@`s#+^K2#Bc5s5NrhdJr"( +H2l.h9`A0&@ibYa!J\^%nimt49VMsD^+fS"\I^ii`7RXY3\1De/ta0=mf`?qra!n@ +mEh&HBILf&KDE<4'\FH3G>S353WJ-k:1FO8p`Zfh(tST>T?AEd*8(N1!L*YAJ>V3( +-j>80TG4%4Th#2""S6\#faFtR^JKG)#M&j+s1=WO_3L=)!UPQ?SNXRC`bDr*)=Q@X +H]K&;_FL0sjP*i@YH[$dHCI/&gitA+(LuIe)1*oR5(WCrAb#f@r]_P$InC^oT@6?\k!5j>,'K"c@ngFH]]kX5E&H(Ehn39?,bqE]Ai02cqA`S>uJ>Q,r\<[,* +6,tWK":-IG4cAn"q3:Auab:#Q9KUj)/lD1CP`)[\Zg7:q(sX3ujSb4Ne(n[o7HO)T +5/3'!588D_n@tO?r7?<^hZ)Q%.$=S2gmK5r]AY9T.">C!/h8g@,l`0d`D(U<64;-[ +CK_CRi.!?\eE$WZ(BJ*ms6SY0K)]Q[e*61Q'B==Lm#T]1J[.4r%u9H?5We5rVd8>m +AS1op(csbVUpr34mG:h9n#H]OTlp_0lUMJFj2^9gbC\"GV828NmH`!dp#58\4kh4\ +0K[#7m<+\#kWrYqp[hj +*"a6h,+@l1N^sorAgI0"76B +MSd3=+R&h/"9Q?2&I=,@#t<^('2.<&gLt=NQj'F\etaSkEf4es +QFgtG"2i^p./5Zm1"%u#]%091"]6B,C'/=TbKHN)H5L"nER-8a=)5B>+SN]0/m#`u[eTq?,!8Dj[obI9r\#Bp:K7F4Np3l_^n->MK=)V1!;i5'X"G +5#tFMWi+d'rT$oCr(HeL>MoF&D#8A?!2%7I:]6JKh-*Io1=pGS[\#h-nb9E^+$'(b +8jJ%FTGj-I!nsYM=1QnJ]r:tiGr;/#>rM_CJ.d(,nEo3N+S9[H34Y&iEYKVHVF#8=LP9ptnum>8Q5(TO)0%*@U*1 +d(Vb"Y:J\@+J7i=(:9LB-CI^XZ9TJSaeN!1,U3es2aaR(E-`an=q;rKN79=T5SsXC +nsbK1,ZU/,/+QA6OfPPHuQh8C5>d]jJRfV,k92n",=g(%e@ZurjT +k0[*1TFhb70-uKWMNU[?f]&&j<\II/JGQ+RH)9b1BmWpm^.gi)@Y]-(%O7?$:#O$& +"uOuIo2l$QH8skfmsU-"h_&p7gUXlOSmK>(f$Q?R4_K*W!RqKR.,8&`DLL8`rZ1J> +gj9Is5D9X)7h.aOO2/fgnYc0(QfK%ZRQCJ33r,4."gn&j3Vt[\rd4Y8hh""Ff;r-eSq<(+R/Sp-)J*`9qt.fXNigoSX[*SGD-(_!7shIa+%?EK(l +q=R^%/K.nAs+^Yo!9B2IJB:%P(B0@?hnf>SIj2K3q54TEaIb]gq9o9gru_(Z_ep9-X8+Pugk3e3!q[o$HAh1g%YnZk+4jh+hs%.F>m6.:JXZL$l>ZLn +(nJ"KE!GF"-\5Y#liM?(AJucWY`*/,JR7)=W)L+UL57T +=V)MZQ#c5YBIU,8$1cDW-j"EL9fB%IkL>+`iu(uV.hhj,S"BdK5[=@I!p'Al8W!du +jaWGCIDAq`?nVJ-k5Upg*ZP!@AL;GNj[HT(@i.$&X-Ukg$q(MUEhR?5Y^VAVK ++FirH&H#=2_Yb[/niu22rkB!1$N=Css6%OGt:s&XgJM/NP:Yo!L. +$;^j:'EK9hN">Gg8cag"]?!0_,Z/OHHiK.YQ=7Qk%.O8bqjriKOoIKD2<7Oa7KDNL +Sc<9YqnMBX\%IU,O21u#h[Y&&J#`NiY*Frm-%!=g$]?KK8[GG\G1;.qC,AEM/8KGf +X4>a0EAa4g9uN>BTm,RIT('?_jD?19/fJ)Kq$$RAF/t5iB4jEoZ>tSa5>5:c^g9n&M4"D5L&`t-Z8OfGPM@#^2QV/!%8!lRYU@BM9s9UE?#spd1pg/7Si9g@;DGCV%WBpN)RKJZT,J:AWa7iGI,UbLSH%-crqEA]#54X(7b),O +ANrP"Pbo39+c#TtT,*_M!@%B,]pYLJO^:!9DN +gN?eFQ^45glAlr:!i@UDTj=k_:]@!0TBFMCKDH\6I+uRrY]JisJ'"+NPQ1>cnPB-# +MLW_IHiHJ.MkKiPoKPbEnaDVMr)Rig?]0(N!XOY`+U*NApn@`f9E"n5rtGLICYo*_ +gVG`6&;-fE5hkpHK]4(QY%;1J'G2HaF#3)&)/-Mt/-W0o>S+sV,,"+_038)5s/]in +k?H2d7-2ofUEo05)>b9aT&cuj.2smGHod5j<%%- +Mkua@ae'nEK)DVCY;CP#ZQoG?b$UP0OT4'66YgGrTa2JRfH\90m,OhuIs0RYgOKR& +a&f_'`_Kmi)]FGU'-a+,1UPojAlIN!X(k?H;W-7.%F_h'643ehF8k%t,lcWS4qRWf +#Us_"4a=%M4ge&B\'H,MVg",Q;9@TfX6VK3gANpQnPe6B&+fpZJ:(b8(g=$Hq +TF7^pFI`BoceWX,MsMOP"s!^`rbr;Jq7h)i+SjXe7Y293HcZ5F/H^?@5[P%k)A.$8 +Anf90NMND=ZF81cbr!mu41Ealq4BB#F$-k#WL>\F1AXm<#E!540(T.c57RFd<:Qb> +SruTB1([$7K?;A'@Y80M,/>"!TccT'K[X`i,d[Q$jF7mD,_#giG@o)!<;El[s8HPR +J,Xl(&)g=AgD9b6,2bQ_Z]/H_$1IoMm[2f3`h39A>6'!C]^(< +"E<^e6Mm;N&*EFIi8fC/ao@%lcUb=).X1-*7)37=#SAb^2ZcO$`<]lG"Y=`rpg96s +G)(boW7j,%DUUH/g6Q$rETm90"(6=e3K/@ach[U\_u$(/m#]Qd(7bE_h[pFbs#^65 +\.-(T.6sXO$4*?r?S)0[s-*B&r&"HL208SF+8MPQs$('Vpq?#.,(NRApq@?uiT1#/ +rhgMspk5(XqO\s^r6GZ6\5/XBne&\5^OE;bH'MBj!]<4'cYuM[KEr,S&0M%C?5]TT +4:#%Z5hZJ!CB_c`p"(lUR/u:-WN1-e2eC-tWDr +>l`UbnGFdMGaf/VWs7XmTRp'lD$7'd^OH10)skuS +_P^XHd=@0@=SNF:L;e(N5@S;[c-<2`'sk=](-qj(m>Mded["OU#bVlFA52_*r,4!] +oq`A7CRDK$9kuL5FXEqVVjRK_.9.pm!7@SdP*C\7P#,Blq>94UCrUK._>j=cHI;a@rlP1VIs/CZQiA4sT8raRnt19c:/'P,`fE#DrepcR1"#LY +K_PN5@.+'^c=j79YVc7>HIl9bYUY5fnsX0IkQ5)e^RS1"aT(Ve^tATmSG]uJ$i.Q` +q7e!rs0EYqKe@nI"jJ(na0bl(+/Ao:aA>&%p5-ntr"%u]5NLm$`W+%iMMM-0OP0q( +MuZJ$=5*at?U3F_&0aA!(F1Pb'W=("22AX.e288X[R8ql)S/ZjJ*nppe!OX^aa]FW +Y8&VbT-*"gFs$[0i'+rd$iUDa)@69!@po$;AUe[fI;&XJpu#ZCn5JrbI#]4s9q6Z" +c(@,q20Jn&q>prbs"/Rdr1:7n4)b.kJqKt7qbP=/o)DM)nNS=t^qYd;n9`"N>Mf'h +mcn$M&q%r/HqBr[:7Vt`r@^P=\;q]d7Wf9?emlsiuH +fV^T4T1F5:KgGlEgpcj]r`,L$2Tpf9Jeh+j\Bh#]7R +^Aq!7)ej@VD6B^/]`42LHs2k0Bep'<3iPk#<:%biLb"nnj77.o7fEY/fOcglqYp^U +$[VMr6'`uL1)1O2s0iE%Np0"2chtO:QiGJ)+\uGD2GWr=7H"4AqWcT:B`)g-p\uuD +^H^Q;+9"CsJ$Wpb/s>N"1JuEJ$]4B/hl]!.,>.; +aT)8t!fgoM'6'F#7;DnUFD,*%KG9g!Jc'G\7jNT#s&Ap6r(lS+a7RcL"+U[P@3\$' +nGSA$3ts`:J43e`6-ZC0ErL!eoPKqGq;5cME=A92/])-X:;;\eo"Qlo!De8piZYi3 +(co/;m0c%0;+i"h7=5KjJ1GsW-\%LjO?^*JI#4JWq+:S#B.j4T]b8O"m1UcFrtM$B +s!bJ]r_n!&q6;'HLAur5TG2qV.0=I:jaW/<#Q31Vn395.?KQC,nk?_bK8731:*LnZH=9O:DHn5q4m_7/UsquL]#!_@t47qncuk?XkqlW +pNDHLZZ`2]\p9'Z8dNg'2RsN8%ntfAKq8#])[8YS4?r%4]5:KQDQ)P:F]H7q$>sqa ++.-\HbG9"oi=KWYQ1I!DgjqNK<:dM&^Z@is*#+1N78>m72T1SRXLW5MSTc. +EUp:[NMWGER*ZZ/\nk928SHRm%7l&'$:+U\Qi]ZbZ>`b]MFiFt#m`r;rl>u`6rsD# +!BL8n8j3&9=d5i-b$7Z)YA8F`pFQ!O[5$T7r_3(#pV/EF(PG#$n!Z@!E2R"S2k)#b7],48OJbe2?K`=?qDk*lU0YJ8,3:DJ) +i$j%%n/,Rfo5b<59'Ce5n!oaC6KJaF'.EiOrepdU/IUar8c+G.-fP*_Z$Kq3krm1IBMLJDK,[hcC+4Sg=gh`V[%+_H!6D)*4&qR+NiW#5$]EiHHTS4RT +0`T":Sj2@U0g#/k,EH_d@p=I-TUU>jJ$]Dr5ZCO`4b>^b2/03=9Z;[i&3#,E>ap^e +!$;4rFY]gh6rgLj*(2$W:dc31Ma8JmabYi4Ip[F8?kJ@"LZ2#,SH$R5S2?h(P5hjg +[G_("f8KuCrZADordt/qq4imu*qb`dh2C'9oPhBZh]EWMh:fki8MfGbiT*e,"X#o2B`9Q="7C0%qr@IN^asI0DNa2,r2DL#rZW$] +"B+/1IW?Aj]nr8@ruqD(mR:I]0RrX-]MG)Ha=uJ'm)bcm`<`uos4&6?KF#A[L]H]A +E#hBVLQS]m*FVl^rk))$aA?c(9'?Gg:bY06%C`%m0nQRKe8t3i4D)G0'EQ(E*T>d# +BqGu,KtmX2FSISNkgNs)&^PZ)pj'2P]_Xi0Y:^U#,J-[8]VcY25.E4ajUuqo%I(s4NB>>Qe$=.T#j.MlPqt%?DFW?Nrgp]3N._ +iDkJI6S[r,-UY?*g[.R9m>X_\\PQDNVVYksb:l0YT8K_G,#nTW9-rJ-dRt^R=it9A +1O8,4cMT2'rbQ^/o.5iW%;iD>qEPrU*^qASkBUV4sE*VQm#Of(Jq&ahTt% +=iN%'^S4*o5L-J=JA\;LSS=PjJ%3RX#Y6%g]?Cjbi&5dQ.=_[@V9nd;3cQ0>T5q[U +g&Dt2rYb]2HeCl-.Bm+gMQ;1p/YM4f2FcYh\,lJihriMtTASH7=G^)*M@(G,`ljG: +B&B0&TDR%\6pR7?#`F4VA.'j5MskhQH^t,F#ah7F05M1B!br$_2/XnaN;sOjEcHLc +VTVB8l%gZ$<[K#L33J'fLb'O/"9Y./+(PL@W;kIt@1gVRCBR%1"DgJh]`/Y!s.?Rj +IW"2geolPVopmm!S?]/+94Eq&,abp*!>%9SS27r3flGC3GijDJJ%b=)X*^TJ;P%+c +<*'a-4<@.ChR1s1j2'f0oRF/+Y8mZYRgZn"^XcWJ2*Y@D".n];R\-.#rmh;;jUH5" +p@RA3nbF+r!LCLis).ROVgZ:url5S#r81\\24(-=9/Uda@a(,0R=[Eq3\u\G6!=+- +-i#.,A'#8k!A^A59Q2oMT7Qe=,,""hVct8G#/Y;YqG#rn!9GgMg'&HuOgG4kd!d@H +-%Z!?6T`M'B_s8Q:\c2e5^^(g("g(hDdW:J'O^pU-BRA&q(mp.70L8[D/-VADP2r^ +8d9d/mu@QtrhZdDaq;EF_L,lB+oi7D>Z.[K^:/_+_S^a]?JC+B`nIri8gBWdFZLX0B;R%ld)>^!/pu&hhBb`8hZ!i +6HX>Ai7^n5ohGX7/:G!1].$G."TiETLf]6:*.o\jU1:?6T?TH0F7u$'D-F[7:([pP ++r\+,-(]rdrL5\p#p?GO>Kt)%b +k+.EVAg?`3Y\Q;`R\Dm_SE#"-Ut7KlEK.K!Y;cT6bPCGIrNX95&VG:es'L%=,Q>ai +7mJI+s3JXTF31./"n=poODn^'e?YCp99mdXD>MP6b6e"EHCrU[J#hD[Ciq\*SQ3Z/%;WM_!;YOH5O*_4s2in=?.Kl7M[.;] +s($'u%0A='59aK96i^Kn&-!;,,:NHb0!M0aBEbbpX$HNr'jP27GD"WY[!r\E"db@b +lcAD(m,"[\!;SF,!)r-=n+ei-^S^11mWG-b<,rb*19#gpk\%(WOr4Y +]q^gA;*!f0qOa)M=*d@#IqZT>[F2cq`=jZ?3VJcW/`(T3'0tZ_aM2YmGP1HTUL@oe +(7JfM]kFP//@i,2PFbQ_B(?-'s"_Fjm@'kO1"C@:n:*6163MBSMk(-.!+;WB=R$&k +)4:PS'M'&lLGgbOlS8O;s#Zq9-0Y`'`V5djbg$:2AV2H +HpPdX?`-gYWAs^GmV<;,_uaH=^fVd7RIX.+Zu6.Zr:BE-_5BbQl@4c!K`>+>41*!> +!:UJ#qn:>d_re^k^'6ODI4^8*X0<3DG`c59GJR!eDa8P;!pqd^U,rEKJ[_nT:ZB!Y +,duSYm,#a+DIV?%9bp.J&HV(2(#Cep;IHKg=_tSal7NGLJ^Ti`K7LrFlKsf+,\k\X +p?e0SA"hpbi2p4#3hLnqn"=7ikLAm+;W_rS.lSm@iHE!*Uq16',61O8,#a"VK^ac. +rOgkIMicBS@9ENd`V,_g'1]@pQOP$Nrh"4Sm5o`a[qkpq7fcfn%n2#;!_O\.qZ"f' +s'l>Mrn!_O+^74s5nJ8<05lVF/rOi;9.h5_J3aO8]]:l>eXe_W +[g[@LAn/PUSgn?mZcQ:JfS`/]H.;jFmX#`D5G94h;s/qs:]CBd=R8m9nKCH#T-cqq;"+_D@9)5#7M0LA\C.Q[H8\TeS0;Gbc9b`&[i"s.f@mFijY/J)Pmu^N6`9 +BD["G\6*VAQN9TAZaWJBT6!+HboR&j`b*uTBNQ2'7JOVp>,p^$nen2^c$1!gi74j4 +=grM<$a$_2Q8c1GQ:AsJ<;d>eK;NRd,<"um)$,hK05*KZXp`g*:a" +mU)pfoJuYFh!s0DoQSpd+$IYo%+tOlZN#Qc"#rK]&>7sOU-C4,kti>V`#To?^Z&Q" +2>8;n\e^#u5l7BW4klRUIT`qFjLB_e8+[>d@MH%G$!+^6_`/!aX]Z.rG4[?f#'Q"kM;nc.\ljPRX7m+iZSDi`TO +pUPT[".-Lds!dkqIXQkTkZo5prn[0_XoB@-2>2O5^0Icl<65+%@`\WMe&BA\H0Kem +s2jaUMLY33If^mD;b>kgb)6OmKE"6o4&6IHL\XYRWngjrh]F"k+LU)5YCg_'6d!0I +QMiN;F_m]KIiCH\*.aUCiH=3HAqA)*\.(6t_4/M0&`$sR`a4RiS[Yl[:Z?`\8UQuZ +H^c"*#iHD)e1B7dNc%^2';E8#rP5H,pc(Gp8:-0nNsOg[_fQr^3/Lfns4JU[C4Q$> +)GY2.hmcaeeM$"`2?6_e2JR47nCB0,^6WuCs1.K/s3#PV?U[N_N0OA:nH0tT`3B8Q +gP9XL3+)XTkIa1Y?eYa9r2'G^%fnM^J3j3jrI<#ipKOKoja@/M'Yq[PW8$u&)b'k> +MG7/9S]?rFAZ=fn/rTH@L*E;Se,sut;pd<)0N)%c +PM*lB1,9IQ^5*"Y8d:n6!`*:A-df`)8FR3@s%)3J,k@"@SE(bf5#,^^JOJX?,6'Kk +H]f**#H?-A/:X=Ts+L$ZHH=">s'5K2/f(;`4#\+OGJ2D7>XJY$c;uCtK`!nJf$4Hp +^X4o9,H,Hn2QXdqbDjqB2/b)/bj9iWJ60mj3edk3`E%,4rD3*9`8$A*[_pZsBF,U? +@iIE0`$f"r`S(j(dG\7Q'K!/6lf\$Da%5*s<:q/fm[:G'1jnlHF7<9*naF+WdC`5Z-!=h8qA/c9Cq&%!5@fbk6:E5*GjdB +W;>!rO;J@)]Ej4FZH'gD"n:RYaMRs0?.(a*31D/m46JlqN;Oc@9SVnDnA!jEaqi++er(erK:G/f1cS#h+(Q/5]"RNLBgTBmEX0p>kXuDiN +jSu4pr\'O2`KCh@JagFGL5Li@A8F:=SO2iWs0MW5^<)%P;Ym"15ks4:s-Ed&0uL_) +$a(oW&$EAClVd_qn-5T/s#dTba?Ouq"5CkVhSbK4-;O?Xi11NIUqe.kklBR0R02qM +lqhtamk^R_%R*V"7KhWUiI@#:X^Ni4X'sqB*$W[ZKr[5 +r)s$Is4m4.o)BkQpd=Thr0TMG5G_.enNZ)am3b7A+Tr%2+QQM==0_]WUT_Fgm!Tq( +_#MPFYj-^Wj0N'=(HDPOr'+<`@jCeN:ZpF/Ga@aXJ(?e^oB7\Gh1P\K7"G%DhnY%& +JhC].bM$PaVc0STM<*'o?kajr8,=eNh2kOc=)rCcD)O;;$BHQd,O*R5<9STWoIM&0 +'FT@$%C,gQU%hD/*[ai-Jn4S3!mE=Y&+bEqjeHYZ7BZVFhg!->k?GX<%I`;>YP-1- +ra0(5\#rcar">$?G(6?:^*Mc8rhnbjX)DB6crq._?YWSARcW(GEOh=CLnAI&5Xa:- +X'2G`<*?.99[/s$*Z=Z9hqUMEp&ci1-pWUh'`Y35QY:qHFG]sG&_T,SD6sDG24p`3 +eVPHba._N(?,s_OSG_4sL&f^,Z9RLqJ@OUl:j#o?a[>jaJ!Ri-!jkMu5H'"4rsLaQ +08Xg;@L0b$-:@hYjY&'nNHquVkL+"6]!QNBFZ,S.bf0gkZe;9e-=WSUgEqN\;#pXn +Ca7fc=74P(Y*0ZHdL6!AGQD&^,_*<#Q14gPcc?#,KE"*>b`hSnc^HRD,=o7\N(]6Z"^Hr'.gJn1poe-g6 +s5(_$^VAOhm;tBnmImFBs)l<\PsnRMPQ0miis]4Lao7>k-ADIIg&I.`_E.p!hHBQ% +g8iDDp!Wm%!)9cN6`Zk#h?9C25ZI`EIlF_EoCk],YEf-)r0[DcC*.hdY^s:Rdl6ea +OTJ#cHjMu.Sq;0Cdnf!OT>lp;muiICr!.HYmu$Ks9cnn0R6u_D">U8438!9MJ9DVL +B*84l,9ZFh'.>e7[[hLQ?.V[?>YFF2IU78"/ucalXQ4s1U&DI7%If4(P7[@M;:$hPD[L^PGEVO&?pnnOgT92:/qu.]&#g%@#l$jQOs8W&D +&]"AN=.Y,>ia;An*p0YrbQ4R2(WCmKBnZ)cr.b53!i5baD;mVdpmbA.;?T[k:F8(' +@/fr;/'tI=3Ybr&TIZl%$H`*KHo;Y/c@j3DOjS";74.RV)0qa.5)MioJGH2-+9*IK +HCRYBWSCF&i]dV<$i.^<+B/6+j\OC`l[PN(X`i@s(A<7T@_$incJ%aED"_eU!I]7( +hS!%W?ESB'Q[n^X]tN'7/N"J?_Z+BO6\%dKZ'[=VBTJeUElA1]2f$uFFg57XBI=Rl +s'$(N41pM-?M^9jNpsH""%rXnmudX:ao;r'r%Ed)<<$%6[#8H7/bFXh]*UiF +\Lr]i%OqDO/8D/o/IX3cFje:g*<'h8k33"/`uNo6`r^s;s:Y]U0V+3kT"s#dq=;h4]'4lACd(E7^H +H,#%1+'3eQ6N01rNsO_ETPl:-jdq&$eFS\QY6_s>"!%S"* +b[48lcY35@nt?#.gXgrIs7u$lTEr&4-a$Ks=3L)_qg>*j2er7rJ0F^rs"CVVBn(^/>="#Y0b2pAbiWQ4 +)ZT\p?NTpc+mOPrB)c!+s!t:"7.pR.r#Z3'q*X10r_M^9RCjtQ0P=%EHtBQb>:MR4 +Wi\/PJ#g&*hu=Ys9#Dar^Th-Ng[=n;o_e9UIrYBMs4?S;[G$N1J#R6ns5u37]CeDa +57s8ns2R8jdPF6&i0agt]NuahWIu%fU#fGpnm?0%2PpFFrfQU>kS>t9Ir(`-J&V>I +H.qd@2;lrZSK]2%.cHhgUB/.+/kp_ZEIT\O;'f*5QhAmn+'_62?J+;+RbQB5R1-T0bbFC^/&Q7;n;U=hY=Qt"5C<$ +6';1MBEuD((Yl_,f?F!eO*@&@NU2*C?=^%]k;dpm_DT(2o)C=^+No6-s%fV(V>aO^ +"DAVOglDI&\DYZD:j!nrrZ=OY3Y=(,Il>''rt:h$_IWp=`BSQqlDK9sIhaUHK+J3W +d00Jb,67`++b6S:Sb,-:^l\T>DBnrJl6Wd +R$]EO`cI!!>k1lAq/JIc;*+4uSTo`/8>VnunXn)ialK(k$]d*_&$r+Qkq\AM5Kd%k +b1BscC9PeQ;,Cp;`TC:%2$qFm)59PI4B(jLA$#A>J9'?/q/Z@A]mHT?rS[Ts$hSAg +HT*Jr"+*,.lEm(:4!D.QUi%FB#";tK0-\dA3F +N:GM'-j"ft<7H3*5'ld^s59%&oq/ofIgS48&jH2.ip7aikOZRD=P5DrZY"A2ECRFO +QHg72fsqB53e3Yci\j&PN0'8D(.K;MLq@-B(*<;n,mY&(,#Ah'$a_*K"nF=Yb]8UZ ++"JL3O0)G8/<1]GKTF+TLL^=9;;^=_cf%OT3r0[i$ClL^.=cu"cS=]L)Z^!&W8B`R +n;N'K'9U(0/2WQHd)j_(,V!YZ2QP0c%/IQH=Bm8>(+YXrQrth`Ze=2QGBZ-^]l&>7 +GCG'grpopHlK4>A^3),GIYk0+GmJB:hZs,J?E^]iD=;RMg[4-(X0L?VXk0Uh[E<6\ +Zs549PmZ)/DBYX[ACH9,8tSlDki.69>Wb+f%o11DdkT`Z_D;:WmQ"sQ036/=$jEi2 +lsHQsIqZA)=`$XZ440$gGR[G^+,Jp+lTDIs&Ub*@@u&Zm@rN/1q8;Ji)4tmKp,Vn) +D-'R"I*(P#4&DIYcG%L^!F@rc3PNWc+:&MZP-8o70CgJ$OL'A\8V+-sg)AsbK""<_ ++E/4T'[,>9id>J5`/->XSNGn6\.3kZ54da"1/73:Ur:NA(C[0 +aOn7'gV-=O$BBn=!7V8kC^9!qiTOTud)L3Ir@5$pl!IGnL(oRG6-j*B*LFK_2W#4u +cBRr#`Q=`opeT_JCO#JGY5pJ-7Srgt+TJI..!p(_cTIfs;KU5">k:Y:4c"W_Eu&F" +OlIa_RF<&$ru$_:&acZrR5R4ls*SjOr@`C*s(@KjAap<,B+%Kj7IX`gr$52aZIPl$ +s/"ETr%I>bNW>pT-^qBis0T7n,/D>5+T7A,s5M;kj]ak;^egI@kAQOlJS=%njqKA*%"]Z&/'#f3Yl'^*;g^a) +oY5ce;r'110s-q[@jYEk+-'[L:s5mq*^"qu? +_#Kt^C\BuWNX-1F@"2#Mg>\b%58X9NoA1Mf6iMVt?S/Ioci9u3/fC3U_Z,0)o"TB< +pjZ'/^KV6EB>)=E6iM!of"833IWPf;!dM.TaF>;c^Ib"N>kW4V08oWLs1c/NC]DQd +["%dLf(i=sV<.$ufcBX,ad7_1#JfMJ4/IHE,H+NCqZ%5t-G]e8mAn6`#hhHX%`s$:6S2=i>l+eF(`?k +otQ:_`W'-)s+^PB"PC,\r]bkAplFcUrVpSR8%Bik]W(aWH-$#6[upXHG3W64pqL:1 +iA2ROr2'F_IM5O%^dh@OHBu;m!d"tMh_e\[/,fe?Ir595NlTiQc*X=0UADVCf2WGk +Y+.f]9GLG=n:*VZP"j=QTHKeS#-uopl0QkmmTI!f/6@4f/V1t\4 ++8EUVd429LE`Vm`X__GGK1me45q0a+.]roMPXjY=9ZZFu="b>WA*EH6&J#_`+_pH!ZWp9#tGS+R6Ts(ZPa +;HPF1((2!Q5&JTB9pq5#/M=k27Wli">Tu.snBWDoFI=K"T60BAJ+c*Ws%C'0?M7M4 +YI'#`5rZC"rWfg[LWk5J;#/$,[ja)IkkPXWYN;=td-"U(J":Y/PBVM?pO&NQ1O/r. +i#8J;`dA4fFtI&9D-n"t%'i.,M*qnO$AatA$FfP26JZ"Afj[4ZNA,_'-dgPi]@buP +fH%l9.Tr6k4qUHO("AH#jS>!VfGe1!o'#io>gUAd(q!bs/O/Z#MD&HUA" +oOK+&Q;(c2Ja`TAONl]Klm@UA/C$@gI$n +]mP:>I!N8GENsh=^=MUfFupZ4J$ZkD+*7.*8&/%'eZ8pAJ8322.@\Kq;nZQg-[!D4 +Xqi_"K1i0@q5c1AIk.0sjJ6&>nPK7XrD2`DUROOF^DU.VGo,um5O/kJJGm6mpO;9h +r,4:mphS+I6/)#pD]^XbB5&lU.XsAV:]rI`bpD;rq=inL+ALq/i8L4.EZM,A&/5`74*W4E*kalCpeP3":.OfL(WTY!2&:rH +]ou=nA3;iBPl=GqZ712SD(*:$>lU.V;3Rb78!T*<*QEiCn+cq\"Pet:LB\^t'#*P7 +Igds@qraO%^".)1\^P8/\*NP`B!]'G_g&Dmb/HUC?d(-qpkSn@#u`Hobf3>EW%erD +UdOXd+VQQ*]`qL9#*SLe_ukCY4SggUaF'Nu^aChKLBB`(;gAu(\.Yq&[=*Mk$;W7j +rr`)sXB,3cn.?4!lTA/Og)l\HI4Ve0h>dA_gAH44hNn-?#Q:i_J:[_^+p+3NJ_/8[ +Vu8^V&*TF])?G)^"n_:deSgP_+:\IA+;!\%<'1j%<<&T8dN.G0WO-LdGr!rE=R&J9 +`rF%^7M#NcZa6L`F=B.GaH,HSi];SkOjAM&jSp4j=8i0XIK/mM!<1:S,%-"qo,m2o +T2KaEgsT+(LkFMuSOGa'Jc2dCr8m(eV;UW^p$u_Vnp9uoC0a6C3r?\Jgh8RrL&X)8 +jn4Y4s',;aHA89s.QL/4?nRd>TN3n7de\?I:@W)GETjI1]\g3To<3=b]FeOCqS2E' +D5j#AL?S$A8mZI#"nUs(EYXo8=:gfuG;^Z&!$: +lCEn0VV(q5k^4_SbGkJ2m0s@h`;jI0c.^e=-b.]EDNk)^H`5n#a^=ZNY562jnGi=+ +%K+PBgBU"Ci`GYeQbB/PmNLZKDstOB>5:*,l]?V0YYrXt=*%$"XL4R$/>atPUcG'O +jPo<,e-,S[_K6\GiA\UQZn:!C8>#,Ir1@sIF&#Z]ALb78'efBt&R5?m!6AMMp&k-2 +G.IB\s50\K\bX/1*NfA17f`JeA_mUF9C=j5JEhV4q\d'H=RlRi$(Lt8MXRIC+&DBS +eHdeYor0Ej3r);$ +%]S:m]I)`i$1k4g^[R\gs*Ges7moVc6fnVj^Sqr%5=dSuA8l*I@bIA3!3!0%[Hdp,/X<6lIt=^<=4=3f-V +o97/1Gd05s0>7SYJ"cX7^@7Zk?RS/#iZI;arVpsf!]'FM=\=7M>]UmhrrN%)C?d2K +=sisX3WrRH.:=k5fMqcuZ\lmIrkB3\E-8L1ID'WC[0cl*GIRF>c!GatK*\-4^=)pV +rGO9c/`j@=S418./euX]Js#sQSh9a#.S*e$&gF"7>Nr/G5Jfmu;n>CPi*'4r@c27jJ#N=*Oq17t[]a+:S:B9uPg#E=_k0&XZ,oB2apQp4#8? +/>Pt9N2;\AXQHXm_GZ$a07pgE$"Lm-YW!O3PG=M#e`$3f$>ALMh7!Fa5r_XTRE"'R6Kpn5lSY#k*"K9 +B58Uk5RW)=N#r+)dO?A*>Q<`S/,p^1DI5!aqqiFbr.E(# ++8FoP09SqO$hBY`+1_rF(B4M3Wpp$$jF74+q\IM+B#8'A]Ce;tdIe+mT:lpK +r!fd$&[XnPcjH82+X9gt$l\q75gBo)5pKoN6j-n&?slA6L^&uK7"B1k065WoQON!2 +ZPhq:qJ;msZ3hA:BjbXeWC("n,iMVchr.m106PT0eRQnW)Od +@k@)ARpYA!7FhDQq]_umQU/m$EnC=ll2Q>E7Cu^qs!#>m+Pn\4(>p>9,[\#`R+eMW +o&heJX$ZdcrrUXKOM?>EY?Y'Lhk,e>ScmHS$-6Ib<.Z7nNs3BjUS3mVnEG292ASHF +LqBPBs+H`ac[l,FiV@:hT8sou#p]F/&B`Pi@P)5HP@8&3fm)'P`pVBdqi./)eimUS ++SJ)(#iQ;6eN9&e-^q79NP04HVc2Dh=12;obsl]dO^A4`ghJ^Q>Y-T6;cZ^e/qS+k +TboX'+r7m%!V4P=^KEQ7_e"Ms(&tn)./p(Clc=D0;?PI$IuXD&SUXB(ci09[^0:QO +;k*f+^'+$,WpkQ:AVXp>r+u8&.g,"Y>"D#2q6;!&.DS5'1`VU@!^Km/6u,3Xi6/4- +AO$==Zi&MP?Y=4CI$5.Q:ZUK&d]l/.3'Q:]=^%5]4losQZqUYB]2>?pd#d':o5I_df:T-D'h!UtTr +CLHu4Hj0THGQ#O!d"D\Ob(%M4;Z=q&Tb/XOisD0:JcA_((=5oo-qI>81S>YZ/?_sqfDgYWp5S4T)HH#l]#UjcoLRLFM +;u%!q=*_gbB@=2S+p(5jg3C%i_3[A%)frt&%0=Dd0'6nqJKk0on!pTU/iQ)rC?,5< +PPpOc!6ZJ>hJV0+mTa,d>[D6Qs8/@(p-8;$g8.O=^R\_MA="c;$_0T<8K/6BKrhGE +f&pN6FJ>scQA,D=7="Zt](q]j:PZCar5 +E5-T;r0Q:n[KAMg-Y@jsZG"K7`X6lM'N>kJ'e]K>r9J!r'Ef?SCY/JtlMj9b%8$a/ +#)sbq@Y4<)9S&'\[ZN#1kU#SFAI*67RlZo;ao,q8GWj%F(It;>YU^3=-=9ML^q@-6&E"8)Sl,3FMt;]IOlpJ#T#HDSlFM25X=* +J@ZnCBH.V'ffjr=JW^)J;i-b=%Gn]e=.M^9%/d9Jl[E][>M03/"'W['Aml^qs39nF +,I:9D03=@g8qP8_`%7a`R;TnV(Eu-D7DH+cU\"!_*p%bX5,k#55C@EWc"dLDJ)=uk +YR^\Kjfe?lZBIqWJ2Re2s.OiFo)H9\kjq=df"Nh#?@Jl."X,#5s%>MV+UJ+k#iMQk +;>'i`ED+"r3Pi2DZYp0\?:r@e$GOg?^WP6:OPt8(4JhuGjgTL?T&@5NKD@E@s#sqj +;q*X:l27>0_?\kCd=.gc!3P!?q'AB]>l=Vl49dnNO9P[R5%!8uH.na.k$.a[C]:#[ +TXF?WT51>VV"r^>>jt,Xi02e'jKI@O(><0GRXYLpQh%lu0#IO7Y6g".J5M.'T:t^4 +p&ZQiruqCuSZl8i^S)B#&lB;Eq31hI&9%tI3$W1S\S>Lfi'mLWm%#YOBArl\HRe?M +jSGMIho[K,ZOh:gT>P;kO#\9#D9pZ(dB%2j@%UGVH-uO=Bd*Or"NVdZ +,@KGZolH,jmmI0%$aBXW;Z:6IKR;Z[a8Y6@r;c_%)ZfFmM!n'<:^"`2-]*KGa<#r&[tKS*#%kK8W3_6*]cEO/l)f>$;gp)/3l4(Y,2sg3^Z/BQj,Uj +27OQSE:A%S\c[_[fCpD/V5iu(>]buZ]-lN?jf3AQTqkN`O5GtI8ADgn.rBEeS+pb! +=ul/''E1cZeWSKu@^.-W*pn.9XmL0j1\#Ejds1o%l?K\;M5KkeBYF3^:ctXX2q>[(p#c!@ZB`F]aLc,EI +"T26lbGbM1J4tO_b:\g*mtu"nPk:HX'57+VI&$7TecE]NaWIbYV>n@B=[Ka"p6R?# +c^[?T\`Yad(hJ1tk(IqsEN=ka5I1s`"Y6b.as3^k7l;NeHIDcPOIo66(#aicc#Q>MO@$2,KBWj6SX0(e. +;F$0E4"ss2RnuIcN<[@:+P(MD6>7+)(4"A]Vj_ +l+g!84.Z,jQMgTJcd_Sk3!E:W/,Vp;b:-rB^[>6Xr"/r=pB$'L3QFPYoS'dnJBij< +"9d'M&il_^TbI-\1QF2;K3Z;i;n@T1?o8"=s3"#YLoPd^"efWJh=n[[FDB7FTC-FA +qS*1)+6s7`8d$-[nHIgA(b!]U>rY:#P2;[hOCLD-qFEI*N@@YfR0a[F!0Ka:6% +-P@7!\S![(\i_JbdM9Xq;Ubt%EH=!GWM^LN-p40almhQE5hD)Ue9tZV!>j9l-s_c8 +%PQR,N%05FhGUk"*`?+ipF^6.pl5%rYlB'hbta:4A_*@8[74o=O^+k5Q>?#eCXrn7=5 +s-rZ&IIjfC5?AdfCjmg_btP:gl;q3d"6U1Er+\dD5:hr6L]8Z8"G?dPs5=Y?HaScf +n1TED-#F3DD[#%W6hR"br&ubo@t)WNr1&!1S9I)O9E]28d"ifmnm;@:%Ycn>INMYh +cbf]?Fs']`mrI<:Z=mSr8X6]sZ3F>(?THtM?,$NPi=Gajq-!]EeL"Ue9jT<9_%go$ +=I'DgF2il5DB.o*!*AF%oE:a+b7+k26-YR+[DkXKa=Uk[Jp4$62tS4 +MZ5K_kCSaneJG4M/!-onUoRuJ;k)"\V@3(M6@H"5dlloE4R9-2HCLc6i;\_i%AiiL +]?To]]m5$Iq>0qss/*PXeV4#^]<+fls&Nj9?!%j@?\/7#0bZ%_Hpga^kbJg+s/F@] +dPmH]9c*_K-3&?UBPIZ$jT"/$P^b3.a8[+[S-&HHhsst8mL7\k^JnIXU15hAR#%:s +pE0CZFnc2&(Y&G^d+RcGjh6F__ +<3A9QYKbqq^Ql"fX4.s/[08Z)!>km.hlL2BDVi%Z)Yj>Is5rl;qZ!9arpM^&aT(Ap +AoSRbVnu?]s/7i6&OS1PbD,/c0`r$<_>gLMABHinnP/.o1R4US&^Uq=+9IdV4tZ.= +pn'ce%D"n/q1G!uIisc4UKN,)dp-qX;!3U0*kMs^j?ER6a7_S5!kA5lE'?uX?Y!Cd +FD9d'8\Opm)$TL@5fG5)Lr/QT'/SPP-u1"lqS.'^!:::V2%5nF=+oT)Ta3&3reGRj +_c`'_R*kc&,61Ne'UmT>bCkm&V>gS`+L=k]s*/-Ts,/s;o];aN'K-EuTTGjTG*MGVe6!s?X*?$ +lt;(60E'5_I&-R@rp5+`J"W4gAbi+Z%gdjAQ4BV9^3o`X,79YGn&\oU!Rh#B)2nhh +m\icP2;8HpE!e:&s8S=-%M3:;0^nVJI(pp+L]"?Wr)EVZ0`PGY_tcP=d3.BM./:_= +ARfb.+2ZXHc6r"Slqb?EQe9bfrX\jtDMAdZB(J`ckPdmmX*pWtH,NZ8OQLJ-kJ;fM +f@Ku%q+ZHLo691@+QQ>_+ML/u>OuZT#'[!:FA"h^c3PXF`]%MW6M4=io(^SW^$a^]ACcZJ2aER9#pX('f-i`:L +r#t/!CXi.\7=a]6YnoZ1h5cP;GRHdY^*N['&HMj;p)iokbF]Q%SleV*7L=EJb$Wg- +^^EFc?(g3II,RVY5PmZh0M9nKE?Q@t(Q[ +cZ<=hXQ'7tEikO-#OV:i/%,fb6iMIOD^2b#p^VMh?bp6;-aN;)JGHU<5I^U[GCEZn +D_n;D7@ljC9''E!K5AmqTTu5cR/TDfDJRV?h+0T&6emZ*I06_ +k?#qL$L^UJl#WK9?d%9Jn)CP0I$gOscY1QVRm1_]*eQM]i)BJ+reSpZD&> +c,C:js*S&:s%[E`r@dGc`W,T6[=IfknpeN[9*KJ&n3$Uj%ueL*UC#g.6pTTfi:g+HT_KrSR)r&?a!,$P>YEq\Pgp.1?7D:V8')LB!M^JFe8( +,5CLrei'#(cP>gO2$33i!bd>hEi#`$s4hDeOT.tok2MN&G[sVp5eX(".tXt,::Q2R3CCN*S5"*h@)mGd6-\9hB2\]?<# +n#&*jWV'S4'6KEW^%cn$]%W1G[)1"gn="BEQaj<6iEp2c5,."A@e`%sd.OTkOl4[c +RaW=bD',/FPH=0#S:.W_1[ib\++,[$s'93?UgHgaP5Kg4KiRQJI:SQ)2uN]i(=7L0 +s0E<#W1s@[nkg]`PXN=%=?5kL`A^K4$XeF4Jc@_3AEW-(rr8r_BZC02*)PsNML`oY +(-kXl$NFnK+$3k4qr"OET%<^EL3E@m;@+F=#QOCY*a7*hi@R0q6+KcojmHlUFfk]. +Lj6o=W;4[YT8EM"jZhi+R[9uMPR&,sIlCj)LP[2SF3WT:(`A?a]DbkTkEGhudCFF" +f88`r@D/0FLArcQ$iq^P\J6arna'hhSXbs?"O.kDs)u6*.dAtpK`@rb`C]hHh"O$( +Q9Rp+4qp*k49P-%`sbC%Gm++BQH'W7EV=mj4=_U@NsH;a+TFiU/f^@>`2=74R=*[a +D_Le0fAM,s!o.U!?gV*(mVq\loXD!7gVWt=rorp]2Z\d%& +/2.k4]o7;F-G?T\T$`.6VB2OSiS%`*r:#0l[T_EE<+NmRl[N(n.fA02OT1efCX]uP +nuSKn/A$mEf4;bsEPVQ4p:pXoO.inpdooSgU\)Yb,(M;%5GLjjjXbpAs$?1i>TG(/ +6&_Q:;etVcP6**O!Ta:OM5b`o>hI(<:b]-R!P`=HO*@ER,c1d13TB)0grV[R +Uqd:R)5FM_;"feQaOk+2G1pQ"GKp5//E%T98bD=!60!Cn2e(itWr.]2!<78K)hSD] +)35I[m_J3+`Rj%Jq"g<;:U'a?I"3#HTAS?%7!;5u+";`,Xn1VNoKOM%lYQe?s"j&X +Wk&4A9e#jc"O$g-Zum8>\Vfg,MqBYOT:]<_^WS^S@"mcLo7Lt7r.2<&cXjl`^T@?I +WTSd%5=`M-qd2^(#Vc[r,CbBo?qi)R0>](R,?0&8.[62(^^D=`05U+,_Z#nRlq79g +h7Y"F!N?"DhO+Z#eGja:c`m4RHS?l@6T\WXG]5[qGF9q3O7.@ik&NWka:%m#9$J?011*WBT>ED4&7XhM6 +[Xd<9@X/(j8hWHDoFW)E=*Tl\4@0DE70)#hdrahgWq_P3)hF3lO-@PLahgTK3"@^_ +8W\T&6"qC'036lduEr5Q> +kOs<'VLNR6h7r/jF8s#4gk>])s0jInc/NK5YgUUoeAPf\<1&rLV&1s'Hoek/gndZS +kL,sTP]o1VR>?)WUV78hB+$L +PU3*i*D'PG[I%69XAZT"[jG5rDIX+c4IaJQ7\V8+]SD".SGq4)r/8o"8!iQ98_`:4 +j.i,S<3-6\MS0!_,^H29dnIB5SE>ti-s.sk0(Hg(d])J+P@"F!&,@#;N1b-+i;.r% +$\W_1%`H]snZffUU]VG;n8LXQ$TK06n:u&*+F_LMl"$r-Rg3N:Nga-aoC=BZU0C^m +)QJ6b/RV$NJ83/)^CpW@c1l<0Z-lCM1>VT6;RC(5^nZjDb]gm'IYYH_mTgY(b<9fM +G^Z&/h?\">He6qVE0]*b/lGp3'$oY$H#1/1=]G_;IY8I*6`i]#pF=N5+hi)ELV)V. +[XS\V-P##\N]p#UJ!rFo/7SdpJ*QX"k6nT/:CuG#dIZmYmj_Z#QMX$$8F(?Tr+bkj +>3qSZK))Yes/@(8W=f-/iW#nArMAKKB@Hb(VXN,+i7CGSjPLBpMi)9?YdWKAC0s&# +cpib;U%k9l77tK,\kb1pQbadM.`$Vo*A86GPl/`Y7H3K^n#*g"i;Pb"r[I0MoDW"! +./4f4ZL`Rgn,X#>-PrBe,U[a]Cjs1:(TQ*D,:V=D,cL2Z[)q'FGpNt9p<^G0.^FL: +=hfr6J2a`-$,4%a^N:?"j876n,XP.s+11b*0H'RK@qtmlTc"uRM`!W.7W?'_S>c[h +6.J3=n<_Y!([_(-:cHO>V@h8Xp`.*@;Z$EWRH<=dl*^$1!.TJ36NI*k8UVt["6(8a +F[7n_RfmA+rr(&E5;VfD[KRrTcRkgaq4A6Jn0CF.q%oVdjSt'YJ]JP'Z_a(+E@t>@ +,6BWG>s("1NR"$t`pLK;:O"@RS587+`QCN\[G:#Lp3Z,grJf#"(2Ej7hTg7^n#H7' +nlP.,V>oGPd/7f4:]@P5r/]",kh#"27ef44RO('Pie"(b\9BLl*D-#qZbaaMA-W$O +,kT?WTq+"u_\UT)/O3YXT+P&9^g-Vl3hQ,+=m<4GY6*#\.`Y'+^gWr$1"AD_P'\P= +UpWJE_uF]urRkcZPKWKCQfGgo[L\P0V]"6iS:-:'K;iWJM[h$7-j@PFTP1[Ks)MQ2 +Dtoqa2$$+\IK+BYcg-/_qd3JQFKBsgp_YGZm7)i50Yj#&H:ILCo6b70qnMJSPb"8D +LfFE+!X&H]'`VCrIk[7[9rrH#CQLcPh=N$Oo4+5:C$GQh#psJ"c:YoUke9&.oKNV(S2p%KA'j1;?lh^Xf.jq1%&mbcf8> +ONtU*CN(@:8dqgT)J"MY2?N=DT-N9\]]U)R>Tc'ob'Ek8s.@^fpbg?Z,!= +B.r^hS,BSNUQn'E?LeEY5,n$0YT`ru:ME`SEs:0hNU-'PHXfG'om&62FrU?NL/hM#_*&+E))X(&unW;qqG#TDqjc&"$ndZ/@'p-p//jPUO18 +jBj2@A')T4.Sk!d:^Rp"155NlAi^nNGD.F2rXjp(`"S`$-?\g-Tkp^X(L-9lDF[)[ +GN5F,.C?j:g"FjN^L,m^eS+rk?!(t+Q?ff44^kJ4(91e!/NsIX$9hD>X\YM4,b/6l +m[AsD`eFH5P2=TnIlngC.RdrHr58Q:'GAsug+dXnA<`RbX+%X9U$ajs-)1j@j&]mu +\8T/W+c;#_cdJK9ckkJ9$2_o]abTB#oB>p]-bi"O6r*at4oIn!p+0u$V_4&J!q;4X +\fHM8l8P]_7!#!@b\k16I_"U)O>3HO?UfRor&N>$m;%j$=$/.+?ZVV"BBH)/gp!7d +FJ]11%0hEeX#$d5XB\rR_#G.9Z/SY6WiLC6?d)LDPQ*p3W;+-$nNX5$CrM3RI&3$/ +V7G=@3d'e#pEi9+*!#S/$4+ah\^>Aeft$8A5J;Ikr"&U1-HQ3'Yq8#Q0>,U/>IB0s +b5OU9Y4i$6KQ-p9?rM(XhS+s8YkFph=oP%h:Hm$n^<>4obLo,+3u#d9ShJ8n2S\8O +9&TS]!8?%ns/120k=1sFrbF;:;h/1B]1-I`@/aO!EC.fT1\j,nIl>'ms$JB5pU-1F +-i\)(L;+T."G&H4^ctHmp:geB,(TVFB*@;3Rr&'G;`9n&28=;^0;#rXmG1f-pO"g[ +h\(,B$etR0[Cl#CF>J8O6QFL^EMo#(!rm&p'Y*W3MsrJXAH%n8,V\`T/TmZqrYP-T +YQ$Tan2\ibJFpUPPq[1K7l'<.h"1PRNd/YFYrVSd%KXj0`0N!JoPumsbl/]kJ9&"( +W1KVN"FsBaUf`!jrD->lGQ"/TIjfQY`,a).GlH`JmU\L>'$lO=p:juB&Wk;a`7#Bn +(s;htB4/Xm*"PH7@jsS=T._cqX4AVP!:ARHngA(-& +13?D@r5V;7f;&1G6s&LAQc-=JAZ'^mdLHW$bH9-S@'L`Jkl@upb_!&IH"PN]S*.GN%ID)-%]duIgRokTS6:sK>0=_.==q4Cmim!f33 +KK")R:^Un%runc'#I$@u5N'ch2R*3kM4Q%n +e:U3eckO96qk'uC%55OFpJHB2n?9GQ^_$SHEs,%lIuXLI#cQ4gl33*gH2BfO:F03- +9B=&B>5<6??co!ZR=FNIo=sguJQpI#o&_Q3)E\)_H&94oU +Y!U=eo3kJdTtkGhbo#Rnn@af&IEai,r3*.+W:L+)+-uEoOT3.HoKW9&:)X.<,iu5# +"lTB@p4YRmCju1Hrf>>ss7_^m5a)8ml9ZUd>P3JEnNTGBjNgBs@gG=u^oIFp\OV7l +9JjpTNTE?]AR>a5J/e`k5qnSFN^o89!mu!pGgmNN&&kGVl1n<0O=OKaQ,Q4&V:D@I +kdFN=mBqCU'/feYHQ2)P(Ti[TqSmet]iSNqs4]R)Il"KkikHP2ikMMi5OUK#d9^PM +/8nIHgVuOr0c"tn-!]IZcT!,A&>E&q?BSF*-8],=m9X`#QL5pJ[oIjiAA'9<`M +R*\SZF^eAqZ3]1C"f+C>\BkMK.R&:;1Quj>L29Z@LOXFfrP#7 +2NkDmLo\-'ZkL0dr^U8Qo:Ps,X8U"cnanq@>5]lEA^+$,rjsBV%0eH[\^ZE+>>>i^ +q6b/K:_;9<'#(rgIR`n3]1bKb?_QU:f"EEE_iBIHl>=^diGs0?dL%Wd6V@D4ZpZ:M +cEuVI^%cnO4qf?)jL`"$i/hnE"e:T4/7[.7l0o*]?+\I%`[ +7/'[V+2IfuV^Y(6jF;r=%S*s"UUUHd>(ZKu6"s3aRI +q()IEf.fU[l"8Eu#s0Wj- +#a"s^oPt(j'VKTB]JWA&%H1^pJ!Z>h565dPQYc69/%9',?dA.5q$Ze4]^`/2q8&&; +ji4n>V#OTs7-J(Zo/84e^Dm=16rp/^g&M#DPrh4$J=4]"p)0_#q$$=jnB\H@H.E^b +rsf!dNEUDY[qCITj$o=CBID&tPHo;Z$[Cn_']^kHA`d(#3g@&09+?IQ6NH\,A>UE> +VZ)'SPp;ro>[d[,dS"%6UFn+qNe8`[pT3TJ]bT#K!HBDK#B8dZ!S4u9Ul/:^4qe/CA( +.L?&5gh)rU235<9[EoIM1kHVHMhje0+]_!#i8=3p6j]_oo\YqT^WPMB9-L#LPfO,<3!*4]F?5rSRRN6e#Q9/Pem6O" +mm-k+s/fp.+O':\\TTQ?rI=2VpjXE9+acjGY^A)*<[>Q^f#uu\5If&J=Ff!&<:bJ/ +Fj]nkq,_ucm!j9j&,Ui9!Wpj+N?@_@S*L$)s#C/5s*/Q(M#(&mh_5mUQ?M!3bSV74>s0uFccuems,Rp&,OGG>EA.QX.6`mn-gbOf +0p1gt8-T[q1n&h6c+J`*UR>0`WX$Z$BVmbOY=ob41V*&jhNQW\6jDeG8S7+k=^sX6 +B)gJ`.u8^VF1'h+#2RikF4jZNs">=\OFP$VVY,E`.,0Y"Ql,ncsHL+`fQ6 +G0HdYBBZMQKCP0c[34S(1U1Koj8tO1kSQfT4#>X8cF.b7mJM>nVsi[UTNa,DMO'1[ +_I[En:[qo-.,o[FZESG9>JA=a>bm\3'.`j=N"eM['Rd;*f3%,+M>-Lh>d+1oZ2[$p +qu6s^rYW$@/TU\I@2m0=KGmYFf;U&Ds3W53dDmoa.YiiHD2H.=Cs5S&F`3I-2ib +/T*T+@fJ7R^ZY[T\`-keaT"tV%t$uN@NE*+q=?`uVnC^g$k@GZjA47=OT3oi2uN\] +,P57dr&ni1)&N:5r7m*[?W9$a"LJjBCrQDXR`U%'QiDQ! +p4N2(bq4"&Q-! +r'+QgpPOa^U*5jWlNI6sJXWG;Xh[>t#oF%.N!8P+:EBuqL4cdJ8=C!EZ]$TU+qi99 +!e^Pq&1mg:]*K6*!Bp_f#JqQ*m5#i/(jQ#+c$nT*#PJfAqZ-Ga!K$nB>7q4.KVE$# +s$TnN*0S;t@X*o$`SlW/%Ds^k?]P'1>nkqQ8N109NC2sMiCq]k[Wutn"^\*]3q0Rr +gN^;P1tb!O3rb\),>CO!6RGp%p=6"P[4)'!f-MDEIKc/*(bs,WH?iU?".2A/R:[3Zc=Z_hM#GtFBK\9nR +>bZ3rhD?$cSn1b)QjJ>LRRTD^)2B2t!'u$qC[`R\l8N>B5D&q&L7^Hu^N/RfldJ") +d7,hT"iWp%=rIGb,/ginjeq$]00.$Vk^P_Os2Y&0^3;FC#m'e^:]/PTahMT?TZBuY +s7;"?28oOUJa`W?8O41Kj)U6X.W&4*9*7frGp(Gs,dj`a2ro%d2ujQp-9NlDhU]g( +'l4!m!d][W61P`\%P^3HOSF0`T=+#tmGS$C@F%-'5K4'N%sq&@Eg<%)*ru<'63-^DdPAqO8_T6dDa2M#62mep +BZ0o-pcl67lamPp^V@Sa1u"p$!/L]4OWXYX-P,:+7KN.1epj+H63':KcbXB+3fMh? +))8c3!<1"^^^09]qV@6N2kU?M"$;(MD*Q+VJ'4r>gUlN#(Hp*`p$gj5q[)lFeld&M +-L^tKb8k2rc-nSD&V3)@bs,WT*4Id:s%he6a3JT+q+j*;.$MppJ=Q#gr,7qb&q0t] +ICg1(i?tt`363o\n9gZPJ%kq]/lC=Sc?LV5q:m>:HCt6+0*,_l(^\t(!@7'q(Os7# +,9AS%5Oq&:*-ka9I*I=aJ)e&RIPalVs5Y[!s)8PH+,>@idP+@5Jo=q&n'ML0^>e[ +EE=N)ieXk`Vg7K&rmj((%jj/eJ:G#nJ)SoWg9(5XNBfWYrcGRN/bnq2T?JAa"NgZTAFaq=Sko7R&cej.MRuh$mf+mX!e:l( +/U=#gcU$SX2;L,4-)6F0/o.H4*P^]rg +0Dg)'c?1aaOT.o(G:j6`+U\<%s%\aTKD_)78obRhG8VU*VL>&*;$/u?M[t)Q4eK?# +lX%SSOEDQR\KTp%c6)SIN;?=>I:3'7@FrVWqg??RFOYYg+oV_/A!#agRE,Z&/$0YtRLB@CGFi\:B +T=.>\'j]8^+jS?n'[\=j_Bg=6IXjOWrssb:_=6B.E,s8"P^E4U'knP>=0siT1I'Lk +:\'"*r"Fnd-b[b1<"":(5f6!oSh'nU#T+&.62j,!>+(>6-,QK("N2Jp,=(l%)uZ"X +HkHj;lt8NSKlH/rq>bI=J*V]Ai(*4n-gomp:[g)l!*BDTr54B6=@SAn&ncr1kojar +/Vtd,n[LCkP=1Q2OoM38l?KZOQ2pECN7Fa4Ps.m,DbZ-UeX;u=\-S=>V!KDj- +W'3'Ir"%iGs"nNjfr"eK_sg2 +P!5N8nu%,5q3aIg+2a@]pa4sMl!<1m%g55j?S0.RaGj^4TP'9;9M&.?+7TckQE;ZHtu!#XiM$E3m/?9._/(%FMi"\9@6[%GBsr(iaSZiX1?('=8\ +')YNl!*X&8O+e:CN-p;@?ps6u"T\Nl+b4tR$U5s]+<85sB0YrC++aI*_%42pO'I5H +*J**-K-.ZkaT&EYV[roWm,MK\30pX:UlkSkr'2_V9dE-3(a+Z(AWa2BbQ>'U>D5C/ +U]sA_&5\RJf0U/oJKUFsigIO^!;YOH5IoJs@TS5"EdXW\(&4EOErK\Vjid^b!e^:A +>!>^gb\a[Xcmf2#ILe47S:7Q;YmDh%`1LHKRb4e*C'h0N`[5i +\Ia;>ZUmXpDu(Z(7Q(*Kk8l):O=3ATH(nFoG=T8qg\DE>;m?(+iR"ptmp-7lXTk%Y +rYsMf=T2JJs6[=Q5lJHU)F!PMr9n9g=T0AY0c5A`6f\cJ%\o>=O?>Mu&lnqY>2\Y4 +)qFY;D!/]ZGWfKlo=)fiqXZtJSYQGl[I>Mr4g/7\`?!(h!VKjZD#b^%>:u+*01X8sZZ\hU\$=Nm8K)\bcH&@:e:n3%^ +!"K%gn,o!T1rt%a#SI)d"F'tiir&g$id\be49G=Kiaj84<*#5(P%:lH<(`)i"?Kd]ciNAh7l)Su?!q4fHO,7WmV4n4$M=%f^i +M`RlRrt1/5Aupk.'Ti6Fru$e1jA-pB!"H<_8@Tg$H7=4sr0E#H.VVnYCoDt7m8@KM +MMqe2"e]k$FQ]A%(]"rK+[*E#J_CE[c%NQLgf$p*?'cue//rbfSYohic2KK+6q-ad/P!2(@ +PrmjGQTm`J!KkSJS%ns:s+g(=T8%l$g91Ksr^YmjWqioMG*!3u(&W_,oo"FteImBR +-)O2Y_#LVG5YD)U/I;HQQA.+E.ajJX/I_a0p(j;8:+5@GpJ-K?+CnrHTA*qB1nlGgr\P9g0@_cY)bho-/j +1^##9)@]r3GYs/ZCBa`*'KF4eYl@"hhNO1rEh)cmVB*HT`^q4*q%4)6[/fn"r=Ep& +ikHHBr*B>$q=?+',RVoc?rC[4rhiTnN=X#-ccP&Qm"S4rXDs`WD2#>>J&8'q@*R4p +=EX?,2gH3UD;]fYS>hoXn7_sqkN%^%T83#&2SYBh5"4A0+Fg[8Ka/4ln=a,o1T+A> +@Srfi>j+J>#>K6&KE#q0V`a-l#uCQ[gu*SeIh"\MrfH:fgY44ZCAsTq&to5ZC_J+b +\L:8c#l)RDmX:Tgk9S1A+R>m0$WV&[%Q. +b>Ad4/'`kQ)7D<Ye)B:akMAROoNgB +hZ3Ga2\uYgP37T#qBh=h!:m&e#OPHos3]*UT`)P!EtJ?nAfXB/cJr?bd0KftIgg&k +0WtmGs7ssTci$iDc2Vjo;K4HgXKdW)T9o@4N#V'&%<[>os6DrLr3@S;5td=W),WQ^ +hSdbPO>Y4e7e)R3!XtQp6Pb\j.uFBbo;DTNkP8*VpHu`Es&8_,s!M&<<,HeM#UBNa +80FS1J,g\.0+IenL:l%ZIr\FEj5L!d7/it<]`1PNd_*=(&SOZ;>A&L`o*b^E!9@KO +c.5UECF1:VP7[tOJ;LKE*=-6OTX+H.'; +1*@?!`0mV.>Fc2S!I1ZS@5u[@^]Trg*NDg;dIi)?Ine?i'6dH=b!5YcCmIT`i-to% +r^1E,@Sd73qB\EB,1J9"5QDB>U'BMt@p0%N`??K&N@4qfe#]j^<=WPUMBPL_L]\[g +)KRd/$BMf5e.&VN>R*s4l9P>Yp,\_L2W77sl^4dl(pPuLhnj\j:]83in6<6qB,8D" +fI['^ZMiQks%U<(0rod:eI(:S#mp]8rkR?`rA&,:,(JC!?NX.)k@ZY[M#<$M5:1l_ +dXm?.k1W[**=W$keXXsY3<.0P;3Mtui;]g/m=8W8p(P7ABRGJ&$M$t&81AYj*!u^m +hXh*g!mm9%/%6&k\GhJXs(Clt*A7B4QZ;Wq^[18knTV)>\lXL>,Qn+*JR7Ve$8)$L +8FjO@oRm'Urf;t%3o^11KFuj2ReBWk5)3j$#dAsT#q*9)^8T.<_d6n`=6\473qFnj +I?r)>9-p`=Oc-Rg(NtnaZ2\+JV.=e81u$o-I`bEYcR8p,\DIcH#^"?.qk&NAr0jej +j[)5j^&H>jqu#k)?Df4-:$+K3F/o:3rc&[oH'iObp`J,qklL;R7R.@P-iRIqoO7?8 +/_N2\41ki8INnes[3't-cS;<#72Y>%]Pgt[.\b!&jnZ#,UD+goJ8=)`4oA^_XBC]@ +M:4p+!"\QBKnKGH]U-](QiDsNjA4ni'L[M:918\R63&l-+l]GZ3X%dm-&t#T#q*1n ++P[J17/d`BZCWOFST$+-Y\Lu#Y9>h=Pj:UZ/-OZCK,q`/&8D=0R$K`:69rEsaIfT( +M:>9h+Ts2IZB9rR3%Q,.#_sUUCq)56.7YO(;A<[a5mRJj#D8ZD".t2F4l27TSe(5f +hh(ma$n)'g\hoNu"90`HV=?rV4Bksh^8C*5&nps-6W@Z!D,6r +L$/M)K@3jm!?iT2e0N]#CoEAEErKH`3`&17qPoaN.P-&Qg@hp'E9/2;+-M-=s59P0 +0H^Gmqc.)JTG0a!@/?_dJ1QXG!1KF:L*u;,#f&65a@l]k[9,lP'M!5N5Y8^7l;Vai +Wa@uNa+KI_`ItpM_VP?eH'7m00jAGEfFZS+"57cP'M/1DPO>%oVLNX=Id2.Tg2HeK +2*?n4bMsKQom]Xc0CV3@Ur+!1>lsN':Hj?!iIf#NPs8g2KY7>e%:lCdN<4h5kC0(7 +'H.%'0m!$d9`aPZ439]H^L%UTM#hilU.l)7"t:Q^J^,JZCWQWggep(kUB']=^`@jJ +1\MT1SEIWd!Ur=pIkBEb/pfJ:Mj1mcO7k[g*Ma[CcQFR?^?SFj6iQP:aIZ\$a+N_Z`1hb;m*k@,slKr4U +^j;TZ;Wi,<:YH04+n@"3!,XQ'&Msi&]_'kFHUGUGlU!i(IhDhNm/lAI&+ZXt8!uTf +ru9gG27I>>H,os_3^8`k5R[RE0mt'R(5?nWr#KRMXr9s6,K)P^(aPRa/a;9+ejc;qI`tH@r=@nl3Acp#2 +Y:Sb:#e1"d]>3a3n9K\9O"fK,5mX._#ON3)it*?R+eU/'apT:^q5XhCs.66D>Cl44 +[ECl80oJ0Gp*lp=kN"iM,4a(W?R9@^(#a_DfO-S9]d0uUUe[%1E;o%0WH5qpf=n"q +^[U?h__uOT9EV-@rpHQFl:]-j(Q%+tp;u+lR5fQFV,lh!PJhNUb7-".d/HK[I +;c\-[-<&UTS,\-:_>fVpKDa=g]k^:pp&B&'$.'7!GgJ* +P?bO_B'c-:/>l(g=TU:?GDQ(jjPS&glK=H5q"P\_!]1\^Va0>@s*$fFof,>;:QGea +A8F\#'aVnVhbO)Fj:?e7aV"Rp5lR_WKDeU25K*T/_h\)\epJh;on3ooc)@d'8'Qk( +LA:fJ`"M]:BTT(`;A'/]H8-RQOq(XU.B=_2a[9ophu=R9H^C7-i5-f?I/gg4I/fBA +0T>s;+TIh2K$(jDs7cl"0PhO*&.!fR>mM]]&r6Uf*>&^@[85Y1k`BmWo=u_5alVu# +H6"7q"9;)gp1:+2L$aOp@#aDU@H!ccA[Z`DBj"g^kLfXa0L,\*/M.)+@q!b%6igBn +KZh_K5c?XT/7O6mRD>SZn$149d-Gu8$[`Lr4q:()-5%WHA9g45N_U-R.\]U>,Z)K) +o9T^;V*8OKf22tOS`KlR7F5L@I5^1q>6Ki#3$&"/fTT!-LJgRK8CUgEQ +F!m7BGj!dcBt=o01=r:WCi="ggES8/nDT%s&K!Gc?lI'Z%D]KN3AXd^$rj_@5;B0f +WRLgPdIL(Va3Ql.?hu@hAdfc:ohQd#8sK6hC(n1ugk6aQa:eQ3@^4Mnl0=#1%2g0_ +L\iH18WtQ:&otUt>]g=[qVZ8;C<64es!Xk$26s314P(p"2]OXe:tUB`Ao8a+M`^ +F=sr:DhgOdeT-XlPL_)Uf7R-qqkpQ=ZhCs68'RQ([VR$f[.Q%TEREFq54@j*oG8JG?e;_Z\a@kK"o9Cm=bnpe0RZS&AGa&U_7oT(&jjJkgH-MF>)`0WK_s=< +eHVf&AG\;1WjQNt;QP((1:S?*A$ijF4a.cie:Okr7(9"f=HG<)B2#odrDlgCQnfX, +$kNU9duOBcF#WDI(d`QMOGZ7s\t?sZ\"V;-qq/<"p1a9L7HaM9bM$OnqGl_4Zi7?K +QGmM9b\>RLR"jL94&#L3'#Bd!j#!Gi9(a:&p$h*8b'KSb'QB9jhgZ;'o/(f9T2WEW@h;L.D's1b&5q_(((&mi4o^8/ +,$#l";W(:!O`$M9j:gT0GH2""LD7C8'.7Y6PZ=*C/=QG[aV[ot'1$L^6KCoBq#fqk +Gc,s9gpY/0/qoc2,)u1kTRJeVQ35TE"V(RSs5O\Gom`D^9en$Bd$&G,EQJ/6k^r0j,QreUQ&J5KHA)LeIi8At +4t.'9bMYDK;oCh>MulpPr#`?t_RSsYRUiKc$N:;%'8ZZ[c4B^I6kg78csEqG9)YC/ +kP\2>IullbcMo:m-AK$euXeint,"gA2&BW?.0P,+[FbPj@ZWqi=1\MatG!bq0Zms-\J/)HKN2erjZ;-$ +Cp718:MN't1#l"CYoC"L4&PT5GH6W2hrgTg"l>=+pQL?FZuJ( +7.;NA^C=^L22T#,jAF\,;=j-Oe'^h*r+Y3.H1TI0a3oZfk/KQ1&S@*`pbqi!k:#IY +K+sL,s$7?%P'/=&5)4,V_)ind5R?<:&`=4>k6?52/;P;,(ih;H!X]_Fr%UJ@ocJW9 +&E!L5"DkCk#Q_mDCVbAgGP'-?]#"D;rg#Gds%--6Ps=q+amq;mo:N"PiO^S30aMYD0Xh,6*XApI:ql=H0Zi +/()qE(soo,CWM3[.UY"U"-EQ#U1(\H5l;^6MNRpeO2:jHhtJP?I#@hD/rE,VUQNtA +>8@O1TULqN39W!gfq3M+:Y`B&\$aju/+MR+H2Wc/J?td@T)qXceftCoJUk[jW)jW) +aC9$ie0U*cA/GIMnn5$Z#0WVn8f>=$eOkI_*4Ycrb!+JR^0Y53"=OEb!lu")gj<.b +s6".roq[e,30X1\K`Jd'Mi'W_(.?IJW5+GZmE#2FZE-e@eQ)B%-'#eH-M;:#"ISgG +Z!2_C[f4clr`R&3YH7`P\?7ZbPFuZO7L\k+oc-eSD,sRGYK6WT@4MFFYUB\'e;\.1 +L'O497L*hP%rV0G-,k?h^k[XET)TbtK,[4hhni+e\MkH!rWFBZIK7*d@B^PH,[RD>kRpn/<:!jKlddRo:;Z;H5U*Bh2L-+r/&$PW:O +T!5Rqg"L&/SWg5A$"qeqZ[12.2s5(KG;h.!RMNT]X2/se-.'(9m+MfI5j?IcSlM*" +q107`DXS*MJ2LB@)X3$I+'=R4Cbj`#TQrcc=Cq*ah6rO0SWm\ED$IE]U]b5E>sMEt +\g0FM#6/VR:f$l&5[s3k!j<9hNpVuG"tJ=QX2#bYD;>a$O!cV%(, +;Y_]2iWPQse-:Z2`+.s<6NHjs&J0coBfH*kp48$&5_rWt5f-c_69q7EjnJ]aBg9Z\ +[(jn0!V!<2m/kk5*PcEk%9SY"/1"I8j,FTRk$g3Sd;"+9!<8f?!:C+e11cSU1$^KE +&ASm((^`H=6:F.!LoUCs?W"o9,iEtMo(Q7Egm5uub7g5=`,.+!g<6JqK=N9.JQiE8 +o036SO1;#grmHUd+:gPEXA'bA84-MY+bM&9P;4%J5`MH-pF/P\s7m4fo09:AP,iMX +s-&1PcN![2GBE3g\`"i9r'?D`+gM'hm=@:tUZfMZZpDt1i.!Qr_,(lu9,MN"Hgn:_ +hufbNQNlhQXJVUOXM=7,J)rP:&CN^tlH6sCTYZRjSqVP/^<&8.XIag3US45HridBn +2":H.XKMf+s+cPUFRs\bI2R17$H2['q',?7p@Z'#!d4DP5SG_uI?fg']e_)Uh\Rt' +=9E('gYK@'lX(/N;8Jl=KB4\4c`].)9h)h6 +-oW!-iNKPeXe9l5m]XR_*E?J(;,S +e$sO*Z[>`aGZm%P>Wm()$\;^MTPNnTf/6D#F8ir/j@3dF'B$nYV%Z`+VfW!Y +4oZm(U(jX!M!t1I6?o^6q&oaS?*MK]o=9X0aIi;nCoI\Carpm!BG>u0r4X:aW?hHJ +i.3ae!b(/;Nj))?M4"MR1\bK3Q:Ueq`T_=,%e\dJk']D6nR3r +\E!To]#BG2'14[*d@N!PZ[gB*+o0=!K5u-#!bjkSs$.G5*N2B<^e"H^A7'J$:_6P5 +'5C)C8%fLOor,$*9OBqsY*lTFNgm%W3InShRjLggPNJ_k;qn[YCW=jgCK<*W1i$^Z +QH?@J2K)_m>9;f/X'UX)/OZ%/_];t?JZY#WQW7lW8r*WnJ(3,skVkjG7_JikK_r6q +qRrQ^ckST!-SGlQ8$;'<4+-)u*=%5;XRA.66t_r$VZ!<>5@`JA)g7CUXb72D!XKl7 +^d0)^bV1CVjtOZD7UqeDbbZh:TcI:"#'L1?Vio.@`QE>RqF/rgZ!n;3^j4bg8_)1k&J<07JHX/j5)<<`g(K'N:@lU1jcM,]pN\%@Qb-(%p@(^6r-,L!"\^i*sM'b +$98&--4scmq'bJMbsblq&T0EOo#J5#+a=Yg$_biNn84GHj,Ej`m5dd!#l=r*?JMt& +[1Kbkk,kk%P;R6`BFB@e+7fI3hZ$u4?rZ"`E&E-.s(f/\l1jFHLff!$qU&J(@j7Y;?BT-t"Z?9P#AW.IfW4 +&[GFLH8VJ.1h8EP&O(!3jSM/Z!r7bYs$bjqptF()K`g-aT=]i@[fgZ-\=>e3K?PG% +foGc8]!p]_g[TXs=A?49(6a^cM$AD7!<8*J7+n+1j2$n_^DQm06@]:UJc_O!IeQ4% +l[dshaQIg3[6O4a62Q^JAtTE\rU2M,]$4eM99J-.,[`,43Q`4P"m`06jCdks$i^/rZh5DVNrS(F +s28FGV9ecgCO-UP$&7"'i."_t"Sli0,ENH%jeKC:S-pM9UQ&[BGm1!)OO*ES59pU* +VENED'Q_:(oY-oE!no3f1%`0kE(BmgXp:5\T/`IMq@KB)#nYL?_h/RE3[fgnLr908 +<`L#<<^p&?Qr4VsMW60(#m$!r4KO8K+AE#H([2nkMFq/cJ +FtN"V-uprp?j'oF$; +k3]C!-Xt_>c24U,JCX%AFV"U&"X/E5Irdi75_UX.BM-0J+:=Y#ABs"'9OgB^%fAL4 +cos2a@EM0K:Ws?B'b`JiiP,>YI44n]`?1]LVlaf7Q@/XDbg6V,q,@<7]a[+(dn?b+ +F:k,E\IakP^Yf%kp:luuIJRC/s#*^ibFcNP5gR)Wcaddd*ftlo/gC:Uf`]]ms)V-P +(fNlE=<9;`Jc(.taq/7D,)"QuYp*k3pWFF!)Gidd@qlFhW)3[os/2r-P5HR)gL]%; +A6MFYrrA2]:94s+oB.1+GN6BpIpb([\WI$Kf2="(/16pQ&rgE:5cR*0&_%)NAL]L0 +MH$N:&Lr\B8tm^:']V7#6bi5Dcg`GBrr\@tg\S]sIcrU#*Y\?2NDK!<^Qe2:+RBm5 +6NDrE`;co3Luf00(PSqW,_(^Xj<$'I"%<,lM-g_S(XfV8O85Dq8G7OIr_KtQa\Wm: +P^q@2?\^:aa7UNLrls*nc'ZLJCD1E[UhuE.4*AKkKJ(0JU5>6+LW9Y=OhpV8i'Ztk +YO1%n[d3rHS,".2p4)eiJ*sZB3ahbC@Xg`4fYX,OW1lJMA][-U?NS!]E!Jo($q$Z90%ftbX!&0#9M=P;=a`0-)I.:M6P9CH%6n:Xq$cujBO_8["%LK3IfaGF4)dlfTP'-9!$SK(>G@[3nq(Z +KVM^`HT#$&IC'>h#<*MPb5"oZB5g"c[XGV0K6ooEM;.dS)Sad]?DqJ=Q-1fb[be!T +^)MIrV=*uXn_gfho>WK]e1c&\cC?pnBC=9K2gsa^*d^e_^7c3Yp;!*1[S0qp-I4P* +B74if4_l]sG@#L^]l)_Te["mKcd!U:%c2AI_"5b#oum3^PH(JFqSeB(Q)jb`l_^5r +L(3'0fASgIff-X/(VuN':hYbAK.sdMr+K^A^^c#"bUtFUU0>]'X:]LBhqrcQ7\Yje&kQi6**P'dR%k;l:UjX/ +&HWi6ke@)=48f#Kp`D=c9M:sCmj-![BW25hNdAOF->c'LjA5]7;uI%IquX\6IT(sH +=H9b,6@lo_o%-s;S#Qn\Z4]K+0*)"ci^PZt5A:^>s%>n0.eWi?&?m&?#4:m!PlR2$ +2i8aW#Bg_C^il`)1XuXP!*DuS0.!&&@c78I=9;0O't=iT1;X;>"JcnL!g:BWR(k33 +/I?c]M>sjV&O,MJO82_`l%.Uact.I[1->Wn`[?;W<:$rUqTVG0bEJ$i"T,rmFd3X3 +7m'*>lP9!661=Jr:B"_;;<$&]r#.:V/V`6qj/*1Yr^9CYUE#h^`VC%IMH%Dq?j)gKaL-t;$8h\^ ++oWQJ`qhbb9<:7 +%Z6t_jcP,RE;u,Xnd3cHb4Z.o?89nV"[X[Nh@V/G1o7EA8G(*SI4_4BrDR!fKb!!T +7r54N^"o"-C!@WAO'F.^YEUipaK@>>XlEKBo?UT@T!b`W[&dRl/j6>/5HUJ;W#f9/ +Z?%eAEgpplm+V29\,:!&5ORh$#QKEO[rCKE(@GOhoKii9$0mT-mRpLD<^$DoHh"[a +om,`?/*618X_2!a?2h#,k',PFYI(4"WOH2Iet22oYB=oQqO+88.H&4QG\^TqeB]59 +`c'CccW!Or?-CjE/'f0D?8h2!epR7iAcMA1SJP!N03XZ[RMX:cB+LgPLXN0&L:)s) +Q-8IcbIYsaL@&VJgTW;TgO8VtrNkt+\"EJ>cMYRaIQ%ORP8W"^Pe/(bQljm.!'^VU +@f\\*aG2hh*FUn,#B1A2!?cl0/J,D=&-1%'i,Gdi_Ebo+m=Rqh$.SbN:&atZa#Uns +aF?@;X8E)Wi.&iJ(HonE="ih9Urbjks+I@k*;^gC)eZ[PanJ$1LROSp7tLU0r_M_X +)?N>,#Q?)c-.AGWZAmp`7;aLD-[C.A7/dKC1-Z=9AXn.V:WW@H,,?md)Ti&P^'m=n +^aS[k.Keb/6Cfg/'`a1n_!6S]]dR>$"3CKh$&!h.Y!\J%SOEH%c74A5J!0CJs'tZh +`R)LtUjG&S"9.1No^4&Q_>f1jFSBOUAL2mo^gaqWHrKUe*UN^l@Mrm5S1"TJ.!*dGouqM9e)\+-`Df[r6bD]0R.)XhO4lqpK$)X"_\9<-WlIq2gK)H*Qpr=+ki +@74<82[N<=Z<;_`k2RDm:f^p*bCLg7%anA9Xat6""l-mHrf`rg)?jfGcXm"7]J+47 +jmEUGcE@IN\lsW;!.'6I!.Q$[)rG:W%`l][OY.6-m$GYUU0(c6XKOa`We`,t=_/KG ++-lo.9#rL@It<^Y?_A#4M9agsB?RY[]/JU)k`2i*VUhc&2WUY!U7i%Td#N5?UjDmM +DZ6_`p..)i+ITs.c@4c(J(&c@ +^&%REe`k\I,OX!aX+YIV#!rk&ljj3L +]`2CH!:k@!5NRTL,+k^A_#LL2N._1m5MuM:5_,&J"A8ZI3;FAPr1EgMTTpZ"%ur\'@_cMnM"XXm6u/!ERK7"In40[s#=kU@&9QEdI\:g[q8V_$UJ@bN-% +k3>KA9=CaBZ;%[ZkLpl2m<>HelI&o-i@XT@B)0gMPLS@.(`V">qPb$Qaiob(e-%PfD7'\4t@oVRuDZa +_0u=QhR-GrjT!u9l.@)hG)X8Cii6_W;SIN%>eH4Oh:AdDBNfi;[.8A.rqG]a[p;)a +HM>-NT7#5'rMFf-GFJ+Vg5XDtG@*7E^M_bZ%mP!hHJPfS +>J9O-C@PlkqMY3PX6TAUMUaN9npu&b83.:\8!T^8NU+1KS&VQHXTu%6&)X`7I,/l7 +qn^J5)d`L"qtXlUdj&dCh5=NdFT<@Hj]VYh*`N0EU>a.EK"+)DcOU-]hHGGf!YkEk +*,6Vea8U9&V',t$$iGJc\)Xg.2L`M=?4/N<^AK^Ur0P%u$Z=M-X*9kIqdTRmnt2uY +e,bZ?cu3OeONVL@BFtl,'?]aY(&tfr!1`u%hu@q@/Y_'meGR3bl$N;W5l28^ImhF3 +_2&Hr+0u:3+!5%R +JeWkeLNH.3#s04Q#`!>GV%7Gr[1je:"jpHOuD6]^Jof3%)i +4n5#"M>[2K-\U:js,I$rVZ>dRr?!^$H2a/'ikG\?rXf+i&0N.Y^E`b?rg0nmP45,$ +JjTN>q8^Gc,6,9=F)#Cp\C3R4PQ0Hq3h6*kq4B?bN*C&DTQ3cMS?d@PcZ\*1]N),I +uJUI;?KDCH?lThRq5hm&1#e9#k2GDRrV$g#"qRcn_>4L-KFo +d"JQYf/7,]%Hp31#+^>cdoYH$Ir;To]&(gD/-/U;6=%cc!!rnR(r?1g0RDm`g9%,;b4FJ"iJanP8TR-DN`r +"g=J\06A.7/aH1U)]()sH[<@qH) +i(mZA&MBTjX(_$.[Wkr]Hr[fN]Q"%`D;'HhEq#t!phf3a[s]X]%:4I$I/TW%b:ihG +(Z(!C[pdhEJHjd2<%G]>J)@efb!VjI\)TcMpc' ++nkqE;O-i<@OmH3!jM]/Q=>he,65uBW1%2CJERS71)(F6r.9T'&9Co@+Eb.X+Y^@! +T^=I;!gk'T9b0D,#Au]A@sI2n4U1KkmM(.qrf=Q=+tc12/VI1,4NBRo&W#V"`[9^B +hl@RTr%H1f?]+'Zj*uR)SIuM(5KNl=s"ekgB+"QtaU_2*jDSoYnjb9!kjob/s'Y25 +6ef,EI8\fRVs.IQ=)TR::K8(3qmX<-4o_\JfcRhDs4$T:07>MR3q2-Ujir"%c[RX] +rhK;6\!F_:L@pnE\L[X"TeWgFT]?f`@+5XqVIK% +L#*Dr#0LWogn^_#1V%0??UJ0fr1A>WLB)@C!0/(is6hpS/]FQ8Y/uHo^Pl0Y0R]dV +Z@Yp\Jj#C8/eQh]B:GnWXW937R[#eoIRj%1Hi.$ugf;Pi7GH%8c1"26I&/g.=e(") +s1!)[Y9^OT+M@6+h0]oCHc.Odat.uU>(77@.i[=n*JN#EKSdsrba*Ku900X`Sm3h+ +CmILk*sS&qnr3Us)#7n76FH4-'\ousf)kp:Cu7j\@jTR;&Q\>rD.f0aF3WG1Nr>H9 +]lW`/M/`HiEr8TbFt923f_qGpO)rGi+^M2DLgUd-fj5@4Xj[J5'gWn8$]?c&S +gQ,]!mQfM+=nCM%SL<=%?%#atn4^(fT6=MHEa?/%=ZuLY2L?p@3\Hm)c73IuMW\E5 +EE*C]\p@;ar2a?"7J"f;/OuGBS\48Lqu+>+&%ZEC^]&TWgMblnT6o_oS@1;!hib0i +hQ'oTp5fhNU&\I/^ajhsL&`,4kXpCJ!cik-J3ZVo9c+83+;!JmJ:ggMW;f_)s4o"D +]&A!#o3_#Tk-*ac+!ps*s"f[qHc^a>TqLbipHNA"^^F$DO[$EVm]QSPr3bOok0=S^ +(BRDX:sT(RrXgj`"+N?C5d\k(57IYcR@'V:eY3;V$KRiR^U>-Dq,mUrpIj4\T;Ki1 +W;u`-rtG4com\:WA,[RUF!Vr6"[\=I%nZb]ir):Bn9PdK*Cc9($#g%CImSua5RD<] +JG=UWd=tC!+TFCY,sX'=#aUO+d$7]NT.&s,N4R0Jrt5-l#T!P7(gD!q>3*.TL(F@= +-[:W3Eg_C`^easuq7hO(TD0M,gs8U.>r;<2Rr2+_JAT?0S\I_SaB25ah +mL/3AIk(I)9+-m^Ql`N.Z>HX0B=cJ,qR*N6B>!qE_t[UmlLQ!WTu)7c-,=VfLY;oo<^j%!D5#"aVbEWT9qAcIL!nVjj7$e\Gj1F-r,8,N5M*5+l'r;%FpA1e'6Y1qNFCA4pm]4^-"A)YA2B_\Q;i`Wpj +N=5R;lfQGn/aMl:pOJU4lW83:2[t[`3k*u>m!d?(o"+8I[Vk9*Ad_WQOQTul(G;QU +8#7H.AYWo"A*8Hc$d$Ukg&/BiiOideWFG"eG4g"$hA>RW?Z^hu:VKi"I]pH\U[cV' +k0JPGGlG^nG%(i]gBSP.]lUpE,u6H";#m7",U1#SJZ]L@iQ"8MSqqPOW[%N#MjnDp +..D`Sq0mZm]_B8X5QditRpjti?&>8TTkQFbb+9m3QNYN^S/!57*P*S4Q]XmYlAQ96MBC?q=FMk6'hT$s'`AB +r(lVXf`0r9rEo,%rs+e:K&R:gInYFn_dG1+oY1@i8Qbt_.JZ""q%1ogae-2ZWX8Ta +.f]@rBhS2pC[e0Kn"jaeXm,J)j6[r=0E9i%l7[R(rn?u/LX)A2]!$VI]>6W,F8r;U +%=J-kjl_-D^A\Y\kC5::nSe7WrbVSnc]0[Xk^N'(kNpVN[5AekmaVa?G%=T))-p1\ +d$W1$gU0HEFBBr::(??''@+^l&06i7&-@B<=qT:a&]c!b3""RQnU#HhP<\I5g/eRe +N3Y@k,7eriRFFp;h@F3f/5YFF.8<\5C$jpk!R%uQ-;8r$Ft'?fLB +jUua5.(!"hhC_IfPboSR?[WL?/fELrT@bP1>5g@hqCi?hkd`+L=;7BW&tOKqbS@IB +VQUMi'C8c0)M]"85)$;a8-i0FIidLfb!n?rbT*8rt&JAV%F4Ra*iq=[Rrc^U7c +eNQ5>M#@)JkK1IL$m6.(NCVf^Q?#5JoT"G$3,A,QBD!b*Y= +s1MNQhZ(\=l@8dGJFEMs=.!r,`#Rmks.pDFXDTO3rWhTg+FiC#SGAKUbp>YG0bAAk +E+ON33?]dMb*Y\P.R,Oogb"ED,64P>'E"KgrtH\/;UkVY&Gu,uh#B%h`]j%ci;SnqVQU+r6TZbL]6JoWGd:5^AL:qndimt+*q;nqnP;_ +^`R];raPZn0D_mPB)M?e/V!OKa7'#Fq;i]h[pPN3caBO;"25YZW&aYa8C8^/pR.YKVjd"S +9shA<\+`V+(m^XS#HaukcM=N\`A@>VM/`/f1^>""N.]ur8htI!_1f2A@VKNG8RSrj +fbGdAbEIhpR,pi#XjZQ2q<2!ZEq'fTku23m$/,d>.#tGH8!f+NP7Ke9T@tYWYtk5: +4T-5br,:B6%g1:Qf]'[2M>X6'q4MS*juVoN)%ZDcKiBRoJHLhpg).Ecc(I09KDCK0NAQ-ojk(1jIemLK< +^N?5Q\?9.egMu^_fBMVHr>3^_YFB^m2!hl:Y.YOOg^9?"8nLhV-Vc0=Ns(Z>rF>ZY +O`Z%Z1ik(5]2U>qY2B2XrTn_kk?iB#^[9nSU3lWOf@`+*BCN\8pY*foN<_,=F$!De +pjXX;6A/-%^k/VN]L0S&5u<)&LMfBKp[EA90EVT.[4r*k,RWTZh/TIq=qTJFRP>R] +nC\/@nGH'K!6>iOc`dAqS=a$e5Xcb\+u4';:#5UWs2T*4:UYSKr4*<`gj-'s +qELVd&HAL:TW(?Ma4n>0&"t>r&lS*E+TPi$ +eoEP7K)2*ZPu@Dks'tj4iW(l$ofo#I5l?Zi5N(K2!L&cC5GXVLFg;"]m41.=kPIKr +-RJ[4pn-.>7%8\9X/XHrH:oJ'$:Lf0OVp#TQM(Vpa0_gYhPOHh7lDWL2k=1QTtgP1 +ht'(??U>$KbTG9Jhl1P.em6qmGSJpP.#o@f5NKNSj`% +iV3A5jbNlfnYcDmIJ[+/BA[te?9+AHFNj*ei93F!b3[iGaMK=mKhi`T:HZ6(JW32!od?Hpio&=aNmSC1j1j!#rDGfMYGR$23*.2W +-9j@D5At-uD"IV:Bqj%ih`<[6oLoKi.uqFjPPp1^G1Da>&nd7odhgP-k>ft9U;Uq< +BYQ=2?TrR.0n1P,\Z+D4dC=NH>f`4+.OWPqe(=00=1.@%pb*NrhDQCfW7neE +CqX\5qK;,slgDH!c0Mk3lS%ACccbKhb@Y+G/NeG:F^-bC<&VEk0tes)4]X*UaVR5A +a0GmDGBcukrC,c*Q`^/Rg,D4a>^Or1rMKPgX5^i;gA3S\R=i#j[UQl]Db;L +1d:q4Vu>)f?cYUkr&E1)ht0-Jni)egoOYDmL\dVN!W#Ekm%^W&+X$ah(i9G/]03]n +^K)?:aSVS;?J,:fper;bjF8-NA@_Pt"?9)T&!I*;'u0tS*jdeh4!9VMJRCRr]bTj8 +O0u12>#qA27e@*n1W`UU#LR\c)3631X@tT'@^`S@[?NuZ6.f?RZ5*]8.Y+&_+pWRL +p(GAJ`jgm,em3GHmIsSuI0*(G7M,/!@QksNpaTEb-(,6uAjeT^Musgsiil/">%MC< +Va2BPb8FSc8?2`XJa-']sbrfeV]Cf@C6uqh2(]_V>/h@&dRZVG2QsJ,$G: +q&a^^^/A88Y4)5aY++l@YCGGPrMSN,J+*F(Djo#OXF:%qs-rJ(hu6(?`LhKAqd\Na +IiEY22r0u'YeL!AY&F>Ys7_"cIIh+7=+>CCok4:@hg(g.2XVW1NNLff7odf:t.)_of!@;Se,c +3i]>K3s&qETFjUMZEUp/)6@KIlejS:W4fr.W[\MO4q=U8JkBIE>86=bVn9!.c19E= +jC.r8:!_8Zp8.*/q.og(NGS-qFSWIVohrBr<4[l8+ejh,MP8;S`)]HM?SZpUT3*U: +/j'QG4GduX%daRW(I&505>(:,A.L32[YlpL+,:UqAQF^ub=/AZ^K;2EojutEQ#,kkeM7,3o52b;MqYQl +RWcD]h0n%jrMF#`R=#=X_.WG1D]nG*GCg]FY%dKk1uT7cLr,Gl^Y#(hVBpPCkDs6*UjAgAS:0=/-js9 +b9S(#CG&W1_856+"qf4FLQrFbjt\^%)`$b7`ndk-LXcIe9Yk=cOK8pMUP=baE?f:' +B?fbDFY.-.$T[-f,@j*Hr<)>E^q`E9[DAsue7e$sY@$kdQMd0ThnYCV[HQu]\A*Wn +F2m#jm:r86[\7L-\=]Fk8Q7Qa,Y-A+d2O3De-V;Td1%eKLlq<8gt(1 +,]6#*YbFTabTm=c7KF&Tan_98kYqLXKT-"Q_*"(UPKk?1Z&qFpc&CAl,`;^t[*Aee>CGX[>r4a4c +!rpT$CT!S8lMm)A:uHiqN6=N*s0*%?>.JOCJGfcSBg:\VXLeU6(30A(PeHa6@b\?[ +qpW7!es>Dq[E9[=35cIC#_XX@q<9@o7qDnX +rs)>Em*uG.n/2^-0C1KNHAs$j>^HlGV`r/#(&+\KH^/&9;$$1dUi>`q[W,WdqljW* +C&ZBkajri$k'?C7,O*d;]Y-$!pE*"f?HJ8#9(nlJG+L.+?BlMCXuLa*\b"[&r_dh# +[IO58bP>r.frM.5BD1^SR[K.%T1<)=,?(s!Z5X-LC5Wb*$R*gYf#n[Eo26LfM7&4/ +]7L,OZ]F53hd.m-`3r55,@%JW&2&LQU*0mm8qM-b/J&W$C(%T&"\JKKXiX57P;-ia +e/BbsfO^2afqn=E$)]C;W@^1^Qs[4h(L340<^m#<9^bT:CPD.ZB6lJhJIQL9R0%V`L]Sm>K3V9oeBjF`9"/Ij_XR +Q/c32qDJrg>KMN5kM$lL5l7>#;1S[a9)gBaZcrsG-P%;qNut5m*2cO!Vi>R?6A"6( +N:33qC7`#a4<;s5)rEg)EWQ*)PCgsaJJGU(1*H@MH`7)Pr6_XriB?&I;:Z.r(?+mL +J,ED2!I<[BcH&l9baQQ;X!i+Y1B@;;:*@2BRuR4S6DKWbfb"jKi&/S,F`^#)?-J^=rdI,,5r7MA,_\TGmVPjX%R,.Tl(CZT2pXVHkG"D&Rs[:#Q/LNsXa<-dFOIqVo +%fAi&0^GdloeojAPmJ3+W\^nqo^#a%VbW^XX%>^C^FGF_.["SiP,_0=_#e?Y;_M8R +:'#0!k5ig,$i#MQO(Z:mK>:Y7plB'7F_e=0FF6nn(AIMq +9l2mNNBmA-%Rg1*`n0CMf#s6A;4ARAK`+:rN,+PpLj`)>F"SD"4@Joj$/7#m=QKi3 +]:9fYN`cos*C*PDo"![$W)=mR=3DF<&gk\kMg"btF/_&P]iN'gcK'8V9,:-g2Y)Ub +"n%f.hM(%)-'!%tg:\?d>bud*=M\R[%kM+>m[C"r:WBe<`O=%=3[U6]+`;s<] +&rendf,Cu6-R$!j'&6(`G`#IuWFDiqS*)&HCW_ +"q):*:)ZON/VXqUri'*R'A"/]3`#'I6JHqjW96NGs$gH^Gf4K/s$?M.@FjXgd_J/g +e.HWqKKj`3S35_j+6?2Q(67a=L(B0>=-_WMAYmkC6%YYX4::uE$N0p'fqdCAO[ko$ +d0dR5,b-O_O:d;R(D(a4%9b"ZBH!`V;?OuD#-(9^ZBo%ngP\?" +^=Jghd[n5#SUgsWZ_+q53dkk[rS4ir>L>YXdlVS=os3Z44A\YP//dmAUtck+Uc]&9 +C4F8Yj`8T)"9a*,b9U\0)0/h%@kJ/$!8%Fe9@sEJ#j)fdl&[t\lnVbP,d'`Z?R;RR +F'.R0N^:UfL`+`0E5[r#DCtWJFEi@H%OU**7KLQJ$X-`g:9K;!KVb!!%_OI":i$R1`)9kpn_!U3Fj0\?/UsZlOO`aE,M;AJL2bWCca<0FA\_)*#+tJKu +pg6`5=A'\8rblcSqgeY;U"bt[CPi#!H..Nd;eRq+jP+M>quB4s<4FKLCB^G`=<`O%]0`"&ZJ/Q!B(L4agP6C7C?r_Zk +?4a=U"mmU%HWe^!g2AJDJIrH/]0rT)KkM"XpK-OqlhdaRq@WX&.M79IXiLb.-fFG$ +_-[W?a8AFVbU<>7ML^2d\X#!-EN`Z9$Gt$-C0)`lr^Z)PW1NTBrouC.*/k=*1&jaY8H+Oa +r9&??dCa_%4XH9kM#9kKUPdp(VL_(ON`GUW!OZYVs&WuFCU>8['?it+K@&-_O6c]h +r0qP^AigI$MV_UW,fFT)ZPoi(%JMrgm)m:9INme5`:U.6q?Z +8SUt>a9&RR+I(t6K_`q#Q[hmI#V!+fN6[#DnfRPB-bB%r&J]cQJ%%T\eTOWsDA.os +$2?#95'+HiS.?Cn!SQO)a:@c'-AC3AkG^#(05Ea$Qp@Q>VE +*^Cq=epRJRG5g'9pLjC!9C)SkH=Ti\mh"Y>8i%FI.X8QK1[G/ZlfLZ^Jm$C-8Z>_#0i^6]2A*nPU8+n7k3<(C +X+8rlQ/8"0Qk7fLkBPU(!uo+Nj`KBbJnoqd\&db8e1XXEQfHkoV0Fp9_&C3!l+g2E +he[Dd9N:n&>RS?^s&L/E*UZF->9WbHrm/N*!<8rGm3=P.iSdU4'>,Z:Zds2m$M`NW +K@]R07GXU-D4^k,V:D4$!m,//2kft5%? +0!O/i>h9Cj#NC6[(r\3If*Tk([`!1'E"=+r?-=t6@F[EpQ>^\riQlXF*0q?grSCT2qWaTLi86-'t[ +23'GHJcCg7*+ht'c58csa?ObKcH^EBolc:Kg$E=l;sN#N6P_BL#\% +0Z=!+*p.tH5NY1_k_9]/5JWoD*CKr'fBH/1Lfslao=r+0Bf*jfR])l+1QCsj.GfQ@GDQ(pGMQP)9X.r!Sbl9"B0KLJH.lG +5$YMKFb!\C?Eh_op.'#&Q,bjOEcrVn.3;3!/Y0=as +8`Q8\Z^EPPQSuRL27Sb/Rpe?jBK^l6i;\1@cWjH$'#b$+/7=J">GZXIia)0'h4W0J +k(0"E+GfNBURQi7_mc("W&d?6RZ6bUO(s6$^jkhWs7PZkCM":Y?dZ905NTn"R&DqYMb$A.EWTEuY!%n3C9JN%RgCW=Ec9 +m@**2665?e8.2q0[)H%S@#Ye@N)=M5OG)(34@0 +:fkjD&`aJq@3AQB![:FPZEs!-h/?icE%s+rJHJSIK,s&BrMf$nSYEeaZQQb]0[ +LbU_;g^k!#!+#qEcqta(E/h%[;!SR1TE\mg@_O-P,VuRU&utamfMD"/g/::R60%*q +c*^7'5>j?d/8heb!uB6o5o.g8=?rVsV#Ut)YQ[uOeWm#]Q)9*9e.nm5"b;.'2>Kg' +$`G5%q2^5U&cZqO\#V;H1l21ji'.J<+2W@=o:,>+s1ns`g.+"B9-,\d@JDpD;R`rR +Y.XQaIh]p+I&JIo[57qe;gL"k.\D-VepHqe:][).TTB\YJfeUPqLs +rt0bI=>16:,N7.4E+3/5)DASr<0G>1?"+Nkqf`hF8c4XSO%!q`_j_F3>#CnEVb`ZF +dS!hkRG36aP[F_U(P&<3XOVRc_A:prUcri@I+en2,N5d8`"()pXIu[$,(G^+1DidZpnq!YP%H)e/Tp@LoTUK6#\"skeT,*d6NZnr3. +r,3d`*t%cpn]fXpAh+V<6HB#EQ(M*5q(_e_koeo682XYRQ>-p!^X:1*r\p;[Ra"FK +L;YEE;PPS46'YQU9PUY&BrFS*01uHVW%U`A.o+)"J?j]QPQIN&NJbZCO:.^EnG`st +;'V`kFNf4?rXo0)U0_o$dio"q"6\&b7@=)rA"FWMOob0Z6,45nheS=`.k>7jRu\;Z +B0!;Znl?gTf,b:NSk4*tjmPD^fY/lj50MbBceWjjLOfE_6f-3`aT:D9FT7&*qg%T6 +nGbr6cKg)\T2kVM<8N%%EVGI`hHSP/\*EPqq?LC/mgMuVR\"sb5PSp*pTaS]WF^Fe +f\M`kfcUjZ`6!a=[T^mUM\bpW +@4\9uP1fpXiVLsWrLL+WHVMNkWqt$L+TASA$7nFDr3,rpXS]I8o,e,Is*$qt2uZWM +s.,n]4&,/S#64@rn[R.qAiG+(nQ3Ef%Bs2(9oL-E,RF9Mj2Md+.JU,]A_p[Mg#4%=)!'jPjVZ]8l`[87]nUB8%!6> +IIL=bQ7%]BM&QR6fM!.arh&c$JcB]F:$KE`3H7PCJ&iQ2rZ>aH(#]693a?im4gQB' +R[87G/#rbe3D"\4]p%7C)6j>6M0_!@63R+@iRH[5+d,:tlI"aEH?J4RXi.2kc_h2S +/q_qjPS0i9Fr=,A*j'K$dB5Bo&HX$GJO_K=$)-.?W,kRZ<[h:e[X=^\^gk^bf^mmA +WSIp-i&jR:S)_Yaq\5Ks-8k(i@X>$32IR3k=7)fuN0f=fE.MgBh_$4re# +(P7,;[jh?@EY&QWciAtJDIcBjC]Qp1k]0OVj?K%B>R4%fq?Ic9G`RMBW4nF$Vmqs(A)@!;LKmopX>jNQH%K"9:WI+-.S_(4`2lL&$c\TLhpr1=1RO%>5YE +ECAk`!=NSGdD,feXGdoV':!tWrt5&IFUV/"FY#kLk#uq$2%/rhMVV`S2d=re(+r]S +X,4%\D.98$WEGeAUeG[)88KhQBdcUao:Jif$E1BU)/X05:BH5U!:_/7E`V@u_bbLO +#n;S[\.c0C49LRed0N".g?`^jW5g<0RO*C5>lr03**88;HYkk8q^1mKsaPAZn94D*=&#hNb::/,%9M#PW8IT'#FS00"3?Zi)(l +s1*CCV.:*7JEOIIc_R&6'Y@`RDbH-6$[>a%kAB>6#OV=ke_SiW1U)D%/?dpp+U`Pa +pSoTjYBfu2g;hNU[l9S)*i[8MSn5u81?Khar,k>dQI!qIi\9bbHl?%>+Jd.r\l-UG +$85XLRmT[Y0f7Q_%6n0c*O!SFXVBQhbeC+&^1BirCZEV#jR$a\U##@oS]&X@KRnd* +!5O)Mi;u`opWbL`BFeX$%hNajn^@TIgn8pX#bEj`Gem'i4(*-$er;gYf3F)+NcjWs?]V)94/ZAQ8'odGk#=kdbHr^K!?]CI$e3Y'Z;nKV*XGhRa+d!EddPf'/YU/!M+Mq3PTpj\et\s"*:rZ>^rgs-`?q!Y&q_iY$bl6*d0 +0t,1GT?m1jQW=8i$?BW@aaY<.7k=dRIj,o5q.'DJ.C2+$,-5rfG)'7W5L0nhKVKG_ +[n]7\gnW1!?bcm)eE9/ed&3*$Bk;_H]O$Sl=&[JD.rlS$\r@&pOdV39S,>!N7rP;A^7>iL5]lj*)k&`oZ][YZ4-F.8(T]rf'RYRt7N#d4<6 +,$gkQJ:V"adrH"n!ZJKIo&s+fDI[8A]Du?34>hlh?]VE5;KF.Jgu6qhAp%e>GWhBQ +b_u+?pd\LQ\!l!F@J+:N]ilcF5JVcK+WpJD;3qNF["-6.!/UPt1h<^.c(:ks$rr3O +_(&lMA>Anln3jY_8@8A$[e3P'ql]_`?.n`Qjdi%:kgn:S$5%%'=Su>:F6Oau$55MV:3.h_e'qKg9sPiED--L%oqWjkXBj +[[GuhVB=a2`UL"W4HiXEj#@2kaRC:>WrJ3UrRZ/GkMk'.,n8bpf6rqIqVDdr@3:MI +!<.TNrfbj?n[On$W#FQSVS4-oSf`rEdC#21p).Jg"LS@'+6=`3:,!fpJ7VmplJtEK +T8NpK2h;,``-*\0](,X9>\lK((r1VF\omXWdr\:<3m4ptfT5gR*F*.Netj#]r"8rd\NMEV +Dl'PI/?cZ2hc4"J.BW*)CBGf_l+E_FpOlR%.r@%4Io04h&0^ZdI8*\SOiE!so"T\H4s&C&_J'nBb +Hhbc:pl?&=aS,Iiql.@)Wc[*J)EP[O,PDqi_de(Aj%1+ +^QGVt,7B;R/:],L&S6Gkq'#Oj;F:We@pQA?#Im#V"6?\m>&A)2L"0pM3D>qY"/eT2 +Ba-@k,r@-_9pM72ns$#WjMoWDL6AA/il&F&"FQGjBPVQq!TA`_:Dt@kCRU,ZM!g_r +R/b*L7t)lc2roLZ-7"p4(\8_4j<1*Y.fPoJ]`ncG;(/?nb?R(gKMR/.o3+`CZ"2js +B`R[kjH/bT>"?s]Jn^IBK6%Bo^"6n=212qr2'83 +?Q5VbCjc#Zk2^O\_1m-+2d9kS-L'"33WZiL/Zs[q/'jrR_A5s#?mnH[J:BSO$E=ZU +K+">fIDnOK!1L?7b*p%r'L6L6O0%WmF%h)AA,]PZ +T%eRf#iK(eV;2$%*^]f[9(OmQ"8G)LCBsi+r"#[_]k[1E1&.o@LLTNqBF_7sOk<\g +#eWV,h\62UT;?YR7(:C[ZqfJVT=j#7F/sR5g+aQ&0c/;!(PDJn0:8K@N^YgW(25qd +gqYOd!4&<#KV9,u5JVb)ARm`:jFfI;Q@\M*5?[<<4u5q-eRnM)`P%P"ShaGmOh@m-`*]A^0"6h+f8l9,ba!B(.6Yk>rhVCM +6A]ohco\@KNKl=2'ikh\RKnam_'5_jW7#%I$s'BUWcG;Gcu$kk1C?C\7Nj#k?!9h/ +2@H_[K&_CM&5$`iV02"N47oMF:.#91aXGqb($-&6GO\a6bpGc4Op"` +5O.hUg?0=\nlnX$+=s2Ie"Y4Xa4OhZGo"I)$81st,JZkOmaW!9fFtN*Lc@/?:2cis +:?=pN33m`2Nqm[E\Rm/$7E#9%"83&lZTqeW2@/2qT$]fnchqDkJGG6X!:t]7?85cQ +2uYoH),=[F[k7f5pg4<7aSR"X>BNW` +J(,]0:IY+jrKfhV%2$>45Hp[arZafAQ%Ni8=KU+17hl-UAn(oB*@=gfOm16!d!%n, +ff75+G*A)Cc]#5qTj_9d5pr@4a$`:IPkp1c&1E_A\0$/FrrN+)C-f7Fk&7tr*kmm[]u\kob2%IuhK^9tI\"6/:P=-qh>'Wd +mh`n;:PSc81j@ejNkc$O0g>QTnGK'\nP>I+^XbRI[GfMgg'9jm5PWlYn&.1#&=]^u +"e%8k,qlt`PP0k.F=aWr\<^aGQY".&oA]iME()^uRh_?UZ?08q#$iOJWd3"'A4g,o +er8*uTU<<]4h[IG]V"3@8b9,5qcApbQaq"l-_pb6bW^j_Jc@5opHi]J3n:ohrWhArT_?Q6gE3bEs"hppDcsJT8f^,]C*7HkV?heO@DI5;/+A%KeNLt;)5fNI +f;m^l+Sbi;R0RW_+oWRCCPM#mi`eo^R`6a-UX;f/M4!G)R$QF8[[H@Al):JAX4.)&_IW9 +CX2UAYmZ5VML0(bRU? +Ph4/][Oj"8CZYQG_VhG'BY.sSTP3a9Ir8=O]q%6i+W=Y0*aZr+>fcV4A[RM/e-r.@ +r+UIQ]%4i%q\3dpH+?EiK7G%Qqed(^;_Mh5$8M,qf-$=3VBqM_Y"Je<(d9DG7.NQ) +a"RN1*'b\B"*OtF7k,5Q'8O,eaAmad/YJpoU#j]i6C0%PI:<5L::MEPTK8]6H9lZS +"08G`61,O5q2^OE8+?PBL=0N0"8kFrr"E/bQK,H2]G?$IAtOq3CImpB-D<&eiMLna +4ccQAr;YnDhXaP0q7OZG#jbKp**'e(>+:N>YgV5V%#:8(7e#H_Cl4UkCCo!_4Wuu@W2Li`=K2D@pHf:Z;m>5NT6-fBqq;Tl9XiKG7#K#;p +o%Qqc)op(\kIfn8P5Po:-Z:9>?]+To!hP(2a@,?k1O,-T):7^b?q:kr+plX^?4:_I=3fQf"b#-o@Tm%.OKXZ +(/d^Np!]E*1-W)SAH?coAO_sV@c;6*B$uq^1eeUr5(u4gD)j!a]e,\(#^/.i;Zbma)$,VBTD@_,C6PQ7lc,"'9nOI=9n!gE-*>O]DHA$86`eYWrH>_poZGu_`CD(R&7>g1HXdPcG]q\"Kfpmc//2c +*r+WBdl%nG4W%0T**YWChZE]tMQ.bTB`sfU+J,3a"o^T\@p;E?<"Jt9)7XE%]hp_3 +ZXNc4,GYO*S'4gb-DHquiPdOroCU>Q"Q*P6!9X;VdcbT6peLYgnGanX')6r@$T)qK +)n@2^\g(N@a]UY?c6!<=:*_X9Z?afZGig,]Fusn[?Jup*[/%(j@nM1TrsSWR'"DBY +5lB5kgr,(8O#e4D]'X5r#iP/JB=+^r&Z(a1?"I0P`S&N\M*#)'c7^$$2_>ehD^kX3 +WoZ^3NQM9/A!E&2';,@tsGd5SsX+/.9k +W)^NN7L,ca2dPgq;u.sOSFEZR)4cKjLOn'WbDY?nNZ(9&^/"W'7gEB^d;6p +ao.rBpuB(OOk@Dc-kd*fZ.d_:1?S&VZbk^`6I0H#&Ji1]LDE-'A-$2JEmQ0L&8"+O +5?=CH#[+rV9g/p`m8DklcmnThd3`1dOlnQ5o]X[DXU]JOj;+is,Q@#.+o%O@)#fLq +`;e3HTAT1r;00SR[Mg'`N[f@eS^q\T[omD3-@2/DdA@/Rg9b\;S'\At-*+-ld8?Ln +-qbar"6$4ZM*rQW$3*R)^l8hCpLDGI;".i;lVd<$Bc9=3759E?1?JhqK6dk'4,reP +]sLikm8`>[22,@JM7:.9 +l$LsoNuF5dr2`EbZD'rHDj=nQ.Pn'(7(SYQ)9=]XP`-ST-?iD/ +L`Ph,4/r0o.K6a9J`Xh$-&ubqehp+h#6$82ghUa+2LAhqk22'jmDSh/@m%Bnr]N_8 +0)dM#H5A$dW&d=j/MgWfEBWp#4qP6D7UeTl!:YIbY%p/.I3dNtO+n,#WX&OtI&$*( +>1rkfb40UoocK"]^L'S!J'8,lK,<$e/oUa#T^g3WSuYHhmjG'YZg9=&rtMMH0#9D7 +lTjXSY$/M?c/9(a;,r]#N)5s."&\`D\%lGk#&`!%hQ.a/ERB=,jUX1#4g8BV=i$3cJ>5B +K@5U/ilO@&jL/T87m+2G1UAa'434!Vi6Ui_VhkEZj4bHh)lWpInQVu6)_'gj#m#98 +7qo*A\Ak+R>LO$]'bh\2hq-h7Xu*ILDCH\Ikd:C5?\sJY)S<8jbMpsV+j7j[f@'?3$Z5: +L+i']EfM!2V2?:.j$ja\&\cce2ANnm""=5[!2J-Wg;AbQTZ=hOJHb]03-]e"fEEnY +IkMZ?h@;SE.g'2r+H6Bm5@am+M#X5mEY._iUfmd#rqG/fK2s_KIgH%cFfCq#cuD2= +'Li6'\HMgBs-r[TUA0f7s(G;\iI_*Sei81Eao<^bAc;f0gM5ti[@t`=HS(KogsA,u +k[4E3KEXd5T@'gZD]i4EmjKRYNs6ig%T1j^M")e+ZN0%f_u_ZbJOL?(!<*uRR?0^i +q+pX(6TZLM=%nAB7HaaE'!1R4$BWc:bB%BU]UBIGBjk5KZn>;(c.,Rc[,8us"o^lr +Hq1\k/m*BC>(%`%H6Si./mEfS\c#KT)u0G,Z1CApqF\7(HBY]4->3@dE;tp]i3i>< +\,]XHCY?PYK&cSO-Z;XR?59*F.D>=MoC,i)UL-\*IaR$P7=FVX\d-!<*rThC:]@K] +g&J(!"8^"7n6tG0M.V45bgXcg,rj5d!qI0!B]to(.*A:$*taA\k#"B4]P(,]Aor%D +NH2k[b)BFS2k`C:Hr8lebF_#WNS[DN4lK*VA^0RR9Wfb(XJHE$q1aa?H"J_*!)$!% +@!mkgYf5nbU'A&8^"Y9ubp`<=`OAJNDUS]Ts%@>pR/\[101>d>rrouL%J=-8:/e_Y +s/4kg7Ut.!inK5u>#6H"S$gQ1oUT>hp0Q.RHs$aQLOd/-b?WdD"LcX0%:@6r^Jo_: +[!bm4\Rcb@WL9QiTLltYmO=HC51)Gam0m7'oDWFtr*]'nB^*8Lir;8CS`d")Gd@`. ++E4G488.L]IJ?;)co%leCJF:'j5>uaU$G^oIp=*7NGb/XZpJpi95Rm@orN32JH$2\ +mj/C:^(;aRLRBrF]W"'N,T,WGk4Abl%.O;2B_kgf$2d"&0;%AZS]M[Do+k-mQ;85g +?icp+`V`W=InXGFl,3Df_mK/BV4`?57Qudl5E[X@`jS6ueZKCu+0K8pX":lp5iO4D ++"OHtXDi&VN,4f@mmtKBA`J-cM4NG,D;jt"IrL<1]IauRaVkBNnjbUEL/ue%MokD+ +<6m$IYf%KO:%aQe3sQYJIVN^AV7Q>S\b\gm4fY`=?f5oo,O8'6# +Y@4fg`+*phe(Uk$TJgS5]:"QODkOQ'l0Z%irjSko^jiHZ1-?g#Y)j$e&C\rm,7[9] +5XlRk?jZT=Gr3;YK],_:JCFnXfAuX=EkBk8f0kr`[o8XJD34HV/teIk +eHKi%Z+%Bil0]58g?kW>c/R^]b]XDBP5h2L +0`MM:j,5\PMDAWo7.@r>o>`W&onrBTF&KSV:pV;cKpWG0J!MsP%^ZT*pNVT8WDi+3 +_jfnAm?HN9)^>)1U@Rd/jpl9*rG+.U9o/_B]fT)_]stiY*=YTVocN,l9N:n*FN5TM +"T^hi"pk"=_Y-\lG6%Z@Lo"Y?s"qCs[;1S9)N&9KIrFcMZ3'gYoQbo9B4>74U]Aul +psjldk/3 +eI9DR&BbDMh_f3=L>@((k.P6s>qlS50_\q^qS(P`rt50IfK`qDF^q[N@kl4AW;+gI +(9kmH>2hNn&mX0_J]"jWUXZPSmL[>:8!"g(boC7g_Zp1\s-ZgKg%4sWZbZT&P/T61 +g)]G#4iQ]CI,TPEk'>(g=i&6P_=Cq3!q:\=D=QmXD$GR\,l*(93=%hpq2^gALn98P +e*rrG5]M%Ll`O*AToY.k8f5umk0&=2d#RmR'pc2&$7[90Ll&IPY?.W8p.8tX< +\(o,Eb:7]O'm>jRFf83lLb7<)W__RQHiAeB0t9+J1S5*Q8O`1l"!9/ss.AnTAGuc+ +,8L>7s%sM%_8bubG(@&SP++d0$QtjEV-oHQcP6=Vrf`6#J$Y;u2Z2bu5^n=boPW[D +(@_@@!l@9(+<<6>r_-#bYh4Uq&$ZN#pcj`3b9Gn@^'l[lp=jV8Q5krWkVd%b$j*>X +O)d9u)QMhs'(KQQ,ktGS!VfoT4orPCe7)gn)VM&TW1Fc-r5!-VL5l6I/jKG!+<5ra +%f[/^<)_OdBK4oT&`f#AXGZJPa*$.8?.Sd'F*sl7Y1#r/0[MihN-7@t9hU8N%^IIl +3m_gfQD4MhphhZn[["-=,qMOm`t:Au&K61[Z&`Dod+n5ePa:<.7k"T@=WJ7T%OG:B +6Ut_oeDp-U+E++!UFl`Ykg9`6?O1LDQVQ86;fisNg/J5=apNWPDQ(?5UM*n6oPI0J +)lZ2d4s(/Fmt_d?"k+"f4qt2N*p'IeQ#&PrYnbt''m2,?2=8@UOIn0Th._YFPq:Rhd+Qi;Th35SkTDrYNr'>Dt +?@2OLq>>1)a+(?Xr(m3r^KI2E'tiIalKCOmb-#\o5A.'P=/=iOuDjH!5;m)H[!I0T08!IU/KcARd%i3/4!Rrfcu2gZe_) +9_Y-'i^gG&bZd8+SiBI_geAYk:E&(\+.d+jVIDV&Lcu-7/2(8;(qsjeG;%7V'CFY: +Eo!Vcn$*'tBPqo]/>a*U5BYG+)Wi_:/%!#"Q@C8j?p;;;-5Q!a"=on;F96/(j4]Wo +*J;`KhB-Uk+"^dU.I6sUX8AZfrX5_(1L\P#`";aRI8e_iF[baXg>E_#G]_sk'o\nt +MUi9hj[@]E]6EaVXm_.t%tsu!jLpH%In*tKK=oO#';go).&:2(c1CRL-&X>5*Fg"q.#i-+,,;]sR<8hT,^>,5a1B +(4UF10Xd0.$ISiV7fPqis&tk00u4<&:@a!N/uAaedXj8HhD%Kh-`4J9@df73frl0l +^WN\V,.lpne(%gr4#n$>U"1YAmOFC.OVU2u#@]S+0X0IVe5F5r3'N)CL_[-4$938j\4.&k3_G34Q-ZQb88',X +ck(UndEiZY`W>c"A.a/fi1EeGH^HB7"tEck4n/r,#ETd.5^3S*NC^`02SSE60LHWF +8b!$.cNrhM,A=-2LQEEn+TB,$N-b%!hie-@&(j`D&DY)(_nT9F'uBq2b.\#HBrN?)TVgritcadtN:Qoa9Rh%qWt(G%`cYJ$$64qXn] +(9R]9T1QKMqGb0>r-t6Ep*si&`.-_u.fLiVJc'HWO5B">^+VPH=E`kUq476pB-3E% +_#M0+%$1+rK +:-K3=Ik!55b#3/Cn=oN\.'bC=?3[McSPkUd""Hp>8#kXK`^IXZB(KH^pc\WC +*WF,4hnPd>ELa@YR^II)5'<;P]U([dOZ^:B4KuWnbf4k%1LG5?hXeKWF`1)]gG,;% +>CCkGSjW>Sk5RQG&,;3H:]^Ankq6q+222qBO5Wn,"us;"jj\!O)[hV^O.4s&isc%&4L)rrH.*Y>kW$ +_#dm!lMJ^3ghlFHB%)@Am2'_VJJ'C)tX(pRIXoGsQ +/7:<3EL+uQ)P].80tK9qI9R-3*m9%r3d.#J[@K:_`Yeaa3(TMArknU8s5Sq83K&r/ +X67pmqspT\MT'r&nfKg[&+Q9rJ4og\ +\s*8(r^U,leg:;/(3$N'-k(Sf9Fu%c_>GZ-aIjUZpMIo='&q%Xq2^6\9QC#di.%X? +nK4I]aGL]$+'F,V7IhP<^KjD9[8kX/s4]"@`0H#[(')2K,lutEOT,j[ql:j[bU%oA +5!=K^_1,c:s3^WMK)b>t'[##9+>0VCaU"kI(tdR3r9+Jq8c1;EtF9"A=16nglB +$7\NC"7Q>B8GKZ[V70Io0`U*hq2^FrUq62m/d(tKEbWJ+b(=#878T\/i2U-n+3^i- +PrNA$>mnN`'u"TLD.#At+5jo)2KFm:M5!2M4RiB(2gpb%!.$%qK&HZCcGi3P3Y_9S +[eiV&q,Fs&8?i)H^]$^,onp6X,(9-0k5#J5p4rS+O+jM#'Lsf&H`uOXmg>+>Pb'', +.(fAGk4.?Vrr5L')"%G#5sH7TranJG^gA[Ss(6cWpSp_:Q+MCr_t-IFjbgb^6"Lom +Tf@[eT0XpNU(sQF6GQru),gYRO?puaV^^&-"f^"g[-3\&Ru6jjEaG9!(%:9dmcF]A +Luo`aQd0ZLDJ'L:(/5g-Q;7Z7B:6*Hl%o@Dab/n.K.C3V%!e7#o,pnk:\%jLV[Aq:=HIpE'PLX7&\4a=;Z,,PZ@R:17>q%l$( +&H%U.2#D`PaPR'sn,#S0q%"p)pSo9b%g)lXr&>W-\V7Z +'&>61Fb%87:t_J!D!lfe4A?bZ9Uj\q%W*I"D'tAcAk_+(S@b7JGFlL(q>X:&nlW0r +5d:C\5FD#ZP3mTB]-jG%-fmGaQ(*gb[iUcMS%)Kp)[jb4NZH)RBuKk]`0[V$0@s/a!4?@.Pf"W>YdDL2R?n=/Oehs:k<)Ga,( +;aYP!Mn,CbAKCVXIR,`s0U%Zl&I!h@jPcIjW[&n%0JtI9NKKPs"9+Xar`GO/;isa0 +F&F?]NJI=nVqDIPZ^2`qc5d3\%f#(>r.S(cd[faXPK_F,oHCF=pr@7tp9-4+X#(*= +4iC2N&+G&n`jpBD\T=P73%fH20m#\L?]0\Qh[.+`NrPHF:>e?Af9nf6j95P`'rH>? +d>*$6`1mslGcYDDhDIcBo\ZSbMQVYh^%IhTS.!,/F5PB_bNnnjDrq"3E=f+6*;Vg"_4sQ%rp4G>!-UbP"^,Yd\F_hff2T:;!)+"%"Y,1s&14q[di3$a<,BIfMI:+^LF=VcQ1C`8,[$KF\ +Pj!ad/LG_/'7B*prQC%h=]XuJ]6BdueKTB&5>s)b2\ +3]1@tb^YC*ZU6Bf_BhGoScNRPYZ#cuI+]:AEg]]I&4 +8V$gqJ+bjCk*454g=4r;fgj`=O+h?\qub.dSd56Z&YcuNNs4iFKq#9a]C.$Nmt@)\,H-+`qpjc +k^XMG"+R^@Jb]96`9EG.9dmO$d9ogo:`B-s*edMD2P1elsKtEGWLmJTs@gD7.&Dr'a5Z3,F +j.$lG+r51dQk?BenpdA290D'Xk"S.1sf>'7_rP*5,VK%@I%UU46^R=$7il[CBo7P.WgWG3c*H4)r#bt]5+*A3K<#qZ9a[sIqDV:Wi0dH",8etp]2e=32a+n +MZ63K.DHR14r7D;Ok3qlDS9B16W+7:CO3:1IiBn@r;SD,MY]'%$b/ +n6_f,M>s^3KGH.p<3q;gJnQ6aXSJM8i6'D;grT7Tj8oJanbOoq"c=lSeF5&1=?QDW +KJ<4Y=ttf>JC['cb.$$_[CC.sU_-&;:l:2D3O+6r,\1j4f< +J!Z1PQ[Jo6I3q^b"X"W.F3*BL*?$98q'Rr[n#0Wh\@2m?%jLh!-Ie]X&:TGK!6>[k +Rr^:`EUOCqItoKubl8C$In0^(_YbdG(r:\3[X`<3#RU95B8RueJ@LF:Oo377$<]W1 +CDFD^r$1tY"8m'n`e=.08+<,Xr,:1`41$hL&0liZ!3D7nOTM1&;-\_Q8D""5Zq"T` +=,?m:8j91J.;2e>JH&`I$NU+:P^/j0c*4j04M-#([<%ek"2?T&o\s2n>F^7MmdN,] +n1\kK4EKp5@hS2-("sRmd]E\$C5#\DM)mi+Q38erYbq>!.W@:i>,P2 +>LDl^N3t%g\]:e>2sG*c!K\#2o>(D#MVB6DOFAfGqmGeW01rVIkPqhO"[U@.nGb6, +Dq6;N=W[KQ\c8:E&)LuDQ]cY8hrM3$.VW?,X4c`+Bf'Pd9hEp\gpC0PEdUGfHs'oP +;0%U>N=&#M:V0JiH:,ZE#DKVHU:DN-dRiTL*PoUFSE2iFWKZr\Yd6F."gkWgRKicK +aW(E^_u$YTgUeH[]hV!F!:YK%l<0&1=hE5onNZ"tIHZbVYer_3J2$[Oq3U)fiC-I#@N$-_6om$SS**Jq=3.@JK$J.!l,OW2K5]XE8:`6a6b"b!oO8)%Ij@)JDd%N +qh#&[pd6)'?O`N:L$dT)q]O/:BD3Jo%Klb25JH"N]<*W0p\s^>\^TA+*+krs[UdKh +=_7jmgQKdhA7fFcAnGY66d7bMhjJiK^Z_eJ[]CmFSltA^B-#1$,:iKNWN*XUeOEC8 +3e(O)1^&F*+7oLfHN#[CnLS;O-?0@eIhX\aS1o)_>Wbk7*@b-@s$rNC;r,TMB&u9A +F(CkG9PqTCmmrFO[mmdYi`gs&R>GTX(J/6bh#uH$6)Z!/8!%O*e5fIbu49(E`?k[O\#U.=; +eftn,'Dt@[l#a!kjo6.lhsnHBPli5iJ$t:8/dlWp!,h9r/*UUL[-8/Upu5"$+M^2' +5H)sJma>Ce&1%0^I;1/9J*;9?c)H!s;Jf#SK"hV7NWS&,TC:c(r)[>8\.`$O#6*Y: +?m5b^]()E\ir>SI>f,t`+B4`elf*D.-E/9.Y;G(=4Pd1lbnoudbX*KVr21,(-Y*-W +RckJUQO>JtF\J,*P\C6)"l/X?"2aQJ"!:f +(VgVH7]h,u=bAC*&@q?lEqQ>`S_rLTJ(GbAh0AKd/#i[L/,]eq#>RE./DpSpK<0EL +AF:PZ%sRj4fcPo[r#g'n9h_HRWORN,@]tUG2j$G1"!(*BnGP0bWWg1&9Hi@mZeBT(Ha(@TW$F:JO&"7,fK()),n:-1M*aB"f"9.>? +5i^@o6/N=5s"LE]^o)Fm[h.a4r3(RDK+t!:IKGTQ<;K9gpVu"TAV6'cir:_;C@sg7 +Bq"'?0CmjE)-'t6G(;e^45T$b$32=nn1R+9pWN[r]sN=u\CC45%DmQ:J%'W>pY0L\ +f:/92hh#9/fa+u*p^`g%TU35XD[&;XA9[M5#'P[;_#H".aKBWD./lo4LX05G/m]p. +>fJG1PHS60c^uN-T$%q#s$t5.,Lu(AB'i^,EhQi-r)_/bUeFN&b?GB)#fBiJ@BHt/ +1gMq$^*t;#/L.t[1U\esW[BU"$bRTPUYRo8$:/njn%Xn#30H7V#R&7Kkr@<`pDgm8pB67e1m1*.h(5*f8p +>&(]d[GGVjZ<)ca+%on6Xs,m'5V9>^fFE`#N?!NeJOK_A*@YZ;:iQgOaW,hW3>+'] +/+Ua5a$"L6rA.a?-Q;-Z#4f>n`;apM!F-E02oZ0mI751N>#G^%\@66WAL=6E9Ke.C5ECTZPJeQB\g;aW([ +Z>u_IgaGJdf8Eo3^WQ\gJ&Zsj_12/!ru#2=D^E$R%phNR_11N1K`+?-IK9E-FPgPs +:FRFiJ#cq5`OEeFk=boHr'qN!AS*(a)lTD:"/AI1!#AD6!@)!=Bj%rA/7AR1G8>^t +"^qcedYNtgBS5b)+;FQF`53;:UlF\:R)Y=>f&ag(ace5PK4"`/P#fZ02e3r0_WH,= +rDHJX)S]pKfJY.IiH,m.:i7LE:,N#p5CbGbFAJN\qKm?EefJ4+/@DCOYL(TWt,,;4Sb?*r*QnU +J#7e"ISDiuS*^:Ps)Z?!BCj\gnou7YT_u&ApN^_<7qg2#W*Jco,:baB7gbTsNY9*d +(\fBJW(D@DZ0_?',(G,mU55Q(4tG^9Q4quUOu=p;f8jIKBIeH!fWf,V69K]L)aB1L +3/j9^NL8L`%daRKJcG5G0@6198QEh5JF2u`=oZ'9T=UHB4-n8QgS%NS)hW_!+K:cY:;%E-l,.D-Dh)HM/"CDXG39XGnkFLN03X9\V +&R*jA[FCHL[(_,C!:BfA3$3S?B,8#Ao?Q[pb3er.pN\iH4S+:u`T+=_ECSImWueX` +C]UkHTJiqXFD_9U5]X=e);>pf5k)N)4FS2,EumMl&M\WrYju,*E@kN"ra."r?-*nd +<6HYHKgWi=Oa?h"[/B\![C1]*Kc;-d&*6.&5FHYZB7CsIiOb_U#U5c_\+ +C%7b&\=SYtXt6q(d#sq".6se*gRDpr0O'i"9bYp>,GL&eI`%&AHV6K*-0SPLFKZ&+ +2F;_UV&`qEX--e>s/M`)?-isnd29Q;Jm""$'m!QU3+u\S>fQmaqoMjOUp<1!GqZ;, +McSWmn#1\FbB^+pr"*B0Ls:/_Dn6aZs%[n>3;[[CPWu9"f^Tfc*l0 +&+Safk1V5#GsY2u>*s4:E0qIP!u(-Y.81$r87@.tVB?PCBH#2J0X2!ZXFdYOn*H7; +qhkDKpttW!23ShE^n:&#J?R6"^l=M.jUQcaU$;'(0?XBm$9B>@!I+H3?9gUXM:c.u +<1/VtLW]d4#R.`r&!Y(4)S_]sMn;"Ps(a/jOH]^DJ^l2CQNblBSFE^SAQ2i0V$Od< +ZA'!$1DR^?=Ml];,6#4:+Lk`RN@DBg/?;E0VaAa0\3&X+NF3.6Jp[D8!5;*ns&CKk +fE-U',s#J>NcukRPQ.L&pF6^TJ'K)7fu-?&C]TLm'Y;1Oq(`$NM[T;hjeqo`rWDWj +qbH\qn1V(nQOkXSqNZNg5H-Ki^F[@=_Fsu1ROD0J3XB@!G^UQuaO&'88c`2s20\XZ +!WH1NI\E47!0*6$IqXZs"O#U,>#it&9&$)Gks.,6(I,sYsqLPrU7Z\T;'q$a)1a;[.?r-2XpaefW?6* +2h"NWr^4C!2uO>!!WQZ?q#@X;_#b__+P)$>>XO>8S3rAZVeu(fJpOkR$_sOW? +Ct*,e&3!5B*@6.T0D:DB:_F*6X+YCjK%Igoe: +DdE*S=sGu"/jO)p@Pd.2gAaa'em2aZp`D_VG)&t2M1pK6+huHnJ9p-:un0YQf/#1iXYLhAi5q`?,K,m=P$UugD.Kf,>J4'1[ +lVfK&.Y^J8Q%OH(O9e*45Suck7/N&X[(]/QLOVVg55Bh[KtUf9n.,4sp^Zdcd$XUD +`d(`$BS4jCc07A*%Klb*`1oDWd;A.38]9sekk%m)!ZL'^[!%Y?HBGj\PJ$fg^@Y1. +EiiY.@7TZ&[hcPoN#9F!LPM!?@5[Ia::LA^bENFnIu0[b +Lf&Pd#e8fR[[.OOQO5;P(E"o>5ik%MKn8W]dcKi\l(#s-EdC$o=6&"M/:deN,bbG& +jM/r&rcc4$28(^We%p,sBW]*2!QY"7^rq@slC%a/k!@c/%tMGC`:8Z\>u+,5qiHhjjr/@/^g4!,`Q;liM=TK#cp0mju7_ +7JmB*V>5\4jA..sH^eB)eX'[XD^jn1s.u]DR/c8['#agcs%3s,_FOjh'S+gU7<>Si +]CDLA"PrU_7"abh72g`8s2>loaMiX*KeBH)+5X()+4k)k)LB\/F/WqKgrNZ@s.$0= +6.EKaCKM<.XUElbH/0E*rB#V)h4WVADVp"apf';'(-HR`SNILD*V+sdT%g@G6f'6" +J9MSu_#N01=>@h65SP$k'mI3LFh>%a%la@0DQ9[4%/H3YK=Dk-11[M +\+ND2m]VI/nDUR.SR9oo^M=AZ@6+Q"pj_NeJ#&V6@6OE8AU?S7\,K*:mLQRJpL6hB +\Z$*kd_D,Z5E*hYMDI#!8HK+7s3ZugEC)nJ^Of75i*RdVV%EMP,SY%0Rhi"$.!al# +HK-mb;LnT':iq0F-tqh#h07'jAf=-7*CS.=gj::bfbP+e8k1i9]a3Y_g1a1HQuCM3 +V1.4+CA`4nU3HQjIg`d%&ug>JedQ6onDBW8d-@1b"TmE?W/^;#\0!Ii6.YDc6FS/raXe;L1A<=A/j +`PCV]am"&8"TeXRDZT.s +i+X;(78;Whr$1:N.7Pn6$K?P#itA\HL!^Bms)([5TjVD,1]mSFbMof6i;Zs4%"`ua +1;s2/M2N1lW;Zn8s0Z>#;Rpm#?T_;jSEe]ZN9dsMXnTi+C]3sLg?)8X7)'nGadQ5V(m1Jf3.PH+I/UF6D +I%tnuf&jVEQl\YorZ]ei+=SS:lU$P1d;d.I>eA97ceql,.Wod'gEq[ +Wu*/P=To9!NrPbMA.8S$.V=Aq!*#p(W7%uO#j*?D7re'0`ihG0\b"^;C,jM/OoRrb4BAE%7R@k,o +K@kE=&E:IR!^oIl$!9HT3nVE$:miX5o=mj<2.O"cViG-,(QY;g%qiV6cgX8bKYTNpQ*jFKJ@j=Y^fiLVjuqJjHJ(;+-kJO-2(2orA#4cJV]Q>Ulh.&[J(:!f+IVP +iKL1U'.<^?PJG!J_;Jbk&CJXFH;?X5!'COT3!^S[k4c!!\s?H8_!kdib&MIhW)qOE +gWFg1\ogpBOq0S`J4=47lZ)$9iK!qOE[%P)HiZqK$Y[Osp"ubEi@#"TMEpdA +r5s?Bj8l!+s37aqRbuc_?_O/INGiXhIWj5e_:q.K.@Xbs/9BR43,S8gB&N[l"c_`b +1F7hT9.&kf^iN$L!9oR`oQqB?Fo\4"!$D!nq)Wl<4+Do_4Qo^/n@o5c7P:1'+2R +*[EBZW)h3&C[Y.Z5Gdds2BCSAp>0@t+5"X1N1#Y0eGR'=287TZ#AQhgFnU4(MJ:@n +Buri^VOO]FT%&uZ?1(*Sd>YS_N`WWG3_oE9GeY-E]cl]FPB6m"=*^>e6!^EDBi*Ok +Y@U/L1<41#imuBD)J9-;`u.kgs%`eVN!7]4s$=I#DVq/uYP^:sHa?dKh5g1*(F^Y+ +NjL`oG=+Y4+YC=SDbJe,;dq> +-^Q)uo6$ikK9&AqEbHlZ!Y-I*DPH82BQb*H,Q=I#\:B![F#IorL5M7sV7S)8](7"P +?V/@:O?mjGZ2etks&RbU+<:I]s!J0S'e3mM(GL`'qDjWZ?b_C&Wpf`/_#/Dn?cT8! +"7"RiL;7O+b9brB!MA"5,$&rFA_]O[Ya=tB07NiVFCtUi(4s;TiT*5.L]Hote=nOu +Xu$B"cB6>lG7u18<`Gq.Q>2(a=qQ=*#igX6MJqE=GBX&GcET>p_"p"hF.MtGGQ'u! +ru_%)&j^tjU"7S6M]X+7r)B9i$Xj&O+,kWX_#iPI8dGIt92^-lruq/]o%uj86N$R* +KIu-:'EB3^,F/=3cH2cuTn!c@Wrs\!&(0_3,$-Ea(=4M7n:g24Khi:2?i_?\$b)ju)q";i +i45Ii2Lt_?,]!SVIne]#$UF63o'Yt(h*9Eds3Mebh/N;][p/k8_ +3&1#[:tt`\`^^*j@kLl6opu.]J6^F='REj_LD_?/:(VY"G^_+`=Xiuqi#aL8,P85s +3HGL\kOJpM:Hh0a!<:G,VVm;P_:X%[aG";Y?i8rIe60n:-_Q$sN:A_#+-J:kRsYb^Wmt_=9bT*PlJ.ngs$T>rY3R +S"s'qq36j6=,cs>6W,U5.Ne42k^?),l>O#_4L85JACSB"_0(95h!:HelfFLG<%kQB +!IAP:Nb*D"/@,r:#<]$pp^an8K^lR--S^%N'Q3c[$8MR-hr]'rc2m86r.g+X!)E;f +WkaJR"'ZO#Xg>)U\604'ggG>EZFb\fqZ:lF@kW'Hm>2:?4:.K<&"*LuInEnX9h+3Y +A^E#Pe>U]`0U.G<^SQgR-Q*N!''gV+.pt@bWGj +Xd8#`>>e*`>Z4[6;cYm3s)40H),l:>"V\/+1Qn2J'NOiNfKF-V.OEifkR<& +%1UL/::_tHs+^R+!h;4L`d]OpS-&O_(4c4A/\o>TQSAH1Pog2.NrMekIffVB._$C! +2+BJYKZbYg#ds(DWqrl1c=e.mWq5`HV55EoVWM+bp5QJ!rNcC/:\]aWKS7nHC*8,C-ng:OMA)PQLD.:@TsI +ZN/oJ(5@=Ld1>kfr$O&RmYIAPs.'pUE]o$SpfnZuIt:K#!r.BR(NApe=b%ipl,gj'=%VFs$->;QQuk9d4V(VPXp;U*raW-nun;LJ#8fc<3M8c<#47b +b'OthG!@8F^<#e/BR7l]4UEn$kGI,Ic@-@h:5[FR--9C]EY(e["T0W5$D8+0W.a"G +rp*0Oba,.%=tp9%K]<$9\LW:3VnA-"hWUe\oL(HVJGYgm-VhUD8&=D>n;277Qf!S= +?qjPb6.;&fi;`G=Wb()-%8,A)X_!XG1ejqS1'?\?qZ-Y!C)p]3Ob)Wu5VB:XDM;bR +`4KiGhLIq:^G5q2n$6YJ%E/:A^86HGKfu$k-s!g?3cOTTqd,M'2[\_hBTAp^lqU;l^K!]o16b1#EU+I0MYqi/O8'qfONe)";qRqr;E9&9Tl6pKF3-+.,@cfFlr[NXj$>)a)/%Gm0G4=TJH,8!5#P8LbKSPh +=ks5\)dh<6?W_&L>0 +rshqU0/Pcu0gCu#4q4V(8F)/><@1:p1aK]Y4&>13YGF:%<8`7tO^^8r0`2O_oi)b* +1Y(P[`.+d[^jh\UAKR6Ms4o4+Q[o&u$NT;2aa"5N?28.2=4(8&E.]V]`4d$aWCBS2^]gmj!/i6= +M+S1I``greR\c,.3qCja_peqjHZ@s +%4\q&g_ZEM.Q716bJjK:'7t_1!Ta9Yr$O&R$ie&u;$dN[p[6<1(Fu$]_'&Nnr!3!5 +We-.o_'LSJ!l,(@_4A^1H"Hqoh!DRfnM4Y;4DOL:B)F?!(APX1C#:&`DGk4-)(h@aBohGUZ2*Lq0m(0\!m&g7:81+pYRA8haam-L9EnCEYUKR-:.;Hi;XNG?SZ^? +4EM8mE@_tlQ-:5*7P1WIS]>B+^EW<2? +CId0d-Q1PG*_(32,m,+G?HiG7TJj#Xe;g$IZcMQfjCOKS4j#=.[fS)\p^^L-^q\dA +(q77L=k.:E+p$qNHJQcoMPgYTiCe%]UBK=qOTePA[6?$^IcaO(D9e%bdt9@9pQFUP +MA#n):ObSSm9+8FP8L>j6j5TG.lt!`.C;W5Fm+5n3\hb&Q*1Gf;h4]%"cs>3j1.5N +Cb6OB5=:/"T;?PE5>+Ge"j#TCSbui\!/Omqh#b;E5FDlTLAqb6=8X`L)ZEh=ItAD/ +f67!D>CG:YIi&-#$qqZ+hn_P8)qA*tfj+J<"'f6S)`n)nmejhP5F^=!K:_3Hr4DJ` +!;MVqdhCAX.Q;obR*jFJiEBi4s#l'J;!@%Lji:#_k#fmMX +$g/$&Ts0bF45Lb[!/!`B;CqL-[]ojstlN.%3Ld^B/13k2a4UEIh)G;7QT*?+A%%TSI +NplN7G'I]am6%Eair=#5>2]G!bAtgV*+EV"ZZbO-H^#WjE,2GHE[Rs +da6hoLu+8]F`Zn=[%!<$"n9IhE@U62#LPa)^RlA6#T>(U!rSrE'js;r!oRhprh)80 +ir9)T$`e',=T%"8%313pOUm1B+,3Q3L/+J7D]ocQ&MlYpf0@C/53b$ikVdtkW;'0X +s%p7PC>-D/DI/f6hJ&9?lh>n7SR(!Z=*(rVoS06-37Q0s)NQ4>Vaie/l; +32g>U?9^`TS#/,rk#m[BN*F-K5+GL?DVb0QXJ +,;'ct,094*JH)83T2t_Eq2$ZR1O&@u1.VX=qI>'Bu0A +U`cc:j<-SN`3/#!s"X>??3'^f"+8T8?Q"MK!6\pS761Ij`IB95c%7]0pfCSnJ&6[o +pcG#8V`W*ZJA9oni1FX-&b(#8her#/2e'HHK^SjJs%?%7_#H0B0CZ_Ks"47jr/\I# +r$VK1-ub@u%ab[$ao;F="Q\QlM,NX"J$`oF]PLNrHR5)j#oPmambNleKtl`ua9@Cq +=")V"N*;$RQYG9FbeY;h04Y*1!BgF4X@$%W$"q%R"ZkcT+$rjJRRS^`OTr%:)(,T",+qo)aHRO>;3Jbl&rXs!_LCVsor^OXm'`[XLj(L%lb-s.>_n +d,.k#^WC^!VV/_D>F]/'QT\1"n&M<]&$\9`UcfAq+,6XVOJR2X9oo+)7kD^_1-@n] +:H6NNd1@F#n7ZjVQ^6fGiq\00W.a"p@26ho8`9WhfM.2%dIFT*q2=11F4%LUs(!o'S$[?LU?Erc`CR=KVA5C+X[OWS_HVUpBr)"ithJ>\A$'LCeE +ls0m*5JWm&nd]cOO5I@\DS\\T5e-e(nDE7:*X.V"PAq0j_#L16.H(6:D#@T"lqb`= +mh6MpCXt]72i7CW4JVu?ntf:Pk+uYh3c4c6\.&<8Obj%`Ze+BhRMn';VNf_F913o; +@Z<7V5!>\j;q.\#.j#C9H2``/p<]=AAHM?W?\eFFcl`^H)^>9* +kJ&,$5@>SN*t.#!U%!hQ2R*.?rrDk2\URX8$*GoIis3sVE[k+51*nV0/i:#Yi)B<]4qucuhcnX-+)GGTV`6:0H:bu4\ons=& +1K'#_J->oL\U%Vp%ciV'Z:0CI^QSqNma +K5gbW>5uS?0lU&f%"F@+_f;UWn^]/K`!>Q=$]1$P--f@\E`BlcC0A83UkkuL<2a?p +0tQ>"K'dN-p;3(#V#X#Z!s_]`#EdJ>U1$=')RF"?d]c2YfVQ&!U*X2heeP?7X=c2Y +I:bG-s&q+",Fl7L)EA6H4:LF8k)S4"cb5BA^'S0E'XRa2_Z#>3NSEq;d$8/0=uuO5 +Gj[r_i]H\SQ!b6s:,r2LMV:Ao*jt[!I.qj8_/IKP^T.nEORN43\GuS.(]R9Pr_LGR +625E\'8-6\]^;<7EaP:20KeBEW:`H8j[!R[LEOQgAUY1up''Hg3INOV?P5Q*'nL8;I7Q0ij"MH(*-.g#59?H8"Gk>6! +GniAe,+^n4NZr\/9qFnqJH&-&@>IGjkcL%tFVA(=PMW\TrllD+Y'"L6P$d5rX`Ze* +H\pD_Js_%=h@.4bSdVmKD$S;1ZV^N#,bdB="K?-uec::"d+dQ"5I;5KfXDhB#=-6G +C;hI#Egg/p:Tmu7[B-\rel_=Or#Tqe,H)7g%:Jr^'9OC*hFD]!=u@c7;/9H#-q:k, +-iD+78JV"-heeZc$^Lg##[oO%UbscNL!,F@77uSe]5rVX^;i[]s'sFn2J]go8Z0me +Vn`Eg6Df;5+6"&SU1Sn>YQ(ENpoN*Ne=W^QO6,t7o$*5a_bZ$#>Q+,>rRtoErZ<"* +2d'(c[erU)>]iY49hJl+H<@R,Z7[H3`Mi.P=b[T1-6nCK_csPp?'&s^'.&F.OTX:P +?)'YN".5?"@@ +N3bX&[!G,*r3-!u#aFTdT)LPnm;Zt=PITslae6e3L5pec^:;Bj7AN:8tooE;R.&03t\j7 +_gZJY4XZ'jccJqZjiGiD0KbpQG@h1MabYQJ;%4gY91?$PN@Y(A5`(4rVP]Vm.%gnV +Pn9L:4JKM^!<7's%6ddgTiLEW20rH.JR3,bJEfk9`Y:nK-(r6"!oY)7"O61I:fA9C +!R+^T__pMU3e^^*kcWXWWj&cZ/=c8a7Y#[]PSsYo&7dood]PpC-?Bn&TU]OnG#Ekf +@%+0A7Xd*7i1H@Obom3*OHY?N?/WI'V[6`#?P`hHrYOf$M[^Zhb--NbMYUQSGQ&C_ +^@:#g$?3l>5g)$X]Iue@Mj:45+lq_t=9',B!/!445bZT+aQS?R?OY.+id]1%d!t#h +X$qbi*A:d;c;t&H;--"F'nY@.f<4,rWeIu58Y0,N:rsgl-rhJj[mb"#+pp4qqi?Fe +kPV;Pfu.%SaEcdhEQm4@FDQli]tnGfV_nMT[jWhT[.]-fSo$so(;11P^WZ!#V0@Lp +S9'Gcrl;,>,Ci"*i.s16-CB=n)XRT`6'Hf%A..R`g$=_gn:fVYKj34mnGg09+nppW +HHmDA&PRT[?31N09tp@kErT4[cJMUu_#IbN\,VS4N;:6*D_JVuaM&`*_8#Y/8;9VO +ikJ!:`?W=l:mCg<$UPT%#9q%fRJCQMbpt9')K'*nbq+U;7kkBC7ZlS;7,sPInm1V! +a3^)IZIR^f70#O86ASsi-%Fo@4E2nl6*-8ET8BI@6t;&QdKqT2blsRfpI7(E,=19'JM^>YC]"[P(ICg($$'uS1aU*f]%(WZ!&<9!4G8m +%i>QBca"IdR<9)=,9pSRiZ8U$N5ihek&K96%"HV4V0\/!#=>R*ci>^$?4pGGB"!;t +&.K^B\(P&BquJK5Pk]nsVD\t:X3c-6ecQtXNaIQ;"8j<,!Y5H_ODipQQd.J[B<%49 +jmFMHhL"Yd[:UKTs%1!*PP*udF*c8t=0%C`8WB`FNNKL>2VMVEb_gSOIPkMcm#q$p^\tWn=Ok0o +U%l8Y[2e1CaRO=F,daCTc#fn_%Q,\2m?=U@ajZ?%s+P.(ceZ$*2$F@sILL_!G@iTj +.g[X\d3%;=OT5p?0`U+!q-U;$K)Rk3/(KIU +XPNYoHH=#WVX@Y&9*3h3hV.Y5J;Irk!oI!W4b6kSX<0 +nXSrj%^poS`34j]!;+NkGsnJ7V#UYu$T'uis$@_2oAA6s#;plrSH"?DmdMl%AV#`` +!Pj2,#+Ptu!WG)?$=nP;)F$PP+S\[P;1#k!h$ke5n/o=$ +Lu!7T=TH/$1iNHi&+TZ;G8U+)'JosF]T#2*^-4u(gk7:c80RPr#QDK%c/2')Jc@\5 +!9tW3LG"5!hhu,3b8/Jk0]i9ennlJJ/Gra4L#n(NC[s0A00`2A#*!Urmu +(CVpXq&moT.)bsYC@RBf4XAqeTG.p1d6jT$i1$<2/`*0ZWlu5$Y]AD4!@:b,],K4E +Xdjb\9=,`I?Fc3Bj3=X&]TJDniu_ +5>S8HVpkbI8%_G;*C.Z`=O"?25JsQ'rf?juJ>7#[S1_Y[3nN&%CRfQe%5_W=TY\VI +;/uhD#RG*omuJDabgPe(63Fc=b.eQ_"=!kkn-]Mq,?O +fR5:j65sBdVZ?VG&W;ahJWBp5i>hn+!<*)u0-b+I@cjk(kASnOW:L*nJ84,8C@?dq +o2`2:URQOIgA'';7aup"#7TisRSY,VNA\;`'\R:nm*t6Sj>g0/Q#gAnX0U$'Ld?hX +:*YLR_Kts/HOaS7LU,l+knlI0-LHH\+-I=_SH/[9:)gk.`P?Z0$L`rLrN#G-We!)t +>qaVl!<94bR2o'Z3aQ@i]ZKL0[(!foC3brpn15$#s2I!c&G`g3jtutse5A)\hAXj+ +&lQK*[,$d7IiJjc2uaEj="o4_80jhGA3WkCa<12Ls%6Sf6rW2]*/Z<2;]/6i>_eo8 +B7tMjCff*V'bn:&5=ah)V=oYYIDgU=%0Cu0-i[1Gbl96)5JSAFT92A_7!:&B"r9I[ +Juo%V:B5S=@K&@[!hqr%[oqp!rYK7IGPjsBT0)g8(eI(1TVGNP:n7iM!7uT0ruq/= +!WT,A(]T'aTE[1moA?A[CLFKnW?thWUQsY#1jO6r$G/d@'.X.N]F6j5L'"@r&!ciiZ0$76d$C&s?V5,V>2]Yi.)a841 +P\%#8-B:Ctp-QRsVN4%j+;3EFFRO632An+4Pf>:I'9F=2O/kN9gqIjgn-N`8PLQm2 +m5SSI>QbL',&e/SGg%00Y$Qn/'&.ldi]q0UrprNRQsW"`!G#:Bg6;L/PWO +C_-Y!6[V'b"R7eMRTXCM4M>Vi!7cSM"55[o*>&7G,lfk"4q$^(&;:!i!GIr6[0h.< +:9FflK>:j05&ece-/_"8$'5g,$(>kODMO`nK@L!bs#.K%O5#"2*k;QEK(K:$eq:a> +JH';L/B%RN1VeK2T\dX@hBjdZWtnJG$V-)>9@`G8ota/k`d@+>X^V(4A`m"Wq+'38 +?J9Y%3i,k!JWA]Y,ML]=DP_e-.=3\MC+Z`>;q?.(4E_c0FW*/Y*&_aIhRt#^FVrPN +,U8>@n\u^Ha6+K$]#jDIj`R:V&A:Mln<8H['A4-!Dtk`,^O&X($%srdc:;a;jdTP8Q!WHkr!<+8_b^9a@)HN]+Ng`N@s,!0Tap?L=BCP@*s#1L(1btLnba&rDCFZsL +_JVu"'S*]u+uKu7ro5kFC]QDm9K*>ej0Cnh4r-?R +#Mh@D!'0j^@/`+aCWVtDR49:T[aHNriU*dQPo!Jmg)F6]U#$U\%k.MB)ucU2?sbBJ +jF9Dol^A4ZV*u$^n%gQ+aBud0p-h?hlKHPUshl[N4LRXUh +WfNm/08!opLWnS-O.V+:+,5?dO$^k[,m/LGDQnXCG!#18;Lg&kr0KTG5b>,UQ]&Rj51T,^Ei4F6[Ntk]iamIRf.Z/m>$L^++dF? +OBR,>[XBg84VHBgMJXE.2NKQ8I@92Jf_U\JIEYE6bV=G5%3?_BF'4r-8/=RUc/n$U +(e^R1=34s];0+rlXFHHe^D"qeT3=9u,FZ:)9n7coVrBF1!eSL#1@;.Q"]#d/+$C71 +Zd0SlY'"=mn/pT'6>bIgp$OTIf?(.!oX5JMl2h*NrsH(ILqQCpJj=!V.(^4'="o4p +@Yc]V,tr)V@5^l*Q\,IW,7rB-c1'Z4791W_eBi!Y[SKp[`;C1=KAZrS!J!Bq@Y"4a +;)*t._#bbY#P+WaTqj.Xr(dl>$G_/\&S*0js3ZRNe39UH-)b=+LCQ1OBA$`;(H^[g +JD"4_:':ML6K^UHZ5I*X@.IP\$'3t3c!CDMitpT#0&J"ZS?(e4d*0OqCX)ikh=6Pp +NVC"ui;Y1jc3MPWLTIiTPhoHOVU65dV%9cX`XTmu9ICQ--If,M2/gZacGHH^,N^0$ +3"MQ>#p"cQ@m@=2ir9,Fi4aGH?SK.7lWPOY42=MF'(g\^i!rC&4C/8uRj"u0J_0^j +_)DObr7s$3\mOWB]ZLGd]uk5FP]Cg#a_s<7.Y=p'gRpU?0Vlfa!2Tm@Y(MThAq^8CWqPM'9=: +lhh`;;LtA'OOGsbIVB_(%(00>S6VH&rqFZ"\=ScWO5]Q1[!3Lb!+5gF8JEB2pW*3P +*^Nm.1'PPnIh;YY61AXib#8"`UE$*H(:Ja@ +X`&d9LTgd9@`SWChB`@J/A1uZB?_OtSGu^Q$A/;85[l,eg'!+QaX2'7noG0P\HVLh +NeAO%+t#$(&C_j]XMiY[`/J#Vbe+IS]a>cC#ou,6*B=BG3t7fM@@4:^:tWN)e@-IF +Ysh/u_2o&kZF`BIK`FT_+gYt?[pdgnI`K4jN&(_TE6.<-jpk^R;1"(ZiQK!UD8?t^ +Ij(Dn%\CD"bsKL2p4%hBIu[W_]]18NoS>;N`ZMsYc2AKoq)?hfU]A01ck-8b*CtHt +)ED$pd$"'-*:Q2c*YIdl$iEG`EfXjY+A@,!P4gR\m=);,^.bG@!WV$\:BUX#q8`UiK]DR-_#bbG +7sgt@1[ouLM%fs\Jn*cdJRE/VdCr6A!;J5D2m6'is*#;cX/+lp.UkRIohQ&__pdS% +4/r=b^]K$rpja4H$c)t,BO5=/"8kHH%\T$&oHju*i'1[.\S@^6K1GHL"a99Rd,H^I +5sQDE5Yo9("]848o-Gb4BE/KN,GfRH%I(P+tnCP.m:V +_)ICJ\,!]jE]Th(JC%_;n,0=I\]rGVRU0Mif\c%`h<^EUifi3WDf6e1=2.K^H(qS$ +9?M^JJElQR<]audd]gr/g#R86q@E';!cgt`J2Z*q +b!pH*Ja1&NBqJ!3s3hnKM&iDM&KnGW91/WLnh$IYeZ@,e[TI3_G)P+3ArpF6ZI&$1 +,u,C3L>^$^N-a!%g:nOH/mt>p?G%KgpSNR55QLh,62*JD's*4.5Fn;9%`i+0s#C"m +$pb!O:s4lT`HP0]$_J)6DIV`]UM"KXQYtpNM#YJS1[PNQr)IZactn3]DZ+8*pc2jY +A,>'&gnWo:fl;:d;e19(A5%31F\#M]'Ku`9>mIZdo8'ZpE6uNt"D"(p(Xfi%)/9lo +NcQb@fTl)H$JJPUNU,A[/UfYJ+5S_bdR9V-S#+\1q,`UPPXl7i*p.A]]``2k.G2@'HfQ!Q6&hAeV.8] +!T=94mNc%oG'0Oc]gN2m6b>K=cH:XE +RTcJ[JEeu@HL2OJ].C`t&ZReG.>?FT0u#a,/,i`iFf7S/"1O,c0JT+-iXCY@pqTt) +Q=F`d+93?^OTuQgGrg0+qfWST*TCEU46Du-$Ws/-^r]a/O +6NKX@!3gu?"hu0_<0/X8roY["+9a0Pk7kl='uoeOA,(J"gKJ.JT![s:2fpX>]FLF@ +#TH!oNLb_r?:QXKs5C/":Gd;?*Bt/F+*)Z.rk%M,X;t@)DF!AKZP2>Jck`Tp_ma'H +epbL(4lc0An;"0"kW&(2?ka/u)`e5/1qH;W+o$g,!;I(+Y^fO.BJ'!K3dW$?6jpqA +Ma&M(HqmfD'MiNM;B5\93]N.iWIC:jq/?1+@_"-b]-!Jj0@]P8eD_'l8jJF?T_+"k +F:WB&LQkmQ#e#&8XH,au%sb#WJZ2H@=`7lhN81JDiS1TY:XXO4s'*V;NHbO"s$8bo +/"X\rof46&[92#Q?B<&G^nWMr"m^PkI70f/C$bQemSZ +s&jj$%M"G*&]dab7'"*ks*qg;7-5np%;Z9.Dm7s(\a;2%./o"#*r*LeC.G>06(Pg# +"\/Fl[2/O%j'/o`DqY?'A[\eKgV>6WG(inLpkoA8'ZX&U1WB=U44\<(0Q00mSkra-lPCJ*^T+LTP6pKMb?g+]f_Y`4J&JB!K)bi%\NMUO:_0C(W:GRBHQ-:eiRQe2Y7(P` +8oFGtPPmdfJVTulRAT2Q1?W(Nm\>$:VZ.3Aq("iIZ:p<-a0UH3-ZM/-@K?C]c]T<0 +&H;e1'lT%k(]+&Ba^m-dKb'Gco\>RRHVKI+.%-='oRfs[BUkhA!L]L*Fco;+M.BFX +Jq"9i58XD9"?.GGHs.Nc/,7/HCbaGH=BZWQ5TCZ^r9j&%(4uMRc3OFM8ONpDO,Eru +Ee3'ho,k#4:J?hgUI0tCs,H)3'9*R_56ql&GQ([C-rhaH:lU>@r,9s0^Z%Id!Nq;V +"54b`a8C.\+nZ?EWT_hsXVpZDb!aHs,[*[(U%)Olc=L?7'B&=_p;I+.]D1j#JBS-# +[K'Mgh&8_7CK_,k/LKCkZ/+N5"lMea`IEYtL\[!`s*S15!B.R0)Vfrd:WZ]DeF9rsuu\RXt0\JH[G' +$^4H(+"Y3i`.NTT-jd`)-ph#W*``pl59!qC&`*_ADYl_PkG#5^&G1ag0-3P2J,&AU +kesWC@%0B"-lP5\N?&03*q=QcR&pqt+)nj.!l,D.fKBfGn'aAlW,3+Gmo!?83W6aC +Gjq5Y(4o>-,M/)->jVR_UO1\hE/Y0Qa=T\T[7P9.I>t$>1 +PBMCMgA>$XG8;uAs"_LcQRM1u8Do!_]jT%X_.c,e=<9l-h@E3NJ7tRAp,E=\!:^!h +A-X8c!T4@M:INdS]MoE5je9cj4hof'jWpSp-=e+ks'E5)?9EQZR_4n\]pDEB%S$DQ +Y([O%_GR&>h8DB/UL9;fi1eSXD#.GY%(o4P[!%!"]Qa=EdYZh5?gm'aCucQoAQl:2h@D2 +:W\Le#A(r<1&8-gBGke(&AULfau:W`0/\o'M=S,FP>`_)/Q$&gJPT\^^iJU&&cX)B +a\hha"X*e[rS@9VCnR"0WMOnJj8a;^MZ4!6$NU+00_9//9[MTYb9[u)?+BstIu^%gVrZO] +6$>+E:Wp;,rrX5f(?Rh.Ijm#II,R<]:9:WNJ<&AJ\G2+GIK9BSpnLdE:B(+mq9ouS +5j8S=G8B,'"2?W!5D&oaq&99\NF.g&<35aGd=HL?ap_69F:c*-88aPSm`Y!N^CC]d +,=[1e(2.r#WL3^a`Le?Y313j0K\@`GREgZ5ie5+Q-t-+h<*P$$1pM9]Y&u+tpJ<_6,@3S?h/N8jJ&^?^ +3R2bOn%>X\gjQV.BRrIT/801,`@SF$8IR1_AnK.-AnGXeZ6Y>Zj50['S'4SRPF>+@ +-<$+s8u:[RQtLhcX0'(]\MI1We5r9sX]DK6s.6XLqn?hHo#J1%Hc#J.?IQL5gJ,#o +6`Wtt*uP*^7h,X!/gGm2""=ER,2th#:Bpp5k!4CbT320g#T:IDGj\5?l>L+cIoYk8 +-2[c@s5:Z=oHG7P:K<`.4Ru$hlj:P,4n;0B'8->,)X\_tR^Q.`[S*'T#PcWe'CgR@ +(8V3a0F1BMquaDHXjB57p+)cWO7$!IqNEP'ZY3S[ +[a=/4CF/n8MKZ*Pp)SB\WoS,O`diJJ&GR$"s-"5kXmQcNE>p.TS4?$3"RG.n:bPQSD\es0n6# +X8&Me.3$%a*I^PH?h/-91J+N&M4O4?h@=?`OEDS$LC_)=cTLGG*V#a@'n*^.0QH)6 +E?ZGanEf%Um=KN:H8hC@f'L$=A$QW,/XO7W62k;F1'me"H6+YncZ<93 +1Rdtl\7%G_%Jo>6#2(T3W9dmg4K2FE1/BL3:c#1P-NYcH&dm`7#A.cHANZ8jW/8@g +>\u^FJ)',N"W"O[]s87Y$c[R"r5D/tfs",WJM!">VrZhCIs,dAqN](AKs%/_$O<_]4_ZCDL +6a,NAbQTE]:U0urg$@80iENPV[?fo/68m]j#QJ`:r;9[m-.tQ-"dfIlZIq`KR35PN4UhJQ +k.[+9K)b0[J6cT>YU%;i._tkN>i$o$)O&c)"nf"@g=JL"p-g3QNW]1PIC/YfTs*YO +HdUoB2sp6dmk_]<2SqV$in-rs,\+:Q\0I55P42 +@6tFUh]I#ob)HS$d]"U=)?VoKMT3INQN']hl^-@+miGJct%=n/J&Z# +g:'oiA@m$I/;WI:9OB$T^k8tF"r]r2$fJdP0Ank7;lD4/TE3qZbIJEI_2:JCillW+gj9+UjdDA$i?joN\I7jd +r(EffK7L^YFPjQWgTtZoe@F\l7Amp@q_kOfn0-"[2W42F&'j)t@TYhdU(l4qk9[gf +$@6%np)+S:(`%)HZZGH?;o&EV6^jChC66n\OSS.&6QM,n0DUS+,IJVBPd,u552 +r!P?6$FPHnZibC!jkr@'HZTA+I#su2!:C5:Jr@,O&82Rj.)>F3rP8G"]ng.rYZerC +$GGqPqm>5,R.:esNoEK>kc;QsgHpC=St@/m]'J3Q!ZlsJg<]2>^4LBsTT?jju(kUunaUm>^mlb!)p>74j:cOQW_ ++?/'cnUEhM`W+*?J%ToDJk\f\=8#4j7@\idnlP?'n5o/[p[LI$0@$(=+PN>Ds4lQt +46=NX%rkg!:%'XmE%5tQ,@Dq5#G2T/O`P28oo^'!*ZL^8Kd +[^`i-S8r[t\TP\bV#7-RfhKN'OYYPOq5'*3HfKl`Wms2\j="k`Jc]VjSaP\doiA1QD +J:7s[r+EtlZ@L,K`@g2FoUhrDC/P$rnW?>1RWlmV\k,cn:/UaXI4Ruho>oGPbg[9% +CUi?GplDIJ8Iq@b*s\sZ2.-2T?+J:1FC2OsSS"qN=74P0DW?5ds's*C@n1&^'`=T7 +N.V(2hU@;\:UiI<[Un'rC"&$WgSsAD.fSlB3ro1aQQEpL_EAIDjjMCRrgRN$RM_k; +X]cSAGO2qcS3"H87/2<4RVfZos54qX5AaF[@8sW[W8Fgnh6q%PYf(Td!WO`A=qZMm +YTERUM\1R:EK`Gr3*q0I:N/W9JB-@4!ZFGDYK%`/>l,t][VaoFm+Z[HgU->"Z0FN[ +iD/m!7R-ZA?22^:Me>>)Y4=AEobm#\:!h=2d,K3Cb*?H/;r$e`gs1]=NJ"H8gjDoS +rgpE=SM7encoYoB>5p^^oW`nLA^OiBkkfma,LEl\ +:#R>1PY>FT-c,g7YO5-2Y&,gQo8=\%0ufN`"!?GPG?+d;`Z5$GkfX+]CMk&5[>e*nWZm9*K!EFFo";%)j$qSBB9(]d84+BBM78&,9>q%me$'P^5JFVX`KO[W&5 +WXQefFVP@uZsJ$tlVLGrYQ;VOU(H=6@Go>A&;54+-HY8^iI@L<,PJSEiD7T*[oMr_44KG0TCG-K-;9U%OoMBnX)sa=j8WE[Vu$?o#l(Y3 +I#%`UdC)Iu^Z3(LPE1n,[kW@nWAIVrpogt'[#IkCmY +B+BB;LkDTd`5`nX0JABj.#PA[NY7XG>B9$S?^FM\*VbRBS-loP7a24 +moVCg=LTMS5'@u.EZ/IM5pA/FmH8+RoCa!BYC$^hrGlB,8)K=npNs +j0)m<(\7Ci=%+!VP,(=*+SZ38;]kf\htKfUM#b3HO_;f]NslIN=?qV,>Ip\IL8K=/ +L56HC%55L=Nf^!t<21DCmG3%>S%\>spuJ]%mRD+!=iiP8]B:BtJ++P5J+#,opn'U- +IrLk[r*]Vm*`W&-,"'r>4FCPg^LGbVrSl)Oo@L#=&**](>9=F$.)"Th\a@t[k1DD, +gH#RkfsIj5G6%*is0b9RmAdR4SM[L>iOu+BA%N5D)o<%4%J8WLBp77>rk!==(2hBn +Hc)FY!9'3rH73>3gjCj-F9:tsUHQ]%3dnG)?%[G]loU>=bRDX67d+/S"kD- +^/LBT@ir\M+([u!7ur$e[QOGr]a"2C8'%:gBUci8YaI@IAbH(;ZT$pJC*;X?K8*t9 +#,U-/$9OjB[lhiNI[\a\1pbFLLT0NP`p4uYFkZA.3cR%21$+j]asU%t3YC;"SpCB* +nlilf5N#oD,cF+l8ij!ITCF")WXCtUAo%1(46VCXBk@-lKt:]:*Q;YSJnf5Cm3u&Sr,g;9*AjgUU]s5`ggT9,#iTXR7sI,Xq![\o!qt;Aeh-e; +s-'7E9J"\KOsQFP5N70Y!iV"VJ:EI33_fX\1YQ(FNB8cZsO. +02KX=$KNk`ZnBV-@MWpP"^'T@!*47AjKV^8S1%]eK_aI#%jsRm-oS9=Xh#.BSbsb2[:g@VhEfs)AN"lj*HFs%5:-Q,br;7(Q>d3*1bHs1'R4 +2C7>58-(1BghJ.@3o:)>n(J9o +4,ohYNa58tcfJ*'dMWUH;?M^B/O9bLi6\/6r*SJCJc>s5<9c]Pl?mO)F(3aJADMst +J&Cp7(;CIOs,Kf4U?+b_JJ6X4nGdC)J*joTEIFN^=1;o9XRk;ECP +E>ao3ZoI5'\_(&\tlQA%td6 +B20*fc?(7\Ft;AaIq\,,)pO:D1N,_[%rLf>)?5g#+?ie3fn,Qh?!U#2-@;#5lACcd'`>hZYIql +rpMgjj)6lI]3)AC871,ZB[o"XhQUIIpWsELirpS++'7qRNein0SrsL2hG'tg-"og5$T_&`)%s`PY1erHk:>@HTut]j=WTLC:qIc2 +F;Q]U/=CeP21"7:[,KQMlT'8:ic_t]Nh(ET*kRL#4,t@C%"Ia`^6";D8]OROdjMlY +V]OkG0)^,t?2N4/HT""mra>isJplt1/bA*f@6lu?E99=? +occ'd-X#/OT;2^dA2UQdqlg'-n3>s,:\k1^[/40DpH?qGH"O"%K)Z_RDG[p5CR1hg +.4t:Y7ulEBguH1Y`1NZL.9+#sbE[i^;3JH+D* +V+Cgii'2Z+/isp(&9KP;U*Tg/T@JG\aWUQfT)jF6!>,1@p)s@lFRNMBD[6;TXdb[E +g0*B_h524_s13(=Lg_`GL01#$r6r:Be;#DfK%S@S;l6t/B!59hNLR@ni;Xi=-u2*; +%uVTDEUDY(mY,F-gQ6D+=LQK_>;kiZftLBi_9<(c[jQt=]3,n-0CTphm(;srq^Ebo +`IhHs_R:JcDXd\Y[f=d$m"HU%ZLGtcktF!`kqdhJ +1NVeBF"*M@(QY0`NkG*)gir);F6u-](uBe!'p]\D9Sr) +fUr9uJ4sUkYcRF6K)`\?UXCdTP)`V_m7'O(qWWP1CTVi[?8oFZBhH'f;QWu!Q"C0o +?`"QEVA!:9)\2l1C'?n&gZJY8r$Od5gd2"FcNO#1)="LIm=7P%d'DgH'a@Mm04P@^ +!cmqoq+j?cocf=d,*39k4(S'h'YOk_.1*B&%sa`gC6BZP,_4giGWe(s"#C& +a+WpPT/D+kc4>c?gmh`@MKB)(q$6pM12KS2Q17)A`G*2`QY\I-MBY1Xi'%0ZSRX]-d-Z7_0G@4g_71"7CC +"b3I3etFG9\N8:J!qr\T>a[)%?"!bP'Mbl_.YUfHKMuldD/l@/a+("2_8/KI`Co1' +S,P5@P/RrpKe(:+\QAf^A/tV@OmF1#NuYb)*K,X_>_Wc[BpdMW9AbS-s+MO8pAZ8+ +485>Vnf%>^1'.hckI8;0=b#r3bqI[dF4s:L$C$1P=uU +U1qRA?Y/PiT(=N4B/a29X!%FDaVsdTWI`C^\bj=1qrB,c\SjZ)^WKSP6mM(pZbc[+ +T=km+]@.D.4V-QP".tPLhGF%N:XVUkF>XSp!eC;bCirh^ +p9ecE]9c5rk)C:@#TQt@!r`E_4c%<*+IY4g3-I"Tms#r+,Ydas0$KbI_gO33+ku2i;^28 +D>F>6Dre:g'Cc#9Y]Aj?F=s&XRIC,ri;]_=bqJTdr:uS)E#uK0'K'3oV`pm-d]"njH$nbPrX5?G([4JupXGIMk!D-)hKSb1 +Z[^l7n&Tq8fPLm#&aMa'\J=N*TuCC)-9CO*>WZXt2eDiSh=Nob)gkfArM$h#=[A*g +q2"N7[qT!9/q\Z5gF%gtr6ni@V`!ZaT%mqN3'b@Gj@oUc4P*n0r_t +`&7[n4es<181g6-+8KG^8FpqTMP#Th)b*gG`DesX.aejKf9>=\KbZaDJ-G$"#*p)p +oZJc=6P<#i"U/?2"_#F(ITFERpA'0Wm*mLt!3([l4`_&k[h)6Q?5!$ZhNK)#p!-Yl8Z2rWjS'chNRZ^R1Ke8am7c](c4ds&C\EjFP@7dgAmV'@;3kfsP%IX_DF`];(kEtHa*cW.=AP/EmFAaa6VVF]82fCU;3WYV:QKSOH^qU6 +Q(g#^f!RWkSuiC9#c_B>l=*.=0qQoMmdL:R"4^6Y'aji,=P'iL.fPZ&3L9r,h?AR. +7F\CBFra,.cQuNAZIocZR7$f"?GJ*Y4(S(u5#^W.1[dcmH]A3?%fM/8_pn/,%10lS +&sd!]^h!1mPEc#B+]-(o='Y3M$P^h;W4pB'"EMfep;)i;B$NGO`C"aq`$ +4XNC?bkrqsqVX-UUQ\jOB3FAh7EId+-N<@to"ZdKJL'?>-ZoId+\DdF.ZbI3rRDMU +HY.''k@o8'CPpR.fe5BCgk%s.X;pU$l+C&=(TFsLWE!tgs\d]d^B;Yj) +AO4e>ft16Me#$."Y$Di\^3/_sGNl$r#k`f;"NK,W0mLNn=oF4XqkeefU7`C]HmPFEu7p1,f;Kl,SAWmUbuhs-M0*dShoWEMT>LSjR%'[V]7B*Ro,\LAFd- +bqE3Bm!h*6oBV):A0k))BC2;1oMqJgGG-e:J@71)ke)-WmN!>T>P'rVAc?6I$j9:i:4:3\=I +\aL12a?BNrM"[48M=YTm6krKBCCT=(?A+S.$UZrTroZKTcI\EDkt:Q>.D!0-HUCFV +rB/6>N/L"2Mh;1%Egm\uD6CW-a3Pr60$':K1Od(G^_V4K[\#r0&s[l)rZhDDXq>WN +!8-F0g,'GD>l81&cO)D#'5(`#9X"M7+i'm5_LrC17@UQr@9#%ZeZi^DEI^:BFb8j" +oHkX=]oFndodq_FgYT-hiBp("-\C%!i$sHV*=7tAD'-]"a]1f665:*L$>p54L&KtX +7/fgRLFS5=Wd%i1m,/-BDmt.L>%WLlAq^Ua$3;NCS!MtWJH"+$rkO$6TeO8H?#7dU +?^U++<:N1+SG6G2`RW`'\Ro_.btIPnu$rj"(0sW39)bKJeBHiB>>?`fT!dTSIDj +_!ZA(JR`u4D0_?j2m^r'P=o&;6[?$r>6V0hG#Rq'>68bk*TpH@iD:9V>T)p**oTp- +mSXiHlOHWnNb[8V)VFX%55$$ZP>!,55Enh^:bS.Q^n:/sb]ApWGrZCdrrbKG$2Q;+ +7]JXFnti^a@X*sU,H&@FNrOmCh&Bscl_Y,\ArYR-bm0OT8-';%T@j_r#eLQuM>tqr +9cpF1J+T>/8Ls59ru^b1kWOuW&/OcWC]9OLi`BKes3NAp0pRRR3$5Yo\R.I35I_+2 +4O#JSP:Z$4O4='p8b_#'mWg)Wfoq_M+*^&$56_',s2bmI?fKU7X$(@A;4f7E6= +Y6".MYj5W$BN_TVA"p-O[N4;kMIdI7i!%%*@.i.0c0q@\Qi9WZTrXe![P$Q?0&p"t->m\tO1,MGX7X90FWrqjp$@=IeI-ZQ7fDI&jH+1s.sN,!n%,'#V)]7`lJ`?WH%9V +:P&ET9qAM'L1"5lk&4A_&,qWZe042IGui@rT9%pFmq#'$Sb^EEI$3!TP7.h,r.jS1 +;ZM=2Xa]9AV[7nR],+%]cf-FG7$MI;b&nr&@]ck4>q-*@C&0(@O1j]pos5>,T;:nC +Uumr>+j&/sJcBQdB;qJsnp,>B5t'DtLn^f)4M;#&5d$+Ub1D]Q[kI(U/q"Zi0+Bch +$@,bk-n(0]Qce^erQTE^"Z_$ZDBf]W9ua(\OAq;miA2d:c,"d.p.=fM&c\&*5PVGK +r:#(4ho_;rPD5M'3)$>+/r;>4M_\iZ^j69m:@bT>qHLX)0RoR"r[s22:&?UT@]URCWnYcP9[$NZD^;1QDB', +Js'XFi"$t=-qRkg+bZpA:f4lJ@2rE8IZ--P/eDa!;IQ@1K>2F,jde-EG:^O\\H>eP +Bic)7:_r"p*pLZRYBQ\BrmA+K+-<[cG-?o!W.m;2=%#."`jc$G:h>ka[DM$JOB6dZ +@Nr=,?IlXOgUH9-I,I)0,+kZ0r5co)atSeN]m*JHp=+^UhB1/3-Z0U-p[kH:"D\8C ++`0^`LBiAu`5^")[Ok[LNSYfWLGk.X:TV/mf_(fgp*(M:[-BcXmQIp!3,p"u[aa8Q['_SUdIc2$$S +m=>Q@:Z*cHq4U=LI8TQ`pQPn,^d$ik\H&pce48q0RblsePbT\M;0U>dF3f0nN;+K' +m3ho[(@>IU/mW_G;>nP%I8^H2Xjni+_(^TB!5+CT8GBmu-d0>OSA^c(*-%*J,'A]ktkPoCNB]Lq`0,l."/Hm$m;?Sc9kc^k)F +r(dhWB3g&eQN,HJJia.O_4T_:`rEpF=2g5iQ=>Kpg9X]*WE%[ +IlqqT=35pmj*t@1J6BZ0ps7mEps3h-\:P74rr`7/k(P)7)9!rS1\fkh5?Yej.mHj" +W)+6/"9(jfiQA1[F"O_\NlEDsBcKBN>.GJXUFMNBph&0^/ +)TSmL19!i$XiiPg. +!5RL&,1<_?ar]1"8=,bn;/MP!FgP4MZl?LS'BF?=q(&4FCEbmtLA8.N"Umqc-b[\> +gfl\L8+rb-#I;H=l.D^(k^1X4F0Blen%On1J(QR'#uD-3D/h]NNH=LEM>qf=/:_:4 +GhOTO6VIT0hi#MKSH]SjqY%`dZUV`[=4"NqgA'O)Mu#Zjp)Mh/KE(D2L(Am- +:I21Cguhc'elSKO/!'L!YNZ_JfRXD^Lqje*C>]0l($fHc).EHWbM\7Ho=.*:*:M^@-10,h/$n$9 +/6W7K2g`#MhLMgPn:q0D'o:,-FQm<5hfe3i[6h5oTg!/)k<-QTO$A6r&:'.3%pf_7 +?fh+f+o#EKCKn&\?U\-Alo;9;BDb7t6d+7KPAoBJT[iAD[7gOb9>3R_bB;>V>$NNLchBJn#DI]npSA;8rcB=4kXFg87i]64g +rsR2Ap\9mJ&GXQqJ&%Z`@LgIs@%*Y1"VCSb$L3`kJc$5[S(*G1WZ3iH)_`f^Hs'5* +f?7lSOp8>XVRPi%i!L]:UY]4HB=&LdE_6T7gKnGK)5OojQ;F=q;tVDnV#Z$qWFf8[ +'+4(6oKkfi0L%kJ1[^BPbQBfBs*,eZRp#pUBa<+Th^>,%J:&@@ebo(bF7j!M!I6^L +q)SOFH4b)?1WP,TpF;>!AMtj.r.5J'fljt(8`m#/ +]JO!WEe&31s8No8<(UIG&!]2VWp4I,2Z3 +/,g+XDg-R-*W@kK6V?o>#XJBpE'\1TO,:]f8-T#Io0;`:ebiG=cL.k/1AKbg59bTs +IjO\H_$`f*I#'f"SQZT@5=mE!aC!0!%$FQ,-1IP]6d*fM79GKG-p]R=1B(9jMIC#G +bEb9.J\l%e`QuTI?;-SHB,W9X*%l[H4(uerY0Si%bMZBuR9-9UbEMbC!`](f##Ug3 +KM(5U?i`^$S78uZ+PN-FI=<^d2eETS1[LIK\M:?WJ&Kfuo5hm&-+ +FD5ZC73l#$Mpb7JbSjb$4gsH[MU/4E@;=X#dCQmfYY+YU;VW5n"n%bg\]gE;58?u(fb^KDan(_"?nn6)8 +m1u?`G""k#rdU9`b3t>Ynui(`hZDDk!^K9c^A'6dMe;IpG/plNcts?l?Xs!mICh%H#1=+<,@Y#f +Bu:UmCcdse9o.3uIgCN,Z13K7d1EI*nt'nE[nilDh5Y=Khn3gXK,<=Kmm6G=CV.#\ +>7S@4%UF<+HVHP%B%Kes+,A,fldcdYAc,?s+3.gf:iPT5l-t^Mr0@<1L#;0ls4Xm\ +lK>sJ?UEAYLTUIe]CS1"s!DIS-t6W7[D(37Z0$=*X]AE'LY+l!dJ,)3CQt^.G/gj&ma5,PG9._r)as=ckH=Q<;29"m#h(q4'bdB +pcht5OKB5'0$r:?@0lc%&HD(:k!+hmjBn]M$O!lWab.=h]trqJ@b9%9lN/PX=Z=JOqWo'2hK4B\)nQ#QNJ3cb'+Y +%-KA?hZj";r-YFkbj8@=r\PCG#%`Mr=RlDDr)HX4N!Zh"fY?O_"Y9SQMEd("WlY3I +i]jXX0alp.70oimSuR_b_GB@iNu-$$1,"JAWD59d8:TKaVr(kZCq<;a[R,2C8p@g< +Dn0%qWD[rS6?fG'J#)j.J'lI54QD04Z@Jrs6"p\rs0h\i'dTX8q5*VGH72E+9hQSGFhKS0eZYDr\GN(W1j>7"QG8Yt6I$QC6 +pL!kCV/\3)05LI#3ksX7^cirlFpd0.s8(>teSbI-!&C9j=MP0NnRZ-=c,7W(J)d*t +-Po2G8Io(d*]X-Ns)A,es6Q9G#`lK.4EAVBpJCG\YoEIf'06YsdG+/"_*@JiPM[kM +s'Gj$IZ\Qd)X+,gcSX5\A"6#=\W3sF(4Bkj8K&9&4M5=Y8&N(mHXHkE%`;MfmQJVI +;igHc#?q-2IYn*!;+1)!@[A%i2/iI0\D)/iJU8S>=;Lg!<@UQ#iHcZ[-=#8Xl"3d5BIcp$S2XRWVY].LYq[2 +9-:l0&dgOX@*,j:%NsD&a_8kto>%4.nc,TEi!0Db*`J;o"jGa)/0O:DQ`jZiDgi,I +p`-g?/TgYYf^`A^G/=3>M-I;M*Hd)ceLtCSFFu(b7a*sDIudk=r/jWYEEsU;/ml3# +Kg4Z5pr=VKB0AK*I4(-YS9g1]]?l)U?)MRnrC@(bh2p61Z_P8+Jp5`3PM\30hDeqK +48?Iims4Gr@7ef:Ji9/.OM=p'_1-45G4I_sA@Onje#Z;s]H392Y#bcI0Co0n`rG0; +)%BB*l^oK2UCV%nQ)q*ENn+.UIC\'=WQ +b*#t@b(Q/qMbGQAb)huHV+]%]`&q-e$F^S.kf[I_Jc;M@s)7^_BE+kI#Cj\l-A_[p +M#Sf\s5ai'!A"C^h0r`4XE=V\i!C4pQhZ&S([)t!?QO)k`p&fek(TMAJe4-_:fHSd +njDqsJO7KY+-dqu"Xs1bOW>`u=Mb"Br&L`+6FTr;J]\%W +V#4a@"omMqEe"XdEStDL,5FY^AsE\+lI0']k3.d=lV(CDXB01Te7`[KBl2I9Emi[? +d9g=Xku`Or[ZqeiFVf3%NPd?r_6f[/E#/p +FP"*BgTCo+l)mrL,&lhMi&guFF6BZ]s3qOCrsmA+Ngsb`AVJ_i@p$(UX'`,F#4i@1 +#H5N"3^@O9*I>ME4HkNF30G5b=#]p-c<9-:==#Q)coY>RK:RJVsIu!V$(SId"ST&(`tr?>.dg@*T#_92AAq>sC.r*kgf'\R@F]h0L&MiJ_;#k\J#c7m@>:CCVG%!\B1pIT`4Tn@ +08jA40'7`bWSDMJ8/3d8=7uS:piFqor)Y`bpd>!-09!plro+YAGT?NRMA2orr_-l1 +nDCSsdYs.cX,>uCDP+>G%=D7g2qH+BQN-m;i]hY:lONid$3a3/J$.:B>N[1)Rt:U` +!XeEeP51Zk%thWY*U.#KT\SLYBO"u6Xg!PI$`G+JnK5UIhrXmoj^=,KI-?mroVj1\ +busQa1OqsX:]Uq!8;*WYRTH(gqDQm.4o&-",o2X,V8e$+S?6KKN9fK7J':,P!(+*6%.bauTs89/j]I'g+cE''h?]ub:8[PQi +YiK')L(nQ5*c#*pk>W&V;A&678I>DY(>r4+l,qpkX:Q,#],$14'SSR]s$N>L_jgaW +NrPjE8.jce&`m=:m8atH:'dLP2K[NX]cu_*aj* +1[i9nShhHVj(G*[F3c!ffhgAWhMogl,1*.u/\\P-&;P\bWg@Hai`R&.Imo1]]*/,j +E;ZX0.<#a*+cqRV^UeN*H1io'F_uF8^f)Y%jplkarWi88G*p?`3=G:Wk\r4j%e53%.H +4a1-e+*o.+BDg)'m#gtnS`@Iteif@>Mi+R..IdA@+n-C"2hCuV*`LSM?FG5+?2ZD1 ++q"\/7KRVm4aH$^!rVf[qcWlQ[=\H2=_b9U40W/@"Ud;2S"pb6[N#/%Euru>n(D^G +LCW*oGFX"'=9)f9A2,5&[S7!G%>"P#7WiJ^bs4Ge&$!P0^jeD$iS4t%n(Dp&WGf6\ +FflrHE9VuY&H?#?[]R0OmQFikjt1g8+FV9)m1Q&d^ZR?XD*%?+F)/^pZJdHuIVdDEiAT4)jU]]HAkM/N/EJi0="fRB9B\MJ>VMop%h7`T>]:E`%h$C ++VC[(MCd/7.+"]SScsE`=\nrN[H4nRph)^TVEL/U5PATXJo]B=@0$P#IO0;k7J[$F +\cE]&JSAGl`*lJ=Qb?;4&h/H\M&&p>>H@?\Eo8YUQ,k +K0mdAdGV?9DCjLYfn[lE;Hos3iBLrUekA(92E;@M&8)?9\,1YGJ!^3%Nu\2q\&#g> +i/\Bar25Z5)?Zo`Ap2Kn]mEiY*F"p\,7%B`d(V'HI+#=B_TqG;+ZF& +i6P@>k#N>Y";M)%E>IJ9'(/p=iEdSi7[nuRS1>eSCST^#@pEcDS=9VuLViX@i]G`n +qRUq5YGh\OHA>'e+2;@?Y%c@%E8Pg56o9)S48$F"nCY/7^&.;e4*A2#EudE?6a(T. +Q[8=L;hminqj6^*AhLmQFaiK(D6M_BY=2B/!V$,X4!BO^f;IWM"5g@hT:lA_ppPGB +b]=DJ1cPKa!S&(]+]na:`GZ4&j-3K`U?c/cEmFtJFJ0uK=J+#5*A)GIhM&b6VleZh'gp$)Ks4OF?[Sn!:gi_qK%I<_9 +HQ&$*(YlH4>r$i;aj"u0jMA_CgCD"@CaY^Vle8bA[L`-`^jks!rt"+=]_I'L+SZK0 +n(I?7o3:+9*t9jl)X0er@il,8;NKRad^DO;VHkS:4)u3,@KN0RCXB:\leWke?LXQs +3;CeIOKZHbl[Yh$i3c?(lSh,1ha068#-WH9h#&)&1@NdrXW(39'2ba(5k)GQeaP/T +('kK1ds@#Jr+BmPf,$1'G;nN,H9kp"ro"AO;`$[E4S.ENI$`Tc6mpC;#+6&Wb@>[p +-0t7BIcI6W]*ZrqUT`rP?e#Rt..;FbfWCdtTpO<2Q-u=sfW(Rpq')l[0YbZ,!:V)i +Q1p]4GJ5V;jG*oS#S@#W_1*pr#f(!PE?KarAjh?!34tnR4,OqBCPRQ33L]qMlLIUX +BpPVI8)"RK=c]DI%qWP*JX3V;)dK%<6WU-7c.A7nCJ.S`ASKkIBhp!'g0Q3H4pd9t +nt(Uo\0a)V'U<-)Ts$h7e0Wkq,fcOD)do6f2pW2-AGVbV8-RZZ,!Q<^M\ImM&Ji4b ++PN\:k:8Lka;OmZGPSG"s&o@D`7U%HM#SX6TD5TDA,Jjdcf0F8NQ%k`@9OO70#4JP +WZI`].9jC>p4?[`s%8=O&2NB+Y?%.j$(??$nN#VAV+Y=p[O:,T(FGu^L5 +W[Z2GP?DBFmp36pRjgOLID0EZ!L9,drU6Ks[f%QYXm51Y +AASE%N:Qg=94t?_XkL8N3-?4Pp)H):^N](C.Zis=6:^O;%X)jHF_Dr6HhuNQAj:8U +i"t>R&2f-*d.nefh=I.Jj&+G/Y:l'XCS)JZ*@[OCnX\>*)Vu9k*Q]Xh?,uI]TJtMP +iB7G.,KOLj!4:XTj$0,Sjh$$nR=]8F>N"5=_:u^-mJIm8$i]1D0;m&TJSUUXR0B(( +-23$)s0YMD5MAWb[uLF@$1WMf6ONM8m^]4PG9clInAopO/-/%]&1IVSTCpR5rE7.> +Cui(pNF,%2_>c#r1%H(WBP0NXpb>"Vf>#G:T\ffp^'CbBC\UI%p-XUD(A>(#2Rm2_ +9^sRJ*Ph[Wq]S)hKDX/a<@;RW8ei2`JLuX`)P-"SaL:CJs/jutC:2<'MqnuR/Xt:/ +=]0060HV,7rs=Et6NI+GDHD>^Hq@= +RFh(`s0*R=6/$jHbegWO_s@s(9?(Mu\Hi,goR[%?G(\s1gq6;Kl&U9D0R>eL)nbf' +A.;#t3H4Y$i:?W`omQXDVeJ&O?RN3[4a9M"HlY*CCY0st-#fG[c!LMO^Im(+nQVo_ +I^K)9bC]amY6d*ii7MNJaUnt&Yj7#OE1EWdt`Z +KE&&L1)n`,_#Nic=ZnEZ@HE+g-2f10"/_oIAnAY&A`,d]TBlO.nM7<.APT:7.9!blMWLAu<9kg,P:ZOmPfZ.;qpO*\sqj!R>\,%F/q^\[oUa;VH +`F>;Q1s@60WEY'Ta%Xa9lGl +V,+n34PjD+C@s5BiOSB;OXG6Rt5asms>#?&.KO(P!2k,]@=f+H_q +D>ULB1TLd+nhZPa;U$<8U/4&3?P';j^Zf&!F2KW@H"Xr/E&!g?_O/Xu44H;I9NCH\ +kH,X"RFn*&Q3F3li:m..A'B_\%2\pma\VDmR?ROsNFJ;T<\U.0\hK>BLfY=20Ae$> +n&X'%0@j:/6KJ1)ZMEA/pA=lun)o0AI./54+8ptga8^VXd6A.(rbJ^k03oJtc"c_5 +[5VlDB$-\S'H"8h^NuGg;OM9'Q=a0[C0I$?N'B42]P>PFm`7/l(K_*,2#jV(q3QW>0(Y=Tn_^p?Y(#r:'$#j4P5em#Rh5Q+DhG(@hX^_/n:$`(OE.JY +EsDR"[d?R#5T\lF[;OA]KmiIJ\oZW?&!8c+LD59sn'VHfd;ghj()p;4f*$b.H-KP[ +R_=`b3cZ;C\&%6;[h/tpD7[:/WRQut^;OS"810:8r($Yq]7Djl\71KI&>$3TB\g&[ +_I)[koRZ_C9pAjSAr@/oWY<_K4idO2@Jdc]pG(TEh1+B?EeWD\k6-6RTDUXn`Ug"e +l0BI,pi[[^E"dnH@,X4eH:Uk5W=`I6)-!UJfPHXslo4]8$(L^L+LqU>^WKTlb;R09 +:Hlt'Te#Y?9'>tOjH"MFT5N8+?lbcHmXR[`&T/7]5Pu!m^`VghU.#I,rY^16\EUV6 +UUgKDMSYXdiP/;U[ok5h;'WVG-[joaiK.44LjA4jY8mS2Z) +-A;.9C,&7tIZ[[+*Q.Z5']@frphH?JW*EP(TH7@2*a&>n\j0f,8MVC`Auj&EeGT`^ +k\)[.?c^KAKKF_a"WHr:2*%7iTe8o;pPmW58#hJ7+*`OqpP?=,M>4D676A$!-GJ.p +Tg9=K,YtD+Kt`fV9U*HEF6),a,9__[-lG`\Ai96o=@!'`t'9U#"qjT=!d +:IGKrperlml +=J3&uTt^F#_0P(SDk0,m]>"1BC&"V+4F[<[Rsb-@-6E,hj,a5hk[PIG5P72=gA^lA +8U[d;CU$3=[.md:Y$j\p;-fVf?+M1YT4-*0rN[sBjZnCE:qkmZT&FYVbPpjp4ded1 +kpENHI$Fk7-TT$FWY1iC!oi!Ni]NTCj#q>L]tjZJ(g>QDNt`gj%0PFY? +rMlgF.(dAFIE4O#an?1g'=XXg_?$B']3`kEP4nKd458T2:>@3m4f;cJ!7(8#(EO45LVs' +CkY$3ecc,@!FDouZj$F2rkF/(rt;V/'4I\p?i3tX`)D(R77mM%9)+Tkj?MA;rgB#? +ZNJa'3;t_=>sNW;r(m=F)&>Juhfl,L'Xd8Sr@)\,0D/Q-elVlGf\+<\PuI2]IaV#@ +b4m'X!UL"sL7deMAc0o(Lt][b`+B5lkT$rg/Id@4]Cplh2Z8!2"%SA]+?@.8)[T$dHeSOEBO6h(+V57qk+HEcSZ!fL5#H$5OqHo/4i\0N`#CFW(DU_p]TCDMH'2* +IM`pn^cu*XYg>56bW;KL3Mk!4YV35^5J4?\35%WkPi_.ul=G["T:>OiS0BmaPo4/U +9X5u,3F[H?R,FUqa8,s:mnac!\Nh&c@*Tt`r"'oLr^1sbSF0P\_u:bQ?])@a/)#'F +\%KihF(PH&&6g&\pNXuEV-]2^-,D]OT9FPLO;Z,H&BonHr#,Mj"&(`FfYRlq?NaJm +Jh3U;EWOAU@&o5_Z2iI5JQY6U\94C[-9o?_=,;U;gtn5=XE#&A3T3;J@QM=q$3.U@ +*u4]lq+ilT1q?2hT5oH//cLKm'e5.l@pV!hkk4(Qc]kDGeNGm-q"[S[:]ITthZ#A% +i4i%coRG.MphpP$rnG*>s"UGF^J@)ncf:)dj"Rch8"\PcNnum)YL]3re*Sf5g62-A ++oQ[b`m_AtVd6jZ"q.#^(;H4fIQgc15F&E2RJ1M[pVft;;.$1lB:;srWh;Q/ib^IG +mf3`o8T:\iF0:OEXbdiKpt$g:W@WSiT+us8,%r-e!rc"TJq("?!H*u#clO>k&Q#Ot +5igh`(Z+3fhrp@g-%%L[olf$%HeWIT2dktd((e3)/>&B;CE4/@p$rGn%$Oj(Op4X= +k4nTg,s*lE-d`+r1rX_2Ia`+dk4Rp +!!+%R#&jMfFS@HIHbY;Q+ioRHR;cS=BkVj)N2 +q94UF3R2&cSH +&_5o%-_`=R<J-'D?Ni'0%:AoR)dTh!fDZu%8gCuL-bMbjA:S$+ +'0>_hf]jjA\W>0a=>Ni>r+;qjcXh8VhpYY0o*=?mJ)i/:p0hgfM3R=D`"i2aj`f=U +Rqqdd>qBU4hbBQQRGg)bR_0[qXQh[AlnKo?3hGKAl=RNiB6?G^"oRn>VCZN'"]`HP +AR#Ot":OFB=[5(K?Y%U\YNHe58*&"hs,:AD+2n1tSg=C9M#>*hW[Epbs%>18_>h0g +'\MTK8b:GCq&4;C^HaTJi-%R`*ln:39j*XJ3:W\)jfs;mL,;3@,>ucBm#fs#^/*ZS +T.kCZN+QIlAn\JN:l;D1fFkPp'^Lbe+JMLp,-6"YCnVYNO?9R/o*2WcuSgLJ-`T+*I9;;p?gJeTVd +W"b:3n3=kfQN,McpbVp,eIVX0D"&E,PI8]/NO#'ReK1Q)=3oRb$'aOiP2U=G%?uHh +guoa%d!4S\C:5Q-Y>#QOP@U]7U! +p>8s$ri[WUs#LMqGkaK64o2#H_:cGI411+b.^[6RFLlO8=u5h51Y)ACZptIFVMfT[ +8p$ClFI#RA((59I$[$+QT2"Aa$t[3=c89<0(Gr!m2PaI*,rsO'')ub,?#&BT<;aK5 +Ih=&>+fL(RI+M8M$i"'C!f77N6q@@BBRmt!;W1T2 +S3]Q*&NRlYBB\,FPB5#0#pbTtJ.Oa(S/[oe"[W5%,gC#irTs4C]PDl-?GDtRB/\'' +lDsbge>!oYof:@_e$L$HOS!n"lIN(X8\&c'3FG]AAlgqSf"1.u[iI62'0L6T*.Jj: +]GW[a6[LZVE43Inj@"`EIsc;W.l[?a*U^s1elP +fL9/a630FZ,T$%o5(jUMpqPjLQ&VQSEGu&2s4d%F:W(d;prb>D$a4fuQd"PaL(L#k +i:s69P*7*SRip`/-05ssirY+%F@q^:K4t9Prb!YSP(R.e*KpOBO".[GOlrK\rqt=F +!\40SrkI@tZ.B.%7UO",T=oQe`S^EC)H(\<2+F^#\[otC46qN\HWVTc_D;Z-r\kho +JH,+K>d;9SJ((rTQPuO+lSAYZ5]5qOSR6n_pf7)dD`RKngU>Cs5/7dB`IE]r2r/e! +8Zqe!@!pG.o!9*C(g@jlei.&_K +RID:M^nk\_\n0RmiLWCqnGep2cnk0;03nB@nUD/M`o1k0KK8d)Yk0D[C"'i!a#L\3 +/c<$a[of5.UCk2cl9FIXSj>CA.9l!EA9hR[YLrVo$8DN`hZbanr(k)DIK(H%[Fi.t +$iZ`'+Mdu:VLI=e0)#41fCg+`p);h&f.G%CcHX"6XdjS5 +r,1P(d"$D\XCL(N?-S1J(#aZ2X\pnR!W.odZ9duU5pu'99q;5W21K)]cY`>'J3WU3 +.iJPQ5H$u(9+?K*pn(:a(-o^.L#c=S&J2M8N3#n)Q`]QX@/n@GbqWX-1le0a1+2pE@%3XHk3t3 +0D@">a"2_PX?\l^a9f,o$*h^O>@l3aZPbY]/=UIQFnYjXJ'q^rW\Il@5Mti-"b5V5 +q\raHM)YS0]Rm`:M\9-YLKT2bJ,O<&5F2P7k*.F9nBY]QL&D:,dWNI368<=U):\F= +Gk(n85ri?RG"5A8\.b]W>(G6/#lhXt2sJ5$?;RSSq"4KaH\uO&`AGkPR:NOA$%!HF +/4X)Q'!6X:8I'nJ'#h8jZ:>EN/m>H.]3!u67qKFs.K>TWWLjN/J$GaCF-6$SQ^.U> +P4J?['u)i5n:(Z'53n;Yg*!_-UAuaY=Qth%6ESNqXo<-jO[A0/;RQ>+)bUo)6ksI0 +PqieWWRcIec%uLSU"il.fbQ0-2N;E/n+lF((*Ma(AH>lns.CP=+$p`./d27T9*?:^ +4Ki5.:GL0+;?2USm^E?@28#ReoWX*7.EsK!O6eBPm-d\12G?1r>]dgh1HRe6qn^1j +]\./u`4T17\BBL:L[&=&nm8ZN%fN[cK;r(4qISE:fO=!r.C=K4B%IMi/iN+GYQTXP +T:d&%!a^qapeUn(MskeO?NU3kL&_$9F238J.d20!,!6*/!3i*D(I=[6f=i!.$1nG< +F"dZ9o8?lKJi>q-!O;:*G9=b\!"cAncNK`07;Q39KJ0n-(E>P;Hkj6jQFFTieQ]-q +QR[]I!+KK/6*X<_K`;VK6]b*7"o*I!&]0HjL/Og\("-MW>19%4hH_)gD(EmB([W<[,Og3*o%p<*:3c(ZW9?.>CqX!, +U6c7&e^#K$ml)bP`t@Y&8_]\BX/\?^i.!2Q!6.Hu`e>F&"YG%'/VZ:$s*Cf3Hg:.q +[1XF#;]?0DlY\'+e]+i.=u6#8:2Go-4Rg*^O]/'o/QV!JWH&BsfEJ:64B4F>^=j>" ++mc,bcNWJ*`T#)"K1LW]iib'AZ`g%+5EG]LMlK`J!U%nkT8Va0:cpk6ic8'(Nq +.RH23N)a;`BSD!L6mr0`C5*Lm8)0pe4Uaocp5,a;@i,"Ijp-4p"?C+KfTEJ#R/]Rb +Cho-"s*[s3j.[!Ke[h_udkH5a_Qps5A+k +[L`0e:'&<&VsMDYomO`-"rk9&UE&tZS=%R1a:VP88"*Js)`;6mp(qJh<[_kCQ[R74 +osRNaDhk.'2l6$B3Mg'gANgpPJ%i)`X*']"YGqM#V#"F+b5XZ-"=_,nFj%L8Q8u$F +=K>;g]ra;"VEJ%]4'eR=?`C;8V7BX'Z!MQ_UIo[pp5R<(p%8[Zj\ieV,/`AePMTQF +[VVFfFIW.V[f6a8YF`N/n9,3up6WgEPXj^%T@,ES;SI>]h]MZT.=caOhlSZXo,]jO ++X=9K+Etoq_1Zb$S58aT_u2%=VBm]+gi2Z9naBiXlWF`9?M*K:'O\fJhPa0AT[iG$ +>ErL'mL8F=Si"0>ZR,"mArQ13RQ#(DFWYi<_1-"2g:knc;87uq +4e_u+0,VQ)jt`%i"apPrc<`_FXar+8RpZT_hB'K3B=rAkJA2e7>uJ(>;HgW:!!WS_ +Mms<\?l1g7\G*o499/4QeAdtKd5c^HN4&.1GlBg=nMAJ1j++DSS!f["Tuh>(na^RZ +!^o[@`(N*<9@L*Q<5orRI;0<)4D$)>;5-*lF +k8Fp3T*3J)`SO$,2M/dbi-"OdTK.i.E>Q:b#WW';)DHGoLW,0rGDo,pf'L$"*I@J9 +7tL\Bbctj__QO'94U8I0#/C:2r?%d(4YJAQM!WiG_nc:PlUOm.[8Y*>#ld>RmE>4* +4!#6n"6:!J#9YdON9eK0f;Z/>G\$9u%B'NJHN76:gT.'[M3sg5+Y]MdYlT +V'ZQ/?tUI/iSM!%g]@>nmN]L'*K'cf/^KG7\!CU'VBq"P=mLDjf(X^prC.NM7MXbU +jc/Q9O0f96[5&l5+0_ib,*-^un:/OM2Xl45jt8cI&&A>D9M;?`J>2I9`SJ!1gC83I +nu#Lg,W=qaJaVhL!<9)VU&k6.l4LK_UMBH7.Q&E.Ph;h44:(p`C2A_(F-K#lr^@`A +_LJj-3)>0jTuKW##m,+r]\*Yk4;n0<3Eiq_2,DMX,O63kitF!,``@4o.l_N[gr&7P +"`f'>)8?HrO=4,d3;)3Sp##t\O@1rq,++9SGWdXZ270SB1pnYUW[oDP5L_KYmZbPn +8S;6?ro^S(5p,Y^O(1Vphh&YYLl[O^`<&]\MXr"k`=mL7qLJ'9@ec-[]ALt^?a-EK +Db5dHlK+BKYYf[I1='Bt!qqJVknZ8:Lk-b*XUp%^=.$62P8n<@n-;h[[nGc6"96aM +Gdd),5?A6Ma`Cnlio>Re#j872^c_L!^&t)Np`F17*(VEh;)KEq5*Pms3*S$VVeqpc +Y`*YV"pO?t7GM+dI&!udh5Y/Ug,RXFA>R!kq7..6[G0,*lE7m>\b;TNO2!dUnIOeI +ros"H0@hG!]-['(*JH`E./daNl'JH.5@0NQ_5tMgN6-N];!-4:aj[-Q7s,pBeR%.E +F)@];PDuT'GOA^#\p+,6^*Mo6Xjll(RA5-"#C#9QWo_jU'pjeX-4@6Xnr-bY/aLlf +W`'n4cQ&2)t#sKc;$ODCb=;28:Nrf7WJZ9nWa=2(:KF^22$G9$L+:Gu. +TX&OogDi[=4)LKh^VK%G#'VX([B55X/-iQ.\i3>Eel1qg\&Rn'<\bUA>D=Rj9YZIg,:*I^k +!;-=iD#S),$5d^b83dZ58$M@+?$aUtGg"ojbM%=YZ?d;!%%hd]S/noHP:+2kQ!u$g +qSp^(`QtG=n*@8X=QmQ#k*4ULhUEABP,!8OE9D1#@[G^s_mstr2eif)NA+.emSeJ_2I;;k8\7aL*$h%kU%?bE9 +?l<>ci!WX*Z>3[QBECPP<"@#Gec0i!#9]r!Fm[q,lid;Po,cb.BOdo*&L^P:0 +Nc;-is++8O$i9K5BDrcir.I95N%oZ!3d8f8T6>*#=(^@@A`*gQD+YGq8HLt99+g'NfZ?N%Lc,sgh? +[r9g)9Bb"dBCg)j]RJ/;Bu9NALsb$Uj-jDoW\3\u2YHtXmG^=IiW8)f*lF.,e&jD7 +g0r/d+F#YM;`eN5)d\2>!4)[3Z6%KC(;i6Ws*]m&3mD25#_GQ;!GXLOn#4:XC;g5R +b>c?WJ=g,K`[OpNjf.`iPW1Zqls+*Qok1>`^pdQ+(,boCIWZ[t7OY_AaSX$!qM*,> +pp+c[St$a$T#Qq7A'2AUV[pmL^@V]6^lb!l!q*nB]Em+CuV1'Qk9X@'?71n"SOD^=Z\'ALB4lQm18If(/ +2(RdW=0Ot+ANYWq@>*/iC&^E*I?k8T'"*_bauIa0GZQBRRZ!ej8Bd.IkF1RTbc^4n +=QH`=+o#ZR,*5qL(n+Ibp!pl@MO./m.>SsEjbVpXnMr%HtF(3=DZKpbn.(g"cA +hbUcX0NFl095V=4=]d;kI7NSeKBo/FZgj_$)95?^n[f."7[*0L,AO3FYJl-j7DVi=T0FV=TG,S#A#ohs*TV9bW-ViA/3LC-^YZd"($GE^#DGGY*'r?;2t"VCZ77Um7u(&u1" +^;9gnJQGF1D\\2`---nBG&DtoXK`!uN37:mml"s#M[hIkf7I3Gn:/9R?0L"Mm+/f6 +dC8.#\ed33mb*4c"!*kCllI]]2$g=(9CITK`0197O<8`*#jBkOT3fkA'feI(rZG^!Pj3S:uFS3s571e?G+BQc#^DKq<[Y25GSUT/oCKWI>%WIcMbQY*W0^<,c1K7[iV'S8*e&t +^_,=FZ'ED3@26Ot Vh34!L9a/+:'!Cq+t3-Cd:Nr]3hlrk3blIIMTpDpjf]C[2V +@^-3?kEI]B_-9%bcTYo64_;mO&?IF0G=(e(U)7.0&sRClqn^Oa"4m78@,Y8PkQPH/ +ir:gl`L!9>nFqk1pDed71[pG+rZsi!Nt)+Q6[8+fL)\BMk6 +o>04$#j'!7Sf`;&q +^:K2;rjBoK#;.,_o*#0m$3)`.\8(.\-i>>-!&8sXc[s[B5m71_e<@eS,k0-URfiAH +r,a_+Fl#_J>N/(`KR[I8d.deLfJi,%9NW"5F#YIpj[E1besVO9WfBaMe3!Sj9=NI&Wtc-48\]X$nQ2:5s)I7ds.#ALfs9,jX--iXaSpHll.r&U +T+=o8gf=@@L@Quc$9qfopqa>mGK8hEDGV*"i3NpXb>t$tO/-,>8\Jf'3bsk'eL(1% +?1tT_R5_7RQS%dk/P;p8MGGTV6Mc_)7j9%I_,E[r--cO]Od:IlD=gh7HbPdPd!k_<^6#*iqE1!Cj"&/X+63 +j,Z>n+f=)g(rjAtJ^_DaWnW"8EQo8jF&p'X?81M][1ajVnNuc-gu3RqTF0rUV/sp3 +QJ'qaB<4O"iAkO&6W!`9^BqL4XPklaHf`-q^os[K&.nG3"ltfD_,KhCOojSAAZuI" +ImENA=*VZTI&*C$0?HkPa8I[,InK-j!lR^`-V-:o83:>-Vi5hcS6r_;J!R`+:4Z78 +9)luGb;/NX`8mu/oukF!pu@lq^P$#;q^$.aR/]9j)7"dZbgeN9%`1c(5uL)F:O[$$GsGor\aA7g7I+0^IYa>p#ld&m#m^7?A<0YP2eY?@]9[#t +Q;'l^s%E,?1p:,-GkBX$E6#'kir=Gq@>Orm>hai0\8fN:%-%;0lW2q.WVfSAh2.9/ +;^X]b`n;()laEo^o>WYW(>q3(s1ZOhVW648auu)_B>.k'SqC6*D8?/lC@)Dus1ef! +$/j&>s.=?KShh^c0P1e4e%=u:*rD*EQ/EidhZ?6=lqdNq4e#6b(14;DT@:hUK-t"1 +bao%lA*4A4iJHIeOUkdjVM6FWDgV2A$X`5,SlKM22VqT3cG@?KW2n16,&1-&N)eoj +XaAM4%j:3D$Q.BslI75:i#e;u;`DGqNCs$U98@CbNa(+o=ks6Ug"daX][Lds%YE:; +DO#A/(UWJiJfa2bk>/-U/u\:emK0nrR=tK>M6'MhPcJ-,^Jm@tj9E@aPuXd&-8@^g +s3NUAi)GD\AAs'5V8*3;T3-BB.!\Uio?4]QA0C6:KunU,fRF'S.;+[".u+0i@6T'0 +?=unlZQu&+lt_W/.2l"9P:tt8PGKPn*N&$>VW,d#p5.3'#ZpP*8iq\T46KaemX$m^*)gBgBlP*Y^aaDYdos]>^h]'9.?aGp1=I/em:$ReOEpjJPh]42[4&Gq:Y2\n. +j?FI&s*RE+^Lh'W@nQI,A$USK./f]#RD8$QIbQ.X7dt,Yo\3^:s0!5G"T;&Ur*OUt +I+G,&0(AG*daJHDHSoQ4Wgf2``nc?Gc.Q\d.1UQb*.\'6$]O(p7D(S<5E\7^o6/=m +rrN+qOktck`<*e@JM5%@!Z_29(^Bp4.HU?^B*kGECA1Z)O5biu0lq]H'ArrH]Wj'> +n2>?0QT;=k(uGPOqgoCg$/KroB\i/g2gB^#?ps)'Zf'WIi6o!je;0=O?+YYY4-%"$ +c@\XT@uic,&V.)*a:B^"F99tZ#'V5On6c]N1a6nB<&sXK=!;u@T +J0>'j_lpp)@Vpl5!2%JYJ1Q5^kI!VpKZGHHCa3RF?>tQ/q;\t=s"U#^*PD#*FS0e? +g)KlL26gTX)0]uHZ2Tc(6AE-[VLj:1D?Rb70[\KX@aG/r'.'3c!n-2cZfK6Q0C0(k +;Xheon>sABZcD=BN"s$dhB^_+:72K._H4PWh?_\'44>X");U10LX)79<)PKO%d3ag +Zg@0:6KfE"aFG:l?k%"+CNc8Y)P5Np@*pfH>/+h0[R7rhckt!=GE\nJ_LI\R)drKH +kO=4Cqn,MVP1fdC`tmStI.!.lgO[ZVn:08jA(SC!-P"NniU:@&\ol>uPLTWa*BU]7 +\j>h='`S\t6D?>YY3q +ocO'k4qF4$iBQt6_iUj7k4l/nD7M8T;qWm,`Aurlf4NjG'@nGs9qY:PL/+96j1u2W +0in/R-@u2!iaXP];cCO&rdlM.4aigf^$t4(4)7eI#[MsM?!DL?,hn)o[\n,iUfe]@ +s20S8+SbET..#J9,MO80c0p[[<0#hH4_e:dBjD,!a"5$scG6Q>@FNsUAl\?QDcG/Y +S(FO$e#JndY+Oa8?(D9oed.7:&!#$C!;K'"#"GWio9;B5MB;H8-Fg;cGaU8MZ60-I*6QE,P"(99[%uq +!TKQort,3/mmQr+>87Gq$br._7QU@a_NoZ;lD2;rCcpH0s,-^tQ=Pn:=_("oI[q>d +>8G+@0#/)6*s`t*M`+-M[Y4\F$sO8l(."Ub`j-pcTc[G<&Qh:Fh][Vb031..J)ih: +gd!eGkJ+h4/;@:;b8>j2dP-%s8D+QYj10#\/PufH?NW49f_WuBU^[7Bd +k7QtI:J1Us?KtT9WS_U7]\AWuc[WnumjS*B6hUnMoIknR5@'BMnZ9qF->!0n9Em]B ++,6T-#l]9ehZ@Ig7:E8C7m7GF3%/K@M-bf`OU+kIk^/$K@X#S>: +0[.3qi*I\u0'J"\J$a4 +8"\K@(g6Cs?hGGp$ln(15Z:eV)-llLST:[$GAP--R"&n!qqWepld6/6@E8@ +)u_m@(9g-\!PUOb,D9I@#P`eb6TkgF54P7)%hYB<$87!3JH)eiCH-s227FM[ggs`Z +,o6OnI[Z4>MDSL%Sn+h7"_oBR^E3C:@\?@GTPcR_Bi@$GTRAf +hc.WXj@tCN,IfKM].X!C9cGDgC@FuaCF:eU;Li'AIPU"mc*_?DWbKG\FiOZ$qjVaL +K`<=s#Q#(d[dSJN*V^&G;3To,GOu^Wri]M5m(&4W62dTg6]++\CZO-Yn(J0I?aY?i +7@aUDqna1^gYd"Wj4Fh=5GA2GEA-LPjeI79Y/cR9,lU%%*.!/N4G<`5q@"X.1Hqp` +W +@sr:4WWr5_6'ZiurduO4i!9nlpDO`glP4q2otUX7+6d<5_uFd!lK:"fVW6;\61"m7mVg^$YR$]8$UNDfB_aLbf5qQ_N(XE1>I +,7'ZtP&Ol8F'sH5&'X3.3:eHO_Binn30VoHM`#Gi#=>C\#Yd +#YhB1AsS0tQE]^K\Y`qcj"e>[=dX%t\.t$Jqg(']2l8`c#_1hG7?Q/T&Qel`i1#oZ +D-nMPFd\T'oReo%Q5N@(ZG2u?^=E+c($=sR=u!3m0*CTmq*=p]IgZfMq1/M==94M&" +Zb&/6F`5CldCMt0e`@;2s2\.-$!c!G4`]aD$lM3OoEEKIrsrlK+:eP9_hXhou" +Sd"JuMo^Cf!i-,=r)H!4(=fB#iUQ1HWE)e#on!-m)7f?r]1U(,J09[KL(O,ed?lH& +q%CM:0E#lHmq&`e1X=PhEs3c]Q[Ib^@rnmf1K2F\6cLcbed!`K8^rFgL8\ +\H^n&A$*CPMo^4MZ]I0I*U.SVT5120:g5K3]uRUQqadu'dCa29Y]e9o%@d&SH6EI- +H4Dgl$lHK^rK-OK^jU1RWFsVc:2!tCB&@^gJVt.ACtc@3IDGjlf1rFIqi2\D:%#O` +6i=^bgj.h4\0gU6i!#cRgK(V06TA&/+) +lR#O*))A-3d(A,;>7MQ0X;X%04(`dDi'1q(J',&$Q_mp;U299-RTB,"QN?ooonn(f +$OQb.%#_aio7?qdg6iC/1EYgkTW%hXGhZ:EqR1_F*M)C$/O?'K)fPCaE:_U?=j"qs +7=$lM=sYRa,qr%N0NB\I8]<>d_39X?5",1+$n4]DZkH`b$-424jT*.W?db>miQT"' +`0YS&QM!dJoM8#?$gdI:8:)4%DM9tnFP +QhP,b\?dGD37;+7Bck4l";O\;1+QW>mA%NX^;2,1aZ$sm+KdIZY`.gC_SsDq,WS9f +Of#YIlPj..&`tA6s$k[3g]@WJ#8MnrjpMIpc)K*-(4^fIf,>Tf.$<(*EYAn6C>f*s +n4.oRMYS=mrmkQLa7riVqH:i6>K$hL-NoV/O$-L8`tCNnKbFCF6;207GNdQX_$#/- +pi`jZ5E>iVV15ipoo!C\?a!$%s8TSVo;@#iGPcFLp>l7\,>7Q6lIRO!^KS5JX$t.& +/P,kOmeVuV-3!@l`/eLV_%(q(..q)De5)AUI^UNlP9m1M0?O3$:B"TJ(&hWDrbqWJ +5AO>JdA4'LV^8E&(bRQ0a>9jBE3)X>X%KT@7f_Z@mT3V#@`Wk&O7Gj9YDiOAE6$oJ +I(pl.D\35P^_Q?QGlen["(dGLG#$adErX=AX_5bf8#`Q$l#S4D,-ZU2n6=fE)*;"K"nVW +1Xr@iOs_-9/JX?rl^KBaSLT\e_@gFnW%>_J6..Y_`\piJ0jVd6id]#`$C'>PUhS(C +8Vd4@i;]:6*T6rjT'.q'FN.Op'J+\K/\9m!XAN^;jm3/!kr#8!ONq+U\=8/D%r$=j +o\i7A3`s;l1]L#Pk.s33[".c`Mrm$p\kW'Fi'1&/U]=rV0nd@C7c=<49+V#l?$_1d +&!eg9Mej>Uh7@fErQEc/GS]bu^[9TpkWM2js*`T4Senn[+*e7rs0j#R&;1jZ*BJ`N +&H(uXGDiPkQcp?Rs,k3p44Kr;"MY<4J`%XF^L>O&G,t/SYo!&N"m\BK]S7T6a0f7t +'5P=1;]ZCUQ>r&0VL0,<0\IC*b"eGBO"T@/-XcXJke\e1dH!JN*r#PkKnf3=$#(?UmcL!U.c3]8c^e`lUr^5c(V-/K4^K!'u>!$Fq4(DPTCYuA[ +gcT@n*sD*u#_N(/r?>(!@7e(S%KugmFL]r$#i^=>r`aMd)KiC5!oCi07Pj1e:_H]M(gE?1e-4!2u].sRt40VbWh18 +3VEUgpqPuVRG2FqV(;,kJ-c'=e-00k_i'a+(&msVQ/Bd]6iI2Z5lcBhrrN*AM#Y9m +aCb,rk*&GBeFZZ%Z.q@OSI\.e:0)ArDY3D +/!8>"efk$Zq884,Z%+KOAH0.ko7p&>hl]TNHKG"d_J>C_#_'<,n[!r85@9Y*Uu:EO +'#.\8r*it&McGV[=/CBiP5C)4iBQH=raee&mtVVVW_;%5jHX8a!OHF_ +^Z6qsJ>``Ra9'60/d1M03kXVj+%Q0tF&pRh +bEj1\=9-p%./lTpWFPl0R;YT5,mdOnD>/k*c'EG>JLg^'pBX9j`:e>WN=_!WE2d+j(Ni +rhQq.-g,i9OO6W]p-$$Jh'1dliVbX'Q5ukU_LFZLE5VK9]:K,*::N.of:ohK".>M" +mL(W7/R-hF3O`k;p'RoE8B)V%41&n`5@aooDBjoX,J1+,LU>n_g&V&1J?scG!;nbr +4'/O+_&(dFkZG0;ZsrhBSm_IlblX_FK=]Y$4!6UEV"O1;FqO^lRQ"aSDr:NGZ$44h +$VJ`3-3=+d+f"YcXcKQiC[H@o"\kka1?Y)qT53S1hC/BVIAY;BE69]=ba7#d6rj4@ +4:%=mrKt)@#O`cKaFiDf':)I=Ui/rg%c5qJ(h>[_7gN&5AFLfAp(T;TX[Q5L9N3a44opfA)=b\J\8Dah)t":o;4T=(C;>LL_lgVCMeI +.to6AGs]^T(''b4Pbaf,RPXu4'2QK5-frR=G2Z=CcQhN,@o[o>PQF`_Yu6g_(7]`( +;5hbA:WZ\3(Z/VcWq4@ip`M\.:]hNlQYHluD[;I))UYCS)_i'd@Op!h.^]m=ZkCN( +'?^".M[a,'%3c.=,77*gnr:Q0/>C:)-'fJrMHE!U8R<2Abs+V*_=%2G_7&kIMC@igNN2A)KkK%J( +8j'iba_AY"I&lRs^E:2hD_i6Gs%F^:X*(E!Ca/n/[k7RU*>sD-+1>a0#OpW4,l@mq +Z&F9>JESp9aks3lJ$COqMmZ+7Cfp+):%.bt,!W7?Q +g$W>KkOH2W(d[Qe$rMEL=m"jL`kb6deo'3Ye)^1SR&/iqjeIWcII?_k)ijuZH*bi7 +QVShpcNMG3gkB7`dUU;V:>ds2!,;!a%%^_s*ggtq[-5oB'f`GVh'o^LJ;Ej=^#uBH +ck1-aM>,Nk^r<4&9)sH8r96j"(BBg)d]q*8?[YP=-Pl3A3If1XHmMkuSVOD$0#KV= +ZR)l,2hDKh5Ih\k0#5A5CY7epDC];DSTC,?pr=i.9Yg,f=lA(HM"!N?nX7_q;r8I_ +1;R6$n(8'hI]/O"G:b(I3ZqKFYBjb.q'XA_^jkNlC%jSYVZ]Tg,Y>k&)H]8-Jm4h@ +B;VKV\81d/k_1tc40PE'0-[i27;SRdQ;0G]kPY-KrmE=R141l;VoWfI11^A_c3j%_ +3"Polej6eROh]NMi'fC>'F2Q:&*uAdYNDq&4eZT-3X"WQIdnlV(9R>XG@]1)B?@.k +6@BgSkL-bQh7-oo#:[Hs907M)'*%FKqS3/_W:1kps)B%Z.lm1a#Q>gVp&X756Z<'N +!kA;WJPGHfGnX9[]L#fuJ\;=t#KRUj_'#=o%"WS=gXjZ<+T=U0lN/(eota]3^N`&: +#5IWl%1P;KgbfBk2YH"L.L(8G?**A,bNln#bldbpK'1EHYr27mj8V.;3V_W02mP4X +MuYUCp`1OQ#KXe-j?E'MCdVZ@gUq"eE6Z+.eaARH\pFW$Ke)&7^K$m?G.DqE;; +[Hcf+QNj">&%6o*6L]J+1]39Ai3#h"[19r!5J168EBMRm_kqS8etff-O7cDY:.QZDId_OQOGs!ATM +LUe&1K?>C?-s;d4.6r0C-&B3YAH@plp\_Mu>4BkCb%@*WT/Aq&*FJ>$K4C+*=b&L2 +\,>\RE61dc@KBiPh\YY[14h%O#M[#V^R^bf'_^pmFmYD[m7_hhWP6"b>Mk#Ya4OldC:oH\F8PC#qgRS;_%ZEHh.&'s;m`D^f7m/Oncc(_6ar)R +SqFiYa0a5=cfrn$&S]sK(;0,nA:_#dd3W^r3^OP_s%2;oJq%#Da;'.B\a__,o2X9E +#l+FOZOCaSrdPj#$0W_r(U+(!,:(-h$cW.D+Z%G2VU&?P&H#n@YQ=nBE#uT8(;*&d +6(Wc;?a3Wrj_fV//3:2o;i/]r,QOd7.IWb**.[sRo`Y^oPkIY2E_2R;_Hn?&f*kfk +/-XWFH1tQHhZnI!O87h>ijW77mCnG13tL4rcG;g@kPW;#@7&$sW.Ceg^cdMR5N@VD +"i-Ci!I3#VfEW'6;Z)+7&0;u`HK$`8gCmfr_#cN3gEHQ'1o\M0^nh;2O(Z(2r>Voq +qqa]u!YGBqBqg.MC*<)fdpAg7 +"kt@Hs,(=XWcJQbhl[RL06Cq5(BFG%+u!_0f4A(8Bi$p.I@]#126=Zbr3%LQ\E +8%RAakLTg,%.ToZ*:JAEE*JD%\HTI[Nj(fd[c-r=JcalA%e]\[eaT`Q#:uG +IR9l&;MA7oWG,7mBhY3tI">TtOb'1L=AWlTF5&8-Napj-J&Z=dXd&2QnUF,`F_R(/ +Bt-kThbD>pp\HB108VM5bSR<:b,RGV\k$6%T)Y*cr_ioH^V#+7c%C?G*#'&j3WHYN$SYo08N!!(p`^&u@9&$3/,(1Nra# +:gH=#s2biQ4Jmk#B8#-aU)W4#oh#BHOoM[Fg>c1[',8I7A;(1b]B.Y:h>cPYbK#+g +qoM'^?iU+q`;a,/q"^r>k8V3d$&J;)Z[!S5^F]ma!.d<&s#4;W[fs3Bi +Z8C.0(g"U:&cMn1Mk^:LXoSLME46G\%H1)'p#_$TjE&$*fcC<5[sXKW%biRnj.=6a +/E-k7LN98$mQSa_]?K#ZRBuK9GPt[e`m]Uk(Q%H2i`BL*MOmh/;&:\b`s&)Vd9@/iKAIsBIhoM7*k$h% +p=pC%NAL;!LF"(5`m7qF8()bU,\ZJJ+3ornRg"?BDgADQLa.GK.<)3$Mn#gS[HkRF6(t`I+P>nACZ(kg2d=P#&Rc*\`^6p!5K\j +543B6mTHr6LA7D1+,Bd(i'=1tWloM5XollLpj_AY^Lbs$Ujip7s*`W3Y^chRSc^qH +h?KQ,`f?G($accd4arD,/P42@jM"um;"W)"TAe\$!V:IDhuX9]/L.nY[NQV,m.tR`6An;F?l(BBj$uG^jc?Q"Q?E]o+ +Vj2F$/%&D.N\'n(;MtPS=0%,,BLUG97b6@R\?@!&k@Q](\j2ZdTd?YgNs>1,(rY=^ +O/pimXjW-,d"+Gf:o)CWD-tdIDR"@eOZ"#]K049'-sqTqp^^Lp[UK4B/>,6SUe$K: +V\E+(c#5:3Wn*.Z<+/)^%k>0,.oG7AWo+G&>U$AncW95kBu!dcPj:_=h2JJ`:V(2+ +VA>*=OR71)g?eddKX-2jI`3CqTYX"cf +XNLcB*u9?`"m1%d&8jt"+5r::rflpN&_('#31e";1Y5B(Lo[T8Tk%Z$Zu`('+:4_ktqUnYUib +T8jWB*Qu2Z<^7ZSrb5a8f@qc>>uURF-0=Si.Kf]XGGpVuJ"k`&2r=W1?_)"1&SFrR +CJtpa/*9(5s,-iZK`B#"\F;"5N0_M.cf;V"ZYue[a/kfMXe!J561BB>S3CK.4Q.k5Erl9Ha6`U8mc#s_ +epDStF)'$6C+E\>,C"$g5.]t62IKZJ@Ss[L?ao@XETl'`n`3X;M%cN7dm0GnLgtn5 +bqF=rMnX/M(;8^Gi"R3W1@Q@k:H`%drkF:lLB&7nqj/aehbH@g>i:jY!cN>*&d&)/ +l.?[Wr!+I^l-NC"kU@1nM?OtBj#_SR4%:\O&7^40(P1F^YoH.95m.-QkB'rPTo4W.cUsRO_gU6u8BIX^$U_]G?;e's,.3i1;^^s&mFu-"HHsDuG:%"Rmc@+2@s!aocS` +*.8e^$fCp'%!Ap"%WGNm+UrLQ=h<5p>\cKFpS(ecC6&ej[3kafY(qHiSh(R +blOq`a/P&M2HFW(h"nH"gYjN@9/\p\i(9-B<$3Q.bf]+XENKK&K:'6W#PnCO@QoSVUW*K0 +c[aN@FOKBj2tf:4+lZq]dDhmHFX$*')$qrDe@R9V;9`s]r.,( +]!%;_mIlM@eS)tPBLPdUJ4-&8N=3>"&QZ"S5TQRg^Abe^VS!b.R30TJacGCrdgofU,(g.^]ebDK0\KK+Bp!70WuBB4rY#YN9:Q9G3(PurA&Bi/8jJSDZNadA+'7,BDsW4mMTAV)D2(! +GX5[sn=/,%(g6SMhXh]=#o_P.)IE+'S?g&HM_388-UAtQ""'TE9V)%+^$qL;r$P-A +n2_CZGg9Fqrh"8`8Gj"g+l9/,j5*m/"OG_,[q+2pT4QBdM.Qt"fSUi&diRk +iWZ,J'[IQnVh(dG@:A/c1&i\@j_`X"=9(7R:t+PM-q+LL!_Z^C_47!9C8b0nad8`7 +b6=feqP;=1K$m1gW0DY'Uhc-hF-7f[!02eqEQsdA/s6TK%;GVpH2Z3fh+n^X?Ua?e +KWc$fIhocl.8PUQM\q?goEE;q^;\u[e,sn:#NL7`YQ3Ho6er-[?jP_^:N-F,fSRWc +n/o#Rehh/]&*%^NXO.RVhtJ07?1FD%\@3J$O/L"c4hE>-f'V`EHaN6$[qn-L6Fabq +e8s\*\Du(YkR-)d3q8SfKECXli7oX@5J2OUr7SOI!<00)+TLD?f=hoI'jLb6:ULLO +DqMK?de]S\$lqD'DHJ_Kkgj#/hmj[=MEP.pj1p5Zi8,c`VT,+iM/kN=MY>hd[-N_I +8)])5@EUdsJ$mkp1"huV-J4p.H$s=:S5_1Dc7"l$H +03$mECP6+STbCJ$)d]Z`?*4uj>;*]g-b`8!Y!OWFpj^mBTo/]oQMXIIupPO8a%dU[Lp*r55@H58Hh< +:@TFa&,P)l`W'5f!jEZO/qK!FU?fR!^@nV(:O]_D0j"FO!5Dm)qX_/^]7%X_!VTb:;D?;1?i@[H5$Sj5kSmGrKNfX_oMTQ,K+'rYNR8(WB8A +5?.b7NM9E@]3c$F`0nqR"Y((.;CR-350"tAl'BN^b$]NBF52qhcIm$C.L(5^k0@;U +pfZbAlg-dms$c>V>!U6]o@UH`d@uu4ld)=o-fN6E"MuD__>snF".Vi24dj(4S`id]$5i-2qpgO,KtpGRKpFVJEpk8_-&5]25UlH(S7 +ErQsH!cf`W%#?7RBT@L2?ZR#:\VmY99:&nr;[g=8ChW?j3YB/^;U3.079.&)4&:m. ++leb=Mu18>s/FMuY[hGYYQ-3W8NL?)8B*YsaXctO>IRW+bP8Wo;Kb5s4EqOGpH!?\ +Ps@b4:O*D#s!#ts\oJ+,5jC93*ViDOqr>-p^GT&s==a/V$(-,&2*k,j263#\K6o+. +;B(t/5k+g+D]$R1mXj30E,NEr44\noH?PgJ+oYiV\\DnN@*>5GE4t;aND9D4*PhA9 +fDZH;po]#U`:ela8$mhQBB2:EMo_odclV,[I&g\pZuO-hA'^j,n/nrd4_MG*"j$gp +Q@7p'A0j9\2G>#"m+R$jO(mF3ipHPbXa7DIR?nW2E)YgL'!@_%??=h.[_.R9K6Z1s +i&CPgG7&m#,&iDr42E-b4HLY_$rN4edF;F4EV58`%g;n4Im!5.C7I5`Rl@k_n=S42 +lI74%18pSJWZr7-ON=gEp^cFmZu&pPWYaAG/_NkLg>?GYl]4m[JE:qFNC"7%U2&T>/p!NB[O/b +p!\`ikM50oJ=/RS#dtXt;EIU#L?BO\B&LG%8(Y1n+2k>&5!q"%Wa^Q;^f)5ie?1,=2BgX6-/+G&Y$DKisdrREhsWk +14EpmA0&5+0ZaGpLJ"RIjn1OQ-$$V-fERgtGL[r,",QRnTU?qK7QD9KX,35=Y+\REs5 +i7.YBD/@7qHcN#NYu;HlnaK0OFpR?aCg)Ahqp%?o[V\?p#E)&oJ9TrO^10*h_Nl4^ +s,]j'"b;HFgGZ-:((G!oFRQ<#2B:Fp4EDThl90+Mh6Z.r\`DGrUmmZ'5u3h+NVUZf +TlJ6(n>YU+)6j)V_:'dGPL"V'T6Ld:GiGU)-_=i#O%S^cPNZu;Uf424Xq_c(39)W* +89qG$A`!R*s,ps2ZuO@#IatB7jcFXgY>4VHjaCP9ZL]me#h2p?S(<)4'ucPRj4K&"&]IRK"9sH!gZ_nAQd\ +i",@-K74/8*%[cNAobR[16B9<9\G%^`R@o +r8P.7:5X[GTMd#?j9Ap(`eBIgN)\?c3!EZ?Zl<)8`fb:L>]X:E3&.TuYccZ#,TI!9 +cER`E8]nch9"o!H^,a>%"_s`bC/!Q8^,iGs/$`&#k,qI,X\Mcb:29aPN&nYrg7.=a +#+7><)0_%gWf3,l%22A,U*)b#dLN>/O01B5Y.Bl)W]!GZHRkrL/A&nY]A>QS4tume +LK6%;0D+59rI\!XLCQrbq]>prB$W&1R], +H"\nEnCP5S^Y)9TrYYZ2fVnTF"]#Esr*1%j +GTQLD9`^4Q6GWUcXpG#U8`7&cHs^jfIQ^'neMIn@_!gF\ridO?QXsTMrL\6ZDc;2( +#CkW\nE6>I^Kl?a$f8cCE?9L?ir2GrpTN9r\!`::V/jJaMX@D6jb^Q.s)TFXF/+\7 +@k@1[D:'-G.-r'qUeNSHE1=^@X+FaFllb&A5jPdBi4g`15L\@Bc-nkqb:^WW1@5P2s%U$U"U1MZ(;:C2,rK:1XQJJlA +Mp<@7:4*%&C4P4tK@atc4Ql5\J=GKi94,%Bk\\6b=MTbTqq?PiAc2C-p`cLQ0;@') +a)8SMh?:XNQ@M(X=T)kE.!k63Z2f7Z0Uhq!^\l68#9c%a(P;Zm]7,R.&"JD-Hs!R= +QJR9G^P(MbX1DQ2oiY6%3d.,[&*@lu41"DQd6sQQ"8jIHBTdNe+.%(7(*I2k=[;UB +@oNWCVb[sb@Ye6a$?W#_$-b +Uc/4XV$#4T\/^J&>Bps0SRJcE@%h7KQJW#+M)O9L0"J-Ba-GJbN53fRs@Z?`k#2;L`X +s%fa%:@&4/CqEbc>M5<"!5f3W:YGnfphO#ODE\Bbm(#hqqm4W(`m.jlkWH@4C*.Qk +F:RLTj';(Wko/KYEOP6oQb'SLZ;b'N&CJ*\'Qcl#RT\o/N=DWN6X +jXk:tg\adibX^=(:Wt*V^1Dt^FkgJRi'1`Qn=L^!^17!af5,0)<1/8QL!X;*W)EHT +8t#Gm\`g35km[*k^<'MI'p`k@UQ$Q.(10;?`]hq6:rZXNigfg +WROULlI'HA&t)K\'n/&p_)ck8>.Gf@Mu_lK_LB/ms.p;mjH&ms,=qQ(3;.?g"]Orb +](@+n=s+RP'am8Zn=KY::D;YN.,"4;"fN:[8c:JPf+7H-KkF`^]U`&KoM%5+tlQeISUpuT94k4S$A]oZI +eO2PA8Dn-WM_#SMmoBde2\KG@'#*$hq!JFHa"4t)Se_%nU'oR#bB]9$_O7NN&q3nTt!c +HFZna\^?YS(o)["PG20;P$#o5=Ng7PlcNo,njr)P#ck7MS!$V6U0u0=5b!?*gDiGi +/3dpS!/A6sWh3_2[s<_;I>WesZB'-[O"&o0W\H2f)q2&255c[6oj-k.:"&/mm./>q +m8p7l0+rI^A%"%b2P8$-?,ph&LFqrhZhB"(-P>.o.,OQ_'aW)<6C(Apk74#[[VSD6 +rZF&?b%>nCZBo(R_DZ(p2ecANCZ:`2/t1@pflZRB&"U:-SpTHc)dqU@$idW$!UGgK +4:bVD'u"Z&bXI_IIG's\]U)L]0-]#>LSMPU2\BgaiN"V!f-">=c]6sq*u%[9c!7bu +p?a1,YX$pOJZI]1E!([c^-,X-hR4IbEg_12^[NFdnPok`Fm/Cm05;JHM&XXOt][VA=`Vu%KF\7FCGM=7?`>.m\'*%"q2+89JO/lqlY93 +I5aaQf1B2Mj0rAaDSHP3GT40lg5l!ZQhSX+p\8#=%bf=*3JW +Q9t(hWo-qID:'0.)W!17=X5ju.@%dcBT-PA@5Am#YChac-ZH +eJr$En:-YU^1Ym#A(Zcbpd$KgN4S^7So#@PXQq^IAP77t2&i?g"/F-dX2-BGJR/.- +Soo6A3BLHh2I"s3%g(otVR/\DZ>/#UYLgkHX^UHJ7u4!)E]MJ,F5[>>hn0E(Tk+nE1k5tI +;Pt!^9f6TrldnppCW:Ls7S+?4RUnVhq0gBh"2s1RmICPjlM9=)bjEg!n'NWKf.D9F +S_p93e"N$9ChLo0"4c_uP+.^SSJp1pO<^E2+\0VleYJF--^Oi&LO9,^#0(6X6eGbH +*6VICg%>qa`06`5h:[1-T@bd1cOBR:(DR\O\%U"Bb_'/ZDc6V-]AB +Un-(-@_dE"H?-,g8r[G'Kk_Si>>7encqGt=A/&`'((k%LLe6^nRH6L*[FW;+5&$ho +*]\fJH"#dH@"W9FPIqH!:68do+:2_R*N##49$.RmbmL5A#O=[;T+Pmi'm[lZ+PjC9 +mIX5GN'QZT)1p9<%7']%'bK.Sr8DAnOJ2OiJ8!f<@j`Uu%u"Ws+7A"#gY6eYTC^VP +hWr(uhn'pqGD2\o$[@..i,080']VZBG1DXod>rBuIpTon5J]o%p`Jt'"n7Ro-K4ku +Em^S-OKs7i702u0iR/1?O%*jXAog@A918KAJsS'B\(u^5"ps0/-iAqJs#EU-2dDJ0 +X?k&`rjNoq^[I'Z@ojNBs,ZREJ(mBhL\UmnMl?4<.i/02a8,a^R."ZTK[30`IkaRj +eHFiTCr2h-"oc^Z65r47V#J+"i6PV*J(f;B]B)PtRXb6FQIL,5-!keR*ApFXWZFp$ +-p5[NXf!Ta-'/LDan[[Pr)d]"S5F:=GtETVH%Jb9S5j-jD,lN(=?o<>9mMD/=h&E^ZTgOQ@4u*T.[C>;InS\5 +:0`:2\Fh%cq`8-j/Dr!io/qEn4C\+1Eqq8L#4O +!-IV9E9`%d_!TeaUW,C`/E"tB@,@BW("=AO*Hd@UFJ$P9.rH(LBpg5XP:H6i+mtFAifGEo'ebHqJ)^\AksD7]_Udtgp1%]gcf=.o-EOpQ +f_Hr`i*F:HS)=`pT8u@jhYA`fGFcEf@DD;8!<.W?PA@,5Sbjt0!J(7MTJK[#RI`8( +(M5N5oNp*aR8KuO;m0?%[\K*mN;mQeV@b_WLM;-O6/,KB'*NNW@)k$Go,lku*oS])YX\daqR-C9c3<"4qIdEGsOPT#sC +)O\*)`FSbA#P`LYjnA!#UEQOK+s`p\Ok-V($N+gCVR6sQD&V[6BGg5j#1^X1V3ib. +8:7a?YtT,19Mh-TB^QPKlmKO#Z@8H'BjZ6K!<'0HlpI[A8c)mmlTe;t?VsI7eFqeJ +8(-*udK]IgqVZ);")$OVj,#_1+"REe?Q^/BMZ:!@V=KaK.WI>aA.T6#\>b395YS=* +V#p\?d?8T\SOR@R&7G:pdB,M0Ih9`::C=hcD2eg!!%1C',<@CRiE?6X=j/ts +q#=^s:W<.%jCXeCC67LM9lr];!!,="+3?qfpn-oMPMr@/k>//bO`-&S(OEP[ICXBMk(+(T46sim(Va[JBj03I\ +HMG9a#lAo&o#ntZ(J/DGB\PHp>Ls1N0,(d$`:$r!XBe8r]StORM;>s+"eD;"@7Us4 +j"21p4QhcqBD*@C5Wa.&":UaYD*J:A0MANP[V0#+Dd0J[8(AHS1"fHKQ%YTQ0 +S9qSfFraR'S'*^SeOM..7B@[;k&-7OPD$m@,>^Ld5)_#9q" +Rmc,f`1=C-H]Sh_D00`E'-HdVcSNQ>b0/RPl"k6\Gm%1?nr?mo_bWd]&^4 +/=KTh!WK9$=s4(%1dmHjn/mShp^`F)8bLif2W4VDl*D-'2L$4_O7msa)cq,#QF(B\ +r\M"m482IA_+1\,g>G[`YRV2"s8&k`g.`Q@Eo?tTHJJe9GnZY]]^%O6:+e%.oe(^S +Q7cu/UmliO4E]UAL&*pD9EG(;lmE=G7NgO>[i@B4)hA]2dhF[(j$K>(75Ntii-3$4 +&)V,!VZHQWY>Yu$T@BqZ<@d<8.KK5SX-LomeLh`KY#ol:bZa0=Rd`@&8$RF?kh3>cD`HgALQ\57:Ed'0jKHT@?BF8^_L];NB_q[_^r]6'AYW +&iN"(:r%Jns$o1jLjA%gr22dQ9Rl&c,Nk-T,74MRr6Z.-E\%3_WKLAaL<*`$*"N'7 +(dX't7Y=H>N@(c.3P%sCWD5Ui<\oB-6*F=IA#dQ!W$X6!l&Cqo7PdN;G=J?CBS#Es +#+(E(YJa$dcOtHR+qNiL+4m4eLE(g8lp-3i:eF3]2CL^b(Z:AET+(X1"I%78og'\j ++'AT`J(t*EpjZuY^;cND(kY#&^B*j(IguBs_LKZ)&%qsm(529$FU)r_7Q]g`AK`03 +Rhp$Z!NlVLSGs_CLEXpe.)j^jf`cEtlj,:o$is7V'R)C?X>ib))'Rb'n?@]I?oD5. +\O:8;7eEGNhV0+N:PSZDCBB*;_Cl]"^K;-HrJCl"qsQTZs4e@h-Wn0ukWa8;63#T" +!<3#EZI&&"r4cbSIPsdunpg@ls+``=T69aRejK4P01lA]Qk]B;)m';A%o;8I=9j]? +YD@\5FbE:`W]8e>1rQsNfWHpT4tk7uSHoTXcO)F[>ThM9No=K? +k\ZdPCWu:Q8@Q!FF\1]u2=:69*ehn27a.Jp/Nus4\P?DCNaKDY#N0O5LgGQC[J;"e +!tWHRaG9cY6l:U@(P.Lr_(2_,eo-/k+IIp8!*nI#F_9*GGg-T?nlW6dL&Yd%NEUMm +qKHcQG&.BVm>"POIVc#/)YiO>]CoT\&[[MN^4iWis$6Y_fC@U^iQF`Q#QueI$?.'1 +]*RJL7pKGMiU@N]?O/nV`V>KiS:RT@Hl'!b\M"%](Pi_`e8H6"O(K'[N<#EM4_J.! +/\V^!I1`]:(1E=/bbO[^NHO;?atftH]H9BPEWG7B$!AU4gX70'U)cPP\hIF6-Ma6% +ML^!;4\M5o3F0FHetMpTNVeaMr@Vdp/Ds7`f'BQ*[ulLHYC*looWUFB"V(2b^X"gLeb;gIu4E<77KFKg+":Lm2/Q5j( +o\6fKHOd1TJ`//F;O1e^-FFQiGbG#(Ieu16A"D%EF^lW!<5CReQ1dF_qt;m[Ei9VLXOmB"_FFm1Au4+T>GR +Y)sX3iQVhd[]I*Ok)HM6M5BtFE^_!!AA._BH`mNS-gi3lDA-*SXAf#$,HHr:ld8)C +_.2f)@pHDNIT?n=^^_KX6'8oW"%<-%U8:=:^`UW.)H&/Yd]QoL +2D1K+ED9YVpu;GN:MX$-P*g'+_iube0DkCB'm2%h%]C0%OoL5C-'[g_1* +)V6VI$Q.;BlVc&H4sP3#qpT=N"$%#c$pF0oH^5)27?a=.UP(Zqf+mHPG!<6coh-N +*807dR39)"2rG^^JEVb`;3.6a*]aEsfa"5mc?@%umD>]k`]e?uIu$,n0ck[-mXXmN +;LfCj#dAh7Kd^=mijhZj[;l=N5co?)SH]N+2kVMjKLU)fK1:0>AnI#l)H/5]7O8tc +WI^E>0;`M2,GYX#l[B2Ki`_@ofiAiX8gtoru^eR[fDruGK$2(.IXpiqhHeS +DuTbDs'qR;/HGSGh?*=6/-Oqa#K=8N61B2js(S(E$kJP$G@R)";6l*gJd_6h&7iGC +%aBWjl`XG,U0Jf=UcV._0MR._/$N>4FBgKiU0[!Kr;rUOnNl[EEt70\MZUY!(AuXM +aBpjE+klt:NYiQ3Mk\LhgLC38'`%'`n=Ot"IQPUMMfHJ'$kRm?_tX*>rpR6=ID;?A +#G$6,knEi>=QnA#K\t#Cs*e;DGDh'fk6_*)=Ub**q>C159hTn]7,mkqM5-@AM#Tp7 +]ih?eFOc6_C2#H"1mqO=@U)J`X%Kd1VHVNHg=FYf)H(D:PHF-`/Si0(i[cTibf_6U +bJJ-KTI@kson!Z.cODYU$*_P@[S?@EptmpU`us'a36T^`dPbXkYB7s7@`NCrgNS%_ +B6_mLA1(d_0R2_:S9#:^KfC(1s#b\r1'g/_R"+DA3UiF=SbapC"nHOtbJls.cC-l\O(- +iQJF"Gjb]8htSkB46)%n)fOR@c_]gm$cm+Rf]UPAEM,3U7h1`E^H`P7!V;HH.):>a +#a!fG5ceD9QZ@o4MkkY)pt(KeE?Y>I203mYJYe-&I4LO/T_F[;d5n*A;?3;3_jXm^ +D?5)dQd-%\l`*_/O\,;Pp$f\@52]Y.e4-$$rNKm7V.rP15/?goAn:*p&H@f6W +XAUm*_&jkNJ3T]uCXf_#g,c.hJAQb-4kKe\J!LQX;?OSu"8kZQ+-lF&LrC%]("6\? +)?c^5MN<9*eL_Zi0#H1h$V0ioj'Z&9pe^_trR*uP5mK+RZ@EF$;&H$USG\or)1_Z\ +;V!Y;]O#VB^(CY1It3$VCg`)R:9^+6:?YmWnWjg7MRFA +k0&J?abIYi?.KfC+Sgk,U9mGWA%0O7NZf8>iO5Og9Shdt\df[Ak.&d5H&0oPPnc-XZL9GTs[g@lYAINn"&]cZZh'gCi&Du#f\$ttH +m@AG:E-M*Cht[e?FeF?SZ@FT,!#Oo@qaK)kOhe4WVH7JV/t'`plgFW.CEKkK88T*V'R" +^'Iqo']:0Em4r_\1l/d,4hL/G$QfU["7SPKRQL8S*@6?U+"sPP@g,\7aCcR#J6Ddi +)-F[K+;Z+WLptXd1$aPWoYPtC9umZ^#LD()^5I6&Sn!P\':5t8bLYB;EN+?2FCE3X +;;6=%lW>SdSGtG(^R-$9fIB>:2"dFh9u*.9Dn4e(GE^6tS[;>#N64EiN;#e["tg<["^p>(9iF$ +i4j0MU1+k_IsAG&rT48Uc'VO,ZV3ctKP.H\"7o+ZL3r*KfCY3Cn.#;^i53(0*BQ+F +4##nfSYAi*+ftarfJ)MkV6of8Ro_$Z9A7X8digD#]_`[=X*.+;CeCIs:Hitf\St6; +Hs4!kN`rnWXlgTZ/fQ[olCf4^B^2sR"7RDRcR.8g))P+Sf1h?B"uJraQ1G729;]e\ +/A:=Gc-fHefdNg&KVlPcB(`.d%Zh$D20,=gkJ5?(cE/)qC;YZa?#"*@lD\Vg3kug? +0gEGq_L<'c+*W8tpt*S7!<:5RUR4[D[47"j&<-dR'r!rnb-8J,9d'DhS4jGHPsM(s8BG'%gN&`[fDruM67&,#J4s2QM9:gF5jFq^UX8S ++TMU++?e#ti6DBX;*OW.47Ym%l@XWeR9i@M"dh(#c0=Y.a_mu4DWkaf2YiK*mR&H4 +;QkpRk_ORFS16^f-+7,R32dV*C(^<@kMCE;"[?K'r:_`a"TQ7"$3$Q.JNp9n_^+a7 +'nJTQo)E8qi3-qj5^uqNpP/qOhm3',29# +oQD.s1O^#tFV6fb +(bg4T1ir"JG8A-1#[R]6+*l2'q/LrbcU1_5aLa''>E7_KXt4DDXMJ'mUR=#ScLikn(\s8<]uk<X&a?]33#sd)E!TXCf*:gN-[`. +3mDH9`WH=-%Y-"D]5?fAo<`65qsaG^h7@l@q4)1Kr"7e#gTV\Uq$1sa[t#Lc)tto%)+W$VeJ`7EETIS'?Xc"82^!BbW3r5C%Vf=rl;9$hf6nu +=\AY*@0=OMUAaq6Ep]AL'Au*M_h>69[fCXR>JGWAMY?t,BNG98qpe=^k(h$-;Q"Q& +=OaO?2lMc!quZ[+s#"in206B01@^'K;0"[7!RK9%4-\3ULku)5C,?LWJ/u!.ktS1V +nrL!M!r87)nGn'G=\(9H>plCM1DShT.3AZiLE!lc&#rNb0:._#TqbTfpT4*(,fG18C^P\t3$]3Rs&/Z+ +Zmc\n!^O.6Pm$k5-8D8fNK2;F*u3UX!HpNM_T>ZUD8p +$-+n)03[dlgarO(s3W+lkB$R9_>2F;s#l\ZY(,O's'^cQ%h"%T//%rh`;c_]_>c=; +pVdR"(DDK%*[6!_NM:iOs1;1CRsY;$9ekV%1;YuMq>R.Ts820dA-%=9?VL%jMCnV( +\U0sT7=WHH_rAGY-iUDMo!mrC8sM6F=FNA>D((=pSPAOre86%(kcR%L0=(fq:4ZTIJHj0%uC)AQT^[=<(jVk\ldQB9qgs +OQ9!1GoP5'bd2/VC7IZ%%_ZX1,J0OG4!'Sim\>Qo';n.FB>HBZIT)%$Zc<*)VZFFD +IkK+C.V.d!GlBgU#Bg*p35kiDcBF$$djQaXi4]P(_hY[jHS+U<:[He/*87[k/B2ib +L3Ljs%]m^TFq@T!Q:"jlI:*(GNoR&oO4W`Tqt,Rq3",$LO*Z6Zo:C`@?%oDt-_:Po +60*OR5-sQ3JWC5WgHImTHria$F%(L7;*.Uh"Y^"&/)WWsJcBto#=AA*hMYJ'F(jEr +;9uu,i406][3%H4FmV>9j1"MTa=?Hkm5>?5ml?=8CDIY@q(m[DP=04"`N9fm/5da0 +l1Z3]7;^Kpf[Id_P(`I+71<*`9^)W0Mb_OKESQVe=qTro[(Tcg4dN%Q!VKE.kWp`g6[s43a;LNZXZ"m8KYElZ[3-RcbA8JH&kP +4l2]Ah%2I/LThZS5He4J7WB;iDGjYMQ-70YEnJORbG;^0Q#a<23MZXZqAn>?:g41iHM*_Mm>;[CX\]9fbLNWLdVmqGF$Y]cNU)h[:&Ur; +KWZlR]O97!B[EtOD;aX.Y_]-ks(Uk8(9%XY:^Ms3#VR.KW<2i*BpB3m.`TS1 +&H#=;=,^jYNG4]q5MlGqIo%AnQ^H6q`9:/q)pl_Heh.jgQVC:+LIfAdRe`_i-%fA) +h5V>UC71XXN%U)H&FdG;-+O=Dl@ns-@(UjV2.-7e5beYqHiY6*q`mio`+fI[!.]Q. +[;o0)C.a*_"&usL^6!_'&u:fN9l1g!#*Hj$BM+WBnZ>]=$qRp73T#HoRSM_$'l +U@Z.$@&I8KF`ra1'GQVJ#QNd_\d$=&,D9i!oSpQ*W4fu']8IG_oFJ0/d:`$ +@sFH:ZO't71iFDIm=0Dc;IdY/ScjlVL=fb>kYh3:\$r0`*n;J +>1KC%'"R6.jiKWmTfgb2Y1@!`p/%@+@%Q4OT&+EMGpJG5QDZg9M_ph(,#@8t`j3]2 +RC`?bg&l%MrH.V:,YECb\LEPD)bm:=1T1R7cM'3?n3NU+P2-Qa0'8Sq%YaW5Cs+0. +6dM4Mn?+/)ctd7NF6^])Fl\U**6:-l#i'X_aFEF_i#fCP*P(f)AF7sJO>psA%PT)-Z`\NioplS+@fo/q2dK#mc9II=Z/D.bk" +-*H%V?mV/rF/VU!(C-Q]J2[tOnm?\X]_X*s"`)CD[qRS$Dr/6I3Ukg@qBbDL4o7!B +nR]MaC]EL3O2WV.1eRc"IA2LaQ#R8E!Z-C% +M?DUi'1]E6o%KL[jZ%ZrgqTcEZUa6_rq"WSg%(s>:V95MNo_1&E4!u3?(Cops)l#P +=d>-Vl0N,pr$>eU(>.6ABK,Lh)?BckjL/W//YIGp#JkeCd_;NsgsBm@q"$>B\E]?; +C?78B"WZV:n:D[/E/dmh$K+Xo.CL["W7nH(^7s2oOX]D5Q`Q?bSUg`uUkT&THE77& +N+Rdl[&@7lUK5Z@Y8gUrmZkm8<[jKgXhj^i24toe3e0G!Z\"rX4f)-bJeS2ls2CEh +PRXXZ5ib/,>ZH.Y7B0=#CB8:Wsm2arf(])V6PY+VbK& +Q[noi$Lg>hO;`Y@OYZfUZJ3pn[5'I1RHSsNGQ9oqqeu]HBH].B['N'#l5=iikQWnj +P<,HF4$HoMS6VhR";Uu(h1UUVHM(6FI\gKTD?k,4f)2?a_B&F9L`-]ciC$fo3Dm6E*- +SIneW.9/nfKtN$T>t'W^lHuVSdn<8k!fa!l@VCLb2ac8XY2B?[P46g#ZB)rEm`A7V +Za@S"CnkdY6Rr5noDD>3"3B^MA4@u8*i_^@-A0\-T-2ss+C22O^j\n+5-FWmIM)To +IT3m3oetsQGCF$!#Nk%^J/<^lQ[hanXT.2nrn@p$:'Mi2s3^O2\-q(!HPRD!^(V5c +pAOLNpT)(^e@Rk1\-k\c;aKZ^W%0aCb:q2i(s*.@!_U%b::?cqD?-pu=LC%RTeg#$ +39C-UVDpMAFMEcR`[&33k5Ro3i:hs/Ilq`>o4c$JYp!02s5a[H0;]n1-iA@^nFKL^ +#ihSM%\cZ]&DM]DJagoV_Rh#+Rr!Xs)W;ruE1BHY90k3",)brnOT1eD_P]!`T61P> +A:T3U#YYO%A5)L!ERF4`M1<3?!r.8p+LtYJ@&`c)Vau7`?[95E30g[.;*^X&<6$k% +W51+9S1R^4/(0e<][]=\Bsp9J-;P;\L<1FpGWKh$$;)]'V1li5#Iq?XQfDrV`a`C5n)AsF2Hg\68-=H5\Rs46+VSH?DlY99fbSG6D(3KfO-\tmE$X9:Q_=t(q +%kMCW-8]]eRYY[4W,4#V"4sgNla34$d10MRD1LJ3DLorc(nrX_o&\[+c#uB[Ul2YgJ6#>^!WA3(XAO$qrq%;&QJH_kWYY_mH\u5GCV=K +r86#m!:0gfQZS^Aj;+n)'0:Go'n65_PKG2HMM2WAUEVp'fE(ut@Z.m?]!\1KNQMQe +EesbcD*Mu(67bQP5'Yke_ +*T.GD!;E&<\]r(bHI^M7@N#Mil;Ztd"eeBpMTL+q&**thH]=hA[C@F"8>%nZ!@a)C +'-=^'8iapTE/Ar"Pjl`t(DoHNPS="h(ZU?*]IZds%rUn69\Fsg-[tRFGtR&kaq;,j +<$FV^MQqk-ObLq$Kuu[b.cL1Q;0f1HH0WeVMI,PiSfgQS(p\]1$6^;0Ja#cjd>AF] +W`D[qjuKR[Ku'Wh+lp!AmBePS4Tn0L"m][rf>[OrU'DS6>Q)V32?iUI\,$WDpj8>/ +]][$YIMV^,rTBLE[oi^>aGUlF@6>n7bjSh[kX2$Vcj+-S0:N#O;oA$\s,kS@I.Im& +s5STE!U8GFR9nA_+%i->DF'q-=f=eYQtfok;L?(30MQjS?#hAiAA(P/\<:f6:O(X: +s"0',#6fQ5,5AK9JY,a.K7MK:)r^\9&DrNQpca&EpVS&d+*J7qX&IsR*W2PHK)8DuST)G`"Q:I=\\L4r!2f7*"lVZbmerg=hN*!YIl^9uknsG] +8+6CajHfg!gEfT6,^,UMf.%j48tRXKb:XP\J.^_.03/')PUJ/L.'Ad`YEgYiej*e= +CLnurV(0[-8ol+*p[LnA"9G[c]QZml#TQ"l:%Z95*ISgY=N:[Iu-_\,bXpPIjkj)eON! +`]]u;diL?m"IkAYU&27tier=5Zu7=FGNQq+p6\&:@Y6>l_r#J/n(E`hgQ!?/O/"l0 +:+Y`;f[RGVldQ4I[J.L`!MLs[W1iGQqrU49nBPr==5eI63s#9:[HKi^+$"IV4nkS( +cb0WeIN,fr`c(OH2sM3-l,-[]YHN9ebtXNu'NDOgLcQR3tQCd3Qt7 +KMBn6X:kp&Fn3r3O93dP^Qu+qEPgR&;4C)g)fL>[DLB"Y%:*TIpg=-#2Nrn3UBd*[ +a#7G#6`Z=old6(FeuA'3Z,Q4Cq8fg*[^TLMEVHps9gTn!<[&r)o6#+:4^SOf:UXo]:212O;H%f^f +I*n*t)-=51UJNZC,^W[r')2u@-e5D6eA-3&^M-?QEF#nZ7AXU&[M41D8#kM3pgc,H0I/0A]m+h+$MIJ4Z)@,m85pq8<)q]/ZZH7>ci`@ +obn,8+UcpVk^[#nUW!B&"8kFbs%1bh=WQ#0D,.iSRTp50PL0kiNa.8%j3('he<6FH(c0nTU"Ub=I:["'=;9IjT +_]Kp)?PqA,9a;e&kZ,moe&q)*VY@$%[&VL=AE2Fo(dn-grVFM#RTXJGR6^a`[59"U +LK8R/.2UK"b%ZRMeMBHbC2()u1r):W5d^Y=!m.ibWK70H+ooGDW)emYTc.",(mQ`j +Q7Wc?ocu4cg,J,eB=sH:-k>YB7);(n_>g2bEe4+*8Imu#,A>LhF;]"OS8YujIthD4 +>Vd+I1DGEXZMpgYIVEg'RIYnHEWPED<=HscL(8`FUce?FHL-nBj-oJU5$?0n0'A?@ +m@PRCS15JBVX2?"o8'/DVHV[6H&>6T#_>'O$k)4i@m_+>lQJ&ZV0= +Wg=J0U8.r11'.p!;Z+Rn5@7RkAb_[D532b@'%!!Ca6Gnh49&gjhJT=NqKFTUI%&c;i3i8Z;1o[Y1?k'e8%b"?bGC8#ar[aP95deNCfWaQXa[QWojk9&nNk^cK1^7#>:Hma0"kTe +BYF?67^;STJ]r/KgjG^Q<2SYGWldSaHS)L5?G;RB[2!_TFZ+MP% +I63`X#1qWW2k6eSSB#e8a*52:G5Wd!6fu^h<\@i>Nl%+UK@kr&m#)GCA^D(D$gKft +j>@%(54b@/pR/p#Bi"9!Mshil^kqA.l?MPMHgJg;EgSk73jiT`LAJ]Qp9'gV4p.;: +rS*Y:qt;85L@7"sgir`%SAfCfhX/5p)sQHJ%G:bC,ipuhr84atg9i!A]`?7[n#Ha< +oJ*e@')6tA`^U'Ir4-I3BJ8Qa2Pp@@&/Urir$lgM:&*F45.i]K8`9W?c/=`EfWj`f +\.;oKH&ZqX\oaE%F.5.2)4m^)0I7oAml@0M44$B4'1L+Y`CuI@9p6kt;^BS$@ce(2 +b=iLp)SpIN[cZVGec29aHnBgA?-Lk=/.6\=^4',e!gA3H4gCP?YSc(Om>("ie]=HG +0O.E@-K+%5C_6AkDDo#"e$c'h>YqGj;lKp!6i<";fS/XcB'l*leVmbTU0?Y.W=C-J +aiE)-"m+0Dc"_K66`1e)_*?:nSQNptY?4f(K!9`/YBa-]"uIi70J/j-W)1""=>V[t +c^)j_<@ctSD;6DeX_BIjLe);bea&;&B[h#:.q]pb_2g-]PqGmjg1;LL"mh,bS9"Ft +DE9U\HB"FVN'g'UD*N)Z20N^m!9St?s'ReO>)bV.N`DY4"nc'W5K*TFHHdt)-D"NG +Mc`0?<&CFP2PF"M2ZWkHIFRrkL&L'nM!ZRs.a^;\?L.HG9T5:&u:l +F/Y'SPe';Ieg8e23J7/g_ckt_Ro2DFSGPu?G>Ptd>c3NLWLD5o2Glqf, +,=$_FdSWmK\3Qqnj-`NZYZVo,+m3Lcnq18S(!h`;13k,0. +&-9u[re'.kG':MR%EK&4d%m8i9ZdA7"8OG]i[Y^P6!0gEW_ai_:Grp^cZneH#IZkf +8p]Qips9H_JL"qT+?s+FZC\%9rcC/m<-_i*JmUQ,m)jkf7=Vn#2kmdEDi1W?U9k.R +#qI3n4@7=NmYQ#O6O-.1@dgqK4[adYPC8mqjHun(P+pl-;[?*[%nTUe8Ps\A:gDJ> +/bAnAMo&2/,IK0)kY7bt9nC[7r-Qbr"YV%]V[K"? +$6DSADMJL9q$V9"pm57]Q9E"epbmbe"^KH.?PX3g/AOH`hb!T_5Mo7lq0t4Hn:P-] +i;\'X)^MXcJD\P@;PLsA;lr6RC<_.+oM1`5e,^6;bp8T2Se35jkHZ3$ci1g50 +7D72rh>Qq+FFR?cn?ho*6+tcFc2HZPWT3Q)H_o0V-.5Ei135JHPA,pQ8W'XTC?;/- +>^oY+QYrfZRGjrKC[44CQ*q7PZ4].0iPSiCTAFUXg@LokpRBqKA-):Qr4r>jQo2qg ++9a_2i#qiim6I=+rh'"\?(.^*K2mJQ]pCnjW+1nSm#glJ(]iJ<@`s.4_fPnNT)XFV +J%kM?m5J@:S[N:1t5si8LB@#?nOD'0i +4`70VTt)l0YV6XD]Yk?YqI/-06C,A%^%IBCNAj%+E/e/h_(D_e?/%Ye.abr3b_CWRJejoB;d`cig>\<%9:paH[ob7i/R,5iZ6G;C!iX-.#$sL:DApYU,iJ +=s`Qc&O,^k"/Q!):Y:(7^%"!!AHAu%cbAi>$Ggim;]Z`D[^\\kjN$d(bW>.>K19Blu0g.dY7n?a.=N^MnA +7'tl6Y$=)`:]N,P5\WL^*3f8ME.2%*nK.d=WtJ_F=TEDi8s]im57:Zs=J.S3mJn$, +@aVo%e]@t&T[lBt5H.d@Ulml//=Fkf@-cFMWQZg)$@r/4B#0RT_5(s/Md-Sdmnq5B +P=b.>9H-gS*R`/YVO$&?OpUF&bT"n\l6YP9U,J]JD<_E(4ZWap\qU*/73I+F4G@DN +%7#B]']=Hs0cc2QidVLBs#$f*C\2gj3*\Ib*:9d5_4<-[S0N;@t2L.+3XdK +^]KZ@HR(iG`'om_:;rS2^o&XXd?XW.Q2PBR4pc6NYtnWbA6`T>SRSC#i#]&K&"Op4 +57NsdpQMriOL,G.D]ekl,uNAH=l(MtfdfYE,>$]S=jG: +hr([.Ii&#-@DVsKr&=M7s%.)t^jfVXli5VGs1@CNJ'.o-/H0&<^V`5f^cj1#N=>PSar0BQn3e.j[!i0Jgcm5EK%g9P0%D1e"Eg-R]GPjMoXjcX]U!jPO;r(>g_X +5-,C0WY]p9+oq-4H8jfH55t$Co6n*/msBHc+THYt#bSI9^UB\ +LJ6[%>'NmA]M\U,8c2#`peA*59oBkPFs#e$ok%-78b]Ed%I1E^dZmjLNFPC7A>QCq"%C850)MaAY(nVcnBMg]Z26(Xq4^UunAsEO@+6hJgD#gF?$(1kU-N+j.V.,!iI.h7u +2T[.F05,@K,Va)'KPd02'j]H;8&C_2B0L$5pb$:V)t_s)VB!7,gB%b5'EGoALaRJgFk@"Q!A:bo>s@S&BWE>kt[Q):&8) +brWZ;]NcVur(HU7$aTSLdQ)8P;_85+/![6:<33'"_NbB(f;`.%mJY[]?e-/AA*^K$nGP5D#sRn;q?d_S"uoeh0*`lY$*tj +^oDQ.=9--dWANa336"7W_ts^be0!rE@RT?KJ[tabX-gY\MHCWKO)Z +1IJ.Xi*S7><=DYtO@lE" +n4-8A+4u-2Uhk[ZCkD+#TiQ1XAq$K:JJ(uaO)-"OS<=W#1K8r]57KBU7('Fhf& +9ja?4!.u9.0F13.pW7Mtc4.#"O0H]^6Hs%]j)q\<[F1e*&H&Fpl=<;Ge&*UP@:Ve* +:f;V<%O74.)GHt4).`pih=gblJ+=!%p1a;ec(AA%s+eETcN!WCJ;h;GE#6#Gj;%E< +_uIL<$bEA?rG*XN48@jsO*)`Nn>"f]A))Y5]'(2f5K81FUoRg&W)7H:;,O8X<!YP3pflOeQo-2O?A +po6dHU/C-=.0)E.'g/N5\ni"-hBI9p.fQGRltqO1f/f$XmM%t=0(G"7,X1LVGge,3 +Y"eh"ZhW:$Z"p6#8L:m9k]3@F-V&QIEPF_@VBK-J/5.W[:c$Qn1YkDCgYT0ehQb7Z +_%,a4$(BQY?JP:Iht*DmnK-?oS3.3g+,)>;7BBf`Xe/*kSb]%*cQfrtrc8kVo@G.u +9>>a'ZsJ0@_>i#/%bI*jHGNV.ih\:E/I-N>n3l4(l1D^7jIZ%_^F">*nGAc_i(Am]f/SU7Z'?)R26=\$c=I>96oruOSs%[6eZL`>O, +BCXc<$_MJ%[;b1bHg&=-ECX"Od&$KlF5Qd0rek\]&^pCqL)2J7g#V +#L;dB%IT\Z*n^smfT!Q7S-'S;n`2qNE.t_;/Y9T"X]fjj/#-WJ,_GE45WTcLF`H#s!DLZD8c:Oi+;ZbYHY>5kR`FG +jJhN+n:0>on'#edZu7gLA3G6N2Z`[9(.5mcFfhMUof@p#V\D`'pfs@Z-Xf._,DtSH'gVb+FR!>DbQ2: +'Qu.DB9Pe=CetO#o'(WA0RrLMH2VjUg3OHgruS_!9='(ANOb`h];=2jVV\POhRKH9 +]V^Cg!01geZpO;Y>i(-Mp+uO"6)B/>NH2i6S8R_q:(9k&P)]251]ZJ8rf\ViYZf_I +=ul)%6EO^!ULFjB7W"UCDZ]nkl8L=7[B>t*K+"o+X')6U.BV#Q,'Z`:R!Sb=:?dFT +eYE]/%Q[?7T@>K'kuQa._Lnk6K5-)b)C8nG"sLcGn:+7Z([d&9Mm.W'KmfBn]ip-9 +1^c@#3B^OL^"PpTQ`m`SMDNHPrDSSHE-J2eLU`,tmS[&p'h.)!!QQ&l +5Tg*e8R'V$h*(-]h>PVmO1\usI_#@*6g.4YpYN=Wl`=#O;=NQ=httLOrH>$N2T>&q +H-M)sXbJk'-60[Z(qqhK_h&?eD_Of([t(ZMV)MRg2HjYlr$tJYTN[7a4T&Ea9I +HNbgDpa`kl6\tQ51sX,,?HTV`B3RjeR7D6k,Io/K>dKe[lZeElK2#"[FcPL,6ku4" +.J33aME[0KEoa$!ZK!Ner-8]TFoo`;O%IZWVe&?Nk/Q&YUMY(IXl&W?%`*WaKH'F\ +5:k_eg!=#2rlP`@_"BVchqn8oir>,E*3K@b8j0EXLYf`fn/iG8#5]P*<>o9FD'es< +,l0MFanQD+`jcgS]9^FG>`3g?_0#$>[btd;8@_o^V8q.:<&X&=3;oXQnXXjgWu_l\ +qaX2LNKQLJ-%@XC!rd]#,E)JhYno:Mj"fs%#J6I[XX7G6Ok<>=2W]i']eR>LN.1lg +T>S[p?t1Rg&J&^1)&o;["-2$<'/nj%NI,*_Q`urF`-ZJrH[-H=&0Ecm5&\KUl,14S +_S-m3IQ:*t@S$K^6%ojHQ/:t0!"4&..KAH5ohc+m@2&Ip3r.`W\]69/J:3+FDWUV= +P[F/7J(1A!qQO&kFcQSBH^.t2.)"nEJ).7'\m=S!AqfE-ML5$e)ccIP-$"&1o&:XS +_0!qf7(oU&Zf4OA]HC_cq,^i]VnB0+1JJH +YdtLYE&VlR.suY%;Yl?Wni14nfIgsbLmD*8nZt9-dk^Khelk.d'-NY^aQiIaPR]U!o3jEg! +%-.Zm,7D"0hX[8l%[[J"mtE)HOXha5o[XJ$Gc\T'-k-W")G?I&B<%aeVWqAn3R]Mg6^Xg)9[?5 +,%OX^r5VV(nC=k#Q.*]5>_)jMI^);M(_VUn<_^c7orD[6_#G]D'`Y@@2_CFubo)gA +C)6mD+mat]U6YnEC?EcSk:c%a0'\eJC)>.\QHU84c^C$B)AJ(Zf`JQtM-f&3#.&o# +^4^ef*M`/\F.^Cf^N=MbK':%Md=W)W)mmo&Nh*ZHDh,eXS(=`H26L(?bo'L,(t:1+ +"97*kbaTr=X+:3o>qTKJ4^_5GCg&BpjT`MMaJW-#tT7c`2]6,/H#rsR+h$SJR>!b1I6Tlh[-]G-Lc]NAuS3U'X'iD:tUIgooh +!k[_(>4!TK%U9"Mhm$a:HqtPBR,F_^MX.a_(NT[UD2i&NNms3G),.j7!,FL_@&(17P'p>'c#u +As%Eea#u0G7!P\,:dt#Q_F2G7'SEIqc,(/6%b$T3B['sFE"\_oZ#kg"oAXG/Z*p/> +leH/(YQ2`GZD3trU><,L4EJDG60p:lS5@;t?Y0>3_J&+q=Js;dqeJm+ +N\o5$9RRLCCTd0Zm0^8g)WcSUq.D:DpGtheoasA,+oIWrr131>k'(\do8Em.d_>S2OaOQ7nS\&WR[M`X2d/=.BIeF6?Fl$Ag_bB.m^HQ03sX+=UgtU#b?AT21GVP#E(LX-FP.r +p0qVW$[;5EO^a"oN7E*oHZ'i@WmSbDtSW[_K,FmCS_(S*0PcB?%A5 +#hDc]Dk,d)SeKoO*1XHUPM4j&I:@<]*,k;/E;,OR^i/cD54,B>^:sg&]rRmPbYh4hDLWPF>^iA.>UJ][ +VS[$_eK\)IM^_Ft;09oW4e=W>M6 +.7aL502:nBR+XYU]&Po_m5f3*C38q>fZjf1"`WoeDs-cVQ:F0tF<+6ZPIa?A15TGm +XR3DEX'0UXb50hNC"9;)NC^[#b+N:jXAl:G*`YcHC1^9+jIu6oXeI,:g1B=@S#-WK +9!q:SoI=cF*,\ZhCf1@D^JHLLUn0cJa.>8V-;mPB9l&ocC1PV'jXfWW9/UbPFhp\m +(X+?S=ujH0l(]0=]pae#.KOA6M0,IB=74ptp73gU:nPZtY4I +]PQH7@!6D+lM?Z4j+MRn=MVRXc&11384b.e.X;6;,=]WS"pJK``T-)l'$JEHS!kel +C8k5#ZumDeNUR0Fbc?CDjV#a?YuMh%]P>ZtJnJ4<[;Q+@4#V\&BFXRYO9/Z#>??W^ +%`A9#ZQas#RE26,/-%M=;__`]-A-f]aM*CI;$Od1BF1kCO-u7A(`*T`P`46qn7f9s +\&L.K/KWjqk1c-n=Mgk]PZBU,:t9>]K";.g(3WmsG$F-R=`HN?\D;<3pgp^2e$SEf +WNDJS/XG)_VJ3H8WE#fEV2!2,!sUfJBuh\[Aq=>fA/'bnAf?ACoRkf-!BGIT=o*Rb +\iuPYI[PriBnh'fSL\[ukG$L%LT?pR;a00BI,#7[hDIj?/Qh7^T].CBKbIOQ4abdb +r06G3$;LmWBA!LJV&%8f.qZ&/@YmN)@)(JJZ,SQ,c6fU30"]7TXmC9VpO#54coH!6 +>9:s?2q]r>acaiS.:>+L0^M(#M7/KIj>$!]M0=sajM]eba4]Rq`gbFU/CFY/MqWD7PO?s.\bR#g;F8[peh`L;"O20S:c +T/8(5gEus3(P23O533_,o`lWX/Dt`"b',)[YC!7@[8B-#7Xh7eplLF$A.[APZ0.K*2WXX9^YQF4Qs8%OV(9i:Kp[j@VX>X]X%?)DG5(D.n'L!'lH!$WbVA`T, +\X6#RSrPh?>K#D0m!n:bZ$/[In&:(?s0b9>^`R[QWYqr7Qo3)+f1tX5:$G\0(tC5W +f?Kf78e!8jFUCW]$\n<]qXsIUAu.b&i4mp=@`#Ekd]s'.@*5cEA5YU/<%9RS'3itB +/8N9XWWe\G6JM>c]1,=)+@gsLJ!SrImY0=p!W:!_9DB8Kj-HN.lZ.M?N8Oa[^0R#N +Z$(_J%H'H_=Z1/#/mBR#"S`KH(CllJN`qd1fiR\`CNrTsK6@tSNg56?AOH0V+Ktk( +s.FL;QT:/B\DO.d?5dc2l*2Z;rcd&XpGs"2>NL$1jP2GSQ%gtbWRbQ[)6eYk&EfAt +1WiEp-)QO<-4**fEI%DCC$G`Pg4#\'/ajDHM[\-WTL@=OE@PV<'okd6SVU*+&HY84 +]LRLb431dSbt$1)?cgJeaTF'o7UIIfX3.$JWZM/^Xs#:_0rh/#[Jc"W'.U)I/+$8N +BPI."TT"2Plt_Wlf4^[8Qcr(*KX,YBAV!p@UX'Z0]^h8V\OlMM,hm\^'$btEr;VEL +3UugYIp"ft#QEo)34#BA"oM0q+'!c=)udT'"9j)MMZ7THrYupZ3g?]pYVXT)[hj=M +HM\aAo0RF\)J(s_rt4Ygao<'B!\;Ws.*eS=_7c;re6uD$@8Dfd#Ai`:$_%UE +`\1fWmT>)Vn`+r^2pKDF^?E@]gPpX46i*UJr)":+_bLj.FXQ-j-O0FOQhXE" +l&u.rk`TKc-8+c,cg^-sk!#>R3Y^KXW!VqpEe;"Vei9l;=OK=#.B_qjGk&TYGfM2*kQE(RJH)RRFRVDGrC8GlnZ)ZjJ"6Q. +r$+`G!bS&0c[c6^]HZ4E.)7D7G6(5;p+M@ho9mNmK^O0Xk<58OBJU#fhAl*3e-/,: +GV[f,M;j[BkE\kcKD7ZhkY\4#j_i0/#8sQaB5YCT^ADgGqSV7%pL7327P-b06@0'< +#fbPSqe*i/?=UeN*KU3sSg2u26ap?I+oBoWXU*a(mbP*]`VD1:.@`/o64pM'SJ6[* +PH$'a17gOrVj7MU([D't=eeS?^1s*)P5'KW[HOn[cZDpN4k^3fmKD*>]7Y&t,^%#"I"+MSb@#6k.XF2g)d#*hOB +@\-Uan:/r\B[!8O?s5$80*h91c-%CK'u5 +95Ld4r$Ums&!%%MeOjYX;<2X&+'$B@.gG?-b`D?;T*t%%6k6UYnuTu(/P6'C+<:CM +OCqiBE2XY^WuSPs#7DAiW.m\?(X.I."'26Pp03iAfHFT%$g>!a_>d7-heAM1CmgaA)N8-AYocf`mM?n+Z[,"g%p86g>3.YB4;QM`[[R;Q9PsInI-_U +LJ^0#W9urN2h.!s+5+VolA?kU#.+X2._5i*Gm2;f!,j1X/'>$O.P=e&S,u^H+WZA= +5+AS2&*@oW0rN]YF58^_`?VkBL"#>2D0@OHF=jHc@b7/DWkBrQF/+j$r."P2p+--;ZTs;( +s.98$o&GH3n+aC5@fM=L%/rdfV>cOurZ_f\9REh@,D,q-o9k%;/:U0%^nDVE%_.Oj +OU7.7.EFLqa9D?GT'0%hFM=@+d$&=cj-WG3oN/)8G8mq3hp?V_8,^n&eLu3jh[>]u +R#q>V6LGAJ`1KGNhiD:)NXu36+#RQs]$Q5(P>d>%1"g<`!1;LL)i/Xnc)$o0aPB)9?(n$]j!gL>tt+$ +JH-JTBaaY!%IB:%_T2&^&)540g4B0T&%f',HE^1kYYY/$cN]4cq7cr3&sC!G;*_l0 +;aFenhbP6W]1M:s'9mD]e?=Oi06hN0%`uEA9mPSfh7lG8Y**!@QCGup!rdBu2&bKH +b9K]k_3Vn:Ud4f&5?h(on#B$s`Nu+$R>Znh>D\7K&saRdcTb'pI?M&6.-^F+C[2"2 +rI4[m:<_KgeEm.TftGJ;qk;GTIgOIC_-aXp!n9YuH@o$7T"Rs"dc!b#.M+0>g/7nE +VkK$a@Cug\1'Q_HLI[!bT&ss_)8CnNbG +!;M;#ht2r4oY1>6N*lZ?ib*f`"u"ngWFbJ:VXV=kk\:r&rh0_pmJsgOlne=eJ**1! +/>N8R!K[J$T7g%Rn!5KhU2o_PaSl@ZMh($oDpc7>UNb#oc8mm@W5c*IF*VfhZDbgm +aDNF@,a\T3F/`JK7(5HchD8Br>7[:c=OabEZ7Eqo&XjjjsetbX4 +?R!j)O!ppsO/7]?1O(CN+o1mfjWoCl](fD$XBl4/!6^,4s)R&tN)no#5rCcpegph# +Mc,_,V,&PK9dQm*F=mf>#P])a?bYj` +nMlVkP,%&IH`f_?NZE?1R^VV;f$,-Wj1ltk.@\g:=YN2'T/!F9cmo.lo5k=is#e1c +M%'8*ZSJ4gWPG;tA6L-7X+Zi^i/\-C!n20h%b.4BCfL]UJGQq"/dTZ_[m9"-YUG;! +!@!#D5E#Z%;NT82$lf:q@FCieb->8\=$:;$#,m=8+"REOJ"Cp7^`OU)CocJod$RN2 +n6\Ga'09J>[BDnL!9o#$on[Gg#_/^I[eS6f/TV5D!g^AMmR*:r!q?<%Zc2\ +Y+4f1lo'/eRqp,gpS']%G1ATRF[0&Oo]<(UO)cS\KLm[dF=E#ic&[(n96B/ID)j-T +SG\mF)\Bel-Mq0F+&BdQ!.b+(r',9(fj6?og`9HkfV&lVhjj^lF7%0BX_V;DZ/C[E +i'6K1F4:TmOT07Mq9^./kXiKRIOCFG*<]=H[^-$L+Ir$nPCrpkUR.-qkb+._-W"Q^ +F()._=T\*MV(i_[D_oBjRjg/+4KZYC4MZ5O[pSURq +nGd\p_RG0MB-#X*#4+c=[a2d--XdAp)#l@bdu@=:6/`e.VN?WCN#5@/&h-qG4I3"+ +V*d:_%-))2Kf[9VJH(TEKJn1qJ'g]$+V7k=-JqJ9%^H?nnk&;:P!6#l5o/tK2=0un +m%`n!lXoL+/)aJIp:;:AW!gHA2gs?AK@lM8`,Bmrmfhe0M^^k]a?NF]5X.t@J"/RL +n#*,!Rj-hfIY.EUM#=*"$7Wck?HUGLb!@7`Z^S1'.R0K-4I2u#ajRa9CJYpe!rU@N +Zq9es@es1S#LUtP:k+-r?[lBu`B$YrQaHOD\TkoO5"M&%?nFT<]mGi]k<>6=n7hs;1^*@H +k!d+gdpj>0pcB;1gZO7!:Q2Hdp"qqn]R>8,E#%T:(_VD;ZTsktiCEAZIh4c?k7bJ3G^ba$eZl`Z\b5>9ct"c1#3SqkTUp6Xd:5kiS3*h4,Prnhj5 +hfS;@Frf4Sl._HQD;KCWfQ/"aTlo1rDRjIf5H/"I;?2p'=gL`R$(mA0]jd*>iq'*` +Z9TpoVofiSL`KE+ns#;*l<2Rs"cN4g.n5dX!qW(I)UbY-)OKb"\_4VQ3m&Qd`@j./""j>'!&A9'ppEE=7kG_gbT6h%*ZH+2JGBd +k_\d'IBPs3LA.6bDd,);fJ[YM2;L*l=Y`6d%H_d`B!ohEoI;$ZLHGXP]Kj):9TqQ8 +SN!Lh*mr)nR$CHMiXr]+G"gGaKeiU>sIe*?+/lo +,IX`n1FXA#a9VJBo-`q*,rV#B!E4N8VX3V6hu\;lM@oFmi'0nC*&9+_kHaA<2(8I5 +Wts1M7C5,65;42*"bs,%d&)>'%1&OCZ7-m"/fqAK)6tn;Zq'WS!8AZFKTdpWnPDpjjT?4mU9)X?n*ailjuF&"[T*Uq^15k +lT>OBkTl4pB7Pl+^T*']ME:&u+e6UO50uc+hgb]>q%b +f;#@`V5VXqgZDJ^iEXgq>h/7>d;g_Op0pq1?V">\Kp"ju@[:;>P'%Ac[OAts.)EGJ +_3rK>k:mutPAJ#o>fmnptu#UreUS0*ArF/l3JMJbMn3LF-NL*#l[E0BQ#gYJ;1BNPT1*V +V]CU+j9S?dF9&QI*T>P5&18$N+'2n;ESu<]bG?qIet$V5/ZR">1`s+X\_0Kk: ++'XYrDC&b=C#gXXniMPi=65[uQ,Ha(W'Kt[FkHR#KNCkNY^hegYYRq +`kLIFILqZl&sN>L8cI`aktb4:+/XmES=tK40LPZ>Benn-C!mO@"OQlVjDW#uY6C0: +a/H3=.5*a/nKFG&;QeG=L0=gL].V*r,g8)(cn`f%-J?E`)'r+dKL"^r5hA\:O))QD +1X*jkXQ`6VV>^1C#P#-IN7n0jV>ZXJ%p5k;N0T"ue=uJ^q!:c6,R"-lCl`eKJe<.W +%7EWEmfuO\$c9W5SJgte)i$4u.-OO\!id%NN3WPiJF*^GZlO*TLg'kBm)&5ib$J:J`sBC8pT$7$,N,^^]@*>#mQ +&3oIE.t-b/I4':0a]0crS71IQ^^@aq[-?!6kkt3sDs?,hM="7>(&+j3PMBW12huW- +S)fu'oLL4CREsoJ15#ir=W] +ir)Ne]ed=u-V&>BqDXtV2JCoN0)_7l4^5)&gMVGOY5S7cKD1p_ck;FO_i/#X:oe!H +W-$%h*;C&\s+G:(8.0;3N[Q7FeX0hr$,_Y*@nI%iAYr!]F'=JXY?/)A=LtG4%*Y%Z +j"0D[//si!BQBme3AVaiqb +>pZe8H6s;H9I2Zo?qrLj.N\nV5JAral,e?Wfs +kuAjgar_)#/4BfY[oh$5sj-X#42d@prc]h7NEr_CeK,VniJGp +gN_]C.0V?<<%T*#kuItRd)pn%p\r27quU&HTlFO[B;!"\<+m9(V)PY`8N +8+VoV +NNWN+I[j73Gt.J1Zc[ccZr(lTA%-H4,i)2_sj$8GZub>?73=p1u\U"FltAO^C<+p.fAgb@T!Cs%O; +&k_#:_ff.XK+.%O*:g`;jEHIemhB4!Q!s,\@dhlu"d$M$5&/&#mD-I"7JjsZNFG,2 +PPu\u5[sl;bGq5GpnVHE=k;,K(&jiEZBrh%9t& +1:m0XkFO$XIa#BG^mmJWRb:K.H]YVEQa$*9n`HVAQ+b7aFOX[KdMK;r^E`lBme`[M +Q318Q2nTDP4N-u&O+Mk35N716guHK;LZ,08,Ws"(M!nU`hDm`Z8d:u)!bbMWrP[(6 +3WAFm!Ed./Ns&en!rd+D'nYr5+#ae!"8Tb:2SDQ##^iM"r(k1qEC'()r-oWr]]1do +gDsD7HVi`#io-@knsb(G[.$1c%P6PcQ-%jgj(kg2M[jU5*3\7Bmb0lWI!BOLkEKr_ +oZ4[dA%N5GKRAW9S]E'AbjUimG0Q\iU_"Noj_-YbGHUksR<-pIV9H\H\b/_Q.e*J/ +R.V7-kgJ.ZWtUPt:&t<<:CKHI5'5,YGD"JWjkOpDhR2Ut%?hg!^'Ch>pDc7lL>ng:_aa48Un'bX!#i^T2CLD?S>jJ'Hlcq^_jHWGp6TNa7R +Ps;+:4/L)ke_n_g'hf^mR8V3>.C"QK%'_9\W1:qmNuM.o!"/f+"8kG`)Jk60kB.,c +C7-)_d>^a!*(Xq&G"g,[m+CD]hJH'd7/J?%GNh9&p2,"*)K-NpY8Gfa>6F_5MgD3- +g;WYe753$O'MlQ#BQ6ORn/p#u)Uut8pkoRh-Okt:qLp&S@577YBO&D1/0*+l)Dabs +!:ZVbWgU5sS[,5aF_I:dCmqVI&=SsT9a\q"nqZlu+G!ZMfR^sRhH;^M?&d1Xif@BY +fXcfU>S-:T +6534I?kT5R2Fd4APfhZ-Q.+,1nF0tgW_]<`>dmY^YXd0`M2$QpCf9t&ABU2a=2"c;k)QL[i,qg\4e$^4+%N*;aZXo:q$n +D\d_\=p%.QO89%W4/E-@-O8'uK' +J)\Y$5=GIEPPk7qiE6JqPDQCf&1.I`TsKF$H9JG$E[_^@1\A\rhVgfbD82ibDcU7a%[Mu4um +n0AfsQB/^-+hgm+.SIEl+F_s0!<.K$:&Ll\q"ldFEV/E#im1N^?Y2iYRmW_P5_#=P +rHWtdo/uP=o3_SER0?t_c!eEfhoYZnRo.$-]nC%cfRrLcN>frKZ%@Y(16AB\P,MS& +D8ZlAcg\D;n[2(]^+;?/)rq(u:C_Qi%!4,q4#n4GGrom4H-Y6D.9d=N2kQls4@r`W +(O3e`mB0b@h0p9SZK3H?51-i[HslOcJ,edZIVfF7pjLVN"mlB4HM&>^%`pVb;-BKH +V1Z"+7o>*B:SLn*!q=2^nBQA^VSE9$Sjhgo[UaND_#cO@nrO$r(4tttG(B\nS)/aD +^1)UWr&CJbp`0LA!.[:If5GsgE)*hRYjsRGf!C1%GS[Lo:Bo,5q9?\p>QFE;Ih&$q +pQ_J5]tVtAr+EhIp+N/KN47U=ef:,[m91W#h`E-/FFNNFQ-&F"6^*Oqod\qrf\%7g +1RmHs:.14o'#?c"AB1RYV.!,ic1DmadLZ@:?ggAs!P37 +_IK"Ls$?JE&Z(j(IR8h:j-tU4SV3W]1:'^QTbK'4Q +0`o/NmRGY6)Gn;\) +bZD+=2LOH4HN:Kf:q?i[^QH.(i&,aQt'R"GR8b@PjqEJtHO$Eaupf7O[4AG;,!*dI$l:e +dO)NA3XOqbh=*8G+GTHP7&A'5MO.^E<4S0bHoYNt;Olac)!Zf7q$Rc'Vl3oo;Oi8g=\?W>I/@@b!jUHU4BA@A76De%1'DREMhU@T[8/$G`m`iW&r?^Z,q+EtSEq"`Rp@36V*cbRS\[ +$?c6rq@A&A'=WKNQbPZ>P^Sj'pm!^Cr)I6NSH&BSr\sbQTgPN&FFZSZ^m?Y8Ur^W< +1I!>4/g\@8PT4QO?@_Vh-7KP6[.$GZLX@lol'W,YTVBA'71%#X&magekT*e9UP6/* +'p^fE%IRNRHJ3Z2omllT@k']6e:*uerR7"ZT-b9O-rC'FJ(mct[$C8i2=YYeEANSV?H^aRp^PW[.CH9ZcFSb0O%!RK3W+#?ujYgsl9XZVNR +q'^8ll\]*CjNn=;D;M8PrakUug$p-]daeM'HUiIg*>dSG*>OKg>XmA,?3(E606dO* +>MKaZftBFdhe-Qed_r9:r3bR[G>@b<08LbM^NFo8/7@fA32?P!_lp`ReEUhi:BDV_ +[#P8&0GiLGJH*+_TP-Q3$!cCkfF"@$ciQD*K:RIQP0Dr=n2+fuh]!;aGj%9s(@]gd +qS@G#.adM3H^iU#C=g$ln%O:/p[F=KSJ4-@a1'9rPk35@4P=9>CpHtD`@41"R6e&M +-_V4sMr6IBZ0qS?Kb'?uSjtHK(O$+0:pB<+Ja_+f"i7Y1^m8d)NZ7!X*W(#1s-@0X +L0`P>gU:sSc:pA;44^O\lAAQQE*PD-WgsYs'l2mQN.AfI9c9%NL<+j43NA,YdRsob +;2>se,Y`(;TJJ^NfJ=GV5^ot9\l9tk7P/sj/OQV$#k7$!7=iIWOX$(rSp[V)[L4W_ +:'U\nC#f9oB='Z$$Lu@J!Cc)0.0M.:d?KJjMS8R%"n3S(iBR(rn(W!Q=C,4\.tq[: +VLHd[DB[ZWbcs9L*(VpK2"N)U_3\\nCZc+?AFsZ5O6X!(L0UF$*l)-WQRC99*7rFW +e)EW-0WN+NkniiKMT1X6,o.4TdV%q4(MJg"NXZYfQ)"3O6TT%gH0Kun3PZ_>J13NF +KTdL:!6>\p(L'kt"m$k`J1L]h(QNJFNR*o/Sq8')Juf!CbfeM;sGp_ +?s[W6GftSVq,(M1g6c,bB2DP$our1]UHfP3[A@m`TLG;q(K3OpphAc;1h/0Wj#%qqM<82pi(6O9eY9EtYU\)4$\Iq8N +3*i'<H7(T@W#OidZe!IM,P. +(o)Q99E)Gh!!2lrLO=kKSd?U%S0^,C$"`7$[PjGMLdfEoi=mi%8r-Lj`$uAY<=-lZ +KOde.>oUu<`OLY)M&E;@S5Tcc:TLoeHTqp3E_X2#Q,7`I% +_!WLW&$h"P\YSDb`-83eYHFR?VdFV1^)s$G^r--o_Yu;Zr7a%8rr95krl6eUn&n*9 +s*KIj[q.%0JL*>'cO5t%c4PMgb'bceJ&s'd4A:!tLN%3bY[(&==Wp"8TYen'k[\DX +`TWYAR:Z\-V"e+@q:MNLF\9:2!/Wsu^Kn.[@oAt8Ej/JGO<`N+WBSC1b,X4B=a67C +q+erCGI(fqO:8u*qSS]>]s-1F&bF>CQ6(h*DI!mXT,a.3BB'*onR;n#Y)3^@VOrSr +4Utc5:ZLJpHLYTU3<('iU-num!>4O4W,>DH_?7Q`HeqYB.-Q6onc-7@j3/?fri4QM +N2Yo1&F/N^n!BRK7=?=jV;;u-(Vgsjn+]M!n^Be0NIWBbbE)dUSqJYG!;Jg.PRbe. +"[LC$Q6?jm]/"MJAa68Pb]CO%]JJTNkMW4ah\rV6anGSB;s/e^r.Bpeh/d'VDU!TU +aX*bS\X6S:JX/cI9HhfI.8D=Z/R.'V"f#AP$epRXq&jhA_pj\/dHFXr%:+>s2#kdL +qU&mEmYTn>f7>NOJPE'1gNIFXCi=.D[0-=cCsW=C&%ZW6Zt`Ks2Iogod>_Fk*._:j +5V62^/:1&N$2?_5+$$D:fgcM:-3O`R_Ze2A8I]d/InhM4(<&,W?!7Lu7&Kf13C?CIjB9-_;%V]R9 +lK&9[3!0%`OU`4$JP_aok@W)27iOu7kLTt[lR/4M +/c\Xq(o8k0'5gN0I^BK!>60k9O(I:+A*#,@C=&ZehlcU5kHqN9]:r=^G,&IMh?^XE +.';W3YW%M +9*_$15[timU[HYpcW\\W9*GTmJbeNOHI8,bam",na]aPQ[>>R!P*>:-PAQNH4(g0i +c[5E3?KmsLm?nB#(PL?M"r3ED?RUak>nH&4NHO5-fQ._m3!o9YF%ue-'CUmE5\DH( +A0okPr[1f[_u]l]pci$W#JGDt@pYZ-"TKSIs&f+0s7$dG2k5n45[KhY9^]CoE'i=j +q1#a`n>>s_^A`b"5/17uGoQ`Bs7,mac%/ZZm*#tl`?,ZP"Ro3P"og$FpmR9B;=OUo +;Bm2JBi-]`Ae\NO_URg@_#G%enPY*3?'8uE#/*7Q<`j0$=(-3K61<`dp:K/$4s!Pt +H;+#/EL#GQQ%%XArglp`Af)Q0J),oD0-ra%+l/)%:ABotO-=0md +Tjj.A=L9h!YDP%lZ".]&9c)"(a#HOjYpg:/U<4+L\&DMj0Lu6@4UK"R8`LU@M)QdS +`TI*GM9r(1lt/5c*eW9[JH]GBnd3W8d9c;ibnqoAe5-'0eg^B4e)Z#&;'G`7f*_1X +Wt`oF'#iEJ%u$U4B]6.3)7_Lq.W[.mRs36>/>b0Xfrd&i2T +cOVqU$F!4o[n6:;8G0:i,=qF2"f9;cf#l97dF&)03SnXJ!F[dCVM"63k?2"B1Ul5c +fmL_?QL4])mR>DaMI(NKG8&ujkE/Cj,*sEuLKrd*L%2(H@f(C;7b!)-[t(s1E^#X@ ++4ku*4BTDJaCV1:V1W0O4k\5RXXb%>"ZFX6DP%qR4#IFu$/!LNT1SrIZ#%NQ$l4BJ +a'R+$.Qn\4$h-B^#in/=^`S"SI";I.(\!1ZGSYgI%Ef@bPSXJH+'b?@$[?c9qO-dl +U]?H.:HUZc3]PA.r3(M,)/0=FbHB +&T*nIOBb^;g=;Q7Xckpd5C-a6*5eC#-EsnY@(-Ulkmdjf>2NP!<6PsI.`HGA*G5,).3K?g0%q>=Zjr&^.'I< +=N9j8Xt[fX+%U<[odQ +"AN'"9NbF)9hQX@OI.9`:KZ-Ip@3a_;VV%.8(8biA61_Md^!nfHd3'Ppk++j!.Q`: +@FF96P\W)#IgHSWOWULQ%9Sl&SEfpW=pY!+rkf3\]f*jFrt0PlhlVXr;@E(+F)JakE+Yan^(H>!n*?G/KD>Hsp`INUDbBa[]H%%P3<,3@k_I!(0`>kfb)iN!ZhDjk +oiF;V^DTlHC0:;$@E%\i'bLB9aKP6_%r:%@\Q9)j<mY.sJH"-j7Bd +]sop:+,.qu[J(2;+F"sb"i*E`P91kgj!atC/%5'u@c_%n19pm_!5InaEVGV%Q$k_I +!&"?Oh`uKAZq."L*C]Z:eWa!^M)rRSNV=*)]o6:: +q>^5IAFIb;Y/_s/s'*e)GO4)9ba[2.RdO@ehQ^b.AT<.9m`'+5QFk`"Djf0-BGX=l +El[9?JH+Z0)3=ph<)Uc$R6q",^ +-:sKOUrk,!7>=&[)W391%j.,G)^Y`>"cOs"N@W"8(;Q,p'.Pdbe5E+])$B/DUW*W5 +lK=d^5lS[&h)Md)]A:j=KO26eDq=T2<]nJj3$o +F\%3DW59J(80-5QLN--RGt9e!$(dCO*s="mn6]^&+>tbblK?Jr_"WHB6N+5f@u:^$]gZDkb1LeZ9d,S&F'!p&m859Pe]j,q+b^_&hA%b3o/CEhAVi%pIek>VO.(]gE4"86;3Q3A;+H#:R3 +2iQB@#2.W$1'ap3G`Hoo>!:3GG20ftB3RN:>D$,-__7`AJu5%V-Ks(NIe9Q=WD*QB +[W:0;>2jX@g:_dOULi#_Fl>_M5+hO`SN2./r`$r/\Xq?0rG.`(02\E1H,7B=$k=*e. +1%tFJa9;S.kWF@-;be;q/"W=M8'?^rc]#-NMNq1gLpq$;+`UtRE;h'eOhi.&56u*, +aE@SD&]b"RrgZe4:8eP!VjT;I=T5`I) +0BmWoruWh:>T-N)?!.Z4>['e)l!h3[L?o9:ZPuN?3gDnn;>dMbqF.Uc(VQd*kN`oMSSIM" +!Oe7g>`9s"09Oso8>X=.\iLT[O&g]r-q&P?0p#!;=26"jr(kZ^!<6!iYRCUbgBci. +&W4f:;@BSKL/;fD?)J9=S^E47H!pRu#QlH2_l!d8i!!c/$S\P%[Mk-BVBGk*r;Wc< +qs4O\9eYJG;%PS/H"Z#]`TGE_c=PbjqMFqKOk$KM.MtG.D[@\SKMh=!&eA +5=o2;O'fC=1]&kDBeOf3ESS]cptW3cS*r(MKAKD:o5=IEfYNK5Q)?>JfE9kOp7=JS +PQId\9qD?l(Ei%iK(tC2!5=El3-s[T_E-n#jA#+F0ET&b%\=&5(2mIkDdPbr`AZ&$ +h`EIm]m];_?2-jjB;G=QJ4H:]O$K&KACZ8#dLbp1`rWOAH.8acM_CfB5*&qLcb_=C +UR1o+?%@Q#&d\K/q+SQqR="?RI<8>*s7F[8HhVA\q^oGRFkcUaVqt_RrnB%W4JV7H +3Bn!s1(+0lE+*/bBd=T)Zd,AnOtbX.gVg!@iJ/;OmkRoe96qR%]5d1%rTJhMh>DC: +]M%]]`A9*Kh_JouY\;&PJ'/15e#)rLH/52i$1OQW7E>9o)Hl*8l(^kP;<M@[pt$fOs7 +L.M;!r^R5.UX=78>R]W,HLg.%PV0Ab:7^<`<=]kB(Pc^F'^2S6W]!Z0fQ+KAhs/49 +?KFn@]LDM;"c]Se/bZi#SjX`%U5TkYVDN>5ePaH;;F]0!q\0\K<>i[maZQ]T;2sX& +R2pe^PaTE;"[F!RaQ.,J$[$O0qK\`:o2;PZO[+s`WgP`AP9VpFMN#=So#;gV0J8eL +0*=q9=+J'ik3O$UVA4KrOtjj`M#8D(&6ma+hA#Qt?h!=\ncP),q=sdsU]A2CoSdRl +e?&n[!1a'5g9kpuRRtgPZ.h-lbTWQJMa-ik0rN/8X8FUAHm!p>ITDA5Aqs5)U^90( +RL/.(YRc:P*GEm37HspiK+8USf`_)qN9bUh8a9Ka6aak;#%)%10=ftjml'Ri$CX39 +Z$qc5,47chP%QDjOi*i3''M=T3K]>0!I6D[>f?4m7lo\o^dABd +1SYt?iIB6D.YP8VhjF9-eHA(+r2I.foCMUa#K?0ZO!<*5E;*u$'&W26T@O]tG0L&'E5YqM$#ORu:,c([44`SG%=2I\Ze4>= +CcRYt2d+0#13=Q'o60KTIdEsedN9g?T;BL+Su;R/"\&Cjs*roqnc-@AbQ$n:ikO'0 +s*76@q#@f<`BQ"(r,J&!rIjI&isHk=;."KFpKAthlJrc)PU8Md"iEEPO.J&jQj\R[:B$#0a#2mlk99`+U(LLEm4$\an'[m#S*It +C"P#,I?D&MHB3LKh42_+l*hb>c8u'Q2^/Of,gr#mK1B-;sKqH<=I"u_)4kAHPn[H:A*_3Dc&aY9*!m0n:S](?= +I1BNrfJq/\EuV]=EqoWF0n,PbYkb&qq%q>+n;dDF1GD;"2!3[lANMepTf'6?nld1\8#<0CMo_'4NuO?oO4c^);j=F?C>qD*3tXc,Pp +"@dH0KIZ/s%<'TS:tl\BVD[CL\':@"/6g[WS(B3\JH+hVQ0#D2rN\HV4Foi!80qQC +RE\f-),X'I,`;Cg7 +HCVQ[O^AlriuA:&U\"M9EL5u)Y0Fb3Hqe5C>UO%_K/Q.Cp8'GQ=;Q.Kk1q`4mP*VOSTVp@q\j_$#7tFJa9X +C`h#QrUS%@%/u0_Hfe+I_74Hhs/,8;HLU]5nCVL>kj?I]gJRn!nBq'iJ`W8MGt'/7 +TsQ+,^*dBkj,j8'Mj\R5GHc.ZP^h=&EdcR'rg%>+oQhGM?j1]lOM@X2rt!e8)`?LP^rcFs5Up[_#MG(in$Mn +:IhAPpQM4l:j/>Lpu7?bXko\Eh%Hnu'lHtfAB#+`9(fO-j*sOK/]6HeYG@E@CR=7# +#o8W`DoqLT8IhUFqHL6&9sXo'ZS+Dj#DC?C>@/+]^W9Q-SA+h^qu?0Y*^F^^=1\Fr +DDF%G$%tR#d*UjLLd]NKFA=YQB2B^"+:pQRL*l'l%5^'PPgd=/I;R=%S!iFW^DY@V +B:M,grbM6o>l!+FYPRP6T`qBtDUn4Qf%PJS/U?1VH'nhdoMHj3I$J +Vfs=3!TVL?ZNh(URh/];>Lr^V`nY@Np*&IXZ$3kd+b:_i:*,-c*J?`"T@a[DHkno" +:'Aq@pZXI=bdg*l4?T5jAe/Qra8uF_!8snqZJ-XqHe*-/XBk:$G3YI(Dcja$^>UOG +$eY2B#L-'Oi/a?,!.V3Qr+Z3Cg(EC/LcYdomHHP<^k^t]n4,uj6Vr+:pqI7AKko$. +D=NW[a9(/o%p]!TS\'*L[;S.8XBX7,)O8Pb4ISJ.GhKGnC0`o/PlF^POX#?3!V,') +K,O7dQXU\h"INUSAs0"U]FrZ^0kU,"B,9;k&30OAB+XjQIB=/6@)]-867H9On.>0j +]RIu8iPW26kk7Km_W?Qr(J?%MG:-0s%VR,D<2(N8GE@+;ADf\c5p`a5cFlR4bd6`@]A?S&+:Rq_Tog$bLSS/[jl1%o!V5*55V1DOupH!`O:t%G_;50sehY=%'UR@o81UDM&3Z@eF-M +qSA!TUCl8/60h%6@G=\Q3o3eeJc_N?,JY\%Nof;]:P%fkf4s6#iuKuDgbA3R$kI<3QPoO`Bt%^$]GDdl;HV7CAC!SqBJ-B75-0/UXA]bg]tb=X$ecbVEj7qI` +)?4ur(4V8Ba9'*=IlJ20J:1h^9HZ&tTnJY=f.>8k?.C8C-`;p+ruHGetYs1`B&( +^J0)j/!kAgMq5]-b"'^+)?38/$H+&5@]5`#KEbNYJZ4:lcX=$SW!#XH^sXlD:]f_! +P]n"g2aT%3KuR5O.JB-\JHaO)s0!(?IH9PSMJLTHcSFV>ga@P?c%#'rXk:C3OWaT% +H:71Fdn5+[;uWO[)=`2Zq+jnRA,cL?\* +>\*^M1OLiq""ji]%:e0HUq'b[%:O'n7p[Pr^.hmYQ*bWE-NUnQ>hp2KluHb534$F# +6d?[sD5[O0,Gkra;m4jFDn2&*XG%1dKl:+WG7enO!N>quaG&u?mfbb2S,pIL5(.0- +Hu(ll!e/6AhU2dlj>g]N,[1ithsqiAneg)&*S@sZ^NHSqrJ#R]8rLWm#n78_'a*+9 +WSXkYS\U@FY5p$h%RR(cKqJngBJ4%.>@p$Cb6U=n-C\$jMFmm%BK5LY/be+]>?tIB +](]_eT^>t>>A.JuGjf7jrq3E7LR&CMg.c>TiqXs#\rXjR+&-#l)l&X>?;,[9Umc-3*/(\^1\+#mb[T8lk,eQ%OHRjAo,'t3qM)qNiZ:lW$&X8 +'5s0MDj4b`b-5[Snm3o*!Nc\P]c*Di%=Y4&Ue$'fUVn&ro7ti+4Y!fKq4KX/?OO]^ +$1ig"TL10,6SYB:%U:-^:P/`b+B#)m$gjODk[%`k +/2Om`&a_*2]j':6'u@sF\"TCb?Gm,nj'9]cZHA'1K^bVYV.K`-+oFja-Up-B7bT0: +/WlqrtGF+&A\+6*c@5T#-u&s+?d[:fE-$k +UZRVrs"XM=1B(KHH)'uD8+:^[s#U&[s68rA@NP2lbjc"3a8I)GO?)/!qQdg7HiD(- +_LF\cni6OLs&40n.6MiYiYE*OY`5J^a$a+Le'QLN$jJ#M@uB[ULcO?4%on/*A@CVu +^<5#TdXDeeYS_^a5^7:@FHB#FP25^Ui=SS3dFb'`>O:cH/3sMNX(#N.=9&[!,K]bC +jSb>M!@l#12,]57.9r')djB*1&I#bA\?htV"(Ku0&-6#Yr?>,X]GP(F-gqpu>tGpGFFUjf08\eE.+gL@`n`);=`iV]TpN +mRCMV%l01>2&1G,$%,Hmrg,YE(JqRGkn;bF,1HE5h>8s`iF@Itj__m)0a4Hda<^C4 +^)Q:`)kXa1ebq7Cm[O!RpG)![KbCL!'^[`>9*oY@n:skR79p\R"@q$RQ\k +WWHLJ(>eQs57!Q>E0jofk4.LCcWBb,f8Nk>JkVekk=aM[;uof*J/u:1cG*g]nE#0Z +gnj2@H$hUP+9&_-3?WGVDr'bc].QEOa]q2g"jlc=ctWD>G=![d6RIiFQhNkeARgP9 +nC?B$n)6F5iI10W6_Da.mu]O0;m6/??ho[%H4<1>hf7u(Sn(VsM??,C<^!X*1 +;m^VP$S5]?Z#.ac@@:#uBMq$.:t0\=Hh^kFZq3i]?=kWUM^@0j@('=aR#@Nq?mkc' +POKPAdFXnFQXGW3e,+\`gYUbUSfJ'f`):@;r5H]tL*\X%pZD/:0,YHW +66W"/!feB`s(K6QGXa?)>UO*=*7M+.mijRB<#DS6UIq_SpO,AMGp7I7P\=DnFEi[K +Dc6.Ej\]s27l@;R:!g)dl;e=1-i#Y64R+Z13m8V'TMmJ9M\H:,.nB;H +ZJ%0=okG(_0S`cI"hK;/#XL$\1S2(h]4Y7md48Zb":EX +ab+nBK6lcf-Y)(IlW#.`[CKQjL%(oEXPn8)n(GAQF8#kI$;cAr+:CelXtscdRL-*o +;Yr*%)hrk0b6>SGme9sU"6'O/*t2;F9-M>mb9nEQgQt2c+16Fsah1uh)@PXl^\-VA +Vb.6R^h9>4k`mWB+*O_&Fa,n2n7]o:+d<:m#2(lF)3"j2,*SaMb>N`HGhBG5/L&9I +2ZS%!dc$U0eY#LSW_qOA+e'i%pbmAWfTf5l1#WgCCkPqrUfdbiR`(GIAVT>KYg$*tq0ZqW!'QNkjrbQ> +XeI&=i"tl)RmJpU@@5h_?Rce\X,0$jRA_p)WP4IlZ?B;uVgu!YKKZq;3/$cSj$0.8 +a281rStnCp,)#Z3>qAS]UOQDJp6UsR:rWQ:#f18/:j#+7('43V5"PbRa64-KoSG"$ +=K3NBgA)s$8NOVM+OS48!FDkY- +4FKns)uq^A:"__-n4+H?,bE?D"?4B555oDX/89Ld@D_Z;pn\c]bE2a2,t_O<'nN1` +hY]1lg`ClUQT=r%-[F"/4(KKB!$_IOGgBan^gHne;n4GON'6TQ7hN%/f"pg:;(X\m\UJD;2QmXVs,Lkgr8pH%phg-^)s>h-r&C!G3gs +0(=^DI*rmW$F!0K=t\S%k)[p=ml7XlkqfM0-P$%Y[lqY7f6]DmRo)PpY[gMD!;O?'g/MP1c*5%ScDEW2;g*6YT; +%1:so9#9nP&HM*&1:KtAX@il*3knaa+d*B(ORm66Gr63s)@9a;Edh:$6"9YODRj05 +^K?-AWm7PD$HXOF`=O[jO>K9XEk_7T'rU%CZZ:oZ]ucIPIHl3b1:(>Q1J$[#WRATa +i-CRS<>gd2oL?70#'+7LH3E`P9a:_aOIoj_9!;*LW0Rsp;i5Bh@uAbK3`Y3eF]&G: +13u\A^s@Ti^uq7L!muj:uqWMEkVN*b>A*kg>p_h&2i]3q^W3f0NTe*1=oh#9qM +I2D@IPJSY8OSjmX*tD0ji`F$M"cV(S#sp_h*sD%r=/mF'K>t)oi>BU^o'kc%eZrVq +#9qVY%Jo>^H5W9?$O/2TN7q;@=d9G#5Jc;Ck"mek[,>1RJ$TXC5N,EFh_5p>p\OiXAOF)qUH!H-k +.,-S0FX#-YM/W,Gs)B'k^t3aKliLj15J5/j!!I-<_LM1WM>o$`+=\dFQN+^a!(c@J +QM=V`^(?A?n/(-j7Jg0gs.d6*4:a6]BJ-d;Mn]Rf@1Gb*S79Ahs@TiV^aJ +It@MQpm^;*I#e($Ci^/`IGd?**uA#Z-p#4C(iF>8/i_>pr_(_WNH/_L$CH-Ng_3C: +>(*C8;`Q=#G8)tC8@sRDi;^-1,fK]kCd!WZ&l"g+P"#N7gMW-'X9h]),bIEdIi0^Q +9o0Fa8-i)15bV^>9HB&<_IttIlNDC<$@J->,lh$ImK1H2/mt[AZ&D=XXpJ->8U0Mg +C:H\Dq[gdm$.$Grk;b4].Tq@c(rtA.f6&7.n+.+cVPCf1I\?&@sFKjI?8-V+B'rq\uuFrf*K=a]`\tZgW$ca +jEPr-ckGU%)toFB?c5Up`3$b6PJ0VQ4AVL'Mk8V]c`FH"2sG!QGb70F^mo\iR"d!X ++7tl*oW<*2UL4gA!,;N75&LB$IjFk1h@"WqT0&^jm"J5e%dC=X!m_a*Y?@$4gm*)b +j]@T9])cEB*dsN\")GAl=.Qr^^]e!?em3XS>^FuVq]OspR@f%lq'Y[Vs7en*7J`QWVH]$8^ZRoJqR\>b\Cf'Wk47P?janEC&DTe.)Nr\`)`6d(X,q5ck*n:+4-cHPN8M14F#3?WhAXFAR(hRsf? +4-O;47l1m+'r+03N-==(7`>YkJ!GU].FIr7B)dNprrBAV`uOF+?!EQq`TiCWAXZ/X +etEq8>4cPlYp-<@k_JcE*cl$>UAg3E4qQY%SGueNBU&o*=[_RhJjCYZ.h'n*\K&9@ +oPinYd=P^(LfXG(3a,%(;??tcd\Rf;XVB0a`b8>FeL=!RL'J*",1@I\Bu'aLD$JJk +FfODCb$c$\TLr*QUFHYq6ED'RXV'B9+s,\"Y/(AB[qbD_r#OM=imS3^\u>dE1B@40 +dht;]Z\L1JoUjBCeP9f/N9`Tig1QN!J+5/,^ZXUi@EJ]M0&R&JVdhVr!<3Y3;Y4F& +MXqp>)074Hd62iOIb_1@1j-21T(n0sTR4J[YiU7EeZ&>`EuBDg%ji9rV]BNllmg'C +X+9]$n>EJ>b[%CWYiC?5%iTX?+0d7^8'GME^sgnMkFEP<\!r`?g8+0(Mf-cI/r`Nl +/SdP@bMKAe!MYRHSs=i?-/Y9tmiI<]mrImtV#?R_Y*I*%F]RZ0[q@X3A34CNPs;Q, +(g;t=>DnnPNLuE3:+^"4H]oLMnT[JR&![!;G.@ZYi.$Ks3fRd/LAZ#*[fBYtE.QLl +_T@kR,g1(P7r1\tTd/(e2>fb\nAc>p^\%Pg^@:#(@N-qQD'(/ +!QXD36;\=I8Ko%j%@(YQ2dUI(#]+i,n8f4+/PEU=P!9',5J\\ +2&Z=WIlXP"O+XGslV[W1U&:comb&98[1^Hh]?6m/)VqAr:%9USUJ;W2FE8YudmG`6 +qO9tOQj\>A;1;J1J/R8%":`r]l7t5oNs3i/80:I"NsSngc.@X2B12@'7/+G)MXn34 +p9fbNYYOk5Y)\=/&N1WIc[Lg[IilAp5&pC0o)(fE_Ej(O=o@hSIt""E],[t +R=!%]GU$=:adN`usGgW!,(sM1Os]^@+4::3GOe56JoU +s3H;E8VH88;[5VKpalIbY2[u9,5cIfJH*WKcE6E'HagqK/q#D.#56i!5R/>4#OqVo +MmaV[2XO`"XW%eSG9:((#7K1kCKN>FT=S8_Un9"uRGN$XpfKO)$bII@=`Ue=$r!!Z +e76[mitF(9@2;5^F$B/M5'(@#KMT8fQo]1.H$6lBf=:CIBDD=#mijQ/8APdA.W1X:eR7jSKe+<_@CGd^n85FgIV! +:aL;f8q:f=Lb:Pg]0`HTs%M@Z==F=H02h:TPOl`gh]_G!r"fqIgPGr]nQmejM[$:g +_*=2m`lH@[LcRBZKZK[q@T`^#T810e;`Lkp)cC8A6Y2rV'.KTL`2s@i%4\(Eas$FM +@CQItW7$\5!O\#(7K4h@^Iie/9"D``V2/0N8(7]JRV\)jX&FZ"6=r-T10W2<5^ju\ +f`![@6YA4PFYip?AFuB`([GZ)l=[M<`IkaEdg%P]*Op3pmfL'U"s4D3h[=tPs0r1K +E@;cIH3tR57'2iElZE2?YQ2HuB\^<'^9*Gogi$4;2D/Z!b\b^$Xui3,C61;`nHd3# +@p_$/=#9/=46oj)j&eD'<'>`5:?(%3R^lT0N\G'P#/:9tC0Nl2AfmR__,*Q>Y6EmEcMk=/V8$#O\dp=VS*ClVdZa!QRO +EQnG)=MWPLrsShT/FXC*YP7i<7o96+r#bVkp^bTp#XcK)T,\!>8d_q!n@/;rs.d6N +pg+ZKVn"AY;]u+HgaJ!YY%BoV5LT`0D^+4+&e9D_^\J]F;e1Z2;=]dG*Q-bqQ"'t! +rhmkarcmuuk!'-mO5M*jM8k;T:]pfrYsHBd60H]acJGCg0'DK+->3j"B[#V&cB#MA:iXPKZ6;')j94, +@PPH]:`&0XU;^Jj/qSSs=e^lTEkq]S6c-)3A7!4FN3N0nm'/-CGE^DA5qeR_>54"qF-Y07F^r]eao*9 +r#oei/-,JFSgQt-_]/um!1K4q63G\es%k%m^]O#pNu%k*"LpMnr',4'6*DIXcN:#Q +]LTCnLOoIeDZ)XLIClZ\)`PAfS]0?dICnnLk]6+-paJ][bE25Xh+ggCt.`R88 +6hgo]6e$U/U\jkhhR..$>L&dpDZ/Js]/R;Z*EkDgRXtL3?,uZVWu]s'NDbZaWC=dr +nY78@r(cXh>#j\e"og.5[2@,XK3_lIo][VaY+,M31X&\+]&s%b>OM!9n;2)$,qlqZ +?b#ap?>86&Y$3W+11tQUp.YJXWU'q^$o!8nrHn-j"n:K8?N0`Alk,5a!S*tucY5"Q)+hJ!Wd'=I[u7%Xgb780D/J@4)q[FqeQ87u:D+ +b$R-!4f+]+8EL>\1PSjG!t+ZAq8c=-n2@,I-H-PcTK4D!3i-5_7qrdUCDCh*#cgl1 +W`&>88p%8"o57G6LGGPlNRORMCaY"HCP7POkT/S@?ib$/(3P'63N?:2Z1h398:2UM +Hnr,,<,1TQKGL]Bc;Q1J\sWZ`mu//"%N,ui)AS7!5/>)90_H$FPLlh3 +Dq(J8O&HdhrRkt/#i0Z"_&W_.-2.rud!si.>d_92Cq?`6*>[Y^R)?f,VbR]8SeQ]EEer'7kfs0Pc +[H],Cna2S;F=_dJY6PK<#7mtloIP\_g@#4-93Dp+p@J])Aq`I__Olns8(V2tf&EQn +A#:&r1YW4Gl\U6EaaI)ZoprtT1(N\8a!HLVRh\B:M71\Ag1>=Gcp0DLQ$fIh*JWD% +r'.W?itK7YXlGE>\;pko`J&HgQ$4-1Zc1EOcS,]tf,# +kPp>8%mgm3=r^fi&3u3bUr3FB8e!]"O45cCSna/3mSD<#a]2[s:urO3.E>)KkiT+@2P?+TJ'7FT/XjiX[Qn^Ko:_O'1"_SQ%fg_>ClY'-sQm6LD_\b2^r^#$YR52T]SN*cVW+r'GB4:E&g +n/HZ827Re0Hh4*?p^^5OpM))FJ+$FicHT*p#If!hDMl7Ep9&RVfsOAM[B#Vd!.Qi. +hmsj14R1m1HoQ@+]h:H$T$/fbn4CV>"dg`oG_&0-"[sqJjC&kI%"\s6><]=45L\b$ +qbFeEle?Do=gtA72g[$b:c,A"G>:)8njMWMG5B^VS-=Ol$g6u2&PS_;):_D&<_jI0 +0Ymfo'uT?.IPl-eB_/.MeLD=2EWsWqbX?QmX8A-2]n0IXMLU[VA2cN]9g5@CKn_8[ +G0'KdP2(Tfc#R\Bb.d965-:2J]3J`&3O2;I;5<,XLu""5)UL2j:o9E;W(DTZ+@gH5 +HX'k/Y=9HT7guHF"r&8.'[QuVAmi>N@'tS6fMJ\Z5`J*h3j@)SiH1hQGpgnGJns+=no+NB$6r]pI&gN#ROjL(GES+X*H +l65js?9se[>f.YWEM-,laG,WH>eJ[eP61#-a<*9h!#hUrT!GM$FD>'od0Ki4P`kVq^:Wa_`iJk!/9l3;ED8e-/a3LkX:Tqc58`Q[K +ihn_ukG@DFZUl_N0h(cjW/#JuqlFRW":dEnX3;u]X=3tI.&Tb1;`=MF0)I,#5T[fL +kPl&D+H:*#cEbHDhAK\D)Ctm?fDPZbDZB/UJl"6KJG`dDh`N4'%%a-Q?qO=YR>h1. +^jgn71&h:R&H3:>h2Q"82ds991L3_iaQs00LL>tV+"2e0M'$CXg\aXZ2uL]c:h9J_ +s-irfs8O2?U[cb2kQVq&;TPd:bEtP0Ck,B'nG3!ll-Wpo_er0taF[h#`&`TSH6o/q*ujAmJ5+s\eF8/%]T!X,b5f'\Js>u`>],o)lD +Q@@t-/F=(9!kLop3DY=V+QXRtc%(muSsXqWP^A7OG?GCr!MEn5&H8kC?!c/S"L]^( +m&$J]Dg;'Wqc<'M@`#;QM>oT=nG=*bhb-rJ[%njt5LLWTFagg$q-Od*^J-6=iW#fb +Q#BjV?IYtAPcIE7jX9M\#I;lAP#'RsT7+X9]LTSu'lK6M/;&G&M(l_EkYDBW7gE.j +]i>)4:7qoL)e=i/;=HN`/HFV,p=2oY(uCR_0s.Mu2/]A?/1%\Y@[rm#"&'Q[rj; +r\$G!1.t2C30UV?SMI(flVRs+:^n:?%Ngprk`uTnOUnNp#cLYjX&JYh;t-fPpT>\8 +f#\.T."n=?V"4N0W^2Wer_4kkSQb!fIuu(iS]pLEo,"Kj!RO[?q1$a6ZWf)`h_6.8s+j7gYXuqbjWUiH(_.E)K[4q` +q*3t;3Hl52iTG,95i;n<"/s$J7]i;5;tLtE%[n++?Fo$kQdp++^$AoEr3f1T-^.SB +;qC(%%^jL3RT?T#9!>XQi+:3J"uKmm!HeG@L,#/Qn.E2&*nVtP&23m'/GXFkL%ar]LKD@i[b.?:K#paDpkE0m;`M?/2nkV<[@K"VP;'BV +569J0S_6W)>#-0l808^?X_Mk:CbS+/l#lssb),X/n!lmuc@i!rF +LZb&B]9:(BR`Vl@$i&EZNl_gS?lnR"s6)L7L,H+Ue,i.'D=opSq1MZIra1G18Gb01 +!VYIY8d[=J%&>!hrgUn'=iC)5UO:8ghA!\jku%5"2eICRJ)S04QR]e8iF7D2s/!=F +1Hg\Jru;pS3IW;7r#u!]G(,n%pA@`tr5Z;F=Q98!\,(U!s0(CBo"QaAi\sF!"0@0, +2R,PK_#Le"Mg$BSauHuW0rKR%o=8$DpU0]e[,3uHH!@_%F7G@h\SEf:_h/jf+V +A/Z)$c4fb$&ahpT2^4QT@8C^DR]BP[.lIrtc*4It,f5Ie,5a#WW?hC/1D(/ts"Q"t +Rn8=kc_F56J!R,7^D1SgJ#VT!QOs)pI7E=b@g`5qUa4^dU\;8bFg$D0qFGbs^1)S` +Ak/;%D=!*WT)O07jm38:'QDA`UZasn^Teos1b-lMBWrNrY@PogUi:q0\A;jN'j>I@ +!MAgn.6E5l`V6\CEIFrS4F$V0iIk%;Ti>p.))p&E&8iC*:/).?aRT"@@]%IA\%GJ$ +HqrGl/oC&u`qD>oRic\&hPY5(9AOu^[/MOM$<=mrppISNX*>e,aI7P&SjWDlY'^nr +q@,Q@.mJntqf0h@\A13"_FrT>\,`N4co.SCI)"#of]8@u&:T$EkZ-^ACDsclj]2'q +k]Q+UF8q)&c]QcDh&pZf:$:@BfjQU[hmq:Pmk\o2Ri6s?H@Z6mRj-7o:j@R"m!92F +h+!n6_#dXX>;iXmn)!Ed^!VOU[<'fjFj$J$pI`L*PtnQ69rl9Fe#F_SMDb"j?/)5n +l\6_Gns.B/k7%g=o[(&?)E$Y?U.6\F)iYa,8GZ!]=MSVld4cK6N&g(7@J@EG%Xn[' +4*24@gEQMs2m$1CpL*,o^>quEgmRNnPNgPo1ZlLTo^cZik=k,N:MT'OS;7t1Z!6<7 +g6BGm'RVEf2p7$NO,;@V,YAATr.&0YWU*HZlJ/&neuKJ3XWH&ji8:LRe>3l/.IZ"d +nn_BHmo8roWI&fgGsN1lJ9eTEq6d2U1oLRA,WY$q[OdROS"S$B:,SXS?8B]qeP;\_ +a,Z=Nh+E-E.KMno+oS3u:-q2J<$)I]mIn!VgWh;h>U=F@JQ];i),RGUJo +mQR&gYFdOs5%igO]tOmK"tpc&)&3T!'`kZbr"#2:h$\_Fm0CN(r3\TMm?Y_\N=4A( +B.U(M'(K2*ScMkfSUd-8/U1+R!MmX +3Oh;^PhnHd:f.fH?,R2pBAG+pEk3'(SLZD;&Qn0$*HImW!;SGGp1WPhVXu8AA-Y6o +@gg(X+QJ@`r"/ia92GiK3h6KDh% +T.8B?>c%@n$IY!X#X9phIX4E[VF:F>IDm!_9)EiIRL)[TIVH]ab76QptC +ZGZ*/Z$ZnE7j7#ab25'RbVt^2]nfI8,8?U[9.5Hk9`#"e]GaQ$D;p/F73s +p>`'(s&FjQT+9kqoTK3%(uVUigKA+G^MLAD'&N?^YBdO*T2PZT$)Lu;.K6t]n].3C +n!_du8m,)i3\M[u])"IV[^;F&)A1LUhpAcYHl +c3pmF$FW8.ZK)3lJT0*H!c3.5p=oX0,^9X&kJ0`H%4Bek+)6?fEfhJZFLZI&k`tK7 +mhU*/C4bYlo&;S7)^"(j5,PF\3-Y>UXITmdRr;6V?6ubg8Q^[ei+)?/o;KVk[VSg. +06ndEH7+pYs$+K*Iu`7`pk&_Ms5=.;T2)#@SK2&3aBAMX?@'q$K_JQ1ZY0@uA-+@F +##@AmR=6/2s+)d#L:d;B\b%(np)n$hf5H&SD.(o)B>_9(qf`"JF5P9&hEC'$IJ<^C +n"H*IA'Y7s+RnF_kjZE.:^]N'!#Dg,dOi/`[cH)RG.hDOC<,J%G'NF[5nc=]XQnDR +)E2K$p?^PBXIE88C6hR:p/eDbK2AW7]Y$ +3S,dnLsuOc)d3uiHdkEp<<l%4j&"N1q$YU[=Y/f=djaGn56Yb/ampV70 +Qo/78'Fc;-@f#,n_So/^;ee$cWe*FRi%?=Z?=i#CL#="ME%C'P6hICCpT@KfJc&kD +Ee+jrP2KM4F9CZ,'0PR@I-TgAS!3u)r#uT==I\!=_s&;mC5Mqlb4K:W>)Na$6_,IM +&TjY$!K"3odHhutLl;i[J80Y!WVpJ9L(3a/0,4t%.d4Wl+*;nm$Qo-GW?0\UB(Z:) +qSLcDagM!@/%#bF24tiA$N0`S0FRf;E6FXX;"scIitc44G1ACW<=nUrk'(]I^"'jZ +g]lEg\:kE=!;u$HC'"G^cTCL5gYJV./-]`A(4Dpc.GcW3k, +Jq)G2H:\'N8C)$q+)![AB7iBi+IITi4gu`-fSga+Nh5*'S/tHVbcHA%o`-0?PG>@%2jr5unj) +9AB\+Gn.@43<4Q@9K!f5PK'h]QV<_4ESYP<$&F#Ahs\I95rF+M*a5Mr)b,]!B! +:O<*4UYusFpO=a@b3t[l?>-hJhr9-MT5:]hcOp1e!c6itXce:[Z9jNX+q;`KB3U%e +#8=LX1H!eT20TG8?9ZkdopGZkN"#XaY$0r!C:Ts;&WnP6hj)<3HDp>I5iG_7_R"?< +om">O'K-]*0H>*m10j# +D\gF/Sl#[ds7;';s6`h\Yt4CPZdSGKeVtsA$cVDkGLQjQ<9mFWQH,RJjQP^R/u42i +-3d\m"B'gBJfmZbYUNaI#r\!XJ6?tJWi_<&O(<;_DNH`%L*3aQXkCMfE'%rj4bt2+ +71e_q*F^$*AfVh.%Ib0'=NFp94"Q"KkT^JK]V`s4lB_?dg;S,3E3$cL1p1DRp?r +\dn):$lng;B&C=,hk"'0Y']&t2nHF1>u0KFN/rq8B0h%OT^+/bHDEJ]Q>a.641;I3e=MVeAX@:t#)4f!nPA,[70jVr'%`O5>H2hX=X]!C%&S>$n +);W=c((`)5@Rar"2D+4^,NkGHV\q(DmX89T".Go87!I/Y-f/Z;\DD;D]1pPHQ?]/7 +F(nPADB]eZ/$jdKuluh?ML[\6!.KWX3QcEo)e5rZX3<8?BWI4rr1Sc"p;gku8%- +^h;*NS@nC>^PGYF`0P+#eZX)DQZeF8g(Cl&9qZAN'/\FpCS3hR)Lr\DU0on=QthSHb-oWeP7TlAE#l/pjt/G]p4cp8iOtfDL^Qp2>]r +mXu$dKb&%LL,L7\efstB+pUBViP"Q9T`a8toD"$Q"Nauer*I8,DUR$X_UqoekYC.4 +?L$2:;lI`MpN@Jf1U=lf1Vo6HAlgoGWZf9,!FMf(47>C\td +RiK/%C_s$mNmQpUrs8Dd,Q=SsD12J74Qkc&5DK*fr5[]E^jl!Qm(c\K\CK!f[h((J +5NM"':p\(O8j]arI#pp)JEb\`(OCSWJ%Yd`qRZ\OCB5Hhj8#>+DXo[b6%T(N=b0F) +Tk:QJ-NK\:Y(l^q +5h8[U@@O3pGDh"H@WK9Ft1*]*I +0/:EeR%_LY>"<%W/(X/B2oSUT$KZG/8MBY^70>cT-6W)A8AFGVs3FOG#lhZ361Dp= +I\NTDZ8lj%! +rX6V[j295ho/N1+*5at#AJ\.O:Q)`&0/U2(RC/R;r\e9SF6`=W&sqLQXa\LJr1(7n +?2h[$o:F(:@/G$-*pB*CXnPEQn)@F+`(Jp[Fb(hmJko3&#Zj1qQGEao>2VnJX2'31 +L?&GXnIp-.1N2qjNq=??iH<'iYSuun"31(f-_JlM?a5sSE2@L*bKn(=p\ZE4I)aO- +n?)]7X53):U93`egWhsoY5S6hB[duc!;N\7GR4?cJ>Nq_:E7G&Ks%t.DdMa3EG1Zl +2M-JZ&H5`_l^(>H+S.tqhkJ\SegXU4FRt4,P4O_[7Lte;5EK2t,W0hVTNaC$,$V-$ +XATJ@8%=kMhpD]^um)?>j#F<^R09d4GS%:MHY$ +C*cVg`Cun[8('u!5Fjf]14A.Mg5&dWY6kO0h^mql7s'/O\G?K;.DSUu.%!1t1'&29 +6o&Q*EZ7*d)EL6(;tU]aICn#,>)9?!Vo"57"P8b%CD@lTQ*+.t=)&&N)AG@#<4sA& +h8mU[;m=FbGqD(SqV\a:PSJ]G,^W'7U7W.LUZK0i(#XmpEk1C%b7dYe]9?u;lmh',1G:HJGat8b!@V:]M:=m +Q[2?'H5(5tkSAMF\*meb)"e1Z9'SdftRl^:3=fah6`m5(V^VJ<_mr?9q>fZ0F +$l:.Ys$pG-;e5_Js(1!mPYkp6r$22>JH#p4$@\'SkJ++uK:5"irpTY9KG*3i`#B`AslU(N*>3Lqi^V"13=gfB3+HbDO6HW4jnR]7Xjr. +q0eO`k8N?@CX>K5T7ACgQfd&CfXi?Eo=NX]%^qd[8Rmus\AWqP9]V6l\mlK,;&?1` +&hB\c7:q2c1[L5ap5"ShZ+t)oYh!$9rP$op"DM#3\=G2N4I>iW#`X00LIT-FG7oDU +6i-J[ibui%IQ+D"a#R.M5FV;MW%K%[7fJH_[9noWs(P$sJ+or?UeCC82V"/%WW<'q +LDgj#Fo$/TFekS.Mpj-kT"Dnfc\T5q&o*R@@M'Dm*?ArB3s"mMkbOS"MrOh1d1M]. +L`A%9'o3Z?_24rpR$J]rPsmK=6rI-/Et6ZNbHg35/HlQ0!R3fsjqJVW-'@39\FfcYQdSd2@e7=:^+KDTQT5&Z?[&>$D\b(M]dO7=MQl^AZu.G +]YGnu6`4:F9YC^#m3$NB9CUS_`68Udj=Nmu7X"+WS$9G)6*nnNa*2G4Cu4AYF)#'s +-JH#\]C.rYb\R!hGO.!`>m(-B"FcgKfYPrXB\BV_m9ADI=50PE9hX(19,=p5bISFH +3[pnOag_E#ri/Nn^jd$?!RjYomdlKO%IVQ6]=3%!AF2_JZ9(V&7l8n)#P^W;eK3Ab\'I$TM*D%6cGg#_5!FV;2_td&mQsJJcBUfnm8^2Y!@I2 +-/Bq9lJu7:'g.qLHs;j;3n;(jqucuRPp76(!DRT#X\-E(npVTO!g/@:7#p\5(Q-`< +*GXA1AOa!M%4n09bj3]9(2LtbS4k!NPD0&.GPV`0q@2lps+cSl7nno^bTDdB+&5X* +B<7`NgJrf:X=L2"T8oC*))6OBr#[dGmQHooH7BrbGJ7J5U3t7pHLXT0#cJ7&#N,^a +fA/h6mTn#FRc2.%q\nBWPQ4A.>8FE9SY+>7D_le*_?1oJc0?bI6ddk +:t4t8+nA,5DU8T=r6GKZgROCKpIrV5(5prU4FR:kdU#?d5-]]I'/ +Q+btlopc>&?Jf_mj!J4$,HfoEQ4lm`dFRO`ZTOq$T$T<$8BhfMja7Y.IC',.YfdeA +Adb-U1Wgb&bSFAnF.I>r=5]LcEQhrTA?d+0RJ%L,]6(W$)U0TddMdcdGs'N)5Jobk +p*7]U5CZU7&N1b2!ef7b(*L3m(mP(\r8@hSIft7.=oZO#ElrsSk_ +b`=Z3ITc^4qO2fRs5m\Ms6ocE%PfiadF\\aAJ,6hDio.3s.cO.Io*`'&d8,.%'bN% ++!5#,(M"=(CnDR2O;)Gr&T2rN@9paTW/*3EMDsfm\M0+R0TiD#STeu9^A;RfH1QUP +hF",la2D%jfXip*6Rh`>$jkkl^'&Rp%e?lq9Z6tm"GPMWdKfpgLLP+LU`(%9W=rXe9rp! +IsZof?'8N>oVjpbMqO&IWF;,,67MFJ<\YcK!.P"J=*'QV6.$9DHupi#UM342dZO;] +.PK4T-4:BWAf8Rl"psKOI%#17b;P+C&4aL,5^,tshb7Tdb@"XJB4&pA3bS"gZFFfJ +%'Ko0U%KRK'V+V$3o=?rWRdn2h@1D)3M:gK[#uC\^WQ1!4U:XK] +258#]&]"D^"eTP?gCH$n638GD1X%)6c,7g@?j7pEEM]18^#JGYdsbbC8E8PafSt4: +c!&P\Ef=0FB*!=1=]H$:eLJ1r/)@p0Z;)WNN5s4a#atfg]Ct7/.4Ta:/$PB(/XGu> +.Cll8]%.O5=(*^W0pbKmf,A5pC'F>]bdCXBT*NG%Al^s7\HHS!2TTeu +SH7dfT=[1"nd'UDi%K)K>JA[qi)g4f^N_Xh)WEkTgK8f`Qpc#:c[V^>J +[XELG!r@(lqR;%\2hD6as(fH!)rZ>Gi>qdY/phDi27gmLggCV0Z%mo.JGgMI`0P@O +6SJ_uNh5$$*84&T4NNsUaBa+X0AX=R%'b::\#VIHDC%BO.+5A^@s1XUS0p#4(/V;] +AiVus>Wu&GJ==_AcP0JlnguX$G$.[;=b<^-;)aSi=0dNTfg:J\BKL`o0)%sJa!mMF +#k5n1O2M!bs2UNL`VXUY+?9^6OCUhB9t/=F+8DHKC&9dfAbk6SM#T7/7JQUGZXn[e +!.UEP"ciA=n,GoKTRcO>i+)cH(;TJtOn#&qo,iWXJ.P=HW9KEJ7%,1AZB4T@O_p,s +4UV50r(i6Z`;'p9%(/h9[K6@TL=df*3$a2;VFEOVd[fg3*D?0Njg!D0M:AtEe#q2Y +]p\$gZOJ:&[:erA0AP$%1:pYZ`F(p@%NOS>)d*??T7< +YO<#e5:3$`T.S^1!rVpds"uA-LC'51c2dc"r`c,rk9USAJD\Xer-Cim=5t2!s*s66 +s-\]@9rm"6HM.5/p!q*OZ\;RMGpc#uH4A!Z.Uii/o(B=l7VQu)`%Xf2W'tO$%F5"6 +M0WcJ?A)/W-IA!tn&%'(5"9e>*QNo:fokn"rqPRr:$ +3W)?=\8uC%;Hh!JoF,lg$M2OQp!*s8]G':I0?iTsW:-9bUr&5/&tdMU/rWuC[(a=' +qp?'!1Y7(rGOJ%IgE3a[O*X*9%tAp*`QnltJe/Zb^5KBpVi%D`V+iqE0/A.\&^SV%dkMZ:@CZ-jn(A%Y#L_a?N:[e&hDu'n#[s^CM +R=5g[n:A%C5#Ur$AF?um3cYU^^rCNeb#MsQ>IRHSkg3sa?Zu#c3^2]W]6V"-N<")e +-c45&JOfa3>MnSN%]NJ,'"(2DVLQ0E8dMOihN=X`H1nM +-;Wf#S]mQ`I:l-dBM#sHrlgqTPNi)h?Z,u$f")me/s/G3VDp:dUWgquR4ZN9aL3WT +1NUc6OtAIG1K/+F)>;A4#C>*qf@T`)ICF.MqsJ_/ZBCa6e(Zt(2qD)SCaW^?qjV[+ +0^#rj;MP7d%'n369N*'V,4,=70p$3,;N#=^FQV3Mb`@2<33];d\#WgE2uF7`R?gC2 +&LYms`6V(Rjs2DI@`cFc99Re=Z$7nVce/j(+(X<;.-iXa5q\t'LJZ'CA4hpoWt(p]P=QT,O(Sj?FJ/+' +OT,FYO%N8=,a(YC;FdQs_m4CZrg,/D!;g+eHUK<06DX\WIn?hYafoC)TR,:iITE?Z +?51hdR*ho8cXGq05+/KI8S5_b2S@tC#G:Kg2/V3SB\N`/a)ut]]]!Y`$VCD,qfV37 +=SoNI3tC)Zm=5;ZrdBR96`5JKGOCS4qp_*^&MSYK13FHqFW*Ab*'AK`r6>/;dBNF" +8IH=[3:5m?rhg(/B8L\bn^@__1OF,j7u?gReCEmWq'T=g:\$E8'Q%+iYOCUKc-Y1s +#rrHW@Fa5=3/(GD(#] +i=[7hm$n$+r6p6,!C6NH\G*a%:]dk`\1&H1J`=!iPo?"9;od%:DHtG.7>,.JD^5'I]f(>2-(7_T9oI3iXLCYs$-9CfJkb@#a>>)s"qAG +,MXV%g+JG"?^fnLQ%Z26Z"U!Q\#8J6V5a.7Pl#1;@gncFVY"uA`'U?bC8Sdi->GaQ +3mBrqA4+u%aNI9<)&%._L,r!sm3\TO.&`tQBKd#s,XS8'W.A>MV`0*X"/a'N?"Jnp +1kM`eTK6S]d1U3'LB]l&UO`J^Z-YclFIe85=b?GH)"EA6`1K&GIi:cTgD&ebcO'N1 +-/nm7-MmaZMZ9k30i.rLqmt_[b[:,UUEX'#7,R`q4Du%MC(`\.&R?n%S]R5ZB$,5E +'q8)C<[Xm]':%%Ai,7E$&:KZ<"967^!,qd`"'n,K_C.:Y,ULSi--@&1+-7CVPY*d0 +P"a>\k1!Lke/2t91l`>c:W;(d"Z`IU].j.f1OM?kI!kP`r!WFT$GHB/4ackWr:4(o +o,Rr(D1`17_'RYU,o/eCj#[!3P$!_RuEdG*T))nPauDrmtE +dZ*GfRENO2_[,2M)-iEumMg(d)s#*]f@t;8@>H`/Xq<"<2Obs+u:g70AB;Q=]HqVPfoV]c,LFffP'"lj@)i'_&5dE5BhLld6g:-\'[e7Y;B3W&Y5qZYB.WnT@F[;Lfs +$j_Rgrr]i0^7>Pu?;RG*!eL9)KOVL/`">k`KHthFN76#$1g*0[Ff1&^dI?[h8L(muCYcgjs/mAiV$iSR0Y_0#5li51JF&3&06GSZ26%A!G?K"-[2?#-+ +s4,7VciQ:=6r.kD":VBj@6PrVt=AR`pq*4?GcYr`b$hAf(q@>7TkP63c +5LJCBgV,fA;iG/!qk"d;PT4@u&nh^O:Ck@J,40CVN^X7l-&Kn^8[c%Kf!WVSogr&R(no)_',t'lF8&R#Kfn6u7ttRrQM,&pfh! +RRr#VLsU)de/na7qa%@PW5nVq=t<:S)62Bn=_iV6C^)b"6(O#I`sY$O/?hEnNiCBB +0(M^slK9MTht/$Vo)GnWk^T5bs*kgfs5u0(s8TJQ!W;s^eqO-@Kb)9)cSNFf*68%6 +/G*@9k[c.8\d<^eMqol%/eeSPAP!:aAQi!mCD%_RO<3OR.l/F/[SBf7;$u.gO:6XU +Hjioh-R4SKOer-:s3t_gHheVY&kud0P:WY#9-F8!#TP_WSegb<_KutbL"+&'>*;dQ +8ZC7V`-g1`9H4!O,QDR)h^h-u*!;g<]ta^:n7'Y@1LMpMhNN.J,p'j7K^M4C +Hc"@GR-iK;lS%`X2h?nMhUl=:F"Ft#Aq*\>R+-ac=[Gi%WWWRc&k#, +i1;ui_#"1N@o#3703mHJ^#tE[s,UDScWiq7*/BPo"/f2ua!Mh>Q(L54qE=UV[gc,O +XB8@M=\pNX?goLHNi75Y)J1`H91K6C2i^gi[,YHXbX;Ti/D!^fXL1XXI)6'iJ*IBa`?<$W>IXEU&]d +b+';o6njRZ$0^FDD9)/:Im'D'dijhT8@B4Y!Ta8< +OELf'>)fej6/+:A*82O%HDEteAD2WYl5'op'+4.>#j$q5%gkgMA,Q2urAtR^ +!"KbTZp-*.";T:N"CXOqhh;F*_1_Li7"Qmr=>YJ)RQ;U[2CtH:'o*V*LLli8rsq?8 +"Ql=U.@AX6=jSdnf.I:O?s:6g;RLKA;?!LdiRl&"@AS@3>W,(f`/'S%XdTEga+.aM +-D5REm/oUG&!RQEHghoGr=,h$*tdJQ)482-e8sj/Hhc$)oYPYPI-%?^d'_3`D:$dd +hrq$](ZbTlUDKQBY20akmpano@ci7A0B8A%j-OZs=0E:(JG`4&VGH2a/s@*tP2Qsk +buhSPhn3Q=?84@`kUS%QV9P7`dFaoMJRq>S;VNU3k&o&c8e`3Ir&OOlQCu-9XHQ\X-)aeJ'MJnH4XUC2g_1u(M +6a@"X5ZXt/ArIITpQ?S/_2:'^7[O2niMO7(*^BT&"7RK'5HbCorm1H/s/]ml+6u.Z +pPVI&s7:cc+7to\s+gV'pY5uTr?NHe?N3g2"mA*6)?Bls=Q$#On2,a-TD\8.rVqZN +^MO!M6M5TdJ't(`i:[--p`JAmNAkAo)?Z@kn+*@nB-_m5!JDS*@PE1u0`c2^NODS0 +gZKnDjsJdEVqC*.XRhY"ne6uij7W##>'Vi;Gu:T<&YeAq6&Ys[+#";G2]@\p#t&`#+TNl01P!O<"f&^-dB0"]H5Au7*MNUj.Q3&S/.gnY +:`U&&'^g!m7XS(>@=h]ok_+])bPh@8CKp([S?HnBX6qpN+&\`[m5#B-##+#6SUAmW,_T"^m=Iq5(_907Tf +]FQr`,Al3Kf:S6b?cDKH,&Y;b"[E>af_po9r@rcuIGEs4o%)"*`>AC!@pqJ('2__s/8%\gE'p<5e,mgNg!eBKr;tr;O@LU-PDE145Ue"PAI4GP +'rfuh(UiX-IH&V!C>$/#>.>]E\0)TuUImSs9Oiks.=Sn[[4<hr&,FdA`cZ\"bn(@QQP +g@XadoU\$_:7VS"CjWWC44[,12U1"MD0h3^#>SP/cWiK.0'k+GaYV8j;m4SiT,@!VggcWbC7^Eoa9f0UA;+/uLKik.>(X-]hYbC(iHj0cqWVFC ++HFgd/ZE0E.A>nR?rN.gdfT1+IQ@J/T6?*q%_Vf$U[u1^Zi4;4$A!Ft?$JF2R,;29 +lu!C'@0e!$Rh37B!PIC++-NFIJKj$HHQ,:odu3%!<%/(#f8ig-Lnn^HBD,OQ"$JmY];55o)L+b+,+C(T>.jC@t!%Gp1pj&8g9#M^R$@%Y#,< +jGdjH7Q#Ib)a)KTcKGTsW!%l(Z:#I4@7i(JjjqL6>@`XJeDlB&#scg)#].FHa(_KA +>iaX3j=dXV^[">ej"UMJ9FC^`XPgk=$iK=Ri4gjRs(tp&^Fb?@j,D'Oa0EB$`W"'n +pt1@Mh'qoj8`[BNA>N[FCb@/*?2T7O7n^Z@S8ki3_7SJqZA>)U4L]Slk +;*b-==f`2A63ElcSPT>R!;J=7n/?.1N>AK[Q.0>0S+UL,:0Dt:F8Vm(lDMLgchTLA +F0C'C]_de3ooaS/hL?ucrK;Bt1&LlYjr'9\LQ(h#)4-sOYJ-f>IV)l5U7],s*l"T= +59k)0-*2`Hln3>S2/2ADi][VK*u"A,Pn.>hTVhs'Ies"9/K!+5(]FC^5u: +qG-D^c>/<3*GbCA5(Wn#4ii&n"kIA48*]!&sD0@*M=u"kA1]6@cHF\W,iecU,$Q0s3l3Qc4qT'2Lm1J]jCfsDIP'QaCZDsR +gR`lUaoge=9Z")_r/)J>:7O!g)E/rY2_Ud[0+g,dL`bUe"TE=.HSo9&.oAdHO:hX. +97t/rODHiG5m:GeM,1U?caj-4CId&8Tri'+`WZXm9S@F5;P"&KB0l)W(nGJl/IEbk +_C(WX8Z$"IdieI$+h^&OlTqmp2e:;:=gmkR<[00]q&S`T2YH!`5h7fS@kHaEdf'FO +]lo'W-d&tq^ls`e?t4hDK^ilpcjn-p)s,t6n-+An.hX]d:7d\B)A>a`i$7m`7f`Cj +0nZ4]+gH9PE.#DC$,@-I(_hV#J-#Z\ajS0C1FF7*:T5,T4t<($.@OY!"$,H]f6Q`jo^Y6a`$ +EJo@j\AaB9\auUcOQTuK/mkG+JUbdc[+/fJe_@I-Vt?69g+mp0VQH@9c5t##*YDfj +84,HnS?7Tr^g7`P0Ud>&1OKfkFr35e*&2gT5(%l&g'\tU$05n-YQ5[DS?8HXm?$:t +:q1)l13Zi#*a +HN>160]2h)oJ5RNT*N^BJH+tc=T79*5lSF65MpcNPDt&Os0p&J).X:!$pY)S`>ao6 +9N+L5oL/QbMF1F%;\m>,,KZ5hD[@CSM_UqqlDEPu')\"$maBLC@K]mIJi/j%NZjmd +P"ZFd-4nW/atSHgbb7-l6idc(aL+Lu=sNWXW-fP+^(%*X(0tU/.Bpc*=C^nQ0mOQ4 +Gnu#-QIl73^/>rJee-a'Np?:'%#5t',Pfn(TE@773Y2%`')?SVic"]js"M[kh^/L+ +AHr#bF+XES&+`T'fRK8(rk!0D57M*eEJ33B#eoL6ilQ\N%$pp<[Q$?SZAg2299CeL +5KG'aA45`>_Mr8esmI.pS=oc)K0qomN5l0u71cT0TaEh@a%]R%mb +J&aMPHsp<1AmTn:\bJ]WI@klm?P>u_I2TNeK!b?4Yl:)uo.D*ZJ+#hueuS#jGJqCc +;*3+@2d\sjK9ta5)c)'PcM_A$ml,MFgqSPnBD2d":OSP#Gq=IWE/*Bo>B8#.K,5"B +n9eO]p8S#K0lGpbZ57Ijh%0F5NpDIu!1S)skVd>6QN6NK9snoO!3:ug.gdRq1@WZa +3H=ni'Q[E7^*$1A'hb#.V+uqGB;HKPZLg.;g9b\=TQ"_TJ6*4k6&2f[+F+#6!<=9N +5Vm`4^?jXPiQ.[1j`1)P:[U,b+p-d36%%kY[t8?*BW@4f%rl18mY,'gl`(O9uY!8V7^Z8=;;*,a2Rf +!>8-rM?=\D74(?d!@^tU)"\0o3!/e:#a!eFef`2[+d6(n/%>pOf[iDIeW=>WA5\+8 +D%a59LjF::]6.RfqQ\mn@K0X7h_)93-d&>+"/#B/`(#_Z`(roNNetde@7gEbBT>=\ +=-rtO*"aD+$Gkln&Kl0%&dH8efTJgY;Q3n=q#&I6;DtaBOT18[mdu.%WG`9"&0flP$[A_cn]C$I*]"E4pgh$IY2"nup?ZSr!` +fDU'JrVMq/gD7G+k1gn'SSH2^X%./pj1)*\gY,6bl7#u_Zq^M#?,O:YB=a^@7L1V&gmf*f"Aee%1h-_oYX0os67mK3)gQsX +(q_jO,Kq>p_OPgV-o29uNjiPQRbc;80k:LS0nOL]61E"BPT\fC5Wq`$??2\>J.eLc +7tKQ>rr4YMmr%WIhlQhpO?E"1mqqR/c2V5J_k$hBq&_g#:GdTK5*LtUFSl(mq1VDg +k7JMGO;MVnP%[.ZfJ&J*=ikh\MCgt'7uO]IN\QA$dj#$1`VQN&]#]ZZ9))Ttq\Y-8 +km2TLnG0)Y!`T`).Std*'Vtk"X$X)V&5a)bC*Q81a>dI!]V.TRM,:L_5`1D+RL,9p +/-qo<#d_ao1+X`W/B+qIZ%Y/gC+$_B%og"[E\J1^*edoqJG,E#I"@X7d$IeO+8X?6 +cfP&EEU5nc@K1$KqgVCOs0#"F^Rr=@Bgh.q6UJQ[$bbg?]5&(oBFmA+&RHt&SM(PH +aHke\bI,g[lVeM@aNAcr[EM`i7[/@*UiBRkH#iQU8Y0U*8sE1nPUSK0r-e^E&M;_, +'no9kZd0)'0*qrd$R^'0)'^RVa/mt;biQIlXVZ80YB'>DB'_[Un(^buT;V+iU@,^g +*IPpfI\1WL2IlJKpAQT)js=S^-+LSAU7b+'\P.]^WX^-dWPoRN72;glJr63h*Mudn +W^J-G:otd=V%s5$5a?(X`M8g+#D@tiEA_Kq+kuRIOT)_jSR2`M^D:)PY4rnt5@tpa +/\([,n)dRBp%s[U+TDu:(SDGmd4[\]SH?OPc2\rc7k`!=f+Eqm +iQEm1qsVTRDb*B:M'qa_V_=e9cb*GS.J71 +'1nXX9j:Y1:0d[;ItPD:j54$@^:- +GjLf:&2=-O&8>5pDG6QW/eu27no$n*e3R,H#d&3%"9g#@5k'n5,,luIF@`i;`jeH> +V]MgD$`TIYaroYU+;B@h;]ppALbB7t:Z[or<\U7X'PWY>-GL0,q&gVSXlVJZODnr) +RWS(&e0.OZ+JVO0dgD%?2.ZV%ZE8aueW<'#UgAc)D\R6QM#-fN\9IJRp0g,]KtQ-% +Xt>_Cn:*)u#TF]l,S/=P@/^3+.(2AtCjp3E\-QMBKJXrBki\#kk)\pB,5$f3%nIsd +UaX,'o2f?#Nq_('%c9+UWh1gqQ\E?%c +0YnD=+d5^=B5]$XT11KJ,-AP`_]8i<`>XcqCKMPlB9V7$! +T8&;q":sts%W$H5Y.:tm4#P@;ek]N%N0oiM9hNDWWr^,8jY0`s>g'E]MfQUhA]:$* +"n,M?/Sb>b'>\06rBt$`,O)ImB5\=MFEb/'8T"de,+-$p=g]#7WhKtY0 +!-eSciE-Y+_$[FM0_$iiA,n=Q3(Bhm1W5e$Cm5NBO:e;/$ISqu+>l\O3VZfC)$&dT +s7c?hJ*DG*J(Xeh#'"PW!2>q`4;%AFs080l6kB7RqENgQPQ89dp\V_<.f_^3hd?BZ +mXKg#oD\_*0*M2RrX>MBDuP)]U1"DJN1+3A9hA%ehMc8@5gEXKEsC?Tc_4,/mN9Ag,@)PY8KKaL)U/GJl]=i+qq:2i'sJOUI#m +D?f[3"T>&1nYm9!]0=Hr%e9S:3#;#HEU;SB$N+n2s48F1T*L$5FUPtM&@uS@3.!3h +Xcpgs:9mfo4o6l4gY&L8qKCq6f\#PZDPaOl]gh4[#(92Q![bdkLI`shQk'tY0HC(u<3b/1:D1+3LLgsdlUJ!7?9.Wr$02X)5 +WtS,`!O:Ma5rbeH?ln;Xe>!#_b2IjX2+",HYui0\6nu,LM:?='[iIBr/8b'.f&!=[ +_k.=qr#_=+6g!LNX=KD4J'PY)P'I0)2>cXibhJalf>.EYW+Q*:'_oh`"OEFqh:nlf +V_L7d#R@PiLMob^JG4%+$66g/Ye<2+h;-ZBEWtpJP^?"5q5J`$Ro?eacrQ# +b)g?A-tNT\P(F6;5R8VXd;^!s$jK?;;JIoV.Zi(i,,CVT.2lAB#?#=KZ05Hu">$2f:&r[2;E,\C"0HAFecT.cq"0="Y%VI)0O/"Ha)J)Gg< +r."=Q6Ipg+k*U8eOEB3L'U9o2b)rR,A`uMPXRSgVF'<-*#'F\cop`tGqe#69(>Yl( +4XV@SRs+4([dTZo!rq^mG+@4Z"$ekcm)2)0kQbi%\^r8!s.S3*Z1l=^nBE-=DA\(l;\S2to>79=?\fBh7XD46f. +)HUaE.SgB8A%IQVFF5AJe(iRBjK +dQHX>38sKH0]UY:irHg-G1a*4fWq[;UpA1J[*uClfds>O0"9:[[llDI7"G][Vugo; +/.XD2^BmSa&M)3f,M4h/`/FpL#66md1Iks +5lLT:!X?:AqaggCs(DB'j<*i!pgXXJ_,VSK"mfYgK_^Q+iEctB+9^mSqWfH`[k.E* +r^WUDJ+s9\h#GW\fDkZ9s6mQCJJ\+!qoS:h;GrM'W2=[*,pUZt::3\e[Rs,u8rg5' +L)WPcl%O@r"?iP1(fIdGA_r#^7ghfN7u +2%/5&(-8T1gF3qA/J/#Os)=iG+&fn=qQMG@[>:W@7F4VedJ^Kg*i#23!/3[s0dm/A +@@!lo7RI$qG:4E]a#IH)c!jXGWF-##Q[n`-Q[g<]]^26U,GpOBN +"feAGF]+n[WCVqLX@XhgLqY7B:d(OP'rFYVZ9fe)[jY4Q(1"(<-rVEmW5n!M7V,3N ++6ug;]Gh@egh2T-%lkFJF0@gk>f4#N=JtVTB-5SU5%RuJc$BHhs#&g@KRf[4=+Ds2 +M_ETOYKZ<>dr3&!]>VV=jF^NRql^:;04T4A'C^_^T,7>8&)SLV(CuQDGTXQl->NE\ +-p0KU7gM(uR6%Yr]6#%-F.[m>1&p$Wk14h;aI3Q)^p.>:O[7U5I`K0"f9'4nM%pst +JK<5-?/r)9PKnTON]k),D6:fp[Fu)O8s&j%.2As7(Tfg@?]`HuQ"J<5dcVkno+I!n +R>_;@k-R\EKMWkZ5@t$;4X1h'/P2=oTE$^,849llGiZac5_fb2LdT9S5b\RH@Zc3: +TQ8PjER'DddB6otU0@K)6l["9i:7ON":8=aE"65o+3(L>c`IQ&cUTB25[mS]Ta6*` +,RQt0@&O?I%_m)d=u?skAO$bSBFtj]!D!d[%tPi6#kCAM$'InP-tt.VRNX'g^46^8 +eVT\Yju75!R^(so4\F;5;J+T1eQe>2+]>ibnYd.J<";'m6)leOoX[Gl\cgi?s18D> +RuoT<&.\KM+7%)^%NUfWrE/Z"o)nRtL4nKVFl:U$NsY&>L%M+WqE[t$8/N$eO%Q;[ +8L4JK;Uc9rg8HQ47#&gKA.D0b@-pU`!X@cMA'3I\l@k-W($k]R4!m]!k,h.>?)L^d +`ths9-K$PU%BsKm#"$,!:"P5Mn!R\cZ")[jo>$XSn?pU;Q1rpbpD>RN^C$FD$X:+M +?\$IGiho'KE#L]Zj>RKkHghs/G4"%)l.pNsa/+.'[:e*n(HaXI3d2V@p7)@Q"PJ"d +#Fp#J?2+l$p^WkkER4Q8?\>Mk=Y^p8;OGQBA,p&Z"7])Ye'/MiE(PE*mtFq*L*O!bqe +-QF$oChhYS-#DnQ!K3+)Zf2"l16-'OO&8KX@:>pSHAts/Q#93b"hi/<]?;kBeM$O! +(G3\Z4kn"/bM+?!gHNO,0#Q9f_[i+%CWZ-':\Qe@%l=AS!EOWI;0nUHp?f;KR-EdTQ,"&d9lB!]PmfIfDas(rreEH%0?5&2uK!>K+QsY^uMWr +TAZUXq`d@G\,M.1^I5s%s5=i^pVXaJ,a7j[e%bNC,ocF[61P$TVpij,FE85RP"Duh +;6=OH9+&cl5/m`H5L`odf1FN1qLOj(N/`RZ`Buq4"T^DNPc6XZ9GB5uKrd]c)c7GV +1!!B8%L4nT#TIKbN=j!-!02m%P]dTcrBq/KLc?)lBeMAb^dhU)\*OP\_tu +[$&S@'.nFfDtHF+i%Ym5_+b,U;Ll9X6G7,!IdB?!dYIW]>MtQpfKXA0s"G_jo75cr +*e>,Ef]u#cP8^-IEhU`enWeKafc79E%-\hECbo(jMXm^-XI`#5@`7QKk)VDoM`.^aX:XbBn +dLWKoRn&]seYo0*Q^UMK7^3`*:a8/"SWp#fS#7#2aAIf<^>Bq8'WQj'W.okB:o573 +XqM27AQj`o*E&_OI&U^_iM2bkq=%:)nrR)O&TRY.H0n>gd80S)58QHXP&pGZFo4;f +n.QIQ/,aPc'`\O>G8[mJq$Z6Z',p,b-A:Pq=Ca/tcN(TN_I\b_6a68-M&Ma@&556n7:A'_4OUJU$r,A'g7@kSJ@- +'CfnV3&W#KRLlp0#eM.,&$C0@QpfG[(d+]#V#;:LQk<1.OFKKEk^Yoh(YZV3(!&K7 +ahbt,$-*L5`fJmsrddQA9;DtfVSWYS6lJRQ0dOXT;Ig8%80<2C1Ta]^XBi8O=M&4o +G[SsACu0VUfm.&SZ#A(!D:#Y`hlp[:^CXr/Ee*HYk.bkI"7r.sEc-iB)MR20.\4$J>+6m'u`W\FaVuotVT>SU/*KY&j''9-;OXhYm>&9E1+nqq.H\ +kW3\@bW0k,SpCeeHZ0!u5/S1cUISWX]V!!/#]no-Jd?SXF(iUR^#mbLjY6`GI"sch +_=HLX#:71LNXp?WCN/*n"6@f"s/IGnc&8pm_Z?B7FWUVB@*uMt_5VejMG +=!s+LiguU%IQB6XqMJ->%nXd`8S1R'/%G_.nOaV6(*E0f[,>u1`$%\N99@J8 +0lNMB?b4V!p@J6#JH'fq!l*n(Iq9";keTH$(V"8q4KO7GHC#".(\=H"1F7gE1;.CB +:!$"mM(/]!WYm()-;fsBWsp_9Jc?YIQLnQ),LDb8mni26e/+Z=9rVJmkrto'?D];Z +`\/<#9HT!a*;/I[`C&t5p8P+rqaY7[A0)KoI46ktJeZIir6QWh2fJV:^+Lh +fR+:>r\Ale/o]ppA*1X>o#uRTb\lLsB)'eloYtVCg0sW1j4K@a#iOL."%'f7HIIU< +o8ZTC^4+\-@F)25bb7?"So+pD=0lmF[:\f'`OpTtPZ5Z0'PgU!HBY +kJV("rmRT5;kig"r/lW!cp:>Z#8+a@"`ac6L.0"8AYT't6EBl$EtCf?kReC>MhSBD +&;gaM<_-F'BSMO^&sm-b1L`Z1!^/S+6BjGPi_kV6bf<8OTd@d-i\=OYG6l>M +P'U:QYt)\X$&R#^A*(pR!^ZcOiG=Wi +XpA`@1]]9SOuOjXjp8lLhZbB)@/imIL?(-[*or9Zk07S+/F-B@):n3o2]ST +>'D$W2BMtS)`DTteZJ%7YNXMSQo+55[fc[!=g];4p0;kOhKY)Fo:LCO3.9sK!B>/` +*]WlbY57_C5/WA1JC=IMhV#R0&A4>sVh+gDHbm`TS.jL?pqD:%i3(N/i4sL)cu&!f +IH\af(i\Cf[-'/gi\u#r0X:)ch=VW?pDVo[j;rUX"$eiuFU*JP%6s=7E:-qRls@'f +>]T%3E9Nh$mf7m)=kf5[S*"Eie!,"8K]V5kZe?Rt8rc1tf=6^??cLYM))NmS)\hsY +k5kk[$[Z`j*p`"=l_M!-jg?&i2JTQ:[HJ'g[-C//M<,e@jdoml?GRpO!CbjOq-L57*fepF2?9VP/^qap +mVbf/Z:GH[j3:It$9"^TK5BEne^oK>@n-,G!J)BDpFh`FIM2bkJ&qZHs8PCa!8]7t +!&"J*&chY*I0oBZs8TS1&8@4EDTE1dN0sK6QiF;tpoXR!6pi<;+Icf16aOq/iNcC% +b]jLF_'C2JOPm;Um\/Tl;fjpZZI9MqHluf@K/T#`Ru!i`Y`Rsjq:g]JD+)aZU)4hH +$r3>=7LEo],E.g$[[%ZEabe`mMZKn##Q\`_$JDD733%kHX%WVa'M+2@=iRj_HV,eb<5!"($9?XqF`3n07/#7Q&070#o_#i'j?Q">-%ok7Y_juP_$=A5$9>/6jor6,F/i9T5Q)_9X*T9V$i1*TJNn,%f,qB9&`6%O4H'UW-%8PLg*[ +:D,U).GHd1]=4bd2lV/5eGUb/qld:XSu83pa>aA&J":r^4S'9*K'-:WmrH]>FEit[ +"TlUoA*8`\ECF*H86\8OS='`NBJRm_;G/E7+dZjh'PkX>_ug\Tq[A,S[2jUG)nE96Ssg/UlZV^\F<"tFmeNOY,g)h +Wl!N,9+$A/dchDXe1i()@FBf7GtKVr<[-dW,8fk61PR37$cJ(r7?4#BGo`0^d."E, +&\7B3EW65lLI.eV.\oa'Lkd,WJfo.\,YCL=eB)u0.Rm0(qb7BB:L_0G&`I5WJW$rH +CB53ajr-(X<%Kll_,_>lc/!9m)J0@e4TGWA-T#*Sh5Og.O7\aMEY]e!Y +B%,i2^3T=aC"&2IbGfdQTlK+iG1EU)`N"+CL-b3]EW)?jj"8I:4hH^^2tjZBd#IcR +i6n.u8q8/,^-d\4?(9,L-h`\_^ogE+^)aj2!+ZSXId#,hhG7D]!@Q/4n-''jI/rbc +^M>M'3-H?*G98hd)[7rcE,R.dr!smdm6`lfmXp:dgg*u>d/0,%s-c=LW^G*eF8r,\ +^q(37@VF5fCX&k"!&pIQD-siG*B\-2Gi#07g8obYW-Z=jB$.Ij]['\tK%J$!r1E&*;9j6f>"[#'W!2;`2 +(B(KJX]dNW7$X:;'U:058)eTM7bC-:fom];W65*"\riiOUafVsKhZr_26kppY0J?( +eh+GLR9\dDck;QIW:c1S$6\KC#IE4@l!BCLr`57K@j(fi+G6HiIr%PmD50SNDJ)1eL-,_2le.C^@_#-Ipn0s\p +@p5,=I0m^4s36)l!?_`Rif&+m`#5J:N'IZi9uH(ig;$O$dc2mo2^San`Hl6OfIM:d +90$[Ci04BrA^RC7#ZE_S7f`?h$\=fi$++Uu`b-sMD_E8A-RE:R/Hk,FbR^q@(U[;* +^aVkWX<*5W*fAW5'"(dUG>oB&eWfXP;$bTKN&"F/0N2H6aHa;-Ziu;F`!EH?#TT]L +h3C7olS1W3nbq0oq'2BtGAMO&k?#+-:]8r?h=(lWo*40u.Zq8Nn^2aKrX?nlRUT3Q +66/M*2#^?YS9bB*1G)do&ZZG/5BrCa!:*O15]8c#J_m]O)VJau?W'q.;q@QdX^@1J +>O%Q[$PuZ](JuutjQ11J_G.nq/S4$u8*\f,g?+=#"2K!Crj1+&i;]_5SUU3oH1ETR +nkI-WgW\,j=_I2E:-W80IE4VIPRB.fI*=d&T6^>`kNlLJafG&df,=H==hCjmpRa3! +%YteYmYspF=llZJ;NQI.;Ms"],oWR'2+%;;a9&cj@a18A[IDsATt-g'GSH[C*0d3< +SG+G=i)+au_lu77I*':)dRV]8!elGX=a\m)6FTPKN-kL`EVq_7nSR"RkOdfM?PAHL+s6eX55?+HqO)U'2KJOP +IKa9]_#M&6i,d]$];V:`JZE^@2kG@LhRUs]56A]ZlZrcia,^qSt!%3 +NVZHC](a`ZG?(-Ko0BAL=;PiU%K(^Khm0$3\?"JlXkL"V>eV>Q4"JRC&5V:p@tt4t +-:Hl@`Q0/)bXk^0`=bEAQk58[+b5LAB-Vdcn!bf9U"op+I,Ip[HqlO9>TVTQ&YU%o +OpZfF_2q21,S:>3(J.^>C&)hggr;e"#"=11Uf=2m'YZK;>r(3p+\WMn![MH$jFKk.%X3]$GqtRJeWkL$`0AS1+Ye^2=asoo%CH:(69WF +)T]Dp@ETG["b.RO2Mctp)%d;s'al,%JqW$n'>=Y4erB;+JRD#.]O*c&pn',R2'.3mf[HsBkll%g"lITtn;_3d2OWE$#+2QW"^q)W72-:jV#b +D9iEh$NOMaD7Z#bgK(E0Q>u;B7B>!.lh4nZc3:8W.?fL3rL-0m\cm`?3J3bhj*C]D +.bUU`e\'Sq--N/SMoIdKL+20:EJ#aVW+jedEQ+S:.9j!#5dq$GY>Y_jdk9k3bj3g% +mo%Ks<6OXZa9M9NkB(GogJf%BO(^+[N)n3q7Wi=6.]l/tX\sjVX[0#98?L640%EK@ +<$A-^L!ZXu.A@ji6EW^?LKLoX'H.e*B1q0,,\!;X`4V7EK$kA&FQJ<0/Q0,>XjJl8 +`P%ol[%K?%^D0Ntp(!u^4&J.WW4nQh5&a#_"s.DRV`pm_:^_@3@T7e?sr^gI!]CBFF[bQ$N4 +n6Q#Xr"%]V9sT>,a(5l7l![&#C +*cItoB.Cs/Q)aHhl\c=G:#t2E@@e2&]s!0>D&;tW\%nWgr-ua%psUAC+,9uJ`tA6t +/+g8Kn?u?\Y[2RR>]d&DU,P`B,u(Zpj^Aa(TeQIC(/^>W7SgSHk8D/:3gA?tOr&OZ@9?uXsXE36`/&7LQX3[3sT/'2+YmrrOG'InkR-j:DN#Ur]/ +W`5%]GSF()+.au#>5]k`?&pl2:q[snH'Yi0NjePAIdi26pa,Uk,A$U:_X>I_^!DN? +5>t=l-Ct4B"6][uGP[/8oKIU$@k8T$I[>$I$i/ic@J:H"gD7Tja=T@Y`9r,4fmAe; +IM8sZdV%gTqqR1[,iI7!$3Yr"LPZDon@@Z-V<-(3g#/,%qnf#p9Y1[OUu&D\d^Z@S +GYb)S?b:;bYPH8Dm81-3?bLIF;:XBQU(sNk"Zb_@"D(ghIS>PS,"NUc,#='8""&[m +A([.\JLIZd5X3C-kck6Kh?#\q68hOD>Vk*B;8BW1hkEZ_1`>s7"t:XIORRcglleYZ +5oGQEC8IG3PIPrg`e8-&^Rd"LY$'dCh]bQ[0G`Y@`1hU,+,n2Hc6YaE)e=F=`8-[$ +]r71k\P<%"kWuZse8,V3PJht\GS#Y8iNT@ ++.@?';Nt(D+:qA*r6dtYm/I,1SH>%$Pp!5$e@Z9n<iCfRWnEGDsPoTCY(,<="J&b/8:A&C6l[NJQ;#m +?,rpm-[]4RYmW[Tm5oKm>IU_<[]S8,E?4qb>t5Cfrk,MTA,;H8 +]&c&/F\>3sbCJc$[Kihq]`Z&s[mk8%H;:)%?B88j\*EuhfR#lkQ)&3-)j>=IJL#8r +P/nMWK*0bnS9)su5TgY,c8fAi-Kgn?0ZGFEgYg`?dkfHoVXL$/F4*,4TE!Z/]4^QB +s1NWYB@SJKD6Tq9qg#rVg/XUND1c(kUI&Fj@U-T_FJX(]3B!TO(KE(,;^rkrVNb2V +RTuk]=K!4+V\#,k(f@;a5,ssO&4DimcGTt7YW`pah[I..%Hj97jcq=WR4.250BO0# +iYEcu)BnT,R*Rs$a>$`4KqdRlXW()NPL/YaBlHWnIMtV"n0HH%1ub2mHDH2,-Mkl` +G-!b;?'=m]rt?:L7pgP6pl#1"UC8"1s"sjkTkVCqY+PU$s3S[Z]5F2^3d;M@kC$XT[pI9CQ7^aS-=3/a;%t1F(93JW +O%]&-9CPu@o2(?d+iQT%9t#ONmM"+a=gC<\fh*7AMk8W1R<]Z(F5T9h+#rpK?jci3 +.VS*,I/YC#i?W.M[!r^rHk`j\-?Sc;FQk?$kotB:12,ohd.13'icX&)g\Bm>nZ2-= +eq)S/m15%8:gaBti\j$>Y271"I#_G0P-FdCIcXkZihrUFYK8l`VR'Z;QhGkXuuC1u%jPUK&aaTgqO"m*l3k/[^:bPBR/"_9_NA_j\( +4TP-%:hc7IXDlWc]9O'ru@$9606c?^;B5=&_e8<%GCW9@jpMQPJ`-MHTPVe-Se]$j:nP +_R_].]/]c$\bs%K]K/&Y4#rhbfq,SA4D`u^CLoj6>F$GP7>O5n$aG5QOG.B/F-F\$ +8g&`Wi/k8)'XZl^<^mW%K")1sU4bCU&^6-aUWX)*9tBV)Naf^<'$>dPYfM-pp5h@? +NsB]+0(qZ!_F9U*FhZoSSbm7ahqMV.5tCW@?0!,"qY/gGYDTlO$^)qT"q1]Y`0a<1(><:u2b+6:1Lg:CeL +]PGs$D:#XUWa@.)D&P6K))eUZlB[5`bG=XpM8p`C$53U +j:)Y-qK;D'mF010cA6qkY[0&+?>/L3knfAL<(r]]:RG/aP@dhD(,Z(s*&]3R:c)W` +%.k-P?g,?[K[e*,dE:\#am:0]G7nsXol<,&nlk_M/YpT[,-=TfX[XhKc%o]"Q<::0 +?_e7(M\""I_1.Lh7>g%mMP5_2+b^83k^TE!Tg6=i37Md/G]*r?P:#TgQ?#Z`FA.j< +a]&VI0MZM6P$RO@RV*]Z@::$cCD^YZ#^W4r*nU;jqqO\Vf5m%),HiDq#T+jP$ +T_gtO:CT'P$SY0D1J[Hu89?#p4OjeF;t[V_F#C`qd1ilbi3C+==GX2K$R"SZLcI+t +,(u)FZ(CXO%\A1:75oVmdh"0#^e_X+5BcRF +C>f9p+'.m7'FBb2MR99R<5qMmKh98faMn$,`-NoQ\u@'flGl+IV)gTDLM+p\pT,`< +m3\pIN*1QD^3gd'n2b7V,\6TpK>!rBKA`08coGcJ7.;[NIQq66V>on5s6PmJgf`RP +FjS#&^r>&i`T+2%PORhCnEGjeq5^I5lfaW.0l-5f2Ll;mS%$ca2c8nfJKR4a+D1R1 +IcX5nk#]_!Q^"'8EN='YD8IXh+`Rn"M;Nj+*3S0EX[/hS7F#fJp5VA=H>\gAWLa)= +OGWSR_Y[p(Ht^r71hi!ZXCim:,9,g.Klt_]\^Bs81`<35S0Jd7a@q.9=k>3pF,#p# +QF'9pc8-:*PDf+Z`VO1V]Ws9=nS;*^m5N0o[Vn.tlG![r^'S$?'X@H[@f'ZVf#PQ8 +OSi=d754b$n&g!L`G2r9*;* +_ashPi2'#sY!LgVI`FXOI:)aESp&;[7M?L9qeGdcau.pGQ,&-%ag+V>!]C9c`b2Kr +d$J=MQO6Ym]EQd>4+Lf^12r44G=4oAFjD6 +M[!5*8lj+JFqYdgR;k`bIRbiJ9r!C!,`,g_RmZ0I&$$a%64sp^j@0gp`!fOoKr`+I +[-Ff;Q)X8]9R8jU^W-qSEoPsa`_O)f.rhZJpM'ZfOR?,BbZ6:PQV.1J8!O$/UAZQA +ZI&_V%-IE,5)hc:."#l4gJ!cU9i\eTa.66R]/8R`%`_O'T/Jf^:_C).3Y>(f5S0sRME'Ti@%!c:hi +3e:)!_P[p&-Vr]7bA)_mHI[@3ot/6:4!sWIArXu['hb?C1^irV[L?o$*J<'O_n%_= +l4&AL-L:LcX86u)L(.?6btRLlB1'F_bHQ!@W!PMc`0qF3l352f6VB,[+B1#XThG4&\sS96u"+#PO#G/gg'4u/Z>0N.s46i+cS +H:@eNJjI'OLku.15-uuM$[[$eqZqm/LdaJFI%ZC^gSLOoWD-5_8[FfN]Lbbgr;T1>'/(LBQ8ss^!WD[8pC,BHeJUDF>Hr.Od8:dco +HK9un3))H$//Cd?M_$Z]eD!o[*5YV@*($b9K5>\X'R6;4IpYA21%A?@#M]2og*bgm +0"U3BGOkL6f(d5a)S\`A_uEoTqEO2;7]?8Es+C6#BNJJor;V9E:C\ZJl@7sTfVq&h +.@tkLb6snTL-JKB[ScNL_E."=4]^RO`GsReZ<@SZQMcOb&,N6&R1Ng[N;DEuPgs$/ +btO;L46I2t6E.Uq?jEq*/^r'BNd[',&-`M;qae#;rGaRum<2=tIliQKj3oC'O4DBn;i +U6a*"Q`oH#Qnp`&'_>^QBGm?](_5?:E.IlW2@2O(m:4<$Qa@b;HT+Mmm;F\.$M]eX +nqW1Uhdme#:[Flp^*g*i-S-9.9V9ALSOLkBq=Y9P<]95A92q[mA-0)Jr*eMT_72i> +ABXaYbBhS+Sra!k1+ju6=eKm$VeK%)TZrRI7`@Kd8NCBj*&AI8O'GeWO/=FuW,d,eYf$5W4O2Oc%2qdr%4B"hnp,+b@bI]Yr.*Q$ZM!o?Y! +?M3(p2=0`?gqKnBU!JG0HGj:uZ/ebbar!Fqp)%Sg1LN=L2O;]51`]C/PQ_ +@*Pcm_kNR@VIdr.=i6tR/XV$F=9e/O@E"I[ff66]<57K5#,PgaAPhZE`^^G1TcH$> +)q'L:l%ib.oRSaWat_ps!.k2CI>75aJ,rP&!0V!ks+P>3d&W?<&b(BSdRu"V-D6D+ +F_KsHCLq^'dlRj-G2;Kp@b,T@>*Du*BYV_5#Q,,23pi^i!kF#L!c-MGWmjt*P`i=k0q!&Zfc#u1G%jrP +ij'YP``Uq[$A@*sLHHbH_9Xr&^t@q+cM9`Yc[=Z+.=X$9i+]"t/]ci@mb]4=3=e=Q +8b&9Bo:+`m9k?^dnA<1)+_*)'2l)-9V"jVZq=S\.qD,pBEoP(GP3,BbpJn&[[)KpW +6uq\jg\n\Gj6sY&69\GSr^/d5TDNp5D3X3JVYL.#>4A0)pjO:Bql**DCFel.3o5KE +hN`K)@B/O4Z?1@KSfWQ=i]kjk*g/%qhD48NqaaE66.XQOe:lNEMoQ.)Vt'*9nJgYJ +lR-[&Y@EoZi`G!T/YC%dT#IJF3*aJ,]M2RZQaljO]_+I6pDnPB.V\c8C#_5d=9CeF +,d2Wl*.FQn]M74s:#5b5f`?5FV.34-rI4JPP2k=@UE9/pL>3RS$MW;Q?L_JTeiOhEeiF&nPZ*P3qP +-f,*Q_g(@uDrK8oq1B3:"dYROJ<'`s++;aZ4]cOVQEHfS+5;giZKQoHG[im]Ed./s +HMH99i7PF2PsNHNDY'0;j.9!rYbb[8Tj5:-<5#8-^+N=ik$Q0kD..L:215BQ/oO$2 +@nJ:p=s@NMP)LTUZ62Z,PGQRHV,-%_LF'h7L&JUX>aj=e[rFJgE*1`]&\I*TT&W-9 +`haZPAXn(WGN:Lja](Yh%ZsHu;"SBjf&EfmMTs$J`!d6ReQM.(DDgAt?>P-[9-PV; +iYMd(T,PZI1nt!(5-5q(#fibsm>f82[-E#adWc:JiQ/9j/H(!tb/ZF,=gdJB9K/G9> +FYAX%Ic'MFF.X.tnJTIVsp,:89GL +:5`#NRG+W2a5[ks"fLO,65q6J9XlG^\9tXI.T"Y +9QX27F\mGp^V._:K1'lO:GI`Jcj[B`$+,fgd$&KN(U]Ui\>/'4JR%Gip\$@OZ9: +;M`jqLr3D`dn5kF8rV@7,A8MPA1Am9L)2_6&Mq7]JFebs"\kfsc*0M]=Pbk/T=qa)[E(I&Z9'\Wc\>6TkB4_>g,`!rk!1*qK6, ++b;lM$@_`>mp/*6lS>F="\368D9:FKrqD\7o2(HecqUirfG0^[RFZkL3dXV)HV<%< +Qb)]>H1`Y4jW\j7/7nc.C#^m]o/qs+Pps3U?>Cg;0]26YLDdpS%PQq%C32DSN$L"#mRr7Rla> +Il&%GCg9lhNtURc@](#j]>gB,3pu*2)^H5uLM+^(Uu=GaE,Jg!?@!+]T1!'8Cc#PTbcmo1^qOt62^:XZK^kFNpXR*G(jkB/D@ +#jU!:Xk[\SFrQ(m^h8-.r[PJp^mGpi#l*>aG2ok*hKAA.A`:XCnZDc6RI(i1q-`L) +DXu;i%jD#-6c72T6H_Qlr?\.S8%T?W?q[mGlMi_h_$Yq2-;!+1A?SX;RAR\.`+[,`EcWQg.\aPgTBbZIRd]BPK"gpS%.9mXmr3h=H2p[W8,%NDt[)eW4eK4a=Mp60I +aLis5\Z*suEA:#BOt%/(0'(G6jj_IddAA-\R>;h';(0#N?GTAb];&`LC0&OJTT8n@ +"0Y?"T6i`)>:m$o4hl=]LLULC@uHH;+>.OG&)&Cl8L9?f-B1VCNef&Vi_iiWF1)?# +>K"rih3&MDibA(E=j=#c[B)f>qMIc!Z3l"OOhRMmZm;9MWa)p.;S&T#ZD(C6U/_@J +BN2.8QZ$SS<-jmQWWk%:IPDN3XfU[ESt3=nISmg$[[D?85^`W*L]:XKu' +76;0WSTRb6FTKKKJ1gWsgESt4;#-;?q^K?,#)$p`9b,[HGjA5>%=hCgoOu]A8"I.[ +lERciOVZR\"&U-L*`mWqVp]c1MBV]tOh:OIV''4jK^SFh(1T^UGY.KE0p[74A-=IR +I$T!u^`:[9K);tZN%PI12[C>^KT0rFJ;L@`RCnWh$ND>FDEfAK%"%En5gGccjE2O7 +\W5cq]u[qsMu9b-ggB6_r]8C(+4KTV1a_k,508lQ'%]jPLR"TLq')RjiTFa6Z5BRr +E-\`"-HTC:mD6J::YJ;9k\9VAgQZQRk$;nLT.b:!EO9aI2GBd2BFeDVO_CUTAJ; +"od&p`!3)1pnUK+V+Z,I@!j^`?KpJ;)3Q&/(j@)idNq9?=bajnW;M2dWFso4ca'@< +A7?N6,jdB5*s%dsa)E=t_EM3%8X?SDL!(71_#!]hVeeM1H/n_[3dC4Df^-53\T^>t +kfJGPk:U$&hmtGQF^D_03d:+=lKZdHHi)Jpl^8F0a^_Z>A!=_Bj1a,km>$@\ +,Q3p6XibL?dOk-T*R0aaNT(\lS9^N]3BV.QWb=te^+>PS.%NJ]Mk?>mRDFU'iM.bT +&MJD-qtYlb+UB@QB(/AbMQ&E,KGjd=DOus<41=sdL3,g@'>[W88B%f)Ks9N&g#PI3 +Hn^Hq9cbB)N!op[IV;dZ?*!ZbRR2u;Z4J>[F!'q%/QdL>0^NdiT+9ddO(Vfq6@NZN +q+Ze$(56raRA[q;DF7aYUjLq.l5bs\Wr1K3Aj8C:gWL.XRE,,s;rkQ>p"j)8.>#bk +55M_ln"6LhIicW-cm/=oJa9P%@l'[kLU?W,[%L>Q +C03H-<&AJt%%Ui*_?"*eP7`GA6"2!Y(&4-W@(9+i:WR*R*$Q-6k[;mG*=b?N5AX.Q +Al:a*?FKd9C$o +.7]Z(6rp>,qD33.?V^ +^L<#_2aQ$s"HDq6&-4+;!>l(ZTa7+F"jQ-'D3Xius52l^?Q3*s +rrW#fW:5:4@fJJ>qU\lQJ"PfudtmL3HGi!iMI&.Hn!=A6jTppq`&^dB.Uo4<_)PVk +C@"+'e!l4`lru4K0l7Q')-k+Y`V!W`$T`sUHq_;q0pd!O06@/kfHA/daS;-R +qB%u'I2\%\%_lTtIa'j&+s6a_Ob3EOpmAVtrr8/T&q8T5FGPZWi$N$uD:0bW>8m.C +DUQnRV%;&r(`/\i/\d75=pl17GA81I:t]]Wf0i4a;i/ba$%`GULq;"[4Ynd`=]eH\ +egjD9N@+H0aM9c4`>elHKEG5k+^Nlp,5)"-'7U9a`?.srm5)]IZc&GGLtGXNlk8!Q +.;TKjOo^!LbdBr8D`e7WrMHb#k2-=nS[hIc\P-?jcb4s.JBPgaV__D8-I2!J:[KKh +D+9&4GSu)OH!UK8R@WS(_mGCk4\6&na5<,n)4cqQ&HU>&+:,&Y31::a_tJ' +@7\/.77M6)Lla^8jb!!sYR:STE=OuG%6EDP?$NRDF=qV''0c>*#09Lfl-El957]68 +(5Z*p+r-urKKo??3mW)6k[l=%O?^$DQ$AiQoeMa(],(&uq=n39V!/ba>^X>JAKi*S +Xl)B]i&GYllkcYMXcP";A2dl\]O'Oj"4BF-arTW&Xe^bh;C.A2Z"IY#ZXT>T9Sql` +40*ABn!Zp/,P?Z@O_>dEKN'3)>PoMX'EjW=U6pS3'XA?E3Qs#+mT8;V?]A/hh6Etg +\#\J.DWLKU&IBJAMOp,.RD:*X9?%#h1+A-=e7D&eYUE.q#["m?c!f"C#nR(.;Wjl* +o=2/A9_&=@]Hj,_TjG73!&c59R+RaB+n=/V+XA.C$:#?A*]+Z9+hH5;,A?pHL](t>?K":T&-p1_m) +!Nl8#?2>Ze#CoRA2SE0JX%+BJJGoqY*2$XdY>76h=-ML9OUWBU)?rdjr'+pmb+&hX +1PQ/`*AoI<0Im>,0A88l)]&r$'a*BA@*#9gfXM>NIniB>*o0QaZBW%EB4?b0g6&XGjbc92e"4,T!:o[KXX4%L-:FJ>$[6p_%O>NuFH&j5T]JcF-.*=JD< +)^>kl/T2SFpUdGC2C]bKWh'tESebLIr^S%M6N-pd2p;:c=X_-a +)Z]o5r."Q-2,K>O!L+]>PAg$l`W'I=n@s;/"4^L(R4)Q=OH,up1:IX_rJ:[#6QsBC +!1ipE(Lf66AiC@c28fcnCV/#H7T'5r9r,.DUS8tW?CK>CD[5NU9EAK_"9iYoh=,8 +"[>BI!*K)!5#VubKs/il!U<<+i'/u5O&l.G8%31;hgQ@>l6TC.d!h\V1J`&jNct1H +'g8o),2`Q^!\!s0)FWR3I(UbD!Zt@LDc=).Ui)ql6@8Nr[eh@\O1@Mp)7OQ'200`l +FVlmBl.%lmZsiHA8\8tc[8oVf63sAPR[@lj\9!Op;l[blMPO/\c4be[,3k7:p*:\<@IlS]i"eK8PU]Mblc="`<-.,L;mhSb\^9,B3jEgELW^gi;tj;%Kd((ohUH*kMq[?HR+r61d==8dY-K/aq"1k +eb>3j8A9<@8fsPA^[ID:<@BU@M,Z]/b&*8YDe,-ME`@3p[fC\V`cYB,q5PF@=!W#s +.t6$!q4SmS:ak#^`(;="\baAo7I[3]V4V+.*7@56iWV7RGlH?0IS)QkiBhoP\31C[ +!!J-H\5HK"=Y._k*,7ptR\a9-,4n8/X1CBRS%2g."hhMo6AMAsV]Z=daWm1O;5n47 +JglLE`KUQ%jCqlYKN&q;7#"gfP[kEZIT@q8=u(5*1jMt]FUjV?JBM8^a^Ncr9+(J& +Ek0XY)DHlA:1;U`AS?/U[!@@a790299(ntl\jc\D.7+SV9e0aMa_/jHdO4+^J8p:E +1&9%%)(PK'!I.m8]tT +.&M,AkZ4SQW2'I-lI9.JN)$4VG(nF(Gma%G0""D&V-Q*h6\WsfRb"H1m8-Am=&*O0 +U-)M1YM`kZ[sAh:V3VE>pqK#]OFH_u;9?=>fn`KK0P+*FAQ:a" +;@LPEYV@%*P@bOcQISpreYAuVkW.'\V9Dbd5libI\oi';'USuU?DGAddukNe79u_E +%`0kL`<:LV:)8@D4%dZ]0IZ-A/8QWp1h^M[lf=]Db3;f>Zr/C'@nTgrMW?">jJ!e3 +.[3lY7:GkB*3I:;bCln98br+tUSs-"M)6iA*>/K8ON'h/c.=5e:g@<[sGo2liT.#:t3UYELV7C\:&pl$L& +$nt$+Iq*1Yn@phUi%CW$SdZ^V8-*%=/dTL3J`DJE5beA+D_3JY='jh!P2Kgb!Co"b +Ij/O&*Da:3mfJpU,bOaF%I5*_=oU6$oUh?,s#hn+?SfS!3\KuQ%>k,3-1nX"o=0fj +X-@fd7slFb\iE'f4:IPm+Ii)OZJIfn+glSWH$h,Wl6'j@3,\8D]F"i6H1Le8Ac"T4 +$iGL:K$Naj-#I"QGM]7c"6cUf$G&YB&cl15oAL$dSX6nF]Dm_.n1j=g/+kcA5,@NI +N%JFFm"jpr$4YVT#1.C*jJ:%mcOkcf?-K5fMq=lGFZ4) +hK[@qn=I73T=oE>]R2JXjtM)J3-5%/)C"+hVZ2Wif5'ET;q'#N#-E[!VO*bE].2A. +8lb`qeM$$aG4F\k1*sQBU)eX?"tkBGS%UG#qt%"utLu1^)N*+aB=BHTS=d$ad7ST0u(pbP@Th=NZh3X +;J5+>B,/nY>5i\<$P31sD0Q:V`:[S\fW7O"iB'2pP! +$S<%U2b:tHitGW3;Hi.Fl`b@d;DHN"eX5PMZ&!/b8@Yc6)rHL/T_p# +#u.]s9sVl+Wrf+bL"(,aQk#3pQ>lp +FE9NdEe$%gH7(H9\TW.s:3NsEAfDnk9>@-b:tVG3;m2BfX%fSA:8uf<.TMKX;2D2N +7Mgtk+Aer$<1k;%XV"TI'[uP2,POJhN:DNc;\Mr"D280']:5e2Y8,I&9\70KbLK5& +fgZ^,B^i7U@"uPuMuRq9rumH2+9Kof5em,%L["gU2][YQ#mjpks5Eub+>?C]2$24= +NRK!Oh"6f&&]"G-#Kd?2EcIu/#a/k_)S_UiV_dJ\'+r/h#kbq97"?,@R3\sjL>P7Z +h[m\D.fHLE^Ph9g0>hQUbBKSp?!1KL4LYc[#HgEB_"@p>Jrc-^r^->$!\UL#JHVR' +qg<7pr:>bM8%Eh-H4o5\r^cti-hWP_[M[J1uZJSr^U\8O(&]j9V,m> +1(hEMoM)s;A(SFfcf%NXd1cE)F8N5k#K(2_6^JK>=P>fr\qWl2OA"8*,B'X%CoE-Ul\/Y/Ld/Nk +4JcaNWaS%XX6>'g_7381KWI?QMcM7N/Ah +%c_XHrIie>`&f'I;3d5LB?;_\WSrD9U(Xc(7K!+gd;g#Nl@lTjE6KRuXCVaoG0'(! +EU@YF2[q*`Si6VH=cMnIn<`g2]6QLH.EHf>bAF#uBpn0S(33I!n`-:plRJ;LG9fTr +aYVkH$i^St)HCF1eLi0o')YK+W +0tT^`$_`I&\Wd79/9F3@CfD[) +82(;d((+K_-mscFQF((oQm'<0;;m07TnG_GkSN"d6NXM[_XW&+0nU[T_\*:S,1/d; +EC6,'&SrG?bG]g*\.\UkFQ@s?$s`._M!bk7l4o6F&rl/jM8'/eef&M+4o\ +1Kr2I7`!FiJ^?HOn9X&L3Y@T[iGbGn3"/as_?Uu5Z1X875Qde-'KZ:)#Jf;X15VZD +kVUDF)s*5tYd"1Mi+!ck^/?-fW03*k/:^tj'1"CR69Lu/19t./U/W\F6o;8A34Md5 +#[6GY8q`MJ=")HrdW%rqE,dc1Qj3A>-5CR'HsDSEC6q\-X'.R2Thaq)ao\6Rm.Y1+ +GA1YU$Ol;85TKD.EZkM\oaCWgncq^>5T_/aT865UF_[[ab+ZEOFE>(e$@3/=g3%I? +[B^_$W6@57eh*iLbc]NiQ)g!><>3P5P0-c<.TI-T'k41CacHSf@mZXFI'pIHop@K: +9&68L6JI;eKDRjPqEB+Gcj0HM2;Ck^4\TtRc#m;C2'$O5*,Uq1XAr*sS*TtorQgl2 +-_UJ=('3WOTZi7c-n!8QB<>9;SN&RaG]uKK(Ej+_mmB_@-)LD$V>?Ci&O9`Uf)HkB +LB#`X%8r^=P8uKD+0(E>j:fdW1&;NVRR6u\4RQ*`:1>d#6kLA8<\#n*.rf_?%%!q- +NsUQq(rL3Njg136Z,UhA_t2O>\KAub5*FrO+HKtZ`t(:oiIQoJja!IHo0$+4#Fs=t +4OaO%LjU(aq-1N,B&^<1l"mh/m([CIqo:moIiT4&pDJg;SJ(hKYim08Ljsk+I+9@D +?bs@9cTdREgmoot+!F_QDc#V0598RaOjq.:0R]%eI[%'ZT"'3>\p9aKGb23ucM=8P +*[e%]-#B4Zot\&!/*G=GQa*$HZ0N2\CKGKp^ifS;]>SR]p0IUAFmR[ZbiZO)mC1JI +f28"rEP!;+pYX#/"46j;5.HU0L&M>J9=XeG_CgcNcp>K$nUV/iP'9*emX7XZATiT" +'Qi%621O=4!%%U@`(fC.J1kUi^J\G;R!m1`&Vdsn9\G3[!!"%t'4[F1g@&a2hbdb: +P.*)S=0LW7i9u%.Z[NLoH5XC#S[4.6oC.IC0rsbPf.J6_:IAcFPG"mBhpG`]T$:mg +M[pasb_fU^.-:6nHuWN)QmYV\KdPeI'6g6E4@p97PTY?,8aIKel=%:!rA:d+W?O5L +_Ud)2M:tiqm\^EIb7M!Ujg0qX[>[O*.GTt7n,[qAe2c:3'?BMkQ[uWk6PiOQc6!Rf +gC9e3*BCMK3OK\b0cN)1l93S]37;YhUK.=tXnn;Tq7_6mY&)CDI-q2&CK3D9=I)]d +Q_CP);H@?BQfQ&nM;`C?a^9I[c^k$#c)p^G_)$>N/maV%S53>AXdV80EtO[>G0FGs +`*64e_+`rF#NEp\crH/A73l1GWAbMAej]F@OgS<5ejjOVQ7Sku8>n@C@YMI52#Z4) +FOIt=>Jd_90Ad_a2%[p8P:s>//bd,CCHYD_0=*#*\Z5838+]dbcWV*.?SF5VNdPTg +r392\["@Ib_p\Rt+LdoBhVsR&HZE0@+m>)/3lbe8hg$*s/rb'3#[&K!)`6VM"o%IT +[/VhRJ(5b7Tj@&0Xc$gME`42AiY+=&@"fF;*d]$nVQ\]FdK_(kDn$l=6 +C3ZEP?rXoP(_cN`k49JKp)`;b)@.FT^_e12Hg-QW;*[qK`IGXr>l]LP$B\S0oB@IugA(f57MhK^R +#M8JLeI.*g$AUKqKXZb+[R`ls84Gb/:?]KW469AOjAUDjded$#qUMnV1pUJeB(TeJ +okS.MC?H#hjpi$pL5>ZCUk21edW+;VFS1ra5GVArI&\8lk4UFk&*k%K\)YKPgBs=/ +lU)-=W<%n%Cfeen;l)WYTscGK2_.ekjHHWU&hC1@APuC%R)fo^La%tC'!X-($\UhR +KgVQ1$O"l]r!Pn[njNV<`']&1WJE#dqBPK^Pu?#[f[sATI%\6;mQXa3U)>"Z>O9!LuD>s>F.0X^$q--&&+ +grRLakaUGb9smgtS*E%=BK=eF3nPiEZW;')fCUdBr'1/+=T?o_eGfZKs-]V^eGn`9 +YZA-;cMu;lN!LPT_dAbpIu9R)e,SZ_LOQ62+m"P]l9eS@J!)fUc4CYg,nd/pj,%8e +Q!eSD7ik4W\M&fl8@%$9@^j?t=^jBCUcHh(^#6Eh@l\CG>RtDU*m=*+bOaH2F$E^` +[TDAh*#*Xh.W@I!E]L1>];kBfjmT>qkc69PXtN>Xa5>h#3ulAN8O^osmX2-Ls*][< ++8\IAr'-5Lr\Pbm@un]dAs'DifQ5g/N0Q-4t@ +i&kiK,'f23K*UGq)MD8/_3hntJG +$cE%>"@<,PPJ<*Bss.6M(,DC@&Xm4/ +7O$ddZ`'/&qPNZ#1G54,H7b/lNNLdFt2%)-@R^*dUrO-9>,-hH.e'n27[ +Db_(R8f"[('Jtm21I2U,NY=++ju/1[23-iUMrc[Oj=gqU@gGK-S!8gmok6fI>3>9: +W'>=d3cu9OlkH$P\i5(I4a(^#EbROQDUNMkW^"q24Ots2V,=t);S9&8ZltdJMU,VF +AM$JbX^LD_E>Fa#LQ>`kN*=a$hRi+R3ACHG?gtM,l_7S@0L!.7c`)omrX87C73&>P +#5SH$L(h\I[4K4L3;'+=fiM.DOT68;&VQ*Q8QS=rU.((MA6+oik@/_V6J+TRrIOPOU0@fB699-`J@d4. +4"\I$C6JAAcM-oUm^L*^*T"G7rbPm>8:LoN7_<`j,5kf&gd!!F@uQ?L,9,ke><0cA +S'Z*MjF5j]ot*E$hfKeWV&RkdGRHNRX4T"-\?-ZON&ircH9#:-Gl>5L4PTY&G)JSS +PkCN`LJE[1NXPWCo(`c8HOY"@69E!E"q:*LNXGOio6Cf7HGC'bE4bl8G]\#=pC$cg +#e9R7HZnqk[e%&l5?ANoI3\S33eRe(]j#"kSI#6 +Q-C/4:Df?l)OJ:T::$8:o@Cs:5cA62RYY"Y@@0p$P4k3oC!0l6? +9*[MbZI*03P\Kg]PR;53.42W.,!8XX3*.@J8(aFd'TagXPTpm4/J+:"'UU6f"@,5: +MMFP=<(Y,0@%sZ@d6JE^mF9:sf#bqJn=`92R@-7`.PQ%=Tr1HqEncG]c!$O;3p`DdYRSla#fA!06ht7L81@m?SENr>]\'fFUd<9$ps/EC5ptsIk)>rr? +ni-Sqhq)J_T/$@5!<7Pics"FAL +\.V`3C8Ns;`'8"GjOkh#<>oC\\H@bOW$iS^&o\4iohn7nGg*,sg60+\4EhralOtGCR_I"B2 +h@sce'U>tqh"NjLkotQ=*UV*Kh"I59NJka2e+iUbV"Kqi25pFK'"0G+pWm;m:;U)Y +6C4l"<_e75TQ;$W"@.aa7HhlsnFC;?dMY##E9/Q_m4rYR.VRjjfZsB`NN;!&\"RU+ +_DeV8m;3Q5^t*?R$j:`M')$4,?@n7^#d33s@0?V)!3pPWL;"f*3Yi`F5UZUTEtCF? +\&f0aqb3WS`1ZR]LZ&PY_3gg?QaCrK]6P@sg0_OLfmj+H;:lM-0N%U#>_2nuV\*^[ +ETlt5cm&)MI,mG3j=tfBM4WdXiA#E=3dL!QET/g(l8WtN0R3)sJ+#\l^0T($_\]Kt +p$LW)=>d:S2lW=o/*Kr-WK[f@'_]`aEm*HhR`8bPcXRj9@hHUq)T?D1cDDDt7LWk[ +"bKl7"'@F4fX4%*F[EWp_&U%;Ct_2A3['<*32.(?ON?dSPrRYuU[2jpjs-3/a%uS. +*!S%^LT_d/?f(012JRM3W-b:DF.c[@UJe`6r`:LQPKVjF/R6-SY$pqQE7p`OfrGH/ +b:9bq$Jp[DY>^4D2tirgC6#XcI`';oFb7%BR!W<]:Q%W+k.PW@dR[l1SBLHXQatb< +Eg`Q2M/Vr!O7Gt&W=Rb=U^)Okf;TC$lXc)LGs4HZrNT#i-!bUOL]=)Di4P36]7/_2 +`mq41Pnm99VNNoG(3(@],Nfe(LNu3LhT,a%R7Tp"F4l$a[J87IZ'^>P<$D9$NVW$[ +mGA?4])rY,s"kVHalU9O;>AqGGZH,6==E^Eamc2Ep]!'bH!%qgP)ap(`W+G)oX7`r +,29?B;C7T+ZC9O\T&$E'bHuuXW0Q,CNT4F\o!jt@E=h.nUC],5o,JEa)WuCocft:X +h78=6=W`FJ&.ck>jTYOf?4P".Rm=[:V-@m`h`P/-!a^L+#\/o +ojZ^'UqGu::Oi$lL4X==Otk,_L"3Jc[!aa]D$3:eL"1.dYU0X&Vcp +;^B8Q`I';:"+n=]TM7YcR6ZOs+Te>909?Yq`.jO32NH=d;O_dUaeEMZ>-TincA:'6 +Qqb3Um:E6bAuYCG9[26ae'1PkotR?/]sX#uQ&:L1QDZ1R%Ggi\!Q@:uhoAdDq*'KA/nebu^+Z:Rg09S't0&gL#^d9V7e` +_cu\jUW";mM*mTWO]U6&(F?`_l@a%Z"JFuTfZ*luJa9[HoOSnX3_OKI;kc9L[+lb\ +(ZW1qDL%eeF.'Ud7G3=!s#=WN0RW[EC@+afl?VCO)0tNlH1).G"]=$piL_c2pN8o,ph@[1?^OMk]tuSbOb+HI&5\/78:X]aNW*oO!%d[.Z+sr@24: +_JeDWJ)NLtnKMoYJcGuY.+)7\!cR+BY%"g>LkCVaNRRCIPHYcm!Jghi+u%IHI!usF +q5nt(=p)7+a)##"DhIYL>USlC`b_27Y.kHQ6k(6Lr$H[Y""]&@28BBi=k"WuP^t5`PpdE-3-MD` +2U*(,2%icF6'M`\XSOat(G'Y%4I^7U-_:Rg4S,(gjuiCjEp?bFkY/V]>c6VEB'YS" +e%S0/6cJ2CnkDW[22AL!8+I1N'T9C71)L@B\($U#1_Z2:K5!roCW^'AE^&Ljp-%F3 +NC%#3G]K&A_ZK5e-tGILj[C3i=&W"H>W2@teYbN!VK:ELIDYg\0YJ8%F%\ +KL1L(K9mj9M>Zeb]Io+YG#Ra6(mB3jEL],1(uAa2ml&2E5/cjr%!0#SYrN54\LpF+ +0$&dUiN4%1o-L58Wci!HJ6tBnf8sCmLiCS:E87We#%nT\nePboLiPq_g+*o_X-SpU +W`QQCQ9aiFC0,NshCcu/C'LXbS$nH<+h_drS_p^nfQ/%cG8f:[(uBEQhK'amEuj.X +*o1uX!I`BUWO=S^_SE?)PY.OKIX<_Wrd+a%T5On^^fd?YQi1S"j5eV'f\OE +\AD3i6Ie(^q.$eAij:])2=^3?C^Jh*G\qi@*bUuQ=/#^sNYQbl42*ipR7K`3ZB3mi +r+RJ'FYiL8:7`984>6Q@Xuj2?>(sD"`Kj+bV@1AT;l%!6bZD,D+6K_2gcrEmhRuV; +\4UCak$KBPSu6C2VJW)=lorNRGas5%OgK=tY^[D&FVRWaN[@,bIiZ.@*CU(%`b]%g +%h4WmlNm>e#T:,^T`Ir@;(_t&SqL![_Wak6i$.8DN-Z/JnN;-hH0k6_KO(M[TFc@9 +]Q\F@qe>'@<@f.hCu?LtIL2$u[#Ma%8&.e!9Agt"-Z,UfLs0'?cVjLDVXD2i5.F2> +I?qD?,S#:s`oBVd<IMISEr^@+g/7@S%ht!0Wl-sqQ`,jn"q`Kc%_MQ_] +E9]3.jui[bEULn^k+T]bOo5u5Cem%!SN4'Ggp7jN0cSiQ5hRpWh(\8m)n,i)UZQis +RRsX0*neaAOf2/!^O3\0(9:.V@Zc3t9>_Vpis_)Qp$Ui3i`W\)%Y/Mg=pUiBK#gHd +aiABL`j!QQ+Lf6"d5XOs5gZT<0kaHDIFY-@a1C<`Ie]&cppji1PtB;3e,B`-hK/(U +$gjNYc-?^ik&](a';`!%Ku-"Kg`<6g>KZ]>E`n=[8b8+ZgBIs;!t&+hh!7@mm]hTq +Z>isF#qh"SOTIpa_6VP8m"`-'^L,Fg)oI%d$Sb$\GYde[ZjV8hT_RK! +m-e$K^0+;-1I^D$l^E<9)#qZRqf%SUDo_]]Xk`aHZ2ODi:2S:t7?uh?Z.Nb*s$%@O +J"p4s0HdW+KKO&3phJm8Dhh?MDjJ"/2Q@R'D&6"ejKfrfEpr%1mc/?Q_`LWaqp('% +9F?$+Q,8l`r$H]*QhVQ_fus]R*!Yqn(sj=3Es/BX@!=rrE:%4Xl="o)j`53J[;/75 +4#rfm31\e`SQa^;#D*f0^E>6^E=`H^P#]O(F!8?+?HnjLq&n-#3ppOf +87XQ7V)F,VJ`bQTVcGTb^HsuX8g^ZD2BA`Z8ZBFf>Uag)=t6LD4L%L0^.C?h4]=Th +\Pk]"9DMo+r4P#alujeUQ1Mfn\hjaUeaqKpF'=gr29;WGaeUl:Ai2*omnbYS2en^8 +D4S+P)f`&5F%cuu1(1D9$ae=R[o*J0&$T++[8ut.ud;.GO5@#riKO4WI&X2U3ubFP?IRBDHDe&/*IE?sji-gGa0e`4e/j!q@pW:OKc +$"k8S?sCCd[T17c<5j8ha*$(ZFkOocd`B+/01;iVR4B!nbnY/T<>p`(U1SL3bbUl. ++$I6No/Ms--u%d*Fo?=B`oE"pY^jb3]20))+`MmS)j:56.E>6DG9QRmNhbILDI#bL +>_\Gna-"k!dIa?aoI!MC-fL<]^fA`Z<:GNS_Z.J7\TcL+O6C])9ejmaIQ_p-gmtW(-Zjrn6NL)&L*?bG0$QU:**.(l@g)<9 +k"/D.cS\+:rK2jGKO2(/SJ@!B\0FLh!`*S,0C)^^2mI`e_Z,BKae,M@3sGB[?\FYo +/B&`Q@E;V-o"g.'T/K[KF^X_Q`T3bMqo^FV?hS>c3Hk2@Nn1+d4MTcRs.;*\?d:2d +):L%b8'Xb$2D`=L,ES7_r4cT>q%Hp()kuD_!Nq>ajC'e6VGlif#/)WBc\*Cb3['9U3XXr%r +p<>.s9o*(2cCpfbXd./-$0TgSjjCp/h(KrM6K,'*_9/5lM945.]u6::KTqBJt8qRfTO3 +5M7J$P%n.']qFaVYT@`o-B07LfeN9W/,[/c$_Z?>W!&(Ct0 +ra5b7"Ah(n98[">c*s)!pG,nFE4K_Eh2:IB%aOlg%8$nU^m3rDT26cjd])BgieCA7bOk4q0O\c +3N];AmHWnWTI-MZHMOp/\[5Pne->%,62g@Oh>2,dk"2J:TtmZbCS)_J6cf90X>N9@ +_#jPY<;m!i0otI1f\s66X +WE=b7E.X3[]V.U'!GB%GHa4fARPFf50jBquP%01#2aVPE=RqDRo8M@C:"\kQj0FVG +;2gWZIC=6'>c',+8"VrrJ3?D`2+ruV$SL_NVs&+HlZ8m?EG&L[5B(.O$/ceZk$L[B +T]C*05N"`(i&n1$ITXXs=2Rh7&gUJ6:)461h\5=70dJ+/mq+/BB-07\H$FUq\fERi +CgGFsO>l4H9_KRa#bF28,+U!\Tj@!Y%\G$8?,c7pNno-sa*84m_t5#b)h8P$ld'hg +_fuN_*O'kL]a24G1m!(u7!n:0Id+7\iOo:FE8eJ=64.H#WV7%j#bb4r"X!n;$!KD" +>_V5YZLl=Zl<-(M`;DZ>',k6`kMc'>0H+mB`;iiHMID#[*/CVH7pO(jTO)j]6%9MB +K;.B]Gl9?68d,)1GY/YP.ZU!;"ut#@n*5bmdNeM*1XBY-0deUFa3I(0X*5m1]?Glh +T6`&=_S^:Ro2u:E+6gBG57`,KJ=d16bHhI.%`\,0NTVHAq!5*]a]I?"(S7FD[m_Y8 +BdB'jSJ,/C$83>m?49 +.\[#"P>s7Rd7Q^3;fTBm=Z,[-ZkD$@j)M(a/;mso+r>mQ5Z*VqR$P&22%"VQ98!_q +78G?Yc75_)Q-otZa`I%;7)cNYV,X10o&uX#/lAe^;3(sFg/SbT\Ql/fJngDf@NGej +-lZ'bbhK,9g89E`\:^"qYmbsb2qfLgjgV[6Gs-'8gMn?-l5]P@OX:1mP +IJgoKU622pM8R4#'=X/Q,r6komu?-u*tug?8ES2ijip@40I\2<(!fO>bfc5'a?E^f +.:6,U=]MRO.G$1$<\Gk%/Lo'/6W?Yb9g^Ut`Ou +fL@W2a&k5h=Z,LHAHmS&T\9h$C^2C)]qG2Z%KJVq@bM-=?U0:2Dm)E:6@t+BSL_]# +i%J'Q5.^Cec)MlFVVNrDG%f'JA=M9:KH34jBf?<7&H?=b:9<2_hR"Wr[HU%Y +h-kim3kF?3afH1G;1l(l!5SmJ=L7C+FrBtVg[qXPF[O)nWu-ja:rkntg@d6[Nb^#r +[*Mi2NuHUABpWsXPdDptYEDL,&rluS-kM7a&.!b.)I%JKTgCi[69raUH]i&%!m+k-rZo42>DWT:iXdVQ9 +)F@8IKD3&M'C.nV>sB8;Vud%&1jmQtHnEirXYbGW;+CnteHiS7$?t!VE70_hdYi:s +_*I`)Fi))s5Pbod5?L1>^[?R^!C6Yas0)H9s*qCYBoHatPS%"_@A%eXNs8_%I2A.W +4.0VPM'S#6a];lm<[,Xd]IB0%RFYX;D=C!&X:K-0C'UQ[&SZ>qD09?*ojD_0_RFjM +>p8WDjAUI@cF-T&!<4K6#i)9&>O7J^neG\HfT6$\UT>7D!"[f/"=@mVb6XBD_$l +,K:#WND2GO"d5R5P*$2'=X+])bO0fE?N#huZ,?'bZP^hR/8-[='(YXMF\(+-a*nZG +GS7Q,*D+3MnH[A5l4Zr:lbUQ"i!%k +7lt-#"fsgq)?FFN=;h^1E#QB4&OnhmV.($SU#-g6!:+nKZN5Ms&P76_ZOKeC70cp* +!7D=K$(lV/3d]dOfQVj.Hu-u@AD;EPf62PfD?@+ipf3C1LhK#fs4W[Pea)/$-R%gE +`GL*d\G(I3m`X8-kL!seer0;TM"#WE7+)'K4:YZ3$LR189qj +-mp5HbdL1LU6L:Djro%rLd;V^$6YJ)Bt<(M##u=h%:]07*lMX0Lo6i +&0tIcYh/%2J-iAI")6=PW\onk4-hK8A18OEDc1:'?Fdq<4jO6P6h6m>n3D3m3A*T- +MqH?*DLXdIpA'C6M]UI44ilebr*Rd-IPmhgnZ_(O3U:PhqdGlm,ni;>Ac#q/r)!oD +;M@37kL_@V4sP#r2N.TKMX>*gZ;ZPV\?56jRbU@rdc$!57,Y7a`P*Zi5AidD%,\U0 +=5#/:3NJ9id=18RIb?Bo'L!fKHncuB?CfenPg$eBA;_^O"2B'sT6;TO:+)dhKZL60 +\M5X&lSu:%=>\24XZAkZJQ9GsB@O_3e:8Ap+QgqMf'R_F/l%I6DP',,J-#Y2+i/s8 +reQ$;n3MWH+l7l<^4]@WPS)l=&YZR/iUe:EZpF]@@K*WBreqH'N)@E<@mX=^,58"S +o*LS6"Gu+a\]Qek3_]R?'Pj#!C6t4O0`[rHf!,>:1n[IqA#_6fFP(V\V/\<1DB@ma/c1A'qa_oJX[l\bb(RTCK,a5Y_ +UR"?dk%3RR!'TDp&3]i5aT*X)[oh5t#9.%=?l$HeuC-5'N.&5dm,VYYI&X^l7# +o\DEZoN(dpGPWG4s)O]Y9=l=1_Hr?MU>-f-W5^Ju1EhcfQZeE)f[A&,P +W_SQ%+ku:U^rgK*J05LMaS,RL2:3kGAnleWlMaiL^=Km>i;-LS +d6iIJFRFHhr2TenR6c.,!E]Zts/#`!@2?hEeo1G8U.moQM^&[hJp$\`[(laKh,H0M +j\kKAPO0Vn*5p"54^"- +ffN.6f+IYqiA>M>B/*Um0E"lf(#hXfofO)Y\jm!lUDVoKO!?&c_(5$B+S`u`$O:RU +gQ%J%%MS:jnfWMfloWN53>92M^lKQ6Y!BW0,%JEtKBtf4?3/XOW_2QJ%?Ukm"""D:o^C/g!+jKWI_ZYAO-C#n5P +F'2N6=k*!RPX7dE(ssp%ZjP3h@KENG/oj#]&bb5tZ9^Ip&3ara6Q&1sfoTH3ks6EQ +[;_CF2u^2jai>]^0,b;7R^Dj39'=Ni2W%>9?B48Fa!2&E8_#QA%OAjgo=q/t8!d?0is3XQRjlC>\%if)8M-Zl6Jh_r#R?I5!/G +]T;0dh"4sl*PQZ0(35?^d5-=+mtF(SqmumaC3rH!NrXj +NfZQ0fMTB$0tsG,hS]HOKFrdb0qi\gr-;42psR.bmS+&O9SqoW8/09>(XLqe+\7qP +aB*ug0<;lmP1Z=PE.Xr!b`TG@Nj$i7MJh7%9SG/@Y9 +JZucP)^8LnjDcrRZ4/_ulC +W@$i[bo8Ql]T_`/<2.1t&.0km>C=3SE!u:tSL,g0\TdY01 +Z?aKj:"Ou.gRMF<\3\ZL:0F38;2aeSC5M!V\)I_u,Dq1!b!]_\LdY;!NiA:=XHJIC +P_uZ-:8k[d:j+3-W/s`T,Rn?Wg2,6R>:2NUf#R6;`Nc>;s0P\C1Q:= +Y2a#ol"d'UB;r_sIoS]PPlPU,-j[91%^jalF3k+-LiguE(Pm+]rcL5Of)M$8Vtb#) +#O&kMIrlDjR)-C;TZi)iHLFl8jG00?79H8\5G/C8N=-s>:^[QX0+?:Ro@;.[<6oYb +8op+fFhX().\5O<-c\_"f9`\ZT;O#lUEtT$56;Sqq*\a>'qo0Bd'HglpK9HG=;goc +)gM]P&#&&/e6IZmKi$6$W2ZlZ<-32``?a%!"PdAn>?`dsXUe=oJQ-t>/V,HYiTpZg +CkQW][=#fM!kt3<0#e!reInS/@bJS&q4;E%`jf7/d:R19e]/R"P9e.f_BL4EH6ZjG +^nIu75,;(&mX]1ZiU+^d4)qsFfd)p(%Gs@4I4*F:CY<7==/NGN:T'kW"EV^\\(C`E1os&?gN6hr7FD*g09F +*km18$8?g1Sq3ns5nQDt%O41EVa.up30\FqZj'po6r55#qNHqf((WM5qsH6Wd"X(' +/U6D?E#!`CGp'bQ6=BW5Ds9$-"<%41:s+\MlE_&mbLgZj(GTU4'kI(^kof>r&iYBn +GIQF%g3u83k=s+eS!][pM8P$b`$4`abOlEGAA1OE^gF_igVCcZTG2L90n +G\l#^W7Q*`h79T>ZDCf!;_$Nu.AnR.O^rLp`)EON`D6b/NU]MtE!6]VqtK[X]S4\4 +O5G;7hlEBqZ_%IC)"$:$ki&c3iNVS+Y\`JZD._8#5r:s]"RQdZD&,O%""0'qj:T=' +7\J*pDB[SCcp_\0r6?)\%\J_"I57qr*QdkrCCZt2S*a;J>Vld-((#tFTmJlMdZGp% +gtsY2I6gj-@UA?=)",RYhI"R8IosGViNU[)1soBWS@_eBj'MHJWH)>YDa3RKGhU%* +i=PGU60-c(nLnKChi-MY-a1CB18,j:YDQRr7t6QD^N +/+lCWmQ=%+3YKAl/1,>Mf.HjH&BEC\P(+URre51X`OO??o1"45oD+61]&7FkMK%2= +CD7tUV6!R;]ZLb(Z6b;n*F>q/`N#KJVA!UoK6-BjF6U_l%'b/?E;GW9"V'Df#B55= +*20-tiF?+)j4ob14AjQ/G@uV.V?E)p7*m6nj$K`Oin;sr%eGj;96X!!Z4BLsAL,7c +++cK?tK`HLLDQunFL>H+Le)\u]fl^f-O_-OYr@*?-2 +_l@q0VLQX8B\-Z0ig/s3]OX9SRmNYSV''rqgh&lNp1HaiACd701CZt`sWCg/W2AZBt'XA7TY +B(_*`'g]I#W*clE<537tL`I6kUK'+H;71'GISklk"U6"K-3@5RDUF"RP9kPOfWnB' +P$oT")ed3ahaGn0?.Ekg(jAj%H>)HS'FXC3Ybs[lVQf;gZ=kJO\'13t:)I"BlW9bN +HrLJgA^\c?ic9Iiob5.MZ/3)>-6M)Qo=J`76<7 +!%f['4^,gAWd4QO'g;OBEMh$1k(]InRQe/`^/g]d>TbOsIqp@mTE&^-+h"QCR!/-:_H,?^O\=M!BI-k=V`#Cm!.BY'O0H9Pp"j=+nt;<%]b3L(Q2U(IctJ +Rm4Hu\.mY#V%rSe+ehjhTiN>9)TnZ@t9D@PtB%1ISJi +V)sY9\,L;+N9rTbmlF203K3F\Nc%u/m90tfgPd18'-B9[]`VT4\h.5rI4rF$B4:FF +&K>*Y9[R0X.f;CC'YqH[m&oq:!6P/!e'u)^iM#EsV#&,"VQ?rr[2l3S*@eN^<,LMoCtb#Rfp",HOt,=S.b^-.gD%ui'$I/.k"nKf +a^Q+]aWgI[Ki1M?$-W7oo3=RLc@B%mhas?Y.#[sFinB#E`hE^M,AIK.`em+E:Q1 +s2h(\BCjQ$g/n,Z9DD]uhnP40^RUSa@e.et`sP$Qk^>94rH*ke40%@#CZ8\phg4"2 +]^A9RT0`I\.eCWdT=e\7>/Y +8oZ1VUcVBr7u(`m?h1L'os[lJh\3PM[k.Oi]&'J/_=r`(^0]@h@`eQBCN1`@@C8]G +BuoZc]PIbL9Js"i"-71[?EK4)TF^KRJ9MrCG9>i6)WgT[iU@q%&"M\h*G#TuB+LQq +Nh#*fSE,KniJo8U`tJU9M8I@jJr'jJ)2f,1huZR"\P#hGA;.=3V/ik^"\#tjU07XV +)HjBV?l4+9%GRidD#hhb*Irg8B&#[07?nsfRWiRMFTYV7[PbCWFe^k)75O8Hma<\6 +^]iR`$n)@r$b#F,GF86AjZ'p=13VS>IcNM7kW-*?0tQO3!ZZHF*V`;Jpe1^D4p59N +B4fV>FN\;*,2HabmPY>e\.8U>Qi:">NYS&WAZJjpFk`?j-TJuS/LYcMfqN?>]QG6* +gq4XqHPViQ\!E[e0rKZ,7/h<`ErD^u_kMV/S:1o$X&hIY.o9'8H\WLsXdHJgk>"?# +C'I]I>=\tCCHASug+=SPAcRbiF+H2iYb%G/S).`"FsmJI$^`\b*&k#>1N+iU\Z9$RlVp.h[mCr6 +=Y[PCUnlnO4Oqk=^r"%G9%26Y'D12.>h..edGIP;`#kNdA??G[UBMDE@]()]VtJsD +O7iO'pg`oFY[)d`rZA$4UJl^"/HgO#s6k8bo"lsK,KrIR$R/:PU5TGR%,V]#T,4Ms +';UoXVj?;>SDgB5l)t4(K\e>>'W[9.lq-?/bi2aIAYBh_HY(9$/TOjPgb^0-gQD6B +OH4aN:\p2LqN+LGd=ZhrK=pG6?F;"2>ES*naA!n";2Lf*5Rn,57Q*TT6rqbK;B!_f +.Ou[#D'9_1?H>\gk2%KuTU+G<0"j,@+ukZ;fT\!d*N`EW?rcCV=)T)+WLl"^P!r"j +De1T(VtE"&V<4e2QE)@a.:k=dV7.p%2_3a%=&5N(6Ods#7D21%ijq\t,='_`T +h58eoa!o:IS$(3KVb*ltkl8T0j!;NSFMiE8drp[f2+fbGUfn1F=cEEn[c9G-@gn9h +:!6tH!ETaNVB.c3TM0l\kVs#6"LWFlB/khO]?ZT_rpZ>&/i00ub`;Y0Mh(KD5k7o/ +oeM?o3+V%r@f(<7(+B*e#hk38%;;DM\=to)Z%kC"ETfGc_op]WB2NaN)R&//T#>P3 +9AFr`I"7Bi#BoT!Id7/JpdohJqoOrkKCXX\*5#&#H^VD8@D]iNM2)Z",jYX%a7$^! +f`c7[2p_l6!L.5J[-%J>9981)NcRl"QYm1lh%q_LLhBJhk73/$gM,TeAG@Cm,`+*n#p1k?k1n9l +foB1Qj.ndL4sP_JSqUGia/[(/ZfN"/69&gbfWoUPBAu>)C.7j>G^@&b-k7L'60dj! +-V\ojHf\T.E'nh0^[OI +mpfJl@r*hTHiWL%Tj'loO@_YpPFi>!]Nmb?%B&/:fm::d +a"Z1pBZ7N#c*j*Xhn.2#eH&0B[;K?XCg/f7,+1BQ9H+b,C)?MX62uK^_g7Jcj,-tq +PQU-c!W)"TTi68K;hd__cdi93apI+f0tdep_g8%Sl[hFGJmK\3"u\SpSiCH:9[mi# +lI<`EA]#9;)R(//)5)/'$3E;D#2O],^E0J0W/'dc1hTcHXBj8JP[,OKOhpFSalf-G +ka*G2O=M!9,9*'t)ff:/4C'T7Y,(MLREu$2fHQ`n1ASQ,L,hW8jgc.X+\'Jc2Hika +nu`HpY*R4Zk5'YK`dX4gb^fb:58t&]>I5W23>TCm#f$a0#V=h$1YtP`hG*%t/eGbN +.A6\AHKg"56'):R@>8Ip+(fIVglT&YKHZF!D$1Qk)!-L@kCoCV"lr1@k!p9O72Y)U +)?P0rqY.mLZU;)U"#\L/<71l51'3^sK2d^P`2hm8[j6e=B(F:1ENb_5m8U;PWb8k[ +AprO]qHbi0a+J3#O0F$iNr*u,pY?P,[hET89]EgTi:u[Va&cbU?]K/T2T5T75Q4@n +mR8YLFWJXs3O3"DC_bsnH$Dq6k=Qd#Ld/S/F/,*-ec-90`F@e_,2piE!d3b%;m.5?^$IsMs<='!nDT/jF%78Z +^dHGeA+/Lm+4;Ck'lX2\FpM1RY(G4H#Eu>Yf!,fk&:A0].R>K'=mY2B9ms*(\ZILfIjB^F<3t+]:YAl +0?S[N..N?3NF/i259Oop9]Z6L<:-$mR_Ls@KEj[\&Gb@Dr/30cH8[MlN]\fSs21#r +5I^:2C,qWDOe#sAUItGU(%.ffrN[L=Aq9cR?EW1G.XaLV]3WL!diip4UJdtG'Ss#[ +#RH^$M,Y[Cl4C@87'D6f\-XnRiiiUV#=e5iP?e+9gnD'%;6p=D"TGoViT(1$ZmCin +-'[S$Ul&5a9%RT(eeG.*8p)u@7Y%RF +^T(O5h,u!+PZSj?i`?\df@:.SF/GPjZ1$= +3O\$,%R>[*!HDT'HOA7%]ms7AS)32@QI!L(cT&E72OU3"1dH'EUS@CBmCLYe`7>r> +k,&e;n,h&c-kfWUS!T*QQ;&9a:jm5KeqQMaDHaf&6GHpEVIINLQ=*`O@8J[g:e/lN +6B?uTW1Hm?T/URi0PXoPD1e@.eZ8j`+"Vr8=YlgU#uq2.4a\/M!.$OdFqQ+-PX#bY +0rU_lYFa+CQo(u2f.Lr>b:/cLa;"8Bi-%iQF\`Y@8eL[K<<&Am5g^a*0E]]%s2tK3LR9_038 +\4l;%/aRd:DS+4\g;!#MfPr-dno$R@cAsb=T/lW11qE!1"7.u\2P"Ijgh&9VJ6Nu$ +UYlWTjH.\T"F5dgn4%6X(=W(BQRV8F[,J]sm(:M*F=/->&Di;Zf@#NM10=ORs4@VJ +Z.B%X,J_LGi,ria!C2F*rKf-gokij++!5NMjB`ZpTF+)(E8WHA4E[Mu +P1@/U-)s6(!RUj`M(2n+U3u9Ml*gA>*Qc`bGL/cbVabqJG3Q4/nt%mRXiZ7!MME4N +Vc"fck'?o;3ID,!StgP.*]BJt2VT3PeMMFuIo]G+Pr_$h.N;p19`7D`18BYD]81[7P\[amoqVqE=G^:j+Wo$LJ$giu,=D-j +1UCF@NBmN9FWt0+V#PHZs)ORg\-mRVi;VJf`jaGZB2CJ0#^oL4qJk]DVaUO1jote! +-i`d*KqAkj5+UY\IoP>J]6\CMKm(Y$m6]mDOqi5h.JVkpjCUT]>E&"M#.]cW^fgZ@H,40WkhXeDW?dT?d/QOse0WL!L^U[i!4Y?08g4V>bR:6KErr=/ +iUfRdHOHQN`*8YY6>saW'?Vmn.:ou#n_f:r6Q;VlDiMOQA^_+F5.]].7/_6H +5M^$]'Ad$o6O/aHNV4'[6N^#dPA&;F#W+%J$l\%ShMsQO$O![nd%iis"7:`-p["h4sdOP=>E_KY14%".SXt)a1O&oc +ot;n6?-VHd/JPlL<.%A>IE^*LTNDrE*!f@O[R3gnNcXBm-F0'1;-^G9'3J6F>3o2:@)@1I@A5,1Oc0K^/$_lFA8t +>q!3`cF[A?J(2\ce'7SuM#GeBcJQk:IQPmIq^%gU=1J:9cLY*mcem;6s%_^.`a?j# +IjO[gk`NT;W'Peo`W#*Umeo?)o+([e1]F21rMUXL?-^,^`k>Nt`@0]A((6=`DGTFg +o\Z"6b7o;b/tJ1KT+\;m'V_.Q*cnEJ(8-g.XI/:NPjY`HlDpB=VZ:j3ZBohie=!0m9hJEM)'.,614T7$.@q[+6@m:>nqNrC:L@ +3mSdmbWkYASbfJhO-obC73EK7!88RpnA[s^9:/+W"2Q,Zd!k2jF;D6,cCA6p'"e:O +IiX4Q).8mD&M"MIK3-Gtj,heaVCqpg$`b;pSLYo32/ZJ=CLAGA@Ml<.G$`.(57cX: +$\X''.j*3-a25CMijIslQ2C-?o/tM(V[ddgG#KbHPNR*FVj8"[BWpsq,GX4QHMtGo +SNA<<^oh/4SjLVl>iWP7_gtM[rggWQ4*CGK[1M)fjfW.USlE\;7*3/@ +l'Y,bY!MWEDB)"gQ7*BM,TaZL*PHDBR)&#Pb8j.7%A)Pe*9$MV([u1u[k#J!`8a7V +n[C=e]W3<,UVQ_;NVQ/Cd\H\Sml0K\4(*)-WOS),J(>n:%GC6J>Ac&*\"5$k=f_Ru +7J40Fs.GP:Z46QE3<+n&g^i`!J([frf9,'"2ar3ujno"Fa<1,PrdLmHhCQug!nZma +UqbfO\K8]&+n315iV:Zdd"aF4f2*7p\@u/^pgIJ'K5pVZ-[:-p;Lg:DqiRfD/C-A% ++',U.CZk_CbOjg,n^AES76TgB.C*0.4"leHdqVX-TUqm",8ctQDF\>;U1Bt:N?V6E(;1ji4,tPTG(Vm%UO00-U+h>^ +&6Q`!pWScg/9'N?f^jha?ujR4@iAk7B*RpVCaBm^"/,[)QQZ>I%'MB"pIPSCK7pf9 +=MV+5\p?XA)9sI'?Vp+98cHM_A_3Yj)eeo=@Iq,t$W4YV;Y2Eond3"("j>%[UZ*`_ +Ee8!W\p'6Rj.(#IU4*ctTFj:q;R9h3[<,*='Vp1aY7;L#_")los(A^=Ap%jm14a+h +%NNg\2GC/QTu5,Ye`0gJ;gL6'.+:d]K0)J1*[ktYRG-AbdTCbrc1J]$N5\jYD94`H`di8a +EF]7mAqoM*cOiP;5;c8oWW/u`0g\T+\,*Fa#_`>E9&ABMp=!'t9b6ZE&c?:**5A') +qI,6La*%BrdJ3aP+o(Z+^cr^l5B$,jCPs:gba88-6PoMZQ1(UccJQlfr,s"YknsEE +!6;E@r+-95Bdp6C!_Oe5IU!PJ]3jJe[5I(#jh0&qVuN"*C;d[$,nHd[c@J*PH*5+4*.H +62u.7;*b.)@gC9!])?gm-F"O37/P&Hn\BSPciZ+Sl+h'PifOb-0^75."WD_/F-m,= +%04$@Es/uSJEC?lnt1.Z!:TP%)U)M1,*qVkO*MA3-ic=gpKtN@7/*7^$DsDI)FJ%= +!%@kI)"f3oh4*CJK`!g"Fr%B0L!K:c5Ds+.`,'r3]^$dD5iJmkWGq0'im,`#D0Q%5 +^5Ulq*PW%:?c6kkDZMpbI3B4D]LM)^@#Mo"(UrG(aAkWCPOg +(H-q&Ml@AROV>VUaT#$E#*6-;G#d;E6f-AIO4?:O*%__0c'gTp33?JTIAG* +Y=+M'MP(sKL/d-GB$#d"k.4eQ>s;b(,'dHTSM#,Y.[RIgRB\;jGFh$JL+d.<+CmaQ +^K\L[RR8_LOg%2@<=fWjEMPa"H8/L>u'OZ+k.4IC%)i4U(TC*SrI$pqNXs +hMeaf:?);4`:J#4jg-8lhl_le36V!(,K6ie7%X#q+4"qVZS(m2"1^)Iq*(5*"Y]Yq +1@)Emgb66XYIfM7W];,d,JRG\rd!,VcSW?$J(AEZ2d)NO/J?Z?Qd&p(L^)rJ:1Z9>Dr[n/,Bn(gHb< +Hh9-rg_<7@Jc"jDDpg1db-*)8#p_ml^Edh1"$7WSiRY\f&L[[#Q6c#;-1dJ!MO()s +Yus?XMUb[k3gs7$Z@(2sC^!D(H:>)N=fH"-Xb!S(]Va'^OeUU=@G&OQl%'6bnpDc) +!6qa=XOjbTdt@aU^k-kQ0X"Lbbc0. +eP3G?qgq)*l&J-cDW?L9KSP"t&A](S]r(NdTaI/r\c;crJj&#%I6Z`*s6QmA3Y8O[ +4UL3Me,gcl)pCU$*RAGu,JXB;/aUFlq-qNuTt[0Z%'A6Vdl>b8i0fMKeh"Fh`q<77 +%jpt-VbP%8!W@egFABGV7\b-!UMgO?tQ:?+;E+uRUA=/sV^O^R$\JF9(hI_Yt>(BuQ +?]D&bk`0Top6n\R7h:h[FO`f=s7+TcHQ98n*iAt].6qf'!*ojAXHsTPl9Ak]R/YtN +1''-4NACWpJFl+%ZiC1N4jX`iLt!R'e6pOc+SFe\bXg9Q(r&n5P&I$fLer,=1f(,-;mJSe\ft9(0BQ#U +KPX,RC3m75Eg1r\KLB:6aPB\[jDaL^#d*l@c.Gj8N0mH7W%cKC!0*?g4g9HQ7uL^J +];tAe43kc3+6#Z&Kd3e3/Z3U"h(Vq2Re^0KCtXedgJ@/tcQ9XA+1`*Z>\<`#M7ZN( +qeeg'Y6O*dTIZq^eHcmg:-l#MK*UpX6T*l-^[06ipS\ +Mq&!`3\R];W"f58r1A^\4:\%c:W"'26Re@q'G11T0jDt6[mC&!Q"?ni8(2CD)]$,N +FNF&N8S@I&!TO.UmdBtoq[h^=C'r24!R)A,7:FDD!"+:HE9=l(+k2u($L%L's0]n] +/!k-H(Tr[hr4nT`dZsF@\_ZV?/HPMDd[G,e5'^m*G%rj]Ait+65PS'e(W?>=[I$3c +/HPQUl?B(>G9g&1hn36GC;+6/Nc`Nq]KcO)ksC8L ++\fCu-<%,uQE_^&8eamd&rmmE(!snf?HSrXM27QEJf7mG1'j%[`A-@&RNmf&80P)n +aFH'%k)PrJnU=2/q49)1U._W:%*;C9-GoRs3n5n<"V5@10BZmF6<3ttN;Z`u!rk[' +,Op.T5X&C;Iba&`l]1m5dp+P\OB:J`a$fj'Md4#"^%Ginji^Z_^E+hR8!AB[MZ7]s +j=a#D^YJ&PBj^:.5P!JQmKXBn+'+%92^2=Gm#T:L2KFoRqLjA\ST0jbn5+ +#h575(2C0fQlsM>>kY6iRuc(DT[F&YS:=_lq5W"a6J8/uh.dMi7,[d/@!*i9plbiT +-JADl7tcQPrZf^toPi%PG9 +4.*mVk7$Q=J.<6;'>OYZr9"RmM$H>b?%2nm!GF[P!eUKG!ChSL,rgs;i):YOanC?D +Via^*aXstNWq4l3r=I4sE5R-lCt6"0N\3Is[V/gLBUCTX,+;NX(fnp!f$#!h5UO6R +ciRqZS0fN]A]ur>d!d#9c1\QGJN99i7HTHEX(EAOle_ZC(Y3;jeZ(!1f7S_"B\-$o +erFHD%]GWcWL!7YQ1SjVp1P-@"+$eO"Af];Bb]&FB,&EVV25!q8pDURH,b]O`eq>n +]?2n*h(Q7Y0g&:W^dh&Hhaf95<6O/u_p9pf5bMFa&Epk0VZ0a/5GnIK!N)g^q\uF.)$$Z8"t>#t +!&Tl:oof6\Oos6u!gSrNj\nJ\1SU+LUP7IkM@=(QR`uRL/`GE`A:)tEEo"S&h>le.p3j=-h%)>obPnk?] +bfn;TbfgoIS=!btAs=mjP&8m8A]r;f`/2CYVVH9^1/;QGST^n0cIJ-@TC;`n[J%2t +>'_`tWO)=ZpR\]e1?!8*p;M:4oC;[M!r9rk]O[<*"S:COj0X'Ue9LBQG_:s"PH9#( +ZoHunrIe:*Ufc*i8FuFZens9c%,]VVYM"O>gLl[bW=614M6)iA?L9)E&,uNZ[`J-iA3r0YD`F +bQQjA%/>_k+r;=j/V+Jk9tp?G=V##.@7@k>;9O1#(-hq/EuKm?h=1%92sk+rQ&]_\ +p)nABE=1_'VsHKPi-umFp[\mPakhXd;6)@>s)D>[V#fBXV]gQtgl#K`(q9,oBasc$ +rqE0'EUlWE^ELgJ_/jf%AoIR?1EU"R'`<;OhtLU8o;(8dUB11_fTUman(Neo=H;5p +6^PPQrl9c,q`8TK)GQ\NFW1nTUFUjuOFTK*7/2,/onN\2n9Is-"nI7nO9!7g"1>-L +e`(H=U0U+)Y"HsA@`=@Jd=u,h#3!RV7es&f+\O_3<\0dcOuP!,g*>ZZ-KpPKab4&R +QC%Z'RpQWB>%;Z7bAA%OAL?EAKj8FS)k2l+[a#8n+rS.ad6'%F5:fjf09L\bYXO\&S]:cfCW_,mF,0-[l\;P3/ebsUFr8j%*0;Irr +Z@2W2bQig7`e8pj7"]*J_oDKL8HR:<@d0)NWKX&j1l4o;1_N)tS=`:b5=^LE?K+'I +p=!LH;keQ$@_#FsH.1ahBG)+oV(U5.$g-LX)0/ekjfSVdlK.W;AT[%>=f#$-lA[gs +$7;He?DD7AX.$"j`R9=A^OAJXDI04VXa5,23].=OeGfo,%7@m=_74(0*&W,GfQZ@g +qeqAlGM/PrJ1Z!)Ja^N3r&Y7Yme]_OZ3S_G?2(TE2^1<_G8jsp3qVA?[ZCXmH/d@3 +5e4,EgNS5h5_onlMCOG;8(KebRh0>j7CI+JLu^i36;_G.W5)9[$ueD/_g]\VCQ8*^ +!oRJmqgnY]iS2%8=+*WTs0-,roW>V3_2_k'=>`=%^!EW_]?cW1dQIk+IYK'k\+of2O7/ONg +Q4P`_#Ff@\QtD!gB7q+6L$NA@BOsZ5kJV&(9@Tt9ODN7-*m0h<\0OBojBF(lgon]b +[rX\#KbT(7dn0PBB8(hCMYt&A;Lj788H-=Trl:&;T0(J6]/"$rEbB`>BA:jaDFj]> +/GetNi4%/:T2C8+,)SMM&\?`6J)%iO<$TMdJB2)0jI/Ee'pXL]&G[gmB<#92$4Fsfu[u3M&Meeg0&EAJFn*"PXN4'.%Ql! +bNhGXrP9:QJc0pZN%O\f6j)c[:@890r^So[?Bc]'8`Zr43Ha)GFt[7kY4(-t%iSj- +Ka]Vd-lf`q@>"HWh$oa`jYMP35/!lpptfMQWS@D.GCpa2W$Z$r"QD`.12$f$!O&>N +mHO[%S+@F;`M-8&#LUUb#[RWlI[[L$)Z`7c&$ukW)ncLiE&Vt)%Qp/hnlFnuMsNUB +1NmW],Bg!r%$/R4$LcG%i@Xa/K2$S-p(*?`*>.d*XMFl*iUA(2l%,K(4CR8qll'd@ +J.ODYlHa_56\"/Y&'TODNHW?b)Sp.YNW\s_%Je(cjM5;4f89`Hk;\A53\NX=BZ$L73PlgnMHC=psG +c1csp4rQA)0rX`C?LG5#9?V&pHZ5Hpr5 +>EekmPOPB-o>>"AZU9@0DeYb*Z2JKF%?6".'`)c@i(KC2(G*rU[Ud*YOTb[`D-ult +Wo,ZWcasiWYF*`[cNo$2,k[[\cMW0Z=$+$.H?MsMlq$S2,Nn3#Cp/Zd#H$(Ul^FHn +eUCVBIF`$fGF@JWYOk?6WkAmocDdq]<"ZOP*2Z"dUQ*\l8A51aZn +V_0#PWdN0HC]mTdJh.mu$N:2Q-\UO""li/OnRTH]MJ+J.,^i!=N@^;d(.s@)>VLjW +OGm5V%'jgf75P68,&^BE**)Rb\6/1#630fcJ1>nk>jB1ZbbZbBX&Oe)ELDaFnc`RV +.hr/`<(^kXEWAP=GY4a$TiXERUSbQ"`=QaMT)0R%LQH^NQ5lN%hRUNo# +*j+`V&^W&qf"B_>G,\cR\5i-8LZ,WVe'$jaBbqIMdmZDT7(fhDI*Tn>kK!jnRA:<\ +<:n4rEW@npd[;$'G;^7!J^;J4:=U-7B$'2*W0pEB`1E +0,60NVZ<#aA'>&D +OqGtcWT>H>EYGgd'D3Ima@Oh+MUB0MCf_8e58dM:(.-mORbA1#]Q.a7J-s%=48tHb ++&qiggFN?Bn%2Fn)5PU&W@o:$5s+V3htEV[gV.&!3<1YHaWK3>q+_#7ElO+5@0;o! +SgHR[=bE(`pLf5G:&Bja]tjHBa>7]O,QKeAW#1fZ!%l5CiKaGAG;s%,;:6quqJ"aE +^%eH$$-TOSRGX!8iTj)-4M1ro!"Zl3\=G)J%(ljjT!A6;eh3,BfTc]h9=P?Lqkl)5 +bR\riYk:,%j*C:ne&^'0+fkobrkIJVo?`tUcXZ[WoAiV9b"XtgOU^QrC(.pfFPh=^ +=23qFYj]Z+mm+)B5c=3M\*H;["`UEjEidr?83g1JJ4$`aIF[&@>0RunAc(aML[+QE +b16:#n?ts:O`ot1iVJ<,oehT,.qmpN/;;u"K[4RihnPMqBop5]PS_S9D!/.h&@Z>< +1p.J?bnO&ritaQ2^b#8bL1%]=L17o/Kmkdeo3tPbW(:V_9l;>FYb*;NS;7XM?rM't +;*"=NOoGNHi:DE3:FDt<\ZYe4:"c0%-%;ALs(#*q1C-i9,_%s"r4&O!T0&p"h][A& +-h?)jh"rqrYLfS!3DfO)+oKeIiL:RYO/Kts^7fOibN/IQp0n/OG7=A'V>8r)d]Ya\ +s4ih;IU;GC'm?pRpU#MGT52GZGC=pHFM6S-n^4<*f6"pVB9b\,Mq9BKkOE%]J*cqS +kG(r&_d'0!AKh5*KS"drmp5#pm>HSS5<,!*Lk/.N(9!'1nCETuk5I +9?0uMA&j:uaW`GQU@D3fj@SkUMJ1Fk8E,k%lE!Z+>qV0bSNS3]_b`D?<4OudI"@u= +BjRol8[pgZ?9hH"4KX!f5q$8Y(#bP3X_-it0siD1o6N8e7Lb91olKse9S6NaS7+"u +W")k>IeBdKeZ"E<,efHP&2!Lc[f\BVpaOn-8b2@6]S=qYD=b!pY;h"9)R2:]]>M +PVisanGfNS0IY,EgAp!tD$Y&1Sp0HfAjNAYVZYrY3QlT3'l\CJ_Zj'?7JjUT`3a5! +!&#N"rl9Te9*".m_b@[#i.'f(iS6N@[Z$T]BcB"1"UQ$q.3$bcJLk3IW+&T'Y7M==c*BD58kDTKH%Fg< +VRgbBk&E#N&Pe/[UKhp#fi*ONq2_FVR[5JmJ!uGglP+kT"U&A-+(Cf`o)D8]Xo]^H +=!GlsU+EH:>N-9W'5X*XJrg^`JnX$`[?@B])^Y:u,l-^Oq.V2[BI!h^VhD"EKdgY( +WSE:FX[s(>:DFnkiZHM,VI/If0>en93,Q^-oO@9LErV-AoApD\=HDN[aiP&==+IV,loAu^/PZ:8/pc@n[R$V?$L +"L[SAc--mXlnglH%PEDPr1IOB:[E&V[n +IcAehKZ,m-J=BW8=U&^#G1I6#"5n;-rZMG1)*oIP&k2el8Xn&c2kS*,!1q025E8uo +(qOrVGi>eIrRcPpaE@'Nq_WmLq3Lt=0L&>3LR+^6ce!!o#[fE8dq%s6n_fNqcb=Q- +i)mI(?`W]^'2V8GcUHs,`So86t?i]]/U+k'W26qqlgY&fYa6aD6D +*JTu%@"C("`&NgAlHiV^LaV#UP^)V&GKse.XPFAU%HgiQ`?mpV_eZRHpSbhHQ%%/8 +h/C;>oEr8WMZ0l`o'<$g+7"f6mV.-d&:T[t4XX`HX@f$H[f65sd5Tg;T.Gf2=q^jQ +>N(KOEKGh!k,#h[QiCl&qWt*L5I\$a*96pb0(h5!kj6geh^:JR:LUFD4C?=Th5TSi +0#es@#l]7gk0n=n]b2c+0'D1&WE0b1^Fn8)B]m,SM,!e=hR]HVUe9p161*!FXnL6: +0(NJjHrH2^[NFRO)A.'9$@*6f]gi3QLA8FY\\&,l +r;*u%qDeEL3W02nK84\L +D*,3HHIf41!3bj@9cJZ`(JGbKF/BBtbEc/?mL@4N/a%\%kM]_@d="e'OT2o8n=&q[ +U]\-]5MP]-[A>ZM^`NEco-Nl%P5"o(TE:2hm.mBS3.(M_HU5N@uDH[*p>Rs +fjbkCpr#=f+bn%XMJ[VPE=%.sN/cQhqKJlWtedm`5lUagi@R!d(8s>FX?T/i[qQV$#ZY+:A/m@%.m-YY"bR/L]iK^aVZaq?6,8Q%r##M^ +5l8ZPL;<_D55OeU\oHT\ESamh^4ZW=e\hILjC#sV;!#Q5_3-4_RQ1;t,i4)G:?i"! +iB#"k"a8CO,iInepAR&7X%AndB6jO&'_mq>ptM7Y^U>Ca$F%FfQII#Bih-[o/-pYs +'7;Y\nD:D-K20/@-@sEVJYb,.l?1L:9B'eKZ;he-f@+lfVeTk@BOmnMdZ`3;OoVLb +8ZJ3#C@G\EMH5&':0P4]Y^Ki0rl$kqMt+@ir+@9jVkn*=,\s!4d4YrNljHLZ$Z&<( ++Q+'dG9!1LqbQYW;@`W'[t^m1/A)mb&rCuT]&ZM$8#m7H%._rMP?rWc$$Bk@ll$2PNU>qlHiI<(e4`%qu,>7u+fD_FEieJBTpjN%k +Re59ihG:1ET/+?VP-FZuS6rHMs*CdBe[iuJH-u(=P[5qt^J2PY?'G/">N>$ns)VT=4]Cm\)I'u!Z]m-!l%_E8;?(s^ +rbfF3Aq@gS6Y9-+-!.Ck2pR7DDq=i(dc+K2kuU:=>8N5O$"4!Bih\d.Q+W9f>geV! +Ai"&QJsge4$=IHdR!C&keVRFQ@u=#b7t4Ej=G +ImiGsR/Ed8`@F@P/#*FTj/U_/Zu464.VQDY%)Z9,H!pVgqn=;9d9GJf\%K`k]nh%' +6T_FQ>H[pT%HlS:qi5sfp'$oqQ>Pf:9c)9iMe1MP$@.1-ESYIKpJTab+2TDBJ-8+; +^CH.Z3=9"#nNYAan3:Faa$J4EU$R^\J-p52MP'BSqN.$!L02a)q1$@$+"m%$LsJ5= +EQ!WGVS\&M$>,SG?!Ah/9 +<9m#:=j;cM%H.7pPG18>_B/)r+l-%l%YeY-=J'TtYYEB8_#^=ONRAAJLdGl,nYM^YrqEq>f7Tlg(gd0&o=Te4)Dc/],j +i0P1PJ#nPp$LmL92Dui[B12rFiMS]0YG_/&P5$1f5AB68XlWd$_8j(21KZ'VEF,q# +;rLo;`d%jM;tSL9[$-\1;j%E#8S[LH.N-?#R469]W@F=8C7kq`l$B!(3=u*9kZp_E +LI)km\euAhg%!eJ,OKe%;]$ZXMs_UI<3EeF=$2;#Pq!:mlB`&J'o +L\8HD[dQ9el_4=eNZG;0]Un^CJ;2\L,Hjl2M-XI'Yn<#5P[T!)qd.if[]BCe*/p?lT3Vc3Re-;BK=FW3m^8D +^p.M;Ym+qGhH$Z4<2eGK\Zqi5Z7L3BXbgWn13gltb/? +C/f*`MRIW1T%I,I(gNJl:S!VDR$ET&9)F^8]gr@ANN1Q$dP>\WAqFHGgmF'LB +rg0tNc\O\*:('9-_X@)):PUse-1cPk/G%Vnj@KGsWs8HLh>$>srn1TEhj08b1.W-BXhp&[\F4@p``,aC;@ +F-.l55:i*^+I3,n(X;@_alsVblP2qROiZi(cg;FkDAUCrFGf/g.pO)p/"Y*;T@0d3 +25sfaC0>Rj?Zkd?GOOA]*o-E3DKu`p+=>,V]]5pdMbZE[@F()m)VkIR]';3C-?S/8 +JX@F:GWd`Hp&_;Dp +RK9b[0rIL@GPd&;oC[p,BM47c43L_r;rIZmB/]Rtl/DQ$;?/WhSGuO$%FTXs5lsju +JBrgXWrOaQs1a1/Mb?nq$]+O5!lXmQ`qgN?Hg*tEl:/:!"2`m$_H\t9h))u3eK_`V +,.pl!]G7/gQ-t'aErlIsLBnTLo2itOGGK2okNil"aeUS]bo^U3Pk=K +2#fM@!^F!c5Q^qq0)ka&`:*&VTALE^gW5QroA<)_7'dSa(d +;cG@pCL'Ul3gX<=Rr"1QA8j!Q]KS<6&mDs/f>b8rlUO-'?5$Q2UH>%m[bEV*04IG[ +$kTo*$D,00m%,aBbbHY9G&8T\r)(fLO??;2!/N.%cPL,n_:(oJ1Vi(MX>G#'I:FZ- +2tK.?'E1QKZC=#bA)*#NipP9>A!\5`XlEp45QsoWR!74`rOY0[M")U( +nPRpO1,TMOU'k=8qH]`HqnA$$?M*"?ZoakcGFb_Bm)KX=M>NTTR#g1g5KDp)A(-Q- +j+/GMhIFI[gVsLX%D.2k%s=>%FkU`uFPP=0":5&[37+a6;_D\(?9-H;gt0(k06:Pr +nbS.@[<8>*GJ1TFo,?E6\?Vc+]V]^d7J/E7.?>[:1*N,^Pt`88NgpXfS.b-\m\8MbYW-;.I.L +cZ:ec$&g0<'8bB?7^3I/V>#nPHaVLaDX8Slf9]EF^TFH+h)>'&+lVdJmQYmEIU8@* +44X:u-1^;Zd#[L-5Aq/57Ml&??Leh0HD;C*9#Q24;uYgqi9+CJcPL0->F.K7P5K,_ +lqnGRcO3?Xa-+4+8t>:Zh6!u]Fi7rri]H7"/HOMHT0!*Sb8?*):pPc@amn_n9[09?,e:Z.U*fP<-"JS;>c/4ChU,*E5!&5&gWU]SWSa^h)EWbBDJ6k33ZRj*tR +/t,KMbGE!6Q;VLTF7tV3Hrg&[c4I8C_Prn/T2goJRRF;_ZTE+5p51er;\V2 +0Q7ZAof"i;V#eph0Dfu95Q'F1,suJZ/!0RqlMrEUFR3;=mp'B+>?UbRd5h:s% +@uqg`ctmt9DL7=Y\rWR1?du;Og%$0dW7 +m)H!ONFA@sI>!c$oqHp6)aqkQU3.KuY*j+X7^?b53>f0'*Hu4boP5raO,L9KNj6!L +F#&Ysd#W4srkJ9doIp2&`;0H5OThuX!abKu?e<,1rjTN[D&)GT#`/'&MtGGWnC.bc +f*ZZL$usYr"M53F+g1)RY,WX_8g/'ZU$deXT#.HjU,i2s(B]AK6?qg*Ge2=JM>:D` +X.0Q/fQKE$d_Ya`O6aajf-b+Cc_go\Aao]/+S_"/Z%&p4,:ncO1,&49XPd0Dr%H+c +hX9F9f>bR0,JYcBU_O6"cLY+EO=JA<]oo+[Q`-1?W0=&7]#`k0OY\4WJQT`iFRFV1\0k2o'G] +0)A=\HB)ihO5nL"GV@^#rsV["Geef^mVle^0>L:$hXkmJGIX:ICr^.@Rj"F11pi"^ +ir"9C6A'e1pN&NX=h5E]N!Asp/@#MsTV26?E>;2a:L&78[Z.UP5*$-'\$Pc/pq?&V +3Nu"Pl/2H_aUMR+OLTm[GhGqWScu>gR%`c[n:QhFKJ29N$'<=1o,e4cmtAoWK[-9b +QJi3r%I0-*eEr0sWdSMG!n^Na#^EBd>+I"#lq.(4`$:c!A(tuc$c#PhDH`"S1`JV5 ++1br:HtkZ`a*cg*C#9SniD\3sA)6h\NKs[+[U$<':$%gD`A4+\3LL6cm99N*1k&h/ +^8t5:X\lZ(\8c&(I:&mo8`:"T3e(0SD,r]/=upetE,J8E&lg-4`"u22odB;j[d`OJ +n>dpFinbZB,DMUrJclR'Jrkd4L34id^INB0WASSb(;-Z*'kh(DXr'0RD^&5&^gC]f +89qg:Dl$!]H(O?/;;sC;qM.G4J#j7-p,$*9EUDSt4rED[@o)b^L[tI/;#&@#iTu-C +fd#=Z8,?RD83@pTH;OCSo9h5:KAI>%mcoZShIWfILG1RQPrm.1c0.bY_ +]D4X]i;.;KlgH>JG8nK@j1&^,6l$.5+*MA"'oBSlZp2/;nNRtqYG_YdCXeq6M13r% +r3A]O0T/aYM-]2XY&SJ?#cUm8"uVcZ,A,V^))kuMXW5>$g_Ku3Xok'G+G]h*VoLKX +pFqpegC#WYPUC^72[',/!98H(AH$7cK.nEVJfd")P5#^1"VRdYl'#T0J1:$$8b'GB +M0$!YQ^CB6!/A?J=;M5X/<.e@c)LqA<"@%n<6T&Uc=p;-@F_>clIO8Ff?LY(*p/G# +)>\Ia_a_4'h2G[bP[N!%@o.*FEnn&(pS& +iR\3o$fLcuO;@n2"5&mn;o_1&8*C/.$J=kg"1AI*BA%I7X?%.K3u4s1_Z(j +mlduD3O%b.O")"'E#V.S/UlpCGX%S.k>#.#UKtID2"9J,Sr6KPPAiZShVKp4"[U1k[4,,'l41Xm"1_C=g#MBi$cs=[$:p4_EA4[V?ekDAoj`,Io\[K%b.]Ur6Kla&lK_ +3LIcF)FQ[O!;X[FkhuB2i+hi?r1E8LLK)(00E^bLIbbTJQM]8c_#JfpJ0Z8m;a9&9 +0?cdBe6OH?9fq!-/'(.8crgJ%!%3B5UUn48C4NXJ^QYD,Pr?g`6;_W<&[\HJInG\nT$N-NnF +344\^NJn^Fb\3QK=AT#bG+s;+P%Ib#/unh@;9qs:cV>eI->?bC2n9O\eJ=`KNNB#S +[4tt7,TrE=!7PeHmQO>=-DQ_XpX+,%nqPK/MT=A<#C4L*'"Rsc;0T]$pV!K]e8l[V +;%Gc9fN5tSTEV60,G!l1l'juGgEEo5W=D6Sj)--+?!YQu&A._kp-30Ror&Zb5Mo1; +6)HfERt,iWCTr;Q[e2V.)e^LN1u%ELf:96+H)?IEuou`,>LkHL-YLn^InF@,R>B +cV-5Di`XVU@*UUua[DH/124oVd@@`\*=]R.[<%M.5 +8nWt@Yb\HRb?nFSc/7$lT9W-I.qR/idnjSu*ANHI%Ulf*SGLjJZr/Vi5>9]E6W;gK +5QZV8j=mZ"a5@]_k6n5 +hj+[7VhkAa/:4h'qYQT$8#(Gbab$-Znc$>`s+\hlgPY?9LPf`*[Ee2U^-gk*1s#4G +Zj6g75AVa?SXTRg:\teds3Ed4O=C+3U@eDir(h+6^TFR7\SLp7U]S\X)4V9$:t%*j +.^iD-EnV]+F?nhd]tc`.C4C-3I7;]dL2-AFJE)PGfl&7f^E'Ud%b7TkP3S,`)3VIH +k;VeeFme5\ke`k%3UD)HqejWO(NUV8g]fIaIa"W1F6c>CcX@^sK/lBNLt>0Y6qu$]#8e&p/]->;N0aZXD4cB[OmX +X(Rf#:E,Qt%<"p*r3'( +N1u?=T^gi'oZ?G1,]A$m[QMPP7p#pPE?<'aO`2HD$!:duXFUpVRl3qYf>5V:DL(Uo +D2lhpcntj9ds7?uH,o(IDOg`8KbRq9Uj>;@E_rr[e?F[@>hsWqN2:nP,iRXC'6=d +n^@GEr;WWC(^GL%J:o&hoSitV\nWJ"Ba)"]qR^G93>6e2e9r=er1#Tk*7V[g<<,4+ +8j;qU55'"\p9K5Z@c:&p0eVJ-Y!Uh0I$]8k.mgcc@lXso^ +9@V*YdEP;7a[t'pSmpNHQ:+M_TR*t?=L?GVg?7h7B-"SH4GO^S->'oI.mE#_cS=5Q +EsX`GiueB'/n8*,'DTi@Si_5TSUBK^EU5tOp)kIQ:EBa7!DYsnmtB,-"juam1%%QW +_g3\hE>jCcq&`#2h>gUXjKXfTM=W_GpmA^S9>H1E%H)`g6QuLiM-IjCF5B=U3<.ln +S;3=0,QK`I(B?MH@9BX^]L^kq +SY[Dr/\n"S5ra5&ETGQsW9%Y(91T-^Afh12:OUWth\4>X;g.?U4V-6KYM"]fF+D;; +F&+p]*k9MNGFlH[5B1-iEa[8)U!^A9W8H0@>$%c[,h)nYmq]?3s.`Sloh`DK8%359 +]jO.f_MZ@p3CTU6Cmnn[]\K>@\I7_Df@K%V(F8R[ealQgMUs]U8KuPlK&IAp;'[]O +K_^XL]Sk#bBmY;:0a/-&!s/lS-k!VGE[L'Ac(f+j6&41>?.1h"soiO +.s=?jiNei[OQ'hQX>`Rg/6cf%OQV&trtj->?VH?+72,T=629VSa`$;eIp#YIAQf#E +3J"*afI(1]Wth>3[Y-i3_nK<@qp.FK?ha4WR1oIH?0Ts(mD!S_DEbb5=H4GgZP,/6uMK,t#Yu;iTZXOMGr\+st(mR%' +N7[5%Ltl=S**u!K#oMdn"VcYAA]:`6sE];rdX("hh=CO!lA>qgNCmnn@3g@r"C)9dlWp+]O9Lj1qNfW;u +CG&^:?:>1]\`/3e_)=E3_"mb20W^?P!&hM<[XG,:YHQe2-jpo(_\Y)#tLs22*g&VcbDgRF[\<)7@+kfH+eEj`-Rm*V6ZiW#]aTFcm^+V6^ +#71C'Xu,jK#4VZ*I4'KOZ0]tuhZ_4@f?Ti71"[PpkRu[b$f)`RG9ZU%7\_40_#-0* +p8q8Rh'B=m@0ll*X+lTa##%[9bR;!60(kT=)7D(;L$ZUR@`Yosd(Ei/WJ^'"=;@Hn +D!)^LA"DOg&a+!&-N9'KdJIk]:\HUFr_N(88bFufEs2-t'F73HO@^Lr!hgsN"98L. +$%C07^.t[nq`t,\+FfpT4meKiD%m!l[L/ULBQ^eHpU6:[ZBasfQO\,##^$lBp>F7& +jM)>P?]iReT#3@)Ocrh%/`4A;C9Wmklqok-4+3V@%eKU8K)`/K^/n6\T2PGJ*om-* +M"a=Kn)H=_P6SalC-cHg&?.q%n%;Um[`$])ST@A;=`MSe$#5k7ROPN,8^V9hW2@S5 +V8]06+?`I'P(Kc'oupskX>lFAFH-2tq8-goNi;<4LlMYBG+)FqKM`oC\9uGpSnZX[ +,!PHV39NFko,@h[B4(Ur6+XPb1:#st[-TofmQ13us3lt7krQQGK!=9-hK,7?o9nISo#.;#>c<&nAF;bEihMm[HP&r!eDnH6`CI1+D!Qfj]SQX> +V[9DK@kV.\K^))A?4SoBRL%?W]GlDY$RFY5_Womt)mDMc?[2o!aF4^b**5)o'(8bJ +&@V'RN.nl/5Ob)Oj%?jnNs:%RQk/O#G&6HNChYcql*_VsG8tA(3FB+(:SVVk\Rnj8 +E`K@aYUj.uOssY#]uH3e=HG1<<%J1C?]`%8,^4Z&Y)"B2b*4F/d4744/nN2d;#/Qe +H.-JZ.:f$3@87?-J6+t=>75&lLfl'GW/'4S28t>f#+na%+Q399&qe![B86>&/EMAaao8Om +IWf)Yk7+9."$GJF1k4/>rcV#sP<&!e\)bgUlC:!ord8&EK.3s)$\ElbeNJ9AnE[OA +X[_ZZ!Pr[kdN=UU:PT)1#N6NY387:p4#6S_.eNmrn,ltjF0apu6V;9>M*CCdHea:& +fJ!/BT@@)iUA(.1Dc^nm>qc5`c=,9iZ^t6@<76OI'#7J*h]UuuW*NVh]=+MWI?ebo +=ZrCRRQ2mb6P3+b'MkX4D4ejoj!T,9C&H(Og^*rfB+,(SLF``;J"/`4Lb!_'1RuN$ +1!YP8`ULEGj2.c@TbOe'#.asLaTVU@$>7e`c7R+DA4iJdQS2gh'[gP"R7\b5WYP-j +Pm\M<9=g*57]i`2D6STp!+<-Ng4moZfJN&!0kt#OCg8i7>rHS,0Pk'*4%1\;[kJ71 +([*V6m0o=^\l7@bA*]Xp#La:8@4jT_UA889aE6Fg +CCH28oc-3gSq&e;a-Cc$T7.Esh"@1FHgOSrZ2;fhItnSQhh?1=JIL1nY<-^;e'P+3 +hn![nQG^M@oC0Kk(`HiF5)bA`SGF%lqNb@k!Q_q0G@JUJ%%D+KM06#,-.ehj&)r=& +$!*.?'c18:_%ilaA]K?*gE*#S)=nWPs$Jq5`H]5%IDm'\!j[>_89ekIP0.# +bHCH_#rTm4L"/"=bUbuq![iI>7hq\FJ@C.\cMS`l9#1@Bed?67iGc2fFgmM#!;YH>$ZX^BAD(r.!.0fX25cQoVpR)/&hlP%oD#a-C#+&NeWC+M +YjXX_.^8;/h=[2io:sniF>tTpd"`O#4Pe7 +U63.DK:%?_EJoXg'3drn3#bVOn2B.d.h"ae!MF$cf8&4,d+nGYk]hr"j$^P +!!;jFS[\J&JLJP%,mo^+W1$q3Rn(cB*hGC@\QE+,L.[^sb +%dTQ]`Kj]P\h^3LfnRY/j2($* +Z0bf,\!rY4?F'KGgLIQ^B.QbmYE&A=V"nbf^f35#;iX"Wj7mjmr7-JMQ]pDnbXoMg +kPk@1in:U,P90'hRc)jN7/1E(oR:%FrbYm"qdoS1[g4XbKKkjbs.:P/Z\J`t%upR$ +:\+aPqkmHEZb*)!/kE`j'7R]WmrPeMha[E+*0f53_-U!h?VN%hQ=39@4GW#BX?;H0 +Kfs.5IB-B-E?jqs&s^)*b+EiZIbWlV1u3Y5?q$@RAL]#Z.q`_!'l9q;C_L^QlJC1D +00V'B\SY?SV-CC%?*^>1`)OCjOO'gjhB\@H-'gG>OYJAP4tW9[/W^,$N+a-2'Z%^b +mV!RbgbH]Od6ugo]LS>5Q3@G!Kki4c[QBRC.7<2$Z3JaW +=k%-c$"3!@4!:DD*E0R^;7@%Pr`)>4Jd105:!6:Z;@H(rbID9*]PYcl'u(u\2e`mnQ]O_NXYim`el52A]\jZTBS +lG33c!E32DkV*6d)a\baTc_G:G?i,th'fms$[@jG]HF1$D**dYq/8>)mR-&RgUlTu +`nB:H6u-V.+NQWphUgS?&0HK,*!At'pesEZrd.HaIP18Is6$IFk&%^1q!`TQZ"P@# +1!)W/_N27iS\#^Km1cT*Wcd?dIFF-t44Cd!C"kdbdkYT;%fAP%R'*X"Jt0-QOGnA$_N,Q""R-K(GcR]@W!)m_g1@+?\n?J!/Ht +7dP:UG&ROf0B$$C`_3id3,hX".9!<1;)Bq4*QUH&8f?8&BKn[I2(i,[+]9mU6MZ+& +5[&bcH6OLc;'nLs[5cUF,p[9Q]prV=4(D:6Zu&;4+$Ct&go9W-A"+OW]0=0nTS\<: +EV-lZG5Ks7*J8LPAJ@$65JS>]J),Y]r4hPU!KfV95@DZo%+ui+4!'W':\]'aoT8d$ +s,R6Ga=/s-"qlFg3BW'Kg^.WLNaSDG0&e8SGhLE4%6c!X5Iu=Y^[s&p=Yr6:fg,P\ +;2][g0]'9s)`?R%J-Z"6,N,s/?H2Dgf`rV^*kdcBK3Nj-mB_eCOl=VjWLL,I'n6UY +T:K5gK35dM](HhT'%VU#@=9YLja#)@Zfn6`;2\3o=f4HjgTK`I(hirjMKm=1FQuBt +gioDREi)$@U7Bqth6@g0C#$%sd0-hkVH=;[<2nVLDqnBo#OE'AG6i>\Wn!145l;-p +;7sWG;N0Df1s1$LY&CGPGb59W++0'@du6`#ue^g!3/isM(aQ6S4+(6 +lk>2skL!#Ce3bo]DbSNCMp$gVCgg*tO4s1gk\'ieKB0'm<]S,IVf]n>o(M:(9E>>*e6iF":&d +AP(gl))5YJ$]9TB6TSsJ.Wnbj_M2OYQJ2"KSSNS,7=.K_%S^\5bhMR?FJ%9/*h"r+ +>1@JZPV;?.?dQC.<'/E&3_,O_SX%W3I)aPB +%^0l6!+aaA5C)Tg7I[/Sm;L_QDqO`"3Y%nKAkmuos%E(!Mfq7a:'4]Q_lNKZ5jW8= +_s=HM)fESG0+n6tJN^aS^_W(5U_0Pmggq$0#>blB?AO$JKSi3X:&!eBYe<__U4F]2>cGYeb3^pm)@dB\5F]"-_.]&06Vf=YY8Z@'N_O_@fTi`=gq +;3pY7#iqWEC6d^d(@Qhn^(B\DUI5+>!;WOsqnGr6W97se#Gh0`V#-Y@r1>&&Rf>02 +9sVW0Zbt%.,0]YQ/C2:;HJ0h-!7$?r+/aP>r'gLg:?F;AVG[Ca1MPK'RE=G-&-G69XJQtG=U+i28=O->6Cb-Bl\W!eN\9?$!_^b"b"*V9_K09!R"*Rq% +*E?K/V&X:&Y_=T4;\s^Lc3ndWLt/:mQ,JPm`i,_&.OR_7-Ujs6q8\/f7Ye"^L9#k4 +r%A@INCOVC+"L9Q@+,^W2&]FQl-dk!ltT7#Q[G3hA3 +b6sViGk218!gNFG>o&"R5"(63:6N\>)de:APO9Jl3 +SR#4:^c0?S+-g!WhcMqW>Aqa@mR8_bO';^L^E9q:6@H;=)f1'&T-XRqj6Ok:Ut=)0 +OF)'('&GD;#Y>+pp\NWY47?eL^.@/$2$2ragaN;2"S0T)*k-72V[;DPU?V0hlgOn% +qZf+d+6=MkOL5VQJ^4pW.t/5#s&m:g_dMV4+50G5b)n4Mct3-HldTd`nXDFH-[+PE +lmEk,;bh2LDjh+WR1(,T%rCQSXJn.]W7J/Qf5D6>[pC0R[iB.l*U0pu0#51,Y2<3X +`VG!PY%1h;FA'#h;l+S)rn=Bhoju+#9+"`;d[-gWZ6pr]-7*0"V;irB&nf/3VaXN- +b!ff7,gS4+0%1O*8E*4+_If,gm;i>TZ+LG#gqrJa]\XKK?;e.8Q+PE\)tM&;B/reW)nj/V0)2@T$'nLn1b^1U(4k[nq#d&DJj/"VqXS<^W?,?;O@]`bT' +!<1Enhj`u/"i[P6nlP!bYKTOESman93moc\2l'_AQl@#8LS(_1o-0q6XbQN(u7!"_6)fB%+"qd/rB) +#Eh$EV!=1=l'9S_2DEAoB8gW(%;R?`WlF]]WCF9oAdr:gl7mo@P#:>NE>@0&P%%H$ +R'81hRMPV\>\I0]Wf^)k+VA'4&mZn*H!N8<81^W+,-F6j\R$S_:$[21iRMB.OpS1h +.D*0*qg,8:/]G;UC0aVVCmk%]?Lm>fnJb^@+RTAq:s2*(3r+?RDVokYcY[dUPmH#` +);0-NJLgd1pi[P.E?!a*+Mh[GIULWB@,7\uafK!gV@2HmfqE?%Fa2(C2%-(#"4s4U +%nI$snb[!A)=7T)-#o2u:B?=QAA2H44KagcW,;F%Dl3-6>r9r2qj3TOH +fL^(!U=qM"RQ&Z3gLk>,VX.M6XII9hXeSo+_#JiFA]&+[_o>\$es2GmXW;lB&tLTf +>V&c8XHMA`:q)dh)8uH0*_*sCa9'H338)J#`S35ga!B3e'EDQ+cp!rp;=G+?T,D+0 +'l'$TB)[PYo:mmfq(KZh?V7PF[o3::fugq\.(EC&4'Tf-S1J4lc!SB)>h[J&&r4/g +PeF''SV`T%M%%2V=N80Jdb.%*_c/JF7QA4Q38^)&=pRo0$s-L;#;M!dHBDj6!*^o3 +fXSc&;2ufa@V`s8=s<',!>jcY-RG*p3/=[f;n*uF]JNLN9!i,DUfpZ`E^H-c>aS\g +R!NkQ>#Ycj!5O(W5Y<$UJ&(k66>^NE3O*`b#*"Vsoc!pJ#>?+*bT02,*(M`$7Ec_V +;%Ge/O'F+=Ng(PSik!CMZ:3*FFl.qkn@JQ0X?3S\6a%pEq`t;Wh'`8Fo'\VZi!tSh +jehLj8H8j:!C[`g_9=k!IhFfgG41sAm6K@,`/M=%$hqLS:Eq!F?[`]ucM$rtAo=P' +.t7.4C_4(NUO*9i+"+6U8pWYrrGhD^o5rKkq`J`Ym>O3DHW^UBB-.2%"+\CBbQ/h% +p`]\'fKOR$QNg\_n/rf+q`uSuR6)MO+2'd\'>>LDJdgXIeC+RZp@9/&a+R;AYR=kN"6OjU.Dr"Udp=QIO6e1id;&.4!$V/lCqL?K4p,ZV2Mo +DbSfAM_@G(VN`4RAKs8$O%gp/4%5mQDeQO=A!).@K9O5_50))sm(59q+'2.-&'I0T +:jG>^pghD8h/UP8\F^e' +F<0"UmTR'$"8s.f0C.5peB8u%LQHrh'N=_RS:&.8E4`te!(60(?+pm,>u8-6dJg4F +l*OnS0L&#"113`5.S]0][,Q4s;A!3B'F:>)<^qj?e#ZZuY%$Wp_hYE52(n+EcKhb$ +6?R$TFfL#jWTn&Y)W_\+Z7523s$P@a5dckgq;0_RM)IBW,]]'$":,<&&35Q:P=[Ym +R2cXT-5]G%3TloRcJVZ%?(Gc.]4\^P];L#<9meb#hS-QR:VZ@&lbC`VaMOcRL>RLL +IEpMmgo,>E(fUq5+#Sb&^URJuCWrSjd]sL;It3#%Y,E:@fP=UCB3a;1:sroOfPO^n +l7jk3YUgm5WaJ!X=q.cF*471ZScNPQW5c96SH<05Mlh8RBe.u#, +/SO7dTmMjR$iJnU_qM`GkhC1Bh57PtgU?M'Zu)"1(C7W!7no&M:r"s +51[[Y]6=5c)#opFro^JH'EJ0"6&2"b/Ygd_QS-MXr17nH]GMeqsC)OogK^+lnka +S('.HN,[OQ8.*NE/akc_'`Wsd"9i0sE1UO8!5a8)@kRX](,BDV)5Q2,A[940gP_,S"]FK3^+ +dt]0)_S%ZjX\hE-&Y^$!Ns^JM1]6?uHk_,/!FBO?*pS9(Su7m`]*??s6c75K*F99K +aAjAMj7EZdZ$P,2Q$eJ_V=HDZ]FM.(b[GSihcchI&8,Kr-PoQlmo-jka +g5?:'nG,f60&"**mbjQ^qqoC8rZlq'HNbAE!lB[S^5lMhRlp;#4CALu/cZL_cf=_K +)S6Asro[d`Tr?b/I#2q=AX;$h?;.b5!!Vq)b%aCDtM8'[]Y$=k1Foh2jd/aH9cpQ:8s/Pn%1_>9J`sLNY)LgGEJ))Ajh2)!9=8;6^ +,lV@HjZaN?hcEC?f-%il&c-XRbg5#iIaN`E@9hf!(R9L&2YtH3Kg/7(rGGm! +3+mD5,CNTAIe#g93N:.\5#1KD3UU +(XM=]c=ito@0$0Ii;\I2rjZRsM!";+Xh7Nur,LQ0EEr#:s)o6h^k0PkH1P)n-C*%, +Q#d]oB.[FbRmG8*"p+ac;;n0V_1:M9IY6DVK7#A);:Dg:_kK?4_ENV3*eU1.K9@DL +Q\/n8k\C*kc_M73?rdd'+MPACZEVqC^X'QshkFf"ZFJelGQ;)+6n[pO'Xa@jf&9M" +lT1d4S5OZWBjD\O +&/2LSfJG7ChLZ-:3OGRUaS7h#kV`IVm"7/QJqld0.qAluF#eP.(r7bQZiLM`9#II\ +)pj7T*KV?GnNZMmK`5S:>!$3bHIDN[34WeR\n]=OKhmd`j8a__T6L$H%>*O#9E+ +U]Bu75]UnL/Va5r#ht;/]?DZqUPR'(fQ0gS3U+0Z1OLPK*+7?%7o?W[@;5^Pflbh`IT +5o,!jk7R#eK(H1nCL?9OYAY'bY^9^G-.]O7SaZpL5g7f7kfjY9_=#t:QeTku/5lOI +phA^gp]=L]8,477oonTV=98<.!UpD.0$k93F4dQ3=M^T3I*cd2msGr/H-a7sS-.TZ +ceNQ=cC#otpAOkHh)kD?4^uHo9b5hQ#BX]B!'2D"mLhRi/6agq>.*bW>JLLS9cH>f +AoD,6266u\Ds=u>e#ZYO/R)7."q+O6.Tf\"Jm;\`.;r?4G=q#NPYCYFIdVbeb+W[f +V'GH!'Vf*Gf6b):&A;rr]KtV\j*?P0rY&JL_igTk#Vd2o$UQ+a!YS`@i4egaE]QMEXTu.t,B[!> +IbB7A7d8rcue*4n[ch-KOX9jSsW3pVW$EpEMG#f-A&rQ2T1!k#[ND +hbQH8QOVUi:(;Z^A)'#a?W8KPAY/lqVrkH/g-9:M5:B"8C(^6%II7%q2?2!NqG4^S +r\D[nY2*VD"ITX!X/61EQ(.esNq0ZCidX%_q,J_Lg^0=)4K&h?/4*;SKZkA=8Speq +*]i6[b*mPA)ttcR>ND5Ae$;!/iYfEr'^9?m,:^7A +auR@3B`()sPrGmJ[%LJ->1>30_NcbmPX@>0RQ#s\^NL9*art6f]'$G*Un7 +`QdPU[CF_,NO\0BI]3rt8]"4uK\3P]XJ_+aZW.&6[)8pq&fYYmH<=_\:0%7Ob!A'" +[:Xqe2,:*;-GQ"UWLYEe/X%-)mG%=T)n,-lpnh%hgTGtnOi-*3D;^6($iE43JKo7X +0R?M8LL=C72_A4J?*eJ(TQR?/X\E"R$c%Z".k(="p]j4!<`j%]l89(*;eX"@H(NSD +6[am_J'@=:s3V(,[j1O"&UX.&^bhm"edh()3^J3N8uWb<]c#n\e/2cIBkB25H\pq\ +9"aX,IMF&))^/5u+^>ucYVoIj-:IKKTQjIeGc@l[!Dbh9-h5DR1<)[FbZ3?B8]fFa +fumOKdo?j]n&eXSEDgl>Wd;1SH+U-Kgf8bcd_X27Mp&X/gZ\=2@.^i(RGZ5^pX#T=HA!,eIt0bNrZN;!omFUo +`<-#pf-q!r!VlOeUS),KaWJR5b[0lV+k4O>q`oL15@:bVIIVp8NW$\&i:"LP!ko'& +s)#OaBB_:07_R8:VQ1+)&q[!*uSX2Ohf=^q*7ll&7IM0*^o;K?pKPl5Z +]RC*Xcu%p(1Nl%NhmMe'ZgHTmGA[sqiVCj7mu/uKaW*OCO.K1'N[P!VjIRqia.?#o +0m2]E2Mh#3"ckXMmQ.di3AbLQ*LJbQ(r"\OX'*oO=oe,oQh*.i?Z)g/YH!h;"Y;.a +/]g._;(\5u#u]:^EV1B$Z(I*S:rG,`,k]?iQZ[j<52gOZ\&@;adZDqP7ER/)am +LK$G[5"!hp&;:Z`g)G>$+Y:PD6lWsbpu,;\N5+V_qSP)eWue&71D\[H=OKU_fs/]FO0cdg)'g8]det4/h@E?\DftYb!rV)WB +J$*SeWXJWO=Q482`I?\9pqpm(?[&RKH!][K%nuu0Lmq.nmo'";)9TF:lG"B@qW4CF +4%L[/*e&@7YahlGi?Qn6/Fn#@Ndi+Wh+U5fil"k2+3c)VT=X+`1oL9MM!8iI>uuD% +=V%?e*pSsrilVl3RKr!J%nLgKYt<^riX;le1b$a1 +&]Z:3AuHh(;OmP^aeRK6VM:>l7![Y6/0At]h92b^Z7(6V@[QXoJl"=7?+XRpZt\.# +7HrU7l.iT3!c;&KK+9^&R;efsdkR[c>A^p6E&$thdBm-OK;6*QC)_Iu7d5RG$(tC +-n[^'Kc'2h*U?"OochHOOCEU>A,V4'!@2'%Ej/haW-q8tWmh;hH5&5"L6`lj&'B^) +Aj*E>PT<99M77,Z'^4-==T"4_!6%j]c2q:d7/juOJ3O24GXG%1,7n6i)R+.9PK%.7 +055%fLgVc/Yn2%%M0i1?_q:#[S>?j',r7\PEkKB^!VQF9Kmk474OHK6MT=JaSQ(.85^k(N'h=9>Q_WR"UuLgDioQoqj/1o' +WT90^$e(rh&1o5K<)j0si#hOc&@1j] +FS4sU%=#DVSc($Om:R"mhhH!!+RHl:SG=#+EoT.jQFBX3fV-/XqnNMRjHMZF`Yq!Q +O2:i0G\,@$H]-[mDqfJ,)8L?)5BtHj]5a@Tj3Csd7uqM6pT6LfLn+d:/P#TE,HKC$ +e$kg$Fdr4bCHAYbeLT3.X4G[Xi/0D"1>kAB!jHR@_"qeG[<+trH/0nHQ4n?('QcV7 +h.QMT3X;1d\K3_/Y'jGS?I%0jrn":bRl3IR<_rSa4c1Q0dKFZa,2,g=oH4NL$OY=8 +&;CHGT1/V:Wo*:-BYTt4Htk??8J59WJfLRZ/]@ujKd-EIP2?f1s*;H?GN5]9YE4[2 +L];//p:CplV[g\J@c5"'UDMAe#Y:XaXJ'pR;qh^N;QN96D\I;gIoSZ:'dN+bVL;oNsd9GV*mHg9+1kqRMO21!.ZI2>$E/p-A:BFNA%OGS14j4@VTB?iZ\m&i^%k976]a/3S$ +WcjMSV/$9:4pd7BO[#M%e"`#PVj,`0P231)e#2`K?TWa6[h&\X#OIN:QXJi]VG&M\ +NQMKGK`M-Me$9:+'6=3Ud5tB;`lC9Z8@T@WY.[p_>Xl`&XK]7;M +E[KNeJ7T_cBFYAjZ3YY4O*7dUi_G=$+NAbkj1T:CI-_LElk9FKM8>FP&A+_l];OU% +@")9)/L-^U&P^H-EQEMoc\ccDA]N9+'53bR&uiDJT*>/md/rBXjM)+8!OcUG2Pr6: +Jq&BGN"o+1gd'-n#;NuPOe+C)=6d-a,`VFBb]%#Z,>kIp"IJc5LFNtYPRV1S`,R(9 +>ou6$I$EV>Ole*TlM`3N9etBj.a_BKhF/pJ=Xqp(1;HK4MkfLY<-DlP3AM9h=\9ep +;jkU$@aZ"M_25;\lD3L\)r[=l)B$%XT;aI)Y,n$4acBN'--Rh7:Sp3@E,*q"EZ+ +NUi%;MVNS.)h189,p[)R+MKhshA"#"G+[Yeq#@.P^6PAC_\+$b6q1FDIV31Sp@1H2 +j)5meIQ+(srM8CZQLJ7(.XXIapV4-grj6/1+5<=#)('`G1&B7)pg$,Pq[(NjQcR5D +LYm5=r;2T]s1YN/]7/cm+/IX5R86>el-f@e'&hRGXb0[VmlP)F[&Er)"8@R*PMZli +c9;h8ejPEnT#+$fYLKcPaOaEI]BR1^f1:1ZAUY2nHPC"4)F,1Z>F,WZ\&pb@b':Q2 +@9!\u?R"W4Lf$$=nlCRr-:7/6j7GNU*0P=\IN1_P_:h-gr7W5K(co +MI$-@HJqftr5[a+GCEK"StfrRTd+a<+fkL%dBuJ[hk@Dng,*[YlW5WQ(/1d+%4Xhi +9*[<^qpb2J<,t$bZb'2B5i(]c]pblK%-\Km;?.Uh-lb0q0gg1CJZfnK/)f*-np1^c +#L)dcN3bSTZFUIN]1nplR,rH/kHs3R3fJ.7NH0?h$O-p6_q*oo`:@>'fq;>:RrJJ- +=YH[J4gda:):Y=Ig7jqaXIiLpRS^'Xm&b_?""pG'32=kHj/m<9]jYZ.;-Fd@@$(5Z +PFVZm4Jg%Q\V?39WuD'.k6Y;g#SOfl&;):m;6iqgW]kQYbQ_No1ZOfL%9>AK-tLcNnJH%WsiajeO5l#=NKC[nfQ=MOVD'Y*"O9Y]Ri=nuYjc+@n%CDu)[&&6$@maNn +@m(,+I0l'n,i)(ICSTg!YO"VT%UqU3j[_j\;:Is&&AscnG*7](PRZ`qL2^]%3RZc# +.H,@A[*IWX(4(tq%]C\n)R!1aTA\:clcI1X!5%-sh<%1m],pr7[Kg2$E% +I2_u#5'%XuB+;XPo"!NINVQEUNAdoh9##?8[DVSkhGV?c)=Lr^J +"8HBd^]=8peDW4\TaUbV9?7:X]gqjX,Kod*l`NUr!XKhDq"?lBQtit +*PL0DmA_1t=*m!3\$[]peB3@BC[*4h.aOI3db".i%P6(Q/X+c1UR*Qa2f*m!q*fZG +&pE^*J'8DW[L:t:7iYIfH&oiQ1cBPlGf>qo:RR5Ha;/2(O(1 +Nk94/IG^dgh4WWNq\4K@Wq$%d`I'eRQO>6mqLn(HG?\jgP(hdBj0V6(?8b^U92'XoKtaVW)Bin/ +O:^YX+k2G.WY)qU#Gt3+m5cQ'"4I1^;LpDDD@Jr))T93&+he\aZjAP#`#.;CN'VHU +5#JL,)8@o&i*"mg?(9V:C>f@WGKDUMffXTcQSGY62Q;s)V4G53:qgl(E+[^J8ngG4 +kd]Pm?+BUap^Z0b:Y^Ut!<;M_B($pVm_WVoLX1A`\bpU"b1Dd!pVDbHI^>A);\pM8 +a3I&m:4q`TK,<=Z;g.UaU\C##Hd:n@A+gqqGAND1f'r@]AhPNj?]5nfp?L_JX(>^S +<_3O=B]esFo)"E(5`c'>)PQFP"odZds#Y1f5lNs*Jptn[DR6t[mGpc'@G[l$dB7bcNadL][UY

    M=;7eYMf#CQ=83=^V#?.#Oo)#mCEVL\\KhgS\T2U;6grE+i1;nk>o +L\#*('&'K9Gi_G&=?Q2H"'tLk5[XVk$h-*&Yqu=1D(d_WptQ[!rAnHqoVK%gFnj+- +oYuY6>PGBA\'\tm&^/dg/"jMQFa+qes)lBi0H]=>#Wp]14q&J#FW5j;*5hKCd?CN3 +';mJCfQujh#d-HJRQHi5og;^onmnRP#]Jq'fgC8f:#nBL.:-fG4'oQjtT&:T+JbH)SCq< +39A*6q]q`:@h'LlfC;Y&9k&8PcSfHFa4)Ia<\Y!I!A.3>9Ml0N7*6D[0R+_;N]]'. +P;GdkVn1lY'_NdTnS0q4,"8;0'cD1ZRL)=+$H2[Z,f_.*?EoUFe;(f0)9CGq)VmXTAqnt_,MVl*J>^[suQA^.BXn.;S^#+khHu)dYVF0VJS<3aA +aS[-G%"C!F\!be.54&r18Va&PPb$/p2sV$uQmDeg$D?`3'-71^d18 +7Ap=JY!!uaIW>,=7ijgS4ba](VDVj!E(N^BBj67B/Rmro^3C5s122n9C7$qk=h8NT +]k^Ys6u4ikD*4IJ*\uKqh>5I2LH^C;h1s9\"](`cO'h[A?O'?^LX(;R>Ik8tJ7K8* +FV;95m12Z?ETo?(iBD;D2?GLMkqmeT4F6obeHfkrit6b%agMDcBAFdR\9,9PN!MI]N"?8LpXn+M67, +!9/A3q"lV#qCE!ZnAW*'q*=pKSWnHFVj3\W0=;F"krc)eV:HSseEEf0kp1dcBZ*qb +``-]lQE#\)WJ+F6ibFGc.s7S%oTC^^ilb;18sQ0pe=nJ#A#g$tX%_P(b[O4=,G2!H +qMX&%Lm!!lA.N+#Q`.=$TH-gi'(h9O+QZLu)q@R%l6>AuR:_!TJN^h^)o]RZYX[f$ +c&h8B(i;@ac>E&78ipCNW+b!hJcP^=RmT5tK7Nt`Nl-5\k_37A[))1bG3[uD%W",t +Tju7_m0aEffJ;Cb-5U"Q?JuD>T!E'u-8eS'UXoW\BQZ-'qFQJ=&c4M>oXYW0>n=?; +`@AW`p7fa4YeVtQ_:ascR;;CFa>P?_/iAlV(2fHSBogefZ=<'$EK'7/Ap,@C-]+6! +iVbQEhD=;,liZif9k4.0$Id[^l)[5$"W3U1n@Q7\3SGs7$h,`5#.";uV(>XK!rbOo +!;n2_b4,3[B;-b`Ua'BB-1rM_AIUJ-5fVM5ju?LZA+X(-K:qREUX$9rFJUIsdhTN` +MV]lf0;cAu6c"\Sj?Zl@@&g^l("'7X0$C'5e*Rkjm*o^WC,A"tR[kXahi=dG@k@qk +,Kir$jV[^L&TV+\C&r4t$6H>WhB'hr?:o[\ZD##':V8QC"$*,PMtV53cEe9>doS/# +=3fimcQ^c>b7KQ'\kGH+&(0j&o,'ORSND_:G6\$g\W)*,d_"s/$5S.dD#CrBqd\:a +]=N@K4,Yuc8,Kabi=FTB7/g<'IB\5@s2?Y.#!QKK!,k@([di,LA+75>?,i?2o@^jr +YQ&Ot7BHUaJcDC4Lm=G8FO\oTRg@n:1Pgk3&)eIp!<-)i#l(?"5',?["E^T%b;_DO +'&rq,c^?;EBn64F^SU[4:VI2Q?8L3GH$DQ^6OcKbG46qo"8Iq;#&p-DjA);Qa"H%T +&G]Zga04]:+:.sBD$Q*CI4$;I$g9%5lNH*j:?X__FWZfoN_/&7!Eatfi;]D->Cca1 +ET.2c41'FACDVEZS9F"Y +Elpr+[=5\una)(!F7,-Y0C?.>6._k=K^9*\"H(4eg>,@FgLNbUR1J(,]+OV3\gq?_`rLC^=mui,]DDm]; +XCNLGM2JL9&rEX0Esq'5=SD4(Cknb'4XC\AEJ4!]Lb4->dm`7;,Hk(b]@?BL6I$7* +I3p4%-Qfrr4EBa(E#B>qQkYURH5o-m'/^sIfI,Y..cf$63&Vo/#)(cdcoW>Tqnu`c +6sP%,?$jLPqB,5S'Z6hJ@n)`XkAf`*etq[j06UH9Fr_N6D9CJ;9%J)VGE#rM>.rV, +^Y(!L4f5V-0fmUNqbaA?Ns"7b4@_\=]ZD,-l._j7i;`\YQ&cMmmm@o[KBM\)AoDpX +n@!,9nb9]\:VO1(]k>+ac8gLnN^N!tA"YWed1c-rm3&Kl2=,:.2]/*d2"X>_Z@117 +V/_OnAr9DJJUc?c<7HoNDqE6KdR^KAIJ-.ll!*hF-Cp)R>ZRnSAs`%/n,cG$78]+[ +5($LOS-EF"9C6t35!4@%kp=$oJ,3&%Y#=Ehi4W#9k(Qi8R7;G#H<@Fe8?#d7Z#TsQ +i"%;8$8#-^.FF/(e.!i]M9*^4LIom@7#<$F+U`3CU:2Ng9a$,?498?YM#B,4f0\l_ +f1'b^KM4obRS;1WM8^-O<>.8Da,-(FUoG`hgK!"j9ksP8]mJM-CQmJW@G_R\-FIX4 +Ta/7ZMGDXqbl9iokT1OF?tdT#O?ZZOkg4bM^^V;;$c!#Hce:;Yp9>r2O;`X,T^sS" +rQCH!)4Q?eqjJtbRn6Z@\3n`,IGGaH@>%M6MD&Wf(%o+;e9\"R\ctA,U83H=+kf_Q +E^6lO9?i?lAgbrGPN\WS42Cc)eecgPVQu*-fZ!(p=Fp07)&s2M'VmH\o(,kGp-0>N +@^#,l;$&c2I'EZ.$NU0)/`\)>Fuk'_m+LB2JU!;"C7ScXZ!:+J+@%r1$"%(\iLE]83kC!#gL* +6/^MFTof1T!#N@j2da.?#0^Kh?I7ZDd@9%q=CE<2r;1.u"OP43;?1fu"Sf0.n,+$9 +q&sbdS`e)/3.)g"DC_A=$Qs*P5c9uXj!d:WDNbn);jSQS!<68B-"h=-0t[OS%g`:q +!Zi9crulXS;s;fa+^;b^H_7,bpXq?8r\ta]BaF^q\^#Zr=SU$Kq49Zp:*2X*+2T?+ +o+-PL^d#6p4t$)2;b(b>A!`G=F#&&Km#9R+r85?@+5kuqlG#@+6-"OnWKceM^"?16 +hbYk/QH`["/`)S=\YE'1VjVa;T(12OpLhdJmoaqVG0.3g)>DDoa4,cZI,(knntB(/ +JU,hT+aRI.)kY]\lQ1U*E"HFjJsZog7Y6Gm(d^J!c9il+5cVmr3a5N.7N@8M6CKn9 +*H&rA]#*?H>5cW,mADPGMb9i?NH1XP+0X`fajR,_n:YcE +pqtHud?)JjJQ$S4;fY_[;UIX:+g$@!N=Y5I#e^s][67oA#We!fps,jl:W`o:b(Yet +":n-/:FaE1BVir:ASbOO*&?ihltb0p?E@k-Ds>$l1T#-Q; +B6:M\F3*Q*@J=-?No#+YhZkdh"ea'Eh$2.+s5/OSh$-Wh&".b\N4sQ"2[up@HAC\9 +S^5M3Ds#>hp==`]MDotK0a9]5RQ0o/3,p$'EZO$"%Ue"5lp4.B0\X"h?0cV%Kr-lb +q?b%)!L&ZVrTA%/3=T`Fe]N`A>1KPD)UFjpdsI%GpR;n>=VZ$^k-pc?[[?6d=lqT2 +kIrgqC,!qL0[6OT8[$,sX#l+(;gk+PoTg+!aO?Pb@,OqW;Pt +QKprP0m_Q-cQ+,B%M;;[82"S32=mWT-/4/s$9H]37s/:s6m2^6DB*Uc5^`?f@WPq^ +]Dq)65hBF`!QY=#\c!B=;=a<_`aK%F?"^le3"Bf0]B;1K;bdd#Nm0BH%_gY0**\/r +Y<,<%]q!4t?sSc6Hh9 +.:BlpQ5mt>>M_\JSm\K^2H@!%.l(q*3VKIX\sa!obDiXc_:EX>X+M"o1=TTs)GU]8 +2I2kJ=(TgFkqkhXn?2okVi`h'aiY<\P+B"D[?V`g>T-#7PU9O)gC\"q2DTB31m992 +@E>)Wi1fT+Wl"@.o(JCD%o<]Ae;s5[cLAg)L-'QK7LTGbTCAtTpHWWuDR"Pt=bCB8 +5j-uWSUW>e@j%:r[Q/b<,:Rliisrp.I!O]]5OIJrrZeTq9UH3Xo1(iNr("dhqT`CB +/R@i]^>&n3"D#N,!e6dCkrXZ;+(0NoY]or(Q +BGHp75%O&]Loh2]lX*(tIhT6].=;eG1`Em@T*VcFla6_=n$DZ6GNBKVjdkSO20\?> +ET4uPnrbkM65r+o+5F?m40\*6[dUu(%:)ke0)0jGmMV44oSHf;k`suN-$eF#fR7Se +\qb&d8&OZB5k)Ob;SK82[]NZ;+d]qjo6a7eF]YF';!lfD5Q1Ei,S,LgCbs0NY6]h>2NUXju +0/98mKe+!&qX]U,Dhb>th*moL,>HJIOjT(ZOd->c..k>HYZ@ZkIYIlD;5s0)"oDAR +lDH9V+/XrI6pDXs'!@lP8*(Zh1dh/P:=?Y=dM#e\/(rrbg0\LI7DX*=Wj0 +_:8j\cfkR+.Y\Ci#0k9./Bc$a4H>emLmWT)1A*&'=t4bt5(6NrF[A!381J@>d_B#9 +X&Zs3)0'C+E#Ou3SlQ/d;"PinnJRVaR8jZf;iHI4@9iNJP[a38>0a^.nN-*g-LTEm +@J.?Sh18/r^1#i[8]/)=qmCS@27_nuVWWWfF[YX[[d-%goNs\?gj"d/YToq3!iGF= +Ao7[Ic@&0c"/g$gVXHi`mmH8UKBKB>BQ-V?CRMV.$["K1I0E"QTp8(MQEAtJ!i!L=eD(1(GcStb?g +&'`o59g&X'>nN:9LZAS0::;@M>U-mf0<*D[A&tr`b3J'siROH@m/m+.$Dha[TlljP +[DV?#434](G$2Ki1]k>kqZ)!Lrq@C&ciUNMjH%2eq=*S>_@".T#6Fj+plic7WNBP2 +C\7%#%Q_mU_n1-GVeat`Do6@$_e!qXgUlh+Zfogllql0SW9X,Wl1D1%MrWs?467bL +jX3(Z$Tile"pFPn)(Z>RF:a3*dS.^-.j+YmQ=`4pMS=X[#QK;Q]?sTm=9-rJd?0Bb +!).2(^ZN;5nc.*g/Ff +$k^2]':>uhcIR.$BY1N,?!/'_r1+\YC^3ZM>oht?"%<7-IOa&n8DVQpd?^_E6Vnn- +aB=s(>F7raR7RuBUi#U!'Ts0TaH;h7Leq?DF:ehuNYA +Yh#I/*4/Gq5'QM;Fl^`mhCjkZYt>N"J;nY@qXfY!1fl2-I`+D +(.8!8'DDOkOP_#7V-C^rp\q3q0aFiD"JBo/^`XV-CP,laYGmI*.j_lF(uT)(G"pp%8Knnn\)[48,]iE`SqF +[nP%':XdAoH-AaRl%g/i&Y^M%F@-5G\&-p+8KK7bn'\ps0?D8rA\6:%iOqaY`Z#t=/']d/GF)K])T0SZl:CF +7M86)8%,$&rd38[pBs$3?8_b<+\jglm,\>M+Z99+[AYOnW:@n/ZVnUYMD6%WnLH@j +(8N-e,pVR6&f-]eB,.abPZ)3p'8ND/HGodZ*r1#:HSl,8RfsZMU?m>#QO-fe#q=`@ +K\r.fi98)J8^Y%naj_X$S2njB2>sNqPIb8q7W]WJIN>4i=6:NiB'iY8S_&OdCN:N, +^mT&c)5[(Zg4/PCJ&^_7G8du1)7XW-](-oUi/Jb(%39a**V2`qrPpRCVJa95DSm9o +6g)bt!VeI;5#X^q&3BF"C\#nd)P)TC4nYXs34bUG4UF1@7+9I4cPaehO17Pjj%\sW +Ja:[s90YIgOk%1_BO7>iTLU*oQ+b0@9q#cj=t'pG:TG2h.ck,Te=JS`B/LXi2OQ\r +-,&NM.ISMt_qIg9WAdMS_khrkR`3$"J,7kEZ._1:cM6WSrO/!-*NQ+O!*X0AS(Ll%V"WOnOcRu!KEYc3tF-1<#)(Vp>"jJTkb +@bH9HQ08F3\1YJX"3/:n()>P([ln+<4#^qS8O<]])HpMVh` +_#fX%`_a4*+1mS"r/\5rh/_iari`(QLjLh,&^41B=/hb[?(9W`2=I-EJDPGKThV`\ +"/#B0cg`U5i>o\l7G#fs:Aq,8j+)W-nPf2AVlka">`)FLMUEL?oc/-@-F5Ca"Y.Z< +`4ul,0?\AccfUO(Z:_IU-/#3srae)pbQ@c)&bt\g^B\&th*K#>VX.cqLnt)&Ook-F +5u=4Wb1iE/rm_c<\lp#q=;XW!j.rer1*F6DS<]\I'i#6W+DYO6@r3hg+Y(n,<7'R\ +!N!I78'TL[*TY&aE.\jTV<5]qF(D4i*)p&NC(.*LSY?J-5L!4D,(G-X[C%IiO&n(q +QSUOc1,4qI23p-ZPI@c]+lHqZ]`+o]NjJ.L:bqt.TS9'6\lRsu4k +^%ASsd'Mi\KC8F(?2iW+"q0Pok5=NFoB/=/&FBSb%p\%bHO@C_L!A8(]GTRfD'biX +JF=J(FqgG2bAZ,(7Hc9"6i5MJ0mZ>4!50h/jYMZeBae7Wo3XQMR(DG)"/95cDsYak +]2%5S:h$1/I(X*q+d!:>JaD;ED##[j/7\X,m*3c=te-\l%tmX>T% +rl!F?iB*Yu\%W4+"JH-Rhs*;[nji^;&3PFL=GQQbhWfHKKaXf8,CV/%;jYTNV_P8/ +dKsP#9E?o#&)+jl[m*2;PbubUt\kY9jd8Q%,1_&G5nI(cI\mFq'oiB%l>28 +XkRBS,Kg",Wppacn["D)f&a2]]jSiRCS\d`)Z +k&1C?HGu@A2hcZ-Y6(RE$b-J0^@\&/n=)Q7)8GY^^YqoDqbBNH!:(ZJJ+UJ6\c;#Q +n*Id7e?8Q,J`k[PES0c-LWsd#p)$5rF?0#G%UeC?D`P2,fY3n(ekY74&"C;sd!jiQ +O\r%S$s__e@VhH[e8*=I2hpS!q##SrQ\3N6m5?98jJ$fi=VnWEg%'D-TqK<7T[#Cm +NEJ`GT=R*n$]#gMW62A6C`XWgY0I=gi2(U36&H +9(sEJ"qK;LLA27RA!bl_!A;/.fln=kEur5ra6l!^`eRS-S-MgJIg%mVIX%(2^V]>? +jn*DohC_amcnT'iSi+O']&O(3J),$j2.1eXG\fB^Hp):d1N[ZnIpgs3D99>f3\e&e +M-$"[/l$I98n43`H^&)e(Zd^>j\bX@m0iLKJG8VN=&CVb)DP$I20*%14Ze/E@t +^-!3DE=+J;3&p^Om,r(V<2o5s1l`qN"g\21lK;FQr`L?I-;i1,8qC_#B4ijkjV0*K +D^ku-?th#`0U*e#a]f_,'+O"d59%e./JY30AZ!/0'+M)8\?WGh*&YC!:jG*:](7M( +b>JQQi:C=;$id1a.MS&//Q%'i<$;g;/M.]29V`+,*uc%Z1,b@R23G8QON(MHO@V/# +G-Ws)1u1Hdc'%1/'g,a_4M;GVGfp"J[BMoUcnNFp#fZF$Mnt4V]o[e.YIQgjR3RcL +A*9OQQ_AQBSo&d,L3N8T07R2UgV2uH0-DqMG]Ad/!c3oT!$6]O!9S_o.J0J4-YVC_ +%'BN>U9k/%6OnA#2??K@a/nm9Muh/-2P'noLu-Ob5Mlb)QNd9WdLYo#=oMjW"n2m* +*[#\%ddS?LfYoMtHjF#m%mn0YZm2sH&pgRAJ#*rfq%UC0.:rt$&Jhj.l0H8T&ATA= +i9T%/:XOOX6T75_D;>H#@lKb5Dh)@!.h-n45`7l$^5I3J"VJ>._XONp$r:JiWY%Q +'p[3!NQNKZRM/r^WBR>'rAU&iDm)hR08pj1FfY6o?\/baoQnHNh_nX;XqUj'[H]E6:#6[)N3aSuRjmG^5RPGl%3PN`Sk^Y1UI893+&OOT +oNhh:\u"0ueDo.;d[0(EQY"XTQ#jB6T5Q,jc:Z(%8gg\iB.PKC/h1JWX(fN:Mf"AlkMDJNp=Q*&;43% +\_#K5fBAXs`2WLL3P[je)O94@"`K!"&VA)R$B+sjrQqYICbcT+iC$>0T68GK54TQF +G8Z)/s28seFQbWAm1eo+]!p?ukkGm=gFB%2agPW8:d&\rfk4:6VOIa(e4p/`.$9M1 +I2#L++'te/=KRqF@%>ndT%cshl*Dr`AQB>j?\/Hg'D&iW7:6/3s&uc$(Bt0`.KPYE +:4<_Q^jf]G$LlqCgtG,S!#>S4G7h7B$qfgfoAGJ@s&E+gL?$&r]7#[d`pYS:[n?*pAbE5epTJiOD$Ne:#ELMC +N%OjdY5"'[q@9TOFg$(72C(Je8d^$0`\#AEU-X7*7sU+rnT/SV5,4ddTfA)>&cRjh +G:ptuQK3)d5M6!OJH^G[IpP/Dq!ZI,ojXl04"1&M$@pqA:+N +PQOX;`KY@_P'?"[H[*^0n.bhH)F4BB_;TDHO`D=*Q@i&fgVH6=*n.b!h&-K0_3EPR +X^L-2J3:E::[:JgpO8 +pd=P3[T7IAjM,@(bADh=?W?O,!9D14pnB]j"6Rr]+CH7>9g70_!Z'bHkL+;J532=c +?q_Xe2W'4kh=-L%?hl?_k<-RZd&,1[&E3(e=,+I^gAab8&Zf=cm!XcK#Pb&U`K=AK +Tkj/Lh?GRL\"EJhgY(4u!H8V>/?E7l(^_PTkY2*H\MMioF"XI#kK,5uoZ65Xho1WW +kU!Og\='Se+!/X?04[g3*q^D!T])2Kc(P[9nKNeLO\N.lXkC*u`Y$ +Rl"K@[9f=[?^`fX(cIoChAkk5Sf-+O)q:MFefJU.TBXjqBY?ke5ca08kN.h'eWk:U +'tO=lVXBmVIn?Wo2_:@oN9V0eFpO<.Aa!uo%b22V*!+NIa2TQcL0bC$S.I+r,3=L! +\dM(7b:4====77pHDi"/48Bl&"Sn,TX09`mkQa3Z[GGJ&"Z^US!B&,Oc*F?aI38?OO%64/_n^U +gsU\cD,@]Y>^p//HskWZh9J20[N5u`N_$8gp>'A?rX6d*41E?+hD=VGZa\f?f3[M8 +)oUbac3"WJ+*.5gS-RG65.ol8G2tf4lAN4Rg.0>)'%?mBlE,^!g9NT^[$WFWQJ#_@ +@5=$%Q""?k/6QMHR6d,bMj]u2WT4adAqirYe^[r2'g49-s)cRT9Z0W;=Z_\Jhs6,riKl(?../Jn\98j +5tBO`(ZDF^TMFjK&/A\Qd0:="e6R,?Tc:$WJk158oZhR"mI@umh0oera3R!CW4_:m +I,-jHP$&^eZ?c9m]::sSjW4>[4e\2HU99G&g]E$1+>8P`oJF5N`gP6nfArHsoOO/q +7&(j3TS]NX>iRN08N9q@#_3;uJ73\gTmqf,i%Hbo&fLp!)OqN^3Lu'!cd<+N15`%d +-T0TJDn9ODnLO&geM%b\AM3=%=;BlulVS\nfNs@NG:=1Dd\rH?_;U'%3k7hShW:Em +^q8A>[*Hi5!saKq#Z;h-Jj(Y-p<15TafuGo<'u-Y?n#0\O^FOTjM_Ih.YkS +DdWYB0=B.L2/)s;"oaJV.6e%RlH"MSJ):W16PYbi^]O96BAO.,5cXlL/0N^SblWS[ +rj2Y:n3?i0%4LMC*T#[.qMYM)lDt)6)/'g&k"(;OTuFXJ8FD"S52AlUoEL.ShEn(M +]EAUM;;K%iBTN9,h+.Yb6C^\8XD9BD8QYEG&/sgN*,P*4DgV?Q;'8)*^ki_:VuPh- +BW+pAeC!Is.uaKb.e)>]caOZ:fo)%V;H#\VZ>?jfX_ZA`HST?4aho%r*'0UJ0tBlE +J9YBdlToSU\nt8.$[T'_5cKB##l[tNaJ8WsnH8'p!G-Y>3k>FkVkE?!HfbJJFnjX& +c0d?a[qV%1e-Q\0?9EV%4XQU%KM4[#*_,g1N$!NO#JrJQG67Xt!.b<(&R`KLO^\*R +K8P'5B1oKiU:6O'Ck!>2h&+0$!/g%L2<_:>!X1%k.TCdJeU4RU#5H(9]\n4P&hPa" +l(\&^Tp]56I1\I.oXPqK[cWuql@ZL$S6cH;80E=UnN8QColW$LE]n<"(m@SWNE;&N^E;O'ggY8*U?g0qm +1'VnhDCsS-:R\*gl]Xj3j&^%#ZduuZ/+,QakXu]j`PsNe33u#C6`.\V9UoI@SnGTj +HRqoFnd$iJLHC2Go#DXX\2ZkG&_mrLrpT6gJ+%HD2]3($j6jqIL4m(O +Fnu!FnWWu_0&ZJ!kJKA.IiUP`h-s=OL3jtY?0m/QRc:7M\4eDXWt2Ad%X//=)c&U@ +.C!!rYV%cMh*:!TN">/9jaY2#m@N2)41LJMoE3g,!kQT#g@QS]i*u-<#7S/iT:4]1 +lGX07E5jKK;NET8DkkgY>2iX7F)B_)]s?nO]'afl`O8h8g8sq\+n`JtEPDW/+Rs,O +G,\hjWgU(DR^OddZcj:FbNpqp4[1,-1eJguaI]jo?39/;`5/j,h;bfjo?)g^Fd_-( +jmi6='DL$i*cD7jDZr@XlXg^.$eiYqm)8Vb`XhEuLA.[B0,W_fGVefA$QgK^8=VnP +:$g:upD@E;lj9c+7nk)Zk"b:6VtPgCH_Tc\V!7j=fn5ML&/saZL%ZX5@#Q*[`o*o2 +@SFFB:;5f7]FiA?hBjj_FoYhK7W9K;h<3,NJZ\j;'cBq+L:)t;#aN43Pe*ag$;uqT +oNW%'F=-)B>hJF`s43m-(]+%J4$@>N30iC//nd.=D6c]L1f&\Pi/2%4IG`2FVpDdJ +f0^b/Mnd[ZF1F*(j\]"U(&t!8%q1b].E*i!aUBgZM%l>1"91ANTe)rM&B=pV5qKo= +VEk4&+HU4Q^c)T'$#ul^qYnT_rb6[EH,`_C(8OD]mT#U\-[qoM!i,`rVM":fLUYB7 +H8P.ZR^`rH_CuUB<(.YSON+aH1L!Op-IaAh;T#*]j@uZ55Ee57"*geW9`Y!SbScM2 +4f@Dd_HU4;n?-/89/*rB4M@/oTZLHdk,t_*(??>HMY,;Yb5:R;20Z+?2hCQPN$#EF +@U;L&>!$W$-8np;5VN0G&5c\T(>t-S +!'AI4b5uAPTDO3\Iol]Z*C/3^'G:EHI`trA;"PVe&A/_k@NQ][5`U#b-b++DmQA'B +K`0"D+-;N_?36uO=9+iQ_4l;'+U'"#Obp6Z"D=rh\7eC6i'03F0egQFCnT&+ZUZ/B +3-iplFQTYnJA:2qjF:FO;'O\J6F5i/m=>4n,!0e->?\%.C:6KASonW@l$K/[3FTO) +0EJps<:CbiW_>[-AloQH#l?L3L2Ol$/@JV$f*>Bad3Gf$&`'4>ma]>NDoM?aU3\Y9 +]c^l[Y:?tVRkTYH&H95j`"DVFis3tW0r/5o5s^D=O[i'gW60qGYGMbIE@1AmB-@EY +k$tFlO9L$Qftd5/:]@UqrAI^e`."%)qiIkoFp,J.'a9$.=6OgVU&,r'Pe_Z'X&HG9 +&cTBDq:g;Ma(!Scb,ru)S*u6"E!LYSYo93%>97n=s.V"]rE=a"!UpT'R.jah2sKX` +iajYi5Y6_:#r7*nC!ofA)kSos?diRndP_+)?^'*VLt4`rQ9R(dhN0)XUBbreHFJO! +.o&8._Pg6oKP4;(ilQr46flEJmFdNUi1?:o",ERu015VDKak[(+*nSYpbJLTs&@%q +a_.:Fj6_X$Sbo>=&!"%>pcUe4Bm\h9m+'5'pHear6@*!nFmk#IHP+g(M;7hAp=O#p +o*t0N)705IUSJ3E`f_chbp$^s:H3$TfPSaCCiHZ[B9e%Sben_WRRCDcN5'!AInBYC +a3^8iCkXgU3HO9FEL]m^*\Knm7l8Yh);?OsE)Z!%s7ps,f13k)]^P##h$_.a,iH%h +(1/0;I^s+)gMHfIL@$(2rd,)lD>q&J5gob6UT,V^^$2o:-Ko`FD7p[e=+jL1>/%`0 +o!7N('fTA4cKq]Jcsm1UF(7f]0!59mrVT4)o-7]2\F=A\65pDa%i>/^+JR%9r\6&' +fo`cVF3^Sh=RKIo3S(Fs;RKn:Z/O9HIH&84mrc2,KltF*M=VIMqt1Em"1.]' +'0GQ;I@YOA'0G!SD8F/NBPL=*Z^W]Xf>CS/C/LDGTjS5P+>HC[BFkf;*72b#R1l5V +`]/@.IJ2_CG7#nZ`P5UW'#[6.Ig8N^7$[@_"Zt'bL6^7P1dcjA8NIV2mR,J5f@eM4 +UM!bWHlCaflQ3R$?+%sq0G2q%o.,l@lAUE5Cbo:doC)I?4nc'p$Yt!EDg2gngN"X= +e]RRD1UIol^*30[ncL5Baa!_.]s@sFG\FI$Qa7\G67ADj(l(m8eU +?"$0:ZKL'6aR))eh06/Epgi],i;N>IkGWf0r#f_l7iV[o5ScVU<)G!,&/0(r\,!@W +_?JlD#5B*Z=@F*8b;sL.rAa5dHHWcQE +W6UgYj@r@6GBB00hlfcWra39&K>8PU^6*]SR?"S'-4\&qlcmr1Alq6#?%?BefYYX +!;-Nl1D.F\+,fY2?3][)"cIJq`&]W0DA`;*9Fm*^8Shorm1FF"pQ(W%^l]PZ])b*a2B%t +aaZ4tJ]G57#d':Y/nq;VWEChk3VFc`a0$`YLIB`p"ar/WY_G'3+V"M52fsg=aE=F_ +`:lXC&(cf.*dA&:J!:P`%k-s2MT@D]<4=X\q*oYa"koo,/muNjE6"^*!)3( +?,$XKg*r>c!retk;TrfpAag_FI:q2V7m2Z+=3r*GBWY_=6e^8W9Q]\bk[Un$Mu5g2 +QJPH9?Gm:_9n%8;419d5D4Z$YmJh8enC^Q%qbB].B&Ia?^5O'.&3ouI0`I4=T"$Om=Zg>[1M`49A&);(&M9J_bSXlN)Q*1p19Tr6Xi/oB +i&iWZKO$M]5Bdl;qPJ?UNQF#$N5OTuoaCopc,*2WK5nD?4^=?q0$_BZ#)A\lMH/72 +GLH+i[LedYa)?7_g*jJmdNe5L>CX.o^J]T/IGOXI)^2:pRCAa,[G'Z<3>VXCM=GN] +]`3&5Z(0cn_$0T/IdhiZVu<)Qn:.g'DVJCO4*"rg8i1:M19-s)F@7s%_!K[2l,JF# +*Pls`7Ve?K#@>OOe:mY5(NNb_uV[FJ15Em]0^SM=`LN3i`6R*!;taE]qKJY +[,uU!l>dbbbkAOJg2OfcJUQ(^8`JF/i-Z$*p17Sg@9_H(RaZ&B2/HSYBiDb,Edkph +P2[=o^K'9=EQ.[EBNp_.U5j,9SgNNZIV%p3D`N9^QdG0k")54$7$VLSB&" +PIA#cOD\`7a9X&Ec?8.rZpd5gIp4A]#nk:7C"h:Eo +K_TI*$3F.6)?qul!n\iN/E='QUU+4#1BPHMhu`.4U%p&13]lI5pWWN+"ckgh]6rna +rN@f,5MV'.=3#L%Y7^F8NTleK%W92q.7dt$.C!(-"CPS.)Za@_TOEj/=ecq;M%/=C +>dtp>cR:b_)U?t^B10q8:^X+biYbJAU,Hq(.rT;QDg&"V]=7R`KBheBc0WUiaB+fZ +&H(F%A0MCp@KHM426]r?:F!EXJT//^J%k>h`6?OYSg^"%LfQbI#=SiB:bF]PX +F#NM97/d2dJ3^0.@&r3^bOBJO.(g;a5Fu6lr27s`==bg@hZ:^r@`Pi4pP^YabT++O +/>&7#q!O)uL"'c$]E'm&E/QTP/,1(hk$r$f@f4jAQ?lCi+;1JjK37Fe'tsqJ%k\uc +GeP'I#?&V9J.$lhJAf*74=O\< +hXR +k.Zup4qm,9`D1ljeI@DUeXk6WJRhX'aA?;/0X]]6fRUq'ebaB9nfbFs" +nSdCg>&hU;Ch%d^q7ZLD,;,B_m]#$.`"YQh3Xusa_aPI,LZ38@4_'5/!Q3-C@\cfI +a&Vq;":H(&ArZ2^a!%A5ArVEf +B^$`[g9=a.bRPa'fOU%O#;""HrP3o)Q=BKMV7?5eaG3^QpH_II7uR0>F-1[_#_iF7 +pRZ,Gl;McE[A;Jbc\20l@O^B);4CAg +I<_;AZS%c7--j[.isOAAMD_3gNN**p^jg4`dUq'n2LSVi1-M5bet(\"r4gt0]$NdV +fM'$17(^j^\:1#um6'E$!t02gO)k]L3sCDo"btb+#9Q)L@,W28M8;D%bT\UTngFH/'O0K[SaFAS3sgTN.EV9$ +kq8Bl?rMXUE#;P&%F9b1+2Gdcs]8T=g0!*h(LFiXI1T.K&"&Q?X +RKpN$?Np#8eHO^"$^A5!?O>]iRiEm:J!':(lLq"J>dF1%F:40?@#((5aU^N6aa^f, +j'O;1^a\3JYB-PDEn5`as-.6&cXKHHej1!e-2Vb5_<63qF:J]XZ,e#9N,Q\(h^`l3 +b.%CS^@Q,kqH`COgZ2qmf2VRX`)Y;$PW!SR(uL/gfEq".N#Xqg+*\[[#NXL,IC^0G +E>[C//q3-hH0Qo&&Wr]spl2`SKsL3)0T5]J5Q)pLoL8!%@;u7id)"Oc!(BTG#j_(TrT0mU\iD8JJbgg9VP0.'-6L5Riq +*5RbW3"F4X$$S41c_eLr-LHV/1/:%?LD/Zd=XLXP@pFYq_g)dY+i1FuoPsmaV>1)I +aEl6+.($sQ,s$Wq"rcYDj#.5gG:ss5o#1-LbV?h;U##7FGj9#;V',GtK"M]!Lp<_^ +A]qHk,N:Seg_Hm,OsPS=]-3NM:0h&1X'E%oVt%:&jk$O&^9lkS./16Ep:1MGT_,rFo=h6o +]t**M?#fd#+FVr7b.QKJAZsZRO3;UIocobK]gfP"k!bKb(%[IKPfKuANdS8OoMB\i +(-K5CpdFHJ7"4,ZR^G6K89)'I<lNg>8Oe2*I;eVs2o"?lZD3iam^64Vjf_BSq#g*ID`2N +JjX.H"[CV#S7=t21T.,)&4F+N+8F/@$AZ]0?ks(.RmmZ7U-LiI#S(#S%f7EuE0a'4 +h(T,Q1(%J/q*Q?fBG/D<59nYg8CnBt1QEq@9;o=.>]r]Ojd"]j;:b,]%\7eTIRb%L +<0>P&4$F'I&SsjZUPT/kY`^gEOqU>C[&G).Y`.)D9^5hkN%ArqfD$#"_76!k,\QhM +35]e)W16)K(,(8IYe_4s8AE]C?[og;ZaH4Ogd8Nt7(V)C]iIq>MK\dC0%Da12:-pS +5^qOL!,>.iq'Z)kr(T%0o'>5k<0#I5(6?;HYp@&_-n^kLOj3F;A@\Nm>Z0h3a])mb +eO`;Tifd3r*`bDEM%Sc9`nM.NCdGl4\3N+uIsccpr)pWR$Etn23)UrjBecbAJfBaX +_8!IHj.4Ee8`f`DEICc187J0OOQW%QoapUT\/[F;c]cUB(-lH@+q('T&1Po0"F#6` +9/tcO38>i'`6(hBGs4$_4Ze-Kjj!q6I=jDMj\A>PW%Wpb2Mdg0`cW^CHJor=)Y$KB +eZ"LRY))G/L:;(h4Hi3V9(AGkdEe*J]pjZU8<%iX@p>1+X:Rok*_&6*=]=8F\c^i& +Y_l9(JRB(?s.d56(\"3%]@kWaffqO.6%o7BFnh*5.oq,%fE;5j1!%64j8YH$c*rR) +d+<:ig]u-"7\I^KnjbJu]'0*>M&)R!qHBm"I0TZerTGI8hs4k9g5?B]91V5:cnR9E +,CRYdpmSGGdErC!.T5ad%i0`@"co8u!?Jg]Hp:VDT+h[^a[$g`9Z2Z+%L +nNL^ID8b)MOJ*K,K:i_eK5VqfM0urWU@og3*@EL<=d4K.O]6)FLGLABdfTQ7; +(L(Ql/-T[MN3gaHfbWN$b&LaS$r8e?]UFk*-YKcB'c)g,aB3;V\%f +l7qAl)=sHZaH(;F:\5E([es?o4q/G`%2-;E$MR#Tk689F7>4qR,Q\p& +(M"RgjIXuMU9I.geE8.]o?q,-f8naPpt]'C\[pdC&VP0*Mpd'7"9(.HH'9rc(8.PuD__P?R['#,98W%.HM7o!pcp[MPL(kBAObNpoP +\)CPQH5*.r82oIP%;c*:>YjPd/O_gm&&1%DHtsqi)PdA^@CO6DDi=g)mJ"BBIC?[J +T]LGqbFR[)%1^M!k0,Ye/2/1(%p*&t+3?;F#P'727tIn2Quj!-i@La#XspXYbRnqI +`Q/3B`dqC4!.dFd5gMYd0-E\J.*"<[e1.W@J^WhFHIFb;O\b4L5k?3!mTOG]7a*Z" +CmR1(23ChVp7P=$pd&V!^GLCQ,67keThYBnN'=[<*Eo6L02R!skrhZ4J<\ca@Z>'2 +;bGo:Jhncac,ePdC>S;FnVtiBWZ+st$]XeE(@m/kC23nTpto7N^/@b?:lg.a;GXr% +#GPJ)A&8u1EfU&Rlu/mlfP`=!CFhba^76j6D(LW_-pZr[f&t$5$bZ'*SiL'M/@K=T +b;kRHPX9F`U,aTa*A@k/-TEmA*,0X&8u*IF/0MkC:%\X@*"r2:IjiZp']MN?FJ6ad +Z&.mrRO@ShQ/_u-#Gq%i;Ybd@!hX#;[tsK!jEW$A4nsDK0&Ma&GRFSpRR4>Jr"h>4Kl1@A +,+)k`RU]*ICJL0s?D2R_WI%XZhouSp5Z6E#Vdl8\U#8U\cj"'_&NO?'6:1U:`chOJ +-)#2A+`492O-]_dX9:]?M!P+?=HDlTNUDlToOu>k=Da+b2R^SlfORaUBqqUsOD#5A +m"=RQKa[Bj?a,>i7]3A]BpKb'X";ANN)`+t?m!7+!'AOHi;YQ.$@5pPpfOEh:o=%l +^'TmV(`oK"5l.HJ"n5#6+A`kfq9N^rb6oG[s&Pr8c@YjuWW@MOCWK#Z,MqV,I\%[W +djEcV:l%=+>M7RL'g$oEg]bO75NU#EVP+&j#l?i`TfN/)nABLZ!FbO6 +H0,hB2>2;14+X=O3rsU+6"M4XlFP%\"[k@@D;iD +$V,*Si>[g\f,L6^4`[8,GOi%gn-rXDI9a]8rTL9d_,WJX4If!W^4*Ma!>gM9ic%ke +1!:-[OVDmperbO$-K4U'oIMcj%+:,T!ZWg5LHBG@2._r`ANAHqAPJX."8mq6'1Yd9_^Cs+?=!/:?1kq=jO#_k=4:q_#b&!#=`S@V +S\2WVgpS6Q^)PM*+4dh=d^^CjCb)_1pob$<;[0\0,Y +FIP4p`>utHKY>=o_ka76d3Q:0"sFg`ZMLt.,bR:+)lq)o+F'lZoPMt\2e`@$6J"\3 +]/95\,l5Vl`1poAUF#i5KuaKTiO:Q@O+gholX8WcY^knfc0bN'G>FI4\>0oX;ZZk_ +I\'sb=9.?nE>XLML;YkWjdLj1FeRGl*2?=o]W6A``b8X[="K"?MiC&K7_sJI^dQ(* +TqQt!Tmqqk:o8MOW?nCXP$2eK]B5R8FfKUI(_VC(P,O0,kbf@sTP(Qt/k(0&/7EK\ +^4._P!*_D;+:bN9#+(.-#kVr[#EVC&LU>8S202g??;U/-'S9IVihSVLEhKMmBmIPf +)Lb<]XgskNgkb[n,5BcP+NSK#UGUMbS*L?Y[c\Xt67o+G();"%76XG2#seKEdTFll +U?MG?@"*3G*ZhOmT+cW*/#kK@?RYOr8G(+BGi\GNEk&9?*+s5AA;5.\ +6q=H]7]Z\Dg5_'"-&">l6QHFu8Amb`63+S0K,lCL/p+`F45PT_7]$)5Cu:lS#GiaC +Jl35pP6>2`M&rBZO2OG1&5N9Ai5Xce0fUfD!`Xnq!l[$Wj;a`K&/f7&&Tooe*6n;4 +7p:U:=^[AA>f'>Dr^deC.`7RtZDT#&o0i#g,**/;F&';6?D^Y=N^ljjA]M)JIc@l;]@u#><_PVX`)1,aNOT0HS +$sU;k=/("FU;3XSYj!9\5K<`lQ!4gtlf*D=(SqN_GQ;aeFA0@8r3&nAZHr>e&A\<5 +hSuGp3F"c_-m/om?G?*1G((l+*s(N!M'1V/_/T@?Z12"V[M<;s+("OT%TqqFJ3sN% +if$`%-TP)nrEGk(rJi>lVBdKHc\ +0T.m[;kM2'he5Yb3WQ;`V?#5WEKacST[0Ej?TF&Ebgl]fE9gHfgABCMhf9H]b>E`M +c>9)c^ZO%U@9^3iZs%'I-+C(Dg/LToMSKVNCh`s//'aD)(+HpDS5Cg,MG)']:#Z#t +6CDWNZ;&Vt;<"CVE6NjiCQdIi+a@VD5CR<:]>UrO>UtKH;rGYa&&EmGIeHH]g1&iD> +!jk`E_^9tYLa+;O+HE,!S33_sp$3t)LUT;(O9f7es1'tIuqIu5KtT=dhn]i&mK6>CUOo5G0T84<](q;@5doG +Eqo$en9`=jO&kc;bP5_.pl:WlY'#-DE`i3ZA;;G5B.Y0q*=86u')M_/9)>oZ1:^!t +Q#K2<_L^7ucF*<V8PXnuTb_U9# +,e+6'?:bPIN;r-iT+$9(-sc&'0(_jQH6nJ7*o\FUU!O7e(V92Nh7s*E>Z\-0)U!t. +GE&L[`=VEHibe4jee%FCi%62Ck#$A]kZA4Kr8S)a7'\\(D?KB@E?46e>CjjIHSLF[lqS&WL+1U6gjXgVL3YtY,17/=Pj"\&K9cu`*`)ABpdpY*kb!C)[W59o\ +G6V3]'^[5BOhB[,]MD^ZS%&U^7k9"XKicJYcth51#NcZ_"`\9)B&kghq@dB'1NOH+ +NO+AX?O)#AY'o<2!rq.3(Z/.,h@O=,6P_qUqoW1>K`g"Z"EFXH0.7[X>N3kRo%Zb# +!keE$^%V3]NIPQ>s&ll)H]I)Y]&`U5q-S/Q?Ni7Vdl5mKRj>H*'__2l]3:Ku3T0qk +dB[>i&rCb=e@n[e?aFEE1hN%1@4>N+.Tio*?F2=l[]Y_EQS"S;cC'L?/2NSce!@(- +-BG$"\Yh%VkOlE+DAu&/1T(YMZU`6lh<&^gSW4^:(ge26rF(]=[AulBHMNatMBIiT:6$V!fB]jQ[1Mu>3D0RGtIl"ZK1N!1Di$Eg:%;Q`^O:H*l +ar3RuM(mZjrE;O[.\Tp2*n\m-#hTg8@f(b$Ki8QACY=>TB"rrG/LPaZ&3JJWSk05/ +$JcVOQ3m(u^R?sWjUIq?=;-It8Y`nc1GA?dhA!*+/mN[a\%%tmUDAok&kD#Q!@b?V +c5_pDHIfn?O<&G[f>6U1fTf*"G=K@P0Lkl9Hu?QUC0a..SZloeq2Z% +R@V'',QKH5o&tZ;F49\6:i&f9uA3uoQ4Le#MPHcJ]XRRV! +ciU+X3ViJKf&QT(>B<68&$&THed*n#@lBmU2qBdRCM;o3Q^WTIlF;R`6 +-ecNOCh",pU>q9nA$)J7bctlS\-*0@kpVC7PYS]"ZgE$#-^Mh#8`Z;-qED_>pPKeg#%J%@u1?7#@7Ri;QaFDr34GHcYa`;4SM[gF]We$>FI9cDd6:*&=sJH`[e;HE +a/UbbrfsX5s3FT8p\Cbr#L3/_a8kcQ8qHo_cP>.EV-##k)LngEr_g95K0^%>u(Z0;"hQdo'1ZmOjB3nN6?]f!:"b%-,6&>=< +l1`,UT3'r8_#HQ=knRe'20fZ70(>F(jopJIUOgV@P`T!i0r]4,,0B3DT\Zsqjb1ZO(A1GI=Fso420%?dWE?cN`BfGa=/Yu2A9Uk]p$'?rt-Yc%Pd`cO!*no?pdPK4h +3S*Hj?H+"tB,K-68&(rVHB^$A(G.T\c.`*BR!I85 +M0R*>e76^=_^2\U&UR2SrFUX9ldU(GMX?W%Q/4GF:+a82\168dCV?oh*kS-8]Y*4N +,58[I4P$MpIT:j0FlPljiq2o7cusq(>^%AQ<.U)gcJF`i4AZ/E>X*XL_5DFoY--Z^ +CX$5950.$`adh*ZRF-hg +[Z[Ogr6Yu)^cJl9NSM'>H+lc5!8t$mT;,)2fdC;%Y#Dr58KApu$S5[0m.O96[@s8E +h>l&%]#4RG4hpQ.7rFUr+i>ipbq)?I!o[Mf5P^@s@h=TUWuoPE$/h6T?u5!kRXf5d +cn,TW^]JU1&;n!u&'Z=&JQ(dDab?N08*(RV(E18Z)\X0m\^t-8mWmr;S>[84^G6aL +B,Z8H-MeblQK'#;7m0u^Ha4LF^(BmPM:O0h%BlT;X_&"pkO^=e\'5`HSLD1u='F-S +??3B<&&J:"H?rL1UT'!q>Y/][LJU5'Ht>ugAO9*Ol!,Z77sfVh6#:Ce6>[bt5Ir0X +&(jCpN2TD\.o-U':Cbo(*1.8:84"dBKpAM2Pbr\X8s6m'$OW'Po7X:)!al&n*7kn% +9+oKPW2gH\9b04jcaTAG>.!%hQ7$;!a=3jG>F=K+SLS[)&p\#a3+lgUk2b/2jLQ*Bq3I>=Wlkpc +T>VQjQVL',>Tr`WYX]s4J.J<[^r*FI#tURR#/oa.SGAqmCp],]LZ>Jid\%g'B&n3,3\2l +2][QPGd3)G(nalZ;DlJomkm0Dq*1-[^^*3p'!>UXD[L';-GYgP.a]8UK\pNfJ$7`- +[;m5-&,N&9#7X;5?if/-XW`.p&JSnBk:#2bTbASH@n-1i>M^"2q60"gYGfo'.9!Ea +1I,u:kIit$mk+uI.cqg8dPFr +;5j:lJ`\J0;K_-qWq6$DRi3.Is#*F+!fIiUF-^B1Fu9Wi(8XZfh"f`q7Gat/nsi^_ +`!N/RGdh5:[X,ijDUkn+hDAlb+CiZr87c/)YeMs\PcH*1DUM1M=@^T*@P%`2k=uUj +Ng$A'U8LdWCiR!8(/)P@Q3bpFN@[tHL4qVao=/:sEG+^t!]pj$qofnJL5\kjb+Lq4 +a>!UZ^4/2hSu$AI_c]s.H&Yt)ce7D3fbVc$C,IUq>OIhS$WED`._A?&?uAtC_fQ82p7ghjuTJg5CTd(<$) +I:e(N]/5@;Zbl$cp=Kh6mkkgN]G!gQ7 +Q6tOPgAdj)Uhr&rCHLOd3j`01ga3PZ]G8tj#_'/AGAZ.2r.nYG\G`raNu#^6S#S:= +JcFXE]D1RdZ$d2?mu`>*r]>eLc+#rR8&,.BCl^=Q"4/2/;Lj451ccsc(AUWArHq8i +pL]=A$ned')$^u1&P[,#Vfu'O&(29f&goQ<+ZZqHc/B5G,k:7E)IZa9=IY/5QB;RX[X$V7^^61[3N +OZKplNa]e:gY?p)TX=_AJ0sU7$r^Jbec*>9nEs_LcmTMK-HrnX`g2ql.`0JFc-:QNk>\bXC>p +N.-FTS;_eY:[Xb85F]oG2I6/eWp8/;e]"6)qX3qF-Y9it9>oH*7$hX+/[a)_F-MD/ +>%8G`WJ&u1*8aSIY:[^pQ6]EFOK.mbC3!cuF2D[t6S#JeW\^EM1*S6t,:_-Y605(S +A`Ada;34NrAO95W%jcs_W\TW:B)#E&qin"(Q_f!CVg&r64UOTK3IKEu^>&Zs!PAO? +HFtIS%Ma\,D'EYU%eM/^&1bbSPMjG.]*i8kg9`jg6R3d9L +/V2OqXtR*3!:2qIqQnaMQfce/GRI7:k\gC1)[i51#S3U,]ERM\7gRp8(]WFnQ`BF+ +eQ\ZH2S[gZs&-<2@t=U2.dcC+2.cmY&p"2s%XtH?pjR)sSqP/")ScN*WS.OapoS#k +0Am#1g/!""G?MLlT3N6\XF1ZF6mPkq@*\1<-=0NH?]ZYmdUmWqrp3k[m!Hb3B+CXp +>X*6E^'[oT,tZIl?iJL*,jG&&q%p:M3u@Hdiq`m4EXI%.9a$#_5VBj9]OrjL"b;"@ +&1.pE;ii."-C042JH!VdrhEla?cDnlcenR^;u'([qHL,^DZuU0>,_8:RF7V9^#jEC +maNAgY#SYte(E9OmE_t<hOI`"e@&9:>c2Og3:0SOSrgE$.r)C"Se%Jcj683Uj+K-[*^o$U-AaN*U6VKF:CNug +E/jqu7%A81M=.E +:/1%lNJm5$$a.J7hDK(d'sk?&F#J^[G/a(H`F&^VO$F[X8n +>ji_JBV.Bgd6;6%0X-P^M]%3CJ"JcJ5TemW&+-\m[qUJ`KPo`JH6.h,-\ro,%4?oqe]E#fn +j9"+c'n9=;*8O5YgeF8"bZ]=V0*1VGs5+P`h\8ZnF&qui(O%YNG1,Q_`KOKaR"@E0 +_q64%TN1!XfpMZs;)bC^DNEapgcSS=@DaNJ-\DSuRDbt,`h@Pe +(3kC!eB]]H.*1"7olCR/WCB/rRW1$;DA#8p$,Tq4B*=nm/E97uYM2-N(&I0pbCIrHL(#F,[Bff2K'lna_ip'e@_mT9B2/;"g +YB)\rDHB+s>C"o^R]H`oM0CJ[\&p_5jkQ./??;+1^4oTg9rtH>Sh$Sc(K)4C8bIB( +Es6!;Kb5X,YpJ)Anr3@d[&C"sTZXWlZ;c;Vr05P@EA4=J.2]9rPSd4F#V>*_PmDu( +pn=Os+0*FH?Cc5?.ano5!%ad"m=tl+-+`7cH=ZU%;d$9F%+cc)Mid9,(;M1T1D64F +EK3`KOYI8,&i870*UCio\-j]+)#r'qhAH96P)b.prh->,hlKfGN&7_[S^Ps"JH%'t +8BJt[BJ)gUn\6lNm6kMHOQjQ1CFnTq;f*-H.99F5&,:3kp(7.[P$F06O(O=+JH+:? +:0Djc\bXMSq*$+Da3'=$?d9t)Zgt#g0$3,Aaf+qW:L#&;bN!r6jFIb,0DteqjM!Kd +rsE7$/jMMe<2+(\!jDd//h_I3[.#tbjWJsRi;Yi.EKIT5Qa-6.#L24;2*#%FW2)4F +#,k$dR*f*Q,+&GOIr+6P2&BDm':R$?rDp<.0`P1c"8p43^.e4. +7/"/`o;PUs=+ED@s-MGF&7c#"ra*).mG.,BSRoU[?`SD/MgttYJZkIhH^2$eR(VoH +;t<+VqBf_>FZMR9I[)IruG\=4a$`[F86X/qQE4' +#iH4/n2YLMHa<:q<8"J>EmSHL^K?@u+A)E4JEHmH$`(OojM*(bhK)^^$fE@B$:o*& +Mso:6Qe!2BVp)FO%7Sas\LYtU.o,b1p8Cug +b-P7+6LL3^_.,Ct8DZ%M36kB]E7\<(,Td\nI-jCi"p0n7;+Hc%`T14h9O +M[3saM)23]Z^\;GPrKO/?-;8P[ICM:4Dt4`pQ+4hpUn-"FA_!r[H:XBo=H.8\h8!7 +-?6T'i`-!p0^D[l@pcJ";RYX_4Dg@mlrZ/U;h3GNc-f'ud^tKaD0j^pVT_4ScF@Et +BZ[M_F8YV.fpb(DYGo2f640B-^U36H6g];#m;kdE!]AbsqqP,@GbR7r5^N@Z.c^Y^ +:EZ7dNp-T>"4_VTq_mu8b4*H?d)TALO0q8t2bm\Epg%-H^E>[_?0-Q3&bo0DnlUNo +^)>O\2E<,VA)t[h!4.2-ih2T##U0/?@-RoFEfW,QNpTk8G0!#)'Of>G;gW2)mtc]Q +XPo+Pq%INBKEaal_;qJ$&mf<4,EN;3klY$\#6gqC'oN\ci/6(kL`Kk+$>,?hd4M-2 +TMNNE<,mZ#r%U_g2FWh!8M`H-ZeK28JGoGJ---I#jK?5>?%/3JFHAMtP<,g?6;%Mj +!E*+i)S'XKd#%:B+o]ku_S]AWq_I9T5f1K5*pMe;lo\3UL2h%,,sc>gNU\)*s'$TR +i0d$t8q>h<5rB.I)Me5>E.L9Y!2a`Ib;>;r(mR@i>IO +(B=h61^aqb#,RcDVATVO)i#5s7Md#a<>ZD8/=8$?JnlEQGJ22scEp>:4J%LY3m)`p +T6'C6H.C0.Uhf/X=6nnMSVq8H8%MeiYH)9e<:an*ii4M=X%MEQ$]4%e8bLaCkHZ>4 +E^EB>6OL-g"_mMGag'sLi>(@YV;frMaeo1:`CWA7_Ebn#V?23350q!"(Jrt+PaCBnPNn?q +m[KL0S!!,knNS8S)[(eX<0*hS=$a,*=(KY>];('fL''-%7'WF^NWSeiFA,_/L)$1h +c0*e,-JFAGNaf9"37TfJeMTGE2F#Zf+Od;^rbiqL;/EeC*"tT$@T`JX5S+D-U +>Q0_oN4Q^!8jB1Ehks#LL7c6D.0#a>q'/u6H_0N3>ik-*aa*sF#Urle.F_nVClu!`>,1T!U-=$AkL9Tn +hefS1\bd+q8#seug=EO@]-e\;T6R_cr3>tFIJ*+lEh=;.j0uKbT'jVkC_BV8Umg'H +&\A8)Ta?4[);Yj)EFDdX,"C#gn_8aPrhki]>S/*LbUj8-`[i0&Ss-K@(uT +pWC!oEaJbQo=9P-[DqFe@.bso6XU6_U+(1$6)rpMRf#OaoKsNXg[7nJ\s'M4S[PIh +)3Y>t0/+Hm;'*t\Sh@'La>(,Q+-GY.`DVb\pr-7jI6/iHZf&.=+:j\q)lQ_:ck\a[ZE=NeZU9%F+`1 +3:fZh.-$eYoP`)kJ*PYba^=ljI-L7--ogLYHJCaH6ia+Fj8Rd(_FW=*&_%*)'>\4m +:so[BagoHD.7Za>a26LWU:@e>"o@beone[J3NY3p4"_7l4mO0/?6#\FHWYI&\WV9H +0)^07_V(rBo9Ccfb*($a7@c%5KU^]J2;k1I"JYQ/.F2)3C.&8Oao0aY7X..*s3iqcUr7hdP[RE-3YW^R +4dKVPb4*mFd^cbqRs4C3[2O!Z@bbh+S^fZpuC_R8cnq.tX +"b?hb7s*-*c[mb@Cq!=AZ<[7#9_,JUL99WHr5YE/U8VFO8crHbkgLa%LGClOs*^m- +'mN4\H4'S%[tSQ+(4,hi:PHOcr(R=?BgN8;8X)%T:RsL>>rYBChbTh*9L8gU5>-]d +^5*P;E_rccQ+Q`:DrVj*paoG>R_B)Qs,JDDoTqm)n/pR!HCD5In+fV-nct0hq#?p< +IS1D#rY&>Ic.N03V4_;VH\UaN6H.j*1k2^]r_&+7k>M-5.TCBY@tEhkT'2Lbkq>`h +/)6r[<4C5Z8SiNP@Wl'mg7U,aY6NXJ^=*\i4ue\pVr9E8$PiBMgTt +=>XSc18`(ppttrmZ0qsI82km#T/C1Z]O4dHBn':Gfrjl3&3D_B?9m).KQ^G5%TI%$nNO48u +\-FP3g*abbGLn(Nf0VQ#,.j0d](hCn9@;Rrnb4uk*fBH>1@hAs49GGQO1V`cE,R&! +G<*2=IU#@6&>mdhDl>!umN?]cZqFQUVX_-`b6J3OqZV@)@nKE9'P-iSUd&$XHT7XF +G;?=:/@N70qfKMS?AUius$Rk"BGg,)#[KYG=#W,%6r;HS!-K\Y2ZcB2&kk8#%`0:q +_%)'*c[8[+5*bo.0.>dpn"_2BhuZKj:16aV(5P$nKKBd)dg\u8!VZU;DumiG+CkL# +_C3IAW5leia!-nk,$Ds]8ot_^%5Z9IOtIU6>dNFsqH!6$?0^D!h(#:-X285F_)59Q +cBYHJ'2;:@+;;pCF.,[-BF +/@+9Za`42M7#EpYf<^Y,E`*nJE/n&qkE)5pA4KD'*Kl`gOT/Gl8*6EHr!+mJU*UqO +S6RRbQ8pu9JBe4`Uo#'.U_mP0#P)M>(^'R?GS:_;%"gcu@[WQq7:B:QB\"/X'p!dK +0]/U\j&!U60.uoG/TY\Y$>_#d&#<7MinDs=#I*/old66pU3bcJLk#&2/fU#Tij?b3 +N%$bq#-%Y0(3Of\AQQ7^gGhn9fp7:lXdd9g\-KB +h86W!?mgH?$W)=bXT\_)1!YCol@Z=>O)f]OqJEA2_s*`dp"Ss#2"2gbSG1&2ohQm4 +;%Ih,%"$-I/Nodf_R0"'/2WW.QaF+q>1psfkC\/QMQiC[EEfj^%:$_#HL-n($"""b=82U"gQ;uaS@X4 +GnBsfou]0Dcgjfq(C]Te]+WlA_#G.El6X-l23[<"o6M,MXW)u&(Y5^+/DAYbR?8g: +8>;bdBlJnQW_a;F'*iNC*SGfZp53@O15>a^HKGD,b*2u=pqL6>3ibam3K>%B)E7Y< +ifY?o;4mZ@s*_pc.R&(!rQHB[k@j*m7\^ac7D#j"gH(bA3JH-Hi4m:rL;NK#Na&F8 +Tln\d_#PV\^G2n>-0a%!FIO.VW6dh?NYQI== +&(nh#N\-*]&>8![Wn[O9HniW).uM6!;D3fRYMt +F9ab0O8?C]!UW4WRXbl<]@X]b5Mo$3;Sa84"MDk:KRudZl+/7"OnH5M%Aa3mghoG +hf:6J1+!b$-LhN(3e;R.e0HedP7[(Jl6`De+>t)FT\Z.50)di7LlQ;7 +".*iM-tkG*Z\dd5!N,4?_L/LGK.2 +j,>eBNA1,@nKqe@=Z4qa48U9Ks1CH%.b-;8Bu+?6E`r)>q)1Z(X3q')dR(F+'!IH2 +[kgIe/[U0S9'JJ2eoFNHO>%*j7?Tcb`W7Fs_2U9_'K$%d)?L9XV2d(q1J5M8\:?DW +$?H4ZjD)mHp@8daGGtFFg+p,(#M[[ZD=tsCr1JK;`?2S]+7_aG5M15SaT>BI!92sD6;r+NTPYKk!PF1qf:527C'BB +i@Fd$%TFNd_#[35a&iRb3i!-+oC%k1_B1X/'ABfH +>1Ggd`Jp9uY.ClLV^ds_m[=#ltGhoe6??r,cm<(n]Z;]Sr(MnY"frQ69 +g*gaZ&3'T2ntL$YM@je'-jBC\Ee6>k**<0:#OQRW`b2kO3r24pXrDMk+$_Y.%7O!;mAq;1q.:e;H]^YC!]pH/7$SodBn1X2FgX +JE/.h!,Y0aYV^5_(i`n&eK^!$W/,mJ$h%BkDHh4r.Ib&'$+Qml!-`Z5$[W-pTSsi, +>(as1LQs#;[^U'Eb; +VFVS&UlD<.a&+(SbV=&;3UsA=oN4L$=j')+Hmtn)0sl!c\8Z>KrrM/^o50[q?8?P`-FrsqNrPU$1L8'l;XYQ,5Z6/3'epnO`"Nqk%D%nJep +R%t[cX;@D$(CYHe77"qM:m-81`A!<`R`mS.'j>2`gSU4gA5l\Ec<+#k>;PP>k2$Fo +)eV(bV,t'E\//<$FnE%UAkF@*\Kii*0IO.s]?ULm+/1rl*e9lDp\A1UB)33PqKuK0 +Ja4\3ir>Y0s5t#fTu[=XV@["JXe+^5=[3]0m0G2dZRa>//',0ZP&8kT)b,Qn*mTq; +C6b,p8P-A39g[*VZBNOSOtF"=XE2fdX+*aQ4Y>-F5lgo=F,77E]_4/_d?9c+aj.;+ +jk6l:[S47Q'%7nYm_GLK"95@2[ZkUo*3plIA4[o2B4dS5o0HRT)Zfb+"B.)',`?3% +PBAW.a"01D,fY&79)6qQb'n,([U7c +D.g%eaNGL=OsX\tOLT=e?Tq"bfL_Ie:RHEn]_Et1s#P9?o(`[11?^`@DcrQ[rTBEQkELu#*8+\V\%h)T +^Z-emaB-#a#[cb3^aHq\r4pt]Ke!9#s&$3$HenZSOS0V's*=FZ05(%Y;SNYVn=O?c +r*H(GS%Gs>4!Y-BWt+KYH$@Fp3g;8a^jiW.'rf.uU&"LqKA'=hdp"^HKm&eOG*`A5 +N=1IuZcC7%:QtkW=&5Do19LZQYF2"O/TE>eY+!p.ra.UdgTkKXO%8Ri,P^#!'j*@i +R?S*t',I-Zq"@[N0gkN_s292b%T(N.,m^a3Y +c)#%GAEe9q-+/7$X9j%*VL6[rGhSs,`jC';m/3ejrdul0HG7TE3(b9D^Qi6Tr5ks5 +Vi(nC>MHl_gV2KAQaR?m-K+o]e^Y,@)f#+Bi4h&2JNU#8'0E.L*`.R_JfP+[g;*sN +IlQA47m#-&mG!DGo*K*k(8Q%`d`?=K\2<4(W_`MLBC!-)LFDY`1o*VWg8g%n=2pn> +5`1q;=j![&DEni8XU#P/j\_(LOaY#ppa@TF5s:peT%2oG.>_!?UqtUFT$=M:K$\i` ++a;m7'[s*R)HGMj=Z@G'C66'6+GdnT9B.fus2RjK\!6Tlg +O;?%AQ=INcdiOB<_rBIV28\96Y'1A?hEIKNHQqGB6rMW19(sO4HR8:?>"5]On5u&4 ++j-=>PGcrqBf1$FLTuZDl>]jJ=F=iNp_pm@EC*=!7p35KZ,@cmcnL`])b8A:U?Y83 +)+`T.WCI'+"VSTD9DHXGpl?j>Y"):, ++7i8f*WSfe>!O_t6SFC/(dg#L@%9o!S]'WZV\_E+arJJ;[Q2l]e7oq;SIM^;RFrHb +4&aqka(r?0i&9ZT.ruA^E[#a2@7@_h"^+,6/h)<@o]Hi^jl?;!RD,'@)*XGY1OOK4 +^4kg#8H7!8Nu%6RnGb('%?"%EAcT!=U?W"=VQ<.*/jC-eYcof$$f< +#0mSK$Ve%%CAYtId]@KC$ +F;l"5QHcdHZ(Mj4$a9^@@-6.iF)CNSXKon_,iVAsh@tVYI'g/N6nnTVr(V7=bpYS" +.26iR5E08DB:Ao$\K"Zlc`t"4\t[-WZIljl`4N+pUSciI$Z.\l=64*qs)k19V;H%Bo"8RQJmP\bYOf_`r<)3%/kt*ENO](@+7=>o+RQB!OHr2E`8t8+ +s+%cY,"M,X>OW'9!rP8R6-jg`iqREulec=_?Yeiu!r8Hun-1sRHg<7Nn.SlT +XnG--s2VYtJ"2#ZpPViIG2^QHNJG9>5Q'R.iNJO$Dt@$(&k=ffBETpP+RO5Ojjkl' +\0\Fq&m8Po1;C2.gotU:)NqkhUIFFu$l6'fZ=3<+l,8^i"iI?_M[$mem>Pp#h\6Kn +EPqA6`lU%&-gSSMi$pF!5`8'Sf>J.n>IN/7Cp"33dH@$6RcmY'J$IM(>iBa-l>OGd:/5Ph3*6'pi4aGOlWc#;j[1@CIYtIUU5`M +e5]f:a9g:KYP_NaVX2Y%3BAN/=OVp89gZ%a?r_6J.r(QS!.<_>`j91,Vd&Q>]'%qQ +FLWL"\>F@aP#XZ5Gl@^L:edc0V:>!:)m9R%g2a' +NY'7IYL4Xeb-2dK\X,IoRl.)j9IV_aod)u%')H)_0%J03GF(9jCa.j>I? +P)%ToO4K-iX',1_nFu'bpZBolc_oQf:[Xg6n!K+!=A66F:[7&>4Bf]5^aG&Ha/-<[ +nsh-f+PfLZ-61Bn0hs,8@oEo)U^2>_)6l]Jc2GZOuf.\Dh0I#7*1flUk +kIk2`<6uY1$(T-^2 +Mc@,r#0.p?D4Y'A6/Yn-$E^UlmIH-OE[r/DQ$.Gt@C88nX6mF.953ba,gpC!;HZaC +Q[Gee-]gh."Sh0[48^i"ZdQNAZ#SW*S5012(\DYDf=liC`1d*9W4g%85?Ye4&e@[o +!bVjGYnE^aj8?(n^]F4;PjJGs!9AEu3X;UT@DPXHADf$&3%J0e2)W900@L]XYO2&A +^a()uZd[/:ai:[j4_jn77e-jDc6/`ib0)^+;6>(bl[5Q=Rq9KGf`Sgn<<-2@@SlEi +g"+hZ(0RGsOS/_P=r@A=Jc+23'+Fkrs1,"PGpd;_$Td+,_(fN-D7TI["U.Z?V05jM +NjY])Q).e%Q(Bju*bS,n0,B5tian.9SegmC7(<9=r%\OEWFK'*Q_o]kE0"S?$15CN +$":snn:*N8&`=2Y+0LCke\+(/OFNn-MY_'C0S1Y@qV33s[T+ +PTDo3CV04u"4l$EHY]+UEI_ELbM\$2jp>Jh.<4`dp?&i0U6h\K-Q2j\["YCqr0P@0 +[lI[M;'k,_M-ahu^4N_q%K+$5[k,p2%[mds?'4,6U&;MHg%WJLMTDDC6Z2/D5( +7gSkHZ:-tlWILstnn[@9ms+-,*9jqK$2cc-`t*m0^Q^-q.[L+B;?c.]Hi3Q&t9h, +bA6F@NI9cml]r??N_O9WgFBCMJ_cSOKA)/ljg2PK!.W)fo*3`d#1]3ONFKrOS(?c= +!X'9O^]cFe$Z3%;*3W#+RK^.O"g/X`)RY2$<9"?LG5) +K=O&*gojE(3kDeZdtKMN;Z@MprABl=N?.o^9)EBrVjg;f2@/J-hAljXSNF"A17Mm0 +?(9!N;deL97V'*;UKs3M2Ztbbk>#>h[qT-W^:%p)I`qEtr.hIY2tAQHcC18q4*c7r +[UIa]p9@3ZYWc^X"CkQ#3k:?V6GKI2,D:Fk9 +GsmdNO=lCV'*R1RHag2)hqe^M.g^L"L`"d(s#:8mKXL-tQI!GKIkg5ZgFJ0L;^S3+ +045I9oEf[&Pd\L0A,'/79]&I`S]/D_i! +B:@g_Q:.T=-I7u_ksTHL3'0[pGXu^i-e=e\-Wd*6()'IYOF7JBI(:L.s$8b6+;Z,M +Pj/4i!rgFDr%r>0.af^,K6V)oQjV8h5/I8_"dIr,)Tt:*b.Q#dH@cbXqV_3U"g"Qa9"+r>cM!,kLXfE1c/89Ou_(>&b;2q8W# +T,2Po0Dl2mo[iH*j4k'\6BZ%kN`cD5S4C'0B3M'"D/@sV=b$U,@S=nLcH/*RZj5^q +q8o$o=a8tCJk7R4M$!V$0`h'%W-Kbuh>XOD*-2V1NluPp.!@'hFf[':W@RSA/9S/! +nZ&V]3)f(j=RL[4+S8tpCV?B.OuWm1fANRc+0$T3"N%o%H!$4j%cJMfs(V!.aM4:P +"BPlc^hqF.I9ApC!r+u5fQpjTH;^sRYobOTdja*seLCRu9C7"2XKaRC+.LB4m/hU" +^Br"(R/`jHKdD5MP6:(a_*i/9dF_gK$4-$9e#rBa;E++Ik"Ir>d!q2>4^@#SNe_jq>J/Rk`P#/>q6=F@ +ZhcP[k3#0VQWHs@//m+HX=pM8S!ED#[d5W>'#=))VSu]#ogO^cPLKWL?gQ:JAUb?p +p*4()1?QeedV3uGW@FQ1jaU=p`a6eha2Ws;fSN@ap)1,@(fe%2\ +;*-XNm$hNDQa^8T-&soa?6/olh>9BAqjr#]NZE0+2O.R[`2-tmZ$jfp7HAH(_r\2V +ZYdm^He#E5q;dN0X'7*Of/G6uY$,+7Ys,d1H%+W0b*($F-/5c4(=a(q7DSs&j*B,9 +-%,T9n:`l\1M`"F$K&I'UV"?j4$gP8n#fBOelA3:9o!>GpRT"5LPN.0q^E2?eUbht +Uj5MtRj%OB"iOi_U@;K#gsBBYS;&^$r-q%9g*+^k"eb+)$%j/O@sPqq0YA3E5n(?6 +iHIfUIpWqAT)$7hhu\FQH2;#jb(qG`4l02:NC-hLD[,6PO-RZp@8@6'Z;>-q7cojt +/kQs_NOZ_?gbBo'%bLB@#H4JKC_)ch.`,[QEC,U'?HX&8"eDnhW:!)q3:UK +eF)#>#?-IBj0@Fu"oS!SlhF=cemki%P`0;kF%VNO0VVu,*U4Pdo6f" +g*DDf(f!#KTc$7JEe/&/^]Lh6,>%_D"Pkao3].I@5^%[BW>D(u)[$mAlES.cj+0cL +!g*j1$cJ9W/0sV-BZ>mJ,(-8%,1A&l!e +WL&)$.ZmSZM_64YQ":-gn;.)J7)XR#/cS-4RJrfoIlQ[`)#s2+]KS(McRS^6$c'iE +l!gL@KaLW""i(2b(V;WqgqX!VfO+a*Pr+8:$7Z&`ZVd2g:_Cg,8$E%&i8C`1,%)%" +(do;iad,]-@?Kq:g4R9oS/FIeT,N-2')!V7oJ9\'kH(Vco+R0#i:IA4#"96T6ABad +gR?BmA?r;(9PZ!(Q;4oc;QM9,`osY+39N$:Gfr0T9jCabqZ%n:4`KF1o`Qg +DS]dcj`@PK>l2B)kFtk@P&cnhEjrV8pY#t`WZ'LS)NW@T4q4pVNRar\8hqc-(Uc)R +28gAKjJXbH#lZPO;UK/9#lb80%(6=1S[Eu2q,QT1BdimBjsk-=^g,^Ah`/ucq4In` +Jk`%IrY9)-DT.i/6BH*CV^5BS5PJ+Vq[AqMS:u^Uo)%4bH8[0#n74bJW4jobIp])N^0L?F7/h%s +j:+oWY"4FM!C)l"J#V6&^0()%V<2^H]0DR[^+WN9?_dmS-r@Wd9>2oUrEQW!F+rbs +$ME\?nH:e!&3$U!I]2si +!3(+qrN=#:6gf1<>X7Gu\m]l$c0?dO>Ns\3o'VaR[;OW&><(:iD,A,/D%Jcc6+1,Y +Bim,N,I5(imFComs46@d[MK'IN?[,X!.T]2:%UZ1hrc2WIh.Z +"4(ZYc`?sr/%dJ`g`!"!B,'c8AR;jqTm**I5$W;L'^/PRN9\t'.jpA>%=NesqO4dg +bLn^$R6SMo4O2e;/6O[f[!iluTjq1!`C[4mETLh>/!9"'r]fo; +hT$2U"qDq:q'`Mmk!Y&VkOd]2=: +M0G:InA0!Uhd0F"h#G+8j7^(?N=DBC;RbD9O[IGmV_gtYh.@Tj]n`3\$NTr(dRBim +dYL'eolg0W=Ml1o(0giSJSiho133LKk,4qnNKU7>otWYihOSFgSH@\1:_Wogd4V*! +E>/c[p`Jo5SuOUH59hgS_nm;):.l7PKEcs6DJW2\s87 +2A5E`EC6b]WND+Ln4nF1mo%$?`InK^R1E@Tm]7#gXV(G2nO&56"#C6Hdto)dWU5UX +IJi_-TJaTZ39Tg4p5"5][$a#R33g0WeEI/f7[XRfa>BH#P/K)8B:6[>r4rFO7^4cp +D,F[X%0HNA)ao5\&G:Ol@OFLI2879a\'UY^4Gg6LA)(N-?(_kSLg_UJi*o#D5upN" +.2[\%-B/!#g-PdB!:G@-5O]bt+.PrdV5(5P)X]j+L@0YsM6U0a%IE>.WV>th7OCVITCJqr)k:eKT/.2[:MSZTqe#h#0c5[4Ns#EJG +YQ21281)h;1KsKDHB&l)T93d3UfsEl%h))h$WRaCSo%neI;tAo^qZu!UUo7HFG4@5 +Q"s*R922O$I@Y;)]mE,;qlTFB;>M0/1pJAl+>IaBY+IJa"?.Jr*LCD6A']O"X1+U1 +OG^W^bM'b'm63hp?OX^TnmdUX=H!/Sh&bDg8`9jmi!]9t$H+Uh[>2^\BYs4%[X\n( +B5TijD!f[&j?Q8UR[UYdIu5Rk)0l7(F.Y(8Zk)Y-&Eaem1RcP,6fQ70_WnJOARI^= +Id-uBs$43Go:3>g,&?RGc3ArLZn0N=^#]OWE40(,f!//hhm0^6hICd8DZ^H2*_-.`f%T<55_2?`ID0C"Y]jQUNt0!4j@.i^s>2=T/fSk]H-ub +4/dW^Y^cOLmuH]Si4mL0?0^&p[(X"^FUbXWd]7.&d\^rLd%u$0]Q)NtbgOW)6G"ST +9?W4Wq*jB]p*%fPkaHf'>4q/F[AT$k&9iSFT%I@FN3[bS"n4^'/'SP#OB9o&%J +Ul3d2FmEgY[MN!3cE5c,KQt7>KoYBB=k:o%f?RQ$$le@'*i0A2F6BXjq=YV*SZXU" +2m#/#bC$pG6664.`hu[AK%S^(*DEJiS^7.-25b,_C!;J+'lrQUb +5SYE$I\]^Vhnr5Wr`9c1eE!#\b+\r6S340Dn-EJcOF/RgB?d=^/7@;GCFe%rEBW^"hdniK;`GIeU1d +4rh:-TH[s3qQNedY56&X9@5!n[3W!=?dK +i3&dl@UEI#;Glg]?00O3jaF_[oh<&'ID,C> +HG;6F*bQBrDH97<0l*+qI1"u@Ou/pXccYdA^jj0!A-CV+]kG;2g7O#[\NeOa#Car6I%:"$?cBn[;^Qn&M>!'0 +=.l2d!U%ju1>8as_H3s#PcP<'91/oG=In"n>pal'qP +UR'*%1NWEjI`;HQ+ER&jC#p'uJ8N7kVmi#pZu04nq)2TYD#ko,_aF%1^+3360^N:> +OobQDs&"!M+t8s-UFcKHo`4o^ZkFS`rZK;^ms*9!H.!4"#f*pL[f9BXnEZKp^RdI\ +rPI%`DL)K_lcg5q[(Q4.jFlV.^olns!$VAOmB,$kF49IBXM+'eu8/OG2V8" +k^.U0j+>eBK#[,Bb5$eAp7g!C8a4]O(jYRq<6pHP)/[j5\da;W7K%gP?-:sM]TqLs +i;]MmDuM_?cj#:=3$j*PrVq/jENrK1ZI=Ylm&oF-e'AfSo"htQ6!.j +1:LI;Y?!7THO`18^]IIFX]!1TV2+c/Y59fMGFQ[[%4jd4gEpsQ[cMUV_u::a?d&HW +bB?sQD4:[R[u_!MS6N?`4(DF9](7NG$*LC,'^Wf"m\NYYr:`a'qh/*#`7,)+nJ3#]lV=cDch4KPiSqufZ*8*@mEoU$4eNTU`pYAS/ +E'H/FGA08+p.STc#3:&Ws)@=a^=[LZalc1B0$Y'`f8'q`jN$bjXid&5jA(VRa#:4. +f<[tKahqmt/JIuI1EL"/8FK^"pIqB+Fp4EsB*VGP7gn7D-F/Wj+JQ2:',W'9<9#s, +8UR+$>cpp4.*":ogZX.R!0jH)#"gf[b:j?[CRRFb.bkkq-Q@dd)2'@#"ioj2`TdY: +8iYduU?e"l!d1$Yr?dd&9c32@'ZVV^B:i-9Lau7p.jHC*kJS,K6@$*!141X]nl,$* +@9!:TitQ)+f_Q*ppM%%Umn_BGU?+Td%/Un;$^/1gQ`GNu=hr!3O2_fnF%'tC!h=fS +JE:%VFK/b1K*@M&fmsl+T3GR/[,kJ:t*dl,Pe0I-S0`;!-O_t\/$UkO:>q'7QH!oX*8 +s(n\]RqBLanfP\4hnH&9WP$5.W$4O9Sg4+AU@gj/@p +kM2Nl@YXJiiE:=anKA(pY+/qDbFX),T9ke54sd`":N#^s4[V. +HgS.imtqg0H@S3cIU)1Bn;83T**Zp"q]7s*jXo +YLfEH=6ob`e+VUI2SK)_c1W+O;f\43P*]L;QX6K6n&::T6k&I9%IOAlX*l'<]^@!C +]69%u>?1g>7oO[AppO'qUso[s!UTla$2[alD.kX*Uc"27;Vl4jTugf8FC1 +7YU.CV<`8XisiQTeSWWEQ1+CDO^/28*V2ac>K1epj4!>jeW\lE8%p6C*Q_U9/J&a5 +Y8A!-)&-p)eZ9Zj4Zl1rXlIORZ/C%ARBL,IjcFIImQ4)[fa5>BpUl>`V-Is-HJdQP +m8WsJ$.-*IH#La:pX&ZnY'YrZYuBc8Xn:gTG]tS64/KZn!m_fi*_fP)AuTFLcim/: +?*h/SB%I1Lk1a.uFD1dTCF?YL:7>1d3&U8tcOC^*4OPa-I'S(Kor?*2%Ir7;_$4k0Qh=` +lL&J%#A?-qp3Q'G@iNH]\F.DbD'S#aO+OP-pft>lfl5j7 +lsRV_?LMq[TspoY4!1BWl:n?qNQU]^.dP!CIuJ/Ia`o3\N1j/S=lWu6\;3 +d>`fC\O@ALjG[p;Nc&A<04d>;caZ31IV1LJ[55>,_4FM=RSdgW@iNB3JI@hl15t5o +ra5alI]rg0EGVca'&8!7dO9id&kt3?,T&u`NM>p^;&P*#35#Tf5pSks'`IMHrU9sg +0Mn9*8?@8ZN*]>>O\^T.WRPF/jJ1\X'Cj%2.90EY.bD&=0O/He.KI56(fHS3;FD!D +_?ZD>_BJr3$NDBq!eC<<"e`hamJbH3)um<.[a\)K"Kfd9s))1YWZ=oO4RPg"< +bFhtdF)CKR/?*1'4H]l(RUm!,B4dWcfX[:f/cPkN+>2-SOCMA9Fb&sg;=%Chk#>RH +>$^/g=*s4KSR5m?kCp8!QQ#e*S.33bk]cI-kH4Tj&RO9C\Q>.Q/g3nT'd-7AJR>M7 +p7n2DKuSHg1T/%jA/R2obZQ6#?B`Xk"9.Hb^`gf-SoV92YW%]["gn#; +PP@oZI#RmI17,5IqbJs`%l4um':7V;TJleQ(n5GGBT`bI)"APA/2m9K=8GbJ$fCrOQD7a&=]3JCHhGcKo/"LSGGdSs4Y<@m3 +fFMk(1p%I\eVVNh2@O=I^L7Y&6(U8,"o(_aX&b0PpdSg6-M?]E[0+IUcft5oFKG&J +ZmsSQD:B)^iB$^Q_[#`14(2D5^Y;S8m.oFd(>?CdIp;;g>N=8?mG2u2Wc-duHe5Z$ +j4^:%I%^2F/+^A)p:DXECYn]p.J@YEG]nnuD[W,j"V,ANLDGF6Wr7(SbuN^%VVA41 +kA,JB9@HpSVQK-]Xm-*!M[,56VJT'9]P`W)g=JK*5m\\+lBI&e<(.6E@g5d@2ot]D +pBC!PRkCf^/hL!$Hb8gC:l'Z++D&h\adgoL$Q96KX&IAM>sV`!-tPY2RT'Ce8X"L( +1],ksR`n@Qm3,mT$A%$/#t=It'EW[VQ6j$&Z&/ri]4pjE-@#li]F[VIe +23`ogNKCbVi=A?(I@FogG+Ua6:n0Wn`=:#=8&,;4;d`ulX3#e]0mDeJ>38.T(;nm. +ipLWYoO-L(,igZ\@0in+\YM;#Z+#M*DYEp6^?-@)=bh'(hl4e['5n<= +cF"tZ\#/\!;bQPTS/G+Ut3(@nrdHm;aoZ;Co@5"(+ES.j>.agNhDZ7+!Z=b&HT:H?^\rL+Jk +,pNt05L38`iq6HGk@[btcbKUL4us+%*Ce+_oZ5AC8OjU8]$)]=-3^]g'q:'j#W&fh +l.Iu%/1[q)&DStG/Qk)7X;hW+Im2DG9=*X+/BK@)Q\ae1'o=cgLaeOQtCH2p$`$m]l#4fMg?+\iIAPA=N$5Oa[Ro!Zi +"90,#1WkYZmTp)$UB&lBH,:get,c?T@5`*ZT!Hf +fT9HVrn%$fjbG>Y]*dZ+,10%?D?,Y>6P#gt`36]1fD/m_W`(>GTgT(Fs1)SDP5kC\ +fk\#-Hj%NOScr)lJc%`6Du[4g(EMX$J,&npVuO9t!bM@(LO[)8lp#B]AY9":o&ak5 +Hg_?\c`*t,r*SQg!;mAUB(hB8pCjIui[/iD`jS6i[Y+Fc"8dWKC9K]cLKaiWRumRB +6Mbmnqc*Z10("s?gT'WuaSuVA%Z&Sqm5HT"Y$5-Cg>5f=ZqFZNd9"ilhXTeGRY/oFgQ#:GR[gpLB +$D6HnZ$-_q3g\(V6mq:hVIp#9oa?(7P=OlDa66_sV(/Opb=3e9#/L6p!n5W9-h8nR +p_W-rZbE[+I'` +XjDi=DQP&/WB7_3,>W:-hu`+^9\(Dt!0A:MERr]gApS:6SrmM=ppLo=h!IjstD8AU]lo4GH>kBDiMp58\#79YtXa9lP2 +kehI%4_#AsD8C@;2Uiq/1nIi*n/q3cdj+rWW&7AjB`ESD5M\/QC7LBd+&C:BbPTmK +=$JF[+>M%^%ZDN95<;M'$46gUMa2NJf5:>,'JH*BLliYC+ncfK; +1-Ytm?@iD@hAtW@`[4I!pEr5*A@YCJ.=I/1KrQm +o[FN_U9QW2>Nb2&kNCR(a>R6$>>rYb!pJ=g!.Xbr$NCejN;l"$;BaMdQjru&.kS'/ +/p=3IF0Deqpd0=TDk; +_E[N3g\_BI(FP3Z2Cnu]g0=a6Q8j/PXs/3a^Xf+$H:aLZkF09AbN9'Nh5=+SNJu`X5*Sf`U8f+.ej,X +=9+`*pVl]q.l>#lliG`>rkG#\qr`ZG^L&tukLJbqFoBi4f:E*Bs4R;go]t&*J#gr/ +is=PhemIO@^<,,SPI#G"JOW6!,Y`-r:[H(slpZtN&Hh`g2SWNa[ ++pNVuMGs8f?Ca?r/]V&qG-u[pbc,a.juQ&qfV3>q%<`k:Bp6-:86fBGG'a$)8rkU$ +q]1sB>r?AI"r;_i=894`,C2WAMe_Eu[E!-DbO'/ec@+B$tQI*)[g,3MHZY<=DAR*!D?T5/m#lL;Tb(0CC2m3Dacfc+IS",UC?$ra"](C8e +ToL/bF"?Zk?^9j-:5>@mcPk][%;a)@f7_u9$$`]AGQ18(KDhGqQ-\h],,s +XOpkp38`G1#QD2jICgbRSP)_E(7cAAD/2Yc9#1R@CsGBHht4LnlIrQ+dN\>JJgpbk +fSptR-',$(`mop@P)p`9kA@.!,eeJGUPJPA-?dk9[^AhVp.2MOKP>#?-oP[CQI%!T +lkdW':o00<&LM_l'<60AF5D$6FVrkj*L%h0MA&IfUANM.mda00T(fn.i-BCg^J,Vl +Ilhpp)`OWb)aW-us&_m`'IF9H;hd_4K3N>/r$gjD@iCN(.P\ +Na;`bTUsT*"RFED^S[hIB4Mo9aC[\^/d8WhAcE_LXgJ#8dIQ05j1j.H/.3UVg_p,"%=or*T+8Ba&-VpGdMU\e"qUW=c0^Z*-=usDC$*?G +dKg-R01eY2(FBR6P/ktboCrnhnmct!PUpYY'`og?+EdPY%Y5.6!#UJ:^Q/Df/cb;" +X^PM%iC9*WQfmqiZo(S7Wf4S5,L15j/;"*km/ZWGV$)>[bZCUbWc3%6R>X0%V'eiU +ca:`8?Y[W"Tba3`9.rgd069)/0LXg(-*N[[[..JGk?"ae=#@;1#Le$["OISZDF:"F +=0-XH.".O;Hr2GAIN`9:PtD>ruYP8iRR(YXaSAf1,G>/&gq\fQ/6jfmtE6BkTG3?`AY5WD2RSk +LlU[AW[&hml6*+='0ArEBBGGfIK.0nDoq_6be3CJf6Wa-,_V-Dhf\=umC+MoaT''U +IkTMABD@].s&]$qs1\7"^[CFjr;UE[oDedQjaU8nHXFMcU1!#Z(`=7]$(?,\.XoKD +5(l7W5.S_&KdoQM+5%iuD%joG*doN9::4"^n6^44$?3]?g5/k7HZY\O^M6WZHMoSX +ZYe12obtsJq[i`AQQPLo)mn1UE(=o]HIKH+BiVK_*A-g_rp/t+9uAM&S(:<'Fq#D! +FYi2Q7ZM+1c9C*tnOioF +FLu+I+_8[0(2it+!ro_l](Aa7>L#sTT&9cH)Q,3_kO=LUhk#>pCQI%Z/hK,RV@BS, +[8X9OX\\38[HZPSiRXuMDVT?:dC;&I?.q7(3oj=9&Gq%YhuZPQ.V71R)ii/^cE._K +C73!"AXe&<(1h"=F8EUH*B90AFm7J_u&-p]k +>m_P#,6k7dp$^AW7oen1IF&,<&-t`Z$G^!rFKZ?=2<8'a<,n*4o=)KS,M +E.N:DlOpmH!GVa,q[BTLK^bdk>h7kXZft)*mBP`p0@=e9/H941U4 +J%[-$C15>or0XO1d8t@_0'>]+k3st._ldVA0'2RTjD`FV"eJ>!*(qH'Q%]$UK(g'9 +Y"'ZVN^VNeBQ&ad>CFf+3XYcinSBmA4&!-d15I?^UTqkj^@mMT?pBb3N$9Eb*+FI;=BT^W>@Ls(C+Mt)q&oC5]@">L]d#JOJ99=2l5[`q@P +aNh>"OE[f`hsts)9mM@c=@g_AJ#%psW1p73$6h9-oM%??* +IC7#5d+M^m*)c9b0lUkdE)t#K2MB=lLX0lBcY`p3*/6#Xi3"%(/FG=WFTdN=+p/.& +;+hl46L#sC*FmM7J7`r5K1lZ%Ju6#foB)m=cP'kCPNC*ndcZIMf;Wf;1Vr@_,l*39 +`2Tc/dPYr!Zu8i"cNV,aN>DoD_\f@uCr\Z$rut3q[WEfEIni:."+SSHs-@Hq-!U*r +6Oe@*U3r@W7gju6:ACP/sCG +7C2]6PL?F)jk?^(m*U'Qe&Ho8CJIS8@cc0:d-^nS1O/Y3HhM:/\*>Oa[V_IWS2q)HSXX:[DV)IL +O4:LQ_n/S)gUlFuVK?MW!V9BpC0)s%QE=o_cHSu-N[=QM$b!H9%Wu7Ha7E>m-8+*= +q+KkknN=1Pf]M_89;BG!G1jTAeZD.*djE+YDijj=bY]WHg'A,$Y[UhGk0OB7H[m'h +U5pP(`f6M1/^B$8JbJ@Y8O>RU$uH"ipF?N_8j]/92N6c+!iK&bEA_Aq_k)A[=b?#u +T*5At9ig%RQJJ%+hJM`e1#]f_"5g4)p+#8mFPn*b(lEof8tV'h[>Jgfi4@Kq]rUo\ +YtrQK?T\#gr"M#*#_&Kl80e]c;U(G(u*?.h9K50 +L_D],.e9c4V4J,g3R8;-V8o)L7\@YPU<_[^d:\7iVod>a:j!=nM@tE%G/lVRSQ[1I +<"1T)&(@RLBXOVBqL6Y1dfkuOE&K5,jqpmAju78!WQ[* +(ZWCd-L`Yhrt?V1,t)#=!WSAC/cl"$"a1b4f%Cq,g"G!JF0Cnj=kq0+/^JVl06F-@ +E<-!q^[:rtbdX:-hOYJ5SK>+46DqUJBS/)*Ln$5Pei]-X=38-rqRD24,kFN1>*QbW +1>b:#J(GmPGE"Yt_4K3\c"hsN#4;P]!XN!aL-qc/1S>E&kJ.7E\8so^6dX&##W`cJ +-XpMF^TbTjULcOgo:M?:S,'68:M:"Y>4NLq&u9)5_97!g(3Xsr#Q3]^Lugf!'(Gp$ +P2]Vr%,_X3!!3OQOX()<0Q5cnOXP^AL][U;Bkk.AO'ge&PT7]Em74:1T#Ih"Ap)Yd +K,rkWO+5@$!WV%iC#>egJG_&'CZ`V9nE%uaVG1K4W`T'rQKDJ5m]T&@s*j:l5K`a# +j3HV5Qj.s;o)TJcs*^jfi;^5^r??<2-[p%oD!iudksb4ol4F[RIMsb$#b6M[^d+C9 +OiB5f@@dF[_0]SEX\WlmWO#(`2tLDFf;J<+;7a0Wll.5$FT?]hA#mb\;0qo_lD7#: +8B`C`;6/d^i:if=^(h]'7WD;ED2c0JsE)Y>iae03Q%fV7*G^Z>\,n +21FQ"_6R'VgT#=3=$&A*S=ZfD4BhJp49uqX\igXTQ(^0WB+5l]>VPhFguoR5^ZYXP +Q`1A5Z6a,kk>/"br0a+.Z&QS)8i_2U&Tb\].rRGSHCmF5+!TF4%di0YG5[*KDI'mB +rHfRf3gUn_GQ.\XA..#$j$0p:1LP=hgp1cOg:$2c9V96>*QYmE8"I(Hr-C2K^$r)T +FRN+QFDF%N+S.6h<_MfF!WEFeFoXL(^oC>nMGsJJD9\gJ6a2*W+*+:*QRuOU/$Z/. +n7[rr2(^o@Y.-=hMBR#@W.P;_aM8eE;b(rJ_+gQioDo/fp%NH"/+tFK'jkm5c8uOD'C:k;1H;i1=1TS,5&(Fre4BcoLWjH#9G$eKT +3E&g9;u1b<:]"Y&QIT(,pZ_An4pQLSb7fKi!kRV +kS&B\0BO6,jt$i6++'0>E/L'.L4.r;UVjiUK_\$SXIl.E:=5Os1X+EAFpiO\-g>LR +j),gJ]YZ,>g/(:3l`s=.$j:3 +WgrdV-P2!,=:hdcb\33kEW5DEPT1;cc_EXa +7_2tO*&2h&$l\E?_*hK,KE%T@.nllNid[=/-HbqdB\tIPY^_TTrAgHp$nNMRkX9gf +a[kjaLf!]K:GhY^(O!V'J+FEMT+lEN&:H8fp7cj32a-"Q,KV,[rH_5ie_KMl7G4!' +3VKq&ST![Oh(E^r015Z@/G#kn+uSNd[:f$'U0O@7/^h5X8cr+\N8Cg3mTu^uaVg:? +atWbbhB;NmaCl]0M:B)[l-QYH[Xo:'9WR]m6H$LpM#U9QC^,p#K-tfs)+aYe:&uJS +kj13O(1-SV"s'3,1.$.$Z@^A7TM9?%VDWU\Q711D-e.3sLrTao92Bg`gEBB]J6=Gk +Ke5KJ7YCM%Dnn+>*j+(R?[\KVraYd/=9(JY3Eu)-;hOp;!kf9.cNOYHqbL^5#LP;: +pp(e:nE#Q2`N)DgO/X(:'%-4Z!;9bU[D78?!?gPmaG84%AN\Zbn2f70a+"MhdcIOHE&/0CQ-JKHR$7#:l5EP6&o0M'G07?(Ri,>\Yr.A6(Ae3PBj=N(sT +q*@X,cmHCQB3\"KPi@$4`lkSlBSP->Fht2"hPkDrG[rTE$@u&sjL?[SI$reR6Z*\8 +Jd'YU"UmKM(V*Ktip+LihJ$HR4LmoEEi\3SY_"O]8Jo15=1BpR.%c"s!11lL"oK:I_QSFV]%gT=n\pK>Turh6;tlsqRp +SOpra:t""k,c5c0dj(IK$Cm/c3.[ggNmVsHBWC+]jR>s&rLNr0%Zi(KTD0KuZ28&4 +&GSK++38ATIg"'^G,c92^FIR?1=+oc[dDm%Rld%hGLETCP&H^[!+Z,,#rbK9d9J@R +0Ch_aeCB/,/&;1dO>@II'8TuraWoW&R,O*mQAiSYkEOqQNJ#gB]Fs.b6HB@Gq +P>,c8iJb)]VOFtB:h!JIT\)*OA#TuVkXmIs;q#Kpa0b'[D +rCd.@0880L$atN0ebrNbr;4b;Ik*>S;R-Jqs.>S&rl9Yh^Qe$2J5>n8:V!Lg.5GE* +i_I_GMkmaf"NhPM!VhRK,F8&5$]!]>Xh\@qfLR(8HOm(2HF0hCmF_U;?[D,0lBTE*H+dFkf!T;BV+]k&Y'!$^>!VE+ +%e/)\G3Ren!WFDRkg6n`XBF'Ts4;2=CcaQ@h:i?uXTk8,K2!kr3:+4uFrf5)UuA+ +<=\L)HOgB,050DCKZRNdkcXXk=Fef,#G$;.e?\2fB-[q#cl)`Zli(QbQBj4t_[S*/ +\i%2g[8.lLjmF.)DX@f"DW!b%n5,O!G:L6eTF-cAf<70`%\G3<5ES0W\QilVX)r8\ +\2UV%2b?-#+GW;jYYOP;4E$,LUH8KT5^W8lbAPQ;gF!@&2DrdeaX^c9^mj#6&FW(@ +CkYWZ:@8NhH`mF3hR#RhPp01hqq!8!C??q[ItKqt_&NJSfOTt4LclA;2rsd7D!)_> +HeG`:g7`-$^3\Y<@F;DalJ&!V$ILI.Cc@S(E\&-u5mp'eh(hms)S$B;!"+8pNs;fg +\1+?Ka]UqZ8%9ds +g[4Qi^'W$\R[N_,iV%t]=^IjIB5f?fa[!id@:<$ZXj#(JUrJ1AQKk93)?He3-De%0 +^$B[LYVochQe54akcRMFY0Q9Y>h&fKk$V32E+T#6A^8Z?4LfQ9UihnK[MB/fF5lI_ +@d\r*\kuDXe0oXOT#5:`2FYf@9Fi<9Wb#*"efTVXD_BhWm]':Zr0&hr)SkUqt^UF>h51l"W%I3d>dMLlO6 +Fu1]fh6Hu]`DGo2s)265iA&;&4&@K]()0-A1$`??[g)X!nQ$rf)\qTqQ+9Xiq2Y2j +m-\Fhhp&C,8@q`jC2kW?ikm1+^qsmg`GmhoU+[qM#FSAY::raYp,"CXLL),9Il +BjWqqPY'928!*u-*JHg&bSp!nNW4N*$#mh!!Uu(G5#;F\.0fNX/EppNj1FNJPG";9 +1U4c/j$M&1LRsRi>lt!QnujD20:OO9+mP45qQecUdUKDUa'D80]Teh13BR1SI_&6^ +WdSlE@nEtS;2SpNg6I9^MhNUH99UqX;VWbNI"t'29SBduY^.W&%HBWr6lu*brS,q< +'C40,F_[8#f!@X`:lpE5E5Gc8ep1N4N7\["#7^-T9S7G\BR#"[HJ'.KKR.R)Nq"hl +/fA,!Fp2YW;`C::8>#H!qn-KMJCqZ)fk/JL"=t]"YNuKGmfkBFq=Od[L>W5+m!Fj- +o/_9/Jq2gqHo(RArr`(H8lqG052:N(-UEVr3K.-j(:/lJKbu8SrubJHQLQS5q`8En +hSHk;#KZqs]WM2^hYUO!_-ceXhIZ!15'R=J+0bc/?2\i5lEt=Nn?3""!;rJ[FeE*a +Fo_DOrV[KmdcZ*[4U2>pnfK,/+""2(9ToSV&)pX`A!bFF8%abG4V(aZi6E:8p'I,h +qg02(0iBA.IR@>2rAp8dPs^>T^2;]&;scNJ7376:Do<1OrVG)N>@ihS!.UPPpU5:L +dAqK/VIAj]*P`a_hQ9l>hQ(\a#&=2HJGbJ4n(OU\l%JO:nCP0o.p^l&RK3>F0A^'H +;Q6p>g2^rWHZY_PDm\ueEpVB2GA/'Z\(u??I"aZM*SKs,\W-$M^5<4^PXEVJ,4!?N2bZXWsG%)!g6-kCET3 +)!f6*\_<.j_oR[ed.(DmfJD+>+aJIUIq6`D;!Zg?TASkg^P9!Ra0rX9GC5.( +?:(T1TV?uJWV>Z)iItI`J/Yh,62btLUlc,$5.."Zj57&Z%`gan5]B6XgT/1#(oS[X +1RDI0goGkV\O#B2]])mLFM;k\Gi"T`V;/%fc+/;@VgT65-1P`)5`STO+JXZ])V,bP +>K[fH39-M#@C^Hc`2RaqV.n2P'>=V9"98+i(;U9b\MLgi#rQ@31LP#T#d'5Q15FG; +6:h"3!l4h1eu5)n#h"H$lFGBD5S:L+?sH=j6,WT6m6LYRV`JdJ3QG==^5b=S-4&#p +dU''>PYlHc]M&g,6:6OLXV/3X]pcn/BmlJ;E,@,d%N/KLi/Ymh,AXq(H.]&RDLOBN +C5LIC2>`@4PAuY`$2;S6kLsYEHIOn_?T1QXl#\M)E&"ZV)eSXQ6Im5]2-/C_oS4[( +LEhCtUH,ePAS%DI3?@kB>:Kg$eHmh8M;SR)O-4j*P'^qQHYR$$C198?OufX"HQM;@ +<+Yg[cg$"EeajocqgN9uASTPDPg6mt*T[Tu[hHAl-$PY#[cbUgVT.-)8(L0#X82\Y +Y_X2m8e!9W<6J#@0Ep;$S3P!]B"#o#gIMBp([qZrA@mplh&3&+Z.SI([TBuXS^G(17.XBpNs@TFe=c +cNQZZJ;/\cYEVF^.ct*^#sOX"r'Tf1,9f2$X9YcLpJ+3JI!r!=0J0=asa>=9D +3Bm$)7glhi*ciV:>Thul7cL8Gj=<5^(3p3eH#_EGdFWGL0'1*Eabr6G)ga>!jr>'g +AI5f9r59cHSLJ`;R,#AGJ:DfmOuVV^=8FoiGg!lY.hZk+Wb]G8/fc!N@Le:>-PJ*IhgVEJ0kRSIIF,S,P5c9p)O +W/m:=Z[MUAk'$?8I3'p90koZBU@K\\nSZ*P)?)4;qsUHifA>BlN'HKC^!4p<9&;6T +qgXla[L&>Pp1rRKBEA,45AmtrrO/)LC,s&PkDSYZ>] +5>>=,Q7RHP%&F1gdHYKJRT=t)O8P9\T\5m/C#Hag-Uq;QK'%0b@@^IaK\g#/IZ6]" +KR]!M^c$&XaIY8PL-]S3*h\VETg@_l[I#J4X+21Y,r]SjXQn1k!V\d[7irEQC3P@* +9rF9dLK;Lo#*3)tNrT=8pZPI8[D@K37TH@lASEq@GpY7Kq%?`jhQAfaeH?+)5D\=)Yomfs)g)M +s/#2U!<0%VjK*`q[sGVfEhQTW6?)BA/e__"r&9]:'RbgBV:@\dRsGi-q_Vj[O>HQo +0ti>Y!U@/-_#">Prf#]dH\<7]G'qJ_'O8H;2lLrde2XgKS6M9(brX`'i4k;4p=k0& +BZVK6r8<=Ecp/HNi%+d&jPJ]JcVA_XnEGKc>I78<:T)&G8`J,Ao4jb=W#e0-13U)C +/7?AjBu?uG6g3'`KM-4UB[mQpgDo?![-N;jNDQPXN,MQ_TT3,nWaBj^>k@F81!tT& +,B%`UCt9.]X67gn_Ikep-HsZo<'_0il@=I;"DC/9=#,Le_%Rc=@rg-3]\"&M"s]4- +&]plF?3p7djMe821QJl,2@jG:+2`k5qKKiTQ$TmAj*N^6"6f)bdp#Dch.tMH\1[Wc +!<;50)X"A";L4BdBFl3?cP9$(LB7CCa;OW;,8?>F-N._WrG(K.Wi%s#M`e"pI;6b- +[?K+/1")U6cb*#nSE^K=Q)>75ruegu+f3GeCm=[%5.Pj:UiNR'G44h2[X4u*%hi.h +-+,s2l?ScF>:&K](uP9?[d-eFDeb-^S`);Y%m\fg_(8SPA!DQcTIY=N=ZO(g)) +C9EA//.4fp$*Q#Ae`DM9A-Q,3YjVmc[j(rIH)qW2Z\6LO%\aaCdL%su*&]7(M9fT+ +V_GcI(h`semF>e$7l%1[W:$o)k8:K,nC^Ka\:P+XagQT&*?tOPi"%S!T22im-^fDA +'C('c"RotZg$[b1Zi.dpplp1PA.Dmic0I.o_=s>L^*Q?9nPRDH-#p` +:]=QBThupOn;AN!)<$UC#\Qha)^a;)o0hA8Of\#H3@BCXkU8H5XO'C4*;:XZ!88Y/ +IQ;t/`Phd,^jf0!.-=E2gr6Y1oB7*6.3mhE*g3aWSJ +rCIEsLS#jO49kcegEjepH10$-fB.@f`d-1I=k@DNbN!Id6 +:csY>faC1="_J-mS$*"]T_Fk#+?s965>1^$s1dY69GN6C9r\Qqs"[.PJni08_WU)e +*BM<9C_+BI!<7K3eh:Uh5F34;r;Urp#inJX5$M9OV7VeW@$_XW;)$4j!%3>e[T,8Yn1l(?Y+@U);bAI!58Y7edWU)0DUbe=?h32Fl:A&l$+7ZC`Q&D6Xr +h'^M9OlpnD;b#cb/oD)T!r."d-Su44oBFa;qXoeE4\MTF%]ROeHKplT0F5EY%:KOT +!6?g5DdW,$![@[QfOI]piqh`M?!>AnV!CJDVH*(CWFO3:\`2kSjYGF817&3QJ(G0? +6qulNEbg=n^.BD5UH34IoHG\h[@u']gc>.Lli'$tDU%B]:[\c&GsK\B4^>@&1V/B& +p_U*pc`4P)lC\H#+Bg=rU^mO3A`URJqVVQjFIb8tM+A@D%Ml8^[@e&";6Yhl"XK>H +%O"cTOgKX.90NGt#341*(FT-Dk4qELJN,J_lkqUYCf +53hmR0B75W"?+J&*f>e3;nfZW/s"_gmo3UaIrloFODo[npAS#\Q)NnQccr3X],<2c +8SqIa*L2K;%hcMC$+fQDG2A8a[GR$`]:7;ROj[6ie:(WK>T2Epb&8ikfA@";],=dN +)H^8Xn?A#I+HLe$LOM1O;4^(eR7\r#,Se>Y$=`LM;:o]VD\gqX/(cM?m0VDM,r=B9 +)G`5'Ea)?Sd34t>(eoTQDp'M2JUmc_<;UWtN="tFI"ERk&eB!er@@\0T5"D2!:tFI +]HQh[nqZ0KF*.I8!/cA;!D3QJodBNUZ'IZS+J3G#hH*X-cfR]'\Wp]J2oGK@Q??39 +jb"O$#:r&BJBcnLCZ'\2U=Ge;bDLP&o%;,EUFm8Oc=D!V-^2<%oDLH"q.PRA-HH6% +Ck2o#5^U4qs0;9ia4_aS5^RtI&HBXbNps-2Eik?Er`^Bt3sEs+E-jEb']]2'_!e

    OLpP491T;O1dXt;- +"eNdB30k6Q9lA&(TUaA=H/YZYU+]Qe>IDJV#=FCs.1L@nDDA;@%bio%dlckq5OI#U +mTr-9no37Mi=)hp9PmrU\'>3ajcPbl\9X0%Y__!pRTb$2Y3rQqgL1I>p:,AQR?fXG +#b'KW>D=X6N*r!]Zl?Si$q@MZ(bCD'gEC#nn@UK'&X*&XX;&#`]`_.hYJW2Xm@0I( +I$b;%O[@RT%::q4o!!GU?[WY:Ihj=2nA:O9T==o +d(=BO#^cuBgV8thrDmM5JIB-4<<<-W(91)m]^*B5GIa39 +T/)X,j-Cka^IjT]o0m.c+.8JCZ@fBXVd/a8:OE_bYT#(K%@T)i07bm]`ss'eL4T8; +a%i2'E#Um7,E6AM@lj=ZSi!h5h'I,Z!=LUVUV6`Dp19.cR;Lu]9*ar+@fNmoYeb6K +jXl.c!M8+dDF[?S54W(HKk?FsBW4Z94>h?Y_5]p]3/GDWCX3g1+_9q&Q_(,CK6s3- +#)M:<_gAne"(1#qHg?FPP7?EhQb4aL[U\T)cK89JJ*Er\\W`H!?Ug*)&[hC^f'8+/ +Q1#(2^+3C_2Ljh5dE8GoeT7RnB[Nh8oGF6ScZTf.h\$HLQ(*_G,G_M0^f6O*;c5%=hn\O5)f!I]E`1;-JPfY`^8HM/99'70DQh:kpn +?(6L2*HOnA+&UsL?if#@#rQ].>[S40@((S#f2egO2R33g$9PVR8Ol[3D]_K&9ZBnH>S88HbsuaBoS: +:s+,_*;[0!i*ehHIP(_E08E\m8\:Bc.nLm#-5,o[Z%^SSGnJoDI.(mCG=I[,HD"=X +a"*;7P"bU@+M5RU>S@AqEB'juXCF?n#mG[5;^C&D6unbf1.qON]O_r--CeH[Nd!0- +!r"uhG[#:iU]/@=bZ2II"ZMPaR-'Ffbou\2bkq_@o5I_1s%n1Qi.%EG%?B&J"P/tL +hW65YrepK'iMYg\[snG6M#ki[cQgDqp5_:_Ipu5PSb) +j#[t,h*)4Sa+RQ'!j?*cY$jY7TJc`]2JDSE)GL6^Qj_j,*^TTD_>bp>BDIs074>Hl +0H3t$J-El:rZMIBQO(Tf.+Tis6m1\u9[J(VajSSp!o-F'@@'9b+"b:0eF76f&Q_p, +(P)F?_*92aN[;!F9jrF=cM/`=qPR$0k?51<1-gd_A-[Qf(k2:1L5$\Pm_(RfQW:`iEp:%?@GK!eT99!n=>GQUQ +C]OE3V#Ri)p\1u\T*dJ_ABCEU8qI",js]-LD#BYQ#P`ooq7f^*]hMCnHO@P'9[2?> +E^)t)c'\e2Dpu]8K[FZ'aJNa^b-g!^1Ur&ZgEqntdDt3Ki4it/]0ki5WQ4[M@J`"= +3i."Z/(RMIS_J\[&GXZ=$"I/R6&kp&VN2;VX5(>2^decf]Fk2&cDFen4BSq*&mqc( +g"&M&Z/\(/=RPLUMN,(u%^jgmL$9%t,s2,NN +AIO$I)U]b@IY0bU,k[p2f0#OfIN=>A[A#25dh_[(0(g^/:'5'8T/-ch!)_Nf!G_[Z-^DUtKc +Mu`O&oPBZdG"DNlUtBho^2&qfVkOIKW4c"/.-NuD'5E)Y-!\WF5LLai'5]$'RFpS[ +bCTt#KDMC;")DLGhRUIofn#`_ +*;=QPHQ'dbL$"t#FV>;uq-s`VTGY=71sXc"aS*n%+M8$I2W5oc&ADCgEH?%K*EP\P\c1Ka*.7XA/W%^RK$An@o-<6MRAim\tfR'A:Uu;p!oB +q#6]Mnl_nK-MRcli9CRHLKS$[YW2JFQKXUrF[8,/Zm$pDj%i8! +Du"M5s61rMs7=W(k@6k3a+PX*o<*K#IffYf6i\X'LY@q<"eQA0r"-pD90Dmj"Nhf! +8Uj8SI\)EYP(pkb?nlUp_]K%DaC0Rf(NoR_Jn&"!F"f:lRb]gHm[N7D/HiK,0j@_3 +7d!nPK>]-#KM!E`d29Q*P0j^9?O@ZI6qsW?B3!>8RV4Mb,\7KXR7!Hpj4]SDIII[7 +)8H`M*([$kc_LGGr5IKj(%VD8,^D(qXjB'QQi]=U@I\'-Glh%SJd4Pf@/aTanpZ8V +W@68FdG+fOI+6'?#^#X`b@p^@3$n&9KK=B)Y+#sh7pEIN`X.XcYq>l-n6b>)]]Rk@p?VnJ_tTo( +@]#.&^ji1^045l8Y6OuQp?6r%5rp)2g-,^bgq'.t!3+d8!<,bCU-nL=&N=Oq(*<=4 +4S9TFpD<3/-au[A"4HD)?a%JO8`j3W-?d*/'$4r2Xe*5>IdWfmj?:H];`Q)[#c@Ta +Z?!69IeM@Had\I^$A&4--V>1NhcdX>Du!jVc`u-W(ddL"M\OBpbV/=h5BAIV1O^q6 +am4'=cWP#g;.B@L] +b25$[\q;gk@*Q)j_7"O+XVk7AD+MVegZHeu$d^fji(7LA8AQ>OXjIBH"@.V;\1Y.n +QRkPl@sBff%Vl(iS)Mim\4(9$_FB&hQ/&j-l]MlWU\RpIG-QDSP[htJh4n,>-W%$< +I+lC?bf<,T@o3Va2.F%bd]92<.B#C;!MBU$gH5rlq%\4Z,'[Xh2[Y`>L1Ht[/)68P +OPsbGhZC2q*nLhu#I@OPFtEErc24u,iHlF_$_6qdTVgtH\l%%,8SH?Oq16nrEg5EC +T-0ZLD=5\"Wm-TN>fqCq)H'QLqC"S0[NMY05NH6*rG_l*5n49E5FZpWPC>HM$J;OQ +-J]uPF>a]'&WeN&rV +"H"Nd7M-Xb$&I+R1-FD>$q,'oiU'@F"`\N?>^otnIMg@mLNS+PTQ#B+oTVq>/AS@E +]>3kMro(\K)W>&uD\J#9rlBQsNn97E_FXe*68&:aRDA)R#0QrVF"OE_X=J@/JH,@! +mI$gEn.5"A#sD?,E%7?&/=lW3$K5cOs-M+SIggN\)Jk8gcaeW6\r$g)Ni7uHrE>u8 +s&&"4*u+^lD'8j'm"aFW@MXRq%5_AoiBPO-"$R@XrFI2@>97LV-Gp'o +;ggB7fV=k15G\U6Ik1H6j'_c`OP"hrUEjZ'Nmf?cl<4VaQguJ8ftmbV>rT2!Y1(U4 +G\T68#[.Q"?Y&g]0RG)HKCZfT3G2lf5(LK_G=E[3Fp.8'/-G(-a3"ff$_Zmc`SK/] +o,R2)Gj.uA!YjI;r.TdE,p<-Wkg^J(&FY2>M#;5`0Oi/j$h+i9kR-PmYeSLg"G$r2 +T(5t@k4<78+P5,b;/'dm[=L4D#hB"ED_kD^ZIEg!d5[2td3J_l[k-A4/1hP;D_b;\ +$asC)C1$(:JH'Q2N:0@J_#JV4](^oIpGn^-VQDut/3cotn:-e:q.&"PWtagIKG@&W +Y7maN;j9L65nPJ3]=jr7n6t7-r><N\43A"r2U#i2Z8c>PIMjp[<`A+Bl6G+&u0n +bktPkL&g[76D&up+Ejh*]lmIn?OQc:(XV)o>I9M%I.)qoWuR%pH4=\tfbP443X@Tm +dBgh$[*_VJhBi%HF&n?gS/UHPSabcT?a+5N-qFq(]#G_SZ\B1$!Jp])IA$ETXrCVR +SMm-nrrTiI("?1;M9UhoIJhMojFWmX\jAl-!T4c0j?C@IuU +h[ZW0S(+cEl.[NYl\&G,:?YlZp)j_Ci3S)7HB4HC2N:3Yh'ME:H-i3.G\B6N:`2AD +iJOG),Ee/)Q,)5[#20=_2Bg(a/.al*Y\hkA`Yi7a*2?@NfWl@J3ZHk])Up`>XqCK\ +%=:$i-lqn6_XYkH`jEqp=hNKlXGU9VSP7=]N1d3CJ:!-KW\d%f)@^d=KlT83HrGR1 +R=MmeLFnMU.uWu3)84S3TF\ +ZD&c@js<0!A31jS2O%`^((]TF_`<@Z3!V(]mN4dnGPK++Etr*"[I$28MmYHi_1*OC +qHu$d-N24%4rJoWR)s#8r3m"'[:9#8Ye4lWq]:fl?c_/3.upsGQY'6_D>f=k&$kR6 +s&3_ai[r;t(\1\KZRB0.j1e4)IMq>?LM2`%,+LiPn/lla_>h:A+0G?/qnHspae*2E +S&;uYr]Cko6VZQ,Fl<1CPfK,qIc&*(^Z3^:o9nbc""m%H?A])EjF9%UHOYgU"cORi +CK_)aFpXDO4"gk&CP)Sc"R:14_H2T(09Z6o!r3N.0hHg@k5/PlHn^uFQjdn,]&/WD +lk`6X01l;\^<8^j_Br'44BN(2Rq7U4"HG)=NN@0@(BOeRkoJ7&[Q.OC9#-NFG]\Ma +2;qn,i`-A%= +idN`,r'5Tbq#U%Oqa2%gpVRe@Z!W(u:I`O#Ld7ND9Fh.#Add8Q/O8\'06ibT0V-Ob +2YZeuTAa0)s,WQtcqdg0T5lc,F1(Ja_Tj/M$G[S,QEFKKqdQqXkA?@?f3To%-rscIMJH%NEni:X].;oMbKAE1\#ddf8&NZ-\5H#rj +)JS6dUsTY?*iVfE*9',)aF)k/(4B(J)F]hnKV8ji4tU3ai&A%%eh)+51fk>u!?,== +f"2If0fYS@uOX]F=SMFR#1& +4f/)!rNR"ReOKZrO4n1cZ%6]<bCi)ARjX4C:6bUJm&-] +!B"EkN)`XodnrTN%Q9,qAj1iA^rt*II+F[J3[i&CU8_./pU&c:$-Pr/8*d%%c\>f9 +Q8Gg?+8prucps>e&I!4]Y+)MT%E8r=#lIrC5/;[!ndu^i#2cV^NbIW$3`G>s3HOs_ +V`_u'JqDPTXV#p)=/fmp!4K6B)E-B`iFC\W'5\O0-XP-]Ms!CBpS5L<`oX/1mLt)8A13XYD=R_M8LS#S$668O"H:_ +#Mk>.n1WpNQ%]#KLNs\V^%+NQ$A.()!o.iWDWr_8<46iTb:/kDB^3Bh2W;[&mEPPf +_k0T1%jfn_&8a`5Sc9c_+;AT5l=X`Rrs'_H6gQb:qHk!&pT\'">rc:>4MX2Zs&p6: +KE&CL%KV#*rd.eR]hUtY0;O;,.u^pHP@@OWBa#fuUKkb;arLLYF1%Ad1?/a5%LrYXboFYs)"/FiTkqDgb*@O1,W<5=ctTW9djf42AC5]OH;EM6In68&XiQW(aS +Vsl;aL7CsD3$-E@+eE^*!WQ/_ru[DP5"kQ/P2q6eoYi*O@T#nA5Kb"0l"/kh0.+U* +8/_eu:W&EY5(?82TB;KAn-A+/U+,rS(q"FBd5=Yc&5tXQM=&d+bu=U9e(A8p7r?#Y +"43uN2$=?7PYVt2)o2GR3#_'ZKhc?,1D'?U"k%$4$K/\p_4a,Kk(l6F#D4MaGlg$4 +q*c8a\)0$1$fk[K%[hsMq(Vi$YU)R5*d?Qb]j3gnp5ep3`@FbD`[!C;qf@M;]S7)B +oh&*ZR\r*.s%Sdd"T89[++h2bD1UYW3]d*(V];oC"+l"'\dnl5^WKf2Q7=s-XR*nZ +!5@<$.K<^jphMVLmJ'TafS]<$P3bo+m)L@T_#L;X"b_BpWEnBY?C9C3AM<**q[q$J +_!)XHp*ed+n +b("AgqVfL<'kpW7&HV;XE;aIgRlm_@'!AijYISbB#DM)+B`7c:IKOm[&] +/Gh6d%#f$M`hqrmp\Z[,mXEL18+c_H\borl$u2U\;\ee[H60MNG/04eNJ)!jH"<7& +G0$)#naURo"7Q@%I);k"s$84N9((LB(Y"rk2=74Ml8[gY^e"t\F8!,(BtD_]+>%G( +734@r@c%Z_g^Tj+Q`6CJEpA0-?p[F(>!kI.7hrPj)7u>/X/ZVFO4(c^Y'S7;(1S$c +"Gq&te?;Tc(fiFi"%R<]FV\3oePX4#]MRb<"r=H$n-$sKPD%WjCH,hnrZJT>b9ZjK +EcOAY!YG@mPRf9/2`RReE3t!l'0<*&6Um^W&j_M97ZEdf1<',VK:PAfP\e^QP?kY>S@uMk(D:0M_Lntk&GVN +5/At4i.M;^9\;kra#YC%,CPC?+oZX.^:%DM>^o8H^V*4ITMU-=hd\n'ce>JX>C]l7 +(Zpn>pYl/H"[\SjTZIUlG_Z9+)n*3jn=SZgS56eZ&pPumd\8kg+8i=bZUOf8D39T7&0upLU)rOU.k[I)dI?p.#,^5o(P3IaI$QockX3!Ug +n)9mch#r38dp?G65=G?q'oDl9P5'GNq1gZDKa9TPs(YN[2%\bskRHI"!q/T@pM'd( +EY92HIjl]9S:GZCQ>nIEo]K>nQO/+JUj.%HO6a-#dHDcn&;D\.2\hHbcN&9 +R;@`&(C>u^Z*K!Q["-W4?kW5fh@EsZnE2GlW87iK#@ITki<9dsbqm#pm_/)-W`2=o)BTGGEOP#lA?2]8)LAue-E:MYQsrcO7t=f)gb-@C`\TdTnl;nPY,H]_iB +bb$:DN*h%*\FMP!rO4!,hb(B7AYIX<5^q(l[.X1X.`f!c^h)N%YN+4Epl!/==`^ba +eEer&"oN.g3WaicA^;8ckfi3qk02<4`FiV[q`OKug('Q/YU7o=!MG=90E>)QPQk@Q +EPVRbGFocPke%2PN.Ek"5onG&72kIT-@Prb!pt@-l2tmJp[(d(i4F03P<.Kl*]pmj +g*+30fKAs#A&,sB_+Qj(&bs6;@5pu.%RF1)3 +QrqUj"*r +CA.?I7]Vp+Z9[[Zr\"h<-C>g;[&+^e_]l-mM;:?gCRo"F:.;NAjEEH"BAEu/'upjm +j8AIXI#;D2Mk^-#q*gF;Kb5@QH0'g.'_'n!\K7E6:_+[u7Z_sdOCqaY7nS;9'&pZ1 +'%7R@TKC1mdI@UL,c#+*!uh!U4%A.ud$R&C3=P3TK$4b4s1iU?-Qmnj^LJYtCC`'S +5P>'9^\QX/2XteT!:g*,$6m>O8d!qj\V!]I&+[-epfO$]:N?5^l[LOLI5NDM`[ub\ +K60cEs5Q3\i."b0_n?J6Ee/?>mnte*LLd#1@\(kb9rJM)n.TVp_LLK$Uu31^6S(D' +pj_B)B>B"=i3S\obmt#_+*&Cn$2u%MIiD(Wo+t:(5?i'Bh:B!9J'3qLD;%80'LE6g +AnD8#P/Suq[`iP8%[ZuJ9p:=VbY4190sd8sEM,>TgcNOQ1#BJ7G2/U7raX0GYq=.^ +/_e&'R;:&)B?\3I`5=Sa#l`Nr1KJ62r(LpBF8 +M^\l`-1/G[GOYLJZZd&BC=37PXHHmuooS+$+j09%e-8onRj+O<]ZP3cj.Y]6BIaF+ +56cU/69V-^9c(3W<:]Re- +d0D#Zte$GjL4oN.=NFd;W"L&"lV%f!f,VagS#:8!Kn1?28mHr_M4 +1.Qji1Ggqm!1iVRMu_X6GZ/ZgkmSC,'[s-\@#-82c'Zl;9^RST3)]W*8_lX\>>id( +BlKY-[`Fc[ZR'ek#_[gbCa&K*;L=q0*mb:!>c0B_B/Xm?N]]Fr4ou>sr3sVO.X`D[ +^$dtVgU:eU06rFl[VRRuhjN31Q8H*<)OVn2/iFV-@+&-BcRnj!$3Va3YBA]]6r2TJ# +D=G,Lo"M0qpp&kSZKqC_Wa17\\---q(\I+2fP3`T:\]pI`YmJC9Pe?8Y(>SR-5QB?.g#N-?bX@10R%&!S^^jkc#BE?hTV=m6C.6S6P1sBo.Cn+?I[EhMgJ1So>*s2,BP.EO@ +9gDcKj:g$f+fWFOCiM4VBA!Af8uFSM:7om@-DBDbBIkC'ccALE]o3B$_ZSUeBGq2e +4!*l=j:73W0).jPOuqS\BI*8D(3$?@1c:cZX.aX/CB)d!20S=ZRFl1DD/n!ME;nnY +2e-m$AbBf*4XtcEP1$=!WmNGYW?(K?4lH.j(oB0S@1VH3%s@EMb..>[jMt#-Rl9B5 +.cld>Xk(^0[r?XY!jp0(*NNDfmab9?+W!t:)HF.f%BE.Hm$,uRVeBoc?FPgHY$a1a +1)_D\!ON%KXa),(8]GE_:9,6Yf16eP;HbXfFbAoh&B%l^J6*;UGXHpGU(<-74nsBm_4MM)RGr^`*I#/HHKLs[gt2Ab +J$&4Q@]tY8OXsP/@MK6tX8j1#a,P#=F[NFGRpN#G&E8T]j&E>uc +!?s/ucdN&O#D%4Cm1'EYKFlWV^bmM1h>DrDH,L64Ws"QJh5XT_^tUu7f,pk-O?#:eci/k3EY#s)2Uo>uuB$bn"+2)B9m357f&WD;\W +`&Gl'>,4`lisE-20on?sDtSe?R7qLOIhOn^">,DGrsO'$cB(sY2Np-%aY2rW//hkG +7$Df!iU>1uZ!o"d!s/=,"nS7^#5HMV9[*hj5lqfWPRo?J._n5=!!JStTN7e[!au"Y +r*]El/j3m&H_ohLQ$'.WKQmEK&%>.?bD#5@1B#KEl`9@#&:`N'qCc_m?TL,a3]=.o+:Hi%12W?s:o-e0jJl=jObYi>W]_CRrgI##T2XF(Bm`Ll6'iI`u_RMq]H&ma+I\l +-hYas#LMC=o'diG%.Xd16AIJ+%PJ]*9^7&OSp\A>YL1_,aT^u!-?01d5Bq*GDaPFK +,*/BmV=[lal__Pfff@m[m,"]pIWV*j4lc4^)YeQnj,)=P^?jVc_#NASYGt@rCd._$ +nDAEcefOkaSpphA[9imsq6]GQK*g0'7:(*G]&k(/c$M0H>p:A^a^6Y?qWF32(K\gV +/%!+lIo`#i1F>'VMC"_@T%:R"*SqQ+JH(BddJ=p*]hlGcQTsR\2CT;\0cbG=cPtd\ +5H\dqpVQ=1?ip*f_]=F2/[1tXCaa6/^friZ=[V\fA7LXo=[gPC`..WFP+hl4FZ\hqJ@eIhJg:!WousQR!WFSm8iARJQS[= +IO%^6TH_tPl0Usk0)V!Y:-G]m*>V1[5WsJQeS.0;M'sAOC'P#qK$tn+[EH));GO!? +;`DpJ/>ojN;VT5_9uC[f&o,+5njH*!/8$e3fa[*/hY*K.^14C]&e_BZ+PKn06sUGE +4k^]GhW)'_?-SGhPI$8dRf2@[a +579U0Z"5"9hYDTTF8NN2gpuTRd)qCc.eqt00f923;&Z+r+RMFo.9#150Ga7 +jd!s]bi*g/E]ip'*ebT6U_RF9&QBI3bO-G +b:)E0)!19Qin\TP9j>d#7q'0^LAfbsnu&#*Cu>e*"ljd_Uf2pn^Y"3LcA[lgdF`id +"Q0klJ`[$pd4G'H/ENm6Z^.BBaOP.ikuY4AJ9PB_4L'G^jL\4>bQ2P8@;?i=YJBF<#G,Zm>GTGp&P,BI^9%"J+KrL +BYRGVfXX$@5?HuR?Z^mUm$&F)\'GB$Vg0N1\W]f>oNPn\SLW[7Xul[XbLtL/5#`G- +I*8eWACaj?n*@k/All-*rPm[4LtjCBF:G`0ID0 +-PU-R$Mjd0j +LM*RgM]=QI_*I9Na&e+'BPK,?"F_RBkAEWB&]% +.Wp+`>lkOCk=ISbd^?HEA`D&lI0.,S;dud^W`Br3BT^h09l*gWnI*-^/tgK.dU;fp +iXDncHi.C;f[[7QYH-"KA3#._If<.AM;P[R8F\p/?GC1VmMQ*K-p@PHqt.q$5G2>* +MneZ_Wm]NNUmb,G4Ha_b.oQH%";,5f"a+\ +'YM,30oC/hERMiXUjcg"YE_*J"s2:1aW5A?L4RPK6QN_dR,/8b!.q82WgES=Pm9K, +-a_T+]?kXR@],dIV^FiX>Gk56Zr+GOPTJ='.)!erlSYK5U]?tt,QRDmn^%MnrdPbp +)Q&@^2I_G%V%"J^A+kOifQi*-mJqp.eZ2>];l)=fZQYT*DUKi?^b=!L*5Dg(_"VbV +p,Vmm7Vnb)i3,giK3R]05kAsiUG8rV6_;&Oq#!#cE-j/2p/Ls"*N<2,g+\rc"HOuU +C(6t3D_]X$XK6:QBE>4Em_S?n;d-Obe2pEshB]!Wb8IEsDO4DVWtP-I3:15J-4.(s +N)hTCf1@s#1Zpa2DN`[8_.Xm&[&`afAS%EV=Cp*_8SDrg.ls:9G**_J`Bbl^Yes`) +GZ:oa+NbNEV=`[Y=pRMN27//PJkCI4Tm/eWSC@Nnd3!0hJq('Z"BKZU>Md7j$KNN< +E"W4l+CbfVhJY(6r4N\m"[@GpfEjaR&b#I;Yi:&`8/e^4TVT>(Ha8k5+Yo8Z7#BR% +:SL)Ar8D)R.dU7jWTa8^$Er47!P_tN +qX&#?cMnj]j4.j6p@urr7pqj0!WM]'dE1;%Ae)O-fU@jO)Z,3l[c[%=3I_jj=Z=XZ +I;k,Z] +jAS:f-[hAGo"I*OSXaBfgh?G6-oZsfE5MX;!1a(9^rJ[p]7.YIpHALW7ClPh\5\.p +.fK'jl0$ed/E;+"DMc%aRLZd"=i]d3B1QX@eMb7.9d$hN +7\*[lF4h55U'Z?1&Dk5Ba#icDALcjGaQMU'*mJk:NPRh7\Dm;m):lB-C.0ZYRg4MO +,AV+[?OgMmO$G-!C:43;G-_dL3DRBY:)D4TE\a#8mI5 +.c\N;'ATp>aUB9jX*Wns\ToE$"g?on&),]hbLr(@+;s:P=7FlrBE#L1G^e$CEWH#e +Fif687PCO3"Mk%M,mJJF2da7QPr?IfJT#1>'r%.h)90`)?adb4/okN]Cifds9dRta +04Xl"&Ej*C*Kbi+4pYt3'VLE#'%^QNmk>E%*K8qprWEKQ]NpG`;\%sJ,&_k#2$3\_ +k"WY!$!X95TF[aiQ0^4eF4'5s'N%m1?S3^pWKBE(``;cKVi;^06SKm'Lo%Yns+(#oa +m$dYG0D5ctJcCIg[rc!4;P3h\mJVf(k^b&@Z/55+ilpdS52q"$0Br',R&f[F"96,! +(jj.X>"SPg`AIhgs&l2_BqaKsU$8`ne#IY-/ouCHIRXktNb"D,5ug2:aUOhh6_FGS +DeEcC7#FG0W$A:Z(56ll%B9TU>-N_iZi!Z3CE"'I@=Z+Y3PN^()4Y8i"H;&JK9-Z^ +.,@E73$X>!deO:X=Ei]<)-f>c#iomM>UQL@)>7Ubk7;]'/Xd'+KeBdP8jeg"<=*f& +e4#Cb>?/:NX"C)qVBOff3;2-Xf3=Q'&"$X;S12TIR#qT +0?bBDZ?cu0%S^[o?HE_eWh^sd)2)V;JaqV_mf<_K1&HP%3bgSpH?d!M2FVZX<9V5K +O'd4YIQuobO!4O-,Jh,FPA2]A:3gl^,,Ibk;?n!LTElY@k_rViZ\I0@!>k=6#(12g +9h.:a_#Hu0o+[s"amIR`H9_OrK'mVNdl=1!_Q'6h2G%&213UhTX(?n130`8B&U$d) +g0/R3]q1O5OVr_9FEir09K^08Y`7IHN;5baPgVqni/^q9dt2VhJg%?ZIgl;nq&qMW +I'Ac\5HMN<&*H/=*_a:,Jq(GbhI8[c1.HNTg2GU7a8+[U_L(bWi9pg_kT,TTE,jjP +q2b"9AgR:>!;n#6S3O/3@/^3+Do.tUjU(hq^2PnPk[RZQnU0&IC;J#1o$75:+g! +(S18Bj*MrecT4F$SgL\UE+>M[B:j8(H(oojgo10ZrelS&$d#$X9BS%EaQ9#c39^Nb_RnJUmb*`*3T5Y(#a^,K6?.ZOBc*Npg]c\S +'rZrbDuR@6PXi>d+S`S.r3*u31;^.UbbY@b?si0/%t=pYl_XM?6@.&rJ%*GpqNJ-j +)^>:/Sb/hEIn4mpN2Rh88Dp.kSihNeBh5Jr[/ +#nTeUE$\Q]0^sR\2ibn)APTAiW7H!S.sXb&16-(0AgWP];Io^E7E>>/g+TM(kmHHM*p/N(&r07JBS]`nsZlP*jK>&`k!?]\: +0"$%\2mT$UG[@P&;.Tgk"8VtBF8G]Db@ZHGX&AuWP53X>ZpF35'aJ"1<1(*$d:$QN +qE!2a)R>NKChaAn7O4/#M\P+FS[8u""I:jjn=$ai2;Xf@*]'!7!A";uO7-%cAuX@e +Ph>fh2J3rn"U#^i1=jij]g.A0/+2W,5dJ0H"gJ(,KDu2U4'=r:Hn!i2cQ9Ql%o/>9 +WkLL#8alPt*o=+j++BY)hoXJdM3U`8ET$jQL3tLEJI65Wj'/n\D2)kZ/"9<&1F=rZ +_]IN0Rb)e:\ql,2n%5p?Nu(-OD4us^JF'bC_N6*_s@7#j[>WZ<]&-rKUrl +g5J(o(*!7Nd_9"c=ilf.6RKkp>NSMnl3$=--W6[oXYW8j&gKRE9[=.(n'Kb;$E3B,&Tdt4WFuK +X[6cXU#_mgO=;"SsPKod&>(0i#>*iUD$4#^4CN4.)X^W$(Kf +I-U5'eet^gH!$Ve!=t=-OF=QnH)?C@&ue7hdG0EL1S`a3,$_Du>S-hN#_c]?ic"t7 +^EiGg$#n^a!5NqM,h`sPB`IfaVuV:m0)X]-E7_Blf?t4sdNCkf^qo:DhK9-)i#)_3 ++8!PMg7A"X+Q_!qHEDePUuun&s382b")$--T.ob)NPZ:>G^Li.jq+4B^[mkn"js-0 +4(_t34#j"nDh7h24.o`jCeff+a.R$^@ccKa5B@P&/9$^ep&?Jgj6*iNrbH(kLugB4 +(k,9^d"8ALo@$?5nGINVkHh5_XXdETs(h>Q/(T\p1NNI0"p"FV9@g"PL1`d<*')pY +legR2.%Jr6TqQfL\f,UusflsTl@4i,;/t:6)?FBt6+eE&JCm8U7neCRf +,5;u?Ss>cVfQ3>I,CoUI.5^JRl(*?Ep,G&3XaJqoTme^dM:u!s)'K^r$A"DZl%De= +'H5RYjQ9DuIrLu:#*Q6[N>gTG-:g-Hd)75m$p=P0Eu6J00nudo4Z5q$;MGn\M98>V +UNZ2Ng0Z0:k_&0nJMjb?*F%=OM>` +`(oUGa42]Xn'=ln<`jdM^cNmXHh5&_$Xp0_YZ?*9SLq'8lZ5R:qV1P(?gGCSGPXOG +GWr@%]5'BM7s"\'*rjK9:JAK`gWu2c\=`]\4"7uF)t1gqIjFQ=Hdj.Xq25F4s8LUC +igOGjh!htpps2PO][a&lfs4-Q]!a)Ug_@mIR:Gj\Yt^o)pqP%F>0g54bLLHA.Miu? +dj\SO$#R!Tm2#9]G5=QF]uEUcgW)K!$hsI:'!BuN;?1kR?Q@c@?2-823nqu^&T/e88^/[G0J;<*4.p-,T#;[ +!j1#"HF.C'TETS.-p5;2g^Y*Jh#EBYnd-3b5jA=o!E9;q>I]NC`D"g\Y)_f&$XJG+S!+JeX1.B?UgCk.UfRRVRF'*WOGH/Dt$8h2)SjjOgu+b902 +W5krb$:r!Xi8i31"p"J8q7hJF>&qI$T98%8J\uu@J03^1o7)h=HP1jm+*[Kuo,h?8 +kPQ.lnT_D>j1ie;2g`9,$,1o\+#j4>5(!3W,6>m`j8Rs-o[-LmA)uhe,4^Rgp8K"a +B7DK*qX#IVrjd/'i7H/dhj'eFOa[-?j.jKd".=]CAH'=d0Yc6Fr9&Fp'6q-F&H?%T +I_c!^,%!9Ne-dc<@X5O?cF^oFai=^%T>@W;q4ntkX=McO>o8#5-fIl3nRs!>P61U& +6I*,.JQIEL.p(q3bk!67%iJ^g:lZ?f*Qq):,*"[N'sUOOKR1RJ#8;,F$2G`t_@T'( +S%p;b[4L)BKkP+g.Z8D?^#mnfU90 +a2dqLZUSY)7[%UeO?$4C77Ch#P1l_)1b5PBC0J*$dB@-kBXe/!N8^1PfmJp&5$,Ig +#J`.ren01@pc8q=K/E]s!<.-FCZDsU_&p(5_qKR6StEt_I+fBo1;ifS$UB0Sb-Y9F +=Xht+<>sg79lIdW!OiR?k5Y5l44o!9r4r4>\-;&FaTN];,,!Eci*WQibFY0cN2Y_T +jRK/TjD*9,F6$!%Y:>M2^1%?H"8lM4\*mS.4hY=X^2/P-"\j8bIJWV:-^]C6fLg^8 +.8RZ]W7o$*dlf394>#(-$`)7Qd1*@,.p"gYPCOZZ'-`HR'cS;`RELehFfIb;fR>n_ +^&l#j74d=Ge3D40c`tKk`,8g5f#0$lrsN;T;Q^@K'9\%QT`j/bP)+iH:)j(DPrl38ktq342YfLeO@,g*JXm]Q;R3/53HtW*I +A(ps>%J]D^^uY$`m_U!8cTOqj,=bkL1SikC=?#/Xq6pIMY7D2J]c2Ig@-lJe`-qZ7 +dS9I\4m1e=^GER7@57SPE3p6f? +&EGq&rn4[H\^^%Y_4PM2:Z_h'2Kj_,dfJmjR40ijeHM`>@[WcGW26kj);VWNZ2"fb +K7hSQ#?E/Sf1tbT%"A\'nL7a=C;1(I09aa<)>ZlBp;f_kZ&uX!j(LnaC'PXgN"-?) +_B6l@s1u/`.:&fm'R"6[$qNicb@K6meC?eE5+2jNm/[O/Fod?&Df9)CGB,$[b*G2a +k7?kASLRNqNnON-9P`0V3W],N(MKlM3^hO.07gj'4c"9RWBJCg()4kbMM)1*pd9R] +pUTbi4b&b#hjo_=Y>2/&#r&0alI&m9JGli%h#An(/V*eVPa\kQa3rJK.8#eg<"BV7 +ATpIk$G\fD$idY6N;=uLdo:YJ,cP::g*'t59e[L+l#fmI_\B:W_JU4Pe]93YrkU!8 +:6*XFKtNEA4kf(q#3.eX\D)KphN/gdEq/#H^ECo30>)S"s7hJr*f:R-hceRH@o!.c +Z2P=$=K +?Z;\u_l$>lM>5E-!9laTqMC`,O2^WZZD-1hL$m+iFqFnPiF:dh6>d>X0:5QI9IL5l +'F28.\%mKLnh@G6o2[KJbC$hU]<`c'?WiU4ap#XfX$Y4>IsEG)K3LXGCE^83'RLnr +23]qbreH`fLt3EbD8:N&WHf\m#""X(-muL)l$XIF^REWTi=FR]`b,\?oD+:*!rZlJ +EALB"Fr(Vf`?kX^o>YeF04l`s7$GDI0ZcTZKffNf9OOn89lf-kYQ4UkeXlR\eSKeG +TgZ4jB`s4+IF\%8>^ceT9BObS_b-2&^"SFhs()R#&tSj3Ck#c'SEGs,7Z6g#=J_53 +^Fl+*;5+0f_sP4>D?"\-Yu>5hWAXkYT3o';n@/Die_KJok?m(6p&E>f9rl7%Kfo0/ +JTAG'[tV5c7`Cu;;o!??mgT"#A&LhU>SP6Z?TZ5?AcXsc>_89AVM4%g#4eV^JdHu- +As$N"e-mdY3N^=flC6mqY%E;1BYhs=@f%/Rii2`))1W+/A=>7:U^j+oaL!+8'mR^: +9N2Z/G:l]]kE>ZM+a&OdH>@oo#%+qgPWgbI5(dd44n_jRGa58]i@3s"'53l#1:k:A +gWl0kjc%iK/V^b_QINp6N"IGB8a#iGg&L"tD!,VFa)lQ]1SW,0->PmF\EB3&%gp$65e?r$dae)$Dgi8t +dtaoh$'>>;?bX1UI/U2?ferAEV.ljEo^?P8dfL!RJ%S#&&m)@^;QL0=Md4a]E&g:: +pPdhI;52I+r/#^Q^^R2T;ga'B^O@E*GqlYCb&0#?H`+3Gp<3gfDpTsRRMV#h\\0>% +cSTPCXSs2H@]/qYFeQTunGh]hjGuSfJ6)u,N9ZU15m)P. +Bq<_3!V#oBUdc*(7?c;W0A;elSU,;qhPJ[nl2o']&_lDE28F]EjC:^u5T1X/5mMlB +r4brcC)9NT$K]Qb.>rEP^.D:l8T!)-3`P@GMUWt1\K/a%Zj6/s-5,R6n=]qAQQ?98 +d%WSL2K`c(T%"'r[*l16"()!-#oN3E2'2OJ/E24-W<%E'l#"SMAiUHkIlnhiO:Z1n +!oR+B<]?.E.jI!'q$.Xb:si+`@lY;&cVCg%4oU#Kb70>8t-r41VK+4\*7Y6Wjsme?\9SoSD4ET;AT$,"Z;:UZlg[]aJ4NuQk@3WT(jGnDIf +#cT0XZ(DW&4skq)71jsl-2HmkHJn\u\AR!6+6?Bbc"EY"jI[sEn1)QP0$b5!BT4DV_&j`5=Ju;TDj6Y4Q3t%+\Ppf/8Y#_%ja%pG4#`Z@c`TA' +(AVr=!9_+%Tj+MG__lba2$Z,@bu3@qcBO=AYDfNKKg.>jZolO%grrjt0s8(mEc=PaBZ$O)c[9TY6/6&d+l.LU?+-Nr"N3"0H^YZ^Rh2'k7@(>/C<(< +_ktq$FA,@R76OBR2=oLbMts#!qNZFCn:+Im^6g'4-cO/T!.V@@,b/&r9[%.E6Y+jC +m][q59cmVUNGFqTn)"[ND4T@3>I:mMic*t^O +hr+"-ZhQ[=ptRY14pM!J6Q/\523VA(eK2j0IR!5^_:6X5@rED\pu+cd;^k)M.XA952S(VhD +J0jp1MVuf4ESY5Mo/se,_J1s6\q$`>X9j!IQJXon/"Od%K.I#qh?8u&NmG"cnVU.B +/;h=e9qV^f0adY?G$lJ;)cJgpPD@'l)*2K^A4f8O5j1UBE]_-"^!BC-fPl@];dI +EJhg0gg1Gb'=HgF[fHGZgEs9^NQaVC$Q`sp)&!0:X[c)T$UG\?Rl\(``'kPB2]?pb +aQHU-f?](u^WE2kWfnAZWQj_8Ado&X!D67b4LKX$=9fH]9J;r,lfP?&HEOG +pnJkgoVP3+NKOc(Y'i(->u'XGcru1'[6sST&aD^im@fEdN7p[1p;ek0Ia9F`NK;4. +#4)MBL'oAYAmtbsO;U..Ke@>u,Q=Ie>JC".hcn`! +XVh$No(6T;As^:/I7io>2&cnE7ddi3Abb@.:)7>_!W"^(i#V ++8#(_&f:BV1[25.`pr?5W#Z&kJo,ae'/3NY-k,,krH.p`1Q3I-d;@p0Ckc&RN>0 +RH./^o[!)RVi-6MV'9i66G>r2kDLn^#nc]+IdBp,fXpLcI9_8'Tq@e[q![C7jc:Z0 +2p`QUCrCjN@3D?GY/Q4U-q]'t\-bKcKVAp-2rRpK3C8B\n@p4gdKI&%&D*pi!X-gK +_GKus5d[gr*'h%!H_r%$C,GUgSCfccfE`XWXd@!bB[_9doH>G)f:5,9Vbb4reHl"f +GnZ*I)J%Y3[F:Oa#AXUQa2,@#_6&nc@:bLe=@3S7`,lcH5PSapSH!Yab/FDk`,r0! +/safDb83gZ5lUG7AcVgOXAS0+c"W^Gl^(;G1hs1VY?GC_u:OKJEq3+7b,"egQu):ojcF%jHh]3!'%& +AUq5tncC=a^d@;i[g_a2B$PG:(W>Nb`!T#?8/eji2]4tuVc10-\J$K]ZCMX,?fdS2 +Tkm@]$2i]C:2[9]i2^3qBi[bl)sjim3Jn\hQT"^!<7pRUG3:'Yh\'4)%XR>pBO"*r +7FPA5\nC$6VrKg;IeQ<,I66Q\$X2g%Y&a_(iaMP-@dq*M%=C8WeM8/m@e-^$H2l<5 +XR)k`bc_P?0#&5&eWHZ2hN.]uB\H2T6j*22^ONK#q?@)Vo]hnD7ChZ.Z!f.Y]p:6W +c&g1Y],iJS8J!b,:bqJ=1&brHF-W\c.9e]#]t;Fj?/d`0dN=nAc$"Q>M;<%2%p29< +&9cIB>de`$@sn#U!,Z\84T]mBg9.:F_;?;?<'2OuJA47W"UQlE3fW2Q1tA&EV+j9d +?Sp1'2I`RIXW(.nu6N&p1V'ldp%#?UfYIOMs>RMQ;l[f.)t'Pf\'JP^6>E +K9B>t7KsFi5Q[h(NPL#d59=d'IlNkQpkJfcPA.-+>+:c54@lT.(o>j*(lq\@7KoPS +`p6W1*3RK`+ZCT2SEu$ODB\,250TFO5oE6[HYjOKZoJ*E6[Y;M2t5Om3):SViUunZ +c`WJP;3/cP900F.ml,(?Kg,^qM8LEe&W%Q]57G7AL]?4+nXK"Knj<[lIGX*N^@HpO +O1d0;:uH?a_uW*+6SW!kd?BUP1(jQ$.e*5Dg^!WSkDc.HfDTVfHi7&em]3EfeVm;C +1%-lF#c)Ntm?Fs@#QWc&-,XZ1eJjeUTN5uYDfe6pG_-M$$,QK`=3nc:*Zks+&%Mf` +gE1Y=@!^I\]VK%H3pQ3>,Wo9\nc\`Sn1Wk/c=Bd:b9DDTNBYPlIYZnN`q"32qRQsF +dHgk<+%e..`;[B2lgG&U0*i[cJ-oZb(PLcj!iS@]nF"j-6nu,W:_`0r;&JFn*9L9? +i`[BkW!GdgN[[LbW/EVPXXk1j\5[bpY"@VFR?&^oLJ]GRe.)qm91oDS2/Hi_)4AB3 +cENqrNge3CQ_Hi=IPKU6&j +U7mQJm*@0*BLZOH)Z7#r[11Q*Gtb=p[=(V#qRfZ'W^OV8u/APFfHV)u6!24i#X +lC2FGX_+e +h/e>#cUG=s6Q7VnNI&@o"/i?[IUDm;6-?SG;JmIGqYEb2O%3,e\A`TF?$Z+>l@4;S +hS*m_\RPB8gA\D-\5T21ade"u!$kr(6]O+DK3SD*`[)enrjS#jH<-j!c<%2]9d1>1 +[Q@#;1XlW1$E.N2fEW$sjn7:^L1dfgAe#a>"htQBFi,.%W]JBGT')0\E^rig*h<7* +mKgN=J?f1S(tsmV%be.]+Oi8*YsTT36lXm:*sXKtU/`!eZ,ONsdZHW,Y!]-cS%ra- +W1>8M4*JWD)i]9>G(\sJ1E;T_5lrB"*e9]Hd_!cR?=;T3l@j`r#RduZb!b8=%?taQ8uk(J&[DC"W=rPb2N3>SkjeT\ +@OV$>7/4I;kXVY+3^mM;19#FHiJBDbiFLp]8-G1a(MK521ru&_?dLL5@@6n9S8qt&X3n.4-Fl;1idm6WL?_0u.*bb]jI;_$ +.o$Qc`VY1p#HNnEgY(o"!.jRSqJblYs7^,@m,-R03-DKhS)#A,SA6uJ@LrHbhL7+s +e5IU*!;h-@LCYQt04_h#s$77_$aJGP.+X.qisjn`gZBSbs$818=K\n6Is@,T-P,us +^E4JBKUVC-(?;_Nd2+_Lcn(RAWrqcr%g;n3EQ.+m;NGb8?X.#fSeqDj.tBn!5Y*t( +83[X_GN&FlZ+l_+J+Gl,Yi-'HN(-qi+MMt>r\&irXh*b +=>o2-!WS(@9TQJQYEB',pJZbXdn>Vpc(UUL/ +mo&QQW6hF_2ksNUXApnulT%p4g%@Ck)lgB+Ds*0b3(&cJ:M6Au&9E-fV'7ch'?YK.WDj6C_`LKc9SYlSY3l`3Dpqm9h!m]D7([76D +e_?AQ=Ri,1$M4h967l#bC2?n;SsGhsZBSbW_>620_e:/f/m;D0f7F"n6bkd9%SC70 +;,Q(n/fGppg)j`9JA+-9[j73aO9a"W/b[h_A]hehO:dI[rX4Qp.mnSX='^>D^VIn37@R[D;f&HI#hoRjm:CFBmQUF_%(#ZqMI. +9gS\YptQnei^^Q"Dmlm]\X8DbPbI?_cKXgV>@8K@h\tter4b%^`>[TB$\(:tq>L90 +iih79h#&;^k0`7n%OZ^L.-[4F'CVG&/sd+Hmk=Cd5<)tuM)O&eWjKDe0`]aTs*,<0 +qu:qX'-&n+bF4#YD&6k_Dq,j'Wd&8_-s:0UcdTLBUh1RBq.ZPh7Cd_60PFnh)48)a +eps/KA7ptInKhjQS2CY^+7Kb\?t0W'"]t1OU4,;)3WJkZ-7TIYHU&LbY$5V=Z'$iH +>nV6d'ibM.2^Gk`>6)j+2fgl4\u/FE'IhtC)qDk2[.-aYbsJh&cJ:m=gh@pUR>C79 +n4/ZS:Xf\F]$tn5HM96,W=jHIBsABI5:`$[1FK8U&ITCTrd]U!88VONX.I`LfJbUt +5muThh55<3i!ChqE2t1kU:jsa3u[JlW)HnJj6MFds6T@WIDG=GiBXaN/WIE@:9GilDJGm<[LBEAD7T4(ldN68\BMg1k!UU&s)hJ#V +N5c0)_Z]XpK_.N)D4G-\8oo>*(Ye@sY6gseZf6E-Wp#$6i:,XdruUIqfJDB2%*T"23aU6Nt`Yt%rJ#H0hhPR@W6"]Rc;n8q)$AN[5- +`En5!2aW%%Lp;m]a3P262hiO&NrT08^kEmiJ<8g+4+dqc-NJ!*K+J_%\c_ZL!aNqH +h2;?ii/p$NK\R,_ESpkB[UL>f/&!AL;p,CpbmI@DQI_a(n'S!CQl4pB[a`/3K7m=P +mBu&7/sRrQco%5.hr9Xs*3FUSSO8Y.cX^/@H4'g3;i]Sp#U>nZchLuWtn\1;h=4T"5!&4GG#XEN_5D\$@Q42q@:3'>$RB(aB@KbpucVqZ(\63NNb`UY* +,Ok0:P-9I"#W3VS2f9?L_4#s6TefMs#p;>mgL;=`eRFaq!W55:_#LV!.=kH>_?kbC +OpE'SJ6:$_grS$IMtK`U8TS@-2QULC[B90ATXjgpphi<56oLf<31)8 +)Zn^!9`6/XCk4X[_B,6pGM1GKVo%j]BcV8brWRWJ6?8DInF:UJk;$NU*k"HX=We7$ +VY"U)*fQ`e8UBN3;p]$$ABS5*cYRFdFC`!8Ud= +H24rKG(?mBn]$k_#lAk^^VC7_A;ih"!_EOa?]j&oB,2Tm&3<^i+h_0Fd\9?K +eu#m.OI>?]$=J='PD\.hUNUG#1/WfN\S38aEIYdk5+g^t!_.eA(_?Lb.]u73gc1Qq +.uS`Uhl8N<<_71+dFo`XZo/euALt6<0_rge9$Iq;iF(+4iPbU,Bod649A%K6N2?n^ +PD!7(7X@$[Y*C\F7Kq$MV=![0]:rEbbW*A]"TOE?%0#d8B`FrTg0WAO*3csIk/@&- +1"Dk6_]L*f$:^6bc@6i53<,CbQ=>"jhCS?t+9(_#!5;pW\+9,tMa)\AdJmAIhlANO +\!u8]s7uT24qE_K639ABN&f$,UBUHdiX9*.qttgrn@-SiGHM,)]+K#ZUA'`EkAToe +Q'4GTN8DS]7cp6/mE5,lr]e:2nNRC%!HnRK%a_)EF%JFO')n#/i$#SUO7*&/IFlg& +GJ!A7HQf9t-i/;5M"s_45+^(`;jqh0pg5<=(,O6KUn-Yd*i"C;OFG$2FInSCQ9[Tk +C5pSciH$XLo7&\/b!:jBMC&UR\7Bos]brbZm69uN#;3M%T]n6r]?Mbn=d(%:6FK)k +"\9C9rEY?Z>`8OS(\kV[4]qdgEUZ/3^q4&?[lba!OiP:FU*qVX5K[Y8<6Q]aUJdn3 +R@%K-T$.VUjUX@l-Dh.rF2a?gI&Su=*,U657W&'3Hfd:i+:)3F"N4!J*2iaM"d%NP +?=a2m5@:5'g,LMD$5N>f&%I&'5-P`g&+G'leIKf10@rPkLS%kZ!4R/)=gol>9>uea +4P=bhi`V*Z35<6Le&=19KC167C*9'SjG[u5'g!^T_&mU^L2)'Q3aZ-um9Drt_;H2S +^mEoFq:A^>n3RN%WeCKXGk)Tj:#NHF\Q5lL?PFdi\I*leEjfF78$sI-?)c4*1,54! +M9ZfFkJH0S"N,05!-YFoSIRE,"efIu1Q!TOVZtGa,9[AF=>/4"HI48] +n/jUh?rl5q3I>Km(UY(]1V!SD3(LM3d`g,e$utZhfVWpds+YK%,?sln7J$P'@t(?. +Ya-Hi(+EU$COqo>I-J(1^;X#FP6TE+/(VcM\]^/cW/Q&4.N4dboe/ZBP1!hGHK%KZ +>[`Z1A*22EB8HN88_r+9LD+Plou$EkXpJC-=#M[k^]**\nk1euq:-e7 +b=^f9;"j7tNBB&i#Mt=QOaJrX"\K,Y!*j@%WR41\^?-joZIM/S60K($UN=q93GSRN +V;"=O[8P?X%cm+n]hU10.RPFN1Shg6OPOX;/-'Df_&c&[V=qc]+9oPagoa4a1K)V1 +<(Z;,07J:B(AW2!m%HV>%<"oIIuOrSQpWijNWok\:j$Hk(o.-mC`Is`+K +Ej]U5fIes4!DR6cJN]g,ouMtA;oBGoHDd2?YM3KY+Q8e^W#l>Z2]-XL393iPJ[=MM#5EZZ)gR*/r'Wq;;'\8N3S3SPb.q@;qqo+/ +E;sP.SSI&S$I4&K4BY8O-mrms5K&QCkJHq4kPDarI.?`5W;f5>[l_pZSUSC$oaE#+ +F4\4&DVbWmI<*:!fE\Q,&+B8loRlY>+"$l7`(p)49KS4p3G3Z&A:!EZ4SY^.bh17j +Jr[cp*_8Cb83B1/3f[s1aJW8k-dgU%#hg^Ao[DIWK/9G1Q$Jh653'f=SNGaVo%]#n +*Hf^!Em=<$K9S[_M)?27^Y^prg%=nJH!NoQ+qm-C[ +`P."QmrpDgGi)*;*+dokF))KVAi[P'cQ9@t,_126`mAi,KKXP'S2Fd$)REt7>;O9L +s7VEAl%1?srhLc7B$Za:\fe)A#P[p2QRl@-D/8'q1$=TU=Nbf%(QH&'qc?X'-X,f7 +[Xp(9%)dZj=f!.Ak>\_IOB&Q=Y'^Yr3Fh^/==d21s83,.`.eXr0Q)HgY%IT +,2GoA0@$e#AH]o3f(j6WlC]/61`5#^\E\HaI0T6H(.XbiafP:h0QqlmK@8/;h""=,SW<:Ga0(Al@J% +]CW#KRFC^&>N9't>@1Vk]rX-*^2/-nAQm?FYN77BYVI]8k*)pa=+N7bQ&n\5H#WRo +FE"bP*iEI/tW59W=5ISIfPZKh.ND4PNGM +OjPa.2Th3J'WFf19"Tu1!fEXDDaSB6;d1Epaitr"R`))nd/3C$>(';L+h>i`l,.:5 +nXV:Je7);.W +SP!T;ML7)rFh^0%p0br`,N4,=!<42MR#$2N(B"4G!N\O#8Uuu\5"01JVDk8Q^r[*P +!eq?UN?mC#+[-_f5F+(ia(Q$DA*NZ7`!8HD56pD5Q2/,ar14OAU+e&>1Hn26Mpfi@?^CQf +GgZ!2,`E.)$%55PJsmMW`VHfH.B?LA44(XcGQ +gdsiR-SA_\pRONni1gegT,hr8qp@M)a5;sJ3(aNA`_SXeb;]q\;_0u%0CT\'cSH51 +&OVRIHQ3GTi>2,A+oGhL1)bM`2?"8UKMPm;4s<0:J*N3lR7lQ&8/bAi7mT9*P'5Uq +6ie)*Sqif^_d/(9h(gUm,OD$IE6]"sB*dk1G`umlN"dQ>^3^e7qo+!hZM`P:6W!hU +A3^2arqL[.JZK67YJpcie%V,[r2ItVt +2i`E(pr6,kOKS]eDrp:cZ^'"0YuuWg3Cib:df/7#%FDeOVqbEr)m#2$="Mg8aTI*3Smt+])"+Xn/Oj1W3V@6#8!p;dm-WI@C3/]9F7i8N2YB^H.PU +j.Xt0&#FdCQ7O9;HM+-.?f0V1h_77/Cj:BYqY[(%X%^-+JDN#Fi/#d"+&r9XPCe7W +EGrCdai6f4G.Ys(8IZ:;V"DooLM\KrQ8E\%bJrnX0`"UE+VEQG;%' +k`'(d:I^Kg$me"QN5%Ros5u*+hDjVjKO-T2DAZ&_kYHQRSb?\,q +>"&3k7s=2_hh]-U)?4"io%$QhceFBXDtBS=kI"B/qLA9ijPtZ\I(oXAl2=q[#9lLk +-cR1gO?UkHS?C("HaCfRa"#Em073_&H2d]RP;i9/P`jnLNHQ*B&bnOBI.)Jcn\6SX +oCkSbSQY+^j=^cF-OHK=')[b&c@3"te@Y0soKM7*,;(%0]Gp^dOX-Q[=cpb2dnUX^ +&HD?HVaZQnFQ/.An;eeH]^L>t^ZWW?T+>3)%MAUL]*'-t%6q'# +6kSr4+n?jEoJ'G,0gaTl-p_74G"!m"er>1)'nUde^d7.JZ,$.\p2JtqE#!Wd6f%%s +On??m7"4T\r*=mO)$*hBU?Mk"IP1+=pur,1%R#UulsbDoa`oeaDp\o=9pqbW0nTEl +g#Qt+e5!e]nu3 +!WM::Dj:WOmk!#JmFm*%-n:4lhCZt@0BL=7IiS#s7<4TXe35B"gFgp+Z%&=Jg!5]5 +;`iN,[CXupYl/gab`+0\(\"Lss.\<@ntPr`.lOd0;lN,).5+hJWm%+_ph9G^1'9!k +/QU15.cFT8^M)K&2P[S_f,P>tYM!""Ff7JO=+R5H@P`h2m*Xp2`HI+&qt(4`fIHCjD49q_PjGJVK=++/M;VG?SC!3(LfJaL$JJ=3LZ>' +NsLgD=M1Og.*H([HjL12)r.ttT!64W:Ql!L\Dpe=MOcNI?^K$Qs.'f$B"$]u=S"#g +n175G/Q-JBVA=I;Moe>-PVKDnNjN[k=H,*"hE5ufNO#DSYLCOCs-B&'\b&2Gs11HN +cP]l[N1q3Ri;_.'/3j.Bn:)T!Fe#$p*ghuVSW2sBLGtTP*]mh7JH';5071X'g7jXX +e:.Df2t'Ups#3/9`VKPg#O1.6,M&D=?*+tBhk(ml^OaVPJko9nG0.*sY1kAP_E39E +T;t;e?_kIb4/;Uh79\2LY=K&d[OOaP@CJ\jg%8-3b2>$D1.rNOj-A**mArJV\'H37 +E+YcL=-+%UZ__?Z&2mo7^^KX%*G#BfF8YQr]cmSA&]d4UZ;!<57[J]_a!M1EVF\TG +8jXWdSa&N\%?u#TGJGYX5JL5f;S=XL$(uDjWD8`4KG";:,;:kL<(,6s_$F5+dK1kJ +.^gBD5YGoL;gUf\:fgPRrub*>TH,a4!U1USXNQ]^g?i;AWRI>&=Weh'^s`VmW=uQS +:]XN0$MRrtQW#s+oI`:V858Oi9mlO3;t^u5D6?P3pDq)Xe/]r"eFI8Mn8T6&SihR-s,h*;TddX8bbF,e\Z +s&e[Z4n-5WkZ9Vh^6Q;.h!TT_A:uuF*d*/Nahg[2!pn_/MKm])GhNf,TIK)5UYirn +^L%q4IVK;iTdB&6"0]5#o;nDXZrV.L4SW]+`$>'H?;HB85']9,a!9I%K:=ODSClbomEn5J($da +L&W+`s%1)j04gapp=K;L9uk;?g[quCeDIZhIF+F&e4J!0eX5O7-'l:j/UH$:P^PL^ +X?S5Br@J!,Xn4cC@^kH=NAA1L8^7b&IiS#d&'-DDTYu$;@g`fVc)8l?lM+E&cA^Ii +`7h>Sr84i73kX5"_fdR)`,:G3rq%o1m-')TiWDcH4].^i+^SC/&"=QlbI60J3+U,j +A4YAk;%?&(7ha%eQo[PQ>WA.QgL4$fUF+JF;p*dPiG2$h>#LmX<;7N!J>e)/Yr.`R +:&MK?UP'J2(0I*%54Cf9-oS'D3uPsY0oS?>Ld-W@D$^/OO&3sD#)UBZ;I/t%I$gNG +'AX0KW"a^.(jW;T;Pm$'*)(&LcX3$r2OWG'?Rd>o_#K6b`C*nUG\Ztl:5:a:;g1^9 +E-Z-rdbsaX2$h%hQ'(0d_i*[_Qd-&S\qj9'8<*jPqWI_l$N&s]/&A*hHbiF>KpR") +?dsH#$3jhf\3rBKeG^pf#p/P-QMbT/k6/9,fBn.;lcC!Y$K+,lC7%f\Ga4;+6uV\o +hjn)Wn-AsQ3-t)ocf%\+Z3V:PFMKQTP_7=-W)"(NWR)8R7NhXA5:A +`=i7l?>:7?'KA\M9MiWK7,)6Oo3cNO +mua7>g`O`e[R017-ELDe[S2J-&,;Fls6!lqjs7&B@c2arA8m7Bje9QU4.-35J#5SJ +VRR'K+nZM"[CUn2&W^DZs01*2L`5EN>,S,6LI.UDF,Cm,Dr38b?82Q]o57ZbhD#K8 +d[9osM;Y6p^H_/>n"1$dE>q])+U!.??M".OQ#,(Fl?3st +4s[/U`mbJfX0eg$Gr+6OQRNGSd"c]5/Es9T-@i; +))`_dXT9Ydq!3[oV!oIQQp);diP9TOgm=h7TMP=q,QODRKVg,q=NrA6\T2Ye2i(\FmSFo,@Aulpo_NtJLm`* +Sd56oXCC0PLVHb/;pl?TX>?rf270kRn7eZtrJ#jkl2n\9infq-8ub4MQN]MPrg*e: +e\Jr'c#!8.4X9gaM0Nr_M5HOfOk-3hf8R-p0rMA8`64.IRs#/7QP4(jj1qc#r4U=j5:3u +11k*/BP"$TD7dWe]uKpZ)V[8TS93NJ(U,DpYV.HV[NNQJ6G!-'2!aIa].\t6o0\qf +*dY*3EU5oXq&aOq'&0_0XkDK^H@G`">^#BShPtL(Tsr@$^"\(FdnjtpZ+c3_M=Wp; +5W^dLEUsl_X.58W=knG)]E=8sqZ:#G\fBScFMR2"G1/Td#.oiH]C_N\^cb.\@_3'e +?uQ-]6>XC[g&WG,NKKg7Ys&H0$$47;/(8tq4(L"6^Q0+LO<&&[J62g#54i_b1coMF +Ncr:U6kXf.*VoRkVH#"j;ljJi;:3F`k`^s-:]9>so/,=TD#WkXY_2+mSmTL5Q)/r1 +.!RaQ=?hr*3cf%L,dgtsfeaCI2jN(u51jOLDgte85]udbC>hiNdf-Nfa?P/oB^Xlq +r.T^@9*6]2#?NNmt^FhKhgfqeFkS67uc!g/Ob9l]kqH4N1b>HU\;LZcT)6Wpo]G9Lqb6+@AB!j[eP&B_ +S=jRNZ?7q(=)\Tf)33Zm$=`G^DP72ON3&`/21EC=p\ +(kV@_bNP>8ma`_lJ=]*s"[;Wn9%ua[-^Gk$X6?nsc0@jTp-($FJG>7K=HapZL]p$U +qBB]S;76@"FW++OZcE"s^F-^G`;jt8!]8kr!3X`j=@FK^h\Yua +6(\P+6XL8MgZ6!FTD!LnPl!R?#FCATCF1<)>sdjj*9R9X%"V0).LeeqJ8\6+4s@?\ +gYm-q.-I"gmQSO&Dfi(S#VT27Wd4j1*Lmq:glbu0V*`V/N3.k;5T9:%hXb(@aWIRE +Ioh!XgBr_^Wgpe:E;d$ni))T!M3RU&b(%hTM1a_sd:Mm9nh*d2$iaphk_$I.]="/U +25,5`"HQb2MJBZ9Q3[?g$`I?a+MM"*G?&TB/Ra-o!W@@hWD*s[@PA#-/8 +U`3q:_`Vf3*cMMEF:c#F&/>0YHIrhaSGC"^cc8?p:k<:uLCFoDTuD=""qu>D`HXG\ +Xi+kcV?l6*f$ar>6[f0##X!d18,PmW?gR*C5kfbFo`()d`<778Y`4nKA7U@IKq>+e +gM3eGp)X"ApZ`j)d6E#s`790,/2U&E(m(bbXc^h.It3$Y#41aRXIZ5C[TcI%mP4C@0btfgNcR&[ +#tXXC(?&[>Ghh;SSZ0MU;sPID].NZ!(pUO81UaWg*GfJg*b4$uW=@UY[r/h2e/gbf +YXOX#d^e3FWhs'g^l+^GRS_W%)EDMV5Ars8Y[T"-re*[73o +TK=/4i3Nj<$2hj>V-/\PZsK@ZE<\Ya#&S]aLPRdn]l3%RK +/.g8o'-Y?&_DM;Sa6q*2r2TF=%cP2o5WS=`e>I[H'\NamZ.#0\$R6[%DTS9 +!9"d-+Q24^((:3=B/AW<,0X]Aa69jJLnWC?<*7OIcj)3`mbEA;4MH.cVK4YiF\3dI +&iPdYV8.R#mjW,Sr_KE#K*7.2FF`h2A\+;c,KV>7P8RYFX%Vm_)[9IQ.gD__-g:_B +JH&;`k88VR4IOt*g&_-'7+RC<2/C&Y4+&*WHU!03!1\"N=?.u>%^-$1"24TK/%Pe? +Yft,]!Q5&F&0/caTGr6Cr"9VGOK"pG^S5&,-ZZsG%Asn^"SWAE2rf_sDu2#^J34`65rBbl&0k8K +J10?2$E#OLm6<1DqRkD@qprOfi:$j9)`SA?,$p@NTO8eWkO;1ps&!ji@t9'8n1!kA +54!_=s'@SG(U+D>o?P4q)^4JD_hWoZ2F?:g5k8mg6NSq<0Q6hBGC,I1"7T[]!V!Qa +rk%2LJcA+>nA<)'bZ-!8U&SKth_;os+or=[_].k@pB:8To_r4m^X7@n8Csq\Pc_)a +T*mb*%QZ'L=s"MSk/EHW_<.&0()F!R>M-QrZhNm">OQ&^g?G.fC?XVg*#)crag[-D +CeY^^alQ[d";tcq;2MH%lhktF[D1+t/AXVQn6UjiG3[]qlAumkhh9]taOk+?fW*!I +WlNt5@VEBmq"RToYJ'cP)h"ET60(n\n=rI%G'_F=fQK@0#Ij!c[apWB`iUE3=ET +]PRDLEp^FBgqUB-ZLLkq@g+HaUl'?Sj!qf^T,t]Fh4C-49]BsRa4rS5]4mZ-_iE3l +&)EJ]I,>b4FA,Lp*EA/#N_/W]l.t:tY@P^\ALXKdUagjFotap\_9Z+dr,>@X[`t^m +Y@K.ao(q4ll[Zh1-P\U[9=3XMl2J(hRUs^Sb5eu^*8+;_!oQ:m[ND%>'d3OAZ.G=4 +=3_64R;A/tSH%9,Ss`QZfE#(TMm-RHda5i[KeBD>*-cGe-^*3a)W3uGoFA#.aHC$g +-`M/LVJ.BN*WE9eB//Y(Q)/a5dRoH5XndBq.<,/-4Fp!'\3LN8GD5YqBb'^+:)9gq +Zhe:QSYMnO]!1>MaHfo,^$c?Y+3roW#+f:o*dXAAn2A&]mtJZojn>f%G-8jZ/B.;a +s/5Yb$R4hG=H1J,\BFj=FB#A;<"7.Q[&=!6pcfHl%bM$LIj)B4KOBC11D*NeG($AQ +E?B#LkP8dffk>=N8AM02"ZD24<=X:@Xk_CWKmHGY-jWU-enu]E2J> +.0sI?SlS,I1?8umd:jd6aMCK+HmM(e/ErR^Cstg0O+fM%nu@8/\:,`13g%uOSdr-r +*`)*hcGjl[ir$\E$oXC$1RB>Y8F6D-CIH_2aciD030ki#K?W3QmD.sL0b(qZYumt\ +[%ttF*gk0dW/pN_<1jcWE^:S2hN-L/+[p63pKP;uFp0u46uNe2G6\k,#=;6`3ZM!Q +aMl^1ihaFHJ;^^^&.mGei/^l0;=u%4Rd_eg!7Lm?1.&?V>psk$KY<'%mgag#_g):J +UDDDP$*3,qjfQ03/_oMPU<8%s-<3`F-hg&nmt2P1TB][2W'('%l/Bt3O+,d?ohu[i +.2`].0cI2RW`1.Ek_UeVLa[-MJ(c5#!XWPu5HNo%D\,#2$)e40)(muA.NYMC&KO(S +mWAf5jga%O>jVOaHnRQHIDe^(OfNo?]+**BRceS3GI&M0m^B8")ufja'sQFD!Lrlq +f.iljZimHlq&9sNE?Yhhs!Ro.@"eQLM``riIq/Jq`pgIPClg$f#W=>o"TF&R^O<0^ +rga6OHOT6OY`IKU3$6[p3MGWtD@kf?&ftF)%PUKH1J.6p89JLaT3t7S%UXa9Ub)c> +kNldVm%BAC;YK60r/E+Trt^$[V&mfX?Z3EYJ[Lo^i@T@.gR*H%R)/Y;@(uN:PIi5u +D:F#-C:>X-]*WPZ5/NrE<]4HO;8fp-XXh6u_LF]QfPZ,7%?iikVOS'.#r#q8>#9ZP +(r96Hc8I#!#6cS"gF/TSIHn*%W@PsUl1%Z.^O49!k?f;"asL6 +H:A"Gi;Bca)Hf3b31$B?*tiMqc9Le?:Lg`ln5NqT%/Bj\Vj5G<;uhE(#h5!e&C(CU +iO?UNaNbT@/8#SW(/fMA;E>WBHN=%1oiXEG^@,[RN1*g/[omU[VIUkqd$o?i;\jRUOQMd"oj(S@.4V-Xr@6#@tI=DW-0ds3\<\uIjCt%2S<`a\fJWj&.?@) +jiIabj7cNUO,!R=N!>W0EN@*)[1F.uE#Q`D.u(`1`e#]!Bn?,k-g_a=RM/YKXm:;SL,jcd//!q>C]gLE"t)XpqM);luVVIgrSQ +JVH]O:0#R-,>/dXQEa%?"e2:GS%.uK?aH2t>"Y&5-HbHsh34S!0fo>%QoUV^%/ca= +BF";7lG7R&^!h8tM6,"-N:tp+3jljGj+:Z=84c%T%/Y2Oj`G'Gc%*#"4F`HC2STKi"XHNh"9/I] +_#FZ8!9[l0VbC'?o+1cf%4kP#8g.U:14e2Y]M!1&-jW-> +WRoI%6&u_mB2&F$)ZB+sK-,1ASggGX8Kt*==9SPW-if/iK7Bl^h1[6WDF"FE&rLE2 +mUgbKnXXO$T#]XffZnWGiHZJ2cF^1LT+CjD[GeaNo3;fW&IEtY@Db"(^gHk4,_0-I +jM(f=r+r=ITD\G,,Sp:)0`U`A"ZGt%UBCj=1]KBe)C+bRR:;9qJCBKO/_K"]n$(5) +\BE65^X4cO6PftppLAq1-WodhCi=N@TKA(pVAFN@*TsMjG[)p$.$@NqaG@97L0HIE +qj@ApQnDF(PORcF\e#ZOaZV(OPbHptG/g48UdNmnLR) +/*Y&Sadl^/TC,JRA4OM]D/isUmRm=jJ,3e>PJ5kBn\`'lJH%Q4;t'0:oBe$Q24JmQW::cP8Z'a7UJ@Tb^fQ'UYLp.(IP')dl[iG:EcaOaCjb^M0RH+kI +r*SRZ&bd>a8aSYY#=R.N]K81`r$R,Fa=tqo6Hqud0jH9$*E^!/N*Bo"jkuKNdJIU! +8R2^l\E%E`U\E;Qc("=BbCpjZ/2eVmmJmAYTlR/H!e3(hr)Tm->2^ZB01PApNa8mM*q<9H5:$C&OC +c-Wb>Q_+LM96ARhK`CWuc*O,kD+LK]Y[#5L>7e1Rr_IQMJcG9SpV8WRO0o!7'E"<@ +n2aj"irm,7L>P8A!^J*jiNlD0`TU'Q)MUT$`^Mbji!7$bbI_=,FkrR;_`ZunZQSc; +^Tup-AHgrD&j;afrr-u`gM;FWQNu(oB^cGlg^j`>]E3$[0._&`s7mNq!(L7!l[0'5 +!mG67b>*Z!Q!=bFiY&ic&ppNcg*`YaEB^D,#5mYhpgsLfq"]#m@2Ec$;n8P&h&l_S +q(fH,q)\:nE)2XJT:e8'0GP\eN$Vf@MD7q_pu\hLg^9$fkJO>X+YRiZK%iHDp(CU3 +T"->P*&i0%,0 +Jq`;Od>^,l.'@,\gl-TB,csFr%3PZRiO+@=)L_iP(7mo3$K3R3+!iB:MUN=0'LE3g +VZcu3V4^/p09_)kJ-9s"8sPRNrZnHoVRkLU?D[N024Rk"[$Pn +^F/]dgYhfV#Q_jF\0+7?dLfg8.=0P_q&CZQ3onoXOIq[5Jd7qM%H=+:gdbF*nU*j] +(sg)$QU>16LVWrQGlmAM_d2Y_DY=,j>Mq_n.<)[Bn-=U0!5J-Qr\Y6HZloGQq\ob; +-P?DehgiJ[kK(XT^]sft,QR7+:'!7G^kUVD+8u6>#(J7bA;boO2pu@=reP>i8.7s%M9u$rqf]skF3<'u +g2"pfAjBM_1X,MpApQ","1FVkhCa4=]!X>&+#[Ug>@7IZgqs#,N2SlD'Y+)e;D;`< +//X'6bl;?!&6h>u^uea,(Q5hnc.]AA3iLB325':EYZM0?JWlsqLH4I/7>5WW"FD6u +?praG0t(&_*>?Zua3)@AU#M(g'4(9o]T(>@bgSqgAdS0A$P-d*Cbd2OZ_AGF`(nup +k8_UK*/RU;S7?_o'3JSbnZt@Xr*r#/m36K%`S\*qr5:Y\X$`<4fO]GQ+T@:IBj^:( +s*SbM(]..F[E\Wp-M-h7ag\>sT5@7!H%5N4gG?fl(MfO*;fWrFibB6mS!OaLkNP=O +OiY-(fI!*gPH@phYikWjpk(BRI/M#;p\P;iU;>bcnC)n)=@`bd3pl_Fs&\c1I1/]I +UJLm:q$/n*jN&Z(\gF_oB+'rSI55&LoqHVlDFRsP;?'7ip7Y.I-^37L!m!PDX@9%4 +"^#(m5P4rt)#3^_.IG%C;QTnopO'eqH'O:DLi@56h6Ai&dXcmSINS)NaU[WEd#2BH$b-^58ClZb;ITp5V-Ker:L,e.r"e,Xs3/tucPciA9B-ife; +8T$CJHp4E=.Agl5Z=Su%Xc$r\(VG!%GMs2CnP'DM%D-mIslGFpFMi`ONn=u:Hsj9(N@H1<]@[VF/`H/P*XdSY-ff<]WG9\P_]4b +/#NUgF&Z)[h=G>9putaO.nY>jSf@bu#VZPpHs/%PoOaKbORfHUFA5ss0^6KKD9A!!!uJC/)ed%oHYVW!K[:=n"kbG$FbnSB>FOBs!R:dYX*f# +*FT:@7B`qi,J:**\ltZWUe-_mM96b>*Wc:V:4^c\J[fkJ6.Nf25 +O8Hf2MVEE;]-r@d![A,MB>^ +=V?L[\E!CI*bU'l$g`.?D.7E)LqA'hl&KoWB@b13m[&=0-iESLBjVt+qSkQ8/jA*Z +r7TBV*tDgGE]d]Pnn+HUQX.H%qLp5jdFI1KK?lb,DOulN:.@-86o@ZX9'9)5!WK9C +PUdWWQ.2DQn=!+]HhEEVb.O=)nuKfaF`W^(H7*2U?ulKaHX]Tm)1:RO"\C%nhtVpO +JkcgVT)kOnRg^hYV[S.>&(k7"0YDRSP^*3!q=L`m]((Z45G$*9Q]!lHe\V?'#L68N +SEU:)+4(J(l).V^/`b"3ahcoE!3VYEjOi%@EsB4RdQ6Q*Sn!EEWBRVl@-MJlW"47h +!;POD(c_H_<,fI:h$`kmZ\_pI#S[2kRCElL7b$38<&P=;?-6,uPcUIm&U+3PAFaXt +`]sh!?_+*DSZDbgo[)oG`H"lNX/)>tW7Gs;_aKl66;XH=PQLat!KS&5Xc97^7t!Fu +IP+e2!!i(Is,[,n6KqBI)[$3BG+/GTJcBJo8&0UN$=-UEfVG`k4VG8;0f8 +[i!T*&CWWUeIL"-Dqqk4[bC,,caP7XZk-M!iES=].6q:M9S!.lpQ(o7^Y&2!B=cu< +NP3(eqW]9M@KIjpe*o]SRgXiVq/YEn;N>3H$_7-<>c*^^=Mh@Gq0\KU+g5hWm,1Lp +d!W!25!o@u!Q7CT0s0Ju%9cULVU3g]QJ!W!UBV\>;iSRic5@71]XpGWuDtP3q`\ +'9D5WU.V8T'V'ZPhgI3F&IuYT&)k8IoOT*^kn[=!_*h+&?Z:iGn9I6rEfaA)NIX8Z +9(jGp+@,]d'N)%?X6p?I#FU*tOIe3I4j>PZ$%Q5@"F/]BLkS;J?9'Wu,Y\:O'qN54 +m\i_]*<.aa.&c>i +6ilZe`=;a#K`>&b?`QN]2'j!f(Z$\3j7++*,2=2hm?mnSrjuA:s2!m]e,0_DBlKlS +5dhX$-hd$,h`))m>l[7rU-7j$q#;EFr;8Ms]T3_W5Ilg1PRAZJYZV.6mEQR1:[l/mgp4O +b7G]BMm]D"ei853^RsK"MaHUaP'+R-g1A40116KUR5FqH)uR1nUUKJeHOUkD7LU6G +24tHE1\U)iG0j2IAM['Qo^JGgHn%\4[EeJW3!f,5:LC6M1\P\-&`:eJ]-VfA@TCLuIZkn1# +'6e6$pp1&o?eD17mCq6!G(tFV.g,/0]bs$U3q`-`eLL#@gUP"uUL*R!TCmn$lOPWh +NN1MPo;1IZN=uoK_u)"Z7*d0%!_&I(AW(99A5K'[XLeAkh)=G'qrl;\;pj%(pi`/N +7d!HG$<1rh%_b"h]S7f3.Ps1$s7Ye5>okd1?]ptCdfkW>6IkuO2`s +J"5nudAoq),_#V0OGl::3ra<8&5[\HT^T@jj)6PbZ)T(K'qG`]Tk +1u4"cb>ui`bJEur$*>*QY#n;]04h2McDp.%AE3^eq:m(Da-$D"qd8NeIlfgc`u48R +;V.G7@l:N88gGG=ekHLUe7XU/ZISb@F\)$Xi)*T2&Qq17,Fsa1Sr8h.nEV2"R,HIF +Lg!hg3e76?gaZ\nn6^J3I\4dcD:$H0E=42c%/?b5NO+a!Zf+VWgq$qo&s*FX5B\6, +=sq1:B5JhIBHKH9]k.%u`ZC:8)$h=t#iGAaL +^knWHSKtO#&;;^Ui:H[I(:SVHLKe`^M$*hY0f`%=#0?TdD*a56s38:Pg\i5f";MUL +X26=dr`IoBN*EFp/QDt>$ud[RTbY)m-aF%(?sMJB)Ug_iC`!jE%#^Q>!n@.Jc28q% +clZ:XHA$#35Cj3JT"$Gd67Khe:b2!')o&11U0J:/dk>EcIpIuiUT,!b<$2.;mNcr\ +<1N`\f)9rWJ7cR.NSZnOH2\)#5[3G^C%/eOhP4,f'*nVB,)Y[\&"804lG=XmL-DUN +4fZhRG`@p&[C$cq378Zc9_]0kTC6BQ5/6F"0NX`_(ItV9*X!#h+;_sS*sDL77?GQ` +%12e;rs?n.Hh\3lpj_8;j?N0a6NYh#J,`4E#li#o^h3S8Ql,u*/M(#9GN^'P1\Gmq +$MCde\X<^dN$I*d7*_li,E-RH>im?:TnG?ga!Pnj)c:rM_`!-/@0"rpE&LY>1EV]Y8%brC&>f0qQds +>hD3'ffU$OV%u1Ks30-40;orSf*q99L3X6'!pN6drCfO>0HZAUdX%71s,!9KSFY5E +^\#$#<;I=%#JD)dYLQ3J!"mjE"G"`q!_)"n1EsWm43R8^7e$?5T-&VgH30.JJM,lK +"o+_?/=^!Aq^e?]_DFoZAsj_/3Lc"`YZ'2$Mo.a[`SbpR3]"96*uT^O:'O+=QO'`@ +5[u"K%Ff%X'KZ8)*B_,e3E*A:_]66AH[eWkDEf"C]s5"^nVdZ#KPN(G2;j2*;ZVnj +ZFdsQIpLD+%e(H$-#!FP-W;lQmMl_>cp]cWA:`i9QNpi(*W"OXWINcmrqjs^^KmCT +Ir>&;l%+>S++'KTR>^B99p%)$3l_Es'4LS*d;>'Z20,S^sm9GcqNUs.t_eQ*elG3!m\NWu_r)F^.!]*9M/atAdLmmJF +mIO2WTGI\ocTifm7+*M]b=kkSn:*p_18!@$NN"HiHM*"qoF:E]l_`H!(*&Ed>Gf4q +bW0s5B-$Hd]3F"HDNDNFY(\8(,iHo^/`Ha,X!;RFQeOTkfgCMpMRd9Q#e?1S#O"3n +k:8o#6FD>bY9=^E:?V-@^s6!kF5QZ[n2;#\`f0]E8iKWK0$r$YNs0#.8*D1/q +r]bdC.T\O"fgRBB^u6ZiGXgm/[3,LF.;d0+XR<>%P^pUFVLdBSJJan9K%&Snb1jTO +OqmB,ck8AH^k0nHn?W&7WkO:0OFVYfnpAgPs0O;\#a\(k1;=N25)?fq +b;:B=lJSs6DO!8Y*23Eo&_&igDAtq52Zj$V"Z=)F\/bjY_Un^GZk?WOac=6]X"fs._1SA,+? +J62ehlb.a^oY9,pn7ttLh6sP1r\k-)2[Q*5Mk2bE*`qnR1oKlRK'O]1-EK#hY(>s4q>Nh3nDIa +$nm"p`SPcGE:Kk%*@;7\Z-Dr.?m$,uJ&$M2^bN]U1NOd>1BV]>k0=u`4'Np?B8[<3 +M_KA886:E\B=@RP1YN/)Vo!:!_*O<9kQ@&_5D-G3)6fS2cMu9D9P[SH/n5(sfM\>Y +L)$ZUC=B=HuIf'!mR^_IF_pK2k?BHkC@3W:#OC'`\ +Ws8>0EGJ(]R^E"4M:gsu*#3_t;t4_2M&k'M/b!0Lpij:*=8IapqC-FUF@&(a +KYTrQ1]hRpX4(2$$2c1H+(.8RrhA;QVT3U)i"cd[J$,E&ScesFSK5MRndmI9`EqJJ +dSgt?3J%42q5Uj/K63Bm**E1)g`RH"QFX/!iS"p+_]qu^8EVT&'_jm5+e=[.)<@'+Hf+/^I5%;G$oS_q2&9?$&Dt3bcnuis4=S(IrfjX$$2>V" +3@/F1NKVSF)V(ET2"$D`o7&4"HSVT=?oolR#6+Ao4<"_^T9!ZSNG]"#O+l+Y7CUJg +Zqhu>lb1'HBeI,6JW4'H/U3mk66Hh]br"JqbNDC9[(16@^YR;O]eQm;"l9;X+0e-KCQ84k[@\O?Q#nGL-&=!uK"r':]E&BO8W:%,k"TNj#MJFIDP>RqH$7Ls#p +9rGJ0"97>'0WDTITa;]QV5R!4WWF9SO'C2pEK;++@\7R`Gu)dc"n)VJ!$D:R"gn%f +B_K/5Ajl7o'Qs18C%ltUi046>AO4Z(!B=O,Hl/,u$1P\oZpGZ*;T5UPO)EOfFcJjI +X8"RNiUm:d[G&^AOf/JG!#;]M2]*nDa[2TJIDImh#]p=G5N6mEEV^6M'W7^F9#t%6 +1*]O7;5K,Lknhsp,KoEOmhHlCU>k7CGUVMH6^>hRTE9(*6@e3kHbfYnm_e^$+Q68W +Iq/96B&8_Tmc3Go(\]1VUE#g-/r)D6D86/HI$]8sJ$,gSci*n+J(UXrcbK)O0?O7o +aT$DWT'$&bgYb:5iW!L7?"6`60_Ung2DY<0Cgu`" +#;c?-!F'4>G9'WYo]\&(IN&2Bs%W1MH[\ZcY@mi!3reMc!W=60IM)gGK)h!uSH+.f +4BA[@Aic;5O'/bo[kN6>ZU/lB_Y7P1E3'Mo!X8@_(9lG$&->\u/g(0oZ3RBpMC14! +PF^uMOQLDIiGFbBD$H:FBTV=oS/#_/*GV#VC6@\gs,=;=m3C"c*D.l'5#plrXXWUL +Q\/P-!;J*= +8%29P46G"X6S0PQF/K^Rg'H>pO%q^GVBrj)7r/aXX( +-Gf`^qi.R>/l0,A48dj$Tmp;eqr>oi.gQ^7qd-+ui0BfH2D.VUkG"G'U!L<_m]&:n +QJ2%I,-e:IgR&!)qJ +;qaMC`m`\>EWPr@"3F8i.FQVfG2U+fV]5CAY<4fNJ8f!C";mHO%o7'`_qh(n'0/50 +o_-+c(6F'\kScm#Pp2rC.&%K%gZn_;gj;``E(fC\4\0(JYZr2GdE?sp&O9.JTr7Fi +q*"lGn9[h=Je`j_(&lP/!A=Y/9*[g6"uqmjJ.fEmMgdiX!R294D$.Y0iCGDdVu_s6 +`@YV_%1\`":E4\4T4-VsBJGnn/a&AjL1%t4#rImZZL4Au\3$(L(P`V7=Zlb)?YYL( +>ZDeh&)^g^p-Xi:O&`@?Q90R"^:Qs&N!:94Y:aib_^VO:8@41h@rllmbImji`uM>$,9q_ng(AO=B`qF8Iu',/N'$[#hfFki;I +ItTC5`f<7&N&s?P3$:#PXRYeEn=R=J)6sX[aum5"iFke'1h5mJSUZgua0Hf\H7"F: +ho(%#25JI\\tF0JYT'f@%9VrW2o%6^k5Y*&0E=2:&W@/*=ASpDo`%K7coV33"GfYG +K`V1"%B+o6bX9B$-PP'&90Ka9H@@s.TtBuJs%s#(V#Ks1ru$;YJ2qc'5ah/$=8J-[MGdp>(SE]r/EO4+./"-C?":2@Yq#L=KC +:\WMej=A?l!H@pU_Ai:MkD'gmkT"L&JpWZXdIe?-Y/#O8DWnRQ8>OeB"0\sS=:<_+ +B`S(/QTXu87\%L-4S\icKN@G'cG>_ZQ3dTWcYA".Dm#.RX]uhIruW500R1%8=3%30 +j8^(mo9Kd]Z,s$X%sD#Momr>-mM=74ZUqjjdHuI<\`B>`\Y*rB2])>Jh&QFkVn*Dj +-U1iSs8Dlj]@2DI(CpK6TSiF.[D>s!-X1;FKoZo +?S"f1+gMpl.bBE>cU9\-_oEu?7sHm6%Iga2a_dL1Z3LjU51piugFDb-9>MWAr_:l" +CSjDL\+$[PfBV:4hVHVm]/d*A;>L%qY%)^)H$K#!&8@&Q(Ca\]/-dXEr;FLrS&.H2 +Lml`u^^'k&+:`;RAXYS!@Pq(MG/m9E9!s8*Ym;>p!#Qg8=LjUO#/CHM(eeZg&HQ,M +pRYo?VDES?E\8rWS4cMua&%A0,cr)n5ZqC1Bl+CbW$M&9Qs+WSX]g62RC`Nl2S/k8 +iU_lN!gSm#/YK@0]W]%qZHsDVYY$4><$(t34.mEWQ%=`0$sTI+6,a&WRX8MERXuU3Gl:n3rWCs,HMgTZ +nnTqhW;$7e_=-R,OF+)2>QiCcrLuYadMDeo=n&YMo0b&WI2mSqp?44^ol6mFqje7N +dJW//s5!&;mKYEKpfIH5-9kVBVg\>1m/K0Is)QU.MlE:talfk^8,1.%3eI+I7d[RW%dg,g*2NKQ(u^7< +0h:G.=Z]5lK-Slrs5EkF'K'ir*l-ZU/)uF3_q'FEa-:"D6s1A0_gd==@)dL,R&L)g +%.8X8peP7d))Br1ZC3Etqg@F%tkVJWQ4+_qaf;@I8/Qo5%n5HueCJ)BP@Lr.-a]s5F6\ +n(Id@['MS-?9=@t%WT=*(3Wb&+gZ,Un06[1\t8LB:]^L1Ff.=!qS;&[G;11Vp6:H! +HBM/J['PCC38krh"RPiV1CPB/&3/U6JTEgNGmsiY2:)ZV";bT1b]Gbg(0IH[)K;/# +R6b4EO<:!BXSL!)=Y7Q$A`3C!\h\lllE?2]]eMf.5G[c+NuL-!ppI) +\Sq#FEoeh^`WS/e9C>tU&^inEo23^Eu0 +1([\#md"t*'ARjk(?1.ijnR,&l10P8DJ`WaL$+e0lqY62'n++PiY-D$/Mk[PPJF47 +S:$Xc2[?Xc:.a?u:p!u!]Y;f)r*R&=+i"5n3$855'>^"Xb-0,#P6$(Cif,7kc=Z]X +;(gZEJVc7?qe-9Y'3XK2]-j@W-R)c:d;@N#)bFsp9PbTC:N]0+g1GAPaIrT)^fUem +*Sl''"U&`E"p4$p+;k^oi$dj"-=#%;hZOPlGCs:332I?(@8MQnJ!>Nln>cd_2;lol +Gh`E#PPTZdTSt_mir"RnUTVh4TDgV#EOLclm9@U@%586N9(!t_`u#U +H:)@"k]"!5#nVC?hLU/@%NpX:\])YFS@un3g1*Q3m\X7`;aIjV^k"f?hIX^TK0:Fi +:_ES-Jpf2qDMHJ\4]a/b;%!RLqjK7K))Eh'(hSM=EpUE#s4Q>Fq+(3a)P@,r/Kkl` +r+Pp@I(;ZfgNuL0-bu?Lk2Y._a&q7q#lsa)1)ZBOl=2II?2r$D#7^VL!oO2Y-WA"k +F-c,-VU%/4i%>dh+3phsg?:uY:bd_j!^PSo)i_WGVtKF2)d8>JJk7Nr5!mo5RF64SoQkr2YD%ih86]'YZGr#kLI*d/1*K=CDk7H +R@3[gnGe=JJ?'2KH?3X3gsOX["(^$QCk34$JY%@o*(Ku0UI_q0Y&YkkCHAGC+W\8g +KfgYDr"Vsc*<>Z6h&'M5U0YAl$F'Rjr,[,jZ>L'b"Mb(j17PJeUM#BW_#HWs=h?BW +lEgb>+=>/V"O1+9dsOXBX0_SS+)[%O^HD#Eg@.=E9tW\To"IrtiQV)cVmLq<2u5JC +6]Q1I6b/L=0UF3>7<^:0F>(YB%Knp"q"KIV/L7_TB65!kj7fY$oC3[,')7Mdo]&$1&j)F9<(fgRR@'!8:*o\< +&3`cU2(Z\Tcc*7B`4!A)<3G?Vc"+2jna?EbSUY]R%.O:O)=OF+pBmJk3H#OTRa(5W +HLU'cipCjtnGhn+&br9;]/K3YL(4@_c.-fYKA_`Af4;:.!)rs:EBL'bQei0;f`JjO +%;t@7YVdBL+/6`N9I0u%UpqEj!,O3`$+sj=HUm+b-^37,('&5^aHk3`&hoOYNdWEG +!<@Uo$a9sdVi+eUVKIpX+OI]N.#]:a+=P$/d'O9:4RS9J&^uB[otB.I6]ncD+9=Ze +JEMf_,&h6=$;cMCJdOaKJ.?k_@j1rH"ms#7RQZ-(Z&lQqg?eB'L\q#kmjB4Yn.C0] +!m#%^i(F=V5QsC\#/]Z4&Ii-#K%co;]Z/FZBP_`go..mO!!iUUp=Nr2j=_iS#E#`@ +3XU+*@#@o4Y&<26ebS%'p&9e=5j@el8ksb2 +HLe+On*_D&B];0k(-Vm.k_Ut]Lj^H^5&Z]c4H9:JqoZZt+)4U\7nha*(T! +#Vo3_8K*3`3)?MM;jiIKXCZ*3>*a:8H +JBnO:16\nD/0%nG^qR/:1TH`3rP3p,BVilf@QMPSNRQmijZKV@I<08drRJ^pf*D>c +Y5ROCn[E.chOb_^UN#dBB.!UtIdBe3C+5p%2+t?G3A;C*T8DrA9p*aj9Da@!A-#K5 +_TJ.NY1^bE#t^mMWWr75P,Wsn"8U!V/L8/mRuRl@;e`AD]HFqj0CZO#k@q-S")URH +(A>d*kn6&82.$FaV^gliHaJC9iNQM@J/h:p2O\9RS&,,SK4>m3R^ho4kA-]+NP/m@ +ERUG-bE@)"1^*T'aZLi0$@HmRibY=\pJ6Op/o0 +2gBrt;?PN[.][cM:@qt9>deGqUDr&C>e:*DQ&R01;t4-<,3c&1BfrsE*+gr?@PS6K +:ug5HEZ:-kdF8qn!!iV`]iZrYbf4^44=O%oRRh8F7b<92%k9hVb2Eirj>N*SWO0q_ +8Wu'@ECel`&_3f`Q.i1J50h=]cNiqHb6+Ag$E/76iIDLH!;La@4dfLs3:Sg1FTI3ajh.P5j2e^&jT&bf>g^;ZsA#1[r0[PDF)rX;9leA +Iet!f)j\Qr$2bSAruKb>G3ha=IGoX'c43Fo^taaR`BXsXb(pKrWrReAho]oO-tnA:(@;e1+4+Y=2oIm8=opCk +oC8=eh>Rl_>AY8l@ErIfkg:$!S\G=X>hY[F"HO.bJ!0Vq3N\`6eMhiXJ+:r@WUg@/^nGM@1h=0].1eqos1J(" +CQeaJ/.atrs5r%f],P6QkJ;<0*DLGsFb;Ze\H(rI_"ic4@lD#I0!u)bkL]NEgbh@J +QY;"S[dqNguqEQiXX_9cM-AqK,j]% +RVY$tH&U4VR]Lg@3:cds`:%nli%>!#p?4_YN;V#panLKV^DDq/rTr&6StH"S$bBF3 +XLXV?^83[Z[%nN5^QtaeMPsV*BUXa[>60jYBUMV3R?0t[A"+63HX:9aB9Tnm!i?TM +,s(`o(TPMSRmC7%c;KG3m.2eqrc?4p@rZS+=DZ/_>h_T6EY0ik0eR2Z(=khU0pg7] +MeWBSQl!4aVt@'&.tIY0kmd=%hgN$_)>J^)+q45Wrr3/^5k+e,)"I$1O6_]WIX>A. +E`V`(nh@cIB41Dlkj'`4_#IK&7R6DX$8jX3*"6jc1P5_!)TSSh$_H$-9b6i#57tt7 +cDA]s&pptt9dAUppoE>fR>P&P4tso1+9CdF:Mn]pFTVU?j6F"nd`K]CP0+`dC`C^f +_D+?L&ICGj!IT6ao,1[e&5rgp*DL4e-GI]1L^llTro!U1)uCjq'qCF1@bdhR'iM;[:':p_:i` +T+6QUJdMMUHTe"pp_9c\bJ6gRGo_fVK\%*;()11B:o+oWBd;rcA8*eD'VbXJ[\R2Rq&h+XiTlPcctue +q>XOrs"A)Yc1!)"+ZC5;r&=W8aXj*VQ2lcFh7qU_&to9GJ)NosDta4q&AWIq"Q11R +cZS"2@'G&<[='Q:]q'.ZVnVnC"3s3-8390n+q[K0U[8mb)]!Y$T&2Fh%l;c[9;;T$ +qE#.klcP`8nfl,a-M3EQ8`js=Vaq^2*:c6X@fs_'Re'RI$_kk)).44,9Sa]S%Y48b +%2jL%RGbRCf3c;aG&[Fpps\C-)p'1\!j%c&)rUdAr`3hNBaF/Z(@/9hGH)B':/K$Cd2-7_]\?^fF_rG5f5j;*cOEG5hoe4 +JB5VtU*CE[-VY6@RFe"#gQR'j[sp0>ANIMR.S2Z:h`[Lb"^]cQ\5k2[D@\73%DfBN +Ia7P9^`oWP?+&i/D+OJ%<83)9F$.0`,dqP>V/n4i0c!l1'Dm#&$>oB4sX]W?FTMo[aWkd?skI]j$_V5to?bG,=inga(tVK2dn,P!C#[_#1,U"ihnDU0N: +$"O(D@_fak5GFGn=rG36KUVVUd^m4acWk?'gAb,]s7;EVh`NU,>[[)EV>W(?*d?n_ +mPajs(&H`$KU+Nn='q$0p+tGDlM+q@boYc[k5Ij\ptHVj=RQWN*RQ/>3#b5'CI<6X +jg3tH-d65()bI_MV;tGA[l%-oE@uD,W#^F$X8/^!+6fF/$[5if:.DHA8&J..g+O4X +7mC)R4+&#;7fHkl$hLjXA&bt0Yc8DOd_L7TNnsVM>Bp%RnA'NXNrMBL/&MfR)$r\' +GpN'e;q$3%k)m%iqq$;%98A7=2ltpjkPptsrbkoa3kG))P7hc+eh!MP\gLGYU^nk^,Y]_e' +'fl/99`%?hKqJO!jip;EWAX]NFSMQ^Y!D8WN1ai4TGH4s=Hm2M>+jrpdb42/n=LII +-?l\g%>clO9&=)NRFiGBm;Tc5e802(bdr3f?,YAOD!Miq-82p5[R+'Mp^ +\gd(<`lO&W!qY%@jK3FHd;_:Xg)9hgeM(Lo=RAGX"HJY9ZRf_3hrYHbhHs7!9?s)* +]KAbQF^mP1mbP=/k"p=B"u@UKXX/i`3&KWWN33hGmIg +b(5X8AQ0nVU/%T>*?G-V[`W)q?t1W;PNcp`p>H?E@i:j,!q6fOrtA_nk>quqV[TF` +s'U'l]K\.WATAaf`Gf'&%\(f^Rf\C5NImDIkSjabRXPKY5C%]7L +JH%sVqi(QZ)N,]L9r2hRLD2d!B7k@B/E&JdAT0GF5?,#Km_>c=jL]:`WDtL=$n,,cH +ru(hsp`$1-cZm0os"psgAA4R_KusuYqeaT[2XT&FqJn*)mq/0] +n$dA.=7`&QJ%5)W2?.C4(FS%SpqNB#jJ_lM/jMeCs,DF*1B+="6ap`V[&k7ZD48-) +RMr^(d,aTBjR(@Sm9d]Ys+3Xi%dK-.SBD\>f9a.C]mT<6`fM9X7%LJs(P#*oP_23f +Lc"3^)J_d1lU2\AOg%XY2qK0LdNcDpA5P4AYd2*LlG:PR8ORE"!%H/Ng>6O`\LVd8 +heIYMjm[1o2YRfG%f99NHCU!2%Ym!fUknO>Q2$Ug.;h7mZbtp#T4:a5$aiG7pf%_, +efgt:Brm+;BlK?#gT(hFdRr:/C=Jm%V"n9'c5UVW@\^Qj`7E,55+9/G,F^\>AOBc( +2-'"l]9rHq\nT9&Afj*L"N+p_jXKH2H"os!S#;c<91T1E`]+hXHsr%U=16n%P +m;On`gZMkf+?29#8htfe9ME&PLu9@6fHCGER%W;@Ifm87,ka(2cg6W-5+&7Bl +"13A*Xa;3s9TmA47=@o70Aml14!l/?OG+hcJ:EuDCmtnN9LUcKLXdXK#QJ/NG>?,c +N^PdUs,4q0B90GrB='-%X32N5r-E/D)CPELmUP]W5cu?rgQYL2)!'(jGFCm"r +@kq'j=t74d_1.kc-1TljUORF;''*2Dc9`GQmNX.bW2opH'^[;"c>K[!c$a-+OIuqqSO<])05i4o-d2o<[E\V3*a0`J+3bHUZXlk`L/!06l6 +s5%jBZ&duJb8XLLm!"g6C[^)qgUEG\\=%!;SaX4rlKWor-uZb`.*I=C@Z.hS7"H;*Qkcr,5[Q[4RZD27QO3B`Y; +'J$gG&GZh@e5@kWZiTZO:JjMGr.Z4lJn9Pg-5 +TCr"9'qgn`QntmQM[KIHQ#A\"!&(P%%D9k+B#hV=:5IOg+TbDR$K[#cO9JhPc +niFJaAPD9+:ubM72?/U>e:@S4cB"*M/KVn*X9ARD_])&i)FgHi!),M4nUE6sqF+ZD ++/]@Z=T&!aSdS'-rI/=b]A2YmA8(qB=Q*@m'-&b%598GV,)i!0$Yr]Gf-#]jQ.QP8 +L_1%ce:[c/a1>lDFhTup3KhhClMpA:I4$XcN4QYN1iJq/[^Hk>J$j^u>1Z\[\.m\) +RETE6cfmf3K3S`/p-Q2%n:,XJ_p&"2?/"cT7`4-iRHCLj1TFA1_Gg%*)(1!$c!MH+%^3u7i9E_T01M\ +lU1s4HFG)mVE^`\c^0Z6Q/U99kW2.K]f``@.'[nG\K*[sIopi35T,PgOjrPmM-!OZ +_L&?@>)qBm$Xa/tg?hbilM.$si;me*GJ)gBHQ-jsm_Fm&!T8Hqd5Q362[.cVM#X%m +D793/W'6:%DW_)2J.AKT,.HQ\s-eM3HFNr7!dRU*>_"$*?4gEenLa'I.hm7DGC.bE +@%l'0G3:5(k,5MG7j6_lCl_\rqbs286(;(EX6YrbpEr;rU>GBshPc=GQn%Y.Z.6^Q +B>U=915B3D&-#IBF5rD%=18TY+'W&#Zi@;Yj_$kqdIZF'8Vc*bbh$_keP'f'!c1SM +KD9Ae[uC?Z`;e$>YMT3SENu^\M-Al1.fW/i%BR$GF]G".R]$^D>n#8=^QHfrY^c4N +SH@0H)Ma@o8,-p-opcCdHN>Htm<;Z6D;O]mC_!R'dF0f)aYVYjZeWiIgo]PGNutI, +)5^3!'8-#M]L_4L2S9mkJCR<3+"^L^cf4uNME`YlLV[?E/kBjY2Jl1XJ!7=\#&R(p +s*WDq\VWE;j]oTa7ZaVQUltJ'$Ws=K=6kGjj$,C&WE4^hqognlA=O%[WPR>c_:tm, +dp'Yp>ff(5Rch>On%G8.\;/,pT,2.Y9/(o)+H&$&po>tQNkh$K@'Ln^Dq-jXe#M=g +;,fh_?tksG0c(34+k!%)hr@b<#;*n+3T'.G]&2ak+&U:5D\]37'[eZVh(k2L0c&Tl +j&JosJC^!o>29tI5*V!Z&#KK;[a+7Xs;[/1cHAO[4.#ds2pOb*F8"AVO^KZd_j#V8')-#oO.rTf_!_1-]NoSS=YNV*XqmH:<*Xu-4p3<7gc+; +Kp,pj\Mb>^!rc!7[3>3NHDaaB8ai06g-)(]4aU'`F0]!c?iA(:d\Vq#=#PiqY4]qC +m?47)GZrkur#2JS3ptVo-4co:0BDi@aWGbWM9]AgcV"QC(DLfZiIV=)^L1&'4A0V< +/)F)I:_+6-`cK4C]W&-!fLC]_6HAMg0)m$J@9_3Ch%Pa7I,#";W/>QAR_ZuRS+IOMXu +5I+H6dg-aD(e:URLj?rXH@T\CY6(?mZr%(E +h+foM<`d7iS,EA2qYkZXT8e$u09Ml4s%33d%uhG(WNk>p3VM!8'^a&e(`)4B"Cj&L +jTo*"R0uhmJV"n&UWG/MfF>48$aA3Y+lssG3OLoUkq:SeMt#WemBk!\;VOg9Dn_]b +E$r&DHL0_sl"Ph8nO#=92tpU*gMM?0%6XSH?o:YjWJa"-jEKu@ZLqII^PRS-JA7'@ +A6ZDu@Gq1*<@[uh?(cSbIrYSDA?$>O'r;FWM'P^p^je,!rs/SWG"@^E/"`uc!^>TC +8tP,OikNY`^<[:FHBOo:[;Wi9u@uK13 +e`PaJ*4XC:ka'<(5jhK(jAXt)>=kC36+)O1J3iW$Y'hLZ3^4M^&LXn8,'at(1PHoS +g2`]E"qg#C6sMi)#Pk9YMb2bKPoKpmpFZXL#CN-s +S.NqVF=9I*_>g4U>4@"s^25&k3.JROcphD-(ddnfG:3:FEoeqmVE6,j#F15'$b@<" +./h=q`=(WRVW-;WUYB*C[3GhtiJ=^MAPaBgpu[rS+^I4$8A[mTs6S/$`N6e-duq[k +P-Ld6Ei0jX2TK+W&=1)VpVJt8nGeH-c085OGoOEeeccB^Dp@J;][#uDF7q`l6;G:q +rm-j[(:3+A!Rtu0nK1=?5%IBoGRiDQ7i@#j8:p=V./[hrJ%8?$p^`8U +q33?qCLoes[.Z#]@47fU.0/@T7:Hoi&,o?G'oj;O)bOl`9PmfW2o7E\a41^>e1/?d +qu!(CI1$dPGJqnhXCIVLrl>O#*7 +EIQ(&6:nc5[@ke'Srd3m/`m-(i\*1%&!cV/+,tPAcO\*XrhCs('OWkkm#f-"94!j$q%.2U'gf/YW#^nkM7P2=kM&70,T0W +o5hkm;YB303?9"8%98CYc6feKVT]nK5NTX'ZUe^2=G.&UjTu>-N/'.lMACT:;2X)omFDr,:`O6oT2++2FLkJPqrP +on"su1Dd8iRuBE1i-d-oX^IW*1mSCH*<0$f!Gl*sm=P.urs\nFIkAnu[J7?jH$`W^ +oY/[3\<:L:L?H-N_tQf`d!_?;#+C'lMCBX5AB%;1D/]gg/H/-`Y\2c$l]CtY)Zp': +/RJ2AEWZ8&*$2Q#hJMqqh<90sdG^,$rdi/Y5/Kk=q$F"o^M[Xe4jH$'[oQXicA=\[nJ9)D1r[0Hij2\d2 +kPqoRX*Dg%;W,t/IKtQ&Y;:K37LR7Nr3I9UiTkOll6FOo":7ZekGq1os1^4HKE$FA +LVFqV.c+^XhO`q\HX&Im+;tQ%!WBh6Oc[/'n)mU;B-Y5mDbftc`R[][8g_'chhg!O +#puPV%iEnZ)UZXr?V$@gB*Tja5LqW?]Orq">2iO-Edfhb;RMTUCBG\40 +@oSm9Li#Hc0Ua7,:(WF.#Dj6la*#(Om6lt+X5SK-Ue:c)grlln&R(rcW`R1uYq"*k +=r[6$(gfLidDdA[\:C:&k2GAJAR$6\^dq]nJq#-6TSTP/,9mJrJ^NcVfTqXZ!?2pB +o.-Y<*7odKpEGGa@@PO/7Z3mlLOL%$O9?Q`5H`fpO2-7\?YFp.?D,."Vb5EVbDj8; +T*RC1gk,E:\%U\\n-JqnpfI08OoO]c.?k!8j6ab/!]O"l**C,H_4(Te^jfre#Lo#g +0fLNcp`h[aT$lY+-Lr\Sn1PAu4`0_6,kQK4)d(l;SGd!FZ +/kt'h\4+;[K>p+E?\_J(FoH()%@'R8_0rqd<-q6:/% +,\(/d#qstrY$;aX>N.jh\)U2i@W85*04j%,h]*9pI"#u&]%-$7H`c,-/U$kM*:F9+ +IM]NmhVr^Ve`fr3h3/o#a,"B(gtG\<=RB3ao0@7pPP+AR#C`g5q4Yq.At.>Q:)c0K +;Ir\BBQtJj?e4:g(jIO0$ZR2'4Jdt,"#<"uf"+GA!%FJVeA!B$8FL"S3Wsi1aidi3 +#J%XNUZ*(=SZoZs*q19.j99B8(ZSGilB=`jH$*/>!.OX3Pf\U[r_kXA5`URWJru'] +&'G%,!:SZYU^>Nt`&`9XZhrp+U^ItDjtFdlo7lCKP\cie`/1(PB;G#&02G4@Ii+[` +,(BfH`rC`#FDT*QEo?Y0(NrPY/o'WlT +i.%L$ohd0ts3Yf%>XHbQI+N=V/IAKSQVL"-]Zq/5JkJP(:FBY@7\^kfl,f`E(k1IH +9ZotOK/0rknc(b]g2\s^*ARIgGR`^5BNWNKUV0Ur1a>*u,lmr]'<0/s19=L??h5b0 +l[qptSk<[k$0Y?9gJ3,<\.X<1c;^4cnEAfLr:OKb2[/BGnN=S/n_?/qnGhiifA$hL +?[WeB9dj/glU;0M-D-=A?_6\6PHT@XE!IsNN?@_g# +,dD27?8*AR3YWg+N"s30NCC3&!c.1FK,.VhYTqsuP9]TIb(Ea)<=,,8IJuT$lle(oS77=gE +Z@4L?Y!9.ba6OUV0>S1i.1t:jJI?/#^>7cG(k.#os1/phV#p[$XGQs/f"WC4W&"7Q +=RU:4j`PrQDY>Xjn;"F*gdN\9pFYR:5Z2"Ij7s\%=2UZZKqP.OMjNqd:=lBfO(X[Y +3'S$8gnu1/F?3*jQXg,n]n$o]i2ZLdk!^ZiBiaqi2TA]q[kS213'^Ged"R%OTjYC4.CfG* +iui\IFbc$Hs(_=YZI7@u'E40s_^Fq4Nc?jXikL>-s+(+^s4-Xh'[bdq;_sAS21dPQ +TWSPDEs\Rt!'PJfO*XA,pl@hYq[dZ/?VK9u-?a`>d"Z3Em3'"UhNiO"6fPldR\9OLYJ#`>6QV;Ig +O].q&JBJPWU\8A:m,m"fs""+L_7p3CisL;_;]Pm[AKR"`S!Fh+LXq>onNUB)Q0.%j +4Ss7`c`J)d]4`XtMZ9PH[c@M>(2`t<]Ho5bOI*,^UVQIpUYkHbcekALI9g9.^$bB. +D0C+AXKOS:s4$%K\)-t!LJop+2'Gtf35KiEPM#V4rK1Hds2!d:7^J`AXmV!]TCf[Z +aSflZ&c0KQ1e3M_1Tfj#JM4KlOWshd%bf?rg&Y[bcE+pIp!p9f@G1d. +XI$jLd^lF<0tn+SMXEUK`UF/GF/VML0g)TpBDNrNFAEt@fYV>gW'uB<0Tm^_r'Z^1&HZhr/n@HQb\6pjWXB3iIKi;Lm[&m0W%# +j:9n\SD:!q_F?>o]m)`]NjmTLs*ABhe>8-J(Y8KFlF=(gb.s5V6]Sm+0cRX38SEG7Q,bk'gUZP?cI5W=hZ$2uGaRW&H1 +'p24Sna1Sk9Z4fB3!% +jRiqr3=Pg%#F'l@h4YQhJYaB975R!j=jZ8/ckTC4*W0.Z;M*iA3NT#/n4@\'@7tQ* +V;!Zt/2#`L\Ge2I;UV8V&[qOENi$7YPc^6bh\BNTd'"9AGiqgu#LX_"WPF5"G$hQ@Ht +>9d%4+(rXss&2U9WM[M\VOIjT4i +=8:>*`B4ig2fd)46o+mP]Si[#3l(be:bbs,PdAF[$sWt%^O?]$Wm;.pEs9UulW+7G^^/n4Jq +C)/rYA&1Cq3J!3 +cU^OGBR0ot1CO3@GM_DDmL=UGRSZA;_=<"%VL:?o[#dk#!J(2ThS>V=K-)_,JH(+) +)>j$j`8=orrH451r\'6/R59Nks*^1Qr24+.Z0gKL6E4Q?WYJ&H#cpu.; +@,6?/A\Vk\TDN\E/`;r697eXiTiD3B%0Nkp6OD9309<-"7b79'eOa99qdUQL&t7c" +7M9DrCcn\632T7_=tS]d(M#_:_aW]p6-Hq=.gA=Tn2,/e)As*-NAo[2e.Dfgg1kuH +iS^oaLMmL#mi\DrFt@Z^%'M.bU#HU557dZoC&0E_f^S/^K.F?12]<5*;Jgk'4ou7- +P0o/KC:G"/*O.c:9!SM$^_lX\W&Ap/S^ahUpFrL1!^U-=i.%Y/JUheB6C5^urqf&R +"gg4e]J$4h-bQH@W!rSor"&,_IYumNJ]5L!@`8:`iMDA1>Jc;OO[Vuq&$+K8.?;(< +,jD?dq]h5HHJAP(eonW:U\:;F"^p> +WY/`]11L/gmK/[us(1^"eAqg'lDs_&U%/?k;/SiWGQHS0F"n)G2@+PUi:+/Z4:F4aQ`k!ro-TU"R>eE=S.FSOeohNe"EbIW5R +;ZCLVLQ0LC5BJO%B3h11StDk=5=^/33rJ)/!,)LO[>+W`T'L![s%Z9F)Y!iX +('":7s!1JI>oRKQdpZn59lJ]+/%#,LbismDYFfR"Dq*FpCcHIAalXt%j0(fDJj/'A +ARP+?>ka/0mbu7,0$f.',Hf)3QhX4r&_`PXn4,D"+WTn8EJf"nlIJ_2<^ +D4/o+G\l,ZI/A#CkZ'[l$`NF`.\l0< +Ep$PPlY)er34=Or#<-coL$l$+`8Q52$nt34s!"KK_*=-;Ud5"Ki-:7t"S=GJe&qGa +&e]oA;37@sY'C,*s5="rs$?"6s3Kdu`rA9K.6j=@ +^dAt=&Y.1FL`-4-5uf.lFap!KcD7lT*:=:>Q;9q._C80u$!qRfo/*utfn_U'?Tar5 +HCmX;WO.jP9+(YK67Tf(DJMqh]+]Tkh!#-N%/biue,!^Ol&8X)=%+WcdsG]UB]aV' +TlhPV0RI%]Qdn`Yh)T5+H?HYt*<#f$nX5=$_>iG`eNW0J\t<9q$N=\Ao=D.di:>Ig +#/?Y63F2)+/)FSNHCI.i3d9d^hXK8-_^mJ<,CbCG=dbM+Jc*^q+9`TXBnm(H%4AN7 +s0B^lmQe4K>5b4"L8/HpOG67r5G;u1]3=K&f'??\nKn;:rkM-?E2>[kQ?IUDecGb@ +#:*a.V!o@$"#$=+Dl+6La_bttj9=OILLZeM6`RbH8A27Rm9.2J.B2k4Q,A14Br&%4 +BPi6,"Wb."(9rEtOaBBlbG1q%^XnkQ<+gIi2rFi>J%,J@\+KuE1`a;%1QRjR&c@t4 +?,O(uid8JUdenFOOl%&es%6j#legX7>931["G+a+lM:0(V7F%-H8ZQd&$D2m=#-1? +V@cGf5b(;#ptoV+\'qA`r+@X4nQuu8hX4ah8jLH!g3jA!@;A,j+L%e%Unf<4W&^ZH +4mH0?jKn:,P,=Gh+!b/dZ+#U4ctE9;Z9JV4LH?E;3>g4cG!#:d4,\')M\&C"aq:$> +UX,.+EW<6?O*!;Z/H/`B?QK&c/jM)@_[)"O;;f3Y6):!Qb*G&>BRHX0m)%Y4iNV2?.)di\kA'XNg7]MQu2s*i.`OAKT6D-dQ3?cYa-jcpeK9 +opUVN2>=9$fZk)#FSf-SJH*j'4^U@+M=qQ]>$opULt$LR8*4$HMf=-!QnN/+eMKaJOQc,s.>1;3q] +T8Y)k,7D;2Ga[8AP9ra+0V0\'1LF+piHDN5!A5_35ZN+Kh4i'J#(//+#Q`j+s&1[Tr\j$!b]MFM0lks6`!$SH8]G>os"BS+/_X)MJQliNLGbY^^#J"TOnh_dD(a +i'5=pa*14tFj)+ek)t(Irune&@C6cR:%(mOinEO6D_S,!,-i9fT87>M.Q#9?b`>4P +%u3(PL"7jt'u)i`%s6-i+[t^NULN9o9o"35g>89YD16t4nhRb.QC9+Big[\Sm@-o8 +IMpTepTMSlF%7PO'<%BK\U?T6jjgX(fotF)V!:YHr)`(YPc8fEG2HQY(j0hHB`>TY +F6gNbXE?V(m0#puH!:J[KqH@&eL2!Wnn0u+l6!-/,[UK)tXCcO00Vf9>,A +%A9E!Jack*!YdAKbQT\E5TV)GJ`FKV#6-HI,fIlE0Y3+,C:QRpFY/c&4]hD[_?EaN +.(!kG]Nu@Q"mbh]V#XnGQi#6,QR?'6NuP3MK*spg!<.2phs5<1cu+%DEIRl&T>CQI +6U+Wo\6`q@Fh-27c],[h2?"A0&DU3%rZ/)mkSgPBlK8r[4)I_'4'W,S#kWG0p&!>h +Jn]#]?s#5A?i>Q+ZCmY;&3?:g=qF4o&H%aa2"CpE@_?2V:/`cs#95)*1Y`M,4m54QnW[^mom]fXf\;/8s>Nj^jjlK(b>U`PQ+fs +9?KJ@o#o=(=h!R32YNu`gTS`Je)D7Ii0ge2=fp.lhot]fOr(c,,>+#S5KP8M<)XY/ +cY*/_%f^&I?B!R[KEGClAui*T[_?C<;;lW1b5#YGbKn]k^GNRX(HOe?2I\mL7+K#o +T\80c^TuS2^R(BpY<;;qoqA#DD1:eiI3SA9QMOR"o!SuP&V;0Z0qjiH&&l"Tn/p-! +e0[cG3DBpF:Si$;cOdo>e%1pO5laf>4an\(QDT&HJYq1Si%^o!ANS4LJa@/LBaUK)E[tcH'Z3.T?6l>L%O\E>no@C)ZZp +H%Yb)liC#ZjS4cTMk&]QYWm-%BC_i8GL!O5^=;YuRZWu^;5K!;Pf\l%7.7oTJ39CV +7C)Vie;).?'.a*o2Yu%W@k_EoIpF#nT5S'8le3';5?o1XT0;u.]Ft,I[pR&]7jD4L +rr`8,8Kp4l6r4e!4.Y$8Ge@5u>hSQ;?mrO*/?F:[\-DL'=A0*YNM";Os5C2n2okeZ +S]$hFE%89:Nq>&'c_+L68@%+ur4J0l0>n>m1J4R[8[#[IL,ZD1Vm>SA]= +iALiJ`f^$,o*%hW!@beq)[pZ'rtOo]Z-RKJ +Ip'6n;6/0`7Yc=Bc4AYmT#(ZP6E]o1TmT!eA.[JCTE +qA2!a4qIi[&qiH7,ZiQfHBdG&)KCkm18A)+,97;Fp+['>bQUBoOU_K&KUk:AiG8kr +7lYs,%/>eZYiW"g`oPscDBDpNs2]]`@@W*eS>S(m_8?g"k@Ht4^rnFOdt"t/7fE\F +rO@hKnL(j<-J\l2Y?T-#=ipQ*s1G1'rakT^c?ort3a,1A +Iu=7Dq:BZ1?Wd4:2mH(]PLG$AKDBkMs&9'R7>db<.H&HST>:I_rc?4H.ep>I?g@7A +9NR.nDt!qVGU:[@Rs/NtcYer$f61K9DH^cdA&s_o3e'-aX8&;t#4cP]s%*tgB!LJT +%Zk;0qE&JJ@[C#Qm_@8^gobh6?0cZ(i'o@pAS;OpaM/p3Q\n\.BR^S\-1)d-:Kb`f)+T3s%jQok+iK2AmkVn^tcI$_#Mqf +$.&g`+?g$4-B<4E*iHu__l5t5#7*:$B23#]04rBDcHFk=.Hm*XKqX!G4oY1W@n6Y"qKh/)CQneUGH_#Eeb"?o:ApGBqX2"Z`W'jP8ATcY\qpA& +e,%u(7Nrc46'W[WFg'5Hi4mLN[+j]iBLqf,GU$J/A5(0;<"k&Q%Q_31Y!N!cqH,E2 +5OG+(hMY4Fhlj-'lBaPi)fi^Vi;ZJcq_UE7;?r`8l]b_os3o@^DN:oC2M4o0R<6he +P41nDqngIHL"!a(EoihhKk*BVIl=o?Q<8Fa/Ec$hLtpOsM#U8SEBU_L.;Q>>p*SB! +gV*2;q%YttU_Mb8!)am-Im-Buk/-Wb.jLPehoMEk-b(-jN+@gjFMGp:`:&K +nTl$IC)]hXaJ'XM8asK'3WFk$RUAbf3e^omir;d:_G8],M/Se;X:A#0p^\>%G.rn^ +#f[K]LH$*Ck!-#QZ",",`?hsQK\&kF98/B:4.ID!,Cj+hk>L^PSkM/Bdbj4n_A`Df +s1A433m>!]fO.9OSS+N3?YcnE(&Whk@D+pF!E,O>\@/(ZdY5L6W6_Y6&R#_EbCciU(JquK5g0BS +$g@/1ee_F@%P4AA:hWj9=G80!Zm4KHj+!uQZOUn7'Q3MkguA48g$aAgd"C2nnq*YZ +Na$kMfpKgkcp;N-P6GVAlX3=;^sSAL)4^B0ESU_Z:6sg&@t(5;3K1eH=gXqaY06bF +.heo50=lPJIqD)t&(>c!ZZn-e6r00^/DeE,cJ*"ms-X5ps7%9#=+-Be59)tbVb+D1 +;5gN]Yd8eOUTVAFY+Mg;q!W'pk@Lb'm),.\5M?$ofL&hNB"S+lP/he!8qQm$CGcNC +q"m',F$mG?Go5r?-3!L"j1i2#Z^h&Grs!=?P:&?l0)eqUrE;RL;Xo77[KbdlhNe$; +Uu'GtN;r/i+-d5l2LdGkc2UK/H-[9h-M:u\9#Gn=VQi&:9Uf[g_mTBi':f"!pX +#*a6*Qf<-Y$Ug.^V,?>K*d:6fs+#?kpqq4jsr.0OY2?J,6P5`@2*TjJ?qSF7Lbj$CPAu +Jq[GUJq(RNncVs\kf=l>_Tj1OqrhbAr6aBc@sn`A>qZki)FLD>BB=(&L-Jo_rU]%< +^FkNFkM"S'huYn;s*8JPn#MZP\ifdQf3\BC2hLgsc%!FKPe9[8)/!j_A,iUJma9*& +\DYc%LA@U80UZiNVpg:gYhJ??QLW>uM8_OBu?HCQFO`S_c)N6M1+* +NU\X`b!9luo\fc5Z0!g>+0KL=bL'2W+2NHXC:j),SFX.7?J+7c6SD"Fo47BC"T3b? +5I?e:K]eG7?PpS6mk9a<*O&9gWi)C%F8]0-XB:8X)A<KILB!\G<4<7-ji?KmQ5qd)*m*Q'm")(B?i#?Z,[f^!4l7(>k@D?GlH8_8TV?:>OfK*[s@\ +_jtCWPusAW@D?_lT2u3q*(;jdbi\o_=rN\SBa?bVO9S[']7b(]0Q9$R#H;%Y!7P=:G1]r#MA_prZ@8VemIcgUnUdHJN9F\1^T*]+oH3`R5pO +kl;88mU/_8Ep\iGr`aF@f8Fh/+6.#Om:aW3hfq0]@1<$ls(\I)LAubb5FBd"+Ff!0 +3LR?jP*[o`ItFlG5?u!QAh<_tVq>XE.DV[jN?^QE_^G[Bi)0Pgql1]G!, +eqHO7e-Iu\pj;Jg3H+*6CuAI!VG]G6nD,U_e.6EYeh7p +NTL)qTrk_dlhL\W@J`$Cmp;^I5CNXY"]0n[m0,DO+K9kQR&KSnD19N8F]W8qddgb. +HF?rM`IYF#_hJ&j_]RO'b>uj7 +[P.Faj<.Tgfk_T%7Jm\kNudOqB>03%iID\>bNDVmfiT-MO&Y4@VK=Q,F#K_E3`NF2 +LjS#d*rkQK"9ngla8U3mj?MNsE^p,ii;^FeiBQm'_>c`?LDg-LT,C`j.L,o#@bTZ\ +QbPU/HPt^$*f^1dj$\8Ni'3FZ-LiQPZoJn*1E4e((/d(4eMJ6b(b4c?;Pl1LQUcouN[Bdi4QA5EtD^M]3d' +$ne3ul6]b$Z,'KNLOZEcn6N<@e]NO%='>KP)u"NJHa#FJf>kH=)XW!kqhSJWEXIlj +]q4?D^Xs(c.FQ;3pIhlDbH1%+""X6Na@F)=.G%c:`CLeIjI +(O6h*]%mVk)a&$lnNT7pbW7E?8_lJ=eWZ5LJ40ai?0aa"5ou">1hML\UnK +$dah*d"UFJXHu-=`2?SsD$l3p0o>NX640CJ^TZH@r+G[EpERV+f0fFF%).PMVrbZl +_gSS?h'NShFdi.fR_)X:D;j^':4$!CShR&86cNa`^hfi#3)%Sh.oe0o)j_Gt_%n'J +k1.+=3kPpU]X:DP]EZE+%EAErX#(Ccmp_,;^MugO@HW.i^JK>?18OqM#q6dZRheA(,(b>$O.fD +a5Vb;)%<$dNQ):Ei`ddng?J5]"dsi&'M8.bd8lSI]eGWAkjF.9unc1AobSO[$%0r;G0a +^V?0\fnV(fYWjb"+YNiq5NE7OSN$,u]Fl&okIL&fjR@7i$,YZj#%29pWUiKCCbea. +E@_*;`Ze"PB-O'a.Y>,-!qJ,r?%\hn6U@Md:YQF?ASdMf74+;RdHCBH!G][6>7/^9 +g#Vk+FBWsKJ&c)Eb@5kn6D=ar'l5FqAau.0iM_D61#cDMO,Fg:j.)Aek4F;en#n,U +0$7L-ONj<]&Oen`>sIMhRY5`t4AW&!5-s*';oe%]HAD)r^%/T8jDP@1)$'1&F)G@'djc:s_=5\RcG^p+k+H[Y(D,J<\ +.UiP`0eMkuZ=rM2@c\_$7/]Aq*J3?9T5p_-S^)p%O)h(Hh+1N*E$T:LQ\O*7)X3<" +k`;a]%4*'RHbsPg@U$hcKd2Fkbd27GV+*m-2.#;>Zoc^m?I3$FLJER)%ibXlYNTrG0`$YETuU>AmR+/UnI: +hl/d6b;81^$VeAc55=LiPPeX3:Ysisq/lZ$>*I,M7t0a.q7ZC@m1/IZ4\is/O%T=l +O^%?@5Rk3F7lV,K6P/7=)JMTl1qH6u>1MU3]Z[c$bjX[4P=4h-?/EO%YJM'i,%4;F +]Dn(<^PApt%pJo>RXH8:GsCZbHeRTU7,W-i270#4Z0_@_M,AF3n]lntED?airE)RJ +(af^FahlG0]g%jSj=[niB/:BBi)An[.^]N4p?uhAkr]e1\=4djQm[3KE6'Sie +KVs9tRpM2?Y+NTq?b?9np3L&RqOTSLDUs+ff-@lPf"+UHp;=ai!&'n&Bk1=ZSM4E!_I1EFp0m>["bXkMVU'EXj.83?hl8g:gN2sn.P-'eYl28[ +j^(020pWsqP=;nQ7gnF`*J(QI2K\&kEG5T1J0jI6p,ifoIeZ +AQ%IUH,WIcS/#[e@\Wfu&W@%nYWa//eXtoeg!7`#&Kt/l5N';<_mT>J**g"HHuZS* +n>Fsr1X\j4bKR"C10gfhoSa4V+$2(ncP06i)P>1=-2E?qZiYK::(q<2@4%uNQko2Y:;IX_Gd^l7@Fl$7^Y$#P1:9:le[F$FMS*,JG6'ld` +?J-5A#e]iq]Fb"5cg..Hl^tpG4InRDQqsFfD_DO5b9*6=f8kCjShQh +T7gdY9*^t/N,JguER/$E!*Oh/]N,eKX,(Rg_SV-/rVfef!Q;.[]l'ZEcO+qo-1-'g +X6C"oP!mg(.0,["dN\JU>!*Al.7m3P7gt3Y#K%M6*1"gPWlq6;e+%FM7klfVB*n>d +&#b`f4JtIFQtK9_qbu2M_[/VN3';"@nlqC4&'spO4ZE/9bikN'POGM&IRN7X/>1Mf +gN'(XlOWqlGPIL;Mf55V??p4SG@tjFDqi\]m`Kp[RpBrQ9jmdlnBgWp2@i39c[&j[ +%fu^@^=HI#=,CW3M/2R@D!VV5*JeX!`O>Dmf5q7^OT^bi0n'cKC%3G`kVdq=J*RFi +')F_*o!g4>Hdii#[)M?NVba#-^SiaB8QT_bpiuI5C!goq(\kX4T" +P.)U@6n`rmrtJi8#X]@,I15,KofqKO^Z0e4+";`aB-1Xr,;Nr%OoG=*Cq")E.: +f4ilRe9'?;jg&`W`ss]iSQ^J(2$8&<"n0hAo0`9B@H"ed[-^_?7B?Lns,@!J-L0N] +c`0.#_4U(E-C4d]Oa0bb6\!qDf']*.%u2ooI5+_2DE#b?Y#SX'=EE\,^$YbV[t8MT +k;*(cJWOWb@/Vg3rYW:,gP=fN7cS3gAgjKm+K7d9]Vfk4-]=j#`7)(QZ?8.4&b2ZC +]?4*!2kmL\<8-NV])C;2.]`B/YZj-tLln]5C(L)(*%aE:r!H[i-A>Bc9(.<9U$DIl +)+E\c,u[s[':C$IhI%8[%li$UW=uruWLuY%SeicmoB53'nqV5(4]M.$*LtFs+CSoc +cZ:^/#((E"(p4#2a*.X`BlP`fP2Mpn$V['%^W>^%g-2?`.U3D2&].m@;FO*p;fV1\ +$!f%[T9K;#@2-QXq`H1aJ"$)B4SgC3#t*oLir<*#!IoItPC7&d?f%<9j8/iUo>gb- +33<22_>j&hQDsZ1\[:e6R?F953giVQpcfTn6fLEr7#>BkI95 +r"&eAaeq2SgT*s9Ij%TKHX7,VI'`KZp0-?PX5,S>m8'mibM(m!I]H!*?PZqn +8j1(3c#<\;]Ncg^47NZR?D(uJ:dibt#jJQ&KesptmrE`kJq'0/jXFaA!>SMi;Pq@R +U@)(G=FJ(j:!J%*8ZB!(li<@6TJ"Xl8+)V<\q56LctAj='UPTqV:r,]iN+!2A, +GbC]#;#f:AFX0Ap"_6UJhBPkNJ9Lss,iAX]7L"BM-Yc( +m<1+11f,ljdVp_:/kB +m/KVD:aeI*1cHa8g"e!sbrX_7$8)>VO)qU;n7T0Lnt2%/g;TSMqDMhIfBm:p'Rg%3 +"TO"%Z8XQq#(M_!c-IBj:Cf-hL>+EO+,8)o9*u(Fq,@6Ns2k9XikKrOuc$I.jOaSj;pVIm\mr:=ogO6Psj*(Eq9g$mI*;m2Ni#BGhJaFTX +)D+-/!.W6+q'klATt-cG1/]QM8a;r!po82C2Xei>)R*L.A32e3\8<`\NX"N2.f3m( +Wn6=icSJZp[[Tnj!<7JcIh82i\lQ^rXbP@19rea$KBmdff't8@ZfCS>e +n.><*D:A(0Yl`tsfEM&6[ThC5Rdco'Gbud=b^n:-]cBs8Jeb("Ar#,O` +Q7TufpLp?C=HRTRT4Y0I`VNCug^fr1V,^97.uo!aFm*&BFR-'t$ChL9""KCfp4cNj +!pPWCPK#L)+;meEL2'oPPq3nJD')F`%]+?/VKG\1oXm`_R>gMMijQOZ`(o![!+'tCG`SCp6Z!DT0"Nr0nJD'2>l@(*I=86n1%AZ%ZY6O +piJaT#R4P?+lCZ]i=(K*`4(s=YZguS3Z"/F042q8X(b-;)BU1&kF*#R&WX0E<2Sr8 +0;cGtjVHjq$QBPWBD!?a@Z1"G(eF&[i'5X3n:-`L0?!W^(0h@K2jrK` +HT[ill^U)k5t*:$gSbaA(u74i.i_=N<$\g_s5Rgp;A<6ijiaWRqNZ1"4r(Zc7;bkI@6=nOI>U)f$3`Wf +H@lU;]F-e`Si=7tf)"4+9>"1Cs%R*U9:D2QWi'^.let65.g5nB/f(IX]M!$c$0f_b +8&4PRFZ,#o\QGfS'q@41'5`) +(btS;_sKkLO8OdPqAR3SJH+RL"sL.\/-UP3TG#VL0#7;0*[e#Ep,b+5[dZuAnb0*D +mF3pi#;TBEa+4k3>QP`QFN\TM&s.?Ukn7k#Op:_2* +B?$@E9,5'\:o*iW:@BJP[*X.slsoKi6j0;ekp6I'!XVj@&4E`s]?<4\qcCFWs/lpD +B?keZordsn=Wp8R0_>,.]rukO%CsI[e`76,qdOCA]/VMJcg]PK3M1eebl(^)$1U7N +^LRQDs1&,B;uGtJr4gQ0^a$W:%-##tc2Uker'(SP^1L*$@pEsRi;[QOa:`uaqIBDh +iqgEGA06Il;W^W?9=`9I&qYU(!?g*G)tXL4:\K8=oHr3"&H.dgcb,nm=/Vn:5);[Z +s/hSg43T>[s$$bohhd),Ch_nu!dqG]P#]=('aD/+<<@^S5(#oPAg=pRW%ID[BuBDG +7Y(N>"_40^a-$Dt;Mb?c'09nKYNBa4aP,X?rhE!'2OEKsm'ClIkA'l0l>T@;!_[\A +\"Ts%D$C8--MRaX.0R"\un&)n@C+h](h=":N")5O9*f +r1F?MJmc6Z29N!K_BijG.[Y@R*\RNm;GVOj']<,to4,(T(W[+12qchhGE]R=FEtXb@9)sp4WIopQ2"@8I;@Sc +.[l!]e-W9bj!>%\*>9=9?idMRet>Q8F2EG"I/`dNc))bj-a0VX*LS#:B`X+]Z6W5q +0(9t\pG&T@&*148Yb,Y/(\2LO!$XaVJKM`nSn/$;0Hgf`'#Q]AY+sC'UQJ4p@KE1Z +Qid&p*TjjQ@Q>H,>C`'rrcS2ZJ?3s1QuAlCg]7AH2q(BRBG5RUUH;<8]S-<^5LQXB +,P0`5+MnB/dlb$P*da"E+SS$@mlbof([M"q378&i++/SCj9LA\VF;JuZk!,Nn3:[d +r0Pq5ebg5"i.!;rs-E:Z$]R+r>/G8_U=j`:`uOO"MOrRnYHtjgiUV67]2$LdP6SB[ +.QnDF09,C^7,pRfm?$b;Q9kT(C>j4UO#/q7m89Bk+_?q*iEcELEQSO^\p#P(Cn0W2Hj!g +Z'Cr=^BOXF!D=?:l)K!D3j0<0#e7A2_UnCPSEeaP3'SF*0?6BjUJ;:*:0=tfk`mot7iL)9 +)M/41_#NrO2JHRJ!3^"!=$=\WPPIBE9*5*:H<'\6<=CQ&%sZtJqipb0.U+mo%FG]Y +`LSJj"=,^m;V3T.IL^@h-Xo6"!W9b8e8&2F7e4J',$e&CDJ%;ajtRnmTNFu_:n3je +#bJg^dI*m)-oVg(Vb+7^.I\F2Ifb+Ur&4W5m-dOLCLEP;R3qG4j*9]`+j29bD62lL +/5rA^era).d3t:m#VEjDLfaZ'p!?/9n?tR>@:SME@P?]Tj\@i>rs,u?d:NWkP>TSS +p!uRe^pW##qEOMB7a>r^^9O6;T@/94E9,`hmkkW'Va3]\Es_YB%A2NVYA^9?n-ud= +i$LLSkdm7f"n_WC7P +"F-hRH[$&,/79o08ub,fgOheR"9Gf\GJ[dEU).8C`WKD@SAr$XJ?Ns;o7qDk\J,5f +?D:2#+nCE''HT!nT\Kd(IkjB-KusmUl3.+TF2m`XQQ#2p!Z+t4mX1^KQ"8Y0=r-08 +j1%:'3ZoaaBo)RNDfG'X'`MFGShus-NH`i"/Z4PV8/PZn)k;uU_.aDd\BtKd#0qf` +T-^\Uk"huP:WD=V@pa0UGk%CR?iNt[rC?;Jml==]AP_=Ja2DD:5;@`n514e7O,]SY +_P,;TUMhA=7f6R@4=Yp^8oDOG5k?8!@4n3i3j9@uG";YR3=Y!W]XnO.$#k57IUG0E +Lal:o+T?:@If5PNAZ>RU,sfk/BP9:94tH/4FL":nr&qbA +osgj0!I6mclJ,/gATt9;9CVZk:G:mj/`k+UrpWhd(C%'r*sB-%2jZ;l3a)A(=7N\3 +#(?\b@2Pj$KRZT-M!.md33PWmIg*Fj1rX0sk-g?J2%I6"O1El$\tT@eqV\2R8-(+H +W!?a$:fnZNqo@`)lAk@5<0mZ,$N)ECn\QrS7d>u*n>Jj2(a/0u(OiX7n,<`iCTRHD +Q''I2A2Itd_ZShHLWqE$5<(2Pfq,]NH(Y9ki5NNq)P]+(rte0t^]GE*09dP0I!2G" +;LgNtK"Lp/DB"INm7dX,\>PtifJ'Z&@"+[:WE-&=OAJdF7@um^+Ei5XnYqdn_8t?N +69F9P*KU?\5qqdm]NkjD8VAp["kC^T0BB3l&Opo0BB:Z4atB=W52V,Ym^N(@+*=_/ +eHFZQ)L;<[L`hK;S]Z"FiA`,=m0,[$ptjRV*`df8ok'L\G7tP+nDE;=!c3qf0T5T\ +-]7N$i-SZ@_/ji)IMG7U!8nsZD+=%;_L>.25MHglT?+E2jI6)ZU>V+1Vm&s]a>"K8T\! +IaC'e*oV_og/iP#8I@@k(i6'JdKJfQdb)69F3eGmYQ2OibF`JY]/2[X2&L\JfS1o3\W7eUQfD#_3WhKpEXABRkMU +?"!?=^eu]6S'o2HY/#CIYR3s8V'MuLrnCAk%*HJ7`^7;r +5=-[2>9W?0-?MuhqrSQHho6*+bTSNgaA!T.4;H^/(HDWr=+k^H00lnJY5+/,J9N.9 +F@8'*"H+;9hTgdOJH&;VV]r?^2PTA_P&r9C$/f)#NK=ML.p9AL$Eub6n!T"!K#c!S +:9lMWQcMP4Rj>hhN`fK@?d+T5r86Zfhrg'('i09>3K:jf_'BV3"t./sNd\c>Ibail +f"p:?LpU$"C@G7LH%TnIAD!SD25]@n24ups)H)nYnl[7<+;fD@NU(fY'KAfV7`kVh +o_b&ml-HmR^D9NC;)7_^[Jf0qOr70n#?(U7rdW0lI`YFaRPjUi%6%I>c'eVc^el$W +6gk[7?i^i=Y9j^,&rk+g[0$@U]W0##?V^GV]aebK@+S2[)O0#HdUB372Cb):J)UE: +Z=GCqBKWNXeVWU_<``iDrsS^Ss"au[\.^$/?o;8B/s\Kp+[o!tA/Zei\`":/&G]h] +fLhGK-cMFVCA)O<7e]S5VAhXrIipO$28P*+7#gg1X)dY=P@23;s]*; +%);r@S=^7:]GP3:J-Cme+TH#hD%KqQr"TgIjIr]NTe5hs7[\bV\::p9^UBV7g$Vf/ +b4B>Fa1Y7B)kF("[Cj[Y6dApC($0GfGlcc9riX@Za!?s6g2*eaB[<+;Mq0.:sBbrYNblHX^kI_B+Hg59"oJAOS&cVe,BIXfP`+hN_ +O<1WmmVdXF%kpENLE*,T5DN6sh)AC6s+C4rWqt&],65T1.KuIGr"H.cro[L"`*7*k +=F;L'd)$SN1m6'AK+!KooIi1KQSB+f\N(\HQc,TD7m^Q3_Rh24q^*Nl?LrrM'?#dh +hMJUjEFI0mgu,ehg"R>u[c^(V[qoOMAg.GD&RFF**Rc`k!.;!N8-F_\!,(Hi@WF(T +(XioaIj^>nj'5!5P%o?];rCWFYm0R6X'sM//@dB$#5_(%H7,FJhT"S(#r4S[R4A1I +JA7k&2(n!NK;ce=j_E0G^h8q$p&H948e*HZ#"5ZQ:,-#Q]:(APJ60a0/hpH1hY'[a +qN!-*:QsfoZBS3+:k6`(qc?^l.fq8qJ-Fk%C5CNkUgP][MJ#-7:Z=(Vr2JS3B#!D$ +I`bMS9T'!;94l\mTDptNp=K5!?aD,2q1)]oq!qq+r&o19$iC6X80.g2^D@BmNbFdS +4J-37:GD*HJm,D^`.mIJA#q`VQsa;WTLHu&LSu%6H+]E$PFo*OomQ\XFEe`7W3j&s +lLna3[M/F#/aQ\8*.#pIYOKMH8Y]hf->lDJp*l)[XPg!aLqs&@S?ZEs:AnK[""D$` +@_<*H5)Z^se:GGd(M@mc3BqO?IeFQUm#o*3(&1RI!AR@6o1nTT'Pq9DgD8"ig--40 +r-<1?\FA73iHJG"UaZ!1jD4nl]W,UHWff*n>rDO9,hO54>n/1>lIn`bh(QmhhhX58qlKM#a=d_#Lst +Sr#-PTua1cEKFO$K)D)PbZ]^-%;6.\JoP].J6SG[Wrec?]Uf6$3uoAc3Bb=.%FcRU +YQ0@VM4GkYnU,i&bJU0<7">FL\Wd[6OEajM=A?^"`_lb:Bi$>X%j_Tt ++XmCR=,+I.%rNB\Y9"$s3=o!1 +2?s9\j8CN^%!1@A)ZG85GYI(!Js*Yo3rnEDQ,GO\$S?mA%#X3dJV6-q!KZ5)AHBUH +BSKB[mu\0b_a,=KrS8!9"fMboc3G2Hqn2=jJI@"p[qhSf8i,4Rm?A0bn,38'YH6Wm +V.T(mj8\e\J*[Gm0==ap5dn@AJ+mf83u7>#;gH31FoIT%c(eM,8%3/`"ct_TqKU=f +K91TWAg"i(MON_:OoH!mZOF3kp&PE+!Y8rI*$ir6q)uo`".fMg500#gHDm_UAnp@(L-EmK$'hr(19:am2/d@JgZ*#QlZ_X1>P1=X,"D8!Hk&pPe^LIc0:'A#=,78DW]s\_Dc+ +^I4,_WlmQ5LO4#Z<"JHSC>m7;V3>7YBAmJ(.DTET#Y>%\(XIU;rrN*ApuG#='lF-a +C6JGF6_KHd*i[Um>l,Sp1r?C5@T89!?[rNGJbbs9T:02d._0mH=MTGSH&peA::j!u +mcE(S'\\QfTi<0@SkMScK(sgRZ4i-\eG1bn($"(q"fG\PX;P[#N;:/Oqc6A'.n;]P +!:hOV["*&W8"GZbBt)pdDQ!G]S-\EZXOI/dZfMq[ItnK9!,/jUf&L]S>bK[Clk@:l +a*Vmcn_&Jfb+kf@AnS/=:Hpk0s4j^:f%uX\qi^Cg8A1]I]VuQoh]Va"@;>W;h"#mh +ou:XUD/TlqQ]6=LI=rZ^Vdf+p!UhKmUINJ1"8G7E&,_4\8.,C3`:Q@*2&b=*'`)i( +XL9dT\sN$$dJ\@no3u5%\VIHSkFt;#U05$8?A4NuR6b^u!?s.l!-Z9^idHkR0FqU) +WeTPB''![5JhS6;^4H)!a\P;u!A*/u$[A04I4YB7J'%il#QDX$7eeA&9N(d_i=SJl +,SgCL:[g:T;I1L((^m8V+GOSEj``Q*pFrBC+^R[T!rUm*4$oM3q`,L>g[>tpQikIt@+\be[@4gcd +VP&R1+>AMBrH*QSiBK=(`h#s@r&#[X2:o5Zq#=RrYNc1s5lLlKj,YA`iA'1]1BD$s +.Gef<]JV:dQ53mrMftc4fPZSOe,itgRI%Lmj4PE"?5uk?Lht#c9OW@,fKjdlhe7') +AD3+hVIqs&IjPZ]X +#cD)TU'm^B[1$feO(`VVK6\n[iuooh$3)1D-Y#7>;e1^b+uC#N90m_TQRmTP.0KRQ +'MPTo@-%X7&,[fe?/B2rV3SiGSGu/dk(9i;F1)]4?kmT(8C9V!,hQ-iN@(;ZJ&\Hr +UnaaE'/GTuA3!Dl$h&KcLKurCI3K[_lcqgtgk*Vhns;[X&.g]/inm_u5>Hk[RJjI* +mRthNp,.]&T/)Y@XYZ7m$Aln#KZr+52\ZVaok(bs&e[)J!tUC +\p#k@%l+nSUZmD,k%auMa8Amj:AFjg:pHH:-i@>n(k*.U>N]r_F#,1f4"FfrQY;j= +nmBEr?^FWkb67g:d17ebaaIij4dXmm[<2>il0/^Ij,;U=KoBHEJGF9nRC3J#J"&Jm +NW@.j%K&L;%09I@V^m(0Qs"3FWqG!jY^cbPLK"@j +`Lt80I]plphs+P*(PA4r#oHef:Au@.DkE!RRG`;4q/,-umA"Yq4b4?O=:^mg2dcER +GoiH!N?8h=RL%JCJ-_dbO9,Q3*X_JR)dt]Be4l20J-CJ9F>\TOcIc3t +@Coh2fJY=F+2J#_1LL$+g@ic^G]c+3c^Gc%kPr!8.LXn:0$#&(?P#csgLN2Ts3lpV +6lD,GK51(tW=IE'+8CgTaV08U/QMD?p!5V,btEi;d&1pnE@#VO+aW&iHI-(0UA>Xe3c)6SG7N>PX%r5'rp40=$ei;?`$c,&\iY_\C%oMibRu8A=31LluVqMgq#_aWci0W=' +n,/7P6)JIpB_(cZfp)UBM>Zn8=,re-T>Mb/gKUW,!'$]5B9F1nN;SIG"o(mA%58(Q +J65)Q#L=b-JX7t-%kmJCcI>0Khbl#n,Ju+QR:G]ab6>BcN7o_>Oai\V,;fOebWR:/ +s8RLbiW#ET`-Yo]e%Z^C!qsJ!^g\Eh+N5PQ['PoRZBch)ro6>MTt,tSlp%tmVdNKM +7s'`qC1e^JB4:a/IJun95LC.Dpj[iZn5^e=$^cokDmR"2Aq"IHotc#La1HG4dQtMr +(H4n[O8G8`*4M;oAm0C>0fTaTJ'Luk90g:BS->6D[ZWebPU_-K+R\oL;YX10c;2ZS +SURQmh(j/!!rqk6r(%068r%Ig0BfKT6lYa;VioH2B4f^ePj]"@9A(O_g!Y(IUZ<*h +B5XN+ioN7:c4XHY^`fbAl0*#@fS@9EK!?>01;"Wd@"_`7j430"'PlTGH]TGZRMM?B"YZosm5>'VNY6+_1VU2Rq9@ptt9*#dOGThW8^[ +_sU:3c)kkRY_ECb-a$GeSUQ$djKW.m?(K3;q[6c"SU#*Po_:[pKlKDh;9k`Gag^mU +DsAWSd>j!@i3%OO4oLVC#BF,j.;H/u!-puIo0^SS+H%-,GqnMa"oQDT%t;iO=I"<) +l4I@W88tUTZq2l`T2+ho'=N#e5&938D[O3U(0YZ-2W9T;THo)..YV$LpF1pfapIg` +$?\uPEWum_EUASjJA;>AI`J^n:LP?emq3qW*!=J619Au'J,)#20]RMJU53U[=X5]' +!<,Q+/#rASnTXtb7/g6/4;I@:r2G807%sYp8n\H;!MGZB1L^@kranKgGs'];cSgVl +*!Q=sXn*Vd@/VVnk<_/r"8Ne\IIsan*aA\c!W28Q+ +J%JFf'=9Fu>*L6`ig/_LBQ0k4rHt=TK%9!'DX<]tb,t!aCrKS_t2(Eg/Ng+;`eL +1%rmm][+TP:LLR[/5HnBMo4ViY^2-gA.:>7]p$iL1\#l$Bd5SL-t`s?Io:5.OoJne +n8cP48sX9FGFZnW#<5ZTR0gYjFW"?T="I1WI6ir8:kun4A4R`R#JT.-$):&jkX!R) +/5E5goB>&o-pHa&>:G7(8Fe;8#XWjuJ)98s +,[^>..E0^5fjgX1a$6HDn/rhpQhHp=KJ +WVZWBr4$9E9.!7.O*?/5P[FM61,2.Y"kR$EOIES2RD%E0I>8UeH]MO*%PrTqDikO) +GW56LhVaTEJo2Q&R7qGpF@uB!jMZh$q-p3(;?1/,lHMBul-ak)a;aE>-M+V'gDCH< +PAoj/!A5f(aT[T=cANBs,/)f7lLcr8-n,^BNRK!T>_%+6g^]MR_:uMAP=(ST$UB-FR +TJ!/>D76oW:>CB?=8$i!*:EO14="[N_\$iq#JgAJ'/uD^Y1AtGI!5#?4oO15#!$7M +$HiqrPT@trIQ4gZ[&E=fk+02[48Voo)Q$dt8@`9$NLP7k,Xc_4J'Ks9P.L[ +i!CeD6@,Lt'(abq/:d5^=9(?7J/?$HP$,h(36+oVQCqcs71.(%`3iPpQ"L7EOB7Wq +?)EjB(RP7$77$)*R-\`$n\:OmYuD#XdMKrW&dntEXTgb@Wr\aaUC6Fd0aqcFHYIr? +F>%K*=3L_bkLVo7s6m5[LEaCn%Sb1gFKLOh_?g'rYY7HAncFp8:+c5mA1e"V%c^\G +(&S10O8Vb]fg\32Q$-so6JcMddf'Mp5[a3dc/a6%0K7a'2#)OX]-<$F36'ng/bC`' +kmBRG="DKcK*^K7s/>`lrBKA[DtHn1N"J3j"92lUrlT^Gp+t!W:>h0o7Z4`V$/oeq +'A1]Bm56*Tks9JLSul%dj#3M-2NGIOJ!Z1*mJ^id3D6Nu'Ii@B&=11g]5j$btQ:!rRDdW^h*T#Z&H,>!W.- +N._RX("ZoL'OlO>V7DtR,Q3tWV<$8Ff0PR@!J!NENo),Oa)Y*pcWm4>/[_kF6fI+( +PI&()eY7n],u`IlR0A*&kke%&KX)]k'I$V%q[/^7-+U-N> +s0Z\-a&MkVA8AU?qH2`T*l2>TEZp:$L!ZK2+bJr+U&^+I\HD6Sb/lt4$;aeEOX]m&L7b:k8E'T +n8Eo/?]!C5EUu7EO_!"93;$0MXc-P@42j2PWXXp:37&8`(Vp&7\&T,RkJP"Kn,t`. +@!/=&nhR]T,Q_?S?j,<2&&87^']9^N6\ipg:AaqXL.[YNO7j=Fe]YP!Rsn^/rQ2dT +BDXI-pLfa3a=99AoAF$M[Cr;MqpM\8'jNPN-[h[r!L:[&)NS1lFM-5aLbo'XI%Cl= +CAYOpJc)_LauYsPrQMi:DG'gF[eV+ZOU;R.uRfr-V[EBaX<. +4Sohb"Y,?%!=QsYEsE)o#^/cME$C-h\e^OK7O>@A!WEYJ;ZK=)F9/c]r7qG*q/*+FVrpY<1<%\1K&$Q'5PHIb?s.b!Br26P[7mQAUE!W@o\f>6$ +U4LsO1q%r^6V#URQNi,uk$,)8FW&#&bo +/:]SNKGghtI^*\%alhkWnuEhl9^$6d'5`DP["S\fJ%?Qt)#4:3O)&.Vk6,'u8)r@Q +U?uZWp9Y)q%A's3gNjgVHFKO&DEipDISMuWY55e@L;dKu?1Et0%ZeRXc-lt>8q-AR +>j#o35Zk+H'?fCHAeG?Q$RKs%Qk-t;iHZWTVY4kC+-&qMNA;ED:1 +Ef0@9-#X2gZ^QZubXau,aMn*2'RQ1pZ,;caqB=+UmkV-I'0PL)m)`7BYp86-6K/9R +Y^t^s]PffN?p53^%!P`=e4QmIr_`T^7/LMp1Bd_bF/M4b!rm6?&Hr-`g6]:`;tg>Y +!.6opb`n:nim.cA5hZs1s.=B%Hn*/9c0P:_pjY;FpeAmRd!e0YH]:8^,t5NtP/1M4 +5V:t=r25R-#>+kac2U"'cbe`*63Vr?C=_UhAEfSnmOl9OVfmgrs"elpV/>tPhSTKq +([/,^Fd\o>S]im9+f.5.^[)UB8(Ff.N;m'B#qGE('h +lq[Th\>gd`UF.NQS6ocL^=hX$j:<2s$`\EB\rJ1VB('#tK($qA$kR&rH +YniSJYot`]`'1?d^Y/`IMAUou![t%nL`o8Z@9\Z"`sMbh"]4f_,Q]GKTS!3Y0`UV> +:CXZVr(l>24bB*"M__lbK4N7NeuW+M]#ZmrjA$7,o8\E=^uPA3V*'a%i)j +cSX+CB`#OD`rDY[Yj^)o"T6/)@1g,^5dnqrU4D)[qGdAO)IJR +47CU:gSO/K8(2UAQHO/?Vqi8O]$$;u\r)hG$+"c(P#;fLPKS;SeuE +*loF@Jg0BgWoIAWOmCXi*MLFnV=BNg"]3#WQ[G4#_gW'3rtm7ILqd9i=m*O8s7gg9 +':\b6r6o*pk`H@^&3F'Wr?d$K4heu/I;KFASI@Y$%D5s?F6j@qm2TlX0$C"n'gZ,H +r;K9:b>`l4%)2m?G[lJmnIfPPpQQ4.13%_fC`cCWUA.O8eD`Voi=JEVO:dEV[E*rg +8ItiVN%s8O;XX=lS?^%L>f$bTiG;u\UnDKe76E0c-2R]D$TL"#5s1bn%!nNm,?MZG +&>^U1;MS?DT'6BT!2VKdSUH+mB6%POZa&j!-sNGJm#Tg,+?t"_T]6"Kbd$nFj('tL +4@8Q;5'uY*%)lD=$lnGQU);lf_Z+i=IIk9\MkN:es1#'U9tHLi^hk`YPGFT;O5`1( +!%J/>J:H0PY_/YfWqcUS>5g\p.DTjj;erp9dVe]RGilJ6F.\@;aZKq*4sBFp!.T.h +^mc0Gf+*@[QfFhAS%E1EQ,*FQi3l2_GM*F8X?%=7(XY,!h`SkpJh-o*DC%p5EA^/l +r.e)jptp2kau]Q0^!@V5qS-ug0rg#u+"'K)@`'RRi*3Db!<42k!%3A:AiX?8aT'S- +DuMljqU^m%i'5$W6fuHndccj5?oY+ElG8//Sa>]Kbg^K9RDHiV]8^Ed"LHf;;rbN-X$uh5]P= +Q1-d$=XA5Dla"4]@XYlErr/Q,2MH_d!9(s9=Q(-meimQAmoqZfW5`%O!DOetQe5c) +kJ&\?"P0DCm/!g@oeBRaTP#.XT0*/:K*=LU0]5s^li+E=mOhYlhJ.BBe8aJC]?Qmi +]=Z<8DH3-2-p)bGTjc76h9!.8$Qj.35Mqt#FDq$9"5+3FjP;k[F5*YOJORS\,:PSF +-XPCN)XVFog>/cM\m=K>Vk$]9c$eFb:^R'g@_n7c_;M=7 +cAQR8]@S50[("T;^&0A7!;"9/NE;mZOZEKoA.-pQK0)"15g=0&:CB-I$i5q* +T<>oY'lK`Qqf/F0YstfL/^ER0JN_mc9HXK^lO:PlkO +%aTO1&s$`PT$W;&V$e7Crj53dHC0msM";CRdoN@+%U]*=h@;U'::e.lH[XZX]tAIi +:RGGXb34h03@c5 +"lgX)i?/8_M!1(sH&2L:54Xu2:/41G^Qa`4+kQ:a%H^eK!'W;R;Css!KTqafW*NBl +9-!b;=V41Jhp,ILYY]EZe(a:%uendR3& +b8L0Wr6J<722JIq(#j`<(ao9O]BpXJ\3*TnCDJ-UAcl3Zncq"2EuNn)FFsaUEU.2K +H::[%S%)e+k6hKn@pE8g0GW?iCk&5RBPG6Kc,HoQ[8^Ch!1o*ARA0sM--/7rNip(8g?kB3_n[]4HWZrX4`!rW2@!3](H!12Z$(0[l^=`k>hMH(r,&l'`2LMMF\U3$U%u"L:8)^_PK-Oh]P\!_4E=C5*l8n>cTu^sP)9&,SM] +ah=`@$:]Z2fdQNq+7ZM#qDHB_/Kh<2Fo.#N09%SCQsL@d5?0IG67)k\Qa>t-s6rTX +HbQMUhC.js"g^t`rZ%;$%Y>i-^d=-(RDcM'X:4N:H3C4&f^!gPm@@kTlOCn<6D32q +p8Gmn?An3q/uST>u$q*$sii_T9h*3<8F>WCA +iUf3eZL/aYJ$_H+jq5uUUVof[nQ5AefCm0i3#$Z2B_$5n1QVm,S]H/rb'Y+3p`eg9 +6gj&jg^EM$4)K!3g1-__`B8(!Hfd=5T?0*_c9@f9.2DDK +#m*om3@Ek>`A,TR.%c)^n,,`&S1qFr?q4%>TREiTD-DHq]"_MhcSh&("!S.LiuCj+ +*s7@%(27U.FH.(uK&FfdVI_O#,0A.N?,5%nSqGri&]6AhAO50%@[a37/Vsc5;n-E` +Sp?2XJt4;AaD.IRU^!S^60:S2pn,J\;dC9Y3ML&3W!RSL?ki$ef>/[/9L5AdkQ4/G +#Ef%VI@?`3F9`fqX'HW6[aGV7`/6_:5;4'M3n3+A7eM8krsmt]8/iD(n!6JX;MKCYRD$E9f9-$<;>mh#?",k-"+,LTH/e=f82cO2k#,H#&^WQX +eqN?Rnd&R+c,Qt0E"/lQ$NM`AOe,r2b/b'u@$cQC?jElTM$XYu#"H9AoV3$'h^%6n +?AKjuk9n4JA?[lfJN2_SRU\V@&!&tWGU0-b2^q**mAk_n9UT!XQ/YD[Wp^4k3X`nW +64)U87/h_]2mITU0UF]'S>Ua)>nHum9DL;H$.1rTr1Dnm-3it?WD_[SDZu@).%)CZ +s6#f(J"0g>p#PErEr._@lr2lm^WnJ2;>cjs>?7^:C+pkt2_K'#Ir&YVP2"W'F0F3c +mKBG2.B70@gF0ti[+,M$,Ocf7It`@sl"kVcduF;gJjB[6Z&^;=2T+uuEO\U859J>G ++[q>'oD#h['E0.jnY\"?f4.d_Xaf3 +$\1b<";68lf-`_a:PN,G!OTmplP/\4R9tTb]4T^-Iaki(aaH&+>M&1?hKo*;Zc2T4 +S%B)E%Ob]di-X0B!68Bb:3+o3^E?)%$_6\O?;o?I8*;TZd.#j7-L7`Q$ps)iiaS5CU64q":p[Km32gDWqLG +kX==TMnQG]na<*XqhK7\*Q`_?R!&!cb1/k9=8+#sg7*2$+929<8GMbl_7\qX-2!4< +bpaIpo?T:Yr6fW^C.%O%f$#YhBigcl2k-$"egK=RMSJjInNpCDabQ9%\$Gh]El*:I +o)C#ls$B>n3FU8pISd"o$PlPOb:&8A714qBb)>c1%Wqsf$ciToa:*.j$QqbPi)p0^ +M$4Ii';ZoNbmF;6,lWQHaF?ERlM@!$s/GL:NWkQ6rua3!.1>odc(C*t)uS7aVZmA4 +fOfMsBaiHMRUH9eO(=VG11S_rHqkefX6./q]/p#ulP3pWFCq[add3N22WIoF/^ +9mFSMJ0kP+r=;N/KEhN).fN?^TL$*g +&)p$.jO(B*WTEdB2t97mi:`#S%NP-j_C^c.]'_]!]g^TLURpYPLqp(ZF7HbM +AKc$rIO"i[<;[7.(Vku^O8PVC'JE9OGZ%3>$NU0UaEJ";'ZNLm$0dH)g,4,*D1 +IP$p6jXEhnHi.&67m7D3OEcedR)AHUkDOokmfg3g8$tBD1eO4< +52bcP3>`G!s7]I@_#I2an9L`9/7jk$8TDCf19aUn=Ar^VTjVM(b0Nc/:>e>Mpg:Od +8\q%gT3%cQ?aM3MP\t_id"A`iZENGgh\Rd!(J_aV$\%^jpgL:&gO,k%I:*)kJ%V\L +MJF#o;b(J1HcC[,k19mqr2Ll,fLQ?'b/!s,h9M(#Gc((25kbDj7DWjZorB2)JLmcl +_.f5j)i&iVqja:d*5C$5H+hhgiM^Db34iKl=TE.XP5/&Zq:5AWaHqtiLcq?IV[Z*< +o;]Za5d"\9*k2frfD.-=%G0ihYL32Mq2s)tPB-a(G)&/pIm!aR]\_Ul'8JdkSZfN: +$#6`[gMm)]4bL+G]#(M[U+Ynp\u*N_D4#_6+Jo-7%!1n*:BMuX[6O?0hR%&&n`L=T +0'sJ3hOj>fWj%\:XnVaq`XCtq0:m+Wj[/X.igM[B02E?b(R7r7PZHn;I[4"31ooP3 +DW[L[UmjBl_#MTG_3`T%0..!1NXri<(OS^9Ila0FMU)>3.$!gJNqK5"^b$qX,qHn? +/0HIa&bqt1@%Z\%#Kktq8EX0/#liHE-i=4RX5e](AD_HC`n>AlDk(Nc>/^,rf":]73A;D=DLC\BZ8VA,*,U0)WA-s:cj!-4[%/3F+D^TT$oT` +j),/%^$Kl)*V;M^%6PX)H9)pi\aKe`DKO>T9cDc:e".6L-PLilEEs4i?%@?7#Yo2j +-[6CHh*2Y+&+IVd?50m+,tgJb:+rHba@R=A(tJ#GYPZOm/#im.t>L'E6ZGk#RcKKAm` +n+JQ50H'F?JpuYXFE";1mq&uaK5TF*_#G]O#?Z,!-AY;BH*+m^fAW9a6*6"e4<'gu +mso?F9e9VP#%fJ5%8FjTqSrQ>jprZk;h-BC@Kr.YL/nGp.l_fsE%.q3ncCT))U]`& +k>Q/\ELn3u-M_/dQ;""VS=KFZ15YqTQJr,0pq_f;L8gi#60=AHLXOV[nM5<0F0o(q +3m9o27Nkb;.aVcDBlcpjq\SQ&(2=(\2,5$LXnq_G`**K(b?N,S04nZ(<8q2*1D0>%@YF2)K.:1Nm?%e'_o'B/pD +f;CbaO8IbV#!L-k*:?.`?Y+Ptg?\mV^WYkje+%4u-SBU(h.jC/ftgE+Y^i5da6Z'H +/9Cn3YQ,7(5Fm4M,QCaZ^p#("TW7qbW'=o+m^= +18T-mX6,3bE\69"r_r;FM.<$q:]_#\Am%O-bgg1hgZ5?;l*uKY=>p!^kI0=`+:A+D +q"'gZs8R$:K;_"O,QAd-;uGOV_)\U51BPu&O4&Z@C'5Wt.p]t]1f6b#W3BV4s"NJm +UY@]NE2J^lfT,m-QnL;HLc;#[b(:B8ZD*$pf8Qri*:9sVdd[Q= +R,daL+-PbqioVZ=YS` +CF5bTFV::s]nmbs$1Cju[>tR#iCN.Q^9NL^8W;(&QVk# +ns+%1;MH1LO5%IVj\s#[pj`<OIrV2!PP?u(.s0Ki"9pU +s-mO*8HD@.!OPl]q7128@t>9!J:`X("aK*#G/5<.WLY[ZLT_6gV`)t4J#W:%!LVAl +'Ia1b&G%QpiqYH"GNmUPR;AnR1M6cns#GFXWr1d3+g?Jm=UJF&>6"bB#m;`cBN&brR$9@u4-3/NVd=gCj!8k9Wb^`m%gr-o[?3uQ\0e&E@L +GM1amB3$[=FXgPe#V4e9MZ8+.6T]658e@#gS.>"Ta),Vjo6g(MF;L`Mb,@58s#unk +DaU!p;mDr%4u(t[^jf?D\cP):rkLiomsLjS"h.gtScs/=!rRh'*IM?=Eho<8'n;aF +"aqhk==mA-WOcbb3FQSNTR!&51E#!;dc"*OT#f-H]GtcH?3/DaJDa"j2Q(5LJ'%,o +@QiQsK:jEZTQ93oDmOaRSUpu]jF&Z;!a*a2/u2)$P4?iO>O!!VPb:eah(]f!]c\65 +qfLN6L%k[>E/#306U!.-WJ+T-IBo7K)hGXPHCQ%!3BR:B;Zf?*jd\ejs#@K1PgLUd +2_0,i11ks,C\]JBC^4]:EP'EQ+hlXACttph)]8'h#0a8.s/#Y?.\\C*%[VjMND8G5c]EcYmArhTi75Q;`5akK28]a-Vr2?VGif+n<\-`(-_ZpJ)%`Pf2H1MSE?9 +^RE3XOr\m,1ZiFU"Bn&gd!6=-!"UWR<5:[D@$\b,'ubEK"VaA7kCpJ/L:ZH$]&\Q) +k6=$0"Fp[.K"qOD$Xb&X=8p;TA386Ipoh++YQH*/!ot$qi32-)AH>Ae"d&=_+T+\: +!0L?GH"PRV0RTJ/ZQsBPK`;%%+iSp:ro;WZ+Fbr_!YhtY"I(H'mKVQ/BPUf=TGsqu +5AnIK>Qil_!m]]urtiuLU>l+N49)Phk^1N,XL?RDG1,1=BR]jB'`[D[,?fUP/]`_` +I$BB<2q"Ki#9j#)<#qPBrsIRV5lghtn=NfJh"mM8e."b1,5`hX&%A3c=OMe^ +CPLS^M0@_oiJn7fgD];R,KFd3_sRaOgE1h0[X-Wpn?Ca9@?9o%ZiA$p&!*>ck(]4- +KqhL&Zi",bKGbuU7K2de8=(Oi(!^l\*ghcM"ocgp%*c_",-W3.@;_![,EO7nK%n'B +n.[!4^6CL`O=jg0*sE:?$,DFcY%cfo9U@TGGs&/1EWfW"*8M"7?0au +E.\SIO2-6&7Cin19N9>c>lSgOIcp<:h_mf1Gs$/<>kJ0nJ'"*`b8;Uh^Dd)Brfa@i +O+3ahs*F5n%]7NMIO^la^%BQ6:=&rHN0?TuVJVWX$k$@;\o)IgHYh#=:^bVS=&%B' +QM\=^$Yl/=j0WHe?ekjoL%(_i@oWZ>!<9$8S,FW3a3&Zq$h,79arJ/W)d'S4,)2^. +fHn1C.96lTLXZN`NX>uAs(!H7`;a`h`.\.tng<<7>KIplbEK!(5SP`"s6k$j9<"(# +Tc!PXpTC(sD=:ss]6Z.oCO9Hl2@ltkkVH(^liLI!Z$+JCmEd$ZDP8,T6ndcZrn-Y3 +#:i6+:F5RU=V1:"+S).#*MW<^+cFtq\r]Fh_g`*Q[VOB(MP]BB8-'jR)>:,J-)X$ +RU,AGR+T(-0":t_9Qj#H6)0C-2(*LN4i_!)Ml?%F# +kOoTf!-eY/f9fikEV'YFj+!P6'h5%-Q5;Vfmkrd(YuYNY+4d=$qS(B0TE.lUek +/pqXlX84I<58u03q5(`:^LWU:82!%X.Qa-l\5`6Rir?.?`+t'`Qh(hBjU/0O`a-lM +\rmYO.4t$u8e?e\^mP>NBd'_]TAjOfU*rgAJ:]IV16[f.2^PTNTN5L6l ++A3i,!macCk!-n&,Q-)Lj'&Ho[qE%t%S_KRJLrjdNN2niW;03r3/%=M!d=J^nf.B( +pGZLA?]2WkMl+1E)X2nO]PrFgRh!\U09(+9`]ZrchnX^Vp5LprNG5*P^&?n:oPg#/ +P!@+MYE;$UrW2o3J"\<%!OR)-NPG'L`^l@P6u@b[UWhtb(3G(VYQ`;/S/-';1YD]I +.(>aU?J;+0g*'dLG]#TG:>:?tT!hkZCpprZmr5\fm:*-VC8XSsn*[R7_qED]CFdS; +hds8"=\Kk*Ilb#b>0r0TM:$t9!?Hjndo$(:M3rbMs)\[fm?72mj3*:bl%j5u%_r/& +,tjR\'j*B?PQOA;dma?:Dc2=oNYL(I(,-Z\!]^=M(IT8o;d5":YkSn@F-.M+$0]d#7Wp)%20X;A_BT"7E#fI%D_#kG7H-YIfjFdT7e+lsh(s%2hj +L9`:lI7bHri1J_7O-aV1UeA7U\*sj)O1`ZO@A11V4_*hW2I7mV`Lk\&S0e[u@cb@4 +k23"RrcS(^n'q:85L6tCi.$K"X8pks6gtVs^h7CV0GjrdI,Zn:aB9iSjC6a>]9Ha^ +s'.tTW,G\&m60OFG89Thir=#5HJlQ$dl\GtrAq:nJq%$^)i4EI_#@&Ms&/da*5<#5 +9_,D7_Z*B3bL?GJmun!-XoMJoLmWbD6o;FJM3=."iA\USM#Tc:rr89F5/;.R?qe,@ +#l&2W,[LL"_0lY-s/1o/$3m@.Zei*n"*$;)[H!MbS`i2,^jJ9t_V]m/VpEZ?it^ka +"92"&\HRf9^t2b,qVhXK)ap`3=,YKM!^dml^8LI!>&"7 +8sR[0qBs28EW-o1ggC.(:]&og_rIt>"]tael#hCWr.eVtp8]7p2G=7^+nI_=&k5#d +!'F7OW:W_cG^jj_ql$t#H@0-?p,![VEV#CHF +JCq7d/b,Gh/iR7S/Ot=?pl%7q&o3 +@AmsMW,k7\;dPOf-o]L:gY\k-Qg`>+cam%C4Cj +[/W/'6T`q%5M0UeTT9-*6dO@egV[p,X3FD+'>#6dG<;AJj(tLNCo?T.YNtJHkN8`3 +mUY86^;\E5hdt-0OdOkEkg&^s=&XKOjusC2L&;JL(^k%"$gp7fM!7i-pAk,XPK,qP +@GOgf'=bq!JJe/X21-m>Zc=6gIj:+AUc/n?2%bQ.9r4QNjE(Ai9o9T[L%e81o\tEg +F?f[9c9Dj*!<,>0J9RR8/H@F3irtbikfS5V6>Kg0oAkD0lWJ-N"f\loJn0Jm? +AB:-OFmoq>s/f]"he1gFZZH\UNm#2_c2W'4!TO,4-*qC+dps7WE;iMEP48L"VA/B) +/RJPV?csGp^qRcZ)Z=\g9udQF15kb]K%^s`\hW)'\PMTPX)TKUeSIe>KKj@e_E4l. +Y5nQC;16X)Ok85_g?Pg\ZMIR9(,: +`4Z61hf[p"\A3p`$ZH.5!qiP4jm74(i$lr^F*fh/?9Sd#L**2qjkp='+;h2Xg%p$. +jlK+0Zd0,QX=u^,ddg;HETo9SOp8Nke]5s5>C]M`0L#B^lr*ee.s4n-4?ZS]6U0Oe +:2S,bX9"]5-S +Qj&S%?`f=\+*%LWK"m@LU+Zk7M/fBqh?_st(J*B78Yhdb]b15`.o8@4SVuuuA"0OX +Y=\i*AoTqS^;!*aTGf*4&?VRrD? +alJt5c8d+76ieJV1)T2ENAieqkdKT_0nAsQCCDujV6H)I +Lp=M+!F[*cG](K1fmbM\jrJ+1+@WpX41u=`4`E/f^=/%%-U;e?0&-hfa6XR`P+;["0JKT+C +i.$EblRXMAB1q+E0,'%h1<]nkl9#XLPA$SeaWY2na"c;lTiQnZ,48:p3IB_H/VKI= +Ru1rFW0$+59g((6>faJH(BUhP+?nM>rpU +3tERVYk[\DJM+a9B.Ie?4`kp)I!@Z]J)7@:Z2fD>\:B2754F%`M743q&RGsk0=+IdLINDGbFGL:ZVip0! +W:So-A[kY*\<*-\cC!>n)-l^uh$]e+#Tjk9p@NVo]r$oiU4=[XZ#X6'@NbDm;"<5C +o(18Es!!F%K1QDh?oo9;+TU.K8[TE:#"):7>QBEG@:@L+E?$0lQl9 +cGFnofM_Kp^or1;X(g.[ESU*'KYn1:!B?;)eKjj(Rs'-=mSF8t!];4XYn_tQ$+tf8 +V[NJ96C%te&2H*0feHt0]NaReOQg8Z@C%37J=4)S-ek^lj0X2'^l^.0(3nrEDJN2S'@fs!sXUNS7Ac09s<% +.;L];BJ[3ICl&"E#9,P$R^K>AS)7p7W3"ChH[^fm7j5$54n>mf`(cMagStQUoQ$R< +(Y<>beER`*\eYemhLGL^G3@?&NlTqXH\MEB/%_ubFPSG6]NN[3S_J+GGr.-WHFra>j!@:%I>lmNbOYGVVh'];S=\,^Has,&f1nec#t +5GS:U&b\X*2ML,@gb>d\:s8bJ$X"VYc[u*hOng@P][B/U5?%-_bQSRQUJVIe;?\4. +j4!:46mGY8rtKs2ftLj!*(H'_(kFp1J&82-%j=.;T3U'CVJ0INE)logP$(?tFdTuF +e-]ieA\Me*EIN0Ij?*Z`Vc3M[PP[0-_9U.oPD(Y&!@j.l4"h,J-Du-D@IR?e`BP2_ +5KGjUQZH)io290#s+<1CGk.n.s$-'=k*4!K3XPhKg\T@&?P`R_i.'Gg:cJ$qr0LR7 +M<7%GL_acbr!t\Rrd,PUPFrT+[^js?;\eZEn';Bbet8) +<=%HY/O8q\$L!9FGOU*$5EYsm)3.-X0H0X;8X>-@mG'jm7G,! +_Z"E7oQoR$r)\U5*A`WCd=mBR+?r*#D_CDL&p&FXIVLqSdKCOW`ei;(+iNd:WPMmH +mfC^!I^gP-!'Z(V$1gaL:U0s+rm;NYrBHAF!;\Cp_u^F+$^%RZDj(@7P;d=gLpYK6 +g+QVj-sfA_dF+$3MDfo8=N0qf\_(-i^o*XaE>^L6Q-#:V<<'_:mK(68e% +(Hji**h@ua&H?[%$u^e./Ht'M6q@?eUF3Kh$=dfi-_i%8d0!n(eKYk(VQ:iA#.-\o +O.17>Kd?Zt.=L/7<=&Le:8QZ^ft`"*q]+D+J3+t.6MTpj[DA!dP3$&of5LmY$!%!Y +=ujgd^Io5VLCO@bpVV6&bS^ZjVBaf0/^);%@S2kO\-`94kP^;Q6b-i^.hn`$.DX\i3W+:]kHOhB?*MM?",$AUB&KoODa\]@=/(Hr ++h_rG*T.!r>j=id#SR4t>^?ld!4%+]a#]^rn8)57lJ%7;=UcI(i#bW*49u/hTC_N@ +Yc[@KGQ90XT>A4m`XdXoI#"$i`s1Hd>c4Z=-EUU9KtK58NI8bnZ4R)rs!c]k.>N5C +HOK'Y&p^o_1-p-f!7#8r"t.:a9^fA-gj5f*;LNL^N(?;Cf#sH4_k%@6Z1m7Gcs(e;+%mKDb +I\,9MDH^oflAjFF[cRotAf%Pn03Xg9[5L6bCB7SR%l4,o$K,pTf2V'rH_unX]bEQ> +(cNXI%55!+f2t.,eB!$uYU"O.+adrT$;l;sT%dfH`.8#3r]RQN0gi&ojof +Ed8ep$?Hn-+E47&fP8@O@`0t.$^e[B40oY^\J@6;*&P#>49Y[g-)&3a&]5IcLCT+b +L>][2!WqlK:>hfQ1_sprUoO`Yr7pCp^t9jq#qtQ=+i6ER73d)J9`Cbs&68PH-Y1G0 +`1)DN]Jr-F0QRfO4/R2u*W^QW(A8(NJH(Bn3+DB^B^M.\Fke"!/,c>us25)V'k6>* +XP=$Zs2X5l?V8q"/mAS[6i?tXBGL:Q2t6SI9'YBuU(o-]_!,#.YhsLJ#hXM2n:)s+ +F?aY@in$2*r>bUIinJGa0oGo'C^1c\i7S;'h[2mQY6=OEqfDo>+l_=6(^2RCoq01E +Kp^I:NmE%DYE#"*`qZ1+s'nHfa?O4[IULL:A^;AB:;9R@;15-#WWdL9P(;R[&qgK0 +\We?OV>Q(Y8]f+Z/rj]JruqC!M]Q7!mb&&=fOkN+ZS*Vc$jYA?e)?Nr$no7)>]:jB +HSLn74p3u7R7_.hH:3Tf0U@6R67Wbf9@&@-ZE)d'(d?*7-).&f3V,N8NRSj,J%>Up +J_9m[jCtXIN08q@T&381!>cjgl5'0%OdZg!s6'MhlG*SIq6[qi,BbKSJ7lflXfJ/! +EPm-25F-g$#;@8,H4BnQ(4CPG"Upb^lHm5pX+(p5ltgHj0!E@eGC^uL!O^CbFZd:I +.@YXf8c)\Io7cVWdnU@92DoFgHaPM'183V-YTji\JGM"4kG/9gg[&7)rq1U)dMbht +NFi2.G_&rmQ1\k4AsJ, +eLH#^;*9qbIH2=9s'"$l;W*&F]n5+QA_f;gS+#Gqh8t(f41F`\?K:^KL25K8=#0*A +JH+u)o$ls1V!=,]kJ[bc!hOAk+6CI+aH*lu)g*P*1]M#8!4[QC'Yp!0!8mp9,Ak7# +&H8F4#e#n=K]5hDTD5&CCBg*`D3FDD)qpkT->*Ls#SH-aXVT;/VD1W"W>.3c_TiW;jmj:R>D?cs>cNGe\Hh3jFJ +o2K1DX2t+:1'5uCe4LP85=kn>T.<,1?h.QMk%ep.]aad5s6KeX<,Y&!;[J7'j1c$% +5^p-r*A#_D$s`kV'K\Os9]uMsEkm/8e$"19s5iXFl0G&1jf7QSVs[rhChp1aAZB-J +!FF]SN;m8qf&oU&hpq+AR(:TXY6&-3-S?R")d,iS>tUOc(U1cF>)4+4LA-Tmqge]K +JrZ<7qr@^dV6HYa`7^nBWE]Pp+#Q16_Nj.3roq$+k=j]fP#O/3e?&5]r5Uja^. +Yk2/al7Z_L_#H@hU*!$oJ@)heQSl2f2^\*5)mX]DCAnM6^Df4%.b4.Y*Pd/\s0j4B +2F7Z#6=)>ApT0,[mP6NrhhdcV@c=%li;\nVdgkcf@H1RM5#$e];`J=(k2ZH3f]i.P +.,tEGs2PNVXT>Qn-4p1:WCdL>q#;LMrg4p`TC='8s/p`!>pD]P-rlDCE#:[ElIWck +5-YbcN>;>LRDktQ:=n6%Pi>=N,"`OoEmOU"[9P#h9I1+K<&oVY_8#2#^h'ppEeU_J +9!Q($pX_2jhW,'$*9]\sGWg=eT3;K(P9JRhc`F&6i/54a)9@;PkI2)%k!MuX#[G.Z +7,.NTUp-r,N\4$ +pm^GF*iir0=6;VEb,No?CA"MFjk_#lH_^%=Wi%O1V+oc3S;t*Qk0LuP8nfO5PZh.W=XflD!B0Js@>Pt%(r=>>+ +b&>][^SMfFb09-M$:+B9Sgjt:iQjd(E.P#-&L/pdS/bNaU1E0>6Y1P3p=D.-^X;HK +6$#7`pV)HI^V6^)/H2\C=1TOSFlXb=(m1`;O/C6Zd:\R;s.B,51*HMpUC/cKOWa[G +nR#m8^n5I1"%tMj%51:bZrg7Q\>b_mDT6%/>(,_uXR>DOjlNT&Co_#`WFTM9BPi(3 +`pBG-krEOc+)K/NTNo8m#O;cC2MuDG.c/GeRHA*=?H)N`]sn4sIS:IQ>/9KsNuqY& +SeB!Pc%bmV$Et7,!Q"\]?k6:CKDP+++CSl;Wh4o\V;V'5:$/r2W#(h^X#I!;\bQ6H +e&QX";k)b+33D4"YS&Y")E>)92?s*""YilLGlAXF0IP@Ls$(h:Q6PtYA_C1[`g!MLFs0DG3s-&uT^jk_0:DW@G9]I6\ +Q?YW$!b";c()3Km!T9`/*47=7mW!2'$TCUcjI_9E-pQcarqjjpIpLkXJoAD(DIN$A +YmH6nI/2+M>JF8,IA#jT5NXUqaq#,hT?1b`^?@p=DQDI:i9p^R'.2nTReof#&H6SQ +^D=@:M-X?J>all?fEiK$poahoXT\AM#'9tIpZ&-ZU[alXrkmW,5,s+Nl$8o[oi[b3 +9>8Y4XARFGM.Zr=r-t%Wp]+G^Rc"1RQ\@hP'6:.K'EJdE,LAVq%Y%$jH:h5SOTGB" +aKAq@6g";4*6M31a2+Uum+L32iE#**34?05Uo6V2=f`0@E$;#Z.BHJpU-^`;n7#ea +76C`%5?kXa-%N`B*n+ZFD5GL/JJXDc#L8(= +.A#R`.aWQ^O3k@ag9hOfcZ@,$+,mQ/M7d&kmNCl?'R2GJlq+**R`rsZ=k`cs4@4.)N)#-l2^8RGQm/.+Z'-mEN +%?g:S*#SkYotR@As-73q%Ea.Rs-)U;*R3sVR`>q\!4Ck%n3M!o=sO8Zd*KhUSYpLZ +drn%V%k`C.!4'2_"feZ"pGL4l)k+r7C#8#dIYot3i1,FrJr>VW]=k)V +K+cqmY/U/+C=MRokQsi!!&)!52S)tQ34=o*%ZZ&]e[ptu%Y;=/Vk*#-"B:.]?nYZe +bJmd..@MR>:7Y[K)df>*m]?WNS>OB*9C1EH*'1G=0tJ5d\f*]DkhEl5h7)b"%piX( +E,[+J=N:C&\?QKXe8#<0`25GVH$ecW>inJc!#0_=/s9g5PS^3-J)4^ohGJ>o&@;O> +#[8\^+X&oHAGD'V^QMU3+7iMsjfcBm6m7!J=hE@]XW5_piSni3cBJ?Z5&kSIMc<3X +@-&sO$%#/ge-WuS"TL\&Xo.q$KE$Au&;#CfoDdjnmXJR!qmhZ;J(`tf(#*[2*s%Kg +`W(W6oB3q4rr5g)E5MX#M#[3IG_H*Ljq"A+So+c*DMRjOBKG>3"Mq"_V$U]t"8h5o +gQqtgOT/am%q;HFU.0@BmBS(j&geb#+/\STrqoI%2P][:&fY[[9TuIk!qeUW=o$V1IK7Lk,MeH>LVsJ6uVWjT5C* +1M:kd`lJ"XYH&ubr")c#*1;"*W),LM>U?[D.AU'##l%c()H^spX&M*X;DeO(KElg+ +;rJ1(Mad(A-&:/pA/51%p?bNs>f#"?B#(2T:O*>D$FCZ?>3Oiuld`jnBc+:(%`(B_ +cN6Blo&%!,_QFfM%K%#hG7aa@Qluh#TD%X0H$'HVgZSN]KO,[/[Pbb<%tO[S)H@fE +Y'93g)E)q)@oioO5Oa0NWYTF'j=(Fq]"[/af(0ld?PNG,KN0VVZosua:bLZ@Y3GS' +2i'S)VU#U_'O[/P];TK\W4;Z<$& ++ob2F*+]!1rm1L,SH>=9Iu9!^0cC!?pmWQ)TI"n'a*S^&_7U5m=?_Hj+8pG&4^rAk +k%DI2fAL9]n#Hii"oeW&?n$Y)rhlCX59n5PJp`CJ*P\A]r4qe]a7-bb(O/e085[\4 +4PfS(/`)F%A]b9:@<`\F#li`J/pKB+He9i1=C/3h4)\t`hU;1hQ-!0% +POH;g9Y\J7+S,sN(.tad)?3@"moReCEdokIb>);JfJ*-2l*1;t3sr/D-$HJ7_+r4"-h-$CI&?OH"(o^@u? +?$U2tj6NDV.m^?^lj*)7HM,-tbo]l7[BSVtLf)5< +pfcVUDj:J%EEPo!I*k,mMNLAQ6\C%,=`sI?i;W`I&'tm2+#2;o45n\:0+i4o^FRaf(:08S`c[;HZ5X4?^^lJj# +LD59%lc8Q4nGanFH9HXpLGap:]W&Q7X>pRE&$om!Op2gUbKCTf`C^_&kPY`%"&"or +nVC+(Uenn*<5l%/)$e)0e3YZcUP^]me6%rm=kuSn!br^*^W#=9r:b6M@.9Uh*;IQ1 +Z^JWWImNjVJ[:)H:h1@o-'=:f_4Vq1W7r/\Rr0[=l_\#usU!]:-hPZr.j5$A\3rpoit- +SDSP*GjnI.n(l0(O,DX4'mAHOKA_bO"mrejaUrL@p$B]!R136'?<1JN:jkc%>m:*o +=j")!UaIIi#=[);&)faK!2oX"Q/[`":\4YdQc6S&`&\YiJcE0os2%_4cO%#%%^F4b +[NN/#R?S%,LF>:[kb!TZZF`_2A +.UP]FSdA8)J.$m4!5*sC2oosWY8A);j[dYZ2#J*Ll!Rlf?Q/il79BHrr=\0-hpPfF +_\W_^69Ss(Iu"jl=B!@lgtSu'4"13X^h!EDK`>QcpdFM/-:@a2)nL-rYF[QY+AMj5 +qL[pA,5?6KDo,%(UN=5pBpfaD%!*?AK$H-L8"7VSR`To?qN(G2Y-pQ>3[hMm.6(HQ +s(tHU=dtq8lO!S7r]b1:ERE03e]gkDgW;.n7cfqc'JJV%E_nDcH7?4K6fJN8%2qSpCgE5Gp@HD0kh8:5(8_7)C$TQ.fG +8*o'>(t]AghhpBD:Kd>W-,KVoHP]qrJk(#/-F'uj4g?%iM/o9=N:8`p^RdQq_k%'b +n/d1/>jn&8C`[or2?`@h&uoi=S?6)B!)$G1LaY^`T:[JH(+) +h$&08AUp%,e,gQ`EMqcU71D=JG.hV!M$_&WhLiUJrN>P(I_NR%#"nFV^>, +@QO32jY&t.i/:d!!U#CT@?^C< +!<-Bd[a0^\+&=tL7qe:al,8k'5$^LihANLli;\a+)pbX?n'M?BJa_1FDt^D&pEkuo +\:AJaS&6S6a"Nf21Y:P2h0,LFipS1e@Z^M\Xa[.SulCJ +pI;1T<&5o +CkW4s;C2=>r]koded:hU4_/.I2,?@^1!0Y20-G<: +^jk>F^>M5;T*51Irr@ODY\@.:H0@\#W$3feM&)s\Mq#Q0.a_EVVC!NkSK=X0klBQ$ +?SorEBEis[C!"iSW(nL)3_b0[=%+$r&tP^?-a1U!J18U^.EK\hrjSec3bhY7:UuJP$A^B>fY=UkEkGo;F7]M"ZgK4qcrCo$5:C$Z!db'km +'FPi&Jehic$T5p!05hMLG8#8[5GS;I2A]XM00#Ai[5U9^CO?n&J>McHQDS:H9u".m +qn^SK]W8H7O+Vb?bG?J*jk:h(5O@@'=6mHTi'M,'6r`27I%qs0!2Y"piiUi>6#`;A +XF2I-lB!T&Ap8'%NWD8oiDU'"Xf0Y/^lCFIVYSTQ"m4H2E'GGR/9NKIG4!8C;Y5iR +%n*b&aos2gk&`MVVnatP`o*ZXIbZc[*(5kUiMr7/[A6Ab/&96'GT>YdRAVHaD'ZXJ +0mhm"/7d7-_9u*;i]:.]>Te:sN%J?66nO,2rNeL.a:9>as!%llVa)r0/P&3^8sl3IJ(*"kWqQ;t8:s +Ka[A-7Ih^cf5.9ZDl-9$_Rh09T(ZhtNgi16oBFp"mhRl@Lg6Cb"c-GgJH(PaSb(_* +b5jct^)_b&rO=<)"U[p->@e9Rg-9B(\_LhBF4R=&-gLAEl7BZ)XN<'VptJ49a0b[& +,lV]Ze:%H0hLmiP,1P_t>f:o#ehgnN5A)VHP`8m]IOX0i?\HRn+DEmZRKlW.PfsAV +o*,i@,X+0q4G&WMnMZ5^0=#0'@dVCM6OV6%4ZK*CP +$',GRmh`p`lMhRu"0]$5Y,1u%:%>iYia%i9FZIRM1=<3T_5/F6W@6X2`)>sY8\ +?FU_PDO>!1W.g31?`\cLp(6<(nj)-m2H^soeMEt11Y\L`-CK9]BdF\(du!R +NBZ,O4lLBJ\;>7TUE0Ok@"?Sq2C-<76#F)E+8m%t$[hU\r:+G[q.*/-Yg9":\m.GK +1c*?,;OI_)V1qYJo#G"0'8t;7LTAl%'c?@Ol.5m-RJ`<[,hJ,B`IB-t0nqGN%Lh8k +%*/tWLSR3"N1B;RR`XG)p>=&=?,4TYKK;(/r'jOeW9m9pT-<6EjGCP((K,1V3p6X$ +B\O@$QVI\I93XZ<,uE-cP>BR>obki#4h3CT.K<>+!<2P'Aod.u%"LR_MOnEK9=:QA +>0rZ!ZnO;I^BkqUmq)'TZh8OF2Wrks00!1+G8E7#20hm"(S+:0!Sp?eIEm99o8hMrT5MAUosQ'=$H:0S(B@51Ft +=G"52r]qMG:G\ccEO_9fHXSgqYWq2.G7cD3Wr^7d$P40sA;d^J0RF:V`Mk23^sPI) +_Cg%PVg''br3:)*QcdRF.lKp3,[1'T9>NCHhK.gCjPZF-iVskl^0+q's,S-F2ImdlEgQd3Er5m1Jq*.bYM93Y +PCCf<74Z+eXb0]$QSsUq7DT5V@G:A.Ca&#ZhF&WIW,>ZDQP\nY6Nt]0>*JKmFBS8_#GG+jECI>;UiKt];I6l +<B'e"'Yp+7n4!V5F/N!5F"gYXR>ts\V93V +MCunHbs.o;s$DF$6A;'Vn-)EM>Qm@O!m4N\.b#ZFW5h%W0:\!"hoV?I@.Oj`B9,n7 +FD2]?JqZ_JM`;W@,MpNSmGRd5_Cf'**&T58CRt!qT6.6,E2*'1gV3a`,Pg`,_+(Mt +$sPrKe6pS6D9f>+5-P9.euKP\577$PL:>BPG!HBgAGs0Yi(g?H%>b%8s'=nkHaR`b +Qp1cR:Y=3Vft801Ho>O2%!Ste]Ro;[K6r,5]`*k*j/a3M"\S<7_@F&j+TEIRN]94T +CR<`!HL9].gA+OPF7*_HDeZ%3d,dCW_8+'!">RZ<09V[Sp("cK3Fm +2[8_No\D1cH,/fr=OcrHXS5!anA"j,7F.OQNk7a0.<%K05;`Lmr-ru,?,.9SJ&mTO +rfd=WErHIEQbku?pB:I,s1?GK?PHu\r6LG.4p?H-rsAZrSpC\a\s2H3Yroi2d6*@_ +l9#&r,7=!tGKLKr!;mMms5?=i>*Pq6[$LeJoE)`27-["^cToL"fo.W4$UXUnRq.8@ +K#MIYd'Vln/hm@41'8=(WK-'Yn+-t.n)tO:VsF?d?,.9S-h4s9s)\J8#RZeFajF)6 +0HbA>p]n_$]UZ1">_j;hE\Y(i2d2[lVSrhW64c:ZP9+lRq1!^_Mj4.;[Q:#2s,liD +bQY3Z0FNsOp;Z>giD5nrJH&2eN)X)i,TWriV&7@;;h.hg3nj_pT5IA9>LATbC%2u2 +#W"rC$,bBU.j*4r:0.C8[Jo%-qOEK*3ChP>p;T>LR@M1d?aFg>(2'a#ce2HnGHt6< +FAN9_eI(op9m!\Tl(9dLr^7=I/b4BCNZ&TX^Re*_?X]"6VTai_"t"`OKe\3qkqcN] +U@i'5@0R6%\^-[NAI0-h-R?N\bCP(W_7f.XICO(=[q@.3Hp(qN2/%*JZY.OpHbm1, +3M:RFTGQ-p,0)U:-[+NH>\H?FFt]-d5_C%b2ehCs%5jlZ%SWB;#1nJ8>d+^!;@b2? +S0^AWb^genM6dI-cq^1nTQgX3K;*>HlQ7N^n=I7:PVoq>b1f^_I_up@V[5TL0)`D$ +'4Cel*L@.PABet'j36-0'&6?9dKJgehMHB9r$=,<"\H`8F_Zj=P>$WT&bA;rT?BF*T*[d\/g2\A;dnVGe;1V +U\O)Gd`"Fk7.@8U=L8LOcA+_#:mGq?8JN9lcE%61T`ad+Ef/[0r]uh]q#=j/jHkb# +@kPO>m9ZW018FoH8s,r"&HMk/o7=^W&sX$2;G_q-J$F$%TqXc!GZd2,>U?sZ-,.s- +Cc]"=-Oi^r7'Zo^^BKHrmQ?BqV2Gg=jrqas:7L,#b\X/8Mo0mR]NZ^fl2\8C>b5H- +2"J,VSp%%5rXW-o_<&>4NC'gLh#&,@-h?L=pMLrEBEA*.3PQ2Qi&\)Vk=,aGgN@GB +BjU2Q?lhjoQYA!<,?O!B0$^[7U8,.AqK^ +#e#1ACB&P,mDZ$IZYo*@Fo%rbHJD#WZL4`hhGI"XVj_d1-2k8MqGh+X2>+J,7DEXr +\(XBufThs)f.)+8P]kq;l@)Q'=Xi(b2D]9feX8m0HYEK5_PgaHCbrtk1f"Ic1aRfN@tjA,pF]Bc]V +p#=eL_;a7doYKOlWZs+daOMt!VZ[p[ +%Xa+>OGs3BTk4U7?a7!"cu=7S6WQ__(BCNH;ks4=10i/RD>q@*q(#2.$r'0N6*ROZ +Vl;H8E$[H@$m,L/Tlu0nq,_jjDm5MVrOcBc1*]*fh29^skb-WcP$a-9s#OCMS+E`Z0cKX=V]O=J'fA^$'(3 +m-CAXlE1fbW?:G8>3m9Z)ICCDBbe?&mb7=Q-LL,g +OZ3iZ&N(%I!"Q7hVflMr+oV^RYT3u,R_a""S9^&[F)^m5;(a=h^Elbgg%0-$HOD.U +T8c?rgY!F`K*sZ3#(_>2\8.7r$NU+=c,HRSK2b#gq/,u`0DR,jEgGU[M'/mRDmYfV +grpKT'sE'7h)M9XLJT5T6p(6TaZ==?8Jgfkj+&!7/#S9c!.W\;n'>S2)#mN\cIhEM +.lh,J%[(J]F2YiS/u+)K_>K@Jn6<\H,lCJtHPQ981Q[b"s%s:e9SU5mdVia+"$OZ+ZN]T@--se&$#.0'nqq +Xe3cp;)+Pm^[a\=lqmiTIlDZnpqIk37\s^K2u+ZmeNO\ls'38t+4jM&;-pDm=K3?> +VSXl]NWf@E.g-Ugs-3\:s)9Ph%PCWcX\b(]e;,PTMn-!]>]!D!("'CS1:!0#%B.;V +62BNr>QM!b?jio96G[jA`*OL'-Z_VkI*E2cRJ(;6F/f6W!%@odhD5(_0,V+:4hT\S +VD&Bk#8[AS-I8T"g8BSdd5p7aC>-'ui7SG_T5o>4D/^\^s%*POBFYO6l);i7XSKB8#X)??-.5855LgcX95MKE-+k/o,X[k.L90i>\Q&15 +lGgMPgFR6nlBOK^aF9qRr]lm":FY[kEnW+AoIH81,oYaakNn)M4aQ^XLTaeQh'g9? +0*)!"@RX\Wr(iT+AFONOOU(g)/HAkX8Gu$orr`:nQ6\XDN;k^6YGDDM_1+TFA\!r] +^QNLf&nc3PQd[85AiO=frkdQR'@k-S)\3Z&V3e"Y+O@A\u+;`SdtDH?f,YlUA=jr.:jFa4_G7P_:t]\T.<`8iAc&LD8$i_Ie,r4*5MZ& +#*n(*r%64\Eq6^_XCKWFiNW,`i4m5'eXB9h`3gKXcZd0'J9ClUg]c`h%$#HjW#$F% +[19Kl*T"&SBo(pg,@1o75ASZK!r`R5lSl]Xp62BI;J-M<"n72Sab#Rfo5>7SiB;[S +3.k3R(@NAg%Le-f@`C.,:DNl!Zh+b#XsLh`J8a:1lY/oNdo! +,7Z#.\'t/n"lNfX+1U=f.p(Tgc141?+6s_L<_FuBM$"Oo.6rGn@%g2#Srum`P(1bG +T0h(i6Ieo8&pSus'",e9e]>q&bbnJFP7\7>.:ZTOdq4IJoB-Vl6jJf.V=X64R3UM5 +RblA[g!6$k8f/.1^7P"3SoT17l.Mi`.snG]pY@Fc\6[m8YI*n-0uErV +@HW&Va1VNlr;TRjc`4&*OPl`FIq8.X_#M#l?h,;s:\ZS[reu%QNcbR$F(=PZpa=&\ +gb:kdPm.2`p]J?^U8J%1r,Jl#YqFa(k=ZN_[:9T3bq6rkX;uF7_@s-MYr/- +!-Z/)K_oE?T6:8)WGkqpE9`1FbQU9$:KIG6qs#O5(aTE[*&K,DV["+Re[WruT2=Pn +/`8i6NS9I_.d-=+;ATnHNMj/_8oc!9)#"!&=oF%tl&<9P4G,n*9:i:/T^k&e0j]p*ZYk"tA(T*#s]VNEf.-ZI1cU@.b;HjXF&\8/eq?d2:KYIrmdF:Pa +\\WV5`lW<&SLb&>aA;i;)CI*;8XG#TS?)]u_M]Am[]apjni+B)MuR2Te.FsU.6YV6 +*b\1;YkX!/eNZ4!^\=u3_:dgp#P7EOW1"6aXmJ+rqhsmfMS:\mTcTY"hE%pUL5@)- +=am5l6UJ@o9;^^@c6[Tb,RlEFN ++0GTn5h8uY$r69Rj1`V/q9,#VrsJ6;LP%Q7]E=9JnfK&mmc%9=GU$j=57\RTr0Xtk +Z"T]0)jACX_3aMpq=#\#Bu5qa90X5o^!B%Dk6&/+1MM"q=1"MO3Q,2TOBGD,l7'gt +dQn3n#[osca^nDo@P[*&V^)4k/]8rB_#I3Gs""!)$U/Q3!Vl-3"B\(tPU5qXA-d/) +9B@)]+38qr/V%LP%'kEDXLZ6Ug%SXJM7o$8%K%#4ot7_]0PGHMdc-D1kp<,*o.E`F +Y-#M>3Tm7SGX!?pF`(Fjm"P3V6?-`\jQk(31%;")0dDAO:"klG`jb*c?>`1AJ-2N8 +]s,0TaH-:lM6l\^\/m!AJlC7a&]^65NcigrNF_Y@YhA%hrS*kI[p+!X^/QP`;^cH_ +/KZ[X5@jV(h!4^2c&3/aa9!F#'fZ`-L4,uF%B'G)7K.6.+-&G4NnN!F2I4$qj-3L0 +\BBdbpD6cboi%HiI9(RX1lEG*BAV64^SR-!TbX,lmr@#4]JI+"a#M2aO;*(Qr/[YX+5q<5 +As(A"&M?,Bq&e^,&4sE_aHJZGbR!p[>]e'3BeM_l!A+4H*C7brC,@ohc\TZ\;*&_% +,R)!._SJ]LM:r@Us(.Bm/43Tm&OI=J*2UU[;3UU`aDh0Nk4C7Mi32^_BB\(?1gPY0 +=#@fN1o%8=-=>5\iT:Oq#UY%pR8O,uf)K:5s)?!@+tZ'67_t]li."Xs_%(rel;W6hAb+0iMlMi7$O!1gW +#,q?SgE+VDo-_A;Q=r][YbK0AdNZo?$Rj)`k%XgSQc7iF/^Xa3=FK@Cq7VFR3Cg[h +5^jNiM]OKC5&RAd*bpgDg>p%4+]L6Rf\ge+oCMJ(Y1W=8KGW+=2I9)3=?rE:[U#EO +_+SY`8piXh>!JSREfs_aQ[NKhq\2*Cha=+^k.,k_\EJTsY0]Bm#Xu#JpF8RK*^T"cETh$P<+I +*GH4Ia[#6Fr=nU7fhLMqB8O'<&cUFrP:H\#J';*tImMYt^S/Rip+q@-s4#^h[K("7 +>L=ecD$7.U/a]k,^ig&$pBhZHoC8(@-u0m=&@[orhP&(3lMO!LN%=6nOoGY+d3jt8 +\,=e6MDbHt^He<:H2#q5*Md6Vl=@V'R5n$6IZlO.PWAo.a1fpDlQ=;N%^Q>SJ+*LN +0E^2DZPf^Q#!X0uIl2@hJ_Aq!IBV8t`g#RpTgKir!+P+m^jgoF,K0VD8I8[kAfU)- +s6!!Ns*"#O`;aU]_lFA:CUciNCr3@j?T#qb_Ij$_j]6c+,94l^10A2 +[G(=0ENR.5>SN"n]k$fFdcL3,U`/O]J$,f$^1qVI;IHYJ +&6B%@J<1t_i"F;m1o4>PQ +od;d\2g#e;R`&I;M0[$3C+h+Z\6dilT%eWQX7oplq^jF#jNe(nGML@BCVQ?5ahALs +hCODcT1&YYJ("$t=.''nj9L$/OnQF=ok?uQJNq*Qp[Y*Jj@8"MOq%Hf=/G[)jdumG +BA!Uh$Y(.WrqR9+_+=&K4nJ?EFF/C0P.chtpa>QtnXhL"geg^Zi!u]Ir\XPJ4Un0b +)r_-65sDR&jHma@>o,:#.fLe/!JLJ`1`/*%:cR0Z&eUu8mr+R*BE7PJ$8Dba;^Pe> +99^(ira99JM?tH;L0Uqq_tP_,j.A07DrclW,"93@?a2"io*rV2>Ir):BsupGm(T;^ +Rc$HnLsO1i.JtJgR22rc[=*ku;6D"k4&Nd,p^]'Vp:MNia^TJ2Iql3G&Z79Fr$CGF +C^npY#:%lVPTf$!D?l3Or9OJqbJIfe@V"L!3!DQa"K"YCBe]h!(dhJc0nfC^=:9GF +'occ2jJOERsbQ$X/o8/_Hr3+tChg/K+bMDGF(Rsl!m*<4'.000, +>nqeD<"rqN($,9D._[L>U5*llQ[,Vnipo?^c/e+6Wl];e3)Ho,OGCN?=T?\_ +;_q.s)m"l:p?WP6#DI,1.ndC?M.2XH$Vb>=U9[K/)W2N8##fOI!.XJJs&H?(Q89'L +J!7:MP;[WH.G6@XhT*&Ka:oCdr',T>]`;&SJl`'L"#^aA^r?df4n*7\dP&/S^DChf +>d=D"s0!A[;!4I/Q'783oC(B1rV5p]KLlUO$Pn4E>DMtqq#<6J":!7)_#NSlPlpa( +]+!)bga2a3[PP.B[_`parc]5b-2_+hCP)=gie$hFXT&M8i#n22S]%E.CcS^1pqI*F +5S3>bJ2d9\s2?%=?go.d"iC<,fSQ/tZ+\d]Gdk.kVf6T"J\TOl*AAkW"h[LEo5WOq +,@,SLV^)iC^J,I``D6U9QNDL-'-OCg')mO"n+\M*r9r(As/OlN$(&X#q(Y%YN6AB& +V3EFLE%_D2.=hlr.oZ]=#A5Okj=5itW.j@a"bKZ$f(:W/jBm9`,kmHtR<;V!9G&qY +AhBqLI3/Edm.oI:hIbL&G4t9Gp=q!AG+XD::bq8d#k")PUA/[,I)=j]P[jB>rNfq5 +LSHbgM$k/tdXb!EkL0nu@c'qCJ(M&'gPB'$^KpN*/NgR=i;&4H_S36V0=]dqW$*=bj)RH;2/ZoL-;$>V/8VnT +Ci@'ecuLY7m8eE-SLcJ)Tr,L1R]dOM_L@#"Oo;@1:O>sY#4bi^ZN:'O8lM;q-TrqQ +GYeqZL;A@7`/RIDjKG1Tp47oJ^gup^_12GUn.5GPi#a0C1k.;(5=GM/!eC:W*r<2# +56c`;PK4ie+6Fi&^T>'mKE#WfU]]6UiZ@/ +PRE9OYZ(4b8=e)e?(]Gg.3Ch_5#HWHi$kUaOE*,OLM9Y]"Oo7/GQG*!(g`*4)WAH4 +09e#A_/E%>$j$:$c0qJ-IG,NBg[faVM/1"6Hj,WAnSbN +cZkoW`RF*V:YY?N*6rmj?1`/7oQF7*ohZ.h;N(%qN:sq!l_fp\BTO!EP/PjGB,VK& +)3lp.Y=-"XN!d3>(&]C=qDg@kH$mu&Uqh?!6$H4Yn"k\"@>DKd$bDnlq'8&'+Xsrb +Pl'1tcBRcj6"l<5Z@@lp0H^1<3G&Bqqt8$e2sG6`D05 +i3#:HPT4PXL(N7"3P(6K`9c*2.M"e5$]D2h]3OhliogR=u +#e*d+%7IWQO+ +s)AtWa'A.17>k]=f#Nqb#l`^lnpelU#QDK_(1H^YR=ENpj*YY-oEKYWR5rW0G,n9F +C;TnBaQ_fVA%fs_ESe5D(R2Ed"901M3i6_/7%UO(!S$p`oJl>i+eXn7ru920@E3+\ +s&o&24,fdk8ENd0D[tkLP5+L0L#i1++;Wc6[U$.MhPEkqs#X'kHSH3'KV-]3a/H3o +JeS2dUiGN's50XY$3.j15jj/\j"nYbU%?=!2;nM<7B>>JI`fQ5!8Q83J,LD4m*bK& +V+T^#MLbq'rXeJ[.elC#&,_2uJT=#L>!S4)?]g(=aed'P:%ptEcUW!!qS]a4cFoeH +_07n\D_`Sn3K)hK+Ul.(iJ]d9fm$he0T(W[61*V!7NGip3BS7iIl;3=Pdl5ugF(n_ +qG$AeR6:QL:Y7eIefF3;^/+WZ=7`Y;4n*q,!db;Ip><3!hQN.+fed+6"P7n_?LQCS +etQ%;9!t`pqeD2ND0VT`.kE":S6lb0@Xfh(#6c>fLbT'P`r7'O9B3Bpo<`S^XE-0C +(IH`.C"F.7ma&lSTD"#qXq*".'#q(MbA8io2I/miWlfR/,oar*l5CQP^A2Aao6!gO +NQh)83FFEtEgj2n2I4XaccQB_+m/e5s"_"krj(H(i.!t`8b`hd^MNJ!RecV#qnJ"u +?"Lj%*eaLN2?f4d_#Nb/CJKt&H@S#4r.4mRi_QYF*n++*i#h'&.?Y&5_ji?17C_PX +n0^poHJ%_4d`+j>3O,!m;kFI"aP&pUhk)fO&=sd*L!gMf;'9]SoJHdN-[I$;_:ZKu +\6M!Y5:b)7-eL+3`(7I%O]"q!nn>sDPV>;H:[-!?CWeFijWYE'+m3P\BB?G+S6/k_smG= +3Ze/>:gH3h"Ub>1II?)l!H.JFll^!H\%nocm,q!LEl?^8U@knH[:J,(FV[ZHI^!1M +>&-Qk89)O#K_/uq?b#E0+0l\;,\HXfF"r'.,K:47g^>>j8t,q@"8(LXf^5*?5\?cF +0MW5\s3Vi`/NpiaXSuD9JbB&#m9_?BRGU+5"T4$(e,,!^a6`&_ESK*nO%"s,s.;oM +hfC*!))FdQg%W%Oj"/#$7u65%i?'Dkm'HL,.D5?7#t\Z'VpVni?0qS-4#_Zjs2@`- +I,"KXkGF5hIVMRlepfSE?a#4pDFKN.qfPL.WA`J/&`E\F6hMtSnh,R\$f9.?0dD%% +PBd(r-A::9D)/8Ea3h$!.7!ht4AWus_aMcg +O?CLtI&3JuQ!ghD-3"#]T0d[-i.&"8A;7[2Ss.'O7'G?:?4CVjdlC44o#KT'XM)m8 +fM3+Kk)/p(^]iY&cCIq^;SpM&bQ'l''2j?Xk8N:YR,(]F&Y$gSerPP:*t2?.#90pm +1hui2`jHD1r)oXmSp/AVCBO'U_!LTAl(\ikk?&#\lO`o:N4IGI$=IhZgYc,->5:gV +;W'dLK6FroK9hR;f"eNpJpcK@\cBd^Br@:sk9ni1VPm4"4I68Dql8NH3-TcPs"F.s')Ckm +Ja.aG5B-^7(@9@:^4[=H5?OGbO\?5pi(o$$M$l;fIuaPArl:SNl!TY/J<=;s$44WS +,T>X1Psg$iJt00,1Yp4)@WM*M.nd* +FQMQU^*71s"TCt>fPa+p,MiQ/!:o#4.ek!_7uPao_t'-"4_c4;*k@u.e\KfpWc=aO +Tn`%IV\FeN[:l8fdT'cGI,\(l6kMs3Ig2VN)J_Zt1T/2qd$_=Tia)PTic?^s6occ& +/NZol4%q@j7"H"rQL4RQ"OD"R/nZBpl%.Du^]etNTCK5dS$1jrrb`MR495B'+[0W@ +s!U,)fDmL(OEPZKDHaOn-d!\thDQn#.Q>J^0\O)eaf72FYp.l31<&3-gSQa^[";dg +So%c$@t-A^nReB1XZY@->l.b$G5&;4SgFY"Kr#4_g_C$:/VfB>`Jt$"k@4]Bi?Kk8 +1YXGp"^OhD#(!I!:2M9H[M!iso)nn6lsGfDWFm=sC%=C-P8kAk^FNXHc^it='k8kX +3Wi.^2Z`hHQ+=P]c"tdR3Zm8j0+1m2Q\SdQ^i9APJR(P?mA+G1^]C*>Lu*0j^Di['EViNl&= +1Pj'[54JTm/DS7Ta5UMf$h>)lD]tjQ?,*C4VZQWXn'Xc%062FtCJuU"Cg?St"o\m8 +?7kts@gc+@+/AlLs6dkAI^0H(.rH>Yi?k[aDS9iWT!,RH#D)oe<1\d_m1TWkP30QB +!WVn^PMjs!@OCibYGEln-I2"tnppa*pie/8KgaJm_u&pU?2Ql]:l^]G +Y+g,kWct_FDi#CPs1&UtefW2g_(&^72R@Qb.K=,2"U`*u@K+ +ca8cf,)I6QKQm%cRR^Ui%,!i_[XC8QrtMbh.An!?#R2k&O=UAO!Or4O(3?XLFFOTL +@+qZ,e?O24i^poL,ObaZ[P)7+=sP9)U;3pOkLos[s,XP]E/O82ei8e&HY,ArGEV?u +AS#n"@ajD.6dn'YS@bJaS0]F(P_F/'JB6*uS?gc"f1-41KWX-Wc?[`_cC*jk`d?PWn7.,R!4q?!oeQ.JZ#1C1THn]pSAeV2D?1Z3 +]A=KJ"3`uE8WC@7Q[fhbG=J5u`19Jd3iH\`nLJN,$nXs1eRI3C46#t!;Xr9mfsNR= +\9bsfoh+>*.`JN)q;!OFSP%_/$Hnu\/*LT!rd)-7>.GJ+h^/%-buOkd7=NbX@q5]> +)g%$"%V$KJpLi4gcFL][q3:>DHVH6;(Wn)j[eO6iV_Ft5Q5#-YD/ +DQMmq3:?Yd6/[(#n\"Q-QO%Os6/9'B4$_$Egi<>*jra+8HoUVf9alQ00190MZ;LmN7SF`a#uY#63GX,jA?;!JI/[1EN6ToI3n90DJaWjTU7 +AW#%GIX[OOCkVTe;Lh3mi2U1,?%T*B?ldXl%rji'+4YiDI;?c_k9Xh,n-#s+!lL(6 +51Wlp3D0;dK+#PfH#D4X^Lp^7"NS5gS=EXibt.]n7(mAL3rljmL&,ubaSFl(L+#rU +qVdnGr')'25]7>]^jjk_W1/H)`j0t'^=B7-M-%U-(Rthdu@0W'/:NkK4=DRC^g+=7Q@co +F/"R3n1tXebR1p(DQYFDd)'@'_!+(AjoPAC3*;4gLor7TgG:J5+'2RY7b[qPhrYik +6g)5*r1eo&9[1CQh4[fkM(TL[#D)%S?oMm.:L7FlZn!*]oM/(D:-T>!l/Vra.DaZ^\B;Dm*!bqp5$$qjE'_I +B7U#cU<^H.BPNGdhj8F*B\sifnhTb;KE)\i_OEVsYDm-WS>!WdaF++S^aI(?p04,b +GJ),fqGc8D.XU)\q`OL-\o&('V4ES?s"^qf69l@_'(?tUJY)(pM#Ylc>GKKZT)lf* +HfTnSR.G/49eY2m%qN,l.WMeWdpulGfm)\<72Lm$QgBkc-*jSJnn[@XQ3u=<9'M=- +l0SiNXYRf=&e!nmRJD/sp1g#VE[gGa1#D:aOU-'c-:Hq(n3NH>\EmHF/5md8;-q"lTH2=653fIh841KAd@'+-W<XmpnR)E"[NA#KD./'pToMGEru*RDrg3T +7/RR>j$\=@MuO;Zp318KVj?A@@Jae!hHI@sL?5+;**:OJb4<:`AH%RZnfRLW5>pPA +O<=MC]YqSHVLF_qIZU"@R%AYVO`QJHI>j_9>iP2dTD8UY!`lh:f9nb0kIB;8enB++THA=Y14gu)5=-i_=\"1P9Sri?hfda +4ik2l6@TFAT3smUiP/P-p9"[V?:u[&aJZaIJf7D0(l5/@0pUA7OT +'\Ik6s3ZjD&LAG?jLhDr:P?`Qs&JI=XL''d,nouiTVZ$]Sk7j)RN"r%2$a+jrt"#b +]k682ejV^1ruFE#pV_bp<;Fq8F^SmhMuhc4_LmS^dlZ4EATlZDY878VQ?;W(`!OHY +-d2E``-=*M>4l"'r8'9EVW\-]Y:KM-kQZMdT)Pq!KqB>@@/bX(*ri3]qT#=u:Zb6c +q%#`@=B@Jk!LUl0T^/H`=!.Tj]ZR-T&PBq7YTa)/#Kk0de#Z5nO;a- +7sQt9\A_OFMKU6sqP?]T#U(C-N"i=Uu'7n71X+.fVIA92Sh/9ab +0XR&]:pm3#I'Joog\W.5i+`d1In(jR)'e=OJ_d8;rOh`!8Hnlj.WWB'QV'#+K^!Z/ +.Y^/Gs"=*4$W5SIPE3lc]B5_28Il%)$W@+GjTh9)`;Dc=h;Q5rTA]aqhL7-:"dsr* +FjE5f/3N4tQQ#sKi"*]j_"R>+P5]7rCNB@GaSbo;s+M.G@3j7s(_"?+P6U@Ypga;5 +J7FtA\DW;1k5q7Xkbaa!cJj!+e/8.a1]Uhg"X:dl_L!"D,U@C!?MGcEUN9P]J&:J, +88cYU"0BSi\NjcL(OXBo5eO&QZ6MPGL$iRRb_,UT?9fkF6qi!SMbp,IB:unf"1\F; +Z@WSIO?UturOa9F$[VRabci`[T[;"F_>DkNDB':(lt;*bZ,>@!;.&fjrh:f#VW7Mg +"i4q[TS=PajTe%%:p[hLN8kQ9@=HM2m#8-8WILeCr4+NCYLN3Jo/"/+ci1X4mhdk\ +"Gi=eJt%6p8G*NM]VR/d"ZJ]_-Q,='AoWo,6F'6N*9+188U-(C_CgmgW$4i5;.Zbp +8p+sX3j(pk%HbFF'Nq8nX#*4E)FJ;a+Us1IF6^,:;G!p^\sIh^P4oSS0e_fq&9k<- +#1qtsCSL>:?Ip13)]\qbHBs8ubGZ3+]-uo2c%O[$2qAjGn(!Dk[8;`H`+8ZT1B4)? +-0$A68-'jD!<2"q00l=_=r,s,lPe0`3!6`j +N:mrNE&""+mN77VJ&jerih5F0`<\/IS_o2t91@,X3U.`=n+4XXqd4M'j:-e1r&U-/ +DtP>*Z5s8PmgNmKq1"\`r+F".O!R^IJH%";,k1h(bLEt5=n +s%L]oGQn+3RW8qpb[l"k-!YfT?2YtOIQHj+pj\)*bC!8n>NZFq]&3%5[Q"A)bT?9L +)nZ3YJl=YA-(EO$O)FNCelf]PS0$[K@hbX\8qL%&gPQ=,s'#X+2["aE-@l,SbRX:* +Pmr9DJ#s`GoEPHW7q-K9%n4qhCfCS<=MF;nT%YDX*?%qo9ru]1()_LWoR8)YEh+?MY1tpE]=h3%oV$0b +:!)%fBjfW@[qZn:&25L!).>_pbIQH"f$]t.j],Q,8,W2Vkb^8\=V,*E^gWmU=7j&d +jMhEje_o_Dc\"eJS4U1Nii^"g>^@+8KLXco3Oj,O. +NDC5Q*p-MZj4LPm1efa^)(kH=M0[[rK%D=WpUiq8_(*/%=b&ke\O;+o6')IZFf7Q, +m-6*Ej^j2em<6F[]lqMY$U2)S"54n_1$TG5)YaF(a6(-)N6QhZG5f#Bj+>%+9#d,< +7QGUrQbS'tVVo/W-hrb0L]?'J?d+!M2,KG5\Ld_9C".T'W#HY0JG?ld(AKlkIc,[j +3$pZD>mA]mJ+#F`SVfH^+7E*i"VI@EjG)88JcP[#rej0Uqm[`_3fO=+akYOS.n.oOq;tGjZTYJkGiQI +?Kf>j^[m'F'bRHpcZ;22pg%:u[r$3Er)\$9b8Oh/Lh[cLG`'mf91T^=4'5;fcDh)r +\-CLQP&(CEi/7CjeRK]FF749*I%dMO9!j!fGF9(J:+!enUSYYEeD1%p(`!IN#'g?e +J($CM8N607MrmR3C.9?7^lo!]`Fs&h&ir2t7U%+F@mISSg)#7Cci:=fq-TuAn-kN` +=2-6LhU4H7Zqp_.D#QZ"(Omp,I:#)Hr;T:?bXXbKp@X&Lo"BMESTYjuKOmS:^1*%q +$uqB?..)dK`sNmEF>&+(DXSd?*fY";re(4mM7s!6@$ksT/V6[/4-FPu,T3^,T?;H] +Jh1:Vn)[[:rRouhbD]LcEE@\jXapqF.C\G[T.I<1(V.KS$G1MGh>`Mq"TB'WScRdL +$FaqR0_AO2\dkn`k6_kC$GTXRoCOm"G]>P/Ip4gkqJ]u?YPHI<^2kXiq%B_Anfu4Y ++De0Z#N$H>s&$GuaYs*-na)1gf]XW:EVZLQQ@sM)B%R6fW)Dp_R^5MWI]XP(NSbU++-R[hT#" +V%`1Wd08@AU4-dp0X@VqR,kXa;BIl>me!<`!<:e`c[fCjSUM1(fG]ILJ=#5m3ht$/ +`AB^fWI#IYeJd/o)gc-=l$@:=c-GC>cR9U%6o9PrmteR1T5`8IHH2G!Ik^%/I/^gP +.Ngt4h?^@d6GVE-g4qs'&-X0ZZq"G9,=2CCG*+GkME0Q/IAJj5LU=?_.!s'(+ltXJH[4=P +PSnsLAc+ubrb$NAJ&kRf/5,u+38kAZ![&*57=N&\s-t2-)*\sYL]-FE3oGN6i%o!? +mhcYO.uaBH24!WI+gTMcW.&r3Dr@9RBP?5@%:``cq.G$1%s&aiJL(2JM(j0g$p);aGgYGg@6a0gAeIQ*SE-?q4dB?aS$8j!0,?&%\o^_=u>ggLSY+9@M+_UXY&$RC?KK>j+emfUHoX2>Z\2Lf9B@_!A=gqOY2)S]M\p`JT2G-NUXB +@85rF39^Y4"\iZU7_\@9id>qY.c>Eg?L:J#4];&k+S3V/i9Hg2jgP;kY'BnCNrct"&]M3diVRU#4S[V2I86ihk\@ne+a(WiK6(%JUWWF^[J(3s4bpnuh3^i5OJlFH(R/`6L@.V.B +m%_W3O_+d/b!(12&'c9)rPX!]n,+QYD0Op!B(b3fMK=E!'`n[?UX`Y'Jr"[^6$n<" +0?XjBP%LGh?EU^YeGPs$\(CYlMZ8uo!*`lnJaoQlB:h$69bQb=N^T(M#\*8?Ed;nT +L`:O41An^63B7&SI>X]r+M:QEL:(goi>\Kl4/DhU]/>kpcUS6\%FC? +6.g+/iP/r%rW6QIr0i@AOK(k_*A*5m2d(2d")n'/R:SoFj@P;0Le^B$RlegqU2e^?+F* +a!j-pUEASP:([!S1Hbmm0+f-`It5:7'Om/Y)eBU._Yi.&A?6td<3_>?:m+K%PVR)T +\W0R]O01<9#Xs=01o)$q8>S&RbSoWJLR+6\es93/gXB=W_M@F):2.n]UjH6"^+MaQ +"rCDqVZb=XKH5+HkIbn]9$fXQlef!u&"d+Y:+f.8>i%C5IRO/gse\P0_X'QKAMq*T]UMspG/pp#^UdB`$M_[__%<[]lh7n!KI)j.P*mcT/\H3\-]/`Eree`MO8SOSb +GqbfgBHXkii'3ZV+4Yl3Skir_"-$kS7_(e0PIAiEchqRj,0nEq4C$$Y[R+b:B6F=I +5DVd[ru8&R/QU1%UMGkF$Z:HJhR1=DJlWc&.tiA5:OH'dC^PsSb!sb&0@fhZid/JM +Hur"FZH2@$kObDdgl.j!nqdNF%pI>7If>NQ/8/TK8rh&[5<>,0@@;j@#FWt(9,8Wn +H>=Oj=*?aQ\l=,X;>e,SEfdF%QE!$2Y;r%%5Er0kcR_+PPFdRD;8S40M'^khLRM2: +$+-XY/]qJ5,0D'dXl:^8dg-2YcE*Z'fWH0#Zh5s946JJ,o89+ori6(roBs6rm?tFf +;m3i-^aY@Cc.d%,-d%3C2o&1)M[0'O#0MIXn##u4r1rkY)WQD1bSbtm>;=Perm'+M#S4@aDuVq +AP#:<=k%.Ui[#Q:[U/c7S>Ht!L'iEkI\9C7,>`]K?8c'W4i;J1N>qAAQ+SInGoP;\gWop>$ +:j8#*&%%',ck#)@NSA+^`;#43!/),JX+iq0!1k9A_$=G%(g85E#'E\YDiK'T7*\Ns ++I\oR+':i,n))QPi4baXAQ/V1jm/Ni$P73QQ+cMGE<+pX78r^T&(P_C.[GOf0>9:R +]D2DeP1b'Ad1)Xf53-Ci8nufqat;[>M^KU!9!lmH7'!>^WPaE]k7"iJH'Ddr`*PPU^7$Ei;^5pm#tkD +=HSPcg]RR8A]$#bkH\K'4MPa!g@9f9C-t%f+R4SE]OaFAR"$6e`WQ7u(X;tjT))te +V]HpsdiWB>>NV&"lfbFMj%eU5d\B^r=*2pcO9e[6s/Mh3Rc@O>`u*5<$)4@(i'7-( +Alf-te3R,X!:[a*:+d.!Pk!b53GFh6iXYqOq9nGE*0X$MJN6a)S>((`pT-%JG#-DC +rqtO<.@8bOcaP3-CeHbf/F)l6aAXaeRU3t'ImlBN);ip]GOAW/aa4'=%tYal7%ATn +\^6dim7$r<4/"%jL,;!u3W1AeF?AS^DrQbmS/[Ok]SaDp/E5H;^lpAf6tj2pt*2Ek5WZKOAi9t-bD_b]:tbK@kPJh^Y6iCHAbqZ8/'Tqngakm'p@WM +BkjIi'+iumTJe2*:9/(7j&uY?!+7mK!dh2epIH*mlCQM2)DoZa26BUWfIs=)(:^RZ +Wg;aI8R:Z<1J:.S\E=+lO3Bg9`PtY21D18s$Y+D>+,Sf17/"?nim]Who>Q"T)(N;gA\e`Q]UnC +r8H1[EQinmYqSgGg@pL:YQt.uUjXA"aM7?HJ=,sHR"'_fW9K*SjsHbfP9taXb5=l& +o0)SMQ@2;qmk2dF+jW:pb;g%U5a[!kOE-GTr#IHi)fG"D>fpD6;1IRpr$/b8N%gt/K"kd5[*,f/d2`Jd8PO6Z@4O-VbbXst<:'H'd:&U*;q)pD* +W0l_Hj90/k_-0,QL;/Tb%-[`p"]:tKJ9m/]E0Qjd4X]tHL;/h!7)X6bF$.BGmp5`] +q>L^uO@hjf8jda,0S9H%++_]:*r&7C^If&^ZP%*hOtO%>:F6cYAg!EYK?K2s($:<% +/%MO5>[Kg3/1$o?MeOe8]Ng\Fdj07>F*+FN7qD?AKl,;oc?Zj>6Sbpbm[L%qDMj5, +WlK\1MosmZAJb9>?;U=t<<2/SZ@R9mkEL(gKluk7n**ZG'HJKS':Zs5T +(34+:2pc@tkIP>;?%P[BX2Or3:$p<9EU:AfoLI*S:US[O*q>W!M +qPRg=q>]`jpZ7=O^6kUSr=)f^-_p*SWbNpH__a`em2N+Y>B.qh:OJFd*k +j?4meYHu,FrZZDsfmBc-Wc;(D0Bf'@+6A3;\e9h,Gf'bg3KPAm5j\A +iX`Wfj^&ee5m-_NkfHHfEfhH]T9g;V.f6";cd1^D*T.%qn`k;1]bc`Lrr9od7_(JYYW/c$DUr-+[l_n:)8 +I[2?7gP;-]bKKM;kPOiYI!PNOg`-iBp2P9<;/YN)&2346W5(d:oMb@5ZiUjGU&[;C +30:KE,]k]t^Td2e&HWGp&G[26`LISSY.Kq]-/j,XrWE)[_E\6RH'hnZ%e)D,,>BYn +(#Z9i'W\#!X'JA10g[m@.#aY%U;p\M/\e)/8Q0]RNp;Z=0+#rkC6aTu14H=qn7h +ER&Pb(\unFr.ubdC#D38=EJa\fE8==CM]Yl;T$"IBFPRT#RB(?2:WO)7oi=ald1&^ +rnrZQA6D%+/0.$3f@&NbH8Lj!NUA6erE)"i>s+!>&0M0;VLP[Nr`A@D5O9dO5l<9@ +pgBZIdLQEZhjHs\2T_\]GNSES`mILD,TY/hl5 +BHt1BM,l;Yd!`C4dU@BTs%JpQZ)PSMi'7#:f95)=#ti,o$tGAXD5rB#LrEfkXL"PC +dm%!5hj_W:s$&PVhA8u%Z&QFT%BZ''qDe5<_sF7=R@:l,R#]Sd8+Y5#Eib%`^jgV< +V*pJGU7aI[TlpV/5M.LtJQ1,G%X3`H8qisM*-+b^;LNG+$f0;^4#s+O0r9.V,AS62 +]bni#f9=_4Q[J1#gtB,he$"q(SgBnP&D_Ra]'4\!KdT`1!;Q'[4QbFoaMrS#oln:m +5kcL!4c*ma*EmY;W:kC"r=+gVLONksa=TQS=!8A!c@+'/EN:\MipG[(bMK[#?C"5* +(@_?c52>\1^$Abo/EGkBb3;ul`b46H*fq>5k;5Ba<1Y`W2 ++G8U%%YD,EVcJHFs/Q^8a6N,T*s98]SrZktbH$k:js*kuZ?Q"fM:+\$_R*3,3fkH$ ++4#3\jsBE1T+P_3`u8VG!.Ste!.Ua%LP3pe.+M[l&qKe=)ErZ5^DFWI70qi?M;Y#m +bt^Roa4TQf#D%'PHZB0RH`HoJolsuic]oS]H`rA-)#/tn&q8JaIo[[HGBAW?)#emX +YKT90#@[TN$A.Z^YCZ_2[*POfqcs'>+b84)[@[^*M#RI)A]R'Zs1KR`0E67@s#q7s +J'%`hR/d'qe;_S.E"g6?+]&%I#0^4YuLf&0plf3oqkqeDJVgb]f#a_q['Z?A97Ddu0FO_o_H1jSO!+!r.J#!a:\A +(*K:H#kGY4"eMFi09DF\X(;$A\3X;)0F6SI;R$,\/70jD>>05Wd8'&nR"Tl]`c7@8 +4V^;%\^2-<'DP.PIPPnn;Au9.KH!mQ7Q&l#Nj<=H/TWBiMZ)(ZR(hljY;`pZ&NmG-gRL&e]'P5VdD&ggbPfL +WpENX[2m[o4n>I9`J$>NUGZQ-]W?T`4>8F4/jMn0mT@p3/ErZ\nptLps+9#4S6o)' +UCfYdaX->q*0CCi0!M&k7_P7F8Rb?f"<-M)hh9fGa6Gp2TaC=O\iN\n(^0"4Jd:fC +\GhQ'atSZ"=O*m8+P^e4VmdC>Do?-iU^J'PWLfMW3,\LdJUa-kuj;G:=1-cs)4F354bo@NGNc-?g3K]m%dkgoA::/ +!D`_"r_K?FF2m)S9"[$Qp0\SWrk+cFIPR(0IlQO"RX5TnZA73lS8`QR^k!t"9N;$p-Up@>#mT:CdBFb_s'YUF"Q5a]n.&V; +VO@YeO6=6Ib%oatNN\7q(.RF&A!f?1pj[eV"]o>(BU(!h:Resq/ocYaOn +JDNAb[Yi(k%=KmM0@ou6/Vi!@UC?a63b/8WmhrFk$0oHr;JB$cP$G!mFi]/OZ=(#' +"TH/,)J0]hg-kqZJq-a"==KM?RNGO*bIkcZ4;6VjS1J_/]2n5RDt +0,j1qR75T#4j\#jZa#IEBZ+K94e`p%&,m[G5F$\90'`i5IlX]\ +KAO(DfX$G0hnP]YEi$!eWM";UmhNWfK7hI!='7\eBFD-[Eh>;=@C5Mm3^d5;7Y?6< +.D,ec^KKRj%qe(tc"dR`Db*C\"?d9WG$^O#9c?Lr/P2H6J!;3[!YZ5]>n\cUmI(kecURQRVJGQK#rHrn&R#MbV`/2>.Y7d_aLst\N=Tncr/(LRk*t*jS +?]fh^B_+tjhfY`!-Hs94G6tL7,B_(&Tlf +H%?+urIFnTQD^8;+_/"qJnH_N'R9%7"@l=;.\m/i.$_LYdQBu +^+?S(19nPKZ`Yd][O/&`-G?RJ#QZf^c';EJ.W[7B'jdu=T`]0T_kaH6%&%rei6)<)34js(T%)Zb$Oipeo]BEe[$dS +nMS9FqL!bh$GUFFe@d0pIA[p.WukKR31!6`Vb>`IosKF@QIi!bG=>FP.croqci0\K +DVI4XAK;!SD\l2ZrkC>Yn:01L\RcU^f*ndVfk;UTo*b%0eUNh$Mite'If"`,F,'A_ +RYM(As2;D;lEL1*/%1\#rRiJc@,%J-T9P-FEsN^R3n5S)"D^&->nr +W$CL`[iaOhRt!>-_dN7:a"/qNoY#9m!2Dm54aD:p;aImO(GY9Ch""Xk-U\?1=&pF.uKDX +9o#.&_hC_/"FXHQWBA<%6Hp-M]tiIFos/N<<*Tqc +f?;5B0GT7%EoTqoR4V+mI?X7 +%seVd]Z@1(P#d#rY3;UPlADe(MGMVWhm//HV<>L-f>9'+-8o,,Pi`jl/K7!NnjsHYpO`0XjH;^'F4"ppl=p +IV5S:_at%IZ',apRF,VUYm[#f8WD0[7r!(^F +5qBP+^$phi@OIKq[f*k?5R`/$*qILRTo&n&P*-`XhBPbRcC5r4"j5X4J>*%[Iq;s- +^[iNh9ORi-PC3+Z!*Mbkn:,?^.,oaDikL-Tr-'C\mtN!H[j&Cs&,_cL]Bq"_^G9W? +.g4d++MR,Rreo\rm,:0Va:CT!T7HrSU4hDQ1]F,F*n8Sulfmh@\qd00;a^-]re(6; +')WPXH1Kh&GsXSr)CO0(9,F +o4e3Fs"abq/-l9*^jdO2"KL)TW!($L6LCIpIlX;bBq:G"(?-k)ks&\iCXhn=`HJG$TV%BNc\(Wp&D9\_snc<#i*eW6V9g&d[LZ^U" +[#&g'f_5S%)An)k)t8>PphG]ZW5[53rOD[.:IQ5e<"Bk+Y9-EK"/p,K3X^cEn@tTC +"]>?is+p][f1-MC9P-C*8\=Fb\i#Sr!rRQ1?el=FfGWC&Q@&f0^J`FeR+&X:fKWs +b%1hZmi>=Gk2FomaLU$nVk&Dj^j;A`43CNK-Q4b]-0WLFDj_%/+&F9pF\'Cj&ot75 +C_h[@"MT2r\";o&AGN2arc4HoRc/$i#3b[V9j,:Y3UsMfq<.<;f*a0S$S$%lQetB9 +Gad@5RWdsf>kR^@`Wor&G=`PD6E39]_eU>a2;AYE\J\:odFWQhcZ1Vih@`e.28FYc +Yk2O(frL4scH(phZlA%Zp\G9#G35C`c9M1UZggYg32KfA>,0/**BOUSSW\ugrr9ib +4L>d2nUOl+#N=i$ISqUJZ':$((j=l1Lrdn7"C_)cXY+k,5@OQ/JH(L#Ot8aeP)@8k +!WNc0'7='_hc]:\^LR.HA,1siSj1j)Z0gGbIUO)+b,HWpGu\:Ok*ONE_Jg1hWHU]0 +c3l?(s53;fLB)SU9`Y<=%K`p;2;(8RhH.:?0BK/<0`M1'48(l9J1MZoNZnOt*^3PW ++$dW&5c`=Q#Z-7!m=YZf(@4L"`.'Kb)%f]`+_\]dJCq$G%7foSVuDGuq\qo]+X?q& +n6\fnkO5*UGjFRBWdGmkY,#6=eSS3T4ja=%Ze'%Z3nc549#l"r"2e^+R#-u2bAMqHZbd!gM'9_/kj06Ig +=;*sh?oQ,WMMr6"O<=!W!oS"EL4W5-0SG$-FHMCZf'j#:G^3G&7:gR@ANC)gP;Nij3g_!2$nkG5'1N!=j5LZ&:.s)/g,6N&]lY'G$UP-f^;48).) +4?qQPWu+/DVp^$QIGFG/7dLgY9?&i&VAs+X$^/4m=m*AFBMC^sU)cRq-e'@kU40b? +Ea@&,Xr:Ct[>u'[mW:;@2QJGJ=BF"8*cof!XW+A@LT9$'FgKi!'2,>H+PRF'1[g7" +2`-.+g4?`q+lGSR/7b1Ppb+/oJcDd.J_/%1@P!T3/_VO-#(L[Os1GjiRf3#IYPl?4 +R`GD-Xe@,s_p8>_-W9@5n6^m>)i234Z(O>Xnn?IrJJ]()ct9>Cqr>f>r!t(0m#Jg0 +'7HQ3\/s-;IX4!4g3!2/\ZD]%\I\g=$##ek"T2UWkZGsqKQh#a?@`&_V=cARZu9p[ +f/\70Cc(Kf>CkpEKAg/PU[qi7Fb88qZbYf,s$'>I?^;3t8UKWbBKt+W +!VKThI).WA\H!NN7r16cA=pgi^ea'M&/ik+`shuN#FCBt;sMqWg9lZ#",$ZV?\Tjg +c7TEEaSeaWIO.=qE`o>.*+g!U'_*D!]FZ;P:s\mbF8`rKCRHFg,8d_&D5"_)l>N?l +S#AW=]F\:"fo_h)R@2lmc9.:75+eR[S8<(7.=..Vr?/9@G!sH3:^8]S%`Et?qKq:X +l_M\Hr[#fq[L5:-.%QTXZe +rN!()k_ZBWn1dCP`9I?"'*N(>t,OaM4MJ:^hgf#]c"@ +`6uc@"/t1>$LeiWqLF+rXq2`!+K@\p!>#q)a0ml\(DfWFpP@ +QsKcB,atp:^^Ka@OQhd_s61oHs"fk"A)!('T,[YjaQmJrBY06NkK)%jMct22$S( +oqCKEn8_`XKq2IT(0+KL?!S1LgH$]sS)liIl"`N\NVpM7g%4.5QQ^&6#gaXJIQOO.0+'GKO)eqcq:8AiM_c,c +2OmIS5EYqa.-5>r!a7\?9f`4tYSL,e2H#/`p@<%^j8&%S@.ch$In0U++VnV'>gDS= +c?V:p>4gcPV +!rlmJ"03iRq3h6o$W&f::409>!rTn7D\l.,&eG1t%HGO@EH'_h\7Tm/l8h"p2pR/E +kkt"1)=aOYMGk0MZl2r3&lE5?qo?`6[UeADS/H^^9L:HtIq"L8(ad$@N)e1)-tCE",dZe0:-3K +b&Y;J"5=b/"fIU7dg@Q)*t,u90E9Eo/u$#aLO_-g`io/EZD.PaK+m_3,_G@grpa$$ +*.gC?uH=d;*\CuL-krq^CmXi$i5)Hit)4e-3"). +RJr)F!+g>HQ1rf]s4tUj"TK\Xs/"]hYU +8p,E$9dQXE.]TX,]"TX(i*T]N<*PId)*pu>>'3h[6WrQnde&9W[//48\oN$[Z)0S] +A4:N;;:]$:\5QTlTt1CUhD3BZlK;cEj)`(4FC//*"W$sBFr4T``VbKZ?rk\`%*+>U ++G^:'HltYb<%I6"U\3_D]VuIgX*fo8]o55Q!!rgV+ac;=(P1uX3&f>^e=7a&h@QB_ +@5G4?IW`gi6e,23XO38c0-3e@=c`-Pl`?:L=.V?LfEJf"05b0M//ht&:D +*`Cl6g'B.C@)DU9fTru(ZeZ30_=mUm?%e"fS8c0"I%`4U)P.*kS9fB5kuM0Hrk9@$<2YZ;21H.]7:)UT^rY& +r]apVgQm[,[OKQbgE['r+&X]c6*6W32i:*`*Fl;?_nE_GVc1j#r7hp!X@3i]O9AY+ +,OE;*@Sq*LJ5j`!Z*8k?S3S)V5gT![d/=:G#6Of\q7aXl9d)JK*t@$r:%MC%qPXf# +e-1E'S3VUr/j%:E*Eek6GVYD,.Y3(!7W^+PELNsj*G.>ICcY3=4rn$Hn1lo3ld'gD +chg$Q@(YJuFOAEV0=pFOqO+87B:]kq]jNe.Mjc^QbqJmj/uIh^8Z^(=_FdFDfO`l%M.3lEo'jJdQW"^SPk"J?IKe0uB:9@M1S5^2mJEf@/^gf +Gju(=lo6XD]\A>U'@$5=k+@2(:7qRCDBBIjqGs$W$tX^Ed8YM#Y(1 +J<1_91=Al1\IX?\!0.-Ug)"Z%'G>0aAj*^uh8O +NqgrFn_r[,#RgPue:3*8a'ZgU=*"JNquD8^=WCra"doLBB"kI*k@b&TMlNs'f7L#sCFeGF/!I!U;a:lq+:G-M9W0U= +X*:?tfTBo@;_-"S9<7)GX.gQU'/%u/T5+%Ob4D\!Y.+7q%F+OXHtZRQC0o.7\"pD7bTpUl'W6./L^M*D)5CnM+'itd8AAH,jY_Z]qRc-ic0.GTcI?< +%7.2u9oHSKnDU6lijBIp);S+K2e3"pMSXFHs-3.IQ`U"c_Utrdf9kSKM^,:G.+S%T +>'iSg-2it*B[ZAaeUK&]2TcU2_>M&+NWB+:RRYk-]c0![](]a'Gc+(!0q7ZFSNa^L[`.*7%_iX0,8+cP'8m[7GnHH*?c42/W(l5ftP&qI[ +!I3h%>_7b3Je.Aq>-r$&*Ar??a'X07gi2/G"Sgjm>`\/7=c[4$ShK4NPQ-A![)!X\ +_`sdd@QF&1@b.U5[$U:@)SL?[eqKF)#go*.FYrQ)&?U-mob5eQdJ?E/UTU7edK@;b +H"oEg:Da&/jG`NOIu(fH:gS64NR+/"KqhUm*TD.bJ`@r.HOs1@!;PjAa8/CD?DEH/V=`hJ$QP +).f7ps"4?PZ2UXN&2"+EJj4uS#a<#Z)/!Ht=qU\-NJ/&K2%#KYI9qRZ2]UbZbp',C +JlLucb-_2(Z1i3UZV&i_"&np;Shk^#"!J3u`C7IEP'\jFeU-bu99s,!f/*aE@"=p] +-.PTB*BT;K`,Ud-$H)4Y^459lf"5J3A2nYUFktYr]h.RPR.tm_ca22K($/\)(+2]0'nAjMfRH._*IoO>C0DOPn[t(Ba^cq`6i(o#U +b^[bkr6MPJ1m&1DLCsTOZ5nm;s57\mNO.Sh>K$Vun:4/V^aVNEoMSGs#S))Er9(ck +k6pe:a8Z77J)ro'!*fm3s3V#9K'."$s5@\f@n?E1=EcT2-jKH:9XM$\(,XD93f6F5 +lBRR'Z4adFLhQc0dl/?PphT/UW+t4O5[BHVRs0j!" +[MJ*r,pl24KR]J]jA;"EC5PMSn`nN-(L#$?;?A1[cqHb"1YN]Ls(T8oi'/A__Eq"$ +.A421P-@k`:)*\Ys%+j_pn.2"'#?=/oBB?\Oo4P>PimO'B4F`?Z+I?9?icYp5lP:f +!(t2Pi_a#Eb/$\>=c"s]eNqNWfpMel\T4p)!N6'GWqq&,H@$QW]:/p`8sK&J?bq/+1%LABOJ(HS@2cTMM_e0WQYi;[Z3q9S<]k?\OE3?t;X=W13^=/Ps6h5m5EnCJ +J&)%J3c=tb>QOe!k8IM*Lb1ea/j2`WhG(UWg!(0%'-ps/aDJSj&O?p +c*CT4]VA'RdFsn(CJM7e,X`Bp2L[tr&J^R$-'](A0VK+G,na'0URJKu7Zc%(&c_M8 +rbM?^Fhn"XB5>7T$K3)uJ&EiXosUF^rB(!g$eX>?.MMql!F;SbS?]%)rqF;p:67[k +(PMM?*tWMk,.>7Oj_J:NR],XDMQ;:8i?.[K=M4t>lDJiQ@K,_$`JaLu38k8(+2cur +2Z,8a$DpV(/b7<";$uJGoBNrKR?+TCSV52ZNdbrsL-' +X-E`Y3tB7%IgSaHe(+4$b^XgsP-=.r:XoK-X5^)^G``.L.-P==chY'J0E6a.p,B/* +5Pc+u7.eMd&H2^L- +2R?b*jh<;.^KTP[lMM=S+2M;RYq*mj'q*_>O1+cf9QW,Jhb4F4-kSNulC]FCFGT\, +s!6\']I53J[DP`A&b)]A\7eGE1[X@Kppg\Q3u-\N_?(#CS]U0b1dJbePfkn6E;8nJ +2$'if(4c4QMH?V56;TG_a8CuefED4WM&8=GTr0ZICkgQ4(%cX7%sD!)qK#Yb\XK+m +]'c`tmOk_oL4ScITj77uXo^/O\$N4b>MG@Da^qM7ZdmaSb%t+GlR;4Yet!,I^5=7L +ipYY4I_"aVY\(Fnf/:8Hr"%"/gUBS&Bhd1g6Mbqnr#6A/WJ'iL\$U,KtXZ^5TQ*pi19K@:8&#E+1^"e[YA7`t%Pe +\$Xa?m#&+]s&4lJ,d0:5hB/lE`J#C(HoFE"rQqeVr85cEhFgV=h+GNZa)T&tLV_XK +OW+hk$P0"G@2+tY_&IXu"6TWLFo7s)#/9n]`HZOQE,"!MECp&P(Q=s[O2W7A"8o>/ +cWoG"Su__oDQ1KI/k,`LJ;#qIrH +k!mRrO9s'E;HS^YTIn4F>IgP5K1l;<`C8m5'5$nCLL!Ur__A[,T>jHC\[abtS1 +;P/U\BW(_A7g"WY+^%B=8rcO-E=jQNi[%e4nRb(m=Vu&k7d:-H3ao&&WhZ7Kdng]l +(WM!s!;Z#l-P`o1_l[B[OPpkmQp?G +mrL;o\m"r9@Y"9I#>op%,9HB47XQe4rV'r)i'6W2m8X1]/o+\#)=bJjd$W?BlmI9L +<=r.8UN-BC59Sq\gUY.D/quTPhqQA*f^E^R3U8,S3\nd6_LIE9WoP&sl\PJBX!?lV +@"1#]ImmqbAh&*o-M^YM;q47KpEW_3eX]u6b?fVPDE +Io@)*h\:"s>q#G&(!$G\md +@_"VuUuDuokMl,Hk]k4&=Zm^>T!4,rm9dFZAA>Z[@%"R)e%o=gtjHo5:Yk-Pmp.bS]S23K$HC6 +B>FQIkue8jD^VF/+"X]'r-oDMRsuJKkBEXZ*^gcmN8K((\I$8t5;7j.Q[Q2Ep.=1, +hHdn\QsS3JRFe$;e9qT"s,,A$rmCN(jNafXPJI4Agf"@:s8!aGoK1dJ<5j4B"ZP^2 +S>nUTr1BoI2c9*/n:.GVIcgUp_Qt>tOo4:"e9P6d>Y=VP$CEX+q-8gbVP=O-\'MD]l!'L0g0I>&MZroWYMH? +2DsM6nhJ5c5M?qR-bQnT>6+H%qGI7LoqG*FT6P8Q1O/focnZN^&RGB+!]>H-qbS&V +T3VA.-9iUkp^dBT+oh!X(*7%RIn\[gp:k[sW(rcU&Ml4ZAB-8!TS\N*?%!#_;cAn8 +)DZ\]H.8e+mo:O)5/36n:uq4t5XG"bWI?iJ"J?5rM><62Eho8e;jMYC`BJQ=5Xl5T +l/2h50o4d)d6eOI=NV+F35e^/=h("=OU-PRL0qsmb?09n#h!-12^2n0hc%K".+p#_ +.ej;K$3@EJStFS5JcPcfI5T`BA$,':1_.$a#2RO3!CL`:OFPW81urEooX3_Ls6ocE +/`C,nr?!9d +re:VL@AH-8q4[225Nj![#IV4#[]CS[s+F^7L#05Y!qqr\q1[aTr.jeAZi%<$=TE@= +n)lpl"U+,_^6Z&FN,&@Q(]VK\qjb.h\PU2I,#/(LC\]P1gAIFZj0/Eh^[tPjk7p/c +-%K@t4QaD>pOD6gh?((!$8Aa3c;,,C^](\pd@^?:=//4js!5gG[Rt6iHdSgV[k'u+$8DE?]0`NTJn37R;_u.j#.s&gI8GeMK_#eq' +c2G2=^\!WLF7+'Sa7*sD(/V7E!f3J;[V+KuF?(D^9SQ0%uW3__& +lqTE]B5.GGa`MA^>n^"6_XF"Q#]l^1>Zf>j'4g+dEqrS)lr.P+h"'&c_\M,0Wc*88 +14([_YSF^RRonI&/hT5gYC!-j64:EA"lKeCG1D!%[dheWm2D#`piP0_dG*mQ@)k)( +++slS66Um:nL3Ypi!d-;X">>!ALV!SY9j(3!";mS*#gf7dT8kJk^&rqK%Le"]LYs7uL%2e3^`'j%l/j#bst +&ATA=!6TkPVFt:t>HMBsdd'!cET$l"G4M>@qN5X_M0%Vfg?jOfs2_P9O5LAVTo9gg +Z#-(9='Yq-YDOqp2U+Peht#(Mn]\(ZVXHLkh9G/X*66iqhBdO]-iCr59\pr058jS! +.(0CN!!3s]S61*!\FguP"T4&Qs#;ns)`W2uW(\%2&MlSKB*S$Fpm:G(O?C*`+6rQa +_RGe/6h70(..bWm!,%sQ5Zo"%*Ck\t"YG6Q/I4n$6)fA74#nKF*bg,[US)q/8-(_$ +#bNMRGf;MCKT7T6)b0/$sX%htc/> +C$RO^F;b_j4#F?:Je:scA\nVld)e8Wbfs+=,F42:!'+n>$]df2q\r[T;pI7!eB +R-7VF`WLeG".mX;*[/o7&gDU?7u/i&W!i*,;*=QgR3*W^GOm$mn__--c63%MV;Vr& +l)T:#nXpK1+9'669oP`rmak=LC8tK0G03HpOs3iZ0tcm\>Ba%Dq79Q#.\qK#IQdYU +Fj(X?fIo.'?>AUE'NMdqWHhLkn:nBZa<3)n91]]o+o-`tRlg`N&Gq<(..!abe_t+l +;D(EGpeM0G&'?jL=VhaCpkS@ql"5LpIP.)A((b#.JkT;!'sHld5ia9o5;9hY1k>Go +9gcTgBVS=@F+6t!`:^iLI;`b(=1Yt!t;Oc'sBr:&;Y0lnIR&h +A>Xs[FI86unN$]`dINuAh$*iJAGh)Ao]>bqc0`UKQ=k9`[^p'X]'$]5je$g_\FDO= +J'2qR8O'XqSmO9)h@3]pJR<4`kO88F2F9>D,aJHj&CEE<;d4gPH_Hkpi:MbB@adOJ +mp!6N/`5QYnNN5QP=7 +8(Y%njraGS=p>WiJXebR>U`VetHrQ='2A4 +W]gD9\CpH+NQ/_=YBY9(o%[d+?`sd^?"BS;.QDi^E9c7HrXTNp0/83aESk#8qWa[; +lFQ?1Q?)?+IXG3]JVJOj(,f]%G=5r6%GCNbEXM.5h$@hcL&?GX4T3&VqoFJ/-$\@\ +q7q(;;./L*nFf6Lm0Cp@4,2u/l'8m?MlY86BKim)eg;67HZ%"a[q22bFj*!fU%s&I +meQl9G3?WXbPqk(64:W/om\=?IYb1X8,r&%qZ!ACIRpS+/at"ls$ptp`oc_;DaK_2 +eb<+BXuTm`pamcm/N&.ROL%bLXX2Q5'Qft;ku1F+WiiYK0-roI8"5"h'J01"4]1rA +pIf3a#oXKqBW#;?r-dsSi'0=iIL.eQAKg)qZVfYIZVc?Y_8/A.8_=4BV4<'p)r+@_ +cA=##:dBH8#iYI0$goVf-dAThB +)Y(82FTJ-$KO92FVZ`!*'uOqBO@qXmB]%_Fmkct8N:e4fg4Z?'0[_,KfB3D0HX"l- +:JV@.YM2hW3?++I`"mYt9aOGk28t9p&).%d8(,n\fl#O?&m[jRmeIOZHm/Dd;=^&0 +*"&Q1JU7<%'cZ44F$Y6a2k]8dFCh,LSeorIiW_Rp@GiMm![PVa!<2I/f9\o`W!2h# +$`#2'Y9sUiY^?0fXoB#>*?'t4+?]Z;A,dGA4.9j.AickL7*>=_;YJ^O\sL:V^I4?: +N[nI-WF*_]rfYh23MZF\9(KSbbHD!?#J=%A>ig7nA*b"23%XR&r(#AQ'E$I'T0p(? +2Z=<[+)S5@".\UA1F&Qjq6UjW-S>Q/pht"(C&%unV/eZ[0=&%D-@[BO"ifFRmM +i?."(H`:"i86gG-i;[THemk*I\FTg]+.r%Sh`P@X6LkB\"8l)(G+,Xc1q8'o!&JH(6)s)FAHkah%< +!r*E]+956uk!+HIS7USg,%BuT.7p6+\l'*>9&GBj(E^2`7q0S_m!O^^i*8IsN?La8[89E"u]&!qRAVnSRFl,l]a^(o6I& +hM(q+?dW3\rdVcPHg?YFnGbUd!;M%p3J+CeWHdkMkNjM7M95=0PEk>9o9m0#YOLG\ +>kTLH^dR8hhpR94ECV[d#N'jTrQ12s(jg@&nBPe72\>=7jHm>U=*Jcrp>W7*?/E=k,qtX$h"g%^UEOWa`LWQB`pcN4K +4Fb=k-DP;KR7LX_;,Bsf9)-pfW2?,S;R?S(5XF3VVkD/Cnmd+tKtoeG)1aFPU(f-Y +o,hCJN%U*ApB(Bh/:c]CU/k&U)#fp&q^he%"j[1YJKNS=OtuP8%Q[iQ,G!k +]Y+udqY.nbh+YlYGLhsCNWtZ[^=626IoR(uq\2(UB]oC:r!ThoM%-(J<[j#Y5LZTB +HB]u)J%p_ni?(JKk=5r4LnP6p5J@C&PY$[AX@VU!_%AC$M&+H6plE1iH2d9K+'o

    s6u.6YMq:3TQjm>[@pE*E9V`qW0(nPG;.E4>Y.DW=B(ocI]2 +^C<(!_H2B@Jd_K0,_(2,%%eG*o-2gOiBP-8bX=6/%(tb_Q+s`8r7Rii/YFU$Rje>e +#j8_t!u3=s'o.r_PEX4R@Pm\M)=@6aQ+nLqH`<;%@MA@Z%J2m:1]H%FIK@9Ws0B-S +Sc=l']rGgn6e?M:fNr$POr[.%^q\7l%pN28kO2eFp]i'fps=+f:n"Tg!`40fK5r#V +_#g5-]UtMuPbFfLOc&R/NV4'NbG9hpPfEWW!D&J(JW?o)Cel@\8iGs%`#CcPejj +(`k=l4!f:K1?8"PqWP>mZP+):DDZeF)ABc/JTD0'i2Z;6moUfEr);K.HI'7^r2M#` +STbDlj)3V\>Onfur^6YirpQduJcBrKYCY7Mf:W9!;N@OnL>_Neu;`0<,_ti7E[CCi!;_2fJk6h%b^1RK(\aq)VoJj5M +6gW*Z:_k?pKDq!:rU7u,5;T@tVcKCbDa2P&7P1:_Xf]t*\+\2Z6+2FplFD&D1@lOGtAkMq2aKA +i'.D[(1>9\=9&j20f:&iIts^ACTbc^lagp*]s`YZHfMoVL&\*LrIdsOJH#qj'Z2#m +!).i4rZL-K^S@p_f$\&F7b`BJD4g$^Q_k+,oo\bPHi*F(-X-JYgL&mRUTPd1&OBAr1c`1bTo_EaG99" +&,mU3Eo900I5(\2>-nG%F;81*s*a8cYF(1B%q%%-6(nk*^jifg:V\(B5=Y*u:qQY% +=kE2&rHP7=B[1e.a*si1H)0?n7^)K(65D4N]W7W:GYq*o*mCtm`a.44*q8*!THJ-pP +5R$NC(Q]^hmg''rF(%@YUp;pVe(7HPi1F`Kca=1)[ikJ?M#dN&#"9mDiRnE8^gKsu +1]3"FT1tVq`;%,t[-^)t`UM2qn-cLn#lJuCpSg.fTE6k4>(q +Y@_:::'tmU!rUpchq/C_DYYX24BL/4 +f/NYl]D`aF_TTapD;X-7H[=:,]4ma:d9O2V]s?[e/+A$cF8iW,iZ6j!T/+"84WaWT +P5C3&rJ'/>r4gO,Ha`MD9Cu0?;MVGpcCfMrCHhgu?a"_O!p9\g!9bBA_"Gi+?dlS^ +s4QoO3B+c=F//T8\(eWJ\A"rPJs*lWikAjlJRNFd4bNRF/fi".s.4OW>;Hces/0Z& +&\@aLB#crb_`[#[qqbQO>RT<[Knh;!J$eu$]aXhZq>Wa*Go*$L=P1",giK4EQGEVD +G_q%Q_<"2d`Ehpp<*TAnh#;l/JB.1a5BPZbU!5g$j[\R#q]1LmlbTPR/VA%W]j"!M +l&*ifY+,Z`hl/p>7pDWt<0rQ(T(5d`!6kI+O[9#is#hP)"8P)hEh<;/g4'$Jg-P@D +8q:RH9G[uOIosP+i?.!D0i_ZWpGdsh5L/CWBCh4cA]37Wl=qX;Pl(gd?:dH[o0;6L +=,Qf9/>,'mlgg7O]VJjEWR"6*pQYAuM[)DcHQ@%c +9WrPOrXtFmhUhBMB6Bg>*rbaRN.#qsbC5kOr:16R(BFc)a,e1&hN2QWDB>!.RT9bHf8l +'QpU[)NO%9NZ,mW1&QmR%g@1VnX^g.s+TOkiI`Ks?'+,Y5K!COG@!9G$^Ui1r`ri* +cA22P*>8"_.PUl-1(=BdM%Mn!U>^[3O[7C:+nTX@)7KVZOJ%qK9F`T\uNlT+^ +3Ts3Wfkr6^fS\Ln,-S[>?mc*ujG3n@5Vu<#$rT.ae?#"nj6-(1bPM/7JRCorYbF(0 +e/&f-J-*AL+*YfH*cfjC3ZHurn>'9;b,mkp0q<9P7d^$qMe:VA)71f<%;fC>L/(`c +>#CEHQ3FW0FEUkYneb!'+G^42[#gk#f<^Ze?;f*@Rb6O%#oa9%@qYK"X3MT"F0Q7! +3bJJ&&(>aj-#RUTB@(CFDfWa:D8UQ<$3;OU*enRhoA#OT3(>/j-A`U8,F\2#M",\H)-VYg)$GbWkeP +m?hM:jB#C:0<^PF>I.&k\8FD=/-5ZmM-Zn6qr)r(o=k;*h=1?+ +j>)C<%0&hceX__cH`?*gU%+3:?/bJq[)f:#ngMU.m47E2FfAYHCfci^V%e]j]6Fb)QlTJ.52k=&(o4O^t[38,fK[FnmhG>)$E1Is0/4K +,j/V2qhYpcA,dGAmg1T=3grWon@tUq]h*BSSCJ:MpB6!WZp$^f5"3uq-Ms"9q)?UQk6nEk4["k(_`\,%!rpZT:Y#P(kC?>&CK'78h"&>`6S\'p +@%dW!n\n^#o7uI);Gct_SSVsi;)]-H?[/f7q\R?R+@Q47^U+ATOokQ^?<5XkWbH]` +gO.N#?=]?9>@kbbF\oKH6$Ks9NO5pT73X-mKQ+1GC5]<#B@,p$iYhc:Z&;o2:uhb/n&R8H13iMWM>bY4*.lATaEA)(mZ +eJW"4A+X#rP)GUt`/D5P7m"W4FF]\J18&)K\p6jP05fgiX^toSP7W@+X@F^uaR^:J +?9Q9,.0\Wh2&*NSmT%_a.V;5jMQ#0((F7LH1t:-mN1*jG'/O9sGO#DW3lRrBaB%Kb +U\(+4WOpoi`O4(27W;hk5IKaH!<0/,`]&=pS"?$"n*HKa''4W3gm3BZNp[PBrr`9q ++QlJD`Hh-cG>+?)p#dLu^L*-*rr8P?GC_PjOaU]?m&c,%pFe3qI8SdLM03QlGPX)h +e3rEOaf;g'nP4JT*;m1>lOunRe]KZi(F@sdan[g%$f+?[n_Ra8(D,3eBd\'u;tnO8 +&$<1!/N7j[ZYP`ilgf$'HT?6m0JTL;#.1BG0;tLPuB*\$]0$;R04rL0sUGaZicSW*fYE&'=^L!$FKA^`W)R7e96O"SN`.8aQ.] +(A^7?3n_i*5u>B(X1/00Ue[:8<[.bf2`\(#/4OF6hn^kL[^AMGnHSh8H)pI(KB0:hc,4$.TgD6$bgkDma@gE>6s^6$-=;BeMJ;1f=hIJ +?S$F.R,>o1Tq1oo8_K"W_*M5C6$_S8MOIVrormYMh_;GYZUfuSr[?q@nFpVD;-u0: +B-7%-J\8c"!9sC2m6TIPIo)j=K0CWtlONE26c-PcidY=@(AL*"n0aYh_#G'pr5\aX +]F'0)K"DuuEco'oR8XP]((_:6"f10i!<:?/YUa#,SZ0QBg65#(LY1LRD?Tq`pr@0U +M7QBn2A<,a7dK'-\qn*Zkc+*`";90mX@M8:N`7'2+SChJK6XI,Lm +r7CC` +-WciPgV,\hs7a14pY9HRk;&2"+WK>5/ZdVPk>g\X'n9TA?blL;rK?4XPh\3VJI@t] +&FV:Bc2dktF5Qi,W"8J!\.IIV$h0dgIL#WY+84*eNmF(UQRbh&3taSuf`)tepBk\H +9).i!LGfLob7sFfW4q:]$N\'CD8XkXqYI)_)LT=&fDPC%gSjr.^U`=6ERuqYj'?mL +>e]Z05`I!93]6iVO(PZrR(3%>eEWgmdf&Yt3l)h>+ARBf4NcN8n@t5l!o8[&Io@*T +[$JV=O%isAQID@%oW;Y('S1f7FpSmoP>Rij.OkIp,V4ubnc,!i@*]4Ped1b"*XR\u +87FE;N)gKt+)M?bYQK^W9YZ/o:pNCVK&5TR&UYG2+!?N>gnNVMkqqe\=qV3B&Q6RjG +TqM0Vm`,9K$\.7dHeuce>6#`>O08lHVl*&#F;u"o2>gf"C49f6J(^_]mPg=32Cdk( +cV3h86[,KU5>cKKWNNWsfl0ngodYWE&cY;X^R5%qPlHn?n=j1_P-X(Y7Fa\(boog! +@:SV#Mte.)+9'rgK4AX/*qeGM;M%qq`5.8?ciPUVKbWS[r+CYnIGBD]r5TS'_S;:k +T,B$1]9;c)HMi'fpW"b1A(?tH!.W?dTBO3Y4(s-D+'S`0!5SX6c8p?E$NsHp/m4<* +)aP)6EujBB-iQL.cR%k_Z!)[-3XW`pTDFYY2W1`]fm*_u\J?F5Uo,+&<5"<&d$5Gi +a,Pg`B#@Jb!@QlV.Cub-7RC*?IY9.-)5)Vt"_%L%1;o.VfOu=?OGcS+q\eW@)mCE* +dNJA+>mp]s*g)#/@hU%Pk4`R_]hi7=WK?`-8nq2J#fps'A9c$d,S:-qu +\Ik2p'Csu.l]-g=&%Y:+Y3@G*s0d@bh*+X(InoP$Uu_,9c*o3lDa(%ASh#\J>+1$? +s**/skT]aXg>j]-pX>jCi;[?,44j8*-d(&5ARF=YG)mHN]EaqR_ft2r4d*<>]\3W +H5:[/./\S`?2i]q"F:\K#Jouk0;:Da/+Xm0'-7)>=FQ?<^6cD7nDD@CSr.r=;AF/[ +#go&aBsC<#KtrbDPl*N'^JdT,g#^/G83%Eh?]\>a!r3OD(#Z"f-P8uRBT3!r^`NNT +Kk"TUUmQ`G:bk=%Hd)M_M?>?9.F-k6JmHUb#%60>J[NO%4Vu)bLD8POk=e);Uuo!rR'5 +%iG'B"4+0+(EUm"5NRA4^cr.X\!:H7X@NL&!%gNh+rn?"<\mP"::M0uIFG@]\%Q`f\#2>;W)n1.Gt+'ct?+T)JHe:b(#V<>+A[GT`JYjMj\)W-K:;X(g;[3RX`%=rO'?29 +m>>AkIn_@1nmeEhc.&tg*d7_2fGsu&^n2/CX#El+(Z`IIJK8"R?hg790KYh%"[GMl +!WKu,p'1:T5s]Fb*:nY.dl>e+2UWh)n:,ITEb\-kE_&mn5JIO*JeA(T^EA%h[Xc"O +.Z(k;#l[5O:m5hpj3CH5Y/('t86mcK)tqfIq"Hk$s*TTeB/O_SHF?\oe>as$%8!HM +d;+ZV!mHC.Mm:?3.WLnMh%gXED^c#qMo-$&e4kgGaDh_Bi<9Ts-N%PI>n"uc-pr%H +'<\XNP6_!j73+5CJ(R3s>"FbNIE#a"Tua!!2l!)2!U,mlfqk?,ljpOk#2@75J"VUT +4,ogIMPla+D'SjFka0.!3k.FCL/*RpAGiO!P>nk%.XnR)I`;?LUX"RKM*&>kA`3T' +?t(K;&,]dY:TXAhJH'EQR8)!9n:/nE[HX%]ribNJ+Um*e8aX +:p=pbgcQKh3>lNo*>?+(D9"[N&YBAfc@jo$.dZP& +#!D9;Xioo-Lm/opB*Q<`:G_gEqu$'GIBWn5n8GB\eQ!=Bc4BDO=c)'t1-`A2V3lct +8_jN3Q!tF@H[_XYJh!-Yr^?PE@.FP^AK_^/'\q_ue6E3pi?.Y5kC:^*C]Tq'qt#;2 +bm+UVrY&omd/#EL;FQPB#2j"MGAI9k_[?=^07!-WaX?3Z`"E"Mr^.:U5Aqso+iFRN +#r!9+%J\'$NoI/1NO[u@YP!t5Wa@gjce6GN5RB^"p,RZF^V:,kiSR7drRC>.;jTYF +Xo+0dm00]U#f$S[^Q7trIs'j= +\K7gNV%"Ia_8H&o+/*5feKA::QX>@dHW$fLg +mC.Te02dB$!.QP9!BpE;,_H'pZ0M;kn@ok4s$WI^pn&@2!Uq?mJta9Z6f!`S>n*HX +>^_2'?)^H?IsBYms5"0hnk1h>]93W=s%KdHKA^+#%+R`JFV+NQra'rJb:7Iub:Ek# +8]N`&QoYsVX/,%1,e^&hWW86c`*t9;Qq0t.fjuA>blI7mOL6?CVaMORgS90_`9u!> +QN9FJXkc\?b(PZlF)^.tDT#BR$JS23'*IA0^F;s-"Fof9VtucZ?0"m!3u,_0Gq$f^ +3pIG#o<]N^2&G#:9'f9KKM!49`U9$)E.NOq&sb4]&p@O&91T4(.5.[[7[2\e]]h4f +H8U0AIe)WEHsnbrrK"S6Bu/DE'!es'u&c0jIj;O/!lK +cC$!-eN*#\YbhNe4m&\-8s?egVi3g3s>T8eRO^1EW7-DJ)L>omh4WpH#geZEE@K%Zt#_V +kdu\KWePGo#8V3hB3*F(NEQ#j`.+^%L`7T]',TGsdEk`i!hfH%r(l-8PCGV/mrdDj +:X+O+M4R:qpM>"["b27E5b8)0r2`=)[b,V6lJRj9daDp!<4ipnQU?d1F@DUN$';K'jf`*9<]"u%Z1YU +!-SAon:D6JgQ8H>>ug7Dj1io,%bbVo2Zo%T$PFgq,J:0M=[B`0 +!>[El>QX]1]ft:Z$,?c9mk4^Q]^Ps#"k43g+S]:Sr#b;A@K)l;O?rk9,leV\K45BE +"A\TuNV9%R]S:d%#.o[X+B-auB*%K)m<*l.s0[:.]&$-Z?Ltp_aT#WK[Ae+^1/sSG +e;&Hp!EY94?L9QWm)Cf]^U9o@f,h:>>Rr +e\mJj%Y23+6c49Oa@+sY?IM[+eP$\PInr94g,70k'DN+6s+027,^eDILD""H4V+s9 +L377UE.(`1'o_(%lN[9Y45Td[_4,XsF3Dk.Ac(Poq>+CG34^ukN[@;LmlYR6A,\\! +'.oluX[hncO#t_L\"30.^!+?*k/M.Vq?)4"/8DSNfG4e8+f![9kff/l:,^4O/CG_t +KT,^;hdR'A"e>_1r)\tF*[TL7`+/fZq%>`f$QiqcP^=!!]"E+K0dPH?oBj^a/O(d%n3p^N$ZiU2B +4,qd)>t&Lhrt-tAe4'tU:q\FpG8"[4s%@A'#K"4^D>5[0nn!LZIY!79EEA8qkC7m*0@i>ddkUZHHcIiFURX[6 +#K)$N_B"YU0`OY;m,8MNcV<]1*:(ta(7)TMB_kMKeM"$%4TBF#0CU'sj3,3IB])*G +aa[&iokR4l+5Em6&1.K;ZHl9g>H'4_%J/JS,+5:+o)nKKf=i`6H3O's.=YK(Gu.@\ +:fDE@Yck*Q)5OGtW3:\Rf6GB`r#16::BKuOg(@ZDCL\uY-[iP)?oUu-S +OL-.s"XaQ/:Cca%!#,L*FT^=NJcuk0ha;5TmmYD&%3U9&9jPl\c?MK#'<+c_)cEBL +A?[NFCBurd]_5O+dqbrVbs/V8^'V\3Djo!*YG^)UD?bG6jaf&5og\:!#Z;^ +I%QphL;6>.mWLA,,pkW@V9[X)OB#7jCFutO(Rl@:c&H7T,Loah2:bXKp%a`VXsOL' +YD'do`/M3n)&)V91/:BVN604gg-qVOds#9YCdc<@VG5k$No&R`TaT&Pgs_A\n(Ops +(Mj`M^MDEd/%SB@s*-6^ZME4UlFTX0kHhHIDT]`hME:h6+8Gj^\b":ubU94SYLt9G +s%h-lpatOe=/Gk*m#r7MK9t48Q['gkIl&gc,^&+&kKTd9_&)`n4VI[YdWpDlJ"u#@ +[J!7;("]lfdp$L,rp0_?>++3p:I5A,=#UF$Y\BKXcq]08q#gO:>&HGVcY`!g+oh#b +Ac^r-s+c.o&7?K04g9@bG^[h!EM&V'HIm6Zr+dg-o8[b_VLj2*hJgkAI=:lQ:U16A3Y_,(3%G\IUK9\9unJMmgJ#>Z\jXT`lk(harX/hMmFYfFbn)#dp96MpI,(TXTMN&@fMiTD4Weptmjn +BH"68oN:Ci9Jm]>s%,3]lHfV4I8+6j,H9#k!Xldo^ZPHcT42(p&\oJ7J)N-?^S2)* +?W!#B5F/1S<%k*krlaOa-.APs\hm+=^X'^h)`[p4(DJ^t65BS-/-)mrZgcu*ZM5L= +3ugR05:GhG-ho@5nDDRlKGO,C^VG>.[M%cdpeTcTs$,I,khks'1DE.RTAc[L(Z5;k +7q3O5"[n%W+DX(Xo&=H89&<7<^hj2HsG-B6rA/1>($\O&!Br+XJe/_nCQ22daoO,jKCXYW.3LI +E(JmYMQu8cZfigoq.!>>4*;g8T'S@u4;KE69@-+cqH.a]s!Y'$`p\oqHK.=gIpCVH +c1A\#Lh77[r-QW'q,ID)k5WmC+O>Q4XNLNUm6IrXjo;LW;n]K#rGmGgR3BP=_oao. +jCa3Y80c8U6ZReY872HC>tm\sq?$\D40&/*KshM\mF)I4mmLfGHdg5&*R0E?0r.L= +Z%IRV,1%%q2MIY1ITsM+>i@G=3:oOqF4R(V'NF>7@^#l*IVr%SkZmJLVe$L1#YT`"T-QHI.OEA>#V97Ol2F_U[RWHY'$@l15V +EG\E7Z/n6g3KY;&gZBB0rP8GU_t977(>efW+"\%nV`/bM2[na4`pkYX1dlP0Fg?3' +_Z)4\V,c">VJR67s!,k,[UWpu3HHj$CH]&/i0]lfO3O[X/1<1,eQt\^U7G,)^dh/g +H57P)JH&8M/A4>K]c8qS#S$(%sQ"9h)T_97[$X&_SY/ +)rr7`R\iFAadM?DR#:0*UVi@73?#/SL&"MH!7JXQ+2)@7`;s'Apa&'TCnQR2D.<-n +Oh"9SY^qL.^ja/=9pjR)7",V,!,gh.m:BQU@?7=\cW;t8W#5rm%>TN7N@3c2B+h=b +-JQghLu6unAO:3\u +N5$Aai.S]Wo,h6aB!0DC.PkgJ8l:n9E1dmS\'dVqZp5:YYQ0QiE9=4p.K:"XbE&V* +`tX-#1>u*g^V'-IXK"f$(9eVS;>tJYBuLd=P\sJTlmel"c#RT&*ZahireIWRYZ2OG +#n-7[c[R);1]1%:pn4P`+#F$O^\Ab4&]^gVp_W]IBO_67oY=E)'?hilGq\ZZJ3W)O +n3:gl;;Q8N3tu7W7cLMs3"&?/ILd;gJ>D=H[/IbPNPBX94`:uV]Nb.#^X2lZrZcXF +TaMsMTn%Y8E_Ub4"34+-Ko.!q6FFELFlh/(sBa&c?8gJt,f- +f,6(@q"H')n-;5o!4>'n"S"]Hn/kj7h@$]_s#C=bbK90Aqu6g__=GC'2L?&@]K]VRF8S;jsf_Q=ipQ=QXGmV<9DRKKufYGmE;It_04@Fb]YC&nU!!6P># +Nb?hGmnGRQ-aL$lB'Ye2l[hp)_c=mnnn7VD17Y:gbG +gt#h=[p38t?+a_g]48C_iLF/^NScU!KqI/):YkWt=Ei!is!OULh,&N@;uOt"pqr/l +fs_Fr4uj\aJs?5(g^3oYL,1kfpXes#^Kt4s#9#r/ZLJ[2I(mqs3NpDlb7AnFD\o[ +&,]dV9`b94BJ81@?+d;NN8_N[SQrY@?ZR:HIm:=40RA()\pQ_f$`^#C]do0?Dsd8E +hoT3>@"$qa?=t"X/M\g<^I_*Wb@j?KfnpR/dGtOas($44"#]Vq`[s"Q'Vs%GhkM_g +f>/U/Q\Mm2b(,(1`8NmV:esP>M!a^S'K+hO*RBVY4r:F69'+BI,P#L7gmk^PgF_X^ +QM:8,#D:P,s,fjia_9/&(ukR".1u(Lh5h&cpg;V2i+MD(JH%h4o.g8,N-r#3Mo'sPk79:7M'.,Gf)Nlmq)hseIYRmuRid4]A^>o9[JsZPq4h0aZJTUX?q'nK@gDdqg5Vq0 +D2DD8>VA:8DL,#`qe@2L[BHUq%]el$;:iM`i?+G*C4+!"^\?1KWj)`0r'L/`pnN6I +RUQ?t<5a8*g?p$Q184db^crk"jl1Z$=h24+HJ$W'?l^E*([,sZskp)3iS +7mQ%JM&7>jg7)f1nDBdQ_=6oP[<>#.fn_$i&pFJs$*Pc:C(31,rK]LJWQ(p@Ep+ui +RS'mj\U[+Q%cr48$iK1<"ZUTDi4iJlaMKFJ@qDDJ^rh8!O!Z)S(g(HhYaIuMT+Nc_ +r!'$*9D<.6L_XY1MBu##OgG"R?$+";3p(*+Z+#s$M0gO"FVIb;h,I$%eH#;9/EOhd +5nTf"CIP5)JmtjVg?WFt"uL/=e&ET?*:O#HlnHoj&opZ"TCdf^MCB7(4WIuhfl_sdu>GFe9hPt+8Gjf\c4F+FBnV'H-Y=aN1LR$l'&:U +qd-]2s5/:]rHW,D!j1c<@i5!Uddf!ZJ'0s`j$juI^i$UI1L$a@*Rif@BbkJa4hdd7 +a8EtX!:YKm0X5C\bIEZ6+8BaUs1o*$F(Zb*:PccuV7j6RR=P3bcT/V9]sQdd>kQG! +FAL_s>eOhg>LB4ri=.o`a"uT@r!.`ph:j`gHL%c,&>UcjJ`hHrn!sD>/VsGAl.pa?ol.f5 +I@`ZrDr>@Kl)H(n\g^Y:S>C*K,Q8(m=u8j7I*Y0aGc1 +l7h=C=n6lT"U+`rLV;do^5>>RU[[0\6f\K[22t+fcF5!gCM-1Zl(_$f_'pX1;+,a6,ZWP(oWp]pA" +iJjKpb("rV]npWV5DTiP,QBt,&,B@\!S0f'Fej_4,b/r.cQ9;h47m6hdVo'.^Mm6R +e6;DV"8<>u`$<(fk"kp-?d+"*mN8IOJ#]^PKiRc9;jH_!AN1-%R/^?8\ltW$j<[H6 +%gX]ZnOgQQ_d?Z_:KkJ>;ZJ?&;C%fbaD(W\Vb%"LQ7 +\QXk"_S;N,Yqu;1gIR!j0peTgtT8\J.h&WKZ +$s0nCh^M +]bFV"8]lXGU.N<.):W$sX<7#j#Q.(:\>`4\Rt!-%VlI:_e=;0FS:`jtZ9CK?QVZ$F +Aa)gnop%@D28`P>0\%n[ok`R$Vh*uJ9_\'4K=/ppd\[rm?(8FikE^g=fDSG(STW>u +;pk6/I!/"7[4]#'UKr+]>^iigpJc/P>QD%V\?_lZ!;p3d:I>'&nGFFLmg3%p1;utaJ7GpCl+7$DS +&b%HM$]tBq..?UQ(Ppn:O3l/omIL50CV]rjochlhIt?`EX?W!-H_t0(ZEJ.L*c>X- +Su"V&7kMQ*!CFr:J,!ADSDOR7&/ +a4U,9.%mLB&Bon?8R1bQM8MQNA];U-MSgMk/HPn,3<)&lS8K8^L]8Mn)=W.*?Q"Y! +ad,C%s1n[)2El_BHh4d+U^Ib[liQ?SlELp4?I1MMD?^LD+oe5=!9T4mdZ5qBg_T4g2Tl5\fSVh71:p'T\Gk_f\R1T?Ps#YjYKF59l8=fbX:2aIN41EJRN]:2/ +:0*j:M-U*4TZrGIJ!lrfW)sKh*KHXB-b%-%f^CI8%bH`J2&tl=*UW!]kS'NR(S;U2 +R3.[o_m6VDMQt(AP9]WsUC,B4@JP`\ajFGRBNOe%Hc&*Q9lE[M=0SJ^<8&!W:LT& +rk%65i4hu[YmJc0EV-ah?Lt9d;ad?m8`45l#kC@R#:]5mA_^6&-UjVr.aLgjE77e^ +n2bk+3/%Ab"B-m01daD4l;LJ!$PhTZl-AF5:GXe)f5OI;\SaD,g3WXmiU4:!@T>Vb +=EeC\IdN!UbcYPu:I"SI4nhWqB1kD$Xdqa68.A4Qeb@\jZO\Q>D`(eUCqP!;Vu(/= +pED&2/fbY;8aF'VHZZ%Fl=/G#/(DE$Peh)!cjZcN2J;4X\,U]Mqc(.:j=>B-YCoDE +9rF.+=#,1_.?WADhXucrkPWG-`P&bCQ.P/R020RQ5TI],pU^$)CGoMa*cQ*QhOH0:\Cb-`oT_%`@'sH*R4OQWqgcc'D)gmRb(j`*kX5bs5(Kd +Qc+MaIhd;)=5isVhL""nNB,V(*hgV\?`n/eko.[T7^W2#h*aPg+@2l"8A`gi9`h@'qGCWUOlbACdM)H,2n/.p>b5SpMMkAUS2q?<0/ca0F[/Y5s:bM',9#K +fN>q:6L9LLWSPAT0X"e*b^RnpR6O#uWe>,t.f +'gqiG+\gpG,-P(jl&k%K$CP`n>"`@K\:GQsVu#^-?6]D2.^SL'M#WbrZF6-Fm'e*+ +bY8l7-A)I-o29gYah33>eGMVn:M[+(Bs((if;A-)dj7_V1*.>.Y7bD^V$t&-@8JY; +.L(:\X;[s1bHgHd+GW/$C#HL[\H-2H!*t`(->n`%%g3XZ1$?$7$uRM%m7n5\nI^K4iL_tESu1C"ha=(A,*rKEAeHm2)I+^Nee)< +57l';ciS6L"T\Hf>Tj'/8D_b&I(0Z@EdE,d4kB7K6P4nPa5rT9s+@!7L!S +02UgUphTL#5[3`5ZC4#,2LfV0lmMS#jjZcqY8'@u:Qj;r#QBW2d6%X:s2kH7Wp`31 +l?tpEO?+`d,NcbP.65P##mp4\W6MagBFY^#D56uu9]/.N58)'9#Pl]&nt1Fs&t8ju +!WP5(\(KBLs'ALK5bKu +q.0-9:_O`ADlCn`rhC,q/)F(B!rc[;:_mT_o?mX63o +5*O8qWiMFpY:&!B10on3:i4m!]m<.1Y[s2HBj>n_\G\r^]rA]O.\D)4^@4f[Z)>@O +Xa@9$SUE&[g/?aUWkGaXaU/E@:`Y^msF0M9?l0(h=nKgr;kOoQCZ?'&`G^T!5l]WSq,a9?J)7d9gXaWOd+T5pWGnULl7%-k-L`)f*/6dF +_l%7'=aC[KKnUkJ#d^X")XuW-J_qo1=Moi"0t9=Ts3d$/e@9lu,Y\4$I>n)Q4Y70D +"TBVR5Z>Di/51L4pjrH^L7C/3*<)IG92;ZB(\lnV.NJ'p&,Ssh!p)Z-s&`J-[X\`+ +bniG9+Q,I"G(C0`J4E]Amp)eOb(!dYa8B!RoZOe\s)'P1o=q[>=[8MWY%gS!s5m5- +n)HXI0GSg2rrN,"`AF]]L+GS]LMr\OblA,BKR\s\;%!Yu\C^k5rrE'7@t'>rQj'1D +NI@"3E:_,n8,RQEr'u0aK'.*\`./B?hcV.gnNKjQ[K$kJ>;(9[[NhW.XhFSPl4!WqFA,[ +J"eIh#tSG"a3+8)oXqB/Z3,+G>EpId/6sJm<\tk^eF!GOGLHh^^OH8JGlgSk'r($]- +,TH%orjV-[./Y#En4/<7^>ct:j[`&#BR:uds7'D=Z@G4I."VE@nK[=?7lB=^[:`"Z +Vh^N*97W&6r^\>$IK]]=:jI8hZ7CHKqmQe9q5`B@q6=!FptMi'r#tuJr]>76G"nB! +?#!B*@LFk=MQm=%'nlj;JO.F983%gN^Ia;d5\LIk&E^`[i'5$7V$?]b_`uHHR/[/9 +]TFkes%Hl'%,"`Y!d*b/8@lqPi4k$!-B1upM5pVf5s"KWQN/.B:9l]8"b'G3Vh##% +=_Q!boT^+h("li_AZg_\dmWg\D_RK8%FDOO\jhgV_u8,Tj"s#E+CDd%$,.]/\d"^T8iDJ+rZng(b9EYniL^/K-^$[tn(@01(d<(9 +6lQ\]!jU_o+*\CD!!sM+=STsR9`C-)6&67;g``T?`/+jZr6:W,\H%+!rWethK:]qg +?h"C6rXXc6!WFHMC(g?'/V)bDd\V,9U(WID6A/-:!:P:Pjo;[dVUOCVPO?$Bp?Elg +M@j@_OM5s9R1K4Nn;mD&cDO1$l#H_"C8c@`!;#)9I&,0PItI]0D@5Mj54> +D'*(q3j1<(=.R@39*>/$)bCY`=XhS7r7=KFFYZ#=J!9_!q!U+\rMD[gL4:EI!WEZ0 +;=`<=r&N00s0WP@uH^@6r.l/MZsg&.85ku2rHB=@Y4 +X`Ziaf7Q9921KA$ln`\$/'W5>!SGs*^[fQ^%DfJj4S.u&DW%pjA +$@mW5q4fO$nm>C+;&4f.MB\7s&BcKFq[!0(s03.8m?mpR./^asVH4FbJ6tb>&JdXu +lctO^ho:<b>t]apXf$d$?R*/%mobc[Ij1K;'\n!a\c"oW]>:+c +,eUgc=l*#ZXYo!AD45l&SUW3*Z_j*rPMVb/^,02;I^A"a1Z?8?@GaVQnN87>J_YQ8 +X7ZgrF,Di&0CRs1pZ,'BO_`I_$$-&6A(RI4gt8XkS)5Ie(dip"$MG9"Xq#Zi!G'!;m(;J2[K+T3"h$+Bh%F +<@n=EpI'4LRV1JJk2CcqEUoZLAP[`(:GB?)^N$/tm#bsr>;hE5KmL5bfJ]=ZB&RSg +o".]j$P-cHo=Sk.ZJ3^$GFhF)f:$W[2U9,<*NG^?QAt>R4+H\Oq2<&>YCfX-[6ApM +\pm#t'l9m*2kbq2D)4W89SMQ2jY8W?\lM'PLpE!Q1>;+?,25N="T4Z_.0DMV'&emt +pf_uq!;Jg.$4(M\r3^Vdoa/pm&f!=Q_aYuO`PfXSD`?j2i\pl;:(*uCRLA0]iU@BU ++7^0Js$m0fUS4t]rWE)E7Ul:V1n8K&!aTV\\s-S'QeO_S#Pc@A_MJ5%JO\St.8Tn6 +BC.EXeRT(=7g##cR/E_`r(&47$iEIr0rMI%uC8]7/oJ)QN)g(c[5Wnr"JiY +DU+CO.11('7RDn.O*4m2RK!Qmc1qA>&CLDi0bltK%ib'^r$2!;F5OlBb)X@]_S<^S +#)+jZEe+5)%C7'O?\:8Bg8i>_s,#M01&R.>q#A\BL0O4's$=.,oMPTPgm3F8`dc?+ +KK0mHo`Za,*`AmlXj*$YU>I'@IcjXY'i9;_*MY_t*Qo$P^#\iT7J2F\2iMrp6YH6K^W^!.o:-t"i +8P)BSVEmR,lfeu[>o&aH`q$ao5u176(Vq3b+2bk+0hL:O82![WK6ZN=H=$/I_9"r$30'oPlDlA53@@M!oOMXH3C_]6u;Ft,UPWT +[Kk/H?BFshntS^g_QInfJ%l^)_]Qr1MM;'C+FbC1M>RVR'?VWZO8+d)5B8+5^J,;e ++@XBPr];:p,PX&L#6HXZ,7BSXH)+EK385U$c[Sk\VQ#SdW$]22!PecZdc6N(M&$Ye +!3'rG_#/uVOfT["=e#tN!>]&m-&/I%B'A0FW^_&CM0VB496F:.Z?W`GG?]=QZ"2j^ +i:[i#bZaV_#PGQ,q(kfWpE#1[_YnN2oHW>.a7?j3W]P!ZRkl?^h;GG]#V%?Fh1\PDs0"R]h`3U1$auLB +]i6ZORNAO[Dr9_`?Z*Nb_4TMJc>O#%2gDej`NqXpq_(XDn*F8nEUCG\n%AbW\D3"[ +5G2Loo&Hk_[K$lM+aQ=3nAg&$/63g+,0aWF.qa&reAnY +\l5c#'99)ff()[:acO5U5at!:iq=l^]eXc$s/f!pUAt6*s&&C.#bTNJro;*2ms"aE +bJ*nS%"LRW-6;\beQdnDC+c0Zd3$Oqa2>OJ6I(]\rk#J?%iF+C3`TUi/lFdPW?h)O'e<hAhl_#k\ +n:*^VGYl:Rd>Q@]?Y0JpUn_'$$mPVU]lnS7SO:WRcYX'N@mp[u\%6WH/M9eV,6nMa +KmqqWJWQK8rok"8/^Aj1s'6`9s)=mo8='l=i+OdL!6kJl_fA$@fb$(qJ,HO:i4i^u +YT@+23?X*d!WW&OlXTjr,)/4EUEL/(r+.$!;UA8NnFp).]]Lu$r8iHWh"JeX\#G;Q +as*=Ld&'(ha"kd.A)^[R3/IJfG)Z?8/->f%JA5=1F;+THWIKU$IKeO4('#Y2K]3\D +n,tF%!Osl5dZA%?c&,^*?3n>!TSckPHu\-]!.lC5^.6V!1dPDEP--@\0GL:i"*Jmd +\rm135TEFZM'"%M<7/VF($"ti7K.!DiFCirRc)n9e[>5LCFf=P/'U!gW\3,2\.73e +Z09JfChcspAYNT-DTSk*r**Crl0Cb?4irF5P(,SEHni831::nT4`[aaeD9,h>Ihd- +BD('(l9XUMj:C-!*#M;SFFGh1rCT+u5]Sd3I.Es+@S5Q^HJ8jFE4tn\G4Vgfrm[NK +iK`iiCVr3Ks.sH:=M6;Sg#Ppnj%XGN9g>boYu,o=BuRL9jr(CU-Ol:*eQ:`#&HhfHQ;B,1Z<]q>am!\'bG/Hl$&%(+U<7,/&/+9T#CiVO7Q3@A9]06G,gThY[g+FlZ^i1J)?J3W6gW#YS^4Br3j +5LWe'n@oHTY.YnJs2'O,JG;q<#k5S7r,6ERpl>Y!`rZPGVqn`\s'rKp,F1+'Im_&& +/e@_ATZdT=GDR1X!_*7kWP=Mo9&ee_ps;V?d,YF?@@A=QmD"5Ur!3!)C,5khJqa\a +^P)WYs,d8'nnc!k#5gDq(dD7,p*K2mBL3%FO[@!l0+S#DquQeqm4ZV=\,?@,&H2_$ +[OOn>s$YT`2h/o9s,#KZim7PBG5b!sBE:K9s-ET!Qj"&D"V$0r,H^e1_gc;Bk\s&. +86>4u-YKZSh&g)![a9@&e4sp_]:7=4aoFjMUfMh3M_.).bm9\7rPfPd+j6e0o]etb +>o)qeW_CJ2f*Dkdq1hq^K,WSi/O)f78EZfdpo-_(P+al>">53V5hg5`.KX_p+X6ho +pfiss#ldpMP6h1K5"?+!s'4JHR^>`oC4WT:2sAC-"s%bO2HC*S"bm_rYsCYM80[s@ +oN#fe#[F%?LsCM:FQF27s6ARo#2oMp#9R.MUjB6F-inkfVh^M(s5sC]M#Rb["TPm- +"r6X!iU1<0s"se+,5bMMp_0(1/7:.$gIcnF&qIa;<9cB-\.*"%I48PGjNs3E!<*Q) +$2>IPn3%tUbYA6Jck^8m8hsCX=fR>f +F[`=-1/1\5%#uHj.'f+LGF(1WWsR#"+'TLc?B\]bWk'laEH\>Gr&Xl#@cdAOl1rl@ +aUS7A.duM?dtO9:"7m6TT8=qf.;iEC*:-EUT?6Sn,m?%'B/k\09?62BMTdf)lk +HBNJJIK,"An[7MqY3bRdhUJ=?8I5-RrrN*\%9%VdkoISKpr/GoM.]PH0>t29I9:Rd +M/_aBi7;Mrb^@C;4l+8H@W"gorPl/8b4FPfHYH,5o0P2+CZG9"IViB)\0ef]Yjp:S +hXk.G_`*J7kDD?jZFR3q+6Ym6o]UZ9:f,^n9P7I)g[MrZM@o#Vn0>YQ_rarqH]UO80kB +J,Et*G2*-C8-o33egU>UD)kn?8TUP2#8_N3>^;u.GQ*]5E5Z+23*:JP"T0o7(+qfh +XuOs0H+LBj/-%LBW?#oX6]*^*+=mN)^nirbMigKUL"6G2s-C[p;#OOHr>AtuBhuuF +=4M:\`DKjm@8"^NA'qM5Io(dUs$+a-FPQm>plC;52#OHenNRWbJnbo!IhKO=nXl-s +'p.N$W_u1&"'FR\#@[TZQ2k59(\GEK0]n&tTRn9eIm3s`Y7qF:F:23DppM::GV>Lg +JRfp7XX85g"7<.q0`RgR0\91kBeP&pOT])3Lp\ZDsO82#i +aX+k+%CEJ==rH.im*5U-e">ITUOq45fkhdFX,@t'B*GB4OI7"@P!/Y8O.Ml_J%[gE +-VDeoPP)d,/>E3e7**GF2Xln%c%gC8#IO^7OTdQ_JkA>H98;qSP%RotBS`09f)5'l +3=5WH[eRD8E:n6+r;DWqT:Lt]*Es@_dGs/eps[DQf`o?$:]#S<+)7dqr'qOL,m57U +&KgJ*`qZ_W5X!Em+7UFU5`G_D)92g;+,Ph+!0*^FJ3Qg.ILRg+ +@:BBQN$,',"QX[>N96C%4^s:Q7!8kb:-P(*,o3$RQ"U2#:^^sZdot/+=dFY=Sh@2haX^8s2&V/>FAFWh]9IfVb!!icSJR>_ +2iM#m*F3=sIk&)+TPAkZA8Jr.i_f??(qj7aQgK.)jk72?"J2MA[QV+lb[!9gO@++HpA/=5IO@7 +N?#Ho)4mPr//95\C:q!oS(Nk[I]Zn#@pI_d0>(-+^X4Zd$Tsd%4lt#&P0Ui#hq;e6 +q-GuZ>&s@ko^4Noo],'?E.QVuq-EM2]RI+7R9=I2r7YP^A5EM0$8\>`)H@$[XS4XFnL`:iJ$U76`8t% +-1@3]V9bO_VBlIH=`.BJAkbH$,br"J%]TK)tFLus67YS70"+.`+-PU6P4(T.Rq#C_7( +)#hB5"Fji)#p^_iY7OXho,mZ*ksj4D,lm)Ba,g[d>CbUm%gqOkcn&!>f.5$qb0DS* +n9=Dj:*'M>dP`i,l>7OVT36\UY,5<9W9CUTC:MRk(I5"E`7RkmJ4-+066GcJ,Z9F5 +Fb(;rILQ8(8AStdiFmRe>epBuXutYB?q9WM4/]hFV_>pTBjl1?)2)*t+_GAKdI(L4 +4LA.52Q4m:$s@:.A-$5rs"V\qNRD@A!:0ak;,h.I$HOD`"(;Rn])XSf63)M;+;I&` +W2-(""E\aT!Nl8dq"FjHJc@>2s#dfC^^Djo]f$Xlcb^)m=J2<).;WT.\V1;F-K[hr +7o'.I3-b!uIgGRo.+0s-ei_C+)Z^b*.\L!fXB0G:?Zp0\Fa?Rf!qLu`7WfRIn-](B +fEC[cnpbjV2;[iO6Ho-33/`EL;Fpd.F-53VkV`%./^dm^1^DrWL%%n-+5d[8/!uttraY^Je8(Ep/IN9%!;*%L!=C=?huZ)VG16\K +<0rF+&5\545U[aJ,Loah-?b'*@Z/+f\-g"e!;qo(:.K`^!do/43^H,6!\TQY.nZ*B +V4nX1bL4?ULZF^Lo!0N?:m@'C"jCpWh-d5H6;u3alMc\frC_`8lhh]C-"iPp/o9$C +>Aq"]Y^K>jnP6pjL!!??Q+*?#VU'=p]+8%?BDPDdI,i^W-in4,8!g80ba+j6ar7u^ +l+K[SSRhBLaQYn3$MC25JH+H_8.H`iSL)sMmamJqg*^Qr>M7i5V;kA;D`ha$gl]k` +s*#3a0=5E\"MfL_Z+-o@I&X^W7X>$!kh\tKls&/u>Lh`)r8+mKp826W$%E$EG89jY +l'?L8MiVDXf*.N()oahe1L$m4=8M`0rTou%l@J*[6M%a)g%;I"CV,=Vc]-,/j`g.E +T>+`WG#p[#>(6L>o^9Fo`!P),51kh%>ZODYi_XYTO%QP8cKV4eZdJs!S%#P\i!%ia1FOGVY"@CKE]+uW2S +"cS!"2pm_Gkhp9eigrr9s3W&id:-7c$38eh'1-'#)#WFlO61q)?G?k^cr2Xb.g#^2 +#"lZ40`U/K!;o(9=q)6d3a@O;+1$L/MimiQq8e=I'"NJD8hJi.';C4Q=u%j_1\ +A+n?Gjs#?U@:R-05&Wu]4.sk:PiiIrfR4AJ_1,VngAa`]C!9?5^V),G;O.@Q2\r=N +r]ckO:m`FY:)0OTTBV13($P^J;/I)a+2U6Kp'"$A'WEKR-:ScDP#=.-Cb6R@Q&YTX +d5$:QfE1(1GO%73T90NPlF:^@a/e?KQ+6mE[j23LPp;7)hqgc-o,&%m!St\CI]B@S +%9/!"@gsaCOS:<5ViE/.nE]kJYE/>TjHoZ>k9f=f>9R%S&89;j%7ftT,AloF5MUfH +dL#_6J$:/9TUcJUDp.O+E4T9+?""LRHWtQ;>B5)-S??6p]8mj"cp=c?r-.E;s"0*b +YQ1U'9X6LM1t=!L0hG9J/0IfY=dU,2!^M(9!('ka.8h< +2^JE4^S;(m&'5pOmkj)-r#N?MLT]"'.7!CS'KOZs5^p>>&;3o!+Tsa4./^V1U#^,j +`;.`tZH3u;"jgN^L5T.:6pk/;Bg9E8U/E_M!_U%"T'_eSH_aWR*/lJ"3"lW>!fJ*, +,:l)e6det)pB?*Oi$pumPRbQ#An1UG24#Qr99Q*)8Z4m&+Gl"474"q-P'r5Y6W.lS +dSs@pMPbu9.UdQ/WfK'K0+3*H.sgV@?!?:5gb6:J;L*Dup?Gi+V6uaBX4::<.kWF* +?@F)ffMp\>Pi"?[DEFHP#6(6m[/p<*E^qh`0/<_ads>qCpY;Qe?Y4GN@/)f^1r6Bj +*FB1D%VRKA]LNLsNu]o\L&B7:]O*A$h/qgOrUc->mORCI8EkJi3@Fq%eG#TYkP$eD +d%Z&*#tb1grbiMZYLR"fZ]Nf]M^CC%\)>M=(2lkWpP]WSW:)lH9Iu%(4OW*@<6W&* +4kqbEh,d?FnBTgj**qT3H=pR>1YpqH/8Qf6=,!!m7@0SR-[)>-cP6sK,tj;q&"b@R +GBLB$\lrH`Ps=O/@^5N#-X/f"kPOt;C]@ht[tBF=6CLlW,1S-bYS%^Z'eq9Q#4ULn +VZQm(_ukEIs!K^SJH);[s3Xnui.&ZL7XuuJOG/\C&a7-(s*M]qaH#j6.hVc:s%Y3/ +J"Mo4r.k1lkPpZ`ajHF&'-X/"s24;JY-hlSV)iJp1$mNMlZf@S4BMdV5O5@2mAW)0 +.Q_>t4(0X&Agrc2hiKpqb8Jg8LD1`dJ%q5u.q\dBs6c=,B2t8"P%6";L\u'=8_''$ +`BM):^jepiR]n>;:nB#8bRi?9!rR'5(H#brpfV'T(;N/WIjmD#IjJ*mb"g1\M'W.` +hk`\A02quK4HC8K$\/_@;/fbqN_@S#pu)[l2Y'`'o:L'R`uITfRL:jq&:. +=1AM7k5tZFr2_e8^"#\mYqputlo>%4RLHD#X"k]#!#pJlQ4fNaRY%R?ZS:&X82:O9 +E)R%<'0qNWKgWFSR;(-C'4gHU)CW&bbJE-Zn_]N8=1AKDpj6g.n-CSSRH6<$)9=uT +$fVWpr3r%qXTg*b!ImGcLk1;@"1,8+"R,Onp@uh("R$AK"Z-^r2@dc"J<=a.iEkcr +Ieg5L`UQ5O\,\F=q"8m(I'cC#d*^Lp>)PI,BX#7X`,@0u=c;;/\*LPID8&r4mnb5t +iNE7,kCI(K58U6r7+hS;16,Y3CR9)JZ#)63!Im/!6RAX]JH%>BTLXH-T0HogR/be? +!"L/01TY$$'a,<1=4"_dn;=G?63@U7$F:$TaL2]pU%'[IliPn@%!?iMZ8=pGP--2V +abqsj!hb@hj;/GD/D8&sg5d3s03sf4iYSI!M@U#Np;oo;8W&[!V9+=H6dj[&8SO&3 +O4M](S?e-EDsj$034i!OFhdk*C@/r/j0d./Wt-`,Y?"t8WP#c>GN4O-k*^\`8niKV +Rf6EhlV(@;,,=0R]lTW(qQT,A>fF7sXJ(R%DTMjeEsHW&]8t@POb4#!J*U1KP9eoI +/B?_%L(h]&o,6MTSC+nFZVMNPen]/?$gt]+0't)'0`5;BQ?SROn#VT,Rb$A0QX;6Y +h^\&DY??/R;-u*anc"XR9qm!.>Bj_(J<$`j00_6DDX$V+)X"PGCKm784n\Ymj22&N +XddQ_VeUe_kJ"DX^9p/[2*KU?/8iUJUF.)CBN+,tHKgiFl%sNPWH*%&d/,^FqTY/H +]u,?*1&+91h4IEMIW6t/om-;@VlfjmbUID_'YcKD.EZagZ>C5/?&J3f9hpCL2u8M$ +dAOo'5=.D35gKoYE^Cl1Ir#.IG2N`%n07^8=P0#=Ym'#5q%Hd053@?-%/Yl*pl?ua +L\s@X/-5aB$C)4'^]OK;r8Th.BKTN'3ZkLg(TJA]<1NV+SbA?Mr(f=G:Y:SdgPV"p +$BCMF7uc;$II=XCaSQW&CTIl)@"@3Z?!+3Oc.?^hG^+.Wn: +a8u&t,]'B7Kgp7T(q+%fV%GlIASS"Z+D')"OK-FBk(87#X3rWN6]R/]]# +:1o*:o"U*FaFCZB1Wk2i3`nZ&"ldQq=U6[b&62!F"_'gZ(T/GOb=P_lEH]!djtP.J +#jGB5FRb^UIj4f#r6Krg`;/jW1#5)I(jprDSqI1g?H$A>O,#82OZ6!s-Ln] +/XKgOBt3h8Fmlm&cB[t'N[,M's"fl^nGauEH^a0u/6&7R_&^4Rh)!1:T?S;ZrZBF/ +Kau0jh]i%Ab:oKa300oh:8GDFU]o,.60U?C(RpDJ]AgFU'r^9tJU^aDs+P(LV1/5e +kEE@jFIq*UD#UPOl(&=,JYaCSgf*n4](?b?^n-#6Q"_-2^LLF'1,!M!k',X^YC+[i +N12oo(#]+tIVN23Bo[[XPrJ\C^[T[Unm&Nl9XXFb^'qg7P#(PF-XN2gGBDFpUrV($ +JR+o;hjc=s(MEJlJ)[f*,G..VM#V:#6lY,M!ImksP_0\Ys%t)Z*u'R>pT[mRN4:'4-@0E]kkW0O5+Oog +J$1@57'qL8'D[,;h0!`#9l)%iU)t;!#5a`H\VMi.(P? +?eHl(kp^988Oj*&s-U2*i.&\4#Oih_EIe(AC=\1rG=\;]n72CjeWO4gR?#E`b^X\Z +3`lC=c_^^AQ_mBu;C1G(U@M8LAZDCa?!*^O:4PRor1KhZa>j)JAJDNr$@e.DrHnA/ +e?W@&+MSg8&;-;3UcETQb9g]P"\]l-JWA/rbf6np>F21ec.(:*p!mS^k\__Pi?$($ +"b.4t5-]_a%f\rUH6JFNE=I+-K;'-3:ld.-1t\+TD[7NW6h6:ns$ZX+[^&2)O^&1l +EW10,R?7Hb!l?k,ha6uQZWqg>6VG;o\#W)&a9\$pn`+1HC+B_ZH +&H8d7s.)W6_o.B$TCifrn^[*hqAd%B*Q0N0l36s@TVr#2R"=H0'0;9tL&\BC!c$,H +>7Hh$R?#jS*S_-K;4"K!-W&>0Wp:&E%hM`+IiG\Di@`MkNs%#lBYX+/T7i:7M?8#7 +X9%5iDhIQ:)6/Sh8m=D(+]f8/J$1.?q!U%Y?+O:QRU]:X+=VAK$Im'=aNo-mI0nXZ +Ag'b'M`>=5@?iUM.-EAlTeAUL5*eX'AX/HFFr<%!91Ham?$o8E?^2Y]d.,a(%0)tQ +3j9F8Lt;fd9gSFH^//VH\-fJ1aS:.%C]R?9kc.3N!iX8g$Eq+,":S)e5f'IJEnm@-*Rf`T(0J1&FX_q@0T3IY+]o+NF.so!#H_sfIO$3*_RUpfuG\pc+p0mp1(\37r +:_YRBhn;2PmI$+_ +*tNb-h"AY"QLA_ep763l4)-e&X#U`RV$<`+\!.(4&H=!e(;Or(>C`Fo!0R;4s4@BV +bT%i#kCPDc'8?"l2/3/s%KFY5ptpf:aG?toFud*=+-I]Ri"%"Qr^H`T9>H+Q,Ji@Q +JMPSlIjld`"K-mKB0p03B>`ikioNblIqlo!c"T)nip +JJ"Q-+>rA&?j@O".`5!u!P!Y[MRl%iqqqW#1Eif^"9S?cpp]Ln;><*/D?!/jaY0Ca +T;[8j&+e]Gi&GRiLF!2IKNnTo%L0LK:\X,l52tPAJ;tDndR#uE+\GR(%dC8a6">)j +b\2RrLpNbl;qgWBlj;8:L?bE7k_h9L#R_1PFUA0B,Mc6gqh2mq$6XSATY7nF[/o2] +K`;Yjs$:9@&cVAEqucj;)BaOr6!`!L@3`5(JGCa[Lfi%<]GGn*;bSu=kU$nd,c60p +@LA^WN/.755dDpl^qNs?M=5:Z&t\3FBDa+I58`F18<7BV@YJrqPRHo +2sD?\ZFg8b95qW_bbF]ZBHT`$cLd[OnfLVMT-]82E[,XjOZR.BhQNt,o";06P:oo, +]?1@32(LcR`BEVd1Y'o0`B63-75FU4qs^/-31*0"^MCF\9RC\VH$e/QpsT*0NFRCg +^9[',h=E;eKn03(0.^e(gQ(Od\p8>LE+;IYEN/NGNc=EoH3G=+WO_OuF_4Q.qT'lU +Wr^Ip4=gG=ch%256HrkEc2HO;Vu9Ci^\i9h^\X:`^A_bT?iTu`kIZL6I6@"WV2d3@ +K3qmUIXT68c%oeXQRXR4c14E)^jdO=d&[Qm?M\jrg$`-1n,#=W!Gp\DWs->Sc,tn3 +iQ(\pAi=tD(RY=4X2XW3r`e'Cs7)[Fk'pHAnos9WCEidKOUAo+ip`$#s)`kPr5\:k +'.?!_-pHZE3orX/pn)R.r!SP>$BA9\O-pU[+od/ui;=,`+',V$jK*e:'Ch]]h;4YB +Br$.+s05Y!'#6h9J@5M^J>t_3;4\s6PbeQ,![5g`V2YQ`t5+O:"3CZpP2[t*Hs_#M;8A_7&R\j!=uliNRSW9;XT:#nMK +rrE)%#`H2^NM%fGl3O[Vr(Q$5^q\Ef%K$ah-pBEj8cZ'qQQ?J7\08J+8DjPE9r9`/ +!al%PH(%Hn8dOi@ck3X +c"rl+AN/')7]+)=FQo4g(_f(7hl!.P2;(8IKb0YK-LforVta!>+Y0nLT%E*Smdufh +#HPWATEOImW<]X@(;Ll&k7]HjL2qg+!+(6.61CJK:i3<*;Zu&Di$aEQhl5_ieSM9! +c&>)TA#0TtKF'DlR/h_hJ;,-:Co54K'4dd;0-U`ae4H$a%WrDXdI)ZY<_4XZ/Bn/t +A0b"L`Z@4_"&F.=;>qdq&3C!ArXd"t#/&pndP+iFAqCE//\iZ!1EqWE&1R^"$OD=k +HsBYdREG(O-ikTH"e\S37B+(P0djRn2m"_4S*H.iC5Td)chjl5Rt>rPq&'VFan6rO +:$'L0nq6l2K;,A5+Bd5\;e8FsKJ?_:H)]4KgXs9n4EeL-ibI<2a,O.YL"FZu8_`%# +Ic<>bV<+?gpj\FVIZ;ear.k3"\*,+"s(,C.9[;QA]DH-9X"W;iq3op,/F-#rNK>t5 +\WI9$cJaD9A#YL#'l=UQ\rNsBg/!1Sojtt:kkGP9m$;bPQ%!4=]C]?VpT\Q6W>nW` +fTja3Y?M&,^X1edBmK3jp(4Mm;usUcV,rtJ\":Vn?0a0"O8&P_?aMqA*SQ0*HuE6G +1!ZqkVB5"L(Rm$1'TYIqSD'6(mHQne< +b5\`Nmm#(rr'0!4Z[Y(,dslN9B7Dr@o685>h&cN2XnV#MD*Pf's8&JUlXQgjrL/t, +4OYgSfHjLTP\-2U;b5gbkuE!5CSbS;#l*Akir-u^,d7C!+O:; +#Q4%I_*=(^^!6]+n)Jm;rWN8E:'F6bJ!g7er*uEud$[/hBn$`1s0T!!PdfSX]C;&c +i2!+ZplF?Obs-)3(6HUe,R&Nk`p=?pHA&XIo&#((5h0"%jM.uEUfQ +#,9kUX3K/FO>PK*+Q?L'dX1ajOE;4JI.U=,s0s:a`"l0:6/A4;-!GgRVYH&6 +&th=:cKa[)f6q<5+t*[]n4IV7'<%h*&]*@3L%bFhMF4O5#lf@#^>GGV4:)YFs&&bg +)$Ks_gGfC>g9-42KcQ`P&F!blm)/O*/e+9D;sC)!Ir9YtnX-j*ulVh7YU,[<8/ +q5s3]O96T`Uj1epM=Vfu5?In26e!)&\m^F2&76PL.g6i78BhJi=E+^fLaZq1>`A]m +BSCJ,?'tj3+=kX"&f-!Ud1r>f5g&0WEme_2IoTABZ`<"leoCHSE`2_'N4Rf/`hN+: +U6N[W,[T.f62>`!MM"`OMlOM"e/o*!VY:i\Nonl`/$r)R+'tt^oW[c3;rRCdq9:MI +eBo.Fq&/,j2.k_sGN:fuI5KX4Q(i8H +fb?Fl!regb>;#$FNA91KP6,C']f1ihh>?F4:O_="kqA!fI!Md.];M+S;(V*KJJcBZT.rQE.0l^Rb;?IrkG75F3?VQ1i;!mJb5Z?\ +h?Wt&(A'TW6+@(4)ubTf?[O,$I*>ZDacD/O+7K,X0`96$97[&?I"eI)Sb1tFs,dK' +nim0T4IlH3rI<;uP-Y@gN$%/2TUlMgLCsU^O1'QOr/^8VY"2JkVC="DA2WW"RWM,< +R\V)og.]:hO;V9lV'6]u_E[N3_dB==l!jQLWi4G@^-7ANZ$_TVXVi'YBYN?19.d5$43P/j%8D@!eDWr>.*fi3j.%\,BT#nd"rZolN#A6;+d# +69@`'<1Ad3:f1Jt/7QnMQ;:!OH*u_C4M#@4*c6ZB>1kKf]SNJih,iCo05D)_G6BG. +_U[Ho27O3J.i+L=>fU8_9JR3@;-:fZEB.PI ++K%?XV9!U+8KLm/=.KNcBukPC$_I;0a$-*SM^S?60#38N?GT>ZZr9L]O3WQ;NJY+5 +ZAd%rqSl3GI>5$S*@.g\'*H$#P""K3_Qjl)8Q'WW_5;eueGREa!nHET"P"2-,'9Kd +k[7]dJbkO^9r/)`b74m%P=/EmK%_J=_Z;R^>BD(nE9>Ek?F<]!1@@iP;SNJP3A1_o_&d +rBgONFjA>!>7]tY;&s<&_%Teo8=KK\I`G$[;Z4]j9`OaGb^]Nq\kdc5I'Daikfdhe +SX7KGgZ#gujGP[eO*8Ib40n\FMEBSXl'KCO:D>are5L:e$SP6KSnk31>4;F#J*en@ +O5@BJIhR(t_RK3T]DX(\\)Z7kl[Q5'>^U"j\S[qNhfNn9D"'a^d$?T\g*?PlB1i4' +54u[/I.?rH?i8a"Q[ec)MuWM7=23Mbr,:.Ms*pAbombm!ec3&HbPuVGci0885Q,e- +s5[OWqpTbe5M\?tbOs^fk9F>.qL1;>qf5fRY?dR]Q"H^9?jYR;ANOERV]>+$!hm/# +`'K((kJ)h1%k[Okn(Pth[9E.Z)R==6jr>Ws#7EaCfj\U3(%)\3/^1.$&3'A-qK?sS +.]3BNs5nmDSGDBi<(q`bs7a-`#2DY_Im!`_rWp;ZIi0Dos%iI6:'\qf"O['!&c5GO +B.OTo@cupM]C<?m8R!A#5DDCr]O>[FTK6^fPqsOqp50aD6eg]ru3K:+a=*3Q(XQSJA8TT"lM%N,SBMniS?[Q8H)$3#7L-qaT"*!MQU#8 +5\:JKUdbRii01[GLN+il\R60V_^%Ss`B]i0&CJo#-E-7"l08JQWh,rbTu\`?6&MZn +\4Xr,A//7p(TU+MAJId:SZ/d6daXH=MH0tT$E,,C2GnT4mMB=,7pQ,`)QKOsX-!fJ +fl8#QW;O^<7X@[^]AN3E0"gd'$NJ[>i'5#lV_pEYJ%Yc=q;:UM')Oo"gDoid>;fSK +cq;,@[=BRHCP"kQ+$NFcF.nF'!3RqS"sUfC"6tJOO%<'S)lW:u%jldRgCNcqi/^29@ah@aW*SY7NL!LWQlog,.bR]&PQM=;%dMZ_6kJA9^%kI0H_9KHe=TScp'97fb9 +Rt+u-AJpq-R/cG4P,^E-!OZuK.^FRAs*;S=RL55t$^)D8JH(Di$BQtA!WIcGK*;8T +H0'hV'hQ9b,XLgH=ku8Oj**]D^2?7-&HV"+YiI-,Z63ja;k/64S_hf>4Z=rV<*\l: +R6/jHA!s$[GluO7_\7`P2tFSXM?`KKS=D5SLK\*"I,mn[D(rN#j'].TM^$p%/&DNj +6g<`sg"mr!f9pIDCgDDGQZW0JIVS_WJ#r[&:[ET%RJN,nDjU<,Wc.=IB/cQ!J+`gb$i&_*X'beL__;"]qnN%nmsXo%BE%Nf:Aie[^@\bhNrT%=Z-T'7J%"#/ +ePtIbF7ue,%,Z3+;LPiEb^YP(7I12;4pV/d8#'GA3F3-s?C[ +>TdFr_>M?2o06+)!q=*7`;\)q\@V'Q4cVjF!I_,fITKs0"[mP8`SlmLJ"6Nert>ik +eFWDZ&So,")#bmf"i,Trs5UQc'@m58Ij4n_FBm&iIO2$jcL-udq;73eIj4lI)Q>Ij +BA':?609Y[$,SpSGesjPU6Y"hZ,n?R`,ZnEDOg!1.))]>"\%&=r*IiSWWF$%%S_m_X5D +pVcjqe?Z=_fG%$7&i:1jXon]H9pl.aU[Z>%f/03^(`<5pr.l?_\6RGen<]u]j_8%4 +"u&91q)+R(D3eO!%d5>B*>RnX=mkLAcl^+HO86b5+A)WCCQP"H-^KDT@?EHF[[BpO +k5Ro6VBJ<*T*Z9`9)rs#)VRS\RS8qG.:<0@i;\?Mj8U8gAWj"[CMEaV]))@XhOui9 +d]b"iPUucbN,_isITr[Vd1&2m?eokK/P.U<-0om+[?p,pPMC`2JtZC;^!e8jZfRqi +'k=R=BC?NDhQ(\B2)jEmkOrF+0`Pt(kKC/srJLWIZ1id/Zb#l(h\l)>9C>5[7agoQ +Bh$WnQ.=(MhKtmBN4NQ1=/tN(JE^IGNA29&_=SG)OE-PSRsYP,Q((Bd6Qk<*T6"UN +ND8j8s0)*s=g'/1$ZL?>Hgc%4Et1-SD\gB2&d09/OhA)5e3k$3EXQ\D9H-F2Eg]rX +GG9]dr:&QE'_e:747Uu'MYckpF:VY3hsismb?MX)=]FTl;0c-Pq2(g?)idj(n\+jL +pt`,D?iOm-_H13mHmsO']u^Xh9:^>apuq&#a5?R)pf7?Z/#ML$mQ9ua^[o^sps3mG +rPL)`+.l[2[qA>uH+s4JMcJ^8cS$M,]j1Lb1]j"EIJVCWO.WWPkY/r;T.^#c@q?`E +;"hT7X5:J!lA\.GkN5i0QE7eN\FQS6_n;B@rqE24!"uK8)*WHe+c1`26&Hl%Qp17+ +NlHLL=UW&dniFPh3>'>`p-f'3#i(RLO#66R]Iu)\Z:>Vr+N:D$!5:+oPH`t4/GctL +$LE$*^n$JV"b2j+89r?ih$YZamb)Z +5KJ=GfhSl\MI_k2e`UB?*E^An^md"N0=^pT)1lfZ^q2[jZtK2)1cNa[mM+6VsS +l)jr2XmeCIRn[Vm"*VknruFRV*2l.kK&cmZ6F5;1Ud4>PeU%mM,MccG6KEr\LOP0X_ +,Q9&pJ7aEF!RM"+TGIST"(FfP$,81Q'gVu@SG]li"@O7s&cp>eUp1:O%Um:%MR\cD +_]C8/I:@X,/cTL_*QX1fOSmMea3d5tD`kd@2.?'Hj>8dddoR*Y0R([^njOd +"NUWW#K6XN]-uYMU\=r)15LIj^F^>g:5(XUN;tPXFJNu4l,:$0pf8Sg+;rk&#U05g +Hl^0*6"__C5FC+7"qfe4/HDdMD'Md+_$8tJ6nMdeWjoPWl%>/S8iCA?X$HGZ +FTDq%TW&/_3]>&DDFVoYf"PEDi=H&^V[0PTS9q^->I%B1T)N*=ma.c+2.:6[)'n:W +?o2q*#Ts0g@pB^P&-:G`L0]el/H@KU_]ILT)TEANnc,JFs#1"3>;qR[X3'r4.e(2q +Ga!?Y2Enip?G6?BC#'(cbrlQ^"4m/nS>pYUfHnk;7ttYh2CX]Cg^hOCTA">PnZJVt +[I8UHkO;F)O$9o1cr8\309#c]gI>NB94.8ASK(q#W%I[C2GS*-4jLCkdH\>2DS9tp +-__/cT=!\Y&e_ +p[fEu43oAX0A(GrH[k@bm]ZHJ6b0sYN;Dk._Z+nanrjk?s7OftX-]kN^KbpSe`tIe +>P7gR0*'cI7uIF: +r\m8)BDfL^r(lC7GQ)\hl[L2[!uC]=;sTT4[N"L]?-BmlAO[i2*#4,(PJcAs)ZC+s +pt?G./p)OD!Nf]$EDRIucSZ1%N5Jo\#g\$=p_)djRBaM-8Bal&rD,o`aIc.WM#SJb +&]&>rg3'@h-h9bd2377Cr(-g>6>[f?Z]h0@dA5P%ob)T>CjZWg-NcRkWO?ke,67*r +'Bp!K3u\VjknN()(nj7JM0<6nG=!Oi8"Vl/+VB6F@9;1_`t%ea4g=P!rZ-[0qmIU# +]C"OJXToZGpHOK74SJ4h_k6;lT?Mn1Vi^op&hW^[QMIa!!<3_*(Eb40,Qc?,:ldMH5E[Vu!4q?I5l3bYCeNf("7q<'npn'sj(NR=V+CST +M+*J#B*.l22EfBhkFk72FZe(R">feBddg1D'o$O=s.d&X4fo7r0qURZ]7Fe3T==2FSZ)gk8D?cD`>lN*)s:n&f5B17dh#n&+uj!O+T@)B`U3X$t!0S>aS +*")A;>+'JZ+$!7@L5SX<;,oU"8ofr3n=odAR[MH=P^t;]= +ka)(-T";c+RXqV97/kj=)p.T!B\TECmGS!_OMA*1*pS72d!F0^#1qKshtVqZHJNK1 +#,lZ?pc%"JXFE2\HIUR>P]Qb_?,ML.*P37\mq\IY2fRcESMrdcEUZrZJ]DscS/i!N +CtHE6h#+!?mZg39$R*ND5hV\1Q-$V;e4/D*:B(W's6Ugi]31Uc`=-rS1;>W3Z8'c> +@*cs&;<>iD6SBjK]P2ZPZL5.KHDgJBH>IjQT#uG-3T[6;Jnp5flcXp]b9tuN2g"aU +Y4A)?/\UMapRTs)[t68_D;t,uO_V?$0oDUoDeAtjZf=iqt+>Dq#BVp/$<30c]qR=?%mTU93)":U&:[,V&qm"o#o'H +J+u;)\*g3idL@KU+'a)/Z2.F?jXl%+HYkVR])&(W$gedg!3#ot`;f$8TV0#e&2LU\ +9JVD("M"`Rg.YQIj#1TQ6e6m?W*N"6JI+9(%ZrZ#4=n65G0S#M3<1K.i4AtM8-GS< +^t48mJ'XPb!T8Gg!l4g9m=4a\(s"qUmAAN:i5Xsri21H(aT$u;?2SmZruZ<)dYkj_ +s+C+9T4S%Nn(fI!FY.,AjcG;bYKA\ +5HqM+s5K>m$U$=nk;OVt8c=^)5H*_j<;pU;r4`eW#63#',QJ1'bSpT+>l]=,3*ZAJ +$^^Yg>5=)n*c][WNVK>XINnf:W0KUas'YhF5T]X.#JG/m>NU(N5lNBO.qCC4&QfKO +MdBlDM9U@\<>W*.5A*4k0Djpf#ps0??@)pLXKXEOC +k5Z/*DUQX"\bMA+_3@Fq9Y*Y08J35_-)&G`p6"i#`ErD68+h\` +CIK(ZNlWrp-R0c*!6`tmm;(l<$70o<_bUdo +4gAe]JZYn!LQgoomRuF9[jNKjbc\Vn@pDba0#afr?L"o2o^8EQ"0Iita@/%sZ#5f0 +g!c0M\Xn[Yj`h.]#8?pj!qKtXXUM'-V*NR*''1?"P./V6(2Ai#.WUf?36LfDCJ#Tt +ErBcHr9Np$lR;d&.^5$Qgf+ku2]hB,4ZOE@CH['jHJkLeaPD:@HNj]C+6e5V/NVp. +]GOuP`_?2T#7A1oK_k*ClDOAT^Nu:Dj6<;!c5(hUU$T*eo[ckSIrMFSHhWbRm?Z`^LK;'tVs9-'W]S,!%QN'9l>&be#'=jS9&rpIDgj&e +"iLAD!;*^^^,9;[L!549&:G4^'aKUGbgU684]J]h;J>4he3X+NtJ/]0V!O=rC&fDC;OT,L&s7AGg ++Fj7g&*.XJ(*NVZ@fk]#^n_S\2+[^9/:ZCRB#?Lsce\_fSiIY+"B'_Q<+7&$r?u#7 +`7GIh[/GA;]o/rHa[PdXeq>ZE_2Mk'X1!#ar9>GYupr8j+pbZ7a7"@"M,6t\"5-XG$riLskr7CfF_!\<9!H\[!r:eEXn&(4m+HmmF +q(GLcal.0M_>j&V`G"!*o+;9/ch'V]p`BM96i@PpO4XW(ih)haqe(XJ;7AOEQ%8dH +oA^c2AVi+Pk6jJKOZ">@jWB)shq&Y5a?ffCHV6j-c,BN,XTU=m\J?b-+;$m,&O7pu +=g5QjeANTXO9d#!)!$`s?L.VVRRBt9&SasngJa5paXDDJ.i%Fud/GC'W;$mFJH(K( +,>K*XT*,=iBEWu9ZiM2Sm/jb[k8J8\N8glL`p*OdkH>!`]>n[.ZmXW+@=87-r!5Yq +`)n*=Z/V&RjF9\[KX3,li_4ZC;dTI1:OqqlJ4Zq`dOS)'-)ms[+M5@mq/ICaV3jpZ +&-Ja,hZEcf>?ZZVk0-K=W!.,hph8b-O+;0PUq6Wo%MhH_@QS@Q>Q*4*\rj,&_cplk +q5!B'P(,()Isb>2U5u/=Y?)1aWmq2RY+k-\nq3f+75?IEt +9BDe\>']eK$KQ[V!K[TVjkjDrqt'X3hjuIrfjeSX4B_CFgParL?M;Nmnh]kP%o04? +dC3Q#>a*O=!m4!>.8)*85WbBO#LJ?IZWGmT,(T98I9]>DH&RD]'9n`e@Oo+:o.+J(ajJ1#DVF^&R=&Y\)rl +b\to:N#KE%+Ynu_F<`o0^LiS**oO*qH0S[_mB= +OojQkOT3`[WrJD>AA`&+$P5>r/VTUSe,Dj*V8-QA)8;d\L!dFr!HHO<::Q"RrXYjM +0-^DA9`@IKrp]uVp4314j.R5B5ELiHiA26dO+o_?U%V,54+%2o^aej'R]8..]\&)b-o4IWYWad^0B]SOf0VZdZoAOE?qH2B-- +m0"c%J*b2Smtgkirf=YO5dp]d&qg2sd5mgnlqqi;.^!"'Li1,H/%j0tg@P`UU;j78 +-0>_OlLCgN!NLR$Z(;Lae\s,AKU%O91e]kJ;f;(hGM3<$kKio,1O^LC^m%WJ[q:Iq +*YFqE@/F`n!UTgmO#2\pqFKX2iK+WR`6@EpGu0:RJ6k5.BSC>`aJff6btGKR5+)JG +XV>LLZbZUVf_50hD(#I?ZH3g\7/k>X[W2BGcVb;(l\=[5UU:X6=]> +!G9&s;5@\9l59?J,0F.]fh-:dA\!iD4F7gl]pe1g9DR/6b29sj)._k*lDYmRfAB!, +)^o>dT"U0TEi=4=Q'>5E[St6CWOp^1.22<3^[S'fiK]hDpQ$#9mf`4eG>Q&"Pki@t +MEa.+o$8,p5J:%dg#\+`'^3[hnGgOq_#G^15%"6H1F9e^U?$q39"oXR_:NQRHh0eu +Kr*3FB3*h%>CXT!EQ1fo=0qj_#+pL"%L'i;1+n9BS.5l<5Z.UT!r/96E^D=V4m;Kh ++uHekLfms'UaU;$FE@0SU"TAGFG1Ad=jeou3SL>Nm8N^:!$6Yn-]3EX:Y0LAAr$Ee +g49?W:>esQlYrg"#c%G3g&`6%Jb-_UgAlIX)XKKLhgEAoIFQl:L)ftRjGsiL#Oo"E +*W+&t+TFSo)>G<75HALd;G*VW8,aSFJcDK4_@XG(g=sYhURNh(`MS%ho^5`!s5mb( +Gt\M*\j\tY7gUdI[/]gYlhjPn"[JY27ed-2PFfi;:QWT8\l#/I#gCU@nlPPTa*Y@, +4#pFBnAuR$dK#5DD&h5\0-/%EJCr:,hU>"kLYr1WJCpH:5(&M>A%Z-l,MEB!CTs@* +UYCoCs.7JCHZ2/crdDD*B`Ab>mG>-mEU03M/\k9(s1?Yk5R/2(AE;XVQpgp\9ZS!E +Z@tj8Hl/[[h$ZLH:YlIS0*-=:V#HE977d1`+fYD0Skjgf.?&P6r*t1)BHKip67fE9 +4=Z!C+3S"4oXBDLrEd#>MLUE:AT+nZ>!`/+r!&EgV\!U`"ZdsMW+n)e3N*,.EpS+Z +$B2,#1Z9+b+q.TKIP[%tDN'<7NcaAY_"E15m=0/S9[)B&#rN^I^n5s?>EKBGH+`Ue5ZP*lEG_cdTA_H@4g/J +%_>^`o9\8M._I9:H@L%e6h(VH,h>)G.R-(%G]p[Vr+,?nAC6*!hLu)T0gWJecaan< +HQedJJrfH7LH94K:)g/[:;4dNP4-5iEQV#5i41,;^iGrJqqm&RX^NP6%f]K&j3%Y>9^\ +qiCu9+tJF ++Dbg;.0.YK#;R_0a>Z'pGY%VaGQddr)mK:t`Q\D155>eD:WW<\)tEsVHk2?hAdK=P +hnD&Vrbj%P7jMUQEVY"KhtWft*KnkBJ(joJ!n.'l+R%fL)N[;1:Bj!*r9MEl_H+hIT5"/dEgjJfn`\AjdWGMmMaL +AcI*]B_`;S_,_6N\G/if4+MRO*Bl:BOlF97Hue_+bak&Lb__A][h8Y8iemLqT]IEu +CRW'=?8W\G:NglDNLb`qTcmrW8'Hr)1OQKTW!EBC2u!V$,bWr`,_j5nK. +s79h)I-c=p"9CmjhOR+/kXlECAjXXq8:!)kLEOA;_LqmYk!3)!Nr1!)2@%G=T5iQa +0!u8gD];&B+>O"m>6+0-E^r9cfNA^s0Y4B@5fPuircK?5gAd'0]((?2b+TaH:p"WF +`]1QOWn?N3Y'2;G\lJ(plC*gC/uktA0\"C8f.T29/@eh'%HQ=^._7c@G'CUi_*n*/ +WR@]B$D*g43Brf#r++*2.H\S`oa3B=?&n#QNY]>KTjI@ik?ZS?1\I!$q=#NumN0AB +c.(SS.^bQL<`73eb*5Da$5an86U@lOUsN#^YZ4TpQ'?>!j]Y0"eh_^>$N02tI2d8* +2&$c)aT$CIUF+Si;eW,Q"&/C`eqO6=;gqQY9Z`]L[PD_;B&'H5H-2]X68\0(g"`e! +B7BSEk)\p]q";OYIsA![DeI8nq*N'`YH581C,PDYJ%]%YO-e$SHEm*F+5!Y!4@8--'.O[K0f"<"aV1ESmV$WmPmXA"iPYHXopZbi^pmA6OSa$$#an%,E9;]EHcKJ,pbXj!UtTkReFMR,k`.0 +g&AH2Gh,_5ZiG1NOAu:&Ur,qC6i_**C9N9b5:qB/pco;,\?^CT5)V +b$I\C9.<4OV(^'#Pic)j)7C9E"8guhNq.'qqD[L12_b.K.=WoaW2'8%!O[3URM,LP +E]oo?ZR@%Ka'9.75Tu!b$"L8ng:*9WFe!d)9KhF^]p-`dB/ejT7k&*PIWg4WmiF"T +b;3_V!g&D7-j$QQ\bc/kIpK&QkN0k$^Jt']$`/>6;uXBV97HJOac@Mlr6IG4^4(.> +gIFjgm9O^TIao[BFa$>7guPY!khJ"->+a6GS'Yu1CMI@Z\u;'_J59a)6rb=,2$?o- +"[#DKLR^](Pdj];jHs'l#/5L_@UUH9.<_]O4ILWJB(cS5k9d!N:?I+,"oD:)4< +9I`.crdt/A$]sKMjV0uZ:QPI:r5n#&X5JV^B_d;V7sd5rO[!N/Y_NYrChCLr[f<\U +NW:mPg6.2LK>tCYX")5ABIgFlBgDIYCO[W#fBhGAcZROXFo_$93/cEPfG +qu`bi-iW!Zh7YYq[jm0?Omb'qgh&b.adb+\DAf)X*MOc$lBVK2!i2Xk#QH#fbtlsk +5V.jh-&Ml[T_PG`+*XR[+1'.b[T;mI"HQ[&-?7Z\63fC.j*H<'%Y;:4$Ns1?0L(\L +eCGJC2Z[n'>3Ej]&G&[Wfn=&UHcFkqlY2m\K)[h=6/H'A#g/\e.pBVn?) +>*AmI$(hnU!;mAuO#eTJh?4s9Q\I&STLB3I9ZdhEZ2#er45EE,:UQ(9gY"QTR.2uQm>*jO[oE&]3 +(bLFu-?Tc=-MBWMm;^I,gOn$(j2KsVB4n1[(3Pq;^N8kB'"UDXc9k4!>TIF +VRrfprR@BAl;8qhI$D2t97?IUot?io&+7R:or6q85#qHgF/>L8XSi"XlDp7Q?%%,Q +q#@\P=+9`*$>9J>raOPfcWC&/FSU]Q('!9UiZF^Bq!O],hmg5Xm3?0qn#LhRU&'VF +eITUMg_jCEUNc*6f@kY]oh1U1$c_j\qXnp@NO.>5;js/NVCn`T6Ta3FDo/\rMd59T +^kn'rPBrJ_9f`kW"9@XDF1iPWdj\[h-V\Fh!N6U7F\LKL +P?-A2V/<]LSS#tJER5$Y[F''r`:b>RFAebbe%]:O[:@>,^1?I2G+1Q/:$qNV3Ka?+ +VKGeiS5Mm*T^j1i`(.bn+`SpDm5)H=':i6ZcIh;^"q\ctJS>tgbVY9TrGPMhp"9:* +NC^5fUVe2C.q$nl`-ecn$ +A-%A#E"<>o-D2(*bQb52!%e/T*qjUWi7:0$V270=KqITO;'KEJcd&oHbY<3[$Su)H +Gr"alQ7)_(4TOl!,l_C:'`@P/3C*M'lMjsJp`E!k40T9Y6PhEmC'?B`8(2)0s+FSr +pL)bh./4*ZcR8K08JP$M8F\nVDjuQje)gm#X_IV.A0,rqgEgo%A9P#tC6k9I/Ok/p +:*if!o2T;bqB,\N%-(%bUOluT1n%Dam!gWUJA]0t,J]MR@Z4:c7/k$H63*B;ifXlQ +P:9>>p^agQ_Ou<*rk"!^JS5>F9c!CHq-WH;"SeTW!+',u7a.#$]!Fs>1@Tk!OdCVd +#[/-dg#!LOENo7!#q'r6fn='5\sat:9iLIL)N[2Ns*5]!_9[f@^_$N50-J8*!(_[? ++M+s4j*rNJ5H4[>s!*QreN*+1ZACqLjF>SZAGb8Xj:hPt7\H75&c_1$r/(Hq63+\= +5fiYKRXmra@$?&Ld4gO)^-JS;NAj/5TN,S=a/.bQUVc?0R[TnP-D[i9d+(nUl^,i^ +,E<5BGq2Z[$D1!-Ssq0XBqaauI0FO0Api\Wp"iR2lUMiM30O%/6[A,Bb'e1=o1O-b +D^?:)#,rCYe/;!KU]#mjG8D;8;2Yn5"Lu.6E`=1])bMMg$2C-$X3>ce(V/-3q:=X> +q1iab&:=P%3T9i4+=l>Ra)KX,A+(rLZ/lJ,*LB+.anZkY0Yo?`(DYeGVQ3HdXL^'nk0EIb'7k9b +]%.q/Y<$?'\'MF![sm/'Y%CR;(K[h-@njZlY%L^acCZ'[crlSm$cLYl&-=*hHn'tS +.41^s%$lMG,+p6XCCTT@kVdL!.ssYj,`M*sK+&qHAT!8=!.Z1_RUia$Y++-Z(.$k< +&`Ak'h;/X)#E\??Bi%^#0=tob6eO]-DkGLd9l"=]r9BT4W]@%p8W6=&A!5K#1bpj%PMb4R@KZHB1+M7V3)^GPqIZt"^d21oJ +*Ta8UG\$.O`BPjLZ2BBc2\>fZFk-//f)sV.!Q]b(F$0^K1-s:3UA[lR#&+$L7fZ-^ +FFS;ZF_(B!HEnQY`CqZ@!(=W^CcAWt(YLV#P:lj@j:@pCJ"h!-dBOPW^_+pZr/VbI +[/=,.OT2+X9_P"'r_Ibb?l69)lP>[[idW4RZHk^W"\0r\s&fn-A"b&G]i9F0kg`;Dsq#9bo>$XDP?.RVW22_o-bfH#B[uk7IN] +&oq=Z8HI6Ii3`P>nNU.[+T$ZX(*FLb56)4]AcItj+2'u,D,7[r!:YKTfbSeQC&]DO +P))\sJPSYYD%#)Xf_eu`IN[n7Utk/Ij?Mbc9RE361N(.$Vf4#IpQ;=9siI +dAMXOO1/^qLX6sa<2bNEJ(uQnd>Us6rn5-f!:>8dp:jZVEbbe1[1N7S+%N@3Q4YbE +h`/^)oegEumCi17nUE\`lSJNQKDF&GCd:&f`IEM[\B-_c?lET\`u=?&7[p^2Y%i!U +OoO7k-Ei`se(09G)^C(H'?1d9`\i)o)im&2hX2tB.%Gd5U@?$AkpqZVSrc`94q$gubQGXF9pO%W3t +:n^ktB13=*+67>iaYfFI=Y'(0eToHj%.?5HY_J%b+od2m,IKOog:Tt)DQ'/tZRAU4 +jfC&>D^qu-aKsA1=R,IZom&GR_pLjU-.Gm!FeG)4>t$p9TMQ"M#;&3tJ)6L_Zhk+p +I&urnc]ihJ4<=O:Z?"XuDcH]$b)/tJYP@1T(&rNmid]6QIX%af2c#]("FT](anP6T +qUVdfn(ghiFc92k!q;D*\j^6 +S'f?7r5&:rCqM&Tef5`n"GapH6'6X^kh[V)p$s-aSBO12V4V\kk8F0,HAl`DJ/C`S +f-u@qItU:e<==Z`C&&E93!(X#kniEA2Z05W`Y8L1(dG@hgI*AQf?#,A^T4],6ibUu +aTD6LTVoMRF5:4_@6"Loo0:n`$B/k7OU*'X2R=-fiNki'Z^_3;rucgY:-iV]2DlYI +9XGGj)1hil!/LX=!W\sCG^s^//kg+g8p0StG$X9\0L&u0hEE%MS;VO67b\O6[b61i +aF=Do8tQ7R2T/:6,h`/%aaj;?YX_AnpqlKFiSC(Js"+3!nrEobV"f/GRpF`<4I;>ok'@gCNFjF+II)LR906kBRRS+s7?g5MZKns*'5e$OFj`6jbL' +/)+[Qc-qeenGd%'3U)kUr7Ch<>b))u8$>A6cnu.RZSZ&`qgWl!:o4C=rn$m'C-h2h +o,em4s726's6.CDm!mjL-Gsf)EJ0&D5LO3_j+-smRO5YT_5V"Xd'_9oIN8g0ntrUK +)saB'AIGrh_lf7Pr2Ulp<'E$6o\=QIbDQOB*[psOnUb]9QLiqE);i2inY;9O5J">r +LNf$^f86$t\:O2L6,n6ZbdWecCB<](.s5#eS_Qfm/>E^Go4I:q,HD2WjZ1=)b#D'& +8t"hI*(sk'XdX:5U-.b?^o"L=UGZ#P-8/Ph:]);B;gpXdX3Yp,b->kqg,J+qC3.UA +9M7Odej+K`<31dWk[E=%SF_7kn(BtOjf,n:pD//>-H6;bjSef2U-CeHo^R"!s*CpO +NI\0^r7>2%F7n^7Xo/*VlHcW7q!=:2cX*^PkP5jB3+)?j1ZcB"](B'bQ2]iqUAo:1 +dJ%Q4roPe$-j>Wq5RP02<-D]ain:`DO3Sie)^+Oc#huQ`_^F),"HW\f0eg?ZjA/.@!6CcDcnpM8;#e]n-j&CS +GoWX\l2lKh!?EFLWKUmLV3_;5"rRI$@*Tmm;ZN2oFCb+Ro>V8JcB'qKY5ICFC0.Y"=X7D.KTD]]Zr4:d55#;ni)VMXlFeuMX/W0S9gLM;!U#W +o]\:=oX,8o.rK#m)0h(bWWSimRtB2ba&bKs>'Y;"o];qI2\&_m]cm,u./<82pojQe +rm-4B5CGNcB+G`,E.@,5mje>9Xj$5RUV=!gnSTi>ocWip48"PeHEa$Q[>f0.:ccoP +#QHm/R"drM=I`+oT4fekX/4g`\]"MV(*hU""WOEPS+>-- +Xr>G[6Wn1Qdj?mZZEbom!iHn79rH +S1Xa4KlT!O\cVG0$i[p;p8Y03"%e7sZ0HtJ5;ZOqs2(!iLn3:SN1DH#K +<6ZS/.ms@;%hJcHF9Js_BEfe?ZD:N]Bh`"k=jSq#D^>u\RY"[!/<\8GgOW%BYUk:& +q/P(@9;u*7fe8>F8V$;DF)Y7@)\@Am:uuWQ.0015[r]qg3#V`8qSE(2_k9gj-m\dn +O.pVW<0,`3#J_h]c@<.#o7-?r'`[+kFRbF3r`j"V=0ZOdnh[e84h)eDq=f'^d/IBI +JW0;h+4-.,FgV0?$\80dcg5`VA4AgeofkOM$NU3Bctd*7)`@;Y#t_.f8b@DfNGV)g=q14%pppgpZkFcEVn9+sKh% +g*bo$a`p=Qpm9%i(I\gd;W/^hNTI]9qN;G]XO-@CtpkJPc=o`U&HJBs- +3REf_jV5c=!A_DqQuJim3^ri9*^5%F(GY=;cM;al4tc@iJ=b/T*0C(bTP<9?39P4M +LPJ,.4Nmu]$UN'ILq>am,:l9:2"M=$>Za*)9#%6i$:r;:LFd^@>g;NB+e5Y;ptJ7W +JMM-ETGr[ZJB4icG6)4W"R?$I%Y[HgbeBXMhN9uT>1:Y]kO/&] +-d+2GB,PHg%Q?F<36ncUf6Dp@anDYPh:a24r6b!8>(4E:"I9"_H,H.h_dQ'S/cLY) +''50K(S"^P]6#1YN?cY#hoP`XD7SQ94825mo7(d1mh_$!rqc-t.%#M[E9WGO/,]D6 +\_fDBI;LY1c4oqdLjIp4lMh*8WE\H55s`e$D/Nl!dg]l<5/WbS"9(6D;PnKWK!p1b +>\f_`%EuUWB"3R%ofj"k@jL?=(b8=69a,8\pI=4nHOXPhToAZ/WI_?30jTCdBPZJQ +[Y!6Jq2a5H(F%?JF#dSh_YjCM"[>4Spu16:Im6m(Y7%1.03.!`5c_'%^?Y!Q.j4sX +HaERTf;nmnV>AZmgqU,!\4$<(TGlBEVKE;2m]4Q4]j[[[=Kd +VJ`/cWoS/b/>7eLBMW?r-adg,PRd?B!R)EBiH*l.739N/^l5FE>2)M;WnZNjf%>3t +Pl5!=W;QT=ka#HIJ-au!gP\hA1#l1Ed+J]`$>Lk!0U#1#^qs&0!5ce1 +:BU/8i'sAt7)39ECGF[?r')"^_"*hEr/(+R)=ramZ&J_Qnflff6f_#qFp+uqUM\rh +eFM\`)Yo0d/_U,I=J`D)dI9XPK_k`+qHolgJ"p>d,(T^$i-u1@r,:/-_#I.Knip7Y +\Bb+?Q^IP4QFV3u]]=bbP=YG_ofiH&a\[2is)ADh-SJI3s-!^2Y>Q#/URL'`RW,,) +q+nRjkR8PTQihW+([^L@#U&\.^b]HQs1e^+It0[(Cfgs#B$$[;mJhm=PXN;'r%B5m +:Qt<5a<0^IYZ2&0pDHD+K95fA7k:H@!:W3U![`Y'Dl[+95<]m>p-/OoJo2\?ptn'L +#63FJd=Fh`"nKE$dtk39j8qV104nliU<1`97H_[[I4O`CU9eW*bEE(lAO(`^m7S>n\9!j%deaZ!C6Vt/4PfI +N3ar5r#`#84I[a3D;,#E3q,?I,/D$)L*SeNRr`HbbrK@4cA/>K1HAf*7"+OT4L>Lg +dCbO3.JWQ2P0[Jb6b>V"s)5!QqUV[In),hVc'=p7%]oZrQ0M-uXn_\1o"RCrqU)kE +5;:&*LroW\_9I.Yb>`rg8`nAm4&1bMV!iBNQhg<@ha>l:MW!`$o\ +q]E1@6Q\0NhDp0O6937g%.V@B\D[D752Pf&-@kgiY&5#3J8G7LEDWMW@$*+Go09oS +FF7j;%IJX+!bhUpFg3%jB`CHu&-)D9!FYo2:D35Ui=e?ZRR+/ch5Ng*oE7d;\RbI4 +A627qm:=r^<;rikchJRsrt#5$0"V%S%C]+rm1Sp'bq(1?Vrp9Iro]u5S!BN0 +r3%'Cf-o8WBE?UW#$mG7^p0=&f`*.C?a,2WriboX.K5ZP*(G9=]Ah5_5QW&#%)OCk +VZHU+q`fdr&c\%.GKGCcEI9&q#5g,BkB"5_9R4S*?HKQUKO3J$-TMIRQ@?3Ah_.A# +p_gT[s&h1/r6tY+E3fD&,%+2G"RM.@(8>7P!MT-W9bT^1gaBp21T1IJ?bV@A!56G^ +`BRZ1#aiTl/HG-e*!'5b6bPi1KGM?]bU:HIiIg'kpHLsQZVf?+W:+X[$oAiV)@?86 +3n#e+1>7EZ&+r/ul*;Jms!FqH.B)Dt%IB=)dt2)I54?M'67X*N<^/`SH!u3e)`O"/ +ogJ>ZmO\4NrM;`(Yk/`mQ2L?tP2DlarU83H_!dmHs6V&-5C]oua*6Wpn"uUnlt9f- +1S-O&:Q@2*DIqZQAaF2?VUeWDElq]Z)05h5^&[U'/Z#N`9R@N'[>NpK`sf&qM/6^Q +rtDf!,`@do1K09lA-4^*VP4M[@2?afa)oqp3,0>hIo@q1@PLP-b1]D^"OqNTj4GL7 +Uld\OTIs"q^O1S.O.13>Ec9fQ+3FQKl24?&(OpV52!ocJs*86'ibeVRS#\c/7c:\s +J%\bZa\1_;^KC%#o2U9';tYppeK6ucnlLmTpu$_/n)%:?N>mejI(9*s;WYm^daP1Q +8&FB^H-/ob=+Lp2Nh*O90eX\Kc7fY,j+J_DY(%T_]/2Z2rOm +q]Flpq]Ab+7K"&#cQ]QL&c[o/9?Ws7!(?hdS,^UkT?*W9npkYP`!,[gK4t22N,DK$ +b9l6i&-4SDcIn%,J-_c,T7/-Y+*uiUJGDe,58.c6q+j>&%!m,j'AfI,0fhW]$[NTu +/V!VC?h1Z,?DWEA[/VB0ejqd\It.<48!cShXoHe@QUW8t.&ZJB4>kq%j.G`?6Nqj"fR?jSoL@aF?ubO?p5fPEH1o/HGHBkNV8?L4W4@sFH$l"A&;1U`W'8jqW$ND%jicu +S69i\:Z^[_ARE!EptL!-s6sR@cQ>GkbC95.fcQigps[noi6T`8rSa`]h`Y$4b6JE) +I!>".hSad`lMjGqX$4N`*V7YZk$9"O_`V\-L5K'-;]on%P_k7pYK`gb!fTSC/V"QE +k=VUq%E#r@)-Y@BD*n5oXCk3b0W#bVn-K[kg\^!75Xd1`o+oe%b9RA#G +j\q@`l:<#g5!ARtO*WgOoAtDMGA^l=E)5?Z:?DH9l+DL0=-WFYCM;grq#Br0_>j<+ +XYl`c=o$_i(&rT^m$#Vhs*VA/7)5G@[ +/:TK]I)!;c<_MNt?.FOIEE-RphW5dQ;Jj0G+G')4N5"p\ur(eLZ3sDeO!j7L_ +0L([anUG]iJ0Q3(s,aO?p`JQ=*,>^K$32=QU5G8HJrY@>3X)gCT+)_9.qDKS7n0e&P*rd+ATj)Fa>6#l[^ur.Fp&HT=GTRh3d6YG_?6USB_/!dZqCaGN6Ks-M>k +ru_4^m:i6T:]#XOP7=MEX)[c%-/SL%#t'0%Zm5&2uu*FY&6Os.e@40cl^3DbN^SPE7?7`JY=A!CX7;FCe6'+@;FfGj2?@ +&r8dKSVXMF"UNeNK/REDhl?#t$Jbo;f,t:t40qt=ZYots('#P/pn,VL988'Kb[L.d +!89:kMD1$r/;bddM@4n_[mo<'o$MUR7ZPg1n`nN8`c2Z=b$)F^3ua>X2iB>gk5NS +-<^7k\<-LcTEb>/An#jf?uUan^Xh2`d\4&cnuaB%B-,k_ZAT6\q?5r0+n<EEk +Y5e"tkl6_D1G(k<3>`%](')Q#(i:FXkO9nRZpR^kr-quT9WHg>c3N);kQP=Trrb.g +cpqL>#Eer3Z7j9@h2:'IDEfjok7[fIrTN75p`JS#)&*(\r_K.$6i7Jca,"I0PA*6E +o=q0ta('TCJ'F?W2'PHfn)G#[&+4*LgtI9+5P;Y"lTkHbcH\K?s3C=["CQ!.q%*Eh +D[fFU(IlW0c\k_f_enCYD;rra<(?0_]Rnkd(U?s>`VSbODtM3G\p)="GGMW +UPS/*qc#sI'5N@;8<80SlbD'H+3]e6$%q6kTWF,7d2%k;NQ>O#da? +5AC<8%3U/f=G%Mg+TH4W*:Q2"!#>[_DAKb/(*F:l1h8TH02K[9Y+A03BHMmlRFjDT +Z$e+0,;l_8IS<1W-6k[Qn7Gl'bPoQQTCO6,"6:qKL]Hp/!oM!'AE5D7K&T#Tru#1e +]WZIohm+)8?49F[inYaa!"V^G$*ZK>`"B6?Io&;6I7F*\,3a$eoA1m6Dk%@#h\R(0 +HP-U?!9aJ,!iEl0aac'n?^k+jr/\q[ob[Sq2$kE-:^TJdY7Zm4^kse!ToXgQ;]l&] +i(*R*]R>Fh'Lr)bU*X=E/RocqP&42V3])5N4"7o)2$1&25qqN9m?o[HOPP +m+ZJ'^@pFJp!:mL^V%>ZXW,EsITu@:mk=P%WFF64+.k1_bF.q94]h/<(WuaII-Eke +0DkWfAUcmL$NC(EK6]\oKDNkJlFfXON>V8]/bdAL]R.9^Nq8Z[l`#+TjXm-A\WiLR +)(8+tTX7B,!P3[6!gHNi$HcAnoWFtkX/Y;DC17i$=\DK?;pu3*9HuFJ80o1AEHY*Q +Sf+QO5uXG&L9loGf259Y_qi*6nekB.e<(c1]+rgo%dIM^q23pH.q>]8.=5R9a&TIEF$KqOT]<^F[=/u7>U$?RZ +Y4qsCa32=!02Ios4&PWPW8Kj@aXuYtrS!NXMRB^ONS]5GQ5uIegZm="Mg,aK\RK?, +IRbWk"S.%'!cSKss2fbX'DlCW6/G5&\'1*pXs2Pg/ +ZlG0m!oSR&HMBloTTkW^JHVdgBJYd#?i\=+gFknjr-N_?m6prsr[5ZLpg:^mYO(;L +qK^0^_a5o(,]H`&X2P8P#[^1fM>U(/.;7URB_d/S?VJXmXIo9[41(2=r(i3"idWBU +M>ufUJGB/U*C8BIL!flIkJ`Et94&HrhZ%Z!r1WZdq2]Af*s_=)s*JQk[oMLp#Q]1# ++2MkmElAdMdl84[T+&i>_JRcL2M_?ZjYu6"c<8R97,)?!*<=01.bL +M+>kAJJ=b#0A'lh`p@qo9bd=E'+qVN9>BtSNqrnY#M[VXa"KsRpTtM-6h*t757dkV +6p>]jT,(H(=r!)`qd66jo%,@[FQ8@2#,>4'], +;i^!SB?RYT3ob+LosDqg62n1/5V''#e?V.]"nOC$V0aH3@/`;t.BjL!5m6Nao!/"9 +`;.@O7/2*Qn-S!fO%mRmqdTL%E<1#$+l&\$"#pH0M_T;fblNh;-11Ve4Pp\6/CWU!g."<:ZSAMgkRh +i@c9)q!cDB5IIf/cj]Um!@*VnBcDfZj +gKO4Beh`I>W'DNfaIt-nb06ON'VtsMaT'uLoG!+g!1mrbIn!ll2,Q_,$NU0uD.9.D +BOW=AbSZ6UWFt-T&C\t&4=nXuV?0IPE?TBhc9/Of\d/-4`CS99$e2YH0L*'1pGXI^ +s8R)Ii6rL"_>dU$ir;Y%&>TJSG87"Ol<>c&Irbegs.>X>s$&PKouEuRJ,O'N#:Bnn +IolXH2]$r=8EoHPb?gPeIO\\!OMC:&8(hl\eWOZIC2A<3E201PP@7[TWi$>,M'2g> +/t*Mk@I(?\d%UM2$BRObM1@63!I.*<4u!#(ck#7\(Bq&@C&_hi&&j#+m=4SDMeAt+ +n.$X5&;3PXkm4e23C"a_Khhg5#NB:\T-\t_T9*ma,fRN<43*HQ5:qBEJ)pZ+4bW_D +h?O#GPnN)+XF8h&;W<0^NW=Z2.j\:S'!kbN6K\b%e>"5LY^jfEA/Ff$aF@TEA![/b +j<+8kOBDjlan@q^o0F`jCgdp1[^([t2#sCUe"E93A/FGGMLU`&@LJOX+@,!`f"T7u +.?VBo/0MII6MM'Om$R..S]Jg^\^s!-ne;5jN,!73n1bR^BId?)Y0:^;N.J$2!" +Ckgq6?kI^E4L^nLT<:YUYspbU?PXmgOT4ia*NcASpA?21*O0:)gAf!4n:1;YLWo7[ +p-36lbQ\@o!SNQ5=Gd&8)>S%_>h+'4M?"r(UZOfE(j.sAOcNZGpm5J*3Gei5AJ]Z_ +@_\g$jHlQ;J$]G[O3E[;62Ws2g,T7"`po%MOY&A_*g$D5uci#A)inV/Z$t:c$1B?aKu#C&_ZGHU9i&.C`XM"/sq7 +EjPZShf+GP62nt*':=Gtmt@&r^g%^`fbT`Epr'*7n=WjhTg=;q-4"B"guOOBCDDI$ +km25I:rQ11JaMJ!<7RoFhk9F,nqRp0arf]*mTLMs-cK5Dd&D"u]`:n"Sa[RU9hl`, +s$u-+9)L$Cb:$m5Krr5%>o9ng->e`80)![XmHrB*\b#,)Tjl/qDZMXlP_mN/Ac*'PE!noHK'eg`mOEs0'bt&qLi6ZH2Gqg&9X>Vp +s4J`6\d>JB,["eZ&F'E;RF+JRfKb-/W#1RU--k?$SpSp]k5>YP)*@LhPkjbFZN'ss +Gj`?>s(@`*8@JSD3tPR$I^SI)Mq.&5i'M5FMG5i=EWh1TU:@#P&H+K=!H-5Et[N.-@_8s.q&J1L9C3'Zfj[Pka,i +0#kdNZ#WAe(;u];^'?KJ>oE^7hrZC@'Vr^_P5I.s7 +JDqXnr_MDiP8X@`aZoQM]SE9;b#<6AOTQ]Hc;4#/C$H?lN.Nn]7MpD'l(WT4n<_iuA'[Y8ZH$'KeV_c%:[ccuRB#'EQPa5RIM<$_VL$ZT7)^MZu9' +iGS-L<;n\QhRTUL_YeNU/4@:lG*)`?'GEqJ0H9:H21MhD;IDQ43TY1Ko#_6e*cuf[ +Ne#>:HM/,kPbWBT.@Y=cbdq!@gp23dI)4PAnctJ4ZDCsPSl"b0E6Gsdli*iA4T&R' +h?,gAl?_@%1SZR*[cQA(lLZQc7_O"[<(<=Nl+L/@ag*Mm?8,d&(gI'qQssf(%j?K, +INa@i,Qq!OK\cbZZu;RXB!00Lpj[(ml`R:._2[X6H_5`["s_dc'6!CJ[eff(rS/[' +M<4l0X[*6fOUGRfZfaXTIh%QiH\;p`(.Hpu'5WR4JcD3kr$aiP/7?E@Mg;Jt!^RSe +%KbY1in+=jY7HP%!HJ+6'EF2H!3kc%9-Jldhu^L@,''N9ga[n'J\S#l09I-fjWh.' +JICO_%uWS3FOA-6m"e^M+q35q,e$FuF +=NuZ'5t:pN"9]Jo=\+,'\Ng01/#8mC<4J$I'N8LHes(uu?-YFXa>hjGh.*0#o.b_M +:&T);lcmt"j;tYQgSpc/e[mfr +[F3I8<`cb"

    iME]n.LY@@6+EI02Bs3Y5"Z6b/6(H"p'RfuDf[[7po.(Pl9\Vj(;6&#+Aj,+oVp(Jk22A[oi(c#t6! +7h,J)rWE(Lr##E\9-]dbT+lc38];bire1^jh1?!WJO:&XG[D5Y(g@rrDj\ +1#;F-r&:T$+QJR%!$B(%mM2aKpdat40PQIgYK`_r+?-3YmN5co,B/`/"-/j>t6XjV/*Ej60jI.+uE94sdg:=[FZ) +1W[tC3S4CpL^QHl!;uRVC?&0ErTLU),6*L^jok_=5gpr0'9U<);#Oq/DUOU@5CrT: +%G2cs+SZWLPMgrG;+gOnPMN2\!'Xe_LAuq'Pe\@tD(D0^QYgbo8nZskS%YY&I'TmR +;Lh!Zp0d.YTPVn,RIc@!!eRr_&',:-KK)BS6:c1M"c.s+[5CF +g5#Gfm`)r58I#9F/)Oju,mXHCV@@);!.P:LaZKSJ +;Y)X;U95X^59pa4+X-b^!@n3GHOWlf!>!=?lY&^!$\-[KRB`+FF;5hdXhQa!nuhLO +pWY78Ap`,#qf?)S0'i0DW7Cp]oBDlPbl$IOS:;dY0DtZehL?SP5ktAq\]hhET(;$' +XB5-pcd,Kb"K>#uf'1Je6aG4d)d1!^b/0tD$HNo%Q(hldi8C4nplE@f<2l*CK7S>7 +EuEP_P[cH0?-Wk]UdthH\e/KS;h`_C&Vr.F.]+Pt`_37%%V(sJRM@Q$bGf!2N)RF> +]gh9Y5O/K/s0je8s6na]qXff4J)-=^t0GGD]T.!S,_TH +ge[C1pYhZ'J%I<(c'qX]FWKd\ql$F$1*;oZ^F,ImV`3Y7D;fC2_Wh'4:rcI\djf5% +l@Grmb)1=)'cVH5"j2^3J3R'@=Z(m<"f'%7";g,5J/eTZ3mbr-](=D-.s1E?JG.)n +5L"\PC\[AdPL044ne;"t7%d7B>6&gI0aP/k.E+SpaGUbJfVdVXr!<;h65&urRo&2o +\a1F,r+SK[5AC!d"TO;X]cEXB/fsMS"TO/Tg&Wo.T*Q;8K_LVLR%<;k,2EWQpY"LG?RO!??$lBI"#G+GVnLDnQ ++)%hX(E`eMUSD7))(%Sj'N!kOQCGt26/^qB&Ko,g.PVIlYq:3@$X(#U9e>-fdZtG+ +5_JgU,';qt0(ZZ9I-snIV[NmXH,`K`ipaE\U_^;TA/#mj6N(kNm3[D!5o2`h8F55" +,*Q3"i,U5 +>Yfs$#lA7(61*,GD=B:qD41_h9Esg\!qh``r^.$CacDI1C%roU97c]dkSj3uq$Hs4 +r%A0ODt0*0oulM"AYADk3G:p]$j7?M/Kimc'Q+Jm?0B>/UKN +7sI+iI,Y2Mi'44C"U-[_1a[+WMl&k^>c8GE[=^O0QrZ#pC#V14*4pc]Yts\#ri8"l +(6Ggihd&(o$!BM"C%;E%>C-'88hOMDaJRr!p)Rn1;^6I",qcmHUDN`bZ@KJ:A1\>H +j:Rk4K(]caa6gNB-YKodkZD6kmEq18-C-cuC!j7KF+/9TpK';tHfJZ14*P:bG]sXj +*hqZo4Gr'hi.(0-pWe-cY)Ki-96pIQitm'@+&>`un5:&[e;7f"4qVRB/^p%\R4-3C +0nKAYs2G#P6jfU3AtY,;o-mlb_mP0g0;h.;cmccO"pXRH3ZeeCh,Hqi(/^QZ)7+-o +Q:&o;6&[CCesqQ$Vi83\OZLqnmR'/%A\KiirGl,b*tKOoIr^iDq#B_kk\U"J+"mKZ +\\)j0]\j.p6-h+!n=TID8$fsX+S4'?ko\]Kn]pN9J%-fd2oh:=*h]+X:-CAM\2]Xt +`h.7W/RqPB2`6[5BNo/L>GA9&Y*?-TPcsJ=YNmTDO1N7E76QM@AHa%E*t8c.MZl\^ +38>t`p`%ojBLZ53)AZuQ(>tea3;h1JLT?h=VZ?5t;;8"%mf3%R8j#_6X'=TI'G9ng +"^"q3TQhZB3#$O[Mhhb&efZ&pp5&sgPEc"Hra`m,\LCkY#QAKP,4LX0Io+H%;L^4+ +YDl0[rrYE9s5LeH^`V*Y832iO!Q'+RGlScjd,-)jB6Ak!^PpUE7tcM$8tR&F^dt#ep/f(]PeIo[U8-CLYB; +cNO9u/NgBHG9ANB!6ZJp[g_bU*,=jN>%D'R/E/(;\Oo@TE`Cd`B)t8H"[#*?-%UQu +M70/UCbi1olENn9-CIRU1_"HL.Q"BKcrUY(pD"PA$]P6WkX84kYmdnbi[nsLn7'dc +Cr8BI=mWLfJc?9n'7gO9^^Jq<]]$-F@?]EI&tA\Hp^cZn3\?t3T9):BQRtCoRhkL# +6_=bYYmL@*HSGB0f\N*P4^CLp!XAO[a`RT*,8d]d`.%gcf6n]/csILqeW!!:+o'+= +8+6b#)F;^6=S$XEEB;_Pr1EQ19dZ\:W8Yd'"sRV#,`I%KkHUJcFL&jkBlp3<+Wm42]tXV)q#l>p1q9B+UF`fF\lG`!=7e+N-k& +p+U\\KXWaos4\4bh]1S#p]],14<<.tNB2>uEO/"LYk!0RNlut2Dpt$a!W6fqLe#Jf +B1IHj!<3'X)@,!>`[LnWFEe_8\L&UEJ<-]nI=,hF!'LR3!.R=ac`kW'.)l-Rd\'#M +[OEEoMPrlCd7=Dd&t*hK[^^U0]Z&J-2ZWmHSm;-Mm,DZ%k5O_o_I)31i6C"shS(N' +UPiebrkBQ#_sr3tU]:rO6TX_Y`U%fpB)M<)o=%4,^Xu+@2b@:u;\t1e>8gXkq2D<' +9N(dI;@EXPD4!rG0j)6kTqi;4n=QKVRS;j_2Ue$\=m%T2M\s1)N5,cM]UF_[B9$k? +:2QfB!XSh8PTXi):iflV/"ZjrcDUBXW8i$Mfl$)+@Ho:#l\,Y`s$%JsIJZ"qdke(T +hOP/Ss6_XG"AJD/2>fS'LZ"4;rrlaoRJYOrln9I'q#%k5O3XZ:F"6iY2lF<34mSqs +e.6b$O)C[EN&"-3dVl@$*RLXDo]Gu((R](1L(qgWb#Ys;Y%]Inf2f=$e0X#),;3#^ +(Z?p<"EU?*g_(h#f=18ZD1#hTceKtRF9rQWH\=Y5!e5Ed(e=8No)G_BJ->f7qd8bt +R7euOFGktZ!<::qBN>4;5eJ)d+TrUs7N_hZ]0sab6QSe!58j$E)L`=gLZ"e2J4oZ5 +#!N%e#?uHRJ&6[@rib9F0)ioe%o:X%IuOUCrY+jT+SaR]8T=/P*pOLMZjZ_6&c]II +G[LQ*s.fu)(=6;b5kDb)@Qc+WreUL&nPK8_p^\Q#5bndD9E[uW+#Ws`n5%u7:kJ]V +,t\$tkQZp"qiQF/BX3q]K?r9H39Ut[bgMSI$5O1qb!t1MGW4b6a*imPqfLI4F@lFS +WQup]=/o%2,^BA5I7TI;F=M#q[oQ:"NGL\HW]#^s49YeAF2g-iA2AA8Z`C +\,\qo7!Q1&hXH6XO/Mr4+kem=7J^F9^`PtKIguF]YlL=a8C)0OX13e1>d>KF\!tjq +=jku+gkdSTYd^MAGD%;\KW-tn?W]ugQ_W(!XU$;+IffXE@r8QI*?`<#=PIr)FZS#7 +*Gh2QVV``Lnm]',c"bAn$XYaW%isq]k"cQZ^adTGk6'F4K(&6O[+O'k+i;Zhr +q)3^+L;.1fYQD0fU]BclanLJMHQo;5,1>:UdE[ON9120P!+A(-5']H_n4aKYp5/L%tnF8Ip:ZY7-FOKk,BB>Qa2o(a=H[^KWA_)U&'@EkF#u/-PHd<( +\CndWlf/rm!WQ52r\+6/Nd3iP\DUFIbl@XD3IeXs)rglnn'?^_T;2JWim7R+T3TJ6 +qrtK&gqrjR+m-s0W1T#IO*R[^6];ZfX?A[\C),jRUoY8AVO/\[AXL:8ri7dR+U%kn +.=`n50pKu5Atl-;jrB+c539n]I#Fu^E[nAPMQL;^DP0N-Ydok+V22CaKUQBqD<%oe +gUYg?.9\c[VMfac7m5s-q8jKHMlk2^p9WViJar7ZeedLJs)TXT]-$*_qnrQos0k3Z +(4bA97g?fgbJ-(;H6hPB9<"Q=Sbh3r[(kCbo(ifp0-=U"BEs.hh.+[`K'C;qft2Cu +2YH4BgQkq[((XCr!Vgl'Y&BTe\!6$aQ>V6IDY\UTU7bjjfN3o65]Nkrb5V6I +fEDEO\-iT#l0/nG]=kQOrlMK0j(4iD[^Ht%r5ie_Ef*Qrs1nYc2BKZ+n@u_rVs4km +PX:i=+MrU^K#B&7C\Y(TOr+G"L47!h.1Z?rK3^fDLQM#?Vj.ej#E]]CRK(m7!2TC> +-1\7"PFjt@Kn!q?Lupp)':.R+!NH!%%K@'H/<].F_?/@Hs$=d)G7P(p"qaH.X:haf +LdMLu!*B4>?g0j/!&*hk@MmFu#Pd=[5dAKXFr5lB^"B1qRtg^UrFFc68Rgs?Ri]=/ +gs61u]OR%?(ucC_>dK`7du+4mG)ORQ7<(Ktoc5qo3:aP7Jc#ZMW6RUYK]&JY^`Pb9 +90gV,+Rlo`l[u!D$rCQVq +:YR0O%d@S\$qi?k(XG.c6!N+ZF5@8^s,LJ%-@[=T=1E!`UiLBAYr>RpJ#%I!XE8A[ +!\OK,'a9F[fEDDr!_C+&'R'RK%2._L!rd*I#e'M^RsaFofn>iU2[.6G2[@Q3O,4)j +!9$/I!.[9_kQ1_.K;A-@!\tDfpM^R6T!=VXPM\8@6oCT&!=aMaq[*0f[`YhL4Q;$] +^FJBG#":Y>&bL!;3pMu)U>@9Vh>m;];#Y=:e`S(jOok"qPdIl-F+.W(dqm&CLT*bFFlP@M+#?+s^:'#JZF!:OTCgcVi +;:@$D?P[3aO:i*c!Zkr-1_dY*5VYNG'aQYuV)#DlPTbZfa^R6l1#*-ch]aPSKF0B1 +U+;WFl_qqueFZ4F[X^qR!,*=,$/3[9m$CO-<`55>tElaTD[lW"oij)pWEfJ5.roRK6+#1T1XIjGj;IBVn]Gdp$p;. +HcK9%_?g.QL)3PX093F$)=*W'Wg<>#=*2O7[E8h\A48)Hn6Fj"?e85j^0d!e:f/oK +]a=kA8SRVH(=8`$^q@DP7 +5ZJ`d7tMfa57.6e?ifsROgROZbQmpl_A9ol$](jC:GN"DCu\rtDBEo@f@X2gJ:B;h ++%38X&cU7q5N+g5rb;Alpg990s26@j8>?ll_LB-GFJ\t(pl@!lLA5+.%ET8,Q/YC+KPi +JV!_1OFmsi`P?u8Rd\Z2JH$\(&GR?OLiEbE!=OE/+Q>fMO.Z#):lc0@rEcNad/bl, +l%Bh3OGD`iSMjj +d7p2\ngAkbC/k!Pr$Po;0H\"D"@!V5q%??#f9$*],2g4^J5>dO])s7,G"5&"H[B16 +H[]&:#-`WlSp[bm_>[o'RVNlYJ2)>>MGTs,MAlI5-:_9Pa4k5]s0mrq&cPQrgBmnX +!f^-mL)F>=L49rRJ'=T'Z,pKV=j/(LRO.JN:SVkC>JR1+3+`oa!#pIt +]O5qWJmCVI!Z(k(M@!tqPLP+0s#'_0pjWt1&V_lfbjblF+IB]Sre#%@,1>RAidZ@P +KAqAj?a2m'0#0F=#qQ/[X%I%E'\QJ1a.Se6r!E;7qGNqW*j@$mrXfA:IahR:o@CqC +X,?a(@fEZ1J*DVn!l(s&6,`e-,-1,M>B\orOsd_L5JJT?Q0WI?HZSmf:1_.`/t:Yo +qVkE@^E,f>nl#2EpL\l#54cHdM#VB59)[j0lM5Gb8JcNaL2jdNooP!`ZAsgdURWb] +"YaBncnd'j7HkXs!Jpc.M(0(A!-JW+SAm6q;YD(e/BG-t=[6pT'j8A/XeA4h"jCdj +&dt'W@Vim7>0PdOKM?!\'(\,>6)otDmD(s"9#O`5ViA%PopIu +OUqmC=[3W"+YVcoe]It%8/<@5\`=.fO10Fui1CE=:5o$8r,:PF8FR/fa+qF"hKE44 +#CiC>f\_*Wr"%6KM/ZHZrdru3`8k/_If>8a&(:,BY3*J)ptsgI=2G#2#le+D$c)mX +JH`iq[CRJ%q,^p-nEHl)5f1aCeQH6IplB>gR!$_GRpd=Q>0F41eH!5`RciIf?D!KkL>-NXcF.#pCeq:_P; +,#/j#itX[5s/cJSOpS7C[htrDV.O5:aWSJ/_ShN;5l;M_Y3\4J:Cd- +5tfDtp=AnE!QqPSID#faJRYa9B/5>e/+L/#_h#C$UjalK3&T9uH(ne)[K+8GdC.U^"!r-\W9BJ@b +D-kmR[r`*D62[nhrbqHXJH+@+&G]C[G"6^c? +&0HT#*tm!!hs;$#qF$gc,%L^,iM][GQFQb8!j@aL)J4I0s%BNR5DHoYOafnq7#lTX +RQj4>!<8_"!oP/A!V!-[r&(8KqGDb&Lp%:oX:A9jL)$tD'_oF'JG=UR=E&>+jk6o8 +(T3T^[OL7IS_64lSGham1]DVSm@?N-f^D3)8Q9e +h9>ZSd>VH"@(nTd4l^*:Ce-YFQ$L#)@s;A8fHSJ-5ri'.'S%qm)$#.,^'-tYJcYlV +n<;W5e##rr1s,k8OA39,.,P(U\;1JFTcu"dfP5/K/N'qeYpl;">JSlp8:^rgU5BMi +XdIheViJp!PE")JrR>D"b-N8Vs/)dIG>j")Nr%ZZp%b.PF9MMmJ;KfC*djd$8<##%jDs1-$)cUWM +7-$!qK)ZSVe#Qf2"d";-kTs@Cr:_B6@pS%XYlN]HXob;'+?Bm$k.daXB2'#t59W7C +Im`aFD)"HX+S[MseJ,")U6GF('lSW;\s5M0`_Z(G04]_eTS).Z.0'_uQbOZg8+iK\ ++\hrO/-):M9/-B='E"1?rc_"Kre(&.ikkU@\%o/S!0RE!Q_:dUHk6YR1%<;R)"L!( +%]/,[peM[G-hEr1^V>dri1Dt1^2j"<;"7plM7Kh.cs&?^;kWG*5U@Op%ZBWA0"oVA +`;E-'D_f`N9%nheLF;N].J6nU?B`,!&L%J1#)HA%h0h"EXLCJ/Y? +F&@""IS7l([^M/9nB:5P;"_.<=b,;DD,Z,#FX)IWnSlrVXEDu_qs8":k-7Z*;Z?t0 +R#SJuKUSh;NK`5q#uBBEX(K4O1MU"0XR`Se9#c5+Md9f06I:NpKX(T>JNn&7D6icA +]7ddYr4QG9?6AolNs@`0kJMAMVdd9MI*O1ul;YuQX0r1HG(f.(#4$Gflh6'q"mh:L +@9";+k&o^^^jmb4\877!g+ed-e,O=%KhW.+f9smm9JTY6&PM +$+C$@="F?rQU#VI:_r!MN]9!!.hC\uEl]\[D3tD\lUi_GCj;C+`?o_a@BeD2?O5$_ +8hg/<&GZG%C$*$M;SYJB)G3lChc^W#I%R! +P4><YQeo\;kKD'UY>BVH0Ce?&Zo]m6Y$T&.n6s6`lhA'r$+aW%;*8'=GbC +,UO`(^-N@E)EtkXISb^3FV"jQJb4YJO20'Mr3pl9(*J2.S@M@%C*d!_(W[+_'3a,2 +]2BYaidm%_>Vt*J!@!#2I(XN+aWQ\;#VVQ1C+:)(rjVDtmJMMFIW#T1eCV_-Ds%_K +LI9ml_-Wq254Oot&&&<7$eKl)_;CoKK#loSIK7oJ1eQ.l[pTb@U^QjFJd]qZTY=Gm +:D:4EqLX(gR]HAcM:UZNQ?J%M#5bl9rf3WR%2>oLRQbl&;+l-8]7`P?s; +q1Ro+DWkiifSu_Dh:UJJ&,[g>!3-MXfkB)/@akUu +4'fj%7/Ht3*[n"[1Z$+Xr7@qP&C6R/&,_3aJK,kd!<,0/3QP\^pR(%bVn'rfXK_!5 +:"iM'^IBLrr9rq3[C:,Urf%B_NK"kq)qUc7TKU&f.j+Ns?3BGEgJL9357qW/55oI- +\'3di56Uuds)E8]W3^fo?8VcbXgo,tq:,cR_gHd>DaYh/B_`72!%.jG&Hd7*baGk[ +GIi&br^-_cOWa++LjOQ;7s#k%hrcZIjXK$R0&,eom-n'D#ZU7$f'ElImg3"]%9Dj( +=7-GHJH#kKPm9dYp5LO3>`g#I&)&*W4YS)7O/bRq]eL*2.1.Brc]=^u@LF=V`tfed +`Nt;^qiaoCG:=i5uD<:EZki,9lNgi/fKhVO[Y5H4J-AKLu'd"J$Q^Nh<] +1)4Vo#hMk*CF:=V+Q39Y^O06N\i7Df)VfRYFOfXG'piNQ*oF]ia86XbI!FN:DTr+O +aiqMl8cS!Z#$AZ*m`P\,Isu>0kBCQ+r(m+fk@850+(2aeNI$&:H9+;S@V+CPOEsE< +`m&'D#E,PJX/J`cW7X5TT>n)k;G:&iL4uM?W>Wo'.Wlk8iI7?i9V79j52,rN-"!P-#,>/nK!k4BSCloN_+6IpTLj-Uh^cD@6,a5+u& +UT)#`SGMA3`]H+W+5K\?EBa'Q&*/-0K=l,o>a!<\Ziq*:r."5:RV;Ql]<_HYHe)sk +Ac1GZZtFTY4YiRW7l]P-/TXsXj(ZWc<9JDGQa`3^C6Pbh_Ye>=Xm7CkR*>TH8rgK; +UTNS@lNK-Fc#!ac+DZ`4pjb#U289MmQ%Y&Ku5Io65/-,H`<#4_`Ku:T;ckbl`_>Hg(]h&SC +8HFA>ZuCdbRed3(5O^lYr+e9H+c_!_?*3D\0(*U1l>/%cD-YiP]Y:.P6YA2QGOl#M +/+C:b(PR_h7mPP3nSh7r!+Z*qFY;q/:\=.l3u=\kEsVTh*Nl3-]YK"&rf+gY7*T(L +2bS,qBrT-Mk'&5*Y0):$_XFGqgF>H@9/TfOU]?]h\Nqap%SD9O^"n8]bl?fP`;f]1 +FnPECR[Vg9[VkfQiW[m1OG4PhL0T6jVIr]"40@jt*qmt]qo/p)js]DY +&b^(:>gTT_g`&a/a.ea76/b#"BX^J57^1(6f^5jDm0cn]%p&$+<=o/uO!4KE*2uVp +i]sRbnNS?`3!pEDff()Vm>VErO75HI@_CM/_/=9YN":LHs)AcE7/4M7P/u?!/n8go=nD2.K]C7#1uu3hObe`HhF56cdO-e +_*8D/0lOt>YAlZqaPS`n6Dl;sObUtKh.3ksnNTu1'i8`>96^.k#6GjJO0#pkP&Tgnn?VCDb>NF[^:8I8GhI7en5e?5/.?iankYtH(U\OrpWV8]=0NP +3M,d'&aIt-"o-1fosC?:fogeu0^D&0k/kc1@fgCp&[ca=27A;JA>:*[L(lHPFI9nk +;CNMp!U*kZ&0ig3J7052ONK8T1TX_N`rUHb?jS'bN"K>j`E"*+(gTF%XCP#-@Vf@H +m:9-cACR^je:lbD8$(2N6`cVeR'5tUhE$Ba5\LL0f%5s8r;)8hh`c\$^nX3B_#Gbu +I=6AAir>k1O"S^&r;Y%3ba/g:5I2s'iMj1GK'>eMR5=eEcGo3:`PV&KAZaCG9p1dD +dT)\4lT<9dnD@MX#lII.rcPW)s+U_Ac%"Y*c$sd#9DB%5%r'1C_HI*kUm6MJ\l')b +%r!?bT=kN%j<^Hg$4\5u>l_!`0-r&t9.:$1$=N06U]2_pqp=jtM>WRrO!B-t1Alq, +(]R(skGku::&%M=rrE'(H3?X9T(jiZC]XIpi;[Qkb@p[OH\7*A.*&s0eeIa9%g]"93^O+T=#ZjMr$$pjZj7ZO0$F%^CQFmS3Nf +f/<&Q*s,Sq?:uLBpq_O!JpKr#:3jWX*cb>94k63e]-cU?%Q^UPp):EBg0q"7'YpXt +"obS!6kKBBU\YM>s2@]l"nd!W^]GV^BPq,]r.eXE;(m&o'g?gniW!Gq%JW.$Mmn*I +D2u$M7b]hTg1Gr=_#JCM_l0=oC97f1.ROCDaKL^mnDdF:S-*#?5MJ!0nIBTP.^!\$ +=[6*0*RV,>$q4C%7Cuh/7skJ!L%8$Z$I4Ulh`5"fDCVg<1Yg_43OarfDMW7m"9f$Y +ir-U>"+L&p^ETggq10+9J&Qu$b/;Vq_pPJ/Z0EPY"d?GHVK=\T*#<-JZ"fr]b!^%gZ%N<(l +AmehtM_Y5F,p3pAFViu2?35VFs7',c>kpsG56V#H_8_=.b@J,*.ag3iBpji:9h"@q +A+^M_3_0B#"-oARATImgmU3liaLYk+C[N+1Iq^7uo%!ujm`]t@@-27Fdf4P9rU!Nt +n)'Q+n_H3thLhuOIW?UpdrY+E3g9BrD5lupaS>'Uf'LB?HW'l=N&!KLbF^5YWi,,.Wq$NO#k2>R72:>n(;;'4D)6]7?2u2?eV*R.,*G +"'d(+iL\Bb/nA?+TYg3i/45:IV+-eLPH1Xm!c"?h3aP1Em7Op"U7V!T02>?Im@aH+ +KR3(ip$!KUJ)6[jou$C9H#1m;:G'Rj+I@QObgLF%>5He0MnXLJ+n5NlHZJ#U/,.B! +AcMJDba8%$4[#/.ct)Fp5hZfK#4UO#PbYBdUb#m#-MWJoVQFZOQ[E:eGs9#N +=^D6Is$.;(i30 +WZr_cVap\qpu?/)!.Vk8M#7SWPPIG9j<[q/O)Fq"q#>R"LVEINRfC2*L:YUIjOjppmXYWrqI0E_&p.=_NX,sgOP1]O>Rm* +>jWXPXUI>\kpU?Qpc%8 +YZ`s;0CDo!#le+58t$Y+JcBk8CN0YoUU%L&1F+drjXtnV++=ED"6'9%1sZ+Q,B3ba +2apmNrmQFa$#fF&]5bbq(U;Ho%$c9Y\Q:"s%>aAaHts8IhSXLb9:W\3=!,I=_"4T$ +O9L+`Rc]()G?_i=Z'[ma?T27iM@jP"sYf0-?Ph*[YjJF8uO^Lp4=L"YIl +;^Ee\3B)]'f8%:7KCbhqDl3efFKCr3DNacbe'EIl`C/DYnZ2eAaAP&f$)3d:Bl-*R%0&Wq[>lp,$P7OsA +nA#;B\NnpO'!ja).Jo>T<(tSeP@slVc6%R$4T!OOG6Ir>D?2H(+5[,e+Sn);$Kk!+Fss58AO58!:h$2Fo*Z.N-FT<'=;pG!n_^Jc=A\pVp^^eYInrX;7mUAN`g4W(/bL&d)`G,_6N"U'X/2JeX[q,O +$l72_#QG,0"3f/UK`qF4''/5>C-N/WOtWa=ib-*i@Bi6SZ'ck4.n'AoJ>XmVXT@dL +"rkQKW.luABYk3(%1uB7aOL8+i3HBdN=(j^MF7ka/ei^BVjlN%K*o=@)hCT\m9bj0:/%IOW]f@9!*gYXo +.Q\7o3uF'3"mcUIp-l#I7fiTs!Up`&UA9:`U_G?[f71_c'CA&\,A9kP8GGt&!Rhgg +/L1??$OBA@!iCB4Y^eGEDoBHugH"IYJ5Nr976Xi;Z3DBf+6l&S[hq[K-=T_p*l7&H#>D)CM6g'B/ee@+@q9 +[O'#L(+S#Ufm"^Hkj6sM];HE-]IE$!40i5ITCJE:dbd$U$5W.*1KbgTQ1OOVgeZ5A +s6T[]O71^Q$6IUJss4'dGIk1e7"HWcq1Jt!iR*%+Z1!Sr6CF75j +`O'3OU6$UdEo^'BH)OZ_P=n?1;UZS`]b9JPopL$t;\,=r+,1X42bj5\Q[h94NUcgM +C&bh)l`5e]^O9qWG>Bj2]-(1&pXJc;+(+`d&((=McgP*-rjhh]s.>iTH"]>%%Y*bd +k4Gd.3T7,'9@NIJW\n)QeNeQp)%hB(rh.D0IjLQes12AJr#6Q-!.T'R&7fJi@0t#& +:1opGXV?-Zd.Z>S?s-T;*mlA[=ue"2.ut,i!;2usU\nklL(2Re&M*p&!=:/0/;Xj0 +^]NF3dnAX+B`^:J^g0<*dME$%i9V3l&JlRH!3$,>S(Mt^0,k@D:.5(21q'QSW"3MA +nW(XZ2Q4BI:%Otkrs(Tp,kCtRrZh(]B)uZu`.+EWP0Qj_;A//BSGXR]5N+i+s!X?> +2#icD+'3@g=4tduJIrGHLaN6u=)rk&rY#>.#9%l-&[hfAZ13r0M1GM3&4G.>hpu"F +lS!:W8fS04PK5tKIB8N!Jn1'SM9/E;+sTUn2#m4eY#^8=mJ$;gGIHZn2ku#Tk.:4n +!H!hl>qJM$Xf*JC\"a'o,(Ht4:!"]N)$/mR5p7/l0SH@ri*3e7d].P\!Bs\j$'@H" +s*gP?-.&^g7]d6MsBgEo6=?j%_Reu6&,b;Fo`rB9/3DIH$s\#$XFTmgK=_to[bJrYj8a@ZlY17P,WD6G,\*t:P)MJr\6 +!6h>u'5j9L[8-L$`][%kY(img5&<[(^KVq,%S3]o+i*!^?=I_&2RO9fgPW'Q-+;*. +9/jnbjLG)E"9&$?K55+T=7*>r]$\.Vp<]Q*Z2"o(0)39LJu@c%T&$sP\UDYNEa\!) +%PE+r^kuduXa5BXg'>*i^UP`mY)/mZA),g.KXX>q2b'T56_t#TO9/DLMq!7e7k-g? +s!,o`/-RI'd0IL`2&*DZ-JK9#de/+Js-8KBEW.=:+=u1Y'L<(VL-g.\UFo<[_$18Q +g:6>tfY[0I\[2NB;&6r:UW)m-1g86S!I5,`O=(Y<"9/H=3Uf8u\VgFVpN6rF-c1]8 +5^pZO#\4/Lj.Vp`@/4n0KJm!<2I>QNFE:t!A-!JoZ@B))o>`r@?TsSJRj4n^5SfIW +pX;.kh3%6jqAfe$_u4T[o%W; +8A0@+j-.q@?_#I;p\d,04H^Z@"oS"5i;Y`3/H>G7ir;-2pQ,1fIWRdJX]%5ugm%EQ +U%INVV`cf'E\;i4l]I +WUc,GDeV]12n'',r]U9BB@b<_mMUbT1q1hm(RJ.t6a23hVY +5T?k97OaKAi%Gs&84SZ$-:g_A.&<-U\U_.biopWe&V41M60lXQ'rP0+m:( +M14IonNR$unkC6R?]h+ufK%#rC:J.%+8,Q$;8FX*r.Mpf6@O"a!)!M3&?m>d+Er8@ +s4&U-WinhD>m-:2;q"29Bi?U@s#ZYWHeQi.6bD"G-CkG#mkdUAAsq6t%st;_r)&:lc/e*lc,;XCf9>1Ed6Lp[ +f+N6cqOD[\9/WLF7\*e^/fX!hpF=/<>loB\>R8+>f%.^`c=0=9N;?H1^W01jK_,Je +Uiq6`iD@#c";Yq^Mu_E..h>'!%RnPO`6.gEj[AuTo"C0+1c6I.:[h_9L&[P9!+Y6usC&]qXpADRDeR8;(=]W`pZPliJ=*DQG0NG3l&On%#EN5)"-I/e$6 +7sAKnGW,/We%$RVT(W*Ml0k33,F@:KT(8dhXP`JVrcSP,eQKo8A'>a:N$"KdI\9NIYqYC7$O]PC](A +;qb2Ca.a!A$mQBrc(kDdeC4]RDGe\2:"8uiZq%7_c\7`>Dk_]qL\fcubR\Scq>ANs +IpL:rk+VV.?,e[n.3hh3>(WZ+dfB:p$V +^Gj"/?MoiY&&`8C%Q&6Xs1QZJpuX4Kptst_(3S#$,5jaJr(edC$-8)[YQ-4'\arH^ +peQous-s%:/L%D!&:sk2o,iJnfLINmXZXrfklCS27NsJD5JVcAM*KWDLoZY6UhLOA +)u)enfF.0rn3Z:DZrs&Gpq33a*\).=(;#;nghO91R"l#4<_:4VMqUh_M(YilmGqCA +/[1&+'J+cV!"32^3*Ok2i%e?r=/u6rr@Nk5AAEr1>YA$ +/q@EP8B4i:2l6b^"VYLpNu1W +SpL1:O9.[*!Q%P[E<6*eH)7]i`8)!5f-'kS'eXWsV,%6`f +\8d.p3?MMAgfE?Hhr*m,4kdYkMg"=cl_5'LKkH$@pP`]H17e9[QBWK[c$Ahuma&(@ +prX&(Is9)ahqbs3f>gt*ErClf*F8?nIRjTB5J2.+IBsm=9bj6;kM80G^*-['32AMF +Wnn\r3>AFsJK5^==3s:3_kHK8.,_Wd,Qsh#:ZtF?&*1Lj#U"dg2T=]6 +_;+)c!1a;JGTaI!SH[XT`F&^k[A__pObTrKjOslI5FMO8X#sckKO%RiG;VF\n^b\W +%3pGDO`3JQ+()!b`EbaR_#MeIbuhJn/6!]!_0;r:GfmoV;:o"aG3&+20&GbtHn:A' +1-5HPGh0bg),JST!Y9d#0^[E.XP[..Mca>6=G7>?r!J%fi2bkA$0A!op+$pNjMHs +),IeG596MJ[1uR>`]H,VI!g;;5L-,*qu1E%!\D,eTOm"E\a6rKQ_4u<"*pMF>k%JL +ErUB"p2@*,IqaDP;d3/g+3V1=RrufB4#<8elWS%@?E()5eQoTQ$NI.7+o#,=((XH2 +OPh4gr?m2HRfE"jIQIIPoq7eeW;`P8(/P4r+#d2[)(U7;UXnujRps6l1hj"XZn!J> +jeGlO>b4(@4?i/Z,TSheM!uss;WJ2i=_(VIdRnJ2W!J&]TNC@FiKHY0>enq,1UhFU +s,[Y"qAPaA7[X2#e5H(]WOo5E5U/"s2$=@KAIZkG.;RL4=4NO:6W-s1M,4q]HOL +"8R^UKHPEDE2Z>npeM$"'%%adaA9Pa8+6tm4_cMWa/eA.c!gbSQpVY,$VEnTBT5(c +])N`4^QdkF,k[8Q",DZ^*6`nU='$!rImHG3IX>!jG8mHHN7UGW[>"6t4h@8/33c?p +8LHF\,Trj_Ct?PV/)UjE=83:E*Y=tte`(-7H`UdE=66BqIO`bIjPN#G4ZD'+D5rY1 +l\';%Ztj9L)_>AZe.rM##!Ir0=+e[sB/*HQ[1u&]AUV7TmOirmCIq/l!O+=Ga/Z(YfqtHsqi(u$Ep&r!VVO>& +o&qF_;qkD-m*BqP=O&R\0DNeoH%3JLd2u+9IH0Cof5=ablLEt#5ITo+LR(02Dq3s? +CG'JWE;-6h4/R6M7O?YV8s'8[.PU\%&oiT7"[jgGrsse+5l3pFj=])MpchXO8Q\6J +s'%2TBD_fNs)XA!O/bH'PQbIoQTto91i`%h+0!R`MW_O8ili!;!0Pl27e8k2W8QjD1Bs$qYuO[)Xn?QJc>#2SAAV^PqcqlL$kCeM7k([ptVh%6*+ +3gt-aGilo5dK#$JmnS@P?g99TVnI7)rJfk%O9iLI>NU:AB7>XLJZOq$aS>OjP`O+q +4ugf5k>pkd2c>,c7_Q@)qX&IkeS4;]6,nte.@4$`pU;mFP\8jgB'rho:$,@Q<6@s5 +-3ci`N0f&-nUEY):gG@o'^`VQc](hKI +IiSH0jA5PHid\EM;?1]%<=869AY3ucs0tWTAh5%/+o%,Q.gYsK*-(t=Tp`A\Gi=?u +bV#Bh@3p(9OX1eW[PBOj\-qMsk_p@d#t]b:P/)Uhf8#*:]$[2OA-"ec#M7JF-7OJB +Z"1'Ki/`$s!;mK^n\o$1H>c/B+,QVZ9#(MtaY-q[+T1E]crBL_r252IAGH0`rtCMZ +C&SgqLVK5G4OO<-5PZD9O-U(NYH\'gpo!pPcm0TI&e-^VM)6e.@KG10+UHq]B`@b$_pI"e1$6bcqDH2q!YK+ +q/=b+ri9&cJNchQGB$i6ED((eF%^LSCGEUhI]EOd5<`0gQ-iQEBpo(i:dCO0CSV`J +bbCHWL1OC+ITK0.jrk&PW"OXi"%%FN!5i/O;N!)0M1&Led#21!WQ0s4#S9-oKUOfY +Zig5Wq)`0mG&C3o'=<2!Ir;QAo9^.A?n0kj&T.g]#l.YOXsUcWcD7WbegOK"Z$(%\:tO^+ +rcu,2Z`^d8T+8*n)e2(?RIR%Xba4?cs71EsDboN0EBoj0o:-9-R44YiQ5lK.brYl`L6rM#'rJ"[P=e!=C`; +[uPbQa1QP)^)B*;W%.#5n9Z[5Y&^_Pb[V9CVu=%k[_W^',UK?gtPRo6u0Oo@5(R +5M;HEGp`,f,jRpDmS82e8H0<8gJWIu@YDdj!oP?X#$t4j<*`2U&!9(;q5ri!4+E\& +3:[m7Q%-*6L#tTb35L))%fR2rs'q(HEG+BdULFRGg0"`NEg0?ZOlAGU*2O=oZ7-"k +>%ra^g;4L[m?CE#9>MFk:ioY#Ld;r'JK]#0,m4UKB1b74GdIW>$D;5c*i_^r4,U"Perps(FiT*,+?RS/22F?(9'eCVYA&b:Dns-(J(--._D&bo]nr(Tcakf[P_ImX/7k!)^]'+qFO)8T\1/-5kGeL_CBIgH\eJ%.qKD++Lr5c0(H"0F)>d3KWH +'S:p?:-c_jlSnnI05)+jC_?gL0t2Pc;a-$A0C\)%Vf]j/"-HSj&Jt;YfP*?Ztlr-JQ6I:A9Vhl8">.*16Jj,h29-iqCn2):Z0e?1J#>R6n-=IH*>L;I/IDCWr]pJ= +Rtj8@Ad:WS#R6CpN;%3.m?ruWNJ9O1!Wnr?LUG]LW&!Qd+U.)u(\RENKs;fr=)139 +I@8?=RQnKGC7hgZm&lCo)IhRC&SqJXJX1:o/!,8!eO8=;[++qHea?k"?!V3"Ij^dL +J:Cp36!*0:mY]P#MH)fse]NbLTYB^A"1;ppJuK3BK(^f3)%u +"B-pIU>Z(IrQJ?'V7q+GqO=Cgs)Q`h^Y^pK]lT)sn(a3&f/rR&FF)@':ZhWPVT$<[ +I`_2:Q9W&qocr3MgbW/Ra#*\S3:D*VHSeP'3YEu*)H%ObFp,D\W#aA:^cTDX!BV)K^glL& +m"0ad&p0Fon@ok!5M6SQr-+X6nD@l^#WSlH"hU[Z=lWfO]* +,G*&>7hPBYqn3??]A2d)F4m?a1XBqeZKg$H%EftCA-$bs7Do6M$TLA:IK+E_,I?W1 +c)!S447/mCBtlj#p;ZXH^YG"Np8PECcXqQH0&s8GC&b\)mBcD`O9N:@46)m$;$ug4 +!U;VuWE1FFT*8?C>$.*rWCtM#B1f.uL")+teG)/KQU9@BGYKmK&W]RGN$Mj21Tp5> +P_M#>TeMB%.>EPmFJ^`Sl0*0(oQ%t.6L&k7j)d*Kh>uoINI!3N]t=ZL!68!o4i.'g +Ng,$"6kc9/GpN.=bDQr3+(+Tp@nXPQCO\R_Yje3ioA@2@qc?9Dcj6;h=nds=h!/7N +a0]MhE/CK&Z(ZUiY)oD<=]JBgpogYd`?0p6;9s^nrj_*Oq2_\,M9np;/:3^3E7RK5EABAN.'SoPCT +J>QVK;bl!kc3PaJV:Yr^;(])(clf:k#sE]M`?\Q9NMHQV-+[]Q!iYCeMZ5s_2=W]K +W,j$S;?Hm1>DigfPoUiU_?,8crQ<(?0@KcNQ+R^elH7FQr(lb\8`IMCqtNt=rsA`& +gV28@qO4r$nbZMn.L=s:`#rGL&WgAS.OENNac`L>d^2WgpE]60A,[,L#jJjl^b@^7 +#lY/.-bKf5PZL'50;%aU6"Ab,&A@6Cq!Y[bJcAKhJ((WP9]I*jnb`1e4\C0)2nus0 +h5"*G9/H`Km@\!7:oYV#):YM&6((:r250MGB%<+Il6[LXI_KNHD](qf7]fjV9nu26!(-`/Uj=^1BRo#jDR'L\^LrINs#/]I +3UP\*)UP'->0tPh6eeM=F&N0F-Mf&ZHH%/"O92rZoi>"NEO\?1.9!2kI=6Kns58u( +2_6g*AAE](XjU^_!Xf8pCB7/FJ%uX]BuJa=o/&sBNF?I0aO(R'ebP.]*ds;@;f3Y< +%X/3a+k*MGkcS+H?G=!F^8drE6Wo,9Yd-'_BAemm3H)2gKZ3l/S!,:Ok[gs+m/2 +b^W.aYQ`rrO\Scp/IQnt_eqJcs#'^%p?f7:r/OB[!RBDL)]c=O[MAF8#RTu9'`Dpke0kZ59;.mj)Dj6"=%[Y9(!$]b&F +T*XSLNTJqYT(V#8`Y2Q4J$]khaFA7foCk1Ss68+![n`j7Iht]X5!P\0ff<#Dq<1fC +4""-[^4H-u+mQNuWE3]1(F+n%m*oM`lG9on43FNAf[SccbaAX9?n19T]qB?Q^rt`F +$2E!D!cTaln7P(&TLB[;d!%B&(hSZaA5[l,IhjoUOP79tg/igm)VeXph;(^eC-a'n +IPsXnC+&\P_lU>.;]hQBHuYaR!4`[7J,@kR_Z0WuZK-r@(OpONhRdtar6pLJY?dLf +infWl[Ue5ZSo]6di;r8=R@tr+c6\cGi'._NWtGB^2'jA^%9loWU#Af=+5fr9Ns9Xa +(OHt0PND-sj8,tAIjsN0q1Ma&m]7Q/T0]KM[E5P;pMc[16=+LN. +MHR,&Hi]1V,R%+)q1M1#ArsuRP)':*6hL5L1!.,O&glD.ogk&FS_>5@=i/h@Q +^je*hR9ZaKS20)KGpk-IJ_eMNJ5LOrkGk4 +Fk=#G6<:nJDsr,"4U8&`Ln4$"ZMg6l`O/jCT7d.h$fG#sngb9andGNZD\JmuaK4qH +Rju*Ub)k%JL3#IdOR(0dT!%WI=rm^V@!'4*)9[Q>?X[8 +"sDj"KD@sK9JIWW?S9P''ET%8aBpI=?Br8b[mQ>1.Er+U"b(/"jqn6S3Z^r;c.$\f +eEmeJkicD358DDG&Yn@teRScWA/Bi5)X +O7?jQH.0[$OoFKomRFh!VF+Ff;&1TlClp-4cd1Zm;aLCL;2/&i;M[+dnl>^6WXDN> +<1SX=m80J@bAm-?Jb^md;Nn-5-n@'/fK)>(.u$=dlCU8kN9.p6;G +XH4?$BMDOsC's#lE,oFlq9d(WE%*k6>O#q)feD"FZpR-)cq8b%(q[&']t'@!rZTES +%m#2fb2r6DDI8pepsiOADk?V#qOH0(B7(g05't6Bj+kuXk^t,Zs)T=Xoj&_S;0X8^ +r,fm;US0%?BZL"Ka*NorB*)@4#_K`jeh0V@2@K:EBA&da&[5fh>-tTNr4or_r1>]Z +:7jGBYEqK%^&]rtTT9r:>lZ!s_Z.:i$t.Wc\lATN0IBs/ef_'*<3%1[hSHLAA:so! +=p3f6PEH+K2oo,-AX62Y$Pp`=2$"LdNEoA@A0O-7>Z&"&/V>fK^T*:5#2M:/nZe,/!cU[gpcB#J& +"M?_WqW8R&GB\AIJ'd<'8)6?mmf(m2oMP+E-U#1)md\6u*ZlKWnBjGB#?I4h,(1U.jOrc*g%PY0TED@0uP\sRH8A`m/C'5+B`T(%>'1%0Bq%2l)I#Q9> +?OBUtp^b/96K=^bUIBiRGAP>>Rgd4N,I-W-f`"S2_(tA^]4SqL)o8cQ[upZu +O7_*[nlI1fs*8Fu&*H'(]&7r4;W++h!I$.$&cVh.1&k#jo)$TBn)*O&psnMdcVD'B +)t86jk.] +D(_22Te(_!J96;U(tRaX.WTh]RGFTZ4$NhY!-:5Z!(.+r=o_PcC&u7Q8SA6+QIK*$ +8GJgUs6og!4L)UE#p;.#2rO[*e0P3A">F$u;cX,hbm]C-*I!(.O2$/3Ih9>(rdW6. +Lu4TiM[1FtHH(9^L-Ma<^Ln:3R/DL1gSu*DrrMURaPYs+_-5NQSA25ha0hQ"5Yqn% +OG+mO,1?Sf+67@*1',!B`NDeud:\(/U"Tb=Y@he`iX5HE;kDg4CIKg?Q9gA:!9cbP +9`/Sc9S%`5iF[)>f`5H0M',G@bXU_2jXN[O.NGiM>:7YBMcS(u5ks4^amTr,+Tsep +?C$/cJeZRNda28Cg_?kEh&o"mMui.B8k#UGS_^-gn*;A>YuONk29ZT?02Dk^:5QF) +9%K)@R?9*t9*"/Wk/RW:!Fpb\"mE8AG3UUMRSpPf==haKs7(Jc\j2W8Q6?2X,qSQ# +?Bl$5d%Moh4.3KFGTg6!'raW#!]<30^Je!=OKu9gHjpVIT-nJfn_kR9H#c8\>YT*8 +k&m#]f"kdKhK:/7`AcVIF)d95]hg.$Yirrhci +(Dj=5m)D4O6N6b5e%ZuIrq_J2?_;QAp7g=9Qb:&Ekl'mIhVXprrT4FDY9"5LoT]C5 +*t7qndGXa/KtafRpndB:cYX!#Z4M>"1?%kM_BVDS-O,b%n"h&h:-$[d%ZS7V>;!8_ +bN>>>k+"jJ@c!7ObaHmFrkOaIboqQbpd+ePHMWGX2QUNk6[Hb&JMChKY*8mcs3^4A +\eJTs(g6p/5-;FA'DrC65O&=f;\@a +ao\[hhk2CE.<;n"i-OE8*iBO<:+SkbVON%CDF(*^A'`gj%oCe>mHSN?'=35h+Hd3,9c\uFL(/Cm/NTk*WP]\eh +h?*2iWGSJF"@W#,rTV3qe>t5pX=HdPri`LtCqq9g^*]hQQ3oJ"d2V?gN=AQp"Af4X +d>8/(L&WrX!dj! +?'j0t_&bG"O*B4$%Q5u#E^B]ZB-]QNV;(0EgZK;5(6n=D2CF7oJ4P+!55: +mVD'leXMO5#&Tc3_'](5&7gU8R1a52<0j5@6N*M[g-O7(e8bF`0%n%k.);(#l +d?G6japR->qoIE-*>Qk0mEW=Cc6mInMkeot9A$kgZ&ap,5cbV/a@J):4[Z[lII)YgL7_>cV8p8@f*hE;SLk(3#'pI;m$,Q-h9iha2= +-`7Ru+%n^R9KPG"HDELk:iFW#>M_7G3\P0Q3H*<.2fHg/1ZuuMZXu;JAk\$&c;jI' +jUVdcBuAZc@4cn(9*5C*Zp#M4?Vg,Fd^osq/fau()AiJS>;ZhBNbLGJ'E/74RroSY +q00p/%J]'?r3j+7[3Pk$8?s@TFilh[WfTK@bgn1`2=`l3]Dt`YClKo3V+OU"QuWX' +U83\^1oW;Lr%Af*K)^OV=91?YapD-hJdkt7$dY_Pi%Z_83%K4-WnhomF5r+[NMlZD +09%qsT5o4&IcK2;?/FE)MUhh=])M*T7aV.4pN_.,d.?]2h#G%AjUYS;^Rgi?p^W0Y89<`6W]`_KC3qb?c[[8;bUI\s/4\%4Ql=b;b6ns0*^J9&,>mo!?[s'qt"7"pNF(Zr]f0+ +%fY@^$()?\oX"fACAe=tBWF6+QTkN\s-(F@1atWg+c:cds(c9H&7b_K!b2.=B[%Z[ +rAX>p$#auAX%/HZrZB]+&H3b+(6*DMipp`7q'S32!G$XP+ZLR2ZEjj&rX3#.Mqr1@ +'E\a!7DPFq#[.=dJ-j:,>s8i$4lX[<+heN7`["^fD'PTb!@62P@3r%h/.o8i`N0!A +.ZUP^M6A@&W%CXbr*Ng!:qmbHi.%9X\lp$2IqSu3E^1iY,%/+aM2QoQn\;"_oDeC] +cQ`DuO%V[U_>bVWKD+'#R@K][Gd^\O'#hVYQ::QjML=BOr2''H-j%s[?W*NZGR+&J +o05@DK]Q@ps)'h1YesD-g7q4]0P%:$s(fIn[2"_U$65#W$\ka.r?el_5SS>f*=H0& +X(H'G3UfO=,m2]\8;,YU\KHC#q,NT*Xe\38@iJq^aSCu#PI/[f"j+*_'dZ#cP&<]C +4F:))4n+'P/MHCS'm31?m@mjT0PC&0EC$$rP[Thg=qt1j"[AL+Wks\ocD!U/j`k/2 +lq$I%"Fo15)#V-i7_3A9[X1c`B3qii6*YK+[W>3KI7CY--!M3"9b$WIHX$BY)K"Ut +UV9^(.[3PH,gn5Cq27%>dWE#N6Rb`^ecrJ`FXd+CW%@:3+?Gd(^aLkc9+#fNW;MQQ +]L#b*Y,&Ui>EdTBcE&OjoQ/PI-+egVT4mTc-\)OV_o%C`6-' +c03&h>!e8C"?e[$GA5:i\LIP[(-UR-iq]&a*l$Omp4qJ,J)')XiBC&qO*f[>hrX_& +(@k\YkX(T%mFZOjp9Y\%J%*Hk.1]rt#O0m,-u@ZDlZL'\QuaXlW<5L6]u:GeG[b8< +K\[ZT\m\&E9^RK^.;6O6MLQX9L"Nu-T$]'-D>g&KVl^E9'I;bJ()b]VF:UiUAn,/itFf35p;B4'>(1F9PQWM06M^&DlBOPmC]&s7A +J'bGbP4I,Q@iq^K^6)u?@S']HQM>i7?dC/-C!Q^le59^'?$d(Vjb$FW0h$^5'Zgo]sntcZ7oItd% +^O;cQfl!qZ1mVo67<3[F<0f,$XU5:n8!9m8U$74rObZC>b!Ml9!74A9rrQJb7[U(h +j=>NO8H(3o]_\/\MR5E&XYMi/,W!"c/Zb$OP",m2TEXF)#6KUdTL)f^^nq?'(-i4+ +V$&ZOqJt"CK)^2?8PAHHqc'@IG3mp"pA/CfGPeY+mO*MT0]hW,c9GRhGl%hs#h\_L +/(Xl)Y5>Cbk6hl@lPZ<:YP3P'b^&*`ZCQMUnQ",IAo3nab;9#p%cWY;:D"=\(L;oMC,n5G,KO:##5!+"A/cU:<,n4T`+%?O.[(%P=X +,"825-knL'Y"0c70:'PYqHk)akA:j=LPsnR`@W7=jq't&YdQ.%,REk)KT)lsQAr`X +Xo9]ZIEi7I?QAaP1$/,W]_h"fY&A/sI:oVPHeUIVIeuUmG<7_^G+8*XMqI>/fP?,l +9/c-^/Tq8Bl!:"t/!"E,0KND$qIVe9!m#+jU?Z:Ms,*V\12:lE@T,iB1-Z]V +,WYd0>Eo?h+p+_/rSQ>E+T0Q$]0#r,Q[gjl@=+lU/IN^!mtcL@Jmc:l*>&;N?QT?'uq;F(HP=]@`rY]WF?Gu?[Z+rD9W%/.,Og$1,c.YFllU!/]NEf"QU$0Qj= +eWMbkN2;a"kiK@bp(:JJ/cbHDo!7?sA:@s.2Qn:1[47rSQsQ@'?d+DQ:CYW[K]^]M +ZD`>ba[+TOIQ<7@ga%@lT,LOViZOnYmU*m(DU6t6c?JNR,]Dh,O/bca]1C8-A&$ke +qY^HC?QYs=qba80cFNjtamJdZr8Q3bV`QhgH94;a%j-#(/'5]08%=H=P\%V)d/Z +k+(=2E-ZNH'mOA*_cr%`2eK9q8'r[]RqCaBWtObsK@AK\Zikb-"5B#ibm%X0)mCq' +ZX`&K-VCS3dsqs?3.NguJeL`t%47&=;>U5?G&O\>K;.*cdSS:l"2/U)FJ.+V31'M? +SK98&3aGIocq8RekjlBTFAkt\c0"ri!N1.]dH\Uk\!$`r)uJeAdO3(+T:^*!q@T'4 +cYWOL8W^^=X\7M$Qi)sF0DKpCcKDN(^&N)hpVXm4n*T[rqVU93m4+/'JRaMos%u],aFDuK#Y`qZU%elgch/(2A,`T_(#+Jc$+:@G!:[::Y\\*)l#L_, +k+7aTXqHP@8Rif0^lU$m2$/GFXROQ;)sn@FO!5kZiJ2F +"SEL#rWJ5e=S"b"D&"02.Xtq,6O\I##iN0R+omu0Rf@qGqVMkcp\:A2n390C&4o.B +'MnM]+oaEe7.F(J*t[B0(Y?"3lf9^;bTi3-jUqKY2&6c&#G_f_^[s`&!I^t)io]c; +FaCfo7JMAbP!<\3,lBm`k!5EB9'Y]ce_f42Wf5Ko8Pf>?Hq%N +$WSqr!1b@NmhP$F#E7a9.F<;#rL3XJF^C\$[gRdQS)<+#1i+\7kc($hDV7.4L-JN"KD,$m +W<&+\lH1=B-X%=N.#A!r^]7`Y:Y0\RC0R]GHir:[NfS-#j0I$ +ct/2P1Vio%g6?SOlBjVD7*-9H;Fa\X_s03h:m3T3:@;1?>_i1`kQjR%&'CU0n(->Z6h_NVkKo?ac! +3%5u"*/6jTQV-Kt),2d-M4Ca"DCO7)pR0$)r7@N5-3K_2"i +j])]^lQ5V8L7R4kUASG^r24Dm^[5Lf^>aT%_g?Cu7([fBMhlr:Sm#^j4Q"$:6a`)XpldWf(q4BhXd +rtEHIhmJ9FLFW&A!>tqbOa*e8=PZD +!(/r=s!.=U:"\)3pqVBjX,uiNWr*)o[LAFPk6^;R4%:]DN`I8lb'+7oC`b3`#h[R, +"t9OX^l5f(,%*fk,+q_tJbB-O;Cu-c7iMQ4!]2VG%'_s!hbNFY\IOrcd].bM_[RF>8 +f9GrA/,F@Q_hOTl@gsU]8TVeVBpBQGrNr898maZRXWH(pN%%CijMaJaG/#jU(m*hg +.LUtfX7U2u\i"b]qSGNsaf)=VZF>:3\95ujE;PRsMLUETJHMB!_#@oihBgFL3Aodq +fhm=#bY\@%`DGOOfg@MLNh>b$/('L2<(LZW:0`4=P_Z+IP +=\H&r.7r#WjYnIlR]4/)r,=#i)LsGj5W;!U`?O#^s%=+$f)LF'"oK2<2d4iT-"r&L ++;,6,U?104"YW?iMs;!mlB-g\.7+e-&-7?;beYZD?B92T5LJrkjIk,D6''3Am`u@nnsPYT*PnN@sS/)1,CS^8+65Y#0:G<"a!^o#`UeLHMg\#:s5h)FXi%+"a0f7Xg>=bg +\S&P^fKE9GCU@QkE#D'Z[K1%Eb*T_4(G+q+cMVZFM?%;jH6d4*/HGc1!;KpSaH5!# +"P(W,C+?]]"qf3[R>a'%4*LU&hW+XSFo-Y.?NJMWS*4_6$r1RF!RUtA5nGDLaC.@; +TMJ]p15Q9Upj`83+'9ojPkG#'VEbWfYC.Y8>^uNECGg`H/-[(YaiF?5XNO"sIm)!6 +s7PqQrbj>1pojZ5:W2Tni+]!9r_Zl\eO39Q2cB.HF0Hm@K%-c +j/f)6@'6hU[\EjI?q+KbV6-7aQU:n7.S<1kllcOZ=BRVbArgU5LUDga!jqdeHZK:9 +fMk&k?')k,"UcsU(s5:0uJs0_->n3>G&12/ej+&E62$=Tr$J.^=/$jX$Oa=+"qi +Jl6""aE.f7NL3OP_umH@LcH#4P!l6P'(ocX-WQ[KTo?$pH`b"tdX,0%u=\ +k+X#/?I,(jB2.VmUU^=OFI,%@(dF]i +)HB`5eB-N]Wa1m+63N("H2-Ol]0LFfGm[d(a?G6T$E#Fr.+52>Q* +BTLL(DU0^-4>-?>?QF6:Aq4o=!.1jb]AA7tY`PIFTKK-M]HG1rMm6pjf]?_fR<%lG +FDQ:\2BRT("C[3sBLR/m4LU0*m@dtc)!&@=o);U)G7(c9)4Q])VP;/L_-!5NVJMkZ!Em:JniO +eE+Ur+!;((k-H+>5r0/(-o`k'M+OSd%mM:"[K\MWmd4j)%Q[. +jSoVfZbZUZ-bs$Y*:SG"aB=jfq`fS[N)+ohm&>-u?g'/W/,t`hS\n"kK_G.8<:?-<3`45@V\d(P)VVC/tD2TKErDmZ(M8N#XBsA+%:^oQ(_rebN`f9 +\n#4iSR4m&YG&@)iC?%1U,*l0#a#(M#kJ`#YH+GLf-`L6cl*kf9hc$M59YM_HQ'@F +Q&#Eb#LIZggan6]BEWR^RJ\u(In=G,4\NR#q%MO^s#V!J$*+/]s4;brjcD^cJ*u#t +/&91h@"*>mla)%Cr"K'r2\cB^"NVM7!;IWFJnjnns2(ek?htk%rrF,YrX>QmT6V[4 +Hp[gBh/)"Us4[kersr9@qlBsL(3Zmh\Aqij1DO,QkfZ[7!U=.-&cq-Mb.gkX"S2$4 +PkXGm62j^Y:NSKg=ci@?XNg@FQ:5".Y[D>$Ch[$/29g>%bcXCMUo"2k?p\-[W\,/h +(QE0f7s/WuLk#M3d=AK4`*>%E>rsjh6L=UQs6j5imp"9]=O\BsdN)NB*nm>5r4d`f +@HR^_s72D5nGh];)rZ:1YsbKUotNhZ725cP>`]$.GcCHUV!,/mip$d.`ZL$*k*hNh +$>*Q]Z]\$5c:uMXMbQnUVo8m<@fqm'r.3p2^SGP8`$k)-81u**'-\U:':qdXhXjA5 +oGG'S"IFGGP;:ahYEHPX96j?n?0FcWs.?F7s.>mYI!s#rVi,FCf=Y1&\?F][=RZ8o +YHEe6Qgb8G.=eRW7I.f,]tQSsobB67mp@1+q;"@*2g5&1Yj?jJ3on\NkefOi4$"6P +qXdoBCKcoZV.P^#o1$,jj1:I(D=Xb\DTNHrXWq^RBCmfTe/iDd/o16*F6FB$-PVYD_W?!_u%_[pg9_j3rJH&!3%4b +O:)_tr&?Di"sO$Io1O>u,$q!KQn7DJWd(cQIZ.8DeLuq"Q\A3c,.;9>G3abjFG/F^ +$9i&33,\G8`9)KFa5b9Jf*aa+/%sRk^"A<^m"I(_McT#1(?\clfIK#L72>X0P]F=Z +q/dMM=9c2u);@s+%O@831U)K\fR<#:O\G*I$QpP9`XZWiHdf4dFOF']drZj\,R>>! +_WISs]rC'kV(Uq8aO_-Y.q\-"W;J$J5;-OHfLbr#!;WrGPV8&l"+NT5h`U2rmGf>h +%6lt[nY^H,pa=:H'868?n,'#q^HU\3Q2lqls7)2kDg&_+[,?=SS8tLaor-PRS&/s8 +h5@m9;c4>T.#I#i,6^MUI!.WESL@lEq1VD#&&D;ms/AU8,%+^TO6"&:s7pU4a1h]! +lW%TjCNJ))T0Hp:5fJ3`)tuQ9jM-N3?bCNkFPB@>Nj4JtGg4!AZ=a+NikTXf464WrO.3Sq/Q")J]>^jVC#Jf1E@5.;!k`i$+Lk#:-)g4G_'3' ++Fh9.+FeuiMZ7&[+"(>jQR8e&r-U./nimt?^$sFb^Ga%3r0S]jKPUS-je#6Q?dY+Q +T(W*'hhusOs#f*:3<6eq;BU;dbcf,R`P7Eje<)n)c3="Wra1q?:XMRSnQl1TF!:$; +%XZ":m^Jq.G^D%+RRMQ.3k<5(.lD$53n;1iaq@&?<`KG\q<:DM]eq%Sg*nhF^BMgl +c3NrZmV3!bAW*[MLk%Kq4[d)mXp-'M2$,h3HsBl6VCM[:E;p#mqVkpnl):G!ZDVSX +TQOt-LPGi8;#G,&051amhm[F3DlXHo!;sTUH.DqaV]GbTl0kYI8mJl-CmZS4'/oY37C]0^\`Lajf?g$ufp(>#neNkR +5rAuHs!$n259F/'s,RsSXutaPo"<`"$Q^fOHEOon.fW_Vq?&,I?3s_Z3G:8Rn>5)* +chlLC)XS%5p.m':lB1`:L$<;YVOAA'r/d#"])F<`o#kbM8JnSCu:ZmJ)H6isAV7$EMEr,b:);#DhDX7\[[!KdNmH'2Eu.$Gu+"A(B;m*d1#u5J,nO-X>U*j#M5Vk>18(<\F3@4^cS=T]_dD +J%4G8bnH[U4QZtMPWq:in_:b@,-Y]mSB_X`3,*ToX7;_CH]BqmWN3n3A\k+8?t"&. +J/ulbN8kWuEXJ!)75R:b*F8YK`fO6V4o^9so+N6AIR`luLOWnIhg_d"Ihq#C*Yf(\ +5A\OGs,WQj[O?[FYi(P0-Ja)HPP;S,mC>l^Q]UFY, +SV-%N!c5:]buoq$3t'($3?su/+p,?okuL)b-T=KG``0?"A5*J(1IKMWk;=NoeUmf5 +K`@Om%X_rk95:Isc6jaYhpTATD.U/bVfFb24C/5aT?6,iEOkS^'_&l!QUrHSTKNrj +D#mhUr+U\S0pNUQX<7EAn,eF31IQBM!;IXfJ;=2)ce##`lI(Io5WenqSF;j@F)"bS +O.OlNi.$hQQll5ZDp34B?eJ]B!6hW[Acm8b'G6uM:gJX^k.gf.O75@PraZ\]580qi6>oi#b6b-?X+onhqMZI;? +[oZd"9R\<_;#*&r+nN/pftmTVXkc*I9,AEB^MIam1,gk.2F7PjiQD8N`YtAI8Rpk8 +TkqqL3iXh7+9`&NO'/N\H5brOg[N0h.JSD=]F#h_(&scrLUJ?*+'dCc?W-?!dLlIc +RZbsWMl)7ii;^Wb#4TC(ZUD"=;or/LBnf0]lparf]j@W"e5#Yg*]N8*D4'CH;tLNsKM7ejoTHNMT'*^ECI!`C-a_OVV5 +RMk3dQ:MX!OLiTUM_3R9"a8a_lMl(;OoBBWEnC?C4H]Y^JH(&5s*+q0J,?VRjT!g5 +m_"EB.=d#cJ!CuP&-:ZQjTWW^^E2qAgOn@t5NML<*T#E1rLXo7J"S`DY8R813i,@d +=(FuQZc:Os8Tj7=q= +L<'s$7'C'`9buP.Nn&t"[,OJ=,>(c?irPu);'D[V7!,\a%e<-AGIGtFXi[*qA55Ki +HYii"WK%5;C?""6inE_Y3R(.n2CcWJr&c-PR=q`)i@&Amo2TND!(!8Yd/]Fe)'T(Y +%n43s(G"IaLtDPX(*n_38]8P=_54.k$(A;9,:CXVMsM3KO,-/r&X\`:cRHbu<+0"] +!IM1;a#Z@Z=^@Z`Ek^0nK)5M$!UR4&nf,K2\&M\0)oLMO-,2B,HFnt1RYcDjj`F5X +I$s#8WF`udAs]Ea8EUpHi*8EgVA=Nk)LJOuH_d0RZInDD_1i%0)n+4c(1XW&-u98+ +4-WBaXPW&r`9AQ]jc'<-]kQiN/L6GX^Q4Jj52lFqUTUQLIraRFkM:Ym:V#L]cT;s5 +.q$a"cU7m='-j.l0DbV0SB$tWHi3K2gL'FDpu%hBRlakmbYp-T4$D#uiFn#$%[m2f +]8O'?5Zd3F/G5,A#79',h#'m:_Uu<6"EY#F$(YGG&=(Zij\0dqB-gJV73);`jR4Mr +4'FF%g[6]ng4VKEJ"%j-p"Eba-Frj(q)-1eR>=\BpC(MFZb +7LBpf!i2:\.0_D'gWid,18QQr;%&$.7,p+#gZt#,CV]k)#gbbQrg$"l3$3q0gmm0J +6>LG7[rIf&g_:;`T:[_MAdSYP!U8p$#U!#Qc2[!3!-(YdW&YmmIk%#npu_U!njh3+ +48_lBJ0P6KWuckCL]b[jHSlK8A9A6_MCGJq("Hlu@*@gY!XdOL!9&D]5>h=h)RBWBq!c=15SX6n4feQ("7tQ3')oq0J&;4^rt!gV +#&L.1p`EZ>ZPoBo!9SLlq&u&Pa#`Mms"AocEVY7\n+tqq%P1i<(#5l4ie@n$gg-1-Qnfm3\i-uFA>DW.5Y^]fbhZ%Y, +r3]IRC:h^OAj_\USVp7J=HgL%^m+C,BN&@Z;qT7R2_fT;N:o;YCCM;Qb5)H[9kdKo ++ilmAJ/+Q=&t)2uVoAoe7/>lO8,ePVp&*]LblroR0\=fKcKmNLPD5K`8TRV%__TYC +/*Y&%kr2g;+!)PQEnoo[e':0@jgt#ia3$:6Q0Va,2gnL*]]9=s?`j?8$QpX$jg@>t +#k#@j#HYT%J*0t,+"hniHd3St%lrnIa2=Keme2*YdI7!$fSTU[!uBTkIEIKmAR(O] +,!_8OULnC%:j6.'%X1F6m,.Hf;#&`&d'LX+B1IR!$jSsW'QA(8jSlltK=8#?SY7jZ +LfY;4WUWC:e6@@TXmg./q?)T[_t5iY03K?%Uk@?8Sr$09uEQdAcscrqG# +1fq[T=2m'ua0n8:1r\MS:PQq5j=ckJP]DLV!M63AlB1ctFMdqo2''L2T.H0Q.-*eIJJn[4XC< +c7^SOY04r58[2nSf[E1R1)IRTNh(*_4OHE?XI=18bQT0j\q"WSCafD>cG5GS@ZRup +SlRnc@]-k%MS5(CoDdK@&leiQru]R5gck1`Nr$NOj`oXI5N2WXIWn'Vlu/ +#o=:GF4kQ'U*OV2%K$b0^T<$dW*k/R2-$]4rtFuG_Z.Vf(.<\@PPoS&BA(6TL]@7Z +=[Rr!iQb]t5MH%Ak'nfY2\285_k0h^s.T3P1g>HS*cVmL8-+MHnGe%GjSup2fGLV%<[VVL6#[?pP1I.o!L2d_Pl'r)=[a7G5tJ[5#Jsr+6kN$/Oms;FbT,7HSNk +:Mac:Ci2ojZqm((\Wtao7VciGs"cXI'WCiTkPl_A&HMn%!`'Cj$3(ZE5[+Hf+6:1; +s53K6;?*g7s*+DP%X58[K`AU5bk:Pif;J60VLZ_np]:.__#JZLT-g^HofpN)fpq4Z +^Uf[>X@<2mIA6'Ur3'cb_W>,)rg2+M!0@9!b'ZKD*ksh(n3=YY'DK6N(&oN7q4g(V +i;\r-6:;WZ0L#R+K@]s-qS1okg2B7np(9gfib.9ocu+>\nkT[D7Z=`gnrF<2#lcR4 +s,6o.78)sZWHkYQC#7>rDeTF+7M(#8i8H*4gd\YZpGT['>u,KZ#W.3N$!\LnU^le +5DdRY[<$#oju@6WI!\4sCMdl\'f\FdW?S4E4;qn3Li05MBrQr'CQ.;]-&?e;&j`g. +9NOb%.QKkm@&d38J.V]CW"jE.P_r4ej!@5UEVK68`_/Y9TCuf@)cb:SW,.;l[7!@u +T]=%6El0J\a=^nCO!na>]P1Uh[%R!GXnR(UiX5\"1.\r>6@NbQH[>sFL\8]iPWs*( +rj!R7kKPC\IQpDMGJra:!LJ"V>i+adH,E8NPo4qV'73T3>utL]b-7oJasA\9I#HJ< +q@0>P>`\#/mENFi0X7/dfWd3u#nH3&^%8H(.IU#qcok`_?+tQ6r6;2M5G!$g=$9m\ +6!&J?oso._)nfMH6X3pW).2n5atpja:I>9[h7D\G_g/G(c[($0R-24*nj4,n*-?UM +-fsJ'q<:Jn\O#D6D9`Y/S9CmF(PEaJg,AX^\smJ_[?X&KdVrVl1p8(a*a<:XL^7*I"kl+AYnMO1>e8:A +>(ETteJ=fbT&)%jZ\hXB"Xuo)3Zh&3@7])i_H+=tBVMml:g4j$!&/mq'=Ff"WMMJ_ +[;Aq@K_aB59TVK*T.1DLMo;#m#n)8"#@_\==I:PuV\9;=U'SR5muY9d6MlH962hAo +,P0N_XtTc?rnQeNg\X;6_YS?ZX"co]^H,@oP:!8=&X+;u^SX"ugVP;]YA#M-7"D;J +7+m6`mpN9m8,alte?DAA"lOl:=\bSA'*/(,)-Pnt+`#qR(ZHM9'9(Y'62SS45f<:> +L]>bMXL>Dls-KG>1%jW>!FUAG]"C[AdB0D7doI"CCIa?V':hkHS^"M'ece_DP)[(i +k5eeupoCLl+'JZ#!9aIqf%&=0=\HZ>W=JZPq1rq=s$N=lj0m[0^=IuMhNgb'N`eC$ +J$f3ek'o_s;#fuhcu,IS/^MKS-H*6c]W'jjnGcltn(d0^#N.XHnaCEj1[@dVc>=,FqS2#-c^p0Y9o+5=YYC +&Z>*D3T%C.[doBtb&cDi2giF@BG]mUr_Vkno)FFD")kpq\FMCY%_hlmg'dQ'=mo-. +pAE)gp\4B.^9Z;PnoW6R".P>'&p>+jo8;T8$$BKBhlmA[kaO8LKBr+mm;P@"&8D +>02\WVi2]Fi?e'*$+tG]TP5(S+d-'=WP*UVF4I.bMWff#.%n::$N/d`)%HS#F:bf+ +8@YU3QV3L,1=/*]UJ0(8Ql7hIf4n+hXOdDD)TZ87K:bi#MNI,Bgc,lND#qN8*s[W& +^T$-h's%>3b$,LC/[;=a(j12,Q4qZ^O6YfH>50i._T5#?k`!AsVp=n$BuAuecJ\_l +A)q"C@dbnBZ_0P:6i+hem%=1&Pia'7dB*>BXk8D%PX!Gkr\3I+]*\%TA(Y^SYdaJ6 +f=S8R:"W/Bp:QjbN"!f?sI7*UeDI/Rqgp:5Ujp\$EBaT'Qh_=-9qm.96@ +cgT/)IeB)KB;[llrh/#<%_KHaO8SitdZ"D=,2g>2e!Y*rEJqj\Sa1IPL3\@)8?6BI +K5YCFYc96R;S5T:NoI=="@[=i;Q?gW"q^hHpOD#*s52Hq>QN/+UJJ[--=49OT)gBk +VEg"IBE?KQ?FoK1qL<4X)#Oo`s,O@)_Z*$o:WWRg;A2;`!7:u*2dM2/5V5C>SrkLY +JE?d9,r>eB%tu^?[/dK@]-Q2`9'[%cB^d[tAR#EhgY1m3pljH#D+3S_C2Kh+[nmD, +,OKPdm6!GFiA7tF-=;ue&>r)f8H)Li[R?Q1O_3V!/0YRWU't]96p0O[a,(B%a<(sP +VMQ6J*<$c)V60LlqT9lps&BJP3gKjRLb[:l?q+b4$%!]fIrU]Kpj_J.CA5AQh]g/!i$eRjrd0tg +gS">6;f/+l&cSCDBdB77t,$(r9d>.?]U:55=PS>%-7t&!W`7IU#Q+7CAkn* +m9.Dj!k3Un!3EgGE9F+uSa-;@?4H]+SBnqknbhSO\)/9HIr1\<1Gf#k%trbWMqMr@ +=AZq1b7eKPaa81#LWp.'H=NZ>^H<*QI4jI>-3)RUqu:EQ#C(7)qtdD?eLBr77suKD +o/U7Rg1C\]2h[-i6bB.&*MQ)a+B$1>!joU73aS29+0+LZlhM--?Bg$T/`Xc?R_qOQ ++oL%*:/+8`AS?.1GXGVEI45Y`J>hVqBMN'-YU)TX9g[_CM"0fkOoH7b]EELa,04-JP!o(PhAS%_McF`U($uT:DN'Sl +BB7($H-nd,i,P1RPLIa[E" +Yg3ehT,kj6fh;=b:X(4-r9aU. +YBQ(8?D:\s-9jX[3nqF>;7k$M*R4o>)n1^\VV431)guQ\J]f +fA_RCa9=#%*Y`NL:)gmK,oJ)0[>bO#Zr:0hab9S3o@%tO#%,:_8lA++1oWH%/nN/c +=Uq5_('FY!&/(^'Qm!&PU'SR5O1'Z<_Z-?b<8urAs3:SEq`h+u"S_]K+b(p8#seKC +s$?Y,L)f]t\gbtas.:&+/VWJUde,0laPTkpY7?&^$NX[I5/03%r'+T1l8&LCJ^3UJ +9CAH)#NS,%IFnM:4m_.!,)?3GiLcteL]A]M9*)MHjjB;bKB[X#U#c/Xd8mOp0&563(Zae<:"?#NUX\`cI=:Dr=@Te#lbS/=%T@[A)Mi%,bPM? +8HN4MJ;6;k5S:ekQZM1uXU2;q6i\D2hPDgod^FB<-VhsD<:OGJBO=TKOW/-NKT4+O +[qIpci;S4PR.BN!jgWEt5OTcoo+:WKrRb2lako3Kq9LM\\#6J14G8-(K)G9rne['Q +TC0T\hLkcJ-hb#dBFj;'cn^PYhAINPhN,sBHbkY,$9Y@Y.I`Q-HW#ADIus\Q$T_PV +h-')?gEBhC?QDX@4b#H'iV!Ec<;\u@^%/8nb-PcP(CUi.^Q%:rE9jh0&!q]q+0YIq +Jp-):5eJc>=h3mGqh!-)B>)'/h#"EV*re4YqU#BtIlc`ig\p2WG*64i9R:CW]/o6L +e,TXjJVQoO#F-6OLg4TI4[pB<]5.=+.Y?1s@>a&8Bea?'LlF[j8d5k.^UsVs2$2L\ +!V!jXQXLQ3XO:((!+H6OpOE2Bm=/U,(JT?:2a=P%6:AWF(1=QOO+;Oi!VVN'5RRSk +7gHtT(!]qiVE4=g%G@Z`:ntY_D:4A<;6!C4e-aT9Ro$u<+(9%K6i[r!e2I*.bb;.Q ++.gSk!ZFSIi/3p(E=*k=Yq00-a[F1a.'aj)&Kh^aKE%cWQ;kqgn1mD%eViA;mJj`_ +J&f^8_uGD9Tord=q2&2Pm?MdKs+tG#Ys@5$^Vg*Hn0^S#TC)-_[\WubYA5]ks#VD) +Jc"=:!JLInlbDc`?cXb@j-8W_o$;pI^YcE`IullKe<>qqoD_WGHa!l"pn5P)n+D1F +[04V/;UX#sdVCE$0pHhF9(:iT`3(Qp:^!^HH8S]T\sa4Zh1VJ1`mcbVC0`O&c'qo" +nX!ep%p5+3KaU2Nh1kh<5l_LI=o_2Gpj^lE38I_!Brmm[$3ha8CTr'Z\Z7#qPd;el4Y/*@1bC_D*q)UYp':UF +Dt8SfE'OOl;Rgqf\rL0H$@r.$K0A6^g__2:5pY#Ra#XR4R:alq9$/#%3WeNIAaE4i +YtlFV^3PZblhlJZT5hK&k5RVh?M(u'o%gJL,l1hL"_)E3H:D[VC/(a(\>)Fk#!NC0PPWFTK8S +q)fgp4m2^Wrn$hW51c]`H"b0@/e3tj,Q=N'J;$qg`!m%Z`&aIF3%mhYhKQ_Vb4'@l_,&;M=`3(6;TchOm +1u"lSa/VL@3F?X7^63r8n!c>tt-($2OE[dcE[8q(D6; +B#DMDj;i"#=qtnZPW9jCl:asFc0!O=701_;;/ +e6;p%!WKi:r#Z=2'2UCgQ[QmAF$]_R4lN$kU$"WIp_,c$@ho'\^Gbdr!WG%0#(^kK5h(9;s3[c0d/GWrCVqn, +!WEOn&Ii:`p$0W0dO=%'2&cINd&Nh?o:nT/WbAVeVP#6LGrhE_k=sqPo_+MOJr,4$B)peOEh6M=>MM8`9b^W.aY?0?nO](-6%6ln_N[0`P&a +llc'.rl9c4 +)1gWJi+:&!hR!Td^jfEYKA`^SJMX8^R+.-0L6>=j.=K62MOkm#(lRn8+t2&J>G +It!^Znq-E$j`145"u#>jX/;ZE"g5Sas,di98)?+Qoa^H$NrP(,pl-VE\ +?+#ktQVUmh3-AZ/bmLSrT"9bGXcq'(>$de;H@7Ld,QDX5o=NP3gT/`ngJuHN<\QaE +C34#1q`N"&N:s]LD/D1:?"^T:bFMYLldnfIW0fPH0l_(7(JJ7OKurmXP@^$GL#o2< +98r`U=`qlDCCuq2^6gT_'[\-hNRe+>+6,L'i]]B@q/)bGK0>0cG;6l7DJu?.=@N1/ +$2L?AN;*4Kk<$a4)CYJ5.,.DYCZMRN@cP1EG"]-O2L_A]DL-c=I;Yi*n6Zgq.??dT +5i;]i,9Y\@kSAE[!XNA1gDI1!<9C_Y$;d)XMtu7:&_s5EpXJd6&,pJa7$XCHo7`jH +;ZMFLlhaS:(-p<*NeGC=EQA2",A5V_lUsK[`;r[:rj=5J._0bkbR="a=)l4#!h>Cm +a=kY(fU")#5WMH_>a6*2aKN\Z!Fbbu`T(G0$c%H=ZkJDV^I%2j(*FE?K*B?9N<&5= +02;0IgH%-@=Coc5^qV19#%7'^d`0;;ftW0/K_$]+Fe`*6t^q$Xqt?\Dq:ZB +Im[3I55>TaDpUtHo=n;aFVWkXR]tMOn/?EPdJQGNs1j+N1"FFV3b<\+ISrgIC.D-I +GDVto/HkXj//<\IPmM)=6sp&eWZVnpS?E[llT_'<[?A)Ni]5_o=]H@<K/As2bb&,^TgrPN-Kn@r12#XCEHVUV^X4"/cA! +:%J%MM"pG[EOFgdgR*RtXNgBXRpkG?S`;KMbh:,Q])^+^lmJ7U9jiV#b=!,eFop-$ +'#SK'/]Q17)+\I50O12:hH+=Y:5^91!O">m[tXJO?IJG?#,>fJ\e3:^*#.%4Yo(/h +3II+U!aeAD!mo0ki^(YodEguQRCf#&Me>AG)Zs%SMtE-Kq( +2JY;b=["Z%@@A +^#po8a2sGW!l-hiEQc*@r4%i.0Y9j.UM=fCP$CD5!;/",< +X=4bZF%7leJqhU@^HBPbELQe&,fooZ8^"l0=oNQ^o$>2hIiMl]3F_iT=9jMj"/e,R +a%]n^AcFo(o&g%KhtOGEmkVJ`NUanu(AoYs?=E?F!nu0*nqBj0kpFQHCCP`%*HQ*:74U*JScFG/Hn$Re\l.nT?J +.=aIs%4hMS?R>6@c1>gLq)/u_[[-]rCBAj_6Y=o(!<;TbhtaC:M`O]+G=?Geb*e*" +k'o'"(On.=\'@\MD4[h9s`H`k3p,Zf:?<1_6[03*RII\@:NTZ@qF +.L?(i'F3\8q%mER#%@sj/-%V\ja$hn(!&$eb=&q5M16Bd&2pHr,qJ +PXkgA+oY!SN$8r+cV^[`J`I^"?_3[EX5>:mUd3k/&aRfAAti>DQ]s0\<")Kf/)Bk13U\eVJ:)-3Gi +W)S;AWoPmXA[Iis5d +''=[^]s+[hCTpMoV^H[S^B40@n6\(U!8,>Dp[S!-%?g;S=3Vm.SBVBWTdYhZ?@*]@8JK"qQFVK"l(7!8PGJ9(aH8S3XCp61HN!eCbrK"g#;dQ;N;2'S6IFKR`? +Jgbf"+DR>fg'7.UP4"/%YECp_k.+n.!;I?DnAbV*_gg\HIt\H`=e5:LF\7rY$(:s,[1==&=A"5^p)0r9Q+C#QJil6u[$%HJW3)XO6NO +,7*"1$o+@@@m&R9qdUT4f+$f5=!A$^]n#h4>f$R=cqQ-"p65J3J>S1;RS3Kp0elM3 +L@jdY_`tVoTC&J8*&^,.?=uu%c^Bd,@tl3-G:bNK?_'X`-d"X.VoO%3*AN%!AF[]K +MoS2^IiL3?l_k\'^oJ-Ir\JSHoBM_-ZQng$%iRZa'O\E?:YUhYs0pT9JH)5oUgKT[ +\^X>;gHmnu:#/`CgK6^H-DSo/2X(r[2Pj*L9!g10GKc+jON#l6`n1@j6')8/,m*Ko +$c)Zn:3D?e?dF5O"W5;[AqL'=g>MN!TA0(.s/Z!Z0`PM3J**YISqm*UAPVMpK0[@@ +J@N)`hckde&I/4n'`6AoqGcU(X&\!f.U`6KF.I9(/RO-/-)sUV94k'!&PLfO9I%QrT+ZAYQ2_Yh_Q +o`Y9W-Ge/05KP-AG[:$`Q%+!WE2[_#f/efE/;8*T0/Rs8?I* +DlSi$Y)(GhL(rhJ.J#Yuk?%!q(#"_.WS7^lCt%CF;ikgZ@Vk'KVFQ>8$P/gI)DF&$ +B;3/QV2l#1bDGEHaRs^Ao!7G04RKXb[RcuJ806[me%+*7E&rN/^i;-Q@5?#T'%:hU +drQERb0D]eQW,%q=sP;/0-Zsi2n[R3qJ%`'!W$X\l0lNSQRAc-B:5M2NCPY,8)EEf +I)(/Q`=+SNbnOscRb:dZJ?T$ts%DLirtd',SfVK.ghCkC4mV^97dnSW3mhBIYJG$V +/]n.ge=.OrR;fmpq!)\hB1pl9f!+U\2?@n^fO=FLDNdS;fY?l4rb[_Dcb!:DChMSZ +D"O>W7\?klI(7]$K*BJWF6Viml0li.@_qs;g!gD9-l3,f(ZE'"IV'qe6l3?5j9Y,F +)TLb\HG@n<=_atYjJl*/Zf\Vb +lJk$lQUdDKrb'o?r9nZKV_Hsb7CAT5-6pQj?QNZ?gV\9AKCNV2mD1jSc`7Jg?UDN@ +LjER&_:Q$@$%a,faL;p_DCqR;?:MZ:`Bi5!&JdthOL%45cdgkB +o-XJ4[.)8I$j['OSPXphX?6E*Z[$5l`f(6UR[[kFaA5Lm*C<#bl<#o=a)eZ +Dm#d$+S"+dqF@ljQ[:-:s#&k#)Tnp$b&TLP[[>Ss0Q0LNXGNED'P"%s[*+H7c&($A ++(kW7CI+$`'I:4)Vg=T)"8k<$$##:t54qN4i."\1Dd.KP=&i(8%C,tUri8&BR/]R0 +It@X#qifps5JSAJBl<4jr9m;.o7+Rns8-P_1&kJ.!KAD>[7.f0"[6Mn5"-s)I&1JI +gaso""isKHG7jXJO^I4D..C+Lq%T(gJdP0LMU_XEAH@)lQ3W#!b6!UO!C\;c+KtiN +"]FFe'(+UBb^Y>r:E^0DLlhj+$Lei2;6#FYi!]&s49D>Z"TL4SCF__eb!W5Qpf_fRAP&ht&D`FM.M!uR +%4RG1$&+X:_U[>ujjN@mSCj:CfLC-QM_YlI5X@/-5]95?T/m'*?r"7OXsS4=AJUC0 +83qJ?'TWS!PQ,KgV!dO?^.(tLZZoN?G]n;V]U@ulpVm*R`Vjh)i?K-#5C=2.K&FCW +N;p.j>8>,n74MY5Y.HgN,qmFj-ds5C@%3En%62L$ZkXk=UrO^V`S)Y,=iZ7mM[H_H +#au[o4#tsB1;MGWho4BsRHNmHmcVO]Fl`>MF7o+4j8%F"YS"B4/hl)qj>rh60&SRO +m50OY35l.S^i;+MirYHi9McWIrU)3oT7bDS]8,t+T>`:D03),pcpVS/7l +\!L!2/2[-65-">3?>T,k$L5rQBYUXF_DhD$);XKQA2!?'+khhtg;n7\A/6E*s,4f1 +5Nmt`MieA_W;^jmUZXk[r4a?7r-"/0H?dO/f)qOuqp"&_-U[)QkSM:LTVmkf*&.22 +YOVQs`]n?LhV#]WZ&8Cs@aZWF-s-9S06L1<23Wfq%a10'![.IiW^(I;s.U5%Ou/4tJbs]k=6Fn].=<*+d:/LZ +[9lEm5Z`)ID]MFNo_`(O)7-t7@Eed5oD>!V7.:`V#AoupZ=.qIs2"`O#!VMuOpf5. +1B44L^Udl=D4tlnJT#PBGCdn0VJq;)+=GK0#CiTI)#V)=Mgb6Y0.@D_eHCH]"]V9( +.]s%0.1ZIf-rC3bQQd6].0.a'7f6?1"-e'3arFpmb%uQUF">4k&`>Aei'5Tj5lC("8,VhS9^/Mg@/^\DAc^O-]V"):0)rll +(I5IFpF3ll5kB?nk4D1L]<["f1L#G1.@C.(U7`-qY:9eI?RBs9r"l9cLf<\0agRU/ +2SoT$&6M1iZun^Fc4i4XA#.^17b<]3DO/%Mq9d(n6bNRdY5G +QT>SjQqtH&L@XY2b4GSU(FL';B8uR/TYQ%DCb-V,M(Sd*'LF#=DkZ@oYIdI>s-GGn +cXEC42&^3e3n.tf\A`]kQ$KrrrNAN@QV39TY;<_u'h/"Llna*2d.m&Q;hufn*F\t* +AGjJ6k-+Y!%sG;ID)_<"X3n&/b\u(]^042.cW]U8lPNa]XJat.aS9o\rO_`EGlMe! +rNg(bNu!m3&!G4])+.ZB9hSieB([U3)T?=RFIk&2;A;8[(c,//2][A$"TP$Xfrg_B +q71AHH1$AAM#T6El@Inb-4@M]6"u75F+,@8KtAC$c)ZFpga:R +Eu\1X`r`68>&$Rshf=!"o;g +A.#5.7us+O3ahl%O7==HR1D4'IU5g\]:*nU-0=8EKC9S>DCm0j>tMcFNqVccg7teB +5IT&@ge#TDhpDZ>MHu0*#r57u7"m+2:UjpW5^Y_DifOf*._=19tiQb?l +q?"ep)'.`:c4'QVbX:aYg& +/RXF%2TMk!maIj"D(,$o0(j)5pigJ*H>8**r*RedL&&K0?pStU@Bn=#Z+S,l;_'ir7BUfBiu!\]ER55:mE5$-$t:E +kUMZKrhJH`WrIG2q2`Y$;Z`+-_-JE4lkd^P_Ga&T#>tQ="o_[);5FE/f`DG)4U!k^ +s5W9Ertk2Ii"*tikNBj?rKI7u[D?5Os5\Z&IFoJWZN(F6>&=P=d$i*2XX9:Br$t<: +)H'nc0.RA]alaJ_K)ttolp-7a80J>)$NlY3_>1s0_T_M4XN7]KLl1PT)QD!`g@>oh +MGnL>GYfu\?'[@G>Wc)O"e&AP2-V:D/C3Z9(CXtRn.3D6n("4UQ7$cgdsC:7qO=ee +>=1hk?CYc"Q%E"l\f.6u+9jG<%,Ti9uMsE!u?l53DHHN8ZQ>BIr4-/hmNe:'%!tmg%m+2nu%= +Yf.IZFn%\(!Vh.W,Oh?)rZ&p>IE^^lbTj7^UM]X%S[8b,bg.)eDm!X-gVeEBKD!96 +f$hOVc]4C*4DooHr0["]6d.b*0d#dm!aG*]1;Y&_ml+1[ho`I@NJ[>g2TEGtbTF=C +;IU"JFq\l`i1l=hE&fjEr$ajX$pJqOhjmA>DT#uKs?gtj/3Le +[g2oM01%*Vn-@p&?Q\lLlsoVP(J;TK$s[!f[*-T7'"?@.TWAhQ;=\gRMGpX`dq<(f +bHgJGfabrD3*U[2,?Or^IsZ%hZo4cBggW'ks,;k![.4'DI[E0BC5i0oc1),+=4_\q +&-j;K2AtO>$i#4$1[Ttjj?'DV[oZFFXu*YrSFlbPd_3&C^M]c.hF;jtMuNHWqU>Bk +npbqD0liF7I%!'Ac?T+ie>u108,%@*(juqfIu2?3\S0)@i]e7Lruh'&rLBMREIZ\( +iSGXITD"Z0hdcY)5POmSgrBXcB#H<8@oWB`Ndgsdf$M-56N23^l,l\PG<8?.Rn1S7 +%/6&,DCm'9;UQ?A0RJMn/ZHC_\4OY:doGA8Z7r2H0LmQSa7_ip6m0o\DZN6B^ib#3 +5k*[C,29Frl<,A'Y#kJL(\fg&Haf7:[Tek"$b<1)Qgi+B0BbdT\tcTc`\(j=:?'jajEoIq>dT9'W;e +UYDFa"],Kf&-(Q"d!tU.>JPL_QO"g%`<(kn +,74S4A&J?*MNffhnfL=Pr8D/\j5N;cIlg'n2'Iq^s6I(DR/[`h2q<[@NT38"K^a[V +"TP-`Wo0;,2Yaoq_<[u4mAMJ%-++Msc.P61mQA&17B+$Kr)'E!W(('=Lt +)j6\t;:T6*b^YF)l`4s9XMei$^)^2-A>3S&"4+M0Ib^o +;P_WlE%ak'JP\9Hhj#/?_VmRt6CbPf6&n(E"jF4Z5V<$PTZ[M9)gb7g6S@:TSaC:= +$F?Lb4:KC-Um]?#[^[Q$EcX=FV?b5<6saj+,OPI7B;Cr48BjE#00ao5K`s(6F=H&ei"U +E2O;N5CC/-r7N:B[=67[m03g]$oVgu$OW7A3@7`DAu[,J3/3b;2[T8OQ!sh[n6D*^ +4PL;s?d3Qki/\'>mc+Cu/4=rlCG&pAS`,ZuDCnPOOdRt1Qon*14u%e+rPn:+2ift&ZnrQU:F\uG^u9t@Nfo(,iq:7%5>H::ZX>TcaUZ%uMe3A0* +T'^%#4G6nX094IN:)]+^+aV9Ib[/!')$Y/-hdZ3[bL-&3NaB"Z?%DObL*)$I:QBE: +%IkS9s*6itLOG[7G$4gKb8g070()Ls3PXo8IBWstcR(,+W*b$u'E,PN1^O(F=_Mbc +!*q'uVM0-!NjQQa]Y0!0:;p=u`uN/UC/6dh9=E?g=qsN\.+#Y8M#>[,B>"K4&cl8Q +aG%\:>k8V7E!fF1m\r8R+SLC$TK:\o3!ekFga_XE(@f/9,-\%/TO."e;sL80p+7le/hHWq64/hC ++]l7m97ERY6*XKg.>!%`KqGs)+&l8U!)cl'^*.>IW*C8MVGLS2;Pl`4:neFN.RCC+ +pc&99J%dH6/-'#_"Vq(eJ25SD0(>!U"f2"Qq@@K1,/TD[s#I#&m"s\^Ma'2rfCN>^ +isLAP!*d6Cr5TQQ,#Wr!@LtA\kQO?gV+6..e&,.!U&'=i[.BJ@n0NntP/N#]q!3Q^ +<<)bIr"!MC$cTt8JDacJelWq"Wi6K.W=GoY#$4mrX/j+EQ!kEh+W<5Q%,bKEmXHlA +7$g-,QUhj0ni=d7K; +SDIuESNE)+kl4Dm'WQM,0mKq:<9'RT:43GR? +#I+Ye"TXuHPACBlduaf%==&_s8`WYJ$]#1\*907 +HH-A5ZiCp3\)ZI!!4fme'@5p?mXuDOfGK=r;h^ftBFA&XBk8o2J%Y_4JdL^8^1T?MeR\a(VdCF>TC&JHl_.YOUV8r`^:o[=_9IKVIZ3BrRj;I2RdfkTWY_7k^6!s_3nNTf3T)VR$J!JM$pg9:F +UK'k +AFs[Kj3Ja)LKeIWDPGqC@!H`E%EgSGDGEY]1h\^L>T9k9"qa\DQfsd]or6f'$@VmUj%/WB0$K^2NmcL%C_\Of+kTW +a\;uG9fiKJqBU,Vgr.&8.n-CZ0&IhAmiW*_a:0!MM,b +=b\m)^HB[Q;?1Qo=1-h]R#9i/G[W*'e\W:=dS +o@L#@.S>(\MS+k)]HH-6'R7c#Y,Xf!*W/RDMT^Eur8lYJEim*f>-HQU0<5R3YMj9T +W=@EX"d`9p&c<.f/G;FW+Wjq@U4n/ou+ +g^D+B!L+;:]N4_jp^bg'J!,3c$OFZd!$\_!+C#0r'NUS:;32\k^n7f;pja,Da7CJL +c]/LZNXj(uf*DNAWX;(] +nB1=+o$U>G^Af5D5E@I]"QtXE,oM/7BtfuahHoIX +(0^cL!J9P"Bp$kB_P\`9=^`*h/AO$lCLW7cKID&&82p,G='gQ<"Xr0Z1=X:+mq`U& +@"GWaIlh^9>9:;)mgP3ed1-0P=IXute'd%]$\M$AM54]CUtoe6R7>W4515E:V#0He +T'W+45PVmd0AFp;!4<-N-*')%Z5#$h,OYa)tdmf"X_e3,In$#:="o&c*`EZpU +]jDg>""+&\K6_M_m;q4cI-Q'2ZjZirQB4cNcXr5<6f4ESP7& +Pk"5\;m]lh/._=o2(6cf'/!\jr$_H]&Gt,m2UNO!HZkVUH@F?Xo\u8&X.s:b!e9Vs ++kDbnhcrPKEg\*5s%if3CB\j#&[O"6Zmq979]h2l&H@H@A,OYpbnHcc?@[?,j?\s= +k@LDUSJ_8!$07Ec\&Qt2a/('5W@H6*=Z(JJM>VR?ShL2F5Ma;/#lg5c./:O\r*RM>DClk[i;<23T/P)GgWoS82s]-\ +k3ufCiM1(e1[5mK0d@;lEtIg_1if"g!6"mD\dLZXq#BS5c6#EPmJG'-F+F;b]=E-0 +9XqKaGB5PVFCQ!]jsH9RgXp_ZRf'\?q"p&0gZoa7XJ,^m5FbAqYFfp6STLNe[(Dd6 +WQ]K9>[5bO9jo8\tF.q +BF(HG;Xj-C`b(pa^i."D1Db!eMYe%R]`W,P`,F40!qfW^E4)!p[u-X3:Dk +97>>:^;0dH,#;4DVm=)Y?4'=fMq-Z1)IIW`Dcr>I^qZTjYe9\E.N"t\%RM46),sWK +=Bb,'r$P*%\eH8/;?2E#^'n.p0)j2je]7\F,5rsAHqt/r/A/l-577UdA4AY5ZD(V) +rqM>V!"&/u$Qr&,@p)'kms+gF=[4)la,33Wq'fap,lZGr:NY>-g@H9p@=Jj(?Q9f2 +%fAd"@c?8f6'uS3*?P?34oah_K#[DS&;#C;2'34r+%?1?i#glQn3NdG\PQ.p&FDVmJ?B%_]>#e.%!T!RN4U#Q5L$9_aJGp0)^`R]S +eS0GCWAD^`#lXu,j1?fS\n[CTldLCaTUDoi!?(k'64gX,E.5G'@/eL:arKch(hcS2 +CJ!o-8a^8J#QE>>K`M/'"[;.'Ok^s.!2;bJ[M)2,81&LJjlHpJlngep"iOSW`6?"4 +O;'`5+_5UucpY*gGb7QIhT,bm4Y2^'\6oXr<8.5p*&!qgopkVq)? +5$Le"r4b3VHZt!\AWUq\55FLZbb$diJ5MC%0qSq1+o!Fn&9,uZ,=id[;\B&U_4P>A&W5tbt +f1[DGQRAJZ^mPLm!S!iCS*Y5ei&Om>EWH7:5^k6mJ$bY?jGFa9MDLbfpeCiuWZAl> +-%9$4XHd++nPd7-16S^o;l43(=Te]#6JM&.L\P42`rCoYgLfQs=X=ia.SY+G)P-/( +Hn[h_@M._i3m)A&d.P!qa'%VFeg8gG5pr`uRdfU2`N/_mYKjC[*A_uLO7?`c%2XN^ +C]s`9DQ[Kc^GnDHp[`"nmlQK"lnS?`^[]JjPEB?eUF4;U:)!O&cOiJBH&LWS4KO[g +QDjeDq0ru^onE4&MuQ$1kM=t3]=!iLH`QtORlOho1]r0:c33daHM4i6NGkn0I^=q* +eT0C,MlG=j(kTMpL*(aeBhPkM\h_^>I+it9Btr;p/RqCidM-irPl-?:H(d%4_7$'\ +HVoSdAgi7\Y+_V2hN4oACS`N>XJMhD$US`oG0r74Es +It&]XVZH>&XZEC\MoHPP1]c-<^)FD_]$ss1IoWb#IYgRg)FaY%'bLQFb5r-&B;U60J*4`fYr0`OGSImm68[K-6DKcn5B6Ln(NWoR"7rttQanPq6j +\3LK&X@Rh(J>OVaH6$'AcpB*irRWBR$j^[)Ya.I]PR +7$/7T0*oSqY7a@Tj?tldTbO.gRFn?._%qQ,!(Hn&s,rW'daT=\Ec2i25InBW!Hiht +rc_Bjp!fh`C_H@s$[Q%*HeS[2!IaA"".^=FH-knQiS8;_n:.Ym$`PA'#l[b82CE]- +-ahRH&c<0>..P4Q+%lW>_#f^=>X8^R1U7"SK2c66gVI$L2OSPC?ifLhdtq7i'j*_K +n>&Ie.1*_PobJ,+AeFDgJZsZ]KEY$XF.,A(^k=s7$R?[O^CANDg>(02grIts`1E7\ +:/qL])e)B[7AMM(#"OK-^qQ1+]"8:i'RI?KoR+F4n6tU_+nEgoL7f?<%%\In!+5MJ +O5\RDkkT&0iNpE+097$'27tnCL3-ha[)qfWp9R-+^[F)W@12JW[FA!<;Vn$9!WKhq +MgfB%G&7`No0no4f1X +98p,"1aP/.$;"3B!WG$WXSPR@=&a4Kf:*aqB3sku +IC=ZPCGkeDJl +Oa;@;jlUk=n/qYKn#U1#\<6U1(=mE9N;qUEqYVK5eIg9Xj.+Lq4Bu)X)rBF;HM&-h"+RU0?=o'$MT +B?Q5t>KZ__+nb\ZJ3WG<9ZlTpb/$QbS^jNIC?g0]\R35?biCUX9r^gQ=r_+=bF^6o +[)rf_c3UdO&%!,U3FqV"1;iKJ&U'm&VlXiMHoiU^!8r?EZjZA1*p9OsE*"?s'dS-b +=b(-fVAE?"\\tfua0S\+;3UJ!;!cZ4p1p_un#?kNLjbpFFA?WHK$NSV4fAMaC^7ge +$U=dac\fVlcWBa)RK26"b[O.fo]!$P"`7q)#YYBm^U#!O+&B,mFn.u&q$iS*: +^ED,m5=\fT!WM7npu(8+Gb<=J+j[Fh:G`+kir[jL7QH#TUVR4@75=;*)hu,!s3__p +A-)41NIV-R]`heP$c)ZFU#8-IPSuiH1 +fE1^+f1?5I&BoV4ItI]/#l`A:TNZ2S9i[.\,)WUTkh#_cMCQc0<6E(.s#F41"V;(> +ZdT#QBn99c=[J#jkmdV*k5p`;;nG\#JH+G:8R:W"Nj%X%ff3`S1`0D$UN:NjF8s +!%Y8RS-?iW9Sd\t"G$^f0(>/15U'SRCr6^d2>=L9pTh'n[H.SH)^i5Eo)]?d@LNHp +"rdmX%ojR_gI_Bd[_])B#/@5c5Mh"C:B%pCl008;?b8mZmN=[L7XDG0o_V+R8"0qF +0;%SV4l&bWIaITQ_s$*IQU=6XS0r[:^[=Ldphs3^Y:fqUdk8mW!rq/M7Jk[V.e*Jh +.b=RCoZ+S&SfTRTpWN^=&`1cS>Nh2CFMaXpE3YPNU'`e.@\5HqT4%Xc`pr4XA)c?: +2FO?J/>KmCBQBu>[BIu&D2$YNTgb_u@Yo'(1jOs4&Ks1[`>2hk<%%I%mbH%.6<>i@ +p9BF338M@gU`%J+$s0g)45D>Abn23l[S'?!:5kd1&3Y0+-7?i:V']< +jUighg77?i7PlcdO'Larn\%'$IX]B%aY*Fn>:Du1LO*_V(%_J!#1iMfcjAM)gVJA6 +T'f*!3aWn]e;<,`BldR*%IruWRE)loS+6IX"9/*l[*oLC0t(l>lUja$.\f`F +UJYE(Q"4261IW$59c>c=,JYeRDd$n+F_[R:>**6?Y`g3J?5R:PkJ'!n"Y2[_=VBhZ +)s)rU!(FO>n39.Z]cb:r0\*iB;Lf1Z;NY%KfLmI^!<-'#4R7o%$b03sf;T3EVZ7Y2 +ZWkU$pg8rfQ8NE0hVpaHJ"bQui*TaW"i5\VelNB3^j]9nLgJ^;LbAoV(T7XfKE('4 +j8!&DmpD^r.$4=enK1I+.E+KX<>#+BTf&/RUb[!<`A&,uO=qZ'bQ^"j(&(.9Q@M(r.6`^GKY` +s7,LA\;mlIcAMD"8lYiTY#"J&9HE/CWh04mD1MJ)S-"3Ebj#fR+/XVc>#Mr$'07)m +g?PCOX2A^'6H8O;*At\LJ=^9K=%]"E=jErdSF2K&AhQ`]b>,DLA]<>6`n7(WbX`&J +8d^uL&i_a/?qR0N,d;fDYDI7j'PcuH +n9(r@Zok9YL0r1]]Xs.#?nHsm\,ba%9r),TOrg\J +rOeCQ(J`6@X1pkBNohNX!rI'p^ZI-e1hk$8SJi +-_kHa+F4j"fS_L)$3%PrkI5CO&CKf +X0ffZ+Q1!gD+7Q\28Od_\+lSfl[ei'k?+pXd8MlAObU?AQ9E`4FMf?[&g^tb0Ism_<#VCGaN*P:lE(9m9gd09d#Q$JFmc**[`r0cg +k'Mlo.)LcT3N%2EB,-7E9nuN)5&>rBL)G$N-5[]1Rno +JDe0Iid\6=n/kURn1;]R`;Hg[o@,6Ql3b5i97<6dIN;Gf6jktbYW6f3HhbI\.%pG- +SnQ@Je[#n\j<'#('S\hc8U'74Nj/=r@Gi.;r9O.8'[kMCdU!a>=qfCf*C0,O#P_Y( +lCmBj6+LX%`iHVtVkhI*?0t^")h?nND;o&*2i#6`>6(?.Y^jmMT`.dmH8`/PY3>s2KBaS+$pm +ne:L'dN\Gk=&jr^rp/NM?^FSr'44h3:fA+,kJ*lN$M,A!J'@pt8J%ZtTNk?u9ZWP. +q8s+p!P5TgZnsQIpg9F(r_NLqq6>JJ1Y$3Q+B8/.!WGII&,^?!fC?Vf/kOChmZo[4 +s7Z*!p@RnF5O]`t_7u$1m4#`N8\k@dL=k)2W!7)Un@u9Y^V%Xc1uGWAktY?2.jP1p +*?aqR$c!ckA[\X;j=D*.bL.E-anF.n7nJk2G(Kfu5Xpu4>533DJCNL+_#G\[aU,Zp +'7;-W"iP^hN[4`?Ym!B:&FO?Pk%"K.&8ohQ47E:l/H$P9pc2L^3O!"!R+^>\nh@u, +A:p^=e/`;hW0;ITjI]KjhHOUerBc.kEsus31;);$ +/u#p78A76ZI!:_3:FckrXJPj/L%8n5mIuU!LDs>+ef!:JbIXsKrn/L<[d>Pmn<9l_VKEN'0'!4,Y:$4<%KE\U#o,e5;H@]![ +nGi'hg!_+n7siL0ftk5*M>Qs6;+)!Prf>IN*0)_j4 +Zo]5*pU;rNm\cbNb_\&MKV-HpeB*,ad-0r3N5n!t&%==^D57:6Dm*fHaF>H#mNs0/ +s3KGZJ(r6I@m<#+#Q0'k%/!5Cs6qg4dTM+g_ABTT%JW05!f#%k?C^/,0=\UQm.,q+>7k`3;:4 +6>;%^1;Zp-4>d4MINN>JJ%XZtKE.K0i.%t@C9UVd_1*UMFTI/hNhULUX9]X7,`D`u +O-LC=RMW.X!PPoAJOcul6_3>!dK>$a:eET"mE!P6nT#r:CNK,#D!HaM2NCp_]SJ7,?/%G60&i8H"M_V[5LN_ +$\/pVT*<^VnNXI4lKSqt!XSSQK[nUBf@Ru!eAjMt-T9I$!<;`^Iq;bVKfU.IO6]$G +IsEY2:DsWWOI$t='Cus50Vp0K5;TAPmApTj-i[D93r%a3E#AS&]!7ka@dGJQZTJ7k +^jkCZ0)-gh5L9_t(X2;8nAgLgJ3WUY[U]s=^T/ft]fr=(2R:%HUdc#"=mCnlL\!YP +"i(lRn1W&\/lm2t(((?U.$V7)Md;T7(V.=-0?#Y!`Aom18Xek1*H*Tg[`aK2(51") +k?Fd:9MGh^S.mXhZ2e\AqCDUt!5!dE=K3>Q;/(9EXk+aN$ +$cJ8]KsW)ToF&ou+*JV$]AN+i3M\_E.!HJ)^0aFF2:(dH=U&7,O?RS-IG.Bk[#t8h +'BSgd__4(\^Zm'"Qt(AR&c%2nU`YqemOb@9\;Y`X.IH7qg(t,*5FOrcDjK?D1C_XS +,_eO1aB&TT5+,(?FrB\f2<9OS!pi;kOZY]nH$,JY6@3F,?oqbUS7ISM675bfs3^2m +BiOeU/T_dD#N-,K6X,@jEURa%^U]C%q,5[WZ^Jj\/bZR@elJ9d/Xb6BZVmP;_ +%V\KEWCc<"m`Z([rrMRV$@q+(!T_:QGPE/l.M36.J#\\T`?@+d(;W5h.PARf2LK7] +pnX:5a`.I`'B>r:-VR6/Re?scV+u:r88.uMch/mhG8hDo11HIQVMaE[Ikle,T*"TQ5RUYam%"9.23Y^\eD,6&ao +T\668[[Gq.aTa8K+3;nS%URX@HkQCNLk?FYh$q-^La!es!j=2+d!5QIhokt,S*>nA +-N@4YFF=>Ul;H)(Mn'W?9KYHtiE3;&TapMiB!_GqVYb,]]=#War^B9a'Wf"gX%Zq1V1)rn]>0Z5Cre.FP>_OLW*nAI!5 +OZGc8O;mL495mfT+o^LO:dM[>;2/__aJ1<@UHHT1;XUD6n>If,Cf#Jf&1rFh\tK%W +i7T]08-'KJKgT5YGUH`=8hWm;-/IO<]>B*rT_Y-eF5-Ms>D%M,!<<$f53@?-6V(Eu +?rKd-K@[f7_@QFF0(gbJE@:PM%f?S,6h"H_oc4Qm'_]"hJ8/9)5]=5!BN6:lW\,.` +?_IoIcimUIYI=L[$31haNcdW'J5-5oJb%fKYm'%u&G?0UIpQAfDJT:Y:]%dnILL_u +%*.oO7X/Q8qM:kngKGf;!h((G+9fPI/qCm\('+7['O"][?<7q/O?^Ia+bAR$;QVhu +p^clGo3YE7XlmGC0,.:aYQ1(:6m2k8AV]B*,7:b$MR+L"ZGP.dnFEY74o_#5M/B8W +mc%d4mD\Qba*oOlkfb]"ljKH]*RFbgg-j0bm\WtA5D"B?!!WC.5lQ^&>nHgEi1=j/ +c3r?hB=9,\Zm=Yj:K'tU?"G_Z^Immk.J?dYG[^Rb$g0\&"[44>Y@>$iOJ+YSZUj@Y +R-gcnFB,-KaO?YjWSdj'3Cuu]s/[(7bU?\k#]!lD$pj)6liP<.P7o)On=G<>!5\o$ +"[iq96"BiU"Zoik3)=kFInsfV0Vp5LE,I"JUQTpg6U"dgE2eQ`Y>3L^4X=G42ZokZ +qak/gP!3_1IG@B)U*X)C,@)oHI8EMHYEt,jSC\[!2 +d`2Hcp\RMEB9H#6O\U$:SnodFD'rWS+al=8qI0KiehT'qoO'K>pDp'Ep7NbLhtmhR ++/$>-LqM[-;$uO;EARkqoR&Zd>pK,p!28B#.'*UI59&TK\9i]Qp[0Bb\#Xc*(U2fc +@h*WamV([aW3dBmRTuJ#<@kH$9m7/G32Ru`VSG"4;'^5ereiR;`q]"R"lu#.]uKY. +45F&(jpZ_m%fR,NVStqhfP_KF4s_[p,3FAAN"s2/Bu'c@P\J!VCU(r]RdGY$0?\M; +-i[uC1HGjjbq8%ZeXGl>+@^kcl(9W+I_1JbB-[6OB$tVP8SDhU!W&jLba/PO0DEk$ +m&*)(9msYpS^IFJ'08DKj$ofU)ia<(2?iZ(+S^Gr#6k-W*Ce3iSfQq*3l/eO5PtN% +ps4EFD&KY[]eKKd3m.F"+Qr`AcM(O:^0,+@`SXX,/][,S&L\\Yr +m]KaeXr5*]$93("HF=WBCi(grimkE+cj&Ed[Bj(BCu +56EH%T)o9c>Vc-#@_U$1rp+V+EJ+ur''0NN!:X?ekR&)u9I;<@i."=qHV'Nm>2i^/ +F03T/#QE>AH8Q?fJ=bK1R12H7bor/M!7`g\5lc;J^)m8=r^EBO$Tn-^m*ch1Y9ObV +5Gp(9Eej2_n#-b7Xpop>+7ii%u?P!1*<^+s$r$O#qnR#K2.6r]\ +s8!>Ss2BK_IjXYl.e(MSo>jPo&1)T-!Vh3DYtKOgn;i=.h\G&[_W8`js5^rO!jZ:* +OThE.5b>dkgc-Bl!Pj285J.OojNi\Q#O13Na1bmDP6bO24sn#]^,\*E[5G=fL#:Ao +gt3Inck+jV&<'T7YH&25XUD,1%4-\K?XFlaC8t&02:f_[`Lf+qrC7N&@A*`K5% +/h;o2/WM5/#ZW=*jQMDA#Y:4";NMPi-k.VkKTo+))G-kSRh*q>_Q'Hk[CP8h\Im)D +R#]$OJ3PdRR1MP08*!,\,g +0`pE)OPd+?ckci_f>tVJjh>:]T@,jZ_HbuGM6BhD/^*c_A@a4CNllIur)B_KPY";8 +js>l>Op&,.R)oLd'3%gk7>+f=R#*fD37H2Ii+&ABpJuP-efY;JYEgofl$@3pY61Orh&>C8R0N.*,>Ym5;40'It3#j +pr(h+amh*L'4Rjb`_B3cT*4Z08=q4A>%_L%'#br*]JZiWif]o9n96CITtZ#5bct&S +gR(4-YB848C5\Y-B#cr8n+P55c_l6&Pk+36$KO^lCnUd9%AcT`Dh[c&?L$S,e47+f +^[_.gB:UWs$"YRIKZcdHZd2q&/Pr[Gesu. +oX"c7ikHp'@W7?%FAdqBcncF^Vp++H^c3AU2#1^Lna.p*?`c3aHo_)H@c#'a][HtK +7qYTRf/[\#Sk?ZAc8gQocsnf-dn@c^n5lR&j#M,*-l.K\9l=9hPka +OV2&mi+Mre.!l?[37"XDs'?8CA.m(o@!;ddg6G^_bR?"9R@.2/ZX\r^8S3./2Gcr6 +Zq[%(8]/c/SN%+#qgJ'#p[cI`S+_2Ar\N.gZ\Hl'WdOcj3R8-DgspQ&gFC(bT:V5$!VJhmGukLN?TjdY +*R4]cS&N_Uj?ih6dW,0hhP@\iekrg*>mr:b."0i\anD^^*ReR"'`>c?)%S/Yc^7H^m?qo9,``.)K=dkBIk'> +/08K_pGORerBucTYXhh!Q@E7!GHdMt<1?*Ob8/TJ7N4eoK +G;ul-m>qDC$GbW!#4jpIFb%Nc$nV7)U&NZo6Nf[Q&,t1ZW`=pK\[2eC0,ik\q2^h5 +8';UAjiW^+bt*P3Ur"fj@2LIQ%YO]&Xb$oWBU*YGg+0=5oKV3tFk?Y/N1Jo,BeVaP +Uqie*'`e&nrZAVY:-,*Jn!s&o'3i7ume-V"WD;s_^%.fe2M1t,S@,0m*I$h!neJB4 +ap.8!'F!.h;P$1qeHp3uBZ:AnUBNCR=#.^Vo]JLV+R."4pT.D +'it+b\\cu2X)!\k2han?_oBhDL5Vi'$=gZWZ*!A[VS),Igc-J]&o+_9CU3!#n7LWj +jk5K%fnnhDbA2-.RlE^OV1JJhb#C`'RA7b?A@\S?5dKQ=XTi*:E_[FTk`#6Q"f6]]D>:MTC,K7C.&s5H:>]$Tmckgt!" +_htVO!j>aYYF#Bp)X.d"d%`Mk7k]1'P4ITPH?DAsHOK6>S[=IOcu/8?+!3b?*(HPA +6_BpBHoY"$q@*>WErk@l"onaRDZNsrJ4H?-V?O+#oY@d$Y:f_;dZRDO`k?jG +p6>`"s)\4g-T.[kki"*DID'FXudYES4])^)X)N=C,&DhbY7f6pNm0T,dil +rl8*X$o($H!1(A"YO]P#9TNb#6qjp4_^A5cY?A-V]'Pq7#5nMrRXi.kVEM\bp<##A +]6gOP(l)T%T:Vd+G+/9H3IEKqB;b!dT*bAdFQi*ji?+Zt]lc+&j&7+nb_LRn-MXpR +iM?Cd\K(M&e^ttH_`%hlc\LK[iiGoG>l]lbOSmqbjJ1bl>Z^rRHjX)@#tD +d(gk(5Ltu,&KpJ^O`b7dHZE'f/?&UAUF_TW)uOVUZ(5(,:X>RcnYU/WF&)^\GKE<] +aju;X?m,^C[&7Bf1hOpA)6WcA2N=&cnmCi;L>W(-:HOK3lp^!p+.'CVC?D"E"aU:E +^I#7LIt]Kcr3&#a[H;jhXT880o!c2'8GV0/J("6Q?ff!#-[#WWBIlU1J#*/f#H,rH +>lPI15^OqE4O]@=1aC@csEs,Y_$a9WVu-laOF5/RhWW9!jA$[ZZ7pV,pH'S(cW6cabM +Q8d5gJD5h`T!4&%DW/`B$RR4#`lBUu7R.jd*ru-BB!YiOW +q%-ajR&qFNdLWE;&#QjkPK$**VM0kJU4dZ`Aah?jFu4SE&E4c?G'+%X>Cna*U!eh& +dB`6BIV9NV[$;1KICVLSp&hb +50D_*FCGYa&>O#=oVe;VR"G1%N:L1%jcNF4?n]+HN)#V3h:ZIG3ng$_mUr5GHuu6W +G,YpVn"W!-JfYZ#h^Hii2!@dWH9d7gB[?nl3iA[D;$W[i#+S]?/h53W/UGWu*d=9T +4m:Q*4?.p,G[hCNZMIJB\>gIH2oO@Rm^o16&Rr9Hi5s*6#UL]C]X +KA:pc^?h=>bFnXF3?S"HdFdXV`]^_Ham[e4I;h=2KI_23.="[VC)DqC4-kNVpkW$5EL'=b5+t[+Ug7ue[d:Xc +p7[>ihp2K*AHVUnqp5`*nNe/`lp2^7U>.ea04k.T'Y6(r\MI$nF"6A`^*P-HpgcSd +p?AeCB>ZqYmS3KA@Pf8.-/W'6$NC(q`mPqN>k[3197D+.mX'!3oSoIf!)Lr1CD8=L +]\Ambp)RN;ql;#-4rF1AeYkPG_EU'pq<$)G(\U6Z]fk_kS"_!nMnCkYFkX>&YUa^S +/S2OKBsTjLK4J_uE+-A3gU-k^P>h`9l6@:6rO?>M5@F*b8tSR!K@f,gESCo,O2] +NttoLr(.97ctVk:DA]@FAIA +@HW:@4Z:$l>\Gi,K)]uiHmo!_otLPEru@S`9DH9;jQ9[Z`AW4FICtMd63NC&+Ii]F +rl6[Da]1S;!Es9Dg<PSoU10bipT)W5(Bn>cq61b`?c&aZn^f2qY2@<_L+@r,6>S(`E_\ +fAi44nk.@1EcENt<%/Va!,>Ab<`FqE!ZT&V2igp?7P!k9'i%Vk>?(!Te;h]k_#Mm+ +eld,Wl2UFr??5g.qL=)Vm@!Hn#8mOBl$ouB+JqCEF1s9^!_e\cs%Yo[D'4>Vs(a`V +jR?g1;)<+%fDg*pUK@$%r.%)P#QJRN"\&fY!1fLX-!Ced:Q("+eaX+A5A3\h6NQ)'C=6 +*8h(r'SL\l8$m<3S+C,.3-"V]j`UChFkd^Sga,\-aKD6K@:au"3FKFV[^?A-mmAH, +d+KRaFXR7=:V#=P]@&u.=%>8J6J]mQCD5'>/FpcQa^o/G.^URqG,1'X?Uql`L9>Wn +l[tGf,NuE=SJ.`]6?H`H*?M%!)@JM]69dS!4#)W<9@++oH<<]5K*a\T6[s6R^\cEb +PN]Opimt,Pk`lV@[E[p;D)_ebg5e=IcVG1e^kL,WY(ZcFJOmGBrRZ:q!IjqFlADbt +Y./mlaTj2[^4%d(g&%-6C%KbCAsZio`#Q4hgnmf6P,2<7)=h4e(mqss%FtGc2bnr' +mTbj'Inq-`;^0h@NXg(8q6PdmH"$i&.*t-V-ZcjKg8\M/`4kl, +0BQ?"@DAEDqX]$eh`%9ZCo+kW"225M]sAfoIi."$!:$J[kb"FTr:FM*aT'`bLc_LZ +5"-_Z.GnKcM1n1\l\7,-Z(T8XjaDB +;C#&ZOs_M(^9JrM+PAN3VX7V;CKW*&=P6c@hHdk,UBIdo,_nVWq=%'r.`_Vihd-8M +fj2V].KAl9mjZTJWQ5)>pJkdVs+fos#\[ZK&-8L`Q3:BADO.#`V&W\.^JU/2bQ9`a<):o0mO+m6tc>`"Oe*oi)H\YDOfd%q6H +iP$$N@'CkgTH-NfInOrl1TS!Ag!e%+Xj7%k!WXt`\naV/R3_8>+=!PW.VmH3h,Ti6 +l5]A;0X3$J_>f+5*[uSXHZ)W-:K;j&+oauWX7H.'\F:;hl5/M7D<&f$q!,"+9fHG: +_`.9(O&&gPa/cnf8rpHum4!8>#cJlEFLeCXC<=U?jT#8N^WFiAYQ'm9EW@9`:e<.% +iX1q^hZ]mll[RAsR<>UJ-EI@G]>R\jEOASG4.uZS2uYK0L\cMUcq6dC@M]%8-iH9g +BUNtN::o8#$S^GJeqZ20@>G(O$h>\Dn&!'DKY?)XXo,C%#Pf='7=#+Z0M)ZE''t+= +W'E/YeVrX+"F9j+*:kSr--Va@Uod]FDpuYn2_71&?$>#3m"1dWa[s86VOi"*1H'@' +N"5/%H^0.d7"OYoJu8AHSXhUA\uM&K/IrDY"S@pHFR^U0$R/f#1!R?WNP))^"\Y<[ +Z#0EC_=A.[6a40[E,>i&HO&9^M2lbQ'5dB*5U0jSNdA?iHAG5M`LsA0Uq"=)0`^J& +a;BT#d:bEQh8c>ZjoleY6RSn]W@#d02Ti9WFJoT7ksFmi[WPETG1(SV"l/!04jZPZ +0o4,9:p,VQ^QtQ`'^3#0oEQUe_Lhbt6Vk8>h`M9dnDYAYg'73CQMT^2 +@#0=TYD3)K7b%H`0^bPD0)b>:MJo]:IrR4Bj3u3/5JI'GK?O8.A*N"mQ9R@>r.6_B +O#_'0.`Eo01QF3NpcJeA@":>obCL@X$>QN#1XO;dm*?Wf.mNXnofngN,%*(2!!@N) +s.M%\?Mr9(?iH5fY:*UI%"B&TrrD<`p6LrGp=-hEe\h>gMuLbD3jAXPJ:*:#c,KALAHjY5"_5SdbSR9]2+MW.3WRT"U3Yb"hU*DT^`2d!LPp#=m.c +j-8:GBnZM=_>hj`I!,kp+9'6i.&-'hrpFOfpXeDt+TfBi[d;bL9;+0"[Ni29r$r:AMp;"4MVd2;5"GWn_8s3&4Xhqs7gi\E^PmDJauf8i?"@s@QP +[t[,[0(/:@oLVjF5't-^;p)1ZPu!h9Pc!2'X0#m4/YhRMTaXY&VR/.#Zmg(#/u-$. +UU*(rX^j*Sqn/&5$n_jeUp%/2hrk,l60C[X%gt"fI +KkE;a%=526^fBWJ*ra6:iZGB"s4RE7PInWij]d.%2Lk6[r\#SZ^YjT[WS(4U\0DF[ +YI3qDjs#1`..JK`UZ&9(Kh.r++OX4 +e7/$%iVTYHHRd_)".EPr%[>W@+*I(N8'gn_#M37Wsp]b +o!SS*mM:bZC!PHqkA=J>h([ko./r]*Iu-b7-R+'O(feP5R,iVs7"DcYck8!`F^o6o +0\7a_9M.*!W)@j'&eCFtS[?A,G=>m4`]WO]rcXIuRq^#Te*,BskQ)d:i^OIdVbH_- +\q&UoJ1Vu1g0r][&(T'b/.G-T)`h";4BNNW>85do(uA:1fhr[C;A@Ts&+;+[&_D]( +)FOGIJa8cOejd$I&ac_3fN0#r\"'n1Lq5[/Z7R)68>>r).Ad*&)H#Q0WgnYt6-'7i +5QZ'q7c:^[F?bhlYEZc1P.It(D66/ElHicfR]3")f$`1"#B"@NDel'6CSr0>VOd"I +?5uF97aQ9$og&oO=)uG..b\B9g@+^!58^A]"]'IocV:+"f:oa_i+8aW*2J%0JqafC +]GU@[e]=Aca8Y_HZ6-6gIsWL3o*\.!k2Y>Z[c4+c/,lu*`UDA*gmEm^CL;D&0n#$D +`@5&k)p+6Ojc/nBrsS\o`s0)L"&MC*L'K_+Gim(E16+2#oo3_IRf>/a2 +hb]M9?+H]8I)5impoc$qJ(>pQ1a[CQVW-Q1:\pgcKnJ%X&j`snW;dUL/?L>L/7%@7 +Uor+@Cc<[^YqJ?)?LkmVe>&3NO3[qk0h6M(4aL]rQ1Y)\ql&s'KGK+/iDBqM^aI1- +s6lHbq.r7]o@)Z*SrN[16M1#=@t)&tk/T:X]CK>.eWj]Q>f(l6H22(h@ddQXqWq\@ +kEdC-\fCFg[4mDeJ,UADo6@J=IsQ>gaT'9cQU/CU`Tp!eC% +^RAAQ&W2'NgeLN?S2LP>(Q4NO'/DCSi8/:X+@eGZ/a^e(CX4&8`2Ck%<\`;Y1>>DW +Ru9$?$Sr]G)5R;%$a[YK+TM-J#5Gs6K%UUqWF0mLAGnGB\goR`W\]!;ZMJ;lqs1Nl +)mY3fQicXSD!_2i*-m#ll$n=FgEb#Es#ZH*V#Mf:S/GNpiW"N&*o7)F!_Co>$J'KJ +lb%qe)IMHn[fDe<(J_RMVHgb?7m8AOYEeFZE1LU4c'3-%jQgsa3dj3eIo62Ar;XD/ +o>1H=:E_0:qfiG;r%J$Sq.'H:MgsRqCrM0`ikqK"3!+1_^/eI:;=RTmLk'l,KI7RW +AYORo.&6hJISf!Y;qT,3mE#M7?.gGCR,Ksc:b-%["\6?P]aYiBp#uJ((I/@i5t8Z, +kn(cIW4SMOu@8Pk+\f=9XZpIp#S!5#QLYW6[m +&,nL,lbWT*?(F9ji.#eH^DHtoIm>N`dRu)bdOMT!at0^"MCNH%8uK&E,H^OZWr7h/ +$';p!&X^tQ!ctqr0L%SnT@1,us!NcmV"mUL0='gA`>!hT@*7:\l"Zf32,-(LVd0bI +N4XN>BAJf#62j&@"o-th/@QKh8cKa[77M-N;-QXtB3&KgPZa6i7VRuN_Db#C%H9^, +YeVO.GDc"3A^!$So^06>K;SEBp*H%gOUmFA4usGKe6L)Ped"FQ0bDLH5HdS,>)3)2 +7t#3:p1o09F`/B*_+?HUYa2_%:>I[[2@LdRDj\'[:tlg8XU,uJ#T>*!m4&Ta08XKo +%dteR)m0TJeqRDpK)@Kj\SALF$>hDnh,d(XY7PQm6^uNORRAtM8$N1o1_cqmdD[MQ +$rJ9,U^f\V.-+_I0V+*PA5YAJ"`<<)XQ.cA9BbC^<-L(Ic:?/hWk4KZ;,1J&7DH$l +[^2R:]9.-D"W_HTT'2o>lCjd9UKn_4>nYMeNmM["BTL66&-RKkUM'Ka!h]q:&TF-. +fr:XJc\6gp#pLK1(V832^U]m0g%iRUo82oU96b0#_G`N/I_-cfJ]X_ZcM/E7N#>cu +D/>J@^^q$^_>ba4][6bOBXg;e`#B`V$AS*p!Z&iUf_bjpJ("AXr+BI0D[3VT25?uk +Xfa"UVJ8PB-'gd[s2DtWm4E$Xm43OqdKb(S(B#m)=ub1Ar%B"!61JlTUc[ZgVprgm +9M#J*$1mTH=8d-RSr3Hj9ZS==!rfQeqXc-"F5,r_m%V/,O7q-jqR_M2miA9QE")Y? +]6+2$J&))VT+:bhCrCr'YJpG2#%$[!rr;uO;Q0L"q6.;L@G1$h97`/M`83NgFo*.u +A&?43cS('&HTdm8!If>?qPQh.nIr3Y)#eo4A')43@Nd:?Sdp85BZ*R@,G*a5roZuf +&'Z`?i/9\X=LuJ=0VM1J8CBA*$lQc6N@c1083f[nm'/2:?lQ_e0sA2QYJ$CPfm"WhVhO.bENDWi +ahTeO3$<^YIUW!r^$C0E0d$<(o%2R*#%tcc`BW>]d^q.6(?RjBn7SHoG8HGpJB:Z) +,U1$;.5I9KNQ*/pJZmZAlEu#A\j3CYWoXC6I,cIgD\NiH_m`nuaHI+?Tsg'nj@T[0 +P5[R`?`pU\H`W\od`GT"Jq<&XTYDq.@%fHnNPO?c@'L)i+`/=URcr\EXXiUg( +b+Adumk)lOX'k['EX&Ps@"(Cirb*P5/1bKeP]u%G5Eu,DrY"O9WH4jTbWd__R$(Or +.'"_nlC\:\rt?saUd3I^s0;M%MuUWk=T8#?1\fkW5h>tnJ%`9cYHB2MJ%#S0"*!?4 +ns:#31Z!LVEG@J&r.L(7Jale9G& ++(K$fiD3T2c_c\P*`] +A!pJ(AC;0%FCf&IH)"r]=]93r9[;i]U5 +>k5gb-lu_7nWs[scefQ;_R@N%GD&/pSc8P1]P-BZXn9AGUAnpjrUD`GO%rUq)Q-j5 +:sN4kEBQ@Y7/+ejerBrnpJDnN<+@^io'"[Q;o=);m)6'4kj_ +&Bto^R9VS"$o&S2*m"=S4+d7+Z[d?-0iR3IbF%_38;rq'S_,LDVn +i9&Cgq(s=OVTqUK2V#U>gBn"p701j`Zuos0%HZ(kAX%$HPl%L0]5&gAl_t+.Bh)@k +a'37jH3,`/#!O<,s&ML4dYaV6-^G'U7CNb)a><4u^(C"oV:Q@rW7;%s02TPo&jE,ci.jc:T^Zh!<3//Pq0r7s4s5Q.@k=VH9,.%pLFd,A`>cq +mhYruB!G5"J,(Kc.q2U+X8Ij^FkXc>QA";$jWN*/-=\*d$%%%od%Aq;T@uh]Ft<6^ +h*J73UV1pa+0,A/;;1n;W#m(YK)@hs)0#I@^n"Ci?^qH$qE-t##GMdjs/uZ:_>iD" +Mu2X/gt.mZ6J;Cl`>,l./\cE4=Sh!UfFR/?+9F_j)h.Zh-q(k;=2TN8-jonPYD7r: +jW99Xf>Fl3S)5Mgr50>%Ic81CiGTbEI:64P?akbSYqGC:RH565b7?[8gh0=SajZ3C +/B(_o8<>J=^c/T-Un91$c6)E2oX/YL)q\HhS;s')[`[t!1.H=>UT"ep5U4YXOM`0Q +q2=`lkPs[o4@/g>rsptB+9G$dRG`Ql2tLAURq:8R):Pb0G,hj]g>#-sYAnU7\f_1I ++TfITpW/sglcGd8k3#LMnps@[fC.Ymlti"Gs/u:k5lO6$SmKbBYJn8nPcloboaW%a +hrlO8-i`p-7_m@L-T2$f,9WKXmP:iTbi1*04%mfk@^E2T@P'KBMG/8s;3B8A2s:Wl +>+UbaEJ@RSR'.41 +jf:G"Zoo_T^/E-^R=2Th4]VmOT9N@".]V+#CGtDA([HPp!igaQs6ofV>knKaG<5"0 +p:<]m>HA91r#>=:K`#WSofk2/0.#":b&t!^K;.'BX"m9s9[ZP>[+m2bZUd`[nd;k3 +b,;90X(s)ip%UX.n3=2"G=mpae1]&D!Pbn)PlE#F[(Uh/ecl&FEj\3LIrG;#nCc4Q +[l=(uFb&@U479m?+(,.6gM2,%.^-AqbRY8o(k_i]!/n#0W[kRC.r>&3Mm`]bDcTk? +e>&OO8q>BmHqk1sfK$[2'r&&%UB7bs:\saL)*&U6$uE +.B!NpIZXR0$L"5foHQ2OcmTWc"->VRBk!/fSo.401p!RR1T2c-:!TPk=]@sN9pSLT +k>2Vu'K^]eI,s=])%L5HNM\^L/g%4STliK)R?+eATKUi4@kI?EIImbu[,:,p;\&HT +'RG0Pjh4s`p(@/ao>+e)%gDEMG@g$JQWJQ]-0^>GaWI-HI.'JObD27PhNe$6k1#/u +;_NEZG$C>Lm8MLVT&V5*fP4_u\NDe'^":`CEePP+r/rD/pbK`0e^ +nTOjD07t.IrY84d[8h;]!AqXR^Z#E[8a+Vs=`K\)Gq(rc+Q9=EZ]=66fR1L!J=c$T`j^Bp)iT)8f`a8W9``!=d'rpR"#JFbkG +1[gK%dTqQ6-1k07'E6;pW6Z_n!:[b-M^@W4Q45O5!i?Irs2kBBgZ0otK'?7@I(Gh> +F5F!h4)=$'g@NlT3B.(&gdMM?F5T(GiCl9m'+VJ.OU(IOg@@AW>k#MGbQdlI%(RL1 +SM!^5bm!q'Ps;iBfj/RqDs"(%>b/PnmjKIHJ%YPh;?#Y;B@6G6qG.O1,1KmbJ0Vkl +.^M0/M>^6saRC.aedW)Er_#\Xu[U,E#QZ@0\[SE%lBZJaD +L]]pQUnMT1a.fAj.ILd*@q<.gUKq#SNi\q_$jZoJ$6TWNl>Zrg.F;/:m&ak2hueRe +GJW5;4HP&:e\[guZk!RGjU]u4"LSDCNVT;\*gHWGG.tPo=Pm%l*[nK,Q+KL9IDW!U +C!E`RG)`MF]s_gb^4)!Fd6CXsS#l-T<-uWch#;&d[h;g85Pajd+Q[ONeUmTgT`->; +:Wd%FXoZW'TCb?Ns(eGG\G0-+)s('(=f"\j%Y_8m=pk8hKYH=5B21Y]5 +Lg/\X/Yr4@qA(bLagemPPWtOb`mFl#!2'4uVF=&I_ZugAd:]NS>3e;:)Sg-j#(3(l +>7J[2&&0CNmkj%_&'3X?.)k?#Q\"u"L.](O)Lr*tl7-Z)>/7/;YpsL-E_/kC;SH5P +=ej9uCrOUNMf@2C#Utl=?::MPqXdk(!je*.m]Z1pTmp:H[1@EOId8+S7HtlU$h_k^) +jo<3rZr`pk590=@d/Y`tP9D$=$NBO*i6@JX>GhNd3CWu7/5F!!084h(-%^@VZJ^4; +r*PK_i^3#!ZSApaa*r8JaZIP0]QJ><)i%p13uIeITs2"eKYMjr%)1Z%1LZn^N;WBa +fT5eZ!,_i,+Qul]Ga[dgAWS%,T9$O..cR!DTeGjBHY.WCPNsHC2]Hmf4Q3A'@D=[. +PA[3mR0gj@I#KQsjc9lfJc>neHpLiM:^;3;-,6W4^ScY=8bg'.K%4&IC4R.NT@mHT +Uj"@`IPQH!at2oZ.L,DdZa2E^UE]XQFB)lP=As@1TWJ+sHk=kJ$Y-Pr+mR>WbK0ul +C&s5tr+Wk1J!Z&SeRjj]F/CM!1H+rc/U,LVL'UoAcGt/h^l4aGmE1oAYpWh&'@,BQ +430Vs.Xs"!F10>XB/Sh?of5K/NaA?Ye`Nkh3McSOjX0*loEW\r+_' +P5k1SK'i$0s0V[(SGil2mO\M\*fS:%E=a(!*KFTV2NG2tB#0rH/>bb'&aqfil?F'r +f\>hA&USs5o;enAasr\9kK2&er'sqalX1U;A$+ +`Iek\!*QR$=lrU`HF23dII2^%DUQl;9^%uDqBhi;&O-^A^TlI1D?7a/X\eM>o0N@s +Y.e%O0*DYFPtF>i/^V:I<[O2k)#`ANmcfK7!nM*mBo*a(UqlnE?F;gld=a6 +p]QcNrUECd +j:Op"Lk_j1pt^bW5p;qD;,@co$)TO/L5T`t4l)ndpu +IsVF3\Y&HWnXqO1LOfDS\:PNHrbt$^lROE.c6n'_-L&e88^Hj$qTJT^hlKHS=WMKo +lnGT6P-;.s1Z7-lhU49K0,<:.Ii]^Hn4arruIu4UQ/oP`g?[>2YN1N +U7]"MipU)_59`UIfL7XY5$j:;b&;'>o-XtBV7b*(((0"lZkmeu/XAF]]LI8Q'/M0X1mk+XO! +Ua'+CUc5"'ZlcM!*KkY*0(YlF_J!`jMGVKZ%D#60>R$!,sVJ#9.`b>#7GpkNRG +ZA@HMe#?lge_/l1^EIsPS25$,JbgI=#af[Q0Q-S(+oa!(m\D'4bMSFLa?Q#U^Ve." +(ii;2QC^@>b5qF>eCb03js[VY/U8Y*,:L.%<7JF +P%P`0fm3rt0?f!=Ls,qN#Raf]1(GQ3$tp!f3jV<01Ig*)"DnQp[UaI!:a6&ZCg[`2 +OOln[\%qc\LoNRFY_kSK_]Gr=L`5h19fuQOTrPe&doiL8.5pjc6.t]B<8b&h>WXI> +?a"4sUi"n%C'O:eDen%dCu=RDpilaJq)^\Ri.$No')ZZleW?8METM,<[[=,>VJ6a- +Z,WH;+llu>nmD%(;I6&YQ-#l!J5TFBQBc7H3!/aCEf?_,,![e31MH[$IDXAdjm&:Xo`C>;DSl3OrX]2X6kD2% +bk1s(!!,gqcp#RGr!Ds^fj=fG5:M)[WLB&9R,>'a=B)\?AW98YXbTA8>bW&*/Rk;M +S3*jqhI=R[DbiI^8[pLcjEGC6s0=*=q41/bg_]PHiK,uo"M]"-i;Ird7q\bZ:X_Os +Vh$.EO6>c::=*o#&,>'ri&_/FhgcEnJY'Oc4gbb_5)FaM6C09gP<.E83IT@YTYHh* +CdQ8o,VGsQnH)0?JKoG+ip8oBK.^IF+ANCX/Z"Ls;MTiQI<;gZu-"6oH/e^VDl1n9jl +a\qB/<2q^5U.V8\;ij$HBp4W-)-Kn"58.l`6jF\$)rX:EGXM"Be$M'CU%e:YDBB8_ +@Qs/5M_RG&`Tr#&\t06\^_[e#5<5)*m`D8k%OODgJ*Q&5Orp,b39=JSa-+#7^!\4K +"@#G3Ekpsn5BaV#R.f?gtg_4mU68+ls5Bq:i5Ara&fhgQD[0[i6/R6af9Vb+GN\ +AqU*uRV\r&aeKo+p:lHdq0DbQ^UO1i[826jpb:@or\Dq"m`@617%&D7'_(rl4h_Q4 +(ZQb0HAV`qM[t''KcWa8a3Gq">.G\I^;n.m]I&ZnO@Sg^T+D7-&Vab^+WFf[qQ`N9&ThTC64gN#0-2l/gr(1Maqchl4+[]Q +Eeo@XF%0nba6M_-_6e.pjWn(s6pi;QDR+/QYo_B5h,h0r2X,#.]9P7*;GV,5@OPS7K;*O +Pn3s=J#fa#U0bX2XMet6XAUDpf85C$s$e`p0$Y>90K/2Pp/od=]&KoPa#u0soAmoY +s4P-YKtY=+*"%/n=Q%p]l0k1;I!D*#o/%2hJ,;ptS[/@4Ve;s#StUXS$A'Di_^"gJ +%\Yh)UDLD5^Mgp^I/V&\]A]ER2$P$ud'em0VE0jAJYF]jL^N>jSD_5rck0lo+>ZQm +1/O&H1qYAY+H"_d9'Zt]FOJ#$W0r7+.&`Pk=s*kJl(,YAXeCuf0M6[;V'?au/YdJ: +:qMl=mE):7,%)-+kZpdE:ITo8BS9bMKdo]gM,L;3'I6I?aqUXC`XuTP6EX=%QAtUf +T^r98LR$@:m-*+/g2!2*]=VdL*"DmV55a/nZ/GYSY/^Kd4mDiZa^8@Hq".h.qkrGq +&^V2\Org#oESd/n^=oY_ougFsQ^*/=l#!ajZJg<:c_c*(6h@E8gOQcP27Bk +=2@FSrbsC"^EpA(4r]s'?YfD./RG5O!YKIk=TWaX2a,Gl@g9#8DAZR5r%O5<8`K +m(OYVLVF[FNgI7&8c^V1i4Y&@A@rTnpD<^T72*D0EA)eR^SfHAk5&ANEG%IcQ_smC +cFB/,\)cA!j"n$tVHfHBa/RC-CNJk-Cn$;n;)$5b/M7$HFRjP0KuVsN +bLW@#e:A05<8biH@?Ni1TJnKen +3t>dH_n\^D7)^l&)&ZqkfXD2$GETIVle=G.lF?GR_O4^V&Z7.P[WL:dP>J= +<)Ueh@K?>00bcL!U#nnq(5k5>;**\c*Jp^H +#se]9Uer6<\l1')&IbI4L/Y'=.?qYZ,Y4d<;BQ7,,aEt.JJ9SU!JY.G=GV8FUbTebk?WO/j2e6TS'.:G;N +W$-AX/ph#,8L7kTk#7[sfDM&tHCf\6*!*`1jGp#e1'/3lq01GdPJr$$"(n2TICM-B +YPdb4OS*6PnZNm?)PI'G1TZc>oKVXXrl6u6r/W3iK`D!oQI!SA=IJ)sVhoDss0Wc" +7`,9us/U6X[4rb")Y68oGibikUKn`90@moDHTJT-;sa@le`@LR)XQB3A4CZSK]PcQn,CG6F\/b2IIPs]a\%BgTs'$KYi.&)8_JH>0 +=Xp-?#Pm\.V)iI]<`?#!PnJg!$iY1WZtp=@rY;7/_>O)0<&%`4.9&k6.TrHKcIXP( +%E];Q'3mq<_c[S>PB5KSSR +NUVGP?!9f!6e?iq!C3)UC0d]OLS<.n/$it`8`p1fD37\N0p-C'%HUVr3F;JSn\KC+c%Sl)W&:S4lCZLjqCk))CoiRu::DM\8gQB6U_#^LE. +n>lHND]YXeGDuOKEI;oe-`7%9.:Hkg?_!m=GohBVJbJ2N3i'P-B4qhW2il>Ch\Yq= +Nh@1H"oWR)i-0(]YKZ4#$6:*Rp[uqIN<.b=+<_9n+4eaDlcfk(=WQ(_)VUAhg9Wo; +-SDVbRbf&[km0Rl6)Z(82$(N!bhjC8D[`O#78TGJ.BRTgI48o66/Uo%O=mU;UA5YFZ>Wf]mr'Oa];<(Bf&Nk9qR%(X3l#1g/1S4n"nm&U>MHe +e)ReO2k[X(dVg-5rVTX5[hk>qnc-pN8cRsCb!0g?Ip0i>qf?I6&[n_^1m2,a$l'V^r6[g-ZE4(0ZoX"-4TRCYUEtNWA +L>SD[]mVZuf(#.%O2\b[39JbDZ/GY3limFLk%rj +D)5d?1bbeLH#>Lc,VN,RMA0R>nrY^'E2e3br6H2R'F;=\,@.t@Nr6.o[\+dfW%NG$ +QIT?ekc#b$)n6gbb@e31t*epFNUoEat0`bl.7]h#3LRfHd^+>s"@,u +B'_P,%MI@&S`.%5I5ZW;rGV\e&H5aH=W2.[>+_]IjD.Fo&f#uOOq5Mr<;n47J_@RV +J&;5>+Fh0eNc@c-!Ue'=cW#aF\eNBWSLPPM.UE+E^CZbN0:\Wl:JL>Ua"]@`"i0ai +Q[h$\kKWe3ZTdf9QS[sDXBEK45O6-Zj!CdO/^oq9B@<53@:>.bZb9t:a2E)aE>C). +!SJ&U13LH-c"O\7,XXT1:^gsP[oCbq;"QH,SN2Q;I+>O.2&?8aCHC+ZY'`TT^3P6E +*@c#ho)6Fg.;?bkZAO`n,F&-Ls1PSD"7D2erm%8Ph8cF>:X8X*?8ltLK>9EGjdE%I +q;:Lf^ud0Qs-mk:->PW_he91>)p8Q`QY=kpC/qN:;+sZ.Yjc4edX1p'^Ibd4V$]Qg +j?]bC&Pj`+:e5gB!T8I5"i(N42&1:?NE]a[Nr"7aIs=#VokqYH$PD=\R/H[a!:>-W +c%+CIom\KZTCb?Ns$Il:f2l(^AS-(i8QBjY5cmN(X^hNWF'#N7gFNd0&X'71jj#YToaJAKtauA +jOpqlJ+h??A&APZ*>>-;"8&SD?t3cH\0)L`hjkM1Po8;`3j^[pUcUTD-U,%eBg>HF +XXB"S*KD6Dg-\_1#\jjgrl=SjU,GNj>%4Ggs-BL[D)l?ha7=.C`;[dhOS#mZHP6s^ +"g+R1YG;M+i\pqe1aGklWD[I,1l+`sh/\07&'fH=IoU_u7ED +*Xi#5Cs3EVQiD.b&aKCG>u1D/mfH^BCOX6,7om:,2rS[Z'-CS>:OtLg?FJIUYXXo; +Pc!c"f`)Ss)LoU'AGq"%rf]kK?TQ'?Ik"j`>Q=GAB6UgtJcD5*;ok@k1&4CSCJJ+o +lG#7[MLU^P+;D'L+oe=j`6E2pW&M+07jlV'f-[[YVaiI])O`$h6Eq0=dLL1\eJ,#S@&YGZt_>o\tSTB$"K[#"4 +#`)D6i;\j6i(FX<3>25CTstp506qGBluW'`LhA8V!VV8]Zo8\FNS!lZ]lc%"m6_Ef +&-jA^*!h+#G]@rGo=LdtB3XGh2b!Xn:$AdY_QT"j?Z[k]BoYEPNSe/LI@i]OSW5_0 +@h\k)X2W'9J&Q(]iW!BV^SJ=M,(R=p:u%3eCr-d7a2l3!&n9RI$\!+*59KqE=@76+ +K[9ZXW,nGbcF4oXop_#>G$_Yt?`P5]Y:\.k>PW;kCN*c7cMcPSSr +a9VeGA+'2Cb0C.iHd,]".g,_<,N_5KIuqE>TCniUf,3.$'2NCM5YXKRJ&)(,J*_Z/ +_7fhlp^9[n4!tA?PD,T"5%rh4>mu8W-YA5_B8\.8M_L?B?tQDMK=)6\kKQa;OML%i +WPZ%g'*W0kH#*Ak[H%'%`jl3U!<+!g_42=i!B1:jXutofnGcC5/nBOjb48\S&.g;+ +Z>3Q#(!EUf=qiM-"Cd*tKW,_@+9b^H9N>H.VDhT*SYt`35f!ubLRK=/DsHM:q&L]6 +c;FRo)!#$.o&$7\o&!km-Dp`YLV-n_`k.7*p7.N<>K>6 +>iL"#!_iJT%TJ7m4E&5<3)0>[6%@c]gb!@3n9rF+I%LV8[AKgABaMnU*hkHT@t@Up\X4$n"m^FJG-ukl?aoBmiSHV +I$43f)-lc_FcQIW"b?I*"n]^C^i/@L?_!d&f"U4cf*kD>\()0R/9V;GmM)52ri#kRei>,fIF^udm +[#tbJJ1;;%#p@c>0QZa`@]dB>gKW.=3n!o:O +Sakm_QU:6BLCIZfD[Ci#BCMsK6Ja\P06$jI7)VUe&KdX*^]U3M@.gJQV^%#XSO:Np +cbd9o!qN2Wa7qU`hoDpW:Ij60O1]`nqSV@@3!'T1]X^9'p&.akm>HYsil+&hcgM2A +i5#kr5=-%L"Mn?=6")5'WM+-6A;/@,$Us19ikV\N\t1G$O>UG5->@ffgc;b*qhdHblH"qG#t +EV3q&_8=Z*Gm55>B]n@6ho3!L$HS[.1d@;@^X,.61VUs4s&@tP1`r*P.[Z_Y"K#c= +Ve6"dlni>B*nJOLXIUpGi5^rBDUcBp7L)!I.=qub?IK?C`E9O#^Zs;33K?4qtA6i-&X(-R<`C4`irqeZ\FlN +1h(((,[6E8QC24&Fpr@1LRC')@:nZXbm(n'];:@IIK/t?4k8($99,3nTAMZ,Dr4dN +]S5pk\RC,i51`cKd9)S?rWdPb>`Besr2\6+Wmr&QarH*ZgNF((:m(kj?U7af?s.n@] +?LMmX]^BmTla]L:q`<)V]sL97LaP[=$^8NPqhWj:dG+8KHr6.%eRA;58*St3T2*a_ +3^Ea1+(]hLM"^KcmF,4t]a=>r!.(=Z4kY?,d%O)FickWDD@^7@<^o`bSN`SfNC2gd +%OE:l"[VeY!(K8IIK"NtS,[\Y&_fIq2Ks8T#DVk8]%mbMiJkYQ48`fgr+R\\?V_;b +Z)%Fs&;1CC;,K"s#t\rLP%cp%!hku#LY,IA5R[f:W>b`ml4^0`J;f$:_#?QOfDg6.kkX)JkA9k2Nd_Q[_ak]eo!Knrsg; +s%r!q]&322)4+Fu]=^,br7kW1VoDu8)7e/\I]^fR@Z/jKB&=72I`jSLZ_jBe6cc2e +FAN_+Th'$cV)HKECEsdD0iKB-.th&'6+2H=6On]'-.SF39")6g/Lk09MUI3(WVkSo +]Nn+Yl#i7Qi[]5k<1X\9)Va6p=tDa?[,(cnXYdb +s7.1-Prg3>s78l.6tK5ks.e3Bh]R_1M5)WMW*4I?SK]!MDcOi/s1s&j#\[A+5618` +q;897U\q;oYP93Cs&::qr71(3N@%#paT2?nq#QJ+%H[noZ&cVbAXiLW1]ONp%,3-/ +qBd!Qh\H7Fr,#T$r7fr33ni4O-\b4?e&DH8gXuS^oX?R,daspRh$eJ2LHF-Um4;YL +NB%!2\"o4`oeiZ@1a`5?"LUhg29B\Nr3,!o88?X#=ga+ZW^KAYM_)SCS6urK6NlVQ +UU?@Ag<)Bl7A?qIfhe)OX@P9MB%0helT&*K:,^2.?*_"2FqZ2%0q8*Cq&o]iX[S$! +MiW%HSEL.P=+L+mNH*9?n3R''H<63CYqCfXnVqkVNUf:_ruA!862pEiq6L5K2IDOP +`BX-iN.>a0T#52\pmFB)O'Tk9Q<,glaqMtA&qK8)ABOrIY9?,9-=[!n"FtrD)B5.p ++;$0%>AMHleXGp)1#>Y7LbSTA0jJY;c<&k=<8=9NCpfs:DUp3o6%8pM^J'dsk*Lmk +J5!nu8=1*3a9$Xa8s'5Tf;U[m<*P;2]EH:Wc%otINV94f`MOQ!DN-KZDCiXkJVl3K]f8XDmhD%*O!Ol!pK%4c&-m?Kkjf12Cgg#r4j!Okjkn@7M:6gHs>n:,p?n;Z\J(WZ(8kFiss/$85P +l9c&NK(l9s3[A,L?4=3MK(9W@,qlpW>A4GR,`u#o=C%L$[Kl;9465OBi;]hJ'IY6r +4Jt0DZllM^:NYRhd/5s-o5FBL]A4k7%c@3O^'-SbMfsf@k!`5b7TG^AEQV`MQ#W=o +7.P=@$G"$EE7.CXo"XObTo#MRGF/(2aC\^icJ"nF?/;]#qTN3*;/HGPf +>(B(j]Ia28-;sgPqY3+"SB_'\?*$=28,6Mo8lVjS!0'$BUIipLdrd^;Q:.W`CQ4`c +0Ya$X[ntPF/1CM6s'U;=9TXa@;td6h=nO&9s-_FRM26 +FibZ3q!iR/.><.7OGm+'*H+q'29:r['JT(;NSFX]2p'!02KR=uS;^VmX%A]7/BVZA +j`X0RFXsm1W,4E>iE5_Re[rf/3VQiH>9,NbY[E3;9mu2S;U>D9DaBS +Ep](DRR>4^)^ctX8rhbuX#8t"-Xc'beCg)aA^W,k^JN.Kr6OE0fD$G;alTc-%[U>C +af).30.fgG/VsSFs*+[b8[8(u/7_A,/1L^#Z5aF`7Y*(/XnkgMWXWptCH"Tg!"iTs +hI>'nCsq.F\jRXTT*q[NagYrZRr#$7RfSdrP>[i"QfL]VD"qYZ?[O"]%1ij(+.[th +5f/[VfeQ\Wr2Ze2\QUl3IhVagAlAO!g7'#Q>HPK=:$S6\AS/7m]hOX;V24S?`""7A +7hhY)Y^J"9"'Vm>_n+l?kIYKr0/bL:2_qPFTEdPLs7`9)7aXkUZ2`^O615gZ+"a/3 +61kJ>Y7@7E%h[^9FW#V67!rtSI56HLeUM'/+2%/\"+!q%:M`$uM2p\kg"V>A>Ztn] +boBATL8'RuN+-.?JJZK#6G'pETR@b@^$m3fLk+lS!o='!LGs(>&!fH!G!,]#Jc,+k +k]lk9Ls;P.J-"gL";-@(p^mM1;ZB86\5@?6[%016Y)[(\PA&37U44DV\e?Y1Ti2qQ +;.b$iKH>!+5MPGl_LU+YA'T#ZF+&DdrS:V;pZB-GReD/H_Z.T[J0tR'*q!<`Dhm6+ +N?6+lOUM$,;0dDU0[`o@.kaO,QDX5O*Bm-F35nIZiWN"j4SZYu7mOB)s*4&]Hru\: +Vs +59;ah.)6c[aoj.C@2sC?%>s@EIDUpd@(@ob9QT-g%_mer]Up*]qo>qYWtUM39f]r" +hs&%(QVWpuIN +.H'aq4FXU6euSuCp710tF3\^K3k[9.q!*h$5PO3gU!J[E4[/4XUW<70]@Yk5^;k#; +%[mH2[6`iQciHRGO$(-DRb.uJO]KL?p%e.CHR=#>p'L2Y&VL*XNOd^Aoi\*g +Z2Xia+\Sr\U5hI(.bNgu#Ad$36?+@IT9TA5s5=1Z?bkan1H<&>C2R3dF\6@N;K58p +ED]D$E@gn"WY3okPlIEiK*]93(`$kWUWNVeb$4qhEa5#r&[4#,dVR=32hS'N +B=5:#>-_!OhE\eUO=M./.`<8SAE$jn-qbc.'FWOpIQEOZb2oFOn$[?7uUN#lY$O)tt0p<]E!8,jDl02J6B-dg]GnTOGp +bS-V"m?_Kn/`@q]i;2Ot5#LX`.J2I"H>&eeWpjG]HIrV`gE#\RTOGJ='!r8PhIretg?q>Gp^o%O,AR3Rq*,CV +qF!NMr5jOTO8t/i>cpPC:.UG5C(Mk'+PepLK9(]I%)c +&`+lDCZItM.q3%cM1Um)m/PKjDFlULIn:XEs+DjW!c7i`:]?f`'Bn;$o`Ol)F]7&ae]G5]uW^8@Ad<]NSH?!qTW&R=E)To7>ic>b.]JR>6s*kFUQ]%#L +P.2E5A?.bkh\GbI"a^+Q%uqE1>gFFIrBrq-SQ\#18'RrJZ4J%Y"=l(Zo[ +T`#%l:Zi`^>&_>03`(V0I^\V43>DCHltq,if\t!uNYGh@Fo(Zrh3Z/%W$fGh*aSNZ +@JA(YiEcNZkb&OGM_ST"-0Q,n:V'lESB`h]fo5PqEE9pHb%u,>a,3X +\pu)*a>?lqMTB[p*j-*'MO2Tf/`3K=(H4ba,7JR6RM$()W; +\sf+cWQ(i;.*<37V0HDTJ=eEZ-W$nG2CXOtPf3'LUUc)@eHJfMq2lD=_TR+V(@;M_ +gjpp.),db%$pK6gX7IM9:0on=5D!5Z[Hddj[JgF,g\bD'Ni\F56MNI=@[kkeKHHD.>1Yp;+^t +6W7$6A*3(#LXk&$C2@;[9s(MZ2KijTCIS5!p33fh3a"^T-ZbrM9ek;<>0BHT3\]9S +Z'MYDk!\+a)=PhS1"r0&N-(o!A3jF'Ig`-W0t"J6Xh-3fX#@B'K%IqaRSl3UBl1Eq +1'l<3<9[5;2Ck.HE]28scgFcpp8U(ECUX]kD](Fi2HdA'?etEYlRjmk@Vah@>JZH) +_W/t3RRd5;KHo^W@>m?Qqd9HRC3At)mf%:fBQ%t +1/fB0On*.eoflIGP>`T?IK7"fJ'l':eQ#2+'a\HJnR2?Ziom\rfS]6EplJFFbU*3` +70**iJk%P`$JC=:0hSCaZ]oM*0M=%C"J@q+)hC^%eOHqZDp&RH9KbRohI4CsjI6;e +&n*[@(1`QJ1pcti%/^+:3Z7ssWpu\^rf@0n'E28_Z7MCf8"nY0<^FZ?0l-=6_2\:C +D2-=&L>5NkG*a@f7%Z63:REiS9cGi=$\Agi/'VU`FdD;quP)];O;/6"gZrY2%(Jq-)a?PKjks,q># +)F7=jd]9/-Jf)'EL/>o(Q;l@sKa)Ii9l29?MeGU1m8$*AU./5/N;FM5*b]P<[+"iM +*nR<:h4F%;pLKW2UE*Z">(*PUIEpF@T!7uB)nbko_[!#HmJQIBl3MNX%+kc&NEDXT +1AUaps%._15Q1lB)XrVo_+UMsE*6V*Z62E'kGur'2ALI?$J3goXK+#G0N\%:/,]#S +I]B^Ho($1.7>-J\^uqVQ,e.u^LshRJ7$BaA8JTXT$QH4)4Bh&t[aG+q]AJ6(hG#P4E6`k3q`\m-S6a2XM%JY_ILm'I/#WMUB/F +Z2SiQHm"&N@ZWT1n9=U"B<5[=07CJlD6,+M8qGC8o\oMnTsh>EQLoajo[OXHJKX0a +V"^8*]1m!F`q1kcZW#38d'8-S>um:MVam6*Vr'2,h_JZPH+`:Jl2#8olgL@Om[.qL +^!r@_glW-<5LEP[C7t4%iUV&bPntgK0CRq,SIsY$5^6d9o&)g%V.BbjItKW%('-Hh +2Y^XUc]W'G1*oRE"'^BNRH3Fa\PiH[o-s\I=TUdQ^HdkbJ'kd]jJo3_&3adBZm;N] +/.V*App^'k_DVQ!s3%73qCE1A$!T%$o6I:\Th<@Z\sP(9i@5jW@c]Zt(haoYpk\Q, +<;qA%5\N6EW5K8_3^EnqGIu7H +XWtn>!+EOEr:ft!H%*+`rYV.#f?-=-ec1m-Z2"KXi]nE:Hj-0"%)3BD76ji]>nG2@ +`6snWKK^:(1Ju1^1jCO>fT'S7m?Xg"#pD@)T\Z`V^M%`!>3C:*e`D7\.r_V>h3C$^ +$T6e=%,Qb^^'bjoI<+d_BkW8UUk'JZY7-b-_o1K(:J-g9r\gFpa)dj#jN=dh[B0Ir +Hmqru(bRHKeJ8B-$[jU\?IiT[f$\[]AZi_3DZ/,O3r**!I.&DR?Tg?`^^X=c,,Mqs +OW[XRacsn&A^Y[k/>6Afe"I3i1'LJ(Kq>IA^=_FlXi'J`DYIf?S=f7_]JN$h2G/I7 +ddHKOqfhcp641B7hr04dp)iX6HoPX1_4:IZ#9;=ks,2MQ?S:+5WVrT:,>a9d5l2oG +hk%4PmZbW`Vk(RaksE*2i(igeM030tTL-J[I'S^'(OU05D]\\nJTVM)02-^U21=EjF[j'!jfc"S`WV-Pi_+W8YSR_MUm% +=a5'\BOt%[GsP8@?9hYB@!P$D,XPK3L0GK);">Ef7#T4*V%o-^j.u^M8bkkj\`#Su +)#2u6#_ND]Ob?Xb!(;?;'+7Vc?j#I#&Pu,0dDge/*('^2m#%lQ*Hl<+"mbe>!G%J6)Fqb5]Ajpf+fH.0!/ +o2s8gM7QohlYk\4E5u5DpA*jA95T"cG8;1c/cb^>*\7EhT-(6@q-T]Th)E6dEBs0C +U@aS#o#D6`/M0!Fk%r3?:sX_e33Ec/F00o>_PnY\U<^nf1-?K:<_ccnb(5apT9+4rJ`.:7' +"9-Ioc,#4BD($YiJ6[[r4list#6=dfAcIYcai=sljcrC;6i[>j`"ReKARB<6_>8sTF^^&@"&@AAhUUP,U!`gZU[-GW +a6Tj&&%("SeBWT2G`['sE^$*]io^49tK*)7BrrV?S7>1", +"*@H?[.d%HG6ZHqS`n%)oNViO6;)f)bdak)m[eh'ds'.LDkbL]M6TIu*Jk]d,Eip< +m6[#jTkXg9KfpnoTOJ-Mc\^<76A#Od])]4]'CGK_H[JY]5K`_u;sY!'ljBQK2*_'W +n:*_8]eGgCc$Z2B?dV'%rBU_(s$]]^_>crtW%\@h%T"$;j2E_u$$M1/\gA#W;c.$O +Ndu\MC#dt8p>r%OAS@KAG?F'J9^6jpm;b\^H<0?T\)-/HcL["GkD7,j612?\[XMKJ +5KNT,nq1[:J\&uHAQahn,b/'Y"U.+jgY;X5?'8#iP(`RT_bEEHV`9N7Pf,/4k7tuD +$_l^9iTd-[;itEmTkAGOCKHAA@"n2c[F5iIH,%F;6L$QXo,a(jF\;h*,U!q#PZF:s +Q_[R+qUMl.UE[ET*SIJ$RP4uj2$BFd=Eh=+oDVLG8%W&ul#h6EO*G@=]fJK,I11<# +DbeqP&-2/o1>\L4]YBL;.0*YhVt1Cj07ip>^GtHHeKG1E51gq#%t&TV!VlX2k/S-1 +hjnA,J(ik:1JU0kjo".EY!8L\7mXg>]_]e?\o1BSIXTH9X.q;>*-1rGT\I'dDK!h4 +^H1B]&b+k[pO`Q33!.e9O$Nn*6r)@QN`Zp@%7qSO`+D0UWb45'RIrUa]frdJQrRqC +`&E#;KSd15dA5 +i'bYd_M$Iur5qiRH"_>7E+Lbk+&c;V;H4-^&5ldN_CZ2X'nlsO_hBBKij't(3W)j0 +JeP!?`IRurr$Wum24QAi^Q'CFjG-&);#[YrScYil,+q6666-Boru*HgfG\#sY`](A +<6U6t1Fr_W?nu38-C3?%,5oifX^fu61$e+-Fk/+lr7U`Pliq7&[?:#E3UL`n[Bq[i +/H<0.f(QJZNqAA*oWInHmLEi)(ULjar6HALbol,?: +r;ECY^s#mr"hl)#gcs!e<^L5N2*K"`VP$#$NhD^F9_>P9H86j/=l\o9d8oC#$Zig< +A#(Pp\n(+'V)$tq.AWsPYp)O`_t)ssdK&c$"_aOO:ZLqB\boE!r^(&!5`iqI-%Q#r.UjP& +!)Pi"+!P)%8JU?Dh@:+D\Fq(f0iZT'9`I +2a@K:^GLgSRfMX#hVCu?2hTtZ,sIisIC)7e]SJaU_L/VIrVts\pAZMhN"79Qmgd`P +mm\rhd@N(fa&U&F6H6^8p35u45B2nc;RYE\<8'_AXZ$;="X&^NTlHc_m0Gm".VX-; +k>F>jZ)8Q\5XL2kmgY +nT_'@RZ'0,`C2rn5n^m\9r@W0>Es[^[ig^#+!o\_.a8@LjnmYpppQEsFd4utLh_--spg;sW!(*L_cr6bmN? +s0=N?SgN')Xm7%lE.,76mYO]mo=aA,I`t$3knK3_YDj-tO/$nAZDVE`!dm+C#QFa> +dG?5MDiQkdo,hI=I"0r>LD0AGST=Z([*t(K*I^'4q"$M4J)0McYIiV?5.?P]mJTh) +\O.RBUqlMCT-%A5nZqHc95a4BHU%`3A'4,j]m^J&q-E;lJ%d`.G[&fuF6(&ikcpSD +@7s:LRFQ&rBJD31N6Fa9]@!/%AB=nS-FVHld9oMKf;I/7;erci(3PdCQ"Z7R;%'*tr(%\Y6Ga&h<hnO[\8`5Um`S5eSMA:)seq>.=Ur +CL_,K2:bh.;j_]AJ46)J0jshWR0$uDNJ1YBP?D+r7fFZtq'l,DS&i]I,I=3m'DrB+ +IoCE'CJ.9(CMH@d>%^?O=%@R\B*Q`3Q#;fc9%cG51?Y&B>Uf@?Nj.G&"Vj3Mr3ahB +S&bVBa0XsK@Sp?kkSq^f3OF\i4]1LHX=]3RNrq&MW(53S)1kDpBWM +*c&n:D9O&8a<%bON*8/62Jo"$\f$lM]Qq*Q[b9e)%^b(QhmWJ[a.=M]q3:eQ%?s\S +5;hrF3lAQn.sPH.r9uuYaT%*l8c\crdtCdlR2YQ].75-L5Lf_kJ%:jo;5T94kosi1 +5b,:Js-6o@oRA?GQbjJfp-G_#6;aIBR)%`.E&hpgQ#Te=O94qeUEPn&AbQWRKrcl>eop=oChPW$u<#&q?Q4qJ^lX +Z,Vlh9d/\hI4hmQ?#Ff"YMJl&%`a@:(f,8&T7V[0T6Peb]f8]4H&VG(FXHI(>k!5* +O$6jf90GL$KNEMLFdVt#."K#lJsZF`"[Oh4s*0G#rWrJ)55HeskT@3k-T($Q!UXV. +rTlGSURUW51[M%!m=lg!h8(qK00\>`jb+oQ1]a/`7J&']-qC4h5Xc)\?r--t8)XF' +D?Pt"&!ZOBnpGE['Wq\Y1Q$hPnq.h0Md:U1rckE'D;#M35NRCArnmW\*4"T(;QB_W +VMX=NZ8fEN(2;cj.2eaDe%eA2Cg,:We@L@bH)jlq2V5,!DEYV`p#gZ3QM:lTq$iVd +%9E8u<pA+Y4ei-!SIE@ad%R0 +ardf=c\iG)#uTgg)Y:YE7I/Zc*_QCkfn,>mU"TPqhU;G)4rS7J%=Y0eNq5o%Qf,d`8TGqG,s?+ +qsbr!hr3bINrM0#"T3YVe%a0is80GJaZ>=Z>]=&n^X%6l*)J(L*Z!](ol6&=Nd+J! +7DCSCrQfo+cVOC%53dWG@/\(4^sa;Ib:R.Df$Jj<@D+Oth<*cn'7PWNml"+^Z9`=e +!&U-B)[si2]&[%3'#\.J.]u5QA!'7`20/W'&\/]Pgfb-7BWW[L`=J7*_kc'A;,o]]a=.Lq/e*J?!8s.CTtKA]IRtB`EhZI1s!,A[o'/d0*L,>M>l4uf)PQ,E;#U^q +g\buP7c8tc8fRj3aIf2mMu\b_qLAGS?%?(E6MV]F"D.Z06P(qeN%BmT7J`WV=3IbI +@q,HsA,ma%>NJZ+r%lGH*EXG6P)RtX1m!36)tlMC[1WLSI@]X$h._Mtktq0XUBB#! +Iq@+sAO5Y:j\&2g_PP +CMtR)Jm`M42T8'W%]$mbF7+;%L5G';dk+e%>Aioc1 +N@?K>"%p_qiEA^J6Y1C*][OT&%X"DLr8"$AadPR_T/LAVS`e-JUZ +MF*a>fHLQDIVml'5C`9pSeGfk/=l1UG*FWRE^2Mfhle +)oBlW#@;(JJ#Rkp[4[O+hR0L&@p)LV^8K?#Aq\+dJ:t4Tb>.d\6)q$bkR2lph&YDPQ+\">0W$K^_.RC"h7cZ;.s7\M# +o%b.pf:b@!f`)+*/HCA0^EeJ0'XBL]=]spuIYGZ3#g`E?q60W:hin8-ng@cFK,a<0 +<>:E76Pi)t&47jTO?*m9j,kJs)3Y=3![En;c>ol*FgZ$fBA)?`g$2^aRu_]]DrZ#5 +VB+OE>s?h&^3+`#QM_kR?D(nSTQEu*A5B"7:D@_k$sn*`ON]4KitgZbC.\6!Ol3Me +U7nRuT9KKLLSWkk718fW8b[n"6E<#T:725A.cWUu;58lM3D8:m3knToT;u"jqjX-p1tt7E +7";9E4k3.ef1lF>U$/::hbCc1Md@8dDiC?5-A_Dod3j#-0,>SB??W((s4pP!S,#YC +48et.a8_ZPqY*"Js4m#>mgDnj\`in54`8#>F=6VRmld9B]uT0b9:in]8,\TqrGmtn +^Fsj2lT@?^ro^UO5Fh9Cp8$<,a*g-5m.8&fpH&V-?^p_8@ES%qZ1U&1!\a3MhUNKc +Zo)O-.<)p`@AGMq^lm&KPLrftg/XcmJRMI2cB-eCV,@Pti4YA7M,mneK/OJ]gT$WE +AMdYC)Drp%Q6a1>RfHNr1.EpVd"k#RJl!=L=Mb@[<h/6(Z8;H5Wcb3%0?5M ++bJ-=@7dE-/6l'o+fu6<=,2G+Prsj5/:S']PUcZLotO?ofn-mA[di.AZuBLO.C'qbl%gTCP42rDVdUkAT(H +TA2\;(\[i^q2&A7^j$WKXY6%`bs!hE4ZAlg2j?*SOL)\dH[ftZ^7O-EmMno5c?.*] +WV0S[NDo56pU+kphnBhmBtn5^rUp=6G:[oR>O("KgqE#McgKi,KdZcffEXY1*@nn/ +\'"E[(ZbVS5qIM#ncSl@qLn!\rK$8o$WCWXOK7oNC9.>0VW:MRm/VD6+"$Nnq:PW8 +!+-9Bro\7[WiG&./)u3/nlQOOIS.Nhm&;RV.n('+D_1XuO,&-Q6%oIN\@LaAn%A +Lh^I%L$U<%XW.r2Isg/^%IJ]PE$Y!Zl1sXA^*h!o..gg8s2oYZB34T[)> +^l\OcPlE3(g\d+>!KNr^@d'O7,`+kYBV^SVRV5#,6ds!d\lcO+d8$;"-r4iFV6,rY +YEL\(jB8(S%A(/5Sq`^tbdVq&[X"?cp_\2Ml?E,@G581^Ke%@;.<`a&p;Dd]`!Q#K +KnZLXU/ZS]Wh:Vs,\:O1-jfTs?7]_b$u\oUUkX8@W+M+*--r?gcl%`@l"[RN5UWDl +Jr4_?8%WZC]4Q&o>3gLFhWnF!hm*$>q7;#GLeo=(dW^b8d% +s)ZiupuNRss'53nfA1m?Sb;sLN;qKWrB7WVIk+g+Za +gRsQ*4*kqjS^k,,(0-Sr*j1iaWc8.EbB\Y:?4`n4+Kc3*+:qKT@ELrj,bhRUEI"LY ++q4@@Mp5$1Y2P(Mq-b1JW0S5.XL8,Fi.$qIganA^O[mnL,6\BjNdI;R-DfM +hn&ne@('jg.[r,_>NWN*c?7r*f6,8af<#G:G:`H(>OL?fW+Y;PhH`YKG+Ta&4?ZYk +f17@3s85:H`TaDG1YcbeK'W$&)iLGGQDKY(PFj4`.@rT'I/8L'r*S3K"9'Z<5koI^ +]nN:=1GqY!6Ba$%VMXC`&-2hea?S>^HqA`Qae'rs4fI> +7lR@]S#?US9/]UP!&Xo]i5b@K,nX?,_/HU;f*YJ;Akb/truD;E^F=TdS2=F`Hal9; +9QT\dD>r/!]H6J7H=Fteq,,`[``uiAE5ML_:!qje`cI_qAe5[S^^"Dps2boPZU;YI +bNR,:kj'JWgLF@bhk-_EE/OALC&3m'LnE!)5=#Vr7m@ueIBu'*Y"TC(Fa+c5i7m0:,R)2+9:RStZ/H2pS"9+ql0+9%3!=MA^@0AN@ +-An[Y,EWE4+,NsZ"JCdLBqcL1(ctoF61?Nn.OOO.I/5sY0(Q^h*r40 +9eJ<4]@?4eDqOI-ORH+Ti+(%?1?XE2o>@t]iQAf(GsRCj0MV`b-OEG/"=6Y;i^@*M +_WkR_n\"c:O?[qj1iHHkBNM]k=0Q;'T4O'f8!27k"au\0*"q4inh+^7qb?^)bbEg-O6`o*WCaP"/W;+l;?5/(5?gS]lLeV2><64]R;TX!WTjEr]cb#s6J\@TC*3arfl<,a*^*SL.8uANEP/*pA$j- +iV2QWqWe64=*f6hZpu0O$"%M+R'B-ID29(9`Xc:;#HTBQ9kh:8WBa6h8uFPSUEQJ9 +/uc6:Z.0KA@ZG.\75l*Z!4]i^@iJ.I_=3.:)@bE7i2u73_?8>'W_uQ.el<< +'G+=N5m3:&LamuWEI"da,",TP.]j2GF2O9Dr[/p5<6YkLo7Ags?pH4Jd0YF7i_ZBq +%/b%0LKbH)J+Gn/.fU@?jJQ%NLeiTL,ls)/i;X]o3`'Ht2Uo%7;"fP>la*\D9G/+0 +]f!Ks0a%=aJH*^s?r?Nfen,\XJ8Bs=c^u,DH!iQemd'I_p +H\DPbRh#(,RI@\SjOV;AgCEpo44Ln0E8Q[#rB$)5mf?/m(7Z6p3]gMcDPHL7<96:? +WisWemQJ@tInB:JP5(_f?b!rjNjG^YN#lgNEPr?M"3Y +?8;\__gd_oIt#?5_^1p6kBpUmn,(n9s.c>Q2[6t`km9H')02H/pgA1,Ch(D*;sZnN +r+L0BE(fI]F'=&llC*,*Oui)e1N=FXZ#)AGX2Imm=e_B;Yr8]'W*4&*=VY\5L$9j+ +)GuSg'L\dJN#rHW+Q\?>m6IVG.Kki6Kug8]JMmE;afR/:s5X.?[JhNqqu0n#q^+T-=n@h;$WFSMMpiW2p0l'A*1Yjk +fY59tk:BA>"&"!I!8IX;?,5GJO@r$Xr6WIZ_$uc_:%Q%88-/9no0V?#`Vu?XKDi82 +&);VIi6J)"4+Lgn4&6:Q3B<=JB"6dcB[PYr-0HZ!/l#JQSuKFba1XCW]l?mD^p<1_ +_ii'#adtVoI.ieni_TC%ZU"ME)G:"nC.!_BRgT.M(-j'>K*Zu[!'^aT,h`7oPQ$WN +q&d0J%R7+b(LP:@-O9Nn[U0%MIMEpsO:QpV9?IoarWg,(SI`0L2*YgS11"c7>VV)3 +*;Ch9L?*)U3q`J![JiZKc*EW5jM#J5roEK1S$^>_Is(juN=,J.Rk*S/O[PhR)+nA<*kb\"G2gW-/*:a+*^HCc[p5IH^Z]6+I +E5G8^Qg@!4T.CQgf_iqbcO\KDjaWY^s"R.FG_])-[klP9;"ai+o](cCs*latHoOfL +ZP.oiT2IDHF3rV-kTC7mOoDg&W!:omOhCtH.K>=I&J-#r4@GDGZk@s=6o<'d]BL`i +O]LC,03h!]HD"_cU=bMT-9]+!ruoYe-`cS29*8n:B>0W..>gJn^^%]/U?XEN0:WT. +,:R>UGo2]u,&&`o"nY#9KHbCQ9b:pW$a]#N3s];#ABD">FI1\j/hK"BpYP-o +SEDk/i]LN*3l>s,iOM_k]dJKi,6FKd^0`6@TS4%We(?n37Wer#HB$m30nplda`Q9fj?LMsSG6l-!Q4)&;OrJbDp2h%Pd.:H(2U!I>tf8I$)%l'aP +s7[t_?dH=_pMQ=)62MI(5l8<'G5:N_.mTQ\m>lVGrI/V>+8<>GES]rAq`Fm55Ch#Y +JEJ!bHN!AqpE'q@r:,mK*tJPU^HLr\qnq`LJ'dZ9qu(uWcOW'gIp_dFq/5XCc?Ob] +deAZp4s)ES^9unnb+/KQ;neKChhpj?;irjIKZp>kbP[*c<] +S,t8k"*g#\jPHaZ*!B"+h2QD[f%A9&R#_#Va9!;tp$k:_?!L#SNe;XFG`=W"U'NQu +6hR`mXZ0E7%T84;Ekr/Rr+oqD-r(5nH +#U"ue$K=)DKPHT,2h`NfMF0Pc9d80j-/&2,2l-?bWI^,Ts5j0gEaAeI&Q$;8WoBk* +s/G9h.`nRE(FD)!^ek$^WAa%lS#NM5_#Gn[7R.&YT\?DDDLiq'?6>9PEoj?Z2^Ng5E7/fs)>jCDuTe,oSW[`4,CW):]GZurcU6jE[,m& +CO^t,>hDb4RR<\U,J%-:-5rWpfn9=4aA,#D3=UgZ3!+a=nIR1Fdn=D^,(cDI#^jq0 +*Nd]-Y/K.i`7)H=RpP6t..O34rhl@f('jdeJ%X9PUOW8/-T-JV5_=-4W@/+IL.ZAI'NOjRI-!FE_f'CXZn)T%dPjp,LB_\h%l, +%u]5(b6"3_%$o]-YMA8Xh!WNlAA=Hu5:g<7e'bFfm2ft\Gd>HUR:aM&>MYogF+9q/lesb: +CMa(g>!n(K@98q.'d9-L]Ru]2kd7#+f +^d7.ad5_^9nqn.;Y9=rCU$&NP9o\lO(`uMI:M8$U$O$5HMZ7&/3<'3R5O\Ff.K;W8 +\GecTho;\o4oVtKo8id^nb%Va.=[CC*q]Bj+k$B$q-s@g'F=X9R-u,Xqu-MMhldRd +r:Y[0Dtj/-pSuPh/p]edr-4RSf-bu146l^Z+oaKf![%Q9TD!PX2!i&PO=P`&B=gK* +F@Ij+`m&]Rc$#pXFf+Wpa;A\g)fAFVL>((1W1Ct4'DbnlH^j&ARUW6=TX(6FRa>IH +s2VS!`DHPP0LjRV_eN1[(.F$F*WcoRNIY%JU61,-`1]e#Lf+qa'SIt2.k..PK`RG9 +(&TcDDuR5jIn9Tpl:#-s8f&T\2j"[PrWeDI8:r[<2$2c<,9RSkr,4;:On)9Qq,>8I +c2UKg-[&qo2`GC.pUlqts6K^)VMsY,+Dn7\!QtF**JqJ#8Se8hMMFF?]gJmC@J8bq +k$%d`ofl]\)f!%WVsmcc4Y*.#Jbe!7_9$Lt">#OF08sKqN+C]@:U1:/)Vb`55d7s(00D0jFFJ/uKujshs/#Z/=(\Ku +rmqt^pE/J"n\=^Q'=+N3b+Ci!"TMEkf@Qha/88:/.`bt48-MuH,c*OnTh$;e>/83i +EDgMP6l%;96jpsg!YN-R7<'Mk]MOh8'I@50;^"&pjV+,r5K77m25%hZq +3GD5kXGo5O%DCERH[iZ^j1W`_'?IkIDpGqB1sbr-dJgI`LG,"]^HBI0ndl8G2?+7n +q,<26^HP'cmP*[V_fe6c@Jk'WYP,f&T`98"n^D4KcY$qk3^ruD$j"s%O.pP.d2+UG +:F@"Jp=!Aa7kY`b*u?`9(8Fg;U'#+C-j6ri+fcW`-]eMZ9ZZ(@*0eN1L?un#6%J;] +#ZZuX++3"Y4WaJ7H_M-[<3dR1OKWF7KV?%TI"hI'#ZMY\-k9gM9E"6M!uU-NFJ'h6 +&qECL(l:Hiq]W,W +*T*L*?V_=?'']ak0b-(:':l[In7FMb"3hXW,Lu":"J&U,pLqqh"9#NfI`[9#'L&F= +4Jd0]Uir/mYFCc%%1'X/\Edo\K!ZgIo&<2[;&lb^O].>e/\oPY\@\IcG=GZj)-MAX +LhN9mG+38oU#Wk%`^q'8?6/QL/b,p-GeTSjZ&S9?]Bce?DGR]d4PK4MUM]tKk.jpg ++j2_TZc:&mFkE'4BWl[ZPe_X3laFlW?EA<0;oq"!:-2TN5?2[m+F*iP3*"R@7.,`s5i:FWjVn++5 +F@@j,a"R-3c#tI/G:2kH8f[:n)XU-8kt3#`8fDMA(heirCPhFqNZj7STL,?a0a?V; +qLZ00a;.EnP:5PYRm,m^1%[rH(:kMSbZdWjojM4B4Y)j\KaU$ROb.=W;`$Z*b? +CgIss5RJJNg9Nhce)B7iNXakMS/U(@);B:G@aA>,$dME]hDJ;TG.RI?rFNd@RI\P- +'itoF?!lOH=Q2Pa;"3s_nKUOL:Z13Hr2J]`W;iJQV1.j>``U7aq.'#c;Ui]sPp&_q +:JpG@Q't*p3^KYqHL%U/T&LeIHJ`[9?MWfb].D(q#>Yqqd]RW +c]:-%1@>#5!V_YAm-m+9GZ-8CJE,6OBO;f,J?P!\^H;iP5ikM!n\\6# +pNX(n>3gYN1/-sE@,S-oY%OS3.&Q"2+%lXV=KZ4=m=-4m1E^C?kR.Pk.=mE1\ +iq5hT!:hMAr-^:4+!gPL!^F9CT-'8ABG5T_fmIn?!hBr:(K@bg +%?0TgqZlF-+0[f^hro1=4'gd(ftf9nGgssoGo)0o84JjTuB$0K'9)_&hc2tSZ.r3o& +hCH^lM4'd"'8N"#bfjnFQ=XZ?N\QPSe1``/E$=>-gQ[jV\'pbBs$$1uMUS!^?gCiu +MC7YWk*tgbpHGel1Ca@1pLC)pdtdWOPIp&K*Rr_bFD*%jpSElTUU)]=C)XZfLgCnF +n(.%nC.2M%U9B=A55qa,]RLs<&b+jAkpNB?Z2^N[^PE\OT!6M\WgB*F#[bFVl=0SH +e9FIX@WY^os*d8sn#Mlg!IpYIp#kSi^Rn@&n]:!6(GrigXSKu6`eU>SD[[Mrgh1T4 +;t"b=iRp-Lj3?]oc'C.[[Zc]Z#93"6)f5E:D0tU9]B\cEl>>;@m2uSdC2@(SX[s'T +@kHpQ+9J2VW"PIs'H*/N(^P0k8H]a'lfY21jeiuHT0_)Is-X0_&-:^=;WkBjSr_;I +O@:>f+mB85qEH32T0rY[rg>;AQo#c`V#D_n!.Oj.NI\l>qk>W'6&l'@It=B5s&f7N +r*NUh!V2TPT1$Kpjo=mL)rc>e!Nc;m"L)+G%[&>Mf`*P+s6r6,eEmE)$68F+QCRt` +nKb#\0['HNZ6(b!FEmlEJ$SFYY5D]!0E9Jd+8uN@",C&=cEq'\ca@[slI^6TF8+'7 +OT5=?IJtVj"-*lp!apJ6r%F(!5!KVgo"R'tGPt?#q#C'HUjr/(X3`"7h_$M4-8ih) +2;8I/ag=ggYY%Ns9MVob9!NnuT>6oQ<+:;2nJ2Ff%Tr!9=[+&*";iPET:&PS@2\IsS,-7[eS>=8[/% +eGi\;MR#0me]HiYr1#?jgoX"j>=%r/!^Hm!B<2>YZSpuXD^ic`;O"Z$,]@0]rjp*9 +cgE!t&\c5c;#e'5NJl?0\!n-$G:C2TO'c8mEf,KVDkH!>n@tNj^R[a-GOlM6^;CK# +YPQORs(0pS47^#J"2bEB\*6+-\L`'VLqpkXmZ8&MU\c3p9!+ +R\;'&<3;+4ZD`k?Z>#OuZO9&)`@r]tMo);fLa"c[N>J\<$e8b(%8iKA$8JH1/3/8^ +]DtU-^BH"#'_rUkpgf#-$8S8@fRf_$5>%Q.eu,;8W(:1s`98om?#Glj'CLb7o^>?.tPJH%XC&RtcI0;@nB3A6_=?`fOsq7fn. +EjnC&7]#tQA_O%SFQ\PR`cDUcdL=jhFBTkGRR:*0!ouBPn@J?k]mJ,QFu0>![2kop +2P^6\W,F9Qo0FT?dF:puDu`\m^79joje,\4d=VeO#L2J=0(cZGq0!GbT*Hs`T1>aX +3[#_c!Pn`%Pjpf\5Fg'R*cj\9nm*gHJ,'&0m]ub5rk?MNs-hNrcYNN.61_:$Rf>FJ +q9*dFn!CDqRa_(K>5P'$c*n3%^H-%QVq9j1\'`/Io2?Zq8(qeAg[(eA*6+K88"(G& +RiM9om!G3A]:0d$)XS[OJG)$n[T[GHJq.A4)[FqE,/C";D[6ptatkZp2(H^+jC]>E +AK'Pq6"re"!SPTM]JFs0E; +<%;o:'A7s:b=0u^+uTKP0uX)$eILMeBE8,'n:tTlZ32G0!:hS\K&KIPkP+4td,iaU +!'0XB(4=33Y@PnMpj\"q]dDIbYkX0dXrD$85Lm?-/cI]TGALJshoLP_!L`Pe/Xs!K,I-)]BCV`9K(G"PMM,U=Wp"#pJDnCd]kGitGR +*!GaK"0DSl6_Ye*'g?^K#mkj4KQrMFN&A=AbV&l[PS!YsZhcg@e%IQdf8tjf7/j^k +ce8BiB>Ldh'^'<8"9mO'!qNZVj^`6?PI38$GACXDp7hJ%,m@L&$9$E*f6$LfLPi2E +MbK.rCL*l>^"K4ZEkD7=I<$D<.t'=tHnilnFdVu::+UnGs7l2TGqnA-hp5d$(;TiH +s8TAqYOqj5oB<)dlA'BEgC`kK50h&hlFdDoH9$;N&DbPqcc;^-rq+Vsk3s:$b`V%< +:>4Spb0Q7E2+%B[.*rD!!Yf+&64BQrLH;%/m=Y=Ho]Tso$Ad6+icLT5*DEg](K)AV +)_^G@D'-BIN`2_5<>\&S$Bq:0,V)`kb&k3-Jm$0O_hn"*)??Fe4\Ejb4E)Ac%hSeL<0cI5"F@F1Rd*N$^n[CMu3=4HAaH9KZl%;9>i3Ym5u`Ln/akiI3a7LB$" +/c)9&rj%mfoe>jT8Z?'/\\Be!R<4nI8L;(nro&=5b`Z52uah6bUnE!nDkqi +s-"8+Z2Ce=4h7BYj:XWjqT!=;T20-MImj&$K'm8-3ZfBjIXBlFeR>47I>0EIRQTJ; +I\t+oYQZ#Q&EW4]pU/DgRSTK't_&cqgm`b_>dLmN"BN3XT1TGND?A5`I]poO+; +;P\2NPhlc:#pM]./8b^^B,oc)85IUY8CYgn,9SAa8E0[2#d=@;1Mt,(J"Agn5?[n- +numF1rg<`Ms,p8-*!+7^(R80f('(;M>6+@mQ[m^m^d)\K',^i#g#Te.!eT59@'bC% +!AN0OZ9/DEb^9C92h+=Z>t4@^gLpi+q*c(s=iC<3B_PtEjkmM4<;r]g#QN,XDZTkk +G&1gQ^\=T0++,;+sG&cZ4+H-RMuKr)6K6;GnNl +s&2tn5:HUt\q*Fl:.fP$8p!^"E$aP2nb@M@?Ib0[%ElpbiWt'1pa,imQ2ZQuTD0ML +_/oL%o!?pajP]$)OLlCQIYmu\$[9C!@IBGolqP#n)_p%#noD/TjDS\g'](#`g/M;; +e&$=t0Emi$4d=U5FGnCSRT;qbkZRqq#>e'"o])7"H!0K +L#4m/r6Ih4pAZ4)W8H5PTKc1KciSSP8:TKULHiQ&J6o9A0CdHW/9Eb@\erGW3PZWj +`gQ@*7\-N->Ir?%6O*-PDG*\/3Q&0IH$$MXC +E0^$;P4`s^o!RZVfeELc.)6/n>G=qM'P<;s;i!1odJ?]/,@^Xs*?%JqU(aru"AL@H +Rhr7['s#Gfjj/c`A`As=3(4$B:ganNQm?r(7@m,87p1^kNZs[fM?MY3g_eSpTN69d +])`r-8KIB"(W&0[i]1XL3>]$#CPC8T&UM!BHG,I47Q]f#/t$p0j.a^G +(-+p:X#3N4/!V7](ATG8$e32&<gF=#O8LAS\<5BV^ +D#ec&RKrgVQ,S#4&"hCe=iB)Ek1pV!hA.Ts-rmlWU3TM,HSf*u:o<)7P!Nl:G-/N* +`?*R0m0Zi4V5esB3cA!\M<`mt$KFb1441HZW7Dm;gRpGpeF/)<\"FuCdCCrQ/HO\N +5d(nVDj@(.lfh2,G`V]Ghi)N(SAJ>-r`J1YkIP7X[if\Pi>h./n:C:*DDZmAit +#m,p5V@&691,I>[5Qk%#RZIe'aWhC*,`[CS9Ol!a[&bs/>rr->RD%.3XSOI77)>uWGS&"*!+"XC]l_Z('#NYDZnSL=V._. +%V49XRfY4(![J$s=TJ*'4pB8aA&l:m[C[#?g0G23\/oI1BCQi+n6?P+/*6eis/4T% +KDuIkD[-4pFDbINgd14TnGe>Ohj+1*HSFga=N&/bZ2BIYJa8*]qZ!\NAE*X$$Yf>N +dpHA&8B<]GcSPh$a.sRRdAb,_.7V]VL/EdsOU6aSnV6"B#4NKJB*d3%O"h'$2?;Z. +7#X]:S69[&AULd@>Po."rXR%Ap.:*a8VmpB@_k\2P_V@jN;;Po6r\gL!padlJ$(mme +r1)"W!ci=jkN;.[aP0is\1b,l$'cEO?e@s_Bl?8*VrD(>@ +qNO2]ANh/F5+ggtT&tEKG^5\&/Gb)TpJ2nPIU#-&c==Zi&-QePTO3QU(\&+LD?2(u +r:gX]s8R^0r9WI5mdH)n#)4aUM&0,_sVWE/SZt,Pm426!9N4l +(cc:gZH5iKM/E[jSdM`:jB0>iU(OYf/j[QDH43qW-G^BnNse'ku-VQI)Ynd +fr5CZSn.NV(mXY'Kd'm7b[%mRXdn_u9^uFaF,8=:X;(I0A1%%u%ek!-pK@UlKd*K8 +I(:0tj_MhM(\sLI$ZtR;ogY-e];"u`!jp*nY7b_R(9U^en`1!^.$5/U3?jSK;eeg\ +C`&9l?"uZlRNdk#W\d#$eu6b0NVT7C*V_4@/!9$r,'l85F'jISWFS21jF#[tGr3>8 +G*Mge&:Vn"c_J/QS`E+3s,#1p-L[de'q%itmK1?PkQ34Ci`?hs*V%c&l?0 +\Sh(K,!0%!3*Ng3Hed@CP +8rUhK!Jl9S!ZV4B]FZ?K!.8$9br@r73JVBnG7b?p?Ha84\R +pqL'ar=9J>'A-SunO7ui]3tCdY]qd^ck31+d%Ni5otN)/r@_V5!`[f(G6)q.&;bq7 +)FncMe2k%?8--inIn:4D[6-f@b5c&RbA`sB8=LQV4)hFKCTQ?]`H/[$,:WTmd(Z=N +#A2`/reL=lr!se5h@KTMftX9\*t>t=*f9h[oi1ol&&l49P6h*(s$/o2Bmc4$WW--$ +)hS)7P7>DC62h'sZ$N)u@R?g]K-U&@,4pHE8;#9&UW?QdQir>W,b+R>gSA@A(Ah\= +gEe3!!XWj+!g41dOqV-kHS2DnM8i[\0PR\h.F&&fj$4e2+i+cZq51[=*ta_%"lT'' +G"Jq +RWa8:@'T)eWTdl[mb&NOX,X-'>9O'4h\tFe2pD/D9^8qjksprMYGW4XchEoCo^uXC +!kt,>5_JRN*:l>-r(Hr`PPG_%s8R6FG6,J(j,:^fbV7BIHom(s*+V8#V4\4kTUC+& +KAI;>Y80tgB2a"Ql0C(rV,6<=0/L5253RXq4A^ehWcmHf3H=ktZFM#KR;:>N2lJ'A +<_:r<'!kRuI#E$l-KY3l(&.SRq=uH9q^D#8'Cpds@F +6Va]m*=I:K-Jm11;b(i.:okRm;8]I5n/]\Z$)$8mn;:/W%$_rhB94H]:b%+'Gc4FS +$'gk^?.<<&oK+f5]rA!5Ges;G(ci>VS4dHHB5t-;o<&nW[n>&kQ)t!idCW]V#JFA) +<.@I4iNNCQacPKe"%r%S._jN5I(r,7DM.M>)W`J\MKjc`:hD%MrEtA#=.DTeKNrd& +:U#lqmcQk*h\@b,(+1u+/9O(?3KqRqR*C%CLN>JBHR2beC;/](I+?]Vl:W"`PK>jK +OkPeG3ot]FpmY@HHRn3ialbWSj]U(.Q71W1jdU9F`ghED3`cRJBV@b0<2,._9rW;? +HH;X@Zk4>5L!Dse7%cX3eD4s5H80`T$6S1%]UOI!m^mHWri\ms!:^i?AJMZEFpoDeC\YJgH:/o-+>s)VN0?fuN[4/E&Ys7Wr'q#;#ss,?pr;^V7ZNUIlEH"#@l +q-f2f"'-fGD3+l7[FQi%0*,)\GDn#+LZ]V&6U_eOe +^b%gZj`1`:!YJ77Cg&,=JI$cqJ4'BZ!J5,t2?f$_"j"6&Lk)!@K4k.d%@'.9J!_"! +`eA+-O93AuYn>\T9Xn+s3.)O_5sdD7#g,[pJdBFh2$*8G9`a7'EHV( +^g?#Manic^"T\@b\s8V>0*:0"&B.cKr*i.tRfX)$ruF^)E!2D^lmbEa/>sRcJ[JJnc`!d.9*p!,(>!-\Q/9pjdL=7QNEDfk5ZEg>#/i]P`M3o'!Y +gu8?bs$rDY55'i$Cg@G>a8U#10Z?@:J*5Op0!s[3+0tnsnc.ji5dKbbrYO2d;_\97 +58=+tq#BJ]nu__#r1B?l"q_EU+p'%h%ndK259sDk3(bPH68pT'-Ql*L!"fbX@iQM- +*h-gP&b'oB2sI=>-2o:@fC,hrqr(iW>TF1bq0,OZ` +!Fks09T!ITZ3NS5T+!0!Du5DN!]#1EZe%kJ%g`!C#fQbi!^Hn(-S,3[70U:(!KN"R +U7t@^L$P3'T"QF&C^XVR/\m'$p$3'eG?WO'l^!"+:Zf>6q2\$V5/RmqE?C0[*fi*n +&2<1e-Ob>GoU?^.48DAtIs(r+,5qhg'StW^!#d?,T4aP[VpMn_;NHPlGTpfKrF%MT +5ooW92>pQ)om.3r0b_6^N%RTK`rbMYIItGfcH$brb=:1]%J&U"Q2-p"EjogpmF"\_ +URo+o4a)li`9u0,4')/p%ja(\!2pM`=] +k<9lKrbP!ro7M'jN[Y!9N2_1[W:]&!l?g3ZDfr):Dc+&HVf\t3iEEeiaV(GhI//V@ +RA!$:p./Lq&ChW9["m\\s2fNBHm8#m13549F#K#)/rb3$6g@"8oS%+8Ze6hKdoRVd +ef9^tD@=mlg8cOPXHBhY/^7EIeP7aMSh-^^^SLhkH0miAkkoHnpP0Cd +?QCs#r.s<%o7l_[)SU44r09j[4(ID37P-6P=IgJ<4um/#CsZV)hf6*NnPI!L7'G'Q +YN&07df:C\12dK2Z:ODA;N!h]o4jVe*E#fodF +s$QagRi,k+*i#>GV$dls'WFb'&)*'RKa>JoB+`X9UeEMqulkJj>d,1"N18_ +s+5(QP$D3".t9f`jS7X3-Xcr^>p]'^g(u^bnX\V0^EYuh[)X-%pb%("gWD:`l3`<8 +8k4[;S-"t9NGJqY=$$I/!_=tp;kUt'ZdH1m&^9\jTssQaoJJ]t4##1N51GOh]oBCF +j4h:!A$'&gCI%l0bm3j!s#6.Z57;=hs-'!\anj>[a_9H2*/_Ef*A?s_NS,TH +s6YNi4,[fsQdY=Ca[e"f9Zab7b-=LY11M,P9VYOG_>fY>`]+$/Z2B4u[0%Y\r'Rmi +68\YXFUs.?qhIKXk_&m[j"8>(0PCLH'5Y`AJG;+>(.a7jMiY7nr_H8.,F>+X#Nt:2 +XNf#_aj+5154sj"It7;9&Gn].$02W$HN^Jao-Yi1,UU%=-Q<(MjQZB.XEAeK)[c-o +#n",(RNKa\.bH=&h?0Chb7RR8^S<2q%jlcB^L0"8?mkmOdXoa*qhqhs`*,D?3$%lkB^6h@>'4ph>c._]gG(^)r^aY_S)^E9lE?c>Y8-)u*k#E# +@'=ml54-/a3hcFn6[#L8Dc[>&WZ&H?3RgPBbYpF:74Q`G)eI-J2W_joEhi<;O]n9r +'r5!= +GlTj&u0INJ0ZOF")74`!QJHdObskiY=_T1`(3SW`mI +X^>DcG.Zird%c2 +LZeSqLQ*HXJ- +kVUiY1u@[`g?16Hk<.8S4OS(2ggrDQ6,9@JGWD_A%U&k@rjRRW?d]!@rD2j;i<\A2 +Z$QMKm"ljD0\\u8!U1rhIkiI)*!Cs2!:#&=p4mAaYo"G]7.ke\BUEC2\7SF1G#T<5 +r#GX,"FiPM0Jgg1/(l=4`AffqA>T"j!Mo4;UP!%mV@9>N#S3ZTnjS*9/Ib:k/F!+c$m4O*rlIjk<$s2-[@K/N@XBE8YFJ`1nu;9!$!V@dh1GkbLU +F`\V;"X[QbO0o#(s-!$sW;fdp`W"3+ncnp-B1@X.s$ZjLNFVb^m+G5uWumrVMK_W_ +E_As?jsMC%n3[24GeugO$Jl94)Za+-noH`Vr=:3&ET%DS54Li;=TG9^fH(%goCM_I +EUgfA\6p+Z-7[Xlb26$X^UBiOURgj1.'T*j0ps]4)Zck;26YTShe6''f;Ucr;/r]eD4s7*\I\*SrJ;ZDpmT\t3J9`^]B!.6We +k6:/&_osd_%+u-cC)F\nq,n7$8I+p+"+$^\]i^/)RU+!s4dDnKPc-\`X#(NigE!!B +(nt(IXBufRQka(l*ho8@X;_tM(E':j)IsBlC_RY$*$/W%iNF2L5VmGi`N-s^/4-9#1h';e*O.oB`4-r^49b0lDm-s@,"b+X&?PUMTU +j\W5k8sQ:%V!'WF8u(l`dNWC)V+.p5.ZD!4L%Ws;4l#WpeFPEYH4%6ONG]Q,gO/9] +$$3S7GH[/7rkDN-^!qT#2l!k:d/]C76`QHFp/!#CbFpDa2n'`]A]K45_FgA>WXGoi^WD\PD>`bgV^hR2K +)E0pH'<8tW!IQn+V2MiM\L:ANk^ +J30_=6iaT.+4R%2gN'66mo.,[r5f@VkXQ4B:S#[cTk^[>inOnesb +N[+[Fj7%ZQs*nC,ia2rOs5uDUhp1HbF@uSF'8DR.)EqC]H%4]3_490tZXtj2[_$t/ +p?I>(o+SRq`p?K]fH22T3\p'73:&CbloHr`4?_m`1#OhOCVGrpn+nnAbdYu!l4g^=iQ%%A$R+Kr(h225lI;hXp()?Ckr9, +';?3^X9h:LBL[\j!KePj5ZFZdnNS/dIk%W$NO"^Aro&@>plfR*EolK!MgpT`bFeXg +($%#U.t9,M6Ds`*9O=DUM#asrE%!/T>YY;im!k-BTgIgB=lrOZjp43TeirZU>3"n>f>rq:VJZa<[mQ-E#HI%lKeVWO/E9hH3(hiO]Cg7&N2m"*7p&XZ] +T4K;=T4m(8e[uJ:rh!oX$M'g-hm(hElU^2Fn33UM\ +J)^Kk_VJUH6/aGX5AP^KrpZ,'m3VR49Cb!-j_^s5&`D5Cj=i!d)AejdM@:7N$r0eO +S?#Z$d\qJu7Hn]ji_/[EYI7*;f"lX4S+Ce^HDcgM^E7k`LtEDLhg(_[C]mQ[![Rcc +;kT1J!$2AhPRID8K_*A"p\+j`gDon0%/&Wpp"^mB)ZK7\p[^\M%m.=?[n=D=07Tll +q.f/oF*rsW3L]OW)C*?n]a7.;0ug%e`qbaoR1ZdH=S.K6m8X9Ri2%jk*3WM`=q^T3ir%">5aA9eK5;P^Xjk-DMR#J%G]^'bgfVRCa"3NLH_0C +=\@d,,]i)]3"'_r`E:u+[+&Q4cpSuZdlIfKY#u7K4 +QHI#jMZ,[-#Lm>ZWClPXjR/=Hp;f&(dZcBqq/VQ=K*""\N/Q\+W&-pi:R*0SjAtM# +NJ;PQ)="OtUQ0$AK3m8dcM4-7;9\DQl@'e#:l7GO5``M'[mLFAr6HUns#HqFjFZ[S +=4P:hs$tal=6mskD#h%m+$cZ3U2t+*T\%k'5mYo9"YXMkb.t,B!)Xh.DDt%M-*F%CqH/ +?k2nBNV0lAp,F/\G%)]3ENn95^X^'%eK4Y7D?8Cm49\\;1*uOD,ZE_o$c#5^8SO26 +nNT8:)?5ons250`>ks$MK@PhVUOk<&&cr%"IqSkus+rK,.IfmcIqSj`qS+]d56k>] +CK6O9"0>c'(DT#i"Kpi4mQcKQUV%9Cne[S$JH*_q!3H3+5lQ%_Nk#<'"S`H=J6.EV +[@lMgE=)tE*SD-G.Mk*HPWDm33<;!Kc8YmJ*:(b&+L6../;gB +ia]s:L=q'.h=;+#p?@l6-Pi'<#\WpkNcZQZ^qT`#,9SeeL\3mQIp5LlG00Rk=TTfH +B-Z=Pq#g+V"M(!F!^ukMWq*hH!]g\UqZW:\-)M?Pcj6QjX]Y7$or<(UC6omhXu7cd +BTg'Z/G2oQoICs+CRrtEh<7-XqHs:n!':u28&/3TpmU]]lpe4ts#:(G(sr8#8#:`C +O77\W(=DT/e8G`e:NN$J%SM4s8-V]G01H[@*`.6n8+lP`eb9uiY1;a,'c@uC17688 ++36/%827*'2_BkBc\oaqGGiJaV8c`9`*4N6i[pPm,_lNt&oX+OT+pD56Qn9"_-pfKW!$6Q>nQ1u +U;Bo0YrX,XjHXNPb-dUW%90$Pl?,sb'cM:[3PHFs?LpK7F#T)(V1-tH(c1FLV=j/LhQGg9P.1De0($^[Fp +Ouh[ES+;g_p6G%>Q1*M/>dn"8A:nWWVco7IC6\Ck?t3b44f*S-ATt3EFt.HTatc%h +kj?gB?3\_Ph+QdR)ZDQ#jWO>B:Qa[J.rtHc4HV\6>)R+E6`ihV`rZ_'EMgFrdMD]' +!Uc4lk"Z9V>!@8I&q/Jc^@"?Q:=+L79E,r7JFb7??XA_-pm]tMs-(Tta<(EX5P;d% +rCF`lJ$>h(+js4;P0?d@m8I+f,]Ij4n'Q7O(.>.lt8oAZk-SJg04e'=;WSBjqi +SK;u91%'@(b\_(VkC>U,56Cr\*ek +Eg.VJr5&@CdKCg-!1tth)L22cjSu=]nQDM%rT#q*aV%EFWepN?:SsO&a--4)=C"TIkD"$?`X8'09%VWDQaUX1TY8@IZeDad`f&2LleT0H +:Q:]ohHCG0&Cbt!f<-!i><>sU!t7O(Y`(F8!+V!-#SW1W!_SrI13#9'q(')&SYabb?#QQXU/Z6s`9R]X\YtN?I +JI"e?Z\V74_oMad%3qY@,GOhI,'6/h[1@q*KD^T+U&GRZ1g0'mG)VJ>+F2Ct]XJpq +k89QD*1uGXOR/L%l3\+'T8>#lQXlXWU4)S$+7[i?JmDKg!tZC58-k8qNJ_\5M=QI^ +(-_qiRcHp8?^/$bpqKe4Z]+q6s2(Kn!)A*>]uY3fnG0rNh/G)ZO>>k,1ZDn)s3raJ2,'59&5L +E*2]tT?dC1K0(YSQQ&WE<*CgfS';Y7o]92_Alr$/>*UAf(*l!(VE!\53p3ZnJ +UqEXAl75B[baUKBtkJXF2R$gLSaaEfutjJea/$5[bQZ:]VSu +63k";57[)(:gk(5T8o,0;\`-#)Cj-&'mfY.B;6?Rc%=PZ8[hou/mC0rKVj.DdPfC) +;"R'UH7=c$]o[M9`k5N8kG18>6Z:SO2]M7Xmn:(PFqo=s&m[.\=3ApM8p%ZL]o41d +HDd)3&b,%pe?QYU2DGK/G"m)AB<.A4Zp#GtMDl/d@N@TB2/Gij7n#]XQeU#N3 +Er'N*FYTM0.YBaDat1KZLVqN5&C+5TGAKK^GW08K(Ungj6HaPkg-=QtT3?A)"Rgk7m6$*]k9htVaiiZ%=(\Y +Gt\WU`r"PIrss:b%X0sdrs)_u=T@Ch_O^Ani;_ACCWTjiM>p-T:gIU/=^Y9mDcWI= +df7>R@kmcV:O7t.9!ge8q3C_>3Y6e(qOdOI2O$pe@!*ZCLkD(6X_%Ous0r#dZHU/b +-$e([ruceF5DJV.&HDdJr'1SG\X7d]JG$ZmIXs:g5FW[5"t9Q0o*V&[Mjtn.aM+m/ +I5QsG*!X(Yp`E"V&5t0$W@E%n.H:(N#J)%8Yi'R"@Rr;9+(KnGL(8#+"$i$MpT'PP@k +6OA9.kZLmO?GLc=^C'OuK89J;fVGBK-`:.f-^r&n)+c"p4Am!as+]`hs5=XCrt37e +"IOtb@/7-=T<0E$ae'^IkZA(1gD>ZAEWYjM-[Y4s%t>`#6<"YpL:-(a6:OELLY2fc +f`d)`#)1c@JfTG1+9J')"@Pbka2b\8`4A8C!<1;Op:l^(UPuso>d1-#GV;?.fN65Q +$^C/TJcG7`!1*ZI_Yh[RBP@Crr4HnBH!:Vr2Xke#964?Rnf/SUlro9_"p?8?!i(Ce +7cQhQ_e&\&Pr_F9[@aL+IATKu>clfQp!VG2R6*"aIQMkV#JE(3Hnp:;*CKTFYeq/h +!IeEp";(TclA>4a=q^GEU>1f&ME`(XrF&:hgNf?nB(\B\2YqODoV5:T[tcM@m]jFP +B(^PHc)GI4Ki75N^/\O6I;>E[Fd>C/VQ1ioZQRH'0RHM[mTL_dG>1hmG?1kjDOspY +^.2V7ZQR:[Qt05;B-7K3i!1YHmZfEW`+4.d%!r\,N1r.+TKLrS`UCn7Vl0\3#7*dI +JjG5K-rV#R2,L,5^3@qANVXXY+]i9oK,RsH%W=an44&$Rd)?RmAdWj<4V..%-1c*Y +o4^#PYSDTAL-B?>n$_1f'CRlQ]/A7@QcmbIV::c.FKHNA!\ZFZYI-.sH?gSA97JiC +qpi&9ZiEUcJpAE'j, +i;]ca2!.;hm,JNBiq@9MFDYK%2^qe1="WI[K`m2"-L#ZMuqcK]iA?;uQU5lBNn"/,`6f5eJsn +>DJ/cA,VILrVUe@pKs9ZJ%oC\o3'!":RQV%S12PZj!.Un/_I_@i:,1-o +ca]O6p,Y(qF^i5RfS'!]M)C4OCZi^h&8&tqB1BQEO*D?<:C_[0#[liAO7p)7s$o-+ +L>#qKs8=OpGX+Zlj0J]Y"gD^@Y[[Ql*tG5$Z*#OaWYS];'q.A[Il;a89]CPBr]`9[ +!;rH+TgKUqM>W51@tV0u63%JSIrPUEBbH7%';0QbGC^4UocMZ<1t0,0VuDJJ?Vk-/ +;Ih/@^OR@(_tq-TfR91LZG60Kn]rk/.k#K?s6&H'A`l*djZ$f442L[k&HMsGA40`8 +'9Ri4)!C0^f.*I%l[QD?EqfDc(Pi(\Ip7V^"?$=oHE,RZ6^9U(]@)VVK*?bnWEe1^ +jT@aT%+P%\e>r[i&CgLaf5DdjemTg90\*OY\n1'`em/9Dr4LCjDlN?DOY1!sJ.k'H +&-8^*KE#Qk>T/MI.6rD7!(tGdQAE3L`L_d2p\[?ZpcF&AS;]#2oM"#(1omDtoD!Qo +fN=o9(rN:#rQ%OCJ4P*er:^H^pk@r.L[#%t)$Yd6Kg_@q"GB6HOAZ%@C&]^@G/OnQ +`nCMlE=UA&p5s@]CQIIPT@T-eJU7]s1J9s\TuqMEjt2nCs()o.EZW`c#p#.GHu:Q560mhs]goEK?d;i&G0&t/$+g_f*8_`Z][u1_^!6P(K>[4oqcT'[ +TjjKTTkbV)V67VI]O5SJiAuG(7\I8)UO^+A?b!)6BN(U4US)jaY]sHQ^oOQXXZkL> +-IoU]?Fq[RXWg"r[$t.0N\5EddOe;.0&H*jjkt=c%Nsm6iqaju&;25F]]WLlN[bH6 +P6L5o5f<$RO:OL`R9A60+5`"%"JSO$k^#A&PD3Br5];SE%r+'7S%+M?H!g=8*qrQn +qU$+-phJR\S`l&6oO8iP;2+su8P!-;!kne'$G]I*e^L_1d1>.gD+Ns`0q#5BjF3V! +r+/1UpI2LI(0:AlX8;Fpdc1YQcUMG0lB?LkK*O!Dh(m:+,3jP&"o_kfTdPK>csG56 +"*JISgG[DQCQQVAWR+oo_biVC8j;$0.DSd1=&3tR[d'I7b.EpMGKKAu8Q!+0BI?80EPSSehl`SN=.OpuIm= +PMSeVf.Kh#Ribur4733pNVUeYY+7S_ApW*fotMlRs2"`rrB^,`>2k$u^1O;"-7A;4 +8"Eq4XaOE5h1,;Dj:B5E`;aDZ-I;PGLqFE+r_JR&'Dil)^[-IF*Z(Ok+0%\'"Hi!M +J40Aeli1Pr)h\14c/4$GT?R4qeGh&[mk/.)047M7@HCiqa;4fd@ejYKgLoY/!g3g; +-mTD__WO"sT-B=eZ42M4B,UVS=lN[)Nt#hlEfp'Bh"A=@lT)GB_"E68&I-V>-21rb?AmK#-/`K=)YQl8GpY5&NJb[qVq( +4SZ;n,2c;h%UckZf3IX5q6msJ!K.uq+U*mLMGrFnFp,/t"\91qH@\^?cU/\279jP.:!#f:fQMITC7C3QEhMd12PFfXHSi"&eiM\"@Pa!JH#_H%/e!P+9IB!rBp3# +!l4iP&a"U3r]q_&_6;cnbE+DNnjE>l,Fpa,8OU(seKD`(I[V+]>bKS\)SaDW5jWV]b0$SO;aEM=A[;!pkCRj^1`/CC%<S1'^g;^ljgL:Q!Q][Xs6hT:2\+28;BNrr,8K0\=u +f>2nIVZ?\o4QlTe0(@[:mou*uZ^Qqf[D/6^qr4jMkL13=:>:5&u9adL7$-e=.X]lM]eE[RladQ-QdK;@PUJb/g3-T+WC> +#Jr`S4C%dlF]"n;h3<., +WCIuWht>1l3%Xt*0_^>7mai]-Q\3dX^$+hnqNWDmK0E6n,i8-+MHBF=1Y +U"?mrKt.bcgkk7ePAIjf^"r^)3,b3$eBM\'4/t]R(C; +FTHP?fp/'O-1p`NXTEFb]Yl@)D$*2Gf8t`.o$DgWKRmB%Cb%'q9 +bT[Os;*l:LBZpJtfgK*O7!Fjb'%"N#oQ;hcirnQ>"Y'?\%/Bk6g>=hjrH2?9s7/CR +s2F`I=S_u>>m'0fmjm,lH=$?TMgA4tIs9')lOtUP*4P]Va\lCAcT0gMGIoWn"nFm) +f_`3H50'OJN_Z`?1:_:BEl7h9qH4d=lXn/@r(6EC794jAgbJ.HH6WCJ%Qp'XMLXXc +Xp,,fs.KU0Y-aMiT+q;Ur@c;J0$Q.#c8>?8s%DX,&ABCki,bapnYH4$@X]FX6C[Y_ +mf22E`J(`5s-&^t[bu@n^^As&_jXiGG6e-a(:`Ek_4Msatik`^ig?mG@>Kb2KMF?ca66Cco]:cr^3R3G1 +#jVH&9gF0q`k7i73E\NG6iackOI*GZ+Q!A%a&?5rHd(af[mlKa>H)+VWU6A*CqN_s +^ +ZRVTnXo<]Cj0NI[:SpiM3b8*7[,hPAp(+$jhOJP5A^p,0JG[k\cZ>X`a4o1j2Z$0* +XrjWKhd=("TmBL/jfZb3h.C4f*`"-HQ\05B9o.bmjO"c?T>Jp@&>H!fofBg3jZKi"c`O0?j0 +1$3X=U>[tCH3ZLnLX&>]kMspsA3?dX?6OX&'YRh(hQFgj4eHlWOndfCUS#QRY>aYo +C%V!RqI'_Z4+Y+.Lu.U>/\jM_!@i'AJ"_NZb=?3&&:C()7\G^N.0TY- +b7S[iij,2N0o5?i9H`ka;r%I`qLVI0(A%jb(Wio`*&dJ>>G,W$05h +!paodD[9_T**C36eD*:4*!-2AI5$.9YVC[Q+(P`R#+-G#2okL`2$`sGhJ9Q4T>f[c +QM^A[!!dnR+:'(n_S^2bUDsqV>uQs"n2TXUFo7b(&-9Y"D1FSOa;2!fGH'X_d*iqB +gIMH#5ZjCPq"D/mt#1Go@`DYg\ +b%32+e'J0OCM"h>>=\lZ3lF.PVS>19I!M+YnhB7uBrTFUAKOYtI57quaW^><%V^AG +C&7h=l+>tfDi`Yc0kPY0T]<^/QQ,m@=*d$TGBofoo3Q*:GGo5n=MpGUgjX=RiA*$kSL?Cf +.aX^.<#C\ZNJ,`nHVj<[Wn`HQejm!&."oEZ0<7:9=G/HkR8RZfn;) +o^r?53/fZ`IP\Q@9jKD>(PLH=V[Lno,'8^%J?'FWm99@OB)9a^cA)L,r7'l=r"i:h +^upmgO%5JL(mb33?hW;NWV^36\;3247NID\VZ"mR'`T,NH>gCMK+%7r[\\RMebo%R +%VX#P?S-V4Q2EgGg&9&K(e*]&$Cnf8oNiF5DU(Eug;dV!BF=7,nWTCBL^lgp#bmsq +#U?B8'Rk<:\puX2'aF#SKZ[h%j(^D]NonqkB79$6M4@Y)kkFA1cb$.u8(#(`W@''< +J"KfXcb@&bq4CllTA7h=E_d?Np:noQhu@6MpQoq*hm<<3Ghi57UGd$Bg"g"W]!L=H +Qi1R1c5lV(DAo_V*1>R^/5<3Q*+)@+)tT1krf[F*!*Sn@KBPh;]af42%X$bKPlpEV +IBCe>kEhH`B>#PUIn/l*8dFAU)PM"bM+tCn$N,tBNmG0GjdMIsR"PDBf`,RAs4e

    e,QZbRcHL`WW+:DB/b>1'*(H"7G/H.>T"^t\gm>jlCao&Xi!UJ5 +(^L2+"^Rk1h.8c5jZ!><.Q7q=92]/nHLkh;f!2f'j7V.SlaBU>ml8h$F1QAC9]PQo +VEK"FHP`+]/@Oo3Uqj5FIg?,GUQF:tR,25fahEl"qSF-hp=.X[Jee>1TWDr3ZiGKf +=$_C!cH7]YZo$OuA?3X:X8$oIn:+u51%Ba$\:HlV"@V_gdY.rL"9]AH$3#K&IR,

    HhNlrg1.^0 +O4is7k'Qfhn(hV/M1Eqal\KMlfGJ8k.A51u\X=D&B0ik&TZInXU7EV.?%>bUQX0XK +[F5)*`;^t[X=S:16Oh,]n]]*uccbi,b4Q;]H<'OLC\dCX78=jl2Z6'V@fj?]`N,#` +.\&,X^/?J(UJ^;6MC"B)8`;8Z4A;t5+5=H@!pLOA(pAO +UUkB(=M?t.H%4K/j^ELFrl[Im@/c?1hFCc$c*@jC+)KRn=MM2?G#q2YY,9pd%CQ)u +C].?:)X@6ilM[.jr0s/WR[SgH>Cp'`K">.`#+G7G2K#"!fK"g&:7Ag/[*ruajr^s13gcj(b^`$$N)H +BT2Q1+SW!5m`Q(3bpmg,7%Yu!V[CcWQKJV&B`YeIMJgqjpDU$_4#l^$nilE,_UF'* +A0)6(I6OV%YYY7N*ne$VUgL:dh@ah"c_F8_;2aCHHp?oQN1/?reK3LTB3pl]PA#jI +CWl.nOP-bUgT)[7?VT[5MX1@T.DS`9kGuZe+O;$h7HD\-*u>U+7=cG/$>/e^aoIS( +9)g7^P_'JA582EJLaqAX>%Q3%JnD^DWIaRrkG9_pA[#+s'4+OoFYn_`]ge)]$+a::$Onum4OX`A6\E5W:-$Z2q;CoWeZs,$kN./o)'U%elm +o/"W!d`fN,;#GtD;keD+nc(i8s%F4/)'Q0_,tgMu$8D6>.FoUfh>mK"0)h4Nq=A[G +2nu$\a4GcCq#dH1]Li;*-cLR4Ted(B5BQ>hH-V)u>USP2Ztou/Iqn+%.0'sAS`?0& +lc+)KY!keI!kq1Hs5WDsT\I!)s8s6lM`C*GodJ/^_KNTXm1gZBD%\ +0OP51&pa*E*ZV=e=h/TnM,RiF"8G*icn;rG#,VAg"G$DKZcpm>/--Fffp4-#?j5.Q +kufoY)7JIQ+=5bPa.<9OJVXO*6@,LLOHtGg"o5(K-`5NB,hqAAG;/eV+i,lfaUo^> +-.N/C+=Gnq5L6C?LgFD>(4ln`#`sm0+U++Y#aHsH;A"WOp5e)1"mt@4O+WL4'#88N9fAhg[t3pejm&YJ"PEi +[i)5*VeuDoAcL@R`s39LpF!WBNr-OEme>Krqs%kaL#J+L6fH)_d;n.?Ss\d1K[K3% +?$H-Q=(Zqp(=`&k?"q".ftH5=JA1I.!Fu +_$96"`F^@!PNh.@$F+5hkk('?Wj+LP8<5f'n+;`,d#gO6a$-+=I<0oGAcE9rB`C)) +oZIR$hbMWAT;=\2L\Ts>HH5j9'@\ZF)LXT#PlB5VjlJ,]m?Z2UmC8'MQ14$'`p?%J,5DG.>NL0_Gd]NJgi0lV%9OI*BI]81t68e/1KQJ.WgmIPCKk9&dbZMB[h +(/#/h(LOIoEZb4?j:=A`D2J/IqU]A^J%B5$rr;hX_^A5A^O*u^d2A"s:O0[%K)Xaa +`cHp'$Nfhtb/<,uXb"WQXB,3?MLRtleI#Bjq6U"Q0 +YH@bUs,#_WjKDV_/q`\;MuY2n+t&lS9K,Jd[)>Ws7g!hW1%R!%$a"Ea +JbFaam>6T[;@[mr!Co27O*.9=mnQ$nP6RQD9a=GdAXr&3>Qs\JG@tAtH[K$j9QO]C +?!V\tFg+=KW)PN`158ot,7o*?\/o5CY,hMn;i%:N6j+)#LN,<[Z7K&R5;Ub8/^la< +_[8[t5_`!2VG/ho^0O>8qO!_3J-`r1!)h83OT1M\',']l''sTXJG=AE\2aAqGe<_R +HX[bPS`u`4lWI3N>>i\D-tVs!!^tgO0eYpE#a5S4A`=!+,SEqp_=5ieAXl@)$RcbR +'Gtp^[23HdiI?P*N+i+H^_@mkIhiS&-kmh>AQX(dGeGFD,+u7l@%`B3VY-I[EB7!5 +q,G8WM_."C\!aqV^%Q@D`eVI$?Q7\hf[e8'I&V%Uq]I3qJDNQ6Zc^"ukL,LV%)2$S%S6q?f4O[ark +g_=2knC,_@BZ;*9A+'ni4rbku3^rmc*orB*1&^VZofjh&^Rd?4I)[#R]baD/ShUFu +]0Q>V*:8O1Zu:sBIS?S]22o9^XaaDH9W<.B)ETjCA"]R3ZSs!c&=^ +)Z4965,JBoj$HCZg3*nc+1IXV^PRjG^R5;gce;(;hEoEB9]!5b+RRp>jf^&nYJgk" +?j3DQ(^j'C!Y_ZV(b7M]MJ/eP1917QX=Ia\,PEF858NPhj@#6UUT+8"%aWqKN;s7f:WZ0_%X\`[5Q9`qU9)]S +!;U\.D#!,#6B8aDBE'@Z)n(/sIe^N+ImudWZ`];p&[8;8j%hf/-2arOp-/ZWO8Amd +m0WA'+T)`uPYC.f'sh)gd]-Dl.X%AO=bin?K::Iqa_K8._>aN1+%8F0Za6)(2X%+M +"F@IAIFX/1W^,Qq8:f=.mnpDm"9-=c+Lo09";n'I"U;AC_mL)f]#^#FnBq)g;%s.q +j,JD9aiu6reDpf +)#mDmo#`nD$?*LI"\s"cFdEc\l#oBGIui"A7L)2lfI"@W/!Yom!i$j4Gb#`0s- +-AdmYpI_5Y#,JOfO:+\"*i1&I+V,q_Did3BJ)Vn2TMgIE)LSf$6WJA8FYG)TY(f,? +lb:-ZiHU"J;p?*@>Ks?qD0ip?o8A7UmgX,`0YHcfSJgqfn;`:A],p1;?IIN`eN;U< +Hed%ph:26_Vl,1@0C"29l,$1)LE@JlHZ1Z^INWO>5B1-TFf[f55EJ%X^CL\KT"b-j +^;A1pih:q)_1Cshrk+qU69mJ+1cA;i5$U4M.j!(QO,aoKNYKcuj;]%/5;eUKh=mP*A$i]eg35G+bLq@r&53MXQk9p#9+ +Rk.i"?+OMCjHB%O(QJH9f'*!07UI5prbT?sqig*Mp"1e.7PD34@1$tI)EY&J*E>oH +bDhN19TQ1(Suc1D^1R>oL3:g$Xh3&pk$ErPB`EPIRMQp&DUfZ_cqlC=WDri'+8);AgbgShYBH!KK<< +aMkqA;TT-EF?R0^*P#]Dg?GPH%GL9FbQR"V5=C_l'U+-;!JJ,-j>7dAc\9fQW1 +_pH-Ep\u!S4RSHi-2/K:RH6s8I1+jSp"f:2=bmUSkr3 +X,c%6irejjY'+a5\CR]d62j,$W*&J^PsGO4CW+LJ^(^9K>],>^s-8ASs.fR#C&!qQ +fAM-&!`o#E[Oa)=XsgU7QJ1oQm6t7b@du\XJ#2pa,Biqhq&B6=LEZ^5s5;8,ETDqnLK<\%j'+.Bq"@[jCrJ[!`5U)uOUs/'IVs&WgE/V*=Z=)3$O' +Bi+W[&j7<<0L(YblB%1iOHb8X*uei3'5J6p@gb[pgAl0npQH72;j3d$dbXV.HfsSg +&.q`F_W$jd!Wm7qL`[rZKSoul6@Kc1!b(.>_M+]iOZ65iM%E=tE'f@dWE5Pg;#(>e +V/1hu+s9FZ$3l`[Tc-Y8[Q:]p)r@:umF2@6fD(EXH;3!Qmr,Elemrj%`Tg@uQ7JN\ +Y'6?ugV*PtmuH0$]+BDQ0D+!]5@At^d,i9?kHuJjbJVP^6`^Y_m*iP%PIei1@/f,P +q;M,5U#_'([*c\urBTN7@.qrTcG760^;7J=_1J9;U>Fj,rU<#,E%]Z93aKj`Y&%?M +5,g1=VFJ]T7Y-`eZ(crWgnS@nB1spiH6Tq#95+Y)%PsIF&9=n(k].'Q2Y9\\shsA9&%:JXbP!5<] +Ikaar$KQ\io$_Rca.0Xere6Y;3*/c(1a<6ZLr:%%JpkZO/Y(1WSVrpB&b3)\nbKu +)`=MECXub@(gg/Y4Y8O;MG$->6Je>%*`'Rt_m)ZX0a7]8=J\8E+O(3:$*O:=l5L +*k8'mXtco-[bU[]m=-;Es&5pPEq:47^i%Sj2sK[8dW9]pLD&%mlH/Mf;*LNnI&9qqBkep.+pn&u3$2G[56uAQK'>XnQ,PdB'&b6/4 +q>km%.-,J^]j +lFPu)8*'\Z>CM^ZlK>q*:P&%IT8&pJ!QZ@G!kqp2Dc'A>aar@Mq85F'Cb%'C(K]6^ +p^V>LUIs[E;7]f-goAomg.JOeNORlk015?'=K@MD?c/tr,6)GYr$sXRl:Qe7mAIgn +r"ARUid\rIrnr8M!^Qa4Z*-\l(J@R-&.eZfaoIEQ7qtY6OST4Cj7VuM[Dh!1Km99Z^Dg#Wo[puU"Om[!>QjZ#0UYHEDYOY4c399De +[-%V40?Ru^Bukd<,ST`uDiFVa%-@'aUTY0T^,\OWl%II1Vi?.NMY>e-g>hdhV2Igd +k.eK>l.Xjfl__H2'EP)G>rMWr7=`L(-]Gc%8H*-0HSeKG@u0bT(\VZ$q*H+hR+cdk +[WY9SED3`Z6G.GA[c;mQa"c-us4`2+^O4&__fOac\$o+W^3\$#c=,UC&,9GecT494E-`;qs4CQ^dHtJg[F,PeC,;5tLS +j`9NS.U/3'!Vg"EB(/62YaeMmrqDUIr<`L*b1Tec!%ie&.q\tZ'S,#4%VSh8#\[)o +-rC*ID/+6LM@C0E\`3N^pkV9?!>mSOj'*I3l>-*,2.;bB[-0q&p(NA2ac%2Lg75(:`1rtB +D9f@=h]XAn4tFr31Y4NpT1sWl2*kESf3,B(u'^' +qkO?bn:,VY-VOP]F'o@58*gUb;&6t-"S9/I;#,bA!PVqZ%PO?i$R=R:6@&K:JBJJ] +$.rKH'>V)To+qc46m@+G,,O>p8JFk/.Ss7cN$:M#>#QHiX:8pnZF[+$"4\Hf26IVc +:/c:YL[$FNeQ\SXZiNo\gRW%u4P/_CY2/,SQd=>+s//-ZrF*_Np)G-43\V8[IjfJ` +0R7smF_W=QHoOb`S.2n)YV$a!5GTqLeE%@Zp$bKq_:WG;fP'6r=P:UqGC!.@4"6Ph +jNUD<@@WnJbtM.Lr]`<4IV(_,RE:-FVZUj31[e+b]f^$/fC9l2)dWK)&X.^(;#`P3 +rM'nq:ARTkl/d4"V??Rm89ZrDI22FKLA"e6m>VnH>@#mV5pZI@B1SX:TRD)&aoB!5 +Zl`']s)Q]_nMufRkDXl%5^e$BLA\e;J&L=U;ZKSodpFbrr;*ebJ"AFpp))nLK@p'B +Zq$mefD;`g43:PBWD@5&&$"2V82fad>6b=D3q9?ZWPu6oHsX+"6#>6lGpc"u1 +m6h37PjZE2G@;J413Y[F^RHG&9bHceLj+qLgG%Se9XKd)g4G/?@o@mfR!9&e1*I$b +aW_XW:Eniia)LiaE\aWA'^IV$:V)^LKjpB)Z+F9-N@O;p3j=["Z*3?4kUcLe-;UeJ +HZF2ViiR(]Xkniq9.sL#]je8=W[.`-s/Ir>5Je36^\#;GV>s9`D3s\E5QiXo[_X3( +mcPP_hd(6sihGK1B,pqob3?$lm]RW%I.=GmZNo_rpA:JSB0S8k4ujXfnu.mL<;pRI +cWK-'=9-ElDDVpcs/-+9,CM,)s'R4f^YPfkc*sUqLQFaIdHK=cn(h%1dpD4l<^TCJ +YOnnp]?QjDEp`G"Qb8W.V^UDi4+Gt[@_Cqqe +r="rU;>-bB5K_m^:]+K&m=Y=j&jJW,AdEDV")n\]+;O^o,tsSg<@MKZ8QjRVQ<)/2 +Og0]JBMbuR<<:?:JCX3\g6E3M!=/iJkYqcaD\Obl599jl%QVH.ZVB]q,6/hEB"'nL +5`[+f7hoL+:DEqf0MiVk$s#h-C%gaI;1QAq00ie.1spif.L,;P\M11GRAbu@'9h-p +@%o]bV_t3]6mlq)$U"7di;Did2N ++#$rMp/h,5Zu7Ale'i0^d]?AG626ac$+j^-GK%Ujj&cA&YnjT!k&K=&..^Qp"JiHYnr**)[E>dSa=j1-BWa]^rFW$?I(2KmmUCs.T38g--]H,QQsS +V>q@d0Utsi_[f%P"$&K*5E!EhEd7RG+T[3\KPCT/.75Xrb*2o6FbiX);I(/DV[77h +"LP*FSSIUWNVbu+lYs8TFHuf0NRL4bbCM_+6OfVWR_4]id#YR/*GlT`./;OaM-ul/k\/m90/]=o,0Tc>QE]5+::) +cGZ1G->6f[Iuf]K*le34O8Z0cqca=!AfEdNi6>m!>A_l>G#R;d-i]eKHZ'n;_tE8C +dMI\[?QV`&H+lLs[Om1_G6R98(Q=qj10qJsG4F9I`!u59g8/H6!K5R\4MJ,`nM@&W +Si?K(,k[Y%kcC:uVQdOOfRlB9Z`TH"oqk5:lXpr:oV-2N]H)EJMZKO%Yg?Q,YT2\m +O)fD<<,DBX(c#N9S'T"h=]ULJ/M2BJ2M&BI/%?k\KjBG3u +XX(k49X,(kYGFca-rM:ILdDlOQ(OEH_dm:O@mob=N9P1M#Q;bWj3^-m&#IoC)mK)6 +8=ts,^-,$_q2EF6Qe"9EG/N=A/bn:3BXQ1Tnpf3\aXbI4GQc[YotN*Ls%@Ons2B?C +Ip9eiO)NbNkITKYQ^;pqH2FDdhbL\qHIVo=6-n`fHXE694[87VHJ8D;1_YAP$a'#- +(h_kOq1JJD<;c%`-AG[`6W!LRNV*,6qj)U."!,+hmD$iZUX?]e_$B&Y>3jX8j5$g0 +,37XSp8Co:JFa3]nun@kJG^q:c6.V5^kqQqUs)S5@RD.7^A@?Vs65%.!lP46MBZhd +?Xc[L+#!VaA3;6-cq,[/Y#Mo]rscsIo!T&Zr[0VhJJ$+eK.;AFq#;H8LZ2\#/!_mY +5**GqQTs1W;84njfj1n6/k(kQDImLb,.(,` +euoE.(f3^2WM9k1WqHu8+,;B@!dGn#)%e[)q43YjBYA_1=&V^]6W.]:HrEaUIFTC@ +.SM9!P_fK1!iCM2a5QfZQ2)9XO)N,%R%_N42u#Th_)XZ( +g&Lk.^.%kidN\\uZmg).QsVD'+9;ET;h3Et7oKC5s545Z5WK(ncbtm36+`<>H^":= +T]>D7#absmXYlq+imf`j.\oZS<3$G-P;=0=^9%b2G,8iLb%"!,VQ]V3fD87nqG5_L +AfH?aS&4DF0DU*plSdBdYcMoP=mf/Z^I:m04q_K0TRPFbOf'(Rsb/]^1O!_pi +N03#LA+'n9ruT,X4[aQs7t,rPo\he'iAC8M*;91/FP-,M8'mrAhk-VU>F6&d(BFIq +C\0EY^r81G?!<"!]j+dVG^[KXZPs2MUU^@M_WA1d0+)X^RTskBNW.Vb)eVAl[bUcTNo_gh;t?is534r-T][71UNd7)8T#0n^VbXDJa]? +HRmh&i?_,BIfq7r$1m30!_E:ja<.p)^Ne??=Lr@aGg$)qkNm]0B)g[CG+YD9-iF#= +@kOr+EI75+a0Q.[ccl3TSYi$LA/l\(/5.b\a&mDX9H5rUoR*P.!L#Q&aFA#fKmGq2 +BW2MGl:LVqg3mu4qfFalZo<5\q-.Ss$1g)UHT"oMeQ7(HX5$S,8*PkRh->Mm58=-4 +OcrE$Lucm+P]Q0ts,0Ne-,h865jYGX?TJ.q*8g&%i.#VfP!?kfqIU$WJ(4/mT_Jcl +bPBG(Og@[B!raR)J&@"*237laj\PXnQTs35#9YLGV0Z/30D@2Crh8lfs,4g;*!WG@ +s*3lQ!7tZ\>pa)17K.%q^IH;G'!\fQ7RWYE!>Dm&e`)<@bs%dNFC.KaqYPpm%3C +QBZV/'RAgU\.cLNAE>d_UA&a!=1Y]=;_kIUK7Y5p)D1a1ZqS`K!)h7HM2$LXMIgh. +F_V6b+okC>lbiqLKQo"Hr5VpY7U$N)#K)2W,=?Ye&C)Y@_AF;-)7RXI1"n6+c7pQc +2)E3/2q+fMa1$#[:C7$u(Wi9k-4r^!//.P'$#YSQk'VplpppC_Yf$N,b`7/#F[R1Qu'OG)A4;bF,!_0._nq(8M=ZP(d&t_CXU&rtuLO(%]mC +LF.(;9r4JS8ptS,m"O]^QY+p#Sb)[?SC>)k(*Cel-ft*j!hnoe(A;SfOt)d0+MGu*CbF)%..MZ[]hiQpuW9Ei4IIXh`k]Cj4Q:S +5J@&_B@d*+H/moW9hRE'0&B]k;W.CtnGgq%Fc2U;s8)BAT?\:'Fi*p;6^2fX.K:5T +4!rcu2gM9?mIBJdF/g6JU>;@[bEd1>Dri +E;8s\rO3R_5ST+S_;:ekZ\\GZ4IXTN-R;&9L\GQku1-kiF,Q9KF^DT1U +C4nUDB7oo/nKIV3q0SOI^&EZ?N.ETBd+cS(Br-I[_>H^ShbrXf>jL4Eok`-(KD?HC +SfiNqeH<[4X\j=VjmE9%1,kk(8lYd4Ig>nr?]`#C+>('fPhAT#X:7\8E!sMUT`(Em +q)c_,FolUlGMk[_'YXYT8o4Q:R4Xeu5g"3Ps!u#?s6Y:QCK*Uo$mEqm_IH.4IW>HV-1IrLpf@cieKQ>. +J7/f3LsmVZrJ;:-X&[G]Onf[QDishR(Q\ge5=PWn^7j%(WdOk`%tOACKhMJ!L^7Z+ +"VA%u!t!]"_*Ec(\5O@KF@eg2,iJs=`J>R,"@94JZ:>Z?FA'MtJ8#jiH\iH@`Vj_7r#]^6r4gi-#5R.+.bld7e3#j<(I:I1:*KiA8-b6+ +_Hu!X=GMhd:>d(1.7YWE_ +FZ`"i,GP3dY[;M1J9:g?)'/f&nGg?V +(NNr-Q8^\ESXHUIRjgK[Q$ZsqoB^#-[E3BDJ9s(N`3YSD?NT3k*5-)B@d$+IbiA.63CT#:Z+b,d-H"C[_42FMKJH#f(7KRA*d$cMm`l'b@V%m!=8Q&]!)@M_Z*37nLeaC6<[Zg$c +;!U#:ldh88f4$V7=Ignl^'2tiRiM(>cL)k5aJI`*TYk[(m_h<>PJC>;hf,N"">^#rNr.!;Z]i.[eNk)fF`9lLa[p/Ilf4M_B:\gM%V]QeM+4N*D +7p2M`8\P/B0QU.4JqAJ]2ur$Cmq2Z/rh1Cq5aPi--_mkt%cGZYI1-Q,0Rm#\(RVt2 +g8hppJ%HJOB`"HS?g_&Ko,u8\Oh$)AeIlM[fL[53jDQR1fZ6$1+Ff%D%7^EiM_00B +jqWT^#l,,Nr%BJ(JFA<:X@6^/U2i=%a+6LqF-6nN%K=EF5FsCf7r9>Yr^ni'#4MYZ +jfbNp44o2]%J!#ir?YbUA:AHkOXh&rr)'GSjNmE"mIN>[!k9"eku&Q)rJfqqe9jL9 +s")%e6j,@)1eQGSo3ioA!3L9"aT#'8s.fLE!9+K=7meCc'!NS[J7]t:8\bDcJtRt8 +9X>jLKcm$_&5:G'%/U/)cji>Y6)QEL6\K+hL>XV4#OW8@G?V'-Rjp`'kR>CEH%]?> +.M2dh\P-lEUV,*c\B(]nh`WpU,:D(p1YXoQb:,U'*ke2(^UckTT-,l=1+CHt.GY&q +O0kNk!@4M1-R;p:-Wq`XB/bIP@KI!Q[V\!62IHn==Q&.g;+j7s)Y +0L]G;8d[QWnAe#:nO7qJ"E]?ZW=lNpZ668)'V-d^(#<&cd+Or?e!A\9Oji3YfAE&^ +cs&?V?SX`C#3^e6oY[LR#kT,8-l4B5d;^e[IrZ1_c^b+4cNo's=n2F&A78f!jM,'/ +J%l-I5K"#P@;d5"[/7Cqs1?\ +UKP5'jt/c'f_4!LNF*`1o?@lrDiYh8E/aM.1$XfZaRdGC"@&KN!'d,7C$l\$f=ED#m_hm[GZ\mQ@I*:(rCMg56 +3n%,lAg:$Yk/UV*m+GM[C0&:0QK,A%_g+V,m@%-uP>t$`XQ&A#-;YT2IMQi33>1YKO%3Ega]0bOC<&b@dj#!?]L>"g2HQC +`4DeC2a+^_q8r54+5b%uI6l&*lrO%&D_OD91WY(9AR,3Fh3VI==M!t8FOt$mNr.ff +O2j38I_P7DrA-'54mi+1):s(.\Gl)ugKtJ:ggBQ\3TG+)dduh8c`n)>6c8PM+^`Uq +plPO.T\nBEN;b%unFs82I8&!O0\FA0n*h]>riT:/@(FTjkLEJ:b5<.\aM1c8Dm4u& +h!L5L<(Yn'15%sm8s,lR?n^]r)>oO>TY_V/Q2Vg*JeAauS>+gN!n-d?WVPd4C,OjJCO]F1dEVb1&*sj`BH<(2EAT+ +.Gbga+E"B*bT9E0s$Hk3K%\meKEqhk1(+WLYpX&upcf)Z58=,X8T)'f*r-ZpK`B6e +rpPn#G[D,tbq]Udg^s>f[LC(g6B/bB#.HraM +fRt:Im.POFF)uq`@Xa7[m.Tko,l`4_Mg%=Uh"H!jK)=St*kso)n^A5^Y:mD5Ia7ne +dr+UoF+*A9l`1D2G@oZL9Y*GOmW!Qkqs^i9gjo,m^X)*DA#,`jl_b9f14Z&>^VOf; +./<\n&RbU@ +&=#K$?cKR=o52;CJ:%M7"JlBk#XALu92Ph2Mp3CH%"S7a;?4[#P(0I=U]Q'R!bpX: +k;s?r^Ja.*gAbQ-E'B:8+b,Ko6`gR*'M8midS$S1-;O5;hV2E"Kc3-QDlES"rW84; +5j/C0o(N1bUM`KZaSZ(VTc6PSFn7E9%aF68q;HB1mJ2%Q3J4?-k!%.%BM4:+0fU>s$09j +$`I,re]O6?Y7MiP$S2.bATT@"98MSWT3M"Q="+'s17_@l*Jl_`K[9ci0!1-2bM:gh'?\Q,3J'E;`$"T9nhGPiYo.j$WTK<"Z&?L-tL7 +P"$&_a#EepiW+rf*iN$aWKi;I=fbT8QS/W^>SjKr=?6)kD*105FB3m_NU(2 +;5m1p%Xu4,#C^%u3o^hXU;7M6A]^ikKRNtO>9XH;.Dce!#dl"Z\gFkg0#"SFWMLoK +[5G(]/Yf?kX/@JG[sMJ3?hWEn5.#HbXpACb?TO45pQSD38tV\(T:Y0FP;Bk.)$'`S +LO$_s,R5Y%0MXo-9U(.u-ROaA-]1D@9D&rEULOIkqQ0MOe3j)nUVfBqF?p`8bj2#f +09/*.12c^7eBUQ?S)<,p=":s*7H"UYPmG8g@*=SIX6=X-R[IXrdT9bo&]rJIL#O8_jWC8NVhhl +L3Pf7MTmZX/0mg!H-s5r,L(8H/2@9MVYY1'FP@XB].VNZ9uIOjctt,GRn\Jm)$b"8 +$J@\(fLV"Y6qu<-"9598-Y8[-_#J\lYsJi"`!T3>.SpIK_#3;[e1qCeH730\na/$( +,OU%c9Oik.g0kH!BEsK.$`qmKgqK!#[EGMtN7tfO*V_C!7@n]X3_&"G@5%=Z2W_Z' +p6TO\?X9Ig7k*?A)?5#-c-fO[Q7d#[I+IY.s$*_oH9aUR[9kZ'_ipE-+2@1;jInBN +rs:=X`VZ7bs!N"Q+2?b(gF`/,DRf^kc2HBi/mk2oJ`?]^PAZEk425dNp#3"Z<3Zo=L`gO2Q[4^Imho&=#K$+3-jjZ\B/jn,PO8ag%+t>6C%/ +.mR'G.4?q6(@U5sm3G/O=1!lm]N\KI/hXJ`n%ePr!;WDQaF%XMq%l)cIutF.s&I84 +,sWr@l:S8F8fo$+n38Q*AcKpgScHi=,59%#mti,nj[I7_JFGcUIie^K#FE2d62oXP ++(-Ar]S5J(P&_XI\Z_1V$i[?lqeHB7b7Pk*&eP-CDcp*?UUlH;H^fM*R +r5nu;i(*bWDNfgj#kH%7+4S(:W8k0u3Cs/';ZH"p^HYQ="KI@W!DT8[V@Z5sl2h&= +TIuiGe.f-&$Nk,NR+akRnPeGjPm)V:q#F?Wk?jYC#A>E11m1D;-sS2,Q1KSr'l735 +?[7c@Bb@,E^aCee1HWW?,MP(5gEeT'0d4FX3R'0&h&],+2sNtgM\KA<-/"$'kis*X +LpK[3[nQqG-_bKS8hudkJKR,+3A&3!2:3j5hD$b95!I[a2Tdp-bVh5p3\8W%1r(<& +[6'2T$cWad3S]Gs[YbR(8_@gN)WO4Bs)ZXOMR#@%ZTZKPIJCcT))hj#6`Qr/Wo]AN +WkANmb,B'o?9kQMY#WMWnXZ:Po"B'8UdQ%H'EmrO!F-+m=9;+-!`1%Ji;WI'ja'pk +H^:"DPi9IaaTM=;oH*g^J/=rq#:M!2'GgKZ(&O>`qiu5i4@,W*'I]9%8EF8PFX0@? +Pr>M:0##%SWMq>W\N-Lg/\SJHX/BT;Dbj\HDuSZ[O`%[.2c;gp`ZZ0>kDi:+%)>Bm +cH$./J:>-4d#F@nqD"XF5?9b$-lB')6esD&UZ\^sakp'c +[T+/W#4g(Jfq.0;/(o8b]\DR.a,PGJf1#'^h6Yo_I$7G&#$Sr+C?/Q5e-7B:s$4dD +C>:_BPT<5-4,l%>)J/=pat]gT=jRMXNJ_RIMJS4ZX=8jhjA&6m25fP6]Cp> +l-I]N?Cdma+/,;HVB6`KS*Tg@lL>t^oGd(hKDGC*I!XMZHJ3b&5#5gL:OA:,CU7)8 +P@mh-7mjt&?&`L!etZ:3;/&2OAa3Y_9s49C24(*J!RZ%W4G +'Sso5M9ZW*);l=2KT=76VnJ`U'9YT0s)+)oGORgV+!BX*qqk&Gn<1u-Uf&k&A6JdR +M$H!^Q2P/6YWZ@O/J.aic55FFH!l)>+iatJI.4eAe;%Djbq&MH(3p:FHiRS0?!L'i +PV?WZrphh!ZGKtdouI$WX:b6Z$CQ#\s*?E+6%]#q:B:N#V6"6o,614G!'b4^#IQ(6 +gD-DILHMqL((.Kg&L1I*T`B)g/.;]N"dPX!qL/W=dG\28jSi5+0+7SJGc76 +5Z&DWn:,G/Ddf5]mQQ]5CcCj58sXNr#4d6n:0OM>O:GcT6[G1a?Hb&m^iE'i?e%M+ +gGSaJWdUiAOnmHGd09'?0CJk93/G/m'Ijk$$i#8a!8#bAcVZ782hk*6)p3_>k +^$>:&r+E6`NFQnjYZD(6Q[5o`^"A6+j?(&>(C$g^#E9c5)*o%N2/<&&C<\iHVXM`u +^3dZjSV+:n%T;1I6N-C%p#BPhO,m+%3CJF4_S>9T0R\HkJR)YN4]>!6XtlH4 +X;S$?hit`=RCn@uZ`Q37>>F+B<6"WVpPlFQl_Eju79Z'JmP'[VAdnYL>>G]89/62` +lMoM=[.s%.a_%nZ'EL&Os4t%eF)&E8[D7Y(^dXguV8.[^A3hiR,aDJp9eFSA++MK+,s`:/WaCu +o\92L\Gs#h4hkju`Ec,\/Y:b*IW>rNlsI>br4dL&jiKZH`HsdApD?PIr>+ks0>5G3 +I"Ctc"nll\#Q/s*i5Ze\n#uS;q[E(s$d;Dl1#>(I6I(4:#G0lB\409+H, +kip+lkm%IkB7'S)\s`CiL-bfE/RFuBhAqS^D2k-(NF#HH2qGMue@O2\]%:V$[8h+p +)Ku<>^DigHY:;YS]HI?lr(-fB+-"n\C47j$!#J&b&+]aMq-PSXK*Y$iO\OoPIP5?.V3Q)?t:.@#0&q+I$u%s@kl +fe;pY$Tr]^49G<766$$P$foXsGTl[FHh5Ddq0R\jq1T)?OY7h,nq@@2QMQ14($0mt +s(hC'/`.)"!'`gR&.nEQ(]P&-J6Bg;!R:2Ndp$#(.Y)_bs?JJSI= +F+\6Y!^Zh9V(C!-\r]4N._s0$X)e0'q\Y215@kTtOm*+l1DnOKIfkjU(n#n*P=:"O +oG=D[!)Y4OR,TVep+[\n#QuCf!TJ..Ya!6_p8]TPPiV]Wja-+YoA#\mSse]/T&g%Q +:X4'6#Y<-\L6(S%pu./cAiXF73+esJi%5rS..<;uj)%*Z`2Xlqs2g,k5\`(.gF"6% +#sd,#s(%J',j3ZFkEH+IUc9`XJ=%b*W!L"k;!ZIn'EQ$dEr`?d&V,[6bNMK4L^D.\ +b[e22)4j:MhPIc):(_7?-jAQ`QWa$s+r[^1J&;6(=TI";(l`R>;AL^JKG+q?&d4#E +JRVK<=<&_%Ylk,X0%[#6gp7DJ59^&i@b9T`(]9e4Qi)6_gg"L14r=-U9D"#[joP]T +hBg.tAYm,2$9p_H.29'6Os%:Tm2]82),XZ[k@Zrn[g7(9daBD5ln\e4qMl2ai0Ven]j$ +Ce9gCfR@2Ua^FD_M:de-npMSTFUe&J%dX9 +S_ne.Mg)iMn+\Nl@6D*fs7caEeGN_R?Z-/IkLlNsY6TJs +-1K+$m*Er^Hut\MmV17Ij36*cq`i`ZG[ZmJo"R9k`P:;e+i`Ji`dFs3'`TC`bi(J[ +^(*2grE[Z3aSb_0cdQ.bfr0%sYApbp]%'h7S%kuO?))R14VOZM>2][/u@=\1aI"o5b\'^g"0#P_XWp(3)9M'i8fD@_fs +Z0Zte50jVD&+D&2EV"N#_AtX":&8MYHr:;P+La1"LCJ_>->:-]!k6 +<=-a=W@mgAk0k(@Ie#UU/=/!H8!i?49!dd19ODK\6_*R;pfrmh554.bqZ)"_H]$^Q +i$U;\:4a!B7=o/TS@\f9)-^"]&sk9U-2tP$"r,L^Gs6rZ]F&[Ao5Zr%rWhhCrrHC( +/heN#q`mWJ93.o_Aqi+HC$3XQ4lChE\/6.b/e[2GjIb_-Olhgpm[BF/$M)MnO"%5V +"Lp]",f_u6"N3_j8*t:%&$lQOYmLuH,R=^bKTABC\5cL]P^DBn5B=JhU-e%[!iLW4kPSS_]R>fcqja!O+P/V,fhM?$qun8LQF3g_f`O>>K*44bEp\ED5?K%V\0$\Pb[KtdP/.!6O-BQ,6Mg!j +p94e6h1f3R_CjAJkHrmPC?eqRm8Udhksj_)9u\!=NN]`-h8Fs(`Ep^c+j*G=dR0BT +kcqndZ.JY^mJtsjI>IN^!R`L3f_nS3Ft;fH'BPWcg/Ap=_hrTci_>&qXf[:hg74_> +2"QUW>m1gPRJpei?CPGqeje>[7-p_(D52c$#5OGhs6j)d=n_k2<<(!UUF'lR,bkX< +V!k*klpC9hhG);NM.Q&u!H8M/m)$&PU'CfT(FZV_E3(`*mf+1qXN@'hAk20kV0#k( +HQWA@OT1)P.K:Jsmf=H\H(A`CR/oj8'Ldi^lm1lgK&.Hob>_1(]m',*HZ'$XJg99_TQlE?@In0ibP&#?VU@_UU:'t0_>&-"$+,ET(hPpo)J=K_Oj%Cq"Sjf +^HBa?l(*Eugs5fEp"]"Cao9S^m'!O']t]a'p?A[U$Q6Ec1TjJcY"`N"*QqV-c'fTs +RN>!i"$T9T4LBmE1V'ucgYL$?BP"S1ZN7r@9aqLBBmi,`ZrnKHf*@`&!!B1oZg.K/ +T[AN;jPT00"(6SNdEG%PmuGkRQ5o9"Qnltq<`L26d'I_\Ii(MSl_9<2kgSV2lAOd1/S06 +dLf\-J,iIIjPT:H+sI$N%7^8&0Ml*^CrkL@#kWH7HTVJfmee\=?DfTo_iY22flC8u +fr^R12G6fG3?-3"UsEf*A_(YF_>/P&\keUR=/PiTBjOrB8"/G"Uq4MoDK./YhC1ua +9!@2VCJtLTr,fOnK)ja/CMfN:-J;A;"Ul,^5[VgbPq`;V.DorF!^)32LHaOo8MR8j +"sld??it-I9;u;3P/jcch=*Xl^QPQmZK1eA.W?L35ju5_P&lBI,6A44(I0WMZiTs:[.Hb3 +H!F?L]MM/&PHm#cN^94F2.V%""[Pq(]t^p7=Pb#KNK![cm1K"k"CV5+E#Ha3+]nfq +#E$Ct6AU/b2a?NN$[?\TXpg(;`F@gpKHJ%'V;7H$0(JcqY`EfVf3kA271'1XK/D5\!IWr*ukt*&(cC4a2X>g'<9GOp"5L@ +Hs/TdURRePT-AgTs/BQkY^"k&/1fN^d#7rZ7Wce1joQbl9s^q;(K&[l[rFd'E74j6N&LHd2d) +cFXUJfoGeDj-s'=*`9#LqM7'0Qoof3OSqXkif:;hgJ>gW([n&Js%hY4r[2Uor^YN* +O,!L<\=X3/(WVX'%[%Q^U-SKr#;O=$&pt!N-pSaPO93c:(lsi^a%;j^X<-[,,ciAM +4IAAOM2rQ<5m%(I+2`Wn';YFllmrsbRN6g[)LG[Sq7;RV(Cf#XqU"oU+.pBraUbjf +rTk-Q!^Hm9AcKpa%t=N:Od@.2J,)CsbE!9BI[Koi"57B0YlF2-hXB>pqq^kf?b`0) +b@`i-4O*d!j5Q+ASc%/nd\#AZq3,%%IrhsWYHOSrhNagR>qGj;C>o%?"j]p[4?AaO +F#N489[:VVXE3FTVakgk8$]#sZR5OTLi_;T\l/$_r5mbH]_5r4>8+WV&Jb4?3?reR +RDAkRi+E8lm#[hG"FZ6%QPC_6Yj4I="tukTQ[aXW"b?E<]rWOk'2JWC!DOU1*?X5CKQjH$\0GZ +A6_uOlkYVR/;Yr-4&K1ANe$9`WF?0$"b0p&ZIt>u[,q+R)\G?:s7oS3_\K]Q:N!H] +IfOu?8,d^C3ipF=,MT/,(pnCe-;g[8!l4_[%1UO":a,`jPW="%<IV8A>n',_H_)u&)04KEj=6rcWUKW)5oMH_(]Sl]t +KmYCE\a65aF&A7j;9SZMF.Sro^7@oJf?T(oXu5WPDgM([b:C'U[b^>,^>Bh$TUYSP +$[^Ta^"%gepisCaHb0B,X#'$Dk:;VJ?[LH!`grF/f,>0+W]tYZ*5<=]Re"f1*R$,\ +/$,O-72ot4DkEp+:+9n24 +!T\`L^AK*BFuTi9093k:#;GpXEemBl!T\jtBU&K%`H2M5r817:rZ=ak)NH2+*66'Z +UFC_u,M9qSdW@U[`54-=CqEtEH7#D7A;gGJAMA/HgCirh5Z#*;fZiA"ULXqtYRFT=mnr_fOu/hI6H\m?CsA:&j9Geh8*+ +rS)tAJ%8q_:Nt!2Ei8KK2_0?%^#B:KWrII[lH_(`I!D@Oh6SK$/>'Si=F>J0E.'e9 +dT$5An[u2hV#P0m$)$#Nod'Y68_u*j) +!mom*(B8:a["J(57f3c"\"lCD)%6"R`&A8Ar^VT*IX6's!>3Zq@Pm,eL&^t:K_[tY +-K:*J:Q5!8bjTSWnZndFNbu%@lkT`,I5YcrDWDr4CdS,Sp2+QQ?,[!='+k.u(*K8G +#oV21b\<'&oL;JgZOP<:7f[W3&N[U.pq1QH[m,WJ+op]])or9U'CtV^(\IK\\p]:' +T)SdPRF=`?n3b$<&L- +A2?Hh<1@^ZFM>IE=^02%Q7&`VG^ufIZP@%+=&HmZO;5"bne_tW3AaSk(!LTOqH:or +.QBI<;d7;h8/Wm+6Wlo\b"uM"a8M(7.t<8AWS/?X7SbU+X5RgCBVn$<@tEUTo0PW` +,6B>.2uQO9JElH&8h!7Vj).!FfQ&]F/n+iZ!=L(O&-uAq20=WSkQ8oMP"Nn191fjU +M*iEc+"sgVXl1W7`H7rHX\nU^qK0=qa6c-9^4a-HM'\f,_LYSdF;L=J.-N5ci+3R$ +r"DBSF(Mt"kuK%TW+?i'9RnP^!?DXnIfbggC=LWB#.hoIZWNVR459-(q54Gp)<"D8 +6[`in;aFW'EG%dU:No6GD[l,a@7:*)A+Gb%q:MK`@TkoTUEPABB',&jq/:<^k';>E +!)_BeC>Z:D`d.E^$cbt\B#Oia5bu20p]gLgpjNnXAr$tRJ=.;Jm'M.X6Ei.M\MRK3 +AiT,9=PETWj2Q')K98Ms8+o+gg%En#S_ofZoimX4\#6J!:1N1L6j#X1)8j0KLjdcC +6C0U=;XQD1?hHp[G=F6QU8E't$!gNFiKXH(p(>KK;i[F?St2dlYFbH+?O:V^&<'p' +OY>6`f'=bd_PF5@dZ#sVAj,6F4a+H"Ei3Khh:C3H[dLc^G2u]iM/+[Llco(0B(hY4 +jFqRZ+jZ?o!?h\"oRDAfE)>iJme8MKY +IsQ%rbK%6a#^Gt&48V3kkJ3""%tF;uOjDlEp?=ZDIs='62p\G;HHQ0H1%499;ubld +f#ROUs63;K:N)TurN3(Ah;nUd2?*?Rn@s*Kpl2]8&'n=lF/J6E;T7IGZCA1s=(b,Y +:)%!$-s@K==rtdm(s8Hha;aHg)o!5*K0cb]68=[<9CjCHlS_#1/1DaaS;8,Q.^rr9,#n\P57s#2u6!5pEB +Ynea(58LOFKdX`\]\\+3_%GGcN$E"V`AbD+rD-Dl"AJn2!Vjst%J^)sqa'pF$PR(0 +&C\@,WY]$sR>a3LNVPR#CbWWINbX$4R8"CnTRE:1[*JsD'+j<8W`V/*#Cm(&;S]9k +H.=?-D/qn6)Dn8@)q,r3)U'X)f!Hc>$:M[!,=%1n3RZYfR6ZNhX4*9gjimh;"*m&G +I/I,j::n&?LAGF]C5#m>5T&Grcq9qLT1`'h4 +P%GH,RYAj16MiW&6WLG+P(u,kWr_CMEYTtTnm[WdO9O,K0Xi7JB8Q5I"JT-*+O"F) +Z5EL53W[=ZVFibiOo.'L\l)3*'G;uj%'ouT +7(8O$+Ff01lh@1YI2h`qC;\*:4#A3$)3KlOe;=cd*a0T3aj^"tcIQ +!)&6[rte@-ZeDBmq:(X>AqE6+?84ZJI!RXDgi1F$&H1n0M)teE*q[5HCgcFQ6T`"? +7O%Ug5#CkD+fP<1?g/cLeO"T78@]Uq^?CD;D2k.8P(sdJ5m=6u/uh?0ktl3^e\*_O +G]hO.lF;-:UilR9IG;CZHI9!7d1LND:0FlbQL3b^%a$dbGqJg6Q%h6#2T8p>5'FUs +nTl/78fRLsHcF%uX:)/@Vjm8>)[[noq1Gi?*_U+C-DRi+j/\p.$=_Pu(eN`TRNo": +m"m'DI4BE&bBIB[]1=etOSO7FJpe@'IL=/'IUkO?r&=R-N?KJFn&%Qj8Wl=.^Z?

    S[24A^Rt?MS=C&r4tDP4(n=a:=\-( +W4k,jH$-^Xnl4:rn^;L=-"f`+Q\UL^#7@C+o#>urLV%U(448FYP<57YD=u\]&@XC8 +X7p[%4uUpd!=oPfI%Q>UF0W@f2W1jHgbpM;Z8)E@NTmjWHk(>H_1;jt4$iio"h+8W +@hd!ODtPf7(,rq[SV0MKQi5h&"HGFP#\&W)!:r^d!'W\'Z0?`^!oTDsb+CXf)IA6m +5K)R-a?2mZrdGs,k@aSmCjanHcg0[)b<#a#I0U".&,"rb5!7Wu4MYPo"i"'\#)KQ, +b\pK8XU;L*c2qg&[U$^j.$IpDY"+`crqG_V6McNKI?C_BUn0J0C +.asS2dL]Cr1b_.%Pd1SpX!BOk^DnF_\l&JXn9J?NJr,5.e/XI.WAejVAtHhl0Un6q +1Eqq\eLEK0=>"/P"@*=p&HL>3d_PlL+]A`JQZ:RHAFR?7R0*B>aE_M<=&k#P8pE`, +84@?OO^U8="XKF&9pA%8):=aLHWP6ms7G2HPBFEV5+,1'gr$hVda>)FIe!j&Ra$@ +/NTcZK_LilI&,-nV+=4oQPL![s0EX/F./ThgKKMSs$Q`lhVVBDrZ<0nNf?WNa.`9p +:/&a,V_p7d=u9&?V^Jp'/MC?:#D;(Jqahn%OP]<=hJ"2tBuKFnUTXKh=.S)@ +RA.]$Ug=AY.DY3PgVO"a4eL.NUpSW?[E5.(("$=0W#eB,YCbOSa80=K^DuQ@?pe]" +K^feuKdH2jZ[r&k_FH8f,d"OrY&pIFmEW8j?#ZY'jJD:M$Q\>A>0B]! +.kgIJK'iGpAlj6g`P/*GN3ha\_oEL=$-AfZ+K5=iAr&R"(m\hp=P+E*6/Rs37-; +TZIZ657.th^kVZo!'^YJO+?6IL[Gc2\53/i?NTUMp/AjmHH6UNM4"adO+g'.r%+J$ +NPd(?O+qs#BY]C*hMI,>bLW,3_u(8ZbH,<,p+8l`!&ZtaJ#eNYA!:sYrqC3WT`,Jd +di\_/`Y>-N_#9>*!!4u$j@BAVS7msc]-iMmJB]4XK*VJK'6JPYrm^rK?)CtuiR;Va +hkS83b;,T1Bm)3Lm3)K*S$pCF>L((0cQ_GR.,'bI.dTIr=k)_Q6qn.#J=!:;UrlAG +77P]0l.#:2,bQ\GYr!c#,WX^X`s/nn/@Ck='[c$Z7&_k;-;'HQU&&,q@%r2m,oYr: +$f,qFSUh7@2d):E$Xc^1@2ed:"X$9c[^H0$FUao2^I5p>s/U,7lVsjpbm;\;P1;-TTCN5W0K2)V4_Q;cj8"k +C5,=#[[FZ9>Dgm\rXuf8%e6Fm2uVXdOA#A`Z_O>.$gRbLJc=r.5sFl>\-1"d9MHt6 +Vj+MDlFC'Br1%q=cEK9H6jGoJZCG(+SNUg;NW;,TiWAUJJ39m5hR*aJMlT#A5Q&cY +:5:pu3598m[q%LCb'`WAEFmP%ZD^""TK=s5>s+rf=ObVLB]eI-1\e@eP>dXdEP^.< +`QV/p6@\jr?KQ"sM5rdpDg(T_h7`\W:\:ubW4.,JT(c'QGEJMA?uNM^cPj%n60u%@ +I+:YGB;Wme8]*cee1H#Yd("M@rl2V>aDr]f68S@67jaW'>$pO%hX[^q2-9\MQ[8k9?%VYgK$u;/ +$h]e6hU5hJ?B7.\M9kl2-(<;Cb*1\`4n*4:)\a^t;<@g=L&X2*YT"8E,t[fm61p+f +'[7eYp2C^.2$_7'"!hFd##G#a!.B,(diK^tO-#mPU3prk./l(gq2a-P5Im6Q2s^9h +H26NnY?$*n[H>AJr3*1-hHcF0>80;.&Dj#mhONP3S*$nMnlNl4c];,0%$rRW(,,:p +c1McEG3+amk7"b/p9a>#)L:^YDG-'!rXX;,!>jMRfGN[hgZb3X4pt?Y[]j%LO?^p& +@KLEldpm!9+Ta4=OI[&*Yj*6FjF7sb,,!VC!<9NS=D_@k<;lm=+)(0es.^>0bjUC, +p;DUQRpCgcG:W_I&-i[naJt24FEZkM\#K"X]g`R_^7j"_#.dJ_^sQDr2!(@^I?9L +lr&jBcJ,;t25rk^K()!$Db;<"!,/U62VM&tm)]3dcEnp"HLo-JTil6H%J^[Eq[*V_ +]S.s*r+9#:P;Vf)rU3a!G_gWOBfc%H+\B/$dte$S@^P#hbpE:k.:\#1Pp_uW@[=Z_ +.4b1fne^5\52gm@b07dWbb("sSt8K?59bRWZk'*'FJY'Gj!RFnM%Y9M7$o*\:usHC +=#F=X8(P.:nIOb@>>kMP%RbihaOBIbBF&9m('+)Y,/5H:r*RW[$:IRm(g7#:`;b.R +N>2.3!]U='$!T%@.dD9L3hB$^Q&7SE6sh(4_lkWP-]%t>q_2Q?9$,FpfHu$HeW`Ap ++'JXESF:I['"FH7:.HY\p3%D/S4fCG\`M$opb3^pN94Z1JZVa)lf;$%/88U`)nM1l +%Zo6fb+qUZ6C*F*5XdWq=;DR2n38s/b)=O[#Pj-Al1O_;9hdXMXnhAOKj@t3]=U`4 +H7K4*C5)q.7tu^2j)pP!d]#eUR%KK?_Rsc+6%9&SL+CjA;I/C#a^k10'E.YrVS +C"_RCG[e=iILUff!%,0%30YguFJ^-!lI9(nl&HVTaZ,p[+8a\dip:4bUPQA7I- +0P;![/]/M%gH/Cs##4ulga>jNR-.[/06\._q/6\lhAR0c)3s%sOkpqL6$BSqgYh1H>o\s@a+eP;)$!Op8'LEVESO$lbJ8-9(]ZsScOBec +#HY"gN>_9dLA[UdAf$*hJ")2oFSH^2J*6s&IGjM2->rt4!oVdNGK7+;&:`^>@pAmR +O[%fS+p+3hrEo=f0[U&W]X3ce&b+3N*i@9,,j);D*p1I1`o__VcorB8Z16aDaPU^m +DO15@U?7ME;6W@c-7OfPEub>_o@rOAi.$N@lW3Sls'Bh?5at+85@d:hn-DR>#cP3T +n-DR#ZDW?T[mgAm^YPN75k6;t,QB^TEl+-DOI>Tf3q!#in@r%iod0XuZbe^u\?>MLT*6B8cR!b1#C5o=+38q#^s)(!KIp^TJnL<"gnh[>GTnFW:/XP*C]O_iq9`\5Eir0b"_>dEgs%*.S[#"`Z"NXLa!`UqH +;S"k.">;O%>+Vg]V^c7,!+^kdN\-O#TXd<,#=/92%$d?^q-EpAVTM,)]H&PRWD]KruFPds.5h1rr*70rdqCP +&L%2=.J`=h&s`E"jH$Hi3GhnZCHK"7LVeN5S/WsUMn&RKq!/^u3:Q?A^Yi6"s.?]) +>dg`)11"58Y:cd*jqOl&o$nWBlAPkU0J,IM-XN4-S=Q"XEnKW +E1nrF9J?90:ZCK-'3Df,hWrg:[rplIQ<,+cDgU7Rod!4[jN$]#Z+EctL(1bja3H*< +R6&b%+7$&G3m+_)dV\Y';>qbsVh"+bT\+9>9,;-b`%2c3;]JAUS77^h1^]/;sN3^bP%$_q^AKC;48=7fH.D<3_Zh!-CYO.9F!;f +GR-)F+R'DE\en,U+#Cm/9-Kkir-qU;4JeO)j?H>W7[((H"H;h7?N3C_:DqeTlJ%4+ ++2VV'aQG=`gQgDQ)Tk"of_=.)+-3'H@/kZk?NPO(qQgF1]Qius]'nRJh4MBo#PnDA +mb^3Ro/Da)r9D]/:[T7uO'@B'^2`dXfA_:+Q2ZujS0+o1IK+]g\'GeBTes,+H[.)5&kgDG?A[qL$o'k$V+KaM0=BDXk"%V42-D +hes`3r@\A#4=6BGbQdZ/J-^,[=`-C?A(^FgR2NB9_KKSn+9e-,%RU/r(,r8ngV$8^ +n37KcD*D]m+%U/<-ae1bk-:TXXo:h3!FNqJbPhb<[sAYeBI*q?0r7>]RL\16@an$Y +HKb-=-&0abn@0>W(MaZog,pdA%Xgb,a3"_.[s,$RojPLQj7JQI!rbqJk +[s/7Kh>-"/H>@]FS/F.p"SEYZ!c7WR=o`sJ#EOBD_Z-S0Y=UTPV\&ESiZI5W'`T;* +Bg._/re5Vk5qW9;%n,K>P8#`9nQqNI!:o%4Ip$W_S=^LX5N.*k(^aQ;+u8N"\\D-_ +7!F2"9O*ARZO%*?&E>ue6ct/`Q!EgnXH%uf"Jlb1YMqA5!gZB_iI@sJq>X)>!3UVN +)A0jLLc(&(XYQ/,X\=;&PtOC_s0[da/$%P"o?R$8 +p<0?cQ^s)t@="hR>kDrsFr94]5 +n\K9g(A+NNpO;tiS,`0J%>]OA5MZ..p\4XG2FP?Cs)TFQB31;r@@d5SMogHo+oT%9 +Ekcn.kbHTQ&-:)n7F1+MO68G!Isk46AO:W]XS1FW`T_q/nGbh!;8iLPNLATG?AG0- +m,^Lg*ceKXN;$MKaSumoq-G;lA)ItqNrf%uSmLp$Om+g0#lb.3%+GG_km]NT6ic3G +2E-OBrdf5G*CW5DhX7;L#,hOs^XX,ra<,*eU3C&PLkl%Ui\)Z'Q%-aPD;IZo-%O5G +844&?G6-Y%Y_NSb%j:\sM7F%&F"WsJFmt2sHsDtJ@]FY<_Z#PB*X/%[^'Uffh#&*^ +k^ZN(Z\Y#M#n$b546ee#ko]^C/HNi2ItEDc6N06sS/hB8s+C6=guu&Jeh-I]%-D[: +$;^j.JAO5E7KbP.EuQ,=&[8YHn>Od_'S&:Zr"&J89pCMgp#I!+;t\\_Occ7hMZ4kY +1%J,]T:\KSa^J.]e#P7&5\57>KSm.To$dee&hdhSs5`q\dV<XE<&iA#iLD8UsN:O[mSCrP_Xi +"dGlhE]BQLcc"V,,Y'o7Q9mKN8S0To`$1UP(guH5XpiOH+4X?d>&R]jW:%U*PArLF +I]9aEH;EXn6/fKhl/P@cUM9DM3W7EP_im+ZUDZ4j^s,a,eNqlbDW+3/on'Ah_OT4V +Qe#!QFJYT`;"mTnmL[)u&\a"M]hXa*5m%(%pI8lkOeK3B`*h^Eb]XOb(VUt7\W;m[ +i#T2n_9F&]L,HnYbmBWKphdo?;<&($P)2o\Z)K!b.N\?aERO&ZCbQ(Zhb+.XpS!4i +jqB2E=kLE&dec)J[s[8Om_F9,MZ(^Bp@In.YHC\c^&,sqa64T*Qg^W#+?k-rc_p7! +qu$72'Ei@A_'<@f0CQl-Eqf&VDT@3R4oQP,d/X(XqhGtc^3;3q4(er8^0L'DO\]T\Z0c +iCWD.-FJ,4Rk4VN5uP$BM3nSDA,h2Zko9WZ=NJGNi$\=gjA.>n-`23,lHQ"D./7fX +'/`!?7/gd1^`Xi:->%i/>r!J[J$4SspciF_"o3q4.3T?8#afjis$?WmHQ%DeIWD>D +H.#oERD`\,ped:1B;Y#s-jI3\GoJa +.N\#-2Y\BZs4XKKZ5P2\!42`>$DNCiq&(0R?<.cORTZ\J) +CEWc-TQ/>YPB@-`!5Bc>Nnjtj#?&8/%%f]+n&PtuF!CVX3jXWH>+EmFa;nS&3u9U* +3;@dE9JF647+W8i8mPns-;LL!(!hPFJrGhW0De)]P;A!eK.upLgl0jRKI.! +6PZ?5eTJnT!pZ"i$?a!NP@uXJd6^Y7bO4LZ"Sfn!TgmN8['g]kF;@4O0JHTDM1Fh> +)70B?LP,UC:[+12%EmrDB+m?"Ig+Zh7F_@Yrti@Q,66eSD9[.`JnoeRG%oWWED_4" +%a"b1Pka6KmD\V<94J_&@jP3r9am4V2H=d?]'S-ec*^"1qNCF/ZKD(jrXA^J +Vf^,RWoY,gRE;K#A&Xd(ZIak0Y7>U@G94?I[f9+oS+je'h=5?ok9Slg@!@R^/Icl' +JR&gq9@]j&1E,)'h9WF)`pa +#._HUHY:Uk&c(BW6i<+*ZPLisg/`7+]B8O3nA2'iacuY(=l%hj`cm`nD_7D#"WknQ +Dbm=7%8/AKLsEEX/G`"Ys/.]?1#*Wo9XGZ4iRc"d7"P@a].hoU9_AT]$PBUuADDq@ +Q5Ihp!#>=d,@koTqlqOhEo7cpFFr:u+8F`^8FE9O7M<1ke"`h.U_tL;6cWf76pQs,f'K'=\7jh#%QC`p1oUF9rUd5kD\"+:p_18ei:DVXs+q$6oR@i( +J'`XtM#TCYi!]SFQO?.`cj9[KSW(Xu"lt.pb@>+q/\m^MH9(aE+!r_aMmU6%p)>86 +5VK"O'?gt(6r]?]TL/ia781KL@>"TNEfOAC+oLf\+Tcs26pT,?"oLjX7KhA*s6C]h +i]dG@hsAUDp`I7%)#:o[StSK,!9F/F!UfuX->,73^YY8V%kW;d*?kJe!&OhS9`L:U +XgL9>Fc*VjV_7-Tq3NcAq+HTIJ@L+=rYOY%_>cC'bc5K*-,.Y\&ckBR:mF>I9W&L7 +b`g@4f0i19mW)&L%3YJ/TRu,[j"I!8jH4EJZGWUZ<'ALJJ16r3&j"kC`SiqTQ&R +T\$%Z&BTV'/]Lbf3Q5K$pH,\_Y3_f3Q[n9$_#I_b2FN)=PtkTELe`*1:\D[)cS1aN +=qanQ86"gePe)*r3`[@f>#TVNjBj>?ruWX&`)P=S07a>SX0c-^ouNPDs!nZCTi_:EU)j+C(PlfTak5R_KF8"-l6+BG>,h +>dD`_bc0aeeO12R.0's!p8i*7h=0i_'S$tWb?_oP)ruMESE&$8$cmg7::3,->FUkY +:%6b(X4Bb_/e4Md1fCtM>'+>Wkj!E_onfF/&-OeSf+s$@QK3`!=-a7+!G<0*BBNjh +AnopEb:eq2H&bH]/u +nmFYl-Te=QK,dM$!m-h!LH0XINJj4XA0[:SUlinYZAhV!1,'4D$`&uU$=E&#H"1*! +Gh/7(f$1g3T6m]JjAo@TSUSD-@HitIe^J6W>71Z)s4^0nLk&"eqR][pRq%>XrpQ,L +[ZiR2RB^B#!S-.7K`C@E_rSE`eS\(P-COA[t +CPn$f:5EeEVmI`mq=i:GN@/S8Ui-A2cSD^b)Y3oM*89d;G4Z1BpUu.N]nu[%PdGj% +(OOB(I*M<+pVP06cMjkR$l +jq@ZCr;ZKj>\ClP>Q.`'p0`;pIk\RbQe_#f2E=lJ$d3HtH1VN,^I,oGBc)"N!oBId +%O$>/\Q-oFC=n&4a7AQLIQc/XA,(.fT:Y=,d(oZ@8RFl9N=$6trWi8Zr&=WD#UXZt +WSRP!_Z,;e$3"T;GPS2a:_43=nd"Y-+=_@H2#'<%V?]7tLGBeA!WUnW8H-'=N&Cp5 +l:$Un2ck6Uq<94dh'f$9U#sBcH_BLKjH'3l6Kj+p7KEN!,/>C5&H6lG\Qf%$e&%=3 +f5Vr8!X=Yq#=X.h)8@1=]tp+&pqOYOd/njugOhLlTfg!_hs`kW(>SCDaM6,b(%Hcu +lR[QA*Nf?3o=q8o-\1ps6kDTj!.r>rIRQ@q>8bJb_U%KZe*Ee4mB75kWt=\N([S3`n#NDC>@+W@J/=la##QV$ +WTNZQ&KMG4*>ZhZeL=;82=W/EP0QBOf]X(egZDp+rB*k(e_cH(CVRp92)JVq10oNGNlH[r=AQ-QcCT6^>]?<>oq:tB +X6rW]SkCU-b@drDQAq4H1i'Gg[7Sc#p'1sY%cgq;aLjEmEIAu&p-@?f_,T@dB_%_! +[+gd)\#,(%buS2@(L:1*?OfG +Tjafdb`[M0W>%FR%0$!A8\n(u9C_^>iU1\sb`%s9`g_@\+o>GIiN;OrHi;J*8&S3# +Pu$RbcPDK./D8l`EkJenHTZ)T5H_[_XBb'+XgD(]##7p\F*d/^Xl]1?jp9R.C=?5=>i!`mEPUa0:R`X=F>;91&7uU8d!( +)E(>\AcF<>[(mg/;p"V;d0jsKk>#ic3_f_U%irI^p0g+r8Nj],rNDD3@\V.H?b]Kp +4dAjX@?(88"87oZlsFA0?`[YM\Jk/*]q$etn(PDU@*d$"mf34N0e^7e!53lln+.(` +^Na\sB?q+$Gl-Es:CmL!dG\Fq3;2rjYs5j_pqO#ZIrphErH"sq4e;,f;jm+!A9Q7[3LSEhSlDCliHbSq%'%J_d6h.a":PQiWk +AE%k,CmTlf=4/[@\lHSRadI;UI*$IZ=WTH@#L3SGCRJLb!&]]0&u"Fl24Tbo$O$-m +TUllT0LJ<1fYhU1Ya[2N(-AYCa_8^bJVYU;BF)[T7k*Ak7:+?QQeL\^Mq%bp$\4iD +<%fua&%)4sL_o^OKa=?[q5Be^hcL.`mfqua-R7\;"RSl*?tK/89O`UkARIOk9S\X> +f+8$J9YdB5TZSAS04j>n%%S6'h]QOp7knB +p89hQ1R^\!/`Ee^E%XFL.(:dqgD\.Kn`Kr_md,7$fL!=ro`OhLpW0 +5lS]V.A&[dZDcX?b]h`]m?Sa]^WDQ@,s(g.V+U]Ls%=lAjjiuQ5)V(%#t.ILLa

    4ja9+HbgLF%hQ1A(re,1OU8SL)ZsVcd7u49@M!E!euuI7m%> +>?5N@?:SpB=+>38"o:^l6_gB\EUi*X*HVFH6)"*QL4pr6WRKFp1RiSST!b_m[B;)# +ekY0sI:f2>(Ti]sXpl,L%cjedUp'G:?g?TTl![rb@G^E\hX*be)<0EmiS>Bp\35t( +A#enE7JG^N%0%mCEV9 +Y/[V9Z-,K#65?qR&t?t[qD1ZZ.ATEC.Ej,kq<_u2R,?%/=m25-#r%Y_a4fH.oD?-.ptXW,F_ks0@LRP@_!cs;a*Du(P]%JG06iW%du54\#pa*r"mmU'l[rkhb4hd#1=i'PSYT1T(FWUtDt +8tShKHo+\-rkC5gRW)V +9u4'TBGN%f!7r.,WF1%:Cg\Ik,5tXfr$iIJ5gAq8k^FO8TP"B:oT519rbo6n>9.gA +J?AeuK`<@6s'>X!%Wd7">:Cj:>%brrKH^ +p-8Er9pZ^L!0TSEX1%O(s5O^,K*^,pIulnA<[`T]j#\rND[Y"mMJns#t%kt+o]0pRuN_g!g7r<=B&88&E_FFGr0t&[5KB=!:s9YSu=,#;,`#U +C.8H98Jk>C1ZEltg/jaMPs;NL(7gZ%H#`u!5Qj=g!%i`SL^%U9RWS&69ZE(es($C! +J4BSSm0+'J"TjET81,US:5Tep+ljWRJd\MWm&g77,/@SSq#\;7?o"UQeXi(+LFq?D +5k5G2ZD$Uiq6u$8RN%-Rl3eNJq=VSZE$-YJa:.i"r4Y"[2\g0Hna[Y@Nnqsg3gl*9 +0N]"\Q?\\R38O]VOXL`sj9He2TFlt[M#[f,0mrM?KIA1!5^YgF4X?=8OfFEl[&B8: +R,[ei"J\-%ihG2?FD$-c$$,t:ae>T=L/qsjEZbP>H_;0'Jfs7uXQUP%llWAO)Ql@6 +KZ4-C#c;0s1Q&/7out'l!A+c25uY"nW'>LKYYhl*McV9Af#Gka#4pDl1G3k@/p8u# +4-Qgh`aA1j^"IDb?>&VWmeFgcEVB^:%u-MH7oSi9A`"I\\iN]3EO'n58oT5OZn`pi +2oT-pXL^5c^-5`15+$E%5N>c\m:RT=F0WQ' +;?n#=B43t.k,)X#)@8+E0E[T9:Vu`1Qp)QI`1n*`P).e*B`'6B7;a^1egku^7(-`3 ++2@/MXTd0,2s&iY.(\55G%[KE_5R/.K]:stk4,MR!;YNAJDspb]jGfR;g>1EEo'eU +Y065HESF,'Jb3&[;N-#':=3iSr9\%B.qHaP,jD@si\cVeodKb[bd[oRh\ANL`fLK+ +\"4^b-'3G9a9+9i&;mD`92!dF?)`;[fHK%KffHCkAs1Nca +mPhD2SQNq5=O0`Y49dlCHhCE"p084&/U-ho[@mD(5$l2,mjdo.MhaAYJ%#79]5PF] +i6=LZ58joU5N]qs0qZ4%kgR!Os5DXnh`a6oahtW<]:JGP2s>b'cM?eVeA!u4rgdWi +J'#%]U5.bTpW(EoN`9=*Y.d##H[_XM9Q0$%GM-6A8LXQcRdrtcf`St4??[=/^VRd@ +Eo!X'\CG\#r6&b##o/55Y+s\>#]RqRD?UCjjH!,Wr$m_._'8@tR=JY`#\`E&%5$T> +##eDF!a>(!+I3#DP:55n3rB73m%OC`s$u4'#s,a+q!:6Y5bpYU'WBQQ8C6]u!0I-7 +M%)*&+8_ZFn\Ldi.i5a(qV#nZJ,_IA!)!q@e,OGjPU%/HaCMFM? +Gg0#lfE_O.4p49-('+>&&cO.%^o<'#MW7#jB5BDIN\?_CMZ83t)tAF!ab:"N('+9i +(te>-COLA16OkaK*:SGn303QYZS.+'J(U/,Z9nM^&ZPf$au$^m`@=S:99N#D1qb#+ +;j3rU8l:mqE:Ym4Ci9>APg9F,$hD&o5N.(lr"%:fDLS"k']c"+J*IBfA?i3AFV)Is +;I'0=-kY +(_R,_r(d=l="&=o;?,shkXYq&!l4^:aT&,ZM_Udl^XJeBG%Ls7)'MdfK0Q7oPRlnp +DC>YGV'2':%V:EI&-3Y6+HYm_VtbY]/g&F(j['+&g^K+)Ap."1[R>oONU[,m!Cp^i +_c$.NTMBVHJ2fg8%ha\0#69lC"q]?++:u^+eL?MLe]/B8=Jc"ag3>qsNA]AK:!Q.b +S*GM0n1hR!L]Cb[.CGj,UT$DWV%Z`L\#h`q]s"Z8^SlM;AD]XDs'YZ<:^%!mHsF*IQ9Sqd^i>nIY$cE:U-ibcAVbQUXS"N)d!=2'8\N\M>AoP6*'S]%]5Ps"o7A=cHl_Hn%jFs/E^qWW;j%kn0M*X^)F'brZ;&248l?:2;R9OE0-FE'V-M +LM0175FNdRNY4dE-/1u5*Du+R[cqsFL,V\i]k$:i1@BX4j"W=a^GCo*Rm8#cU'srA +i;YMMH(7Ku8A_6eaHMHoc-K??2KbMunfL"32&&FGa5u4S3nF-7f!ZUbGn$6*KWVMg +hl=VdX3'8m;#[a0qrp*6DbJ_1CS]kFgA^(dg@L#QqEIC=J,-L#`:sl;J^/-B%;O +Q/!3/>9BrZfO)>?=1-gh^VOln3uH0Gq0+hs)m-cWM&jqt62jU_8.e1nf>*Z<-O7kMNRnBm.ac@i*&_=fCnn:+9X#jE174h]^f#TFC"!ZK<_ +L]Cm.!:Ku&3(Y(I6TP?@rWI^%Ifb"L+E?]u!#cYo-apMt!'C<@r/Y2An43H_q>8B, +r_<3)0oOjhR:G>]jaWF11g3"^\0+4,W5OnfY7rn#[GrkJAl=o^Z8XYL)! +2cDIP.Ma'+q!:#p.foW^HHg8uL#]X^[R:*3lMn(IV8<1cs+q#sj\Lum*6`-eB+t8_ +E+anF)*poMhj!QXVgGeMDG?QeR;A'5>O%AG@\,lMWoL"WX-8g6ag)Pf\<%1h434%d +LHlNe\a3MH4t_=o#`&`f@5a%j4oe.Tjp0Wk:83G0WY1W/P_F8lPd(oK%/EK+XI(Ul +U/dOupqLFai4iO57Phf*[-!.RbneMFn.u30L%V-KJoKWnIXbD'f2^aP'\KOuAg<1` +:&d!Urth!?(R2oSQGaN)`KaIN6:XjPP>kbUKH`ZTQdda@s474>/O9/E"Cu?F5F!)] +LZ74WbQ@g"]ij:nL/8q)*sZ-C)T0,YiTtM3.i_"tEeD385TnbKkIC\"^ua`=I;HNN +Bm!9Ip&q-B?=uHq0cj74N[bK.'`,^STKNuTXuQn=ZT)!4B"[uqGg5MAam*!eV30R+To?&Nm-?f+ +2`cuTDV]fGp!!`D],nGop9\u*>HKuY\*\RHD9)P@45]Hipf@-OV`jP\HYKtc3*6"* +f(sI6%2>EU-ojFi$Ci%b=WK%smE8Z"b#n%%\^\Ju%4@\;*^+/n^T;d!8[`Ju+#WZi +Zh5*\F_9gMrr12'ZC"dMRDcB_B#_a4bTO>>9BI+=Ql4o*:hDK,j2j!.Z#H1Ze,F:6 +2JIN-Zjeid1?8DH^Fc#(Ug<:hh`>clgWLX&^\<0Nrq;pkhD!#3GDrr[bfIr*YRl)) +aN6_+%PmSl-9`(fTZheBPtsCC%.Ja3K',@CrO3HBp7pHC,Nk@bXu4bcio],3Q9UUM +q=8:PA"1/-ljQ%&HLe*Q(;>*sp8]!hql^!ld=-@biChq3pT70HSK>C#fRM<&H"p*^ +3@%A-Ee:#(;e:NgnA,&T9$^t6$Wmr@N`#uli2pOCW_fFM!WHG=1H2k'D*A'Zm3lCJ +,K2&.i0Sg;'`G$JlN[3a2gj+>)hIq5f4+QJVWGQuK)NGiLsB"p>P[LQY5B*%d(:@a +],BaBe))fGqK43cH7sHR=#S2qkY[C1$M>AmA0_X.j4)o+c]p +hU^[mO:\`!A8sE";apBER%YBfJ5O"@5m8Nmf`+O,3b_Kh/;j(K"=QeDJH&\5dT*IZ +\q%2]lsMK.N;.n;H9X7Chi@;L5RREJq<'YlMK7VkJSG/5-NZlj,3"lDFmj=jCb+qN>F"8FEH,q?41\p53O3N/No@&NSl'drjr'0rUXNWgTGL&kF5:%cP +^K(]WT,r)M!L"^@6Rlj7M!:jJiFmg"4rpTiIg,ak` +Mfr49cnD:ocLo?Cqc#TOk9(A@^D>2"3\Z%>R,9sPm$MSi>Vl7:Vje]*C;;a$[#E+` +r+GNg?OZp@d`?M/GI]e,nK3bbK)DsCiT9FIg<(8:\i`S/!*TACdS;V])"@3LW9s;R +L]%bVs1k"pJ!<#O`eOheQkIgZ+J,'d/2#$e!jk!77"E,+CbiI@p:o)5Oc'4$5P@cj"O6?"Qm2cbWgI+6X]H!E=&Hc'XZgFnBS$N^eJ/"pe9U7_1!PP>d +?u<^,i.10-[()Ul5-?Can:X70W$iLLKG)^H>!X>T\^HF8@'t-&XURh'[AMR1Q8^nh +7-It7mT\Qef[q8#g_ni"*BA7WoYR$D)p[Ce+7D3SI/j#9bL[`GI%g/Zo5.9n5oq=RG? +T3e3-Ih2>sl@8:fmKLE+n(/Vf!6"n3^OM7AS!SE8H9q%YB;@chMY\]][Joj:XZs>1 +otu+]hmpGhopbea\8Lf8$c?=81\Tb`jIJAB[r7>`ZoNGHD31'52opJ!fX?@mhi\%2 +=eNj&kN!/bgRiE1/%c84i0#js\92OnHB]4m> +ZtE]SS!?$HKuEnL=Y\j'Q/jpkJ$2=.,/A!05Ae3;$sFCart-&mR0'"G6(aj%G99*R +,f@m6s'(%4l*f?4MHt<##C?\Eo^m\F>WAH +Re1\JS6[jr8&X*^`P54#imE;S>AS@4QK]1&Ufoj[=rLAcpW7+?j6cfO]rJV,25[;I +!7er8,G8B.Dl0Eq9%/UM6aDt_GT]sB2?B1D-UP_&b1-i(*t +_M("ldXm_Yhe[!M_gh0*.+/c%oKhRLnm>N@h,p.K?QBAj$=q3A#T/7$Le!4fJH9Eb +*0+Mhs+BT,VBc7R]Y5!'^^,sNiGm`LUgS4H0KgqT?hc_baNt66#NWYgs$)DA:KO^o +-c;R$oN3U"%J!k$G8(#FZePXJnXl"p2VXYA["8-hs0ht5r`95s;XhF[/8ti>XoR)< +k;`E7XoOsL%dni2n[%&c-C;3hl"P1J)Z^QS<_!6`!K4'g^J/Z)>!6]]J,!`+&Rce/ +L'W;F+HhnO'M)h85U*=(Ut+sG4KF?PRM4A?7(+t`glYc($=m1uYHAs3Z$H8kp@1d; +7rLE@Fm!*.IAbdPT2L(4Pl-Bed]`'F?J'+>FgpYYDV;Y673CsZIC.&ipa#]hjr<() +f=0*c[a=p+`U@]0^:@DGnp8P:)k=r,,S+almb)9co>K&2[^pD3lFhVXkH:rTHT$rA +Zef]j>5R+njm>q89n)dFp=a:=AS3@]:\NtR[8UccMMNYsR(jDNX3j6kR]An62/'Q* +c.5b`%EsM1`WCepRq_#hCg.cfL_6bIcf$pSP[348mOOj9(n3pTBM2P9%;!epEUql_:NCS47sJ6FbFd +8Z6AdS75DW;d[Vm*'r:@[)>@9-IV'cbQZ'kKNt@45QrB1abX#nF<+kf!Ne;TD%>:] +Tecal:-`(r;V)8SCO,-tVS!9Ge@BmI[LSk/+_>1Tk6+70JjUUq2 +jH6oWkaO$`?_(a+m7$='G>c3R''%S>YlC'Yj^AmF* +"-9#kI;jDsY-$*@2hnJ>4XC%ma1A0[`JHPF`6;tX5I_)##4eZt-cPseAej_oV/K*D +=AT/;eIbMK<8/>C6OKdNM:&>5)HBJ38Ea&FK.;Qo!>XDlaS-pp-I4a7=dKi#J56&t>BNVhe(=QZ*YD!+if6bVo*s+h1pn*'A5r:[]J,#Yo)o=pL1G%/a@GiM9%17[J +J7pqC&KkEja>.(9g4M>]&s?!Y%/b%1"jKr[;:Ld +N.@;Q3J/L;c[DH*-&H8+m)9Vor3I"8q^f%'9nE&cd]ZN$S][$u&b[UthYm&1nEJUL +r$HG6esqc0A.l#fWHKC#!NYI[&-6kaFK_L#/]Woa@pPI@=!lFb$?RUXSI'Kf'`TBCqBV?`bfEqcERfd_Ho(:0T62j^Ye&*9l_BRk7 +@kaq(nDE9lUk_/tM:;OHIhaI5[5%/8!Csrr"FZ4)(nLfJ7`)u-YHJc?D_e^@o"U"U +^OMV,36E0[jc&1)i(_H%kN9bAIn1P7FNXE/?iIG4pERK(iIN\'GC]_lhY*VW"8ndI +*_Rl,%f:`%gJ(eqG%\-OQ^o^+6S +7jj>0\"7eTV=:=HZEGn45.8[`-nE"^X1U=EEq==l9e0Mbc-'D%ANOeQ[,$j^]/_@S +cH!]NhK[mZZTDB;@^ogVM\T59OL_+uBrfTs)4E$BA*%C2Jf:Va`[]A(2.NYH0iO"m +lB&H:A`VoT3@kDriT4!(&RH\W[8>&;B`0o5U7\hL_)aYbI*kc9PHFgB&,362jIV%i +hJE&kCYe`'p-3M*ke>=#qpn:*?5^C+h044?qrZ8195J:D&fbJh4pHtC1nF-\[++O(is3m;9eGn-9%DR7Dr?!\5 +s5.n7T0Fq)knD434JM[Z&$*HjhpIMQm/#Ef?*E,3o?9)jO.R;;E-LBg'-fH=%:!0+ +5F^rkM]LU\FftUFQX/m`>nI,5)jO7qKDas_LCE6^T9_KNS2533N5@ZWKR`@,!"k2O +Wl9r_6IuQpruhQ]s8*C9;F21]1s;(pC!%V%1(F_g5\<4^A?,b,C-W,+3n3G1+jk%d +:B5r5(2SZ6jNV$E"T(glLEQR$o:H9p:5&?&&$-#?r)Z/_5&o-a84!5([l!?4,[E/J +iK+.=LHKC<#>?nF!LNc'quAH9fUbQ#"7hd0\)k<`34*rlLq'GZQBKOUASXb!4ODoQB-PhgI?DV#??UIbVNhC=Z1Z +1aC.s6,ibP''0)Mr=A8@J,3-Or:0(Q\tKn_W7!T,#5NW/gVNer%DEN?$;YMS_U&>% +1p,.O>RL"2niU>CrT`*)SZJZ,iVoScWo[k8C'9,ct:T;2. +"Aiti-A1:tjQm?IM#Td<^'N=,AHVoK-^`hGj-WZ%)iR*!Yl-kr0@8eQi=p6X)I779 +%bP&]TB"l3i.!in->dR^C9W.jIUl'JBsp^ +,Seq:\ro +V-*'VdFRV.lKa)Flho1HS:\)EVS0t6Xf9InrfTf._^._57\m"Mdq*3e"HAY=JJ-I> +MY\jME!7/Z6t=csA1`KSJU0r?aKVn:Fq3!EB8._6\;ZhTU.d@Hm:'u,jII-an3XoW +#t2Q<[ea4A]K7=C3tghj1ECZmGMT[T"5K=3]fk2">N4kkd>mPJ*S4JacY2DfVZ`Ik +SE5dbYb]K?GJmHQkmOcoQg"8fr4bL*F:9,i[R,_\KF3qt6D[Yg&gaTX9F[8h_^Vh; +J@6EDaTf+VO#^L0^K8\8X[5)aBWDha'VUM6[!i-pjPfrbYCsKqSr&MVkWmHiU3E`` +:Qjp3>L`[:O+)7&oe-hPs7&)?pWjTQf?7`t^019rf;eTDs%tBCs6<+6=$L0S08]U@ +s5P[rGSb&LmpE3smE5t=nKR._cguYVPk\le8H8,$o"TH@p:U\l5.R`]h4*/+]@?Mt +q.#JWl@!Vai/[J])cum^Vn,1n4-?9WrjGg&9?*Z!R,oKDg98Km(np:$h,u*[[OIV] +6>E-W3Mtm*Ba&VT^)7ZbqEg*Tpu`/#LhW5-PY2j<@tV`!=AO%qP8Z@HS3V&J=R5Arsb;"GFRZ?NBX% +Z5KFl^p,"-rc`'2%YMi:'Y@$O[or"(b!H#5^``aV/r0LQaK#UCQ(`^P&lM09l<$Fh +O/eE6Q;jS9I4,.^rn2Bp-InMZG%D=m`;b[5)m9*e>mYF^gs[-bF[b\'@<:TtnIP1l +6Leu]<,"h`MaR'3@28QeY$V3[s(V2i5P+iu7/le+MT%cl4%MTaC`L@!IFh/NGJ5rA +fFcYV-M1S8_pO^J.q:W/20T(8@n4\gT%(58qT#uq`;fG8,#\?⪼OaH[WrU[\%YV +&PW-hT^FT@En?paFYlOjK31gLIR4),./*>4q6pCiW^$f`-@eCj:u! +EQ9F$PLBU&s89NoX9+U^+8g&^2U`Je[+VQ<[/XG*2NP2hlY<>!Id,fej=!1hl2/-g +VcS5%s3iF0i-V]jgHmNOLo43l]'KZplGiCU8jWGQT7d/XE5<(=&&c"D0Yn2EH'&4V +*9TWkKHcpops1.7%2%Oj!$$U.b6+7u^*a3K(PDO?r.Er--n4^u[.)&KJAgI]H+Xd* +m1Hrn;O:VQF@l1G$9W34`7pOLf5HFZc:'7%Lf`Ap/:/dNE]I(h`(J.XCPAs.jLqsX +52&<[3CWo)/G6o%kV(5T+.bsZ4&(-9Qgq0u`](?Qf@SDs=`bsD?;BR0,uXNB^\giF +c&M.g$VoPmVr3eAM*M;J/]Fm=0jK%r#.q@K1*lUh:nC&`bL=gdT:k9T]U:"'$aL +KsERL[0EbbUSXi(a*Os9mtj@& +WYU$LB7,J>qgKP,rqu%:o@HbYq85k%IseZ`ZR[o*]j69Rr#a=6Lkl#&IG"1;nPB&^ +C'PV^1DEOa12JR?1Oj`7Ap! +!P*8Lk+YcS,/?FD%PF\(Mp3LKE#jc5O/G5L>r1:?-kf'M_PeS+&3%^o3%TOf?4IMYTOk@R$S[K*a-+fC +'`h@K^h+-=9Oao&ij+@@[oMFmMqnl"a!p\,A:C^Ei.D;sLP&2[!gL5$81%P>WA[(( +h+>d3?O2ohFJ`Q=a[eMg57ojnpOiLX@]h=gr7a8bk5F;sZ(SPN.P*Ajj-3_+Y$;lG +JH$F@P`%ooKg?[]oDNEmjor$o'VG;r,[5on,:^mHg,@XDRr&JU[mdYEEPp22l$1d* +I)XgQC-)p,foH.aeuiQlc@V"S'=)"lOuG29Vls$]j6KZXO^.[Z#*@5R[K@79T"TQKuF!6s5(ShJO0"V +0GdoCl"P._s'uEq2Za&",@G3hYlAS8A5Th2Q2!QVk_b(SOl=W,)L@V8j_W8c1>2&O +je%,hn0FC9PMYoa7c+lTN[N@fl4Q]=J7(_3O%I"sYrKXCY'";!arqkWA0 +;W$dVKDF>0U!q81]#5)VaPhfMfP[$XXWiZlBX[ES-B?@)i3cS**#1!,P*\'!kf:Z53^7;^eo?WV:2/Y.is6PnbDaq)sb[!G>T2o$JGX[WbQDp6[-t)>#0JuPd&qdS_;%!ZG;bKW2 +JUqVK!VqO>5d61SQ_A@YMO(#*"\Mdf9KC8D//L#s"XmZ,;FaQ@Ki/UW;eVSM2Y5:p +13>ksj'K]e-[Lid0["^qJ\I>DIYcf7*F2 +/QP#6rnr/dr+Z*$K;L:1J)7?Oq>ZMZec"Y]%DR1fn_]quro,Gu^OKM`\!KP`l[I]7 +mZ`*HMtik/V6=3)4`e3rOcpg[C`^98\a)SfCF@oGmV<"%a)1*t=VWbAHs3LZ;.Z4u +QtN5W.B.;RC`!s+`[%E9M/cL2*eb;ig;d`a$0e,h:uO$7T`n>EOGeXV<^AADdhq3D +R)B;h1iAR)*V-!nRW;#NUe?qo$^5IWas0H(GSr/!ZQB=%=AWdQDNW9\$oFc\95idu]7IpT8!g^T,K)tf&FQYp*?AR(8&3XB2n$9)/augJrjU1Q+9KXDY@^2%),ZS3_#?QS!$_H^!>R29 +\RATX*`Lc>S"2>eTd;7>W;3ccSeZUu\t`oee[Of.flHb%]6f)BcL+EInI>B#mJ +Bl>VNMq?Vtn?duLS^Ch_ptnU>:OpdUe?NPY'9d71rsDrf0-qZ@`=aZ[IkZ?p/gKr] +KLLDQ`5dPAqu2ZHF8cCSr(e!dKd&YK5B,P10)-OLn+#uMDa<4i8I7EQ6P)Mj&*=CW +jaiMPPl4GJ4'b1.;dcp>$+0NI-Bleai9U%U]ro2MYB:[4K'(LgX^KVGGK,hjXZuig ++S5RMRkWL?b:SI6IAc1IKj+eG<5J3*P_m=XG5FqiASNQEA2e`4u +M5-D(bU6Ysqoo:'/estK0&E$H?^n&uRAsVcBj-qdY0QRq\NIi;)6_AHetrYOUYTL4 +NY7\Q5f]X3CXHf5Y5Gm>d*Y0L7kpd@(q#T;3_N:N)*1glGm)@@Tqqs(BV]GajP_ls +,"=uCLb#I#RS(%E;Oo4a*%goZ[3;n3cJ"l75H-*d2--8j1nA^Xc?Y)Y"h,=.mhDA$ +o?V-iB,1Vse%G(A`q)OhN!R$SFe1pO$'Pe"`1al.X&>5q_L%T]o +G`!qW.3hZH!,)VS,uh7)MA9X\q,B0]`E4P.6no%f&P=:T6K()"f@6HT7EirGH-_H4 +^M]].%&+h/nUD>QrCtLc\srM6Q]M(4WVkYg:lOjg\PiCj4*>e'k6JE%r9NS*J)f)R +RrCmerYPM4l'jP[_Y6CJ62giL(E7hr>gq#kAH2:!mf.a'qj7&]^Nhp@r>YNM>nXet +:A32?io.#j44i2hca9=%[;?,Pr0'm'Uk@9L?Z*8)8BnJ]oNFCSe9L^;@g +M@Cf)_[ulS!#YobV\KaDA/5<]5\OI=?;O*,#F]-'OFKd=dh-W.L$e7nCOT-Qo%Oe"(TC5\u]ENgm#La/3M#?Y(.edbcDkSb[CJ,R-U$Q--@e4M`qR1gc]=pI: +cef8'o_.]t+:Gb(&_7uCV^FL2e%EYFO^9QY`P9)$MZ85,3;>]Mm8#+6s&\E$T4H7Y +QlfGRNY;B@Mn`anHMA0CofJ_bp\hk*Sskk(:ZMER6Y"!L;*k49Dd\Y7p&FLeDgYeL +2'n]ZO[/jU&TDD..#n:j!Y@h5UDb>l@;R1\^X(HuGIIah3!+-lT1o)@M#dBPYN)VC +d6tV\@Mdo/NW0@85W#Y!-_^#UADXiG*>R&c(ES)O^_>]&@KPQh:m+TVrYH.Ae+Gqt +pcf>NdJi<[s)A/]aPRdJqBPP5Uk,HQ+D!c"6P7;Os1K&3>=?Z,B4#4q8R(=3'Z\FOX1#[)cFgL?r7h1o]0iY)A'DtlN+]Kl>2H`td78QI +>A'r(Sn!6<4:D)eK6,uN]`%\VQ$P*?hMe*DrQN=LXF#)8f1kAo%'io\o@(CLM8%-b +5]qIY/j<5V)ls,S]f!(_W_ccb\$ak^/2.E6$OpdT$4W9[YtnF%eQ.K0$^&";.8,c. +'Xa5\>3SN7=9[uh6A*Vde=l\-Ui1i#!4`5+^R,V!0qq:Q&N&?r&oB&EKd._hMm@tK +Fe#us>HkA7]PXT.Bs-@$YtmBiB:Z\V%c/'BkHqea]WUMc@cQ.0L:u!'c/WB9/]D]q +T!&>Yb:%-L_jEsY),(KKk5;EXYlAhO[f@ZTm.lm@!',t!Wu,IMi%7@RI,rI@JKlS] +6pW.,0Mi=S0L@M49I";;J^0LRS(Qp*2=mgH,;1_2^#Pp&PsTsd+8Gb0GNA_n_7\Yh +OoJe7(]P(Cri>Qgs.'@Ag=ll*Gh_CO`;^s0c@Pn/5;3"Ms8V'.g>BSK$i^1fZ]5"S +7)RnfDYio%%6qM3mZX-mmp9gXr>b^f?/D?GU3kL&[q&GSP0I2sFk3=H3E/;&b,nat +QP>`G\RA+)T9SUf,hd7CAlPM$d>_Q+6XPWLgHc5*R?Ea>^s"6X^q;G4TE%<:EZhk; +"[X,1:aZeCV96.Eo4g%clp^tU\nJ@#EhM%E70*-eGA.)H(9 +^L5;:>kJAgh;YM/P]d>\%Umd79KL*Kd[4;["moM?OTY1g_#FFfaJTbuQ:/+[`B!sN +r!@l8Il[W63%=@J+RFS35@@b"puWm:73aL#p]R@P8&.Cj.:D.>/OI6j'-`:E"S2`V8pV]]OM>B,WWIQ89l%i$(::O0Rja7J?G5'f +;n9;NQ_rG"%kW=[32nT*Wq9e9L]@mt&W"(*5g"QC&:Q(\1odpAWuP&Q%g9J@4f7\l +=LqQ"8"nkup@b*eo^;thc@cDJL51;#O^WLOQ`#H*o'-RM6W"^%:]/hP/cdPDaDmCs71=2Jk^8!4UJJ=uZb +rXX8-_#LZY(\g4YlqOCNjp1MP!4gR-5lN`gUWl%`Whbaj4>@/`h\?_`_a;-N&qL#H +7R5miM?4)]0I[DX7?Or2S.^Jb.fXHm&;"/[-U1#u3WFH=GjGP3a79atEnV&Dj?P>D ++u'9!2BW)!5q2?='Due/)K".,7Z5&ik3G_CNK-6NF?.hOdlr'IGhN*3U"(tcps6oAPhqUVlX>8[+ +;K%69o,d%hB')8Jb\K]c-]XmB)`-2o,hQ;YCFeq!QMJ*AC!-DAjaku'LbPiJeVce=\PfQ-tib;bL+^0e6@o%ld8BN>%/^$ +(V^?:O+WXkTFusJnQF6/%[7+r+%)G;Pj(]k&U]5lb&$%j@`aA5nq:>=+P6&TLV(Gc],6e1`V)+ +s*O'Vo)I/XdD+J+p[=1(s6%i7IsS(&MZ9_aX2DP1hh>g[nPeFs1AQElpAb-N%lJSs +7*D4:'YXeLO"Z37^A_K3Y5d0bF[(/Up93neq>MsG+(!jof(o0/p?KA4U@EdLc>@:3 +V1eJ?7)%s*X.KENghjhCp?CaNRORL<\7B#I9NW8M!LaTTTH+e`L.(nN0J"u=Lb3+G +Z6JY(P*$%>dCk'K +I=m9X?fGM$g`/ds"H5[*=TI)k@Yb#r"-72f0!WSOEI4J]r*/mjAV$Ku(fHMEM1A.( +3>0=%%u4?X[32jSK>-K07nkV5-eD;V>:Jg+M"`PLe#AuJYV.Cr:P[]90k^S%f`-RZ +idW`QLW\L6U9ICkS^Yj?q<"oY/Re"i=H@C1Nbs&g5l7IrK4gprk@6jqiks@n7Y#9< +U[Wpi0>p?+`^W]nLA!q9$Zj1]"gMA][9D'T8"8,Z(%`N-"FG4ha7Xod$QjnfIl0#K +5KB'g,M<45o04uE$':LgH>L$"-<#\#f=&3=4?]m:R$!G)8ojnQ)TLk95;OaVT>>_> +63Q4fLR.!^N?T/>7QnIXhJ-f6>As_Tl1kk[J&p4Br_LOo6NeYJplNg$&hD%S+-aE?AJTJd,EoP*C>rq@'g4L +Fn@hV1I@SjFNk7c-=^^-f+(IXdFTp(mMe#5WS4okoA-4\.a!TIt$Ml-1%@cpg*S +.oB-ZFN^$]fb'])hkheEfHhB(38Z8pl$.GSs&:io.7a=_N:%ljB-FU`L]P?tq9)EF +MHKEn#,FM"):\a>5ReX78(6CG%V68L:'oAsVZJ=^C;pKHa#ifod+Nik3)ad6Lq2-^`:"fV +n_)4e8n?+8"t!A6"u-,I5T+;C"R(RCqI(Nf!CD*$=Q5^03%(C9OLZ_n +SnY0J7cp2*HKPQC^ca@E/4dF3;W!WgW"B2Tih0T\.1e7B",?k45@ip)&V(f$g^DJ: +LYFfq#<1o.*M`\%9S?HZGUS7.6M,"G)>Ik#!\cj#YR7*?4qP&X!2BEGY4hB=(GlIc +1dM4.?kE*T!r!nr3R>1\nncePSj2:c%R;3FX?$;8IE7g=L=9X5OFG7,toVc[ZYK/ +q/q/Kc,%')U.67s*I03RG]j8(ZAH<(KAM@hfL):tTl9AYrVtd.6/OkW^:o=c85jHI +d&VWfZj!)\G4Q8+Et#0.UUV9cdHM?CAj-8nlhLh5_NVkkE1$]Rcnsb(s*+\c/A]5r +@n"V%,%aJ&[QX/qU_Lq._>c&T$.W9U[9'*j)Dor/9s]s_aL"K@*bAZ^j[c24)=(ZU +O"rNIXQAl_e_^5sTn89_r3`g-[>?-/dfiSJY:c-BC+F?(5@\21E:U/OnS`IbhrZ?$ +bu/qQVVo%b]PE7(ViNUB<6Q$]KrU9K!1=Z/rhgq=2c*0MK?F.cJV?#d_/oVY!uV\+ +dl;"`!k\g(s83]`"TNDGd)?oF5CVQMn'/h'&)Drsc/eir.I(iT8,mrQj4B>+rVidZ +cX&0;qBj:uQEKi4=$Pb2c$pl9HLUrIn$0.tn`6V1F8s!]$hF>XX4=t3Z1?Mm`@2%7d@I*+/ +Mc6.F&^L._DgY)82B\G.i(jjiB*O(=!s3`,#T+7/$jsfM!;N1H!5+rC??kWo,Xg+NjO +pt5Lj*l]>lCAG"9Sn!dnS)IPKp':TsPQ/2nU]2oe4NddJ0J2i(Md/NZ^qhRA!R9We +o)IMF(*U]?584.qco_!?A/Cli-NsuX5dcXpj7fO0q%*:.:&CPH\^[S15K,r>GAC7^PS"ILN8L,"C27%kWaJbB-#nG[;Iqc&Jt^-/:Z8t(ckr&6e;g]!m&YsE4* +bkT0D"PE^.jbXpX-?!560n%O9e.$1Lf<7qV>+UCc1G)0L>+*p6`%r1\r_`\`ros2c +QM*Pmr-7-qTD>^S*el'upMTYD]km#$P0imD8@aADq8!6cMm6S>b6"9&h2"+DQllQA\\4$4^B:JN +r1*QCkCANL%Ea)BKg2XI7%bAndJY0l;CNG\qtpf)8*4td]JdFCb\S'Zbn?"=]P&!1 +kSb6oFYiVhF,Mo6g3T`4#$i38]18jNAsACN +J.T7NaOZ0+%EMAdeC>SUe.ub6o9ZKK(91JXW3ZCmK#E=Se]E/sS.;KmalD"44N(>$Hg=/:dTCR47&Hf\Cd3H*TQ^66)071]9Q*qnYXlpb?fUBe0i +^);%I!4pgC&+`TKTKj[+;'9r&KUAES4]M)f*t,2p'0oE,##5:aj7DLHp-3p\'=hd= +#l,&I7Ed#dX>(R("nZNT9#`XhkRB-?H3jI_'4:G0&%>L;q@NMjPPDM7]]\nR3kWSt +US^gOi=J3MR/Q9)m*@k+IgWm)H]=Z3gMlh$9io*LqKr-Y@; +f.U,%gU!1FqGd,P5$IWCqoaJ68,5)]/:R&8rYmroo&YW_Kt"umcM/nJgiY%b#F'"2 +'PM.T;\Rs6;>/pee%e5tE*q(-5C9bu2qHVDHcNp%66FU"^eH8.1!S\H?1BA_e +NW;i$^^_5BOA0-OY_eDqZQg5ZeB3X&H2G:*M\".@>EJN8kj=r#U_?O`i#AC++ZQ*Y +p\Q0^UBidsoTPS*4n(C$Hj1Je_QVb&KMV7AR@C+"`;^)85Na`YJ*Sr31da(>V;X[i +!kNZc@?I[!BKAN`6*"]&FP6qO^<(%rRuA>nM$nk3$!&#&Af')?_Yjc089-]';O:0K +^V$n6hl"7?`?#0oUnM_-_2uR8rDn43fP6aJ:6)H/qVO=]a8V]s(e0eKJ)c@?0J#!i +a\3pWgSimkVJo;2]^sA[D0s^dZ/Y!&oH)frpRFnK?d+O9=rmq3KBGPq\#`B]\hSUP +c0ZHk[r(PSlc/iSlZ5"gc!#71RpKM^WUX9*m]M*?BK4K^O5S"79Od5Wi]fM"k5WW( ++[#QrNMPQd+TH\.3"d,8q"e6uDMP"aMuHdR:FX:2Un_F/E;lL*IsapP0eNM,:JXVr +AgGj3'(Gq=&?j +&EN;Fdjm;=gc>E7h>a(Ni3<6V8HFhO>5jX$O8(pRio4hT[/Y.!mq"TEj+&[UJ7EbC +XDXWS,IDedDSMEtMZD\.-,b?3r;ibH-[0oJ0"2Pmn$%8Ws,6_#cosHm>&.e,>'P3P +cg0X[4\\L4Hdc?4(Tt90.(^d2FPh+&q#?`q"YX^/^:@J?[8R7tT$D]O1unNtlFe4X +4.FaSX6B45og/e0`Ikm87^p^<9>bERlMhWa&Ko3P= +$H?:.=&V.ni3&6SL_Z"aWaMYgbrubdh:gO2X.GAG;j(CT+] +`HZ[7jRMRKkjP\I(N)6+d['0kY21nj@$kBK:2i[X)[;$8qZc>4Hg:o`RRVT`_`aNP +?7r%hGTV.8NsAqjNduReXp#ps!9OMCcjl!A$mQVLW0!g*F^,"#Vkc.Yl]_Zs9(ks< +GA.lgc6^2//+i$.!0lA;NVa/)n:/60j\PeM#T7pGIl[Wj,Q_F5/tE(@WrqtqJWQ24 ++D0ZmIc6Cg8Up8UUS5`CS6o&&r,D^.4hCSZIdHYM>gH3.;*Quus"TM>$&/bpEKXh# +iYn`Jn4.,]jT%^J`;dFE`\14caSB2HCWo4taiq_.Z7S1"!dG7W(5K]j'\Ze9]3@rg +*!,Q4RfNH(GJ_f-pUTAd\"(1.=:GBQ.XZ]Q+&kl!l_ok?C(1IeGmO. +-%07#bWIA]N6.V($MTc7Yg=$2G,3>P=.J(oHp)i85*5f@o^MOSj#=&6lt_)lk4TIp +GAbY;mb46;mbD+Omat^QrPUf$J_cu@(f_&eZeO%8omac(N7@k=!Dre0s0%M1i8k8m +huau)jQ'*!r.thJnaH#-@t3k=itffmn@r#Q%fW)k;,'d-M>om@3WHu%nGhR:(a<@. +;G9T2)#QT[0*D7?[7[Cqa^$U=s42hWhAoY)bFsY+_!`R$!1j7S'Ym,@r54*:O<79Q +i"+W/OaGk]QWjO,?d8T-PZL6t^u[WEJ,4s*52PuH(D-UVB_dFY>U'2UOE2Ooo$/m- ++9#bg1thE(DalfV!70pG5QnlP!!N)C_!RKOg&Z.V!E]9-r1,Hs_^Y<3`AXdH2A$9h:%=,:JpSPUu't(hHE@OA^L$GoTU.T@RYJo;)_7CZ[M%*-bd/< +.]7Xqq@([SZrD\?.a`&9ROUVPV3oA&r2T4!#p0*,r81OCdDLG@.Xt;%&jcq)0GLob +g7%TEKnZ;%in:aQrUFq*@&7eK.PtJ&R''31f.J^WY:N]$<30cU=1&dYMRF_0WrVsn +:"gHkfFUFNAZ"[j54*h]_N5HS&HH):p-3"'l/LTRp:o(-L]:C +:)unLega+ZbXXTM1*11=4u^W1gVikfSG"?U%0]iL.R/?E(I2B$Jm# +)m=_7\b^I:eSjq@rT#A%>Xo^qcE*5o3sS[r1>VA`FC2YICSJgpD*_l +p>U#J=9QVCCHLa\X!J:qo5=3,p)NlG;fP'$iL$[nNn;O9e;b!hp@ZsXpXXH\e9,ue +p)e2il>@O*Ae4X?+UH)63=l5VLX;L]IBq +'*&UE*;7-jY2P;g*V)EWlT^F1'D=Z,CuH;7>a(.[[c7Y7V#]In$BWeHkjcJU@",oW +Oo`N!I_4$@;g:V-Ago<]WDb0aBZ7C`O`aE8R?[dISW4SeAP-O*<*C[2Q3eG^`9!Nn +q9tVDW=-o,hnKsF4I6ntQlW_WFc:#6%M8Wbgg);jn(sJIkN'Ln]8MGH\n?6OmZWRE +>rrEHUY9E['E^@%]S"l2gqJepkdWTU"VYJDTl^RQL*g)BSMUqV?DX\'&7F +17GdrqT^B]g\aC]oSdg2!Y9bj(]VGRj!-n=!p'7i/HDdM%6>6<+P>I=,/=QbVkG># +jg;(]"n6e@KAQ<`^F^F=LD0_2"n7:4.p6)/h*o*;=2t+ +6EV.J5SdaT$C=i^oZ6C+r9\Za"2FFR0j"q`DoR"JCJAWYM$*<-g4U6>`9@o`mfIPC +!VT!Ura_u$I&u>iFQpIh`K./;]Q]Zn#^C\(?.hKH +YFl[Ekii(ne3uN.ADB)d\osIKgW`u?@jG$4J/C:kPc58Hs796(#5hgt9cfJtE%r7Z +7_Qe)GR+,eZMFn)<'iQ,QRbet+:n;g9+["?j +>@_Agg_CKHh9=]ZE]1!nl^#P5F"OV2Y$!L1A<#+WF'Zd`T=-Uf*=WtUL:ok.V-)PZ +l+5'=[81X=5.atfg.'c]lU>`pjh76OYCt&\QQ2c?J;YSq4lMgamC68H[4^S!ZQV2dJJ)Obl=no\"Ma+q/j86e)oLQRg +Nc;fpqfk9kIl;BL6bV.P,IIL@ +EkhZnWh2@;.%&od5AtOt%kJ.tJAi[LcN!_qTj^&B%\f@&R)\sA8KM9._S^Ha^O:Y@ +rc`($DB7>Ef_7TN!W2j,)V>2^57fYj"8X>G)%](F*39Whks\pkg.RKca;g%lHO;pE +]jE94JG-Z[Y#idOT_e^o)Nf)tIVESXDqUc\'DZ^kjPMZ-eW4e@F+UH^B[T\^%9"8& +<+@N^FL+\lG-2/LYF2Kt(?[AEo6#RqTcP?IchUpd_#>^s5=pJ2!tg'go)Q*aLqsmW +i.(a:+94so#o3THNgs#F#-gN%nd"c5/2F\;M@Z.086pOFr_#TD][/c.&6l$Njsc.0 +>%sOigSa0*R8+C`X;i>DVe]%pg9^LU%<(u9\Go6Mkh=6/0)IRSn(=5f2ui=$]t4T3?X#oMU_H:MG?+:FuSgF,LFeDe]$As81dfI*!VDl3lMCOr7QNNkmN8`*iD] +kqera")^mn'`VbqQG;;'U"+9:0"'!L2c!-unF1oA%tAWj*ua4X'E!nOjT!n9l23ZX +(P_^Wg^%gQZs\i-$XO*>g!!'%!Shk+cd`ZtZitkSl9ceH#mK&Q#@FOhe*M-+]e5#Z +.! +nd"dP/bSb40MQjDa(,ugS0G",Ci`3_`7XQs.B&gDY5k/b+Rr&e/ +Y9;ujV*`u)#l@T.l!g[\[cOLof#sXE;=_q+ +aU9I(hDeGUjnt6bY/u%b!YZaX\q%7ciiSnmXAU^Jd*lFqQj5cf@GJE7]Pa6b\t?7T +8YoUhO;'_NaMOf2p%3O$'kjn)0j]US9)$.lrQtO.<^_(r0O%1QW\UHYPS09*@3n9E +H_d'8ga/ua=&tB+eh-SQRRTM"oZf.nNR[81J?TV9XhJV$D-]?j9@D'(# +^6V*(#,rDTN$mf[mN5nEC;rbi`_2D[!?,T9,nE<"U0-n&E$OG2'!%!bd5ln1bS]&cn;D2Yc1?`$bP^>6(6jci0R7O68a7amh=Jn'TP0hh`-W +9Mk+t!TJKQ"dr0IJ)GeW&Us/#7g#jumQTPfi)mpU+F)5qCi8?Y?iJ"A\hUYe!6mJm +4+[?+d*=I^B,WY6q^ftB&c!+?n&p3cJX@OSa7;/+f`:HYC&dMDD"u%G0\El>Z@CVh +[r=S8O;,JrE6o`=@O=W=spam>IV6+665% +c,p(`1fo'@;.*6:d4#^*k%.'GL@7+Cs&=C8I!YS!Q*HfS9S*-13]RT1^LqKahGXgV +T:UPjbQ@:lfu8`l/HHBSf)Y=!_)!!m"lOkQkC7dai-6[&'+qaB$t1g +oN3:tP"L`['DSqIOFJt6n+H?7S]@l+/sE-\@-f/Xs2&72m$sO>T+qTjS583aei*[X]b7OdPn3s+MooU0H(bCl#k2=$L[_k)?C/gp<4X(W +VaKb%YHIE8/Mdi(JD&&f7=`;Q\c5%(s%*T[47Fg=As;'M7/fs5^APhFoIXG[q#2p6 +,HYbVDa7iV5Y3U2_!PNS]*k[2i:JG=#P^Sgr\BC;$K_SP+V+?>dDgU\9hPMPj48^:78\]_='7 +hn,3Ec<9;aH]0I:kk0)j@Knt/6VmKULTHHHQ5DPS;K<`)<[oXj/g:!3jnq>? +:4`F8l/I2WpA\MTl78`l(GiWp=l=d@)&%mWIOPo\Yb[84D1^1WH1bi@_e`,4ILNu& +j4?hrAm*]a>4\aiKNbhaJ%j214N]/PgLj+kdX*a_Al8rKV,-Qnc_U;g"9Ka+oT7E4 ++J;i@d$1O3FqK';`Eq,;O>bEO"']A4JJ&Y=- +K-6g2J8^PXX:"t($gZsR+,84X![N(q@$l1uDUtW&k +s7">s/:[OZ<1Sbp+FrF$Vc!fY+J;n*!'/c0W\Qu%5F]a/dNc`@,XJ>m3dO>#BpIWU +O%J[s8-f6k1hafDg!nsA\C,]([+1Snl-&N&(A?=]*,Jq;[_?;,L@Un)ja6mSf*jt, +YJ5)$E+n$Oo)7*6r+Rq_GLS&;ktN+^_"h +;5nS,kD*`cXAf0r6NEgS>Y(su_Z#>ehe:4\QU6;n62nh&O9KNK52`a0G[:,o='j7C +=67c/#uV@3;ZP?s>tnapROT-pjC@ujF+SiYi$SoAHt'b)QD?des+9/1s#7@lS(km? +#U$2HadLHaNP`8Q5*:?!MMp4Q?7^R[^VudYZ=fOFCo[pnG*0s$E:>_s^m#X".f@%9tU&X[J?L:+e:T0'Yqk(DCQ<>'FaGm)bnLmM/AS"?m +$fb=1'$?N;CY.OFdV$Y$F(DcWo/G57DJLJ=&"SiM%OUIOTFO"Hp0f`dFX,C36R,HA:BnD<&DaIc4]+dtAJ9B6 +*.>=-IS]l)3%]DFh;.Qb]+\H +K"`d-?BSW$`%G35#2t47EUQ0")/>]i&&;p#8_ +&)Q;U^W8RE+_X`Ij"dr_hDtT#[i`'+NIi8C0D\Di(!j@/NjR:QD%A +'5TibBHNfKViSS(4G=%mJGaEWYKg,tZUs[_ +bTE^p9L/NZ;bQeP/>Fi@@a&iEnsoE$;Z$V."PEYAVDZt>bP"!fI*UMt-b%Nj>8&Qd +Xi.&"6pPV^@KOWc\h_#MQq%/P#CKZ\YJf`Ai=j`C'%-h*rhkX'&b1@!e'J"@&Gap: +F6f8SoaLu8afC.4XQK;1puY)%#FGC$>DCOkBs=n%2R^RJYdm.3%pm06ouI-*2j=CC +@.$@o_&NHF:ZV!8AN9kG+Fe_lf+uFNrjTB@Teq>(WQY[ui$DoNNRH(RU4N!sI2f^% +]=E^o=XSu&cAs550GX`lH#e9(dlX:rfbO>LmYIcbd\!,*4ZQ8eFGTX05Q/D>g-$D& +?#@+N^X\+B\'ap7BU&Nof`*!65-`@s8iJ$E_>iI"!;VuA"$^M=Ymg.%nefC+=??4a +Q:R@>139u\9=%OI/P]0Ei>=(u0!2M7QJ:b5gf^tj]6X3;gT'5XFlSsL-b>5ED(2(C +0:URjgsB6/YrTS"3G2!CA>DRT]ifU/ +]J&]\CZ=57&HD07e2r$ADsp#rO/"3-[-'>&+)QfnmpmJ:0_Q8;"uB,''PaL"_#2H5 +fhSLZ]&HS24Dd_RXZ)g]O>EN@,=:ti"off( +)M)MKdGY.K5t-:QO03:amuCJc27P+bLli9=%&FBk5G;gO8/ae +LGQ\n0_;@BdQbJsjfbSF5ilmQpm^fgm[HfSK6iCAT%'2'V/h0IH?(5!%L0\&S;,WP` +^fQhgYiK@8n36pPLD3RMZ*i[55Z>\RcitS_-_X`IATUPTP=rXZbb,r+VJLWG`J-`*YE_$bk +\HJ/j"Vb)Q)"-.f*RkiHW46(VsnVI:SQe/O_TcaX8M)W$-\+T?9f/o@[nDa +)9;u5rZ&[Sk4;-"s7+KO7/l[W:aQotm/$5&U8X4!-i\LU,fFc-Vk7F=*ep)-,$F4nI=K1 +>oC-DUn8W!5<#&'i-=Tu;qV"$m#8Zk48Qp%OY?-_G^_%#l#$'ja-]X20d#Xkgp`,` +^*D/-^VrjXek2FZTSYFWO+:>aR^,eUM"6$B_jn`nV_XJCZuGsS0?Wms5U;Rd,WMM,oFJV+31pZ>bue&h[dN^chXV]cnsojt +3sOe.#0XJo/K3UFhDf"7Zsjair]XqIho.A^Q#?2B(](LKS:E!tD-$OYkco)LX`=0pt=s8>-L5 +Tme>SIih!E1c,u)p8"EeoB7!T&Pr0>06a.77^2nW\V^$`1N`!`7oJNAb+*gM98`b[ +f0i\I(!jQ;VM5)F\JKp?cgn%65PmYUBE%eEN3DZoq!;8^a!XdA5FhN"p$nCC76e=- +o"hS%/H;jLgDiLh@Bg?_Bg`d#*F:+K,'QY2VLEP_[o-/>5(f%SQZ2'j\e +b*(ABVA9uGPndkpXj&-j0.gb#Q*EXSMK5koJuu&''O[ub3<\h9aBpU`Jc@;O+ip8U +UB#$OJE:]NK(DD@Q%8Eihl.>%]ETGN^PN2l310iHEU^3M*?5-;!34SqYG[8*;Z'U# +3<'G@!guHrg33j=_#+]$4CW=4J*7l$Wu0nnQ2Q+*)5q-E!k413>1KV2;(I%`,R++] ++ofa]7ef.+h^M#>kGTo`](/gK7@IPFDGXp^P\GGd3_#TVe:[d(AubjNrjQi;";(0/ +lra7Os1($HcfX@$q#_BCa6`aBMh_OjrD0#F!7_RR!<;*)1H^'eE)FsejqmjK6D%I% +E)AS_I_2i4Ne[UrF?gBkM(6bZHn11$X2,rpEoA8<5lOu&o0N@CD9KA)/D@hKFEap9 +mso0QI%_c%^&(,mT\bBKtFu"t9:[Gs#O9fDj?-_0l9TkH!a/k0Nt94a;\tqnGQ&&J."[VccjJ +aK)IRKD_G<#Pn5u*2f-?F0S"0H:tj#R&MfJF@NFLbVr\D(;&)YPe!=;JD6>V76lmZ +S*TTU"27#Q"9H.@Rig\:EsM<-_*!&ehr`7kq6I,es.+%agFFLU'+AJ3emrn\#R4MD +4#ZSHmRfOkPB_7)iVq6g7jfLPg!77C1ZIMe&[+ku]Y'0mdSBWHPI+uB&75Co(\.!, +qL';"kM'tIk?&;Okb-0=BKHoAW!N't5TPb[SI>[UaN[.3(;lBZe>4i8qB+d**lf[g +AZgrN_"E0sp4$C24XZY[c]21sQ+V!SL4!im8Ki&BZDnpjc,%8V%4cbZ+[CKd]:IkClVejs#an5af'B?0@ +GGIDWST*[Ugq,MbIW0A`a1!=AFOQtF%l$:2Ljm)TrFS.Oe\R&S2D?5QAXk[-Y$4QR +-:U-iW&'Uf:.(RG7UHiJMFS>Zo^+3L/foF18=u$=R4R'N$r(B`GKX[3-)$eMVZjGA +!IYKN2U.raaIl>Id>+*&!23g^D'J%jk'9DV'0bQD:>0X+d%=ptk;P34s52]SF7nL& +I+n2`\GlNm'uAF%L#@G#!EXa;s)0Np^OD:;`P%\3SBL]fM<*P6:42`Gph0=h5:_.8+#LSl/OM +$C$jQQPZQ$'W%.\UmIBZPa,gGQa8.kfLe./:armc!J%&!Mig@i%)kHWW5SZG"6Lsj +63oiP!,;GBF@)r+]hn_JGi4'd()XN2Won^j,R)6s,6,4X%3MPBM=[dllnF^o-IkMY +bg3Juo."U)..qSr:E@32Va50-64"@D5-r%?_Y^ds#+eHm57^L8UAanZO+8.K#]'hK +a<(EX"1ei\4"PkH>EPqLA,cWl<?tLH*ZITL +O&A,rptt.@'8Z\+&HJFL&b,D)J1lSr,P4f&8V$`P/cbr*_@]W/&-Zlp]G%[14VsA! +]^lWohrr8cq6tXU2cSm^fAuT^((@Z))t-ld[CKs`:[+I"qb2:eI#7 +;L,7OU-qsa_C,9(s!GDf2h*BhDEAR4Kjk#!L[_1cPYE( +8EEW^IU[c0>O$Bi!:jc_s7#3<0PuAREC8mM@fCR!dfJB(gj-sQ) +oYoZM*:q7[W+UhRALD4::?2"St[Lt?i,Y3_riSt\G*Jt=$V4k@ljfSM5f1!_bN +j-Ah?%m")f]@F9n10X\rZsbCubsU#ZjQ.TkQY+@46h.?SB#kX"("AhYMKjGm!kg.V +;%eZ_^FD,Cr*L;]aA8JlV(`9;d;-'e.[I?IV(B5R'XVK(Hh5h?/D8gkEbjjLdESLU +r\MH\]q`0,;;Ifu4ZoW4e_A)$j.iW:+m%;So"N@aHT!M2g6FnM-(XE>h`Suo@j:lF +SOlZU^Nj,?mr+'[k]#nGN[Ohf8G$7"1]Mh9mGihX&,&E(3>V/_B)3`HN&r'Djta:o +)f8;>?.$0A^F>$uRY[grcjZP\?V8'kLN^REFQ68@isOM6+52RtGV"TjKH_>U,-YP8Q(m:E*jh?uck5QVKc8+$o]enk(:i2e03Zc-eejRJHaOs@FS("AOf&C-DsE4MD2[7O)qu0bmrTp6>@[Vd'Y?CtY=qhD;Ce!" +'ad^HS6K=F!Q17hB9ZGm(!BNJb/++Wji(pK`G*9jiJ.&-&.dZk9Gb.;'oc^Nm_M37_ +\4$=]-il$',L=b*A39j.l//C-O +[;l8*SK#Gf",QsXPD^*om,jaeH2kd;o<:qA(-E`D_XdqUeYnfF\D75%$V#.E4AJ*1 +J']bk/Kfk"TNqg8!,75kOkOkXkJW]-[!8G5"1#(ugtme`J`_R3p1;u:XP2?51t,O2 +Dl`HE0GOPuRjLh<:/R;V:@%FU2Br0bHZ8(ra1`+/me%28&]"DoG<$fJg)=#!eZWe' +=P;K>ZDhhTF42V#IoFPEk#U:5Br6dfVWK)$r/(HIncu116_/]Tr_9hKSLIX>Q'.n+ +Pq&b*/L0hQ.jW(5ji'P/m!l;E5bf#,s4^5X;!S)'s"7Nih4.t4at/8uGX;NmX>9:[ +c^baAk7[er#-ph0+,YOrqn`15O!j"crrW7S70MmPd:GmWHe@BGk,WBb8 +kPV=)eF>_er:/U'a6^$)VM72JH14Zs\QdOrP/1Q6PO14HP7o@=?;p"cuV[aUij5u71&+5'&3 +TG?;1^r5efTkgcD7SDJ_B+,M2]2p^j(SggHRU:Z:Ht6[7o]^Hl1..VNO=)b5&fC\8 +If^93:5E#9&@j8SEbRYB+o%uC'h`G`.2dli$`q@ej\qr0/po%eH=9V'BJY'8R2_Q-=/su(OK&Zg_E8$#f +Udq=m?bRlT2`J_TnOW"RDG0ck7,k,R7=TWBq8m(-?OnkTg`Q(3[YP`Hcss3A]E]HS +U:l7oP`:"K;>29a+-1TNd68/R/sgbQ;VL)pkh%q<>8]J!JPIW@#rT;"8NTt(Ng?E7 +7Hi=QkiNWDaDuB5)<:5GW#K<+p2Lp2+[4o4NWaG5&)F[QH(R"[WN)mu\esmqUQ9#$ +T2GUfiL\\orVmKP(]PV-i&(h$!6Y=_cgejO5)'KA*tD! +3n>*eD.gf\`ot-%FGDpYDtPk33ONR5FfIorU/\$@eK01NQt31!T>I0 +Z@%I[mrF!QGoe=L=Q"[+V-=XQ$7['JTH2$?!he6jXE07<4Q7-[Us5O^=Xi*1QeU,Z +2q1_bhRS#Ke*>aP-%#L3pCA^V),Xa*Y.JtKiUD$^Z(9HU].M($XX@`I-hfSqm<,#V +lb_eVojIL).74"]/YJ-dea:V^f=.hqDdCiC$JJk9d^HT\Xbb?T>122(*;2UOCArb` +WQtNAX0^/\ajh66'H0Ps8u,;__Z(#0It70@_HVPu.Z$GaMZ8(.rBgO2*/ibrW;Js\ +B"\nMs-Na%2Z098#=/9ao7I.s_t3^ird8O2WpO.r-_UBu:N$6Hf@4hts$?PRB7&Wj +4WtW_;Ld-jEI2Zfa)&rGRF>+XIn`P719F68=g$a`W4?ZF:obBO9Naikb\G.1R>1Y. +aG!Ikjo>7/IrW8r*t/i_"r@NF$A4U:a\BhT5bZ4/r\(L$nuV7aX!>gUO?&j+gYj*( +^APQmXKtMQ=HbqtS8r:1g)XDe>/A7f/UCi<`+q1$GP9I74%oV0/E=5:Qk0::cS/_` +'leMhnalhAOcB#1RaJ,"aBRH`TEX`1UFA3?W0k;127$d1Z_;nPR6u1TnNi]M2I2Nd +:aXX+8bDR!##LUV0>T?#&fGHtR&^lF=2[T[M.\G9@,r]_$I\.]Gu:EK8ZELVCINtR +0-n_i2V0T"BNm@J[;N"R*4L>cZrmN%@WC>uEke%Yd +j$7fMd00dBS<\9C9Y82m:92b/(&1i(;oRt]3lZ$;,&bSa\42/FEmS,.\g[lT.AgHm +6p9EAq.;;?r@)d^:W,290Rp_oDXLEFeL/eK50(D61*;DRTAC.Z\Bt8B7GG`0'J)cX4>)_.aNN$Ijlu@,9L@D0.Kp'-(V'6A +ndK8a;+$aY/%4[f2Q[P1COf>6,!'kHKeA>j^tRBI+jH_I%hgc2p](T4ap_95O0in7 ++94'DJV=oIY_*^7\Eq98f5lNJ/j?G_K^iQ*?BYA(H\Wbd3!6*#K^U[R!N/4O+UIu& +TbiH5@&mI!`\/YBhkX@^rc/%40B;B@3$<\P"G*F$6WK_OY/gbGCR!b,(?L%S%RpK6 +Mu\IO`\ZBFQsW=5.1DM)4,hi)HLWNcgYKnNW.#jaWlJbn%=;`;`)f;cr/B +=1\Sm:/%3_e,]%M8Cgj7\n\N:re%>28(TDblrpf&D=[FuqE`>(Ns)Or4Xq4P2cQWso;J)mXZ4pj_ +L]BOETN6W,,CmkrY>LX'2s33'"X&;6^Us'\:qAO"8nUm&SMm/Me<+7MLZrF,N6 +SRRXQ6Ag8r=+Zo+?G'D[j<(Hj5SO@,3d_fb;PQ_XaSRn@OrU7r4 +NVe,-PlDO)",3#V,up4'$'j"7pKh3Xb?NnDkLl@c2L@=^h["J_"[R&Jr'.D/^_s+H +R;G\]g6Huu2sR*!kbko1>a^UuF!'F$Zh3/DVgiZ9n\Kd4pLb_"1E5lHFO\Lg,+]g6 +Gbg7;D[X.iC=,a.`iKd><28Cbg6U:]Noc5>--YX:VH:dEMatj'>9+K"5298MiNe& +mb4!%aR5F^YDIhis.c?oT:Cl?r+s6:GJnuVBCsf"7"'?Lq=gZ.^YZd(r=%_G;;)DI +dA7%hcA@RGU"N\XK8V'4d>WU!BF-`2#+_%.0Ba8I=O7.FoWTECW^74tUPEU26e1D: +.b]LUgAP@S"Zsl&,beoQUFn^&6sW!8&3ZqH/Dq=IJa4V0;e#inA.lClEt +_5$!Wi5aah3'@J>98BQoS!Ij=Z/H"mm5b:A]O#F'$NPqoT6COURhT>D125g4j]]`]RKBqkL/ae=7,E][qD@EG,Y>m1$*Nkl5M,$ +LNdEm:E+5E1#0L`Ht';@e?&&]IQQMK=c.W&i9XdjR6AjE1QGD_G-W<6_tpd.NoCKe +8iens!Z`3cjhU`<1(jU>XX/rYGSEt5^[0G0>NTlV[c$:D2Y"OenI,ALr7*9b=tYFN +!(6g<'`[*@XXX_VDpmg!s'<(Pf`/A]Y8tiJK7GfYU$N$[)ms."=q";u=mEA*\X[o( +-!=XEpLf2,ra:3E5K58ue%X=W9`eZJ0)LM*8`_&0kB:lteL]_[RT_Vj+/l=#;BKqs +X$(k3QIjH+Gr"`HFJC]nCJ%UlP0;@s`U]h!MUW8aAsnea-r^8U3.0j&i +DXU33]0H/OBFOG2gNsh,n#(SQ%?9pC]Ongg&'kEl1r4kAs3aC54jE28+Qdu#I9MW^ +r:I-XSP)ht@T8UcF%bau1#9W8=q49KU0"-Fb#7H"8W;X3>#?JU9`f`c+s#aZ/:k>g +0dQes,g--)ZrM]+q;PrF)QmGhXttpg>^4.fJ)din[[#lYCPi(2rP-T:Ze[IOh6MCM +]RA3f3ME&HrspPJ#>.L4f8TDUS*DMk*Ko5@-9+#JER4?3VtS-EIXTH/%`eSEp:/Go +`.)c.d_F)JI<.%'S+BKaT8iX*2YneVQlka+^:TAJV&qkJL;;lJ27+<(U0/Af-UB]P +734*OB3"/%9>j7Xo+=4cnJ_%!'7-O8p1-72C-?mCKDoC#oC6GTA+fJPi(ptJC4d=1`0/a/h)khhqh\pAkCgnXXF +Qau9>V#H//T,Ic%I[^!qeH-@M"G$Ne]?(!'".>cg8Gi0]o".C)/mfFnaE5p;'9S(I +,,Ws^%Fdd)]4(l.A;:aM:ks'sLI[Ssi.&-XO:88`AngLr'!qHp%hnC,A`MOtI>+5^I]_..RXKOOI.,]`;H'5Nk1%+ +Td#$2!YGCL!(sG +=I^r35JcJm;i=1l+FdYr]/U?&(s;k$IfSP*_".4,l[0mKrhAPJUn_,#TiSIq;74YS'E33Ki$S*J@blGtReaT(=-,KToG4tHF.oE"9GrIFk5 +4rXJ6!7*HJm&+K4oDZ3"?(FF"VMumk2O/K'Y1V5Un(K>M\>`rt-:R+;$60)r[P(1& +ju/"qdepr#^XHJ%[HHLqISJLks.`PMkPIJ?q+Ht[%/:p!!"]0a?hp>Ds58C^s0:]R +"AjHZ$:sb1b'oF:@*ja)#0NuIM:3X@\?gQH/:X9f9)MJBJ!WNULB%^Jr<.]rdeP'W +YJe\8>lX]$1BUXF&-:<=/E6]`>'$mS*d2$F0aM] +/Cu/5-khLg+9Kt`'eJ^$"g:6=:h`<9D3e3MK:`pR$AJQ\e,SA4+S*SPpD/#/\IL"$ +"<$9=[`IU;"7LOG33sPdi[pGNm<'l0[3XP9+oXgcU'HGi4g?U(V-O[OcHtG&LYeZB +r,rjh]*H#0NNR(fcOHYiHdIE2dJ3ugX#oTs!DSlTX_Z>(jesUMQR:1[Z6.8>D%[;8s+UG*9T3)D`!$,'-DQ07[#uL3r\pt>"BFf7UT!": +]d]?#fY]=Pqfl5i[iLZ7'nuV2Rd",EAs&\5Pc0cPlVX0CGg. +k\'0*T51.8d)!6)M>lLH`_4"-0D@nBh@fcR=7e9h;,9&>4Pi>QV@G`m),X+TVC)0( +lIa.)M_;Q8*d'7[&(7(gAA$VbHZobo@5n&+%HnnSdFW"K5?s#9HS>d's/A])q=bW_ +YPeFLQ1,IfpA`oX]^IMY^3P8%pUeoSF=-gs@gOgG&Gu7Z7HZ$7*Sq^BV_`X7I;GB3 ++_XSi)VHGXa.s<"Zo)$I`fNL^5KTQm2652#b"SRnX&JOG._\g&K*Y:%=Xn!;>0.44 +I#.S)i&\Q+#K;0pr[q-8a".>ars[ET#(5>GBV.[XV--i1aWc95nIe6Ro[j)7*s+^8 ++4e.rQf6M.GRU#0!q>J2I9Mq<_ku?g7s#*N3fuo_FQ%KMcp$H!.M`0mUZMZ? +!WWX2a?Tb#=oM7nKE"paP>Y*PY:JcLrtG4/f!0b0rrL:BD!ZWdB;,?J>*`48eerok +"Me+-fm/+q_$@jJ-_W5tHH\_U)f9Dd/X&SQhO\GgV4/Knif=TA"3DF,^VgHN8--XHd7gs)$\D&clD>c!@1%S'&0UWT`SD!o6!h +@.P.F=1;q^-:QZYH#lJYe,%LI[aM`&./lm/Z@Dl][sNtLR;]Lp(emo!?DPsSeS9Y- +,uc9]&-9<@YEgWe3?#FRFP;d\XK2s^W0RDDj9O-2\!cjSNWJMtC0r]RUO[Q7Ocp&Fa384>OZAic3,rN\%XQ(]W"@brApU*_sMkW=3gIWd$dO +28?"X5R>qtr3\Y*29/qCnl&HeXs4&;#!W9=oUKU>(]2aUa5MuJO[t*sj#6(HaJRq.I +npkYHe%6TS;':7U8S0bj=Ue7j?L3p:ZKD#Mg!pRP]:7Osbh,d^FKd(Uk_>=C^!)Ce +Oj?cj?C>98^(\19NS`MZIqtMCp+GL^4!NN"B3Z<^4M-GiWBKgT\ja+(,H`bslr(b3 +XC69'B@JSHqIF];Li*hpC3`JfLFMCSPd;Q+1Dn)F[#u,&S;^JrdoIdBQ\5!q,*jTF +/PV6X9Nne_0kt]+"P!J4P*_GJW/-FTZ5D3aoUmt7_'iu@82p?cqs0OV\#pS*Y^0OJ +g<'>W*Q&UR`g&V51cm%+41FIRk)X]?Lh^U=l.4_AK7/I.qK)B?8.H/Omi5-<\39B( +l0'W&UOI80k+e+pT:_,Cf=@\!mS]AHfMk$+8s`XLsM+/s*OqF_g_7',_@0> +<:q'.d8-4[1%`U2 +]%i@q,Ze?85Qq#[ZK-e-`8A6Vr8C6+IidSK +qZ6DW(Kd<$:Aa)!Upk9$H!u3P`@:X("8&'Ed@R4kJO!2@,K=LoYSn-K`\atSa_8Vp +BnII:KVpF903hAPCR"'u72hP#V-lRl5b/Fa'eE>g>?T5Q<#$C\*0NWr^s*tVp2OU6 +9>-WB^KCVm&"9fFhBI=\0&C4dWj:QpFFb?gUt.#lNZ,t_=$Y>?AFS`/"Q3LnJ4ba8 +BAiQE>^1SfClaV_8Uuk,XBMl672Q +J@:uMX2)ncF\3^C$J30[;N27);6T(#oK8\9V`/W*-6>#O#hArc#S?;$K`+\8!8*l2 +!2Y*%f`;HZ#P/UN'\%4)$W6s0JV-C7'2o-.Yhem_&9.4jh>Aa-rsNT>QoZ<"B^kV_ +r4LTQiVH0oL)U'R#%<"NCoB..]ISK;]5U:/dbP!UAr"uE1KJo%[^$i"jYuX)*Smj2 +D8q5j5Pm1Fpn.:FMulc#>0'h(oW7H0?MfQslYSG&ReYNQ$TA%Io%kthhHiWXT=VdE +e'k5_l-nPrkg=5%XR,:O[Vi$o#\\$-WGa?q/c58o-iSn$eRZtR)oPo3NWA'gC(ub` +YC08ZBp;MCeE8L`:F:Rbp^d6DDuL)pkCONH+665q%mp>d=mE<9]WQ(c2qnk[r/'D^82$K5,fK.h!0eSr3eCelb6=2h&J5aamA3eMaanA3D[mE]eX +10sn+lmN3i/W7RJ=OqN)`clWDN(Lrp[PNB^hXE?7R3aVq!?;rccH_5Q1o` +q`INAm\Mg3^E:_N*p1<9R<_r@iRi3j40e/e6[4.,J11>WkeIo\33:]\s2HbGL1f%.pO +#G^,nLjZh,!PC:GhA`ABY`o(VIekENdf+N^q9@U6+7G6cn\O:eQ`0$Wq#>BUrsQ:9 +JH+f8ABhe=rTUW_`tp%'T94O$R.*0VrFr@jdQN/,ES+*BqS.QqHpci0G +L&Fg.mo61ef_ep]'BH@B%/\qYk(b=+mS"T5GR(Mc[_l7t*a5:-@0e,LRjI,KMMT+Cp.=T=0Ts$'!S`;]sLIY38A +r/X>KA0QO4"<7eFPN-'N@n6'@*\@>+YHZ"o!r:FPKXCi0]q=1!i3`[kD,m&^h +aT"bSRCgTTG]V]ueXIZq(N9*#XOcpf@3fD#Ej +4l.J;pL:`ZcXP-WR:LWW7&u$A?DLJUWVTso,!`a_!f6lUQ(Obl?fHQj"g@s%a:7V6 +Z5U&7Unme;BqVUE;g!5Ke!:G#^X%OZ4s$'/AVW[8LAVS".lS"<@/XDLFetbrQi=1k +'i\5o]37*QZ#.5W+,pd"3H?hQH/<.A!"oEdP&oCba"8b;?MQeW`EbQG%=XjlrPMq5O8%$#H4Ao@Ri)/f(A[DTOgt"(?gLQ!VrEotT#qK;TA'='[fF&b +_H[]!?(q1G-0shh=lB=jB;IgT.K89kV[V +M=>>b=m@qus.r"s#/L-o`;,E)9oa%_fpP3igZZjqE0(caJ<88_`)HjF +Jn@4AImNVNEi:b[b&iD"'WKD0LYmCMZuuM +>."tEO["A/Ic[;2q&b,a!:[&c?X(AVq]C:cs65$N0DmIWf_GNghrGK)+`W%ir3$AK +0+:+Eq9RL6B7[m]&b`tVhZ';G5O`'35']uF"J9W)OJBD,QT +*e4q1q]?"D[Zc%i`;dl$RuZhlJcGUn'O1I>r.g#K6UQU16\^.XrP5)"Dbs4R@+,m6 +B=GHhUE"K/RYttHf*&+>Hi;.)&R`$/)N@l(K8bZf#t+d]s&0R3@K-E;r(gAmk?(HO +qE2A-n,E_^j:Cpf8LLCr0@[!*n-$@X>]_#Hh< +WPoU?V$_CEmB;^MpD6AZ'P=-8s5STf[h.G1k$-tj(V6.@b?$DJ2hXI:M''h$NiM`BCLanqfZde&1CK+nXhroN32]C^[i, +JaR5=B<,aUH5QOrs +S]4d"7UM8ZWb8D(O'7[NBOtfJ8ljH!bfCNeba"U'2(Z,#KhB2]@YQlrg0YOmY=rdl +Z+D"R/Um4TNGPgJGBA8lkf.0Yh(-MAGXrQ_04Pl^B +B1R?t.0%L:lbVs;:[a-iom/%cj$#XIpU<6E^V#Zth0mCd&#S?MFtTUrI5q%X#8=h` +-g'<10)FQ1oCF1=J%;0@i@Or@mBh(,627sEJ$#(-k!12>q#1uf&%a#LRYgt6Cuh4# +Bn)R?J%>j_q-*F"lfk%s&Fm6q,QDj;NfW)Dd,'CgQpQ;TN]u0NriucWZCh-\!SXFV +cmRG>:Cj2GDEr]dKasA4aT)-Y0mMBt6O?lo:U9,SA^A'*-&;NY.:CZo)9pom!U_349CRYs.n/AT_NDJS9oB21r%(1NG[EmIJH/rcKUCtd( +`,iQ5OT-nI.:?.a/4rL4o0`?2+;]\1obA*FR3WaQ@K1,#nT6IV'I[JicPDT`/*26h +!Ts=kg)!Zb+p&O?%LeTq_uA96Nrm00s5OBhB.diuZ,mQ-pqP6-W:l#E&+Zp-'K0## +>AAh.256*l8S>^NMZ;NKl.5Mf$iIb]B'XL,s#$QeYF?rSEN8Akm1i-7WK0bhO3d#b +m(p$Ds&T>Rs%i!h:rU3FaA0XTgN>Y`b5-E7"pZt-s5@BLKlJV0SDJGM5LW#k;KN0##ZT%g;Y8 +jc?TYYMfTEoM1L_1QZ&>V)c2f&-8q!W]QR`,*2gbIHbonL[48sd_M_=\L+72M4 +V/CBp'7-;b?"h5Z4Wa.@YI2feCX@#Lg9_3p/ObYEW%K1B'qji>'Z`_W*1BMiF(S9j +!MaAC\Z:mpebgP;/:E8FaSKi&!cW\K`Gh` +$qjokG>1J"H$kN.[F:;L`Qdi^M?`CB_l^Ho*"$V.dpND^?B2HWYDr)5H-^O!,& +.2;g:\&o0hMiWm>cV:6Ag7@lQCs8B(@fC=n*#/FFT8W[ESqQPS9V/k`MJ$+D7plUa +nDjK;hd0FWp"@E9hA5YWXj"Q:5lYM`p>IjqG<=1KPN[4SZk'T$5+G)lhG=1+[EQt" +O8/SqJFUa@kd-a4+S1*<)Y*Q+Qr)5KpO7KAqJ\[qb?jhmIH]-qcp+['?QPqiGq8aV +0`(VOqYD'*l8IAJ-gBNYNVr32/=-=TjLIp_ps-(@3T8tiq#AAb_N=\&'$X^Jb[&M_ +]:=Lrsg^CE[SsKBF\G3[kKjW`XrTkNiO/Gq64-oO0%C&^rNP06Sp@a(4K,YHC +S!(@_&f]W2*]r+njZ'>HJ@^c2D`.,Qrmu,u>4t;_/!M%/"d3LNoY6^Go0q^N`W&:# +=51&sm*YWRSdupe%mn9#WZ:K[,0(@*s!u'!DV(tQ&G\!KIhi%`-P8WA +O,!_MoY7tj>_NL`k>VB#-fC,Cd4+j!s5QrVjaXt(X'6,9H5hq?ImF2C(]$?Y(*HaF +r/]t>?iJ[3e#DqiZ2a?"Y*ArPrW&qtrE`b/=8O-B">rJoD5Eo!9$gZlConG[f*OP) +`;gVPIB?Z1s,i:dgGWRAn*F764g8bkd..8`Wt(0-&>87LmA5S^.ZR6(O3_K7m+%Z^ +V[#uA)G]`[SiL2qi-(399E^m$otQpis/,0en:,q[HuQ+nbWdjl+:O6s#&steqhK+n +o^qGm!]FpsG&46nlR6Ub?!#XfVQ+eCM#b#3M:9#fC8E+>LQ?Q(Wc01RUCdUt+[cWSq]$EXeXWqjR7Dmei1/%r)%C:]J90CDU1 +k5N:@0;dYh(*?2IJ0@2;Nt93ZIrF\_kO:n/(WVr9hb*q'%CiYBo7$>p=j16HFMA<6 +IO"6Y:W81]q+[q]([.o7\:[s4*HieG9-= +X"T@n?BBd1.<9s>cY*:Es*OR6T4`Y$2Z\eE4l_kM#d%[h(#X,6\'mif;PtSWE#U9_%5B.:r+@Ta%-))]6t\\.ELCZOp0+9 +.Vhe=%EtFH1t_E9r(YIa%-W+Gp$LUf2K/#eBQrWm,/>L?Eb70gUr<;/s+<7O;?!JN +s%p+!jDS%+W&Ij[%#;p]W8NcC+3^pfi+g*Mja;Zo6=>o8Q5NI;= +OhZmXZa,%O"h%tnoKTMYac3H0Y1pkeGd44L50Ed(+TV):X5_`4$^tgaD0<_goqK`$ +ai6e)aM5Ja2V:=l5P\5^$?l%Wmcn#hB5lQ'Olb2RXZSSJqTrdCXZO^e:+M@L-u\"< +5-\11EhE'TXR=:rh:YpUU]:l+.KWB4&,H7H[-i!n*jn? +r%7>IZRZ?dHBR[0Pkg4o\WN=#L,L"8A]U6kCjC-s +;H12]hGCYZ9bk.ZPh,/2om,5=613DM3&3"EoW_V`ZeI!'AEJ&jJ4,O&8O:B@^"+%_&-^^gsq +I6Ede;mG_srMk`:*"_*c=O$(g%Jonb`k,MuoNX:^$^_ +Im&_b?`hJh)hAaj$Mf9bs&ANl;tA.IThQbT)C/X!<`o>d]a]2VM>WiK"[ItrYK]uG +jp*g>$T`_3*Sro''&`G/:N!3\Vo>29rWa>^;A]Jb>")Bu8-o4h)0g$.a/NO%<3O@G +m9AXc!#5O&7/dr*Hi@,Z/h-Zb#Lc!#5;3%jrWi64rtG2As4`g+"^N6qaoG@oN'r3! +7'%sUBo3&ol9@X!GV?6SH&rm?Yefg/:<(EBXE[Oo-jZ8dMhtI +rsdrL:RIdQNt2RE&b,COhHgW\Ih6-*&6&gir.Q9nL)N@!SZ&Ak\@thMTh2o%=jl&j +e^lG.\m@;ER5;D\L&cKt4aGekjR)@gf_b\.HOo>MY3Z"0fI;tsKB^&[HLI5-0YR2j +?j7GJKAHV1"9DC78Prbm0ib2*b3DqZ<3^#X'B/%/nEMr`XUT"tu6 +SGugZXnAg./jY)R8Ds5jH_5BQ=fqi7\DqPO4WEd$Y4$\p\FRHLWj='1O5-[^lmD[p +2t[1N6!si2IoHmh9>chIA:$.')F/Og>6++t6h0]i-! +2G6;ks!Dt&]1aEI?@i.G088@\n/lKb6GoOS"'0qTLL";sE=)Vcr*D3^0,bGBU;D.[ +!;o?F^Sra]O!HI[=h(qNpQBCM&eJIPA:Qmlcl&Q<6OZJHDY)8X?<"W>rs^ltc3!pA +W-c\;7]P05fCZ<>4_Wu=aWe)o7>(eeWt>8oOBb\M>D?cMTJ6p?3QUBen-:ELaAe].3n[Br8q\>jkPi@+Ir)H2> +RNc#XBYUP:J@T*\Xa[IUDR[<*]=jBgWZl_'[YJThnRb[;qi3u(Sr'&u=U/N`6Nmj" +N3a>sq+FlTdV-VJGg559c-fPn@0+pEF-S,>V0B$K+$%Pkrd)_@Rn>n*?2XHk(]Wf2 +`W,a-aepeVrm1DgY?rZ=q\JkqL%;n50#JIoc2%I7&l1+p>f])3`7o(Rt[*q=rF +#lAoH'DF]ms,$5@_#Mi9ggBUCO +lCK3@s)25Ys1*dZ+.=XmD7"+2;`#NJP72CS5%`0\n@KO,r+CMKrt,(X)Dpq25Q!f= +a5PpUir]-%r!+e4YZ2Tc!;S-kSX]-Cat`h#rgH.FX;VT"N.l%5j,k5k\3N@4!eC=' +![.Lfrbps!KO"TU&"5IA"4Laf@/)f?1^nlG\fAC.ro]K[7>%&u4Xr:Pe"C17PtVb% +TQ@_[?9qs"^jdIkFoeK^/3j84pSKYE#n%:Hr#Ig94KU)i74C,F#RfkYK1lUQ^&\oI +rog/F"t_>*;Cm,XP*(htJGF+@oAR<+/,ueTJ6QtMs76W:ru;"..PQ#Hb^U11!_*q: +5_9Uk(ug:f*6eFNN0fLp;b::V`R"2HZMaj_s6'IBs(k_0=FbWF-Wq=R+.eRt:W\Vj +8WV+sgpgI7b=_a#CJC]c>/,-cj8L"/5nBSaK*=A`9)&,*#"#7"!'Pce=rMk^-Hk'] +bn$?.\aP9EH]I=X=OTP7?WtlFpb2+Yi"'!qG9&a2"TGSEZ9odcQNJ"hbgZ1[M&B$* +6CAi[-\?^Bg7\Wl[Y$L5gpKr[3,DJ<$ZQdp=b$Y$n?JBLKu:Q:SqBIhKk!=k5I&08 +_]LWBHmODJ6\U?kYNFkF"5Et#[oj\;-ioi7.=8S@!'_)3`+,N06:ccZ548>V!WU%D +DD9*(8!okZe*3d_4Co:0M:Gl=7c"c1Re^YSF%o]Lf`ke;!h3Hub/<*[,iC>rj9P$i($<9_=,Bu%?#mSl$K/,>o6"dX +QC`%VW0O$=ni/*cY)7m6ReDSmrVVKpG:LNk'HnQYQAEWkjNc6EmKg$P +TU@W5:J+Z)pW9mKUtZ;eP$5Fm;fP',=er5Mdao<01mOG48h=huNqQppH]b4C^SbYH +YA'?4@>D%(S$/+lW?XjjGA&=73d6U_pC\6,T(C?AK=';7CHd`d/qXPC&5fq/#:KT.Kg*ZXpn$U$VCBAUf#Y!&%MlV-s:t +.cB^"Mf4nH$%QL\9>t8QE8^%ppAr^W4LaLSVfs;Tg[XrX]0R6?]^G@d"..=M"3aW$ +J*FbF(Q0r\;"U[Kp4!5uE:/UtHu_I8X)n.k5$)bBY;j:'IeaH:CbMO+Br]kMgIEA*68A9OSqW6Jks*794fc-[3 +?4m,'H2`:U_u7Ofb^Y#-qUbI?5.S#\K2Y3W*\D^+?kMo.p0?iOVKa8u^G!gMr""*U +:]d;H`&]G3\\cqpANa?LT]jKG$Yf1e_L$^ukq42CbI;usK3!#OjATe/Ma +/(+:Z@=.YuJ3W&.L1s`:m:rXq76uMA2;PED5jZN-ZK1h2"o]L'7EmQffNYsEGSQmu +p;bn0J)V],G*'qIOZLuiRo4&Kphs1?!WH3jDG^_eJ-^c,84Wr4HjYB`$g%r)+TM=*BJ*F5_MuP0 +ag_o"^Gj.#mh6-%o7tc2iJ[d35;jpW*YO+fL="b+?id='P/EL3UXM!Vs,$Qo;bK!W +Xu>(GhK%51[X51G\G'+gK?"^7!DiiJ"[Eef#do:+]?0^/9r>6URdNCX4ECoO)*>O* +\U@145Hjl@s#2j+T_5?;J63DfbUl*qTHBY$&!*Hq&H1k=7'\BW6!YVJ\&Rg:E8&^L +AuAYg"@V!4Rk>i'>C+Z:XV/:fl`rk>a3+"GckZPtf/$B^^jj%TLDLFo#-fK3CkqM@ +qEJjFgZRPSn:(oI$Pj:Lnq$fPrdk+;+N*HAff\@3=..T5[Wou4I]>OrZhkU_Ru[Tl +5JN9&aY;F_"rdZ^+3DaRRcm3sgLTDLm!Om(C(Q0Ii=rhm,%=,1Hqirt$HV.^5=3.m +%e-s^AJU1`Iptaf@g/TtEh*9D1h10ZE>-Y35M_q(Pj]"==27jV] +]9Y>/Q8PHA%;CjLXHGEj8K)I!emD2[fC!V.m'Ka\!pgk@MgO_^HCDu-X^l2Pg"o/@ +.EGgAfr$A2Frq!5m0e,C(MV\?C#0nKg(pk:-0e($7T-++gMkci1!Oj)4U7j$V,L^k +Cm[DUill%':r(BWP=k@>s,jJQ(u&$KuF]:T#qNT?/h/H+o38q-bAfrpu" +qt('_B.oO"q'u',2sH.6n'Wj(niu=YA-&%S,[Yr4o[9dO(LJWidu-ZP4F[@0rlkiJ +/j9g"qkod`qr[!gQX2\&Q>>A#?[`I_b.Q#d?$mMRV>-PCiNnCo5)8dTo-Jm4%R*bl +5=KBIU(Ml-qso*(bO9M&r0P.NHFE4@Xil\(\KRN*kK,%Ig=[/C:,`I-@.lRCT2%p7A6N/[dkLOC=i!h7Ss0\ +D\C)AdY(?KCR=e-j8aZLp^-NCNJ-T;6?V>IH@l(fTW)V\(!E\O=HldH=j6s>W9>)tR/C1aO5R3P_%X$Pi[mE=iga*LOrC+: +A%88G&a8fQ+9MP*!rjk.6?lZK67eU("T5a/Kh0.-@9C\SDAraQ +=Ps?q:[(.UT/>>Q_JnWAh*"7bV`.QTs'&O80`P3Tb\M"&!rS*!EIg#+gn8R#!g`Z$ +T'9DVpnW'YV]H]AJeB2tBT5ELK)SrJV_)sO,TrFZm'7"i(88o$o%562Y*N($0l\2]lj +-'JBs"o\m=$!o%*#t-be&Bber3dc%=LMCs1b\i:A?G&*]FA.-_\8AD25km:lR5k5! +HE7O=Q#3sLX:CiHagX9a)#d9sf]>SD42%:Lk;W2InDE:kp]A5QcZt'1D_aWLk@43: +Pt\"eSns[LUXeoEHATUm2TL-7PsM;f*SYQ@P>6#<_A::K/7C/l)+p[XlXHmM$0BGu2/-=%IDN.']F&DF'$#&8"t-QDZ^:rdtH9j7_3f'C\LSkp.[N(VgRUT0ue#OGBQ5*Rha5/8=([' +2Wc<`Q6l:.M4Ri4SA0XSiQ^Gu\srWl&!d+Q8b.?n9E+mJe3BP&If2>+i`h@h!l$uh +2m!cFi8&Fd*=*YmG`@+@Hl6+XIC\Feq-l1HkLH;F?7QeBM5`Zb3?[qKFeq$lVc +f-qMC/&1]`q@l3n`L"Z4$"6d-#4sD@d[S`Zn1!h2/q:sSr&lmPiSmA_90_?@qNu?g +7uYW`?sUs*'`Nr'&!oKp!;sUB+StqOaWKCOrVeqZG;!uco+M)TcguY^2"+cmj8QrK +m%9kXrV/5:_qQU.S0Yjq[,cqkdO,jRc:jC>*haS!Pg(M41&lkI-LfpQAiZD%LRePF +2k-0t`[gQI<`$B=d4Ja`:7A2@Y3cFfU?d5u>/&T2$lHr\ZNU&qZqC38'bL?L]N#O? +Jc)1g?lo<2ObQ!r_tDL+]'1gELFV4sr`9RA$c*^AXNr=Bnio?L7e^l4oh9nN6d;ha +)4#.YOFL39*bAYI-%f$C/2K#KRCJqmc2G!H7m$_Wno'5sf-p>1&H"ZqM+93ddRd_Z +WC&,C/!+T',-1kDm4'iF5I"$Ce,s\K>/J`73#A_"j3SO=],^)7;+MU8Lo!q/; +[hohK8PO71 ++I*SV'AsD[58G&g@QaDnpBH``_`n:8^sR4_(jOR8YYU)*j'\V#?Q0WjQu#I&SRk)# +!kIO38%0&jF:2@q&]j!rTV;=5.F.j:$h-fo$N,+YAU):WeLKhJ&a4!ZZ7U>70+Rq7Ap588*gjSK50q',F3W>S]2^KR1^',K4Mq]*pN_ +QL%W=&#j`me[pRDUaa3iK0;:T9KTEG+WWA&NnR#om%eb3C!B%L"Ue.E)t2REl.?[8 +hA9'(H+GG2"kBBB+X-F*Hod^':CS3j&Nq#%-)iQceaDm)m7h`jkj@/sDVgt&,_:LG +K.WnX%aq:rf9h$5Qt?;N]4cHg"9(4qKO;J=qVhtC+'oQW +kDS,&5KEWPB=U6VRe)T$ZN'WIp.t*bIs`X0rL4[CSYX#JIJIr"j\IIn534$"c476=f*ajbL/ +X55EXFLSkX0+&T8;3NP>Oelm"[%h*G=[`$t!roA03agjp9uSl2dm41,?M\Ob +>o-4r501N`dhhcik#LXF5S`m'poh`$GkEgC[oVKf8aHdl^+`@^Y5A[d#U6JWX'0>g +8:^_E/YKn/C]Ra4!$UdrDX*)_=R:>4kN-)[!WW1Y"^K=PXVQHabeS_LU;28Q"Q1'h +/#IW<:]bI@!%7lf1ApZ.i4fiu^Q9*M*##GKLc2,-$#j@YlGhqS;[7)V?ELtbXmqF2 +QI?diG9?4^pK6&G2^f_,Y(S/GeT[$&n%E%IpCYB8LV%Ide9UZ%p)=fPNmBT^+8CmK +?n;6f*O;jgEhMOP7FTIccbu%XQN2I4D95au1f>7mcp7kZ8I>g870SB.ieUYKogtoB +H-BV-n]F1+"AXfGYQ1jq"PVT0s53Xu\(gZXJ%#g%584gM_W!s_S-8i3KrN!*re#b$ +d23u)T)E(MP[,n+#N+ZrLUDR=qg%b%C^^;-L`eq"C^9_k/In)sM/N35H3IDgY:tH3 +$UC;>qVFI@aQ&U#P&[3V`k;71L7[>QSmY5K(:lQB&#_8@U<0Z(Ip\C%8 +URGKK#1?`el&^2f%H_LTX_;Ls8N5l<6H:(A,::h^h2XM_St'DB^TJho+@l(*W.>sI +,)(>F`;E):fM%?pBlPpdAZ#*h8i+KH!StJsH<4:T,MTeNTQtkgnp^aU!(qcB;Qbp7 +WnoaJg,n7ARLGEU`i1cmXI0)!8hUkK@geeg^W)nq#<5]nXn1)G:5ubs"D)bG4Fs?p`Wf]IZ')B!::IeV?!%a +ik%%cE4egcl[:+-"?q;)/L2&$?UmS9\im7ZcPM4LT\o6%g>rk/`7Za_q_?qfQTehb +qLmf,QMc5ds$&b\!VO=+%flV,0Abc>p6spgQOs(EH&h\2:U:#_&`umW@si;5j.fhH +n(c!/jV[s8,P6%m[)\I]!<7BnBfD$ued(G\rNssSGq-[lpc6/@iLbl@,u%XX!DRf; +.qskn8ln*NXc;,e7VQe=G.5ju*=PXYbGa@Q<'6L:M"r,Ka\l_RT="aq5No6knlChH +;&ORhkL+plB"X0aQ19nfJJnE%)^g`Y(D\((9'[[XQ)k*:[*%lN;dTm%Mk#8ao5qG)9emb6ke^#!bliV)": +[GSXFEb_T8j?h>#RJMoB)U/56#>GC$"76-df>.FU@414;TiK.U6-%EJF`uUu2.IJm +cjYnU'4%9^A>1du6IRIf?FerBSfS^I6M5,Mk4=EjUjN150,$5EfaKYmO +*FJmlC]Se\&!R&P4c`_FjVbHgK:7jSqYrn+Itq;0nFua3i;gGe3?FB)0`MH1(W`c<^YOV&0/T+#3WiH[@2J^rfu7Xj@^^G8g:LLY +8rcLke/KY[P8)i6CmI+UkY-ZcKZ9_/BRnAh(Elg9+=h +f:JXUB1fdGZA5f#-69@.4([;u]5-.AhX.cT8N-fDVh1]`^"\S%G9"$W3b[53ePhVg +PC"nC8IHnWjX4mHI*t7Ai.2LN:=@GQ[57\se,a?1em't@gcqal4eUL`d]mG`D9!cP +s(50;$#MMQ&n",8D:H(:e@'SV;`#(8SaN=KB:*bXgk-:"'2,FUX6`@.fe4GBIsM1O@86DM#343D8.bK7M"&`5fK!V8\s]HTTRi^b'gpYC#Pbd; +Pd`JJ5]*XqY#`->NN)53HeiO2!Ur=H'8?+BBTFuaOokY>c:[$pYcfAiW69l6$Lj[E +2pXm]JGHP1L:Ki"4@+8lLEQKss(b+,28e9C:t>^q+s1bipm38-Jr#7*1iOT8]FZRInt:?L)! +h.e\,^U$Ssl4uGr&kBDJq!T?^SLWmSr0Ium(LGPl6L#-se77L-bQ@Q$a6Kp[3$p-l +SqL4=4TC7)!92uR$b3YFs)%bi2u2nO?d&Ionk'4Mad3!ZTaZ,jblnKB*"pI($\A?$ +UZ.PYn_m%oPQ7Wi1fnt?6IkVFlBBE!VcN)$=`"9X\8AADp;8Ii"Eti'!CsZu;!Y56 +c(LO92P45j3'Pr5Co2)[Wqt)0Hn56Kenfk_bl"U4hobuL'drL#H-*KKDk8I[a;kmS2_>%/YmC+\5X`)R5qjB,J$YDS[3OgPSHfG]pB3n@ +T\('DJ?E/>*j,n_\,,qgH!s4qZiE,%hs$9US-]+K*^57,NK-Y:mE/9Sfb"-E3ghQ? +o]C<84M_)l"OLhu!^oL&3_s0=MTc9u8WU#k[DCIs:&obe-ERjEUE3l3KiZ=\mUANe +d1'*aK'ZJ'KCD*uh_Aai*HGsXl.L?;B@ktrSTb=Jh\Fsr7st2NYoU<.GMM38Hd[&m +L:qig:*+Q)#$eOB;j_Q]dN3D^Msd2)onlZh;4gA*kMl1sfbk#Xl_"Yjb=9_QLCCf$ +g)ZqoHZJMcL>0-G\"L'>_l]_s:==rM1nrkkC=tK0\o'&u4IN"hU`&bkM#XB=)0[03 +4gDhJof2?6>:;_HL]$mZbC9FI)>O6$(WcXT1A/glkSISR;YljsaS@g&?d8U+;[?66j7mLp2^H +JcG@85@bDIHO99C=Em8!&n(_%ojbi"O8238LO8ha!TsD(?fV1c4T@@dZiB]4qEI,4 +q2:$l5J4K'rS,KO4AL?NU\rmSq[S[/!jf=a/IAc +0Ao0PJ":c4p&P:6N*M>Hi/dkK456]Nc/dHQntsFtjsI;4H<2msdE9gZYVH/@)GraI +!WFHmo(aC&TCp]mq!1G&oqRYh!rV#61OjuU1SK-AA>jr$R"p1:?TmJL!+k!WN)Q0g +80IiN/OBaC!BgeiY0HHSm`LaNcq`-#;)E.Ol;fZl*ai-aEI2g>#WOtJd"Pokk`.a5QXO^V5]QX2d4TFX!CXR +lA\b1'`]!\ouJ=p3EQ7q!Zk"k![-T.#gf7i"$2QI0=D]C&;K_6"d`0GLU&*P"o(I[ +:W?qE\]Y)d=acmI\s>-LLqLF_I/*;('<&hnTTd6q.5^"Sh_) +"9oUed*:!^1Y#d4c44H5#E"=,7@?ZV0<*VgADFHR+Dl@H!pB"Ylo>a5I:%*jGETT, +l.^VI[Hk!VFf1u>D@Mici$hlR;^T/9@37#c'bC*cfmkmuX8o(!Z@9PJ"7cd'4IiLB +2=.(!m,b"nU62%\_tmZLcJ=lf,H]#g7bUDD:/XhYb3;GWo?NkImoWut7&"M/T/o3\ +4uatornk'%HBQqVq9c>*TA3A&q>\WNbCBJ)]O$'WNA@<=(mt?% +kC`UqV4_b`/\-,"[(uU_mo(C=?iH.-q0UUa+0mTr;2kN=oC>B[&0Up,;#Pa>]n?4a +MM8NRH+h,/r&+qRm6XeTnS@&]crM6Yrrna=,UOd*qH.bQr!E@tjq#<.9,I@_s8Vi: +]O*nDqNiSW3T#nd/-#*\fitA3cQ=`fAu5:h>k,L+-$N+MhE"\\C8ge"5l!m\HmqLP +LslT"0SXID_:*9jF>DR]SM:Q:Au+0e%n(qs*,Xsrt/TQO<6=C +OobU[mGap9*$rJ:?cPM&=4)$N+*W01gS_GagHnq!QXVCI8sQMi+B9?u]0LrI!<92! +2^p`]%POlOO&,P&"h6KC!;MU,!5fnq5OdR*"RP;'TYQ%)d/n?)Q30$&3Ztkrs#-G; +s31L;$fJm/53Qb\k:N@b!rr;qmuFV+rrWj.1r^p$cNbWmUQ(qF"MjpW*oMUr7\D;^ +C)E39?i`n\_9/]qF\K$i!fkN,K+*1YjOL5,i.f7 +SW7Nu2bn[tO*%h&oLO`/khrn)#BlWZm('gEkX;^e4]/p-V,'UKj$?-q-le1:ZS?_` +;)+<:*_9>.it_&Bg`I=+LP2&qDpZPMH5M8Eb(.%4hme/,[<#HTA#P$bopijqZmp?7t1LIkYq;Im'^H0:1]8%7gH]7[e +:G[Ik?0:T^@"%`(qi@p7^Nc%CeSDOq?`3nfc15f0U(CtRBNY%7!5s*rqqicR/q^A9 +]FXpQ&D1XWHhc4VqeT4SHpC3(*k\e+JUrtK$NFeE!:0f:%K+A,5NJR.^U;h&"oFX8 +,UcUpQ[U?qp^sc6k^D=>n*pAP3r@Bm=hf6RM9bo\<;[R,dbhfnr9CQ8n#fS1?Da1!$gW,lOisdA]"#4[;`!h!,IOE0MA_G=%9+TLMBBaMOHN8FV['\8M +RORq&j`RsWTSD1[VY\`,7^AbNWbSEsg6q(UY$K:rCB]0>-t!`&Rt'M4G'VeA"o3Z# +=#V%6*")f=)aRE$e^H2eC+9Tj@U +aDVW7FTSIqlVMY=.^hd^13G=1*Aec5PqCLg(.r4\Gt1d^W3a#CeC?ldZr,=i<)#GK +gHik][$\F,Aoq*!#ES0R$a;,.&dH8n]EI0>kBDK'$mM<0lmIHIs$ZF5Vb[a;+RsQ3 +61-?;#`QRP-+%e$YQ["N>JiZ<*Z9O+7Y#S)#Z6[8nVL5>^(t,U)[J`/"6n'$n84(i +/DMgd)m&ks@$>*d\H'p??4bflMXCanGk68Jd>a4g[Pu.tL,lsRf,%lmURXT.*DW7` +Weg\'kl,S];K@45/]aeL,AEDB7gHQa6ThT'kn3o2:9ZE)>TO]`9Sldt7;LA8__]^c +1GY[LKFS80RAi/$NJ9m;Es%do,m18D860b?aS\[uie7F'CqjYp%rfm/`i)oJc?a]F9Soc1k-oIbqc5"pq/`) +hl+N`kNJ19Xb)?(YNT@Eg?G&dEcY7l+Wm::fN215Zde2;M +RXo!<-7[8.`0l]p0YkqZR9:mp@kC:H*5MIom_6R9n^ReqaqBGgCQ^N=c:]KDhb*T;t +J,KQh/:_Ec:dKa<:bhuI$g>:0Dt>(4s$+R(M]Zs0r-ofHILZ;hr&=GDDuKk.s$CrV +o>_q6;J^ +[@b$lU/*'P@km+bN0B:+P6"_&a7MA@DC`nB@T\TQZogJQ +VR.,<\=oe\g\_;$!rqlX^BFSak0Hd$kCDrMH;+==q*,a5`5;Guq.fNjG?+_>Tnk,4 +5A1UQ[!\4+LZ*/OZH_3!%"F*3Rc^pDVK,L''=kf]$j86A"^hAeXlZS6]ERNQY?1ee +59AF5q2\SXYQ\k)5Q$"VFi"4[6!ffn8cT9LquleIm+k!`6KH5Vrt0.2(lsVCK0sc* +b^]DB_F"X6[dkLJaFBK!*%^#4.,P3>C/ZPX)ZDh2X9/FhI\n@*ncUtgJ&28ii]h/Y +oqM?(YOKr+o4E'VKcGSV$nV^fN/J3@8jP`;_o-V5>_[A, +UAcRX5!aYFdZ@jF:b\6pch%MkPCiPQ1QT"XefH +jKs(\s+drJ?ier&bDEL;:S18e,I9LY8Y*?q[['_$r(+^$??91>WVJ#lUSg.T5rY?.=@ +<"08e$teh26hQ(a%I%tODf6*S+!Q"bME3Q'-^3`.7>UP"ENk@:F`MHrGPcUrq_A!N +;$;Ber+%*9j/K%HDfr@h-J;Ui8C\>*:`E'W`1g7.#_XC%JRF`r=:P1PGq*ZK\]HIY +n?;6Xq260)1DsO.,G1%N/K+S&e.DUGSs/hiNc7XiL`DN\H"T4B9s3ZQL/J!ht ++MpA*5>(pG]^BqYjF:H:\Q)K@;"*g.CkqMpm+Zq]@,'r@Xu!aQ+3NqtHt_,1,sC'j0V3,$*GcY91^p4C +"Ul_gfr0-T//\(J#s]T>Q_c:Q_qNqal5dd"D.!BXk>+$N>e,8^ZYdpjIE]9L_RFG5 +;DOL_c_ROu3R6osk`0r.R8rt@8(]JdAAbuhk(,uA,P=Xb!/'h +r8YC]h`Gu=jWjYo3`LLa>j9MDZM"4F\QJ*,KK:*I%L-)75uud]$G"R.2n,',iSUD# +o6e,1aK;T+3;1tROK5G*.P"/l.S?_AMX.97)JdGH%00VVJ+b5h7="m%j[B*O`9Y1- +/YL)iU;MUt^Qn_IrGddeS45A:!$db;9VFj5>Re[`?P=Br9Hn2F(f +UGfcnC4[AatSX +2jtQJs2Usa8U5em7Cp6SE1duf5;_.L1t7c72J+]R_#IrX1(pVg3;\>A2nm`e[6)<- +Ri'K@61Z)_I%-=C)GUnng3P"L"69gqX%27^9+]Xs<'C^-E(rkD/Vf]9f#f`,<0p4(es#+RQ'/tdN- +c;tP0IajET%oL*'q/PakSY&XOB3j%*HUY[ar?a7_$k*SZsjhZl8i&jh#B)X3+s%7F!D5W +#c8Dggj1-Z+%dK]L@5F88,GT?6"q:_-_%e__\J;(:F3qM.shGfe*2_:I76C5;]PJG`o09q]mb +7\i^6@0)?a!au#JI:s?d?lnoN&X4a%ZuJDdTB&3Lrq[8 +0CMV7&FckCaOC4fLPHd;L)WX;RtlcfL'F+P1QD@HodL%,=q'Qqs(uoc=FepK[+Rn3 +4KhE9nF.HiI.I#dK_b,05hM-q+Sf(aH7P2-<>b^+IJ'7-(I1t#\4eSU%K(GI.NC(O +fcf$brc#@9T7Js!J2%!B>KLaLe8>F_UZ(HZ4QC$bMRKS%="Ib\KGOXgq2#%X^8@BX +q60)GBY%kh"kfq.8fID^RC=8[DneKGo<&qp9^h\11>O4pd=_$`f5D%" +1Xt;G>Q3jpjSsM)k_kdc^H>j)(4_E_?,M/$kKfFl58T+br0[M/@Lih;]WM+HoK?Z` +"/,Vq2SqWfs4?hVE;8a'F7=tM7AW1l-#d^pEr^jg2O6tdXHs&8:58V28. +fI/&+!$DZjfHJJj%(Y\Y4nX@"&T:C[!^Ja=A,)+pX0\tO5puU,.c>fK71+0*Udl>V +pl1G$&)_Bj@UemIg?H1LW'14\jkUC,cFQnM>:i`g9aMTJ\(F2m!n[R] +*D@#H^m.pX:&a1*@&OFZU32GIr"loo6>o/Z(4Z_m6EY3W]pnIe#XI;05@FR`#l^cq +cHDkjKCGuMBXY@0N%;peWYb!THrFjiBbcOM-nB=P9^ +mS#hM:lj,L"5ls%3M;33HS!1\7@6GN<]7l#nu'1&4u[7hg49"6]&MN+IgEE$,db8t +>W_:Z4lfmOodh2,/M9'G(MoK=!.W5mgmAH,O82"$![U*%0&&RQD#r`&R_]%Zm/d(t +6S4M>XWVihNG"a$#$OLmRg5/Mku_dP>a#Fo`<0f/J_uK*J\fgogsug_^Xld\^`Q:1 +n@PJ^Y;?lR`krP5(fDhi$1RCZNp.L>Rm1N/k'sn4-18O@1I/uEOH6$Igl@)peOq,Kn"$PL/8nh02pbpDX?b/ +!#J2%lG*J:AO>>oSGrup2ctae*hH9Q_9IJ:gV8A4;utM2Fpb1UN<8'GX_-nlI(\VU +Xa1r*G.e0e&/niN@W1h0=Su0<[d]ocmA=)eAO0U!VJ#XfD>R``l,LfiHP<1[qFP>+ +4!#Ktp@QuI"TR^Lcd!7i4\'p2Q??eCE;Pm\j@%PpGg4BE_pVZI_IVj-6jGgjVRu#( +'XL_D;j/6N'-6!4'-g58fnT$=ols(e)Adu/5SZ!;.3!JrN8'9;JR?V"8aCqAHdS(a +GhPZZ#ubIMoXQf*G*%XUX\D>eB.Hd&$W!ru+'=jnIT5spEW3#:3#?IEJ+!lua*4lh +niNqfIZXL=O0F*,3;jXh;L\42q0tS0reD;IrqJhM(GD&&oh$\&*t<;,kC5G"J/1O0 +o8/qS#n%k6!YH4;I4g7=$L/`'JH$EM`2t'^p]UDt$N.r"Iut*90Ba)+<>>Ps5NmIH +:q2kKiI1M?la)qG^\oe`Q]IF0c2GUG\msXPa*0b[G27fB#AtD2?0`k674J1A>1CO` +gUR<6LKS"@#q\UIUenanU[2H5[*g)mf2S)9<0T`3(C?!&r$">nJ<3hX()4kZ,urC! +q?gtY%C?Q[jSlHqUW^U6r1n^$t#nSg^M-WB3Bj2lc5h8q.=9.?1TaE3` ++7gS`I0`iL*>Q7g`;d&=ime>Cn2GJDJc$)VFY.]81NaB1jGqs"p)6?uQKmI][TYq/ +6[;`XV7VSqeHOH>FOUof5T>:DNEq6X&G]IJ>+S*6c8E3jI,I5N+'V!]s#q0"J:N`; +fF`f3O5]P,$6:TKaJRtL<'2,)^D)>KK&X'SmAqGgYLg'F=tNsR@:?.C88,&q4tj,? +/?OK&1Em4R>F8N<(U,\%0aJ.pCqUFRE%*%n'^S>b-67aIi3NF)![>hgi<#,Y#B'r< +kCR3V,T$P:*=-Kh-!XifT8*W(VKq2>5%lF`"&5Z'PE>9\foml97$9POB-H3Rfcr6& +E0L[OL?Sd@HH+:PpmM(%B!7,\YVTlm4XU78@FHQcb' +&g]elrE"ip0rUL8/L>-_cC4LO3_2d>UmklKch*AA)B_2/,sj5/pZq'93BBDU+!g@9_N_VMRSo.i)kr:Z'>aeYDcRG4CeY%VAe_I5-"@b(,d6=+: +R2W)tb1'2[b!h3cUE!?LhJCg +?C[7`bZ:?agK**>JjQS4\2_uD;(JY5f^WZHZqQX+#6'BT?#d6itMg0a9&/hU'bmSXfT1/P1EZ0m:r;p>1i[pY.725=$SYZ +qlVs`:sj3\g/7uWiW!#Q^0gu?oC#1"^\R$,]/&E2pXpfhCIQ;i"o^U64I^q>m``_# +FUJYr^T5'gPJ=62Up[54[BcU)U:=?4e5V"K>ku.GJ>/I/":)d1TQUOr#`>V^i](%K +,3W),K*Qt_KFrIZr2omP-ci%q\)UBL]TCK$2)&_IqRk_'>jOdoB-@=XHm9dTLbDQs3$3#EDp +'T>NaKd5AD'O>8hMkM"#fEh.Z!P^XOu\t= +NPlY6r1OB@)e'9])1[-er%pqbI3)RqMk5%;j#2"XIqi4M2f2fg[iJ^qXKK&DYPC*- +"5![8XG7=iLCUEiJH)9p/)J5kfA_C!DUpp1AHWAX)qU(0Bu`\C+I)nIh.&Ip"TnI[ +%_%FWZqh)4mFqX9JGn0+pA*,0fo]=^FKQWJ5*9[ar^.!Rm;9uk)PI:i#HC/JWh?9D +n#H"?+SJmP'B"r/1&h>n)8Ue_ELKRmMH0BsC,:9\K:,1W3m\s0g,V1EKf&o28].)D +KC)$oekC"'?_S\YDb`\>,D#]AVqU=6oJko6?j(AF'TI*"o,L/iJB/gu6L4pXr5/h$ +?LTP"s)bi\^E6Uo?5K7Y>q,R +/:Vn3qW(NDZSr.r^/=\%9nIS,'mssJn)JcIs4;[X^O:nF(@\M';1mXEYE&qs +.(I(5Jh%d+[g[a57R1q`(3"j%oc'tc?=Eg37J^qo0)O>>V*DZj]'nO`pU-dnSm:-% +pJgD):&]p=9TPHK1NUqX1'?F,iMsL-m;7fZ7VZ4G=bWqJ,Wm4-mg&Wo=aL2K+?l\/ +pb1S@K*>52bJ40_CME/F1hAI9gWDBrnkf^d6LAd:s1"3L5LBPLq&eOUq*,Y-<`dTE +qJ*=-WW?0ig*)C-.M;[iH2a;E:r?Yr636j3Rjd)%'M/5jeR\pNj!Wj$lp.?D5i)RB +8m88mBn.c8])Y\6:^d3A4+`P%i,c_,--MCMNrp)CFq+o5sd]g"+>0[mF?kt:YCLS%^ +(4]XtF'PNt8g@SkQC!4j(]ae4!1T%##7fc,^`?jo!d&VtNJ@$/R3428@-Cc1^_T1? +'?9E8E!MT^iElM&06\\\c1UYc3L!q7i_4keOFJrP"6QM0<.aqq?8i%PE`%u0C>;:< +>,W#H2;CQOr'*9nZS6.q$WpPPY,r6]^j>e>7Kf;%LI5GMCPTg.LOob4r6#=n9+0%a +H,oiaD.^pQ2//GJd#X6%7rrFL!7V8u%^;d!<+)HrsNg\n.hDGp(cC,sWqb#K]*hX7a**^]FS?C67=U7CYT/qM;a-jK44@.Y.PU +87c_PVSBTh+Pi8&BGh"5)/]Q=:JWBN-C(7gERj-K@ViQ4i.$*Fs0:."\1-dJK0R6= +XMb/bJ6KbOWUP1E`%W:<0-g`#K7NlmBW!!1M^iWd["5$Q^MA+)_Pp(JUtD/n!PIA@)Sr]`9O +rM^+]s.4:Qqp-jUT3,Qp>nZ#/>^?([``lNu:G+/XXf.;_lIOR55T>JQl\E`?f1^tN +H<(Ooo1#sWJ^ONULrNWh(,/QJgFJdg$"f@uff3n.b+/o]An13'"F%RUY`M.PQS.F] +MrU)G;`+emlW305O3UhaLaEng@1,&AUuI5@nu&XW4F:?2dS6GpD\Kj7ReQ?cbb%!_ +s8,^Pc]Z%fM2_!i\qH2tq#>4lSdUgL@.eW9n?6XP'C\LT*u+S[R9s"M-nPM:$h-B!"@OC5('&*M +.4Z)6fEIKkeA@4g]IBdu1os3db&7D\F)d.on;jjSD;fL8aH'>G03A5Dhl6[+`0\@1 +9e&?3Ldl3`dr=su0#/Q#c%-]9;Q3bj,nn22n4fq20X6GJ&E5o\$deN"5QqPB=_Ltp +'qY1kWPM%M`ll_+8'",M&tiV5\-)fJ]@5*Alb`FK!3AGI"Tg>Bo:j,R0S4P6Ru":p +UR]%X-PPZTip=Ja#:[Y6Lbc85=uO57#gkCnH9/k*o?hkI[$7&KE0eNbjJ\%XZ'L0i +P`U_70N.h'nu4@gb_0kOngY%"=:8U;+'FYV2@`4a(3"67]tEP1$^dEgJd"hDRO6(7 +j/<=3TLKJr"[EFrj!.ji\IOGTZX\7]r37sUB1LZ'LYBcRgJJ?W^EqXhP)]X7lXJh! +Y^iCU#4d&d>m%qZJE0&QTnpH5)Q\mQ.j=`_-gT3/NIr)OJ++ILqbq"IR0b_d_1*cI +s3@<[B-u0_(N0BS-R6m`5X#%jfNAIfW'HC'XUO.8SUN$W+.r$3a"'G[plB_bR/aZ! +\0h5';RN!p^ie%,I4/9W8cW4Pb:@d_A6-=\A/6#AI"O_Xf*c`Z!a75%)[-(a2NuL! +P7hF:])3l@Xphol6B93]]cpjJqsbEC-?4K&*h$LiYkAWPig>C:$TDUkV]B>YY!";L +*QKDZDGP(1duX;(#iu$5SPMuJ3]DKC\bX[O([4^HF3cI3mlOWmpM0<%X.;2PWfA1L +*a^]Ch6;EiCfitBl^#HSk-`+J"V.XQTd!TF<28$k9]tD'gO-/L0%VeLPqFqfY388C_U&KeC"uN8hV=r2%e_f&Y +H/M6DK,_*dY*V>#>k5`\d2TRbhnPBJUO*hB"Ud:_s4(VUs7p3YVr'*'r5J4k]n0k( +U&=q&.D?T>D$39tK@/rADr)]$LOZ%Z'=6`U6asNi_S>jH1",l%r'qO,"o-sjZ2jYf +/2RrBnBDJ'(OsN>pdJIu<2.B'5U-[V`,VP3L%m;C +FB/g\X+:41FPC:W-N@&PDE:f_'5s741h'0,)Q)MR'@D.$A]+MiUoP\SQNp2.Nul.o +jF=bL*Q!:dk!+qIJj"Xt\Zupe^Ku3de0)rH!4Dl4(Z_WIC:$62BETcQ.?OQarseA6 +Z>pa%oDcs&+;Mr"jQ+4nP!]L]\C^5"o3JS#W!^lW#n%09W6 +H>.@X-0CpkSWnr*4CfrM>0$_c\_gBS1K&UHZ:@9R*!K)PB.*BD;l6BolbQ!p.KL"_ +N^=&&Wdkr;OnIB'>3,1Hn=c*IkhqVJES`Q;l^46O9i"rS"cV`_!ld]oT^bk_TGXsn +BPtHjCN7"6@B9hn5M?GA#>G6(ZQ2jDN20[5qHD@IVVZQG,2.i)!$XHi%1W[eK +/7u1]@s]U[':?Q\':afFRM=3DG.YA'\*[dcRrX]ZR^Pd+B[Zdjh^`K4[!0j,kPWVL +5&38U-Lgmsc\kp)+p]2c4UBrUV?@2)/"J0t4#A8lo#2IaiIB7n#Trt/3li9.jhO[3 +&HJJ6p*"F3S(X#)mi/:n*NAs8G:RS\`J'Jt*dticC7%]kGT, +XR0P']=lSuCi+?5DKK^5p$6U'CgEOo"X!jJ/SK7I/b_?'I[_0P+L"c4?`ruPe-$P% +r3KD;R7_`#kH!PThbpRp[GJCDi3T]eG)jnjot,HG%l`q:D?i-K"74L1L)Zl[(Sc*h +,?:)].kbh,?ThQkpjUVKkAn:[,P0[!11%YXorG\rDqn.aj$U1t*i"2R1caNE6Lr&` +!*K6q4T +J\QUb(XFG'#U$pP$P*(6_j>E$I?+!Oj4$"OQhcOi=[0PFofp_As2*p'7fp^agJdh_ +1Ar!Z66DK^5J#CY9pOPiaH26L!d711'tL%iK96+QOFL%!EU8/t2s1>&s-08UeC&ru +ntPd!P4\Tk'"8d=!m0*J8q*pHR+j79dnB9ANh:sUC#Qq4/e?JboN+bTCFmJ_6%,`CXe8=PVeaLo +("8C6]\jG,`ZL?E!-_$^eCUjP:F2LK`s<4R/dg%%-!RTqEKl<-CtU\F2>sc+V',K^mpBU +hA6LjQrj@"45QLVCL["*=[Tk_,Q2mt"1a`64CDVPa3r9PmphCP1+XclBK'eH80@,m +&*U"k3_HT1A8Wcag.4B/Ui$q#fKq;@8KCMN0Vi1r[/b)kX4O$SH9.gPK`8s>A)e5B-;2BbqkYNfc&.SAhK7*cmalhp^\@Pq*%Ymcf[b>V);-TBVXih'i8g=TZ +r9mmVJc+>))>6;9+6-Of3Ys*ps%ChXKh:7q3,`hr@i=7`#T2j1+7*i0.F;@U)?FLV +\n_LrCPN/[7Yg9K8(g`7ccl_2AgPV.E^Ka4k?>48ecN@`c)t(9Il%#eXRf/>s()H7 +(QK!rL\`ti=VC[eYm1,e,9YtF^5Cd`$M,t*09#io15<5`JcERaL]%`'*Wgr>VkuG; +`'c79032>=Z8^VWHU:C]:5?=\%7*\ +mB5h."&=D_]g/AV)K(N=<"Zl#ltq=g#WZH^4$RE="]m^X'i&/9+h1>Z^l*]Q_7.S\ +=ZJYbf5KjJ?%RT2)"`V.?N.N3(_?*o"2(0IG;mCqO0DE6ld!IaDiRoV\Q/1XDTO]4 +o=fSqpHL:lI.-da?X0;=IP:ZE#O-.k8q$.WFQ!1Mi@4R"B68G=EL7]ds's:EPf+q& +U&/(;!^?J?_$.P[4ZKV-@+aX-rlH/]fm$$&a'Zmlrr0kV!jo33mpF3B'#?km-K-gK +`6]&=s&ZMr'_(!DTDRJdfmeI6rk!qV^UQW0r[hjBs"A72HdmQ;/mIV?q=L>61uCrW +s#7Qs/8\UO9#58*d_u&W?JUI<$?B9@7U\ST?LI>On]s6#`BX'G?>`A$mKn]"ikj3n +@c[R#12$]er)t#G!);88(4N\qR%of)')jQJD_2mTPZTt=-bIO`9pLXMeZ4>1CW!Q? +Ym0^1B8Y`AjooAlJJhd5QLCI-q.=m.BeFdN+(qd;e1M83ch>scROUf:-2kDA;ephp +F,i`0<(Z;5>dU(9r6ME=k>\+++1,BVP(H9&!4t(I(_KA'*!)d;1^&K0!9BZ's4)o& +O#@dcZbTbOb8A?.0@/%hZ9+o9#$D!O!cefA^nDJ;8,o>"i)BWiVnjbO2V0!&i +%_e21,!b"P,EXa$"<;.Z:]r3)?_!:9_(!a0@Ltgj`t8<2!5:KjKl.N3&dq)F\-khK +&H*+OOLQ`ofJT2Te\f6@\s^QUAp4an@:](R1!Mk_?;pQJ.;."t448m$m`cV*1Ib*_ +OWF^+(B^4e5!@/lTpj[a4`09J0*7,MS=qCQ"SbcRm<.09$`s[B@d:A +*FW;V/83mroRm1Qj.e[T@lkVF*!%#E,EMu:(OubsqTXIB1]g.Udf9s[NI]j=Am89; +XgG13-jfo4,6+jfh5cMVGS69kdf5>[qB,W"5:r=li'Rl[)<6a:E^g8dW!gE?2^O32 +i!&t/d@l*t)u=mG+9G@Ho[KSn-AVYC%<4g`,sKQ+1fsoe^u02)n7XEIs7dj$l$jZRkW-PP +RcOP=!l%/$.VE4\D&HjD#T+DY0uFMYQ1]6cF;mlb]?h:,S_=^)HP%10+mpN@e.d)j +T_L0+ju6;!Qn!YaF4F^=HE0Q[Fi3?V8;'9=)13gYD^t37+h@u+-&8=6I'"bto'Ynna7]jI5*u/M2s#McesrAh62fp$[,5/c^%F(E(]m$M8(W+$Lus8'biHNO[19nN=/\PeY&S"C#Es3]"@J4P7H +O'^*4i8s!co5QC1r2#oN%7MnSq(%6Q7tAJ9_\[CnqCLr.O3rH(2=m_I8?^m5qQHfX +^(`g&M!rEU[M,S%dK3o]$YJIL6TY[5ItaC=("!3OXG6#)o$YbAJi&R:;Z=3$LBmn[ +"8OstiE(pE;G)jGb=`NJkLEBt[<+DJ,(QX*A0_+5j`NKu%($b1k_^_rV:48^o6lls +K/L*"3esQW4imoY#NkfT"^u(q+'oXLQA"a\GPD`$^bsBFO4RT+CZ^CeB)8RKHch(( +Mu'dah5trCb3V#HZWT\5jASJ>RG`R'[QZ0aGlDk`E#0V6'\Ms*$@9JGCpj9!fVsi*Q0[U'8V8 +2'O.E!:W4?^JW.*#QXJ>A1S-mRjV]SE!2iEJ>R/Ps.NcW+9dQ,+T0R(@mP?;(p=;k +NZ+eTFeORo!.=TYihSPCO$NET_*Qb/DFCCb"a"l!(moiYMq)Z0R#0Rlqg3@6Tl0"c +OYHj.F>/%6FH`$Ih+m?P-%kp'<0Y[?'%!9&bJ&N)mojQOr8/93l,!DI +J12Q-gPcVNs.A'd#9X2Yj!Y(-0+].`Kngnh_/Nd)i9g3J9fNl3 +!rF'.!5'jS!I2HLTCW$K,dBaMS.?2c3l6UW\CL_Kd%iA._hSn8JuMejL@G6.XFfqkj0V610d8aHs*dm0F9.#k"sOFid9i`Z[^Qdtl92i! +l(%pAcijgd<=OpQrK$[J:/#Z)Vn!%q^ZPR\3A(:D0Je?3n%fSiDW-P7e.3#>o_2AZ +RS+e*H@*fYh;pe*kTHWq>u?BS(fC>Sm&qurFVORmPn!GOO7lTSA*EQtWptZ*"T>_A +"*NXV%=k:Y#(C]R3+6`9!6p7WTi8A"Ak1%27g^8(N50C5Ho15d-2#J]g'tCQ2_C.F +XjV*r1%0)Ch-,+jVM@ +Ko#7C_Z*uerVIE%5(#QkY>;L]HBe-C!u\:JTmZ9I1n$:?+qNo%*N^lX,$QC+ +#e;ceD.ZNgFtU0j/E6Y?`3KR2`6!1ma1>9-=:^[09$BY-'ch/YstN1^TYiYK6eEj5Rh^pdB62 +!aYj@!Q?KU"Jd;I!+ASKp:mDp$kWL==)r$"4M"-UD\Y%]7*Kg/ninI`!`G=Zao_BK +ruXYE5PP3:=^],HB]9m.BlkirZ2YJA!/OXJD2l(+&HSsK#W3J%l@8/M#Qc]p'InUL +s3LgTn5?sVH9G`TYV\kmc8"LJ.6F5Q`s1o5n1lc7*?XB=:J0b]V$lnc`T>)SX3t^!Fk^+` +N]$n:#QR6mpn+Mb>Hn,DVWDl_Eq',(ODo;D0J^QgmcQn>%/.!Cr7M%2"fTg&f9sir +CN*>c3jdSm=5H#8FiaSrT@_9q%n?V;!<=BmnGb$]R%<=?pj8kbThjX+NRjdjJHdV$ +qo\e_bFho)1SGC"1j%2LetjlSBM5#nn'REm)G6]pLU-rqr5k`*l@A,]qO,`cR5+^5 +-jSUc%r3O_&83*6=T+2TAoXR!C+4.^AHt]()j\^3iu.aCRHN]\(RJgE=BW*oEc6.: +hX7MU_d@b;Ljpit]b"m='2e/#\*n/GQ2e!)r/WiX!B]%A4.F=k>8qnXs7X$93tVM< +\.OOB'eso2bh3hUU*)^(i08n#;K+gMje)%W9hb4!ek9th/R.SuXOPA.l^P?_"rS+YhjSYq( +JG5[,Hgf'ui;KQ/5'RheJ-$;_5/.j6YN/q-m7I0S'ke/Ga_ScOq`fIZ:;HecH,fa- +1>dG5JbeM4K',ufr;Ga4K)m00*/HtX^QA']QZ><8NqMg=9P@Pb\1deNGEKX];sN!. +O1lICrZTV\G?Bpnc_NfecPT6%>+2mRWK8jRHj6d*?=-F,0$PdnbUgt2"97C=5?Rf- +r9(lq.0]Mr7>7]e`T^q[Bsje*%2EE0!.GYO>YbJ%X+fa[2.L]9DemanY%6$Y1=];e +f2i8*?Q4dC^VMkH"o3fHf2j9g2)0Z$h("5_nfu>*Lk*8bf/q#m:"MHRCTWoVV;^^- +rpg@okpR,EVY7YtF^@-8;bEDJW5?+sDC(bF"lWMt+&@J5i3<6$$QuB;&THD&$4[dp +5fKY#"ZQSI&-6G[J,nF`&n]Sl#&,goOT,h]5]RT91'2b?$Tf*6L]!2u4,\r$^`#Xj +]-7KT-8*\2pLlDFJc)_I?]qp",O'[t!HL1V_4fg1+Q(Vh7^DPPh6iHAlW!h^.p +=HsEui.!G'/WsmhOhGr?F.Ppg# +%Lgl5rBE+da53c]?DN#=%-<+Si(8:qC&30PdGB/3qTb:Hi.EDu+$N9li$mrHc_>M()?_B.Abqe( +:$1IP=0jJM'NT?'`o5:r@m8j!ddcd,6YqS0B_"8a1uM>ELbCA@2FI:_I$dZL\q5$_Su>FnTXgY0_HMO79O1I^_$VfGD#$P#k2DqnEKq_X]kuD8"$\a,2;,H#:.'7M&JhV\4 +0T@c9RY$m_a@Z%U21(Zo]N2GtV?o;*@@&/&0uc>QGbAt"fY8FEhH:rTh&,&(SD)kh +hV1eU;IFS.jMpicqW)714nD[PS$I% +r/N)&D*bfm0=&l[Q\iV3k,/f4;uG;tiRBgt%06oZ8"^#&j:2T="p]Du,=9S`G\0R^ +0\<\EUp.#1i18&?],oqXqOBU^ld34[m>sE<crnk8/2d*dp((16O4K=p +l7E[Jrh#Y5a9`ZBe:D:8=T?kgji!JiPrZ'/dO4K8K<"BL@%1Q8Fp/dn>^f9%B +?#2D7R1,oWYG&J`=?9tZ2e1igZ_p:YVT7u:fO$r%N(bi\=F'V&J]47B+3n#AJ4DE-h> +h%1-g$d6kNS+abEHVja_[&5hlkI,`_U&)MG"9+(9@"6o:5WYFjkWl68JXr0G40%+o +BE1.8YT?\2"3hFm1?<9ZH5Zl,W:ORli6*BP?_c4+"7lbG_Our3oPjoHT*tXZ#.`LI +L3'WR45uNtgFddN<9]`f./h_VZr_0Ur!us:`;&0ee@2SYhKaqTraV0'*iHaP,Ro"m +$-HTrc1oc*Bn(Hm`*3`ns8BhaIs\JpEF4[Pl%@_=c0%tJ@[U#UQRh!:2)E57EN@_>eo6%d8C?K]h)ERMo +Cjr%!ZF@na.N.M$12Cq?Xk=TrZUIJ2AcYH8WXO&RAo1C[9jnA/06P6JFmpqSD(&!= +QXt,n;rK1,9H^@XOOfTUop!INCD"q;UZ3*iT(iKQ_8u@!@ddc`PDSnk"R$$(mmbr"3h!+dhOGCV2Es-Ai6V,5YEU@:i#9s +=pAu[CuUD0_l#kp3&!%oL.'^UjA1/[mN75AK&k6kDe"CpeLH3[`U%6)&6g#PE\CS'0iO[QRV*02R5."%I(1p2#!^Wr9mum +#OL0+;]g"'`k:-n^)iC9AnYR"3eI;@o#I9SScSP/B(iZFP7b#6Rt)4*LfiZ!&L%DT +6U'!*pm_Q'>n%D(?<&&,!0D]Ys!r/h"X'ZBL^9@`9i(9+/AN[+L]H<$#Tr+l4t`Cr +P0M<&Q5?#VrcVKraPZ=3"J)*6hkeA@b5]Wmb/IL0pj[Tf5%0RUd/5Qf56e%$J=-N= +2g&+>%7pI=\e:,)_%jj6k.\-ss7,kKpj`.Us2=aU2N@e(r4+fg%Y@&b!lncL_u:`! +XMb])?q(aTf2L,L;[0o8aFCFlI&%S#Ir`%/J?"'9K!RECc:8+N!7j0;A'P;))*Y8[ +_dA.AA!R??a5Xa>Bp$09aKmL>*=MQ.q&99L7;0+T10_%??a..E[(fdn``@65;0Ua= +53ilP%>[dbjnllkrD:[DU_We,oQKtsSY#eDI2G'Y]M$=qGLMZG5F-fIs7('_HDi\: +Cr-)]q\5Z[!3CU`>Cp3E?Fe*(om3h;5N(YuHp$nJ369naZd2,+DMqRYYZ:PlDXlmc +'n,+)ofKW:IrIL@Vg$OPBD +GOLZ$hfn]OpE+bBJYR_)HM"=pTGL/plP&OO4e;E-F,@F-+p&%+SRUfspm6j7*sqam +r#umPKm0.9^&8Z*n)c3I/.IeX=6=W]hkrVj-:jSpl/d/WkHoo03^IWefX%rP#>k3:o5N.:OMht;1t +YIcK84>r@;mE$2Zcg"J-'I'i=9BA$-)f!IEV&ABKaR.o82#]kdp0t<6P3[UTnl,'- +5aK\J0m/`r1E0S +"K2JpNWikF6qf<7s6fntrK!k,Z2`5J+%9L6rW($sXG.!/kYW.>]uZB;5)Uc-Dq.N,W_6H8.h#Y6(f +))"q*8XD90M)2a+P,(=U[diECBI!.BS!3R&j@n@10 +[k1.er7M$bo9.XL+p-+&O:[eO&OQp[4X$?"#U'8h7Kha]U<.E>bZ*WX9[hEVMfhI* +?Yal=gHkg&[l3I7!Wqd"%($"Zk_\fDlI5_SOc0^+%l;_]mk_4C`k&=uMG?<6rl$om +&]c%tJ5>t_ci9a45D9'q3D%`o8cL/B!;n*gKS9BGa6TE&(`fb#)+"ohglM.^&,):k +"(oW+MICQPQN,H*rQH+*G+?6"`IkU4FVaA-K>^H(nBXg>>sPJBMO#(+0h90E7fqF# +hmp[K0@^4AW^;W/iJ$j)Kp(iBdZ^_3qmP4,#CG@I4lmE/r:^8PHQa]$s/D"'rQVKD +>1d3hKIg49U:$Z'`2`6^QTi&^o]/c`.q`PqG*2IR9<95l`q\4u +atN5-I/Q[W+7@>K_ZQgglQ5Y&J(NuYkGF;oL/'9;dg'(*[7seE&P:';E9ertn/qQ=DC6V. +c1<:(a\BlETUB0"rr95&TlI$3aQ1s8UjT1?b1%h\3nn]"qSJB_P)8&$\Y8TEZ"j.M +3nuU&b`8grT^i?`A+T0bA=-Z+(k<-:(U10a,9if+[nu7VWJ9713f)J*d])*OZ(BtS +NdR$;VaHa*hSa(>UG_tD3\nI=">)lZ,.P?H%DsVia<(N`(KY6Xq.'GSU'9JM;'=f4 +r,cC;R&-'sUAd@s!Qg$\s)_2D;43KgI\QBd!;ZAt(1Bn=Kfs:/DR>\+W2mVYcrQrX +/H?n$g6CgX-)(uZ5mmZr2l0E^KYRP`mD25j*.qC=Du9C&Fu1?=f"(WR8,DoY3$@m>CeQ2hSeJTg%uG +G%BfICih=JJI"6C3H9@\o'H'$L.Hmk;qlBj6mD[DO@Xrt32F>ZGUO?kO8+#gDB2,q +f@K4jo"J?HrQ!jVf:'_sD-7n/U#pfQH=lnXjoBL@sc._ul`2XYV +ZV&fM,+DTr>ig<5f`(!D^iE._WG5qB._]/qAZadoeaGK-4&,hQ5M\!M%4HGc7#lZtf:osN_1K*(HfE5'XI +:#^:d!4PndECg2?V\(,git;+"U4?0])F($ +>lR$T;%!6dm\A,@]_T?_Lk%]SSFZA`l$n*oh';D$3^G0fqS0d9`r7@RQc*)7B^\0@ +Y+Bq)E@R.OK?R\.Ac2I4`bSZ>5IjJfnK#"*?`*h\%Y4IAGL:kZ&*#VC6!qG&_h3n4 +5/d6F,D\9#VLeYP^Q8-l1fb"9E"_/UG#*?He$R/Dp]"9[THbSps7K2f +Z`-Z.VQ,KP=69-hoD?.W/A7n/"6GH9Q+"dQ.-(8aoar\bb'Pi?Jc$hkId*h^%%\=2 +jJ5:fdO![lnDdPmFL`R!eDQaqJ_jhc;@Y:oFFT.d<&E*-#VCR(SZPnt>%J*GT.9$N +`3j,^74,K+K3aqs:m+rVaXogV1]N+*.H#\Br\WC3@/iS%5kps@aIeLUJ$i/i7OMbV +oJ.24rGNtTKu(+-.1an&a>YOUb\6&bYGWZgRifbeXT7R#0*HS-baQCTD37Ckd\AD^ +7Q4*XG-r-1Zkq\ETX?cr7f[W33!);*s29kHN!1YL&-R*L&YL5+!$VU16mFt7&V0V0 +&N>am3ZWePhXRg7)YP?8O.$6hs6;LGMji4E;#E]6%lHS7^R-R(>!eS[L]q;qJWrrr +ECj%<\r0;(c4\WjGdWs0l?%2LH:)6^Z"fWps,+[+;["21Lg(tmFqXd5D28">SM9hg +A1p,ic2V!HB=;C^;IKK-"Dga6-f\,QitFN?j'0p9G\@X4+R88"6!8)DF02l1QCD]K +EmR!.5CUHJ?_*BG"9;$n:Q5aZ-cFJt18s%qrl^476N.n4]u0%$%!]BNriU4U)"r*i +9@.=%/-%m$s7mH"5K(U<76WUG&-7=)/U+h'DR\9Y(3hui!6jp) +$!^f.a3+qgg$rJKI\>uG0QBAAobF +bTG]1nY=jcoD=kP?bQ9fWu$i0HL^k#h=R-(hgF7@DS,m\Q`ko;mUHe/J%U[$"oe'` +lFW*i]_<*H>pAl&Z!ZF7$CX@0[L&Ut!m`)TGb4W\VnrE*2]nk`jU^p3m2kDNm\KmW +m>UL>U4)cJqR:?SI/>8e[r8"XQf#Q;!MT#3];WKUL\gjqeDEH?!;$6e*qCuP,lRMh +oN*[eqtL'L?ePADF2@/`E*Rnar+EJja'?;,'-6=,$"(o/l/Z! +8LZ/`.5$8f5V"r)#muJ2^_@Q[,6&IRc+aSbQY->h)Np'*BTY*Yc.QBkgtp?SOe/;h +^=)i9nmtIjF/INt:`im"`,3/(QN.GT[1Ac;8hm4F&n'TMhY2m#]L%F)9cJ;)[3;p* +i17&p:00aiYV]691Jfi8.NU;[64;/hSo&/_ehgs=,]j3^/n7h6eQMgm/0"[mqoOsB +N1`9Hqd0DS_fZZ=4Z2JR4T:E?O%+[gM$Uora+42pp+[TYm* +87`KlKED,h67j?'nbH]\^dW[JJ5$e;!:W>6eW\oUP(oF,\uL_#MY,WnN1.'ERZ]N) +U-.CaB,a_JKE&Ohs7bqL##7VmpL9oplXQhO`m8HFq.%]ec/,ZfGA_$oJ/#d2YdZWV +N.C;_AMZ(K&J,CnLOEGlVp;WG-8spC63OAA,438]K-F\U!Vh.^=u8#aR[k);:!:?/ +M]i&qQF5gtql84YdsIVi6N5g"5S*WS&qh&GNWA8KTJ&8As(OXc^CpH/0d\.C0-2$r +!eUTuUOb9o!^Kj_b=?AXBr`,Pog"ZcVT4TVk0FYtMcRZ/6lrgTT2CjiuehtrTKIE-`2=HJ>a"> +h\,W>LYU%F$iJoCFiXm\UEjm4`#,CXij=Ud"e73%$R'Q&p^dHI2e\iPp'G#2'ELB> +-fV2]oN.XlPhd<2^V!G:i'uQ:-OZR&bC=e%kKBck?U%9\`gP'N31'HCo9nLMOS.u[ +&`:0]p)c45s5pj:^phU=juXCY"X;lSDEm^8.nAL,pZN^e56bBnC&oAMHgf(X&"b$D +m9-f>k[Wf[d*6$&+$N1=+Fj/;'oIM1s'M@js.1)$J]e!_G;B3a2TBrY@K--mqDZ^K +^3Yh4Q*\]>B(kedu,h +pFTFVB@6@fb#m?p@.@3'nQ8TGOB&YZf:cn,7J=b;P'V"k#lBV.Ol +SggZb!D3;M2UK8>Gl6a/s&DB8p`DqTR+enjZB5Yfr +H;t-jJq$$i32Ak&#DTV]K:4oI%`""*BL%XB-\)oHo]Qs@6FqeBGp!0OUHmK3#8(pW +&\u:M3!+mM5S;AAU]Ejj:`EEA-8efNMBW&h$Vj1,+oekYDNI.caZp!%OWlBG2,L^0 +:^<\'.%;(\s%B?LJ!#Ngo&oNu\NDA>kJUem\b.4c>o'o8$ukLY"B0.pWOe6lX$a%5BE?\4L5g_Gn:)PNs8)5a\-;e[DI`6lbpgk^_Z/eC +k0NKIiWoO'(Q`G6+9J2D=:b'o"/_`he*@!\UBj-a?]\^rgQq>JO;.[P0T1!MLk(1^ +g#ANLBjS'MTNYW#Y:[4DX?)]M"3KHL*V,U`rfNi[LkSP_!gJ&WY.VFg(]QoHrsi=+VnUg1;.;AHRCXon]iG"5kglu#0m'5; +JcA%8r$4)IHn9aI6=G6Soa&KBJAcU21g)ql,%/iN(@"\j[JC=LF@Fn/HZ>Ld4Nh*& +(To?IZ2$O@L"EL"4$fmk":C+B-24ng]`]Am3iF-blZ"h/>tU/ViC<6(qXlio!P?F6 +%VLk+f#7%dfqfji/F7,/HB_72a+IdgpdjYG"""lOal/C>Z +I)+n'\D1nmV`*P:h!pZ%$!#atXhGDu]RR9KTrk'0[r\.sB"kKDrVTFA(O*4"]V"l> +.FM9mB79Gpk#Y%sGaA!<1R.X*@D)1WrD/L%J+k)af+0QaEWoJB;"K1s!Y_s@ +V:3X)G&q"QRJlq%jqi4*?TCRTqH4_Y-/Z7qr9DP&+5B3Zi?^6A?bJ2S]B:ubGrk'Q +gIJujIdFW'B-2;'IsL;"RH`bZd57maF%f@bZt-QnBMDBn6q@u3BuC]N-5:'WV5Z]t +1]et;V8[,1B/,MM`C99J/q5<%:/B$IVNb%eRDl5%8-MifB5:I,7VtnM=8nSt?tu?Z +@C0QVU`etWNrY&b<:bm`@!ckl\ZdmsI!IaqJdC$#$[+Yf9kQi; +=r1GFp<6OIAu_rmAW+I&W!+jkNYqBbFa9jmln29JE@r50-"+mJA8*WQ!%@Ec<,[%o +6MhL#J@LafW?R8$a<6!-2PTuEaZ5/H/f!H./,o_I_ikH,DZff.)=Z.sQs)LChjAft +Ak*N8;iton''=,X#.e#Z!-:eNY:?_OSWE+7ISIY_[V/cEJ/QlPNp$j)_];15Th`Nk +8+Bu"8\^sp[f^2-\#us1]g/XDKa9qLJ=IpNU'MTLmU`Ud'ipR2`4$7\G_qG1k\'@YHWW1pgBiY_FbVuXV1g:c#es9KV/+!hIf&JIZ +>fNEA\d;:lcl3Ad`I_\E)M0*A2?`)^li46h8)sX0NT6f)d32*]J@MdiAF:3)R!;)1 +I:m1%pg9dFeS'h_RUWt=\5QbgeQECQmkIk9';><'"pXCJ8(\saZ#S(a,k%u;Ve61> +1C/d:S-"KE5;O[SS)XN_K)i/QF*aQ%>u`Rb5<$E0Jq%21C4>U!$u]]P?sS<*?OW2a +QG;>1&o]^`4A]=a6*NrY5%VOeN6^V\.YAcG&9'T(S;dQY![7Vp2uj$LM+`RuBlh@D +G+$r/p"AHbB+QruOD$m"os_%.2&fu[7]^*@]aNJK)<0&hpj0tM]FuDcTHh:-IP\.E +JH)><#Pl7JM4ftQFi`AOE%7P?n^nac:D4B*IhehfIOS+q*4bKq:6#,J,cAjYMF3j +2cmfp-_LYK5%:d!JPchKb:\kJsN>5"p=` +!@[s35J558mNj.Y=3bEm][(nQQheR3iFDE5-fR-^DmE^%@_R5Qn]Dmf\rqP-8,Do_ +nolaKrptlgDk5/N\%VF;D5DV/G4bUgZ,^-?m^4+HH1(IQp^,hc`aUl@84t\EOMUr( +Y_7fbim[Z3A56uR6WI`@Grg$J&-:mhBR=Z#8Ht3*P2ZoF^D/.ae%Q.af9VQP+Bo=U ++^qc.EhN1dFlj.$$HDrNB8-e\/1%a(E[]arbOt1:lqI]ij:>])o:M!fKRZ?WH9Ne\ +huu)u636upho7nFj__Q\EI1u)W=$*7[YaROOV4Al-lOlW%!:S(C3"W(F#oQL/pOd[ +U#8b`lJQhW("aS7T0\QmZk$_4iIAX?IQ7Y(-E7d,RYCBGUjZ59A(ma.Fr:aa +D.MiF/.b!Y+*/QS(=:Tj?B!.QmM8ia._AcIe)-T!oX3lUC\".D+42=r&*rdVFTl:O +6GV#H1tGf?0ba_b>mZ5p`C!pe5snLV-StmY7W!^j/RR+uLX,*aYaT)rWK[_gLW0eF +?bd3ZR3Eg7?on'jVZ7uh+p)GMnir.bA.+J="[Z]_#L:@CL]("2dIL.7)ai/1es%Xf +ck$l]^Pje.:r3/ODAJCb"/R(:?V@B!+kK\U_>MiK'JP#N3N8f*X11!mW&UiR330P; +Z_T9rI[n#*,6IDp/c_dL6iaRoGSCGZIshIss+\cin:-%l=lA)d49jkrC1Cj?4q/*S +jhDHHgCB9^6jD71rRh.["X&h:OQ01\1tI4iKOr72+O%#k>ACSuX6hF>:92r_dsE++ +I]"pamKn/c,(@mf?T^F^&,2d[,%?-0^dCABrGT0RXf\Ndl$qsMYS;uphEG@l&%;r- +DlD4PX0gS9m/IQ$WM-)GOFIGT*utT'Wp]1C#IobPElKXmb;eB7_4?tQ:JM;5r7Lh# +!WSo4iI_Hr\N.M(4De(_t8:B +e>lICeL:*aFm:2kb%'tC-Xs&N`DW=2:O.6O1LJhKmbc<=p3S_>q8obZIW=PU2sr[9 +/\iI3Sj3qiT1f"/K;j?G3;qZ'1>[:9rU#BghK]qk&5%HH$$=*BaT%<^TY[7-hFDq[ +$N8N0kl6*orlR7`k>12%\\if*K6_sRjPE`4I=^1SU$P8oXN.)pBB-?8\&_q5V7a*t +p!paWYF`u9nlFH\ii+S";kFF;J(/W#lIrL)(6>#W9Snh+4*i]\%HKf,Xg<_]Bgs?@ +N=rfr8/H9FJpK2]-53t@RYirW"V;9-#n_]JH76nPJGCoBs1/Qej%K/h30+kKGM?X@aTtLiU7n_6'ENqTL]F.WaO@S'MZDN65QlW8+T_r5aV7:)"edQ! +qPa`#o+Ufs-OtqoZmgrfjf^Q(@LeG#!UuZDqqDrZs$M\FIj'UeTnimb^ueK;!p]&A +2)V1uW[s5rQlk-.8!q&q.'oR#"&i

    ;Cf[h-_)o2KF[86W@ +blT9jO0L;ef9TBDI`H2mCJC.W^T.b'+82.enqS:pSHVDH/GohSqd"O310eU3N`@s> +s,$E@aT'3RNtCT+./o4^%3Y\2NbU!A_>d\UM-^_TYNYDS"F?#,k+3J:Irc(oIpfKh +A<_lJ5<^4jq+o6@B4jWBniqe@Am1Rlr$CDW>Z_PFn28/U1Yd[e*YeNs[)]m/ENG%f +]4K#l<7Arq8lZ)\)%@@4`$"BbEh@Z)A.1!#&^p#!iL>a6&G:gTgZD +IT=2E&C+p&\6 +Cd?gVXVRE;/3+9"p8Sj3`L=KWmB#f(.`:-./PkVk[q*fdElMI.hOf9POF*1PFeb:P +X.OO(T6WFf^Cm9s)pGlHOSW?-mEnbaoC:.SEVm>NT0E+D(QSLaQ=E.oSFXZ_WP?2$ +jnJKis24f^#b6/2JWr)88GhGtFEeFk$0"05R.fZ0R;Y8[rr%J62m-&.F!('f=(kG. +J*.UDf?ViP3h?6AakR+[EpeoBD\&(=neU'@b.GQTH^?D_5!SA,+T8PNrTSIYZg6]8 +ihYaI=jMaWT*hG[mRqoZFo;6He2s!p0?g6[=OI;(Ut$O76jIQo__AMSoe!8R-<@I3 +d3PS^q,Cc)R9u9aomC%F7Wiu+1]is(kR'j#!>T0KR-HF>5cjd:iZJ-%7M,`[J%[H8&]=*YT3>r,L&('*_*lRj1B-nr-n(gD$:F.q"F]n/TVX/!(n +2'c<6)5dIp"8K1,Nf")>%u1&QDB]uY+#O*&f`0&!^b#Nu!RsL9B-dH5 +XJUQjea:R]/YcU4Ap_p+?UG=^s(ooqMBGr?+okn$e':5GK8P\EJD`)3r%FoQKiIPk +4bjI=Ir0%'F`eJ65J!pQX/I;=TK"aB-mPoL1G:rp.*+bL%FiU\2F&(_1rVtcbnjoN +jX,e6TM$6tki;[1XUpp)3#*G[,%(K5T)_RgbQ&dc1E3Zueu/7P;nEXbam^.O'-?!? +%;c7Rj+4pe^FA.sL>3eu_Z.QZr+Hku\q&(U:7C&p068beU$O/NYOSd[rV"Ne_[q!8 +7m*Z9YBSg[Mg)"7Ka5(r_jf$S1q47a92#B]\jB^*mP<-)YZ3RA#ho<:rBGkqIg62^ +rVZe]M.8LjJ"<_Vkj5D7100jTWrYQd[Jdf3!)`d&$3(C;s/2d1r1^`)EKYTVg3S3; +N&E?5#S+s`[(%Mh=qVtoKM.5N.#Gd"LPL322Q<%AdK*VHA+Ql/2*a:"NJH9KBUM"% +Db!1_iHB<-="&(d>28"or?$&1:^c2r:&kkkpttIP,1[.D?$6D0Fst&,Q@QFZ7mK&, +_1_`gi;e\I/J:fH!.GP6ki@6?IO#3VUCR.Y\#XoEAn&?Fa_>Or>NNu-2OEUGkdG&9 +Ni'EjaBFos;!kq>'U64g/%17rT#^Js,?uf&WQTNZ^::1B'tptrb[5AF7+oA +p>XQP]X)C!(Mu/k40gNOZg[Eq5N$1'c;/j:5^h!Tq4'P4W[mC!Hhs#Xe6"FVK*Q$3F3#f7NoL:%_IS"-$FP6c",`_#/^\8B,4*Q9Uc: +5lO`bP,<#lA+V@3"6[#;]8DI0CdiUVJGaatR3;Odp2#ijTI+kJQSYA^)hXTO0`OOB +!-(JOQ;!gCY%M/Ju*ZDVBKVZ]Ki"sT_TWt6JJ>F46H8W3`0M#b)>M'CkV +rYHno"R'N5&;l!*/V&G(r=A0WfF^Bhrmu7&-iL5-B'U!*&)^._3!G*R!"Bg>kl[X` +J=d-4?S&%0`Dh@t(us$Q<`<1F-k#PmNuA.\ad.`ffUccmHuXdgH6 +Ka='PT\ZJYcdA6c,/CP9ichbb4jXA":]s1@M$Oq_[lshCs,$!uR<'=rVrL^RegL3d +iunAmcmWQLRt!6@+8>Ft^"l6Y/jn!>:2*78"<\J3U1HQq@K=Y=cUgG%Y/NlN.TGjt +T+*&5br_]k5h;="4^E"0nG+ul%@maJ*_*eEZH=Od_Lu0$resZf53],6gWjXLg +,gUt-*1)\E#R:T[]tM)HmY6a^b%0TEo'bcj2fr%bg&I-4*uO&I%J*'0pm,pYpk4G_ +H5%VX!p@lCZdhX'BktiGX:WPs=BlPmq3RGe.nem`,lt6],d]0O7XLe?p=L'CV``m< +"JLA?,ik#@AI.n&_0H$OZ(N7$Ej`Zmc*+aRh=jU"aG*IYs`Be9J@'q,aRMT,O +h,qkW?"9G6;U8Q'00(fUnrOn'r6)FO,@B.gnVu!Hh`62a7lR(8l"qesH,jnqJR@;g +T$S)Io_.=Kh24Me5J')?B:%(S].<3T4"3D)PPW2?Zle)Kr8&aR2fE/Soc3:Oh4.!E +o'TX6]G)hVmk*1t=)D,6j%Dh198,H#H)F^7c`cHfCsT^63W^,`/iWAs(ZPuEoen>r +dL,g0I/<*lXPJ`WfI4it86I[pVi8i^Ngr:MDKfkkV]'EGD/3UZA9=XR9MUn3>S&+8l`Cs+LRT"6Un8H7JG]p@:0/#qJ>jb8+omT(WH7U,!-C&N +R$M^j+o_q'oRj,ORRi#[F6rpZ9juP(KqR&qhkjhTj*a,UIuLNlkAUd8FeY!BLk'(H +RQeM=!#c$(,o-RP`AG;Z`oU&9fH_XB.$YNX2#(!9.P)e*(M2\EOKeWjEtK1/"ge8. +_f\?`^mUhQ/P[T9^hm)&Z,,9.!SalXm(a7A,K9n<.=bo[iMZp+P, +q"m>UlX"Y^@gn"f*P-c,la+(ZqENqgipc,r)%f7:p:ksRa^`"2s+mR#l8?V1III?t +DZF]F2uliaE'#fT&O9"mR+;Aq;-t+B(9uo0c#%;$s"?#^c-OJ/-Y3r,n]7X/r,4L" +NQ,/8^&lN^d1jLi@)BAS&,Tc2b]QJUI1:frgu/kEka6=0>Vn;]Ssbbebt@0^A4uO" +jTToBp_!Tn0JA0d++sO,Z6EFWQE76^NEWbj#riTgN/(lhBTGXqp,.WQri)3>?^6j? +pA\_m^DYRNdf3M;s*1b*E<+l?c&YZ6DAX=p6`8cMIuPg:nG +bq,,R0:+p,bZUY=Y?\A5XkKeWp6mj/Xj'mT?RrUhQ.]53!QRfhIAt/n=DV1iF=\4& +(\8i/XN`'fhD4?4)YmDMoc_ZS:PFgX@5I$2HYMA$S+?N9-[p)/b"jZ2qcQ8\a+&L$ +p^_7)p>6[H]mB5$Wu$frf7g!`H"6,hh3@TgS&jKtRI]a7W8G1Kp5S[NYFT+9[)[Q' +li.Uk*_qkGfmVc5H?[!-e'm.7]m&kWX6M>+^%^B1?e'W'cKsuT!CnLK"9^as&9J2. +!dTAc-pSt!6Bilj$AJ,uk[e@K:%-je^+?0]K6g+5[%uPc3nd_!fNQp6qg*;ef`#haobFt: +=F`a;Pu@Se"HK^dp_l?,L@jU$b+nopV:5B3E"K5dQ\k.>d\BQ<3l'!qGh&0hT)XAu +f/*E0]c"THs3HdM6,/H[)l3LTdoF92KaNG$Im+>Jpn.9N@h)8Lbjst"p/;1Ss2jC9 +\JcMo?u?*4ILQ9!%=)/=`LB!Yp3A9iGJD_q4Z_D+3'2M`4^-2Q5;K/Z',`Y8k-IL7 +G0-ag-F2kbmpP$jTRn-fi'51pUj@tVT8Dhldie>dr$,eXCAU9eGG<`^G+u?>aC"GP +PF_@hA['U#-io]3e.DTm$]$i(@4+/#jSE\/[tX4b-%PNE#pfR4'XkKX]DHI^#iH3r +T(*nh!+[>$YO__cCg!fGeeuhB+#M`m9JKcAS7aVKGrG\p!O"*L/3miR+r3H;K#0/E +(3"B.7.u4MmD0V.G[p@8gA8\/b(;tM8;\14+7TMq\@b=Jc6M73L\1qkAKo)VU3&Yd +$H^b*72Ck!?7dGS9=aG)EN`M.!K)olQA`HZ?k018j^f,W!3-#1G(EfeT>?!iGqt,u +O3%?3i4U/@DYhZ%^>lkA6h#"4L(/"72EM(s[f,+R^Ra&$)V'Q>PC]ps47M +"TDdZ:k63iMLKUeHZ]FJJE@X:lltr[fC8eKn\9$Ba;t>jPOT[/35F>4??&!$2N,oh +YoegPDp=WUr=oJ7TtkCkYSI[d`1pOE+8@,85r!jg/ffMFeB003b44i(1O)bWY*Y27 +o,EdJJ<7QRe_G)P%KQM,c7s(NpJ917m.g4!Wl61si2N45agj+?jo;1=Xoq#f +^UZ`Qqcbh#!V7BBea+Ae*=mM;Zdm\T\p:VdgEVpe2="WQa6i"\VkjS6+G/35fGBVW +bXHo1*%O:uD]QXF&(lYE$C*DYeBo`^\&^9__Ycf,'p#K8U8hlp#muH`_14K^'SSm5 +:e3QeJV5G>&uRP6CegGDMphinXFgtTYS&c/3bd?ITLmoo@D@)ke-qRd3ntfk4??8n\sCuqtM5skZ25bU81kP +%#X%Y8UOMNh38(=bP\7ap@C'*AVp(2:OS\,h.A(%M9"Z?j*kO<)5NMqF/oD-kl(cTQS?U2]^H+_>g0^mFAQ1)A_gL +[nE.\+5Q4i^,R5$G.+r?24O2"hr0'2n22Bh*<&FP,*:](qKNL"K`+!^ccJEKN0-F0 +n-=6`#lXi44_.tdnjf5m3tioi([CU)E5SU3i<&h5?7o=[iIBG%!Q!Yg_N"=T[fFX% +(@WF,\'P'5CjH;S%\]Sg*`CMbZf;%!/U/3Tjk/9TnJd-SJAldA#l)/aZ?2b2nXmVs +9M5Ecl]'CSs%O7kjPN6J2nr`TYSWr^J4p3"&cb^e5[>T_+#L=a@*?Iii4"?r'UJg` +>bOcp+I=ZP-'&5H"Wi/E_d-ZV(V&f8*c"'W>QdKcHmf$:_"gr3OcChg0i^-BkS!NZ+j(@KW'Rt09I=2u[Etnp)G^Ybun:*W]4roE0e^jol6oZ=>n5+nuT5B)mED"mDnE:BAf;hobCRp`J):j16#a5OCKfT`<#t=J`*QSUNAS/ +%\"<>58V*WjDi<0_]R-Z%W.:.6id6Rm^Q4UY*@G'r+]''WbL!@D6-c4:"6*^Ja1bf ++oit4hY;mZKP+]WTSk^CNX)mb32i$:^lCJRqR=\h/o[H_:G*)!e<''4FTt-DJ*+.W5ts6V=O^p;rXXq[.+/[H\o4]meUI*48rtmC!q= +4Xh9,.joqS"cAXM(&_#1r6jEhkr_6t7Tr9jpTLrKn +Y";5cBsB56QDBq#$l'?l;(9\(AQbcjmA@,[WkA+jg>Y>4J,"f::]@taci;&?:]J>= +:]?\^+91L_+8d5lI!T-SK902aol-](D!Ed.ciU4S^p=E[QgDf>jkN:KnNRb5a^"dUFfpU +562/-qUos0i"+/Cr("rK5N)QQ;[NBjkKu997T*KO#RUS5l2QkCDc8GdLGZsLm:B`- +6o4UV;=lq,r=4O;N;"\Y]E%Zu-OY-LJ!",g%f\Q_*C^s0';!,J0`2q<.`c>$\]o6* +RHh+$pd9`MZ(]f7C]!&0Ib)agnAjP4q4GRo#=/ZXoW[8C65JKFN?Ea9k!/H$s6U#f +##%_WNQ[GY8+`1.]W_J:]Zh=?seLN-[p'*!*WS4f6@IfQ*c'Rr4fUFSK#rdSB34J[u?&Vp@cmIL!DlM$N]a8G\S3H_Sd=XY$4SY-!K0L5eI[dWdX$>L?U%j +!<6qgd>0K66"\ikM0o%a(Z0)m3\2hoR0b\PLd<1l4JQXK'\$#EUk/u)"mg*".5&5: +O0K'D!;tfd\V>tC!e5I4n_O>WL@GChaFT5as0/$tOYuOd(4u[Ek`(5qp^a'H>`i8J +A2ddSnGf!:-^IJJ,+)`>%K+!4VWNOKWm5EM!a"Zr/5\Pnu!=`R#Uk/UGi.+W/@Eej<4>"-5 +Rpd0HSm^GD-ZWVrn-SeS/di]\Z-g6ce,_qqho5e/8ijIr(E_S-naLQ?4r1WkjT5/` +r':>_3WE#a2AQP6hp%%jQr,4UiW/`u@gtCW4r`6fAeG"]c^;Lt-ijefQLXLG`N?6G +UeO2TV>L0$O2;9MjrbYC[](=<:#-D8S:n'&F5,6KL+E5`uJ3f]LSg!!#el0rUO[+UhQ0U>K*@L +iNQWJ!'Z$=ATansg92I8D9I4NIbAdnDMI#GPJE)Jn-,gOd*a`(K6eKSML9aa#(:P+ +H?g6M1%g-iNSRBbp7&c2o)BPf)f8Z:cnpU72A@oCcGZK884XX/9$e$.*gt%tG1hR% +X'd9dd-HkQ%KD7a62sP&[%<(iV.)O^e3Pg1<6h&ao9r/-ZrPP_bIVjogT).q2:J4, +*&^NdQ?gt<%U4ZR;a5)hOk]%\NK\6Y9ibO)>n(?Dam]_C:B^/7)_^ZVI4N^oF/7uc +p1IJ#VjB=:dG$/ZhPG-9Rt!':&c="$olt^:)B;-2+*A+9WEXcDC;3p.XVZ13\ZJbA +,G3dCjNpkCU1/!VO4QYklsqoY,7OD'`e,4^XeY$>7(qSW7I56Vug`)E^a4cM$c[KXrr;n +T?PIkBk]%Gc6g)e,cdDD>ici!bKU!drp]ZnO+)TNkSKGdrq6)"It%4@?X1;=QWHEJ +pK'TUbH(C;h8KX!B(Ed1N'@/XqB,^[c:@oAotbQ*Oo@r<_gg,5pilV-s4>/lIWtSN +[ms*kGf'CUA'Rk>r0INED+??q8I'BP[jLLlej/WfC:h(6.Bg.:.ujI!,HIi@(%mfm +QB]ru@S"S%]pgHJ1G+:+<)_4C>a19Tdii&AcO"sNIOE>%@(YfBVj.QXJ6*J"H4C[#]rf/^0!]Ci"J&g>;aJDs-*,h?(@WoOQ'_L4JMDUR==I#:o(O +VgR9)OB[\7l3TaGOLg]Q0#]2XI5>Q&jl6(pj)'U!#rsoA@Zg+ftHn0O&%Y_ ++SF;@8@)#N`OG^#?06qQn=#Ce=A,ukSA9M'lWpQ(f$?)"lpMW3ICUn.,,^i\[7')^ +dm*pQ/SM&;m0q)'E@8ou)%R@A5O[\Kr4J22-K5(pGor?=LR3X=)&G7!6p!`/G)4nh +8s$<4XR(J&m5D3h7[?K(7`1?+d3j]k1`@;,rulE82UT&h$5"-e"9nm<_VsJ%&QF$j +MHWBtSC10YW+T;&:/r1"CDK^qNt9_S0UeQMa5B&6plP'd`q&uknu?GM76t4A;H4?u6:@"F +5R`p>jYh+k(*9tNcli`VOV&-@M#.n.J1;L@0>r(WWVq/2\5PR1+A/Ek4QQ+';=J`G +"S,porKggY[,m`5SYLp^=LS]RAmf,`C'7';b;rJ7XR-eRTQg1g/0+ck]Kh!H;X;+K_LGD/YOm0P5^U!-/cVA"A-)JLQ^fm^@ZeoWHoIPMT>Mc# +/=ln?Tc-8T=5>uV=srD)RC3:5pds("m[2!7h5IGE0$ldNlDOa_UH[Bc8bc)i=[IkX +\MSp]hP5ab30.^kUr`Fe$4YCJ'^:DG7Phi\*?$TIu+&:.!/l*?)a#o +0;R/Sac1+O':qV])O+!u-Y'3Vpf]/(>5.7q)IA*BY@Z=laVj(+TcTe@+K,X2;'dV& +;8be3MCDHU0+RhIpg-T"e5.cQ"A1/Uf3r5Pm4=X;=;Mgq#leV4M-?`D6+iIo\eU8*rK1@_8#D$pdOAZJ(6jjErU.@]t]n`n9t"# +5KUV5Zbcs^=o%j]f0B-;Xfq^iHf=.Y`Lh9eD`Xs9/ac>30Wb]f2+b8,DcL@mn6Z@o +hrfJP,:EDm-:Io@_Z*Y+<)6.PeV'_8'BZ#F$]'$f+V-Niiqi0$EjDo_dk`OYRR'Ts +SM"^)BZGh%K?&48a5ZOTj6^\=i9RH"m>#'Y-\i*!Shp$pnD@]Jr$q^S&T01\[X`P- +,-h=O(3&)9H/;hPfj>s>n7Jurm<"NtD>QQT_YcpnIlZ2052<[8Y6+VTRk'\[E_qpA +Eq$bub&K^"/4b(I0c)c^j1+ajq^Du8J65;<5OZZhi.&1.dI*c1O80'pa/VW#K#e-k +Ec)7n?b9hNo&#'aan?2Ph-m.o,H$M*,-P:]d$2goG!]RebEM`B:*R`\$A\j$%Cn+nieeOhYcUf5/?^kt]69=V!O'>LR#4LdJ1:gNJ6V;]Lqs.KAes;95m]fnIknU+ +KB/XfY3".Yq%NOUW<)ggY*lY5k8afkU'HEd^Z[aEkf(=om276VrMTtLJ`]$GV^!eI +?p?L[pd:jtZPRF[Np#"(#lJaPntU55IQe-X`RL%kpu?pOLr]Meg'=22*l,Nsf`kGL +gSI##pC3)O'qdL)*gDJ``$6aV'#EmI8-+U%Q0@foAIAF$&hD)6MVpW=L2='/09ek] +LaiTn"lD0J_Y4iPYse@>)g%jOa!h-1F]a;T'7Y2/]np5TZgANo_o?d[fr,&CF+PPF +?3LlOr^6H6PU%+FkVrPrs4?E5._GdsKT\MW/VEL0&q+cT5G7X/$)I4kcE1^mgb#&cB[R01MEZr:a5ElEg!; +btAun0mnWL#X9?f`a!age,nU$b^Z;NpLa=n-[Ds.s7_h$#8I+["Wu?mc-GUGd2ck! +U:]6N=Ei#;gsqgdhIWVB'9t\N2N#D`$j-"3XV<&(1@RNh?kE=TQPXje;B?mp5O1bk +*XI6;rh,G;/cSZm#k%rqncl]#SeGK%?ik8XOmP]-9MnQa=hH=K0YB%'Oge@rr9>TLd$LE)I_n^8[ +GTJB]B,C=`_LPPf$Fs\U[lieO`Vh;4]`o-KKk$L +?OdDJAaYOWhOOW1N#;,5Yq3l`QNQ^`r#9K!#['D[Po%sAi`\WS/#qm22'>-24T>N`RlMea^QD!I-dA8a=q'oh'kF)& +E`c?PMTT73:0aH^&/V(89F/0ogrmYhJC]GQFuElUQ]5]K%E]f]6Q,;"\)dj-gfH^h3Aei +[9&!?R0APsEWF%,U0!Oqi;`E-enop8ZE9SF<25>F?^*Kji`\rVMXe"r5$1Ls`B%k,$F-+*]EOFMHdaH,;pCcJ.-:FKfac#QFI*RrWfaDidW]Pe&91j>u41XQ@.="6bTNs6nMFODS*n2bY\com`Y*6oecYAmdFZjq2=#G1u +b9,%*hl3W&FmZDEQuoErI0WU@G3je@&+G'g&\\eUmf<>X;:'ad^HD5jr'=+Jf'C'7 +0)QgBMnA*;AoU:,1iaIVWc?3+(0!a#M#Tc-(m"q!LMmt'&*8>n-1Q>4LFWr8o;MhB +Il3-VSk_jI])=V^"8SK(?.WZ6NG_NJSrIKY^l&E*A$,hBK[H*6.;OtHt4FV +WlV`NqoKr+fQ;],(@^D=:?JdCJNNr4bGj`RWX&=FNaE5PK>+qK7;L'5ej6Wn:)6gP +4(O0`D9,;O5b=Uu4$b"%ikBs=!O(#AJaZPNE$GP"@SCng!5%;4c$ts_>K+=+c+?t*%!gcJd2t=]cCdnoU@HKPOCmU+br:dgK_#QD6il@dF6sPsKMDH,i +TcbqPUlb*3%:u<6+DS?L7.o[L=9/7q]dCW4+G2ThMi(2/Q3.'+-*]-gjp1+tZ@6OC +a;BWF\;^I_C$N_l;kLcqJ+.m_]q;,Br%&>(mrdj1'CURQli!F8URM"RSGpFel$MBq ++%HcIVQ:/NpN"q_XbI_$34"`X2c?/lL0r+0X'&:G-`uGEQ-sELZMk@6;$hdKhomL; +=W+0/OanQ+"8iBM`MB5u<`RW6lD1^@/?/ILoX7p9XU'FbA:qr,&jBh/JF85O'6#Zh +:*soj>ejgC67m`VHWj=>/E!#s(/Bl3_hL-m6!e9$$J+dBTWknp=XN//m_R/HB*%pcI\g,U`\,&Jl\[Ac_pq +G[WOdi=@F8q2#&7:Wpi@'&b."36X:]^QNLD58'jSuj]&,^&As)IW8nK6\-ZSF67RD6mo+MZ7C^1RI_ +_LJ>5Ng9Yr'Z,?1!F!M-.79l(X4j5b`5][&c%i#V:WNctUrgOeH;dF"i\ +BL<2[DoAG_R<<>HJIDKp+hV5"J5e)n3Zo]kr%<99V$suP"]f-I[=j#['Ab7X:WfZc +>$>.VkaIpG7f!hK8tWLIV]`d(,CMmXaW-Vq#Ok.(mLotA'TZnM^,Z%_GAUD@r'-To +!WM;ag%gU,./@?brdT80q=Z<1qI5cg6-gnOVdnK+^jOPW&"B"omC6g:3r9iIk@R9(!aC%4mD[4 +2slrKkB5C)e+;Ag]haF_kMm\4_*mA*g[3Js('(q9J?0PsbN6Z_H#C:uP!bDIYZtO= +n+6Ps^n9hNs6mgS@YEHD4OhWshpZ\Zl?<'7ffX(-%ViDb8d-4l_^D\>dA^=YIis4H +M!^B/I>WPV4r7DtJ4djWKWgu&56SG,'QQR.7r97:+UCt/Hm\2Gf2KO"5.pb$49b4B +Jep&HEYP_crAMTD\!^#Ed6Id.>L8C#YDqpnE.`%7#f;'n&<@-]$#r!Pa^uj7s)M73 +peR8Qptohe#lg"urtSoB_#JE]SKONq58k,'".Y16)>DKV?=q[\AM<[p'R_s#HuTo# +pbqdDe):+!r'.d`]Jm:6h_os\2LM%GIM+j?`-q-[IJ4uXiE(3qf,hhMcq95,)9Dk/ +)r]jf]PO+%95nr$?6LQ9B;NS[*;EkmY;JpE:p[5)o$@1tK`;VPgN`t`Hk6Em8H.YT +lCt;iXJ%6RDPX@.C9-7o>pq#]MZR;f,jMb/#@SA/C-Of@AHpqGHf+r`$`XBuY'e3t +ri+ke5iAh1>;GK1JH&XpeR>.um0kBh^e0c=/sH?i0Y\IdJH[mZ6EgHg!X9t(Gt!0b +;h>D4[T5pN5[<7/E9"oN'4V!J_<-s\ekL>ji"+e]?\Ya"Guc4;K!.'Gk(CE6pd_0< +$!dQ@9cSbBaZU3n'O1W; +b+]3gp!QFDU-Zu::&Y%[s(hE@jSsDsN4c\0^UXc%F7+Man@qWq[mS>Mf.^p1!rUg7 +K6i!ESDM=5PhA%-@jN"Y,skLSK&VmlrJE#cJ(Q%R3Q4:>#f)-4?Zr^3-&q\"M=>gD +LA5-!*]L#V_/E%"E:3GM#.G +G;['o]%]Ab\(/ac1"r[2h60fOaM+3MFu,[m9>ji8W%&F7W%S8Jq`Q.%B[C4D9*4ZD +`D.BCG9Ve[?#no3n\GA,%Xu2bg,S<8>ke%)Y;-%\6id7C7CF"-%J_JZgp]d +#Y*aWU8F[:`qA#1!<7S5(\7(Yr(eUbRC"f;PN%EaZMJh!mT[%L3+[%l`Q]JT@n`-1=1<^X2P#g%BeXn9ir6Ib;&XfJ($H +e:!]`Ii(+^4ecsHied'r"/b/lDj0ig.cYe$L>"&0s.X= +/$&/72\K5Q:i`GB^"T+n5M>T72#U_r752Whu>$kh=K'r-ue;(@[r\!1G5di6q#4j0L.la8;".9ii]l-p]DR\r70Boi%%#6OjT +=LTs3mYT8573J.U5qKgMAV7c%L]r=!=e*n'0EbcDBZ]#bT)-=s6%S>;]Peo +JNo5<_a"/!FV7(ni:TP"ogK/Es(^hG5'B+/3T,2;2dQ0;kABUjG1%l_U@=f2-mi!q +rgB-9)3t9o6%o8G"EC%bk*8D!BRr81PXHKI$\2D'i_^tRCRN6jg%*Y:*d7s[IRuc( +RP(&;bEi+`r,UK3$]%0AR:FB +[b8bGEnQ-HJ#Mi&-8(B%j>foX\H"j)tWtgkl7UCqmIg?""16R`GP"jV>keA46K'!7H!M+\(3SA%Z(=XU>7pW:^0Di3Y71/HC@Z +TNd's#iKVZ+p%^\-5'W\&,HmqT`IoKU&Kpjq3IX'=D5KB'kkq_`ZTO^a[/1\^ +]!JKkHNZAkGcX&5e=S;/)rPR6*BL>l[eJ\p?HW41%hb`uam^SLqn7(_?pEjf#-e#S +Ra4LsHJVi@4"^Lt&blm5;p[r<$U?'&k7[dW1&jm4_0uRMI,"8QOBiTBXsRH? +C!UDB\"3,.(BW?Fi!rXu)_&E%r6N<*n+4;Y4ffDo)WfFc_n#hbd]o9_:N4tm\Qs67o0A72t4))#1IED>8m*-KGOXho6- +ZX-2$1(dYM/56>fM1F[$JH#q:2"$Zk-h'?:(ZE'2icm_V%W>l&S&N<`4aku.SojRm +@G*?6R=YB>D$TS18^@eB-NX[#2$q$fc3,bM"s6Qf+Q3:M`.dftj6?]e%oda\g#`@c +"7VpGn]^[uf)=[L"`M:j@Cqn1VIS)"BG_3NK:]O+f!H@9c,N>MIU&opb$iB@ig9ltPj6-8jlq!?WMt9gq+n37&q-VCicmnN*s6DZK#l*<" +WrW,anm3B>o+"EnEEP`Is',JH[QqY.X6(bmZF*9(!7G.ek@KXC$r!Ic(7.so@R?07 +E\h-A,/fN;fjPsQk[sW1FA`JHkH.2\()=qRq[bs[ZpB.Q5RA!D0o6(XWdkAi&4=XW +?ieM/g(Di[$Gd!o"VDTfTHjP4C5R"2hQP#4-#'8*6fcRNC[f"b,l+Au'+uR<*,q2] +r/EP&Sn:0])GV]rH/QGLD\;aTcT2$A(BFH3;[W"9hE3m8CSK,"C=sIIo\pAXW-dQR +bP[c[$rA?/HP$P1I=?q&8,%7B/iUc6=pDNC'I'`MlLP#UT`bAp1SR&KG\G+C"TeYp +pKKn,Mnej0!!7YeIo+k@90'Kg%"D*lb4V7VK`BUZL'a5m9KbP7OSfXe5M2ghj$.)Z +IiSHKNmPFPob#@H+PdnKK_U%OACN@'2@4=']%+_!rnm#C+U`h-:Z1]Zq;6Ocs5Aa+ +"L-*R5W#7$7egmZU[)^C-+6'Z"4+(3R^G8a%g)8R>uqIY^f(QBMO!X +Il"eC"o*#FMLb_E4-3h7[RrYo.1!cI,UEe$jC+N5Am\+jf$^^\N-dAHh8oCMF8i:H +?kY!!Pn.`0@ln'!F8hMI8$sP0n;kU^R&9U=rriikSd*%R?`Q/L7 +5'13-RkEI,C#a;353`o-S,'0iB"a^>ai+qos&97("hYHqI;_QYO8'5]QT47A!rnBf +P%6hC$*OC0bOOCXlEHI4l/aasP$qNO?T)`P/]UT?q#p8(nM +i'5?`F6q3d5(S+*-NXfQ#,T73d0k>[IEX>f#'""K9^iB3o4.nDD?o^Hdo_11c1t^;9@=,<+q"5a ++oZgVDAV2=kI=5;\!c20d_,Ef^OFS6II'm^m5k0f]^phhrR626`uFT1D<))oK).=a?B]auM]Em">Lk5BHS/AJ;9af-9@4_>g4S8@57!e]^e4Z>(XqAOR-dE[#Ld9RGF.(. +B^<&@mD7^`ofS.mDpo%ZE;/p_Ygqer8VRs6m,A/+S]0ZDNb2+f_:b$ +DPPQo&Gj3E9X9IW&[SJMp=5s[-1,$^Gb?$Y:OC^A^@dG7,9,9/LUY&#Z?SGWiIAP: +Jaa3>8!E%\FqjpG^?=Oc)uf=$+"-PJ\qiZIIl._M(de.D4r`.Hp2.9EdrjDahDcNW +:\s2OG(.&cfr1]I$Sb(&,0PWbko^0oYfe8EB@8"/-/9\2ISe6;!rb^V!iQ00/ns%C +ap(r=iQXu2\!]Y^+658rU39Eh>h#lW:&AeVnYG9XmC9t95gOEV5[<6&r!rf-nb&-V +$T-nYo5Cl6(Q//.rjTA>3WC`bXOF/:n:0*B@K+T.9,uQaZ_=/U_EL!.M[t)MV%]>t +p]e5.g@LALm"bL2Y=ml^$/hEY`-rgn#[YaNn&F\"F8jKeGQH_&I%9feq6l4`k,1O$ +U3'L4\n\1KLAPdO.(.Dt'ZL2F)#f+t=HngOZ8#3".0ZGHEme,Wg8rBM2.a7jm'g"" +mF>WX%5d@QO*_i8":G2kkd>lTs)MG798^MQ&lE'g/.Mp(!M>JC+-e_gVHq6HT2KLj +-0r[)C&_uIcGPjR0D6*PX[t]WLd.1t5*a8SQ`@c!Y!:XnHZX56W'7%l%1JGcD0L$V +?G>;M!*b$6+r,mj86SR,MCEfsV5U8VQ)q9W^*Wm#gV1oPNM_OYr8P_oW,!08_u'j)`dH4"d6Oq/eIP.6Z.Y9KiUal +%N-eN#&gsNr"n`9b5?o,d+.;eJm%"u!4BWd]b;X38%G9ZN)N@k4V`I$6TC3+KGV*f)bl+ +]Ss(U#o3Nu!q8DfjFII7s5t/:[h4j3o'cq_o.Jl^q%NPDD-VU@d"JP4p?L=K4j0^6 +=u8pJRSFWJeVDiR)ikPJF8/#P2q\e@A[VtIjuR-n952CKL#O4 +$fDPf&]PiC\3gt:$_oT\=9'c;QHf1H>cU=+s1@OU@9,06S4uVebF7\,DS'W86+9Q9 +kbda9eH0tUJ?/pI(=7=P#VM?57#=pLHd&6d!bnS#)5E> +fB\rL>+0>-e#o<"X%-^[BU'!sc@`guk#4?4kq,#u@3Z46&7(-I=o!3-H$0jf +#[RW21&oqu-!Ja"M0O],r.OcG@OS*=:fY?(/_KAoWJI(GXX(fIq0Roq@8Sd*"=8X$ +&A1P(`0mPXQOjq]P&aT$J[ARE@K+Gs9biE_b,DpEi'6IE&dn\$J@a[!s/>r.r6M:7 +L_-Fn,4Lh18/1\(]l'FbNkR0N!4.]"da6@hjM>?NcGK,6f/njC2Io$QME[;^5*^pg +Q-$d;X?9I!/-,\\O-__:FM36B[W!6^J^4_=ROB)?-6i"s>`hg^MCBjMP#>[VFMN@\67BP!H^s8@NNpckn+i/`Nn57uj=dF6t/D/U3n7W@&:]]a1%\(<"p[8C3e +W%,A).1,)./@Y[%aBV8TF&?g9`(C59J@1K.*cP$U%]\8LR3./-#H#S1.C+XKY6Z?h +^l&8ULB6X9&[6iR5^;3LYlc3F9I:1_"YUbOi'YZV].ia0)j\O]E1,p#=qMf[8k#>p1Wpa+-Jk:*\%UQs3XrHdpr7-$+`*. +eo+&FGKLk")9+JD8P?r:5XcGrU>E?ZVj$([Tk,[6Mrf)fi:t ++HKgfZUqf1M9dC9R`,:,!'H6-,2dsh.#F@'EJo5F9#A&\TDt/b +3_n5)kQ_PQRIF"\d$%\6rU)T6Q$JG7> +Z&A,I/=lr^bB+m+ZV^G3K7=`!!W">.YOTdoVr(fgqK(kqI +`Fa$:`o6mU>XS_;L4pap;2@f$#uN-#Y,'%ES@DFX[S!i#7FnT`#7]4nZSLdJ:4$d* +m`'Nj'*Ku.4]Df>2>!?< +3klNpjru?qJV9X:o26fN[,U=m:crCn$l/5g`ti&o3%X7r_7\G2jT)eUfY\2G:dkrg +To&h0:R(@TOaN[d#;qP"SIkRE2MT7=6c7K61c0+gNFdap^BEUVW0bl="ik;=GG[Ig +P@:'P2NnU[VQcKFP(0oAs)(mc"SEkP+-W=-rql0q7":m,r1?si!SUcWqe6'Q/a0*f +I<<@+G^p/G/s,dZ"n6LV85SgG$\\O$6,CeNrWE'SG';$R-h,1N`H/mcN_b]g\8e(E +*2dTmrY]tD5@&\.-2\"mq`fG;o7NJrHY@1CiBM)VA,\jp-UKGYTnY.cM8"?-R/J4H.g7?4<.f2./@A`SK&3N!#FI/6V"^M(6R@1qDM^^*hC6O't^kO +j8p7X]TgLtOHHeRkSc"m9O>KCJngr-.ZQi",Zr&q2S-5OZ.S[&qqs'.!TL +<%4lZE\\01Hnf&YZcs$[:9$(`8N,H*p$-E"mnb3E)ec$f;q@?s6D]u"@t30c2p8? +PJGGqIm*@.N'%n$(D>m1jgbiD8i4!Wg_W;ZTm>CJOT-C%9u)6,!JXn9TVEmPjmWGB +a3E%0UPR4E!WV+==!N2kbO/@c>G]h=._ZLUX@@faOB3$5bIe<.r[5Ma3N[*/KR^9f +'>`QZ,-;p7gEG2SV!$`nkG)\;(Kd]"J +8j+:6tT*Y$u!IiD9o=K9cLBu%/Z[i;=lL^HiZqK*.=PR4g>*CV@8/DtL +FG6G_ns)iCp%l\gXmP5&UW!lU/ee5bNu4hkF%!q87jEX_$#i5_Wu3%t?2hUbJ##1*47&ol1ZuO!_Z?G\^r$BU +]0'?V5KMsFJNWF_ho1:!(fCp.PE1CBpk\F*CbZ';O&=moLqS*`\:Ea9l9:L.h+Al` +9+F9)TX\V)@g/FoTNr>A&]qpkEQB=2R=o!),!YRh=?t@]M5"=<1i/Go46QN;#<0!J +m6V5#FPJF=OmkSQgE?EhRf\&[s7te!FR0BZrnkA*re+VOI8ASjS#6"]Y.pp+%SOjU +_L=-InD@t-enS&^.AA5DB)GGN;#!PI3(":5r_C!hHX[!UIsAhck`]$JW<2'3EQY$V +$NU0-_jep7?8m$]9UuTu-_m3#=MSLeGZcb.<@ug&gJn[t!nT"OkQY;MDI:aJ](49r +3Uf]I/hG%bA=nm`k@<9B`.DR7Qs)b!LB;(1_>UeQ&IJJjs5Uh%e2%THL/D/:BV(AK +e2l\)1/AF2&'t,f(AI.jh.h*pT@qI7\'UjgBS@FRp:J7,-T_1e[W=SAT%D*W]5cTG +-]Gm6>>U4!YK4\End0RbM6L>^U'F0d!#ah%*4mN"^AS6Y( +Tq\V=WN5UO'cFRs@sr36B:H3A7mGQUSS;e-'AI9^cGI>'"P!Y7ImfHY8^[oJ-7R3: +A@]'0Q][D"&=0&Z<)ce:oM)1DD/96gJ!0ip(3!8Z\ZW'keCU;#2ICQM^)?:P*q$]Z +m=.&Tr-!D]h>,I-Q\XI(*PK!EI!p;0c,K[?aj0MJk0r;cD1D5U[rU"r?F8eBL3-N6 +K+#$n]NUb&:,)8o7OCf'#*tta2R*61"G!`W>;R_V9X@qHs3i-?j)%*sZ9]2LagX#f +2Q#)d%WGhWnEH&+Jb^o3)QaOoLR_//4Jl$a\Tl`%7a6=Dr!E2f_]Sp*,.Xc%XC!14 +#Mj=kcSIdliMFlt4FnU_F`,aa!B`kCG:4Xn$!9I2JcrkZr7;==YR?+Xi!tnma=`C\P]?.An37T? +!rge,-YRX]0C*&RLG$TFd39cp%pd+.r8g#?lUOlGR@'$8:Z`2#;.=A"X>p8]U(ZjJYU0gn0+Sdrt +`\#`JZ[1#'^D^otD-mftk5:"\"9/VnFh@jM_Yco>a&#Cd9ele""UadpIU:Zbrlmg` +CGg**3k#$PI<*CkfO=,`ZF3dORk'G?-#H?W7^0P=%f^h=-O23q)+AIO>-&^AKC&f1QC'E143DnYH)I60Umnq2C3If.[iOCUXCV"Q0@[Wj#di2fbE\P#[IgF(&67`4iPXKNZ0ARg$3^ +bsi%$+J<(r5RnuK-9%sKG@NZB'Y>EWag10X*)=s3\qJ%qSjiL^j!DV5_G%K&1Q23e +Mt9t6?K@[5`X\9N4SFu1%t$r42fj/0lONo#1&(?Y_uUF@fDP2q,s +"QM$EiBK(!/7JX9r"0\MIZXV53!_;%r""'e5p%V97Fo;+s24so*crh>RJ).Ar:83P +Ik(oJO-787L#L@nn>?BgD/,qNNB7)c[.6j7nDE,K4R`1]$^eh]hZ"bs50Ct)s1Q*@8";WInbXC%FeU4#49OZ.%8c1rHjE'Xeg-ts?lm-#4Q5l\ +]---uj?<'sDm^53W^V*"Z,9W?W1,p7V:>INAsCh?ntp-;HsX0cPcU(Cq4_?1X?2Q5 +Y3WOZVW5F+q^sq$Bu^Z$FW@0ZIF,Co]#K3FeF+7ZIJn+m!0)aF-H&Q(_A0)U3`]a&W4=gkg:%IRq73Tl&, +$6T?nMk"k63=-6`LnL956`91?R@\fOH/F[.f@h2C`cBDT]i@j*f[K6UR^SjG>&1&* +EZV7TG\>T5:9GV5fARC`*@WlMDnH3N/#8tJ"fb]`jac-/"rsG)JSKK736\UWH=?WZ +\8kh`29MuE$6.2ba8SorT/X9HXmPH*c:sc=T3';)X(*2u6:V$X!Ak)/.GuMrH.G9A +T.r^&CgZnI0\o$`m!LPt!Wc]O8943oMH_st`@STB7adbLHZb0FAt +`5a#&P/6[u+4gSlc4R5hO3lU?6P':j>aSiUa2HIW?l2b$Fr=A]`qP^-GScbL;=0A3 +ah^rdVZk]d&(CLE.e,3cQ2io6f1=c!-3'g(I]&IE"8%I7UU&dQ)78T3Xni(6Bc;): +De!a0p>cbE\LlF;a#!#PNDf-KrKBJp0GOjf:jh4./GS7h'RsT7(SC?>-@52Ma'I(I +B4TL,#0%NA9gK?b#l0i>UEg#a8a>j6U#9W-ig:51Vs#@d5Rbea4QeRhX$NA,RP+@r +airYFS@7fZ=VOe(l*sJDTok:dr8kPc'#Cf>5:"r#'Q+E`!o=L8ofird-+bCbh'&b2 +Qk]QkBC+MYr'b9%kZ*fHU]@r_D&b7o`ls/D$_I,4__%GQ.>q])Hqr@"1NrMobrt1X +cG1f8pImY31XHV.4J$g2!V;+WY'Z7Od:]?^?'dZc<4!/Y"8oZq(\;TUSh2)Efo11, +6f.L/LBE6>KFmf!0c)q:>_NjtfiSddi.(dYrl9.irM!N*i%ciMd,@mT']Wj*lk^\Jl&^UeSL,S#lG1=PV8%3G1J'>5!k"kgi;h65E +s'0)gnNV"W#Q3K4l]U3HB"ODd^^l:k%s)MO/^FGYhm9As*9GiGr7Z.Adk%W0b3X:_ +%Eo>TLU-re^'Wr:B_&,"DbgK9NI[M\c5Zd*Qe7(IeOk"QqS/D"ai@F;d?S5JpDQK5 +F$:E4]X)eQZrr06XJ"@#5Hp[afH([r#(^(Z+T0pJK9fb"XY\1ib]>[(?N'0`X:k?4 +K)?Mda$T,\KTeaTPg3eJ(W9C]_3I?#[1m3n`7>FYCIcLS!O?)$jF;Q=5m1","dgG-GfG4riBB+B +!n_-cN.>)\[[`e<#Y9^u+TeCpOTjWaDUhU\BT4amjd-WGIA7 +OUR)cpaCti1a!A>aV%5H#8*@a.^H&),XWhh1`]r!/m7Y\:iM +SU:.2q*"FKO+!h9[rV]MO"7Qn&aKB]?p#9r#R-;tGaoD&:'>'j_8K;$3=Q+Sdkh+C +BEYhTj6jujQO+#?hb+^jkYBM*Zk@n-K).^iScJQg+ib9^?hNg!n8=d*=f.R;Io&?3 +o(Qgq)?5?eXbo]VI*r#c(QBkf7mR(KgIX(-o&&7ET]W0CBBjsHWXJSbf":P:ht`m6 +5d13Ig/0s?7SX(*b``S=\@m[mT+:e$Y"?u(j*sPt!6Cf&5P4X@c7ZE%H^WZ6Zf8;D +qoq*s#oJb_IjaX<:AVF55$m$Z(%[!]2krQ3beFX[pr1DWs2Wrd#RK0d+9J2>#aeaY +j,-@AjO2K:!%'SFWV5hW=, +d2C>/;%>#TNBeZ/-i8KR;B\djaA)k=j8S/9Ii6J&%PEsI4(%&EpU>\K!>%FkbAA6\ +2h;mX`!e)r;sM>]HBH)FWb6#3KD*o$jc+FpZ[K'(;Iq).CSs&8H%"`4Gk7Z?Fk,V[ +T56er'nli0"!raicQ:iYhP>t;X(?-n?GD=Bh +iEPM^"L:e_)iL.6ple-jCFDS\oT(SSsNWd&Rn]&s)kgdf"[@UXQG/f!ZXV\&-:,Z1t.fn-b0]U5:QU-%nFL'QKZ5C +jO9NMPl6K3Dj2CE5MY]*nQtn/Hh^SHTZfKT@/IWVVf.o_Dp[siQ^p=f2M/+@$ZL#Kofqm3;=kb" +qWWoD+FgGXgu1^Un0'a[GTE>B=EK7`biI1LP`GZ/P8th\'TdemNq*^YAq0l^WW*6\pB!+7JLNW6\0S7MJ$$fn4>^N!D8Ti*2-:efb)3s22$jYJXID +M#Wh+=TG@3cD@(UkiSE?idYVrXB>Ki6]hXjkMtPoG-=qGiITmTM4XOSDLEu5Hg?1$ +s"=L'rh#36O*0(!JX%#Nc&DQKZ!33)Qc!$,rcoSIlWHroeTjA#I*40`osDjQ=Ah&TH8U642(-kk#$K*A6Y=i8R:S?IPEE[B0ORZ_F0C +!51WD,:]R6MTP7%F9T0&r=,_!62M@Xr:0b5W]n1H>hE^( +2im^bJOXA6CY"sXE'DhOahcPBQ$[\q4$?h1"7724KDC)A`n".RHem[9k'o?,!q>:m +IX'#MBlCmr6i@NI"-.nhXD2q?+9O's;GNE3\Oe^WCI<-=!XfUt@M%fH>N\EiL]F1_ +]7A*lBr#Ah9e_W;Kp2I(;#r>7:>t`#O'kB!pt)eHk"RbDq;-'#?HhRpu@s ++APIP=&N??NIXoo?&l!Q"On'Sj2>+W"YIOd=TOG1nGd!HQhpaR3DP$P\sA!EJR@l1 ++TFYhm/?XV"Sr)j=n1)[)qL*GPqInul\qG^/ +h"6&9)Bp:=RY^c/K5Ndc!%OlFf;SN[o'^,;n&WJq"!Slos#^'N/W\Z4--lp:P/lZ5 +gKT(fh"Leu$P@@TG'X$!INNY4EmXltc_h<*H&;Gjffqmdmkj#q??bT/SPa_2MJg9s +8AX0er=@TmLSlP"ro#8fIujV^l"M9%RKEE/8VZ+A$B`kimN*O$Yk-2XJ$*^AqB)%( +aA+,ZR![1@jYp!19]LT^36:!P*=)1go:bIkqi!(R%`F +rf?cp,@FDMn9TDOaLk(GiMMS7A\Z +gap%FMD!c"Hb8@r.JX.':K[=K.fi4!]N:"'&uq_&\s?59V):e5?VB^b3W26E=3@Oo +@WnrB<>>%OV;bgX]Q[1L5Qi(We.;Y?]"3S'!BS7#'ZU'ao"R@Gh4C(I=Fb8rFi>0t'=]7?Iq*pWs2U%' +"B'b<5kr)qK6F$4HPW!]Ul"Lo?A" +mQO6f=V(U:5J%J1gCLDQ"B;/+&5Z/BRb>rl4:h@>G:9-`s*-"?rg-O_s3d%EP03fC +07/1u]E!"D`kM^[uHZRg@@gHiOCRi +OI7OE-F/bj6i^EkT7,4'Jbjj]YO0r?qP!\N9&Armh?a>[HmFdIh<%`[YeKXfZi8e7 +g,fXSmD"12olo3apY:aP]_-L')]/TBl@3676JUHIrqH6u=SWI]nokp?>kn$8Yh)A+ +5N6mH#2CX,J8["&+"%4V7)13u!hmt#,oiM46mb:KnSi:G#fS1/3;c(ZWTd5kLAE!^ +c_LQjs1P`]VAB74qn"qDK&Og:BED?G!<1A0:WNKaid[R0T.L/Ds1KP)4ii,3mdgJH +>8?1K76ZHk^MM[NJ]`IA-Q7Ws?RZhgrYP#3')<'s6u`GU?i\DH!WRfMl%O'JfPciN +al]5!rVqSC]g1XT0\m3qVUe*^!1nXZklo6:+^&to5RC3cFT^;jhUr`K*e +)c(P?7T*EJmN.%jbJDGO5KZ4R?Hp&`;#GtGP@XGUjM0&%gLZnMo'c7$_D"ads$sp* +\uD'7(4MW+/S6Q8S;j)GYNHoF"Xb*0N<#(7!.pq`\)`lV6IKmVKLPO.l`\9<0eWnj"8rLXIta4.;-3Qe7L/Ol'uH^ilu'8 +1[6tQEma]ZX_*t/Kftau704@ki&h6IB/hp(TRcM>F0YS:"j-*a4@(E%d=13HWH:Nj1VHCJ(='+;PZGum(ZhA:-oA/Q_G?aCjJc>o/!N9qU +47B0KQ+5@W!9LDXQ2dLO2oNep@i^-anN05BSro[i1doP0"Qs:2:Wsn!%>Dj>O`:pr +1\61'+3YfMk3nD#VKCoa%KoZC!.=;_`dZhF1]dr)24=:_,JZ8"pJ+Wdqmue$>:4ib +$$)mB\6Co[e;W/(-;WCQPZ]mOR&"5JADA:oq30ffFH;L#+3DfaG^Gh#TDG1B5>KN- +"*E5*F6#IhKHr)>WW=uXd!>4%WQrSP1VB6[r\gVCikk15f`n2 +in&U"a?J_*NY?H4W`St&tg.gXf+95C[&8L +HJX,N*e-Irj2's20QXn^?rH@0^HQ5TJ-=TO73cRH*_bIX>P9 +C76dppak-1>t9*u[]2r_!qPBWXB++>/-r;1SSEbIDn%es=CON7aI4Wrr$M?W=h+.DCIZ*e->(E0g4t-E(b*<7hbBEdW,QfNA +*u+`Eomg3"Ps9P7Qce3$_[T.&pDEQfBKlWr4ork'28A1$fa_mF^^/kZLj"N0s8/bg`POo5BA3rZ5N@6T^IF66s-X.;j*s0* +7[d7D3=6>u[>IFjhn&cZGuZEp0dd,@cqU,tFl?8a?99e5m$b+JJ/WU#%aqObq1 +Y3s5>hhm+'s36c0MZ:R*ruf*/q_h)RcG!pCmpU^&!!b#g?\r!R@[>!3H(b>9J70Tj +L`7-Jo; +"e]sLrhh*?p]pUZ>reLZS4j($(CCf\&*H[p7)2OOHJPkl06#`$>`aR?-T*d;0@W+G +*LA'B-"1XtE)bIR.e71iV&rP`XoL#jQ9-222h?/pZ!dgaAP()9Xb+X>]Y?U?>T^."?;>roQR:`[bu-X +#CUBM+?l+reUB9bOb%6p\sXk:R4&k):l9-Z6_!-`<@$MUn@R9S?4IP=E!"kOd8A4> +SH;([jFcgta;VJe$-mO&X'OAg/9+5h^jeH^]X3>6:W;3u8>$.]_i5*&:Q,7&4Hmqf +U[l0,^\Ij/I'881!$D@Bq*FkDB@3?DTEX9QE4,N#I-nJh^->r75$Vcdk3DcAEb_<[ +)hSnKl5fEKL[:l="3uDm'QSRPSd^*fi/a(oocXfXONm4^b.eJ$6T_Ndedq>l&J/(N +T^im,#%Y@7C%8C[lP0A2m=83gD`8(c9uDqfI?%`L4Wo.SStVg$R/0#e66]hpIj@&U +0Z$?Ca!\XmIqfT%,pr[rn2#H_!b<$*55rS%rn%>bLuKc0@!IhX*1DKc!<;pW/\j2+ +k3P"$s.-_U]6_]$*4,!kS)k;qq4pkKArqrQ7/eE-q5X\:@".U[684#/s.'`NIoIm; +UCeX0^CpqN,bb=p4[o'C_5Bs^6uVrjaQ?9;4uN7fp-8AF3a`>gR8E4h<*F\+QO\W: +L4<-_s6]RmYf/g*XoJ?QOCpcD?&JECnS#[]L$nei1-UTm,ZitX<])$gs2Xo"`(i09 +!"RHT8dZbM]9M4g@p=ZNAhL`f$*HPsV#:"F@bB!6HL]fnn2t#%5lOjd%a0gS2@"Kn +9CSi((>MU("Jq(PJ@tT-:OO8Kcg)p6JErNg^?kpr5G,^Hq#g2;BR.Dn+FgKj/!d6A +q7NASikLnCeA?fPOB:sjQ+S%YJ"Y_'V>efPfP0Lu6iaRoi-u+1hmaLHk,5$Jo,5G4 +(;:V%;ktE6rr_3ZKPgfF/1-?,.2*(e!(bbp8_Z9;\)9-A#s1*5&u7\gN)tN?hZQB% +YbW+u!el^e+-_BTP@7$6YY,Nm"=0bSkr=NMN22i)P`,*bOrP^>VSlJb\f)cc\*Um) +VXj1)rp:s40Em!iCY-ah)h9T_Q07D[9mOBjd1j)TZ.q0qciT`f2B"ZH' ++3dT94IoCM)spSe;dQXr!0)N4rQbZIdhVr_61>D]j!K_8id*bK2_\pH3a8Dj'0&R(W^gB`d!oi2E`u>no"ZE +RoKB9pP.##jsNCs!+^`SC>?Lko!:-$FtU]cL=qgpk:jSPcZ'nXmB_5t7>5qaj51pd +T%[QLs3%PBf/psD+85^)[.fWJ]mmaG5lUfTE=_,+!WUUSJ1fbe+(bJ&(>r?ba+s&S +DReS2dM?6`p1W^g@%k+qH^(!P,q$$]M7A$DMK"o)Ll?igp(VpQEM!eDId)H&+0GR2 +q%&DV*IR$hqD["&5lSF@pH./Z!Okk7rE_&=PDKLKns9e^q30ff^nr3(arf^51_%%? +6QYPIa<)coY$pfaa?S`)#mJ0FUL!n\7\XD?fQ[WFIit&IAmkNY-ICbefDetKGF8.< +!V(0,*Uo0s#fJ*1'5;8<*1[R.0F1"=ktl=W#[Mrk;IB_,n$1h%"s1r%\`mSYVB*-\ +-KrF?#5JDp^Ks#(nlLa(/q,d!Em0Rdq-f+\F-;2.a$`GS*\s^"uaLc8AhY*!@ +^5uo5h;0D7]N.&gDs'=lAn_8DY>u8srZ@tBJ6G150auM`TJHbW:Km-a2TIX;$g-tQ +HbFKLml^8mfS_g8khWm.#\#t5ia.YWQ4r[$lMlM3:,Timq&cIGLk'N'>=rf#-T:Sd +NGOiC%X7WZ709tP3on2B12ctOF.&!C4A@FAR.E9Q]Zrt0OlN.;9UoJ88uVcZPqW,l +(4c2;olT="H<5R3s7]7^rNY[c7OnG$l(Y+&GJH);N+;(hG;]o[\`QW-@C19]Wpt.mfYt^k' +8U:+L^G$VZmGJc_aabf)(P]VN_KNl,'+aXI&HP?d8lWlhp[cgR;2Pr^2/=Jc\6t`=8N\N +24FPugH4Ald&@M?J$8H.??Q$^\P(8a^75Q2)?J2B5RVN5R6nN9)@i2YK4kVmri]o' +\/@[7ofm9oido(IG;8H]a5$1G8j!UD>_M"G@6ZghhF^1]aln%/^6qc;$OIlNj8V(\ +%o#R&,6;,l[h\CKqHW>m-cbUb"!.^%lbk<=67tAF'D"P)`.e="r*P8abH^`Dr*SBEg&:SILN6*PGXsQrXRrgRbos_[1)lQj9O`fFH;);+TLE)Ij>XDNJ^&AfSL2at1fI,PCBB0TE +](3n>^(_TT$D6,O4P@@Ho,IKMbSu[PI\gWZ?m:-t?Y:9c=oe%V5lL.p:O0W0eGfR< +I+SCVbRNs1)h!SC!RspA)1@)]u6qNcHCZNPcBS@1G( +j!];lGIMY=j'3";<[*)Ppu@Q2ik8k-GtDj[hQ3YN%^fq>qTB)G:>WDF%jt+c!M3%K +c_k)=r_WOM*]l[e=&nS*^@A:t-d06pj!Y]C#D)9"G7uBO2G3C[J@5H'B?1$d1'3%5 +!0Zhc?&G$d_#-dqElt`tm,U^kY)1o`h0)e;E[[a; +56&>%g6KopBq;u49t]u_ +o=5^)!3RRY!Iq[^r:[)aqm5:ScglS!b/+,Xo3(UuT9TB=Z\lUN!W;mbN(X0Fr*O?N +J%Xol3o>0rXZ\LC2mpVgdd1skWg]A-6mpF!M;Ai+C@Ai@nfG:=n+?-q\F"?IoDSMT +BA3raT.Km!cC_5`pUnc>G.?Gsm_"_hqnA#;kakSUiKMFAD&fA6#CP+]J:0E<*$rT( +E7d5RncqGZPZW/Ej<'FWcj5HHVi]t]DZk1C&*cn4&-4p5I^%t49>L&]NK6ur^\'f/ +#B^LdP:7'&!;o0$PGA-OH)J32)tSib;7s%*Rq +#QJR=PLTV)c.JV6PklJoGLm=KrSM;ihUMS`IqSk17/d1O5OnadIA[/h6hd_$4WsbN +TU$46!:pm"+6F+TbR8'ad>od_(B_qb;u6YQ$of5g$=LpckeW!Y#2R7'=)lkn`fc +!s:Vqk!Z:Tq!b,D$Ue8_+QDGp[\L!rLmrr#JH)>B:ITHCs-^BYqL:^W.MkJXq\"^8 +qOrj;5eE^NCT&mi>*/d/M#mNMjU^KdjA5^BGeJ54q0P<^HU$Ba5!AZi+7B7QDZ-+3 +\9?8b!r,2,8)RI$?h::o]526@*s%lhPe_Q3!Lt/t$:lT4UYV-F#g@dqqLb/SmLT(VEJJ8heW34q=CT`)?53]`rBs$T^.+,^[XO[18#j7nrP&b?H\BW +!b/4k6i`foNqnt@L]G!u+qe!P7^q=MB.h.VUj,-Z]tL$>G"QKJIM2+la@*Mbs'^@- +ZkEa^LO@j*pQjFA@CNGnCi$^peRiDR/1YTg7K@T4DDmK)Ckr0*:l;GXe(e,M]2dcU +"C)r)K.h%QT_ +m(H*J[ju\VfR=>C5kk2[F.h[qNc+`I04a4+#^H@q(;,ulqXuso]6u^#CXinUodG"P3NZfhMKC0nAJ8Xo'pcf?2:D3]J@jqA%@_1V"piIe40gCBg+6[P6n#HB] +C]87eJ)c#@;\m22K`](>^Mf#"5JB0=5==4'D,I?id\W9!rS+? +r9b'l&+[c=ld5[_qOkjhq2]p'jq?q"bZ)*]B +XVA_c!OW$O;,u=0;EtTireL@M>Q9?R]>_O;qZ"I($fq?P!T`_Hr"u)7"PimY&Kha\ +mRj_fINeKbn!Bb5^&jFB<,DbeJAg"+7ZoIC^1mkTC.L6G!s)ScrO":Zjsd.WF#> +/@p>=PZfXcUNs7\<2PYrF]O-c^Vs6&D:BQYLC +LkYWRrrMl&ho0Zh[qsV)Z[();=97c#(.8/ZOm'Tgi.QlV%]j(A6oQVfDDf,`aDF6H +S<^Q:X#i95*SLRB3YUJs5M+=jDBK!+^YMQ^H8#9LTCG.6 +rsW<=pqDW9=D0f6'9oY@>s/7b%!;Q#&i;*Sutg +.7>1eU-X+a1Wj&]+R8Mer_WMks3guYp`I:AhDLL=q_A&;%R'Lb)uh]js7`r%\/34G +_Y>3+L%q\IT6]JsY5H%&5c=T//O2*FD<)tXED?YdJ\2Ql&+o_5OP6E@OG^C)OVR@M +*jg,E\B1Kb+FbcLH5_5Zi*23Gfsm`-e5rK*JGT;U/8nORVakuabtZXmrkE7-._nEp +!Tkd#GMP+r;#F["/a$]o8Q"Bf.[uj>j+N@BqsKR*+8o!u5@rG-;"C^T'E;"p63)*. +eGj7K>Q:T0/(FUg$/`T!EW&drs(Cd%cR+B+RJKBEO +%?^ct`46V)Cr.QbJ'>Ng%.Y;9(QuZ6['5eJV6O,SM@N71_oclq:sq=dS^\BNP@(cJ +f-f;,1]@9+naXd"FD[^6hcO6'D:JS-n(jTW4J*_o_"UVj8oFjZ7H$aq!%XT_5o4K] +B"U;b5e[f:QYd2-^dVIa0G2Yd^Z<$V4`?32=q]bn,pM\^[m^ECE-&=3HRcTiiraMg +!d]BT3PbNJHndi+jo\C\?i9qhq(G1/YJe6Nf90J?^KcVf]#VYdKE&\KUu(eEIZd]e +01,kQ,C$1b,.J68koHF]jd1#i,_PXmnEGUZ&H5=6'>(6rDB^9` +"pJF"%eKh\Ij`QWgGBf"oXYciG1is$[N"arCcq3)B/(![$Z?_h>NOph/[Jc=#Uj]= +/%hq:cIpjWk!O44*arf6oqKAXh$:PnYoW[G4>EVd=$aPWeaU(Mo?URpbt'*8-Ys&B!I +s+DOV!i!`O^7G:VquO!J(As5;fQQT\YIX`&El-QK!O[jV<^DHPO;A#>5[h-.gr0k% +?>u!Y5996+A:$PYbGD\RWQfNo%9`bOs#:'(0+PdqgBkEs&.5MW_jj\B1Rj%^?NU#o +.%r4i`[J%M@hU_PY,bT!Z%a7 +62oC6+jC-$<0)B*dk90%i6a*+s*]3ZhoGiuq^id[:BYe;:BCKJr"FNF.O=5[s2"^\ +!\bB#$G2ROI^L^hWeujQde8h$oGFTJeQ#i&l[LTQQ2g0@*Bm'bn,E7KrVe5Gr9t[- +IXDI>-PHM6\"75G!H8BVUChHh@D4nn6pMP/9nCJ$'o`3^!lY#X78!jU^MHQH:1T]. +n]cpOp(Y&!PMVjen'<;q_rWN?GU5\tdG:]n&HZ;NXSG0B%+P;6!l4;as/Gpqs5B/2 +!:UEr^VjKI?4'*'KA`0X?R\YrR] +)E!Egh4iB.j>JZ@M#dQ:M#_EN[13df>hE_V[k=\nl_EkE`m/(0kMn#JmJ54rR/6j/ +GIr?O`"q8QYECQOaQV:>[g0_kDn."jT5QW;]NY>Wc@.oZI;Z/N2HF]+Qbi^"`J*@= +/G4K>3"bb@ph^+.dst$]?E\aSP^OnqU"dQaG@VlT*PLWec?qnV]A!E34s'B^Mst1g +XW6g-2lh_8g(C6+aJBjL(l!Gc0`Nga57k,]KH6g2-Y59s*Y!D!"UKC/_#Id&:[(>6 +/34Lli8c7H%=F:fFJ@Ce1]eNoGDPthJ+QtFj'gpA7'HXhO\&oIqL8S()XpF`('#S0 +p`Ej:2XS$AV4:g;t,5u$+VPsjoLKp=mFT:@_@W;epk5^u6bA'dmg +8-J.A;]#PX%__F's/X-SWY%8nU(P0l!VcG_WW0Zg\c;#Gk)jQNJ(1lNZEI[6h^h;: +5Z)X90>dRXE);'L+FJuqI"i"@O@aJUZ.6#WnWl1K!U/SLIMhQ+KnCpSr,Y1'KoE60 +rd=D8s/`B:3psEV5f6'1g`G4RO/u1UT2eR_qhA-X>>Gn-LNb7t*oh_uri=m-l@/fp +chP9=Ai'Fa$NV."$M^79+39h8Z6g&L5YAd`)hF61"VF#u+45MuIulnf^Jd/1eGkL- +IHECVrY1GYRK$[n%oE9RI=bFKT*Yhq;%_i-^*d^HRq>SB8SEpZJi;$i.\oo^Pf`4C(UPY/S^G&cf[OFh&1uKY?rsH-Z)#n*l*- +HPl-+0N4Woli/3bK@I&A;#`J3lh%?j>%&EqJD:O>66dTh?-IsfN>_g0ATTo +0"_*ILeef(kl3;/!;cEoeU+^`rLO.YFJV;dX99,#f)dAEXpKtUu)0A6BSp&>j(G9[*D +7OEZl1>VX'1qhrlJM5jT&&%[rC>YMJ"Fe\sUK%5a(q'NgeKC6p#RDcu6+5!_D"0?JGpWFC@MLWNO%"7T`:hj@"+MW!0ECMrDN.c^o[5bCVGc- +62i>kY1nR+p#(,O'V_"+=oeCT-N1;C2f#3$TL;2/&iD/sIp)YFEF)RNf!X+Ph/1teGAMZ*Rt:G]b?`A?HXJs#0V4U'-Uhl& +gH"X/k-+Wl#JMe$o*;uLpc!`/QW*IV;6UEN)aX'j(=NZL368h([4Pr@b7p*@.qBk9 +nI:^NhDqR5aW#hU&NBY`J&uDqa<(ISBDd6Zra[6SYHht\0>`i7I?Q)lr\_m +<8P(6GNZPm9>feTnbEA5_#/4gTmXAZ!?WDRT:amM!C&^DIs_Q1"FLK)rt>8-*;K0e +s7kd[?LL$1GOFWu)t8)hH+5We?C>[`NAm29i;1:jT`7gI+Te-8X+;A"0Hu&tuJFIS+\l))%iN3nV*rO"^ +C,lk0Y=6Tib3<2`\=AkNP-Y%>1`r&p,Ed.>Opm-ZrnWZ?i3dRlL,G`@=f5BKb3'(P +#3DR^]Atg,H`;JK$?Q;?mi"U0IV=uf#HMLgS,7F:hK]RV1-Dphc_oO$kM+m$mJM&X +puQ,^[sQM0*lNQA(IJc3Fq6gUh*/U1mJM":rUVVDq>P?@q>"Lkj4X9aPP+)cqk:`E +='N3r@%EXMgA/:SoQ2rF"o_:&`oHJ!YdE";ACD&+eU>$BpWrsi,l`p1r*L!9cj5I& +mmMOCp^$]aG2\5GPJTMoUn??*DcZVeFL]F7ZlcN2'(_1$Y6dU%,>n1`^-:eTAO"!m +!JGlb&`4uJooI&nj$4TQOjFn@ro"/+*1Vt@VP7MSV%7s="b]]]et8<1I.:oXI+At# +Tsu=WGhVnm'nLA97u6RCV*F"$F`9jJ!,m,)':SOr-9U,k[qD4#JC?5jZ]&&3ULs+b +d4hTP%;RhR)WdWP]cnpZFCJ0J&)A"g%F +;Q>K%RSg2)gn9F0G2-Vu!:VaV_h>ugBeiGO$o:BO["/>\F9-J!e@R=5n^6%fYJ`8K +9Mqh*r-/tWMLUn2=CWDgc2[Jlk40TJYfh`U^W=Ya:kp\,rJNq>#EUhQK!9G"#^I2< +mY16Wfi_#Z&^W_'[6W.#%XIRDEDZd+4/r7(W;lJb2=Q)FcCt5mDuou+csR!dB;"__ +"r7:!j]IJ*8o/+'>hi8?`l8&5LVs)HH:\PEDcPn3LFS75Rc_2Y]/eIi[?-7^^^>-Ik#s5s(YpTr*SurpI;q;s3h]kl@1,^^@ctBcg(1Q +=?T*nrm^su7l0J]/l6qMV4a4[rl6tcD&`Q:&"hd0!9>6-D_h;^q>5Jn5@=>Ls0neJ +.bPj-s*@g5jaUI#SMq6ZK\r;7W;iKm9oT!Wn#Se+s7T6L/g^W*61@*(rVtimrHdo6 +IaA&8\,M/4mNjUf9j;mq]mpaV:]5ck.L5e,>ppetP`Hj\CHgNf5Mk]ihQ)2f3q2As'[Z/q./k+a?Q=!Zs.TE>rbn)]^@Hc):&SEo +Xl-#e3!Ec""@Q:r0ig$J.;(T=r8IDD;-lq),(LeEr*P7OE`$Nt/1(93;ubVElo4nK +N-fVHc%i4%Is5MZ\GGf'c8\]*Ag5M^9Ernh(#I(i\fq7iN&]sQ?3O'23Ib?4[D +IiEjC's8+6>.q3+/Ai0=be$;!K1R*T^&EeYUBB&\Hj,T#namRJEt)A7Cq5, +'DFf'X&elA6bZUgTYibM0m2P!ZB'\[UJUP'KE1S@rWfD% +B`p-h&&r74Y5G32YpiJ]"1&P(bC0fc"l=n6rL`9U!BpE(,Q8d_rjUWOWW,\iYPbSg +pOW5W;Z-P_p5T)$;HY+:?`Fs=JH,An_]XGoH:4MKU]0`'h_YZ%\!Y/9oRAe$YSWsb +TW_:EnNZFdqV(sGr2WO:htitf+"Dcqq;:64i&CW_k+;lDrW1XK79pCn+"9`*+NG23 +#=RYGOq`iaFaLO;0,Jots4hoQbG"or"IDK)kV)'"hH?Hej?E`p7j,#Q%^3f[D!M$e +MMG2LD?p)H/IIR4Ym#Z-"M(:6H!XKV<]*s6g8dU>9iXtqp!m?4[-`:+Ii[UBO<-lE +Q/GEVc`?oOhtXrF!b_p;s!)CsnGh,?V"gR1j+"gUd]8r1l/i#]E=q7R[^%-d2>jh& +BEV0-1\H7Q7I]j\^25q,!Vtnf,.dPBMgph<3O91cY801J^>82cb`b]5*?*RKDCC +s72uGs!oT![Z[""MLXaT\I.DIJ#Rb,ei\*:s62iLq9e^-;>^T3e_^[OMG5aIi677-3=j!j^hCnUHq*s2AC_he1bkj+!&):*Z7Yb@aMk +=UVbGf4CKF=r'H5`Ir"Mj'0V/mH1_J+"`R@Q?3:0(g; +`;_(?s0-Q(?lsNZL[54mSBIiPJ-\unh^PE#H]gRH(s`@nALkd++"RZ."8V7^"8"<$ +pq,JQoiWD-`i/g1Y47ll/FFO>Hf+T?,9Ytb&,VJqq#f"eAMQlJcbS?9:N'qIs5Ud% +;"bRors/RJf)itJYq]=%'Q?Q4H\10ks-!ZhM#Sh`%tsbunYLL&s-V7iMZ1)F=Y-?3 +J,PI04.oR_5m0IO:]2r5%WiRtj=cc-aQ.srIq3HtO=LQ5rgFFtr)*Ii;4XCZr.Fq; +,tIP0`GCtIBEX^6&b%+N[-F4g:eqYBHMt%TN7'4AXY.9[_N>>s4+Zo@t(]dE8JIjs!$s8+2(O"Vj& +&'oP%K7K`tiN![Vf(8*TUo,5@jNmW$bsal.B02pah^65_*^,/LVZ*cc3(Nl0s&u"" +c*G.bQ#6s0&O;tFfh\oQrt"\UK-IM1^J4V:k25Qc!+u4/N(XOU)>HE0Pbe5?\5,Nk +mi_V]mk)rn@/Oq>9Uf#1R@"PdPb;dmm#r(FU:P>MgN[ON +]sN&,::MAh"\uOXT?b")T5P<"5G)dX_,&=:$bgKOA!?T<=CL9?=e7k4T0FdK2_rk3 +s2FJPej#Vm@[/!TN^=-31ni.<=r[\0Q$k\^T3?e3TT]$uAafA(n"J;9h.cQe5lQM@ +r8I)Gq*Taaq.\qdje)%%&cWB^]d#6RA[G\5:Ct85s(!B/dEt.'o.(ND%Yn[X-Zd&m +Ijd$E#pa0Jq_,H"[(?H+5G8IXo'`D6#eKRsgs!\+_(g.APdCFci.%)7eF"0Y>"=DK +LUDj;[<=g5*q$H:!t'=g68lnQ!,2D7r/(G>rX.0ep`G:_\e-EA]b0@E0dB`BUofR* +12AVjckJ1_#U'+Z;?4ic<=u*Mn&J,I^ +"G?mHbKblp$Xo;"E]h?6`]9IJ7No0>9QNj@*`MtYQR1R>kV"%[W;k2EC1mX`s&Tpe +H.?2A>#9eMO#),Zr)]Yai=@R?qKP!nB-?eql@7Lhi*?n#2JA]`rU3p`rhg?iYqGbF +Iu-W[YQ(]Nf+t ++6Ma\Jp`O'aIh>#?VG=$1)nd1<\;WZ0($"UKVp]t`N]hdn(SN+rL)g<6MGk=!+)gf +Y;`;\Qk"oC=**V_3:BZdZ6Y#5\^3ilF +aT%tkG>afFj^rlD?eo=EOKgc$Mt^K*jO@BpF+8^%VMSdAcuY +r'juT)+XJfh;i#MptL.-5%>^bidWBULD.DL\A,p__LLnO8]Lf*g7s-Q(n@>_8,3Cf +^@cuCqS2n7jL5`jn,*\U,9m3a%c+r&b*QY4lURWs)%IVc]H9eSN,VNd'mjSJ)dqR>-4oXo:.M.HLY:KXhcH],>#aN0@1e&R+[ooUW +G3bC12Joe[`T*$]`;`cNgB)p*))U%rPeUh#aFBq*$aaajr/\`;ndmJjs8I;YIf\c` +rg-ra#@+b*nNX/Sc1DLdm\?d?R+I23hFArWi8J.E>kjj+!_)Q9YSCEQhsbc&G]D=rWc%ZoRAMf"QUR/6id!(aI8I[Z/:TA +nbh!:>l-V^pHo=]SHAWaNUoI-@ar\\45"U$#DfoM1$8apnUK>Jr_ImMZMom^$]uH[ +pj\(U/O98A8o@VV)i+6kUEZmE&[\UNkB6#h;fU8ed!GTJ9)E_*J%=-_r$`0XeGh`q +=F9l6dIH8''DUh2"[Tp(IDP)nqklM\/9nk7;Y`Q6R"3pA_p,#!jF;'bb3@Zs>cVe1\>UF+2/=aL)lX/uO?' +I!DJ%>HO`&RlPXB^O!W?^+?UoX6eRZkJ"\0E4k^cbQaXWGdcq3_7Dj;\NaDtgsqa/ +O-ZpZ(Tse0(R=g&dO`bjcSVsa8!tgcs06$D1`m]3#=an&!^]XEU'i`#R>2d%!P>&_ +VXH>,hO1d"!5nmd!FA?Z$,rWi6(!l#t0^eaBM@-$ctc27!Qi&S8pL.^D%7!73aS@%AEVQA;Z2igD0 +L9u4WC"1,B20&RkE`h!H(&3J6&,`&ZYGG_ZWT\@XZ4mg$!3u#;OFWs[X[7Z%:TbM, +`(fh51>C?lJGLA>J'4!U71,QL'u3sT#n_,W#5P=FMX$DBC&"`+4*aZ9s`dXOCbEWIs/m4h$Aft +TESGqr_L&7);f&W^Qf?#MIDa!1!8C7mEeoL8dSg\aM46K5KI`l$JbJA[roJNXmKF7 +90iK#Vojd9*4CCtdBkhQ527&@hO21Z=R^ep^CBu&ao9bQoB0a&!"hJI3.S^CraNs, +f/q1!Xn`jZDZol#>T:NZ<;s\PW>b\1s3lVu+nSQjAZ#tt'E]0=4bs+-5A:-`rl1j? +_t0Gh_h7dFUZM*RmJ@^sNusnArsTGA1]i,N2UZTlV[(Je44WZA*qfO8H/o-R\DTR_ +m/uuYTJp$C]D$eAnUF`[2oTb!B6E6os(b%]j*rfc>4-5._#NLQS%a!kj*uNk\4c)Z +IjlZm`[1GZ(lEems/8R%G$sk/13.27Q&%`XAhM#cREBh$^&7;RdQ]^ +BDh5h^\,?upg7qXJ-_hIhc'1sPl:0d52uTk1@RT*MF!>RfDk;Nn/&%[mqMVu\j8T` ++2X5P`_e/`9pTe"Fi%a[F?$-H:?RSZ1#oLaogXJN4C^+k1+?VX:$K(7!8TgG;#eG2 +:-V&%!XAZlnS`GCdDpb(*h<8Zq%*9#s5iT1Ri4r2[,(3>FEdPtK,d?O+NT +Y`"nk*:=.-GPGeuCSuk$pn)gST:O8=V>sL:Pu&dej/G"`2)Q8>3;/C:h8:;Ge8:Db +s":XF]i/DsFs.-ir5DF#`9ZV3nolN,Y/.&L3(f+0A7.M7S>aRp4Ilr,J3nJj&WdMi +rptCW.f[&dCF3bbUs$K658jG_;s41\q/lc^XF'Y*5(aa[#TWL=k'qt$rpOIupM>Zu +!Urq^f6GmsokC:Z=kVoR>W`8pM[iHm`LW'36h4W(J%0%"? +NW5&&jFWY=E1$J$1c8:Y:1*'p/&cclBtl1%0Hbl7Ge*Y/Ef14<;WZ*Pmapd<0/j=! +s0&m:^YU!j'@/32_#<@Q+oqVG7DSb2C%^h7\TG-!>$O0=<-+[#58$0oT1%S'-S<+R +ris7!pt!VenE+-(W3>Qhu-Wi0-Elm1o3IbVBY1>ssI4*<%-d28(^ +IV7k`;m'HmA&UZbq0t]\YqY%K375Q?J.@2+X#":J]4OC'X`(kkGC +LdloVao<7MF!JJs5kUs]dK2PSs%!+sn[BGDGoe"De?&f*0sD +:mqIQN0RDk!g)FA>Q8X0o=!13cMuZ*Ld_<3"kC_eq-SKu`@_k&YV>r!^YYV_X[^<,*EPY=?=tA1T\]p5\oR1.2-'cf$_R+:(L^=F9R* +nA!l.+)h]Ds$QePIkF,S3\Z[&:,;BR!J3,<^hb/J9BAC%E_\"K7/g`C^L;f.o;`l< +CR!_a!$TSe)>gZ7T?&69^0`RWm''hhj[0rX0%qLd:;_njXU\KWE=TgGS +#HU99]9;tSj;9#W+#gl@.km3GNYE/,;L7mb1Nf:Y\N6:n3a9*H0A`3Ph4Ll7>c`Ls +bB9uGBA3CY7a:_qFs!@OcIZ8Xbg5@2No4CNS\jfmX$)^5GF'94-EbOmQ*W?h+2pA6 +5NH1mc./8sWG9HiT1gW01#Mc7@646gX@sVL8`bT'DRA +q-bb&El@TA6I)_6c9u32Rf$=A&+_TO4p%;TMjaTJ(BFI?OTH"*!VIZ[pn("UWD_[S +A(Oi(WXr0+c/"-=C4+iXpm]pq]Z'UB!p":rqs%i+r!OJg"=]HE. +)/I4Sr'\TlCs1gcgZftsn97npeE^u*/Y6[;B05$3212T)(c2W;<^FeZN&ZO3/lBJQ +8:#t=T-t^&AGcKu2SdWCKRiqp`l!H/+occ&!t$(jdBp8-+6rD_MKhZ1&X]"+k1hkR[bZmdR#&^\alT%^gkl +Df":6aoBH/%+tD0r3&;!hE>Q-4_W^*Dq`u\=Pmk$rb"m>nGiDq[07[%0M>GkVrupe +bl8I@-^F`0>;GnZIL?om+oh:0C[cfuo06X^kKM733"YMYL$Cf%%0@\I!TB(O_"jTl +5[+ugs7Pcfot:k1IqPAaq/c-Lpn(ao@#u+3H4efWJUeq*g_V$1"k=8*FpOFIH\d4C +l/q6PC9b!FKG+^!\_K#,"=00\2?JL1iuo.O!B^HJB>;JZNnF#n$07=&Et?!,e)I"B +?Ubam)N&m+$1D!EVZJ;3nNY)!-3<2"%8>s`1EbkOc+'[8Ni@\S,&S3j,NoQh>X=a"6TWe:PT?@,6C'bUH@-(E4c'n^%jqk^Oj4kg[a?HBteUKEHsSj'IUdk)'? +HKtH)=o^A\:2.EA!K)@6Oak;*7aHQ$&(i8Bk&Ik-jj9#j%nDWV_n+YIc*d\ms5$?i +PQ0cm&'PP\s$QdDj#kD4AuoiqZup,ZAQ%2D1"&0Eg+kkK1bh:6K:SWP?CG_7Rr;d) +e`q>39RH08UCO2n#C1Vh4J,(#[gIaGOp,9LJAd"C&9IC$DK2fL3q>1H%CDq%[:"I3O^k.*Q[W29P=7]?oR.c*Sd/I*gE:Wal"^\7F +@j*;Z2WL"I,/A*WZP5)4"lE]'RSB=]Mb3[Q('-*n)*F2dJ0+/EcrsNS1bl^?!@iRV +Xnb!QfDC'dY_i-^h9>\eXoDFqbOU0SU+9H3`O9e)f5T51q\cBT^gDZ@Wl<67erbo5`m'JSKF7oQo +rb$S/W`7m(&i/*Z%Ih,Y8%?#[W0I#W;W,3p5toE+ETHa(4sM+kP;B!p8XrQhG6Iikm8p,b*OQVPBpccJCB +jmP&9"K[4VgH`P$X8-+_i]e)'="Bjf7[a90MpnVdYKn#&0$(*$.%A="bKAPZX3JsL +gV;V5K&:LrJ)'JlIWt9#rFBk5!71uo-Fa?VE4,Xh\Tq7_KD[C%r6*'5e)G`u2oB6U +OFJ`?R7p:"BWi4:O.'FJ["(RFpis7Qs$,=U4^;jZ?*<"RY?[VMsEn\bA09 +:$VC`h+ilJ^Wh/YB%=9AFWF3Z^EuiBiZMXj#Aq,8K:M;-lTET!\bo@XC +dugX*d)\DOCVD\%2_k=SBq#Vj9l(4$hI-ems#7A*&HQ/8&*=\c;#e;5[K,pf=Tm+- +T%,^U3Q/l,]=CoCr/]j0h8ut:h7C=S?aaN][rlVmSU3/Z]q5HRPIGOmW]NQ3n&V^.^Ns,qPQ1%ds/(0c +H`jon338@p(Zh4Yfu@`?V0K:J.$@%BRjF=CotMsg),5cR@f[meAk3sa6Nm2/P(6!D +G1PV6pccn4U+UclKj+kHSe7(TNj8r3H5VnhkF&'gUn*\trn6tYSGuajchX3-ptu!& +p1jtcURpZFJ%sa9mUlHBmr0iY=TruOFM\">3Xe7mdlO,bpA`^l1 +5m@0nLaf>'WnP8]OrohOXL@[Wd +J4K1lVgC9Fi9=k:iG+8\TW3JAk5S=c(NVt%#V.j@4rb[bsfO31L5gth-@8QZ(dJlroJ(rl\ +Rbk=?OrU\jr-EDMmHhdGV:gUrL%j?&:&>k]LYFDL>o!P`2D#G!C/'9:_F"Z`Xh9I\ +GTQ6;bg?IN:*,5=BE19n+"[IX$@!#>?@B.Mf>KocTIFjE>A>u<,?RXh5\dZH3"7Gl +@r"5F;]G4ngV&m_KG5PCA`ikAJtPgP[!)R("_p=GW!`YM5SbIG*#9]U!tK^N_K/iS +%%EMXcASir)JZ^g!L'qA?mFm;/I\2H)/.8h>#U@!fF3dRnZUJ%"-'I?;>#Dj!n@+l +>J#kG!ifITn?X;sf&^s+Qa4o4nJ0r"K*K'3OeNY3"dCc^i;%KM;X#W!/1dPZ]I[u,?0i[8mORSCl[o_W8+8bX5eGXJ;j/?.C-']R+9fD)T$L*uFpZor +H4g@brHHuYi9!8sOC"RaZ2f<(hq/i$[Gco1P5ulFCMj=t;\R""nr1,e*j;CLd"`+' +Q05XH!>:9u.V*Q7F:^daC(MKk83hPA"Mda<^h#e7n=uO:eLg(i$*\h"<+(8GQ,tbs7c(s +rWgsQh=*Y&lMp;;lX8IV2?N`ipg:MIe#a"lX8f^9^o-:eLVDIe[N5%h]DortNr]pl +O8<3Lr,3S]=@hNqYfT`;'<(rK`;e7b&H3b"`_lbsS>B5P`Q@Go%-RZ2aqEAoJ$]"I +B&h)%(A%PJ\+'FchsU4=*f0_9GQ3%7T>*VWi"P#Dm;D+Qh)=m)LW[U=Q2g7)1AkK. +/Y3bk:205/q.EU*A5\[`N%soG7n,`h4;JEdL-hXIU];;66pP3"/H@!QUpu!lJ477J +V:G_'*dkO\Ub+hh!q61R59>6!3./;;5@![6l?U8P03A"4djab5X;p5k31Crp4Vh@.I+g8`e:LH)'5F2XkAM#=S#5.K&XUoDgKF +(k/C$:R7s#T[N;'O>eqL]QS>q41Y(:4D`U[=g^WZ,BB$XG%Y34X.RL`F%O$Y:jq;\ +^%2X'!u1cGlnd/85.;@\4($XkF]O_/0j1lpBX>Mh'-nYK(M?!N3%%fW=HKWsuE; +Mm\8VA8\.99=Eg.BPKCA><)80Ad1m2Q3H&J!.3u8)?J@f)fO[7jgB;^l$rClf6lJ\ +h*0mXp.]QG_od/1nd"JBg0+dZglFb<6>%[oaF>1np:>r9>KU1pC&>JXo(3ZBA=_7 +cY0c4s&o(!Q2bC?3Q:ogZ;.4>j+*s+kC46JeY/(RkZG#2&YPK>#j;BJ(&u're.)4P +`16k@HX50G!N?#=CYaL)J),,ao=o`AM#-#5e&BH/Gm\XAk59/L83$YHbq]:jLMR;_ +bthsND"9sfT9(crga@g#:s_Q.Vod>,G)Z7sp3c#b@eTM9mdY#mrr7)P#601dh1X=8 +n8J&gr;ZJp$^gi0>S-gV^J4[Ap$$e_l#IO<^Sh(G0$RapcBSdd-b5(=!j.qbiN*W/n(Sk6T);u#[_(,"[G4D*s6d:?;YbBF5KqI)HmaZ<7MUnj +Gb@p]s/o`*%eB)UH.1RQq7dOL?4-Hl$O+9f#Wb%smcjPD=7VQVs2DdhMihFghr;$. +p&)boVEV[5k(TK+B@N!haHV%o&io*qVQOI@=eIIY>"[L`nMHgc->4te_\:,k#lE[ZoDF::D^@$ +q;c3I:Ac1AN&(E#qa&<\g5"[[aPU-ir-3sZhVkL7S\o0.'%HeC8*<,+J'Q/\p4(XO +Q*F)i#Y:C25hn=/R/cF9^i7dW80U;PV\&q/9:5YWNW2QXi8=#!9E3aI3][4E=*XE; +f<=`SRq2;(h!J_!J"HHUoK`;85JT(Ws!99\\bOKhmK5bU/8QC.bl=J?Wl[Kl(*@mM +;9*8mZ@#?^q7gr*78?_Z_>dk[2h/NqqYbjZr[5UULY]^0mK)JM^O<'Oa5clc:HLH6 +DrIm/rdVR_6NjX,2]!jJSGsc46qeJX%RhR/To^H8(^'lS_WMm/GSFKQ+G'S*!"]2$ +pf':39,7SL.q^4KQfT7Ms+dXBZiD1cO$;E752?n>F&;lY_>fQTHhIA&.o)FOPOO&c +2,ZsRjV%a,G^hp.?V'g?@^7'u"k_$`([\9G59crpQU(OI+$TSSb_M\]F27]QraP#> +('-;g=8hTq8N_/F_:B.m43]O9&I^r(L9JqU`+D*T +pKBQ"2<>=$'PijTR& +8-f'kX8PQg+$4^@rl;/o^nh$e]"@JQkWRqni]-XR+8`XDrb`;ApOJX7)aqC7ThZ0m +LdGu&(BM<[D>eYj5l=FHR\t70bj?`e/'S-;#;*#=YZNth";oDNJ6WP_Dq/OC\Y+<7 +VX9rK)r&VnJD)IQUB(sW:Nj1mnnKDV^UDOq^.Af[JGC'Nqs(WWII^Po1I'n]ruW9P +^`c!/jj34A3rdkJCt8?2F01)-(KT`!p!*4'm5F`hCQGrurdG7=rXfZ[nuqdPHN=TT +h^/R>n^uoXZl!TGQm7gW+@n>%b'b>!f/;K0aZ]%mYC"@R6SC,+#cg:SWG$Z3Q!2CD=&'*< +jjI2kHl!B6HUV+gV]3m9>OA>84[o8D6-$bsU>b'Gs8I'%VS%2"Z +3d\LLLgB#Kot%56&)EO@c?o`&5+`k:W:ORqS'5=.\!prPK,iP9"2t,k6,TB_[u\[= +EO[7Nhj!T3bI3]_lO,jl_3]!2S4gjg$g4K+2=cXcobd$aYhn +MiF3!NP"LNLYGOlh.GJ,3Fgr5Na-8&C7FTOZ=o/&^nYfj*fZ@HI8GIln0-@qgZGu?HA[n9_:4!YjecHiBAt+eg)q>!g"=SG2=GM!\c[`/rr@m@02/)4 +s1uE_?S1'f,p37I[fdY>r:LNUg3UduYf5L:*f>IB+)J!']*ADY+/h$u"q:WbqN0$d +Ke+jT$2F>k$d$!_>93$6r3]88b*l^`Pu$Ij@_P"3hOiQ]s+_QVrN5,=c@0tQ7O)kp +Fa.o?F]MZlYD_OT$gFX,HkbFn"UDnN=ia1rk!6%!6flYlO34's.Ib+JJ$Rr2J=3$l +1JQ@C@([6U@r]s'r#[^(7SNb'C6TeZA8DCeLAu^>2de(2^C%bn"D)KPd/*dqs/%(` +2u,]7f)OOp^Y]"1K5Th$Eg%F3IkU`g2YY[7J$nVj1S:-9K+9VneVL+>nn5pk0emd= +;NdS8fa6[hYIEhnpntHH2W"8Mj8Li1cOmtjcQ_;. +]C"Kn2W=ZhQg5_G=.V%1n';be:Zp[9Mqf$<5B<:lHOMDKINj96VpFBI\/d?&UYk0t +\Xa):HdM(!7K`jHj2]3hZ.NkR]n@/oFL#n+[a"`rVEc[6^rJMWH3"Lt#Lrd#\*$t\ +e$\eaEZmAo6@t.0Be>H.LAr"QLb(o"l0W,Vk0HGll!k$U8&WpTio<:B`%QlC6XFcR +dgPcRF^AaR!Vq'^h44DXn\:'"=8(?5^WRgD=i`>[2H.B@fOrb->I8eoXj!`+lOW*g +C?7*MDoh.p?b3KjAi"p9r[Uo6:B&^1riJTfMIr]Z$1^UJ:FABWikO@M!rbsq%mn)c +!7)j"!E)2TZafn/7/Ts:qhdhXprKMhWHV"A%;YS(=;O5_A_8:Fm$\Gh&-3P@P>0fq +d^PkWE,-EGH7nf=aP=oFIV+%Z,+h$&B97)>1*D(.Ej\:6+RA>a1]2)=?RmAcD1R85 +*o8X,mV*'@>9f`J)A%?Z5**Ci@qo]hoa,*!Y,Jsr70)pDKFdmV^!OJA9?I'W$@ZtL +d!HCij8_\J^;PX/?fD+=g)2MY)kbM>QaMGs/'u7H]:!:dGF4>+H>T_VW#9V&pVKl?"f2kR +!]bM9ra<4;Xk'&6G4dJKae+unQ/[-\*U5ur^UifraIds(5]0-2YCBZ>J(=R\s'PX& +qZrkoLV:h!LI=Bt^&O]Li/l:Cr6N4]IgI"ur!KJDD6"Tr/*jjE(#Iqja3BJ6p=+84 +AuGWP52SMu!0R8sPC:OqD7^THGo8]B*JQ +oC++n-U3QsLB.?=7t>rFq*4=K'tk:)s$c)Rpe1Y=7DK%YXp^3,9uCs,,!^c']G77QB' +5^3+tJ+%kBq2rStIHb^#/ims7*'VQ)Sn5sfoAeWcmqm@]'CBfjOf/-nh:uW"c*^Eg-1qO+mkU5'aq>::-P:mgh]hSmqqMsde(j+?8R[(+6#+j@RZ@6'LJlrZ +7(Y,&W;@'KV#+BE^!49^otJ\9hR1+,GH=Y+L&f?:Z@qAGr]%#(N$\HS[VVDJAqV +,(SOfnnrsK]k4Iq\DLaVBJpqi=c^W^::I[bV.j1me!T='%t4EM8,mXK!O+^I>e'2Pm`0,3j,NNlcMM_HI_'S`hHBK.e57_SgXL1gN69qt:d12^.2c#QYP5@">h5?mhb1HT0_ppg +T%,^Uk@NPL:RVR_VqL.'pg8;I(b6D&Q_hg)nhf+s:\Xl?_>bTq2Ep#8ICY.5M>@2L +E!Vj,8IBjDqS1&A\cA-"h/<7A^%lGTrU],#o095S$e+;scI/_Ve6"o;XS-Y`s'CJF +Q-D4.04IrW^>(*$ +5Ee=FJF)iKUKgnQAEV'k'++6%:a2WmLD9M>`9D`"r0*gFlX@as.fYU>S&jL96fm4M +WdQApDMHWg"WZZB"sUj"5m0_lfe91?X@li2M;:^1ni(6.W;1ctC<=kpoO(n/[jB_` +m+l"6L&BPhH<01?Q=`RD1W"i[!Lrm\O*+,[n+1H7*dBmt,7O4X7HMkk+'!_bf*eB> +2c9W]JFFA*LG%OY('42nCB;".Ips#:-[e!#s&l1G1Tbsb5Q5h;p>*a*]BWC$F:_tZ +2q'S9Rd?JTk7MC/qVTrJr#`2P_Z-/(=cm&+5"^6I8(,oh3:d5c59JhHr9a;F#KR*2 +EZlL>"3C7_kldBYH8]M*aYA"n-G1`+n<F:SoMg$R[J.D$%h<]5.WdPVIR08?_SF1^s,6n#q&3f+^Y/VQ +IldUZp4$As;<>"Z:UZ4"PI4oAIqc^D$Tr]_$Z_nipCb52?!! +h??+"GogmppG^[X/,e#[@C`ad;HPH4_kP7EdHljsGYje6%*/<60olA]4Q#oum8DN] +J%h3JoC*"a0A=F&r6;!^aPSrenA(^r=4%:MO]`7gJ'pHHYRCMR!E!A,Z(WhFQ\P1m +OKQX-lN07e-i*\[s.(4=#l]&:F?-fup'm^<SB;5Q_`6`41pae@E +;&m7tW@I"dX7OGK](fLcs*kLXYJ^C=s*q`>q>G=5cc:[Q4WSR\+8EU24ntX<^crn6 +rdMW*r/VWOoB#P`o>Sbugs*qlr-s:W5<2o+O(>"Ql0o8>)tD%&/t(&=#QHk;@]N(: +1%kUJ=j!o]KcV('Hdp\="FDT +,#E^i?)%?a"A$Ha'n%`)'X&mKDl:V3/5ElaSU%]TpjRFpOUcs][6*G':X_MKTH`9=Ve6%+;JQ_ +#MT7lJO6pMMBM:"Ju#l>_.Gl1rgea.(Z^&]mQEe#?GQKd^<9XYF`QCeV7=XiDKBi; +QQO!j^:_b`,7?+:]k@k4Om[Uq1E-t'!s8BQE=W,,cW6thPd'`u,-DC[IKFs:UDcp7 +IQ_@E.&(440]FF+NIhX(RKBBiBK,M,3hGY1s*!Q)]'eU*s._GZ@t4E(k'n*FbA/\f/*&1b\V4)* +IqkkFp<7ha[Sck4T.7RQ9E0c>i8[i,o5+aFnDP/MJ--C&;pem8eYSIHZiGgH@Q$)2 +DMO(Kn:,F%4Icfi,m&]js'X!i9n#Pe[,>Trs)Ig`igF8n+3;4Rs%NDp"!5H1GZJDJ +A+2]\J?B':0`NLP""C8p@@!)_0M.=bP),ejXR?J[\lf/dTGDX5#WV+&rs?(3s*+Ht +r/(GT>I>CjOM>ZPrYMh:ca3PKq-!6+dWF.N]aduEb-]_iLR%frf/nL=2KsAdWZ>r!PIAXS)O5F5:hd=Yk<9D$GeafW-57UK.UU-BiI'S_ITQ7bnFg4H& +""q:ml+lB+MZ6Rl3M0HqVWO&HM-+:>)B97a'L>NE^f=[a%=*9i:PnN,QS@(E#8$^C +]7#\lYHs/.hql?5:S$AHCrT5c%OAM$nbjLVXoDgY-@[jc%T>UQDq1QVQP_Pgm6M73 +ql0L;2g`!bHT"oOesN)qK_"f+lHRc< +gXXt9>>#De.h2,;h?M5JUOK,gXd/A?p3.Z(m?AH,A-I[=%sjG+fIo#JTI"4-_PJ\/#]2C#h+8eD*c=AS1kWt6'. +K&jR+`!q1!KY'D8C'0P[gU#&USp9,:j%d^(LF08jhY0P'q8j7*ri#$`&-Pi@3Gg5) +ij3hB*%%7WhdP$AlSKjj%M(VopL*uD:k+i7Oh"M3Ua*`@15LJ#fZ>8^H4bU^r+K@= +e8)CGe.nN?e]M&ba0."o%F"VEP7`H +`W*oF,WdW'+qkng7n;$:AL=3HqiutfN0io)AH)UHnMfVOVuE8KRNRs2JLB+7]Y.M_&<)O.5uZ"S>.l@MStFI5a=j,(TLHG:<#' +#X,ejmMuo2^L]i!YC5Bq(O;PN%p)qpeklBjkdL9erhg?Y/teCa5F'")Ggo'R4FVc7q04ps<13W. +]`c()XSdCR_=mp,m;<;@-e.5WT:ZVZ@K-;i,P1qY^&L<&/3TQKhk/-Jj>kBG(o=9eR^m +c;/Z2"^IYdYonSd)3rW.\3/S.05c;\@D.n04io6Broq?pr-73\7.-QFtLss(1r5]s4e;1nt/gf9ocVVN4+6KAYX:U@DEps?n&Y@^+p%`cACY#E($,lt!CT11"_\$+0iP2YRtm_"KOo`Xf?=1F.OrJ"!?Oj14ub^fSO>O0OnN%iNBjfMT@MN. +c_I^38io&sb_MSc/CFXr?Ui"9EQLq\mk-2=a::>o#&gUG+V5bo-#8TJ1OTCG_e8D!o23.-iApY*/g9Yo%Eikq#J/@'GNn? +13.(=*lrC.ruO)\jeqamsk%+mWU6J^&4l0>KFq4,Q_0.J'nDF +qd5`g3WBcg+0smua,5@AriGme?YZ$8BUYN0R[e78T5;/(2r%_C*\6r^MX.8$q/GTq +E8ghA)'GR'JPS5o^ZYUMEH>]n>So*=2.rgK-\[P-Fubn\-XAuP.\!BN)O?5=Uk?R@ +!;N1es.AEA*R@U(nAdUj^[h]L5Pu&fB8k!G8]CV"eLGXO&^"3V>>9Y_cn")T<6bGW +)nH#09nC$G=W0.h=:-bXEtRp.s#hP'n(8?.f#`J]V".mNrIt9\GXVNmW;d0YI::Cu +_CG1A9rNd\s&O16.J#dYD9->g!BZfP/SgFGI8'Y)fK_>L:e2+fk!ho??Vk&D7rD)B +m`rfocDR9eB0TZTI(ib]I(fING4gp.Fn"SMqVU^"rpdK%]lWcEYKo^tRfCKM5b7p6 +J%o=im:m+0s*VBTTtKl63XbtSe"cD^U$D[WA%.^%3QY-I*g>5n,hg2oPZ^ZWDpnC, +/te7oJC?^GE@LkCZ(N1E^iV?GJ1fb:^JQIsSe\`]$SX2.2F5%M5r#iHa?ok6l-sMn +4r$4%%gT,$MNWBXACD#kan7]ABF=BN.1l-/"%<,gf]8qgnF'ds5LI2O`5%ubIt"H4 +S',B0V&tdnrhb.chlQ'rr!aQ"p-7@"LA]TV]fR1h>AN%orZ]E*Q2g30?tPil4iNn\ +BP6lX+))c +%FFqN0B2D,Wk3eWYD2.WPQPjD#%((R?dF\F=r;b"1@og?*YImbHfaYog:O,>\%8,S +Ap&anORms+c2[1Ts(NjAb+Aq=1E_2I$F,s6tM"jNk3ZRK#D+Fcl%O3)9gq +=c?F>Fqg_=20e%).KkH]+/n3Z'iZ"[YJOKQE\KKq6g1ZCcIbl^\ +`IR.tfSR=pBcg%P6DT!f,U-<&Op?bF1Emn3Aei]aYt.:B'o:md=G +4L:&\e-kmeo(F\'Wt>?&YS_MuP-S6pp'0F+(4'X-fN^ +llWCc+"h71:s]]F.21UgeoI1T>]q9W5m0klP@KE!>]_5- +)9mh(dsc>e"'b8+a$+$^A+5!)2Z_"F+7Bo+J'J!unIPXa(]RZM*MXLs84lP&/(94r +pci7/bh.^/s0PLlZHA>rrj\ecq7jWUTHbSd!1j+aBE+b+L]:NU&,GN,fX5(gbl4GDL:]2rrm0acb6_Qdr6BQ(BO?#,iIkN)0F\"mFIWiddg[ibP&#-m(Y5$9W9MnU +8i:ARiN=$gYE&S5]%:#p;>`)m#pFcuq+lsYNN65_@kHU6Iq\2[+l7=.;p"&[Q +p$1:aNAoSY\&R6JBE;p>%>Y"f^6KW<`T[.Oo06=Vj"olcaSU4-B]%]J#dbCVr`[QR +Q%)J(NXO/H,&%!3e]#$hnel(3?7F9!,pbNFnkTq9PITO--:/%RO%Bl_3$0j?Sq]Y0 +E$s_-argVt:A3PA@J)f`plI]II[L00o&t")b$RfZn3;OuJbs9@D-CtDl9th\`mI'p +9n]Df5qtX5.V4#K%%&+jo,"A!:P(q,>;P6%5-DK>=2t)Kkfs+55l[]hY6P&a][Q\]\[N2VA@^CT#H% +dTa9_8HFIJk'mGL!BL6*TQ^;RnNT/$#3qZ>dW?%HLL?h6MNg&H0Ws3YR3l@8S9ZbtD^ +#n7,e?p#.e&,<_a*Ztt`!r1ZK(U5tVfkO(>mrPe:5#qEB[o;N2,N>'*s7(&8.Orc_ +ji,^l?2P]ir\)Q%Qf`3ZoCq@cVb;&Cm\_QZs'P6G-l]ocCpB>AkhEi:IsaBOIj"-197(8>luI!8Ak?U +L"(-$\WfZ:VW/'+kZm+l(@j\MW5>7U?pNuN%=-_aQa;b>410p9p)]Pf9`7b?ccf0L +-%PCPL]?%ZDgg_C]9Ol85@'Bpq`IaL]<)2%%:?`@T5F7Vr!c/EqSRcA>/L252Z0ZN +Cp*E7i=GiN5Q+jb,6@jK!;KnYdeJIp$#CcX*\Ir&VgqCsYV&2prJG@n-niN5WrLc+ +!TZI9k\>Ku.5p_#QP!3LS5NLlIo)kTl"7Z5TP/-8Rur.6#?0AOi!*d^.f^pHn]+""`UEp"VnB#5q).OPMc +5oA3&)7W;[5p[tp9ZSmHZ;OAB&s8h'8GU>b$t4o>?k[gfoY=A"&1M!B#gj"k$mL:[ +?nurCr+:9d_eAKC#&?.h!>?ZMP7<*LeL==-[0+#?Cr1EeW +(&p)l3]R/G%KEiX:Xd1$oA!g8b^AI#9I#GpZ17R`@gs79]utKie`@s6!#(9@V[CZ: +\3II[SD1jPL$e0\>UdSj`q,@j<)aZT>PW*;"X"Te:1_o7/->As(%B6YLZ&E%D ++<^GZq7nsVPFA20 +G-Kj')4N`eMZC*ahbA[3#iPXL>A`A3Tr3'7?f;uGKrn?a_bta%;Q`&a( +62nZ=-t*%pKCOas!bMPfD*mq6P2]j8FnGO#P1hp[7/gUu`(#.8?E.*uRY[%aS+;De +'2n`=OUFtm,F.9J9kikcD#(fX"TLjgbl>H6YJ^CTh/C11^H_-!5,eMZFMRhST3jJ" +r8dJ0dD%ll#^G'9Rt%C:\![1[?*&E%00PiNErJ:!Nh"%Co#HBNi'?)H+$jMTs-DMf +s'&;m+)ql82JU4Ls!AVA!/F;-E@CuR'd=cD4Hk`M,6^UL,aT@Y`*tX=!)g(I^s!m[ +9HT%f_Vt#mE6RbmnbIpP*ehWt!<5.I&j,W5YM7arEKO=K+oIf/V4d+'#PtMC2YO'J +n%ug@PJ4V3_6E-"q=d2Mbksm.-Fq#B6h,*GpN,oW*V>JZhjrX&Xkp`#](3J]\\?S$ +][4dus1I\k@K%?I2UqV2WH`$"i,'L\EA/qj2q7k6+XQ0Amik$Fob.6/&j#rAr;3=/ +M#ZS8s'NVBj3!]eaqqCaQh(kD(FEK3!]Cg$D6LV/?[o-0H=PTQCKf9&gq,bgJom]_5cZtgP +&XG3SE0_7@7)15q+N?nYS481.@hF6l_*t^=o"ReQmuiihFPg"t#5L(H_tHe$c^ZK! +Sd[;T$BPP79'@Cr[hA/LYK!sML]I;G,!`Q^JGe):LRB7!000Tgp=7Ph7a?8"%Qn8, +)0_JlE17o]:(ABP0&li+La4)/8c-+5DK:h-aO0+UM;6ZWZ9-,QU6c[<115R;V9 +Z.[)3LY1nYA`28&kV:e;ckhekOKf0B5X&uXZW&/W?c:NFRK&]Q2`KFk*q3a3s4_;f +-S7ShrSPH!!-fXWhf#F3J,Pt2n)WB!RV3NUI6TRbmegnuU:c*aW*C=t>\`^W7Pk2@ +=15_6g[0^f\no&@ZW'QZb;7"(a`N!3.hrSP\Xe+["'&\d1S;q9B%#.A!GW3#SrJp> +a@+$'_6Ies>QgSl-FjnBo7,>eV#IQ\V%HPs5?`S)-2aAr*)()]n:/l^prn\YWu!(IFTX:nc.nP;Uc./qod)Y3X7Us/k/r)*BU9*p6!OFK;X:,=CG'dm_7=_?k[ +Vn=R>?C`X_s*&+6&*:ckQ/sZaV6?/Q9o[hjXT.D[TLq\V]eQTCDOp4!c&^,[_qk^W4^]e5tscpn(#@ +&)g&qek7TL5)ZDtC)X<\#+`22O)rh6*G=hp`M9brfGm$"CnEb[9h&_[/3FB9TrCRr +1)(Q9qGl&C9![07j./(-+Ve5lO;,r5g+3P?(_J^J)8scSk]&rP.%q +s0""tY0;Cas!^qeQl'[;.@We^hiA7)J<@\BSGD\[s!>j?&U^3+eGg:4Uot"KDa8.N +oG[;`OS6?N'>GfN"k$HSI4l>@Qc&B\UPbD9`Q+7WdK1CFOH#oD(JkM2J(W0bV5qC'B1Kt\VtE7j!*joDa)0#le6Y3\/=P +r3%k] +s*+L8%K9BH6j3F6.fXf=qf41`I+$=bJ.7+W!aNaVn-(2D_E\$kE`/)'9W

      rSM8fB.V"$"G;m/ +5HXqj1IjeTI5%0P"G;Q+JYZ$As/4'CL]Eg7!VnPOa%b!s>QkBQ'*!=P"YO!Y>;sWS +5Mu:/b^\3tN](+0s0ZUUiLg;8B0cHN,G%IWJGcMR&JRWks6;,5A8:)gALG-q]1@bX +ID+/1&gmW`>[C)P:/d$jG5*7G!roS_J"Fqt5F/3rX3CQ_2=kHll5qsks1Fp"fA5H$ +mT9DX\Ftu]D0!_P`rc>As7>h*jBqBUIm+n&r3-+s4dl=hmsfcKs3(%:3MA"D#)W@I +EqhlaP(juTpto`LAc?o;M#T&qDAiOUru3fY!d\9PZU)ST^gS=.3g;hs=>2E@#u)GX +o0iA,BP@7p2\Gab$2V]i/GQng?MB8rUNF,'?*5>9*u%K5JAWP>g\[KN.=)#u]AMe6 +q7T2nht\?#9-\tWn6e'H;#b_A@X]eVZ_$ONGF85`r(V2g/=cWR`P2hH5NK$$lTa/J +r]d4Xhc]:!>3XaCl<7I)/^KOq^:9^uD":B2I5[)o^#T#tcSsa\R@c2bIfo]]p:W`p +TF4!faCtdM$_WKp9,.C`s1a6ejjFshs!*.L`s,mE%f\:?L'[QcNs&eK7aHJ;]ghbVP1)k2.41@]O4m&`VM7R$J,jCIrUFKJMG09nH=V&ci]E1shRcs9us +YKfk-ZAH^']P&eXc/>\%L&Xs>l9]YURp"!HQ;ad9A83TV,WCRU;Ier:+d3a;a%)3m +\VM^S*m97fW]gLJ#`'K"Q?SAPCh'OMTMqLu"QYXG>0('54rbgf$muR?Op?kjs-!_] +nkWEb;`4C&qk&cHcMqc>+9&`fnLa(>DsI?;SsC>pDmb_dO.*7:&D$`N5`:-5T,%Br +r+d6QWq*@M7Hj&KPQ+$IRZ.e2q4Ao;9'O0MS?db+^Gtq/1@j['oqaeEH[56*Q*LA% +)ZB-*OoMslrQjlLs*I!%5EGF9cf+>rE\/Oh`E^h+UogP4K9V:RNgpSf;2:nT<9ODd8J;$FX5@"+_\'_9:?Fuk?p +0'=q&A6Ii4%O)(H:799So=/A]s727Q(A>kD;_CJqpR*kl:U!rEEYF:"V"mK2ZE\TfY,i;^H^Cre.3 +"Dh[:j:?N*7!/DQqTf,M\Lqifn%LAq;FG`J:2k:&=\Fl@6[MSB7C=!?Pq_E0KQ5&J4=j58Fju;pLFc!A"&NA%f>X +.2n3CK]F:*rVm3dpSYCD1lhS>Rfk#$RSK%U$7NC$!>,I"`&5AE1b":_\LP)+`HVVM +E4B9)`Z*L+[5G56DS+ZV:,fL8S[O#RTD*aE7q16+RI?Y3]_j%In+V+apO1nnXS-n. +C9]fBs/VmaX>EK?A_H]U`JgC"$g=Yn)ZI$'!mf1cQBS')LbiNK0Jq7#D-^4JV=H8r +R?W"RTjC+\8h86p7&U:+Z<4%a)X`YT&H!D8*KP!'\0dfo`ff%H>TIE5?ia(0"@L>I +'(5lcT>KWN#pfRR5j/'SG5n2,s,8T1_Xekjn:*R3kG.eHn$q-""hG*>T6;?>n5&gD +DZB$6ppWi=,8(:^#*X]u8go=Zc&H_5]hk*S\As:)(r=[9`DNZsRl+!:iiSQ\oqIM_ +^W$JXg\t+MGqh6b@-:>o-o]DVW!"7@hb%+'s+1EYfEJeXiV],Z>QFO0]#&qZs3m,A +Rr]_d]/5l_)Y"DIF3\F@HO"Qja3_s.XZ:]K?FT$n[sW;OO1'HDnGg9$?[aU0$R#?X +5F"S7%o,FtE6i0`/q=+erX[]Xs6MB'r8(&mUHWeU,6rh?+dDV+CVT&GH#o^DL5 +mgdVB4W=.dkl-'la8^9=nSN()+`r2NT"j$!tniTtt +\Q$9HNSd50"3:P\+TmgrM4"A4!;R<]iIPC]MZdUDeFPlGL!r2fcKOWXfK+`_(T,dVp(UsSI42CiB +i;ZRA1I-E2+Ph_ti@TS`![O05,-=`GfGF=dme[Ao!0dIec#p^#1dHs*s/h#k:]f:n +f>aoD$3-q&/q3Xnm')D@"E6T9l$4T[$OAe3L?CdW]"6o0h'q5uYQ-chi_DN9r9ql0 +Z8VVP_ZY2JEN]T)?f.)+e%IirSip,/i@b9R'GUG7I%Ndqh_3-/g+FM#dU5#cZa8KB +a@M-kOStpHFAF\EhoUuLY65I:+0cNbJ56Q-diu[.V+NRl(-X87#kj*M$6%"cL[>ig +r3b>i72(3ddSjN;D61mKC\$;@^0M`U!WMs6m9rIOaFBYR5Ki@W0+<5*1_SrcXpi"C +=tXR@YNs*q"MLd=hpb+JlD("22;@H^kc="4BmKK`0Q`C4IDQnHdKA#o6YW)DXO?GB +PsuQ#WRg(O(ND3`9#3nqUgj`#:F<&/P"MTn+=9Nc(XO?BLh[^CD&;^9@1,oi7Sk24 +J9a/`!84HK(RG."#\Z'p+"Vs3J+Cl"rprb_D>k3BjH&aPR%3s`i'6bm2W"'$!=Ae$ +%rMj0\kL"I,l@cB2=ul%f#G#Uq>;I^ZKrfrRjo/FQ:kTdcg,.%9c:gV7Z$rjt&Zm4lreo4'.d!H%V(+#qSRc4Tgbr^6J\Zo2o/YnR-*k!mb\TBpq& +cQoKl340?p#1T`It8A,sH,/M5CI14YGi6qpNj#Rac8.0O)l4m"si(U0D`XJH,WL_6;t@Ur#P +9H+@T69u?T;rLe0qB4k"5QL^.&$c0:s+!V!g-0C2hT^\-]P145Dn#ZGrf/AK\L39f +.66?OhZ=_Vq2_C9Qp7sb_uCGZk^9"MM;Pkoaa^'\N.pj&PnF)lYN`eF^gH$77X^9I +gb@q[GR<:`s,U;Oj!,cFI5+n,(`,1RM4KhCf9CsX5;7:3j?t9o3Xb^^8W?$c4uT"S0#]F^'7nLZJIcr81XYltuG0pqKL<%Png@ +quHauS3\:tGo>OY^8P2KBahanNe%,6@j? +%-EOD- +Q +Q +q +72 -796.5 490 262.5 re +W* +q +q +[1 0 0 1 0 0] cm +0 0 595.28 841.89 re +W +Q +Q +Q +q +1 w +[] 0 d +0 J +0 j +[1 0 0 1 71.72 52.89] cm +q +[1 0 0 1 0 0] Tm +0 0 Td +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.34766 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.02148 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.01758 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.69141 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.36523 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.69922 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\001) +[3.624 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.69531 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.36914 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\006) +[6.096 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 476 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\003) +[4.476 +0] Tj +Q +Q +Q +Q +showpage +%%PageTrailer +pdfEndPage +%%Page: 2 2 +%%BeginPageSetup +%%PageOrientation: Portrait +pdfStartPage +0 0.706376 translate +0.9983 0.9983 scale +0 0 596 842 re W +%%EndPageSetup +[] 0 d +1 i +0 j +0 J +10 M +1 w +/DeviceGray {} cs +[0] sc +/DeviceGray {} CS +[0] SC +false op +false OP +{} settransfer +0 0 595.28 841.89 re +W +q +q +[1 0 0 1 0 0] cm +q +1 w +[] 0 d +0 J +0 j +[1 0 0 1 36 805.89] cm +q +[1 0 0 1 0 0] Tm +0 0 Td +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.99609 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.66992 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 227.66602 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.33984 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.33984 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.33594 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.00977 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.68359 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.35742 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.36523 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 283.69922 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.37305 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.36914 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 297.70312 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.37695 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 307.71094 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.37891 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.375 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.04883 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 333.71484 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.38086 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 339.04688 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.38086 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.38086 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 357.04688 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 363.04688 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.7207 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 373.7168 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.71289 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 387.7207 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.39453 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 397.06055 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 399.72656 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.40039 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.39648 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.39648 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 429.07031 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 433.06641 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 439.74023 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.41406 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 453.74414 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.41797 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 467.0918 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 473.0918 -9.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.66602 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.33984 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.01367 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 227.00977 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 233.68359 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.35742 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.02539 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 253.69922 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.36719 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.70117 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.375 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.37109 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 281.04492 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 283.71094 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.71094 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.37891 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.38281 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.37891 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 315.05273 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 317.71875 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 327.72656 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.40039 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 341.07422 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 343.74023 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 353.74805 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.42188 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 367.0957 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.76172 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.42773 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 379.10156 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 385.76953 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 391.76953 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.44336 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 405.11719 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 411.79102 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.46484 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 425.13867 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.47266 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.13867 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.47266 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.47266 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 449.13867 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 455.80664 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 462.48047 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 472.47656 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 479.15039 -24.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 213 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 219 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 225.67383 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.33984 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.01367 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 239.00977 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 241.67578 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.67578 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.34961 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 261.02344 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.03125 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 277.70508 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 281.03906 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 287.70703 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.38086 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.05469 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 307.72852 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 311.72461 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.39844 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.73242 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 325.72852 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 335.73633 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.41016 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.08398 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 355.75781 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.42578 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.09961 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 375.77344 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.43945 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 385.10742 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\022) +[8.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.44141 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 401.11523 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.78125 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.78125 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.45508 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.12891 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 433.13672 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 439.13672 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 445.81055 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 455.81836 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 462.49219 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 465.1582 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 471.83203 -39.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.66602 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.33984 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.01367 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.68164 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.35547 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.02344 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.68945 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.36328 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.03711 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.0332 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.70703 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.38086 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 283.04883 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 293.04492 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 299.71875 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.39258 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 313.06641 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 319.74023 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.4082 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\024) +[7.704 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 335.07422 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 341.74219 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.41602 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 351.08203 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 357.75586 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 371.08594 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 373.75195 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.42578 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.42578 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 393.09961 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.0957 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.76367 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.4375 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 419.10352 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 421.76953 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.44336 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.43945 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.43945 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 451.11328 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 455.10938 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 461.7832 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 468.45703 -54.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 213 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 215.66602 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.33398 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.00781 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 239.00391 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.67773 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.3457 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.01953 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 265.01953 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.01953 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 277.69336 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 287.68945 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 293.68945 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.36328 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.37109 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 317.04492 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.37891 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 327.04688 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 333.7207 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.38672 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 339.7207 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 343.7168 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.38281 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.38281 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 355.04883 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 361.72266 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 371.05664 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.05664 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 379.72266 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.05664 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 389.73047 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 399.73828 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.41211 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.41211 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 415.74609 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.41406 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.41797 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.41406 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 441.08789 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 447.76172 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 453.76172 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.43555 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 467.10938 -69.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 213 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 219.67383 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 233.00391 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.66992 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.34375 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 249.01758 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.68555 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.68164 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.35547 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.68945 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 273.68555 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.35938 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 293.68945 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 299.68945 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.35547 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 305.68945 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.36328 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.37109 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 325.70508 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.37891 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 339.05273 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 345.05273 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.71875 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.39258 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 361.06641 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.40039 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.07422 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.4082 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 387.07617 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 393.75 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.42383 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 407.09766 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 413.09766 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.43164 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.10547 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.43945 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.43945 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 445.11328 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 451.11328 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 457.78711 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 463.78711 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 467.12109 -84.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\012) +[8.028 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 215.66602 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.33984 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.01367 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.6875 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.02148 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 251.69531 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.36914 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 265.04297 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.71094 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.38477 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.38086 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.05469 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 295.72852 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 299.72461 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 309.73242 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.40625 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 323.08008 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 327.07617 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 333.75 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 339.75 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.41797 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 353.75977 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.43359 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.43359 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.76758 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.43359 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 379.10742 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 385.78125 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.44727 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 395.12109 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.45117 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 415.125 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 421.79883 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.80664 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.48047 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 441.14648 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 447.82031 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 457.82812 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.49414 -99.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.33398 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.00781 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 219.67383 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.33984 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.67383 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 237.67383 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.34766 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 251.02148 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.69531 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.36133 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 267.03516 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.36914 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 283.04297 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 285.70898 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.04297 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 293.03906 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 295.70508 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.70508 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.37891 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.37891 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.04688 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.38867 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 335.0625 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 341.0625 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.39648 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.0625 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 353.73633 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.41016 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 363.07617 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.75 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.08008 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 389.75391 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.42773 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 399.09375 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.42773 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 411.09375 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 417.76758 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.76758 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.44141 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 443.77148 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.44531 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.44141 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 461.11523 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 467.11523 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 469.78125 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 476.45508 -114.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.66602 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.33984 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.33984 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.01367 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.34375 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.67773 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.35156 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.35156 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 261.01758 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.68359 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.34961 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.34961 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 275.01562 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.34961 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 287.68359 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.35742 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.35352 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 311.02734 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 317.70117 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.03516 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 327.70312 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 337.69922 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.37305 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 351.04688 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 355.04297 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 357.70898 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 367.04297 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 373.04297 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 375.70898 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.37695 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 389.05078 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 399.04688 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 405.7207 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.38867 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.38477 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 429.05859 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 435.73242 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.40625 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.41406 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 459.08789 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 461.75391 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 468.42773 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 478.42383 -129.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.99609 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.66992 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.34375 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.34375 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.01758 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 249.69141 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.36523 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 265.69922 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.37305 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.04688 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 281.71289 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.38672 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 295.7168 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.7168 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.39062 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.38672 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 325.05469 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.72852 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.40234 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 341.06836 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.74219 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.4082 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.4082 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.4082 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 365.07422 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 371.74805 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 381.75586 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 387.75586 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.42969 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.42969 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.0957 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.76953 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 415.76953 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 419.10352 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 425.77734 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 435.11133 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 441.78516 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.45898 -144.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.33398 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.00781 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 227.00391 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 233.67773 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.35156 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 249.68555 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.68555 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.35156 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 261.68555 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.35938 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.36719 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 285.04102 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.71484 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 295.04883 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.72266 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.39062 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\015) +[2.256 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 311.72461 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.73242 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.40625 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 335.08008 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.41406 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 351.08789 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 357.76172 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.43555 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 367.10156 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.43555 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.10938 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.10938 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 389.10938 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 399.11719 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 405.79102 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.45703 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 415.13086 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.46484 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 425.13867 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 435.14648 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 441.82031 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.48633 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.48633 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 453.82031 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.49414 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 470.49023 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 476.49023 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 479.82422 -159.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 215.00391 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 221.67773 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.35156 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.35156 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 241.02539 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.69141 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.35742 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 253.03125 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.36523 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 265.69922 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.37305 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.36914 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.04297 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 295.7168 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 303.04688 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 309.04688 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 315.7207 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.7207 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 327.7207 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.38672 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 337.06055 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 339.72656 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.39453 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.06055 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 355.73438 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.4082 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.07617 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.41797 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.0918 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 389.0918 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 392.42578 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 395.0918 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 401.76562 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.43945 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 411.10547 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 417.7793 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.10938 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 437.10938 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 443.7832 -174.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.66602 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.33984 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.33984 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.01367 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.01367 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 241.68164 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.34766 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 251.02148 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.69531 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 261.69141 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.36523 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 275.03906 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 281.70703 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 287.70703 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.38086 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.05469 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 307.72852 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.40234 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.07617 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.41016 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 327.07617 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.41016 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 343.08398 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.75781 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.42383 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.42383 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.41992 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 375.09375 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 385.10156 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 391.77539 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.44922 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 405.12305 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 411.79102 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 415.125 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 417.79102 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.46484 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.46484 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 433.13086 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 439.80469 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.47852 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 453.15234 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 459.82031 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 466.49414 -189.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207 -204.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 213.67383 -204.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.66992 -204.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.34375 -204.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 227.67773 -204.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\012) +[8.028 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.66602 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.33984 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.01367 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 237.6875 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.02148 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 253.02148 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.69531 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.70313 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 273.03711 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 275.70313 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.37695 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.37695 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.04297 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 297.7168 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.39063 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 311.06445 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 317.73242 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.39844 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 327.07227 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 333.07227 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.40625 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 343.08008 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.08008 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 355.74805 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\013) +[7.356 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 363.75195 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 367.08594 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.75195 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.42578 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 389.75586 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 395.75586 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.42188 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 401.75586 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.42969 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.4375 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.4375 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.11133 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.44141 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 451.11523 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 457.78906 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 464.46289 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 471.13672 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 477.81055 -234.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.66667 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.67448 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.67448 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 233.34831 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 237.3444 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.3444 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.01823 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.35221 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.01823 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.68424 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.35807 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.0319 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.69792 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 287.37174 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.03971 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.04362 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.71745 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 315.39128 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.39128 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.0651 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.73112 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 333.39714 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.07096 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.40495 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.07096 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.07878 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.7526 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 375.42643 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.09245 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.10026 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.76628 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 397.4401 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.4401 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.11393 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.11003 -249.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.66602 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.33594 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.00977 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.68359 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 35.34961 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.68359 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.35742 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.6875 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.68359 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 69.35742 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.02344 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.69727 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.69727 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 91.37109 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.04492 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.71875 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.72656 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.72656 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.40039 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.40039 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.73438 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 139.40039 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.07422 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.74805 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 155.41406 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.08789 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 175.41797 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.75195 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 181.41797 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.0918 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.0918 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.75781 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 203.43164 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.10547 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.7793 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.11328 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.78125 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.56836 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 237.23438 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.23438 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 249.9082 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.9043 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.57812 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 275.91211 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.9082 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.58203 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 293.25586 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 299.92969 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 305.92969 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.60352 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.9375 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.61133 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 335.28516 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 341.28516 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.28516 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.95117 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 353.28516 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 359.95898 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.9668 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 373.30078 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 379.97461 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.9707 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.64453 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 393.31055 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.64453 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 405.97852 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.65234 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.65234 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 421.31836 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.98438 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.65039 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.65039 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 435.31641 -264.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.66992 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 29.34375 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.01758 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.35156 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.01953 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.02344 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.01953 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.69336 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.36719 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 77.36719 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.04102 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.71484 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.38281 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.05664 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.73047 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.06445 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.73828 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.73438 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.4082 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.08203 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.07812 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 158.08594 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.08203 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.75586 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.08984 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.76367 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.76367 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 197.43164 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 205.43555 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.10938 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.11719 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 225.45117 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.125 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.12109 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.79492 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.46875 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.80273 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.79883 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.46484 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 277.46484 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.13867 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 293.47266 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.14648 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.81445 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 313.48828 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.16211 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.16211 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.83594 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 339.50977 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 345.50977 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.84375 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 355.51758 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.84766 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.52148 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.19531 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.86133 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.86133 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.85742 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 401.53125 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.20508 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.21094 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\010) +[8.016 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.21484 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.88086 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 427.54688 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.2207 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.89453 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 447.56836 -279.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.66602 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 14.66602 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 21.33984 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.33594 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 29.33203 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.33984 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.01367 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.6875 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.6875 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.36133 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.03516 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.70117 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 77.36719 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.04102 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.71484 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.04883 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.72266 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.72266 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 113.39648 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.07031 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.07812 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.75195 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.42578 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 147.42188 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.0957 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.42969 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.10352 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.77734 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.77734 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.11133 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.78516 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.11523 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.11133 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.78516 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.11914 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.11523 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.78906 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.78516 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 241.45312 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.79492 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.46875 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 261.46875 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.80273 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 267.46875 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.14258 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.81641 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 283.48242 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.15625 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 303.48633 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.16016 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.83398 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.16797 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.17578 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.8418 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 339.51562 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 345.51562 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.18945 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 365.51953 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.19336 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.18945 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.85547 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.85156 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 391.51758 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.85156 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.51758 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 413.52539 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.85938 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.5332 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.20703 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.20703 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.87305 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 445.54688 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.2207 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 461.55469 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 468.22852 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 472.22461 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 478.22461 -294.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 2.66602 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.34766 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.68164 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.35547 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.02344 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.69727 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.36328 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.69727 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.69336 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 67.35938 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.35938 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.0332 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 89.36719 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.04102 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.71484 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.71484 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 115.38867 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.0625 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.05859 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.06641 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.06641 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.74023 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 155.41406 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 158.08008 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.74609 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.41211 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.41992 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.08594 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.75977 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.75586 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.42969 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.10352 -309.58984] Tm +0 0 Td +/F82_0 12 Tf +(\007) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.00391 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.34766 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.34766 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.02148 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.69531 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 51.36328 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.03711 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.04492 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.37891 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.05273 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.04883 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.72266 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 101.39648 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.73047 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.4043 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.07812 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.41211 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.08594 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.75391 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\015) +[2.256 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.08789 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.0957 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.76953 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 167.4375 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.11133 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.78516 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 187.45898 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 191.45508 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.12891 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 201.46289 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 205.45898 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 215.4668 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.14062 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.13672 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.81055 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.81055 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.47852 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.82031 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.49414 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 265.49414 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.82812 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.49414 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.16797 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.8418 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 287.50781 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.18164 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 307.51172 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.8457 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.8418 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 317.50781 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.18164 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.85547 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 333.52148 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.1875 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.85352 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.86133 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.19531 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.86914 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 365.54297 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.2168 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 374.88281 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 381.55664 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.22461 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.2207 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.89453 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 411.56836 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 415.56445 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.23047 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.23047 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.89844 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.89844 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 443.57227 -339.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.00586 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67969 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.35352 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.02734 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.69531 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.69141 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 55.36523 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.03906 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.03516 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.70117 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.03516 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.70898 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 91.38281 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.04883 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.04883 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.04492 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.71875 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.72656 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.40039 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.06641 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.06641 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 145.40039 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.07422 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.07031 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.73828 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.74219 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.41602 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.08203 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.74805 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.42188 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.0957 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 205.42969 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.10352 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.10352 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.77734 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.45117 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 241.45898 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.13281 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.80664 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.80273 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.13672 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 265.4707 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.13672 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.4707 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.14453 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 285.47461 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.14062 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.80664 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 297.48047 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.1543 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.82031 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.82812 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 323.50195 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.17578 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.8418 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.8418 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 341.50781 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.18164 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.85547 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.18555 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.85938 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 371.52539 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.19922 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 391.5293 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.20312 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.86914 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 407.54297 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.2168 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.88281 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.88281 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.88281 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.54883 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.87891 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 447.54492 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.21875 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.89258 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 464.88867 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 471.5625 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 478.23633 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 481.57031 -354.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\022) +[8.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33398 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.00781 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.67383 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.34766 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.02148 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.0293 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.70312 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.36914 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.03516 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.70898 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.38281 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.05664 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.72461 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 89.39062 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.05664 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.73047 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 105.4043 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 109.40039 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.4082 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 125.4082 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.07422 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 131.4082 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.08203 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.08984 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\037) +[1.836 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.75586 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 157.42969 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.42969 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.76367 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.77148 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.77148 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.44531 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.44531 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 201.44531 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.11133 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.78516 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 213.45117 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.11914 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.11914 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.78516 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.45312 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.12695 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.12305 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.79688 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 265.46484 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.13867 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.8125 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 281.47852 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.15234 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.81836 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.81836 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.81836 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 305.48438 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.1582 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.16602 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.83984 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.83984 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 341.50781 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.50781 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.18164 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.85547 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 367.5293 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.19531 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.19141 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.86523 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 393.53906 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.87305 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.54688 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 413.54297 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.21094 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\015) +[2.256 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.54492 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 433.55273 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.22656 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.90039 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 453.57422 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.24219 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 466.91602 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 470.91211 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 476.91211 -369.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.68164 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.67773 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.35156 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.02539 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.69922 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.70703 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.70703 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.38086 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 79.37695 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.04297 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.7168 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.05078 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.72461 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 111.39844 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 115.39453 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.72852 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.0625 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.72852 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.0625 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.73633 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.06641 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.74023 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 155.41406 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 165.42188 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 175.42969 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.10352 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.77734 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.45117 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 205.44727 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.11523 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.11914 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.79297 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 233.4668 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 239.4668 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.14062 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.80664 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 251.47266 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.14648 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 267.48047 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 273.48047 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.48047 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.1543 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.82031 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 295.49414 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 299.49023 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.15625 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.15625 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.83008 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.50391 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.51172 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.17773 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.85156 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.85156 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 353.52539 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.85547 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 373.5293 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.20312 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 389.53711 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.21094 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.88477 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 405.55078 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.88086 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 425.55469 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.2207 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.88672 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 437.56055 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.23438 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.9082 -384.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.33398 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 21.33984 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.01367 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.6875 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.36133 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.69531 -399.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.99609 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.66992 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 23.34375 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 27.33984 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.00586 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.33984 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.67383 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.33984 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.01367 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.01367 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.67969 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 67.35352 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.02734 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.70117 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 87.36914 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.70312 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.37695 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.05078 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.72461 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 113.39062 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.06445 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.73242 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.06641 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.74023 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.73633 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.07031 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.74414 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 158.07422 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.74805 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 171.42188 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 181.42969 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.76367 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 191.4375 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.11133 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.78516 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207.45117 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.125 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.45898 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.12695 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.12305 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.79688 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.4707 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 251.4668 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.13281 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.4668 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 273.47461 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.14844 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.82227 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 293.49609 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.16992 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.17773 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.17773 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.85156 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.85156 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.51758 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.19141 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.19922 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.19531 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.86914 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.20312 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.19922 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.87305 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.20312 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 392.87695 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.87305 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.54688 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.54688 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.21289 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.88672 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.89453 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.89453 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 437.56055 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.89453 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 447.56836 -429.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 21.33984 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.01367 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.00977 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.67578 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.67578 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 47.34961 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.02344 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.03125 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.70508 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.70117 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.375 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.04883 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.05469 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\010) +[8.016 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.05859 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.72461 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 107.39062 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.06445 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.73828 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.41211 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.74219 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.73828 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 157.41211 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.74609 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 167.41992 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.75391 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.42773 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.10156 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.77539 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.77539 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.44336 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.11719 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.7832 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 225.45703 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.13086 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.79688 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.79688 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.79688 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 249.46289 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.79297 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.79297 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.45898 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.12695 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.80078 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.79688 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.4707 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.13867 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 311.47266 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 315.46875 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.13477 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.80859 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.48242 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.14844 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.81445 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 339.48047 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.48828 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.16211 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 359.49609 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.16406 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.16406 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.83789 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.83789 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.17188 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.83789 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 397.51172 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.18555 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.85156 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 413.52539 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.85547 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 433.5293 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 443.53711 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 449.53711 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 456.21094 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 462.88477 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 465.55078 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 472.22461 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 478.89844 -444.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\012) +[8.028 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.66602 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.00586 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 27.33984 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.33984 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.00586 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.33984 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.01367 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.02148 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.6875 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.36133 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.36133 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.69531 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.36914 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.70312 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 93.36914 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 103.37695 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.05078 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.7168 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.7168 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.7168 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 131.39062 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.05664 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.72266 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.39648 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.07031 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.74414 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.41211 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.74609 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.74219 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.4082 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.08203 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.75586 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.42188 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.08789 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.75391 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 201.42773 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.0957 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\012) +[8.028 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.76172 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.43555 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.10156 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.43555 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.10156 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.10938 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.10547 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.7793 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.45312 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.12695 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.13477 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.46875 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.14258 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.80859 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 303.47461 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 309.47461 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.14258 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.14648 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.82031 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.82031 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 343.49414 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.16797 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.8418 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 363.51562 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.18164 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.18164 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.18164 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.18945 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.86328 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.85938 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 405.5332 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.20703 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.20312 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.21094 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.20703 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.88086 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 445.54688 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.2207 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.2207 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 464.89453 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 471.56836 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 478.24219 -459.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.99609 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.66992 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.00391 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.67773 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.67773 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.3457 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.01172 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.01953 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.69336 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.36719 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.36719 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.04102 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.71484 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.71094 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.71875 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.71484 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 115.38867 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.0625 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.05859 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.72461 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.05859 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.73242 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 147.39844 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.07227 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.73828 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.07227 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.74609 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.41992 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.42773 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.09375 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.76758 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.43555 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.10156 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.77539 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.44141 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.10742 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.78125 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.11133 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.11133 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.78516 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.78516 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.78516 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 267.45117 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.125 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.79102 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 283.45898 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.13281 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.79883 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.79883 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.79883 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.79883 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 313.46484 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.79883 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 323.47266 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 333.48047 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.81445 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 343.48828 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.48438 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.81836 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 357.49219 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.82227 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.82227 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 373.48828 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 379.48828 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.16211 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.1582 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.1543 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.16211 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 407.49609 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.16992 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.84375 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.84375 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 429.50977 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.18359 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.85742 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.85742 -474.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.99609 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.66992 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 23.34375 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 29.34375 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.01758 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.69141 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.36523 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.69922 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.0332 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.69922 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.37305 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 77.37305 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.03906 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.71289 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 93.38672 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.06055 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 103.39453 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.0625 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\037) +[1.836 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.72852 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.40234 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 125.40234 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.73633 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.74414 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.74414 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 147.41016 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.07812 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.75195 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.74805 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 177.42188 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.08984 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.08984 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.76367 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 203.4375 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.10352 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.10352 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.77734 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 221.44336 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.45117 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.125 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.79883 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.46484 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.13867 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.80469 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.80469 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.80469 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.4707 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.14453 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.81836 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.48633 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.82031 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.49414 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.16016 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.82617 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.16016 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.15625 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.83008 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.16406 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.83789 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.17188 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.16797 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.8418 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.51562 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.18945 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.18945 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.86328 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.19727 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.19336 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.86719 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.86719 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.86719 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 433.54102 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.20898 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.88281 -489.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.02148 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.01758 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.69141 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.02539 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.02148 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.0293 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.70312 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.69922 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.37305 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.38086 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.37695 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.05078 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 95.38477 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.05859 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 111.39258 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.39258 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.06641 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.06641 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.74023 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.41406 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 147.41016 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.08398 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.08398 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.75195 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 175.41797 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.0918 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.75781 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 187.42383 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 197.43164 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.10547 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.7793 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.7793 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.45312 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.12695 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.80078 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.80078 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 249.46875 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.14258 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.81641 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.8125 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.14648 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.1543 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.82031 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.82812 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.82812 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 305.50195 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.17578 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.8418 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.17578 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.50977 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.17578 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 333.50977 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.18359 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.85156 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.85156 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.85156 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 365.52539 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.19141 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 374.86523 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.86133 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 381.52734 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 387.52734 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.20117 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.875 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.88281 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.88281 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.55664 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 429.55664 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.23047 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.9043 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 449.57812 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 459.57422 -504.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.00391 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 14.67773 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.67773 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 27.35156 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.02539 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.69922 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 47.37305 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.03906 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.03906 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.03906 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.04688 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.71289 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 77.37891 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.05273 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.72656 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.72266 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.73047 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 111.4043 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.07812 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.75195 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.74805 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.41602 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.08984 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.75586 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 157.42969 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.42969 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.10352 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.09961 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.77344 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 187.44141 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.11523 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.78906 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207.46289 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 213.46289 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.79688 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.4707 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.80469 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 239.47852 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.14453 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.81836 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.81445 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 265.48828 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.16211 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 275.49609 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.16992 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 295.5 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.17383 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.8418 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.8418 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.51562 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.18164 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.85547 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 337.5293 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.86328 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.53711 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.87109 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.87891 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 367.55273 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 374.22656 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.56055 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.23438 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.9082 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 399.57422 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.57031 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.23828 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.58008 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.25391 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.25391 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 439.58789 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.25391 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.92773 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 455.60156 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.26758 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 464.94141 -519.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.01367 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.00977 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.68359 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.69141 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.69141 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.36523 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.03125 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.69727 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 69.36328 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.36328 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.0293 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.36328 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.03711 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.71094 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.37695 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 107.38477 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.05078 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.72461 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.72461 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.05859 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.73242 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.06641 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.74023 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 155.4082 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.08203 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.74805 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 171.42188 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.08789 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 177.42188 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.0957 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.76953 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 197.44336 -534.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.00391 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 14.67773 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 17.34375 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.00977 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.68359 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.35742 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.69141 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 43.36523 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.36523 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.03906 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.71289 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.7207 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.7207 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.38672 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.05469 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.72852 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.72461 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 111.39844 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.06641 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.0625 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.72852 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.72852 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.40234 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.73633 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.07031 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.74414 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.74023 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.41406 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.08789 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.42188 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.0957 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.76172 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.76172 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.76172 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.75781 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 221.43164 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.76562 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.76172 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.43555 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.76562 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.43945 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.11328 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.10938 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.44336 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.45117 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 285.45117 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.125 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.13281 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.13281 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.80664 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.80664 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 327.48047 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.1543 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.82812 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.50195 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.17578 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.84375 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.84766 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 375.52148 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 381.52148 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.19531 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.86914 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 401.54297 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.2168 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.88281 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.88281 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.88281 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.89062 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 439.56445 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.23828 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 449.57227 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 456.24609 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 462.91992 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 466.25391 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 468.91992 -564.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.66602 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.66211 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.33594 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.66992 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.33594 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.33203 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.00586 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.67969 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.67578 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.34961 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.01562 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.68945 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.35742 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.02344 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.69727 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.69336 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 101.36719 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.69727 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 121.37109 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 131.37891 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.375 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.04883 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 151.38281 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 158.05664 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 167.39062 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.06445 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.73828 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.4043 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.4043 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.40039 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.07422 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.08203 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.75586 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.42383 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 233.41992 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.09375 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.42773 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.42383 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.09766 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 267.42773 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.76172 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 277.43555 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 281.43164 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.10547 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.77148 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.10547 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.7793 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 309.44531 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.11914 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.11914 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.79297 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.78906 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 339.46289 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.79688 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.46484 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.13086 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.80469 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.13477 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.80078 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 387.47461 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 393.47461 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.14062 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.81445 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 405.48047 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 415.48828 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.16211 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.83594 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 435.50977 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.17578 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.84961 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 451.52344 -579.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 17.34375 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.01758 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.01758 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.68555 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 43.35938 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.0332 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.0293 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.36328 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.69727 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.36328 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.69727 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.37109 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.70117 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 87.375 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.04102 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.71484 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.71484 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 109.38867 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 113.38477 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.05859 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.72656 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.40039 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.07422 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.74805 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 153.42188 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.42969 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.42969 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.10352 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.77734 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.45117 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.125 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.13281 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.80664 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 219.48047 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 225.48047 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.14844 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.81445 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.48828 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.1543 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.82031 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.49414 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.82422 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 275.49023 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 285.49805 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.49414 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.16016 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.16016 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.83398 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.16797 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.16797 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.8418 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 333.51562 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.18164 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.85547 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.5293 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.19727 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.87109 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 365.53711 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.87109 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.86719 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 375.5332 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 381.5332 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.20703 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 397.54102 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.54102 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.21484 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.21484 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.21484 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.88086 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.55469 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.2207 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.88867 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.22266 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.89648 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 453.5625 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 456.22852 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 462.90234 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 468.90234 -594.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.00391 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 14.67773 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.68555 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.35938 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 35.35547 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.0293 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.0293 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.69531 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.36914 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 67.37695 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.37305 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.04688 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.38086 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.37695 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.05078 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 105.38086 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.71484 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 115.38867 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 125.38477 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.05859 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.73242 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.73242 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 151.40039 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.4043 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.07812 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.07812 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.75195 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 185.42578 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.09961 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.77344 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 201.43945 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207.43945 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 213.43945 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.44727 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\037) +[1.836 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.11328 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.78711 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.78711 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.12109 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.12891 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.79492 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.46094 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.13477 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.80859 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 273.47461 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.14844 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.81641 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.48242 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.15625 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 299.49023 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.16406 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.16016 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.83398 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 323.50781 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.83789 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 343.51172 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.18555 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 359.51953 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.51562 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.18945 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.85547 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 381.52148 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.1875 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 393.52148 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 399.52148 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.19531 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.86914 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 419.53711 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.21094 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.87695 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 435.55078 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 445.54688 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.2207 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.89453 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 462.22852 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 468.90234 -609.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.66602 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.68164 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 35.35547 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.02148 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.69531 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 51.36914 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.03711 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\015) +[2.256 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.37109 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.37891 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 77.37891 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.05273 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.72656 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.40039 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.06641 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.74023 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.07422 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.07422 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.74805 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 135.42188 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.08789 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.42188 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.75586 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 147.42188 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.75586 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.75586 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.42969 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 175.42969 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 181.42969 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.0957 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.76953 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 193.43555 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.76953 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 203.4375 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.10352 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.77734 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 221.44336 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.10938 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.11719 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.11719 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.79102 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.79102 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.46484 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.13867 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.13867 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.13867 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.8125 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.82031 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.49414 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 305.49023 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.16406 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.17188 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.8457 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.51172 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.18555 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.85938 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 351.5332 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.20703 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.88086 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.21094 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.88477 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 387.55078 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.22461 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.89844 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.56445 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.56445 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 415.56445 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.23047 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.56055 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.23438 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.9082 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 447.57422 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 456.9082 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 463.58203 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 470.25586 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 476.92969 -624.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.02148 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.69531 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.36914 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.03711 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\013) +[7.356 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.04102 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 51.375 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.04102 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.71484 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.04492 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.71094 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 79.37695 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.05078 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.72461 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.7207 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.72852 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 109.39453 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.06836 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.06445 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.73828 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.73438 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.40234 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.40234 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.07617 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 158.74219 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 165.41602 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.08984 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 175.42383 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.09766 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 185.43164 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.43945 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.11328 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.78711 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.79492 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.79492 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.46875 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.13477 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.80859 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.14258 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.81641 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.49023 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.1582 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.82422 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 273.49805 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.16602 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.16602 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.83984 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 299.51367 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.1875 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.86133 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.86914 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 329.54297 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.2168 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.22461 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.2207 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.88672 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.88672 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 365.56055 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 371.56055 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.22852 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.89453 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 389.56055 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.23438 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.23438 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 411.56836 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.24219 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.91602 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.25 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.92383 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 441.59766 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.26562 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 451.59961 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.27344 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 464.27344 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 466.93945 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 470.27344 -639.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 21.33984 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.01367 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.01367 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.6875 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.69531 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.36914 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 67.37695 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.04297 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.70898 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.04297 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.7168 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.71289 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.7207 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.05469 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.72852 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.72461 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.39844 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.07227 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.74609 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.41992 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.08789 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.76172 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 157.43555 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.76562 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.76562 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 177.43945 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.11328 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.78711 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 197.46094 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.12695 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.13477 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.80859 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.48242 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.48242 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.81641 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.8125 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.48633 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.1543 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.82812 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.50195 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.83203 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 273.49805 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.17188 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.17188 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.8457 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 299.51953 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.85352 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 309.52734 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.86133 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 325.53516 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.20117 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.19727 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.87109 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 351.54492 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.21875 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.89258 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 371.56641 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.56641 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.23438 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 392.23828 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.23438 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.9082 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.58203 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 415.58203 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.25586 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.92969 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 435.59766 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.93164 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 445.60547 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.2793 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.95312 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 461.61914 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 468.29297 -654.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.66992 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.33594 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.33594 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.00977 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.34375 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.01758 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.69141 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.36523 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.03906 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.04688 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.7207 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.7168 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 95.39062 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 101.39062 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.05664 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.73047 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.73828 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.41211 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.08594 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.75195 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.75195 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.74805 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.42188 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.0957 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.76367 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\015) +[2.256 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.09766 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.77148 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.10547 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.7793 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.45312 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.12695 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 213.45703 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.13086 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.80469 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.80078 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.13477 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 237.46875 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.13477 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.46875 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.14258 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.47266 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.47266 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.14648 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.14258 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.14258 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.81641 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.15039 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.81641 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.48242 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.15625 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.83008 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 317.49609 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 327.50391 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.17773 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.8457 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.51953 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.19336 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.85938 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 363.5332 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.19922 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.19922 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.19922 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.86523 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 387.53906 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.21289 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.88086 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.54688 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.2207 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.88672 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 421.55273 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.22656 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 441.55664 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 445.55273 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.22656 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 455.56055 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 459.55664 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 466.23047 -669.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.6875 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.35352 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.35352 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 37.35352 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.01953 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 53.34961 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.02344 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.69727 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 69.36328 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.69336 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.69336 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 91.35938 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.02734 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.70117 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.69727 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 121.37109 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.03906 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.71289 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.38672 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.05273 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.05273 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.71875 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.39258 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.06641 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.40039 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.06836 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.73438 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 191.4082 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.73828 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.73828 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.41211 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.41211 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.07812 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.74414 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.07812 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.75195 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 251.42578 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.75977 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 261.43359 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.10742 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.78125 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.78906 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.46289 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.13672 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.81055 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 311.48438 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.81836 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 317.48438 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.1582 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.83203 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.16602 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.83984 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 353.50781 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 363.50391 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.17773 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.85156 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.52539 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.19141 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 395.52539 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.19922 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.86523 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.19922 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.87305 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 427.54688 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.54297 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.87695 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 441.55078 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 445.54688 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.21289 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.88672 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 461.56055 -684.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.99609 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.66992 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 23.34375 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.67773 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.35156 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.35156 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.01953 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.69336 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 59.36719 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.36719 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.36719 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.04102 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.375 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.04883 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 95.37891 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.375 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.04102 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.71484 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 111.38086 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.38086 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.05469 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.7207 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.39453 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.72852 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.72461 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.39844 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 165.39844 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.06641 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.07031 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.74414 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 193.41797 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.41797 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.0918 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.75781 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.42383 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.09766 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 227.43164 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.10547 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.77148 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.44531 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.11133 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 249.44531 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.11914 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.79297 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.4668 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.13477 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.80859 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 285.47461 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.14844 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.15625 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.83008 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.16406 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.16016 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.83398 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 335.50781 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.18164 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.18164 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.85547 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.18945 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.18945 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.85547 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.85547 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 385.5293 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 389.52539 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 393.52148 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.19531 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.86328 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 413.53711 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.21094 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.88477 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.21484 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 443.54883 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.22266 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.21875 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.89258 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 463.55859 -699.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.33398 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 21.33984 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.01367 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.6875 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.36133 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.0293 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.02539 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.69922 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.0332 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.70703 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.70703 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 87.375 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.04102 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.71484 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.04883 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.72266 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.71875 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.39258 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.06641 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.39648 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.39648 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.07031 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.74414 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.74414 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.41797 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 175.41797 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.75195 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 185.42578 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.75977 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.43359 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.76367 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.4375 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.11133 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.11133 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 225.44531 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.11914 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.78516 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 237.45117 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.125 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 253.45898 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.45898 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.13281 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.13281 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.79883 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 281.47266 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 287.47266 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.14648 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.14648 -714.58984] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +Q +Q +q +1 w +[] 0 d +0 J +0 j +[1 0 0 1 36 805.89] cm +q +0 -250 200 250 re +W* +[200 0 0 250 0 -250] cm +/DeviceRGB setcolorspace +<< + /ImageType 1 + /Width 512 + /Height 640 + /ImageMatrix [512 0 0 -640 0 640] + /BitsPerComponent 8 + /Decode [0 1 0 1 0 1] + /DataSource currentfile + /ASCII85Decode filter + << >> /DCTDecode filter +>> +pdfIm +s4IA0!"_al8O`[\!!#/mE=]te*!AFKR!#0X!E-) +'[!CQnf!#/pV@:T?&sZDXJPst(FE6e=s*eFz-l;pa'itA8\2.5i=s*eFzo25Ph!!!! +"(=.&61GSq1!!!!"$pjpo"lB:Zo-OG;#Ef&erK7-[qkF,jrM'>m"5EkV^hdM'EcqE +_z!!*-W!!$kPF^kCOz!"T&0!"]Hh92Y`i!#P\9!]#(8<*'&"!#bh;!A\t76V[U]!#>P7!Q4t2!+>kE!+ +c.N!+>kF!$D8>!(R#U!%e1i!-/'X!+l4F!,2FL!,2F2!(R#U!%e1i!-/'X!+l4F!, +MWX!+l6)!-/&^!)!<#!,V^I!-/'J0WQnd0^/Q[!)`ee!($Z.0\8#G0\%kN0VgDgAU +%!FG5,fZ!$D8>!(R#U!$Hl6?YO#rO73-G!+c.N!+l4F!,MWX!)`ee!($Z.!,qpH!- +/'J!$D8/!9!op!+>kJ!+>k*!(R#U!%e1d!+>kE!+c.N!+>kH!-/'X!+l4F!,MX5!( +R#U!%e1I!-/'X!+l4F!,MWX!+l6/!-/&^!)!<#!,V^I!-/'J!-<>YB0$#8!)`ee!( +$Z.A'^5*AVX%S!)`ee!($Z;!+Gq>!-8-U!-/'R!-S?R!,MXV!+c-Q!-A3T!,MWX!) +!<#!,V^I!-/'J!)`ee!($Z;!,qpY!,hjL!,2FE!,MWX!(-`p!,V^I!-/'Jc?-Ws[s +S/u!)`ee!($Z.eHEBedgX9E!)NZ,!-/'O!,2FL!$D8>!(R#U!$D8P!+c-Q!(-bG!, +V^I!-/'J!)NZ9!,hjL!,2FL!,hi[!)`ee!($Z.!(Hs,!-A3Z!+Q";!,V^I!-/'J!( +-`p!,V^I!-/'J!$D8>!(R#U!$D8!* +0(d!$D8P!+c-Q!,Nbq!+>kH!,qpH!-/'N!,2FL!%e1i!,)@N!-A3Z!!%+PG]Woc!! +#B)E-ZJF9hbU;!!!!)!"&`;!!3-#!!WE'!/ +:Lc!!3-#!"Ju/!/ppk!!<3$!!*'#!!!$Bk(pXBl +.E(2_Hoe1,(CA3\WGQ&2(gT`!#^BW!!N?&!! +*'"!Y0gQ!!N?&!!*'"!Z$S9!!<3$!!*'$!!%g&!!<3$!!*(m!!&)\!!`K(!!Fu21, ++dl!!3-#!#,D5!ZmF+!!3-#!#,D5!]#l=!!`K(!!E9%!!&/_!"&]+!!*'"!_/=S!! +N?&!!*'"!`"m]!"&]+!!*'"!`kHh!!<3$!!*''!!&/g!!<3$!!*'2!!&/h!!N?&!! +*'"!a_N"!!`K(!!Fu10JK.#!!<3$!!*'#!!&Yn!!E9%!!*'"!W\kq!!E9%!!*'"!e +@!T!!N?&!!*'"!bS/;!!N?&!!*'"!cF_D!!<3$!!*'$!!&eq!!<3$!!*'"!!&er!! +<3$!!*'"!!&es!!<3$!!*'"!!&f!!!<3$!!*'"zz!!*'"!6>-?!!rW*!!+i/0K2$I +0fh$H+>Gf92)[KL1&s'`0K2$I0fh$H+>Gf92)[KL1&q:S&Hr.7!]g;]#$1d)!<<*" +z!!*'"!#tt=!!*'%OFR<2!':/_OFR<2!'C5>!hj:LFDs8o05bh`@:X:cAM.J2D(g- +BE%`pu0JL\FDF# +`=A;U76Z7!V+>b]-/heJ5$6UH64E=tE3`8@8+F%a>DK@jZA7dtKBQS?83\N.1GBYZ +`1G3TdB.ku"3B8`H1+tC1,cGWJ<.Ec$#/Eb.EA+. +C04B!J3_!poF(dpZD/a'(F('0)Df.`G+1_cAi`gX7Qq/A06fO&D/a?'FC@??!<:dR6N@/g!%;)SAnPdkC3 ++K>G'A1VH@pm)L51SAMNX0fQ'Rc(R@9kFUnsrdW2Zf&Za@-K\%&u[_Sa=2`lH0Bb0 +nbge^i@)g"PEEj5f=akNM0qnac;Dp%J.Tq>1-0!$;4A!s&E'!/9Ll%"PPE_>uQ^F20U8+N\VPgAlWj8t<[C3QS\[oYq` +5Ta:aN;TQe'uq!f@\d1ioB+]k3(smlLOQ=p%J.Tq>1-F!"8r1!!3`7&HG#qr%bV*\ +&O0ibT:BE2\(B1c_Y?RKs]E=-J'g-G3bNP?bkg&a*`rG>ckghAguhnF+?=C:Ot'0? +uAk63T3=`5;>D,`e+C-m#(WIPXD#3N!&n(_r7Pkrr>u?&8pHo$CP6h-tAT2[PG!Q: +=h7eDdnB-A(NRHG'[?6\4>#=\MO>!e&&+3ce>K)d6+]6!9bQV`1IXS3&IhFg\;/n8L@"b1tOAT5?5`D@+-C= +K=g-(5tM(+s_@Ecj#_A$uE)"ZX.D9Z5!+:!lR$sp(ZmM_&#"Q9Ph+m-A)D/&@7&+i +rC"CO2P)-l^0::3TL4"e'F#>bN61B?@TJTM?)APW5W>G45GSkkJH>%T9hdbJ.cdT' +4&p,ZIC"c36hPcEE)k\;:IcO>\Ud7Krrn]\f:/9?'P9!7X5m?g9&?RLiiU8dSh4Gc +IM21#@>)\F&RF;cK`#\c]UP"5QK.96#Z8Glr$c)4gb+<$:I\\qi=ZMMrERIMO:R+J +j%iib,cO'VP6j*)jqBtbbtO&2F+6Qeh&LZ[+@`PA)D?EBXJIu!)XuL;pq)"P1[^'T\a +(!F$"HWW$2q-K24%lRG)eB'rZC@AIP*-N.>ZLENl?Ii:h)@CUO#.Ka>3A+_&`BM\P +A$Sq7\f!J57YpPb:E@"edL6rU];7XPB$.K+.I8i8'?pi^4Vci+0upB.sPjhQcGX?8 +K5Cqi+k8!f8+Ao"3>'%2-KS9;bfu,3)-Za9F&hpSh!%E<]AE?G,_N/(.9KMFH@4`7 +LpQ.,.\<=%or+;L-.t;C%)Mi%uL3_Vd*FD/hT$-%^(jE[%Y8IWWIR`=)m(X'uB/cJ +^_8!,hh;ll^%iVM?6@K)_qp-Ih=pT\dTehZPQKh6/d+89%`XX[]?488Yndc1`(.WK +k-\uFiQN,o]o[%*kLjj3[&j>.u1g(Gs&$`DBV$o)81>+4Y*#5nG]gJUF4a&39%D]2 +^pH_,6t[4B905UTF$/To^,5IE9tlS*&d7Mfc=! +.:2?Nl=2LU)8N\)ths4Var"%c2:J>epRU33FMXOPAJV1`^NL+?>:`fp_T%GoZR +T@+tpb,=koPYT111+cT;d2nj_Q5UV"cO'>\O(upMG+O4K+cpn5bJ2A_)a,/2MiC'b6ad[4!s@>M[A=>QN37kqp7I36MeFa)o&T[8.gn5jK +(f6maFcR[UmqL+g7u4OX[b"8=$_f_7PY7"hkLEX,<^FpjUTF-eU6 +VFc9&D^X:,RE%IpAOD?LE;_OIGmP3E#-UT0&h%60qlaKn.'g*("_)_HcAn[9XjNH3 +i@U5Y2k^oghu.5CjT!`o.GjQJF+J,]UT#EY>F^QlmqUaioZ_W2oF!4M$]Ci4Gt\JKQt*rIE +3b=dh8tVZR(*TaOpK>BEcRI%LI.$,B+(?kmK^E*9r;Vr5,04nRk/NEb_L]()!E^32 +:VDGgM,69atekg3tqld,<"SGPE%^''r0bafk9O;'_W8HQDSj1SAY.2;b%[Rj7b-/$VpPS*\&J3&*rFR[k6-i>2U" +)t%F\_QNXl8t3dRGiW-c$uOW]LNl,;78!Z+e/STg3Rh/Do1rFHqO1'.=['o$iF/e7 +@)ldq:[`^EMu@4k62:!NN0ee;L%Y:M;lk7UZ9\;lW;U6BCHl"KnZ))D":k:pfdFY[ +d(L@Q:TQEA))qAiTJd6mgqh"H?l5C3"nf:6]SZoH5blBF(_AHXA,J9CE(.F$n7.DH +CcXipi#.6)LQ&.SN[qh44HA_F;L>:D(=jNP^if!$qr\nGpA8of%uDE?])F[hs>ZT\je0UF-qFWl/a3a=(%01ZdK=kg-?DS[NVf_K68j'JAE;=eB$e5,)k +*)_3nSG\')nitSY350b?n-a%P_+=)q,^RtHpmLCJK%?k&pSTraYg?;af`!0J6X_oR +>9(I)H""qPRY_:W;sgPa,NB-_5,q$g3)NBbTSl_@H?2[-\W,A=*?SG.]HlJ!+aFYY +LLZNpNAJ>GT&0J;NT)3Ih;N./&8_@Bf`*GQSG*^6jj.#_^ki\n(PH!nfJYBBHQX1K +g$]b%]n^7n?qm-YVO./%YMH'pcHObnHp]MKY.V@WA5KS2#T1FYVIs=mLd7mk6-f*k +]9!/VQ$`_g4R3GT^Sf6>?hIPb*$_ATHg>mqAQKr%2.R:]##hJj&opHdbN+eJ2)aRD +d)d9rZ+b4\"4`;;rYl-@PK^2)8B1:\mW"2JpLn1I!=SP/&=/rgJZBU*?OcIRA.9sR +aQ"uP`RD4mCO4.%R3bZm\QbELK(Vn.ZK6%$9im!HIMAmNao^JFX(g9p3+M.p^BaTA +,4[0:DN-J[C=jk77dmuL`G3?kGC^p*`(`tqV56aakKY=MTlQ/,7Ztb]5\(+W!1*?#f%'KT!\cq=.C/34h3A`L'`]#pEcK>\ps2F[' +RB.i]7hP6E'X,tTA&!m_r[%n,G<3?r\OWlKd74F?RokOU30&l&f"b[CZlOn'%*o=I:]U;D<9T6gI9A:Qch5r%ip%C;Vk/P&1BYa +gsh`m9Q`fi&@@KH[D:l%gl?_"l2EDO!\GK\\0!4iCp5a+m[R^DI +W]#VpP!+^eD\l[@uB9kic@9)o=$'s9OX!ZEQ)jDC9[]dn<@+2S`TgA">!0:>:]aVn +.YoplrF`k_uS'&"4jBZK'(L21,M#US8nQ/l`432,4!-mXn7.!Y\4ad:-nInC!Y6CV +JK9"[lJNLHI)j'bj\7MbCWr]Vn^ZO2'4h[bCU#nN&Cl3r+krNrXIW^!.n-s1B.?+Q +2%QB9)d8%GbePg<]a*^-BG.6NqS`*2jhe3FVq;err@YJJ&)CS&q7s[[J0%LiST8HS +i+4Nf>K7TFRKiG?0[DlH6I'kO^?mB!Jlj(]m8m'!e-9_.[Zfd<0 +\FAQ%P.BnrLq!agi(i7p.tGsih)$6*,e"el+Ru'SYWiWTm?Y<#u03<7hYh%$:]e+3 +B1#8$"?0'ah;WPrrDBdCZ9b9BO;(\BpnV`4%F/Mpu]lX_?#V$1rYTd"7e/lF>uBZ# +Nr#='?C\70&F.a"Yr55.+jcM0RJ3-O2AS[i+J$u&pRdDr7A\"]HQ;7!"aAdgg9f"( +0^S!g6\,ORp^+7$+U^"+mA951Q9r7hBDikrr?JpFA'+4rrBj\NlLoA;f45%-UP&YG +)V8\AL0feNM)ej;uuJiSh:j'.*TSR`]4:=d[MT)Q%4uaWoS\U(%h7fL%.7uAap;KD +)so72rWr=h[asp46;XfIh;Cdhc$nF!iL&iEOd6D+GM,2@,eTUplYFu-g\sS=[hiWj +.Q1,F5cG@D)E."4fdg*\tOoc-#6pAq(78T/dTO*iA:T+S.dI)%NU)>)!@Qk-:C9T` +DLf;Q>S5#5@e[EHn"RRML7rZC@;].dp1:<]J4O5R(aKg;UY-b=!\7g\WT?^[K&i>5 +FA(-L#D.X5PQVX^u'\B!4C_^'F2FF#/%A6`f]3J&8;pRL-810&Dp`..f_>XRMIt.O +gc1S[W*5f!MBs]elYQXXG!343N8Z%3Q,GJFlt`fl,!B"i7-Z;b"+j`pk'N&_(4Nua +j,4jT>JEBA1)Jf0A3C0A2OddQYoaW-hRMTB8@O:D1:Eg^"uPK +"M^XF,jo9q2gMEE/Y8dqH$JQ!t!=k]XEk&D8_(Jq([M/XWOoor"*f)n%fNp56nPGX +^fC]9^rSHg$=Z@6IRG:d#6sErRhNZV?bA7o2#-,rrDs/TAdQ_'F4?+rLuC^nK2*-Y +*5^[,h?#N8=N5bN]HN;rm]FLp=qI!#L$4G%DW(3F.A[!r5mB0":WZ&(",sq4]IJ7P +/c<1D!otD"]]/A5X(J3!;o7b!/!d*#C&nXqX1I-`6XAXWG*'uWmY5,8P4H,TGG +VfH\Me!l?=3i^i(2X98HhG@,^jj1*J)K.$!<3&84T>CjrG1D"Jc6g;n&.8YQ`B;]H +rc\h<`.SV>hqmPQh$eNAfJoDA7GTNH-bS!T:eIr[cV3Jn^`4NAVcJoj.F>hq\+6ZN +^*]:p8qrU@sIR4q&&;d`Hn0CDu2IZqcA6qILGq#WG8m\27JG2>0]dN@9Hpmf87se0K[!YRLAW't<-0'\2i#/oF:^9 +T7^-n1`)!BDpQ\g\qK1rr<2cdD+V;hmMUAnB8j9,h2`@@A\9Eo>E//@;-J#mj*)g_ +N>S$=TT[&R%>/Q!KLgqj84g"*B/4oU1Xr0EVHg;X$]LJ3:9.!"+Wna#;uA6qg8>C_ +6,]2Z+oXnnH7cHLH[2&*\r;h>NBesgF46s`iTH/u8nr9O3Yn:1?IbP#cA(PZ>Gr$gQLD +![j#7iL8UEUHE[O1DHFomc`a(W4Fgpc+gn%GkfFN*d!jpgW)*L;bcac[)l"B(L#S! +&44iN^Wqs,/1Sf>c`fGIVoTdC#UX'gLO@s\P>buJ[,!!+/OHc;h)Ugrr?\YK`s:b@ +0lps07b>i.FVgd;^hgN>N+HI4/O#kDEpb76Os%WT8h`)^%^Y+Ikpg:X\KVH90di +]r\Bqn(=#duJA_!RNa5UOdca?@kLhUC^M+[2"B4&95e#`V!IJ?1hQ\p52> +"`f'ho7i1cu%m/gb_'0^O9T(OT">4mH6pnHBX;OaYgC`m:5[@in@UFnTWuW&5.^gO +2cL@X06YVP2Xf>kX-d%'47o/?gqb=Jp'%QL&4JU&i8bO!#tna5pWB&Y+!9Pk/[Dl5A&+=!5OGpU&P+#+8?upVuH`#$ +Lf-rp&>!lGi$W]n9?b.2guL5hb]<.Q$6iBlAZI. +_mgZUgu4B\PLk5rX+q\\,H]OnA=q:5I0.RT]oV`rh.uHn6U?_*C<8eO\iG)6^1I8UIrF5$+7N3))e_:im58>1Qo,ro=\kr"c/IJ%o](!$Y +)[tY\jnBqd>bX\DHj8gebu.*i1*"I5@n18g1]@^oU'aF:AVs&)gc+fc=pZ_8UQ)?J +>XX$j'F14h[?JQ\@#6)aFV`i(a+cukA"! +s2N>Ba7AQhLI)#-9HI"0%4&p#r8qacuWBFgS4M +,mKS4hQnHbJ9<7\O4uJaIj%GQbn&i-kC+ocNA1HWlq&Lb:lms<_cHZ)`[O/YB8C2@ +/g3KG[jI?:tI)"L=4U28Fg!1Rn'Y4>i(J;/U3>qf'e%GMk)'c.+lOMNSa_rTUi`)J +3:R/6FMI:$s-&]IV'67572,oe[$fW[G#I;@UmaZf<)2'&:[2k[kZuNTT@T^>Hmp*4CWra_B_Wk +Ul3T/e-9T@9`e6.tPe/?13i]QF?QQDZkcZm9kmWcNK2>lWgoU(KL6JOkKN4b$?W8a +Ssq3?'3?g#Z+L",EdE*A`C`kLV\$W>GTN6B7"dHH/mJ+4:=XS4s'KM!+[.!o.PMkWkQR8'Z% +tg/dc+f+J$fR$2:JfNJ/NK!R&Il^]b,c6ba)XG&9R0=N8K#N:HjBZO"<.:?Jmin.r +X+qN:PmpLpof1GC!h0+b0'_a:S-!HNRE#_&dJMo?%F]-N0B_(Dtkj_!;mEN_B6!^n +%a:"J\&%AA\%T$ictA[Y0V64m'b"eO[bNY+i8b@q2fRUKB`G"(]Ferrr<5In-Ak*+ +8-QA[t+JYB1ne9a%&dn(?\LTNgD&^g;H^6&Tj)X.ab9%oYsuc:)/M5%:$0nAlO%$- +:_uT-R\R6^N'8Ri3@e$5d,i*n5#\O,VP"'+a?TOSPS5mKkJgY>u@dNMUph5^Ok<&EBjq/gR!;%a5`WV5A-%6E +dSGngYk@7C!1U9]J$EiCCqkoV'L,!Efq.B+FMT4aiVBVcQ(ORY,lr1nZ?^>^p<6KZ +*;?=i[D>//ckn[!.R:\chnfIiD10C5pqB!/`JDofY$Fs4I/Wa4HL0H&/ELB#EY;ui +uL[o:AD0q!!V?[R/[/&0A1urSf7,PMKN28>e>Mj>2&9\>ag)cfmf:JK5`=*,:m+d4 +$g\#ik/Xt$"s8<#4$XRrMokt@6]G&'>>dYIP2V;N&2sU/,MtRdO56W:.0J?!K-W4O +.YoZ:[J&LX7gB"ritBC%i%[q*A9aj][Pu/ClN*&u%#*Xr*i+0_[@'[ttnl +q-/QHbt(FM**5[petJ1O'^7W%8Tts9fWd)Yg!e>0.bsj?>#R]:*#09(&#\cDtke0_ +#FCj*X%O.d68iNB>eskn,EA]dC757*n,jA8]`3Irr=PH[f6>Z:]CF2$++X*'B&V94 +t(m,\u)1%kr7#Gl*KW?pMFD]Ub"9c7ZKnZ3\`q31X@3)mB=11?hcRthuN93fm%F*e +ZG2P[TU"0V(is-VoA+VR^Kklier0,?"Up=)(e-`gUI[)FNo8:Hmnrc4pV)2+7NAS! +pfjdZj*4sC)T:=<;oFU7GPKoD&9]2cl:NK/8ndhV^(C]=A;Q13shH +ijk@i?''7hT";GMI3Xi,Vg4gBe#pa'%fCGRf(5+!.oTIa>^m':p8Wcia.35=87Qh) +Pr4@HoYZ?6m<12.%a>J?&Si8TUR38NiX1g!;[r>%Q43i4r4',rr@XGHpZ7F&:!f`h +Bge5eMK@T/8Qt?=eme;G3pHgrXb`ch(T:f4*QA2BU;V$ZK<"#Q$7=eWlt_.O4M'l^ +Yljkn4TC-5I+)_'!oVB0_e/6hH4OH4nap:OPN"2/;qAtUc=(-.K4f +qlSR1i's")_kY[WC,OU0E)8#X(D;2"YBFH-=IfOLc24["._LJ!VqgA)Y%a%ShIbOZ +11mqnKtMdWG6W\MHWDc2)Tq;A/=mD5b^]VrQkfp._BsUW`q'Dlp#!$XPSE[?OIO_V +1tiH!!u*'OO::5-Z#3e,9M,Z=oe,#!SGS2iVQaF'r%!?OuX,/$B[/qKB^X0K7C_I5 +IDlVT@iK'qscV%rr?cZ(?o-5cn>5(F/Hd'W5"Z%QS!IkJe6hs.'14.!;]d7%3$q-W +Yr.,W*TRrBDt)+l]6GFh*:CHrrDCGi(r+Fp7$e[P+:;RkF1.pgrLc!/j)2Qp8N;:J8"EDtQQ2=KX6`$nR?HgoSnVOF4et, +6O.\J3QTtce>_akD&B2.CJU[Bi^OK,FP[@la^/]g3HY.`arNm)pg;Wm9^3FgZ$8,1 +iN.K8&O'ZY-4b[$h"EUKk-c8:rR3k4D$jMDaqg)1',09GHYZSdJj1LKCJ'k#N;WRD +iMLG4tcW*8@=].De#a2OND>YPUVcR&rAB>Jfd.(C\"+XO8KU@0+@j\_-&lTq_raU` +-rlW4ra9h9T*#Ze?+pV[OA*9TF,cgHiihTf8=^A'PZ"Zf5^`?OV)ZG(',+eq28b`#@,mDC +d7E)GJ]=A`08O+mta.k[$+FF<_KDDD(b7PFZD$!d\@FN(bA8Ec#[56J)K*&8&7@sn +HQ>hLZ>k"R5F[be5FUrdc-]>@>MAIa,RX`7TUGJTF\9Ai/@1$F2Qn>=,(g("?)l^D +55tE5V-DfNNGiH^M/`h?bO7p1"*e<4tG&Ze`<2tdoBl!FuOC@\EfcV[=/7._'m(VO +.,P8'dr]r*uBAVHr]]dJF2F4\^PZ&5=T?=Y((Kh,^Y-+0Cac(]UX&7E'As+5N(8!* +,,2VVP!@uj1ucX$DqOF4ZZ_afnher7(Z_ia".TQ4IQ9H&,lnXidV=@b6>UE5]\5O`e;mDWZ+S+]1srXF*UrM8s2`J^"N: +eOHJ6O&B_,/!$k0Rd0cnu]cp?,>=WKd+5hrr@Y&J&+>HIJY#^GPHC?:YH'6+,(fR5 +8FFQV#E![di-rh6+;P(/)LSu/-$YDAI!0\3hO]9O)sPUipO3gi@bp\?8M*MO*<4KU +MJSJe%=LOn:?F\9d@;iYnG-V#TEiRhJtkE-iOFEqZgJX_*GbM."!hLaXs->Zo\Z1' +6pES&Wuc-52?Sd[P/b:MF=3^j38h:MZ/2,!8rDL+7T:1Jc>^[m;##_HYX@][BGH@/ +tUH:1;"rV8MWm#"Hl6!"+eo9q>UHp!!MNc!.k=NC9d`!T_PGDO,-t(K=k"?r?(o64 +?7P^>U8WK6Tr,reD=ZnRPVUO+]N*odu9k3p,E!OJpVsp+7N.2!U;j*p^-!8DZFM*/ +u8/8W;kE'lI3?TEk)dk)N#Ue`s]mL@j!!S&FQ@\%Fg +3<6m4u+W.g-:M7opM"*"Q"5TW?;ep07f]TYFk^trr<2irr@Y$rm(@`RABURp4Lfd4 +XEdF0f@gc/BM_3OKk!+D]oI4o3'_/F45^2chnKdi@""7q`fHrn/dkl;"(RR!49b?(Zh)[3c>VCJK#Rsc=#EmkY?JR2:cRC!:WaEgWF::o +6rFY3i:8,&GV^V+oi^/+ZchETFZ&B!.kp_0%,1]GXLN!Q\-o5gRCo1i[r`X2til0" +BSL"1F9P+"<`9)4Ru)0?[,!N4K7XRcNif`nO*#tpo.c`/JnBmfd#^U9CW*gplaV`K +?.9l(rjtF-,ah?Fto7.<:3RHkTT9Z5DEMJ5N&KS&52t'`Es;X(i-^;Yn1%pqq$\lF +O%Ykl6Ec-N^8enF!*I+-j.$[!.t4NGW*IY*tZ1t+73cRLtMW2Ye#Tt"Nng,6@CYAE +S)3.':LM>P5oTh@J\UIhu:ZZFdm/2"C:7.E62TBT +I5`X87PXckF"H:`RR7QNQ"2eZ90M6J&,._9B"&%]m0GbdoPI>Pc<-f!UsUZcS-:u>a&?=WI_pTAr +rDFrX*A!l>(?G`Mo.77)N!uMe+NMo$TKc5Z.'>&R';li[ZDaRMO"5N!)F:0p$_eo- +h_M?j$2qWnD7mP1T/-/nFOU7m\k])fq:o9o@P"0&cUBZNdmY8!k4HR>'F/qVp%'6)cQGo12r +`8u5oAANn6$s+`c2RbqRm3"6?:5IX2YKCQ7r?G+ZMi>C(OJso-+g +9*QkjL8gJbi`p4?S1#n=oeM,rr@YBBi][&rr@Z$rl1kRWg)7Jd%/_`P'Em! +,r1g5l1M. +d@PFfm,BJ]QXY0Lk-hE0,JH_PE%0cotTFt+0ria.Z[0M4t,+AgeB:D8S_aS/20&"C +m%4bW7g26kE/"Nano=j1:?Xjm*Xk\2D$i@BlA)4\.F(Z1.dGb#8Qacr"(C:mH;Cs5Y1l3*aK:#TUh6@L&ANg>Q]-=6*./:kcS]mf# +h)&[,`2U#KDtq(1roN)8&gfA!VEEGj?LYui;We0N\?U6?OOLQXZ`J(D4n>6WQ*\:8 +<7MJ>l9N0[NRWN3Q>sj%I4feBsOrlnc\O6G,g*@O\msN0Jl +X/*_r"l-F4r3W(r#8M7Bg2amUsT#gAT-P(ZI>$nI>pIum5IF5rafYh^o?N`(`9I=K +rth#pM>Beo::@Oa#!f#cCE;CU@1N1h2Wre!l%_;*o/=3>e@hWe.p9s%4g6$Lb@)+q +ToBH'(-T5:I2rlQGLi^0M(8`<:Mo:.k+ILk;bYR[.6VcLsRZ4Un$Rt'&uT2B[u31J +hh:/n$/uJ%E>;IdtV(G]85s$rr?0/9#C,bjt&hZ`rX:)+'_j5KnFkkfbk-[DPPL'& +"HH$p"fWJ"q +#!R/Wl#/u`I+rj!!K7iXknX+13b;I'M2`5G]dK*I%QSe)l\B*4Td7*J%K&cZ,m6BS +=0HU[/U,YRm1K1'n4bA$[`'7rr@Z@8Z6lop@i]qS?@_Bk"4TV7;!c2m0&B0/*Dug2 +llo`q>3GV`4te<\@c\U8,*F0%es3<%-lPjgL\lOV_9moBOamZ:0;/NPLOU(S6+rnr +"L>Z?P!'?pb/gX9+4$nf3U#P84"06S=eb"ccgVRqUa4/b*#t]DqbC7ERV#i>'X.;7 +H8Sc*gQQKrr<1bn[[5#Dnk"97D$";L2@"<7^[t=)0qqmHBWAt!K_gnIPps>2b +Wn6(aEcJn3q@T@YKbDg,R'\[nCM$S(Q!"i\@A!na(N9[9`65Dm)X,)!-Hp'c8'[e4 +g@6mhu"IrV5CNt!!HO$cQ(>X_CKsN^s>?6YdcoK>lGK9^Zk4sf*gn+N.8X&Io2j-E +MU/F!.ltRDV`A294E-^:[inN.Z7%+/,mf^\BWop +3tit!"kTBn/@A6:hopC\V*K4%:pI`DP_`j"-<6SR_M_7i6g`jIb_[Ng]R-B(-H*a: +m:f*"88Mh$6O4OiJbEgNc8=],oGog12m\E?a5$[T@l$ZZFb$snF`G`HCQ9nIbPVGW +rC2U0S2UkHW;:RLll\<6X!j%M(/nk%OKsB;bT]lWWhD0 +``+S!B$56#PXi"r'QQ;)J.:9YO5bT!;Qm!#R-mk.]BS.U-=YdfBCJ>59iN#f0q`V#J0d*k'K<2M!_k7/VnL#G%.``a!40P<@I4r?VJXH_H)hhtc?kMTrG1rtiQ$R(]Qs$[_g3k[C +q#UD)SLlt*O1[N2iTEU$IK"B-KtB_W6S@+Gg1Zgrr<9;rnWmT[?p:PWp,ruln`l.i +7p-`G/dC7MW44N[4@AuL;U#q07D6l]u>GmJ&)Qn!8rmO')do@%X9ehLW1tV,l[i`p +!#@tDC<)<+RmTV,fl:;d:C9;W=pOq1AoE`k7dsn&t3L=nE@66,`NqPncC:k,MN^`! +S@fj(%Ipo&I-fJ&)!;h! +5TIEgO'e3O)8j9$pHkh(f?;OY!545_LOhmGeD,r%QPeI#O,S"rrBlKHn2)[58V(GS +%W5h>\X3<&YPEu,*:t;#UJsp89'[T/+7\@Wr1Zk^)d4RK$;*L`4rWhQKkX1n?M@XZ +89>5,MT,"!;gfu=7)07_ZsR63c303(8kp:>cD:uN+@Ni&Pnu2B9@u=\Hh`(^o,ErH +KU.cp`KJ@;kWAY(PBJ9#^eA(8#II(=uYnj&TB+a@YSsicjZO=qnTu^7a=7@21/R,! +!Hc<)JhXF">$=m57t'ZP_)FWbK_X*fZL!,*1%Yl#Npm?([o2k^*TEF?+@Y&9dZ(oF +%s2t"/>o-8-F,9!$9\e%6AdD1(H<\o]aVJ/,oSQc\(V6)#+1Kf^aJ>C!F?4!d3saliE\[s?I&[,G"L6Ag7.Bri(u['7]O=8jHV.?pf_]H@F#X,G"c>S4ghA+M[;>)D6r +^'ffQ[dqf,qT&rCekE9-(kTo4 +hHa#a7I./1m6";LVM!ZVb]W5H)&A?BY!W?b$'nk)]:Cs/DEop&%+dp7*9s0>[,p-^ +2.S'+=b$S@U=iZ2'0@iO%GliP">M]#g+kIkHXF!!*&`30HL`o9GR/eI=#S?mJd0=E +dT+)q!\1hUZ)/9iWbC=CGltI%^lPKrN]37i0K'l`WTg`Z0@HZ$%+1\[t3F]?]sFdr +r@QVqqM)Z\!tM>!.q_*/1:qTBj0LE4oYN0m>plr!VESn'SH*'n.10O!8tI2!]4,"% +d;=80?:Q5F;I+_MV1nM6q"^DLQ$1>Z&#i7lsEB`+n:os&R4sHnRfLRAb^1TVGA-`` +7"(MZ\^RRN$96_h"qHS\,H`"i)eDmi2ZeWZ<)i^9$;/4hrrDg@n,*cuiVroF5PlYh4tZR&-\uY_h+pn];]L#JOhi#Y1hm::=Fm"9,hM>M` +D?:VIN^\l)KimSrrCM+Y6b..Jm56Lq]"b*fu5Eek,!Ji,\8OCVdm6sJY"io%hemBL +jD5i=a4h257^?F]ftH78kF.7i7uK4H!,->\8fhPnd^l`!!J8MAqBs^"`07XpP5oa8 +Z2=hG[7_5hTBf(0jH=b>!apYQ?(X@,ZjC*:]u<'4Cc'PXTTE*rrDOgag^'D0h1\c>15q!8t(Gn4a9TfcTo^e\3J[4n3`QFfV,CKNYC!3>LR/1M)FbM:;B>Kh%I +O4NeX5b_F+8BUB[;N'dG,N4TSie?[BBX:^aim)!/%#lWL!4aF5cF^'Q,J +@]9;HUM`KO"OB2!)5D'\*KR2#Dr4j:\_Eu.q-bhAJ:^5pt4d!XUQk'@_j'Kh_FIXX87$um38n=AX!aF#[HPM!;HKnfIQ[G^AW]^H +*6q+7I\lrihY.AcC6Dnh+`Nq0McSfnOeHHSY\rK2ZQn!`p`[PpV\`=ZfBA^] +Q*J@m58"Y!THa(HJZriH"BXAop*Ui'eHF/qAnd2pR$W[t!=r?aLpAe*s!iI +u(6li2#3H\a_>-/hH"1.7T4BVC_0Jng<$,/;sfQLLZBKp4)iiGO:gq*.HDo%Ib\$l +k/1knE5^G25a`'ofOT[Nc&sfD^)4EBG0^#UMucKblDL._*Ln-WJell\R8I@2*6:7K]spN_bbrq\U[8F/P*o(bGoBDec +8*?a@pNUD$-AiEI`jeP!6+$oc(rMtj68VmOGRmqEAYh)9m:V%o$p1>I +k;e)-tQWlPFmaa@#jAW6A]1_dDd&iJ1)Ah8tXrOH[F4FnH+6b14Le1O^a-NK#.O13 +"3F.I;6E@!!.X1+,*.]B +TRq7q`lT:fGGb-k,`7Q?*3HF.^=j[lOeUnT[Q,Z +LB)`Md+bCZ]Z`=Nh"\A,`r?tmi_cT0I2VImp?9J_E[Nj&,7O8_iOR?n%uhpJ$imPB +/bb@%h;aql]Fcs/$#P5p+uh?(GS6E^H_7C+QRUgGP])P('jgFQf66kGbuR9RXUNrl +WU0HH"Uc/h>tBI1sZO0M^(PT*S;7f%pgXcfBg!H!!`o0,Q3As!,!l>^gFH*g>2L]X +R<50cJ$Rk;UK>lEp.&F-XpMBJfEk:J9!1**s7#/_&i`+<0dumNtgi^HQ'+hYuufaF +2l3FNXn#B1&E[)!5WHY5M7^`9^pYbHi`_`iAZTU>GpHGT08^Lm,Zll=eTBV55Wbg"B +OT*b&WMs-@CUX1#H`Kl;1ENQ"THgJ!V!M;_2GAKgC5%`e>JUW>2RP8^f])-Gq*jjY +uj%Wr#L@Xl1LT37CMtcj.-(B.g`E)WM^[Pg^tL20V$5,#WTt]4i"7gIr8Wg.!P#XF +o,Km&(^a\A3FR#[a6jl&]K+W_.09I8#P-EmeKkVA.%:"kA[*`%S&D@?aZNHYP9EQm +E;M6lB\%S1iJ/\L66T*l;q.jL9C1Eok6g<`i/A0E#aW(!hM3#ae00E[GUa=BAqtN] +3/i3'%ChGOe*GPCA+T(4CC-1^@?=o!5VgG?eEujpVjlca!=jVNMq4rZ +G4r^]A)GUNJlKEgRFu^1V,GbPYcT"HU]q&H4RsO(B4BN.'L>blJ;rUpks+sM)>!;T +%h7c4r_#Af68#=jfJI^Ah.u"RjJnO$G>9tOn0[m&I\<\rL#l]O,JK#mlta>Bm$6:= +)_F>G?:!Gf,CVrNAO'r7iSgDU))2F+Cq`^!eu4hmf*9=_10:A4op,cTC>K\M;A@0, +Hc9WrrC#ArM@38XB/^J]@5f2dV.]b%5NHjJCl.?)!'<-VP1"9L6$\*o/(2nT5R%]b +C4]>f.ZnLhu"4Up+QB;5@r`0?Of;3h[c.Q;6+C([lQ1#D#?3bei,PIj`>1"Puis21 +s)#5N:,H4+QN"L`;V$de208NCp##,lS:p:mHPNAo)aG'T8e=./q+3'k1`GVFZ;6=o +j&M2f-E&e@i!sX +fSe`^tA](6fini+#'iri"s=k+5s-b6h&RCHJR\edkji5=m?gOR"+UEu6 +U2YVm7Zs?:\[fa:&*V`hA3[q].H6Gb".\sUNErqpG-1,O@Pd].-oV.X7%/*2/D4?K +t\4LftS/;4stT-9(;emf7Mt5.;u?l1!'@pDj@b_HfLRmfC;)Dk^7a;STe>j[7%U#VJ9EKK +$MO%`#"Ni6$"g=9'm6Jh^MrlN(7+b5Td%Vn5;;u9Mk0%>A_cNr4f^,kbZG<0?.*L5H^g8#a1&P8FXj[H/`;Uj&C]t3G6:jE9eU0=Q=8pS'%e*9l((]?kX/YF9B +$X6"-dsq7R%tA4@e5rWKT*GD,]\Bf&5ECUZp,NpAIW,9@p]rJ"hNU$L&6"Dh6c.Qp +*:0g^+K;]]LhT;=oSJR^O-^ac%Yo.]8q_2+Ro@!eR4F:"G1EBo,0h"gC?J1[h*d$c +SP\!?)@I8n87"::hfn\nAW>t.@AOH;?nrW1.]=R1!NGQcPC4q57_nKSPrM:@'U\n_ +`[C1)PXF"gugZc?6u?D)#]]l-^?eUUAPUfb(:L"][gR9HSut=oVruN5A!^Ecc1GAH +pMX%4s!(^2Id0H\u\br?"7K+QP*dOJ08rbq%s#;XntK`)Yh/=>=Q("KtIRJJA)or5>-6!eZ9h3k +>o3X:e]m"9GkU*;'HI(Ilud%1',Pt!5UQW)X0FMrOI@nh#(ZKm2cecHqe1PCY)Y:5 +Pe!I];&%iqSMu5<3Nodo+oY>rEOSg`S+s^,F2[H?eF8O*3YiIE0A865fuXq4'1A]V +u.25>lB\i:Y@BinJ.!ui]iCrBDrQkUR+#9b`^!t0grX^U.#/$H4nQ3O)f_G49PT`" +AeugIN.!H>PN,%nO,$QHStHe-]2j$Bl55/BdDFQWh>YeGT:=G5-h]18>:C4j@J6`9 +1^?^$Q=P98aV(m?P7k)(;*eGG_a)l(G.a,HYmmV5!$6c!]Sp*5j/8DO7"`$L?NSl^ +='dSPJ7V"Y7'X,9H(si.8s(RW@ZmQ]QT,K+5O5j2ThfnTLg/F?QO^5$3(!;>K>SDQc>UD:C^]\C24II-VfOsV=,"m- +d11n86c3]ida6rmNJ$'^G6Qg8L1OnT//IC#IZ*ao,Ybulj)i`(jH_n#mmPX3kS>KT +A")SKc4bZ)7-Pq?3t#JTR,jo0oYi<9eM8-?_`N;)7GL1^Rq!6rX[5eqS9 +j6E(b:dgtXBA'5E,jHZ3$7o@>!Zl_J(dZ:$_+bXM"@$YIrr[bN,dH,p5iRBSkaM7^ +E3CTi/.I6!VQHmN;Am]/$B6V;ttq%?#r?`rrB=lDQIaC2rrC$]bD/mddk7"[[iBq,TFa-'8(N2SAH)X^LbD5cn4V]9a +bi+&58A9P./a3magoa'"1Xr=Ed4e&d,i0*6NBCm%RB8cP2Gb9:Onnk_QS]d$SME$r +)N:MO6o&(itsm8p\OUp3I@q-m\hc/pJl$^R$^a"1m+m"s^W6j!P@%L625j8KAb!+GhqA ++:PZYf-Me@Mj<*0On=Jq^_H'a5l-*@JH%4X>W!1[SXlsUgp$:MYV2a$aK4T`h+1I9 +@A1mY3R,4D[,L%:2NHE[k2ja8&6n#2D@,4-_>9?BEN2P$;GiT5INM(jC3uOJG`$r@ +:R(g:YVR*fCNkKnH\t92C6-$_B8^j[JSSU!!OLLnP@"dg69E/Hqs4/WT/`Fn5"jF9 +%m6&X_/0sfiWraI)%Nk??,V@_1.5.*sPhWYLoKQ5P<-3!^@C<(#R"?DE6fho51VYG +iUJ$*.Bn4Q"Tm--'*a`3CAI^JI[HuH/buOSofbJO,+\6H""+i[r]%R>KU3'l?[o%9 +>u]>4RZnYma&Ha5OaD#^[HTKS94Y9pG`!hV/k.Zn5rJ@b,^XV^ +9i[JI?G9B/)d[bpc];#g2WV0ao;?]^5)^gJn"Miibs?si1`FGFV!-b7aq92nIdjcP +=s=WjRTE1'l-bAKU,El![S6b8'Ctr1]dKCrr<3HdjM^+!.ouTc$'.$D\fj`Oj4nb+ +T=XXkU=Z]^,k[>pa>rUrn62:#K(J@^C>R1qfhrte,CHDpiB:cV'UbBU=Sjr(3Wa(k +?tg@^_e%me/tsAr"iGp?gS/6%^2[Wc5OHVr""4U8SrD*h$W)ZC#pRUOL/g10,nfN: +$[AH/b4E&dgq>4/cHL[Q:m1:/sgZ^Nl0"Sq;;/WnTSuRrkl<0g9n<]19ecjMW2Bc:\ZEQpVW&GHB^@\#'lleik5r@bnV@#Vrr?-irrDs3NoT,#M>e<,p +:@DhiV(Z#_G`qkX'*GB9??'+o>7)d_&Uf,m;,UO0-6["jSo5T/hQ;8^YkW22S!6K+ +FA:f&L'XAi8nt(CZeJ$J&0@_+,RB[B"t\(a'C=U0)&Sp*!7+0$$Y-cMpqg\aVEn:! +:Wr`par\'^M=A5,@:.hi'.ToZbh$LMIV_6g:"=%5#&OcT.s1_Z]IW]@!\Cm",-[SK +;8&E+SZ]K/*^sSBrR(Ema7odM0uM2Un1^P9(.`Y*i]`cr4ggbRP3@XGrpYThY`?'4 +Nua8gA_0B>Yk,C(W1aW]AVT5q/,URqcnU^GUh/4cWm]fZ]a*GZ*m3>)og!D3XeQ4j +[nVVrrE$)9cs9J) +rpq&i;%b:rX=]Ue)']`;/*(Uo!e+e='u$eZeqW7c@2Yh\Nf$)+o1gf3>1=8&orGlGe<8 ++"Y"+aEnC:Q+?nG+LMBg.`n9ApDcLNPi5OO%l\4#REPl25->U-DQ\m[(i +rT"*'ql)-N0')T]TrY8YF+!mI[kj$a3tdF2Y.Xc)(atCid`E`/rT`0WX98(RN?1Yo +,T%XE*VhN%;]85^M/fs4@%qi?NAWact?dmaoo3XK7'ILcV2[i$V,Co,KgFh5C9M>d +&l.tX*C/*pCaK=XuA$:q:;qP'Y/mfn>`f'OE*R^>g^2C,3i?Y>SuKFi\7;@A5F*_j +3iHB=2/1ipbe)`p;d,nIN*Z5*uhU>!PEcQgZqaOWT!V">9bWc0eA$0@(_Gemf">Y_ +4D%n:BbP1[AWBsW605L3d[g#IgeXGZ6M%36hE@Y(cD.orr<0OROc/J,PUQqer1S%K +5D3]UB!Xl^uSSFh[lkaVss]d?@d&XrANdjH?0aW!<-<>AInPogZ.2YET&p*8)R*/^ +tNFSK04=g^8lN/2=&9)Y,c7TQU%@]bg&N-^upbkrL#n.dN=Ht4RmWOXZpMA([/UC1 +sIO[l+2V:MldG:M`<8D=Uhqj`$2` +ubDM:f-3O'=\<7V1Te\Vm#&d77e.!fXlBM+bTS +T!W7)"Lc4,C/+e"9K$^;A!Qc9QVOmo<4.'i0Jg%O2J^Jrr?`ufV9kH^,pKurY)QB5 +@"1E!VkdRIXf2:hCnhaIq9]BHs"sK0R-C3p3m=s`Uq`$G^er'k3W8%Z".X:r/K:u/Yi(d+j4fIjZ2Xg^^Cun4!.m7ZIk$E=]JCan#'#u9C\_PJN +;]XN+H0RJJ+<+q*%:;8R*l2'q2dP%aNaeXkCLl_d68B%pli`Z-\o&e12;.7V/)p\K +gOI]i`1t"N;Um]MU]5!-0Clj%SY_^"-19ff(_n8^:X83HoBHUPtTqXZU]RG]0o:Tl +lu\`K"paU#jok:Emjk?\C5BbD)G!SEMqX=B/2/$%h,WTrrD*#CZ#-li0rQm8fo#P! +5VDo/U3>nT*@0f$>?a?G=YG/npPH$0hE@K1b+i!QYCh.S3[MCD9ai6.hH*,cKOYl"SDpgQ&q@a!J$`k.ItXC8nOJ7TT7ZT:C[:]/Z +1QF%JRCLcI@ri;r&r1&pcj)"G>t1&'f\i(I-mnrrCcR/[N@p!!f"pS<"6%K?nAe< +]7Yk!!^#`ef$`+2>!IDpiS9B-BY'Y`"o4T-m\;!JBe@FfC]!?N.=aOM=B8)f!OId$e3_%UqAe%]Bc]q0!7^+.X"S_UPi:u)XUJh-1GK74>-H=rZHp +h0R55N,2H1\!R#;R0Gt;/Zs>MAVe-8.o2'CBR)J4jJ8#M(I(]b[/)rkh,?3q-!"&,])f6FN3dBcJ%#:6I[%he.W^l#5+ +*5CBfM<7Xg&D$Un@Ou,6fhSN!HN&TS.k<8.h?hc:oM +ed+tPMrCR4.,KZ/s>&#Cr%*Aj!U*3k,`;O)kB%P"Q7NDU7UXBOe:3T5<6L:K +6)I9?23o`:GEg,0hA,0:?a,gO)T04I!2^@=I8b0,JAP.J&/l5gh<=f([W$tSRPW_k +=LsANl;e5qB[BPjb*?[lE[;[4X^F(hu4*$doQ5$f>7%?h*nfk.k?5#*Wi"iEUSECY +J+kjbm[9eL.#F!i$,p#rrD!;I6uf2p3?@H*SW?lHsS-nH7cblrr>CMlX-;'!XI)R$ +2M&6_isE`[@di1#2RK-=:`G0o3^i?iSg[/fAa`R'd?EbYfQ5k3To9WoY&F!9cF\p^ +uOaLq7$&4MF@!d`W#qG9!#5UB@qj20:=-k5+GFSK`;%g;1(P#O4p'me,`$`INA3@' +h:EHUD&4R=7_"4l%=4F:B^[tbEO+F.;ScGrr@YZc$-)`EO)'(q;b>,4sOJX9ulV9F +mLeVIlH6pMLWJZVrt89U$r!]9U:2a]Qjmg`_RYcOo1Y[*--T4YK9m( +bDNS6M)WM*BRcZZoh;7*ut2TD$u%b%edn5c+G0AXD\gKEp6_i18Q'"EVE7X4m2qhK +F-`7.cEo'D:/rumd'22p`X2:(]5&dp3]h_5piY5.fag%FIDnf]Rfl[aI`d\D[G;@I +G;\=No.InI0\(_.ETG:?.`7le'&A2sFCnr(R0?^OO)<*Y,OH`8@o0Ofu_DR +Vo+bb<$*ME,d2:f.]WMGoVgk]<4Xtk.a?t"5p45L>EiVd(CfC"br;=TXV7^h#p9RJ +\^!4%$$#8lX'ep!5X&)Q+sg,Su.TeG^^lO_eB&MYF&(7%!Ac$8Np-d;4M]g_LOcK! +;<8?^Yp43S%Nrl1l_\W("jZXOQ=>F_YK"5jpj_8g588?T6^$V'^9P!7m]>Kqr3BnpF +Z^0,qAB]1,8eN^cP?$Vr"c>)QWO1eJl-oEK%4.spbqfRUBN7O'7`l^8c\cfrr@Z>I +a.isrL1A8q^5Zs*:U.er$_Osn6OhrXLuP%7f@6lruTrhqPI3''ZP1Ee;raB?Daj1qQ +P'1NCE6%D9/rJQ7SlLON^I\,!OnTY^6/EqJqj:*_/MRarn@+=$*LpJn:OR4H_gE]` +0U8^Q@N:;rO8`Fl%l%)H[&DnTofpiMg9iC@]Y'qY$N*s4Al+Y"V-tc'A-H&0h&!4k +a*tSWibBt3+,4#i7I6Yr#@T`EmMc*!!rQ*8S)aS%%\R9[A%"n"DXldABTOaR71]bB +ERK0JR@nET*0O`n5Z?dVJ&1-[$bNc9\.q,Ji,UrU3"-iAV=T,3$TQ+7htV;r"VHq$ ++p'urrBl02=3'957sOIItTpsV*lnGepKF8l&EudC8$MhgrD:aa"UE0UC\:L+<3*@o +SsTr2']UQb*7s.q9TB9j7.o?A)WKg58/9fFOr)hdIj!@&A"^+[D#0VSX)1*&dOaR6 +4uam!&k;p&RVMen4q=)+HOBXg;EV?n0c-rN.*,PU3o,`0,.N:[c5V(nFSMN_-,*Mp +FZfL +k8#>+JafZ2>cS:!!^CA!5V%p&))C()pnQshb%auFmrm6LJ54c>Bs'uoVH_ccL#.NZ +'g8=Nf(cM-4Vhh!;tpVXhWfL:CV[TpbUpe(AKuNO2O]#[(D:]2L;QX`O'SgZRYtTV +VUW9n"%D`$(S#HOBDF)qe,gLp]^-liI6Ca_S'1IL;3MKG@]0-a04.Kkuo&P&K$rYg +a&5pdQ[&]6s+:.jogYC!(gPe?QY)mDo1u<,k&^S9>oH6<_2oqg=s_d.](>7-,aL9^ +fV32(*d]_kHV5PPP2E_Y<61h@X*OCBB!Hj@T`C=5_HpW-)HJg/@I?-&Y><@ +'%ch1.Mg^blJY><43Y8J.L5'KKlii_gd2$#rGE`p_1J.&j/&1T*X]NBsndqi +;;%t=qm^6D)BY6DKIomuX&fqeCMi^fb)$gC_b0JEM]^L9AL&V%VIrr?qB#2f,XdX4%cK +7&)fqE1*B--h^JbC="_4RXm*l10ZF3*&MdA7Q0cl(RuhG]7"UCG1H,,Y)irL.J(X$ +=3SfFnpXLO)nYkJ&/eKP3Cfm:.m/omno]R\C5guX!G#Eo"WLiAHN%-!Pfbn^>p5bT +:Pu^3WSs8p=9,MFFOYmUZ$C`$XOBKGa^3NnMM.pGk",YG^FJ<-!3]u'r +rDGaTQlpR.J1pnr(u&rZS.m-qq)e%p5o@\f]'O_rX/8TqlO)Chm;17%^U/P_-uA+K +u0jE>En"a'lnU2P]AfdgKRn-E3B,DrZfA^GqPBZ@-II*TBL_9pm?V7Jn4Aa_MkJTF +RRu8^)ZVamHO-Op3ue)nN-F'8&:6"Hq_o[Y8)J[qSPi*47+4`#1_L5jnkta,@H.qF +2@f?6u7pPZ'!i\p%8t?%X_nR$QBskM%DD`@`c\N$OAs0=KYsT('MfARRHFrT-r +:oV,"31?@i2Q_V[g&Vt\,H[cn_1n!?O6WUpbGM"$/`bOCDP=!&FCPi(02dlNWB.Oc +7A_I[%4@q\o!p\9$O`'IdoD$l@8d@X_QI;n417I(c8D-Hl[>-F.JsDF6e__=DY;p\ +&t,^9eHsR=O7:6co(9JYMapbFchM&hE+^+CSge6Be?OU;=MAs#pceNb_/( +bcKP335?K$%,CINQ!9"kR@.=ng8UaO7n%An*O-E^ts[i%iOS%bW]86WE\d.*n11^( +84qCNIEj?J;&AJXIrGF)NDP9^H^Z0?]R77^ouI-icX.64rsFUiX*Bh[?\D/G]"oTQ +Ga0sL(_WUOtj5"'cTdcWQ511pc%5j#_-MpLs$WP^[RQTK-$E0mVDJ'rXAf,>lN8\) +%Z[;g3CnkFUPb\rB^5Sqciib`ZMOrM$BI=?fG[X\qQ@&c#p,(CCRtA9QnT1JiT7u< +QP4iaiA:7$ha]$W:.)8fs=,p07Fdd8^nm1)I#1D9fl+1k0$U'c4'uKtN_(9sHT+6?Y08;LsV:M>a( +6taS]$G^Mqh_R&I.B4>4\>'9!-T0n[.UPOO6sXgpl^A-meD6&2Z7f_?Q]9u+10NY0 +DlNfi4k.$HmRe:mBfSGk1OV/8X9b][%*775m\Ll#Qg.hah\'+^I%8,pBg*EU_rH_]IALFQ5#IY$Ttoi+D-ZhiT(Vg`7W:-9*I6f^!Vq*B!!RrP[ +QG#Ymh:'^rr@aIWQ4"Fmfn$V4a-Z`=<.,E*nBhe^g^ot!CnBL>5Ij+J+-CoJc7h%] +oKCGCCYqR7enUClO07=q?JDH>dj7DY)St@5fi,bI.@4:"k)mNp-'VtSTV+>IqREj> +'rZG2^?C?fdkqN&% +j-%6mVR9ILS!QU+]V(TA'fOc?^78Q7h=@p6MLf_:\`Yij)+27NM:ej]91P(i +n9nT3p4h\G`CLY#_>+8-saPo1CK:HBD/0L4(\,nRB3IuXNDF[baXF7XtsFV^[O_]L +%+A7*'\1t6c\m=?.)+1-;-KU(fc1BWg<^ne6a.X]a>fh$^Y+tQ%U8n#r*Q+2H9]G%HW'WYieoBL!I#"D!5Wi.B$UK\_fM]"b4ESlbmM[m5gq'o74`D_aQg9pZd^lkp +VhL9Mu+TD.C@R^2Lc/u$A"Hg,\W8@=J4jForD9Bm!J6*`>6>F`'4_o'6O\: +B^5cZ!X_,naF7d)_gfsKF4H?@Ps'bJ-RIjfB7Y]&i]Mbn3gXQ:nQJu9CXB]HoZJLp +j9uGQJ)HXFpU8lFf^nocQ0=4#`T4UOLu2:V1+6TI`t"d/[n/cnGC$6UL91p*O#bh3 +[FcT7t8@rT=*!ur\$Xt%h!b)nO9%W5DJO-[Adff=7*4Z4L(-$X!,E_2R+uhOd5q8Y +LGeip\(6K:W+o:?52o3pt*?Cp<=B(H\ilsD5N.3b#$tTC`[BPhtT2"i2;?X-i3u/R +]"('!"8"ZG,&d3N>Bn_b/QC[b.5#dGncJCOA]$L3'F>/T+?CiIg:J-X:3tn^)`OUY +dWTNqeVJ$:33bEBEP9Jpkdt?+u@:!4TXN"'"a,q&u`S%-[,X+A,V5_2#(tGiX`s\m +4DY]br*aPCR[nnSPgtl8Rs1Y:,B8/3TP:cTJ&N>DhQ`BnQ^Gtn085nX?Wt`)mXmX^ +-)DqJoq";IM_dOn`0=onG`L.*sZJJ[YNg['#t37CNtHj(niN)i\SUX!&?VC!9a@^i +2[q41`m9VHF(!*28(X9Dtm>(-b\8MEg%Rr>Gr#%PJPQ[r9!o;amc]ii*b6Uao;@D9 +4,EJn5jP]Do+^?^+\8JIa=SrnE5LLr$_)Hp%iC2Me\U@3+0VF\(Qg+ebC6UmQ^KB1 +t'-`ft[j'0s(FMi^nNYNP7)UfMa%%a#X/"7uFk'&migqRZ(_1%e@IK2l#NK09p_7HpiB%/$q/,[/hCg?'+ +W3?UPCP]:YNMPW$s/Jp_rApiC`NKAs>9nM2a5pkh.=0(/H0l=Z.@cP0d.P +;u0lR1tc#\EYXd_9`$!.n-s3P5&tq`&Bf-]"O"DsY,XYK&])LIOLFE +X3np;Q&8,o[Ta];9op@+(LG\0[1/OcT/cMr\a;6r"CU?JG?J?lf&DTBk%pLAitF@X +AF29j]D&7QsUg5>T=6`lTcO_9,>ka&&'RS+-(YQjPTH2rr@U+fpU%jnGC-FBES!54 +R7BYn"$'JRep'Gl>bS1[J)I@'hQkgKBT'Q@fg#@1;Z/id^CpGJ*g +rcFS^!-P+."B(PRX!;1*e))t7)>7<1V&L"Ep8A.:0@5nn3-*sIi.$nn/g$:'k8[[C&8`Y.%>Z/7G7r +Su.*rrD!ZdkoWR_AM+)N8pX./\E1"CQ^P8)tNlu#('1ldYE!%mK$1(E+4n2m?hsbG.hqA#0i21[5f +6s\P5-`;Wl!L`Gn6Wdt4($Y'_iWY=B)>YioK]r/-nB6VB>E+T,gf6S_ab1.44X'OKJ3RnL;S>RXh-1u.n:!+^JWguSCrS6ul]J ++t6!r",@V-gY>Wi=Co/G+l,7X2k(oLLI]II7!^)@AJ=ShBne0iX.@G?6-?0&i5OJ# +Jn1E;\.WLp=]DO5N%e#RV0\0fj`:FIK7%0D)K'>Gj*qX)(f9-R4dIIS]%rf=]*7Y1 +!rQ3F'jk\[#HWNHA$H^cD.$XjFQ^CIr(acDo6q:Ma+L[`'4\*@8uM1^PD7Em1c&m0 +9f'>9PspHN^)56p,i&;q]+o?/sH0(GW/jrNS^^E*QO[WpBT=8Q$`4j2"QT4`3J0 +/i^.R+-qV+CTos]YM$=Z91J51t1j9QP2e+HF&;Fqjo>3n,4D"8Kiuj@q"8!Q++7X"2lf(-iYV5CuiD5$9# +K(tR28kpeji#lEnJ[4L4L%cmJ_740Jd)E7I`2#HC=bZ\2<9RYGr490"\o=M-6*rY9 +IfSQiDdH]VQR\mg\W=0$$gA5"JN^Bb#IQ^[OTT%S>Vtf]#-&o")!7W46;&o +,/eUG9484b.c>,n`sOBrrBkc]HZ2GrK23f9=T.jHH)i0N.'&*UelR2gA4+t=As=c' +h;.s"TWq(rr<45d\kSG4q=ub,!dT:D!^!(5$+1<#9LE^]rUId5Vb:$9mBb,q;@)l= +5aYkh:]mtS"f]09"S1ljA&Cb1&,Y4iEkr1LF8[k[/__to)_'DI$+F/"OaN?_oLXTO +mUpThDL`kQ(+<%NO\tqouT5YlX)RL_;5pAn=RW0NR2p12hpI5(d.*t+]9M/pS/7!@ ++ba70Rr#j;0ef'80?P:G9.%)5E]WnPl#fi%'q\2L:M`QgMsF^j8KF#&"ES!"H!)Oe +:[c$+s*W;4X">6/,;ZDAJW_!g8pVEnk%#`oE'Ok73n"?cWcb![hgLeQkeUlo:oks% +_I8HUn_@Q6,CCqi_P>uMs5^3r"!a!R`!>=-C$!Ob;8s*DkT@-Q8rfqE'!"om!lJD! +0<=BphTaZko*j%a)e#'&[F2tKaMD_[0b!:;m\gFrrBH5F7>Aee*h`Gr&^'gGj]"4` +L8KK,]C>N:q$T6oH9dOJ.pp:a4pSHGhuRUg@aRt0!OJ]JABXs:q/U2D9G+M0Cg +:pSd!8u2uZhmVtBHt!=o[3[=Pmbb4_ta63DZh/hn&5R>\C)m0]P4RN.YfLP2+s%(* +oe(ZKmo_F_4OU)!6.l`CX6o(l(@fk]q>nj5MBiuPP0@n^cBa7*RQ*Li^];Za*Q^Zi +1"=m^LH('[\0I-&R5ACk5okJdu7@Rqb=iY5A-T/97*oUF:5t1],:M"`HgqM''ii5n +Sdb_i4VKW;R>/\b;_]#BQ;)%'ER[W5$IaQp;OCQ^Yl"3Z[AT@<#<41)c"u^poh!G" +WZ.8rrBJ"Q-]@jA+k,4d1jG=oj!WNMqdir&$&P.39f:fW58K6')RWJYrMa`a%]]-p +;jA_E[\SUbdE^^mRA:lTFZoEi.b8?C&9LWZ8f^\^,XXI3jAF8hEC2l;%Lf49mgs\! +!Gs"3cFBZj(NTIdJ2m'#=$3!%VtG=!;K4d$WWt>Wa1)"WIDBRpfNi?ai4W87m4TRI +DX?YmRBc!EF0F[!5WIk_km8=P654fNF3W[cJ5!TSLa*IQ[ +^>Lm:U`Nm-UIaQWsWg`l9Ch*U[b-`mlZCNao;?kMQnY$"!/DD`c7S`"6#mI-Jb285 +O_N'g>+@BEq8P)1:f7>dT-+hIKb/I)9+JUFNo2!rr<2pZWfZh.0o6OR01+5Xb5PT) +]X?"qf*@/[m)3!*V2NL'_YWeV+ALutJ*u\gkYcfIkmc.Ah]Oc[1g4+l<` +^P=)iRQcCbrNIS@3>:SCZ@YS3(,fW_*"9jdrS%=r=-$%*.>'oA+1hZcg]$]_YS0.! +&h'Br._QZAPdZN=D-auf=cQ=rr<2@eE":@XZpPnHaX,_D\baB5i<`SJ3OHDgB&M'! +^F=^SU<6n!!KQo-gc]d<$[V2BCSr!E^o3eV&;L`57.D^chK,?MeB,9K'qU98Z)D&n +TkSA0OAf0Y#G&f&cSWMrPn:isa(A]Ak?RPN*p*h7()ofr?d_G;EU$H)?Vf^\tA$&,kfOl$NACZ,3mcf&/:P +E3(;f_,&Ul$rIRA`)2@!8sl,!aad`LKb@BE%l.t#PF`#fPgOEq7j;N,(@c0pPUP+` +T#iskbBG]?<_FEN/tun +59E:_a6TGUqa5%mL^./X%0[Ji_0U=c_9ZcC&n+EdMGJ^EVWi,f8+pm:h&8\!!Mog_ +?,"9ph&]Lh$5=[=1Q:&j9Igu`4t6+nI+g<)rgh.PDbJ0?:W%B;\T9-C&\/AL.4JXEi$QMg^!SWLV>'"ZX;sPGZ;2\/JK<9" +XbJbf-e7[Z2Xe-nG$juphs=6PPduSD:Q9jn>7-V6*^!GZ'O.c@J'G%hjo-@iX*Q7c +)gX11pKI]KK)+USt1XO!9SAP=&?Es'*J]6.c;`U%c\dkU#@\Ti>]**P-H\2XXtV`H +OJ)+aK2ZFL?/FDKuq.8W-C`ipjUnBIdM4P:^:@s>N=VZ+5(;dAZ(Id!+t+&C*"#X$ +p0_mXQ,#q$<3uDArY/r1$Z1#pr!(<(]-Mr=jsQ*rZV2Ln]nm^I<_+IDlI&1FVIm`_ +FY*%J;&bCDun9Yf+[Y_i@g#$48@sm_`nKMbM?Z4]#rc0g-b%!FSe^INH-f,7QA_/m +(P'#_4MH,B-2kfj"HA*'(JJIHoQ$(Z]b%Z8!4-fRRYqc`3I-1a%U-Z!.'rdrr<5Ir +Nj-oVBpt\naj\^>/L^$_C*b']87_kG4_'%[ks^+*b0*Gnd4HUO8]NPqdj*?1nV,[N +(X];SXP"jJ.B*)ZGuA,]%52\^Yjd[[(hMM/["I<$a*]"G7r(<9juRS!Y.?Li/6eOJ +)K.tg2cR>%h.5@W,#e=rr>O[5T^I7dQ\HDIJZig@b3_Vqa5U`;=R_F2:RLFh+Hr\U +3i,b0.@]R#a)%k9)P\MT<3MfnDjKU3W!333V[rppi!u5*;iC*k^"P><]2joUM3KSn +4%sA5;DOu&-@@gkL?*WrWrN(Jjt-(n_`^]ia7?;*RX:%nY_$k+Or>B7ta@?[dt+UO +L%.!)f&pdSh&D8rr@[drr?bkXI1pBBMO&5.F7*-!-pEYn6B>T7df,;H]O6e->;KX3 +RK[3h$>5Lc_8ZYFrZD2muP5mn&M'SX5S^Q#lUS`,jN)rn>3hB2c4;@[MB.7d_p?!n +OX2$9,U[\4p^+u(%0rsrrCu-hm=3drU@R+nX#:nq#:?!4?[C)rUc0TGD6sdF2$$S8 +,s+,q[8,Yi:W`RMgc]2= +8r7-r)X*5iJYm#X(cVm[+h"NA%3C'rZMKND:*2!c9\%o0O$4Vl +hjnhU+O.WG,H=+mcLF:rXEuJU0t?k<4VMAR$""T]\6Ql3;^af0 +gHg9G^o]"X8>h?51WiK2=]/9[Lgb@gq6\4@Zbr!2WnJ?AKa])=Fl$dE!K"Ypl5:R" +YBVL^+\^e)J@!,NdhYK/N31N9.Pp!Ig*K/&A&-gIj,0NjGk&gbPWp!lX-nmr"1aEh +sKn%pK6rE_7=Zg9d`B\DKHcR;4PNKh-2eZ6EbN4"E)O_?H$j_!!]fZd#oE4C-EtU!\AJY(*jNJU)6VVZ+pa'"mIJYNK1oba`u1Q@oda$LL&qYP9><`?4k&YD0\5W +dJQqd3j^i*[%r]cBu1kc^Q.c:S!'$ObM`TL2i+F)h7W?HhN2ceJ`j;_?eEh'jX(d1 +,IMr!V7SOn6#\jg?Nqc\&\LKLq-HHM)=]^5(_)qLLpJc"\elmmIdS^qCdm/!"u'@e +USHVV:D[kV+OQH2dm&e\BfF4F)76Kd?=Ug0"?>i4fI\i/HP.5A2.mp +YrBm:BAS78)iEYM.kW8ZX+j0T=MPmC5nRT0NN.jT,d:e'H%hfoNmSCLj8d5?jH8_Cm]Y=`r +XJf#rVIH0(\*R)mhq/ohIo(ad'!$^;f8AA7=c],#ga*QL_p,R4_s".(@UWfJ5gIfl +g6qUrr<[\0&6Anj(t6T;Dei0*"Hou4EEb(F +_'(jr*->EnOY$&'X"E_278]H?#T_m0A;$],s`4?\I(slmZK;3W#tMCWd(qdC!V`tn +@\O83:NN-kD`JR,AFp*gUGosa82M8;@o&]]p>/emq +a1[_Pq[qdi\jiW8I^&P&)5._Lt;4E&&WRrbV[BmmUUZ[K#B5>Cs7F'.Rqg*L&'4!QYKT_'bZOqI30+FOkUK +1VA#A$hE\+_F_G#`Y?'+EB8iEEWY[FXo-S%RC>nW2-a8d4g)/dLI'U0*ej2*aa5))Yh[9Dll&ub0#'.C7N6=,N#MT)b?EFK@a3bf/ +\?LTY+fG&)Ym&*#hu31l!F^F+NLBF?jF^'m-D$0^Pn;HEF3,XAT?(Lepopdrr@Z5l +eWkpnB\sboUQaXoXfSh>@&0gFl.7c)$UIsK92%5QHu5-X5iusj(Ku\Ro];_iWcrV& +D&@FW`[E+8_mC8<@gAg_"q-X?gq9h+au"Oa70O4@=g`aT"+&KX6l`Y:1saTo +RD_OZWtc9Hoge?UUhelWZrkd@'C'o:^ +9Q[`eFattWnP71C[Iu`Z4fm[?p7qqc/'+g +p>2ah=?sU/"Nl\H40iLk(#X\dCqO6]),0U?$kOCMR;]9!6YH#f\CaBfnA9@HklY]" +k`dP2=3ol?A[WXCVW+Xn?`hoBrHMqC8M(Qrr?\0kUk"63%_XlK(e1U.V^Pnkan-jK^]],/6aO]Phe5_55P%HZaH6,gi?lk +LU,laKLYS?nVRST`5#.:-WL\N5C32-UH%+d^L%K\"aW5QoJ,*+`I\J0Q0K]*F5OrrD!8b@riKfO=-AQ%T`oK +1qSR;3N5mnIk#h";a86ER<=k?5naAK(O[-,faU$@m*):F$Ldrj6t]BpVQn>e1^_an +CuqdX++so4ggaC5sE[E+"ET5G,[G(i8Ad^K>6?,rrBELX_%%_j.<+;#JaXsI9XZoZ +l5O0Br#IfGksPAj:)O5Q"8AYj&ip78M;lom,uYJ^4-Y(N]DX4tZ"LJlTCl1]BWbhJJJPlPHWH +QQ)Ra8?qGqcW\`c9X(lnFp)_7Dh57pVI(/.W>M^j^'PN@u\9rF'f-niCJ>IN@)8]F +^c0L1>V2=2X1>;O[/:9`rrBAQr#@*Slh4\8'/,$O1[s-udl^d4c`UO,h +SXIQ[CqsqQ(aHu=UG'C;1^TfS.C_Bpe`r>_JAOW_oNZ!,"^HHH5m["UJ-kKQp[S4h +B'4r5_!rpK!?9s5U/8_TF]ESrr@\NIq_&ht+O,)H'Prs%lIqC6qJ#c#O7P48*&(,lYTC@!d`fCT(r +r2B[L%oZGcP1VMm1CJiF;a/fkL0e+bGE\'M6hqllr4C3sT,dk2i9@?[j+kIc( +O/GRp`JLt+s.B\P#s$l^;"/sn/42:dI6,sfOUHK\mL0j-C:+lkiC'ck^T]Vn>n`Fp +I:&kDX6nfci-'m*fL$u"WmWV=8iO?YP]l>9^n@Ci<]E'pe:HkUU\et\rQ.QYe&'l` +_VmkKSEEcX4IRt.;c#^)<;9mX*`d*bn<$+YA,U@D:M9Kj9>t3jWaokT +r/9C/ZbOWgWZ1<)^C!,ZiQ9^^3!GRGfsR5D69B1g,T$]Oml:pSfG+%hMG%m>CI.6- +b?pYg?FlefhVI:7MGT=>j'#%gYj(PaSiu!a(bVO^]"\r[ML%T<5/\pY2aGA'8-83A +%5dqlmM]O!Gj.`!.n6d*b!2%)>J8nOe'De8eI-8^E:5Kh+'!q!!Pj]QJCTa:XNeS! +#kgLrr@ZHGXq`Gn+\8qr#sWJZggeCFXn.":hh\=,LWdP%;?Klk(#W*p_0ArVLQFtT,hjG=,n;"iO4>pO +DuG$H]4Khi^L&`YWXN"Ns4b7+8ch.`n/g^W6^06eeV"P?2X-_jDeNoK:\;.j59?&4 +ZmJOYntA`@]n^?$suJ(#L%5s%P,j)!8r``(*3tc!.n=#'%>7]j+G6gq.dg7<3p!r: +=7'W4EFifqaYAFXF(6BIC),D\kTu[1S9c$l!1^a)USKa_^"psCeSCbYUPCuKd'"-3 +:EHG!UBN8giaI8,lIrJ!R$\R!SilGTAT-7It[u*rr@Yfrl*_Ke6md5Bm4mRMT>cPT +pF5rVq1oil'hPSF]W/2")U%9*=o8s0C;?i3'9M(97.2Y]5Wp8lPFV/G6#e&%%IFb1 +:d$)epopG#Sm\G,#G;4!!cq0lEH\`l-_\3B\3j79hqbqmJ51$"Gg'g#D58`"TJJ<] +H.Emn6bNjj""tM5g:t#^Cnd2I/`=KkmflTC[XP1e]S!+Irnc;hsa?7*-cP`B8gs-( +,^]e`d4FXK`a=+!@>#.,hLG3)9umti*Z7=1ZA;9NIPRoKX0FerM'$]pqboY[@(*;g +Nf6P0tRrlOdHgH?[RmD*7rH-n5OaM>G$HF*0XEd3$Pa"Y)+lr;2&7\00)8C[='sF8_]\m.tkfsi +n[)\.;l"smNt4B@.MInBcQQ_A'(q(5PR%n:]*',`oEY^&^lI6:Z$YUIV!uV2_f)T[ +?nj,:;q*K*D=>[=&Gd[Y:;&u9EVS`V"<,.*r4)G:4M3JPX"Zd*uikq,Jnb&Dd3,8! +<'e0re=4"Sh^PMd-h5=6#,mbp`eM8-i=Y(_1LK0,#*B@5*n$+]meR0??iL,B4a\bnDlmZa\%U9M5d$hcFSe^r%"RDX7jN/<5TaRhb +[7e0Z9e@lHjXPRHN*tZbQuKMpOC4JKmbp"5Q,p(\+\&QC\pPJ'"T>Mrr?n-&<&Ql6 +.stPN*%gN +9@M=o]$\Z6uBJTCZjU-9$qaO$cb!)!tQ$ibW,#R577/KgC^U2!;R`9$L_aVkCL>bI +?^"A]Q%F(j#P,(k.\H/a_^\:?S6iE;BYj,K"`=s&@)#=e<_0eoGRMf^7^Fb\\[@dJ +0&^;A(nXHrr@XU>DlkBDt^#*_2EQk?NWMGb5>>pTgC@9J":t"rr<2XGbb5=^M.$tP +P1U]`Aos"4NbG9/HC3[F`6D1^]TlI39k;?J+tTZOo;9V)8nNWB(6%X\K+/j2>,u@< +7f?$n5bN#F`5q;k6fPrlIKte$0esPT>8eYd@M1kpiB4U5On%UrdZK`*R%(7EQRo3P +MnTrHnhKP#6rTiO%dR$W3)@*E_QO]KDQ<@Qi@$h$iQm]gpf:d9W/7CIJf__^;<"G=!.NLL1p?$QgLTB(J,ld:rr?IZ%UC=QVtfe.`I*Y4;h]reQ +P)s$rZI9E^8odq5@e._[fX9brK89=h]4m4b(5.;W^[$0"GUXkn?;U?A?TYC/Grne[ +rWLl_JN^:]Fh^7(&[9?E\J4.j":O#&RWpEr#?5h[f6?4T>h`K3H'Y=n6Gd\W\N5r5".2J4EthJ2kl,Ttdfh%lYA-L +Ng#X?MXU$"SIX"rr@Y3HjEs6/Y[/*BaR5fC!4]9cNcRPhRjfcK,2mRObl0R"j(*Gn +*+/lq^#ntn<&1ThZ^*`gnW-S]c88Pg-^_?.97F@\Hbp.F$Lh[hsb]*KljAFfpCYB_ +55h>rr@b\U5&6@UK2Ln!("L"f`r?WC\05$K\dHYZ%-)(B>`.fjesT]A09VJ[b']Wr +"ig:;oenh*^.NJRp)I_!*S4qU`Q5n5A/r5N,>?mH=?gk[#/%ET*rm#K7`7^UDaqX +_`Z0l;/dcng[D0mjN9Fc[8gZH3BRM0KZmY^.8.6/FC42]DhkApUH1*.eo3)no->qR +lN6R/C4VlIK'9MQ2R7kj:2+h$CAVLkPf`l:k".uhLE?`IqLk&%0$;8oOZ$],EbFsn +-6UtNC:Q;Ehm4aP@,j:_DdV3]`cWMh# +9t=GkujSY!IQV<'J[O@!Bu+EHm$U\X2uoi23B)$QPaNR3SWre=M>&[1e:,ES:5P\o +geom"%W-;Y5[8V(q'=14@u;m:W:ag>Q'mceSG]fS-L_H\cm12+WHYr_#2"2MZ$'F! +.rSq+oSBh*.Gp%IQ[S1IPpmj`?#`_p:$pFJ)PkbGZmfs/%,7R?oVjIQ/jC;q.0EBH +WTAPr"*mYC&\/:+Rl2:jY&2\3q;@:rjDX:rr<5Td@g"_r"F"eQJ0CjD)SAj/BCdFM[\K +oK5+AK-1;aVFO*a1r4C&mI%sMjN:.b:YX8e,i-,7oB2beUAFi-',sW!ZP%o<'"m'B +J\:/TN9q+44"5oV9amF-sZcP_*[U>$[b26Q$qUe*&`CkX,e#d&@trO%ZDN4B$Zj`= +G*+HgQBuegcl$m%fBc):5kKg&\]gLpn=F!X)fYDU5EQA0,q@.)=s"eHJf=N4UY9i! +.qJjM_6-Aqfbg)W.m<1Al[PON&$D("Rpo;I8[&PEKYei!;ZWpp71Vb`#Uep+,t.`n +a$ll`MM?bkVT)i6AgG\=V\YI^^s6O(%K?2BE%r7rqM-9Vnua[pn@1Ip5AI]J@(p^l +0alV7Ze[7(.ZPk5p?WE+D/YFWXf2q?_SXLVO%dtq`af+dnj).$]Pee3>ZiH.>b>dg +PYS5pa)Pc[/U+1rkj0iYjf9>p3]]_p2qo^P_Nfo#L\,TO4f$B9NGIn*uk,NO!S;kg +3a'T!sOho1S#G,3BaI#8b2L)hEBkSSi1GQb'"g6$V +8%h81humBOY5$DhB0%_hjig%W!).@0j7m;peTpe=%/G3_r>:/ACDu+/37(BaLlXrg +[^t_o\V30^ZSJMU5M#6DVMO^K?m-ir0RE`Jn'8Tr\OMOr*/\-?go_hc\Pp=k4#:,h +T&OC1mm3e*Dd$e*cZ.Yb@I-?&%U]6qU0k)h]H%c&aB&I*5$30j@A;`dob92R=/5n7 +tQ3k?9A/e"BFr^L5#-CidK+5'P[N>j."^=J3;$U\.!\T8N8jh?YPlB\rMP/9`TW8r +rBlOIKfMqp1sFDTm3J5pkQ"shTo6mAVQ"X[:^s@hY's'A\t!k,7nf.E)/@`pV=Q=- +Jud\J_-5)h@TXb^*W'qg6sNMRBG1Fhq?\AA1b>?,X=*bb8q@Ee_g-RGDC;=GP4/*O +'B#LjV,U:;oU(F,_C$CrGu=KkdIG$7Baj"&>8<<\ci(]NT6Z+-b%*6MHY_Cr$T7d^ +Pa>@[j:cQ2 +bbUTXf+,!+uKO0I&C"J1?#RK(r[D-L]';X.u!GK7?@aX,,J! +$OeP4$t!"^Rq,"30V6hQ9osTW +-kCRGPXDk?cOOF)eXe>1QY=bV@WX0Ndr]SVl]_`c^U\8>0.f7cShqE=l\%3lnQMgk +N.6KWRt@!PB^^#="iQI507LeS=PKf8m$+54Gng^f,M&Vpr.PdN4p#oi=CI2$p8T>W +bO=pHqoH%P5A%B&$sosSZ,%h+-8_m&gWs:*1b5,J]2TB0=J7=r%W;n;F0E+5QhfO6 +-(;.+-chLT+l_a8,GCGC@Pqem<'9=Gl/eb5+K+0gC%G34K,+6jUMrlLqrEbmpDaFn +8'Jf)paQ%["$&.FJ"@q4Ra$K4rr+TSkfdO7;VgX4ltDn;jh-7;+b_&Th\n(49>9(W +d)9Ypm%o;poDFo6MFaT"K(nli'nq)8l\f/6,NQ)TYU3'eSk4=nn"Hu(`s%JI +f,0lqLCe-^,Q&.qONE.rSs#gn[%76Z566H]1?j_=eg/WgJSjR]N(Y.8o5V-KAILT=0 +jgtG'bpnfBD"/74Nn!:WgCarO2?0e:2.MJhd=Ym!@,RN +M)Fn-:^3I"4=b9n41FR*0gE&ikLF077AU.jh7.0!(f_J-]cii"Rn@+&g80FMf? +-d)">j&gfO6'Xn\PKUJfXM(SO*\Ic+Y#loSFsi$Hs-=ZfP7!>eGO"[!"/XN2O^'uY +53=c*f=%[aGIAlh.HS;@(Jk.(qbl"6,C:.Ai5TWSj':J`>=Rb'dE/i=0TTU +.JnLTtXS'SEGr,ERqgAiSVHN!0$mm[+"EMT8aW<[e:4d.6!"?Q:^+KrS2Ed]G:d_! +8oU"9k8ai67qOMdJAbLQ8e7>S&;[67KtYN"8M\o8E&Er +)/F"eNP1%Bb\Kbc?qB@Ogk)#oCe]#&W6@ap&OSXO%8S]n]K>>a,RHN21H)#!lnNXe +9%"fK2@_&>!PJI/:/>,gOUIO`I@/9.b9e%o*%)tUd.%a,J"o\+1"kMJ/UTC>`_5 +8RK#$VQO$Gc*kH)dUJep@e3Wfd)[BA$`d%4H9Ca=4XgbUoDKc3QFb]^\B55Nef +5s,p@rHKtfCn68ObQ^.4&.q[5Sq-PG^&08O5Sh$dVfor(@DG[?lj.\7DtN'(;YOH, +mqnU%GOCU/$KV57*3`bi9Vo/-V^+AqYg9SHmPW/RKf8CU29]P+:N?5>2WGP`In6o9 +b_qRAE7j*%u.e!B`9s[rrDG/nMh[Wt?OA&hXd\jPY;d;i/4Y+Frqc!DgS&=r#EGHgd0;V&<S6,\*Nj[g.d12u\"L5@t#4khsY$+:T"QX(W? +Mj51628d!GT4P(I]KW!41s%DX=`HK;d8,'HZG][3Hj0l2@(92EERe\;ni`SR`FdMn +T`X-cK,kHjcEV.#mUniPt^4thh^lRKQ@0ll6/H*E=iV(VVHu7duW>I/#_(HnLo+]&\2!N +LfjU0f-ba]U5-Y;'e<>BcRQn_Jfu\lZ_\3rr<3udQ_Pd"T>6&n6_,2)p:ekf)"qh/ +mGG(Y#cKu.F`![3S;I]oJQ5Ir#F>XdneQ-+n-jdPfRtlnK1BNF!SuOp1\GRB2q9;$ +"[AN;?dXHqEa)$"_mf;Z=ECkQ*l9[(436k0:&rhYUZs_^b)pAlt#WO)'9rMBqGZ[c +6!SYnLql9gV#[>)*h*F=_\j1AN7Z*Fo?#^q0Z8a@L&7bq-\IB0l>F(>CTQce[T\Qikp-`W-QDd]U +i\bGf>KoC6S/>G%u+F.,.9nKBY8IM$B]pn.RJ\d]eD,?m6Dadr*''*4s0\9pi\bgi +uMHIA`S]2.si_:FpFNP.Q/>tLl1shp6#@TK!YA)pjr<@Hl)<3_oQu&HGdRCQWLbgB +#0YigrtKqOJ2RB%$p^04*X[JM!nq+0b3"i!I7LUtdo8!%,pa`'93 +<&u1e9hgk^tK!bV:o]9P?]E@$QO(/_f-!`.)p9s9`TUZn3?WLn3=s_1B.=a2+sor\ +/':Ko1lg$FAqHO7HAXpV0+sdnc"O+:C^1$ppX.R:=H:pH#@1&ji"'j,Qml7X]V\n. +\JRG*<>#H=K7gk#>pU7RhA"H-?X\kR/;IGY=\fmGlIbrV8$^d#(0POOa[g8n<_)I0 +#)<'3g\(#ZCPQ/MWqT[@6_WQ`Dd>F33UD=.$]0AC*gt'/9#jpGH^7LrX26T%1c56! +.m[hcnLG<::@X@n9srMZ.VX'jY<=/TI@_@pp`3ZghF3TH_\UqT.BIK*Bd8DF#qqH>N+YOepk0?hS51Ro&.8%CGF +:9o!cj/;)!:W>$PPtnLi:F`2qUfl9rp\P.J,C(^dAHpDJnfPj"*`Fo[?R1oDVLC9Y +NPocrWp'6a.e("5eJ+WFQj]nC0'"A'r&BZpu`9IlRqcU2#9q +^MGin=RZAg$?JCVhD]KU#!UMM.g%$e09'iKIRiZ#2t+9&8TdLltpD`&kD#Jhr\eqG +2kU/9Oka9I-'>\AcDa^Q>=PDi:qF6/4WA]60XM=7`cmq,%Cd\Jpfd4j6](kmtA(J! +;J&#Ylqcfcm#s1n9e+/c[-auo:p1XM)2Lm:.j]GI:;)q)JMAmZ!3h)lFL$"C.SJ2/ +#9\6OEn_e,D/5uT^RQ*+!sqRA/!oNS1Ro!;c]q*0*f:XaH3,F)-i3rnF'OR +b>LN'4'nt&T'mf[f\#,Nk^9rD@:>DZ&[J?M#RHl:PiBt]Dhk_S,U]:--iW-Dm\ib\ +"4k0+1'-7%k/eBq+KZ(*cm@k!VY5:<-$3&l],\gBNb^#QH<#/4)fLEe@%PWIT;lsm +hK"#-c9#SYDDX!/8X[IBk;*JL +&h4\p0X+VqorK?rrDOS^Pt9-N;ajq8CCRAe%aia]^_SeXfe0@*dG,a_-%O/Eb!Br, +L`+Yq[\b%5OY2N!V.?1(@FJdI_rQ1%lB(o]LSd?><*f6V'`pZnC_%%#0(muPbbb8Y +ihr5%WUfA)ZKfRFhT-rHc+aH*M8,b=fY[mJ$&L48S#+X%.d[nG8o>KT]Z[oF.[7YC +[:GE?7""s@GlPe;>>.1f]p$o`O[!MX%=(?AW/#>CFLCur$VY^891U\j61:>rr@SO$ +@flm4u/oI+5_E9gnOSN:0"a-:i80aGVOk6FSK_A4?JT,6fZ3]7IaUde>ks-/U1 +r;H`fn*1PkFA">:&b3G7Wf%(^M8gfl_^45,(Bo5hI3q(0b=P_GP;+&gn3$0.6KV-- +3!suX@kU)@d1?Zmm$;::\Z?1O$DQhh\^Kc'mq7Qr$C?9]L/U;VJ9.XgbS81A!Q.FL +.HkM^A/]nWrat7`ulJTQ\qt`FUnprOB^MX,KXRRW[tuE*J]RUc$@(j8#F"Wn-[Uhm +9V4c^:PZs.Rmu/(pCoCrr@\\^*`+`%HXG4_*<^omiuL6p:Af(%0.UUKR6L^i1+gNl +Pnd6r#HI@"5pOFem9?k6(m+KF38.)[a_rc1>G4_Em.iO='g.3ETs*1!<.gb>c@Ijr +q`f,rM_W0YC89T0:ko)f(t$'n&FGWg:o=`TD1MU"(rJ$rWDJc5F//Fc`Ys4#KP)r] +S]>E(aK3DTQI>``r?&.FT.d"%oUfYn%K5*MSfAn7g;N,lB[Uf]\RNtPK_^F!&.Z:n +gFV6qSgm"8+/1Pj7eT>26_nI"6\U&G91Pi$fV*cr#MgAVQ1m8Qge3`N[@"+[#J8Cj +8rPUq99em*e9H:MWc_.i:%fu_g^W>p`&7-?sXDfW`Va7IFlb,B-2g-IY`Hi"Y"\ +C>@>2K&Npo$%n\44G$.S/`"p?EY1C*ZCZeB?qQTQUG(VdbG +W7n?!7?A-#3C66_clp__&ph\6+f_N6CtbOA$lRe`$<7^3_9[AV&a"7)HVbHHr[&tfZ#IsVX[/6Ul-.-=k',sj +Z_J!,<0+n/$..5i95[D[eF#fC_rPjZpbA"UNm,s`S8e<@C8ZhF3]$'&#'"'B+%MAr +rBl%4uJ-rKPq=1EZ:]'3QJlloaEMa0[;[A)S[@X*IknR^'f<:rM2Q&j,Xj[C-%'HN +E-muD027g4SF5.I*F*J#1LQtld'>BMngG(?dF9E3Rcrs\(.@^1:&EimcN4@,KFhYp +]n:1!.'.m'o,H]e,>JCJQ@%;>u'0!7]"em$j),&P'[49jm`9M$/g44,^kiQT5:[%? +mSU5\K?NSH2cZMn5`nMh]KQ@l95pW:Z?4C2/)DeRQHB!=:b'b]a1Tb- +iX/O-%=(p'u&/01#i)V'5@;tRiE2&Qh'@pXFu0f4GMp.%!bd>2dg0,jNn.qmH;WP< +Jqn/%F':pq`e:2YJuW[6Ze%,oOR,-*4$aYMXl`SrXK>7Q@)!Wm71'[pjlop3a7/UP +Bh.A#'a^=Noj9c(]k0VH-VEL!!SS]+n4O23mWZUrr=,)>/dCqja_&aBm3G"6qgn:j +>m#M/k/VW]dNpmpc2ICrr@Xbr(,Qm.SH[A7fNJVID%RgQ2LRE0pGMfjlF;WG@R8h= +6n31(Gl$tV=89oT`'TB;rO@%4ta7WnHPpKfWDNiF^*%bqeDN?35Hl?+D,OW%VtUp: +Rg`T^6&ZSQ7/p/>6o7=2CmWNX@YYCZ(MRk9%2nkf>P@;gK.DHo\!l;:p7]b^%.dkQ +u[W@G4+)Df&lB,g,C'TQ:?hoJ6219orC:tDCoK!Y.c,bpjHO.WT%^J8;JH\h.'uS* +0(:d$00U/[VB\K +"h61=!C(8iip.E(!&H\P;r=4LZ^oEBkeFnVqNuCLY05C7LAeSuQ"ZA5sc^ +]P]4C`9@T9dTH%=Wp\G]B1?DFruRA0j:uR_R=38YpbT2FB+tEQ+u7j3pYuOn46oH` +[CTNHoiX3W'XZT.0pW7SCd@^UgG[Wa7E6buijnLa>OT+NrDH,o4+\<,!;7Mf? +f>8)*'Q>r*Y[#qCRkY%lappWIr8@[BBDmD,uUNm.TeC@#EB'PfBKL:'pnn*4l?k'p +\kKtj0+?s!V)534A#&R_Ud(>heH?oXY&m#U@&W7)E2,Ha=>,o:qLe+p]pd)Qp`%)O ++a!Hh:/Nq#JMhMYNQMr$e(tCWdhi-r]Mg),`CW,D?7e"O;qd"TFV,&RK!9FhG;hn> +kiU&cQ5WL'2r&bZu&$.\i/Cm\q1g^0dD(n`^5sK%AXSrh.uCB#_'aa*r(6Sb3QpY( +"W@0Y(M%]^S +jN&p*;Wag+1%=3&<#Ijm)c#gpfD2"3mQ_]XXgT72&P/*#j0I161(#Q$si!5Tt1(&5]*: +Q./,&@]T+Ec!I']"PJ3;XobFr"`5O$SiRZRRU9p"!\hei4ms_p20=QKR[B*JD[cD_ +?BB=YBB?s>uc8RU1jD![W:8MG'X)fIl4=YbN:lTj[D1*T?Io7!,%AI:E6?PKAJTZY +hSg)eH"NF-5XJoSDq8-rr@WjpbdFRrr<2Kn90GbC%0lY1>c8.B$oQ^*d\*NOR\9?? +u\FQ\)=HEg3:IG"R1-aI])5,SNBcUMY>bInKm&fA!'-P@]VVC?O_JA^TkP)49bV)I +a=-j>%R!6Nr2BsC5[57geBu0dHtM`cMh:Y!*^VQJ_,eX47J;a/:7ocRiN<^Y^M^2d +c":[U'sZ5)#L6rM2:]M`U%?? +O9MLGMmBW>\IFc]8'=Vf':]6E6Og1FFTg\DooiL/+g;@rKVIf1;YbY#E%P418PM=2 +$Z5PGF[h2i]lp&*Bibl^-4g0_/4L"IB2jup(m-F_ +V^,022B%e$9j#N/5(1=Ssucd>/8TZ]B,8&Es1@fbfn__'9g[dfDHGc:H*d-7$=L%` +t+"`aW2X4_]E_@(%pI8A/"_c7LD^"fYGYmW6B%g_\^NDO0n,Z+F!N;VMDfChH!_>o#kGeaqo9qksH\#KN:jH+pUWJHb[Do8\[KI3WieS./B,$n!!!$*7oqQgWM/mg'BO#\QHkjON^dUiE[LZesBQ1LgS.[g +>/'o'pNR.>M6t8f%66pBbD`INKY&C/21Yh9C\/T.^+LW7(sbJg'K7I'ApKG;Eu0$- +KjWS8cZl9o="SN1Z/`L\.raA?8fqaOMd43Cn>EG(q1V,H4ED(V39GA\R:na*gl7Y' +dL5pg-hBN.g+DJ4_^PXOD02!93(d`+A,H%X/_5%DU!/CP^eH/9f?8q/*f)rct3!fm +6D85C"_94G1k!'`Y>lanY%5:O+UocGls2Ld]M2,")OIS;,`K1F,h^>gF-L:(s+p&R +K1R(<4'FC:S3i5o)jlP.#.]8GAA\di!,-JHOB!9`h9SJ_8fA%$1*uYVVlrh!+QE.V +H]A$ZY!SF)gKXWhqN`3,%G6D$BpSi""0pOeL_bdC`DG183b'p6m>/%_8.-?"!N%Wl +Okq]"HF^:(4EK7DZ(uP)1TJrO:$:!:rqiiD5U +,P2P!I_4KrmeK>J74^>N`QX/qG`eBS\;[kbQ7G2"!ghXC\FG.%c_UDcT(bt"aQrNP +'/%k>AE,!"1`n$1Y:jKQAA0L%BZjGr7X,b?9iKml_$o?_TW`Srk#.9FMMK +0Tm0<7@N`-'ckCi3K]tpNqJ[-PoU#0-]b[3U'_1 +8,9#Uc(LQ!K25CU.Ppj0:#!D6>W8Y5hP)&d<(c-pNH-NaWkYd%.6,li-.;RpZB&NX +?dPN4HZ-B@m!SGDf[EG+.m0'IM;h+gYHQ>;M6-*;tmi2IS\R4C%s'mg<[g#M4-qdI +L9e]gYO=0I.LcSo>P7L*_\.B\XMK?,fFB:k@'N3N^sne24,%3gUqmq\aL:.?V%*!WN.O\,70CV#D0T?bk(ge%ZSR^PJubRBFF6[>[6"um^AGWbS)/cPgeG +k;i\YCqB7B-dKWU&7:,R:.QPaW[2^Yp[,E(k;4ZoM`WU[(:Vs>b'ThM@4&V3#p9MP +^H0pGK9),5Zd_hi4f>.\q4S=NdZ?8)93\;&*(o^\,QF2K(5bXm:*um`h@URph&tgP +K$\e:_HLrHe-_&"G'ot`e@DZ%KYS(7&,E,4p^Af37$HbNITlJRVHQJ?cT3e2"Op=n +Olh`elKnRTuZ;/"4_iJGgk^Cn>`+5?4?/ioEf(HrB36ra9;''`4Sh'DRj8T,"T +A!>drX*B/cKjISKfhG.nHVfV#)Ar*hMF)TfOS+eJMU]meQeDJ#V%h#Iq9#hn.,uX+ +&DM)fB7AFfqh?AJ&,Rm"3%)8mV?LoOLPB1*cW?[*=?[%*QXlaT`,=HI/(D7^C^ac- +2]='nCO8k,B>ZoY+f;cK`cKgV\SUVg,e#%CVp4Y)$9l2k>?IZ&@()s)%S[e:J0M88 +GQNglnTO.J3Q7_i8Yn"C7ZOI!.n;IZ41!c!kQ+TUhQW;)"TCi"T.*sikGmYaDL?%8 +E()H+l#Z0:eks?j'T3Dn8uCH'qbC;O%QgK->>.2FehG?MSscdAlgNRnSk:FlX,JI) +q1OiZF&D_0EY`2Oujb7ZiS5B$32q6(5]BpjV^q;i7B'E?sh_'rr>8^&'@AEbJI%,X +mNQ"IT_GDT9XC7,ciY:o#!\-R%/BrblE/$rr<2j\Inhm?5PHrdWS4Fnm9d2,[X\2k48*RjVE\ROMr"m;IBRmeD.d&Ae]"\W\,IJ]e!!Vp^Lm' +@DP0#j:>-onY]6>NOLaIhd?6nN6HYpj$`V!!O_EEHXHGTq)a.)MXT$WO9f?>AL-Y0 +goS;0@062S,r]:2>*ND`GZ.?`Yj.siTV(Jg<$;16YE+O]?g`e`30RsM`qVqj%Yr!X +lc6u&0maDb%raQ\Z[1V!.m`jKD%+LelfZKI7j+T,[eXHn8nDf^4h&T*0torProNV8Q,7?3U*&7Fgm47l^MSUX`*Q:=k#42ST_U@W +R!_Re[>W#V"kn1d0M)ufoS,]$mpJ)f#2Pq\gMb+_.Bjt`uTk +J(?:`I4$t/iOAs?J,/*n?;1uqdFjjpc%`(isq(/*u9*F_U<6!1R@&nH`OCc>.eAo; +\;5C@-*V&o"uFcqlVq[i=Wm/Ogtbs+D:l<0DI-eT),AO)lgd[ok?:S25Jq^J0uEec +ZV9T`kJ>>49jQVH:0A'dCB>C7?Ru3DWcjI75Y(nPQfAj2iDBddc[f6>Z6$tfb` +M=9Rpjs$6r[-&WcllaWinS8>9S,i"nQ=Kk$DcU'YFP7B(0p.Y7Fe6a;V^&46<56A! +0R@EH/kE7mUuZO%pZJB[aoU6h9C!lRimS'GA.qa$E/aeAgDmA5pVNl,hKEf^T^f,' +rks:C0G8crr@t_fj)NCMI5^[+./&7m!0>t5)c.pi)]jXr"Bm_#q,Wc48+$L]KNC]> +0LKr)L]]@[YN)9ce7_4Oq"R7A<1/YO%nDiQljo6"XA"rD,*Y?4Yd.8H/Zq;#^O5!<1@]p;1OX#5hh!A,q3Pl^`L5M^TBD=Vii2DP +e1g0X%:Ku)0;TRR6H=Vcl=[l[h&dDF:kDJ$&rJ!OMQ4CA'! +!@[)0o5181fNrbB,o%W62YNp(\gE-j8KC:WIZs<(YZ(;gR+>:InWquj)gp`mT^Th, +mB@Sa$>'!F8/Q:nf7u%,Pt!U'4",t5N;U,`fcpFl@*V?4!/rNhLIh\J ++t_KQLPu:ldP?Qt@ug6SVNq/]#.jfVG#gpM7nNK2Ism=2V=D!g\JbDI@ +Sef5J3Q1Sg]%9K,ds*m\nKq>rLe9jRE@Y#cf-eP!)e%#8JcP@$\=MeEK&Kqinki-_ +(dd)40d`$Ktk#"[<3HXm8\:';UKAm3SPK8f-+W(i,1i6ZcdZr`3'JqUCG#[k.CrZDs\))0 +3?^W/H-.W+7U_2o_D+TGM8t6BlEYM,83!",IM]1GCM#X#pU,pf.&f[Quq6\,&ch5U +p+N^B5`J*m4Xdh?DCR\6#-j<:TbR'&:HgQr_s]$LPhQE\L"0rj['bfam%"K5*"1I +tle^[aT5da.cdg)`\!^H=t!r'tIoF6c$?Ui_+b,G[Ag?5Nr)IP6)O;`_I2NCj$Knr ++tW3-._a0d:k<#GE/;BajD`:c_oTPdcrmUZ"@!YfJO$`FkhT;:R9YCkXZ>\P!-V"I +qMg=%FZjlI5a^Fik06)A<,(mZk&o&N1>4GjB)7"=8P)D>+`Ub:Ph+INP%3A#[8<5= +k4-Hqb,\o8J+RU]TNPrqIED-i,,!I>km2??j8qbYfGS=?h'peFM!@Z@kgHQfjIH$c +NQT4CD,(WD#?X^bFeIH1;C`T\pkDc)VSH;bkkt.H61`'5T$*sdG>=0D8@tlQaK=a+ +Hq@_[JM[`)pd7ti3ZBMbP&,qIN42q*Aa\1H2dlYCWV#2DI9kD5DA.\e$!NZDqUt?5 +IV`MW?cn`=2_i_a9h,2!$N2jSq(j,4AZ:OYP]1NrrC$,pjWXLAtS'hPasR/)heCqh +i-1Dp@e&Z>cb/]_%6Ya]&8%4W3=MXpaeYKmgAj!<;9DMrr^F!!P.Q3HS`d"5J\Mb6?Y/d +V3_b:/k$peNOGoj,X!,e>j`'PU(\MorasjF8cZ/T%/>KS2s8W2W@2LY +%tL>(u`06GU#WuCi0=GY\,qu3_OXc`k%g-+W'^Sj52ipBEPsHR^A!&)>ns1oUT_BQ +I>%hr5OG`KfH!\Rn;r%h6W$%=^TLZF0/R5A@;29d%5P?uafe`dah7;EqRR;/9OD+2j(Y] +jeVC:"^ZM*c5pRmm%s8rrBkH5N&/`%=B&_^Ys*lIOk1061J$4r\Sjd%=6K)enbu]m +m=aK'K8(#/LF/7m\e?i7k2V/-`)9_61?PDG+@928&H`@akN[Z4(CKu]HdC^V;[n.M +'&rT_+WTOc%>9!p;qMg.X\CQ4NNDMO1NF!X9VR;l8@RO5<=:1Dh>\@\,QG[VYhC[k +3u]DWH_(TE61#EAD2@QBdC3R&`$7*'G-$:`^$`[hDosJtOZ!_$fB/7 +Vm4_cTtoV`/-C;E$.+qX7AH-=+u.?4"MIIpeS*J.W+ah6ZnuA]F_i?TFY\Cn4nn=g +dN9RTkP(llp!=MEMubh!5K?@L\qi^_LbQ#r#D%je$hmZM30M4$=OP76H^MNko&i'l +G1U)Jj//7h!V@eeCgT3-ZP$o`. +X:lkhBe890*j-j/@srr>XkphTR4p +]9;[(B2o-))^:cpfC2$ebgBG7YaFC\@5AE8MG#$'`5Q$!8->%@_V&*._%G@E.E*Ho +$?Yc(q'=BIqSgT#lan5i?<43rrC5;I^Si.q+_pj\(s64!V5FO(%9%@>0lPPX'T.:g +V*TWH"SB9R$FM'pVS.VN.=Af(%-PY]G%'nHnk-g(7J!f$nJ40mu;0o_*1KS[m&re_ +u:!]@2>X8+*`Ml7m9b8fq$1Q!@[cJ3rM5-f=htmp7I(G*0T(tT@nD*#Pa(=ZiMM4q +KAbe7WeNP]VYF(TM-KR0KAa/OZXkcm/I';i-;sDq[GRE4p-E!em\Z%EltNTD8MeQ> +<*hc^<3_.@FFU_/t``5jppY+F,1OJT>-PqD8%%01EF6Ng +6?('?/Kp=iudm=rrB2"BaeAO#3Hc`8Y1"rp&:0UZ656m(tp2o@F.5dJ+`%ge4BMrT +.,QNa#WuoXIacLS\A1drrE"49X+Kj6b=0ipE0?`ks,>"9I-5d8]nJd:Mg%2.KZq1N +1;p91'9a#!dlq4G=>d1PJ)^ +[M.9(C1-$4EBIsrr@$S1V")B5MYP8%srI)@.Kpk6_7&n[,_]_,iq=!W7* +u+W#Zt97;Jt?_WdLbF4;C]rCQrinF?YfoC6.u +(AeHrMO/9S,NaFq_bdomZq6rdRsR-LAq6@!D.:["bGB'nunppit8U.[1srYNWB(Z4 +hK=i^U2gme,KEa7=Sq8W9(T+dB:,g--).3BE6rhf-f1DYGStG^Zae)pD\e!a8Z-He +(T9J1kFUa]Hhr%dCb]BWD +:2/Q*.Q-dg[aat8eB';;3J*m$b-'+ocK6E8cW;j9I52j^]T^Zq'rUF(hhdt6]M3P2G/*O5\',fYXn5?r@INmab:[Mk +Vk?5h];^s55piCU]1hX +mU(-.39iVUl(t5DT$.s++WF1/QZiWb[CUGWNTn3N\n9=YP9M1^c3t]])McIaO$RJm +5pWsHLpf,5Tb@Q!5V]ci0BoD5N&(n^VXhtp1MkISNQ2O0X/rIC$!$TX0"Y/FW-Z@\ +KjnWU`b[Qfc5QMX*u,i!.n'q/C=bc.5L%lY?3Mq%Iq9HVB,$7_LAqh`s?%!p:i\Q;kbg(G9%-"ugrr@WRHq^p;( +74Q06aaJU/p0]BR.WK%SX@:R:rl5S=gu>gNcIDgfBB,g(b^[5aTH<^bI=+UqmN\Z5 +N%+HEIZZZiXVrc'p`!MJ#GUjN%mkT`l_),R7SIL!W!&`m^MgepVHOk?_0KkL"uF[n +5^W^es^1KrNT"_8P!AAY+A>2hV#]uH]c9rKIi%g&\oDWYPJ-4o;hT)em1Mu[GM.hX +s8kklPssud!)a1COe09(^(K:Lrr@XVB +%')Vg0Mf'N"-(bgcrDoOV?h=0HG66H;Fk/0Dlhti>\=trX&DZ%;O&Rjp:lJj3%b#" +_(oN=]OB$@e\U)LJulk:MTege+H/;nFicY?P\,P8-DIbV`5'FYOesSYP:-d;mEY.! +.n^/P5aB0EN/`Z7#o\-A^&9#j'^1\J^&4:^dI.1?NZ@s&"q(I! +-J5@#rTn%#5hcT[>tmBp010S0YE?h*!/>G/,(jnrrE'!K&Z\Mcp(pKa6^S2#Jcoon +Q1T"U%%uk7D8Jfm`sZ[G21=f)%O8X&?Ob<@\]IFi5Vm;*Ta=0O1GgIB:RIpH52\@T +X^W;P'h*;*!5(hD9&/nB+tCDAcD`5YQ"TTML5a"Y>+e9n:0J@L6LQ1#'bVi]!p4;F +/nP'q(pi\o52npABUokk(YO>$32lNr8m*04Pi"'/IVdRi"70%:,QYrn1qjSI4.o +W.j]rrBl-4ppVj&Uk\s6MPKh/&Bs-Mk\L'k2P,oN2-ecUPX9=&..V]a#32-rr<4ur +KSMV1trm34o@)p)T'\kAR0$u:!p8I7U-t6qj[Y^jbPnA/,;o37+1"-AQUNpT&='\6 +pNhoK6-]JEQRHJlf#N0c'`&WaV=VQ5d;f>.Z>c`#L!m$1Xi`3IoVS[_.u8I%6!r_d +_&fSHbo\&eMP3iVR=.O7"6j4/;QUGiO:E>3Q]6fjnh8l_/qnSDqT&*fDI/ha2U'LF +k,UOR]W^b^6q.GL!4EgOGVps;)<&77@i6tC\ctMcH=/P4)H^SQJOa0_>m]fSEC5=; +oq$6$QVH&e&3"G)d&!U/0?4ono9V4k(#[u5Cn=RM%r(?H9pXjVa]TO[cBI7uT1 +9>!e=T."gRf6N/Zop'=Q"AIn`p(l;DRkHIHiq.fJN2sof_A6-I'C'.,AcjZ[#IVma +QTD/#OnAslQbB7r"bXmSJGcA(s#_\" +3KV(G^L0tZrp$R+c/AKTu[.`5TeH6^q]Xm5J^-HrrBGs`N<(un&:=gkScCKeh,DG#Wg*D#@uYA2HLi3L,'cFr3(='!^44FB'!dr\S#HST?W6ic8;F +^`AU?ch=&.mgJFHs16_*/7k0dAUM8&]Z/-i:bJ3

      >5; +t.a+InN#q<.FZmn*KYJI`-[rZM8M2MngAXnM/L%][?AR5?ij)eS!3r"=QP\1@7/OE +)14]m!Ja4#87$\?P%T$CSQ8^nQ^PMFh\CMnA?imMWKTLhXeof)h"1;-[;L2Y')=5! +L/[%?d0%6SZSI)CKR:KKp\UQM95Y2YQf0h2/__W[TR/3_SFTW,=oeQ'r +rD$R^TnMW9le"U8&Fg?:$eM,#T/Uk;p1eAP@"oHp\kPsq_;da!:Wr>,N>kYj1Ck3_ +NoW^\a\/Ag-1R=HSS@EUko"J+b(mCJ"IS@MYncWiOY.^CRNa=3:e+0H\:)BXWk2.g +?(t5?C>tai;6=UoW<99!!aMC`kN)_&ZT#K?Q(c%B=Y=:#aY3[]!Z-^Ks3de(roH.H +gtIerr@[KiR"Ct6tRcE,-']GQcslk39>=T4_dWK6RnG`+JdAQbtC#YLWV(4FE9d/lU`gGJs1?"Vg"E`?`" +S1D0NFD;tXGWj,>L2l6!1!,B?ljU6*@M$U:Y@86\kg3#pagp]f4!U<3:tHVmcVJ+\ +VrH9pD7"]j3K/%]=>s([J`n.a#icYWKm&GpUU:;[J[.Ubke`drrDs3O7lD_e>Es:e +p[uV@Y>u#rJLA"O9s)2_)uAW=7nj(2jd6TqbSm+W9@;IbGK\9!_poUbn8'=R0Gk<^ +-AFu1b%Lih#Ra3:Oc(q0Yr^>!<3$^l)rntqe8$%rPu'\r;-C,Sni6H$23c"5 +DO-&rBICa4b%W>E-ER_pST"1qL;8iKiFt:-IH[\&Gu/=r4]PRiLTte]3o(qrq:?dh +l#F-&k">%h]H7AfoKD.piOQi!)-s5F#nk5kK;iR&,s=n8knZJZm.B84Orc)J+^F"d +k]#pXkE^S88S),d!C+gI;,`,QRd!,G*qn"#lalbgHZhnGRMZTrX+p<\\rN'>(W7c% +W,%4fA"pp$ZL9hSANq?O42_8]mMWp)n8iG$C,XB%HW(Tl2f0i"70fnqe%pm\6;m/L +A,nVlijTujpd>8+&?VOS'?Au6GMc3G56FD+5u\`;Cd$'RrVeF3LJik3;!3BNC6rUq +[$'c/Yn#[QY(a>*hSE"0)6I`5BbRd/r/t0*P^=mpjNnX+UDQJR-h=Y2Fn?V4(nEaV +=PU,%l_ZgH1KF]na#uLPup`kfp1VCq_<]u7!-_!rr@T4)#*qWmjj,iGX'X%n$4i!UiVRG!W:MnF6E'TTC1LWIEs7Oc#&k25BZ[JgPUU,KC4d"! +<%6*4>)2t8M5>fU1P8B!.)5gaBi.1i5WpR?p'X)39NK9oUl\Di%0d,L8gE_k(kX:@ +lUEu5Q:_8h=q6Kfmeb1Vo&H6%X(KKu6Z_2dq8+>@-b +JLg0A]/hPrrB?DGiHFlrLgp`ZsNes,(IF'rZ?Mfn@l*c]gKZ.*7Ki_AGB4!3MMk^O +]BgfN&1C[!V1luoG.1/@g.OhMqK"X3.*U"+7pDAY!k4Mi\GD\<_Il8!EG+ +0-LQE-M3oS)@Z@V>D]d'>pHJa2Dm+pt*G"hD_^hVmNaR'AH`k[eY*PNW;.3pTMd?! +$'`f%u5'I/fmB$Cp(B?gao7nCjOYsJG_/LnZJ2Khm4K<;752a4<3B/d^id"Lp^%G> +DjkolV_DD+nFK,jNF1\OE[U;n*TH5 +rsQ&V:6Td498]!VeRCMLT//0_F&8SF3.kPWVgE2e(VofHE^UA-Z^91OVZXsA;us/m +b'&oHY1#r'E/H"n?pTXL@K\Y`NKq4gN[Q#WFg6f-!rK5q;+Y^*poBk8LJ]^Jdukt@ +Gsk$TAgWM*2'NW.c>Np0434A_`:q-rJ.Ie7GdY2$%`=C\,$[N!Vomu!!Jnn!/#JX& +cVh7"9"#Q$1W)r-cH#b)0gX/nNtPQqIZ4*@.pjN;`%l49af$.$44ls>DuY5ej&Y<- +c+F^G.m[EnJZLjOdg2N!;6S(#P5,orr@[Z-b,NB#f#lG44pE^BJ25FMRgnY(UDt[5 +Xl5FB"pY0ifN.tFpr66DtD\55G(0p_u9p=i"+L:pom3b_UjcGQ>,;pA@f7EoqIXD(p_ +#Sp*%u+9Fm9af*P<<$P`XP::Li"VNo0>;0j[gN3FmD:0:&b1tSpnpUBZbJ$MWSqKh +tX1Hao"2@lX)1B!5W;4i.YMAIpgUUDo9:Fp]km#O6p0/)!O'jg?Q9!\'VOjeA=4#) +j@PH5n#dO/,*@CVe-C#kAhBf+5uNm$3("A_gVoAM*J3j[B(/W5qRkNFF?`O@kL$U\ +.XpSF-7)b7ni&q3)(Xs%:/q"Xt=MRT't[7Oj+;N_u::lJ>%jPH-\jM_\G&Lr#2F/0 +X-aXn&7h=pDA.Xi03u.C\0^*R\J!aNA[i[q`@rth"2]#;u(LZ?FcK$M3Q0Q[R!D[$ +SU\_I4,3e&,%j[!.lAA0)_LD!8sn"[+E-4*pW_Z[d\LW>>7the8N(Q(?03aF +-fiKN!t-nS<70!O8(SRp0*J9`p\d?Pi>#W>BRRK%G%W[_4CXr^n:+$rrCdI;_si&L +pVRKi#aAI7T&8F]8??D$Brr&q0n'pnL+'K6#XO(rrDAX&&h?>/!p/F48+1&L4A-Rh +\&+&HleX6'V6K;`LKh?UA%s'^U,dRWgK)!3+ToJ:IEBQBC('H!!^u&c6!t>AZpmG1 +%Fl5:]B#:?@cmlh'@FqSd.Tu$e$_N"W"YQ"U\Zb@,<'#3@jh]"C3fEB]O3 +@%J1c2LZ1?*C`:!:tcdG2Mdb$e[+]@=NpVhX^43;Sj$lFc1-cZ]*69aliX4=oF[]W +[@`GM?7dL8F^QB)ufmZ`XR<(j)=]:(W%1#rX4SA5M>bSq!Keig@rM\r(el!?BO^1p +,=I$/d^C>J[6eq*S(&4pFi(p_p-2A7r%f$inqUgRh,9RBAn?$;c.28](gDpUmj7Z& +CU#BiX1j))9Pj!el[$[3_O +2n)H);97srYi>sm-^]QA[/t^Kq&PV.=4+=#JLM;%IrEg&Dd*cpHs\u:&b3H2*=HAX +'Al6;1&8?[]Ks97JoV.,:_c%\_s'RXSbH5r7rDeI5n6TVTmL`3CYE9Giu'mF2^5YG +J4bU"oM_?XQeP9l4OKljqofopj.n3_A^>QTTVZ53q]NrBOX%=_P6m +$!g6Ac$;5bROjY62j4r,JWgK5/1Lp#OV?V2(X8[jS4M,mU:gdq;f"J[pZ1!N-j1Ln +SkTFb?A.F]&r+BU7@fa!:W./k6(V;qp5Mo.n_B_A8uZ&+)/jo_"%IhVm5Vd)p$_;\ +P.Kc-uUh[e:Yl`T]?^BYL]GChT0QlF\-F(AUO#%4HZ7'r(".^EISZWIqZq&1)[r\V&"N!i`I)e6o2PE"=Fo8IkY_PE2pA1Z/)Io2N5-^Y/].q1(-OO5Emkj88W;lmMqjOrJUI/ro2!8b*r`o"3hPD +dNAkU"X]Kn2OX^!jpDJo0NBV*.Pb=UI6[gDnRGi_G?T3ja_OS!.b(LGM\`=CEPO@: +6?JX8bRmX"M!_'pW(VQTFZ`o4sFZh"TJIRQsXVM0YY!EFKIW-]5+I*QLa$d9q$UVh +(bK+XaT5BX4uF^>M[mbKDO>r>OCrX,VNc,H\TZM91>2 +R\`Iq5>hL(d&m;RepF>8\?R">0:%i6V5$;Cb7>D1uNiP?I'C+8GPe%obbIL9teGmO +NIKo\',XJa9+EI5>AnP+Li4c2qV'P(2supj1DX=,FjSCg-7)`nF,Lb6+7JmGIfkPeJQO+6fY1hI,&SE-EJn8B9TMk +4Y^]ZYO@UO6:ejE_/iS1%GQXS9,jjle[.U;Z3OV70rnnK#)oZqFS8X2.t8u:N0`#j +dpTsmDD!0d0J!VEdi#K"P;7@W'o5!GfDSJhBV.Il\pnGt%.[m6;`O%?hOkA!Reb9&rOBL]E-3s`q_7qbJiM/Hc +"FT6Z#i>e&OX$(-$O29)GPa#!0BJIkP!QuiR,e$m+(XAl;>i'W0]pDh#ccbceJ..D +A-]#_`?EU1]d57]FV[.Ws_lX0%BQ+".uP7`-[ANho8Y*cJrG-$Y9S*<]rG:j):cAq +l>2#^+ln+dsf@71jj55hW&)$I@TNg*N/smZA$?A7X4uaI +0M#oZRa0SV6F;Q7mP#^ES#)J\'Wai#Q,SEkHFds#KBE91:2n>pjLYPUjm,E#(9#_I +I$/4rO0@D^*]!\IqJ//_b@lQFOB?&AgAcdPssj8!18q:-nnVl!Qk,?rXK_=O/dS)r +r@X]lM[ZbcTC +5F50aWa?rislE^*`2aLN_^DOIfRK1I8m@u=^tD1!.uRnnn?.SU-8j&!.k8 +SfV-7iqGk(tDd(H(UT\(sWBk^;eR8i7[Q'95irG`buX9(>c@]eimJ__&J]q%AU2_P +'c5Hlp'F;>'KliT=-]FJbEI-\G2O+Lrd_tAXe`X[m$<`l+W^",VX$h7@iKJ!QQO;5 +:&Efrr<8QGWS@`G,)##Ihqrr/]ODu292$:D3'l@6mO9lgc-8`KH>HkJ:GB`rrCu;2 +Li3fm,=mflbn67eFXW[p%uuNg"TZG3e;2mN&bT^-S:;8*"G-tpM8bJOZjlQq,@P]1 +?U7<5ItV)GQ&'#nD2]L^Yka9dG2b`k4,mj+WI^h2;=C79^O$S4F9p)!/"eJ%-,J)X +/l6k]4k<6j0)#N=3DLXr"5[bJ)I7dn%acBH`/inU3J:T(:s@,To>Ve ++7_BnZUi\iLE@PE(_4eL?[nd_S-`2r$s=jn6M*23ibK,-2:bSB#HV\''7l!mH/'PT +FV=TJ)K'U(W>$P^V9K6WPn\f5:gJKX8rP.P._<:HsH[OK6J!GqKA)c47O@`Vn(>p/ +BY\`'PXjE!kT)XSYIPQlh*ofBlaKGUdSjIO>]He'R`B%7?^K,<5"&iR42E +[:sPlp/nFJ3OU[i2M4q97*<+-f;h6"8KK(nI0jjYeKX,>".?fEa*sVaUEb[a7`g:QJqNt1FFGRGNdVg8SKhaT+d%WYKQk[]Rf1'@)U@V%d +R+BqW;04'!PRGR;&@/6*KV<14,j,IEF73Cm,KX[rk^i6Q6dKGn1o[CJ+PPKl>-3!:BHJ&(N.EZZa%.,X#'LM$^>5dM]!mm%sWrrBlR]KHdcr +r?b?fcS+Xe)%oXY5$P@H^e^XF'CEdPo:ti*Ejb,1Vc!faWI4BjK.H%1J&W&SW><_A +G_luis5JtLZ93<'2,\/>`qOpUNpHoq?."C\nA!/m!::e>cTYs'O2$n>"PUY1ahe!=G;=0TGs^M3S[% +p[68Nrj\Me$UW]B`?QAX.+pmqIG.p1'-GH_S7un@R#iZ[NC)b2Z.mMIq"P(R`l51r)Ild\*(j3N`@Bj+U9/T1`=1"3:C?s.CmZghV^X`"#%A +Pd4DiJotj?4$W3"eO4kD/K.llP&TM?5U$<9r,#H +18jZLVuCcZjf@ST^0\b5TucMmn>-ZAAaqeLRYpg^%5* +tnPjr%,PNI$P`DrDr5dD'J]91UkH]D%]IB"F(*&0&ujST_]JFT+sS)D[^?'dWe^LD +o5ji>Kin_ho1d>ZXRm6@sAf9o9R30*jbt1+1#m@DYu*d!!SAWrr=(X_FSS:$QJ.k` +!"?k?@)aUK?eL`@_U#!AO)\Q"g7_=[K*O?_Zrr\=AS[-u].`ae"`L].q@Hp56km +NM*'RQ\U@=&sKh$Srrj.Ge%l"G;ESV,R>C4?*.Dr':4>^)cNhZe*1RJsP7'ehm/ud +%],S*Hu^U@:2$Mj-EZS5q,#Hh6h2b8]Uo1.smHM'^5>;iV`$&^Y1V8^Pi'u_=/CCN +X$JEhB1:MI*b/LPm66$Fo/T*8sF!WL!74K?E_B/1tTM +O:j?g%,Cb"0DQPWlHmNC%AP26&J*>N03ZilX.\nMCaEf&:@G$_ZeWa5WoAkL"7dKR +b.6Yrr@Y$Get:CpuVBCHG4qbG]7LXIQ[ +]'Q8*!ZL1g,>N-:-"sPI`HYach?,g20TUk9t-.iQ37[(qbjC@n +7.u3c$renl5sW(>@0>6MXB)Ug&D%NPJGNDJa&'k:qpLV:&Af*(5gq!Gs$GM#`i:\a +DCb3.6-dTGH^;k*tS=Hm\$hT26=,Up6fGXTaMo4j.dTQIh,]_Gh*n]fiikKD(s'c, +RLa*+'a:4n!]8BIVZANEc)[)AJWq-1?Y@o]49qcJ`j^-?O1S,D=9.99.2\;H$RRgo +GR,,*.NIpR<.?O^Z4>5l4<8!eWSc3%ODAV2?ltHIkV/RZtu#1@!-t,*6bKD[f_67o +H2"5SnTd(9^kN_I?Ro/@,jIbo/85\EYir!l5QrEoVTI`3=Wl27=B)#oS%bC*n"(/$ +tWEaB8aL]u!+(;u&m[r8;mircm9I?;X +#,NB>]naj,FiTk^cn$[VYImJiW4+GCs_H8#i=%UcPb%3j.(2&\Dj_'LDArOeTD$qN +PVr?^H4>7j87Xt*QL!riV(B5GlZC'@qInkRH`>kGL-#nTFR;`nQ1KY]Ug'haQ/5!A +=U4G0P"S\7iCR#pj12!EUOMgMIK>>PI8Dp'MC&n&k_ +E+Fk!5loC[;E%Ae:,s-CX4UMfE17&`=raM53U*GpO/oqEfc^>m%KS_)WED'H%V,[^ +7>W*lX-4?Zf\5BIb0#d^=.!0e\mp+7YH>/E=1Q\ +;iojXS-ZLrSnMNaSu2>-]d]elTcIY=#!l`S(jDlXn;+15`BXeM_'KItt2Uh!nB?`(4=or +J*3A>"[(Y5#=LSf.5LGil=>M+$G)#g.%:)J&,NBB*?`D/UM'f0,&bkIlIQbk0V=U0 +DT8`/\!(rQ5J]7eiHLK;Hnm:D:P/;'&-oa'V3F.EXl(B*.$5QT,E$%*jP_F"fh>0W +9Z@1;VLA!%Ik1(NT8k-+T\[dnf]7_)p0f2'@:sFq\;:F:EHtgSD"EHi84rg_RbB,k +FRIj,l[iNj6?[#L),_S+WR6Gk"h?*J9p8Q\_rNTBaq!q$Nf`?:RGC:&Riu?rr@Dqp +Dd6Y4tA`.44l@uo/:L&+#J@T3:+P6NoQb5A)_NWPZkCsIq=;*s1\5qCl$b0Z1 +kJcX +d`Uqp(_9MEok$^&'Tg7KDk9,q>il(Cg^dRLfk^k`YRK\5J*NEhWD>AnOXA7>PXSGV +RSg+],UqoEh0T +HdYR2^bU)!-eV;H@]Fl+7OA0RRdAX"OZaeCElBj=Rc-CZ]L."PUDN"1P+dtIP8[^_ +jcH[eET`-P/7\qW'.]YaToB:l1s*6!!b*#Jj"J[$'jM\Cj%XF`**8T.F(N7,1"X!W +9r#`5Fh*X$Qfu?19J"I6'=$Th]DIBl\]*>Uui>G/pu/bA9"2(33G$WpRWbo6,g_S% +o0"5I:gm0]K%K3U.CG',b(/L(cte>aisf_rLY&_LUC1&G=r]+rp>+grr<28ZkjhR; +pU#WggP9a$[*d:in!W!!&7hB +p[g`/pD.;&B_q(rQs>$kstHA6ZXel2[>H+LDr&L!8"Em&)7!AT?!QCJ&+B6"mLLX4 +:0t@'2e8nM&T@^hNOaNo%dI6f1&c$FLQ9X)^c+lI(7)WPJI,a[J5sUL>u1?Aag4^O +pQE(^PcY6$[!pq>4^&X#(Sk7i`+ud37l%KnnFF::\VVEeabs&&7[C5Q0J94LY:4lg$$=2HQUJGSB988SY6fg3-1DL-Wi(s[BMbC?,. +J[22ERW[sqH(ScR[h_(!!T%c@^o4?[t.R?RlDd2+^`<_f"LM/j%mVbA0eh_CSD[#5 +%6t?UiW0J&_=Z7nF+D)M@KRc"\eG+R]h9aqqYjY*8=b$Xm0,eV3tjE0UuZ6:_[Oa2 +GXC)5>"$E0:/D&ep)SK!^4"6C8]#>Nus-W^bGcgDHaL8j(Z\dT@1C,iTJ/Z!/OJI2 +VOdSoLhhc+]l/p,4cd$m_3%laVc?4N;b^+Vu.:I`8;/U)?.eTQ\g)OnLF&\2!mchB +WRrW+gmQKr2/5)raE0mnd]Y)*d@Va#>!W,^TscSW3/sBEd7*dhfr':q\@Bd4-J?"* +Of[hIQUtOP==DV)d's56`J*VPrNWiA145F$:Np>ti?(<6mX96)iCAOh_;M?], +B)P2:^8NN!!_IdYCaN$rr@q*)itRd9)+II:8`$dNcHFG4SIPqouB]''V=EK2i^GeM +Z!&F!.lglD;d^9TZMk6I_*LNPE2,A"Q#hk.a;#<\F8E2B4CnjL>-h?G7N]!.t)]& +mg%,^)Xb8Wgu5k\kLkbNObE)qK;]sf@_'9rcOd1OHLLfcecqWdGM^iJ1OjZLfg%:" +t>B/SqAoi0DlI;9g/RW!.pJbg6OFWM*H<89Sn67@.PPtlG]M:?pn1FboT,6-(;=pAY-cf80G,_*9e&^\s'S%^iVo+Zq,\*Yd^b=@2QN` +Hqt+p?gV&fT&I,O7UTrHl(`N%=C29!!a6`pJ\H0O[k5A)I(rl'7mMI/;72Mr\?\Fq +ps(j,9WVDH?pM)UW#9!3_0)a-I1t^C%&bXaG#Jk$q*lO;Ta23 +YU748c\ItgV6@=`P8[u`ENd]jGNQGHA0"^[foqN+0Vk_nh9b^rh"PqrrDg@GJYpMk +#j$t9$k;i-h.&]8C@(U#D:$3ftV4d-J!VOGh1=B'qM8M0;3&`m`7MGb*2:SCVp)Oi +/KeYGG"A#cN;4*pYe'8a9*&1[ge?nHl<"]ZJsm+HZ4ceETQQB:Yc95orj"_DBB;#V +u.:qp0ZFYo-i,UXKUQB5p6W@aUU3NQ+4gnaS*l,j*,GoDNWmUYXo^?#=%+i&nEa0TmhS-X!#;tup6 +)T1]rr=YD2t;C5^UsehQ +2Iq8179_DBjUn_C.A!2EMJdKhO9G=6hM2m/C!/Z]=F5%1'o*YaMR<]m:SE(1]?`_] +=?nb!SgE0kb>O47.5b.>$\4K(EO-a?SdH#7=\A% +q">\5TSA20Y9(oZ"\rc]1(*&JA^%:o<@rW8DZ/ATpc+HFL/VRUu4$2o3`WTkBQEE) +Ypj5nBG +,(ee82'D7A%9"X@>.VQ`"h%]*Hn$K^5Rid1J7$TPK:4Pm2%$,1?<'cofYV?*gQ;?_ +qu#-hJ/W%ddA`j'nhm97jZYS(1[&c#8t;dOc[!=H="8\XioI)ZrXY:+"[:\rS]m:V +n0+nkFj^5gIpSjTsL'Y7ba_(XDS,geLUB`*eTPU\`TsW*tTMjZ&VOUV,b$7Cp8A!# +F,Gc39P;K`HeA@r(:>6[r%mrUmG"uZ:/[[%N2']2\dEZWe,K\.rOYKHWc"i>KXF*! +5etJ;D9eO",r:.5TR?3-<"gio:PCmJXQ?bY&X7Y7QVZ+#lJY]k#nfiA[I;Oh8WQ>$ +^U,9Sknk`j3FOCOkRtK$](,u4-,e[m!\iI)TgS'd:oIn'EHZ%D9mNaQe=QU!^Mi;0$J:HKEnXpB]Gf"`:" +&=+Eb"(0?iL)[lHacZ\r&f:)2]ar<+V%7(VbsikE35_Ala&ad)Y4CW>AZ:R7gq-gZ +3C]DRE(qq04BWq#C]IrD@q>qI\'=JN<,_TN]\49>%`\c-'6) +Y,ArP0/ql?/s'k@3uNYGBW%X+2WOiW;L"L/_1&c*ho[37QdG9Sd5GnI63'g-\&sW2 +5iAcpa$0(50W6X08[&+!sD2XWC3i3%jrn=51T+G3mY5&?@_RM9"/'Y-NCQC#R<7)'k?>R7U69I +"ZT<-j%0Mk*p`WW.c?IT,>UG_H54$n4WK\rX+YS6gg+"9O!_$b@,X2jr`9n:&Y[Q= +j@&[NCa52q*npaDd$b*&s#HXl%G3bX%?cANmPL*s!V$MX`DT_]bE=MiFI*CVr?UKd95T.k+%j)ukWa1>o6:*EmnB3>rIMUDpQlQK)m*1=SRe)U8.EIWIk4[J^DJUIL +Z%?QSn#CTddFBMJ($K6Rd;rr@jkRNa^jnnamq8^ALDpr7A^Rm&0i@C4Ia@!7aVBW%*^^ +]g)+JbU/^!ri7QN:5lWrr?T5JI=>u\h:W7bp;^X>r$4=a7llh.fB1qj$_C5$q#+m/ +-MR)Sk%s/*1iXt8Z"n\C(dj4;An)>+7+RRk-BA7."fR%7hUMX[.o6*e-rf(moqt#f +"\NMb/D6>1)QS)#j['->gC$+WT"@6lLhLW]epaM$tSfiBEN!095FA[m_BY.AGUcE= +7dm;%JZ"%IP(MMIN7TU::$fjXBJbX4f!9me&L\6OO<56n^A6Nn]rSJEGEn]%6S3MN +o1(iN;gWm!:WO/(OTcg4V,JE4se8.=Z'@Crm%pXZY,\+>Ae=)h@<0\NL7SSjCWp+n +c&T@G^-Den;$CHisdiNg>7)'6X<;$Uu:$VNr%]];?"r6UT_E(%i&=+C%8-8em!*n_ ++?UMRJrbXn5nNa_%2E1+2;XuIN>>G[I8m"/,1cq?"mB3+OQ-l;r7>VXQF37W^k:+" +1Stn$N2$n5A!]O*[nG.n@8Fp]PF4aAS#^u%\mUom3OCX!8r\\Q[QIqZ":c*UTSPa" +0DM&K&??QrY!!@-^M@1`BUYNI5^c4?5P6ZSpQq&>J_i$_O=GTZBA>,_]GCQfB"[`( +*E6&pn@1!p*%t6FlM;>?PC.(6_WC?/aeh(V4F`IE60Mbm[7qPO!ZRBY]:47"3l6gr +r<2Apj)D);>lIO"0)2mpueSk7p")H^:$i47r77Lc5.?hIaI"0_%_S3_^$l\SnJ5la +S(PFb@.li/!!q:a?B`eho!4u>*"MsjQsJJ%tUiqM[PY#Pp5YOHXh;J.LA7PnuJd/Sm_A1UY5J?q+qU5ASRRJ&6@*/UTn6, +]sK?T\QQ53#N$o:T"rh*t:GD:Pt_^nB]GQG+?dY2NY4/PM*7]GD1$(0APD`A[:,cT1lK?eHZ*nRe;W^ ++]97nWNAWhXa%3g=a*!6=GR=*B41DTFWdPl(sFOrrBlJ^)6,o1DDmZKPp1/OuOu,N +-3$aCTcaiQXR*kgn%?hbjPV +."HB#O+9mK67UN>3`\RA+L8T)=ttlbQf4b-?D5mhI0jcg>51Ve$ +D"1eTmkKuZ\A%?Kse#5Ra\m(3+Rig#\S>+'l\0ZCYpsEJ9SSJSLq7q`:&RBAVgV:o +n\6NE]/NRNnr#=q9/d<"`F9so\Y;9)86P_GRMH=,h,W1*Hh9?WSHeXUYieSc*>u9R +Idi[#sM4?/WBFs&(_g;kqrcMFK&c\;(+GY`1_J&[K.KDl.'i8<4;=1ss;h\874l?:X4j,t@B'M/YQ:kB[CG)Z?lDu_,8`I8hp-j%]+DM#\KppB%ZeK2?2joB'7iEM!b([.s=Pl"pi&Z1upEg<))R2&J=5hlVB"dp@I)V +t/b(c"5+l0k##^m4Y+)I-5MK+cfh+CHNT2hr=*YjG?DbK5G +HdCem-VNl[<0;OnPA9ma@H*H:[[D*UHIJpV#Vj+i_aT2Dj4ih"A`mq%H-94)=%.XO +$T5L0"#8c`:,=eZ@/jaa!'([[5e&]ppU2&FH6%=>EU5AeLk>E>['tM8T,ln)f[t@O +5_8p&#U.X:7`s?;dncmMoO!'6:cZR(>(Vtrm,*#XT3*"T\^eZZPP,*6\;dFJ\"4=S +;urUo+&8pQ#!k=N`bbJMAS`5JT$Qd7W:9_EUk/c)(r%#dFQW?[RZ)0Lb;lli^*rZ# +gNM^%t*OXXF3Q]eg+IXKGKbTN(&QTM0%4QH3;e%nsTR7I9aTMb1?HL`^H-P,$GLT' +SDJ?^H,r>bJu-dcAM>E0(kXfJqkl4Ed#[ShDF;jCpn=OTXQN;%iYdY.fE4XQDufId2L9K2>b@%*1&,PCpPG5-plFDD( +JE7$\,Od3DJk^\(`,3$7E5CM)>htE8c*PIqa038P=P,WLMpSiYgBNkkJS"UMN]ui$ +:bM*l8P\]=c&4h")\.]^^"r=R,Eir!)FFIlSYb]ZT+')>EB/$_'^**[Ta. +_/fDi:gK8k8>"1p2'a/Jq@A-9Pll.^Du4;e$[CK%qR7J&^M-L@O.#Ka8mbG&^J6h" +FtnBW*fM/4YH>$N]?I"^:/l5aq($#*3q5*7mZgmFI$+4^7D0fdm9BSLbb9+Qs70BG +0a%O??N=73fWmR*A:0'@Vh7R^6Uu>@#jQp`)u2nbIG#G.8fD:A/G7Gg='<^a8WY5O +NR,bWWpk3PQT)=LXZY2kVT@8-^H>G=V?!T8N4d/\i3L8Y!5e&F'fOJ>3b]Rdkm%nC +M(/H6().,8s(9Z8e.\ASE#bdC4GT\,)Ruil-c,^4I6%:)gWq\!mCOV?4%N5BAHAN8aTL>prrA$an +I;#"PYjd_W.J5]Z<5n`8;3j8\6f?(aMA_hJ^%d+!5_rUXhq9?>;TN8_n,!_I/#e3l +m-'D'k.^#:^4%nUpg!6(gtgjK'2:fcL&skGSn^\fQ#KI26RQaF+%JncTR5-l5)0k\ +XF=W&?%/Wa)(Q^&`5Idr+Gf\iLb]\4p_/gKW=jFLW91]k*P0uD'"oIFV8ep5C^@[C16XhRS]Ef#[PBb(E-DnS-j-Et]MU;fWgTuR8 +bR>\2jC@]]D7s2c^16<..-tK4XII#Z)tm4a;8NNm +,g0rF8*/($#U&4NtlhblSkB2[mqCp'.V#T?9'u-kHQ+g!(d3qBuR4'>jk\K'?#gk( +*/klYl!5JYFbaoi3msN[\<^$"IdmFU[<'\SJW3[aNsR3&:T967XJ4H5)?'b,ZsIY[ +8<\H.KZd8j)ak/`rqEK=+L;N_.UD4rK7*["K1c*W6>1(W;*8WNb@-*b)tS[6D/9.b +]AXAr%X31dGh*^1T;!%gq?/`i)aQo0os2\XrE+7+,lLkVn[-"_e;o[Sg]Z@iM?QM4iA\oG![J!XR1$S)X@]&pqt?#`)K5!@j-ag3=COu?k!kK&!IddhX +r%'DdB,oWq[n'!_^)O9R@V"'&.b/]+&Q8bm>LiuBU9eCHqg[:PJs+S]c'Y9o>(UMZ +HWHl3+"Bo"0\XdCR+jP5A&q'm-g]ag+SIK=/UPb6b;PR2#Hr=E-Dj#Ti6o$blE0Y# +-Pfd/@`f27XKJ`aM.iLb^[Gr^+#J-76&bj$P5GDR0,6X^,!?EY@5gB"DTVpcY[gp( +WR=^@k.;@0]>=o7P]B$I$*o"p0kN%TC$@Q*Ai#E>7gRDN7)#O_]@gSQj$N5jICjp? +75pJ"%_sYO+ZgE3rT#JOqG'jM3]HLS=C&,ACnnPn61=#?4ssH!TEtj6In!JSR54C; +X`sqdYV*M>_=cb8/N]k4\;i+pj;KH:XaCnRnnhV\"5Et.@N7SaQ'fh]OPbb0*<;UI +(fO["_RO0iOoWXLPDYS[#5qW8bKTdBru5P*S#o#Ve?qQ,(Md'rXK&*Vr?NkTt/R@oZe,kl0/-U3Ej]"d1]: +P9c+pe1KFr#@m$d.H-DlL2YdZH6q`NTO6FR#o70h6c4>7VgsU(hG:s'ip8"<&X1?a +'Zu>0UDR[&l9Si.=q;!LAVL$hDE.8)D].<+(6W(pMg'q>-ca.BTG^oYZOdI!9Ci=sFRnXs3.[l/&iH"b4?J]P2;WI_f>(Np9S8+ +h=0W8$AFR+-Fft4rVipE8/Hf8Fn'f+'^cid_BmR@?"JdQhEu>Knchj;=C>6)"Gh<# +PF`#@/ZG+%OR\F!!NAtQ/\tV0o"`,Q8-L'kh)&0MH'Z]i)^=(K:I*FYHYYmimkcVe +MS&F[GPn9)EVC2=0V);@ZZaP!'qc`";-UP^2j!UUdlfQrCJPP^5ZUNDdc(caB:kI) +s8<^N9"WJHOkmZ0/ef>VHJ.mmcrr_=dj,h,:tddUW1G"N^`iMmp+r7N:G-mE;oq-= +RQ[NHpI]3JmEb'ZLIppCA* +_M$X9FKJ!%">V+J&+a!?gVe,-`=p644K*2LVo9M).*Ip[BMDL\shLSJi2cb^XuWH7 ++hTOn@b+gm>1;an=U:i\;1l]K]';o+&DKcpg7D'HYI9q!!V6.28r"9qX_L`C==\@c +PQh=Cr<1S'9?>oJL2ro]5Uo?@jeGpnlNbX5O?/S]2mM5d-Zli1.BWS+]Y++_'3ce +?;nI_k[1(Z2jOZ^66qW'(&S/F\n_^MqI&4rr>([q![q,nFQtb$A\LBlJgM.1=\!od +d`E:)+f9GDP.u>O'uC$kSC3UP.,5n(8: +b4O`?9PcZ83g_d1,]Zp(uq1kI?=Z;gtc5:ElkS[3&3/4YCW=Ep90U.i2geh4TjWdN +rh/(hEA6hJb,fjPIJnM6(ean2#0@a)ks96AHgCbK&$>?r1\RRNF;^ar[c#0!lO&hI +PM@"YNikpe[q1EW2(d#:f^Qe2I;4)iL09&fhD2[B`]G&m$nXgoun +Xp=EL/IoEp0d[gjoU"XX8";X.ben\p/F,onU8[n;-&&Z!mfe\<&W&\4H_a<'j&2^E +aF4=59/at(eXPoaoJd%!5'dZtJkW%tu5 +TKK\K#?SdLP:UMl:;X4TOa@?=5DOR`k3t%Ca=hWe>?'#;mkoGT4&LDi',?b:8CC!FG +#ZN,4pqoBGg9:Z-j.mS?P2!phq^JkCH0iWA$K"66L0cE:3m#4m`u,j=71H(P@"'V& +r(19,]$a6Ri@II*4c2@hiqUg,9O*DiNn$9l0W]S"r_rdq.5EE8MHqK(1dV(&6+o:2 +GZ1oVkmL0nhbOIBS,q\$;J/e")@6`A(TfmK?"1H-bRqPLJ*!^i +LB;D6ZZeine?a>Ftb2A\MKAU=ep0]=X<#Tl/Y&84r&=/_]%q)Xfc#U:=mfii9UaekWh;EUURncimb]iGocb%M!ZhR&EYl8lPjj&S:/l=E +R^HHcQ<) +$>\?..FS.\)^cXlQ+DEDr$\G+<^3)V![)j#oW3mT+cYe>'s<>?&#b]>XdTTeK-1.F +beEl2_n("#7DD"Q5E'2bN_iD5T_LZBp[[>f;-"02)2(TU>^9?QMPb\$_;epd`p>X+ +9kic>8Z:"U3mg*/A*L,pr<ElfJ03i7'5Id3,O_fNe!n-=oRjS27)HS%_3edHijD>)4jJ +Q]p38C@6kpMr0s;GSRb#=J#6$bOa<_cDZ=_V0_+O_H,I6Q(hH@QOoA%"Zm"a54H=r +rCt8\)9K*?6/Guk[Z%/q%5`inEeS$R#3dfJ+:o'ur(>L<`IA0$4 +_$k.obC-5,/nCA\T2VN?9PrMQGc4IVb5&kV6)$D7D93*?j@puiFT($^)Sj +i,?j;EK5t\'/.G@n=bDMmSWX?pNcpDJR$O!i-kApj!(5(7BTFC>L(dC.Whh0/8<\+ +euREoTsC +k8#TobGp'hb.ak$QUm1+"gs'[lbEDrZ#G\lcj_KTAo`pfQ" +CiDmGnlZ!`D$S9nE]<;$>ZT[Rgp\6>QhR#@i$f5"M03'1#Rf['EMsd7CAKBO2Kd@9 +t@UZXqSU>PZCiA8Wn)(mJ#Sc!KJB._#l:14\*mA&%%8:oO2_Kd?@:gj>0FWNo>]mE +WB>aOE9Uj>589=J$b5kn7pEDfB:W:.t8(D&nl_nikH&Nr1QgT*>-OCY_VS8H$U&Nilm$1@5*KG!oqKLWs5.+q1.-1]d<"MS))1i +&1I>p%7J"r=AiWAD,6JuWOr-F*%I`*_cu9fIZHp,BRrV +5oB"$02ssR$?NuEYgAIAX&b$@l<@K^DLlQ0R2[Bem9-)O'%6feHP?agGIUC)"5efBcQ^>Ys?'Pj,KB/"u$6O4H:flf4)7F> +O_E1r)<\LRYdA(TeaDiWoN3aLdi_;:;7]1pd)O?$n +&>ie\pfpd_=>Q3eEq)3nF"&d_/4N2OunMc#4O<)7hHsLEW[>r_j^54K:3jZnM=OZd +'O1O#4GKT1m+`Ifd"G9Q?Qn&,(Z9EXp"fQJae$c$8#]HO5!;$iI1-I+%[+%=mWLbC +UB8\^]P]hJu)sUL*pG6*"ZVG-f?43iE<84hA"GUoWS!/TO!.HG3;TUkSiXe!n=ku@GSf%(p= +*rrMO0eaFeFs_]U>U$^iKkX@4_olC]pm"$GiP%qQ0r/i\`Y6.FS<[t)XX[i*N_+/A +<&Pg7g&lS+&:sK.t5a/_ES$84$XY4Gff9-;SKjE!.lb()ufp5.pp1M$HC'n?"-13Z +&Dl,e(!tr,Q>th`Xdh@3SG(K2ZqhW.7QNE/*bIU,FmUlr"X#+ZeOnEQFa>AVmk@>k +(#Zh^-27"4@fDO"Xs.7*8XmW['E=DaIqnAA"J+"+,t],WTig/bKH:8G%g&:a?[Qh4 +Y,R=MRPqHrX)`r\.2YYhF-2t?[ql\^9uJg9_u+-/>sSA*Ni-S@.V>;H+30nDS"`l^ +Z0QfcOkM`4/8F4@=^F&S/SI*(2.Z3=8#Y`0YFh+7.pda,6S!<,SBk%?afEZjO1eUm +91>O&N/o%)8PLuN#$?/1&9D0m-B-P+0u3'mGh-UCfAIZni7sl#G\E&:Gb$6H&)YAr +Sd3P2ZCnV#I`*2C/9_Ri;&RA'M7&!kP'u2IU8<37[#pkRrD=XS.H2Vf8+n#2Q^cbZ +f8L0An&5GP#";g7lW]'K`HJ.OXPmn\!P?s:R+Hf4BK[Tn5\dFjRB@'QXm8*Ga@63% +J''T)o!"9l+J#(bE1h=ENnJp(Rk0[rX+?h*hf6O+7-.k*#e4DJ"LEmFW:nhcj,2Q_ +8kC7P?e'VIMp7AE>Ot9Gc*1:]J,Mk<*UL%1FTHMIh-<,[CrBILA-jHUd3^sVS?:6! +8tTW$pV!uLT`>u&PPA=e!;_u&%$VAjl6TE)n-o_YK;h97'5DuFh3BSVc8XU_"gqV= +bj_25TXK`4PjNm;f<1u)7_!EdWhjCVs-Le9Q7S*l%1E`D\'8$T(BP+$CD/<8Fq8Md +^-S\D)?H_=.4?SaS?((]T+B@qQ'F\3dG4uUC6kKl7[d=.6iOd!+kU/c/HoK:0YmD/ +]ZQ*\HlPOL=uXLgt`ZSP&'Xjjp$l<&egG$it_Frrr<2,rr@XoHp.8hq`$NjrZc.RN +TRPgU=SbSHR*?gm8BQ#O>tAFXT3b_q`=`+eiIsR!/EH3!"irW=kr;jc)4s3[c1F4N +Kl8Jl2B?GhP8rAam^+d0+?Xhq)VQYVb78XZ&%A-5MQ=c$T8L]B*!P[[J>laI`FK^O +2'VI5@]G%2;UtJ05/ +>\M!nm&E:DU)#s]DhjBitMONrrBl$]QrJJ?eB#N6lmQ'1mjU/!'n'$[6o!SS!dDuO +eRJo,k6cMJ)HR+[4d*F+F_8$%CE']=5h=cJ&@^;H=!GF*E"&#AMdVQLQ.Eg#)9m?\CTQrKt;LC460P%DIOrG^DffeMLQ(2eEir"NZTFS)mIGj[eQXCVg#Ni +0]#`8^E'>[(#CenA!u?`=EC3C-)*G2!;Nb`6o/Vb`"Zp0Ri;;^F7)ojRf)Ue_g-t4 +>[.5ENWu[Ltc\em0JMda)ucXqd9?LIKK:=qec6inUA74D(ZhV80;?l0#[$`ZWU=.% +."H(?/VQ*d[aIUT&g+ANj*AZrE_jr?p$qeD6i.br@o`O/J +s^p"pjSMrrY=%EQ@$k`8tDN9iS]__Wb!Ps$pq3!oj=je5>8]rqb2B$plPQ+"8MW,g +9_>KUKVE:\SaR"r8t,/:AE;sDhicL]=@B;];@pc6I:kO +kZ@GdB\jeWsj1TB/BYNJ$R\ne*r"JXB46opnd#s%VQcJeg7qd:V;e+=t'f?6Q_Q\D +JI`pa.BTFn#;%U$u*bj'UMi_ +:iOYr*Q/`c_mkRnBO8Q=%#!Kko8b=7nc\MN#n5'&aK74NFe*V6QL +)ptXkuj$'3dNb5S<@m`of16:RK29=rr@Z%HjB1>#pf$^XeoAS.@(jY[k/I?P(^g\A +)!#,lfTqRbPhiWn6blaZgl;$m4%(T5Jm_1f$:%cUsR3-Gmo3q+E"4D12T#6f=cQ'rr<0:pa9S208P+b%\`DK7 +Bk<\Hm.NsY750oC)1MFg/pbQ;$h2j=6mJgqToK2gtUf&AbJ?Dl:gcWnT"obflslH +NE$nYGjAY!![-n>cV)m\lrtY"+Oq#]*!W3HXdf^5Pu;s!)BuLrr?[r!ab,?)Rc3=H +sL<#.U(QH)[I_EXp>m"Qog,l#6=Ch%.A_=n6_g@ho-MX3`_BBppT>E@^?NT*7@6j\ +,`U_jZ,F3*oQdr_iNML93l3(anX,'VmEe.UZ6btp+/,Qo&c'G:jifm"Es$:e(;HtJ +Gor(>>&<6pfkYeWsc>sXJr):N>.P01:!ldQZ1l?raW]X%6eVVp>u)0DgY]U4FaHj1 +54GZE)6N7^^<=J^eb7p1hPY'^Q(hS]5 +e8jW'K]JR;1a(<#AW99rHd(]r("/Sm.=Fnc"!Z*gZ7WI.o%h$[BX%Zpf>PBTdB,8?KGb;kl`1"NRQ%dj&h_[%n`2Z]p@Yrj_o0TZu]^JZQd+@>iY*a?)lP!thC_K]D+ +3Co^1To:jH2r6N[=.p2L@75ninJefI7%AiXRoY!p?Ls?8=DW=ORJJ\-'g[[aG9mi$ +o]M`5,-JTCL(^Nn6Ni]4RG(%'3m!94@;,W6dCa[d1I'F(cYC=3sd%l\D97,+,]tLi +LO$0a1%MO]6eTM[0pI3 +e7ub7`6O#o)Z!ri%Ru/P!B14cPuZAI1l9Le(u%GB'+/Hkma+dSN>S+PP0?TUbg#`D +NWP6EX$Mh%@3FANt"ThUt],0:M6A'gF?C.EBTlN\T&3?;@ihNP9?c\DGP4 +I:"::'JbL*aQT3UO7'736jF8aQlfBr[T@RoeGFunNOL-D`kP@(QkeVf-nQ_5$DJ@R +)&p.*#S_YX&c-W5AT#q^(+=kC?*5NG(a8QjOG?i4q5t8V"YY2b4&P(Wg$X5PAo>$O +\2L!,0+`UhD4J?uI*.f!hSO +oYROoO!/Q++l[;Vg*qjHVDmQ[,H8;-nd0Aagmt5A8,3,:q#Kh7>2sqo!sje;8KG!M +rHR^O6jo]lIWU@M/P#MEASg&`lW.%&\QW8)4Dt3@rUIk#H`#R"+P*Y*Ve+'LICoMU +Kd#T)8')4d,VV;l)npJTBPlM5klF7R$-NlI%U]1i#;5@g61dBrXnW0M4f6$IgfKTU +IN`j8T[LFq**PO#4RCqb,';C8Wi^c<0@O53M6kc`HSMar$o7e\$C?PDD]$jB^G/P: +r!trg9)$D*-r:kY;,MrC"d^tS#9IXEN&YTm%fO,p;P&%@j#$Egu]p>G(\.LSFs%k: +[n(iIJZ2-!+n[n>ON0]kZVBK&SU0LjAmh(4#'%2\&q?\a]t?;1Sjdr`+sF.AG4T_) +H5?_Ct:Dcjh9BKW,D&\>q4RR+1&mD +ED&ULNbf+fAP13m(>2p092[&:O?_+j@letDg94&+:EbE=XL]J%!WjiO3=l>f +5*QVWn^^A\^ocNOPANb&maLUJBAanF!1:_8/]4a#9iTsa)ZE"I7dd?h"[43nI3iJ> +MUmYYqo(0X#/53\720j.0p8>8LkE._6)SqHn3f2:S??BgM+^G?6@Uu,b>q]*c()d6 +P'lmbB"g^],!]!Cuo`1D''/\F2]1:pn7e'(YU>-G+CT2Q0^%RXsT5sY8b?2%!cs76 +N!L_F'b9n58?iMMbS?]DJcbsF_ZBMp:45X7uf?*.?JH$'am)9U_hm5^uPU7iR.,n* +n]<*?OJGu`#=?JEU!o_;G.EW/DqdFZu0!-rpDRm\I@0[?D6SOU7AX`;4oV#2\@aWU]iNrqD+Y2A@D6fTZnt=4=ee?kXE_[;L +KmX\Y'@P>]K+uBC<'Z/Hn413"'B5Sm4L"R5-aCW2j:kr8=Y:<@Q#'"E1jJUi3gV_H +qh?'-e%H?D/W.!8gCl^>A+ah/a_>Dm;AjfNZ_JZaT4$Iq6l5^%u@t3DS@=Of\>_YE +p=Ea0O&8b7S%*9TFpRK(q/AD=NKhSR>A[eV`dP0pnu);,kuGaY*H)5?UN;V.U6aJ< +`2:07f&97,:_,'7a.%#?b$[5,\5&:_e9&H2u=d:S=L[4K)WXH4\k]h[DQcB0a@4`a +>2(j5OSM:[>kT4#e<#gi +jbTgpsCqJK>KbceELr.P:*Oo.N?A)aU,`YV\T(qfC8hhZE\Om=qAH +CVFlG^^rN5#@C[_2_;IEHVJJJ0uQVlag`I(o#Gj[db&X4)9la8-EqXd+KW`73b'"i +`@9i`LMM#af?I#nYQ5gi]DQ?gN!*f>IBE41gc8 +Sfc^nQh]U0M$jZ8=GqP(a$Ki!mZ(#8Ngif9),G/LS\4g50d@_9f_Y/XR,%?*TYCpk +P;3/GIWj)3SM!ZD:LTC'[lSMeN_uH*Zr&bC8,H%hDDL/@'-q[WL^XmjGfgQH1+!?O +<-Dt)_]na;]Zl-RXAgaZ*\A]"GYJ`.0m"7*e!W\r,!Eli@C*%p,FSFobc"Yj*W-B6 +/jHNOJlc`+>7U!g:Ik0>c\t\GjXRD5qAMBLb-:ijEo5UrWuiKHgp1W3Q=2IU9$/'N +Jl6!K`;%(COBC84op,i*Y@jKNdfEf2O@pQ75N]6e&\+k;6"Jm1fuCe>)U[[(F`RTt#r'E+Y'+@ja8uZ&J9]3V\s7IFIG1o-Vuu/mgCs:dn7:D]] +Kr4c='C5RfC^Xs/0Koq0B=qI(2^j("7tb_+7-c4P=P-R^qU4Y%72!qh($r+AqG3s? +)9jY(#=DX^.00p7JgbG40jY"NkKJ%4dFRZU.KF:@6m#I=9ehgS$E(B?M:UF7*]Deh+!t2_[gA-t'l%7n*nJFTbd4+4S")5>0e5n\[NtmL?"8s64/jJ.s) +>l"len%_;q_8BIS^blVO.B76N1ZpHE`tr%)d=f5XJJ!gjuiKdf>c_;;!i>C(EP3u[ +Z#$An(Qume?YFjZhL(8WR'RV +7mO-1hY'KVHGgL]Y7jZaS%*53WSr=rX5XoA)^h*q`K6dd/GYB)T+ni>JqsIqLu??g +'$">6R7J=8,BGm6-7Dl\%R/h-'VR\_gO^_O=[@^0083'R:\5T//I_ +IFNkTofpkGCE-UY$8p(2i6m&3/tSfVl(M8[[jEcp(WtT3D7Ks'FjVbID78MnPnLb; +T`*2ij(Q9``K$%$37W<_4M0/aMh+%4ZENYqpk]H"?,eiQZrC>A.lcEIjW]o4alt!5WtF3-]rHK>%&RD;fu":Z?`G/TcqBQU])+; +D?ru3A6DjE#IgZX4+H5pchqD/O2Yr,]Q\Bku=_Se=%;Ts%rL83\VD]9.P*nej_)N_HaiOc)4P.Bpp +c'etM#El,"/gBJiLYtsZ$\R/bjrY_H_aqA1h#1OYM'b7c:^)C4u,)WTF^8t8@3Ti= +`0=f5Bn67o4RGTV@+L"lem04ASfJ,WQcAeZ,DI%*.r#&o='VTM3n2ON;]\enBtHCB +HIRuY1/VPj06*@GpIk&^(*+-T9p+hS<#i.L&V,Vo_kUM-?t_Cf!W])!(!UZH?HdIk +HsaFObahe>:A)S+)j2",6OiI!,$1o"6J#9Oi!"7uu5O5?r]rSl_CD5NCLS/HSXfiXk^S6((?@Is;IJ/VoIUlSl" +RT#NIA?itf/joN!;mRup8lmDBV$E?j!MNRWC%Rmjk3jB7:c9VDm_"0M9jH<%/Wj+m +Vi";-MlJ"nHZqCNBi=bZi!01JQRu$I)pqZHB7B!;b:r&i^k_qg8fgZ7P]%G_gj95P +5(gP!!R3;V1(2?C-6@n*>+mk!=W$Fq?1c`(X*`-2jdTMZ@fc,1'\ZUQIGg@'Dl"tp +Q=66r-Im[b8Q'BEFYnB$Re$haDJqB*0cd`BTQ9:VSC0p@Pm#PBHSB(4&-r`"a8D87:`6-_"CJ+tR2i;UmM;r5NJJ +)MZXg0==_bSM?!X73.HrX+/P:MKeK[F-Wj3r&L7en%Ngnlj-`.L61&ckgR+qo%ZsX +#eT)'cZk8nTC@&^+]8pX_ZKYaPUR[FSe/3I%lm,$F(`_!@)dUE_5GU!W=g!+(!F_j +NoJsDla`N93BSLCT,1QG,Qq#G"EPRVJH!WrOd!=ci+7ln^V4P)16Y-!iqNGogZUK6 +1H;8i6N@YidV>OHGu5.+Eio"@,F[Ah7Jfma"5Nt=Rj]t^s6ptlJ\u"0'V +;LW/g7_`21&a%3`snF.[2AoDO/HFt4b[u!(iie)nq7ilpH6e+?N\/];b`9brUT5.7 +j''-0qLH&=6V`r+S]=(btD40"$eM$VU%g.T)MO^4)Mc6O6+$NY_f52%=Pl[#7Q8.d)6s(YqoAj%) +o:Hr:"Oc@TP?YND[Sq0ph*%GTXK_^qhMJ=?^3spg$Ycr8>G?Q$in*6:/,\V,P[62%ihAl'Ia!%`iW"DZ`a35FOVU%'d:nUH+#\+ju[077!'l +B/6IFfJ4-UbJi-k_fPb66mErDA;J9H(Y[Ppg:!di,9HH3l\MbbC:74pba=1B7-'b< +XumZ086oHo,*8#D>A7oWP0lSiC<@_Yn)WXoT(*hR<4poK')&a'\V+ca.H.#R0bXcd +`R-/#i<^!36:ZSin8\giT.Gqm-ePlZ]s3i2JDYEh8HHS.Q`Wi)HJ0+D?3X-J2:['a +8R36`KP(\3]3\h\o!3sM?UO4B7pDN=$fZt%WVY_fF3';$9G>LCZ-jgn9@.7:Xo;NW +p-cklYa'-ZJ3:kEiU'0X_u:a3+MQ[Ws(%4k-odHlBY`b[a=Z+2Uat,n7I;!V=^3oT +#d@*pS\rA=@b.t$+R3q&XNph3#M6]#?&tI7I_Zd1ZON.rr?K]GahQ\CKF.NPUr#/Z +D5Mm@gEB=#uS>EW(G9MVnLRH4<5`=T>UVV?QStu6aA@n&@5SH54lJ!0jd!Hi#WXj; +m^--ai+7&IiZ6[bG]eu\+'ZZiEJGu:"&*L4.End$sH'EJOQhBR/EgY\:Qn0(XiSdV +J-sW[IEJ%`d3B5Wncm7:t0*K36W!b4iq;YkTc:+W)@n>P&bm>IgS@Wl[KR)qu6X0G +\=qJc.89eW#@f^*Nc*EJhUnY+V1EdM2qdAL+m!<#cIC8bJ_e7I_^iDf;?7g33@!2V#2#)%&fOXW<@GH +mk2md=>Kp@SLl$HIHQ:;r5?7@=KASZQ*""\![]BR8K5Q5^/A:0s_EV"$BXhA)Y*S` +a<@".F^1[$7hZ2`]Lk+=P-ekKU,a]_/7.-D`KrQmp$pCVWbHI)RLRsr)>da$lYWjg +<4-,[Cr-l:0C[%FX;e5N.8K\:'CVud8dUS!$Bs2h\fnO0?)D*(=BM_lSNir)(N-uJ +L/.ua]9c@JL6Ya>kjB>7@'"nr?G"boVN2PrNRYO/j^bjeKhRnP#rbq+i2:`(_URB; +/.lN4tG!WRe$#BOYuO9-89mHUBCY9]C;q^MtfqDq];SA0/Rg[[ +MGdRBq6X8EA0bP0g# +@VSEmKY5>N"3LMNH>d4!"&>=r$7a(bh#nbXgXd=@:UuBm3<$=bcF*56#F>%D6/CnJ'RR"MN;'lW2FjXUE$iM0F)TVeIj\kp]FD5 +Qmf9;6^];;g*H37cpVFG]M-N4sSQ>%h7i>QEl-SGH&Om&DNKQr?C/X)Zr*,Ep"8a( +u]UUm9#(Er&4^N[C*Q.93A)GV#$m?$!ol"86u$slkG^MgNgaq>EA,X%(q;IXr.7=+jS$t>b&r.$c/k'=H>c<"?LqZg08g=omE7<, +p#i2PdHQ(lD&;E#5@-t/ToW*5DhR/AZ&(A(q`Xgi`G_fQVGm4`0d+<+NY-%ttMJ`QhHu@m%J2FnHi]UB&( +:0n&T+@U!Rn:#T/upcpcn"M"NMBNZ+.-Lp&&:8oFNhH]ebfI;j2%%m?c:bR_jZ%_- ++Se)D')'Te:ltc,a-:Cg!CSr5^"c/o.VGO$*MIiqKA@e40&bV-cVT,j;PD-E-+/gC +#V]`A?;+G'H*;?P/(mn5ldF-??bBn&BCMZ>(3NTpchfn[D/+d^U05.#+$L#q4:dQj +Wtrm)Cgnh=jJ:B,k/!nn);gAL"D9*XY[rMe\kNI?6@U;ZZ15DO]@boN*1aDG=hj8% +]V4.X*V_QEMLE[TNp:;[LtrKVeV@Z(/ViI9$tp!,A?'u"3A>UA'"j%4ClsMSgNqS_ +Z'TUhtU:3FC%/\4DPpoP&'Na6Fn@0)+Am699%hC\T6+%o*3j+J>mDC/Q__"iU(-k5 +8C>K0B+s.DsfHJJ;IZ"!m;sU?.^l>ZHR:)=]MouNI],LO:9rFXo(Jdl/16=2pgdbA +=e4?MT7/K=s:=W)1g)P@%]+^il<9Wc#u>@;q,#9p`\OJd3kCH:#d0!elc3.6g?5\* +d`=jHWN5fY7J\CC(CX,r)Bo3\(C:0htApi/d70C%(k1tm:D+:"9M=Zn11C9!.qF[g +WTTs"ASB8r$-C3gY56XI5As!4KNQ*I)IuiB+LQd,NekA\#T+: +-)oeo1/f\$BEcU7C6)DP$jHX-qP,(r&DOV"qROr&&K>E9UVf;!^T]?O\?rZm-BpFLHj"R/lp1Amp;L4dr +YesIlWDYFd)FeU8?);oRT<;%-ZYA?C&n7%D#XKMmtH#:KV&m)$b&FU2%[@C"PMi;2 +uMG4(k-VShL'//7FWB-TAq"$?!>2cY4^>b@LScafH;R:*.ZhLIP%Htm-J>VT"D7l6O]S#L\@=Vd$$dNim.4>a^t2JF5$8.d3+<%`$H@WnaA+L2` +dfob9=@:-,SE*4#gH-ZPK%S)!">A)^m[?AaP>3\mIMA(oV]-g(q*pEMa[?jD0b! +8IB]Kig1Z%bT#_,KXgh2WXPS0\jEMu(":g'S%Ll]j=F@=U84&oX+,'\g+ +HdnYXs5siL'R567#%75J0.i:,E[MWqE_ +?4b$bc^mBH[:!>eQsiWFmTD4D]d\8>\[HVW&A@=)NNd1LR/)W>sQHTX&s_A#@''r&\5&T)oT"6qj>0B$ctGY1 +%:uU27Ak1NF'oK$W3p"EGZT(OJGqgjhg5_%T(-P)4M9-!!3G/GJq)Klu:hFrZ,>Xl +M?j,0B7[eOA?:rT7u]DM.DbuAW:k&i,,M=29g0.g30+J+MJ05Y;M5&$@#R;; +kM"jAG\!_+dDdp*59ja9'hkJmO]*NE\J8Z4^1ARTbkaqAWFIaNA"JEZ-r&@auFjn] +&I#>'UF;k)tC;Z^_$hPAE+o0n;#bI(u>*U^%%u-iZ"8g1nin\(p9\rMV`Y\6'DXnr^1"?DaC"RhH,QW%qPO7\/4J +A*q4*tNa'nMc7TV74hO!4T?X)#/h'DDb-H9h7Q'L93jJ;C)hPD6TQsA3n-7a^ks-! +5JK2Q5]FoL:f7@h.h97oWukjWCq(7Zo50WI-B@>D_BiR@HGd&`\ecV;ice0*Y$\;D +7gO$R_?JRSX5(6m1SeDq^-e=Fkt?0i3i*?mu+QO[%I!3:&U]:g +?/_%)tP&2#%5aE#i2mkgPYN:YDXJ6A(2do:;<^IiZF&HfS<7[`a\#_Br[2llYOcRE +k%9$NX6sm#Ofu*/"Y:6LQ,>'0TnQR/`i2nk>T/@Ga!/C96&:gelD/`u$iq +%]Z4!%*4[fg,Bum)aHeH`.mS92G'%]N:PUr8u)6b[lq!RRk7LI)%HZA.IFq57H5kn +OLM=%b#O7&8F-?8IJWj$Jn5FAs-dq1GJo6Bc]^]@>IuJJJFA[j%U3*rrB?(g>2/7H +A7DVKZ*Js]O-MGI!dnSFOm +hg;5`f8=_<)d2Yq\G1\upNH,nR`n05hAF_`NcT&IB +b1L:ipK<8,GsO"N!f]EKE@C(p>6;N*=DN;`7?TQ@lrn29XHH),3s!Ik`,k/:M08MmOX2HFT>eZPG/![d*l9Z9i;78O47SiH1u\ +h9]H-WnqF\'q_MXVqDO7PWXTNTJ*p4LsW?=@<:0!](HQ:eIl\41\M[pS_kopTXFel +7F4*4YG=3_pdh]0-u3,/5P[Z*_@fFGa6q?fWeEn&-l<2ngg;En=Y1FCi(ldko7K";3Veo,+6&&StL-@ZVCE0:Trck^ElK^[VPellrsJT(J'jfe3>a3H/%(<*9J,!`;3mJ6'>"*=DOG^[SW1! +$1`*],qmbHJu"h2eSqSlrqf@n\ZnP\X.?7#[B*&hR5.h4fcR$4h&qEr!21FJ61GOL +S=K8H]:=[cV=De\CRI$&Du8X#Bo:@Q:,T8\D+QN-]pY8?"NQ(_bg,#f1jgLZk"%!2 +Tk1SlP&&E@#%LgJF#2G,]29[b%#[k#;RCXNgUoCKs1D5j*q;IH68('P/kAL*i!L/R +q3&[rUR\O,]R'CDmVd.`\Ee6PC\Z2U[bJtLVFhC+5b93)XG6YT7_9G?>--?3lrV1q +amQ%&NlCl2b9mI?%:@T[tDq3dC^3elFn4a,ggFrjO82T]oQc.bk]HcC"f6%rM;2\-!fq_/tC=.dLQ'eu6YF-8;f +jC!E!Dlclpadf2M*\J>J.8FWq0;.146-4?A +&@6S`*.8`(!X[;5l.jrag0`4)dg^!5sm^HF@6es-Id2/@R=.VR78(e# +.aW3Xj6BUbjmtGG9sr5W&3a$R,h\-]AGfJmQ%CR]bR=3c%YEeDmdX#-_NK`%se56_ +`r%CZPH_TTHTGo!Ot?KcT&'&U"*u=`QW..DkH,hb'iR<_f#\tWsQ!&!TfSU'oB^6h +sp?AgK)'n9g+XMhY`c%jGoH*_*t;]o4d.?8VUdu_h&DlTFNa?`;V%2/W!l?`I^a.7,WN:6 +4Y+%NH/ZsJ7=uVSkmUl5]^4sW<9khfeXZ3OmC=\=XR!#g3H8L`.u06*"V3t)99Z3g +4TbX[kiOS&Y![QNbZ9%N:1;h^pof@I&]SCH@&^n23a(%]]t4#!_tf7I[-+rBT$r17 +jA]tG7h6&JpG<8@6]ZfY10Su`Q/HKKEq-,>Q,6q"-@'(_@=G>e.M.QRRWILOZO<-F7Q2o!Ign_PrZBafe1<[VO&N%i5Jo8Y'?=mm#"?E +&nD%UVe[sWf[^LkKAFmDM.Sb8Vr+HGD2c4d3[s8.-IP%%h;b:6HM:L1#+p;WoDS_, +81TU\5S9mHPgBki$9>c3`I#amUM(tgbI!9gX:U.A'A!kN.K;]e['$.=O'J\EcY4I/ +P8\Snc@f^orER!kbF4Vr59Xrr?q^1V\kPJr]M_ZXI%0_li7paN_OD*>(?6\(QhU62 +0"?gfFM_rX4nnC?@AnE4hl7QnT3931Zsa$-I,1HPa4$0n4l`*RS`B&B6a(=lri5;03IX\0HB$ +nhbAqp[&;B.&kX@!:\)B,PBkVKZYXSBq=El#BsEGi4Ohm\FgB[VELQK%&SmB7c1#P +1t6j2R1'qNF04)[\,O!S7f,$W6NT.T%ZmuD4F/0SqDdDl'_UMiL>'CFGQ[_hH!jUk +/-Aj+]0Z@L]/mkao:a(C[>>R^m<>lfDZE8kOCa:nM=!sm?Wmgl!u-2r`>8hR%1,EA,;9EF8KLiCBcODo+9epC)c!bZEc +5ds23q&A)!2^G2)G#gs5;c\1[(hTpP3-%7`*OW/FA]?=?BQjB>6ei](!+H$nZJ3Qp +s\9\&K*1mT,m]qdUD>5ahu?[/k?"(!&!Q/*'PS30^mHh+aAttm;<[iX*]7natFs3kir7)g +PuT\'jjB(XlG=&6It#>iZ>Q)7^RWh*,sa0P?Q$f$HUm/)!<-87VDk#RH&c"BNeF'[ +tHpSfVXG"C?qhRl7\f%D["D1Vud[A\(VbZQiLS-.k(dpNp6'u_&`sYa`F:o?97Rf] +R3G!3up1jJm#/XdR_t&8N!'M?s=Kd=cD"WL]QQD^#FO#58Jo'RYSgG'9,LTS'?P1F +V#\+jG$]r!E"ZogjjP,[WUK]CWF"%M.rp3j1c:4aN1?)DOQbGcpPDke\O@#E]Sj$J +3;9?i1@[fgDqmW&o+$NFbtMAp#tdPJ6BJp/@(HGJSpo#@qeIQ@659>1u!N54nXKuClT6KbT76Jh,VF&X()X47L,h@KC7gR":ZA@d>Al-Zj*qt=Wn^f1Z +o7FKK'ie5HXm)'^a.GP`e5pVe&!V>gOMIN@b*RVgTMal2V/>`oAjL"I,i3 +Els&3ourI1F!Kl3Njh#YDrHmnB4/hFFP*OJ`U#UH`+F*m]IVh])qmmVZ^O\:MaMX' +\uF&pNjN*-Fp)naAQ&9S':C,SKTl##CnBE8:Km1nY +5i:jLP?q@:Kt"S(rY`_Ogop-P##>QHP&LaTf/&'8&h"Cle_csY)L<@Vs'S/c%=$^i +,4C)EqrIfgp-HBI[Iqu[.PMq*t\?Ui\CH5$+*T9>[#oUWQpT?[aobRRiYEF275pWS +192.8l9e_j#O:QdX3WWqQr/H0;hNlkZeD?-*lS5NOTV>K@+BlSd"V5Lb3&@nLq`g0'cKc+L6?m2&Tnm4(Si%VBUuA(N<6`, +joh8YFROHM]*qCAN]MIYm>G`@q-=DU#[U+4NQ#\Z?foMh4.B`.- +(p.Sj$jW\FUABJq]&E;g"L>ETP,B(eMWmR`]b6ReoY\@`euZ>d`0DF%3n.;=c,.IX +1D7qc2_/NXr#%V!#Kaj$K13UG`P,$eZWO6f,AePT6MT(-Y)t.XMDF`,VR63INL(Rh +[j8*n^1QVfjaIC%U(qBj^o.65llQ"mh+snBRliD+`s4IO4b$D/E_e(;ajL,TBAe0X/Tc\F>+k%R"UV5Ab(Nr +&t&[#4m3FZ*Cg!mbN#&+@U8WQ&]rhh[TZ4i.)>Ep%!NcV4hZ<6`&Q`qGi2T>Kc3_9 +-Ls0^2hJW+]&AkF,Kgtp<^7.&,8)ai+,k$/Cm\aH;[#n8OeM2i4V&@_eDB0, +lHHoo3T,hM*cS/n>T@&\"6IOXP'6QDe71XE,MKTfY +V0o&0'6bba4On2X`O?VMetZ:cs_,2\8Rl,TFP]sV-DMq=K;Oc61=2ILZ-KfqbK[[E +^Inf@cZUY"`OW(abYhEW'7bpK>VX$:Z(8T'Ba)i4E3V,4kmR4lF@Y6bBAd*\$J00,ag3t'mh&j2hC5)>m9 +-d;N5\CME5j;I]d_OFG.CTjSj#B0P/]>i1?*Y$^@L]CpN"J;hoQ6B:U_a"E]T8A]P +Q=8)3?Xl\qnD`-7_Dr]fV`$1jRbMtP[ +4_p6iMh2^)mVR=4I7PnakC7d:X_2?4b7Griebd$Q_E=Ye_g.YN!DjeacU&;OC4T1f +E:4ZJ3J=Ki2,"Mgj'sk*S&&59-KFPGWe*VJk$I1Z!;e`!6m\Ch'"=0-` +fW0C/^?Bu[73gU1OYnt1Oq3=e"i7o(ra%:*@eY(S/=jWJhb+@ZiM>\>?d^SRt_cr? +Csi;/fsV,^-]-e]-#%KUQ*n1b5a8[F%DCKM`Kk&3R8e4!(j"m'/>>FF/UY7TC_kYZ +n$0<+?l/;V.=9Q8L06d#n]>3r[,$R5sBHMd_0Z!-j%Tt"c+1DqUCEB(.TM*EK'Ebg +UZ1n+K<@DIIg@Tj>.5&B,Nk2!G%+1JVZ>c*-S=Kd:%j!(b\%^k-Lnp&ihlAD)!pYe +!ul&[uYrY,W?38mL2"F39WRf!9iJF8F[H$cTLAJ/sST%[-HAnOIKAt"9*W)K[7+To +Va"I&jg&n1_W@Y!&F_s1".5H--_O^L+-aI#Xu[#K=3][:'N`)*.%7M#\&c`L1?8#= +T6O22^*$)CmG]>GI(q^\J\?&^=Q_,6eXHWSc:rh3h;XKA>DbJb`B&WP0opOL?2:AH^IJ>Y[%k-iufM +`#o\B9<7IN)iI8bXr:d,67:[`@Pl$B_+r>URcR$.sVodY.XK1-?)]]Z-EY_I\!O4e +@[rA]qs5:(@=QD8(q(,)*u-Frr>@[MPmZQ$LoI')sa%PX<#J"8N+XD7L0mtmGA+u. +`:8Z(?$:A-.^m2X0dIYhG--.'?>R0KrF9_4nbP3_eH<]nEqtJFZ1?:iHPKHoW=Z[_ +tVMlG1RgPNFKM?QrHa;hLMWs$1"40Tg+9ah/'U3o-B;ZAJ&cZ4iZje-in2_9;H;@@ +=F&CAKtdL[j8eo!H.l"VHe#6">J8V[Pe,IMna%NGmH(_PH@n"/OoU:ji=H*p@ +L!n'-tR6W0L]hj@$4;/G=.\N,'3:0T,/Pa6eRC`Ao*l8]K\7r7=Oe9aY]]e%R@g>cQ:tIQ +9U*>c*a3eCeAXRjj#pYe:1ZKdisN.6!u$4BSM"YUl;eWcaVq")HnJg*k<#AMaGkC" +]GuN.kJfQZ:8&acWhR;[rd$[6IPQ%V"%cPD)e)p)jgA+PG'D_n2kKg9:Zp='K7Oa$ +eh<.,iW*[E(0ilc4@U.4r3pg"il[@,hEfB[CLJeDA-]GJ`T4Ai+M@@*g?+%_7gr=r +(-R#iCeOB-KU2KIB$n-pfq@s3'@Fd2eh7Dd3ZKSHmbUk710(tb9O>,f4sq#B>,nFY +!0\#>K^H!*UJ8kD^sS5/$pnE!,'ogrcQCjpgW:$EnoA`;8M?c0:])I!(J]%"3BMoP +QB$2po;,8S(jJDg,\:^WZlJDa:8a*ggk'GU@@ +*ne-P!P]h95P7S`Ya,PBps4)2S!M\``2s<7`IC%bqoA"&77[cBED9n/P=J`CIpA;) +5;?3f+C:f9!-Hl#fomC!:$X#C\pgtY9L>f@T\O\_Uo-YZX=a`?87o5kHpQ+_Fo;P5 +pfEr-&m)5-`LQ9W9c\44"Ve-%nAEim;Ro(R<;Y%fnOrR<;$06pRhqaH+ZL$DH>k9K +ErPH.\nM0D&=7&N/Zo1bo(nQW9O1A.;nM"QA&d];Fq/pQMH@4O/JljcWOL0f$,BE< +KYu[)Z8H&LQG'Q:f#H%m9Pdi^P^b-K6d/0W7KP=,AlHj,TQBt)h@*3cWU=anF!EZg +KWJd]4i#0[YPqKPn?]>_H;=EJdsm5%<.n"nNFS[La;O2gdaJ#)7SPeN&(9Q(p?s7O +/sZ,X_W(6c1]h^63Xa+!(QKI$Sj"ecoF&E^p'/jZj?e7up5p6RW3R+Hr*Te]dfC-9>l3qNK75FYd:0f$QhJ0 +c7oY4fb((<#+2>YZQKY4h,\(N,CdASoC]'GRB.d`4i! +4QFj5L-h:QVuuHoYgf5dj?Jn&>sRjCK`;capsF<<]tNDIHqZb]TAq\UuR>qro$;\! +/KcXa?K4sf(]^]]NqL4@hB*V9SBr<5T&Eh]m!d9NdHVL:XMeF\$HH[)ib/%T>nA_K +/^Xp]iO#0?E1gTa43g:)<^P7H=r'dkI(*?bAtbVZKVX'<>aFU+Qr'Q5T_YdSn*d)" +9ZqD=G\^45E/Wo6+l+sh>/&>\'nlF96\$HWH-aTB;c&u70#VtBB71K)ptO"&t:1$LqBqJ5nM$@4j.9T`t]Zk8C/A\SMYsKd#pk3R2:?QIk%h.p.pDjX&*LrpG+3[9a6Bc +*^;ff8eCfX]Z*>`_D.GTuro"*G_:SS`6.aP)MIVTFG"*Z3QF9*:e?N-dCAjh$:&If +-Q0u47pKZCh/]:P6G3=\B]+C!odH/9#[Edm@=1Vp$+H".8gZH_ir[e!lWiL;T#FS[ +4T$a/=>K5`^c_q$V9;d&rjSb%rYVrAH7m=-\/n0V*(t]H@BuLc%eB:g4+p)ciil&f +c)%GfMnqSW!mq7!>qgppM8pY@LbBD]iCD%SB,k&DYC6sAkeCO([S3nmnG@qXW)&>TD[#Hp#196rIT? +;&cLbo*trg%)/]I+UU%J":<;)U`kQ,&juf@UD,-FWW1?H0)L5obhS"dEjAq7C9j?QjZS=1n;sDfKPANp9>c: +l4a5mc###UZ9jc]_g=93_e_K^)YP&iR9*b1>0UP9,*9ja[O)ZeV8Cs#;>*(;5O.QW +i:Tu*'R4,mN>q^Ik$'E;+hj!?J`t3R5eg#"hOCejEamAIg(RtDBlg$['t0p\][2'T +9h(JnP[,WUV8;,/1ds7#cD*eA:^AV@F#Vgk*#"kcR;55j.$0\o=$*;%XqeV),PPd3 +9sCG>qC>3NYRFp+$K]M*guJ'UOV7%od`[2[CsiKZA;`'B43;9A"jK!1OpsfqD8i,/ +eR&b;AD\%2lF)t"3\(K=g([L9#FsRbX +.cna_#*<&q;Vn->\*!t)WinOHLk1JRTk\S\;Z@RYZ$A4qcn3?OY5]fChMp3aPHL/)='O-B7WEbd +BqW528h39!6I5j"!3,:O%R1H++nl8`$a$#:Xp64:puHB1.qFH>BYI&9EC+DSC`%^- +'jT:377,3:pJ0cR:>!5#EeI\8+ur_K\d!2e9!dA%B!UF6Gs)m\#)LT]`<*I*X'+%D +o^UEP'u`0c/6(bPRNicdj4JXdbsT=Zh?NPbu]cR1hU5 +)8//%K3BOCBg+QQ'?WPid"]?]_CJ.fcIPpU3o8Rop\L[>72kGr7<*`'=F*N(aoY?S +nJ4pnSc7'-V`.a$N[4sVWJ_VZ%[?+a%<:T%J\"q+YGXaJ`N*+KTnK``1qjDe4L'7Rqb"iq`3p +s1PRm.'Ipf&7jF"H@JVFH';NR%Ck(*N0)'>rPK4lGN!%h\OB?R!Es,TFo8a>1]ZFI +4j7284a!'$bOP]k$h*dA#ML2\?Dtr_2*+*BO)Eo$F:Eh7eb.i4=V*tVNA +!f3r#f&G*VCNb*+0[DS0(g(Wd/D9:,je"1Q2?u5T_oZ0 +1"+#?`ZZu_[/&u/]5Q[OclV*nW&lE`nFhJi!P:5L"`:,CB`u]rB#EXf/6>78+8h5m +<"=^0.uWE!g\UOq5![pCuC>K+,tQ$O"Qbu6sFABR)n5(;:s#_3BUf&aTd8Um4.XCT +UKf(XE#Er7N#3f[qqs^d%:%7T[/[+i$\!pqtgrrLBXDs/4D=.U#M/Z=%$c(P;`^h?+]`;okF)`>pbQ,`8f#Qh=hY8l21D]R_kX +_)nfGaYCsfJj8JJ;W*1GKK&0ik'&'`4dD+dd6.g97'A?BZudNd]>oJ!:-$DSUckhc +4ldY]`<;lLqEh0T\`;t/"Z!?>B= +K_:!DBj>[*(OikXI1(a&?\KI-`qWK`][s1[FCt]pmSE+\PP&pg<>_kQ/_mLAhZ:V" +acm>'N7hgj8K=`q^X%Km;:deeE*ot3+J)=D.?j,iIFsJ)2A6AC_@7n!;sJ(`UO6u/ +,WT@%uIBp]f/Hjf(Bo`i3TYH-+jg1Ll-^j&[j!%i-k4_?B=#@:W!\I'N/1M=;W$MT +;?fDKR?!2UZ%,XLQt_RTZ56eF,i?Sk.Qu;oOU3.GV#St7ZOXd0A^QdM?$l(,6fF'+ +W7WWNp"j$c#>3"o(o?@:TkM[CRVfI^E-b`'aSBeq+nOCV/nS.FQD42+]j1Sj4]#>< +W:BfRg9jLkLR5V:m'sC4)iJA-si=EEmI6/WM7+MOh1rVMX-I!+5$Qr7k:s!%)?lYc +j!MH3eq6-:Qb\q!"k6Hd\jA"(-O]-,9bse$@1gu._`6,d4C-t+A`.^Y=`(Z;h\YDA +1SW^TW/F\NjJgo`AanNcHDGZ1BX.jqYJbXc5ciMWo2eFk-[p%m#=a=b28OK9>)L%hJ0dOh8tZ\F-&5%,h^O*]3*HS5q7`A`l6_ +dd3<.S$WknkN6J4-gJcf&IA25i?Y;*l![7>51sp2NYfP(tI1TQ'=8NYefC%Wp1M&p +m^7`[IB0N=4:7P];Q3(`k:cV[pRlB@_K:+)/^f%1Q3N:HZU"[g?BDW2i=a)FX-$o3 +cc]Xd#Ir85T_[^_nE;#OMm<.e_@4LLF(.9#.AiR9-U%.,.LEU!WIC7Wkm+b)1S]a%;IFIWIO3R8\l"Nk95R9]al^o2C<^ZDi4c +@5L[Y9h9q/j27QJeuk>\hSe7K'mi#BF@;Edcb8D"lmo +cWtgdXWr9E7%;$E*+o;T5N11;sp')5:1DBl=u.g\ckJ)NgVmeW1*7OO9n_lSF,X2! +7mJ*(;rW.#7$du2':RN9SXo`5gO2=*RX\l'7t5EoV?XDbI/2PcNc9-/6TQq@V, +Z7EUf'goDVk)?iUeP\9`Gfj<:--mc-;Vj+4m2qTS%Kp^hZ++Z8BsB_ON9hLaN5IA2 +sda2!=.C*@m@hFgbpOFIL,Q`blBZAn&U[qi>c7qkuo[:X(7Q%2B8t>.oF,u'Tf97b +++&RHFCu!:eEMbggO7eqp5M9D%bBb+KFkg\_Ptdo3",^lUN#ki+7pB*2q$WF7325* +_ZAeo&]EF9nqq)*/ktWPrY1X_?"@=C]b<[NCGo2GP]Yi#VFHIELbr< +.QlN2^\;fM=R62F5QR(S@SdWgq[E3CM\(je]2*!]LO)M!Rfd!AKQnR\ +]NNb#!)!V["#>ipo`NJ%/;u"i#O%;\@,0?BuW\^Y*'PQd"Cm2RA,Oo*4cnGj9bnY% +9)t/n_]2920Wo8Hp57Ce@)oADC>=M0<3O2Y/nYfn"rJCk_lA8ZG0-n:]Gqk\LXSs?)00DHp?T,BZeff19WA]I*.#'6!II64@j>)eK)>O!7UeCnb#:H`S)W%g=FpOQS$3p3+-2 +4g:Kims9.%hdYfGVI1[]`$M]3!r3u=6_j,M?*HSrrBl"A\,+/o#J_Y2k#`%bdSAf! ++_'0:_C624)Nl^N"a#keY1tUfE)3pFS]UKibON+p%At,]J@ZHJ.:lu!:k=RKcKRl/ +*[`9DH[mX`L0h?o-4?!,o[(YkN:rf][p6E*D=<1:UHHRe8@D'`EQo]M +V!!`gJB>>kV.rL5oIf-dXXMLeplVF>$$Kdn$IS]AZFKglH4n\+^fGF%>!YXiKX&la +>c-oL#RUW-`;L>m*K*a_caFT66_)YBlb8lJ3u$NG2dM^h=&q/_:/Aul5Feu;X4ls^ +7*[q6;$uL^e)lB@(Nh]5g=!-p/Y@5NdQ+hU$9-^rrBQdF);)nN!L8]c8Z*JJ[ZjT0 +IR%`:\\JTI4BGanQr#0/RXIh@#%.)1N`b/#QPi!^tchihhmtS@jl_OXkd1j:AW]=: +I^rD`p-*'NFCl1=]gAgg-Hh(XaOCi^e"B7iT7APePj<\c.QR4(gq-YU0Nc86H,3l" +-S#oTc@-.U&5,B4<5>-TNNlFrNn&2/`k$Cf/NaGCk?in,b1BN:^?1DIa#M9rmS)4" +?M7`E8Y3B@a_2&HNL]H`]KJH4C"=_S2i$em[+.?=4gsqOS.Cm.T?IiH-^r#aY_Pfu%E=NAiTiI@V +'\aT)l"YHr4"kZkAL@o1]d++rl*T'^SSanQ[_0N/sGCp)](LboY"c^63mf/2-&:F +Xg^D +5;%%B/h#ml,?;?mO"^g3B/hE^?9YaP9B"c_(u8H\RWKH:b?K6'XENQ7LS)2A1 +0:&q#IT#3pp\'\!nN[l8beXL$i$WIgBi*.6V@"<-1q%`H#9%rr@X\p]_QnG +a%(S_CM`hjW8c\.sSO.kB@t79T1m=3'O'e5!nSa#VQH>B;%B#*AS +k.Opd,%Sq83#lFXZOFhB85i]Bl:>.M;JR^Z+*(DU%*33jDCK0CgmE?3oQ*I/+hl\h +s@,rDhai#^PftdZgWGJE:':bQHNr[m2QPqL-uueJ3&9AX'R+!Hj0$u4Z5MGHlgc`> +<@Z7C2]i_Dg,UL\gDCCTsOYQ\S1;S#*MX5 +R!H2,uei-&j+T?rlTPO2b%ELY)@..W3hlXW=KC%'>^rXKk+)tSQd0dWDV-*?dW(Jn +3gHeZK[r)3\&3Wh;j`(7gl!ghY8_`grp)N/_cCk\jqK44!`LPC0gr2+D&?[GS8d.> +CbJ*OsXb"JK4&S]B(CS!Us-Pq[H]aB_n9[[GriZpJn]-7('S4nC?$S;3hFiXO;$+V +-&sPh>dPIF'c3Q!!QNST_O$:!!MolQTR4cle@D:cV;1'j0lRcgtU#PK[T8ph%P:=/ +BX`%kZ]$.e[ggYr@^s(g-6I"C>hVZ^.>Z\0:Pp0>].\mOJ".Lc2cqWc/8"tqagWDq +^VLWI4dV)$"39,_aPT]<1_QVRrih8KW;>>3YU8:!;rA"emDA-r)!,!)]O_&6KVco` +.$`h_sWF-O-B!Oa32L7BCP4n4lQ45]h;"VpNrBfa +Gn-`c1qToFhbR8h6d+'F5;.=C&fd59m>Rt/+LFpHUKq_bi&u'g6:=.%AK\"%24J$.b, +Z6Ag2a7!OM/tD>1N'?$cgZ.O#ALX;BItUd6fLQ<0O]oLVnV[([dXHP1jnI4NcY"sI +/a1^\qkWo\')fcA6#"\r<0/$9+D&I51BVgr"RP.iBF["+G'H[b@Fac$ofWEcdKKBl +VRJ.Moq6r_58CXV"14jr\udtrrBB!_)tGl7>;jm.]s("1@q-h]c"-mMJlci8@1f+^ +PL5Abdq&IZU'*CjDe,e.d*]EO/?7t5OO.?&n#M'#FAk!.*!m/"9>#h[2./a3G1FqCk+rU9qqr:7ne')`0lZG*I-^s-G*MS^'Qp"4K7E%4ERU[F +J2l"V,m0A!V,(%es3'_gJ8r`RAj@(A0H-uZiL/m#uQ)gg.*4;pAY-Yd_d=Z,)^I4% +->Q>;0cKWEbS*4.;DV@K-=0E6:^6FWRqkh[+h.G40mF@9A,t\]E'0! +deC\Tb9*'nd[_X;@+P[\KOVdne.`^8*)u^ORFHF1Q:8Aic8Bk\Hj"5V`jf!J3>+B6'CF1$+ST20 +h:q/IOlF-@<02@D[kf0Ahem%M`K*#mpEGOn4V*\K($W"YfFXg'&#("]Na?s54;<"X +'B,urm%IJGrXIj"].+N$i&?5j?>3$]`.t`DTea._fmH$AasfY;jEe,+VsY4@bG`!% +"^j;^7NqY8$i2o0e;W\]!&WA`nA5EN.hY\VJ4A"KHZ +dg2I+7.qn\(e(:@d$)]ggK:p3cuHWp:gd(e$N7jnD1L:G8E2FDRgBbgeJc*"-kKZK +Rqm1l!!q]`h3#o4EmT8SgVs(/I.+YCNqH7nO[V2"*VnC-R8a`*.Vh&MW(WiT?Mt6' +:eu`A1N+.!Q?,`>XrUofZ0kkUflQG<3nA2B/n!M_<2o:5edb4_nmq+$JRhR5o5dCA +CDS2;#^M-_i@5lH^aCSogn`@j)'NaV`B]Rk^/2lVqj5;m:c#^@$?u$*o4h3nt6LUP +JCAaQ$R>\2YK(m>m5oW88[D^QI+t%fA1IW^BHm\-oDCjT=#KR6f +JprY^Q*RDJ2l:jLA&((:F1YHM.M/liWln_pGP!>\"e +^1!FaeY6L4BE['nHheEp&9]//MV&ahkVI82G]d)]/rR@M<")u%TcI$9;;3*_pDL2) +N:+q,7r!Eo4Ee;,just)+18q`I +@<)+bq)s9=E'E?fXB1Yl77IlLJ,:L;3+kgPJ_^P"oC72pF=rIa:,pZ0N,[qU^j.7n ++p(C@:YC?W"C984hC,PHJi:jjo-BM$F'&TT0)g3mI^:r,e.m?H=LV>EeEWO@XpR-k +J@helnq;oSE/H0<]JG0O-foek;J[639eD"C>'/snE*hi4E-XuNS^FQjkE3D6>5%qOB=::e*:9?EK$?D9g7`9ZBF=pr +B:n%V9?;UOCS?35&X+k#;9l^@n+/ASM'PJFA&sO2d:Ws"EIlZTES^0+QKUnI&[,G# +6S/MC[_Se7u-`q18!r,L`qT]Q1A`<78^SVh_"oT,`uGNf\Hp\&7.5h)hQ9c=K.]R? +^S#Zj/tZ*=2c9NENZu6KffJ,043.;Jlsqd6K\9t>]3EF6#+h;=au6=3b05jRK38RZ +I@p_0':W$TJ5/'&$j9ZRpIJ!atg3f6R`.aJh,H#*6g^&Hn?V[U%>XYjT'E.P#U%uh +eLSaL?eM6eVhPKlE\T]FQL#FRTGo]4!!lNiJR9^)4OCjYD3?4[e`r1%prK?K>h#a= +#X?OQ&/fSN0O&0Y#bDkVunQB9QODVoSk,Z9'Y&=4,#WH#4UkB![p_I4gTc'Gae0/4pcETOAFFdeArPWfD"ou#%(OXG7='Rk\(8spa +[p#r7RhI/WuQUMo:LV8.B*"d?"qS?Iiju+\kuIYr:mHg]9."(X(/Uk@N^6V'ERbM' +7"S^6\Kp&Uf=X)JWD4M`*i1>lT/Mn`u1c]7oT61"O,i$"jc=4V;Yeh4Yr)Y?.`BKJ'A#F5'?n&m$0a\nE9\UXN\YjPh +BJPHRa6`L@%M@9H-<\H$38IZP\i'oJ&J2J6>hYI.Md3'rjf<7ff3kU,u^_EcL[SIs^epoYf\-d8%WKQHbQ +9G1-ec]0OI@rEJE&aoFC6;%^efRR^#.!(G!&;UnW_7Vh8EJN]lh($S9&;m:Q'G=0I +^M&IV\sop_)]:f!N.7=A(Y:i2&^4;\/@9H?WsEbq6B#fc`KO_X8qkHnt6JE`q`YQS +Hn=QfC++C*^g!Za5rc,[an;$;j&hUrrCWLme+Z)/1-?Jgr#0Na5oQ*^)Hd_Y$$-+] +'p'MCj\$]X]]GZ-AV?;F_q\A-j0nNW(Nm\RMiC#V@3@#$/RW>mJ,r +Uc`Cc`R]WHd#R.p&o[#GF%'oCiq^RFkg]bL:%8C/Xc8+"g-!B%bO>+56g;C>5=N&h +mBkNr%kNK^D&dNTO/!?nNVdk8S]kMl1/*&)/Pr9CXN&CnXDkZZ*V!lFZgh`\+`Eef +(u5nUM]YWd7VR=SH*t9Q07joBbD9C\jjIj0GltD:I4ZS&#aob-<;iQbU4J_"pa@?/ +'"j-2r"+3`!5.EkM`kGVnrA?UmuOfEtI\aIK4KU:UehCY9\,*-,827Yc=T)X60@Jr +X'2*_lH+S[<)2Jr"T=lAYe]'V=Ea-mBtYqY(,CQLNcLK# +DrNXX&-Nom9W6!MgP4p.&T:t)8N1&r'_XbOBePNHPjU'E^HDlN]pnDYE[B2'nc4.M +DS9%hnZs8J-:X2M24BOor)*o$Kq:LfJuZAj0sXg"!qKce5;SQELtgU&#l^Z6U%;F4 +A\=bDK&/U"9!X_Pa\DZV=M\uNp\WRE-=56!kqdd5N&_pZcm(-QfFEcVoT=#]fV"Q) +!gQgFas0<8*oh[!m9C>ZoaL=E/9Z>\GR+m5.eAUY'"HsC$.c->UpZE6[N=QWO//3A +CW@ih%[dlr+d0?oPd=A/Yo# +6B:u6Yf-NIhUTu&d((JDrcn4H/&4kq9TB'he8HOl!F*p(T9l*M4)d\#Nb,^jGps/Q +@&c=uZF^GrFCrrDgCa6i2dr +nSVDe$U8!G^EQ^inLQsngiT?$(8mIR)Dl^)cJ4X`cs<@o=(6_YEPUO"-62<=0BfR: +BtRp!A(fm!:cS@/&_)7[2,ItCn8'uo8kKOhcUNc>=h**K\4:pmbj%JTEt4;psM&5Z +K/31lWgjRge8TR^2c3SiG-BGX[fn%S^+lYE\gfYNT/$7aT->/J2QtgQm._L3rRnqk +uLQQ&V(BHIVbM]j/'@;)fAD2WlIqD0\q0@"07a\)uX@MOi@nh;Je]^ouQ1(*G_$Xk +M/hmT'*dVg5Ca38LO/'%\?rm&6P\9JW;3/2(4_iQt$4'29SR3m!=_"`gXA4\Dr]:* +A"dlDP2_ifu>H[hZu>6pF6!MY1t`6[_P,(H8_%9K +&RN)P"2seNRbP^Y3V%c`^:)?>]?6l_/GBia:M@ +'n[S)n>U?+q6=^NDu%'12$Z"$r5)'.M_^nThEOUruf@#e'k +.'7.`lo-H$-kaB!**E2N:ZnoR5lOF;%$EV:pMp_Oua3%duC`(Nqi@"gc$R1*_Bl4p +?P+lkZ9Un>XXsH3oK$L.Ti2,jVf5tG8^D\$33jJfO]W1&\%pXo]sIeo"<8m2X\25> +]KO;i3^(cim5i:6-*H3+%cNd7cd2dO/4=\pr's6P;b&Ram5F8JP;-B\&KJ#'s!7Sonb`Cie!9]Q9Rba%B2`I\;/#7*nQ?>uko +DS+1X3p]=Q#6(N4YSTU*4"9n=RI,JgmYJOXL@W@>T@VYq6#s,*M`\SF"6n$!(!`H/ +&C9q>F(:Cp'O;(!Q!M]p.COI!4%atmQh(BKj*!=6*+jF>#K*ZQhRrb27cuUM"7m#M +C!,^OH;gZL\\G?cT09$eMKJp>?\&HYsXRH\Z\b)fs#\5aP+(?beN?@+A:u"I-j +CrM5Ej@ALJihu*kM^'mX`qQ]G7i4g/onZ^b^YNShgbYWFakLupVkeTe<;'C?dK%p+5.[l^8fRLn +RpdfQ@j5C4R)Lg9Da4Ael@eBTJcAF_B0']#l(F34rrm:%_p+'[1L@uk^Vi-u9?ot^S-6&g@;MRTL2$Ll?,GM.4.99#>0;U-C>q ++q=e?O+Kn>4:R6__QsVZmjE@:4F+3'r>d:6cJR\A?t]R[iAb#ROp#LDqF*.^TAou& +9qcIRbeq2RS7*)[Yp:J!.<=LnD?kX9tuJUkg_a#'-M]X5Te*SL%#2`r\V'Tn&%=%] +%(5DU[*s3'8A$]>/8S=QTiSuM^oP%2m'l#'HK%?:%\'grK,_r-I3QQ-U\d]2L$ +.n^amM1.3iFe'#<^W%RUkfZApf#hc^4.m`%dJ-FhY]eS +CQ@ipa,q?m/_D\Xg`lSp2'a/"@VGCh-\YZLLma\*N5Gp\Zrc+Ns7NO7ia)?RP>G:7 +DH@ec@*L$&*iT=7jPn+,2Lpo>2=?7]CHOAS\4fNSdLEQHJGVg5:-B4Dj<)6*=k?5 +I6?IMmV:Qf)GDCqbN!OqKA_6V`;E!PRIk%7;PZTp!>BU_4V>%B#jlVIJZS#,_*cXI +gKW#Q?<;pE,0)[rr@WRrm?%P>(7&68$HcjGCDO2^S>'b8AESpb=bQErO"/3@LFLSO +fCR^&)q^s'J\lrJ3ATZ/t[E.fRi']W>q2L*eU9hIl^b/r#*;UDel@kECs2 +'4b%Yooe$l'#BfTDTXXn]lDL]bB>Vg,B +7[NK!W;o^?gqpaa1*rm4?[Q?rC1mj+"gdt!4%bNU^I,%OVOqUTD1U3G\hqlDgIH(:Ui_b;a29-D#9gR! +Ve8*htT+5`VqHq\=:=k3q2jEQEn3:ng3p,/2F-[NJqU1YjSEO9tgQ,K4k+)!*K7$i +A0oi[-T^dT'LF$4YQs;!*]?\K>Q!ig;n-j[^Iu'jq&`8B-WRY#35,sX=k$-;bHQCI +2BfJ]oAnY)AlL4$Q)se_N-s'TH`FI#0Kc_gUE%COdib"nC*HPAtnH[,7!/LRS%/a&tSOV]a\ +abH.imVK9%d0EBHejgOdNHC\MiusX/AN/(r-,`,imn,]kT2oCiEHXbAj)3_9*@R`p +IT'CN,IIn#/^;2BF%:0Cbf3rlir07+Tq8Iu%RHQGqEB'/Q`;Qb,N#* +a5d0gLbb>"7s\#jVBs9WZbfg4*7D1S?i4[#hha6ltc'*^\B'4PH+W_Am=I`(48\h4 +6acaCX%@fV8ep5F@g>RU6ZFm0#QfR\k'^f.P`gC^CgJGi/_4ar!BK3`NEoY$L%!`) +LesiV&F(Ea-jD(i4!Z+UM.dKIETDt8H/[)/FHGqE^tGjG@tJq@[?8VUfr_TH +L-(",Q?%QO),lWJ3DB>(PcCNi!#CP`JJK,P"GBY]Z`t1l!%>7QQ;?_Rlm"qAJ'V"3 +He>.(G-m6oThPC,1RnR35T_C4R +aA'mUropuX?#>=qUDtI&X6dj6oZ'%^\>ueh!11DHUbJQh:(dZ1h8'3j0.K>b:]2&M +"WOu'1r20G\J=2f*]dOkVs7Va>Bi6D5;)p6:t1g7X`QgehkGr9;2gA8=\8@Z!WXM +!,-2&$9TdK$mFLr5[R8!(rC;E[o&Q`)inAh8O;c3D"B.P6?=^PlEa*D,$r +QC2:!;.K6Jur8#WG<:9qG_;n6ignZ:uM2#H,_pB#%Tn0)MAGLmK]+CB8lc4%Q`UFQ +f63^nC]&,$Eq^bNcXt*Q.0JnGDM?]>aQ[kRjB=FcMmm#l!4p9#3HB(Omh'h@%^p,F +90AF?3cA(([YdES1YhnKC=iNYGLIB<;uqe5&ug0R+Hj=EBH'H`r5Jq4F(1K+-=`OM +o3MYMg-94hLIa"Ob[;Z$F^5*MSLZ'n +cgUV%2iDYjo'J*&8;I5!??,<4\$1dm0cHd@&#a\9mY.'RnITR(n01,qHM10_b(M$Ng89rl0MD +/:hHVikCDnBHd_J2nU>T"V')CJ2+&>1?G3bsp[Rg[ci;?if(ZF,neu*C>5cfUoB"b +M5/G[pLhWn_!:Xn`i*;/cO81]P+(Jar'`Y7".E$!-m)mHG%t.X:)&nZ8@\_5QD?Oh +"RHPFnfhNCA_?3>8l5j93TpnMug69;g#[[;=D0(./!dFd(^(*i3/T_\O?5AiHks>N +E%%c8%9,=I=/#C;LO10^)\-BVOnKXmZr,;jU?#b&ag?UqG]B0.Q>bVKU19#;E3@aE +QI0t!<$I'GQ.XH#P`>O_C-:Ds'O=1J#Ui\(r.=(%=bD +/V-:%$%U&B[>\F!W.@dF*htN!dSe._7,^j5`.PH7oPV4cEt6m84etASG]($2LkD#o +'bNZr"]t*GDBX^5E$LMM;gKo@AOQ.f/j +V4>0k\@Jgpp\)EZed"CLnIMK7(uq$qK+'r"hSD:0.JS;5r>hi^1Ia6m`4gK"TWUM+ +8E%;lShHGbk#PX3T_9VfYd=H?%oeWiHN6iAN]#SK3;B=JF']=9'V7MEi[$"?^ZBc> +),]EAi,+;7+EnuM2i0O:GP/-#(STLA1Ypq'A&B#UNC*-Q-/-cV-a^=6+Ec;Rd,?0U +E',OcCW]/C-i\8a7>t$6'%VbHHB6hEa^[7rkWEVrX. +HRO*3iVT3-itoG8CdH1FZ/eE=#^ +K-0L1G2FGK@f^f%/84ZRXa4'\\J@J!'6#'6aql+NDS[4V`=P(ZKUR$/9p4;3>AZo_ +&?lrfM;a4ZiSA,VOq=k[5Q0o$b'h%c_`ppXFH(5/auQQmfqQ^Ii!6E,>-<.,2-`g+ +q:mSLM03!,N^@l4EK)c/%lKg'7!:`;L'F+8GZ=.(47a]]PZWLFMAc/[Ml$6E]20ZE +tFKP8#%Nu4fAJSrrD*2LAhJEIQ)[B_YSJpmfm+N#U>(@'l;p:-K6fq&j[5Dnpt<'" +5nSUC@U\=\O9Nihd!O_91ce)o-%"f*s_n,8ASjpp +>cLR3.S!NMo-&it]<&*X_\YUZjuPi +ioi[YE(5GK>Ua$5DX5U9<49XU__N/0h!'q^"Y*UP&`jiO,=ag%mDseqE'Vm(EXLZ( +\2jC3n:l;F^o3**P3gImi@od`f_g%)[`9DP"nbgr(QVX+13JQr&K%4iCLUK"fkr(F +<,F#3fE:Mg2;`%Ub[M^nm(?DpC-(g:1n1%%/N(4'` +NmcK[+=Yf4W`b2g>XAB4 +5Quk2^u.^UDRgX9Kks(B?$RKG*/j'5e6ZJnCth-GXG6VFN&Uk[''q\jM5.8Rs"_TQ +/5/I[Xkk3(?G^XB/BVg!"&YG_tJWMf"4/#Hq!%u%de7$l"1pPYBRmtl/>1Ih%g +USg%MK#[#Lm!PKh_7-QN;hBM#Trb-pj;#&1W6aE&3_ap0fSfcVtfGnf)cn.5%d9Aa +bD+[:!^nD0ECOXO@=Mg_XEQ#$bdZcbMtgIT+a=_i\(Ad>>&\-hY;"^;5HQ5rS(N#^ +h1)Z1'bs3?8l@2Y7LDprr<3(eLu'G;=KA!ItB8$*dGFq\(#dcUNiBM%h\+k.\7N?K +ZYmM">RkE?qgHId@]T0WClM_[MHnB@$CH`a-D"dl/fTre-= +^bG:YF0FZ7C>D5l6K]TLfNcR*W.UM$JjZ,.\gX +Su/2m`_ABT?Q$-!5>*J-gNr"eNO4_L@%"e2:KYUm__-4bE#=t^[4&@)dR[JDi!h/.FYGoi1C_a(:#h0U^;@*,1WqFr(F/%r#Z0PKFI>uGn4%U)MR)GTZ)9[-%3 +jca5DJAbUBrULsOL1qRk9D6mH1dg:oUG=9Rj=?=MeMD2AK8D1?C4jbJ>8'dEQFm;` +7e7qP(W,-g^ULpUtC\t9G*?sY%m(,R3'%k4l,eUWng@+>!F]3Z*">S%QRi[Mk&t<6 +/l5^6"`3hK?o9n`He%EVERthi +P(fTX..:%RE.`9Zt-B^ME*&UVt$Dg9h/#\[>ntQHuNS=^c)kA3+KsW]MpiARW4&,S +FSa65spo9)Ni1`UO[F@;T4,DWc!t7QT$CnYX2^Y"J3N_Y%q).8]ics7Qt$TrU4"aj*sc;8:Ye_0_j/)n*Ig-;i3dW:G%\!IU!g8)qsjr$jSIi>MgHn.iq9LjH.gb*Z47G:u9*E[Jg0fd]R%uk:` +cGDcKK_=aES$(aaWnf+fc-dnCsW@kp2/lLVUd#R4e,Nno\52^#40>a_8B`hL5p/H+ +)*>EG2,>)PtqmJ'QL.Y/+_QZT;eIlf[1s+@lq,h&itJa349',Y]Gqu"_p>sb6JW8] +X-7B82YQB,A&_KAcn7_5T+a6G$F]K-3nP:k>N^qo6g#=4Q<91fYC)M9PP(\%+Id8( +JG\o#^YS$8GZ-cO]iS&BjF_/9TEl\PR-e6NIemk#Er/O7a;B48rm%PoqJ("E+l*D_MKAg!YB +N]VLL6VjGY8mBHe+6H:@5%po)0CI/IKa'-2ZT +pjn0);(Nhb@'M'N]s>O%ifA.F'HI,St+B8F`=Pq.?*K7J)(r-J^j3c]^W/R[.d+eJ +R,(sfa(KPLBE-LNnMAOEgi>KTorn#V;iaeSV3WMl(:RsPG1i6>XbVsgH#m3!sGgBj +]t2dH7/ET_l?:VEH(1E`sSV-[piIi/)Q,KXXKB.Jg_k#b8.^VB`gSA-ibt!BEJW6d +VSQ#jL2V3;aoS^)gn._=WM5\?X>Dile[W$Xd68oFWHN*u_htSYh`P2HB[?jn&nU77B)]G9I] +SaDm12UrI6*D*3nR\4F+rqcI*/8PtaEnCX%sjut[*`Y:q_q06_0:^e.?QiBjIShS" +8"h9kPQ(P_`fbFi:rl%I#--9FHHbQ1LgFGG69(N>0DM0nM[=P;#4.k$ +]co1?&dJ]!Z=eKC&n4_O&;!4iaPC8(C[eq:at!tb6kMh@Bm)&rZArL-gP-"^C5U/h +i+00/hQI<$0#63=4>Y)Kq>1lC/?Y2;Iht#n])/tqD7*Vr"`%SJ)OTCA7RgUMa=8L+Q0')qqc5(Ihuoq)#39&JjY]:M?g-EZ"e`Lgt.bepC%$ +r@$Y4A]oOqhp-j3)e/`8I(+*aGpQ"\33K&a4+&UUKa^#(cE4DlWNe'FeY,DDYnG/F +]ndu?Pgm%qUj>bYn\L.ar)r]%(f7ok-3d#OCEFkpi7k&E$7aC[`U>Mdt/tLgN^'nG +nC_9F4]u]-GIT6[p?V$-ZL6)6YUIW.[$\:%k)H8I;6fiOgstK+a?+]J1Op\I(SJJ) +94*bc$\@E;u"DlJ&V)i\a[V:nDhLE4SX?flT/9pHAXl1JIrJ=>H'\&64VE\J2_OV7 +R9#FA%0UtSl@Bhf+mftp.G*cnY:K('mp8FSSDlDk;CeQQBrV^XuCed!V9+):BPk-i +n))K*u"Q-JklC3(RL_;La:GL03cBgC]MG2AeEe&;hA.m0L-)`Y%u"+ZCd3U-h5d-n +)@*=i6X:N[X +;a65\TUr+'GA5-AR_Rgt9[i55\]j[*U#e23sk[j0+5Rj'/Qp]$th?'A^cOX%J2;9 +XC-%Vb\UUSdhG#[6(j-XG51N?,p:q.A@'( +K#,53k:Z5mAu#O#9J`l+8uT\n:/J^+n0^R'KXOc8mNWC2-h3@'ZSkti=NO2A*"G%dI%,l( +pB\u(&n9M7(s@Nrr4X6_'`\Bi)O;3X@ +Oj,*WUR=Ym;i%fMgbg(+6"r7FK7;3grM)6ZtVg/<&2R.4L#P-TkS_NS\]d@%\OA__ +a`k\rl:TDe,3qhDa#_qVVSqOQZmB-fl&"D!PEto+[@QBT,3R.Wum2+c:X +l/XLC)S/(H/LpS/T0hm"k0V+jP:f]8K]oM(Y8/*RMd=WN#CYdCg]eS/"$+.nEpg>^ +'`;15MFT+^(5Qr:`VVhYg[u&$HI04:!'nu.Z.N!p+\d%97/&!#3VqA8N"i^g^-n +Ooe6:D@uu(&W9T,[r%[:DkY?SFl'mngfjO,b+[#m,![-jhiQi?FVfmiVH&u[9Z8i6 +APG%1<#uVl4YGGT`*[/rrB1c>$`c-ib^%!4\Jfl?>%R'b0)(k`Z'q%L:q6s+uiqaE +XtE28qLn0-sn3535ueF+'J[9hZW^Nc(`/1[>G6b@b/fcm?E\#599\QVN3uf<#8[L' +PE[Kp8r$[(]`6H2$Y@=?d.%l<`r(@;b8-"fFa1iJ2-J6L7H4.aO+6]Smu/!rr@TNg +g[V_#\UVi@'+Mb#)YF.J/B?(^\ZL-Z_+Ri\p%6TkSUD32$mo.g_e#%#=<)*4+0`[i +EmoKP#('Ff_=O05,E79C]qU" +^oB0&V4,sFOtT=h&;j>kA]FJN+@@CN^k5ZiiENdLG$_P3t-uHY^ZQkJA!]C8s,N#& +:p4?7g.@E*4iIHo4O^@WV37.PH9F2Ka7@q:^"B_#RbPfbAp2?oYjdu_6gmZ3A2X.) +&N`'bZO&%*NF7sFkj2V84OQ[X6@-nd\WO078V.n8eTfnG5>#>VtPiII$9TDfM4VqD +Vn?J`!!0o5>ltLacFD]-d9ELSj30QP5(U0+1mIEGTF%S)_61Bi8EF%qBl?@C#ZH"X +tg6UlWi.gL@):/WiD)i?;3+YaXj(K/.U]l:H0$hgtQ)8n=d5G:VK7Y=[(.Co9N^4\ +e"J8*?HUQfPN!SFl!!H79sdr%j[tcr7Gn=J0^Ic8:T,Y_f4Zlp8[\.Jf+hsji9@7' +\3(f%7dI60s6(I'^Fo%7%RtHObPOi5C5N`%r/coL%L:6&bq)bb:B67:sk"aI>n5([ +2OCC.?\:[^NXB7rYV\OYJQ)!U2=u#0iM:H[^(S!nm4l!JuD,`H1n:5T\@YuoSI*M% +on`X#OrK/CI^:)FJ/ao$3V>kk&]id^a'0!;>q#0nge9!]W5R,Bp$j]rQB! +:I57OSE0]NEo2&8fUH_fG6]u?8RR=]=O*'Bl%[mUE)6AmHoc2jL"peM +K\iIO_qJM7NX(q'&5D%SM?^j[<9)0mi]'#C5qC^\!U%Rgd:Q%"jSlO8#:B4082M!W +j.lH3MO'HnG`K(Ok)jG@m#3GCT#R/%JYD9`>$("N.OemJR*?P&o^XS!-t+6"O^kAh +sHcY_)XU1"MZ&][P?$=CY..blQ]!L,+a.Nb;lK/\WYXD[^UA40e!q>r8<5R('*BgY +A)NN%qn!Mj50#I)lIAt2Rg5YCi-`YQNr+`cf%UFn,+P@-a_EGGI-(K(3r9J*-O?f- +j.OI>KsB;p+S8WCrD;E#nrh;No"e4R`FrBL9.d5QFtdbXmSgJ8rLgX*YYHmS=K6n^.ZFRhF&EG-OJluKd'U4>@,4 +6UOP*ni-/E&^g&i,df,WHc&7d*oAPMWu3cnl,`;:td];lq2p+=4'(\KmjhBl0YJ7S +h8E58Kf%9HY3M&3$0a[T"^KQQQNVYX74!s8F6"f%J]A1Z6FJ?bVLC.W+^CEFRmBjI08%g0nD%37/17n8\Jj1.Rq +:f+Eo[=M#h"l_qb)rn`1@O]05\*o$bc`R\SJP*4Ko6$p0dG9boJWQ)fef0c++psq@ +p-6KJYqb%5MI@X/>M>L>h:LcO+k/&`gJ.AN'%p90+?Z+$Q4&AoBg,h!,K_nj68fN2 +Z,hG\r6M._c4/Ei:A5bYi0PBaibS%S&^Jam6YA]("`)G_8Mrd3L+@E +j$0T1$ooON1XhbqOc_*r"m]==+m.^r]'Zt7I`O/;BG!sOP&D'gZ@N-*9V:EWkJ^f_ +/R*lr):7ca2E;,Z/Anf^s%,lPGPd-mYd>74D3P/q)SD*,#,*bk@)/Lh1Jk^SfDHW- +BMJ!II7m@Ec^A]-_(L'8+q^Hc)Z"sA2uFE2j,\a"0_gu6[IFuh* +PZ4;X\r:m*g=N0&&h6CI`99@OV/YE*I;30Jf%?QM^1#3i.?5_?OD#crX6X&eRXL+4*N^n:Rj(a'@;!KrCoCb2I@Vg+?`^O0nBV:$lM^-*p(bfOtJgjbtN`H4#U +gop@HVakt'$VS^UQSaf[m)Zrc1m1I>qo/gc''O%eT2uWE53S)2G!!pP\T>D;b=O,N +p>oOXs/,Ak1-ob!@.'Q`"3c."G=e[5Sea2$Y(j&oE8YgNHd\TX8=FB\)XWZ"MY;V> +HMSMW^@ndcJ&0^E8H[3"LG$ZW"(isK@'8rds_Q"1EQjL%%8+0K>`3eD`UOZ2*GtjQ +M0FrZGoC'<@9tfTF:f#XA9LYbFTC!Iq[(KijH[KRNED\QChT:Xh7/LDl:693$MQAO +B]kI,BCkHKfr:(!;smc1eW>O[eIgK(&3ulGkqr-R_GY'QoHr*9N+FI%,JplECubn#K%Jo^YP7.B&ciU;N8"LZk@nC?XNm(2>_W'P3AH#Y8Zm\%(H)K#SkO6%`+3!48APrWW3NdsMP`^D?UB=c^H6`MQhplP_+>W9W)@.G5Snnc`-u"D`a)ET\_q[ +X-FA^[3]6^D:QCpYW>BO/DDiSRnhT>M3aT("2KQ2/PZj\7cF\)19@VZt1)UC5f)rm +JpdBW;'+D[&B"JZ7FUWZ7WXOZ&^oiU2* +ro)g49[[%>K>S&pn?_ke7'!E=m>Ys)K\h]+@_.Qnk&`3:^ujIUEc<$+FmjEO&FF*( +OTN'nMcFZ4;m[]iQf$Dk"Fd6TCeAVXkXgdAU[c9]5`q>UHSDu:98i5^n+> +g;6*pn-It(ANM[qaI'2MIt=T1i$ks-.4dp/55OMP7^:UXG%BYH8F-nnP=i&J3bXOKe=7&p_JY<*I#D"PV.pHYT6+- +b5C2p&I''EVF\qcYbc'MJ%_4"iZM)OYB/5AJsn$5T9Y&]934I(MM+pSYr'eSKfMLi +FA-Er\m_FiE\kJ=jsMkY9UMSH/+)6rrBaba+dpV%hqSE?%Z/N\]ZJBpWfmgG*ro3UK'LnElCAkd:5#@ +X/uk!"Ka5PO2ONB\mm5Dgrti:Ao_5oOCe8**2pC[]Kq\4AgBuLh(;TjI@ +u`M+.qE[G*7Ui-0F!P%P]-g-id`%`EjmHrD^9d9%.?D>lW_qDA_s7l)BD@A&GrKoE +2W6-4*rV10VT,k*+3rhm++UJBS)5ASdEa%5TL<2QZ6[Gf\Q"+ePP=Reu"G#8M#M]Y +;Buh9SbGg)SihqP[O=!(itC/#I)HhBtD:MHU:D8kX"O'!:@-f20@KH0O1^D0`YZ\U +n:=^AE`8h__beo=\:oj:7h$j1I1Hg0@m9eDZ\U@PKEA!f,9WE,3pM3Mr_-L +c;W_[$!UX5AlD9J2DBF6s0l\k4[dK!+@Ys*ouKIJ"8?5%9_NBL9qcdbaaLN\Cs53o +:!:cl?5TlRjs/KR)j.a4\C5f5<@\8FG>Fm]X)sPfi3CD[!r"OJ0FF>Ke5L0a'H(&" +%Gp4\"#Ku39./_hR'FL&f7oZY7'/^-R"@QTHT%Fh*1)K#O>PW(0-u(\!On>B%n<M;B)#JC+X=p<6&oh1,-]LCeM)c"n=#Kj]3bA?h0Ns<0G^O^h>nScNpV +!7T5:WhU3"TJJ.+KhkCL\_dkAG8#Ag.%tKOmct[fdj$b5[hdlC#`p_mTgGD"_G6tA="!\>$h]UI1&ptu"Dlt;1DgSBb>q2#C= +PPHGp"U'B6X3%UL;36Kk"32nj%V%)[>\u"2PW=7_+4!L*+j^^h%ieBT+):T:JS:eI +[Zr-p?D9dKjU`I5/O]YIqcY,2SWh"pl`NodB@H^0&*Er8E1h7Bh6r6gHB>1nAjqBB +;XO^ViU6,ZR>2UD`0E:npU%igMrh?+.2j)NR@leeJ"F>d/W +d6!sDBDa#@V]rX+%:*4iMVJhmgI:6[bF^.gH#96Y&4NcYc_SFgNMnug46X%$L*!h[ +71-j#&U9X1R:KI-Q\KQO7@SUcDRjQaSu5Vr!Rp_'$/P1UZ'cq+P]EoGoK$OY8lQ7_ +WUW;*[kqSi>7@CX^091(@i1r +rG4sob._[d^MHA.#frr=VsN0rAT3 +6L:OY's$p%Sa85gOkRmo<5@dIN<[R7:q%"qR-;$Phe1P8l`E_+5HulT>NPK?P9-#; +RnO[9))C,Y,<2/f32]<$"MFD>ce!N9XcZ;D'dDh&9F*aO2P'OYG0_8/,SPX[X_10m +NM2KH@"_cbO5qChPP-HpnkB_O3!Z\7,L\.At-"%&Fk4#Vjjae!"WD^AA^WFKga=Nl +]`[A"`K.1[1C6KJT=/-%s\*S`>0$%hsp"'S""RBkr.H)ZB8Kfr$;:BK(Ah7n5@.7a +$0jn*\XrhLUj^6Ye"9Bl.gsj8?\'^6Bd[7RO:ISS(M6M!c&Uk4rMl=2/?EU#?jc&+ +C**'nG0e;n+1[#e+0.BfM@-Ln6;N9h\+q&Ht.B>-QN#[l4m;%f)McU)+d-QR!fT +*5'cnDD0R#k#DKCKbV^BdO2/AT4t)*kW.nc/,@n?X-^Z(83i,:ej!->Ac1N':sKiC +4j;_YJ_N;)QMYA4277VOU,Nf9@TfBIGUf!.c3]"J3Tp!-UgoLBu`1\V*k3Y_Hhg=r +r>:)fAJdk!:Wi-g-;S.ighI@t?Q8s=_rLYF;X"[Pnahj.c+!9e[1kUa\).3LB. +BR?EUZiRROMc9cp`HsIJ2>MVVDi[>d%lu;9]j58FfY+KKB+Vq6&' +$#]_'IZXs_"8\4!V9BR5@F6U^!5nbCO5BVO/]+E*tR,-'3Js)PF6!:=$nIZ.,a0ln +@/JZ[tFU-rL`aiM8'BT:BgYZplB-7U1q+(/_>c#bq3aS3uKVo8Co66O)&hbcG3I&E +.cDaKL0Q(;'ttg\bT6E?Q\I-;V!rBU!%5PC0-imO4lH!(QS?+?rJAq6!:k2)n6bJF +R)16c&K0*"n./J%>9hu67;eO[("8rm7;,:>:\?Y +4Z^ef/B?Ybi6U"@99Z:^Pf;'I"C:L"2BVpjg*IcCMTjY$uuB2r\aRSIpo3hRtZog4 +!#8_F89:X5PC-9Q?44)p5lcXGGneri5nP;T"-O-J5l==V0\XVN.bmfJ- +_@_`&I$D+]\l?M+-E,bb'f'(>5oBrR^[l(#9>0kJ?o0JVd@V.T#!m +^ZGmLe*[!?Z%S`)!%g9*,eT$O#U/s@iD`^rc(E\lR6l2L`rN4%3SEJ'1"P'_lZ6Sl%;2 +rbXK[\tNCGbHkrqrlh!oH2A/2KT73cKup81mNebd[3u%P?kVJW*W_$ZRccucm%`;W^!0*6QWI,)rJN#$P6$i-GYnq]3*r!Kf0?-gVc8c'@Gi+UniQ"LYG5=F4GE/UA +.ghKjH*1%`b5IR5MF\TEAb_bYQ[ao1oKKrT&6Ylp(htoV/4ItWm168TJER)!fVVP\ +@D0XMi3j62,La(2kRIk#oN-u;uZi9>NgtVLK(d8iu$Sl4-6&i4+1cE\+&7**#G\uX +8PY+LL(+'(jY`_`XQg#-cNYEcY4AfeXV;2L0#rN!Rr+ZRcO)_N-_=hPHEPslZH#5` +'agMYT<].\+&"01`qPMFk)3%_p<7H)2:[mEtj9IA17Wnp$8%Hh!iXGFF@W`edY72l +V*I8rM=.o`:0:-Cr[#A`]B-fe*=[Kr&j9[@,UmL\^=a`m':CuPG8uPMHY/r"rZ#^&%rjhO%;itb:Y>*]RhsIVW;L"`_a9W@kX4E4o45fCWUBV"5[0f>[72,nF!-UBlT*LG@NIa;PSif4 ++_d9oj8O7d%2cWZr#NPX2C%P_ma1N$6A/gkP.U4ofl'sdXE]kLW=]Q*Xf<`]7L)V> +?fr0E_LJ6QtMQ="HJ]I_0e?60>YD\?7%^?LZH(ReF\9n27n."V^r@%!)&>nJoH%fW +*b"r*WhcE'5?`CgVg,YZ5u&ee`EAHn=I>Qrr^>Pha:iT5;R",E_cQ8`>0Q_(:+hiqA]tL +@9p*?1$?MD;"6e($N5AN3lq5Bd-b=[FCpjOKbsO?\@#'-f#bQk`P'gl7)OSg,J7E1/8)K!?SA"(P])b9;"orR$^r#h3Ckp>eP]F1q<9 +3#(lj>5$h,dS/O_1kE[A\]R5_&+N04ZhZd`#j^ML>0\Y4iXB$&-0=p&qCMjoNo1,# +?(kZr+5Bskf9C=@H/:45V3/hISAZ>n/J&`F;;3WHtr/WCU2[l@IX5eLMl=,,/)G4Z +n`,nkOPdRnA[QPXEZi#Sa`M]4A4[,>L*gbQ5MUg7!\e4H_IU[>$DLQ=>N5`O,10r7 +LWuTYu#b.%tAJ]!MO)oqr+0-ll5:07Z431j5lF5JGs#^F[+NR_`)5q*tBrH*tI0=R +1+-)9`(T(7<1W^c^kre]97XE?Y[:feHsM5fjiA*SI#'ih=ucb\&"kS7/ChZXCCoPIFBA)g\PdI,3VbCEN8YWWrV\%G`$?\a+C/MZ6X]c-*,0 +M=s7+,p8'f=V^Sc\_KPC\qBf<3hU)4J\+RJ]E)pDLZ3q5=/k"n;G7(mt>:;o#98-B +]lN,D3+oHrrBCWLL\<\_9X,riMf1_fbMSk"p;0_ +84E20=a'Z-5Nr0[qbdpJorL/m5Q7,Og$Dhc +%nh.6%aD1=68NTn4lLlaf"/P*p_Fe.:r#[_Kp^M)ZlfmrM?`QB^q%,SF-f,a7Q#8R +NSiDNablAfn[dqSt=rG]RSl7YI%Hpen*=USSo]WH$:$1DjD+#$abX6O>)!5?G5Cf# +^eQ#*05%!prgL"KO`:>blib94FgI+@.W,#l!?C0+5,OH]@81@T>Xq,U*O^C:#(Q'q +#O7,o(ic)#O=VeBqnAl>VCHndnu,UfJ&/)Cri +Lc#l-^AdhQCc[Z;KTRXS2+TtQ96\>)#icIJhrkTr&s!h5DQ\^A(%-<1j:^bO<)X0P +1`$s[:lj=0ImCN%n@un:XXbWh=rl-!!VoDoQOtuBWhQ\W'p%R][::&RrNra0t@IB5 +s7g&(%kA=htS],_!d4@ECFO#.!/,(E2(NV7^S6jlaQq?mL=V,h!%:M`QJGpJUg3]2 +3Fn`AdadLZsf3m,S<9`d;OcP8Yu3:_E4q0hH8Z#i&]$ab9GurD)E]!Di*t4XJPj2p +,U2o>T`WU`bYD.pil6=a'T&=`;]fB]N).[H'%3#[G(tJrNg=YB0ZF,#u]Ob`o-k.6$OaE7=DTXmC@! +J)7.A$#`\6/%'W(7sMa'O.iL0ZYPi$Kle^VB6OTZocB'EK1%Zf!"D.V;uCUX!riEN +2[eDB!>pU+(s=C`a2$)osl4Bj.ua$?D6kc%8,n[5K4:&E1VnWuE2;h9a)b@ZNgLG@_1-?(0:9Ge/% +bHZ%d?O,j@p][mC&OHBItYDHe*->JH)J:9=\1eZoH'4tjmd#bDf<5RCm%Ope:&-1l +UCE8Ru(!B:LVh^+cl#;j-S"n7q0"nrG#3*ar,kp41OU:6B?Tf!Kh6XkWHd16DGF?: +'n(TeV(G:B]X@AQ7Io_K$-lWdbm[D2p?R]aS\+,fsWZ]UrZ5Wp0%I6LJ*BrNV"5UCeH.2SP+7/.3c_@:V*2";"T@KB547iXIoB(k>%C+T6 +8!8gH2Qk.#>ho#AcDa$\Ap_iEE$S$9.j_;Bk/<-@r==!XdaQ9qHr)(P(VnZ5T0:th'Cj5)W:nVTStP5[5"+>@ +.pkF8F.9!IHoCYV&NRf@jm:*K$QCI^0n"+"9KlmKKg+#^,?a1-ccW6>SkV3/cq,p#5\pG@DFh_H2LB3>!)'AZ" +0k+@5>$6+!5L**RD:2RiTj;tU;Jd@D=1.TO1[[qQH.3`Uu)ibLF#4]L'R6F"hSHE7 +O=g1*E\NRVgp#54J[DDA)n`.B9s,AaEH`/T']"Bjm"aD_&(guNS-tH)\aY>i%R/2o +q2f'a$YjG@C8$%0u>R:q%l3iiji$&,\XXO':3#.@.5u;E +QrV%Fd;Be0)()sW5!V_QRBOO!8fh9Bt(@\lbm:mG8::)6-t2%>HkVN/Y;58Lor*NR +%[g,g.C>mj6mN?;pT"l^(Cb-T.g790p[YfkQtQiAA5B_,`a:OD +A*johC-\+`I0cWC=\fQ[]oc3nP'sT,Li2jrrBoZ,>">6k!^l7hHAQY&dMV!FJ"CSn;?,)g.gr-m`I=*C`4?Vqe!9Dp +;+5eofFQaDe4!<#,n41K94&igXB- +QNS@eP?^)f3U"F@J%k^kr(S4Y0drm%C'JR3QIfufBZ=Z/)s?q[5gDeeYbf'h=c3!: +L8=I^+Sp[NT9I3cZ8k-*[A0tKK`DKb(YXj,RYt"3j.HC`^n*=BEP;errC&&.(=#B@ +GY-^1$+Z6-9ei`7Jf:D[bG!_?\,A]ZoaRj`GFUG;E`X.B1r+d>Bk/SkP.F)41uhBS +@u&#pDcc3A:UK6Ol!M3^)2XirPGD*-[;URkSXkX$m,)iBj\8DM +*(q]Xmqaqctg);i5Ku<5%SEQ0Nf=sG?j0c!9Upun:0GYL#nHG;7b?Gj$,8MHoN1_f +/7'u4!hJ$a2aX8M6,i:[A0#@2,in6&)gkaGYe%1pikc+(A=3W=P^:l>M_#s,5U=_I +E8I>Gh"e@ +?(D1nsnai@\b*QYDU%-"G=bg>$m)mi&7(*#m"WZV%!EMqgQ`Bf>Ds[o@cEb/-d^X- +Y/?hi&Sblpj5j,1cgfShs'N>F9p\!68?e@lb?Pb:+^)p\HrYj:.q8:/uSU2V:M&-:GWi,#bi +umMBjZ<$_)=W;8l]OTmU1ZHsb!/NaQVg+iL]"RI,DJ_%h%@i,A^e:=^P_$=GHh5=Z&H;aH74dmZ6 +[bEgc9^#A#FMReMP7"uh>.YkeQ_D[>QN>5c[r^)i$`&j=)B/JfdnUMrdM?t6>ia +1Oe0GJQc@hg@SpebhLW^OMY[+cUQqQ0;qn:[PLJm0<8t4@(QkM0-el,8V-mB2RQ33 +EG=W"TV["c1*"0p3sHu;rYa0):kmsa[A%d4?cKoj`_0O+"06"-Q-A0W>3YPQf5-cN +D%5tCY#m(')2]/\cGj_mC1_=f\dr4"4uWl^Z,jY_nr#[FaXd"nV;`K0=;VJnmc:KVcL7XKs,Rjff\T(j3prAN[ +IV?Ec>h?grrDB8L_44+L/Pu_7a(f!:IA=b1%K,]('KVeBSc$urN5]hN)(?aLT,YA+\&M5oduE&L)uDo]J&o +rlX+N@GkXQW_fcSo!S'OR`-G1Yu;;W@rBVAH)W:\LmUJ5Tb5qX5f_9>*GFm;B)Z>N +WLcI`u#lJPC;uD$b1S]eSUTm];/jfgj9D'=R2d)njRUFD>sT!,p],PX7l;>[s"6X@ +.KD<&"A^Vcj/9n%u>TdWh6WqUpa]Foo +_[&C>VI:mYBp#J`BF:O]E>4A2cNEBY(XKX/%d'Z>>UUbQt_(*XsW?G/d5a`*tc12ohK5kW,%d +kp;7%;aDG/8/8%3hN6NIg]$eKF_j;o3C#Y$]2q*Ze%C/`FVcF5gbu.TqqmH)f`+Zc +if>TCMeA5olc>)"(rn-bo5H]:72Z,JB%:)2n'No'4'n0.^UgP&!_>!T2Y)>lV\f7g +`3Q/4[*6jNP<2)#%risFIs(K'jO@%>6m(6Q*-oW!;Z[*<-"B9A*^4b +Z:Yf0Ut^qiB;lsZ^L:E-V7u0oEpcL,O0"_X'l!MeOh9WQL_gQO/^0'!2f,241+O0; +]tHF5OOZdeXJ]ZVhJ]gArCU1/9//NNmugS8_"1&`#X$jV[BWq9hhNA?-(lTQH%G;Ro +VU@?c'^pG7mWcsrr>Xkf@%CV"k$%\"Iop]7(@5PIC +h7B_bO0p"+:LkgN9;Rj!&E\"380dCBUN!J0WQX:36l^1gjr``L#S#;N@G5dA<,G6! +[lZ4i5XG3T-Kt'He(;#08/RE=l%_7bpBo.ThYl&IdnYb,MpOZ),&J81rF320pVI*$ +e+mZ9g6haNqb4uhP^qH)<6f>p#_r/h1iBLg99$QQ8[M_Fb?2o41^X@CCmjMhT5D.6 +#j+X@tr$OV9R+A5 +T9_C\]<`1VpuFV(h8CnRb334#L:66+%cN(]LJJ77QPT^E@e8qd`hRg`reMjeG*h>O +,Ghh4udWtVqHij^NUn,/DWl"N1=tHMT##@Z_HcoZJ'%*(ah%`NWe:knAZf=Eac.#H4`bdNF^i8T`W4[Z]<-d-u<)4K0^%*D%CGX +F4bfYD;9^*Kf_H8X""udl?>H3!T.DdC$*] +Km8+pNhmcVjt3j@k4L^!/AT7PR^11Nd7Y9Iu%P'hNdfJ1?VfG(Qf$^",#+9. +Acea7F:\a'_nDJF]d>!i4c^_H-e&_10EV94&[LPr40r3[<2uhGa$$kD9^>.Z$gtKWC=LK<`-4%]?E?B.#_8 +8cgGc6M%<$NC+6mnKB9-8lXc#NYk`!Eg<[Lc6l"^!@QCWsBoim\;A(O;)1Pf+hi\Q +OWrp*iUM,S.+fjetHdd/6;0S??)QhlTc1:1hN3Hk7S'\.Q.FTjn6r9O_IZTY9DU3#IbtM_R'ik0+fP[Dl$]"k9Pa.J3!BEgmi4K[/@_2T +!Y-RQSM7CE^\4]m?f%"ULQ8IL::Rc5N4GnbgWJ6rrAXL]1[Jh^i+gAiEm:*lLX66p +6XKjU>3mM#bPeG,tAinI_6!fHnn%rOWleKcUrrUQA7B`5m*)G!1mu'nB\t.]OIo;( +$27jB1;DJRck^a>k)$`h%F`l:ZB^.!-q;qQ$%D2k.Eb\qS;0jFnAp\3QGa +Z@(4Obk8i18Gt]EVS._2seB?Qt$T2*R4L7EEpMgn@M]I2/2G895jI'A6Gd#Nq\Dh, +V,*!Kf`B=dt9n*XnK0r*GXHb9.ATCkt1bcAMHO0J.[>mb@KO^N]r^[`,EO%IBW'W;jr`M?bD+ +CfuB:U.+0Ht30V"S_F5Z#so^=C']VGA>4H`G]qHDBZN!1',235,0?kidNtkEZ5^EbUm6+fsj\G +;#DEOk;Up!-bTN`;uFO^E82!5ffO2G9l\m7\"b%&,J^6@ZkPC>Z>ZM4qlZ>&!K>[!WKf5Grf-p +m9<2'BAi@IPe%lG!"5:KRMtUlSo +qZpZe"HaQ\Jn:./o+rdrLCMIQK9]k[t7\"Pdom5!Us*BOi7q'm:@mTc%sXOWqR_:) +)Gp'bRon5%S.Yr=5S>U^qUCnG<=#L?SMAU#3J(o.t:+Z2ME`OIq@lQD6sblA,p,R- +j0':`;3d5XU.Xc\[P!LFcc!#>kG`4]"KWTZ4MP'-RqQ6Q[$F,Dhi^.he&J:3(Q,_5 +bUh\Y"7hAQ/]OoIq5ChSk8hn&OR,cYQ=o'#Kla$IG]crUQL4R@s02!9sS9]p\Gf"m +o-[,OEiI?Ju:=aQ"KmT\h>#d\)%+sniTWZ,WtYcUb4fG$@abM^Ob4Aa@Fe^`E]P[eA^ouj#Qc%D]k.[EtLqZX7j\C5)M`]U0Y$)iEp +CkO:Z7l;dh+@!!(#cPtCgP[@m;fa!,JoZ1JPhmBfG0dHgc"R+8?Gn@@s9[Q(YT/L) +;MVh6U$JI'-qM0Ze+?[j$4?O8@(i\F0o.U%Q8rlN]A,[a7a=H=?A.NTQ18EI>OiC@ +m72fCn&M)Y"A%WIX;EuDrLB\$JaK`NgpgeKl:k9!)7(W2R/X?*4GnX7L*\3p1rt22 +VURH.j(7>ELqca+Hk-8*(g_=0-@.m!#gFJ:c]gn8T.rr?6+V^uC_CSIZ@4&_crW3n7eXWt,&/3-N\UjCAlU=n;Sb +R?+^[!\mQ\;t]77nsj4Q]Bu1OQ"#`]6MC")T7H5IWL?GTU>-$)!X;uC!*lMY`h-_N +"A=_S=\O(p!!JB=e9D_-,G_)c1CH)!(tYKSSbK\://Ba&!f[b_">\!f!-FJV&H-N? +p6i'Y2PJ0m=EiqFV&F5OM^Ui#s@*"YeXbCNS3@EFJ&kR>.q>%`T5en1fKS&KD3-;j +SDenF[Xc%RFb8mj6Vs2N+8+5QLr3BNoH^pmbUOJp-lS=I#Nu9oju0Nk>Jr*C(nE3G +E+!F[^(V5Yl]Fl_isnb=guj*G-\RP1P^Q0dpIIkmoh4sGQ?a`/)%3#Cp +?UG.d]M];]CYK0/L:gEYc^q-lg=Jo"?/5;rOuBAE(C)SO=;Kl*.cigY=!.3p+lfZk +2SGsJO2?W!'9,8\'i/bf*Q^6;jJS%$.!!RAntIK)L]8bIOEYhJcu3I2p)[u1P!NUc +V2[h>rT_`%[D:Q+%o8J21ZU1)(lMi)iWs6F29(+,H/r%FIGhX_5,j5X"f0HnsG7Bc +&S8h36+YP[8VG3=feg@)JUWnYdY7bra[H?GN0n\/!b)uDmF+'l:e\k`uR#3m>*i]u@'@u^oaj +bI5lE][>enI,&HT>UN,A&&j5i[ZHgNn.>V0r-ME8Ef-rRCB*s(s(tj"1$O[oY2.9? +KN8:'3*Fu!OeK\$sdq/*OsEBZLmNT=P:diQS3F2_[Z^mEefd7kjL,Ur+2Xm( +"Ve0/',q1DfHJ$b`UBC/DG^Z)iX.aj#ApXT+#h6&U6ga;KuJ(n8)Mb-'d,n#"5mEL +2b&nIG"kukHAd[,*!+)f1GKDRdT/YC8%FQ7T,;YZ%+- +\cc"r4C3/h7%JU\Bj2AA$:8=d-e;5mX-C_SKKUb*,uFa)k?!P"(CSf].R35*`FG'5]C$Co0;8^ +q^>1\:#b,Ig(;P?4-3TQhJ*jHX/Z9:$ce"an#DUi%b#YE8$>1inXmJW3a.F\qo`r` +2Orbl3QZ/CZ^]3A&6lC`h*`GBO;3$o\U+&L!4S^4oUu'1!KoJ<#G]!"^=",HQ3sBn +.3AX)giLtCDN>rY7a-L27iU,P/4'l:dg`16sGKC7fWr--j.!-^LXMUr#+u5icoA@! +)i\GP`Kr"7SCFXZ,r"@?^Sj/OqWu`"+UPq*sm(NN?8QPeMTqk)daS0EpdPRd\MFP8 +`UU*@8pVqA#VK14Y(,_g3n>`CEWDcD/X%[W[4*G@2OEfK#9Ug7X4*RmPGI.1+d"RM +)U$Z[FFZh9.F/IqdFMl#/MDoS.\#m!L_l,gD`oXjcY,1GG=B6("2%n,J1B=a_opX[ +ptJ!,Tat=-chJdO +gCkR\Knl3($L)arLcWTE/W^+G +r3'o^?\;HlTc;uYO,cXmtTOj`83`JD=45XWbtpU2QL)FficBr5>j#U-hs=:rn;i9c +BkVW&&Q]g;Iu%f]%"-gQ]"\H[P9_eB8i?Lgn_:1C/8HJUgbX4+XSK9=+I(hj52_n%H;+78umZUN^a0hiNPI.8 +%itM%CE[PH`6j2DRR-0QVpO?A43/bn]o8R:65/q9d$r%o=0p5 +@&Al&[dk"@qQCqaAuUPYcBh8\+YI]4>M2'.G^4+_fc6prA.-so\7rSqESQHJ7N$!o +WH3jd)-W/*ceDAWgmr!A"F4ZJ1rR@oUOeqZL-S4_&n3.1O.k)[2.1@b8DIH\AJ@28 ++j!_[J3$;:_IdNDYr;$a3q6j2Eu.Eh"`q3?G`/kai*d8/UX$+20*Knf*/6Zn6B"R2B&JO +9R^C1-N=L"3UToIb$+?pV[//n/oP$*;H5OKk@iP^r6G$rZLCmNne0:4(c*_9kh""$ +e];6QBuTfa`e6j/1uO2mF_,?#lsr^G?*!BCouN1T&J1OQ4-$;q@?cl)T0Y(SG7j"8 +Bg0lWF:Iq`p\C^:]k7[%];!t/Rq?QaWtE$oEa?j'Q"J?/]+[%c`I1fSiZ?WLOLRS'\.=#K +[-JHacZ@"bPqP^5@c;%$sg:8KOK/n6-e8)#GA9+[D*'dAY1aFb:(SJT?auA:Jus<[ +mnteU5B>8P1`7.R6Pr:$4]8fI8:OXq;&3iEMdIThXUt\O[7fnN)i]QBX>_ff;Q$Li +F=uklfr9S5E0$UeVB2BhmM^2`fai19EKW-(%O<%c\8s6_gC!r:k@b^1"]>YEV5t81 +K:lgRTVI(E=#rgREo5PrYi;8eW'J]a(Sd>!p#8m59kIqkNY+ZV7!,&]q%GikbmMEn +F'/E'YKMHpar%&!F'LO-j-$kF?<]"At%l@jC?]WaS[bp2Y4D@l"Xc`[ie53W#kWf/ +t$8!q2)[!`nef4#O4>lX\dNS'&!.<]^\Sf_-Ato(uY?@".;*R>:.Jj-,__XrrBgD<(9!YH"#r?f +@Djk=Gt>6o:]Psq29tQqteiVB]!6/H +[Ts7=2\!"gaOd(IaOHoa5#hp_5uC^7e4u?>H'3ka,o$]C_,]t*l<=t1eS,1Q;P1u% +o;r+dXW^q[9KT^V3"o*R?&:P@QPS+e@If.jH_)E<(r7_fK-HQ53K=a>i&f@C(OP)7 +2;rZ&n":%5%ks^>k.2bF*P#hdNJh[fclUB!>aQE(`DN!4QkAVJ.h]q:=;,?Ih_ +C6+7*e9iS9'>jb1p'gPj&">!b@tTIm;4@KRi/!hX:K2E:L5Pgk*l"[8jb23o+j'Df +hunGoVa(HLK%fB`3@p":C=2X*HGWiiEli>N9AoF0mjN%.m_Y^;f1LL+l\OK>"!3Yi +WWi_YXf*,!D"5XOc&M.)'m]?o#55]=Cq%k6&&,n(*fg<\[M"pSuL^V3qD'2?Ufrm/ +tN%EN0"86N03Ah>u]h0_YQXTh$Dtnb+N4Fbg+6H3!>A-ae0,Qj5GKBg>Y[\h)QUEj +sK?r?SWBVjTu1V>r@CkoSX_.g;OZeMVOOcEgt^iGA\s%,T`PB=%U!pk*;=TcKb]5/Q=AA8$2&dR>$\Rrk?FZj^J"j#F.W^UJIXFWjh6n[fYZbp<,iS +TcPT1;u(FU_6/b#f6GX!(X-gCt#QRm&m]u@(:QIM!dq_TiKlJM'JO0GTOl.kh[etn +LfH0X8$kjZHa=4D_Gq.V6G2^5iXi,CJF:qnEnEUQkrZt0ehk>>G-o@YVlSQ3Pi``2 +:tD.]##RZE!HYgJ3PL`X(Q8mS*_cUi0/h&[&;kd/$-sFq%Bc$:0bBn,MV*>[9;:5R +'!aYEh$c[i&U&H6_nLdB8SOd\^GA)\VGONqm3nM.hKMm&4$Fq#.,2/R0-]ua]"URE +V6:n/`MT!6)DPMiN&liYde2"A!;KF+HcGV`:K^`BUo-qAjAB!N:GW)l/abVE?:_h[ +ogq'ZkuZ45UZ`^HoiC=W[\CWbFq,f,AB(!kY6B$KmjABqRf)8FO.U1OY6!%c$,VUG +kPGc4CiC@$D-''rr@h'-I8?"CN$"`Wtiq;0lC`6PedmHo04eEWATaTOmCZ"E_mQil.Lsq +aZ-J:\uI5ZWlqeP.EmOhPm:8[mGAShQdFkXh:pJmarpL:0KDtl.>3;1JTT7*Nt?Bf +_$'e)eR0Q17jkNj-oIENr%k0_]O`E>%R0?Ntt#n9QON`GfCgN\?c%4(cc7+6r,JOerr?e=5oajr>SMehh4HUf4.-D6.4r\ZcQiVYop,9(r +[N:UnBRpj_qe3$'LeuhDCc!u!1k-BfJ\3I!::#SUusc!?iZk:rg'-N^-,ti-JIGXS>9>`:*o]FD2LK5Te;aZPMsDP +Gn?)#:ZKsDXdCZoJpHG;7G#mC,7Z2?R,ND>E!uGM=Fs?+n4`hen(-V;oCRt%,Y[2^ +9-\;M%MhU:\P@AR]IV#D&^imYT8H^Z^BXU1iK"&^F +VHQ:l1^LUB$lrZbd"2hZc;0:)g6bJZ;*);M"'lNbtIbSp_=$gILu;/_d=$696P#dZ +nE6N>FTDb$Z>tE7: +%-EOD- +Q +Q +q +667.28 85.39 490 262.5 re +W* +q +q +[1 0 0 1 0 0] cm +0 0 595.28 841.89 re +W +Q +Q +Q +q +1 w +[] 0 d +0 J +0 j +[1 0 0 1 37 55.78] cm +q +[1 0 0 1 0 0] Tm +0 0 Td +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\004) +[6.036 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 413 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 419.67383 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.34766 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.34766 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 439.02148 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 449.01758 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 455.69141 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 462.36523 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 465.69922 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\001) +[3.624 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 469.69531 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 476.36914 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\006) +[6.096 +0] Tj +Q +Q +Q +Q +showpage +%%PageTrailer +pdfEndPage +%%Page: 3 3 +%%BeginPageSetup +%%PageOrientation: Portrait +pdfStartPage +0 0.706376 translate +0.9983 0.9983 scale +0 0 596 842 re W +%%EndPageSetup +[] 0 d +1 i +0 j +0 J +10 M +1 w +/DeviceGray {} cs +[0] sc +/DeviceGray {} CS +[0] SC +false op +false OP +{} settransfer +0 0 595.28 841.89 re +W +q +q +[1 0 0 1 0 0] cm +q +1 w +[] 0 d +0 J +0 j +[1 0 0 1 72 805.89] cm +q +[1 0 0 1 0 0] Tm +0 0 Td +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\015) +[2.256 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.33398 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.3418 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.01562 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.68359 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.35742 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.03125 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.70508 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 53.37891 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.375 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.04883 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.04492 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.71094 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 77.37891 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 83.37891 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.05273 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.72656 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.39258 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.06641 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.74023 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.4082 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.07422 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.74805 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.41406 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.08008 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.08789 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.0957 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.76953 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.76562 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.76562 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.77344 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.76953 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 203.44336 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.11719 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.11328 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.7793 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.7793 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.44727 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.11328 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.78711 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.11719 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.79102 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.46484 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.13867 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.80664 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.80273 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.46875 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 297.46875 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.14258 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 313.47656 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 319.47656 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.14258 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 325.47656 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.15039 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.1582 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 345.49219 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.16602 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.83203 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 357.49805 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.83203 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 373.50586 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.17969 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.8457 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.8457 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.8418 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 405.51562 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 415.52344 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.19727 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.86328 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.53711 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.21094 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.88477 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 451.55859 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.23242 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 471.5625 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 478.23633 -9.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.66602 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.68164 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 35.35547 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.02148 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.02148 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.6875 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 53.35547 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\015) +[2.256 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.68945 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.36328 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.69727 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.37109 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.04492 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.71875 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.04883 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.04883 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.71484 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.71484 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 115.38867 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.38477 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 123.38086 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.38867 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.0625 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.72852 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.72852 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.72852 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 161.40234 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.07031 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.74414 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 181.41797 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.08398 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.0918 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.76562 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207.43945 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.10547 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.10547 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.77148 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 225.44531 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.11914 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 239.44922 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.12305 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.79688 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.46289 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 261.46289 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.12891 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.80273 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 277.47656 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.81055 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 287.47852 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 295.48242 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 299.47852 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.15234 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.81836 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.82617 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.82227 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 335.49609 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.16211 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.83594 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.83594 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.16992 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.83594 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.84375 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.17773 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.85156 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.52539 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 389.52539 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 392.19141 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.86523 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 405.53906 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.87305 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.86914 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.86914 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.86914 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 443.54297 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.2168 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 456.89062 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 463.56445 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 470.23828 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 473.57227 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 476.23828 -24.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.33398 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 10.00781 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 14.00391 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.67383 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 37.34766 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.68164 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 47.35547 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.35156 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.01953 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.02344 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.69727 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.37109 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 91.37109 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.04492 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.71094 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 103.37695 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.05078 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.38477 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 123.38086 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.05469 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.38867 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.38477 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.05859 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.05469 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.72266 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\037) +[1.836 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.38867 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.0625 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.0625 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.39648 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.4043 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.4043 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.07812 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.08594 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.08594 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.75977 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 227.42578 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.09961 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.77344 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.10742 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.78125 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.11523 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.12305 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.78906 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 273.46289 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.46289 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.12891 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.80273 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.46875 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.14258 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.81055 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 307.47656 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.15039 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.15039 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.82422 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.1543 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.82812 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.49414 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.16797 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.49805 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.83203 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 379.50586 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 389.50195 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.17578 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.84961 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.18359 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.85742 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 421.52344 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 427.52344 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.18945 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.85742 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 443.53125 -39.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.01367 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.00977 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.68359 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.69141 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.6875 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.36133 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.36133 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 77.36133 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 87.36914 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.04297 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.7168 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 107.39062 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.06445 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.07227 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.07227 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.73828 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.07227 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.74609 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.75391 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.75 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.42383 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 175.42383 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 181.42383 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.09766 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.76562 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.76953 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.44336 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 215.44336 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.11719 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.79102 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.46484 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.13867 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.80469 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.80469 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.80469 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.8125 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 273.48633 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.16016 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 283.49414 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.16797 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.8418 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.17578 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.8418 -54.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.66602 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.00586 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.67188 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.67969 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.01367 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.6875 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.6875 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.35352 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.01953 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.68555 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.68555 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.35156 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.01953 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\012) +[8.028 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.68555 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.35938 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.0332 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.70703 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.04102 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.03711 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.71094 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.37695 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.04297 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.70898 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.04297 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.04297 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.7168 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 161.39062 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.06445 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.73047 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.72656 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 187.40039 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.07422 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 197.4082 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.08203 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.41211 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 227.4082 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.08203 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.75586 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.42969 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.4375 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.4375 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.10352 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.77148 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.44531 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.44141 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.11523 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.7832 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.7832 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 315.45703 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.13086 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.13086 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.80469 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.80469 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.13867 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.8125 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.14648 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.82031 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.1543 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.82227 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.82617 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 385.5 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 395.50781 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 401.50781 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.18164 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.17773 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.17773 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.85156 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.18555 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 437.51953 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.18555 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.85938 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.85938 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 455.52539 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 462.19922 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 468.87305 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 475.54688 -84.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.00391 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.67773 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.3457 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.3457 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.01953 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.01562 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.68164 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.35547 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.35547 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.02344 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.68945 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 93.36328 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.35938 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.0332 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.70703 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 113.37305 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.70703 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 123.38086 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.71094 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.37695 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.05078 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.38477 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.05859 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.05469 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.72852 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 167.40234 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.73242 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.39844 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.07227 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.07227 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.74609 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.08008 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.08008 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.74609 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 227.41406 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.08789 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.08398 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.75781 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.42578 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 267.42188 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.0957 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.76953 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.76562 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 287.43164 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.76562 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.09961 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.77344 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.77344 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 315.43945 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.10547 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.77148 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.77148 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 329.4375 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.77148 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.76758 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 355.44141 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.77539 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.10938 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.77539 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.77539 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.44336 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 387.43945 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.11328 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.78711 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.7832 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 407.44922 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.7832 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.45703 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.13086 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.79688 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.79688 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 441.46289 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.13672 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.81055 -99.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 2.66602 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.01367 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.34766 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.02148 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.69531 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 47.36133 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.69531 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.0293 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.70312 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.69922 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 77.37305 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.03906 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 89.37305 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.04688 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.71289 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 105.38672 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.05273 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 111.38672 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.06055 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.73438 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.74219 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.41602 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.08984 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.75586 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.08984 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.76367 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.42969 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.76367 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.75977 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.42578 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 185.42578 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.09961 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 201.43359 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.42969 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.10352 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.77734 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.77344 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.43945 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.77344 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.44727 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.11328 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.78711 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.46094 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.12695 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.12695 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.12695 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.79297 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.78906 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 297.45703 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\013) +[7.356 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 305.46094 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.79492 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 311.46094 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.13477 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.46484 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.13867 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.8125 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.14648 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 357.48047 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.1543 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.82031 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.48633 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.16016 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 385.49414 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 392.16797 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.8418 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 401.50781 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.18164 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.84961 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.8457 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.51953 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 435.51562 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.18945 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.18945 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.86328 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 461.53711 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 464.20312 -114.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.99609 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.66992 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.66992 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.66992 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.67773 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.35156 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.01953 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.01562 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.68945 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.68945 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.68945 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.69727 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.69336 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 107.36719 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.70117 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.03516 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.70117 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.03516 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 129.36914 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.04297 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.7168 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.7168 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 151.38281 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 158.05664 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.73047 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.73047 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 177.39844 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.06445 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.73828 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.41211 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.74609 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.41211 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.08594 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.08594 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.75977 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 233.42578 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.0918 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.42578 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 251.42578 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.09961 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.77344 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.44727 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.12109 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.79492 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.12891 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.79492 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.12891 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.80273 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 313.47656 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.15039 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 333.48047 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.1543 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.82227 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 353.49609 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.16992 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.84375 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 373.51758 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.51367 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.1875 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.18359 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.84961 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.18359 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.85156 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\012) +[8.028 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.51758 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.19141 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.86523 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 429.53906 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.87305 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 445.54688 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.2207 -129.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.6875 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.01758 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.69141 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 43.36523 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.03125 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.69922 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 59.37305 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.70703 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.70703 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.38086 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.05469 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.72852 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.39453 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.06836 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 113.40234 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.07617 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.75 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.42383 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.41992 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.08789 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 158.75391 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.75 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.42383 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.75781 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.0918 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.76562 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.76562 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 197.43164 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.09766 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.76367 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.76367 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.42969 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.76367 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 227.4375 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.10352 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.77734 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.45117 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.125 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.79102 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.45898 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.45508 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.12891 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.12891 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.12891 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.80273 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.4707 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 307.4707 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.14453 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.15234 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.82617 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 337.5 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.17383 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.84766 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.84375 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 361.51758 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 365.51367 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.17969 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 374.84766 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.51367 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.1875 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.19531 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.86133 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.53516 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.20898 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.20508 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.87891 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 427.55273 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.2207 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.2207 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.88672 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 449.55469 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 456.22852 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 466.22461 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 472.89844 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 476.23242 -144.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\022) +[8.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33398 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.00781 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.67383 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.34766 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.02148 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.0293 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.0293 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.70312 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.36914 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.04297 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.37695 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.05078 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.72461 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 93.39258 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.06641 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.0625 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.73633 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.73633 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.40234 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.07617 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.08398 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.75781 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 145.42383 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.08984 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.76367 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 161.4375 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.11133 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.10742 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.77539 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.7793 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.45312 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.46094 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.12695 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.80078 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.80078 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.47461 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.14062 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.80664 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.14062 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.13672 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.81055 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.14453 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.14062 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.81445 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.14453 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 287.47852 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.15234 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.14844 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.82227 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 307.48828 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 313.48828 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.15625 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.83008 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.16406 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 339.49805 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.17188 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.8457 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.8457 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 361.51172 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.18555 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 374.85938 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.19336 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.86719 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 397.54102 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.20703 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.87305 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.88086 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.88086 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 425.55469 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.55469 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 437.55469 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.2207 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.89453 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 449.56055 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 456.22852 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 462.22852 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 464.89453 -159.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.66992 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 23.34375 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.67773 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.3457 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.6875 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 47.36133 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 53.36133 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.69531 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 59.36133 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.03516 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.70898 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.375 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.04883 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 95.37891 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.05273 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.72656 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 115.40039 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.06836 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 125.40234 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.06836 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.74219 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.74219 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.4082 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.08203 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.75586 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.42969 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.09766 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.77148 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.44531 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.11133 -174.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\012) +[8.028 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.66602 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.00586 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 27.33984 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.33594 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.00977 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.34375 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.33984 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.01367 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.00977 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.67773 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\037) +[1.836 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.34375 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.01758 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.01758 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 87.35156 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.35938 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.0332 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.70703 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.71484 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 123.38086 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.05469 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.38867 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.0625 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.05859 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.73242 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 157.40625 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.73633 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 177.41016 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.07617 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.75 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 193.42383 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.09766 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.76367 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.09766 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.76562 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 219.43945 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.10547 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.10547 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.10547 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.7793 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.45312 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.12695 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 267.45703 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.13086 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.80469 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 283.4707 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.4707 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.13672 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.81055 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 305.48438 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.81445 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.14844 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.82227 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 325.48828 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.1543 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.82812 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.82812 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.49609 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.16211 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.16992 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 363.50391 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.17773 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.17383 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.84766 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 393.52148 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.85547 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.5293 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.19531 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.19531 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.19531 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.86914 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 433.53516 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 439.53516 -204.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 21.33984 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.01367 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.00977 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.67578 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.67578 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 47.34961 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.02344 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.03125 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.69727 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 69.36328 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.03711 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.71094 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.37695 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.05078 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.71875 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.72266 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.71875 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.39258 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.05859 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.06641 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.73242 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 139.40625 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 145.40625 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.74023 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 155.41406 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.74805 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.74805 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 177.42188 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.0957 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.76953 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 193.43555 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.10938 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.44336 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 215.44336 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.11719 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.11719 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.11719 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.7832 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.45703 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.12305 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 249.45703 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.125 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.12891 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.80273 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.81055 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 287.48438 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.15234 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.82617 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 307.5 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 311.49609 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.16992 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 327.50391 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.17773 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.85156 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.52539 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 357.52148 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.18945 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.19336 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.86719 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 381.5332 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.19922 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.87305 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 397.54688 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.88086 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 407.55469 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 413.55469 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.22852 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.90234 -219.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.34766 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.02148 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.69531 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.36914 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.70312 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.36914 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.70312 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.37695 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.05078 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.04688 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.7207 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.05469 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.72852 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 101.39648 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.07031 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.06641 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.06641 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.06641 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.06641 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.74023 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 139.40625 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.08008 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.41406 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.08789 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.76172 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.42969 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 175.42969 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.10352 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.77734 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.45117 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.11719 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.79102 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.125 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.79102 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.46484 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.46484 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.79883 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 239.47266 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.80664 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.48047 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.47656 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.15039 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.1582 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.83203 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.50586 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 295.50586 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.83984 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 305.51367 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.84766 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 315.51562 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 325.51172 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.18555 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.85938 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.85938 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 351.5332 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.20703 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.88086 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 374.21484 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.21484 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.88867 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.88477 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.88477 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.55859 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.55859 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.22656 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.22266 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.88867 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.88867 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 435.5625 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.89648 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 451.57031 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.24414 -234.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.66992 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.66992 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 29.34375 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.67773 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.67773 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 51.35156 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.35156 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.68555 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.35156 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.02539 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.69922 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 79.36523 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.03906 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.03516 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.70312 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 109.37695 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.05078 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.72461 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 129.39844 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 139.40625 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.08008 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.75391 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 155.41992 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.75 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 171.41602 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.08984 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.08984 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.75586 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 193.42969 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.0957 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.10352 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.77734 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 219.45117 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.11719 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.78516 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.78516 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 237.45117 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.11914 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.79297 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.78906 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 267.46289 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.13086 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 277.46484 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.13867 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.13867 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.80469 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 295.4707 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.13672 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.13672 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.80273 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.13672 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.81055 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 325.47656 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.47656 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 337.47656 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.15039 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.14648 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.14648 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.14648 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.82031 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 373.48828 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.16211 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.82812 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 389.50195 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 399.49805 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.16602 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 413.50781 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.18164 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.18164 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 429.51562 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.18164 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.85547 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 445.5293 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.19531 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.86914 -249.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.34766 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.67773 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.01172 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.68555 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.68164 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.01562 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.68945 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.01953 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.69336 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.70117 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 89.36719 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.0332 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.70703 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 105.38086 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.04688 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.05469 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.72852 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.72461 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 135.39844 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.39844 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.06445 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.73828 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.74609 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.08008 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.75391 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.75 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 187.42383 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.09766 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.09766 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.76562 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.76953 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 221.44336 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 227.44336 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.11719 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.79102 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.46484 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.13867 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.80469 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.80469 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.80469 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.8125 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 285.48633 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.16016 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 295.49414 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.16797 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.8418 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.17578 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.8418 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.50977 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.17578 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.84961 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.8457 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.51953 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.19336 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.85938 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.19336 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.86719 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 374.19727 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.87109 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.87109 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 392.87109 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 399.54492 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.54102 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 415.54102 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.21484 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.22266 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.89648 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 441.5625 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.22852 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.90234 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 457.57617 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 464.25 -264.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.6875 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.02148 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.68945 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 35.35547 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.36328 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.69727 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 51.36328 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.03711 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.03711 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.70312 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.37695 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.05078 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.72461 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 93.39258 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.06641 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.0625 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.0625 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.07031 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.74414 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.74023 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.41406 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.74805 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.41406 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.08789 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.41797 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.41797 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.08398 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.75195 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 185.42578 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.42188 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.0957 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 205.42969 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.09766 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\022) +[8.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 221.43164 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.10547 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.77148 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.77148 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.44531 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.11914 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.12695 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.12305 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.12305 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.79688 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.79297 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 293.4668 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 299.4668 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.13477 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.80859 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 315.47461 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.47461 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.80859 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.48242 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.8125 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.8125 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 357.48633 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 363.48633 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.82031 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.81641 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 373.48242 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.15625 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.83008 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 389.49609 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 392.16211 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.82812 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.83594 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 411.50977 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.17578 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 417.50977 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 421.50586 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.17188 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.17188 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.8457 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.8457 -279.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.34766 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.34766 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.02148 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.02148 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.35547 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.0293 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 51.36328 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.03711 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.36719 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.0332 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.04102 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.71484 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 91.38867 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.05469 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.72852 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 107.39648 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.39258 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.06641 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.0625 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.73633 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.73633 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.73633 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 153.41016 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 157.40625 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.40625 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.08008 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.41406 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 185.41406 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.08789 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.08789 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.08789 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.75391 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 213.42773 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.09375 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.76172 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.76172 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.43555 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.43164 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.42773 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.10156 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.77539 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 275.44922 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.11719 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\024) +[7.704 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.7832 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 297.45117 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.125 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.79102 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 313.46484 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 323.46094 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.13477 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.80859 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.14258 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.81641 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.8125 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 363.48047 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.14648 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.8125 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 375.48633 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.16016 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.15625 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.16406 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.16406 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.83789 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.83789 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 421.51172 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.18555 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.85938 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 437.52539 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.19922 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.87305 -294.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.00977 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.68359 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.35742 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.69141 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.36523 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 59.36133 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.0293 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\037) +[1.836 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.69531 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.36914 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.36914 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.70312 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.71094 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.70703 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 111.38086 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.38086 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 123.38086 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.38867 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.05469 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.72852 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.40234 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.07617 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.07227 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.40625 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.07227 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 175.40625 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.08008 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.07617 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.75 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.42383 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.0918 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.76562 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 219.43945 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 225.43945 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.11328 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.78711 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.7832 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.79102 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.125 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.12109 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.78711 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.46094 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.13477 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.80078 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 281.4668 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.13281 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.14062 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.81445 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 307.48828 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.1543 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.16211 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.83594 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 333.50977 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.17578 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.8418 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.84961 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.84961 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 357.51562 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.84961 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 367.52344 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.53125 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 387.52734 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.20117 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 397.53516 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.20898 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.20898 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.87695 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.21875 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.89258 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.89258 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.22656 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.89258 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 449.56641 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 456.24023 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.90625 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 465.58008 -309.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.34766 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.01367 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.67969 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 37.35352 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.02734 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.02344 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.03125 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.70508 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.37891 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.38672 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.05273 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.71875 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 93.39258 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.06641 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.0625 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.07031 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.4043 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.07812 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.75195 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.75195 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 139.41797 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.0918 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.76562 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.09961 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.77344 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 175.44727 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.11328 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.78711 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 187.45312 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 193.45312 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.45312 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.11914 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.79297 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.80078 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 225.47461 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.14258 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.81641 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.49023 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.15625 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.49023 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.82422 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 267.49805 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.16406 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.83008 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.83008 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 285.49805 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.16406 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.83789 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.16797 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.16797 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.8418 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.8418 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 335.50781 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.17383 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.50781 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.18164 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.85547 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.18945 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.86328 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.53711 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.21094 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.21875 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.89258 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 407.56641 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.24023 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.91406 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.24805 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.91406 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 433.58789 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.26172 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 449.5957 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 456.26953 -324.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.99609 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.66992 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 23.34375 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.01758 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.68359 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.01758 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.69141 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 51.35742 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.69141 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 67.36523 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.03906 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.03516 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.36914 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.04297 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.03906 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.70508 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 101.37891 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.05273 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.7207 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.7168 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 131.39062 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.06445 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.39844 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.07227 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.07227 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.74023 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 167.41406 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.08789 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.08789 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.08789 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.76172 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.0957 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.76953 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.09961 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.0957 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.76172 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.43555 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.10156 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.10156 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.77539 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 241.44141 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.11523 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.44922 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 267.44531 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.11914 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.11914 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.78711 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.79102 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.46484 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.13867 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.13867 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.8125 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 323.47852 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.14453 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.81836 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.15234 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.82617 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 355.5 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.17383 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.83984 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 371.51367 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.1875 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 387.52148 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.19531 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.86914 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.53516 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.20117 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.875 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 419.54883 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.88281 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 429.55664 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 435.55664 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.23047 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.9043 -339.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.33398 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 10.00781 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.00391 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.67773 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.35156 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.68555 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 43.35352 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.01953 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.69336 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.02344 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.69727 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.37109 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.04492 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.71289 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.04688 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.7207 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.7168 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.05078 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.72461 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.05469 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.72852 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.39648 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.07031 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.74414 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.41016 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.08398 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.41406 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.74805 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.74414 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.41016 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.41016 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.74414 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.41016 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.08398 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.75781 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.76562 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.76562 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.43945 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.10547 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.7793 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.45312 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.78711 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.46094 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.79492 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.80273 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.80273 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.46875 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.80273 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.47656 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 299.48438 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.1582 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.83203 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 319.50586 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.17383 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.84766 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 339.52148 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.19531 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.19531 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.86328 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\014) +[6.78 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.19336 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.86719 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.86719 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.86719 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.875 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.875 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 407.54883 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 413.54883 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.54492 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 429.54492 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.21875 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 445.55273 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 455.54883 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 462.22266 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 468.22266 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 474.22266 -354.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.66992 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.33594 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.00977 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 35.34375 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.01758 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.68359 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.68359 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.01758 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.69141 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.02148 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.69531 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 87.36914 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.37695 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.05078 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.71875 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 113.38477 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.05078 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.72461 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 129.39844 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.39453 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.06836 -369.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\010) +[8.016 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.00391 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 14.67773 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 21.35156 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.02539 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.69922 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.70703 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 51.38086 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.04688 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.71289 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.38672 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.38281 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 79.38281 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.05664 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.05273 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.72656 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 103.40039 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.73047 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.4043 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 121.40039 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.40039 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.4082 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.07422 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.08203 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 153.41602 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.08984 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.75586 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 165.42188 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.75586 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.08984 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.75586 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 187.42969 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 193.42969 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.0957 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.76953 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.44336 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.11719 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.78516 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.45898 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.12695 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.12695 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.80078 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.79688 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.46289 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.13672 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.4707 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.14453 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.81836 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.49219 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.16602 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.17383 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.84766 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.52148 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.1875 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.1875 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.18359 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.85742 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 353.53125 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.19922 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.86523 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.86133 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 379.53516 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.86914 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 392.86523 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 399.53906 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.87305 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.86914 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 413.54297 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.53906 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.20703 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.87305 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 439.54688 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 445.54688 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.2207 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 461.55469 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 468.22852 -399.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.6875 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.35352 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.35352 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 37.35352 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.01953 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 53.34961 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.02344 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.69727 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.69727 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.03125 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.70508 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.03906 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.70703 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 95.37305 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.04688 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.04688 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.7207 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.05469 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.7207 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.39453 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 139.39453 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.06836 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.39844 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.07227 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.74609 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.41992 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.09375 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.08984 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.76367 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.75977 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 203.42578 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.09375 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.75977 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 219.43359 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.10742 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.77539 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 239.44922 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.7832 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.7832 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 261.45703 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.13086 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.80469 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 281.47852 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.15234 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.48633 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.15234 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 303.48633 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.16016 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.82617 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 319.5 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 329.50781 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.17383 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.84766 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.84375 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.51758 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.84766 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 365.51367 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 375.52148 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 379.51758 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.18359 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.18359 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.85742 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.85742 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 407.52539 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\013) +[7.356 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 415.5293 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.86328 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 421.5293 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.20312 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 441.5332 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 447.5332 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.20703 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.88086 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 467.55469 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 474.22852 -414.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.33398 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 10.00781 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.68164 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.68164 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.34766 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.02148 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.69531 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.0293 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.70312 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.69922 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.37305 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.37305 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.04102 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.71484 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 91.38867 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.05469 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 103.38867 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.05469 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.72852 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.72852 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 121.39453 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.06836 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.73438 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.74219 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 147.41602 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.08984 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.76367 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 167.4375 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 177.44531 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.44531 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.11914 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.79297 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.79297 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.4668 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.14062 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.81445 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.48828 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.15625 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.83008 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.83008 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.49805 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\013) +[7.356 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.50195 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.83594 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.50195 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.17578 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.50586 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.17188 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.8457 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.8418 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 309.51562 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.8457 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 329.51953 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.19336 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 339.52734 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.20117 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.86914 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 359.54297 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.20898 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.20898 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 371.54297 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.2168 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 391.54688 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.2207 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 407.55469 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 413.55469 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.22852 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.22461 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.2207 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.89453 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 453.56836 -429.73633] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +Q +Q +q +72 85.39 490 262.5 re +W* +q +q +[1 0 0 1 0 0] cm +0 0 595.28 841.89 re +W +q +/DeviceRGB {} cs +[0 1 0] sc +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +1 w +[] 0 d +0 J +0 j +[1 0 0 1 162 201.89] cm +0 -30 288 30 re +f* +0 -30 288 30 re +S +Q +q +/DeviceRGB {} cs +[1 0 0] sc +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +1 w +[] 0 d +0 J +0 j +[1 0 0 1 162 129.89] cm +0 -30 360 30 re +f* +0 -30 360 30 re +S +Q +q +/DeviceRGB {} cs +[0.8863 0 0.4706] sc +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +1 w +[] 0 d +0 J +0 j +[1 0 0 1 162 309.89] cm +0 -30 180 30 re +f* +0 -30 180 30 re +S +Q +q +/DeviceRGB {} cs +[0 0 1] sc +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +1 w +[] 0 d +0 J +0 j +[1 0 0 1 162 273.89] cm +0 -30 216 30 re +f* +0 -30 216 30 re +S +Q +q +/DeviceRGB {} cs +[0 0.6196 0.8784] sc +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +1 w +[] 0 d +0 J +0 j +[1 0 0 1 162 237.89] cm +0 -30 252 30 re +f* +0 -30 252 30 re +S +Q +q +/DeviceRGB {} cs +[1 0.9255 0] sc +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +1 w +[] 0 d +0 J +0 j +[1 0 0 1 162 165.89] cm +0 -30 324 30 re +f* +0 -30 324 30 re +S +Q +q +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +1 w +[] 0 d +0 J +0 j +[1 0 0 1 144 325.89] cm +[0 -1 1 0 0 0] cm +0 0 m +240 0 l +S +Q +q +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +1 w +[] 0 d +0 J +0 j +[1 0 0 1 144 85.89] cm +0 0 m +400 0 l +S +Q +q +1 w +[] 0 d +0 J +0 j +[1 0 0 1 72 301.89] cm +q +[1 0 0 1 0 0] Tm +0 0 Td +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 1.28271 -11.7373] Tm +0 0 Td +/F35_0 15 Tf +(\003) +[11.43 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.77783 -11.7373] Tm +0 0 Td +/F35_0 15 Tf +(\007) +[7.83 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.12012 -11.7373] Tm +0 0 Td +/F35_0 15 Tf +(\013) +[8.205 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.28271 -11.7373] Tm +0 0 Td +/F35_0 15 Tf +(\011) +[7.785 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.625 -11.7373] Tm +0 0 Td +/F35_0 15 Tf +(\017) +[8.145 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.7876 -11.7373] Tm +0 0 Td +/F35_0 15 Tf +(\023) +[4.815 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 53.78271 -11.7373] Tm +0 0 Td +/F35_0 15 Tf +(\007) +[7.83 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 29.62012 -47.7373] Tm +0 0 Td +/F35_0 15 Tf +(\000) +[10.095 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.45264 -47.7373] Tm +0 0 Td +/F35_0 15 Tf +(\016) +[3.135 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.62012 -47.7373] Tm +0 0 Td +/F35_0 15 Tf +(\024) +[8.115 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 53.78271 -47.7373] Tm +0 0 Td +/F35_0 15 Tf +(\011) +[7.785 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.44531 -83.7373] Tm +0 0 Td +/F35_0 15 Tf +(\001) +[10.065 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.27783 -83.7373] Tm +0 0 Td +/F35_0 15 Tf +(\026) +[8.1 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.62012 -83.7373] Tm +0 0 Td +/F35_0 15 Tf +(\007) +[7.83 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.9624 -83.7373] Tm +0 0 Td +/F35_0 15 Tf +(\017) +[8.145 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.77295 -119.7373] Tm +0 0 Td +/F35_0 15 Tf +(\002) +[10.755 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.44043 -119.7373] Tm +0 0 Td +/F35_0 15 Tf +(\022) +[6.03 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.27783 -119.7373] Tm +0 0 Td +/F35_0 15 Tf +(\011) +[7.785 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.62012 -119.7373] Tm +0 0 Td +/F35_0 15 Tf +(\011) +[7.785 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.9624 -119.7373] Tm +0 0 Td +/F35_0 15 Tf +(\017) +[8.145 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.44043 -155.7373] Tm +0 0 Td +/F35_0 15 Tf +(\006) +[10.02 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.61768 -155.7373] Tm +0 0 Td +/F35_0 15 Tf +(\011) +[7.785 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.95996 -155.7373] Tm +0 0 Td +/F35_0 15 Tf +(\016) +[3.135 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 37.12744 -155.7373] Tm +0 0 Td +/F35_0 15 Tf +(\016) +[3.135 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.29492 -155.7373] Tm +0 0 Td +/F35_0 15 Tf +(\020) +[8.625 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.45752 -155.7373] Tm +0 0 Td +/F35_0 15 Tf +(\025) +[11.655 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.7876 -191.7373] Tm +0 0 Td +/F35_0 15 Tf +(\004) +[10.755 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.62012 -191.7373] Tm +0 0 Td +/F35_0 15 Tf +(\011) +[7.785 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.9624 -191.7373] Tm +0 0 Td +/F35_0 15 Tf +(\010) +[8.205 +0] Tj +Q +Q +q +1 w +[] 0 d +0 J +0 j +[1 0 0 1 72 347.89] cm +q +[1 0 0 1 0 0] Tm +0 0 Td +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.26123 -17.0166] Tm +0 0 Td +/F35_0 22 Tf +(\005) +[12.98 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.30225 -17.0166] Tm +0 0 Td +/F35_0 22 Tf +(\015) +[4.598 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.41455 -17.0166] Tm +0 0 Td +/F35_0 22 Tf +(\023) +[7.062 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.74072 -17.0166] Tm +0 0 Td +/F35_0 22 Tf +(\016) +[4.598 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 205.85303 -17.0166] Tm +0 0 Td +/F35_0 22 Tf +(\011) +[11.418 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.20068 -17.0166] Tm +0 0 Td +/F35_0 22 Tf +(\020) +[12.65 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 237.63916 -17.0166] Tm +0 0 Td +/F35_0 22 Tf +(\012) +[7.964 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 251.07764 -17.0166] Tm +0 0 Td +/F35_0 22 Tf +(\002) +[15.774 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.18994 -17.0166] Tm +0 0 Td +/F35_0 22 Tf +(\022) +[8.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.75146 -17.0166] Tm +0 0 Td +/F35_0 22 Tf +(\007) +[11.484 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.98682 -17.0166] Tm +0 0 Td +/F35_0 22 Tf +(\021) +[12.628 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.42529 -17.0166] Tm +0 0 Td +/F35_0 22 Tf +(\014) +[11.946 +0] Tj +Q +Q +Q +Q +Q +q +1 w +[] 0 d +0 J +0 j +[1 0 0 1 71.72 54.78] cm +q +[1 0 0 1 0 0] Tm +0 0 Td +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.34766 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.02148 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.01758 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.69141 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.36523 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.69922 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\001) +[3.624 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.69531 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.36914 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\006) +[6.096 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 476 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\005) +[6.132 +0] Tj +Q +Q +Q +Q +showpage +%%PageTrailer +pdfEndPage +%%Page: 4 4 +%%BeginPageSetup +%%PageOrientation: Portrait +pdfStartPage +0 0.706376 translate +0.9983 0.9983 scale +0 0 596 842 re W +%%EndPageSetup +[] 0 d +1 i +0 j +0 J +10 M +1 w +/DeviceGray {} cs +[0] sc +/DeviceGray {} CS +[0] SC +false op +false OP +{} settransfer +0 0 595.28 841.89 re +W +q +q +[1 0 0 1 0 0] cm +q +1 w +[] 0 d +0 J +0 j +[1 0 0 1 36 805.89] cm +q +[1 0 0 1 0 0] Tm +0 0 Td +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.02148 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.68945 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.36328 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.0293 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.69531 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.36914 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 55.36523 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.36523 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.03906 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.03516 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.70898 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.38281 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.71289 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.38672 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.05469 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.7207 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 115.39453 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 121.39453 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.72852 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 131.40234 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.40234 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.07031 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\022) +[8.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 153.4043 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.07812 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.74414 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.74414 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 175.41797 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.0918 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.09961 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.09961 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.77344 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.44727 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.11328 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.11328 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.78711 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.45312 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.12695 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.79492 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.79102 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 249.45703 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.45703 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.13086 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.46484 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.13086 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.13867 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 287.47266 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.14648 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.82031 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.82031 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 309.48633 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.16016 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.83398 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.16797 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.16406 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.83789 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.17188 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 355.50586 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.17188 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.17188 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.83984 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.51367 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.1875 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.86133 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.19531 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.86914 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 413.54297 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 417.53906 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.21289 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 433.54688 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.21289 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.88672 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 449.56055 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 453.55664 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.23047 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 466.9043 -9.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.02148 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.69531 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.36914 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.03711 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.71094 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 53.37891 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.05273 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.71875 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 69.39258 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.39258 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.06641 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.0625 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.73633 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.4043 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.07031 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.74414 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.74414 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.07812 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.75195 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.08594 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.75977 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 147.43359 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.10742 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.78125 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.78906 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 177.46289 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.13672 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 193.4707 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.13672 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.81055 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.81055 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.14453 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.81836 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.81836 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.48633 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\012) +[8.028 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.15234 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.82617 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 253.5 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.17383 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.50781 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.18164 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.84766 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.84766 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.18164 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.85547 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.18555 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.85938 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.5332 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.19922 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.86523 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 333.53906 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.21289 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 343.54688 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.2207 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.2207 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.89453 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.56836 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 379.57617 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.25 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.24609 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.24609 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.91992 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.58789 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 415.58789 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.26172 -24.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.34766 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.34766 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.02148 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.02148 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.35547 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.0293 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 51.36328 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.03711 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.36719 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.0332 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.70703 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.70703 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.04102 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.71484 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.04883 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.04883 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.72266 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.72266 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.72266 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.38867 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.0625 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.72852 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.39648 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.39648 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.07031 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 158.73633 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.74219 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\010) +[8.016 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.74609 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.41992 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.09375 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.76758 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.44141 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.44922 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.11523 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.12305 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.79688 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.4707 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.14453 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.81836 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.82617 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 265.5 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.49609 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.16992 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.84375 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.51172 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.18555 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.85352 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 309.52734 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.19336 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.85938 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.5332 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.5293 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 337.5293 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.20312 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.19922 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.87305 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 361.54688 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.87695 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 375.55078 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.22461 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.89844 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.89453 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 405.5625 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.9043 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 419.57812 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 425.57812 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.91211 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.57812 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.25195 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.92578 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 447.5918 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.26562 -39.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.01367 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.34766 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.02148 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.69531 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.36328 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.03711 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.70312 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.37695 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.05078 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.72461 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 77.39062 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.05859 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.73242 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 93.39844 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.39844 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 105.39844 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.07227 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.74023 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.74023 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 131.41406 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 135.41016 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.07617 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.75 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.75 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 157.41797 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 167.41406 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.08789 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.76172 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.75781 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 187.42383 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.75781 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.75391 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207.42773 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.76172 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.75781 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 221.43164 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.76172 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.76172 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.43555 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.10938 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.7832 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.44922 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 273.44531 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.11914 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.79297 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.12695 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.80078 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.13086 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.13086 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.80469 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 325.4707 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.13672 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.4707 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.13867 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.80469 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.8125 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 357.48633 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.15234 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.82617 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 373.5 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.16602 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.16602 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.16602 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.83203 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.16211 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.83594 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 417.50977 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.18359 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.85742 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.86523 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.86133 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 457.53516 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.20117 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 462.86719 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 465.5332 -54.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.01562 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.01953 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.01562 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.68945 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.35547 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 51.36328 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.0293 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.03711 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.03711 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.71094 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.04102 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.71484 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 103.38867 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 113.39648 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.07031 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.06641 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.74023 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.4082 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.4082 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.07422 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.07422 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 158.74805 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.74414 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.74023 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.74805 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.42188 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.08789 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.75391 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.42773 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 205.42383 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.42383 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.09766 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.09375 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.76758 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.44141 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.77148 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.77148 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 251.4375 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.10547 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.7793 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.77539 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 281.44922 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.11719 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.7832 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 297.45703 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 303.45703 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.79102 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 313.46484 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.79883 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 329.47266 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.14648 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.8125 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 345.48633 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.82031 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 355.48828 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\022) +[8.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.82227 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 371.49609 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 374.16211 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.16211 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.83594 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 393.50977 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.51758 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.19141 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.86523 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 419.53125 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.86523 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 435.53906 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 439.53516 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.20898 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 449.54297 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.20898 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.88281 -69.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.00586 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.67383 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\012) +[8.028 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.33984 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.01367 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.6875 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 53.36133 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.69531 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.69531 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.36914 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.04297 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.04297 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.7168 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.38281 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.04883 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.71484 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.71484 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 115.38281 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.05664 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.73047 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 135.4043 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.73438 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 151.40039 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 161.4082 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.08203 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.75586 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.75586 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.08984 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.76367 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.09375 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.09375 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.09375 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.76758 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 219.43359 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.10742 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.10352 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.76953 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.76953 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.44336 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.11719 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.79102 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 265.45898 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 275.45508 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.12891 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.80273 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.79883 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 295.46484 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.79883 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.79492 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.46875 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.14258 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.13867 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.80469 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.13867 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.80469 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 353.47852 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 359.47852 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.8125 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.48633 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.82031 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\037) +[1.836 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 381.48633 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.16016 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.16016 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 397.49414 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.16797 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.83594 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 417.50977 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.84375 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.83984 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 443.51367 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.17969 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.8457 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 451.51172 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.8457 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 467.51953 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 471.51562 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 478.18945 -84.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 17.34375 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.01758 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.35156 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.02539 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.69336 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.35938 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.0332 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.0332 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.70703 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.70703 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.375 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\014) +[6.78 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.70508 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 95.37891 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 101.37891 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 107.37891 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.38672 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.06055 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.06836 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\037) +[1.836 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.73438 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.4082 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.4082 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.74219 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.75 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.74609 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.41992 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 185.41992 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 191.41992 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.09375 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.76172 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\015) +[2.256 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.0957 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.76953 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.10352 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.77734 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.45117 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.125 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.45508 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 251.45508 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.12891 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.12891 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.12891 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.80273 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.46875 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.13477 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.80078 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.80078 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 293.4668 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.80078 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 303.47461 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.14844 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.81445 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.82227 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 329.49609 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 333.49219 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.16602 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 343.5 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.16797 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\012) +[8.028 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.83398 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 365.50781 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.18164 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.85547 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.18945 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.86328 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 401.53711 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.21094 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.20703 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.88086 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.21484 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.21094 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.21875 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.89258 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 449.56641 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.90039 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 462.9082 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 468.9082 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 475.58203 -99.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 2.66602 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.34766 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.02148 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.69531 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 35.36133 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.02734 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.70117 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 51.375 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.70898 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.38281 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 67.38281 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.05664 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.73047 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.73828 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.73438 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 107.4082 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.74219 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.07617 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.74219 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.74219 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.74805 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\010) +[8.016 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.75195 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.42578 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.09961 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.77344 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.44727 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.45508 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.45508 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.12891 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.125 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.125 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.79883 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.13281 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.80664 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 221.47266 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.14648 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.82031 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 237.48633 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.48633 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 249.48633 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.15234 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 265.48242 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.15625 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.83008 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.83008 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.16406 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.83789 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.17188 -114.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.66602 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.66992 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.66992 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.34375 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.00977 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.67578 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.3418 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 55.3418 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.00781 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.3418 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.01562 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.68945 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 77.35547 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 87.36328 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.0293 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.70312 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.69922 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 107.37305 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.70312 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.70312 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 129.36914 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.70312 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 139.37695 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.38477 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.05078 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 158.72461 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.72461 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 171.39844 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.72852 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 187.39453 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.06836 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.74219 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.73828 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.41211 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.08594 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.75391 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.42773 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.09375 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.76758 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.44141 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.10742 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.10742 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.10742 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.77344 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.76953 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 281.4375 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.44141 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 293.4375 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.11133 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.78516 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.78516 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 319.45898 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.13281 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.80078 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 339.47461 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.14844 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.81445 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.81445 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 357.48047 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.1543 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.82812 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 374.16211 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.83008 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 387.50391 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.16992 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.84375 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.17383 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.83984 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.84766 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.84766 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 435.52148 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 441.52148 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.85547 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 447.52148 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.19531 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.86914 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 463.53516 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 470.20898 -144.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.34766 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 35.34375 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.01172 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 44.67773 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 47.34375 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.01758 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.69141 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.35742 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.36523 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.03125 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.70508 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.70508 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.03906 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.71289 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.04688 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.7207 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.38672 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.05273 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.72656 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.40039 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.07422 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 153.4043 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.07031 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.74414 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.41797 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.08594 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.09375 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.76758 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.76367 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 203.4375 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.4375 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.10352 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.77734 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.78516 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.45117 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.125 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.125 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.79883 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.12891 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.80273 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 277.47656 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.14258 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.15039 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.82422 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 303.49805 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 313.50586 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.83984 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 323.51367 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.17969 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.8457 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.8457 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 341.51367 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.51758 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 353.51367 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.1875 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.86133 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.86133 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 379.53516 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.20898 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 392.87695 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 399.55078 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.21875 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.89258 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 415.55859 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.22461 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.89844 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.57227 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.24609 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.91406 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 451.58789 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.26172 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 464.93555 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 470.93555 -159.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\015) +[2.256 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.33398 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.3418 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.3418 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.00781 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.00781 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.68164 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.67773 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.67383 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.68164 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.01562 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.68945 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 69.36328 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.03711 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.70312 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.37695 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.04492 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.71094 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 101.38477 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 105.38086 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.05469 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 125.38477 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.05859 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.73242 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.74023 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 155.41406 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.41016 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.08398 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.08398 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.75 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 181.42383 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.09766 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.76562 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\015) +[2.256 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.09961 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.77344 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.10742 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.78125 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 221.45508 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.12891 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.45898 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.125 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.79883 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 251.47266 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.14648 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.14258 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 265.47656 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.14258 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.14258 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.81055 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.14453 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.81836 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.81445 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.14844 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.82227 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.15234 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.81836 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.82617 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.82617 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 337.5 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.17383 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.17383 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.84766 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 359.51367 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.17969 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.8457 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 374.17969 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.85352 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.51953 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.19336 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.86719 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 399.5332 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 405.5332 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 411.5332 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.19922 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.19531 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.86328 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 437.53711 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.21094 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.88477 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 457.55859 -174.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 2.66602 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 5.33203 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.00586 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67969 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.67578 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.68359 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.35742 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.02344 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.69727 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.69727 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.37109 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.36719 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.04102 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.70898 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.04297 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.7168 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.71289 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.04688 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.7207 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.05469 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.72266 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.39648 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.0625 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.73633 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.73633 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.41016 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 145.40625 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.08008 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 158.74805 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.74805 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 171.42188 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.0957 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.76953 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 191.44336 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 201.45117 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.78516 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.45898 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.125 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.79102 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.125 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.45898 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.13281 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.12891 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.46289 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.13672 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.4668 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.14062 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.80859 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.14258 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.81641 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 283.48242 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.14844 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.82227 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.82227 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 305.49023 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 313.49414 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.16797 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.83398 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 325.5 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.17383 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.84766 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.18164 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.85547 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.85547 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 361.5293 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.20312 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.21094 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.88477 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 391.55859 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.23242 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 411.5625 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 417.5625 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.23633 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.91016 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 433.57617 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.25 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.92383 -189.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.66992 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.33594 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.33594 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.00977 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.34375 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.01758 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.69141 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.69922 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.37305 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.03906 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.71289 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 87.38672 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.06055 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.73438 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 107.4082 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.73828 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.73828 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 129.4043 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.73828 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 139.41211 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.08594 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.75391 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.08789 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.76172 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.75781 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.43164 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.10547 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.43945 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 201.43945 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.10547 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.77344 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.44727 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 227.44336 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.11719 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.78516 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.45117 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.125 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.125 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.79883 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.79883 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 275.4668 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 283.4707 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.14453 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.81836 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.81836 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 309.49219 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.1582 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.82422 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.49805 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.83203 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.16602 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.83984 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.83594 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 351.50977 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.17578 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 363.50977 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.18359 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.84961 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.84961 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 381.51562 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.18359 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.85742 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 401.53125 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 405.52734 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.86133 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.86914 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.86914 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.54297 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 441.55078 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 447.55078 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.22461 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.2207 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.88672 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 467.56055 -204.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.01562 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.68945 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.36328 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.03711 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.70312 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.37695 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.05078 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.38477 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.05078 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.05859 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.73242 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 91.40625 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.07227 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.73828 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 103.41211 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.08008 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.07617 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.75 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.74609 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.41992 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.41992 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.41992 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.09375 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.76758 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.44141 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.10742 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.78125 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.11523 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 191.44922 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.12305 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.12305 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.78906 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 209.45508 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.12109 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.12109 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.78711 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.12109 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\037) +[1.836 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.78711 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 239.46094 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.46094 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.79492 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.80273 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.80273 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.47656 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 281.48438 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 287.48438 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.1582 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.1543 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.82031 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 307.49414 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 313.49414 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.16211 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\013) +[7.356 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.16602 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.5 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.16602 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.83984 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.16992 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.83594 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 363.50977 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.18359 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 374.17969 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.85352 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 387.52734 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.19531 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.19531 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.86914 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.53516 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.20117 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.86719 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.86719 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.5332 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.86719 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 433.54102 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.21484 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.88086 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.88867 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 459.5625 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 463.55859 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 469.55859 -219.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.68164 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.34766 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.02148 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 35.35547 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.0293 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.02539 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.69922 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 59.37305 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 69.36914 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.03711 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.70312 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 91.37695 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 95.37305 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.04688 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.7207 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 111.38672 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.7207 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 121.39453 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.72461 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 135.39844 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.06445 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.73828 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.73438 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 161.4082 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.08203 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 171.41602 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.08984 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 191.41992 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.09375 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.75977 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207.43359 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.10742 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.77344 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.77344 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.77344 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.43945 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.76953 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.76953 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.44336 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.11719 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.11719 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.79102 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.45703 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.12305 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.78906 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.78906 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 297.45703 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.12305 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.78906 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 315.46289 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.46289 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.79688 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 337.4707 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.14453 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.47852 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.15234 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.82617 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 367.49414 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.82812 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.50195 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.50195 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.16797 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 389.50195 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 395.50195 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 401.50195 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.17578 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.17578 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.8418 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.51562 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 429.51562 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.18945 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.19727 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.87109 -234.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 2.66602 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 5.33203 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.66602 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.33594 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 29.34375 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.67773 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.35156 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 43.34766 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.02148 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.69531 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.36914 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.04297 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.71094 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 83.38477 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.05859 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.38867 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 103.38867 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.0625 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.73633 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 123.41016 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.08398 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.75 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.75781 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.43164 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.10547 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.10547 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 165.43945 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.43555 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.10938 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.77734 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.45117 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.125 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 203.45508 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.12109 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.79492 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.79492 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 225.46875 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.14258 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.47656 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.15039 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 251.48438 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.1582 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.82422 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.82031 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 277.49414 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.16797 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.8418 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 297.51562 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.18945 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.18945 -249.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.66602 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 11.33203 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.00586 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.00586 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.33984 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.01367 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.6875 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.02148 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.69531 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.36914 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.03711 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.37109 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.04492 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.04492 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.71094 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.04492 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.04492 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.04492 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.71875 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.71875 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.38477 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.05859 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.05859 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.73242 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.74023 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 155.41406 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 165.42188 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.08789 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.75391 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.08789 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.76172 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.75781 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.76562 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.09961 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.77344 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.76953 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 215.44336 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.11719 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.79102 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.46484 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.13281 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.80664 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.48047 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.81055 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.81055 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 275.48438 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.1582 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.83203 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 295.50586 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.17188 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.17969 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.85352 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.52734 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 327.52734 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.86133 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.85742 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 341.53125 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.19922 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.87305 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 361.54688 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.87695 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 371.54297 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.2168 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.2168 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.89062 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 397.56445 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.89844 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 407.57227 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.90625 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.58008 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.24609 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.24219 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.91602 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 449.58984 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 456.26367 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 462.9375 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 469.61133 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 475.61133 -279.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.99609 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.66992 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.66602 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 27.33984 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.33984 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.00586 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.01367 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\037) +[1.836 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.67969 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 55.35352 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.35352 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.6875 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.69531 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.36914 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.36523 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.03906 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 95.37305 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.04102 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.71484 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 115.38867 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.72266 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.72266 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.39648 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.07031 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.07031 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.74414 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.41016 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.07617 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.74219 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.07617 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.75 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 187.42383 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.09766 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.77148 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207.44531 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.11328 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\012) +[8.028 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.7793 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.45312 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.12695 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.80078 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.13477 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.80859 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 265.47656 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.15039 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.82422 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 281.49023 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.49805 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.83203 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.50586 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.17188 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.83789 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 313.51172 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 319.51172 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.17969 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.85352 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 339.52148 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.19531 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.86914 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.86523 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.19922 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.20703 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.88086 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.55469 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.22852 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.90234 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.57617 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.24414 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 417.58594 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.25977 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.25977 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 433.59375 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.25977 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.93359 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 449.60742 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.27344 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.94727 -294.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.66602 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.68164 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.68164 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.35547 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.68555 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.35938 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.0332 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 77.36719 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.04102 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.70703 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.70703 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.70703 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.04102 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.71484 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 115.38867 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 121.38867 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.05469 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.72852 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.40234 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.73633 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 153.41016 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.08398 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.75781 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.43164 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 177.42773 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.10156 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.09766 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.76367 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 197.43164 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.10547 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.7793 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 213.44531 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.7793 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.45312 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.12109 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.79492 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.46094 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.13477 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.13086 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.79883 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\015) +[2.256 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.13281 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.80664 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.14062 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.81445 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 295.48828 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.16211 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 309.49219 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 315.49219 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.16602 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.83984 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 335.51367 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.17969 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.17578 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.84961 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 361.52344 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.85742 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 371.53125 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.86133 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 391.53516 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.20898 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.875 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.88281 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.2168 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.21289 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.87891 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 427.55273 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.22656 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.89258 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 439.55859 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.22461 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.23242 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.90625 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 461.57227 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 468.24609 -309.74219] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67969 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.35352 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.02734 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 35.36133 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.03516 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.03516 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.70898 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.38281 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.39062 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 77.39062 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.06445 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.73828 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.41211 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.08594 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.75977 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.42773 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.42383 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.09766 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.77148 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.77148 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 153.44531 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.11914 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.79297 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.12695 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.80078 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.13477 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.80859 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.80469 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.80469 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.80469 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 221.47852 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.15234 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.81836 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 237.49219 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.82617 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.49414 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.49805 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.49414 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.16797 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.8418 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.8418 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 285.51562 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.18945 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.85742 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 305.53125 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.20508 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.87891 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.20898 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.88281 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 345.55664 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.23047 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.22656 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.89453 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 375.56836 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.24219 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.91602 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.91602 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.25 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.92383 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.25781 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.93164 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.93945 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.93945 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 443.61328 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 447.60938 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 453.60938 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.2832 -324.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.02148 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.6875 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 29.36133 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.03516 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.03516 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.70312 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.03711 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.71094 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.38477 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.05859 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.72461 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.39844 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.06641 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.06641 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.73242 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.06641 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.74023 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.74805 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 123.42188 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.0957 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.76953 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.44336 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.11719 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.78516 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 165.45117 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.44727 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.12109 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 185.45508 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.12891 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.80273 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.13672 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.81055 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.47656 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.15039 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.1582 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.16602 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.83203 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 253.50586 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.51367 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.51367 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.17969 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.17969 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.85352 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.84961 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.8457 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.85352 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 309.52734 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.19336 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.85938 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.5332 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.20703 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.88086 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 341.54883 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.22266 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.89648 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.9043 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 371.57812 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.24609 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.91992 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 391.59375 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.25977 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.93359 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 407.60156 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 415.60547 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.2793 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.28711 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.95312 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.96094 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 451.63477 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 457.63477 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 464.30273 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 470.30273 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 476.97656 -339.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.34766 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.34766 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.34766 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.02148 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.6875 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 47.36133 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 51.35742 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.02344 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.02344 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.69727 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.37109 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 83.37891 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.05273 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.72656 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.39258 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 105.39258 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.05859 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.73242 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 121.40625 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.74023 -354.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.66602 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.66211 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.33594 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.66992 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.67773 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.67383 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.67383 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.00781 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.68164 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.67773 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 75.35156 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.01758 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.01758 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.68555 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\015) +[2.256 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.01953 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.69336 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.02734 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.70117 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.375 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.04883 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 131.37891 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.05273 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.7207 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 147.38672 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.05273 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.72656 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.40039 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 167.39648 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 177.4043 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.07031 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.74414 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.74023 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 197.41406 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207.41016 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.07812 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\015) +[2.256 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.41211 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 227.41992 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.75391 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 237.42773 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 241.42383 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.09766 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.76367 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.09766 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.76367 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.4375 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 273.43359 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.10742 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.10352 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.77148 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 303.44531 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.11133 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 309.44531 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 313.44141 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.10742 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.10742 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.77344 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.44727 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.78125 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.45508 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.12305 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 357.45703 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.12305 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.79688 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.79688 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 375.46289 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.13672 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.81055 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 395.48438 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.15234 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.82617 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 415.5 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.17383 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.8418 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.8418 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 441.51562 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.18945 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.18945 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.86328 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 463.5293 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 466.19531 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 468.86133 -384.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 25.34766 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.02148 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.6875 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.36133 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 51.35742 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.36328 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\010) +[8.016 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.36719 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.04102 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.71484 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.38867 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.0625 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.07031 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.74414 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 115.41797 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.08398 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.08398 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.75 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.42383 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.09766 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 147.42773 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.10156 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 158.09766 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.77148 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 171.43945 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.11328 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.78711 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 191.46094 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.12891 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.80273 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.47656 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.14258 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 227.47266 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.14648 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.14258 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.81641 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.81641 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 253.48242 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.15625 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.16406 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.83789 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 283.50586 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.83984 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.83594 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 293.50195 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.17578 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.84961 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 309.51562 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.18164 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.84766 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.85547 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.5293 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.20312 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.87695 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 351.55078 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 361.55859 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.22461 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.89844 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.57227 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.24609 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.24219 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 391.57617 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.24219 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.24219 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.91016 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.91406 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 421.58789 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.25391 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.91992 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 433.59375 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.26758 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 443.60156 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.27539 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 456.27539 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 462.94922 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 469.62305 -399.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.02148 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.6875 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.02148 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.69531 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.36914 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.03711 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.0332 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.70703 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.70312 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.37695 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 79.37695 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.71094 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.70703 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 89.37305 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 95.37305 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.70703 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 101.37305 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.04688 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.7207 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.72852 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.72852 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.40234 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.07617 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.75 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.75 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.08398 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.75781 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.0918 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.76562 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.43359 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.10742 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.78125 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.11523 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.78906 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.12305 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.79688 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 235.46484 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.46094 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.13477 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.80078 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 261.47461 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 267.47461 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.14844 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.82227 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 287.49609 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 297.50391 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.83789 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 307.51172 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 317.50781 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.18164 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 333.51562 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.18945 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.52344 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.85742 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 359.53125 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 363.52734 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.20117 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.86719 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.20117 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.875 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 395.54883 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.22266 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 408.22266 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 411.55664 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.23047 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.23047 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.89844 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(\015) +[2.256 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.23242 -414.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.33398 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.67383 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 21.33984 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.01367 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 34.6875 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 41.36133 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.0293 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.69531 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.36914 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 63.36914 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.04297 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 79.37695 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.05078 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 95.38477 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.05859 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.72461 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.72461 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.72461 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 123.39844 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.07227 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.74609 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.74609 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.08008 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.75391 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.08789 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.76172 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 175.43555 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 181.43555 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.76953 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 191.44336 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.77344 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.77344 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.44727 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 221.45508 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.12891 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 237.46289 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.13672 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.81055 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.48438 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 267.48047 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.14844 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.81445 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.48828 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.1543 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.82031 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.49414 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.82422 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 317.49023 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 327.49805 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.17188 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.83789 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 343.51172 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.18555 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.85156 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.85156 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.85156 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 367.51758 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.84766 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.51367 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.1875 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.1875 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 399.52148 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.19531 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.19531 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.86328 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 427.5293 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.20312 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.86914 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 439.53516 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 449.54297 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.20898 -429.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.33398 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 10.00781 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.01367 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.34766 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.02148 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.6875 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.6875 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.35352 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.02148 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.6875 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.36133 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.03516 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 87.36914 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 93.36914 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.03516 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.70312 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 109.37695 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.37305 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.04688 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.71484 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 135.38086 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.04688 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.7207 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 151.39453 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 155.39062 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 165.39844 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.06445 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.73828 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 181.41211 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.08008 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.08008 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.74609 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.08008 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.75391 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.76172 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.76172 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.43555 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.10938 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.10938 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.7832 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.45703 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.13086 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.80469 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 275.47266 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\037) +[1.836 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.13867 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.8125 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.8125 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.14648 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.82031 -444.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.66602 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 18.00586 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.67188 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.67969 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.67969 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 43.35352 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 53.34961 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.02344 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.69727 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.02734 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 77.36133 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.03516 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.03125 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 91.36523 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.03906 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 105.36914 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.04297 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.71094 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 124.71094 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 131.38477 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.71484 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.04883 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.04492 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.71094 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 161.38477 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.05859 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.72461 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.39062 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.05664 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.06445 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.73828 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.41211 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.08594 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.08594 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 215.41992 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.09375 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.09375 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.76172 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.75781 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 251.43164 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.10547 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.10547 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.7793 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 277.45312 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.12695 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 293.46094 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.13477 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 306.80859 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 313.48242 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 319.48242 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.81641 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 329.49023 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.82422 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.82422 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 351.49805 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.17188 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.83789 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 367.51172 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.51953 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 383.51953 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.18555 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 392.85352 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 399.52734 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.52344 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.19727 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.86523 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.86133 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 439.53516 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.20898 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.20508 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.87109 -474.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.6875 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 29.36133 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.02734 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.69531 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.36914 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 55.37695 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.04297 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.7168 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.7168 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.38281 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.05664 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.72266 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.73047 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.4043 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.07812 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 108.74414 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 111.41016 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 121.41797 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.41797 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.0918 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 138.08789 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.75391 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 147.42773 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 153.42773 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.0957 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.09961 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.77344 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 181.44727 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 187.44727 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.12109 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.78711 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.45312 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.12695 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 215.46094 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.13477 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.80078 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.13477 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.13086 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.79688 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.79688 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.46289 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.13672 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.4707 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.14453 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.81836 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 275.48438 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.15039 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.82422 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.49805 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.83203 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.50586 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 307.50586 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.17969 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.85352 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.86133 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 337.53516 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 341.53125 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 347.53125 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 353.53125 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 359.53125 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.20508 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.20508 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.87891 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 385.55273 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 391.55273 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.88672 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 401.56055 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.89453 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 411.5625 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 419.34961 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.01562 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.01562 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.68945 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.68555 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 451.35938 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.69336 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 467.36719 -489.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.02148 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.69531 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.70312 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 43.37695 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.05078 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.72461 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.7207 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.38867 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 80.0625 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.73047 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 89.39648 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 96.07031 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 102.74414 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.74023 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 113.41406 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.08789 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.75586 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 133.42969 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.10352 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.76953 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.44336 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.77734 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.44531 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.11133 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.78516 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.11523 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.11523 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.78906 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.78516 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 217.45898 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.13281 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.46289 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.79688 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 237.46289 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 244.13672 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.13672 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.80273 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.47656 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.15039 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.82422 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.49219 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 285.49219 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.16602 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.16211 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.83008 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 311.49609 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 321.50391 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 328.17773 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.85156 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 337.51758 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.19141 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.85742 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.85742 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.85742 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 361.52344 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.19727 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.20508 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.87891 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 391.55273 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.21875 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396.88477 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.89258 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.88867 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.5625 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.22852 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.90234 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.90234 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 445.57617 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 452.25 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.92383 -504.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.00781 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.00391 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.67773 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.35156 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.34766 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.01367 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.34766 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.02148 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.69531 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.36914 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.03711 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.0332 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.70703 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 111.38086 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 115.37695 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.04297 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.37695 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.05078 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.05859 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.05469 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.05469 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.72852 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.39453 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.72852 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.72461 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.39062 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 185.39062 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.06445 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 201.39844 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.73242 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.72852 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.39453 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.06836 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.74219 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 227.4082 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.07422 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.74023 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 239.41406 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.08203 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\012) +[8.028 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.74805 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 261.42188 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.0957 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.76953 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.10352 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 290.77734 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.77344 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.44727 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 311.45508 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.12891 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.80273 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.47656 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.15039 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 344.82422 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 351.49219 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 357.49219 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 364.16602 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.16602 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 373.5 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.16602 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.83984 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 389.51367 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 392.17969 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.85352 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.18359 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.85742 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 425.52539 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.19922 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.19531 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.86914 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.86914 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 451.53516 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 458.20898 -519.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.34766 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.01562 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.68945 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.68555 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 43.35938 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.69336 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.35938 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.0332 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 69.36328 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.03711 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 86.04492 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.71875 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.39258 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.06641 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.74023 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.41406 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.08203 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.07812 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.75195 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.42578 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 153.42188 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.08789 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 165.42188 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\033) +[3.756 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.75586 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 175.42969 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.10352 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.10352 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.76953 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 197.44336 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.11719 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 213.45117 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.125 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.79883 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 233.47266 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.14648 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.1543 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.82031 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.82812 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.50195 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.17578 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.84961 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.18359 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.85742 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 305.53125 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 312.20508 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.20117 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.875 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.20898 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 330.20508 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.21289 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.88672 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.55273 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.22656 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.22266 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.89648 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 379.57031 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.9043 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 389.57812 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 399.57422 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.24219 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 414.9082 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 421.58203 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.25586 -534.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.01367 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 28.6875 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.02148 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.69531 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.0293 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.03711 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.03711 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.71094 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 67.37695 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.04297 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.70898 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.70898 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.375 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.70898 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 91.38281 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.05664 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.72266 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.73047 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.4043 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.07031 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.74414 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.75195 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 139.41797 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.42578 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.09961 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.77344 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 166.76953 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.10352 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 173.4375 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.10352 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.4375 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.11133 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.44531 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.11328 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\024) +[7.704 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 204.7793 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.44727 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.11328 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 220.78711 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.78711 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.12109 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.79492 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.12891 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.79492 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 255.46875 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 262.14258 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 268.81055 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.81055 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 281.48438 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.1582 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.83203 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 301.50586 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 311.51367 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 318.1875 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.86133 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 327.52734 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.86133 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.86133 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.53516 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 355.53516 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.86914 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 361.53516 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 368.20898 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 374.88281 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.54883 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.22266 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 397.55273 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 403.55273 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.22656 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.90039 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.90039 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 429.57422 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.24023 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.90625 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 437.57227 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 443.57227 -549.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 2.66602 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.6875 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.68359 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.01758 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.68359 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.01758 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.69141 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 55.36523 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.37305 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.03906 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 70.70508 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 77.37891 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.05273 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.04883 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.72266 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 101.39062 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 109.39453 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 113.39062 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.06445 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.73047 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.73828 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 135.4043 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.07812 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.07812 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.74414 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 157.41797 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.08398 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.0918 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.75781 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 179.43164 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.43945 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.10547 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.11328 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.78711 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 215.46094 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.13477 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.80859 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.81641 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.49023 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.16406 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.83008 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.49609 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.16992 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.84375 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.17773 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.85156 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.85156 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 293.52539 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.19922 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.20703 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.20312 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.87695 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 327.55078 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 334.22461 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.22461 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 346.89844 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.89844 -564.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.66602 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33984 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.33594 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.00977 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.68359 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 35.34961 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.68359 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.35742 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.6875 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.6875 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.36133 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.03516 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.70898 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.375 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.04883 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 97.38281 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.05664 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.72266 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.05664 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.05273 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.71875 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.71875 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 125.38477 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 132.05859 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.39258 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.06641 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 154.74023 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 157.40625 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 167.40234 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.07031 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.73633 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.74414 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 189.41016 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 196.08398 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.41797 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.0918 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.08789 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.76172 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.43555 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.76562 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 243.43945 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.11328 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.78711 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.12109 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.79492 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.46875 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.14258 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.81641 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.8125 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 303.48633 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 307.48242 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.14844 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.81641 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 319.48242 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.15625 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.82422 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 341.49023 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.16406 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 350.83008 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 353.49609 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.16992 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 373.5 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.17383 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.8418 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 393.51562 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.18945 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.18945 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 409.52344 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 416.19727 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.52734 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 433.52344 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.19727 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 446.87109 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.86719 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 453.5332 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 459.5332 -594.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.00391 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 14.67773 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.68555 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 31.35938 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.02734 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.02344 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.69727 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.69727 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 66.69727 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.70508 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 83.37891 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.04688 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 93.38086 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.05469 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 104.05078 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 107.38477 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.05859 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 121.38867 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.38867 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.0625 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.73633 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.40234 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.40234 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.07617 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 158.74219 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.75 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 172.08398 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 178.75781 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.75391 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 195.42773 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.10156 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.43555 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.10938 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.11719 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.79102 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 241.45898 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 247.45898 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.13281 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 260.80664 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.47266 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 270.14648 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.82031 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 283.48828 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.49219 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.16602 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.17383 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.83984 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 317.51367 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 323.51367 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.17969 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.85352 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 335.51953 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.19336 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.86133 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.85742 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 365.53125 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.20508 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 376.20117 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.86719 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.20117 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.875 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.88281 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.88281 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 417.55664 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 424.23047 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.23047 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.9043 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 439.57031 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 442.23633 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.90234 -609.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 12.67383 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.34766 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.02148 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.69531 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.36914 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.70312 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.36914 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 51.36914 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.03711 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.70312 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 67.37695 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 73.37695 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.71094 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 83.38477 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.71875 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.39258 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.06641 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.74023 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.07422 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.07422 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 134.74805 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 141.42188 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.08789 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.08789 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.76172 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 159.42773 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.43555 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.10938 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.10938 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 185.44336 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.11133 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 198.78516 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 205.45312 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.12695 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 218.80078 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.79688 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.13086 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.13867 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 238.80469 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 241.4707 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.14453 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.81836 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.48438 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 267.49219 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 274.16602 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.16211 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.83594 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.16992 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.84375 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 307.51758 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.18359 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 319.51758 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 329.51367 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.1875 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.86133 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.53516 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 356.20898 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 362.87695 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\025) +[7.908 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.66406 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 373.33008 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 379.33008 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.00391 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 396 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 402.67383 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 412.00781 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 422.00391 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.67773 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.01172 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 435.3457 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 438.01172 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.01172 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.67969 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 453.3457 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.01953 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 466.01953 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 472.69336 -624.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 8.66602 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 15.33398 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 22.00781 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.00391 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.67773 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 45.3457 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 52.01953 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 58.69336 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.36719 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.0332 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 74.70703 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 81.38086 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.71484 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 93.38086 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.05469 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 103.38867 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.0625 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.05859 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.73242 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.40625 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.40234 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 144.07031 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.73633 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 153.41016 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 157.40625 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 164.08008 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 177.41016 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 184.08398 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 190.75781 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 193.42383 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.75391 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.75391 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 219.42773 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.42383 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 226.08984 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 232.76367 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.09766 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.43164 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.10547 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.10156 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.43555 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.10938 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 269.44336 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.11133 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.78516 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.79297 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.79297 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 305.4668 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 315.46289 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 325.45898 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 332.13281 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.80664 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.81445 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 351.48047 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 354.14648 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 360.82031 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 367.49414 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.16016 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 380.16797 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 386.8418 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 390.83789 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 397.51172 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 406.8457 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\035) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 413.51953 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.19336 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.86719 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 433.54102 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 437.53711 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.21094 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 448.20703 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.87305 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 457.54102 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 464.21484 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 470.88867 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 474.22266 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 480.89648 -639.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\015) +[2.256 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 3.33398 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 10.00781 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.3418 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.01562 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.68945 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.36328 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.69336 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 50.68945 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 57.36328 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 60.69727 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 67.37109 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 76.70508 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 79.37109 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.03711 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 88.71094 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 95.38477 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 99.38086 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.05469 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.72266 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 119.39648 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 122.0625 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 128.73633 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 135.41016 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.08398 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 148.75781 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 155.43164 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.76172 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.76172 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 181.43555 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 188.10938 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.7832 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 201.45703 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.13086 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 211.46484 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.13086 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.46484 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.46484 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.13867 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.8125 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 249.48633 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 256.16016 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.16797 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.8418 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 279.50977 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.18359 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 288.84961 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.51562 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.18945 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 308.18555 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.18555 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.85938 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 324.85547 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 331.5293 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 338.20312 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 345.5332 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.20703 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.88086 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 361.54688 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.88086 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 377.55469 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 384.22852 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 387.5625 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.23633 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 400.9043 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\017) +[9.084 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.90039 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 417.57422 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 421.57031 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.24414 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.24414 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.91797 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 447.5918 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.26562 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.93945 -654.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 17.34375 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 24.01758 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.01758 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 36.68555 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 43.35938 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.02539 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.69141 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 55.36523 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.03906 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.71289 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 82.04297 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.70898 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.7168 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 101.39062 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 105.38672 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 112.06055 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 118.06055 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.72656 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 127.40039 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 137.4082 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 143.4082 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.07422 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 149.4082 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.08203 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 162.75586 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 169.42383 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.09766 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.77148 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 185.4375 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 191.4375 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.10352 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.77734 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 207.45117 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 214.78125 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 221.45508 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 228.12305 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.78906 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 233.45508 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 240.12891 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.80273 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 250.79883 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 257.47266 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.14062 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\023) +[7.38 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.14453 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.81836 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 284.81836 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 291.49219 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 298.16602 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 304.83984 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 311.51367 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 314.17969 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 320.17969 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.17969 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.1875 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.86133 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 349.53516 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 352.86914 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 359.54297 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 366.2168 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 369.55078 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.2168 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 378.88477 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\011) +[8.196 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 387.55078 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 394.22461 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 407.55469 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 413.55469 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 420.22852 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.22852 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.89453 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 431.56055 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 440.89453 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 447.56836 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 454.24219 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 457.57617 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 464.25 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 470.92383 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 477.59766 -669.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 6.67383 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 13.34766 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.02148 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.69531 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 30.0293 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.69531 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 39.36914 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 46.04297 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 55.37695 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 62.05078 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 68.71875 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.71484 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 85.38867 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 92.0625 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 98.73633 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 101.40234 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 110.73633 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 117.41016 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.07617 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 129.41016 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.08398 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.75781 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.75391 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 150.08789 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.76172 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 160.75781 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.42383 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.09766 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 176.77148 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 183.43945 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 193.43555 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 200.10938 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 206.7832 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 210.11719 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.79102 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.79102 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 229.45898 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.13281 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 242.80664 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 248.80664 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 254.80664 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 261.48047 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 264.81445 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 271.48828 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.81836 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 282.81445 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 285.48047 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 292.1543 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 294.82031 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 300.82031 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 307.49414 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.16016 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 316.83398 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 326.16797 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 336.16406 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.83789 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.83789 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 355.50586 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\021) +[7.488 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 363.50977 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 370.18359 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 372.84961 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 375.51562 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.18945 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.86328 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 392.19727 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.87109 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.87109 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 411.54492 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 418.21875 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 428.22656 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\027) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 434.90039 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 437.56641 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 444.24023 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.91406 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 457.58789 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 460.25391 -684.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.99609 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.66992 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 19.33594 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.00977 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 32.00977 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 35.34375 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 38.00977 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 48.01758 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 54.69141 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 61.36523 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 65.36133 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 72.03516 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 78.03516 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.70312 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 94.71094 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.71094 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 107.38477 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\034) +[5.868 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 114.05859 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.72461 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 120.05859 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 123.39258 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 126.05859 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 135.39258 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 142.06641 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.0625 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 152.0625 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 158.0625 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 168.05859 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.73242 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 177.39844 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.06445 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 182.73047 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 192.06445 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 194.73047 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 201.4043 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 208.07227 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\012) +[8.028 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 216.73828 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 223.41211 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 230.08594 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 236.75977 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 246.09375 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.76758 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 259.44141 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 263.4375 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 266.77148 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 276.7793 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 280.11328 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 286.78711 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.7832 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 303.45703 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 310.13086 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 319.46484 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 322.79883 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 329.47266 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 333.46875 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 340.14258 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 342.80859 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 348.80859 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 355.47656 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 358.81055 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 365.48438 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 375.48047 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 382.1543 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 388.82812 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 398.16211 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 404.16211 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 410.83594 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 417.50977 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 423.50977 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 430.18359 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(%) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 436.85742 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 443.53125 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 450.20508 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 456.87305 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\037) +[1.836 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 459.53906 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 466.21289 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 472.21289 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 475.54688 -699.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +( ) +[1.824 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 2.66602 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 9.33984 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 16.01367 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 20.00977 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 26.68359 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 33.35742 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 40.02539 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 42.69141 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 49.36523 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 56.0332 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\020) +[7.68 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 64.69922 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 71.37305 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 84.70312 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 90.70312 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 93.36914 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 100.03711 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 106.71094 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 116.70703 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 123.38086 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 130.04883 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 136.72266 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 140.71875 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 146.71875 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 156.72656 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 163.40039 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 170.07422 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 174.07031 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 180.74414 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 186.74414 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\000) +[2.268 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 193.41211 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(*) +[5.856 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 199.41211 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 202.07812 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 205.41211 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 212.08594 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 222.09375 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\036) +[1.848 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 224.75977 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 231.43359 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 234.76758 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 241.44141 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 245.4375 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 252.11133 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 258.78516 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 272.11523 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +($) +[6.192 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 278.78906 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 285.46289 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(&) +[4.164 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 289.45898 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 296.13281 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(') +[5.532 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 302.13281 -714.74219] Tm +0 0 Td +/F82_0 12 Tf +(\002) +[2.292 +0] Tj +Q +Q +q +667.28 967.28 490 262.5 re +W* +q +q +[1 0 0 1 0 0] cm +0 0 595.28 841.89 re +W +Q +Q +Q +q +1 w +[] 0 d +0 J +0 j +[1 0 0 1 36 53.67] cm +q +[1 0 0 1 0 0] Tm +0 0 Td +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 0 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\006) +[6.096 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 413 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\031) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 419.67383 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(#) +[6.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 426.34766 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\030) +[5.892 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 432.34766 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\)) +[5.808 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 439.02148 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(!) +[9.228 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 449.01758 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\032) +[6.18 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 455.69141 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(") +[5.844 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 462.36523 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\() +[3.252 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 465.69922 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\001) +[3.624 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 469.69531 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\026) +[6.168 +0] Tj +/DeviceRGB {} CS +[0.0863 0.0784 0.0745] SC +/DeviceRGB {} cs +[0.0863 0.0784 0.0745] sc +0 Tr +[1 0 0 1 476.36914 -9.625] Tm +0 0 Td +/F82_0 12 Tf +(\006) +[6.096 +0] Tj +Q +Q +Q +Q +showpage +%%PageTrailer +pdfEndPage +%%Trailer +end +%%DocumentSuppliedResources: +%%+ font T3_35_0 +%%+ font T3_82_0 +%%EOF diff --git a/test/document-a4.sla b/test/document-a4.sla new file mode 100644 index 0000000000..95d518f409 --- /dev/null +++ b/test/document-a4.sla @@ -0,0 +1,277 @@ + + + + + + + + + + + + + + + + + + + + + + + + + +

      CUPS 1.6 Software Test Report

      + +

      This software test report provides detailed test results that +are used to evaluate the stability and compliance of CUPS Version 1.6. + +

      Document Overview

      + +

      This software test plan is organized into the following sections: + +

        +
      • 1 - IPP Compliance Tests
      • +
      • 2 - Command Tests
      • +
      • 3 - Log Files
      • +
      diff --git a/test/str-trailer.html b/test/str-trailer.html new file mode 100644 index 0000000000..e04310fdd2 --- /dev/null +++ b/test/str-trailer.html @@ -0,0 +1,2 @@ + + diff --git a/test/testfile.jpg b/test/testfile.jpg new file mode 100644 index 0000000000000000000000000000000000000000..418cb93597c7a3cb5a425fffaf2250373663b3ae GIT binary patch literal 204156 zc-ri`RZv_(*ETx1Bsc`OFk~1kfgpnhm%#_u2o{{dJ-7vD20|Eoa3?UhI|K;s?h>4k z07*{1_pAE{rXZuF zq#&c9pnAeYOGV86q!GLK_u@LR^LO%`Aosk(mJgGJp)oyvI>v5{5KNQKMwzR zVEx~y{2K}D9~Z)ZUS$dZHWoH64h|s!J}xf7e_{R+p}?hN7r>L%p?V6Z7JLbN8@NWJG~z7b&UCv!sc3&a&cN)|!4i+*twCZ+aOY2pljkTgr?%)sd8+(I z;ENyP13gpJloj4_XPWw5Z7x%_hxohww1@34%ps8FRd@$eos5l&dT^b1B6 zNa+iB$-ez3y{X7hF_7lvhmN%hk=A1}{$G9=(O-`m@Br}ayWM-^X5nwN zs_8sh>ZwhMkvhze8dR^-gUDmIG}ZWz0l8ubbVx0I*UD`54Wp~+1L)gomFcbBrAir8s*J9VabUQr|d=WcgQvBn}MO zeD%(rHaqWoD0G&C;+m4DXjnrbQr3@7A071!-Tcm&fR`m)?nBjoYdr`|IVK<=;hI@K zM{K5jQ2=e&3KrC&qGY!{MOH1;d?&A- zd=z9q{!V1eB9*!0B5*Koln< zZFHaDLZi5-b&%?UvuCU4`4izU>=c>u;r34KZ$xCq6NdKOQ18&Id>5BztHSUPWbi~) zCao@kO0{bT99Nv^goa?(EhyT9)Xh-n}43lYphzUCqFaP+i=Bh-=ZgU+W%|^#oZ(TrVu<0YnDjxgd9pl-q|(6&AcHt>(<*H zG+rw!wkPveHOub<+3=*$;c+P2cglA1Oy#);xni=3rT6iCe*wC;494sYb}q4yGJneK zFq-iyAum%8-WL(fb%NFWC%5x*9*VTlZ>}9IL;7F5?P^tw{&c1sU8#YgSd<|EfmBo_ zuEwDM`b)jyJS6NstlKKlAz6=Gdu)+9OO&-Gb2fx;6H6O%1vCBbIu`x*CfcSiwit~j5z4SR3@MC2FE!g28Y%haH|4cm zz@b{MMe^=TSM|)X`)JAs-xU1tQ9bu+(2qRD{*V>Y)!7g5iJ9HWC_1=R_OljKn+EZB z4Wkq`;G%UyUzoD*z|R5StF?+Xx>^&6O+V$}yWp>nJe2M*bnoASnSP^FR6%a4&w|n7 z;2Mv8_pOgsRnjrNC^1`%Xw;`L=BU@rypK7d+OxSs=sCYa99#d$^}UvUt)lGJ1DP zV2+D)s&*A=wcRTGG(Q%QAIpUu%au#~-+RXYU;Tf%N?~z2cg#yAeh8+p#KTReMOj@k z>XPjxcC+?y<(vgl&)&$8m@`Yh`mM`FICbELbQdf;jMzxxY%|3LK*{^xemtT=NQBvFrYS zLX>b-`7-$>^5n>u{RId)6a69Y`wQ?tD7aGEsNQ_6b=wiU3%w@2^9BAdj0N%k!YKa3 z;Qj~mb|jc5Ta~3?WdGYy&KPZIXSl$twL7kdNgtk{@>IXi@|)ifUmFG_NNM_7-7#zZ z$~(7sdDH!0j<$zM$n&4F|H0VAo+ZndTn=9=>ju3?(LS*K7e(d0nC=ffg#UQcj0#x_ zU2=J{7)wVVxr|YTUgT5}{&zfWG!N|miKo^mM_)35SvgHBZ}!#a=Xf6o-+gGmHjF|$ z6SMvWX#aJP8wm!b-DJ`z>cpnlpoH!8Q`HY;2LfLBT5vpcDsm}PF3e@AG}Y&Ay}UM* zG>uII@N^BhDSIb?@2Iaq21xQGX@1}BHp4_iR_e=M-I;<&-45G#`;xl7hvNDkR;(m{ zV$UyLlaYEGNWac6rOh&JS;PJQm13lxwoSt+4>A|O^>sQd?8|els&VGuHeFW~v4 z*|`PKU56TvYNgEU&4Eib3;pk(iI=~rSp+td{{pW58}pM=PvUn&cGxJjE9CWGfJMOb z`zywOTzdyly!aYNqNCUP*f0l^sn2_Q!r}zS>k+SlwgWm9eSEgV2PIZq(hsV9O$cHt z9iYlFrB?E~`DD!J(fffPFMrcio+kMS7tL&^|BR7SQWeLr9PMp#rkj~M;qj?JAet)| zU#I;i(V8i4M~rdt-_TsU%CqZ! zMrERPqNP@^8seNV@Ez@m$VT{Jzrgzgid^%x;yFP71$;hJTJKq4{WoLBr1+KsNi;xm zN>6lQN|P# z$UdcNZBVeF&#|#VBH`Q7HE4|gC9uL1Q1cMB?IXGtpfS4LtErHm$ z@^OiLd!j5%HlpQ~VgFY2f6=l1RdmjJp2m8=t_asN?3Q?j1aBpe7wifiNCkiH&IHxn^Z^WDVaSnbo7yk_b3k+a!a zN=wp16%zl(Q+|U!Ar%#SFiqBcT7OuKG|z(W%r?X43OPx!VU?iyVEE}%45q$C?vB!E&k*kM!}Wm6d)-rEh2<8- zEr@tj55Ec?9!C|!^{IsXg!98y$!pe&_MX6#iEA$=SoH8TqByOOIgz9BA~Cq^7^-bT zSURB)TLsfQqnn zs*$XZv-DFFjj5Vq6QCJRsQi?B11u+?8>~ipp-{6!>8+%fI{!b zXO|4}9I;J7kkNUSLx$KO5pSkne#e>p4fc6{$HInO2$E?7 zeS2BP0_Es)pk%Tjk7L)wS`eGC!f9T~F;{j@uX#YNfdATG8nm^su=T6XSNhpiY zV!gTVp#yC$G-nUYrWw5@a*K;Hm`X}1JaPi>c;9?) z^b~|MmM8W4<3Ykkdo7zT&#=qNP0*YU?=_(NW+3z_&cCI?$o}MLf$RHeAsPIZur8^% z8vc!DD5>(-FJFf3Oup`NJ#RF1Qz--Gx69Fh9j8rRM_ISB}<8u(Hjql z0;Pvzmg;EDDa3k7=#5Z8!n5Mm4@8;Y zvwvKr+iVEC1n)$kYv(+3*_G*MUPwrssE#R#O9?vHynL!l#>QG(iHViq;0RO>ve1Y2 zcaII`xxBe7p0OZ6`X^GraF;h~LbqE`0B>EZ`azzDQzX{DfjoK9Ni;N4->c6gpk#X2 zvz-pfY}Lb1dF<<7A}SmsSNsG?f%Wg)TR8{K5}o`DE5ZrInJ z+{iDBw%b%k6Tb#0O-B4n^KM!6zZSM@qz!eeEXq$jJk0LGV;U4qOB=BYl{eJ4)#1H` z1ab(WH&-~XHZ|cZ(8&c#)CD{*XhKt>2c#K0p<;jz`PvwavSOI?q)rLK{v3v95` z_``6l#MU?Cnxam=9kY>_g$xVX&Ki7m>0p*zf#hVfy28CI8gd3yO{I90ixD{en%dhz z@`7Y0I{@Rp$?HdzpCH`&!r;oakPRo9E1}|VSx8R zKQ1I1t%@J{5c?|b!$kDQYxAkz2MJ#@wJ-Kw{d}v|+~P>4)Rw5~jA~8HQ|&AL5xcin zM_N^Vgr%~(KIyAZz7k#L${ez;moFlSAl!djX z!SW+t)HWue+O&inio@o1)?LlKQEpj4@(NsVtNIF3R)f+|6gE)rqzF%N-ZtQ2sWeRr zJFS*65FxOyrR}=tvUt{(uMHChI%;uXUdD#4=4pHx5h!d#|1OI>)V{!PH=$e7U(LM{ z&a#RjSFOg0Wqem-dL(&l9DrrYt9`=7bAcaG7#3x1GLIEb;)7TFSATnrNjXfy+v7Ta++qF20WCDgRWx3 zt>pJ218pv;*^wg#KY}YZR=Nkh3I!}n%*8$7kZ?Ll&?)uy5>d9B&E<+CJu;_%12^Aw;5r7pu^olJ-Zi-xsP5_P3k` z#dOttf}=zYE>L=@-?N2}QQx8D@!&P|r`os(Y?Q%AKg!i9iKqZx0~br-Z6c|30wL6F zvK42o=W-UbvXbwWLl{hxxZP7!F0{f~;;3=c|C1M)fu^e{b<41uQ~6p1lUuj&5qAvw z>yO%RqL1RdosRjru<_hGHAHQY^mCsLgT!Y3eD{SqIrCHksK2+nce1Riw}wum&|LTS zkWzF-LZO!&2%Gs?az&~pC4Qa$wu+e&sRQvsv)S=kDa~j>fG`#&x1`%lk*1}%i=pM# z6&BVR4RlK4$whHbBCc-qEpNCJ{{n&^L_D^?fIhHJ@Xr<)_B=_9QEz-oYr)a9oGvl- zPBdRac3>{cyuuy#@rD)i1L3}^)9p~3+M7?`+i!023L6Wdtd<79jmJs%-)_AemCX8C zxT|b^Yg6A-S*8^A1dZeq$NVhVetfw6b(TBkT2+k-$j?yTSI2YoS$qf)w)6Hfa*WW)O}2 zs`aRMMqHk%Y38S|)Vuu8IT&h#lUL(V+XUlWhvE!=;;|#&GH69J$Exev@KLQ8TCB=&kXEEnP;PdIJ8q2Vuzmg56&+2~582`M&ch0d zImvVRH~X#3vy}tKFgsIAG@ob!RmTW-v((dJxDGLQQb&tK?7G{R2yD%lrjEtNrTl+v@^rjli!RVohteR;-ws7W~Msl>0mEYle}0!Nb5dxs)bYIprz|* zH4Xq55*C&ZFs zp(4Awg~o~pBjs79ZR~L5G7Sk4Di}i|e;!DEqhrV;|r$^JT zd-Nc_D$Z)5_UO3<8QLy$rKyXbt~A#FH6 zmwvs_>=fCeUCYIWz2Vas zwqk7D&3Jh`uV>IkebwD*XN3%n<*E#q`=9R)!>-ptJ8*p2Ci8dblelG7Y;tt9pO4 z;Bg>s2&mIOunwKUP7Q^ z!^@S#F~$tA)JYOW;TR3^kr)~@vtUn}L@fLS%}CY2VIsJ2QVNwQkkxpj3x#^^8d2+A zC69o(iojLMiBhvjq^&2=edu&Bs827ccvYJpoJQY?UvEFNdAUQ-=_U=mf)odRwwSgc zF>&;Jw+SvRCE_cD-oJZUOyKA*FoI%8(L}X1BG9`=5N!$nsOO{>LNmRB3Z5OXv{>08 zSdDTHFRA4NSGXoA5mYQ;4hyoM{THAgaQD$g$uFZ}CRI+i++8}UqtX;NB-nk;UZEy( zi*HPph~C(EKpU{1Nc;u^VT`OFku%dFnphOt?qm!2by>2{^m3(m!+Dt+N1vDmr=w zzKfk5Qw}_SHlrWTX04p&@U2tV!>*+KY1hGP^kLg0xH{3P*wcZ_nr&OR?pKaE&ne8U zottD^Oc$GEl9n&BRZN&>6nzIpQ@(TUdwjYs_)k4c|7w#xSZR83@PQxcJ&^kB-2trP+S%X{km+9cZ)x zww?%BY~WY58cAP)u$@$*C~faOvzdpbc#}221^i)>CS-Gu(i`6IugE2->U4eGcmFg^ z6*r`2a6;`<2J=qym*6}j4yL2w3-K3Th`5hrnN}hJ1mquP_vZ?X7}Z~>$ZI68Bi)L9 z`2|<3rS%@YaaJDFk;potM2ox|VPIk>10|~ftsWvPFGn=YRX(Dp8=?%p*Mtfh%Z|O< zAw*jV3G9Hms|18OIsLR3xGf$+IOqqx+)oQHohKFe94pfi$oC#V5Es;YTx{R{=Px}A zL1gM!)9{RKhI8a}-i;zZl7vTF1&<4mH4(Z0c3V4$Ba7n6>=K(20@m;+`p6^PjSXm- zO#c_a;Ja16eQapzoOaJmKQmg1*Z^5uzQbB^cK@p9HtiZ@-JR)wR;bPu&lkR{5T2T% zGCcU)TF0a=wI0e89@jW4f^}Z8$XGIWA5ECHrOXC5S)y~9f6W83eL3u6Oa}!h>I*LB zM-0QaBo4Pe90vHzk!L>cgSC};6m80BxPL=1=k~!mhPl`|wm#>01mZnznP~_~$(v}> zd-90e^w?XKLs;aCCcofV$BUV(aNxU2ch!MMR`$i5o-8UYC-Z??Pk9w+4g6DexyHuq z#k|zt1t+8SGVcU74}O@M0Ec_fU@;hS9CPu3!cVT_5hlyQa8Ub{=4`4@Rq664@c%hM(=5)6sk!LilBRQrYpuMC+0wsV*JsdZ{>1Q)$ zdB`!*s!!{TNXZd#sWmtqwcm(iaOf*vIWaOhNjiH~Z6YkQ z>uHU$l*gI6tTwawE*mU3PY)(9!spE}7rK$x3xLP=itH{8_L);6u?~*+= ztwff?v)}m4^;3VuJi#tDNHbRD1_4NZ{SnVFt%$+My;qUZ%isWY$|;c&ZT ziNxTTnV|wIU!h8%ivO$41il+ul)a;&P*SLrU2Gw_c^B%K>L+fTSRZ^Gp%po@~cVtQ`t2~mA+vSnnN0D@2(ddRhtN)gg z`jKCvQEBd$^3v&LAR+JJ(Ta?TesNbwo%?Q1zv-%<$o3=2{Tv$xP_IS5j(+)a@u$B4 zD$NJcsrW5y^08IKDhJT+7@zV|2Doj=>Fi5RmSvRGbRV&f5p&hw{q$Xkz$gki+Z2~1 zsC&!-Pu|ydD_322Z_nb6z;SaFu>G`r_}Z*~gWaxMQ|u3S1jOUk5`Hxgg9iw~v(n^v z(@iBiY75Eg>e5r2777Q3V1yUqt(YpWty7u~*QZ_}R1uJ=( z?~-8IC+A^rPo+`!a&;xb@v%HrjX!JV zWj!px$mk@i=bw5|_&=zJ2#?9suG0;iA6|d_3gj4JuZ~ixMMcY(i`WXfpWA_PB^TG{ z^~6m~Mi4AA2K#|u43sn!#$Y^BizNl_OO7F!5$f4KT3nNFM|vi@V@Xfujj)541Kf8l zE=VSSXn*v~ew>l696c2&VNcIytK~b~^%o#kdCF@MPGI}WriC+K+st<^RCy5_k(%P) zqkmd4TkDqSC6Rd^Ada}dv9=!endM*uIoUsR}|IMv5qG=5BGRaN%U4d zxBSG2ZX#zuTA0X1;BEc(cSJgj8ez?;kRn5eyYYQ!uIZq06lH0y&mJM9eO37QCQJ8C z#2d>DNcXE~eZ6x1Ql3ktHiJixqKX|>+dmpiNJ~vFJ{5T66agfl%!O@mu*;924-n(!POLqWjCDv~<3z_$Gs1p73{SId)O}YD+VPZQqQ$OoACN7fAJAUuV5*Sb~%m zJ55S4Nlx`-I(m6pxiO`llox~=A3>lCYcCWGhHns|L4IW|mVw1h@oAav{7UZfF>F6P z3$)i+a9Y}k^;wZ%ey8kgwJC=MS!gt_r;jv9!k-5iteg`xQ|j;@$WcBTl^Hx|r-!W9 znH}W9b*99+b6C+QlhzB*t5Z=?1)czCGAB6&8`u}d^He25Q545xHdlp^9s_J8N?4Wh zAkvZ6KOrK|1+1#Gqa_Zl_{Gj3a0C*xdCzqnL7mgUVL#Jo4R~=1AEE|6Y+=Nn%8wHJ z5)#l2sjm3Wr9a8b0%=QP=&QW_D9>($n4K#K$g^I%5sX>1^j28glTfUNjn}!BY)~vR z_Zid_P3`F|PQZ(7>J<)^y5N%Uw$g> zFv5tOR1F4UG83a_u+O$;Z*Iw{)-8X z5-yVrP9rHd&kQsJkt%RM)N2;A)Fi=ko*rL7_)fs~vhhVaig%PtQE`=Wk!r_mCPGq| zJ_^eM5r=$?vOm8GXw^kP4!l(efhUy86YUC>RmV_7GGag5_t$irFF$9E_R$o)1~=jE z%2JK{+bsRMo5FI{i~Qm&W3fe0K-^zIgtAMl({4v`Dte<3_Y7^2ks21F&{nqg|A?TvN06hRZtba zM3UCTszo5@qK(_0!=0?LVSyU|@s*0J^Pt%B=QT|nW@8QvzNQTQ@-@5V|@=-O3zQq!@j@sY|f?)TIL zzt-}{Wi1I4Rz5bPrbmHXNq5SY2@gGG=4du<&@@bM| zFGWiuJRQUhM5P|?%T0<JRZjClYxW3SCS98I#a)Y-1ZePDD>UBNFmdnDD2eOh~R2Oxh zFZ57&3I~$OMex>YD$BqA>7~(dIJ)Uewci)P=>(1Yi8nwfXEsAEEiP2z5*PnvLTDH# zT3sUCj;zOtH224o>j0Khg^g@W7XjEc9JkPQHbiih%Y5Y9*J5P3T$&ur0G)k8POaHx zMdPK&mO=XH!Vvt^X&6fW#40VAo)m3I9|SPjwF)8AEav*Ydum zUwNLiXtRbW?2qS}9+tKoLI2oeKi|zT?0Bss89E=%klm-IDbDrE*ACR|8h&C@AOLk#h2W=uO={!`@hj>%r!eLZQ? zH50y7T9%9yB_Xg}P2`S+GZ~O?eN#2S7x60_t+Nk-N$d)IKO@WeV(w45ay1Dag#ZLJ|ppXltsBc+;;~iXHe~hH9H} zPq4X(?wjfKJ0!Cv^4s9oKRJB_(~AJ;EOUa^ZaG*#mr~C z`nsD*8<&#_6B9E#do{60DGeT$%3EA_70Z}sB>Z6d$Tu??racL7WP}PeiFA#~4E+e( z{%rWs)wZCwmfTaXU<@hg8w2i|BMa(%h#ebNQyJv}dvdU)Hg;r&M_{O70B(s5w@T+% zC@Y)lvvqKi$kr+ z&yx4%YtxakaXh`IAbk5zP?nJ~q*W^8GDp>QvQn!adYL%%5&rURDWlIc_c$z2@1oy1 z3iQT}dT6UAMbcq5_S;XOkr`FrbK#z{y4?I%Mxo?IA!_A1u!Q`|A51lX-Q%yF1!8BW zJ|^7CVjKqE-p+Iee^+n2pV|TG#&NNOIg+PQpA~z4%xyzrzL*@- z)FmxSkiFDg;bf7&1s7)Us7a@Y_l57Xb8vuKCCls5oZ>YP5UU{N&@NmXxlHz|aro$m zt4-tc-IXRJ``k=p$_Pp~Jg!`e{j`YXxsRw6h#{848Y~$zNATRhL8ERfdTOtSFTtsJG< zTE$bX%eDv}r$DEW6_u*0AjpdUm^yuwYcffZHUq`=W0y|exL-3ggPHK5Q!w#(As+C1 zgzav*NngC{;`EL|m89U*J`@c;=X)T`48kg$Jt;q!1MNBlu~t&+2=y$l35dVUv+Y37;!@z;hMtu9Pn2E7ec6>wB4;!0V$!gi>Sa9l04N`Jk#C;k)5(m?Su``F$)&StN z@b=_UJSanl2|{lhZT=vS0NyV>ZhQ4UOXOiSf6e?lY9v-Wy*@0QYyWG==j_$h99++Y z0XqkEX&Ofb#ms0& z+1AW&Rh5_|T0S6F&Cg@TCo{e%M@hiamv@X)2PoIts@A%hh(HapLUYmcOHG_;Pe=L%II)R*ny@`ounf>$RKYjN;tF0=o^szw+324S0N`Cn# zP3y`3=O5*sNUO2T<_uRs9IpJPbL)K9o1*0cUf&5^hQ@Dcsh=_ z>3JaY5Qk9hq+Fy`sf_)~&!0}srL9lKNfnTAwx}R`scy8qoZd7Zw~myB48JXKJHS#* zn4t##(#seIT;lr{qp96Um4visSFrB_?C9#QI0Q~jv3>n1LPd=ADGn>QQsS%xb*ks) zKW26;VVw}&FAN?VasHl0bGg{FA7W#)5aMYkg0`ciWv+DM>PN;-s2={=1MY$!VwQoh~by9;0oa^)cJ~}eQNl_L&f*D%mfR8d- zf&@#Z*(`i0Nn*Ah^j{!&U@4>r!z$8Z%MzrXvpSrm3ws}1aN*xi&ORx>dic^g3VS*5 zZCNeQL$$I=8ySb2+%(iZJL(tjn*R9w3RK++9?W)z?qKMAiJ%SCp%K`rHJbSX8082f zT1*pF%yRv5A5mH!4mmp)L29D8Oou!SoTZHaTUHUFvf8Q8AE|7j+ zdtpn4xyjq2WoB`8+bz zM#??5b(s&G&Z-xF^!I!WxTe?goMNa(^sd^9hzGa(G5P*#V0U1hU~IYSWNxE(@W} zEF!lUf&^1Sm~oWtWZpujzwXno)ui_V$9&3$IXszg1#`*1v@V)ja5kIl#TwZYiw~SF zjmjU~W@2~2s^GzE+*|wOyGWICf{-? zHryw0I%z~wc8{Zs6kbP8vh!w_h#cFog^a2~3?yosm!vXqrZSLQ++Xy7l$(_t{=^4 zmqwcg9s=uN?}`3?1?kpjcwW;qmYqZ#e*ps3RQ)M_T?Q+#>~le~+ww9*qyyz|LaF23 zZ6hn3CE`IZeVwrWVH}^ZS0f)Ho_+Yp!ol}qDqzMlNI-K8)%?OFG_^%`Fs95hj(vz3 zsG2a6Qmtlc&C!gdx^h{X4`~>Kct#@JF)6y&vw?xF+{&~=pNGML+<0uK6pH}&g-cX~Qu2<(4f!q?sHiL_|C01CO_3V6ufM62nem1& z+8}y;RMN{-;VvSANE&-OvE@A@)o|fU$+oib|3wiQ@#dY8w9JqoxO1*&-c6I@JjrTul z8q}i^kyAjC`nqUx3&#K8DxjV|GxFZdESNOPVCoCp844~rlnfQ7F4yi7W6pl9C?)Lu zsnwq^odIl}ix44wBGcoam>n3IpH$rRydWP%oWHwX1LQW~k#DVYJ( z$rA*hmRbG+tjxJk^*J@<@G-OUXHSP=Lg2_tmG4QZ+se>Kd2$tL7?$FEYVF`D)7f%1 zt9_>XW+sZ!p5&HDy%E?jrz~?#p|e(EOS?iKi3m(k&R=p z#dtZUD4g-$7BTI|t{$C0T~TH{9g>pmSBrAdcrF8vDCM(%Y*GfKzg+3yQ?%3+cJgvwG<}3IjM_o)eT1@4?c5AbuOZUjPbKy$IG(hJCV#>Xh2OpSBjl9uP1ynyZ%t;3`jD_f$@Lfp2)yvYW0b{dV@*A78rx<#zva| zG(H%ET(UT6u+doUkW1-*-@MRH_06tW*R)sG_^pY*K1HkaU2%(0V5e%HQ(b6-lN1O9 z=Bb*f!1SYrA?^-Kst0gQu&8QK>OaWgQ3b+`u;h>@LJrS8sU)BPf&(;C#J#js)( zt-)?as-x5*Vrg0NwyQ=WK`c>+(cf-dow{6H1=eKQmMHDf1#`A}h)5>pNt1 z6u7Ju77PYg@*ZYwR#LME<{8Bj98x^JOgh^$SQ{>GQlCCdxclTum2+@2J%nQ@#CX)s zvABqeiyC7QA2+h<^u15zgXj6>kA3LK~0`sA$4F34eVrP-#ncB2M#&eXs> z+B2h$RI7-5ZB3 zeUHedsQCc4_vBItgN^J6evU4h#f)_|tiGO?9G6BX!UP91Tu}{KKvV-0ukl^!;f=O} z0QShR%8b+i%w@cDw&giSiitr=?-HUTGY2FSsaQ+W~>H3 z9=X8n=Tre|h}{{bv<2-~A*^L)H!fT>kkma2CYZKBqZH4`*Vpd+4U(Y{eRHpnh`@|< z^P9mBVp7kMR4@Q66Q<>jp||cXWOsg*Y7m;kMGsNhux%RDU*dup$wMDRUyv6F@><3H za&dp9R`2vQZdb!Nj)^;D0+*#RpyRKu=zBCEJSsCzFuNLZZ04Ufx0srCesk0zpvpm- zws?j3c$4FB8{V3VihIS$c|1@)y?Iu%gaggM7SGF^yYKNov84hytR4=J6mcvVC7W5K z&o#y5v~{S0=%Pw0V32|F5{~}i{VJosUoScW88G=Jx3}s5ql0>K`8Db`HI;(-4poZA zA$Ga!5kHWBrTPb4!0#)xmY{1ZIjejveL9um94p*`yHN|?jA%6aL76|dU$$r4Fk2Zp zrrE%6$ZuyO_m^X{>hD8%?I&DXCH5fbk_EDrSd7072JR^YzGk%g8GIoWP-5n$m61hb z?*q&*Z2RiRm8_G%`PdZ^YnhZP*v2KDEVIpmrb}ME0sS29El+O8?c4)nzavtMaL&ra z;P&LFPqRwpKM?}rFPsl(&nN>M_nx`Kqkd&IM~k*PHdv&#=!#z`=f_sBOP7>-=D6=b zW^YleEt>^Hq#$PMxyXL<|I` zQpdl0tlj@w3tj^R235rIzR%zP_9@dR@fMD?LSY+vy!8d`Cp-lizH_bvVnX2Lgk&lO zk3Y!{M#~n?d00$QEI08E%*b}YNs_`1ARG(1cN$)PGJCBWHD+NIH<0x<@l2ex%)R8} zOL*b7I%$t`zn?UmR;{HB&$1aXh5;RlXZ>nOY7*rTHLC!xxEU30*K*+Kn8rT!zqx5^X12?Pb=VdAyNW$?+4$ncakWK zQ9cVx6!HEC_)x_J69{Jj_<%$eQ{l@!P4!Bq23Kj@SS8*Ef`6l}jN&2B7C1*e7i90R zz4Vwd!x+K)lgIhU#G##r3StAm76;?|IG$%hFQ3@O!P`H$wTFIv^CwljgX%eDMelDk zAG@5ud}P4iB+foel)W%SAWvEBota|B7nd-u_03mVfQ|7`F=vteH>JmO>i->#`ZZ;pNUCfQciHE;Ol7$A0QkY$DF`Z z3i`iRWe(2iIj;mI_mgdW5ruUmoSZ^nZiax>_;d?@I58HkeS7Z%kDV72wGzRr78bVB z?&lFO5o9IxBD{=o4*C8F48wrk!5X!9?BTxLXdvcr$?(s|vHb*&`_GS36{7>umC86T zLx7v_NLn~@177t)xYc5x?a0C{heXIK`WAH4JLTo_n_~(ZxaN3sqjUL$4 zN2z!G`*ValAl^o`YtOb-1#$d$s_5`iEVl_1{T^pel;peF$sjR)>FFpmxUr>HITWst zr_9r~yDf_sWPeJy4mKKwsW~<>WrmUFrhPqv&+-DxSPBQlW_w}|zc;FxmB0gxw&?iX z^3ipP4v`tP3)xMlvLr2!9_30U(64`Bw#V`dE9Honcf## zD7dFI>=~*A@CjN|3w&Fk^SJ}Nw<&$xubpw`By^WbK1qB_SxSDD-VBHjM>J0-bp^4iuMc{EWaVSAM zR%K(P$+HGh$L`SlF}hKFX6d$$I@JIlkVg~P|G*D34dk3&vXsKctT^iXJX&yl&v`5? z(9>++?=q*!jL4a?*1f3xk$v>aEsRULGD~nkoE8;|N)LIzOpR*WQAg(mh}Xg=*t5`5r|Fu`ni`7QpcpfM&awBXPXj774>^!j=a z>kf&V;+M{#x76Xm$NTU*@uaN`-<4PFr;?eNoLsN6qu7@es<=iMfvQ|G2*0Hkdj z!V3ms*;X#UPT2;i1^Jx~x(pWF{HKPc9^a9maeo>*K?h9TB)@>|d>%X*k@E7JodOYP zVBqQX@Y6|D%iJNbl&pkP_3?(Ky@iiJ+SQ%|nw35KUU4>}M=9ze0Hy`mY6kxUn5Q@@ zs>2GqsgUT0qUOg#z8sH1m&V4eryKcFYx^w5MG+}?3UeO(aegIdC1hl2aei*^FZa*a(uam7ZSIKJJZAETcepV`&U&msvvB?h8ycM_FB6R{?Y&0eUhz051f zOGP*gh5!f;bC$i3i%Ot94Dc2Z^OE+aNvzsZEtFzQ!=4_O_&W4=Yw!4WU0reh1Kj^$ z%3wTE1{Sn^c0&2R&&CIshW9IzWn%0R$HHpp?$eME?mtak_Ou*BtW$B4VGmObWqIv3 z1-~-J>I9g|vZOfiXZy6{ORB*T75oR)pQ+Am(d$~ix>-4HAorTvf(LR6^A%9Jad#_O zuu)4Zu%(()$GNwp)KDW_U)c%SlmwO*nvx5fQ7UDC>#OXx5?M+l_zA?uGE1h)&3}Nr z?D{4!uAwm%N)7KlZYdQCT8!c_L?6XD#xF{+P2;MK4)(+*QK@QOgylV-ESo2OmN1-h zdqdQJYdHab70*I>quTX0uWja!rZYxxgApI17ei28gp`MP&zUYaZJ|f7s%K5$u z)r;PI2D52O50@2dcF(EkgSTAVC;7l9iN0H#Ws-llWkmF7&4Z=x zTIyGA(_dLh4sMhI)#>ZMG16pHyT6oCG^*wog({SuOWe*fS`qjs;cntV z5Ps!F+==lCXu?`IFUa74$=I3n6N(-V588F`edFeLuRd5LEi5eldWd+KGxy|J+Y~4friVW^aX)v#ZQTa>yiI`&E02yoO8*t}Vzg zo<5_(#*a{hx#^RkSg-(EOqEzhJ)t(<{vegBBPc|-OT|md@!L4D0dZ6{>JJR7fwn2# zEw?S2M^^iF<< zFVjF;rq;xi89}}~zP5Rm`$mTi=kl|jrD(y>VC6qCfe2_t{VXoQe@Nakhke|S z!?S7o&Z~yuc8l5OpAq9csj~>}VpblgFqjCv!CacE+XxfZiEt`^w#0LA=I0%2}&y}{2QkhT< z#Ad$y5K+7H?3lhYJ5wui&&;59{Y~=kV>)Zglv#e7EVtAm@7eOY!-uq~(_A zF&}&eW|g6k{#tpDJ;3e_$i+k3Du$ts-aA}d*?+1G>GiSCx0=ce zQxyLJ7`}NBPrJDS)w=&mrF$nH@2~W;mP_YcJWc9J?RwQ%e_yy*f$vj}MMe<Q?!Ez?5w*VChhb{*SBy0 zfhb8u?ee`YxtCo<-{ikI9J@YvMO!eZSz#0hf*!I|81j)O<|K#McNcoUf3ln77A+_* zBx>l%e<0P470h|e;g)OpIy?2m@}QY4++=D~NP&BUdt*gc@+J6#s8Es~D%4W}cy%K> z@I5}aq)sLh2jZZbB^e;8WyO??O??(Pa=EQBh4JopX%s`u>~g}Rsvu*O(egD9KUk-c zkP24{22HxOYqp!kKPaqR`bA4~b8iEuUGv|d-AHe4_}d(7#K7!>Y(AnGvo9^HrC^c& za$}7Q>W8=9RUPdbGPOzOb3sVm`H!$tcl}xPaPsE9Ca_cjs8*v1r2S zF7$pE=Z}lhlUoH*yCwDe4I`9%6Lj`m)*94yYhr}%LCxL1%!%y0A4z^Ye^|cpxa{Qm z?nhiwa?^oA&?oHMEyJDtZ@!LjZq~u3X2YkfQvCm03eot5$#53iFV0MNchDS#Jt>5g z%f|!m+gYvgY1VF0K=%Fx{mwmvmhr9Dx}%L+RAkO4y!X_ zOqHxIt1XPAooa|1gvm_e?R@|sg|?I=*@=5+sLm_{Q740d7SW_1j^+0nD_Ok+-qv6$ zV~me*CM9zrzB%&rddG?V_M`9bF<8Nyim|9n6c&gnB0ElnP|dpU@to(ya-2hbn~q4p zx|x`e?F**lSw&BZ{;Yc;CJ%$+K6orP;&0-}9L3AXXcIRSgohRSOQ)Un`kcSEXdUQ0 zj_j%4YbpmKi$IQUKc%FYN5L0K*LQ(OwLg9VZ@p{Srv3qTm8d|jAcJm< zr0KnrQa$T3Nd`ESm=nEyFN2ey`@k{8I^nZYw>xbc!{t0SYOw_C=0bals z(oumGy;E@OdPSO);|rUGMGVr@0bsok$s^N7 zM>EgQdOBDoONaFtPpp(Qzdt{~V^vM_q|yPZXIr`r<0BJ#Lvwy*eGLWulf^pDAW`81 z%~X)`PgP!)edo1>q@#e`-lvj1Z7Al_RzT;R6J=JOV*A0O{P=5nN!-(9i5eW#JVRWu zR$Yk%Pw}Ta7l`hZzF&JD=icoOx)IHcYOlORB?$>MyOBfn#M5-kt8w=aqaRoi-co;{ z8o~@accV9IdZxml9uL3O*Rym53b=XT5CpNG{THlZi8n}4_I+3i!mX-h_CI_{7OoT zWktvb_^J_*zOt-L)Vil}wOY=8yf6ef!ticGaA-=+2W=+jyLEm3&T_S)-#muYJF zdGaSw@DBq+ftgRGcbGg&=>B+HyIH9SMOQ9qVIcu@sZH_-QjbvFt3fIOz5{8DJ9tRR zv3B&WL49q&7Vgs8dyku-A2nu`u_MkZqD_Cb_~XaPw*v@rW&YF#XbO9&GE4 zkE`R+Nnru$s0mRm37&DAIiSz^K}vM)oeiss0_~VYY?Q;h+&B4!)VFl>Rio(bA$-IE z->E=F%l(BzF$4wug2$cc*=sQtwP(oNa{&YV!eHf?VUw~FFUAZe?Ki(X&=3hBwZuxB zTUt{M7fE>fbOuxCO8Te2GA(Veu`l&cEaN$-dqQ6)5~IyEYpqE_$EUvOOai*LUE(tv z$KQx9K)F_j++!-;la4|p`zcZC$K+@%7 zapwHii@`-X&wNz!s6g^N{xU7-y*}#^i$r%BdM0%uTuWKKtLkr}$*kxh&>nux{fFi7 zhyCB=7{oLSnxq7HmbCkK$u)i_w9G1uUzDE^UBL<8dt>QR;xp=h|*&6|ptDev$tJwT*$UGlA-{j-5wg@#< zo%Q-=TrM)W^1(6iI=?6+)<2o^NfKQ+3vNy6xBKo>{3OuDrSX-I=wv2=qW1Pl3GwqGhCGs?I^#&hxWkcfC!vfD6sC^i6ub zr$vtVf*Nt_Cy38xBdbkj&L(oI&aELv2|$)#y6QZ#Ja5ry1kv=@N^3b8^|>P-NB#;* z^z^)QeOAE+YWnzrV#A#rF239$j-M{94p1Ul*Y^=do{K7ueeX2w=b7V`-A`7wfDP~o z2Rrp1TTDjhh2C_o>6L*ww&~R`-AzW>g}**m66pP`*dXF{)qJvb{u?D-CBp9_NX}!K zmP&pfX1vE@m>!ub8C2oIZ3r9lDkw=i<_jS+p|20bYbxG4HlL4A4Q-4PMpm=~Uqc;T zblENd)R}&e1rF|@uB!1=dejDq*x=E~pQ)|cm-ER)tt@~5u<1W2i7fu804iaVwkpKY z;(8MqVTY&G8*66%+`IAW|C_S=Nn1$w$w-Ue@}UZ@R*k6i+Pg$(4D{ z$4AWj*aEH`W{RNO@lFL)&p~x--|H5e7g-iYaO#K^qg4_6Ito-?hUz?8;2EHbL{o`& z2LD9WAVtLVMu)$fwdOretT*6V)eYFIgCYV$%9p4Yhq#ws|+%T4t z+S)L6!nBk4HeMD&Y6p^kxP@a%1Z=AJ0wxLoBKADXtSr0a&c2g{>G=Q!A~G$0u!}j5 zIe+tT`GwR7e=%FpiuS44Q9D0Y@TUehm0(7tKj!ih@t&^CnZHAnj|s37@9L~Bq=cbO zh5o>u@Q`diD7}^LS+MkC!W_f$h!L4~{lIl|b}_XE!iSxHO%|D6NI~gibI_TPWnfS} z4@)hSq~aL3WC;1oTJ09OdsdUsYr7Rwx;vbPmqsKz@MZqs?CNBhQ2Aj8>Do-`x0HCu zDptYGhQYg0GORE#C({pG4Qy206<-aCbv!^F)lx*Y^?&rX+6Lt94!!s&Yy{OY^9Fn+XgjY3JEi{{W10%df}L5}MBEQHt!}uV*X3 z1Yg2in0^=jx@_H?$#jqoMAV4mSOj{V-D`qP3)rc8SwDl`q|miZNFf=XsIbL*cBb{BX|z-XMhS{RJs5`?{8J zH8Qzy6si6%fy{Es zxT4ip{M>PGnRRwLCwh>_Zh-f|v?Mq4T&fpEU)xf5G8*!Q)um|>*VgBu)@|uZWI-p9 z3frGkXFPUy4A^JEk*L8_xhKhu@B|%1`Z|1r$L$N;2?q9>Jr$aa=Py0{)<3u!EiB+;NUR3-4A%G!IcO&M#A`?Z0!`LaBgv^gCY1ZBg1PY(}Jw4=3vmPU|U2^ElVOAUOq{nS@~%H*~6 zXrYsB0dlvMSgoDceJN7YM8AB|YHmP%o~H}5TcJ2`O z!lOO2H(tt^r!+D4D~^;|#3DW=?(nj7AgYNA8hIfXtt1>8Lr}{8*N*0onkDaPShXO@ z%%(NGqVG5=9m#G8_t+>*R*#SW1CW+Ae36bH>w-a%wK-4x$Ocew0A^nW``YnbODU;fZOXHe2{vSh5plVk9QgQ#l zbX%%>&G-LuDDcj|v@W#&)*?tT%eSlRWo(jS&e6Hm!dre|GO`WqXFvvoyZgDK1gK~r z$fg8%3=@1!iFe}@>6bQ^_|;L$hXTZ9I#*FToDJhvX~|>#Oi->~+(mn6<_+i2uTP(Y z)m4XnYSZCnQi}_RU}{|H8)0gBHXOXh#z)9+;(f>h&sN!w5kU2+bmIkaxwNwSfxTw~ z$D0}1?=3wr{4ce!vu;E2yBZ%Qpc*d>VbQl0qtnt-J=0b;4A$IpBgtYbl)OaeQN3{r zq7Wk(xo=!xxh4Z$k;riBS~}zof+*OHVgL>6B#y^4ySOY+G2x^wcbd zddw+diqi;@4bgFbJevL%bTJ*nrz@te`IwuNPuT5jTWjiTY(n#mL{R$YfzmTnO=2U! z>(09e02S2o7j;w=T15yQk$M|3h<(_J8YC`ITcG?9#v0nq!~}177Kl3p#J3q&X9$bS0TDySKdAHx z7LozXVHS(gGl*5O196u@DgyE!K+*lzW<9npO%_Lct4`{+!qEi+hh)FgaKwyE-f-*# zruz=S|Nfq_ayfVKmh*T5`9!QP;A6kdcbCi90*(Vgx`|c)0EwWSk9~ELV}RM>4-z z!o{=FDd(QOMj(@CIj;MM4R?+WOxN7LQo-9dka!CH@+QbSO+XTDlYpqRptxcH0DedO zdcAN%*lwVBeJTp3f!d2_kwWcoE+($mX)Zwn$j7Qg`55>6?uB_VZ#Ha-?59TxYtN3s z;TdVCT+_6X0|DEspNr#BXn_h9H2phq7A^%*j~NIuGashoeASt%tU$!e#f8&*XBgId zBX7sO9~A=)9Pmt}oyX4}?&{V?ffq_)moU^ESIk?v)9OuhY`J&PeZ^)5)bS@8a^=&2 zZPdcTEz1D4?$AyFr43nXzgTOdgdPbojW|m66`2bz>+`xOZsqRcQSXq2x8|3Vi^BHh zsEMgZeHII@P>|w?!ao2@USY<9rQo8SurTdI-Ui!%OX023#+5ih^>kY+0VQ5du|qs| z6`S^7M{Z6D&y$d1ds$IV8}v4b!2}#oMcX?88bmeKz)0r-Mo`pZb_PNo3)w6TOq)~+ zGHwjJRHJ*nndXxdJqo?5+Dh#<{L=@Jk+zCJB@Zj04xqJNIQWgJ=Y!F8zqE zJ%YLD#{wxIEVYSSf# z0xw-U=yexD?;9mr>#)f71@DXl3G`{#sRnBIFdQ}YRki7AzZ55J;O4xCGaaGXxc{<0 z1o&8iTq26OsFO+6%CwTl<5!6aP0$FPpw*)4P)_`?&VYp^OYV6db!x+D9hIr}gItY* z4i4>g@K?BZtT<7i#@vEsb;%(t&lhyEpP*s`MVv=O{FDu0y{chdoy|5cM@?F#2Vv)3 zy$^r9FNLEV&Us9uKCgiW)mCx|@BQ(jn?2$}g^drsQ;{mhEpQx435S?P?tnQ*&A3WR+BKomx%S zzR(3I7;g~_@Nk|=re7ENoE+_zKJ1QRiA)M_F+2Y%))g@Eg5|AYT>iu`$K$Z5zFq|q zCf?B zjL*W<`-j z_?qIQQ4f=vv62c~&QTx3S^XjvlN8*D>w01lnD(b>#-QeF2Iq@Ow2v!|0414!yhDt8 z?uROc@71ef-pR&)zRL@nq8b^AIf6T^d#f47^JyoKA=q*|d&R)r0Ub{RFAZ30!rht! zbiY!?+`lHq&eA-|bM1Xyo+CL0EH6OFR%ghmhgC41Cu*Qacs zLqEI`XBpV8v|G0@zQesFUiGcXvvL>toKo{!Q(dj6mbOY4%Vu(X^PyXG3e5{+EKjVB z4M0jxSb1=zr+Heno5pnMw?0DRyK&gK3#yxzJDUwgW#q@_3mT-J4kaN={MF)zXdL|B z_ee1_yF)UDIwAg!Orx#ln=?va%)7jgPTTtZgCY{1=+y%8y~p-sX#cwpi_?=60-qWU zy_k-gpix*>1swnMz6Np8Ouu8u!2|7{tWlBwoO;vh2t_DP$t2<|($gYw4wpyzNIu#` zb{(k$Ls}pCfE#WhP8wS)LPl7Lp(?PgtRpwBdCQ4+bDN=%3gQXQ zH79r_)#shzpPUGF^s>sGi2_#lQlrm?)1Cp6`X6%D`~yfoPvbVld45}yAa{=*<^SxR zw;n(FnRJ&ticdINVvs;zou5wMbK>;Z$Ij?QyQU!Y4vT1ykD=wT5tAfaqtIk(Ls+7N z7^V0A!i=u(FkRVX;Tq2+j}C0q)RZRS2q4H~HNdyQ>vQmyYL~VS-%q!co?Bn}2aqU& zjmSvcpya?s#Qf56KI8w(-K>nQL|>#Oo6WOem5h>Ql*39ujm@3i8>5C=Pgr}u$ zAvU*+&WA=!loU*mg3A{_7s(508gR7j({c{5)NQMxuOffZ$j9s4AM@DSb?xR3=a;@T z+){Jk`h<;QYK=C6AxGUIc|=a|cOh=3!sk#IuB)WiCBZxuj%w8jb?_Q}U~J*#`p8`_ zzpA6pG)14WUO^lO>rTtZI4ge{zzu6yzbg2O%6pJqef5@|Yp!S2bMca)1hJlwBrdqR z!3jKNJ!ZMF$?9F-;%u1?LYc;P=Zdq28dV5}XPkuTSFm~o*xNBjcz;8N{{sXyTa=)x zB??*6eM;%-Gv+9JZlPb4$g4cMUr@p=;p4&yzm;xM7gLS1(thP~P>eyd-b7UKJ8LW^ zvOte0jou2k4Hc9(J)i#dM{3F&VeuM79z$=;CKHCAh5vU6TP0{!ztWRO3S^4C>yF;S ztw8o^LN8)Sw1?1&RY{;U-lJO)rUCZrVVS_iT&FBPXW)(F+I9{E4~Iy6hdnXD6j2KP z$EPlZQa`;ok0i2C1{ptPa%WCbqO=~;(>G%#(fd5RFvo_4>N00I3kVwfxtl@~nPRO? zNwXrPJd}rD;*uy+cj)%Ku<;U}F(p6~lLg;S0jz3t@oNw#&@`-DeI+xfEe2o7+8N!W zE3=7Zx|9*7<~ZgeN+(|0w|+PP3cW%8Q7%n_^x0-A*EM~sPeNf>B;@mn9o+7?8E)yuc9N_H5`~Cx1 zw4^IDY^Vf%3tuynlqVf(^G*^N+e6@XOtlGQzS`rkSzV|zF`YEjBZ|R$Pvtyr-|Tr| zrh_6e`Wm|E_cDzh*v?N}f62co+~6a~|JJgEbhEllI#a8ffAyo_;vhrBL%K zP5t`TBKGGL?|bFVZrNn<=%K3Ue;QN9DBn%T61PssJ4DF?2Nr$uMS^axY|4D&-R(WY zR?&7RM&IshI3>2JWtXsi2N?#?p{r`8B;`j(k19w$z^xDNQnHVgk*6zDv&mCV zzT>Qx`#4!)eqfvh+$lqfzH-wF+NCMgbQA#{3JC{0IE~;KSCxh${NiExJ%%A$76PozkjvdW355Zft!?p7N(#BaA@a!H29YWwOSW8ZqgX}b&@!i zAlHzp1mpHM-FdD^4IpD&gGSfjl}o3o{ZtB`Fh3eIsf7$Ix{tIRH6M$RNNkTT8K}Z8X1gZwWK$-OTgxc}Ry9U$9-A+@; zu>dJpVZLZH%lbs{a_Zc!h^a;CeMEWhn~sT+0+OxTgXQKi z^E!~Ys$TJi7=Xb9#_xIKDW2rNlSLQBC z*DRoGUdM=lglPPr)LjOEN-=KX+^b_O(6XS!h95sMYv`$nj>0LB*2yh8ac~nsD`Yxr zB%DT#tz(KbvEsQN9sEbk~gM}#DL>YFHIIW$$%2aQ1@H++K5!U#{}#6&M%ZMR8LuyKGJO3y=SBqS|P zPlV@Acxohw0rS<=0Y-aMZ=_@@sn6b`C1{H@x&n`WeVq z4t|7}0Z}CjDjICimlg;$ItreS#t4}C)Q$gOfqbjoC`-h1z1btBF^v?tJcaTd)=rto zGpc-Ze7Ck{R;|{U9&~;Fv_0}(qyFyZqWqnz3h`)~(H+Qnz>sLY5I?4F` zuW0)+Qa5^{XFyuZ&(kVTCa%^vDIPj((lZ0W&AY{)?ue-V&X|J?@QH<-%Jr*MOeeY* zs#>?TC0A;v@e}N%jLJL!jhHn7PT^Ul@*rMi|4){Qikp??)$hrwQynxA>@j)`BF7e2}2RQ~{GznMU8 znM`1OXdSxJrIogoqPQU2wfIXczN%M)_N;k==8BcD~6X z&D~DgSwTBM_#O4>UehmZ!*`IDbnYsgLc6t*kT~@Gfe#J_5Er2Iq?8Q#yXULxyW&XK z3`e=KC`H)>a@li2>DV|8-Q`WOgD#o}avd634}p@V2Bwtq#cKQ-@!7y5XpEey5H z$@oB-|B|eeD+E|+HxGpjG!KsT5ytEz8T|MI5z?D_Qb>6w@elI15sJEvGR2F?M*N@9 z8m*l=ddVvM$aQ~+K!Z|$t;fk&+Y$kBQZvr$`n=)vkk%pC_Vd!@XYUSvSZ;~ERi0yG zDZ@ZhLxX^goRHx4k)Fw1b6fWA$f%v>7@krqSPoSxu9{F}Kz55J__8bzh)6}D5}T(f zm-r&iUKTCgl__oKchO8%-djK_X5R3?X6Cnx|BhjH^nG1lKqT~Dhu7R+j9J7)^PInt zOt%u{3&YZ#VZ2IpI;6xc5n6jg=%mg!4!~d&x;~9x@SMrnTfW*>N2ba$5ntM;u!T*ElGjedan|lP=S& zvYVs4Z=8#n>zF`2 zr+pU`Uh?jDy!VQ}l-2OJy0oDV+6S)#IQL=myoxcbGBu`5YVZ|tP|A#u1TYB%?PT4{ zC`jMK1AwN@3HQaSA07@0Oj%&6;}M^L?IO{}(z!EVWR~f<$km)u`@~xq8p$H?HR&0z zEJGV4AG5k8jE#!-`EJ+sxPrGb^i$d7`4w1t3D4Z=DnmWBmd^hHob_iVD@l{e<7+;+ zG1gY;78!D18Nb~Dc_fr{9%Mt-K(f#baR* znTQryrOrUY4%TwT7EMc!8~L%V!?8Y%UI^CIkVbqOGQE|~{OneuS%aCHU83RB2c#@^ z@nU!qEy^es0}9G0j>33AXVHKB$?<=~|JQQ#R^u=1B;$C(5V-Sw*WzmMYi28gM_g5XdfAm;2o38@`F9myIKJKqB8+Y$4b`&jcAE=Kl8)3ais- z#Yw{z*g1;?kG!4CI7RH^x?=a-2XL1Fs7hu^bA;ZE7?D3B!EX5nzZ$7wKltQ9E4IVg zmY3?8z@H;yw$B5a61$=oy>aoJA<^~d@Y~%_m4Dw$&|4YLBKLTs(AE&d51c zRxw`O`dHP`9f@a)RBdRiQby;VY=iy*LIW>6*QX@p&0|N|mb!+Bw<*CtuHSLTFi37Z zDrz2>-oJz2W68My6g2<5n_duvSy0#eGCnKx(KXh1_7%u^(j#juT^)Muim;K`;vcSR zTXWWfMR;TRp^Ub@ft#;EUS?IBtCv9ew`g;= zGrAJukBP~G+QPT-<-asnvmUT#mUL6UonG5(yqh>Q%2&hqxqrAnr^Vg2Syyh;SV+ho zHdP`joF%=jYB49ugnb$)lO$ys07jA>Xix0@D3wRDtq<{Bxf%#P$x=qkCQHxu_gaXW@kc6|D)3hk`jG~Fyu%STl zNu8Kq)J%!A#DY!ob1n;wgF&KhZcp{3#JQ#m*GD3MMcHCe$6rln_X4tX!pK4;eXFPq z1#{P5A#;U1@8Q-gkwxbus7SL(8zzY;LuS6TW8*o@zjBWNw|8i+(e%G8GDL2 zeZ~drs5+#_WO5-;Bnl+Y&Q6tH>khnCC-B$R+LvTvi2bJfThR70cv>5lsk{)d{4l-7u z0|LR~AR2YPqYrlar$(^xA9&;l)4T?^Ibd(lLS%H&ls*Ahb7Dfw&v`L<0aX-GDjzo? zyuIpv9M|M2ih@EZZq~kM=Gn8lPyDf?%{U=UCHXBcDe8=H>7;#^Jfi|G93_vT0Ed8w zz4ube3&oXNPyC1Jh0rkzo;-T_fP&Be@vDvL{hQ*3nt^#ca6URviykBXRKX&^?cHb+ zv{l7T+RK8r0iPtO@E_ozZg~X&*PffOvkSUTu60vSRx+1nq&!sV=$NYO>b*-olp$`w zc4@<=RasVYtCodHjxXXfJXq{gMEd;#A;ZA~J@Ax=#Me;wVkyL`q5lIAC&chByMkyJ z;Offmqxh$T8a>BZBj7%<`V+*07>IF8r_c@0mK0~5_& z(DT;Pj4%4Vi)OY%-KjB?ab#LW)+?s)ochmGUt&tS0lmzy9^I1zkF6q)D(L$X zwJUJg?M1>#@E$NpBIM-`ZI&CS9Bc$}d34MET0NlWL5%2IT)ioy7FbU$^i>KA!)#eP zY?paF`l8qN0%LSaT2*%zRyb}N!;E32vrI=~a<2XDS|vhOV*Fo(Vp47yb7p1Qqdo{U zeGui+2;KQ+nY(5+dVfwrkPPNiG)W%a2;5NG;MJylCp!Lr=s3%$CLHhKj}ifCP&x#J zjZzS#rP&4&qf;qy^ypL?=`k2c*9Ht2C5-NFrKCF~B_!RRD zmE+~IkeI7|jYv}X;$gy<%IiK!2_03fO-0if{pdR*>;1|G1VM?!xGXsWf@6Old|)p2 zVnT+2hn3m+3&hL^ETq{X;c#EQFo6ivmMT~X0tgGWcz>0>Hg_Bdw9ew?5uxVc_a1N! zVd#s~)zwFwR;B%X=&1ral{YaXO&7Nlf(yIYefoYB)c0ooemWTm5|g&Np4tI*$ET}) z0*ob4?C#i-;+GPo}MQee{*OF z78aB>MX(3EkS3i)SMfCy2pC5tt+vH}zZ^yHwOd1^!W^pn}JJ#_5MPe5IcVr}Wd;rB&bLRBI)1 z{h>{%tYzALDb_)(je1cXlFjaW7F2hIfNr4;xx$`OzORBR_<9~Z_(o@4-^N4s6bnvn za_aoFLyRPrpy_)>uA~-J+p5?kbL`H$xt}pM(y9z{e5h6-aQzF+?EnR@?^VzG$E3i_ z;F?lK*{yrpTaqp>wdeoxx~pq0Zwbpm9CbH`7#f@$4tfc0oPU)n@Y`!y8hl+zG40{DP;%+N(7#Z1pzzjwZ~7p-%H0 zfa*I@7uPch?gHH~f7YSrrZa}7^)37J!-pd?*2`ao`J>WtP@qZwe(LFu3iS33Ycpbu z9z$3DH4XP))z4bEKAj}g_aDq!$>YY1Jk!9_b?4!FE8jJmRarAdw! zvPE+1%D(RSI-n}6C)$3xl`s?DOkK#p)J}AQ*AtRt6=l3`JXsYOXj)4@1y(q$x&zhe zG22{Ll?`*#T8<^`uQ`EQ2*F2n5mplV)$j?pC;m(D$i8mI*9zO(#?#_J6P&y#l%Y-H z!;(&nDk7}o5!B2elxA$l2_bE#q zxBj3Y5ZNdCA$32Iw{ze6S`52gx+~?4GUKX+)g7pVvwJ)4U|ADbUzh7uq+t%MgdeTi zHoTt%Bkt%Feg#qJdWP3=oRu6i2ZIC4&sRcsv4k9m#cW9!@(x6*e>{#86sn)`WxaRH z+Ue?#0l84r2tUbYy0li+?24@kjijkgx}?Tl3SZ4n*%Yy7q!x^xZqzTlK%B@ZVqz<^ z*`ls+nQNm+sd&#j!Yi}5-7@C8mHb{WMhH_XH7*>VO@OI&8E+;6HT-4jPMCKCbw8Pn z#t%JxomW_gW=?)HA+KT&k~G_Eua=7#3@nlQPs9A5M(9p_qkKuY29v4s|G(n>KaqK8 z?!M&b)*?g_bGELtC8Y ztYmoG>$kSbvC-jEu=m_4&jn0Ebbm6Soghl)&*RI!b^7G+`59GxR3E#y_0Dbc1CBX$Ul9q-6J!w8A zQb6e%fLtXu6jpl=n`T7iI(gS!{sZKR#2eNzZ8mRj%;Ngrh+~24&GwmHdpO0~_Zh=q zO|4NSI5vo`YP_G~P~OwA_+yPH^uyszM)kzdi>FcheB^I6(Oq(T{{STwOkYnl*=4AB zonF5E2M{O3IkPv%Xtq2spQvB7ShZL&W9PyN_cC_zfTwEh<>zmUH^%K}-xGdER9Au# zbIIyxh9{>OVo!*EUyGz@m?Xh3%L7=`*%Y<)fso3d`#ILvr}N|a54zX;&P|3mbC^3@Jfv0jh(E@AXkcUWEKdP3cMcJ6K7}_W_b~&u zl^!|IDT*!&CPZqm^0eyK81w+V_nb>lN3dQm296s8;@Zl;XCz}tbDcScl@9UygcynK z(YqA<06cDbtRL6TLs}?Z&pf=(vTw+AmL52tpmSsY6g!4u^M(;8qrGjw3$xhQkZ07a zdREvv@5JVbh$t+4S~UXI9+?|P3sR$;Os>%$m&OYM&9uaAe#N)YQiHw7qKD5Gd~>2O zk1vStc$Z(dxFS=tJrDRwZv7~Ltrl29zp2%Hd4gi;q|(fQU_PzLtt($?|1VXX9+g>{ zzHMgHlrfYE>)xr(`>k4p1aTT;`wI*>^_NOpNCrmDnc_b#bYE{>qGU+#6H(nXwHbf z%vx(^X!#@gG&B!2 z)6GDP`q-3#BI5caY)HSvmz{$1-8nedzl=k2t41Uf4}v*vFnjy=UJ2}^Sh&n51TsiK z{A1PCdV@ip0e;Z|OIEXtU%rj8;mS7OhpU4;0p1j(CPU(WlT|4YpYljdG-KqdL0|?R z^}-e227cK`G;hvtR|rz7eiA>mvLX*(LlN$+Om&lrq(AAXK#*I7*VQ?tNLo zf9d*C75jABGMWC1XV|BEcm}>vUP`bB)Y8LbzxadFB;hE0X4NI^pl%4PlOypOBuk<>fV+Hb^oyY&l=I~A1o*9kyf>9(}l`VMO& zyK3E6R`^LXJIi}9(;526CD@*k=aWT#Tj)w&+X#QsVb@kpj>p>l_s)LLOWHDt5DpLC zE`xWFAu+!XJg^O8FEj)1i4Zm{-S2HM)2P#rJESH++M{ib8=i>2&@q z+)VrOCM(QOca{}ad~%IbK_91@yHPobbX_lE3eObx$#*Z$*PW5)qSwE?2pN0hh#vmv zW@o%=|K@yO*kJ22lmuhZ9P;=_^L?Pn5xb-1KBK^9-Td^3v-S1-gp8`G${4~39S04l& z=9PnlG2SujPvg{%tq+@+vWjyBky1ibaI9FePxzRiokcqB(r{{DX{OM@~Bw4b6dopu$nf!>Jczv4txqx_X9U08BKivp$3q zrEIF;s;djB_QsRRdBCO`^B@X8K6xb2kgWfU2qfJ-$ zE}!Nap%=4r#)KdG9g}k)COmhUF5HWZe!GKYN~P$$f`wK%YszSWLbYyOEOC>P2<)u~7X3 zP)1sbMpIfW`wrlAA28!ya^&tS$uoe|o4lTYF?e47O6Sl2!F;jqmb~SvGi-W9Sa&1e z!Rsbj>o@#IX+sImx6%@m>-m$-+RrmfR3_^WxDz8212}GtKEHUQtNSF8*4ovWqqy1- zHSU@gU1a`Wvg%BKph(>6K5Y%TS~b|j9WMXZl{k1%f&h>95o8yhBb#TNL~rGe|yKfY0wrKJ(kv5hvg3g z{x!n0P2Zgv7VP@+Vl;Zb)q0r#BzN-jgZ`v`JY8lZ2ch8H-@|%Q>L-bok@EmmYt^W1 zdl%(P6xv<*OJEG96Ium%FTEO)8@`gMC^DF#qBsXlJkZv<6c0O z9H88sj1!dT%XYNR;CU9=i6*vs{s|iUv$vwFJUGVm;Zg85ggPG^mEfa6f1Uxh!~}&Y zPi2+8toXUw&td&MMwCO=nr0?iM9PDy5p`{oXq?7c zmD9&?>$ocuq#Ro_sGyY3cgoWA3 zb9K($Z27%O6w6LjdE)jIGhW3Gd#`IkDm%|HziDS8Z! zo)5s7x@0iu6#@cIpS5=N6Xu#vMNex_ixM~O*~lBV*WkM?_(8CiSrjH9_jQgOc#F#L zdeBM@CtRen&t<{fMl2Hv%r6;1LR$^*e3si@_w@+fA?5j9zPR$`R$kr{P_dT#B>b1|}ZTTHCgb!o_9w7e0-J9k_u= z(R7IN#4^sM`amN`lP7*2u&>i;3kl!j`wFt$N_UPhojb_;Bc4JSxWTN};eR^^$oay^rH%1j%3FAhk>r;JOE8h;;Hbk2jSS-%MN$t%+FMYyrNaOtTaeYYKHykt89jBdAl|{BcQez z-)tyl8Ybx#T15R>nENREOp+y+Jp3QP{CT1Yo;APMs%AoaP9#>8Eg^X}h9EY!|C_foho zTole4c85FWa~c;gR;Y}*WkVbfHDA4_{Ku{z1NQg5KQ4PoBKReipqjC3P}nA@N5InF zzN=CR4-OlaC!;mMpQvj)$<4*r3~vtmrF<9)KzvOI#e?7*QUVRaRjy^Jtb;?H8B2nt zs+MjvLNX{+dkH^1iuv2}WbQp5kI39(FBhG{1r-%R^8EdvowFvBr!xc1?vL8#c1*#r zp3o{kXzi16I{zN_dq&cSSu~@R+Enw!&{&1|oW76H{OOe>dQMeltX?Z@8E6`AF$`8x zpV_DV?jO`87o}eC25d32QIe+tDWT3L{qz)x(n0b((_H`0Q}Z8R$OU9i zd8v)ea=gE#{zZvK8`ozOVgj@*B^b{VJaBI9T_1&Zm^lZ7>*aCQ)sMgL^H&iU22%?1 z?*w?U>?po9nyqH-v?k6aD8Gd=w)lfp$$yS^|8Vy0vF2Q-t~c)e2LL{PSR0j&15uuE z?J_pS0H%}-~qfubhq0+uWZ9MSn_#r-5utit(aP3lbaYWP{tLK8DQaM6bqV zLDgJkdBPryM^&;H1aPh^N}W61IhTr8^S3-!-qCHMUv)Hx%Fm>Rg#fFB$`2L)8doHl z%_hHTD-nfympx;xq%rg`71vGFUUhLNs)HJrDfVykc<2djMK3S8R+GRM^&aMr4tmjS z^8vWUenzj{x3u7Z8PA3#*Yyhuked))?(RZa0tY%RT(LT34Mr47h4Sq#&rX<*v+}=5 z#Xfr^`eO~HdL`;OQ266dO+EwQiV_|bC%WpO<@eHJX>MMciDFtZh4bUzK-fX5jWKp; zN$zPCgK9099Yr3Etu^&tL8>1AZ-|xnZy>vbZXZKPrHV3+wq;ql z6_q?gCw?_NFqL@7218)Mx0Nf&mfM+emA2M1$wLBK3Qe+0DwYrd2;XG308}iOkM)#1 zv+5rp@6T26_x?juO3u7StT5^;<l~{+${?&LsoQz``7I8HT{8!3r;yQ- z@dRw=3%_=BNQHK+j_=l1SD^t zF1(T3_)Nx>dk!A^U6s9GC(Q6e27fY8OVE_+Bu<;A`|?np?Q#f&Q1(n=C{(+k#nK2Zb2Uj7YQTp6g5lp!-5)!jks64)Tp$Y#8XC|AbIv6- zU_O?U)Q&ttudb>-JcNu`L})%+N=v_Qe}y$$=I1hYo->QINScXf_LSk_wao5WrEo24 zr;h}$!REKC1wnw59Z6_OZJEz0nB$Lcu!*{uFS%+p{OjPP$;gAQ$$m%VEW@CV#+UkyQePWuz#M0ADmpz>9BWQiSZ}i$pNTTtu+xZD#xhK1YPC0Z3EvmxPn7jo( zzVS$*{L%TO$K(la$hYXg_DOPz@3G9wR?E#_#C%lst9MjPlJOnt@IlB;%DbwVv3I&F z_fN(vXCBo@6VfA&eM4ShxsGq>jroU`P&Mjp1N;R1QYa}9bj8_Ix*SV_cV;+IG{F+< z$Z|g*zpi-xM5)B)HC2@7PnSo6PaS@&jqSXtjKd3NXTJ#Gr1`?Vd@A`zKK7@w-#Ty{ zFqTYKUoXuH?|vMrZdA<;SuuRWgLpvO=o2qH-cWT&(%*ppv%uv$l*>-MOOhl?cUrsd zRwC6o6SwS0tQkn(gs43I`VVkFCvxUCD1|bm{sYvt(#E7sEoP@5a$H)aJi2sg z9dUj}f7bBJx5s&UB}U5(EzFlz-t&#T;mIj38g)MS`!VC0;)czuvVQ>Ye6K<3pL&Ech{h&4nzVTX z7EGG!p1W7+Y_PHiQ%T*{bkQ=8cvho&_{?7T*_SNl^5!R#bw}h$VuL0g!!?xAZ;o7g z&#;pft_J0%(ROK9C4+Th_b$s<4oiVFls9G=$0<3I)mfcy4&sjHHp0FLAXATonxB5^OfwBz!1veNr&WE84yFJx}ia^KIxt$}COAh_f5AT2Ddt z#_M)yIX+?TzMxxws=1C2VBe{YWgY!6_pi-i+8&lWyWrF_>NVBJ(pbksXizO+8FFIZ ze2*yQIiEpen3ZKSPr19~unDU$c#ie8DU&K_*e5bP>0>ia3SY|~fz+im(()`RN>S-OV ztm^^%wT@luC#TJ$x5f-$2y*fn_0-*LGR88OpyXU)*L2OI3E39N`U z&GJ*BZN+_7o*%Q5>kn`Q-omOvXA1r2F6G-hHLsc3O%<~GM6y|$G29y36i_GwO&%ZG zF~+l za{cNOkDoC!Hd~n|c)HOPP1bpkioS7mqv=rP!Fs9ZHk;(r|?vvLw zHCq5=87yQr1DMx$Kr1GuW)7+JeY{2q!9= z)s|&5+A$$Z4?=nxo`(2A-KvZr&uz*0%mc=s*1bXBJk%Yj+dt0UxuFhEcCw@Bky$PX zz@h7A2Inje(6CsAkQcb>yD%Bxq{-E3K1 zQKO@6!{y5jQ*WYf8<~9Hwn2928A;1bf~yW8jr0PibZeRV|9z{e zJ8F;e`{jHj_8uei1+c8hT$*j^H_y=ST6X7fZunbbV*cCl6vhMM&_u7R-&r+)nr9p8 z%@^x^3esh6RRaVKKUL!1Yy>!ffoLwp{l|X3*?_UT#j(X`yry^8BJ=mz*itPgn^UrR zwNrw=Zgnj~YA*aPn1VfMEa4(fjk}W_s{dF11BUbv8#E7bj{_m-u7H zqkyZN`E+%PRl#?Z3;O!Rv7p^J>l9P7(=|iZWr$9?6eo2tu?a~P9g>N}$(Oj$7$-aF zRmGrT-a%_^k@G6E@7_zkvv+Kygnqc!@^qGnpy@)m=~w8imojPE5+HXgDU{MMK_6ni z!yr9!%G~;#@g=X!htl&W{Q??yEuMVh8$F0GXLiD zapntmaf}1|;u7YuzF1u$W>@9j-nT*kq$l^IGbzF{(EUk|_wERSZBD4x0spqKJgtDQ zL!U!x=c`IAKoO6o@D^s*&lWt`eOqX&?1IgNRSk)0BX_TO{K1Y+a%1NmTT zqbJFc?Xb6e1x%J6wfLhn(3oFiZ!%UCn#n|1yUXKuUM9;&A}h%G{lhtjxdIumjMe=x zmk`nCo{noTM^@Rl@x24p1gRd7BnEUWoSZX{ENAN%6YNQ`rJs~m{ z#7J{jJ$!tJmhv+vg|0L!R@q^n>+rj0^PA10RLT3|Oy`p0bJ6!iDUut+Nz7+*oo0m% z->5nErbitbSaiH@8t}065X?FpP1`Z-8mH|ndFC?7dH4Zr`G)6ka+K%WSZ~z`TvqJ0 zG|FG#Q?;8UeQkX&G(Nu9C|IVOf4{5EE&2N*Z<1+es|0dO<-fp|o~h$#j{&w|TK`lj zatfLw1jq(lCF{BeS1$KYYur%W#skz~Xswr#iX1eif8#eg;SnC0(EeF8aVwP>Y!WR1 z(H-7P4R1 zKlrdTs1k>#&Lq-yl#B3JdcJ>d(ulb;%}_mU+^N~W;cN}$BCBU&OrDhPQTQ9Q=n9uOHFzjTYjSx&3D|Zse;UorESeVe*s9Bc-*veD$kA$OPvD`}Elac0`|(b)y|99qSCzL}C~_135gP zNK7VUOj_bicFdW=k5fd{`zL?c%X6s9thFnEKz)``Pnuu$^0dsI#0~yhmDmajt4}}l6qd1Q@7mXKWoqMkF2Dg~ zG3frbpJFY%F+cZX=V|&q89hgbg5=QNB7s$hZ*Y1g^vmb8lc8+!#hOA|7YRfSiSMe5 zmG-g&^rER-P`?L9(l>x6niH7tMJ~bqiyPxQ-@1O(Z6LzJDx&?6FYB%eWjmtAxyK~k z#gmYjD8wH=Udb(pwxpal;RX3SzB_|Jp&1g`9V{l1vo^S`Wxmmjah_Qw&w{uCGoo$Gu3qB*h$=2P!$C3c=O;3&Mlf0z~0U7zpIIlb`o?@ znSofp=d2^+;B6EUVWRmP!2RRn-xt!>&5o*15b2z7mEj?UcWHW|EY<FCkA)tIfRmvAGOTnseKSq?nHTQ*8v_q(*%lYiBW8^+#ln+w(OtZZlN(@IiiruHW(%;cnH3-?=8K_47xo? z;RGx0M2;J|M7Yuy5VTvz5@*)3_$)cSUm>GhnIxn~!*58|VL*~{8qrTuI9Zl6{L{4o zNfvz$SS!BUd@$5ll%S>t*G(P1xA$R=x{p(wI}z(s+K>r z>ENz<)=S4u4j}sn2sM7Fw&wftY~t@1kJ^ERM=ox1OGZ##@78oJUpc{qWBBxoktz-@ zt*%8IQE<~Ew?vdhn)G^4IXU!6?0rPR-7}ZCBc3MywQBg_K&iAF1oPgUUR zg->umtBoFu<%*Y%>a@WVLTV-}R8xXHNKYa-*}Dm}tDrM(X`!!7cK5E!kZqVEdfK2> zWAgzG`xbjp%4JmK>7z<_^iTPX&PVaz=*-1*%$>NVb4i|X8=RADF}z(=3TEth)qnPN zYoBYz;Z0e*&(>OPi>noC);YjEMrmPF9wx_hG#) z&&dD7gJkZIXNbgxA}#9n{2ATQ(ZRuV3z?+h3Q8(It~(v>6OXc2hIiJY$4HK1M#$1N zW_(rk8%VFr)F1%3zLt5V!3aC!2p?pbmi~4iV-dQ$@W@XEWa?I^Km(QK{J2TjH+|h& zhxiEXwf0PL)2s4Vc_Lt7WDOZKt|1~}+1G%KEPb(WFtA(*iB%fvzy4h+k*_%b_K|MD zs>y<3x?Xo^5bLWDSCR2XZM0;pW9!tA?E)%J2*Ll#BMHLQp4Sp@lw_*`MiD%pM234XyT~5&OvA~ zhuiJpbm^CrH+;j1Z24yejDsDu`sBs}bxSJxaAz#Pt0ts0EqX>BH8qm5UOf>gWI7C zgZj+fnOx|fn$1qtTks;m53W%uNTate=@9%EI&G^x$Cq=wTIaJFtlsub*W9Xt2vgc# z(yRhUc7`r=jBC14+5s;tLt%a}3OXka^lIht4?;-7ooVr|UASZoY$NL-oz5 zTf@EN`@-d%Hj)Ll%@m%!$$6-=7~Q>ikEx_ST4bkyKf9?i{4W-q{A^~OT1HTr3;1&lXqut9@UV_DUErc&{#p3K7N?*t?!_&Sbt{hZXbV#Wq}HhOB|iN%Self^EbS{2mw z;+#7f`NX4Lr*%B{!y&@U$Qni*gLO-5s;?~1NvM;9ziwk13nZ9}5*8>;7PNyFly!r181dyN52!S+%|&WoKK*T7!85S1#8Pr`)1?%^=n^V2kYRv@&8V zb<8JBp&qLBQ3^odkz0dM!f0R=w=0B;^P9PZp87AZTZ-lv#fN-Q>?Xk=y?$~;-thYJ z3YmTXNO^l(#*D?Gm2?})S|V!mQ=@>MC^<5t^5v%oycpRqhA>R{7^#5ykBZ$O4iU4v^{q&%aXFzv=VUg()BXZd;V^#Mv{zTVn3J- zG^GmZrCSPGIaF1#um+J?I)vqhloa@aW!$dRQv|YV92-Qnj~>+N24dBQ=IWT4Hy4^n zt(ZosKCfxLDN;_abiMc~{OYXtLg1Y<<}w?z5!g-2OqdVMqa9f zcWCCW%=={LS$0+u#FnlVhc8O=AA0spy z5u~M~b+Ph)RC7kHD0}CP$U%PK!^(#~UznD*%@nD5zi`1P>mn9mM)JMTe;N3G}Vqi>G{bX&@6vR1XSlLHIn;FfvaURA85RTyu+PAui)WP&URmC9HDGL>?n zd@Y#?+#{IxtmwELE$GNlnC(GDL26IxjTLE&%DnT-u5|PmZc~3PEdV_|t;TG$#edkO z;`cN1{mu2s0CX2hu4u_Qxj<%CZ9dDeL6RjktMK^PoP)^0agn<6^HvnT;bYl$;XR=X zgB}_FlY%!FGOZm#HSkI*g`_T9j;%n09UUL0!VMILcO^aYTzSyDDO3aFd01fXCC`RT zXJ%+*412a8eu^uqk9AbK6sZ&!V!(2QA*IQiz3^gntSu7~Wy=3e=E!^vzE^22BYWQ2 zW3km=J2wYdrc~>WwlRKY=0oRXTpv;zlW)EoexI$dj-WiffHCNg8shFkSXt#s>ZCk%adjhxc7pUNf$n4%l*kp)GDA?A&PC zJq-C#QGn^peAsYUOo|On$IA{6{x}2tdK>k%h2^V`W)8fb8Lv$#m$}*XZp28NQ_Miv zEYSODf4G-#hr2)nkD$Q`QB!3exXQ%L+03ZSq#>P|;~q}{$G-OqSFk$_^^(llbAvyH z@n{!DybxG^UR>m2QETbeNIj)wHr3=xIbz9b0UcR5elsQ!Z;|-h`Kt4a8-lPR;6na6 z%Y(?GE;n@yu(31e`d+e3h)_0v`g0gG3Hdgm=fNQ;y&mdYnS%@bO%_?BnAXf?wh|1I zxICp8NN+Hs>~_Vfu`VlC{{xr=&aXUo(#-po)d3Y&HOhghg3r-pCD#r--lmHbQ&n~a z4_l9({1Mn|{RFetC&CvO96qvviVWm$c7?QtvB%k;X zkm`e=9(vjUf=wTA zb&Zz`OgOECa@Q_sfrq{hZ$+g zpMQtZG2ED)=CMIw$_j1g1_^EcJXrtoKMeNxvuSNF1_#k`9;}MBgkg~85N`a0>uRQAU_ z*eb9kzT@d3M-0M`Lt7|maJ)1eVUOhn-QS4;M}{osqg_=e(Hw!f(-b_Ll=JL@4yUyK z3IU)UB1y)O-#4BF3h7yK9_-tz<~M|W0TB6AIYgu-{jL`twDt8nxL~R} zG;}LcVl{bp5kI9G$ZL`8rKF-uE5j8<7)2IWM!qpn85R_C+O6byZ)lNr|8)rSz;)og zTwveV(M!^kQ+=+VjV~z-d!eezKF)(QL+!?#gB`^^lyb1{g39)p=VuV}^nqb}&TJ1*6&kS}o z(y}Ak)!54$lm&t*x71-HePpe^8UPo)JPO z4c?|Mj25GQ92Pr+~^H=;{X#X#b^sQ(FWSZ7+NFU%UbQ z1v7bXrz+ALqVG>X314$rB;&>6odGmIJ9kYEC9D}D{zehiht7`pyE4U6u-_93I=F_x zq-1nQ*}2`GPC05&Qca>BYYJOGvx_TgX<;Q~%qD$z;ua0;8#}rmiIb-d#C#Emp;YGQ z7aYptZh6uiz>p&F!yDxWrBzG@wY_a1mO&J~8()+10};BH)F z{G!im*M(03#jE|oDV$}q;XuMtJj%-@OL}MMeyvW*FWS$|9f={S?ai z(lp6guH{s|bJnzW)sOnx`tRCUYBdansgn4qzxE_7m!BF)%6}4>o|x7d5;vFlXlZ33 z!|}B|U*hq&vged~Y$D8071$(WzYjXGe*loytcwHX%?cPTj9fc%V! z{?nvpk@J4161gU2T!#r{e`|$E4?(1f;kmBflX(x7E_j1Tk{x>sm!t}sa?mBX zmH)h8E$V-_TBku$c6TB1)s`KvLpfe%TCq};C%`wd%{gZIWZIEDD~-WOwu*i}n*ZK@ z$iedj;?lD{Nb}bmGWX_QCpb(E4ua7UjVSzr8ko@Z-R+qKrz*O<|6T&zvlM6gDn=d0 zMNWRF(Wqu|NrI<%t-O?ByI3{*wW<|@beCsSjmvr#Ukz+u{&4uAS!qe!15{)|bd!uriaf#hTE` zaZ`r_nm_JHSpTB7Y7A4v-ksI$zRrB^?W2xGrPwTw*dEoK&|_ZZq@MrkfWMn9FVwb? zf!&NqzLHn|*Ub-SIUFrHs<3Gq=B1!jQM!Sdd?X|9fNDX}&puL$sA{T_;gbdAhg7)O zp_1=y3^&OlXJjT?+IuYxD@br|4TmrV3fPbB6X*L2NHkfIwzt6^$Nq(J>;Q*oe~3!o zXI=}uHsZc;u*6GvdnNPJLQ$?eh;Vl1Dw9Y(LLyUMKsoW9eVfX>&z%cDe#`X_VB1qZ zVE$(E8R9|&XJYB)2y7y88MJ`;@&BH!|1uzy_v8c4slu|Nd#-sZ8d5!P@6`Z)pYEuw z^0`TXdT7jewJkcJXecL0su*@ZnoK|!utwvjSPW7e@|#>u%XEVW21UT4xdPaY+`{U7 z)FkF%>FsnI95J_2j*`^2F)K*A(OINX?mlF2o3A&!>>_qnF}1%?KOJjf_t?#9R%F zjrfrMd|KXPHJXujNi2;2TglAiBaYD5htNLt3T~3>gSHOoMzwgkk#|vY?-=nf%2cA5 zS!-nWOIbXm5p$dMD8-i2wkRcDL>&z~o@G=(-A`Zd+^D35Yst;{CnXbx3YEE0OffV9 zO&h2J0kwta8xqOF-r5_?W^s+h`3}w|0>2l-cbh%Ey zg=uhzDOaX%zbs09*1mR1>Ubtz@O_*Y*$=TFa{rmW{Eg;*wgoaKgNX^5ic-+j4Sv2k zS&U>?JE9WSBv-ioIXNf3`IoiT>H~w@SwY6AF$S3K2W^+)pssvczQ1EpS~6~xb-CNl z;X5Y|Rqio3<`4E8yd@dT7yZ0bp(aU9&`ykYtY4xaQ`<^g4|{8YI{Tve(bBE#l$T;u zb-sI@@|05X_5!EApt(q9xSC6_w3YEh+ct!+WA?fFOlJCI4nJpCKN43r{`i*N?}*J| zunY=gl4~Ass11|+uc=!1z5NI9EL9TSqMGP^$hWcp7XuF5=^RE?SEUucV`sm2b6LsM zyn3jDKKmspO3jaRef$)wA)n{MPU@d;)pppi6}GY*e=6(-Il529HWP{AM38 z<*=@A97y85ull6aosK`;0DxUooO`)4sr>6s<-)+Dxs*_WBp%+Nc<=B;;{>~hc6V9% zoX>J??LXkB?{moCbC_@B!O+PcYTe5PgTAa3Wsjf`U3{y`5F`D_6gC~y8_d6vwz+zY zNkehJYADoBkfk{V><`mkHp)zBYt-AP-z*ozs15B`xJ?v=0g36;)ZIgj>*DKOTWaR}J^Un{ zee%H-G>MOvy}@Z)oVi|x$sHP_Z`Ct01>9O2<=U~joU3Ohp>u34E1mSIU*n_jZpot` z6NS4%zO5$u2Op@vzhoi#zoXm2f2=V-=KA4Nk#>+@2J+Y|!$eE6-0dQyk-)?|s+DPj zw-}vv>LMQ|ET;$I#p!gP{&xJs?tM|Z_Zf$2CIn>x^-UPODE5p(e&Ht12Tgnj;0-_3 zs-pmS(N|9MFZgU#9q#PbZr_DXVxKl%qt^c5TQ2xWf?Az{HkyGdtUrA)*cZGOYVWDb z1!QN*eI02SV@yjrV_7IP8)<4h^IDupc1vO~DsT5`q-AR_wcn{y)0jE1V6-iyM!kMr+gBqoTyBy{SBKlDpND^55q&+ekC8IRE!+)6|Y1L z9fWa-zKjgfYD}(p9zco!E%2CWcD%(M_Lexx+1k6eC83%lDX)andy@_-?nfhLLiCKf zjR!f8h8y}9pP5cH#eUFVB~S$S-PwR^3C&dAyie9U9#@sRxc~aTKW9JsVQ?3gpX_AK zi*`*gf59Og5kt}fO}U+vhSN_&e6t)`PX0&ueWD+|5Z9IpUM;1ph64@x{{eo@xg80(!EbE8^qoXLo(%^8-JMUrXEdL_dF3y@R-Q8jwsOg zB2TJNr+@)gXzf_U_6m}#GLmsErBv|&dwWRNNuK*hf@B= z*)BZ)s>@L?+oR7@=VrVJ=)7w9PjS=L;kuxQl8%ZR4Mn`J;I|3{LbpP z@cW9Xx6XVjw%a}7fXv$x!q9rL(wXK*@RPm2hi_h_+No#)|8_uN^;gpR9cG(Su8qK?AT zz>h1eZ7r6C*YbgKYT|MwyE)_=UmHnx%8M>VUxB`Z^c$ihMnlN?KglXt3GJ+f>>pJ8 z7WVC&S@t*fM%sEsq@T_H49e;v}Q4j7x0B?%|JXPnp{`V%mqG-AcZdg0`=uAp%Bx;>}6b(Px6 z8njOE%%vql&rHHm81kuxvwk1-%lWZ+{n@M!s?@>SY|4J!;rE3ALG+SVEJ0q5prkhbiNHkFZZU?6~tJ z8JIVkS<{U&S<785q@%unZ1D$A^mDTBLFbGkgB#sy+z~xHr6M?~0L^y~ z+dr$$F-fcl$6LQu%hzyKQ0!7&GE|N0FL`grKx=5tZ#4AnZV`49p9A(57pj4!GD|n9 zxa)Q~M$B#$D9pJRKhgXORD!(&DEYP(9DVp=KVKXcLc>ezxDFiF0S3 zO{u#Ff)!W(hY|Kq1w5PEdYJrFA1$+rSq{PA1~m|l8Q%#LIz$Fg@Y9r!h;^zVvoWTZ z;03{ccg}QOAO>56yq>kUM`W{=vWw-ZKcq&2Pbu(j=Xfpz6GNFB-?+C(Bp{`pX1~gZ0E>Dv4}P{IEZ< zL1zENwx)x(%9;X2#Y!CC7`jjVkNKB0V?%zcpMVMLKU`KxeUm{zhXoN}-m?Z}Cm)M$ z9!hGuUCw@Tm&ZZtm6niYymyYh_F?505rh79H|G(qi^jGxZ}z$2*oDEh`sP6-l8gDE z3eRYQtLjd#qIfFsnn$M?t)N0h8djFO#*{3EdAaROJ=hmqQ2`2a5QbGcxSzv^`v&?~ zl@o1!e*l#Vp6G=+uu#90@qYlN#19IK zh%keXq*JqLYpvWFA1SXyqUZTy}!MS_5I^oUoq2F^5tZv z#K5p9QN)t||9+?q^dF#tA6cl^2fo2QKNe`(nZd1A}o zXpgA>06FVpu5koE2~S}z>%Nj4BdP%3(fkuzV|cZ$DgF@e|-FqHx&BWDjdWwph*4n7b3Y7+2%0iyJTEJQ=y>LE`3y<$=9jtD;<_l zarpW+i+5OzpN|i`3{4@dkz7wDJg7j;;|`ta^PU|m&V9NTi4I_U4*AG$)zbyCWQ9wm zhJ_S_K?oNy?_5kA_-0g=VO)qQ5(`CE!3CGSnW)uNR7oTF#bjfvh`b<2J0Fb%3iRfH z!U_6w-IKb$^=W%g)Z9RplxQHtlDISsr}ymy9AE#r=;~fUczhcCC?hWNUCOLzHxO#5 zc#VXnsE}Lyc^D8qT54m0yO5A+U1EOQ3_gt1t=wuT_xAq&z#;#U{?ePZPNBL>DBL3r z8pq4YCTqKnZtAUn;A#GV!S&6-v&Si841+Zt@2c%g1ADeAY?}_fibfHQX;<>KniTQM z>ABN0cH%C2Wv}i#I}e&|(iD?&J8qTbzc)aUXYf+B`uzFbl@Kn8`&6p^!+) zNC|_RS=Fx1nQ3f8HgX8kr>oK!{9xz57z)>!{e z);$>4&%4C_1z>uI`MU%@cO<@+DK)cOabA0=a%xv)-JH=nsZ9ML`x`6?A>Z`P;?3jU zPfjB69||R{jS8D%xz}hPo~I-5DoSnQm|>NCk?y^g2UedZ_r?VvtR$4n4i?t#_4mJQ z$5%&_UVHf+UD1)U%tF|h7R^%!Nt-81T$4ADrXtJ zl05Q8GORwn4ef39x~2!WK?5y+?0a8gxv>PYoQacr>l0odp7o@4vddmFpsjDG2sLtG zWkOnsYqb;1D2f$BsaytHwDV&zRnuy8{!aGuD+A&3f%kuaR@r8+uk_6RmFJpmoxQI6 zH?3OuMb13QAH}*!b4{zE2&nqWV=Qf2Yfyold68En!viJS`#bo1)wDlRSbJVVHre0r ze)=PhFzL}0HRssC2pWZqrN>EmxXHDcpF2lFApIOmpf54mZUy1nJlY0YMmK_k^}s;} z*mP7SIr?k?!T(ZnMOZYtd4DRr9=plQ{kqDk0;i9wkLkXi<*Z%>;pExeW_J);`d-PM z(uDjZgJ;LmG&D^vp=d9W+kz*SL4gDZE6i0+>%FReu?EeQ)>))HmpBC6%J9t2d(Jy@F zz}0)U;*hEMGWdr-wrR0iQQYO8Xx>DYn%wp|I#-{X;FVM&mV8VpE?UF2;ZFF+l0L*A z)+ee}TEuGqEVfz6j^9iQRk=YLH3|p~?W0w)OIdlHie`>{fblUSq$(VrmysuSfw0HR z0(`-oo_d~qii$@eaW%5^1#a224wD&d-La-IVow>gLK!KEx+49Hyo37SPclkrJyrH~$hxT%iof+76Tu)o=N)QB6G{ zVrzEFV_nAzsLmaOFijP?+#sBAJ&a(|>?zmq2~(JydN)*K{hRy9=+wJ%h{L0M;;5@g z8!+G|$yxd4d@Zj?3~jxbdQ?2M7X5CJeorrl5$3z76h5xvJkW|9d(FeBrbfAO&1UAa z)f<`1WAROOIj3!~(A`?;0BZdSv*_GF@W}NK7v|VThTvCnLh*Vu6a{>^-^t7ZQXKa* zrSXLJJ%x=YgXg~tgp-EqTY4Urib*r^xi{zcd@vHR z(pjeWQN7IhN>{9~*z`)qdA&6_NuE@tebWEYb`?m$@7@$@M;FJG+?L`Z-S@yUrCaZG zqxlenjblviryBfrHju@av+SGs=fm*f!@lNbt8nN{ea0>;pWVc}(7E3)=#~z%NFE9F zQgr<}-lDBbzNBCpjHHzt5MDH(qxOG;H};)~&gc5d`D0orFf3`h{oV_Df&PhjMplGa zH1nl1v{r6R`8+-UFehJ!}8Tci0aRE|`KNmT$Gj1`1%S4QD zod3AnGnVVD?X%D`IuzDTDEo79-+p%Qi0h;G*srmHail@if}YTY9}6Q)3P(i_B-`!z z`n46;Dw!pn^0C3`=CK3PBH$lvE_6g3Xk~C>sklS)>C3{=v)id}~*91I3ql?r_Csd?5 z`i10zc=1>6k>(YPgM-r*u0DQtm8rn%aWQt$tU-1 z(lmSMjYqXRUdw<&YFNuUoN2u0olRWq4f?80b#y$%CiZ4xsWA_De}mw0p3@T#kz&t3 z$_9akxh$+G;gIF}<6kbO6I`8A!SKfCN!{6+1$r)?WeqgK0-eU` z52;-fs4jHFZx(ou4SvZc9@T$vWF7MWZ2PDj)LJ3bSek1ilTy7Dw~Dvxh~B4<{{y@w zpmm#@VoF5vUPAiOhXw^h#zEq^Sbn)U!$!G=n6x^z#|8heEq7%*oXAqcQ;iFfk#Omq zuY<0+M>zg^$)?9TCZ7h)7p|Pv&xSPZg=N8GHi)CjLLcH+5Al z_1fAc=*FYh(31T@9a6DT)Ojm5L-ov{~IHl4l!uwLsb*R`mA4TVoK*O&Lb9rkHs>B%P z3a*>uwfw3#a5VSWm$X$Q)L}A1kI3=cVtLm$ZZ5gs6b1$=OIO`Jy5J0ok(FGHrflx9 zTl5)aKyqYQBYy;7kJ`NcPQ701rB3|hXoFL%xEKA1Fk`kbtO5JQL+fhb-fzw}^SG)vemrTuMM0gV`GK z(D7oi8E0Ms9;V{mf%xERObfb_?A+1wC(e5Z_43v@ZbHnYqZ2jt;#c@wz(f2Tzk>I# zKU#?ptcM`|k6R`GwUk6lZ7hR%ueooBe@cG-9yTPZg=3QAv64Vt@>clh+?-ZHTrA%* zV}w-_a~FJ$f1Y5v6$NpWYIku6m|P{-DPgR@e4y+J$iFuEQ%+_KnAK46o5~-bu**iT zzr6XK&;$S=T!Idtd*`!7Pr}aNTxmAI#WQu+=(bfA_RFAFLtN^!H$Y}28$8qCynIvz zaoTRcF&1T{N2g&=qVlbcbF8-mSaKK*D^tN`VYalz!(mZY^w(AGDMUwb+RblUB zFwAyv03rC^xjciJlO-ZmKV+8WK8DO(tt#c3+vJv}-d@)`x=P6KEyLih>e}(GN4A@F zvY0B^$c%Cd@a-2-heJp?N+*V?wz`^!$3jJKPyNTw_IW23sV&Aej7e4OyI+kVA?3Z}vsP1rBG@HL*YC{>N;ouPs+j{R7UfD6TJ92Ry=%Bt5 zxcleEWKVia_=7AuIfD4G;1fDxmOOm8IMv)|2(n8Tvv2-p7S{ z_h6O2kH?1VKQzWaDby#uBZZ_~7y5ZDi~KTdq&fPEC4Y`&Ou5|6k|MiC57IyDnW|gH zBOFGlS#^vB-UR{~GVbKxG9bL623B!z(>sluuixjDzp(-Y%wf>9MB=v^i(S-jih{Bd z=FDLCS_S+Biu@^(DeBgCUHB+xTP>XA&-|e?kK2EM4bJNJvT|rby@4kJBJj^-Wm?qW0lsbUrMcZ-^9bmk5)qYzZC;~D z7^;@p{11SWWR+QVx?*JBwa8#94JE9I+?7sKVDkoAoQi(DJe0qZLuO{G13AMfV!e&x zX{y$^Lk!-?ZSaS3=5-Cc>oRpH>k@6@x(^zF=htJ2!->l(W0HeBITbe$n=-~b+(J-0 z@!Sm}AT83Vpe0cNdZ_kkpTA(CHYcB+%1Sc^9GGRA-PQHiq7N=&ggbNfB#*t|4&S&Pm9k*M=ANm$>V*-^b(4SX2CcK z3iRJK++1(>@2~17WajwctzQfOU6?W~jWWnA5Hk6q8EHnAp8u*7W{n>B5pX*RPQ zoW?C`@|10ETg*X;!%~6!9Sif;JBflGqor&$eFy{gFs{t(-J>(vUb?7I6?4y$_m9UYA)EyHhY>k;xUCWT#-TcMPO9_tWvbcTW zKW7(FdA2v-jyX@)@rc&r_`Y|^Y~2Sh0T|M@^6|j_n`M@L$f)F>Su%_-GF5E<2piLdJPQ@gV3NEL&{rk=L>mS= zLEw6o(JU2wuo_(YwsKor*>TtyCt0mHBYWcVXWmxz`H{>phjp!a5f{xFASo&m8513~ zt>Pj-W4i#Gzb9n}^4a8y&a+NJLQY7zhv6yo;gwQ5-&hDx=4b}IZ&d!fY5hUeNYw!p zGT@d$|E_?2;d2iJLfg2ZW0ZD2?<<|RIlSY%IpX;m%7*CMbn5mp$PfK4yU4o9;rFeY z?7Off5KnEAyW8{hL-*&Kh=ugRM;3nCC@v9*CmHD<`=h1e?j`DsM^X=>fyadCvby|d zyO|;NXY=Ai+O{6H@$Xiq5kjd`!?kFFpWqCrG|_&sx>nQnjoQWviwG1Mnj*(tj$ZL0 zP(k1(zY)B9Rz7!XL*lDWPv)Z?sZPFdrLeS|tre|CV%&{cyaOoKIHBZaaBhcJCD0Fx zFXCCp<3wlf?DWiAINoZvxC_xG&y)c|yvQvnw?fhmv!?+I&UPW~v@@7a-e!niLgr-Z zm;^>4@AR;OMbE(Qq6wr7uQI3r8Zoj2Dk{^?j(;#W6! z206r@FK7Oy=?vDacu3uEVXWXs=1}hC@+-x)1Fa|hn|kN-D5@`KCTqJLC}lmxId#{E zHZhUFO74>bF7kAw+vcBH*ygo*St~iJ0rlN;_yc+_x#c8{%Mujr%Q^;YnZssC@pqC1j!k@Rt zCzLDJB;KE-?y3o^lhL(M$V4p5ey6GGVVeQ+#G~^qc5Nq!NdBx(PkOYB2_wc8nqyyR z^VFz$Wc80GbKgyJYv$eybHj9J}GxryH!oLfsDhyL-GaHiS!T&C9{azpIET?tlu-_-)krKxM(T3oeV5 z41@5Z4^ZYRM|7zQtSAJJ{IqNu$lyYWVBtP7h0l=G9|i8`T~UlRY+@+K&V;v2DrS3{ zjaL681-o_@nI6fp>Yi7%Z;pnw4G^4XN7fCslj-!O(a?dU92%?h?{)25 zUS9G=PIS>u?J6hv@+YAwlk}!0S>BYbFqQ^-fDY}zdsF@Q>ZRLF_daymidlNeKq-{)7+Ed*6`^=%>xN>)vgF%9rpsOsb~L+` zQjLGCg9tCgQ(c-yo;+JIoNbh9=r>NBz9Oq60tku<-}(bC*}E?%`-TNN66AJ-9m-s# zmN+M>|Eg(?2V-9Ctu1~CQM}-Nr1-Nc!CuSz1oM0VS!S0MYh&D{t!e=g9{eF`7qbs! z_FT_t-9OaaGO1gEON3Osa)``zXi#%8K`#D~JgO?wq@^_VKeX`>wO4}b)_3CjVR?x@ zpswfWqyl7zdvx+QwcDM(UIUq>eyFR}qvGbN>rZ3*3RQ13&%;vc?kSKxOdDr9v>+E=*pn6|Drp55)9MGts0@)7no&M?^b+V%_;VZcFD@ zKWOsS4;>xRsdKUDyD4&itm9#zOPI&~LXBsL^M&*C8hM&T6G8vuH)ZR=KRmWJZt#>zX$q1A$0|@0%V|-KAc|0cZE|4sFq7C zN&q9j^%PPVK~Iz8`&Enk+eC<1sOiL??J6s{qip(Shj+_wRw@KyoxnqVF1hBT)Dinz ztp9i=VM`jr<-n)fkikj&EFhKYCpIH5OKd*QD*cNnFy;+EX+OrzFR82%e`e+DhRk^2{fjKCqqb8&S-;n`*Fr)t|fYMnGTu-G~lEq<}4GlMy_8+O>qgO zZEio8^{XbNk}3eZ{NuN#b-x8VP3ABlrzS(4w>-(6AqT*pe_IuwWK=d(gDrUxS0<~0 zvafx1@zE;V0&Wanuo(a zE^DFBGC3_jh?YhV%u2cxa^@m#d=mw&9Fy4(YU)QP@f+?Yc@{Xy)l4~7+1h2q+RYGA z-ksl)|Gp3apDmr`H>#(_5C{xJv{4~0K1CY=kZE)U3mT5S-}bhR(a9|Jw8uR)BNTcl zBCi_43mUr??K;ByL_bi`C^RMr@L9kCF!jL|9?$RKX)YI3TzaKkIJG*^`CZU@&vvO^ z1ns3OTpeUM#90@W*D>qr?9*nm%56Hup1Lg(^}URg%Gv|Pe7^25^6vOHsPQ7MFO_+O z;Wg%^{4Bodu1Q*?oHE0>BFAp<2W)c!rKZ%K*hN;Aaqn%l92hnoJu=HyS*#8uvckM5 zzCXK|O-pdn+c${+`3^yD&#d9BtedEhw-4o0C!QW*gTtR#`)1I~R4i z^;hofKR`F~6@L3QtCh8ulin1BeJnli6n(VqYb>JaAnR_TskOp-rHtc@ubmb&Q~&d(wU8ls`w7Gn*iY z#w!2vR;b)ttR@`#q_m-~ukSvX6~Qh!B*MRoNe*Ui2=(Dq8PisO*Ze7iXKIS5Au@$x z&aKG$b;X`)sZ>@iAkNGMA_+8ulB{qBus=bZPe;XW*I~sUGcJ6@e}NaFb2n~Mo*BE! zE#A>w_E(9KgE6Dz07hK9tOGT{ueUf3@Ox)s{sCE!g5$NG(R3{fD6A`tshk>cJ`9Vu z%5wL=>RX)=C42a=tiNF!#IGnI2n*Y0>X7csY!`UbqimgWn(bti*#iz%J|+W}xz6Q> z1TvqUUe7>(-4zD5PJ44BE<~WYWAGYIl8^Z%w9PAp=)%4)j?vg%Sg5dSpI!t_Zkd6? zDa1==>7A6WYMk^)<^^iXT%>2R%&I6x*8dm7z>x+Fq1naDSoDQaccjC`gi4>e@Ch%- zCe9+@=Qb#-_+a(NUwMew~;5F!&{~!^kr@;r|7o=J^UpUnGiK$j^ z7*Se6Ez$l0vrFE@#m<;Ha#m_*8d^MlK0uFg2JjoHIXB4sX^>i)qBDVSEb~^WW0{&9 zIkyV(>-y#km1y%4@%&uNiLf?9HgD~FryxHdHaLM|yZS@Y!Qn$|8>h0M0ef9awUsF} z6w1$+no3X7kJ)|i=Y|jpr+XQfm4-!X9FxA%_!CqW0K&XE(n=x90Ucc^zN%i%j2#OG z)+u&#COkCSO|L5nl4BhcF^dN7I${tO+@n)*1at$p=9+sWQ$*?)@bOj9Tx1N5!lx0nc(nXyHTKoUYz#%Dawr@MCYjc><>ds#X31#c zRD?u44OeEOin8)T&>?~r#vk=vS^pV5C74a8kH(MvvYZ|(73Z9DPcg0QWh}O4cXKxz z%4hbMlgc;2kOj<%$*+-v+>s&WC6_?6zsg>&p)a~ldfDm!13c#Bcf>~C4aEig2cX{g zjM6ogE#V>076Xi6E&7pZ5e1?o9bwx3OrCw-S=Mtg=t!i$Vztg~i4ym_sjZumIaKqV zf1{Gg5J8;xmegI<&I*Kyfle^1@UUvX@AG+-1Aa(T!wn%!nkTSGPR(keGLZaqC~-Nd z>t+0c5;7A zZMc?Y=g`H}{y)A-1x=dtbvM_^)-pGH^1bBNrBJ9SjnA;xWmv5-dpz|>m>-4#9A-4< zV(Dmpw>6c=eW_4tj!Mj_5JyGRvy!*}me+ z*rAO;kyO>*oO8+Jq4if8nFqFRW@D>M_l)lsdMbX`Xkg*-P&B5Q+g>(cRvBJ?5eN?` z4RKE>t)`Xs&k$EpryS9O%F^x-yySKb^=2`zFM8#w`l<`bn>Z2OU%`q%#R_0RinY7d zKKz4u3&_;P3|xCd`zyg+YrZNDhC0auUI({Vcrx=nx58 zj_GA~`K~Ka`={vvy$O9dWLc%<_Fw{YFBg0G{rhmz!`A5nTcNq>pkh(5;RJ(Zl3|2GrIAkwwV>v= zP++u9uI#e6GsYW=QVb5n3TXNCIAYW)D=aS=Y~d4Kz5LwYo4!6$^uaxE@eW|9LVXda zNZ-id>Xhd*hLLYrbL=N5*8 z<~1dF(C-$xA!6vr_U@5V7zF;5&1|oLDD*At2zx{1J`kT@r{*_6MHDSA)@#Ek1 zYduooO}8_Aw!!b@kgA0KitC%7ZS*pDyL)gi%pJ-6KLLr3+NWFyGs^W>EFD2<-9r`9 zddO!7A4>b;c(hw$*~JC#1yQljAE*!un)$@XgJxdFR95h}=R2>%TblYtQcA!OvX?9# z>Q$M`y+V=;@kcpqQ?;9ce8N_3*MdE*O`GUFT~}OA>X95@EPnz*JadM=IJz?VBX&N) zEo$A&lfiDows1cYqeE%%N-32&Mrc=%)a_RRdyF_M?5+mzXw~}77HU4o5@k|Es9iwS zsQ##@;STImXk>s|fLX_bcI?VZy1U^496Yvm#6D#fAxK36Yq!6ZkBX~rm=28PqbNM1 zZ-QiD+qcGuQ^89n>HvF{>T=EI6k^dOM^E94-DT4c zf>)Ui@V{<^%6EKF6XiXyynPOOgBu9#sxMG=;SLQm^Le1sYPaX;1v7U@6Rpvg)$}slS zAa~SJ*&OG~gYqU9D zQKU$kOUl+FnJbl$!YKfPL8{VjJWd|_6jD~`4c#n_0j~ZDf3i4o)9`+Rabbi8{hL-f zy_i1NFBj4F#?W^3WVFANuivjIcR&GW7;N6iwT*swYwg%Pa>`FeSW264KgzNwk)Xj?^fR{^=zAhmHji-49*k{G`qGG6WCZ7yQ{N!MUBWb zBRE&N_aiCUFY?Y;{3{J@7U~**Co+hMMaBq(2<;XWWGP(+P_Xcmyn#k)z*wq;!7BJM zP$Gf+Cw?wBRh{~U*LfAgTn3JQ!tgDZ(wPKAO^qcu+75e^dhO*eIjPu4Lv!?{_+9Af z>Zq}=G-+SsS9CHT;0}q(PlF7nM+^rf&CQAci)*iY6 znZjO?L@4m8iw_4d9CY)C{C2zjYdiC!gnz)b#>+p}6no4ltlCvK`sptsNLP4DZrjD) zJ@Qnm+>Z}31T*o;f3dr6DZC^VoEEGKo~|b}oJ}TAkE;$S&lx$PF)rYlND{E>#l|k> zKv=0yN|_#nlzM?PJ@40b2!)@VotP`WA(a^wnZ$A*U=Yr|o{27{dXv)f;+f-RpAgvZ z*~_zGUxSGSt+bIEUV*-33(7A30p`gb{NC1@Q*&pF%Ml+nIS95)Crv!73J{4p8t$H) zt-QJr;fxc1vFq|(KF1Pt#^?T)JHzvpCUbY5g`lD#1r;~*tVo=bCZ$Cxr@c6_3qqE1 zB0Rk4oh2Y9CreglX-Q3WnAW6r_#Ysz@2Ms8!Q{+zjMKFuIXWrqDZ%i_WjeSHBqidb z1qq7AmIBB(oV^Lp!HC_}REJV_`IW!b0mtk7!OfNJitgUCq=5OfD`T*I7zd9zY=JDtJIdxnw-N(6{*AMVFPSzY&+w5&_`BWRFAcckUv?e-8LMS0EL}&jJk`?^7HP6LT($s z+;eAF8(3Eon)u$pp*~V-LPM;8wA}*J2Rj0TR5WNYC#&#d)*rfgdMlALvTyVooxV|P zrsmR}z2|!DzW7~rMz^VTS^cGe(1JT&Shex$8Rb&`J1zIjo)|VN)x3!DkM%nh*0DC4 z&97CoWw^UF!@Cp@Nmf@47i-Z9T+=j;%}5e50Ig!kz<0Q{l2emw0u_}vH9(N!B{QS) zpK_up{6mNZ!LmVMa!WUhi!y8t(-TP_eTDy}dTDd2-Bk=+Ue2R=q^yr^~LnNK2wPFO_GTG=46GMitE4mq*_1X#(9)tArja*FPK>6qxHN)oZ3W7 zBsAHIk!m*S&nxdU_eKXyQdot4fN5i};5)l@#K(C1S>dr<_xC4o5$VK2tKAjuW`(n3 zv3>;J(3(SpnjI%MIGdJo8N|LmIl-wV_rjcI&_Dp)nm`Yx&Dh~(T09cW*A3#IH*)gA zJsvf&Feq1IosJR_o==MC)L}wleYj;`zjZrxAMCnNWhy6$#w{UH+YWB@g~@o{QpKTN z^O-VALEz`B_Y;%bH=m{t37!rI^KfJGhYu!aA#d)4`zfI_r|42uK4xeI8?|+|%#Dc3 zr_U#KZc_Ja3{Dr2CJ%g)eYMJ;{b0A|!F(79{~12BWKCuHZ3y5%a7f+xDrxg8IOl)% zPWWQPK>12D)>@E^CKh%>0XQ4$SrX|skCkII5no!hHVIm=QjL8q)vG7ikmHp%AG^?^ zBl=@el@<{c8A;kPz{)u|M#y_!ZOl9LJUcD?KADdaj*eIAE_K2;(riJloAUKIUT9*> z8m3N)f7p%k8&)h(Fh4)!P$DGZxJ6&tP*S$6#%5Y8Wt{T^vz6qOv$5nLaYSB~hClKnXgTC_H*KJrMX z+Gzs){K67m*J3#Egf26+jhj-#4^sgGL~h=!Rn?i@&n#Z$lRMjFiAkQ3qgtmJ=Vu+y zP=Ascm;P21_wJh-pWR)G3fcKWP0!c-IaYc~>`qAvKJY9Ee~;U=8wyWI0KGXs;!Um} z@$|Z$$*uR)G+dTkdQU~-SPb-#()B9(K{#BtF?o=>{d&4GQo>N0(0t`?tL!8C9dGam zIRzOrQRLo4O1NlDeh@YQ&&f6~XqC8 zptH0E;|StN_p=wU^3I`N(pBJ2Tg)Dwj0A3t5Gj9)OkV6H)o4iFmmIR}crh zWLep8SR$7;%;HDYY{l^Onpmi77+^KzfqYAZj~ z5xnzuWU^t5<}2pgTyxC78mlfUD?fYS$2*Dyl|EgCEsg+j_5{Jn57I?%;$%3F(NTpx zqyCdosa{}(>Wlq{G47$J4*rY23G-P}nLoYIQFXl64h6O5Pc8~3-G9fiW{&jr)v^(D zGVhkWmdkL_v{^s&pxyFUju@RI`Y~s=ecm_o7`D!!xO3^fp|m-MYHNlO`N2E8hW~)j zh0IZ12Vv<`4&sWt78#eeg<4p7#v)yDoP&^ZjU4=fUZBw-_tu(iVF}sSQi{M4R{*y_Xz&2*o0czjs^i^=%t5o@FT)XQo4|f zQFB|kf*KDcpSde4GcS`7o^+-Yw9mDDHXb4KA+ee}-*MR2Jw!n82(C!&=6?(vQd+lO z3M$jJw`hJSrF`JoKri11lM#Rx`*Cf(pM_Py2*HAQaw}Nr#fF`bN8!Hg@do5-dM%HLlq) zLxL@DRc6Gs#?kwH5c3EY-p0P?nVM)I@^0^ao>MGQpvRM9Gk(LlW5(>lX4z}c<&@>w zH6KK9Th=CTE~(D?)6L(O&KBEewb7-oGDxA0nofAy7r;^^v@Y#pCWv+r44;*)q#05U zxvsQqxtBQNRZWj>j%c;^Vx*`&!@9oBO{q}RN_(DZG$%!5mOM3(wE7gRHAdT92%=E9 zi>xwERTFW{cNC3!E><46^(sPyPyb!k*f-jLHpF1Q7A0DuTP%2{AgJ=Qri~NdH^v|S z$BzmF_L}BDJ8b+Si4k~N;k&frp!`19*y_{P-ba0#fDik~;!5Ihs*c(dQSPHMrEI%i z7LlqEp}xQ>K2N|>P^6b@NW2pbs81m^S`dyhk)TL%!I{A1Y=pd%`ULKaP{GY(xK7j4 zI{{Zwdh>qSxFA1iaTz!cUH*i!bD;z}#QTTR+sC{rBt7 z+-^-6J(6z?Y8gll#X+*`P|t$ya@0D|n$>b?2UYRCTPgvO0HAdOD=>dPhkf4M@i28c&or1V@exJAuxM_y9msiIR2+x*t-*3 z*L5a_rF+U}Not;7cQHNfq44QTM4WUanMELYQUPWm#{lk%^&uzN<{jx5~4O;r<~V2DN7OLTJVJFAqU z8C(qx3=DdV7hmXxtBa@qP%l$55;+o8ecB6tvqL>d(L5KYFiI9~?S&2akwcDGQwEz? zHoLAzy$Y8YraD!WIs77>o|1Tz@K7yH)cb9XWmq3Fw7BhJUdLSOLW3nU*r2Y8l1kOJ zJ?-OdT*yI5g)J+Fi`H)Qf1dNp|6_DmOssZmo0#pYODemH{r6RgaAU?1gH_1($`hV4 zi5AI}^6m^6Q|*{tEHEjx{q?EYp^7p{+5*muy93w6QH?Ed9%Qh+pjoT#OcOUI&oV>n z#7_-+&myKWg#H7}rfg6OKW${2wU7*Gm6?ck%x z%yn3#_|2u5*9z*vzU0+^FEYplJpx&Wc2_OVutbafzP&`u8r{~3t|;p>#Od2RbpDGw zyB6G(ga?gx-Q1f{2zUn{Aoo{Bz%`+@YHxDZzeUA?FCkGgb8983K7nf05=y(TKus#x zlOc3pJNTY!lvDn}3Avo5*5JWic`}cpTpF7x-xT%x*-9-z^oPSG}Od12~7b5%UWJG)yy`9Q?W<|3$3p0@I_N*$ad>8_`=7RjvCmpm$Qzmz*t zsU%=F1;9}=>9|X>U8|b4t;=$&Sb8dHOa>n{^*RZ!6Yn;|*GF<|6)sEx>5ulsb+Ffa zzK&n`T&UTG6xZJueC2+79uWEcSV9kzd1r8V#41IOBeHNsSa^WahJHhm=fLvK-wd$* zOInMupn%!3Zuxn)E7Yyx5zVQ}s7kJQWtvlC6NhXCoK$H#yrSK!GCnPx zd#w$E_?}i!fL>IxFz(2$#moi`&y?j zlVc#UVWHA1W2RwSZJHuH0%DiM?MyzeWr7Fleo1zGqfCl8yUtlQ2Iq9OaYcWPc678# z&zq~&#Km&2a>!?Ma!H82>LOFLAN)=Pou%(&=Y*YK*=^>_Ti$$=*_89~!`n5Vt<>Yu zut!>D(0ExsQR1YN+HT(CI**#05ttf{%r&jmr$Q7WOGBQOONXCAobo@~Dp3kB)`5y5L%c1EKbvapf zUp{fzuZufk&aFt|!OIwpDzgfwh78HA_BkiZya(p>f*X_VqKnGa8AjagKfmqPTVa%OBN)r zx^DBHg<834o+_neCQbWmt;*l~LiM&^Ss_c|#CVjELcoT2SCU2;SDx0za?ZQC%k$Mh*QGsAuD*@Q>r@8)@H-jhbT{&ezL)0iX1?6PZ^Ru-s#`0f4VxTckT*(&qC5}u=Ip4rpd-i?UR*bCHT0efcI%#K8FA$WXTxchAYQ5M@ymeY zbhw#8T+ajHm7{K|R|dUWbEGm3Gq0OWF4lFsB3^FeCSbnP$hxyP^XT5jI*7zAV_Rqu zcGl^xZ%)s4A`+E*%ko!iADW+nA{o1wh?$W#=555S9nK$ycPV4PUq7gzhw_s~dK{jIG8tX?6pW` zZR>zj%2W&Ak$<>T0TB^52X7f;^7H|B4z%Uve8MPy*qjUCXat(qL1TTLr{SwOWUhEnq?E*~!QQt>|+_<04|J(zoxG5*UnsO{b+XjS?RNoBN zd9+2IEHI|m?dm*cLw#l&+~-mT_jUKXnErw*kQf3O97kx@e zd}pZ(fyP)j4t-*aQ%}`fw4RbR2!>9{{QmQkR5BJDT#PX}7*V*6pTPkgov1G#bws}O zD>Ks9-k7N86Uv#N7ii*o-BGM8UI-$_BI}S^eZQnVr2sNq#(WxkEm^;&a-jTSvI$;L+c->#0{F)+p7_|RbOA=6E7C!KNlO3Jv^uw}oRWUC{%G{dUWs?+)>sLuSJ~&9rsS4# z0TlCWiZ!i1R_7=hFZ_vrk2Gj_5O#th42xy8QEOil$bi_}&#l3<j>7M+8JOF{zhM*NU494vSe85Guglb?;2G{Imq9)WP~Vm+1((wTamHZCmpa}edyrFX;^5SK9f({lN>CekpD*HG-c^yJNh<_$>+fm)8wFe zz6MJf(Bxj$<280)IN)}s*YuE{ZHde9f2oZ6o+&MGG2@GVgCiog$#xr%82pPq#tXe>Zc#G=Y*oc7;h0sZwY4K>9w853>=z2pBc!=qDG>NP}iH1Ghd76 zEfU`>elCnFm-Z^BTh;Iy-Ipo-;MU|4)uZX`)1OO{*?cifcdwUIwmU|nc^(F0a_JxV z;oUWIK<+C{Z($a|!Y+I3zniNT)E(_aG+MW|5;nPsAurcpmx2V6|HHbxnanWZZ7eL@ zv$}as(T~HslzMiP;OAeiRbfO1 zh8jO+)|+hOE<#jKevSu|f8O^fs(x1M^Tw}zSyFk?y>(9|(KkA6)7aD?9O5a#wm^2;JO3gcRZ2D=t>P83hGz;+1I}F zbtgcwF%C(lfvM%oy1_zdRS%$@bOx4BdQv{|%<;xHSr&u2E)99|R5~KGbN|P)NHc6O zu^|xur={<L z(Ktvzv(3?eNU>9fBrXWB{<~PkDQ>&xop}=%ogTb3yhuMVO@yFKe`Iu(N>t~UFd1^+ zDRyd!AQCDh?>zUbdUoNgva%ybU^=w0?GgCa!fZ_aMQ08c;Ui7(LlZ+(?~k0LPZN>v zmvKrggwpcTYj&ZESG$j4MSSg_tZI(bx@#MwzL7>8E>-4AcDcHja%T{IsV{cKv#_&0 zQM0KgQDO@b{JXlOF|BZh+ShsUYrz;DAhjwl4l8WjppMeun7zX@+j+fvAN1G@EnEk4o}B4b1aK#HfAv& zGPRAG5OvQ+MI58bVt{QUde4FUcu(Pry>)Q&{ly>A?n1&zkzvdnAZ?T_^TXL269%z6 zN-sytgn7a}w*WknI zXu|Tk;T3ciRk8ebR7Ars+sCD)@d;CN)4xZU{UOD*AT zvPf0f$3>dbszWx!77#_^V#tWxK_QvJCw;m69?`|LRIP^xtBg46^0r#v&t&wH9AK^7 zSjYP9!u)TNfWJU|#w6_$C4{T^>4=bcvCF;{o=V0YRldrjv|z_6>J0cVrIX#8Ita`j zpUs7fyAL7{Uf4o9NlHm=Pi~c!xfHnHCo+Fx?@YVVxp2h2oZ=RorqU;gAM#(&mo_ACs+|}65JzhE&1ov&olm)-H2sf#vid1A z&MQ@`LtQ$<(F^We>w&i-b)f=!a`~Z&_vth*Oi$8%2tLCEKaN}RZIH1Y%x)CqSiv8r zzkLW=YT($O8C1huMGwMSC^C^nIs8eL7?U-tD%VKZBpDOTs|id{-l&;C6b(d;c)IZ% z+|x7o+Tv#Wve}W^TG88q{p`vxE#%w0xa_m+T>{OU&bkm4d+rS;9FZp?Aui0H)nkgj zrv=G?Bfn)4b&7RzS!kQhs!M8HMIuC7yIDF<6LoDjNT{FBpG8gW$j~b|ykj(m$M(i% zDREaZcnUp1kS}j#yWN=HeV}&ZCxj0XLP{LC^%69J4BwSTYef;*!`c<8D{PakYQDD( zrjD93UC7HC{xT1Yij4$hl&NPs&6$_{kIlhSy_VlbhuXpHM_s!C`18WSUm9BFxY^`) zv_$M0fE?VvP2BG5*l6iiues#cF}2fWhfGirsw!$8e6~$nlob=JqyzavweTjA?t~hJ zLmgX7;kVY%4EY(XcO*}TXx$5>{X^=tL)@Nc=YJwBE_%g{Tj!#w6q6tD|sh5$O(sc z6ECMkrs&^^LV(0al1qw$TmHK&vMB-OW+56rU6L!f@3Z?x@FkN5DIEgt*si-IvsV$?` zeTY;0dEdN=Tu_ln+je94f?y#pSaBsh!!LFvTz+(x*Jn<^!n_j6zE1=mGbOV0^fas@ zWhD>!Q?i`(?CiQL9)5kmKI^2PM1iWwhL5gFc29`v5&scixv?}4sCx66xrF4z$+cW6 zv@xYwhe%qn`*dddbt@B5yt();|&r;h?ybn9X<(#nRKm{aKE7mrdQzwq$7B24QoQ zX)RPsiMSO#xNM}}J=|U}O`GA@G`z2`{-=8LsJd2_NfoRISMp)u2)FJ?jO$68PmBPa zr9{Hpc&xX}rKd2el#4Q9DWL)1GX-;)TaryhjB&T2DVw$P^M$PB$h?kL(2p9hI@ajysv(|Pw+J+*&?TToE8X3YoAI=gF_QcNe~u{IIY zDX`u0N>96&TXTlFNlWx2{j!15asHxa_7Ks_)P?aOn?R?H<#gS2<~i|eMQR=%XK>mp zx@(6FY17s+&Ujl;-d`%{lbj=Dp5?dGBXNsQFPp?e5jcp9XKU?Bw}&Az{mnUqZoSC# z7;9;e_AzKj>8$2ODM8f=l#PY4Ot`1b3m zya89_aU-7WMuvpaiVWK!=ZrjW*ep&Nas@Mqmq5`E2&YDHX)TwgP;binHADia5X10s_?AB19xe=~;F>+eL0}HVrvQMAL ze__TIu$Z&y&_0$C7-DWO1w5>kv#sXi#xOPH#&(l>4N8;}NXvXdldaeZlzGn7Vw6=} zwA_CE!OcCxqNAJS&4cCR@Wp#1o4Xp8&CP%G75p^OX+{P6x~BxVJk`P5?6hgNXqU_P zJTMSBW*$O>sgP3r3^k2~ZARpszqQbUTOm0Pz~dJ`*T@vQ5`Og0mA;uSZMywyKKzyD z?J;L8z;nBNL>0)a=SG!rFtYUoCjsz>)Na;m{O4isyp!iMW}Q`HeTVvjox=8DRX^`9 zep_m9{oXTuQOtc_V^F@A`co@HK;t_q+%mtJG0;4} z_lnVms;``QEW4Ho1Q7}eR^wi9IvD?+Px0BrM*oIPy1?_$$?P*9JsySVXq9Fc?b6(m z>c`X>svIldk3+HIn=GTNM*B~mAnlp&tV> zZZ|9Us#I_}n81vC2F0*oY2LNv=AUWmA2K8oVrA3P+YNH=MWc>;O79hjcz;q&gWdQg zawL#|E%Nm=$P-|%x!n`$>GIzAMXO)=vVWy-lfs_Ixh$=pd~OMR@3iNXD>(@QjLO6x zdAa&?l7n13!Wn>!{LkvscXN{bp_DLldO|3pZ_AX1M_Ni;ScdG?fWJyhJimC2{<}HD z#!V?hiGNuBi`h{`a<9>k@+qfM<+GVXi z0LTq3prNTbB9Bv$OkM1icUKxPe)#+;XWTloh`Sr26sHy?K{z27($zFd0+NkqshfQG zE1h#SIRM_6kja$;1o46S)o?xPP@D4gUd|MdKm@_{ zEHzwN+4D&l>+Qq4rq&#Hh&Edo>qO{mqdO${AJ*jUwc%4+(By#Z`NhMmsY9h{lDXJ3 z1T|H9vk;a%^;!V*doozJ`(!exS?bSpB!P>zEFWwmM1h9Ai>*8ZkhA6e4@;igxjnam zFOQodDj?9zhYiuonlk4A^s&lcs^kuP56qkO@mlnPS`5=qeSMoOvye5v#G)K-80kI6 ziIsYh`uf!cQ~Dh3Rk;j*zI)%Taz%p5i?S7TzPioV=qV_Lt?ZYrX9W+X)brAZFY((v zk&_N=DNk9YL$V(nmJe2Tq$t1Xk^WHuPHnFDhbvB8C8cj^zihPr?0q$qh=}>+DuLP$ zwKtxEZM@;l-awUcF(b(X=xQjI#H?jDzcli3%C-nw{Z34g-cKo;EDjW{i9(T7`EroT zX|~{(UOB*_r3B|tBKlMAo03oOOXb7u&utbq+y(6ynz58O2c@*qLmr@O`J$`M*Nfka z23vaHQNLhJp+`Pha%zl56X9ZENnt~m^~AyLCXCE7P$NF?*hc(nXrI@& zR=R~)AL1((o<7!QtRU_Zx-o@lM(*TzjZ#G4cUT^v+Uj1Xmg8?166vlH8)~<7P{)sG z>VSB38Cfs{4aWno!0B`CBCUm1mlO30B}!H494c`4%f#V9Mn|;#gF&h(g{$f2{sLZQ zOB3?bleG!3yO#Ka-8wan$)0Wpi&t0P7yaK1vbz{xx;`B;lt#9 zTh#ekE7+EMP3oU~c>x8q%L6?Dm%W;kH-;c%8LrNVpG^^$`tcvq>T8^vCaM`NBU&b}X)rjAa}xga`4!}7sd?JWzW9#?u+ z-lmG!K*&4Mi!G}M=&aof#WSa@AD1rAF&as!gdQ{S+VD*q$P$ zJuE1VLBXxooT)6fj9vqFMN6YOnwk*N$P9L=O~fy#f}%Ggj!WIaf&lq{ z+*I%XqOAY0^qte-$(iUR)EleRoX{rk4^J0}h+oP#MYTFw%TH-8wZA;b(A7{o<>$8K zs`t71-Dbou&K}JkYCB>ZnwFaLY&G2Fi696>JKfdD?N83p@L4E=G0sYWoQ(#!OA^g* zhl4fUinlqxrgKYz*oRVb(n}Hz6UiH&QLoOG&mP!e`W-+z+M!a*9D>T=4^d?K%^m=U zhnqRAl@>u=$(Gmiw;P&zwl+Ao4`1P#eW7y*Qy+B>*0MIh|mBX3!bt!GEA~Adcs;nr3kA*ccYhzOVy7WiEq62$gRT(<7o}Q4+A@iX} zO6DZQSt<1WmhJeZ_J;Wk?xQZS4|NoLDS~b-w0ThK1n6^;zF9x|x!H3p z(am}0t*@pV`$Q0uvQ{=>EeO+EvCFa1IGW1UvTWKUAUz$Cg15C9lJEFieZNRIsajgS z-t_Kb)HrWLA6mae+ayA$;2Mss1mHFoh^{4Sct#fer5KB!@k*nJ#tYYEqtXwKy)D`E zVo7y(!yn)f1p^8jH|6bix$`sm<_z^CjSmcy5ZJ~O+I~{=OsO&=W-i%}u`>_;3?Ukb z7<{T)aeQt@>PR&9>c#D0uh%UZ68^$xR#kW zC+mMwRowHkd2IeO$#6JrVRSD8IGpXYlzAw`UvJvz1FDTof%vS^*B*Re!9%FtZRPeK zno-ZO@_vzKAsC{~;5F;vr&VDWJ^b1U=!x>~0Yfxo=k;?e1`THs=o8)+jTYb63+!S= zI4Y}PCTaDj{nM$Bg6}IL%ELFu^sVQxTcsz|m1H8lTk>W(^8=!Q+tugpHm=BIe77tH z)4(JBBpfeo*+hqDCT@R1X*tZDHZ(yn#fa6`L6_>+NL%YB53ZMv`XvS3^#N~zZikVNhJ$0s0Qpp7heJdRNhZ` zw-Io2_TMB1TuD942YT_vD**-A`N##j)xD9f!mZc4c~`$;ZC~5yL5*NqCVDZyJJfK~ zl*1uZ1wJ??_o+OcA(4?M z`wpq&E3s<~cTk=xUOLK&f?M8wEnz8AU;;Umn^NVVfyUS%Y!@shx06cfM8A&~6+`MA zCnQo?KQd>R04OzxJ+k(7cvgIx&Un)ZhZ?5qI&e_%P6)i%cy(a9s{ks>XxoL{FJOrMU|1{H+h)kZC1xB9XOx37h4zzX(VTm4cB$(lGxCZ= zMo$s|5ZU(L`0!88FjcE^MQ#=Xxxe3-E8MytTyUo;O5d7IfwA>2^tC=oaceI{3|rYQ z8@$$_zF_qJu`H9^hz0iU;!#D2`;OZ+-LX_#)))lRRTIQj63|t?Ev(LzfCU}+8kGEA zK4P~H?DeiYDW%;(Nk=5*N^#Gd|HFEoTrBFULSi0IUMNZ0oD`YQbmYJG)fIa04{~S6 zlcmK-D^5UGT4Wy%H|i8rg(*B5{N)H?gj%5_H=0i0oY|%_Rk;BYX&YvECq)u8I?F-I z&6-;70#tO$5i`Z@R8@AQw}0McnFTB{eHyk#x9UUmn!Th>lOKXB$LSg{fsP{B)eTa zD#WW*w$mg%09+*c{sp5KYAlfdxvB4IGsuuCHTV8E-L}3A(;GPqX>WYl9uJ}NPZM2E z6Ho7DcGc`f+LN2!x3Ct9^d#z|g1|^A?1LJp>=TTU^JV*&G3_4eP76uFjTy6m6seJ# z79S^P&$JDFi5_{WEh&Acfm2i&_licm`w34|`P{Ug~Ez=PoSaC}u_8%6PaA%dB zJ%e}c?pxgqH7@4Voz)gUX)Ye$=27#5t2~Xg=4I}V?eecGph*yu{7Hegc&=jdRM3wW zxxvM8#lzq1)f{gxced$_D6e(dNL-?b}0Of>!E`#CA)DV|E<9 zo_#+egG>`(3*62}%xzT)cmZtJ43FOh)Ri+N`YiPIRAU|Y#9_0O+ucP>j-_Nb$5U>7 zdLu2gZg{hk79!VH)oR_y@ej+#gg_CjU*-SjPrzU4o_+ZQf9QM$&ii5YUpJ}8<3Z`2 z>#Hjd^-j=%Q2OyQR_(BaN3Pb4R9=Ty3!;Ud|gXRSIDII>nFmax)pn?=oQ z)-hDuiL4P3J4J;MqI!ab6UVo^MsRgJ#Yz_YDXL%abs`p#jpyNy6prfeEJeKQd#4BG z`eQ4>4s@o4L-yNF`ooDjWOkQOYKTSx!J@^n-kJ)ae1x-<2}80CrfAN$e=4$J?&d@c zPV=IE%0cGjlPTxNsrK#2;RY7jHE7Zo4% zw){>+KI&jbywM+)OjIhg(OtQKh`gqdapld%jpwuNIzJTX+1Z`{u-w&UZ`L!>r-o;4 zw7%8Fi;>KgUr!$!HhlR!oLn1ifVDYUX7*`V!x!6)1AUX}r`ks#n z97|!AjQ@ve;^bMxanA&>ZL#M|?-Rn4iD!T1r}Z@O_kB@Gdtq7PWD1&9G{OZE(2TDf z1!`h((tpGS6j>=mG{(%*%0}dB4a1UiA9E@+H))B~9JB;QzMnEE>?-TT8Ds&Z3nx>D zzBKvnfw2y*QpjS65F42|IKA(8+Eu-ji<*$(=rXvoWYr_5W{;LVclTCEYspj=HhpIx}oZPI@9N@itO5eZ8b*Jtv7vHlGA`7 z!V<8VZ!pli0_qOlr6hNF7RWv8c#w&TfJL zKW$~H(CtBmwK|ZWMWGfmNm>!~%uO!pLWG>}ZM#|BTwdRsC7)~{eR0nfhdLgdi2*q& zFUH-q`n0B)ps`|wR)D8aH7kq-R6p6x%4u{}O1{T*dzFrd$S#En*I52<7?;FkD|+r~ zYkxL34R(<;Th~Tl?T%PVL}Zy8(;s0~1y87vfuVEM^lsdA(z9rd70lzzO8@sMr2~pR zCL|GvrTXCK_QbP=r(!$a=a*kc3@KkX|1BxC@GjwdVG7pGx~Oi@`%1ujaVR zTuH3Xu8Kfx7{y(PBJ*pdF6P~5HtX1m5_uK>52tX)L6N4BYTA@Ck~CBO*|*bG^CqYD z4>7$Yfi^0RzS8XjX`a~2bo49@77o`+Tuq)h(8FQt22VKkMjM;UN_NGKLwwbWpBnN$g=v9 z0#X*ccIVa_0eCIv2cC*&OmpRq!4iy6T~cmx{2J_drOJLR*v70MR^ND-=fZE&E(hqO zr}T5s6Cn-Xf|@T&E0cnmkB6r?yI5wv)M!<{Fy_=Z^_{`ZwO;F%>PRkW`YFEn!x zUvkY&6$g{heOdY3NL`?LN!Pt4e2FTDeKW&WcChN%SJCv{Vf^?HKU5<#^GGEYjf@$e z-Lo4HlJC0u-_A1w<&_N86gPG%R*uFkT*z|qouTSYem_7)3MWxKZLyKcXNWOxg>>sjiw~xJ1Wg5@oS;aV zkrB;FL0g#{Qhf{H1%I-!++eI8Wo;&_S;V~2Y;;7u|3dB`x|55`(tGWvrn&qw^1q7u z_=q_~E)2fyFXQf3?F_7|6TTx}6Ua=<$ZPaVRFndwe=Q?_AU)MGPk$FjJ&|~4W=bef ztmop5U5N9_>PM0Hm%}N&eFxbgaN&z<}36{7J){^jOx5&8nAul|Z72 zp|JO&nx|wYGfZ&`mFe_b(RT`#S!oeDNkBlj-|xVACRVYD!Lg6cPI1I52_M=E_pP|u zdEEqrwba6Wwz8gQ%`!}QCXc;Ls6zC?V0Rx$>f>Z1Z2=vWxv6|AtbJUV|Cbk#z}uKJo9%~Rwx zVVgj_$Yr}Ht1~Uw+5$iCKaUaT`sV1ShAU7i)jhbdHZ+?kDX$^Qmsa-NEF*PSp3E<( zc+|D=9~Qd2s4sbwAqbJ?B~nCW83@0dw^ohv;%ff5nT(cOi^Z|R0i1dvA zDR0RZ(MibMZ`Ta$*1Hn-94S?E-6!uw#92(J>ttnuCw`Z&&G0st^aX?1OkbS-!x}O0 zs5AKf2{HCjk%3%d<-|z~3$4UpSU~RSH}H<%|8nu>g*(B2cw4=J*5y0)i+L%|Zn27O zcnP09x^vgmoa}$5V9s+x%lM@{@QN>$kSL0RJcMw^iKpH&R@qHY@vrLK%;HODyw9$6 zYj?zDg2{(+^;CVD5KUYwxjb>q?;SBetkRo`gi*Zz82ODf{U$)A%rDY>x$<#y*^_;! zUFz>v*JDs_1H|dHb^xKb`dcLQ-oQ&c8<2e2s~gHz^h4-V{Us>znD% zJm0t)IH;#C;Lu+l(oZ|&N7Lq2QdA}i|77c?dk|}J9yxQ?W3A05|DH@hm0P!I3%KEl zN7O?RL??7uH}k0`okO0Pc+osb4xrwMiGHusQ%oI47cG?89HcEv*fU5W6+k*;8j3%! z^ImjV8HdTWiiTb3r)pG-yvEv_Y4+DvzHqUrChlWm&FkaghpSCr)qV16Q2a?#@|Ae{ z-_Wa>t>F4$`6#}=clVVT`3SSJ{Knj$go+?K5}Et9ndJg*oOZ){tOA%o-rLclf|I44 za>XL*V@s~|x?5-O?;oDbPSKQv_K+m?K^OH>zD9H5nu312`d^@)wb6e$lyP(aol@N* zbgY`QDNi$}Jx|k+e}a4zlNw^;9pm*}J06=7jUEPZ&xo4mm^s8ScH)$ z2#Q4p^GiKGGv-rwH*6j{R?y&C5AY)vo@APuJ^0#%_i~GAH*Cn%%+O-_Yi8JFhDQ+p zPp8Cl1<%khsarl;_6hKWP1obIzrht!TyDwfL%P1tLql|t$?4}1DX-sQ`4(VFiDjB) z(-SJwr!p9jQ++TWgXW+tNb+L0OWiU?;0=Paom{~&aFhDJ$3rtdL3m5Umkofy{}!EN z_v&@^Q8MxbV(c%f{vU zy6@=)C!eKm53|~fuVP*qNct$I)#gT!rRSZZ?p@}iz4k(Uo#7=fyqk~UtqV{wcyRVe zL;<&KH~p=`U``DDMCr-$OK!>woq22$iPBka9Vs zUO$R&UraE<8V|VQo*=hqsBJT4MK#H?&q{^dyv~IGVd;lWhRQBY2RYTT*w?#!BXyF~ zomphOJiRs`&NY20$NgpR7~Ht(e({5n{`+e5UAAviLDb@maO$R}0jF7!=zU{xOCspB z5_0MKF0?MD$t`^zZ@qVq3v2|JN=YlqdQbH6Tu{}1CjED!(qQO&8hhS9PNM<=2@ruV z<9n(WR~_DW;>bUxp-5S$*CdIv!uaC}I@=J4iDZQ9M{j{=OyKvN@s5?+WfudE3BB_> z)nvYzZKl|>Wn!=t1>$y#t<}U+cLzlufoYn5Sdm+U$sBr@MII)H=-GzK-|$OfL58lwe2R;Ie|0B+3oI`|Rfo!(Lc<3hK!bY|FETQr1RU z_hldAwwKXpSpu|x`|W=rnYzg8bxuLvY}hze&RrpRQtK7!dWqtNEG)NSuDZ}MgdIV3 zd0HrgXe=bc&s9@iL{CzfhIFpU48|{5* zNM>4n`>6dfbv1MH#~Wp#*GSa9PH(5^)!d9G42<;)euOdvHLMcXcZ9^GYz;q{V2WnF z@KT~s`$O^=eS+}cwb$tv140w@5=ZFo^DmFkh$u&CKrFIa6WIIbOiC{@+>d_cTRNjK@nY{?^fmRR!uO z_AM2DONj@IiBL z0%vA!+<5j+o7GXV2Uel5=aq)(#vwFvW_NheeBErP$N#X1n<(od^L#n4C{x($Fqjc+ zwvP+k?q(rPzmfw9fN^}*zlXWoYrW+5(lS&sLlj*%x6lJ091e^SXrDDZCtWiSV8{U& zHWD)rns@15oR#Y5qS%y#J2adXI*@JbN}9 zb>lC^-s_ovcH{NK9GbLnx;AVqy<=zy=< zTv1q}66XpTn9F;`vqM9~Dzg`yNK1{7%JDE|aGF`3%3woZ!WU1vhc-&(AnXJ9Y(U_Y zxPMaeg-cvlM*mrS>iCefUeVCv*Ej~y~OR|A@ox6l1?GN4YK8~pR5u80AOJwhT42Du^-5+yCmy-uq$kMLBG2T**tlFHQ`N~Fzxf)X*4t%IyHLQTk$ zPDe{xB(E7%93;yge{750xB|Ai7LD@$VX1+YeX5H>ltqCNUoPO9Czs^{6AEHJNv9~u zzTs2Xmt%nUyz^y6ddV2m%*@dQxm!;0dQngeJu-&I$<>OLDd}P7;|Uz3GJ;&{@#O!l zAx<{5Sv@)cTj>nKPG%RHmD+psF#>54gEa}Ta&P}^E}H+jR$ksA|H#rpgK?AIP_@4x zb^}Xd`CA>_4)z;ao^2E@NXs{=$S2b{hA(|bGfN{;IQ}wr2N_okN*}UYHl@CQn)fNwg?90a&p1sq&FAGLH&*P}`&*YlBmis0fFw7a07o5omVztj)~P390^kU3KD{h!{a%{kg)RaS|b1$?Fr3x%uh?rzTfJPz#$HNb@3vwv8|@rOx{H(S#=aO|wV!oz#*(}fi)Z83`?_I1 z;*ad+wRK;KE{Z*Pf9%8l|AI;LB_w?o!`D4B7)Kks3FXQub4Gup}mkv3)&K-kJ_0>3Z8^zK82jNz%Ftv?fSF{Qn{9L1#qVCR`rl6vCV%$bK zq@2CAhxNWUbgQ@;Sxk*g9sUBB~D+b06SkL?2K7B>Qzi3$`!nnYVN2{zxYghk%Lhqx+_Ze{D(#B zdHow_cgg*%L{$ZyAv;}Az;9cx4d46;bbjd${AV0D*JQhSSe-8jvjC(hKM=H%b1eQ7BN|M{(^q^91m;~o0vD9xMEJG) z_Y4=c5yR>m23tZ6ISY$sA)uaqmTmQ@M=OId6l$TRJY_E*{f+j3v7Bk)!)C;wV4yW(oJHN}GH>NxWUl^)O z*sd;aK9SoW)c2GP@^}+YqD!s>UEI<)WAj3EQJ8YxT%7jL#$ad~8RefmHN;6}a;D5} zEAOsA;e#33GucpWh-xROf3bf5l`y(r6U`)mGFdM z!=f5D_G%fL0KfA~(0S47BuGX%AQC`lZ16DPt&6n3YI?G!CBHIgqZ43BUWR*MgKdI^cd6F14~AP!y78Igk8ULL!5n zia3}+$TZ0dGOY;O_*&`iYG6Wr4JzAS4uuKG9?2Y}tg`Z?pbyy0wqK!^&EHN?4#F`` zGTSJkKgMskRb#vD#WL;q4V|IBJMTKDeKsVTPGpLao2cq9BlR!3vlqR#;6I7rRkjN8 zYdh%0fv)2?3$TF&R>_q2Kfv}cR}>$UZm|6~h;MPzgERr1S5NRuTB_k-nD|+1Yi-Io zMVR4VihiSUsc`6^{s%VgnUJ~S2Nyy!IS0uhlkRQLUF|?R@h+&%grjWQjCDGOY*USo5Hp+U`=Wor^BKfTXVD{kfrVr1Rj=;0~Q!j@2;L zu;HM)!LKnsEjbjeEce+lR@DSJ*;&>F1SzE4;-D5TTVam==an3Z1)0 z=rczk?Qkt2F{r^B-EG~3x^H*EHmFp)W?Uxc*&@I?;zK+SvlikfbyGP(|6sz$&Xi?w zxCw&|c$jQ`O=wOiXjaXWj@J2>d7MN#rj&I<@BMbc?{mbJ{rp=yr0bN%w5v8&Y0m|W z-14F^ucgv1p+^%W)X}BWe8o!-9jG!iaxvlG7JpQ6sIoyH`rDM#bW)8kUr*-~^evLF zictlcrUtBJ+N7Z~pQTy9-ZzBdFdnonsu84t>xqM9A`8{QI6;nZ`_HlNQ2VF%(fTny zU<2K-&01Aau+Y6$HF%rtQH1ZcvDXcc28gfKchKzaExMr3f`X@Wr_$ng4LPE;Y)KEx zJ?8gE`vFTI5N);YOtohCk@w=~)m(qhS2?kjgVmzcOTu2ohNaE+EGWhm&sbBu`96_@ z+Q#;s_$m6VXTlZ%+#D^t+u)8qsA|vrpovm zMKiF!W#qU==2lYoOw(K5xUv(*6X%q)^)c=ECn~yTIv5rFLv{G-qW?>xLI{6%`ImSs zX_bXWG;n3eD4e;@LNb0zKu0Z+jZviBpp%V<%Yrl4%{l9)=lCjS9#@3_Jlu6@GyD@U zuIk^(dtUw@fah$WYl@ke*KmUNyLpO1`U;_DO!Z5yn%L=Hpp57j3d7rm`+`%Q-P zSBR^k-0YW6+1hk^ZzpQK7T!E^%bGDM1l1YWECF+Edsr{d%iT}D`?$P%q1HHxz_VbK z&|(PO!BHk^+AnR0*Tvp(fn4xyOt zoLp54K)NPnHW}*yfdMny<|XdpX(h}KIg>LZe_n7lvKSA=ACiy?E2a^&H^m(?P^BJo zn@CBa?AV_1N+rRj(Wr|jzlmA6JgJ1KSb2}6&ZolaXC|1iS-DW$nE$nm*^*btU)2Wi zm3vW^k!?UowvXHON>}R=qozVjc%oC(l-=4Z={5U%!|8KXd8|;cTzbkil9mdLH;hYl z(p{^2aAG0I#^<&H+)Vz&n_mL=AsR%<&uiO6$XZLRz`{zL8*AnKiqo}sOrwT847#5n zSv2OMSpol&-#i}ep#%%tp*bCap2y5OPfoePQN{ovOpw9-`9GP5p=u1So>gzQ1b|h^ zW~9fF7qji{|I__aeBs&KnC7aQj-@`(^-gOlP2t@L%4QrWCSRP&!~|cgCHZh0TWD@J zy^4-VC|?6l6?G79SHFe1J0=z~l^K01@BZ2h4_7nDGp~c&?N7zXT6u0FJk=@~nu2|S z(HY~W{qN!vbBp!zT|VylY36kuL#)f>`pC#Qx1yr+4&0VyI{z5c1>!&}E`i2{t&j&c z=X#5~3uXen=`G>VxKdy!jiO_%Mw_MoU9R7|>hRBI%rfbAq5}Lr3YD%Nd-B4o;`*hF zT0A9k;-iSf-_C|QXwJA*o}0wqE>A$YGoK{|O#8n%g1zH>sipK|rINlz!Ijq zD8lEiao_W)rp7ybwA5p$w#XL?6=6P*yobUa|B8w6M3J`4i?=yKz|%$0i$qj@G;SND z-`pbK!FzRMbkW~$4C5R|>6~x#)n!-HrQx(G&BrB$0LI5tiZd|-GxZ>{txRsUu!9{< z6|Vw#T9CP6YJ;Q9#jeGt-vxwv+&dbD-D8NS9-(|E)v%c`gJ!gRT4F6Gn=}e}Kt9Io`8G9z(E{dr$JzCLCrP>kXaurq=iGy^6CHme2kSYngyE3a7kG3s^uh z!g^G6g~QHt&K-qyrOksP$dxM@12i5sK6dgjLevoPb>D*-7bL2j42HR*np-&Q4{Km;7)v)0ehD?7Oq(PN_y_Q%2aosl6nsnYAZgkzj#8Ykh|BS z_$S9N!{jWwyR8GHL?huw6O-SlUK@JoF+SRt6Ufr@#5of~Yo^gcEO=Pkttx81fX3?r9D2tA-7y#v&e_86P_$_UHy$+&HbAyL&iqM zTgJOC6hKEcB7%^4IhEcpmQgdE9mHR@r0oFK78=~HDuKL?9PpoCY!C_r$Tyo*S#C8u z;c7JAVRH^3ZGky@gEUFm*(PybOqj3wI~#7O{$VBH2-O?EGYVPZd!juH5-vZ zCX|jR65K8C{tf)xvlkrpkHqGSprIq%ZikTq=%28`(PWbSnPNug$+g-} ztpQG2FJj?iM~zRlroH$iBxi)5?=k>lEz%*T_IP@)3=EVb*JjpPNhfkP?Ps*~vj7Hn zh}M6A5Q%E8b$qwk2cKrXXlenP1~zo$c5Je}caXS1k!tcmh2D6uX}+k%T$!%yv@~_i z$~HFno6^P{rlvFl$duM`R|pKvY+vP$h`hE)o^|ZmkfqrAkvO5rK-D-h*vcG@jm?MV zBuK^;$}x<^8~rHShd_nWhW?PZy1J9j*(4 zJ9&Z3II;4H%aIn3fa|%(Rw1OGf!jiAOuKjg@O9{!ak-qc z*+a}e$SmaK1;}5_`L21l>&noBI;ikDC?-5@c19fIGx}#*1Cd=X4!IBN*i|^OJH-P* zB0?J4!Vu@O=`spg&43{oX2e;i5^E06lOgrh`>vnD1G6g?-d4{9)zu$qKq>o|&tP%v ziP?yTTHdJK^^i&~3&zr9Pf_*jR@?rZT~R*L^jge`Of-$cm^hSum6U6!f*`!7Gw^8x zBARscd>93>rxnzWr{Y@R#QMxIrhEqwH7j82jCL51D9aap+ozQeYB{89Q0k5kezRZE zEq>Ym?Og5b3}a3CISFq=R;vT_@;GMg5^xoQb$T5mUiD0(Di<}3$W)NEg##tJ+oAl&h5HyPpH z(7%GALN?QG4|!-`d1c!BFiHOtqWljqVquAx3h26uG;cLBS@_U0ldq8+xa?cn$aMOQ zk;d8blY25Ulg=ujAz^&z!>M#5E{&c5QO)z7sgKyaEEDX#p((FDdq4H~Ig3uaJ7R(b zcIZh~SnhirG~@6q1jdy#`VseAEJJ&HE^#_?kWLA4Tu1Qm-&!l|bq6I5d6#j(i6uE1 z&_0rWMw}kssq0?hpDI9TT0h&$1A+KOSXu|5%EN?6N<_W51 zK4vT{dJF~}np&Xc2T^gjxv2G|;PjqZ1(m^@wczHmS|k8E6gE>vXb-U*d3nl(m#HF0 zW(XKq$Y^!$6x~*QONr5@MI;VSM^J93%GI?56Itr<^n7Rddo}BMk1^qm@RWpS%HK-T zM(GVn4CCS6Kx`Ac=Y7?;My`nN<0v}{%0uGBglTkMCo3BV`vm5MdE+jCWH6n0h*)?J z!DHtl2P>rPTBe>mq1mt?yC?~f!0IDE<*D4s3#!h*8i-Iz(tHyVA5xA@uAAJi?Yv1~ zM+c(GPfPHtM#NfhN?uM6h|GjsPP(=s^&^b=f&UK&qaU%;EE7FbwoD|Bth`w_yX|3gctepK^dG&&zh+;tNd&5)AbI@cCS|)@2?FW8_Xm; zks%}qywOBb^*>KB=Gp%dBa-FBk{W??*9cZSRBdSx%1q`b@d-J6c$}pLzxS~9QMG># za%wcHw=ScpzL2@nEtfGQp4({Gt>x7xr{=LdQO}_uv>1^*QkSxG$?p9>UrTI!5%YokHAiZCTt5nNBSWMJ*jxGd7-Q9%A|sx8M~*10;U3CpHVLU+9 zUwP#Hr20jg7Mc4CHqHcNZ}WaiL1q_Y0*;O$?D??fs&A*nKHSSRSScdohV1vcD^ z8Gaw99-Wn71)d~>g!z$!6cn?c9XFXJAI5Gz#kzr%ci$Q*J((<1$FZGUX*{KUli!7QEsz1N$S|j6q1ODGFn9D zFGiBa`1>!rm-7%m@Cz9bZ>mhjrFkGfzBy_^XAu#-eq$|$^6#8@%I<(aBh{;p#dN_& z-0Jt{>7g@8>8Uo{!66rL8JXo$v2SguYt6o%5`$i%?Nk_m8rCPm9Z<7Rt#+ax@fx51|;!43;wVof}SK(a;EiY6fQ*o${ho zJCb!O)?@BNw$Pk`5zY-{TC5f$BqPwU99ZnK8&_0j-g(JoqQj%V?~%w+|BSm~DE%Td zqsk4iJS#mV{jj08@1xF3%D4Qhs3C)6&>I0+{;V!d_g}w^WBSjjWTduA@rt@q{9Z;x zzr-!?9=$ULfmDdxg@n>~WJ7#S&m`HubN>e*?$T(>2|2%|lM4Ap3ubNFT@Y zPIU$?sDIG~`VU|_E$(BoF9QKO4}B4|BuS%$G9oeb-Ul2Z6JypZ0?Hm+r0O#KnAmRP zwLHohz2A!!E%kX4>574>uylwe((C2-4*?I&P9AX5)c4M19mFk`nUdaBR@sX|heKq3 zo-q&pq3E8HnH`yNq6-g-^Drtu4rr@yiY!~CimNJZi7Pp*@C>DJY!)hS+OjL!D68kx zd}FhFJ0$s*n7#=7M!Sg+6quAyl`RlI|saB)y}lX2Z3R1JL=6om>iE>&eS zHIQS`I!554^s3RH{8$W(9nNeWc^@~itX$sbM z{REiF%!q?e=_JWJKRG!A47eA@K3QUsN44B>92j8T9 zHvil74L8Zmq6<(R&r_y7hVZ6sYh{B8lrhbs_1dZ$9L{M`HG1KyDPc|3mFJH?@$U^) zSE6%Rh`nA1WWSi?s!ps9-O-SK}eY_0NK4Dxfgld+wZdj36r!d^y)AK4$6o%0M z|LX<+X9WKTs0bw#CjbkpC|_;!5)`Y=s!h}8Ds~2)c_nHNjCOSA*WP(E&%QINyPx;4 zE%JJrmFLt1q-O!i!lZQ9i?V9nF6{mb^Apb6p_T=W_#wNL_1{tI^SESgr4y2k>K6LS zH}d(TF5~qShC^Q7v!-1~)BO4A+%W{AZ8}NI6xb!6B-eyP1AYNY;!AqMAF*u4S!Z4G zZUL%)i0~k|@{=|N>pbzR$k!la{65jMdGoZ#N>-w67(o>nOz8>nMg9>X)Fk(*)7yc^ zuTq|*NtW5}3yE|gwx7yl-l&lA0qwSKeQ-R6jSOlV`zDQ##ioZsHKCHNWzh6FZd2LY zO&r$J*Xi@H$!y2wRB=Pmy`%RlDDQ1+ho=;}81=>ZS#1Uxl|Elykuw~IF3Z5jL0B6h z>^IJbc>{c7u?}R^nmuHmdytaC*D#naAP39B)ElYnR0naBbf8*$P2{7dsUCpr zn(8!58Zz_kwo)%v*1_^YjmzjYi?Fy!etbkW$tcEtN*AK9&wCS@Z+bF>V++d?-g$NF z-#^MKq$X$OL=zR^E7-Qv-rSiGeGtjXd7*7%WVXtcRGQAWB#@`p4yhmbk@{eM@G5bY z|L13R_l(|;BOm*1M@8&zH-wdMsK%0_GT8)$EiLM&Hgo|?BY zXQ@U1pcc0p*Q`r5@Ud;%UA4C`jYL!(eyi=abGH^2V-mn-NrhqTTZ&KAk|h>?7*?(zWCikC>0UDTLB%%>@{c)mR7cxGW1;asYiB`oeHAkWdYvdQiBydku~UJ* zV&OL;_V}4PO!+QKMTUq%FdOU}lHe9$51WgikQ_kB+?3`uNpayak*4XQgy&q^ zqvf2U3dxUX{o^mnQXV&s*1u#lU>-WEya!GsM_=^Ais@Pv4!CnF^nH#y4pu{LBo&f-NPA4tsy ziD3+hppdQB-71Wk(jXHs7?f!E>6e?RQDdl|z=zj5-ZFaYgLr6M$WZ$Tw)`mYhi=rS<<*UTx>MkOJ8I}LZA`ykU9~z+aUTXZk%4W^o!l(X=;!sg!> z&2!k_hg|z8bWx1V!ZPH0ecnoft~oQAw_@;@KAdENGhu|LIrht~EV~cR0wGeb^4bR6 z?ZJYY-W0&#{Ri1EWqJ9c0kppph9p@S%_;69I=_$!LYO*c`N%Hojq@}Cxm!VS&SQB= z+t&^_MENPOWp5-Ad{?5y67@g4xBl7H)x|C6EFM99U3-KDF*HQe+Hg;XdJB zp}T)Tz8_7NV2UsD^FS|L(njXqap@@D``Da!PS=5w%XCb`e3&5`B!z*huMAebj*A+E zm&L81CtXR`Sv~lytomIXEN@~(s9?ZU%U(+sfK{q^?P=HG8ow&7eUIYzqnKZb%|lo* z7C2jW(rE$06;AgotUuQtzbXmbe~gr=%P2>N@XLD^7+XB_gU`mVd>=wNvaC176urMz z$Ja11X>_Sra9uQn5H-ZI93sP5R0P0K7nOLeWl2Ye;eGiScsqs$RXxFaxH#xn>$n75XmOQOm+X*hF zCy=rGZJ?#Tr?!?d;fd&cvqhMhzI91De*FQ#It-pU_R%;~D9c16;NIl#=MC}#TVm5X zEpS2{rQU*xLAdm0LL0T6`vd43hw>hO?W!+tkr|9oKYHmU))Ut8o|cbCjBt<1 z(NB~J{5Q8kKT>qs+9sLnH%rR`ATO3n+YKLW-+QgBHMqN}V`r9IJ)zyoY+uuG^D2i! z>u+VX2+>IWgG=}lXG985ezL9z@Y&FejJU~S4XTi@_}xV?;=9gG?5&IM^vU&K9JJszcxvTkJH3UK zV+zw9t|^Y#D2vG`El)@0d6e&NW6ZOkRsM^c>NO3R?hOkSx%wEa3IwX6xXVj}JuLYL zLgX|B9olKGq@yb3D=Af`O& zGv$24q(p$i7HG8mWX=u8Je`SLA)J%Cy2}l=oC$9Ut`g^HOKlw{QUNlZh=S%NK>EbY zbEvOsgVYy4)fHXNM>w{$nd$_!t?fsQj0&x06$h`_p7-c!r4ZcP^Q+2j;j}bnsbe;N z3h^g65R5=RDL*vv6gjgyI9_c8W)%$1w&D|DxpGp4h4GN_+OjgUh$aWh8`WKY{pm!y zOY@ZTT#82ryn4`A@9wZb5B2%^a}b;Tvii#Ekwrnjs{YQxe}L_BqnCvw>+e{pGA{k= zOOp9T$Q?aH{nzmkSPW0yx}5;D_r|zD`>MMmlf3- zGAzK0*!?uHSc-O+$T$BxJ2YZAfHKRA5%NaAkHWm)Y%_`P_vg5c;pugK{9Yjua?$50 ztlZhJd^YgiU_5?HvOd`^Pt+l!s4yc6miv3mCJ(wmsT$lVwdxM3ry0sQ-a??#dj@dgRE7jlNtalAfOMh6tDc-~uNdLI)P;4bL=w zvF=(P_rgynJYP~&ga8Yl_9+Z};YlekU#`v;Ut7=Q*U%YB{|`{uyJ>0MgcU_a$GF-jeL$eb4NL@Fh$!FqC8Phu*$vh+ar=4U*j1MZNqR5 zHdN~mi+07>fA6bdf%c)Ay+_^R?NqB7q?vmIrWYZ*Z2(kQrjOqX1?kS8CHQl@aTEcS z3!taxG=oF7@tT9+_v?pW=d@eLwA>*c+V>L#Fl7|b{@(ZNg_$AXo4vVNc4i#orJa<< z-8A*H~rNxH)m&rpASc8C!;14MMkp7CWpnP5kyVhbCAFti}^Qr!sDT|pJ@;@n> zt^LK&ZbxKBWxb*r$vY+E;#dAH#WmxYpe=_+C^B03D141BD@RaUpe{nni~Eo5ij@FU zuO7A-097@9kIb|S2W#>TD05jjn}OM8s%!%CHSkr$dHcv zsq`2grrWcb)ZEmk_IrGm!= zd578+zmlGduX%?Z2kZ z*u9j!Pcj=_F!0(rDS17}?~}t~H8q;%e>gq8WwG5MMU`ECONARrA#--iO^5v>#@8hD zZ!319`*rIuq42wl9Sxem23}NiwdfJ#sWb(DQ7HyrUE6!rZXl0OuT%M{Fws(iIyoRM zV&Gsq6_m_f5H67XRZV>VP?++COFo3VE77gstPZf)tb6}s#wWrET|^m&+k-nJA`6?& zi0tT$UfXItK`J8s>%4uCV#d_1MI;PWC(9-!jBo+<0g+zh&hWfHSczpjIx5@)dId2` zj$A|HZf07IuTfmynloftob!z71|i576r;S?Ewgvu#=-rUkDS3^dovI%-#y3`z21b* z!)?ARblZ(Ss(J}cP#zk(COS8qy8mV;fi#W?vgQVKe3xMqkTezJwY-h5T~z%E>6!@S zo)~~7&WH|IFvXe3#x4g`sdCTco>Pnrf6+eY@Y!X~RG3H3*lgxd_=yl&^C?mZi{(Ov zrP@Ihe5x%RMi!sb{0}(P>ey7wr=Q7bV?5~IC=QW>ZQ5(w z#?F64i^gusOtGM2oY+w?q8-E3*Znf3;dE^lA||D!rIVlHyHQ`6b%}{1{uWU#v5}@+ zk>ph>ocUyg2wM~+9IHiln=*e$n%>TKXz!TX+N6(eV3${Bl3=p@XP^mWSXRDA8i<}X z=x&r~?hClF+n#qH6{%k^n35ot`uV@ZoQk-#=8eXRDCaP zPFi4)Y-GQT8y zK9dC?M5mTNSyZvtP?P?EC85y6Xc(nv$8dX~URNh(q;&>h(9wq4Sqf+%RNN6dmC)Gb z2?BB2d)*tkT}-mQ93vxn=UY^Z$_|appb(|NOQqv~P!CU9d?uDiKuw0ci6DTT+3bZe zk&{;iA3~u%Du5`EWfpY2H#+!6B6on2UkgCh@wE9Z9h^7MBSGN)eR~RXzjA_)f|zva z2QLQUvV<<3#ByOK zUwx%8m?LFTn*24~;J0@d4SJptD=EzRn+?d;orSAIC4UlV)p6IKzW$6(}9ndn8Qku*7sV1OJ_A!*~2KW+@a#by`ctAD^yQsc8*O;DqkSbT_$ z9dUd7uVhst!rOUosgK(v(HM&+UE|FkkAS}ztf1R{?&~#6g3w3-LHh{GwN}N=%B48E zQo^tApSJ5o>?1ZKC?RG*fTla2@~v%lV5NL3HLkbq?%sWbOaC~ zjq}aQ@I32lQ9zNp(hX8F#G79FO~_VSrFX8If@XD1qs-T(B3RPw?n$k_`Ud*v-v%wq zF`q4&;PG-Au#GJx%tr%Zce-(arO6+833D-K6xw7rXJQNf)vG2#5 z*~CZo32=p$p{8KZ)0zZ1f_8`VMByPiS*ow|dZo%9RFY8-AHqX;1F6re8jK;3*M@su zGRs_Gac`)Y#cTNGUpM8z^!FZ(lnf;nyaF@4Le+9c8MtgQd5g5VNWCn>;hcqa-N*Gx zSs3$t2af%2=6$*Mb{@sVx*E!-KPCNX#Wvz`WVLhJUT#`$?~i~q?e)=qd1S#;?oMI3 z$oD@Y6r=%Me;dMNpx_bP{=T!PLYa$@J1YF~~H9cRjNRH{4l7iKaEqk7|{{53ON)eZ}DOWR|5PK^+8_`3c$CkzD?n_2{!Ud|2dj>CqtHvp5LAj-E*0VzGu!gF|B)auSZ^8><)> zVM7mR@L9KEJ9Mc_o>*8E67b0|%42VFitBr@Mlhb{gnbg^0Jq8->+$SK)^r z5Pi>#77^~1t%BcX+pXg2G1w!oQHm^SrlYxkB;Q_|WLP{Y9o@{KKY5-1s1?mqXG4q% z`#9iD(*G&&_}X*jiDsN)&E@@|nL``W7^$iIA1JtG^sj&5?YcfbLI@aqPJJ%yhT%6f zh3k56Dc(x9Cb&zD)rBw%~SuPV+5aV_>|&+LhDTjx!fg@w-usYF8HND zb81scB8)^P&XlX{9(#h4?9xA@-@VZSps8GbE@Zy(?gZWEHdpC7*UiAIismv}wH+1N zPQ^R8n#>UvtmL=EbuLykcM`yg0R;;AM<}V~S&k0fXM_>WDbM>@IVN4h(*KIQKWEp- zHYnMQQ1IvYePNA=*9YbgvMs!hHz7-XtqN;nEMN?Y>p;qF=Kjy+g~ou)^c~Hoh)sqR z@F!z@UKjkfmLghbn#A=A$kJimpa4z=i!A@edYq}kuOGUfNj?x+1$)1b&{$990e5Zh z!KsYjuPqBxmV1@qR5P(`eDcUIrksb*a8A$~LnJMo9E#d+@2`9Fz?^D~%d|kJ*7k3! z#SWJb>pVaD(%=2ve|H;Dars4?VsCKd|AaRN|F7`ov?3z%y8=r0g<|cu-?Jmzy}=R^ zjo0dJCz-a{cImg3<>A<^!@Xjtm#)3|$Pdd_7xyxLRoyK_icsM>Df8B!dOp#<>;8eR z??_q|i0Q>}V%)fns3A0!RkR)eL&u#d~+lI9= z+M~e7*d@QAtxz~JJhL=p-zS<4qZ1(GCJZKKH2T;zJ$OdWve^yu&0q|ozHyxR=JC6i zka^=TxG5M^tiklzrmgl9zXyaT1(U)LlLR$L1q>S}rTWkEG?f{}+(dX=Kjit6O_dE# zZW3B8YkK32OCd8@k>%z8V0-+ni2ud3m8GSxU%CUn@e#k3>qwZWKrNddvty?Kq+fO` zpI{`d{ZP=+Th0>EX{ipM`cflW`g`@XXvX0z@yp3V?qLk*up@mldudy&9v`>8t@$J4`Aqp*3T%)p8sQ1FU zyDG6G8hBI<_6DVLi4wrV~Yu6~3PGNVjFA`3VH1#>%Nxsw*vUNK>j6pP6e*)8b- zW(Gbew}#u`wf&l`?$WxiOjlN~P4MEi5GppkECO$0uBsY!pBi=RvRJfKM(o)ZQd;5f z>`O1sEcb_Hy{;o82<57xV{;-n(Q;l2#S6Z5@J^(Ec3R zE??w&T*gWXYcJ4|(21L~1CekuT}MI1qEPuZ$zA^el)jpCd+HDUI%pan;M5YBo4QcP z6|#{34xcv)rF5_7FmUdR@z1D%Y1Y_1IN64~o^?ld=_+9J!}731j;tofG0rg-d*d%# z0-Oam)A>Fakfx3NFrO>Eo*QO@Mr^;|{Z=VK@NR%6K}53|?(CAL>9xx&pQJ#a4PetEwF&UOo>^S!#c8=>k}+iuiU3gbB~{>Op->A8%O@w~{LqFO~s3;!^LpWsmcrAn{QTG42;9Lx`b=?!UZ zJtHV%wcvmu{@~hfZ9jj>H=-Zsd#YQwz`PrML`d2mPpaY3#)lxVsi$ro5lJYSO=xCv zlkj(W2+SB~7XyqED|SUB3%;^y#_%Mzx;_^StW$0v_P1WSrC2cok|xKW56b_^+|v+4 zfS`)KJ23)s8o1Q7YHr19n|0M1CS=J*2U#G~gKuz+CB}DLWb%wL4>s}-&H%zhcDCQV z)76?y)PlMAfVsy6R9J?SvM&ilNg&C2}def`8zD|+6rboTc*$B1|SW*P}tKkd<9V))c~0Z@aHqq_AHrpBj-~r973RE zd;$$1`N%)Uv>|jJSL=xwW_#KgUs$~(|GbXi%iJOFmu(=O6Bq^m4$aFhaVxfAsvpe# zRs~ZmTUa6Q$#yfu&%*QxUPqEN-0`d00q{E``!bU~%o}sa8hh$fL-B}r1 z0)SoaLXg;zD2VMcfILKva!j7p)92r!UYdxIVGnP29xE%e7r82f(&fTP}+gz4eK8)=) zwr+Q>9ol;1FuAiQ&z(hwxazCwx=`ntqNPZtLR8foUN^&yB-plCf)K5SyO38dQwB?S zYItsPYW7n)DhS)o@|~c?5;u@8D+Bu^n*#pZFMpfUaci}hP!y$S67G;GhC(9sne&Fs zNJGdR;@x?HO}N0_2BAZ1e(fz`3ZmX}W?=a9t+35@BTY-Wg{gmO>9b|^6bX9I54U%< zmwZ1x-9e4tQ@*6+1KL3JdA{ptP0|HEDha)JuW_hrh*2+lu@oI|FjSTKIxMHCPCAKb z{3{XAL&rrG^~^Wb-d)cz|6c{J&OuELBIB!dQMF`hPBoGMRp~GGuTmjBjXcD845(EM zN5B7~E$m-W~rQ)Tk6c^Jr@eX-8`o4J}5(9$G zB?H{o7RLl!L8e2jTvxTaIMaja^2pkLZ!5Rz?u@J)XhQ_uXO&;S=Coke`ZKzb9jI{W z8kj+uy?ptX*5ETw&Mj&nHFZAX0hVKD&^!f&3dQnq{gSfhx?2Hzzx zg0hUC`epyPY>zykD~j#ehwI1DGOuNBzc{Zh`(3JUxmTxVvk%T1oQQL2{>);?R$rAV z%*v(&5ZoT~i*A%q&tLbDv9QJQq6x?&l#;Tmxe>;aX(`ezzehe(_cCKoB3*-_pT z$Nodtp$)uyP%%ZRod<%9Y@hcHsoa=al6kq*Z#*=XP7Qi-_#%<244^Pz*SB#p9O|{c z6!RsvDOR%EV!ZY~_-WGu(0@7#Lp16LQThVjn%(U6dosy}s4K4^qZ|@q9Tp3vz~}7L zWa8_aGBW5@qndfF+(ZUG_V~I_(H{c2OVu5TW9bwaYPpSB_wOetHOjnwbvVuyjmnzA zYs7^>0RU-=ttH9N!WIvjQa?0pSTfCcYB?JcT5uqD?Yl?hbOPMAgLj8vU*m08N)l*# zG8PQq%*I4VAwxqf|5U{8^QYk23J*xLgl8~`8 zk8Mk=mCS$iV{&IikJ6iLN|i0yB{L=$Q7r+B{*}p9ar)J(Gl3?)x`}>Th!SehEOVyN z5sZ`xu}K`0uOx4r>m4qu!MxDPzaqAEX0&P@Z?1;qDaH1fDc;|djpDhQy8&<+h-c%%r* zU-58RfRp?j*PUpN7;ww9sB|NapwLbo1SwZI7aug)gd94nMNekFK;x$C=pQL`B(Xy% zmLc;*v+x-YAQDcOqg%5Rj1-Ai$OL^V_k+K%F`@6?5|FI*N7 zqw2f5)Tahmq6^9GEyYXJ&72-Q`re$l70!9M;a`_OVbQq7itfVa#*p4~qj~C%>6JEE zVVORof>5wI3aB6IP;d&$G7M%MG)G%WtR>Oti$LpMpx5`{9+@V}5&eJw;N%zEr+kba z`RpBU{hAEkD^G+_3MQbuj({8ZBxyCvOpN)4s$=*$osr3F?cf=_%qoS4X=sJ|J7st^ zJZpkJ!?ypi{)!{!w*Icg9RV{1*J9UDsN~EmqVPS)izo6G*FRMf1vvkQDfd>5_ZcX zeCjOefllHpoCZ`@s7wey(Kz_K7m?P|ji431t1z47t!Q z2^z8Gqa5E?$T_L;#5lkA5p!xrb;P@dX{VpMsx%JjgVCJk7wJN|uW@!)KUu5rKY4gX zV=Kvy)Q`#x8L&C_R$T7HY!430zJ&Q(6B5kq;)<#0?O;h+hrU+`r`~<3fqD^MkR?2V zU_G)62J^A!(n5p-(6fuT4EYJEg;>@9<3y_wv#EZ~Sa@x%hsdKFREgB!p zHsUqvspj)4OhlZQiEb+PE?gmZvDzNwET5jcm8-1!`|u1)uD+!JgQY03)t7~)$FP)a z?DOlGlW9qKTR9rvUneH?FV!}Uc^gfKGaJzkSKd#G?O}V1O3c`1>~CdNuMyc_$`?tFL9%zY zj|ohp;1d{-^gc@J@XFwCr5h1p$h0KiG;lg^rA#BsL&gegKxvrt){MD~{%{_Tl!r%L zDQEp)`3#*<9N~k$%lBi#^88&7S1(CRZb~Tg!%18h z6O$**{sFpM3=cjUX$u$pKc@&x0^pJCOmgn=qYnQ7ADbQ%O+Jta2t1C;`-DNA4 z*WDC6fA{l>=xko@w|lv+y6<+VAOmDKjut73@C!GTX=4A}7w74h)@U61z*?J|8|pfN z%8#)&B3)20S?G>`av-T3AtBBfZi5QyAR2X}K^BkFc-VR)C#+pb|6&`6cfp?A&P6_; z95{W>{P6ky zDA~jqH{2u1^mJ#0E`ONd2OkBjtpW}rx7mS%;J-x!+h+`B zx-RcEBUFxu43fW2&&LW77h1-|`4Js3sPb^m=*NKA+3WuP!Jc$(#aVNAbsarC&Pak) zEzwT@hO@Y!o37M(?qMEfJxQDZ@?BC3#1y%X9ho<*gyHN~P;Avz@03QHOGg)u&eDjGPKe&qAhuWap068-J*g^cD>#rGbxP|uI};+M`&LzKvfyZ>k?=Eyo&k_p zL=ci;W%a=BNNMEndURs3ork(XrkwtAc5=X+OjJZei#4i?vFa-o_tK#4W!A5Fb&LCI0r^Q-im|x}-{1xJ z1mBA$51$kFGSHPtqqw`tW{nuIwodNVBi< z%hZ9%6x%ISwqWeE>=VsRTp~EGx)Huk{q{%Vm0UWTbPJ1LdY^ zPI})q#q7;qo?I#Ci@k@`6QYQY%7PH8q(9>H_a@V6k!Vr66DAXMwN`*xxb{#$*lV(K ztNe)>zUNsiql*+1#jn=6x6W3J5aR7yWkp5BfkeM5SS)(u2H-Vw2Q1rYKWZ>^9|n9y zYyl?rWwCGKw{K90=rkxKjABjvHlF6>^uV!~L`K3ALR|C@5bpU5U&X!J(8_z)!pVOg@&Q+=%6x;dTQEOV3GoA;jQwaB$3iLxZiz?*<`zBf%g z{{RSqZo&uu0E_>H1`5%`PXil~Job&`Lcwc-gI3!U173!;FaHAMohKyTo#OolhDP ze+W!0dK?Ezj66ns^p_SsUH#W%6w{k?%kh)rqkmS)BM^`r&XI#piiYixkF@fJA-t|R zh~-R}nY+R@Z zS?`jb;SO%UeIDhC*GtRJexPzb%=^bxJu`_u#uMkDblsi0`TDtAb(aRqID7t@GRaG; zKHIute(Uy6bqv-Dg6UUUaujAm?1q4Q4+Wez#Z12MXHEXnIU_IaFSRNIIn*_6V+;!( zG7s2tjx#tdmYx2gWkKkO zHPU<99y2VB(?4Y~>(M|CDG}fWdUnfn`KxghM(Ta;3GrDeVZeYQENyOq@`x&JCf-#| zm)++j-ica`Z*fbh{~-Kh3QriVxjkzB;Tgo6Ph5yo5e=KO)RrYF4q`2eEWTjojn6$?HiSFtt+} zoyT}j)~Rujd-8l4lYB9)tQNeyEzw5tmWKk}$@5>iehovCp0+7ZUqfNQ7C%CrGOo>u z7=VjnO;FRNGg80csj7H@tP5qnBc&jK`>wZAU7le%?~9vpiLU9zZ1ZgX1KgjSnh=mz zGIj=YJ|g;PBQoJvBK>8G!gNzg0t{Y;v3j-&u%Nf-NQW2OY-4?*PrJlr5=)uk9=tzM zzMryoJ9+`Ts&N1M3Z($H^F}%P>^^??EKS&z`ocz|ll|C@613l*OJFMPf{uBa&J?h( zWQ61Hi|@Bpf8M)b&~q>4giSd<(E#SmO@KstlCs-#93@yj;jg&9FBt_$vOM) zwYrk<)W)`*FN#DA;t7;UrJ=p9cnzcvU)#(IdlWX)nJaV{ylJwO&dIYMuIM1_*LFV7 zKg||)E}BHBuBc8VL`Tr&ul-{gCV2T5ZpBUM3b1wXO1GlAf*h1O4s;xzi5%3;Y$M%X z85BRw>`FJOx*YDaH!)NBLa7%6kly^R2ijfRb;2>$$sm*nEFI!|H#LY;4c#K(za8;( z=i&6|TwAt@#0^oklA1C%r!m8;^r{6yBNG?&X6fb_>iGLiSrv#9IsCRAfx-r0o~knIpiE^F1|Wh| z6V;OUJubGPb|s11&x`ub1reL^M55GUz0CLpF>-adH9h4c=f1PNgVFuL8+-A^gD{)P ze45*hKN8-ftut?%3ou|7HgYiEM8=A*-%AM-NO`9xwI_efENL}k5K2}0^kV9he&KDu zf8G2Ku&Z}HwINydPu`sh2A}$ebzOYVhvEy$D32JA?}S^EGjzCYl^?ddukRM5__QzF z$~)FYEIMBfJ2OcGV*KnzOSngD7-hUCi1k+eY@TU<{^J&yrtqkq=q8``A7H6)r)XJo z!4x?6&J;}>2F(a7lME6LjtiUsGe3C$Lin^#bx>EnXPcv3ZPD1?1p{nlCb{0nDxCRP zfQj7h;c6V;%C*kg@t@rAk#kk#tY`7OF8NtK6rfDF59WMpJTk2!iAZ;03^P?AdQ;_Bt zN3(4wkxWI`@t_GrF79I-0MaqaT{?ojFR||XT6DD&FE|g56Q;qz3^E)1S(X7 zO)7MeWKop+-1WnIa2?+=m_C~*VJ`S}&R_38*2peMOvimQiOQt4`LB@lZ@inV8Psix z`o6jge7El6mBe&Y{ht{K6Q_e~^yKphL>H^q{&@p_&tfCIR2dN>N>Aw8P$99@2TZ0m ztocxLBals}rtW3%71jM^Z$U<9MYlh{<@F=8hOq$LKqik7=gYPS`6=vxu$7F;Lu%*x%kGD?)j}tWTZ@Ue!lF^%}zyyFUpNTiGe>VTEB-bIg2F7Fn{M4rJ$(_}^HkRS|twac>e^x?cmKd^y z#SXa}2kHCgC*R0seG1%!7^i!TnCt*3i19tTuu^u5|Bxnx zp}oJoK+5*=woK3+hZUO;{VFIx&?RjT-QL?CYJq2LlTtL_eth({Fgoua;6u$bn`di@4%cU2J!av~wWqI2}=`>1Q2eq!jP_F{!6rx+R-Hue*<9Z)+hQIbV)zeVas>*;2Lq&9%Vz z(%4`_jNzvqZyD%SL^`-TC~kn1OV1mJXN^!-epT0~~u`(kRoHDgi7@y7$on5qXK(_C#d zf@oD!M5=2`{STHiwk2d}&njxT-!N4%>ltP^SLh!+wKsg_y_Kg?^PK##c#>_ZLCC5u z4~vJt>SS;b8VDeL?4}VF*V&}_`s=7>{eTe_HwA%(5s7Az_k;Gkq|mrK!`Cbe&%CL* zL%XOTF`B7J!uh_$5DAFyFJ-8KRNr;!*%@!G)+$z`ae>$5~s7)ojsFbv>E*0oL5pH3IdvzU^Y0&OhdTJaEzL%udy1}t%I z1mY~(cs{v%PMu@fn0Z0}d$ZXQS~ zK_cO2rgdDj8u0B~&q{iUJBf z*agNaal9StNiS~{*Y|iya0*}C) zOB#pi16CnOGMv7eG-zNDJNNaYm>a8(AdUd4z=anpV1Ts8J);8IK|jid^S*P7z;DgY zr}3Zmg(=smHgFS&%v#rqik&>f0Hl_)RR0q7m1(rNI+C)J84aoT+M6osps}qx@KT0m zIs4)-EXgxvAVHWd-*Y=*dHvl*hTN;kY=YMdcJYiiJx*ntW;=1F0ok0cIUzL+#aC*->Q{U@?<;Tfo4lsr0{!(-D@i^KK% zoxzCw`^K5deB6gz4RD8cAMx9GqoQX_uAJ~Iogt$5?C7`+8ioLdr)$1sUmRR7FdR6L z@lgkJiTSFaqhym)5i?C$Mky2NoZ6@XJUY27EVI@$Ju5+E*4A^dl#YA|&Jg1VcI@=7 zYdnIlg*s!rCe*Agd-V0N{g0-@`MWcAiA%Sb{%24Q99xsG6O%VUGnm8%)ga1yMm>-T()?|N_N+cZmd zM0vzu_!?9S#|-OwUP%EmBgeq0dTt9i#r>YV6x)sFN6_)5e_C@sE zy~!(X=S-_#cvaLBUU+VBiw1A48t;D(nZ(hR6;@)|(AV)`EoMR#GW=J$Xd877)@X#v zPLC@(VS&fzcVT$;xV<#nB(dMi|Fxi&gyD+m^)lF(gzS3RSOA z>~ViKe?jAtryh{I_Nx;CVm3IR-X*2g?)DqaEfJT!Z)r>Gs+0BhG=XO-5rB-3nMOqY ziX&cZFEq(>lT7vdKhh-zN~X^C?H-TFiF!lChdtlDxLwdtY^ix-S@9>O)PYYp>|jn; z8r&&M69C&(tDoiWtJ?^NnxpD~*4+ukFdo*7e}D{q1fzA9(>_Yso|U?F1$hJKH0l~s z6@G3f*B*G2(xqYY+EKvSl#jxQ*D5r_ro+Lm-FIppDEHmhkueLFSXjm$?#(H9iMzkV zxLEFCtkt|fpT2D}wuQzLaNS>bUlJLhSnmbm?C;+!^aX4!w6ydV{R5DTuL3SQ!iug| zsMl8wp3#85Onsi&iTjZ(Kh2w0UXG$VFzt|Dpv;b>0GYAVN>?bXWK%NNz3* z70*Y?L@UN&s!XJMfL$T}P31@lmf$wG^Jc(_Z+v4dij{DS-1*S%fmm56PL-tKBo52YB{_(ck^ z>2A(Dl8e;K?VFmF<>N4Ko+#$B;0f-^%`;xywmJ~mrMCRGR%}V+TUg(V(K`= zR8`oGcNdSi>)6qEs+H) zY#y?n%t`TssEe>rmvC#yvKv^FTEk9@{D7)vU46NF!1RaQMtN%;3RxBRu3A|Ic3mjn z-(1MAYRcFrC~{JG8aJIAx4LLLONPF6tY0NDI~(vxJe#!FqMgnUz1|6EC==F*a@L!8 z^lKl)zi^5V=;irixr&f4hYk%?-3YAOT!)wbxk#p%=#l6N%9I`&62Ep*cFpikOLmbW za|P87i3$o~-vVOOZ+T~i+$hi#s@fF{R_FF_)u>#;D2=o@$fj>(#3rsGqP- zjKupW{0_Vp8oO(l^PEQNjFjWZD4dN37Y-8^b)A4x)QKOx5FcE>hYz6P^LCJZULYpR zh@9AFKz}GZGNWz4C?l#S;CZo&s+M?T@`AQiVIy&|EXU1x1;46v!?S{X*fYUe73-ft z>&LeNUOY0j!(QMFS!C^l#9}pw>}`qiMjToFYy~Z9Jkv#2!)2Kq%(OdRte8QNwAXjf z7Bh=qTMl&S=X5a(dueBU;2h2u=ij-MqEfH)e%af%{y_wr1Gfc)oLDOJ_n)e~(1oxQ zF!W{pKLGWI#ZHCI#}#)dhdIBy}=^IIJEft}mee6n@UTfa!f?_pMD5uT8id+{!lkE#Q&r@Um6H zqe1(8AVy||jQJVgFCLGW4Ytfbi1GKrx$}pjb7Lu+FjohN2~TAiu#mTddN<1;#ph}L zr0xW$e$yI-FQw1-3`1@%4073~awXBDx~j(#ucE4@e(S2okdgNSl@1mvJKfh&gr zt3HCzjX@^H*dPaZ86H>-Mtm-gUQ7J~V)n%ND=U$5aj2z2w&9c{s8^J%tx((M0=h3l zQOCQ{6P3zP2s7JMCNE$m<_f`;FVD;@SL*hAZ4VFq{!TXd1HbVYhW=97g*wTy+Iu-g zrm)e(Ba3;h($qBj-iTHbiV)S6cmib`nbj7&l{?Y6QD`nX5kD?H)&F_#e(9@lNmeMo zx&|&tlD|u-qfgQJx`(Cfm9~cK7R4hLWkyUf)~O&_MVYBS^ZLry0DZfmFhd0H;N^;OF52+E zidOp6UOp`98NKo~R+>rtH}(JgFqvR!790UMl-;+bYQnmbq{*J~N{H8DF}0^2+=chl zm3D89Iys~VYM>49?A(My}QQAfNchX&wk(%TOGC{1*-a$zqXF&_rW{WERB5q$(X*vA}ocBn2l7Jw9Kyq$yi0`y-FYKd(7FbB; z(1a(>N_lPY@O3`viiY0eJ60D=aSFpUA28wOF+&p2ijbUb_krKaS14cbTyi^e9t ziq3x^*)yKjK`02b64ZOF_tVE?+WFd_I|m*(1oY)M(=8MjA9xCM68Me`>hynRwMi}d z#@V%Zv}JNXMb{JEKx7KhX+`PbfW%0V@+{T%ryYi+t*Z|v#zMFW})5z0`qe4 z$W@CBXL{+zJkFON3xyl!4a_^p-F;Xl7r>#1p-?b0+-ozNmRi8icB0EjudvjC0|(6N z;05Z%ZtZkx4jz#zSmNzp0r%?5pZ=t^uJjb$b;siWj$Lyc`?_<`FA^?Al?ky1 z+r?zz9B6aVPioQVW-5RKF=f{d56d5w3NkB^!5dNk5sDY<3KH)^t?-jnu{JFIQz> zI-Q0fH=l+NA4oAQG$TZE0g(u!5lKVgt$zK)q%RX`C4#4Q3oPaJaP0`VsvA8(rz_%G z_Dn&%A*!O`FRylYia;9sP#moZK^HtB2AMVC0$|==-WIUWoB!Mqc$i_=xjUKJ>dpJn z=;*}`A3U?|m2%MN+xmn$oo(3Fv+9?o>&;X0v9}){(Wb0?OwhWPQ|j1KE>5IaHJn z2FxH8lC8yfv1t^=!+I===CQAhKRnFCp13K=TnOATNbbt~_W*3D>h+R`YO%c2ho!DH zZT%;#Yl_E^gLajeN-4lQXuhrTCRrT%&=k-5yI4 z@VZ67^$%@Y{C=E~()jkP2Hk$QibuDTiYwOz`sw`+{#|R13ryJX_UB1w` zFF6l2Z?d4IfU8z#ttsuVtXQ4?kjNrmwUUkxHNaehl}`>1d<}m3e2O&LZ?KO;uUnq9 zN-o@jh3Mzo;tCSBf95w(s$BZTPEkH#!|?%%ig1y1_mp^K!`TN|;%@J2Is;sO44Eh&W46YpDl+rU9YdZ|-G6Z&ssSB*AzInh)D1qv~>eWe@ygl#U1c2lY|R zVF1|MZso~pbz5mp-fA@WmIs&Q1gq8x1_%IiXf-ixzqmS6UbW%WgSM(MO=4fHu-(lZ z{rKoJ8IeIT=^x};HQb@^qd{PNpBFOPyHve5{~t1zA*SeE@h_^SIA@ldG@^?g#PM!) zDd=C^U0SkS-uKT&CrvF+V&mYb2&bn>#ZYx6SycI(?Uwf3+;jSEqTw2o>H5H_T8z!x zhk6KIvl8q;6@#mV8AklmPljSW6Y1`v=X(?tltsLxqyuaQB+vr@L!D94E$r6W!lK-S zy0dd%|B06}_1xr95%XB!!uG&)ib!lqLpi4CR?T^zS4bho0du>{J3;yRbvq-O;Wpm; zH#nU!Gmg>sip9Q#+ZN$~Qp50k`)M$VwdHjweX{TLxV7)%t2ouJ^$zR7zWD78`&1!R zJ~32Fvs*)5Cm!?x+~?VlojU2j|4PMIt9?aTS)KUzWT+vZHud)ctxvJK7J1SW;AP85 z^Swpe@2fne#O!#nj46Gfn4qnmJBt?jEw6k<0#w5GJh}d-B@1IxNbNT1bytb)Z@Q3& zQ;=dvnO^RIby%5ST zzp@4EV7y|)32v<<&pz={a0*&IL0`hp!#D{9sr|oj-6dH(@-spXSoCs9vKu0j_Qn@z zp5iUKfWxjG$|Ge|AZel*#y};w;3IJ$ddLxf4H-K=?UndiR#AP!U3Sgi%d=r5{t+2k znS|4|0xeCD$?h@@95KTa&-63L@of+F$-yKvyk)j#9r72HRmRvipc_!~u%zjnCw&*( zFnG;_e#JK}?cC8aW1BH=QRJ(^=KP>pqF@pTs-^4^vkjJs{3Lj$k3b_#9O^~TnCd)x zH#%KT!S7vX1CsaDKE+DrQs|O*!sCOs4R{y>Ia}08XhYJ}$5Ie=3`GIegCl@LCf0=2 zG+g8CbmPa%$B$vaZeJM8T$fxBnGsax_Mcaw-aX?05B_q>V)2-oMP*p?!G}MStM#8~ zm*+_B7&bi4{pI>`R@98D*~E$#ja5;9BNjWdbmzdYVmSEYwRMkD&<2!(z>uT(2VmU- zC9Eb#EBkwThwXU}-N6>C+rdF4V|eZQ+d>od1GeDr4r;ObEJ@(GUxoVlJWj_)54uv< zPrLqz{FFHnjKwoFauqqqo8xYaTcU-e`1hjHkq6b&L6w;@1G*$eC-bB~Bs|*BMBb%v zx|2Eb!12ES0G|s=Gpm<-Xj5G;J-LNQS%Ghn)a(5e{Vfc+oo@vDMIxS&%q`j0!O(x( z2wmimK}aEw$AqjW`N*m|pOL z1Rx-_WG~DoV5hRYRf*e4f|~LZn}eCV12=Dro2XxK$u?R4i|~lg1}*uC_?|GFQ*AYy z1Fsc(@X_hQv|Flc#5%SqFq=$3`Iyk9(wo%1A^2Fr*62W(eRW1?nYJ5VxNDI_b~7;s z3@e858bFC{Wc{aWkRIbTgBEufMJ!=_J*B+4u``EKfY1DD6o7v=jm9}K;#e$lQ z@}(f+SA#3fJKfyC^!}u&8mk%npjN>4@5&P@CB(-?y2x9D6gWJqfxIw^49B-`w@o@a zFPu$Z6wESv$b`}GG+j_!nnkjr!{&i~9yl(*o;34i=?UUY(p7+kS!6gI%t@c&J3V79 zBv+2DtOZ(I9Rz2~I5%CKtVr`;<|@El)>s{FkBBVG0_U3(W(vi?Zp`oGyeN)Q*w)H~ zB!W!5+NV)7M^jbLC%+$>OUI3`a&V?F7izrLrB*QIchnMedZrF^QuYW@Rh{{vG+jAI z8O}UGzP)Jp=GB$J>!V2HT9*Y|HwzQ9n4$NpO1OkvzU8zQwMGZoGlcxf%2eisXDXNl zz-yUh>HKtBeb=$zfUx5x>v7oJp|`hK2_hs&oyaO4lQuOW9+9#3e&eNYp{XZMV9ZUF zoY=}cxNu4x0-~@CIuMjgVSAPG`92#KldAzN3XXm8J2AFc?LS%X`T`WJ%N5PC#gTWY z6F`V=c;~ZOzD#JTeiU{pu=!ro=pJ%y7_~1LA@LFg*qc-7#yJ`-SSw}%o#a$y&!V0) zxQJ!Qx|>qh8g%Iu32I35Z)W!|#&Gztl=)PJEi`_aU~zS9PW>)f2eKQ4>tYhx7b3xl z9x`{Eq#NHGHPg_7ueb|0_&=x*09P`oSNBItJwYs=pG0ZbViwTlv07>{n~wsvhO_i^ zVTHsQ`y!*m+41dtzvAYdyrUP}%%(|FBGX{eL$F|D1A_Tp82I92$@88nreKhvWb~I( z62bgxmpUs#VZVrq51J;&*S=;FVnx%BH))8o ze$fprqa8xQE`~1ayipm15%XTAtGXi)K;E};brn2tv)iz~l!}3^`VE9T*ge%5uqpFg zYoIq%RRjR;&D#VXv1iFhp?>}9{ye9Q0g)6U2#xkpx}k|NryP*0pAzSx(&iD1ZTWLZ z3{Ek-7Dks)wcYr9yyz(vlU)J@!!{h7KK^jI7}YbE*?4k((aapGAa-B2aZoQ;i}M}e zZ<5BrPk>MeqQS(GrnLe68ps2p+)`Wguj5>)deYyKAn3ecCBmZmZKQ~zJD!%B!tc~Y z>|5_?aagGwgdbHb_@A8vTZtEKv&=byYL0mA(CI(i1C2IOPFKHduia>L)+4Rcxr}pPis>dg^65E3B+6^%5FPIbPH=lAKAXa_$o_R~ z*$t791GL|xPR2{G@~b_Q8M3~dOW#k~{BASGJF@p0Mn0VkKQY!yjGd~w;oF)u`_Lz8 z`{wVKWNONXvy8Zf9+R7mZ_HPSJb;Gn^qPK5mU1eyV*zK}jO$8H8K`cX+#v|AI3mS- zrB^rk<*gYJt}pzFUisp3syJ7sW3#YnAN2+eiw$K{E%Lb9^BqsLfEhN74yH)1!|Q+l zEkWj)*@mfZ)6&LvzA!125g?o4{trNX;GyaM=4JIEc*mfZ(Q41VXnOai5yNLwB2D0i5S)YoiwobGq(a}V^T@|F!NGDzyD6#`xUm1f<-ZAns(^kTwFKgbc4hO z*W}4>WFcJc^mdDT8n-I5QedS5j?QoXw#g~89V*`S`}=nGAGlU+q0vB%C=A^rGQ3oHU8t9J4)D^KKqS(*iwK_tu{d zawp~RI=-L}D^)OELFilaZ8)397r!=1Nt)j8qQau*Y11*P1y8HLItGETI4dG z?3W}wP5Ld-CjL2CFtX<91l+m_{`Kq^<3Yp$IY$HXY&U# zA~PMHRU?wln7BCKYlHQ9gRi~(>t0Q{`Qcz@)S@@SIf`8;Iooo-&+fH;oh(*vwI9hJ zKb>FoV|}KDlhPJiCnZ@dWIA|4F>LrnBgC;y?VvUT?Hh8BCYdzWT#L#?B~; z=!Bn@^n^pl!W7zlfRMz|wP@_&-J@d_+om}un#TGJ*cWXv>_<2Ea3IpxjY~hNlcv7W zU7lJv_MWlwM2g}bU0DbgV^dSgd^|9~G_zmP<5C&vfX%zUW>BMo!-tSl6N}4wgXc}c zvEAeJsRzkp$8q$vY=^=qIKz^5GC2dH;fZIKv76jxEw(vQD-Om#LM>nCr&WJ$XlGLR z=7Fu4F3S&Gt2%( zZ536w*YL56X^L~jm^lxB!nda-?}gIVc8tJ4UGJS zQZMIahf+VYBL-t5+4)S=`8_pw$tjqOn2AY(;Srl243E`g8h=#|8n8J|k3Z&zq1@<& z4VEz352}i>y+SV4OU#a_2aV~NCG&$_Q@@D@Q8nK4!6YYoZH~Y zT?iqN8D#n((AUuTM8oUs?6IJrU@i1uhC%K zy4^vqySU~rAm##>v%=+h!!{M`M7*0N=L7tT(w^25NQ-+?C+i!%KcGH3^ZGg%>wplens6!0tnZIi&zXgouZon$ z&ad(_P8>F{#JXtM>iz?~#gQQt%skm%XTubfO zCpB8!n(u__A4q|GF0}puF8NzfJkoAp>qwqnVlsOkkUQGT`^}lnF%L)T z*eZ3U|5YIJqtu#cksSAgL%LaH;8W@HfX{X;E_8_GsUQ4IB>qi^XVKS|QH@$|Jm#yO z;5WaV>OHFurFRTV8&_hhNqk3Ky{dw7PN_ptgJk67%uLdKB&L6;8`s%b2FY}_jl+^2 zl_lc9%?S3oS~^NzJq$6mEqT-b?$|2r}D`CwmL1> zY|y;@JkgR|miELh%bYe-kuR~nyV2WHnt6NuTjUp%sZ~B1@nQoLpCK#3fVVZh~B!G0GNSs)_?SmTge`T|Hck zxpHo!*IMN6oh^stg*oJ?Dyq&kUEf@u;^;Ola&Cr#)JJ>g`w#q!A0sByn5}l*d1~v} zV$NjLmtFh?UPPqrnLJa|*quT>%%G}5Qj_|W(s6CXifwo%2w{-Ac*dp;hgsPNTje)8 z6UlRRq?BJ3`N_FfLX_qKS8@LUdB}ULZj1c&ET%8JazlczAiPXh2IvK3hx5dPVmuNh zVarcSWDG#vQ{N7TLrF_G#!h%mCaRVPLPNjbKUg*LjFry9!PE`5{YyC(CiSW-vFu`} zT5fY8DAJ;gyUeLH#37bgW}4yKtnS=s#FQLX>XG0km%}lS3mqmzWWDX<+M*|qx;5;W zD&zV(O_hF6O*!pD!}pbyY#VOl29X?(7WTfukyRe1i)f=S@3N|tm;{k7s%|uE425Mn z#6{}?r|yf3Ci21z?9VN^&fwb5jOVIUKI{1^P9-*@;9#hzV7}!@60Mv!P$0?T=LcTu znA6T1d%aDV)+Wf9*iL=m!qLQVrcYffMA4a905W3htt4SgHs zkvB;-mnO1a-=P<35F9gQ><;_h%3YYL&YC=;C@uRS*HXgHBl@%R+yn#3<&XijNU1Y> zOG$$lz4SwKM4@Tq6o%g}01XMb=S{&yU+VPwdY;$P!;Vh{{`F-Om1gB!CYc6EsAvUP zJ)`@ZRU!?(o*A(ua0;6aR9x#H)EaPlW&gqHsiDqmtwM7cf3DWiMV#|TDK1{`8{SWi!0$54&^H;7QDItJN#v4+Qg>JvdPv6pQTCsr--^w9v@2vy%W_- zlH>KtOg?{D5n{8OxB7w?Cwr!_>#{d=q!&ieE>~4Qv#BElYL(k44WA;B!Qd!eKbV&f zXrT*}JDeNh?4-KPOl5Z7y8Up8E_}eA@!8(@hX(U37>1G2s*_r>221v)~^$;-dVqH_8GuFQ73IKlH$2 zF!#gx4y9v*fJbpJq8zs)HlA}APrkN3ukg0W`yIiR+(JYR41-3)}u$DbQjI)z$CI($j)G*pLgIANw7z#2 zT)s_)U&Nu3!-+_CIQ*0zNuQW7sM6|ex_{tZ%Io-$^`lZuQL6-p+`D;QCZma1kQ2Y@ zjm>LF0}2`Ddrzk1X>6Z7pE=Btj*ozTdKb;I5W8OD$ra^#jTKO*_bWSDsFF z0@jtpBDX;z-VL0Yw=VTy}kVtHzuxH#J@&bgEF=cg3pQ$*7Z)AU13^?FM)d6&`$PbrDqAwQJnHnNI{NTGaePnzoM}Wb&;Z@haZQ!K7h&N=>3Bp;dk>Ygi zzS74dws~EZr6J#5uo!s*f9C`@ltt^zf1aEZ+Wtnsdt%5dq5a}Zv5u#6(bdDqjh69G z4B7!5XZ6=7$7+uYt+^Z%ZV~>q!(%Z<)ZTj}Q9+O*C`D_<-n3PF1|df65u`@#7^U`X z(O9+j-n%XB@BP1@-}8Qa&i!_u=iKw$bMARApTU4%=8Dh)9jkqtcW4M`p@nDP?@zvc zhE>aG0+SmKIq2BTftsl=+}N{>gG-l^Hs(9{%(rZw%WBR{G$|UiJyZM8-#qOW!!f`9 zz|(UNXs+pS4T~&wrs`q`C{+zpFfArpr4w6~D9{IS) zy1HN;S#cP)zG7))Jrc*AzTeg34ON(x_c9#_EXTI{JY}vf-T=epwxUE-7SC0KlVq#L z$}?mI$NRKNPx}P?5ss9QbFUV?Uxegv*bQB+7!cuvyizQ@&rIxjf_l#!f$|eiFKrIP z+LD%MAD4-A3~NaHaZZ<1U`7IS*z89iHtmmiAkA8-W2>8~N@rZ&gU#((&v8?UHE2`}{wD(j(=| z(Yr!QdmM!Sl~PUog-VEI$nNcwi#n*HOcCB*_hHcaLD zhP>u-Bxuq4SypF!uK$`Gi={hJP9e{Nx)ulKk4>M%Y|e9sd_tb4VD065K5V6~q5ODu zlOsz`-&093q-Prv9cG%K2;=^tz>6*FPCT7%{+)Xhsy;8};`$(1tI^cvJQ+4e zKHS>&FSQRfhf1Mujr5_ShZ6JfS`A_aBcm00&VX4yL|&5rMw8)34n})LO8pvv#+DR~ z3E~0DfqxCwh{@tAxnop~(Sts>qU{MDt!KxyMbPxzJ$@zBUEl|^rGn!81nc~%cMms2 z($dO`#IC4PdMkmvBB`#tdkV{VJ1=_Cec1(t42cd*Z2GcTZu=zgm8tt@0+(NH`3HB! zbj?;8RN%JU){xs@G5gpU;DeU_I10$L5E~9gz-Q(mDv7+=1^9B4q{?)+q!X7O~V|S9uMf55a0Bb!rRGrjc z&1jfzvBc;%+|=2W<<%0s9lkv3a|je!@o7dYAS)N3u^$4WkTL!S(fuyb2k&2DM5o%g zhe$p9Zta)*}NoNmYeN-t&sokLQ z;l5VtPfau$E4^aLNz5-lMjoZn9}}PEuZhq`ow}F>xpnA4zo|m0h?dUDy}_Mh^Pin5p@p@Z;OXP@t}y61Kh`rP z@su*%@8q0NN1Y!MhS1qG1W09bx7JpD6Fzp9Z_Iom)4rmv@)mN(W#?_6rl)tKM3 zz}%#FnN3NDz$&Nc)6+K!ck@w$fP;8)P}OOAoivsze|=FVNYrLCwcV_H)Ckp+k1ppS z70M<3pAha$o4ak>7M6l=hFV&5oA(z(z}?qK<9yLJo5wmSag)z&Du8qiT+Dgk9gL6a zH&Z(1itFFs@1%g<1%dmN6D&mvgGlX75&LF3r6DD_8h<_z>Jt!psloPdP3hyT{tX>hu5siawvaJ8gP!mR>V7Bme{r zjTxtY{ggWTN~588uQz}Uv0@LIo)+R2<$q#@aFp#O+KFs_!f$bkhhajkubAAk%Qd4n7Z zo!a`b_0NNT#eVpe{TY`V4$tLm4g~)!P<9I#7fxiIoKdu2B7vCmR>*WJg()n|L8>!u z*pGAqLAo^;^W47`E~?Qbf%{2~XO}y2;LZbhtN+R`w88gutspn*+$h$~3SNN@g#Tsv zky*+Y+EewNkDuH=iK}?7Wk(@9_#pnWHAo0l6aDTW^cUC1z_n#bJ?drlHfUJ(-m^c2 z3kgCn)?C&p`jnvyQjffS!QiYKzII6a6T)vvs{?ojQxpHo1;%g3-k`C$`Km~1_|J@{`~DNy4)EHZg>s{vVJu@%oHQum3zT2n8LH{msygTVw1S|B1fG{ z1TKmC_s^;lmGias2bV9icqOK~rZ1udD&Fqt_v7MK+d~kP$-)K^xB;9>tK;tDdUTI& z?W7bX-)vJQr_{@O-)+o_$^tz}n@%ZCv(@9rwV|LyI~b%;L1GYXbs42YtUR?r%6Opo z{t-s>Ys=p=Mt#O_OSF-G&T1Zbx5QPZd%B@GDr1vYgWyw~0c55?isxa=m#j)V{IXk1 zO6qhN^w$dcXz%;C5VxDrs2b&zm7PG4>+|$do}$TI0TBvR|NC1(&7 zfzwNvli?ia%%dpc|e)w?ksdM@MCvT3=8<&nU=B6KICQPRW7Cyvg@u`7^FQ zRtrtzo-8M)hD&p_xo_rK>4XUB$EJ#sLAvx^ekAX%SGgyxSjr0xn@E+Z7B$dR2GB)T zq*d0c?uNhhgN?OoL2~^0J-*I5G!yQFNZMWM<*B5Dlpik;GB`9jNs9&KFd9@ zvM}EN32WEmF4l`(;ZwOaQ7f~wwMD$q(J~Cf$bQ4zYYIRY#HT#_Ca84I(kfy%v*PfQ z!}AT19s#qFEeMPE$v>CcKORrPOYHt#{~0U%Jb(j&?%=QHHiCxK7j0bxANVj?`b!#R z+E)U*KvYJ?R~QBtwWpdckGH=e2j?k;O=?tT+Vex+5Yf2H?rgP8Kfa}WrkyEzR+vJ> z?J~jKxt5!(eCUiZU;i%Xpj!P?U^@Dv&!O`XX%%xN_m15;&tIGZ=94YLZfJLM;Z`J2 z@arqIp<AGs9f3G)l`19|T*a|K-V`_Y*usrBha+8bSXD)UvJDD`XQ%8>d}KqSb8=|3WOQ&uzl`y^*OsA8{2?K*s9g6N1=P#Mp8*XwSI za_r8U%nI8?XD$uv+vdKR&JF(5mpF_3>FY4y>lX(?I=Gc47G#G0hGRPB`35Jj=eLiU zB~^{%pX2BX?=_=0Z0*@W{F`Gy{nRbIbMX2GmAf#LDOM=5lrA(~jwf)rZ6390wtbkI z3h51UrqFv=wlJh7^90g0Wl9(s>(|g>zlKW`L{4L-3_>u+gs$z{!Rzo=*^_a$JO&~qfv`Yd6JipR=SMQ6~m^o+K#{*^ktlbQ@qr zOtSx03=Ous?CXg+Bm04*@|~4}9QREM9A9_ecW{$$a`6Lf@pIjIiXd=^ZayJDA$dLx zy{2QC1M-VYtIf5(Bps_8?rDbrC4-w(LH^)Tl7U(ag(~nTdL4RJ>&Jj_!v#R-WyJ>j zeDOJ_-+*s}z%{B`o6q5%^T>DFE%gJ3#T{3ZC}0m-mx#MBXo^Eo;euqBTD)fP>OC2({cV)ksqSZ zR_wg>O&B?aQNx<|U{cNZ3)KNM>rJxeZ2KFv?>0zo(IV~#MvtYVi0Fn9cX!^bRUyhQ zup*Z2+_B$iJ+U+~@-_0ueIS^3sSE7B@~PDMj2xMiCz4LV)h>yu8^RzFisww0`l>jv z6s3g}Amc5i6&_ngguR1kI}|0Hzu=pShMiLpU?1j4nIw%1ZjxRos}$FFBzXGm4OAZS z`$7U=O-yOI}#SsyvL;J6JAARsH(SO_~X}yF|OR{ZQ)dIbzroqeGEf-E=wLi zr!6r)Ny5vJX<@Zx7Pk7<@Tm*LuwP@~_g1IVI&T2L<9jaJ6kt^Q=>nnV5gPv&0HMN~2*JM%I`@v2EmY zVPcJ)+3?iCx*Wht>(jBwLUd{(_n>gn8SPbM(e6`F8s7~utOsvmaWWEfR6!@zYr#_jc9Hz6Y0tg8)qjT-u?&aEUzD|7> zY`4mmc})fsb`~dPkR!59Zn}NRxRXoe+o1Do-phUHGG=LVrousx*VYZp@1e4f)UC?+ z&h`;Z`;eMTB)FjEnUB>&ZjVaOAxrpV+<-)F^^{60_tdD{y7^4SlLdxB?ItdPe~t2l zFc8r`SG!*70P)yw>=LiJ;9`NhssFSdtR(8aM$!6bH; z7vvgJXvzL+w%x#%HZ*_9DKLwz14b#Q^6;)yl$wQ z`7mXR=WhB?af1e(4*d**L3wNEON*ZLtvedL@wdY+x!H;+g2)e@kGIJ)oIfu3M1DW> z4D6hl^jye{q`}B>1?CEhknWKv*`9wAgg{;7IL8*U}}gMU75Z?MXu`ofSo9NmE}+M3v0*8uXU|5UM-{(R3+!$fa*a3%FO8#}g81(e^zk2!4$LQ5`s;(O?e?`;|J8g48C zN~@bUlbV_-9qJWA0XtrX4;3VN(s;B){P+gYB@yf3<+=yl7S!pJt}q65ViLfk-!G;n zFnj%-(IMtOtE6fH>+UASR!`mJ#;+doY8S(n?ZRcx-^LtV& z>!`krIi&QWXLgc+@Q-moX@S?x#Da0jndR!JUthgI*70XvXK05ozsj$1HI8wMo1~VI ziuQ+j3B?$zYSx3iF+0aJ7pnan50l8Yg?RIU1yqBF$5YccMaR$9-jS^;#OIfPpARk1 zsHm2~G)I>5l~&{e+^IU6en^R3yZW+jDoH9C6QpG|qtlYYP1Zwdg(DhHtMncS#?I`| zd;W1Kb%yw4Kq!P?s0f;m$a4BD*H7e@eaz@*8i~EQa~%kv4(R$(!fEAvf>W0rNxw^} z@_&#YVxN(_!@j~|COkDx>zmO^t9tmlSU%dyq?#Lv8<45mI@RV^$~4;o~Ma`Rj#`n|_Ys}o~d zT*dh*y*S7%;_G)r5kibUW#bvVP}C)nDmt~=6%wO*+&x|uILide2Wh;2*VD1VvM<}n z<-uO80aumOxbwx~nbseqj1Nvw2!BUzB&g=hThK+Z8AuOx4OCNr#)m-+FbdLq zzqga|d=T(M>o$ZOr%HPqw)-Jgfy~RnZ>$9dkZuyRtkw7xr!got4U*3%Jk1r(A;?2W zf@YX5ikf&+Mp)5|XLw=wegFNteytWh^tc8X@s)k-NlIP*+HM8bN|T^Rzc|7Hp7hDX z2nY{-GjD&h{`d3et*XX7Jwy?#^mB>87k*c{3RpR{2YtO!xh@alc~O0aQR%9iDOI|+ z#!uo=nFqV~Hb+$?^|WJ*(yI+?=arg70L15${}EC2rSH&_XX8iHGdiTjagjEJ;H(Yu z-mmU|?}RBQa)e(Puxd6kV62+%>4kI2xPvMh;ib!4d3|Sp)yi+Dd|O7$b}J2cVn~qm zW9t5x)DN9IE^)qudA_r$W~M9;WaTBid)k zx4zp~nFIiZ_jpbt2hF)(!O&$r#3xtY3ghop|Ab#q{5ya0g(Hjh^3?@O&G#}noW}=; zO5q?H@qxsRD<@t5Dm*cJbALGXYONPL$hyj8Um>6mV0twE%PhP1=>45AEH{<$^IuRY zsfrARu~E=;3g=kWYQ$)0eDr(^_w?Pz?J)f0TzB$!Wr=^a-N8L;?bWBb&#o@wWiLE<))VBATEH=9yE}i? z`;anx!_|L})mKO2vlg1O+j4LLC(l)v_@Hx|9ikBF3kUemKJwX*S^WxnXEK)Higif= z8w->xg9nqF+c+xnIqR*g_TmTo(hhsZ(xdc{SO|UfG^@?<2#Au1^v2_3$ETsjYf9-R zDxu<1-~acp=l?>g^8Mrsw6 zg1o;|0>)N=RacnJgd#Ele^)Cf|H6u+`si?73J)PF_TvRluko#l?<4;Xhf#e=eEs2t zn8G$V)BUdB!C#{E0w+-{eu*tV(Cel<0~T;iy|a*u{jSoEN5gn+=QUYPOUuQdTsv%M zS06t!0u_A_g@XeiYE)NeC@cT;` z^XeL{oMX{I!msy{^Xs4zriaDKjsK?4{&0+k*--h6KK|tT^fyBdZB)P3DCDbVC^z8u zIqoY_=Vsjfr@wCP$}2`USHcTc}Gj9+H)cW=hN7h2H}&c$$s# zle^{pHNy#Yw?BM3EP}e@?lV}d`K-xk-|K?-K-{>$kDww7Kzjx(sOiYQtlC|ht^kF# z8oF5eDRz7S^V0H9ujIwbcQZdBIgCt>K04a6P7iMkFWMO7ySrJ17 zMs{tZ)VZw&y7|f3yFTHJ%b~FJpkt^>8szq|?DU!VkQTUcfsjcJ0+~>57s_}XjHz)e z4aZRCBo||yV}xie=K>JwH*ZvK|9-B_pUM{~mE6r{*VL^JH-$-Go8;_u^uL?XdD_<`~lyy{9fKjzJ`NF|$u+vd<49 zo_%EEzjBah%%qso`}-yZj^Rd;<5{v%htZ24_#h8BIDW#$FqnWag$=un*5fUm8f`&_ zRueA5iK(S!JyfHMN9)eO-*~jJJtX_|%g;}3IiWW^+05zKsp^pGpkX2rUIz!7M3V8; z>N~UEb9w)DysfBCBS7&UC1C@?PRFkttq{?|e#-m!9YFsbSF=m{LHtx6xe<8K-zSRp zY%SlSo2m8YwWDVOlY@MwK%Zw3dDS69RYkc{xW-g1@d|9<*S+F2J02;ewfuldK~nQlSFZga3Fm%fFLH&&9SPQU zj^Z{}=z)Du+>JCGX^GSYktKnR(NC(dhE4B@CSEyJf_(2)dmUfwubWd8N0kK;}Aw>%`% z^gS(u8sT&ZHwfq}h!hkgQCM()*XErvlKZSo!OLCAg*)pzzMLak)1Ul3+#o>nse8*a z!RX`I^m%DntNj%xTJ%?OBCmhmK*BM1&=9iQ+-r^qsQo`*tQ^Ze$e%ahNuC5;&&+N+ zt*jAkP7_voi>O?bFU%)+=l=HD8TlyBrR{&jVu*7|=imdv_XcCx|07amVQr1g){3t* ze9+`Bg7CstkbdYygBI8a18?FRTAZ^B5UG9UaWCOtaDxc_I5rTGlQHM*U#-rzJ!+l; z4?i|4F%dTQzgUAnWXzlS^{OE^Cy13L>->lxgM_J9X^3Hxrpsw7Rhyeyeg_om4FHPp zzrv}a!idyZL@P=D+8QjeJBUV~1zP=Py}4!9+oKWW&PPm=9E-gd(>-&!H8 zroa5uZAM=NGZZ!ZuG((!pUni&k+<>nDUR-k;u>+b_CSC`i6Hyn(+$o*c+TQ)%66t( zN&fBt(r9nkuo7c91-qd9#eBWp;LOPmR3s%2vFEkLXK^!iKA!yX-J#X11Rgw%{VTYV=H;qknTAoai z&Jz_D=bTkL9(1O))-^)#D2adHd}sk%i*zCRV}9P}g)eJffkhX(kxn2>cnTEaZOE6) z-Dw%Qy7|)Z0bBbp`%j0K*zKpYN9VDyDK4Z&)0Un*EQs-||Fh>`oVG^Se%{E_2|)EU zwcgvm_<0KT1Q8`Z$alEKy&c)|G^b2*k0eZ@^<$@)B|T-k$DXBa7(BHWb3CJ;sIGgA z*$+~ftx8-CD~2w-;Dde&R-v4(jKJ0CtJwpp&oH?2h5pPR0wiR8&EE52+Z2l@`dsS> z--?6hIgxwB4z1IjQPmuE;l63PsdWltvh%5ZiIxyY`Bg*>fUz@NuQs_DQzT?89~+f+ zHdThdxsXvE_EBzKU(G|?^W_jzYc*GvJmV7PGY}HdyhC~GKC3$(f2)Kj?{AS`=S)H0 z@*l=Q8j>Y6x_REq0BQB$OqnU+(%59F@GpZg2g={g8{v{OIF7cT z!}wA@a*%%PSQl?Wf!YpB(Qj(cb1glkEzXTzH~1`op;Ntr$KJa@v3riDh$D|u{fy*M zd};jWqV+93NY8Wh+`#bG$xziJccsF8m}$+r8UAyg6MzeslKY_5(w}4oC>3pqt`==Z zun$2bSdNETq+^>~6FwkI{b6%#!!soj`@_|z+^|hONVIwo8$T(%D6U0*o&(TWc5& zvYU4LD+%L ztP-N+lTy$oPW#pXf~;}H%WxYiDn`kdVx7#%ZRi(#6qu*%qCy?yPEeooQX#wp$ZW7V z1Xg1~TdQgc5UwLMM;!=?Q}h>nr$(Ktnoj z6_EQ?P*|}yF+0DieT!*of^{oyG^@`6JKe};th5i}P0~tLqTOqcKN87bu~|MDguS%) zXoimP*d{3jmK8xUIcZ{uMvy4sV-C9{|Ch-jV}6r2KvEy}Pwz`P4;L`~*XieJv`r}s zwyU<@fPl2|C)kOM98gweXeP&N%eJmL-m5oVZx+5*_EbXoob7Ii$&HM=&X#dUQ&^0* z+N@I((a!l*2!*xGCDD)Ttv;&5}?}-MB}VSSI^(f_{lHS=Fzo%ZGl&Y z15r#um^_|n0Z`!v-EkuxZOc#No)L8$aP0d`Cdx-;L;+Tw0uMH)@;rJbpvtNh=cwaK zn;R8KjIzJGG4rYplH(_{WO33L1e@T4pb@*Lzo4~N#9dRO0EZI{9f)7;_c!pqs__GLK0qSLlGKaRLIHXDuey z)F35#>kM5sQ)m)eDT2L)U}!rp1EGAt{Rj#XW24Drs`*$nKtLabm<$JV5~*A?z4yAS3cO?dkF}oayL+951xIPcxB~IQ{U?a zkbfEUHa)+hko%50V#)H?fYs~3>#ZltIC{TKvf=pSbX?QDH7`;!kxVBZt55Bn5*u{D zrXnF2J5=*U1-JecJypHk&$5@t&5txuDXpm+(Icsy#@s|F+eV(D`?vvu&+B-@=Oxvj z_Watdj`Z8qWUjJ1Tr$=P2w_bwB>;EF(zHwz-&&8{Yxu>-2hJ^`KytfOdgTt69jLO@ z59KVk!BoGNNMRr%PG2b(rw9cvWPIlFqkZ>?zH!~yw-x6kL}*kI0Ya-?6pd5v??L?) z*1WwzvVTv3Dh{a`3!NH+$nKCL`k;G zC_W}cLD9d2jYh1%ZUH>bsv|i|f#Wf&de8M;g+c|yi8yp4i5i_b^8-)2huBVJY&>Pw zP4h+aTYgv^l!7xFnW)v~DVN$hhJgQ~_p;en3j%8bEOq2J81hR&xfJ8V1#+@$U zg6I}#iI*plo+@jJ2Yb|@`v5?lNA?qP>^qgl1B0GA(%%!D=>*!bSNQlG%LF?}3LTpG ze;pZbfxjJT2dWGSvQ{bdW!{s}D;1)7L=x|0wD6?y;@()Zadhz0o2!=jZpvt9Z(f8oedH*U~f>J4<6XC^)1)jIW^!p zWS^-OF+|(2V`4V$JADz98F}!#)zT0gdA(bK&a|drGN4LD*QHF&*e7EHKJ5y zqgnkP?6tkuRIbfaPtvB@kYW3BX1%{LkW1G(BT%++&!O)tOGI+L6Cs7UB%BHY4f3$+ z)qVS^{Z7Z#!T+h4;MyotV8R+k%?ob`W5%jbu4BaE#D(?`9z6`o@_3LK{3xIEuMsB2>Gk1(;G8+U~to|xYPk$Gjg|Q7zE8sYPzAT~HlabGX zPbBi*AMV;(vcZfir7ZCXq>2HbpFR&@4L7ZC`8d2_OG!?y{5w9^@{!d-2-C~NiUaeJ zkMJ!)`4f`u3Wo&DBA5OiNzB@&J%uwB{~!IbL2)Av}cwruR0~nsR5l#FCESb9z18?95OPj6}Z<3Zu3-gQlgLFw)ci-cBX>bAJ11$WjG`V z!w8)cYqfcRC-#x6Nne%PKWv!nmju!U(1JD+IQ`%8wDyD$;Lqc90{MR|4IK#``&W)M z{Bg^RTeYTgKDt0Q9lngjq2Oq4Yw9fKIp#MTFnLe)YeMDn2X z{k@X-_j^|+2xj)_r%! zImp>i`3>i90>~d%t!3H-Xzb0>cQsma$g5SBpd1%XQaO`~PksjIM@SmYd&A_WOSx8W zFIIs#lP#;wmdnh9Pl9-Szj4>rsJ!8@rJ#;UIcqCs06C1h6LIr@o+?m{ELP~%03^9C zZcIs`L%U9JzPtYsQP+NJN@-1D4L-%VOfj*XbgM1~O76Y5n)Xdd;3 z)25}JD)GBuF5mOfE=4K~GUl@h$dlFT1@h19N(786`mW2m!!sYU!ys$ouek{r zBFZ6Zu5;ah8`Bp*eqJ7TD$OsBVX;(Q##i~rcXXk5!j|Ri59c#`XXR)d0Rxs1 z?nO;Z>F9g4a-59eD%t^5TJCAPHAC)H1!76U)C5X?cJeH?ZovguHgUor#fT3P#1YL! zrjE0eEUno1N|cP$LgIDTT6$qAX6z6XcOi$PRj-5Nm9r6g*VGVtR#N?(J*WgmbX(KP zBmvn}X&4M69sZ34wQz(GD@a4q<4S9jX&f(@LA8Sm}v_-sbjF zHp!x3t&BO^7VHzL{po4`d^D(QfX-8Hal-}avlbu*48J{Y`{NxY& z`~9%;>ejr`Z}wkG`Qu;J>&H|gOWRDVA!kR7b=wN79BCGr>`poN>~*p!g&`*B(xc8| ze-civK%b$CP__Uuz?UI^W=G}Q_^FHq&WOc(F;8h5_RE(ATkaU0Sg~X7$nOSMFhFy` z#K(n09rY9}osJD@@(V#{yuEkczMn_VFy~aJTE&GCwHF+$Cx3TKF#Z-G3lAbNCb&1o*41)hBKZ_}Nh8qe z$}#@Ss-_k#@Iak1H`78t@a!}9Jqkz83Y_*i8-R;(sjkd?+)C_4--?*T1DTF^HnIB) zl;jQ@3O`V5i8}>v70+l_;!mo|sQ+qHyFg1gBz^Hq;nfNv{QcUi=R5lI!z!q)^j3GI z=wE-Y;8Go@9P2MwVK11qhL^}6?t}N_ zak$a0k|?t%FI#iy91W|_6>65Mcmv=%G}-hP)@QQ*x_T)Vf*eUVVupkxsD^gx$O}B# zSXifO%xuJ-)ioQ8(0rYA4*ITq2?P$}*{?k;S6T+VYKplWn#2659Z3r$tC-&Y8v%Wb z{miXzLP?ZIBS6l6B_OC;@1t-0jJOlq6j`yRN1<(7KqImtvRUJ(pZcME>U-9wOzh*Q zd+fZsjKM!#Pu;rp78Vif7B? zJVM`wRw!3crI-Bcv?XCQjx^6sVTI^Wq(ph<+f1}$xv!N2a$M1Hz1qD&f+nZ*s~zah zv=mqxW_n?#a+_+gR?eUwWW@QK%X$~EfKk0X;bPN>fq4&T4ACZ8RB%Mj{qIXjVOIhkPD`6M91)e;9j1l(v1ZQpJ z74+0(Dh$1de?v{o;Vmh zzU-`X;7YBuKj;T&M=92$*Lw!~Fvv?-7gdk(OpEjlZIu5EtJC)+(Tcy>8oX|K>S3Wu zC|B6|dYtogVgc>=@W<~{Rm}|WfJL;oj_qUdNMbaomC5m-?LfRc2pMAbKTzG*(4ZKaPiOxwP?V z`A+e>+?}jy+<@GU*`=83^ZL(YMRa_mq*u9PK^teLQOaNn)u~szUekgi6kvJ5^$!;z zm5*}~Kk^6Q;2&P@G-hOy2T}dWu_?Y`v@|PO^56sk{$q_h-)xn-@r=GV7!MQ!vt;U6 z60ggKtIY9%TjW;fr07hp4c6Hyt(^US*2bwCa|Ow%(gw1rnS9Iz8Q9*ZnT0$^Ai}1v z32a{bZzUdq-f=dTdUz-Ob&~lxlWtaJrYH>EbUQUVHHZOZC6bwTAL>s{E}p7mRho*_e4HCreo$ZT(<4cbs|^*Bz=GL*e~ELr^(S^Zf;U zFiKMGlt)7`hzWI;HocClg>MaXyS-dXHw-c{oAhKKT@oclQVy1>} z-H})1hdogMd5y<;C4DvB;{$2YSZ7$!{@c3bo4w&sJEunv2&O@Y!_|6;M<2bW^j~C} z{gXZ_pEyAmJ9cI4`HwO;i#@QDmC%14Cp40-ryPM7=SUv!5POQ}uOX>~(Mz|W=^Htm zmF*c)i-okgokMfEo5&Q%(Z>Y+@s6@%=rmEqZV$s)c%860@g7TI!rhk*(zMy(%)STV^v zzcO|09=R2tLg^#MA#eG<1IKEg)e4@OdV}3vlW?R$`yhGcoAN#X;mHjkl@O93f#J6= zaFjXy9&Y7K|2$Y9P~l7oQdK@%8u>B=25c{v&upB^)XDH`K0EFRWxA#=ZQ*a2=`2Lw z@2lil(|>yS4ddNVfi*5(vYP1qgD%+_`n8jA!DgPr< zHGBN^8Pn4H)m_FM6XY{-eIVz*S%u$FS6!ZTN3L;`?atIzWjI)OMtArNrSAuoxn>L8 z+didPy@PRQVG`a4L&p*-+0}O*b{%I1JtRd%JPOhCBdqT={PY^p%Ia#z>*Zo!*dQ~- z_h6FVp2E^Oi3c4&qDz!(C*4`98)YN&fSW})3x4_IlQH4pfoBuHXC2@#M1o%pF>LQU z4%@wg%wDTsyZGWoGnPuu&dH`6u&# z#Q*X?On3vjW4)^n%`2+A0V=bR!hDaA10m6UR6cF7k^M@%O}b!0g$nKoWzqKhZ^CFhwgb9$EaU40%tT=v8QR{dIs2mRG= z;_JnI;+FPR<};XMvmwbAUO}O!bSXu@a2>99H!K=Jw&u?2n_pvc=O<9vih3;uIGc}& z$lKWm<&7j!Iy`4fV;$DvyY;@hMljo8Nk1e8jeeV#|IUw@;CSpc`u9@D#FfhI1QH^w z4_x&5g)6@*nux+YP5;mWdnRu?z3>)}@(h znNKyB1ndx`=qmg6Jn#3^x~WX3lOtp*jeESubVb(GJ%MId(;FLxG5--+asEfNJiHxr zzO`azhGic+vihuCToosE=R7dH9+<)Y^a<0fOtUEg9lkq-hWOdi+`M64A?WA2idI?y zuXlJsfpU-eqhZYJ+=M(!(_H6nWAIr;oaVchiRILCOS%K<|A>}m-bwuYv*$rwYwWd_ z(u<6FODEY5b$GCQ#CfR&FAE)}5|YvLf!@oKSn-Qu zuRW(1LdNg+2Q~^u^Uo}nWEZBENZa!XkgNKsZ!@p3p=x{L$CFRz!_$k(;Ti`Dwhz|g zP1B1Ka^U6S`7xq@1)BMLkpf!YbdpuH?RTHV{;>O$2HrJ387L0^7WQ6VY5Xha4V-a= zZO?yh>WUbs5A4-3e`Ke=V!ZQ@`}e0?wYlw9$4u8$q-%yl4U5r!`q7{D_31i2Q-asl z8JRZ!qFdVQQB?of<^x9dpjm7+rLGkPNv$Nb1cflG*hLyGrr-&gOlgG`N0pqREj6KQ z0$53T>?{|u4qL0AB+w=GjMFxp_C<8E)8WuGA;;C%;w{_a(fg0p&Ep);TVdH5yC(5j zx79QfyGF#l-xzegmaN&Wa()ocUTHTsR*QhM#eJI@RQdn*?wsscJK7`(RaEke`c;K8 z80)Y&tN*{Dum2aa=uO+MbL_raKPY4t*dF=7PU>(`L!QzgQ})u4OfmgS zl~wI#O{9D3{1~1K3G5mke$tX*1x)>zKpZP+ttEQ@&%{WyVURqj4`AAiWz=~$x?KP( zm7l^z1}05O&l^)FT3Lc%T;o`#S~ygqTnV)$xit1ZEeku?hQQY*)e@KF(xQ|CGuBRr zL0iHXTS@JeLWe6BAaQ{mQy4LovmQ|(65m`5E!6L+H!+*P{4Q;gF&s^FV z3%{Ls!67v~1#KDDii3rqiuRR);=c>L@r6nnIKBdVef;X#X6-eF_l`&eLb_6x6@qUE z1G~$zXTqBJ>c=t>#xoQ!O88*QW&pw7IgjkPe*W9QM6q`|85uvoyK9l=4U6h(<>JC^ z;=FuBj5+LszH9!)I-mL@H`D5jXKQP)B`2L5o3_L|Ip*%Qgrfl%BevxoDn>E{s;sH0 z8~UK)R#^1j;ub-|Ib$T>Z*rDl-vL8ok6pm1*P+DjUswkcBwYxVtfQ` z@znd}`^n?fO`?*NVMr5K(n#{x_JUFI_~;y;D%MLTvs_WOp>ff$>thj&z(`*aJu>?B z$JN`X{}DY;2y9)<(?pleMGrf&QxY^RPYdZ|Nhe=2EPcK+lcGEPD^e~5EZI9C( zc#i!#-0H04e1E)z)d0hYskE46F2Ht+DB1mIfAAt3@nWnS`F*-? zr3~Imb^GN_gCNp;oF#PLEN!2iW!-WrEYH2-KO%NpV*I%AH2OGcKygSEAGw;CN8&;E z=ZcZX?_1GS62cC+a_9W*+0fM=kh$<-L@wi~pod@JiRHTIIsqX_rFl7qfNVI|p^N!z&XhV2NC)?G?7HNmz|xAhBlkT+$AEc*i*L=R*1HsIP0b=U z5`D04Qa+aPVa|mL6-7Y7*$-(ps+)b`^XB?H3S#%ullxR%K5D!D;Xsf>nFzz%-f2xi zIKHtNFv*cuf%9_IAMfdFc-~KYAXrTXu#b(()p`HQ>!*weqnAetPRv=Q*ihvn?eMc7 zpYts{n5PYtv7K0;6EejqgSO0WHGP5K_T|{%pX&^_NB{@GEHz{Q*88(yvzeHy9Szd* zb(HAMguS^isHuS1fjOs-!wk#RRF``GC;PEA)o~8Z8B)&u0%{JH=^A51@>lE4!vCX6 z7^0_~S4qqPdG|D+aYw22ZhD_Bt%8x|5*1QN7*Y(_bny;xwS=KdB3cJ}8npI?VQH25 zy8c`EJX>4kMs(RDIwo)5f)dQDKFX&`l@)&4-YOO~yZUwdcKWas7*%cJ4%-g&S!p?R zv2O%nK!DN7gsYu)DNZeKiT9VRe~kr$XTUTamFH83&y?FBnYC1IVGbi^ zJ<0p;Z2q{28X)G;lRKRL6Ysg*AdV5?qlNzwv7p~sI_fFOO?aC|;uK+XJZw;)5&=Rn zLK2zp;1Lg={ssu2{52>MuF}hr!jzKx(Z)mwM6=SP*z?TbxN*<%+EC&*- zmqAZ;oy8Sz1!xN^`5X@KJ1a&ui91ot5A~#YI?oJjW|*=zKM!Ei{J`~rd*dMxGqjeYaZmImb;zB0R5p1~F;;1L;BB};ko zh;21RC0hxGnM=o;8`%x!vy1p4^B^`Q;ozUrHn=#nGe-zhT1j(?)iZ$R{zI5l?2S8VKN@djTc1BFdlN<*d6It)z zZvvF+5J>ucg_)@ll&*I4*D?`5$`MZ}m|B!jI9EFMf-UIZht%RSsjE zEosQJv-lab$#{tCdD74OF3{mkfA@1M&5axK9BugwY!N%?oeQBTVw_@J-258uyvCO) znhek6?(b=@+3}xl+7(_C4`0FK-@it(lyDKT^3gP7@^!;@Vqf~K zU521}5JHu`eAigsn6R9%^7a3r<1C|^aNNH=1SFJ@k`@GnQA#>eLSY*+V2lw1L;;az zbSte$juGNUZonw%5|K^?>6Y&9RG;tf{qui$zrD}7Kj%8LwG71KOCrDvkja z+dem+H7ngNcQyxKPq@~jUrQ7dS7ce1s|eOm#N#gM!`_!xUrn;uLewvUN0Sel1&nF(P zeeZ9Ii^MJkM+VT&RaA>Y5~b%O&D3cO=x|A)-W)2UZl2D~+jEk16WKpd&i8M9Nj2vX zHq~J83H@+tq|X2NT>QjfXVENW-N5h%>8FyPvtcmhxy#zkwf!!pTB`&uTd8F)F$kx* zn4+dtK}N53)KWy?q{}B8&Q%K^)sGXECX(aAr5!Fg0+SkdYFNDPVHtcfMwLO2?}eMD zeI2GoQ^r2zTX6;!dpWkkrSenS_v%;D6ZYG$Y%$Xv;;<#4ImErOr87TQHGG8{j=2?Q ztV7{+q0I}=x;-ZexB!)+&9?NVK=WTb!At##KWY$sf?qi`Q- zWL)-|xZOx#49j_xn%9J=M^X1oK^+V{m+3p*kc|_vr*S^Dh2SZx7_+$S1fpYg` z(_%@_yN9R9H0GrIS1~cumC331uDsE_%V_@IjL17>G2=VOjy!cckg{GE@pWgR+jj+Ov-m}Z$bKaR zi^8S7oD>YTA&&#LYKo*mb}`!vW0QlAi@tVI>a@D$7VOm-3(V=uSzM}P4m9d8Tx26y zD@$t%VGiySoL$#Iw>7KkZyk9L!&oIJX*2OuA78uo*{Wlt1DDtw4dyAn>=F?BG>3qH z;sF*kev9=R+tv>b_mLVjUBTuS+>jNUU_>Af?LG9m(-4#xM;ps;=<2>Yr$YNS?@#PF zkGkdu_R++IyEq&pIr59UGTOS6cwwr ziJ2tlMB|dMb}vsSN{q#}eTyZkbyI~PYVfi4l+l8Pr{VKAV9f!6q#A*<5AGSSo-j4X zulJLH?xrwqrv(UL&7F)7f5zR&Z*S8J(X|`c6>o8G0Naiz;_r;8({@UQEg99)eDta~ z@qe}9pXlUbZBO@|n-YXb9a;s@Qc=*`#Kw!luN4>`HKGF_AZw${LZm=$w*~R#G2tP7 zFSljKKKsz#`rhIo>aG^UW7Mpe_3KGBw6{rKYRQ~P|Fg#3<0m&^DQ9c}ys;_QKdbtD zy&vL3--H+4d9-;Dn&T#u-{xxTMd!hw>5UTSO)9R)o|#VNYur?O7*TwqtzL;2MZ2*% z{%n?`NU5rUFDF`bDB=HK^&s}WkV(nCLG{=!E@cMSitwIb^&40H@AI)w_iQ8zgY?ful#x1@-VS!e@ z0_XuW*rk^1^W(l^d7D0+I~M4gYSqmO=CYZhpbe^4ne?eOLJ*tmZHe*Nv` zi%*CiF>f)Q0^oueqn($VRAF#ylwQ1FlDu_sJF4|pola4w;zVt?II!oZX`lU;Kx5elHLzB z`@eDRaoG=5A8bJ2ia+^c_vzL8sYC^rchD&M4FeT|Pk>RS5(e#*E$G z_tQ#&c_}@$6O&{Pm5}JvN3?Th`b+^KLcWL7za1XZPmG&{F3U6e7w_@#i-MG#vsM$#zv zj(9PL)o))M6=F-@biQm%cX<4*(?`JV3XW{gV`@#fSyi{`Mo)IndEQ?3eTvH154IWm z0AlbbR>jDGP(0OdO6kVoWk_00?`H}jC8gdH2;}XlXwB-f5 zC3mettkO^~w1!6`x*(%C`1PQxA9&o^lUR*0+Q-@wt?Qi53a)_9(`KGAQlgOo&Br zivkB<=54qOOXN0JCF$7v;|CM)j*Co4%o5LERpm`In|Q(!tfVs`Bns4MAZm#2by4mUZ+!oi z2gfy>DQXhFEF@86(8&aY-wcfNI5zSanwH2pa}5&DraY%H>r^O^uov8EY7Qu5!dJ__ z1bUw!9=<$b>z44c2w?n(H`!9J*Mb?MZiA(;ln|jRs5gv=K2TeHHdm2wX>`cRwv=A<&XRR0r ztqm@*{5-mJbKw(HGIw2EPr19bSu}2xpT|9tYy?6MSaL4Mji3lm$`49L>2Q2Hp0~M! z8VVg5lYrotRU+%qXmy(YHHGO>{Zfp_$%$`w*IL(i>cX6=&pC4;n{I_r_U*6hk2aCo z_Ljqz$pto)N#ElYhgj4H6uU*{GXlmlKV_nlRkc=AJTN&@sL&L6P6X?17mwbg8Yb90 zZfppAi~pryJ|*AoTUKxBATW&PtDwzh|AfKGT7Lz+OU0qJ=0PA5R?G)T^CzvHMcpmNR^bGFo9qI>GpC7U8DHt}De2~2i%@!5Ut{8UN?wbv9=$vvGuls!}j z8niV3yq_>u%)nd?Wqy-?=b5flT%SsJ#q+ykL`s_~XnaFv?+lPuRFHmOOY~vj`$Y>) z5)?xyDB!l{CawN_1daMaUQBh1ec7?@{og~eh0)?!j>8Ez*Ahxe@1YF;@9gf09~PQ$75kDt(Zk5Q!2KMR-c|$b|+pr+GF9WQp{Oev7!dAl?vBG zuc}zD6;}LI*GZBu9I-VTMrDsdF~Kw87^qR-+e1IUUrYv5sAywJJBP7LO)3p~6p|$j zkIh&z!5HOJ33E(4ggOn)$96oUC>Ft5g{gU2*^kC^!iZZCr_f}pc+;#p3>h-$WTw9( z2rbM;G6+w#ceH1v zu;t-C*eUo2B35oe?KGNCls!{F}t7>})DC`(dI zW{}>OYaUkT)Sc4z@o%#|GfFHW$}NWqrwDmZ`vh}3-ME?$=Y;~c{_z5qc=4yArCUI( z*$fo>B}4Z$M%%AA)Z@U&G$bs7!TE)IO*~WOZ{63dwK7`a8JEYxECQr;!7;T=8B43^ zZswZN;ZzycbVsskCthKPTjdQBaa?y9p40l7%+djEvMG-E44XR~AguM3Q+knsVxQn< zJnrr@@OJ&We$^ZRr1?nhqd{K@shtE6t~-(wKukNP*gh43@GXnnwi;F%JDafn!gm7Y z5@Y{~BnGv~wK{!}Sa(t^Q3LoH#$LX3cd!NxF-EFU(y&tuYgoX+5*P*n)GIA_b+fU6 z%7XMQ=-I1n?6HLq`NuS!4M&X=ZlD4cd9t}XZ*6x#;T;ozue`#z;?x6bW42YnO_YNo zRVCi6kw$Vsu57D7NQ^TS>wU!fkyXdMeZEJw$OGz8&QVOiA7c5evI7+YEBj5>FH0L3 z9U4jVc8O&*_pB(*-d3FZHj>WrebrhM2LIee@hd-(LjDtM0kpkjRz@i$CuK?o;alV4 z-isOfn}9c~t8tHxw?8ScOH}M%O+$hIyM8o|aORWg40R^mdlr#R^XsE^gAl(OV;N+( z|75uyN%(L z%D{)xv!@NSMRboKWaHknFbMlScPM1Pu=T7pQ2&#(F#p)9;$d_OW9b;Af9m_4pQdan z(Hg7{&m2Cw_uQYA0zQ5A2;)aZzc!*^*xOMOPOiE*h5lLQ_BIE)2?EUc;_ z$koSMXZ3S{ z)f#J(YeIIC`5UZ5NjeXg*N@=$U;IdRWCyYz?-GUXSody)y75wM2s8%#)Z{Qxx!%+p zJld}8s5t?1*t>*36sSkpO{Q1G;9h@pXq5w2jUGREVaaWz1dO1GZ21S!ynPQFrXJOK z_GB>AuPXEI4XS!MJHbioJIPT}ec&&wLA4dCZ9EFxAHKK>BLml5U8QMQXH?@|hQNqV z1iK7ua$sKO&Y!zo-O@Y>d$SM07r>oepz#!oKr=0Pk6H1{W!qlqO0;mwbzIHZ|9f1hTP2}(7OBE!=1p@ zdg~lYyO@do1@Wf0*RRW7eY>v`$eRZJKF4pdvL1()*mz3|V!4h&q*N(kWBtnv^PQtA zszr4;7f25!)B((=!T}Z)-n~A%CEQPstvhSayq=UuHc<~r9aQZ!U0(x5AKFGq#IPYY zEZYQc2PNd!`56@)GO_fU%FOBS`q{)PWegUvH4oF9@D&&8L!6THp$AQW@5v$K>r{Y$ z$5tE!=8$jc8dy;Wqm@&O(2?9fhhn>8Uf)aY3*wU3p%^8wfJ%bW zrsJFB5l@eMcg8kTC7vcEz?I_=%E8dk1Et)Bi*Ss)z7P8@tM#!L>(_kh9Hl~_29mh! zC&JM6bSQc9hwU|nci~eXW@vKMJ^`!_(;qggj4e;KFX=M(?h|JpVpp6-=5uM>^qH=x z<~~}BuCO)Xh|5O^Ncb<*kitow+|~SwG=Z_*?bBcX?qWs-yLm0KZ*<4P&mwW zr9MnaM7Qk(+l?6uifY=wUOq!^DY~=h$G!(~M|z}#LRRFw&?Qh{zgx*cBE)y{IN0`4 zZ<^+o%V9Pl*<0N%(_U!`Wn<2`41qZ>%c-UV|vuW)zS`pb;~J?jlWg>-U+Zv+M#DX(J9I$rf#0HbIQ!{ zw_CAYm$N2%Nk28u7NfgAw?`6l6_F_;sS;O3qKeV5 z>n+WUxKF((7ZHVgT4)9SI}b=ck6R6qEC%FMqvcye&u){xF;JmPIpI5f@LcMj^0b(y zu1!HmC`mB%orf@05#LqIg#)w2@LPKC_V#DZ9+;ms){<&2W|{?V z*{%NpvrH)FiQ{fbrC_jo_*(A!bBArGJ#wB@|@|uPm^- zeRjB_t0EY4@N@3d_4(cctJ%53Q`g8(&TCMVz6lbkMh@BIEk|9d$K=-yZRafm%Qsx! z0jN}}eaFnAk6+ZCKB|dl)C=C=YNmSS=c}Ip2C}RCEu373HeX*hp0jlZJ1|o^2tOak zl%Hk;ukDLDjR1h`YC`10crhzuy$9~=~+O<$_H?) zg1v_JV(D_Nv`Vhag__8nCX9BheP49ItRBGb$(!Q7x>CrLF~<7hB0A5 zlZtB-Fjh-5&E2EC)sajQv*twi2W!IuC}yNN8Fg4CP0rXiJYo%rJxbaF<%t-Eb5-D+ z2g-`dHnq_Qq^I){e0R!k3E;I$i;32pp)jq#sIfv_rH%}FNl7|H@v6!kOYQ}{L^{>52EnrjN zaZlhw}pkD?b0di6H+X{m4z#_c?sg~Ezxa!S65KD7l#MEFmpwB zua$uM(FYNzs9JmGyG(|M4(TPCXZ3F$5v}a>^}LM|&;J1iYLaV~0JdhRgZ}2rN(d*M z%?D!N@`LH^JujD$s!%QcML$f;)t2R23&-q>n6%dWyv{w+zZM0mFtjOIm_R@GwcpPa zQNBWtAB=rW46h?mNi;5c<)-xPK_<>ZrJos(bLF6xzQV10pfBQv*Sq`?S#e9hwJ)af z=%E_?>*!~&I(gM5uOpmC^s7g3M1j56_a|s^LbVF}@mtZ>S6z`QD{loi>7jt zHU(IbbN-TxI)N{AYfFk6^>dQh{&p@hq_YIS(o230@}B={(NRr^E77K%PTMPb9Kqv8 z0{c)EkFXRkKcv{(ZTXU(dz}Bltv4gs^Y_sB-(Ts`766*0p8W6=zbqKVx1eDfie(XelC%szXGLWiJ; zkgJXw~K`>14R0k~FQV5@P)8gn9QLVlwrdw#aDCM#hvx#nuAzq(z| zFUhSCVBkP$k> zm_yhJ)t>g>406UykCnZTqH-sM&6!9^!h>78F&~qEHTFf@^9i$0Wn)Ha%>~bg#)R1x zyxgJi92k?ru*C26AIjq1pZD9CeWHBiW1Fewx?ZdIMMKsK> zFfq=R?rr!%kzWKy_T&E4z}#U018Jq&r0m{@4f8)!Qw`-#msuO~QE)^>WUJ0(55KQ) zU`8D-)k>=(a`2tM}H?BuCm`cpm*(`8${LU6;T41d49Ta)Xe6cW$F*xoeAnWDyfFS=qg zlEpv&?mNf?&f0pLR0wh(nR@h!@`CgLOBOF5?O;OH7cfLmUvA^_!6-J_OMNYgoV8bk`ZSGGYR&Rni zKnM5M9NZm?HAS3t9~M(p-H7ZNT@6$siPuccF(QcB)`?n^e1E5}ueLs?K#NyP9!S3c zJ^NU_(l!1-lX8WUhn9l}qrezQRX6li5Ebz$0wpmtfTS@tBl4CV4s`I5;hbSVYa)ap zp;NR8a0Cr&n**& zZaK?VEPGeq`LP7B`|c{bL77)zTj6HI^^~CFu$Z&2plyaI5v1QPdTR+>IFUbhH1EVL zejc3wH+Ro01#S7oEu;(j3wko*`Mf@7yzQ8tn+3muCQK8@t)J;Y476@snXU$u;}ZY+ ztO8JZam z>LQJGKV(yxI?I??s7-|Re7jP8;$_TiNuwpYTRJwZEQw9$Jqn$GmOYk{*RC2Y8?UU1 z9xuE#KC@F=n6(He2vCI7Og3ML*?uFmAW}{I=r&C>JSjr8+HH35*N;RsW_;4c=!s>u zSmpwJHCe5jaKiqYcJ{aXuqf^~j08|XYnczlWzxXFq=irhvig{1I>R01tP_l)_05ui z0tBeI%Yx0%1H@kKQonSFiy$_G`CmeYlA5|^YvZ21L}Rawnc~Yul(Mr^!5lqvix=0) z+cv!n*~vn*R`S$tBz3E0D{CrO@x%PX}u%@{Omk7S4?5jG41lbpY;>)P)P?dEhgq}o@>AJ%MG~k<6kVqdR84M{Q^;t zi7Le4Vk^qf5g}F0`P6h_8YP{s7-g37(HTQ8ErY*U^O%K}dVdd}a#NwTA#4q0^S6^| zTWhbZ`+_%r8!JSBEOwmHcJJFnLirv zEzkyw=`V5RRAnUzwPBvb73SYr$N!1?(!%9SobQx{S=HUFB;dGrP$`?#rkWo{9&@O^ zU4UrJ3ai-jTscGXlsAp1MUh#LN(P!5S6)}CR`+xbynVNxLTuC^2GVn1Eb0i-6DuW=G=4*2zG%6`(ZN&^(?i~T`+5<(Po@1$7>$!s9d~%Ge6Ko5 zS{o7JM{vneZtlX{FJ(|}D1~us?IA}>AE-&^bZ?nar7XK-p1k|%QHELd*m9spqB5jM zRzN9j{!r{qqj!m!%rhd1Y}S&bTlNaHv*4kupj>PHBhRL|m45(JRBC(Ub3(RQ&=6w+ zHa76xZvLx_imrBRFPpyMg!E#xh-r8`{yr7ETbxNBIspoL9SJxjBWz2QPX^mJBR7e z7lAlWtP*%~H<~nh=rC^SC=|W?xMy!VSu}ia;)mT^4bzFgqt=Q40IIUF)s$I4L94OO zq4pAN5&}1yI8QfyP{tw_eRz^(5q~3^f_r%2;3@ny>qzP8d4qbY2 z?`=%R()NG*6n$Q}l>{OcS*{L_Zcz{&IQlmMfvtF7dy{WkP+I0`2)nRzht!{1MXv?d z(Hbe-rXK(IPS#*f*J|+5c{2Njg!Zir{bxGBNkiYC8~TZ)<^wcO3!m7lz?)J? zEbxZ4`d&>}Lv>*fXUCv($$t2!RO3Y8no_rjfku$pogLxMbP1-0C>@(qguB>cU6VX$ zGY&D9*kC#B0z3N^TswOo>RUFQtqlR{3s6yn#U9NB*2qcLqM4W}P!Af9U&~Z(7p$oE z8yOGWo^-ShmZ6KELkx@6}E4-uZ^71E_Sq%yQd*d(fnXx4!%Rx!Ee64 z+bqKDBcU!T{;J{gALZn~#vIG?y;!hVEVifY>P3dk8-8C>Mbx0VX&72WO?Gv_V=@aXAmwVu^$>;PTzd=I^wXX{UiE(@(6(ahY)C-Vmou28P#6 zj-$lm$QXP++jr){FUJC`?6pt_y}Wjf+Omj;p8&q;MqZUas>OlJ#7;zT=aHR=ac$iN z{B=sgV~#K0coU;&9M7<-Y7=iBuM6H}vh;a(Z5Vb%4W9M`bXb{fa)PAm1=#bEZ4z(m zps6@Ul*%ofCkS3O<)b)8N<{|^_>D3>FX>95pcTH0vHeqb#A1-mDARLH*)AgDHY4lr zM?f2;wqKo0OxVMM!`A|wT*yF}qbz2<_aL6xs7|<^|M!ZwyLHPL4mjK+n_}IIVPXpk zevK(v-w}8jwg{xCFI0M*p!_EL*T#$E$OKUMd<;dx6GCC!l3?I-8z0&F z?}Fo-;Q$JJT)+Vs_r}&DWGx97_YZ)14^uG85lngv=I`7mn+U6Wver;DCGKj8)BKtv zqEw&nb=i2mVH?+5RJ>eM724*z^(ZrT1?i;o>b^F(RUZJjZO!UwRIU8{HGe+#Rkc}K zZ3TNwvqr!*kt0kpe7ZKN7t#NRwf4X>@lZP-@`pPAUCxa~K@VGm%+-YmG4xE1dChMg%E{U5;5A|eBg06BG#WNOwl9kOJY8HC7O$_;Eqs-gFvfN zwU6-=PY-sPA)fak{{W33nWc-_sddg%yZVb0--6!m|0xgf|Dino0rY&8X=94O=NhVi zc?z@V{30OK=Iu-|SeHz?qrIZJ{FRfyn-M$JtqL@B? z=4et>*j<6P=lxNFw0Y!|{mZ*@BsC}OI?~;5JM$@E`d?Up{4Vjid|(bHP*#2{${k%D3w7G&}rF2z zi~o4yq-M;5gOGM$z7{!jpeo_P&*2xOl82 z_d9^#vafaH7%3_5smXJz>6Q)$xQ-|#jo@hw^3RoXI>cj9q-Jf9Z6Pqauh3aC{K3>> zRam!;!uCO_EKCZE&nbPFC{ZQUoVOwoYeEIxrxvbTO1UYu^b|&d7@IFg#6p355BU|k zIacDmv~1A2B`_@4Uq)DINBK+PRN3UcKIN@It{Ivmvkz47$3D<6^V8JWJ40KPl0_16 zzQej7PUV=semrOLOkrHwt-H6`dXr{L`jK?NA4H;P4XIW~U+9Jru{|<{6-RgQS&3m znmR4OhVH3v?1kTWscf}i1S<2EAU4+77G-_<_luX-mqs(Eqss)rx@v+1N|1rBg=6=k z;A2OOPSq1Mc4RD(qg|eu-5xj8S~V|3#GUFzc}adQ_emN-Th4Cr@t)eSFTpN; z^i?7+6J43ju-rVDD1Kn=kz>mB7V1fQsT)C&gs+{m?i0wwO9{U(p@-Q1E?lpD0R$T7 zmGJGtrO5#v15#!=2b5JUc(<^dFN4LbB%o_MjDV%i{6qrcYkzx+|EsFL z&}ZM;Yx9v=DV_KWEAFe1pU#mz^x5`dqrmKU#aUsA?>um4Z~L=i)ktS;g)?e-b5_WA$e?3)if-*tXd#G=2`t<%1mlDF%MYNE zP073#<^6@Sa!ogQ;T3m6wd91A=uOuwhEA0@S%q{*bTkmwVI|Zl#-qZwoa!oWdc>?4 zMZxnGJ7cmS=NtNNOA6#vFM3uCj1J_8x;1MpOxV!HP>o**_ebuKfFC|XTh9)8TDOG4 zn-luVr{H5*weXyTEG3Yi=8}k$b}VYLJG=ZJ0IX9=#e;<2d}m?C^jI;DHtxC}p-oWc zm1$s^kB4~hEG-o#{qdg}LKv`y)9te%5I6mJE-s5)xj@GmLU@r1h}TMq8FjPJ)5Q9L zAnSJJktxfzbc*2JoO%FE2;><|cz*qEai0N0%&~OqVD;`tR$i&V5CSu9Ufohd-#~A3CIG)j& zJGv7?*I~ZoL0gcUv+6&?qsvD;V?xhl>&=kc#XgDQdzO)NtX8p90RO9~kIBi{_3f%j z-&q;AbdDThvVfy{kWx4U*G9?d;H^($?Z$4onQ;mt^t+~{2YaMvZ?r4@e3reRFj(#+ z7Po3{#H^Oxn6{xHQY_JIuVS_Euj~3OQV<}z`}2Jtf~uLI7Qa7A^M zMQc+K=Q^7{EJ5EBA`w|;j&gKv#MxkRfq6yl&QoiX202(}%;>}JU;K}*MEq(qfo2*} zlMKFxavxKUV*H=u$U;tSky{i#OQ!d=5<;UsYb6Vwxx8f+<0I?4cbMnHHe{RFHjHJ# z`SFt`qICu#@a1fJ2NAyxMsHQWgR`+y4AZ-morEc;pd3VH>=nuEy_h2(qj^`i|T(oR6P>m*_Ai^G3#>B1fps`^_@4C(PB| zqmBqypO57w=KH|mh}v)3?1HTMR_N2#&yT4ntioXCGU2~DL}A$lOi499tlm7qV-I)* z{{a?6AHK8`Oh~}6##jm22{Pp3?|)YqHGKfW)3ni&L)OaoK_#skV8FIA_1wA~_r?w1 zS0e(l#gZDeoapelfXf3E7NttVWww~-YJTTUWyy<7XDGUw@TU^x<-q$R|IZ6WwYy=f zqNF-=A=EA;m98}k#oP8>U{Tb$EfB)AKcrtKpwqb{vi(Ii;~&8INk)_Fz1D!6Yq~xt z*VfA<`$M7f-KT>5ZCLz!`*pibO=UW>{tP>`q?ltwJu)GbJ}C z`ljo)A|dspF^rysa+5hCSF2JrYiEj{PgpzYrf8BHSEo5s7N_i*&qjqvW>cwfI z8B~2(pU>%OdkX?>>RwFHb&^d7e6cP@+F=Z&8E;x5psDFPhV(D;lqja zxmV)>m??=Io5;yO1#Wwm)!sp!m-!>SRGdSrD{61KG6$bCFE_A@S+UOBLb(4Vm)=eW z$E6}<6TJ16H)%iid2A||ik4d1EDh&BwePZiRFAC1Fcu7)Ugpllc)$X>YO1D-r6t95 zVoPNo{6=yzSq>Ke9Tiv-o1l|Y|JD;5E@Y=IM{kwM*{l4!21&TVeD#9Xt%o~=%N#a2n0 zN3p=?(J$evy&^QuDbw1vB6=#mR9rYgCd7uqx>59!?4!-AbCwEz*7`);Jqogz`73*# zR*+^dOqNN3n*ozIV`NpgA3m-_=nWl?sBX-x(kItfRm0iUg{8+IkU2opGFh7m5{fKv zKi*gMiY^RrSx`=URZf-7$J{;*3dV|?MBAsATXt+e8E4>nf^-$W0 zI_)nt*GRX$p(DuC%i#N0agvG1Ci)PU_rqqT?*6rG^cyXr3cTtd!rhq1f?8CsG(1)e z9TPbZ`)8{lRjKcm?1F@bLe=;4Nid2oec0oIMRxnB1P%zV=ls$XYA8qTeqCb5^VqqG zWT`d%R@$N7Tm1l;GCudywMqRe|BSo#MTy+nCao+JZV#blWHU~0S)cD3bEI^|)alTm z?cproo|eu>D#<*xUEWU*#xb>QIa`JOKKF)=+=qQVZ~L-&uGL=j<3H3VQpuhs6EagdKpLTM{05ggx&k zh&l?AUGk|KNowJq5TZ}Nq+PV%*}t!x7?ULcgv~E~?Z*(Sg*2fEye5X}X3~I{?@|8r z6`6OZG=rVmP>{*o^_6{gOUh@E4T(>LDb$C3S>Y_bk9k*YxT;hnJmJWOsJ7&ch0RW? zP9f8li}i?uyDH)vOOLv{*XW9r-&3N}z*1j(;I&OP&@3n+In&^>-4wWzP( z-YZsg5y4_$@#z2PPc;8yAPFpuLo8|`J{2=62N?A!2Z#m)$^dO0?IqWp?P;3zm0SJ+ zK&1tj+1MdPWida+&&+9J>{J5Ac=s-FymnG6+?b;*ayns(HJosxq<^9~4RbArH_0C< zOfDmiJsB-vU)zxQ2k?d8Clpvicv-$jfnPA9Fb4ks->Ux{Sh{Nu>6ldP*Y*2-h-c6i z`;kmR6jR#uSvhn+B82qlc#Z#>;#8p3GYf^Ob-71H_4PHDCX5&eb%=_&fR+jsB!mT* z_YOS{OW?*l*oX(eG2u)&K|vRAGr6i)@r8|(Gc(ISfc;oVXY$L+jK8S!jLRZLJGX&i7>+zg&ykg1aOIkgaGNwlaHLKf~aibP` zvU6>jJAc|u<`ZXYVSB@lh8+LT2L96oFP5HvMDR>9sgB;2a3ravIV56h5JyhhG0>zPX zcfKke)O>zxlVxgV`2PKOke1c?YtPvsCJk2-zX+M`TIF2*7x8gZwm7Oq$RNtLL%tw9 zd)Wa(*&!VKr!vzP{tdX(XUU8Cd-60S4@Ix08huIz`{UrfleswTz=)l4CEc!@lu9Ta zYOM?n1RNA>ar^nBPQ%~VG=HE}EWhXZCm8~oYxY^ixHJP`tj7tM&&ND z24M!l`reaLijZc|OS^kQN}h!;%wD~#@!S*zxl*YaIcR*VUj5G7>cDG#Ko(#SH{Xa`Rd%>9Y_7V?r4+91Um=R5y<)&cv3BAKTR| z#t4d`c0w7IDsBR03M9yjf(PEs&7$U&B7O$TUF)}^BMPGs@*iKk_HT44+6 zO{{Kl&Aa(0cPadaY_o59jS`=kB^T`U$yhN7cwnkj!BOoAcdg20akZ+RqN$O08LmQ}|-7lkfd?e;#zoIgtRn;el~Ph0_T^hi+8j4kyOyT}TPi zRr1um=9U^kOL$#vnP0@8@Q-!b?IgZU?hkr3RV=KVt+3`I(DaVhlYvS*xxNKZaaoj?${f`Sf zzsbqE`rHD$Id_Y_O}+lB_@YhtaU?o3M(Mgkt_#T;szu%MQ^~|2XFHr^j(yB^7Q1=< zm>B;d{)2r4XrjLz1PxUkxRfBt`fT2V(-X zMsK3CnvT*BBzVd^EOmpqw2B&#!L5GG;z{UQe#Mwh>c65DL9Lk$pszW$odI+NA^s{7 z$9ek=^(P{Q5eZs1{SZa(9$emI(Lm-fZ8jwPZB7K%6&CDw8#I3LhA~1?HD+${jOreT zHb->~eO~?qWO%%m7{TBx=S;GSy}A}{*oXx96&M;|NwkLkQICS9-yRtahNeVbOY$#l z0(01+utT*yti|C`*eiyc*e@{{>Ciy*Ob{ZGlxu7#eshM4@=sw?w0>TToH@zHKU3$n zSrNtJ?st~w=Qn?{9rhjOH6{xs+lyBxVzk;beAf&mkONpk-0oAS;o;J6`-eIy%jH4ONAjV9SvE_U$T{t%FZoajv#wrrJ)-0> z$JYr5e|V;9qR=lrw|=HfBk;L)OsZYq|%z=hgmB2)A*Dunjkf{7cfG4!aik= zO7CXfF7$>X^O%>v8`0(Fjc=oq`nsl>lKZoj`ZkA;8C6t52{16VB5I_Vm4Uq0%jNZ^ zg;2dD;bDh1gm7twv zUp;n{UMbSgBcJWO_xBM`;Q4k#&;MiZt)k+Hf_BkC5+u001RV$ve9#~XFar!QxF!S$ z!5JI^gy1g2;O;KL-5E3_xVyUq3ldKL|D1L2!@2)i>&nA@I8VE4ZRu~QzVQ;0y_?;zwizAa;gu9hh5Ets8cqv0nddw?H8@H z@SEds=}gCwJPGME3Rhu%@K-9VEk(V zDe=1*t6Qk>S14sBs9gXs#{KM3P_b+7+X5#$?Brl|jGL&8}@px-v z9DUO$5cSW=oFKn9-QaTWUvhgnZ_d7L(+G|}<0&a3Y3H6&v_yN~ue8_J;p*RbBZvBC^SfoS zC8Di`SuO2yqR-6?F^>EWP6i9-X14mGH=9O6oNO2?i?Y&_FJykTU46OeKKNGj>0~Nq z{khTxBb;ivZAt^6C2V@EYkwwfl%Y*SFOGS-0B5UNgI0^7G+gi>t#WxYC_3#0iNI|f zJ&a!Kn#UQt9@LfTM#RhOU+=Qiv8WWDXX&~MfFemE*Q_UG#hufm9(;_a9{U1RYs)ZC z@^5ZEe_v0^r%9)yzuu!I!an93`1`4b<&tZ>Rc?%4Zn)Pko0W-3MlvL2wRQet^5A<> zMHi0FmY?9k$scKhPt|@T#%BrwI2@Y{zi%6h1jw8gTrJ;FzOj{Pc(W^v8UbpaBzDRP zeA|H1gz^Wr?JxeJ>`2`g-z5k=j_%nU!O`nd9)V)H@4DOM9j+?Bf?YEr&w;=V3(ljS zaLydR|KcZevj+$l{_*%Ka^9Z{Z(tb^JpBD&`@*uL zrS~A_^PBOq$V}zrt~V2v@3r)}{EqZhM1}pbwN|u(=vg<-iPOz*Gqsb5s&ee8i^1Rv zuGn-wFX#2b&TGURfJ2b_% zfU7puWHH{i@c2u;?^idUAsy3yeGRm=xy4xKD_j+H_eR8*@#Bc{q+{YWRYV4q#ayL*k2%YwOzr$$|kd5jmIg=R+vb z$A$r~3GZf2_+cgQUT^5{(2|&;_ z{R-+{yN`*i#{RKU=?67<6;<(eBydCgd?cb|=>5*5hZ_aAx&7(6{G`eyxVNb@kf zH%xDili=sD?A5Vhh+fk$jwv=xupilT2=sPwxF{44!2%A!*_kd_8Zq^fp|R6 z6jL~wzBNeCo$8$vw~M{OSY4-TI46!UWPb}7aHU!Wk+-okvAl>?^mUn;NPfp;?Q#0k zA<6pU;$E%l=QQ^4Khna@bKcPAwJ(Rqp8i5N9|Yh4Phcgvc}|wJbYGe`?tRt?SepFI zA^pL8*4AnNf@)at7-~cEZOzJih~HeABKhz$njcaA>5kj39-GFuo`5aOSiL*$m_EcA z2__+i$zB2QK&bpmjZwH3cj8UZ_Grrv@Z0#OX! zm0sFuEEaJT*zw)-AxkD<<7qdD4fMl7Imc%j(je5hrz-8Z;wF;|%s+g{Gz=J#)(-!Z zO6YsZpipLzCtvex;sVxyNWj}VJ;~qm@4~9)U$au~3S>zfroBJxa~iv!PJ$ayr!$LL zf5V1|$jK?jRy2VYC2yp96S>iyWXsdwGRW8jrnNOc1t0A(ck_ulm|~$sANiiRH*{)N zxXy|LCUtpSAuCe^QQCs>oABIB5g2Pb34oWj4WRJv_@2}9iD9o|WkBj!$t7`PI`-BR}Gpi!$jP zD{W*pVOr^=>zQ|-b_9Tv!J(Aw7X-q1n=@Q(|B*%1hWShseG^6gR(a>98AHxiN6s+g z>2A4Kn)~I@XO|Ifkhl?!^d7qs+bn^;oT6p^O!x{9W4>#(^DC}jTjkJ=?sxEKZyfsO z1e}Ng-g57&1}b{Jqr2Y*m8@a!-uJO&Wmuw_*Wuv&M3mL?_|>5+O0`zwfBM!SARwlL zBZ0Te(0Nu~oxWRtq)~dL7nNszle$X2U2{OEFf8Yf@Wx)!K_ZdPx(zkabMUU%+}vEi z^s>Z>x1RundKiN|S;A*6IU zI6zV!g6AIA?z!rycB4wC>nccZ4Frs4DjVMD)W$7gOFnAj4Dh5f1I42++=04ik-OU5 z;=bc6n}=sYe|prY+GL-hs^u~u1O$EDX1{0SlX=><+z|pnWW|ic;9P+v%YevyW|=MG zCF=oILDJp=?*2LCWuBdpdusatv@TtgnPnvI#_oumvA&&Lt)_%i`J61Ln30gtLO(0x zH1TzkG;3nxU*Hn}L;Gzqq2Z4B0zniv(9qmMtaws)GF2*-s!U=Z7y7LxSM8F0fcnkL)`k|vGj zId~I!Z*AMorZ?atdRCFhfj{5hL6W#OrL$RS2AlGRFE2ggas9w8>|c(|Utl5gv{f)k z`PP_Ub*_n^nO40H!q*hM&r#ed5(0^=)N1`s$t7&zjP6pO$f+0W4J43(Isu!*u&4d4 zR>)>1k17l0bXgidsQ~xfG4a+W%Pr*=Fc}VhheyEj$cr{LF=ovd%MonivAM$nSjP#2 zgjQXZ_E!GRN*_1x``Dnwx@cIKV2L0QU8`?IPH{3B!v=)^RSEeVEcD2`t-efWp27vf6Z;_(xFps zi83w$GZ@_Kadt+<`o&rssL1W$1-+?OC{sZpzi>#=CF zr?G{uP1|{SJq$hphz%=ZZ=(_nwsQ@ZvNC8dG>K{7P2nUVS}LSf7M3BUvKAYcVcDwDfo!A6x130ey9 zcA>a1NhkQx522pW%YuRi0}Yf23UGC+Is6K-@QACS@WmgkLJvaVJUSC>!W~%@lD)5B zyi;!cAp9mIYa@bNWI(uczw`}RR6G6A=}Fng=CYjIc)fj+!J@4Y;@Kwt>CLZ-?w!ID z^+#$tzJ?KnC1XQN|8b%!`M;p=s|V5QiVBsS2te`x?^l{;*`iz?8^e~Kt1|io|15_# zMxL?~=~Abhama2LHd68#my+hvZQy@^btAdv$Q7ndZgu#_` z-(rVxBY37~B4^LeoZvX@PiS$vMB2Fhkf&cz%K;*I0y@vNy&UyJ-}LPFtIDN9C)l>Gy~y{+C`R8YPkTov49=x zn+eQ|>F{hNx=&ZKzn|13@JtujGnE5KDlEjPk1fScD+;$Z(x3GlMo=w_$4C7h5j*R5 z22Pc$Sn|p(7&^#X$@bxuKqCn}SU0D?mmin6AoH2&NUB!I97UNo{UApMjT=0l1ZzFMtbJt)*LKKt8IU7CcRCVkcXD6 z&8+EGR1C|1G~UsDZrCR8%z=1T$9|s<^hb{t+4AoH8~|iM=o=*lek=ameVcvcIfZtg zn2hW3q2;+2Iv5AjJ=5eMO0Io0#*sG{GI+>e0nueSW?&SigW4L8butMBQl_X+=bMUk zq2^b8O|;^|Dh2kH?HlU+a~=I(JIAkXUVO~KUtRKyg{ford^;r1XeRBftgRhOyxVT8 zuRu*@!@iZ(Rxdg?`Z#lPSj9f1L}6CoA1K7i%-KBw_;90Lf-9?{PKI>Ie1N~yV(he5 z9_|;&XP9y-XXk3x2PGf8C9QNBD*EAe-3}y1eC5S|4#diD>#g5o%&pn{QZEX} z?MVEYh6@jNuNSj=eEcpWZX?*l@LjJAy?|W~$V7Meymp3rUf}VhUE2+|c}HHLIsefAG2%wAgO(*Oq5L3CGCw09}j z`P}bqlX6OzKV$+w9tB4Ux_1xBa(v1$!neZuoNTD6`W+_X zm1w43%ay>4K-=vyb?V3h7OV_0$0e6eU%yqSELoO8x~y1Dn2vTNSqxpc_gZ!VoYjKK zEtV`N-~slAUe4Esjb7g#6QJmT#Q>^+3b|SzbKW4p0xGZQeFFtSXI>HBv^cAt7iu98{00m6lNpcq$l+r?J=_XllE1R_Kz z;~OAg#iN+?n*zG3u{Q)4}$C=PWjO^(~wu?l!gzrccAEsvjjF&4qjF@GaM$JM= z>3#8u)KdGj%cpf7u~?*5LdqM0hC{Gt0(y{Ne%uQFXIDk_;sG>cvMq~6V;!wktMZ%Z z^P}D4k{4w?!K$t)OSbk6d@mAWE)O2GN}$NQ{RWcDQiUhL)o}UD8HaOtVNvEtN#)*m z0qB=UWUgcpAsdZ@ZGpU(dN*K7^E-RcvU8;fY)$G9VC!RwJ^XFr#7IQ1l1@D#Bd+Hm zmX4cbsYD-1#Or0I6US$tdk>|QOE_v++!YZ=HNo9@_0?qSrtmTet9G!d06UMthXK8# z1mETN{fxkTolxkgPXp}DTRmK9YN6a$$rm1BW6fL>BipZ@d7G5eVQx;J{;*R{IZ#M# zfEm)g+~=-r7ubx?fO>65l6wFMnb)2vaquo$To)7{d4h*oQ4} zD4BUU>gEkS)USx#3Vk{1&sjZ{Q`ku>6?bP?>PRH-`wgAQF2>w`; z*xvDRC(oT8m2Ui$2w%6>IaQ3~5e$xvYy7D?xswBgT{t%YbB%}U5a{R2^MwPv1^sK* zN?5BppAfCOmfekjprL{^Pss277(|GrG?0-HV@*mX3S&n`Ir3TC}G1JRVrifu~0G z@V&c7$xmunD_U(^C9h4u(!aP{ii;t*ikjcMmG&w(5}rllwjW^7L|S1!)JHs$8D%&& zbr5$3G`OI?w$2-$zEiA`+r{qQn>Z$7s0uhOV!<-f9U6FpyDXSjR)=b7ycM|6Eav{~ zkiX>>?m$@biLl?ix)R!vPXBS z7!UoWRHoizRgj05R0sLtd)(D2HOcPxTe?pFKzy>qFY@Cnu@%as+EAanmnZMZatl|5 z?~R>;0H=6xsxFMrHmF|4y2|6+41`!pIzwi}iD?w0f#?N92w~dVj1u%qx12Z7-^<)s zKvY%OvuUWY?3F!4KGd&FY1YUKg5Hfq@!5Q+16hW4v3U18-uto%apa_}x$AbNE2$H> zP`wzqvDxDxJNh*`_Wb?Z#)L?|5Kt_h)7y9XU!4+V_kM?e+=tkvtH3gbeQ>_qyI7Jh z*J=J+6XK$PCL{@Y&+KIJhj^UTp$a;9_k-RI*-v;SRlB(V#h?LGOMd3~mYjR#W|dHQ zX#=@-u+EQUI@{`mFK;Uyan?Hc|Khywod4;o`sFGKB}TO_adR)WI#l^M!8xh#TNYCO z)9{#HO~V_CK}U|FS`Fe`J=*=sUZeK>JUX3$wA1PF#Qnmiqnfj;Xj=U$`MuIXk$1#~ zqv(tS)y3Up=hxbdht#t(OR6UT=IMX-Y&|9Wn0KWm^-1lR=ly>SdpEj?w|fYE{5$q< zpz&^_53)~yd5N1hEC1Qp>oSW-?Hl-*GFt?_rKn~nU#-uMK7{zFgP?QJ%b;A6GgTKgaEPcVIFTJOns zIGc%{yBNs*9$8-5M{K?s{d`|-LZqOst;p7IU|QL*i(MbdxH)i_0UlZihoSL*_ukr zJ9)8j&08h?1i)EM#xXCDvS^+PK!{mLYp>k@(M>KM!OgF5QYM*fk5*74IA_Ls0$B2P zJplxqvo39BZk@x_co|#QZhIVkxeg&j=$oQ;-SUA|%Ptm)Q__U)bpZ(Mqtk#akE-nYHH@BSJgykQDw0)} zT_+HC)GxUZtaA$4!8gyP{s^0**2lQwi5ru8DeHjXAxO43wsg6!%6bNhKvHefxb}OG#umR@y6HgYJd9O zO}IxjhqNo4JJe6V|6bw1I-jmNNHSa}VA8Pgp0x7_BHX(P&*BTdO)6II4R(0TiO2Da zemZ_RN_hT*#y8C)N*F=KA3F1BiVbANk>m32vmV7=O7|y#TRR=FpW_LzKQeI5Asv1+ z`fu4J1?+FwXnt6kMHBb5!NeT5*S`KG&3eaa8GM_~X&GYw&sNg!xU0?;fLq@W+=m(E z6Isss|EQ2`hwl-HdAd10DM;HMW&urJZ@ZR4SEe$66T`4kID zv*2;iO|S<`*vGcT%CpCDaUFzgXvgM31q?*+n>O;3KnFuB7qfTMVnq)e%&9RYia~41 zFnkoLn_M;%RK217a%JPB9YUm$h{G zvz59s=vp_qa)-+UwS18me)}`erf%Ne$kg$8K`6fUTQc{=ZVFWFu6au2J=%F1!M+;q zdt>k_uu<~s#q2LzjquzG>?x7)O>LkZnfK-0z^2OPp<*auR(F-Js^?Was_VFiPDV;+ z_MZa=e~i@m${eHlegTqLTNso1USHCo>?*GhH1<`~Ju5}jmFaVQ*`JnX`-_>NL zwpNjSEW%^`IN40?!(2S6Pth?M9is4KaOl&IzH#k`q2u=bDl-?;FV2_ovkgcww)-!6 z+g;>y$<+{SSMhmZ^cIl31U+HD8XVKMnmXa$dC9WQ)X`}7FsDxqd5nkm92UEO~X z&z_M4ndF6?+ai3o`6ZI)`EKt-ZbKNiL6(+9HmfT^)DSIRtkksEcsS?ssJ68m;)l&Y z&SS*&fAM*g|1JLiO%zl~_v~v}Ay_*sA~Rm%46=(ZTAn;lhz2uQv$A@|8BKoP5S;=z$kVSJb@{AI-MXivG%vP^dJ`kg0> z5avmEWs_RrNhB!ur+dqMQ%2*a{$`5c70R=|W^iXGh!qI_tt50Tn7{HYtm_qgE0fu@ zgDzIa+RAS9RRWnGX_1K&!Q6=uW=D7x=(KPwvgzepr6uLQAHDHFWXB@9;FvhA&OYo{ zAR_XX6JuM=;WD7x_Dle@zVb3KnjT)g!I40Zi@6Cf)6j7e+7zQgea3l5j=Q&5V2`#> zOiJD(DKm8?hX$k7RJQmG-m2hbCCTEi7g%ihx);9QALZC4Ef{{)KO2B=n&ef`1Vxu` zaiPojHbw}~es#x%bdU0gcAs{=N+CEZ^st}o?MQKqn9hT&XozbS5Ep*{&Ch+SSF(MS zxKwlB#B|JPjWiu>0iD7oV3Qe|m6kNHeP|Hra*)GvD*AA4*-4zZ<=$-GeXUz2*TyGP zH3PXvdOY#FU-3f_`~m~n-`!Hv{OgO;WufJ9d~V>;pN%TEpuXT}SltugCsXtXe|*6D z|NO%6|3lIFtLl>SgzMo!NWF{Ycq2t)?-SrI`A70*+^-Gv`d8cfa-nmqcK6m8`J9!X z_MGzx;y^Zq(%gGbfC3Y?5U()HTCSt*p+XX5b>QZSPv$?rWcS>s>>Cb=CgEv5Sj)#LUtgY*!?_1_Utcf^yE#eK9p-0E2uT=!wXah6&AYIqQ=$R6Z3|L1AbCZ4wvmr9v zZi9A|mXY8`rxC-!hn~o0zE_b}axg77sav|TBp$v;b$a22Pv@%y=hh(J9*W(rnVkvO zAqWn&f!`PorumjR9V`AIz0EsKDS%5Byvc=qm-I5X^BbFig&2u#Nre~4g_RlX490Mx zAgx}O+3YDWrn~GOpH}xGBvcm%02ja zZ4e;Dbu)ZAndx|XZ-{Ym8SooM9apLkuGZ+>(1v3cwaNlo;Cr5SfMTEdy63_NwnItVRIjSns@Z#sa_w%Nc1+1!{cXL%CxTDXjUUVLCBs^YwPZbQLI)%qx{=TTAKfQ1jO2NTk9H0#S#Vk8dVH zZi$7};#Yi&awmrG%1Pm${UkTuHsUpU-)HS}lN4B0EFW&`gaKfi%1SIU7SL;w9`Ac# z8|OfGakgn$K^|1`^v45jyKNvYt`mSapIAi{$d@FpK2}_NF|M;$rc+BvXOsoxC0`eg z^vXOFha?JJCsd0q!j)Q}d-8#IGMm_v+Ab*`-<(1R@+f&GM(CaOIUu3b_ZIk}AW$|Z zL*T8OCGmMDl;ni?mCDGc!B&CAn%e0e0v=pUq1OUvAIxtu?)USu%P&i8_ZstDwiiY5 z*;P*-(zn^bhz+y4D#7&4>q^U(sxJv?{=P-L*c~gjiD^^U4uVASXbD(q2HEXE1sR}% zU^>E$ZvV}c>3Hn}{&-rFL^&zJ)mQh4e4)B^+Zw@XA#dzw5SC=_?rj;(wy%T{M{RZk z#$1lx_Qd-y0#4qkF3oI4breT21Auu8kXRT3>qABzLS)gJ5Dy+g>_XdBebkA5hMZx2 z&sf)Tmu_c&;Ec_sF^te!INzX?tkt<1IMCl)u=s1yr__lG5&qkuT`74?Ot0hI0xzRJ z8C#w!p@bM?I*KY$9KHe{4cDi!FozN>8sP&AgI}+#MX_MentxG6GdLfu3;+p_#okT~ z%z~ThpNS|}RL<1WNEcNj8xq*ifEouwY+S5X8Yt_EwzK*xx-a95SjHZTfgaY9X=zTI zy$$3Nk^(_&5CIb%NR6<2Ix-7)-(9-sB<794K#Rmjx{IW$p&%|UBM zg_^4{V!4t&dxH;GO$vVr=>WGa$oaF4q|E}4!N7a=oI%z#%r#t$^cJ~JRYD(iZ-;07 z1p`HI*HQ1m-S|OQP(N>68l@+I9;}3(8jdWM#OVELaL2p;o5aZKb&*`t=O!q`RZ z>{7<8NPmmsE?{zD={on|2=$Ofv8yz~@utTzO9(9{F6uJC>KIf&OwXM(FV4G653Ibh z3x|erc(YL|_^ss$Ux)~%aF!OK@M zv!eCMuiJ)M_A?OTcSKqZzH)g2OkLXA>^TH)ooy$$DVrB}#7>cm7l|Y4$vK~KI|-8$ zd1G&@TYZIfD%9{H3d1OOn3TU(`A&>(ghuE%`^O5KNi8&-^c!CfYS=a0Uf;eFWq9=o zF^Mmd&Q`|?Kz>^S0Mv2+KVo7S!%d58ys<5^ccotM+ArPv<2;;HlLhUZG><}rS(LzX zvJk&x0X06{BZL4K{rIY)`s;feAG{ju23gGbtwcErLay7KV&mBavP7 z-W=4SXTNmsePq_RyZ-WTXUA4&Zxn9GYa$}6dxjZg>y*xUT#xoMFX)4bZwWLCgu96o zmUu$EcKL>vdl1(Sx(6IaxJl)Xcb^p}J=gPiw?ef41jxs_^<6m9F&_>149UySl}10G zjx?o)EiBo;<~KIB+dJvS$Zu^_QWfLZZOs?z^;CAD_%-43ff=;8!O=E}DIIVt!uKu- z8cpqXkS08%TcGSbp=eTnDEHF=%AYtP=JYcP?WYK8vvqvntrS)FI`)!A-|1WJ$K)HF zh);Mf6*hW?L*RS%g;tJM$*Yg4AceP!<|~=EJ!?P0UoG0j>etQF9IO}Y#^?qIKbO%2 z5A8tJd|YCvAJra6`)+K=f6VKa*+vLGKOQYz6)fQ$kYVQ3$B+!e^v8K#-+z~Ji~cWl zr^J1$y&zQ<$|lTiraDXwwlhE=Nzwy%Y3VM=o;mkO5uPn)9lUKarm21H5xv_}1@W~; znH)FH1fqMgnqpw|x~VzVaWK$L1G9?0%cgDnX2;1s(@h85kN7DiY?n`h}IwufN-5@}y}<)T}-M3S>$ z2ceR;JkgAd-X>S4@-Yx#L*1n{qRf7ejP8BH+|1;BmHbMns806BtG~yQ494K zH^e-WBp9L`Baw~1prByHJ;*X3hQc?HRJLe1Cl9uZJ|`#1R<%+YwD3;!7iBQ%ILNzo zIDqW+{PeZnI&4hKcQZcJZ~TEsP=vJgz;Hw2ByM*6A})LTi0;()33DnLn&p%1*UNVG z-QsZFh>E(fO5}RLw1FA5LqBOyH-y2*t2(Y3CxqXgL945q0NDLPmUCd} zkgZl1^;AZucrmRrAdi2sykAV9K6DBT<0AdmsE1gs}Z>x%}yx&n6I2zfuj}3kyi4-OgL&$Uqk4}OR zryRldZi#%}@^)U+=-zF)4@U(fO<_tQWFmL|S`DrRmNEcx`hK*zW`>vTmF{iEpfrC_ z<+f<}DQP9#qJmd~VUgLsye9hlRz9-@esyI<0Xh*>eKIgZS*9{D{DM`%THce>70qg} zyPKNq`S5?DY}4(}mKr=KGxrGJfwCs`B|&*MUKBk)?`vF;PI53xNg^h}UG)T8g^sI( zX;O}h+u|1Vx`seh23Tb6Jru1E+@rT~$4MzFKdjY=29f4U7MHBRB%#NY1m+|dtO6)w zFdIbni(V32GK&1g-j~o#IL8anZ(uXl#?Y%_t?M2-_;e28Apj`D4EB5m{Ao^vK%*js z{HNw3V~F4BbZQ|ZMtvg+BA%V77Ou0ad{z~s2XZ@hTu}eQp;p^Vt>?g{ar>7dxN`LA z{?kfvSy{~qJ?M3rDAkfS#SrjCt9AU;;#$8vmXS7V8DU#}Y)!cvfST!B;t5kmpC~jP zF=|*FOx{04WLqiIS=>mxH?CFB`&J7*fumwjJlb7`rH>%!-wGb9XHcDE{$pRS&S)5=lKqJR^yg7VmPb)8(l*MLEK%D4^gr5IwP^#NVlt<2a{ zQ7Gvq7|J#C-dYp!sgS2eJb|(F6A7%q3ck_~hC?@N&pp>P?ZOGERM(nuqr6+*#eM4J z#UFy6vxX@s56ZtQV&PMoU2x2=tKj6m9l+^Fn_>zW99NeP_NWmhTi^hFlN_-++X`XL zXCt^Oc_BBCrigFF#7f5vAjeBj+5W%z^dN3ic3X>*3jUs2c4JH`hBg!b`<{WdeyNCTX$bQz_a`;$P7?a;fXVVeC*c>2qJ zt7)s_JOhS7fc3C9m#1J@Q&AwO<)}lrO+{pP>*`U9ocdkUam6m09k$G?0#!EgZmu2! z7X)rxEZ+f6zl^x%zVvw4$6r%@N?z~AG8c(^*i}n8!q4SmVOvT%+dKm+XiQ0w{?Wgf@cf=%3Ew5#FDdpNgH!rvCGtdV zUZ^b0e)idRlkN+;Q(k;b7;rTwt&!vZ8FD?uw7qSf)dOu2!zK%;pl+?-tG61Q%|&;Vb#(o@E_jHLPN(7x4@NpM5S+ zkisQ{C4on9hf#DzNNnOy$YL~_aGCxRPA2h{%IH&1ow1(1%1wSwfP6vqzCrKrw-tmufDG6;zM}cf z(x*k3yMj12@r4^_8e#R%L@g=(r4EqQ7!wmL;R{HRU!Zyyudrjl}wH zzC3oU>B)U6l_&DrC2FKYY!OtuQQ;Th2bp@Z7F(l~=-=L8a@7oJk30zm4hv0jBs;;vzXe^QviOi`JE5ZSf5zHUlyO_&0){-|e!0{T0D1L{- zt-DZK1tePDV2NM2V2qS1VM1b~C%~7DXdy-7^Ne;nr2=Xru_whM*%l91-S9kt9HDrxL?a1@_nKerBh(I^aV1@+>@R!G$a({Nx=PiL;n zxOU`ha8JnWSbq!ZmsIWX))#q0zrw*QF{#OH*Q=2*ox`m`BS(~3fJms*LTJH|cnISX zcd(Pid#|^3j-tJ_wr+I``6j$DW~BoLS8M}XdDHP05?o{f@x8hjsf+4wrO_lAx$GSEtUdtL;-UVEgOWx1ZUC9vc{#i|5uaAxm%+` zkn5)xRVEx{iK z9zS|BE}LDS;pKm9ctc-J9<*K7r)l{}&tHjF;GtX^c{TYuU#`DgD=s#*kpsZ}s2%-c zg>8nmHO=8_BxfO%M&kpNVWs|;b%UCYQx+8ZcJ}3D1=wn^g)|;=f9(Puy7abG^e4c- zkV@pKZ8~9|;yxB@YxI2i!~(Ij-Jrd)%T5=avrB8GDiVxXTgK2VfUtNo;hk7h6Xr*= ztVeuq)ISkf5vbQQ_8YygYec4qq6x27=lCJ=OO&kawymXV2l4j^{BxTNXD6Nj6g}(r zk25oU+v z2kaN}5n4%86PFb>ST&@v*{_|HB)>P24&z|7X=o)uM5WnvA^a)pmB8cz!vvbQt+pQm z-$syGYaS+VTyw)g%2tCcn`X4XC_nuRHL_T7&zt>w`SRzl30On8etQS>iA)k7mYT2R zg&tM^hDwx5MAcPRLHu1=55-CSoJt#77&o?7mz!kMc%5hUhU78`6*N^bSxYLrS#Wh4 zmS&@sr(%_qvR|ph5Sz@388!v&+(NNbW;5S3^~W*&XwL0n9Q9OBBf#vlOZTH7AyRIQ zE#I}ii4ccv>{#S>_eIIot46ps498w2Du?5Q8hU9Sy(+^j7p0cfn+LylMH@%zzNsqA z1VtM}umm4*H!W{0TFmFw^YqZF458K4v+2#7iYJt?&?nf!b?fuiF|5PRs>6BdAyS+Q z(^FS13hgwFlj)lho7CCl|J6PNL-VYoCLEvyz1!R8_UvdEPgB~_60D~GlTA6Ul+wD8 zMXW5!X^-Ob(MQ>7X|nG zD?WEY`$Dc-mKv+ePq8$00L@X|cj9*4P{wUELu4Ae_&Mj8??9ua&-LkQhLlmI&b=Ah zJ#z0?3fC{X*aJhb4h~97W(@5{(|y(*=Lvvw)Va9uP>R7lfP1#Qo;$rTfbNBtKMIT; zbaWkpBAYuGF^{wGOSWuZ1LF@{1D^na*YA&yeYF{lmpC3cYMuZNHW$TME&t0(GW2hl z<}~JaC+lz4rIPNyyid>v#e8rjvhDS$^~I zEG*1^3DH2SyDGuOMbo3b_LHs#{$(VX%jpV1XUioeW)>;je7(rbQaxq4TDijs39B@mZ#35zuK4>UUGJn0ytrqQM<|8lMriE@&x z*M_`OGu}SdC}?^5ZU$)4<4urVn=D@_&%+zL2)AVHc>dyjbUp{(_Kas&9lIfbsMM=k6u1n~S zy_W;s5HsuVhw{5?EYUQ^LtJ!DalXC1H)7}A6o9Ml7jFkJDSEOXQ)JnBk?FBGzsY`E ztyePEEt(qIC3>i@%j!FE`0yPPR?tz|3B%3y9U<0V!w=LOIvga=U5dchhz;Y0z3`UC7ROIrpEI*mOdRMAU zxbptJw1Ps^y`je^Kf`xS#yh`alxLj-v*ZT~EH0z-Bnjz^#leI9XtW7<1>E7t~BKC+QXqaq<>nu!a|V%QYvEw-W?r`3chhnEOxRz4VkaB9 z{8>Xz*P5pQy^mtz03XHjhyOmj`PLciE%tZP@}l_eam`^pi{NYB{6p#XIPaJbmgjY= z_a7p2mB^%YzkfFPj*H_6`q|LS?|fnd^#o`rwC7#pCEA!jvxL)!3!~oB1yk(n46%|V z0AyfJu`XpS_S#9arx;)HZk-xYPsY zBxoortpDX52$Ir-YtCLqqzi$G*+hB#Y8*q-#N@6|v3w4sPZhkekH>yHxd*RTYVQ|O zH}M$67zAzVp&T;|G!f|}G8_q@yeN;7HVaG`l~hL*LmHszVXcSm>pxubI)iZV@&oVQ;m3_b(jy0&Imvm)xvce%#a z7}gCznn504f)}El%Fk z%{{lKs5?Vy^$}wAn(n%(gZ~>XZygq8*ENm~0wN%wq%;ahgAz&)DJ33O^B{zyR3G{Jc|F!ANlju1|3$t4BjDY!i@TiN%5pzCS*ZGpIXr=LQth_Uw?ZclSe~PimOdaSCZSJM3`pKkPQn1)x8&Pe9<$}caQ)a_N<@ge$x^^ zf12>~Ji*G)bGtjMy0!1CIW^4~MM!lDcpGp+@k_{=d`5rdRXYh`FH}D*iR4EQxa{AG zcI1l;50AYSH;@Ej2rQw9afvNvLT3@cqAE80eRR=g51C8VH;AQ~4Yk@3)o&irsy7dU zmZCf!4l|pFCU(ZOl9=9+RQ45n5x6amjir=d_`38Bo?m|PR!QFENSzWD2UbOG^}@Z- zNl6QvxLJCB%{VzsZMuDy_1)e;a1g2TA3lbGF#Qd4H5lb$6wgZ*9Sh48A@1rL)hw_- z>(aHof1mIXX#=_iL!^SXPL$FWtE%X{T#qo@>vde&I!EeHL~D$b&4Dada+0CLiHve| z`4O=(Z;4*PJr1J}bNk^f6$52o9G~Nhj%OIdtzB!27!{=I2gK=OxCum}n=-){ zJ%9n^b6XRB zHiPP2EFHO?;{b+(`3LeZ;$s5~ZG$*r>#In>@}T{BzBjM8;~^4ou+eZ05>4IrI1~7G z)}nQ2Voz~MaOWsDry{g8LQjK0O9fIa$AZ(7zrkSo=Mj{seWh{`YF4BSCF^wXA&OYA ztv-}PL#y4k2}CahiFas3`1OUk|8Ena*znJJ(6%b_CZ0__Ijki-gXGuU$&h*$SX4$A zvFfQVA6&YU9v1h?=6sL@x^n$eN-BK1@j}yIc6x2HuEhIT9YSks_5V0Q&Z0V}p|m&c z;2KN1I9qU@h;v4%ytf^{=T{4k5|FEKtB5WQj8UYy?i7&8u#4u$b&c?h@Aux5FmjFdBbO%dAFQI~==r;^B11bnb2_3kw@4l(i=zg75`^ zs@R<1u!7(flb*Tuo=sQ0lSd0j7okEpD=3<$auULZj?T)I!V6s$vfsaiP8V+H{c07M zPHvitMuh8nvThwOo(;{Wv!UJInur+0V3AX^qiPJykGq#h-KT_UTjOHhmh`?Vw{5l2 zwvN1)+J?FKOv7es}>V&Wp(wzDx$2g!BH>`qC6N;7v}kiiqIH=@`V* z@vvfeQ^oPAECy06{(U^ZzBEp5oH#qijeJ(U<9{FEL3!8&jh-dO`(p}ss#A70nw_uu zk0zA9Fy~j7KY7$|%$rg*>mz{@s3-`l%OwL*oVP}f`|cwVdX2Rukq1Qd+^Xkv`1obl zLVGRv_`I40`A>ExgEMWdPo}bz(Fp{46IEiHilOSN!kn(}kgX|8-GixmwQq;Z~Kp z;89Ib;D&tLFyMf%=gDGVmR#!kQ3cgg$EG%?>vzJU#-dS8YHiQb#_;eo|Dg~` z(lL%owtacX{>YNYEsTb;vDb{4J>*gv952fXrd2{H+2l2H~3CR0&A+wpyH)b#<_E!3xG};Eq z9x-}d!0B_JEw8DD_Y88O##yYCj92VEjy<~QjMmtUEWIfJ%@V9FJNW2T@v!d(v^e*| zcJ+hv0a@=t!s6$5SgS8ecP%d%Z|lc%Bo^Im((RF-`UI+tT2;L>=J9yj*z&@*BbL1Pl3=*aOwpwf-OhfJg#y{)w#U0Q!f=hhM@CaSkuY z6&l`?eK;pwPO}QZyPq1!%O65aoRC;ew#|mg@=boRonx(HIKnQ@UJuWvaj>Dv-m#12 zVKASo&b_*aI4`n^@8v^SW$Eq_^MFL?+uoIoHL(69&>h+-oItJ|12!C~#7~FL84@N< zwH(U#gL_mY-cPH=_C-cXt9m(Fs;ocl;Ly!X^sIYP*;BaBvR`W=0ujGM}f>E^- zwxHUwzL9~YlCxhHl?qk#*a)_6V|gd*aVVit6-BSTbPtG*v0Kj$azX3OW;?mMfyr@) z!mlGC-KksWE4n^Yl~=PvT73fdP;qVTx450j9nxtENri!1E?ct9<4T|m8BAA4jQqP8 z(XQ|OR+#ahr4fatknVFT#*IDEvnR`>_nXw6<`m!BdfP*|o*z>XDlddBFE{(FSkSV9} ziO0C?syiq5tg&-Zl$|Z_z3OIxdq0KrWW9S{ zK=;+lL;JXgIuoYGA{ApX3)Y$I<7PS76xxQM37t>(+zyI6kMy>dBUJV>xexj?oH=qp zx;ttQ&!=-$a1}GFCOsv6Il$!lMyk{xYY1Qz}48rmf&C($H*K)YC0Yv!iC88$E7bOPcYNOD#ucjlcBb0~3X z3%?(K8hPMN|8warZHL*yd?!3RVZGlsdW66?W6PPXDXCW}XVQpaFOv&Yw`R?2@|`f* zghJ;DosfrWL(p4kW6`AE&wn;Cqumbk>~W66a7nT75M5-3*WyG-0Kub_;>}>F(RBqp zI}`~BiG5iYO|+S%*oN4@(cUB3AW1)0wy0pv7J+Djr(S^v^R7GMK%ap-wZM%r_^N)? zhO1YP^5rEGu(y2A`y~ESL+!D0q{z|5?!kM}tNt6-;<#q33Md!RsT~b;)e_8lMc03n z$J7(XO7`p(x0gl#QEr&tD;`ZQ873sam@^LvDDJr-I@8w(9UIwwyn8YHd?l-lUm3WF zkO?{U19xzJk46IC2%uDw&PaxEZcDE3aoBkn`eUGs_5Q?3Si^oNnYIZMkO@7YC?bbb zS8Ndd{Bil?1Tfj~BGRVgs`M(8A8M=Ijs#p0{v1X5!5;$*u+8J$d)z{bKY4ON8&$}6 zd#ffL3Al##s->PiK}6DG_lRBr-TSHL#&4Dg*Emy2%p6fMicXg$p?Nt^MYA_m3QHt5 zixMH^o?!QmlFXYo$3|P5fqNHA57t!D;)(VmSc!w}BOKOW_GPeGIV!)GEum3jn&$!Y3%^d{`2z3zD4I71>ZtGNDdlt3Cc~5vC0@dWQcHH`pKRV z1UTr?d%~tYiVJIA0=-((mu8zBR{Fz)nX!yqJj7)$>v!^=L+~3_10fiAxUyP;TU^E8 z7aq-wLhthMl%6)MhB_$(;BdC)?)1|LXczA%tl-h3Cr$1B=k3r6@K@lf2wEtf5BAQt zfn8t;!THmGi`9yD5xtaO`AQ`{HrP%L{4n}beZp42pdac=&HQBx3)1hyzQ*62`rzld@~%Wr|H#_RJ9I&B`6m~8yYD{@SW&Iv z#7iWAIV)Wqf&`2$Ugw>RKKO0bLa%_5?)xD_N-lG+;dskUw0r`i|J5@+wAG4~EOo;l zn>|2p>Z{%Dlz(+anTBL)+C_Mqf!rt@P5Nz#zaD;9@>`$NHX#A?cd{Ehb<|mYoz{&`&T)-acc};93 z-+{Nly~`D?bl00qhaU^%->o0a{4Ix!9#dXd9$9%Px43B>p>ZrOO{`&7`ICyJN>O0# z=Z3A0Pz<=1tFenStiK6<5_!qAt4TC+0gPRR(_QBxERcY6NeHnjO3?m#W{>_DEMt8o zb+cA>-LO+TvM~c+*FXp6)&2A!G~o+pI_Af_h@-L(hI~kXw{q%wYid>i5^(%DH;heg z?T_2UDVTZ1Qctyq!SLnv;vw-S_Dvw-;ArxD%gg6^G98Z5t;%2jSm)-4|K-I}t}`P< z&l8^;zDuT(c(zhRDR8!jU^y6yapq+D3;s8bQmozRfR>j4mr(&@kRsaKuRx8*V1DQ8 zp7`Eu}i_A$q9d;ZZ$YFx_uP?^q<90K0>6L|D*wg z3lX*Xl{PbX>~9|``7a+z%;>j8@BPhF6Q+MUPLrGeUFkViaI2ZOD!3-=0sPc`v!cN4 z_ipKB79Y(B#_9OsO&<#9RW#(UBf8bLS~2Uhiv(Ca0 z)X+=(Vpw@q<7MKy#>vK-Pjn|@SK&&lAGHpR)Q+E2tP&J}zxnEO^@k>(`dyR1`b(1s z?1x{Wbp0HDf~vH&sLNmCeK+NQXl(Vnh^U$4(5r6U3;QX{tig)wo_Zs%8Tgr~brj6X zhX4uaOATW#V+#98H5H5Ov0K(-<^o8-1^7Re_|h*Wj#{Wz=Ij5YxWs!ve4iHID?x{~ zk{_X*Vovl5pG9c+qV9gyEjyEoWv-pfY|x|?lc3V^#a;wF>(3U{4BH%hE>Ha^xJ<23 zD)RC8m(Z6AnTt;rc$z+ORo>BdzRP@Od+$lMd_MytLwKRz!BhJi4xDYmJ)9pQDy{>; z+1jA?hvK4Q{?C_fw{UuQI^hwoy)Uf~m9-vg>AX*f~^}Ka3o&~LOFez^ZtPm zZJfCtG(`+o4E{?1;f@s-NrUhV!(=s4?9asr>bQgU>F-CpObRwd=;y+l>(1FU%oU`&3ZzprEq(oPCwE0!P$FJ{Z3+GmyLcrN3FsALi@kE4Q^cHIH zfw11c+tKn`u-m{~)YzE5tZ`fPmtUS`7rac5kHxrn2LYXwNc zPpr2`Me|wD>m;AJCXG*rHgmE5dM(x09E1MlG`hHTiIoj%rTmA}^NCY=rGgSKAos;w zf=KRdErhQfscB(8+Qht4U6OU8#Km+3sm<+TJ%Jd>x3`mzyUP`OqqLBClgZlPGWM$# z*(#m&p?J|v(xpVv^MvZcOi3f17rU3euVvW3`=*bnGd*xwep`7&Tx>jthM=_4=veHy zmB0vmL{Bkr=A|;9(aGN~ATZ+kx;(Mz#rtpJUJHy>SDM#NYa=OyZ2Qs*mR3BFYH<-z zZR_S7sS7+vmryL5YloH5a{Ynbq3K}Qm2GYf3cqOOy$lvPV{oML2_k6TiAQ0FFU?eR@h)!XGMXys7G zD{!sydhX)M&Zp1R^!}LKzUVZT{QmetAO}}*S2K56aFh%mYt?C57rvuN$8y)boY_^F zgBktAg5K)|%eNM3&wnZNhXb0Zi-h zdEyCatLb+FixI}Yty>?6r@dM+)Gg*+lgxwwkW8KsJs@*i5YwM_~Th28q{M{EO6Z|ptD zgNk_bXgB~BFD{=p>8-g@t?WNYs@2oO+`gZo7G*yq#g8UBFc_X1s4?`whfyJ|plfA3Vfox;*YXI{g#eQ1mIj3YFn&j#%fw;V#wWuR+g zzqKbC3MakG+hNr=Ku}{hi3!|k0qUFn5FUGg(w@Jy_qNYytt)*>p#+o4orab08N_H8 z642RKpwrMOcx0D6HZ^3uvQHs#-ycyy0B;dQ2NmPzE>X745G;9=}2rCrv4nHGD z1$<{0U9?``5G3mT`siIM?6_-`aZ9V=lI5(r5Ni-fygD4CtFCjpK8<^xRYIkiDCY~&{P*V<(_hGlItCyKnIp-SL*Yn37wuRp3w}o= zYw=eR6WXs+h>OaKnip1sr?|ia%o11~LW9Q79KKQq)`Sj}L0=icpnmZ;+bf2h#fQQVWCwx%{{ir8gzKI?67VW3T?6V4*GRcR zMHJ1J#f{H=sxV%0nAASUtLWYT==$`%nQph{~NTo_ZLzRQ+MIXvhw1lu;I~?n+t|>X)@QLvY4;Xjn`# zL|&|w*8$%Pg%h2QZ5=}28H1(ZyvJeQV6^ky5LnzL*JRlDutZN`uBF+mF`yz%H10(` z)2Z@=GE3piA+22_xR}cyE}cQ@oP`9qU+y@omfw}+D3>)8V&7EqtbFnIGBs|dCMYhE zVeGqB??msUN0?U4?vGLCj};If!8HWg+QLf({nqi%HpQwR@cJB5)F#&h1s}*OzviCZ zC9d;-s35Y$($W(xR!VygeYN2{;~n{XFLW1FhsAE!5@+B^ zOE&SXZ5&ByC37siR4#=k|JoDw2fEm80LeB6+D7&^~cWp~4N zyWOR^9J6(Xx0r~j*K(xm zJ1s!D^%7bPnCzy_>37`{Br5T-Av~f*N_V0KzxA1YKebT(COBEYAa1Agfkh#*o^NLo z#}`xVmUugVHYW}%%h(7-&-ydYFtZuwI;eHJkzBM$Eu^cWMAG+MJ;y(p@a;Eq&%{3> zSK~Fpbt~J~EUXy!-nD;Br4?)D!XI7azsP(jRW#hIhmA;ze}wa3W{P)=xlL@BtO_GT zHzSy%iTRERoYG;pnU*r>DI;mo^hI{dc31^Jj=WSKB|Fc53*Mu>EiD|J+;b_{$ebYZ?( z6bPJ%0JEZ;y;hm+5T746r?;E_Nrqz$lfy9&euio<-}Hs;qRo(V(iN84tjhKEi0?m2 z^DsXpYej``2VxRhV`bS6i8VoMiw6#6NxSc2?!B*3ke)07u!&>KqTi9-NFQR(rW1ZRW}DjH zc40^Cq?ZiL3TMlxZ_w&Gr3Q(Q?P(Y%_xMVlJ6HGty0|_z7r2&~vG|-_!l#B9G0khN=%L%`f4gr&6lEYsdYu! zbRa>t)ld3_y<2NmBAkm|b`XNrdD0y%=v?rNTgMP8zbf9Qy|${>+T6nn>_ys@@2h<; z-w*#RC&gfni<_e$P9z}&y}(|AoT`veiTc|(+PPif8QS%*DOPhn)tV}P{+uH2TSDVb z?G9GCtNcTLETcZPWvWwM+qJ7l(gWKZoUGz%O)B&Lvm8k$(mi z$7G^naofVU1R0|qmAGU=iO^lm<`{?SI~!Vf#Fz@5)ORFSKe68sbkAHmOyCF+NSj(n zyy5SZw{6hO&X6%BRLx-J?-TwgJ|^?xnL*?jNBG^8+w29#IX$F=MAO`(HZ7IZ4FF67 z2D3tS6&lu7SC{K+o+I;RCLd22>&L69iGv>_-C154SG&V&29X|_%A}``M|7vE(jL9O zf2Jo$CQ6)oSGZ<}4=(n(R%lC0hHt?m4rqPW5NNRTW&@_}mcA1Au!!epGKqv~)qavZYd96Lp7@{`i0hO61>rJtK~@?kp>4G8cQ zxUBdxCH+p?6RZ61=U*Nc*jQLzmcSi^V63`(Kr3(78JJqOWQQO8b9#Y;aE6kN>q|Ylc#BpStz93W^;dE57ggl#9 zBd=p`%x8em68T`hnX=}W>vX}Fkvqs4xtU~hBG4i7dR;;W8vHlY8=i_w|g$SDW zle4)eA`Z(j$=c>cMk4YU34LmwyHwO2M;;bmr)VU{shYmh zd=hPNjA?kn8$l430AZoNBt`=MojjF->q4vepa%;;KcB|&&s;lS;h|+nfY00w?paU5 zFUalo;)$LE&Nr+&BpRUs)Bq4u!IPj{uzEEXYuzdMyzkgkbqqS z!;_;+LBGk9JS1Sc8M?WC=z>1=Te5f-T=)Q?0pu_Cik`rS&t2_2Bsg8C9Vpx+;fjSB zEq?}zpq2mg&Jx;5Fj)NE&s6d0SSK*MwI_5R` zk|@0U2K{W$qgi?4UF^w?;Y0-nYCCT?lYVXr6z;x#2_2-r?uZ}%)htvVmO_cyC;sTO zc1MH@%9PSUZe(C9jYxo?`}Ox2tv=%x;Tyv*msI9UtRB!0DDf6g|HIDyBUMG+a+kVi zx6Mxqpp6@zdTbNdS^b#}nAamP(_>N;!fxi}UuKsk?^FstTHFksQ1+g0^$eu{SNELX zJTEKkW(54Fc+H?>Gu!izD>^)^=MP5-@$oOm!BaP`kBScdmNTtM2PEKg7Yspi@#-4! zGc7(b_J5}Ktx#A4*w`ApTo1klcZ9eZPJO;6(nY{)F3ENye*v;g1V9Z)fvu`goK)*V zC49KYE5y}{8&(Tk*a@sk=xPwU%$t;G0M=vz3!umo_v>kZ`af*sivF7qvW4~Ww`b~= zNPyDQn~y$@pexz}>g$@5NeSBR9$=#KFsmW`X(!ap#}i%uA09;Cyo>O^_)3tPv7Pse zz5cuRiah?Ms<%FtY-%Y(Yk;KQ$L^aILZO~~Yj1^h$9lU0#-vHsY5v00QYssy_l24W z^)5F;eSKOg_;Js9n!T^XFR^y#k36ypF6cKep@jx_Q2RLqXfmudF9g2CG;>(d)VuuCtkh?| z{|{QLtGL9sJ-&E6c(!Uv_MpYy@WG&%uD%N6kfe|$Hp6BTp8Rg6r(C7JeLaGHE#*c) zr0eko=>iSWlZ3(vD=Xa>YJE0%3S7jcdq*(|7b>JbXkz&(te-XHI%+_^>F2ZZ-b!J) zIo#rBmi?|I`{0PqJuJ2(nZgDv`~KL#8WdZ`M3+zmk(?E$8QAP=p&g3v+*wnzYLE&Gs_#KMJ<8wud?o6|tx7_l@cI8xrzy7ys~ zWgqJ9Vs4_39}ML!n!V9mRplnalmG>hZ@0?^3gHH)s*a&m0B9Bf+a zurL6;@WY#LVp*&3hmh^{&BzLEIc+vOSD7go$KyG?wRb{hx*8fJOo>=F+5UJQcB|tb zEM9MGwoA2k=-#<#YK6tWkd*XTI&ox3=F+JYJ8&l)qRNilaiYD0^w5FgohrfSaLbq1Ygz3oM21qdpI?Km6lvZA-P#b7sVHyU z{Z#zA9_u`RY%AcxhG0*w02M@l|0kGNC^5E8)=OW9H8(=FHCFX&vDVeyyc=g!)FN!v z8U*tM9v2}Fi%YX6IN>M_iKYw!PV^#bRVSlQk5DRxI4lg!{=fO@mSSho=)%|fnM8v3 z+3$)c)76-;5drilg`!}8%-B^moM<1u?MxoE9y>cMICD!*UZfCxLF{4W5g2#5+<7la z_pEjG*Uaz?TD=(FMn8Cfq7Jr$~t1 zee{v5*lEsnbucB~rSj;}Toxf;E6)jsX^^B2=rr^WT@0wrQ3q6a3&QLd9}(@5*>k#2 zY94wS*4}+oxNL`rQ#EdYIPHziP8;>=TFJ3sY-7_RubiM} zY5!h1(PdRQC8vIh4M{JX1nH7sLEA3%!%a|je3M|?69}V*^Qk@Og^>T=u}5fNm6MLv zG=mJg=|d~LqFOnQM7oF47#R$)U&Ul28Qs>=iaT)QA4ui)dvWWbdB36~+1|`2y#3Z* zKALrtCh>=pn%B0$g3u#$lRu0xMy|C zqusac4z-gDmr0C6!W?&WNhu@`9*j}C%1s2znnDB;w>vbI1v4E(@cUM9OOm6gjZIhE z8S9hGUg&Go*fNqI|(z=EDLPM+j-BpAkOYsw{((iqQQ3q)bzq5Y$Ix`+^uAVTGBGJ7!=%nD3 zO75mRd}hJXsOg%cZ|sMKaZ3?x?0lcTQqSBIul2+}nN1;lM>{65Lcz|}jtV`U(K~*{ zO=zM(+VQJj6n#y1h*K5~O40)xr~BOBR<3>CX6NJRcZLvK$f<}Juh&Bu2>IOa>mHnV zYS(M$!f>1QiU#Xjo}zUNe&Ud_@|+$gTb?Pa9>9O6BZrlUHq+@jA*%Q4-q$oig{OJk zaSaBpMt3A+xaDREcr-LN5HXkUvSl9YE?g>vZdZg>I3fdbPjOlt85zgzX*U6%g>YqTIP%Nh+ArTeyx-{ax*em9BPC~s` z<6~XtrZP!LKwhtlq77sdJeI$Sqq%O>!e;WG-5$yk%z3EqM%iyHe-9>~;wXbH(I%2e z=zz93^SH_ax)eb}3w1cz{;XH#us*MzCIbqd*YBU9jjId{o^z40j1=3}bifi!Gq82O z#0n=83MxC{lF;Z@!|`Sta=vbc7<$X!(x(Q(4u>lRk))Z-ve5mo@# zYo~a7QaHKfb0?ONx3SVQkyOugx#5O%P!v z=++B4J|0-?>my!D_Demz@W8i+^O0LW*mR78X>DX#M3>@b9UNdI0SymemK-FjZ`HjGyYhByfSRoTD-`i@ z5e)hsmPRlY)>_F4hBh-H7EQMEc5iK7Uw44_dPGwdq=i37zV^Uxc=t$UfbmF<)rcLsCWuzAS4J}*JwVR!!7_BVCegZ!*l zi{gl2;CUdN&#I5s+_%_QNi-VF0=Y1);FaC(HM1K0pef*3D@^{f$`K1J<_ z2}OL$_rTF!#AsGQ8T;F@XhHAX=tHgU=;N9I*vVzZd7}L^*FA7fG&~2WC_3w*bV4!4 zm`w4S!--Pmnp?I_zWu6O(NR8^ppt99_lxz^%8e)AQ*sS9OHTVdjyweAVKc*LSta)c z*Ab;a`d#BAL}c`C5Unm*^r;P4kJUYz_34Je=Jf6*I21f}*tMWhsUi6-I zM3;s4!kz)kNrZY69~>9{1bcst*9o`u(%KRit>>cMnng6K{sJ%bIL@O{hkXST9MKvz zg3GwV;CER04UWnf>U#!g&pIG+9BLX5yD=L4t7U9F##f%xG&#K5%hG?hNc;X{!uMlm zy&HusI7$1eS_C79cS&+%)XjaN^GGD%74zjI_dQBPrFsalaf>0lLB*Q>IT&v$_Q0RX zXAHWFNsFNb&jU~6#H>ZA4U&~yM%P^0l}3C0jr}YIT5At$2T`epy)`}q-~Wjl+Gs|3 zr0~^2b}GV1`>3!<0mH7sl(nL2)p_qpRbtc4P(tNVXS-T(pWP?J(xkZ~jtkt{2l0(_68K@w_u`!LcJdY0mF~$D?5>5gdIN}fc+zeTf-FVMt!w7_mnrwuljRR)FGC8* z%J65)Z8?|dfsZ{oKnx-Rs-ZE979X*9Qk5#As%sO8@;!o;qWBd{DsYc(-TnL}oZVC+ z)~=DW=ZnDcRETIE?}%Bu(+85B^%v@P{Ly%0`M8bLuw`lOImMpfV&PoEPE1|56{UyT zmO>y)KYr=0+fQt0`Ym<;6qO<7etrM3m zUq{r3gZ_~Z=loMeb~5bc8op>Sf1LC=U-3N?{fw9XwM-8apGh?#1!wGYk9~ zlD+mo1!brYzg1~sEKzTRtjfKHgA<^yPovN0fSpr#h%FSHa)W->`A-0S&~8|vORy{p z4VLCr`zNv>#*#3AYL@>MIh3El>Ky)vVHtiUZc2zyjG8{u%5p>UvqsJyHj#4C4fM~Z zE&3zaLVKkS_E4Wz-cI(}GZc}?@yU$7`3P)gwPrdUgt}E^`p*28^rBk@j=u38 zy(<3U>&DyP0nYBb&N)&0irN#-6JgZEKR?0%dFUNs%;-OBHc{I^^Lv5ii~6zc5Vd_A z=IpyDNK9fC8dT>3z3nf0to(`Fc8CHLF7X$1fG2xo9Ry^FxPRKvyFU@~<8v1h;6-({ z+r4fOCJKx-73s4jRyBO4_jg@g&KPi);_PDj`HDf|j?q@uS)}LnF6AMY+L>1`)%C!k z^r_cqZWvoH$4~OG{i>SL{J21NIsqJD+9WaM67fh+>Xq(;Q;&QGiWq7C-7riuX+2ZB z%K9=x%ZH+aHufLH_{g+{dtH0e6E|LzmCdz3aYAu^Nw5T+;EHE0`<{lP0!+g9FDj#b z@$Ba3S}&PVT>o$u=3v;XRo%a#86)kXi!UdKyxdC0Ox<+cX{$VBb_bM{wwzV8@SL1 z2ktrR1CzetX35RCWINk6y51%`?8_D;{D+qy?iEudAt#04R!hcTd7D=ml!)5z-bczS zA3y)bV*vPXY*J6Qy@dn}Yjo)wF>H`qy<{|9WpXt`;r#9VOP~Fe-ViTKNO{} zXDV8&ptsI|Y6Mb9K&kc2@~dS(&#aQ~{F_T!{#<$AgNRiP1T22e^UJo^Y0mytpIgH6 zcDL(Wy&$K|*1q@L)h^EP-;J9E>WhzAUl3-ES4F7(^~wH`gHi>G8GbGN&YP4d1xEwB zWqp`wX!AKkqqO1&<}yWoF^Fw@C&P?aXxx%YogGoojSRw&Q9 zEnHk>wE^{j`t4_)h|cF*Z5Q3k`vFIP1*{$AeFCV1J{W=*8r)z8ZtdQp*42&rT{OuTR z|3~s5Na8;=((eiETjIZ%gWDl%i(7QdYx3g1D@O&q)xvODz2-auE@O#$9ytHY{SEJF zA@%}48h95o&z#$8*WE5n3~EEoNA+Lu39ssjL>O}gRqanYmsR~$OsiA!OUGy=G@q?G z=~9!?pSp|wxH$_l|GVL18+ARU{2k1r=QW91<{O4TGT?vf69xHPA7AKa+NTxvzVKZ* z_Z9cYCCh5mRQW7_AM4-j;j+9WVw_xhKDAqszhV$p7%WKwB62qNaHAD_+ zPxP?Y#P&@goWrC(4(wn+RZ&6LoXY_xc^o7~nZ8}936x7mH!N@Q$& zR-Z_2dH<`M)d$mVF2l{>K?}zz7Qo$n2I^f%K5xi+W2bKde_Rm2vFd@)lnhoj4Rwf> z8Jn<7S7yq6$H>-|Vmz{y23kY)2}6^yd@Qo`{P>LF4&SJ`CXMa81yh{fO$_0>AjYT9 zoS&!%o4bMxUlW&p3MTffe=j#x^l6Ai?$OptN%OmdyAn+@O=3-q->998*jW={6Qw-epwRy`h@8xP5qP@Sy6^ z6v-8#!+NyPBZ)Y@w=<-gbtSUc7;$d00C${Q8L-Ckz8V(=`-F|mIXC%=h=xgJ$uJw! zyw;ew=B=D$dG`LW)gU&){Unq^brzpc;OGN89)M}RhpQw z3unF=B4`e^&fgjyZ7=v=uCXnen!Ir!!{XKZBqQtCcq|c{4C3a#BB#mIC=+p@-Q5jA zz`Ad^E%5VN9Jy6JGo#DOX*E-nBIQkgq_gk)3=YS6yA)nN?SxMLEGkQ_ST332+KdK! ztB3i<{{ByNc>)q}o8KK40$rPQ3%ezKHlDT8i;o1TT-@l0HmTh_SY6(sK?1^6=b5p| zX5_&ER6=mc|hx`WVq|sr{hk_KgZSm*q-!|TSFP^W4J%O7IU4I8+ioHaj zw!N*nyv23Ab!mtMKy<;`(7}RvW-P8CBmkz-0K>g$68#0UT_1)vxC%xj+9Lds0M6)t zU~F@JJ%Ze(o(cNgZMfUy7Zdc&E#zp&{IMoOlL+b4j!AA%1+Bi8CKB*>+P@kJP;9x$ zZFLs2Ijy>m<2uzi9osg!U@+V&+0LI-!SQDI`4$c*@#gh>1@9+=vBE6*h;L;n(0)Q4 zjkKX6O%l*a8WGl-swj=dvMS>_Z78BIJEYoY=N88s$QZ-Zr=A>HM-?9*kqwBuC`^DM|5qh^3l9x zI7tclY&A!rJ+lzf{lvzJah)`9(Cu!^WGkbd0BarwT=!U6rjIwe3tTU=M4%|HZ<;mL`mW1&iReq$Ur@ts-! z=Y3_TVcj;Nrqvxa3-dcU?NN&sTm6E%5)vS8cYXk9{*Si~pUVmxdQ>`s@647>7toqr zJ3L)Z+Fj1h!Q+_W&9>9XshR&Ke5GL{=(J|e8CgFRHT~_~E=8klYgOASs34c#Ond|4 znXpgzO#V*4&{TUD6$|NEcaqRe3Kyg18~MAHF}!8U+0WFKg#&}oq8BL2U!P)8P#5sL zbc~!>6>^UC!EWT_VO1wJNyfFsupF;C*ND@IQzCH4rYwa&bquJOj|Yb276(EfrbT?H zT2|x61sS|fB7Mh9R~rE7kPk1$J}N#SJ*uf!3NBx?`;zu%?#bS;tWB?TSwx}lLuMSg zJvtI{UbyKhxN*4DSQZCuskmZhHz;RdK}sin(^U?EGh0LfQDCM zDa1~~kn1r>LeKPNv795utry}FpNKMd#(Ec>rs8wwgiWk1xs!_aK2j>kCl>drC*-io zbC-n$i!GbT?R%3RZVcwOkKIhiKElyqQP_X7RNLgF(9R+&p_RxS0l4i3aW}v;ZM_%T zxk&Bg9uSKo@#clo?7%IML?}pojzTk$gixF^V+*IiTyaq|ic;#@ir;wWhlf>@l50Z0 zZ4TwQ)SD-P0zn~@D=u!8x1(hESXL472jh*I>$WYrIdi$8?eC)%B_mud6b2>EiaQb- zA&eZvN!J42^mX~E$F$<49~zhm8oHiKODfoe=WFED;ig5(If3*XJ)P27 zZPJ`-qer|cknY)K#O21x=;SD66|iZT1XQ}bW_iQz_Oo!kD5*Ggl8yTb$)3agw8x>NQPgiU~uxolfvaPJkhaAK8KjE#X%gsG(KvP zGGk>P62uFpZ`jYBKRqa(p*ejA4CT*U=pr5F{&e`G=w4=EJiZXVmF*N=#i=@*oQTeI z$_RBc!$~CIHAm%C>&Y|vr&D)%7b19Jl8zr7^YbBH2}N-`i~U%*?LCv|8Oa4p6fLY@ zI5g94CKo{inY#BnChYN>3evb@%Bu6()^=vCHd^eqgz{I%sp~cceH&67r-X0Jj>z+q zlgKTKiV_8N?&wriP71uiv0GIN)Lsp}dxP8K-lr1C{6qGAp9DtoD@(=to$1@9ZeiiC zw^cXmQ7ao`dO;c$sgpv{%W@`_^hLd~V>~9aE-ApuBxzMR105YRP)B zb$sFHiv6Hw5piTIRtc-Sr_ZD^;Rk(WS#jdIro1wO1Ag-k!6CV4SeY?_7)SY+opMEx zcj;VPBX_#SoXdg(Gv+^snL1a1IQx^9zn^twwPz9|`cO6)_`I9}ardzBXUX@JoU|wK z4L{ip1h{ROL%>q9%e`>3)EIxs(~&jz(@4wwj#w6=r`DoO|Ke_snr z-RW+Z{o-16ImN0K?v;} zPngfm;0^bVfT2ks5`gCus?piuygI3C`RrImbH*=a-MY}`M>Fs=dzslmed51CaoJM8F^LwEu!vE63Pgj;x@fc5af3HdQ#f6RNv)pOxh=iKx^rLWS^UOmbtY$XM*5tik8QIh zB6tX%0#26szR;7kWwSh&eC4X6Y5<(g%3D5GM*?zuf?8FzrcINt>W>Oxl+Vvi5&E=g z_7vcp8nBt+#+0SBXcruHka2}@+*6wWcaZ5Lis@#B3*yE@K^$}lH^aBrO||>!j-JuZ ziPW_c%$6HD0v4NSQUvwo#*r|~te@|%wQ?Z=B@G0doM4Zc&~DN2LodEpFQO>!D+FS=b2U$g0^)6TJ9>Dm4lO@MRFhna7kQV@#hqC4-Xl0xc;2|);5W+q4$wWBV5Iok7 zfEXsq%$!}osc3gJt~7{qvs4b3S7cx+{~bs@>=|^pmAo}I2}> z#Ns~q0(|wcw<2MU-_yhEW$}1eghs6a=Ams#b=kI7G=EJ{+7|YLeoB#mQ%C-{@zmxjiK|=h1RHP30+w+G$tJ9x~4kzk_=-8{!{zzC`Tz7gM`;T zaSJccVp2UgrV~^XizNl_uua~tU!|y)2xg@OD>X}UD)t6dO?jzsJ8M>;pE{xxF~jGjwU>@FOeBm+;T$iA6f3yG1}kQaTq&>5x!B zx|Z&Sl}1pImWD;TJ9qcHc+T;7&g*$TzvuT4uX}E0X6K%{=Bl}__q50`INyC0>+Uk) z9=ExyRIBC5iS#Y9PxJ}0jkIe6q%sck<|$#NT*ASV_^cFn`}Ax~ zusdXxvRmcj|(Db};VppB*Njsy(jUgAuHwlQe@VhyN%c?6UV zur4kZk$~0;@SSAy{89|Kj{_1guLBKMMEmBpJGPm0*aK}Xlv1$&%_{oKeNx;S0n)AV zJ&$hFw9+z`rYW9;diN|8VB`K4j<*Tk2((|0g)gd4g7fAOs>Xr^#L#1LfrDPe9%KtT zVh?66OZ`Uvn+KL5^)J(41nVXufN7`atMdOCJw<%5v=I+as~_2|niuhrDNCZ!$lp5d zPWMSW-~7IimTW}x0Vjocn@H98naa=XQsxYmdnwvmOwLF^bu4Ia*qTMH=^sVoKTs2- zjDe_+K9T>K3#)4i*9*8>hVJ6XD6oTzRKTfq@I}qBLz*{Z4J_+2C(mH%(8ox?oyI7Q z=zp;EpLyGA2+ad&K|54WC1yYK$7?Pg!XtOneN7PPikif)jubB`=0&eMu5KjnA!6V| zU|#U$gDFvmT`3~%f#knjR+AKe`e?mTmZG)=5+LYw*sJ;m6@;X5Xl=eBnk7U}3MzjU0_y_(jWWg!}2~m~6qXpwIrbj^h-8 z-S=0-S_$SyCKq2*PQUo#_{9wUJtn(4M1eFymu8MwB$?qvqSJ_@{R?5j0i+8qi1VnSHGgOshQ{1yRQhpg@nEw7mdit~z5aq_ z*$Y+pJ7E6q7ib4(cESY+|1y_r*_gpJaSPSYxa7~1592KR?{nv;D>MrXW2#zE$@z+w z!`&~SPjY4W>-e*zhcZY&B}ew^by~dldd4p-A?P2jJMAtc;Mf&>m;`FJ4*7>ftU>}( z8{kXEJUt?OAS@){M{U{{2^6J9Jo~ec+Z360ZlFu()tKp+5@g0Np-^D~#H8s#!A)zH)lHxI`7{b7z^39{;~%YxxU%ub zZRT4|_cU=Y=G{(LgOPw~c3({$AId4M9_(i}GsOchVZ@N4a0S^*(g)(@_z8eTDfTl# z*G;9{ggZj zy6-VM_w2PdGQy!7|>p{%& zKUc=s{UvC6+7*v&dNFOeSZVH>7|v;&$znY8W6pDP&koV6QEqJV2z<-Gs`ceF=FnJW z^>Y?h;WszlM3WDO0i)X5Pv!&bqxNr(+$rX#PCU?ms`?{FnWtj9_ZwcdhU1*@LSES& z-Beir1G~rM_&oQ&+7~-kel#%={4A&ZO1Pqhy}}WMZ)__4&b;MmeFaYKb6HAG_=Z?~LwCn)Z)rMm67 zj@CjdVv0-r7W(+y&Fol4RQ9j4^>3Jpj3(|n)lG3k&vI6owgpg;&&6E6!L3avX5PXC zl!k@V=h&ee=zHfl21VK!q;qEiKa|AxIuqa!I8D*U@bA)hCw+nqkk`wmP-C9Vv#dRS zN2qSongB_Pcc4(gGQSraO}}7?RmdvRood_eeZSM`uZ4|Vj3K@jj%(N94&>G+EZJ*Wvlly^-&4A z`24REhzc@wj#(JIm@&uxsZlTyAExK;0GbRMEn$;ev-Ku<^9_yFSyCrFN(}ct%c`}> z9Xx2U!w0q(r+%!y>g^+ptCXUWQUdX?P3+F!0-sVUpT+yW)GnKEB|rjpfa&{eYt4B{ z#S#^ejY|Cgs~*cTZ;Q$I(PLe#42`A3$1T3_J(NlA*m-Tst(belFBkCO|{{xu5*uU_9Rb4-hx_1^16Arm|CFo6d<66+Y|0(Do z;1%so<5q7D#C-UAP?99>XJe;o}N#C)}bEOq2>D{F?)y^v{exFcBlx%E7hSzJ|VU z-YfxcEn4O43G_8w;?Usq2)IZa4&v`4 z21EiDlrN0e#4I7FD{@`nMs}L*f_aop`}ebQ=^1!GdtYKB6}-m;=Q^V0`(4F*dG#4s zb}EC91YD0tXO`00qmuwgxO;cCczMDu)<8~bD2NH$YVS3Q(OTMW?v+tQ?R(=sV^t|4 zIh0ASxfgP$yKPRx42l}_)4Z^FACq~Mray@UOhq6xX>{C>fTLRQNj50K2!C8h8O9ZA z5%U5G2;vHOaHNsFiH8Ir5(g2Z*Vx1zP$VE1CCzKdt_pq30}&6-qG?f5AIJEn)EY!) zoC*nmbELxyt}%c^z({}sO69cXIoG)8zg~3er{M^q);p2iaNfadWEW#n6e~xKV!!~g zU)TsjHx3LAbsh49DQ~2R(w~jJp0nYZZTjkRJ0$ltrOKBa4FBxe2H9G+;k)ILC$)%V(KMZx?}4bv-0QGs{wUS(zLrAt)$%>>YGYLWh=QIwc@I9 zSQ3xBym`!tH9j(;b;?O!bs`9NG{X%&dZ!oQi(#M!OBEJyj!I>vRth9VD3HRQE1y; zli7!|Fnur8?Z=%gJ1k${d#BzUyRtb|Axvf?sE&>8YZ*-^$CzTnqOz@Kd8LQ4W+9Rb zXsdc9BZ~&lD@5lxLSq^DDa}tP@%XR9e>C`MqxQv6Y8Nv;B_p$%zx>-9euBwM6!v!< z$&KH5Cz->rQ+aIarA^@0IchFLnXsZ?tdd}joxvRu z_5MK_5kU+Qccb8t@HeOp)Q2)YP`32SS@R7DNPQkh$7qphIsp{=;r*7mxQbyzz6Hxz zK|Yx!V{FE|2n(vC!;$p6LLQ#;)*+u}RBCe9W0Tx2=H9VCwZ}}NG_OWafOIC@V9`Mn zDQN@Do&Ai1!T3-w;NRa228GbJ!`4!tYzZcAF1g7vTLtz=I!i^a4F)b-Gb?4}6l}+d zG2N6II`U;&M8s{6uW5Z4nA+j^KxjVm09fal5Rw2}E~{ncm8i!|+WEYmBXM30vCdW)W}ZSfP!R zT+A?1cYiEQW6f6$Js-(??&yu3nA=>uOmOqrak9}^l7Egs8!g0-@&O$XC* zh2(us)Ae0-d_m;Wm^kAu!gpYQC4m=e);3)5&x`7`mnpGy>0^&I2-HlClzvbO|_b6 zA@1c~WIV@RpafpbO1!b6Eg*<->}$fp_@v=E-$u!34|6+9CdHf9Y3@}U-5Py_QOMR^ zl}>ExG%qIOsomfhKM+b?@4(T}ZReEDaPuMW2BVAwWMontjm3pgPG~EY&_f+0zcp*H zmN@+i*3CH}d^CR7!7T_@_#zp%xIV~IPd+|Q#V!8^(?mb7yQ}u+IxkihT##20?Xrl}C&kKNw8{K)t{t)`|scDhdYC8zK- z=KC*WEX$eLR#J+PTYW08m2UOUyeyiicMIRrg8C=5pLT1Q=xr+)+&V}M7?pgj@EJ3L z!^$8e>~TWwlQ(_kw@N>K+PkJEN=VzY{cNtbiG_KRF#oB`hL= zQGsdnGWI`~d$BBqMuQFsIGaWZwx95Rf@R4_fCkgnUoVChTVv0t%dn3s_h>(4^DPr^ z(TI5^^BAfA@z)SDs2x-xGTjF!y++xLSxIt;-cokEg#QHHjG3tF5W#OrLjrCYm;18w zNLyyLV2qZ(j2M!_4B*WhyU4$HjnATHt!@oh4@IdI&8$3!y4a^qOA!*`9nX|5hX z@;=rN9xr-cE})niu_E8`cg@TD?p(2_^}Nk#>Hg=}IA`BS%utpxJG*~AB@XCBY558P zT9_>mQF6a`oc|gDrrERCPU#}u64_jdrjO)0vp<$u3Av*hb`o}A<1X_)n|Pb3T)69g znPu*qP2+5VQ`6NVQ5fhrFnw9`Fu9llP6lnF0TzibU1TzyYkIFV+cB-vp+-Z$THli! z5^xoGmQAE;jCP_TD& z6fE5{sp*H)syrEtI?-%utj$33txBgwjzRWhy7R0-WYT2R^pBg#R%|Q9%hpd&0@IMv zArLad;cdRy{o*sRN?O&2Z$o%6$h_HRTv|JEh>+BPKmI7c6mcN?p1iB;Vpb(wD~s#wl694QexZ<)ZF-e#Piv zN8ERbcM2@M9XfhAOp=RfE_8Ep)4yt8ei{plc*^~niK2-)=D2x^c`iA$d=BZ}+pYsk zC=LkOd?BnB;&|nACxpACd)kIJQ7wkB%1_2JHuP$${jw?;fdSNNc5T}Td%$4bw?<(S-Xqnw933*l( zO0>?jnw-5{8tJ*|x@~y-*)te%Gi(R}KbZ0Kn!f|LyBX&bNF#EU((i>*$5Loy_{}|e zZ^M}3kiaJLLX14>@nbeOG~KfgBCLCXLOxYt+ge^iC7n8PqodUi>TVPy5Szs3_Gn1^ z6J_Kq#7DA}LjDLw1yKIxi509u#DczyPapx7I@d5rrRb$_Bp^rmgrEkcVUmFM4%qyD zIqFzWv*%GY@vi-P8Y9I25X|UTFq#S6!Sl`eKux;bncwB|FW0S=v-(Bv$I~&;o5I?^ z-;Xoyf2Jak+o_Jl+e*=-A_OT~c!Osx46zz>vu@cku zR~8I@HM3Z$99RNP1a0mHAOV4zyd!mFh-R)onMG^U?KQfER++iyt-H$S+P$ceHxK{J za$S7`(sScl-A4j$9zQK~`jlQGlyrE81O#|C&#|rj!5e(0;i~7E-Kdc>?EQsgdJG*( zK`AC-)tsP0LU00Ve#`8SULv-}DI)DOkEhq` zcry$ak8m!km3QzMMD>i7W@w`O10sJ3666gfxa@?s%JeuIwe|T+ z$>18GgFDP10ckB~Qm6&!1eH>xA5K7T6-xXaj+rC{Dh{3ZQ9mOuvL0T`KaLgJEd*nT z$*%o9CL(>XM%Hz7)hdbQtp?3X~`7!VwojhZDmA|0rg6_|Wf(7~NYpUl2nL@{>wEMKG=bXtgsu-(0U1ON}q2 z^LpP9CC5)eHDKWDkBqn)m-co;FBzXIWF&BNMyYBBd`J+LeFN}RL(f|fUp9vIwg|!CEwOSqkCuX5zG|7~f zSz!mtH~#!~eEok5oWc_%K-zw%XMpuvJ)%Gkp=m5wMiF`;dj{&=1O6d`j;^s0;{SvP z{kF)lwz>2S2~gi3ZO?>Ld0YLIMB4rP_{|EJKfK+DFGpMEuj5Ex2=D!S@-ryix+bQ% zWSi+bhD_QOOE(~#Uz3+Q>w~|WqwJ`;M>joNC$)WfPsc3m)>ESLW)|8cg&rHh-tYa; z`KIf1=+7X$p=RdV+9$!mGB*&}Exd*oo`qs@(wQox*`=^@BKZ~1@%KRX& z5Jz!aqR%MNOYyNYGxNHOJOks6vFY}thn!bJH+Ca1vY?o_{b;Y8ALSs3JG*3dbswpE((*%TGm(pEaR)hBj z16oEv@Gdm`tUMZH*JuRUo?D~EHQ3tb&Je0QQM^;s%$*Y55Dk0p<1+<1vuK-JPe;I_ z%bH}6fIDYGvph(^j0(Yp&)VrRsy@``1%jA&_4w2h#oN=dyz;=8VIB)c{M9YpClA$! z2C*OK&Sd8Rzkh(-X#Ci1b~kp=!Sc4mZ>Cqe{gpZy>17=LEr*T1m%@B3j~^Fv=%z2_hgajQ1m@O*{NB(SMs9duTJA8;G|%f z3g&;1Q`q*+{6&Ep;d~W>J{WYIKVvd|IA2KCy$K>(?79KHK-`FbH8=jSCcw9dWPGl@2U{Pkhp3@^)i^N!8k!CFZ=Y4@oOln zU)&yO-2>z*Y8|WL<)4Pn%@C~j>NW+v(DsU=sDgTL1Z=4`iCP|8oT!9OzWY-urP=Ai zHww!_M#MY!6B()7uu;B_aTm^2!5Z1PT+#kJ$!8Vvnpr#5&XmTFwNSp4SC*#wjWI(w_jP4I0|9Jl%(tV$a*~F%@9WnGOTLT}E(vBl#m zZ!Tbb92lzq_{34NfTOW1Y5=zVTKDBe`C+n_VLc~>RhZzipk!ruz2;NA}!;3Bp}p1-qEnh|BQvC z(}+>Rb@8EL*KPf~rkQeLTeLY41wB8Ch#LjF5i0klSSuG12?Qgyll85W53q+g; zq-0$xj{-TY)a7xW#1Kr4yKM!=whG9R=KB)IeZaOG>}*XO{r+t__&Hw7aj!(UWhQx* zf1;L`4m}2~3P~3+N%{V=|Mt*`w`&qjxTS_RL(wqBHyGhu$@6?B$6S4Mr$`8woUq+I zuCo<=I4yrD9t-X*+eYGCr6<*ybWa7f(6nU9gs3mjEu@@|(o*U_@G~E7@1Gg%j&@1) zEnLYkFutSuY!orgftkZq#-`9P4j|`_+)!+vWt!XZmU}YkTq{{!$;9}+v`-;o&6cx1 zgs`=of#VBf2_u;tmmi;%CYtS#ri`*7CVXP*3gf@{R(Ht#gE+$4Je z8>77JAgk#jbqeRON%u$JO3a$QJ94``)+Tn1KSH^lB>MBQVT&ifDsJD;SLv(G;CjMZ zS#>nop5#{CTif2MAT@M{#h&j5q+Y}`J~7ZA|9L^KL5{fLROpzr6C6Sm-NGfP*d2Me z9JKMDIg(Ycvo8C^%z+1<>y;NQ}h2~}O(715tu&y<2iXol`>qXd7*6&Dh) zfdniK{j)fczpOCE>r?#c2t40hx-ZSzdFG1Z0v*=@-pAnmwSSb>QFUovJrbcg-DPW5 z4XY5Psz3tvkbod00B(HcDN0od{=2lAz!|Xp@^&@?zC!s=A#drxd&%1xy%VN=Q-|D^ z)P13P@?EZE<*DT|?6m;pe#2DlaWm)&r^4$8icglQV|3ChC2EsGtB$#H5eyD#G}L{S z38zDxz6XU_B}R22NWdE{E^icjb@{thNxGiI7&v4@A7Qfx(gF`d)&6_k{PCD$TRfvR(1WlB_Q7HLSPk;G)pWjl;Jzw2l=qFd!OC>O~MQyT`QS&a^ksWBqF-n{7 z^2O#$VAAWa|Rp)ehV#MTZ8#o-cHP@fNVIP8cfCw|mDi|QPr=oGOt3D1h3TqTaTTEY=q1mLq z`u(wP^8(zFw0IGR1bj&!;!-O+Dmx`=A$Io0=m53VgT_V9e}B9{r0*@C`DP;lkHL~H z^3Y-EjM}l_g7SDm|M?ZTtw{|D@N>5~6TrIYfpVr7lXy|MS}E3;1y5VYOsb)$w$ZV* z%^F$E>$rzdHyNiwe+0&bc zKsl?+s(VuJPLr9AQm+aRTWj+-Kpn>@iJgiB$iVf|Q7hE=e>Cv1RUSnxCujhijVMuG zH-;FRxevuIX>qlYD9>4@4?7(coIUt_`Ssp~lCP%dADz>zf>3&cR{l}hXc_rW)lEeA z^d1sW%zy;2mZc6$>e*y=!oJ>H!mo8YC^URXNi6FqA2`Hl7AIdfKVkGz=&f`PY}8mZ zZO}@}exD~`pQKPVVw;L;cy;&W?z5rfcNawXGwhdRM?-5WAcv}5?Gc^kO3|V#oYq-~ zxoa5Gh?fzg>EAMZ7cJ(>@}kAxTn43kwkCU+`DmZv*BJ=pwBA0ZC(C}Ag9ySny)Am| z!|VBHha>&;8_-Wajf*YUh^Nr(G$+vLmCK_%PLh{Gtl$C@Tnn;cxk1o9<(dUki?r$- zQ%F>9^t(>ALt|q&Bi)!wmBM~YDb_MN5qX2-817O%oS0hmGUeKD_Z1) z%Cu&EGi;!`fdhx_R(Ca}Ku(;{fh+oARuLJT&!O5&q7GPY??$5ck}d4+15DF0vsOYm z6?u7m@`O0RXl$G;aRg_|nY``Jv`skP$%aJ}c6>iU>!*>pDZKKaA|d9S2tPcblR3X? z=8Qk0$tYX*d}@z7hqy*Q#?`%;{_)8+CWCy4UPdRo*2M&XktVsV0`x(j%+h-yV;WW; zNy@BL{=(36{8=KoNp9r`_9b;B8Xmj1!LsB^$p!N*sg7&~qE>?N_5xQ8eKR)4fV;RH z3{^4r#!uehIAM^f(BtUPZt&49-o9f*ws^w6PvvpMn_FL(Mz3EAtl@a26?0P~mw5qj z(-ojzLxjbP1Pt?BV(CG*?_D1i9Gd6(5X~kIc-|R-Ug%l-gdT39e7TeU zNrh1FZ$l1~1Ue=Q9-u(!eGlx{Cu#c1R{8X&kN{0OBmiiC*`ZIir}WJejxkW87b|b8^DGtle!BvT(8+DC1zG8g+@3 z=3@17A{jRyJ4U3|?<-4j>=P!=%<;iGJ7dfqm}w}=Xg?I{$I*C=Bg71_#009y(QBO- zk%ys^b-}6i>mxIgIPX0Nets_TnoIjV#7eI^lTG|x5VZ$Z9N3!sVfpj}U7Hx2JqnJ_ z0IK>gv_*d;w0LY`a-hDKo*p$VT5bbjH@>TlVT3a3??gLtP25%wbO!2q6>Y>t;!Min zrrtSSI92ya$(T0GyR?5tatWNu_N1uf7~l`We*I#TL%5E+XoYg<& zb-Yt-aQ*x$tly4wiN$j-uH3UGAeG&YNBx2sM~O=^+TcPQrr%knRNa1t-9!Z$VHZ9a zrO+eh5oBPe^CZ-q5;Ss*l|PfnNO8lP%{G{B<>+}IyZ#hc+&1{4?XGH!bM|vWlA5!) z>^P6H4kcnH~@I=XTHM3PwlHA?>hO{i<3L$kvp8SJO!4u)d` zc#_y?XMO(dB;RMHWYw*=Agrtm0rLUstjsu`li#E&=bh^sdkAKT?QHLkIR-9SYPWN7 z(|u-keATU*WT+M&zd!}qjk;hB>Av^jgJY9$LMWp>!=s>F8>`_&^#UqPdIomaH;(Px zMXyk++PZ{@+A&b*ayav!>~^a6SsP!q?c&aQysoZP!s>+C`H}UB+T)wj$>D<+={7t3 z(~r0+vt+DP)q)KpW&&MQVUR$ETvHqsHvKn}l+3FXIE|JWtaxN=l6$a=A`pe);`9W% zup5kbn2_4z{ajQ?e8m(?k4@c2M*WYq5E@Azvgs#$kxz)@Op5~8rM0t50E~5d<{TVt zQK!6^Y=feF!dZTM6N7eXpqHYomp@hqjm4FHyieAbPE28CtxDd<`L|bjx|V#eGug4T z!q}`1|GIHS*N4)#M>`tI4FbQ1f@VF$PBP4 z+6DZXOIy(YQdeP6YTW~_opQA}xJ!PGFIjj9{&~^ftI6CInNL^KG(R(xQ9r90EOzag zVA<)qNTNS?2D>dLxa9 zJcIB1YvdADXK!ctxAP6vKOq9ULJyNpzg-w%t`)lfgh|wLn)=yPxFF*jKBWK>V7QeZ zX0Zsp*fkf{PI!A1aoPGaciG&+Wvz8tDCfh>3m|NfW_am-LBv~|f^M^ndmX}hwvXF){7 zZ-4Tk&_7cM-8tW=w1hMMwu1y}Z+5}f68=F8Sxi(-|lN6fRBqBd=eyu0C-gU{xmHj(+PKtjE z6`|VY0fjiur3acKBib#k$Aoy?F_PUK-{6(QSD1P0bmNP2(@lXS!|0=<+{Af<`)V=7 zo5U}N{vCbkqztDAmAe!nhPbaUH8TdiS|k0}n_p0&EWdGgtKi=mzh6`RyW`I;2=D6y z7FiZBp>G2xbTIwu0R;2ckLgq0MU>~mPn8!~^NUQgK-a;4t?O5qXpF!4k5pg5HA1cy zp}RPh4-p?@uL==Pnk{9=YBYzzNI)}dZ_0(M%t6t=e9`)!K)K8CDPQY%8Ww(Fh5pOx ztFedwt7~xHSftS%{{F0J_g%V(54Onh;dFlVWOP{Tx!~b=Dp%?tY2*d-8A9@57;{Pf z*FPgF(afD4FR6?~uez=OP5H=lQ=!%}uez@GnG2(I>uZ-p9;; z;N)y#e~lQ?!u5+6TWPKy z8`=N7^pwMkx4HL>CdCx-6KMoBhvmK3M6xXM9jf*N%o)8?`r&UD?UF*g;x)9rhu;Fu zGk(a5k|1P9CcogJqyA#cZvUMv+x^a#{{XkfB~tmtmS6tCmM4B=%h#fUso=ldYDRTz zxyMB095Q5fn+97y@O*1dFTS?F*B&=FDNadCY^8d8)!+>hkhQ8f-*y8DsHR64`aJ3) z!uot22bmC1}~cM)OY9|v@$w**`q`SAMus$xu}LIfJeC-FVO-yAv` z`c;L`TUAW;8}MFOayr@AG(enMJl^w%4f-3AWBvq_#!~hEXEwP?%u!Pw_ckH3L^m)t zJ$EmYkbvIwyy%im=#DWG@W3}YeMn49w5}8hXaHY@URYktH6fOffCHv=2>kzO!tD@< zt)d(ALD9wh6{bjrQ(c1SkktPM;6hEU*+a#C#5F8+no~;Ktha^jOVm*05+cHqvmqx$ zNo=NVrut@4J#dC=v358qMDd zFm9&c*G9`(E@NAb7W`nN98LFOx%v#eJdH*IM(fTs$IgiCwbgqx{41p?wTEmTKJ-{+ zF&ZD9&+Mc5z(nJ_j|5yKA_1tMdlwCLM-j0CIKj}kg<=#@Q8Yx>+Wp+;8vE>Z$ckLg zd@~r43q=th!_w{?2nkS|U)I*%l|H)YE|V1f4;4!)Jeih8<;|@t0rB;5$~gk%H^AhL zMV41dgxCo8G<5UKdGe1UXlD2)Cl&3 ztF!E?Xpy=Z!3NKjJI)5fyO4k%$|ExW>R-9ck%|-s>TA8bIS7MJX{76+c+iY7LK4oq zZoSylPe5(e`(E#(xZk)Sp!oDWtm0@WEJ%sRD62|#EC30>tv4dfx56GcdFAp|e~b+Y z@Z&^LE@&RLj{t^9z-w@;XmUCCMmBX}+2)g$yLDTf7!yfdzCqwU%=uHM%cb?~h1<{_ z<N%$(0*YOx(1#j@}>}g<6pMtqzttPmzr*!P!wH(p4Rq| zoh{Ebkh_>ttfa1GRSnp^{xWEc8xlnwmya8xDn(4V_n#H72UNT>JN!mr?umz14c-*| zm@ShWvKtaPL<~i;a=^ZdoO}I(ZG|%7XYayk1@Rui!mAzwtw?UYsg|AcZot8l*NE7G zc?Pn*a1fGi6u-2pGv?r$Djg#Ka8nH1BSc!Q{1%WN>cCQ{e8V9pjYFA4w&JPlS+HmP zSA7o$dQ9NnSs}gfp6s%i(<$EII>lSlud!I+4@=bRJux>nLP_sY6_}sS>R5#kCb$3| z#irhg;fYbQ9$C{5H)#Cq+wYR(oaFZ60iWFjngxJ$D`mY-Hw-hkT!tQpxo?rY< z%I5zvkyr3LHjyE=JQ}`Q6F<-yK}v=C*|bj z6zt|ylZ>5N4j*ydCmi)6`Nt(pG?bBs@6__oqTik|g_jOEM(!+g&ip8o*H@u7h#SFW znNSmV&WFfojva?$8YV`)HYBzbTdT`nBV{x1roo>2p72Ucl9X^@3y{lf+C&SHyjRb# zM6~R&&`R1|k}zu%vjX{$mh6APM}Xii^YsEea{YusfrI|^PXUFMpeo=sdt9_Duh8K5 zqkw?6$-Yj1?j`>-b9R?f{(dVd$ZVsuAs ziWZ_H_w7vApuL3hHa+sEX8dBOw(1_AV_oVU z+T~7K+nKUC(VC9Tz5C;<8~G@avJOn8I&}*VFfvf)q^dt5GsEH@;vOzKG5%nm1@cbg z10LNgpA5h0P!$zs7Tj!d+2s{WJhytsQ`rZTorz80`!hs9^Vdd^g3l`RcWxgIk+5;P z^2R6RE;#!sTlf?@(TAT^1-Z9w09W=O@>QHa9UmgYj?;^pj%Fu`8RpAmv%(%Lj>E8L zW7%41v(NHHw_CDEn|+fb^3n=d0&SHS5R^oq^-;(1saY|Li6rx?;9M#&#IN4??JWT> z`&=4@qe_JVh7!*wmBhIcYB)FJ0w(ES-~uLwljMa3KRp+h&>96Xf4Yp{!5=f+m0(r?k59`keX;$*jvZT0YzU?O++Xcsnj2m-w=d0@IX zoHs4v^cKff9*3itQ4Hvxle4ZREyIfbHhox`xGR31pc@vtJC!Zs;Q!XX`WqpAM;*%q z;}f+Q&tXLDBptm;=^{3 zc~{+{@!RkV%UbplRc0PKZ z8=laWm;C;~K#N*IEchUGz3(YKaDe|S8tM8DymLnIlR|oXnfObuDtZa|AGF2!WiA}( zQy#>a?9}7mu>=bTbEp&RE89BN?H%$TZzs;WBv~pJbjv;dV#($)F}m7)LnIT89)t3H zY`e2r#1{5&k{S^zYt4@* zh|R_I&vKZO{w(|5>@-gHmM8x@MF{Xbr6^UzA)I}B8Q2~-8dw~kD=$w8{E@J(s{EF$ z&~Xm9?qb=TB6+-jkFj^_L!y@+<;y!!HC?iHPm*%EGI41cN-W4nZ#u|Y1aRQf!A62O(`w|^1}f?W^~$}fY9w{ngaXMbx~i%iVe!(zEvmAuvJ+I8m^ijo zMded*4r-6^=7=!0qTU1Zl!^CNarJ<4l8Id7ZzJ;#A5Tpyb|$b)XeHDK`eS6~wG+%a zL-bP@CKRV9U3qVswy;dv)%qp0uF2{L;*OTk-bwQU2xE%>IFf7Q(#zMEYUnooqQOWo z(kExuDX&gbB-qCr7-huNoi8VMYb5pj8brSd85fsUOC1wm{~Ou}ooeDb5~^S*W-m^iwM<3^)W+E@7t@do3aR5!~7QJR(}lg*44r}`NkD?H&~mFn8yX*~28 zjhZ+)EHtL>kGr!}4zoL^k9U|8>r&zxZmThrQ2Dly*?x-Ss|!vc$NMgJI6&z!1jTnu zXy<5OTOQyo*b<~Dc_Y}{=Mi#?&Ai>8k6)II=mrjxyn#fAqv$irIOWKa{8&XNfJG-s%QkrNmFP(AC3;8`P;@J|WL^nTYmHE6|Tnb{cR<6dM(I zhMTlqWXZZ`$FRFr{I;+boYg_2ZI&xVVCA8dxR{WgSRwGvTgZj>Lz9--e+NEvX zAJJxD?sv?MXAjAEONQ@huds#l^|a<^lk)uk9BuV2d2Rq*ah0&}uLj{&325`{YdWy(nQg>bxl z)D;V>$%>&YEeo4mFHkL{AQ>b5M!+8}|{qOc-TVgq_d_-7JJ`~mUoBt4r%U4>{@{f%S75W>VJrRRQ0qt1TG zNgZoW{&ch3ZuiTVAKR8Zo^~AJ7|4Y6jn}4nkaZNv&*=Flyen8qJqyn+rG)lMhQ~41 zFT~Z&FvnH9ELzEHj3mpB4#^YnQkBQ9#*lkTHO!X^+nX+kQnd)08uY5!xlsft()A=q z^`(fbe0LxUVzG;e!$VgR9Mvq12ZjfkJe+V28zf1246E^J>52UG=D<0nNkO=gz+xj@|_p#E#*k!F3E-5Ag0rkmPrtZdJCizmZ0AI zLdM;J9d!@;gVKWs2~{7n;Zc?uWyY_;_S7sB>D@0z4N960P-0pa8PD>f_ z?u)v^n(`Zr+U17~M^!jNjuT`&sR}#<(_t@@fEktm=}wWo_&kO&@_QPT~)pbzX-D_PLjR8sP!hO``7)2=zHk6ftka5IUqr zmfy-??Pnr$u5&Io@_%VK3A+^~tugI&QXfI_ii>|K)Uk%c(?CguP<~G>COec}yUtrlheqFMTCsQuVB+TeDr>;>q+o z;}i~iC<8wsK~RZgoL_w4`f^0nVZsbYo2wE<1In};mD}9mg!Fi>-*(_9g3YQ?-~h#O zW*qfoJfjl{9sx?c zrAF`x=(=r5Ddg*wR%VB%mVU@eR;=NLP`jy)1QzpAjTSs#62xlFd-UY)!u{f}5%uc1 z9q*{lN+c5mCv*e*eI!VwoR^pT3}8s4uCei5C5_LFG7MrkuRAE35Wh16;`= z!-|#*!KUf8pHl=o{e~iGo^wcK~NUkpV$&zqN*6zsc9~k(7<$*W7cDte2~V z_%1cHOSx4EUqxi9-Iw_ILeIfZ!><@uj51=$TjN37x6_-Gfqak0y48mmW#UE-nShr8 bL+t^$8>%2v=^7IZo85up{}w6_LQeiachriZ literal 0 Hc-jL100001 diff --git a/test/testfile.pdf b/test/testfile.pdf new file mode 100644 index 0000000000000000000000000000000000000000..433577179840d3d626229e113e7471fd44b7e638 GIT binary patch literal 279746 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°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 literal 0 Hc-jL100001 diff --git a/test/testfile.ps b/test/testfile.ps new file mode 100644 index 0000000000..d02503a1b5 --- /dev/null +++ b/test/testfile.ps @@ -0,0 +1,598 @@ +%!PS-Adobe-3.0 +%%BoundingBox: 0 0 612 792 +%%Pages: 1 +%%LanguageLevel: 1 +%%DocumentData: Clean7Bit +%%DocumentSuppliedResources: procset testprint/1.3 +%%DocumentNeededResources: font Helvetica Helvetica-Bold Times-Roman +%%Creator: Michael Sweet, Apple Inc. +%%CreationDate: D:20070606214000+0500 +%%Title: Test Page +%%EndComments +%%BeginProlog +%%BeginResource procset testprint 1.3 0 +% +% PostScript test page for CUPS. +% +% Copyright 2007-2011 Apple Inc. +% Copyright 1993-2007 Easy Software Products +% +% These coded instructions, statements, and computer programs are the +% property of Apple Inc. and are protected by Federal copyright law. +% Distribution and use rights are outlined in the file "LICENSE.txt" +% which is included with the CUPS source distribution. +% +/SEXTANT { % Draw a color wheel sextant... + % (name) white radius r g b SEXTANT - + % Loop through 100 shades... + 0 0.010101 0.98 { + % Set the color... + dup 0.75 le { % Get "white" value + % Start from black + dup 0.75 div % val2 = val / 0.75 + + 0 index 5 index mul % R = R * val2 + 1 index 5 index mul % G = G * val2 + 2 index 5 index mul % B = B * val2 + + 4 -1 roll pop % Discard val2 + } { + % Fade to white + dup neg 1 add 4 mul % val2 = (1 - val) * 4 + + 0 index 5 index mul % R = R * val2 + 1 index neg 1 add add % + (1 - val2) + 1 index 5 index mul % G = G * val2 + 2 index neg 1 add add % + (1 - val2) + 2 index 5 index mul % B = B * val2 + 3 index neg 1 add add % + (1 - val2) + + 4 -1 roll pop % Discard val2 + } ifelse + setrgbcolor % Set the color... + + % Draw the polygon... + newpath % Start a new path... + dup 5 index mul % r1 = radius * val + 0 0 3 -1 roll 0 60 arc % Draw the inner arc + + dup 0.010101 add 5 index mul% r2 = (radius + 0.010101) * val + 0 0 3 -1 roll 60 0 arcn % Draw the outer arc + + closepath % Close the path + fill % Fill it... + + pop % Pop value... + } for + + % Draw a line around the polygons... + pop pop pop dup % Pop R, G, B, start + 0 setgray % Black + newpath + 0 0 moveto % Center + 0 0 3 -1 roll 0 60 arc % Arc around octant + closepath % Back to center + stroke % Stroke it... + + % Draw the label... + dup % Save radius + dup 30 cos mul % X = radius * cos(30) + exch 30 sin mul % Y = radius * sin(30) + moveto % Position label + + gsave + 30 rotate % Rotate label + dup 0.05 mul % Offset to the right + exch -0.05 mul % and down... + rmoveto % Offset label + show % Show label + grestore +} bind def +/CENTER { % Draw centered text + % (name) CENTER - + dup stringwidth pop % Get the width of the string + 0.5 mul neg 0 rmoveto % Shift left 1/2 of the distance + show % Show the string +} bind def +/RIGHT { % Draw right-justified text + % (name) RIGHT - + dup stringwidth pop % Get the width of the string + neg 0 rmoveto % Shift left the entire distance + show % Show the string +} bind def +/NUMBER { % Draw a number + % power n NUMBER - + 1 index 1 eq { % power == 1? + round cvi exch pop % Convert "n" to integer + } { + 1 index mul round exch div % Truncate extra decimal places + } ifelse + 100 string cvs show % Convert to a string and show it... +} bind def +/CUPSLOGO { % Draw the CUPS logo + % height CUPSLOGO + % Start with a big C... + /Helvetica findfont 1 index scalefont setfont + 0 setgray + 0 0 moveto + (C) show + + % Then "UNIX Printing System" much smaller... + /Helvetica-Bold findfont 1 index 9 div scalefont setfont + 0.25 mul + dup dup 2.0 mul moveto + (UNIX) show + dup dup 1.6 mul moveto + (Printing) show + dup 1.2 mul moveto + (System) show +} bind def +%%EndResource +%%EndProlog +%%Page: 1 1 +gsave + + % Determine the imageable area and device resolution... + initclip newpath clippath pathbbox % Get bounding rectangle + 72 div /pageTop exch def % Get top margin in inches + 72 div /pageRight exch def % Get right margin in inches + 72 div /pageBottom exch def % Get bottom margin in inches + 72 div /pageLeft exch def % Get left margin in inches + + 4 setlinewidth % Draw wide lines + 0 setgray closepath stroke % Draw a clipping rectangle + + /pageWidth pageRight pageLeft sub def % pageWidth = pageRight - pageLeft + /pageHeight pageTop pageBottom sub def% pageHeight = pageTop - pageBottom + + 72 72 dtransform % Get device resolution per inch + /yResolution exch abs def % yResolution = abs(yres) + /xResolution exch abs def % xResolution = abs(xres) + + % Figure out the sizes of things... + /wheelSize % size of wheels + pageWidth pageHeight lt + { pageWidth 9 mul } + { pageHeight 7 mul } + ifelse def + + % Create fonts... + /bigFont /Helvetica-Bold findfont % bigFont = Helvetica-Bold + pageHeight 3 mul scalefont def % size = pageHeight * 3 (nominally 33) + + /mediumFont /Helvetica findfont % mediumFont = Helvetica + pageHeight 1.5 mul scalefont def % size = pageHeight * 1.5 (nominally 16.5) + + /smallFont /Times-Roman findfont % smallFont = Times-Roman + pageHeight scalefont def % size = pageHeight (nominally 11) + + % Draw rulers along the edges... + /CENTIMETER 72 2.54 div def + /MILLIMETER 72 25.4 div def + + /Times-Roman findfont % Font for ruler numbers + 11 scalefont setfont % 11 points + + gsave % Left side inches + pageLeft 72 mul 0 translate % Offset left edge + + 1 setlinewidth % Draw normal lines + 72 72 pageTop 72 mul { % Height inches + dup dup + 0 exch moveto 24 0 rlineto stroke % Draw tic mark + 24 exch pageHeight sub moveto % Draw number + 72 div cvi 10 string cvs RIGHT + } for + + 0.5 setlinewidth % Draw thin lines + 18 18 pageTop 72 mul { % 1/4 inches + 0 exch moveto 15 0 rlineto stroke % Draw tic mark + } for + + 9 9 pageTop 72 mul { % 1/8 inches + 0 exch moveto 6 0 rlineto stroke % Draw tic mark + } for + grestore + + gsave % Bottom inches + 0 pageBottom 72 mul translate % Offset bottom edge + + 1 setlinewidth % Draw normal lines + 72 72 pageRight 72 mul { % Width inches + dup dup + 0 moveto 0 24 rlineto stroke % Draw tic mark + 3 add 27 pageHeight sub moveto % Draw number + 72 div cvi 10 string cvs show + } for + + 0.5 setlinewidth % Draw thin lines + 18 18 pageRight 72 mul { % 1/4 inches + 0 moveto 0 15 rlineto stroke % Draw tic mark + } for + + 9 9 pageRight 72 mul { % 1/8 inches + 0 moveto 0 6 rlineto stroke % Draw tic mark + } for + grestore + + gsave % Right side centimeters + pageRight 72 mul 0 translate % Offset right edge + + 1 setlinewidth % Draw normal lines + CENTIMETER CENTIMETER + pageTop 72 mul { % Height centimeters + 0 exch moveto -24 0 rlineto stroke% Draw tic mark + } for + 1 1 pageTop 2.54 mul { % Height labels + dup + -24 exch CENTIMETER mul + pageHeight sub moveto % Draw number + cvi 10 string cvs show + } for + + 0.5 setlinewidth % Draw thin lines + 0 0.5 CENTIMETER mul + pageTop 72 mul { % 1/2 centimeters + 0 exch moveto -15 0 rlineto stroke% Draw tic mark + } for + 0 MILLIMETER pageTop 72 mul { % Millimeters + 0 exch moveto -6 0 rlineto stroke % Draw tic mark + } for + grestore + + gsave % Top centimeters + 0 pageTop 72 mul translate % Offset top edge + + 1 setlinewidth % Draw normal lines + CENTIMETER CENTIMETER + pageRight 72 mul { % Width centimeters + 0 moveto 0 -24 rlineto stroke % Draw tic mark + } for + 1 1 pageRight 2.54 mul { % Width labels + dup + CENTIMETER mul 3 add -24 moveto % Draw number + cvi 10 string cvs show + } for + + 0.5 setlinewidth % Draw thin lines + 0 0.5 CENTIMETER mul + pageRight 72 mul { % 1/2 centimeters + 0 moveto 0 -15 rlineto stroke % Draw tic mark + } for + 0 MILLIMETER pageRight 72 mul { % Millimeters + 0 moveto 0 -6 rlineto stroke % Draw tic mark + } for + grestore + + % Offset page to account for lower-left margin... + pageLeft 72 mul + pageBottom 72 mul + translate + + % Set text font and color... + mediumFont setfont % Font + 0 setgray % Color + 1 setlinewidth % Draw normal lines + + % Draw the color wheel... + gsave + % Position the wheel on the left side... + pageWidth 18 mul % x = pageWidth * 1/4 * 72 + pageHeight 54 mul % y = pageHeight * 3/4 * 72 + translate + + % Size the wheel... + wheelSize + + % Draw the colors... + dup (C) 3 -1 roll 0 1 1 SEXTANT 60 rotate + dup (M) 3 -1 roll 1 0 1 SEXTANT 60 rotate + dup (Y) 3 -1 roll 1 1 0 SEXTANT 60 rotate + dup (R) 3 -1 roll 1 0 0 SEXTANT 60 rotate + dup (G) 3 -1 roll 0 1 0 SEXTANT 60 rotate + dup (B) 3 -1 roll 0 0 1 SEXTANT 60 rotate + + pop + grestore + + % Label the color wheel... + pageWidth 18 mul % x = pageWidth * 1/4 * 72 + pageHeight 43 mul % y = pageHeight * 19/32 * 72 + moveto % Position the text + (Color Wheel) CENTER % Show the text centered + + % Draw the gray ramp... + gsave + % Position the gray ramp in the center... + pageWidth 36 mul % x = pageWidth * 1/2 * 72 + pageHeight 54 mul % y = pageHeight * 3/4 * 72 + wheelSize sub % - wheelSize + translate + + % Loop through 100 shades... + 0 0.010101 0.98 { + % Set the color... + dup setgray % Set the grayscale... + + % Draw the polygon... + newpath % Start a new path... + + wheelSize -0.2 mul % X = -wheelSize / 5 + 1 index 2 mul wheelSize mul % Y = val * 2 * wheelSize + moveto % Move there... + + wheelSize 0.4 mul 0 rlineto % Right side... + + wheelSize 0.2 mul % X = wheelSize / 5 + 1 index 0.010101 add 2 mul wheelSize mul + % Y = (val + 0.010101) * 2 * wheelSize + lineto % Move there... + + wheelSize -0.4 mul 0 rlineto % Left side... + + closepath % Close the path + fill % Fill it... + + pop % Pop value... + } for + + 0 setgray % Black + + newpath % Start a new path + wheelSize -0.2 mul 0 moveto % Bottom left + wheelSize 0.4 mul 0 rlineto % Bottom right + 0 wheelSize 2 mul rlineto % Upper right + wheelSize -0.4 mul 0 rlineto % Upper left + closepath % Close the path + stroke % Stroke it... + + 0 wheelSize -0.2 mul moveto % Center bottom for label + (K) CENTER % Center K at bottom + + 0 wheelSize 2.05 mul moveto % Center top for label + (W) CENTER % Center W at top + grestore + + % Label the gray ramp... + pageWidth 36 mul % x = pageWidth * 1/2 * 72 + pageHeight 43 mul % y = pageHeight * 19/32 * 72 + moveto % Position the text + (Gray Ramp) CENTER % Show the text centered + + + % Draw radial lines... + gsave + 0 setlinewidth % 1 pixel lines + + % Position the lines on the left side... + pageWidth 54 mul % x = pageWidth * 3/4 * 72 + pageHeight 54 mul % y = pageHeight * 3/4 * 72 + translate + + % Size the wheel... + wheelSize + + % Loop at 1 degree increments + 0 1 359 { + pop % Discard angle - not used + 0 0 moveto % Start line at the center + dup 0 lineto % Draw to the radius + 1 rotate % Rotate 1 degree + } for + + pop % Discard radius - not needed anymore + stroke % Draw lines... + + grestore + + % Label the lines... + pageWidth 54 mul % x = pageWidth * 3/4 * 72 + pageHeight 43 mul % y = pageHeight * 19/32 * 72 + moveto % Position the text + (1 Degree Radial Lines) CENTER % Show the text centered + + % Imageable area... + pageHeight 15 mul % Height of imageable area + + pageWidth 4.5 mul % x = pageWidth * 1/16 * 72 + pageHeight 35.5 mul % y = pageHeight * 1/2 * 72 + 2 index sub % y -= height + pageWidth 28 mul % width = pageWidth * 1/4 * 72 + 3 index % height + 0.5 setgray rectfill % Draw a shadow + + pageWidth 4 mul % x = pageWidth * 1/16 * 72 + pageHeight 36 mul % y = pageHeight * 1/2 * 72 + 2 index sub % y -= height + pageWidth 28 mul % width = pageWidth * 3/8 * 72 + 3 index % height + 4 copy 1 setgray rectfill % Clear the box to white + 0 setgray rectstroke % Draw a black box around it... + + pop % Discard height + + % Label the imageable area... + pageWidth 4 mul % x = pageWidth * 1/16 * 72 + pageHeight 37 mul % y = pageHeight * 1/2 * 72 + moveto % Position the text + mediumFont setfont % Font + (Imageable Area) show % Show the text + + smallFont setfont % Font + pageWidth 14 mul % x = pageWidth * 3/16 * 72 + pageHeight 36 mul % y = pageWidth * 1/2 * 72 + pageHeight -2 mul add % y -= 2 * smallFont height + + % Page Size inches + 2 copy moveto % Move to x & y + (Page Size: ) RIGHT % Label + 100 pageWidth NUMBER % pageWidth + (x) show % "x" + 100 pageHeight NUMBER % pageHeight + (in) show % "in" + + % Page Size millimeters + pageHeight sub % Move down... + + 2 copy moveto % Move to x & y + 10 pageWidth 25.4 mul NUMBER % pageWidth + (x) show % "x" + 10 pageHeight 25.4 mul NUMBER % pageHeight + (mm) show % "mm" + + % Lower-left inches + pageHeight 2 mul sub % Move down... + + 2 copy moveto % Move to x & y + (Lower-Left: ) RIGHT % Label + 100 pageLeft NUMBER % pageLeft + (x) show % "x" + 100 pageBottom NUMBER % pageBottom + (in) show % "in" + + % Lower-left millimeters + pageHeight sub % Move down... + + 2 copy moveto % Move to x & y + 10 pageLeft 25.4 mul NUMBER % pageLeft + (x) show % "x" + 10 pageBottom 25.4 mul NUMBER % pageBottom + (mm) show % "mm" + + % Upper-right inches + pageHeight 2 mul sub % Move down... + + 2 copy moveto % Move to x & y + (Upper-Right: ) RIGHT % Label + 100 pageRight NUMBER % pageRight + (x) show % "x" + 100 pageTop NUMBER % pageTop + (in) show % "in" + + % Upper-right millimeters + pageHeight sub % Move down... + + 2 copy moveto % Move to x & y + 10 pageRight 25.4 mul NUMBER % pageRight + (x) show % "x" + 10 pageTop 25.4 mul NUMBER % pageTop + (mm) show % "mm" + + % Resolution dots-per-inch + pageHeight 2 mul sub % Move down... + + 2 copy moveto % Move to x & y + (Resolution: ) RIGHT % Label + 1 xResolution NUMBER % xResolution + (x) show % "x" + 1 yResolution NUMBER % yResolution + (dpi) show % "dpi" + + % Resolution dots-per-meter + pageHeight sub % Move down... + + moveto % Move to x & y + 1 xResolution 39.27 mul NUMBER % xResolution + (x) show % "x" + 1 yResolution 39.27 mul NUMBER % yResolution + (dpm) show % "dpm" + + % Interpreter Information... + pageHeight 15 mul % Height of interpreter information + + pageWidth 40.5 mul % x = pageWidth * 9/16 * 72 + pageHeight 35.5 mul % y = pageHeight * 1/2 * 72 + 2 index sub % y -= height + pageWidth 28 mul % width = pageWidth * 1/4 * 72 + 3 index % height + 0.5 setgray rectfill % Draw a shadow + + pageWidth 40 mul % x = pageWidth * 9/16 * 72 + pageHeight 36 mul % y = pageHeight * 1/2 * 72 + 2 index sub % y -= height + pageWidth 28 mul % width = pageWidth * 3/8 * 72 + 3 index % height + 4 copy 1 setgray rectfill % Clear the box to white + 0 setgray rectstroke % Draw a black box around it... + + pop % Discard height + + % Label the interpreter info... + pageWidth 40 mul % x = pageWidth * 9/16 * 72 + pageHeight 37 mul % y = pageHeight * 1/2 * 72 + moveto % Position the text + mediumFont setfont % Font + (Interpreter Information) show % Show the text + + smallFont setfont % Font + pageWidth 49 mul % x = pageWidth * 11/16 * 72 + pageHeight 36 mul % y = pageWidth * 1/2 * 72 + pageHeight 2 mul sub % y -= 2 * smallFont height + + % Language level + 2 copy moveto % Move to x & y + (PostScript: ) RIGHT % Label + (Level ) show % "Level " + 1 languagelevel NUMBER % Language level + + % Version + pageHeight 2 mul sub % Move down... + 2 copy moveto % Move to x & y + (Version: ) RIGHT % Label + version show % Version + ( \() show % " (" + 1 revision NUMBER % Revision + (\)) show % ")" + + % Product + pageHeight 2 mul sub % Move down... + 2 copy moveto % Move to x & y + (Product: ) RIGHT % Label + product show % Product name + + % Serial Number + pageHeight 2 mul sub % Move down... + moveto % Move to x & y + (Serial #: ) RIGHT % Label + 1 serialnumber NUMBER % S/N + + % Draw the label at the top... + pageWidth 36 mul % Center of page + pageHeight 66 mul % Top of page (11/12ths) + moveto % Position text + bigFont setfont % Font + (Printer Test Page) CENTER % Show text centered + + % Draw the copyright notice at the bottom... + pageWidth 17 mul % Center of page + pageHeight 10 mul % Bottom of page + moveto % Position text + (Printed Using CUPS v1.3.x) show + + pageWidth 17 mul % Left side of page + pageHeight 8 mul % Move down... + 2 copy moveto % Position text + smallFont setfont % Font + (Copyright 2007 Apple Inc., All Rights Reserved. CUPS and the CUPS logo are the trademark) show + pageHeight 2 add sub % Move down... + 2 copy moveto % Position text + (property of Apple Inc., 1 Infinite Loop, Cupertino, CA 95014, USA.) show + pageHeight 2 mul 4 add sub % Move down... + moveto % Position text + (Need help? Contact your operating system vendor or visit "http://www.cups.org/".) show + + % Then the CUPS logo.... + gsave + pageWidth 4 mul + pageHeight 4 mul + translate + pageWidth 15 mul CUPSLOGO + grestore + +% Show the page... +grestore +showpage +% +% End of "$Id: testfile.ps 9771 2011-05-12 05:21:56Z mike $". +% +%%EOF diff --git a/test/testfile.txt b/test/testfile.txt new file mode 100644 index 0000000000..46bbf08483 --- /dev/null +++ b/test/testfile.txt @@ -0,0 +1,60 @@ +All work and no play makes Johhny a dull boy. All work and no +play makes Johhny a dull boy. All work and no play makes Johhny +a dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. All +work and no play makes Johhny a dull boy. All work and no play +makes Johhny a dull boy. All work and no play makes Johhny a +dull boy. All work and no play makes Johhny a dull boy. diff --git a/test/testhp.ppd b/test/testhp.ppd new file mode 100644 index 0000000000..bc32a12883 --- /dev/null +++ b/test/testhp.ppd @@ -0,0 +1,187 @@ +*PPD-Adobe: "4.3" +*% +*% "$Id$" +*% +*% Test HP PPD file for CUPS. +*% +*% Copyright 2007-2011 by Apple Inc. +*% Copyright 1997-2005 by Easy Software Products. +*% +*% These coded instructions, statements, and computer programs are the +*% property of Apple Inc. and are protected by Federal copyright +*% law. Distribution and use rights are outlined in the file "LICENSE.txt" +*% which should have been included with this file. If this file is +*% file is missing or damaged, see the license at "http://www.cups.org/". +*% +*FormatVersion: "4.3" +*FileVersion: "1.1" +*LanguageVersion: English +*LanguageEncoding: ISOLatin1 +*PCFileName: "TESTHP.PPD" +*Manufacturer: "ESP" +*Product: "(CUPS v1.5)" +*cupsVersion: 1.1 +*cupsManualCopies: True +*cupsFilter: "application/vnd.cups-raster 50 rastertohp" +*cupsFilter2: "application/vnd.cups-raster application/vnd.hp-pcl 50 rastertohp" +*ModelName: "Test HP Printer" +*ShortNickName: "Test HP Printer" +*NickName: "Test HP Printer CUPS v1.1" +*PSVersion: "(3010.000) 550" +*LanguageLevel: "3" +*ColorDevice: True +*DefaultColorSpace: RGB +*FileSystem: False +*Throughput: "1" +*LandscapeOrientation: Plus90 +*VariablePaperSize: False +*TTRasterizer: Type42 + +*UIConstraints: *PageSize Executive *InputSlot Envelope +*UIConstraints: *PageSize Letter *InputSlot Envelope +*UIConstraints: *PageSize Legal *InputSlot Envelope +*UIConstraints: *PageSize Tabloid *InputSlot Envelope +*UIConstraints: *PageSize A3 *InputSlot Envelope +*UIConstraints: *PageSize A4 *InputSlot Envelope +*UIConstraints: *PageSize A5 *InputSlot Envelope +*UIConstraints: *PageSize B5 *InputSlot Envelope +*UIConstraints: *Resolution 600dpi *ColorModel CMYK + +*OpenUI *PageSize/Media Size: PickOne +*OrderDependency: 10 AnySetup *PageSize +*DefaultPageSize: Letter +*PageSize Letter/US Letter: "<>setpagedevice" +*PageSize Legal/US Legal: "<>setpagedevice" +*PageSize Executive/US Executive: "<>setpagedevice" +*PageSize Tabloid/US Tabloid: "<>setpagedevice" +*PageSize A3/A3: "<>setpagedevice" +*PageSize A4/A4: "<>setpagedevice" +*PageSize A5/A5: "<>setpagedevice" +*PageSize B5/B5 (JIS): "<>setpagedevice" +*PageSize EnvISOB5/Envelope B5: "<>setpagedevice" +*PageSize Env10/Envelope #10: "<>setpagedevice" +*PageSize EnvC5/Envelope C5: "<>setpagedevice" +*PageSize EnvDL/Envelope DL: "<>setpagedevice" +*PageSize EnvMonarch/Envelope Monarch: "<>setpagedevice" +*CloseUI: *PageSize + +*OpenUI *PageRegion: PickOne +*OrderDependency: 10 AnySetup *PageRegion +*DefaultPageRegion: Letter +*PageRegion Letter/US Letter: "<>setpagedevice" +*PageRegion Legal/US Legal: "<>setpagedevice" +*PageRegion Executive/US Executive: "<>setpagedevice" +*PageRegion Tabloid/US Tabloid: "<>setpagedevice" +*PageRegion A3/A3: "<>setpagedevice" +*PageRegion A4/A4: "<>setpagedevice" +*PageRegion A5/A5: "<>setpagedevice" +*PageRegion B5/B5 (JIS): "<>setpagedevice" +*PageRegion EnvISOB5/Envelope B5: "<>setpagedevice" +*PageRegion Env10/Envelope #10: "<>setpagedevice" +*PageRegion EnvC5/Envelope C5: "<>setpagedevice" +*PageRegion EnvDL/Envelope DL: "<>setpagedevice" +*PageRegion EnvMonarch/Envelope Monarch: "<>setpagedevice" +*CloseUI: *PageRegion + +*DefaultImageableArea: Letter +*ImageableArea Letter/US Letter: "18 36 594 756" +*ImageableArea Legal/US Legal: "18 36 594 972" +*ImageableArea Executive/US Executive: "18 36 504 684" +*ImageableArea Tabloid/US Tabloid: "18 36 774 1188" +*ImageableArea A3/A3: "18 36 824 1155" +*ImageableArea A4/A4: "18 36 577 806" +*ImageableArea A5/A5: "18 36 403 559" +*ImageableArea B5/JIS B5: "18 36 498 693" +*ImageableArea EnvISOB5/B5 (ISO): "18 36 463 673" +*ImageableArea Env10/Com-10: "18 36 279 648" +*ImageableArea EnvC5/EnvC5: "18 36 441 613" +*ImageableArea EnvDL/EnvDL: "18 36 294 588" +*ImageableArea EnvMonarch/Envelope Monarch: "18 36 261 504" + +*DefaultPaperDimension: Letter +*PaperDimension Letter/US Letter: "612 792" +*PaperDimension Legal/US Legal: "612 1008" +*PaperDimension Executive/US Executive: "522 756" +*PaperDimension Tabloid/US Tabloid: "792 1224" +*PaperDimension A3/A3: "842 1191" +*PaperDimension A4/A4: "595 842" +*PaperDimension A5/A5: "421 595" +*PaperDimension B5/B5 (JIS): "516 729" +*PaperDimension EnvISOB5/Envelope B5: "499 709" +*PaperDimension Env10/Envelope #10: "297 684" +*PaperDimension EnvC5/Envelope C5: "459 649" +*PaperDimension EnvDL/Envelope DL: "312 624" +*PaperDimension EnvMonarch/Envelope Monarch: "279 540" + +*OpenUI *MediaType/Media Type: PickOne +*OrderDependency: 10 AnySetup *MediaType +*DefaultMediaType: Plain +*MediaType Plain/Plain Paper: "<>setpagedevice" +*MediaType Bond/Bond Paper: "<>setpagedevice" +*MediaType Special/Special Paper: "<>setpagedevice" +*MediaType Transparency/Transparency: "<>setpagedevice" +*MediaType Glossy/Glossy Paper: "<>setpagedevice" +*CloseUI: *MediaType + +*OpenUI *InputSlot/Media Source: PickOne +*OrderDependency: 10 AnySetup *InputSlot +*DefaultInputSlot: Tray +*InputSlot Tray/Tray: "<>setpagedevice" +*InputSlot Manual/Manual Feed: "<>setpagedevice" +*InputSlot Envelope/Envelope Feed: "<>setpagedevice" +*CloseUI: *InputSlot + +*OpenUI *Resolution/Output Resolution: PickOne +*OrderDependency: 20 AnySetup *Resolution +*DefaultResolution: 100dpi +*Resolution 75dpi/75 DPI: "<>setpagedevice" +*Resolution 100dpi/100 DPI: "<>setpagedevice" +*CloseUI: *Resolution + +*OpenUI *ColorModel/Output Mode: PickOne +*OrderDependency: 10 AnySetup *ColorModel +*DefaultColorModel: CMYK +*ColorModel CMYK/CMYK Color: "<>setpagedevice" +*ColorModel RGB/CMY Color: "<>setpagedevice" +*ColorModel Gray/Grayscale: "<>setpagedevice" +*CloseUI: *ColorModel + +*DefaultFont: Courier +*Font AvantGarde-Book: Standard "(001.006S)" Standard ROM +*Font AvantGarde-BookOblique: Standard "(001.006S)" Standard ROM +*Font AvantGarde-Demi: Standard "(001.007S)" Standard ROM +*Font AvantGarde-DemiOblique: Standard "(001.007S)" Standard ROM +*Font Bookman-Demi: Standard "(001.004S)" Standard ROM +*Font Bookman-DemiItalic: Standard "(001.004S)" Standard ROM +*Font Bookman-Light: Standard "(001.004S)" Standard ROM +*Font Bookman-LightItalic: Standard "(001.004S)" Standard ROM +*Font Courier: Standard "(002.004S)" Standard ROM +*Font Courier-Bold: Standard "(002.004S)" Standard ROM +*Font Courier-BoldOblique: Standard "(002.004S)" Standard ROM +*Font Courier-Oblique: Standard "(002.004S)" Standard ROM +*Font Helvetica: Standard "(001.006S)" Standard ROM +*Font Helvetica-Bold: Standard "(001.007S)" Standard ROM +*Font Helvetica-BoldOblique: Standard "(001.007S)" Standard ROM +*Font Helvetica-Narrow: Standard "(001.006S)" Standard ROM +*Font Helvetica-Narrow-Bold: Standard "(001.007S)" Standard ROM +*Font Helvetica-Narrow-BoldOblique: Standard "(001.007S)" Standard ROM +*Font Helvetica-Narrow-Oblique: Standard "(001.006S)" Standard ROM +*Font Helvetica-Oblique: Standard "(001.006S)" Standard ROM +*Font NewCenturySchlbk-Bold: Standard "(001.009S)" Standard ROM +*Font NewCenturySchlbk-BoldItalic: Standard "(001.007S)" Standard ROM +*Font NewCenturySchlbk-Italic: Standard "(001.006S)" Standard ROM +*Font NewCenturySchlbk-Roman: Standard "(001.007S)" Standard ROM +*Font Palatino-Bold: Standard "(001.005S)" Standard ROM +*Font Palatino-BoldItalic: Standard "(001.005S)" Standard ROM +*Font Palatino-Italic: Standard "(001.005S)" Standard ROM +*Font Palatino-Roman: Standard "(001.005S)" Standard ROM +*Font Symbol: Special "(001.007S)" Special ROM +*Font Times-Bold: Standard "(001.007S)" Standard ROM +*Font Times-BoldItalic: Standard "(001.009S)" Standard ROM +*Font Times-Italic: Standard "(001.007S)" Standard ROM +*Font Times-Roman: Standard "(001.007S)" Standard ROM +*Font ZapfChancery-MediumItalic: Standard "(001.007S)" Standard ROM +*Font ZapfDingbats: Special "(001.004S)" Standard ROM +*% +*% End of "$Id$". +*% diff --git a/test/testps.ppd b/test/testps.ppd new file mode 100644 index 0000000000..a200c4a501 --- /dev/null +++ b/test/testps.ppd @@ -0,0 +1,183 @@ +*PPD-Adobe: "4.3" +*% +*% "$Id$" +*% +*% Test PS PPD file for CUPS. +*% +*% Copyright 2007-2011 by Apple Inc. +*% Copyright 1997-2005 by Easy Software Products. +*% +*% These coded instructions, statements, and computer programs are the +*% property of Apple Inc. and are protected by Federal copyright +*% law. Distribution and use rights are outlined in the file "LICENSE.txt" +*% which should have been included with this file. If this file is +*% file is missing or damaged, see the license at "http://www.cups.org/". +*% +*FormatVersion: "4.3" +*FileVersion: "1.1" +*LanguageVersion: English +*LanguageEncoding: ISOLatin1 +*PCFileName: "TESTPS.PPD" +*Manufacturer: "ESP" +*Product: "(CUPS v1.1)" +*ModelName: "Test PS Printer" +*ShortNickName: "Test PS Printer" +*NickName: "Test PS Printer CUPS v1.1" +*PSVersion: "(3010.000) 550" +*LanguageLevel: "3" +*ColorDevice: True +*DefaultColorSpace: RGB +*FileSystem: False +*Throughput: "1" +*LandscapeOrientation: Plus90 +*VariablePaperSize: False +*TTRasterizer: Type42 + +*UIConstraints: *PageSize Executive *InputSlot Envelope +*UIConstraints: *PageSize Letter *InputSlot Envelope +*UIConstraints: *PageSize Legal *InputSlot Envelope +*UIConstraints: *PageSize Tabloid *InputSlot Envelope +*UIConstraints: *PageSize A3 *InputSlot Envelope +*UIConstraints: *PageSize A4 *InputSlot Envelope +*UIConstraints: *PageSize A5 *InputSlot Envelope +*UIConstraints: *PageSize B5 *InputSlot Envelope +*UIConstraints: *Resolution 600dpi *ColorModel CMYK + +*OpenUI *PageSize/Media Size: PickOne +*OrderDependency: 10 AnySetup *PageSize +*DefaultPageSize: Letter +*PageSize Letter/US Letter: "<>setpagedevice" +*PageSize Legal/US Legal: "<>setpagedevice" +*PageSize Executive/US Executive: "<>setpagedevice" +*PageSize Tabloid/US Tabloid: "<>setpagedevice" +*PageSize A3/A3: "<>setpagedevice" +*PageSize A4/A4: "<>setpagedevice" +*PageSize A5/A5: "<>setpagedevice" +*PageSize B5/B5 (JIS): "<>setpagedevice" +*PageSize EnvISOB5/Envelope B5: "<>setpagedevice" +*PageSize Env10/Envelope #10: "<>setpagedevice" +*PageSize EnvC5/Envelope C5: "<>setpagedevice" +*PageSize EnvDL/Envelope DL: "<>setpagedevice" +*PageSize EnvMonarch/Envelope Monarch: "<>setpagedevice" +*CloseUI: *PageSize + +*OpenUI *PageRegion: PickOne +*OrderDependency: 10 AnySetup *PageRegion +*DefaultPageRegion: Letter +*PageRegion Letter/US Letter: "<>setpagedevice" +*PageRegion Legal/US Legal: "<>setpagedevice" +*PageRegion Executive/US Executive: "<>setpagedevice" +*PageRegion Tabloid/US Tabloid: "<>setpagedevice" +*PageRegion A3/A3: "<>setpagedevice" +*PageRegion A4/A4: "<>setpagedevice" +*PageRegion A5/A5: "<>setpagedevice" +*PageRegion B5/B5 (JIS): "<>setpagedevice" +*PageRegion EnvISOB5/Envelope B5: "<>setpagedevice" +*PageRegion Env10/Envelope #10: "<>setpagedevice" +*PageRegion EnvC5/Envelope C5: "<>setpagedevice" +*PageRegion EnvDL/Envelope DL: "<>setpagedevice" +*PageRegion EnvMonarch/Envelope Monarch: "<>setpagedevice" +*CloseUI: *PageRegion + +*DefaultImageableArea: Letter +*ImageableArea Letter/US Letter: "18 36 594 756" +*ImageableArea Legal/US Legal: "18 36 594 972" +*ImageableArea Executive/US Executive: "18 36 504 684" +*ImageableArea Tabloid/US Tabloid: "18 36 774 1188" +*ImageableArea A3/A3: "18 36 824 1155" +*ImageableArea A4/A4: "18 36 577 806" +*ImageableArea A5/A5: "18 36 403 559" +*ImageableArea B5/JIS B5: "18 36 498 693" +*ImageableArea EnvISOB5/B5 (ISO): "18 36 463 673" +*ImageableArea Env10/Com-10: "18 36 279 648" +*ImageableArea EnvC5/EnvC5: "18 36 441 613" +*ImageableArea EnvDL/EnvDL: "18 36 294 588" +*ImageableArea EnvMonarch/Envelope Monarch: "18 36 261 504" + +*DefaultPaperDimension: Letter +*PaperDimension Letter/US Letter: "612 792" +*PaperDimension Legal/US Legal: "612 1008" +*PaperDimension Executive/US Executive: "522 756" +*PaperDimension Tabloid/US Tabloid: "792 1224" +*PaperDimension A3/A3: "842 1191" +*PaperDimension A4/A4: "595 842" +*PaperDimension A5/A5: "421 595" +*PaperDimension B5/B5 (JIS): "516 729" +*PaperDimension EnvISOB5/Envelope B5: "499 709" +*PaperDimension Env10/Envelope #10: "297 684" +*PaperDimension EnvC5/Envelope C5: "459 649" +*PaperDimension EnvDL/Envelope DL: "312 624" +*PaperDimension EnvMonarch/Envelope Monarch: "279 540" + +*OpenUI *MediaType/Media Type: PickOne +*OrderDependency: 10 AnySetup *MediaType +*DefaultMediaType: Plain +*MediaType Plain/Plain Paper: "<>setpagedevice" +*MediaType Bond/Bond Paper: "<>setpagedevice" +*MediaType Special/Special Paper: "<>setpagedevice" +*MediaType Transparency/Transparency: "<>setpagedevice" +*MediaType Glossy/Glossy Paper: "<>setpagedevice" +*CloseUI: *MediaType + +*OpenUI *InputSlot/Media Source: PickOne +*OrderDependency: 10 AnySetup *InputSlot +*DefaultInputSlot: Tray +*InputSlot Tray/Tray: "<>setpagedevice" +*InputSlot Manual/Manual Feed: "<>setpagedevice" +*InputSlot Envelope/Envelope Feed: "<>setpagedevice" +*CloseUI: *InputSlot + +*OpenUI *Resolution/Output Resolution: PickOne +*OrderDependency: 20 AnySetup *Resolution +*DefaultResolution: 100dpi +*Resolution 75dpi/75 DPI: "<>setpagedevice" +*Resolution 100dpi/100 DPI: "<>setpagedevice" +*CloseUI: *Resolution + +*OpenUI *ColorModel/Output Mode: PickOne +*OrderDependency: 10 AnySetup *ColorModel +*DefaultColorModel: CMYK +*ColorModel CMYK/CMYK Color: "<>setpagedevice" +*ColorModel RGB/CMY Color: "<>setpagedevice" +*ColorModel Gray/Grayscale: "<>setpagedevice" +*CloseUI: *ColorModel + +*DefaultFont: Courier +*Font AvantGarde-Book: Standard "(001.006S)" Standard ROM +*Font AvantGarde-BookOblique: Standard "(001.006S)" Standard ROM +*Font AvantGarde-Demi: Standard "(001.007S)" Standard ROM +*Font AvantGarde-DemiOblique: Standard "(001.007S)" Standard ROM +*Font Bookman-Demi: Standard "(001.004S)" Standard ROM +*Font Bookman-DemiItalic: Standard "(001.004S)" Standard ROM +*Font Bookman-Light: Standard "(001.004S)" Standard ROM +*Font Bookman-LightItalic: Standard "(001.004S)" Standard ROM +*Font Courier: Standard "(002.004S)" Standard ROM +*Font Courier-Bold: Standard "(002.004S)" Standard ROM +*Font Courier-BoldOblique: Standard "(002.004S)" Standard ROM +*Font Courier-Oblique: Standard "(002.004S)" Standard ROM +*Font Helvetica: Standard "(001.006S)" Standard ROM +*Font Helvetica-Bold: Standard "(001.007S)" Standard ROM +*Font Helvetica-BoldOblique: Standard "(001.007S)" Standard ROM +*Font Helvetica-Narrow: Standard "(001.006S)" Standard ROM +*Font Helvetica-Narrow-Bold: Standard "(001.007S)" Standard ROM +*Font Helvetica-Narrow-BoldOblique: Standard "(001.007S)" Standard ROM +*Font Helvetica-Narrow-Oblique: Standard "(001.006S)" Standard ROM +*Font Helvetica-Oblique: Standard "(001.006S)" Standard ROM +*Font NewCenturySchlbk-Bold: Standard "(001.009S)" Standard ROM +*Font NewCenturySchlbk-BoldItalic: Standard "(001.007S)" Standard ROM +*Font NewCenturySchlbk-Italic: Standard "(001.006S)" Standard ROM +*Font NewCenturySchlbk-Roman: Standard "(001.007S)" Standard ROM +*Font Palatino-Bold: Standard "(001.005S)" Standard ROM +*Font Palatino-BoldItalic: Standard "(001.005S)" Standard ROM +*Font Palatino-Italic: Standard "(001.005S)" Standard ROM +*Font Palatino-Roman: Standard "(001.005S)" Standard ROM +*Font Symbol: Special "(001.007S)" Special ROM +*Font Times-Bold: Standard "(001.007S)" Standard ROM +*Font Times-BoldItalic: Standard "(001.009S)" Standard ROM +*Font Times-Italic: Standard "(001.007S)" Standard ROM +*Font Times-Roman: Standard "(001.007S)" Standard ROM +*Font ZapfChancery-MediumItalic: Standard "(001.007S)" Standard ROM +*Font ZapfDingbats: Special "(001.004S)" Standard ROM +*% +*% End of "$Id$". +*% diff --git a/test/waitjobs.sh b/test/waitjobs.sh new file mode 100755 index 0000000000..7a79fb69be --- /dev/null +++ b/test/waitjobs.sh @@ -0,0 +1,60 @@ +#!/bin/sh +# +# "$Id$" +# +# Script to wait for jobs to complete. +# +# Copyright 2008-2009 by Apple Inc. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +# +# Get timeout from command-line +# + +if test $# = 1; then + timeout=$1 +else + timeout=360 +fi + +# +# Figure out the proper echo options... +# + +if (echo "testing\c"; echo 1,2,3) | grep c >/dev/null; then + ac_n=-n + ac_c= +else + ac_n= + ac_c='\c' +fi + +echo $ac_n "Waiting for jobs to complete...$ac_c" +oldjobs=0 + +while test $timeout -gt 0; do + jobs=`../systemv/lpstat 2>/dev/null | wc -l | tr -d ' '` + if test $jobs = 0; then + break + fi + + if test $jobs != $oldjobs; then + echo $ac_n "$jobs...$ac_c" + oldjobs=$jobs + fi + + sleep 5 + timeout=`expr $timeout - 5` +done + +echo "" + +# +# End of "$Id$". +# diff --git a/test/xmltotest.c b/test/xmltotest.c new file mode 100644 index 0000000000..8f601cc269 --- /dev/null +++ b/test/xmltotest.c @@ -0,0 +1,529 @@ +/* + * "$Id$" + * + * IANA XML registration to test file generator for CUPS. + * + * Copyright 2011-2012 by Apple Inc. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * This file is subject to the Apple OS-Developed Software exception. + * + * Usage: + * + * ./xmltotest [--ref standard] {--job|--printer} [XML file/URL] >file.test + * + * If not specified, loads the XML registrations from: + * + * http://www.iana.org/assignments/ipp-registrations/ipp-registrations.xml + * + * "Standard" is of the form "rfcNNNN" or "pwgNNNN.N". + * + * Contents: + * + * main() - Process command-line arguments. + * compare_reg() - Compare two registrations. + * load_xml() - Load the XML registration file or URL. + * match_xref() - Compare the xref against the named standard. + * new_reg() - Create a new registration record. + * usage() - Show usage message. + * write_expect() - Write an EXPECT test for an attribute. + */ + + +#include +#include +#include +#include + +#ifdef HAVE_MXML_H +# include +/* + * Local types... + */ + +typedef struct _cups_reg_s /**** Registration data ****/ +{ + char *name, /* Attribute name */ + *member, /* Member attribute name */ + *sub_member, /* Sub-member attribute name */ + *syntax; /* Attribute syntax */ +} _cups_reg_t; + + +/* + * Local functions... + */ + +static int compare_reg(_cups_reg_t *a, _cups_reg_t *b); +static mxml_node_t *load_xml(const char *reg_file); +static int match_xref(mxml_node_t *xref, const char *standard); +static _cups_reg_t *new_reg(mxml_node_t *name, mxml_node_t *member, + mxml_node_t *sub_member, mxml_node_t *syntax); +static int usage(void); +static void write_expect(_cups_reg_t *reg, ipp_tag_t group); + + +/* + * 'main()' - Process command-line arguments. + */ + +int +main(int argc, /* I - Number of command-line args */ + char *argv[]) /* I - Command-line arguments */ +{ + int i; /* Looping var */ + const char *reg_file = NULL, /* Registration file/URL to use */ + *reg_standard = NULL; /* Which standard to extract */ + mxml_node_t *reg_xml, /* Registration XML data */ + *reg_2, /* ipp-registrations-2 */ + *reg_record, /* */ + *reg_collection, /* */ + *reg_name, /* */ + *reg_member, /* */ + *reg_sub_member, /* */ + *reg_syntax, /* */ + *reg_xref; /* */ + cups_array_t *attrs; /* Attribute registrations */ + _cups_reg_t *current; /* Current attribute registration */ + ipp_tag_t group = IPP_TAG_ZERO, /* Which attributes to test */ + reg_group; /* Group for registration */ + + + /* + * Parse command-line... + */ + + for (i = 1; i < argc; i ++) + { + if (!strcmp(argv[i], "--job") && group == IPP_TAG_ZERO) + group = IPP_TAG_JOB; + else if (!strcmp(argv[i], "--ref")) + { + i ++; + if (i >= argc) + return (usage()); + + reg_standard = argv[i]; + } + else if (!strcmp(argv[i], "--printer") && group == IPP_TAG_ZERO) + group = IPP_TAG_PRINTER; + else if (argv[i][0] == '-' || reg_file) + return (usage()); + else + reg_file = argv[i]; + } + + if (group == IPP_TAG_ZERO) + return (usage()); + + /* + * Read registrations... + */ + + if (!reg_file) + reg_file = "http://www.iana.org/assignments/ipp-registrations/" + "ipp-registrations.xml"; + + if ((reg_xml = load_xml(reg_file)) == NULL) + return (1); + + /* + * Scan registrations for attributes... + */ + + if ((reg_2 = mxmlFindElement(reg_xml, reg_xml, "registry", "id", + "ipp-registrations-2", + MXML_DESCEND)) == NULL) + { + fprintf(stderr, "xmltotest: No IPP attribute registrations in \"%s\".\n", + reg_file); + return (1); + } + + attrs = cupsArrayNew((cups_array_func_t)compare_reg, NULL); + + for (reg_record = mxmlFindElement(reg_2, reg_2, "record", NULL, NULL, + MXML_DESCEND); + reg_record; + reg_record = mxmlFindElement(reg_record, reg_2, "record", NULL, NULL, + MXML_NO_DESCEND)) + { + /* + * Get the values from the current record... + */ + + reg_collection = mxmlFindElement(reg_record, reg_record, "collection", + NULL, NULL, MXML_DESCEND); + reg_name = mxmlFindElement(reg_record, reg_record, "name", NULL, NULL, + MXML_DESCEND); + reg_member = mxmlFindElement(reg_record, reg_record, "member_attribute", + NULL, NULL, MXML_DESCEND); + reg_sub_member = mxmlFindElement(reg_record, reg_record, + "sub-member_attribute", NULL, NULL, + MXML_DESCEND); + reg_syntax = mxmlFindElement(reg_record, reg_record, "syntax", NULL, + NULL, MXML_DESCEND); + reg_xref = mxmlFindElement(reg_record, reg_record, "xref", NULL, NULL, + MXML_DESCEND); + + if (!reg_collection || !reg_name || !reg_syntax || !reg_xref) + continue; + + /* + * Filter based on group and standard... + */ + + if (!strcmp(reg_collection->child->value.opaque, "Printer Description")) + reg_group = IPP_TAG_PRINTER; + else if (!strcmp(reg_collection->child->value.opaque, "Job Description")) + reg_group = IPP_TAG_JOB; + else if (!strcmp(reg_collection->child->value.opaque, "Job Template")) + { + if (strstr(reg_name->child->value.opaque, "-default") || + strstr(reg_name->child->value.opaque, "-supported")) + reg_group = IPP_TAG_PRINTER; + else + reg_group = IPP_TAG_JOB; + } + else + reg_group = IPP_TAG_ZERO; + + if (reg_group != group) + continue; + + if (reg_standard && !match_xref(reg_xref, reg_standard)) + continue; + + /* + * Add the record to the array... + */ + + if ((current = new_reg(reg_name, reg_member, reg_sub_member, + reg_syntax)) != NULL) + cupsArrayAdd(attrs, current); + } + + /* + * Write out a test for all of the selected attributes... + */ + + puts("{"); + + if (group == IPP_TAG_PRINTER) + { + puts("\tOPERATION Get-Printer-Attributes"); + puts("\tGROUP operation-attributes-tag"); + puts("\tATTR charset attributes-charset utf-8"); + puts("\tATTR naturalLanguage attributes-natural-language en"); + puts("\tATTR uri printer-uri $uri"); + puts("\tATTR name requesting-user-name $user"); + puts("\tATTR keyword requested-attributes all,media-col-database"); + puts(""); + puts("\tSTATUS successful-ok"); + puts("\tSTATUS successful-ok-ignored-or-substituted-attributes"); + puts(""); + } + else + { + puts("\tOPERATION Get-Job-Attributes"); + puts("\tGROUP operation-attributes-tag"); + puts("\tATTR charset attributes-charset utf-8"); + puts("\tATTR naturalLanguage attributes-natural-language en"); + puts("\tATTR uri printer-uri $uri"); + puts("\tATTR integer job-id $job-id"); + puts("\tATTR name requesting-user-name $user"); + puts(""); + puts("\tSTATUS successful-ok"); + puts(""); + } + + for (current = cupsArrayFirst(attrs); + current; + current = cupsArrayNext(attrs)) + write_expect(current, group); + + puts("}"); + + return (0); +} + + +/* + * 'compare_reg()' - Compare two registrations. + */ + +static int /* O - Result of comparison */ +compare_reg(_cups_reg_t *a, /* I - First registration */ + _cups_reg_t *b) /* I - Second registration */ +{ + int retval; /* Return value */ + + + if ((retval = strcmp(a->name, b->name)) != 0) + return (retval); + + if (a->member && b->member) + retval = strcmp(a->member, b->member); + else if (a->member) + retval = 1; + else if (b->member) + retval = -1; + + if (retval) + return (retval); + + if (a->sub_member && b->sub_member) + retval = strcmp(a->sub_member, b->sub_member); + else if (a->sub_member) + retval = 1; + else if (b->sub_member) + retval = -1; + + return (retval); +} + + +/* + * 'load_xml()' - Load the XML registration file or URL. + */ + +static mxml_node_t * /* O - XML file or NULL */ +load_xml(const char *reg_file) /* I - Filename or URL */ +{ + mxml_node_t *xml; /* XML file */ + char scheme[256], /* Scheme */ + userpass[256], /* Username and password */ + hostname[256], /* Hostname */ + resource[1024], /* Resource path */ + filename[1024]; /* Temporary file */ + int port, /* Port number */ + fd; /* File descriptor */ + + + if (httpSeparateURI(HTTP_URI_CODING_ALL, reg_file, scheme, sizeof(scheme), + userpass, sizeof(userpass), hostname, sizeof(hostname), + &port, resource, sizeof(resource)) < HTTP_URI_OK) + { + fprintf(stderr, "xmltotest: Bad URI or filename \"%s\".\n", reg_file); + return (NULL); + } + + if (!strcmp(scheme, "file")) + { + /* + * Local file... + */ + + if ((fd = open(resource, O_RDONLY)) < 0) + { + fprintf(stderr, "xmltotest: Unable to open \"%s\": %s\n", resource, + strerror(errno)); + return (NULL); + } + + filename[0] = '\0'; + } + else if (strcmp(scheme, "http") && strcmp(scheme, "https")) + { + fprintf(stderr, "xmltotest: Unsupported URI scheme \"%s\".\n", scheme); + return (NULL); + } + else + { + http_t *http; /* HTTP connection */ + http_encryption_t encryption; /* Encryption to use */ + http_status_t status; /* Status of HTTP GET */ + + if (!strcmp(scheme, "https") || port == 443) + encryption = HTTP_ENCRYPT_ALWAYS; + else + encryption = HTTP_ENCRYPT_IF_REQUESTED; + + if ((http = httpConnectEncrypt(hostname, port, encryption)) == NULL) + { + fprintf(stderr, "xmltotest: Unable to connect to \"%s\": %s\n", hostname, + cupsLastErrorString()); + return (NULL); + } + + if ((fd = cupsTempFd(filename, sizeof(filename))) < 0) + { + fprintf(stderr, "xmltotest: Unable to create temporary file: %s\n", + strerror(errno)); + httpClose(http); + return (NULL); + } + + status = cupsGetFd(http, resource, fd); + httpClose(http); + + if (status != HTTP_OK) + { + fprintf(stderr, "mxmltotest: Unable to get \"%s\": %d\n", reg_file, + status); + close(fd); + unlink(filename); + return (NULL); + } + + lseek(fd, 0, SEEK_SET); + } + + /* + * Load the XML file... + */ + + xml = mxmlLoadFd(NULL, fd, MXML_OPAQUE_CALLBACK); + + close(fd); + + if (filename[0]) + unlink(filename); + + return (xml); +} + + +/* + * 'match_xref()' - Compare the xref against the named standard. + */ + +static int /* O - 1 if match, 0 if not */ +match_xref(mxml_node_t *xref, /* I - node */ + const char *standard) /* I - Name of standard */ +{ + const char *data; /* "data" attribute */ + char s[256]; /* String to look for */ + + + if ((data = mxmlElementGetAttr(xref, "data")) == NULL) + return (1); + + if (!strcmp(data, standard)) + return (1); + + if (!strncmp(standard, "pwg", 3)) + { + snprintf(s, sizeof(s), "-%s.pdf", standard + 3); + return (strstr(data, s) != NULL); + } + else + return (0); +} + + +/* + * 'new_reg()' - Create a new registration record. + */ + +static _cups_reg_t * /* O - New record */ +new_reg(mxml_node_t *name, /* I - Attribute name */ + mxml_node_t *member, /* I - Member attribute, if any */ + mxml_node_t *sub_member, /* I - Sub-member attribute, if any */ + mxml_node_t *syntax) /* I - Syntax */ +{ + _cups_reg_t *reg; /* New record */ + + + if ((reg = calloc(1, sizeof(_cups_reg_t))) != NULL) + { + reg->name = name->child->value.opaque; + reg->syntax = syntax->child->value.opaque; + + if (member) + reg->member = member->child->value.opaque; + + if (sub_member) + reg->sub_member = sub_member->child->value.opaque; + } + + return (reg); +} + + +/* + * 'usage()' - Show usage message. + */ + +static int /* O - Exit status */ +usage(void) +{ + puts("Usage ./xmltotest [--ref standard] {--job|--printer} [XML file/URL] " + ">file.test"); + return (1); +} + + +/* + * 'write_expect()' - Write an EXPECT test for an attribute. + */ + +static void +write_expect(_cups_reg_t *reg, /* I - Registration information */ + ipp_tag_t group) /* I - Attribute group tag */ +{ + const char *syntax; /* Pointer into syntax string */ + int single = 1, /* Single valued? */ + skip = 0; /* Skip characters? */ + + + printf("\tEXPECT ?%s OF-TYPE ", reg->name); + + syntax = reg->syntax; + + while (*syntax) + { + if (!strncmp(syntax, "1setOf", 6)) + { + single = 0; + syntax += 6; + + while (isspace(*syntax & 255)) + syntax ++; + + if (*syntax == '(') + syntax ++; + } + else if (!strncmp(syntax, "type1", 5) || !strncmp(syntax, "type2", 5) || + !strncmp(syntax, "type3", 5)) + syntax += 5; + else if (*syntax == '(') + { + skip = 1; + syntax ++; + } + else if (*syntax == ')') + { + skip = 0; + syntax ++; + } + else if (!skip && (*syntax == '|' || isalpha(*syntax & 255))) + putchar(*syntax++); + else + syntax ++; + } + + if (single) + printf(" IN-GROUP %s COUNT 1\n", ippTagString(group)); + else + printf(" IN-GROUP %s\n", ippTagString(group)); +} + + +#else /* !HAVE_MXML */ +int +main(void) +{ + return (1); +} +#endif /* HAVE_MXML */ + + +/* + * End of "$Id$". + */ diff --git a/tools/checkglobals b/tools/checkglobals new file mode 100755 index 0000000000..4d15387eaf --- /dev/null +++ b/tools/checkglobals @@ -0,0 +1,32 @@ +#!/bin/sh +# +# Check for global symbols that don't need to be made global... +# + +for file in *.o; do + functions="" + + for function in `nm -g $file | grep "T " | awk '{print $3}'`; do + found="" + for file2 in *.o; do + if test "$file" = "$file2"; then + continue; + fi + + found=`nm -g $file2 | grep $function` + if test "$found" != ""; then + break; + fi + done + + if test -z "$found"; then + functions="$functions $function" + fi + done + + if test -z "$functions"; then + echo "$file: OK" + else + echo "$file: $functions" + fi +done diff --git a/tools/listpublic b/tools/listpublic new file mode 100755 index 0000000000..d864e7ee63 --- /dev/null +++ b/tools/listpublic @@ -0,0 +1,12 @@ +#!/bin/sh +# +# List public API symbols... +# + +for function in `nm -g *.so | grep "T " | awk '{print $3}' | grep -v '^_' | sort`; do + found=`grep $function\( *.h | grep -v DEPRECATED` + + if test "x$found" != x; then + echo $function + fi +done diff --git a/tools/makeipptoolpkg b/tools/makeipptoolpkg new file mode 100755 index 0000000000..a8ff8dfd43 --- /dev/null +++ b/tools/makeipptoolpkg @@ -0,0 +1,88 @@ +#!/bin/sh +# +# "$Id: makeipptoolpkg 10143 2011-12-02 20:11:30Z mike $" +# +# Make an ipptool package for CUPS. +# +# Copyright 2007-2011 by Apple Inc. +# Copyright 1997-2007 by Easy Software Products, all rights reserved. +# +# These coded instructions, statements, and computer programs are the +# property of Apple Inc. and are protected by Federal copyright +# law. Distribution and use rights are outlined in the file "LICENSE.txt" +# which should have been included with this file. If this file is +# file is missing or damaged, see the license at "http://www.cups.org/". +# + +# Make sure we are running in the right directory... +if test ! -f tools/makeipptoolpkg; then + echo "Run this script from the top-level CUPS source directory, e.g.:" + echo "" + echo " tools/makeipptoolpkg $*" + echo "" + exit 1 +fi + +uname="`uname | awk '{print tolower($1)}'`" +if test "x$1" = xlsb; then + shift + uname="linux-lsb" +fi + +if test $# = 0; then + echo Updating to get snapshot version... + svn up + rev=`svnversion . | sed -e '1,$s/[a-zA-Z]//g'` + fileversion="`date '+%Y%m%d'`-r$rev" +else + fileversion=$1 +fi + +if (svn st | grep -qv '^\?'); then + echo Local changes remain: + svn st | grep -v '^\?' + exit 1 +fi + +echo Creating package directory... +pkgdir="ipptool-$fileversion" + +test -d $pkgdir && rm -r $pkgdir +mkdir $pkgdir || exit 1 + +echo Copying package files +cp CHANGES-IPPTOOL.txt IPPTOOL.txt LICENSE.txt $pkgdir +cp doc/help/man-ipptool*.html $pkgdir +cp test/color.jpg $pkgdir +cp test/create-printer-subscription.test $pkgdir +cp test/document-*.pdf $pkgdir +cp test/document-*.ps $pkgdir +cp test/get-completed-jobs.test test/get-jobs.test $pkgdir +cp test/get-printer-attributes.test $pkgdir +cp test/gray.jpg $pkgdir +cp test/ipp-[12].*.test $pkgdir +cp test/ipptool-static $pkgdir/ipptool +cp test/onepage-*.pdf $pkgdir +cp test/onepage-*.ps $pkgdir +cp test/testfile.* $pkgdir + +if test `uname` = Darwin; then + pkgfile="$pkgdir-macosx-universal.dmg" + echo Creating disk image $pkgfile... + test -f $pkgfile && rm $pkgfile + hdiutil create -srcfolder $pkgdir $pkgfile + #sudo chown `whoami` $pkgfile +else + pkgfile="$pkgdir-$uname-`uname -m`.tar.gz" + echo Creating archive $pkgfile... + tar czf $pkgfile $pkgdir || exit 1 +fi + +echo Removing temporary files... +rm -r $pkgdir + +echo Done. + +# +# End of "$Id: makeipptoolpkg 10143 2011-12-02 20:11:30Z mike $". +# diff --git a/tools/makesrcdist b/tools/makesrcdist new file mode 100755 index 0000000000..694fb2357d --- /dev/null +++ b/tools/makesrcdist @@ -0,0 +1,90 @@ +#!/bin/sh +# +# "$Id: makesrcdist 9852 2011-07-06 20:14:24Z mike $" +# +# makesrcdist - make a source distribution of CUPS. +# + +# Make sure we are running in the right directory... +if test ! -f tools/makesrcdist; then + echo "Run this script from the top-level CUPS source directory, e.g.:" + echo "" + echo " tools/makesrcdist $*" + echo "" + exit 1 +fi + +if test $# = 0; then + echo Updating for snapshot... + svn up + rev=`svnversion . | sed -e '1,$s/[a-zA-Z]//g'` + version="1.5svn" + revision="-r$rev" + fileversion="1.5svn-r$rev" + fileurl="http://ftp.easysw.com/pub/cups/test/cups-$fileversion-source.tar." + url="." +else + echo Creating tag for release... + rev="1" + version=$1 + revision="" + fileversion=$1 + fileurl="http://ftp.easysw.com/pub/cups/$version/cups-$fileversion-source.tar." + url="https://svn.easysw.com/public/cups/tags/release-$version" + + svn copy https://svn.easysw.com/public/cups/trunk "$url" \ + -m "Tag $version" || exit 1 +fi + +fileurl=`echo $fileurl | sed -e '1,$s/\\//\\\\\\//g'` + +echo Exporting $fileversion... +rm -rf /tmp/cups-$version +svn export $url /tmp/cups-$version + +echo Updating version information... +cd /tmp/cups-$version/config-scripts + +sed -e '1,$s/^CUPS_VERSION=.*/CUPS_VERSION='$version'/' \ + -e '1,$s/^CUPS_REVISION=.*/CUPS_REVISION='$revision'/' \ + cups-common.m4.new +mv cups-common.m4.new cups-common.m4 +cd .. + +echo Configuring... +autoconf -f +rm -rf autom4te*.cache +rm -rf tools +cd .. + +echo -n Archiving...gz +sed -e '1,$s/@CUPS_VERSION@/'$version'/' \ + -e '1,$s/^Release:.*/Release: '$rev'/' \ + -e '1,$s/^Source:.*/Source: '$fileurl'gz/' \ + cups-$version/packaging/cups.spec +tar czf cups-$fileversion-source.tar.gz cups-$version + +echo -n ...bz2 +sed -e '1,$s/@CUPS_VERSION@/'$version'/' \ + -e '1,$s/^Release:.*/Release: '$rev'/' \ + -e '1,$s/^Source:.*/Source: '$fileurl'bz2/' \ + cups-$version/packaging/cups.spec +tar cjf cups-$fileversion-source.tar.bz2 cups-$version +echo "..." + +if test -x /usr/bin/md5sum; then + (cd /tmp; md5sum cups-$fileversion-source.tar.* | awk '{print $1, "'$fileversion' cups/'$fileversion'/" $2}') +elif test -x /sbin/md5; then + (cd /tmp; md5 cups-$fileversion-source.tar.* | awk '{print $4, "'$fileversion' cups/'$fileversion'/" substr($2, 2, length($2) - 2)}') +fi + +echo Removing temporary files... +rm -rf cups-$version + +echo "Done!" + +# +# End of "$Id: makesrcdist 9852 2011-07-06 20:14:24Z mike $". +# diff --git a/tools/pdftops-darwin.sh b/tools/pdftops-darwin.sh new file mode 100755 index 0000000000..07d3224c5e --- /dev/null +++ b/tools/pdftops-darwin.sh @@ -0,0 +1,44 @@ +#!/bin/sh +# +# Script to simulate Xpdf/Poppler's pdftops program. +# + +options="" + +while test $# -gt 0; do + option="$1" + shift + + case "$option" in + -expand) + options="$options fit-to-page" + ;; + -h) + echo "Usage: pdftops [options] filename" + echo "Options:" + echo " -expand" + echo " -h" + echo " -level1" + echo " -level2" + echo " -level3" + echo " -noembtt" + echo " -origpagesizes" + echo " -paperw width-points" + echo " -paperh length-points" + echo "" + echo "THIS IS A COMPATIBILITY WRAPPER" + exit 0 + ;; + -paperw | -paperh) + # Ignore width/length in points + shift + ;; + -*) + # Ignore everything else + ;; + *) + /usr/libexec/cups/filter/cgpdftops job user title 1 "$options" "$option" + exit $? + ;; + esac +done diff --git a/tools/products.php b/tools/products.php new file mode 100755 index 0000000000..9c2e7430ea --- /dev/null +++ b/tools/products.php @@ -0,0 +1,50 @@ +#!/usr/bin/php -f + $maxlen) + $maxlen = strlen($data[1]); +} + +pclose($fp); + +arsort($files); + +$current_count = 0; +$current_files = 0; + +foreach ($files as $file => $count) +{ + if ($current_count == 0) + print(basename($file) . " => $count products\n"); + + if ($count != $current_count) + { + if ($current_count != 0) + print("$current_files PPDs with $current_count products.\n"); + + $current_count = $count; + $current_files = 1; + } + else + $current_files ++; +} + +if ($current_count != 0) + print("$current_files PPDs with $current_count products.\n"); + +print("Maximum length of Product string: $maxlen\n"); + +?> diff --git a/tools/testosx b/tools/testosx new file mode 100755 index 0000000000..4547977f23 --- /dev/null +++ b/tools/testosx @@ -0,0 +1,137 @@ +#!/bin/sh +# Make sure we are running in the right directory... +if test ! -f tools/testosx; then + echo "Run this script from the top-level CUPS source directory, e.g.:" + echo "" + echo " sudo tools/testosx [version]" + echo "" + exit 1 +fi + +if test `whoami` != root; then + echo "Run this script with sudo, e.g.:" + echo "" + echo " sudo tools/testosx [version]" + echo "" + exit 1 +fi + +# Get the current working copy version... +rev=`svnversion . | awk -F: '{print $NF}' | sed -e '1,$s/[a-zA-Z]*//g'` + +if test $# = 0; then + version="1.5svn-r$rev" +else + version=$1 +fi + +# Setup an install directory... +user=`whoami` +topdir=`pwd` +pkgdir="/tmp/cups.pkg-$user" + +echo Building package using temp directory $pkgdir... +rm -rf $pkgdir +mkdir -p $pkgdir/Package +mkdir -p $pkgdir/Resources + +# Install resource files into the Resources directory... +echo Installing resource files... +cp packaging/LICENSE.rtf $pkgdir/Resources/ReadMe.rtf +sed -e '1,$s/@CUPS_VERSION@/'$version'/g' \ + $pkgdir/Resources/Welcome.rtf +cp packaging/installer.tif $pkgdir/Resources/background.tif + +if test -x /bin/launchctl; then + cat >$pkgdir/Resources/preflight <$pkgdir/Resources/preflight <$pkgdir/Resources/postflight <$pkgdir/Resources/postflight <$file +done + +# Install CUPS into the Package directory... +#make INSTALL=$topdir/install-sh BUILDROOT=$pkgdir/Package install +make BUILDROOT=$pkgdir/Package install || exit 1 + +# Figure out where PackageMaker is installled... +if test -d /Developer/Applications/Utilities/PackageMaker.app; then + PackageMaker=/Developer/Applications/Utilities/PackageMaker.app/Contents/MacOS/PackageMaker +else + PackageMaker=/Developer/Applications/PackageMaker.app/Contents/MacOS/PackageMaker +fi + +# Create the package... +echo Creating MacOS X package... +rm -rf cups.pkg +echo $PackageMaker -build -v -p cups.pkg \ + -f $pkgdir/Package \ + -r $pkgdir/Resources \ + -d packaging/cups-desc.plist \ + -i packaging/cups-info.plist +$PackageMaker -build -v -p cups.pkg \ + -f $pkgdir/Package \ + -r $pkgdir/Resources \ + -d packaging/cups-desc.plist \ + -i packaging/cups-info.plist + +# Create a disk image... +echo Creating MacOS X disk image... +hdiutil create -ov -srcfolder cups.pkg cups-$version.dmg + +# Cleanup temp files... +echo Removing temporary files... +#rm -rf $pkgdir diff --git a/tools/testrpm b/tools/testrpm new file mode 100755 index 0000000000..7d4e9e4a3c --- /dev/null +++ b/tools/testrpm @@ -0,0 +1,32 @@ +#!/bin/sh +# +# "$Id: testrpm 5304 2006-03-18 01:21:37Z mike $" +# +# Test script for making RPMs... +# + +# Make sure we are running in the right directory... +if test ! -f tools/testrpm; then + echo "Run this script from the top-level CUPS source directory, e.g.:" + echo "" + echo " tools/testrpm [rpmbuild options]" + echo "" + exit 1 +fi + +# Strip command-line arguments so we don't create a release tag... +args="$*" +shift $# + +# Get a snapshot of the current source... +. tools/makesrcdist + +# Build the RPM... +echo Building rpm... +rm -f /usr/src/redhat/RPMS/i386/cups*.rpm +rm -f /usr/src/redhat/SRPMS/cups*.rpm +rpmbuild -ta $args cups-$fileversion-source.tar.bz2 + +# +# End of "$Id: testrpm 5304 2006-03-18 01:21:37Z mike $". +# diff --git a/vc2005/cups.sln b/vc2005/cups.sln new file mode 100644 index 0000000000..e510e163c1 --- /dev/null +++ b/vc2005/cups.sln @@ -0,0 +1,78 @@ +Microsoft Visual Studio Solution File, Format Version 9.00 +# Visual Studio 2005 +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "libcups2", "libcups2.vcproj", "{CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}" +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "libcupsimage2", "libcupsimage2.vcproj", "{CB4AA6F2-3E84-45BE-B505-95CD375E1234}" + ProjectSection(ProjectDependencies) = postProject + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} = {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "testfile", "testfile.vcproj", "{CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}" + ProjectSection(ProjectDependencies) = postProject + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} = {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "testhttp", "testhttp.vcproj", "{90B0058C-8393-411F-BD3B-E2C831D4E883}" + ProjectSection(ProjectDependencies) = postProject + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} = {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "cupstestppd", "cupstestppd.vcproj", "{6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}" + ProjectSection(ProjectDependencies) = postProject + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} = {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} + {CB4AA6F2-3E84-45BE-B505-95CD375E1234} = {CB4AA6F2-3E84-45BE-B505-95CD375E1234} + EndProjectSection +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|Win32 = Debug|Win32 + Debug|x64 = Debug|x64 + Release|Win32 = Release|Win32 + Release|x64 = Release|x64 + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}.Debug|Win32.ActiveCfg = Debug|Win32 + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}.Debug|Win32.Build.0 = Debug|Win32 + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}.Debug|x64.ActiveCfg = Debug|x64 + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}.Debug|x64.Build.0 = Debug|x64 + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}.Release|Win32.ActiveCfg = Debug|Win32 + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}.Release|Win32.Build.0 = Debug|Win32 + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}.Release|x64.ActiveCfg = Debug|x64 + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}.Release|x64.Build.0 = Debug|x64 + {CB4AA6F2-3E84-45BE-B505-95CD375E1234}.Debug|Win32.ActiveCfg = Debug|Win32 + {CB4AA6F2-3E84-45BE-B505-95CD375E1234}.Debug|Win32.Build.0 = Debug|Win32 + {CB4AA6F2-3E84-45BE-B505-95CD375E1234}.Debug|x64.ActiveCfg = Debug|x64 + {CB4AA6F2-3E84-45BE-B505-95CD375E1234}.Debug|x64.Build.0 = Debug|x64 + {CB4AA6F2-3E84-45BE-B505-95CD375E1234}.Release|Win32.ActiveCfg = Release|Win32 + {CB4AA6F2-3E84-45BE-B505-95CD375E1234}.Release|Win32.Build.0 = Release|Win32 + {CB4AA6F2-3E84-45BE-B505-95CD375E1234}.Release|x64.ActiveCfg = Release|x64 + {CB4AA6F2-3E84-45BE-B505-95CD375E1234}.Release|x64.Build.0 = Release|x64 + {CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}.Debug|Win32.ActiveCfg = Debug|Win32 + {CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}.Debug|Win32.Build.0 = Debug|Win32 + {CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}.Debug|x64.ActiveCfg = Debug|x64 + {CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}.Debug|x64.Build.0 = Debug|x64 + {CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}.Release|Win32.ActiveCfg = Debug|Win32 + {CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}.Release|Win32.Build.0 = Debug|Win32 + {CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}.Release|x64.ActiveCfg = Debug|x64 + {CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}.Release|x64.Build.0 = Debug|x64 + {90B0058C-8393-411F-BD3B-E2C831D4E883}.Debug|Win32.ActiveCfg = Debug|Win32 + {90B0058C-8393-411F-BD3B-E2C831D4E883}.Debug|Win32.Build.0 = Debug|Win32 + {90B0058C-8393-411F-BD3B-E2C831D4E883}.Debug|x64.ActiveCfg = Debug|x64 + {90B0058C-8393-411F-BD3B-E2C831D4E883}.Debug|x64.Build.0 = Debug|x64 + {90B0058C-8393-411F-BD3B-E2C831D4E883}.Release|Win32.ActiveCfg = Debug|Win32 + {90B0058C-8393-411F-BD3B-E2C831D4E883}.Release|Win32.Build.0 = Debug|Win32 + {90B0058C-8393-411F-BD3B-E2C831D4E883}.Release|x64.ActiveCfg = Debug|x64 + {90B0058C-8393-411F-BD3B-E2C831D4E883}.Release|x64.Build.0 = Debug|x64 + {6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}.Debug|Win32.ActiveCfg = Debug|Win32 + {6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}.Debug|Win32.Build.0 = Debug|Win32 + {6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}.Debug|x64.ActiveCfg = Debug|x64 + {6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}.Debug|x64.Build.0 = Debug|x64 + {6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}.Release|Win32.ActiveCfg = Release|Win32 + {6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}.Release|Win32.Build.0 = Release|Win32 + {6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}.Release|x64.ActiveCfg = Release|x64 + {6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}.Release|x64.Build.0 = Release|x64 + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection +EndGlobal diff --git a/vc2005/cupstestppd.vcproj b/vc2005/cupstestppd.vcproj new file mode 100644 index 0000000000..479afcc1dd --- /dev/null +++ b/vc2005/cupstestppd.vcproj @@ -0,0 +1,356 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/vc2005/ipptool.vcproj b/vc2005/ipptool.vcproj new file mode 100644 index 0000000000..271187b542 --- /dev/null +++ b/vc2005/ipptool.vcproj @@ -0,0 +1,199 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/vc2005/libcups2.vcproj b/vc2005/libcups2.vcproj new file mode 100644 index 0000000000..049818580d --- /dev/null +++ b/vc2005/libcups2.vcproj @@ -0,0 +1,1611 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/vc2005/libcupsimage2.vcproj b/vc2005/libcupsimage2.vcproj new file mode 100644 index 0000000000..bb1f9efb4d --- /dev/null +++ b/vc2005/libcupsimage2.vcproj @@ -0,0 +1,395 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/vc2005/testfile.vcproj b/vc2005/testfile.vcproj new file mode 100644 index 0000000000..edebe6333d --- /dev/null +++ b/vc2005/testfile.vcproj @@ -0,0 +1,202 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/vc2005/testhttp.vcproj b/vc2005/testhttp.vcproj new file mode 100644 index 0000000000..77501a2d74 --- /dev/null +++ b/vc2005/testhttp.vcproj @@ -0,0 +1,202 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/vcnet/config.h b/vcnet/config.h new file mode 100644 index 0000000000..09c0ce184a --- /dev/null +++ b/vcnet/config.h @@ -0,0 +1,788 @@ +/* + * "$Id$" + * + * Configuration file for CUPS on Windows. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#ifndef _CUPS_CONFIG_H_ +#define _CUPS_CONFIG_H_ + +/* + * Include necessary headers... + */ + +#include +#include +#include +#include +#include +#include + + +/* + * Microsoft renames the POSIX functions to _name, and introduces + * a broken compatibility layer using the original names. As a result, + * random crashes can occur when, for example, strdup() allocates memory + * from a different heap than used by malloc() and free(). + * + * To avoid moronic problems like this, we #define the POSIX function + * names to the corresponding non-standard Microsoft names. + */ + +#define access _access +#define close _close +#define fileno _fileno +#define lseek _lseek +#define mkdir(d,p) _mkdir(d) +#define open _open +#define read _read +#define rmdir _rmdir +#define snprintf _snprintf +#define strdup _strdup +#define unlink _unlink +#define vsnprintf _vsnprintf +#define write _write + + +/* + * Map the POSIX sleep() and usleep() functions to the Win32 Sleep() function... + */ + +#define sleep(X) Sleep(1000 * (X)) +#define usleep(X) Sleep((X)/1000) + + +/* + * Map various parameters to Posix style system calls + */ + +# define F_OK 00 +# define W_OK 02 +# define R_OK 04 +# define O_RDONLY _O_RDONLY +# define O_WRONLY _O_WRONLY +# define O_CREATE _O_CREAT +# define O_TRUNC _O_TRUNC + + +/* + * Compiler stuff... + */ + +#undef const +#undef __CHAR_UNSIGNED__ + + +/* + * Version of software... + */ + +#define CUPS_SVERSION "CUPS v1.5.0" +#define CUPS_MINIMAL "CUPS/1.5.0" + + +/* + * Default user and groups... + */ + +#define CUPS_DEFAULT_USER "" +#define CUPS_DEFAULT_GROUP "" +#define CUPS_DEFAULT_SYSTEM_GROUPS "" +#define CUPS_DEFAULT_PRINTOPERATOR_AUTH "" + + +/* + * Default file permissions... + */ + +#define CUPS_DEFAULT_CONFIG_FILE_PERM 0644 +#define CUPS_DEFAULT_LOG_FILE_PERM 0644 + + +/* + * Default logging settings... + */ + +#define CUPS_DEFAULT_LOG_LEVEL "warn" +#define CUPS_DEFAULT_ACCESS_LOG_LEVEL "actions" + + +/* + * Default fatal error settings... + */ + +#define CUPS_DEFAULT_FATAL_ERRORS "config" + + +/* + * Default browsing settings... + */ + +#define CUPS_DEFAULT_BROWSING 1 +#define CUPS_DEFAULT_BROWSE_LOCAL_PROTOCOLS "CUPS dnssd" +#define CUPS_DEFAULT_BROWSE_REMOTE_PROTOCOLS "" +#define CUPS_DEFAULT_BROWSE_SHORT_NAMES 1 +#define CUPS_DEFAULT_DEFAULT_SHARED 1 +#define CUPS_DEFAULT_IMPLICIT_CLASSES 1 +#define CUPS_DEFAULT_USE_NETWORK_DEFAULT 0 + + +/* + * Default IPP port... + */ + +#define CUPS_DEFAULT_IPP_PORT 631 + + +/* + * Default printcap file... + */ + +#define CUPS_DEFAULT_PRINTCAP "" + + +/* + * Default Samba and LPD config files... + */ + +#define CUPS_DEFAULT_SMB_CONFIG_FILE "" +#define CUPS_DEFAULT_LPD_CONFIG_FILE "" + + +/* + * Default MaxCopies value... + */ + +#define CUPS_DEFAULT_MAX_COPIES 9999 + + +/* + * Do we have domain socket support? + */ + +#undef CUPS_DEFAULT_DOMAINSOCKET + + +/* + * Where are files stored? + * + * Note: These are defaults, which can be overridden by environment + * variables at run-time... + */ + +#define CUPS_BINDIR "C:/CUPS/bin" +#define CUPS_CACHEDIR "C:/CUPS/cache" +#define CUPS_DATADIR "C:/CUPS/share" +#define CUPS_DOCROOT "C:/CUPS/share/doc" +#define CUPS_FONTPATH "C:/CUPS/share/fonts" +#define CUPS_LOCALEDIR "C:/CUPS/locale" +#define CUPS_LOGDIR "C:/CUPS/logs" +#define CUPS_REQUESTS "C:/CUPS/spool" +#define CUPS_SBINDIR "C:/CUPS/sbin" +#define CUPS_SERVERBIN "C:/CUPS/lib" +#define CUPS_SERVERROOT "C:/CUPS/etc" +#define CUPS_STATEDIR "C:/CUPS/run" + + +/* + * Do we have various image libraries? + */ + +/* #undef HAVE_LIBPNG */ +/* #undef HAVE_LIBZ */ +/* #undef HAVE_LIBJPEG */ +/* #undef HAVE_LIBTIFF */ + + +/* + * Do we have PAM stuff? + */ + +#ifndef HAVE_LIBPAM +#define HAVE_LIBPAM 0 +#endif /* !HAVE_LIBPAM */ + +/* #undef HAVE_PAM_PAM_APPL_H */ +/* #undef HAVE_PAM_SET_ITEM */ +/* #undef HAVE_PAM_SETCRED */ + + +/* + * Do we have ? + */ + +/* #undef HAVE_SHADOW_H */ + + +/* + * Do we have ? + */ + +/* #undef HAVE_CRYPT_H */ + + +/* + * Do we have ? + */ + +/* #undef HAVE_SCSI_SG_H */ + + +/* + * Use , , and/or ? + */ + +#define HAVE_STRING_H 1 +/* #undef HAVE_STRINGS_H */ +/* #undef HAVE_BSTRING_H */ + + +/* + * Do we have the long long type? + */ + +/* #undef HAVE_LONG_LONG */ + +#ifdef HAVE_LONG_LONG +# define CUPS_LLFMT "%lld" +# define CUPS_LLCAST (long long) +#else +# define CUPS_LLFMT "%ld" +# define CUPS_LLCAST (long) +#endif /* HAVE_LONG_LONG */ + + +/* + * Do we have the strtoll() function? + */ + +/* #undef HAVE_STRTOLL */ + +#ifndef HAVE_STRTOLL +# define strtoll(nptr,endptr,base) strtol((nptr), (endptr), (base)) +#endif /* !HAVE_STRTOLL */ + + +/* + * Do we have the strXXX() functions? + */ + +#define HAVE_STRDUP 1 +/* #undef HAVE_STRLCAT */ +/* #undef HAVE_STRLCPY */ + + +/* + * Do we have the geteuid() function? + */ + +/* #undef HAVE_GETEUID */ + + +/* + * Do we have the setpgid() function? + */ + +/* #undef HAVE_SETPGID */ + + +/* + * Do we have the vsyslog() function? + */ + +/* #undef HAVE_VSYSLOG */ + + +/* + * Do we have the (v)snprintf() functions? + */ + +#define HAVE_SNPRINTF 1 +#define HAVE_VSNPRINTF 1 + + +/* + * What signal functions to use? + */ + +/* #undef HAVE_SIGSET */ +/* #undef HAVE_SIGACTION */ + + +/* + * What wait functions to use? + */ + +/* #undef HAVE_WAITPID */ +/* #undef HAVE_WAIT3 */ + + +/* + * Do we have the mallinfo function and malloc.h? + */ + +/* #undef HAVE_MALLINFO */ +/* #undef HAVE_MALLOC_H */ + + +/* + * Do we have the POSIX ACL functions? + */ + +/* #undef HAVE_ACL_INIT */ + + +/* + * Do we have the langinfo.h header file? + */ + +/* #undef HAVE_LANGINFO_H */ + + +/* + * Which encryption libraries do we have? + */ + +/* #undef HAVE_CDSASSL */ +/* #undef HAVE_GNUTLS */ +/* #undef HAVE_LIBSSL */ +#define HAVE_SSPISSL +#define HAVE_SSL + + +/* + * What Security framework headers do we have? + */ + +/* #undef HAVE_AUTHORIZATION_H */ +/* #undef HAVE_SECPOLICY_H */ +/* #undef HAVE_SECPOLICYPRIV_H */ +/* #undef HAVE_SECBASEPRIV_H */ +/* #undef HAVE_SECIDENTITYSEARCHPRIV_H */ + + +/* + * Do we have the SecIdentitySearchCreateWithPolicy function? + */ + +/* #undef HAVE_SECIDENTITYSEARCHCREATEWITHPOLICY */ + + +/* + * Do we have the SecPolicyCreateSSL function? + */ + +/* #undef HAVE_SECPOLICYCREATESSL */ + + +/* + * Do we have the SecPolicyCreateSSL function? + */ + +/* #undef HAVE_SECPOLICYCREATESSL */ + + +/* + * Do we have the cssmErrorString function? + */ + +/* #undef HAVE_CSSMERRORSTRING */ + + +/* + * Do we have the SLP library? + */ + +/* #undef HAVE_LIBSLP */ + + +/* + * Do we have an LDAP library? + */ + +/* #undef HAVE_LDAP */ +/* #undef HAVE_OPENLDAP */ +/* #undef HAVE_MOZILLA_LDAP */ +/* #undef HAVE_LDAP_SSL_H */ +/* #undef HAVE_LDAP_SSL */ +/* #undef HAVE_LDAP_REBIND_PROC */ + + +/* + * Do we have libpaper? + */ + +/* #undef HAVE_LIBPAPER */ + + +/* + * Do we have DNS Service Discovery (aka Bonjour)? + */ + +#define HAVE_DNSSD 1 + + +/* + * Does the "stat" structure contain the "st_gen" member? + */ + +/* #undef HAVE_ST_GEN */ + + +/* + * Do we have ? + */ + +/* #undef HAVE_SYS_IOCTL_H */ + + +/* + * Does the "tm" structure contain the "tm_gmtoff" member? + */ + +/* #undef HAVE_TM_GMTOFF */ + + +/* + * Do we have rresvport_af()? + */ + +/* #undef HAVE_RRESVPORT_AF */ + + +/* + * Do we have getaddrinfo()? + */ + +#define HAVE_GETADDRINFO 1 + + +/* + * Do we have getnameinfo()? + */ + +#define HAVE_GETNAMEINFO 1 + + +/* + * Do we have getifaddrs()? + */ + +/* #undef HAVE_GETIFADDRS */ + + +/* + * Do we have hstrerror()? + */ + +/* #undef HAVE_HSTRERROR */ + + +/* + * Do we have res_init()? + */ + +/* #undef HAVE_RES_INIT */ + + +/* + * Do we have + */ + +/* #undef HAVE_RESOLV_H */ + + +/* + * Do we have the header file? + */ + +/* #undef HAVE_SYS_SOCKIO_H */ + + +/* + * Does the sockaddr structure contain an sa_len parameter? + */ + +/* #undef HAVE_STRUCT_SOCKADDR_SA_LEN */ + + +/* + * Do we have the AIX usersec.h header file? + */ + +/* #undef HAVE_USERSEC_H */ + + +/* + * Do we have pthread support? + */ + +/* #undef HAVE_PTHREAD_H */ + + +/* + * Do we have launchd support? + */ + +/* #undef HAVE_LAUNCH_H */ +/* #undef HAVE_LAUNCHD */ + + +/* + * Various scripting languages... + */ + +/* #undef HAVE_JAVA */ +#define CUPS_JAVA "" +/* #undef HAVE_PERL */ +#define CUPS_PERL "" +/* #undef HAVE_PHP */ +#define CUPS_PHP "" +/* #undef HAVE_PYTHON */ +#define CUPS_PYTHON "" + + +/* + * Location of the poppler/Xpdf pdftops program... + */ + +/* #undef HAVE_PDFTOPS */ +#define CUPS_PDFTOPS "" + + +/* + * Location of the Ghostscript gs program... + */ + +/* #undef HAVE_GHOSTSCRIPT */ +#define CUPS_GHOSTSCRIPT "" + + +/* + * Do we have Darwin's CoreFoundation and SystemConfiguration frameworks? + */ + +/* #undef HAVE_COREFOUNDATION */ +/* #undef HAVE_SYSTEMCONFIGURATION */ + + +/* + * Do we have CoreFoundation public and private headers? + */ + +/* #undef HAVE_COREFOUNDATION_H */ +/* #undef HAVE_CFPRIV_H */ +/* #undef HAVE_CFBUNDLEPRIV_H */ + + +/* + * Do we have ApplicationServices public headers? + */ + +/* #undef HAVE_APPLICATIONSERVICES_H */ + + +/* + * Do we have the SCDynamicStoreCopyComputerName function? + */ + +/* #undef HAVE_SCDYNAMICSTORECOPYCOMPUTERNAME */ + + +/* + * Do we have Mac OS X 10.4's mbr_XXX functions? + */ + +/* #undef HAVE_MEMBERSHIP_H */ +/* #undef HAVE_MEMBERSHIPPRIV_H */ +/* #undef HAVE_MBR_UID_TO_UUID */ + + +/* + * Do we have Darwin's notify_post header and function? + */ + +/* #undef HAVE_NOTIFY_H */ +/* #undef HAVE_NOTIFY_POST */ + + +/* + * Do we have Darwin's IOKit private headers? + */ + +/* #undef HAVE_IOKIT_PWR_MGT_IOPMLIBPRIVATE_H */ + + +/* + * Do we have DBUS? + */ + +/* #undef HAVE_DBUS */ +/* #undef HAVE_DBUS_MESSAGE_ITER_INIT_APPEND */ + + +/* + * Do we have the GSSAPI support library (for Kerberos support)? + */ + +/* #undef HAVE_GSS_ACQUIRE_CRED_EX_F */ +/* #undef HAVE_GSS_C_NT_HOSTBASED_SERVICE */ +/* #undef HAVE_GSS_GSSAPI_H */ +/* #undef HAVE_GSS_GSSAPI_SPI_H */ +/* #undef HAVE_GSSAPI */ +/* #undef HAVE_GSSAPI_GENERIC_H */ +/* #undef HAVE_GSSAPI_GSSAPI_H */ +/* #undef HAVE_GSSAPI_H */ +/* #undef HAVE_GSSAPI_KRB5_H */ +/* #undef HAVE_KRB5_H */ + + +/* + * Default GSS service name... + */ + +#define CUPS_DEFAULT_GSSSERVICENAME "host" + + +/* + * Select/poll interfaces... + */ + +/* #undef HAVE_POLL */ +/* #undef HAVE_EPOLL */ +/* #undef HAVE_KQUEUE */ + + +/* + * Do we have the header? + */ + +/* #undef HAVE_DLFCN_H */ + + +/* + * Do we have ? + */ + +/* #undef HAVE_SYS_PARAM_H */ + + +/* + * Do we have ? + */ + +/* #undef HAVE_SYS_UCRED_H */ + + +/* + * Do we have removefile()? + */ + +/* #undef HAVE_REMOVEFILE */ + + +/* + * Do we have ? + */ + +/* #undef HAVE_SANDBOX_H */ + + +/* + * Which random number generator function to use... + */ + +/* #undef HAVE_ARC4RANDOM */ +/* #undef HAVE_RANDOM */ +/* #undef HAVE_LRAND48 */ + +#ifdef HAVE_ARC4RANDOM +# define CUPS_RAND() arc4random() +# define CUPS_SRAND(v) arc4random_stir() +#elif defined(HAVE_RANDOM) +# define CUPS_RAND() random() +# define CUPS_SRAND(v) srandom(v) +#elif defined(HAVE_LRAND48) +# define CUPS_RAND() lrand48() +# define CUPS_SRAND(v) srand48(v) +#else +# define CUPS_RAND() rand() +# define CUPS_SRAND(v) srand(v) +#endif /* HAVE_ARC4RANDOM */ + + +/* + * Do we have vproc_transaction_begin/end? + */ + +/* #undef HAVE_VPROC_TRANSACTION_BEGIN */ + + +/* + * Do we have libusb? + */ + +/* #undef HAVE_USB_H */ + + +/* + * Do we have libwrap and tcpd.h? + */ + +/* #undef HAVE_TCPD_H */ + + +/* + * Do we have ? + */ + +/* #undef HAVE_ICONV_H */ + + +/* + * Do we have statfs or statvfs and one of the corresponding headers? + */ + +/* #undef HAVE_STATFS */ +/* #undef HAVE_STATVFS */ +/* #undef HAVE_SYS_MOUNT_H */ +/* #undef HAVE_SYS_STATFS_H */ +/* #undef HAVE_SYS_STATVFS_H */ +/* #undef HAVE_SYS_VFS_H */ + + +/* + * Location of Mac OS X localization bundle, if any. + */ + +/* #undef CUPS_BUNDLEDIR */ + + +/* + * Do we have the ColorSyncRegisterDevice function? + */ + +/* #undef HAVE_COLORSYNCREGISTERDEVICE */ + + +/* + * Do we have XPC? + */ + +/* #undef HAVE_XPC */ + + +#endif /* !_CUPS_CONFIG_H_ */ + +/* + * End of "$Id$". + */ diff --git a/vcnet/cups.sln b/vcnet/cups.sln new file mode 100644 index 0000000000..45d459319e --- /dev/null +++ b/vcnet/cups.sln @@ -0,0 +1,101 @@ +Microsoft Visual Studio Solution File, Format Version 10.00 +# Visual Studio 2008 +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "libcups2", "libcups2.vcproj", "{CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}" +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "libcupsimage2", "libcupsimage2.vcproj", "{CB4AA6F2-3E84-45BE-B505-95CD375E1234}" + ProjectSection(ProjectDependencies) = postProject + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} = {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "testfile", "testfile.vcproj", "{CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}" + ProjectSection(ProjectDependencies) = postProject + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} = {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "testhttp", "testhttp.vcproj", "{90B0058C-8393-411F-BD3B-E2C831D4E883}" + ProjectSection(ProjectDependencies) = postProject + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} = {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "cupstestppd", "cupstestppd.vcproj", "{6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}" + ProjectSection(ProjectDependencies) = postProject + {CB4AA6F2-3E84-45BE-B505-95CD375E1234} = {CB4AA6F2-3E84-45BE-B505-95CD375E1234} + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} = {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "ipptool", "ipptool.vcproj", "{B246D91E-61F2-4433-BFD2-6C2A96FBD4D4}" + ProjectSection(ProjectDependencies) = postProject + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} = {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3} + EndProjectSection +EndProject +Project("{54435603-DBB4-11D2-8724-00A0C9A8B90C}") = "ipptool-installer", "ipptool-installer.vdproj", "{6AEA263B-92C0-426F-B5FF-F7F5917B704A}" +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|Win32 = Debug|Win32 + Debug|x64 = Debug|x64 + Release|Win32 = Release|Win32 + Release|x64 = Release|x64 + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}.Debug|Win32.ActiveCfg = Debug|Win32 + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}.Debug|Win32.Build.0 = Debug|Win32 + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}.Debug|x64.ActiveCfg = Debug|x64 + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}.Debug|x64.Build.0 = Debug|x64 + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}.Release|Win32.ActiveCfg = Release|Win32 + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}.Release|Win32.Build.0 = Release|Win32 + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}.Release|x64.ActiveCfg = Debug|x64 + {CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}.Release|x64.Build.0 = Debug|x64 + {CB4AA6F2-3E84-45BE-B505-95CD375E1234}.Debug|Win32.ActiveCfg = Debug|Win32 + {CB4AA6F2-3E84-45BE-B505-95CD375E1234}.Debug|Win32.Build.0 = Debug|Win32 + {CB4AA6F2-3E84-45BE-B505-95CD375E1234}.Debug|x64.ActiveCfg = Debug|x64 + {CB4AA6F2-3E84-45BE-B505-95CD375E1234}.Debug|x64.Build.0 = Debug|x64 + {CB4AA6F2-3E84-45BE-B505-95CD375E1234}.Release|Win32.ActiveCfg = Release|Win32 + {CB4AA6F2-3E84-45BE-B505-95CD375E1234}.Release|Win32.Build.0 = Release|Win32 + {CB4AA6F2-3E84-45BE-B505-95CD375E1234}.Release|x64.ActiveCfg = Release|x64 + {CB4AA6F2-3E84-45BE-B505-95CD375E1234}.Release|x64.Build.0 = Release|x64 + {CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}.Debug|Win32.ActiveCfg = Debug|Win32 + {CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}.Debug|Win32.Build.0 = Debug|Win32 + {CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}.Debug|x64.ActiveCfg = Debug|x64 + {CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}.Debug|x64.Build.0 = Debug|x64 + {CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}.Release|Win32.ActiveCfg = Release|Win32 + {CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}.Release|Win32.Build.0 = Release|Win32 + {CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}.Release|x64.ActiveCfg = Debug|x64 + {CE75FC5F-E0CF-45DC-AD27-84666D3FBA30}.Release|x64.Build.0 = Debug|x64 + {90B0058C-8393-411F-BD3B-E2C831D4E883}.Debug|Win32.ActiveCfg = Debug|Win32 + {90B0058C-8393-411F-BD3B-E2C831D4E883}.Debug|Win32.Build.0 = Debug|Win32 + {90B0058C-8393-411F-BD3B-E2C831D4E883}.Debug|x64.ActiveCfg = Debug|x64 + {90B0058C-8393-411F-BD3B-E2C831D4E883}.Debug|x64.Build.0 = Debug|x64 + {90B0058C-8393-411F-BD3B-E2C831D4E883}.Release|Win32.ActiveCfg = Release|Win32 + {90B0058C-8393-411F-BD3B-E2C831D4E883}.Release|Win32.Build.0 = Release|Win32 + {90B0058C-8393-411F-BD3B-E2C831D4E883}.Release|x64.ActiveCfg = Debug|x64 + {90B0058C-8393-411F-BD3B-E2C831D4E883}.Release|x64.Build.0 = Debug|x64 + {6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}.Debug|Win32.ActiveCfg = Debug|Win32 + {6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}.Debug|Win32.Build.0 = Debug|Win32 + {6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}.Debug|x64.ActiveCfg = Debug|x64 + {6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}.Debug|x64.Build.0 = Debug|x64 + {6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}.Release|Win32.ActiveCfg = Release|Win32 + {6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}.Release|Win32.Build.0 = Release|Win32 + {6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}.Release|x64.ActiveCfg = Release|x64 + {6BE0CDD3-4ED7-409C-A80F-19DF73664B1F}.Release|x64.Build.0 = Release|x64 + {B246D91E-61F2-4433-BFD2-6C2A96FBD4D4}.Debug|Win32.ActiveCfg = Debug|Win32 + {B246D91E-61F2-4433-BFD2-6C2A96FBD4D4}.Debug|Win32.Build.0 = Debug|Win32 + {B246D91E-61F2-4433-BFD2-6C2A96FBD4D4}.Debug|x64.ActiveCfg = Debug|x64 + {B246D91E-61F2-4433-BFD2-6C2A96FBD4D4}.Debug|x64.Build.0 = Debug|x64 + {B246D91E-61F2-4433-BFD2-6C2A96FBD4D4}.Release|Win32.ActiveCfg = Release|Win32 + {B246D91E-61F2-4433-BFD2-6C2A96FBD4D4}.Release|Win32.Build.0 = Release|Win32 + {B246D91E-61F2-4433-BFD2-6C2A96FBD4D4}.Release|x64.ActiveCfg = Release|x64 + {B246D91E-61F2-4433-BFD2-6C2A96FBD4D4}.Release|x64.Build.0 = Release|x64 + {6AEA263B-92C0-426F-B5FF-F7F5917B704A}.Debug|Win32.ActiveCfg = Debug + {6AEA263B-92C0-426F-B5FF-F7F5917B704A}.Debug|Win32.Build.0 = Debug + {6AEA263B-92C0-426F-B5FF-F7F5917B704A}.Debug|x64.ActiveCfg = Debug + {6AEA263B-92C0-426F-B5FF-F7F5917B704A}.Debug|x64.Build.0 = Debug + {6AEA263B-92C0-426F-B5FF-F7F5917B704A}.Release|Win32.ActiveCfg = Release + {6AEA263B-92C0-426F-B5FF-F7F5917B704A}.Release|Win32.Build.0 = Release + {6AEA263B-92C0-426F-B5FF-F7F5917B704A}.Release|x64.ActiveCfg = Release + {6AEA263B-92C0-426F-B5FF-F7F5917B704A}.Release|x64.Build.0 = Release + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection +EndGlobal diff --git a/vcnet/cupstestppd.vcproj b/vcnet/cupstestppd.vcproj new file mode 100644 index 0000000000..4739f37199 --- /dev/null +++ b/vcnet/cupstestppd.vcproj @@ -0,0 +1,353 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/vcnet/ipptool-installer.vdproj b/vcnet/ipptool-installer.vdproj new file mode 100644 index 0000000000..21f6b25b8c --- /dev/null +++ b/vcnet/ipptool-installer.vdproj @@ -0,0 +1,1577 @@ +"DeployProject" +{ +"VSVersion" = "3:800" +"ProjectType" = "8:{978C614F-708E-4E1A-B201-565925725DBA}" +"IsWebType" = "8:FALSE" +"ProjectName" = "8:ipptool-installer" +"LanguageId" = "3:1033" +"CodePage" = "3:1252" +"UILanguageId" = "3:1033" +"SccProjectName" = "8:" +"SccLocalPath" = "8:" +"SccAuxPath" = "8:" +"SccProvider" = "8:" + "Hierarchy" + { + "Entry" + { + "MsmKey" = "8:_009DC3ED78164048934ED35E89A0A1C5" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_1A1324305D78463BBFC62269C56DCF0B" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_1E4B2A9BD6A44926B719E0D7E8FC0952" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_40C483299484486C9416B704F0EF4A6C" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_41260853156249FDB39A8E386F935492" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_425F2537D86148B9A4233AD27D426738" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_65CBD149C9DA448FBACE2B02766A6537" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_6F893B2A3B7048CBA39359FC368BCA27" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_80DBA85C961E447CB92C95CE1F38C56B" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_842C04D73EBC4F5DBC2FD58D5B98D5D1" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_8AC6B9D7EF6B4C7A8C7B5AD85AA397C3" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_8CC3562BCDED4676A672115D19D898B5" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_8E8C3CEE3CE746F0B1D2DAE5DBB941F1" + "OwnerKey" = "8:_E97571D3FBE048DABDC59B37762D800F" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_8E8C3CEE3CE746F0B1D2DAE5DBB941F1" + "OwnerKey" = "8:_EA282F32A10B4ED1A81AA6133B997C6A" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_9967519E7058965D4C7DEF47EB39CC50" + "OwnerKey" = "8:_EA282F32A10B4ED1A81AA6133B997C6A" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_9FBF78D7B89EEA843380D5F10E1954D7" + "OwnerKey" = "8:_EA282F32A10B4ED1A81AA6133B997C6A" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_AF057921D20E4520A3C6420F0729A744" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_B9E79062FEF64745915546DDD5BF8D85" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_C430646D6E7C4CBDA84F951AE95EB76F" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_D4BEA026ABFB46DD960AD8EFCAE45E31" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_D5CD9D9AB1644688A1D54B1589BDF724" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_DAED4A66293E4B81B094ACBF3266D11C" + "OwnerKey" = "8:_8E8C3CEE3CE746F0B1D2DAE5DBB941F1" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_E30823E684384F4DB7385B14F714AD95" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_E3C6C5A7FDD94965B68960844461D5EA" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_E4C0F67094B94E05AB69787080727089" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_E50F2FB950DD47D993DC8FB577266549" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_E7DD6A8D195B4752B0525628FF9586E4" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_E97571D3FBE048DABDC59B37762D800F" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_E9CA670571B5447682529CDB7BC3C100" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_EA282F32A10B4ED1A81AA6133B997C6A" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_EA9CE1B09EF1442BAA3C3F5AE7865ABF" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_EC7FE3265BDB4C52B5B87A8AC56AFDDD" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_F16FA7F9826E461E955A95B2EEABA975" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + "Entry" + { + "MsmKey" = "8:_F22F5380A6E14A43A15A452C7C6F6C07" + "OwnerKey" = "8:_UNDEFINED" + "MsmSig" = "8:_UNDEFINED" + } + } + "Configurations" + { + "Debug" + { + "DisplayName" = "8:Debug" + "IsDebugOnly" = "11:TRUE" + "IsReleaseOnly" = "11:FALSE" + "OutputFilename" = "8:ipptool-windows.msi" + "PackageFilesAs" = "3:2" + "PackageFileSize" = "3:-2147483648" + "CabType" = "3:1" + "Compression" = "3:3" + "SignOutput" = "11:FALSE" + "CertificateFile" = "8:" + "PrivateKeyFile" = "8:" + "TimeStampServer" = "8:" + "InstallerBootstrapper" = "3:2" + "BootstrapperCfg:{63ACBE69-63AA-4F98-B2B6-99F9E24495F2}" + { + "Enabled" = "11:TRUE" + "PromptEnabled" = "11:TRUE" + "PrerequisitesLocation" = "2:1" + "Url" = "8:" + "ComponentsUrl" = "8:" + "Items" + { + "{EDC2488A-8267-493A-A98E-7D9C3B36CDF3}:Microsoft.Net.Framework.3.5.SP1" + { + "Name" = "8:.NET Framework 3.5 SP1" + "ProductCode" = "8:Microsoft.Net.Framework.3.5.SP1" + } + "{EDC2488A-8267-493A-A98E-7D9C3B36CDF3}:Microsoft.Windows.Installer.3.1" + { + "Name" = "8:Windows Installer 3.1" + "ProductCode" = "8:Microsoft.Windows.Installer.3.1" + } + } + } + } + "Release" + { + "DisplayName" = "8:Release" + "IsDebugOnly" = "11:FALSE" + "IsReleaseOnly" = "11:TRUE" + "OutputFilename" = "8:ipptool-windows.msi" + "PackageFilesAs" = "3:2" + "PackageFileSize" = "3:-2147483648" + "CabType" = "3:1" + "Compression" = "3:3" + "SignOutput" = "11:FALSE" + "CertificateFile" = "8:" + "PrivateKeyFile" = "8:" + "TimeStampServer" = "8:" + "InstallerBootstrapper" = "3:2" + "BootstrapperCfg:{63ACBE69-63AA-4F98-B2B6-99F9E24495F2}" + { + "Enabled" = "11:TRUE" + "PromptEnabled" = "11:TRUE" + "PrerequisitesLocation" = "2:1" + "Url" = "8:" + "ComponentsUrl" = "8:" + "Items" + { + "{EDC2488A-8267-493A-A98E-7D9C3B36CDF3}:Microsoft.Net.Framework.3.5.SP1" + { + "Name" = "8:.NET Framework 3.5 SP1" + "ProductCode" = "8:Microsoft.Net.Framework.3.5.SP1" + } + "{EDC2488A-8267-493A-A98E-7D9C3B36CDF3}:Microsoft.Windows.Installer.3.1" + { + "Name" = "8:Windows Installer 3.1" + "ProductCode" = "8:Microsoft.Windows.Installer.3.1" + } + } + } + } + } + "Deployable" + { + "CustomAction" + { + } + "DefaultFeature" + { + "Name" = "8:DefaultFeature" + "Title" = "8:" + "Description" = "8:" + } + "ExternalPersistence" + { + "LaunchCondition" + { + } + } + "File" + { + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_009DC3ED78164048934ED35E89A0A1C5" + { + "SourcePath" = "8:..\\test\\document-a4.ps" + "TargetName" = "8:document-a4.ps" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_1A1324305D78463BBFC62269C56DCF0B" + { + "SourcePath" = "8:..\\doc\\help\\man-ipptoolfile.html" + "TargetName" = "8:man-ipptoolfile.html" + "Tag" = "8:" + "Folder" = "8:_FFAEDEA1D38D4088A03FDD6F17E2CA5D" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_1E4B2A9BD6A44926B719E0D7E8FC0952" + { + "SourcePath" = "8:..\\test\\get-printer-attributes.test" + "TargetName" = "8:get-printer-attributes.test" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_40C483299484486C9416B704F0EF4A6C" + { + "SourcePath" = "8:..\\test\\testfile.pdf" + "TargetName" = "8:testfile.pdf" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_41260853156249FDB39A8E386F935492" + { + "SourcePath" = "8:..\\test\\ipp-1.1.test" + "TargetName" = "8:ipp-1.1.test" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_425F2537D86148B9A4233AD27D426738" + { + "SourcePath" = "8:..\\test\\ipp-2.0.test" + "TargetName" = "8:ipp-2.0.test" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_65CBD149C9DA448FBACE2B02766A6537" + { + "SourcePath" = "8:..\\test\\testfile.ps" + "TargetName" = "8:testfile.ps" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_6F893B2A3B7048CBA39359FC368BCA27" + { + "SourcePath" = "8:..\\test\\onepage-letter.ps" + "TargetName" = "8:onepage-letter.ps" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_80DBA85C961E447CB92C95CE1F38C56B" + { + "SourcePath" = "8:..\\test\\ipp-2.2.test" + "TargetName" = "8:ipp-2.2.test" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_842C04D73EBC4F5DBC2FD58D5B98D5D1" + { + "SourcePath" = "8:..\\test\\document-a4.pdf" + "TargetName" = "8:document-a4.pdf" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_8AC6B9D7EF6B4C7A8C7B5AD85AA397C3" + { + "SourcePath" = "8:..\\test\\get-jobs.test" + "TargetName" = "8:get-jobs.test" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_8CC3562BCDED4676A672115D19D898B5" + { + "SourcePath" = "8:..\\test\\testfile.jpg" + "TargetName" = "8:testfile.jpg" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_9967519E7058965D4C7DEF47EB39CC50" + { + "SourcePath" = "8:CRYPT32.dll" + "TargetName" = "8:CRYPT32.dll" + "Tag" = "8:" + "Folder" = "8:_FFAEDEA1D38D4088A03FDD6F17E2CA5D" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:TRUE" + "IsDependency" = "11:TRUE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_9FBF78D7B89EEA843380D5F10E1954D7" + { + "SourcePath" = "8:Secur32.dll" + "TargetName" = "8:Secur32.dll" + "Tag" = "8:" + "Folder" = "8:_FFAEDEA1D38D4088A03FDD6F17E2CA5D" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:TRUE" + "IsDependency" = "11:TRUE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_AF057921D20E4520A3C6420F0729A744" + { + "SourcePath" = "8:..\\IPPTOOL.txt" + "TargetName" = "8:IPPTOOL.txt" + "Tag" = "8:" + "Folder" = "8:_FFAEDEA1D38D4088A03FDD6F17E2CA5D" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_B9E79062FEF64745915546DDD5BF8D85" + { + "SourcePath" = "8:..\\CHANGES-IPPTOOL.txt" + "TargetName" = "8:CHANGES-IPPTOOL.txt" + "Tag" = "8:" + "Folder" = "8:_FFAEDEA1D38D4088A03FDD6F17E2CA5D" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_C430646D6E7C4CBDA84F951AE95EB76F" + { + "SourcePath" = "8:..\\test\\color.jpg" + "TargetName" = "8:color.jpg" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_D4BEA026ABFB46DD960AD8EFCAE45E31" + { + "SourcePath" = "8:..\\test\\onepage-letter.pdf" + "TargetName" = "8:onepage-letter.pdf" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_D5CD9D9AB1644688A1D54B1589BDF724" + { + "SourcePath" = "8:..\\doc\\help\\man-ipptool.html" + "TargetName" = "8:man-ipptool.html" + "Tag" = "8:" + "Folder" = "8:_FFAEDEA1D38D4088A03FDD6F17E2CA5D" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_E30823E684384F4DB7385B14F714AD95" + { + "SourcePath" = "8:..\\test\\ipp-2.1.test" + "TargetName" = "8:ipp-2.1.test" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_E3C6C5A7FDD94965B68960844461D5EA" + { + "SourcePath" = "8:..\\LICENSE.txt" + "TargetName" = "8:LICENSE.txt" + "Tag" = "8:" + "Folder" = "8:_FFAEDEA1D38D4088A03FDD6F17E2CA5D" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_E4C0F67094B94E05AB69787080727089" + { + "SourcePath" = "8:..\\test\\create-printer-subscription.test" + "TargetName" = "8:create-printer-subscription.test" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_E50F2FB950DD47D993DC8FB577266549" + { + "SourcePath" = "8:..\\test\\document-letter.ps" + "TargetName" = "8:document-letter.ps" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_E7DD6A8D195B4752B0525628FF9586E4" + { + "SourcePath" = "8:..\\test\\onepage-a4.ps" + "TargetName" = "8:onepage-a4.ps" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_E9CA670571B5447682529CDB7BC3C100" + { + "SourcePath" = "8:..\\test\\gray.jpg" + "TargetName" = "8:gray.jpg" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_EA9CE1B09EF1442BAA3C3F5AE7865ABF" + { + "SourcePath" = "8:..\\test\\testfile.txt" + "TargetName" = "8:testfile.txt" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_EC7FE3265BDB4C52B5B87A8AC56AFDDD" + { + "SourcePath" = "8:..\\test\\get-completed-jobs.test" + "TargetName" = "8:get-completed-jobs.test" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_F16FA7F9826E461E955A95B2EEABA975" + { + "SourcePath" = "8:..\\test\\onepage-a4.pdf" + "TargetName" = "8:onepage-a4.pdf" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + "{1FB2D0AE-D3B9-43D4-B9DD-F88EC61E35DE}:_F22F5380A6E14A43A15A452C7C6F6C07" + { + "SourcePath" = "8:..\\test\\document-letter.pdf" + "TargetName" = "8:document-letter.pdf" + "Tag" = "8:" + "Folder" = "8:_EB00D0298C7E441EBD0257AC04FB3560" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + } + } + "FileType" + { + } + "Folder" + { + "{1525181F-901A-416C-8A58-119130FE478E}:_BEC0EAE20C954C78B294B83E6696156E" + { + "Name" = "8:#1919" + "AlwaysCreate" = "11:FALSE" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Property" = "8:ProgramMenuFolder" + "Folders" + { + } + } + "{1525181F-901A-416C-8A58-119130FE478E}:_D02CDADE99F344CF92CA1A8D0278861F" + { + "Name" = "8:#1916" + "AlwaysCreate" = "11:FALSE" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Property" = "8:DesktopFolder" + "Folders" + { + } + } + "{3C67513D-01DD-4637-8A68-80971EB9504F}:_FFAEDEA1D38D4088A03FDD6F17E2CA5D" + { + "DefaultLocation" = "8:[ProgramFilesFolder]\\ipptool" + "Name" = "8:#1925" + "AlwaysCreate" = "11:FALSE" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Property" = "8:TARGETDIR" + "Folders" + { + "{9EF0B969-E518-4E46-987F-47570745A589}:_EB00D0298C7E441EBD0257AC04FB3560" + { + "Name" = "8:ipptool" + "AlwaysCreate" = "11:FALSE" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Property" = "8:_6F223FB51798428A9F2D64A5A7F2B49C" + "Folders" + { + } + } + } + } + } + "LaunchCondition" + { + } + "Locator" + { + } + "MsiBootstrapper" + { + "LangId" = "3:1033" + "RequiresElevation" = "11:FALSE" + } + "Product" + { + "Name" = "8:Microsoft Visual Studio" + "ProductName" = "8:ipptool for Windows" + "ProductCode" = "8:{DD8D418A-BF66-4B07-B944-4914E7F8E5E9}" + "PackageCode" = "8:{23839540-E821-4A65-95D0-7DD3512E362C}" + "UpgradeCode" = "8:{BAB6EBBB-515D-4155-9FEF-D98DA76814CA}" + "RestartWWWService" = "11:FALSE" + "RemovePreviousVersions" = "11:TRUE" + "DetectNewerInstalledVersion" = "11:TRUE" + "InstallAllUsers" = "11:TRUE" + "ProductVersion" = "8:12.02.0200" + "Manufacturer" = "8:Apple Inc." + "ARPHELPTELEPHONE" = "8:" + "ARPHELPLINK" = "8:http://www.cups.org/str.php" + "Title" = "8:ipptool" + "Subject" = "8:" + "ARPCONTACT" = "8:Apple Inc." + "Keywords" = "8:IPP, Internet Printing Protocol" + "ARPCOMMENTS" = "8:ipptool for Windows" + "ARPURLINFOABOUT" = "8:http://www.cups.org/" + "ARPPRODUCTICON" = "8:" + "ARPIconIndex" = "3:0" + "SearchPath" = "8:" + "UseSystemSearchPath" = "11:TRUE" + "TargetPlatform" = "3:0" + "PreBuildEvent" = "8:" + "PostBuildEvent" = "8:" + "RunPostBuildEvent" = "3:0" + } + "Registry" + { + "HKLM" + { + "Keys" + { + "{60EA8692-D2D5-43EB-80DC-7906BF13D6EF}:_BC4E66686BCA4F9A8B24B6CF2728DACD" + { + "Name" = "8:Software" + "Condition" = "8:" + "AlwaysCreate" = "11:FALSE" + "DeleteAtUninstall" = "11:FALSE" + "Transitive" = "11:FALSE" + "Keys" + { + "{60EA8692-D2D5-43EB-80DC-7906BF13D6EF}:_F9AB9B310C7545D993D690F529048AA2" + { + "Name" = "8:cups.org" + "Condition" = "8:" + "AlwaysCreate" = "11:FALSE" + "DeleteAtUninstall" = "11:FALSE" + "Transitive" = "11:FALSE" + "Keys" + { + } + "Values" + { + "{ADCFDA98-8FDD-45E4-90BC-E3D20B029870}:_4E5BAC705A1D44E78C90C6D2A4A7BE20" + { + "Name" = "8:installdir" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "ValueTypes" = "3:2" + "Value" = "8:[TARGETDIR]" + } + } + } + } + "Values" + { + } + } + } + } + "HKCU" + { + "Keys" + { + "{60EA8692-D2D5-43EB-80DC-7906BF13D6EF}:_A4C9879F42874B6B92960A55F2D98922" + { + "Name" = "8:Software" + "Condition" = "8:" + "AlwaysCreate" = "11:FALSE" + "DeleteAtUninstall" = "11:FALSE" + "Transitive" = "11:FALSE" + "Keys" + { + "{60EA8692-D2D5-43EB-80DC-7906BF13D6EF}:_1ACB03C307FB4B85BB27C9913FB58B09" + { + "Name" = "8:[Manufacturer]" + "Condition" = "8:" + "AlwaysCreate" = "11:FALSE" + "DeleteAtUninstall" = "11:FALSE" + "Transitive" = "11:FALSE" + "Keys" + { + } + "Values" + { + } + } + } + "Values" + { + } + } + } + } + "HKCR" + { + "Keys" + { + } + } + "HKU" + { + "Keys" + { + } + } + "HKPU" + { + "Keys" + { + } + } + } + "Sequences" + { + } + "Shortcut" + { + } + "UserInterface" + { + "{2479F3F5-0309-486D-8047-8187E2CE5BA0}:_045DF90B1FF941A9BA7A742CFC0A6C00" + { + "UseDynamicProperties" = "11:FALSE" + "IsDependency" = "11:FALSE" + "SourcePath" = "8:\\VsdUserInterface.wim" + } + "{DF760B10-853B-4699-99F2-AFF7185B4A62}:_345BD86390E841A98B74ED3E07945F8C" + { + "Name" = "8:#1900" + "Sequence" = "3:2" + "Attributes" = "3:1" + "Dialogs" + { + "{688940B3-5CA9-4162-8DEE-2993FA9D8CBC}:_067C143A8731427180B1568AF8C07375" + { + "Sequence" = "3:200" + "DisplayName" = "8:Installation Folder" + "UseDynamicProperties" = "11:TRUE" + "IsDependency" = "11:FALSE" + "SourcePath" = "8:\\VsdAdminFolderDlg.wid" + "Properties" + { + "BannerBitmap" + { + "Name" = "8:BannerBitmap" + "DisplayName" = "8:#1001" + "Description" = "8:#1101" + "Type" = "3:8" + "ContextData" = "8:Bitmap" + "Attributes" = "3:4" + "Setting" = "3:1" + "UsePlugInResources" = "11:TRUE" + } + } + } + "{688940B3-5CA9-4162-8DEE-2993FA9D8CBC}:_A2F2CC6EB9D7453599E7598D4D0629A5" + { + "Sequence" = "3:300" + "DisplayName" = "8:Confirm Installation" + "UseDynamicProperties" = "11:TRUE" + "IsDependency" = "11:FALSE" + "SourcePath" = "8:\\VsdAdminConfirmDlg.wid" + "Properties" + { + "BannerBitmap" + { + "Name" = "8:BannerBitmap" + "DisplayName" = "8:#1001" + "Description" = "8:#1101" + "Type" = "3:8" + "ContextData" = "8:Bitmap" + "Attributes" = "3:4" + "Setting" = "3:1" + "UsePlugInResources" = "11:TRUE" + } + } + } + "{688940B3-5CA9-4162-8DEE-2993FA9D8CBC}:_C2B41369B7334F419318792EBA031412" + { + "Sequence" = "3:100" + "DisplayName" = "8:Welcome" + "UseDynamicProperties" = "11:TRUE" + "IsDependency" = "11:FALSE" + "SourcePath" = "8:\\VsdAdminWelcomeDlg.wid" + "Properties" + { + "BannerBitmap" + { + "Name" = "8:BannerBitmap" + "DisplayName" = "8:#1001" + "Description" = "8:#1101" + "Type" = "3:8" + "ContextData" = "8:Bitmap" + "Attributes" = "3:4" + "Setting" = "3:1" + "UsePlugInResources" = "11:TRUE" + } + "CopyrightWarning" + { + "Name" = "8:CopyrightWarning" + "DisplayName" = "8:#1002" + "Description" = "8:#1102" + "Type" = "3:3" + "ContextData" = "8:" + "Attributes" = "3:0" + "Setting" = "3:2" + "Value" = "8:This computer program is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License version 2." + "DefaultValue" = "8:#1202" + "UsePlugInResources" = "11:TRUE" + } + "Welcome" + { + "Name" = "8:Welcome" + "DisplayName" = "8:#1003" + "Description" = "8:#1103" + "Type" = "3:3" + "ContextData" = "8:" + "Attributes" = "3:0" + "Setting" = "3:1" + "Value" = "8:#1203" + "DefaultValue" = "8:#1203" + "UsePlugInResources" = "11:TRUE" + } + } + } + } + } + "{DF760B10-853B-4699-99F2-AFF7185B4A62}:_6B654A06090344BA9AA443E0D0296737" + { + "Name" = "8:#1902" + "Sequence" = "3:1" + "Attributes" = "3:3" + "Dialogs" + { + "{688940B3-5CA9-4162-8DEE-2993FA9D8CBC}:_FB4E7BBC3DA242309FFB58F9A6194A93" + { + "Sequence" = "3:100" + "DisplayName" = "8:Finished" + "UseDynamicProperties" = "11:TRUE" + "IsDependency" = "11:FALSE" + "SourcePath" = "8:\\VsdFinishedDlg.wid" + "Properties" + { + "BannerBitmap" + { + "Name" = "8:BannerBitmap" + "DisplayName" = "8:#1001" + "Description" = "8:#1101" + "Type" = "3:8" + "ContextData" = "8:Bitmap" + "Attributes" = "3:4" + "Setting" = "3:1" + "UsePlugInResources" = "11:TRUE" + } + "UpdateText" + { + "Name" = "8:UpdateText" + "DisplayName" = "8:#1058" + "Description" = "8:#1158" + "Type" = "3:15" + "ContextData" = "8:" + "Attributes" = "3:0" + "Setting" = "3:1" + "Value" = "8:#1258" + "DefaultValue" = "8:#1258" + "UsePlugInResources" = "11:TRUE" + } + } + } + } + } + "{DF760B10-853B-4699-99F2-AFF7185B4A62}:_712819C7C4F042ABB708949BD4426628" + { + "Name" = "8:#1900" + "Sequence" = "3:1" + "Attributes" = "3:1" + "Dialogs" + { + "{688940B3-5CA9-4162-8DEE-2993FA9D8CBC}:_2F49D4FACB954AF2B786D2AD9206D053" + { + "Sequence" = "3:100" + "DisplayName" = "8:Welcome" + "UseDynamicProperties" = "11:TRUE" + "IsDependency" = "11:FALSE" + "SourcePath" = "8:\\VsdWelcomeDlg.wid" + "Properties" + { + "BannerBitmap" + { + "Name" = "8:BannerBitmap" + "DisplayName" = "8:#1001" + "Description" = "8:#1101" + "Type" = "3:8" + "ContextData" = "8:Bitmap" + "Attributes" = "3:4" + "Setting" = "3:1" + "UsePlugInResources" = "11:TRUE" + } + "CopyrightWarning" + { + "Name" = "8:CopyrightWarning" + "DisplayName" = "8:#1002" + "Description" = "8:#1102" + "Type" = "3:3" + "ContextData" = "8:" + "Attributes" = "3:0" + "Setting" = "3:2" + "Value" = "8:This computer program is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License version 2." + "DefaultValue" = "8:#1202" + "UsePlugInResources" = "11:TRUE" + } + "Welcome" + { + "Name" = "8:Welcome" + "DisplayName" = "8:#1003" + "Description" = "8:#1103" + "Type" = "3:3" + "ContextData" = "8:" + "Attributes" = "3:0" + "Setting" = "3:1" + "Value" = "8:#1203" + "DefaultValue" = "8:#1203" + "UsePlugInResources" = "11:TRUE" + } + } + } + "{688940B3-5CA9-4162-8DEE-2993FA9D8CBC}:_770E25BC453A464EA8CD51381FDDDD9F" + { + "Sequence" = "3:300" + "DisplayName" = "8:Confirm Installation" + "UseDynamicProperties" = "11:TRUE" + "IsDependency" = "11:FALSE" + "SourcePath" = "8:\\VsdConfirmDlg.wid" + "Properties" + { + "BannerBitmap" + { + "Name" = "8:BannerBitmap" + "DisplayName" = "8:#1001" + "Description" = "8:#1101" + "Type" = "3:8" + "ContextData" = "8:Bitmap" + "Attributes" = "3:4" + "Setting" = "3:1" + "UsePlugInResources" = "11:TRUE" + } + } + } + "{688940B3-5CA9-4162-8DEE-2993FA9D8CBC}:_9534A64B629F4F868D6D7A384C76DCB2" + { + "Sequence" = "3:200" + "DisplayName" = "8:Installation Folder" + "UseDynamicProperties" = "11:TRUE" + "IsDependency" = "11:FALSE" + "SourcePath" = "8:\\VsdFolderDlg.wid" + "Properties" + { + "BannerBitmap" + { + "Name" = "8:BannerBitmap" + "DisplayName" = "8:#1001" + "Description" = "8:#1101" + "Type" = "3:8" + "ContextData" = "8:Bitmap" + "Attributes" = "3:4" + "Setting" = "3:1" + "UsePlugInResources" = "11:TRUE" + } + "InstallAllUsersVisible" + { + "Name" = "8:InstallAllUsersVisible" + "DisplayName" = "8:#1059" + "Description" = "8:#1159" + "Type" = "3:5" + "ContextData" = "8:1;True=1;False=0" + "Attributes" = "3:0" + "Setting" = "3:0" + "Value" = "3:1" + "DefaultValue" = "3:1" + "UsePlugInResources" = "11:TRUE" + } + } + } + } + } + "{DF760B10-853B-4699-99F2-AFF7185B4A62}:_770CCEFF81BD46A182A6E816A41A0E81" + { + "Name" = "8:#1901" + "Sequence" = "3:2" + "Attributes" = "3:2" + "Dialogs" + { + "{688940B3-5CA9-4162-8DEE-2993FA9D8CBC}:_75E0C1FD245D493DA0D7E3E0BC0C365F" + { + "Sequence" = "3:100" + "DisplayName" = "8:Progress" + "UseDynamicProperties" = "11:TRUE" + "IsDependency" = "11:FALSE" + "SourcePath" = "8:\\VsdAdminProgressDlg.wid" + "Properties" + { + "BannerBitmap" + { + "Name" = "8:BannerBitmap" + "DisplayName" = "8:#1001" + "Description" = "8:#1101" + "Type" = "3:8" + "ContextData" = "8:Bitmap" + "Attributes" = "3:4" + "Setting" = "3:1" + "UsePlugInResources" = "11:TRUE" + } + "ShowProgress" + { + "Name" = "8:ShowProgress" + "DisplayName" = "8:#1009" + "Description" = "8:#1109" + "Type" = "3:5" + "ContextData" = "8:1;True=1;False=0" + "Attributes" = "3:0" + "Setting" = "3:0" + "Value" = "3:1" + "DefaultValue" = "3:1" + "UsePlugInResources" = "11:TRUE" + } + } + } + } + } + "{2479F3F5-0309-486D-8047-8187E2CE5BA0}:_933B15E9A383418F8ADF3B13F68458F3" + { + "UseDynamicProperties" = "11:FALSE" + "IsDependency" = "11:FALSE" + "SourcePath" = "8:\\VsdBasicDialogs.wim" + } + "{DF760B10-853B-4699-99F2-AFF7185B4A62}:_E310DA8CED734E00950A3C5D630CE987" + { + "Name" = "8:#1902" + "Sequence" = "3:2" + "Attributes" = "3:3" + "Dialogs" + { + "{688940B3-5CA9-4162-8DEE-2993FA9D8CBC}:_CB172E69F9C74901BE1040336CFD4F72" + { + "Sequence" = "3:100" + "DisplayName" = "8:Finished" + "UseDynamicProperties" = "11:TRUE" + "IsDependency" = "11:FALSE" + "SourcePath" = "8:\\VsdAdminFinishedDlg.wid" + "Properties" + { + "BannerBitmap" + { + "Name" = "8:BannerBitmap" + "DisplayName" = "8:#1001" + "Description" = "8:#1101" + "Type" = "3:8" + "ContextData" = "8:Bitmap" + "Attributes" = "3:4" + "Setting" = "3:1" + "UsePlugInResources" = "11:TRUE" + } + } + } + } + } + "{DF760B10-853B-4699-99F2-AFF7185B4A62}:_F44F9BE9B54940848289669635E4A5A2" + { + "Name" = "8:#1901" + "Sequence" = "3:1" + "Attributes" = "3:2" + "Dialogs" + { + "{688940B3-5CA9-4162-8DEE-2993FA9D8CBC}:_14751F327A634B989283C8F28CFB6DFC" + { + "Sequence" = "3:100" + "DisplayName" = "8:Progress" + "UseDynamicProperties" = "11:TRUE" + "IsDependency" = "11:FALSE" + "SourcePath" = "8:\\VsdProgressDlg.wid" + "Properties" + { + "BannerBitmap" + { + "Name" = "8:BannerBitmap" + "DisplayName" = "8:#1001" + "Description" = "8:#1101" + "Type" = "3:8" + "ContextData" = "8:Bitmap" + "Attributes" = "3:4" + "Setting" = "3:1" + "UsePlugInResources" = "11:TRUE" + } + "ShowProgress" + { + "Name" = "8:ShowProgress" + "DisplayName" = "8:#1009" + "Description" = "8:#1109" + "Type" = "3:5" + "ContextData" = "8:1;True=1;False=0" + "Attributes" = "3:0" + "Setting" = "3:0" + "Value" = "3:1" + "DefaultValue" = "3:1" + "UsePlugInResources" = "11:TRUE" + } + } + } + } + } + } + "MergeModule" + { + "{CEE29DC0-9FBA-4B99-8D47-5BC643D9B626}:_8E8C3CEE3CE746F0B1D2DAE5DBB941F1" + { + "UseDynamicProperties" = "11:TRUE" + "IsDependency" = "11:TRUE" + "SourcePath" = "8:policy_9_0_Microsoft_VC90_DebugCRT_x86.msm" + "Properties" + { + } + "LanguageId" = "3:0" + "Exclude" = "11:FALSE" + "Folder" = "8:" + "Feature" = "8:" + "IsolateTo" = "8:" + } + "{CEE29DC0-9FBA-4B99-8D47-5BC643D9B626}:_DAED4A66293E4B81B094ACBF3266D11C" + { + "UseDynamicProperties" = "11:TRUE" + "IsDependency" = "11:TRUE" + "SourcePath" = "8:microsoft_vc90_debugcrt_x86.msm" + "Properties" + { + } + "LanguageId" = "3:0" + "Exclude" = "11:FALSE" + "Folder" = "8:" + "Feature" = "8:" + "IsolateTo" = "8:" + } + } + "ProjectOutput" + { + "{5259A561-127C-4D43-A0A1-72F10C7B3BF8}:_E97571D3FBE048DABDC59B37762D800F" + { + "SourcePath" = "8:Win32\\Debug\\ipptool.exe" + "TargetName" = "8:" + "Tag" = "8:" + "Folder" = "8:_FFAEDEA1D38D4088A03FDD6F17E2CA5D" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + "ProjectOutputGroupRegister" = "3:1" + "OutputConfiguration" = "8:" + "OutputGroupCanonicalName" = "8:Built" + "OutputProjectGuid" = "8:{B246D91E-61F2-4433-BFD2-6C2A96FBD4D4}" + "ShowKeyOutput" = "11:TRUE" + "ExcludeFilters" + { + } + } + "{5259A561-127C-4D43-A0A1-72F10C7B3BF8}:_EA282F32A10B4ED1A81AA6133B997C6A" + { + "SourcePath" = "8:Win32\\Debug\\libcups2.dll" + "TargetName" = "8:" + "Tag" = "8:" + "Folder" = "8:_FFAEDEA1D38D4088A03FDD6F17E2CA5D" + "Condition" = "8:" + "Transitive" = "11:FALSE" + "Vital" = "11:TRUE" + "ReadOnly" = "11:FALSE" + "Hidden" = "11:FALSE" + "System" = "11:FALSE" + "Permanent" = "11:FALSE" + "SharedLegacy" = "11:FALSE" + "PackageAs" = "3:1" + "Register" = "3:1" + "Exclude" = "11:FALSE" + "IsDependency" = "11:FALSE" + "IsolateTo" = "8:" + "ProjectOutputGroupRegister" = "3:1" + "OutputConfiguration" = "8:" + "OutputGroupCanonicalName" = "8:Built" + "OutputProjectGuid" = "8:{CB4AA6F2-3E84-45BE-B505-95CD375E8BE3}" + "ShowKeyOutput" = "11:TRUE" + "ExcludeFilters" + { + } + } + } + } +} diff --git a/vcnet/ipptool.vcproj b/vcnet/ipptool.vcproj new file mode 100644 index 0000000000..cebe443aac --- /dev/null +++ b/vcnet/ipptool.vcproj @@ -0,0 +1,385 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/vcnet/libcups2.vcproj b/vcnet/libcups2.vcproj new file mode 100644 index 0000000000..ee586489df --- /dev/null +++ b/vcnet/libcups2.vcproj @@ -0,0 +1,1612 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/vcnet/libcupsimage2.vcproj b/vcnet/libcupsimage2.vcproj new file mode 100644 index 0000000000..23f3f5b919 --- /dev/null +++ b/vcnet/libcupsimage2.vcproj @@ -0,0 +1,388 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/vcnet/regex/COPYRIGHT b/vcnet/regex/COPYRIGHT new file mode 100644 index 0000000000..30c1f7a488 --- /dev/null +++ b/vcnet/regex/COPYRIGHT @@ -0,0 +1,20 @@ +Copyright 1992, 1993, 1994, 1997 Henry Spencer. All rights reserved. +This software is not subject to any license of the American Telephone +and Telegraph Company or of the Regents of the University of California. + +Permission is granted to anyone to use this software for any purpose on +any computer system, and to alter it and redistribute it, subject +to the following restrictions: + +1. The author is not responsible for the consequences of use of this + software, no matter how awful, even if they arise from flaws in it. + +2. The origin of this software must not be misrepresented, either by + explicit claim or by omission. Since few users ever read sources, + credits must appear in the documentation. + +3. Altered versions must be plainly marked as such, and must not be + misrepresented as being the original software. Since few users + ever read sources, credits must appear in the documentation. + +4. This notice may not be removed or altered. diff --git a/vcnet/regex/Makefile b/vcnet/regex/Makefile new file mode 100644 index 0000000000..3882b37864 --- /dev/null +++ b/vcnet/regex/Makefile @@ -0,0 +1,130 @@ +# You probably want to take -DREDEBUG out of CFLAGS, and put something like +# -O in, *after* testing (-DREDEBUG strengthens testing by enabling a lot of +# internal assertion checking and some debugging facilities). +# Put -Dconst= in for a pre-ANSI compiler. +# Do not take -DPOSIX_MISTAKE out. +# REGCFLAGS isn't important to you (it's for my use in some special contexts). +CFLAGS=-I. -DPOSIX_MISTAKE -DREDEBUG $(REGCFLAGS) + +# If you have a pre-ANSI compiler, put -o into MKHFLAGS. If you want +# the Berkeley __P macro, put -b in. +MKHFLAGS= + +# Flags for linking but not compiling, if any. +LDFLAGS= + +# Extra libraries for linking, if any. +LIBS= + +# Internal stuff, should not need changing. +OBJPRODN=regcomp.o regexec.o regerror.o regfree.o +OBJS=$(OBJPRODN) split.o debug.o main.o +H=cclass.h cname.h regex2.h utils.h +REGSRC=regcomp.c regerror.c regexec.c regfree.c +ALLSRC=$(REGSRC) engine.c debug.c main.c split.c + +# Stuff that matters only if you're trying to lint the package. +LINTFLAGS=-I. -Dstatic= -Dconst= -DREDEBUG +LINTC=regcomp.c regexec.c regerror.c regfree.c debug.c main.c +JUNKLINT=possible pointer alignment|null effect + +# arrangements to build forward-reference header files +.SUFFIXES: .ih .h +.c.ih: + sh ./mkh $(MKHFLAGS) -p $< >$@ + +default: r + +lib: purge $(OBJPRODN) + rm -f libregex.a + ar crv libregex.a $(OBJPRODN) + +purge: + rm -f *.o + +# stuff to build regex.h +REGEXH=regex.h +REGEXHSRC=regex2.h $(REGSRC) +$(REGEXH): $(REGEXHSRC) mkh + sh ./mkh $(MKHFLAGS) -i _REGEX_H_ $(REGEXHSRC) >regex.tmp + cmp -s regex.tmp regex.h 2>/dev/null || cp regex.tmp regex.h + rm -f regex.tmp + +# dependencies +$(OBJPRODN) debug.o: utils.h regex.h regex2.h +regcomp.o: cclass.h cname.h regcomp.ih +regexec.o: engine.c engine.ih +regerror.o: regerror.ih +debug.o: debug.ih +main.o: main.ih + +# tester +re: $(OBJS) + $(CC) $(CFLAGS) $(LDFLAGS) $(OBJS) $(LIBS) -o $@ + +# regression test +r: re tests + ./re &1 | egrep -v '$(JUNKLINT)' | tee lint + +fullprint: + ti README WHATSNEW notes todo | list + ti *.h | list + list *.c + list regex.3 regex.7 + +print: + ti README WHATSNEW notes todo | list + ti *.h | list + list reg*.c engine.c + + +mf.tmp: Makefile + sed '/^REGEXH=/s/=.*/=regex.h/' Makefile | sed '/#DEL$$/d' >$@ + +DTRH=cclass.h cname.h regex2.h utils.h +PRE=COPYRIGHT README WHATSNEW +POST=mkh regex.3 regex.7 tests $(DTRH) $(ALLSRC) fake/*.[ch] +FILES=$(PRE) Makefile $(POST) +DTR=$(PRE) Makefile=mf.tmp $(POST) +dtr: $(FILES) mf.tmp + makedtr $(DTR) >$@ + rm mf.tmp + +cio: $(FILES) + cio $(FILES) + +rdf: $(FILES) + rcsdiff -c $(FILES) 2>&1 | p + +# various forms of cleanup +tidy: + rm -f junk* core core.* *.core dtr *.tmp lint + +clean: tidy + rm -f *.o *.s *.ih re libregex.a + +# don't do this one unless you know what you're doing +spotless: clean + rm -f mkh regex.h diff --git a/vcnet/regex/README b/vcnet/regex/README new file mode 100644 index 0000000000..e6ce373444 --- /dev/null +++ b/vcnet/regex/README @@ -0,0 +1,32 @@ +alpha3.8 release. +Tue Aug 10 15:51:48 EDT 1999 +henry@spsystems.net (formerly henry@zoo.toronto.edu) + +See WHATSNEW for change listing. + +installation notes: +-------- +Read the comments at the beginning of Makefile before running. + +Utils.h contains some things that just might have to be modified on +some systems, as well as a nested include (ugh) of . + +The "fake" directory contains quick-and-dirty fakes for some header +files and routines that old systems may not have. Note also that +-DUSEBCOPY will make utils.h substitute bcopy() for memmove(). + +After that, "make r" will build regcomp.o, regexec.o, regfree.o, +and regerror.o (the actual routines), bundle them together into a test +program, and run regression tests on them. No output is good output. + +"make lib" builds just the .o files for the actual routines (when +you're happy with testing and have adjusted CFLAGS for production), +and puts them together into libregex.a. You can pick up either the +library or *.o ("make lib" makes sure there are no other .o files left +around to confuse things). + +Main.c, debug.c, split.c are used for regression testing but are not part +of the RE routines themselves. + +Regex.h goes in /usr/include. All other .h files are internal only. +-------- diff --git a/vcnet/regex/WHATSNEW b/vcnet/regex/WHATSNEW new file mode 100644 index 0000000000..12953433d3 --- /dev/null +++ b/vcnet/regex/WHATSNEW @@ -0,0 +1,108 @@ +New in alpha3.8: Bug fix for signed/unsigned mixup, found and fixed +by the FreeBSD folks. + +New in alpha3.7: A bit of cleanup aimed at maximizing portability, +possibly at slight cost in efficiency. "ul" suffixes and "unsigned long" +no longer appear, in particular. + +New in alpha3.6: A couple more portability glitches fixed. + +New in alpha3.5: Active development of this code has been stopped -- +I'm working on a complete reimplementation -- but folks have found some +minor portability glitches and the like, hence this release to fix them. +One penalty: slightly reduced compatibility with old compilers, because +the ANSI C `unsigned long' type and `ul' constant suffix are used in a +few places (I could avoid this but it would be considerably more work). + +New in alpha3.4: The complex bug alluded to below has been fixed (in a +slightly kludgey temporary way that may hurt efficiency a bit; this is +another "get it out the door for 4.4" release). The tests at the end of +the tests file have accordingly been uncommented. The primary sign of +the bug was that something like a?b matching ab matched b rather than ab. +(The bug was essentially specific to this exact situation, else it would +have shown up earlier.) + +New in alpha3.3: The definition of word boundaries has been altered +slightly, to more closely match the usual programming notion that "_" +is an alphabetic. Stuff used for pre-ANSI systems is now in a subdir, +and the makefile no longer alludes to it in mysterious ways. The +makefile has generally been cleaned up some. Fixes have been made +(again!) so that the regression test will run without -DREDEBUG, at +the cost of weaker checking. A workaround for a bug in some folks' + has been added. And some more things have been added to +tests, including a couple right at the end which are commented out +because the code currently flunks them (complex bug; fix coming). +Plus the usual minor cleanup. + +New in alpha3.2: Assorted bits of cleanup and portability improvement +(the development base is now a BSDI system using GCC instead of an ancient +Sun system, and the newer compiler exposed some glitches). Fix for a +serious bug that affected REs using many [] (including REG_ICASE REs +because of the way they are implemented), *sometimes*, depending on +memory-allocation patterns. The header-file prototypes no longer name +the parameters, avoiding possible name conflicts. The possibility that +some clot has defined CHAR_MIN as (say) `-128' instead of `(-128)' is +now handled gracefully. "uchar" is no longer used as an internal type +name (too many people have the same idea). Still the same old lousy +performance, alas. + +New in alpha3.1: Basically nothing, this release is just a bookkeeping +convenience. Stay tuned. + +New in alpha3.0: Performance is no better, alas, but some fixes have been +made and some functionality has been added. (This is basically the "get +it out the door in time for 4.4" release.) One bug fix: regfree() didn't +free the main internal structure (how embarrassing). It is now possible +to put NULs in either the RE or the target string, using (resp.) a new +REG_PEND flag and the old REG_STARTEND flag. The REG_NOSPEC flag to +regcomp() makes all characters ordinary, so you can match a literal +string easily (this will become more useful when performance improves!). +There are now primitives to match beginnings and ends of words, although +the syntax is disgusting and so is the implementation. The REG_ATOI +debugging interface has changed a bit. And there has been considerable +internal cleanup of various kinds. + +New in alpha2.3: Split change list out of README, and moved flags notes +into Makefile. Macro-ized the name of regex(7) in regex(3), since it has +to change for 4.4BSD. Cleanup work in engine.c, and some new regression +tests to catch tricky cases thereof. + +New in alpha2.2: Out-of-date manpages updated. Regerror() acquires two +small extensions -- REG_ITOA and REG_ATOI -- which avoid debugging kludges +in my own test program and might be useful to others for similar purposes. +The regression test will now compile (and run) without REDEBUG. The +BRE \$ bug is fixed. Most uses of "uchar" are gone; it's all chars now. +Char/uchar parameters are now written int/unsigned, to avoid possible +portability problems with unpromoted parameters. Some unsigned casts have +been introduced to minimize portability problems with shifting into sign +bits. + +New in alpha2.1: Lots of little stuff, cleanup and fixes. The one big +thing is that regex.h is now generated, using mkh, rather than being +supplied in the distribution; due to circularities in dependencies, +you have to build regex.h explicitly by "make h". The two known bugs +have been fixed (and the regression test now checks for them), as has a +problem with assertions not being suppressed in the absence of REDEBUG. +No performance work yet. + +New in alpha2: Backslash-anything is an ordinary character, not an +error (except, of course, for the handful of backslashed metacharacters +in BREs), which should reduce script breakage. The regression test +checks *where* null strings are supposed to match, and has generally +been tightened up somewhat. Small bug fixes in parameter passing (not +harmful, but technically errors) and some other areas. Debugging +invoked by defining REDEBUG rather than not defining NDEBUG. + +New in alpha+3: full prototyping for internal routines, using a little +helper program, mkh, which extracts prototypes given in stylized comments. +More minor cleanup. Buglet fix: it's CHAR_BIT, not CHAR_BITS. Simple +pre-screening of input when a literal string is known to be part of the +RE; this does wonders for performance. + +New in alpha+2: minor bits of cleanup. Notably, the number "32" for the +word width isn't hardwired into regexec.c any more, the public header +file prototypes the functions if __STDC__ is defined, and some small typos +in the manpages have been fixed. + +New in alpha+1: improvements to the manual pages, and an important +extension, the REG_STARTEND option to regexec(). diff --git a/vcnet/regex/cclass.h b/vcnet/regex/cclass.h new file mode 100644 index 0000000000..0c293028e9 --- /dev/null +++ b/vcnet/regex/cclass.h @@ -0,0 +1,31 @@ +/* character-class table */ +static struct cclass { + char *name; + char *chars; + char *multis; +} cclasses[] = { + "alnum", "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\ +0123456789", "", + "alpha", "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz", + "", + "blank", " \t", "", + "cntrl", "\007\b\t\n\v\f\r\1\2\3\4\5\6\16\17\20\21\22\23\24\ +\25\26\27\30\31\32\33\34\35\36\37\177", "", + "digit", "0123456789", "", + "graph", "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\ +0123456789!\"#$%&'()*+,-./:;<=>?@[\\]^_`{|}~", + "", + "lower", "abcdefghijklmnopqrstuvwxyz", + "", + "print", "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\ +0123456789!\"#$%&'()*+,-./:;<=>?@[\\]^_`{|}~ ", + "", + "punct", "!\"#$%&'()*+,-./:;<=>?@[\\]^_`{|}~", + "", + "space", "\t\n\v\f\r ", "", + "upper", "ABCDEFGHIJKLMNOPQRSTUVWXYZ", + "", + "xdigit", "0123456789ABCDEFabcdef", + "", + NULL, 0, "" +}; diff --git a/vcnet/regex/cname.h b/vcnet/regex/cname.h new file mode 100644 index 0000000000..02e86e912e --- /dev/null +++ b/vcnet/regex/cname.h @@ -0,0 +1,102 @@ +/* character-name table */ +static struct cname { + char *name; + char code; +} cnames[] = { + "NUL", '\0', + "SOH", '\001', + "STX", '\002', + "ETX", '\003', + "EOT", '\004', + "ENQ", '\005', + "ACK", '\006', + "BEL", '\007', + "alert", '\007', + "BS", '\010', + "backspace", '\b', + "HT", '\011', + "tab", '\t', + "LF", '\012', + "newline", '\n', + "VT", '\013', + "vertical-tab", '\v', + "FF", '\014', + "form-feed", '\f', + "CR", '\015', + "carriage-return", '\r', + "SO", '\016', + "SI", '\017', + "DLE", '\020', + "DC1", '\021', + "DC2", '\022', + "DC3", '\023', + "DC4", '\024', + "NAK", '\025', + "SYN", '\026', + "ETB", '\027', + "CAN", '\030', + "EM", '\031', + "SUB", '\032', + "ESC", '\033', + "IS4", '\034', + "FS", '\034', + "IS3", '\035', + "GS", '\035', + "IS2", '\036', + "RS", '\036', + "IS1", '\037', + "US", '\037', + "space", ' ', + "exclamation-mark", '!', + "quotation-mark", '"', + "number-sign", '#', + "dollar-sign", '$', + "percent-sign", '%', + "ampersand", '&', + "apostrophe", '\'', + "left-parenthesis", '(', + "right-parenthesis", ')', + "asterisk", '*', + "plus-sign", '+', + "comma", ',', + "hyphen", '-', + "hyphen-minus", '-', + "period", '.', + "full-stop", '.', + "slash", '/', + "solidus", '/', + "zero", '0', + "one", '1', + "two", '2', + "three", '3', + "four", '4', + "five", '5', + "six", '6', + "seven", '7', + "eight", '8', + "nine", '9', + "colon", ':', + "semicolon", ';', + "less-than-sign", '<', + "equals-sign", '=', + "greater-than-sign", '>', + "question-mark", '?', + "commercial-at", '@', + "left-square-bracket", '[', + "backslash", '\\', + "reverse-solidus", '\\', + "right-square-bracket", ']', + "circumflex", '^', + "circumflex-accent", '^', + "underscore", '_', + "low-line", '_', + "grave-accent", '`', + "left-brace", '{', + "left-curly-bracket", '{', + "vertical-line", '|', + "right-brace", '}', + "right-curly-bracket", '}', + "tilde", '~', + "DEL", '\177', + NULL, 0, +}; diff --git a/vcnet/regex/debug.c b/vcnet/regex/debug.c new file mode 100644 index 0000000000..99ce7da6dd --- /dev/null +++ b/vcnet/regex/debug.c @@ -0,0 +1,242 @@ +#include +#include +#include +#include +#include +#include +#include + +#include "utils.h" +#include "regex2.h" +#include "debug.ih" + +/* + - regprint - print a regexp for debugging + == void regprint(regex_t *r, FILE *d); + */ +void +regprint(r, d) +regex_t *r; +FILE *d; +{ + register struct re_guts *g = r->re_g; + register int i; + register int c; + register int last; + int nincat[NC]; + + fprintf(d, "%ld states, %d categories", (long)g->nstates, + g->ncategories); + fprintf(d, ", first %ld last %ld", (long)g->firststate, + (long)g->laststate); + if (g->iflags&USEBOL) + fprintf(d, ", USEBOL"); + if (g->iflags&USEEOL) + fprintf(d, ", USEEOL"); + if (g->iflags&BAD) + fprintf(d, ", BAD"); + if (g->nsub > 0) + fprintf(d, ", nsub=%ld", (long)g->nsub); + if (g->must != NULL) + fprintf(d, ", must(%ld) `%*s'", (long)g->mlen, (int)g->mlen, + g->must); + if (g->backrefs) + fprintf(d, ", backrefs"); + if (g->nplus > 0) + fprintf(d, ", nplus %ld", (long)g->nplus); + fprintf(d, "\n"); + s_print(g, d); + for (i = 0; i < g->ncategories; i++) { + nincat[i] = 0; + for (c = CHAR_MIN; c <= CHAR_MAX; c++) + if (g->categories[c] == i) + nincat[i]++; + } + fprintf(d, "cc0#%d", nincat[0]); + for (i = 1; i < g->ncategories; i++) + if (nincat[i] == 1) { + for (c = CHAR_MIN; c <= CHAR_MAX; c++) + if (g->categories[c] == i) + break; + fprintf(d, ", %d=%s", i, regchar(c)); + } + fprintf(d, "\n"); + for (i = 1; i < g->ncategories; i++) + if (nincat[i] != 1) { + fprintf(d, "cc%d\t", i); + last = -1; + for (c = CHAR_MIN; c <= CHAR_MAX+1; c++) /* +1 does flush */ + if (c <= CHAR_MAX && g->categories[c] == i) { + if (last < 0) { + fprintf(d, "%s", regchar(c)); + last = c; + } + } else { + if (last >= 0) { + if (last != c-1) + fprintf(d, "-%s", + regchar(c-1)); + last = -1; + } + } + fprintf(d, "\n"); + } +} + +/* + - s_print - print the strip for debugging + == static void s_print(register struct re_guts *g, FILE *d); + */ +static void +s_print(g, d) +register struct re_guts *g; +FILE *d; +{ + register sop *s; + register cset *cs; + register int i; + register int done = 0; + register sop opnd; + register int col = 0; + register int last; + register sopno offset = 2; +# define GAP() { if (offset % 5 == 0) { \ + if (col > 40) { \ + fprintf(d, "\n\t"); \ + col = 0; \ + } else { \ + fprintf(d, " "); \ + col++; \ + } \ + } else \ + col++; \ + offset++; \ + } + + if (OP(g->strip[0]) != OEND) + fprintf(d, "missing initial OEND!\n"); + for (s = &g->strip[1]; !done; s++) { + opnd = OPND(*s); + switch (OP(*s)) { + case OEND: + fprintf(d, "\n"); + done = 1; + break; + case OCHAR: + if (strchr("\\|()^$.[+*?{}!<> ", (char)opnd) != NULL) + fprintf(d, "\\%c", (char)opnd); + else + fprintf(d, "%s", regchar((char)opnd)); + break; + case OBOL: + fprintf(d, "^"); + break; + case OEOL: + fprintf(d, "$"); + break; + case OBOW: + fprintf(d, "\\{"); + break; + case OEOW: + fprintf(d, "\\}"); + break; + case OANY: + fprintf(d, "."); + break; + case OANYOF: + fprintf(d, "[(%ld)", (long)opnd); + cs = &g->sets[opnd]; + last = -1; + for (i = 0; i < g->csetsize+1; i++) /* +1 flushes */ + if (CHIN(cs, i) && i < g->csetsize) { + if (last < 0) { + fprintf(d, "%s", regchar(i)); + last = i; + } + } else { + if (last >= 0) { + if (last != i-1) + fprintf(d, "-%s", + regchar(i-1)); + last = -1; + } + } + fprintf(d, "]"); + break; + case OBACK_: + fprintf(d, "(\\<%ld>", (long)opnd); + break; + case O_BACK: + fprintf(d, "<%ld>\\)", (long)opnd); + break; + case OPLUS_: + fprintf(d, "(+"); + if (OP(*(s+opnd)) != O_PLUS) + fprintf(d, "<%ld>", (long)opnd); + break; + case O_PLUS: + if (OP(*(s-opnd)) != OPLUS_) + fprintf(d, "<%ld>", (long)opnd); + fprintf(d, "+)"); + break; + case OQUEST_: + fprintf(d, "(?"); + if (OP(*(s+opnd)) != O_QUEST) + fprintf(d, "<%ld>", (long)opnd); + break; + case O_QUEST: + if (OP(*(s-opnd)) != OQUEST_) + fprintf(d, "<%ld>", (long)opnd); + fprintf(d, "?)"); + break; + case OLPAREN: + fprintf(d, "((<%ld>", (long)opnd); + break; + case ORPAREN: + fprintf(d, "<%ld>))", (long)opnd); + break; + case OCH_: + fprintf(d, "<"); + if (OP(*(s+opnd)) != OOR2) + fprintf(d, "<%ld>", (long)opnd); + break; + case OOR1: + if (OP(*(s-opnd)) != OOR1 && OP(*(s-opnd)) != OCH_) + fprintf(d, "<%ld>", (long)opnd); + fprintf(d, "|"); + break; + case OOR2: + fprintf(d, "|"); + if (OP(*(s+opnd)) != OOR2 && OP(*(s+opnd)) != O_CH) + fprintf(d, "<%ld>", (long)opnd); + break; + case O_CH: + if (OP(*(s-opnd)) != OOR1) + fprintf(d, "<%ld>", (long)opnd); + fprintf(d, ">"); + break; + default: + fprintf(d, "!%d(%d)!", OP(*s), opnd); + break; + } + if (!done) + GAP(); + } +} + +/* + - regchar - make a character printable + == static char *regchar(int ch); + */ +static char * /* -> representation */ +regchar(ch) +int ch; +{ + static char buf[10]; + + if (isprint(ch) || ch == ' ') + sprintf(buf, "%c", ch); + else + sprintf(buf, "\\%o", ch); + return(buf); +} diff --git a/vcnet/regex/debug.ih b/vcnet/regex/debug.ih new file mode 100644 index 0000000000..5f40ff7917 --- /dev/null +++ b/vcnet/regex/debug.ih @@ -0,0 +1,14 @@ +/* ========= begin header generated by ./mkh ========= */ +#ifdef __cplusplus +extern "C" { +#endif + +/* === debug.c === */ +void regprint(regex_t *r, FILE *d); +static void s_print(register struct re_guts *g, FILE *d); +static char *regchar(int ch); + +#ifdef __cplusplus +} +#endif +/* ========= end header generated by ./mkh ========= */ diff --git a/vcnet/regex/engine.c b/vcnet/regex/engine.c new file mode 100644 index 0000000000..0b88dcf1ed --- /dev/null +++ b/vcnet/regex/engine.c @@ -0,0 +1,1019 @@ +/* + * The matching engine and friends. This file is #included by regexec.c + * after suitable #defines of a variety of macros used herein, so that + * different state representations can be used without duplicating masses + * of code. + */ + +#ifdef SNAMES +#define matcher smatcher +#define fast sfast +#define slow sslow +#define dissect sdissect +#define backref sbackref +#define step sstep +#define print sprint +#define at sat +#define match smat +#endif +#ifdef LNAMES +#define matcher lmatcher +#define fast lfast +#define slow lslow +#define dissect ldissect +#define backref lbackref +#define step lstep +#define print lprint +#define at lat +#define match lmat +#endif + +/* another structure passed up and down to avoid zillions of parameters */ +struct match { + struct re_guts *g; + int eflags; + regmatch_t *pmatch; /* [nsub+1] (0 element unused) */ + char *offp; /* offsets work from here */ + char *beginp; /* start of string -- virtual NUL precedes */ + char *endp; /* end of string -- virtual NUL here */ + char *coldp; /* can be no match starting before here */ + char **lastpos; /* [nplus+1] */ + STATEVARS; + states st; /* current states */ + states fresh; /* states for a fresh start */ + states tmp; /* temporary */ + states empty; /* empty set of states */ +}; + +#include "engine.ih" + +#ifdef REDEBUG +#define SP(t, s, c) print(m, t, s, c, stdout) +#define AT(t, p1, p2, s1, s2) at(m, t, p1, p2, s1, s2) +#define NOTE(str) { if (m->eflags®_TRACE) printf("=%s\n", (str)); } +#else +#define SP(t, s, c) /* nothing */ +#define AT(t, p1, p2, s1, s2) /* nothing */ +#define NOTE(s) /* nothing */ +#endif + +/* + - matcher - the actual matching engine + == static int matcher(register struct re_guts *g, char *string, \ + == size_t nmatch, regmatch_t pmatch[], int eflags); + */ +static int /* 0 success, REG_NOMATCH failure */ +matcher(g, string, nmatch, pmatch, eflags) +register struct re_guts *g; +char *string; +size_t nmatch; +regmatch_t pmatch[]; +int eflags; +{ + register char *endp; + register size_t i; + struct match mv; + register struct match *m = &mv; + register char *dp; + const register sopno gf = g->firststate+1; /* +1 for OEND */ + const register sopno gl = g->laststate; + char *start; + char *stop; + + /* simplify the situation where possible */ + if (g->cflags®_NOSUB) + nmatch = 0; + if (eflags®_STARTEND) { + start = string + pmatch[0].rm_so; + stop = string + pmatch[0].rm_eo; + } else { + start = string; + stop = start + strlen(start); + } + if (stop < start) + return(REG_INVARG); + + /* prescreening; this does wonders for this rather slow code */ + if (g->must != NULL) { + for (dp = start; dp < stop; dp++) + if (*dp == g->must[0] && stop - dp >= g->mlen && + memcmp(dp, g->must, (size_t)g->mlen) == 0) + break; + if (dp == stop) /* we didn't find g->must */ + return(REG_NOMATCH); + } + + /* match struct setup */ + m->g = g; + m->eflags = eflags; + m->pmatch = NULL; + m->lastpos = NULL; + m->offp = string; + m->beginp = start; + m->endp = stop; + STATESETUP(m, 4); + SETUP(m->st); + SETUP(m->fresh); + SETUP(m->tmp); + SETUP(m->empty); + CLEAR(m->empty); + + /* this loop does only one repetition except for backrefs */ + for (;;) { + endp = fast(m, start, stop, gf, gl); + if (endp == NULL) { /* a miss */ + STATETEARDOWN(m); + return(REG_NOMATCH); + } + if (nmatch == 0 && !g->backrefs) + break; /* no further info needed */ + + /* where? */ + assert(m->coldp != NULL); + for (;;) { + NOTE("finding start"); + endp = slow(m, m->coldp, stop, gf, gl); + if (endp != NULL) + break; + assert(m->coldp < m->endp); + m->coldp++; + } + if (nmatch == 1 && !g->backrefs) + break; /* no further info needed */ + + /* oh my, he wants the subexpressions... */ + if (m->pmatch == NULL) + m->pmatch = (regmatch_t *)malloc((m->g->nsub + 1) * + sizeof(regmatch_t)); + if (m->pmatch == NULL) { + STATETEARDOWN(m); + return(REG_ESPACE); + } + for (i = 1; i <= m->g->nsub; i++) + m->pmatch[i].rm_so = m->pmatch[i].rm_eo = -1; + if (!g->backrefs && !(m->eflags®_BACKR)) { + NOTE("dissecting"); + dp = dissect(m, m->coldp, endp, gf, gl); + } else { + if (g->nplus > 0 && m->lastpos == NULL) + m->lastpos = (char **)malloc((g->nplus+1) * + sizeof(char *)); + if (g->nplus > 0 && m->lastpos == NULL) { + free(m->pmatch); + STATETEARDOWN(m); + return(REG_ESPACE); + } + NOTE("backref dissect"); + dp = backref(m, m->coldp, endp, gf, gl, (sopno)0); + } + if (dp != NULL) + break; + + /* uh-oh... we couldn't find a subexpression-level match */ + assert(g->backrefs); /* must be back references doing it */ + assert(g->nplus == 0 || m->lastpos != NULL); + for (;;) { + if (dp != NULL || endp <= m->coldp) + break; /* defeat */ + NOTE("backoff"); + endp = slow(m, m->coldp, endp-1, gf, gl); + if (endp == NULL) + break; /* defeat */ + /* try it on a shorter possibility */ +#ifndef NDEBUG + for (i = 1; i <= m->g->nsub; i++) { + assert(m->pmatch[i].rm_so == -1); + assert(m->pmatch[i].rm_eo == -1); + } +#endif + NOTE("backoff dissect"); + dp = backref(m, m->coldp, endp, gf, gl, (sopno)0); + } + assert(dp == NULL || dp == endp); + if (dp != NULL) /* found a shorter one */ + break; + + /* despite initial appearances, there is no match here */ + NOTE("false alarm"); + start = m->coldp + 1; /* recycle starting later */ + assert(start <= stop); + } + + /* fill in the details if requested */ + if (nmatch > 0) { + pmatch[0].rm_so = m->coldp - m->offp; + pmatch[0].rm_eo = endp - m->offp; + } + if (nmatch > 1) { + assert(m->pmatch != NULL); + for (i = 1; i < nmatch; i++) + if (i <= m->g->nsub) + pmatch[i] = m->pmatch[i]; + else { + pmatch[i].rm_so = -1; + pmatch[i].rm_eo = -1; + } + } + + if (m->pmatch != NULL) + free((char *)m->pmatch); + if (m->lastpos != NULL) + free((char *)m->lastpos); + STATETEARDOWN(m); + return(0); +} + +/* + - dissect - figure out what matched what, no back references + == static char *dissect(register struct match *m, char *start, \ + == char *stop, sopno startst, sopno stopst); + */ +static char * /* == stop (success) always */ +dissect(m, start, stop, startst, stopst) +register struct match *m; +char *start; +char *stop; +sopno startst; +sopno stopst; +{ + register int i; + register sopno ss; /* start sop of current subRE */ + register sopno es; /* end sop of current subRE */ + register char *sp; /* start of string matched by it */ + register char *stp; /* string matched by it cannot pass here */ + register char *rest; /* start of rest of string */ + register char *tail; /* string unmatched by rest of RE */ + register sopno ssub; /* start sop of subsubRE */ + register sopno esub; /* end sop of subsubRE */ + register char *ssp; /* start of string matched by subsubRE */ + register char *sep; /* end of string matched by subsubRE */ + register char *oldssp; /* previous ssp */ + register char *dp; + + AT("diss", start, stop, startst, stopst); + sp = start; + for (ss = startst; ss < stopst; ss = es) { + /* identify end of subRE */ + es = ss; + switch (OP(m->g->strip[es])) { + case OPLUS_: + case OQUEST_: + es += OPND(m->g->strip[es]); + break; + case OCH_: + while (OP(m->g->strip[es]) != O_CH) + es += OPND(m->g->strip[es]); + break; + } + es++; + + /* figure out what it matched */ + switch (OP(m->g->strip[ss])) { + case OEND: + assert(nope); + break; + case OCHAR: + sp++; + break; + case OBOL: + case OEOL: + case OBOW: + case OEOW: + break; + case OANY: + case OANYOF: + sp++; + break; + case OBACK_: + case O_BACK: + assert(nope); + break; + /* cases where length of match is hard to find */ + case OQUEST_: + stp = stop; + for (;;) { + /* how long could this one be? */ + rest = slow(m, sp, stp, ss, es); + assert(rest != NULL); /* it did match */ + /* could the rest match the rest? */ + tail = slow(m, rest, stop, es, stopst); + if (tail == stop) + break; /* yes! */ + /* no -- try a shorter match for this one */ + stp = rest - 1; + assert(stp >= sp); /* it did work */ + } + ssub = ss + 1; + esub = es - 1; + /* did innards match? */ + if (slow(m, sp, rest, ssub, esub) != NULL) { + dp = dissect(m, sp, rest, ssub, esub); + assert(dp == rest); + } else /* no */ + assert(sp == rest); + sp = rest; + break; + case OPLUS_: + stp = stop; + for (;;) { + /* how long could this one be? */ + rest = slow(m, sp, stp, ss, es); + assert(rest != NULL); /* it did match */ + /* could the rest match the rest? */ + tail = slow(m, rest, stop, es, stopst); + if (tail == stop) + break; /* yes! */ + /* no -- try a shorter match for this one */ + stp = rest - 1; + assert(stp >= sp); /* it did work */ + } + ssub = ss + 1; + esub = es - 1; + ssp = sp; + oldssp = ssp; + for (;;) { /* find last match of innards */ + sep = slow(m, ssp, rest, ssub, esub); + if (sep == NULL || sep == ssp) + break; /* failed or matched null */ + oldssp = ssp; /* on to next try */ + ssp = sep; + } + if (sep == NULL) { + /* last successful match */ + sep = ssp; + ssp = oldssp; + } + assert(sep == rest); /* must exhaust substring */ + assert(slow(m, ssp, sep, ssub, esub) == rest); + dp = dissect(m, ssp, sep, ssub, esub); + assert(dp == sep); + sp = rest; + break; + case OCH_: + stp = stop; + for (;;) { + /* how long could this one be? */ + rest = slow(m, sp, stp, ss, es); + assert(rest != NULL); /* it did match */ + /* could the rest match the rest? */ + tail = slow(m, rest, stop, es, stopst); + if (tail == stop) + break; /* yes! */ + /* no -- try a shorter match for this one */ + stp = rest - 1; + assert(stp >= sp); /* it did work */ + } + ssub = ss + 1; + esub = ss + OPND(m->g->strip[ss]) - 1; + assert(OP(m->g->strip[esub]) == OOR1); + for (;;) { /* find first matching branch */ + if (slow(m, sp, rest, ssub, esub) == rest) + break; /* it matched all of it */ + /* that one missed, try next one */ + assert(OP(m->g->strip[esub]) == OOR1); + esub++; + assert(OP(m->g->strip[esub]) == OOR2); + ssub = esub + 1; + esub += OPND(m->g->strip[esub]); + if (OP(m->g->strip[esub]) == OOR2) + esub--; + else + assert(OP(m->g->strip[esub]) == O_CH); + } + dp = dissect(m, sp, rest, ssub, esub); + assert(dp == rest); + sp = rest; + break; + case O_PLUS: + case O_QUEST: + case OOR1: + case OOR2: + case O_CH: + assert(nope); + break; + case OLPAREN: + i = OPND(m->g->strip[ss]); + assert(0 < i && i <= m->g->nsub); + m->pmatch[i].rm_so = sp - m->offp; + break; + case ORPAREN: + i = OPND(m->g->strip[ss]); + assert(0 < i && i <= m->g->nsub); + m->pmatch[i].rm_eo = sp - m->offp; + break; + default: /* uh oh */ + assert(nope); + break; + } + } + + assert(sp == stop); + return(sp); +} + +/* + - backref - figure out what matched what, figuring in back references + == static char *backref(register struct match *m, char *start, \ + == char *stop, sopno startst, sopno stopst, sopno lev); + */ +static char * /* == stop (success) or NULL (failure) */ +backref(m, start, stop, startst, stopst, lev) +register struct match *m; +char *start; +char *stop; +sopno startst; +sopno stopst; +sopno lev; /* PLUS nesting level */ +{ + register int i; + register sopno ss; /* start sop of current subRE */ + register char *sp; /* start of string matched by it */ + register sopno ssub; /* start sop of subsubRE */ + register sopno esub; /* end sop of subsubRE */ + register char *ssp; /* start of string matched by subsubRE */ + register char *dp; + register size_t len; + register int hard; + register sop s; + register regoff_t offsave; + register cset *cs; + + AT("back", start, stop, startst, stopst); + sp = start; + + /* get as far as we can with easy stuff */ + hard = 0; + for (ss = startst; !hard && ss < stopst; ss++) + switch (OP(s = m->g->strip[ss])) { + case OCHAR: + if (sp == stop || *sp++ != (char)OPND(s)) + return(NULL); + break; + case OANY: + if (sp == stop) + return(NULL); + sp++; + break; + case OANYOF: + cs = &m->g->sets[OPND(s)]; + if (sp == stop || !CHIN(cs, *sp++)) + return(NULL); + break; + case OBOL: + if ( (sp == m->beginp && !(m->eflags®_NOTBOL)) || + (sp < m->endp && *(sp-1) == '\n' && + (m->g->cflags®_NEWLINE)) ) + { /* yes */ } + else + return(NULL); + break; + case OEOL: + if ( (sp == m->endp && !(m->eflags®_NOTEOL)) || + (sp < m->endp && *sp == '\n' && + (m->g->cflags®_NEWLINE)) ) + { /* yes */ } + else + return(NULL); + break; + case OBOW: + if (( (sp == m->beginp && !(m->eflags®_NOTBOL)) || + (sp < m->endp && *(sp-1) == '\n' && + (m->g->cflags®_NEWLINE)) || + (sp > m->beginp && + !ISWORD(*(sp-1))) ) && + (sp < m->endp && ISWORD(*sp)) ) + { /* yes */ } + else + return(NULL); + break; + case OEOW: + if (( (sp == m->endp && !(m->eflags®_NOTEOL)) || + (sp < m->endp && *sp == '\n' && + (m->g->cflags®_NEWLINE)) || + (sp < m->endp && !ISWORD(*sp)) ) && + (sp > m->beginp && ISWORD(*(sp-1))) ) + { /* yes */ } + else + return(NULL); + break; + case O_QUEST: + break; + case OOR1: /* matches null but needs to skip */ + ss++; + s = m->g->strip[ss]; + do { + assert(OP(s) == OOR2); + ss += OPND(s); + } while (OP(s = m->g->strip[ss]) != O_CH); + /* note that the ss++ gets us past the O_CH */ + break; + default: /* have to make a choice */ + hard = 1; + break; + } + if (!hard) { /* that was it! */ + if (sp != stop) + return(NULL); + return(sp); + } + ss--; /* adjust for the for's final increment */ + + /* the hard stuff */ + AT("hard", sp, stop, ss, stopst); + s = m->g->strip[ss]; + switch (OP(s)) { + case OBACK_: /* the vilest depths */ + i = OPND(s); + assert(0 < i && i <= m->g->nsub); + if (m->pmatch[i].rm_eo == -1) + return(NULL); + assert(m->pmatch[i].rm_so != -1); + len = m->pmatch[i].rm_eo - m->pmatch[i].rm_so; + assert(stop - m->beginp >= len); + if (sp > stop - len) + return(NULL); /* not enough left to match */ + ssp = m->offp + m->pmatch[i].rm_so; + if (memcmp(sp, ssp, len) != 0) + return(NULL); + while (m->g->strip[ss] != SOP(O_BACK, i)) + ss++; + return(backref(m, sp+len, stop, ss+1, stopst, lev)); + break; + case OQUEST_: /* to null or not */ + dp = backref(m, sp, stop, ss+1, stopst, lev); + if (dp != NULL) + return(dp); /* not */ + return(backref(m, sp, stop, ss+OPND(s)+1, stopst, lev)); + break; + case OPLUS_: + assert(m->lastpos != NULL); + assert(lev+1 <= m->g->nplus); + m->lastpos[lev+1] = sp; + return(backref(m, sp, stop, ss+1, stopst, lev+1)); + break; + case O_PLUS: + if (sp == m->lastpos[lev]) /* last pass matched null */ + return(backref(m, sp, stop, ss+1, stopst, lev-1)); + /* try another pass */ + m->lastpos[lev] = sp; + dp = backref(m, sp, stop, ss-OPND(s)+1, stopst, lev); + if (dp == NULL) + return(backref(m, sp, stop, ss+1, stopst, lev-1)); + else + return(dp); + break; + case OCH_: /* find the right one, if any */ + ssub = ss + 1; + esub = ss + OPND(s) - 1; + assert(OP(m->g->strip[esub]) == OOR1); + for (;;) { /* find first matching branch */ + dp = backref(m, sp, stop, ssub, esub, lev); + if (dp != NULL) + return(dp); + /* that one missed, try next one */ + if (OP(m->g->strip[esub]) == O_CH) + return(NULL); /* there is none */ + esub++; + assert(OP(m->g->strip[esub]) == OOR2); + ssub = esub + 1; + esub += OPND(m->g->strip[esub]); + if (OP(m->g->strip[esub]) == OOR2) + esub--; + else + assert(OP(m->g->strip[esub]) == O_CH); + } + break; + case OLPAREN: /* must undo assignment if rest fails */ + i = OPND(s); + assert(0 < i && i <= m->g->nsub); + offsave = m->pmatch[i].rm_so; + m->pmatch[i].rm_so = sp - m->offp; + dp = backref(m, sp, stop, ss+1, stopst, lev); + if (dp != NULL) + return(dp); + m->pmatch[i].rm_so = offsave; + return(NULL); + break; + case ORPAREN: /* must undo assignment if rest fails */ + i = OPND(s); + assert(0 < i && i <= m->g->nsub); + offsave = m->pmatch[i].rm_eo; + m->pmatch[i].rm_eo = sp - m->offp; + dp = backref(m, sp, stop, ss+1, stopst, lev); + if (dp != NULL) + return(dp); + m->pmatch[i].rm_eo = offsave; + return(NULL); + break; + default: /* uh oh */ + assert(nope); + break; + } + + /* "can't happen" */ + assert(nope); + /* NOTREACHED */ + return((char *)NULL); /* dummy */ +} + +/* + - fast - step through the string at top speed + == static char *fast(register struct match *m, char *start, \ + == char *stop, sopno startst, sopno stopst); + */ +static char * /* where tentative match ended, or NULL */ +fast(m, start, stop, startst, stopst) +register struct match *m; +char *start; +char *stop; +sopno startst; +sopno stopst; +{ + register states st = m->st; + register states fresh = m->fresh; + register states tmp = m->tmp; + register char *p = start; + register int c = (start == m->beginp) ? OUT : *(start-1); + register int lastc; /* previous c */ + register int flagch; + register int i; + register char *coldp; /* last p after which no match was underway */ + + CLEAR(st); + SET1(st, startst); + st = step(m->g, startst, stopst, st, NOTHING, st); + ASSIGN(fresh, st); + SP("start", st, *p); + coldp = NULL; + for (;;) { + /* next character */ + lastc = c; + c = (p == m->endp) ? OUT : *p; + if (EQ(st, fresh)) + coldp = p; + + /* is there an EOL and/or BOL between lastc and c? */ + flagch = '\0'; + i = 0; + if ( (lastc == '\n' && m->g->cflags®_NEWLINE) || + (lastc == OUT && !(m->eflags®_NOTBOL)) ) { + flagch = BOL; + i = m->g->nbol; + } + if ( (c == '\n' && m->g->cflags®_NEWLINE) || + (c == OUT && !(m->eflags®_NOTEOL)) ) { + flagch = (flagch == BOL) ? BOLEOL : EOL; + i += m->g->neol; + } + if (i != 0) { + for (; i > 0; i--) + st = step(m->g, startst, stopst, st, flagch, st); + SP("boleol", st, c); + } + + /* how about a word boundary? */ + if ( (flagch == BOL || (lastc != OUT && !ISWORD(lastc))) && + (c != OUT && ISWORD(c)) ) { + flagch = BOW; + } + if ( (lastc != OUT && ISWORD(lastc)) && + (flagch == EOL || (c != OUT && !ISWORD(c))) ) { + flagch = EOW; + } + if (flagch == BOW || flagch == EOW) { + st = step(m->g, startst, stopst, st, flagch, st); + SP("boweow", st, c); + } + + /* are we done? */ + if (ISSET(st, stopst) || p == stop) + break; /* NOTE BREAK OUT */ + + /* no, we must deal with this character */ + ASSIGN(tmp, st); + ASSIGN(st, fresh); + assert(c != OUT); + st = step(m->g, startst, stopst, tmp, c, st); + SP("aft", st, c); + assert(EQ(step(m->g, startst, stopst, st, NOTHING, st), st)); + p++; + } + + assert(coldp != NULL); + m->coldp = coldp; + if (ISSET(st, stopst)) + return(p+1); + else + return(NULL); +} + +/* + - slow - step through the string more deliberately + == static char *slow(register struct match *m, char *start, \ + == char *stop, sopno startst, sopno stopst); + */ +static char * /* where it ended */ +slow(m, start, stop, startst, stopst) +register struct match *m; +char *start; +char *stop; +sopno startst; +sopno stopst; +{ + register states st = m->st; + register states empty = m->empty; + register states tmp = m->tmp; + register char *p = start; + register int c = (start == m->beginp) ? OUT : *(start-1); + register int lastc; /* previous c */ + register int flagch; + register int i; + register char *matchp; /* last p at which a match ended */ + + AT("slow", start, stop, startst, stopst); + CLEAR(st); + SET1(st, startst); + SP("sstart", st, *p); + st = step(m->g, startst, stopst, st, NOTHING, st); + matchp = NULL; + for (;;) { + /* next character */ + lastc = c; + c = (p == m->endp) ? OUT : *p; + + /* is there an EOL and/or BOL between lastc and c? */ + flagch = '\0'; + i = 0; + if ( (lastc == '\n' && m->g->cflags®_NEWLINE) || + (lastc == OUT && !(m->eflags®_NOTBOL)) ) { + flagch = BOL; + i = m->g->nbol; + } + if ( (c == '\n' && m->g->cflags®_NEWLINE) || + (c == OUT && !(m->eflags®_NOTEOL)) ) { + flagch = (flagch == BOL) ? BOLEOL : EOL; + i += m->g->neol; + } + if (i != 0) { + for (; i > 0; i--) + st = step(m->g, startst, stopst, st, flagch, st); + SP("sboleol", st, c); + } + + /* how about a word boundary? */ + if ( (flagch == BOL || (lastc != OUT && !ISWORD(lastc))) && + (c != OUT && ISWORD(c)) ) { + flagch = BOW; + } + if ( (lastc != OUT && ISWORD(lastc)) && + (flagch == EOL || (c != OUT && !ISWORD(c))) ) { + flagch = EOW; + } + if (flagch == BOW || flagch == EOW) { + st = step(m->g, startst, stopst, st, flagch, st); + SP("sboweow", st, c); + } + + /* are we done? */ + if (ISSET(st, stopst)) + matchp = p; + if (EQ(st, empty) || p == stop) + break; /* NOTE BREAK OUT */ + + /* no, we must deal with this character */ + ASSIGN(tmp, st); + ASSIGN(st, empty); + assert(c != OUT); + st = step(m->g, startst, stopst, tmp, c, st); + SP("saft", st, c); + assert(EQ(step(m->g, startst, stopst, st, NOTHING, st), st)); + p++; + } + + return(matchp); +} + + +/* + - step - map set of states reachable before char to set reachable after + == static states step(register struct re_guts *g, sopno start, sopno stop, \ + == register states bef, int ch, register states aft); + == #define BOL (OUT+1) + == #define EOL (BOL+1) + == #define BOLEOL (BOL+2) + == #define NOTHING (BOL+3) + == #define BOW (BOL+4) + == #define EOW (BOL+5) + == #define CODEMAX (BOL+5) // highest code used + == #define NONCHAR(c) ((c) > CHAR_MAX) + == #define NNONCHAR (CODEMAX-CHAR_MAX) + */ +static states +step(g, start, stop, bef, ch, aft) +register struct re_guts *g; +sopno start; /* start state within strip */ +sopno stop; /* state after stop state within strip */ +register states bef; /* states reachable before */ +int ch; /* character or NONCHAR code */ +register states aft; /* states already known reachable after */ +{ + register cset *cs; + register sop s; + register sopno pc; + register onestate here; /* note, macros know this name */ + register sopno look; + register long i; + + for (pc = start, INIT(here, pc); pc != stop; pc++, INC(here)) { + s = g->strip[pc]; + switch (OP(s)) { + case OEND: + assert(pc == stop-1); + break; + case OCHAR: + /* only characters can match */ + assert(!NONCHAR(ch) || ch != (char)OPND(s)); + if (ch == (char)OPND(s)) + FWD(aft, bef, 1); + break; + case OBOL: + if (ch == BOL || ch == BOLEOL) + FWD(aft, bef, 1); + break; + case OEOL: + if (ch == EOL || ch == BOLEOL) + FWD(aft, bef, 1); + break; + case OBOW: + if (ch == BOW) + FWD(aft, bef, 1); + break; + case OEOW: + if (ch == EOW) + FWD(aft, bef, 1); + break; + case OANY: + if (!NONCHAR(ch)) + FWD(aft, bef, 1); + break; + case OANYOF: + cs = &g->sets[OPND(s)]; + if (!NONCHAR(ch) && CHIN(cs, ch)) + FWD(aft, bef, 1); + break; + case OBACK_: /* ignored here */ + case O_BACK: + FWD(aft, aft, 1); + break; + case OPLUS_: /* forward, this is just an empty */ + FWD(aft, aft, 1); + break; + case O_PLUS: /* both forward and back */ + FWD(aft, aft, 1); + i = ISSETBACK(aft, OPND(s)); + BACK(aft, aft, OPND(s)); + if (!i && ISSETBACK(aft, OPND(s))) { + /* oho, must reconsider loop body */ + pc -= OPND(s) + 1; + INIT(here, pc); + } + break; + case OQUEST_: /* two branches, both forward */ + FWD(aft, aft, 1); + FWD(aft, aft, OPND(s)); + break; + case O_QUEST: /* just an empty */ + FWD(aft, aft, 1); + break; + case OLPAREN: /* not significant here */ + case ORPAREN: + FWD(aft, aft, 1); + break; + case OCH_: /* mark the first two branches */ + FWD(aft, aft, 1); + assert(OP(g->strip[pc+OPND(s)]) == OOR2); + FWD(aft, aft, OPND(s)); + break; + case OOR1: /* done a branch, find the O_CH */ + if (ISSTATEIN(aft, here)) { + for (look = 1; + OP(s = g->strip[pc+look]) != O_CH; + look += OPND(s)) + assert(OP(s) == OOR2); + FWD(aft, aft, look); + } + break; + case OOR2: /* propagate OCH_'s marking */ + FWD(aft, aft, 1); + if (OP(g->strip[pc+OPND(s)]) != O_CH) { + assert(OP(g->strip[pc+OPND(s)]) == OOR2); + FWD(aft, aft, OPND(s)); + } + break; + case O_CH: /* just empty */ + FWD(aft, aft, 1); + break; + default: /* ooooops... */ + assert(nope); + break; + } + } + + return(aft); +} + +#ifdef REDEBUG +/* + - print - print a set of states + == #ifdef REDEBUG + == static void print(struct match *m, char *caption, states st, \ + == int ch, FILE *d); + == #endif + */ +static void +print(m, caption, st, ch, d) +struct match *m; +char *caption; +states st; +int ch; +FILE *d; +{ + register struct re_guts *g = m->g; + register int i; + register int first = 1; + + if (!(m->eflags®_TRACE)) + return; + + fprintf(d, "%s", caption); + if (ch != '\0') + fprintf(d, " %s", pchar(ch)); + for (i = 0; i < g->nstates; i++) + if (ISSET(st, i)) { + fprintf(d, "%s%d", (first) ? "\t" : ", ", i); + first = 0; + } + fprintf(d, "\n"); +} + +/* + - at - print current situation + == #ifdef REDEBUG + == static void at(struct match *m, char *title, char *start, char *stop, \ + == sopno startst, sopno stopst); + == #endif + */ +static void +at(m, title, start, stop, startst, stopst) +struct match *m; +char *title; +char *start; +char *stop; +sopno startst; +sopno stopst; +{ + if (!(m->eflags®_TRACE)) + return; + + printf("%s %s-", title, pchar(*start)); + printf("%s ", pchar(*stop)); + printf("%ld-%ld\n", (long)startst, (long)stopst); +} + +#ifndef PCHARDONE +#define PCHARDONE /* never again */ +/* + - pchar - make a character printable + == #ifdef REDEBUG + == static char *pchar(int ch); + == #endif + * + * Is this identical to regchar() over in debug.c? Well, yes. But a + * duplicate here avoids having a debugging-capable regexec.o tied to + * a matching debug.o, and this is convenient. It all disappears in + * the non-debug compilation anyway, so it doesn't matter much. + */ +static char * /* -> representation */ +pchar(ch) +int ch; +{ + static char pbuf[10]; + + if (isprint(ch) || ch == ' ') + sprintf(pbuf, "%c", ch); + else + sprintf(pbuf, "\\%o", ch); + return(pbuf); +} +#endif +#endif + +#undef matcher +#undef fast +#undef slow +#undef dissect +#undef backref +#undef step +#undef print +#undef at +#undef match diff --git a/vcnet/regex/engine.ih b/vcnet/regex/engine.ih new file mode 100644 index 0000000000..cc98334e75 --- /dev/null +++ b/vcnet/regex/engine.ih @@ -0,0 +1,35 @@ +/* ========= begin header generated by ./mkh ========= */ +#ifdef __cplusplus +extern "C" { +#endif + +/* === engine.c === */ +static int matcher(register struct re_guts *g, char *string, size_t nmatch, regmatch_t pmatch[], int eflags); +static char *dissect(register struct match *m, char *start, char *stop, sopno startst, sopno stopst); +static char *backref(register struct match *m, char *start, char *stop, sopno startst, sopno stopst, sopno lev); +static char *fast(register struct match *m, char *start, char *stop, sopno startst, sopno stopst); +static char *slow(register struct match *m, char *start, char *stop, sopno startst, sopno stopst); +static states step(register struct re_guts *g, sopno start, sopno stop, register states bef, int ch, register states aft); +#define BOL (OUT+1) +#define EOL (BOL+1) +#define BOLEOL (BOL+2) +#define NOTHING (BOL+3) +#define BOW (BOL+4) +#define EOW (BOL+5) +#define CODEMAX (BOL+5) /* highest code used */ +#define NONCHAR(c) ((c) > CHAR_MAX) +#define NNONCHAR (CODEMAX-CHAR_MAX) +#ifdef REDEBUG +static void print(struct match *m, char *caption, states st, int ch, FILE *d); +#endif +#ifdef REDEBUG +static void at(struct match *m, char *title, char *start, char *stop, sopno startst, sopno stopst); +#endif +#ifdef REDEBUG +static char *pchar(int ch); +#endif + +#ifdef __cplusplus +} +#endif +/* ========= end header generated by ./mkh ========= */ diff --git a/vcnet/regex/main.c b/vcnet/regex/main.c new file mode 100644 index 0000000000..0221e7713d --- /dev/null +++ b/vcnet/regex/main.c @@ -0,0 +1,510 @@ +#include +#include +#include +#include +#include + +#include "main.ih" + +char *progname; +int debug = 0; +int line = 0; +int status = 0; + +int copts = REG_EXTENDED; +int eopts = 0; +regoff_t startoff = 0; +regoff_t endoff = 0; + + +extern int split(); +extern void regprint(); + +/* + - main - do the simple case, hand off to regress() for regression + */ +main(argc, argv) +int argc; +char *argv[]; +{ + regex_t re; +# define NS 10 + regmatch_t subs[NS]; + char erbuf[100]; + int err; + size_t len; + int c; + int errflg = 0; + register int i; + extern int optind; + extern char *optarg; + + progname = argv[0]; + + while ((c = getopt(argc, argv, "c:e:S:E:x")) != EOF) + switch (c) { + case 'c': /* compile options */ + copts = options('c', optarg); + break; + case 'e': /* execute options */ + eopts = options('e', optarg); + break; + case 'S': /* start offset */ + startoff = (regoff_t)atoi(optarg); + break; + case 'E': /* end offset */ + endoff = (regoff_t)atoi(optarg); + break; + case 'x': /* Debugging. */ + debug++; + break; + case '?': + default: + errflg++; + break; + } + if (errflg) { + fprintf(stderr, "usage: %s ", progname); + fprintf(stderr, "[-c copt][-C][-d] [re]\n"); + exit(2); + } + + if (optind >= argc) { + regress(stdin); + exit(status); + } + + err = regcomp(&re, argv[optind++], copts); + if (err) { + len = regerror(err, &re, erbuf, sizeof(erbuf)); + fprintf(stderr, "error %s, %d/%d `%s'\n", + eprint(err), len, sizeof(erbuf), erbuf); + exit(status); + } + regprint(&re, stdout); + + if (optind >= argc) { + regfree(&re); + exit(status); + } + + if (eopts®_STARTEND) { + subs[0].rm_so = startoff; + subs[0].rm_eo = strlen(argv[optind]) - endoff; + } + err = regexec(&re, argv[optind], (size_t)NS, subs, eopts); + if (err) { + len = regerror(err, &re, erbuf, sizeof(erbuf)); + fprintf(stderr, "error %s, %d/%d `%s'\n", + eprint(err), len, sizeof(erbuf), erbuf); + exit(status); + } + if (!(copts®_NOSUB)) { + len = (int)(subs[0].rm_eo - subs[0].rm_so); + if (subs[0].rm_so != -1) { + if (len != 0) + printf("match `%.*s'\n", len, + argv[optind] + subs[0].rm_so); + else + printf("match `'@%.1s\n", + argv[optind] + subs[0].rm_so); + } + for (i = 1; i < NS; i++) + if (subs[i].rm_so != -1) + printf("(%d) `%.*s'\n", i, + (int)(subs[i].rm_eo - subs[i].rm_so), + argv[optind] + subs[i].rm_so); + } + exit(status); +} + +/* + - regress - main loop of regression test + == void regress(FILE *in); + */ +void +regress(in) +FILE *in; +{ + char inbuf[1000]; +# define MAXF 10 + char *f[MAXF]; + int nf; + int i; + char erbuf[100]; + size_t ne; + char *badpat = "invalid regular expression"; +# define SHORT 10 + char *bpname = "REG_BADPAT"; + regex_t re; + + while (fgets(inbuf, sizeof(inbuf), in) != NULL) { + line++; + if (inbuf[0] == '#' || inbuf[0] == '\n') + continue; /* NOTE CONTINUE */ + inbuf[strlen(inbuf)-1] = '\0'; /* get rid of stupid \n */ + if (debug) + fprintf(stdout, "%d:\n", line); + nf = split(inbuf, f, MAXF, "\t\t"); + if (nf < 3) { + fprintf(stderr, "bad input, line %d\n", line); + exit(1); + } + for (i = 0; i < nf; i++) + if (strcmp(f[i], "\"\"") == 0) + f[i] = ""; + if (nf <= 3) + f[3] = NULL; + if (nf <= 4) + f[4] = NULL; + try(f[0], f[1], f[2], f[3], f[4], options('c', f[1])); + if (opt('&', f[1])) /* try with either type of RE */ + try(f[0], f[1], f[2], f[3], f[4], + options('c', f[1]) &~ REG_EXTENDED); + } + + ne = regerror(REG_BADPAT, (regex_t *)NULL, erbuf, sizeof(erbuf)); + if (strcmp(erbuf, badpat) != 0 || ne != strlen(badpat)+1) { + fprintf(stderr, "end: regerror() test gave `%s' not `%s'\n", + erbuf, badpat); + status = 1; + } + ne = regerror(REG_BADPAT, (regex_t *)NULL, erbuf, (size_t)SHORT); + if (strncmp(erbuf, badpat, SHORT-1) != 0 || erbuf[SHORT-1] != '\0' || + ne != strlen(badpat)+1) { + fprintf(stderr, "end: regerror() short test gave `%s' not `%.*s'\n", + erbuf, SHORT-1, badpat); + status = 1; + } + ne = regerror(REG_ITOA|REG_BADPAT, (regex_t *)NULL, erbuf, sizeof(erbuf)); + if (strcmp(erbuf, bpname) != 0 || ne != strlen(bpname)+1) { + fprintf(stderr, "end: regerror() ITOA test gave `%s' not `%s'\n", + erbuf, bpname); + status = 1; + } + re.re_endp = bpname; + ne = regerror(REG_ATOI, &re, erbuf, sizeof(erbuf)); + if (atoi(erbuf) != (int)REG_BADPAT) { + fprintf(stderr, "end: regerror() ATOI test gave `%s' not `%ld'\n", + erbuf, (long)REG_BADPAT); + status = 1; + } else if (ne != strlen(erbuf)+1) { + fprintf(stderr, "end: regerror() ATOI test len(`%s') = %ld\n", + erbuf, (long)REG_BADPAT); + status = 1; + } +} + +/* + - try - try it, and report on problems + == void try(char *f0, char *f1, char *f2, char *f3, char *f4, int opts); + */ +void +try(f0, f1, f2, f3, f4, opts) +char *f0; +char *f1; +char *f2; +char *f3; +char *f4; +int opts; /* may not match f1 */ +{ + regex_t re; +# define NSUBS 10 + regmatch_t subs[NSUBS]; +# define NSHOULD 15 + char *should[NSHOULD]; + int nshould; + char erbuf[100]; + int err; + int len; + char *type = (opts & REG_EXTENDED) ? "ERE" : "BRE"; + register int i; + char *grump; + char f0copy[1000]; + char f2copy[1000]; + + strcpy(f0copy, f0); + re.re_endp = (opts®_PEND) ? f0copy + strlen(f0copy) : NULL; + fixstr(f0copy); + err = regcomp(&re, f0copy, opts); + if (err != 0 && (!opt('C', f1) || err != efind(f2))) { + /* unexpected error or wrong error */ + len = regerror(err, &re, erbuf, sizeof(erbuf)); + fprintf(stderr, "%d: %s error %s, %d/%d `%s'\n", + line, type, eprint(err), len, + sizeof(erbuf), erbuf); + status = 1; + } else if (err == 0 && opt('C', f1)) { + /* unexpected success */ + fprintf(stderr, "%d: %s should have given REG_%s\n", + line, type, f2); + status = 1; + err = 1; /* so we won't try regexec */ + } + + if (err != 0) { + regfree(&re); + return; + } + + strcpy(f2copy, f2); + fixstr(f2copy); + + if (options('e', f1)®_STARTEND) { + if (strchr(f2, '(') == NULL || strchr(f2, ')') == NULL) + fprintf(stderr, "%d: bad STARTEND syntax\n", line); + subs[0].rm_so = strchr(f2, '(') - f2 + 1; + subs[0].rm_eo = strchr(f2, ')') - f2; + } + err = regexec(&re, f2copy, NSUBS, subs, options('e', f1)); + + if (err != 0 && (f3 != NULL || err != REG_NOMATCH)) { + /* unexpected error or wrong error */ + len = regerror(err, &re, erbuf, sizeof(erbuf)); + fprintf(stderr, "%d: %s exec error %s, %d/%d `%s'\n", + line, type, eprint(err), len, + sizeof(erbuf), erbuf); + status = 1; + } else if (err != 0) { + /* nothing more to check */ + } else if (f3 == NULL) { + /* unexpected success */ + fprintf(stderr, "%d: %s exec should have failed\n", + line, type); + status = 1; + err = 1; /* just on principle */ + } else if (opts®_NOSUB) { + /* nothing more to check */ + } else if ((grump = check(f2, subs[0], f3)) != NULL) { + fprintf(stderr, "%d: %s %s\n", line, type, grump); + status = 1; + err = 1; + } + + if (err != 0 || f4 == NULL) { + regfree(&re); + return; + } + + for (i = 1; i < NSHOULD; i++) + should[i] = NULL; + nshould = split(f4, should+1, NSHOULD-1, ","); + if (nshould == 0) { + nshould = 1; + should[1] = ""; + } + for (i = 1; i < NSUBS; i++) { + grump = check(f2, subs[i], should[i]); + if (grump != NULL) { + fprintf(stderr, "%d: %s $%d %s\n", line, + type, i, grump); + status = 1; + err = 1; + } + } + + regfree(&re); +} + +/* + - options - pick options out of a regression-test string + == int options(int type, char *s); + */ +int +options(type, s) +int type; /* 'c' compile, 'e' exec */ +char *s; +{ + register char *p; + register int o = (type == 'c') ? copts : eopts; + register char *legal = (type == 'c') ? "bisnmp" : "^$#tl"; + + for (p = s; *p != '\0'; p++) + if (strchr(legal, *p) != NULL) + switch (*p) { + case 'b': + o &= ~REG_EXTENDED; + break; + case 'i': + o |= REG_ICASE; + break; + case 's': + o |= REG_NOSUB; + break; + case 'n': + o |= REG_NEWLINE; + break; + case 'm': + o &= ~REG_EXTENDED; + o |= REG_NOSPEC; + break; + case 'p': + o |= REG_PEND; + break; + case '^': + o |= REG_NOTBOL; + break; + case '$': + o |= REG_NOTEOL; + break; + case '#': + o |= REG_STARTEND; + break; + case 't': /* trace */ + o |= REG_TRACE; + break; + case 'l': /* force long representation */ + o |= REG_LARGE; + break; + case 'r': /* force backref use */ + o |= REG_BACKR; + break; + } + return(o); +} + +/* + - opt - is a particular option in a regression string? + == int opt(int c, char *s); + */ +int /* predicate */ +opt(c, s) +int c; +char *s; +{ + return(strchr(s, c) != NULL); +} + +/* + - fixstr - transform magic characters in strings + == void fixstr(register char *p); + */ +void +fixstr(p) +register char *p; +{ + if (p == NULL) + return; + + for (; *p != '\0'; p++) + if (*p == 'N') + *p = '\n'; + else if (*p == 'T') + *p = '\t'; + else if (*p == 'S') + *p = ' '; + else if (*p == 'Z') + *p = '\0'; +} + +/* + - check - check a substring match + == char *check(char *str, regmatch_t sub, char *should); + */ +char * /* NULL or complaint */ +check(str, sub, should) +char *str; +regmatch_t sub; +char *should; +{ + register int len; + register int shlen; + register char *p; + static char grump[500]; + register char *at = NULL; + + if (should != NULL && strcmp(should, "-") == 0) + should = NULL; + if (should != NULL && should[0] == '@') { + at = should + 1; + should = ""; + } + + /* check rm_so and rm_eo for consistency */ + if (sub.rm_so > sub.rm_eo || (sub.rm_so == -1 && sub.rm_eo != -1) || + (sub.rm_so != -1 && sub.rm_eo == -1) || + (sub.rm_so != -1 && sub.rm_so < 0) || + (sub.rm_eo != -1 && sub.rm_eo < 0) ) { + sprintf(grump, "start %ld end %ld", (long)sub.rm_so, + (long)sub.rm_eo); + return(grump); + } + + /* check for no match */ + if (sub.rm_so == -1 && should == NULL) + return(NULL); + if (sub.rm_so == -1) + return("did not match"); + + /* check for in range */ + if (sub.rm_eo > strlen(str)) { + sprintf(grump, "start %ld end %ld, past end of string", + (long)sub.rm_so, (long)sub.rm_eo); + return(grump); + } + + len = (int)(sub.rm_eo - sub.rm_so); + shlen = (int)strlen(should); + p = str + sub.rm_so; + + /* check for not supposed to match */ + if (should == NULL) { + sprintf(grump, "matched `%.*s'", len, p); + return(grump); + } + + /* check for wrong match */ + if (len != shlen || strncmp(p, should, (size_t)shlen) != 0) { + sprintf(grump, "matched `%.*s' instead", len, p); + return(grump); + } + if (shlen > 0) + return(NULL); + + /* check null match in right place */ + if (at == NULL) + return(NULL); + shlen = strlen(at); + if (shlen == 0) + shlen = 1; /* force check for end-of-string */ + if (strncmp(p, at, shlen) != 0) { + sprintf(grump, "matched null at `%.20s'", p); + return(grump); + } + return(NULL); +} + +/* + - eprint - convert error number to name + == static char *eprint(int err); + */ +static char * +eprint(err) +int err; +{ + static char epbuf[100]; + size_t len; + + len = regerror(REG_ITOA|err, (regex_t *)NULL, epbuf, sizeof(epbuf)); + assert(len <= sizeof(epbuf)); + return(epbuf); +} + +/* + - efind - convert error name to number + == static int efind(char *name); + */ +static int +efind(name) +char *name; +{ + static char efbuf[100]; + size_t n; + regex_t re; + + sprintf(efbuf, "REG_%s", name); + assert(strlen(efbuf) < sizeof(efbuf)); + re.re_endp = efbuf; + (void) regerror(REG_ATOI, &re, efbuf, sizeof(efbuf)); + return(atoi(efbuf)); +} diff --git a/vcnet/regex/main.ih b/vcnet/regex/main.ih new file mode 100644 index 0000000000..5a0118ac44 --- /dev/null +++ b/vcnet/regex/main.ih @@ -0,0 +1,19 @@ +/* ========= begin header generated by ./mkh ========= */ +#ifdef __cplusplus +extern "C" { +#endif + +/* === main.c === */ +void regress(FILE *in); +void try(char *f0, char *f1, char *f2, char *f3, char *f4, int opts); +int options(int type, char *s); +int opt(int c, char *s); +void fixstr(register char *p); +char *check(char *str, regmatch_t sub, char *should); +static char *eprint(int err); +static int efind(char *name); + +#ifdef __cplusplus +} +#endif +/* ========= end header generated by ./mkh ========= */ diff --git a/vcnet/regex/mkh b/vcnet/regex/mkh new file mode 100644 index 0000000000..252b246c7b --- /dev/null +++ b/vcnet/regex/mkh @@ -0,0 +1,76 @@ +#! /bin/sh +# mkh - pull headers out of C source +PATH=/bin:/usr/bin ; export PATH + +# egrep pattern to pick out marked lines +egrep='^ =([ ]|$)' + +# Sed program to process marked lines into lines for the header file. +# The markers have already been removed. Two things are done here: removal +# of backslashed newlines, and some fudging of comments. The first is done +# because -o needs to have prototypes on one line to strip them down. +# Getting comments into the output is tricky; we turn C++-style // comments +# into /* */ comments, after altering any existing */'s to avoid trouble. +peel=' /\\$/N + /\\\n[ ]*/s///g + /\/\//s;\*/;* /;g + /\/\//s;//\(.*\);/*\1 */;' + +for a +do + case "$a" in + -o) # old (pre-function-prototype) compiler + # add code to comment out argument lists + peel="$peel + "'/^\([^#\/][^\/]*[a-zA-Z0-9_)]\)(\(.*\))/s;;\1(/*\2*/);' + shift + ;; + -b) # funny Berkeley __P macro + peel="$peel + "'/^\([^#\/][^\/]*[a-zA-Z0-9_)]\)(\(.*\))/s;;\1 __P((\2));' + shift + ;; + -s) # compiler doesn't like `static foo();' + # add code to get rid of the `static' + peel="$peel + "'/^static[ ][^\/]*[a-zA-Z0-9_)](.*)/s;static.;;' + shift + ;; + -p) # private declarations + egrep='^ ==([ ]|$)' + shift + ;; + -i) # wrap in #ifndef, argument is name + ifndef="$2" + shift ; shift + ;; + *) break + ;; + esac +done + +if test " $ifndef" != " " +then + echo "#ifndef $ifndef" + echo "#define $ifndef /* never again */" +fi +echo "/* ========= begin header generated by $0 ========= */" +echo '#ifdef __cplusplus' +echo 'extern "C" {' +echo '#endif' +for f +do + echo + echo "/* === $f === */" + egrep "$egrep" $f | sed 's/^ ==*[ ]//;s/^ ==*$//' | sed "$peel" + echo +done +echo '#ifdef __cplusplus' +echo '}' +echo '#endif' +echo "/* ========= end header generated by $0 ========= */" +if test " $ifndef" != " " +then + echo "#endif" +fi +exit 0 diff --git a/vcnet/regex/regcomp.c b/vcnet/regex/regcomp.c new file mode 100644 index 0000000000..5df5d53fc1 --- /dev/null +++ b/vcnet/regex/regcomp.c @@ -0,0 +1,1603 @@ +#include +#include +#include +#include +#include +#include +#include + +#include "utils.h" +#include "regex2.h" + +#include "cclass.h" +#include "cname.h" + +/* + * parse structure, passed up and down to avoid global variables and + * other clumsinesses + */ +struct parse { + char *next; /* next character in RE */ + char *end; /* end of string (-> NUL normally) */ + int error; /* has an error been seen? */ + sop *strip; /* malloced strip */ + sopno ssize; /* malloced strip size (allocated) */ + sopno slen; /* malloced strip length (used) */ + int ncsalloc; /* number of csets allocated */ + struct re_guts *g; +# define NPAREN 10 /* we need to remember () 1-9 for back refs */ + sopno pbegin[NPAREN]; /* -> ( ([0] unused) */ + sopno pend[NPAREN]; /* -> ) ([0] unused) */ +}; + +#include "regcomp.ih" + +static char nuls[10]; /* place to point scanner in event of error */ + +/* + * macros for use with parse structure + * BEWARE: these know that the parse structure is named `p' !!! + */ +#define PEEK() (*p->next) +#define PEEK2() (*(p->next+1)) +#define MORE() (p->next < p->end) +#define MORE2() (p->next+1 < p->end) +#define SEE(c) (MORE() && PEEK() == (c)) +#define SEETWO(a, b) (MORE() && MORE2() && PEEK() == (a) && PEEK2() == (b)) +#define EAT(c) ((SEE(c)) ? (NEXT(), 1) : 0) +#define EATTWO(a, b) ((SEETWO(a, b)) ? (NEXT2(), 1) : 0) +#define NEXT() (p->next++) +#define NEXT2() (p->next += 2) +#define NEXTn(n) (p->next += (n)) +#define GETNEXT() (*p->next++) +#define SETERROR(e) seterr(p, (e)) +#define REQUIRE(co, e) ((co) || SETERROR(e)) +#define MUSTSEE(c, e) (REQUIRE(MORE() && PEEK() == (c), e)) +#define MUSTEAT(c, e) (REQUIRE(MORE() && GETNEXT() == (c), e)) +#define MUSTNOTSEE(c, e) (REQUIRE(!MORE() || PEEK() != (c), e)) +#define EMIT(op, sopnd) doemit(p, (sop)(op), (size_t)(sopnd)) +#define INSERT(op, pos) doinsert(p, (sop)(op), HERE()-(pos)+1, pos) +#define AHEAD(pos) dofwd(p, pos, HERE()-(pos)) +#define ASTERN(sop, pos) EMIT(sop, HERE()-pos) +#define HERE() (p->slen) +#define THERE() (p->slen - 1) +#define THERETHERE() (p->slen - 2) +#define DROP(n) (p->slen -= (n)) + +#ifndef NDEBUG +static int never = 0; /* for use in asserts; shuts lint up */ +#else +#define never 0 /* some s have bugs too */ +#endif + +/* + - regcomp - interface for parser and compilation + = extern int regcomp(regex_t *, const char *, int); + = #define REG_BASIC 0000 + = #define REG_EXTENDED 0001 + = #define REG_ICASE 0002 + = #define REG_NOSUB 0004 + = #define REG_NEWLINE 0010 + = #define REG_NOSPEC 0020 + = #define REG_PEND 0040 + = #define REG_DUMP 0200 + */ +int /* 0 success, otherwise REG_something */ +regcomp(preg, pattern, cflags) +regex_t *preg; +const char *pattern; +int cflags; +{ + struct parse pa; + register struct re_guts *g; + register struct parse *p = &pa; + register int i; + register size_t len; +#ifdef REDEBUG +# define GOODFLAGS(f) (f) +#else +# define GOODFLAGS(f) ((f)&~REG_DUMP) +#endif + + cflags = GOODFLAGS(cflags); + if ((cflags®_EXTENDED) && (cflags®_NOSPEC)) + return(REG_INVARG); + + if (cflags®_PEND) { + if (preg->re_endp < pattern) + return(REG_INVARG); + len = preg->re_endp - pattern; + } else + len = strlen((char *)pattern); + + /* do the mallocs early so failure handling is easy */ + g = (struct re_guts *)malloc(sizeof(struct re_guts) + + (NC-1)*sizeof(cat_t)); + if (g == NULL) + return(REG_ESPACE); + p->ssize = len/(size_t)2*(size_t)3 + (size_t)1; /* ugh */ + p->strip = (sop *)malloc(p->ssize * sizeof(sop)); + p->slen = 0; + if (p->strip == NULL) { + free((char *)g); + return(REG_ESPACE); + } + + /* set things up */ + p->g = g; + p->next = (char *)pattern; /* convenience; we do not modify it */ + p->end = p->next + len; + p->error = 0; + p->ncsalloc = 0; + for (i = 0; i < NPAREN; i++) { + p->pbegin[i] = 0; + p->pend[i] = 0; + } + g->csetsize = NC; + g->sets = NULL; + g->setbits = NULL; + g->ncsets = 0; + g->cflags = cflags; + g->iflags = 0; + g->nbol = 0; + g->neol = 0; + g->must = NULL; + g->mlen = 0; + g->nsub = 0; + g->ncategories = 1; /* category 0 is "everything else" */ + g->categories = &g->catspace[-(CHAR_MIN)]; + (void) memset((char *)g->catspace, 0, NC*sizeof(cat_t)); + g->backrefs = 0; + + /* do it */ + EMIT(OEND, 0); + g->firststate = THERE(); + if (cflags®_EXTENDED) + p_ere(p, OUT); + else if (cflags®_NOSPEC) + p_str(p); + else + p_bre(p, OUT, OUT); + EMIT(OEND, 0); + g->laststate = THERE(); + + /* tidy up loose ends and fill things in */ + categorize(p, g); + stripsnug(p, g); + findmust(p, g); + g->nplus = pluscount(p, g); + g->magic = MAGIC2; + preg->re_nsub = g->nsub; + preg->re_g = g; + preg->re_magic = MAGIC1; +#ifndef REDEBUG + /* not debugging, so can't rely on the assert() in regexec() */ + if (g->iflags&BAD) + SETERROR(REG_ASSERT); +#endif + + /* win or lose, we're done */ + if (p->error != 0) /* lose */ + regfree(preg); + return(p->error); +} + +/* + - p_ere - ERE parser top level, concatenation and alternation + == static void p_ere(register struct parse *p, int stop); + */ +static void +p_ere(p, stop) +register struct parse *p; +int stop; /* character this ERE should end at */ +{ + register char c; + register sopno prevback; + register sopno prevfwd; + register sopno conc; + register int first = 1; /* is this the first alternative? */ + + for (;;) { + /* do a bunch of concatenated expressions */ + conc = HERE(); + while (MORE() && (c = PEEK()) != '|' && c != stop) + p_ere_exp(p); + REQUIRE(HERE() != conc, REG_EMPTY); /* require nonempty */ + + if (!EAT('|')) + break; /* NOTE BREAK OUT */ + + if (first) { + INSERT(OCH_, conc); /* offset is wrong */ + prevfwd = conc; + prevback = conc; + first = 0; + } + ASTERN(OOR1, prevback); + prevback = THERE(); + AHEAD(prevfwd); /* fix previous offset */ + prevfwd = HERE(); + EMIT(OOR2, 0); /* offset is very wrong */ + } + + if (!first) { /* tail-end fixups */ + AHEAD(prevfwd); + ASTERN(O_CH, prevback); + } + + assert(!MORE() || SEE(stop)); +} + +/* + - p_ere_exp - parse one subERE, an atom possibly followed by a repetition op + == static void p_ere_exp(register struct parse *p); + */ +static void +p_ere_exp(p) +register struct parse *p; +{ + register char c; + register sopno pos; + register int count; + register int count2; + register sopno subno; + int wascaret = 0; + + assert(MORE()); /* caller should have ensured this */ + c = GETNEXT(); + + pos = HERE(); + switch (c) { + case '(': + REQUIRE(MORE(), REG_EPAREN); + p->g->nsub++; + subno = p->g->nsub; + if (subno < NPAREN) + p->pbegin[subno] = HERE(); + EMIT(OLPAREN, subno); + if (!SEE(')')) + p_ere(p, ')'); + if (subno < NPAREN) { + p->pend[subno] = HERE(); + assert(p->pend[subno] != 0); + } + EMIT(ORPAREN, subno); + MUSTEAT(')', REG_EPAREN); + break; +#ifndef POSIX_MISTAKE + case ')': /* happens only if no current unmatched ( */ + /* + * You may ask, why the ifndef? Because I didn't notice + * this until slightly too late for 1003.2, and none of the + * other 1003.2 regular-expression reviewers noticed it at + * all. So an unmatched ) is legal POSIX, at least until + * we can get it fixed. + */ + SETERROR(REG_EPAREN); + break; +#endif + case '^': + EMIT(OBOL, 0); + p->g->iflags |= USEBOL; + p->g->nbol++; + wascaret = 1; + break; + case '$': + EMIT(OEOL, 0); + p->g->iflags |= USEEOL; + p->g->neol++; + break; + case '|': + SETERROR(REG_EMPTY); + break; + case '*': + case '+': + case '?': + SETERROR(REG_BADRPT); + break; + case '.': + if (p->g->cflags®_NEWLINE) + nonnewline(p); + else + EMIT(OANY, 0); + break; + case '[': + p_bracket(p); + break; + case '\\': + REQUIRE(MORE(), REG_EESCAPE); + c = GETNEXT(); + ordinary(p, c); + break; + case '{': /* okay as ordinary except if digit follows */ + REQUIRE(!MORE() || !isdigit(PEEK()), REG_BADRPT); + /* FALLTHROUGH */ + default: + ordinary(p, c); + break; + } + + if (!MORE()) + return; + c = PEEK(); + /* we call { a repetition if followed by a digit */ + if (!( c == '*' || c == '+' || c == '?' || + (c == '{' && MORE2() && isdigit(PEEK2())) )) + return; /* no repetition, we're done */ + NEXT(); + + REQUIRE(!wascaret, REG_BADRPT); + switch (c) { + case '*': /* implemented as +? */ + /* this case does not require the (y|) trick, noKLUDGE */ + INSERT(OPLUS_, pos); + ASTERN(O_PLUS, pos); + INSERT(OQUEST_, pos); + ASTERN(O_QUEST, pos); + break; + case '+': + INSERT(OPLUS_, pos); + ASTERN(O_PLUS, pos); + break; + case '?': + /* KLUDGE: emit y? as (y|) until subtle bug gets fixed */ + INSERT(OCH_, pos); /* offset slightly wrong */ + ASTERN(OOR1, pos); /* this one's right */ + AHEAD(pos); /* fix the OCH_ */ + EMIT(OOR2, 0); /* offset very wrong... */ + AHEAD(THERE()); /* ...so fix it */ + ASTERN(O_CH, THERETHERE()); + break; + case '{': + count = p_count(p); + if (EAT(',')) { + if (isdigit(PEEK())) { + count2 = p_count(p); + REQUIRE(count <= count2, REG_BADBR); + } else /* single number with comma */ + count2 = INFINITY; + } else /* just a single number */ + count2 = count; + repeat(p, pos, count, count2); + if (!EAT('}')) { /* error heuristics */ + while (MORE() && PEEK() != '}') + NEXT(); + REQUIRE(MORE(), REG_EBRACE); + SETERROR(REG_BADBR); + } + break; + } + + if (!MORE()) + return; + c = PEEK(); + if (!( c == '*' || c == '+' || c == '?' || + (c == '{' && MORE2() && isdigit(PEEK2())) ) ) + return; + SETERROR(REG_BADRPT); +} + +/* + - p_str - string (no metacharacters) "parser" + == static void p_str(register struct parse *p); + */ +static void +p_str(p) +register struct parse *p; +{ + REQUIRE(MORE(), REG_EMPTY); + while (MORE()) + ordinary(p, GETNEXT()); +} + +/* + - p_bre - BRE parser top level, anchoring and concatenation + == static void p_bre(register struct parse *p, register int end1, \ + == register int end2); + * Giving end1 as OUT essentially eliminates the end1/end2 check. + * + * This implementation is a bit of a kludge, in that a trailing $ is first + * taken as an ordinary character and then revised to be an anchor. The + * only undesirable side effect is that '$' gets included as a character + * category in such cases. This is fairly harmless; not worth fixing. + * The amount of lookahead needed to avoid this kludge is excessive. + */ +static void +p_bre(p, end1, end2) +register struct parse *p; +register int end1; /* first terminating character */ +register int end2; /* second terminating character */ +{ + register sopno start = HERE(); + register int first = 1; /* first subexpression? */ + register int wasdollar = 0; + + if (EAT('^')) { + EMIT(OBOL, 0); + p->g->iflags |= USEBOL; + p->g->nbol++; + } + while (MORE() && !SEETWO(end1, end2)) { + wasdollar = p_simp_re(p, first); + first = 0; + } + if (wasdollar) { /* oops, that was a trailing anchor */ + DROP(1); + EMIT(OEOL, 0); + p->g->iflags |= USEEOL; + p->g->neol++; + } + + REQUIRE(HERE() != start, REG_EMPTY); /* require nonempty */ +} + +/* + - p_simp_re - parse a simple RE, an atom possibly followed by a repetition + == static int p_simp_re(register struct parse *p, int starordinary); + */ +static int /* was the simple RE an unbackslashed $? */ +p_simp_re(p, starordinary) +register struct parse *p; +int starordinary; /* is a leading * an ordinary character? */ +{ + register int c; + register int count; + register int count2; + register sopno pos; + register int i; + register sopno subno; +# define BACKSL (1<g->cflags®_NEWLINE) + nonnewline(p); + else + EMIT(OANY, 0); + break; + case '[': + p_bracket(p); + break; + case BACKSL|'{': + SETERROR(REG_BADRPT); + break; + case BACKSL|'(': + p->g->nsub++; + subno = p->g->nsub; + if (subno < NPAREN) + p->pbegin[subno] = HERE(); + EMIT(OLPAREN, subno); + /* the MORE here is an error heuristic */ + if (MORE() && !SEETWO('\\', ')')) + p_bre(p, '\\', ')'); + if (subno < NPAREN) { + p->pend[subno] = HERE(); + assert(p->pend[subno] != 0); + } + EMIT(ORPAREN, subno); + REQUIRE(EATTWO('\\', ')'), REG_EPAREN); + break; + case BACKSL|')': /* should not get here -- must be user */ + case BACKSL|'}': + SETERROR(REG_EPAREN); + break; + case BACKSL|'1': + case BACKSL|'2': + case BACKSL|'3': + case BACKSL|'4': + case BACKSL|'5': + case BACKSL|'6': + case BACKSL|'7': + case BACKSL|'8': + case BACKSL|'9': + i = (c&~BACKSL) - '0'; + assert(i < NPAREN); + if (p->pend[i] != 0) { + assert(i <= p->g->nsub); + EMIT(OBACK_, i); + assert(p->pbegin[i] != 0); + assert(OP(p->strip[p->pbegin[i]]) == OLPAREN); + assert(OP(p->strip[p->pend[i]]) == ORPAREN); + (void) dupl(p, p->pbegin[i]+1, p->pend[i]); + EMIT(O_BACK, i); + } else + SETERROR(REG_ESUBREG); + p->g->backrefs = 1; + break; + case '*': + REQUIRE(starordinary, REG_BADRPT); + /* FALLTHROUGH */ + default: + ordinary(p, (char)c); /* takes off BACKSL, if any */ + break; + } + + if (EAT('*')) { /* implemented as +? */ + /* this case does not require the (y|) trick, noKLUDGE */ + INSERT(OPLUS_, pos); + ASTERN(O_PLUS, pos); + INSERT(OQUEST_, pos); + ASTERN(O_QUEST, pos); + } else if (EATTWO('\\', '{')) { + count = p_count(p); + if (EAT(',')) { + if (MORE() && isdigit(PEEK())) { + count2 = p_count(p); + REQUIRE(count <= count2, REG_BADBR); + } else /* single number with comma */ + count2 = INFINITY; + } else /* just a single number */ + count2 = count; + repeat(p, pos, count, count2); + if (!EATTWO('\\', '}')) { /* error heuristics */ + while (MORE() && !SEETWO('\\', '}')) + NEXT(); + REQUIRE(MORE(), REG_EBRACE); + SETERROR(REG_BADBR); + } + } else if (c == (unsigned char)'$') /* $ (but not \$) ends it */ + return(1); + + return(0); +} + +/* + - p_count - parse a repetition count + == static int p_count(register struct parse *p); + */ +static int /* the value */ +p_count(p) +register struct parse *p; +{ + register int count = 0; + register int ndigits = 0; + + while (MORE() && isdigit(PEEK()) && count <= DUPMAX) { + count = count*10 + (GETNEXT() - '0'); + ndigits++; + } + + REQUIRE(ndigits > 0 && count <= DUPMAX, REG_BADBR); + return(count); +} + +/* + - p_bracket - parse a bracketed character list + == static void p_bracket(register struct parse *p); + * + * Note a significant property of this code: if the allocset() did SETERROR, + * no set operations are done. + */ +static void +p_bracket(p) +register struct parse *p; +{ + register cset *cs = allocset(p); + register int invert = 0; + + /* Dept of Truly Sickening Special-Case Kludges */ + if (p->next + 5 < p->end && strncmp(p->next, "[:<:]]", 6) == 0) { + EMIT(OBOW, 0); + NEXTn(6); + return; + } + if (p->next + 5 < p->end && strncmp(p->next, "[:>:]]", 6) == 0) { + EMIT(OEOW, 0); + NEXTn(6); + return; + } + + if (EAT('^')) + invert++; /* make note to invert set at end */ + if (EAT(']')) + CHadd(cs, ']'); + else if (EAT('-')) + CHadd(cs, '-'); + while (MORE() && PEEK() != ']' && !SEETWO('-', ']')) + p_b_term(p, cs); + if (EAT('-')) + CHadd(cs, '-'); + MUSTEAT(']', REG_EBRACK); + + if (p->error != 0) /* don't mess things up further */ + return; + + if (p->g->cflags®_ICASE) { + register int i; + register int ci; + + for (i = p->g->csetsize - 1; i >= 0; i--) + if (CHIN(cs, i) && isalpha(i)) { + ci = othercase(i); + if (ci != i) + CHadd(cs, ci); + } + if (cs->multis != NULL) + mccase(p, cs); + } + if (invert) { + register int i; + + for (i = p->g->csetsize - 1; i >= 0; i--) + if (CHIN(cs, i)) + CHsub(cs, i); + else + CHadd(cs, i); + if (p->g->cflags®_NEWLINE) + CHsub(cs, '\n'); + if (cs->multis != NULL) + mcinvert(p, cs); + } + + assert(cs->multis == NULL); /* xxx */ + + if (nch(p, cs) == 1) { /* optimize singleton sets */ + ordinary(p, firstch(p, cs)); + freeset(p, cs); + } else + EMIT(OANYOF, freezeset(p, cs)); +} + +/* + - p_b_term - parse one term of a bracketed character list + == static void p_b_term(register struct parse *p, register cset *cs); + */ +static void +p_b_term(p, cs) +register struct parse *p; +register cset *cs; +{ + register char c; + register char start, finish; + register int i; + + /* classify what we've got */ + switch ((MORE()) ? PEEK() : '\0') { + case '[': + c = (MORE2()) ? PEEK2() : '\0'; + break; + case '-': + SETERROR(REG_ERANGE); + return; /* NOTE RETURN */ + break; + default: + c = '\0'; + break; + } + + switch (c) { + case ':': /* character class */ + NEXT2(); + REQUIRE(MORE(), REG_EBRACK); + c = PEEK(); + REQUIRE(c != '-' && c != ']', REG_ECTYPE); + p_b_cclass(p, cs); + REQUIRE(MORE(), REG_EBRACK); + REQUIRE(EATTWO(':', ']'), REG_ECTYPE); + break; + case '=': /* equivalence class */ + NEXT2(); + REQUIRE(MORE(), REG_EBRACK); + c = PEEK(); + REQUIRE(c != '-' && c != ']', REG_ECOLLATE); + p_b_eclass(p, cs); + REQUIRE(MORE(), REG_EBRACK); + REQUIRE(EATTWO('=', ']'), REG_ECOLLATE); + break; + default: /* symbol, ordinary character, or range */ +/* xxx revision needed for multichar stuff */ + start = p_b_symbol(p); + if (SEE('-') && MORE2() && PEEK2() != ']') { + /* range */ + NEXT(); + if (EAT('-')) + finish = '-'; + else + finish = p_b_symbol(p); + } else + finish = start; +/* xxx what about signed chars here... */ + REQUIRE(start <= finish, REG_ERANGE); + for (i = start; i <= finish; i++) + CHadd(cs, i); + break; + } +} + +/* + - p_b_cclass - parse a character-class name and deal with it + == static void p_b_cclass(register struct parse *p, register cset *cs); + */ +static void +p_b_cclass(p, cs) +register struct parse *p; +register cset *cs; +{ + register char *sp = p->next; + register struct cclass *cp; + register size_t len; + register char *u; + register char c; + + while (MORE() && isalpha(PEEK())) + NEXT(); + len = p->next - sp; + for (cp = cclasses; cp->name != NULL; cp++) + if (strncmp(cp->name, sp, len) == 0 && cp->name[len] == '\0') + break; + if (cp->name == NULL) { + /* oops, didn't find it */ + SETERROR(REG_ECTYPE); + return; + } + + u = cp->chars; + while ((c = *u++) != '\0') + CHadd(cs, c); + for (u = cp->multis; *u != '\0'; u += strlen(u) + 1) + MCadd(p, cs, u); +} + +/* + - p_b_eclass - parse an equivalence-class name and deal with it + == static void p_b_eclass(register struct parse *p, register cset *cs); + * + * This implementation is incomplete. xxx + */ +static void +p_b_eclass(p, cs) +register struct parse *p; +register cset *cs; +{ + register char c; + + c = p_b_coll_elem(p, '='); + CHadd(cs, c); +} + +/* + - p_b_symbol - parse a character or [..]ed multicharacter collating symbol + == static char p_b_symbol(register struct parse *p); + */ +static char /* value of symbol */ +p_b_symbol(p) +register struct parse *p; +{ + register char value; + + REQUIRE(MORE(), REG_EBRACK); + if (!EATTWO('[', '.')) + return(GETNEXT()); + + /* collating symbol */ + value = p_b_coll_elem(p, '.'); + REQUIRE(EATTWO('.', ']'), REG_ECOLLATE); + return(value); +} + +/* + - p_b_coll_elem - parse a collating-element name and look it up + == static char p_b_coll_elem(register struct parse *p, int endc); + */ +static char /* value of collating element */ +p_b_coll_elem(p, endc) +register struct parse *p; +int endc; /* name ended by endc,']' */ +{ + register char *sp = p->next; + register struct cname *cp; + register int len; + + while (MORE() && !SEETWO(endc, ']')) + NEXT(); + if (!MORE()) { + SETERROR(REG_EBRACK); + return(0); + } + len = p->next - sp; + for (cp = cnames; cp->name != NULL; cp++) + if (strncmp(cp->name, sp, len) == 0 && cp->name[len] == '\0') + return(cp->code); /* known name */ + if (len == 1) + return(*sp); /* single character */ + SETERROR(REG_ECOLLATE); /* neither */ + return(0); +} + +/* + - othercase - return the case counterpart of an alphabetic + == static char othercase(int ch); + */ +static char /* if no counterpart, return ch */ +othercase(ch) +int ch; +{ + assert(isalpha(ch)); + if (isupper(ch)) + return(tolower(ch)); + else if (islower(ch)) + return(toupper(ch)); + else /* peculiar, but could happen */ + return(ch); +} + +/* + - bothcases - emit a dualcase version of a two-case character + == static void bothcases(register struct parse *p, int ch); + * + * Boy, is this implementation ever a kludge... + */ +static void +bothcases(p, ch) +register struct parse *p; +int ch; +{ + register char *oldnext = p->next; + register char *oldend = p->end; + char bracket[3]; + + assert(othercase(ch) != ch); /* p_bracket() would recurse */ + p->next = bracket; + p->end = bracket+2; + bracket[0] = ch; + bracket[1] = ']'; + bracket[2] = '\0'; + p_bracket(p); + assert(p->next == bracket+2); + p->next = oldnext; + p->end = oldend; +} + +/* + - ordinary - emit an ordinary character + == static void ordinary(register struct parse *p, register int ch); + */ +static void +ordinary(p, ch) +register struct parse *p; +register int ch; +{ + register cat_t *cap = p->g->categories; + + if ((p->g->cflags®_ICASE) && isalpha(ch) && othercase(ch) != ch) + bothcases(p, ch); + else { + EMIT(OCHAR, (unsigned char)ch); + if (cap[ch] == 0) + cap[ch] = p->g->ncategories++; + } +} + +/* + - nonnewline - emit REG_NEWLINE version of OANY + == static void nonnewline(register struct parse *p); + * + * Boy, is this implementation ever a kludge... + */ +static void +nonnewline(p) +register struct parse *p; +{ + register char *oldnext = p->next; + register char *oldend = p->end; + char bracket[4]; + + p->next = bracket; + p->end = bracket+3; + bracket[0] = '^'; + bracket[1] = '\n'; + bracket[2] = ']'; + bracket[3] = '\0'; + p_bracket(p); + assert(p->next == bracket+3); + p->next = oldnext; + p->end = oldend; +} + +/* + - repeat - generate code for a bounded repetition, recursively if needed + == static void repeat(register struct parse *p, sopno start, int from, int to); + */ +static void +repeat(p, start, from, to) +register struct parse *p; +sopno start; /* operand from here to end of strip */ +int from; /* repeated from this number */ +int to; /* to this number of times (maybe INFINITY) */ +{ + register sopno finish = HERE(); +# define N 2 +# define INF 3 +# define REP(f, t) ((f)*8 + (t)) +# define MAP(n) (((n) <= 1) ? (n) : ((n) == INFINITY) ? INF : N) + register sopno copy; + + if (p->error != 0) /* head off possible runaway recursion */ + return; + + assert(from <= to); + + switch (REP(MAP(from), MAP(to))) { + case REP(0, 0): /* must be user doing this */ + DROP(finish-start); /* drop the operand */ + break; + case REP(0, 1): /* as x{1,1}? */ + case REP(0, N): /* as x{1,n}? */ + case REP(0, INF): /* as x{1,}? */ + /* KLUDGE: emit y? as (y|) until subtle bug gets fixed */ + INSERT(OCH_, start); /* offset is wrong... */ + repeat(p, start+1, 1, to); + ASTERN(OOR1, start); + AHEAD(start); /* ... fix it */ + EMIT(OOR2, 0); + AHEAD(THERE()); + ASTERN(O_CH, THERETHERE()); + break; + case REP(1, 1): /* trivial case */ + /* done */ + break; + case REP(1, N): /* as x?x{1,n-1} */ + /* KLUDGE: emit y? as (y|) until subtle bug gets fixed */ + INSERT(OCH_, start); + ASTERN(OOR1, start); + AHEAD(start); + EMIT(OOR2, 0); /* offset very wrong... */ + AHEAD(THERE()); /* ...so fix it */ + ASTERN(O_CH, THERETHERE()); + copy = dupl(p, start+1, finish+1); + assert(copy == finish+4); + repeat(p, copy, 1, to-1); + break; + case REP(1, INF): /* as x+ */ + INSERT(OPLUS_, start); + ASTERN(O_PLUS, start); + break; + case REP(N, N): /* as xx{m-1,n-1} */ + copy = dupl(p, start, finish); + repeat(p, copy, from-1, to-1); + break; + case REP(N, INF): /* as xx{n-1,INF} */ + copy = dupl(p, start, finish); + repeat(p, copy, from-1, to); + break; + default: /* "can't happen" */ + SETERROR(REG_ASSERT); /* just in case */ + break; + } +} + +/* + - seterr - set an error condition + == static int seterr(register struct parse *p, int e); + */ +static int /* useless but makes type checking happy */ +seterr(p, e) +register struct parse *p; +int e; +{ + if (p->error == 0) /* keep earliest error condition */ + p->error = e; + p->next = nuls; /* try to bring things to a halt */ + p->end = nuls; + return(0); /* make the return value well-defined */ +} + +/* + - allocset - allocate a set of characters for [] + == static cset *allocset(register struct parse *p); + */ +static cset * +allocset(p) +register struct parse *p; +{ + register int no = p->g->ncsets++; + register size_t nc; + register size_t nbytes; + register cset *cs; + register size_t css = (size_t)p->g->csetsize; + register int i; + + if (no >= p->ncsalloc) { /* need another column of space */ + p->ncsalloc += CHAR_BIT; + nc = p->ncsalloc; + assert(nc % CHAR_BIT == 0); + nbytes = nc / CHAR_BIT * css; + if (p->g->sets == NULL) + p->g->sets = (cset *)malloc(nc * sizeof(cset)); + else + p->g->sets = (cset *)realloc((char *)p->g->sets, + nc * sizeof(cset)); + if (p->g->setbits == NULL) + p->g->setbits = (uch *)malloc(nbytes); + else { + p->g->setbits = (uch *)realloc((char *)p->g->setbits, + nbytes); + /* xxx this isn't right if setbits is now NULL */ + for (i = 0; i < no; i++) + p->g->sets[i].ptr = p->g->setbits + css*(i/CHAR_BIT); + } + if (p->g->sets != NULL && p->g->setbits != NULL) + (void) memset((char *)p->g->setbits + (nbytes - css), + 0, css); + else { + no = 0; + SETERROR(REG_ESPACE); + /* caller's responsibility not to do set ops */ + } + } + + assert(p->g->sets != NULL); /* xxx */ + cs = &p->g->sets[no]; + cs->ptr = p->g->setbits + css*((no)/CHAR_BIT); + cs->mask = 1 << ((no) % CHAR_BIT); + cs->hash = 0; + cs->smultis = 0; + cs->multis = NULL; + + return(cs); +} + +/* + - freeset - free a now-unused set + == static void freeset(register struct parse *p, register cset *cs); + */ +static void +freeset(p, cs) +register struct parse *p; +register cset *cs; +{ + register size_t i; + register cset *top = &p->g->sets[p->g->ncsets]; + register size_t css = (size_t)p->g->csetsize; + + for (i = 0; i < css; i++) + CHsub(cs, i); + if (cs == top-1) /* recover only the easy case */ + p->g->ncsets--; +} + +/* + - freezeset - final processing on a set of characters + == static int freezeset(register struct parse *p, register cset *cs); + * + * The main task here is merging identical sets. This is usually a waste + * of time (although the hash code minimizes the overhead), but can win + * big if REG_ICASE is being used. REG_ICASE, by the way, is why the hash + * is done using addition rather than xor -- all ASCII [aA] sets xor to + * the same value! + */ +static int /* set number */ +freezeset(p, cs) +register struct parse *p; +register cset *cs; +{ + register uch h = cs->hash; + register size_t i; + register cset *top = &p->g->sets[p->g->ncsets]; + register cset *cs2; + register size_t css = (size_t)p->g->csetsize; + + /* look for an earlier one which is the same */ + for (cs2 = &p->g->sets[0]; cs2 < top; cs2++) + if (cs2->hash == h && cs2 != cs) { + /* maybe */ + for (i = 0; i < css; i++) + if (!!CHIN(cs2, i) != !!CHIN(cs, i)) + break; /* no */ + if (i == css) + break; /* yes */ + } + + if (cs2 < top) { /* found one */ + freeset(p, cs); + cs = cs2; + } + + return((int)(cs - p->g->sets)); +} + +/* + - firstch - return first character in a set (which must have at least one) + == static int firstch(register struct parse *p, register cset *cs); + */ +static int /* character; there is no "none" value */ +firstch(p, cs) +register struct parse *p; +register cset *cs; +{ + register size_t i; + register size_t css = (size_t)p->g->csetsize; + + for (i = 0; i < css; i++) + if (CHIN(cs, i)) + return((char)i); + assert(never); + return(0); /* arbitrary */ +} + +/* + - nch - number of characters in a set + == static int nch(register struct parse *p, register cset *cs); + */ +static int +nch(p, cs) +register struct parse *p; +register cset *cs; +{ + register size_t i; + register size_t css = (size_t)p->g->csetsize; + register int n = 0; + + for (i = 0; i < css; i++) + if (CHIN(cs, i)) + n++; + return(n); +} + +/* + - mcadd - add a collating element to a cset + == static void mcadd(register struct parse *p, register cset *cs, \ + == register char *cp); + */ +static void +mcadd(p, cs, cp) +register struct parse *p; +register cset *cs; +register char *cp; +{ + register size_t oldend = cs->smultis; + + cs->smultis += strlen(cp) + 1; + if (cs->multis == NULL) + cs->multis = malloc(cs->smultis); + else + cs->multis = realloc(cs->multis, cs->smultis); + if (cs->multis == NULL) { + SETERROR(REG_ESPACE); + return; + } + + (void) strcpy(cs->multis + oldend - 1, cp); + cs->multis[cs->smultis - 1] = '\0'; +} + +/* + - mcsub - subtract a collating element from a cset + == static void mcsub(register cset *cs, register char *cp); + */ +static void +mcsub(cs, cp) +register cset *cs; +register char *cp; +{ + register char *fp = mcfind(cs, cp); + register size_t len = strlen(fp); + + assert(fp != NULL); + (void) memmove(fp, fp + len + 1, + cs->smultis - (fp + len + 1 - cs->multis)); + cs->smultis -= len; + + if (cs->smultis == 0) { + free(cs->multis); + cs->multis = NULL; + return; + } + + cs->multis = realloc(cs->multis, cs->smultis); + assert(cs->multis != NULL); +} + +/* + - mcin - is a collating element in a cset? + == static int mcin(register cset *cs, register char *cp); + */ +static int +mcin(cs, cp) +register cset *cs; +register char *cp; +{ + return(mcfind(cs, cp) != NULL); +} + +/* + - mcfind - find a collating element in a cset + == static char *mcfind(register cset *cs, register char *cp); + */ +static char * +mcfind(cs, cp) +register cset *cs; +register char *cp; +{ + register char *p; + + if (cs->multis == NULL) + return(NULL); + for (p = cs->multis; *p != '\0'; p += strlen(p) + 1) + if (strcmp(cp, p) == 0) + return(p); + return(NULL); +} + +/* + - mcinvert - invert the list of collating elements in a cset + == static void mcinvert(register struct parse *p, register cset *cs); + * + * This would have to know the set of possibilities. Implementation + * is deferred. + */ +static void +mcinvert(p, cs) +register struct parse *p; +register cset *cs; +{ + assert(cs->multis == NULL); /* xxx */ +} + +/* + - mccase - add case counterparts of the list of collating elements in a cset + == static void mccase(register struct parse *p, register cset *cs); + * + * This would have to know the set of possibilities. Implementation + * is deferred. + */ +static void +mccase(p, cs) +register struct parse *p; +register cset *cs; +{ + assert(cs->multis == NULL); /* xxx */ +} + +/* + - isinsets - is this character in any sets? + == static int isinsets(register struct re_guts *g, int c); + */ +static int /* predicate */ +isinsets(g, c) +register struct re_guts *g; +int c; +{ + register uch *col; + register int i; + register int ncols = (g->ncsets+(CHAR_BIT-1)) / CHAR_BIT; + register unsigned uc = (unsigned char)c; + + for (i = 0, col = g->setbits; i < ncols; i++, col += g->csetsize) + if (col[uc] != 0) + return(1); + return(0); +} + +/* + - samesets - are these two characters in exactly the same sets? + == static int samesets(register struct re_guts *g, int c1, int c2); + */ +static int /* predicate */ +samesets(g, c1, c2) +register struct re_guts *g; +int c1; +int c2; +{ + register uch *col; + register int i; + register int ncols = (g->ncsets+(CHAR_BIT-1)) / CHAR_BIT; + register unsigned uc1 = (unsigned char)c1; + register unsigned uc2 = (unsigned char)c2; + + for (i = 0, col = g->setbits; i < ncols; i++, col += g->csetsize) + if (col[uc1] != col[uc2]) + return(0); + return(1); +} + +/* + - categorize - sort out character categories + == static void categorize(struct parse *p, register struct re_guts *g); + */ +static void +categorize(p, g) +struct parse *p; +register struct re_guts *g; +{ + register cat_t *cats = g->categories; + register int c; + register int c2; + register cat_t cat; + + /* avoid making error situations worse */ + if (p->error != 0) + return; + + for (c = CHAR_MIN; c <= CHAR_MAX; c++) + if (cats[c] == 0 && isinsets(g, c)) { + cat = g->ncategories++; + cats[c] = cat; + for (c2 = c+1; c2 <= CHAR_MAX; c2++) + if (cats[c2] == 0 && samesets(g, c, c2)) + cats[c2] = cat; + } +} + +/* + - dupl - emit a duplicate of a bunch of sops + == static sopno dupl(register struct parse *p, sopno start, sopno finish); + */ +static sopno /* start of duplicate */ +dupl(p, start, finish) +register struct parse *p; +sopno start; /* from here */ +sopno finish; /* to this less one */ +{ + register sopno ret = HERE(); + register sopno len = finish - start; + + assert(finish >= start); + if (len == 0) + return(ret); + enlarge(p, p->ssize + len); /* this many unexpected additions */ + assert(p->ssize >= p->slen + len); + (void) memcpy((char *)(p->strip + p->slen), + (char *)(p->strip + start), (size_t)len*sizeof(sop)); + p->slen += len; + return(ret); +} + +/* + - doemit - emit a strip operator + == static void doemit(register struct parse *p, sop op, size_t opnd); + * + * It might seem better to implement this as a macro with a function as + * hard-case backup, but it's just too big and messy unless there are + * some changes to the data structures. Maybe later. + */ +static void +doemit(p, op, opnd) +register struct parse *p; +sop op; +size_t opnd; +{ + /* avoid making error situations worse */ + if (p->error != 0) + return; + + /* deal with oversize operands ("can't happen", more or less) */ + assert(opnd < 1<slen >= p->ssize) + enlarge(p, (p->ssize+1) / 2 * 3); /* +50% */ + assert(p->slen < p->ssize); + + /* finally, it's all reduced to the easy case */ + p->strip[p->slen++] = SOP(op, opnd); +} + +/* + - doinsert - insert a sop into the strip + == static void doinsert(register struct parse *p, sop op, size_t opnd, sopno pos); + */ +static void +doinsert(p, op, opnd, pos) +register struct parse *p; +sop op; +size_t opnd; +sopno pos; +{ + register sopno sn; + register sop s; + register int i; + + /* avoid making error situations worse */ + if (p->error != 0) + return; + + sn = HERE(); + EMIT(op, opnd); /* do checks, ensure space */ + assert(HERE() == sn+1); + s = p->strip[sn]; + + /* adjust paren pointers */ + assert(pos > 0); + for (i = 1; i < NPAREN; i++) { + if (p->pbegin[i] >= pos) { + p->pbegin[i]++; + } + if (p->pend[i] >= pos) { + p->pend[i]++; + } + } + + memmove((char *)&p->strip[pos+1], (char *)&p->strip[pos], + (HERE()-pos-1)*sizeof(sop)); + p->strip[pos] = s; +} + +/* + - dofwd - complete a forward reference + == static void dofwd(register struct parse *p, sopno pos, sop value); + */ +static void +dofwd(p, pos, value) +register struct parse *p; +register sopno pos; +sop value; +{ + /* avoid making error situations worse */ + if (p->error != 0) + return; + + assert(value < 1<strip[pos] = OP(p->strip[pos]) | value; +} + +/* + - enlarge - enlarge the strip + == static void enlarge(register struct parse *p, sopno size); + */ +static void +enlarge(p, size) +register struct parse *p; +register sopno size; +{ + register sop *sp; + + if (p->ssize >= size) + return; + + sp = (sop *)realloc(p->strip, size*sizeof(sop)); + if (sp == NULL) { + SETERROR(REG_ESPACE); + return; + } + p->strip = sp; + p->ssize = size; +} + +/* + - stripsnug - compact the strip + == static void stripsnug(register struct parse *p, register struct re_guts *g); + */ +static void +stripsnug(p, g) +register struct parse *p; +register struct re_guts *g; +{ + g->nstates = p->slen; + g->strip = (sop *)realloc((char *)p->strip, p->slen * sizeof(sop)); + if (g->strip == NULL) { + SETERROR(REG_ESPACE); + g->strip = p->strip; + } +} + +/* + - findmust - fill in must and mlen with longest mandatory literal string + == static void findmust(register struct parse *p, register struct re_guts *g); + * + * This algorithm could do fancy things like analyzing the operands of | + * for common subsequences. Someday. This code is simple and finds most + * of the interesting cases. + * + * Note that must and mlen got initialized during setup. + */ +static void +findmust(p, g) +struct parse *p; +register struct re_guts *g; +{ + register sop *scan; + sop *start; + register sop *newstart; + register sopno newlen; + register sop s; + register char *cp; + register sopno i; + + /* avoid making error situations worse */ + if (p->error != 0) + return; + + /* find the longest OCHAR sequence in strip */ + newlen = 0; + scan = g->strip + 1; + do { + s = *scan++; + switch (OP(s)) { + case OCHAR: /* sequence member */ + if (newlen == 0) /* new sequence */ + newstart = scan - 1; + newlen++; + break; + case OPLUS_: /* things that don't break one */ + case OLPAREN: + case ORPAREN: + break; + case OQUEST_: /* things that must be skipped */ + case OCH_: + scan--; + do { + scan += OPND(s); + s = *scan; + /* assert() interferes w debug printouts */ + if (OP(s) != O_QUEST && OP(s) != O_CH && + OP(s) != OOR2) { + g->iflags |= BAD; + return; + } + } while (OP(s) != O_QUEST && OP(s) != O_CH); + /* fallthrough */ + default: /* things that break a sequence */ + if (newlen > g->mlen) { /* ends one */ + start = newstart; + g->mlen = newlen; + } + newlen = 0; + break; + } + } while (OP(s) != OEND); + + if (g->mlen == 0) /* there isn't one */ + return; + + /* turn it into a character string */ + g->must = malloc((size_t)g->mlen + 1); + if (g->must == NULL) { /* argh; just forget it */ + g->mlen = 0; + return; + } + cp = g->must; + scan = start; + for (i = g->mlen; i > 0; i--) { + while (OP(s = *scan++) != OCHAR) + continue; + assert(cp < g->must + g->mlen); + *cp++ = (char)OPND(s); + } + assert(cp == g->must + g->mlen); + *cp++ = '\0'; /* just on general principles */ +} + +/* + - pluscount - count + nesting + == static sopno pluscount(register struct parse *p, register struct re_guts *g); + */ +static sopno /* nesting depth */ +pluscount(p, g) +struct parse *p; +register struct re_guts *g; +{ + register sop *scan; + register sop s; + register sopno plusnest = 0; + register sopno maxnest = 0; + + if (p->error != 0) + return(0); /* there may not be an OEND */ + + scan = g->strip + 1; + do { + s = *scan++; + switch (OP(s)) { + case OPLUS_: + plusnest++; + break; + case O_PLUS: + if (plusnest > maxnest) + maxnest = plusnest; + plusnest--; + break; + } + } while (OP(s) != OEND); + if (plusnest != 0) + g->iflags |= BAD; + return(maxnest); +} diff --git a/vcnet/regex/regcomp.ih b/vcnet/regex/regcomp.ih new file mode 100644 index 0000000000..0776e7185c --- /dev/null +++ b/vcnet/regex/regcomp.ih @@ -0,0 +1,51 @@ +/* ========= begin header generated by ./mkh ========= */ +#ifdef __cplusplus +extern "C" { +#endif + +/* === regcomp.c === */ +static void p_ere(register struct parse *p, int stop); +static void p_ere_exp(register struct parse *p); +static void p_str(register struct parse *p); +static void p_bre(register struct parse *p, register int end1, register int end2); +static int p_simp_re(register struct parse *p, int starordinary); +static int p_count(register struct parse *p); +static void p_bracket(register struct parse *p); +static void p_b_term(register struct parse *p, register cset *cs); +static void p_b_cclass(register struct parse *p, register cset *cs); +static void p_b_eclass(register struct parse *p, register cset *cs); +static char p_b_symbol(register struct parse *p); +static char p_b_coll_elem(register struct parse *p, int endc); +static char othercase(int ch); +static void bothcases(register struct parse *p, int ch); +static void ordinary(register struct parse *p, register int ch); +static void nonnewline(register struct parse *p); +static void repeat(register struct parse *p, sopno start, int from, int to); +static int seterr(register struct parse *p, int e); +static cset *allocset(register struct parse *p); +static void freeset(register struct parse *p, register cset *cs); +static int freezeset(register struct parse *p, register cset *cs); +static int firstch(register struct parse *p, register cset *cs); +static int nch(register struct parse *p, register cset *cs); +static void mcadd(register struct parse *p, register cset *cs, register char *cp); +static void mcsub(register cset *cs, register char *cp); +static int mcin(register cset *cs, register char *cp); +static char *mcfind(register cset *cs, register char *cp); +static void mcinvert(register struct parse *p, register cset *cs); +static void mccase(register struct parse *p, register cset *cs); +static int isinsets(register struct re_guts *g, int c); +static int samesets(register struct re_guts *g, int c1, int c2); +static void categorize(struct parse *p, register struct re_guts *g); +static sopno dupl(register struct parse *p, sopno start, sopno finish); +static void doemit(register struct parse *p, sop op, size_t opnd); +static void doinsert(register struct parse *p, sop op, size_t opnd, sopno pos); +static void dofwd(register struct parse *p, sopno pos, sop value); +static void enlarge(register struct parse *p, sopno size); +static void stripsnug(register struct parse *p, register struct re_guts *g); +static void findmust(register struct parse *p, register struct re_guts *g); +static sopno pluscount(register struct parse *p, register struct re_guts *g); + +#ifdef __cplusplus +} +#endif +/* ========= end header generated by ./mkh ========= */ diff --git a/vcnet/regex/regerror.c b/vcnet/regex/regerror.c new file mode 100644 index 0000000000..9ddd25ca9b --- /dev/null +++ b/vcnet/regex/regerror.c @@ -0,0 +1,126 @@ +#include +#include +#include +#include +#include +#include +#include + +#include "utils.h" +#include "regerror.ih" + +/* + = #define REG_OKAY 0 + = #define REG_NOMATCH 1 + = #define REG_BADPAT 2 + = #define REG_ECOLLATE 3 + = #define REG_ECTYPE 4 + = #define REG_EESCAPE 5 + = #define REG_ESUBREG 6 + = #define REG_EBRACK 7 + = #define REG_EPAREN 8 + = #define REG_EBRACE 9 + = #define REG_BADBR 10 + = #define REG_ERANGE 11 + = #define REG_ESPACE 12 + = #define REG_BADRPT 13 + = #define REG_EMPTY 14 + = #define REG_ASSERT 15 + = #define REG_INVARG 16 + = #define REG_ATOI 255 // convert name to number (!) + = #define REG_ITOA 0400 // convert number to name (!) + */ +static struct rerr { + int code; + char *name; + char *explain; +} rerrs[] = { + REG_OKAY, "REG_OKAY", "no errors detected", + REG_NOMATCH, "REG_NOMATCH", "regexec() failed to match", + REG_BADPAT, "REG_BADPAT", "invalid regular expression", + REG_ECOLLATE, "REG_ECOLLATE", "invalid collating element", + REG_ECTYPE, "REG_ECTYPE", "invalid character class", + REG_EESCAPE, "REG_EESCAPE", "trailing backslash (\\)", + REG_ESUBREG, "REG_ESUBREG", "invalid backreference number", + REG_EBRACK, "REG_EBRACK", "brackets ([ ]) not balanced", + REG_EPAREN, "REG_EPAREN", "parentheses not balanced", + REG_EBRACE, "REG_EBRACE", "braces not balanced", + REG_BADBR, "REG_BADBR", "invalid repetition count(s)", + REG_ERANGE, "REG_ERANGE", "invalid character range", + REG_ESPACE, "REG_ESPACE", "out of memory", + REG_BADRPT, "REG_BADRPT", "repetition-operator operand invalid", + REG_EMPTY, "REG_EMPTY", "empty (sub)expression", + REG_ASSERT, "REG_ASSERT", "\"can't happen\" -- you found a bug", + REG_INVARG, "REG_INVARG", "invalid argument to regex routine", + -1, "", "*** unknown regexp error code ***", +}; + +/* + - regerror - the interface to error numbers + = extern size_t regerror(int, const regex_t *, char *, size_t); + */ +/* ARGSUSED */ +size_t +regerror( +int errcode, +const regex_t *preg, +char *errbuf, +size_t errbuf_size) +{ + register struct rerr *r; + register size_t len; + register int target = errcode &~ REG_ITOA; + register char *s; + char convbuf[50]; + + if (errcode == REG_ATOI) + s = regatoi(preg, convbuf); + else { + for (r = rerrs; r->code >= 0; r++) + if (r->code == target) + break; + + if (errcode®_ITOA) { + if (r->code >= 0) + (void) strcpy(convbuf, r->name); + else + sprintf(convbuf, "REG_0x%x", target); + assert(strlen(convbuf) < sizeof(convbuf)); + s = convbuf; + } else + s = r->explain; + } + + len = strlen(s) + 1; + if (errbuf_size > 0) { + if (errbuf_size > len) + (void) strcpy(errbuf, s); + else { + (void) strncpy(errbuf, s, errbuf_size-1); + errbuf[errbuf_size-1] = '\0'; + } + } + + return(len); +} + +/* + - regatoi - internal routine to implement REG_ATOI + == static char *regatoi(const regex_t *preg, char *localbuf); + */ +static char * +regatoi(preg, localbuf) +const regex_t *preg; +char *localbuf; +{ + register struct rerr *r; + + for (r = rerrs; r->code >= 0; r++) + if (strcmp(r->name, preg->re_endp) == 0) + break; + if (r->code < 0) + return("0"); + + sprintf(localbuf, "%d", r->code); + return(localbuf); +} diff --git a/vcnet/regex/regerror.ih b/vcnet/regex/regerror.ih new file mode 100644 index 0000000000..2cb668c24f --- /dev/null +++ b/vcnet/regex/regerror.ih @@ -0,0 +1,12 @@ +/* ========= begin header generated by ./mkh ========= */ +#ifdef __cplusplus +extern "C" { +#endif + +/* === regerror.c === */ +static char *regatoi(const regex_t *preg, char *localbuf); + +#ifdef __cplusplus +} +#endif +/* ========= end header generated by ./mkh ========= */ diff --git a/vcnet/regex/regex.3 b/vcnet/regex/regex.3 new file mode 100644 index 0000000000..bc747096d6 --- /dev/null +++ b/vcnet/regex/regex.3 @@ -0,0 +1,509 @@ +.TH REGEX 3 "25 Sept 1997" +.BY "Henry Spencer" +.de ZR +.\" one other place knows this name: the SEE ALSO section +.IR regex (7) \\$1 +.. +.SH NAME +regcomp, regexec, regerror, regfree \- regular-expression library +.SH SYNOPSIS +.ft B +.\".na +#include +.br +#include +.HP 10 +int regcomp(regex_t\ *preg, const\ char\ *pattern, int\ cflags); +.HP +int\ regexec(const\ regex_t\ *preg, const\ char\ *string, +size_t\ nmatch, regmatch_t\ pmatch[], int\ eflags); +.HP +size_t\ regerror(int\ errcode, const\ regex_t\ *preg, +char\ *errbuf, size_t\ errbuf_size); +.HP +void\ regfree(regex_t\ *preg); +.\".ad +.ft +.SH DESCRIPTION +These routines implement POSIX 1003.2 regular expressions (``RE''s); +see +.ZR . +.I Regcomp +compiles an RE written as a string into an internal form, +.I regexec +matches that internal form against a string and reports results, +.I regerror +transforms error codes from either into human-readable messages, +and +.I regfree +frees any dynamically-allocated storage used by the internal form +of an RE. +.PP +The header +.I +declares two structure types, +.I regex_t +and +.IR regmatch_t , +the former for compiled internal forms and the latter for match reporting. +It also declares the four functions, +a type +.IR regoff_t , +and a number of constants with names starting with ``REG_''. +.PP +.I Regcomp +compiles the regular expression contained in the +.I pattern +string, +subject to the flags in +.IR cflags , +and places the results in the +.I regex_t +structure pointed to by +.IR preg . +.I Cflags +is the bitwise OR of zero or more of the following flags: +.IP REG_EXTENDED \w'REG_EXTENDED'u+2n +Compile modern (``extended'') REs, +rather than the obsolete (``basic'') REs that +are the default. +.IP REG_BASIC +This is a synonym for 0, +provided as a counterpart to REG_EXTENDED to improve readability. +This is an extension, +compatible with but not specified by POSIX 1003.2, +and should be used with +caution in software intended to be portable to other systems. +.IP REG_NOSPEC +Compile with recognition of all special characters turned off. +All characters are thus considered ordinary, +so the ``RE'' is a literal string. +This is an extension, +compatible with but not specified by POSIX 1003.2, +and should be used with +caution in software intended to be portable to other systems. +REG_EXTENDED and REG_NOSPEC may not be used +in the same call to +.IR regcomp . +.IP REG_ICASE +Compile for matching that ignores upper/lower case distinctions. +See +.ZR . +.IP REG_NOSUB +Compile for matching that need only report success or failure, +not what was matched. +.IP REG_NEWLINE +Compile for newline-sensitive matching. +By default, newline is a completely ordinary character with no special +meaning in either REs or strings. +With this flag, +`[^' bracket expressions and `.' never match newline, +a `^' anchor matches the null string after any newline in the string +in addition to its normal function, +and the `$' anchor matches the null string before any newline in the +string in addition to its normal function. +.IP REG_PEND +The regular expression ends, +not at the first NUL, +but just before the character pointed to by the +.I re_endp +member of the structure pointed to by +.IR preg . +The +.I re_endp +member is of type +.IR const\ char\ * . +This flag permits inclusion of NULs in the RE; +they are considered ordinary characters. +This is an extension, +compatible with but not specified by POSIX 1003.2, +and should be used with +caution in software intended to be portable to other systems. +.PP +When successful, +.I regcomp +returns 0 and fills in the structure pointed to by +.IR preg . +One member of that structure +(other than +.IR re_endp ) +is publicized: +.IR re_nsub , +of type +.IR size_t , +contains the number of parenthesized subexpressions within the RE +(except that the value of this member is undefined if the +REG_NOSUB flag was used). +If +.I regcomp +fails, it returns a non-zero error code; +see DIAGNOSTICS. +.PP +.I Regexec +matches the compiled RE pointed to by +.I preg +against the +.IR string , +subject to the flags in +.IR eflags , +and reports results using +.IR nmatch , +.IR pmatch , +and the returned value. +The RE must have been compiled by a previous invocation of +.IR regcomp . +The compiled form is not altered during execution of +.IR regexec , +so a single compiled RE can be used simultaneously by multiple threads. +.PP +By default, +the NUL-terminated string pointed to by +.I string +is considered to be the text of an entire line, +with the NUL indicating the end of the line. +(That is, +any other end-of-line marker is considered to have been removed +and replaced by the NUL.) +The +.I eflags +argument is the bitwise OR of zero or more of the following flags: +.IP REG_NOTBOL \w'REG_STARTEND'u+2n +The first character of +the string +is not the beginning of a line, so the `^' anchor should not match before it. +This does not affect the behavior of newlines under REG_NEWLINE. +.IP REG_NOTEOL +The NUL terminating +the string +does not end a line, so the `$' anchor should not match before it. +This does not affect the behavior of newlines under REG_NEWLINE. +.IP REG_STARTEND +The string is considered to start at +\fIstring\fR\ + \fIpmatch\fR[0].\fIrm_so\fR +and to have a terminating NUL located at +\fIstring\fR\ + \fIpmatch\fR[0].\fIrm_eo\fR +(there need not actually be a NUL at that location), +regardless of the value of +.IR nmatch . +See below for the definition of +.IR pmatch +and +.IR nmatch . +This is an extension, +compatible with but not specified by POSIX 1003.2, +and should be used with +caution in software intended to be portable to other systems. +Note that a non-zero \fIrm_so\fR does not imply REG_NOTBOL; +REG_STARTEND affects only the location of the string, +not how it is matched. +.PP +See +.ZR +for a discussion of what is matched in situations where an RE or a +portion thereof could match any of several substrings of +.IR string . +.PP +Normally, +.I regexec +returns 0 for success and the non-zero code REG_NOMATCH for failure. +Other non-zero error codes may be returned in exceptional situations; +see DIAGNOSTICS. +.PP +If REG_NOSUB was specified in the compilation of the RE, +or if +.I nmatch +is 0, +.I regexec +ignores the +.I pmatch +argument (but see below for the case where REG_STARTEND is specified). +Otherwise, +.I pmatch +points to an array of +.I nmatch +structures of type +.IR regmatch_t . +Such a structure has at least the members +.I rm_so +and +.IR rm_eo , +both of type +.I regoff_t +(a signed arithmetic type at least as large as an +.I off_t +and a +.IR ssize_t ), +containing respectively the offset of the first character of a substring +and the offset of the first character after the end of the substring. +Offsets are measured from the beginning of the +.I string +argument given to +.IR regexec . +An empty substring is denoted by equal offsets, +both indicating the character following the empty substring. +.PP +The 0th member of the +.I pmatch +array is filled in to indicate what substring of +.I string +was matched by the entire RE. +Remaining members report what substring was matched by parenthesized +subexpressions within the RE; +member +.I i +reports subexpression +.IR i , +with subexpressions counted (starting at 1) by the order of their opening +parentheses in the RE, left to right. +Unused entries in the array\(emcorresponding either to subexpressions that +did not participate in the match at all, or to subexpressions that do not +exist in the RE (that is, \fIi\fR\ > \fIpreg\fR\->\fIre_nsub\fR)\(emhave both +.I rm_so +and +.I rm_eo +set to \-1. +If a subexpression participated in the match several times, +the reported substring is the last one it matched. +(Note, as an example in particular, that when the RE `(b*)+' matches `bbb', +the parenthesized subexpression matches the three `b's and then +an infinite number of empty strings following the last `b', +so the reported substring is one of the empties.) +.PP +If REG_STARTEND is specified, +.I pmatch +must point to at least one +.I regmatch_t +(even if +.I nmatch +is 0 or REG_NOSUB was specified), +to hold the input offsets for REG_STARTEND. +Use for output is still entirely controlled by +.IR nmatch ; +if +.I nmatch +is 0 or REG_NOSUB was specified, +the value of +.IR pmatch [0] +will not be changed by a successful +.IR regexec . +.PP +.I Regerror +maps a non-zero +.I errcode +from either +.I regcomp +or +.I regexec +to a human-readable, printable message. +If +.I preg +is non-NULL, +the error code should have arisen from use of +the +.I regex_t +pointed to by +.IR preg , +and if the error code came from +.IR regcomp , +it should have been the result from the most recent +.I regcomp +using that +.IR regex_t . +.RI ( Regerror +may be able to supply a more detailed message using information +from the +.IR regex_t .) +.I Regerror +places the NUL-terminated message into the buffer pointed to by +.IR errbuf , +limiting the length (including the NUL) to at most +.I errbuf_size +bytes. +If the whole message won't fit, +as much of it as will fit before the terminating NUL is supplied. +In any case, +the returned value is the size of buffer needed to hold the whole +message (including terminating NUL). +If +.I errbuf_size +is 0, +.I errbuf +is ignored but the return value is still correct. +.PP +If the +.I errcode +given to +.I regerror +is first ORed with REG_ITOA, +the ``message'' that results is the printable name of the error code, +e.g. ``REG_NOMATCH'', +rather than an explanation thereof. +If +.I errcode +is REG_ATOI, +then +.I preg +shall be non-NULL and the +.I re_endp +member of the structure it points to +must point to the printable name of an error code; +in this case, the result in +.I errbuf +is the decimal digits of +the numeric value of the error code +(0 if the name is not recognized). +REG_ITOA and REG_ATOI are intended primarily as debugging facilities; +they are extensions, +compatible with but not specified by POSIX 1003.2, +and should be used with +caution in software intended to be portable to other systems. +Be warned also that they are considered experimental and changes are possible. +.PP +.I Regfree +frees any dynamically-allocated storage associated with the compiled RE +pointed to by +.IR preg . +The remaining +.I regex_t +is no longer a valid compiled RE +and the effect of supplying it to +.I regexec +or +.I regerror +is undefined. +.PP +None of these functions references global variables except for tables +of constants; +all are safe for use from multiple threads if the arguments are safe. +.SH IMPLEMENTATION CHOICES +There are a number of decisions that 1003.2 leaves up to the implementor, +either by explicitly saying ``undefined'' or by virtue of them being +forbidden by the RE grammar. +This implementation treats them as follows. +.PP +See +.ZR +for a discussion of the definition of case-independent matching. +.PP +There is no particular limit on the length of REs, +except insofar as memory is limited. +Memory usage is approximately linear in RE size, and largely insensitive +to RE complexity, except for bounded repetitions. +See BUGS for one short RE using them +that will run almost any system out of memory. +.PP +A backslashed character other than one specifically given a magic meaning +by 1003.2 (such magic meanings occur only in obsolete [``basic''] REs) +is taken as an ordinary character. +.PP +Any unmatched [ is a REG_EBRACK error. +.PP +Equivalence classes cannot begin or end bracket-expression ranges. +The endpoint of one range cannot begin another. +.PP +RE_DUP_MAX, the limit on repetition counts in bounded repetitions, is 255. +.PP +A repetition operator (?, *, +, or bounds) cannot follow another +repetition operator. +A repetition operator cannot begin an expression or subexpression +or follow `^' or `|'. +.PP +`|' cannot appear first or last in a (sub)expression or after another `|', +i.e. an operand of `|' cannot be an empty subexpression. +An empty parenthesized subexpression, `()', is legal and matches an +empty (sub)string. +An empty string is not a legal RE. +.PP +A `{' followed by a digit is considered the beginning of bounds for a +bounded repetition, which must then follow the syntax for bounds. +A `{' \fInot\fR followed by a digit is considered an ordinary character. +.PP +`^' and `$' beginning and ending subexpressions in obsolete (``basic'') +REs are anchors, not ordinary characters. +.SH SEE ALSO +grep(1), regex(7) +.PP +POSIX 1003.2, sections 2.8 (Regular Expression Notation) +and +B.5 (C Binding for Regular Expression Matching). +.SH DIAGNOSTICS +Non-zero error codes from +.I regcomp +and +.I regexec +include the following: +.PP +.nf +.ta \w'REG_ECOLLATE'u+3n +REG_NOMATCH regexec() failed to match +REG_BADPAT invalid regular expression +REG_ECOLLATE invalid collating element +REG_ECTYPE invalid character class +REG_EESCAPE \e applied to unescapable character +REG_ESUBREG invalid backreference number +REG_EBRACK brackets [ ] not balanced +REG_EPAREN parentheses ( ) not balanced +REG_EBRACE braces { } not balanced +REG_BADBR invalid repetition count(s) in { } +REG_ERANGE invalid character range in [ ] +REG_ESPACE ran out of memory +REG_BADRPT ?, *, or + operand invalid +REG_EMPTY empty (sub)expression +REG_ASSERT ``can't happen''\(emyou found a bug +REG_INVARG invalid argument, e.g. negative-length string +.fi +.SH HISTORY +Written by Henry Spencer, +henry@zoo.toronto.edu. +.SH BUGS +This is an alpha release with known defects. +Please report problems. +.PP +There is one known functionality bug. +The implementation of internationalization is incomplete: +the locale is always assumed to be the default one of 1003.2, +and only the collating elements etc. of that locale are available. +.PP +The back-reference code is subtle and doubts linger about its correctness +in complex cases. +.PP +.I Regexec +performance is poor. +This will improve with later releases. +.I Nmatch +exceeding 0 is expensive; +.I nmatch +exceeding 1 is worse. +.I Regexec +is largely insensitive to RE complexity \fIexcept\fR that back +references are massively expensive. +RE length does matter; in particular, there is a strong speed bonus +for keeping RE length under about 30 characters, +with most special characters counting roughly double. +.PP +.I Regcomp +implements bounded repetitions by macro expansion, +which is costly in time and space if counts are large +or bounded repetitions are nested. +An RE like, say, +`((((a{1,100}){1,100}){1,100}){1,100}){1,100}' +will (eventually) run almost any existing machine out of swap space. +.PP +There are suspected problems with response to obscure error conditions. +Notably, +certain kinds of internal overflow, +produced only by truly enormous REs or by multiply nested bounded repetitions, +are probably not handled well. +.PP +Due to a mistake in 1003.2, things like `a)b' are legal REs because `)' is +a special character only in the presence of a previous unmatched `('. +This can't be fixed until the spec is fixed. +.PP +The standard's definition of back references is vague. +For example, does +`a\e(\e(b\e)*\e2\e)*d' match `abbbd'? +Until the standard is clarified, +behavior in such cases should not be relied on. +.PP +The implementation of word-boundary matching is a bit of a kludge, +and bugs may lurk in combinations of word-boundary matching and anchoring. diff --git a/vcnet/regex/regex.7 b/vcnet/regex/regex.7 new file mode 100644 index 0000000000..0fa180269e --- /dev/null +++ b/vcnet/regex/regex.7 @@ -0,0 +1,235 @@ +.TH REGEX 7 "25 Oct 1995" +.BY "Henry Spencer" +.SH NAME +regex \- POSIX 1003.2 regular expressions +.SH DESCRIPTION +Regular expressions (``RE''s), +as defined in POSIX 1003.2, come in two forms: +modern REs (roughly those of +.IR egrep ; +1003.2 calls these ``extended'' REs) +and obsolete REs (roughly those of +.IR ed ; +1003.2 ``basic'' REs). +Obsolete REs mostly exist for backward compatibility in some old programs; +they will be discussed at the end. +1003.2 leaves some aspects of RE syntax and semantics open; +`\(dg' marks decisions on these aspects that +may not be fully portable to other 1003.2 implementations. +.PP +A (modern) RE is one\(dg or more non-empty\(dg \fIbranches\fR, +separated by `|'. +It matches anything that matches one of the branches. +.PP +A branch is one\(dg or more \fIpieces\fR, concatenated. +It matches a match for the first, followed by a match for the second, etc. +.PP +A piece is an \fIatom\fR possibly followed +by a single\(dg `*', `+', `?', or \fIbound\fR. +An atom followed by `*' matches a sequence of 0 or more matches of the atom. +An atom followed by `+' matches a sequence of 1 or more matches of the atom. +An atom followed by `?' matches a sequence of 0 or 1 matches of the atom. +.PP +A \fIbound\fR is `{' followed by an unsigned decimal integer, +possibly followed by `,' +possibly followed by another unsigned decimal integer, +always followed by `}'. +The integers must lie between 0 and RE_DUP_MAX (255\(dg) inclusive, +and if there are two of them, the first may not exceed the second. +An atom followed by a bound containing one integer \fIi\fR +and no comma matches +a sequence of exactly \fIi\fR matches of the atom. +An atom followed by a bound +containing one integer \fIi\fR and a comma matches +a sequence of \fIi\fR or more matches of the atom. +An atom followed by a bound +containing two integers \fIi\fR and \fIj\fR matches +a sequence of \fIi\fR through \fIj\fR (inclusive) matches of the atom. +.PP +An atom is a regular expression enclosed in `()' (matching a match for the +regular expression), +an empty set of `()' (matching the null string)\(dg, +a \fIbracket expression\fR (see below), `.' +(matching any single character), `^' (matching the null string at the +beginning of a line), `$' (matching the null string at the +end of a line), a `\e' followed by one of the characters +`^.[$()|*+?{\e' +(matching that character taken as an ordinary character), +a `\e' followed by any other character\(dg +(matching that character taken as an ordinary character, +as if the `\e' had not been present\(dg), +or a single character with no other significance (matching that character). +A `{' followed by a character other than a digit is an ordinary +character, not the beginning of a bound\(dg. +It is illegal to end an RE with `\e'. +.PP +A \fIbracket expression\fR is a list of characters enclosed in `[]'. +It normally matches any single character from the list (but see below). +If the list begins with `^', +it matches any single character +(but see below) \fInot\fR from the rest of the list. +If two characters in the list are separated by `\-', this is shorthand +for the full \fIrange\fR of characters between those two (inclusive) in the +collating sequence, +e.g. `[0\-9]' in ASCII matches any decimal digit. +It is illegal\(dg for two ranges to share an +endpoint, e.g. `a\-c\-e'. +Ranges are very collating-sequence-dependent, +and portable programs should avoid relying on them. +.PP +To include a literal `]' in the list, make it the first character +(following a possible `^'). +To include a literal `\-', make it the first or last character, +or the second endpoint of a range. +To use a literal `\-' as the first endpoint of a range, +enclose it in `[.' and `.]' to make it a collating element (see below). +With the exception of these and some combinations using `[' (see next +paragraphs), all other special characters, including `\e', lose their +special significance within a bracket expression. +.PP +Within a bracket expression, a collating element (a character, +a multi-character sequence that collates as if it were a single character, +or a collating-sequence name for either) +enclosed in `[.' and `.]' stands for the +sequence of characters of that collating element. +The sequence is a single element of the bracket expression's list. +A bracket expression containing a multi-character collating element +can thus match more than one character, +e.g. if the collating sequence includes a `ch' collating element, +then the RE `[[.ch.]]*c' matches the first five characters +of `chchcc'. +.PP +Within a bracket expression, a collating element enclosed in `[=' and +`=]' is an equivalence class, standing for the sequences of characters +of all collating elements equivalent to that one, including itself. +(If there are no other equivalent collating elements, +the treatment is as if the enclosing delimiters were `[.' and `.]'.) +For example, if o and \o'o^' are the members of an equivalence class, +then `[[=o=]]', `[[=\o'o^'=]]', and `[o\o'o^']' are all synonymous. +An equivalence class may not\(dg be an endpoint +of a range. +.PP +Within a bracket expression, the name of a \fIcharacter class\fR enclosed +in `[:' and `:]' stands for the list of all characters belonging to that +class. +Standard character class names are: +.PP +.RS +.nf +.ta 3c 6c 9c +alnum digit punct +alpha graph space +blank lower upper +cntrl print xdigit +.fi +.RE +.PP +These stand for the character classes defined in +.IR ctype (3). +A locale may provide others. +A character class may not be used as an endpoint of a range. +.PP +There are two special cases\(dg of bracket expressions: +the bracket expressions `[[:<:]]' and `[[:>:]]' match the null string at +the beginning and end of a word respectively. +A word is defined as a sequence of +word characters +which is neither preceded nor followed by +word characters. +A word character is an +.I alnum +character (as defined by +.IR ctype (3)) +or an underscore. +This is an extension, +compatible with but not specified by POSIX 1003.2, +and should be used with +caution in software intended to be portable to other systems. +.PP +In the event that an RE could match more than one substring of a given +string, +the RE matches the one starting earliest in the string. +If the RE could match more than one substring starting at that point, +it matches the longest. +Subexpressions also match the longest possible substrings, subject to +the constraint that the whole match be as long as possible, +with subexpressions starting earlier in the RE taking priority over +ones starting later. +Note that higher-level subexpressions thus take priority over +their lower-level component subexpressions. +.PP +Match lengths are measured in characters, not collating elements. +A null string is considered longer than no match at all. +For example, +`bb*' matches the three middle characters of `abbbc', +`(wee|week)(knights|nights)' matches all ten characters of `weeknights', +when `(.*).*' is matched against `abc' the parenthesized subexpression +matches all three characters, and +when `(a*)*' is matched against `bc' both the whole RE and the parenthesized +subexpression match the null string. +.PP +If case-independent matching is specified, +the effect is much as if all case distinctions had vanished from the +alphabet. +When an alphabetic that exists in multiple cases appears as an +ordinary character outside a bracket expression, it is effectively +transformed into a bracket expression containing both cases, +e.g. `x' becomes `[xX]'. +When it appears inside a bracket expression, all case counterparts +of it are added to the bracket expression, so that (e.g.) `[x]' +becomes `[xX]' and `[^x]' becomes `[^xX]'. +.PP +No particular limit is imposed on the length of REs\(dg. +Programs intended to be portable should not employ REs longer +than 256 bytes, +as an implementation can refuse to accept such REs and remain +POSIX-compliant. +.PP +Obsolete (``basic'') regular expressions differ in several respects. +`|', `+', and `?' are ordinary characters and there is no equivalent +for their functionality. +The delimiters for bounds are `\e{' and `\e}', +with `{' and `}' by themselves ordinary characters. +The parentheses for nested subexpressions are `\e(' and `\e)', +with `(' and `)' by themselves ordinary characters. +`^' is an ordinary character except at the beginning of the +RE or\(dg the beginning of a parenthesized subexpression, +`$' is an ordinary character except at the end of the +RE or\(dg the end of a parenthesized subexpression, +and `*' is an ordinary character if it appears at the beginning of the +RE or the beginning of a parenthesized subexpression +(after a possible leading `^'). +Finally, there is one new type of atom, a \fIback reference\fR: +`\e' followed by a non-zero decimal digit \fId\fR +matches the same sequence of characters +matched by the \fId\fRth parenthesized subexpression +(numbering subexpressions by the positions of their opening parentheses, +left to right), +so that (e.g.) `\e([bc]\e)\e1' matches `bb' or `cc' but not `bc'. +.SH SEE ALSO +regex(3) +.PP +POSIX 1003.2, section 2.8 (Regular Expression Notation). +.SH HISTORY +Written by Henry Spencer, based on the 1003.2 spec. +.SH BUGS +Having two kinds of REs is a botch. +.PP +The current 1003.2 spec says that `)' is an ordinary character in +the absence of an unmatched `('; +this was an unintentional result of a wording error, +and change is likely. +Avoid relying on it. +.PP +Back references are a dreadful botch, +posing major problems for efficient implementations. +They are also somewhat vaguely defined +(does +`a\e(\e(b\e)*\e2\e)*d' match `abbbd'?). +Avoid using them. +.PP +1003.2's specification of case-independent matching is vague. +The ``one case implies all cases'' definition given above +is current consensus among implementors as to the right interpretation. +.PP +The syntax for word boundaries is incredibly ugly. diff --git a/vcnet/regex/regex.h b/vcnet/regex/regex.h new file mode 100644 index 0000000000..6918a55212 --- /dev/null +++ b/vcnet/regex/regex.h @@ -0,0 +1,74 @@ +#ifndef _REGEX_H_ +#define _REGEX_H_ /* never again */ +/* ========= begin header generated by ./mkh ========= */ +#ifdef __cplusplus +extern "C" { +#endif + +/* === regex2.h === */ +typedef long regoff_t; +typedef struct { + int re_magic; + size_t re_nsub; /* number of parenthesized subexpressions */ + const char *re_endp; /* end pointer for REG_PEND */ + struct re_guts *re_g; /* none of your business :-) */ +} regex_t; +typedef struct { + regoff_t rm_so; /* start of match */ + regoff_t rm_eo; /* end of match */ +} regmatch_t; + + +/* === regcomp.c === */ +extern int regcomp(regex_t *, const char *, int); +#define REG_BASIC 0000 +#define REG_EXTENDED 0001 +#define REG_ICASE 0002 +#define REG_NOSUB 0004 +#define REG_NEWLINE 0010 +#define REG_NOSPEC 0020 +#define REG_PEND 0040 +#define REG_DUMP 0200 + + +/* === regerror.c === */ +#define REG_OKAY 0 +#define REG_NOMATCH 1 +#define REG_BADPAT 2 +#define REG_ECOLLATE 3 +#define REG_ECTYPE 4 +#define REG_EESCAPE 5 +#define REG_ESUBREG 6 +#define REG_EBRACK 7 +#define REG_EPAREN 8 +#define REG_EBRACE 9 +#define REG_BADBR 10 +#define REG_ERANGE 11 +#define REG_ESPACE 12 +#define REG_BADRPT 13 +#define REG_EMPTY 14 +#define REG_ASSERT 15 +#define REG_INVARG 16 +#define REG_ATOI 255 /* convert name to number (!) */ +#define REG_ITOA 0400 /* convert number to name (!) */ +extern size_t regerror(int, const regex_t *, char *, size_t); + + +/* === regexec.c === */ +extern int regexec(const regex_t *, const char *, size_t, regmatch_t [], int); +#define REG_NOTBOL 00001 +#define REG_NOTEOL 00002 +#define REG_STARTEND 00004 +#define REG_TRACE 00400 /* tracing of execution */ +#define REG_LARGE 01000 /* force large representation */ +#define REG_BACKR 02000 /* force use of backref code */ + + +/* === regfree.c === */ +extern void regfree(regex_t *); + +#ifdef __cplusplus +} +#endif +/* ========= end header generated by ./mkh ========= */ +#endif diff --git a/vcnet/regex/regex2.h b/vcnet/regex/regex2.h new file mode 100644 index 0000000000..58fd8d8a43 --- /dev/null +++ b/vcnet/regex/regex2.h @@ -0,0 +1,134 @@ +/* + * First, the stuff that ends up in the outside-world include file + = typedef off_t regoff_t; + = typedef struct { + = int re_magic; + = size_t re_nsub; // number of parenthesized subexpressions + = const char *re_endp; // end pointer for REG_PEND + = struct re_guts *re_g; // none of your business :-) + = } regex_t; + = typedef struct { + = regoff_t rm_so; // start of match + = regoff_t rm_eo; // end of match + = } regmatch_t; + */ +/* + * internals of regex_t + */ +#define MAGIC1 ((('r'^0200)<<8) | 'e') + +/* + * The internal representation is a *strip*, a sequence of + * operators ending with an endmarker. (Some terminology etc. is a + * historical relic of earlier versions which used multiple strips.) + * Certain oddities in the representation are there to permit running + * the machinery backwards; in particular, any deviation from sequential + * flow must be marked at both its source and its destination. Some + * fine points: + * + * - OPLUS_ and O_PLUS are *inside* the loop they create. + * - OQUEST_ and O_QUEST are *outside* the bypass they create. + * - OCH_ and O_CH are *outside* the multi-way branch they create, while + * OOR1 and OOR2 are respectively the end and the beginning of one of + * the branches. Note that there is an implicit OOR2 following OCH_ + * and an implicit OOR1 preceding O_CH. + * + * In state representations, an operator's bit is on to signify a state + * immediately *preceding* "execution" of that operator. + */ +typedef long sop; /* strip operator */ +typedef long sopno; +#define OPRMASK 0x7c000000 +#define OPDMASK 0x03ffffff +#define OPSHIFT (26) +#define OP(n) ((n)&OPRMASK) +#define OPND(n) ((n)&OPDMASK) +#define SOP(op, opnd) ((op)|(opnd)) +/* operators meaning operand */ +/* (back, fwd are offsets) */ +#define OEND (1< uch [csetsize] */ + uch mask; /* bit within array */ + uch hash; /* hash code */ + size_t smultis; + char *multis; /* -> char[smulti] ab\0cd\0ef\0\0 */ +} cset; +/* note that CHadd and CHsub are unsafe, and CHIN doesn't yield 0/1 */ +#define CHadd(cs, c) ((cs)->ptr[(uch)(c)] |= (cs)->mask, (cs)->hash += (c)) +#define CHsub(cs, c) ((cs)->ptr[(uch)(c)] &= ~(cs)->mask, (cs)->hash -= (c)) +#define CHIN(cs, c) ((cs)->ptr[(uch)(c)] & (cs)->mask) +#define MCadd(p, cs, cp) mcadd(p, cs, cp) /* regcomp() internal fns */ +#define MCsub(p, cs, cp) mcsub(p, cs, cp) +#define MCin(p, cs, cp) mcin(p, cs, cp) + +/* stuff for character categories */ +typedef unsigned char cat_t; + +/* + * main compiled-expression structure + */ +struct re_guts { + int magic; +# define MAGIC2 ((('R'^0200)<<8)|'E') + sop *strip; /* malloced area for strip */ + int csetsize; /* number of bits in a cset vector */ + int ncsets; /* number of csets in use */ + cset *sets; /* -> cset [ncsets] */ + uch *setbits; /* -> uch[csetsize][ncsets/CHAR_BIT] */ + int cflags; /* copy of regcomp() cflags argument */ + sopno nstates; /* = number of sops */ + sopno firststate; /* the initial OEND (normally 0) */ + sopno laststate; /* the final OEND */ + int iflags; /* internal flags */ +# define USEBOL 01 /* used ^ */ +# define USEEOL 02 /* used $ */ +# define BAD 04 /* something wrong */ + int nbol; /* number of ^ used */ + int neol; /* number of $ used */ + int ncategories; /* how many character categories */ + cat_t *categories; /* ->catspace[-CHAR_MIN] */ + char *must; /* match must contain this string */ + int mlen; /* length of must */ + size_t nsub; /* copy of re_nsub */ + int backrefs; /* does it use back references? */ + sopno nplus; /* how deep does it nest +s? */ + /* catspace must be last */ + cat_t catspace[1]; /* actually [NC] */ +}; + +/* misc utilities */ +#define OUT (CHAR_MAX+1) /* a non-character value */ +#define ISWORD(c) (isalnum(c) || (c) == '_') diff --git a/vcnet/regex/regexec.c b/vcnet/regex/regexec.c new file mode 100644 index 0000000000..dcb11b285c --- /dev/null +++ b/vcnet/regex/regexec.c @@ -0,0 +1,138 @@ +/* + * the outer shell of regexec() + * + * This file includes engine.c *twice*, after muchos fiddling with the + * macros that code uses. This lets the same code operate on two different + * representations for state sets. + */ +#include +#include +#include +#include +#include +#include +#include + +#include "utils.h" +#include "regex2.h" + +static int nope = 0; /* for use in asserts; shuts lint up */ + +/* macros for manipulating states, small version */ +#define states unsigned +#define states1 unsigned /* for later use in regexec() decision */ +#define CLEAR(v) ((v) = 0) +#define SET0(v, n) ((v) &= ~((unsigned)1 << (n))) +#define SET1(v, n) ((v) |= (unsigned)1 << (n)) +#define ISSET(v, n) ((v) & ((unsigned)1 << (n))) +#define ASSIGN(d, s) ((d) = (s)) +#define EQ(a, b) ((a) == (b)) +#define STATEVARS int dummy /* dummy version */ +#define STATESETUP(m, n) /* nothing */ +#define STATETEARDOWN(m) /* nothing */ +#define SETUP(v) ((v) = 0) +#define onestate unsigned +#define INIT(o, n) ((o) = (unsigned)1 << (n)) +#define INC(o) ((o) <<= 1) +#define ISSTATEIN(v, o) ((v) & (o)) +/* some abbreviations; note that some of these know variable names! */ +/* do "if I'm here, I can also be there" etc without branches */ +#define FWD(dst, src, n) ((dst) |= ((unsigned)(src)&(here)) << (n)) +#define BACK(dst, src, n) ((dst) |= ((unsigned)(src)&(here)) >> (n)) +#define ISSETBACK(v, n) ((v) & ((unsigned)here >> (n))) +/* function names */ +#define SNAMES /* engine.c looks after details */ + +#include "engine.c" + +/* now undo things */ +#undef states +#undef CLEAR +#undef SET0 +#undef SET1 +#undef ISSET +#undef ASSIGN +#undef EQ +#undef STATEVARS +#undef STATESETUP +#undef STATETEARDOWN +#undef SETUP +#undef onestate +#undef INIT +#undef INC +#undef ISSTATEIN +#undef FWD +#undef BACK +#undef ISSETBACK +#undef SNAMES + +/* macros for manipulating states, large version */ +#define states char * +#define CLEAR(v) memset(v, 0, m->g->nstates) +#define SET0(v, n) ((v)[n] = 0) +#define SET1(v, n) ((v)[n] = 1) +#define ISSET(v, n) ((v)[n]) +#define ASSIGN(d, s) memcpy(d, s, m->g->nstates) +#define EQ(a, b) (memcmp(a, b, m->g->nstates) == 0) +#define STATEVARS int vn; char *space +#define STATESETUP(m, nv) { (m)->space = malloc((nv)*(m)->g->nstates); \ + if ((m)->space == NULL) return(REG_ESPACE); \ + (m)->vn = 0; } +#define STATETEARDOWN(m) { free((m)->space); } +#define SETUP(v) ((v) = &m->space[m->vn++ * m->g->nstates]) +#define onestate int +#define INIT(o, n) ((o) = (n)) +#define INC(o) ((o)++) +#define ISSTATEIN(v, o) ((v)[o]) +/* some abbreviations; note that some of these know variable names! */ +/* do "if I'm here, I can also be there" etc without branches */ +#define FWD(dst, src, n) ((dst)[here+(n)] |= (src)[here]) +#define BACK(dst, src, n) ((dst)[here-(n)] |= (src)[here]) +#define ISSETBACK(v, n) ((v)[here - (n)]) +/* function names */ +#define LNAMES /* flag */ + +#include "engine.c" + +/* + - regexec - interface for matching + = extern int regexec(const regex_t *, const char *, size_t, \ + = regmatch_t [], int); + = #define REG_NOTBOL 00001 + = #define REG_NOTEOL 00002 + = #define REG_STARTEND 00004 + = #define REG_TRACE 00400 // tracing of execution + = #define REG_LARGE 01000 // force large representation + = #define REG_BACKR 02000 // force use of backref code + * + * We put this here so we can exploit knowledge of the state representation + * when choosing which matcher to call. Also, by this point the matchers + * have been prototyped. + */ +int /* 0 success, REG_NOMATCH failure */ +regexec(preg, string, nmatch, pmatch, eflags) +const regex_t *preg; +const char *string; +size_t nmatch; +regmatch_t pmatch[]; +int eflags; +{ + register struct re_guts *g = preg->re_g; +#ifdef REDEBUG +# define GOODFLAGS(f) (f) +#else +# define GOODFLAGS(f) ((f)&(REG_NOTBOL|REG_NOTEOL|REG_STARTEND)) +#endif + + if (preg->re_magic != MAGIC1 || g->magic != MAGIC2) + return(REG_BADPAT); + assert(!(g->iflags&BAD)); + if (g->iflags&BAD) /* backstop for no-debug case */ + return(REG_BADPAT); + eflags = GOODFLAGS(eflags); + + if (g->nstates <= CHAR_BIT*sizeof(states1) && !(eflags®_LARGE)) + return(smatcher(g, (char *)string, nmatch, pmatch, eflags)); + else + return(lmatcher(g, (char *)string, nmatch, pmatch, eflags)); +} diff --git a/vcnet/regex/regfree.c b/vcnet/regex/regfree.c new file mode 100644 index 0000000000..9a6acf1733 --- /dev/null +++ b/vcnet/regex/regfree.c @@ -0,0 +1,37 @@ +#include +#include +#include +#include + +#include "utils.h" +#include "regex2.h" + +/* + - regfree - free everything + = extern void regfree(regex_t *); + */ +void +regfree(preg) +regex_t *preg; +{ + register struct re_guts *g; + + if (preg->re_magic != MAGIC1) /* oops */ + return; /* nice to complain, but hard */ + + g = preg->re_g; + if (g == NULL || g->magic != MAGIC2) /* oops again */ + return; + preg->re_magic = 0; /* mark it invalid */ + g->magic = 0; /* mark it invalid */ + + if (g->strip != NULL) + free((char *)g->strip); + if (g->sets != NULL) + free((char *)g->sets); + if (g->setbits != NULL) + free((char *)g->setbits); + if (g->must != NULL) + free(g->must); + free((char *)g); +} diff --git a/vcnet/regex/split.c b/vcnet/regex/split.c new file mode 100644 index 0000000000..188bdb775b --- /dev/null +++ b/vcnet/regex/split.c @@ -0,0 +1,316 @@ +#include +#include + +/* + - split - divide a string into fields, like awk split() + = int split(char *string, char *fields[], int nfields, char *sep); + */ +int /* number of fields, including overflow */ +split(string, fields, nfields, sep) +char *string; +char *fields[]; /* list is not NULL-terminated */ +int nfields; /* number of entries available in fields[] */ +char *sep; /* "" white, "c" single char, "ab" [ab]+ */ +{ + register char *p = string; + register char c; /* latest character */ + register char sepc = sep[0]; + register char sepc2; + register int fn; + register char **fp = fields; + register char *sepp; + register int trimtrail; + + /* white space */ + if (sepc == '\0') { + while ((c = *p++) == ' ' || c == '\t') + continue; + p--; + trimtrail = 1; + sep = " \t"; /* note, code below knows this is 2 long */ + sepc = ' '; + } else + trimtrail = 0; + sepc2 = sep[1]; /* now we can safely pick this up */ + + /* catch empties */ + if (*p == '\0') + return(0); + + /* single separator */ + if (sepc2 == '\0') { + fn = nfields; + for (;;) { + *fp++ = p; + fn--; + if (fn == 0) + break; + while ((c = *p++) != sepc) + if (c == '\0') + return(nfields - fn); + *(p-1) = '\0'; + } + /* we have overflowed the fields vector -- just count them */ + fn = nfields; + for (;;) { + while ((c = *p++) != sepc) + if (c == '\0') + return(fn); + fn++; + } + /* not reached */ + } + + /* two separators */ + if (sep[2] == '\0') { + fn = nfields; + for (;;) { + *fp++ = p; + fn--; + while ((c = *p++) != sepc && c != sepc2) + if (c == '\0') { + if (trimtrail && **(fp-1) == '\0') + fn++; + return(nfields - fn); + } + if (fn == 0) + break; + *(p-1) = '\0'; + while ((c = *p++) == sepc || c == sepc2) + continue; + p--; + } + /* we have overflowed the fields vector -- just count them */ + fn = nfields; + while (c != '\0') { + while ((c = *p++) == sepc || c == sepc2) + continue; + p--; + fn++; + while ((c = *p++) != '\0' && c != sepc && c != sepc2) + continue; + } + /* might have to trim trailing white space */ + if (trimtrail) { + p--; + while ((c = *--p) == sepc || c == sepc2) + continue; + p++; + if (*p != '\0') { + if (fn == nfields+1) + *p = '\0'; + fn--; + } + } + return(fn); + } + + /* n separators */ + fn = 0; + for (;;) { + if (fn < nfields) + *fp++ = p; + fn++; + for (;;) { + c = *p++; + if (c == '\0') + return(fn); + sepp = sep; + while ((sepc = *sepp++) != '\0' && sepc != c) + continue; + if (sepc != '\0') /* it was a separator */ + break; + } + if (fn < nfields) + *(p-1) = '\0'; + for (;;) { + c = *p++; + sepp = sep; + while ((sepc = *sepp++) != '\0' && sepc != c) + continue; + if (sepc == '\0') /* it wasn't a separator */ + break; + } + p--; + } + + /* not reached */ +} + +#ifdef TEST_SPLIT + + +/* + * test program + * pgm runs regression + * pgm sep splits stdin lines by sep + * pgm str sep splits str by sep + * pgm str sep n splits str by sep n times + */ +int +main(argc, argv) +int argc; +char *argv[]; +{ + char buf[512]; + register int n; +# define MNF 10 + char *fields[MNF]; + + if (argc > 4) + for (n = atoi(argv[3]); n > 0; n--) { + (void) strcpy(buf, argv[1]); + } + else if (argc > 3) + for (n = atoi(argv[3]); n > 0; n--) { + (void) strcpy(buf, argv[1]); + (void) split(buf, fields, MNF, argv[2]); + } + else if (argc > 2) + dosplit(argv[1], argv[2]); + else if (argc > 1) + while (fgets(buf, sizeof(buf), stdin) != NULL) { + buf[strlen(buf)-1] = '\0'; /* stomp newline */ + dosplit(buf, argv[1]); + } + else + regress(); + + exit(0); +} + +dosplit(string, seps) +char *string; +char *seps; +{ +# define NF 5 + char *fields[NF]; + register int nf; + + nf = split(string, fields, NF, seps); + print(nf, NF, fields); +} + +print(nf, nfp, fields) +int nf; +int nfp; +char *fields[]; +{ + register int fn; + register int bound; + + bound = (nf > nfp) ? nfp : nf; + printf("%d:\t", nf); + for (fn = 0; fn < bound; fn++) + printf("\"%s\"%s", fields[fn], (fn+1 < nf) ? ", " : "\n"); +} + +#define RNF 5 /* some table entries know this */ +struct { + char *str; + char *seps; + int nf; + char *fi[RNF]; +} tests[] = { + "", " ", 0, { "" }, + " ", " ", 2, { "", "" }, + "x", " ", 1, { "x" }, + "xy", " ", 1, { "xy" }, + "x y", " ", 2, { "x", "y" }, + "abc def g ", " ", 5, { "abc", "def", "", "g", "" }, + " a bcd", " ", 4, { "", "", "a", "bcd" }, + "a b c d e f", " ", 6, { "a", "b", "c", "d", "e f" }, + " a b c d ", " ", 6, { "", "a", "b", "c", "d " }, + + "", " _", 0, { "" }, + " ", " _", 2, { "", "" }, + "x", " _", 1, { "x" }, + "x y", " _", 2, { "x", "y" }, + "ab _ cd", " _", 2, { "ab", "cd" }, + " a_b c ", " _", 5, { "", "a", "b", "c", "" }, + "a b c_d e f", " _", 6, { "a", "b", "c", "d", "e f" }, + " a b c d ", " _", 6, { "", "a", "b", "c", "d " }, + + "", " _~", 0, { "" }, + " ", " _~", 2, { "", "" }, + "x", " _~", 1, { "x" }, + "x y", " _~", 2, { "x", "y" }, + "ab _~ cd", " _~", 2, { "ab", "cd" }, + " a_b c~", " _~", 5, { "", "a", "b", "c", "" }, + "a b_c d~e f", " _~", 6, { "a", "b", "c", "d", "e f" }, + "~a b c d ", " _~", 6, { "", "a", "b", "c", "d " }, + + "", " _~-", 0, { "" }, + " ", " _~-", 2, { "", "" }, + "x", " _~-", 1, { "x" }, + "x y", " _~-", 2, { "x", "y" }, + "ab _~- cd", " _~-", 2, { "ab", "cd" }, + " a_b c~", " _~-", 5, { "", "a", "b", "c", "" }, + "a b_c-d~e f", " _~-", 6, { "a", "b", "c", "d", "e f" }, + "~a-b c d ", " _~-", 6, { "", "a", "b", "c", "d " }, + + "", " ", 0, { "" }, + " ", " ", 2, { "", "" }, + "x", " ", 1, { "x" }, + "xy", " ", 1, { "xy" }, + "x y", " ", 2, { "x", "y" }, + "abc def g ", " ", 4, { "abc", "def", "g", "" }, + " a bcd", " ", 3, { "", "a", "bcd" }, + "a b c d e f", " ", 6, { "a", "b", "c", "d", "e f" }, + " a b c d ", " ", 6, { "", "a", "b", "c", "d " }, + + "", "", 0, { "" }, + " ", "", 0, { "" }, + "x", "", 1, { "x" }, + "xy", "", 1, { "xy" }, + "x y", "", 2, { "x", "y" }, + "abc def g ", "", 3, { "abc", "def", "g" }, + "\t a bcd", "", 2, { "a", "bcd" }, + " a \tb\t c ", "", 3, { "a", "b", "c" }, + "a b c d e ", "", 5, { "a", "b", "c", "d", "e" }, + "a b\tc d e f", "", 6, { "a", "b", "c", "d", "e f" }, + " a b c d e f ", "", 6, { "a", "b", "c", "d", "e f " }, + + NULL, NULL, 0, { NULL }, +}; + +regress() +{ + char buf[512]; + register int n; + char *fields[RNF+1]; + register int nf; + register int i; + register int printit; + register char *f; + + for (n = 0; tests[n].str != NULL; n++) { + (void) strcpy(buf, tests[n].str); + fields[RNF] = NULL; + nf = split(buf, fields, RNF, tests[n].seps); + printit = 0; + if (nf != tests[n].nf) { + printf("split `%s' by `%s' gave %d fields, not %d\n", + tests[n].str, tests[n].seps, nf, tests[n].nf); + printit = 1; + } else if (fields[RNF] != NULL) { + printf("split() went beyond array end\n"); + printit = 1; + } else { + for (i = 0; i < nf && i < RNF; i++) { + f = fields[i]; + if (f == NULL) + f = "(NULL)"; + if (strcmp(f, tests[n].fi[i]) != 0) { + printf("split `%s' by `%s', field %d is `%s', not `%s'\n", + tests[n].str, tests[n].seps, + i, fields[i], tests[n].fi[i]); + printit = 1; + } + } + } + if (printit) + print(nf, RNF, fields); + } +} +#endif diff --git a/vcnet/regex/tests b/vcnet/regex/tests new file mode 100644 index 0000000000..e4d928dad6 --- /dev/null +++ b/vcnet/regex/tests @@ -0,0 +1,477 @@ +# regular expression test set +# Lines are at least three fields, separated by one or more tabs. "" stands +# for an empty field. First field is an RE. Second field is flags. If +# C flag given, regcomp() is expected to fail, and the third field is the +# error name (minus the leading REG_). +# +# Otherwise it is expected to succeed, and the third field is the string to +# try matching it against. If there is no fourth field, the match is +# expected to fail. If there is a fourth field, it is the substring that +# the RE is expected to match. If there is a fifth field, it is a comma- +# separated list of what the subexpressions should match, with - indicating +# no match for that one. In both the fourth and fifth fields, a (sub)field +# starting with @ indicates that the (sub)expression is expected to match +# a null string followed by the stuff after the @; this provides a way to +# test where null strings match. The character `N' in REs and strings +# is newline, `S' is space, `T' is tab, `Z' is NUL. +# +# The full list of flags: +# - placeholder, does nothing +# b RE is a BRE, not an ERE +# & try it as both an ERE and a BRE +# C regcomp() error expected, third field is error name +# i REG_ICASE +# m ("mundane") REG_NOSPEC +# s REG_NOSUB (not really testable) +# n REG_NEWLINE +# ^ REG_NOTBOL +# $ REG_NOTEOL +# # REG_STARTEND (see below) +# p REG_PEND +# +# For REG_STARTEND, the start/end offsets are those of the substring +# enclosed in (). + +# basics +a & a a +abc & abc abc +abc|de - abc abc +a|b|c - abc a + +# parentheses and perversions thereof +a(b)c - abc abc +a\(b\)c b abc abc +a( C EPAREN +a( b a( a( +a\( - a( a( +a\( bC EPAREN +a\(b bC EPAREN +a(b C EPAREN +a(b b a(b a(b +# gag me with a right parenthesis -- 1003.2 goofed here (my fault, partly) +a) - a) a) +) - ) ) +# end gagging (in a just world, those *should* give EPAREN) +a) b a) a) +a\) bC EPAREN +\) bC EPAREN +a()b - ab ab +a\(\)b b ab ab + +# anchoring and REG_NEWLINE +^abc$ & abc abc +a^b - a^b +a^b b a^b a^b +a$b - a$b +a$b b a$b a$b +^ & abc @abc +$ & abc @ +^$ & "" @ +$^ - "" @ +\($\)\(^\) b "" @ +# stop retching, those are legitimate (although disgusting) +^^ - "" @ +$$ - "" @ +b$ & abNc +b$ &n abNc b +^b$ & aNbNc +^b$ &n aNbNc b +^$ &n aNNb @Nb +^$ n abc +^$ n abcN @ +$^ n aNNb @Nb +\($\)\(^\) bn aNNb @Nb +^^ n^ aNNb @Nb +$$ n aNNb @NN +^a ^ a +a$ $ a +^a ^n aNb +^b ^n aNb b +a$ $n bNa +b$ $n bNa b +a*(^b$)c* - b b +a*\(^b$\)c* b b b + +# certain syntax errors and non-errors +| C EMPTY +| b | | +* C BADRPT +* b * * ++ C BADRPT +? C BADRPT +"" &C EMPTY +() - abc @abc +\(\) b abc @abc +a||b C EMPTY +|ab C EMPTY +ab| C EMPTY +(|a)b C EMPTY +(a|)b C EMPTY +(*a) C BADRPT +(+a) C BADRPT +(?a) C BADRPT +({1}a) C BADRPT +\(\{1\}a\) bC BADRPT +(a|*b) C BADRPT +(a|+b) C BADRPT +(a|?b) C BADRPT +(a|{1}b) C BADRPT +^* C BADRPT +^* b * * +^+ C BADRPT +^? C BADRPT +^{1} C BADRPT +^\{1\} bC BADRPT + +# metacharacters, backslashes +a.c & abc abc +a[bc]d & abd abd +a\*c & a*c a*c +a\\b & a\b a\b +a\\\*b & a\*b a\*b +a\bc & abc abc +a\ &C EESCAPE +a\\bc & a\bc a\bc +\{ bC BADRPT +a\[b & a[b a[b +a[b &C EBRACK +# trailing $ is a peculiar special case for the BRE code +a$ & a a +a$ & a$ +a\$ & a +a\$ & a$ a$ +a\\$ & a +a\\$ & a$ +a\\$ & a\$ +a\\$ & a\ a\ + +# back references, ugh +a\(b\)\2c bC ESUBREG +a\(b\1\)c bC ESUBREG +a\(b*\)c\1d b abbcbbd abbcbbd bb +a\(b*\)c\1d b abbcbd +a\(b*\)c\1d b abbcbbbd +^\(.\)\1 b abc +a\([bc]\)\1d b abcdabbd abbd b +a\(\([bc]\)\2\)*d b abbccd abbccd +a\(\([bc]\)\2\)*d b abbcbd +# actually, this next one probably ought to fail, but the spec is unclear +a\(\(b\)*\2\)*d b abbbd abbbd +# here is a case that no NFA implementation does right +\(ab*\)[ab]*\1 b ababaaa ababaaa a +# check out normal matching in the presence of back refs +\(a\)\1bcd b aabcd aabcd +\(a\)\1bc*d b aabcd aabcd +\(a\)\1bc*d b aabd aabd +\(a\)\1bc*d b aabcccd aabcccd +\(a\)\1bc*[ce]d b aabcccd aabcccd +^\(a\)\1b\(c\)*cd$ b aabcccd aabcccd + +# ordinary repetitions +ab*c & abc abc +ab+c - abc abc +ab?c - abc abc +a\(*\)b b a*b a*b +a\(**\)b b ab ab +a\(***\)b bC BADRPT +*a b *a *a +**a b a a +***a bC BADRPT + +# the dreaded bounded repetitions +{ & { { +{abc & {abc {abc +{1 C BADRPT +{1} C BADRPT +a{b & a{b a{b +a{1}b - ab ab +a\{1\}b b ab ab +a{1,}b - ab ab +a\{1,\}b b ab ab +a{1,2}b - aab aab +a\{1,2\}b b aab aab +a{1 C EBRACE +a\{1 bC EBRACE +a{1a C EBRACE +a\{1a bC EBRACE +a{1a} C BADBR +a\{1a\} bC BADBR +a{,2} - a{,2} a{,2} +a\{,2\} bC BADBR +a{,} - a{,} a{,} +a\{,\} bC BADBR +a{1,x} C BADBR +a\{1,x\} bC BADBR +a{1,x C EBRACE +a\{1,x bC EBRACE +a{300} C BADBR +a\{300\} bC BADBR +a{1,0} C BADBR +a\{1,0\} bC BADBR +ab{0,0}c - abcac ac +ab\{0,0\}c b abcac ac +ab{0,1}c - abcac abc +ab\{0,1\}c b abcac abc +ab{0,3}c - abbcac abbc +ab\{0,3\}c b abbcac abbc +ab{1,1}c - acabc abc +ab\{1,1\}c b acabc abc +ab{1,3}c - acabc abc +ab\{1,3\}c b acabc abc +ab{2,2}c - abcabbc abbc +ab\{2,2\}c b abcabbc abbc +ab{2,4}c - abcabbc abbc +ab\{2,4\}c b abcabbc abbc +((a{1,10}){1,10}){1,10} - a a a,a + +# multiple repetitions +a** &C BADRPT +a++ C BADRPT +a?? C BADRPT +a*+ C BADRPT +a*? C BADRPT +a+* C BADRPT +a+? C BADRPT +a?* C BADRPT +a?+ C BADRPT +a{1}{1} C BADRPT +a*{1} C BADRPT +a+{1} C BADRPT +a?{1} C BADRPT +a{1}* C BADRPT +a{1}+ C BADRPT +a{1}? C BADRPT +a*{b} - a{b} a{b} +a\{1\}\{1\} bC BADRPT +a*\{1\} bC BADRPT +a\{1\}* bC BADRPT + +# brackets, and numerous perversions thereof +a[b]c & abc abc +a[ab]c & abc abc +a[^ab]c & adc adc +a[]b]c & a]c a]c +a[[b]c & a[c a[c +a[-b]c & a-c a-c +a[^]b]c & adc adc +a[^-b]c & adc adc +a[b-]c & a-c a-c +a[b &C EBRACK +a[] &C EBRACK +a[1-3]c & a2c a2c +a[3-1]c &C ERANGE +a[1-3-5]c &C ERANGE +a[[.-.]--]c & a-c a-c +a[1- &C ERANGE +a[[. &C EBRACK +a[[.x &C EBRACK +a[[.x. &C EBRACK +a[[.x.] &C EBRACK +a[[.x.]] & ax ax +a[[.x,.]] &C ECOLLATE +a[[.one.]]b & a1b a1b +a[[.notdef.]]b &C ECOLLATE +a[[.].]]b & a]b a]b +a[[:alpha:]]c & abc abc +a[[:notdef:]]c &C ECTYPE +a[[: &C EBRACK +a[[:alpha &C EBRACK +a[[:alpha:] &C EBRACK +a[[:alpha,:] &C ECTYPE +a[[:]:]]b &C ECTYPE +a[[:-:]]b &C ECTYPE +a[[:alph:]] &C ECTYPE +a[[:alphabet:]] &C ECTYPE +[[:alnum:]]+ - -%@a0X- a0X +[[:alpha:]]+ - -%@aX0- aX +[[:blank:]]+ - aSSTb SST +[[:cntrl:]]+ - aNTb NT +[[:digit:]]+ - a019b 019 +[[:graph:]]+ - Sa%bS a%b +[[:lower:]]+ - AabC ab +[[:print:]]+ - NaSbN aSb +[[:punct:]]+ - S%-&T %-& +[[:space:]]+ - aSNTb SNT +[[:upper:]]+ - aBCd BC +[[:xdigit:]]+ - p0f3Cq 0f3C +a[[=b=]]c & abc abc +a[[= &C EBRACK +a[[=b &C EBRACK +a[[=b= &C EBRACK +a[[=b=] &C EBRACK +a[[=b,=]] &C ECOLLATE +a[[=one=]]b & a1b a1b + +# complexities +a(((b)))c - abc abc +a(b|(c))d - abd abd +a(b*|c)d - abbd abbd +# just gotta have one DFA-buster, of course +a[ab]{20} - aaaaabaaaabaaaabaaaab aaaaabaaaabaaaabaaaab +# and an inline expansion in case somebody gets tricky +a[ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab] - aaaaabaaaabaaaabaaaab aaaaabaaaabaaaabaaaab +# and in case somebody just slips in an NFA... +a[ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab][ab](wee|week)(knights|night) - aaaaabaaaabaaaabaaaabweeknights aaaaabaaaabaaaabaaaabweeknights +# fish for anomalies as the number of states passes 32 +12345678901234567890123456789 - a12345678901234567890123456789b 12345678901234567890123456789 +123456789012345678901234567890 - a123456789012345678901234567890b 123456789012345678901234567890 +1234567890123456789012345678901 - a1234567890123456789012345678901b 1234567890123456789012345678901 +12345678901234567890123456789012 - a12345678901234567890123456789012b 12345678901234567890123456789012 +123456789012345678901234567890123 - a123456789012345678901234567890123b 123456789012345678901234567890123 +# and one really big one, beyond any plausible word width +1234567890123456789012345678901234567890123456789012345678901234567890 - a1234567890123456789012345678901234567890123456789012345678901234567890b 1234567890123456789012345678901234567890123456789012345678901234567890 +# fish for problems as brackets go past 8 +[ab][cd][ef][gh][ij][kl][mn] - xacegikmoq acegikm +[ab][cd][ef][gh][ij][kl][mn][op] - xacegikmoq acegikmo +[ab][cd][ef][gh][ij][kl][mn][op][qr] - xacegikmoqy acegikmoq +[ab][cd][ef][gh][ij][kl][mn][op][q] - xacegikmoqy acegikmoq + +# subtleties of matching +abc & xabcy abc +a\(b\)?c\1d b acd +aBc i Abc Abc +a[Bc]*d i abBCcd abBCcd +0[[:upper:]]1 &i 0a1 0a1 +0[[:lower:]]1 &i 0A1 0A1 +a[^b]c &i abc +a[^b]c &i aBc +a[^b]c &i adc adc +[a]b[c] - abc abc +[a]b[a] - aba aba +[abc]b[abc] - abc abc +[abc]b[abd] - abd abd +a(b?c)+d - accd accd +(wee|week)(knights|night) - weeknights weeknights +(we|wee|week|frob)(knights|night|day) - weeknights weeknights +a[bc]d - xyzaaabcaababdacd abd +a[ab]c - aaabc abc +abc s abc abc +a* & b @b + +# Let's have some fun -- try to match a C comment. +# first the obvious, which looks okay at first glance... +/\*.*\*/ - /*x*/ /*x*/ +# but... +/\*.*\*/ - /*x*/y/*z*/ /*x*/y/*z*/ +# okay, we must not match */ inside; try to do that... +/\*([^*]|\*[^/])*\*/ - /*x*/ /*x*/ +/\*([^*]|\*[^/])*\*/ - /*x*/y/*z*/ /*x*/ +# but... +/\*([^*]|\*[^/])*\*/ - /*x**/y/*z*/ /*x**/y/*z*/ +# and a still fancier version, which does it right (I think)... +/\*([^*]|\*+[^*/])*\*+/ - /*x*/ /*x*/ +/\*([^*]|\*+[^*/])*\*+/ - /*x*/y/*z*/ /*x*/ +/\*([^*]|\*+[^*/])*\*+/ - /*x**/y/*z*/ /*x**/ +/\*([^*]|\*+[^*/])*\*+/ - /*x****/y/*z*/ /*x****/ +/\*([^*]|\*+[^*/])*\*+/ - /*x**x*/y/*z*/ /*x**x*/ +/\*([^*]|\*+[^*/])*\*+/ - /*x***x/y/*z*/ /*x***x/y/*z*/ + +# subexpressions +.* - abc abc - +a(b)(c)d - abcd abcd b,c +a(((b)))c - abc abc b,b,b +a(b|(c))d - abd abd b,- +a(b*|c|e)d - abbd abbd bb +a(b*|c|e)d - acd acd c +a(b*|c|e)d - ad ad @d +a(b?)c - abc abc b +a(b?)c - ac ac @c +a(b+)c - abc abc b +a(b+)c - abbbc abbbc bbb +a(b*)c - ac ac @c +(a|ab)(bc([de]+)f|cde) - abcdef abcdef a,bcdef,de +# the regression tester only asks for 9 subexpressions +a(b)(c)(d)(e)(f)(g)(h)(i)(j)k - abcdefghijk abcdefghijk b,c,d,e,f,g,h,i,j +a(b)(c)(d)(e)(f)(g)(h)(i)(j)(k)l - abcdefghijkl abcdefghijkl b,c,d,e,f,g,h,i,j,k +a([bc]?)c - abc abc b +a([bc]?)c - ac ac @c +a([bc]+)c - abc abc b +a([bc]+)c - abcc abcc bc +a([bc]+)bc - abcbc abcbc bc +a(bb+|b)b - abb abb b +a(bbb+|bb+|b)b - abb abb b +a(bbb+|bb+|b)b - abbb abbb bb +a(bbb+|bb+|b)bb - abbb abbb b +(.*).* - abcdef abcdef abcdef +(a*)* - bc @b @b + +# do we get the right subexpression when it is used more than once? +a(b|c)*d - ad ad - +a(b|c)*d - abcd abcd c +a(b|c)+d - abd abd b +a(b|c)+d - abcd abcd c +a(b|c?)+d - ad ad @d +a(b|c?)+d - abcd abcd @d +a(b|c){0,0}d - ad ad - +a(b|c){0,1}d - ad ad - +a(b|c){0,1}d - abd abd b +a(b|c){0,2}d - ad ad - +a(b|c){0,2}d - abcd abcd c +a(b|c){0,}d - ad ad - +a(b|c){0,}d - abcd abcd c +a(b|c){1,1}d - abd abd b +a(b|c){1,1}d - acd acd c +a(b|c){1,2}d - abd abd b +a(b|c){1,2}d - abcd abcd c +a(b|c){1,}d - abd abd b +a(b|c){1,}d - abcd abcd c +a(b|c){2,2}d - acbd acbd b +a(b|c){2,2}d - abcd abcd c +a(b|c){2,4}d - abcd abcd c +a(b|c){2,4}d - abcbd abcbd b +a(b|c){2,4}d - abcbcd abcbcd c +a(b|c){2,}d - abcd abcd c +a(b|c){2,}d - abcbd abcbd b +a(b+|((c)*))+d - abd abd @d,@d,- +a(b+|((c)*))+d - abcd abcd @d,@d,- + +# check out the STARTEND option +[abc] &# a(b)c b +[abc] &# a(d)c +[abc] &# a(bc)d b +[abc] &# a(dc)d c +. &# a()c +b.*c &# b(bc)c bc +b.* &# b(bc)c bc +.*c &# b(bc)c bc + +# plain strings, with the NOSPEC flag +abc m abc abc +abc m xabcy abc +abc m xyz +a*b m aba*b a*b +a*b m ab +"" mC EMPTY + +# cases involving NULs +aZb & a a +aZb &p a +aZb &p# (aZb) aZb +aZ*b &p# (ab) ab +a.b &# (aZb) aZb +a.* &# (aZb)c aZb + +# word boundaries (ick) +[[:<:]]a & a a +[[:<:]]a & ba +[[:<:]]a & -a a +a[[:>:]] & a a +a[[:>:]] & ab +a[[:>:]] & a- a +[[:<:]]a.c[[:>:]] & axcd-dayc-dazce-abc abc +[[:<:]]a.c[[:>:]] & axcd-dayc-dazce-abc-q abc +[[:<:]]a.c[[:>:]] & axc-dayc-dazce-abc axc +[[:<:]]b.c[[:>:]] & a_bxc-byc_d-bzc-q bzc +[[:<:]].x..[[:>:]] & y_xa_-_xb_y-_xc_-axdc _xc_ +[[:<:]]a_b[[:>:]] & x_a_b + +# past problems, and suspected problems +(A[1])|(A[2])|(A[3])|(A[4])|(A[5])|(A[6])|(A[7])|(A[8])|(A[9])|(A[A]) - A1 A1 +abcdefghijklmnop i abcdefghijklmnop abcdefghijklmnop +abcdefghijklmnopqrstuv i abcdefghijklmnopqrstuv abcdefghijklmnopqrstuv +(ALAK)|(ALT[AB])|(CC[123]1)|(CM[123]1)|(GAMC)|(LC[23][EO ])|(SEM[1234])|(SL[ES][12])|(SLWW)|(SLF )|(SLDT)|(VWH[12])|(WH[34][EW])|(WP1[ESN]) - CC11 CC11 +CC[13]1|a{21}[23][EO][123][Es][12]a{15}aa[34][EW]aaaaaaa[X]a - CC11 CC11 +Char \([a-z0-9_]*\)\[.* b Char xyz[k Char xyz[k xyz +a?b - ab ab +-\{0,1\}[0-9]*$ b -5 -5 +a*a*a*a*a*a*a* & aaaaaa aaaaaa diff --git a/vcnet/regex/utils.h b/vcnet/regex/utils.h new file mode 100644 index 0000000000..1a997ac8fc --- /dev/null +++ b/vcnet/regex/utils.h @@ -0,0 +1,22 @@ +/* utility definitions */ +#ifdef _POSIX2_RE_DUP_MAX +#define DUPMAX _POSIX2_RE_DUP_MAX +#else +#define DUPMAX 255 +#endif +#define INFINITY (DUPMAX + 1) +#define NC (CHAR_MAX - CHAR_MIN + 1) +typedef unsigned char uch; + +/* switch off assertions (if not already off) if no REDEBUG */ +#ifndef REDEBUG +#ifndef NDEBUG +#define NDEBUG /* no assertions please */ +#endif +#endif +#include + +/* for old systems with bcopy() but no memmove() */ +#ifdef USEBCOPY +#define memmove(d, s, c) bcopy(s, d, c) +#endif diff --git a/vcnet/setdebug.bat b/vcnet/setdebug.bat new file mode 100644 index 0000000000..c80bbd1afc --- /dev/null +++ b/vcnet/setdebug.bat @@ -0,0 +1,5 @@ +rem Script to enable debug logging for IPPTOOL +set CUPS_DEBUG_LOG=debug.log +set CUPS_DEBUG_LEVEL=6 +set "CUPS_DEBUG_FILTER=^(http|_http|ipp|_ipp|cupsDo|cupsGetResponse|cupsSend)" + diff --git a/vcnet/testfile.vcproj b/vcnet/testfile.vcproj new file mode 100644 index 0000000000..417721930b --- /dev/null +++ b/vcnet/testfile.vcproj @@ -0,0 +1,201 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/vcnet/testhttp.vcproj b/vcnet/testhttp.vcproj new file mode 100755 index 0000000000..49df722618 --- /dev/null +++ b/vcnet/testhttp.vcproj @@ -0,0 +1,201 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/xcode/CUPS.xcodeproj/project.pbxproj b/xcode/CUPS.xcodeproj/project.pbxproj new file mode 100644 index 0000000000..66174ef801 --- /dev/null +++ b/xcode/CUPS.xcodeproj/project.pbxproj @@ -0,0 +1,4554 @@ +// !$*UTF8*$! +{ + archiveVersion = 1; + classes = { + }; + objectVersion = 46; + objects = { + +/* Begin PBXAggregateTarget section */ + 273BF6D91333B6260022CAAB /* Tests */ = { + isa = PBXAggregateTarget; + buildConfigurationList = 273BF6DA1333B6270022CAAB /* Build configuration list for PBXAggregateTarget "Tests" */; + buildPhases = ( + ); + dependencies = ( + 726AD704135E8AA1002C930D /* PBXTargetDependency */, + 273BF6DE1333B6370022CAAB /* PBXTargetDependency */, + 278C58D6136B641D00836530 /* PBXTargetDependency */, + 270CCDB2135E3CDE00007BE2 /* PBXTargetDependency */, + ); + name = Tests; + productName = Tests; + }; + 274FF5DE13332D3000317ECB /* All */ = { + isa = PBXAggregateTarget; + buildConfigurationList = 274FF5DF13332D3100317ECB /* Build configuration list for PBXAggregateTarget "All" */; + buildPhases = ( + ); + dependencies = ( + 274FF5E313332D4300317ECB /* PBXTargetDependency */, + 72F75A711336FACD004BB496 /* PBXTargetDependency */, + 274FF5E513332D4300317ECB /* PBXTargetDependency */, + 274FF622133331D300317ECB /* PBXTargetDependency */, + 276684131337FA8D000D33D0 /* PBXTargetDependency */, + 2766836B1337AA25000D33D0 /* PBXTargetDependency */, + 274FF5E713332D4300317ECB /* PBXTargetDependency */, + 274FF6E21333B33F00317ECB /* PBXTargetDependency */, + 72F75A731336FACD004BB496 /* PBXTargetDependency */, + 274FF6391333348400317ECB /* PBXTargetDependency */, + 274FF5E913332D4300317ECB /* PBXTargetDependency */, + 274FF648133335A300317ECB /* PBXTargetDependency */, + 274FF65E13333A3400317ECB /* PBXTargetDependency */, + 274FF67213333AE400317ECB /* PBXTargetDependency */, + 724379531333FECE009631B9 /* PBXTargetDependency */, + 724379111333E4EA009631B9 /* PBXTargetDependency */, + 276683FF1337F7C5000D33D0 /* PBXTargetDependency */, + 7243792B1333E962009631B9 /* PBXTargetDependency */, + 276683D71337B24A000D33D0 /* PBXTargetDependency */, + 276683D91337B24A000D33D0 /* PBXTargetDependency */, + 276683DB1337B24A000D33D0 /* PBXTargetDependency */, + 276683DD1337B24A000D33D0 /* PBXTargetDependency */, + 276683DF1337B24A000D33D0 /* PBXTargetDependency */, + 7258EAEF13459ADA009286F1 /* PBXTargetDependency */, + 720DD6D11358FDBE0064AA82 /* PBXTargetDependency */, + 7243793F1333FD23009631B9 /* PBXTargetDependency */, + 724379C31333FF7D009631B9 /* PBXTargetDependency */, + ); + name = All; + productName = All; + }; +/* End PBXAggregateTarget section */ + +/* Begin PBXBuildFile section */ + 270CCDB9135E3D0900007BE2 /* libcups_static.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 72F75A4C1336F31B004BB496 /* libcups_static.a */; }; + 270CCDBA135E3D0900007BE2 /* libcupsmime.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220FAC13330B2200FCA411 /* libcupsmime.dylib */; }; + 270CCDBC135E3D3E00007BE2 /* testmime.c in Sources */ = {isa = PBXBuildFile; fileRef = 270CCDBB135E3D3E00007BE2 /* testmime.c */; }; + 273BF6C71333B5370022CAAB /* testcups.c in Sources */ = {isa = PBXBuildFile; fileRef = 273BF6C61333B5370022CAAB /* testcups.c */; }; + 273BF6CE1333B5950022CAAB /* CoreFoundation.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 273BF6CB1333B5950022CAAB /* CoreFoundation.framework */; }; + 273BF6CF1333B5950022CAAB /* libiconv.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 273BF6CC1333B5950022CAAB /* libiconv.dylib */; }; + 273BF6D01333B5950022CAAB /* libz.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 273BF6CD1333B5950022CAAB /* libz.dylib */; }; + 273BF6D31333B5C30022CAAB /* Kerberos.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 273BF6D11333B5C30022CAAB /* Kerberos.framework */; }; + 273BF6D41333B5C30022CAAB /* Security.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 273BF6D21333B5C30022CAAB /* Security.framework */; }; + 273BF6D71333B5F60022CAAB /* libresolv.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 273BF6D51333B5F60022CAAB /* libresolv.dylib */; }; + 273BF6D81333B5F60022CAAB /* SystemConfiguration.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 273BF6D61333B5F60022CAAB /* SystemConfiguration.framework */; }; + 274FF5D913332CC700317ECB /* cups-driverd.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF5D613332CC700317ECB /* cups-driverd.cxx */; }; + 274FF5DA13332CC700317ECB /* util.c in Sources */ = {isa = PBXBuildFile; fileRef = 274FF5D713332CC700317ECB /* util.c */; }; + 274FF5DD13332D0600317ECB /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 274FF60A1333315100317ECB /* ppdc-array.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF5F51333315100317ECB /* ppdc-array.cxx */; }; + 274FF60B1333315100317ECB /* ppdc-attr.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF5F61333315100317ECB /* ppdc-attr.cxx */; }; + 274FF60C1333315100317ECB /* ppdc-catalog.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF5F71333315100317ECB /* ppdc-catalog.cxx */; }; + 274FF60D1333315100317ECB /* ppdc-choice.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF5F81333315100317ECB /* ppdc-choice.cxx */; }; + 274FF60E1333315100317ECB /* ppdc-constraint.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF5F91333315100317ECB /* ppdc-constraint.cxx */; }; + 274FF60F1333315100317ECB /* ppdc-driver.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF5FA1333315100317ECB /* ppdc-driver.cxx */; }; + 274FF6101333315100317ECB /* ppdc-file.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF5FB1333315100317ECB /* ppdc-file.cxx */; }; + 274FF6111333315100317ECB /* ppdc-filter.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF5FC1333315100317ECB /* ppdc-filter.cxx */; }; + 274FF6121333315100317ECB /* ppdc-font.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF5FD1333315100317ECB /* ppdc-font.cxx */; }; + 274FF6131333315100317ECB /* ppdc-group.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF5FE1333315100317ECB /* ppdc-group.cxx */; }; + 274FF6141333315100317ECB /* ppdc-import.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF5FF1333315100317ECB /* ppdc-import.cxx */; }; + 274FF6151333315100317ECB /* ppdc-mediasize.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF6001333315100317ECB /* ppdc-mediasize.cxx */; }; + 274FF6161333315100317ECB /* ppdc-message.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF6011333315100317ECB /* ppdc-message.cxx */; }; + 274FF6171333315100317ECB /* ppdc-option.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF6021333315100317ECB /* ppdc-option.cxx */; }; + 274FF6181333315100317ECB /* ppdc-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 274FF6031333315100317ECB /* ppdc-private.h */; settings = {ATTRIBUTES = (Private, ); }; }; + 274FF6191333315100317ECB /* ppdc-profile.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF6041333315100317ECB /* ppdc-profile.cxx */; }; + 274FF61A1333315100317ECB /* ppdc-shared.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF6051333315100317ECB /* ppdc-shared.cxx */; }; + 274FF61B1333315100317ECB /* ppdc-source.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF6061333315100317ECB /* ppdc-source.cxx */; }; + 274FF61C1333315100317ECB /* ppdc-string.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF6071333315100317ECB /* ppdc-string.cxx */; }; + 274FF61D1333315100317ECB /* ppdc-variable.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 274FF6081333315100317ECB /* ppdc-variable.cxx */; }; + 274FF61E1333315100317ECB /* ppdc.h in Headers */ = {isa = PBXBuildFile; fileRef = 274FF6091333315100317ECB /* ppdc.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 274FF6231333321400317ECB /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 274FF6241333323B00317ECB /* libcupsppdc.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 274FF5EE133330C800317ECB /* libcupsppdc.dylib */; }; + 274FF6321333334A00317ECB /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 274FF6361333344400317ECB /* cups-deviced.c in Sources */ = {isa = PBXBuildFile; fileRef = 274FF6351333344400317ECB /* cups-deviced.c */; }; + 274FF6371333345900317ECB /* util.c in Sources */ = {isa = PBXBuildFile; fileRef = 274FF5D713332CC700317ECB /* util.c */; }; + 274FF64A1333398D00317ECB /* cups-exec.c in Sources */ = {isa = PBXBuildFile; fileRef = 274FF6491333398D00317ECB /* cups-exec.c */; }; + 274FF658133339D300317ECB /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 274FF65C133339FC00317ECB /* cups-lpd.c in Sources */ = {isa = PBXBuildFile; fileRef = 274FF65B133339FC00317ECB /* cups-lpd.c */; }; + 274FF66E13333AB500317ECB /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 274FF67013333ACF00317ECB /* cups-polld.c in Sources */ = {isa = PBXBuildFile; fileRef = 274FF66F13333ACF00317ECB /* cups-polld.c */; }; + 274FF68513333B4300317ECB /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 274FF68613333B4300317ECB /* libcupsmime.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220FAC13330B2200FCA411 /* libcupsmime.dylib */; }; + 274FF68813333B6E00317ECB /* cupsfilter.c in Sources */ = {isa = PBXBuildFile; fileRef = 274FF68713333B6E00317ECB /* cupsfilter.c */; }; + 274FF68B1333B1C400317ECB /* adminutil.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EB51333052D00FCA411 /* adminutil.c */; }; + 274FF68C1333B1C400317ECB /* array.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EB81333056300FCA411 /* array.c */; }; + 274FF68D1333B1C400317ECB /* attr.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EBA1333056300FCA411 /* attr.c */; }; + 274FF68E1333B1C400317ECB /* auth.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EBB1333056300FCA411 /* auth.c */; }; + 274FF68F1333B1C400317ECB /* backchannel.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EBC1333056300FCA411 /* backchannel.c */; }; + 274FF6901333B1C400317ECB /* backend.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EBD1333056300FCA411 /* backend.c */; }; + 274FF6911333B1C400317ECB /* conflicts.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EBF1333056300FCA411 /* conflicts.c */; }; + 274FF6921333B1C400317ECB /* custom.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EC21333056300FCA411 /* custom.c */; }; + 274FF6931333B1C400317ECB /* debug.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220ED1133305BB00FCA411 /* debug.c */; }; + 274FF6941333B1C400317ECB /* dest.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220ED2133305BB00FCA411 /* dest.c */; }; + 274FF6951333B1C400317ECB /* dir.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220ED3133305BB00FCA411 /* dir.c */; }; + 274FF6961333B1C400317ECB /* emit.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220ED5133305BB00FCA411 /* emit.c */; }; + 274FF6971333B1C400317ECB /* encode.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220ED6133305BB00FCA411 /* encode.c */; }; + 274FF6981333B1C400317ECB /* file.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220ED8133305BB00FCA411 /* file.c */; }; + 274FF6991333B1C400317ECB /* getdevices.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EDA133305BB00FCA411 /* getdevices.c */; }; + 274FF69A1333B1C400317ECB /* getifaddrs.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EDB133305BB00FCA411 /* getifaddrs.c */; }; + 274FF69B1333B1C400317ECB /* getputfile.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EDC133305BB00FCA411 /* getputfile.c */; }; + 274FF69C1333B1C400317ECB /* globals.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EDD133305BB00FCA411 /* globals.c */; }; + 274FF69D1333B1C400317ECB /* http-addr.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EDE133305BB00FCA411 /* http-addr.c */; }; + 274FF69E1333B1C400317ECB /* http-addrlist.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EDF133305BB00FCA411 /* http-addrlist.c */; }; + 274FF69F1333B1C400317ECB /* http-support.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EE1133305BB00FCA411 /* http-support.c */; }; + 274FF6A01333B1C400317ECB /* http.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EE2133305BB00FCA411 /* http.c */; }; + 274FF6A11333B1C400317ECB /* ipp-support.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EE5133305BB00FCA411 /* ipp-support.c */; }; + 274FF6A21333B1C400317ECB /* ipp.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EE6133305BB00FCA411 /* ipp.c */; }; + 274FF6A31333B1C400317ECB /* langprintf.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EE8133305BB00FCA411 /* langprintf.c */; }; + 274FF6A41333B1C400317ECB /* language.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EEA133305BB00FCA411 /* language.c */; }; + 274FF6A51333B1C400317ECB /* localize.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EEC133305BB00FCA411 /* localize.c */; }; + 274FF6A61333B1C400317ECB /* mark.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EED133305BB00FCA411 /* mark.c */; }; + 274FF6A71333B1C400317ECB /* md5.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EEF133305BB00FCA411 /* md5.c */; }; + 274FF6A81333B1C400317ECB /* md5passwd.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EF0133305BB00FCA411 /* md5passwd.c */; }; + 274FF6A91333B1C400317ECB /* notify.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EF1133305BB00FCA411 /* notify.c */; }; + 274FF6AA1333B1C400317ECB /* options.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EF2133305BB00FCA411 /* options.c */; }; + 274FF6AB1333B1C400317ECB /* page.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EF3133305BB00FCA411 /* page.c */; }; + 274FF6AC1333B1C400317ECB /* ppd-cache.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EF4133305BB00FCA411 /* ppd-cache.c */; }; + 274FF6AD1333B1C400317ECB /* ppd.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EF6133305BB00FCA411 /* ppd.c */; }; + 274FF6AE1333B1C400317ECB /* pwg-media.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EF8133305BB00FCA411 /* pwg-media.c */; }; + 274FF6AF1333B1C400317ECB /* request.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EFB133305BB00FCA411 /* request.c */; }; + 274FF6B01333B1C400317ECB /* sidechannel.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EFC133305BB00FCA411 /* sidechannel.c */; }; + 274FF6B11333B1C400317ECB /* snmp.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EFF133305BB00FCA411 /* snmp.c */; }; + 274FF6B21333B1C400317ECB /* snprintf.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F00133305BB00FCA411 /* snprintf.c */; }; + 274FF6B31333B1C400317ECB /* string.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F02133305BB00FCA411 /* string.c */; }; + 274FF6B41333B1C400317ECB /* tempfile.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F03133305BB00FCA411 /* tempfile.c */; }; + 274FF6B51333B1C400317ECB /* thread.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F05133305BB00FCA411 /* thread.c */; }; + 274FF6B61333B1C400317ECB /* transcode.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F06133305BB00FCA411 /* transcode.c */; }; + 274FF6B71333B1C400317ECB /* usersys.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F08133305BB00FCA411 /* usersys.c */; }; + 274FF6B81333B1C400317ECB /* util.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F09133305BB00FCA411 /* util.c */; }; + 274FF6BA1333B1C400317ECB /* CoreFoundation.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F49133306BB00FCA411 /* CoreFoundation.framework */; }; + 274FF6BB1333B1C400317ECB /* Kerberos.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F55133308EA00FCA411 /* Kerberos.framework */; settings = {ATTRIBUTES = (Weak, ); }; }; + 274FF6BC1333B1C400317ECB /* Security.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F4B133306BB00FCA411 /* Security.framework */; }; + 274FF6BD1333B1C400317ECB /* SystemConfiguration.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F4C133306BB00FCA411 /* SystemConfiguration.framework */; }; + 274FF6BE1333B1C400317ECB /* libiconv.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F51133308C100FCA411 /* libiconv.dylib */; }; + 274FF6BF1333B1C400317ECB /* libresolv.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F53133308CB00FCA411 /* libresolv.dylib */; }; + 274FF6C01333B1C400317ECB /* libz.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F4A133306BB00FCA411 /* libz.dylib */; }; + 274FF6C21333B1C400317ECB /* adminutil.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EB71333056300FCA411 /* adminutil.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6C31333B1C400317ECB /* array.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EB91333056300FCA411 /* array.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6C41333B1C400317ECB /* backend.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EBE1333056300FCA411 /* backend.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6C51333B1C400317ECB /* cups.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EC11333056300FCA411 /* cups.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6C61333B1C400317ECB /* dir.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220ED4133305BB00FCA411 /* dir.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6C71333B1C400317ECB /* file.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220ED9133305BB00FCA411 /* file.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6C81333B1C400317ECB /* http.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EE3133305BB00FCA411 /* http.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6C91333B1C400317ECB /* ipp.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EE7133305BB00FCA411 /* ipp.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6CA1333B1C400317ECB /* language.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EEB133305BB00FCA411 /* language.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6CB1333B1C400317ECB /* ppd.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EF7133305BB00FCA411 /* ppd.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6CD1333B1C400317ECB /* sidechannel.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EFD133305BB00FCA411 /* sidechannel.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6CE1333B1C400317ECB /* transcode.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220F07133305BB00FCA411 /* transcode.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6CF1333B1C400317ECB /* versioning.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220F0A133305BB00FCA411 /* versioning.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6D01333B1C400317ECB /* cups-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EC01333056300FCA411 /* cups-private.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6D11333B1C400317ECB /* debug-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EC31333056300FCA411 /* debug-private.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6D21333B1C400317ECB /* file-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220ED7133305BB00FCA411 /* file-private.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6D31333B1C400317ECB /* http-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EE0133305BB00FCA411 /* http-private.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6D41333B1C400317ECB /* ipp-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EE4133305BB00FCA411 /* ipp-private.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6D51333B1C400317ECB /* language-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EE9133305BB00FCA411 /* language-private.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6D61333B1C400317ECB /* md5-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EEE133305BB00FCA411 /* md5-private.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6D71333B1C400317ECB /* ppd-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EF5133305BB00FCA411 /* ppd-private.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6D81333B1C400317ECB /* pwg-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EF9133305BB00FCA411 /* pwg-private.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6D91333B1C400317ECB /* snmp-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EFE133305BB00FCA411 /* snmp-private.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6DA1333B1C400317ECB /* string-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220F01133305BB00FCA411 /* string-private.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6DB1333B1C400317ECB /* thread-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220F04133305BB00FCA411 /* thread-private.h */; settings = {ATTRIBUTES = (); }; }; + 274FF6DC1333B1C400317ECB /* config.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220F471333063D00FCA411 /* config.h */; settings = {ATTRIBUTES = (); }; }; + 276683671337A9E0000D33D0 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 276683691337AA00000D33D0 /* cupsctl.c in Sources */ = {isa = PBXBuildFile; fileRef = 276683681337AA00000D33D0 /* cupsctl.c */; }; + 276683B11337AD06000D33D0 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 276683B21337AD06000D33D0 /* libcupsppdc.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 274FF5EE133330C800317ECB /* libcupsppdc.dylib */; }; + 276683B71337AD23000D33D0 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 276683B81337AD23000D33D0 /* libcupsppdc.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 274FF5EE133330C800317ECB /* libcupsppdc.dylib */; }; + 276683B91337AD31000D33D0 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 276683BA1337AD31000D33D0 /* libcupsppdc.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 274FF5EE133330C800317ECB /* libcupsppdc.dylib */; }; + 276683C31337B1B3000D33D0 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 276683C41337B1B3000D33D0 /* libcupsppdc.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 274FF5EE133330C800317ECB /* libcupsppdc.dylib */; }; + 276683C91337B1C1000D33D0 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 276683CA1337B1C1000D33D0 /* libcupsppdc.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 274FF5EE133330C800317ECB /* libcupsppdc.dylib */; }; + 276683CD1337B201000D33D0 /* ppdc.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 276683CC1337B201000D33D0 /* ppdc.cxx */; }; + 276683CF1337B20D000D33D0 /* ppdhtml.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 276683CE1337B20D000D33D0 /* ppdhtml.cxx */; }; + 276683D11337B21A000D33D0 /* ppdi.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 276683D01337B21A000D33D0 /* ppdi.cxx */; }; + 276683D31337B228000D33D0 /* ppdmerge.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 276683D21337B228000D33D0 /* ppdmerge.cxx */; }; + 276683D51337B237000D33D0 /* ppdpo.cxx in Sources */ = {isa = PBXBuildFile; fileRef = 276683D41337B237000D33D0 /* ppdpo.cxx */; }; + 276683E21337B29C000D33D0 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 276683E51337B2BE000D33D0 /* libcupsimage.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72F75A611336F9A3004BB496 /* libcupsimage.dylib */; }; + 276683FA1337F7A9000D33D0 /* ipptool.c in Sources */ = {isa = PBXBuildFile; fileRef = 276683F91337F7A9000D33D0 /* ipptool.c */; }; + 276683FD1337F7B8000D33D0 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 2766840F1337FA38000D33D0 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 276684111337FA7C000D33D0 /* cupsaddsmb.c in Sources */ = {isa = PBXBuildFile; fileRef = 276684101337FA7C000D33D0 /* cupsaddsmb.c */; }; + 278C58D9136B645C00836530 /* libcups_static.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 72F75A4C1336F31B004BB496 /* libcups_static.a */; }; + 278C58DE136B645C00836530 /* CoreFoundation.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 278C58DA136B645C00836530 /* CoreFoundation.framework */; }; + 278C58DF136B645C00836530 /* Kerberos.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 278C58DB136B645C00836530 /* Kerberos.framework */; }; + 278C58E1136B645C00836530 /* SystemConfiguration.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 278C58DD136B645C00836530 /* SystemConfiguration.framework */; }; + 278C58E3136B647200836530 /* testhttp.c in Sources */ = {isa = PBXBuildFile; fileRef = 278C58E2136B647200836530 /* testhttp.c */; }; + 278C58E4136B649200836530 /* libcups_static.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 72F75A4C1336F31B004BB496 /* libcups_static.a */; }; + 278C58E9136B64B000836530 /* CoreFoundation.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 278C58E5136B64AF00836530 /* CoreFoundation.framework */; }; + 278C58EA136B64B000836530 /* Kerberos.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 278C58E6136B64B000836530 /* Kerberos.framework */; }; + 278C58EB136B64B000836530 /* Security.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 278C58E7136B64B000836530 /* Security.framework */; }; + 278C58EC136B64B000836530 /* SystemConfiguration.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 278C58E8136B64B000836530 /* SystemConfiguration.framework */; }; + 278C58F6136B652300836530 /* Security.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 278C58F4136B652300836530 /* Security.framework */; }; + 720DD6CD1358FD720064AA82 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 720DD6D31358FDDE0064AA82 /* snmp.c in Sources */ = {isa = PBXBuildFile; fileRef = 720DD6D21358FDDE0064AA82 /* snmp.c */; }; + 720DD6D413590AB90064AA82 /* ieee1284.c in Sources */ = {isa = PBXBuildFile; fileRef = 724379CA1334000E009631B9 /* ieee1284.c */; }; + 72220EB61333052D00FCA411 /* adminutil.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EB51333052D00FCA411 /* adminutil.c */; }; + 72220EC41333056300FCA411 /* adminutil.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EB71333056300FCA411 /* adminutil.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 72220EC51333056300FCA411 /* array.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EB81333056300FCA411 /* array.c */; }; + 72220EC61333056300FCA411 /* array.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EB91333056300FCA411 /* array.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 72220EC71333056300FCA411 /* attr.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EBA1333056300FCA411 /* attr.c */; }; + 72220EC81333056300FCA411 /* auth.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EBB1333056300FCA411 /* auth.c */; }; + 72220EC91333056300FCA411 /* backchannel.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EBC1333056300FCA411 /* backchannel.c */; }; + 72220ECA1333056300FCA411 /* backend.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EBD1333056300FCA411 /* backend.c */; }; + 72220ECB1333056300FCA411 /* backend.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EBE1333056300FCA411 /* backend.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 72220ECC1333056300FCA411 /* conflicts.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EBF1333056300FCA411 /* conflicts.c */; }; + 72220ECD1333056300FCA411 /* cups-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EC01333056300FCA411 /* cups-private.h */; settings = {ATTRIBUTES = (Private, ); }; }; + 72220ECE1333056300FCA411 /* cups.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EC11333056300FCA411 /* cups.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 72220ECF1333056300FCA411 /* custom.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EC21333056300FCA411 /* custom.c */; }; + 72220ED01333056300FCA411 /* debug-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EC31333056300FCA411 /* debug-private.h */; settings = {ATTRIBUTES = (Private, ); }; }; + 72220F0B133305BB00FCA411 /* debug.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220ED1133305BB00FCA411 /* debug.c */; }; + 72220F0C133305BB00FCA411 /* dest.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220ED2133305BB00FCA411 /* dest.c */; }; + 72220F0D133305BB00FCA411 /* dir.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220ED3133305BB00FCA411 /* dir.c */; }; + 72220F0E133305BB00FCA411 /* dir.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220ED4133305BB00FCA411 /* dir.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 72220F0F133305BB00FCA411 /* emit.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220ED5133305BB00FCA411 /* emit.c */; }; + 72220F10133305BB00FCA411 /* encode.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220ED6133305BB00FCA411 /* encode.c */; }; + 72220F11133305BB00FCA411 /* file-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220ED7133305BB00FCA411 /* file-private.h */; settings = {ATTRIBUTES = (Private, ); }; }; + 72220F12133305BB00FCA411 /* file.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220ED8133305BB00FCA411 /* file.c */; }; + 72220F13133305BB00FCA411 /* file.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220ED9133305BB00FCA411 /* file.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 72220F14133305BB00FCA411 /* getdevices.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EDA133305BB00FCA411 /* getdevices.c */; }; + 72220F15133305BB00FCA411 /* getifaddrs.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EDB133305BB00FCA411 /* getifaddrs.c */; }; + 72220F16133305BB00FCA411 /* getputfile.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EDC133305BB00FCA411 /* getputfile.c */; }; + 72220F17133305BB00FCA411 /* globals.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EDD133305BB00FCA411 /* globals.c */; }; + 72220F18133305BB00FCA411 /* http-addr.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EDE133305BB00FCA411 /* http-addr.c */; }; + 72220F19133305BB00FCA411 /* http-addrlist.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EDF133305BB00FCA411 /* http-addrlist.c */; }; + 72220F1A133305BB00FCA411 /* http-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EE0133305BB00FCA411 /* http-private.h */; settings = {ATTRIBUTES = (Private, ); }; }; + 72220F1B133305BB00FCA411 /* http-support.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EE1133305BB00FCA411 /* http-support.c */; }; + 72220F1C133305BB00FCA411 /* http.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EE2133305BB00FCA411 /* http.c */; }; + 72220F1D133305BB00FCA411 /* http.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EE3133305BB00FCA411 /* http.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 72220F1E133305BB00FCA411 /* ipp-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EE4133305BB00FCA411 /* ipp-private.h */; settings = {ATTRIBUTES = (Private, ); }; }; + 72220F1F133305BB00FCA411 /* ipp-support.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EE5133305BB00FCA411 /* ipp-support.c */; }; + 72220F20133305BB00FCA411 /* ipp.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EE6133305BB00FCA411 /* ipp.c */; }; + 72220F21133305BB00FCA411 /* ipp.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EE7133305BB00FCA411 /* ipp.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 72220F22133305BB00FCA411 /* langprintf.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EE8133305BB00FCA411 /* langprintf.c */; }; + 72220F23133305BB00FCA411 /* language-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EE9133305BB00FCA411 /* language-private.h */; settings = {ATTRIBUTES = (Private, ); }; }; + 72220F24133305BB00FCA411 /* language.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EEA133305BB00FCA411 /* language.c */; }; + 72220F25133305BB00FCA411 /* language.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EEB133305BB00FCA411 /* language.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 72220F26133305BB00FCA411 /* localize.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EEC133305BB00FCA411 /* localize.c */; }; + 72220F27133305BB00FCA411 /* mark.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EED133305BB00FCA411 /* mark.c */; }; + 72220F28133305BB00FCA411 /* md5-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EEE133305BB00FCA411 /* md5-private.h */; settings = {ATTRIBUTES = (Private, ); }; }; + 72220F29133305BB00FCA411 /* md5.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EEF133305BB00FCA411 /* md5.c */; }; + 72220F2A133305BB00FCA411 /* md5passwd.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EF0133305BB00FCA411 /* md5passwd.c */; }; + 72220F2B133305BB00FCA411 /* notify.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EF1133305BB00FCA411 /* notify.c */; }; + 72220F2C133305BB00FCA411 /* options.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EF2133305BB00FCA411 /* options.c */; }; + 72220F2D133305BB00FCA411 /* page.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EF3133305BB00FCA411 /* page.c */; }; + 72220F2E133305BB00FCA411 /* ppd-cache.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EF4133305BB00FCA411 /* ppd-cache.c */; }; + 72220F2F133305BB00FCA411 /* ppd-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EF5133305BB00FCA411 /* ppd-private.h */; settings = {ATTRIBUTES = (Private, ); }; }; + 72220F30133305BB00FCA411 /* ppd.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EF6133305BB00FCA411 /* ppd.c */; }; + 72220F31133305BB00FCA411 /* ppd.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EF7133305BB00FCA411 /* ppd.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 72220F32133305BB00FCA411 /* pwg-media.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EF8133305BB00FCA411 /* pwg-media.c */; }; + 72220F33133305BB00FCA411 /* pwg-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EF9133305BB00FCA411 /* pwg-private.h */; settings = {ATTRIBUTES = (Private, ); }; }; + 72220F35133305BB00FCA411 /* request.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EFB133305BB00FCA411 /* request.c */; }; + 72220F36133305BB00FCA411 /* sidechannel.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EFC133305BB00FCA411 /* sidechannel.c */; }; + 72220F37133305BB00FCA411 /* sidechannel.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EFD133305BB00FCA411 /* sidechannel.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 72220F38133305BB00FCA411 /* snmp-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EFE133305BB00FCA411 /* snmp-private.h */; settings = {ATTRIBUTES = (Private, ); }; }; + 72220F39133305BB00FCA411 /* snmp.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220EFF133305BB00FCA411 /* snmp.c */; }; + 72220F3A133305BB00FCA411 /* snprintf.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F00133305BB00FCA411 /* snprintf.c */; }; + 72220F3B133305BB00FCA411 /* string-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220F01133305BB00FCA411 /* string-private.h */; settings = {ATTRIBUTES = (Private, ); }; }; + 72220F3C133305BB00FCA411 /* string.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F02133305BB00FCA411 /* string.c */; }; + 72220F3D133305BB00FCA411 /* tempfile.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F03133305BB00FCA411 /* tempfile.c */; }; + 72220F3E133305BB00FCA411 /* thread-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220F04133305BB00FCA411 /* thread-private.h */; settings = {ATTRIBUTES = (Private, ); }; }; + 72220F3F133305BB00FCA411 /* thread.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F05133305BB00FCA411 /* thread.c */; }; + 72220F40133305BB00FCA411 /* transcode.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F06133305BB00FCA411 /* transcode.c */; }; + 72220F41133305BB00FCA411 /* transcode.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220F07133305BB00FCA411 /* transcode.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 72220F42133305BB00FCA411 /* usersys.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F08133305BB00FCA411 /* usersys.c */; }; + 72220F43133305BB00FCA411 /* util.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F09133305BB00FCA411 /* util.c */; }; + 72220F44133305BB00FCA411 /* versioning.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220F0A133305BB00FCA411 /* versioning.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 72220F481333063D00FCA411 /* config.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220F471333063D00FCA411 /* config.h */; settings = {ATTRIBUTES = (Private, ); }; }; + 72220F4D133306BB00FCA411 /* CoreFoundation.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F49133306BB00FCA411 /* CoreFoundation.framework */; }; + 72220F4E133306BB00FCA411 /* libz.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F4A133306BB00FCA411 /* libz.dylib */; }; + 72220F4F133306BB00FCA411 /* Security.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F4B133306BB00FCA411 /* Security.framework */; }; + 72220F50133306BB00FCA411 /* SystemConfiguration.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F4C133306BB00FCA411 /* SystemConfiguration.framework */; }; + 72220F52133308C100FCA411 /* libiconv.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F51133308C100FCA411 /* libiconv.dylib */; }; + 72220F54133308CB00FCA411 /* libresolv.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F53133308CB00FCA411 /* libresolv.dylib */; }; + 72220F56133308EA00FCA411 /* Kerberos.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F55133308EA00FCA411 /* Kerberos.framework */; settings = {ATTRIBUTES = (Weak, ); }; }; + 72220F6613330A7000FCA411 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 72220F6813330A8500FCA411 /* ApplicationServices.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F6713330A8500FCA411 /* ApplicationServices.framework */; settings = {ATTRIBUTES = (Weak, ); }; }; + 72220F9013330B0C00FCA411 /* auth.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F6913330B0C00FCA411 /* auth.c */; }; + 72220F9113330B0C00FCA411 /* banners.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F6B13330B0C00FCA411 /* banners.c */; }; + 72220F9213330B0C00FCA411 /* cert.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F6D13330B0C00FCA411 /* cert.c */; }; + 72220F9313330B0C00FCA411 /* classes.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F6F13330B0C00FCA411 /* classes.c */; }; + 72220F9413330B0C00FCA411 /* client.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F7113330B0C00FCA411 /* client.c */; }; + 72220F9513330B0C00FCA411 /* conf.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F7313330B0C00FCA411 /* conf.c */; }; + 72220F9613330B0C00FCA411 /* dirsvc.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F7613330B0C00FCA411 /* dirsvc.c */; }; + 72220F9713330B0C00FCA411 /* env.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F7813330B0C00FCA411 /* env.c */; }; + 72220F9813330B0C00FCA411 /* ipp.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F7913330B0C00FCA411 /* ipp.c */; }; + 72220F9913330B0C00FCA411 /* job.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F7A13330B0C00FCA411 /* job.c */; }; + 72220F9A13330B0C00FCA411 /* listen.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F7C13330B0C00FCA411 /* listen.c */; }; + 72220F9B13330B0C00FCA411 /* log.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F7D13330B0C00FCA411 /* log.c */; }; + 72220F9C13330B0C00FCA411 /* main.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F7E13330B0C00FCA411 /* main.c */; }; + 72220F9D13330B0C00FCA411 /* network.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F7F13330B0C00FCA411 /* network.c */; }; + 72220F9E13330B0C00FCA411 /* policy.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F8113330B0C00FCA411 /* policy.c */; }; + 72220F9F13330B0C00FCA411 /* printers.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F8313330B0C00FCA411 /* printers.c */; }; + 72220FA013330B0C00FCA411 /* process.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F8513330B0C00FCA411 /* process.c */; }; + 72220FA113330B0C00FCA411 /* quotas.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F8613330B0C00FCA411 /* quotas.c */; }; + 72220FA313330B0C00FCA411 /* select.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F8813330B0C00FCA411 /* select.c */; }; + 72220FA413330B0C00FCA411 /* server.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F8913330B0C00FCA411 /* server.c */; }; + 72220FA513330B0C00FCA411 /* statbuf.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F8A13330B0C00FCA411 /* statbuf.c */; }; + 72220FA613330B0C00FCA411 /* subscriptions.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F8C13330B0C00FCA411 /* subscriptions.c */; }; + 72220FA713330B0C00FCA411 /* sysman.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220F8E13330B0C00FCA411 /* sysman.c */; }; + 72220FB613330BCE00FCA411 /* filter.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220FB213330BCE00FCA411 /* filter.c */; }; + 72220FB713330BCE00FCA411 /* mime.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220FB313330BCE00FCA411 /* mime.c */; }; + 72220FB813330BCE00FCA411 /* mime.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220FB413330BCE00FCA411 /* mime.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 72220FB913330BCE00FCA411 /* type.c in Sources */ = {isa = PBXBuildFile; fileRef = 72220FB513330BCE00FCA411 /* type.c */; }; + 72220FBA13330BEE00FCA411 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 72220FBF13330C1000FCA411 /* libcupsmime.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220FAC13330B2200FCA411 /* libcupsmime.dylib */; }; + 7234F4201378A16F00D3E9C9 /* array-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 7234F41F1378A16F00D3E9C9 /* array-private.h */; }; + 724379081333E4A5009631B9 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 7243790D1333E4E3009631B9 /* ipp.c in Sources */ = {isa = PBXBuildFile; fileRef = 7243790A1333E4E3009631B9 /* ipp.c */; }; + 7243790E1333E4E3009631B9 /* network.c in Sources */ = {isa = PBXBuildFile; fileRef = 7243790B1333E4E3009631B9 /* network.c */; }; + 7243790F1333E4E3009631B9 /* snmp-supplies.c in Sources */ = {isa = PBXBuildFile; fileRef = 7243790C1333E4E3009631B9 /* snmp-supplies.c */; }; + 724379131333E516009631B9 /* runloop.c in Sources */ = {isa = PBXBuildFile; fileRef = 724379121333E516009631B9 /* runloop.c */; }; + 724379221333E928009631B9 /* network.c in Sources */ = {isa = PBXBuildFile; fileRef = 7243790B1333E4E3009631B9 /* network.c */; }; + 724379231333E928009631B9 /* runloop.c in Sources */ = {isa = PBXBuildFile; fileRef = 724379121333E516009631B9 /* runloop.c */; }; + 724379241333E928009631B9 /* snmp-supplies.c in Sources */ = {isa = PBXBuildFile; fileRef = 7243790C1333E4E3009631B9 /* snmp-supplies.c */; }; + 724379271333E93D009631B9 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 724379291333E952009631B9 /* lpd.c in Sources */ = {isa = PBXBuildFile; fileRef = 724379281333E952009631B9 /* lpd.c */; }; + 7243793B1333FB9D009631B9 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 7243793D1333FD19009631B9 /* socket.c in Sources */ = {isa = PBXBuildFile; fileRef = 7243793C1333FD19009631B9 /* socket.c */; }; + 724379401333FD4B009631B9 /* network.c in Sources */ = {isa = PBXBuildFile; fileRef = 7243790B1333E4E3009631B9 /* network.c */; }; + 724379411333FD4B009631B9 /* runloop.c in Sources */ = {isa = PBXBuildFile; fileRef = 724379121333E516009631B9 /* runloop.c */; }; + 724379421333FD4B009631B9 /* snmp-supplies.c in Sources */ = {isa = PBXBuildFile; fileRef = 7243790C1333E4E3009631B9 /* snmp-supplies.c */; }; + 724379511333FEBB009631B9 /* dnssd.c in Sources */ = {isa = PBXBuildFile; fileRef = 724379501333FEBB009631B9 /* dnssd.c */; }; + 724379561333FF04009631B9 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 724379661333FF3B009631B9 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 724379681333FF3B009631B9 /* IOKit.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 724379671333FF3B009631B9 /* IOKit.framework */; }; + 724379C71333FFC7009631B9 /* usb.c in Sources */ = {isa = PBXBuildFile; fileRef = 724379C51333FFC7009631B9 /* usb.c */; }; + 724379C91333FFF3009631B9 /* CoreFoundation.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 724379C81333FFF3009631B9 /* CoreFoundation.framework */; }; + 724379CB1334000E009631B9 /* ieee1284.c in Sources */ = {isa = PBXBuildFile; fileRef = 724379CA1334000E009631B9 /* ieee1284.c */; }; + 7258EAED134594EB009286F1 /* rastertopwg.c in Sources */ = {isa = PBXBuildFile; fileRef = 7258EAEC134594EB009286F1 /* rastertopwg.c */; }; + 7258EAF413459B6D009286F1 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 7258EAF513459B6D009286F1 /* libcupsimage.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72F75A611336F9A3004BB496 /* libcupsimage.dylib */; }; + 7263EE2713330D2800BA4D44 /* libpam.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 7263EE2613330D2800BA4D44 /* libpam.dylib */; }; + 7263EE2C13330D5C00BA4D44 /* Kerberos.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 7263EE2913330D5C00BA4D44 /* Kerberos.framework */; settings = {ATTRIBUTES = (Weak, ); }; }; + 7263EE2D13330D5C00BA4D44 /* Security.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 7263EE2A13330D5C00BA4D44 /* Security.framework */; }; + 7263EE2E13330D5C00BA4D44 /* SystemConfiguration.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 7263EE2B13330D5C00BA4D44 /* SystemConfiguration.framework */; }; + 7263EE3013330DC100BA4D44 /* IOKit.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 7263EE2F13330DC100BA4D44 /* IOKit.framework */; }; + 7263EE3213330E1E00BA4D44 /* libpthread.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 7263EE3113330E1E00BA4D44 /* libpthread.dylib */; }; + 7263EE3413330E3C00BA4D44 /* libresolv.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 7263EE3313330E3C00BA4D44 /* libresolv.dylib */; }; + 7263EE3613330E4E00BA4D44 /* CoreFoundation.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 7263EE3513330E4E00BA4D44 /* CoreFoundation.framework */; }; + 7263EE3813330E7500BA4D44 /* libz.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 7263EE3713330E7500BA4D44 /* libz.dylib */; }; + 7263EE3A13330EC500BA4D44 /* libldap.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 7263EE3913330EC500BA4D44 /* libldap.dylib */; }; + 726AD702135E8A90002C930D /* ippserver.c in Sources */ = {isa = PBXBuildFile; fileRef = 726AD701135E8A90002C930D /* ippserver.c */; }; + 726AD707135E8B11002C930D /* libcups_static.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 72F75A4C1336F31B004BB496 /* libcups_static.a */; }; + 726AD708135E8B11002C930D /* CoreFoundation.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F49133306BB00FCA411 /* CoreFoundation.framework */; }; + 726AD709135E8B11002C930D /* Kerberos.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F55133308EA00FCA411 /* Kerberos.framework */; }; + 726AD70A135E8B11002C930D /* libresolv.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F53133308CB00FCA411 /* libresolv.dylib */; }; + 726AD70B135E8B11002C930D /* libz.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F4A133306BB00FCA411 /* libz.dylib */; }; + 726AD70C135E8B11002C930D /* Security.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F4B133306BB00FCA411 /* Security.framework */; }; + 726AD70D135E8B11002C930D /* SystemConfiguration.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F4C133306BB00FCA411 /* SystemConfiguration.framework */; }; + 726AD70E135E8B5E002C930D /* libiconv.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220F51133308C100FCA411 /* libiconv.dylib */; }; + 7271883D1374AB14001A2036 /* mime-private.h in Headers */ = {isa = PBXBuildFile; fileRef = 7271883C1374AB14001A2036 /* mime-private.h */; }; + 72C16CB9137B195D007E4BF4 /* file.c in Sources */ = {isa = PBXBuildFile; fileRef = 72C16CB8137B195D007E4BF4 /* file.c */; }; + 72F75A5C1336F988004BB496 /* cupstestppd.c in Sources */ = {isa = PBXBuildFile; fileRef = 72F75A5B1336F988004BB496 /* cupstestppd.c */; }; + 72F75A671336FA38004BB496 /* libcups.dylib in Frameworks */ = {isa = PBXBuildFile; fileRef = 72220EAE1333047D00FCA411 /* libcups.dylib */; }; + 72F75A6C1336FA8A004BB496 /* error.c in Sources */ = {isa = PBXBuildFile; fileRef = 72F75A691336FA8A004BB496 /* error.c */; }; + 72F75A6D1336FA8A004BB496 /* interpret.c in Sources */ = {isa = PBXBuildFile; fileRef = 72F75A6A1336FA8A004BB496 /* interpret.c */; }; + 72F75A6E1336FA8A004BB496 /* raster.c in Sources */ = {isa = PBXBuildFile; fileRef = 72F75A6B1336FA8A004BB496 /* raster.c */; }; + 72F75A6F1336FAB6004BB496 /* raster.h in Headers */ = {isa = PBXBuildFile; fileRef = 72220EFA133305BB00FCA411 /* raster.h */; settings = {ATTRIBUTES = (Public, ); }; }; +/* End PBXBuildFile section */ + +/* Begin PBXContainerItemProxy section */ + 270CCDB1135E3CDE00007BE2 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 270CCDA6135E3C9E00007BE2; + remoteInfo = testmime; + }; + 270CCDB5135E3CF700007BE2 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220FAB13330B2200FCA411; + remoteInfo = libcupsmime; + }; + 270CCDB7135E3CFD00007BE2 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 274FF6891333B1C400317ECB; + remoteInfo = libcups_static; + }; + 273BF6C81333B5410022CAAB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 274FF6891333B1C400317ECB; + remoteInfo = libcups_static; + }; + 273BF6DD1333B6370022CAAB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 273BF6BC1333B5000022CAAB; + remoteInfo = testcups; + }; + 274FF5DB13332CF900317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 274FF5E213332D4300317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 274FF5E413332D4300317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220FAB13330B2200FCA411; + remoteInfo = libcupsmime; + }; + 274FF5E613332D4300317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220F5A13330A5A00FCA411; + remoteInfo = cupsd; + }; + 274FF5E813332D4300317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 274FF5CB13332B1F00317ECB; + remoteInfo = "cups-driverd"; + }; + 274FF5F2133330FD00317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 274FF61F1333316200317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 274FF5ED133330C800317ECB; + remoteInfo = libcupsppdc; + }; + 274FF621133331D300317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 274FF5ED133330C800317ECB; + remoteInfo = libcupsppdc; + }; + 274FF6331333335200317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 274FF6381333348400317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 274FF6281333333600317ECB; + remoteInfo = "cups-deviced"; + }; + 274FF647133335A300317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 274FF63D1333358B00317ECB; + remoteInfo = "cups-exec"; + }; + 274FF659133339D900317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 274FF65D13333A3400317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 274FF64E133339C400317ECB; + remoteInfo = "cups-lpd"; + }; + 274FF66C13333AAD00317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 274FF67113333AE400317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 274FF66213333A9B00317ECB; + remoteInfo = "cups-polld"; + }; + 274FF68113333B3C00317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 274FF68313333B3C00317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220FAB13330B2200FCA411; + remoteInfo = libcupsmime; + }; + 274FF6E11333B33F00317ECB /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 274FF67713333B2F00317ECB; + remoteInfo = cupsfilter; + }; + 276683651337A9D6000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 2766836A1337AA25000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 2766835B1337A9B6000D33D0; + remoteInfo = cupsctl; + }; + 276683AD1337ACF9000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 276683AF1337ACF9000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 274FF5ED133330C800317ECB; + remoteInfo = libcupsppdc; + }; + 276683B31337AD18000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 276683B51337AD18000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 274FF5ED133330C800317ECB; + remoteInfo = libcupsppdc; + }; + 276683BB1337AE49000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 276683BD1337AE49000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 274FF5ED133330C800317ECB; + remoteInfo = libcupsppdc; + }; + 276683BF1337B1AD000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 276683C11337B1AD000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 274FF5ED133330C800317ECB; + remoteInfo = libcupsppdc; + }; + 276683C51337B1BC000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 276683C71337B1BC000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 274FF5ED133330C800317ECB; + remoteInfo = libcupsppdc; + }; + 276683D61337B24A000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 2766836F1337AC79000D33D0; + remoteInfo = ppdc; + }; + 276683D81337B24A000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 2766837C1337AC8C000D33D0; + remoteInfo = ppdhtml; + }; + 276683DA1337B24A000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 276683891337AC97000D33D0; + remoteInfo = ppdi; + }; + 276683DC1337B24A000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 276683961337ACA2000D33D0; + remoteInfo = ppdmerge; + }; + 276683DE1337B24A000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 276683A31337ACAB000D33D0; + remoteInfo = ppdpo; + }; + 276683E01337B299000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 276683E31337B2BA000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72F75A601336F9A3004BB496; + remoteInfo = libcupsimage; + }; + 276683FB1337F7B3000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 276683FE1337F7C5000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 276683EF1337F78E000D33D0; + remoteInfo = ipptool; + }; + 2766840D1337FA31000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 276684121337FA8D000D33D0 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 276684031337FA1D000D33D0; + remoteInfo = cupsaddsmb; + }; + 278C58D5136B641D00836530 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 278C58CA136B640300836530; + remoteInfo = testhttp; + }; + 278C58D7136B642F00836530 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 274FF6891333B1C400317ECB; + remoteInfo = libcups_static; + }; + 720DD6CE1358FD790064AA82 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 720DD6D01358FDBE0064AA82 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 720DD6C11358FD5F0064AA82; + remoteInfo = snmp; + }; + 72220F6413330A6500FCA411 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 72220FBB13330C0500FCA411 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 72220FBD13330C0B00FCA411 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220FAB13330B2200FCA411; + remoteInfo = libcupsmime; + }; + 724379061333E49B009631B9 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 724379101333E4EA009631B9 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 724378FC1333E43E009631B9; + remoteInfo = ipp; + }; + 724379251333E932009631B9 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 7243792A1333E962009631B9 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 724379171333E532009631B9; + remoteInfo = lpd; + }; + 724379391333FB95009631B9 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 7243793E1333FD23009631B9 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 7243792F1333FB85009631B9; + remoteInfo = socket; + }; + 724379521333FECE009631B9 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 724379461333FEA9009631B9; + remoteInfo = dnssd; + }; + 724379541333FEFE009631B9 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 724379641333FF2E009631B9 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 724379C21333FF7D009631B9 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 7243795A1333FF1D009631B9; + remoteInfo = usb; + }; + 7258EAEE13459ADA009286F1 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 7258EAE1134594C4009286F1; + remoteInfo = rastertopwg; + }; + 7258EAF013459B67009286F1 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 7258EAF213459B67009286F1 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72F75A601336F9A3004BB496; + remoteInfo = libcupsimage; + }; + 726AD703135E8AA1002C930D /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 726AD6F6135E88F0002C930D; + remoteInfo = ippserver; + }; + 726AD705135E8AC5002C930D /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 274FF6891333B1C400317ECB; + remoteInfo = libcups_static; + }; + 72F75A651336FA30004BB496 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72220EAD1333047D00FCA411; + remoteInfo = libcups; + }; + 72F75A701336FACD004BB496 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72F75A601336F9A3004BB496; + remoteInfo = libcupsimage; + }; + 72F75A721336FACD004BB496 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 72BF96371333042100B1EAD7 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 72F75A511336F950004BB496; + remoteInfo = cupstestppd; + }; +/* End PBXContainerItemProxy section */ + +/* Begin PBXCopyFilesBuildPhase section */ + 270CCDA5135E3C9E00007BE2 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 273BF6BB1333B5000022CAAB /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 274FF5CA13332B1F00317ECB /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 274FF6271333333600317ECB /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 274FF63C1333358B00317ECB /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 274FF64D133339C400317ECB /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 274FF66113333A9B00317ECB /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 274FF67613333B2F00317ECB /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 2766835A1337A9B6000D33D0 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 2766836E1337AC79000D33D0 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 2766837B1337AC8C000D33D0 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 276683881337AC97000D33D0 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 276683951337ACA2000D33D0 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 276683A21337ACAB000D33D0 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 276683EE1337F78E000D33D0 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 276684021337FA1D000D33D0 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 278C58C9136B640300836530 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 720DD6C01358FD5F0064AA82 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 72220F5913330A5A00FCA411 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 724378FB1333E43E009631B9 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 724379161333E532009631B9 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 7243792E1333FB85009631B9 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 724379451333FEA9009631B9 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 724379591333FF1D009631B9 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 7258EAE0134594C4009286F1 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 726AD6F5135E88F0002C930D /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; + 72F75A501336F950004BB496 /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = /usr/share/man/man1/; + dstSubfolderSpec = 0; + files = ( + ); + runOnlyForDeploymentPostprocessing = 1; + }; +/* End PBXCopyFilesBuildPhase section */ + +/* Begin PBXFileReference section */ + 270CCDA7135E3C9E00007BE2 /* testmime */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = testmime; sourceTree = BUILT_PRODUCTS_DIR; }; + 270CCDBB135E3D3E00007BE2 /* testmime.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = testmime.c; path = ../scheduler/testmime.c; sourceTree = ""; }; + 2732E089137A3F5200FAFEF6 /* cancel.c */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c; name = cancel.c; path = ../systemv/cancel.c; sourceTree = ""; }; + 2732E08A137A3F5200FAFEF6 /* cupsaccept.c */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c; name = cupsaccept.c; path = ../systemv/cupsaccept.c; sourceTree = ""; }; + 2732E08B137A3F5200FAFEF6 /* cupstestdsc.c */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c; name = cupstestdsc.c; path = ../systemv/cupstestdsc.c; sourceTree = ""; }; + 2732E08C137A3F5200FAFEF6 /* lp.c */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c; name = lp.c; path = ../systemv/lp.c; sourceTree = ""; }; + 2732E08D137A3F5200FAFEF6 /* lpadmin.c */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c; name = lpadmin.c; path = ../systemv/lpadmin.c; sourceTree = ""; }; + 2732E08E137A3F5200FAFEF6 /* lpinfo.c */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c; name = lpinfo.c; path = ../systemv/lpinfo.c; sourceTree = ""; }; + 2732E08F137A3F5200FAFEF6 /* lpmove.c */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c; name = lpmove.c; path = ../systemv/lpmove.c; sourceTree = ""; }; + 2732E090137A3F5200FAFEF6 /* lpoptions.c */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c; name = lpoptions.c; path = ../systemv/lpoptions.c; sourceTree = ""; }; + 2732E091137A3F5200FAFEF6 /* lppasswd.c */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c; name = lppasswd.c; path = ../systemv/lppasswd.c; sourceTree = ""; }; + 2732E092137A3F5200FAFEF6 /* lpstat.c */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c; name = lpstat.c; path = ../systemv/lpstat.c; sourceTree = ""; }; + 273BF6BD1333B5000022CAAB /* testcups */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = testcups; sourceTree = BUILT_PRODUCTS_DIR; }; + 273BF6C61333B5370022CAAB /* testcups.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = testcups.c; path = ../cups/testcups.c; sourceTree = ""; }; + 273BF6CB1333B5950022CAAB /* CoreFoundation.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = CoreFoundation.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/CoreFoundation.framework; sourceTree = DEVELOPER_DIR; }; + 273BF6CC1333B5950022CAAB /* libiconv.dylib */ = {isa = PBXFileReference; lastKnownFileType = "compiled.mach-o.dylib"; name = libiconv.dylib; path = SDKs/MacOSX10.6.sdk/usr/lib/libiconv.dylib; sourceTree = DEVELOPER_DIR; }; + 273BF6CD1333B5950022CAAB /* libz.dylib */ = {isa = PBXFileReference; lastKnownFileType = "compiled.mach-o.dylib"; name = libz.dylib; path = SDKs/MacOSX10.6.sdk/usr/lib/libz.dylib; sourceTree = DEVELOPER_DIR; }; + 273BF6D11333B5C30022CAAB /* Kerberos.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = Kerberos.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/Kerberos.framework; sourceTree = DEVELOPER_DIR; }; + 273BF6D21333B5C30022CAAB /* Security.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = Security.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/Security.framework; sourceTree = DEVELOPER_DIR; }; + 273BF6D51333B5F60022CAAB /* libresolv.dylib */ = {isa = PBXFileReference; lastKnownFileType = "compiled.mach-o.dylib"; name = libresolv.dylib; path = SDKs/MacOSX10.6.sdk/usr/lib/libresolv.dylib; sourceTree = DEVELOPER_DIR; }; + 273BF6D61333B5F60022CAAB /* SystemConfiguration.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = SystemConfiguration.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/SystemConfiguration.framework; sourceTree = DEVELOPER_DIR; }; + 274FF5CC13332B1F00317ECB /* cups-driverd */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = "cups-driverd"; sourceTree = BUILT_PRODUCTS_DIR; }; + 274FF5D613332CC700317ECB /* cups-driverd.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "cups-driverd.cxx"; path = "../scheduler/cups-driverd.cxx"; sourceTree = ""; }; + 274FF5D713332CC700317ECB /* util.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = util.c; path = ../scheduler/util.c; sourceTree = ""; }; + 274FF5D813332CC700317ECB /* util.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = util.h; path = ../scheduler/util.h; sourceTree = ""; }; + 274FF5EE133330C800317ECB /* libcupsppdc.dylib */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.dylib"; includeInIndex = 0; path = libcupsppdc.dylib; sourceTree = BUILT_PRODUCTS_DIR; }; + 274FF5F51333315100317ECB /* ppdc-array.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-array.cxx"; path = "../ppdc/ppdc-array.cxx"; sourceTree = ""; }; + 274FF5F61333315100317ECB /* ppdc-attr.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-attr.cxx"; path = "../ppdc/ppdc-attr.cxx"; sourceTree = ""; }; + 274FF5F71333315100317ECB /* ppdc-catalog.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-catalog.cxx"; path = "../ppdc/ppdc-catalog.cxx"; sourceTree = ""; }; + 274FF5F81333315100317ECB /* ppdc-choice.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-choice.cxx"; path = "../ppdc/ppdc-choice.cxx"; sourceTree = ""; }; + 274FF5F91333315100317ECB /* ppdc-constraint.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-constraint.cxx"; path = "../ppdc/ppdc-constraint.cxx"; sourceTree = ""; }; + 274FF5FA1333315100317ECB /* ppdc-driver.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-driver.cxx"; path = "../ppdc/ppdc-driver.cxx"; sourceTree = ""; }; + 274FF5FB1333315100317ECB /* ppdc-file.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-file.cxx"; path = "../ppdc/ppdc-file.cxx"; sourceTree = ""; }; + 274FF5FC1333315100317ECB /* ppdc-filter.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-filter.cxx"; path = "../ppdc/ppdc-filter.cxx"; sourceTree = ""; }; + 274FF5FD1333315100317ECB /* ppdc-font.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-font.cxx"; path = "../ppdc/ppdc-font.cxx"; sourceTree = ""; }; + 274FF5FE1333315100317ECB /* ppdc-group.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-group.cxx"; path = "../ppdc/ppdc-group.cxx"; sourceTree = ""; }; + 274FF5FF1333315100317ECB /* ppdc-import.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-import.cxx"; path = "../ppdc/ppdc-import.cxx"; sourceTree = ""; }; + 274FF6001333315100317ECB /* ppdc-mediasize.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-mediasize.cxx"; path = "../ppdc/ppdc-mediasize.cxx"; sourceTree = ""; }; + 274FF6011333315100317ECB /* ppdc-message.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-message.cxx"; path = "../ppdc/ppdc-message.cxx"; sourceTree = ""; }; + 274FF6021333315100317ECB /* ppdc-option.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-option.cxx"; path = "../ppdc/ppdc-option.cxx"; sourceTree = ""; }; + 274FF6031333315100317ECB /* ppdc-private.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = "ppdc-private.h"; path = "../ppdc/ppdc-private.h"; sourceTree = ""; }; + 274FF6041333315100317ECB /* ppdc-profile.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-profile.cxx"; path = "../ppdc/ppdc-profile.cxx"; sourceTree = ""; }; + 274FF6051333315100317ECB /* ppdc-shared.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-shared.cxx"; path = "../ppdc/ppdc-shared.cxx"; sourceTree = ""; }; + 274FF6061333315100317ECB /* ppdc-source.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-source.cxx"; path = "../ppdc/ppdc-source.cxx"; sourceTree = ""; }; + 274FF6071333315100317ECB /* ppdc-string.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-string.cxx"; path = "../ppdc/ppdc-string.cxx"; sourceTree = ""; }; + 274FF6081333315100317ECB /* ppdc-variable.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = "ppdc-variable.cxx"; path = "../ppdc/ppdc-variable.cxx"; sourceTree = ""; }; + 274FF6091333315100317ECB /* ppdc.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = ppdc.h; path = ../ppdc/ppdc.h; sourceTree = ""; }; + 274FF6291333333600317ECB /* cups-deviced */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = "cups-deviced"; sourceTree = BUILT_PRODUCTS_DIR; }; + 274FF6351333344400317ECB /* cups-deviced.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = "cups-deviced.c"; path = "../scheduler/cups-deviced.c"; sourceTree = ""; }; + 274FF63E1333358B00317ECB /* cups-exec */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = "cups-exec"; sourceTree = BUILT_PRODUCTS_DIR; }; + 274FF6491333398D00317ECB /* cups-exec.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = "cups-exec.c"; path = "../scheduler/cups-exec.c"; sourceTree = ""; }; + 274FF64F133339C400317ECB /* cups-lpd */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = "cups-lpd"; sourceTree = BUILT_PRODUCTS_DIR; }; + 274FF65B133339FC00317ECB /* cups-lpd.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = "cups-lpd.c"; path = "../scheduler/cups-lpd.c"; sourceTree = ""; }; + 274FF66313333A9B00317ECB /* cups-polld */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = "cups-polld"; sourceTree = BUILT_PRODUCTS_DIR; }; + 274FF66F13333ACF00317ECB /* cups-polld.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = "cups-polld.c"; path = "../scheduler/cups-polld.c"; sourceTree = ""; }; + 274FF67813333B2F00317ECB /* cupsfilter */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = cupsfilter; sourceTree = BUILT_PRODUCTS_DIR; }; + 274FF68713333B6E00317ECB /* cupsfilter.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = cupsfilter.c; path = ../scheduler/cupsfilter.c; sourceTree = ""; }; + 276683561337A8C5000D33D0 /* cups.strings */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.strings; name = cups.strings; path = ../locale/cups.strings; sourceTree = ""; }; + 2766835C1337A9B6000D33D0 /* cupsctl */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = cupsctl; sourceTree = BUILT_PRODUCTS_DIR; }; + 276683681337AA00000D33D0 /* cupsctl.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = cupsctl.c; path = ../systemv/cupsctl.c; sourceTree = ""; }; + 276683701337AC79000D33D0 /* ppdc */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = ppdc; sourceTree = BUILT_PRODUCTS_DIR; }; + 2766837D1337AC8C000D33D0 /* ppdhtml */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = ppdhtml; sourceTree = BUILT_PRODUCTS_DIR; }; + 2766838A1337AC97000D33D0 /* ppdi */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = ppdi; sourceTree = BUILT_PRODUCTS_DIR; }; + 276683971337ACA2000D33D0 /* ppdmerge */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = ppdmerge; sourceTree = BUILT_PRODUCTS_DIR; }; + 276683A41337ACAB000D33D0 /* ppdpo */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = ppdpo; sourceTree = BUILT_PRODUCTS_DIR; }; + 276683CC1337B201000D33D0 /* ppdc.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = ppdc.cxx; path = ../ppdc/ppdc.cxx; sourceTree = ""; }; + 276683CE1337B20D000D33D0 /* ppdhtml.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = ppdhtml.cxx; path = ../ppdc/ppdhtml.cxx; sourceTree = ""; }; + 276683D01337B21A000D33D0 /* ppdi.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = ppdi.cxx; path = ../ppdc/ppdi.cxx; sourceTree = ""; }; + 276683D21337B228000D33D0 /* ppdmerge.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = ppdmerge.cxx; path = ../ppdc/ppdmerge.cxx; sourceTree = ""; }; + 276683D41337B237000D33D0 /* ppdpo.cxx */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = ppdpo.cxx; path = ../ppdc/ppdpo.cxx; sourceTree = ""; }; + 276683F01337F78E000D33D0 /* ipptool */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = ipptool; sourceTree = BUILT_PRODUCTS_DIR; }; + 276683F91337F7A9000D33D0 /* ipptool.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = ipptool.c; path = ../test/ipptool.c; sourceTree = ""; }; + 276684041337FA1D000D33D0 /* cupsaddsmb */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = cupsaddsmb; sourceTree = BUILT_PRODUCTS_DIR; }; + 276684101337FA7C000D33D0 /* cupsaddsmb.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = cupsaddsmb.c; path = ../systemv/cupsaddsmb.c; sourceTree = ""; }; + 278C58CB136B640300836530 /* testhttp */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = testhttp; sourceTree = BUILT_PRODUCTS_DIR; }; + 278C58DA136B645C00836530 /* CoreFoundation.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = CoreFoundation.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/CoreFoundation.framework; sourceTree = DEVELOPER_DIR; }; + 278C58DB136B645C00836530 /* Kerberos.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = Kerberos.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/Kerberos.framework; sourceTree = DEVELOPER_DIR; }; + 278C58DD136B645C00836530 /* SystemConfiguration.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = SystemConfiguration.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/SystemConfiguration.framework; sourceTree = DEVELOPER_DIR; }; + 278C58E2136B647200836530 /* testhttp.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = testhttp.c; path = ../cups/testhttp.c; sourceTree = ""; }; + 278C58E5136B64AF00836530 /* CoreFoundation.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = CoreFoundation.framework; path = /System/Library/Frameworks/CoreFoundation.framework; sourceTree = ""; }; + 278C58E6136B64B000836530 /* Kerberos.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = Kerberos.framework; path = /System/Library/Frameworks/Kerberos.framework; sourceTree = ""; }; + 278C58E7136B64B000836530 /* Security.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = Security.framework; path = /System/Library/Frameworks/Security.framework; sourceTree = ""; }; + 278C58E8136B64B000836530 /* SystemConfiguration.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = SystemConfiguration.framework; path = /System/Library/Frameworks/SystemConfiguration.framework; sourceTree = ""; }; + 278C58F4136B652300836530 /* Security.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = Security.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/Security.framework; sourceTree = DEVELOPER_DIR; }; + 27D3037C134148CB00F022B1 /* libcups_s.exp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.exports; name = libcups_s.exp; path = ../cups/libcups_s.exp; sourceTree = ""; }; + 27D3037D134148CB00F022B1 /* libcups2.def */ = {isa = PBXFileReference; lastKnownFileType = text; name = libcups2.def; path = ../cups/libcups2.def; sourceTree = ""; }; + 720DD6C21358FD5F0064AA82 /* snmp */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = snmp; sourceTree = BUILT_PRODUCTS_DIR; }; + 720DD6D21358FDDE0064AA82 /* snmp.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = snmp.c; path = ../backend/snmp.c; sourceTree = ""; }; + 72220EAE1333047D00FCA411 /* libcups.dylib */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.dylib"; includeInIndex = 0; path = libcups.dylib; sourceTree = BUILT_PRODUCTS_DIR; }; + 72220EB51333052D00FCA411 /* adminutil.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = adminutil.c; path = ../cups/adminutil.c; sourceTree = ""; }; + 72220EB71333056300FCA411 /* adminutil.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = adminutil.h; path = ../cups/adminutil.h; sourceTree = ""; }; + 72220EB81333056300FCA411 /* array.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = array.c; path = ../cups/array.c; sourceTree = ""; }; + 72220EB91333056300FCA411 /* array.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = array.h; path = ../cups/array.h; sourceTree = ""; }; + 72220EBA1333056300FCA411 /* attr.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = attr.c; path = ../cups/attr.c; sourceTree = ""; }; + 72220EBB1333056300FCA411 /* auth.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = auth.c; path = ../cups/auth.c; sourceTree = ""; }; + 72220EBC1333056300FCA411 /* backchannel.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = backchannel.c; path = ../cups/backchannel.c; sourceTree = ""; }; + 72220EBD1333056300FCA411 /* backend.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = backend.c; path = ../cups/backend.c; sourceTree = ""; }; + 72220EBE1333056300FCA411 /* backend.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = backend.h; path = ../cups/backend.h; sourceTree = ""; }; + 72220EBF1333056300FCA411 /* conflicts.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = conflicts.c; path = ../cups/conflicts.c; sourceTree = ""; }; + 72220EC01333056300FCA411 /* cups-private.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = "cups-private.h"; path = "../cups/cups-private.h"; sourceTree = ""; }; + 72220EC11333056300FCA411 /* cups.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = cups.h; path = ../cups/cups.h; sourceTree = ""; }; + 72220EC21333056300FCA411 /* custom.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = custom.c; path = ../cups/custom.c; sourceTree = ""; }; + 72220EC31333056300FCA411 /* debug-private.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = "debug-private.h"; path = "../cups/debug-private.h"; sourceTree = ""; }; + 72220ED1133305BB00FCA411 /* debug.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = debug.c; path = ../cups/debug.c; sourceTree = ""; }; + 72220ED2133305BB00FCA411 /* dest.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = dest.c; path = ../cups/dest.c; sourceTree = ""; }; + 72220ED3133305BB00FCA411 /* dir.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = dir.c; path = ../cups/dir.c; sourceTree = ""; }; + 72220ED4133305BB00FCA411 /* dir.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = dir.h; path = ../cups/dir.h; sourceTree = ""; }; + 72220ED5133305BB00FCA411 /* emit.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = emit.c; path = ../cups/emit.c; sourceTree = ""; }; + 72220ED6133305BB00FCA411 /* encode.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = encode.c; path = ../cups/encode.c; sourceTree = ""; }; + 72220ED7133305BB00FCA411 /* file-private.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = "file-private.h"; path = "../cups/file-private.h"; sourceTree = ""; }; + 72220ED8133305BB00FCA411 /* file.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = file.c; path = ../cups/file.c; sourceTree = ""; }; + 72220ED9133305BB00FCA411 /* file.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = file.h; path = ../cups/file.h; sourceTree = ""; }; + 72220EDA133305BB00FCA411 /* getdevices.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = getdevices.c; path = ../cups/getdevices.c; sourceTree = ""; }; + 72220EDB133305BB00FCA411 /* getifaddrs.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = getifaddrs.c; path = ../cups/getifaddrs.c; sourceTree = ""; }; + 72220EDC133305BB00FCA411 /* getputfile.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = getputfile.c; path = ../cups/getputfile.c; sourceTree = ""; }; + 72220EDD133305BB00FCA411 /* globals.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = globals.c; path = ../cups/globals.c; sourceTree = ""; }; + 72220EDE133305BB00FCA411 /* http-addr.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = "http-addr.c"; path = "../cups/http-addr.c"; sourceTree = ""; }; + 72220EDF133305BB00FCA411 /* http-addrlist.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = "http-addrlist.c"; path = "../cups/http-addrlist.c"; sourceTree = ""; }; + 72220EE0133305BB00FCA411 /* http-private.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = "http-private.h"; path = "../cups/http-private.h"; sourceTree = ""; }; + 72220EE1133305BB00FCA411 /* http-support.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = "http-support.c"; path = "../cups/http-support.c"; sourceTree = ""; }; + 72220EE2133305BB00FCA411 /* http.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = http.c; path = ../cups/http.c; sourceTree = ""; }; + 72220EE3133305BB00FCA411 /* http.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = http.h; path = ../cups/http.h; sourceTree = ""; }; + 72220EE4133305BB00FCA411 /* ipp-private.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = "ipp-private.h"; path = "../cups/ipp-private.h"; sourceTree = ""; }; + 72220EE5133305BB00FCA411 /* ipp-support.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = "ipp-support.c"; path = "../cups/ipp-support.c"; sourceTree = ""; }; + 72220EE6133305BB00FCA411 /* ipp.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = ipp.c; path = ../cups/ipp.c; sourceTree = ""; }; + 72220EE7133305BB00FCA411 /* ipp.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = ipp.h; path = ../cups/ipp.h; sourceTree = ""; }; + 72220EE8133305BB00FCA411 /* langprintf.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = langprintf.c; path = ../cups/langprintf.c; sourceTree = ""; }; + 72220EE9133305BB00FCA411 /* language-private.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = "language-private.h"; path = "../cups/language-private.h"; sourceTree = ""; }; + 72220EEA133305BB00FCA411 /* language.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = language.c; path = ../cups/language.c; sourceTree = ""; }; + 72220EEB133305BB00FCA411 /* language.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = language.h; path = ../cups/language.h; sourceTree = ""; }; + 72220EEC133305BB00FCA411 /* localize.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = localize.c; path = ../cups/localize.c; sourceTree = ""; }; + 72220EED133305BB00FCA411 /* mark.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = mark.c; path = ../cups/mark.c; sourceTree = ""; }; + 72220EEE133305BB00FCA411 /* md5-private.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = "md5-private.h"; path = "../cups/md5-private.h"; sourceTree = ""; }; + 72220EEF133305BB00FCA411 /* md5.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = md5.c; path = ../cups/md5.c; sourceTree = ""; }; + 72220EF0133305BB00FCA411 /* md5passwd.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = md5passwd.c; path = ../cups/md5passwd.c; sourceTree = ""; }; + 72220EF1133305BB00FCA411 /* notify.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = notify.c; path = ../cups/notify.c; sourceTree = ""; }; + 72220EF2133305BB00FCA411 /* options.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = options.c; path = ../cups/options.c; sourceTree = ""; }; + 72220EF3133305BB00FCA411 /* page.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = page.c; path = ../cups/page.c; sourceTree = ""; }; + 72220EF4133305BB00FCA411 /* ppd-cache.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = "ppd-cache.c"; path = "../cups/ppd-cache.c"; sourceTree = ""; }; + 72220EF5133305BB00FCA411 /* ppd-private.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = "ppd-private.h"; path = "../cups/ppd-private.h"; sourceTree = ""; }; + 72220EF6133305BB00FCA411 /* ppd.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = ppd.c; path = ../cups/ppd.c; sourceTree = ""; }; + 72220EF7133305BB00FCA411 /* ppd.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = ppd.h; path = ../cups/ppd.h; sourceTree = ""; }; + 72220EF8133305BB00FCA411 /* pwg-media.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = "pwg-media.c"; path = "../cups/pwg-media.c"; sourceTree = ""; }; + 72220EF9133305BB00FCA411 /* pwg-private.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = "pwg-private.h"; path = "../cups/pwg-private.h"; sourceTree = ""; }; + 72220EFA133305BB00FCA411 /* raster.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = raster.h; path = ../cups/raster.h; sourceTree = ""; }; + 72220EFB133305BB00FCA411 /* request.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = request.c; path = ../cups/request.c; sourceTree = ""; }; + 72220EFC133305BB00FCA411 /* sidechannel.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = sidechannel.c; path = ../cups/sidechannel.c; sourceTree = ""; }; + 72220EFD133305BB00FCA411 /* sidechannel.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = sidechannel.h; path = ../cups/sidechannel.h; sourceTree = ""; }; + 72220EFE133305BB00FCA411 /* snmp-private.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = "snmp-private.h"; path = "../cups/snmp-private.h"; sourceTree = ""; }; + 72220EFF133305BB00FCA411 /* snmp.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = snmp.c; path = ../cups/snmp.c; sourceTree = ""; }; + 72220F00133305BB00FCA411 /* snprintf.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = snprintf.c; path = ../cups/snprintf.c; sourceTree = ""; }; + 72220F01133305BB00FCA411 /* string-private.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = "string-private.h"; path = "../cups/string-private.h"; sourceTree = ""; }; + 72220F02133305BB00FCA411 /* string.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = string.c; path = ../cups/string.c; sourceTree = ""; }; + 72220F03133305BB00FCA411 /* tempfile.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = tempfile.c; path = ../cups/tempfile.c; sourceTree = ""; }; + 72220F04133305BB00FCA411 /* thread-private.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = "thread-private.h"; path = "../cups/thread-private.h"; sourceTree = ""; }; + 72220F05133305BB00FCA411 /* thread.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = thread.c; path = ../cups/thread.c; sourceTree = ""; }; + 72220F06133305BB00FCA411 /* transcode.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = transcode.c; path = ../cups/transcode.c; sourceTree = ""; }; + 72220F07133305BB00FCA411 /* transcode.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = transcode.h; path = ../cups/transcode.h; sourceTree = ""; }; + 72220F08133305BB00FCA411 /* usersys.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = usersys.c; path = ../cups/usersys.c; sourceTree = ""; }; + 72220F09133305BB00FCA411 /* util.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = util.c; path = ../cups/util.c; sourceTree = ""; }; + 72220F0A133305BB00FCA411 /* versioning.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = versioning.h; path = ../cups/versioning.h; sourceTree = ""; }; + 72220F471333063D00FCA411 /* config.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = config.h; sourceTree = ""; }; + 72220F49133306BB00FCA411 /* CoreFoundation.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = CoreFoundation.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/CoreFoundation.framework; sourceTree = DEVELOPER_DIR; }; + 72220F4A133306BB00FCA411 /* libz.dylib */ = {isa = PBXFileReference; lastKnownFileType = "compiled.mach-o.dylib"; name = libz.dylib; path = SDKs/MacOSX10.6.sdk/usr/lib/libz.dylib; sourceTree = DEVELOPER_DIR; }; + 72220F4B133306BB00FCA411 /* Security.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = Security.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/Security.framework; sourceTree = DEVELOPER_DIR; }; + 72220F4C133306BB00FCA411 /* SystemConfiguration.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = SystemConfiguration.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/SystemConfiguration.framework; sourceTree = DEVELOPER_DIR; }; + 72220F51133308C100FCA411 /* libiconv.dylib */ = {isa = PBXFileReference; lastKnownFileType = "compiled.mach-o.dylib"; name = libiconv.dylib; path = SDKs/MacOSX10.6.sdk/usr/lib/libiconv.dylib; sourceTree = DEVELOPER_DIR; }; + 72220F53133308CB00FCA411 /* libresolv.dylib */ = {isa = PBXFileReference; lastKnownFileType = "compiled.mach-o.dylib"; name = libresolv.dylib; path = SDKs/MacOSX10.6.sdk/usr/lib/libresolv.dylib; sourceTree = DEVELOPER_DIR; }; + 72220F55133308EA00FCA411 /* Kerberos.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = Kerberos.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/Kerberos.framework; sourceTree = DEVELOPER_DIR; }; + 72220F5B13330A5A00FCA411 /* cupsd */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = cupsd; sourceTree = BUILT_PRODUCTS_DIR; }; + 72220F6713330A8500FCA411 /* ApplicationServices.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = ApplicationServices.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/ApplicationServices.framework; sourceTree = DEVELOPER_DIR; }; + 72220F6913330B0C00FCA411 /* auth.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = auth.c; path = ../scheduler/auth.c; sourceTree = SOURCE_ROOT; }; + 72220F6A13330B0C00FCA411 /* auth.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = auth.h; path = ../scheduler/auth.h; sourceTree = SOURCE_ROOT; }; + 72220F6B13330B0C00FCA411 /* banners.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = banners.c; path = ../scheduler/banners.c; sourceTree = SOURCE_ROOT; }; + 72220F6C13330B0C00FCA411 /* banners.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = banners.h; path = ../scheduler/banners.h; sourceTree = SOURCE_ROOT; }; + 72220F6D13330B0C00FCA411 /* cert.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = cert.c; path = ../scheduler/cert.c; sourceTree = SOURCE_ROOT; }; + 72220F6E13330B0C00FCA411 /* cert.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = cert.h; path = ../scheduler/cert.h; sourceTree = SOURCE_ROOT; }; + 72220F6F13330B0C00FCA411 /* classes.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = classes.c; path = ../scheduler/classes.c; sourceTree = SOURCE_ROOT; }; + 72220F7013330B0C00FCA411 /* classes.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = classes.h; path = ../scheduler/classes.h; sourceTree = SOURCE_ROOT; }; + 72220F7113330B0C00FCA411 /* client.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = client.c; path = ../scheduler/client.c; sourceTree = SOURCE_ROOT; }; + 72220F7213330B0C00FCA411 /* client.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = client.h; path = ../scheduler/client.h; sourceTree = SOURCE_ROOT; }; + 72220F7313330B0C00FCA411 /* conf.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = conf.c; path = ../scheduler/conf.c; sourceTree = SOURCE_ROOT; }; + 72220F7413330B0C00FCA411 /* conf.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = conf.h; path = ../scheduler/conf.h; sourceTree = SOURCE_ROOT; }; + 72220F7513330B0C00FCA411 /* cupsd.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = cupsd.h; path = ../scheduler/cupsd.h; sourceTree = SOURCE_ROOT; }; + 72220F7613330B0C00FCA411 /* dirsvc.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = dirsvc.c; path = ../scheduler/dirsvc.c; sourceTree = SOURCE_ROOT; }; + 72220F7713330B0C00FCA411 /* dirsvc.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = dirsvc.h; path = ../scheduler/dirsvc.h; sourceTree = SOURCE_ROOT; }; + 72220F7813330B0C00FCA411 /* env.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = env.c; path = ../scheduler/env.c; sourceTree = SOURCE_ROOT; }; + 72220F7913330B0C00FCA411 /* ipp.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = ipp.c; path = ../scheduler/ipp.c; sourceTree = SOURCE_ROOT; }; + 72220F7A13330B0C00FCA411 /* job.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = job.c; path = ../scheduler/job.c; sourceTree = SOURCE_ROOT; }; + 72220F7B13330B0C00FCA411 /* job.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = job.h; path = ../scheduler/job.h; sourceTree = SOURCE_ROOT; }; + 72220F7C13330B0C00FCA411 /* listen.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = listen.c; path = ../scheduler/listen.c; sourceTree = SOURCE_ROOT; }; + 72220F7D13330B0C00FCA411 /* log.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = log.c; path = ../scheduler/log.c; sourceTree = SOURCE_ROOT; }; + 72220F7E13330B0C00FCA411 /* main.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = main.c; path = ../scheduler/main.c; sourceTree = SOURCE_ROOT; }; + 72220F7F13330B0C00FCA411 /* network.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = network.c; path = ../scheduler/network.c; sourceTree = SOURCE_ROOT; }; + 72220F8013330B0C00FCA411 /* network.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = network.h; path = ../scheduler/network.h; sourceTree = SOURCE_ROOT; }; + 72220F8113330B0C00FCA411 /* policy.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = policy.c; path = ../scheduler/policy.c; sourceTree = SOURCE_ROOT; }; + 72220F8213330B0C00FCA411 /* policy.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = policy.h; path = ../scheduler/policy.h; sourceTree = SOURCE_ROOT; }; + 72220F8313330B0C00FCA411 /* printers.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = printers.c; path = ../scheduler/printers.c; sourceTree = SOURCE_ROOT; }; + 72220F8413330B0C00FCA411 /* printers.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = printers.h; path = ../scheduler/printers.h; sourceTree = SOURCE_ROOT; }; + 72220F8513330B0C00FCA411 /* process.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = process.c; path = ../scheduler/process.c; sourceTree = SOURCE_ROOT; }; + 72220F8613330B0C00FCA411 /* quotas.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = quotas.c; path = ../scheduler/quotas.c; sourceTree = SOURCE_ROOT; }; + 72220F8813330B0C00FCA411 /* select.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = select.c; path = ../scheduler/select.c; sourceTree = SOURCE_ROOT; }; + 72220F8913330B0C00FCA411 /* server.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = server.c; path = ../scheduler/server.c; sourceTree = SOURCE_ROOT; }; + 72220F8A13330B0C00FCA411 /* statbuf.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = statbuf.c; path = ../scheduler/statbuf.c; sourceTree = SOURCE_ROOT; }; + 72220F8B13330B0C00FCA411 /* statbuf.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = statbuf.h; path = ../scheduler/statbuf.h; sourceTree = SOURCE_ROOT; }; + 72220F8C13330B0C00FCA411 /* subscriptions.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = subscriptions.c; path = ../scheduler/subscriptions.c; sourceTree = SOURCE_ROOT; }; + 72220F8D13330B0C00FCA411 /* subscriptions.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = subscriptions.h; path = ../scheduler/subscriptions.h; sourceTree = SOURCE_ROOT; }; + 72220F8E13330B0C00FCA411 /* sysman.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = sysman.c; path = ../scheduler/sysman.c; sourceTree = SOURCE_ROOT; }; + 72220F8F13330B0C00FCA411 /* sysman.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = sysman.h; path = ../scheduler/sysman.h; sourceTree = SOURCE_ROOT; }; + 72220FAC13330B2200FCA411 /* libcupsmime.dylib */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.dylib"; includeInIndex = 0; path = libcupsmime.dylib; sourceTree = BUILT_PRODUCTS_DIR; }; + 72220FB213330BCE00FCA411 /* filter.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = filter.c; path = ../scheduler/filter.c; sourceTree = ""; }; + 72220FB313330BCE00FCA411 /* mime.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = mime.c; path = ../scheduler/mime.c; sourceTree = ""; }; + 72220FB413330BCE00FCA411 /* mime.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = mime.h; path = ../scheduler/mime.h; sourceTree = ""; }; + 72220FB513330BCE00FCA411 /* type.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = type.c; path = ../scheduler/type.c; sourceTree = ""; }; + 7234F41F1378A16F00D3E9C9 /* array-private.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = "array-private.h"; path = "../cups/array-private.h"; sourceTree = ""; }; + 724378FD1333E43E009631B9 /* ipp */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = ipp; sourceTree = BUILT_PRODUCTS_DIR; }; + 724379091333E4E3009631B9 /* backend-private.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = "backend-private.h"; path = "../backend/backend-private.h"; sourceTree = ""; }; + 7243790A1333E4E3009631B9 /* ipp.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = ipp.c; path = ../backend/ipp.c; sourceTree = ""; }; + 7243790B1333E4E3009631B9 /* network.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = network.c; path = ../backend/network.c; sourceTree = ""; }; + 7243790C1333E4E3009631B9 /* snmp-supplies.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = "snmp-supplies.c"; path = "../backend/snmp-supplies.c"; sourceTree = ""; }; + 724379121333E516009631B9 /* runloop.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = runloop.c; path = ../backend/runloop.c; sourceTree = ""; }; + 724379181333E532009631B9 /* lpd */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = lpd; sourceTree = BUILT_PRODUCTS_DIR; }; + 724379281333E952009631B9 /* lpd.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = lpd.c; path = ../backend/lpd.c; sourceTree = ""; }; + 724379301333FB85009631B9 /* socket */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = socket; sourceTree = BUILT_PRODUCTS_DIR; }; + 7243793C1333FD19009631B9 /* socket.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = socket.c; path = ../backend/socket.c; sourceTree = ""; }; + 724379471333FEA9009631B9 /* dnssd */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = dnssd; sourceTree = BUILT_PRODUCTS_DIR; }; + 724379501333FEBB009631B9 /* dnssd.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = dnssd.c; path = ../backend/dnssd.c; sourceTree = ""; }; + 7243795B1333FF1D009631B9 /* usb */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = usb; sourceTree = BUILT_PRODUCTS_DIR; }; + 724379671333FF3B009631B9 /* IOKit.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = IOKit.framework; path = SDKs/MacOSX10.7.sdk/System/Library/Frameworks/IOKit.framework; sourceTree = DEVELOPER_DIR; }; + 724379C41333FFC7009631B9 /* usb-darwin.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = "usb-darwin.c"; path = "../backend/usb-darwin.c"; sourceTree = ""; }; + 724379C51333FFC7009631B9 /* usb.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = usb.c; path = ../backend/usb.c; sourceTree = ""; }; + 724379C81333FFF3009631B9 /* CoreFoundation.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = CoreFoundation.framework; path = SDKs/MacOSX10.7.sdk/System/Library/Frameworks/CoreFoundation.framework; sourceTree = DEVELOPER_DIR; }; + 724379CA1334000E009631B9 /* ieee1284.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = ieee1284.c; path = ../backend/ieee1284.c; sourceTree = ""; }; + 7258EAE2134594C4009286F1 /* rastertopwg */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = rastertopwg; sourceTree = BUILT_PRODUCTS_DIR; }; + 7258EAEC134594EB009286F1 /* rastertopwg.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = rastertopwg.c; path = ../filter/rastertopwg.c; sourceTree = ""; }; + 7263EE2613330D2800BA4D44 /* libpam.dylib */ = {isa = PBXFileReference; lastKnownFileType = "compiled.mach-o.dylib"; name = libpam.dylib; path = SDKs/MacOSX10.6.sdk/usr/lib/libpam.dylib; sourceTree = DEVELOPER_DIR; }; + 7263EE2913330D5C00BA4D44 /* Kerberos.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = Kerberos.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/Kerberos.framework; sourceTree = DEVELOPER_DIR; }; + 7263EE2A13330D5C00BA4D44 /* Security.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = Security.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/Security.framework; sourceTree = DEVELOPER_DIR; }; + 7263EE2B13330D5C00BA4D44 /* SystemConfiguration.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = SystemConfiguration.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/SystemConfiguration.framework; sourceTree = DEVELOPER_DIR; }; + 7263EE2F13330DC100BA4D44 /* IOKit.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = IOKit.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/IOKit.framework; sourceTree = DEVELOPER_DIR; }; + 7263EE3113330E1E00BA4D44 /* libpthread.dylib */ = {isa = PBXFileReference; lastKnownFileType = "compiled.mach-o.dylib"; name = libpthread.dylib; path = SDKs/MacOSX10.6.sdk/usr/lib/libpthread.dylib; sourceTree = DEVELOPER_DIR; }; + 7263EE3313330E3C00BA4D44 /* libresolv.dylib */ = {isa = PBXFileReference; lastKnownFileType = "compiled.mach-o.dylib"; name = libresolv.dylib; path = SDKs/MacOSX10.6.sdk/usr/lib/libresolv.dylib; sourceTree = DEVELOPER_DIR; }; + 7263EE3513330E4E00BA4D44 /* CoreFoundation.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = CoreFoundation.framework; path = SDKs/MacOSX10.6.sdk/System/Library/Frameworks/CoreFoundation.framework; sourceTree = DEVELOPER_DIR; }; + 7263EE3713330E7500BA4D44 /* libz.dylib */ = {isa = PBXFileReference; lastKnownFileType = "compiled.mach-o.dylib"; name = libz.dylib; path = SDKs/MacOSX10.6.sdk/usr/lib/libz.dylib; sourceTree = DEVELOPER_DIR; }; + 7263EE3913330EC500BA4D44 /* libldap.dylib */ = {isa = PBXFileReference; lastKnownFileType = "compiled.mach-o.dylib"; name = libldap.dylib; path = SDKs/MacOSX10.6.sdk/usr/lib/libldap.dylib; sourceTree = DEVELOPER_DIR; }; + 726AD6F7135E88F0002C930D /* ippserver */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = ippserver; sourceTree = BUILT_PRODUCTS_DIR; }; + 726AD701135E8A90002C930D /* ippserver.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = ippserver.c; path = ../test/ippserver.c; sourceTree = ""; }; + 7271881713746EA8001A2036 /* commandtops.c */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c; name = commandtops.c; path = ../filter/commandtops.c; sourceTree = ""; }; + 7271881813746EA8001A2036 /* common.c */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c; name = common.c; path = ../filter/common.c; sourceTree = ""; }; + 7271881913746EA8001A2036 /* common.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; name = common.h; path = ../filter/common.h; sourceTree = ""; }; + 7271881A13746EA8001A2036 /* gziptoany.c */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c; name = gziptoany.c; path = ../filter/gziptoany.c; sourceTree = ""; }; + 7271882013746EA8001A2036 /* pstops.c */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c; name = pstops.c; path = ../filter/pstops.c; sourceTree = ""; }; + 7271882113746EA8001A2036 /* rastertoepson.c */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c; name = rastertoepson.c; path = ../filter/rastertoepson.c; sourceTree = ""; }; + 7271882213746EA8001A2036 /* rastertohp.c */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c; name = rastertohp.c; path = ../filter/rastertohp.c; sourceTree = ""; }; + 7271882313746EA8001A2036 /* rastertolabel.c */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c; name = rastertolabel.c; path = ../filter/rastertolabel.c; sourceTree = ""; }; + 7271883C1374AB14001A2036 /* mime-private.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = "mime-private.h"; path = "../scheduler/mime-private.h"; sourceTree = ""; }; + 72C16CB8137B195D007E4BF4 /* file.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = file.c; path = ../scheduler/file.c; sourceTree = SOURCE_ROOT; }; + 72F75A4C1336F31B004BB496 /* libcups_static.a */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.dylib"; includeInIndex = 0; path = libcups_static.a; sourceTree = BUILT_PRODUCTS_DIR; }; + 72F75A521336F950004BB496 /* cupstestppd */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = cupstestppd; sourceTree = BUILT_PRODUCTS_DIR; }; + 72F75A5B1336F988004BB496 /* cupstestppd.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = cupstestppd.c; path = ../systemv/cupstestppd.c; sourceTree = ""; }; + 72F75A611336F9A3004BB496 /* libcupsimage.dylib */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.dylib"; includeInIndex = 0; path = libcupsimage.dylib; sourceTree = BUILT_PRODUCTS_DIR; }; + 72F75A691336FA8A004BB496 /* error.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = error.c; path = ../filter/error.c; sourceTree = ""; }; + 72F75A6A1336FA8A004BB496 /* interpret.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = interpret.c; path = ../filter/interpret.c; sourceTree = ""; }; + 72F75A6B1336FA8A004BB496 /* raster.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; name = raster.c; path = ../filter/raster.c; sourceTree = ""; }; +/* End PBXFileReference section */ + +/* Begin PBXFrameworksBuildPhase section */ + 270CCDA4135E3C9E00007BE2 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 278C58E9136B64B000836530 /* CoreFoundation.framework in Frameworks */, + 278C58EA136B64B000836530 /* Kerberos.framework in Frameworks */, + 278C58EB136B64B000836530 /* Security.framework in Frameworks */, + 278C58EC136B64B000836530 /* SystemConfiguration.framework in Frameworks */, + 270CCDB9135E3D0900007BE2 /* libcups_static.a in Frameworks */, + 270CCDBA135E3D0900007BE2 /* libcupsmime.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 273BF6BA1333B5000022CAAB /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 278C58E4136B649200836530 /* libcups_static.a in Frameworks */, + 273BF6CE1333B5950022CAAB /* CoreFoundation.framework in Frameworks */, + 273BF6D31333B5C30022CAAB /* Kerberos.framework in Frameworks */, + 273BF6D41333B5C30022CAAB /* Security.framework in Frameworks */, + 273BF6D81333B5F60022CAAB /* SystemConfiguration.framework in Frameworks */, + 273BF6CF1333B5950022CAAB /* libiconv.dylib in Frameworks */, + 273BF6D71333B5F60022CAAB /* libresolv.dylib in Frameworks */, + 273BF6D01333B5950022CAAB /* libz.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 274FF5C913332B1F00317ECB /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 274FF5DD13332D0600317ECB /* libcups.dylib in Frameworks */, + 274FF6241333323B00317ECB /* libcupsppdc.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 274FF5EB133330C800317ECB /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 274FF6231333321400317ECB /* libcups.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 274FF6261333333600317ECB /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 274FF6321333334A00317ECB /* libcups.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 274FF63B1333358B00317ECB /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 274FF64C133339C400317ECB /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 274FF658133339D300317ECB /* libcups.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 274FF66013333A9B00317ECB /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 274FF66E13333AB500317ECB /* libcups.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 274FF67513333B2F00317ECB /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 274FF68513333B4300317ECB /* libcups.dylib in Frameworks */, + 274FF68613333B4300317ECB /* libcupsmime.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 274FF6B91333B1C400317ECB /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 274FF6BA1333B1C400317ECB /* CoreFoundation.framework in Frameworks */, + 274FF6BB1333B1C400317ECB /* Kerberos.framework in Frameworks */, + 274FF6BC1333B1C400317ECB /* Security.framework in Frameworks */, + 274FF6BD1333B1C400317ECB /* SystemConfiguration.framework in Frameworks */, + 274FF6BE1333B1C400317ECB /* libiconv.dylib in Frameworks */, + 274FF6BF1333B1C400317ECB /* libresolv.dylib in Frameworks */, + 274FF6C01333B1C400317ECB /* libz.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 276683591337A9B6000D33D0 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 276683671337A9E0000D33D0 /* libcups.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 2766836D1337AC79000D33D0 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 276683B11337AD06000D33D0 /* libcups.dylib in Frameworks */, + 276683B21337AD06000D33D0 /* libcupsppdc.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 2766837A1337AC8C000D33D0 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 276683B71337AD23000D33D0 /* libcups.dylib in Frameworks */, + 276683B81337AD23000D33D0 /* libcupsppdc.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 276683871337AC97000D33D0 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 276683B91337AD31000D33D0 /* libcups.dylib in Frameworks */, + 276683BA1337AD31000D33D0 /* libcupsppdc.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 276683941337ACA2000D33D0 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 276683C31337B1B3000D33D0 /* libcups.dylib in Frameworks */, + 276683C41337B1B3000D33D0 /* libcupsppdc.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 276683A11337ACAB000D33D0 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 276683C91337B1C1000D33D0 /* libcups.dylib in Frameworks */, + 276683CA1337B1C1000D33D0 /* libcupsppdc.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 276683ED1337F78E000D33D0 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 276683FD1337F7B8000D33D0 /* libcups.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 276684011337FA1D000D33D0 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 2766840F1337FA38000D33D0 /* libcups.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 278C58C8136B640300836530 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 278C58DE136B645C00836530 /* CoreFoundation.framework in Frameworks */, + 278C58DF136B645C00836530 /* Kerberos.framework in Frameworks */, + 278C58F6136B652300836530 /* Security.framework in Frameworks */, + 278C58E1136B645C00836530 /* SystemConfiguration.framework in Frameworks */, + 278C58D9136B645C00836530 /* libcups_static.a in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 720DD6BF1358FD5F0064AA82 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 720DD6CD1358FD720064AA82 /* libcups.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 72220EAB1333047D00FCA411 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 72220F4D133306BB00FCA411 /* CoreFoundation.framework in Frameworks */, + 72220F56133308EA00FCA411 /* Kerberos.framework in Frameworks */, + 72220F4F133306BB00FCA411 /* Security.framework in Frameworks */, + 72220F50133306BB00FCA411 /* SystemConfiguration.framework in Frameworks */, + 72220F52133308C100FCA411 /* libiconv.dylib in Frameworks */, + 72220F54133308CB00FCA411 /* libresolv.dylib in Frameworks */, + 72220F4E133306BB00FCA411 /* libz.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 72220F5813330A5A00FCA411 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 72220F6813330A8500FCA411 /* ApplicationServices.framework in Frameworks */, + 7263EE3613330E4E00BA4D44 /* CoreFoundation.framework in Frameworks */, + 7263EE3013330DC100BA4D44 /* IOKit.framework in Frameworks */, + 7263EE2C13330D5C00BA4D44 /* Kerberos.framework in Frameworks */, + 7263EE2D13330D5C00BA4D44 /* Security.framework in Frameworks */, + 7263EE2E13330D5C00BA4D44 /* SystemConfiguration.framework in Frameworks */, + 72220F6613330A7000FCA411 /* libcups.dylib in Frameworks */, + 72220FBF13330C1000FCA411 /* libcupsmime.dylib in Frameworks */, + 7263EE3A13330EC500BA4D44 /* libldap.dylib in Frameworks */, + 7263EE2713330D2800BA4D44 /* libpam.dylib in Frameworks */, + 7263EE3213330E1E00BA4D44 /* libpthread.dylib in Frameworks */, + 7263EE3413330E3C00BA4D44 /* libresolv.dylib in Frameworks */, + 7263EE3813330E7500BA4D44 /* libz.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 72220FA913330B2200FCA411 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 72220FBA13330BEE00FCA411 /* libcups.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 724378FA1333E43E009631B9 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 724379081333E4A5009631B9 /* libcups.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 724379151333E532009631B9 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 724379271333E93D009631B9 /* libcups.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 7243792D1333FB85009631B9 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 7243793B1333FB9D009631B9 /* libcups.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 724379441333FEA9009631B9 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 724379561333FF04009631B9 /* libcups.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 724379581333FF1D009631B9 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 724379C91333FFF3009631B9 /* CoreFoundation.framework in Frameworks */, + 724379681333FF3B009631B9 /* IOKit.framework in Frameworks */, + 724379661333FF3B009631B9 /* libcups.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 7258EADF134594C4009286F1 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 7258EAF413459B6D009286F1 /* libcups.dylib in Frameworks */, + 7258EAF513459B6D009286F1 /* libcupsimage.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 726AD6F4135E88F0002C930D /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 726AD70E135E8B5E002C930D /* libiconv.dylib in Frameworks */, + 726AD707135E8B11002C930D /* libcups_static.a in Frameworks */, + 726AD708135E8B11002C930D /* CoreFoundation.framework in Frameworks */, + 726AD709135E8B11002C930D /* Kerberos.framework in Frameworks */, + 726AD70A135E8B11002C930D /* libresolv.dylib in Frameworks */, + 726AD70B135E8B11002C930D /* libz.dylib in Frameworks */, + 726AD70C135E8B11002C930D /* Security.framework in Frameworks */, + 726AD70D135E8B11002C930D /* SystemConfiguration.framework in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 72F75A4F1336F950004BB496 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 276683E51337B2BE000D33D0 /* libcupsimage.dylib in Frameworks */, + 276683E21337B29C000D33D0 /* libcups.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 72F75A5E1336F9A3004BB496 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 72F75A671336FA38004BB496 /* libcups.dylib in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXFrameworksBuildPhase section */ + +/* Begin PBXGroup section */ + 273BF6B81333B4A90022CAAB /* tests */ = { + isa = PBXGroup; + children = ( + 273BF6C61333B5370022CAAB /* testcups.c */, + 278C58E2136B647200836530 /* testhttp.c */, + 270CCDBB135E3D3E00007BE2 /* testmime.c */, + ); + name = tests; + sourceTree = ""; + }; + 274FF5D513332C2C00317ECB /* daemon */ = { + isa = PBXGroup; + children = ( + 274FF66F13333ACF00317ECB /* cups-polld.c */, + 274FF6351333344400317ECB /* cups-deviced.c */, + 274FF5D613332CC700317ECB /* cups-driverd.cxx */, + 274FF6491333398D00317ECB /* cups-exec.c */, + 274FF65B133339FC00317ECB /* cups-lpd.c */, + 274FF5D713332CC700317ECB /* util.c */, + 274FF5D813332CC700317ECB /* util.h */, + ); + name = daemon; + sourceTree = ""; + }; + 274FF5F41333310400317ECB /* libcupsppdc */ = { + isa = PBXGroup; + children = ( + 274FF5F51333315100317ECB /* ppdc-array.cxx */, + 274FF5F61333315100317ECB /* ppdc-attr.cxx */, + 274FF5F71333315100317ECB /* ppdc-catalog.cxx */, + 274FF5F81333315100317ECB /* ppdc-choice.cxx */, + 274FF5F91333315100317ECB /* ppdc-constraint.cxx */, + 274FF5FA1333315100317ECB /* ppdc-driver.cxx */, + 274FF5FB1333315100317ECB /* ppdc-file.cxx */, + 274FF5FC1333315100317ECB /* ppdc-filter.cxx */, + 274FF5FD1333315100317ECB /* ppdc-font.cxx */, + 274FF5FE1333315100317ECB /* ppdc-group.cxx */, + 274FF5FF1333315100317ECB /* ppdc-import.cxx */, + 274FF6001333315100317ECB /* ppdc-mediasize.cxx */, + 274FF6011333315100317ECB /* ppdc-message.cxx */, + 274FF6021333315100317ECB /* ppdc-option.cxx */, + 274FF6031333315100317ECB /* ppdc-private.h */, + 274FF6041333315100317ECB /* ppdc-profile.cxx */, + 274FF6051333315100317ECB /* ppdc-shared.cxx */, + 274FF6061333315100317ECB /* ppdc-source.cxx */, + 274FF6071333315100317ECB /* ppdc-string.cxx */, + 274FF6081333315100317ECB /* ppdc-variable.cxx */, + ); + name = libcupsppdc; + sourceTree = ""; + }; + 274FF67313333B0A00317ECB /* commands */ = { + isa = PBXGroup; + children = ( + 2732E089137A3F5200FAFEF6 /* cancel.c */, + 2732E08A137A3F5200FAFEF6 /* cupsaccept.c */, + 276684101337FA7C000D33D0 /* cupsaddsmb.c */, + 276683681337AA00000D33D0 /* cupsctl.c */, + 274FF68713333B6E00317ECB /* cupsfilter.c */, + 2732E08B137A3F5200FAFEF6 /* cupstestdsc.c */, + 72F75A5B1336F988004BB496 /* cupstestppd.c */, + 726AD701135E8A90002C930D /* ippserver.c */, + 276683F91337F7A9000D33D0 /* ipptool.c */, + 2732E08C137A3F5200FAFEF6 /* lp.c */, + 2732E08D137A3F5200FAFEF6 /* lpadmin.c */, + 2732E08E137A3F5200FAFEF6 /* lpinfo.c */, + 2732E08F137A3F5200FAFEF6 /* lpmove.c */, + 2732E090137A3F5200FAFEF6 /* lpoptions.c */, + 2732E091137A3F5200FAFEF6 /* lppasswd.c */, + 2732E092137A3F5200FAFEF6 /* lpstat.c */, + ); + name = commands; + sourceTree = ""; + }; + 276683CB1337B1CC000D33D0 /* ppdc tools */ = { + isa = PBXGroup; + children = ( + 276683CC1337B201000D33D0 /* ppdc.cxx */, + 276683CE1337B20D000D33D0 /* ppdhtml.cxx */, + 276683D01337B21A000D33D0 /* ppdi.cxx */, + 276683D21337B228000D33D0 /* ppdmerge.cxx */, + 276683D41337B237000D33D0 /* ppdpo.cxx */, + ); + name = "ppdc tools"; + sourceTree = ""; + }; + 72220EAF1333047D00FCA411 /* Products */ = { + isa = PBXGroup; + children = ( + 72220F5B13330A5A00FCA411 /* cupsd */, + 274FF5CC13332B1F00317ECB /* cups-driverd */, + 274FF6291333333600317ECB /* cups-deviced */, + 274FF63E1333358B00317ECB /* cups-exec */, + 274FF64F133339C400317ECB /* cups-lpd */, + 274FF66313333A9B00317ECB /* cups-polld */, + 274FF67813333B2F00317ECB /* cupsfilter */, + 273BF6BD1333B5000022CAAB /* testcups */, + 724378FD1333E43E009631B9 /* ipp */, + 724379181333E532009631B9 /* lpd */, + 724379301333FB85009631B9 /* socket */, + 724379471333FEA9009631B9 /* dnssd */, + 7243795B1333FF1D009631B9 /* usb */, + 72F75A521336F950004BB496 /* cupstestppd */, + 2766835C1337A9B6000D33D0 /* cupsctl */, + 276683701337AC79000D33D0 /* ppdc */, + 2766837D1337AC8C000D33D0 /* ppdhtml */, + 2766838A1337AC97000D33D0 /* ppdi */, + 276683971337ACA2000D33D0 /* ppdmerge */, + 276683A41337ACAB000D33D0 /* ppdpo */, + 276683F01337F78E000D33D0 /* ipptool */, + 276684041337FA1D000D33D0 /* cupsaddsmb */, + 7258EAE2134594C4009286F1 /* rastertopwg */, + 720DD6C21358FD5F0064AA82 /* snmp */, + 270CCDA7135E3C9E00007BE2 /* testmime */, + 726AD6F7135E88F0002C930D /* ippserver */, + 278C58CB136B640300836530 /* testhttp */, + ); + name = Products; + sourceTree = ""; + }; + 72220EB41333050100FCA411 /* libcups */ = { + isa = PBXGroup; + children = ( + 276683561337A8C5000D33D0 /* cups.strings */, + 27D3037C134148CB00F022B1 /* libcups_s.exp */, + 27D3037D134148CB00F022B1 /* libcups2.def */, + 72220EB51333052D00FCA411 /* adminutil.c */, + 72220EB81333056300FCA411 /* array.c */, + 72220EBA1333056300FCA411 /* attr.c */, + 72220EBB1333056300FCA411 /* auth.c */, + 72220EBC1333056300FCA411 /* backchannel.c */, + 72220EBD1333056300FCA411 /* backend.c */, + 72220EBF1333056300FCA411 /* conflicts.c */, + 72220EC21333056300FCA411 /* custom.c */, + 72220ED1133305BB00FCA411 /* debug.c */, + 72220ED2133305BB00FCA411 /* dest.c */, + 72220ED3133305BB00FCA411 /* dir.c */, + 72220ED4133305BB00FCA411 /* dir.h */, + 72220ED5133305BB00FCA411 /* emit.c */, + 72220ED6133305BB00FCA411 /* encode.c */, + 72220ED8133305BB00FCA411 /* file.c */, + 72220EDA133305BB00FCA411 /* getdevices.c */, + 72220EDB133305BB00FCA411 /* getifaddrs.c */, + 72220EDC133305BB00FCA411 /* getputfile.c */, + 72220EDD133305BB00FCA411 /* globals.c */, + 72220EDE133305BB00FCA411 /* http-addr.c */, + 72220EDF133305BB00FCA411 /* http-addrlist.c */, + 72220EE1133305BB00FCA411 /* http-support.c */, + 72220EE2133305BB00FCA411 /* http.c */, + 72220EE5133305BB00FCA411 /* ipp-support.c */, + 72220EE6133305BB00FCA411 /* ipp.c */, + 72220EE8133305BB00FCA411 /* langprintf.c */, + 72220EEA133305BB00FCA411 /* language.c */, + 72220EEC133305BB00FCA411 /* localize.c */, + 72220EED133305BB00FCA411 /* mark.c */, + 72220EEF133305BB00FCA411 /* md5.c */, + 72220EF0133305BB00FCA411 /* md5passwd.c */, + 72220EF1133305BB00FCA411 /* notify.c */, + 72220EF2133305BB00FCA411 /* options.c */, + 72220EF3133305BB00FCA411 /* page.c */, + 72220EF4133305BB00FCA411 /* ppd-cache.c */, + 72220EF6133305BB00FCA411 /* ppd.c */, + 72220EF8133305BB00FCA411 /* pwg-media.c */, + 72220EFB133305BB00FCA411 /* request.c */, + 72220EFC133305BB00FCA411 /* sidechannel.c */, + 72220EFF133305BB00FCA411 /* snmp.c */, + 72220F00133305BB00FCA411 /* snprintf.c */, + 72220F02133305BB00FCA411 /* string.c */, + 72220F03133305BB00FCA411 /* tempfile.c */, + 72220F05133305BB00FCA411 /* thread.c */, + 72220F06133305BB00FCA411 /* transcode.c */, + 72220F08133305BB00FCA411 /* usersys.c */, + 72220F09133305BB00FCA411 /* util.c */, + ); + name = libcups; + sourceTree = ""; + }; + 72220F45133305D000FCA411 /* Public Headers */ = { + isa = PBXGroup; + children = ( + 72220EB71333056300FCA411 /* adminutil.h */, + 72220EB91333056300FCA411 /* array.h */, + 72220EBE1333056300FCA411 /* backend.h */, + 72220EC11333056300FCA411 /* cups.h */, + 72220ED9133305BB00FCA411 /* file.h */, + 72220EE3133305BB00FCA411 /* http.h */, + 72220EE7133305BB00FCA411 /* ipp.h */, + 72220EEB133305BB00FCA411 /* language.h */, + 72220FB413330BCE00FCA411 /* mime.h */, + 72220EF7133305BB00FCA411 /* ppd.h */, + 274FF6091333315100317ECB /* ppdc.h */, + 72220EFA133305BB00FCA411 /* raster.h */, + 72220EFD133305BB00FCA411 /* sidechannel.h */, + 72220F07133305BB00FCA411 /* transcode.h */, + 72220F0A133305BB00FCA411 /* versioning.h */, + ); + name = "Public Headers"; + sourceTree = ""; + }; + 72220F461333060C00FCA411 /* Private Headers */ = { + isa = PBXGroup; + children = ( + 72220F471333063D00FCA411 /* config.h */, + 7234F41F1378A16F00D3E9C9 /* array-private.h */, + 72220EC01333056300FCA411 /* cups-private.h */, + 72220EC31333056300FCA411 /* debug-private.h */, + 72220ED7133305BB00FCA411 /* file-private.h */, + 72220EE0133305BB00FCA411 /* http-private.h */, + 72220EE4133305BB00FCA411 /* ipp-private.h */, + 72220EE9133305BB00FCA411 /* language-private.h */, + 72220EEE133305BB00FCA411 /* md5-private.h */, + 7271883C1374AB14001A2036 /* mime-private.h */, + 72220EF5133305BB00FCA411 /* ppd-private.h */, + 72220EF9133305BB00FCA411 /* pwg-private.h */, + 72220EFE133305BB00FCA411 /* snmp-private.h */, + 72220F01133305BB00FCA411 /* string-private.h */, + 72220F04133305BB00FCA411 /* thread-private.h */, + ); + name = "Private Headers"; + sourceTree = ""; + }; + 72220F5D13330A5A00FCA411 /* cupsd */ = { + isa = PBXGroup; + children = ( + 72220F6913330B0C00FCA411 /* auth.c */, + 72220F6A13330B0C00FCA411 /* auth.h */, + 72220F6B13330B0C00FCA411 /* banners.c */, + 72220F6C13330B0C00FCA411 /* banners.h */, + 72220F6D13330B0C00FCA411 /* cert.c */, + 72220F6E13330B0C00FCA411 /* cert.h */, + 72220F6F13330B0C00FCA411 /* classes.c */, + 72220F7013330B0C00FCA411 /* classes.h */, + 72220F7113330B0C00FCA411 /* client.c */, + 72220F7213330B0C00FCA411 /* client.h */, + 72220F7313330B0C00FCA411 /* conf.c */, + 72220F7413330B0C00FCA411 /* conf.h */, + 72220F7513330B0C00FCA411 /* cupsd.h */, + 72220F7613330B0C00FCA411 /* dirsvc.c */, + 72220F7713330B0C00FCA411 /* dirsvc.h */, + 72220F7813330B0C00FCA411 /* env.c */, + 72C16CB8137B195D007E4BF4 /* file.c */, + 72220F7913330B0C00FCA411 /* ipp.c */, + 72220F7A13330B0C00FCA411 /* job.c */, + 72220F7B13330B0C00FCA411 /* job.h */, + 72220F7C13330B0C00FCA411 /* listen.c */, + 72220F7D13330B0C00FCA411 /* log.c */, + 72220F7E13330B0C00FCA411 /* main.c */, + 72220F7F13330B0C00FCA411 /* network.c */, + 72220F8013330B0C00FCA411 /* network.h */, + 72220F8113330B0C00FCA411 /* policy.c */, + 72220F8213330B0C00FCA411 /* policy.h */, + 72220F8313330B0C00FCA411 /* printers.c */, + 72220F8413330B0C00FCA411 /* printers.h */, + 72220F8513330B0C00FCA411 /* process.c */, + 72220F8613330B0C00FCA411 /* quotas.c */, + 72220F8813330B0C00FCA411 /* select.c */, + 72220F8913330B0C00FCA411 /* server.c */, + 72220F8A13330B0C00FCA411 /* statbuf.c */, + 72220F8B13330B0C00FCA411 /* statbuf.h */, + 72220F8C13330B0C00FCA411 /* subscriptions.c */, + 72220F8D13330B0C00FCA411 /* subscriptions.h */, + 72220F8E13330B0C00FCA411 /* sysman.c */, + 72220F8F13330B0C00FCA411 /* sysman.h */, + ); + name = cupsd; + path = .; + sourceTree = ""; + }; + 72220FB013330B3400FCA411 /* libcupsmime */ = { + isa = PBXGroup; + children = ( + 72220FB213330BCE00FCA411 /* filter.c */, + 72220FB313330BCE00FCA411 /* mime.c */, + 72220FB513330BCE00FCA411 /* type.c */, + ); + name = libcupsmime; + sourceTree = ""; + }; + 72220FB113330B4A00FCA411 /* Frameworks */ = { + isa = PBXGroup; + children = ( + 278C58F4136B652300836530 /* Security.framework */, + 278C58E5136B64AF00836530 /* CoreFoundation.framework */, + 278C58E6136B64B000836530 /* Kerberos.framework */, + 278C58E7136B64B000836530 /* Security.framework */, + 278C58E8136B64B000836530 /* SystemConfiguration.framework */, + 278C58DA136B645C00836530 /* CoreFoundation.framework */, + 278C58DB136B645C00836530 /* Kerberos.framework */, + 278C58DD136B645C00836530 /* SystemConfiguration.framework */, + 72220FAC13330B2200FCA411 /* libcupsmime.dylib */, + 72220EAE1333047D00FCA411 /* libcups.dylib */, + 72F75A611336F9A3004BB496 /* libcupsimage.dylib */, + 274FF5EE133330C800317ECB /* libcupsppdc.dylib */, + 724379C81333FFF3009631B9 /* CoreFoundation.framework */, + 724379671333FF3B009631B9 /* IOKit.framework */, + 273BF6D51333B5F60022CAAB /* libresolv.dylib */, + 273BF6D61333B5F60022CAAB /* SystemConfiguration.framework */, + 273BF6D11333B5C30022CAAB /* Kerberos.framework */, + 273BF6D21333B5C30022CAAB /* Security.framework */, + 273BF6CB1333B5950022CAAB /* CoreFoundation.framework */, + 273BF6CC1333B5950022CAAB /* libiconv.dylib */, + 273BF6CD1333B5950022CAAB /* libz.dylib */, + 7263EE3913330EC500BA4D44 /* libldap.dylib */, + 7263EE3713330E7500BA4D44 /* libz.dylib */, + 7263EE3513330E4E00BA4D44 /* CoreFoundation.framework */, + 7263EE3313330E3C00BA4D44 /* libresolv.dylib */, + 7263EE3113330E1E00BA4D44 /* libpthread.dylib */, + 7263EE2F13330DC100BA4D44 /* IOKit.framework */, + 7263EE2913330D5C00BA4D44 /* Kerberos.framework */, + 7263EE2A13330D5C00BA4D44 /* Security.framework */, + 7263EE2B13330D5C00BA4D44 /* SystemConfiguration.framework */, + 7263EE2613330D2800BA4D44 /* libpam.dylib */, + 72220F6713330A8500FCA411 /* ApplicationServices.framework */, + 72220F49133306BB00FCA411 /* CoreFoundation.framework */, + 72220F55133308EA00FCA411 /* Kerberos.framework */, + 72220F4B133306BB00FCA411 /* Security.framework */, + 72220F4C133306BB00FCA411 /* SystemConfiguration.framework */, + 72220F51133308C100FCA411 /* libiconv.dylib */, + 72220F53133308CB00FCA411 /* libresolv.dylib */, + 72220F4A133306BB00FCA411 /* libz.dylib */, + ); + name = Frameworks; + sourceTree = ""; + }; + 724378F71333E3CE009631B9 /* backends */ = { + isa = PBXGroup; + children = ( + 724379091333E4E3009631B9 /* backend-private.h */, + 724379501333FEBB009631B9 /* dnssd.c */, + 724379CA1334000E009631B9 /* ieee1284.c */, + 7243790A1333E4E3009631B9 /* ipp.c */, + 724379281333E952009631B9 /* lpd.c */, + 7243790B1333E4E3009631B9 /* network.c */, + 724379121333E516009631B9 /* runloop.c */, + 720DD6D21358FDDE0064AA82 /* snmp.c */, + 7243790C1333E4E3009631B9 /* snmp-supplies.c */, + 7243793C1333FD19009631B9 /* socket.c */, + 724379C51333FFC7009631B9 /* usb.c */, + 724379C41333FFC7009631B9 /* usb-darwin.c */, + ); + name = backends; + sourceTree = ""; + }; + 7258EADC134594A8009286F1 /* filters */ = { + isa = PBXGroup; + children = ( + 7271881713746EA8001A2036 /* commandtops.c */, + 7271881813746EA8001A2036 /* common.c */, + 7271881913746EA8001A2036 /* common.h */, + 7271881A13746EA8001A2036 /* gziptoany.c */, + 7271882013746EA8001A2036 /* pstops.c */, + 7271882113746EA8001A2036 /* rastertoepson.c */, + 7271882213746EA8001A2036 /* rastertohp.c */, + 7271882313746EA8001A2036 /* rastertolabel.c */, + 7258EAEC134594EB009286F1 /* rastertopwg.c */, + ); + name = filters; + sourceTree = ""; + }; + 72BF96351333042100B1EAD7 = { + isa = PBXGroup; + children = ( + 72F75A4C1336F31B004BB496 /* libcups_static.a */, + 72220FB113330B4A00FCA411 /* Frameworks */, + 72220F45133305D000FCA411 /* Public Headers */, + 72220F461333060C00FCA411 /* Private Headers */, + 72220EB41333050100FCA411 /* libcups */, + 72F75A681336FA42004BB496 /* libcupsimage */, + 72220FB013330B3400FCA411 /* libcupsmime */, + 274FF5F41333310400317ECB /* libcupsppdc */, + 724378F71333E3CE009631B9 /* backends */, + 274FF67313333B0A00317ECB /* commands */, + 72220F5D13330A5A00FCA411 /* cupsd */, + 274FF5D513332C2C00317ECB /* daemon */, + 7258EADC134594A8009286F1 /* filters */, + 276683CB1337B1CC000D33D0 /* ppdc tools */, + 273BF6B81333B4A90022CAAB /* tests */, + 72220EAF1333047D00FCA411 /* Products */, + ); + sourceTree = ""; + }; + 72F75A681336FA42004BB496 /* libcupsimage */ = { + isa = PBXGroup; + children = ( + 72F75A691336FA8A004BB496 /* error.c */, + 72F75A6A1336FA8A004BB496 /* interpret.c */, + 72F75A6B1336FA8A004BB496 /* raster.c */, + ); + name = libcupsimage; + sourceTree = ""; + }; +/* End PBXGroup section */ + +/* Begin PBXHeadersBuildPhase section */ + 274FF5EC133330C800317ECB /* Headers */ = { + isa = PBXHeadersBuildPhase; + buildActionMask = 2147483647; + files = ( + 274FF61E1333315100317ECB /* ppdc.h in Headers */, + 274FF6181333315100317ECB /* ppdc-private.h in Headers */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 274FF6C11333B1C400317ECB /* Headers */ = { + isa = PBXHeadersBuildPhase; + buildActionMask = 2147483647; + files = ( + 274FF6C21333B1C400317ECB /* adminutil.h in Headers */, + 274FF6C31333B1C400317ECB /* array.h in Headers */, + 274FF6C41333B1C400317ECB /* backend.h in Headers */, + 274FF6D01333B1C400317ECB /* cups-private.h in Headers */, + 274FF6D11333B1C400317ECB /* debug-private.h in Headers */, + 274FF6D21333B1C400317ECB /* file-private.h in Headers */, + 274FF6D31333B1C400317ECB /* http-private.h in Headers */, + 274FF6D41333B1C400317ECB /* ipp-private.h in Headers */, + 274FF6D51333B1C400317ECB /* language-private.h in Headers */, + 274FF6D61333B1C400317ECB /* md5-private.h in Headers */, + 274FF6D71333B1C400317ECB /* ppd-private.h in Headers */, + 274FF6D81333B1C400317ECB /* pwg-private.h in Headers */, + 274FF6D91333B1C400317ECB /* snmp-private.h in Headers */, + 274FF6DA1333B1C400317ECB /* string-private.h in Headers */, + 274FF6DB1333B1C400317ECB /* thread-private.h in Headers */, + 274FF6DC1333B1C400317ECB /* config.h in Headers */, + 274FF6C51333B1C400317ECB /* cups.h in Headers */, + 274FF6C61333B1C400317ECB /* dir.h in Headers */, + 274FF6C71333B1C400317ECB /* file.h in Headers */, + 274FF6C81333B1C400317ECB /* http.h in Headers */, + 274FF6C91333B1C400317ECB /* ipp.h in Headers */, + 274FF6CA1333B1C400317ECB /* language.h in Headers */, + 274FF6CB1333B1C400317ECB /* ppd.h in Headers */, + 274FF6CD1333B1C400317ECB /* sidechannel.h in Headers */, + 274FF6CE1333B1C400317ECB /* transcode.h in Headers */, + 274FF6CF1333B1C400317ECB /* versioning.h in Headers */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 72220EAC1333047D00FCA411 /* Headers */ = { + isa = PBXHeadersBuildPhase; + buildActionMask = 2147483647; + files = ( + 72220EC41333056300FCA411 /* adminutil.h in Headers */, + 72220EC61333056300FCA411 /* array.h in Headers */, + 72220ECB1333056300FCA411 /* backend.h in Headers */, + 72220ECE1333056300FCA411 /* cups.h in Headers */, + 72220F0E133305BB00FCA411 /* dir.h in Headers */, + 72220F13133305BB00FCA411 /* file.h in Headers */, + 72220F1D133305BB00FCA411 /* http.h in Headers */, + 72220F21133305BB00FCA411 /* ipp.h in Headers */, + 72220F25133305BB00FCA411 /* language.h in Headers */, + 72220F31133305BB00FCA411 /* ppd.h in Headers */, + 72220F37133305BB00FCA411 /* sidechannel.h in Headers */, + 72220F41133305BB00FCA411 /* transcode.h in Headers */, + 72220F44133305BB00FCA411 /* versioning.h in Headers */, + 72220ECD1333056300FCA411 /* cups-private.h in Headers */, + 72220ED01333056300FCA411 /* debug-private.h in Headers */, + 72220F11133305BB00FCA411 /* file-private.h in Headers */, + 72220F1A133305BB00FCA411 /* http-private.h in Headers */, + 72220F1E133305BB00FCA411 /* ipp-private.h in Headers */, + 72220F23133305BB00FCA411 /* language-private.h in Headers */, + 72220F28133305BB00FCA411 /* md5-private.h in Headers */, + 72220F2F133305BB00FCA411 /* ppd-private.h in Headers */, + 72220F33133305BB00FCA411 /* pwg-private.h in Headers */, + 72220F38133305BB00FCA411 /* snmp-private.h in Headers */, + 72220F3B133305BB00FCA411 /* string-private.h in Headers */, + 72220F3E133305BB00FCA411 /* thread-private.h in Headers */, + 72220F481333063D00FCA411 /* config.h in Headers */, + 7234F4201378A16F00D3E9C9 /* array-private.h in Headers */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 72220FAA13330B2200FCA411 /* Headers */ = { + isa = PBXHeadersBuildPhase; + buildActionMask = 2147483647; + files = ( + 72220FB813330BCE00FCA411 /* mime.h in Headers */, + 7271883D1374AB14001A2036 /* mime-private.h in Headers */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 72F75A5F1336F9A3004BB496 /* Headers */ = { + isa = PBXHeadersBuildPhase; + buildActionMask = 2147483647; + files = ( + 72F75A6F1336FAB6004BB496 /* raster.h in Headers */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXHeadersBuildPhase section */ + +/* Begin PBXNativeTarget section */ + 270CCDA6135E3C9E00007BE2 /* testmime */ = { + isa = PBXNativeTarget; + buildConfigurationList = 270CCDAF135E3C9E00007BE2 /* Build configuration list for PBXNativeTarget "testmime" */; + buildPhases = ( + 270CCDA3135E3C9E00007BE2 /* Sources */, + 270CCDA4135E3C9E00007BE2 /* Frameworks */, + 270CCDA5135E3C9E00007BE2 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 270CCDB8135E3CFD00007BE2 /* PBXTargetDependency */, + 270CCDB6135E3CF700007BE2 /* PBXTargetDependency */, + ); + name = testmime; + productName = testmime; + productReference = 270CCDA7135E3C9E00007BE2 /* testmime */; + productType = "com.apple.product-type.tool"; + }; + 273BF6BC1333B5000022CAAB /* testcups */ = { + isa = PBXNativeTarget; + buildConfigurationList = 273BF6C31333B5000022CAAB /* Build configuration list for PBXNativeTarget "testcups" */; + buildPhases = ( + 273BF6B91333B5000022CAAB /* Sources */, + 273BF6BA1333B5000022CAAB /* Frameworks */, + 273BF6BB1333B5000022CAAB /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 273BF6C91333B5410022CAAB /* PBXTargetDependency */, + ); + name = testcups; + productName = testcups; + productReference = 273BF6BD1333B5000022CAAB /* testcups */; + productType = "com.apple.product-type.tool"; + }; + 274FF5CB13332B1F00317ECB /* cups-driverd */ = { + isa = PBXNativeTarget; + buildConfigurationList = 274FF5D213332B1F00317ECB /* Build configuration list for PBXNativeTarget "cups-driverd" */; + buildPhases = ( + 274FF5C813332B1F00317ECB /* Sources */, + 274FF5C913332B1F00317ECB /* Frameworks */, + 274FF5CA13332B1F00317ECB /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 274FF5DC13332CF900317ECB /* PBXTargetDependency */, + 274FF6201333316200317ECB /* PBXTargetDependency */, + ); + name = "cups-driverd"; + productName = "cups-driverd"; + productReference = 274FF5CC13332B1F00317ECB /* cups-driverd */; + productType = "com.apple.product-type.tool"; + }; + 274FF5ED133330C800317ECB /* libcupsppdc */ = { + isa = PBXNativeTarget; + buildConfigurationList = 274FF5EF133330C800317ECB /* Build configuration list for PBXNativeTarget "libcupsppdc" */; + buildPhases = ( + 274FF5EA133330C800317ECB /* Sources */, + 274FF5EB133330C800317ECB /* Frameworks */, + 274FF5EC133330C800317ECB /* Headers */, + ); + buildRules = ( + ); + dependencies = ( + 274FF5F3133330FD00317ECB /* PBXTargetDependency */, + ); + name = libcupsppdc; + productName = libcupsppdc; + productReference = 274FF5EE133330C800317ECB /* libcupsppdc.dylib */; + productType = "com.apple.product-type.library.dynamic"; + }; + 274FF6281333333600317ECB /* cups-deviced */ = { + isa = PBXNativeTarget; + buildConfigurationList = 274FF62F1333333600317ECB /* Build configuration list for PBXNativeTarget "cups-deviced" */; + buildPhases = ( + 274FF6251333333600317ECB /* Sources */, + 274FF6261333333600317ECB /* Frameworks */, + 274FF6271333333600317ECB /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 274FF6341333335200317ECB /* PBXTargetDependency */, + ); + name = "cups-deviced"; + productName = "cups-deviced"; + productReference = 274FF6291333333600317ECB /* cups-deviced */; + productType = "com.apple.product-type.tool"; + }; + 274FF63D1333358B00317ECB /* cups-exec */ = { + isa = PBXNativeTarget; + buildConfigurationList = 274FF6441333358C00317ECB /* Build configuration list for PBXNativeTarget "cups-exec" */; + buildPhases = ( + 274FF63A1333358B00317ECB /* Sources */, + 274FF63B1333358B00317ECB /* Frameworks */, + 274FF63C1333358B00317ECB /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + ); + name = "cups-exec"; + productName = "cups-exec"; + productReference = 274FF63E1333358B00317ECB /* cups-exec */; + productType = "com.apple.product-type.tool"; + }; + 274FF64E133339C400317ECB /* cups-lpd */ = { + isa = PBXNativeTarget; + buildConfigurationList = 274FF655133339C400317ECB /* Build configuration list for PBXNativeTarget "cups-lpd" */; + buildPhases = ( + 274FF64B133339C400317ECB /* Sources */, + 274FF64C133339C400317ECB /* Frameworks */, + 274FF64D133339C400317ECB /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 274FF65A133339D900317ECB /* PBXTargetDependency */, + ); + name = "cups-lpd"; + productName = "cups-lpd"; + productReference = 274FF64F133339C400317ECB /* cups-lpd */; + productType = "com.apple.product-type.tool"; + }; + 274FF66213333A9B00317ECB /* cups-polld */ = { + isa = PBXNativeTarget; + buildConfigurationList = 274FF66913333A9B00317ECB /* Build configuration list for PBXNativeTarget "cups-polld" */; + buildPhases = ( + 274FF65F13333A9B00317ECB /* Sources */, + 274FF66013333A9B00317ECB /* Frameworks */, + 274FF66113333A9B00317ECB /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 274FF66D13333AAD00317ECB /* PBXTargetDependency */, + ); + name = "cups-polld"; + productName = "cups-polld"; + productReference = 274FF66313333A9B00317ECB /* cups-polld */; + productType = "com.apple.product-type.tool"; + }; + 274FF67713333B2F00317ECB /* cupsfilter */ = { + isa = PBXNativeTarget; + buildConfigurationList = 274FF67E13333B2F00317ECB /* Build configuration list for PBXNativeTarget "cupsfilter" */; + buildPhases = ( + 274FF67413333B2F00317ECB /* Sources */, + 274FF67513333B2F00317ECB /* Frameworks */, + 274FF67613333B2F00317ECB /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 274FF68213333B3C00317ECB /* PBXTargetDependency */, + 274FF68413333B3C00317ECB /* PBXTargetDependency */, + ); + name = cupsfilter; + productName = cupsfilter; + productReference = 274FF67813333B2F00317ECB /* cupsfilter */; + productType = "com.apple.product-type.tool"; + }; + 274FF6891333B1C400317ECB /* libcups_static */ = { + isa = PBXNativeTarget; + buildConfigurationList = 274FF6DD1333B1C400317ECB /* Build configuration list for PBXNativeTarget "libcups_static" */; + buildPhases = ( + 274FF68A1333B1C400317ECB /* Sources */, + 274FF6B91333B1C400317ECB /* Frameworks */, + 274FF6C11333B1C400317ECB /* Headers */, + ); + buildRules = ( + ); + dependencies = ( + ); + name = libcups_static; + productName = libcups; + productReference = 72F75A4C1336F31B004BB496 /* libcups_static.a */; + productType = "com.apple.product-type.library.dynamic"; + }; + 2766835B1337A9B6000D33D0 /* cupsctl */ = { + isa = PBXNativeTarget; + buildConfigurationList = 276683621337A9B6000D33D0 /* Build configuration list for PBXNativeTarget "cupsctl" */; + buildPhases = ( + 276683581337A9B6000D33D0 /* Sources */, + 276683591337A9B6000D33D0 /* Frameworks */, + 2766835A1337A9B6000D33D0 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 276683661337A9D6000D33D0 /* PBXTargetDependency */, + ); + name = cupsctl; + productName = cupsctl; + productReference = 2766835C1337A9B6000D33D0 /* cupsctl */; + productType = "com.apple.product-type.tool"; + }; + 2766836F1337AC79000D33D0 /* ppdc */ = { + isa = PBXNativeTarget; + buildConfigurationList = 276683761337AC79000D33D0 /* Build configuration list for PBXNativeTarget "ppdc" */; + buildPhases = ( + 2766836C1337AC79000D33D0 /* Sources */, + 2766836D1337AC79000D33D0 /* Frameworks */, + 2766836E1337AC79000D33D0 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 276683AE1337ACF9000D33D0 /* PBXTargetDependency */, + 276683B01337ACF9000D33D0 /* PBXTargetDependency */, + ); + name = ppdc; + productName = ppdc; + productReference = 276683701337AC79000D33D0 /* ppdc */; + productType = "com.apple.product-type.tool"; + }; + 2766837C1337AC8C000D33D0 /* ppdhtml */ = { + isa = PBXNativeTarget; + buildConfigurationList = 276683831337AC8C000D33D0 /* Build configuration list for PBXNativeTarget "ppdhtml" */; + buildPhases = ( + 276683791337AC8C000D33D0 /* Sources */, + 2766837A1337AC8C000D33D0 /* Frameworks */, + 2766837B1337AC8C000D33D0 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 276683B41337AD18000D33D0 /* PBXTargetDependency */, + 276683B61337AD18000D33D0 /* PBXTargetDependency */, + ); + name = ppdhtml; + productName = ppdhtml; + productReference = 2766837D1337AC8C000D33D0 /* ppdhtml */; + productType = "com.apple.product-type.tool"; + }; + 276683891337AC97000D33D0 /* ppdi */ = { + isa = PBXNativeTarget; + buildConfigurationList = 276683901337AC97000D33D0 /* Build configuration list for PBXNativeTarget "ppdi" */; + buildPhases = ( + 276683861337AC97000D33D0 /* Sources */, + 276683871337AC97000D33D0 /* Frameworks */, + 276683881337AC97000D33D0 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 276683BC1337AE49000D33D0 /* PBXTargetDependency */, + 276683BE1337AE49000D33D0 /* PBXTargetDependency */, + ); + name = ppdi; + productName = ppdi; + productReference = 2766838A1337AC97000D33D0 /* ppdi */; + productType = "com.apple.product-type.tool"; + }; + 276683961337ACA2000D33D0 /* ppdmerge */ = { + isa = PBXNativeTarget; + buildConfigurationList = 2766839D1337ACA2000D33D0 /* Build configuration list for PBXNativeTarget "ppdmerge" */; + buildPhases = ( + 276683931337ACA2000D33D0 /* Sources */, + 276683941337ACA2000D33D0 /* Frameworks */, + 276683951337ACA2000D33D0 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 276683C01337B1AD000D33D0 /* PBXTargetDependency */, + 276683C21337B1AD000D33D0 /* PBXTargetDependency */, + ); + name = ppdmerge; + productName = ppdmerge; + productReference = 276683971337ACA2000D33D0 /* ppdmerge */; + productType = "com.apple.product-type.tool"; + }; + 276683A31337ACAB000D33D0 /* ppdpo */ = { + isa = PBXNativeTarget; + buildConfigurationList = 276683AA1337ACAB000D33D0 /* Build configuration list for PBXNativeTarget "ppdpo" */; + buildPhases = ( + 276683A01337ACAB000D33D0 /* Sources */, + 276683A11337ACAB000D33D0 /* Frameworks */, + 276683A21337ACAB000D33D0 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 276683C61337B1BC000D33D0 /* PBXTargetDependency */, + 276683C81337B1BC000D33D0 /* PBXTargetDependency */, + ); + name = ppdpo; + productName = ppdpo; + productReference = 276683A41337ACAB000D33D0 /* ppdpo */; + productType = "com.apple.product-type.tool"; + }; + 276683EF1337F78E000D33D0 /* ipptool */ = { + isa = PBXNativeTarget; + buildConfigurationList = 276683F61337F78F000D33D0 /* Build configuration list for PBXNativeTarget "ipptool" */; + buildPhases = ( + 276683EC1337F78E000D33D0 /* Sources */, + 276683ED1337F78E000D33D0 /* Frameworks */, + 276683EE1337F78E000D33D0 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 276683FC1337F7B3000D33D0 /* PBXTargetDependency */, + ); + name = ipptool; + productName = ipptool; + productReference = 276683F01337F78E000D33D0 /* ipptool */; + productType = "com.apple.product-type.tool"; + }; + 276684031337FA1D000D33D0 /* cupsaddsmb */ = { + isa = PBXNativeTarget; + buildConfigurationList = 2766840A1337FA1E000D33D0 /* Build configuration list for PBXNativeTarget "cupsaddsmb" */; + buildPhases = ( + 276684001337FA1D000D33D0 /* Sources */, + 276684011337FA1D000D33D0 /* Frameworks */, + 276684021337FA1D000D33D0 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 2766840E1337FA31000D33D0 /* PBXTargetDependency */, + ); + name = cupsaddsmb; + productName = cupsaddsmb; + productReference = 276684041337FA1D000D33D0 /* cupsaddsmb */; + productType = "com.apple.product-type.tool"; + }; + 278C58CA136B640300836530 /* testhttp */ = { + isa = PBXNativeTarget; + buildConfigurationList = 278C58D3136B640300836530 /* Build configuration list for PBXNativeTarget "testhttp" */; + buildPhases = ( + 278C58C7136B640300836530 /* Sources */, + 278C58C8136B640300836530 /* Frameworks */, + 278C58C9136B640300836530 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 278C58D8136B642F00836530 /* PBXTargetDependency */, + ); + name = testhttp; + productName = testhttp; + productReference = 278C58CB136B640300836530 /* testhttp */; + productType = "com.apple.product-type.tool"; + }; + 720DD6C11358FD5F0064AA82 /* snmp */ = { + isa = PBXNativeTarget; + buildConfigurationList = 720DD6CB1358FD600064AA82 /* Build configuration list for PBXNativeTarget "snmp" */; + buildPhases = ( + 720DD6BE1358FD5F0064AA82 /* Sources */, + 720DD6BF1358FD5F0064AA82 /* Frameworks */, + 720DD6C01358FD5F0064AA82 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 720DD6CF1358FD790064AA82 /* PBXTargetDependency */, + ); + name = snmp; + productName = snmp; + productReference = 720DD6C21358FD5F0064AA82 /* snmp */; + productType = "com.apple.product-type.tool"; + }; + 72220EAD1333047D00FCA411 /* libcups */ = { + isa = PBXNativeTarget; + buildConfigurationList = 72220EB21333047D00FCA411 /* Build configuration list for PBXNativeTarget "libcups" */; + buildPhases = ( + 72220EAA1333047D00FCA411 /* Sources */, + 72220EAB1333047D00FCA411 /* Frameworks */, + 72220EAC1333047D00FCA411 /* Headers */, + ); + buildRules = ( + ); + dependencies = ( + ); + name = libcups; + productName = libcups; + productReference = 72220EAE1333047D00FCA411 /* libcups.dylib */; + productType = "com.apple.product-type.library.dynamic"; + }; + 72220F5A13330A5A00FCA411 /* cupsd */ = { + isa = PBXNativeTarget; + buildConfigurationList = 72220F6113330A5A00FCA411 /* Build configuration list for PBXNativeTarget "cupsd" */; + buildPhases = ( + 72220F5713330A5A00FCA411 /* Sources */, + 72220F5813330A5A00FCA411 /* Frameworks */, + 72220F5913330A5A00FCA411 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 72220FBE13330C0B00FCA411 /* PBXTargetDependency */, + 72220F6513330A6500FCA411 /* PBXTargetDependency */, + ); + name = cupsd; + productName = cupsd; + productReference = 72220F5B13330A5A00FCA411 /* cupsd */; + productType = "com.apple.product-type.tool"; + }; + 72220FAB13330B2200FCA411 /* libcupsmime */ = { + isa = PBXNativeTarget; + buildConfigurationList = 72220FAD13330B2300FCA411 /* Build configuration list for PBXNativeTarget "libcupsmime" */; + buildPhases = ( + 72220FA813330B2200FCA411 /* Sources */, + 72220FA913330B2200FCA411 /* Frameworks */, + 72220FAA13330B2200FCA411 /* Headers */, + ); + buildRules = ( + ); + dependencies = ( + 72220FBC13330C0500FCA411 /* PBXTargetDependency */, + ); + name = libcupsmime; + productName = libcupsmime; + productReference = 72220FAC13330B2200FCA411 /* libcupsmime.dylib */; + productType = "com.apple.product-type.library.dynamic"; + }; + 724378FC1333E43E009631B9 /* ipp */ = { + isa = PBXNativeTarget; + buildConfigurationList = 724379031333E43E009631B9 /* Build configuration list for PBXNativeTarget "ipp" */; + buildPhases = ( + 724378F91333E43E009631B9 /* Sources */, + 724378FA1333E43E009631B9 /* Frameworks */, + 724378FB1333E43E009631B9 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 724379071333E49B009631B9 /* PBXTargetDependency */, + ); + name = ipp; + productName = ipp; + productReference = 724378FD1333E43E009631B9 /* ipp */; + productType = "com.apple.product-type.tool"; + }; + 724379171333E532009631B9 /* lpd */ = { + isa = PBXNativeTarget; + buildConfigurationList = 7243791E1333E532009631B9 /* Build configuration list for PBXNativeTarget "lpd" */; + buildPhases = ( + 724379141333E532009631B9 /* Sources */, + 724379151333E532009631B9 /* Frameworks */, + 724379161333E532009631B9 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 724379261333E932009631B9 /* PBXTargetDependency */, + ); + name = lpd; + productName = lpd; + productReference = 724379181333E532009631B9 /* lpd */; + productType = "com.apple.product-type.tool"; + }; + 7243792F1333FB85009631B9 /* socket */ = { + isa = PBXNativeTarget; + buildConfigurationList = 724379361333FB85009631B9 /* Build configuration list for PBXNativeTarget "socket" */; + buildPhases = ( + 7243792C1333FB85009631B9 /* Sources */, + 7243792D1333FB85009631B9 /* Frameworks */, + 7243792E1333FB85009631B9 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 7243793A1333FB95009631B9 /* PBXTargetDependency */, + ); + name = socket; + productName = socket; + productReference = 724379301333FB85009631B9 /* socket */; + productType = "com.apple.product-type.tool"; + }; + 724379461333FEA9009631B9 /* dnssd */ = { + isa = PBXNativeTarget; + buildConfigurationList = 7243794D1333FEA9009631B9 /* Build configuration list for PBXNativeTarget "dnssd" */; + buildPhases = ( + 724379431333FEA9009631B9 /* Sources */, + 724379441333FEA9009631B9 /* Frameworks */, + 724379451333FEA9009631B9 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 724379551333FEFE009631B9 /* PBXTargetDependency */, + ); + name = dnssd; + productName = dnssd; + productReference = 724379471333FEA9009631B9 /* dnssd */; + productType = "com.apple.product-type.tool"; + }; + 7243795A1333FF1D009631B9 /* usb */ = { + isa = PBXNativeTarget; + buildConfigurationList = 724379611333FF1D009631B9 /* Build configuration list for PBXNativeTarget "usb" */; + buildPhases = ( + 724379571333FF1D009631B9 /* Sources */, + 724379581333FF1D009631B9 /* Frameworks */, + 724379591333FF1D009631B9 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 724379651333FF2E009631B9 /* PBXTargetDependency */, + ); + name = usb; + productName = usb; + productReference = 7243795B1333FF1D009631B9 /* usb */; + productType = "com.apple.product-type.tool"; + }; + 7258EAE1134594C4009286F1 /* rastertopwg */ = { + isa = PBXNativeTarget; + buildConfigurationList = 7258EAE9134594C4009286F1 /* Build configuration list for PBXNativeTarget "rastertopwg" */; + buildPhases = ( + 7258EADE134594C4009286F1 /* Sources */, + 7258EADF134594C4009286F1 /* Frameworks */, + 7258EAE0134594C4009286F1 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 7258EAF113459B67009286F1 /* PBXTargetDependency */, + 7258EAF313459B67009286F1 /* PBXTargetDependency */, + ); + name = rastertopwg; + productName = rastertopwg; + productReference = 7258EAE2134594C4009286F1 /* rastertopwg */; + productType = "com.apple.product-type.tool"; + }; + 726AD6F6135E88F0002C930D /* ippserver */ = { + isa = PBXNativeTarget; + buildConfigurationList = 726AD6FE135E88F1002C930D /* Build configuration list for PBXNativeTarget "ippserver" */; + buildPhases = ( + 726AD6F3135E88F0002C930D /* Sources */, + 726AD6F4135E88F0002C930D /* Frameworks */, + 726AD6F5135E88F0002C930D /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 726AD706135E8AC5002C930D /* PBXTargetDependency */, + ); + name = ippserver; + productName = ippserver; + productReference = 726AD6F7135E88F0002C930D /* ippserver */; + productType = "com.apple.product-type.tool"; + }; + 72F75A511336F950004BB496 /* cupstestppd */ = { + isa = PBXNativeTarget; + buildConfigurationList = 72F75A581336F951004BB496 /* Build configuration list for PBXNativeTarget "cupstestppd" */; + buildPhases = ( + 72F75A4E1336F950004BB496 /* Sources */, + 72F75A4F1336F950004BB496 /* Frameworks */, + 72F75A501336F950004BB496 /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + 276683E41337B2BA000D33D0 /* PBXTargetDependency */, + 276683E11337B299000D33D0 /* PBXTargetDependency */, + ); + name = cupstestppd; + productName = cupstestppd; + productReference = 72F75A521336F950004BB496 /* cupstestppd */; + productType = "com.apple.product-type.tool"; + }; + 72F75A601336F9A3004BB496 /* libcupsimage */ = { + isa = PBXNativeTarget; + buildConfigurationList = 72F75A621336F9A3004BB496 /* Build configuration list for PBXNativeTarget "libcupsimage" */; + buildPhases = ( + 72F75A5D1336F9A3004BB496 /* Sources */, + 72F75A5E1336F9A3004BB496 /* Frameworks */, + 72F75A5F1336F9A3004BB496 /* Headers */, + ); + buildRules = ( + ); + dependencies = ( + 72F75A661336FA30004BB496 /* PBXTargetDependency */, + ); + name = libcupsimage; + productName = libcupsimage; + productReference = 72F75A611336F9A3004BB496 /* libcupsimage.dylib */; + productType = "com.apple.product-type.library.dynamic"; + }; +/* End PBXNativeTarget section */ + +/* Begin PBXProject section */ + 72BF96371333042100B1EAD7 /* Project object */ = { + isa = PBXProject; + attributes = { + LastUpgradeCheck = 0420; + ORGANIZATIONNAME = "Apple Inc."; + }; + buildConfigurationList = 72BF963A1333042100B1EAD7 /* Build configuration list for PBXProject "CUPS" */; + compatibilityVersion = "Xcode 3.2"; + developmentRegion = English; + hasScannedForEncodings = 0; + knownRegions = ( + en, + ); + mainGroup = 72BF96351333042100B1EAD7; + productRefGroup = 72220EAF1333047D00FCA411 /* Products */; + projectDirPath = ""; + projectRoot = ""; + targets = ( + 274FF5DE13332D3000317ECB /* All */, + 273BF6D91333B6260022CAAB /* Tests */, + 72220EAD1333047D00FCA411 /* libcups */, + 274FF6891333B1C400317ECB /* libcups_static */, + 72F75A601336F9A3004BB496 /* libcupsimage */, + 72220FAB13330B2200FCA411 /* libcupsmime */, + 274FF5ED133330C800317ECB /* libcupsppdc */, + 276684031337FA1D000D33D0 /* cupsaddsmb */, + 2766835B1337A9B6000D33D0 /* cupsctl */, + 72220F5A13330A5A00FCA411 /* cupsd */, + 274FF5CB13332B1F00317ECB /* cups-driverd */, + 274FF6281333333600317ECB /* cups-deviced */, + 274FF63D1333358B00317ECB /* cups-exec */, + 274FF64E133339C400317ECB /* cups-lpd */, + 274FF66213333A9B00317ECB /* cups-polld */, + 274FF67713333B2F00317ECB /* cupsfilter */, + 72F75A511336F950004BB496 /* cupstestppd */, + 724379461333FEA9009631B9 /* dnssd */, + 724378FC1333E43E009631B9 /* ipp */, + 726AD6F6135E88F0002C930D /* ippserver */, + 276683EF1337F78E000D33D0 /* ipptool */, + 724379171333E532009631B9 /* lpd */, + 2766836F1337AC79000D33D0 /* ppdc */, + 2766837C1337AC8C000D33D0 /* ppdhtml */, + 276683891337AC97000D33D0 /* ppdi */, + 276683961337ACA2000D33D0 /* ppdmerge */, + 276683A31337ACAB000D33D0 /* ppdpo */, + 7258EAE1134594C4009286F1 /* rastertopwg */, + 720DD6C11358FD5F0064AA82 /* snmp */, + 7243792F1333FB85009631B9 /* socket */, + 273BF6BC1333B5000022CAAB /* testcups */, + 278C58CA136B640300836530 /* testhttp */, + 270CCDA6135E3C9E00007BE2 /* testmime */, + 7243795A1333FF1D009631B9 /* usb */, + ); + }; +/* End PBXProject section */ + +/* Begin PBXSourcesBuildPhase section */ + 270CCDA3135E3C9E00007BE2 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 270CCDBC135E3D3E00007BE2 /* testmime.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 273BF6B91333B5000022CAAB /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 273BF6C71333B5370022CAAB /* testcups.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 274FF5C813332B1F00317ECB /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 274FF5D913332CC700317ECB /* cups-driverd.cxx in Sources */, + 274FF5DA13332CC700317ECB /* util.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 274FF5EA133330C800317ECB /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 274FF60A1333315100317ECB /* ppdc-array.cxx in Sources */, + 274FF60B1333315100317ECB /* ppdc-attr.cxx in Sources */, + 274FF60C1333315100317ECB /* ppdc-catalog.cxx in Sources */, + 274FF60D1333315100317ECB /* ppdc-choice.cxx in Sources */, + 274FF60E1333315100317ECB /* ppdc-constraint.cxx in Sources */, + 274FF60F1333315100317ECB /* ppdc-driver.cxx in Sources */, + 274FF6101333315100317ECB /* ppdc-file.cxx in Sources */, + 274FF6111333315100317ECB /* ppdc-filter.cxx in Sources */, + 274FF6121333315100317ECB /* ppdc-font.cxx in Sources */, + 274FF6131333315100317ECB /* ppdc-group.cxx in Sources */, + 274FF6141333315100317ECB /* ppdc-import.cxx in Sources */, + 274FF6151333315100317ECB /* ppdc-mediasize.cxx in Sources */, + 274FF6161333315100317ECB /* ppdc-message.cxx in Sources */, + 274FF6171333315100317ECB /* ppdc-option.cxx in Sources */, + 274FF6191333315100317ECB /* ppdc-profile.cxx in Sources */, + 274FF61A1333315100317ECB /* ppdc-shared.cxx in Sources */, + 274FF61B1333315100317ECB /* ppdc-source.cxx in Sources */, + 274FF61C1333315100317ECB /* ppdc-string.cxx in Sources */, + 274FF61D1333315100317ECB /* ppdc-variable.cxx in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 274FF6251333333600317ECB /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 274FF6361333344400317ECB /* cups-deviced.c in Sources */, + 274FF6371333345900317ECB /* util.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 274FF63A1333358B00317ECB /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 274FF64A1333398D00317ECB /* cups-exec.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 274FF64B133339C400317ECB /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 274FF65C133339FC00317ECB /* cups-lpd.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 274FF65F13333A9B00317ECB /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 274FF67013333ACF00317ECB /* cups-polld.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 274FF67413333B2F00317ECB /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 274FF68813333B6E00317ECB /* cupsfilter.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 274FF68A1333B1C400317ECB /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 274FF68B1333B1C400317ECB /* adminutil.c in Sources */, + 274FF68C1333B1C400317ECB /* array.c in Sources */, + 274FF68D1333B1C400317ECB /* attr.c in Sources */, + 274FF68E1333B1C400317ECB /* auth.c in Sources */, + 274FF68F1333B1C400317ECB /* backchannel.c in Sources */, + 274FF6901333B1C400317ECB /* backend.c in Sources */, + 274FF6911333B1C400317ECB /* conflicts.c in Sources */, + 274FF6921333B1C400317ECB /* custom.c in Sources */, + 274FF6931333B1C400317ECB /* debug.c in Sources */, + 274FF6941333B1C400317ECB /* dest.c in Sources */, + 274FF6951333B1C400317ECB /* dir.c in Sources */, + 274FF6961333B1C400317ECB /* emit.c in Sources */, + 274FF6971333B1C400317ECB /* encode.c in Sources */, + 274FF6981333B1C400317ECB /* file.c in Sources */, + 274FF6991333B1C400317ECB /* getdevices.c in Sources */, + 274FF69A1333B1C400317ECB /* getifaddrs.c in Sources */, + 274FF69B1333B1C400317ECB /* getputfile.c in Sources */, + 274FF69C1333B1C400317ECB /* globals.c in Sources */, + 274FF69D1333B1C400317ECB /* http-addr.c in Sources */, + 274FF69E1333B1C400317ECB /* http-addrlist.c in Sources */, + 274FF69F1333B1C400317ECB /* http-support.c in Sources */, + 274FF6A01333B1C400317ECB /* http.c in Sources */, + 274FF6A11333B1C400317ECB /* ipp-support.c in Sources */, + 274FF6A21333B1C400317ECB /* ipp.c in Sources */, + 274FF6A31333B1C400317ECB /* langprintf.c in Sources */, + 274FF6A41333B1C400317ECB /* language.c in Sources */, + 274FF6A51333B1C400317ECB /* localize.c in Sources */, + 274FF6A61333B1C400317ECB /* mark.c in Sources */, + 274FF6A71333B1C400317ECB /* md5.c in Sources */, + 274FF6A81333B1C400317ECB /* md5passwd.c in Sources */, + 274FF6A91333B1C400317ECB /* notify.c in Sources */, + 274FF6AA1333B1C400317ECB /* options.c in Sources */, + 274FF6AB1333B1C400317ECB /* page.c in Sources */, + 274FF6AC1333B1C400317ECB /* ppd-cache.c in Sources */, + 274FF6AD1333B1C400317ECB /* ppd.c in Sources */, + 274FF6AE1333B1C400317ECB /* pwg-media.c in Sources */, + 274FF6AF1333B1C400317ECB /* request.c in Sources */, + 274FF6B01333B1C400317ECB /* sidechannel.c in Sources */, + 274FF6B11333B1C400317ECB /* snmp.c in Sources */, + 274FF6B21333B1C400317ECB /* snprintf.c in Sources */, + 274FF6B31333B1C400317ECB /* string.c in Sources */, + 274FF6B41333B1C400317ECB /* tempfile.c in Sources */, + 274FF6B51333B1C400317ECB /* thread.c in Sources */, + 274FF6B61333B1C400317ECB /* transcode.c in Sources */, + 274FF6B71333B1C400317ECB /* usersys.c in Sources */, + 274FF6B81333B1C400317ECB /* util.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 276683581337A9B6000D33D0 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 276683691337AA00000D33D0 /* cupsctl.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 2766836C1337AC79000D33D0 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 276683CD1337B201000D33D0 /* ppdc.cxx in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 276683791337AC8C000D33D0 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 276683CF1337B20D000D33D0 /* ppdhtml.cxx in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 276683861337AC97000D33D0 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 276683D11337B21A000D33D0 /* ppdi.cxx in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 276683931337ACA2000D33D0 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 276683D31337B228000D33D0 /* ppdmerge.cxx in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 276683A01337ACAB000D33D0 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 276683D51337B237000D33D0 /* ppdpo.cxx in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 276683EC1337F78E000D33D0 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 276683FA1337F7A9000D33D0 /* ipptool.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 276684001337FA1D000D33D0 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 276684111337FA7C000D33D0 /* cupsaddsmb.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 278C58C7136B640300836530 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 278C58E3136B647200836530 /* testhttp.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 720DD6BE1358FD5F0064AA82 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 720DD6D413590AB90064AA82 /* ieee1284.c in Sources */, + 720DD6D31358FDDE0064AA82 /* snmp.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 72220EAA1333047D00FCA411 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 72220EB61333052D00FCA411 /* adminutil.c in Sources */, + 72220EC51333056300FCA411 /* array.c in Sources */, + 72220EC71333056300FCA411 /* attr.c in Sources */, + 72220EC81333056300FCA411 /* auth.c in Sources */, + 72220EC91333056300FCA411 /* backchannel.c in Sources */, + 72220ECA1333056300FCA411 /* backend.c in Sources */, + 72220ECC1333056300FCA411 /* conflicts.c in Sources */, + 72220ECF1333056300FCA411 /* custom.c in Sources */, + 72220F0B133305BB00FCA411 /* debug.c in Sources */, + 72220F0C133305BB00FCA411 /* dest.c in Sources */, + 72220F0D133305BB00FCA411 /* dir.c in Sources */, + 72220F0F133305BB00FCA411 /* emit.c in Sources */, + 72220F10133305BB00FCA411 /* encode.c in Sources */, + 72220F12133305BB00FCA411 /* file.c in Sources */, + 72220F14133305BB00FCA411 /* getdevices.c in Sources */, + 72220F15133305BB00FCA411 /* getifaddrs.c in Sources */, + 72220F16133305BB00FCA411 /* getputfile.c in Sources */, + 72220F17133305BB00FCA411 /* globals.c in Sources */, + 72220F18133305BB00FCA411 /* http-addr.c in Sources */, + 72220F19133305BB00FCA411 /* http-addrlist.c in Sources */, + 72220F1B133305BB00FCA411 /* http-support.c in Sources */, + 72220F1C133305BB00FCA411 /* http.c in Sources */, + 72220F1F133305BB00FCA411 /* ipp-support.c in Sources */, + 72220F20133305BB00FCA411 /* ipp.c in Sources */, + 72220F22133305BB00FCA411 /* langprintf.c in Sources */, + 72220F24133305BB00FCA411 /* language.c in Sources */, + 72220F26133305BB00FCA411 /* localize.c in Sources */, + 72220F27133305BB00FCA411 /* mark.c in Sources */, + 72220F29133305BB00FCA411 /* md5.c in Sources */, + 72220F2A133305BB00FCA411 /* md5passwd.c in Sources */, + 72220F2B133305BB00FCA411 /* notify.c in Sources */, + 72220F2C133305BB00FCA411 /* options.c in Sources */, + 72220F2D133305BB00FCA411 /* page.c in Sources */, + 72220F2E133305BB00FCA411 /* ppd-cache.c in Sources */, + 72220F30133305BB00FCA411 /* ppd.c in Sources */, + 72220F32133305BB00FCA411 /* pwg-media.c in Sources */, + 72220F35133305BB00FCA411 /* request.c in Sources */, + 72220F36133305BB00FCA411 /* sidechannel.c in Sources */, + 72220F39133305BB00FCA411 /* snmp.c in Sources */, + 72220F3A133305BB00FCA411 /* snprintf.c in Sources */, + 72220F3C133305BB00FCA411 /* string.c in Sources */, + 72220F3D133305BB00FCA411 /* tempfile.c in Sources */, + 72220F3F133305BB00FCA411 /* thread.c in Sources */, + 72220F40133305BB00FCA411 /* transcode.c in Sources */, + 72220F42133305BB00FCA411 /* usersys.c in Sources */, + 72220F43133305BB00FCA411 /* util.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 72220F5713330A5A00FCA411 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 72220F9013330B0C00FCA411 /* auth.c in Sources */, + 72220F9113330B0C00FCA411 /* banners.c in Sources */, + 72220F9213330B0C00FCA411 /* cert.c in Sources */, + 72220F9313330B0C00FCA411 /* classes.c in Sources */, + 72220F9413330B0C00FCA411 /* client.c in Sources */, + 72220F9513330B0C00FCA411 /* conf.c in Sources */, + 72220F9613330B0C00FCA411 /* dirsvc.c in Sources */, + 72220F9713330B0C00FCA411 /* env.c in Sources */, + 72220F9813330B0C00FCA411 /* ipp.c in Sources */, + 72220F9913330B0C00FCA411 /* job.c in Sources */, + 72220F9A13330B0C00FCA411 /* listen.c in Sources */, + 72220F9B13330B0C00FCA411 /* log.c in Sources */, + 72220F9C13330B0C00FCA411 /* main.c in Sources */, + 72220F9D13330B0C00FCA411 /* network.c in Sources */, + 72220F9E13330B0C00FCA411 /* policy.c in Sources */, + 72220F9F13330B0C00FCA411 /* printers.c in Sources */, + 72220FA013330B0C00FCA411 /* process.c in Sources */, + 72220FA113330B0C00FCA411 /* quotas.c in Sources */, + 72220FA313330B0C00FCA411 /* select.c in Sources */, + 72220FA413330B0C00FCA411 /* server.c in Sources */, + 72220FA513330B0C00FCA411 /* statbuf.c in Sources */, + 72220FA613330B0C00FCA411 /* subscriptions.c in Sources */, + 72220FA713330B0C00FCA411 /* sysman.c in Sources */, + 72C16CB9137B195D007E4BF4 /* file.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 72220FA813330B2200FCA411 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 72220FB613330BCE00FCA411 /* filter.c in Sources */, + 72220FB713330BCE00FCA411 /* mime.c in Sources */, + 72220FB913330BCE00FCA411 /* type.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 724378F91333E43E009631B9 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 7243790D1333E4E3009631B9 /* ipp.c in Sources */, + 7243790E1333E4E3009631B9 /* network.c in Sources */, + 7243790F1333E4E3009631B9 /* snmp-supplies.c in Sources */, + 724379131333E516009631B9 /* runloop.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 724379141333E532009631B9 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 724379221333E928009631B9 /* network.c in Sources */, + 724379231333E928009631B9 /* runloop.c in Sources */, + 724379241333E928009631B9 /* snmp-supplies.c in Sources */, + 724379291333E952009631B9 /* lpd.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 7243792C1333FB85009631B9 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 724379401333FD4B009631B9 /* network.c in Sources */, + 724379411333FD4B009631B9 /* runloop.c in Sources */, + 724379421333FD4B009631B9 /* snmp-supplies.c in Sources */, + 7243793D1333FD19009631B9 /* socket.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 724379431333FEA9009631B9 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 724379511333FEBB009631B9 /* dnssd.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 724379571333FF1D009631B9 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 724379C71333FFC7009631B9 /* usb.c in Sources */, + 724379CB1334000E009631B9 /* ieee1284.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 7258EADE134594C4009286F1 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 7258EAED134594EB009286F1 /* rastertopwg.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 726AD6F3135E88F0002C930D /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 726AD702135E8A90002C930D /* ippserver.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 72F75A4E1336F950004BB496 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 72F75A5C1336F988004BB496 /* cupstestppd.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 72F75A5D1336F9A3004BB496 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 72F75A6C1336FA8A004BB496 /* error.c in Sources */, + 72F75A6D1336FA8A004BB496 /* interpret.c in Sources */, + 72F75A6E1336FA8A004BB496 /* raster.c in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXSourcesBuildPhase section */ + +/* Begin PBXTargetDependency section */ + 270CCDB2135E3CDE00007BE2 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 270CCDA6135E3C9E00007BE2 /* testmime */; + targetProxy = 270CCDB1135E3CDE00007BE2 /* PBXContainerItemProxy */; + }; + 270CCDB6135E3CF700007BE2 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220FAB13330B2200FCA411 /* libcupsmime */; + targetProxy = 270CCDB5135E3CF700007BE2 /* PBXContainerItemProxy */; + }; + 270CCDB8135E3CFD00007BE2 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 274FF6891333B1C400317ECB /* libcups_static */; + targetProxy = 270CCDB7135E3CFD00007BE2 /* PBXContainerItemProxy */; + }; + 273BF6C91333B5410022CAAB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 274FF6891333B1C400317ECB /* libcups_static */; + targetProxy = 273BF6C81333B5410022CAAB /* PBXContainerItemProxy */; + }; + 273BF6DE1333B6370022CAAB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 273BF6BC1333B5000022CAAB /* testcups */; + targetProxy = 273BF6DD1333B6370022CAAB /* PBXContainerItemProxy */; + }; + 274FF5DC13332CF900317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 274FF5DB13332CF900317ECB /* PBXContainerItemProxy */; + }; + 274FF5E313332D4300317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 274FF5E213332D4300317ECB /* PBXContainerItemProxy */; + }; + 274FF5E513332D4300317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220FAB13330B2200FCA411 /* libcupsmime */; + targetProxy = 274FF5E413332D4300317ECB /* PBXContainerItemProxy */; + }; + 274FF5E713332D4300317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220F5A13330A5A00FCA411 /* cupsd */; + targetProxy = 274FF5E613332D4300317ECB /* PBXContainerItemProxy */; + }; + 274FF5E913332D4300317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 274FF5CB13332B1F00317ECB /* cups-driverd */; + targetProxy = 274FF5E813332D4300317ECB /* PBXContainerItemProxy */; + }; + 274FF5F3133330FD00317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 274FF5F2133330FD00317ECB /* PBXContainerItemProxy */; + }; + 274FF6201333316200317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 274FF5ED133330C800317ECB /* libcupsppdc */; + targetProxy = 274FF61F1333316200317ECB /* PBXContainerItemProxy */; + }; + 274FF622133331D300317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 274FF5ED133330C800317ECB /* libcupsppdc */; + targetProxy = 274FF621133331D300317ECB /* PBXContainerItemProxy */; + }; + 274FF6341333335200317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 274FF6331333335200317ECB /* PBXContainerItemProxy */; + }; + 274FF6391333348400317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 274FF6281333333600317ECB /* cups-deviced */; + targetProxy = 274FF6381333348400317ECB /* PBXContainerItemProxy */; + }; + 274FF648133335A300317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 274FF63D1333358B00317ECB /* cups-exec */; + targetProxy = 274FF647133335A300317ECB /* PBXContainerItemProxy */; + }; + 274FF65A133339D900317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 274FF659133339D900317ECB /* PBXContainerItemProxy */; + }; + 274FF65E13333A3400317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 274FF64E133339C400317ECB /* cups-lpd */; + targetProxy = 274FF65D13333A3400317ECB /* PBXContainerItemProxy */; + }; + 274FF66D13333AAD00317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 274FF66C13333AAD00317ECB /* PBXContainerItemProxy */; + }; + 274FF67213333AE400317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 274FF66213333A9B00317ECB /* cups-polld */; + targetProxy = 274FF67113333AE400317ECB /* PBXContainerItemProxy */; + }; + 274FF68213333B3C00317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 274FF68113333B3C00317ECB /* PBXContainerItemProxy */; + }; + 274FF68413333B3C00317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220FAB13330B2200FCA411 /* libcupsmime */; + targetProxy = 274FF68313333B3C00317ECB /* PBXContainerItemProxy */; + }; + 274FF6E21333B33F00317ECB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 274FF67713333B2F00317ECB /* cupsfilter */; + targetProxy = 274FF6E11333B33F00317ECB /* PBXContainerItemProxy */; + }; + 276683661337A9D6000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 276683651337A9D6000D33D0 /* PBXContainerItemProxy */; + }; + 2766836B1337AA25000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 2766835B1337A9B6000D33D0 /* cupsctl */; + targetProxy = 2766836A1337AA25000D33D0 /* PBXContainerItemProxy */; + }; + 276683AE1337ACF9000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 276683AD1337ACF9000D33D0 /* PBXContainerItemProxy */; + }; + 276683B01337ACF9000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 274FF5ED133330C800317ECB /* libcupsppdc */; + targetProxy = 276683AF1337ACF9000D33D0 /* PBXContainerItemProxy */; + }; + 276683B41337AD18000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 276683B31337AD18000D33D0 /* PBXContainerItemProxy */; + }; + 276683B61337AD18000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 274FF5ED133330C800317ECB /* libcupsppdc */; + targetProxy = 276683B51337AD18000D33D0 /* PBXContainerItemProxy */; + }; + 276683BC1337AE49000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 276683BB1337AE49000D33D0 /* PBXContainerItemProxy */; + }; + 276683BE1337AE49000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 274FF5ED133330C800317ECB /* libcupsppdc */; + targetProxy = 276683BD1337AE49000D33D0 /* PBXContainerItemProxy */; + }; + 276683C01337B1AD000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 276683BF1337B1AD000D33D0 /* PBXContainerItemProxy */; + }; + 276683C21337B1AD000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 274FF5ED133330C800317ECB /* libcupsppdc */; + targetProxy = 276683C11337B1AD000D33D0 /* PBXContainerItemProxy */; + }; + 276683C61337B1BC000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 276683C51337B1BC000D33D0 /* PBXContainerItemProxy */; + }; + 276683C81337B1BC000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 274FF5ED133330C800317ECB /* libcupsppdc */; + targetProxy = 276683C71337B1BC000D33D0 /* PBXContainerItemProxy */; + }; + 276683D71337B24A000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 2766836F1337AC79000D33D0 /* ppdc */; + targetProxy = 276683D61337B24A000D33D0 /* PBXContainerItemProxy */; + }; + 276683D91337B24A000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 2766837C1337AC8C000D33D0 /* ppdhtml */; + targetProxy = 276683D81337B24A000D33D0 /* PBXContainerItemProxy */; + }; + 276683DB1337B24A000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 276683891337AC97000D33D0 /* ppdi */; + targetProxy = 276683DA1337B24A000D33D0 /* PBXContainerItemProxy */; + }; + 276683DD1337B24A000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 276683961337ACA2000D33D0 /* ppdmerge */; + targetProxy = 276683DC1337B24A000D33D0 /* PBXContainerItemProxy */; + }; + 276683DF1337B24A000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 276683A31337ACAB000D33D0 /* ppdpo */; + targetProxy = 276683DE1337B24A000D33D0 /* PBXContainerItemProxy */; + }; + 276683E11337B299000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 276683E01337B299000D33D0 /* PBXContainerItemProxy */; + }; + 276683E41337B2BA000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72F75A601336F9A3004BB496 /* libcupsimage */; + targetProxy = 276683E31337B2BA000D33D0 /* PBXContainerItemProxy */; + }; + 276683FC1337F7B3000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 276683FB1337F7B3000D33D0 /* PBXContainerItemProxy */; + }; + 276683FF1337F7C5000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 276683EF1337F78E000D33D0 /* ipptool */; + targetProxy = 276683FE1337F7C5000D33D0 /* PBXContainerItemProxy */; + }; + 2766840E1337FA31000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 2766840D1337FA31000D33D0 /* PBXContainerItemProxy */; + }; + 276684131337FA8D000D33D0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 276684031337FA1D000D33D0 /* cupsaddsmb */; + targetProxy = 276684121337FA8D000D33D0 /* PBXContainerItemProxy */; + }; + 278C58D6136B641D00836530 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 278C58CA136B640300836530 /* testhttp */; + targetProxy = 278C58D5136B641D00836530 /* PBXContainerItemProxy */; + }; + 278C58D8136B642F00836530 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 274FF6891333B1C400317ECB /* libcups_static */; + targetProxy = 278C58D7136B642F00836530 /* PBXContainerItemProxy */; + }; + 720DD6CF1358FD790064AA82 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 720DD6CE1358FD790064AA82 /* PBXContainerItemProxy */; + }; + 720DD6D11358FDBE0064AA82 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 720DD6C11358FD5F0064AA82 /* snmp */; + targetProxy = 720DD6D01358FDBE0064AA82 /* PBXContainerItemProxy */; + }; + 72220F6513330A6500FCA411 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 72220F6413330A6500FCA411 /* PBXContainerItemProxy */; + }; + 72220FBC13330C0500FCA411 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 72220FBB13330C0500FCA411 /* PBXContainerItemProxy */; + }; + 72220FBE13330C0B00FCA411 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220FAB13330B2200FCA411 /* libcupsmime */; + targetProxy = 72220FBD13330C0B00FCA411 /* PBXContainerItemProxy */; + }; + 724379071333E49B009631B9 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 724379061333E49B009631B9 /* PBXContainerItemProxy */; + }; + 724379111333E4EA009631B9 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 724378FC1333E43E009631B9 /* ipp */; + targetProxy = 724379101333E4EA009631B9 /* PBXContainerItemProxy */; + }; + 724379261333E932009631B9 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 724379251333E932009631B9 /* PBXContainerItemProxy */; + }; + 7243792B1333E962009631B9 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 724379171333E532009631B9 /* lpd */; + targetProxy = 7243792A1333E962009631B9 /* PBXContainerItemProxy */; + }; + 7243793A1333FB95009631B9 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 724379391333FB95009631B9 /* PBXContainerItemProxy */; + }; + 7243793F1333FD23009631B9 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 7243792F1333FB85009631B9 /* socket */; + targetProxy = 7243793E1333FD23009631B9 /* PBXContainerItemProxy */; + }; + 724379531333FECE009631B9 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 724379461333FEA9009631B9 /* dnssd */; + targetProxy = 724379521333FECE009631B9 /* PBXContainerItemProxy */; + }; + 724379551333FEFE009631B9 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 724379541333FEFE009631B9 /* PBXContainerItemProxy */; + }; + 724379651333FF2E009631B9 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 724379641333FF2E009631B9 /* PBXContainerItemProxy */; + }; + 724379C31333FF7D009631B9 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 7243795A1333FF1D009631B9 /* usb */; + targetProxy = 724379C21333FF7D009631B9 /* PBXContainerItemProxy */; + }; + 7258EAEF13459ADA009286F1 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 7258EAE1134594C4009286F1 /* rastertopwg */; + targetProxy = 7258EAEE13459ADA009286F1 /* PBXContainerItemProxy */; + }; + 7258EAF113459B67009286F1 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 7258EAF013459B67009286F1 /* PBXContainerItemProxy */; + }; + 7258EAF313459B67009286F1 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72F75A601336F9A3004BB496 /* libcupsimage */; + targetProxy = 7258EAF213459B67009286F1 /* PBXContainerItemProxy */; + }; + 726AD704135E8AA1002C930D /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 726AD6F6135E88F0002C930D /* ippserver */; + targetProxy = 726AD703135E8AA1002C930D /* PBXContainerItemProxy */; + }; + 726AD706135E8AC5002C930D /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 274FF6891333B1C400317ECB /* libcups_static */; + targetProxy = 726AD705135E8AC5002C930D /* PBXContainerItemProxy */; + }; + 72F75A661336FA30004BB496 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72220EAD1333047D00FCA411 /* libcups */; + targetProxy = 72F75A651336FA30004BB496 /* PBXContainerItemProxy */; + }; + 72F75A711336FACD004BB496 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72F75A601336F9A3004BB496 /* libcupsimage */; + targetProxy = 72F75A701336FACD004BB496 /* PBXContainerItemProxy */; + }; + 72F75A731336FACD004BB496 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 72F75A511336F950004BB496 /* cupstestppd */; + targetProxy = 72F75A721336FACD004BB496 /* PBXContainerItemProxy */; + }; +/* End PBXTargetDependency section */ + +/* Begin XCBuildConfiguration section */ + 270CCDAD135E3C9E00007BE2 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 270CCDAE135E3C9E00007BE2 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 273BF6C41333B5000022CAAB /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 273BF6C51333B5000022CAAB /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 273BF6DB1333B6270022CAAB /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 273BF6DC1333B6270022CAAB /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 274FF5D313332B1F00317ECB /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/daemon; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 274FF5D413332B1F00317ECB /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/daemon; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 274FF5E013332D3100317ECB /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 274FF5E113332D3100317ECB /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 274FF5F0133330C800317ECB /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + EXECUTABLE_PREFIX = ""; + INSTALL_PATH = /usr/lib; + PRIVATE_HEADERS_FOLDER_PATH = /usr/local/include/cups; + PRODUCT_NAME = "$(TARGET_NAME)"; + PUBLIC_HEADERS_FOLDER_PATH = /usr/include/cups; + }; + name = Debug; + }; + 274FF5F1133330C800317ECB /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + EXECUTABLE_PREFIX = ""; + INSTALL_PATH = /usr/lib; + PRIVATE_HEADERS_FOLDER_PATH = /usr/local/include/cups; + PRODUCT_NAME = "$(TARGET_NAME)"; + PUBLIC_HEADERS_FOLDER_PATH = /usr/include/cups; + }; + name = Release; + }; + 274FF6301333333600317ECB /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/daemon; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 274FF6311333333600317ECB /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/daemon; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 274FF6451333358C00317ECB /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/daemon; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 274FF6461333358C00317ECB /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/daemon; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 274FF656133339C400317ECB /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/daemon; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 274FF657133339C400317ECB /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/daemon; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 274FF66A13333A9B00317ECB /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/daemon; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 274FF66B13333A9B00317ECB /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/daemon; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 274FF67F13333B2F00317ECB /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/sbin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 274FF68013333B2F00317ECB /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/sbin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 274FF6DE1333B1C400317ECB /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + EXECUTABLE_EXTENSION = a; + EXECUTABLE_PREFIX = ""; + INSTALL_PATH = /usr/local/lib; + PRIVATE_HEADERS_FOLDER_PATH = /usr/local/include/cups; + PRODUCT_NAME = libcups_static; + PUBLIC_HEADERS_FOLDER_PATH = /usr/include/cups; + STANDARD_C_PLUS_PLUS_LIBRARY_TYPE = static; + }; + name = Debug; + }; + 274FF6DF1333B1C400317ECB /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + EXECUTABLE_EXTENSION = a; + EXECUTABLE_PREFIX = ""; + INSTALL_PATH = /usr/local/lib; + PRIVATE_HEADERS_FOLDER_PATH = /usr/local/include/cups; + PRODUCT_NAME = libcups_static; + PUBLIC_HEADERS_FOLDER_PATH = /usr/include/cups; + STANDARD_C_PLUS_PLUS_LIBRARY_TYPE = static; + }; + name = Release; + }; + 276683631337A9B6000D33D0 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/sbin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 276683641337A9B6000D33D0 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/sbin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 276683771337AC79000D33D0 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/bin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 276683781337AC79000D33D0 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/bin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 276683841337AC8C000D33D0 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/bin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 276683851337AC8C000D33D0 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/bin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 276683911337AC97000D33D0 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/bin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 276683921337AC97000D33D0 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/bin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 2766839E1337ACA2000D33D0 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/bin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 2766839F1337ACA2000D33D0 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/bin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 276683AB1337ACAB000D33D0 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/bin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 276683AC1337ACAB000D33D0 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/bin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 276683F71337F78F000D33D0 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/bin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 276683F81337F78F000D33D0 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/bin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 2766840B1337FA1E000D33D0 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/sbin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 2766840C1337FA1E000D33D0 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/sbin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 278C58D1136B640300836530 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 278C58D2136B640300836530 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 720DD6C91358FD5F0064AA82 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/backend; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 720DD6CA1358FD5F0064AA82 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/backend; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 72220EB01333047D00FCA411 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + EXECUTABLE_PREFIX = ""; + INSTALL_PATH = /usr/lib; + PRIVATE_HEADERS_FOLDER_PATH = /usr/local/include/cups; + PRODUCT_NAME = "$(TARGET_NAME)"; + PUBLIC_HEADERS_FOLDER_PATH = /usr/include/cups; + }; + name = Debug; + }; + 72220EB11333047D00FCA411 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + EXECUTABLE_PREFIX = ""; + INSTALL_PATH = /usr/lib; + PRIVATE_HEADERS_FOLDER_PATH = /usr/local/include/cups; + PRODUCT_NAME = "$(TARGET_NAME)"; + PUBLIC_HEADERS_FOLDER_PATH = /usr/include/cups; + }; + name = Release; + }; + 72220F6213330A5A00FCA411 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/sbin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 72220F6313330A5A00FCA411 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/sbin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 72220FAE13330B2300FCA411 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + EXECUTABLE_PREFIX = ""; + INSTALL_PATH = /usr/lib; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 72220FAF13330B2300FCA411 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + EXECUTABLE_PREFIX = ""; + INSTALL_PATH = /usr/lib; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 724379041333E43E009631B9 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_MODE_FLAG = "u+rwX,go-rwX"; + INSTALL_PATH = /usr/libexec/cups/backend; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 724379051333E43E009631B9 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_MODE_FLAG = "u+rwX,go-rwX"; + INSTALL_PATH = /usr/libexec/cups/backend; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 7243791F1333E532009631B9 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/backend; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 724379201333E532009631B9 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/backend; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 724379371333FB85009631B9 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/backend; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 724379381333FB85009631B9 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/backend; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 7243794E1333FEA9009631B9 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/backend; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 7243794F1333FEA9009631B9 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/backend; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 724379621333FF1D009631B9 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/backend; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 724379631333FF1D009631B9 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/backend; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 7258EAEA134594C4009286F1 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/filter; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 7258EAEB134594C4009286F1 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/libexec/cups/filter; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 726AD6FF135E88F1002C930D /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 726AD700135E88F1002C930D /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 72BF963C1333042100B1EAD7 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + ARCHS = "$(NATIVE_ARCH_ACTUAL)"; + DEBUG_INFORMATION_FORMAT = dwarf; + GCC_PREPROCESSOR_DEFINITIONS = DEBUG; + GCC_WARN_64_TO_32_BIT_CONVERSION = YES; + GCC_WARN_ABOUT_MISSING_PROTOTYPES = YES; + GCC_WARN_ABOUT_RETURN_TYPE = YES; + GCC_WARN_SHADOW = YES; + GCC_WARN_UNINITIALIZED_AUTOS = YES; + GCC_WARN_UNUSED_FUNCTION = YES; + GCC_WARN_UNUSED_LABEL = YES; + GCC_WARN_UNUSED_PARAMETER = YES; + GCC_WARN_UNUSED_VARIABLE = YES; + HEADER_SEARCH_PATHS = ( + ., + .., + ); + OTHER_CFLAGS = ( + "-D_CUPS_SOURCE", + "-Wno-shorten-64-to-32", + ); + }; + name = Debug; + }; + 72BF963D1333042100B1EAD7 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + ARCHS = "$(ARCHS_STANDARD_32_64_BIT)"; + DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; + GCC_WARN_64_TO_32_BIT_CONVERSION = YES; + GCC_WARN_ABOUT_MISSING_PROTOTYPES = YES; + GCC_WARN_ABOUT_RETURN_TYPE = YES; + GCC_WARN_SHADOW = YES; + GCC_WARN_UNINITIALIZED_AUTOS = YES; + GCC_WARN_UNUSED_FUNCTION = YES; + GCC_WARN_UNUSED_LABEL = YES; + GCC_WARN_UNUSED_PARAMETER = YES; + GCC_WARN_UNUSED_VARIABLE = YES; + HEADER_SEARCH_PATHS = ( + ., + .., + ); + OTHER_CFLAGS = ( + "-D_CUPS_SOURCE", + "-Wno-shorten-64-to-32", + ); + }; + name = Release; + }; + 72F75A591336F951004BB496 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/bin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 72F75A5A1336F951004BB496 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + INSTALL_PATH = /usr/bin; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; + 72F75A631336F9A3004BB496 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + EXECUTABLE_PREFIX = ""; + INSTALL_PATH = /usr/lib; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 72F75A641336F9A3004BB496 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + EXECUTABLE_PREFIX = ""; + INSTALL_PATH = /usr/lib; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; +/* End XCBuildConfiguration section */ + +/* Begin XCConfigurationList section */ + 270CCDAF135E3C9E00007BE2 /* Build configuration list for PBXNativeTarget "testmime" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 270CCDAD135E3C9E00007BE2 /* Debug */, + 270CCDAE135E3C9E00007BE2 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 273BF6C31333B5000022CAAB /* Build configuration list for PBXNativeTarget "testcups" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 273BF6C41333B5000022CAAB /* Debug */, + 273BF6C51333B5000022CAAB /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 273BF6DA1333B6270022CAAB /* Build configuration list for PBXAggregateTarget "Tests" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 273BF6DB1333B6270022CAAB /* Debug */, + 273BF6DC1333B6270022CAAB /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 274FF5D213332B1F00317ECB /* Build configuration list for PBXNativeTarget "cups-driverd" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 274FF5D313332B1F00317ECB /* Debug */, + 274FF5D413332B1F00317ECB /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 274FF5DF13332D3100317ECB /* Build configuration list for PBXAggregateTarget "All" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 274FF5E013332D3100317ECB /* Debug */, + 274FF5E113332D3100317ECB /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 274FF5EF133330C800317ECB /* Build configuration list for PBXNativeTarget "libcupsppdc" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 274FF5F0133330C800317ECB /* Debug */, + 274FF5F1133330C800317ECB /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 274FF62F1333333600317ECB /* Build configuration list for PBXNativeTarget "cups-deviced" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 274FF6301333333600317ECB /* Debug */, + 274FF6311333333600317ECB /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 274FF6441333358C00317ECB /* Build configuration list for PBXNativeTarget "cups-exec" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 274FF6451333358C00317ECB /* Debug */, + 274FF6461333358C00317ECB /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 274FF655133339C400317ECB /* Build configuration list for PBXNativeTarget "cups-lpd" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 274FF656133339C400317ECB /* Debug */, + 274FF657133339C400317ECB /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 274FF66913333A9B00317ECB /* Build configuration list for PBXNativeTarget "cups-polld" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 274FF66A13333A9B00317ECB /* Debug */, + 274FF66B13333A9B00317ECB /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 274FF67E13333B2F00317ECB /* Build configuration list for PBXNativeTarget "cupsfilter" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 274FF67F13333B2F00317ECB /* Debug */, + 274FF68013333B2F00317ECB /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 274FF6DD1333B1C400317ECB /* Build configuration list for PBXNativeTarget "libcups_static" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 274FF6DE1333B1C400317ECB /* Debug */, + 274FF6DF1333B1C400317ECB /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 276683621337A9B6000D33D0 /* Build configuration list for PBXNativeTarget "cupsctl" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 276683631337A9B6000D33D0 /* Debug */, + 276683641337A9B6000D33D0 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 276683761337AC79000D33D0 /* Build configuration list for PBXNativeTarget "ppdc" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 276683771337AC79000D33D0 /* Debug */, + 276683781337AC79000D33D0 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 276683831337AC8C000D33D0 /* Build configuration list for PBXNativeTarget "ppdhtml" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 276683841337AC8C000D33D0 /* Debug */, + 276683851337AC8C000D33D0 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 276683901337AC97000D33D0 /* Build configuration list for PBXNativeTarget "ppdi" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 276683911337AC97000D33D0 /* Debug */, + 276683921337AC97000D33D0 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 2766839D1337ACA2000D33D0 /* Build configuration list for PBXNativeTarget "ppdmerge" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 2766839E1337ACA2000D33D0 /* Debug */, + 2766839F1337ACA2000D33D0 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 276683AA1337ACAB000D33D0 /* Build configuration list for PBXNativeTarget "ppdpo" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 276683AB1337ACAB000D33D0 /* Debug */, + 276683AC1337ACAB000D33D0 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 276683F61337F78F000D33D0 /* Build configuration list for PBXNativeTarget "ipptool" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 276683F71337F78F000D33D0 /* Debug */, + 276683F81337F78F000D33D0 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 2766840A1337FA1E000D33D0 /* Build configuration list for PBXNativeTarget "cupsaddsmb" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 2766840B1337FA1E000D33D0 /* Debug */, + 2766840C1337FA1E000D33D0 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 278C58D3136B640300836530 /* Build configuration list for PBXNativeTarget "testhttp" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 278C58D1136B640300836530 /* Debug */, + 278C58D2136B640300836530 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 720DD6CB1358FD600064AA82 /* Build configuration list for PBXNativeTarget "snmp" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 720DD6C91358FD5F0064AA82 /* Debug */, + 720DD6CA1358FD5F0064AA82 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 72220EB21333047D00FCA411 /* Build configuration list for PBXNativeTarget "libcups" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 72220EB01333047D00FCA411 /* Debug */, + 72220EB11333047D00FCA411 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 72220F6113330A5A00FCA411 /* Build configuration list for PBXNativeTarget "cupsd" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 72220F6213330A5A00FCA411 /* Debug */, + 72220F6313330A5A00FCA411 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 72220FAD13330B2300FCA411 /* Build configuration list for PBXNativeTarget "libcupsmime" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 72220FAE13330B2300FCA411 /* Debug */, + 72220FAF13330B2300FCA411 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 724379031333E43E009631B9 /* Build configuration list for PBXNativeTarget "ipp" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 724379041333E43E009631B9 /* Debug */, + 724379051333E43E009631B9 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 7243791E1333E532009631B9 /* Build configuration list for PBXNativeTarget "lpd" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 7243791F1333E532009631B9 /* Debug */, + 724379201333E532009631B9 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 724379361333FB85009631B9 /* Build configuration list for PBXNativeTarget "socket" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 724379371333FB85009631B9 /* Debug */, + 724379381333FB85009631B9 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 7243794D1333FEA9009631B9 /* Build configuration list for PBXNativeTarget "dnssd" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 7243794E1333FEA9009631B9 /* Debug */, + 7243794F1333FEA9009631B9 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 724379611333FF1D009631B9 /* Build configuration list for PBXNativeTarget "usb" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 724379621333FF1D009631B9 /* Debug */, + 724379631333FF1D009631B9 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 7258EAE9134594C4009286F1 /* Build configuration list for PBXNativeTarget "rastertopwg" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 7258EAEA134594C4009286F1 /* Debug */, + 7258EAEB134594C4009286F1 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 726AD6FE135E88F1002C930D /* Build configuration list for PBXNativeTarget "ippserver" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 726AD6FF135E88F1002C930D /* Debug */, + 726AD700135E88F1002C930D /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 72BF963A1333042100B1EAD7 /* Build configuration list for PBXProject "CUPS" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 72BF963C1333042100B1EAD7 /* Debug */, + 72BF963D1333042100B1EAD7 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 72F75A581336F951004BB496 /* Build configuration list for PBXNativeTarget "cupstestppd" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 72F75A591336F951004BB496 /* Debug */, + 72F75A5A1336F951004BB496 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 72F75A621336F9A3004BB496 /* Build configuration list for PBXNativeTarget "libcupsimage" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 72F75A631336F9A3004BB496 /* Debug */, + 72F75A641336F9A3004BB496 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; +/* End XCConfigurationList section */ + }; + rootObject = 72BF96371333042100B1EAD7 /* Project object */; +} diff --git a/xcode/config.h b/xcode/config.h new file mode 100644 index 0000000000..6574617cc3 --- /dev/null +++ b/xcode/config.h @@ -0,0 +1,736 @@ +/* config.h. Generated from config.h.in by configure. */ +/* + * "$Id$" + * + * Configuration file for CUPS. + * + * Copyright 2007-2011 by Apple Inc. + * Copyright 1997-2007 by Easy Software Products. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +#ifndef _CUPS_CONFIG_H_ +#define _CUPS_CONFIG_H_ + +/* + * Version of software... + */ + +#define CUPS_SVERSION "CUPS v1.5.0" +#define CUPS_MINIMAL "CUPS/1.5.0" + + +/* + * Default user and groups... + */ + +#define CUPS_DEFAULT_USER "_lp" +#define CUPS_DEFAULT_GROUP "_lp" +#define CUPS_DEFAULT_SYSTEM_GROUPS "admin" +#define CUPS_DEFAULT_PRINTOPERATOR_AUTH "@AUTHKEY(system.print.operator) @admin @lpadmin" + + +/* + * Default file permissions... + */ + +#define CUPS_DEFAULT_CONFIG_FILE_PERM 0644 +#define CUPS_DEFAULT_LOG_FILE_PERM 0644 + + +/* + * Default logging settings... + */ + +#define CUPS_DEFAULT_LOG_LEVEL "warn" +#define CUPS_DEFAULT_ACCESS_LOG_LEVEL "actions" + + +/* + * Default fatal error settings... + */ + +#define CUPS_DEFAULT_FATAL_ERRORS "config" + + +/* + * Default browsing settings... + */ + +#define CUPS_DEFAULT_BROWSING 1 +#define CUPS_DEFAULT_BROWSE_LOCAL_PROTOCOLS "CUPS dnssd" +#define CUPS_DEFAULT_BROWSE_REMOTE_PROTOCOLS "" +#define CUPS_DEFAULT_BROWSE_SHORT_NAMES 1 +#define CUPS_DEFAULT_DEFAULT_SHARED 1 +#define CUPS_DEFAULT_IMPLICIT_CLASSES 1 +#define CUPS_DEFAULT_USE_NETWORK_DEFAULT 0 + + +/* + * Default IPP port... + */ + +#define CUPS_DEFAULT_IPP_PORT 631 + + +/* + * Default printcap file... + */ + +#define CUPS_DEFAULT_PRINTCAP "/Library/Preferences/org.cups.printers.plist" + + +/* + * Default Samba and LPD config files... + */ + +#define CUPS_DEFAULT_SMB_CONFIG_FILE "" +#define CUPS_DEFAULT_LPD_CONFIG_FILE "launchd:///System/Library/LaunchDaemons/org.cups.cups-lpd.plist" + + +/* + * Default MaxCopies value... + */ + +#define CUPS_DEFAULT_MAX_COPIES 9999 + + +/* + * Do we have domain socket support, and if so what is the default one? + */ + +#define CUPS_DEFAULT_DOMAINSOCKET "/private/var/run/cupsd" + + +/* + * Default WebInterface value... + */ + +#define CUPS_DEFAULT_WEBIF 0 + + +/* + * Where are files stored? + * + * Note: These are defaults, which can be overridden by environment + * variables at run-time... + */ + +#define CUPS_BINDIR "/usr/bin" +#define CUPS_CACHEDIR "/private/var/spool/cups/cache" +#define CUPS_DATADIR "/usr/share/cups" +#define CUPS_DOCROOT "/usr/share/doc/cups" +#define CUPS_FONTPATH "/usr/share/cups/fonts" +#define CUPS_LOCALEDIR "/usr/share/locale" +#define CUPS_LOGDIR "/private/var/log/cups" +#define CUPS_REQUESTS "/private/var/spool/cups" +#define CUPS_SBINDIR "/usr/sbin" +#define CUPS_SERVERBIN "/usr/libexec/cups" +#define CUPS_SERVERROOT "/private/etc/cups" +#define CUPS_STATEDIR "/private/etc/cups" + + +/* + * Do we have various image libraries? + */ + +/* #undef HAVE_LIBPNG */ +#define HAVE_LIBZ 1 +/* #undef HAVE_LIBJPEG */ +/* #undef HAVE_LIBTIFF */ + + +/* + * Do we have PAM stuff? + */ + +#ifndef HAVE_LIBPAM +#define HAVE_LIBPAM 1 +#endif /* !HAVE_LIBPAM */ + +/* #undef HAVE_PAM_PAM_APPL_H */ +#define HAVE_PAM_SET_ITEM 1 +#define HAVE_PAM_SETCRED 1 + + +/* + * Do we have ? + */ + +/* #undef HAVE_SHADOW_H */ + + +/* + * Do we have ? + */ + +/* #undef HAVE_CRYPT_H */ + + +/* + * Do we have ? + */ + +/* #undef HAVE_SCSI_SG_H */ + + +/* + * Use , , and/or ? + */ + +#define HAVE_STRING_H 1 +#define HAVE_STRINGS_H 1 +/* #undef HAVE_BSTRING_H */ + +/* + * Do we have the long long type? + */ + +#define HAVE_LONG_LONG 1 + +#ifdef HAVE_LONG_LONG +# define CUPS_LLFMT "%lld" +# define CUPS_LLCAST (long long) +#else +# define CUPS_LLFMT "%ld" +# define CUPS_LLCAST (long) +#endif /* HAVE_LONG_LONG */ + +/* + * Do we have the strtoll() function? + */ + +#define HAVE_STRTOLL 1 + +#ifndef HAVE_STRTOLL +# define strtoll(nptr,endptr,base) strtol((nptr), (endptr), (base)) +#endif /* !HAVE_STRTOLL */ + +/* + * Do we have the strXXX() functions? + */ + +#define HAVE_STRDUP 1 +#define HAVE_STRLCAT 1 +#define HAVE_STRLCPY 1 + + +/* + * Do we have the geteuid() function? + */ + +#define HAVE_GETEUID 1 + + +/* + * Do we have the setpgid() function? + */ + +#define HAVE_SETPGID 1 + + +/* + * Do we have the vsyslog() function? + */ + +#define HAVE_VSYSLOG 1 + + +/* + * Do we have the (v)snprintf() functions? + */ + +#define HAVE_SNPRINTF 1 +#define HAVE_VSNPRINTF 1 + + +/* + * What signal functions to use? + */ + +#define HAVE_SIGSET 1 +#define HAVE_SIGACTION 1 + + +/* + * What wait functions to use? + */ + +#define HAVE_WAITPID 1 +#define HAVE_WAIT3 1 + + +/* + * Do we have the mallinfo function and malloc.h? + */ + +/* #undef HAVE_MALLINFO */ +/* #undef HAVE_MALLOC_H */ + + +/* + * Do we have the POSIX ACL functions? + */ + +#define HAVE_ACL_INIT 1 + + +/* + * Do we have the langinfo.h header file? + */ + +#define HAVE_LANGINFO_H 1 + + +/* + * Which encryption libraries do we have? + */ + +#define HAVE_CDSASSL 1 +/* #undef HAVE_GNUTLS */ +/* #undef HAVE_LIBSSL */ +#define HAVE_SSL 1 + + +/* + * What Security framework headers do we have? + */ + +#define HAVE_AUTHORIZATION_H 1 +#define HAVE_SECCERTIFICATE_H 1 +#define HAVE_SECITEM_H 1 +/* #undef HAVE_SECITEMPRIV_H */ +#define HAVE_SECPOLICY_H 1 +/* #undef HAVE_SECPOLICYPRIV_H */ +/* #undef HAVE_SECBASEPRIV_H */ +/* #undef HAVE_SECIDENTITYSEARCHPRIV_H */ + + +/* + * Do we have the SecCertificateCopyData function? + */ + +#define HAVE_SECCERTIFICATECOPYDATA 1 + + +/* + * Do we have the SecIdentitySearchCreateWithPolicy function? + */ + +#define HAVE_SECIDENTITYSEARCHCREATEWITHPOLICY 1 + + +/* + * Do we have the SecPolicyCreateSSL function? + */ + +#define HAVE_SECPOLICYCREATESSL 1 + + +/* + * Do we have the SecPolicyCreateSSL function? + */ + +#define HAVE_SECPOLICYCREATESSL 1 + + +/* + * Do we have the cssmErrorString function? + */ + +#define HAVE_CSSMERRORSTRING 1 + + +/* + * Do we have the SLP library? + */ + +/* #undef HAVE_LIBSLP */ + + +/* + * Do we have an LDAP library? + */ + +#define HAVE_LDAP 1 +#define HAVE_OPENLDAP 1 +/* #undef HAVE_MOZILLA_LDAP */ +/* #undef HAVE_LDAP_SSL_H */ +/* #undef HAVE_LDAP_SSL */ +#define HAVE_LDAP_REBIND_PROC 1 + + +/* + * Do we have libpaper? + */ + +/* #undef HAVE_LIBPAPER */ + + +/* + * Do we have DNS Service Discovery (aka Bonjour)? + */ + +#define HAVE_DNSSD 1 + + +/* + * Do we have ? + */ + +#define HAVE_SYS_IOCTL_H 1 + + +/* + * Does the "stat" structure contain the "st_gen" member? + */ + +#define HAVE_ST_GEN 1 + + +/* + * Does the "tm" structure contain the "tm_gmtoff" member? + */ + +#define HAVE_TM_GMTOFF 1 + + +/* + * Do we have rresvport_af()? + */ + +#define HAVE_RRESVPORT_AF 1 + + +/* + * Do we have getaddrinfo()? + */ + +#define HAVE_GETADDRINFO 1 + + +/* + * Do we have getnameinfo()? + */ + +#define HAVE_GETNAMEINFO 1 + + +/* + * Do we have getifaddrs()? + */ + +#define HAVE_GETIFADDRS 1 + + +/* + * Do we have hstrerror()? + */ + +#define HAVE_HSTRERROR 1 + + +/* + * Do we have res_init()? + */ + +#define HAVE_RES_INIT 1 + + +/* + * Do we have + */ + +#define HAVE_RESOLV_H 1 + + +/* + * Do we have the header file? + */ + +#define HAVE_SYS_SOCKIO_H 1 + + +/* + * Does the sockaddr structure contain an sa_len parameter? + */ + +/* #undef HAVE_STRUCT_SOCKADDR_SA_LEN */ + + +/* + * Do we have the AIX usersec.h header file? + */ + +/* #undef HAVE_USERSEC_H */ + + +/* + * Do we have pthread support? + */ + +#define HAVE_PTHREAD_H 1 + + +/* + * Do we have launchd support? + */ + +#define HAVE_LAUNCH_H 1 +#define HAVE_LAUNCHD 1 + + +/* + * Various scripting languages... + */ + +#define HAVE_JAVA 1 +#define CUPS_JAVA "/usr/bin/java" +#define HAVE_PERL 1 +#define CUPS_PERL "/usr/bin/perl" +#define HAVE_PHP 1 +#define CUPS_PHP "/usr/bin/php" +#define HAVE_PYTHON 1 +#define CUPS_PYTHON "/usr/bin/python" + + +/* + * Location of the poppler/Xpdf pdftops program... + */ + +/* #undef HAVE_PDFTOPS */ +#define CUPS_PDFTOPS "" + + +/* + * Location of the Ghostscript gs program... + */ + +/* #undef HAVE_GHOSTSCRIPT */ +#define CUPS_GHOSTSCRIPT "" + + +/* + * Do we have Darwin's CoreFoundation and SystemConfiguration frameworks? + */ + +#define HAVE_COREFOUNDATION 1 +#define HAVE_SYSTEMCONFIGURATION 1 + + +/* + * Do we have CoreFoundation public and private headers? + */ + +#define HAVE_COREFOUNDATION_H 1 +/* #undef HAVE_CFPRIV_H */ +/* #undef HAVE_CFBUNDLEPRIV_H */ + + +/* + * Do we have ApplicationServices public headers? + */ + +#define HAVE_APPLICATIONSERVICES_H 1 + + +/* + * Do we have the SCDynamicStoreCopyComputerName function? + */ + +#define HAVE_SCDYNAMICSTORECOPYCOMPUTERNAME 1 + + +/* + * Do we have MacOSX 10.4's mbr_XXX functions? + */ + +#define HAVE_MEMBERSHIP_H 1 +/* #undef HAVE_MEMBERSHIPPRIV_H */ +#define HAVE_MBR_UID_TO_UUID 1 + + +/* + * Do we have Darwin's notify_post header and function? + */ + +#define HAVE_NOTIFY_H 1 +#define HAVE_NOTIFY_POST 1 + + +/* + * Do we have DBUS? + */ + +/* #undef HAVE_DBUS */ +/* #undef HAVE_DBUS_MESSAGE_ITER_INIT_APPEND */ + + +/* + * Do we have the AppleTalk/at_proto.h header? + */ + +/* #undef HAVE_APPLETALK_AT_PROTO_H */ + + +/* + * Do we have the GSSAPI support library (for Kerberos support)? + */ + +/* #undef HAVE_GSS_ACQUIRE_CRED_EX_F */ +#define HAVE_GSS_C_NT_HOSTBASED_SERVICE 1 +/* #undef HAVE_GSS_GSSAPI_H */ +/* #undef HAVE_GSS_GSSAPI_SPI_H */ +#define HAVE_GSSAPI 1 +#define HAVE_GSSAPI_H 1 +#define HAVE_GSSAPI_GSSAPI_H 1 +#define HAVE_GSSAPI_GSSAPI_GENERIC_H 1 +#define HAVE_GSSAPI_GSSAPI_KRB5_H 1 +#define HAVE_KRB5_H 1 + + +/* + * Default GSS service name... + */ + +#define CUPS_DEFAULT_GSSSERVICENAME "host" + + +/* + * Select/poll interfaces... + */ + +#define HAVE_POLL 1 +/* #undef HAVE_EPOLL */ +#define HAVE_KQUEUE 1 + + +/* + * Do we have the header? + */ + +#define HAVE_DLFCN_H 1 + + +/* + * Do we have ? + */ + +#define HAVE_SYS_PARAM_H 1 + + +/* + * Do we have ? + */ + +#define HAVE_SYS_UCRED_H 1 + + +/* + * Do we have removefile()? + */ + +#define HAVE_REMOVEFILE 1 + + +/* + * Do we have ? + */ + +#define HAVE_SANDBOX_H 1 + + +/* + * Which random number generator function to use... + */ + +#define HAVE_ARC4RANDOM 1 +#define HAVE_RANDOM 1 +#define HAVE_LRAND48 1 + +#ifdef HAVE_ARC4RANDOM +# define CUPS_RAND() arc4random() +# define CUPS_SRAND(v) arc4random_stir() +#elif defined(HAVE_RANDOM) +# define CUPS_RAND() random() +# define CUPS_SRAND(v) srandom(v) +#elif defined(HAVE_LRAND48) +# define CUPS_RAND() lrand48() +# define CUPS_SRAND(v) srand48(v) +#else +# define CUPS_RAND() rand() +# define CUPS_SRAND(v) srand(v) +#endif /* HAVE_ARC4RANDOM */ + + +/* + * Do we have vproc_transaction_begin/end? + */ + +#define HAVE_VPROC_TRANSACTION_BEGIN 1 + + +/* + * Do we have libusb? + */ + +/* #undef HAVE_USB_H */ + + +/* + * Do we have libwrap and tcpd.h? + */ + +/* #undef HAVE_TCPD_H */ + + +/* + * Do we have ? + */ + +#define HAVE_ICONV_H 1 + + +/* + * Do we have statfs or statvfs and one of the corresponding headers? + */ + +#define HAVE_STATFS 1 +#define HAVE_STATVFS 1 +#define HAVE_SYS_MOUNT_H 1 +/* #undef HAVE_SYS_STATFS_H */ +#define HAVE_SYS_STATVFS_H 1 +/* #undef HAVE_SYS_VFS_H */ + + +/* + * Location of Mac OS X localization bundle, if any. + */ + +#define CUPS_BUNDLEDIR "/System/Library/Frameworks/ApplicationServices.framework/Versions/A/Frameworks/PrintCore.framework/Versions/A" + + +/* + * Do we have the ColorSyncRegisterDevice function? + */ + +#define HAVE_COLORSYNCREGISTERDEVICE 1 + + +/* + * Do we have XPC? + */ + +/* #undef HAVE_XPC */ + + +#endif /* !_CUPS_CONFIG_H_ */ + +/* + * End of "$Id$". + */ -- 2.47.3